【北京迅为】iTOP-4412全能版使用手册-第六十七章 USB鼠标驱动详解
iTOP-4412全能版采用四核Cortex-A9,主频为1.4GHz-1.6GHz,配备S5M8767 电源管理,集成USB HUB,选用高品质板对板连接器稳定可靠,大厂生产,做工精良。接口一应俱全,开发更简单,搭载全网通4G、支持WIFI、蓝牙、陀螺仪、CAN总线、RS485总线、500万摄像头等模块,稳定运行Android 4.0.3/Android 4.4操作,系统通用Linux-3.0.15+Qt操作系统(QT支持5.7版本),Ubuntu版本:12.04,接口智能分配 方便好用。
第六十七章 USB鼠标驱动详解
67.1 鼠标驱动注册
在 Linux 的 USB 驱动中,鼠标驱动是最简单,最适合用来学习的。USB 鼠标驱动是内核源码目录“drivers/hid/usbhid/”下的“usbmouse.c”文件,这个驱动在任意版本的内核中都有。
我们将“usbmouse.c”文件进行分解,提取其中一部分来分析,本文档主要分析 USB 鼠标驱动的注册部分。
USB 鼠标驱动的配置
我们要分析的 USB 鼠标驱动源码是“usbmouse.c”,这里要注意的是要支持 USB 鼠标驱动有多种配置方法。
这里给大家介绍一个概念“人机接口”HMI,人机接口在 linux 中包含很多设备,例如: 鼠标、键盘、显示屏...,凡是用来直接和用户直接通信的设备都可以叫做人机接口。从原始的DOS 控制台,到手机触屏,未来的人脑控制等等也都是人机接口。
在 linux 中,默认使用的是 HID,也就是人机接口设备,内核中默认配置的是 HID 驱动,并没有配置 USB 鼠标驱动。配置部分涉及到 Makefile、menuconfig 和 Kconfig,我们一起来研究一下。
在内核配置编译之后,如下图所示,使用命令“ls drivers/hid/usbhid/ ”,可以看到“usbmouse.c”并没有被编译,可以我们的内核明显是可以支持 USB 鼠标的。
接着使用命令“vim drivers/hid/usbhid/Makefile”,打开 Makefile 文件,可以看到编译条件是宏“CONFIG_USB_MOUSE”。
然后使用“vim drivers/hid/usbhid/Kconfig”命令,可以看到“config USB_MOUSE”Kconfig 的配置,如下图所示。
接着我们使用命令“make menuconfig”,然后在其中搜索“USB_MOUSE”宏,如下图所示。
如下图所示,我们进入到“HID Devices”配置界面之后,却没有发现有“USB HID Boot Protocol drivers”这个菜单。
那么到底是什么原因呢?我们回到“drivers/hid/usbhid/Kconfig”文件,如下图所示, 可以看到“USB_MOUSE”是在菜单“USB HID Boot Protocol drivers”下。
如上图所示,注意以下的提示,这部分意思是,菜单“USB HID Boot Protocol drivers”需要依赖“USB!=n && USB_HID!=y && EXPERT”,也就是必须定义“USB”, 没有定义“USB_HID”,这部分菜单才会显示出来。请注意这种配置在内核中经常会遇到。
menu "USB HID Boot Protocol drivers"
depends on USB!=n && USB_HID!=y && EXPERT
我们到 menuconfig 中将 USB_HID 取消配置,然后就可以看到 "USB HID Boot Protocol drivers"配置菜单了,如下图所示。
如上图所示,接着进入"USB HID Boot Protocol drivers",如下图所示,USB 鼠标和键盘都是没有配置的。
前面介绍这么多,主要是给大家介绍更多的缺省文件配置的知识。
为了进行后面的实验,只需要去掉“USB Human Interface Device (full HID) support ”就可以了,如下图所示,其它地方都不需要动,这样是为了我们动态加载 USB 鼠标驱动,一步一步的实现 USB 鼠标驱动。
取消配置“USB Human Interface Device (full HID) support”,然后退出保存,重新编译内核,烧写到开发板中,准备后续的实验。
USB 鼠标驱动的注册
做好准备工作之后,我们来查看一下 USB 鼠标驱动的注册部分。
USB 设备驱动的驱动注册的函数为 usb_register 和 usb_deregister,如下图所示。
这部分内容就不再赘述,它和前面介绍的字符驱动以及 I2C、SPI 驱动等等类似,由内核提供专门的注册和卸载函数。
我们需要注意的是,USB 驱动是可以热拔插的,再插入和拔掉设备的时候,肯定都需要对应的代码,如下图所示,可以看到 struct usb_driver usb_mouse_driver 中有定义usb_mouse_probe 和 usb_mouse_disconnect。
插入 USB 设备,进行初始化则进入“usb_mouse_probe”;拔掉 USB 卸载则进入“usb_mouse_disconnect”;其中的参数 name,则是驱动名称“usbmouse”,既然有驱动名称,那一定有设备名称,请注意前面介绍过的 USB 描述符,USB 描述符的具体内容是在USB 设备中的,相当于设备注册是在实体的“USB 设备”中!另外 usb_mouse_id_table, 这部分用来匹配,在初始化 probe 的代码中我们再进行打印测试,这里我们先使用默认的配置即可。
我们将代码进行简化,只进行 USB 设备的初始化和卸载部分,代码如下所示。
#include <linux/kernel.h>
#include <linux/slab.h>
#include <linux/module.h>
#include <linux/init.h>
#include <linux/usb/input.h>
#include <linux/hid.h>MODULE_AUTHOR(DRIVER_AUTHOR);
MODULE_DESCRIPTION(DRIVER_DESC);
MODULE_LICENSE(DRIVER_LICENSE);
static int usb_mouse_probe(struct usb_interface *intf, const struct usb_device_id *id)
{
printk("usb mouse probe!\n");
return 0;
}static void usb_mouse_disconnect(struct usb_interface *intf)
{
printk("usb mouse disconnect!\n");
}static struct usb_device_id usb_mouse_id_table [] = {
{
USB_INTERFACE_INFO(USB_INTERFACE_CLASS_HID,
USB_INTERFACE_SUBCLASS_BOOT,
USB_INTERFACE_PROTOCOL_MOUSE) },
{ } /* Terminating entry */
};MODULE_DEVICE_TABLE (usb, usb_mouse_id_table);static struct usb_driver my_usb_mouse_driver = {
.name = "usbmouse",
.probe = usb_mouse_probe,
.disconnect = usb_mouse_disconnect,
.id_table = usb_mouse_id_table,
};static int init my_usb_mouse_init(void)
{
//注册 usb 驱动
return usb_register(&my_usb_mouse_driver);
}static void exit my_usb_mouse_exit(void)
{
//卸载 USB 驱动
usb_deregister(&my_usb_mouse_driver);
}module_init(my_usb_mouse_init);
module_exit(my_usb_mouse_exit);
上面这段代码要实现的功能很简单,就是加载驱动之后,如果有 USB 鼠标插入,则会打印“usb mouse probe!”,如果有 USB 鼠标拔出,则会打印“usb mouse disconnect!”。
如下图所示,我们加载该驱动,串口控制台提示如下。
如上图所示,这部分是 usbcore 打印的,和我们的驱动中的代码没有直接的关系,在加载 usb 鼠标驱动之后,打印“注册了新的驱动,叫 usbmouse”。
接着我们接上鼠标,看到打印信息如下。
如上图所示,红色框中的就是驱动中的打印信息“usb mouse probe!”,其它部分全部是由 USB 主控制器完成。这说明匹配成功进入了 usb 鼠标的 probe。
接着我们拔掉 USB 鼠标,如下图所示,打印了“usb mouse disconnect!”这部分信息,是和我们 disconnect 函数中的打印对应。
如上图所示,其它部分打印信息也是由主控制器来完成。
主控制器会统一管理每一个 usb hub 以及每一个 usb 接口,大部分工作其实都是有主控制器完成的!
至此,本文档要介绍的内容完成,后面的文档我们继续分析进入 probe 之后,我们需要进行的配置等工作。
67.2 USB鼠标设备信息
在前面的文档中,我们知道 USB 的设备注册信息是在“真实的设备”中保存,在 USB 设备被检测到之后,主控制器将设备信息读取到驱动中。本篇的内容比较简单,主要是验证这部分内容。
在内核中没有 USB 鼠标驱动的时候插入 USB 鼠标,可以看到一些打印信息,这些信息是主控制器部分来完成的,我们在 probe 中打印 USB 设备的一些信息,来做个验证。
请注意,“USB 的设备信息”的标准术语严格来说,应该叫 USB 描述符,我这里把它称为 USB 的设备信息,主要是为了和前面驱动中的设备注册对应起来,便于大家理解。因为我们在前面所有的设备驱动中,都有设备注册这部分,在 USB 驱动中,可以将主控制器获取描述符的过程类比为“设备注册”。
在不加载 USB 鼠标驱动的情况下,插上 USB 鼠标,也是可以看到打印信息的,如下图所示。可以看到 idVendor, idProduct, bcdDevice 等信息,我们后面就在 probe 中添加这几个参数的打印信息,对比验证下。
在代码中,我们添加如下函数。
static void check_usb_device_descriptor(struct usb_device *dev)
{
printk("dev->descriptor.idVendor is %4x!\n\
dev->descriptor.idProduct is %4x!\n\
dev->descriptor.bcdDevice is %4x!\n\
dev->descriptor.iSerialNumber is %2x!\n",\
dev->descriptor.idVendor,dev->descriptor.idProduct,dev->descriptor.bcdDevice,dev->descriptor.iSerialNumber);
}
然后在 probe 中调用,如下图所示。
struct usb_device *dev = interface_to_usbdev(intf);
check_usb_device_descriptor(dev);
完成代码请参考打包的程序,加载驱动之后,插入 USB 鼠标,如下图所示。
如上图所示,可以看到在 probe 中获取的数据,和主控制器中打印的信息是一模一样的。
关于主控制器获取描述符信息,可以参考前面的“iTOP-4412-驱动-usb 文档 05-usb 枚举流程”这个文档,它经过了一个复杂的通信过程,将信息读取到内核中,然后在初始化的时候,会将其传递到 probe 函数中。
67.3 鼠标URB请求块的使用
本章主要介绍在 probe 中,urb 的初始化。USB 中所有的对底层的通信,都是围绕urb 来做的,所有的工作都是一个套路。先带大家分析下内核自带 USB 鼠标驱动部分代码,然后再改写为简单模式,对urb 进行监测。
先结合前面介绍理论部分,对内核中自带的 USB 鼠标驱动代码进行分析,后面我们的例程会进行一些简化,让大家对其更容易理解。
初始化和 usb_interface 分析
下面是进入 probe 之后的一段匹配的代码,用于验证接入的设备是否和驱动匹配。
interface = intf->cur_altsetting
if (interface->desc.bNumEndpoints != 1)
return -ENODEV;endpoint = &interface->endpoint[0].desc;
if (!usb_endpoint_is_int_in(endpoint))
return -ENODEV;
在请求块理论部分,我们了解到鼠标驱动是“一个中断 IN 端点给带有特定端点号的特定USB 设备”。在 USB 设备描述符的文档中,我们了解到“设备描述符→配置描述符→接口描述符→端点描述符”这几个概念,用户可以直接看下“usb_interface”这个结构体,里面内嵌了接口描述符信息和端点描述符等。
在上面的代码“interface = intf->cur_altsetting;”中,获取鼠标硬件传输过来的描述符信息。
在“if (interface->desc.bNumEndpoints != 1) return -ENODEV;”中,判断端点是否只有一个,鼠标设备只有一个端点,如果不是,则报错。
关于static int usb_mouse_probe(struct usb_interface *intf, const struct usb_device_id *id)中的变量“struct usb_interface *intf”,看到这里很多人可能不太理解。前面我们介绍描述符的时候,并没有 usb_interface 这个结构体,实际上是这样的,它本质上是个“配置描述符”,它是从从机(鼠标)中传递过来的信息,配置描述符的组织方式可能有点不一样。可以这样类比,当我们要到大学报到的时候,我们要填写学籍档案,假设学籍档案中信息都在户口簿上,例如:姓名、籍贯以及身份证号等等,这些信息从户口簿传递到学籍档案中,信息其实是一样的,只是在户口簿和学籍档案中组织形式不一样,名称不一样,实际包含的信息是一样的。我们也可以看下“usb_interface”这个结构体,部分代码如下所示。
struct usb_interface {
/* array of alternate settings for this interface,
* stored in no particular order */
struct usb_host_interface *altsetting;
接着看下“usb_host_interface”结构体,如下所示,可以看到结构体中包含了接口描述符和端点描述符。
struct usb_host_interface {
struct usb_interface_descriptordesc;
/* array of desc.bNumEndpoint endpoints associated with this
* interface setting. these will be in no particular order.
*/
struct usb_host_endpoint *endpoint;
在“endpoint = &interface->endpoint[0].desc;”代码中,我们的驱动获取了端点描述符信息。
在我们的例程中,可以简化为“interface = intf->cur_altsetting;”和“endpoint = &interface->endpoint[0].desc;”,不做判断。
管道和管道数据包
代码如下所示。
pipe = usb_rcvintpipe(dev, endpoint->bEndpointAddress);
maxp = usb_maxpacket(dev, pipe, usb_pipeout(pipe));
前面介绍过,要和 USB 设备通信,唯一的方式是通过管道,每个驱动和 USB 设备通信, 都必须创建管道,使用管道进行通信,而且管道必须使用内核统一的函数来申请!
初始化USB设备结构体和输入子系统设备结构体
如下所示,代码有详细的注释。
//为 mouse 分配内存,申请输入子系统mouse = kzalloc(sizeof(struct usb_mouse), GFP_KERNEL);
input_dev = input_allocate_device();
if (!mouse || !input_dev)
goto fail1;//申请内存空间用于数据传输,data 为指向该空间的地址,data_dma 则是这块内存空间的 dma 映射,即这块内存空间对应的 dma 地址。在使用 dma 传输的情况下,则使用 data_dma 指向的 dma 区域,否则使用 data 指向的普通内存区域进行传输。 GFP_ATOMIC 表示不等待,GFP_KERNEL 是普通的优先级,可以睡眠等待,由于鼠标使用中断传输方式,不允许睡眠状态,data 又是周期性获取鼠标事件的存储区,因此使用 GFP_ATOMIC 优先级,如果不能 分配到内存则立即返回 0
mouse->data = usb_alloc_coherent(dev, 8, GFP_ATOMIC, &mouse->data_dma);
if (!mouse->data)
goto fail1;//为 urb 结构体申请内存空间,第一个参数表示等时传输时需要传送包的数量,其它传输方式则为 0。申请的内存将通过下面即将见到的 usb_fill_int_urb 函数进行填充。
mouse->irq = usb_alloc_urb(0, GFP_KERNEL);
if (!mouse->irq)
goto fail2;//填充 usb 设备结构体和输入设备结构体
mouse->usbdev = dev;
mouse->dev = input_dev;//填充鼠标设备的名称if (dev->manufacturer)
strlcpy(mouse->name, dev->manufacturer, sizeof(mouse->name));
if (dev->product) {
if (dev->manufacturer)
strlcat(mouse->name, " ", sizeof(mouse->name));
strlcat(mouse->name, dev->product, sizeof(mouse->name));
}if (!strlen(mouse->name))
snprintf(mouse->name, sizeof(mouse->name),
"USB HIDBP Mouse %04x:%04x",
le16_to_cpu(dev->descriptor.idVendor),
le16_to_cpu(dev->descriptor.idProduct));//填充鼠标设备结构体中的节点名。usb_make_path 用来获取 USB 设备在 Sysfs 中的路径,格式为:usb-usb
总线号-路径名
usb_make_path(dev, mouse->phys, sizeof(mouse->phys));
strlcat(mouse->phys, "/input0", sizeof(mouse->phys));//将鼠标设备的名称赋给鼠标设备内嵌的输入子系统结构体
input_dev->name = mouse->name;
input_dev->phys = mouse->phys;//input_dev 中的 input_id 结构体,用来存储厂商、设备类型和设备的编号,这个函数是将设备描述符中的编号赋给内嵌的输入子系统结构体
usb_to_input_id(dev, &input_dev->id);
input_dev->dev.parent = &intf->dev;//input_dev 进行初始化。EV_KEY 是按键事件,EV_REL 是相对坐标事件;keybit 表示键值,包括左键、右键和中键;relbit 用于表示相对坐标值;有的鼠标还有其它按键;中键滚轮的滚动值。
input_dev->evbit[0] = BIT_MASK(EV_KEY) | BIT_MASK(EV_REL);
input_dev->keybit[BIT_WORD(BTN_MOUSE)] = BIT_MASK(BTN_LEFT) |
BIT_MASK(BTN_RIGHT) | BIT_MASK(BTN_MIDDLE);
input_dev->relbit[0] = BIT_MASK(REL_X) | BIT_MASK(REL_Y);
input_dev->keybit[BIT_WORD(BTN_MOUSE)] |= BIT_MASK(BTN_SIDE) |
BIT_MASK(BTN_EXTRA);
input_dev->relbit[0] |= BIT_MASK(REL_WHEEL);//设置当前输入设备的种类
input_set_drvdata(input_dev, mouse);//为输入子系统设备添加打开,关闭等
input_dev->open = usb_mouse_open;
input_dev->close = usb_mouse_close;
设置 urb 等
如下所示,可以看注释来理解。
/*填充构建 urb,将刚才填充好的 mouse 结构体的数据填充进 urb 结构体中,在 open 中递交 urb。当 urb 包含一个即将传输的 DMA 缓冲区时应该设置 URB_NO_TRANSFER_DMA_MAP。USB 核心使用transfer_dma 变量所指向的缓冲区,而不是 transfer_buffer 变量所指向的。URB_NO_SETUP_DMA_MAP 用于 Setup 包,URB_NO_TRANSFER_DMA_MAP 用于所有 Data 包。*/
usb_fill_int_urb(mouse->irq, dev, pipe, mouse->data,
(maxp > 8 ? 8 : maxp),usb_mouse_irq, mouse, endpoint->bInterval);
mouse->irq->transfer_dma = mouse->data_dma;
mouse->irq->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;//注册输入子系统
error = input_register_device(mouse->dev);
if (error)
goto fail3;//一般在 probe 函数中,都需要将设备相关信息保存在一个 usb_interface 结构体中,以便以后通过usb_get_intfdata 获取使用。这里鼠标设备结构体信息将保存在 intf 接口结构体内嵌的设备结构体中
的 driver_data 数据成员中,即 intf->dev->dirver_data = mouse。
usb_set_intfdata(intf, mouse);
return 0;
usb_mouse_disconnect 操作
代码如下,拔出之后会进行以下操作。现获取鼠标设备结构体,然后清空;kill 掉 urb, 删掉输入子系统;删掉 urb;释放内存等等。
static void usb_mouse_disconnect(struct usb_interface *intf)
{
struct usb_mouse *mouse = usb_get_intfdata (intf);
usb_set_intfdata(intf, NULL);
if (mouse) {
usb_kill_urb(mouse->irq);
input_unregister_device(mouse->dev);
usb_free_urb(mouse->irq);
usb_free_coherent(interface_to_usbdev(intf), 8, mouse->data, mouse->data_dma);
kfree(mouse);
}
}
请求块 URB 的使用
在通信原理中,有信源、信道和信宿的概念。虽然这个概念更多的是指代硬件方面,不过我觉得放在这里来理解 USB 的几个概念,非常好。
类比生活中的例子,小 A 对小 B 说了一句“您好”,小 B 听到了。小 A 就是信源,空气就是信道,小 B 就是信宿。当然,也有数据源、传输通道和数据终端的说法,实质是差不多的。
那么我给大家说这几个概念有什么用呢?不知道大家还记不记的前面的文档中介绍过的几个概念。主机只能和 USB 设备的“端点”通信;通信是通过“管道”实现;USB 通信是通过请求块实现。最后数据终端就是我们的驱动程序了,我们要在驱动程序中获得 USB 传输来的数据,这个实验就成功了一多半。
接着,我们来看一下将要在 probe 中添加的代码,其中去掉了输入子系统的代码,简化了 urb 部分。
static int usb_mouse_probe(struct usb_interface *intf, const struct usb_device_id *id)
{struct usb_device *dev = interface_to_usbdev(intf);
struct usb_endpoint_descriptor *endpoint;//鉴于我们在中断处理函数中,需要监听是否有数据传输,这几个变量设置为 static 变量
/*int pipe, maxp; signed char *data;
dma_addr_t data_dma;
struct urb *mouse_urb;*/printk("usb mouse probe!\n");//check_usb_device_descriptor(dev);//在 usb 的数据传输中,"数据源"是端点,“传输通道"是管道,"数据终端"是内核“USB 驱动”中的buffer//数据源,数据通道,数据终端全部是在 urb 中设置endpoint = &intf->cur_altsetting->endpoint[0].desc; //数据源pipe = usb_rcvintpipe(dev, endpoint->bEndpointAddress); //传输通道
maxp = usb_maxpacket(dev, pipe, usb_pipeout(pipe));data = usb_alloc_coherent(dev, 8, GFP_ATOMIC, &data_dma);//数据终端//urb 必须使用内核函数申请,以便内核同一管理mouse_urb = usb_alloc_urb(0, GFP_KERNEL);//通过 urb 设置数据源,传输通道,数据终端
usb_fill_int_urb(mouse_urb, dev, pipe, data,(maxp > 8 ? 8 : maxp),check_usb_data, NULL, endpoint-
>bInterval);mouse_urb->transfer_dma = data_dma;
mouse_urb->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;//提交 urb
usb_submit_urb(mouse_urb, GFP_KERNEL);
return 0;
}
然后看一下,中断处理函数。
//监测是否有数据传入
static void check_usb_data(struct urb *urb)
{
int i = 0;
printk("count is %d time!\n",++count);
//重新提交
urb usb_submit_urb(mouse_urb, GFP_KERNEL);
}
完整的代码,请参考代码文件。
驱动编译加载之后,插上 USB 鼠标,如下图所示。
接着,尝试按鼠标左键,右键,滚轮,滑动操作等等,如下图所示。USB 鼠标进行操作之后,驱动进入了中断。
那么传输过来的数据在哪里呢?其实就在 data 这个数据指针中,在下一篇文档中,我们增加输入子系统部分,在应用层打印数据。
67.4 鼠标驱动与Linux输入子系统的接口
本章在驱动中添加输入子系统相关的部分代码,在鼠标左键,右键等操作之后,能够打印对应的信息。
输入子系统初始化
在前面,已经学习过输入子系统,这里就不再重复,代码中有详细的注释,如下所示。
static int usb_mouse_probe(struct usb_interface *intf, const struct usb_device_id *id)
{
struct usb_device *dev = interface_to_usbdev(intf);
struct usb_endpoint_descriptor *endpoint;
int ret;//鉴于我们在中断处理函数中,需要监听是否有数据传输,这几个变量设置为 static 变量
#if 0
int pipe, maxp;
signed char *data;
dma_addr_t data_dma;
struct urb *mouse_urb;*/
#endifprintk("usb mouse probe!\n");//check_usb_device_descriptor(dev);
//申请 input_devinput_dev = input_allocate_device();//设置 能产生哪类事件和具体的哪些时间
set_bit(EV_KEY, input_dev->evbit);
set_bit(EV_REP, input_dev->evbit);
set_bit(KEY_A, input_dev->keybit);
set_bit(KEY_B, input_dev->keybit);//注册输入子系统
ret = input_register_device(input_dev);
//在 usb 的数据传输中,"数据源"是端点,“传输通道"是管道,"数据终端"是内核“USB 驱动”中的buffer
//数据源,数据通道,数据终端全部是在 urb 中设置endpoint = &intf->cur_altsetting->endpoint[0].desc; //数据源
pipe = usb_rcvintpipe(dev, endpoint->bEndpointAddress); //传输通道
maxp = usb_maxpacket(dev, pipe, usb_pipeout(pipe));
data = usb_alloc_coherent(dev, 8, GFP_ATOMIC, &data_dma);//数据终端//urb 必须使用内核函数申请,以便内核同一管理
mouse_urb = usb_alloc_urb(0, GFP_KERNEL);//通过 urb 设置数据源,传输通道,数据终端
usb_fill_int_urb(mouse_urb, dev, pipe, data,(maxp > 8 ? 8 : maxp),check_usb_data, NULL, endpoint->bInterval);
mouse_urb->transfer_dma = data_dma;
mouse_urb->transfer_flags |= URB_NO_TRANSFER_DMA_MAP;
//提交 urb
usb_submit_urb(mouse_urb, GFP_KERNEL);
return 0;
}
在 probe 函数中,申请输入子系统,设置能产生哪类事件和具体的哪些事件,注册输入子系统。我们设置了键盘 key 类事件,能够产生 KEY_A 和 KEY_B 事件。
中断处理函数
在前一节式样中,我们已经成功实现了,鼠标操作进入中断处理函数。这里我们将采集数据和输入子系统对应,具体代码如下。
static void check_usb_data(struct urb *urb)
{
unsigned char val = 0;//printk("count is %d time!\n",++count);
printk("check_usb_data!\n");//USB 鼠标数据含义 data[0]: bit0 左键, 1 按下, 0 松开;bit1 右键,1 按下,0 松开
//按下之后,val 获取按键值
val = data[0];
if(val == 0x01){
//左键
input_report_key(input_dev, KEY_A, 1);
//通知上层,本次事件结束; KEY_A=30
input_sync(input_dev);input_report_key(input_dev, KEY_A, 0);
//通知上层,本次事件结束
input_sync(input_dev);
}if(val == 0x02){
//右键
input_report_key(input_dev, KEY_B, 1);//通知上层,本次事件结束;KEY_B=48
input_sync(input_dev);input_report_key(input_dev, KEY_B, 0);
//通知上层,本次事件结束
input_sync(input_dev);
}
val = 0;
// 重 新 提 交
urb usb_submit_urb(mouse_urb, GFP_KERNEL);
}
鼠标左键会打印 A,鼠标右键打印 B,然后其它鼠标操作会打印“check_usb_data!”。这部分要和后面的实验结果对照。
usb_mouse_disconnect
断开 USB 鼠标之后,需要添加释放输入子系统的代码,如下所示。
static void usb_mouse_disconnect(struct usb_interface *intf)
{
printk("usb mouse disconnect!\n");usb_kill_urb(mouse_urb);
usb_free_urb(mouse_urb);
input_unregister_device(input_dev);
input_free_device(input_dev);
usb_free_coherent(interface_to_usbdev(intf), 8, data,data_dma);
}
67.5 测试
驱动编译之后,加载驱动,接上鼠标,如下图所示。
操作鼠标,会打印如下图所示的信息。
接着在控制台使用命令“hexdump /dev/input/event”,鼠标左键和右键打印如下图所示信息。
如上图所示,打印了 0x1e 和 0x30,转化为十进制为 30 和 48,如下图所示,KEY_A 和KEY_B 的数值为 30 和 48,和驱动代码中的设置一一对应。
相关文章:

【北京迅为】iTOP-4412全能版使用手册-第六十七章 USB鼠标驱动详解
iTOP-4412全能版采用四核Cortex-A9,主频为1.4GHz-1.6GHz,配备S5M8767 电源管理,集成USB HUB,选用高品质板对板连接器稳定可靠,大厂生产,做工精良。接口一应俱全,开发更简单,搭载全网通4G、支持WIFI、蓝牙、…...

【青牛科技】拥有两个独立的、高增益、内部相位补偿的双运算放大器,可适用于单电源或双电源工作——D4558
概述: D4558内部包括有两个独立的、高增益、内部相位补偿的双运算放大器,可适用于单电源或双电源工作。该电路具有电压增益高、噪声低等特点。主要应用于音频信号放大,有源滤波器等场合。 D4558采用DIP8、SOP8的封装形式 主要特点ÿ…...

Kafka 数据写入问题
目录标题 分析思路1. **生产者配置问题**:Kafka生产者的配置参数生产者和消费者的处理确定并优化 2. **网络问题**:3. **Kafka 集群配置问题**:unclean.leader.election.enable 4. **Zookeeper 配置问题**:5. **JVM 参数调优**&am…...

实战ansible-playbook(九)-profile配置- 确保 CUDA 和 MPI 环境变量正确设置并立即生效
Playbook 分析 --- - name: 确保 CUDA 和 MPI 环境变量正确设置并立即生效hosts: pod2 # 指定目标主机组或具体主机名become: yes # 使用特权提升(sudo),以root权限执行某些需要权限的任务remote_user: canopy # 远程连接使用的用户名vars: # 定义全局变量,用于Playbo…...

气膜馆:科技与环保融合的未来建筑新选择—轻空间
在全球城市化进程不断加快的背景下,传统建筑方式面临着越来越多的挑战。如何在有限的土地和资源条件下,快速、高效、环保地搭建符合多功能需求的建筑,成为现代建筑行业亟待解决的重要课题。而随着科技的进步与建筑材料的创新,一种…...

git回退到某个版本git checkout和git reset命令的区别
文章目录 1. git checkout <commit>2. git reset --hard <commit>两者的区别总结推荐使用场景* 在使用 Git 回退到某个版本时, git checkout <commit> 和 git reset --hard <commit> 是两种常见的方式,但它们的用途和影响有很…...

Preprocess
Preprocess数据预处理 文本 使用Tokenizer将文本转换为标记序列,创建标记的数值表示,并将它们组装成张量。 预处理文本数据的主要工具是标记器。标记器根据一组规则将文本拆分为标记。标记被转换为数字,然后转换为张量,这些张量…...

stm32 spi接口传输asm330l速率优化(及cpu和dma方式对比)
最近一段时间做了一个mems的项目,项目的方案是stm32g071做主控,读写3颗asm330l的硬件形态。最初是想放置4颗imu芯片,因为pcb空间布局的问题,改放了3颗。但对于软件方案来说无所谓,关键是如何优化spi的传输速率…...

数字时代的文化宝库:存储技术与精神生活
文章目录 1. 文学经典的数字传承2. 音乐的无限可能3. 影视艺术的数字化存储4. 结语 数字时代的文化宝库:存储技术与精神生活 在数字化的浪潮中,存储技术如同一座桥梁,连接着过去与未来,承载着人类文明的瑰宝。随着存储容量的不断增…...

flex: 1 display:flex 导致的宽度失效问题
flex: 1 & display:flex 导致的宽度失效问题 问题复现 有这样的一个业务场景,详情项每行三项分别占33%宽度,每项有label字数不固定所以宽度不固定,还有content 占满标签剩余宽度,文字过多显示省略号, 鼠标划入展示…...

Hive 窗口函数与分析函数深度解析:开启大数据分析的新维度
Hive 窗口函数与分析函数深度解析:开启大数据分析的新维度 在当今大数据蓬勃发展的时代,Hive 作为一款强大的数据仓库工具,其窗口函数和分析函数犹如一把把精巧的手术刀,助力数据分析师们精准地剖析海量数据,挖掘出深…...

前端工程 Node 版本如何选择
1. Node 与 Npm 版本对应 这是一个必知必会的问题,尤其是对于维护那些老掉牙、一坨坨、非常大的有着长期历史的老破大工程。 1.1. package-lock.json 版本 首先你要会看项目的 package-lock.json 文件中的 lockfileVersion 版本号,这对于 NPM 安装来说…...

推荐在线Sql运行
SQL Fiddle 1、网址:SQL Fiddle - Online SQL Compiler for learning & practiceDiscover our free online SQL editor enhanced with AI to chat, explain, and generate code. Support SQL Server, MySQL, MariaDB, PostgreSQL, and SQLite.http://www.sqlfi…...

【数据结构】【线性表】特殊的线性表-字符串
目录 字符串的基本概念 字符串的三要素 字符串的基本概念 串的编码 串的实现及基本运算 顺序串的实现 串的静态数组实现 串的动态数组的实现 顺序存储的四种方案 链式串的实现 基本运算 方案三 方案一 字符串的基本概念 数据结构千千万,…...

app-1 App 逆向环境准备(mumu模拟器+magisk+LSPosed+算法助手+抓包(socksDroid+charles)+Frida环境搭建
一、前言 本篇是基于 mumu模拟器 进行环境配置记录。(真机的后面博客记录) 二、mumu模拟器magiskLSPosed算法助手 2.1、mumu模拟器 选择 mumu 模拟器,下载地址:https://mumu.163.com 安装完成后打开,找到设置中心进…...

在米尔FPGA开发板上实现Tiny YOLO V4,助力AIoT应用
学习如何在 MYIR 的 ZU3EG FPGA 开发板上部署 Tiny YOLO v4,对比 FPGA、GPU、CPU 的性能,助力 AIoT 边缘计算应用。 一、 为什么选择 FPGA:应对 7nm 制程与 AI 限制 在全球半导体制程限制和高端 GPU 受限的大环境下,FPGA 成为了中…...

【IT】测试用例模版(含示例)
这里写目录标题 一、测试用例模版二、怎么用模版示例如何使用这个模板 一、测试用例模版 一个相对标准的测试用例模板通常包含以下部分: 测试用例ID:唯一标识符,用于追踪测试用例。测试用例标题:简短描述测试用例的目的。测试用…...

react dnd——一个拖拽组件
React DnD是一个流行的库,用于在React应用程序中实现拖放功能。以下是对React DnD的详细解释,包括示例和API说明: 基本概念 在开始使用React DnD之前,了解以下几个基本概念是很重要的: Drag Source(拖动…...

3GPP R18 LTM(L1/L2 Triggered Mobility)是什么鬼?(三) RACH-less LTM cell switch
这篇看下RACH-less LTM cell switch。 相比于RACH-based LTM,RACH-less LTM在进行LTM cell switch之前就要先知道target cell的TA信息,进而才能进行RACH-less过程,这里一般可以通过UE自行测量或者通过RA过程获取,而这里的RA一般是通过PDCCH order过程触发。根据38.300中的描…...

Flutter解压文件并解析数据
Flutter解压文件并解析数据 前言 在 Flutter 开发中,我们经常需要处理文件的读取和解压。 这在处理应用数据更新、安装包、存档文件等场景中尤为常见。 本文将介绍如何在Flutter中使用archive插件来解压文件并解析数据。 准备 在开始之前,我们需要…...

21、结构体成员分布
结构体中的成员并不是紧挨着分布的,内存分布遵循字节对齐的原则。 按照成员定义的顺序,遵循字节对齐的原则存储。 字节对齐的原则: 找成员中占据字节数最大的成员,以它为单位进行空间空配 --- 遇到数组看元素的类型 每一个成员距离…...

TSWIKI知识库软件
TSWIKI 知识库软件介绍 推荐一个适合本地化部署、自托管的知识库软件 TSWIKI介绍 tswiki 是一个适合小团队、个人的知识库、资料管理的软件,所有数据均本地化存储。可以本地化、私有云部署,安装简单。在线预览。 主要功能说明 1、简化的软件依赖和安…...

深度学习安装环境笔记
1、输出cuda版本 torch.version.cuda 返回的是 PyTorch 在编译时所使用的 CUDA 版本,而不是运行时实际调用的 CUDA 版本。PyTorch 在运行时实际调用的 CUDA 版本取决于系统上安装的 CUDA 驱动和库。 import torch from torch.utils.cpp_extension import CUDA_HOME…...

使用android studio写一个Android的远程通信软件(APP),有通讯的发送和接收消息界面
以下是使用 Android Studio 基于 Java 语言编写一个简单的 Android APP 实现远程通信(这里以 TCP 通信为例)的代码示例,包含基本的通信界面以及发送和接收消息功能。 1. 创建项目 打开 Android Studio,新建一个 Empty Activity …...

学习Python的笔记14--迭代器和生成器
1.迭代器(Iterator) 概念: 迭代意味着重复多次,就像循环一样。 迭代器是一个可以记住遍历的位置的对象。 迭代器对象从集合的第一个元素开始访问,直到所有的元素被访问完结束。 迭代器只能往前不会后退。 1.iter…...

车机端同步outlook日历
最近在开发一个车机上的日历助手,其中一个需求就是要实现手机端日历和车机端日历数据的同步。然而这种需求似乎没办法实现,毕竟手机日历是手机厂商自己带的系统应用,根本不能和车机端实现数据同步的。 那么只能去其他公共的平台寻求一些机会&…...

教学案例:k相同的一次函数的图像关系
【题目】 请在同一个平面直角坐标系中画出一次函数y2x, y2x4的图象,并观察图象,你发现这两个图形有什么位置关系?为什么? 【答案】 图象是相互平行的两条直线 【解析】 一、教学活动形式 这里设计的教学活动形式是“画图 →…...

EmoAva:首个大规模、高质量的文本到3D表情映射数据集。
2024-12-03,由哈尔滨工业大学(深圳)的计算机科学系联合澳门大学、新加坡南洋理工大学等机构创建了EmoAva数据集,这是首个大规模、高质量的文本到3D表情映射数据集,对于推动情感丰富的3D头像生成技术的发展具有重要意义…...

Elasticsearch vs 向量数据库:寻找最佳混合检索方案
图片来自Shutterstock上的Bakhtiar Zein 多年来,以Elasticsearch为代表的基于全文检索的搜索方案,一直是搜索和推荐引擎等信息检索系统的默认选择。但传统的全文搜索只能提供基于关键字匹配的精确结果,例如找到包含特殊名词“Python3.9”的文…...

【Flink-scala】DataStream编程模型之水位线
DataStream API编程模型 1.【Flink-Scala】DataStream编程模型之 数据源、数据转换、数据输出 2.【Flink-scala】DataStream编程模型之 窗口的划分-时间概念-窗口计算程序 3.【Flink-scala】DataStream编程模型之 窗口计算-触发器-驱逐器 文章目录 DataStream API编程模型前言…...