众所周知,专业的程序员都知道,程序入口通常为main()函数,而驱动程序真正的入口是XXX_probe()函数;
我们看代码第一步,就是从入口函数开始的,天才第一步,雀氏纸尿裤。从总——分——总的方式学会linux驱动
Step1:驱动模块的加载
Linux模块的加载手动加载:1、通过执行insmod命令
自动加载:2、通过模块编译进内核,在内核启动时模块自动加载
自动加载是调用module_init(hello_init) -->从而程序进入hello_init()函数当中;
下面以led驱动为例分析:
static int __init aw2013_led_init(void)
{
return i2c_add_driver(&aw2013_led_driver);
}
module_init(aw2013_led_init);
由于这里aw2013-led是led 的芯片挂载i2c总线上面(可以通过查看硬件原理图得知)。
根据platform的总线bus、设备device、驱动driver的设备驱动模型;在这里总线是:i2c ,设备是:aw2013-led芯片,驱动是:aw2013-led.C,
Step2:将dirver加载到bus上
#define i2c_add_driver(driver) \
i2c_register_driver(THIS_MODULE,driver)
在函数i2c_register_driver中,将传递进来的参数driver添加到i2c驱动列表list当中,主要调用driver_register()函数;
首先判断在目前总线上该driver是否已经被注册了,如果没有注册则调用bus_add_driver(drv)函数。
下面来看bus_add_driver函数主要是driver_attach()
void driver_attach(struct device_driver * drv)
{
bus_for_each_dev(drv->bus, NULL, drv, __driver_attach);
}
遍历总线上的每个设备,找到与该驱动相匹配的。在__driver_attach()函数中,首先driver_match_device(drv,dev)调用platform_match函数判断device与driver是否匹配。
static inline int driver_match_device(struct device_driver *drv,struct device *dev)
{
return drv->bus->match ? drv->bus->match(dev,drv) : 1;
}
然后一旦匹配函数match成功;就走进driver_probe_device()函数中,调用驱动的probe函数
继承执行的话:
if (drv->probe) {
ret = drv->probe(dev);
if (ret) {
dev->driver = NULL;
goto ProbeFailed;
}
终点:在这里我们可以看到drv->probe(dev)函数被调用的位置。致此敬礼
=================具体调用的bus_type类型的match函数实现是在哪?还需找到