当前位置:首页 > 嵌入式 > 嵌入式软件
[导读] 当我们查看Linux kernel.C的Virtio代码时,我们会发现许多相关的文件。那么,verdio,我们只有Virtio\UC',Virtio\ubk。C、 美德\ubalon。C、 等等。与这些文件相关联的是什么?其次,有许多相应的程序功能。这个电话怎么说是真的,比如说通过互联网。从哪里开始?为了澄清这些关系,必须考虑模式Linux.DiesIn本文分析了linux-kernel-3.10中病毒组织与设备初始化的关系。

当我们查看Linux kernel.C的Virtio代码时,我们会发现许多相关的文件。那么,verdio,我们只有Virtio\UC',Virtio\ubk。C、 美德\ubalon。C、 等等。与这些文件相关联的是什么?其次,有许多相应的程序功能。这个电话怎么说是真的,比如说通过互联网。从哪里开始?为了澄清这些关系,必须考虑模式Linux.DiesIn本文分析了linux-kernel-3.10中病毒组织与设备初始化的关系。

总线及驱动的注册

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

图1

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

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

l  virtio总线的注册

点击(此处)折叠或打开

/*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,

};

static int virtio_init(void)

{

if (bus_register(&virtio_bus) != 0)

panic("virtio bus registration failed");

return 0;

}

core_initcall(virtio_init);

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

l  virtio-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)

int register_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)。同时设置device的vendor_id为0x1AF4(virtio的pci 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总线相关的设备对应struct virtio_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设备struct virtio_device剩余部分的初始化。

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

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

点击(此处)折叠或打开

#define VIRTIO_ID_NET        1 /* virtio net */

#define VIRTIO_ID_BLOCK        2 /* virtio block */

#define VIRTIO_ID_CONSOLE    3 /* virtio console */

#define VIRTIO_ID_RNG        4 /* virtio rng */

#define VIRTIO_ID_BALLOON    5 /* virtio balloon */

#define VIRTIO_ID_RPMSG        7 /* virtio remote processor messaging */

#define VIRTIO_ID_SCSI        8 /* virtio scsi */

#define VIRTIO_ID_9P        9 /* 9p virtio console */

#define VIRTIO_ID_RPROC_SERIAL 11 /* virtio remoteproc serial link */

355定义virtio u id u caif-12/*virtio caif*/

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: 驱动电源

在工业自动化蓬勃发展的当下,工业电机作为核心动力设备,其驱动电源的性能直接关系到整个系统的稳定性和可靠性。其中,反电动势抑制与过流保护是驱动电源设计中至关重要的两个环节,集成化方案的设计成为提升电机驱动性能的关键。

关键字: 工业电机 驱动电源

LED 驱动电源作为 LED 照明系统的 “心脏”,其稳定性直接决定了整个照明设备的使用寿命。然而,在实际应用中,LED 驱动电源易损坏的问题却十分常见,不仅增加了维护成本,还影响了用户体验。要解决这一问题,需从设计、生...

关键字: 驱动电源 照明系统 散热

根据LED驱动电源的公式,电感内电流波动大小和电感值成反比,输出纹波和输出电容值成反比。所以加大电感值和输出电容值可以减小纹波。

关键字: LED 设计 驱动电源

电动汽车(EV)作为新能源汽车的重要代表,正逐渐成为全球汽车产业的重要发展方向。电动汽车的核心技术之一是电机驱动控制系统,而绝缘栅双极型晶体管(IGBT)作为电机驱动系统中的关键元件,其性能直接影响到电动汽车的动力性能和...

关键字: 电动汽车 新能源 驱动电源

在现代城市建设中,街道及停车场照明作为基础设施的重要组成部分,其质量和效率直接关系到城市的公共安全、居民生活质量和能源利用效率。随着科技的进步,高亮度白光发光二极管(LED)因其独特的优势逐渐取代传统光源,成为大功率区域...

关键字: 发光二极管 驱动电源 LED

LED通用照明设计工程师会遇到许多挑战,如功率密度、功率因数校正(PFC)、空间受限和可靠性等。

关键字: LED 驱动电源 功率因数校正

在LED照明技术日益普及的今天,LED驱动电源的电磁干扰(EMI)问题成为了一个不可忽视的挑战。电磁干扰不仅会影响LED灯具的正常工作,还可能对周围电子设备造成不利影响,甚至引发系统故障。因此,采取有效的硬件措施来解决L...

关键字: LED照明技术 电磁干扰 驱动电源

开关电源具有效率高的特性,而且开关电源的变压器体积比串联稳压型电源的要小得多,电源电路比较整洁,整机重量也有所下降,所以,现在的LED驱动电源

关键字: LED 驱动电源 开关电源

LED驱动电源是把电源供应转换为特定的电压电流以驱动LED发光的电压转换器,通常情况下:LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: LED 隧道灯 驱动电源
关闭