mirror of https://gitlab.com/qemu-project/qemu
You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
132 lines
3.7 KiB
C
132 lines
3.7 KiB
C
/*
|
|
* 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));
|
|
}
|
|
}
|
|
|