Linux内核input子系统详解

目录

1 input子系统整体架构

2 input子系统驱动框架分析

2.1 怎么添加input_dev

2.2 input_dev和input_handler匹配后,connec函数做了什么

3 input子系统读数据流程

3.1 open输入设备流程

3.2 read读取输入事件流程

4 应用程序读取的输入数据是怎样的

4.1 type:表示哪类事件

4.2 code:表示该类事件下的哪一个事件

4.3 value:表示事件值

4.4 事件之间的界线

5 APP读取输入事件代码

5.1 get_input_info.c

5.2 input_read.c

5.3 input_read_poll.c

5.4 input_read_select.c

5.5 input_read_fasync.c

6 input应用程序调试技巧

6.1 确定设备信息

6.2使用命令读取数据

7 tslib

7.1 tslib框架分析

7.2交叉编译、测试tslib

7.2.1 交叉编译tslib

7.2.2 测试tslib

7.3 使用tslib编写程序

参考文献:


根据费曼学习法把知识点讲出来能够加深自己对知识点的理解,于是我录制了一个讲解input输入子系统框架的学习视频:

1 input子系统整体架构

如上图所示是Linux内核中input子系统的整体架构图,

  • 用户空间:用户可以直接用open,read,write访问输入设备,也可以利用tslib等一些库来使用输入设备。
  • input handler事件处理层:用户往下首先是input handler事件处理层,该层用于处理下面核心层上报的输入事件,然后往上给用户空间提供访问接口。
  • input 核心层:input核心层起到一个承上启下的作用,接受来自底层的输入事件转发给上层的handler,另外核心层还定义了一些注册函数。
  • input device输入设备层:这一层是硬件相关的驱动,从硬件获得数据,转换为标准的输入事件,然后向上汇报。

2 input子系统驱动框架分析

如上图所示为输入子系统的设备层,核心层和处理层之间的关系,当我们用input_register_handler 函数添加handler的时候,会将其添加到input.c里面的一个链表中,同样当我们用input_register_device添加input_dev的时候,会将其添加到input.c的input_dev链表中,然后无论是我们添加handler还是添加device,内核都会对两者进行match,当发现某个device和handler相匹配的时候,就会调用handler里面的connect函数,然后再connect函数里面去申请input_handle结构体,这个input_handle结构体里面保存着匹配的input_handler和input_dev。

2.1 怎么添加input_dev

接下来将分别看一下我们怎么添加input_dev也就是上图中的左下角的input_dev是怎么被添加上的,然后再看一下右边connect函数内部的细节。

上图是以内核中的按键输入驱动为例,看一下是怎么添加input_dev的,从左边可以看到,首先我们编写设备树文件,然后设备树文件会被解析成platform_device,然后利用函数of_device_add添加到platform_bus_type的设备链表中,然后我们还会编写一个platform_driver驱动结构体,然后利用platform_driver_register函数将驱动注册内核中,无论我们添加驱动还是添加设备,内核都会去对设备和驱动进行匹配,匹配成功后就会调用驱动结构体里面的probe函数,也就是上图中的gpio_keys_probe函数,然后在probe函数中ongoing会申请input_dev结构体,并且利用input_register_device函数将设备注册到input系统的设备链表中。

上图是以内核中的I2C接口的触摸屏驱动为例,看一下怎么添加input_dev的,这个图和前面的按键那个图类似,注意区别就是,对于i2c设备来说,它是挂载i2c_bus_type上面的不是platform_bus上的,其他的地方是类似的,就不再赘述了。

这里需要补充的一点是在connect函数中还有一个gtp_request_irq(ts);函数,这里是申请中断的,然后进一步调用了ret = request_threaded_irq(ts->client->irq, NULL,gtp_irq_handler,ts->pdata->irq_flags | IRQF_ONESHOT,ts->client->name,ts);函数,这里设置了中断处理函数是gtp_irq_handler,在gtp_irq_handler函数里面就是真正读取触摸屏数据的最底层的硬件函数了,里面就是调用i2c_transfer函数读取数据的了。

2.2 input_dev和input_handler匹配后,connec函数做了什么

前面看了怎么把input_dev添加到内核中,接下来看一下当input_dev和input_handler匹配后调用connect函数内部的细节,上图就是connect函数内部细节流程,

在connect函数里面我们可以猜一下,无非就是分配、设置、注册结构体,具体看一下里面,首先是申请了一个evdev结构体,然后我们可以看到这个evdev结构体里面又包含了input_handle结构体,然后这个input_handle结构体里面有两个重要成员就是input_dev和input_handler,这两个就是用来保存匹配的input_dev和input_handler的;

再往下可以看到list_add_tail_rcu(&handle->d_node, &dev->h_list);和list_add_tail_rcu(&handle->h_node, &handler->h_list);,这两个函数的意思就是把handle又分别保存到了input_handler和input_dev的链表中了。

