接前一篇文章:
上一回开始,沿着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函数的深入解析,请看下回。