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

接前一篇文章:

上一回开始,沿着device_set_realized() -> pci_qdev_realize() -> virtio_pci_realize()这一调用线,对于virtio_pci_realize函数进行了详细解析。最后,讲到virtio_pci_realize函数的最后一步调用了virtio_balloon_pci_realize函数。本回就来对于virtio_balloon_pci_realize函数进行解析。

virtio_balloon_pci_realize函数在hw/virtio/virtio-balloon-pci.c中,如下:

static void virtio_balloon_pci_realize(VirtIOPCIProxy *vpci_dev, Error **errp)
{
    VirtIOBalloonPCI *dev = VIRTIO_BALLOON_PCI(vpci_dev);
    DeviceState *vdev = DEVICE(&dev->vdev);

    vpci_dev->class_code = PCI_CLASS_OTHERS;
    qdev_realize(vdev, BUS(&vpci_dev->bus), errp);
}

virtio_balloon_pci_realize函数首先通过VIRTIO_BALLOON_PCI宏将VirtIOPCIxy类型的设备转换为VirtIOBalloonPCI设备,相当于从一个父类转换为一个子类。代码片段为:

    VirtIOBalloonPCI *dev = VIRTIO_BALLOON_PCI(vpci_dev);

接着,得到设备VirtIOBalloonPCI的VirtIOBalloon部分,这就是实际的virtio balloon设备了。.代码片段为:

    DeviceState *vdev = DEVICE(&dev->vdev);

接下来,调用hw/core/qdev.c中的qdev_realize函数。代码片段如下:

    qdev_realize(vdev, BUS(&vpci_dev->bus), errp);

qdev_realize函数代码如下:

bool qdev_realize(DeviceState *dev, BusState *bus, Error **errp)
{
    assert(!dev->realized && !dev->parent_bus);

    if (bus) {
        if (!qdev_set_parent_bus(dev, bus, errp)) {
            return false;
        }
    } else {
        assert(!DEVICE_GET_CLASS(dev)->bus_type);
    }

    return object_property_set_bool(OBJECT(dev), "realized", true, errp);
}

qdev_realize函数主要做了两件事情:

1)设置virtio balloon设备的总线为VirtIOPCIProxy设备中的bus成员,也就是把这个virtio balloon设备挂到virtio总线上。

2)调用object_property_bool函数,将virtio balloon设备具现化,这会导致virtio_device_realize函数的执行。

注:实际上在老版本中,并没有qdev_realize函数,它里边的两项工作是直接在virtio_balloon_pci_realize函数中完成的。

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

static void virtio_device_realize(DeviceState *dev, Error **errp)
{
    VirtIODevice *vdev = VIRTIO_DEVICE(dev);
    VirtioDeviceClass *vdc = VIRTIO_DEVICE_GET_CLASS(dev);
    Error *err = NULL;

    /* Devices should either use vmsd or the load/save methods */
    assert(!vdc->vmsd || !vdc->load);

    if (vdc->realize != NULL) {
        vdc->realize(dev, &err);
        if (err != NULL) {
            error_propagate(errp, err);
            return;
        }
    }

    virtio_bus_device_plugged(vdev, &err);
    if (err != NULL) {
        error_propagate(errp, err);
        vdc->unrealize(dev);
        return;
    }

    vdev->listener.commit = virtio_memory_listener_commit;
    vdev->listener.name = "virtio";
    memory_listener_register(&vdev->listener, vdev->dma_as);
    QTAILQ_INSERT_TAIL(&virtio_list, vdev, next);
}

对于virtio_device_realize函数的深入解析,请看下回。

相关推荐

  1. QEMU解析 —— virtio10

    2023-12-18 03:46:02       65 阅读
  2. QEMU解析 —— virtio12

    2023-12-18 03:46:02       49 阅读
  3. QEMU解析 —— virtio(6)

    2023-12-18 03:46:02       71 阅读
  4. QEMU解析 —— virtio(8)

    2023-12-18 03:46:02       49 阅读
  5. QEMU解析 —— virtio(27)

    2023-12-18 03:46:02       37 阅读
  6. QEMU解析 —— PCI设备模拟(6)

    2023-12-18 03:46:02       61 阅读

最近更新

  1. docker php8.1+nginx base 镜像 dockerfile 配置

    2023-12-18 03:46:02       98 阅读
  2. Could not load dynamic library ‘cudart64_100.dll‘

    2023-12-18 03:46:02       106 阅读
  3. 在Django里面运行非项目文件

    2023-12-18 03:46:02       87 阅读
  4. Python语言-面向对象

    2023-12-18 03:46:02       96 阅读

热门阅读

  1. 解决Qt UI界面卡顿的优化方法

    2023-12-18 03:46:02       70 阅读
  2. C++11——lambda

    2023-12-18 03:46:02       56 阅读
  3. 【力扣100】141.环形链表

    2023-12-18 03:46:02       58 阅读
  4. 重启docker容器后,ssh无法访问且浏览器无法访问

    2023-12-18 03:46:02       43 阅读
  5. 构建一个简单的 npm 验证项目

    2023-12-18 03:46:02       59 阅读