再往下就是注册字符设备驱动程序,其中包含一个file_operations结构体,那么当应用程序调用open/read/write函数的时候就会调用这里的file_ops结构体里面的函数。

3 input子系统读数据流程

接下来我们看一下读取input输入事件数据的流程,先看一下open的流程。

3.1 open输入设备流程

上图就是open打开输入设备的流程,当我们调用open函数时,进一步会调用evdev_fops结构体里面的evdev_open函数,这个函数里面首先申请一个client,这个client就对应着我们的用户程序,然后会将这个client和evdev进行绑定,也就是把用户和输入设备绑定了。

3.2 read读取输入事件流程

如上图所示是read流程,先从最上面开始看,当应用程序调用read函数时,进一步会调用到evdev_read函数,然后在evdev_read函数中,如果有数据那么直接读取,如果没有数据,那么休眠等待。

好了,接下来再从图的最下面往上看,前面说了没数据就休眠,然后过一会硬件输入设备被操作了或者被按下了,然后就会产生中断,这时候就会进入中断处理函数,然后就会利用处理函数gpio_keys_gpio_work_func进行处理,然后这个函数一层层看下去最终就是调用input_event上报输入事件。

然后input_event其实就是核心层了,然后看一下input_event函数做了什么,这个函数一层层调用最终是调用到了input_to_handler函数,这个函数的源码大体可以看到,如果input_handler定义了filter就用filter函数处理,如果定义了input_handler->events函数那就用events函数处理,这里就是用events处理的。

那么这时候就去看input_handler->events函数,这个函数里面利用__pass_event(client, &event);把数据传给client,这个client是在open时候分配的。然后唤醒前面休眠的程序,这时候evdev_read函数就会被唤醒,然后读取输入事件。

至此,输入事件的read过程结束。

4 应用程序读取的输入数据是怎样的

应用程序得到的其实是一系列的输入事件,就是一个一个“struct input_event”,它定义如下:

每个输入事件input_event中都含有发生时间:timeval表示的是“自系统启动以来过了多少时间”,它是一个结构体,含有“tv_sec、tv_usec”两项(即秒、微秒)。

输入事件input_event中更重要的是:type(哪类事件)、code(哪个事件)、value(事件值),细讲如下:

4.1 type:表示哪类事件

比如EV_KEY表示按键类、EV_REL表示相对位移(比如鼠标),EV_ABS表示绝对位置(比如触摸屏)。有这几类事件(参考Linux内核头文件):

4.2 code:表示该类事件下的哪一个事件

比如对于EV_KEY(按键)类事件,它表示键盘。键盘上有很多按键,比如数字键1、2、3,字母键A、B、C里等。所以可以有这些事件:

对于触摸屏,它提供的是绝对位置信息,有X方向、Y方向,还有压力值。所以code值有这些:

4.3 value:表示事件值

对于按键,它的value可以是0(表示按键被按下)、1(表示按键被松开)、2(表示长按);

对于触摸屏,它的value就是坐标值、压力值。

4.4 事件之间的界线

APP读取数据时,可以得到一个或多个数据,比如一个触摸屏的一个触点会上报X、Y位置信息,也可能会上报压力值。

APP怎么知道它已经读到了完整的数据?

驱动程序上报完一系列的数据后,会上报一个“同步事件”,表示数据上报完毕。APP读到“同步事件”时,就知道已经读完了当前的数据。

同步事件也是一个input_event结构体,它的type、code、value三项都是0。

5 APP读取输入事件代码

5.1 get_input_info.c


#include <linux/input.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <sys/ioctl.h>
#include <stdio.h>/* ./01_get_input_info /dev/input/event0 */
int main(int argc, char **argv)
{int fd;int err;int len;int i;unsigned char byte;int bit;struct input_id id;unsigned int evbit[2];char *ev_names[] = {"EV_SYN ","EV_KEY ","EV_REL ","EV_ABS ","EV_MSC ","EV_SW	","NULL ","NULL ","NULL ","NULL ","NULL ","NULL ","NULL ","NULL ","NULL ","NULL ","NULL ","EV_LED ","EV_SND ","NULL ","EV_REP ","EV_FF	","EV_PWR ",};if (argc != 2){printf("Usage: %s <dev>\n", argv[0]);return -1;}fd = open(argv[1], O_RDWR);if (fd < 0){printf("open %s err\n", argv[1]);return -1;}err = ioctl(fd, EVIOCGID, &id);if (err == 0){printf("bustype = 0x%x\n", id.bustype );printf("vendor	= 0x%x\n", id.vendor  );printf("product = 0x%x\n", id.product );printf("version = 0x%x\n", id.version );}len = ioctl(fd, EVIOCGBIT(0, sizeof(evbit)), &evbit);if (len > 0 && len <= sizeof(evbit)){printf("support ev type: ");for (i = 0; i < len; i++){byte = ((unsigned char *)evbit)[i];for (bit = 0; bit < 8; bit++){if (byte & (1<<bit)) {printf("%s ", ev_names[i*8 + bit]);}}}printf("\n");}return 0;
}

