USB Hub 检测设备

server/2024/12/28 14:17:27/

在这里插入图片描述

系列文章目录


xHCI 简单分析
USB Root Hub 分析
USB Hub 检测设备


文章目录

  • 系列文章目录
  • 一、引言
  • 二、hub_events
    • hub_port_connect_change
    • usb_alloc_dev
    • usb_set_device_state
    • hub_port_init
    • usb_new_device


一、引言

    USB Hub 检测设备 一文中讲到,当有 USB 插入时,它会激活 hub_events 函数。

static int hub_thread(void *__unused)
{do {hub_events();wait_event_interruptible(khubd_wait,!list_empty(&hub_event_list) ||kthread_should_stop());try_to_freeze();} while (!kthread_should_stop() || !list_empty(&hub_event_list));pr_debug("%s: khubd exiting\n", usbcore_name);return 0;
}

二、hub_events

在这里插入图片描述

static void hub_events(void)
{// ...while (1) {// ...tmp = hub_event_list.next;list_del_init(tmp);hub = list_entry(tmp, struct usb_hub, event_list);// 描述 usb 设备(Hub,整体,不是接口)hdev = hub->hdev;intf = to_usb_interface(hub->intfdev);// hub 接口设备hub_dev = &intf->dev;// .../* Lock the device, then check to see if we were* disconnected while waiting for the lock to succeed. */if (locktree(hdev) < 0) {// .../* Autoresume */ret = usb_autopm_get_interface(intf);// .../* deal with port status changes */for (i = 1; i <= hub->descriptor->bNbrPorts; i++) {// ...ret = hub_port_status(hub, i,&portstatus, &portchange);// ...if (connect_change)hub_port_connect_change(hub, i,portstatus, portchange);} /* end for i */// ...if (!hdev->parent && !hub->busy_bits[0])usb_enable_root_hub_irq(hdev->bus);loop_autopm:/* Allow autosuspend if we're not going to run again */if (list_empty(&hub->event_list))usb_autopm_enable(intf);
loop:usb_unlock_device(hdev);usb_put_intf(intf);} /* end while (1) */
}

    在这个循环中,检测 Hub 的每个端口是否有变化,有变化则调用 hub_port_connect_change 进行处理。

hub_port_connect_change

static void hub_port_connect_change(struct usb_hub *hub, int port1,u16 portstatus, u16 portchange)
{struct usb_device *hdev = hub->hdev;struct device *hub_dev = hub->intfdev;// .../* Disconnect any existing devices under this port */if (hdev->children[port1-1])usb_disconnect(&hdev->children[port1-1]);clear_bit(port1, hub->change_bits);// ...for (i = 0; i < SET_CONFIG_TRIES; i++) {struct usb_device *udev;// ...udev = usb_alloc_dev(hdev, hdev->bus, port1);// ...usb_set_device_state(udev, USB_STATE_POWERED);udev->speed = USB_SPEED_UNKNOWN;udev->bus_mA = hub->mA_per_port;udev->level = hdev->level + 1;// ...choose_address(udev);// .../* reset and get descriptor */status = hub_port_init(hub, udev, port1, i);// ...if (!status) {status = usb_new_device(udev);// ...}// ...status = hub_power_remaining(hub);// ...
}

    在这个循环中,主要涉及 8 个重量级函数,先点明它们的角色分工。

    第一个函数,usb_alloc_dev(),一个 struct usb_device 结构体指针,申请内存,这个结构体指针可不是为 Hub 准备的,它正是为了 Hub 这个端口所接的设备而申请的,别忘了我们此时此刻的上下文,之所以进入这个循环,是因为我们的 Hub 检测到某个端口有设备连接,所以,Hub 驱动就义不容辞地要为该设备做点什么。

    第二个函数,usb_set_device_state(),这个函数用来设置设备的状态,在 struct usb_device 结构体中,有一个成员 enum usb_device_state state,这一刻会把这个设备的状态设置为 USB_STATE_POWERED,即上电状态。

    第三个函数,choose_address(),为设备选择一个地址。后面会用实例来查看效果。

    第四个函数,hub_port_init(),端口初始化,主要就是前面所讲的获取设备的描述符。

    第五个函数,usb_get_status(),这个函数是专门为 Hub 准备的,不是为当前的这个 Hub,而是说当前 Hub 的这个端口上连接的如果又是 Hub,那么和连接普通设备就不一样。

    第六个函数,check_highspeed(),不同速度的设备,当然待遇不一样。

    第七个函数,usb_new_device()。寻找驱动程序,调用驱动程序的 probe,跟踪这个函数就能一直到设备驱动程序的 probe() 函数的调用。

    第八个函数,hub_power_remaining(),电源管理。

