18b8036a6SElena Agostini /* SPDX-License-Identifier: BSD-3-Clause 28b8036a6SElena Agostini * Copyright (c) 2021 NVIDIA Corporation & Affiliates 38b8036a6SElena Agostini */ 48b8036a6SElena Agostini 58b8036a6SElena Agostini #include <rte_eal.h> 618cb0756SThomas Monjalon #include <rte_tailq.h> 718cb0756SThomas Monjalon #include <rte_rwlock.h> 88b8036a6SElena Agostini #include <rte_string_fns.h> 9*a9af048aSThomas Monjalon #include <rte_memzone.h> 108b8036a6SElena Agostini #include <rte_errno.h> 118b8036a6SElena Agostini #include <rte_log.h> 128b8036a6SElena Agostini 138b8036a6SElena Agostini #include "rte_gpudev.h" 148b8036a6SElena Agostini #include "gpudev_driver.h" 158b8036a6SElena Agostini 168b8036a6SElena Agostini /* Logging */ 178b8036a6SElena Agostini RTE_LOG_REGISTER_DEFAULT(gpu_logtype, NOTICE); 188b8036a6SElena Agostini #define GPU_LOG(level, ...) \ 198b8036a6SElena Agostini rte_log(RTE_LOG_ ## level, gpu_logtype, RTE_FMT("gpu: " \ 208b8036a6SElena Agostini RTE_FMT_HEAD(__VA_ARGS__, ) "\n", RTE_FMT_TAIL(__VA_ARGS__, ))) 218b8036a6SElena Agostini 228b8036a6SElena Agostini /* Set any driver error as EPERM */ 238b8036a6SElena Agostini #define GPU_DRV_RET(function) \ 248b8036a6SElena Agostini ((function != 0) ? -(rte_errno = EPERM) : (rte_errno = 0)) 258b8036a6SElena Agostini 268b8036a6SElena Agostini /* Array of devices */ 278b8036a6SElena Agostini static struct rte_gpu *gpus; 288b8036a6SElena Agostini /* Number of currently valid devices */ 298b8036a6SElena Agostini static int16_t gpu_max; 308b8036a6SElena Agostini /* Number of currently valid devices */ 318b8036a6SElena Agostini static int16_t gpu_count; 328b8036a6SElena Agostini 33*a9af048aSThomas Monjalon /* Shared memory between processes. */ 34*a9af048aSThomas Monjalon static const char *GPU_MEMZONE = "rte_gpu_shared"; 35*a9af048aSThomas Monjalon static struct { 36*a9af048aSThomas Monjalon __extension__ struct rte_gpu_mpshared gpus[0]; 37*a9af048aSThomas Monjalon } *gpu_shared_mem; 38*a9af048aSThomas Monjalon 3918cb0756SThomas Monjalon /* Event callback object */ 4018cb0756SThomas Monjalon struct rte_gpu_callback { 4118cb0756SThomas Monjalon TAILQ_ENTRY(rte_gpu_callback) next; 4218cb0756SThomas Monjalon rte_gpu_callback_t *function; 4318cb0756SThomas Monjalon void *user_data; 4418cb0756SThomas Monjalon enum rte_gpu_event event; 4518cb0756SThomas Monjalon }; 4618cb0756SThomas Monjalon static rte_rwlock_t gpu_callback_lock = RTE_RWLOCK_INITIALIZER; 4718cb0756SThomas Monjalon static void gpu_free_callbacks(struct rte_gpu *dev); 4818cb0756SThomas Monjalon 498b8036a6SElena Agostini int 508b8036a6SElena Agostini rte_gpu_init(size_t dev_max) 518b8036a6SElena Agostini { 528b8036a6SElena Agostini if (dev_max == 0 || dev_max > INT16_MAX) { 538b8036a6SElena Agostini GPU_LOG(ERR, "invalid array size"); 548b8036a6SElena Agostini rte_errno = EINVAL; 558b8036a6SElena Agostini return -rte_errno; 568b8036a6SElena Agostini } 578b8036a6SElena Agostini 588b8036a6SElena Agostini /* No lock, it must be called before or during first probing. */ 598b8036a6SElena Agostini if (gpus != NULL) { 608b8036a6SElena Agostini GPU_LOG(ERR, "already initialized"); 618b8036a6SElena Agostini rte_errno = EBUSY; 628b8036a6SElena Agostini return -rte_errno; 638b8036a6SElena Agostini } 648b8036a6SElena Agostini 658b8036a6SElena Agostini gpus = calloc(dev_max, sizeof(struct rte_gpu)); 668b8036a6SElena Agostini if (gpus == NULL) { 678b8036a6SElena Agostini GPU_LOG(ERR, "cannot initialize library"); 688b8036a6SElena Agostini rte_errno = ENOMEM; 698b8036a6SElena Agostini return -rte_errno; 708b8036a6SElena Agostini } 718b8036a6SElena Agostini 728b8036a6SElena Agostini gpu_max = dev_max; 738b8036a6SElena Agostini return 0; 748b8036a6SElena Agostini } 758b8036a6SElena Agostini 768b8036a6SElena Agostini uint16_t 778b8036a6SElena Agostini rte_gpu_count_avail(void) 788b8036a6SElena Agostini { 798b8036a6SElena Agostini return gpu_count; 808b8036a6SElena Agostini } 818b8036a6SElena Agostini 828b8036a6SElena Agostini bool 838b8036a6SElena Agostini rte_gpu_is_valid(int16_t dev_id) 848b8036a6SElena Agostini { 858b8036a6SElena Agostini if (dev_id >= 0 && dev_id < gpu_max && 86*a9af048aSThomas Monjalon gpus[dev_id].process_state == RTE_GPU_STATE_INITIALIZED) 878b8036a6SElena Agostini return true; 888b8036a6SElena Agostini return false; 898b8036a6SElena Agostini } 908b8036a6SElena Agostini 9182e5f6b6SThomas Monjalon static bool 9282e5f6b6SThomas Monjalon gpu_match_parent(int16_t dev_id, int16_t parent) 9382e5f6b6SThomas Monjalon { 9482e5f6b6SThomas Monjalon if (parent == RTE_GPU_ID_ANY) 9582e5f6b6SThomas Monjalon return true; 96*a9af048aSThomas Monjalon return gpus[dev_id].mpshared->info.parent == parent; 9782e5f6b6SThomas Monjalon } 9882e5f6b6SThomas Monjalon 998b8036a6SElena Agostini int16_t 10082e5f6b6SThomas Monjalon rte_gpu_find_next(int16_t dev_id, int16_t parent) 1018b8036a6SElena Agostini { 1028b8036a6SElena Agostini if (dev_id < 0) 1038b8036a6SElena Agostini dev_id = 0; 1048b8036a6SElena Agostini while (dev_id < gpu_max && 105*a9af048aSThomas Monjalon (gpus[dev_id].process_state == RTE_GPU_STATE_UNUSED || 10682e5f6b6SThomas Monjalon !gpu_match_parent(dev_id, parent))) 1078b8036a6SElena Agostini dev_id++; 1088b8036a6SElena Agostini 1098b8036a6SElena Agostini if (dev_id >= gpu_max) 1108b8036a6SElena Agostini return RTE_GPU_ID_NONE; 1118b8036a6SElena Agostini return dev_id; 1128b8036a6SElena Agostini } 1138b8036a6SElena Agostini 1148b8036a6SElena Agostini static int16_t 1158b8036a6SElena Agostini gpu_find_free_id(void) 1168b8036a6SElena Agostini { 1178b8036a6SElena Agostini int16_t dev_id; 1188b8036a6SElena Agostini 1198b8036a6SElena Agostini for (dev_id = 0; dev_id < gpu_max; dev_id++) { 120*a9af048aSThomas Monjalon if (gpus[dev_id].process_state == RTE_GPU_STATE_UNUSED) 1218b8036a6SElena Agostini return dev_id; 1228b8036a6SElena Agostini } 1238b8036a6SElena Agostini return RTE_GPU_ID_NONE; 1248b8036a6SElena Agostini } 1258b8036a6SElena Agostini 1268b8036a6SElena Agostini static struct rte_gpu * 1278b8036a6SElena Agostini gpu_get_by_id(int16_t dev_id) 1288b8036a6SElena Agostini { 1298b8036a6SElena Agostini if (!rte_gpu_is_valid(dev_id)) 1308b8036a6SElena Agostini return NULL; 1318b8036a6SElena Agostini return &gpus[dev_id]; 1328b8036a6SElena Agostini } 1338b8036a6SElena Agostini 1348b8036a6SElena Agostini struct rte_gpu * 1358b8036a6SElena Agostini rte_gpu_get_by_name(const char *name) 1368b8036a6SElena Agostini { 1378b8036a6SElena Agostini int16_t dev_id; 1388b8036a6SElena Agostini struct rte_gpu *dev; 1398b8036a6SElena Agostini 1408b8036a6SElena Agostini if (name == NULL) { 1418b8036a6SElena Agostini rte_errno = EINVAL; 1428b8036a6SElena Agostini return NULL; 1438b8036a6SElena Agostini } 1448b8036a6SElena Agostini 1458b8036a6SElena Agostini RTE_GPU_FOREACH(dev_id) { 1468b8036a6SElena Agostini dev = &gpus[dev_id]; 147*a9af048aSThomas Monjalon if (strncmp(name, dev->mpshared->name, RTE_DEV_NAME_MAX_LEN) == 0) 1488b8036a6SElena Agostini return dev; 1498b8036a6SElena Agostini } 1508b8036a6SElena Agostini return NULL; 1518b8036a6SElena Agostini } 1528b8036a6SElena Agostini 153*a9af048aSThomas Monjalon static int 154*a9af048aSThomas Monjalon gpu_shared_mem_init(void) 155*a9af048aSThomas Monjalon { 156*a9af048aSThomas Monjalon const struct rte_memzone *memzone; 157*a9af048aSThomas Monjalon 158*a9af048aSThomas Monjalon if (rte_eal_process_type() == RTE_PROC_PRIMARY) { 159*a9af048aSThomas Monjalon memzone = rte_memzone_reserve(GPU_MEMZONE, 160*a9af048aSThomas Monjalon sizeof(*gpu_shared_mem) + 161*a9af048aSThomas Monjalon sizeof(*gpu_shared_mem->gpus) * gpu_max, 162*a9af048aSThomas Monjalon SOCKET_ID_ANY, 0); 163*a9af048aSThomas Monjalon } else { 164*a9af048aSThomas Monjalon memzone = rte_memzone_lookup(GPU_MEMZONE); 165*a9af048aSThomas Monjalon } 166*a9af048aSThomas Monjalon if (memzone == NULL) { 167*a9af048aSThomas Monjalon GPU_LOG(ERR, "cannot initialize shared memory"); 168*a9af048aSThomas Monjalon rte_errno = ENOMEM; 169*a9af048aSThomas Monjalon return -rte_errno; 170*a9af048aSThomas Monjalon } 171*a9af048aSThomas Monjalon 172*a9af048aSThomas Monjalon gpu_shared_mem = memzone->addr; 173*a9af048aSThomas Monjalon return 0; 174*a9af048aSThomas Monjalon } 175*a9af048aSThomas Monjalon 1768b8036a6SElena Agostini struct rte_gpu * 1778b8036a6SElena Agostini rte_gpu_allocate(const char *name) 1788b8036a6SElena Agostini { 1798b8036a6SElena Agostini int16_t dev_id; 1808b8036a6SElena Agostini struct rte_gpu *dev; 1818b8036a6SElena Agostini 1828b8036a6SElena Agostini if (rte_eal_process_type() != RTE_PROC_PRIMARY) { 1838b8036a6SElena Agostini GPU_LOG(ERR, "only primary process can allocate device"); 1848b8036a6SElena Agostini rte_errno = EPERM; 1858b8036a6SElena Agostini return NULL; 1868b8036a6SElena Agostini } 1878b8036a6SElena Agostini if (name == NULL) { 1888b8036a6SElena Agostini GPU_LOG(ERR, "allocate device without a name"); 1898b8036a6SElena Agostini rte_errno = EINVAL; 1908b8036a6SElena Agostini return NULL; 1918b8036a6SElena Agostini } 1928b8036a6SElena Agostini 1938b8036a6SElena Agostini /* implicit initialization of library before adding first device */ 1948b8036a6SElena Agostini if (gpus == NULL && rte_gpu_init(RTE_GPU_DEFAULT_MAX) < 0) 1958b8036a6SElena Agostini return NULL; 1968b8036a6SElena Agostini 197*a9af048aSThomas Monjalon /* initialize shared memory before adding first device */ 198*a9af048aSThomas Monjalon if (gpu_shared_mem == NULL && gpu_shared_mem_init() < 0) 199*a9af048aSThomas Monjalon return NULL; 200*a9af048aSThomas Monjalon 2018b8036a6SElena Agostini if (rte_gpu_get_by_name(name) != NULL) { 2028b8036a6SElena Agostini GPU_LOG(ERR, "device with name %s already exists", name); 2038b8036a6SElena Agostini rte_errno = EEXIST; 2048b8036a6SElena Agostini return NULL; 2058b8036a6SElena Agostini } 2068b8036a6SElena Agostini dev_id = gpu_find_free_id(); 2078b8036a6SElena Agostini if (dev_id == RTE_GPU_ID_NONE) { 2088b8036a6SElena Agostini GPU_LOG(ERR, "reached maximum number of devices"); 2098b8036a6SElena Agostini rte_errno = ENOENT; 2108b8036a6SElena Agostini return NULL; 2118b8036a6SElena Agostini } 2128b8036a6SElena Agostini 2138b8036a6SElena Agostini dev = &gpus[dev_id]; 2148b8036a6SElena Agostini memset(dev, 0, sizeof(*dev)); 2158b8036a6SElena Agostini 216*a9af048aSThomas Monjalon dev->mpshared = &gpu_shared_mem->gpus[dev_id]; 217*a9af048aSThomas Monjalon memset(dev->mpshared, 0, sizeof(*dev->mpshared)); 218*a9af048aSThomas Monjalon 219*a9af048aSThomas Monjalon if (rte_strscpy(dev->mpshared->name, name, RTE_DEV_NAME_MAX_LEN) < 0) { 2208b8036a6SElena Agostini GPU_LOG(ERR, "device name too long: %s", name); 2218b8036a6SElena Agostini rte_errno = ENAMETOOLONG; 2228b8036a6SElena Agostini return NULL; 2238b8036a6SElena Agostini } 224*a9af048aSThomas Monjalon dev->mpshared->info.name = dev->mpshared->name; 225*a9af048aSThomas Monjalon dev->mpshared->info.dev_id = dev_id; 226*a9af048aSThomas Monjalon dev->mpshared->info.numa_node = -1; 227*a9af048aSThomas Monjalon dev->mpshared->info.parent = RTE_GPU_ID_NONE; 22818cb0756SThomas Monjalon TAILQ_INIT(&dev->callbacks); 229*a9af048aSThomas Monjalon __atomic_fetch_add(&dev->mpshared->process_refcnt, 1, __ATOMIC_RELAXED); 2308b8036a6SElena Agostini 2318b8036a6SElena Agostini gpu_count++; 2328b8036a6SElena Agostini GPU_LOG(DEBUG, "new device %s (id %d) of total %d", 2338b8036a6SElena Agostini name, dev_id, gpu_count); 2348b8036a6SElena Agostini return dev; 2358b8036a6SElena Agostini } 2368b8036a6SElena Agostini 237*a9af048aSThomas Monjalon struct rte_gpu * 238*a9af048aSThomas Monjalon rte_gpu_attach(const char *name) 239*a9af048aSThomas Monjalon { 240*a9af048aSThomas Monjalon int16_t dev_id; 241*a9af048aSThomas Monjalon struct rte_gpu *dev; 242*a9af048aSThomas Monjalon struct rte_gpu_mpshared *shared_dev; 243*a9af048aSThomas Monjalon 244*a9af048aSThomas Monjalon if (rte_eal_process_type() != RTE_PROC_SECONDARY) { 245*a9af048aSThomas Monjalon GPU_LOG(ERR, "only secondary process can attach device"); 246*a9af048aSThomas Monjalon rte_errno = EPERM; 247*a9af048aSThomas Monjalon return NULL; 248*a9af048aSThomas Monjalon } 249*a9af048aSThomas Monjalon if (name == NULL) { 250*a9af048aSThomas Monjalon GPU_LOG(ERR, "attach device without a name"); 251*a9af048aSThomas Monjalon rte_errno = EINVAL; 252*a9af048aSThomas Monjalon return NULL; 253*a9af048aSThomas Monjalon } 254*a9af048aSThomas Monjalon 255*a9af048aSThomas Monjalon /* implicit initialization of library before adding first device */ 256*a9af048aSThomas Monjalon if (gpus == NULL && rte_gpu_init(RTE_GPU_DEFAULT_MAX) < 0) 257*a9af048aSThomas Monjalon return NULL; 258*a9af048aSThomas Monjalon 259*a9af048aSThomas Monjalon /* initialize shared memory before adding first device */ 260*a9af048aSThomas Monjalon if (gpu_shared_mem == NULL && gpu_shared_mem_init() < 0) 261*a9af048aSThomas Monjalon return NULL; 262*a9af048aSThomas Monjalon 263*a9af048aSThomas Monjalon for (dev_id = 0; dev_id < gpu_max; dev_id++) { 264*a9af048aSThomas Monjalon shared_dev = &gpu_shared_mem->gpus[dev_id]; 265*a9af048aSThomas Monjalon if (strncmp(name, shared_dev->name, RTE_DEV_NAME_MAX_LEN) == 0) 266*a9af048aSThomas Monjalon break; 267*a9af048aSThomas Monjalon } 268*a9af048aSThomas Monjalon if (dev_id >= gpu_max) { 269*a9af048aSThomas Monjalon GPU_LOG(ERR, "device with name %s not found", name); 270*a9af048aSThomas Monjalon rte_errno = ENOENT; 271*a9af048aSThomas Monjalon return NULL; 272*a9af048aSThomas Monjalon } 273*a9af048aSThomas Monjalon dev = &gpus[dev_id]; 274*a9af048aSThomas Monjalon memset(dev, 0, sizeof(*dev)); 275*a9af048aSThomas Monjalon 276*a9af048aSThomas Monjalon TAILQ_INIT(&dev->callbacks); 277*a9af048aSThomas Monjalon dev->mpshared = shared_dev; 278*a9af048aSThomas Monjalon __atomic_fetch_add(&dev->mpshared->process_refcnt, 1, __ATOMIC_RELAXED); 279*a9af048aSThomas Monjalon 280*a9af048aSThomas Monjalon gpu_count++; 281*a9af048aSThomas Monjalon GPU_LOG(DEBUG, "attached device %s (id %d) of total %d", 282*a9af048aSThomas Monjalon name, dev_id, gpu_count); 283*a9af048aSThomas Monjalon return dev; 284*a9af048aSThomas Monjalon } 285*a9af048aSThomas Monjalon 28682e5f6b6SThomas Monjalon int16_t 28782e5f6b6SThomas Monjalon rte_gpu_add_child(const char *name, int16_t parent, uint64_t child_context) 28882e5f6b6SThomas Monjalon { 28982e5f6b6SThomas Monjalon struct rte_gpu *dev; 29082e5f6b6SThomas Monjalon 29182e5f6b6SThomas Monjalon if (!rte_gpu_is_valid(parent)) { 29282e5f6b6SThomas Monjalon GPU_LOG(ERR, "add child to invalid parent ID %d", parent); 29382e5f6b6SThomas Monjalon rte_errno = ENODEV; 29482e5f6b6SThomas Monjalon return -rte_errno; 29582e5f6b6SThomas Monjalon } 29682e5f6b6SThomas Monjalon 29782e5f6b6SThomas Monjalon dev = rte_gpu_allocate(name); 29882e5f6b6SThomas Monjalon if (dev == NULL) 29982e5f6b6SThomas Monjalon return -rte_errno; 30082e5f6b6SThomas Monjalon 301*a9af048aSThomas Monjalon dev->mpshared->info.parent = parent; 302*a9af048aSThomas Monjalon dev->mpshared->info.context = child_context; 30382e5f6b6SThomas Monjalon 30482e5f6b6SThomas Monjalon rte_gpu_complete_new(dev); 305*a9af048aSThomas Monjalon return dev->mpshared->info.dev_id; 30682e5f6b6SThomas Monjalon } 30782e5f6b6SThomas Monjalon 3088b8036a6SElena Agostini void 3098b8036a6SElena Agostini rte_gpu_complete_new(struct rte_gpu *dev) 3108b8036a6SElena Agostini { 3118b8036a6SElena Agostini if (dev == NULL) 3128b8036a6SElena Agostini return; 3138b8036a6SElena Agostini 314*a9af048aSThomas Monjalon dev->process_state = RTE_GPU_STATE_INITIALIZED; 31518cb0756SThomas Monjalon rte_gpu_notify(dev, RTE_GPU_EVENT_NEW); 3168b8036a6SElena Agostini } 3178b8036a6SElena Agostini 3188b8036a6SElena Agostini int 3198b8036a6SElena Agostini rte_gpu_release(struct rte_gpu *dev) 3208b8036a6SElena Agostini { 32182e5f6b6SThomas Monjalon int16_t dev_id, child; 32282e5f6b6SThomas Monjalon 3238b8036a6SElena Agostini if (dev == NULL) { 3248b8036a6SElena Agostini rte_errno = ENODEV; 3258b8036a6SElena Agostini return -rte_errno; 3268b8036a6SElena Agostini } 327*a9af048aSThomas Monjalon dev_id = dev->mpshared->info.dev_id; 32882e5f6b6SThomas Monjalon RTE_GPU_FOREACH_CHILD(child, dev_id) { 32982e5f6b6SThomas Monjalon GPU_LOG(ERR, "cannot release device %d with child %d", 33082e5f6b6SThomas Monjalon dev_id, child); 33182e5f6b6SThomas Monjalon rte_errno = EBUSY; 33282e5f6b6SThomas Monjalon return -rte_errno; 33382e5f6b6SThomas Monjalon } 3348b8036a6SElena Agostini 3358b8036a6SElena Agostini GPU_LOG(DEBUG, "free device %s (id %d)", 336*a9af048aSThomas Monjalon dev->mpshared->info.name, dev->mpshared->info.dev_id); 33718cb0756SThomas Monjalon rte_gpu_notify(dev, RTE_GPU_EVENT_DEL); 33818cb0756SThomas Monjalon 33918cb0756SThomas Monjalon gpu_free_callbacks(dev); 340*a9af048aSThomas Monjalon dev->process_state = RTE_GPU_STATE_UNUSED; 341*a9af048aSThomas Monjalon __atomic_fetch_sub(&dev->mpshared->process_refcnt, 1, __ATOMIC_RELAXED); 3428b8036a6SElena Agostini gpu_count--; 3438b8036a6SElena Agostini 3448b8036a6SElena Agostini return 0; 3458b8036a6SElena Agostini } 3468b8036a6SElena Agostini 3478b8036a6SElena Agostini int 3488b8036a6SElena Agostini rte_gpu_close(int16_t dev_id) 3498b8036a6SElena Agostini { 3508b8036a6SElena Agostini int firsterr, binerr; 3518b8036a6SElena Agostini int *lasterr = &firsterr; 3528b8036a6SElena Agostini struct rte_gpu *dev; 3538b8036a6SElena Agostini 3548b8036a6SElena Agostini dev = gpu_get_by_id(dev_id); 3558b8036a6SElena Agostini if (dev == NULL) { 3568b8036a6SElena Agostini GPU_LOG(ERR, "close invalid device ID %d", dev_id); 3578b8036a6SElena Agostini rte_errno = ENODEV; 3588b8036a6SElena Agostini return -rte_errno; 3598b8036a6SElena Agostini } 3608b8036a6SElena Agostini 3618b8036a6SElena Agostini if (dev->ops.dev_close != NULL) { 3628b8036a6SElena Agostini *lasterr = GPU_DRV_RET(dev->ops.dev_close(dev)); 3638b8036a6SElena Agostini if (*lasterr != 0) 3648b8036a6SElena Agostini lasterr = &binerr; 3658b8036a6SElena Agostini } 3668b8036a6SElena Agostini 3678b8036a6SElena Agostini *lasterr = rte_gpu_release(dev); 3688b8036a6SElena Agostini 3698b8036a6SElena Agostini rte_errno = -firsterr; 3708b8036a6SElena Agostini return firsterr; 3718b8036a6SElena Agostini } 3728b8036a6SElena Agostini 3738b8036a6SElena Agostini int 37418cb0756SThomas Monjalon rte_gpu_callback_register(int16_t dev_id, enum rte_gpu_event event, 37518cb0756SThomas Monjalon rte_gpu_callback_t *function, void *user_data) 37618cb0756SThomas Monjalon { 37718cb0756SThomas Monjalon int16_t next_dev, last_dev; 37818cb0756SThomas Monjalon struct rte_gpu_callback_list *callbacks; 37918cb0756SThomas Monjalon struct rte_gpu_callback *callback; 38018cb0756SThomas Monjalon 38118cb0756SThomas Monjalon if (!rte_gpu_is_valid(dev_id) && dev_id != RTE_GPU_ID_ANY) { 38218cb0756SThomas Monjalon GPU_LOG(ERR, "register callback of invalid ID %d", dev_id); 38318cb0756SThomas Monjalon rte_errno = ENODEV; 38418cb0756SThomas Monjalon return -rte_errno; 38518cb0756SThomas Monjalon } 38618cb0756SThomas Monjalon if (function == NULL) { 38718cb0756SThomas Monjalon GPU_LOG(ERR, "cannot register callback without function"); 38818cb0756SThomas Monjalon rte_errno = EINVAL; 38918cb0756SThomas Monjalon return -rte_errno; 39018cb0756SThomas Monjalon } 39118cb0756SThomas Monjalon 39218cb0756SThomas Monjalon if (dev_id == RTE_GPU_ID_ANY) { 39318cb0756SThomas Monjalon next_dev = 0; 39418cb0756SThomas Monjalon last_dev = gpu_max - 1; 39518cb0756SThomas Monjalon } else { 39618cb0756SThomas Monjalon next_dev = last_dev = dev_id; 39718cb0756SThomas Monjalon } 39818cb0756SThomas Monjalon 39918cb0756SThomas Monjalon rte_rwlock_write_lock(&gpu_callback_lock); 40018cb0756SThomas Monjalon do { 40118cb0756SThomas Monjalon callbacks = &gpus[next_dev].callbacks; 40218cb0756SThomas Monjalon 40318cb0756SThomas Monjalon /* check if not already registered */ 40418cb0756SThomas Monjalon TAILQ_FOREACH(callback, callbacks, next) { 40518cb0756SThomas Monjalon if (callback->event == event && 40618cb0756SThomas Monjalon callback->function == function && 40718cb0756SThomas Monjalon callback->user_data == user_data) { 40818cb0756SThomas Monjalon GPU_LOG(INFO, "callback already registered"); 40918cb0756SThomas Monjalon return 0; 41018cb0756SThomas Monjalon } 41118cb0756SThomas Monjalon } 41218cb0756SThomas Monjalon 41318cb0756SThomas Monjalon callback = malloc(sizeof(*callback)); 41418cb0756SThomas Monjalon if (callback == NULL) { 41518cb0756SThomas Monjalon GPU_LOG(ERR, "cannot allocate callback"); 41618cb0756SThomas Monjalon return -ENOMEM; 41718cb0756SThomas Monjalon } 41818cb0756SThomas Monjalon callback->function = function; 41918cb0756SThomas Monjalon callback->user_data = user_data; 42018cb0756SThomas Monjalon callback->event = event; 42118cb0756SThomas Monjalon TAILQ_INSERT_TAIL(callbacks, callback, next); 42218cb0756SThomas Monjalon 42318cb0756SThomas Monjalon } while (++next_dev <= last_dev); 42418cb0756SThomas Monjalon rte_rwlock_write_unlock(&gpu_callback_lock); 42518cb0756SThomas Monjalon 42618cb0756SThomas Monjalon return 0; 42718cb0756SThomas Monjalon } 42818cb0756SThomas Monjalon 42918cb0756SThomas Monjalon int 43018cb0756SThomas Monjalon rte_gpu_callback_unregister(int16_t dev_id, enum rte_gpu_event event, 43118cb0756SThomas Monjalon rte_gpu_callback_t *function, void *user_data) 43218cb0756SThomas Monjalon { 43318cb0756SThomas Monjalon int16_t next_dev, last_dev; 43418cb0756SThomas Monjalon struct rte_gpu_callback_list *callbacks; 43518cb0756SThomas Monjalon struct rte_gpu_callback *callback, *nextcb; 43618cb0756SThomas Monjalon 43718cb0756SThomas Monjalon if (!rte_gpu_is_valid(dev_id) && dev_id != RTE_GPU_ID_ANY) { 43818cb0756SThomas Monjalon GPU_LOG(ERR, "unregister callback of invalid ID %d", dev_id); 43918cb0756SThomas Monjalon rte_errno = ENODEV; 44018cb0756SThomas Monjalon return -rte_errno; 44118cb0756SThomas Monjalon } 44218cb0756SThomas Monjalon if (function == NULL) { 44318cb0756SThomas Monjalon GPU_LOG(ERR, "cannot unregister callback without function"); 44418cb0756SThomas Monjalon rte_errno = EINVAL; 44518cb0756SThomas Monjalon return -rte_errno; 44618cb0756SThomas Monjalon } 44718cb0756SThomas Monjalon 44818cb0756SThomas Monjalon if (dev_id == RTE_GPU_ID_ANY) { 44918cb0756SThomas Monjalon next_dev = 0; 45018cb0756SThomas Monjalon last_dev = gpu_max - 1; 45118cb0756SThomas Monjalon } else { 45218cb0756SThomas Monjalon next_dev = last_dev = dev_id; 45318cb0756SThomas Monjalon } 45418cb0756SThomas Monjalon 45518cb0756SThomas Monjalon rte_rwlock_write_lock(&gpu_callback_lock); 45618cb0756SThomas Monjalon do { 45718cb0756SThomas Monjalon callbacks = &gpus[next_dev].callbacks; 45818cb0756SThomas Monjalon RTE_TAILQ_FOREACH_SAFE(callback, callbacks, next, nextcb) { 45918cb0756SThomas Monjalon if (callback->event != event || 46018cb0756SThomas Monjalon callback->function != function || 46118cb0756SThomas Monjalon (callback->user_data != user_data && 46218cb0756SThomas Monjalon user_data != (void *)-1)) 46318cb0756SThomas Monjalon continue; 46418cb0756SThomas Monjalon TAILQ_REMOVE(callbacks, callback, next); 46518cb0756SThomas Monjalon free(callback); 46618cb0756SThomas Monjalon } 46718cb0756SThomas Monjalon } while (++next_dev <= last_dev); 46818cb0756SThomas Monjalon rte_rwlock_write_unlock(&gpu_callback_lock); 46918cb0756SThomas Monjalon 47018cb0756SThomas Monjalon return 0; 47118cb0756SThomas Monjalon } 47218cb0756SThomas Monjalon 47318cb0756SThomas Monjalon static void 47418cb0756SThomas Monjalon gpu_free_callbacks(struct rte_gpu *dev) 47518cb0756SThomas Monjalon { 47618cb0756SThomas Monjalon struct rte_gpu_callback_list *callbacks; 47718cb0756SThomas Monjalon struct rte_gpu_callback *callback, *nextcb; 47818cb0756SThomas Monjalon 47918cb0756SThomas Monjalon callbacks = &dev->callbacks; 48018cb0756SThomas Monjalon rte_rwlock_write_lock(&gpu_callback_lock); 48118cb0756SThomas Monjalon RTE_TAILQ_FOREACH_SAFE(callback, callbacks, next, nextcb) { 48218cb0756SThomas Monjalon TAILQ_REMOVE(callbacks, callback, next); 48318cb0756SThomas Monjalon free(callback); 48418cb0756SThomas Monjalon } 48518cb0756SThomas Monjalon rte_rwlock_write_unlock(&gpu_callback_lock); 48618cb0756SThomas Monjalon } 48718cb0756SThomas Monjalon 48818cb0756SThomas Monjalon void 48918cb0756SThomas Monjalon rte_gpu_notify(struct rte_gpu *dev, enum rte_gpu_event event) 49018cb0756SThomas Monjalon { 49118cb0756SThomas Monjalon int16_t dev_id; 49218cb0756SThomas Monjalon struct rte_gpu_callback *callback; 49318cb0756SThomas Monjalon 494*a9af048aSThomas Monjalon dev_id = dev->mpshared->info.dev_id; 49518cb0756SThomas Monjalon rte_rwlock_read_lock(&gpu_callback_lock); 49618cb0756SThomas Monjalon TAILQ_FOREACH(callback, &dev->callbacks, next) { 49718cb0756SThomas Monjalon if (callback->event != event || callback->function == NULL) 49818cb0756SThomas Monjalon continue; 49918cb0756SThomas Monjalon callback->function(dev_id, event, callback->user_data); 50018cb0756SThomas Monjalon } 50118cb0756SThomas Monjalon rte_rwlock_read_unlock(&gpu_callback_lock); 50218cb0756SThomas Monjalon } 50318cb0756SThomas Monjalon 50418cb0756SThomas Monjalon int 5058b8036a6SElena Agostini rte_gpu_info_get(int16_t dev_id, struct rte_gpu_info *info) 5068b8036a6SElena Agostini { 5078b8036a6SElena Agostini struct rte_gpu *dev; 5088b8036a6SElena Agostini 5098b8036a6SElena Agostini dev = gpu_get_by_id(dev_id); 5108b8036a6SElena Agostini if (dev == NULL) { 5118b8036a6SElena Agostini GPU_LOG(ERR, "query invalid device ID %d", dev_id); 5128b8036a6SElena Agostini rte_errno = ENODEV; 5138b8036a6SElena Agostini return -rte_errno; 5148b8036a6SElena Agostini } 5158b8036a6SElena Agostini if (info == NULL) { 5168b8036a6SElena Agostini GPU_LOG(ERR, "query without storage"); 5178b8036a6SElena Agostini rte_errno = EINVAL; 5188b8036a6SElena Agostini return -rte_errno; 5198b8036a6SElena Agostini } 5208b8036a6SElena Agostini 5218b8036a6SElena Agostini if (dev->ops.dev_info_get == NULL) { 522*a9af048aSThomas Monjalon *info = dev->mpshared->info; 5238b8036a6SElena Agostini return 0; 5248b8036a6SElena Agostini } 5258b8036a6SElena Agostini return GPU_DRV_RET(dev->ops.dev_info_get(dev, info)); 5268b8036a6SElena Agostini } 527