You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
823 lines
26 KiB
823 lines
26 KiB
#define _POSIX_C_SOURCE 200809L
|
|
#define _DEFAULT_SOURCE
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <errno.h>
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <fcntl.h>
|
|
#include <dirent.h>
|
|
#include <unistd.h>
|
|
#include <signal.h>
|
|
#include <sys/wait.h>
|
|
#include <linux/input.h>
|
|
#include <sys/epoll.h>
|
|
#include <sys/inotify.h>
|
|
#include <wordexp.h>
|
|
#include <ctype.h>
|
|
|
|
/* Value defines */
|
|
#define FILE_NAME_MAX_LENGTH 255
|
|
#define KEY_BUFFER_SIZE 16
|
|
|
|
/* ANSI colors escape codes */
|
|
#define ANSI_COLOR_RED "\x1b[31m"
|
|
#define ANSI_COLOR_GREEN "\x1b[32m"
|
|
#define ANSI_COLOR_YELLOW "\x1b[33m"
|
|
#define ANSI_COLOR_BLUE "\x1b[34m"
|
|
#define ANSI_COLOR_MAGENTA "\x1b[35m"
|
|
#define ANSI_COLOR_CYAN "\x1b[36m"
|
|
#define ANSI_COLOR_RESET "\x1b[0m"
|
|
|
|
/* Macro functions */
|
|
#define yellow(str) (ANSI_COLOR_YELLOW str ANSI_COLOR_RESET)
|
|
#define green(str) (ANSI_COLOR_GREEN str ANSI_COLOR_RESET)
|
|
#define red(str) (ANSI_COLOR_RED str ANSI_COLOR_RESET)
|
|
#define test_bit(yalv, abs_b) ((((char *)abs_b)[yalv/8] & (1<<yalv%8)) > 0)
|
|
#define die(str) {perror(ANSI_COLOR_RED str); fputs(ANSI_COLOR_RESET, stderr); exit(errno);}
|
|
#define array_size(val) (val ? sizeof(val)/sizeof(val[0]) : 0)
|
|
#define array_size_const(val) ((int)(sizeof(val)/sizeof(val[0])))
|
|
|
|
#define EVENT_SIZE (sizeof(struct inotify_event))
|
|
#define EVENT_BUF_LEN (1024*(EVENT_SIZE+16))
|
|
|
|
const char *config_paths[] = {
|
|
"$XDG_CONFIG_HOME/hkd/config",
|
|
"$HOME/.config/hkd/config",
|
|
"/etc/hkd/config",
|
|
};
|
|
|
|
const struct {
|
|
const char *const name;
|
|
const unsigned short value;
|
|
} key_conversion_table[] =
|
|
{{"ESC", KEY_ESC}, {"1", KEY_1}, {"2", KEY_2}, {"3", KEY_3},
|
|
{"4", KEY_4}, {"5", KEY_5}, {"6", KEY_6}, {"7", KEY_7},
|
|
{"8", KEY_8}, {"9", KEY_9}, {"0", KEY_0}, {"MINUS", KEY_MINUS},
|
|
{"EQUAL", KEY_EQUAL}, {"BACKSPACE", KEY_BACKSPACE}, {"TAB", KEY_TAB},
|
|
{"Q", KEY_Q}, {"W", KEY_W}, {"E", KEY_E}, {"R", KEY_R},
|
|
{"T", KEY_T}, {"Y", KEY_Y}, {"U", KEY_U}, {"I", KEY_I},
|
|
{"O", KEY_O}, {"P", KEY_P}, {"LEFTBRACE", KEY_LEFTBRACE},
|
|
{"RIGHTBRACE", KEY_RIGHTBRACE}, {"ENTER", KEY_ENTER}, {"LEFTCTRL", KEY_LEFTCTRL},
|
|
{"A", KEY_A}, {"S", KEY_S}, {"D", KEY_D}, {"F", KEY_F},
|
|
{"G", KEY_G}, {"H", KEY_H}, {"J", KEY_J}, {"K", KEY_K},
|
|
{"L", KEY_L}, {"SEMICOLON", KEY_SEMICOLON}, {"APOSTROPHE", KEY_APOSTROPHE},
|
|
{"GRAVE", KEY_GRAVE}, {"LEFTSHIFT", KEY_LEFTSHIFT}, {"BACKSLASH", KEY_BACKSLASH},
|
|
{"Z", KEY_Z}, {"X", KEY_X}, {"C", KEY_C}, {"V", KEY_V},
|
|
{"B", KEY_B}, {"N", KEY_N}, {"M", KEY_M}, {"COMMA", KEY_COMMA},
|
|
{"DOT", KEY_DOT}, {"SLASH", KEY_SLASH}, {"RIGHTSHIFT", KEY_RIGHTSHIFT},
|
|
{"KPASTERISK", KEY_KPASTERISK}, {"LEFTALT", KEY_LEFTALT}, {"SPACE", KEY_SPACE},
|
|
{"CAPSLOCK", KEY_CAPSLOCK}, {"F1", KEY_F1}, {"F2", KEY_F2}, {"F3", KEY_F3},
|
|
{"F4", KEY_F4}, {"F5", KEY_F5}, {"F6", KEY_F6}, {"F7", KEY_F7},
|
|
{"F8", KEY_F8}, {"F9", KEY_F9}, {"F10", KEY_F10}, {"NUMLOCK", KEY_NUMLOCK},
|
|
{"SCROLLLOCK", KEY_SCROLLLOCK}, {"KP7", KEY_KP7}, {"KP8", KEY_KP8},
|
|
{"KP9", KEY_KP9}, {"KPMINUS", KEY_KPMINUS}, {"KP4", KEY_KP4}, {"KP5", KEY_KP5},
|
|
{"KP6", KEY_KP6}, {"KPPLUS", KEY_KPPLUS}, {"KP1", KEY_KP1}, {"KP2", KEY_KP2},
|
|
{"KP3", KEY_KP3}, {"KP0", KEY_KP0}, {"KPDOT", KEY_KP0},
|
|
{"ZENKAKUHANKAKU", KEY_ZENKAKUHANKAKU}, {"102ND", KEY_102ND}, {"F11", KEY_F11},
|
|
{"F12", KEY_F12}, {"RO", KEY_RO}, {"KATAKANA", KEY_KATAKANA},
|
|
{"HIRAGANA", KEY_HIRAGANA}, {"HENKAN", KEY_HENKAN},
|
|
{"KATAKANAHIRAGANA", KEY_KATAKANAHIRAGANA}, {"MUHENKAN", KEY_MUHENKAN},
|
|
{"KPJPCOMMA", KEY_KPJPCOMMA}, {"KPENTER", KEY_KPENTER},
|
|
{"RIGHTCTRL", KEY_RIGHTCTRL}, {"KPSLASH", KEY_KPSLASH},
|
|
{"SYSRQ", KEY_SYSRQ}, {"RIGHTALT", KEY_RIGHTALT}, {"LINEFEED", KEY_LINEFEED},
|
|
{"HOME", KEY_HOME}, {"UP", KEY_UP}, {"PAGEUP", KEY_PAGEUP},
|
|
{"LEFT", KEY_LEFT}, {"RIGHT", KEY_RIGHT}, {"END", KEY_END},
|
|
{"DOWN", KEY_DOWN}, {"PAGEDOWN", KEY_PAGEDOWN}, {"INSERT", KEY_INSERT},
|
|
{"DELETE", KEY_DELETE}, {"MACRO", KEY_MACRO}, {"MUTE", KEY_MUTE},
|
|
{"VOLUMEDOWN", KEY_VOLUMEDOWN}, {"VOLUMEUP", KEY_VOLUMEUP},
|
|
{"POWER", KEY_POWER}, {"KPEQUAL", KEY_KPEQUAL}, {"KPPLUSMINUS", KEY_KPPLUSMINUS},
|
|
{"PAUSE", KEY_PAUSE}, {"SCALE", KEY_SCALE}, {"KPCOMMA", KEY_KPCOMMA},
|
|
{"HANGEUL", KEY_HANGEUL}, {"HANGUEL", KEY_HANGEUL}, {"HANJA", KEY_HANJA},
|
|
{"YEN", KEY_YEN}, {"LEFTMETA", KEY_LEFTMETA}, {"RIGHTMETA", KEY_LEFTMETA},
|
|
{"COMPOSE", KEY_COMPOSE}, {"STOP", KEY_STOP}, {"AGAIN", KEY_AGAIN},
|
|
{"PROPS", KEY_PROPS}, {"UNDO", KEY_UNDO}, {"FRONT", KEY_FRONT},
|
|
{"COPY", KEY_COPY}, {"OPEN", KEY_OPEN}, {"PASTE", KEY_PASTE},
|
|
{"FIND", KEY_FIND}, {"CUT", KEY_CUT}, {"HELP", KEY_HELP},
|
|
{"MENU", KEY_MENU}, {"CALC", KEY_CALC}, {"SETUP", KEY_SETUP},
|
|
{"SLEEP", KEY_SLEEP}, {"WAKEUP", KEY_WAKEUP}, {"FILE", KEY_FILE},
|
|
{"SENDFILE", KEY_SENDFILE}, {"DELETEFILE", KEY_DELETEFILE},
|
|
{"XFER", KEY_XFER}, {"PROG1", KEY_PROG1}, {"PROG2", KEY_PROG2},
|
|
{"WWW", KEY_WWW}, {"MSDOS", KEY_MSDOS}, {"COFFEE", KEY_COFFEE},
|
|
{"SCREENLOCK", KEY_COFFEE}, {"DIRECTION", KEY_DIRECTION},
|
|
{"CYCLEWINDOWS", KEY_CYCLEWINDOWS}, {"MAIL", KEY_MAIL},
|
|
{"BOOKMARKS", KEY_BOOKMARKS}, {"COMPUTER", KEY_COMPUTER}, {"BACK", KEY_BACK},
|
|
{"FORWARD", KEY_FORWARD}, {"CLOSECD", KEY_CLOSECD}, {"EJECTCD", KEY_EJECTCD},
|
|
{"EJECTCLOSECD", KEY_EJECTCLOSECD}, {"NEXTSONG", KEY_NEXTSONG},
|
|
{"PLAYPAUSE", KEY_PLAYPAUSE}, {"PREVIOUSSONG", KEY_PREVIOUSSONG},
|
|
{"STOPCD", KEY_STOPCD}, {"RECORD", KEY_RECORD}, {"REWIND", KEY_REWIND},
|
|
{"PHONE", KEY_PHONE}, {"ISO", KEY_ISO}, {"CONFIG", KEY_CONFIG},
|
|
{"HOMEPAGE", KEY_HOMEPAGE}, {"REFRESH", KEY_REFRESH}, {"EXIT", KEY_EXIT},
|
|
{"MOVE", KEY_MOVE}, {"EDIT", KEY_EDIT}, {"SCROLLUP", KEY_SCROLLUP},
|
|
{"SCROLLDOWN", KEY_SCROLLDOWN}, {"KPLEFTPAREN", KEY_KPLEFTPAREN},
|
|
{"KPRIGHTPAREN", KEY_KPRIGHTPAREN}, {"NEW", KEY_NEW}, {"REDO", KEY_REDO},
|
|
{"F13", KEY_F13}, {"F14", KEY_F14}, {"F15", KEY_F15}, {"F16", KEY_F16},
|
|
{"F17", KEY_F17}, {"F18", KEY_F18}, {"F19", KEY_F19}, {"F20", KEY_F20},
|
|
{"F21", KEY_F21}, {"F22", KEY_F22}, {"F23", KEY_F23}, {"F24", KEY_F24},
|
|
{"PLAYCD", KEY_PLAYCD}, {"PAUSECD", KEY_PAUSECD}, {"PROG3", KEY_PROG3},
|
|
{"PROG4", KEY_PROG4}, {"DASHBOARD", KEY_DASHBOARD}, {"SUSPEND", KEY_SUSPEND},
|
|
{"CLOSE", KEY_CLOSE}, {"PLAY", KEY_PLAY}, {"FASTFORWARD", KEY_FASTFORWARD},
|
|
{"BASSBOOST", KEY_BASSBOOST}, {"PRINT", KEY_PRINT}, {"HP", KEY_HP},
|
|
{"CAMERA", KEY_CAMERA}, {"SOUND", KEY_SOUND}, {"QUESTION", KEY_QUESTION},
|
|
{"EMAIL", KEY_EMAIL}, {"CHAT", KEY_CHAT}, {"SEARCH", KEY_SEARCH},
|
|
{"CONNECT", KEY_CONNECT}, {"FINANCE", KEY_FINANCE}, {"SPORT", KEY_SPORT},
|
|
{"SHOP", KEY_SHOP}, {"ALTERASE", KEY_ALTERASE}, {"CANCEL", KEY_CANCEL},
|
|
{"BRIGHTNESSDOWN", KEY_BRIGHTNESSDOWN}, {"BRIGHTNESSUP", KEY_BRIGHTNESSUP},
|
|
{"MEDIA", KEY_MEDIA}, {"SWITCHVIDEOMODE", KEY_SWITCHVIDEOMODE},
|
|
{"KBDILLUMTOGGLE", KEY_KBDILLUMTOGGLE}, {"KBDILLUMDOWN", KEY_KBDILLUMDOWN},
|
|
{"KBDILLUMUP", KEY_KBDILLUMUP}, {"SEND", KEY_SEND}, {"REPLY", KEY_REPLY},
|
|
{"FORWARDMAIL", KEY_FORWARDMAIL}, {"SAVE", KEY_SAVE},
|
|
{"DOCUMENTS", KEY_DOCUMENTS}, {"BATTERY", KEY_BATTERY},
|
|
{"BLUETOOTH", KEY_BLUETOOTH}, {"WLAN", KEY_WLAN}, {"UWB", KEY_UWB},
|
|
{"UNKNOWN", KEY_UNKNOWN}, {"VIDEO_NEXT", KEY_VIDEO_NEXT},
|
|
{"VIDEO_PREV", KEY_VIDEO_PREV}, {"BRIGHTNESS_CYCLE", KEY_BRIGHTNESS_CYCLE},
|
|
{"BRIGHTNESS_ZERO", KEY_BRIGHTNESS_ZERO}, {"DISPLAY_OFF", KEY_DISPLAY_OFF},
|
|
{"WIMAX", KEY_WIMAX}, {"RFKILL", KEY_RFKILL}, {"OK", KEY_OK},
|
|
{"SELECT", KEY_SELECT}, {"GOTO", KEY_GOTO}, {"CLEAR", KEY_CLEAR},
|
|
{"POWER2", KEY_POWER2}, {"OPTION", KEY_OPTION}, {"INFO", KEY_INFO},
|
|
{"TIME", KEY_TIME}, {"VENDOR", KEY_VENDOR}, {"ARCHIVE", KEY_ARCHIVE},
|
|
{"PROGRAM", KEY_PROGRAM}, {"CHANNEL", KEY_CHANNEL}, {"FAVORITES", KEY_FAVORITES},
|
|
{"EPG", KEY_EPG}, {"PVR", KEY_PVR}, {"MHP", KEY_MHP},
|
|
{"LANGUAGE", KEY_LANGUAGE}, {"TITLE", KEY_TITLE}, {"SUBTITLE", KEY_SUBTITLE},
|
|
{"ANGLE", KEY_ANGLE}, {"ZOOM", KEY_ZOOM}, {"MODE", KEY_MODE},
|
|
{"KEYBOARD", KEY_KEYBOARD}, {"SCREEN", KEY_SCREEN}, {"PC", KEY_PC},
|
|
{"TV", KEY_TV}, {"TV2", KEY_TV2}, {"VCR", KEY_VCR}, {"VCR2", KEY_VCR2},
|
|
{"SAT", KEY_SAT}, {"SAT2", KEY_SAT2}, {"CD", KEY_CD}, {"TAPE", KEY_TAPE},
|
|
{"RADIO", KEY_RADIO}, {"TUNER", KEY_TUNER}, {"PLAYER", KEY_PLAYER},
|
|
{"TEXT", KEY_TEXT}, {"DVD", KEY_DVD}, {"AUX", KEY_AUX}, {"MP3", KEY_MP3},
|
|
{"AUDIO", KEY_AUDIO}, {"VIDEO", KEY_VIDEO}, {"DIRECTORY", KEY_DIRECTORY},
|
|
{"LIST", KEY_LIST}, {"MEMO", KEY_MEMO}, {"CALENDAR", KEY_CALENDAR},
|
|
{"RED", KEY_RED}, {"GREEN", KEY_GREEN}, {"YELLOW", KEY_YELLOW},
|
|
{"BLUE", KEY_BLUE}, {"CHANNELUP", KEY_CHANNELUP},
|
|
{"CHANNELDOWN", KEY_CHANNELDOWN}, {"FIRST", KEY_FIRST}, {"LAST", KEY_LAST},
|
|
{"AB", KEY_AB}, {"NEXT", KEY_NEXT}, {"RESTART", KEY_RESTART},
|
|
{"SLOW", KEY_SLOW}, {"SHUFFLE", KEY_SHUFFLE}, {"BREAK", KEY_BREAK},
|
|
{"PREVIOUS", KEY_PREVIOUS}, {"DIGITS", KEY_DIGITS}, {"TEEN", KEY_TEEN},
|
|
{"TWEN", KEY_TWEN}, {"VIDEOPHONE", KEY_VIDEOPHONE}, {"GAMES", KEY_GAMES},
|
|
{"ZOOMIN", KEY_ZOOMIN}, {"ZOOMOUT", KEY_ZOOMOUT}, {"ZOOMRESET", KEY_ZOOMRESET},
|
|
{"WORDPROCESSOR", KEY_WORDPROCESSOR}, {"EDITOR", KEY_EDITOR},
|
|
{"SPREADSHEET", KEY_SPREADSHEET}, {"GRAPHICSEDITOR", KEY_GRAPHICSEDITOR},
|
|
{"PRESENTATION", KEY_PRESENTATION}, {"DATABASE", KEY_DATABASE},
|
|
{"NEWS", KEY_NEWS}, {"VOICEMAIL", KEY_VOICEMAIL},
|
|
{"ADDRESSBOOK", KEY_ADDRESSBOOK}, {"MESSENGER", KEY_MESSENGER},
|
|
{"DISPLAYTOGGLE", KEY_DISPLAYTOGGLE}, {"SPELLCHECK", KEY_SPELLCHECK},
|
|
{"LOGOFF", KEY_LOGOFF}, {"DOLLAR", KEY_DOLLAR}, {"EURO", KEY_EURO},
|
|
{"FRAMEBACK", KEY_FRAMEBACK}, {"FRAMEFORWARD", KEY_FRAMEFORWARD},
|
|
{"CONTEXT_MENU", KEY_CONTEXT_MENU}, {"MEDIA_REPEAT", KEY_MEDIA_REPEAT},
|
|
{"10CHANNELSUP", KEY_10CHANNELSUP}, {"10CHANNELSDOWN", KEY_10CHANNELSDOWN},
|
|
{"DEL_EOL", KEY_DEL_EOL}, {"DEL_EOS", KEY_DEL_EOS}, {"INS_LINE", KEY_INS_LINE},
|
|
{"DEL_LINE", KEY_DEL_LINE}, {"FN", KEY_FN}, {"FN_ESC", KEY_FN_ESC},
|
|
{"FN_F1", KEY_FN_F1}, {"FN_F2", KEY_FN_F2}, {"FN_F3", KEY_FN_F3},
|
|
{"FN_F4", KEY_FN_F4}, {"FN_F5", KEY_FN_F5}, {"FN_F6", KEY_FN_F6},
|
|
{"FN_F7", KEY_FN_F7}, {"FN_F8", KEY_FN_F8}, {"FN_F9", KEY_FN_F9},
|
|
{"FN_F10", KEY_FN_F10}, {"FN_F11", KEY_FN_F11}, {"FN_F12", KEY_FN_F12},
|
|
{"FN_1", KEY_FN_1}, {"FN_2", KEY_FN_2}, {"FN_D", KEY_FN_D},
|
|
{"FN_E", KEY_FN_E}, {"FN_F", KEY_FN_F}, {"FN_S", KEY_FN_S},
|
|
{"FN_B", KEY_FN_B}, {"BRL_DOT1", KEY_BRL_DOT1}, {"BRL_DOT2", KEY_BRL_DOT2},
|
|
{"BRL_DOT3", KEY_BRL_DOT3}, {"BRL_DOT4", KEY_BRL_DOT4},
|
|
{"BRL_DOT5", KEY_BRL_DOT5}, {"BRL_DOT6", KEY_BRL_DOT6},
|
|
{"BRL_DOT7", KEY_BRL_DOT7}, {"BRL_DOT8", KEY_BRL_DOT8},
|
|
{"BRL_DOT9", KEY_BRL_DOT9}, {"BRL_DOT10", KEY_BRL_DOT10},
|
|
{"NUMERIC_0", KEY_NUMERIC_0}, {"NUMERIC_1", KEY_NUMERIC_1},
|
|
{"NUMERIC_2", KEY_NUMERIC_2}, {"NUMERIC_3", KEY_NUMERIC_3},
|
|
{"NUMERIC_4", KEY_NUMERIC_4}, {"NUMERIC_5", KEY_NUMERIC_5},
|
|
{"NUMERIC_6", KEY_NUMERIC_6}, {"NUMERIC_7", KEY_NUMERIC_7},
|
|
{"NUMERIC_8", KEY_NUMERIC_8}, {"NUMERIC_9", KEY_NUMERIC_9},
|
|
{"NUMERIC_STAR", KEY_NUMERIC_STAR}, {"NUMERIC_POUND", KEY_NUMERIC_POUND},
|
|
{"CAMERA_FOCUS", KEY_CAMERA_FOCUS}, {"WPS_BUTTON", KEY_WPS_BUTTON},
|
|
{"TOUCHPAD_TOGGLE", KEY_TOUCHPAD_TOGGLE}, {"TOUCHPAD_ON", KEY_TOUCHPAD_ON},
|
|
{"TOUCHPAD_OFF", KEY_TOUCHPAD_OFF}};
|
|
|
|
struct key_buffer {
|
|
unsigned short buf[KEY_BUFFER_SIZE];
|
|
unsigned int size;
|
|
};
|
|
|
|
/* Hotkey list: linked list that holds all valid hoteys parsed from the
|
|
* config file and the corresponding command */
|
|
struct hotkey_list_e {
|
|
struct key_buffer kb;
|
|
char *command;
|
|
int fuzzy;
|
|
struct hotkey_list_e *next;
|
|
};
|
|
|
|
struct hotkey_list_e *hotkey_list = NULL;
|
|
const char evdev_root_dir[] = "/dev/input/";
|
|
unsigned long hotkey_size_mask = 0;
|
|
char *ext_config_file = NULL;
|
|
/* Flags */
|
|
int vflag = 0;
|
|
int dead = 0; // exit flag
|
|
|
|
/* key buffer operations */
|
|
int key_buffer_add (struct key_buffer*, unsigned short);
|
|
int key_buffer_remove (struct key_buffer*, unsigned short);
|
|
int key_buffer_compare_fuzzy (struct key_buffer *, struct key_buffer *);
|
|
int key_buffer_compare (struct key_buffer *, struct key_buffer *);
|
|
void key_buffer_reset (struct key_buffer *);
|
|
/* Other operations */
|
|
void int_handler (int signum);
|
|
void exec_command (char *);
|
|
void parse_config_file (void);
|
|
void update_descriptors_list (int **, int *);
|
|
int prepare_epoll (int *, int, int);
|
|
unsigned short key_to_code (char *);
|
|
int is_empty (const char *s);
|
|
/* hotkey list operations */
|
|
void hotkey_list_add (struct hotkey_list_e *, struct key_buffer *, char *, int);
|
|
void hotkey_list_destroy (struct hotkey_list_e *);
|
|
void hotkey_list_append_command (struct hotkey_list_e *, char *);
|
|
|
|
int main (int argc, char *argv[])
|
|
{
|
|
/* Handle SIGINT */
|
|
dead = 0;
|
|
struct sigaction action;
|
|
memset(&action, 0, sizeof(action));
|
|
action.sa_handler = int_handler;
|
|
sigaction(SIGINT, &action, NULL);
|
|
|
|
/* Parse command line arguments */
|
|
int opc;
|
|
while ((opc = getopt(argc, argv, "vc:")) != -1) {
|
|
switch (opc) {
|
|
case 'v':
|
|
vflag = 1;
|
|
break;
|
|
case 'c':
|
|
ext_config_file = malloc(strlen(optarg) + 1);
|
|
if (!ext_config_file)
|
|
die("malloc in main()");
|
|
strcpy(ext_config_file, optarg);
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* Parse config file */
|
|
parse_config_file();
|
|
|
|
/* Load descriptors */
|
|
int fd_num = 0;
|
|
int *fds = NULL;
|
|
update_descriptors_list(&fds, &fd_num);
|
|
|
|
/* Prepare directory update watcher */
|
|
int event_watcher = inotify_init1(IN_NONBLOCK);
|
|
if (event_watcher < 0)
|
|
die("could not call inotify_init");
|
|
if (inotify_add_watch(event_watcher, evdev_root_dir, IN_CREATE | IN_DELETE) < 0)
|
|
die("could not add /dev/input to the watch list");
|
|
|
|
struct input_event event;
|
|
struct key_buffer pb = {{0}, 0}; // Pressed keys buffer
|
|
ssize_t rb; // Read bits
|
|
|
|
/* Prepare epoll list */
|
|
int ev_fd;
|
|
ev_fd = prepare_epoll(fds, fd_num, event_watcher);
|
|
|
|
/* MAIN EVENT LOOP */
|
|
mainloop_begin:
|
|
for (;;) {
|
|
/* On linux use epoll(2) as it gives better performance */
|
|
static struct epoll_event ev_type;
|
|
if (epoll_wait(ev_fd, &ev_type, fd_num, -1) < 0 || dead)
|
|
break;
|
|
|
|
if (ev_type.events != EPOLLIN)
|
|
continue;
|
|
|
|
char buf[EVENT_BUF_LEN];
|
|
if (read(event_watcher, buf, EVENT_BUF_LEN) >= 0) {
|
|
sleep(1); // wait for devices to settle
|
|
update_descriptors_list(&fds, &fd_num);
|
|
if (close(ev_fd) < 0)
|
|
die("could not close event filedescriptors list (ev_fd)");
|
|
ev_fd = prepare_epoll(fds, fd_num, event_watcher);
|
|
goto mainloop_begin;
|
|
}
|
|
|
|
static unsigned int prev_size;
|
|
prev_size = pb.size;
|
|
for (int i = 0; i < fd_num; i++) {
|
|
|
|
rb = read(fds[i], &event, sizeof(struct input_event));
|
|
if (rb != sizeof(struct input_event)) continue;
|
|
|
|
/* Ignore touchpad events */
|
|
if (
|
|
event.type == EV_KEY &&
|
|
event.code != BTN_TOUCH &&
|
|
event.code != BTN_TOOL_FINGER &&
|
|
event.code != BTN_TOOL_DOUBLETAP &&
|
|
event.code != BTN_TOOL_TRIPLETAP
|
|
) {
|
|
switch (event.value) {
|
|
/* Key released */
|
|
case 0:
|
|
key_buffer_remove(&pb, event.code);
|
|
break;
|
|
/* Key pressed */
|
|
case 1:
|
|
key_buffer_add(&pb, event.code);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (pb.size <= prev_size)
|
|
continue;
|
|
|
|
if (vflag) {
|
|
int ci;
|
|
printf("Pressed keys: ");
|
|
for (unsigned int i = 0; i < pb.size; i++) {
|
|
ci = i;
|
|
while (pb.buf[i] != key_conversion_table[ci + 1].value) {
|
|
if (ci >= array_size_const(key_conversion_table) - 2)
|
|
break;
|
|
ci++;
|
|
}
|
|
printf("%s ", key_conversion_table[ci + 1].name);
|
|
}
|
|
putchar('\n');
|
|
}
|
|
|
|
struct hotkey_list_e *tmp;
|
|
int t = 0;
|
|
if (hotkey_size_mask & 1 << (pb.size - 1)) {
|
|
for (tmp = hotkey_list; tmp != NULL; tmp = tmp->next) {
|
|
if (tmp->fuzzy)
|
|
t = key_buffer_compare_fuzzy(&pb, &tmp->kb);
|
|
else
|
|
t = key_buffer_compare(&pb, &tmp->kb);
|
|
if (t)
|
|
exec_command(tmp->command);
|
|
|
|
}
|
|
}
|
|
}
|
|
|
|
// TODO: better child handling, for now all children receive the same
|
|
// interrupts as the father so everything should work fine
|
|
wait(NULL);
|
|
if (!dead)
|
|
fprintf(stderr, red("an error occured\n"));
|
|
close(ev_fd);
|
|
close(event_watcher);
|
|
for (int i = 0; i < fd_num; i++)
|
|
if (close(fds[i]) == -1)
|
|
die("close file descriptors");
|
|
return 0;
|
|
}
|
|
|
|
/* Adds a keycode to the pressed buffer if it is not already present
|
|
* Returns non zero if the key was not added. */
|
|
int key_buffer_add (struct key_buffer *pb, unsigned short key)
|
|
{
|
|
if (!pb) return 1;
|
|
/* Linear search if the key is already buffered */
|
|
for (unsigned int i = 0; i < pb->size; i++)
|
|
if (key == pb->buf[i]) return 1;
|
|
|
|
if (pb->size >= KEY_BUFFER_SIZE)
|
|
return 1;
|
|
|
|
pb->buf[pb->size++] = key;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* Removes a keycode from a pressed buffer if it is present returns
|
|
* non zero in case of failure (key not present or buffer empty). */
|
|
int key_buffer_remove (struct key_buffer *pb, unsigned short key)
|
|
{
|
|
if (!pb) return 1;
|
|
|
|
for (unsigned int i = 0; i < pb->size; i++) {
|
|
if (pb->buf[i] == key) {
|
|
pb->size--;
|
|
pb->buf[i] = pb->buf[pb->size];
|
|
return 0;
|
|
}
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
void key_buffer_reset (struct key_buffer *kb)
|
|
{
|
|
kb->size = 0;
|
|
for (int i = 0; i < KEY_BUFFER_SIZE; i++)
|
|
kb->buf[i] = 0;
|
|
}
|
|
|
|
void int_handler (int signum)
|
|
{
|
|
switch (signum) {
|
|
case SIGINT:
|
|
if (dead) {
|
|
fprintf(stderr, red("an error occured, exiting\n"));
|
|
exit(EXIT_FAILURE);
|
|
}
|
|
if (vflag)
|
|
printf(yellow("Received interrupt signal, exiting gracefully...\n"));
|
|
dead = 1;
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* Executes a command from a string */
|
|
void exec_command (char *command)
|
|
{
|
|
static wordexp_t result;
|
|
|
|
/* Expand the string for the program to run */
|
|
switch (wordexp (command, &result, 0)) {
|
|
case 0:
|
|
break;
|
|
case WRDE_NOSPACE:
|
|
/* If the error was WRDE_NOSPACE,
|
|
* then perhaps part of the result was allocated */
|
|
wordfree (&result);
|
|
return;
|
|
default:
|
|
/* Some other error */
|
|
fprintf(stderr, "Could not parse, %s is not valid\n", command);
|
|
return;
|
|
}
|
|
|
|
pid_t cpid;
|
|
switch (cpid = fork()) {
|
|
case -1:
|
|
fprintf(stderr, "Could not create child process: %s", strerror(errno));
|
|
wordfree(&result);
|
|
break;
|
|
case 0:
|
|
/* This is the child process, execute the command */
|
|
execvp(result.we_wordv[0], result.we_wordv);
|
|
fprintf(stderr, red("%s : %s\n"), command, strerror(errno));
|
|
exit(errno);
|
|
break;
|
|
default:
|
|
while (waitpid(cpid, NULL, WNOHANG) == -1) {}
|
|
wordfree(&result);
|
|
break;
|
|
}
|
|
}
|
|
|
|
void update_descriptors_list (int **fds, int *fd_num)
|
|
{
|
|
struct dirent *file_ent;
|
|
char ev_path[sizeof(evdev_root_dir) + FILE_NAME_MAX_LENGTH + 1];
|
|
void *tmp_p;
|
|
int tmp_fd;
|
|
unsigned char evtype_b[EV_MAX];
|
|
/* Open the event directory */
|
|
DIR *ev_dir = opendir(evdev_root_dir);
|
|
if (!ev_dir)
|
|
die("could not open /dev/input");
|
|
|
|
(*fd_num) = 0;
|
|
// if ((*fds))
|
|
// free(*fds);
|
|
|
|
for (;;) {
|
|
|
|
if ((file_ent = readdir(ev_dir)) == NULL)
|
|
break;
|
|
/* Filter out non character devices */
|
|
if (file_ent->d_type != DT_CHR)
|
|
continue;
|
|
|
|
/* Compose absolute path from relative */
|
|
strncpy(ev_path, evdev_root_dir, sizeof(evdev_root_dir) + FILE_NAME_MAX_LENGTH);
|
|
strncat(ev_path, file_ent->d_name, sizeof(evdev_root_dir) + FILE_NAME_MAX_LENGTH);
|
|
|
|
/* Open device and check if it can give key events otherwise ignore it */
|
|
tmp_fd = open(ev_path, O_RDONLY | O_NONBLOCK);
|
|
if (tmp_fd < 0) {
|
|
if (vflag)
|
|
printf(red("Could not open device %s\n"), ev_path);
|
|
continue;
|
|
}
|
|
|
|
memset(evtype_b, 0, sizeof(evtype_b));
|
|
if (ioctl(tmp_fd, EVIOCGBIT(0, EV_MAX), evtype_b) < 0) {
|
|
if (vflag)
|
|
printf(red("Could not read capabilities of device %s\n"),ev_path);
|
|
close(tmp_fd);
|
|
continue;
|
|
}
|
|
|
|
if (!test_bit(EV_KEY, evtype_b)) {
|
|
if (vflag)
|
|
printf(yellow("Ignoring device %s\n"), ev_path);
|
|
close(tmp_fd);
|
|
continue;
|
|
}
|
|
|
|
tmp_p = realloc((*fds), sizeof(int) * ((*fd_num) + 1));
|
|
if (!tmp_p)
|
|
die("realloc file descriptors");
|
|
(*fds) = (int *) tmp_p;
|
|
|
|
(*fds)[(*fd_num)] = tmp_fd;
|
|
(*fd_num)++;
|
|
}
|
|
closedir(ev_dir);
|
|
if (*fd_num) {
|
|
if (vflag)
|
|
printf(green("Monitoring %d devices\n"), *fd_num);
|
|
} else {
|
|
fprintf(stderr, red("Could not open any devices, exiting\n"));
|
|
exit(EXIT_FAILURE);
|
|
}
|
|
}
|
|
|
|
int prepare_epoll (int *fds, int fd_num, int event_watcher)
|
|
{
|
|
static struct epoll_event epoll_read_ev;
|
|
epoll_read_ev.events = EPOLLIN;
|
|
int ev_fd = epoll_create(1);
|
|
if (ev_fd < 0)
|
|
die("failed epoll_create");
|
|
if (epoll_ctl(ev_fd, EPOLL_CTL_ADD, event_watcher, &epoll_read_ev) < 0)
|
|
die("could not add file descriptor to the epoll list");
|
|
for (int i = 0; i < fd_num; i++)
|
|
if (epoll_ctl(ev_fd, EPOLL_CTL_ADD, fds[i], &epoll_read_ev) < 0)
|
|
die("could not add file descriptor to the epoll list");
|
|
return ev_fd;
|
|
}
|
|
|
|
/* Checks if two key buffers contain the same keys in no specified order */
|
|
int key_buffer_compare_fuzzy (struct key_buffer *haystack, struct key_buffer *needle)
|
|
{
|
|
if (haystack->size != needle->size)
|
|
return 0;
|
|
int ff = 0;
|
|
for (int x = needle->size - 1; x >= 0; x--) {
|
|
for (unsigned int i = 0; i < haystack->size; i++)
|
|
ff += (needle->buf[x] == haystack->buf[i]);
|
|
if (!ff)
|
|
return 0;
|
|
ff = 0;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
/* Checks if two key buffers are the same (same order) */
|
|
int key_buffer_compare (struct key_buffer *haystack, struct key_buffer *needle)
|
|
{
|
|
if (haystack->size != needle->size)
|
|
return 0;
|
|
for (unsigned int i = 0; i < needle->size; i++) {
|
|
if (needle->buf[i] != haystack->buf[i])
|
|
return 0;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
void hotkey_list_destroy (struct hotkey_list_e *head)
|
|
{
|
|
struct hotkey_list_e *tmp;
|
|
for (; head; free(tmp)) {
|
|
if (head->command)
|
|
free(head->command);
|
|
tmp = head;
|
|
head = head->next;
|
|
}
|
|
}
|
|
|
|
void hotkey_list_add (struct hotkey_list_e *head, struct key_buffer *kb, char *cmd, int f)
|
|
{
|
|
struct hotkey_list_e *tmp;
|
|
int size;
|
|
if (!(size = strlen(cmd)))
|
|
return;
|
|
if (!(tmp = malloc(sizeof(struct hotkey_list_e))))
|
|
die("memory allocation failed in hotkey_list_add()");
|
|
if (!(tmp->command = malloc(size + 1)))
|
|
die("memory allocation failed in hotkey_list_add()");
|
|
strcpy(tmp->command, cmd);
|
|
tmp->kb = *kb;
|
|
tmp->fuzzy = f;
|
|
tmp->next = NULL;
|
|
|
|
if (head) {
|
|
for (; head->next; head = head->next);
|
|
head->next = tmp;
|
|
} else
|
|
hotkey_list = tmp;
|
|
}
|
|
|
|
void hotkey_list_append_command (struct hotkey_list_e *head, char *cmd)
|
|
{
|
|
char *tmp;
|
|
if (!head)
|
|
return;
|
|
while (head->next)
|
|
head = head->next;
|
|
tmp = realloc(head->command, sizeof(head->command) + strlen(cmd) + 1);
|
|
if (!tmp)
|
|
die("realloc in hotkey_list_append_command()");
|
|
head->command = tmp;
|
|
strcat(head->command, cmd);
|
|
}
|
|
|
|
void parse_config_file (void)
|
|
{
|
|
wordexp_t result = {0};
|
|
FILE *fd;
|
|
if (ext_config_file) {
|
|
switch (wordexp(ext_config_file, &result, 0)) {
|
|
case 0:
|
|
break;
|
|
case WRDE_NOSPACE:
|
|
/* If the error was WRDE_NOSPACE,
|
|
* then perhaps part of the result was allocated */
|
|
wordfree (&result);
|
|
die("not enough space")
|
|
default:
|
|
/* Some other error */
|
|
die("path not valid");
|
|
}
|
|
|
|
fd = fopen(result.we_wordv[0], "r");
|
|
wordfree(&result);
|
|
if (!fd)
|
|
die("error opening config file");
|
|
free(ext_config_file);
|
|
ext_config_file = NULL;
|
|
} else {
|
|
for (int i = 0; i < array_size_const(config_paths); i++) {
|
|
switch (wordexp(config_paths[i], &result, 0)) {
|
|
case 0:
|
|
break;
|
|
case WRDE_NOSPACE:
|
|
/* If the error was WRDE_NOSPACE,
|
|
* then perhaps part of the result was allocated */
|
|
wordfree (&result);
|
|
die("not enough space")
|
|
default:
|
|
/* Some other error */
|
|
die("path not valid");
|
|
}
|
|
|
|
fd = fopen(result.we_wordv[0], "r");
|
|
wordfree(&result);
|
|
if (fd)
|
|
break;
|
|
if (vflag)
|
|
printf(yellow("config file not found at %s\n"), config_paths[i]);
|
|
}
|
|
if (!fd)
|
|
die("could not open any config files, check the log for more details");
|
|
}
|
|
|
|
int iscmd = 0;
|
|
struct key_buffer kb;
|
|
for (int linenum = 1;; linenum++) {
|
|
int fuzzy = 0, tmp;
|
|
char * linebegin = NULL;
|
|
char *line = NULL, *keys = NULL, *command = NULL;
|
|
size_t linelen = 0;
|
|
|
|
if (getline(&line, &linelen, fd) == -1)
|
|
break;
|
|
|
|
linelen = strlen(line);
|
|
if (linelen < 2)
|
|
continue;
|
|
|
|
linebegin = line;
|
|
|
|
// Remove leading spaces
|
|
while (isspace(line[0]) && linelen > 1) {
|
|
line = &line[1];
|
|
linelen--;
|
|
}
|
|
|
|
// Skip comments and blank lines
|
|
if (line[0] == '#' || !line[0])
|
|
goto parse_end_free;
|
|
|
|
for (size_t i = 1; i < linelen; i++) {
|
|
if (line[i] == '#')
|
|
line[i] = '\0';
|
|
}
|
|
if (!iscmd) {
|
|
if (line[0] != '-' && line[0] != '*') {
|
|
fprintf(stderr, red("Error at line %d: "
|
|
"Hotkey definitions must start with '*' or '-'\n"), linenum);
|
|
exit(EXIT_FAILURE);
|
|
}
|
|
|
|
if (line[0] == '*')
|
|
fuzzy = 1;
|
|
line = &line[1];
|
|
linelen--;
|
|
// Remove leading spaces
|
|
while (isspace(line[0]) && linelen > 1) {
|
|
line = &line[1];
|
|
linelen--;
|
|
}
|
|
keys = strtok(line, ":");
|
|
command = strtok(NULL, ":");
|
|
if (!command || !keys) {
|
|
fprintf(stderr, red("Error at line %d: "
|
|
"No command or keys specified\n"), linenum);
|
|
exit(EXIT_FAILURE);
|
|
}
|
|
// Check if keys and command are not blank
|
|
if (is_empty(keys) || is_empty(command)) {
|
|
fprintf(stderr, red("Error at line %d: "
|
|
"command or keys not present\n"), linenum);
|
|
exit(EXIT_FAILURE);
|
|
}
|
|
|
|
// Remove whitespaces in keys
|
|
tmp = strlen(keys);
|
|
for (int i = 0; i < tmp; i++) {
|
|
if (isspace(keys[i]))
|
|
memmove(&line[i], &line[i + 1], --tmp);
|
|
}
|
|
|
|
// Remove leading and trailing spaces in command
|
|
tmp = strlen(command);
|
|
while (isspace(command[0]) && tmp > 1) {
|
|
command = &command[1];
|
|
tmp--;
|
|
}
|
|
for (tmp = strlen(command) - 1; isspace(command[tmp]); tmp--) {
|
|
if (isblank(command[tmp]))
|
|
command[tmp] = '\0';
|
|
}
|
|
|
|
if (command[strlen(command) - 2] == '\\') {
|
|
iscmd = 1;
|
|
} else {
|
|
iscmd = 0;
|
|
tmp = strlen(command) - 1;
|
|
while (isspace(command[tmp]))
|
|
command[tmp--] = '\0';
|
|
}
|
|
|
|
key_buffer_reset(&kb);
|
|
char *k = strtok(keys, ",");
|
|
unsigned short kc;
|
|
do {
|
|
if (!(kc = key_to_code(k))) {
|
|
fprintf(stderr, red("Error at line %d: "
|
|
"%s is not a valid key"), linenum, k);
|
|
exit(EXIT_FAILURE);
|
|
}
|
|
key_buffer_add(&kb, kc);
|
|
} while ((k = strtok(NULL, ",")));
|
|
hotkey_list_add(hotkey_list, &kb, command, fuzzy);
|
|
hotkey_size_mask |= 1 << (kb.size - 1);
|
|
key_buffer_reset(&kb);
|
|
} else {
|
|
for (tmp = strlen(line) - 1; isspace(line[tmp]); tmp--) {
|
|
if (isblank(line[tmp]))
|
|
line[tmp] = '\0';
|
|
}
|
|
linelen = strlen(line);
|
|
|
|
// if line is blank skip
|
|
if (is_empty(line))
|
|
goto parse_end_free;
|
|
|
|
if (line[linelen - 2] == '\\') {
|
|
iscmd = 1;
|
|
} else {
|
|
iscmd = 0;
|
|
while (isspace(line[linelen - 1]))
|
|
line[--linelen] = '\0';
|
|
}
|
|
hotkey_list_append_command(hotkey_list, line);
|
|
}
|
|
parse_end_free:
|
|
free(linebegin);
|
|
}
|
|
}
|
|
|
|
unsigned short key_to_code (char *key)
|
|
{
|
|
for (int i = 0; i < array_size_const(key_conversion_table); i++) {
|
|
if (!strcmp(key_conversion_table[i].name, key))
|
|
return key_conversion_table[i].value;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int is_empty (const char *s)
|
|
{
|
|
while (*s != '\0') {
|
|
if (!isspace((unsigned char)*s))
|
|
return 0;
|
|
s++;
|
|
}
|
|
return 1;
|
|
}
|
|
|