Определить устройство событий клавиатуры Linux

Этот другой вопрос, Доступ к ключам с устройства ввода Linux

предоставил работающий код C о том, как распознать нажатие клавиши-модификатора в текстовой консоли Linux, когда другие клавиши не нажимаются.

Однако, чтобы применить трюк, вы должны знать конкретный файл /dev/input/event* который регистрирует события клавиатуры.

Как определить имя этого файла?

2 ответа

Я предпочитаю следующее решение, потому что оно анализирует файл устройств внутри.
Если мы проверим содержимое /proc/bus/input/devices, мы найдем что-то вроде этого:

# cat /proc/bus/input/devices
I: Bus=0019 Vendor=0002 Product=0001 Version=0100
N: Name="vmouse"
P: Phys=vmouse/input0
S: Sysfs=/devices/virtual/input/input0
U: Uniq=
H: Handlers=mouse0 event0
B: PROP=0
B: EV=7
B: KEY=70400 0 0 0 0 0 0 0 0
B: REL=143

I: Bus=0019 Vendor=0001 Product=0001 Version=0100
N: Name="sunxi-ths"
P: Phys=sunxiths/input0
S: Sysfs=/devices/virtual/input/input3
U: Uniq=
H: Handlers=event3
B: PROP=0
B: EV=9
B: ABS=100 0

I: Bus=0003 Vendor=1220 Product=0008 Version=0100
N: Name="HID 1220:0008"
P: Phys=usb-sunxi-ohci-1/input0
S: Sysfs=/devices/platform/sunxi-ohci.3/usb7/7-1/7-1:1.0/input/input6
U: Uniq=
H: Handlers=sysrq kbd event1
B: PROP=0
B: EV=120013
B: KEY=10000 7 ff9f207a c14057ff febeffdf ffefffff ffffffff fffffffe
B: MSC=10
B: LED=1f

I: Bus=0003 Vendor=1220 Product=0008 Version=0100
N: Name="HID 1220:0008"
P: Phys=usb-sunxi-ohci-1/input1
S: Sysfs=/devices/platform/sunxi-ohci.3/usb7/7-1/7-1:1.1/input/input7
U: Uniq=
H: Handlers=kbd mouse1 event2
B: PROP=0
B: EV=1f
B: KEY=4837fff 72ff32d bf544446 0 0 1f0001 20f90 8b17c000 677bfa d941dfed 9ed680 4400 0 10000002
B: REL=143
B: ABS=1 0
B: MSC=10

Поэтому единственное, что нам нужно, это найти номер устройства события в блоке, где появилась строка "EV=120013":

#include <fcntl.h>
#include <stdio.h>
#include <string>
#include <string.h>
#include <unistd.h>

using namespace std;

string getInputDeviceName() {
    int rd;
    std::string devName;
    const char* pdevsName = "/proc/bus/input/devices";

    int devsFile = open(pdevsName, O_RDONLY);
    if (devsFile == -1) {
        printf("[ERR] Open input devices file: '%s' is FAILED\n", pdevsName);
    }
    else {
        char devs[2048];

        if ((rd = read(devsFile, devs, sizeof(devs) - 1)) < 6) {
            printf("[ERR] Wrong size was read from devs file\n");
        }
        else {
            devs[rd] = 0;

            char *pHandlers, *pEV = devs;
            do {
                pHandlers = strstr(pEV, "Handlers=");
                pEV = strstr(pHandlers, "EV=");
            }
            while (pHandlers && pEV && 0 != strncmp(pEV + 3, "120013", 6));

            if (pHandlers && pEV) {
                char* pevent = strstr(pHandlers, "event");
                if (pevent) {
                    devName = string("/dev/input/event");
                    devName.push_back(pevent[5]);
                }
                else {
                    printf("[ERR] Abnormal keyboard event device\n");
                }
            }
            else {
                printf("[ERR] Keyboard event device not found\n");
            }
        }
    }

    return devName;
}

ПРИМЕЧАНИЕ 1. Этот код работает на устройствах событий с номерами меньше 10.
ПРИМЕЧАНИЕ 2. Пожалуйста, проверьте размер буфера "devs", в вашей системе может не хватить 2048 байт.

У меня была точно такая же проблема. Вот мое решение:

#include <iostream>
#include <string>
#include <stdio.h>

static const std::string COMMAND_GET_INPUT_DEVICE_EVENT_NUMBER =
    "grep -E 'Handlers|EV=' /proc/bus/input/devices |"
    "grep -B1 'EV=120013' |"
    "grep -Eo 'event[0-9]+' |"
    "grep -Eo '[0-9]+' |"
    "tr -d '\n'";

std::string executeCommand(const char *cmd) {
  FILE *pipe = popen(cmd, "r");
  char buffer[128];
  std::string result = "";
  while (!feof(pipe))
    if (fgets(buffer, 128, pipe) != NULL)
      result += buffer;
  pclose(pipe);
  return result;
}

std::string getInputDevicePath() {
  return "/dev/input/event" +
         executeCommand(COMMAND_GET_INPUT_DEVICE_EVENT_NUMBER.c_str());
}

int main() {
  std::cout << getInputDevicePath() << std::endl;
  return 0;
}
Другие вопросы по тегам