f-stack/dpdk/drivers/bus/cdx/cdx.c

671 lines
16 KiB
C
Raw Normal View History

2025-01-10 11:50:43 +00:00
/* SPDX-License-Identifier: BSD-3-Clause
* Copyright (C) 2022-2023, Advanced Micro Devices, Inc.
*/
/*
* Architecture Overview
* =====================
* CDX is a Hardware Architecture designed for AMD FPGA devices. It
* consists of sophisticated mechanism for interaction between FPGA,
* Firmware and the APUs (Application CPUs).
*
* Firmware resides on RPU (Realtime CPUs) which interacts with
* the FPGA program manager and the APUs. The RPU provides memory-mapped
* interface (RPU if) which is used to communicate with APUs.
*
* The diagram below shows an overview of the AMD CDX architecture:
*
* +--------------------------------------+
* | DPDK |
* | DPDK CDX drivers |
* | | |
* | DPDK AMD CDX bus |
* | | |
* +-----------------------------|--------+
* |
* +-----------------------------|--------+
* | Application CPUs (APU) | |
* | | |
* | VFIO CDX driver |
* | Linux OS | |
* | Linux AMD CDX bus |
* | | |
* +-----------------------------|--------+
* |
* |
* +------------------------| RPU if |----+
* | | |
* | V |
* | Realtime CPUs (RPU) |
* | |
* +--------------------------------------+
* |
* +---------------------|----------------+
* | FPGA | |
* | +-----------------------+ |
* | | | | |
* | +-------+ +-------+ +-------+ |
* | | dev 1 | | dev 2 | | dev 3 | |
* | +-------+ +-------+ +-------+ |
* +--------------------------------------+
*
* The RPU firmware extracts the device information from the loaded FPGA
* image and implements a mechanism that allows the APU drivers to
* enumerate such devices (device personality and resource details) via
* a dedicated communication channel.
*
* VFIO CDX driver provides the CDX device resources like MMIO and interrupts
* to map to user-space. DPDK CDX bus uses sysfs interface and the vfio-cdx
* driver to discover and initialize the CDX devices for user-space
* applications.
*/
/**
* @file
* CDX probing using Linux sysfs.
*/
#include <string.h>
#include <dirent.h>
#include <rte_eal_paging.h>
#include <rte_errno.h>
#include <rte_devargs.h>
#include <rte_kvargs.h>
#include <rte_malloc.h>
#include <rte_vfio.h>
#include <eal_filesystem.h>
#include "bus_cdx_driver.h"
#include "cdx_logs.h"
#include "private.h"
#define CDX_BUS_NAME cdx
#define CDX_DEV_PREFIX "cdx-"
/* CDX Bus iterators */
#define FOREACH_DEVICE_ON_CDXBUS(p) \
RTE_TAILQ_FOREACH(p, &rte_cdx_bus.device_list, next)
#define FOREACH_DRIVER_ON_CDXBUS(p) \
RTE_TAILQ_FOREACH(p, &rte_cdx_bus.driver_list, next)
struct rte_cdx_bus rte_cdx_bus;
enum cdx_params {
RTE_CDX_PARAM_NAME,
};
static const char * const cdx_params_keys[] = {
[RTE_CDX_PARAM_NAME] = "name",
NULL,
};
/* Add a device to CDX bus */
static void
cdx_add_device(struct rte_cdx_device *cdx_dev)
{
TAILQ_INSERT_TAIL(&rte_cdx_bus.device_list, cdx_dev, next);
}
static int
cdx_get_kernel_driver_by_path(const char *filename, char *driver_name,
size_t len)
{
int count;
char path[PATH_MAX];
char *name;
if (!filename || !driver_name)
return -1;
count = readlink(filename, path, PATH_MAX);
if (count >= PATH_MAX)
return -1;
/* For device does not have a driver */
if (count < 0)
return 1;
path[count] = '\0';
name = strrchr(path, '/');
if (name) {
strlcpy(driver_name, name + 1, len);
return 0;
}
return -1;
}
int rte_cdx_map_device(struct rte_cdx_device *dev)
{
return cdx_vfio_map_resource(dev);
}
void rte_cdx_unmap_device(struct rte_cdx_device *dev)
{
cdx_vfio_unmap_resource(dev);
}
static struct rte_devargs *
cdx_devargs_lookup(const char *dev_name)
{
struct rte_devargs *devargs;
RTE_EAL_DEVARGS_FOREACH("cdx", devargs) {
if (strcmp(devargs->name, dev_name) == 0)
return devargs;
}
return NULL;
}
static bool
cdx_ignore_device(const char *dev_name)
{
struct rte_devargs *devargs = cdx_devargs_lookup(dev_name);
switch (rte_cdx_bus.bus.conf.scan_mode) {
case RTE_BUS_SCAN_ALLOWLIST:
if (devargs && devargs->policy == RTE_DEV_ALLOWED)
return false;
break;
case RTE_BUS_SCAN_UNDEFINED:
case RTE_BUS_SCAN_BLOCKLIST:
if (devargs == NULL || devargs->policy != RTE_DEV_BLOCKED)
return false;
break;
}
return true;
}
/*
* Scan one cdx sysfs entry, and fill the devices list from it.
* It checks if the CDX device is bound to vfio-cdx driver. In case
* the device is vfio bound, it reads the vendor and device id and
* stores it for device-driver matching.
*/
static int
cdx_scan_one(const char *dirname, const char *dev_name)
{
char filename[PATH_MAX];
struct rte_cdx_device *dev = NULL;
char driver[PATH_MAX];
unsigned long tmp;
int ret;
dev = calloc(1, sizeof(*dev));
if (!dev)
return -ENOMEM;
dev->device.bus = &rte_cdx_bus.bus;
memcpy(dev->name, dev_name, RTE_DEV_NAME_MAX_LEN);
dev->device.name = dev->name;
/* parse driver */
snprintf(filename, sizeof(filename), "%s/driver", dirname);
ret = cdx_get_kernel_driver_by_path(filename, driver, sizeof(driver));
if (ret < 0) {
CDX_BUS_ERR("Fail to get kernel driver");
free(dev);
return -1;
}
/* Allocate interrupt instance for cdx device */
dev->intr_handle =
rte_intr_instance_alloc(RTE_INTR_INSTANCE_F_PRIVATE);
if (dev->intr_handle == NULL) {
CDX_BUS_ERR("Failed to create interrupt instance for %s",
dev->device.name);
free(dev);
return -ENOMEM;
}
/*
* Check if device is bound to 'vfio-cdx' driver, so that user-space
* can gracefully access the device.
*/
if (ret || strcmp(driver, "vfio-cdx")) {
ret = 0;
goto err;
}
/* get vendor id */
snprintf(filename, sizeof(filename), "%s/vendor", dirname);
if (eal_parse_sysfs_value(filename, &tmp) < 0) {
ret = -1;
goto err;
}
dev->id.vendor_id = (uint16_t)tmp;
/* get device id */
snprintf(filename, sizeof(filename), "%s/device", dirname);
if (eal_parse_sysfs_value(filename, &tmp) < 0) {
ret = -1;
goto err;
}
dev->id.device_id = (uint16_t)tmp;
cdx_add_device(dev);
return 0;
err:
rte_intr_instance_free(dev->intr_handle);
free(dev);
return ret;
}
/*
* Scan the content of the CDX bus, and the devices in the devices
* list.
*/
static int
cdx_scan(void)
{
struct dirent *e;
DIR *dir;
char dirname[PATH_MAX];
dir = opendir(RTE_CDX_BUS_DEVICES_PATH);
if (dir == NULL) {
CDX_BUS_INFO("%s(): opendir failed: %s", __func__,
strerror(errno));
return 0;
}
while ((e = readdir(dir)) != NULL) {
if (e->d_name[0] == '.')
continue;
if (cdx_ignore_device(e->d_name))
continue;
snprintf(dirname, sizeof(dirname), "%s/%s",
RTE_CDX_BUS_DEVICES_PATH, e->d_name);
if (cdx_scan_one(dirname, e->d_name) < 0)
goto error;
}
closedir(dir);
return 0;
error:
closedir(dir);
return -1;
}
/* map a particular resource from a file */
void *
cdx_map_resource(void *requested_addr, int fd, uint64_t offset, size_t size,
int additional_flags)
{
void *mapaddr;
/* Map the cdx MMIO memory resource of device */
mapaddr = rte_mem_map(requested_addr, size,
RTE_PROT_READ | RTE_PROT_WRITE,
RTE_MAP_SHARED | additional_flags, fd, offset);
if (mapaddr == NULL) {
CDX_BUS_ERR("%s(): cannot map resource(%d, %p, 0x%zx, 0x%"PRIx64"): %s (%p)",
__func__, fd, requested_addr, size, offset,
rte_strerror(rte_errno), mapaddr);
}
CDX_BUS_DEBUG("CDX MMIO memory mapped at %p", mapaddr);
return mapaddr;
}
/* unmap a particular resource */
void
cdx_unmap_resource(void *requested_addr, size_t size)
{
if (requested_addr == NULL)
return;
CDX_BUS_DEBUG("Unmapping CDX memory at %p", requested_addr);
/* Unmap the CDX memory resource of device */
if (rte_mem_unmap(requested_addr, size)) {
CDX_BUS_ERR("%s(): cannot mem unmap(%p, %#zx): %s", __func__,
requested_addr, size, rte_strerror(rte_errno));
}
}
/*
* Match the CDX Driver and Device using device id and vendor id.
*/
static bool
cdx_match(const struct rte_cdx_driver *cdx_drv,
const struct rte_cdx_device *cdx_dev)
{
const struct rte_cdx_id *id_table;
for (id_table = cdx_drv->id_table; id_table->vendor_id != 0;
id_table++) {
/* check if device's identifiers match the driver's ones */
if (id_table->vendor_id != cdx_dev->id.vendor_id &&
id_table->vendor_id != RTE_CDX_ANY_ID)
continue;
if (id_table->device_id != cdx_dev->id.device_id &&
id_table->device_id != RTE_CDX_ANY_ID)
continue;
return 1;
}
return 0;
}
/*
* If vendor id and device id match, call the probe() function of the
* driver.
*/
static int
cdx_probe_one_driver(struct rte_cdx_driver *dr,
struct rte_cdx_device *dev)
{
const char *dev_name = dev->name;
bool already_probed;
int ret;
/* The device is not blocked; Check if driver supports it */
if (!cdx_match(dr, dev))
/* Match of device and driver failed */
return 1;
already_probed = rte_dev_is_probed(&dev->device);
if (already_probed) {
CDX_BUS_INFO("Device %s is already probed", dev_name);
return -EEXIST;
}
CDX_BUS_DEBUG(" probe device %s using driver: %s", dev_name,
dr->driver.name);
if (dr->drv_flags & RTE_CDX_DRV_NEED_MAPPING) {
ret = cdx_vfio_map_resource(dev);
if (ret != 0) {
CDX_BUS_ERR("CDX map device failed: %d", ret);
goto error_map_device;
}
}
/* call the driver probe() function */
ret = dr->probe(dr, dev);
if (ret) {
CDX_BUS_ERR("Probe CDX driver: %s device: %s failed: %d",
dr->driver.name, dev_name, ret);
goto error_probe;
} else {
dev->device.driver = &dr->driver;
}
dev->driver = dr;
return ret;
error_probe:
cdx_vfio_unmap_resource(dev);
rte_intr_instance_free(dev->intr_handle);
dev->intr_handle = NULL;
error_map_device:
return ret;
}
/*
* If vendor/device ID match, call the probe() function of all
* registered driver for the given device. Return < 0 if initialization
* failed, return 1 if no driver is found for this device.
*/
static int
cdx_probe_all_drivers(struct rte_cdx_device *dev)
{
struct rte_cdx_driver *dr = NULL;
int rc = 0;
FOREACH_DRIVER_ON_CDXBUS(dr) {
rc = cdx_probe_one_driver(dr, dev);
if (rc < 0)
/* negative value is an error */
return rc;
if (rc > 0)
/* positive value means driver doesn't support it */
continue;
return 0;
}
return 1;
}
/*
* Scan the content of the CDX bus, and call the probe() function for
* all registered drivers that have a matching entry in its id_table
* for discovered devices.
*/
static int
cdx_probe(void)
{
struct rte_cdx_device *dev = NULL;
size_t probed = 0, failed = 0;
int ret = 0;
FOREACH_DEVICE_ON_CDXBUS(dev) {
probed++;
ret = cdx_probe_all_drivers(dev);
if (ret < 0) {
CDX_BUS_ERR("Requested device %s cannot be used",
dev->name);
rte_errno = errno;
failed++;
}
}
return (probed && probed == failed) ? -1 : 0;
}
static int
cdx_parse(const char *name, void *addr)
{
const char **out = addr;
int ret;
ret = strncmp(name, CDX_DEV_PREFIX, strlen(CDX_DEV_PREFIX));
if (ret == 0 && addr)
*out = name;
return ret;
}
/* register a driver */
void
rte_cdx_register(struct rte_cdx_driver *driver)
{
TAILQ_INSERT_TAIL(&rte_cdx_bus.driver_list, driver, next);
driver->bus = &rte_cdx_bus;
}
/* unregister a driver */
void
rte_cdx_unregister(struct rte_cdx_driver *driver)
{
TAILQ_REMOVE(&rte_cdx_bus.driver_list, driver, next);
driver->bus = NULL;
}
static struct rte_device *
cdx_find_device(const struct rte_device *start, rte_dev_cmp_t cmp,
const void *data)
{
const struct rte_cdx_device *cdx_start;
struct rte_cdx_device *cdx_dev;
if (start != NULL) {
cdx_start = RTE_DEV_TO_CDX_DEV_CONST(start);
cdx_dev = TAILQ_NEXT(cdx_start, next);
} else {
cdx_dev = TAILQ_FIRST(&rte_cdx_bus.device_list);
}
while (cdx_dev != NULL) {
if (cmp(&cdx_dev->device, data) == 0)
return &cdx_dev->device;
cdx_dev = TAILQ_NEXT(cdx_dev, next);
}
return NULL;
}
/* Remove a device from CDX bus */
static void
cdx_remove_device(struct rte_cdx_device *cdx_dev)
{
TAILQ_REMOVE(&rte_cdx_bus.device_list, cdx_dev, next);
}
/*
* If vendor/device ID match, call the remove() function of the
* driver.
*/
static int
cdx_detach_dev(struct rte_cdx_device *dev)
{
struct rte_cdx_driver *dr;
int ret = 0;
if (dev == NULL)
return -EINVAL;
dr = dev->driver;
CDX_BUS_DEBUG("detach device %s using driver: %s",
dev->device.name, dr->driver.name);
if (dr->remove) {
ret = dr->remove(dev);
if (ret < 0)
return ret;
}
/* clear driver structure */
dev->driver = NULL;
dev->device.driver = NULL;
rte_cdx_unmap_device(dev);
rte_intr_instance_free(dev->intr_handle);
dev->intr_handle = NULL;
return 0;
}
static int
cdx_plug(struct rte_device *dev)
{
return cdx_probe_all_drivers(RTE_DEV_TO_CDX_DEV(dev));
}
static int
cdx_unplug(struct rte_device *dev)
{
struct rte_cdx_device *cdx_dev;
int ret;
cdx_dev = RTE_DEV_TO_CDX_DEV(dev);
ret = cdx_detach_dev(cdx_dev);
if (ret == 0) {
cdx_remove_device(cdx_dev);
rte_devargs_remove(dev->devargs);
free(cdx_dev);
}
return ret;
}
static int
cdx_dma_map(struct rte_device *dev, void *addr, uint64_t iova, size_t len)
{
RTE_SET_USED(dev);
return rte_vfio_container_dma_map(RTE_VFIO_DEFAULT_CONTAINER_FD,
(uintptr_t)addr, iova, len);
}
static int
cdx_dma_unmap(struct rte_device *dev, void *addr, uint64_t iova, size_t len)
{
RTE_SET_USED(dev);
return rte_vfio_container_dma_unmap(RTE_VFIO_DEFAULT_CONTAINER_FD,
(uintptr_t)addr, iova, len);
}
static enum rte_iova_mode
cdx_get_iommu_class(void)
{
if (TAILQ_EMPTY(&rte_cdx_bus.device_list))
return RTE_IOVA_DC;
return RTE_IOVA_VA;
}
static int
cdx_dev_match(const struct rte_device *dev,
const void *_kvlist)
{
const struct rte_kvargs *kvlist = _kvlist;
const char *key = cdx_params_keys[RTE_CDX_PARAM_NAME];
const char *name;
/* no kvlist arg, all devices match */
if (kvlist == NULL)
return 0;
/* if key is present in kvlist and does not match, filter device */
name = rte_kvargs_get(kvlist, key);
if (name != NULL && strcmp(name, dev->name))
return -1;
return 0;
}
static void *
cdx_dev_iterate(const void *start,
const char *str,
const struct rte_dev_iterator *it __rte_unused)
{
rte_bus_find_device_t find_device;
struct rte_kvargs *kvargs = NULL;
struct rte_device *dev;
if (str != NULL) {
kvargs = rte_kvargs_parse(str, cdx_params_keys);
if (kvargs == NULL) {
CDX_BUS_ERR("cannot parse argument list %s", str);
rte_errno = EINVAL;
return NULL;
}
}
find_device = rte_cdx_bus.bus.find_device;
dev = find_device(start, cdx_dev_match, kvargs);
rte_kvargs_free(kvargs);
return dev;
}
struct rte_cdx_bus rte_cdx_bus = {
.bus = {
.scan = cdx_scan,
.probe = cdx_probe,
.find_device = cdx_find_device,
.plug = cdx_plug,
.unplug = cdx_unplug,
.parse = cdx_parse,
.dma_map = cdx_dma_map,
.dma_unmap = cdx_dma_unmap,
.get_iommu_class = cdx_get_iommu_class,
.dev_iterate = cdx_dev_iterate,
},
.device_list = TAILQ_HEAD_INITIALIZER(rte_cdx_bus.device_list),
.driver_list = TAILQ_HEAD_INITIALIZER(rte_cdx_bus.driver_list),
};
RTE_REGISTER_BUS(cdx, rte_cdx_bus.bus);
RTE_LOG_REGISTER_DEFAULT(cdx_logtype_bus, NOTICE);