/* * IOThread Virtqueue Mapping * * Copyright Red Hat, Inc * * SPDX-License-Identifier: GPL-2.0-only */ #include "qemu/osdep.h" #include "system/iothread.h" #include "hw/virtio/iothread-vq-mapping.h" static bool iothread_vq_mapping_validate(IOThreadVirtQueueMappingList *list, uint16_t num_queues, Error **errp) { g_autofree unsigned long *vqs = bitmap_new(num_queues); g_autoptr(GHashTable) iothreads = g_hash_table_new(g_str_hash, g_str_equal); for (IOThreadVirtQueueMappingList *node = list; node; node = node->next) { const char *name = node->value->iothread; uint16List *vq; if (!iothread_by_id(name)) { error_setg(errp, "IOThread \"%s\" object does not exist", name); return false; } if (!g_hash_table_add(iothreads, (gpointer)name)) { error_setg(errp, "duplicate IOThread name \"%s\" in iothread-vq-mapping", name); return false; } if (node != list) { if (!!node->value->vqs != !!list->value->vqs) { error_setg(errp, "either all items in iothread-vq-mapping " "must have vqs or none of them must have it"); return false; } } for (vq = node->value->vqs; vq; vq = vq->next) { if (vq->value >= num_queues) { error_setg(errp, "vq index %u for IOThread \"%s\" must be " "less than num_queues %u in iothread-vq-mapping", vq->value, name, num_queues); return false; } if (test_and_set_bit(vq->value, vqs)) { error_setg(errp, "cannot assign vq %u to IOThread \"%s\" " "because it is already assigned", vq->value, name); return false; } } } if (list->value->vqs) { for (uint16_t i = 0; i < num_queues; i++) { if (!test_bit(i, vqs)) { error_setg(errp, "missing vq %u IOThread assignment in iothread-vq-mapping", i); return false; } } } return true; } bool iothread_vq_mapping_apply( IOThreadVirtQueueMappingList *list, AioContext **vq_aio_context, uint16_t num_queues, Error **errp) { IOThreadVirtQueueMappingList *node; size_t num_iothreads = 0; size_t cur_iothread = 0; if (!iothread_vq_mapping_validate(list, num_queues, errp)) { return false; } for (node = list; node; node = node->next) { num_iothreads++; } for (node = list; node; node = node->next) { IOThread *iothread = iothread_by_id(node->value->iothread); AioContext *ctx = iothread_get_aio_context(iothread); /* Released in virtio_blk_vq_aio_context_cleanup() */ object_ref(OBJECT(iothread)); if (node->value->vqs) { uint16List *vq; /* Explicit vq:IOThread assignment */ for (vq = node->value->vqs; vq; vq = vq->next) { assert(vq->value < num_queues); vq_aio_context[vq->value] = ctx; } } else { /* Round-robin vq:IOThread assignment */ for (unsigned i = cur_iothread; i < num_queues; i += num_iothreads) { vq_aio_context[i] = ctx; } } cur_iothread++; } return true; } void iothread_vq_mapping_cleanup(IOThreadVirtQueueMappingList *list) { IOThreadVirtQueueMappingList *node; for (node = list; node; node = node->next) { IOThread *iothread = iothread_by_id(node->value->iothread); object_unref(OBJECT(iothread)); } }