libusb_init(NULL), 如果传入一个NULL, 则libusb 内部会有一个 usbi_default_context 变量在内部保存上下文. 这样以后调用 libusb 函数时可以不指定 context, libusb 使用默认的内部变量.

详见 USBI_GET_CONTEXT(ctx);


usbi_backend->init --> op_init

查找 usbfs 根目录

usbfs_path = find_usbfs_path();

  1. 先到 "/dev/bus/usb/" 目录下查找是否有任意有效文件(非'.'开头的隐藏文件), 如果有的话, 则 usbfs_path 为 "/dev/bus/usb"
  2. 如果没有的话, 就到 "/proc/bus/usb" 下寻找. 如果找到, 则 usbfs_path 为 "/proc/bus/usb"
  3. 如果还没有找到, 则查找 /dev/ 下的 /dev/usbdev%d.%d 格式的文件. 如果找到的话, 则 usbfs_path 为 "/dev" . 这种情况比较特殊, 会置标志 usbdev_names = 1;
  4. 如果还没找到的话, 就没办法了. 返回错误.

比较内核版本

libusb 里有个函数 kernel_version_ge(2,6,32), 使用 uname() 系统调用来获取当前运行内核版本信息.

sysfs 里的 USB 信息

遍历 "/sys/bus/usb/devices" , 找 usb 开头的目录. 这个目录表示总线. 查找其下是否有 busnum, devnum, descriptor, bConfigurationValue 等文件, 以判断其能力.

如果 /sys 下的信息完整, 则以后就可以直接从 /sys 下读到 USB 的信息. 否则就要到上面查找出的 usbfs 里读.

libusb 里的代码这样写, 看来整个 USB 的趋势还是倾向于到 /sys/ 下获取信息.

usbi_io_init(ctx)

pipe() 创建了两个管道 ctx->ctrl_pipe[2], 并把读管道加入到 poll 监听列表里.


cnt = libusb_get_device_list(NULL, &devs);

查找设备数量. 实现中是去查找 /sys/bus/devices/ 下的目录, 条件是: 非数字开头 && "usb" 字符开头 && 名字没有':' , 其下的 busnum 和 devnum 就是总线号和设备号

每个设备用 libusb_device 结构表示, 唯一标志符是 session_id, session_id = busnum << 8 | devaddr;

struct libusb_device {
/* lock protects refcnt, everything else is finalized at initialization
* time */
usbi_mutex_t lock;
int refcnt; struct libusb_context *ctx; uint8_t bus_number;
uint8_t device_address;
uint8_t num_configurations;
enum libusb_speed speed; struct list_head list;
unsigned long session_data;
unsigned char os_priv[0];
};

最后的 os_priv 会使用 initialize_device() 分配一片操作系统相关的空间给它, 以存储设备路径等信息, 在linux 下为:

struct linux_device_priv {
char *sysfs_dir;
unsigned char *dev_descriptor;
unsigned char *config_descriptor;
};

int libusb_get_device_descriptor(libusb_device *dev, struct libusb_device_descriptor *desc);

libusb_device *dev 中有所有该设备在 /sys 下的文件路径信息, 那么此函数就会读该路径下的 "descriptors" 文件, 该文件是 USB 设备描述符, 其中就存储了 vendor, producdt 等信息.


同步IO

控制传输, 块传输, 中断传输

int 	libusb_control_transfer (libusb_device_handle *dev_handle, uint8_t bmRequestType, uint8_t bRequest, uint16_t wValue, uint16_t wIndex, unsigned char *data, uint16_t wLength, unsigned int timeout)
Perform a USB control transfer. int libusb_bulk_transfer (struct libusb_device_handle *dev_handle, unsigned char endpoint, unsigned char *data, int length, int *transferred, unsigned int timeout)
Perform a USB bulk transfer. int libusb_interrupt_transfer (struct libusb_device_handle *dev_handle, unsigned char endpoint, unsigned char *data, int length, int *transferred, unsigned int timeout)
Perform a USB interrupt transfer.

异步IO

最新文章

  1. QTimer的用法
  2. Redis学习笔记(4) Redis事务、生存时间及排序
  3. CORS(跨域资源共享)简介
  4. Linux free命令详解(转)
  5. SQL设置SQLServer最大连接数查询语句
  6. C# mvc 验证码3
  7. 挣值管理不是搞数字游戏(3)——进阶指标:CV、SV、CPI、SPI、EAC
  8. 【转】 UIView如何管理它的子视图
  9. GitHub详解(转)
  10. Linux文件编辑时光标操作
  11. nodejs-- vuex中mapActions
  12. odoo订餐系统之菜单设计
  13. json包
  14. 怎样消除adobe flash player设置
  15. BugPhobia沟通篇章:Solr模式配置与数据导入调研
  16. 洛谷P4338 [ZJOI2018]历史(LCT,树形DP,树链剖分)
  17. python---django中orm的使用(4)字段,参数(on_delete重点)补充,一对多,一对一,多对多
  18. C#中 DateTime , DateTime2 ,DateTimeOffset 之间的小区别 (转载)
  19. java HTTP代码示例
  20. vim中project多标签和多窗口的使用

热门文章

  1. 使用pipework将Docker容器桥接到本地网络环境中
  2. NYOJ 747 蚂蚁的难题(三)
  3. nginx报错502
  4. HDU 1565 方格取数(1) ——插头DP
  5. PC端有什么好用的倍速播放的软件?
  6. [HDU-4825] Xor-Sum (01字典树)
  7. 【loj6191】「美团 CodeM 复赛」配对游戏
  8. 什么是JNI?
  9. 商务旅行(codevs 1036)
  10. Objective C语言中nil、Nil、NULL、NSNull的区别