virtio内容众多,代码分布于qemu,linux,dpdk等中,而且分为frontend和backend,可以运行于userspace也可以运行于kernelspace,极其难以理解,不看代码只看原理性文档往往流于表面,只有真正看懂了代码才能理解virtio。
以qemu和linux中的virtio-net举例分析代码,这儿只分析qemu部分virtio代码,在qemu中创建一个virtio-net设备,tap作为backend,有2个queue,那么qemu中tx和rx各2个,再加1个controll queue就得创建5个queue了
- -netdev tap,id=hostnet0,queues=2
- -device virtio-net-pci,host_mtu=1450,mq=on,vectors=5,netdev=hostnet0,id=net0,mac=fa:16:3e:d8:fe:81,bus=pci.0,addr=0x3
我们先看数据结构NetClientState,重点关注peer和incoming_queue
- struct NetClientState {
- NetClientInfo *info;
- int link_down;
- QTAILQ_ENTRY(NetClientState) next;
- NetClientState *peer;
- NetQueue *incoming_queue;
- char *model;
- char *name;
- char info_str[256];
- unsigned receive_disabled : 1;
- NetClientDestructor *destructor;
- unsigned int queue_index;
- unsigned rxfilter_notify_enabled:1;
- int vring_enable;
- int vnet_hdr_len;
- bool is_netdev;
- QTAILQ_HEAD(, NetFilterState) filters;
- };
先看qemu中的参数-netdev,创建了2个TAPState,每个TAPState包含一个NetClientState,函数qemu_net_client_setup参数peer为空,所以创建的ncs中peer为空。
- net_tap_fd_init
- └─qemu_new_net_client
- └─qemu_net_client_setup//这个函数参数peer为NULL
再看qemu的参数-device中有netdev=hostnet0,而-netdev中有id=hostnet0,根据name找到了刚才tap创建的2个NetClientState和queue个数为2
- typedef struct NICPeers {
- NetClientState *ncs[MAX_QUEUE_NUM];
- int32_t queues;
- }