keysym-utf: reject out-of-range Unicode codepoints in xkb_keysym_to_utf{8,32}
[platform/upstream/libxkbcommon.git] / test / keysym.c
1 /*
2  * Copyright © 2009 Dan Nicholson
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 #include <locale.h>
24
25 #include "test.h"
26 #include "keysym.h" /* For unexported is_lower/upper/keypad() */
27
28 static int
29 test_string(const char *string, xkb_keysym_t expected)
30 {
31     xkb_keysym_t keysym;
32
33     keysym = xkb_keysym_from_name(string, 0);
34
35     fprintf(stderr, "Expected string %s -> %x\n", string, expected);
36     fprintf(stderr, "Received string %s -> %x\n\n", string, keysym);
37
38     return keysym == expected;
39 }
40
41 static int
42 test_casestring(const char *string, xkb_keysym_t expected)
43 {
44     xkb_keysym_t keysym;
45
46     keysym = xkb_keysym_from_name(string, XKB_KEYSYM_CASE_INSENSITIVE);
47
48     fprintf(stderr, "Expected casestring %s -> %x\n", string, expected);
49     fprintf(stderr, "Received casestring %s -> %x\n\n", string, keysym);
50
51     return keysym == expected;
52 }
53
54 static int
55 test_keysym(xkb_keysym_t keysym, const char *expected)
56 {
57     char s[16];
58
59     xkb_keysym_get_name(keysym, s, sizeof(s));
60
61     fprintf(stderr, "Expected keysym %#x -> %s\n", keysym, expected);
62     fprintf(stderr, "Received keysym %#x -> %s\n\n", keysym, s);
63
64     return streq(s, expected);
65 }
66
67 static int
68 test_utf8(xkb_keysym_t keysym, const char *expected)
69 {
70     char s[8];
71     int ret;
72
73     ret = xkb_keysym_to_utf8(keysym, s, sizeof(s));
74     if (ret <= 0)
75         return ret;
76
77     fprintf(stderr, "Expected keysym %#x -> %s (%u bytes)\n", keysym, expected,
78             (unsigned) strlen(expected));
79     fprintf(stderr, "Received keysym %#x -> %s (%u bytes)\n\n", keysym, s,
80             (unsigned) strlen(s));
81
82     assert(expected != NULL);
83     return streq(s, expected);
84 }
85
86 static void
87 test_github_issue_42(void)
88 {
89     // Verify we are not dependent on locale, Turkish-i problem in particular.
90     if (setlocale(LC_CTYPE, "tr_TR.UTF-8") == NULL) {
91         // The locale is not available, probably; skip.
92         return;
93     }
94
95     assert(test_string("i", XKB_KEY_i));
96     assert(test_string("I", XKB_KEY_I));
97     assert(test_casestring("i", XKB_KEY_i));
98     assert(test_casestring("I", XKB_KEY_i));
99     assert(xkb_keysym_to_upper(XKB_KEY_i) == XKB_KEY_I);
100     assert(xkb_keysym_to_lower(XKB_KEY_I) == XKB_KEY_i);
101
102     setlocale(LC_CTYPE, "C");
103 }
104
105 int
106 main(void)
107 {
108     assert(test_string("Undo", 0xFF65));
109     assert(test_string("ThisKeyShouldNotExist", XKB_KEY_NoSymbol));
110     assert(test_string("XF86_Switch_VT_5", 0x1008FE05));
111     assert(test_string("VoidSymbol", 0xFFFFFF));
112     assert(test_string("U4567", 0x1004567));
113     assert(test_string("0x10203040", 0x10203040));
114     assert(test_string("a", 0x61));
115     assert(test_string("A", 0x41));
116     assert(test_string("ch", 0xfea0));
117     assert(test_string("Ch", 0xfea1));
118     assert(test_string("CH", 0xfea2));
119     assert(test_string("THORN", 0x00de));
120     assert(test_string("Thorn", 0x00de));
121     assert(test_string("thorn", 0x00fe));
122
123     assert(test_keysym(0x1008FF56, "XF86Close"));
124     assert(test_keysym(0x0, "NoSymbol"));
125     assert(test_keysym(0x1008FE20, "XF86Ungrab"));
126     assert(test_keysym(0x01001234, "U1234"));
127     /* 16-bit unicode padded to width 4. */
128     assert(test_keysym(0x010002DE, "U02DE"));
129     /* 32-bit unicode padded to width 8. */
130     assert(test_keysym(0x0101F4A9, "U0001F4A9"));
131
132     assert(test_casestring("Undo", 0xFF65));
133     assert(test_casestring("UNDO", 0xFF65));
134     assert(test_casestring("A", 0x61));
135     assert(test_casestring("a", 0x61));
136     assert(test_casestring("ThisKeyShouldNotExist", XKB_KEY_NoSymbol));
137     assert(test_casestring("XF86_Switch_vT_5", 0x1008FE05));
138     assert(test_casestring("xF86_SwitcH_VT_5", 0x1008FE05));
139     assert(test_casestring("xF86SwiTch_VT_5", 0x1008FE05));
140     assert(test_casestring("xF86Switch_vt_5", 0x1008FE05));
141     assert(test_casestring("VoidSymbol", 0xFFFFFF));
142     assert(test_casestring("vOIDsymBol", 0xFFFFFF));
143     assert(test_casestring("U4567", 0x1004567));
144     assert(test_casestring("u4567", 0x1004567));
145     assert(test_casestring("0x10203040", 0x10203040));
146     assert(test_casestring("0X10203040", 0x10203040));
147     assert(test_casestring("THORN", 0x00fe));
148     assert(test_casestring("Thorn", 0x00fe));
149     assert(test_casestring("thorn", 0x00fe));
150
151     assert(test_utf8(XKB_KEY_y, "y"));
152     assert(test_utf8(XKB_KEY_u, "u"));
153     assert(test_utf8(XKB_KEY_m, "m"));
154     assert(test_utf8(XKB_KEY_Cyrillic_em, "м"));
155     assert(test_utf8(XKB_KEY_Cyrillic_u, "у"));
156     assert(test_utf8(XKB_KEY_exclam, "!"));
157     assert(test_utf8(XKB_KEY_oslash, "ø"));
158     assert(test_utf8(XKB_KEY_hebrew_aleph, "א"));
159     assert(test_utf8(XKB_KEY_Arabic_sheen, "ش"));
160
161     assert(test_utf8(XKB_KEY_space, " "));
162     assert(test_utf8(XKB_KEY_KP_Space, " "));
163     assert(test_utf8(XKB_KEY_BackSpace, "\b"));
164     assert(test_utf8(XKB_KEY_Escape, "\033"));
165     assert(test_utf8(XKB_KEY_KP_Separator, ","));
166     assert(test_utf8(XKB_KEY_KP_Decimal, "."));
167     assert(test_utf8(XKB_KEY_Tab, "\t"));
168     assert(test_utf8(XKB_KEY_KP_Tab, "\t"));
169     assert(test_utf8(XKB_KEY_hyphen, "­"));
170     assert(test_utf8(XKB_KEY_Linefeed, "\n"));
171     assert(test_utf8(XKB_KEY_Return, "\r"));
172     assert(test_utf8(XKB_KEY_KP_Enter, "\r"));
173     assert(test_utf8(XKB_KEY_KP_Equal, "="));
174     assert(test_utf8(XKB_KEY_9, "9"));
175     assert(test_utf8(XKB_KEY_KP_9, "9"));
176     assert(test_utf8(XKB_KEY_KP_Multiply, "*"));
177     assert(test_utf8(XKB_KEY_KP_Subtract, "-"));
178
179     assert(test_utf8(0x10005d0, "א"));
180     assert(test_utf8(0x110ffff, "\xf4\x8f\xbf\xbf"));
181     assert(test_utf8(0x1110000, NULL) == 0);
182
183     assert(xkb_keysym_is_lower(XKB_KEY_a));
184     assert(xkb_keysym_is_lower(XKB_KEY_Greek_lambda));
185     assert(xkb_keysym_is_lower(xkb_keysym_from_name("U03b1", 0))); /* GREEK SMALL LETTER ALPHA */
186     assert(xkb_keysym_is_lower(xkb_keysym_from_name("U03af", 0))); /* GREEK SMALL LETTER IOTA WITH TONOS */
187
188     assert(xkb_keysym_is_upper(XKB_KEY_A));
189     assert(xkb_keysym_is_upper(XKB_KEY_Greek_LAMBDA));
190     assert(xkb_keysym_is_upper(xkb_keysym_from_name("U0391", 0))); /* GREEK CAPITAL LETTER ALPHA */
191     assert(xkb_keysym_is_upper(xkb_keysym_from_name("U0388", 0))); /* GREEK CAPITAL LETTER EPSILON WITH TONOS */
192
193     assert(!xkb_keysym_is_upper(XKB_KEY_a));
194     assert(!xkb_keysym_is_lower(XKB_KEY_A));
195     assert(!xkb_keysym_is_lower(XKB_KEY_Return));
196     assert(!xkb_keysym_is_upper(XKB_KEY_Return));
197     assert(!xkb_keysym_is_lower(XKB_KEY_hebrew_aleph));
198     assert(!xkb_keysym_is_upper(XKB_KEY_hebrew_aleph));
199     assert(!xkb_keysym_is_upper(xkb_keysym_from_name("U05D0", 0))); /* HEBREW LETTER ALEF */
200     assert(!xkb_keysym_is_lower(xkb_keysym_from_name("U05D0", 0))); /* HEBREW LETTER ALEF */
201     assert(!xkb_keysym_is_lower(XKB_KEY_8));
202     assert(!xkb_keysym_is_upper(XKB_KEY_8));
203
204     assert(xkb_keysym_is_keypad(XKB_KEY_KP_Enter));
205     assert(xkb_keysym_is_keypad(XKB_KEY_KP_6));
206     assert(xkb_keysym_is_keypad(XKB_KEY_KP_Add));
207     assert(!xkb_keysym_is_keypad(XKB_KEY_Num_Lock));
208     assert(!xkb_keysym_is_keypad(XKB_KEY_1));
209     assert(!xkb_keysym_is_keypad(XKB_KEY_Return));
210
211     assert(xkb_keysym_to_upper(XKB_KEY_a) == XKB_KEY_A);
212     assert(xkb_keysym_to_upper(XKB_KEY_A) == XKB_KEY_A);
213     assert(xkb_keysym_to_lower(XKB_KEY_a) == XKB_KEY_a);
214     assert(xkb_keysym_to_lower(XKB_KEY_A) == XKB_KEY_a);
215     assert(xkb_keysym_to_upper(XKB_KEY_Return) == XKB_KEY_Return);
216     assert(xkb_keysym_to_lower(XKB_KEY_Return) == XKB_KEY_Return);
217     assert(xkb_keysym_to_upper(XKB_KEY_Greek_lambda) == XKB_KEY_Greek_LAMBDA);
218     assert(xkb_keysym_to_upper(XKB_KEY_Greek_LAMBDA) == XKB_KEY_Greek_LAMBDA);
219     assert(xkb_keysym_to_lower(XKB_KEY_Greek_lambda) == XKB_KEY_Greek_lambda);
220     assert(xkb_keysym_to_lower(XKB_KEY_Greek_LAMBDA) == XKB_KEY_Greek_lambda);
221     assert(xkb_keysym_to_upper(XKB_KEY_eacute) == XKB_KEY_Eacute);
222     assert(xkb_keysym_to_lower(XKB_KEY_Eacute) == XKB_KEY_eacute);
223
224     test_github_issue_42();
225
226     return 0;
227 }