国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 服務器 > Linux服務器 > 正文

詳解Linux驅動中,probe函數何時被調用

2024-09-05 23:03:12
字體:
來源:轉載
供稿:網友

最近看到linux的設備驅動模型,關于Kobject、Kset等還不是很清淅。看到了struct device_driver這個結構時,想到一個問題:它的初始化函數到底在哪里調用呢?以前搞PCI驅動時用pci驅動注冊函數就可以調用它,搞s3c2410驅動時只要在mach-smdk2410.c中的struct platform_device *smdk2410_devices {}中加入設備也會調用。但從來就沒有想過具體的驅動注冊并調用probe的過程。

于是打開SourceInsight追蹤了一下:

從driver_register看起:

int driver_register(struct device_driver * drv){    klist_init(&drv->klist_devices, klist_devices_get, klist_devices_put);    init_completion(&drv->unloaded);    return bus_add_driver(drv);}

klist_init與init_completion沒去管它,可能是2.6的這個設備模型要做的一些工作。直覺告訴我要去bus_add_driver。

bus_add_driver中:

都是些Kobject 與 klist 、attr等。還是與設備模型有關的。但是其中有一句:

driver_attach(drv);

單聽名字就很像:

void driver_attach(struct device_driver * drv){    bus_for_each_dev(drv->bus, NULL, drv, __driver_attach);}

這個熟悉,遍歷總線上的設備并設用__driver_attach。

在__driver_attach中又主要是這樣:

driver_probe_device(drv, dev);

跑到driver_probe_device中去看看:

有一段很重要:

if (drv->bus->match && !drv->bus->match(dev, drv))        goto Done;

明顯,是調用的驅動的總線上的match函數。如果返回1,則可以繼續,否則就Done了。

繼承執行的話:

    if (drv->probe) {        ret = drv->probe(dev);        if (ret) {            dev->driver = NULL;            goto ProbeFailed;        }

只要probe存在則調用之。至此就完成了probe的調用。

這個過程鏈的關鍵還是在drv->bus->match ,因為其余的地方出錯的話就是注冊失敗,而只要注冊不失敗且match返回1,那么就鐵定會調用驅程的probe了。你可以注冊一個總線類型和總線,并在match中總是返回 1, 會發現,只要struct device_driver中的bus類型正確時,probe函數總是被調用.

PCI設備有自己的總線模型,估計在它的match中就有一個判斷的條件。

static int pci_bus_match(struct device *dev, struct device_driver *drv){    struct pci_dev *pci_dev = to_pci_dev(dev);    struct pci_driver *pci_drv = to_pci_driver(drv);    const struct pci_device_id *found_id;    found_id = pci_match_device(pci_drv, pci_dev);    if (found_id)        return 1;    return 0;}

再往下跟蹤就知道主要是根據我們熟悉的id_table來的。

-------------------------------另解-----------------------------------------------------------------------------------------------

從driver_register看起,此處我的這里是:

int driver_register(struct device_driver * drv){if ((drv->bus->probe && drv->probe) ||   (drv->bus->remove && drv->remove) ||   (drv->bus->shutdown && drv->shutdown)) {  printk(KERN_WARNING "Driver '%s' needs updating - please use bus_type methods/n", drv->name);}klist_init(&drv->klist_devices, NULL, NULL);return bus_add_driver(drv);} 

klist_init不相關,不用管他,具體再去看bus_add_driver:

int bus_add_driver(struct device_driver *drv){//1.先kobject_set_name(&drv->kobj, "%s", drv->name);//2.再kobject_register(&drv->kobj)//3.然后調用了:driver_attach(drv)}
int driver_attach(struct device_driver * drv){return bus_for_each_dev(drv->bus, NULL, drv, __driver_attach);}

真正起作用的是__driver_attach:

static int __driver_attach(struct device * dev, void * data){...if (!dev->driver)  driver_probe_device(drv, dev);...}int driver_probe_device(struct device_driver * drv, struct device * dev){...//1.先是判斷bus是否match:if (drv->bus->match && !drv->bus->match(dev, drv))  goto done;//2.再具體執行probe:ret = really_probe(dev, drv);...}

really_probe才是我們要找的函數: 

static int really_probe(struct device *dev, struct device_driver *drv){...//1.先是調用的驅動所屬總線的probe函數:if (dev->bus->probe) {  ret = dev->bus->probe(dev);  if (ret)  goto probe_failed;} else if (drv->probe) {//2.再調用你的驅動中的probe函數:  ret = drv->probe(dev);  if (ret)  goto probe_failed;}...}

其中,drv->probe(dev),才是真正調用你的驅動實現的具體的probe函數。

也就是對應此文標題所問的,probe函數此時被調用。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持VEVB武林網。


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 梁河县| 温泉县| 胶南市| 葫芦岛市| 中卫市| 乡城县| 深泽县| 定陶县| 德格县| 来宾市| 温州市| 彰化县| 阿鲁科尔沁旗| 锡林郭勒盟| 泸水县| 松江区| 定襄县| 恩施市| 县级市| 虎林市| 武冈市| 新巴尔虎右旗| 织金县| 博罗县| 邮箱| 泽州县| 海宁市| 绥化市| 通山县| 隆回县| 仁化县| 谢通门县| 镶黄旗| 新绛县| 太原市| 阳山县| 兰坪| 洛宁县| 绿春县| 上思县| 五家渠市|