|  | /* | 
|  | * 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); |