USB协议介绍四 设备端_linux 5

先自我介绍一下,小编浙江大学毕业,去过华为、字节跳动等大厂,目前在阿里

深知大多数程序员,想要提升技能,往往是自己摸索成长,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年最新Linux运维全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友。
img
img
img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上运维知识点,真正体系化!

由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新

需要这份系统化的资料的朋友,可以点击这里获取!

https://github.com/RoganDawes/P4wnP1
Using RPi Zero as a Keyboard
内核驱动
在介绍应用之间,我们先看看内核的实现。还是以 Linux 内核为例,具体来说,我们想了解如何通过添加内核模块的方式实现一个新的自定义 USB 设备。俗话说得好,添加 Linux 驱动的最好方式是参看现有的驱动,毕竟当前内核中大部分都是驱动代码。

因为 Linux 内核既能运行在主机端,也能运行在设备端,因此设备端的 USB 驱动有个不同的名字: gadget driver。对于不同设备,也提供不同的内核接口,即 Host-Side API 和 Gadget API。既然我们是想实现自己的设备,就需要从 gadget 驱动入手。

g_zero.ko 就是这么一个驱动,代码在 drivers/usb/gadget/legacy/zero.c。该驱动实现了一个简单的 USB 设备,包含 2 个配置描述,各包含 1 个功能,分别是 sink 和 loopback,前者接收数据并返回 0,后者接收数据并原样返回:

drivers/usb/gadget/function/f_sourcesink.c
drivers/usb/gadget/function/f_loopback.c
代码量不多,感兴趣的自行 RTFSC。另外值得一提的是,对于运行于 USB device 端的系统而言,内核中至少有三个层级处理 USB 协议,可能用户层还有更多。gadget API 属于三层的中间层。至底向上,三层分别是:

USB Controller Driver: 这是软件的最底层,通过寄存器、FIFO、DMA、IRQ 等其他手段直接和硬件打交道,通常称为 UDC (USB Device Controller) Driver。
Gadget Driver: 作为承上启下的部分,通过调用抽象的 UDC 驱动接口,底层实现了硬件无关的 USB function。主要用于实现前面提到的 USB 功能,包括处理 setup packet (ep0)、返回各类描述符、处理各类修改配置情况、处理各类 USB 事件以及 IN/OUT 的传输等等。
Upper Level: 通过 Gadget Driver 抽象的接口,实现基于 USB 协议的上层应用,比如 USB 网卡、声卡、文件存储、HID 设备等。
关于 Linux USB 子系统的详细设计结构,可以参考源码中的文档: Linux USB API,以及其他一些资料,如下所示:

https://bootlin.com/doc/legacy/linux-usb/linux-usb.pdf
https://static.lwn.net/images/pdf/LDD3/ch13.pdf
https://elinux.org/images/5/5e/Opasiak.pdf

GadgetFS/ConfigFS
参考现有的 Linux 驱动,依葫芦画瓢可以很容易实现一个自定义的 USB Gadget。但是这样存在一些问题,如果我想实现一个八声道的麦克风,还要重新写一遍驱动、编译、安装,明明内核中麦克风的功能已经有了,复制粘贴就显得很不优雅。

那么,有没有什么办法可以方便组合和复用现有的 gadget function 呢?在 Linux 3.11 中,引入了 USB Gadget ConfigFS,提供了用户态的 API 来方便创建新的 USB 设备,并可以组合复用现有内核中的驱动。
在这里插入图片描述

前文提到的基于树莓派 Zero 实现的各类 USB 设备,大部分都是基于 Gadget ConfigFS 接口实现的。基于 configfs 创建 USB gadget 的步骤一般如下:

CONFIGFS_HOME=/sys/kernel/config/usb_gadget

#1. 新建一个 gadget,并写入实际的设备描述
mkdir $CONFIGFS_HOME/mydev # 创建设备目录后,该目录下自动创建并初始化了一个设备模板
cd $CONFIGFS_HOME/mydev
echo 0x0100 > bcdDevice # Version 1.0.0
echo 0x0200 > bcdUSB # USB 2.0
echo 0x00 > bDeviceClass
echo 0x00 > bDeviceProtocol
echo 0x40 > bMaxPacketSize0
echo 0x0104 > idProduct # Multifunction Composite Gadget
echo 0x1d6b > idVendor # Linux Foundation

#2. 新建一个配置,并写入实际的配置描述
mkdir configs/c.1 # 创建一个配置实例: .
cd configs/c.1
echo 0x01 > MaxPower
echo 0x80 > bmAttributes

#3. 新建一个接口(function),或者将已有接口链接到当前配置下
cd $CONFIGFS_HOME/mydev
mkdir functions/hid.usb0 # 创建一个 function 实例: .
echo 1 > functions/hid.usb0/protocol
echo 8 > functions/hid.usb0/report_length # 8-byte reports
echo 1 > functions/hid.usb0/subclass
ln -s functions/hid.usb0 configs/c.1

#4. 将当前 USB 设备绑定到 UDC 驱动中
echo ls /sys/class/udc > $CONFIGFS_HOME/mydev/UDC

这样就实现了一个最简单的 USB gadget,当然要完整实现的话还可以添加字符串描述,以及增加各个端点的功能。使用 configfs 实现一个 USB 键盘的示例可以参考网上其他文章,比如 Using RPi Zero as a Keyboard,或者 Github 上的开源项目,比如 P4wnP1。

有些人觉得 ConfigFS 配置起来很繁琐,所以开发了一些函数库(如 libusbgx) 来通过调用创建 gadget;有人觉得通过函数操作也还是繁琐,就创建了一些工具(如 gt) 来通过处理一个类似于 libconfig 的配置文件直接创建 gadget,不过笔者用得不多。

FunctionFS
FunctionFS 最初是对 GadgetFS 的重写,用于支持实现用户态的 gadget function,并组合到现有设备中。这里说的 FunctionFS 实际上是新版基于 ConfigFS 的 GadgetFS 拓展。在上一节中说到创建设备 gadget 的第四步就是给对应的 configuration 添加 function,格式为 function—type.instance-name,type 对应一个已有的内核驱动,比如上节中是 hid。

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化的资料的朋友,可以点击这里获取!

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

)**

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

  • 21
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值