QEMU源码全解析 —— virtio(12)

88 篇文章 19 订阅
本文详细解析了QEMU中virtio balloon设备的实现,重点介绍了virtio_balloon_device_realize函数。该函数首先调用virtio_init初始化设备的公共部分,然后通过virtio_add_queue创建3个virtqueue,用于与虚拟机的数据传输。这些virtqueue使用VirtQueue结构表示,并设置处理输出的回调函数。
摘要由CSDN通过智能技术生成

接前一篇文章:

上一回对于virtio_device_realize函数进行了详细解析。在第2步中virtio_device_realize 函数调用 了具体类的realize函数,对于virtio balloon设备来说是virtio_balloon_device_realize函数。本回就来对于virtio_balloon_device_realize函数进行解析。

为了便于理解,再次贴出virtio_device_realize函数源码,在hw/ virtio /virtio.c中,如下:

  1. static void virtio_device_realize(DeviceState *dev, Error **errp)
  2. {
  3. VirtIODevice *vdev = VIRTIO_DEVICE(dev);
  4. VirtioDeviceClass *vdc = VIRTIO_DEVICE_GET_CLASS(dev);
  5. Error *err = NULL;
  6. /* Devices should either use vmsd or the load/save methods */
  7. assert(!vdc->vmsd || !vdc->load);
  8. if (vdc->realize != NULL) {
  9. vdc->realize(dev, &err);
  10. if (err != NULL) {
  11. error_propagate(errp, err);
  12. return;
  13. }
  14. }
  15. virtio_bus_device_plugged(vdev, &err);
  16. if (err != NULL) {
  17. error_propagate(errp, err);
  18. vdc->unrealize(dev);
  19. return;
  20. }
  21. vdev->listener.commit = virtio_memory_listener_commit;
  22. vdev->listener.name = "virtio";
  23. memory_listener_register(&vdev->listener, vdev->dma_as);
  24. QTAILQ_INSERT_TAIL(&virtio_list, vdev, next);
  25. }

调用具体类的realize函数的代码片段如下:

  1. if (vdc->realize != NULL) {
  2. vdc->realize(dev, &err);
  3. if (err != NULL) {
  4. error_propagate(errp, err);
  5. return;
  6. }
  7. }

virtio_balloon_device_realize函数在hw/virtio/virtio-balloon.c中,代码如下:

  1. static void virtio_balloon_device_realize(DeviceState *dev, Error **errp)
  2. {
  3. VirtIODevice *vdev = VIRTIO_DEVICE(dev);
  4. VirtIOBalloon *s = VIRTIO_BALLOON(dev);
  5. int ret;
  6. virtio_init(vdev, VIRTIO_ID_BALLOON, virtio_balloon_config_size(s));
  7. ret = qemu_add_balloon_handler(virtio_balloon_to_target,
  8. virtio_balloon_stat, s);
  9. if (ret < 0) {
  10. error_setg(errp, "Only one balloon device is supported");
  11. virtio_cleanup(vdev);
  12. return;
  13. }
  14. if (virtio_has_feature(s->host_features, VIRTIO_BALLOON_F_FREE_PAGE_HINT) &&
  15. !s->iothread) {
  16. error_setg(errp, "'free-page-hint' requires 'iothread' to be set");
  17. virtio_cleanup(vdev);
  18. return;
  19. }
  20. s->ivq = virtio_add_queue(vdev, 128, virtio_balloon_handle_output);
  21. s->dvq = virtio_add_queue(vdev, 128, virtio_balloon_handle_output);
  22. s->svq = virtio_add_queue(vdev, 128, virtio_balloon_receive_stats);
  23. if (virtio_has_feature(s->host_features, VIRTIO_BALLOON_F_FREE_PAGE_HINT)) {
  24. s->free_page_vq = virtio_add_queue(vdev, VIRTQUEUE_MAX_SIZE,
  25. virtio_balloon_handle_free_page_vq);
  26. precopy_add_notifier(&s->free_page_hint_notify);
  27. object_ref(OBJECT(s->iothread));
  28. s->free_page_bh = aio_bh_new_guarded(iothread_get_aio_context(s->iothread),
  29. virtio_ballloon_get_free_page_hints, s,
  30. &dev->mem_reentrancy_guard);
  31. }
  32. if (virtio_has_feature(s->host_features, VIRTIO_BALLOON_F_REPORTING)) {
  33. s->reporting_vq = virtio_add_queue(vdev, 32,
  34. virtio_balloon_handle_report);
  35. }
  36. reset_stats(s);
  37. }

