Linux libusb USB开发(一)—— USB设备基础概念

原文链接:添加链接描述

在终端用户看来,USB设备为主机提供了多种多样的附加功能,如文件传输,声音播放等,但对USB主机来说,它与所有USB设备的接口都是一致的。一个USB设备由3个功能模块组成:USB总线接口、USB逻辑设备和功能单元:

a – 这里的USB总线接口指的是USB设备中的串行接口引擎(SIE);

b – USB逻辑设备被USB系统软件看作是一个端点的集合;

c – 功能单元被客户软件看作是一个接口的集合。SIE、端点和接口都是USB设备的组成单元;

为了更好地描述USB设备的特征,USB提出了设备架构的概念。从这个角度来看,可以认为USB设备是由一些配置、接口和端点组成,即一个USB设备可以含有一个或多个配置,在每个配置中可含有一个或多个接口,在每个接口中可含有若干个端点。其中,配置和接口是对USB设备功能的抽象,实际的数据传输由端点来完成。在使用USB设备前,必须指明其采用的配置和接口。这个步骤一般是在设备接入主机时设备进行枚举时完成的。

这些单元之间的关系如下:

设备通常有一个或多个配置;

配置通常有一个或多个接口;

接口通常有一个或多个设置;

接口有零或多个端点。

这样的概念太抽象了,可以这样看:有一个设备,如支持视频和音频的一个播放器。那么,对于上面提到的4个描述符,对它们设置的时候,它们分别对于哪一个描述符呢?

从我现在的理解来看,这样一个设备对应一个设备描述符,支持视频的功能对应一个接口描述符,支持音频功能的对应一个接口描述符。为了支持视频,在下层有多个端口同时工作为提供视频数据传输的支持,所以有多个端点描述符。

1、设备描述符

设备代表一个USB设备,它由一个或多个配置组成。设备描述符用于说明设备的总体信息,并指明其所含的配置的个数。一个USB设备只能有一个设备描述符。

struct usb_device_descriptor
{
_ _u8 bLength; //描述符长度
_ _u8 bDescriptorType; //描述符类型编号

_ _le16 bcdUSB; //USB版本号
_ _u8 bDeviceClass; //USB分配的设备类code
_ _u8 bDeviceSubClass;// USB分配的子类code
_ _u8 bDeviceProtocol; //USB分配的协议code
_ _u8 bMaxPacketSize0; //endpoint0最大包大小
_ _le16 idVendor; //厂商编号
_ _le16 idProduct; //产品编号
_ _le16 bcdDevice; //设备出厂编号
_ _u8 iManufacturer; //描述厂商字符串的索引
_ _u8 iProduct; //描述产品字符串的索引
_ _u8 iSerialNumber; //描述设备序列号字符串的索引
_ _u8 bNumConfigurations; //可能的配置数量

} _ attribute _ ((packed));
2、配置描述符

一个USB设备可以包含一个或多个配置,如USB设备的低功耗模式和高功耗模式可分别对应一个配置。在使用USB设备前,必须为其选择一个合适的配置。配置描述符用于说明USB设备中各个配置的特性,如配置所含接口的个数等。USB设备的每一个配置都必须有一个配置描述符。

struct usb_config_descriptor
{
_ _u8 bLength; //描述符长度
_ _u8 bDescriptorType; //描述符类型编号

_ _le16 wTotalLength; //配置所返回的所有数据的大小
_ _u8 bNumInterfaces; // 配置所支持的接口数
_ _u8 bConfigurationValue; //Set_Configuration命令需要的参数值
_ _u8 iConfiguration; //描述该配置的字符串的索引值
_ _u8 bmAttributes; //供电模式的选择
_ _u8 bMaxPower; //设备从总线提取的最大电流

} _ attribute _ ((packed));
3、接口描述符

一个配置可以包含一个或多个接口,例如对一个光驱来说,当用于文件传输时,使用其大容量存储接口;而当用于播放CD时,使用其音频接口。接口是端点的集合,可以包含一个或多个可替换设置,用户能够在USB处于配置状态时改变当前接口所含的个数和特性。接口描述符用于说明设备中各个接口的特性,如接口所属的设备类及其子类等。USB设备的每个接口都必须有一个接口描述符

