Linux设备驱动之USBhub驱动

整理文档很辛苦,赏杯茶钱您下走!

免费阅读已结束,点击下载阅读编辑剩下 ...

阅读已结束,您可以下载文档离线阅读编辑

资源描述

Linux100中文网驱动------------------------------------------本文系本站原创,欢迎转载!转载请注明出处:一:前言继UHCI的驱动之后,我们对USBControl的运作有了一定的了解.在接下来的分析中,我们对USB设备的驱动做一个全面的分析,我们先从HUB的驱动说起.关于HUB,usb2.0spec上有详细的定义,基于这部份的代码位于linux-2.6.25/drivers/usb/core下,也就是说,这部份代码是位于core下,和具体设备是无关的,因为各厂商的hub都是按照spec的要求来设计的.二:UHCI驱动中的roothub记得在分析UHCI驱动的时候,曾详细分析过roothub的初始化操作.为了分析方便,将代码片段列出如下:usb_add_hcd()àusb_alloc_dev():structusb_device*usb_alloc_dev(structusb_device*parent,structusb_bus*bus,unsignedport1){…………//usb_device,内嵌有structdevice结构,对这个结构进行初始化device_initialize(&dev-dev);dev-dev.bus=&usb_bus_type;dev-dev.type=&usb_device_type;Linux100中文网…………}一看到前面对dev的赋值,根据我们对设备模型的理解,一旦这个device进行注册,就会发生driver和device的匹配过程了.不过,现在还不是分析这个过程的时候,我们先来看一下,USB子系统中的两种驱动.三:USB子系统中的两种驱动linux-2.6.25/drivers/usb/core/driver.c中,我们可以找到两种registerdriver的方式,分别为usb_register_driver()和usb_register_device_driver().分别来分析一下这两个接口.usb_register_device_driver()接口的代码如下:intusb_register_device_driver(structusb_device_driver*new_udriver,structmodule*owner){intretval=0;if(usb_disabled())return-ENODEV;new_udriver-drvwrap.for_devices=1;new_udriver-drvwrap.driver.name=(char*)new_udriver-name;new_udriver-drvwrap.driver.bus=&usb_bus_type;Linux100中文网=usb_probe_device;new_udriver-drvwrap.driver.remove=usb_unbind_device;new_udriver-drvwrap.driver.owner=owner;retval=driver_register(&new_udriver-drvwrap.driver);if(!retval){pr_info(“%s:registerednewdevicedriver%s\n”,usbcore_name,new_udriver-name);usbfs_update_special();}else{printk(KERN_ERR“%s:error%dregisteringdevice““driver%s\n”,usbcore_name,retval,new_udriver-name);}returnretval;}首先,通过usb_disabled()来判断一下usb是否被禁用,如果被禁用,当然就不必执行下面的流程了,直接退出即可.从上面的代码,很明显可以看到,structusb_device_driver对structdevice_driver进行了一次封装,我们注意一下这里的赋值操作:new_udriver-drvwrap.for_devices=1.等等.这些在后面都是用派上用场的.Linux100中文网()的代码如下:intusb_register_driver(structusb_driver*new_driver,structmodule*owner,constchar*mod_name){intretval=0;if(usb_disabled())return-ENODEV;new_driver-drvwrap.for_devices=0;new_driver-drvwrap.driver.name=(char*)new_driver-name;new_driver-drvwrap.driver.bus=&usb_bus_type;new_driver-drvwrap.driver.probe=usb_probe_interface;new_driver-drvwrap.driver.remove=usb_unbind_interface;new_driver-drvwrap.driver.owner=owner;new_driver-drvwrap.driver.mod_name=mod_name;spin_lock_init(&new_driver-dynids.lock);INIT_LIST_HEAD(&new_driver-dynids.list);retval=driver_register(&new_driver-drvwrap.driver);if(!retval){Linux100中文网(“%s:registerednewinterfacedriver%s\n”,usbcore_name,new_driver-name);usbfs_update_special();usb_create_newid_file(new_driver);}else{printk(KERN_ERR“%s:error%dregisteringinterface““driver%s\n”,usbcore_name,retval,new_driver-name);}returnretval;}很明显,在这里接口里,将new_driver-drvwrap.for_devices设为了0.而且两个接口的porbe()函数也不一样.其实,对于usb_register_driver()可以看作是usb设备中的接口驱动,而usb_register_device_driver()是一个单纯的USB设备驱动.四:hub的驱动分析4.1:usb_bus_type-match()的匹配过程usb_bus_type-match()用来判断驱动和设备是否匹配,它的代码如下:staticintusb_device_match(structdevice*dev,structdevice_driver*drv){/*整理by*/Linux100中文网的情况if(is_usb_device(dev)){/*interfacedriversnevermatchdevices*/if(!is_usb_device_driver(drv))return0;/*TODO:Addrealmatchingcode*/return1;}//interface的情况else{structusb_interface*intf;structusb_driver*usb_drv;conststructusb_device_id*id;/*整理by*/if(is_usb_device_driver(drv))return0;Linux100中文网=to_usb_interface(dev);usb_drv=to_usb_driver(drv);id=usb_match_id(intf,usb_drv-id_table);if(id)return1;id=usb_match_dynamic_id(intf,usb_drv);if(id)return1;}return0;}这里的match会区分上面所说的两种驱动,即设备的驱动和接口的驱动.is_usb_device()的代码如下:staticinlineintis_usb_device(conststructdevice*dev){returndev-type==&usb_device_type;}很明显,对于roothub来说,这个判断是肯定会满足的.staticinlineintis_usb_device_driver(structdevice_driver*drv)Linux100中文网{returncontainer_of(drv,structusbdrv_wrap,driver)-for_devices;}回忆一下,我们在分析usb_register_device_driver()的时候,不是将new_udriver-drvwrap.for_devices置为了1么?所以对于usb_register_device_driver()注册的驱动来说,这里也是会满足的.因此,对应roothub的情况,从第一个if就会匹配到usb_register_device_driver()注册的驱动.对于接口的驱动,我们等遇到的时候再来进行分析.4.2:roothub的驱动入口既然我们知道,roothub会匹配到usb_bus_type-match()的驱动,那这个驱动到底是什么呢?我们从usb子系统的初始化开始说起.在linux-2.6.25/drivers/usb/core/usb.c中.有这样的一段代码:subsys_initcall(usb_init);对于subsys_initcall()我们已经不陌生了,在很多地方都会遇到它.在系统初始化的时候,会调用到它对应的函数.在这里,即为usb_init().在usb_init()中,有这样的代码片段:staticint__initusb_init(void){…………retval=usb_register_device_driver(&usb_generic_driver,THIS_MODULE);if(!retval)Linux100中文网……}在这里终于看到usb_register_device_driver()了.usb_generic_driver会匹配到所有usb设备.

1 / 82
下载文档,编辑使用

©2015-2020 m.777doc.com 三七文档.

备案号:鲁ICP备2024069028号-1 客服联系 QQ:2149211541

×
保存成功