Linux内核驱动——USB主机控制器驱动分析

一、USB总线

        系统启动 -》初始化usb子系统 -》向内核注册USB总线 -》向USB总线中注册3个usb驱动(分别是USB接口驱动、HUB驱动、USB设备驱动)。

        主机驱动注册为platform平台驱动 -》遍历平台总线的设备链表,执行platform的.match函数(platform_match)进行匹配,成功后执行.probe函数(ohci_hcd_s3c2410_drv_probe) -》在此函数中再遍历usb总线的驱动链表,执行.match函数(usb_device_match),成功后执行.probe函数(预先注册的3个usb驱动) -》检测其他usb设备插拔。。。

//----------------------------------------USB BUS------------------------------------

【USB总线注册】

// usb_bus_type提供了驱动和设备匹配的匹配函数

retval = bus_register(&usb_bus_type);
struct bus_type usb_bus_type = {
        .name =     "usb",
        .match =    usb_device_match, // 内部会将 设备驱动 和 接口驱动 分开匹配(提高效率)
        ... ...
};

//使用bus_register接口注册USB总线,会创建出两条链表用来分别存放向USB总线注册的设备和驱动。

klist_init(&bus->klist_devices, klist_devices_get, klist_devices_put);
klist_init(&bus->klist_drivers, NULL, NULL);

1、在usb总线注册USB接口驱动,该驱动被放在usb总线的驱动链表中。(一个设备可以注册多个接口驱动

// 注册usbfs接口驱动,提供了在用户空间直接访问USB硬件设备的接口,完成USB设备在用户空间的映射

retval = usb_register(&usbfs_driver);
struct usb_driver usbfs_driver = {
    .name =     "usbfs",
    .probe =    driver_probe, // .match成功后,执行
    .disconnect =   driver_disconnect,
    ... ...
};

2、在usb总线注册一个hub接口驱动,该驱动被放在usb总线的驱动链表中。

retval = usb_register(&hub_driver);
khubd_task = kthread_run(hub_thread, NULL, "khubd");// 创建一个死循环线程hub_thread,用来处理USB设备的断开、连接等事件

static struct usb_driver hub_driver = {
    .name =     "hub",
    .probe =    hub_probe, // 通过usb_fill_int_urb创建urb请求块同时注册了一个中断处理函数hub_irq,用来唤醒hub_thread线程来处理USB设备事件(热插拔)
    .disconnect =   hub_disconnect,
    ... ...

};

3、在usb总线注册通用USB设备驱动,该驱动被放在usb总线的驱动链表中。

retval = usb_register_device_driver(&usb_generic_driver, THIS_MODULE);
struct usb_device_driver usb_generic_driver = {
    .name = "usb",
    .probe = generic_probe, // 通用USB设备匹配成功后执行
    .disconnect = generic_disconnect,
    ... ...
};

二、USB主机驱动

【主机驱动器HCD】——platform模型

USB HCD注册在 平台总线 上。用来处理主机控制器的初始化以及数据的传输,并监测外部设备插入、拔出,完成设备枚举。

1、设备(或者采用设备树描述

struct platform_device s3c_device_usb = {
    .name       = "s3c2410-ohci",
    .resource     = s3c_usb_resource
    ... ...
};

platform_add_devices();    // 注册设备信息

2、主机驱动注册

retval = platform_driver_register(&ohci_hcd_s3c2410_driver); //内部会调用platform_match 进行驱动-设备匹配
struct platform_driver ohci_hcd_s3c2410_driver = {
    .probe      = ohci_hcd_s3c2410_drv_probe,
    .remove     = ohci_hcd_s3c2410_drv_remove,
    .shutdown   = usb_hcd_platform_shutdown,
    .driver     = {
        .owner  = THIS_MODULE,
        .name   = "s3c2410-ohci",
    },
    ... ...
};

3、 平台总线:此处的platform总线(虚拟总线)完成 USB主机控制器 的驱动和设备匹配.

       USB BUS的USB总线是完成 设备驱动与设备信息 的匹配.

struct bus_type platform_bus_type = {
    .name       = "platform",
    .match      = platform_match,
    ... ...
};

        调用platform_driver_register函数注册主机控制器驱动后,遍历平台总线的设备链表,调用platform_match进行设备-驱动匹配,匹配上后会调用驱动的ohci_hcd_s3c2410_drv_probe函数。

       ohci_hcd_s3c2410_drv_probe流程解析:遍历usb总线的驱动链表,调用usb_device_match进行匹配(第一节的USB总线),匹配成功后执行设备驱动的.probe函数。如执行USB通用设备驱动函数generic_probe。

 

//-------------------------------------------------------------------------------------------------------

驱动开发中,所有注册的USB设备,都是向USB总线注册,由USB总线实现驱动-设备的匹配。

入门级的设备驱动程序可以看上篇文章的分析—— 内核鼠标驱动源码。

 

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值