1 /* 2 * QEMU keysym to keycode conversion using rdesktop keymaps 3 * 4 * Copyright (c) 2004 Johannes Schindelin 5 * 6 * Permission is hereby granted, free of charge, to any person obtaining a copy 7 * of this software and associated documentation files (the "Software"), to deal 8 * in the Software without restriction, including without limitation the rights 9 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 10 * copies of the Software, and to permit persons to whom the Software is 11 * furnished to do so, subject to the following conditions: 12 * 13 * The above copyright notice and this permission notice shall be included in 14 * all copies or substantial portions of the Software. 15 * 16 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 17 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 18 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL 19 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 20 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 21 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN 22 * THE SOFTWARE. 23 */ 24 25 #include "qemu/osdep.h" 26 #include "keymaps.h" 27 #include "sysemu/sysemu.h" 28 #include "trace.h" 29 #include "qemu/error-report.h" 30 31 struct key_range { 32 int start; 33 int end; 34 struct key_range *next; 35 }; 36 37 struct keysym2code { 38 uint16_t keycode; 39 }; 40 41 struct kbd_layout_t { 42 GHashTable *hash; 43 struct key_range *keypad_range; 44 struct key_range *numlock_range; 45 }; 46 47 static int get_keysym(const name2keysym_t *table, 48 const char *name) 49 { 50 const name2keysym_t *p; 51 for(p = table; p->name != NULL; p++) { 52 if (!strcmp(p->name, name)) { 53 return p->keysym; 54 } 55 } 56 if (name[0] == 'U' && strlen(name) == 5) { /* try unicode Uxxxx */ 57 char *end; 58 int ret = (int)strtoul(name + 1, &end, 16); 59 if (*end == '\0' && ret > 0) { 60 return ret; 61 } 62 } 63 return 0; 64 } 65 66 67 static void add_to_key_range(struct key_range **krp, int code) { 68 struct key_range *kr; 69 for (kr = *krp; kr; kr = kr->next) { 70 if (code >= kr->start && code <= kr->end) { 71 break; 72 } 73 if (code == kr->start - 1) { 74 kr->start--; 75 break; 76 } 77 if (code == kr->end + 1) { 78 kr->end++; 79 break; 80 } 81 } 82 if (kr == NULL) { 83 kr = g_malloc0(sizeof(*kr)); 84 kr->start = kr->end = code; 85 kr->next = *krp; 86 *krp = kr; 87 } 88 } 89 90 static void add_keysym(char *line, int keysym, int keycode, kbd_layout_t *k) 91 { 92 struct keysym2code *keysym2code; 93 94 keysym2code = g_hash_table_lookup(k->hash, GINT_TO_POINTER(keysym)); 95 if (keysym2code) { 96 return; 97 } 98 99 keysym2code = g_new0(struct keysym2code, 1); 100 keysym2code->keycode = keycode; 101 g_hash_table_replace(k->hash, GINT_TO_POINTER(keysym), keysym2code); 102 trace_keymap_add(keysym, keycode, line); 103 } 104 105 static kbd_layout_t *parse_keyboard_layout(const name2keysym_t *table, 106 const char *language, 107 kbd_layout_t *k) 108 { 109 FILE *f; 110 char * filename; 111 char line[1024]; 112 char keyname[64]; 113 int len; 114 115 filename = qemu_find_file(QEMU_FILE_TYPE_KEYMAP, language); 116 trace_keymap_parse(filename); 117 f = filename ? fopen(filename, "r") : NULL; 118 g_free(filename); 119 if (!f) { 120 fprintf(stderr, "Could not read keymap file: '%s'\n", language); 121 return NULL; 122 } 123 124 if (!k) { 125 k = g_new0(kbd_layout_t, 1); 126 k->hash = g_hash_table_new(NULL, NULL); 127 } 128 129 for(;;) { 130 if (fgets(line, 1024, f) == NULL) { 131 break; 132 } 133 len = strlen(line); 134 if (len > 0 && line[len - 1] == '\n') { 135 line[len - 1] = '\0'; 136 } 137 if (line[0] == '#') { 138 continue; 139 } 140 if (!strncmp(line, "map ", 4)) { 141 continue; 142 } 143 if (!strncmp(line, "include ", 8)) { 144 parse_keyboard_layout(table, line + 8, k); 145 } else { 146 int offset = 0; 147 while (line[offset] != 0 && 148 line[offset] != ' ' && 149 offset < sizeof(keyname) - 1) { 150 keyname[offset] = line[offset]; 151 offset++; 152 } 153 keyname[offset] = 0; 154 if (strlen(keyname)) { 155 int keysym; 156 keysym = get_keysym(table, keyname); 157 if (keysym == 0) { 158 /* warn_report("unknown keysym %s", line);*/ 159 } else { 160 const char *rest = line + offset + 1; 161 int keycode = strtol(rest, NULL, 0); 162 163 if (strstr(rest, "numlock")) { 164 add_to_key_range(&k->keypad_range, keycode); 165 add_to_key_range(&k->numlock_range, keysym); 166 /* fprintf(stderr, "keypad keysym %04x keycode %d\n", 167 keysym, keycode); */ 168 } 169 170 if (strstr(rest, "shift")) { 171 keycode |= SCANCODE_SHIFT; 172 } 173 if (strstr(rest, "altgr")) { 174 keycode |= SCANCODE_ALTGR; 175 } 176 if (strstr(rest, "ctrl")) { 177 keycode |= SCANCODE_CTRL; 178 } 179 180 add_keysym(line, keysym, keycode, k); 181 182 if (strstr(rest, "addupper")) { 183 char *c; 184 for (c = keyname; *c; c++) { 185 *c = qemu_toupper(*c); 186 } 187 keysym = get_keysym(table, keyname); 188 if (keysym) { 189 add_keysym(line, keysym, 190 keycode | SCANCODE_SHIFT, k); 191 } 192 } 193 } 194 } 195 } 196 } 197 fclose(f); 198 return k; 199 } 200 201 202 kbd_layout_t *init_keyboard_layout(const name2keysym_t *table, 203 const char *language) 204 { 205 return parse_keyboard_layout(table, language, NULL); 206 } 207 208 209 int keysym2scancode(kbd_layout_t *k, int keysym) 210 { 211 struct keysym2code *keysym2code; 212 213 #ifdef XK_ISO_Left_Tab 214 if (keysym == XK_ISO_Left_Tab) { 215 keysym = XK_Tab; 216 } 217 #endif 218 219 keysym2code = g_hash_table_lookup(k->hash, GINT_TO_POINTER(keysym)); 220 if (!keysym2code) { 221 trace_keymap_unmapped(keysym); 222 warn_report("no scancode found for keysym %d", keysym); 223 return 0; 224 } 225 226 return keysym2code->keycode; 227 } 228 229 int keycode_is_keypad(kbd_layout_t *k, int keycode) 230 { 231 struct key_range *kr; 232 233 for (kr = k->keypad_range; kr; kr = kr->next) { 234 if (keycode >= kr->start && keycode <= kr->end) { 235 return 1; 236 } 237 } 238 return 0; 239 } 240 241 int keysym_is_numlock(kbd_layout_t *k, int keysym) 242 { 243 struct key_range *kr; 244 245 for (kr = k->numlock_range; kr; kr = kr->next) { 246 if (keysym >= kr->start && keysym <= kr->end) { 247 return 1; 248 } 249 } 250 return 0; 251 } 252