先看一张熟悉的图:
可见usb设备都是接在hub上的,设备的检测自然跟hub相关了,这里以root hub为例。
看usb hub驱动代码,跟usb设备检测相关的有如下几处(按在hub_probe中的调用顺序):
1.INIT_WORK(&hub->events, hub_event);
2.usb_fill_int_urb(hub->urb, hdev, pipe, *hub->buffer, maxp, hub_irq,
hub, endpoint->bInterval);
3.status = usb_submit_urb(hub->urb, GFP_NOIO);
第3处即向root_hub提交一个中断传输的urb,我们来看这个urb的完成函数hub_irq:
这个函数比较简单,最后又调用usb_submit_urb提交中断urb,以实现对中断端点的轮询。
这个中断urb最终会调到root_hub的定时器函数rh_timer_func,具体调用过程请分析usb_create_hcd/usb_add_hcd,usb_submit_urb,这里不赘述
这个定时器函数里,调用hc_driver的hub_status_data读取root hub的端口变化状态
kick_hub_wq实现如下:
向工作队列hub_wq提交了一个工作任务,工作函数为前面第1处的hub_event,
hub_event最终会调到hub_port_connect,处理端口设备的connect,disconnect
如果是connect,则会调用usb_alloc_dev/usb_new_device创建新的usb_device,踏上match与probe usb_device_driver和usb_driver的漫长之旅...
可见usb设备都是接在hub上的,设备的检测自然跟hub相关了,这里以root hub为例。
看usb hub驱动代码,跟usb设备检测相关的有如下几处(按在hub_probe中的调用顺序):
1.INIT_WORK(&hub->events, hub_event);
2.usb_fill_int_urb(hub->urb, hdev, pipe, *hub->buffer, maxp, hub_irq,
hub, endpoint->bInterval);
3.status = usb_submit_urb(hub->urb, GFP_NOIO);
第3处即向root_hub提交一个中断传输的urb,我们来看这个urb的完成函数hub_irq:
这个函数比较简单,最后又调用usb_submit_urb提交中断urb,以实现对中断端点的轮询。
这个中断urb最终会调到root_hub的定时器函数rh_timer_func,具体调用过程请分析usb_create_hcd/usb_add_hcd,usb_submit_urb,这里不赘述
这个定时器函数里,调用hc_driver的hub_status_data读取root hub的端口变化状态
kick_hub_wq实现如下:
向工作队列hub_wq提交了一个工作任务,工作函数为前面第1处的hub_event,
hub_event最终会调到hub_port_connect,处理端口设备的connect,disconnect
如果是connect,则会调用usb_alloc_dev/usb_new_device创建新的usb_device,踏上match与probe usb_device_driver和usb_driver的漫长之旅...