话说上一章,介绍了host获取设备描述符的过程,主要的思路就是先获取9 bytes,从这9 bytes中获取配置描述符的长度,然后一次性获取全部的配置描述符,配置描述符其实是配置描述符信息 + 端口描述符信息的总和,在设备内存中的分布如下图所示。有些设备可能不只一个配置描述符(业界称之为:符合设备),也同样是这样的分布。

获取了配置描述符后,对其进行解析。

static int usb_parse_configuration(struct usb_device *dev, int cfgidx,struct usb_host_config *config, unsigned char *buffer, int size)
{struct device *ddev = &dev->dev;unsigned char *buffer0 = buffer;int cfgno;int nintf, nintf_orig;int i, j, n;struct usb_interface_cache *intfc;unsigned char *buffer2;int size2;struct usb_descriptor_header *header;int len, retval;u8 inums[USB_MAXINTERFACES], nalts[USB_MAXINTERFACES];unsigned iad_num = 0;memcpy(&config->desc, buffer, USB_DT_CONFIG_SIZE);    /*将读取到的配置描述符的前9个bytes拷贝到设备的描述符缓存中*/if (config->desc.bDescriptorType != USB_DT_CONFIG ||   /* 检查读取到配置的合法性 */config->desc.bLength < USB_DT_CONFIG_SIZE ||config->desc.bLength > size) {dev_err(ddev, "invalid descriptor for config index %d: ""type = 0x%X, length = %d\n", cfgidx,config->desc.bDescriptorType, config->desc.bLength);return -EINVAL;}cfgno = config->desc.bConfigurationValue;      /* 获取配置的编号 */buffer += config->desc.bLength;              /* 跳过配置描述符端点的长度,接下来就是端点描述符的起始地址 */size -= config->desc.bLength;                /*  得到端点描述符的长度 */nintf = nintf_orig = config->desc.bNumInterfaces;   /* 获取端点配置的数目 */if (nintf > USB_MAXINTERFACES) {dev_warn(ddev, "config %d has too many interfaces: %d, ""using maximum allowed: %d\n",cfgno, nintf, USB_MAXINTERFACES);nintf = USB_MAXINTERFACES;}/* Go through the descriptors, checking their length and counting the* number of altsettings for each interface */n = 0;for ((buffer2 = buffer, size2 = size);    /* 依次获取每一个接口描述符 */size2 > 0;(buffer2 += header->bLength, size2 -= header->bLength)) {if (size2 < sizeof(struct usb_descriptor_header)) {    /* 剩下接口长度少于标准长度,则退出,不再解析 */dev_warn(ddev, "config %d descriptor has %d excess ""byte%s, ignoring\n",cfgno, size2, plural(size2));break;}header = (struct usb_descriptor_header *) buffer2;if ((header->bLength > size2) || (header->bLength < 2)) {   /* 接口描述符长度合法性检查 */dev_warn(ddev, "config %d has an invalid descriptor ""of length %d, skipping remainder of the config\n",cfgno, header->bLength);break;}if (header->bDescriptorType == USB_DT_INTERFACE) {     /* 接口描述符类型检查,是接口描述符再解析 */struct usb_interface_descriptor *d;int inum;d = (struct usb_interface_descriptor *) header;if (d->bLength < USB_DT_INTERFACE_SIZE) {  /* 接口描述符长度合法性检查 */dev_warn(ddev, "config %d has an invalid ""interface descriptor of length %d, ""skipping\n", cfgno, d->bLength);continue;}inum = d->bInterfaceNumber;    /* 获取接口描述编号 */if ((dev->quirks & USB_QUIRK_HONOR_BNUMINTERFACES) &&n >= nintf_orig) {         /* 合法性检测 */dev_warn(ddev, "config %d has more interface ""descriptors, than it declares in ""bNumInterfaces, ignoring interface ""number: %d\n", cfgno, inum);continue;}if (inum >= nintf_orig)dev_warn(ddev, "config %d has an invalid ""interface number: %d but max is %d\n",cfgno, inum, nintf_orig - 1);/* Have we already encountered this interface?* Count its altsettings */for (i = 0; i < n; ++i) {        /* for循环的俄作用只有一个: 改变i的值,使得数组项被依次赋值 */if (inums[i] == inum)        /* 如果inums数组中的当前接口配置描述符被识别过,则跳出 */break;                  /* 未被识别,则++i,从而增加nalts数组和inums数组 */}if (i < n) {if (nalts[i] < 255)++nalts[i];} else if (n < USB_MAXINTERFACES) {inums[n] = inum;             /* inums 数组,里面的每一项表示一个接口配置的编号 */nalts[n] = 1;                /* nalts 数组,里面每一项表示接口配置被正常识别,和 inums 一一对应 */ ++n;}} else if (header->bDescriptorType ==    /*其他配置符的处理 */USB_DT_INTERFACE_ASSOCIATION) {if (iad_num == USB_MAXIADS) {dev_warn(ddev, "found more Interface ""Association Descriptors ""than allocated for in ""configuration %d\n", cfgno);} else {config->intf_assoc[iad_num] =(struct usb_interface_assoc_descriptor*)header;iad_num++;}} else if (header->bDescriptorType == USB_DT_DEVICE ||header->bDescriptorType == USB_DT_CONFIG)dev_warn(ddev, "config %d contains an unexpected ""descriptor of type 0x%X, skipping\n",cfgno, header->bDescriptorType);}    /* for ((buffer2 = buffer, size2 = size); ...) */size = buffer2 - buffer;config->desc.wTotalLength = cpu_to_le16(buffer2 - buffer0);if (n != nintf)dev_warn(ddev, "config %d has %d interface%s, different from ""the descriptor's value: %d\n",cfgno, n, plural(n), nintf_orig);else if (n == 0)dev_warn(ddev, "config %d has no interfaces?\n", cfgno);config->desc.bNumInterfaces = nintf = n;/* Check for missing interface numbers */for (i = 0; i < nintf; ++i) {for (j = 0; j < nintf; ++j) {if (inums[j] == i)break;}if (j >= nintf)dev_warn(ddev, "config %d has no interface number ""%d\n", cfgno, i);}/* Allocate the usb_interface_caches and altsetting arrays */for (i = 0; i < nintf; ++i) {j = nalts[i];if (j > USB_MAXALTSETTING) {dev_warn(ddev, "too many alternate settings for ""config %d interface %d: %d, ""using maximum allowed: %d\n",cfgno, inums[i], j, USB_MAXALTSETTING);nalts[i] = j = USB_MAXALTSETTING;}len = sizeof(*intfc) + sizeof(struct usb_host_interface) * j;config->intf_cache[i] = intfc = kzalloc(len, GFP_KERNEL);if (!intfc)return -ENOMEM;kref_init(&intfc->ref);}/* FIXME: parse the BOS descriptor *//* Skip over any Class Specific or Vendor Specific descriptors;* find the first interface descriptor */config->extra = buffer;i = find_next_descriptor(buffer, size, USB_DT_INTERFACE,USB_DT_INTERFACE, &n);config->extralen = i;if (n > 0)dev_dbg(ddev, "skipped %d descriptor%s after %s\n",n, plural(n), "configuration");buffer += i;size -= i;/* Parse all the interface/altsetting descriptors */while (size > 0) {retval = usb_parse_interface(ddev, cfgno, config,    /* 解析接口配置描述符 */buffer, size, inums, nalts);if (retval < 0)return retval;buffer += retval;size -= retval;}

解析的结果是配置可选项存储在inums[n]  数组中,且在设备端分配了足够的空间来存储在这些配置信息,为接口配置符解析做了充足的准备。

retval = usb_parse_interface(ddev, cfgno, config,    /* 解析接口配置描述符 */buffer, size, inums, nalts);先整理传进来的参数
ddev: 需要被解析的设备
cfgno:配置描述符的编号
config: 配置描述符
buffer:指向接口描述符的起始地址
size: 接口描述符的大小
inums:接口描述符个数数组
nalts:接口描述符对应的数组

USB 设备驱动之设备接入梳理(六)相关推荐

  1. Linux设备驱动和设备匹配过程

    Linux设备驱动和设备匹配过程 1. 设备驱动匹配简述 2. 重点结构体介绍 2.1 `struct device` 2.2 `struct platform_device` 2.3 `struct ...

  2. 字符设备驱动、平台设备驱动、设备驱动模型、sysfs的比较和关联

    参考原文:https://www.kancloud.cn/yueqian_scut/emlinux/106829 对原文笔误地方做了修改.重新排版 目录 字符设备驱动.平台设备驱动.设备驱动模型.sy ...

  3. LINUX设备驱动之设备模型一--kobject

    http://blog.csdn.net/yangzhu1982/article/details/6186016 Linux设备驱动之设备模型一kobject Eric Fang  2010-01-1 ...

  4. 【驱动】linux设备驱动·字符设备驱动开发

    Preface 前面对linux设备驱动的相应知识点进行了总结,现在进入实践阶段! <linux设备驱动入门篇>:http://infohacker.blog.51cto.com/6751 ...

  5. linux设备驱动--字符设备模型

    linux设备驱动--字符设备模型 最近正在学习设备驱动开发,因此打算写一个系列博客,即是对自己学习的一个总结,也是对自己的一个督促,有不对,不足,需要改正的地方还望大家指出,而且希望结识志同道合的朋 ...

  6. LINUX设备驱动之设备模型一kobject

    LINUX设备驱动之设备模型一kobject -------------------------------------------------------------- 转载请注明出处:http:/ ...

  7. linux_设备驱动_设备树

    一.什么是DTS?为什么要引入DTS? DTS即Device Tree Source 设备树源码, Device Tree是一种描述硬件的数据结构,它起源于 OpenFirmware (OF). 在L ...

  8. Linux 设备驱动开发 —— 设备树在platform设备驱动中的使用

    关与设备树的概念,我们在Exynos4412 内核移植(六)-- 设备树解析 里面已经学习过,下面看一下设备树在设备驱动开发中起到的作用 Device Tree是一种描述硬件的数据结构,设备树源(De ...

  9. linux fb设备驱动,linux设备驱动归纳总结(八):1.总线、设备和驱动

    linux设备驱动归纳总结(八):1.总线.设备和驱动 xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx ...

  10. Linux设备模型、平台设备驱动、设备树(device tree)、GPIO子系统以及pinctrl子系统介绍

    文章目录 一.Linux设备模型介绍 (1)设备驱动模型总体介绍 (2)设备驱动模型文件表现 (3)设备驱动模型工作原理 [1]总线 [2]设备 [3]驱动 [4]注册流程 二.平台设备驱动介绍 (1 ...

最新文章

  1. 识别迷雾中的物体,谷歌提出最新目标检测算法Context R-CNN
  2. vim切换编程语言_把 Vim 打造成源代码编辑器 - C 语言编程透视
  3. Elasticsearch-Jest 配置ES集群源码解读
  4. 全局对象_C++全局变量初始化
  5. 数学知识总结——矩阵
  6. Nodejs学习(三)-安装nodejs supervisor,提高点效率吧。
  7. 设备底座几个常见固定方式
  8. Win7电脑快速获取超级管理员权限的方法
  9. 参考文献中杂志名字问题
  10. Qt入门(11)——Qt插件
  11. oracle blob update,Oracle数据库中对BLOB数据的操作问题
  12. mysql表 spid program_oracle 解锁某张表 和编译存储过程卡死问题处理
  13. 小程序切出去重新进入
  14. OpenERP QWeb模板标签笔记
  15. python if并列条件_Python中if有多个条件怎么办
  16. sqlplus登录缓慢的解决
  17. 某厂机试算法刷题一览
  18. CS:APP 计算机系统 课程大作业
  19. 2012年每周推荐阅读汇总
  20. 那些你可能用得上的在线办公神器

热门文章

  1. 【总结】FLANN特征匹配
  2. 手把手做一个JSP入门程序(九):购物车的基本实现(Servlet)
  3. 计算机存储器的有关术语,关于计算机存储器,不正确的描述是()。
  4. 小布老师讲座笔记(五)
  5. 【递归 动态规划 备忘录法】Fibonacci数列(斐波那契数列)(C++)
  6. apk包的破解与反编译
  7. Ubuntu 13.04下安装RabbitVCS,类似Windows的TortoiseSVN
  8. 【深圳】大湾区第三次.NET技术交流会(网络直播)
  9. Nginx负载均衡策略 - least_conn 最少连接
  10. Android Studio使用技巧系列教程(二)