Linux 有自己的 input 子系统,可以统一管理鼠标和键盘事件。

如果想模拟键盘事件,但是系统没有键盘设备该如何是好?

基于输入子系统实现的 input 可以方便的在用户空间模拟鼠标和键盘事件。

也可以自己做一个字符设备接收用户输入,根据输入,投递 input 事件。

还有一种方式就是直接往evnent 里写入数据,都可以达到控制鼠标键盘的功能。---没有键盘设备的话,向哪一个event里面写?

好文:

http://blog.chinaunix.net/uid-23381466-id-3883164.html

https://blog.csdn.net/neiloid/article/details/7893732

相关书籍:

https://blog.csdn.net/huaweimember/article/details/51001672

http://baijiahao.baidu.com/s?id=1601404998850288910&wfr=spider&for=pc


weekend 探索成功, 使用uinput驱动来实现 linux系统用户空间模拟按键事件:

    这几篇真是太棒了!:

http://sipgreen.iteye.com/blog/1774676

    http://sipgreen.iteye.com/blog/1774676

    http://blog.sina.com.cn/s/blog_602f87700100llew.html

    https://www.cnblogs.com/myblesh/articles/2367648.html

    https://www.cnblogs.com/myblesh/articles/2367648.html

   https://www.kernel.org/doc/html/v4.12/input/uinput.html ----linux官网上关于uinput模块的介绍

  首先在内核中添加uinput模块(也可以将这个模块直接编译进内核里面),然后使用wriet系统函数向文件中写入键盘事件!

linux2.6.x内核提供了uinput驱动,它可以帮助用户将数据(包括用户输入的键盘或者鼠标或者触摸板数据)注入到Linux内核,这对于编写用户自定义的输入设备的应用程序是非常有用。

驱动程序使用/dev/uinput的设备,将数据发送到内核空间,内核将数据发送到X-Windows或者shell终端。此功能可用于所有涉及图形用户的输入。

uinput驱动在许多linux内核里均被配置为可加载的模块,你可以通过$ modprobe uinput 命令加载luinput驱动。

$ lsmod 命令列出linux系统已经加载的所有驱动。你应该可以看到“uinput”驱动也在这个列表中。但是如果你是在kernel里把uinput设置成编译进内核,而不是编译成模块的话,那么你通过lsmod是看不到uinput模块的。但你在/dev下会看到uinput的设备结点。后面你可以直接打开此结点来向uinput发送数据。

相关头文件:

#include <sys/types.h>  #include <sys/stat.h>  #include <fcntl.h>  #include <linux/input.h>  #include <linux/uinput.h>  #include <sys/time.h>  #include <unistd.h>  #include <errno.h> 

step1:打开uinput设备并设置设备参数:

uinp_fd = open(“/dev/uinput”,O_WRONLY|O_NDELAY);

If(uinp_fd < 0){

printf(“unable to open /dev/uinput\n”);//也有可能在/dev/input/uniput

Return -1;

}

成功打开/dev/uinput设备之后,需要通过驱动的ioctl函数(点击打开链接)来配置uinput设备的参数,包括鼠标参数、键盘参数等,即设置Input Device关心或者说会产生的消息:

ioctl(uinp_fd ,UI_SET_EVBIT,EV_KEY); //设置设备所支持的动作,#defineEV_KEY 0x01 // 按下键