usb_alloc_dev

struct usb_device *
usb_alloc_dev(struct usb_device *parent, struct usb_bus *bus, unsigned port1)
{struct usb_device *dev;// ...device_initialize(&dev->dev);dev->dev.bus = &usb_bus_type;dev->dev.type = &usb_device_type;dev->dev.dma_mask = bus->controller->dma_mask;dev->state = USB_STATE_ATTACHED;INIT_LIST_HEAD(&dev->ep0.urb_list);dev->ep0.desc.bLength = USB_DT_ENDPOINT_SIZE;dev->ep0.desc.bDescriptorType = USB_DT_ENDPOINT;/* ep0 maxpacket comes later, from device descriptor */dev->ep_in[0] = dev->ep_out[0] = &dev->ep0;// ...dev->portnum = port1;dev->bus = bus;dev->parent = parent;INIT_LIST_HEAD(&dev->filelist);#ifdef	CONFIG_PMmutex_init(&dev->pm_mutex);INIT_DELAYED_WORK(&dev->autosuspend, usb_autosuspend_work);dev->autosuspend_delay = usb_autosuspend_delay * HZ;
#endifreturn dev;
}

usb_set_device_state

void usb_set_device_state(struct usb_device *udev,enum usb_device_state new_state)
{unsigned long flags;spin_lock_irqsave(&device_state_lock, flags);if (udev->state == USB_STATE_NOTATTACHED);	/* do nothing */else if (new_state != USB_STATE_NOTATTACHED) {/* root hub wakeup capabilities are managed out-of-band* and may involve silicon errata ... ignore them here.*/if (udev->parent) {if (udev->state == USB_STATE_SUSPENDED|| new_state == USB_STATE_SUSPENDED);	/* No change to wakeup settings */else if (new_state == USB_STATE_CONFIGURED)device_init_wakeup(&udev->dev,(udev->actconfig->desc.bmAttributes& USB_CONFIG_ATT_WAKEUP));elsedevice_init_wakeup(&udev->dev, 0);}udev->state = new_state;} elserecursively_mark_NOTATTACHED(udev);spin_unlock_irqrestore(&device_state_lock, flags);
}

hub_port_init

static int
hub_port_init (struct usb_hub *hub, struct usb_device *udev, int port1,int retry_counter)
{// .../* Reset the device; full speed may morph to high speed */retval = hub_port_reset(hub, port1, udev, delay);// ...for (i = 0; i < GET_DESCRIPTOR_TRIES; (++i, msleep(100))) {if (USE_NEW_SCHEME(retry_counter)) {struct usb_device_descriptor *buf;int r = 0;#define GET_DESCRIPTOR_BUFSIZE	64buf = kmalloc(GET_DESCRIPTOR_BUFSIZE, GFP_NOIO);// ...for (j = 0; j < 3; ++j) {buf->bMaxPacketSize0 = 0;r = usb_control_msg(udev, usb_rcvaddr0pipe(),USB_REQ_GET_DESCRIPTOR, USB_DIR_IN,USB_DT_DEVICE << 8, 0,buf, GET_DESCRIPTOR_BUFSIZE,USB_CTRL_GET_TIMEOUT);// ...}udev->descriptor.bMaxPacketSize0 =buf->bMaxPacketSize0;kfree(buf);retval = hub_port_reset(hub, port1, udev, delay);// ...}for (j = 0; j < SET_ADDRESS_TRIES; ++j) {retval = hub_set_address(udev);if (retval >= 0)break;msleep(200);}// ...retval = usb_get_device_descriptor(udev, 8);// ...
}

