/* SPDX-License-Identifier: GPL-2.0 */ /* Copyright(c) 2020 Intel Corporation. */ /* Copyright(c) 2026 Advanced Micro Devices, Inc. */ #ifndef __CXL_CXL_H__ #define __CXL_CXL_H__ #include #include #include /** * enum cxl_devtype - delineate type-2 from a generic type-3 device * @CXL_DEVTYPE_DEVMEM: Vendor specific CXL Type-2 device implementing HDM-D or * HDM-DB, no requirement that this device implements a * mailbox, or other memory-device-standard manageability * flows. * @CXL_DEVTYPE_CLASSMEM: Common class definition of a CXL Type-3 device with * HDM-H and class-mandatory memory device registers */ enum cxl_devtype { CXL_DEVTYPE_DEVMEM, CXL_DEVTYPE_CLASSMEM, }; struct device; /* * Using struct_group() allows for per register-block-type helper routines, * without requiring block-type agnostic code to include the prefix. */ struct cxl_regs { /* * Common set of CXL Component register block base pointers * @hdm_decoder: CXL 2.0 8.2.5.12 CXL HDM Decoder Capability Structure * @ras: CXL 2.0 8.2.5.9 CXL RAS Capability Structure */ struct_group_tagged(cxl_component_regs, component, void __iomem *hdm_decoder; void __iomem *ras; ); /* * Common set of CXL Device register block base pointers * @status: CXL 2.0 8.2.8.3 Device Status Registers * @mbox: CXL 2.0 8.2.8.4 Mailbox Registers * @memdev: CXL 2.0 8.2.8.5 Memory Device Registers */ struct_group_tagged(cxl_device_regs, device_regs, void __iomem *status, *mbox, *memdev; ); struct_group_tagged(cxl_pmu_regs, pmu_regs, void __iomem *pmu; ); /* * RCH downstream port specific RAS register * @aer: CXL 3.0 8.2.1.1 RCH Downstream Port RCRB */ struct_group_tagged(cxl_rch_regs, rch_regs, void __iomem *dport_aer; ); /* * RCD upstream port specific PCIe cap register * @pcie_cap: CXL 3.0 8.2.1.2 RCD Upstream Port RCRB */ struct_group_tagged(cxl_rcd_regs, rcd_regs, void __iomem *rcd_pcie_cap; ); }; struct cxl_reg_map { bool valid; int id; unsigned long offset; unsigned long size; }; struct cxl_component_reg_map { struct cxl_reg_map hdm_decoder; struct cxl_reg_map ras; }; struct cxl_device_reg_map { struct cxl_reg_map status; struct cxl_reg_map mbox; struct cxl_reg_map memdev; }; struct cxl_pmu_reg_map { struct cxl_reg_map pmu; }; /** * struct cxl_register_map - DVSEC harvested register block mapping parameters * @host: device for devm operations and logging * @base: virtual base of the register-block-BAR + @block_offset * @resource: physical resource base of the register block * @max_size: maximum mapping size to perform register search * @reg_type: see enum cxl_regloc_type * @component_map: cxl_reg_map for component registers * @device_map: cxl_reg_maps for device registers * @pmu_map: cxl_reg_maps for CXL Performance Monitoring Units */ struct cxl_register_map { struct device *host; void __iomem *base; resource_size_t resource; resource_size_t max_size; u8 reg_type; union { struct cxl_component_reg_map component_map; struct cxl_device_reg_map device_map; struct cxl_pmu_reg_map pmu_map; }; }; /** * struct cxl_dpa_perf - DPA performance property entry * @dpa_range: range for DPA address * @coord: QoS performance data (i.e. latency, bandwidth) * @cdat_coord: raw QoS performance data from CDAT * @qos_class: QoS Class cookies */ struct cxl_dpa_perf { struct range dpa_range; struct access_coordinate coord[ACCESS_COORDINATE_MAX]; struct access_coordinate cdat_coord[ACCESS_COORDINATE_MAX]; int qos_class; }; enum cxl_partition_mode { CXL_PARTMODE_RAM, CXL_PARTMODE_PMEM, }; /** * struct cxl_dpa_partition - DPA partition descriptor * @res: shortcut to the partition in the DPA resource tree (cxlds->dpa_res) * @perf: performance attributes of the partition from CDAT * @mode: operation mode for the DPA capacity, e.g. ram, pmem, dynamic... */ struct cxl_dpa_partition { struct resource res; struct cxl_dpa_perf perf; enum cxl_partition_mode mode; }; #define CXL_NR_PARTITIONS_MAX 2 /** * struct cxl_dev_state - The driver device state * * cxl_dev_state represents the CXL driver/device state. It provides an * interface to mailbox commands as well as some cached data about the device. * Currently only memory devices are represented. * * @dev: The device associated with this CXL state * @cxlmd: The device representing the CXL.mem capabilities of @dev * @reg_map: component and ras register mapping parameters * @regs: Parsed register blocks * @cxl_dvsec: Offset to the PCIe device DVSEC * @rcd: operating in RCD mode (CXL 3.0 9.11.8 CXL Devices Attached to an RCH) * @media_ready: Indicate whether the device media is usable * @dpa_res: Overall DPA resource tree for the device * @part: DPA partition array * @nr_partitions: Number of DPA partitions * @serial: PCIe Device Serial Number * @type: Generic Memory Class device or Vendor Specific Memory device * @cxl_mbox: CXL mailbox context * @cxlfs: CXL features context */ struct cxl_dev_state { /* public for Type2 drivers */ struct device *dev; struct cxl_memdev *cxlmd; /* private for Type2 drivers */ struct cxl_register_map reg_map; struct cxl_device_regs regs; int cxl_dvsec; bool rcd; bool media_ready; struct resource dpa_res; struct cxl_dpa_partition part[CXL_NR_PARTITIONS_MAX]; unsigned int nr_partitions; u64 serial; enum cxl_devtype type; struct cxl_mailbox cxl_mbox; #ifdef CONFIG_CXL_FEATURES struct cxl_features_state *cxlfs; #endif }; struct cxl_dev_state *_devm_cxl_dev_state_create(struct device *dev, enum cxl_devtype type, u64 serial, u16 dvsec, size_t size, bool has_mbox); /** * cxl_dev_state_create - safely create and cast a cxl dev state embedded in a * driver specific struct. * * @parent: device behind the request * @type: CXL device type * @serial: device identification * @dvsec: dvsec capability offset * @drv_struct: driver struct embedding a cxl_dev_state struct * @member: name of the struct cxl_dev_state member in drv_struct * @mbox: true if mailbox supported * * Returns a pointer to the drv_struct allocated and embedding a cxl_dev_state * struct initialized. * * Introduced for Type2 driver support. */ #define devm_cxl_dev_state_create(parent, type, serial, dvsec, drv_struct, member, mbox) \ ({ \ static_assert(__same_type(struct cxl_dev_state, \ ((drv_struct *)NULL)->member)); \ static_assert(offsetof(drv_struct, member) == 0); \ (drv_struct *)_devm_cxl_dev_state_create(parent, type, serial, dvsec, \ sizeof(drv_struct), mbox); \ }) #endif /* __CXL_CXL_H__ */