/** * @brief General-purpose keyboard conversion library. * * This provides similar functionality to xkb: * - It provides mappings for keyboards from locales * - It translates incoming key presses to key names * - It translates incoming keys to escape sequences * * @copyright * This file is part of ToaruOS and is released under the terms * of the NCSA / University of Illinois License - see LICENSE.md * Copyright (C) 2012-2018 K. Lange */ #include #include #define DEBUG_SCANCODES 0 #define KEY_UP_MASK 0x80 #define KEY_CODE_MASK 0x7F #define KEY_CTRL_MASK 0x40 #define norm 0x01 #define spec 0x02 #define func 0x03 #define SET_UNSET(a,b,c) (a) = (c) ? ((a) | (b)) : ((a) & ~(b)) char key_method[] = { /* 00 */ 0, spec, norm, norm, norm, norm, norm, norm, /* 08 */ norm, norm, norm, norm, norm, norm, norm, norm, /* 10 */ norm, norm, norm, norm, norm, norm, norm, norm, /* 18 */ norm, norm, norm, norm, norm, spec, norm, norm, /* 20 */ norm, norm, norm, norm, norm, norm, norm, norm, /* 28 */ norm, norm, spec, norm, norm, norm, norm, norm, /* 30 */ norm, norm, norm, norm, norm, norm, spec, norm, /* 38 */ spec, norm, spec, func, func, func, func, func, /* 40 */ func, func, func, func, func, spec, spec, spec, /* 48 */ spec, spec, spec, spec, spec, spec, spec, spec, /* 50 */ spec, spec, spec, spec, spec, spec, spec, func, /* 58 */ func, spec, spec, spec, spec, spec, spec, spec, /* 60 */ spec, spec, spec, spec, spec, spec, spec, spec, /* 68 */ spec, spec, spec, spec, spec, spec, spec, spec, /* 70 */ spec, spec, spec, spec, spec, spec, spec, spec, /* 78 */ spec, spec, spec, spec, spec, spec, spec, spec, }; char kbd_us[128] = { 0, 27, '1','2','3','4','5','6','7','8','9','0', '-','=','\b', '\t', /* tab */ 'q','w','e','r','t','y','u','i','o','p','[',']','\n', 0, /* control */ 'a','s','d','f','g','h','j','k','l',';','\'', '`', 0, /* left shift */ '\\','z','x','c','v','b','n','m',',','.','/', 0, /* right shift */ '*', 0, /* alt */ ' ', /* space */ 0, /* caps lock */ 0, /* F1 [59] */ 0, 0, 0, 0, 0, 0, 0, 0, 0, /* ... F10 */ 0, /* 69 num lock */ 0, /* scroll lock */ 0, /* home */ 0, /* up */ 0, /* page up */ '-', 0, /* left arrow */ 0, 0, /* right arrow */ '+', 0, /* 79 end */ 0, /* down */ 0, /* page down */ 0, /* insert */ 0, /* delete */ 0, 0, 0, 0, /* F11 */ 0, /* F12 */ 0, /* everything else */ }; char kbd_us_l2[128] = { 0, 27, '!','@','#','$','%','^','&','*','(',')', '_','+','\b', '\t', /* tab */ 'Q','W','E','R','T','Y','U','I','O','P','{','}','\n', 0, /* control */ 'A','S','D','F','G','H','J','K','L',':','"', '~', 0, /* left shift */ '|','Z','X','C','V','B','N','M','<','>','?', 0, /* right shift */ '*', 0, /* alt */ ' ', /* space */ 0, /* caps lock */ 0, /* F1 [59] */ 0, 0, 0, 0, 0, 0, 0, 0, 0, /* ... F10 */ 0, /* 69 num lock */ 0, /* scroll lock */ 0, /* home */ 0, /* up */ 0, /* page up */ '-', 0, /* left arrow */ 0, 0, /* right arrow */ '+', 0, /* 79 end */ 0, /* down */ 0, /* page down */ 0, /* insert */ 0, /* delete */ 0, 0, 0, 0, /* F11 */ 0, /* F12 */ 0, /* everything else */ }; #define KEY_SCANCODE_F1 0x3b #define KEY_SCANCODE_F2 0x3c #define KEY_SCANCODE_F3 0x3d #define KEY_SCANCODE_F4 0x3e #define KEY_SCANCODE_F5 0x3f #define KEY_SCANCODE_F6 0x40 #define KEY_SCANCODE_F7 0x41 #define KEY_SCANCODE_F8 0x42 #define KEY_SCANCODE_F9 0x43 #define KEY_SCANCODE_F10 0x44 #define KEY_SCANCODE_F11 0x57 #define KEY_SCANCODE_F12 0x58 #define KEY_SCANCODE_NUM_1 0x4f #define KEY_SCANCODE_NUM_2 0x50 #define KEY_SCANCODE_NUM_3 0x51 #define KEY_SCANCODE_NUM_4 0x4B #define KEY_SCANCODE_NUM_5 0x4C #define KEY_SCANCODE_NUM_6 0x4D #define KEY_SCANCODE_NUM_7 0x47 #define KEY_SCANCODE_NUM_8 0x48 #define KEY_SCANCODE_NUM_9 0x49 #define KEY_SCANCODE_NUM_0 0x52 #define KEY_SCANCODE_NUM_DOT 0x53 #define KEY_SCANCODE_NUM_MIN 0x4a #define KEY_SCANCODE_NUM_ADD 0x4e #define KEY_SCANCODE_NUM_LK 0x45 #define KEY_SCANCODE_SCROLL 0x46 int kbd_scancode(key_event_state_t * state, unsigned char c, key_event_t * event) { /* Convert scancodes to a series of keys */ event->keycode = 0; event->action = 0; event->modifiers = 0; event->key = 0; #if DEBUG_SCANCODES fprintf(stderr, "[%d] %d\n", state->kbd_s_state, (int)c); #endif event->modifiers |= state->kl_ctrl ? KEY_MOD_LEFT_CTRL : 0; event->modifiers |= state->kl_shift ? KEY_MOD_LEFT_SHIFT : 0; event->modifiers |= state->kl_alt ? KEY_MOD_LEFT_ALT : 0; event->modifiers |= state->kl_super ? KEY_MOD_LEFT_SUPER : 0; event->modifiers |= state->kr_ctrl ? KEY_MOD_RIGHT_CTRL : 0; event->modifiers |= state->kr_shift ? KEY_MOD_RIGHT_SHIFT : 0; event->modifiers |= state->kr_alt ? KEY_MOD_RIGHT_ALT : 0; event->modifiers |= state->kr_super ? KEY_MOD_RIGHT_SUPER : 0; if (!state->kbd_s_state) { if (c == 0xE0) { state->kbd_s_state = 1; return 0; } event->action = (c & KEY_UP_MASK) ? KEY_ACTION_UP : KEY_ACTION_DOWN; c &= 0x7F; int down = (event->action == KEY_ACTION_DOWN); switch (key_method[c]) { case norm: { event->keycode = kbd_us[c]; if (state->k_ctrl) { int s = kbd_us[c]; if (s >= 'a' && s <= 'z') s -= 'a' - 'A'; if (s == '-') s = '_'; if (s == '`') s = '@'; int out = (int)(s - KEY_CTRL_MASK); if (out < 0 || out > 0x1F) { event->key = kbd_us[c]; } else { event->key = out; } } else { event->key = state->k_shift ? kbd_us_l2[c] : kbd_us[c]; } } return 1; case spec: switch (c) { case 0x01: event->key = '\033'; event->keycode = KEY_ESCAPE; return 1; case 0x1D: state->k_ctrl = down; state->kl_ctrl = down; event->keycode = KEY_LEFT_CTRL; SET_UNSET(event->modifiers, KEY_MOD_LEFT_CTRL, down); return 1; case 0x2A: state->k_shift = down; state->kl_shift = down; event->keycode = KEY_LEFT_SHIFT; SET_UNSET(event->modifiers, KEY_MOD_LEFT_SHIFT, down); return 1; case 0x36: state->k_shift = down; state->kr_shift = down; event->keycode = KEY_RIGHT_SHIFT; SET_UNSET(event->modifiers, KEY_MOD_RIGHT_SHIFT, down); return 1; case 0x38: state->k_alt = down; state->kl_alt = down; event->keycode = KEY_LEFT_ALT; SET_UNSET(event->modifiers, KEY_MOD_LEFT_ALT, down); return 1; case KEY_SCANCODE_NUM_0: event->keycode = KEY_NUM_0; event->key = '0'; return 1; case KEY_SCANCODE_NUM_1: event->keycode = KEY_NUM_1; event->key = '1'; return 1; case KEY_SCANCODE_NUM_2: event->keycode = KEY_NUM_2; event->key = '2'; return 1; case KEY_SCANCODE_NUM_3: event->keycode = KEY_NUM_3; event->key = '3'; return 1; case KEY_SCANCODE_NUM_4: event->keycode = KEY_NUM_4; event->key = '4'; return 1; case KEY_SCANCODE_NUM_5: event->keycode = KEY_NUM_5; event->key = '5'; return 1; case KEY_SCANCODE_NUM_6: event->keycode = KEY_NUM_6; event->key = '6'; return 1; case KEY_SCANCODE_NUM_7: event->keycode = KEY_NUM_7; event->key = '7'; return 1; case KEY_SCANCODE_NUM_8: event->keycode = KEY_NUM_8; event->key = '8'; return 1; case KEY_SCANCODE_NUM_9: event->keycode = KEY_NUM_9; event->key = '9'; return 1; case KEY_SCANCODE_NUM_DOT: event->keycode = KEY_NUM_DOT; event->key = '.'; return 1; case KEY_SCANCODE_NUM_MIN: event->keycode = KEY_NUM_MINUS; event->key = '-'; return 1; case KEY_SCANCODE_NUM_ADD: event->keycode = KEY_NUM_PLUS; event->key = '+'; return 1; default: break; } break; case func: switch (c) { case KEY_SCANCODE_F1: event->keycode = KEY_F1; return 1; case KEY_SCANCODE_F2: event->keycode = KEY_F2; return 1; case KEY_SCANCODE_F3: event->keycode = KEY_F3; return 1; case KEY_SCANCODE_F4: event->keycode = KEY_F4; return 1; case KEY_SCANCODE_F5: event->keycode = KEY_F5; return 1; case KEY_SCANCODE_F6: event->keycode = KEY_F6; return 1; case KEY_SCANCODE_F7: event->keycode = KEY_F7; return 1; case KEY_SCANCODE_F8: event->keycode = KEY_F8; return 1; case KEY_SCANCODE_F9: event->keycode = KEY_F9; return 1; case KEY_SCANCODE_F10: event->keycode = KEY_F10; return 1; case KEY_SCANCODE_F11: event->keycode = KEY_F11; return 1; case KEY_SCANCODE_F12: event->keycode = KEY_F12; return 1; } break; default: break; } return 0; } else if (state->kbd_s_state == 1) { state->kbd_s_state = 0; event->action = (c & KEY_UP_MASK) ? KEY_ACTION_UP : KEY_ACTION_DOWN; c &= 0x7F; int down = (event->action == KEY_ACTION_DOWN); switch (c) { case 0x5B: state->k_super = down; state->kl_super = down; event->keycode = KEY_LEFT_SUPER; SET_UNSET(event->modifiers, KEY_MOD_LEFT_SUPER, down); return 1; case 0x5C: state->k_super = down; state->kr_super = down; event->keycode = KEY_RIGHT_SUPER; SET_UNSET(event->modifiers, KEY_MOD_RIGHT_SUPER, down); return 1; case 0x1D: state->kr_ctrl = down; state->k_ctrl = down; event->keycode = KEY_RIGHT_CTRL; SET_UNSET(event->modifiers, KEY_MOD_RIGHT_CTRL, down); return 1; case 0x38: state->kr_alt = down; state->k_alt = down; event->keycode = KEY_RIGHT_ALT; SET_UNSET(event->modifiers, KEY_MOD_RIGHT_ALT, down); return 1; case 0x48: event->keycode = KEY_ARROW_UP; return 1; case 0x4D: event->keycode = KEY_ARROW_RIGHT; return 1; case 0x47: event->keycode = KEY_HOME; return 1; case 0x49: event->keycode = KEY_PAGE_UP; return 1; case 0x4B: event->keycode = KEY_ARROW_LEFT; return 1; case 0x4F: event->keycode = KEY_END; return 1; case 0x50: event->keycode = KEY_ARROW_DOWN; return 1; case 0x51: event->keycode = KEY_PAGE_DOWN; return 1; case 0x52: event->keycode = KEY_INSERT; return 1; case 0x53: event->keycode = KEY_DEL; return 1; case 0x35: event->keycode = KEY_NUM_DIV; event->key = '/'; return 1; case 0x1C: event->keycode = KEY_NUM_ENTER; event->key = '\n'; return 1; case 0x37: event->keycode = KEY_PRINT_SCREEN; return 1; case 0x5D: event->keycode = KEY_APP; return 1; default: break; } } return 0; }