在同文件(hw/virtio/virtio-balloon.c)的virtio_balloon_class_init函数中,将virtio_balloon_device_realize函数(地址)赋给了vdc->realize。

  1. static void virtio_balloon_class_init(ObjectClass *klass, void *data)
  2. {
  3. DeviceClass *dc = DEVICE_CLASS(klass);
  4. VirtioDeviceClass *vdc = VIRTIO_DEVICE_CLASS(klass);
  5. device_class_set_props(dc, virtio_balloon_properties);
  6. dc->vmsd = &vmstate_virtio_balloon;
  7. set_bit(DEVICE_CATEGORY_MISC, dc->categories);
  8. vdc->realize = virtio_balloon_device_realize;
  9. vdc->unrealize = virtio_balloon_device_unrealize;
  10. vdc->reset = virtio_balloon_device_reset;
  11. vdc->get_config = virtio_balloon_get_config;
  12. vdc->set_config = virtio_balloon_set_config;
  13. vdc->get_features = virtio_balloon_get_features;
  14. vdc->set_status = virtio_balloon_set_status;
  15. vdc->vmsd = &vmstate_virtio_balloon_device;
  16. }

virtio_balloon_device_realize()是virtio balloon设备的具现化函数,它用于实现TYPE_VIRTIO_BALLOON_DEVICE的具现化。

(1)virtio_balloon_device_realize函数首先调用virtio_init函数初始化virtio设备的公共部分。代码片段如下:

    virtio_init(vdev, VIRTIO_ID_BALLOON, virtio_balloon_config_size(s));

virtio_init函数在hw/virtio/virtio.c中,代码如下:

  1. void virtio_init(VirtIODevice *vdev, uint16_t device_id, size_t config_size)
  2. {
  3. BusState *qbus = qdev_get_parent_bus(DEVICE(vdev));
  4. VirtioBusClass *k = VIRTIO_BUS_GET_CLASS(qbus);
  5. int i;
  6. int nvectors = k->query_nvectors ? k->query_nvectors(qbus->parent) : 0;
  7. if (nvectors) {
  8. vdev->vector_queues =
  9. g_malloc0(sizeof(*vdev->vector_queues) * nvectors);
  10. }
  11. vdev->start_on_kick = false;
  12. vdev->started = false;
  13. vdev->vhost_started = false;
  14. vdev->device_id = device_id;
  15. vdev->status = 0;
  16. qatomic_set(&vdev->isr, 0);
  17. vdev->queue_sel = 0;
  18. vdev->config_vector = VIRTIO_NO_VECTOR;
  19. vdev->vq = g_new0(VirtQueue, VIRTIO_QUEUE_MAX);
  20. vdev->vm_running = runstate_is_running();
  21. vdev->broken = false;
  22. for (i = 0; i < VIRTIO_QUEUE_MAX; i++) {
  23. vdev->vq[i].vector = VIRTIO_NO_VECTOR;
  24. vdev->vq[i].vdev = vdev;
  25. vdev->vq[i].queue_index = i;
  26. vdev->vq[i].host_notifier_enabled = false;
  27. }
  28. vdev->name = virtio_id_to_name(device_id);
  29. vdev->config_len = config_size;
  30. if (vdev->config_len) {
  31. vdev->config = g_malloc0(config_size);
  32. } else {
  33. vdev->config = NULL;
  34. }
  35. vdev->vmstate = qdev_add_vm_change_state_handler(DEVICE(vdev),
  36. virtio_vmstate_change, vdev);
  37. vdev->device_endian = virtio_default_endian();
  38. vdev->use_guest_notifier_mask = true;
  39. }

virtio_init函数的工作是初始化所有virtio设备的基类TYPE_VIRTIO_DEVICE的实例VirtIODevice结构体,其对VirtIODevice的成员进行初始化。

