0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

linux设备中virtio组织关系及设备初始化调用流程

454398 来源: Chinaunix 作者:lvyilong316 2020-09-25 15:47 次阅读

我们看linux kernel中virtio驱动相关代码,会发现有很多相关文件。首先有virtio.c这种文件,其次还有virtio_pci.c,virtio_scsi.c等这些文件,还有virtio_net.c,virtio_blk.c,virtio_balloon.c等这些。那么这些文件是什么关系呢?其次里面很多还有各自probe函数,到底是如何调用的,例如以网络的virtio_net到底是从哪里开始初始化的?要理清这些关系需要以linux设备驱动模型为背景展开讨论。这篇文章,我们以linux kernel 3.10代码为例,分析一下virtio的相关组织关系,以及设备初始化调用流程。

总线及驱动的注册

linux设备驱动模型的核心有三个概念:设备(device),驱动(driver),总线(bus)。而如果我们把virtio的相关关系梳理清楚后,以网络virtio_net为例映射到设备驱动模型,就得到了下图。我们这个小节后面就以下图为背景展开。

图1

linux将virtio实现分离成两部分:和物理总线标准相关的(如pci,scsi等),和物理总线标准无关的。

图中左侧部分即和物理总线相关的实现,这里以pci为了,当然virtio也支持其他总线类型,如scsi。virtio-pci是virtio对应pci的驱动实现,所以virtio-pci是一个pci总线上的一个驱动。它通过如下方式注册到pci总线上去。

lvirtio总线的注册

点击(此处)折叠或打开

/*virtio.c*/

static struct bus_type virtio_bus={

.name="virtio",

.match=virtio_dev_match,

.dev_attrs=virtio_dev_attrs,

.uevent=virtio_uevent,

.probe=virtio_dev_probe,

.remove=virtio_dev_remove,

};

staticintvirtio_init(void)

{

if(bus_register(&virtio_bus)!=0)

panic("virtio bus registration failed");

return 0;

}

core_initcall(virtio_init);

如代码所示,这个总线的名字叫”virtio”,通过bus_register就将virtio总线注册进系统,可以在sys文件系统中查看。

lvirtio-net驱动注册

最后我们看我们经常接触到设备驱动的初始化,我们以网络驱动virtio_net为例,其对应的驱动为virtio-net。其注册过程如下。

点击(此处)折叠或打开

/*virtio-net.c*/

static struct virtio_device_id id_table[]={

{VIRTIO_ID_NET,VIRTIO_DEV_ANY_ID},

{0},

};

static struct virtio_driver virtio_net_driver={

.feature_table=features,

.feature_table_size=ARRAY_SIZE(features),

.driver.name=KBUILD_MODNAME,

.driver.owner=THIS_MODULE,

.id_table=id_table,

.probe=virtnet_probe,

.remove=virtnet_remove,

.config_changed=virtnet_config_changed,

#ifdef CONFIG_PM

.freeze=virtnet_freeze,

.restore=virtnet_restore,

#endif

};

module_virtio_driver(virtio_net_driver);

#define module_virtio_driver(__virtio_driver)\

module_driver(__virtio_driver,register_virtio_driver,\

unregister_virtio_driver)

intregister_virtio_driver(struct virtio_driver*driver)

{

/*Catch this early.*/

BUG_ON(driver->feature_table_size&&!driver->feature_table);

driver->driver.bus=&virtio_bus;

return driver_register(&driver->driver);

}

最终通过register_virtio_driver函数将驱动的bus设置为之前注册的virtio总线,完成总线的注册。这样我们就能在sys文件系统对应virtio总线下的drivers目录看到这个驱动了。

所以我们再回头来看图1,可以看到virtio设备是横跨两类总线及驱动的。

virtio设备的初始化流程

梳理清楚virtio相关设备,总线及驱动关系后我们看下virtio设备的初始化过程,我们还是以网络virtio_net设备为例子。这个初始化过程如下图2中的黄色部分所示。

图2

首先是系统启动kernel初始化阶段,pci子系统调用pci_scan_device发现pci网卡设备,并初始化对应pci_dev结构,然后将去注册到pci总线上(dev->dev.bus=&pci_bus_type)。同时设置devicevendor_id0x1AF4virtiopci vendor_id,device_id为1

