LINUX中USB驱动架构—设备驱动

一、usb_driver结构体描述一个USB设备驱动

usb_driver

复制代码
struct usb_driver {
   const char *name;
 
  int (*probe) (struct usb_interface *intf,
  const struct usb_device_id *id);
 
  void (*disconnect) (struct usb_interface *intf);
 
  int (*unlocked_ioctl) (struct usb_interface *intf, unsigned int code,10 void *buf);

  int (*suspend) (struct usb_interface *intf, pm_message_t message);
  int (*resume) (struct usb_interface *intf);
  int (*reset_resume)(struct usb_interface *intf);

  int (*pre_reset)(struct usb_interface *intf);
  int (*post_reset)(struct usb_interface *intf);
 
  const struct usb_device_id *id_table;
 
  struct usb_dynids dynids;
  struct usbdrv_wrap drvwrap;
  unsigned int no_dynamic_id:1;
  unsigned int supports_autosuspend:1;
  unsigned int disable_hub_initiated_lpm:1;
  unsigned int soft_unbind:1;
};

二、编写USB设备驱动

在编写新的USB设备驱动时,主要应该完成的工作是probe()和disconnect()函数,即探测和断开函数,它们分别在设备被插入和拔出的时候调用,用于初始化和释放软硬件资源。对usb_driver的注册和 注销可通过下面两个函数完成:

复制代码
int usb_register(struct usb_driver *new_driver);
void usb_deregister(struct usb_driver *driver);

usb_driver结构体中的id_table成员描述了这个USB驱动所支持的USB设备列表,它指向一个

usb_device_id数组,usb_device_id结构体包含有USB设备的制造商ID、产品ID、产品版本、设备类、接口类等信息及其要匹配标志成员match_flags(标明要与哪些成员匹配,包含DEV_LO、DEV_HI、DEV_CLASS、DEV_SUBCLASS、DEV_PROTOCOL、INT_CLASS、INT_SUBCLASS、INT_PROTOCOL)。

可以借助下面一组宏来生成usb_device_id结构体的实例:

复制代码
USB_DEVICE(vendor, product);

该宏根据制造商ID和产品ID生成一个usb_device_id结构体的实例,在数组中增加该元素将意味着该驱 动可支持与制造商ID、产品ID匹配的设备。

复制代码
USB_DEVICE_VER(vendor, product, lo, hi);

该宏根据制造商ID、产品ID、产品版本的最小值和最大值生成一个usb_device_id结构体的实例,在数 组中增加该元素将意味着该驱动可支持与制造商ID、产品ID匹配和lo~hi范围内版本的设备。

复制代码
USB_DEVICE_INFO(class, subclass, protocol);

该宏用于创建一个匹配设备指定类型的usb_device_id结构体实例。

复制代码
USB_INTERFACE_INFO(class, subclass, protocol);

该宏用于创建一个匹配接口指定类型的usb_device_id结构体实例。


usb_device_id结构体数组实例

两个用于描述某USB驱动支持的USB设备的usb_device_id结构体数组实例

复制代码
/* 本驱动支持的USB设备列表 */
 
 /* 实例1 */
static struct usb_device_id id_table [] = {
  { USB_DEVICE(VENDOR_ID, PRODUCT_ID) },
  { },
};
 MODULE_DEVICE_TABLE (usb, id_table);
 
/* 实例2 */
static struct usb_device_id id_table [] = {
 { .idVendor = 0x10D2, .match_flags = USB_DEVICE_ID_MATCH_VENDOR, },13 { },
};
MODULE_DEVICE_TABLE (usb, id_table);

当USB核心检测到某个设备的属性和某个驱动程序的usb_device_id结构体所携带的信息一致时,这个 驱动程序的probe()函数就被执行(如果这个USB驱动是个模块的话,相关的.ko还应被Linux自动加载)。拔掉设备或者卸掉驱动模块后,USB核心就执行disconnect()函数来响应这个动作。

上述usb_driver结构体中的函数是USB设备驱动中与USB相关的部分,而USB只是一个总线,USB设备 驱动真正的主体工作仍然是USB设备本身所属类型的驱动,如字符设备、tty设备、块设备、输入设备等。 因此USB设备驱动包含其作为总线上挂接设备的驱动和本身所属设备类型的驱动两部分。

与platform_driver、i2c_driver类似,usb_driver起到了"牵线"的作用,即在probe()里注册相应的字符、tty等设备,在disconnect()注销相应的字符、tty等设备,而原先对设备的注册和注销一般直接发生在模块加载和卸载函数中。

相关推荐
ZHANG13HAO13 小时前
RV1106 通过 4G 网络基于 libdatachannel 实现 WebRTC 实时视频传输”
linux
..过云雨14 小时前
13.【Linux系统编程】从ELF格式深入理解动静态库
linux·c语言·c++·后端
一名机电研究生14 小时前
华为、阿里巴巴、字节跳动 100+ Linux面试问题总结(一)
linux·华为·面试
讨厌下雨的天空14 小时前
环境变量与地址
linux
阿巴~阿巴~14 小时前
深入解析UDP服务器核心开发机制
linux·服务器·网络协议·网络编程·udp服务器·recvfrom函数
wa的一声哭了15 小时前
Linux服务器配置ssh免密登陆多台服务器、服务器别名配置
linux·运维·服务器·网络·arm开发·python·ssh
beijingliushao15 小时前
93-MongoDB-Linux
linux·数据库·mongodb
YongCheng_Liang15 小时前
openEuler 22.03 LTS 部署 ELK(Elasticsearch+Logstash+Kibana)完整教程
linux·运维·elk·elasticsearch
go_bai16 小时前
Linux-线程
linux·开发语言·c++·经验分享·笔记
清浅儿16 小时前
Linux权限知识点
linux·运维·服务器