(新手)Linux 输入子系统实战教程 —— 02设备信息查询 + 输入事件读取(阻塞 / 非阻塞模式)
Linux 输入子系统实战教程 —— 设备信息查询 输入事件读取阻塞 / 非阻塞模式完整学习文档本文档基于Linux 输入设备事件读取程序编写包含完整注释源码、核心原理、逐模块解析、真实实验现象、错误原因分析专为嵌入式 Linux 驱动新手打造完全贴合百问网开发板 / 虚拟机实操环境覆盖你实验中出现的read err -1、阻塞 / 非阻塞模式、电源键设备等全部核心内容。一、文档说明本文档讲解的程序在旧版查询设备信息基础上升级新增实时读取输入事件、阻塞 / 非阻塞运行模式切换功能完整包含代码注释、原理讲解、编译运行、实验现象、错误排查所有案例、现象均来自你真实的实验操作可直接对照学习。二、程序核心功能保留旧版功能查询输入设备总线类型、厂商 ID、产品 ID、版本号、支持的事件类型新增核心功能实时读取键盘 / 鼠标 / 电源键的输入事件打印事件类型、编码、状态支持两种运行模式阻塞模式默认无事件时程序卡住等待不打印冗余信息非阻塞模式无事件时read直接返回不等待会出现read err -1兼容所有 Linux 输入设备键盘、鼠标、电源键、触摸屏等。三、核心前置知识新手必学3.1 Linux 标准输入事件结构体struct input_event内核向应用层传递所有输入设备事件统一使用该结构体struct input_event { struct timeval time; // 事件发生的时间戳 __u16 type; // 事件类型如 EV_KEY 按键事件 __u16 code; // 具体按键/坐标编码如回车键、鼠标左键 __s32 value; // 事件状态1按下 0松开 2长按重复 };3.2 阻塞模式 VS 非阻塞模式表格模式打开标志无事件时表现有事件时表现阻塞模式默认O_RDWRread 函数休眠等待程序卡住不打印read 读取事件打印事件信息非阻塞模式O_RDWR | O_NONBLOCKread 立即返回 - 1循环打印read err -1read 读取事件打印事件信息3.3 事件位图机制内核使用 ** 二进制位bit** 表示设备支持的事件类型1支持0不支持程序通过双层循环 位运算解析位图。3.4read函数作用从输入设备文件/dev/input/eventx中读取内核发送的input_event输入事件。四、完整带超详细注释源码// Linux输入子系统核心头文件定义输入事件、结构体、ioctl设备控制命令 #include linux/input.h // Linux系统调用基础头文件 #include sys/types.h #include sys/stat.h #include fcntl.h // 提供 open() 函数用于打开设备文件 #include sys/ioctl.h // 提供 ioctl() 函数设备控制读取硬件信息 #include stdio.h // 提供 printf() 打印函数 #include string.h // 提供 strcmp() 字符串比较函数判断 noblock 参数 #include unistd.h // 提供 read()、close() 函数 /* * 程序功能查询输入设备信息 实时读取输入事件 * 运行命令格式两种模式 * 1. 阻塞模式默认推荐 * sudo ./02_input_read /dev/input/eventx * 2. 非阻塞模式加 noblock 参数 * sudo ./02_input_read /dev/input/eventx noblock * 设备说明event0电源键event1键盘event2~4鼠标 */ int main(int argc, char **argv) { int fd; // 设备文件描述符打开设备后用于操作硬件 int err; // 存储 ioctl 函数返回值判断是否调用成功 int len; // 存储 read/ioctl 读取到的数据长度 int i, bit; // 双层循环计数器用于解析事件类型位图 unsigned char byte; // 存储事件位图的单个字节数据 struct input_id id; // 内核结构体存储设备硬件ID总线/厂商/产品/版本 unsigned int evbit[2]; // 存储事件类型位图2个int 8字节 64位 struct input_event event; // 内核标准结构体存储读取到的输入事件数据 // 事件类型名称数组与内核 EV_XXX 宏一一对应用于打印可读事件名 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 , // LED灯事件键盘大小写锁定灯 EV_SND , // 声音事件 NULL , EV_REP , // 按键重复事件长按键盘自动重复输入 EV_FF , // 力反馈事件 EV_PWR , // 电源事件 }; // 步骤1参数合法性检查 // 程序运行至少需要2个参数程序名 输入设备路径/dev/input/eventx if (argc 2) { printf(Usage: %s dev [noblock]\n, argv[0]); return -1; } // 步骤2打开设备阻塞/非阻塞模式分支 // 判断传入3个参数且第3个参数为 noblock → 非阻塞模式 if (argc 3 !strcmp(argv[2], noblock)) { // O_NONBLOCK非阻塞标志无事件时read不等待直接返回-1 fd open(argv[1], O_RDWR | O_NONBLOCK); } // 默认情况阻塞模式无事件时read卡住等待直到有事件触发 else { fd open(argv[1], O_RDWR); } // 打开设备失败判断无权限/设备不存在打印错误并退出 if (fd 0) { printf(open %s err\n, argv[1]); return -1; } // 步骤3读取并打印设备硬件ID信息 // EVIOCGID内核ioctl命令获取设备总线、厂商、产品、版本号 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 ); } // 步骤4读取并解析事件类型位图 // EVIOCGBIT(0, 大小)内核ioctl命令获取设备支持的事件类型位图 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]; // 第二层循环遍历当前字节的每一位0~7 for (bit 0; bit 8; bit) { // 位运算判断当前位1 → 设备支持该事件类型 if (byte (1bit)) { printf(%s , ev_names[i*8 bit]); } } } printf(\n); } // 步骤5无限循环实时读取输入事件 while (1) { // 从设备文件读取内核发送的输入事件存入event结构体 // 阻塞模式无事件 → 卡住等待 // 非阻塞模式无事件 → 立即返回-1实验现象read err -1 len read(fd, event, sizeof(event)); // 读取成功判断读取长度 输入事件结构体大小 if (len sizeof(event)) { // 打印事件信息type事件类型code按键编码value按键状态 // value1按下 value0松开 printf(get event: type 0x%x, code 0x%x, value 0x%x\n, event.type, event.code, event.value); } // 读取失败非阻塞模式无事件 或 设备异常 else { // 非阻塞模式下无事件时read返回-1打印此错误你的核心实验现象 printf(read err %d\n, len); } } // 关闭设备文件while(1)无限循环本行代码永远不会执行 close(fd); return 0; }五、代码逐模块精讲5.1 参数与模式判断模块通过strcmp判断是否传入noblock参数选择阻塞 / 非阻塞打开方式是两种模式的核心分支。5.2 设备信息查询模块使用ioctlEVIOCGID、EVIOCGBIT命令读取设备硬件信息与事件位图保留旧版核心逻辑。5.3 事件读取循环模块while(1)无限循环持续调用read读取事件阻塞模式无事件休眠不占用 CPU非阻塞模式无事件立即返回 - 1高速循环打印read err -1。六、编译与实操运行命令6.1 编译代码gcc 02_input_read.c -o 02_input_read6.2 查看系统输入设备cat /proc/bus/input/devices你的设备event0电源键event1键盘event2~4鼠标6.3 阻塞模式运行默认推荐sudo ./02_input_read /dev/input/event16.4 非阻塞模式运行sudo ./02_input_read /dev/input/event1 noblock七、真实实验现象全解析对应你的实操结果7.1 现象 1非阻塞模式疯狂打印read err -1read err -1 read err -1 ...无限刷屏完全正常不是程序 BUG原因开启O_NONBLOCK非阻塞模式无按键事件时read函数不等待直接返回-1while(1)死循环 → 程序高速打印read err -1。7.2 现象 2读取电源键设备/dev/input/event0bustype 0x19 vendor 0x0 product 0x0 version 0x0 support ev type: EV_SYN EV_KEY设备身份主板虚拟电源按钮无厂商、产品信息bustype 0x19电源总线类型支持事件仅EV_SYN同步、EV_KEY按键按下电源键才会触发事件运行模式阻塞模式无操作时程序卡住等待不打印任何信息。7.3 按键事件正常输出示例plaintextget event: type 0x1, code 0x1c, value 0x1 // 回车键按下 get event: type 0x0, code 0x0, value 0x0 // 同步事件 get event: type 0x1, code 0x1c, value 0x0 // 回车键松开八、常见问题与错误排查表格报错 / 现象原因解决方案read err -1疯狂刷屏非阻塞模式无事件正常现象改用阻塞模式或添加休眠降低刷屏程序卡住不动阻塞模式正常等待按下键盘 / 鼠标 / 电源键触发事件CtrlC 退出open xxx err无权限 / 设备路径错误加 sudo更换正确设备节点event1/event0编译报错 strcmp 未定义缺少头文件添加#include string.h九、核心知识点总结程序完整流程参数检查→打开设备→查询信息→解析位图→循环读事件read err -1是非阻塞模式的正常特征不是错误阻塞模式适合监听输入非阻塞模式适合多任务并发处理Linux 所有输入设备统一使用struct input_event传递事件操作硬件设备必须加sudo获取管理员权限。