我一直在使用以下 C 代码来尝试在 CentOS 6.0 机器上模拟击键:
#include <stdio.h>
#include <string.h>
#include <fcntl.h>
#include <unistd.h>
#inlcude <linux/input.h>
#include <linux/uinput.h>
#include <sys/time.h>
static int fd = -1;
struct uinput_user_dev uidev;
struct input_event event;
int main()
{
int i;
fd = open("/dev/uinput", O_WRONLY | O_NONBLOCK);
memset(&uidev, 0, sizeof(uidev));
snrpintf(uidev.name, UINPUT_MAX_NAME_SIZE, "uinput-kbd");
uidev.id.version = 1;
uidev.id.vendor = 0x1;
uidev.id.product = 0x1;
uidev.id.bustype = BUS_USB;
ioctl(fd, UI_SET_EVBIT, EV_KEY);
for(i = 0; i < 256; i++)
{
ioctl(fd, UI_SET_KEYBIT, i);
}
ioctl(fd, UI_SET_EVBIT, EV_SYN);
write(fd, &uidev, sizeof(uidev));
ioctl(fd, UI_DEV_CREATE));
memset(&event, 0, sizeof(event));
gettimeofday(&event.time, NULL);
event.type = EV_KEY;
event.code = KEY_1;
event.value = 1;
write(fd, &event, sizeof(event));
event.type = EV_SYN;
event.code = SYN_REPORT;
event.value = 0;
write(fd, &event, sizeof(event));
memset(&event, 0, sizeof(event));
gettimeofday(&event.time, NULL);
event.type = EV_KEY;
event.code = KEY_1;
event.value = 0;
write(fd, &event, sizeof(event));
event.type = EV_SYN;
event.code = SYN_REPORT;
event.value = 0;
write(fd, &event, sizeof(event));
ioctl(fd, UI_DEV_DESTROY);
close(fd);
return 0;
}
如果我是正确的,这段代码应该在机器上创建一个虚拟输入设备,然后按该设备上的“1”键。当我执行代码时,它似乎运行没有任何问题(我没有在我的示例代码中包含检查以确保正在创建设备和正在编写击键等的代码,因为它会太长了),但我看不到任何实际击键的迹象。
我的印象是,如果我在直接登录到机器时从终端窗口运行它,我应该会在运行它的终端窗口上看到一个“1”字符。如果我通过 ssh 登录机器并以这种方式运行,则击键应该在机器上而不是 ssh 会话上注册。但在这两种情况下我都没有得到任何东西。
我误解了这段代码的目的吗?我做错了吗?还是我需要添加更多才能正确模拟击键?