[AMD Official Use Only - General] > -----Original Message----- > From: Saravana Kannan <saravanak@xxxxxxxxxx> > Sent: Wednesday, September 7, 2022 6:03 AM > To: Gupta, Nipun <Nipun.Gupta@xxxxxxx> > Cc: robh+dt@xxxxxxxxxx; krzysztof.kozlowski+dt@xxxxxxxxxx; > gregkh@xxxxxxxxxxxxxxxxxxx; rafael@xxxxxxxxxx; eric.auger@xxxxxxxxxx; > alex.williamson@xxxxxxxxxx; cohuck@xxxxxxxxxx; Gupta, Puneet (DCG-ENG) > <puneet.gupta@xxxxxxx>; song.bao.hua@xxxxxxxxxxxxx; > mchehab+huawei@xxxxxxxxxx; maz@xxxxxxxxxx; f.fainelli@xxxxxxxxx; > jeffrey.l.hugo@xxxxxxxxx; Michael.Srba@xxxxxxxxx; mani@xxxxxxxxxx; > yishaih@xxxxxxxxxx; jgg@xxxxxxxx; jgg@xxxxxxxxxx; robin.murphy@xxxxxxx; > will@xxxxxxxxxx; joro@xxxxxxxxxx; masahiroy@xxxxxxxxxx; > ndesaulniers@xxxxxxxxxx; linux-arm-kernel@xxxxxxxxxxxxxxxxxxx; linux- > kbuild@xxxxxxxxxxxxxxx; linux-kernel@xxxxxxxxxxxxxxx; > devicetree@xxxxxxxxxxxxxxx; kvm@xxxxxxxxxxxxxxx; okaya@xxxxxxxxxx; Anand, > Harpreet <harpreet.anand@xxxxxxx>; Agarwal, Nikhil > <nikhil.agarwal@xxxxxxx>; Simek, Michal <michal.simek@xxxxxxx>; > Radovanovic, Aleksandar <aleksandar.radovanovic@xxxxxxx>; git (AMD-Xilinx) > <git@xxxxxxx> > Subject: Re: [RFC PATCH v3 2/7] bus/cdx: add the cdx bus driver > > [CAUTION: External Email] > > On Tue, Sep 6, 2022 at 6:48 AM Nipun Gupta <nipun.gupta@xxxxxxx> wrote: > > > > CDX bus supports the scanning and probing of FPGA based > > devices. These devices are registers as CDX devices. > > > > The bus driver sets up the basic infrastructure and fetches > > the device related information from the firmware. > > > > CDX bus is capable of scanning devices dynamically, > > supporting rescanning of dynamically added, removed or > > updated devices. > > > > Signed-off-by: Nipun Gupta <nipun.gupta@xxxxxxx> > > --- > > > > Basic overview of CDX bus architecture is provided in [patch 0/7]. > > > > Please NOTE: This RFC change does not support the CDX bus firmware > > interface as it is under development, and this series aims to get > > an early feedback from the community. Firmware interaction are > > stubbed as MCDI APIs which is a protocol used by AMD to interact > > with Firmware. > > > > MAINTAINERS | 2 + > > drivers/bus/Kconfig | 1 + > > drivers/bus/Makefile | 3 + > > drivers/bus/cdx/Kconfig | 7 + > > drivers/bus/cdx/Makefile | 3 + > > drivers/bus/cdx/cdx.c | 437 ++++++++++++++++++++++++++++++ > > drivers/bus/cdx/cdx.h | 34 +++ > > drivers/bus/cdx/mcdi_stubs.c | 54 ++++ > > drivers/bus/cdx/mcdi_stubs.h | 76 ++++++ > > include/linux/cdx/cdx_bus.h | 93 +++++++ > > include/linux/mod_devicetable.h | 13 + > > scripts/mod/devicetable-offsets.c | 4 + > > scripts/mod/file2alias.c | 12 + > > 13 files changed, 739 insertions(+) > > create mode 100644 drivers/bus/cdx/Kconfig > > create mode 100644 drivers/bus/cdx/Makefile > > create mode 100644 drivers/bus/cdx/cdx.c > > create mode 100644 drivers/bus/cdx/cdx.h > > create mode 100644 drivers/bus/cdx/mcdi_stubs.c > > create mode 100644 drivers/bus/cdx/mcdi_stubs.h > > create mode 100644 include/linux/cdx/cdx_bus.h > > <snip> > > + > > +static int cdx_device_add(struct device *parent, > > + struct cdx_dev_params_t *dev_params) > > +{ > > + struct cdx_device *cdx_dev; > > + int ret; > > + > > + cdx_dev = kzalloc(sizeof(*cdx_dev), GFP_KERNEL); > > + if (!cdx_dev) { > > + dev_err(parent, > > + "memory allocation for cdx dev failed\n"); > > + return -ENOMEM; > > + } > > + > > + /* Populate resource */ > > + memcpy(cdx_dev->res, dev_params->res, sizeof(struct resource) * > > + dev_params->res_count); > > + cdx_dev->res_count = dev_params->res_count; > > + > > + /* Populate CDX dev params */ > > + cdx_dev->req_id = dev_params->req_id; > > + cdx_dev->vendor = dev_params->vendor; > > + cdx_dev->device = dev_params->device; > > + cdx_dev->bus_id = dev_params->bus_id; > > + cdx_dev->func_id = dev_params->func_id; > > + cdx_dev->dma_mask = CDX_DEFAULT_DMA_MASK; > > + > > + /* Initiaize generic device */ > > + device_initialize(&cdx_dev->dev); > > + cdx_dev->dev.parent = parent; > > + cdx_dev->dev.bus = &cdx_bus_type; > > + cdx_dev->dev.dma_mask = &cdx_dev->dma_mask; > > + cdx_dev->dev.release = cdx_device_release; > > How will these devices get resources like clocks, regulators, etc that > might be provided by other DT based devices? Is that not possible? > > I also see that v2 of this series has a "swnode" implementation that > was used to set the fwnode of these CDX devices. Why are these devices > no longer getting the fwnode set? Also, swnode doesn't have support > for the fwnode.add_links() ops. It also doesn't seem to support the > parent of a swnode being another fwnode of a different type (DT). I'm > not sure about the history behind that, but maybe swnode is not the > right fit or you might need to add support for these to swnode. All of > this is to say that if you set these things up correctly, fw_devlink > can work for CDX devices and that might be helpful. In the RFC v2 we were trying to use the platform bus for CDX devices, and in that implementation sw_node was used, but now we understand that using platform bus for dynamic devices is not a good approach. So, changes in RFC v3 are to have a CDX bus for CDX devices. Regards, Nipun > > -Saravana > > + > > + /* Set Name */ > > + dev_set_name(&cdx_dev->dev, "cdx-%02x:%02x", cdx_dev->bus_id, > > + cdx_dev->func_id); > > + > > + ret = device_add(&cdx_dev->dev); > > + if (ret != 0) { > > + dev_err(&cdx_dev->dev, > > + "cdx device add failed: %d", ret); > > + goto fail; > > + } > > + > > + return 0; > > + > > +fail: > > + /* > > + * Do not free cdx_dev here as it would be freed in > > + * cdx_device_release() called from within put_device(). > > + */ > > + put_device(&cdx_dev->dev); > > + > > + return ret; > > +} > > + > > +static int cdx_bus_device_discovery(struct platform_device *pdev) > > +{ > > + struct cdx_mcdi_t *cdx_mcdi = platform_get_drvdata(pdev); > > + int num_cdx_bus, num_cdx_func; > > + uint8_t bus_id, func_id; > > + int ret; > > + > > + cdx_controller_pdev = pdev; > > + > > + /* MCDI FW Read: Fetch the number of CDX buses present*/ > > + num_cdx_bus = cdx_mcdi_get_num_buses(cdx_mcdi); > > + > > + for (bus_id = 0; bus_id < num_cdx_bus; bus_id++) { > > + /* MCDI FW Read: Fetch the number of devices present */ > > + num_cdx_func = cdx_mcdi_get_num_funcs(cdx_mcdi, bus_id); > > + > > + for (func_id = 0; func_id < num_cdx_func; func_id++) { > > + struct cdx_dev_params_t dev_params; > > + > > + /* MCDI FW: Get the device config */ > > + ret = cdx_mcdi_get_func_config(cdx_mcdi, bus_id, > > + func_id, &dev_params); > > + if (ret) { > > + dev_err(&pdev->dev, > > + "CDX device config get failed for bus: %d\n", > > + ret); > > + goto fail; > > + } > > + > > + /* Add the device to the cdx bus */ > > + ret = cdx_device_add(&pdev->dev, &dev_params); > > + if (ret == -EPROBE_DEFER) { > > + goto fail; > > + } else if (ret) { > > + dev_err(&pdev->dev, > > + "registering cdx dev: %d failed: %d\n", > > + func_id, ret); > > + goto fail; > > + } else { > > + dev_dbg(&pdev->dev, > > + "CDX dev: %d on cdx bus: %d created\n", > > + func_id, bus_id); > > + } > > + } > > + } > > + > > + return 0; > > +fail: > > + cdx_unregister_devices(&cdx_bus_type); > > + return ret; > > +} > > + > > +static int cdx_bus_probe(struct platform_device *pdev) > > +{ > > + struct cdx_mcdi_t *cdx_mcdi; > > + int ret; > > + > > + cdx_mcdi = kzalloc(sizeof(*cdx_mcdi), GFP_KERNEL); > > + if (!cdx_mcdi) { > > + dev_err(&pdev->dev, "Failed to allocate memory for cdx_mcdi\n"); > > + return -ENOMEM; > > + } > > + > > + /* MCDI FW: Initialize the FW path */ > > + ret = cdx_mcdi_init(cdx_mcdi); > > + if (ret) { > > + dev_err(&pdev->dev, "MCDI Initialization failed: %d\n", ret); > > + goto mcdi_init_fail; > > + } > > + platform_set_drvdata(pdev, cdx_mcdi); > > + > > + /* Discover all the devices on the bus */ > > + ret = cdx_bus_device_discovery(pdev); > > + if (ret) { > > + dev_err(&pdev->dev, > > + "CDX bus device discovery failed: %d\n", ret); > > + goto device_discovery_fail; > > + } > > + > > + return 0; > > + > > +mcdi_init_fail: > > + kfree(cdx_mcdi); > > +device_discovery_fail: > > + cdx_mcdi_finish(cdx_mcdi); > > + > > + return ret; > > +} > > + > > +static int cdx_bus_remove(struct platform_device *pdev) > > +{ > > + struct cdx_mcdi_t *cdx_mcdi = platform_get_drvdata(pdev); > > + > > + cdx_unregister_devices(&cdx_bus_type); > > + > > + cdx_mcdi_finish(cdx_mcdi); > > + kfree(cdx_mcdi); > > + > > + return 0; > > +} > > + > > +static void cdx_bus_shutdown(struct platform_device *pdev) > > +{ > > + cdx_bus_remove(pdev); > > +} > > + > > +static const struct of_device_id cdx_match_table[] = { > > + {.compatible = "xlnx,cdxbus-controller-1.0",}, > > + { }, > > +}; > > + > > +MODULE_DEVICE_TABLE(of, cdx_match_table); > > + > > +static struct platform_driver cdx_bus_driver = { > > + .driver = { > > + .name = "cdx-bus", > > + .pm = NULL, > > + .of_match_table = cdx_match_table, > > + }, > > + .probe = cdx_bus_probe, > > + .remove = cdx_bus_remove, > > + .shutdown = cdx_bus_shutdown, > > +}; > > + > > +static int __init cdx_bus_driver_init(void) > > +{ > > + int ret; > > + > > + ret = bus_register(&cdx_bus_type); > > + if (ret < 0) { > > + pr_err("bus type registration failed for CDX: %d\n", ret); > > + return ret; > > + } > > + > > + ret = platform_driver_register(&cdx_bus_driver); > > + if (ret < 0) { > > + pr_err("platform_driver_register() failed: %d\n", ret); > > + goto fail; > > + } > > + > > + return 0; > > + > > +fail: > > + bus_unregister(&cdx_bus_type); > > + return ret; > > +} > > +postcore_initcall(cdx_bus_driver_init); > > diff --git a/drivers/bus/cdx/cdx.h b/drivers/bus/cdx/cdx.h > > new file mode 100644 > > index 000000000000..db0569431c10 > > --- /dev/null > > +++ b/drivers/bus/cdx/cdx.h > > @@ -0,0 +1,34 @@ > > +// SPDX-License-Identifier: GPL-2.0 > > +/* > > + * Header file for the CDX Bus > > + * > > + * Copyright (C) 2022, Advanced Micro Devices, Inc. > > + */ > > + > > +#ifndef _CDX_H_ > > +#define _CDX_H_ > > + > > +#include <linux/cdx/cdx_bus.h> > > +#include <linux/irqdomain.h> > > + > > +/** > > + * struct cdx_dev_params_t - CDX device parameters > > + * @vendor: Vendor ID for CDX device > > + * @device: Device ID for CDX device > > + * @bus_id: Bus ID for this CDX device > > + * @func_id: Function ID for this device > > + * @res: array of MMIO region entries > > + * @res_count: number of valid MMIO regions > > + * @req_id: Requestor ID associated with CDX device > > + */ > > +struct cdx_dev_params_t { > > + u16 vendor; > > + u16 device; > > + u8 bus_id; > > + u8 func_id; > > + struct resource res[MAX_CDX_DEV_RESOURCES]; > > + u8 res_count; > > + u32 req_id; > > +}; > > + > > +#endif /* _CDX_H_ */ > > diff --git a/drivers/bus/cdx/mcdi_stubs.c b/drivers/bus/cdx/mcdi_stubs.c > > new file mode 100644 > > index 000000000000..cc9d30fa02f8 > > --- /dev/null > > +++ b/drivers/bus/cdx/mcdi_stubs.c > > @@ -0,0 +1,54 @@ > > +// SPDX-License-Identifier: GPL-2.0 > > +/* > > + * MCDI Firmware interaction for CDX bus. > > + * > > + * Copyright (C) 2022, Advanced Micro Devices, Inc. > > + */ > > + > > +#include <linux/ioport.h> > > + > > +#include "cdx.h" > > +#include "mcdi_stubs.h" > > + > > +int cdx_mcdi_init(struct cdx_mcdi_t *cdx_mcdi) > > +{ > > + cdx_mcdi->id = 0; > > + cdx_mcdi->flags = 0; > > + > > + return 0; > > +} > > + > > +void cdx_mcdi_finish(struct cdx_mcdi_t *cdx_mcdi) > > +{ > > +} > > + > > +int cdx_mcdi_get_num_buses(struct cdx_mcdi_t *cdx_mcdi) > > +{ > > + return 1; > > +} > > + > > +int cdx_mcdi_get_num_funcs(struct cdx_mcdi_t *cdx_mcdi, int bus_num) > > +{ > > + return 1; > > +} > > + > > +int cdx_mcdi_get_func_config(struct cdx_mcdi_t *cdx_mcdi, > > + uint8_t bus_id, uint8_t func_id, > > + struct cdx_dev_params_t *dev_params) > > +{ > > + dev_params->res[0].start = 0xe4020000; > > + dev_params->res[0].end = 0xe4020FFF; > > + dev_params->res[0].flags = IORESOURCE_MEM; > > + dev_params->res[1].start = 0xe4100000; > > + dev_params->res[1].end = 0xE411FFFF; > > + dev_params->res[1].flags = IORESOURCE_MEM; > > + dev_params->res_count = 2; > > + > > + dev_params->req_id = 0x250; > > + dev_params->vendor = 0x10ee; > > + dev_params->device = 0x8084; > > + dev_params->bus_id = bus_id; > > + dev_params->func_id = func_id; > > + > > + return 0; > > +} > > diff --git a/drivers/bus/cdx/mcdi_stubs.h b/drivers/bus/cdx/mcdi_stubs.h > > new file mode 100644 > > index 000000000000..7b6f79d48ce9 > > --- /dev/null > > +++ b/drivers/bus/cdx/mcdi_stubs.h > > @@ -0,0 +1,76 @@ > > +// SPDX-License-Identifier: GPL-2.0 > > +/* > > + * Header file for MCDI FW interaction for CDX bus. > > + * > > + * Copyright (C) 2022, Advanced Micro Devices, Inc. > > + */ > > + > > +#ifndef _MCDI_STUBS_H_ > > +#define _MCDI_STUBS_H_ > > + > > +#include "cdx.h" > > + > > +/** > > + * struct cdx_mcdi_t - CDX MCDI Firmware interface, to interact > > + * with CDX controller. > > + * @id: ID for MCDI Firmware interface > > + * @flags: Associated flags > > + */ > > +struct cdx_mcdi_t { > > + u32 id; > > + u32 flags; > > + /* Have more MCDI interface related data */ > > +}; > > + > > +/** > > + * cdx_mcdi_init - Initialize the MCDI Firmware interface > > + * for the CDX controller. > > + * @cdx_mcdi: pointer to MCDI interface > > + * > > + * Return 0 on success, <0 on failure > > + */ > > +int cdx_mcdi_init(struct cdx_mcdi_t *cdx_mcdi); > > + > > +/** > > + * cdx_mcdi_finish - Close the MCDI Firmware interface. > > + * @cdx_mcdi: pointer to MCDI interface > > + */ > > +void cdx_mcdi_finish(struct cdx_mcdi_t *cdx_mcdi); > > + > > +/** > > + * cdx_mcdi_get_num_buses - Get the total number of busses on > > + * the controller. > > + * @cdx_mcdi: pointer to MCDI interface. > > + * > > + * Return total number of busses available on the controller, > > + * <0 on failure > > + */ > > +int cdx_mcdi_get_num_buses(struct cdx_mcdi_t *cdx_mcdi); > > + > > +/** > > + * cdx_mcdi_get_num_funcs - Get the total number of functions on > > + * a particular bus of the controller. > > + * @cdx_mcdi: pointer to MCDI interface. > > + * @bus_num: Bus number. > > + * > > + * Return total number of functions available on the bus, <0 on failure > > + */ > > +int cdx_mcdi_get_num_funcs(struct cdx_mcdi_t *cdx_mcdi, int bus_num); > > + > > +/** > > + * cdx_mcdi_get_func_config - Get configuration for a particular > > + * bus_id:func_id > > + * @cdx_mcdi: pointer to MCDI interface. > > + * @bus_num: Bus number. > > + * @func_id: Function number. > > + * @dev_params: Pointer to cdx_dev_params_t, this is populated by this > > + * function with the configuration corresponding to the provided > > + * bus_id:func_id. > > + * > > + * Return 0 total number of functions available on the bus, <0 on failure > > + */ > > +int cdx_mcdi_get_func_config(struct cdx_mcdi_t *cdx_mcdi, > > + uint8_t bus_id, uint8_t func_id, > > + struct cdx_dev_params_t *dev_params); > > + > > +#endif /* _MCDI_STUBS_H_ */ > > diff --git a/include/linux/cdx/cdx_bus.h b/include/linux/cdx/cdx_bus.h > > new file mode 100644 > > index 000000000000..6e870b2c87d9 > > --- /dev/null > > +++ b/include/linux/cdx/cdx_bus.h > > @@ -0,0 +1,93 @@ > > +// SPDX-License-Identifier: GPL-2.0 > > +/* > > + * CDX bus public interface > > + * > > + * Copyright (C) 2022, Advanced Micro Devices, Inc. > > + * > > + */ > > +#ifndef _CDX_BUS_H_ > > +#define _CDX_BUS_H_ > > + > > +#include <linux/mod_devicetable.h> > > +#include <linux/device.h> > > + > > +#define MAX_CDX_DEV_RESOURCES 6 > > + > > +/** > > + * struct cdx_device - CDX device object > > + * @dev: Linux driver model device object > > + * @vendor: Vendor ID for CDX device > > + * @device: Device ID for CDX device > > + * @bus_id: Bus ID for this CDX device > > + * @func_id: Function ID for this device > > + * @res: array of MMIO region entries > > + * @res_count: number of valid MMIO regions > > + * @dma_mask: Default DMA mask > > + * @flags: CDX device flags > > + * @req_id: Requestor ID associated with CDX device > > + * @driver_override: driver name to force a match; do not set directly, > > + * because core frees it; use driver_set_override() to > > + * set or clear it. > > + */ > > +struct cdx_device { > > + struct device dev; > > + u16 vendor; > > + u16 device; > > + u8 bus_id; > > + u8 func_id; > > + struct resource res[MAX_CDX_DEV_RESOURCES]; > > + u8 res_count; > > + u64 dma_mask; > > + u16 flags; > > + u32 req_id; > > + const char *driver_override; > > +}; > > + > > +#define to_cdx_device(_dev) \ > > + container_of(_dev, struct cdx_device, dev) > > + > > +/** > > + * struct cdx_driver - CDX device driver > > + * @driver: Generic device driver > > + * @match_id_table: table of supported device matching Ids > > + * @probe: Function called when a device is added > > + * @remove: Function called when a device is removed > > + * @shutdown: Function called at shutdown time to quiesce the device > > + * @suspend: Function called when a device is stopped > > + * @resume: Function called when a device is resumed > > + * @driver_managed_dma: Device driver doesn't use kernel DMA API for > DMA. > > + * For most device drivers, no need to care about this flag > > + * as long as all DMAs are handled through the kernel DMA API. > > + * For some special ones, for example VFIO drivers, they know > > + * how to manage the DMA themselves and set this flag so that > > + * the IOMMU layer will allow them to setup and manage their > > + * own I/O address space. > > + */ > > +struct cdx_driver { > > + struct device_driver driver; > > + const struct cdx_device_id *match_id_table; > > + int (*probe)(struct cdx_device *dev); > > + int (*remove)(struct cdx_device *dev); > > + void (*shutdown)(struct cdx_device *dev); > > + int (*suspend)(struct cdx_device *dev, pm_message_t state); > > + int (*resume)(struct cdx_device *dev); > > + bool driver_managed_dma; > > +}; > > + > > +#define to_cdx_driver(_drv) \ > > + container_of(_drv, struct cdx_driver, driver) > > + > > +/* > > + * Macro to avoid include chaining to get THIS_MODULE > > + */ > > +#define cdx_driver_register(drv) \ > > + __cdx_driver_register(drv, THIS_MODULE) > > + > > +int __must_check __cdx_driver_register(struct cdx_driver *cdx_driver, > > + struct module *owner); > > + > > +void cdx_driver_unregister(struct cdx_driver *driver); > > + > > +extern struct bus_type cdx_bus_type; > > + > > +#endif /* _CDX_BUS_H_ */ > > diff --git a/include/linux/mod_devicetable.h > b/include/linux/mod_devicetable.h > > index 549590e9c644..9b94be83d53e 100644 > > --- a/include/linux/mod_devicetable.h > > +++ b/include/linux/mod_devicetable.h > > @@ -911,4 +911,17 @@ struct ishtp_device_id { > > kernel_ulong_t driver_data; > > }; > > > > +/** > > + * struct cdx_device_id - CDX device identifier > > + * @vendor: vendor ID > > + * @device: Device ID > > + * > > + * Type of entries in the "device Id" table for CDX devices supported by > > + * a CDX device driver. > > + */ > > +struct cdx_device_id { > > + __u16 vendor; > > + __u16 device; > > +}; > > + > > #endif /* LINUX_MOD_DEVICETABLE_H */ > > diff --git a/scripts/mod/devicetable-offsets.c b/scripts/mod/devicetable- > offsets.c > > index c0d3bcb99138..62dc988df84d 100644 > > --- a/scripts/mod/devicetable-offsets.c > > +++ b/scripts/mod/devicetable-offsets.c > > @@ -262,5 +262,9 @@ int main(void) > > DEVID(ishtp_device_id); > > DEVID_FIELD(ishtp_device_id, guid); > > > > + DEVID(cdx_device_id); > > + DEVID_FIELD(cdx_device_id, vendor); > > + DEVID_FIELD(cdx_device_id, device); > > + > > return 0; > > } > > diff --git a/scripts/mod/file2alias.c b/scripts/mod/file2alias.c > > index 80d973144fde..c36e1f624e39 100644 > > --- a/scripts/mod/file2alias.c > > +++ b/scripts/mod/file2alias.c > > @@ -1452,6 +1452,17 @@ static int do_dfl_entry(const char *filename, void > *symval, char *alias) > > return 1; > > } > > > > +/* Looks like: cdx:vNdN */ > > +static int do_cdx_entry(const char *filename, void *symval, > > + char *alias) > > +{ > > + DEF_FIELD(symval, cdx_device_id, vendor); > > + DEF_FIELD(symval, cdx_device_id, device); > > + > > + sprintf(alias, "cdx:v%08Xd%08Xd", vendor, device); > > + return 1; > > +} > > + > > /* Does namelen bytes of name exactly match the symbol? */ > > static bool sym_is(const char *name, unsigned namelen, const char *symbol) > > { > > @@ -1531,6 +1542,7 @@ static const struct devtable devtable[] = { > > {"ssam", SIZE_ssam_device_id, do_ssam_entry}, > > {"dfl", SIZE_dfl_device_id, do_dfl_entry}, > > {"ishtp", SIZE_ishtp_device_id, do_ishtp_entry}, > > + {"cdx", SIZE_cdx_device_id, do_cdx_entry}, > > }; > > > > /* Create MODULE_ALIAS() statements. > > -- > > 2.25.1 > >