keymap-dump: unbreak some complex lines
[platform/upstream/libxkbcommon.git] / src / xkbcomp / ast-build.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 /*
28  * Copyright © 2012 Intel Corporation
29  * Copyright © 2012 Ran Benita <ran234@gmail.com>
30  *
31  * Permission is hereby granted, free of charge, to any person obtaining a
32  * copy of this software and associated documentation files (the "Software"),
33  * to deal in the Software without restriction, including without limitation
34  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
35  * and/or sell copies of the Software, and to permit persons to whom the
36  * Software is furnished to do so, subject to the following conditions:
37  *
38  * The above copyright notice and this permission notice (including the next
39  * paragraph) shall be included in all copies or substantial portions of the
40  * Software.
41  *
42  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
43  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
44  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
45  * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
46  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
47  * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
48  * DEALINGS IN THE SOFTWARE.
49  *
50  * Author: Daniel Stone <daniel@fooishbar.org>
51  *         Ran Benita <ran234@gmail.com>
52  */
53
54 #include "xkbcomp-priv.h"
55 #include "ast-build.h"
56 #include "include.h"
57
58 ParseCommon *
59 AppendStmt(ParseCommon *to, ParseCommon *append)
60 {
61     ParseCommon *iter;
62
63     if (!to)
64         return append;
65
66     for (iter = to; iter->next; iter = iter->next);
67
68     iter->next = append;
69     return to;
70 }
71
72 static ExprDef *
73 ExprCreate(enum expr_op_type op, enum expr_value_type type, size_t size)
74 {
75     ExprDef *expr = malloc(size);
76     if (!expr)
77         return NULL;
78
79     expr->common.type = STMT_EXPR;
80     expr->common.next = NULL;
81     expr->expr.op = op;
82     expr->expr.value_type = type;
83
84     return expr;
85 }
86
87 #define EXPR_CREATE(type_, name_, op_, value_type_) \
88     ExprDef *name_ = ExprCreate(op_, value_type_, sizeof(type_)); \
89     if (!name_) \
90         return NULL;
91
92 ExprDef *
93 ExprCreateString(xkb_atom_t str)
94 {
95     EXPR_CREATE(ExprString, expr, EXPR_VALUE, EXPR_TYPE_STRING);
96     expr->string.str = str;
97     return expr;
98 }
99
100 ExprDef *
101 ExprCreateInteger(int ival)
102 {
103     EXPR_CREATE(ExprInteger, expr, EXPR_VALUE, EXPR_TYPE_INT);
104     expr->integer.ival = ival;
105     return expr;
106 }
107
108 ExprDef *
109 ExprCreateBoolean(bool set)
110 {
111     EXPR_CREATE(ExprBoolean, expr, EXPR_VALUE, EXPR_TYPE_BOOLEAN);
112     expr->boolean.set = set;
113     return expr;
114 }
115
116 ExprDef *
117 ExprCreateKeyName(xkb_atom_t key_name)
118 {
119     EXPR_CREATE(ExprKeyName, expr, EXPR_VALUE, EXPR_TYPE_KEYNAME);
120     expr->key_name.key_name = key_name;
121     return expr;
122 }
123
124 ExprDef *
125 ExprCreateIdent(xkb_atom_t ident)
126 {
127     EXPR_CREATE(ExprIdent, expr, EXPR_IDENT, EXPR_TYPE_UNKNOWN);
128     expr->ident.ident = ident;
129     return expr;
130 }
131
132 ExprDef *
133 ExprCreateUnary(enum expr_op_type op, enum expr_value_type type,
134                 ExprDef *child)
135 {
136     EXPR_CREATE(ExprUnary, expr, op, type);
137     expr->unary.child = child;
138     return expr;
139 }
140
141 ExprDef *
142 ExprCreateBinary(enum expr_op_type op, ExprDef *left, ExprDef *right)
143 {
144     EXPR_CREATE(ExprBinary, expr, op, EXPR_TYPE_UNKNOWN);
145
146     if (op == EXPR_ASSIGN || left->expr.value_type == EXPR_TYPE_UNKNOWN)
147         expr->expr.value_type = right->expr.value_type;
148     else if (left->expr.value_type == right->expr.value_type ||
149              right->expr.value_type == EXPR_TYPE_UNKNOWN)
150         expr->expr.value_type = left->expr.value_type;
151     expr->binary.left = left;
152     expr->binary.right = right;
153
154     return expr;
155 }
156
157 ExprDef *
158 ExprCreateFieldRef(xkb_atom_t element, xkb_atom_t field)
159 {
160     EXPR_CREATE(ExprFieldRef, expr, EXPR_FIELD_REF, EXPR_TYPE_UNKNOWN);
161     expr->field_ref.element = element;
162     expr->field_ref.field = field;
163     return expr;
164 }
165
166 ExprDef *
167 ExprCreateArrayRef(xkb_atom_t element, xkb_atom_t field, ExprDef *entry)
168 {
169     EXPR_CREATE(ExprArrayRef, expr, EXPR_ARRAY_REF, EXPR_TYPE_UNKNOWN);
170     expr->array_ref.element = element;
171     expr->array_ref.field = field;
172     expr->array_ref.entry = entry;
173     return expr;
174 }
175
176 ExprDef *
177 ExprCreateAction(xkb_atom_t name, ExprDef *args)
178 {
179     EXPR_CREATE(ExprAction, expr, EXPR_ACTION_DECL, EXPR_TYPE_UNKNOWN);
180     expr->action.name = name;
181     expr->action.args = args;
182     return expr;
183 }
184
185 ExprDef *
186 ExprCreateKeysymList(xkb_keysym_t sym)
187 {
188     EXPR_CREATE(ExprKeysymList, expr, EXPR_KEYSYM_LIST, EXPR_TYPE_SYMBOLS);
189
190     darray_init(expr->keysym_list.syms);
191     darray_init(expr->keysym_list.symsMapIndex);
192     darray_init(expr->keysym_list.symsNumEntries);
193
194     darray_append(expr->keysym_list.syms, sym);
195     darray_append(expr->keysym_list.symsMapIndex, 0);
196     darray_append(expr->keysym_list.symsNumEntries, 1);
197
198     return expr;
199 }
200
201 ExprDef *
202 ExprCreateMultiKeysymList(ExprDef *expr)
203 {
204     unsigned nLevels = darray_size(expr->keysym_list.symsMapIndex);
205
206     darray_resize(expr->keysym_list.symsMapIndex, 1);
207     darray_resize(expr->keysym_list.symsNumEntries, 1);
208     darray_item(expr->keysym_list.symsMapIndex, 0) = 0;
209     darray_item(expr->keysym_list.symsNumEntries, 0) = nLevels;
210
211     return expr;
212 }
213
214 ExprDef *
215 ExprAppendKeysymList(ExprDef *expr, xkb_keysym_t sym)
216 {
217     unsigned nSyms = darray_size(expr->keysym_list.syms);
218
219     darray_append(expr->keysym_list.symsMapIndex, nSyms);
220     darray_append(expr->keysym_list.symsNumEntries, 1);
221     darray_append(expr->keysym_list.syms, sym);
222
223     return expr;
224 }
225
226 ExprDef *
227 ExprAppendMultiKeysymList(ExprDef *expr, ExprDef *append)
228 {
229     unsigned nSyms = darray_size(expr->keysym_list.syms);
230     unsigned numEntries = darray_size(append->keysym_list.syms);
231
232     darray_append(expr->keysym_list.symsMapIndex, nSyms);
233     darray_append(expr->keysym_list.symsNumEntries, numEntries);
234     darray_append_items(expr->keysym_list.syms,
235                         darray_mem(append->keysym_list.syms, 0), numEntries);
236
237     darray_resize(append->keysym_list.syms, 0);
238     FreeStmt(&append->common);
239
240     return expr;
241 }
242
243 KeycodeDef *
244 KeycodeCreate(xkb_atom_t name, int64_t value)
245 {
246     KeycodeDef *def = malloc(sizeof(*def));
247     if (!def)
248         return NULL;
249
250     def->common.type = STMT_KEYCODE;
251     def->common.next = NULL;
252     def->name = name;
253     def->value = value;
254
255     return def;
256 }
257
258 KeyAliasDef *
259 KeyAliasCreate(xkb_atom_t alias, xkb_atom_t real)
260 {
261     KeyAliasDef *def = malloc(sizeof(*def));
262     if (!def)
263         return NULL;
264
265     def->common.type = STMT_ALIAS;
266     def->common.next = NULL;
267     def->alias = alias;
268     def->real = real;
269
270     return def;
271 }
272
273 VModDef *
274 VModCreate(xkb_atom_t name, ExprDef *value)
275 {
276     VModDef *def = malloc(sizeof(*def));
277     if (!def)
278         return NULL;
279
280     def->common.type = STMT_VMOD;
281     def->common.next = NULL;
282     def->name = name;
283     def->value = value;
284
285     return def;
286 }
287
288 VarDef *
289 VarCreate(ExprDef *name, ExprDef *value)
290 {
291     VarDef *def = malloc(sizeof(*def));
292     if (!def)
293         return NULL;
294
295     def->common.type = STMT_VAR;
296     def->common.next = NULL;
297     def->name = name;
298     def->value = value;
299
300     return def;
301 }
302
303 VarDef *
304 BoolVarCreate(xkb_atom_t ident, bool set)
305 {
306     return VarCreate((ExprDef *) ExprCreateIdent(ident),
307                      (ExprDef *) ExprCreateBoolean(set));
308 }
309
310 InterpDef *
311 InterpCreate(xkb_keysym_t sym, ExprDef *match)
312 {
313     InterpDef *def = malloc(sizeof(*def));
314     if (!def)
315         return NULL;
316
317     def->common.type = STMT_INTERP;
318     def->common.next = NULL;
319     def->sym = sym;
320     def->match = match;
321
322     return def;
323 }
324
325 KeyTypeDef *
326 KeyTypeCreate(xkb_atom_t name, VarDef *body)
327 {
328     KeyTypeDef *def = malloc(sizeof(*def));
329     if (!def)
330         return NULL;
331
332     def->common.type = STMT_TYPE;
333     def->common.next = NULL;
334     def->merge = MERGE_DEFAULT;
335     def->name = name;
336     def->body = body;
337
338     return def;
339 }
340
341 SymbolsDef *
342 SymbolsCreate(xkb_atom_t keyName, VarDef *symbols)
343 {
344     SymbolsDef *def = malloc(sizeof(*def));
345     if (!def)
346         return NULL;
347
348     def->common.type = STMT_SYMBOLS;
349     def->common.next = NULL;
350     def->merge = MERGE_DEFAULT;
351     def->keyName = keyName;
352     def->symbols = symbols;
353
354     return def;
355 }
356
357 GroupCompatDef *
358 GroupCompatCreate(unsigned group, ExprDef *val)
359 {
360     GroupCompatDef *def = malloc(sizeof(*def));
361     if (!def)
362         return NULL;
363
364     def->common.type = STMT_GROUP_COMPAT;
365     def->common.next = NULL;
366     def->merge = MERGE_DEFAULT;
367     def->group = group;
368     def->def = val;
369
370     return def;
371 }
372
373 ModMapDef *
374 ModMapCreate(xkb_atom_t modifier, ExprDef *keys)
375 {
376     ModMapDef *def = malloc(sizeof(*def));
377     if (!def)
378         return NULL;
379
380     def->common.type = STMT_MODMAP;
381     def->common.next = NULL;
382     def->merge = MERGE_DEFAULT;
383     def->modifier = modifier;
384     def->keys = keys;
385
386     return def;
387 }
388
389 LedMapDef *
390 LedMapCreate(xkb_atom_t name, VarDef *body)
391 {
392     LedMapDef *def = malloc(sizeof(*def));
393     if (!def)
394         return NULL;
395
396     def->common.type = STMT_LED_MAP;
397     def->common.next = NULL;
398     def->merge = MERGE_DEFAULT;
399     def->name = name;
400     def->body = body;
401
402     return def;
403 }
404
405 LedNameDef *
406 LedNameCreate(unsigned ndx, ExprDef *name, bool virtual)
407 {
408     LedNameDef *def = malloc(sizeof(*def));
409     if (!def)
410         return NULL;
411
412     def->common.type = STMT_LED_NAME;
413     def->common.next = NULL;
414     def->merge = MERGE_DEFAULT;
415     def->ndx = ndx;
416     def->name = name;
417     def->virtual = virtual;
418
419     return def;
420 }
421
422 static void
423 FreeInclude(IncludeStmt *incl);
424
425 IncludeStmt *
426 IncludeCreate(struct xkb_context *ctx, char *str, enum merge_mode merge)
427 {
428     IncludeStmt *incl, *first;
429     char *file, *map, *stmt, *tmp, *extra_data;
430     char nextop;
431
432     incl = first = NULL;
433     file = map = NULL;
434     tmp = str;
435     stmt = strdup_safe(str);
436     while (tmp && *tmp)
437     {
438         if (!ParseIncludeMap(&tmp, &file, &map, &nextop, &extra_data))
439             goto err;
440
441         /*
442          * Given an RMLVO (here layout) like 'us,,fr', the rules parser
443          * will give out something like 'pc+us+:2+fr:3+inet(evdev)'.
444          * We should just skip the ':2' in this case and leave it to the
445          * appropriate section to deal with the empty group.
446          */
447         if (isempty(file)) {
448             free(file);
449             free(map);
450             free(extra_data);
451             continue;
452         }
453
454         if (first == NULL) {
455             first = incl = malloc(sizeof(*first));
456         } else {
457             incl->next_incl = malloc(sizeof(*first));
458             incl = incl->next_incl;
459         }
460
461         if (!incl) {
462             log_wsgo(ctx,
463                      "Allocation failure in IncludeCreate; "
464                      "Using only part of the include\n");
465             break;
466         }
467
468         incl->common.type = STMT_INCLUDE;
469         incl->common.next = NULL;
470         incl->merge = merge;
471         incl->stmt = NULL;
472         incl->file = file;
473         incl->map = map;
474         incl->modifier = extra_data;
475         incl->next_incl = NULL;
476
477         if (nextop == '|')
478             merge = MERGE_AUGMENT;
479         else
480             merge = MERGE_OVERRIDE;
481     }
482
483     if (first)
484         first->stmt = stmt;
485     else
486         free(stmt);
487
488     return first;
489
490 err:
491     log_err(ctx, "Illegal include statement \"%s\"; Ignored\n", stmt);
492     FreeInclude(first);
493     free(stmt);
494     return NULL;
495 }
496
497 XkbFile *
498 XkbFileCreate(enum xkb_file_type type, char *name, ParseCommon *defs,
499               enum xkb_map_flags flags)
500 {
501     XkbFile *file;
502
503     file = calloc(1, sizeof(*file));
504     if (!file)
505         return NULL;
506
507     XkbEscapeMapName(name);
508     file->file_type = type;
509     file->topName = strdup_safe(name);
510     file->name = name;
511     file->defs = defs;
512     file->flags = flags;
513
514     return file;
515 }
516
517 XkbFile *
518 XkbFileFromComponents(struct xkb_context *ctx,
519                       const struct xkb_component_names *kkctgs)
520 {
521     char *const components[] = {
522         kkctgs->keycodes, kkctgs->types,
523         kkctgs->compat, kkctgs->symbols,
524     };
525     enum xkb_file_type type;
526     IncludeStmt *include = NULL;
527     XkbFile *file = NULL;
528     ParseCommon *defs = NULL;
529
530     for (type = FIRST_KEYMAP_FILE_TYPE; type <= LAST_KEYMAP_FILE_TYPE; type++) {
531         include = IncludeCreate(ctx, components[type], MERGE_DEFAULT);
532         if (!include)
533             goto err;
534
535         file = XkbFileCreate(type, NULL, (ParseCommon *) include, 0);
536         if (!file) {
537             FreeInclude(include);
538             goto err;
539         }
540
541         defs = AppendStmt(defs, &file->common);
542     }
543
544     file = XkbFileCreate(FILE_TYPE_KEYMAP, NULL, defs, 0);
545     if (!file)
546         goto err;
547
548     return file;
549
550 err:
551     FreeXkbFile((XkbFile *) defs);
552     return NULL;
553 }
554
555 static void
556 FreeExpr(ExprDef *expr)
557 {
558     if (!expr)
559         return;
560
561     switch (expr->expr.op) {
562     case EXPR_ACTION_LIST:
563     case EXPR_NEGATE:
564     case EXPR_UNARY_PLUS:
565     case EXPR_NOT:
566     case EXPR_INVERT:
567         FreeStmt((ParseCommon *) expr->unary.child);
568         break;
569
570     case EXPR_DIVIDE:
571     case EXPR_ADD:
572     case EXPR_SUBTRACT:
573     case EXPR_MULTIPLY:
574     case EXPR_ASSIGN:
575         FreeStmt((ParseCommon *) expr->binary.left);
576         FreeStmt((ParseCommon *) expr->binary.right);
577         break;
578
579     case EXPR_ACTION_DECL:
580         FreeStmt((ParseCommon *) expr->action.args);
581         break;
582
583     case EXPR_ARRAY_REF:
584         FreeStmt((ParseCommon *) expr->array_ref.entry);
585         break;
586
587     case EXPR_KEYSYM_LIST:
588         darray_free(expr->keysym_list.syms);
589         darray_free(expr->keysym_list.symsMapIndex);
590         darray_free(expr->keysym_list.symsNumEntries);
591         break;
592
593     default:
594         break;
595     }
596 }
597
598 static void
599 FreeInclude(IncludeStmt *incl)
600 {
601     IncludeStmt *next;
602
603     while (incl)
604     {
605         next = incl->next_incl;
606
607         free(incl->file);
608         free(incl->map);
609         free(incl->modifier);
610         free(incl->stmt);
611
612         free(incl);
613         incl = next;
614     }
615 }
616
617 void
618 FreeStmt(ParseCommon *stmt)
619 {
620     ParseCommon *next;
621
622     while (stmt)
623     {
624         next = stmt->next;
625
626         switch (stmt->type) {
627         case STMT_INCLUDE:
628             FreeInclude((IncludeStmt *) stmt);
629             /* stmt is already free'd here. */
630             stmt = NULL;
631             break;
632         case STMT_EXPR:
633             FreeExpr((ExprDef *) stmt);
634             break;
635         case STMT_VAR:
636             FreeStmt((ParseCommon *) ((VarDef *) stmt)->name);
637             FreeStmt((ParseCommon *) ((VarDef *) stmt)->value);
638             break;
639         case STMT_TYPE:
640             FreeStmt((ParseCommon *) ((KeyTypeDef *) stmt)->body);
641             break;
642         case STMT_INTERP:
643             FreeStmt((ParseCommon *) ((InterpDef *) stmt)->match);
644             FreeStmt((ParseCommon *) ((InterpDef *) stmt)->def);
645             break;
646         case STMT_VMOD:
647             FreeStmt((ParseCommon *) ((VModDef *) stmt)->value);
648             break;
649         case STMT_SYMBOLS:
650             FreeStmt((ParseCommon *) ((SymbolsDef *) stmt)->symbols);
651             break;
652         case STMT_MODMAP:
653             FreeStmt((ParseCommon *) ((ModMapDef *) stmt)->keys);
654             break;
655         case STMT_GROUP_COMPAT:
656             FreeStmt((ParseCommon *) ((GroupCompatDef *) stmt)->def);
657             break;
658         case STMT_LED_MAP:
659             FreeStmt((ParseCommon *) ((LedMapDef *) stmt)->body);
660             break;
661         case STMT_LED_NAME:
662             FreeStmt((ParseCommon *) ((LedNameDef *) stmt)->name);
663             break;
664         default:
665             break;
666         }
667
668         free(stmt);
669         stmt = next;
670     }
671 }
672
673 void
674 FreeXkbFile(XkbFile *file)
675 {
676     XkbFile *next;
677
678     while (file)
679     {
680         next = (XkbFile *) file->common.next;
681
682         switch (file->file_type) {
683         case FILE_TYPE_KEYMAP:
684             FreeXkbFile((XkbFile *) file->defs);
685             break;
686
687         case FILE_TYPE_TYPES:
688         case FILE_TYPE_COMPAT:
689         case FILE_TYPE_SYMBOLS:
690         case FILE_TYPE_KEYCODES:
691         case FILE_TYPE_GEOMETRY:
692             FreeStmt(file->defs);
693             break;
694
695         default:
696             break;
697         }
698
699         free(file->name);
700         free(file->topName);
701         free(file);
702         file = next;
703     }
704 }
705
706 static const char *xkb_file_type_strings[_FILE_TYPE_NUM_ENTRIES] = {
707     [FILE_TYPE_KEYCODES] = "xkb_keycodes",
708     [FILE_TYPE_TYPES] = "xkb_types",
709     [FILE_TYPE_COMPAT] = "xkb_compatibility",
710     [FILE_TYPE_SYMBOLS] = "xkb_symbols",
711     [FILE_TYPE_GEOMETRY] = "xkb_geometry",
712     [FILE_TYPE_KEYMAP] = "xkb_keymap",
713     [FILE_TYPE_RULES] = "rules",
714 };
715
716 const char *
717 xkb_file_type_to_string(enum xkb_file_type type)
718 {
719     if (type > _FILE_TYPE_NUM_ENTRIES)
720         return "unknown";
721     return xkb_file_type_strings[type];
722 }
723
724 static const char *stmt_type_strings[_STMT_NUM_VALUES] = {
725     [STMT_UNKNOWN] = "unknown statement",
726     [STMT_INCLUDE] = "include statement",
727     [STMT_KEYCODE] = "key name definition",
728     [STMT_ALIAS] = "key alias definition",
729     [STMT_EXPR] = "expression",
730     [STMT_VAR] = "variable definition",
731     [STMT_TYPE] = "key type definition",
732     [STMT_INTERP] = "symbol interpretation definition",
733     [STMT_VMOD] = "virtual modifiers definition",
734     [STMT_SYMBOLS] = "key symbols definition",
735     [STMT_MODMAP] = "modifier map declaration",
736     [STMT_GROUP_COMPAT] = "group declaration",
737     [STMT_LED_MAP] = "indicator map declaration",
738     [STMT_LED_NAME] = "indicator name declaration",
739 };
740
741 const char *
742 stmt_type_to_string(enum stmt_type type)
743 {
744     if (type >= _STMT_NUM_VALUES)
745         return NULL;
746     return stmt_type_strings[type];
747 }
748
749 static const char *expr_op_type_strings[_EXPR_NUM_VALUES] = {
750     [EXPR_VALUE] = "literal",
751     [EXPR_IDENT] = "identifier",
752     [EXPR_ACTION_DECL] = "action declaration",
753     [EXPR_FIELD_REF] = "field reference",
754     [EXPR_ARRAY_REF] = "array reference",
755     [EXPR_KEYSYM_LIST] = "list of keysyms",
756     [EXPR_ACTION_LIST] = "list of actions",
757     [EXPR_ADD] = "addition",
758     [EXPR_SUBTRACT] = "subtraction",
759     [EXPR_MULTIPLY] = "multiplication",
760     [EXPR_DIVIDE] = "division",
761     [EXPR_ASSIGN] = "assignment",
762     [EXPR_NOT] = "logical negation",
763     [EXPR_NEGATE] = "arithmetic negation",
764     [EXPR_INVERT] = "bitwise inversion",
765     [EXPR_UNARY_PLUS] = "unary plus",
766 };
767
768 const char *
769 expr_op_type_to_string(enum expr_op_type type)
770 {
771     if (type >= _EXPR_NUM_VALUES)
772         return NULL;
773     return expr_op_type_strings[type];
774 }
775
776 static const char *expr_value_type_strings[_EXPR_TYPE_NUM_VALUES] = {
777     [EXPR_TYPE_UNKNOWN] = "unknown",
778     [EXPR_TYPE_BOOLEAN] = "boolean",
779     [EXPR_TYPE_INT] = "int",
780     [EXPR_TYPE_STRING] = "string",
781     [EXPR_TYPE_ACTION] = "action",
782     [EXPR_TYPE_KEYNAME] = "keyname",
783     [EXPR_TYPE_SYMBOLS] = "symbols",
784 };
785
786 const char *
787 expr_value_type_to_string(enum expr_value_type type)
788 {
789     if (type >= _EXPR_TYPE_NUM_VALUES)
790         return NULL;
791     return expr_value_type_strings[type];
792 }