5.2 input_read.c


#include <linux/input.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <sys/ioctl.h>
#include <stdio.h>
#include <string.h>
#include <unistd.h>/* ./01_get_input_info /dev/input/event0 noblock */
int main(int argc, char **argv)
{int fd;int err;int len;int i;unsigned char byte;int bit;struct input_id id;unsigned int evbit[2];struct input_event event;char *ev_names[] = {"EV_SYN ","EV_KEY ","EV_REL ","EV_ABS ","EV_MSC ","EV_SW	","NULL ","NULL ","NULL ","NULL ","NULL ","NULL ","NULL ","NULL ","NULL ","NULL ","NULL ","EV_LED ","EV_SND ","NULL ","EV_REP ","EV_FF	","EV_PWR ",};if (argc < 2){printf("Usage: %s <dev> [noblock]\n", argv[0]);return -1;}if (argc == 3 && !strcmp(argv[2], "noblock")){fd = open(argv[1], O_RDWR | O_NONBLOCK);}else{fd = open(argv[1], O_RDWR);}if (fd < 0){printf("open %s err\n", argv[1]);return -1;}err = ioctl(fd, EVIOCGID, &id);if (err == 0){printf("bustype = 0x%x\n", id.bustype );printf("vendor	= 0x%x\n", id.vendor  );printf("product = 0x%x\n", id.product );printf("version = 0x%x\n", id.version );}len = ioctl(fd, EVIOCGBIT(0, sizeof(evbit)), &evbit);if (len > 0 && len <= sizeof(evbit)){printf("support ev type: ");for (i = 0; i < len; i++){byte = ((unsigned char *)evbit)[i];for (bit = 0; bit < 8; bit++){if (byte & (1<<bit)) {printf("%s ", ev_names[i*8 + bit]);}}}printf("\n");}while (1){len = read(fd, &event, sizeof(event));if (len == sizeof(event)){printf("get event: type = 0x%x, code = 0x%x, value = 0x%x\n", event.type, event.code, event.value);}else{printf("read err %d\n", len);}}return 0;
}

5.3 input_read_poll.c


#include <linux/input.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <sys/ioctl.h>
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <poll.h>/* ./01_get_input_info /dev/input/event0 */
int main(int argc, char **argv)
{int fd;int err;int len;int ret;int i;unsigned char byte;int bit;struct input_id id;unsigned int evbit[2];struct input_event event;struct pollfd fds[1];nfds_t nfds = 1;char *ev_names[] = {"EV_SYN ","EV_KEY ","EV_REL ","EV_ABS ","EV_MSC ","EV_SW	","NULL ","NULL ","NULL ","NULL ","NULL ","NULL ","NULL ","NULL ","NULL ","NULL ","NULL ","EV_LED ","EV_SND ","NULL ","EV_REP ","EV_FF	","EV_PWR ",};if (argc != 2){printf("Usage: %s <dev>\n", argv[0]);return -1;}fd = open(argv[1], O_RDWR | O_NONBLOCK);if (fd < 0){printf("open %s err\n", argv[1]);return -1;}err = ioctl(fd, EVIOCGID, &id);if (err == 0){printf("bustype = 0x%x\n", id.bustype );printf("vendor	= 0x%x\n", id.vendor  );printf("product = 0x%x\n", id.product );printf("version = 0x%x\n", id.version );}len = ioctl(fd, EVIOCGBIT(0, sizeof(evbit)), &evbit);if (len > 0 && len <= sizeof(evbit)){printf("support ev type: ");for (i = 0; i < len; i++){byte = ((unsigned char *)evbit)[i];for (bit = 0; bit < 8; bit++){if (byte & (1<<bit)) {printf("%s ", ev_names[i*8 + bit]);}}}printf("\n");}while (1){fds[0].fd = fd;fds[0].events  = POLLIN;fds[0].revents = 0;ret = poll(fds, nfds, 5000);if (ret > 0){if (fds[0].revents == POLLIN){while (read(fd, &event, sizeof(event)) == sizeof(event)){printf("get event: type = 0x%x, code = 0x%x, value = 0x%x\n", event.type, event.code, event.value);}}}else if (ret == 0){printf("time out\n");}else{printf("poll err\n");}}return 0;
}

5.4 input_read_select.c


