【北京迅为】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,和驱动代码中的设置一一对应。 

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/888902.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【银河麒麟操作系统真实案例分享】内存黑洞导致服务器卡死分析全过程

了解更多银河麒麟操作系统全新产品&#xff0c;请点击访问 麒麟软件产品专区&#xff1a;https://product.kylinos.cn 开发者专区&#xff1a;https://developer.kylinos.cn 文档中心&#xff1a;https://documentkylinos.cn 现象描述 机房显示器连接服务器后黑屏&#xff…

docker系统详解哟 以及相关命令 Centos Kali安装相关详解 Docker-Compose 亲测

目录 who Is Docker 概念 centos7 安装docker kali安装docker docker安装nginx Docker常用命令 容器得常用命令 Docker-Compose install 常用docker-compose命令 who Is Docker 软件的打包技术&#xff0c;就是将算乱的多个文件打包为一个整体&#xff0c;打包技术在没…

Java项目实战II基于微信小程序的旅游社交平台(开发文档+数据库+源码)

目录 一、前言 二、技术介绍 三、系统实现 四、核心代码 五、源码获取 全栈码农以及毕业设计实战开发&#xff0c;CSDN平台Java领域新星创作者&#xff0c;专注于大学生项目实战开发、讲解和毕业答疑辅导。 一、前言 随着移动互联网的迅猛发展&#xff0c;旅游已经成为人…

Windows 和 Linux 系统命令行操作详解:从文件管理到进程监控

1.切换盘符与目录操作 在命令行中&#xff0c;切换盘符和目录是最常见的操作。尽管 DOS 和 Linux 在这些操作上有所不同&#xff0c;但它们都能实现相似的功能。 (1)切换盘符 ①DOS命令&#xff1a;在 DOS 中&#xff0c;切换盘符非常简单&#xff0c;使用 盘符名:&#xff…

【数据库】关系代数和SQL语句