然后当我们加载virtio-pci驱动时,当调用module_pci_driver(virtio_pci_driver)将virtio-pci驱动注册在pci总线上时,在linux设备驱动模型中,这会导致对pci总线设备链表上未被驱动绑定的每个设备调用pci总线的match回调函数,即pci_bus_match函数。原型如下:

static int pci_bus_match(struct device *dev, struct device_driver *drv)

pci_bus_match函数将linux设备驱动模型核心出入的device结构转换为pci_dev结构,将device_driver结构转换为pci_driver结构,之后调用pci_match_device函数判断pci设备结构是否有匹配的pci设备ID结构。如果有则判断设备的pci ID和驱动设置的id_table中是否一样,如果一样说明设备和驱动匹配(这里设备的vendor_id和virtio-pci的virtio_pci_id_table匹配),将struct device的driver指针指向驱动,然后调用pci总线的probe函数,即pci_deivce_probe函数。这个函数再次将struct device强制转换成struct pci_dev,将设置在设备中的driver结构强制转换为struct pci_derver。它再次校验这个驱动能否支持这个设备,递增设备的引用计数,然后调用pci驱动probe函数(即virtio-pci的probe函数virtio_pci_probe),传入它应该绑定到的struct pci_dev结构体指针。这就进入到了图2中黄色部分的函数调用链了。

在开始梳理virtio_net初始化调用链前我们先看其对应的结构struct virtio_pci_device,将其展开得到图3。

图3

我们看到virtio_pci_device可以分为两部分,一部分是和pci总线相关的设备对应struct pci_dev,另一部分是和virtio总线相关的设备对应structvirtio_device。

virtio_pci_probe函数主要负责完成pci_dev部分的初始化,已经virtio_device部分初始化,然后调用register_virtio_device函数。

register_virtio_device函数将virtio_device的设备总线设置为virtio总线,然后调用device_register将virtio_device对应的设备添加到virtio总线上。这个添加总线的动作,会触发virtio总线的match函数即virtio_dev_match调用,同样该函数会比较设备dev的pci id和驱动id (virtio net的devid为1),如果匹配则virtio bus的probe函数virtio_dev_probe将被调用。其中又会调用对应驱动的probe函数,即virtnet_probe。而virtnet_probe将会完成virtio net设备structvirtio_device剩余部分的初始化。

到此,virtio net的初始化流程就已经梳理清楚了。virtio net设备创建完成后也会分别出现在pci总线和virtio总线的drvices目录下。

最后附上virtio的其他类型设备id:

点击(此处)折叠或打开

#define VIRTIO_ID_NET1/*virtio net*/

#define VIRTIO_ID_BLOCK2/*virtio block*/

#define VIRTIO_ID_CONSOLE3/*virtio console*/

#define VIRTIO_ID_RNG4/*virtio rng*/

#define VIRTIO_ID_BALLOON5/*virtio balloon*/

#define VIRTIO_ID_RPMSG7/*virtio remote processor messaging*/

#define VIRTIO_ID_SCSI8/*virtio scsi*/

#define VIRTIO_ID_9P9/*9p virtio console*/

#define VIRTIO_ID_RPROC_SERIAL 11/*virtio remoteproc seriallink*/

#define VIRTIO_ID_CAIF 12/*Virtio caif*/

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • Linux
    +关注

    关注

    87

    文章

    11304

    浏览量

    209474
  • 总线
    +关注

    关注

    10

    文章

    2881

    浏览量

    88080
  • 设备驱动
    +关注

    关注

    0

    文章

    68

    浏览量

    10888