struct usb_interface_descriptor
{
_ _u8 bLength; //描述符长度
_ _u8 bDescriptorType; //描述符类型

_ _u8 bInterfaceNumber;   // 接口的编号
_ _u8 bAlternateSetting; //备用的接口描述符编号
_ _u8 bNumEndpoints;      //该接口使用的端点数,不包括端点0
_ _u8 bInterfaceClass;    //接口类型
_ _u8 bInterfaceSubClass; //接口子类型
_ _u8 bInterfaceProtocol; //接口所遵循的协议
_ _u8 iInterface; //描述该接口的字符串索引值

} _ attribute _ ((packed));
4、端点描述符

端点是USB设备中的实际物理单元,USB数据传输就是在主机和USB设备各个端点之间进行的。端点一般由USB接口芯片提供,例如Freescale公司的MC68HC908JB8和MC9S12UF32。USB设备中的每一个端点都有唯一的端点号,每个端点所支持的数据传输方向一般而言也是确定的:或是输入(IN),或是输出(OUT)。也有些芯片提供的端点的数据方向是可以配置的,例如MC68HC908JB8包含有两个用于数据收发的端点:端点1和端点2。其中端点1只能用于数据发送,即支持输入(IN)操作;端点2既能用于数据发送,也可用于数据接收,即支持输入(IN)和输出(OUT)操作。而MC9S12UF32具有6个端点。

利用设备地址、端点号和传输方向就可以指定一个端点,并与它进行通信。端点的传输特性还决定了其与主机通信是所采用的传输类型,例如控制端点只能使用控制传输。根据端点的不同用途,可将端点分为两类:0号端点和非0号端点。

0号端点比较特殊,它有数据输入IN和数据输出OUT两个物理单元,且只能支持控制传输。所有的USB设备都必须含有一个0号端点,用作默认控制管道。USB系统软件就是使用该管道与USB逻辑设备进行配置通信的。0号端点在USB设备上的以后就可以使用,而非0号端点必须要在配置以后才可以使用。

根据具体应用的需要,USB设备还可以含有多个除0号端点以外的其他端点。对于低速设备,其附加的端点数最多为2个;对于全速/高速设备,其附加的端点数最多为15个。

struct usb_endpoint_descriptor
{
_ _u8 bLength; //描述符长度
_ _u8 bDescriptorType; //描述符类型
_ _u8 bEndpointAddress; //端点地址:0~3位是端点号,第7位是方向(0-OUT,1-IN)
_ _u8 bmAttributes; //端点属性:bit[0:1] 的值为00表示控制,为01表示同步,为02表示批量,为03表示中断
_ _le16 wMaxPacketSize; 本端点接收或发送的最大信息包的大小
_ _u8 bInterval;//轮询数据传送端点的时间间隔
//对于批量传送的端点以及控制传送的端点,此域忽略
//对于同步传送的端点,此域必须为1
_ _u8 bRefresh;
_ _u8 bSynchAddress;
} _ attribute _ ((packed));
5、字符串描述符

在USB设备中通常还含有字符串描述符,以说明一些专用信息,如制造商的名称、设备的序列号等。它的内容以UNICODE的形式给出,且可以被客户软件所读取。对USB设备来说,字符串描述符是可选的。

struct usb_string_descriptor
{
_ _u8 bLength; //描述符长度
_ _u8 bDescriptorType; //描述符类型

_ _le16 wData[1];

} _ attribute _ ((packed));
6、管道

在USB系统结构中,可以认为数据传输时在USB主机软件与USB设备的各个端点之间直接进行的,它们之间的连接称为管道。管道是在USB设备的配置过程中建立的。管道是对USB主机与USB设备间通信流的抽象,表示USB主机的数据缓冲区与USB设备的端点之间存在着逻辑数据传输,而实际的数据传输是由USB总线接口层来完成的。

