VIRTIO-BLK整个过程数据流如下所示:
IO请求发送过程
虚拟机中通过FIO等下发IO请求,IO请求通过VFS/filesystem,然后到BLOCK层,传递给virtio-blk驱动,virtio-blk驱动通过virtio_queue_rq()下发IO请求,并通过vp_notify()通知QEMU中VIRTIO设备数据准备完毕。
虚拟机中vp_notify()访问寄存器,它会陷入到hypervisor中,并分发到QEMU中,并最终由virtio_pci_notify_write()模拟操作,通过函数virtio_blk_handle_output()将数据下发,这里如果支持IO_URING时,可以通过luring_co_submit()提交IO请求;如果支持AIO时,可以通过laio_co_submit()提供IO请求。
QEMU提交IO请求如同普通物理机上用户态提交IO请求一样,依次经过文件系统层,BLOCK层,最终下发给硬件处理。
IO请求完成过程
硬件完成IO处理后,IO完成信息依次通过BLOCK层,文件系统层,最终返回到QEMU中由函数virtio_blk_rw_complete()处理。该函数通过virtio_notify()往虚拟机注入中断,虚拟机产生中断后由函数vring_interrupt()进行中断处理,并调用virtioblk_done()作VIRTIOBLK的IO完成,并将IO完成信息依次通过BLOCK层,文件系统层,并最终返回给虚拟机中应用。
在上述过程中IO请求下发是通过vp_notify()访问虚拟机中寄存器,最终由QEMU模拟MMIO操作通知IO下发;IO完成是通过往虚拟机中注入中断通知虚拟机IO完成处理。