当前位置:首页 > 嵌入式 > 嵌入式软件
[导读]Linux设备驱动的难点在于复杂的,庞大的结构。理清楚结构和一个结构与另外结构的关系,以及linux设备驱动的层次性和层次封装抽象性。对于linux设备驱动的结构有点像C++中的类,而层次与抽象有点像继承的关系。

Linux设备驱动的难点在于复杂的,庞大的结构。理清楚结构和一个结构与另外结构的关系,以及linux设备驱动的层次性和层次封装抽象性。对于linux设备驱动的结构有点像C++中的类,而层次与抽象有点像继承的关系。

一、总线、设备、驱动的主要三个结构关系

structbus_type  

---------------------------------

struct bus_type中为devices和drivers准备了两个链表:

struct klist klist_devices

struct klist klist_drivers

struct device

---------------------------------

struct device有两个成员

struct bus_type *bus 记录的是这个设备连在哪条总线上

struct device_driver *driver 记录的是这个设备用的是哪个驱动

struct device_driver

---------------------------------

struct device_driver同样有两个成员

struct bus_type *bus 代表的是这个驱动属于哪条总线

struct klist klist_devices 记录的是这个驱动支持的那些设备,没错,是devices(复数),因为一个驱动程序可以支持一个或多个设备,反过来一个设备则只会绑定给一个驱动程序.

二、总线,设备,驱动的关联

总线将设备和驱动绑定。在系统每注册一个设备的时候,会寻找与之匹配的驱动;相反的,在系统每 注册一个驱动的时候,会寻找与之匹配的设备,而匹配由总线完成。一个现实的Linux设备和驱动通常都需要挂接在一种总线上。设备与驱动的关联通过总线的match()方法进行匹配,驱动挂载总线时与所有设备进行匹配,设备挂载总线时与所有的驱动进行匹配,所以驱动和设备的挂载无先后之分。匹配成功后会通过调用驱动的probo()方法来初始化设备。

三、总线,设备,驱动的注册

设备与驱动需要挂载在总线上,需要指明驱动与设备是属于哪条总线的,所以设备与驱动需要注册。而总线在linux系统中也是属于设备,所以总线也要注册,同时要先有总线而后才能注册设备和驱动,所以总线要先注册。

总线在linux系统中有俩种,一是实际存在的总线 pci usb 等等,还有一类是虚拟存在的总线 platform ,platform总线主要是用于集成在SoC系统的设备,使得每一个设备都属于一条总线,相应的设备称为platform_device,而驱动成为 platform_driver。linux驱动中platform总线用的非常多,以platform总线说明总线,设备,驱动的注册顺序,注意这里是以先调加设备为例。

1. platform_bus_type -- 总线 先被kenrel 注册。

2. 系统初始化过程中调用platform_add_devices 或者platform_device_register ,将平台设备(platform devices) 注册到平台总线中( platform_bus_type )

3. 平台驱动(platform driver) 与平台设备(platform device) 的关联是在platform_driver_register 或者driver_register 中实现,一般这个函数在驱动的初始化过程调用。

通过这三步,就将平台总线,设备,驱动关联起来。

1. platform bus 先被kenrel 注册。

------------------------------------------------------

do_basic_setup() --> - driver_init() --> - platform_bus_init() -->bus_register()

2. 系统初始化过程中调用platform_add_devices 或者platform_device_register ,将平台设备(platform devices) 注册到平台总线中( platform_bus_type )

------------------------------------------------------

系统启动阶段,总线的驱动链表还是空的,所以启动阶段的platform_add_devices() 只负责将设备添加到总线的设备链表上。

linux 2.6.26/drivers/base/platform.c

int platform_add_devices(struct platform_device **devs, int num)

{

...

ret = platform_device_register (devs[i]);

...

}

int platform_device_register(struct platform_device *pdev)

{

device_initialize(&pdev >dev);

return platform_device_add (pdev);

}

int platform_device_add (struct platform_device *pdev)

{

...

pdev >dev.bus = &platform_bus_type;

...

ret = device_add (&pdev >dev);

...

}

device_add() > bus_attach_device()

void bus_attach_device(struct device *dev)

{

struct bus_type *bus = dev >bus;

int ret = 0;

if (bus) {

if (bus >p >drivers_autoprobe)

ret = device_attach (dev);

WARN_ON(ret < 0);

if (ret >= 0)

klist_add_tail (&dev >knode_bus, &bus >p >klist_devices);

}

}

device_attach() 的返回值:

1 设备和驱动匹配成功

0 设备已经注册,但是总线上没有与之相匹配的驱动( 系统启动阶段,由于总线上还没有驱动,所以设备在此匹配不到与之对应的驱动,只是将其添加到总线的设备链表)

-ENODEV 设备没有注册(registered) -- 设备在哪里注册?

如果设备和驱动匹配成功; 或者设备已经注册,但是总线上没有与之相匹配的驱动 ,bus_attach_device() 将调用klist_add_tail() 将设备添加到总线的设备链表尾部。

四、附录linux内核中的platform的几个结构源代码

[cpp] view plain copy

1. // 所在目录:kernel/include/linux/platform_device.h

2. struct platform_device

3. {

4. const char * name;/* 设备名 */

5. u32 id;

6. struct device dev;

7. u32 num_resources;/* 设备所使用各类资源数量 */

8. struct resource * resource;/* 资源 */

9. };

10. // 所在目录:include/linux/ioport.h

11. struct resource

12. {

13. resource_size_t start; /* 资源的开始值 */

14. resource_size_t end; /* 资源的结束值 */

15. const char *name; /* 资源的名字 */

16. unsigned long flags; /* 资源的类型值,如可以是:mem,io,irq,dma等等 */

17. struct resource *parent, *sibling, *child;

18. };

19. // 所在目录:kernel/include/linux/ioport.h

20. struct platform_driver

21. {

22. int (*probe)(struct platform_device *);

23. int (*remove)(struct platform_device *);

24. void (*shutdown)(struct platform_device *);

25. int (*suspend)(struct platform_device *, pm_message_t state);

26. int (*suspend_late)(struct platform_device *, pm_message_t state);

27. int (*resume_early)(struct platform_device *);

28. int (*resume)(struct platform_device *);

29. struct pm_ext_ops *pm;

30. struct device_driver driver;

31. };

32. // 所在目录:include/linux/device.h

33. struct device_driver

34. {

35. char * name;

36. struct bus_type * bus;

37. rwlock_t lock;

38. atomic_t refcount;

39. list_t bus_list;

40. list_t devices;

41. struct driver_dir_entry dir;

42. int (*probe) (struct device * dev);

43. int (*remove) (struct device * dev);

44. int (*suspend) (struct device * dev, u32 state, u32 level);

45. int (*resume) (struct device * dev, u32 level);

46. void (*release) (struct device_driver * drv);

47. };

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

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 隧道灯 驱动电源
关闭