在 linux 下使用 ioctl 重新映射键盘
Remap a keyboard with ioctl under linux
我实际上正在尝试编写一个小程序来捕获来自 linux 下特定 USB 键盘的全局键盘输入。
我正在用这段代码进行测试:
#include <stdlib.h>
#include <unistd.h>
#include <fcntl.h>
#include <errno.h>
#include <linux/input.h>
#include <string.h>
#include <stdio.h>
static const char *const evval[3] = {
"RELEASED",
"PRESSED ",
"REPEATED"
};
int main(void)
{
const char *dev = "/dev/input/event2";
struct input_event ev;
ssize_t n;
int fd;
char name[256]= "Unknown";
// int codes[2];
// codes[0] = 58; /* M keycap */
// codes[1] = 49; /* assign to N */
fd = open(dev, O_RDONLY);
if (fd == -1) {
fprintf(stderr, "Cannot open %s: %s.\n", dev, strerror(errno));
return EXIT_FAILURE;
}
if(ioctl(fd, EVIOCGNAME(sizeof(name)), name) > 0)
{
printf("The device on %s says its name is '%s'\n", dev, name);
}
/*
int err = ioctl(fd, EVIOCSKEYCODE, codes);
if (err) {
perror("evdev ioctl");
}*/
while (1) {
n = read(fd, &ev, sizeof ev);
if (n == (ssize_t)-1) {
if (errno == EINTR)
continue;
else
break;
} else
if (n != sizeof ev) {
errno = EIO;
break;
}
if (ev.type == EV_KEY && ev.value >= 0 && ev.value <= 2)
printf("%s 0x%04x (%d)\n", evval[ev.value], (int)ev.code, (int)ev.code);
}
fflush(stdout);
fprintf(stderr, "%s.\n", strerror(errno));
return EXIT_FAILURE;
}
问题是我不知道如何通过其他方式更改某些输入键。我尝试通过更改事件代码在当前红色事件上调用 write(),发送的密钥仍然是前一个,我尝试将 ioctl 与 EVIOCSKEYCODE 一起使用,但调用失败并出现 "invalid argument" 错误(我是不确定是否正确调用它)。
如何正确更改输出的密钥?
使用 EVIOCGRAB ioctl 获取输入设备,以便通过读取事件来使用它们。通常(未抓取)事件在您阅读时不会被消耗。 ioctl 有一个附加参数,(int)1
用于抓取,(int)0
用于释放。
要重新注入任何事件,只需将它们写入 uinput
设备即可。见例如。一个mouse example here。 (事件结构是同一类型,你只需要先写一个struct uinput_user_dev
结构到uinput
设备,描述你的新输入设备(它提供映射的事件)。)
换句话说,您不重新映射:您消费并转发。
我实际上正在尝试编写一个小程序来捕获来自 linux 下特定 USB 键盘的全局键盘输入。
我正在用这段代码进行测试:
#include <stdlib.h>
#include <unistd.h>
#include <fcntl.h>
#include <errno.h>
#include <linux/input.h>
#include <string.h>
#include <stdio.h>
static const char *const evval[3] = {
"RELEASED",
"PRESSED ",
"REPEATED"
};
int main(void)
{
const char *dev = "/dev/input/event2";
struct input_event ev;
ssize_t n;
int fd;
char name[256]= "Unknown";
// int codes[2];
// codes[0] = 58; /* M keycap */
// codes[1] = 49; /* assign to N */
fd = open(dev, O_RDONLY);
if (fd == -1) {
fprintf(stderr, "Cannot open %s: %s.\n", dev, strerror(errno));
return EXIT_FAILURE;
}
if(ioctl(fd, EVIOCGNAME(sizeof(name)), name) > 0)
{
printf("The device on %s says its name is '%s'\n", dev, name);
}
/*
int err = ioctl(fd, EVIOCSKEYCODE, codes);
if (err) {
perror("evdev ioctl");
}*/
while (1) {
n = read(fd, &ev, sizeof ev);
if (n == (ssize_t)-1) {
if (errno == EINTR)
continue;
else
break;
} else
if (n != sizeof ev) {
errno = EIO;
break;
}
if (ev.type == EV_KEY && ev.value >= 0 && ev.value <= 2)
printf("%s 0x%04x (%d)\n", evval[ev.value], (int)ev.code, (int)ev.code);
}
fflush(stdout);
fprintf(stderr, "%s.\n", strerror(errno));
return EXIT_FAILURE;
}
问题是我不知道如何通过其他方式更改某些输入键。我尝试通过更改事件代码在当前红色事件上调用 write(),发送的密钥仍然是前一个,我尝试将 ioctl 与 EVIOCSKEYCODE 一起使用,但调用失败并出现 "invalid argument" 错误(我是不确定是否正确调用它)。
如何正确更改输出的密钥?
使用 EVIOCGRAB ioctl 获取输入设备,以便通过读取事件来使用它们。通常(未抓取)事件在您阅读时不会被消耗。 ioctl 有一个附加参数,(int)1
用于抓取,(int)0
用于释放。
要重新注入任何事件,只需将它们写入 uinput
设备即可。见例如。一个mouse example here。 (事件结构是同一类型,你只需要先写一个struct uinput_user_dev
结构到uinput
设备,描述你的新输入设备(它提供映射的事件)。)
换句话说,您不重新映射:您消费并转发。