本文主要是介绍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函数的深入解析,请看下回。
这篇关于QEMU源码全解析 —— virtio(10)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!