| /* | 
 |  * SPDX-License-Identifier: GPL-2.0-or-later | 
 |  * | 
 |  * Emulation of a CXL Switch Mailbox CCI PCIe function. | 
 |  * | 
 |  * Copyright (c) 2023 Huawei Technologies. | 
 |  * | 
 |  * From www.computeexpresslink.org | 
 |  * Compute Express Link (CXL) Specification revision 3.0 Version 1.0 | 
 |  */ | 
 | #include "qemu/osdep.h" | 
 | #include "hw/pci/pci.h" | 
 | #include "hw/pci-bridge/cxl_upstream_port.h" | 
 | #include "qapi/error.h" | 
 | #include "qemu/log.h" | 
 | #include "qemu/module.h" | 
 | #include "hw/qdev-properties.h" | 
 | #include "hw/cxl/cxl.h" | 
 |  | 
 | static void cswmbcci_reset(DeviceState *dev) | 
 | { | 
 |     CSWMBCCIDev *cswmb = CXL_SWITCH_MAILBOX_CCI(dev); | 
 |     cxl_device_register_init_swcci(cswmb); | 
 | } | 
 |  | 
 | static void cswbcci_realize(PCIDevice *pci_dev, Error **errp) | 
 | { | 
 |     CSWMBCCIDev *cswmb = CXL_SWITCH_MAILBOX_CCI(pci_dev); | 
 |     CXLComponentState *cxl_cstate = &cswmb->cxl_cstate; | 
 |     CXLDeviceState *cxl_dstate = &cswmb->cxl_dstate; | 
 |     CXLDVSECRegisterLocator *regloc_dvsec; | 
 |     CXLUpstreamPort *usp; | 
 |  | 
 |     if (!cswmb->target) { | 
 |         error_setg(errp, "Target not set"); | 
 |         return; | 
 |     } | 
 |     usp = CXL_USP(cswmb->target); | 
 |  | 
 |     pcie_endpoint_cap_init(pci_dev, 0x80); | 
 |     cxl_cstate->dvsec_offset = 0x100; | 
 |     cxl_cstate->pdev = pci_dev; | 
 |     cswmb->cci = &usp->swcci; | 
 |     cxl_device_register_block_init(OBJECT(pci_dev), cxl_dstate, cswmb->cci); | 
 |     pci_register_bar(pci_dev, 0, | 
 |                      PCI_BASE_ADDRESS_SPACE_MEMORY | | 
 |                          PCI_BASE_ADDRESS_MEM_TYPE_64, | 
 |                      &cxl_dstate->device_registers); | 
 |     regloc_dvsec = &(CXLDVSECRegisterLocator) { | 
 |         .rsvd         = 0, | 
 |         .reg0_base_lo = RBI_CXL_DEVICE_REG | 0, | 
 |         .reg0_base_hi = 0, | 
 |     }; | 
 |     cxl_component_create_dvsec(cxl_cstate, CXL3_SWITCH_MAILBOX_CCI, | 
 |                                REG_LOC_DVSEC_LENGTH, REG_LOC_DVSEC, | 
 |                                REG_LOC_DVSEC_REVID, (uint8_t *)regloc_dvsec); | 
 |  | 
 |     cxl_initialize_mailbox_swcci(cswmb->cci, DEVICE(pci_dev), | 
 |                                  DEVICE(cswmb->target), | 
 |                                  CXL_MAILBOX_MAX_PAYLOAD_SIZE); | 
 | } | 
 |  | 
 | static void cswmbcci_exit(PCIDevice *pci_dev) | 
 | { | 
 |     /* Nothing to do here yet */ | 
 | } | 
 |  | 
 | static Property cxl_switch_cci_props[] = { | 
 |     DEFINE_PROP_LINK("target", CSWMBCCIDev, | 
 |                      target, TYPE_CXL_USP, PCIDevice *), | 
 |     DEFINE_PROP_END_OF_LIST(), | 
 | }; | 
 |  | 
 | static void cswmbcci_class_init(ObjectClass *oc, void *data) | 
 | { | 
 |     DeviceClass *dc = DEVICE_CLASS(oc); | 
 |     PCIDeviceClass *pc = PCI_DEVICE_CLASS(oc); | 
 |  | 
 |     pc->realize = cswbcci_realize; | 
 |     pc->exit = cswmbcci_exit; | 
 |     /* Serial bus, CXL Switch CCI */ | 
 |     pc->class_id = 0x0c0b; | 
 |     /* | 
 |      * Huawei Technologies | 
 |      * CXL Switch Mailbox CCI - DID assigned for emulation only. | 
 |      * No real hardware will ever use this ID. | 
 |      */ | 
 |     pc->vendor_id = 0x19e5; | 
 |     pc->device_id = 0xa123; | 
 |     pc->revision = 0; | 
 |     dc->desc = "CXL Switch Mailbox CCI"; | 
 |     dc->reset = cswmbcci_reset; | 
 |     device_class_set_props(dc, cxl_switch_cci_props); | 
 | } | 
 |  | 
 | static const TypeInfo cswmbcci_info = { | 
 |     .name = TYPE_CXL_SWITCH_MAILBOX_CCI, | 
 |     .parent = TYPE_PCI_DEVICE, | 
 |     .class_init = cswmbcci_class_init, | 
 |     .instance_size = sizeof(CSWMBCCIDev), | 
 |     .interfaces = (InterfaceInfo[]) { | 
 |         { INTERFACE_PCIE_DEVICE }, | 
 |         { } | 
 |     }, | 
 | }; | 
 |  | 
 | static void cxl_switch_mailbox_cci_register(void) | 
 | { | 
 |     type_register_static(&cswmbcci_info); | 
 | } | 
 | type_init(cxl_switch_mailbox_cci_register); |