#include <linux/input.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <sys/ioctl.h>
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <sys/select.h>/* According to earlier standards */
#include <sys/time.h>/* ./01_get_input_info /dev/input/event0 */
int main(int argc, char **argv)
{int fd;int err;int len;int ret;int i;unsigned char byte;int bit;struct input_id id;unsigned int evbit[2];struct input_event event;int nfds;struct timeval tv;fd_set readfds;char *ev_names[] = {"EV_SYN ","EV_KEY ","EV_REL ","EV_ABS ","EV_MSC ","EV_SW	","NULL ","NULL ","NULL ","NULL ","NULL ","NULL ","NULL ","NULL ","NULL ","NULL ","NULL ","EV_LED ","EV_SND ","NULL ","EV_REP ","EV_FF	","EV_PWR ",};if (argc != 2){printf("Usage: %s <dev>\n", argv[0]);return -1;}fd = open(argv[1], O_RDWR | O_NONBLOCK);if (fd < 0){printf("open %s err\n", argv[1]);return -1;}err = ioctl(fd, EVIOCGID, &id);if (err == 0){printf("bustype = 0x%x\n", id.bustype );printf("vendor	= 0x%x\n", id.vendor  );printf("product = 0x%x\n", id.product );printf("version = 0x%x\n", id.version );}len = ioctl(fd, EVIOCGBIT(0, sizeof(evbit)), &evbit);if (len > 0 && len <= sizeof(evbit)){printf("support ev type: ");for (i = 0; i < len; i++){byte = ((unsigned char *)evbit)[i];for (bit = 0; bit < 8; bit++){if (byte & (1<<bit)) {printf("%s ", ev_names[i*8 + bit]);}}}printf("\n");}while (1){/* 设置超时时间 */tv.tv_sec  = 5;tv.tv_usec = 0;/* 想监测哪些文件? */FD_ZERO(&readfds);    /* 先全部清零 */	FD_SET(fd, &readfds); /* 想监测fd *//* 函数原型为:int select(int nfds, fd_set *readfds, fd_set *writefds,fd_set *exceptfds, struct timeval *timeout);* 我们为了"read"而监测, 所以只需要提供readfds*/nfds = fd + 1; /* nfds 是最大的文件句柄+1, 注意: 不是文件个数, 这与poll不一样 */ ret = select(nfds, &readfds, NULL, NULL, &tv);if (ret > 0)  /* 有文件可以提供数据了 */{/* 再次确认fd有数据 */if (FD_ISSET(fd, &readfds)){while (read(fd, &event, sizeof(event)) == sizeof(event)){printf("get event: type = 0x%x, code = 0x%x, value = 0x%x\n", event.type, event.code, event.value);}}}else if (ret == 0)  /* 超时 */{printf("time out\n");}else   /* -1: error */{printf("select err\n");}}return 0;
}

5.5 input_read_fasync.c


#include <linux/input.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <sys/ioctl.h>
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <signal.h>
#include <sys/types.h>
#include <unistd.h>int fd;void my_sig_handler(int sig)
{struct input_event event;while (read(fd, &event, sizeof(event)) == sizeof(event)){printf("get event: type = 0x%x, code = 0x%x, value = 0x%x\n", event.type, event.code, event.value);		}
}/* ./05_input_read_fasync /dev/input/event0 */
int main(int argc, char **argv)
{int err;int len;int ret;int i;unsigned char byte;int bit;struct input_id id;unsigned int evbit[2];unsigned int flags;int count = 0;char *ev_names[] = {"EV_SYN ","EV_KEY ","EV_REL ","EV_ABS ","EV_MSC ","EV_SW	","NULL ","NULL ","NULL ","NULL ","NULL ","NULL ","NULL ","NULL ","NULL ","NULL ","NULL ","EV_LED ","EV_SND ","NULL ","EV_REP ","EV_FF	","EV_PWR ",};if (argc != 2){printf("Usage: %s <dev>\n", argv[0]);return -1;}/* 注册信号处理函数 */signal(SIGIO, my_sig_handler);/* 打开驱动程序 */fd = open(argv[1], O_RDWR | O_NONBLOCK);if (fd < 0){printf("open %s err\n", argv[1]);return -1;}err = ioctl(fd, EVIOCGID, &id);if (err == 0){printf("bustype = 0x%x\n", id.bustype );printf("vendor	= 0x%x\n", id.vendor  );printf("product = 0x%x\n", id.product );printf("version = 0x%x\n", id.version );}len = ioctl(fd, EVIOCGBIT(0, sizeof(evbit)), &evbit);if (len > 0 && len <= sizeof(evbit)){printf("support ev type: ");for (i = 0; i < len; i++){byte = ((unsigned char *)evbit)[i];for (bit = 0; bit < 8; bit++){if (byte & (1<<bit)) {printf("%s ", ev_names[i*8 + bit]);}}}printf("\n");}/* 把APP的进程号告诉驱动程序 */fcntl(fd, F_SETOWN, getpid());/* 使能"异步通知" */flags = fcntl(fd, F_GETFL);fcntl(fd, F_SETFL, flags | FASYNC);while (1){printf("main loop count = %d\n", count++);sleep(2);}return 0;
}