管道与USB设备中的端点一一对应。一个USB设备含有多少个端点,其与USB主机进行通信时就可以使用多少条管道,且端点的类型决定了管道中数据的传输类型,例如中断端点对应中断管道,且该管道只能进行中断传输。不论存在着多少条管道,在各个管道中进行的数据传输都是相互独立的。

7、USB端点分类

USB 通讯的最基本形式是通过端点。一个USB端点只能向一个方向传输数据(从主机到设备(称为输出端点)或者从设备到主机(称为输入端点))。端点可被看作一个单向的管道。

USB 端点有 4 种不同类型, 分别具有不同的数据传送方式:

  1. 控制CONTROL

控制端点被用来控制对USB设备的不同部分访问. 通常用作配置设备、获取设备信息、发送命令到设备或获取设备状态报告。这些端点通常较小。每个 USB 设备都有一个控制端点称为"端点 0", 被 USB 核心用来在插入时配置设备。USB协议保证总有足够的带宽留给控制端点传送数据到设备.

  1. 中断INTERRUPT

每当 USB 主机向设备请求数据时,中断端点以固定的速率传送小量的数据。此为USB 键盘和鼠标的主要的数据传送方法。它还用以传送数据到USB设备来控制设备。通常不用来传送大量数据。USB协议保证总有足够的带宽留给中断端点传送数据到设备.

  1. 批量BULK

批量端点用以传送大量数据。这些端点通常比中断端点大得多. 它们普遍用于不能有任何数据丢失的情况。USB 协议不保证传输在特定时间范围内完成。如果总线上没有足够的空间来发送整个BULK包,它被分为多个包进行传输。这些端点普遍用于打印机、USB Mass Storage和USB网络设备上。

  1. 等时ISOCHRONOUS

等时端点也批量传送大量数据, 但是这个数据不被保证能送达。这些端点用在可以处理数据丢失的设备中,并且更多依赖于保持持续的数据流。如音频和视频设备等等。

控制和批量端点用于异步数据传送,而中断和等时端点是周期性的。这意味着这些端点被设置来在固定的时间连续传送数据,USB 核心为它们保留了相应的带宽。

struct usb_host_endpoint{
struct usb_endpoint_descriptor desc;//端点描述符
struct list_head urb_list;//此端点的URB对列,由USB核心维护
void hcpriv;
struct ep_device ep_dev; / For sysfs info /
unsigned char
extra;/
Extra descriptors */
int extralen;
int enabled;
};
当调用USB设备驱动调用usb_submit_urb提交urb请求时,将调用int usb_hcd_link_urb_to_ep(struct usb_hcd *hcd, struct urb *urb)把此urb增加到urb_list的尾巴上。(hcd: Host Controller Driver,对应数据结构struct usb_hcd )

8、USB设备参数查看方法

1)调用libusb库中的函数进行查询,后续博客会提及

2)Linux下查看方法

Linux系统下可查看USB设备信息的命令有:

lsusb

cat usb/devices

3)Windows下查看方法

Windows系统下可查看USB设备信息的方法有:

点击设备管理器,查看设备属性

USB查看器usbtreeviewV3.1.0软件 可查看USB设备详细信息

USB抓包工具–Bus Hound 可查看USB的通讯信息

参考链接

https://blog.csdn.net/sjin_1314/article/details/25366459

https://blog.csdn.net/zqixiao_09/article/details/50984074

https://www.cnblogs.com/tianchiyuyin/p/5139948.html

https://blog.csdn.net/qq_25544855/article/details/79313496

https://www.cnblogs.com/rusking/p/6107989.html

https://blog.csdn.net/encourage2011/article/details/53525297

https://blog.csdn.net/ericwuhk/article/details/79228117

