Skip to content

Commit f272f31

Browse files
akihikodakiMichael Tokarev
authored andcommitted
virtio-net: Add queues for RSS during migration
virtio_net_pre_load_queues() inspects vdev->guest_features to tell if VIRTIO_NET_F_RSS or VIRTIO_NET_F_MQ is enabled to infer the required number of queues. This works for VIRTIO_NET_F_MQ but it doesn't for VIRTIO_NET_F_RSS because only the lowest 32 bits of vdev->guest_features is set at the point and VIRTIO_NET_F_RSS uses bit 60 while VIRTIO_NET_F_MQ uses bit 22. Instead of inferring the required number of queues from vdev->guest_features, use the number loaded from the vm state. This change also has a nice side effect to remove a duplicate peer queue pair change by circumventing virtio_net_set_multiqueue(). Also update the comment in include/hw/virtio/virtio.h to prevent an implementation of pre_load_queues() from refering to any fields being loaded during migration by accident in the future. Fixes: 8c49756 ("virtio-net: Add only one queue pair when realizing") Tested-by: Lei Yang <[email protected]> Cc: [email protected] Signed-off-by: Akihiko Odaki <[email protected]> Signed-off-by: Jason Wang <[email protected]> (cherry picked from commit adda0ad56bd28d5a809051cbd190fda5798ec4e4) Signed-off-by: Michael Tokarev <[email protected]>
1 parent 6624ff3 commit f272f31

File tree

3 files changed

+19
-16
lines changed

3 files changed

+19
-16
lines changed

hw/net/virtio-net.c

Lines changed: 4 additions & 7 deletions
Original file line numberDiff line numberDiff line change
@@ -3021,11 +3021,10 @@ static void virtio_net_del_queue(VirtIONet *n, int index)
30213021
virtio_del_queue(vdev, index * 2 + 1);
30223022
}
30233023

3024-
static void virtio_net_change_num_queue_pairs(VirtIONet *n, int new_max_queue_pairs)
3024+
static void virtio_net_change_num_queues(VirtIONet *n, int new_num_queues)
30253025
{
30263026
VirtIODevice *vdev = VIRTIO_DEVICE(n);
30273027
int old_num_queues = virtio_get_num_queues(vdev);
3028-
int new_num_queues = new_max_queue_pairs * 2 + 1;
30293028
int i;
30303029

30313030
assert(old_num_queues >= 3);
@@ -3061,16 +3060,14 @@ static void virtio_net_set_multiqueue(VirtIONet *n, int multiqueue)
30613060
int max = multiqueue ? n->max_queue_pairs : 1;
30623061

30633062
n->multiqueue = multiqueue;
3064-
virtio_net_change_num_queue_pairs(n, max);
3063+
virtio_net_change_num_queues(n, max * 2 + 1);
30653064

30663065
virtio_net_set_queue_pairs(n);
30673066
}
30683067

3069-
static int virtio_net_pre_load_queues(VirtIODevice *vdev)
3068+
static int virtio_net_pre_load_queues(VirtIODevice *vdev, uint32_t n)
30703069
{
3071-
virtio_net_set_multiqueue(VIRTIO_NET(vdev),
3072-
virtio_has_feature(vdev->guest_features, VIRTIO_NET_F_RSS) ||
3073-
virtio_has_feature(vdev->guest_features, VIRTIO_NET_F_MQ));
3070+
virtio_net_change_num_queues(VIRTIO_NET(vdev), n);
30743071

30753072
return 0;
30763073
}

hw/virtio/virtio.c

Lines changed: 7 additions & 7 deletions
Original file line numberDiff line numberDiff line change
@@ -3257,20 +3257,20 @@ virtio_load(VirtIODevice *vdev, QEMUFile *f, int version_id)
32573257
config_len--;
32583258
}
32593259

3260-
if (vdc->pre_load_queues) {
3261-
ret = vdc->pre_load_queues(vdev);
3262-
if (ret) {
3263-
return ret;
3264-
}
3265-
}
3266-
32673260
num = qemu_get_be32(f);
32683261

32693262
if (num > VIRTIO_QUEUE_MAX) {
32703263
error_report("Invalid number of virtqueues: 0x%x", num);
32713264
return -1;
32723265
}
32733266

3267+
if (vdc->pre_load_queues) {
3268+
ret = vdc->pre_load_queues(vdev, num);
3269+
if (ret) {
3270+
return ret;
3271+
}
3272+
}
3273+
32743274
for (i = 0; i < num; i++) {
32753275
vdev->vq[i].vring.num = qemu_get_be32(f);
32763276
if (k->has_variable_vring_alignment) {

include/hw/virtio/virtio.h

Lines changed: 8 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -210,8 +210,14 @@ struct VirtioDeviceClass {
210210
void (*guest_notifier_mask)(VirtIODevice *vdev, int n, bool mask);
211211
int (*start_ioeventfd)(VirtIODevice *vdev);
212212
void (*stop_ioeventfd)(VirtIODevice *vdev);
213-
/* Called before loading queues. Useful to add queues before loading. */
214-
int (*pre_load_queues)(VirtIODevice *vdev);
213+
/*
214+
* Called before loading queues.
215+
* If the number of queues change at runtime, use @n to know the
216+
* number and add or remove queues accordingly.
217+
* Note that this function is called in the middle of loading vmsd;
218+
* no assumption should be made on states being loaded from vmsd.
219+
*/
220+
int (*pre_load_queues)(VirtIODevice *vdev, uint32_t n);
215221
/* Saving and loading of a device; trying to deprecate save/load
216222
* use vmsd for new devices.
217223
*/

0 commit comments

Comments
 (0)