收藏 人收藏

    评论

    相关推荐

    【明天会更好】关于应届毕业生档案、户口、party组织关系...

    关于应届毕业生档案、户口、party组织关系的问题:作为过来人,我觉得应届毕业生档案、户口、party组织关系等这些问题必须得自己搞清楚了。不然有你吃亏的一天呀!{:soso_e116:}
    发表于 05-14 16:52

    c语言编程多个文件的组织关系错误

    c语言编程多个文件的组织关系错误 因为keyboard.c,display.c需要reg52.h,为皮面重定义我建了reg.h中有:#include 已知:main.c: #include
    发表于 06-15 14:51

    如何初始化函数? 如何从代码配置器调用函数?

    :V3.30XC8:V1.37 OS:Mac OS 1011区域:如何初始化函数和调用函数?设备:PIC16F1847外围设备:EASART,ADCMOD:异步
    发表于 10-15 09:38

    处理器和PCH设备之间的通讯流程是怎样的?

    QAT相关的名词组织关系是什么?处理器和PCH设备之间的通讯流程是怎样的?服务实例与硬件是如何交互的?
    发表于 07-23 08:12

    StratoVirt 的 virtio-blk 设备是如何实现的?

    virtio-blk 设备支持一个队列:request_queue。该队列负责 block 设备初始化以及 IO 命令传输。StratoVirt 为该队列配置了对应的 event_
    发表于 06-29 10:37

    LINUX系统引导和初始化-LINUX内核解读

    Linux 的系统引导和初始化 ----------Linux2.4.22内核解读之一 一、 系统引导和初始化概述 相关代码(引导扇区的程序及其辅助程序,以 x86体系为例): \
    发表于 11-03 22:31 53次下载

    objc源码NSObject如何进行初始化

    + alloc 和 - init 这一对我们在 iOS 开发每天都要用到的初始化方法一直困扰着我, 于是笔者仔细研究了一下 objc 源码 NSObject 如何进行初始化。 在具
    发表于 09-26 09:58 0次下载

    Linux内存初始化

    之前有几篇博客详细介绍了Xen的内存初始化,确实感觉这部分内容蛮复杂的。这两天在看Linux内核启动内存的初始化,也是看的云里雾里的,想尝试下边看边写,在写博客的过程
    发表于 10-12 11:16 0次下载

    Linux内核初始化过程调用顺序

    所有的__init函数在区段.initcall.init还保存了一份函数指针,在初始化时内核会通过这些函数指针调用这些__init函数指针,并在整个初始化完成后,释放整个init区段
    发表于 05-12 08:40 1614次阅读

    uboot中进行LCD初始化流程免费下载

    本文档的主要内容详细介绍的是uboot中进行LCD初始化流程免费下载。
    发表于 08-12 08:00 11次下载
    uboot中进行LCD<b class='flag-5'>初始化</b>的<b class='flag-5'>流程</b>免费下载

    UCOS2系统内核讲述(二)_ 初始化调用函数

    UCOS2系统内核讲述(二)_初始化调用函数
    的头像 发表于 03-25 09:57 1772次阅读
    UCOS2系统内核讲述(二)_ <b class='flag-5'>初始化</b><b class='flag-5'>调用</b>函数

    EE-359:ADSP-CM40x引导时间优化和设备初始化

    EE-359:ADSP-CM40x引导时间优化和设备初始化
    发表于 05-21 09:39 10次下载
    EE-359:ADSP-CM40x引导时间优化和<b class='flag-5'>设备</b><b class='flag-5'>初始化</b>

    eMMC初始化命令发送流程

    eMMC初始化命令发送流程通过stm32单片机驱动eMMC芯片,在完成项目的过程,命令发送和响应很令人头疼,所以整理出一篇文档供各位开发者参考。初始化命令发送
    发表于 12-05 19:21 18次下载
    eMMC<b class='flag-5'>初始化</b>命令发送<b class='flag-5'>流程</b>

    Linux终端初始化和tty驱动框架

    initcall机制 注意上述流程,我们来理解一下 initcall 机制: 普通我们写一个程序,想要它被调用,需要在主流程调用这个函数,
    的头像 发表于 09-28 16:33 712次阅读
    <b class='flag-5'>Linux</b>终端<b class='flag-5'>初始化</b>和tty驱动框架

    OP-TEE的内核初始化函数调用

    generic_boot_init_primary函数内容 generic_boot_init_primary函数是OP-TEE建立系统运行环境的入口函数,该函数会进行建立线程运行空间、初始化
    的头像 发表于 11-02 18:18 761次阅读
    OP-TEE的内核<b class='flag-5'>初始化</b>函数<b class='flag-5'>调用</b>