https://blog.csdn.net/wy_stutdy/article/details/73326330

  • 2
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在Linux下进行USB设备开发有一些基本的步骤和工具。 首先,为了进行USB设备开发工作,我们需要一个支持USB协议栈的操作系统,而Linux是一个开源的操作系统,自带了完整的USB驱动程序和协议栈,因此非常适合进行USB设备开发。 其次,为了在Linux下进行USB设备开发,我们需要选择一种编程语言和开发工具。Linux提供了各种编程语言的支持,包括C、C++和Python等。在编程语言的选择上,可以根据个人的偏好和需求进行选择。 通常,我们可以使用Linux内核提供的libusb库来进行USB设备开发libusb是一个用户态的USB开发库,可以方便地进行设备的操作和数据传输。通过libusb库,我们可以在代码中进行设备的初始化、设备的打开和关闭、发送和接收数据等操作。 在开发过程中,我们还需要了解USB协议和设备的通信规范。USB协议包括设备的描述符、端点的配置、数据传输的方式等等,这些都会对设备开发和使用产生重要影响。同时,我们还需要了解设备的规格和报文格式,这将有助于理解设备的功能和通信机制。 在编写代码之前,我们可以先使用lsusb命令查看已连接到系统的USB设备,以及其供应商ID和产品ID等信息。这将帮助我们确定设备的型号和特性。 最后,在进行USB设备开发之前,我们需要确保系统内核已经加载了相应的USB驱动程序,并且设备已经正确连接到系统上。 总之,在Linux下进行USB设备开发,我们可以利用Linux操作系统本身提供的丰富资源,结合相应的开发工具和库,使得开发工作更加高效和方便。 ### 回答2: 在Linux下进行USB设备开发通常需要以下步骤: 1. 硬件准备:首先需要有一个可用的USB设备,可以是USB键盘、鼠标、摄像头等等。确保该设备可以连接到计算机上,通常通过USB接口。 2. 安装开发环境:在Linux下进行USB设备开发,需要安装一些必要的软件包和工具。最常用的是libusb库,它提供了一组用于访问和通信USB设备的API函数。可以使用包管理工具(如apt、yum等)来安装这些软件包。 3. 编写代码:使用C或者C++等编程语言,结合libusb库编写代码来实现需要的功能。首先需要初始化libusb库,并找到待连接的USB设备。然后可以通过读取和写入数据来与USB设备进行通信。根据具体的设备功能,可能需要实现一些特定的协议或者处理一些特定的命令。 4. 构建和调试:根据开发完成的代码,使用编译器进行构建,生成可执行文件。然后可以进行调试。可以使用调试工具(如gdb)来查看代码的执行过程,以及检查变量的值和内存的使用情况。 5. 测试和部署:完成代码的编写和调试后,可以进行测试。连接USB设备到计算机上,并运行开发完成的程序,验证功能是否正常。如果测试通过,可以将程序部署到目标设备上,让用户使用。 总结来说,Linux下进行USB设备开发需要安装相应的开发环境和工具,编写代码实现功能,进行构建和调试,最后进行测试和部署。 ### 回答3: 在Linux下进行USB设备开发是一项相对简单和广泛支持的任务。 首先,我们需要了解Linux内核对USB设备的支持。内核中的usbcore模块和相关的驱动程序提供了对USB设备的通用支持。这些驱动程序可以识别连接的USB设备,并通过系统的USB子系统将其呈现给应用程序。 在开发USB设备时,我们需要了解USB协议和相关规范。USB设备通常包括硬件和固件两部分。硬件部分包括USB接口芯片和设备连接的外部电路。固件部分是运行在设备上的软件,用来控制设备的功能和与主机进行通信。 在Linux中,我们可以使用libusb库来与USB设备进行通信。这个库提供了一组API函数,允许应用程序读写USB设备的寄存器、控制传输和批量传输等操作。我们可以使用C或C++等编程语言开发应用程序,并使用这个库来进行USB设备的访问和控制。 另外,Linux还提供了一些用于USB设备开发的工具和调试工具。其中,lsusb命令可以用来列出连接到系统上的USB设备信息,包括设备ID、供应商ID和产品ID等。usbmon工具可以用来监视USB总线上的数据传输,帮助我们分析和调试USB通信问题。 总的来说,Linux提供了丰富的支持和工具,使得USB设备开发变得相对容易。我们只需要了解USB协议、设备硬件和固件的工作原理,并使用适当的工具和库来进行开发、调试和测试。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值