USB Hub 检测设备

在这里插入图片描述

系列文章目录


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.mzph.cn/pingmian/65302.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

MySQL从入门到入土---MySQL表的约束 (内含实践)---详细版

目录 引入&#xff1a; null 与not null default&#xff1a; comment列描述 &#xff1a; not null 和 default&#xff1a; zerofill &#xff1a; 主键&#xff1a;primary key 复合主键&#xff1a; 自增长:auto_increment 唯一键&#xff1a;unique key 外键&a…

易语言 OCR 文字识别

一.引言 文字识别&#xff0c;也称为光学字符识别&#xff08;Optical Character Recognition, OCR&#xff09;&#xff0c;是一种将不同形式的文档&#xff08;如扫描的纸质文档、PDF文件或数字相机拍摄的图片&#xff09;中的文字转换成可编辑和可搜索的数据的技术。随着技…

基于SpringBoot的蜗牛兼职网的设计与实现

一、项目背景 随着社会的快速发展&#xff0c;计算机的影响是全面且深入的。人们生活水平的不断提高&#xff0c;日常生活中人们对蜗牛兼职网方面的要求也在不断提高&#xff0c;需要兼职工作的人数更是不断增加&#xff0c;使得蜗牛兼职网的开发成为必需而且紧迫的事情。蜗牛…

shardingsphere分库分表项目实践5-自己用java写一个sql解析器+完整项目源码

前1节我们介绍了 shardingsphere 分表分库的sql解析与重写&#xff1a; shardingsphere分库分表项目实践4-sql解析&重写-CSDN博客 那么shardingsphere sql 解析底层究竟是怎么实现的呢&#xff0c;其实它直接用了著名的开源软件 antlr . antlr 介绍&#xff1a; ANTLR&a…

光谱相机与普通相机的区别

一、成像目的 普通相机&#xff1a;主要目的是记录物体的外观形态&#xff0c;生成人眼可见的、直观的二维图像&#xff0c;重点在于还原物体的形状、颜色和纹理等视觉特征&#xff0c;以供人们进行观赏、记录场景或人物等用途。例如&#xff0c;拍摄旅游风景照片、人物肖像等…

TiDB 的MPP架构概述

MPP架构介绍&#xff1a; 如图&#xff0c;TiDB Server 作为协调者&#xff0c;首先 TiDB Server 会把每个TiFlash 拥有的region 会在TiFlash上做交换&#xff0c;让表连接在一个TiFlash上。另外 TiFlash会作为计算节点&#xff0c;每个TiFlash都负责数据交换&#xff0c;表连接…

渗透Vulnhub-Solidstate靶机

本篇文章旨在为网络安全渗透测试行业靶机教学。通过阅读本文&#xff0c;读者将能够对渗透Vulnhub系列Solidstate靶机有定的了解 一、信息收集阶段 靶机官网&#xff1a;https://www.vulnhub.com/entry/solidstate-1%2C261/ 因为靶机为本地部署虚拟机网段&#xff0c;查看dhcp…

ElasticSearch - 深入解析 Elasticsearch Composite Aggregation 的分页与去重机制

文章目录 Pre概述什么是 composite aggregation&#xff1f;基本结构after 参数的作用问题背景&#xff1a;传统分页的重复问题after 的设计理念响应示例 after 如何确保数据不重复核心机制Example步骤 1: 创建测试数据创建索引插入测试数据 步骤 2: 查询第一页结果查询第一页返…

两分钟掌握 TDengine 全部写入方式

1. 背景 TDengine 写入过程会涉及很多概念&#xff0c;这些概念目前你是不是还一团乱&#xff0c;参数绑定写入、无模式写入、websocket 写入、RESTFUL 写入 、各种连接器写入等等一堆的写入&#xff0c;都是做什么的&#xff0c;不明白&#xff0c;这里花两分钟时间给你彻底整…

快速理解24种设计模式

简单工厂模式 建立产品接口类&#xff0c;规定好要实现方法。 建立工厂类&#xff0c;根据传入的参数&#xff0c;实例化所需的类&#xff0c;实例化的类必须实现指定的产品类接口 创建型 单例模式Singleton 保证一个类只有一个实例&#xff0c;并提供一个访问他它的全局…

