xref: /openbmc/qemu/ui/keymaps.c (revision d713e3fd4c1da510a31313e1d3cc51c89b8036f3)
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