使用/dev/uinput的简要介绍(含demo程序)【转】
转自:https://blog.csdn.net/zhongkunjia/article/details/75142699
uinput机制有2个很大的优点:
1) 不用自己写驱动(比如弄个红外遥控器、车载线控)。
2) 创建/dev/input/eventX节点,在用户态下向/dev/input/eventX写入事件,即可模拟键盘、鼠标等的事件输入。
操作流程:
1)打开UInput Device
2)设置UInput Device
3)写入设备信息
4)创建Input Device
5*)向Input Device发送Event
具体的demo代码如下:
//uinput-demo.c
- #include <stdio.h>
- #include <stdlib.h>
- #include <unistd.h>
- #include <fcntl.h>
- #include <string.h>
- #include <linux/uinput.h>
- #include <linux/input.h>
- #define KEY_CUSTOM_UP 0x20
- #define KEY_CUSTOM_DOWN 0x30
- static struct uinput_user_dev uinput_dev;
- static int uinput_fd;
- int creat_user_uinput(void);
- int report_key(unsigned int type, unsigned int keycode, unsigned int value);
- int main(int argc, char *argv[])
- {
- int ret = 0;
- ret = creat_user_uinput();
- if(ret < 0){
- printf("%s:%d\n", __func__, __LINE__);
- return -1;//error process.
- }
- sleep(10);// help you to 'hexdump -C /dev/input/event[X]' for test.
- report_key(EV_KEY, KEY_A, 1);// Report BUTTON A CLICK - PRESS event
- report_key(EV_KEY, KEY_A, 0);// Report BUTTON A CLICK - RELEASE event
- report_key(EV_KEY, KEY_CUSTOM_UP, 12);
- report_key(EV_KEY, KEY_CUSTOM_UP, 0);
- sleep(5);
- close(uinput_fd);
- return 0;
- }
- int creat_user_uinput(void)
- {
- int i;
- int ret = 0;
- uinput_fd = open("/dev/uinput", O_RDWR | O_NDELAY);
- if(uinput_fd < 0){
- printf("%s:%d\n", __func__, __LINE__);
- return -1;//error process.
- }
- //to set uinput dev
- memset(&uinput_dev, 0, sizeof(struct uinput_user_dev));
- snprintf(uinput_dev.name, UINPUT_MAX_NAME_SIZE, "uinput-custom-dev");
- uinput_dev.id.version = 1;
- uinput_dev.id.bustype = BUS_VIRTUAL;
- ioctl(uinput_fd, UI_SET_EVBIT, EV_SYN);
- ioctl(uinput_fd, UI_SET_EVBIT, EV_KEY);
- ioctl(uinput_fd, UI_SET_EVBIT, EV_MSC);
- for(i = 0; i < 256; i++){
- ioctl(uinput_fd, UI_SET_KEYBIT, i);
- }
- ioctl(uinput_fd, UI_SET_MSCBIT, KEY_CUSTOM_UP);
- ioctl(uinput_fd, UI_SET_MSCBIT, KEY_CUSTOM_DOWN);
- ret = write(uinput_fd, &uinput_dev, sizeof(struct uinput_user_dev));
- if(ret < 0){
- printf("%s:%d\n", __func__, __LINE__);
- return ret;//error process.
- }
- ret = ioctl(uinput_fd, UI_DEV_CREATE);
- if(ret < 0){
- printf("%s:%d\n", __func__, __LINE__);
- close(uinput_fd);
- return ret;//error process.
- }
- }
- int report_key(unsigned int type, unsigned int keycode, unsigned int value)
- {
- struct input_event key_event;
- int ret;
- memset(&key_event, 0, sizeof(struct input_event));
- gettimeofday(&key_event.time, NULL);
- key_event.type = type;
- key_event.code = keycode;
- key_event.value = value;
- ret = write(uinput_fd, &key_event, sizeof(struct input_event));
- if(ret < 0){
- printf("%s:%d\n", __func__, __LINE__);
- return ret;//error process.
- }
- gettimeofday(&key_event.time, NULL);
- key_event.type = EV_SYN;
- key_event.code = SYN_REPORT;
- key_event.value = 0;//event status sync
- ret = write(uinput_fd, &key_event, sizeof(struct input_event));
- if(ret < 0){
- printf("%s:%d\n", __func__, __LINE__);
- return ret;//error process.
- }
- return 0;
- }
编译方法:
root@ubuntu:/home/ubuntu/Documents# gcc uinput-demo.c -o uinput-demo
root@ubuntu:/home/ubuntu/Documents# ls
uinput-demo uinput-demo.c
root@ubuntu:/home/ubuntu/Documents# ./uinput-demo ------执行时要用root用户登入才行
这时,在开个终端,切换到:root@ubuntu:/dev/input#
执行hexdump -C event5 ------event5是创建uinput时生成的节点
10秒到后,显示结果:
root@ubuntu:/dev/input# hexdump -C event5
00000000 bd 5e 69 59 aa fd 05 00 01 00 1e 00 01 00 00 00 |.^iY............|
00000010 bd 5e 69 59 aa fd 05 00 00 00 00 00 00 00 00 00 |.^iY............|
00000020 bd 5e 69 59 d7 fd 05 00 01 00 1e 00 00 00 00 00 |.^iY............|
00000030 bd 5e 69 59 d7 fd 05 00 00 00 00 00 00 00 00 00 |.^iY............|
00000040 bd 5e 69 59 db fd 05 00 01 00 20 00 0c 00 00 00 |.^iY...... .....|
00000050 bd 5e 69 59 db fd 05 00 00 00 00 00 00 00 00 00 |.^iY............|
00000060 bd 5e 69 59 dd fd 05 00 01 00 20 00 00 00 00 00 |.^iY...... .....|
00000070 bd 5e 69 59 dd fd 05 00 00 00 00 00 00 00 00 00 |.^iY............|
adhexdump: event5: 没有那个设备
00000080
root@ubuntu:/dev/input#
也可以打开记事本,再次执行:
root@ubuntu:/home/ubuntu/Documents#
./uinput-demo
10秒到后,就会在记事本里输入:ad。
这样,就可以通过应用程序的执行模拟鼠标、键盘等的输入事件。
使用/dev/uinput的简要介绍(含demo程序)【转】的更多相关文章
- 浅析BMP位图文件结构(含Demo)
浅析BMP位图文件结构(含Demo) 作者:一点一滴的Beer http://beer.cnblogs.com/ 关于BMP位图格式在网上可以找到比较详细的相关文档,有兴趣的可以搜索标题为“BMP ...
- Android Debuggerd 简要介绍和源码分析(转载)
转载: http://dylangao.com/2014/05/16/android-debuggerd-%E7%AE%80%E8%A6%81%E4%BB%8B%E7%BB%8D%E5%92%8C%E ...
- WCF简要介绍
什么是WCF WCF的全称是:Windows Communication Foundation.从本质上来说,它是一套软件开发包,是微软公司推出的符合SOA思想的技术框架.WCF为程序员提供了丰富的功 ...
- 简要介绍BASE64、MD5、SHA、HMAC几种方法。
加密解密,曾经是我一个毕业设计的重要组件.在工作了多年以后回想当时那个加密.解密算法,实在是太单纯了. 言归正传,这里我们主要描述Java已经实现的一些加密解密算法,最后介绍数字证书. ...
- [转]Android系统Surface机制的SurfaceFlinger服务简要介绍和学习计划
转自:Android系统Surface机制的SurfaceFlinger服务简要介绍和学习计划 前面我们从Android应用程序与SurfaceFlinger服务的关系出发,从侧面简单学习了Surfa ...
- [转] Android资源管理框架(Asset Manager)简要介绍和学习计划
转自:http://blog.csdn.net/luoshengyang/article/details/8738877 Android应用程序主要由两部分内容组成:代码和资源.资源主要就是指那些与U ...
- Activity启动过程简要介绍
无论是通过点击应用程序图标来启动Activity,还是通过Activity内部调用startActivity接口来启动新的Activity,都要借助于应用程序框架层的ActivityManagerSe ...
- Android应用程序的Activity启动过程简要介绍和学习计划
文章转载至CSDN社区罗升阳的安卓之旅,原文地址:http://blog.csdn.net/luoshengyang/article/details/6685853 在Android系统中,Activ ...
- Dalvik虚拟机简要介绍和学习计划
文章转载至CSDN社区罗升阳的安卓之旅,原文地址:http://blog.csdn.net/luoshengyang/article/details/8852432 我们知道,Android应用程序是 ...
随机推荐
- Alpha版总结会议——班级派
一.开会的过程 在周一下午上课的最后20分钟内,我们组进行了“班级派”的alpha版的总结会议.首先进行的是分析目前的版本情况,每个人说了自己的进度,包括已经完成的以及即将要完成的.随后是分析前段时间 ...
- Python网络数据采集
一.正则表达式 * 表匹配0次或者多次 a*b* + 表至少一次 [ ] 匹配任意一个 ( ) 辨识一个编组 {m,n} m或者n 次 [^] 匹配任意不在中括号里的字符 | ...
- 2018软工实践—Beta冲刺(7)
队名 火箭少男100 组长博客 林燊大哥 作业博客 Beta 冲鸭鸭鸭! 成员冲刺阶段情况 林燊(组长) 过去两天完成了哪些任务 协调组内工作 整体软件测试 展示GitHub当日代码/文档签入记录(组 ...
- Beta阶段冲刺-1
1. 新成员 新加入成员,克克飞同学,任务是去弄公众号相关的部分. 队员 个人简介 博客地址 杨晨露 每天都在开会的PM http://www.cnblogs.com/ycll/ 游舒婷 每天都在装死 ...
- Prometheus 和 Grafana的简单学习
1. 下载 暂时不采用 docker化部署 prometheus下载地址 https://github.com/prometheus/prometheus/releases/ prometheus的e ...
- php常用几种设计模式的应用场景
1.单例设计模式 所谓单例模式,即在应用程序中最多只有该类的一个实例存在,一旦创建,就会一直存在于内存中! 单例设计模式常应用于数据库类设计,采用单例模式,只连接一次数据库,防止打开多个数据库连接. ...
- BZOJ1050 HAOI2006旅行(最小生成树+LCT)
暴力枚举路径中权值最小边是哪个,然后求出边权不小于它的边的最小生成树,即可保证该路径上最大值最小.暴力当然可以过,这里使用LCT维护.注意数据中有自环. #include<iostream> ...
- jquery 绑定文本即时查询功能
bindFilterFunc: function () { if ("\v" == "v") { // IE only ...
- 解题:BZOJ 2989 数列
题面 学习二进制分组 题目本身可以看成二维平面上的问题,转成切比雪夫距离后就是矩形和了 二进制分组是将每个修改添加到末尾,然后从后往前二进制下进位合并,这样最多同时有$\log n$组,每个修改只会被 ...
- cookie工具包
package com.taotao.common.utils; import java.io.UnsupportedEncodingException; import java.net.URLDec ...