test: Add environment checking to rulescomp
[platform/upstream/libxkbcommon.git] / test / rulescomp.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
24 #include <time.h>
25
26 #include "test.h"
27
28 #define BENCHMARK_ITERATIONS 1000
29
30 static int
31 test_rmlvo(struct xkb_context *context, const char *rules,
32            const char *model, const char *layout,
33            const char *variant, const char *options)
34 {
35     struct xkb_keymap *keymap;
36
37     keymap = test_compile_rules(context, rules, model, layout, variant,
38                                 options);
39     if (keymap) {
40         fprintf(stderr, "Compiled '%s' '%s' '%s' '%s' '%s'\n",
41                 strnull(rules), strnull(model), strnull(layout),
42                 strnull(variant), strnull(options));
43         xkb_keymap_unref(keymap);
44     }
45
46     return keymap != NULL;
47 }
48
49 static int
50 test_rmlvo_silent(struct xkb_context *context, const char *rules,
51                   const char *model, const char *layout,
52                   const char *variant, const char *options)
53 {
54     struct xkb_keymap *keymap;
55
56     keymap = test_compile_rules(context, rules, model, layout, variant,
57                                 options);
58     if (keymap)
59         xkb_keymap_unref(keymap);
60
61     return keymap != NULL;
62 }
63
64 static void
65 benchmark(struct xkb_context *context)
66 {
67     struct timespec start, stop, elapsed;
68     enum xkb_log_level old_level = xkb_context_get_log_level(context);
69     int old_verb = xkb_context_get_log_verbosity(context);
70     int i;
71
72     xkb_context_set_log_level(context, XKB_LOG_LEVEL_CRITICAL);
73     xkb_context_set_log_verbosity(context, 0);
74
75     clock_gettime(CLOCK_MONOTONIC, &start);
76     for (i = 0; i < BENCHMARK_ITERATIONS; i++)
77         assert(test_rmlvo_silent(context, "evdev", "evdev", "us", "", ""));
78     clock_gettime(CLOCK_MONOTONIC, &stop);
79
80     xkb_context_set_log_level(context, old_level);
81     xkb_context_set_log_verbosity(context, old_verb);
82
83     elapsed.tv_sec = stop.tv_sec - start.tv_sec;
84     elapsed.tv_nsec = stop.tv_nsec - start.tv_nsec;
85     if (elapsed.tv_nsec < 0) {
86         elapsed.tv_nsec += 1000000000;
87         elapsed.tv_sec--;
88     }
89
90     fprintf(stderr, "compiled %d keymaps in %ld.%09lds\n",
91             BENCHMARK_ITERATIONS, elapsed.tv_sec, elapsed.tv_nsec);
92 }
93
94 static int
95 test_rmlvo_env(struct xkb_context *ctx, const char *rules, const char *model,
96                const char *layout, const char *variant, const char *options)
97 {
98     if (!isempty(rules))
99         setenv("XKB_DEFAULT_RULES", rules, 1);
100     else
101         unsetenv("XKB_DEFAULT_RULES");
102
103     if (!isempty(model))
104         setenv("XKB_DEFAULT_MODEL", model, 1);
105     else
106         unsetenv("XKB_DEFAULT_MODEL");
107
108     if (!isempty(layout))
109         setenv("XKB_DEFAULT_LAYOUT", layout, 1);
110     else
111         unsetenv("XKB_DEFAULT_LAYOUT");
112
113     if (!isempty(variant))
114         setenv("XKB_DEFAULT_VARIANT", variant, 1);
115     else
116         unsetenv("XKB_DEFAULT_VARIANT");
117
118     if (!isempty(options))
119         setenv("XKB_DEFAULT_OPTIONS", options, 1);
120     else
121         unsetenv("XKB_DEFAULT_OPTIONS");
122
123     return test_rmlvo(ctx, NULL, NULL, NULL, NULL, NULL);
124 }
125
126 int
127 main(int argc, char *argv[])
128 {
129     struct xkb_context *ctx = test_get_context(CONTEXT_ALLOW_ENVIRONMENT_NAMES);
130
131     assert(ctx);
132
133     if (argc > 1 && streq(argv[1], "bench")) {
134         benchmark(ctx);
135         return 0;
136     }
137
138     assert(test_rmlvo(ctx, "evdev", "pc105", "us,il,ru,ca", ",,,multix", "grp:alts_toggle,ctrl:nocaps,compose:rwin"));
139     assert(test_rmlvo(ctx, "base",  "pc105", "us,in", "", ""));
140     assert(test_rmlvo(ctx, "evdev", "pc105", "us", "intl", ""));
141     assert(test_rmlvo(ctx, "evdev", "evdev", "us", "intl", "grp:alts_toggle"));
142
143     /* 20 is not a legal group; make sure this is handled gracefully. */
144     assert(test_rmlvo(ctx, "evdev", "", "us:20", "", ""));
145
146     /* Don't choke on missing values in RMLVO. Should just skip them. */
147     assert(test_rmlvo(ctx, "evdev", "", "us,,ca", "", ""));
148
149     assert(test_rmlvo(ctx, "", "", "", "", ""));
150
151     assert(!test_rmlvo(ctx, "does-not-exist", "", "", "", ""));
152
153     assert(test_rmlvo_env(ctx, "evdev", "", "us", "", ""));
154     assert(test_rmlvo_env(ctx, "evdev", "", "us", "", "ctrl:nocaps"));
155     assert(test_rmlvo_env(ctx, "evdev", "", "us,ca", ",,,multix", "grp:alts_toggle"));
156     assert(!test_rmlvo_env(ctx, "broken", "what-on-earth", "invalid", "", ""));
157
158     xkb_context_unref(ctx);
159
160     ctx = test_get_context(0);
161     assert(test_rmlvo_env(ctx, "broken", "but", "ignored", "per", "ctx flags"));
162
163     /* Test response to invalid flags. */
164     {
165         struct xkb_rule_names rmlvo = { NULL };
166         assert(!xkb_keymap_new_from_names(ctx, &rmlvo, -1));
167         assert(!xkb_keymap_new_from_names(ctx, &rmlvo, 5453));
168     }
169
170     xkb_context_unref(ctx);
171 }