d7b71780c6d640a37c0d10663260d36726f69dcd
[platform/upstream/libxkbcommon.git] / test / keyseq.c
1 /*
2  * Copyright © 2012 Ran Benita <ran234@gmail.com>
3  *
4  * Permission is hereby granted, free of charge, to any person obtaining a
5  * copy of this software and associated documentation files (the "Software"),
6  * to deal in the Software without restriction, including without limitation
7  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8  * and/or sell copies of the Software, and to permit persons to whom the
9  * Software is furnished to do so, subject to the following conditions:
10  *
11  * The above copyright notice and this permission notice (including the next
12  * paragraph) shall be included in all copies or substantial portions of the
13  * Software.
14  *
15  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
18  * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
20  * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
21  * DEALINGS IN THE SOFTWARE.
22  */
23
24 #include <assert.h>
25 #include <stdarg.h>
26 #include <stdio.h>
27
28 #include <linux/input.h>
29
30 #include "xkbcommon/xkbcommon.h"
31 #include "test.h"
32
33 enum {
34     DOWN,
35     UP,
36     BOTH,
37     NEXT,
38     FINISH,
39 };
40
41 #define EVDEV_OFFSET 8
42
43 /*
44  * Test a sequence of keysyms, resulting from a sequence of key presses,
45  * against the keysyms they're supposed to generate.
46  *
47  * - Each test runs with a clean state.
48  * - Each line in the test is made up of:
49  *   + A keycode, given as a KEY_* from linux/input.h.
50  *   + A direction - DOWN for press, UP for release, BOTH for
51  *     immediate press + release.
52  *   + A sequence of keysyms that should result from this keypress.
53  *
54  * The vararg format is:
55  * <KEY_*>  <DOWN | UP | BOTH>  <XKB_KEY_* (zero or more)>  <NEXT | FINISH>
56  *
57  * See below for examples.
58  */
59 static int
60 test_key_seq(struct xkb_keymap *keymap, ...)
61 {
62     struct xkb_state *state;
63
64     va_list ap;
65     xkb_keycode_t kc;
66     int op;
67     xkb_keysym_t keysym;
68
69     const xkb_keysym_t *syms;
70     unsigned int nsyms, i;
71     char ksbuf[64];
72
73     state = xkb_state_new(keymap);
74     assert(state);
75
76     va_start(ap, keymap);
77
78     for (;;) {
79         kc = va_arg(ap, int) + EVDEV_OFFSET;
80         op = va_arg(ap, int);
81
82         nsyms = xkb_key_get_syms(state, kc, &syms);
83         fprintf(stderr, "got %d syms for key 0x%x: [", nsyms, kc);
84
85         if (op == DOWN || op == BOTH)
86             xkb_state_update_key(state, kc, XKB_KEY_DOWN);
87         if (op == UP || op == BOTH)
88             xkb_state_update_key(state, kc, XKB_KEY_UP);
89
90         for (i = 0; i < nsyms; i++) {
91             keysym = va_arg(ap, int);
92             xkb_keysym_get_name(syms[i], ksbuf, sizeof(ksbuf));
93             fprintf(stderr, "%s%s", (i != 0) ? ", " : "", ksbuf);
94
95             if (keysym == FINISH || keysym == NEXT) {
96                 xkb_keysym_get_name(syms[i], ksbuf, sizeof(ksbuf));
97                 fprintf(stderr, "Did not expect keysym: %s.\n", ksbuf);
98                 goto fail;
99             }
100
101             if (keysym != syms[i]) {
102                 xkb_keysym_get_name(keysym, ksbuf, sizeof(ksbuf));
103                 fprintf(stderr, "Expected keysym: %s. ", ksbuf);;
104                 xkb_keysym_get_name(syms[i], ksbuf, sizeof(ksbuf));
105                 fprintf(stderr, "Got keysym: %s.\n", ksbuf);;
106                 goto fail;
107             }
108         }
109
110         fprintf(stderr, "]\n");
111
112         keysym = va_arg(ap, int);
113         if (keysym == NEXT)
114             continue;
115         if (keysym == FINISH)
116             break;
117
118         xkb_keysym_get_name(keysym, ksbuf, sizeof(ksbuf));
119         fprintf(stderr, "Expected keysym: %s. Didn't get it.\n", ksbuf);
120         goto fail;
121     }
122
123     va_end(ap);
124     xkb_state_unref(state);
125     return 1;
126
127 fail:
128     va_end(ap);
129     xkb_state_unref(state);
130     return 0;
131 }
132
133 int
134 main(void)
135 {
136     struct xkb_context *ctx = test_get_context();
137     struct xkb_keymap *keymap;
138
139     assert(ctx);
140     keymap = test_compile_rules(ctx, "evdev", "evdev", "us,il", NULL,
141                                 "grp:alt_shift_toggle,grp:menu_toggle");
142     assert(keymap);
143
144     assert(test_key_seq(keymap,
145                         KEY_H,  BOTH,  XKB_KEY_h,  NEXT,
146                         KEY_E,  BOTH,  XKB_KEY_e,  NEXT,
147                         KEY_L,  BOTH,  XKB_KEY_l,  NEXT,
148                         KEY_L,  BOTH,  XKB_KEY_l,  NEXT,
149                         KEY_O,  BOTH,  XKB_KEY_o,  FINISH));
150
151     assert(test_key_seq(keymap,
152                         KEY_H,          BOTH,  XKB_KEY_h,        NEXT,
153                         KEY_LEFTSHIFT,  DOWN,  XKB_KEY_Shift_L,  NEXT,
154                         KEY_E,          BOTH,  XKB_KEY_E,        NEXT,
155                         KEY_L,          BOTH,  XKB_KEY_L,        NEXT,
156                         KEY_LEFTSHIFT,  UP,    XKB_KEY_Shift_L,  NEXT,
157                         KEY_L,          BOTH,  XKB_KEY_l,        NEXT,
158                         KEY_O,          BOTH,  XKB_KEY_o,        FINISH));
159
160     /* Base modifier cleared on key release... */
161     assert(test_key_seq(keymap,
162                         KEY_H,          BOTH,  XKB_KEY_h,        NEXT,
163                         KEY_LEFTSHIFT,  DOWN,  XKB_KEY_Shift_L,  NEXT,
164                         KEY_E,          BOTH,  XKB_KEY_E,        NEXT,
165                         KEY_L,          BOTH,  XKB_KEY_L,        NEXT,
166                         KEY_LEFTSHIFT,  DOWN,  XKB_KEY_Shift_L,  NEXT,
167                         KEY_L,          BOTH,  XKB_KEY_L,        NEXT,
168                         KEY_O,          BOTH,  XKB_KEY_O,        FINISH));
169
170     /* ... But only by the keycode that set it. */
171     assert(test_key_seq(keymap,
172                         KEY_H,           BOTH,  XKB_KEY_h,        NEXT,
173                         KEY_LEFTSHIFT,   DOWN,  XKB_KEY_Shift_L,  NEXT,
174                         KEY_E,           BOTH,  XKB_KEY_E,        NEXT,
175                         KEY_L,           BOTH,  XKB_KEY_L,        NEXT,
176                         KEY_RIGHTSHIFT,  UP,    XKB_KEY_Shift_R,  NEXT,
177                         KEY_L,           BOTH,  XKB_KEY_L,        NEXT,
178                         KEY_O,           BOTH,  XKB_KEY_O,        FINISH));
179
180     /*
181      * A base modifier should only be cleared when no other key affecting
182      * the modifier is down.
183      */
184     assert(test_key_seq(keymap,
185                         KEY_H,           BOTH,  XKB_KEY_h,        NEXT,
186                         KEY_LEFTSHIFT,   DOWN,  XKB_KEY_Shift_L,  NEXT,
187                         KEY_E,           BOTH,  XKB_KEY_E,        NEXT,
188                         KEY_RIGHTSHIFT,  DOWN,  XKB_KEY_Shift_R,  NEXT,
189                         KEY_L,           BOTH,  XKB_KEY_L,        NEXT,
190                         KEY_RIGHTSHIFT,  UP,    XKB_KEY_Shift_R,  NEXT,
191                         KEY_L,           BOTH,  XKB_KEY_L,        NEXT,
192                         KEY_LEFTSHIFT,   UP,    XKB_KEY_Shift_L,  NEXT,
193                         KEY_O,           BOTH,  XKB_KEY_o,        FINISH));
194
195     /* Group switching / locking. */
196     assert(test_key_seq(keymap,
197                         KEY_H,        BOTH,  XKB_KEY_h,               NEXT,
198                         KEY_E,        BOTH,  XKB_KEY_e,               NEXT,
199                         KEY_COMPOSE,  BOTH,  XKB_KEY_ISO_Next_Group,  NEXT,
200                         KEY_K,        BOTH,  XKB_KEY_hebrew_lamed,    NEXT,
201                         KEY_F,        BOTH,  XKB_KEY_hebrew_kaph,     NEXT,
202                         KEY_COMPOSE,  BOTH,  XKB_KEY_ISO_Next_Group,  NEXT,
203                         KEY_O,        BOTH,  XKB_KEY_o,               FINISH));
204
205     assert(test_key_seq(keymap,
206                         KEY_LEFTSHIFT, DOWN, XKB_KEY_Shift_L,        NEXT,
207                         KEY_LEFTALT,   DOWN, XKB_KEY_ISO_Next_Group, NEXT,
208                         KEY_LEFTALT,   UP,   XKB_KEY_ISO_Next_Group, NEXT,
209                         KEY_LEFTSHIFT, UP,   XKB_KEY_Shift_L,        FINISH));
210
211     assert(test_key_seq(keymap,
212                         KEY_LEFTALT,   DOWN, XKB_KEY_Alt_L,          NEXT,
213                         KEY_LEFTSHIFT, DOWN, XKB_KEY_ISO_Next_Group, NEXT,
214                         KEY_LEFTSHIFT, UP,   XKB_KEY_ISO_Next_Group, NEXT,
215                         KEY_LEFTALT,   UP,   XKB_KEY_Alt_L,          FINISH));
216
217     /* Locked modifiers. */
218     assert(test_key_seq(keymap,
219                         KEY_CAPSLOCK,  BOTH,  XKB_KEY_Caps_Lock,  NEXT,
220                         KEY_H,         BOTH,  XKB_KEY_H,          NEXT,
221                         KEY_E,         BOTH,  XKB_KEY_E,          NEXT,
222                         KEY_L,         BOTH,  XKB_KEY_L,          NEXT,
223                         KEY_L,         BOTH,  XKB_KEY_L,          NEXT,
224                         KEY_O,         BOTH,  XKB_KEY_O,          FINISH));
225
226     assert(test_key_seq(keymap,
227                         KEY_H,         BOTH,  XKB_KEY_h,          NEXT,
228                         KEY_E,         BOTH,  XKB_KEY_e,          NEXT,
229                         KEY_CAPSLOCK,  BOTH,  XKB_KEY_Caps_Lock,  NEXT,
230                         KEY_L,         BOTH,  XKB_KEY_L,          NEXT,
231                         KEY_L,         BOTH,  XKB_KEY_L,          NEXT,
232                         KEY_CAPSLOCK,  BOTH,  XKB_KEY_Caps_Lock,  NEXT,
233                         KEY_O,         BOTH,  XKB_KEY_o,          FINISH));
234
235     assert(test_key_seq(keymap,
236                         KEY_H,         BOTH,  XKB_KEY_h,          NEXT,
237                         KEY_CAPSLOCK,  DOWN,  XKB_KEY_Caps_Lock,  NEXT,
238                         KEY_E,         BOTH,  XKB_KEY_E,          NEXT,
239                         KEY_L,         BOTH,  XKB_KEY_L,          NEXT,
240                         KEY_L,         BOTH,  XKB_KEY_L,          NEXT,
241                         KEY_CAPSLOCK,  UP,    XKB_KEY_Caps_Lock,  NEXT,
242                         KEY_O,         BOTH,  XKB_KEY_O,          FINISH));
243
244     assert(test_key_seq(keymap,
245                         KEY_H,         BOTH,  XKB_KEY_h,          NEXT,
246                         KEY_E,         BOTH,  XKB_KEY_e,          NEXT,
247                         KEY_CAPSLOCK,  UP,    XKB_KEY_Caps_Lock,  NEXT,
248                         KEY_L,         BOTH,  XKB_KEY_l,          NEXT,
249                         KEY_L,         BOTH,  XKB_KEY_l,          NEXT,
250                         KEY_O,         BOTH,  XKB_KEY_o,          FINISH));
251
252     /*
253      * A key release affecting a locked modifier should clear it
254      * regardless of the key press.
255      */
256     /* assert(test_key_seq(keymap, */
257     /*                     KEY_H,         BOTH,  XKB_KEY_h,          NEXT, */
258     /*                     KEY_CAPSLOCK,  DOWN,  XKB_KEY_Caps_Lock,  NEXT, */
259     /*                     KEY_E,         BOTH,  XKB_KEY_E,          NEXT, */
260     /*                     KEY_L,         BOTH,  XKB_KEY_L,          NEXT, */
261     /*                     KEY_CAPSLOCK,  UP,    XKB_KEY_Caps_Lock,  NEXT, */
262     /*                     KEY_L,         BOTH,  XKB_KEY_L,          NEXT, */
263     /*                     KEY_CAPSLOCK,  UP,    XKB_KEY_Caps_Lock,  NEXT, */
264     /*                     KEY_O,         BOTH,  XKB_KEY_o,          FINISH)); */
265
266     /* Simple Num Lock sanity check. */
267     assert(test_key_seq(keymap,
268                         KEY_KP1,      BOTH,  XKB_KEY_KP_End,    NEXT,
269                         KEY_NUMLOCK,  BOTH,  XKB_KEY_Num_Lock,  NEXT,
270                         KEY_KP1,      BOTH,  XKB_KEY_KP_1,      NEXT,
271                         KEY_KP2,      BOTH,  XKB_KEY_KP_2,      NEXT,
272                         KEY_NUMLOCK,  BOTH,  XKB_KEY_Num_Lock,  NEXT,
273                         KEY_KP2,      BOTH,  XKB_KEY_KP_Down,   FINISH));
274
275     xkb_map_unref(keymap);
276     xkb_context_unref(ctx);
277     return 0;
278 }