/* * (C) Copyright 1992, ..., 2014 the "DOSEMU-Development-Team". * * for details see file COPYING in the DOSEMU distribution */ #include #include #include #include #include #include #include #include #include #include #include #include "emu.h" #include "timers.h" #include "sig.h" #include "keyboard/keymaps.h" #include "keyboard/keyb_clients.h" #include "keyboard/keyboard.h" #include "utilities.h" #include "video.h" #include "env_term.h" #include "translate/translate.h" #ifndef SLANG_VERSION # define SLANG_VERSION 1 #endif #if SLANG_VERSION < 20000 || defined(USE_RELAYTOOL) static inline void slang_set_error(int x) { #ifdef USE_RELAYTOOL if (SLang_Version >= 20000) { SLang_set_error(x); return; } #endif SLang_Error = x; } static inline int slang_get_error(void) { #ifdef USE_RELAYTOOL if (SLang_Version >= 20000) return SLang_get_error(); #endif return SLang_Error; } #else #define slang_set_error(x) SLang_set_error(x) #define slang_get_error() SLang_get_error() #endif #ifndef VOID # define VOID void #endif #define KBBUF_SIZE 80 static struct keyboard_state { int kbd_fd; hitimer_t t_start; int kbcount; Bit8u kbbuf[KBBUF_SIZE]; Bit8u *kbp; int save_kbd_flags; /* saved flags for STDIN before our fcntl */ struct termios save_termios; int pc_scancode_mode; /* By default we are not in pc_scancode_mode */ SLKeyMap_List_Type *The_Normal_KeyMap; unsigned char erasekey; unsigned char Esc_Char; int KeyNot_Ready; /* a flag */ int Keystr_Len; unsigned long Shift_Flags; struct char_set_state translate_state; } keyb_state; /* * The goal of the routines here is simple: to allow SHIFT, ALT, and * CONTROL keys to be used with a remote terminal. The way this is * accomplished is simple: There are 4 keymaps: normal, shift, control, * and alt. The user must a key or keysequence that invokes on of these * keymaps. The default keymap is the normal one. When one of the other * keymaps is invoked via the special key defined for it, the new keymap * is only in effect for the next character at which point the normal one * becomes active again. * * The normal keymap is fairly robust. It contains mappings for all the * ascii characters (0-26, 28-256). The escape character itself (27) is * included only in the sence that it is the prefix character for arrow * keys, function keys, etc... */ /* * The keymaps are simply a mapping from a character sequence to * scan/ascii codes. */ typedef struct { char keystr[10]; unsigned long scan_code; } Keymap_Scan_Type; #define SHIFT_MASK 0x00010000 #define CTRL_MASK 0x00020000 #define ALT_MASK 0x00040000 #define ALTGR_MASK 0x00080000 #define STICKY_SHIFT_MASK 0x00100000 #define STICKY_CTRL_MASK 0x00200000 #define STICKY_ALT_MASK 0x00400000 #define STICKY_ALTGR_MASK 0x00800000 #define KEYPAD_MASK 0x01000000 #define MOVE_MASK 0x02000000 #define WAIT_MASK 0x04000000 #define ALT_DKY_SCAN_CODE 0x80000000 #define STICKY_ALT_DKY_SCAN_CODE 0x80000001 #define SHIFT_DKY_SCAN_CODE 0x80000002 #define STICKY_SHIFT_DKY_SCAN_CODE 0x80000003 #define CTRL_DKY_SCAN_CODE 0x80000004 #define STICKY_CTRL_DKY_SCAN_CODE 0x80000005 #define ALTGR_DKY_SCAN_CODE 0x80000006 #define STICKY_ALTGR_DKY_SCAN_CODE 0x80000007 #define SCROLL_UP_SCAN_CODE 0x80000020 #define SCROLL_DOWN_SCAN_CODE 0x80000021 #define REDRAW_SCAN_CODE 0x80000022 #define SUSPEND_SCAN_CODE 0x80000023 #define HELP_SCAN_CODE 0x80000024 #define RESET_SCAN_CODE 0x80000025 #define SET_MONO_SCAN_CODE 0x80000026 #define KEYPAD_DKY_SCAN_CODE 0x80000027 /* Keyboard maps for non-terminfo keys */ static Keymap_Scan_Type Dosemu_defined_fkeys[] = { /* These are the F1 - F12 keys for terminals without them. */ {"^@1", DKY_F1 }, /* F1 */ {"^@2", DKY_F2 }, /* F2 */ {"^@3", DKY_F3 }, /* F3 */ {"^@4", DKY_F4 }, /* F4 */ {"^@5", DKY_F5 }, /* F5 */ {"^@6", DKY_F6 }, /* F6 */ {"^@7", DKY_F7 }, /* F7 */ {"^@8", DKY_F8 }, /* F8 */ {"^@9", DKY_F9 }, /* F9 */ {"^@0", DKY_F10 }, /* F10 */ {"^@-", DKY_F11 }, /* F11 */ {"^@=", DKY_F12 }, /* F12 */ {"", 0} }; static Keymap_Scan_Type CTRL[] = { {"^Q", DKY_Q | CTRL_MASK }, {"^W", DKY_W | CTRL_MASK }, {"^E", DKY_E | CTRL_MASK }, {"^R", DKY_R | CTRL_MASK }, {"^T", DKY_T | CTRL_MASK }, {"^Y", DKY_Y | CTRL_MASK }, {"^U", DKY_U | CTRL_MASK }, {"^I", DKY_TAB }, {"^O", DKY_O | CTRL_MASK }, {"^P", DKY_P | CTRL_MASK }, {"^A", DKY_A | CTRL_MASK }, {"^S", DKY_S | CTRL_MASK }, {"^D", DKY_D | CTRL_MASK }, {"^F", DKY_F | CTRL_MASK }, {"^G", DKY_G | CTRL_MASK }, /* if backspace generates ^H then the terminfo definition takes precedence here */ {"^H", DKY_H | CTRL_MASK }, {"^J", DKY_J | CTRL_MASK }, {"^K", DKY_K | CTRL_MASK }, {"^L", DKY_L | CTRL_MASK }, {"^Z", DKY_Z | CTRL_MASK }, {"^X", DKY_X | CTRL_MASK }, #if 0 {"^C", DKY_BREAK }, #else {"^C", DKY_C | CTRL_MASK }, #endif {"^V", DKY_V | CTRL_MASK }, {"^B", DKY_B | CTRL_MASK }, {"^N", DKY_N | CTRL_MASK }, {"^M", DKY_RETURN }, {"^\\",DKY_BACKSLASH | CTRL_MASK }, {"^]", DKY_RBRACK | CTRL_MASK }, {"^^", DKY_6 | CTRL_MASK }, {"^_", DKY_DASH | CTRL_MASK | SHIFT_MASK }, {"", 0} }; #if 0 /* better not to use this (see comment below) */ static Keymap_Scan_Type Linux_Keypad[] = { /* Keypad keys */ {"\033OP", DKY_NUM }, /* Keypad Numlock */ {"\033OQ", DKY_PAD_SLASH }, /* Keypad / */ {"\033OR", DKY_PAD_AST }, /* Keypad * */ {"\033OS", DKY_PAD_MINUS }, /* Keypad - */ {"", 0} }; #endif static Keymap_Scan_Type Xterm_Keypad[] = { /* Keypad keys */ {"\033Oo", DKY_PAD_SLASH }, /* Keypad / */ {"\033Oj", DKY_PAD_AST }, /* Keypad * */ {"\033Om", DKY_PAD_MINUS }, /* Keypad - */ {"", 0} }; static Keymap_Scan_Type vtxxx_Keypad[] = { {"\033Ow", DKY_PAD_7 }, /* Keypad 7 */ {"\033Ox", DKY_PAD_8 }, /* Keypad 8 */ {"\033Oy", DKY_PAD_9 }, /* Keypad 9 */ {"\033Ot", DKY_PAD_4 }, /* Keypad 4 */ {"\033Ou", DKY_PAD_5 }, /* Keypad 5 */ {"\033Ov", DKY_PAD_6 }, /* Keypad 6 */ {"\033Ol", DKY_PAD_PLUS }, /* Keypad + */ {"\033Oq", DKY_PAD_1 }, /* Keypad 1 */ {"\033Or", DKY_PAD_2 }, /* Keypad 2 */ {"\033Os", DKY_PAD_3 }, /* Keypad 3 */ {"\033Op", DKY_PAD_0 }, /* Keypad 0 */ {"\033On", DKY_PAD_DECIMAL }, /* Keypad . */ {"\033OM", DKY_PAD_ENTER }, /* Keypad Enter */ {"", 0} }; static Keymap_Scan_Type Xterm_Xkeys[] = { {"\033[2~", DKY_INS | MOVE_MASK }, /* Ins */ {"\033[H", DKY_HOME }, /* Ho (rxvt) */ {"\033[e", DKY_END }, /* End (color_xterm) */ {"\033[K", DKY_END }, /* End - Where does this come from ? */ {"\033[F", DKY_END }, /* End - another xterm */ {"\033[5~", DKY_PGUP }, /* PgUp */ {"\033[6~", DKY_PGDN }, /* PgDn */ {"\033[7~", DKY_HOME }, /* Ho (xterm) */ {"\033[8~", DKY_END }, /* End (xterm) */ {"\033[1~", DKY_HOME }, /* Ho (putty) */ {"\033[4~", DKY_END }, /* End (putty) */ {"\033[A", DKY_UP }, /* Up */ {"\033[B", DKY_DOWN }, /* Dn */ {"\033[C", DKY_RIGHT }, /* Ri */ {"\033[D", DKY_LEFT }, /* Le */ {"\033[G", DKY_PAD_5 }, /* 5 (putty) */ {"\033OA", DKY_UP | CTRL_MASK }, /* Up (putty) */ {"\033OB", DKY_DOWN | CTRL_MASK }, /* Dn (putty) */ {"\033OC", DKY_RIGHT | CTRL_MASK }, /* Ri (putty) */ {"\033OD", DKY_LEFT | CTRL_MASK }, /* Le (putty) */ {"\033OG", DKY_PAD_5 | CTRL_MASK }, /* 5 (putty) */ {"\033[Z", DKY_TAB | SHIFT_MASK }, /* Shift-Tab */ /* rxvt keys */ {"\033[a", DKY_UP | SHIFT_MASK }, /* Up */ {"\033[b", DKY_DOWN | SHIFT_MASK }, /* Dn */ {"\033[c", DKY_RIGHT | SHIFT_MASK },/* Ri */ {"\033[d", DKY_LEFT | SHIFT_MASK }, /* Le */ {"\033Oa", DKY_UP | CTRL_MASK }, /* Up */ {"\033Ob", DKY_DOWN | CTRL_MASK }, /* Dn */ {"\033Oc", DKY_RIGHT | CTRL_MASK }, /* Ri */ {"\033Od", DKY_LEFT | CTRL_MASK }, /* Le */ {"", 0} }; static Keymap_Scan_Type vtxxx_fkeys[] = { {"\033[17~", DKY_F6 }, /* F6 */ {"\033[18~", DKY_F7 }, /* F7 */ {"\033[19~", DKY_F8 }, /* F8 */ {"\033[20~", DKY_F9 }, /* F9 */ {"\033[21~", DKY_F10 }, /* F10 */ {"\033[23~", DKY_F1 | SHIFT_MASK }, /* Shift F1 (F11 acts like * Shift-F1) */ {"\033[24~", DKY_F2 | SHIFT_MASK }, /* Shift F2 (F12 acts like * Shift-F2) */ {"\033[25~", DKY_F3 | SHIFT_MASK }, /* Shift F3 */ {"\033[26~", DKY_F4 | SHIFT_MASK }, /* Shift F4 */ {"\033[28~", DKY_F5 | SHIFT_MASK }, /* Shift F5 */ {"\033[29~", DKY_F6 | SHIFT_MASK }, /* Shift F6 */ {"\033[31~", DKY_F7 | SHIFT_MASK }, /* Shift F7 */ {"\033[32~", DKY_F8 | SHIFT_MASK }, /* Shift F8 */ {"\033[33~", DKY_F9 | SHIFT_MASK }, /* Shift F9 */ {"\033[34~", DKY_F10 | SHIFT_MASK }, /* Shift F10 */ {"", 0} }; static Keymap_Scan_Type Xterm_fkeys[] = { {"\033[11~", DKY_F1 }, /* F1 */ {"\033[12~", DKY_F2 }, /* F2 */ {"\033[13~", DKY_F3 }, /* F3 */ {"\033[14~", DKY_F4 }, /* F4 */ {"\033[15~", DKY_F5 }, /* F5 */ {"", 0} }; static Keymap_Scan_Type vtxxx_xkeys[] = { /* Who knows which mode it'll be in */ {"\033OA", DKY_UP }, /* Up */ {"\033OB", DKY_DOWN }, /* Dn */ {"\033OC", DKY_RIGHT }, /* Ri */ {"\033OD", DKY_LEFT }, /* Le */ {"\033[A", DKY_UP }, /* Up */ {"\033[B", DKY_DOWN }, /* Dn */ {"\033[C", DKY_RIGHT }, /* Ri */ {"\033[D", DKY_LEFT }, /* Le */ {"", 0} }; /* Keyboard map for terminfo keys */ static Keymap_Scan_Type terminfo_keys[] = { {"^(kb)", DKY_BKSP}, /* BackSpace */ {"^(k1)", DKY_F1}, /* F1 */ {"^(k2)", DKY_F2}, /* F2 */ {"^(k3)", DKY_F3}, /* F3 */ {"^(k4)", DKY_F4}, /* F4 */ {"^(k5)", DKY_F5}, /* F5 */ {"^(k6)", DKY_F6}, /* F6 */ {"^(k7)", DKY_F7}, /* F7 */ {"^(k8)", DKY_F8}, /* F8 */ {"^(k9)", DKY_F9}, /* F9 */ {"^(k;)", DKY_F10}, /* F10 */ {"^(kI)", DKY_INS|MOVE_MASK}, /* Ins */ {"^(#3)", DKY_INS|MOVE_MASK|SHIFT_MASK}, /* Shift Insert */ {"^(kD)", DKY_DEL}, /* Del */ {"^(*5)", DKY_DEL|SHIFT_MASK}, /* Shift Del */ {"^(kh)", DKY_HOME}, /* Ho */ {"^(#2)", DKY_HOME|SHIFT_MASK}, /* Shift Home */ {"^(kH)", DKY_END}, /* End */ {"^(@7)", DKY_END}, /* End */ {"^(*7)", DKY_END|SHIFT_MASK}, /* Shift End */ {"^(kP)", DKY_PGUP}, /* PgUp */ {"^(kN)", DKY_PGDN}, /* PgDn */ {"^(K1)", DKY_PAD_7}, /* Upper Left key on keypad */ {"^(ku)", DKY_UP}, /* Up */ {"^(K3)", DKY_PAD_9}, /* Upper Right key on keypad */ {"^(kl)", DKY_LEFT}, /* Le */ {"^(#4)", DKY_LEFT|SHIFT_MASK}, /* Shift Left */ {"^(K2)", DKY_PAD_5}, /* Center key on keypad */ {"^(kr)", DKY_RIGHT}, /* Ri */ {"^(K4)", DKY_PAD_1}, /* Lower Left key on keypad */ {"^(kd)", DKY_DOWN}, /* Dn */ {"^(K5)", DKY_PAD_3}, /* Lower Right key on keypad */ {"^(%i)", DKY_RIGHT|SHIFT_MASK}, /* Shift Right */ {"^(kB)", DKY_TAB|SHIFT_MASK}, /* Shift Tab -- BackTab */ {"^(@8)", DKY_PAD_ENTER}, /* DKY_RETURN? */ /* Enter */ {"^(Km)", DKY_MOUSE }, /* Mouse */ /* Special keys */ {"^(&2)", REDRAW_SCAN_CODE}, /* Refresh */ {"^(%1)", HELP_SCAN_CODE}, /* Help */ {"", 0} }; /* Keyboard map for basic terminfo keys F11--F20 */ static Keymap_Scan_Type terminfo_basic_fkeys[] = { {"^(F1)", DKY_F1|SHIFT_MASK}, /* Shuft F1 */ {"^(F2)", DKY_F2|SHIFT_MASK}, /* Shift F2 */ {"^(F3)", DKY_F3|SHIFT_MASK}, /* Shift F3 */ {"^(F4)", DKY_F4|SHIFT_MASK}, /* Shift F4 */ {"^(F5)", DKY_F5|SHIFT_MASK}, /* Shift F5 */ {"^(F6)", DKY_F6|SHIFT_MASK}, /* Shift F6 */ {"^(F7)", DKY_F7|SHIFT_MASK}, /* Shift F7 */ {"^(F8)", DKY_F8|SHIFT_MASK}, /* Shift F8 */ {"^(F9)", DKY_F9|SHIFT_MASK}, /* Shift F9 */ {"^(FA)", DKY_F10|SHIFT_MASK}, /* Shift F10 */ {"", 0} }; /* Keyboard map for extended terminfo keys F11--F48 */ static Keymap_Scan_Type terminfo_ext_fkeys[] = { {"^(F1)", DKY_F11}, /* F11 */ {"^(F2)", DKY_F12}, /* F12 */ {"^(F3)", DKY_F1|SHIFT_MASK}, /* Shift F1 */ {"^(F4)", DKY_F2|SHIFT_MASK}, /* Shift F2 */ {"^(F5)", DKY_F3|SHIFT_MASK}, /* Shift F3 */ {"^(F6)", DKY_F4|SHIFT_MASK}, /* Shift F4 */ {"^(F7)", DKY_F5|SHIFT_MASK}, /* Shift F5 */ {"^(F8)", DKY_F6|SHIFT_MASK}, /* Shift F6 */ {"^(F9)", DKY_F7|SHIFT_MASK}, /* Shift F7 */ {"^(FA)", DKY_F8|SHIFT_MASK}, /* Shift F8 */ {"^(FB)", DKY_F9|SHIFT_MASK}, /* Shift F9 */ {"^(FC)", DKY_F10|SHIFT_MASK}, /* Shift F10 */ {"^(FD)", DKY_F11|SHIFT_MASK}, /* Shift F11 */ {"^(FE)", DKY_F12|SHIFT_MASK}, /* Shift F12 */ {"^(FF)", DKY_F1|CTRL_MASK}, /* Ctrl F1 */ {"^(FG)", DKY_F2|CTRL_MASK}, /* Ctrl F2 */ {"^(FH)", DKY_F3|CTRL_MASK}, /* Ctrl F3 */ {"^(FI)", DKY_F4|CTRL_MASK}, /* Ctrl F4 */ {"^(FJ)", DKY_F5|CTRL_MASK}, /* Ctrl F5 */ {"^(FK)", DKY_F6|CTRL_MASK}, /* Ctrl F6 */ {"^(FL)", DKY_F7|CTRL_MASK}, /* Ctrl F7 */ {"^(FM)", DKY_F8|CTRL_MASK}, /* Ctrl F8 */ {"^(FN)", DKY_F9|CTRL_MASK}, /* Ctrl F9 */ {"^(FO)", DKY_F10|CTRL_MASK}, /* Ctrl F10 */ {"^(FP)", DKY_F11|CTRL_MASK}, /* Ctrl F11 */ {"^(FQ)", DKY_F12|CTRL_MASK}, /* Ctrl F12 */ {"^(FR)", DKY_F1|CTRL_MASK|SHIFT_MASK}, /* Ctrl Shift F1 */ {"^(FS)", DKY_F2|CTRL_MASK|SHIFT_MASK}, /* Ctrl Shift F2 */ {"^(FT)", DKY_F3|CTRL_MASK|SHIFT_MASK}, /* Ctrl Shift F3 */ {"^(FU)", DKY_F4|CTRL_MASK|SHIFT_MASK}, /* Ctrl Shift F4 */ {"^(FV)", DKY_F5|CTRL_MASK|SHIFT_MASK}, /* Ctrl Shift F5 */ {"^(FW)", DKY_F6|CTRL_MASK|SHIFT_MASK}, /* Ctrl Shift F6 */ {"^(FX)", DKY_F7|CTRL_MASK|SHIFT_MASK}, /* Ctrl Shift F7 */ {"^(FY)", DKY_F8|CTRL_MASK|SHIFT_MASK}, /* Ctrl Shift F8 */ {"^(FZ)", DKY_F9|CTRL_MASK|SHIFT_MASK}, /* Ctrl Shift F9 */ {"^(Fa)", DKY_F10|CTRL_MASK|SHIFT_MASK}, /* Ctrl Shift F10 */ {"^(Fb)", DKY_F11|CTRL_MASK|SHIFT_MASK}, /* Ctrl Shift F11 */ {"^(Fc)", DKY_F12|CTRL_MASK|SHIFT_MASK}, /* Ctrl Shift F12 */ {"", 0} }; static Keymap_Scan_Type Dosemu_Xkeys[] = { /* These keys are laid out like the numbers on the keypad - not too difficult */ {"^@K0", DKY_INS|MOVE_MASK}, /* Ins */ {"^@K1", DKY_END }, /* End Another keyscan is 0x6100 */ {"^@K2", DKY_DOWN }, /* Dn */ {"^@K3", DKY_PGDN }, /* PgDn */ {"^@K4", DKY_LEFT }, /* Le */ {"^@K5", DKY_PAD_5 }, /* There's no Xkey equlivant */ {"^@K6", DKY_RIGHT }, /* Ri */ {"^@K7", DKY_HOME }, /* Ho Another keyscan is 0x5c00 */ {"^@K8", DKY_UP }, /* Up */ {"^@K9", DKY_PGUP }, /* PgUp */ {"^@K.", DKY_DEL }, /* Del Another keyscan is 0x007F */ {"^@Kd", DKY_DEL }, /* Del */ /* And a few more */ {"^@Kh", DKY_PAUSE }, /* Hold or Pause DOS */ {"^@Kp", DKY_PRTSCR }, /* Print screen, SysRequest. */ {"^@Ky", DKY_SYSRQ }, /* SysRequest. */ {"^@KS", DKY_SCROLL }, /* Scroll Lock */ {"^@KN", DKY_NUM }, /* Num Lock */ {"", 0} }; static Keymap_Scan_Type Dosemu_Ctrl_keys[] = { /* Now setup the shift modifier keys */ {"^@a", ALT_DKY_SCAN_CODE }, {"^@c", CTRL_DKY_SCAN_CODE }, {"^@s", SHIFT_DKY_SCAN_CODE }, {"^@g", ALTGR_DKY_SCAN_CODE }, {"^@A", STICKY_ALT_DKY_SCAN_CODE }, {"^@C", STICKY_CTRL_DKY_SCAN_CODE }, {"^@S", STICKY_SHIFT_DKY_SCAN_CODE }, {"^@G", STICKY_ALTGR_DKY_SCAN_CODE }, {"^@k", KEYPAD_DKY_SCAN_CODE }, {"^@?", HELP_SCAN_CODE }, {"^@h", HELP_SCAN_CODE }, {"^@^R", REDRAW_SCAN_CODE }, {"^@^L", REDRAW_SCAN_CODE }, {"^@^Z", SUSPEND_SCAN_CODE }, {"^@ ", RESET_SCAN_CODE }, {"^@B", SET_MONO_SCAN_CODE }, {"^@\033[A", SCROLL_UP_SCAN_CODE }, {"^@\033OA", SCROLL_UP_SCAN_CODE }, {"^@U", SCROLL_UP_SCAN_CODE }, {"^@\033[B", SCROLL_DOWN_SCAN_CODE }, {"^@\033OB", SCROLL_DOWN_SCAN_CODE }, {"^@D", SCROLL_DOWN_SCAN_CODE }, {"", 0} }; static Keymap_Scan_Type Dosemu_Alt_keys[] = { {"\033[1;3P", ALT_MASK | DKY_F1}, {"\033[1;3Q", ALT_MASK | DKY_F2}, {"\033[1;3R", ALT_MASK | DKY_F3}, {"\033[1;3S", ALT_MASK | DKY_F4}, {"\033[15;3~", ALT_MASK | DKY_F5}, {"\033[17;3~", ALT_MASK | DKY_F6}, {"\033[18;3~", ALT_MASK | DKY_F7}, {"\033[19;3~", ALT_MASK | DKY_F8}, {"\033[20;3~", ALT_MASK | DKY_F9}, {"\033[21;3~", ALT_MASK | DKY_F10}, {"\033[23;3~", ALT_MASK | DKY_F11}, {"\033[24;3~", ALT_MASK | DKY_F12}, {"", 0} }; /* shift state from previous keypress */ static unsigned long old_flags = 0; static const unsigned char *define_key_keys = 0; static int define_key_keys_length =0; static void _do_slang_getkeys(void); static int define_getkey_callback(void) { if (define_key_keys_length == 0) { define_key_keys = 0; } if (!define_key_keys) { return 0; } define_key_keys_length--; return *define_key_keys++; } /* Note: Later definitions with the same or a conflicting key sequence fail, * and give an error message, but now don't stop the emulator. */ static int define_key(char *key, unsigned long scan, SLKeyMap_List_Type * m) { char buf[SLANG_MAX_KEYMAP_KEY_SEQ +1], k1; unsigned char buf2[SLANG_MAX_KEYMAP_KEY_SEQ +1]; int ret; const unsigned char *key_str; SLang_Key_Type *pre_key; int i; if (strlen(key) > SLANG_MAX_KEYMAP_KEY_SEQ) { k_printf("key string too long %s\n", key); return -1; } if (slang_get_error()) { k_printf("Current slang error skipping string %s\n", key); return -1; } if ((*key == '^') && (keyb_state.Esc_Char != '@')) { k1 = key[1]; if (k1 == keyb_state.Esc_Char) return 0; /* ^Esc_Char is not defined here */ if (k1 == '@') { strcpy(buf, key); buf[1] = keyb_state.Esc_Char; key = buf; } } /* Get the translated keystring, and save a copy */ key_str = (unsigned char *)SLang_process_keystring(key); memcpy(buf2, key_str, key_str[0]); key_str = buf2; /* Display in the debug logs what we are defining */ k_printf("KBD: define "); k_printf("'%s'=", strprintable((char *)key)); for(i = 1; i < key_str[0]; i++) { if (i != 1) { k_printf(","); } k_printf("%02x", key_str[i]); } k_printf(" -> %04lX:%04lX\n", scan >> 16, scan & 0xFFFF); if (key_str[0] == 1) { k_printf("KBD: no input string skipping\n\n"); return 0; } /* Lookup the key to see if we have already defined it */ define_key_keys = key_str +1; define_key_keys_length = key_str[0] -1; pre_key = SLang_do_key(m, define_getkey_callback); /* Duplicate key definition, warn and ignore it */ if (pre_key && (pre_key->str[0] == key_str[0]) && (memcmp(pre_key->str, key_str, key_str[0]) == 0)) { unsigned long prev_scan; prev_scan = (unsigned long)pre_key->f.f; k_printf("KBD: Previously mapped to: %04lx:%04lx\n\n", prev_scan >> 16, prev_scan & 0xFFFF); return 0; } ret = SLkm_define_key((char *)key, (VOID *) scan, m); if (ret == -2) { /* Conflicting key error, ignore it */ k_printf("KBD: Conflicting key: \n\n"); slang_set_error(0); } if (slang_get_error()) { fprintf(stderr, "Bad key: %s\n", key); return -1; } return 0; } static int define_keyset(Keymap_Scan_Type *k, SLKeyMap_List_Type *m) { char *str; while ((str = k->keystr), (*str != 0)) { define_key(str, k->scan_code, m); k++; } return 0; } static void define_remaining_characters(SLKeyMap_List_Type *m) { int i; for(i = ' '; i < 256; i++) { char str[2]; if (keyb_state.Esc_Char != '@' && keyb_state.Esc_Char == '@' + i) continue; str[0] = i; str[1] = '\0'; if (define_key(str, DKY_VOID, m) < 0) { continue; } } } static int init_slang_keymaps(void) { SLKeyMap_List_Type *m; char buf[5]; unsigned long esc_scan; char * term; char * kf21; /* Do some sanity checking */ if (config.term_esc_char >= 32) config.term_esc_char = 30; /* Carriage Return & Escape are not going to be used by any sane person */ if ((config.term_esc_char == '\x0d') || (config.term_esc_char == '\x1b')) { config.term_esc_char = 30; } /* escape characters are identity mapped in unicode. */ esc_scan = config.term_esc_char; esc_scan |= CTRL_MASK; keyb_state.Esc_Char = config.term_esc_char + '@'; if (keyb_state.The_Normal_KeyMap != NULL) return 0; if (NULL == (m = keyb_state.The_Normal_KeyMap = SLang_create_keymap("Normal", NULL))) return -1; /* Everybody needs these */ define_keyset(Dosemu_defined_fkeys, m); /* Keypad in a special way */ define_keyset(Dosemu_Xkeys, m); /* Some terminals have kf13=shift+f1, some others kf11=f11=shift+f1: often the ones which stop with kf20 will have 2*10 function keys: for those we assume f11=shift+f1. The ones with more commonly have n*12 (xterm 4*12=48). That's much better for dosemu as there is no confusing anymore The Linux console is most problematic as the default mapping may be different from what loadkeys does. KDGKBENT works only locally and in that case raw keyboard is available anyway. */ kf21 = SLtt_tgetstr("FB"); define_keyset(terminfo_keys, m); if (kf21) { k_printf("KBD: Many function keys detected in terminfo, trust it\n"); define_keyset(terminfo_ext_fkeys, m); } else { define_keyset(terminfo_basic_fkeys, m); } define_keyset(CTRL, m); term = getenv("TERM"); if( using_xterm() ) { /* Oh no, this is _BAD_, there are so many different things called 'xterm'*/ if (!kf21) { define_keyset(vtxxx_fkeys, m); define_keyset(Xterm_fkeys, m); } define_keyset(vtxxx_Keypad, m); define_keyset(Xterm_Keypad, m); define_keyset(Xterm_Xkeys, m); } #if 0 /* The linux terminfo entry is missing smkx/rmkx; manually setting keypad mode using an escape sequence causes some confusion (NumLock LED does not work) */ else if( term && !strncmp("linux", term, 5) ) { /* This isn't too nasty; most is in terminfo */ define_keyset(vtxxx_Keypad, m); define_keyset(Linux_Keypad, m); } #endif else if( term && strcmp("vt52", term) && !strncmp("vt", term, 2) && term[2] >= '1' && term[2] <= '9' ) { /* A 'real' VT ... yesss, if you're sure ... */ if (!kf21) define_keyset(vtxxx_fkeys, m); define_keyset(vtxxx_xkeys, m); define_keyset(vtxxx_Keypad, m); } /* And more Dosemu keys */ define_keyset(Dosemu_Ctrl_keys, m); define_keyset(Dosemu_Alt_keys, m); if (slang_get_error()) return -1; /* * If the erase key (as set by stty) is a reasonably safe one, use it. */ if( ((keyb_state.erasekey>0 && keyb_state.erasekey<' ')) && keyb_state.erasekey != 27 && keyb_state.erasekey != keyb_state.Esc_Char) { buf[0] = '^'; buf[1] = keyb_state.erasekey+'@'; buf[2] = 0; define_key(buf, DKY_BKSP, m); } else if (keyb_state.erasekey > '~') { buf[0] = keyb_state.erasekey; buf[1] = 0; define_key(buf, DKY_BKSP, m); } /* * Now add one more for the esc character so that sending it twice sends * it. */ SLkm_define_key("^[^[", (VOID *) DKY_ESC, m); if (slang_get_error()) return -1; /* Note: define_keys_by_character comes last or we could never define functions keys. . . */ define_remaining_characters(m); if (slang_get_error()) return -1; return 0; } /* * Global variables this module uses: int kbcount : number of characters * in the keyboard buffer to be processed. unsigned char kbbuf[KBBUF_SIZE] * : buffer where characters are waiting to be processed unsigned char * *kbp : pointer into kbbuf where next char to be processed is. */ static int read_some_keys(void) { fd_set fds; struct timeval tv = { 0, 0 }; int selrt; int cc; int offs; if (keyb_state.kbcount == 0) keyb_state.kbp = keyb_state.kbbuf; else if (keyb_state.kbp > &keyb_state.kbbuf[(KBBUF_SIZE * 3) / 5]) { memmove(keyb_state.kbbuf, keyb_state.kbp, keyb_state.kbcount); keyb_state.kbp = keyb_state.kbbuf; } FD_ZERO(&fds); FD_SET(keyb_state.kbd_fd, &fds); selrt = select(keyb_state.kbd_fd + 1, &fds, NULL, NULL, &tv); if (selrt <= 0) return 0; if (!FD_ISSET(keyb_state.kbd_fd, &fds)) return 0; offs = keyb_state.kbp - keyb_state.kbbuf; cc = read(keyb_state.kbd_fd, &keyb_state.kbp[keyb_state.kbcount], KBBUF_SIZE - keyb_state.kbcount - offs); k_printf("KBD: cc found %d characters (Xlate)\n", cc); if (cc > 0) keyb_state.kbcount += cc; return cc; } /* * This function is a callback to read the key. It returns 0 if one is * not ready. */ static int getkey_callback(void) { if (keyb_state.kbcount == keyb_state.Keystr_Len) read_some_keys(); if (keyb_state.kbcount == keyb_state.Keystr_Len) { keyb_state.KeyNot_Ready = 1; return 0; } return keyb_state.kbp[keyb_state.Keystr_Len++]; } /* * If the sticky bits are set, then the scan code or the modifier key has * already been taken care of. In this case, the unsticky bit should be * ignored. */ static void slang_send_scancode(unsigned long ls_flags, unsigned long lscan) { unsigned long flags = 0; k_printf("KBD: slang_send_scancode(ls_flags=%08lx, lscan=%08lx)\n", ls_flags, lscan); if (ls_flags & KEYPAD_MASK) { flags |= KEYPAD_MASK; switch(lscan) { case DKY_INS: lscan = DKY_PAD_INS; break; case DKY_END: lscan = DKY_PAD_END; break; case DKY_DOWN: lscan = DKY_PAD_DOWN; break; case DKY_PGDN: lscan = DKY_PAD_PGDN; break; case DKY_LEFT: lscan = DKY_PAD_LEFT; break; case DKY_RIGHT: lscan = DKY_PAD_RIGHT; break; case DKY_HOME: lscan = DKY_PAD_HOME; break; case DKY_UP: lscan = DKY_PAD_UP; break; case DKY_PGUP: lscan = DKY_PAD_PGUP; break; case DKY_DEL: lscan = DKY_PAD_DEL; break; case DKY_DASH: lscan = DKY_PAD_MINUS; break; case DKY_RETURN: lscan = DKY_PAD_ENTER; break; case DKY_0: lscan = DKY_PAD_0; break; case DKY_1: lscan = DKY_PAD_1; break; case DKY_2: lscan = DKY_PAD_2; break; case DKY_3: lscan = DKY_PAD_3; break; case DKY_4: lscan = DKY_PAD_4; break; case DKY_5: lscan = DKY_PAD_5; break; case DKY_6: lscan = DKY_PAD_6; break; case DKY_7: lscan = DKY_PAD_7; break; case DKY_9: lscan = DKY_PAD_9; break; /* This is a special */ case DKY_8: if ( ls_flags & SHIFT_MASK ) { ls_flags &= ~SHIFT_MASK; lscan = DKY_PAD_AST; } else lscan = DKY_PAD_8; break; /* Need to remove the shift flag for this */ case DKY_EQUALS: if (ls_flags & SHIFT_MASK ) { ls_flags &= ~SHIFT_MASK; lscan = DKY_PAD_PLUS; } /* else It is silly to translate an equals */ break; /* This still generates the wrong scancode - should be $E02F */ case DKY_SLASH: lscan = DKY_PAD_SLASH; break; } } else if( (ls_flags & (ALT_MASK|STICKY_ALT_MASK|ALTGR_MASK|STICKY_ALTGR_MASK)) && (lscan == DKY_PRTSCR)) { lscan = DKY_SYSRQ; ls_flags |= MOVE_MASK; } if ((ls_flags & SHIFT_MASK) && ((ls_flags & STICKY_SHIFT_MASK) == 0)) { flags |= SHIFT_MASK; move_key(PRESS, DKY_L_SHIFT); } if ((ls_flags & CTRL_MASK) && ((ls_flags & STICKY_CTRL_MASK) == 0)) { flags |= CTRL_MASK; move_key(PRESS, DKY_L_CTRL); } if ((ls_flags & ALT_MASK) && ((ls_flags & STICKY_ALT_MASK) == 0)) { flags |= ALT_MASK; move_key(PRESS, DKY_L_ALT); } if ((ls_flags & ALTGR_MASK) && ((ls_flags & STICKY_ALTGR_MASK) == 0)) { flags |= ALTGR_MASK; move_key(PRESS, DKY_R_ALT); } if (!(ls_flags & MOVE_MASK)) { /* For any keys we know do not modify the shiftstate * this is the optimal way to go. As it handles all * of the weird cases. */ put_modified_symbol(PRESS, get_shiftstate(), lscan); put_modified_symbol(RELEASE, get_shiftstate(), lscan); } else { /* For the few keys that might modify the shiftstate * we just do a straight forward key press and release. */ move_key(PRESS, lscan); move_key(RELEASE, lscan); } /* set shift release in a "queue" to wait for two SIGALRMs */ if (flags) flags |= WAIT_MASK; old_flags = flags; } void handle_slang_keys(Boolean make, t_keysym key) { if (!make) { return; } switch(key) { case DKY_DOSEMU_HELP: DOSemu_Slang_Show_Help = 1; break; case DKY_DOSEMU_REDRAW: dos_slang_redraw(); break; case DKY_DOSEMU_SUSPEND: dos_slang_suspend(); break; case DKY_DOSEMU_MONO: dos_slang_smart_set_mono(); break; case DKY_DOSEMU_PAN_UP: DOSemu_Terminal_Scroll = -1; break; case DKY_DOSEMU_PAN_DOWN: DOSemu_Terminal_Scroll = 1; break; case DKY_DOSEMU_PAN_LEFT: /* this should be implemented someday */ break; case DKY_DOSEMU_PAN_RIGHT: /* this should be implemented someday */ break; case DKY_DOSEMU_RESET: DOSemu_Slang_Show_Help = 0; DOSemu_Terminal_Scroll = 0; if (keyb_state.Shift_Flags & STICKY_CTRL_MASK) { move_key(RELEASE, DKY_L_CTRL); } if (keyb_state.Shift_Flags & STICKY_SHIFT_MASK) { move_key(RELEASE, DKY_L_SHIFT); } if (keyb_state.Shift_Flags & STICKY_ALT_MASK) { move_key(RELEASE, DKY_L_ALT); } if (keyb_state.Shift_Flags & STICKY_ALTGR_MASK) { move_key(RELEASE, DKY_R_ALT); } keyb_state.Shift_Flags = 0; } return; } static void do_slang_special_keys(unsigned long scan) { static const char *keymap_prompts[] = { 0, "[Shift]", "[Ctrl]", "[Ctrl-Shift]", "[Alt]", "[Alt-Shift]", "[Alt-Ctrl]", "[Alt-Ctrl-Shift]", "[AltGr]", "[AltGr-Shift]", "[AltGr-Ctrl]", "[AltGr-Ctrl-Shift]", "[AltGr-Alt]", "[AltGr-Alt-Shift]", "[AltGr-Alt-Ctrl]", "[AltGr-Alt-Ctrl-Shift]" }; int prompt_no = 0; switch (scan) { case CTRL_DKY_SCAN_CODE: if ( !(keyb_state.Shift_Flags & STICKY_CTRL_MASK)) keyb_state.Shift_Flags |= CTRL_MASK; break; case STICKY_CTRL_DKY_SCAN_CODE: if (keyb_state.Shift_Flags & CTRL_MASK) keyb_state.Shift_Flags &= ~CTRL_MASK; if (keyb_state.Shift_Flags & STICKY_CTRL_MASK) { move_key(RELEASE, DKY_L_CTRL); keyb_state.Shift_Flags &= ~STICKY_CTRL_MASK; } else { keyb_state.Shift_Flags |= STICKY_CTRL_MASK; move_key(PRESS, DKY_L_CTRL); } break; case SHIFT_DKY_SCAN_CODE: if ( !(keyb_state.Shift_Flags & STICKY_SHIFT_MASK)) keyb_state.Shift_Flags |= SHIFT_MASK; break; case STICKY_SHIFT_DKY_SCAN_CODE: if (keyb_state.Shift_Flags & SHIFT_MASK) keyb_state.Shift_Flags &= ~SHIFT_MASK; if (keyb_state.Shift_Flags & STICKY_SHIFT_MASK) { move_key(RELEASE, DKY_L_SHIFT); keyb_state.Shift_Flags &= ~STICKY_SHIFT_MASK; } else { keyb_state.Shift_Flags |= STICKY_SHIFT_MASK; move_key(PRESS, DKY_L_SHIFT); } break; case ALT_DKY_SCAN_CODE: if ( !(keyb_state.Shift_Flags & STICKY_ALT_MASK)) keyb_state.Shift_Flags |= ALT_MASK; break; case STICKY_ALT_DKY_SCAN_CODE: if (keyb_state.Shift_Flags & ALT_MASK) keyb_state.Shift_Flags &= ~ALT_MASK; if (keyb_state.Shift_Flags & STICKY_ALT_MASK) { move_key(RELEASE, DKY_L_ALT); keyb_state.Shift_Flags &= ~STICKY_ALT_MASK; } else { keyb_state.Shift_Flags |= STICKY_ALT_MASK; move_key(PRESS, DKY_L_ALT); } break; case ALTGR_DKY_SCAN_CODE: if ( !(keyb_state.Shift_Flags & STICKY_ALTGR_MASK)) keyb_state.Shift_Flags |= ALTGR_MASK; break; case STICKY_ALTGR_DKY_SCAN_CODE: if (keyb_state.Shift_Flags & ALTGR_MASK) keyb_state.Shift_Flags &= ~ALTGR_MASK; if (keyb_state.Shift_Flags & STICKY_ALTGR_MASK) { move_key(RELEASE, DKY_R_ALT); keyb_state.Shift_Flags &= ~STICKY_ALTGR_MASK; } else { keyb_state.Shift_Flags |= STICKY_ALTGR_MASK; move_key(PRESS, DKY_R_ALT); } break; case KEYPAD_DKY_SCAN_CODE: keyb_state.Shift_Flags |= KEYPAD_MASK; break; case SCROLL_DOWN_SCAN_CODE: DOSemu_Terminal_Scroll = 1; break; case SCROLL_UP_SCAN_CODE: DOSemu_Terminal_Scroll = -1; break; case REDRAW_SCAN_CODE: dos_slang_redraw(); break; case SUSPEND_SCAN_CODE: dos_slang_suspend(); break; case HELP_SCAN_CODE: DOSemu_Slang_Show_Help = 1; break; case RESET_SCAN_CODE: DOSemu_Slang_Show_Help = 0; DOSemu_Terminal_Scroll = 0; if (keyb_state.Shift_Flags & STICKY_CTRL_MASK) { move_key(RELEASE, DKY_L_CTRL); } if (keyb_state.Shift_Flags & STICKY_SHIFT_MASK) { move_key(RELEASE, DKY_L_SHIFT); } if (keyb_state.Shift_Flags & STICKY_ALT_MASK) { move_key(RELEASE, DKY_L_ALT); } if (keyb_state.Shift_Flags & STICKY_ALTGR_MASK) { move_key(RELEASE, DKY_R_ALT); } keyb_state.Shift_Flags = 0; break; case SET_MONO_SCAN_CODE: dos_slang_smart_set_mono(); break; } if (keyb_state.Shift_Flags & (SHIFT_MASK | STICKY_SHIFT_MASK)) prompt_no += 1; if (keyb_state.Shift_Flags & (CTRL_MASK | STICKY_CTRL_MASK)) prompt_no += 2; if (keyb_state.Shift_Flags & (ALT_MASK | STICKY_ALT_MASK)) prompt_no += 4; if (keyb_state.Shift_Flags & (ALTGR_MASK | STICKY_ALTGR_MASK)) prompt_no += 8; DOSemu_Keyboard_Keymap_Prompt = keymap_prompts[prompt_no]; } static const char *exitstr; static const char exitstr1[] = "Your locale (using the LANG, LC_CTYPE, or LC_ALL environment variable,\n" "e.g., en_US) or $_external_char_set setting in ~/.dosemu/.dosemurc or\n" "dosemu.conf does not match your terminal: one assumes UTF-8 and the other\n" "does not.\n" "Non-ASCII characters (\"extended ASCII\") were not displayed correctly.\n\n"; /* check for u6=\E[%i%d;%dR cursor position reply */ static int is_cursor_position_reply(int i) { int j; char pos; char *u6 = SLtt_tgetstr ("u6"); if (u6 == NULL || strcmp(u6, "\x1b[%i%d;%dR")) return 0; for (j = i+2; j < keyb_state.kbcount; j++) if (keyb_state.kbp[j] == 'R') break; if (j == keyb_state.kbcount) return 0; keyb_state.kbcount -= j + 1; if (keyb_state.kbp[i+2] != 'R') /* eat but do not report */ return 1; pos = keyb_state.kbp[j-1]; if (strstr("utf8", trconfig.output_charset->names[0]) || (strstr("default", trconfig.output_charset->names[0]) && strcmp(nl_langinfo(CODESET), "UTF-8") == 0)) { if (pos == '3' && !config.quiet) exitstr = exitstr1; } else if (pos == '2' && !config.quiet) exitstr = exitstr1; return 1; } /* checks for xterm and rxvt style modifiers in the escape sequence */ static int get_modifiers(void) { int i, mod, replacepos, modifier; /* always starts with ^[[ or ^[O */ if (keyb_state.kbcount < 4 || (keyb_state.kbp[1] != '[' && keyb_state.kbp[1] != 'O') || /* then either a digit or digits semicolon digit (or rxvt) */ !isdigit(keyb_state.kbp[2])) return 0; for (i = 3; isdigit(keyb_state.kbp[i]); i++) if (i >= keyb_state.kbcount - 1) return 0; mod = keyb_state.kbp[i]; replacepos = i; if (mod == ';') { if (isdigit(keyb_state.kbp[i+1]) && is_cursor_position_reply(i)) return -1; /* ^[[1;2A is special: get rid of "1;2" */ if (i == 3 && keyb_state.kbp[2] == '1') replacepos--; /* e.g. ^[[15;2~ */ i++; if (i >= keyb_state.kbcount - 1 || !isdigit(keyb_state.kbp[i])) return 0; } else if (mod == '$' || mod == '^' || mod == '@') { /*rxvt */ keyb_state.kbp[i] = '~'; if (mod == '$') return SHIFT_MASK; if (mod == '^') return CTRL_MASK; return CTRL_MASK | SHIFT_MASK; /* '@' */ } else { /* e.g. ^[[2A */ i--; if (i != 2) return 0; replacepos = i; } mod = (unsigned char)keyb_state.kbp[i] - '0'; if(isdigit(keyb_state.kbp[i+1])) { i++; mod = mod * 10 + keyb_state.kbp[i] - '0'; } mod--; /* after the digit must be a non-digit */ if (mod == 0 || mod > 15 || i >= keyb_state.kbcount - 1 || isdigit(keyb_state.kbp[i+1])) return 0; modifier = 0; if (mod & 1) modifier |= SHIFT_MASK; if ((mod & 2) || (mod & 8)) modifier |= ALT_MASK; if (mod & 4) modifier |= CTRL_MASK; /* now remove the modifier and figure out the unmodified string */ memmove(&keyb_state.kbp[replacepos], &keyb_state.kbp[i+1], keyb_state.kbcount - (i + 1)); keyb_state.kbcount -= (i - replacepos + 1); return modifier; } static void do_slang_pending(void) { if (keyb_state.KeyNot_Ready && (keyb_state.Keystr_Len == 1) && (*keyb_state.kbp == 27) && keyb_state.kbcount == 1) { #if 0 #define THE_TIMEOUT 750000L #else #define THE_TIMEOUT 250000L #endif hitimer_t t_dif = GETusTIME(0) - keyb_state.t_start; if (t_dif >= THE_TIMEOUT) { k_printf("KBD: slang got single ESC\n"); keyb_state.kbcount--; /* update count */ keyb_state.kbp++; slang_send_scancode(keyb_state.Shift_Flags, DKY_ESC); keyb_state.KeyNot_Ready = 0; } } /* do_slang_getkeys() throttles pasting. So we call it here in * addition to the SIGIO handler. */ if (keyb_state.kbcount) _do_slang_getkeys(); } static void do_sync_shiftstate(void) { if (old_flags & SHIFT_MASK) { move_key(RELEASE, DKY_L_SHIFT); keyb_state.Shift_Flags &= ~SHIFT_MASK; } if (old_flags & CTRL_MASK) { move_key(RELEASE, DKY_L_CTRL); keyb_state.Shift_Flags &= ~CTRL_MASK; } if (old_flags & ALT_MASK) { move_key(RELEASE, DKY_L_ALT); keyb_state.Shift_Flags &= ~ALT_MASK; } if (old_flags & ALTGR_MASK) { move_key(RELEASE, DKY_R_ALT); keyb_state.Shift_Flags &= ~ALTGR_MASK; } if (old_flags & KEYPAD_MASK) { keyb_state.Shift_Flags &= ~KEYPAD_MASK; } old_flags = 0; } static void _do_slang_getkeys(void) { SLang_Key_Type *key; int cc; int modifier = 0; cc = read_some_keys(); if (cc <= 0 && !keyb_state.kbcount) { if (old_flags == 0) return; if (old_flags & WAIT_MASK) { old_flags &= ~WAIT_MASK; return; } do_sync_shiftstate(); return; } if (cc <= 0 && keyb_state.KeyNot_Ready) return; k_printf("KBD: do_slang_getkeys()\n"); /* restore shift-state from previous keypress */ do_sync_shiftstate(); if (!keyb_state.kbcount) { do_slang_special_keys(0); return; } k_printf("KBD: do_slang_getkeys() found %d bytes\n", keyb_state.kbcount); /* Now process the keys that are buffered up */ while (keyb_state.kbcount) { unsigned long scan = 0; t_unicode symbol = DKY_VOID; size_t result; keyb_state.Keystr_Len = 0; keyb_state.KeyNot_Ready = 0; key = SLang_do_key(keyb_state.The_Normal_KeyMap, getkey_callback); slang_set_error(0); if (keyb_state.KeyNot_Ready) { k_printf("KBD: got ESC character\n"); keyb_state.t_start = GETusTIME(0); break; /* try again next time */ } if (key) { scan = (unsigned long) key->f.f | modifier; symbol = scan & 0xFFFF; } result = 1; if (symbol == DKY_VOID) { /* rough draft version don't stop here... */ result = charset_to_unicode(&keyb_state.translate_state, &symbol, keyb_state.kbp, keyb_state.kbcount); if (result != -1 && result > keyb_state.Keystr_Len) keyb_state.Keystr_Len = result; k_printf("KBD: got %08x, result=%zx\n", symbol, result); } if (key == NULL && symbol == DKY_ESC && keyb_state.Keystr_Len > 1) { int old_modifier = modifier; modifier = get_modifiers(); if (modifier != old_modifier) continue; keyb_state.kbcount--; keyb_state.kbp++; modifier = ALT_MASK; continue; } modifier = 0; if (result == -1 && (unsigned char)keyb_state.kbp[0] >= 0x80) { /* allow for high bit meta on dumb ascii terminals */ scan |= ALT_MASK; symbol = keyb_state.kbp[0] & 0x7f; } if (key == NULL && symbol != DKY_ESC) { /* undefined key --- return */ DOSemu_Slang_Show_Help = 0; keyb_state.kbcount = 0; break; } if (DOSemu_Slang_Show_Help) { DOSemu_Slang_Show_Help = 0; keyb_state.kbcount = 0; continue; } if (symbol == DKY_MOUSE) { int len = keyb_state.kbcount - keyb_state.Keystr_Len; k_printf("KDB: mouse str=%s len=%i\n", strprintable((char *)key->str + 1), keyb_state.Keystr_Len + len); if (len <= 0) break; k_printf("\t%s\n", strprintable((char *)keyb_state.kbp + keyb_state.Keystr_Len)); /* Xtermmouse support */ int pr = xtermmouse_get_event( keyb_state.kbp + keyb_state.Keystr_Len, len); if (!pr) break; keyb_state.kbcount -= keyb_state.Keystr_Len + pr; keyb_state.kbp += keyb_state.Keystr_Len + pr; continue; } keyb_state.kbcount -= keyb_state.Keystr_Len; /* update count */ keyb_state.kbp += keyb_state.Keystr_Len; k_printf("KBD: scan=%08lx Shift_Flags=%08lx str[0]=%d str='%s' len=%d\n", scan,keyb_state.Shift_Flags,key ? key->str[0] : 27, key ? strprintable((char *)key->str+1): "ESC", keyb_state.Keystr_Len); if (!(scan&0x80000000)) { slang_send_scancode(keyb_state.Shift_Flags | scan, symbol); do_slang_special_keys(0); } else { do_slang_special_keys(scan); } /* break to allow DOS code to chew the keypress */ break; } } static void do_slang_getkeys(void *arg) { _do_slang_getkeys(); } /* * DANG_BEGIN_FUNCTION setup_pc_scancode_mode * * Initialize the keyboard in pc scancode mode. * This functionality is ideal but rarely supported on a terminal. * * DANG_END_FUNCTION */ static void setup_pc_scancode_mode(void) { k_printf("entering pc scancode mode"); set_shiftstate(0); /* disable all persistent shift state */ /* enter pc scancode mode */ SLtt_write_string(SLtt_tgetstr("S4")); } /* * DANG_BEGIN_FUNCTION exit_pc_scancode_mode * * Set the terminal back to a keyboard mode other * programs can understand. * * DANG_END_FUNCTION */ static void exit_pc_scancode_mode(void) { if (keyb_state.pc_scancode_mode) { k_printf("leaving pc scancode mode"); SLtt_write_string(SLtt_tgetstr("S5")); keyb_state.pc_scancode_mode = FALSE; } } /* * DANG_BEGIN_FUNCTION do_pc_scancode_getkeys * * Set the terminal back to a keyboard mode other * programs can understand. * * DANG_END_FUNCTION */ static void do_pc_scancode_getkeys(void *arg) { if (read_some_keys() <= 0) { return; } k_printf("KBD: do_pc_scancode_getkeys() found %d bytes\n", keyb_state.kbcount); /* Now process the keys that are buffered up */ while(keyb_state.kbcount) { unsigned char ch = *(keyb_state.kbp++); keyb_state.kbcount--; put_rawkey(ch); } } /* * DANG_BEGIN_FUNCTION slang_keyb_init() * * Code is called at start up to set up the terminal line for non-raw mode. * * DANG_END_FUNCTION */ static int slang_keyb_init(void) { struct termios buf; k_printf("KBD: slang_keyb_init()\n"); /* First initialize keyb_state */ memset(&keyb_state, '\0', sizeof(keyb_state)); keyb_state.kbd_fd = -1; keyb_state.kbcount = 0; keyb_state.kbp = &keyb_state.kbbuf[0]; keyb_state.save_kbd_flags = -1; keyb_state.pc_scancode_mode = FALSE; keyb_state.The_Normal_KeyMap = (void *)0; keyb_state.Esc_Char = 0; keyb_state.erasekey = 0; keyb_state.KeyNot_Ready = TRUE; keyb_state.Keystr_Len = 0; keyb_state.Shift_Flags = 0; init_charset_state(&keyb_state.translate_state, trconfig.keyb_charset); SLtt_Force_Keypad_Init = 1; term_init(); set_shiftstate(0); if (SLtt_tgetstr("S4") && SLtt_tgetstr("S5")) { keyb_state.pc_scancode_mode = TRUE; } keyb_state.kbd_fd = STDIN_FILENO; keyb_state.save_kbd_flags = fcntl(keyb_state.kbd_fd, F_GETFL); if (tcgetattr(keyb_state.kbd_fd, &keyb_state.save_termios) < 0 && errno != EINVAL && errno != ENOTTY) { error("slang_keyb_init(): Couldn't tcgetattr(kbd_fd,...) errno=%d\n", errno); return FALSE; } buf = keyb_state.save_termios; if (keyb_state.pc_scancode_mode) { buf.c_iflag = IGNBRK; } else { buf.c_iflag &= (ISTRIP | IGNBRK | IXON | IXOFF); } buf.c_cflag &= ~(CLOCAL | CSIZE | PARENB); buf.c_cflag |= CS8; buf.c_lflag &= 0; /* ISIG */ buf.c_cc[VMIN] = 1; buf.c_cc[VTIME] = 0; keyb_state.erasekey = buf.c_cc[VERASE]; if (tcsetattr(keyb_state.kbd_fd, TCSANOW, &buf) < 0 && errno != EINVAL && errno != ENOTTY) { error("slang_keyb_init(): Couldn't tcsetattr(kbd_fd,TCSANOW,...) !\n"); return FALSE; } if (keyb_state.pc_scancode_mode) { setup_pc_scancode_mode(); add_to_io_select(keyb_state.kbd_fd, do_pc_scancode_getkeys, NULL); } else { if (-1 == init_slang_keymaps()) { error("Unable to initialize S-Lang keymaps.\n"); return FALSE; } add_to_io_select(keyb_state.kbd_fd, do_slang_getkeys, NULL); } sigalrm_register_handler(do_slang_pending); k_printf("KBD: slang_keyb_init() ok\n"); return TRUE; } static void slang_keyb_close(void) { remove_from_io_select(keyb_state.kbd_fd); exit_pc_scancode_mode(); if (tcsetattr(keyb_state.kbd_fd, TCSAFLUSH, &keyb_state.save_termios) < 0 && errno != EINVAL && errno != ENOTTY) { error("slang_keyb_close(): failed to restore keyboard termios settings!\n"); } if (keyb_state.save_kbd_flags != -1) { if (fcntl(keyb_state.kbd_fd, F_SETFL, keyb_state.save_kbd_flags) == -1) error("slang_keyb_close(): failed to restore keyboard flags!\n"); } term_close(); cleanup_charset_state(&keyb_state.translate_state); if (exitstr) printf("%s", exitstr); } /* * DANG_BEGIN_FUNCTION slang_keyb_probe() * * Code is called at start up to see if we can use the slang keyboard. * * DANG_END_FUNCTION */ static int slang_keyb_probe(void) { struct termios buf; if (config.X || config.console_keyb != KEYB_OTHER) return FALSE; if (tcgetattr(STDIN_FILENO, &buf) >= 0 || errno == EINVAL || errno == ENOTTY) return TRUE; return FALSE; } struct keyboard_client Keyboard_slang = { "slang", /* name */ slang_keyb_probe, /* probe */ slang_keyb_init, /* init */ NULL, /* reset */ slang_keyb_close, /* close */ NULL, /* set_leds */ handle_slang_keys /* handle_keys */ };