一 对于教学数据库的三个基本表 学生S(S#,SNAME,AGE,SEX) 学习SC(S#,C#,GRADE) 课程(C#,CNAME,TEACHER) &#xff08;1&#xff09;试用关系代数表达式和SQL语句表示&#xff1a;检索WANG同学不学的课程号 select C# from C where C# not in(select C# from SCwhere S# in…

IS-IS二

目录 ISIS建立邻接关系的基本条件&#xff1a; 1、接口链路类型一致 2、广播型链路上&#xff0c;接口类型一致 3、Hello包级别和类型一致 4、L1区域的ID要一致&#xff0c;L2的邻居区域ID不做要求 5、L1-2在区域ID相同下&#xff0c;即建立L1也建立L2区域ID不同只能建立…

echarts全屏,vue

echarts实现全屏并且不失真&#xff0c;全屏图片需要自己换 html&#xff1a; <!-- 图表全屏盒子 --> <div style"position: relative;" ref"charts_orders"><!-- 图表 --><div class"chart_box" v-show"sho…

杂谈随笔-关于unity开发游戏

最近有在做unity的游戏开发&#xff0c;都是自学&#xff0c;甚至没有完整的课程体系…… 在犹豫要不要出系列教程&#xff0c;帮助新手快速入门的同时算是巩固一下基础知识。 那这篇文章先谈谈我对于引擎开发游戏的一些小观点&#xff0c;算是做了这么十几个星期的微不足道的…

️ 在 Windows WSL 上部署 Ollama 和大语言模型的完整指南20241206

&#x1f6e0;️ 在 Windows WSL 上部署 Ollama 和大语言模型的完整指南 &#x1f4dd; 引言 随着大语言模型&#xff08;LLM&#xff09;和人工智能的飞速发展&#xff0c;越来越多的开发者尝试在本地环境中部署大模型进行实验。然而&#xff0c;由于资源需求高、网络限制多…

[光源控制] UI调节光源亮度参数失效

📢博客主页:https://loewen.blog.csdn.net📢欢迎点赞 👍 收藏 ⭐留言 📝 如有错误敬请指正!📢本文由 丶布布原创,首发于 CSDN,转载注明出处🙉📢现在的付出,都会是一种沉淀,只为让你成为更好的人✨文章预览: 一. 前言二. 串口调试助手辅助排查接线问题二. …

设计模式の单例工厂原型模式

文章目录 前言一、单例模式1.1、饿汉式静态常量单例1.2、饿汉式静态代码块单例1.3、懒汉式单例&#xff08;线程不安全&#xff09;1.4、懒汉式单例&#xff08;线程安全&#xff0c;同步代码块&#xff09;1.5、懒汉式单例&#xff08;线程不安全&#xff0c;同步代码块&#…

net.sf.jsqlparser.statement.select.SelectItem

今天一启动项目&#xff0c;出现了这个错误&#xff0c;仔细想了想&#xff0c;应该是昨天合并代码&#xff0c;导致的mybatis-plus版本冲突&#xff0c;以及分页PageHelper版本不兼容 可以看见这个我是最下边的Caused by 报错信息&#xff0c;这个地方提示我 net .sf.jsqlpar…

第427场周赛: 转换数组、用点构造面积最大的矩形 Ⅰ、长度可被 K 整除的子数组的最大元素和、用点构造面积最大的矩形 Ⅱ

Q1、转换数组 1、题目描述 给你一个整数数组 nums&#xff0c;它表示一个循环数组。请你遵循以下规则创建一个大小 相同 的新数组 result &#xff1a; 对于每个下标 i&#xff08;其中 0 < i < nums.length&#xff09;&#xff0c;独立执行以下操作&#xff1a; 如…

18 设计模式之迭代器模式(书籍遍历案例)

一、什么是迭代器模式 迭代器模式&#xff08;Iterator Pattern&#xff09;是一种行为型设计模式&#xff0c;允许客户端通过统一的接口顺序访问一个集合对象中的元素&#xff0c;而无需暴露集合对象的内部实现。这个模式主要用于访问聚合对象&#xff08;如集合、数组等&…

CV工程师专用键盘开源项目硬件分析

1、前言 作为一个电子发烧友&#xff0c;你是否有遇到过这样的问题呢。当我们去查看函数定义的时候&#xff0c;需要敲击鼠标右键之后选择go to definition。更高级一些&#xff0c;我们使用键盘的快捷键来查看定义&#xff0c;这时候可以想象一下&#xff0c;你左手按下ALT&a…

CSS 属性的可继承

一、可继承的属性 1. 文本相关属性 color&#xff1a;文本的颜色。 font-family&#xff1a;字体系列。 font-size&#xff1a;文本的大小。 font-style&#xff1a;文本的样式。 line-height&#xff1a;行与行之间的垂直间距。 2. 列表相关属性 list-style-type&#xff1a;…

Rust学习笔记_18——HashSet

Rust学习笔记_15——Union Rust学习笔记_16——Vector Rust学习笔记_17——HashMap HashSet 文章目录 HashSet1. 创建2. 插入3. 检查元素是否存在4. 遍历5. 移除6. 工作原理7. 示例 Rust 中的 HashSet 是一种集合数据结构&#xff0c;它允许你存储不重复的元素&#xff0c;并且…

uniapp远程摄像头流界面上显示

用到的插件&#xff1a;dplayer、hls dplayer官网&#xff1a;dplayer dplayer官网npm安装的是最新版本&#xff08;1.27.1&#xff09;&#xff0c;真机运行异常了&#xff0c;可以安装历史版本 dplayer历史版本 远程摄像头视频流格式&#xff1a;m3u8 可以用来测试的视频流&a…

001-mysql安装

[rootcentos701 ~]# hostname -I 10.0.0.200 172.17.0.1 [rootcentos701 ~]# hostname centos701 [rootcentos701 ~]# rpm -qa | grep mariadb [rootcentos701 ~]# rpm -e --nodeps mariadb-libs-5.5.65-1.el7.x86_64 [rootcentos701 ~]# useradd mysql -s /sbin/nologin #创建…

ubuntu20.04设置远程桌面

安装xrdp sudo apt install xrdp 2、 检查xrdp状态 sudo systemctl status xrdp3、&#xff08;若为Ubuntu 20&#xff09;添加xrdp至ssl-cert sudo adduser xrdp ssl-cert 4、重启服务 sudo systemctl restart xrdp最后可以远程了&#xff0c;注意一个账号只能一个登录