VirtIODevice的vector_queues成员和config_vector成员与MSI中断相关;device_id、status、name成员分别表示设备的id、状态和名字;isr成员用来表示中断请求;queue_sel成员用来在进行配置队列的时候选择队列;vq成员表示的是该设备的virtio queue,这里分配了VIRTIO_QUEUE_MAX个queue,并且进行了初始化;config_len和config分别表示该virtio设备配置空间的长度和数据存放区域;use_guest_notifier_mask成员与irqfd有关。

回到virtio_balloon_device_realize函数。

(2)在virtio_init函数初始化了VirtIODevice之后,调用virtio_add_queue函数创建了3个virtqueue。代码片段如下:

  1. s->ivq = virtio_add_queue(vdev, 128, virtio_balloon_handle_output);
  2. s->dvq = virtio_add_queue(vdev, 128, virtio_balloon_handle_output);
  3. s->svq = virtio_add_queue(vdev, 128, virtio_balloon_receive_stats);

virtqueue是virtio设备的重要组成部分,用来与 虚拟机 中的操作系统进行数据传输。virtio_add_queue函数在hw/virtio/virtio.c中,代码如下:

  1. VirtQueue *virtio_add_queue(VirtIODevice *vdev, int queue_size,
  2. VirtIOHandleOutput handle_output)
  3. {
  4. int i;
  5. for (i = 0; i < VIRTIO_QUEUE_MAX; i++) {
  6. if (vdev->vq[i].vring.num == 0)
  7. break;
  8. }
  9. if (i == VIRTIO_QUEUE_MAX || queue_size > VIRTQUEUE_MAX_SIZE)
  10. abort();
  11. vdev->vq[i].vring.num = queue_size;
  12. vdev->vq[i].vring.num_default = queue_size;
  13. vdev->vq[i].vring.align = VIRTIO_PCI_VRING_ALIGN;
  14. vdev->vq[i].handle_output = handle_output;
  15. vdev->vq[i].used_elems = g_new0(VirtQueueElement, queue_size);
  16. return &vdev->vq[i];
  17. }

virtio_add_queue函数是virtio框架中用来添加virtqueue的接口,其3个参数分别表示要添加的设备(VirtIODevice *vdev)、virtqueue的大小(int queue_size)以及处理函数(VirtIOHandleOutput handle_output)。

virtio_add_queue函数从VirtIODevice的vq数组成员中找到还未被使用的一个queue。一个virtqueue使用VirtQueue结构表示,这里对VirtQueue的成员进行初始化,包括这个queue的大小以及对齐等信息。最重要的是设置VirtQueue的handle_output成员,其是一个函数指针,在收到虚拟机发过来的IO请求时,会调用存放在handle_output中的回调函数。

VirtQueue结构的定义在hw/virtio/virtio.c中,如下:

  1. struct VirtQueue
  2. {
  3. VRing vring;
  4. VirtQueueElement *used_elems;
  5. /* Next head to pop */
  6. uint16_t last_avail_idx;
  7. bool last_avail_wrap_counter;
  8. /* Last avail_idx read from VQ. */
  9. uint16_t shadow_avail_idx;
  10. bool shadow_avail_wrap_counter;
  11. uint16_t used_idx;
  12. bool used_wrap_counter;
  13. /* Last used index value we have signalled on */
  14. uint16_t signalled_used;
  15. /* Last used index value we have signalled on */
  16. bool signalled_used_valid;
  17. /* Notification enabled? */
  18. bool notification;
  19. uint16_t queue_index;
  20. unsigned int inuse;
  21. uint16_t vector;
  22. VirtIOHandleOutput handle_output;
  23. VirtIODevice *vdev;
  24. EventNotifier guest_notifier;
  25. EventNotifier host_notifier;
  26. bool host_notifier_enabled;
  27. QLIST_ENTRY(VirtQueue) node;
  28. };

欲知后事如何,且看下回分解。

举报

选择你想要举报的内容(必选)
  • 内容涉黄
  • 政治相关
  • 内容抄袭
  • 涉嫌广告
  • 内容侵权
  • 侮辱谩骂
  • 样式问题
  • 其他