usb_new_device

int usb_new_device(struct usb_device *udev)
{// ...usb_detect_quirks(udev);err = usb_get_configuration(udev);// .../* read the standard strings and cache them if present */udev->product = usb_cache_string(udev, udev->descriptor.iProduct);udev->manufacturer = usb_cache_string(udev,udev->descriptor.iManufacturer);udev->serial = usb_cache_string(udev, udev->descriptor.iSerialNumber);// .../* export the usbdev device-node for libusb */udev->dev.devt = MKDEV(USB_DEVICE_MAJOR,(((udev->bus->busnum-1) * 128) + (udev->devnum-1)));// ...err = device_add(&udev->dev);// ...if (udev->parent)usb_autoresume_device(udev->parent);// ...			
}

   
 


http://www.ppmy.cn/server/153934.html

相关文章

【UE5.3.2】生成vs工程并rider打开

Rider是跨平台的,UE也是,当前现在windows上测试首先安装ue5.3.2 会自动有右键的菜单: windows上,右键,生成vs工程 生成的结果 sln默认是vs打开的,我的是vs2022,可以open with 选择 rider :Rider 会弹出 RiderLink是什么插

Elasticsearch 集群

集群结构 以三台物理机为例。在这三台物理机上&#xff0c;搭建了 6 个 ES 的节点&#xff0c;三个 data 节点&#xff0c;三个 master 节点&#xff08;每台物理机分别起了一个 data 和一个 master&#xff09;&#xff0c;3 个 master 节点&#xff0c;目的是达到&#xff0…

YOLO11改进-注意力-引入自调制特征聚合模块SMFA

本篇文章将介绍一个新的改进机制——SMFA&#xff08;自调制特征聚合模块&#xff09;&#xff0c;并阐述如何将其应用于YOLOv11中&#xff0c;显著提升模型性能。随着深度学习在计算机视觉中的不断进展&#xff0c;目标检测任务也在快速发展。YOLO系列模型&#xff08;You Onl…

软考-信息安全-网络安全体系与网络安全模型

4.1 网络安全体系概述 网络安全保障是一项复杂的系统工程&#xff0c;是安全策略&#xff0c;多种技术&#xff0c;管理方法和人员安全素质的综合。 4.1.1 网络安全体系概念 现代的网络安全问题变化莫测&#xff0c;要保障网络系统的安全&#xff0c;应当把相应的安全策略&a…

虚幻引擎是什么?

Unreal Engine&#xff0c;是一款由Epic Games开发的游戏引擎。该引擎主要是为了开发第一人称射击游戏而设计&#xff0c;但现在已经被成功地应用于开发模拟游戏、恐怖游戏、角色扮演游戏等多种不同类型的游戏。虚幻引擎除了被用于开发游戏&#xff0c;现在也用于电影的虚拟制片…

Android14 OTA升级速度过慢问题解决方案

软件版本&#xff1a;Android14 硬件平台&#xff1a;QCS6115 问题&#xff1a;OTA整包升级接近20min&#xff0c;太长无法忍受。 该问题为Android高版本的虚拟AB分区压缩技术所致&#xff0c;其实就是时间换空间&#xff0c;个人推测AB分区压缩会节约硬件存储空间&#xff0…

EKF 自动匹配维度 MATLAB代码

该 M A T L A B MATLAB MATLAB代码实现了扩展卡尔曼滤波( E

Python 和 Pandas 版本的对应关系

在使用 Python 和 Pandas 时&#xff0c;确保它们的版本兼容是非常重要的。以下是 Python 和 Pandas 版本的对应关系&#xff0c;帮助你选择合适的版本。 安装示例 如果你使用的是 Python 3.9&#xff0c;并希望安装与之兼容的 Pandas 版本&#xff0c;可以使用以下命令&#…