ioctl(uinp_fd ,UI_SET_EVBIT,EV_REP); //设置设备所支持的动作,#defineEV_KEY 0x02 // 释放

   (//EV_KEY和EV_REP来告诉uinput驱动是包含键值的键盘事件。)

ioctl(uinp_fd , UI_SET_RELBIT, REL_X); //包含鼠标事件

ioctl(uinp_fd , UI_SET_RELBIT, REL_Y);

ioctl(uinp_fd , UI_SET_EVBIT, EV_ABS);

ioctl(uinp_fd , UI_SET_ABSBIT, ABS_X);

ioctl(uinp_fd , UI_SET_ABSBIT, ABS_Y);

ioctl(uinp_fd , UI_SET_ABSBIT, ABS_PRESSURE);

for(i = 0; i < 256; i++){//  ---------------------???????

ioctl(uinp_fd , UI_SET_KEYBIT, i);

}

setp2:创建设备并写入至input子系统

struct  uinput_user_dev  uinput;

uinput.id.version = 4;
uinput.id.bustype = BUS_USB;
uinput.absmin[ABS_X] = 0;
uinput.absmax[ABS_X] = 65535; //sam 把屏幕设为0-65535
uinput.absmin[ABS_Y] = 0;
uinput.absmax[ABS_Y] = 65535;
uinput.absmin[ABS_PRESSURE] = 0;
uinput.absmax[ABS_PRESSURE] = 0xfff;

ret = write( uinp_fd , &uinput, sizeof(uinput) );

ioctl(uinput_fd, UI_DEV_CREATE)

如果在代码中,上面的执行完成之后,马上执行下面的step3的话,会导致没有任何效果,我加掩饰100ms,好了!原因还未知!

step3: 自己构造input事件并写入

所有来自用户进程的事件都会通过结构体“struct input_event”(在/usr/include/linux/input.h中定义)传送给内核空间。

可以通过下面的代码产生键盘事件:

event.type = EV_KEY;

event.code = KEY_ENTER;

event.value = 1;//1表示按下,0表示抬起

gettimeofday(&event.time, NULL);

__u16 type; //类型,在下面有定义
__u16 code; //要模拟成什么按键

__s32 value;//是按下还是释放

write(uinp_fd, &event, sizeof(event));

event.type = EV_SYN;

event.code = SYN_REPORT;

event.value = 0;

gettimeofday(&event.time, NULL);

write(uinput_fd, &event, sizeof(event));//发送同步信号

上面的代码将要发送一个enter键给内核。该键值最终会通过内核发送给用户空间的应用程序。所有的按键定义在“/usr/include/linux/input.h”中;注意,当发送按键事件之后,要马上发送一个同步事件,否则内核不会马上处理你的按键事件!


my code as follows: 注意需要修改/dev/uinput文件的权限

void testevent()
{
//int uinp_fd = open("/dev/uinput",O_WRONLY|O_NDELAY);O_RDWR
int uinp_fd = open("/dev/uinput",O_RDWR);
if (uinp_fd < 0){
qDebug()<<"unable to open /dev/uinput";
return;
}
if(ioctl(uinp_fd ,UI_SET_EVBIT,EV_KEY)<0) //设置设备所支持的动作,#defineEV_KEY 0x01 // 按下键
{
qDebug()<<"unable to set EV_KEY";
return;
}
if(ioctl(uinp_fd ,UI_SET_EVBIT,EV_REP)<0) //设置设备所支持的动作,#defineEV_KEY 0x02 // 释放
{
qDebug()<<"unable to set EV_REP";
return;
} for(int i = 0; i < 256; i++){ //---------------------??????? ioctl(uinp_fd , UI_SET_KEYBIT, i);
} //创建设备并写入至input子系统
struct uinput_user_dev uinput;
memset(&uinput,0,sizeof(uinput));
uinput.id.version = 4;
uinput.id.bustype = BUS_USB;
strncpy(uinput.name,"virtual device a",UINPUT_MAX_NAME_SIZE);
int ret = write( uinp_fd , &uinput, sizeof(uinput) );
if(ret<0)
{
qDebug()<<"unable to write(uinp_fd , &uinput, sizeof(uinput)";
return;
}
int err = ioctl(uinp_fd, UI_DEV_CREATE);
if(err < 0)
{
qDebug()<<"unable to creat device";
return;
}
//construct input_event and send it
struct input_event event;
unsigned int key_code = KEY_TAB;
usleep(100000);//important!!!!!!!!!!!!!!!!!!!!!!!!!! don't know why
//unsigned int key_code = KEY_CAPSLOCK;
event.type = EV_KEY;
event.code = key_code;
event.value = 1; //1 means pressed signal
gettimeofday(&event.time, NULL);
if (write(uinp_fd, &event, sizeof(event)) < 0) {
qDebug()<<"unable to write key event";
return;
}
gettimeofday(&event.time, NULL);
event.value = 0; ////////////////send sync signal///////////////////////////////
event.type = EV_SYN;
event.code = SYN_REPORT;
event.value = 0;
gettimeofday(&event.time, NULL);
if( write(uinp_fd, &event, sizeof(event))<0 )//发送同步信号
{
qDebug()<<"unable to write sync event";
return;
} close(uinp_fd); qDebug()<<"UmProg::testQevent() end:";
return;
}

linux udev学习:



linux系统下,在用户空间应用程序中模拟发送系统键盘事件的更多相关文章

  1. linux系统下的用户文件句柄数限制

    linux系统下的用户文件句柄数限制 文章来源:企鹅号 为什么要修改用户打开的文件数 系统默认单个进程可以打开1024个文件,对于一些应用如tomcat.oracle等,运行时经常open成千上万个文 ...

  2. 05 Linux系统下的用户以及用户权限管理(权限管理介绍、用户管理、常见命令介绍)

    这一节我们介绍Linux的用户以及权限管理的前半段,包括:1.权限管理介绍: 2.用户管理: 3.常见命令 权限管理介绍 权限管理: 为了访问计算机资源,我们需要对其进行授权才能访问,根据什么东西来进 ...

  3. (转)linux 系统下虚拟用户的作用

    原文:http://blog.csdn.net/luoshao20120430/article/details/16900653            http://blog.csdn.net/u01 ...

  4. 记录linux系统下所有用户的操作信息

    在日常运维中,我们需要清楚服务器上每个用户登录后都做了哪些操作,我们需要记录下每个用户的操作命令.下面的内容设置可以实现在Linux下所有用户,不管是远程还是本地登陆,在本机的所有操作都会记录下来,并 ...

  5. linux系统下root用户和普通用户的时区不一致

    1. 发现这个问题是在root下执行 date -R 和 普通用户下执行 date -R,发现时区不一致 2. 在一些linux机器下,发现是一致的 3. 什么原因呢?找了半天,最后发现 时区一致的机 ...

  6. Linux系统下C语言如何调用scalapack中的函数

    在并行计算中经常需要调用scalapck(并行化的lapack)函数库里面的函数进行编程,这里简单介绍在C语言如何调用scalapck中的矩阵向量乘的函数. 注意:scalapack中的函数是用for ...

  7. 在C#程序中模拟发送键盘按键消息

    using System.Runtime.InteropServices; 引入键盘事件函数 [DllImport("user32.dll")]public static exte ...

  8. linux系统下memcached启动正常但程序无法连接的问题解决

    在虚拟机linux安装好memcached之后,试着用java程序连接一下memcached的服务端,但却出现了以下错误 com.schooner.MemCached.SchoonerSockIOPo ...

  9. Linux系统下超级用户密码的修改

    1)重启系统:在虚拟机刚启动界面,不停地按上下键,停止系统的自动引导(界面底部有提示) 2) 按 e 进入编辑模式 3) 编辑内容如下:完成后按Ctrl+x    (具体编辑内容为下图:删除倒数第三行 ...

随机推荐

  1. Hello 2019题解

    Hello 2019题解 题解 CF1097A [Gennady and a Card Game] map大法好qwq 枚举每一个的第\(1,2\)位判是否与给定的重复即可 # include < ...

  2. 使用DOS命令将类库封装成dll

    1.Windows键+R.输入cmd进入DOS 2.使用 cd  加路径找到需要封装成dll的类库文件 3.csc /target:library /out:dll的名字.DLL 需要封装的cs文件

  3. CF1174B Ehab Is an Odd Person(排序+结论)

    做法 一个显然的结论就是如果至少有一个奇数和一个偶数,那么是可以随意调整的,也就是升序排序 否则不可以进行任何操作 Code #include<bits/stdc++.h> using n ...

  4. js 将网络图片格式转为base64 canvas 跨域

    function getBase64Image(img) { var canvas = document.createElement("canvas"); canvas.width ...

  5. kvm 学习(三)存储池

    创建kvm存储池 1.查看系统已经存储的存储池 [root@runstone ~ ::]#virsh pool-list Name State Autostart ------------------ ...

  6. [PKUSC2018]真实排名——线段树+组合数

    题目链接: [PKUSC2018]真实排名 对于每个数$val$分两种情况讨论: 1.当$val$不翻倍时,那么可以翻倍的是权值比$\frac{val-1}{2}$小的和大于等于$val$的. 2.当 ...

  7. 关于SQLServer 中行列互转的实例说明

    这几天在做一个招标系统中审批模块,其中关于报价信息这块,用到了pivot和unpivot来实现数据的行列互转,下面简单介绍一下,实际案例,便于回忆和记录相关的条件下使用的情况.pivot 与 unpi ...

  8. Java 多线程编程(锁优化)

    转:https://mp.weixin.qq.com/s/lDuguEhuWiLY8ofBRy3tZA 并发环境下进行编程时,需要使用锁机制来同步多线程间的操作,保证共享资源的互斥访问. 加锁会带来性 ...

  9. java之map遍历

    java开发中常常会用到遍历,所以下边就列举四种map的遍历方法. public class testMap { public static void main(String[] args) { Ma ...

  10. php如何生成 uuid(总结)

    php如何生成 uuid(总结) 一.总结 一句话总结: UUID的全拼为“Universally Unique Identifier”,可以译为“通用唯一识别码”. UUID是指在一台机器上生成的数 ...