KVM-QEMU vdagent实现机制分析(二)

1.       虚拟串口模块的初始化注册

QEMU中与虚拟串口(virtio-serial)通信相关的文件主要有以下几个:

Virtio-pci.c

virtio-serial-bus.c(h)            //从名字上看也像

virtio-console.c

vl.c                         //qemumain函数在这里

Virtio-pci.c的最后一行是这样的:

type_init(virtio_pci_register_types)

virtio-serial-bus.c最后一行是这样的:

type_init(virtio_serial_register_types)

Type_init是一个宏定义,它的定义在module.h里面,它调了module_init这个宏

Module_init的定义如下:

#define  module_init(function, type)                                         \

static void  __attribute__((constructor)) do_qemu_init_ ## function(void) {  \

    register_module_init(function, type);                                   \

}

因为__attribute__((constructor))类似全局变量类的构造函数,会在qemumain()之前被执行。因此可以看成是virtio-serial-bus这个模块会在qemu进程运行的最初就被初始化注册(实际是将真正的模块初始化函数virtio_serial_register_types设置好,并将该模块加入到全局模块链表中)。同理,virtio-pcivirtio-console和其他设备模块都是这样的。

2.       虚拟串口模块的初始化

真正的模块构造是在vl.cmain()函数中调用到:module_call_init(MODULE_INIT_QOM);开始的,该函数遍历全局模块链表逐一做初始化。就virtio-serial-bus virtio-serial port的初始化流程如下:

KVM-QEMU vdagent实现机制分析(二)_第1张图片 

上图只是一些主要API的调用流程,大致介绍一下:

1)        Virtio-pci模块注册的时候注册了virtio-serial-bus这个虚拟串口设备,因此在模块初始化的过程中virtio-serial-busvirtio_serial_init初始化函数会被调用;

2)        virtio-serial-busvirtio_serial_init里面除了做了最common的初始化之外,主要做了两件事:

a)建立了与虚拟机vdagent通信的数据输入输出队列;

b)建立了与虚拟机vdagent通信的控制消息的输入输出队列;

3)        在控制接收队列中有vdagent发送过来的消息的时候,handle_control_message被触发;

4)        Virtio-console.c里面也在模块初始化中已经注册了virtio-serial-port这个设备模块,并注册了VirtIOSerialPortClass类的回调函数guest_open

5)        因此当virtio-serial-bushandle_control_message里判断是VIRTIO_CONSOLE_PORT_OPEN消息的时候,virtio-console.cguest_open消息被调用;

6)        Guest_open的下一步处理在Spice-qemu-char.cspice_chr_guest_open函数中,接下来qemu将这个事件传递给了spice server去处理了。

你可能感兴趣的:(云计算,linux系统,qemu-kvm)