Захват HID клавиатуры события

Код ниже работает нормально только для одного устройства ввода. К сожалению, мне нужно захватить около 12 различных устройств HID (считывателей RFID), поэтому я хотел бы знать, знает ли кто-нибудь, как адаптировать код для работы с 12 различными входами?

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <errno.h>
#include <fcntl.h>
#include <dirent.h>
#include <linux/input.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/select.h>
#include <sys/time.h>
#include <termios.h>
#include <signal.h>

int main(int argc, char* argv[])
{
struct input_event ev[64];
int fevdev = -1;
int result = 0;
int size = sizeof(struct input_event);
int rd;
int value;
char name[256] = "Unknown";
char *device = "/dev/input/event3";fevdev = open(device, O_RDONLY);
if (fevdev == -1) {
printf("Failed to open event device.\n");
exit(1);
}

result = ioctl(fevdev, EVIOCGNAME(sizeof(name)), name);
printf ("Reading From : %s (%s)\n", device, name);

printf("Getting exclusive access: ");
result = ioctl(fevdev, EVIOCGRAB, 1);
printf("%s\n", (result == 0) ? "SUCCESS" : "FAILURE");

while (1)
{
if ((rd = read(fevdev, ev, size * 64)) < size) {
break;
}

value = ev[0].value;

if (value != ' ' && ev[1].value == 1 && ev[1].type == 1) {
printf ("Code[%d]\n", (ev[1].code));
}
}

printf("Exiting.\n");
result = ioctl(fevdev, EVIOCGRAB, 1);
close(fevdev);
return 0;
}

2

Решение

Вызов open() для каждого устройства, а затем использовать select() (или же (e)poll()) для отслеживания всех файловых дескрипторов вместе, чтобы вы могли определить, на каких устройствах есть данные, доступные для последующего read() в любой момент.

Обновить: Например:

struct event_device
{
char *device;
int fd;
};

int main(int argc, char* argv[])
{
struct input_event ev[64];
int numevents;
int result = 0;
int size = sizeof(struct input_event);
int rd;
char name[256];
char* device[12];
event_device evdevs[12], *evdev;
int numevdevs = 0;
fd_set fds;
int maxfd;

device[0] = "/dev/input/event3";
device[1] = "/dev/input/event4";
// and so on...

for (int i = 0; i < 12; ++i) {
evdev = &evdevs[numevdevs];

evdev->device = device[i];
evdev->fd = open(evdev->device, O_RDONLY);
if (evdev->fd == -1) {
printf("Failed to open event device: %s.\n", evdev->device);
continue;
}
++numevdevs;

memset(name, 0, sizeof(name));
result = ioctl(evdev->fd, EVIOCGNAME(sizeof(name)), name);
printf ("Reading From : %s (%s)\n", evdev->device, name);

printf("Getting exclusive access: ");
result = ioctl(evdev->fd, EVIOCGRAB, 1);
printf("%s\n", (result == 0) ? "SUCCESS" : "FAILURE");
}

if (numevdevs == 0) {
exit(1);
}

while (1)
{
FD_ZERO(&fds);
maxfd = -1;

for (int i = 0; i < numevdevs; ++i) {
evdev = &evdevs[i];
FD_SET(evdev->fd, &fds);
if (maxfd < evdev->fd) maxfd = evdev->fd;
}

result = select(maxfd+1, &fds, NULL, NULL, NULL);
if (result == -1) {
break;
}

for (int i = 0; i < numevdevs; ++i) {
evdev = &evdevs[i];

if (!FD_ISSET(evdev->fd, &fds)) {
continue;
}

if ((rd = read(evdev->fd, ev, size * 64)) < size) {
continue;
}

numevents = rd / size;
for (int j = 0; j < numevents; ++j) {
printf ("%s: Type[%d] Code[%d] Value[%d]\n", evdev->device, ev[j].type, ev[j].code, ev[j].value);
}
}
}

printf("Exiting.\n");

for (int i = 0; i < numevdevs; ++i) {
evdev = &evdevs[i];
result = ioctl(evdev->fd, EVIOCGRAB, 0);
close(evdev->);
}

return 0;
}
2

Другие решения

Других решений пока нет …

По вопросам рекламы [email protected]