数据可视化echarts学习笔记

目录&#xff0c;介绍 知识储备 一端操作&#xff0c;多端联动的效果&#xff08;开启了多个网页&#xff0c;操作一端&#xff0c;多个网页的效果会跟着改变&#xff09; cmd命令控制面板返回上一级或上上级 在当前目录打开文件&#xff1a; cd 文件名 在Windows命令提示符&am…

OpenCV相机标定与3D重建(30)过滤二值图像中的小斑点函数filterSpeckles()的使用

操作系统&#xff1a;ubuntu22.04 OpenCV版本&#xff1a;OpenCV4.9 IDE:Visual Studio Code 编程语言&#xff1a;C11 算法描述 在视差图中过滤掉小的噪声斑点&#xff08;speckles&#xff09;。 cv::filterSpeckles 是 OpenCV 库中的一个函数&#xff0c;用于过滤图像或视…

C语言期末复习笔记(中)

目录 五、选择控制结构 1.算法中的概念及描述方法 2.关系运算符和逻辑表达式 3.条件运算符和条件表达式 4.两种多分支if 5.switch语句 6.逻辑运算符和逻辑表达式 六、循环控制结构 1.控制循环的方式 2.控制非法输入 3.选择三种循环的一般原则 4.猜数游戏 5.嵌套循环…

利用Gurobi追溯模型不可行原因的四种方案及详细案例

文章目录 1. 引言2. 追溯不可行集的四种方法2.1 通过约束增减进行判断2.2 通过computeIIS函数获得冲突集2.3 利用 feasRelaxS() 或 feasRelax() 函数辅助排查2.4 利用 IIS Force 属性1. 引言 模型不可行是一个让工程师头疼的问题,对于复杂模型而言,导致模型不可行的原因可能…

MySQL和HBase的对比

Mysql &#xff1a;关系型数据库&#xff0c;主要面向 OLTP &#xff0c;支持事务&#xff0c;支持二级索引&#xff0c;支持 sql &#xff0c;支持主从、 Group Replication 架构模型&#xff08;此处以 Innodb 为例&#xff0c;不涉及别的存储引擎&#xff09;。 HBase &am…

mybatis-plus自动填充时间的配置类实现

mybatis-plus自动填充时间的配置类实现 在实际操作过程中&#xff0c;我们并不希望创建时间、修改时间这些来手动进行&#xff0c;而是希望通过自动化来完成&#xff0c;而mybatis-plus则也提供了自动填充功能来实现这一操作&#xff0c;接下来&#xff0c;就来了解一下mybatis…

【软件工程】十万字知识点梳理 | 期末复习专用

原创文章,禁止转载。 文章目录 图CRC卡片用例图类图状态图活动图泳道图软件质量因素自顶向下集成自底向上集成人员与工作量之间的关系时序图关键路径软件结构基本路径测试判定表数据流图(DFD)体系结构设计问题数据字典挣值分析等价划分程序流程图PAD | N-S燃尽图甘特图对象模…

STM32完全学习——FLASH上FATFS文件管理系统

一、需要移植的接口 我们通过看官网的手册&#xff0c;可以看到我们只要完成下面函数的实现&#xff0c;就可以完成移植。我们这里只移植前5个函数&#xff0c;获取时间的函数我们不在这里移植。 二、移植接口函数 DSTATUS disk_status (BYTE pdrv /* Physical drive nmuber…

Redis - Token JWT 概念解析及双token实现分布式session存储实战

Token 定义&#xff1a;令牌&#xff0c;访问资源接口&#xff08;API&#xff09;时所需要的资源凭证 一、Access Token 定义&#xff1a;访问资源接口&#xff08;API&#xff09;时所需要的资源凭证&#xff0c;存储在客户端 组成 组成部分说明uid用户唯一的身份标识time…

软体机器人研究报告:设计方法、材料与驱动、感知与控制

软体机器人因其出色的可变形性和高适应性受到了广泛关注&#xff0c;这些特性使其在医疗、救援、探测等复杂场景中展现出独特的优势和巨大的应用潜力。研究人员对软体机器人的设计方法、材料与驱动技术、感知与控制策略等方面进行深入研究&#xff0c;取得了一系列成果。 本文汇…