6 input应用程序调试技巧

6.1 确定设备信息

输入设备的设备节点名为/dev/input/eventX(也可能是/dev/eventX,X表示0、1、2等数字)。查看设备节点,可以执行以下命令:

ls /dev/input/* -l
或
ls /dev/event* -l

可以看到类似下面的信息:

怎么知道这些设备节点对应什么硬件呢?可以在板子上执行以下命令:

cat /proc/bus/input/devices

这条指令的含义就是获取与event对应的相关设备信息,可以看到类似以下的结果:

那么这里的I、N、P、S、U、H、B对应的每一行是什么含义呢?其实就对应这个结构体

I:id of the device(设备ID)该参数由结构体struct input_id来进行描述,

N:name of the device 设备名称

P:physical path to the device in the system hierarchy 系统层次结构中设备的物理路径。

S:sysfs path 位于sys文件系统的路径

U:unique identification code for the device(if device has it) 设备的唯一标识码

H:list of input handles associated with the device.  与设备关联的输入句柄列表。

B:bitmaps(位图)  PROP:device properties and quirks(设备属性)  EV:types of events supported by the device(设备支持的事件类型)    KEY:keys/buttons this device has(此设备具有的键/按钮)    MSC:miscellaneous events supported by the device(设备支持的其他事件)    LED:leds present on the device(设备上的指示灯)

值得注意的是B位图,比如上图中“B: EV=b”用来表示该设备支持哪类输入事件。b的二进制是1011,bit0、1、3为1,表示该设备支持0、1、3这三类事件,即EV_SYN、EV_KEY、EV_ABS。

再举一个例子,“B: ABS=2658000 3”如何理解?它表示该设备支持EV_ABS这一类事件中的哪一些事件。这是2个32位的数字:0x2658000、0x3,高位在前低位在后,组成一个64位的数字:“0x2658000,00000003”,数值为1的位有:0、1、47、48、50、53、54,即:0、1、0x2f、0x30、0x32、0x35、0x36,对应以下这些宏:

即这款输入设备支持上述的ABS_X、ABS_Y、ABS_MT_SLOT、ABS_MT_TOUCH_MAJOR、ABS_MT_WIDTH_MAJOR、ABS_MT_POSITION_X、ABS_MT_POSITION_Y这些绝对位置事件。

6.2使用命令读取数据

调试输入系统时,直接执行类似下面的命令,然后操作对应的输入设备即可读出数据:

hexdump /dev/input/event0

在开发板上执行上述命令之后,点击按键或是点击触摸屏,就会打印以下信息:

上图中的type为3,对应EV_ABS;code为0x35对应ABS_MT_POSITION_X;code为0x36对应ABS_MT_POSITION_Y。

上图中还发现有2个同步事件:它的type、code、value都为0。表示电容屏上报了2次完整的数据。

7 tslib

tslib是一个触摸屏的开源库,可以使用它来访问触摸屏设备,可以给输入设备添加各种“filter”(过滤器,就是各种处理),地址是:tslib。

编译tslib后,可以得到libts库,还可以得到各种工具:较准工具、测试工具。

7.1 tslib框架分析

tslib的主要代码如下:

核心在于“plugins”目录里的“插件”,或称为“module”。这个目录下的每个文件都是一个module,每个module都提供2个函数:read、read_mt,前者用于读取单点触摸屏的数据,后者用于读取多点触摸屏的数据。

要分析tslib的框架,先看看示例程序怎么使用,我们参考ts_test.c和ts_test_mt.c,前者用于一般触摸屏(比如电阻屏、单点电容屏),后者用于多点触摸屏。

一个图就可以弄清楚tslib的框架:

调用ts_open后,可以打开某个设备节点,构造出一个tsdev结构体。

然后调用ts_config读取配置文件的处理,假设/etc/ts.conf内容如下:

 module_raw input

 module pthres pmin=1

 module dejitter delta=100

 module linear

每行表示一个“module”或“moduel_raw”。

对于所有的“module”,都会插入tsdev.list链表头,也就是tsdev.list执行配置文件中最后一个“module”,配置文件中第一个“module”位于链表的尾部。

对于所有的“module_raw”,都会插入tsdev.list_raw链表头,一般只有一个“module_raw”。

注意:tsdev.list中最后一个“module”会指向ts_dev.list_raw的头部。

无论是调用ts_read还是ts_read_mt,都是通过tsdev.list中的模块来处理数据的。这写模块是递归调用的,比如linear模块的read函数如下:

linear模块的read_raw函数如下:

因为是递归调用,所有最先使用input模块读取设备节点得到原始数据,再依次经过pthres模块、dejitter模块、linear模块处理后,才返回最终数据。

7.2交叉编译、测试tslib

7.2.1 交叉编译tslib

在编译之前要设置交叉编译工具链 

对于IMX6ULL,命令如下

export ARCH=arm

export CROSS_COMPILE=arm-linux-gnueabihf-

export PATH=$PATH:/data/chw/imx6ull_20230512/bsp/100ask_imx6ull-sdk/ToolChain/arm-buildroot-linux-gnueabihf_sdk-buildroot/bin

交叉编译tslib

./configure --host=arm-linux-gnueabihf  --prefix=/

make

make install DESTDIR=$PWD/tmp

确定工具链中头文件、库文件目录:

echo 'main(){}'| arm-buildroot-linux-gnueabihf-gcc -E -v -

把头文件、库文件放到工具链目录下

cd tslib-1.21/tmp/

cp include/*   /data/chw/imx6ull_20230512/bsp/100ask_imx6ull-sdk/ToolChain/arm-buildroot-linux-gnueabihf_sdk-buildroot/arm-buildroot-linux-gnueabihf/sysroot/usr/include

cp -d lib/*so*  /data/chw/imx6ull_20230512/bsp/100ask_imx6ull-sdk/ToolChain/arm-buildroot-linux-gnueabihf_sdk-buildroot/arm-buildroot-linux-gnueabihf/sysroot/usr/lib/

7.2.2 测试tslib

把库文件放到单板上:运行程序要用。先在开发板上使用NFS挂载Ubuntu的目录,再把前面编译出来的tslib-1.21/tmp/部分文件复制到板子上,示例命令如下:

cp  /mnt/tslib-1.21/tmp/lib/*  -drf     /lib

cp  /mnt/tslib-1.21/tmp/bin/*            /bin

cp  /mnt/tslib-1.21/tmp/etc/ts.conf  -d  /etc

对于IMX6ULL,首先需要关闭默认的qt gui程序,才可以执行ts_test_mt测试命令,关闭qt命令如下所示:

mv  /etc/init.d/S07hmi  /root

reboot

在单板上执行测试程序:

ts_test_mt

7.3 使用tslib编写程序

驱动程序使用slot、tracking_id来标识一个触点;当tracking_id等于-1时,标识这个触点被松开了。

触摸屏可能支持多个触点,比如5个:tslib为了简化处理,即使只有2个触点,ts_read_mt函数也会返回5个触点数据,可以根据标志位判断数据是否有效。

ts_read_mt函数原型如下:

假设nr设置为1,max_slots设置为5,那么读到的数据保存在:samp[0][0]、samp[0][1]、samp[0][2]、samp[0][3]、samp[0][4]中。

假设nr设置为1,max_slots设置为5,那么督导的数据保存在:samp[0][0]、samp[0][1]、samp[0][2]、samp[0][3]、samp[0][4]和samp[1][0]、samp[1][1]、samp[1][2]、samp[1][3]、samp[1][4]中。

ts_sample_mt结构体如下:

实现一个程序,不断打印2个触点的距离。

思路:假设是5点触摸屏,调用一次ts_read_mt可以得到5个新数据;使用新旧数据进行判断,如果有2个触点,就打印出距离。

#include <stdio.h>
#include <stdint.h>
#include <stdlib.h>
#include <string.h>
#include <signal.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <getopt.h>#include <linux/input.h>#include <sys/ioctl.h>#include <tslib.h>int distance(struct ts_sample_mt *point1, struct ts_sample_mt *point2)
{int x = point1->x - point2->x;int y = point1->y - point2->y;return x*x + y*y;
}int main(int argc, char **argv)
{struct tsdev *ts;int i;int ret;struct ts_sample_mt **samp_mt;struct ts_sample_mt **pre_samp_mt;	int max_slots;int point_pressed[20];struct input_absinfo slot;int touch_cnt = 0;ts = ts_setup(NULL, 0);if (!ts){printf("ts_setup err\n");return -1;}if (ioctl(ts_fd(ts), EVIOCGABS(ABS_MT_SLOT), &slot) < 0) {perror("ioctl EVIOGABS");ts_close(ts);return errno;}max_slots = slot.maximum + 1 - slot.minimum;samp_mt = malloc(sizeof(struct ts_sample_mt *));if (!samp_mt) {ts_close(ts);return -ENOMEM;}samp_mt[0] = calloc(max_slots, sizeof(struct ts_sample_mt));if (!samp_mt[0]) {free(samp_mt);ts_close(ts);return -ENOMEM;}pre_samp_mt = malloc(sizeof(struct ts_sample_mt *));if (!pre_samp_mt) {ts_close(ts);return -ENOMEM;}pre_samp_mt[0] = calloc(max_slots, sizeof(struct ts_sample_mt));if (!pre_samp_mt[0]) {free(pre_samp_mt);ts_close(ts);return -ENOMEM;}for ( i = 0; i < max_slots; i++)pre_samp_mt[0][i].valid = 0;while (1){ret = ts_read_mt(ts, samp_mt, max_slots, 1);if (ret < 0) {printf("ts_read_mt err\n");ts_close(ts);return -1;}for (i = 0; i < max_slots; i++){if (samp_mt[0][i].valid){memcpy(&pre_samp_mt[0][i], &samp_mt[0][i], sizeof(struct ts_sample_mt));}}touch_cnt = 0;for (i = 0; i < max_slots; i++){if (pre_samp_mt[0][i].valid && pre_samp_mt[0][i].tracking_id != -1)point_pressed[touch_cnt++] = i;}if (touch_cnt == 2){printf("distance: %08d\n", distance(&pre_samp_mt[0][point_pressed[0]], &pre_samp_mt[0][point_pressed[1]]));}}return 0;
}

参考文献:

八、INPUT子系统和内核自带的GPIO按键驱动_乔碧萝成都分萝的博客-CSDN博客

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

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

相关文章

chap认证带客户端IP分配案例

PPP协议两边的网段可以不在同一个网段&#xff0c;因为数据链路帧用0xff表示帧&#xff0c;不用arp&#xff0c;所以可以不同网段。 R1&#xff1a; aaa local-user test password cipher admin local-user test service-type ppp interface Serial4/0/0 link-protocol ppp pp…

Git 标签(Tag)实战:打标签和删除标签的步骤指南

目录 前言使用 Git 打本地和远程标签&#xff08;Tag&#xff09;删除本地和远程 Git 标签&#xff08;Tag&#xff09;开源项目标签&#xff08;Tag&#xff09;实战打标签删除标签 结语开源微服务商城项目前后端分离项目 前言 在开源项目中&#xff0c;版本控制是至关重要的…

在钣金加工领域,迅镭激光切割机广泛使用的原因和优点何在?

激光切割工艺和激光切割设备正在被广泛的板材加工企业逐渐理解并接受&#xff0c;凭借其高效率的加工、高精度的加工、优质的切割断面、三维切割能力等诸多优势&#xff0c;逐步取代了传统的钣金切割设备。 苏州迅镭激光科技有限公司推出的激光切割设备的柔性化程度高&#xff…

Centos下用nodejs实现一个简单的web服务器

WebRTC是音视频直播中最常用的一个框架&#xff0c;在使用的过程中&#xff0c;我们就需要实现一个服务器端。本文以nodejs实现一个服务器为例&#xff0c;讲述一下在centos下如何用nodejs实现一个简单的web服务器。 一、安装nodejs 在linux环境下安装nodejs有多重方式&#x…

消息中间件——RabbitMQ(二)各大主流消息中间件综合对比介绍!

前言 消息队列已经逐渐成为企业IT系统内部通信的核心手段。它具有低耦合、可靠投递、广播、流量控制、最终一致性等一系列功能&#xff0c;成为异步RPC的主要手段之一。当今市面上有很多主流的消息中间件&#xff0c;如老牌的ActiveMQ、RabbitMQ&#xff0c;炙手可热的Kafka&a…

数据库概述 -- 数据模型知识点要点详解

数据模型 概述概念数据特征组成要素分类概念层数据模型逻辑层数据模型物理层数据模型 主页传送门&#xff1a;&#x1f4c0; 传送 概述 数据模型是对现实世界数据特征的抽象&#xff0c;它描述了数据的结构、操作和约束条件&#xff0c;为数据库系统的信息表示与操作提供一个抽…

JWT

目录 JWT组成 第一部分header 第二部分payload 第三部分signature 注意 JWT认证算法&#xff1a;签发和校验 drf使用jwt drf项目的jwt认证开发流程 drf-jwt安装和简单使用 安装 简单使用 drf-jwt使用 jwt内置类JSONWebTokenAuthentication 控制使用jwt的登录接口…

人工智能基础_机器学习014_BGD批量梯度下降公式更新_进一步推导_SGD随机梯度下降和MBGD小批量梯度下降公式进一步推导---人工智能工作笔记0054

然后我们先来看BGD批量梯度下降,可以看到这里,其实这个公式来源于 梯度下降的公式对吧,其实就是对原始梯度下降公式求偏导以后的梯度下降公式,然后 使用所有样本进行梯度下降得来的,可以看到* 1/n 其实就是求了一个平均数对吧.所有样本的平均数. 然后我们看,我们这里* 1/n那么…

启用NTP服务解决Linux系统时间与北京时间不同步问题

一、背景 1、服务器的Linux版本为Linux version 4.18.0-348.7.1.el8_5.x86_64 (mockbuildkbuilder.bsys.centos.org) (gcc version 8.5.0 20210514 (Red Hat 8.5.0-4) (GCC)) #1 SMP Wed Dec 22 13:25:12 UTC 2021 2、NTP即Network Time Protocol&#xff08;网络时间协议&am…

Unity3D与iOS的交互 简单版开箱即用

本文适合的情况如下&#xff1a; Unity客户端人员 与 IOS端研发人员合作的情况 目录 From U3D to iOS 实现原理 1.unity工程目录创建2个文件 NativeCallProxy.m、NativeCallProxy.h 并且放到Unity工程目录Plugins/iOS/unity_ios_plus目录下 2.创建C#调用脚本 定义对应.mm脚…

海康监控摄像机和录像机接入LiveMedia GB28181平台实现远程调取监控视频

海康威视各种型号监控摄像头或硬盘录像机&#xff08;NVR/HVR&#xff09;接入LiveMedia GB28181平台配置过程都非常简单明了&#xff0c;但有些细节需要注意&#xff0c;避免走弯路。 1、基本要求 (1) 网络要求 总体来说&#xff0c;只要监控设备和GB28181平台的网络是连通…

centos9 stream 下 rabbitmq高可用集群搭建及使用

RabbitMQ是一种常用的消息队列系统&#xff0c;可以快速搭建一个高可用的集群环境&#xff0c;以提高系统的弹性和可靠性。下面是搭建RabbitMQ集群的步骤&#xff1a; 基于centos9 stream系统 1. 安装Erlang和RabbitMQ 首先需要在所有节点上安装Erlang和RabbitMQ。建议使用官…

7个UI设计必备课程,小白必看!

无论你是想提高技能的资深UI设计师还是网站开发人员&#xff0c;又或者是刚转行不久的UI设计新手&#xff0c;学习UI设计课程都会让你做出更美观、更有影响力的UI界面设计作品。现在网上有很多网上的UI设计课程。通过这些课程&#xff0c;你可以自己学习、掌握一些UI设计的基础…

【Jmeter】生成html格式接口自动化测试报告

jmeter自带执行结果查看的插件&#xff0c;但是需要在jmeter工具中才能查看&#xff0c;如果要向领导提交测试结果&#xff0c;不够方便直观。 笔者刚做了这方面的尝试&#xff0c;总结出来分享给大家。 这里需要用到ant来执行测试用例并生成HTML格式测试报告。 一、ant下载安…

k8s之集群调度

目录 调度 工作机制 调度过程 调度算法 优先级 指定调度节点 调度 Kubernetes 是通过 List-Watch 的机制进行每个组件的协作&#xff0c;保持数据同步的&#xff0c;每个组件之间的设计实现了解耦。 用户是通过 kubectl 根据配置文件&#xff0c;向 APIServer 发送命令…

linux远程桌面管理工具xrdp

一、概述 我们知道&#xff0c;我们日常通过vnc来远程管理linux图形界面&#xff0c;今天分享一工具Xrdp&#xff0c;它是一个开源工具&#xff0c;允许用户通过Windows RDP访问Linux远程桌面。 除了Windows RDP之外&#xff0c;xrdp工具还接受来自其他RDP客户端的连接&#xf…

C# Winform串口助手

界面设置 修改控件name属性 了解SerialPort类 实现串口的初始化&#xff0c;开关 创建虚拟串口 namespace 串口助手 {public partial class Form1 : Form{public Form1(){InitializeComponent();}private void Form1_Load(object sender, EventArgs e){//在设计页面已经预先…

Map和Set(JAVA)

本篇文章建议在了解了哈希表和二叉搜索树后食用更佳。 链接: 二叉搜索树 和 哈希表 (JAVA) Map和Set都是一种专门用来进行搜索的容器或者数据结构&#xff0c;其搜索的效率与其具体的实例化子类有关。 Map接口 Map是一个接口&#xff0c;不能直接实例化对象&#xff0c;如果…

【css3】涟漪动画

效果展示 dom代码 <div class"mapSelfTitle66"><div></div> </div> 样式代码 .mapSelfTitle66{width:120px;height:60px;position: relative;&>div{width:100%;height:100%;background: url("~/assets/images/video_show/err…

javaee实验:搭建maven+spring boot开发环境,开发“Hello,Spring Boot”应用

目录 mavenspringboot实验目的实验内容环境的搭建 在开发中&#xff0c;maven和spring都是非常常用、非常重要的管理工具和框架&#xff0c;今天就在这里使用idea进行环境的搭建和创建第一个spring程序 maven 1.1maven是一个跨平台的项目管理工具&#xff08;主要管理jar包&am…