Convert rest of names in xkb_keymap back to atoms
[platform/upstream/libxkbcommon.git] / src / xkbcomp / vmod.c
1 /************************************************************
2  * Copyright (c) 1994 by Silicon Graphics Computer Systems, Inc.
3  *
4  * Permission to use, copy, modify, and distribute this
5  * software and its documentation for any purpose and without
6  * fee is hereby granted, provided that the above copyright
7  * notice appear in all copies and that both that copyright
8  * notice and this permission notice appear in supporting
9  * documentation, and that the name of Silicon Graphics not be
10  * used in advertising or publicity pertaining to distribution
11  * of the software without specific prior written permission.
12  * Silicon Graphics makes no representation about the suitability
13  * of this software for any purpose. It is provided "as is"
14  * without any express or implied warranty.
15  *
16  * SILICON GRAPHICS DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS
17  * SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
18  * AND FITNESS FOR A PARTICULAR PURPOSE. IN NO EVENT SHALL SILICON
19  * GRAPHICS BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL
20  * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE,
21  * DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
22  * OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION  WITH
23  * THE USE OR PERFORMANCE OF THIS SOFTWARE.
24  *
25  ********************************************************/
26
27 #include "xkbcomp-priv.h"
28 #include "expr.h"
29 #include "vmod.h"
30
31 void
32 InitVModInfo(VModInfo *info, struct xkb_keymap *keymap)
33 {
34     xkb_group_index_t i;
35
36     info->defined = info->available = 0;
37
38     for (i = 0; i < XkbNumVirtualMods; i++)
39         if (keymap->vmod_names[i])
40             info->defined |= (1 << i);
41 }
42
43 void
44 ClearVModInfo(VModInfo *info)
45 {
46     info->defined = info->available = 0;
47 }
48
49 bool
50 HandleVModDef(VModDef *stmt, struct xkb_keymap *keymap,
51               enum merge_mode mergeMode, VModInfo *info)
52 {
53     xkb_mod_index_t i;
54     int nextFree;
55     xkb_mod_mask_t bit;
56
57     if (stmt->value)
58         log_err(keymap->ctx,
59                 "Support for setting a value in a virtual_modifiers statement has been removed; "
60                 "Value ignored\n");
61
62     nextFree = -1;
63     for (i = 0, bit = 1; i < XkbNumVirtualMods; i++, bit <<= 1) {
64         if (!(info->defined & bit)) {
65             if (nextFree < 0)
66                 nextFree = i;
67             continue;
68         }
69
70         /* Already defined. */
71         if (!keymap->vmod_names[i])
72             continue;
73
74         if (keymap->vmod_names[i] != stmt->name)
75             continue;
76
77         info->available |= bit;
78         return true;
79     }
80
81     if (nextFree < 0) {
82         log_err(keymap->ctx,
83                 "Too many virtual modifiers defined (maximum %d)\n",
84                 XkbNumVirtualMods);
85         return false;
86     }
87
88     info->defined |= (1 << nextFree);
89     info->available |= (1 << nextFree);
90
91     keymap->vmod_names[nextFree] = stmt->name;
92     return true;
93 }
94
95 static bool
96 LookupVModIndex(const struct xkb_keymap *keymap, xkb_atom_t field,
97                 enum expr_value_type type, xkb_mod_index_t *val_rtrn)
98 {
99     xkb_mod_index_t i;
100
101     if (type != EXPR_TYPE_INT)
102         return false;
103
104     for (i = 0; i < XkbNumVirtualMods; i++) {
105         if (keymap->vmod_names[i] == field) {
106             *val_rtrn = i;
107             return true;
108         }
109     }
110
111     return false;
112 }
113
114 bool
115 LookupVModMask(struct xkb_context *ctx, const void *priv, xkb_atom_t field,
116                enum expr_value_type type, xkb_mod_mask_t *val_rtrn)
117 {
118     xkb_mod_index_t ndx;
119
120     if (LookupModMask(ctx, NULL, field, type, val_rtrn)) {
121         return true;
122     }
123     else if (LookupVModIndex(priv, field, type, &ndx)) {
124         *val_rtrn = (1 << (XkbNumModifiers + ndx));
125         return true;
126     }
127
128     return false;
129 }
130
131 bool
132 ResolveVirtualModifier(ExprDef *def, struct xkb_keymap *keymap,
133                        xkb_mod_index_t *ndx_rtrn, VModInfo *info)
134 {
135     xkb_mod_index_t i;
136     xkb_atom_t name = def->value.str;
137
138     if (def->op != EXPR_IDENT) {
139         log_err(keymap->ctx,
140                 "Cannot resolve virtual modifier: "
141                 "found %s where a virtual modifier name was expected\n",
142                 expr_op_type_to_string(def->op));
143         return false;
144     }
145
146     for (i = 0; i < XkbNumVirtualMods; i++) {
147         if ((info->available & (1 << i)) && keymap->vmod_names[i] == name) {
148             *ndx_rtrn = i;
149             return true;
150         }
151     }
152
153     log_err(keymap->ctx,
154             "Cannot resolve virtual modifier: "
155             "\"%s\" was not previously declared\n",
156             xkb_atom_text(keymap->ctx, name));
157     return false;
158 }