Edit

kc3-lang/libxkbcommon/test/buffercomp.c

Branch :

  • Show log

    Commit

  • Author : Pierre Le Marre
    Date : 2025-03-28 09:38:58
    Hash : 500b260b
    Message : xkbcomp: Fix parser failure on floating-point numbers Before this commit we used `strtold`, which depends on the locale. But the XKB syntax is fixed and uses a period as decimal separator. So ensure the syntax is correct without relying on `strtold` and truncate the result, as the parser does not use floating-point numbers.

  • test/buffercomp.c
  • /*
     * Copyright © 2009 Dan Nicholson
     * SPDX-License-Identifier: MIT
     */
    
    #include "config.h"
    
    #include <assert.h>
    #include <stdio.h>
    #include <stdlib.h>
    
    #include "test.h"
    
    #define GOLDEN_TESTS_OUTPUTS "keymaps/"
    
    static void
    test_encodings(struct xkb_context *ctx)
    {
        struct xkb_keymap *keymap;
    
        /* Accept UTF-8 encoded BOM (U+FEFF) */
        const char utf8_with_bom[] =
            "\xef\xbb\xbfxkb_keymap {"
            "  xkb_keycodes { };"
            "  xkb_types { };"
            "  xkb_compat { };"
            "  xkb_symbols { };"
            "};";
        keymap = test_compile_buffer(ctx, utf8_with_bom, sizeof(utf8_with_bom));
        assert(keymap);
        xkb_keymap_unref(keymap);
    
        /* Reject UTF-16LE encoded string */
        const char utf16_le[] =
            "x\0k\0b\0_\0k\0e\0y\0m\0a\0p\0 \0{\0\n\0"
            " \0 \0x\0k\0b\0_\0k\0e\0y\0c\0o\0d\0e\0s\0 \0{\0 \0i\0n\0c\0l\0u\0d\0e\0 \0\"\0e\0v\0d\0e\0v\0\"\0 \0}\0;\0\n\0"
            " \0 \0x\0k\0b\0_\0t\0y\0p\0e\0s\0 \0{\0 \0i\0n\0c\0l\0u\0d\0e\0 \0\"\0c\0o\0m\0p\0l\0e\0t\0e\0\"\0 \0}\0;\0\n\0"
            " \0 \0x\0k\0b\0_\0c\0o\0m\0p\0a\0t\0 \0{\0 \0i\0n\0c\0l\0u\0d\0e\0 \0\"\0c\0o\0m\0p\0l\0e\0t\0e\0\"\0 \0}\0;\0\n\0"
            " \0 \0x\0k\0b\0_\0s\0y\0m\0b\0o\0l\0s\0 \0{\0 \0i\0n\0c\0l\0u\0d\0e\0 \0\"\0p\0c\0\"\0 \0}\0;\0\n\0"
            "}\0;\0";
        keymap = test_compile_buffer(ctx, utf16_le, sizeof(utf16_le));
        assert(!keymap);
    
        /* Reject UTF-16LE with BOM encoded string */
        const char utf16_le_with_bom[] =
            "\xff\xfex\0k\0b\0_\0k\0e\0y\0m\0a\0p\0 \0{\0\n\0"
            " \0 \0x\0k\0b\0_\0k\0e\0y\0c\0o\0d\0e\0s\0 \0{\0 \0i\0n\0c\0l\0u\0d\0e\0 \0\"\0e\0v\0d\0e\0v\0\"\0 \0}\0;\0\n\0"
            " \0 \0x\0k\0b\0_\0t\0y\0p\0e\0s\0 \0{\0 \0i\0n\0c\0l\0u\0d\0e\0 \0\"\0c\0o\0m\0p\0l\0e\0t\0e\0\"\0 \0}\0;\0\n\0"
            " \0 \0x\0k\0b\0_\0c\0o\0m\0p\0a\0t\0 \0{\0 \0i\0n\0c\0l\0u\0d\0e\0 \0\"\0c\0o\0m\0p\0l\0e\0t\0e\0\"\0 \0}\0;\0\n\0"
            " \0 \0x\0k\0b\0_\0s\0y\0m\0b\0o\0l\0s\0 \0{\0 \0i\0n\0c\0l\0u\0d\0e\0 \0\"\0p\0c\0\"\0 \0}\0;\0\n\0"
            "}\0;\0";
        keymap = test_compile_buffer(ctx, utf16_le_with_bom, sizeof(utf16_le_with_bom));
        assert(!keymap);
    
        /* Reject UTF-16BE encoded string */
        const char utf16_be[] =
            "\0x\0k\0b\0_\0k\0e\0y\0m\0a\0p\0 \0{\0\n\0"
            " \0 \0x\0k\0b\0_\0k\0e\0y\0c\0o\0d\0e\0s\0 \0{\0 \0i\0n\0c\0l\0u\0d\0e\0 \0\"\0e\0v\0d\0e\0v\0\"\0 \0}\0;\0\n\0"
            " \0 \0x\0k\0b\0_\0t\0y\0p\0e\0s\0 \0{\0 \0i\0n\0c\0l\0u\0d\0e\0 \0\"\0c\0o\0m\0p\0l\0e\0t\0e\0\"\0 \0}\0;\0\n\0"
            " \0 \0x\0k\0b\0_\0c\0o\0m\0p\0a\0t\0 \0{\0 \0i\0n\0c\0l\0u\0d\0e\0 \0\"\0c\0o\0m\0p\0l\0e\0t\0e\0\"\0 \0}\0;\0\n\0"
            " \0 \0x\0k\0b\0_\0s\0y\0m\0b\0o\0l\0s\0 \0{\0 \0i\0n\0c\0l\0u\0d\0e\0 \0\"\0p\0c\0\"\0 \0}\0;\0\n\0"
            "}\0;";
        keymap = test_compile_buffer(ctx, utf16_be, sizeof(utf16_be));
        assert(!keymap);
    }
    
    struct keymap_simple_test_data {
        const char * const keymap;
        bool valid;
    };
    
    static void
    test_floats(struct xkb_context *ctx)
    {
        const struct keymap_simple_test_data tests[] = {
            /* Valid floats */
            {
                .keymap =
                    "xkb_keymap {\n"
                    "  xkb_keycodes { };\n"
                    "  xkb_types { };\n"
                    "  xkb_compat { };\n"
                    "  xkb_symbols { };\n"
                    "  xkb_geometry {\n"
                    "    width=123.456;\n"
                    "    width=123.0;\n"
                    "    width=123.;\n"
                    "    width=01.234;\n"
                    "    width=01.0;\n"
                    "    width=01.;\n"
                    "    width=001.234;\n"
                    "    width=001.0;\n"
                    "    width=001.;\n"
                    "  };"
                    "};",
                .valid = true
            },
            /* Invalid: missing integer part */
            {
                .keymap =
                    "xkb_keymap {\n"
                    "  xkb_keycodes { };\n"
                    "  xkb_types { };\n"
                    "  xkb_compat { };\n"
                    "  xkb_symbols { };\n"
                    "  xkb_geometry {\n"
                    "    width=.123;"
                    "  };"
                    "};",
                .valid = false
            },
            /* Invalid: comma decimal separator */
            {
                .keymap =
                    "xkb_keymap {\n"
                    "  xkb_keycodes { };\n"
                    "  xkb_types { };\n"
                    "  xkb_compat { };\n"
                    "  xkb_symbols { };\n"
                    "  xkb_geometry {\n"
                    "    width=1,23;"
                    "  };"
                    "};",
                .valid = false
            },
            /* Invalid: exponent */
            {
                .keymap =
                    "xkb_keymap {\n"
                    "  xkb_keycodes { };\n"
                    "  xkb_types { };\n"
                    "  xkb_compat { };\n"
                    "  xkb_symbols { };\n"
                    "  xkb_geometry {\n"
                    "    width=1.23e2;"
                    "  };"
                    "};",
                .valid = false
            },
        };
    
        for (unsigned int k = 0; k < ARRAY_SIZE(tests); k++) {
            fprintf(stderr, "------\n*** %s: #%u ***\n", __func__, k);
            struct xkb_keymap *keymap =
                test_compile_buffer(ctx, tests[k].keymap, strlen(tests[k].keymap));
            assert(tests[k].valid ^ !keymap);
            xkb_keymap_unref(keymap);
        }
    }
    
    static void
    test_component_syntax_error(struct xkb_context *ctx)
    {
        /* The following used to trigger memory leak */
        const char* const keymaps[] = {
            "xkb_keymap {"
            "  xkb_keycodes { };"
            "};"
            "};", /* Syntax error, keymap “complete” */
            "xkb_keymap {"
            "  xkb_keycodes { };"
            "  xkb_types { };"
            "  xkb_compat { };"
            "  xkb_symbols { };"
            "};"
            "};"/* Syntax error, keymap complete */,
            "xkb_keymap {"
            "  xkb_keycodes { };"
            "}" /* Syntax error, incomplete map */
            "  xkb_types { };"
            "  xkb_compat { };"
            "  xkb_symbols { };"
            "};",
            "xkb_keymap {"
            "  xkb_keycodes { };"
            ";" /* Syntax error, incomplete map */
            "  xkb_types { };"
            "  xkb_compat { };"
            "  xkb_symbols { };"
            "};",
        };
    
        for (unsigned int k = 0; k < ARRAY_SIZE(keymaps); k++) {
            fprintf(stderr, "------\n*** %s: #%u ***\n", __func__, k);
            const struct xkb_keymap *keymap =
                test_compile_buffer(ctx, keymaps[k], strlen(keymaps[k]));
            assert(!keymap);
        }
    }
    
    static void
    test_recursive(struct xkb_context *ctx)
    {
        struct xkb_keymap *keymap;
    
        const char* const keymaps[] = {
            /* Recursive keycodes */
            "Keycodes: recursive",
            "xkb_keymap {"
            "  xkb_keycodes { include \"evdev+recursive\" };"
            "  xkb_types { include \"complete\" };"
            "  xkb_compat { include \"complete\" };"
            "  xkb_symbols { include \"pc\" };"
            "};",
            "Keycodes: recursive(bar)",
            "xkb_keymap {"
            "  xkb_keycodes { include \"evdev+recursive(bar)\" };"
            "  xkb_types { include \"complete\" };"
            "  xkb_compat { include \"complete\" };"
            "  xkb_symbols { include \"pc\" };"
            "};",
            /* Recursive key types */
            "Key types: recursive",
            "xkb_keymap {"
            "  xkb_keycodes { include \"evdev\" };"
            "  xkb_types { include \"recursive\" };"
            "  xkb_compat { include \"complete\" };"
            "  xkb_symbols { include \"pc\" };"
            "};",
            "Key types: recursive(bar)",
            "xkb_keymap {"
            "  xkb_keycodes { include \"evdev\" };"
            "  xkb_types { include \"recursive(bar)\" };"
            "  xkb_compat { include \"complete\" };"
            "  xkb_symbols { include \"pc\" };"
            "};",
            /* Recursive compat */
            "Compat: recursive",
            "xkb_keymap {"
            "  xkb_keycodes { include \"evdev\" };"
            "  xkb_types { include \"recursive\" };"
            "  xkb_compat { include \"complete\" };"
            "  xkb_symbols { include \"pc\" };"
            "};",
            "Compat: recursive(bar)",
            "xkb_keymap {"
            "  xkb_keycodes { include \"evdev\" };"
            "  xkb_types { include \"complete\" };"
            "  xkb_compat { include \"recursive(bar)\" };"
            "  xkb_symbols { include \"pc\" };"
            "};",
            /* Recursive symbols */
            "Symbols: recursive",
            "xkb_keymap {"
            "  xkb_keycodes { include \"evdev\" };"
            "  xkb_types { include \"complete\" };"
            "  xkb_compat { include \"complete\" };"
            "  xkb_symbols { include \"recursive\" };"
            "};",
            "Symbols: recursive(bar)",
            "xkb_keymap {"
            "  xkb_keycodes { include \"evdev\" };"
            "  xkb_types { include \"complete\" };"
            "  xkb_compat { include \"complete\" };"
            "  xkb_symbols { include \"recursive(bar)\" };"
            "};"
        };
    
        for (unsigned int k = 0; k < ARRAY_SIZE(keymaps); k++) {
            fprintf(stderr, "------\n*** %s: #%u %s ***\n", __func__, k, keymaps[k]);
            k++;
            keymap = test_compile_buffer(ctx, keymaps[k], strlen(keymaps[k]));
            assert(!keymap);
        }
    }
    
    /* Test some limits related to allocations */
    static void
    test_alloc_limits(struct xkb_context *ctx)
    {
        const char * const keymaps[] = {
            /* Keycodes */
            "xkb_keymap {\n"
            /* Valid keycode value, but we should not handle it
             * with our *continuous* array! */
            "  xkb_keycodes { <> = 0xfffffffe; };\n"
            "  xkb_types { };\n"
            "  xkb_compat { };\n"
            "  xkb_symbols { key <> {[a]}; };\n"
            "};",
            /* Key types */
            "xkb_keymap {\n"
            "  xkb_keycodes { };\n"
            "  xkb_types {\n"
            "    type \"X\" { map[none] = 0xfffffffe; };\n" /* Invalid level index */
            "  };\n"
            "  xkb_compat { };\n"
            "  xkb_symbols { };\n"
            "};",
            "xkb_keymap {\n"
            "  xkb_keycodes { };\n"
            "  xkb_types {\n"
            "    type \"X\" {levelname[0xfffffffe]=\"x\";};\n" /* Invalid level index */
            "  };\n"
            "  xkb_compat { };\n"
            "  xkb_symbols { };\n"
            "};"
        };
        for (unsigned int k = 0; k < ARRAY_SIZE(keymaps); k++) {
            fprintf(stderr, "------\n*** %s: #%u ***\n", __func__, k);
            const struct xkb_keymap *keymap =
                test_compile_buffer(ctx, keymaps[k], strlen(keymaps[k]));
            assert(!keymap);
        }
    }
    
    struct keymap_test_data {
        const char * const keymap;
        const char * const expected;
    };
    
    /* Our keymap compiler is the xkbcommon buffer compiler */
    static struct xkb_keymap *
    compile_buffer(struct xkb_context *context, const char *buf, size_t len,
                         void *private)
    {
        return test_compile_buffer(context, buf, len);
    }
    
    static void
    test_integers(struct xkb_context *ctx, bool update_output_files) {
        const struct keymap_test_data keymaps[] = {
            {
                    .keymap =
                        "xkb_keymap {\n"
                        "  xkb_keycodes {\n"
                        /* Out of range (expect 32 bits, got > 64 bits) */
                        "    <> = 0x10000000000000000;\n"
                        "  };\n"
                        "  xkb_types { };\n"
                        "  xkb_compat { };\n"
                        "  xkb_symbols { };\n"
                        "};",
                    .expected = NULL
            },
            {
                    .keymap =
                        "xkb_keymap {\n"
                        "  xkb_keycodes { <> = 1; };\n"
                        "  xkb_types { };\n"
                        "  xkb_compat {\n"
                        "  };\n"
                        "  xkb_symbols {\n"
                        "    key <> {\n"
                        /* FIXME: Unchecked overflow */
                        "      [MovePointer(x=0xffffffff + 1,\n"
                        "                   y=0x80000000 * 2)]\n"
                        "    };\n"
                        "  };\n"
                        "};",
                    .expected = GOLDEN_TESTS_OUTPUTS "integers-overflow.xkb"
            },
            {
                .keymap =
                    "xkb_keymap {\n"
                    "  xkb_keycodes {\n"
                    "    <> = 1;\n"
                    "    indicator 32 = \"xxx\";\n"
                    "  };\n"
                    "  xkb_types { };\n"
                    "  xkb_compat {\n"
                    "    group 0xffffffff = Mod5;\n"
                    "  };\n"
                    "  xkb_symbols {\n"
                    /* Computations with 64 bit ints that then fit into 16 bits */
                    "    key <> {\n"
                    "      actions[1 + -~0x100000001 / 0x100000000]=\n"
                    "      [MovePointer(x=0x100000000 - 0xfffffffe,\n"
                    "                   y=~-0x7fff * 0x30000 / 0x2ffff)]\n"
                    "    };\n"
                    "  };\n"
                    "};",
                .expected = GOLDEN_TESTS_OUTPUTS "integers.xkb"
            }
        };
        for (unsigned int k = 0; k < ARRAY_SIZE(keymaps); k++) {
            fprintf(stderr, "------\n*** %s: #%u ***\n", __func__, k);
            assert(test_compile_output(ctx, compile_buffer, NULL, __func__,
                                       keymaps[k].keymap, strlen(keymaps[k].keymap),
                                       keymaps[k].expected, update_output_files));
        }
    }
    
    static void
    test_keycodes(struct xkb_context *ctx, bool update_output_files) {
        const struct keymap_test_data keymaps[] = {
            /*
             * Ensure keycodes bound are correctly updated.
             *
             * Since expanding bounds is tested in the rest of the code, we test
             * only shrinking here.
             */
    
            /* Single keycode */
            {
                .keymap =
                    "xkb_keymap {\n"
                    "  xkb_keycodes {\n"
                    "    <A> = 0;\n"
                    "    override <A> = 1;\n"
                    "    augment  <A> = 300;\n"
                    "  };\n"
                    "  xkb_types { };\n"
                    "  xkb_compat {\n"
                    "    interpret.repeat= False;\n"
                    "  };\n"
                    "  xkb_symbols { };\n"
                    "};",
                .expected = GOLDEN_TESTS_OUTPUTS "keycodes-bounds-single-1.xkb"
            },
            {
                .keymap =
                    "xkb_keymap {\n"
                    "  xkb_keycodes {\n"
                    "    <A> = 300;\n"
                    "    override <A> = 1;\n"
                    "    augment  <A> = 0;\n"
                    "  };\n"
                    "  xkb_types { };\n"
                    "  xkb_compat {\n"
                    "    interpret.repeat= False;\n"
                    "  };\n"
                    "  xkb_symbols { };\n"
                    "};",
                /* NOTE: same as previous */
                .expected = GOLDEN_TESTS_OUTPUTS "keycodes-bounds-single-1.xkb"
            },
            {
                .keymap =
                    "xkb_keymap {\n"
                    "  xkb_keycodes {\n"
                    "    <A> = 0;\n"
                    "    override <A> = 1;\n"
                    "    override <A> = 301;\n"
                    "    override <A> = 300;\n"
                    "  };\n"
                    "  xkb_types { };\n"
                    "  xkb_compat {\n"
                    "    interpret.repeat= False;\n"
                    "  };\n"
                    "  xkb_symbols { };\n"
                    "};",
                .expected = GOLDEN_TESTS_OUTPUTS "keycodes-bounds-single-2.xkb"
            },
    
            /* Multiple keycodes to single keycode */
            {
                .keymap =
                    "xkb_keymap {\n"
                    "  xkb_keycodes {\n"
                    "    <A> = 300;\n"
                    "    <B> = 1;\n"
                    "    augment  <B> = 301;\n"
                    "    override <A> = 1;\n"
                    "  };\n"
                    "  xkb_types { };\n"
                    "  xkb_compat {\n"
                    "    interpret.repeat= False;\n"
                    "  };\n"
                    "  xkb_symbols { };\n"
                    "};",
                /* NOTE: same as previous */
                .expected = GOLDEN_TESTS_OUTPUTS "keycodes-bounds-single-1.xkb"
            },
            {
                .keymap =
                    "xkb_keymap {\n"
                    "  xkb_keycodes {\n"
                    "    <A> = 0;\n"
                    "    <B> = 1;\n"
                    "    augment  <B> = 300;\n"
                    "    override <A> = 1;\n"
                    "  };\n"
                    "  xkb_types { };\n"
                    "  xkb_compat {\n"
                    "    interpret.repeat= False;\n"
                    "  };\n"
                    "  xkb_symbols { };\n"
                    "};",
                /* NOTE: same as previous */
                .expected = GOLDEN_TESTS_OUTPUTS "keycodes-bounds-single-1.xkb"
            },
    
            /* Multiple keycodes to multiple keycodes */
            {
                .keymap =
                    "xkb_keymap {\n"
                    "  xkb_keycodes {\n"
                    "    <A> = 1;\n"
                    "    <B> = 0;\n"
                    "    override <B> = 300;\n"
                    "    augment  <A> = 0;\n"
                    "  };\n"
                    "  xkb_types { };\n"
                    "  xkb_compat {\n"
                    "    interpret.repeat= False;\n"
                    "  };\n"
                    "  xkb_symbols { };\n"
                    "};",
                .expected = GOLDEN_TESTS_OUTPUTS "keycodes-bounds-multiple-1.xkb"
            },
            {
                .keymap =
                    "xkb_keymap {\n"
                    "  xkb_keycodes {\n"
                    "    <A> = 301;\n"
                    "    <B> = 300;\n"
                    "    override <A> = 1;\n"
                    "    augment  <B> = 302;\n"
                    "  };\n"
                    "  xkb_types { };\n"
                    "  xkb_compat {\n"
                    "    interpret.repeat= False;\n"
                    "  };\n"
                    "  xkb_symbols { };\n"
                    "};",
                /* NOTE: same as previous */
                .expected = GOLDEN_TESTS_OUTPUTS "keycodes-bounds-multiple-1.xkb"
            },
        };
    
        for (unsigned int k = 0; k < ARRAY_SIZE(keymaps); k++) {
            fprintf(stderr, "------\n*** %s: #%u ***\n", __func__, k);
            assert(test_compile_output(ctx, compile_buffer, NULL, __func__,
                                       keymaps[k].keymap, strlen(keymaps[k].keymap),
                                       keymaps[k].expected, update_output_files));
        }
    }
    
    static void
    test_masks(struct xkb_context *ctx, bool update_output_files) {
        const struct keymap_test_data keymaps[] = {
            {
                    .keymap =
                        "xkb_keymap {\n"
                        "  xkb_keycodes { };\n"
                        "  xkb_types { };\n"
                        "  xkb_compat {\n"
                        /* Cannot be negative */
                        "    virtual_modifiers Test1 = -1;\n"
                        "  };\n"
                        "  xkb_symbols { };\n"
                        "};",
                    .expected = NULL
            },
            {
                    .keymap =
                        "xkb_keymap {\n"
                        "  xkb_keycodes { };\n"
                        "  xkb_types { };\n"
                        "  xkb_compat {\n"
                        /* Out of range (expect 32bits) */
                        "    virtual_modifiers Test1 = 0x100000000;\n"
                        "  };\n"
                        "  xkb_symbols { };\n"
                        "};",
                    .expected = NULL
            },
            {
                    .keymap =
                        "xkb_keymap {\n"
                        "  xkb_keycodes { };\n"
                        "  xkb_types { };\n"
                        "  xkb_compat {\n"
                        /* Out of range (expect 32bits) */
                        "    virtual_modifiers Test1 = ~0x100000000;\n"
                        "  };\n"
                        "  xkb_symbols { };\n"
                        "};",
                    .expected = NULL
            },
            {
                    .keymap =
                        "xkb_keymap {\n"
                        "  xkb_keycodes { };\n"
                        "  xkb_types { };\n"
                        "  xkb_compat {\n"
                        /* Unsupported operator */
                        "    virtual_modifiers Test1 = !Mod1;\n"
                        "  };\n"
                        "  xkb_symbols { };\n"
                        "};",
                    .expected = NULL
            },
            {
                .keymap =
                    "xkb_keymap {\n"
                    "  xkb_keycodes { };\n"
                    "  xkb_types {\n"
                    /* Try range */
                    "    virtual_modifiers Test01 = 0;\n"
                    "    virtual_modifiers Test02 = 0xffffffff;\n"
                    /* Try various operations on masks */
                    "    virtual_modifiers Test11 = 0xf0 + 0x0f;\n"
                    "    virtual_modifiers Test12 = 0xff - 0x0f;\n"
                    "    virtual_modifiers Test13 = ~0xf;\n"
                    "    virtual_modifiers Test14 = ~none;\n"
                    "    virtual_modifiers Test15 = ~all;\n"
                    "    virtual_modifiers Test16 = ~0xffffffff;\n"
                    "    virtual_modifiers Test17 = all - ~Mod1 + Mod2;\n"
                    "    type \"XXX\" {\n"
                    "      modifiers = Test12;\n"
                    /* Masks mappings are not resolved here, so:
                     *   map[Test12 - Mod5] <=> map[Test12] */
                    "      map[Test12 - Mod5] = 2;\n"
                    "    };\n"
                    "  };\n"
                    "  xkb_compat { };\n"
                    "  xkb_symbols { };\n"
                    "};",
                .expected = GOLDEN_TESTS_OUTPUTS "masks.xkb"
            }
        };
        for (unsigned int k = 0; k < ARRAY_SIZE(keymaps); k++) {
            fprintf(stderr, "------\n*** %s: #%u ***\n", __func__, k);
            assert(test_compile_output(ctx, compile_buffer, NULL, __func__,
                                       keymaps[k].keymap, strlen(keymaps[k].keymap),
                                       keymaps[k].expected, update_output_files));
        }
    }
    
    /* Test various multi-{keysym,action} syntaxes */
    static void
    test_multi_keysyms_actions(struct xkb_context *ctx)
    {
        struct xkb_keymap *keymap;
    
        /* Macros to define the tests */
    #define make_keymap(keysyms, actions)               \
            "xkb_keymap {\n"                            \
            "  xkb_keycodes {\n"                        \
            "    minimum= 8;\n"                         \
            "    maximum= 10;\n"                        \
            "    <AE01> = 10;\n"                        \
            "  };\n"                                    \
            "  xkb_types { include \"basic\" };\n"      \
            "  xkb_compat { include \"basic\" };\n"     \
            "  xkb_symbols {\n"                         \
            "    key <AE01> { " keysyms actions " };\n" \
            "  };\n"                                    \
            "};"
    #define make_keymap_with_keysyms(keysyms) \
            make_keymap("[" keysyms "]", "")
    #define make_keymap_with_actions(actions) \
            make_keymap("", "actions[1] = [" actions "]")
    #define test_keymaps                                    \
            make_keymaps_with(make_keymap_with_keysyms,     \
                              "a", "b", "c", "d"),          \
            make_keymaps_with(make_keymap_with_actions,     \
                              "SetMods(modifiers=Control)", \
                              "SetGroup(group=+1)",         \
                              "Private(data=\"foo\")",      \
                              "Private(data=\"bar\")")
    #define run_test(kind, condition, msg, ...) do {                      \
        const char* const keymaps[] = { test_keymaps };                   \
        for (size_t k = 0; k < ARRAY_SIZE(keymaps); k++) {                \
            fprintf(stderr,                                               \
                    "------\n"                                            \
                    "*** %s: " kind "#%zu ***\n",                         \
                    __func__, k + 1);                                     \
            keymap = test_compile_buffer(ctx, keymaps[k],                 \
                                         strlen(keymaps[k]));             \
            assert_printf(condition,                                      \
                          "The following symbols " msg " parse:\n%s\n",   \
                          keymaps[k]);                                    \
            __VA_ARGS__;                                                  \
        }                                                                 \
    } while (0)
    
        /* Test: valid keymaps */
    #define make_keymaps_with(func, a, b, c, d)      \
            func(a),                                 \
            func(a", "b),                            \
            func("{ "a", "b" }"),                    \
            func("{ "a", "b", "c" }"),               \
            func(a", { "b", "c" }"),                 \
            func("{ "a", "b" }, "c),                 \
            func("{ "a", "b" }, { "c", "d" }"),      \
            func("{ "a", "b" }, "c", { "d", "a" }"), \
            func("{ "a", "b" }, { "c", "d" }, "a)
        run_test("valid", keymap != NULL, "does *not*", xkb_keymap_unref(keymap));
    #undef make_keymaps_with
    
        /* Test: invalid keymaps */
    #define make_keymaps_with(func, a, b, c, d)    \
            func("{}"),                            \
            func("{ {} }"),                        \
            func("{ "a" }"),                       \
            func("{ "a", {} }"),                   \
            func("{ {}, "b" }"),                   \
            func("{ {}, {} }"),                    \
            func("{ "a", { "b" } }"),              \
            func("{ { "a" }, "b" }"),              \
            func("{ { "a", "b" }, "c" }"),         \
            func("{ "a", { "b", "c" } }"),         \
            func("{ "a", {}, "c" }"),              \
            func("{ "a", "b", {} }"),              \
            func("{ {}, "b", "c" }"),              \
            func("{ { "a", "b" }, "c", "d" }"),    \
            func("{ "a", { "b", "c" }, "d" }"),    \
            func("{ "a", "b", { "c", "d" } }"),    \
            func("{ { "a", "b" }, { "c", "d" } }")
        run_test("invalid", keymap == NULL, "*does*");
    #undef make_keymap
    #undef make_keymap_with_actions
    #undef make_keymap_with_keysyms
    #undef make_keymaps_with
    #undef test_keymaps
    #undef run_test
    }
    
    static void
    test_invalid_symbols_fields(struct xkb_context *ctx)
    {
        /* Any of the following is invalid syntax, but also use to trigger
         * a NULL pointer deference, thus this regression test */
        const char * const keymaps[] = {
            "xkb_keymap {\n"
            "    xkb_keycodes { <> = 9; };\n"
            "    xkb_types { };\n"
            "    xkb_compat { };\n"
            "    xkb_symbols { key <> { vmods = [] }; };\n"
            "};",
            "xkb_keymap {\n"
            "    xkb_keycodes { <> = 9; };\n"
            "    xkb_types { };\n"
            "    xkb_compat { };\n"
            "    xkb_symbols { key <> { repeat = [] }; };\n"
            "};",
            "xkb_keymap {\n"
            "    xkb_keycodes { <> = 9; };\n"
            "    xkb_types { };\n"
            "    xkb_compat { };\n"
            "    xkb_symbols { key <> { type = [] }; };\n"
            "};",
            "xkb_keymap {\n"
            "    xkb_keycodes { <> = 9; };\n"
            "    xkb_types { };\n"
            "    xkb_compat { };\n"
            "    xkb_symbols { key <> { groupswrap = [] }; };\n"
            "};",
            "xkb_keymap {\n"
            "    xkb_keycodes { <> = 9; };\n"
            "    xkb_types { };\n"
            "    xkb_compat { };\n"
            "    xkb_symbols { key <> { groupsredirect = [] }; };\n"
            "};",
            /* Used to parse without error because the “repeats” key field is valid,
             * but we should fail in the following 2 keymaps */
            "xkb_keymap {\n"
            "    xkb_keycodes { <> = 9; };\n"
            "    xkb_types { };\n"
            "    xkb_compat { };\n"
            "    xkb_symbols { key <> { vmods=[], repeats=false }; };\n"
            "};",
            "xkb_keymap {\n"
            "    xkb_keycodes { <> = 9; };\n"
            "    xkb_types { };\n"
            "    xkb_compat { };\n"
            "    xkb_symbols { key <> { repeats=false, vmods=[] }; };\n"
            "};"
        };
        for (unsigned int k = 0; k < ARRAY_SIZE(keymaps); k++) {
            fprintf(stderr, "------\n*** %s: #%u ***\n", __func__, k);
            const struct xkb_keymap *keymap =
                test_compile_buffer(ctx, keymaps[k], strlen(keymaps[k]));
            assert(!keymap);
        }
    }
    
    static void
    test_modifier_maps(struct xkb_context *ctx, bool update_output_files)
    {
        /* Only accept key and keysyms in the modifier_map list */
        struct keymap_test_data keymaps[] = {
            {
                .keymap =
                    "xkb_keymap {\n"
                    "  xkb_keycodes {\n"
                    "    <CAPS> = 66;\n"
                    "    alias <LOCK> = <CAPS>;\n"
                    "    <0> = 0;"
                    "    <1> = 1;"
                    "    <2> = 2;"
                    "    <3> = 3;"
                    "    <any>  = 10;"
                    "    <none> = 11;"
                    "    <a> = 61;"
                    "    <100> = 100;"
                    "  };\n"
                    "  xkb_types { include \"basic\" };\n"
                    "  xkb_compat { };\n"
                    "  xkb_symbols {\n"
                    "    key <CAPS> { [Caps_Lock] };\n"
                    "    key <any>  { [any, A] };\n"
                    "    key <none> { [none, N] };\n"
                    "    key <0>    { [0] };\n"
                    "    key <1>    { [1] };\n"
                    "    key <2>    { [2] };\n"
                    "    key <a>    { [a] };\n"
                    "    key <3>    { [NotAKeysym, 3] };\n"
                    "    key <100>  { [C] };\n"
                    "    modifier_map Lock {\n"
                    "      <100>, <LOCK>, any, none,\n"
                    "      0, 1, 0x2, a, NotAKeysym\n"
                    "    };\n"
                    "  };\n"
                    "};",
                .expected = GOLDEN_TESTS_OUTPUTS "symbols-modifier_map.xkb"
            },
            /* Invalid: string (length = 0) */
            {
                .keymap =
                    "xkb_keymap {\n"
                    "  xkb_keycodes { };\n"
                    "  xkb_types { };\n"
                    "  xkb_compat { };\n"
                    "  xkb_symbols { modifier_map Lock { \"\" }; };\n"
                    "};",
                .expected = NULL
            },
            /* Invalid: string (length = 1) */
            {
                .keymap =
                    "xkb_keymap {\n"
                    "  xkb_keycodes { };\n"
                    "  xkb_types { };\n"
                    "  xkb_compat { };\n"
                    "  xkb_symbols { modifier_map Lock { \"a\" }; };\n"
                    "};",
                .expected = NULL
            },
            /* Invalid: string (length > 1) */
            {
                .keymap =
                    "xkb_keymap {\n"
                    "  xkb_keycodes { };\n"
                    "  xkb_types { };\n"
                    "  xkb_compat { };\n"
                    "  xkb_symbols { modifier_map Lock { \"ab\" }; };\n"
                    "};",
                .expected = NULL
            },
            /* Invalid type: list */
            {
                .keymap =
                    "xkb_keymap {\n"
                    "  xkb_keycodes { };\n"
                    "  xkb_types { };\n"
                    "  xkb_compat { };\n"
                    "  xkb_symbols { modifier_map Lock { [a] }; };\n"
                    "};",
                .expected = NULL
            },
            /* Invalid type: list */
            {
                .keymap =
                    "xkb_keymap {\n"
                    "  xkb_keycodes { };\n"
                    "  xkb_types { };\n"
                    "  xkb_compat { };\n"
                    "  xkb_symbols { modifier_map Lock { {a, b} }; };\n"
                    "};",
                .expected = NULL
            },
        };
    
        for (unsigned int k = 0; k < ARRAY_SIZE(keymaps); k++) {
            fprintf(stderr, "------\n*** %s: #%u ***\n", __func__, k);
            assert(test_compile_output(ctx, compile_buffer, NULL, __func__,
                                       keymaps[k].keymap, strlen(keymaps[k].keymap),
                                       keymaps[k].expected, update_output_files));
        }
    }
    
    static void
    test_prebuilt_keymap_roundtrip(struct xkb_context *ctx, bool update_output_files)
    {
        /* Load in a prebuilt keymap, make sure we can compile it from memory,
         * then compare it to make sure we get the same result when dumping it
         * to a string. */
        const char * const path = GOLDEN_TESTS_OUTPUTS "stringcomp.data";
        char *original = test_read_file(path);
        assert(original);
    
        /* Load a prebuild keymap, once without, once with the trailing \0 */
        for (unsigned int i = 0; i <= 1; i++) {
            fprintf(stderr, "------\n*** %s, trailing '\\0': %d ***\n", __func__, i);
    
            assert(test_compile_output(ctx, compile_buffer, NULL, __func__,
                                       original, strlen(original) + i, path,
                                       update_output_files));
        }
    
        free(original);
    }
    
    static void
    test_keymap_from_rules(struct xkb_context *ctx)
    {
        /* Make sure we can recompile our output for a normal keymap from rules. */
        fprintf(stderr, "------\n*** %s ***\n", __func__);
        struct xkb_keymap *keymap = test_compile_rules(ctx, NULL, NULL,
                                                       "ru,ca,de,us",
                                                       ",multix,neo,intl", NULL);
        assert(keymap);
        char *dump = xkb_keymap_get_as_string(keymap, XKB_KEYMAP_USE_ORIGINAL_FORMAT);
        assert(dump);
        xkb_keymap_unref(keymap);
        keymap = test_compile_buffer(ctx, dump, strlen(dump));
        assert(keymap);
        xkb_keymap_unref(keymap);
        free(dump);
    }
    
    int
    main(int argc, char *argv[])
    {
        test_init();
    
        bool update_output_files = false;
        if (argc > 1) {
            if (streq(argv[1], "update")) {
                /* Update files with *obtained* results */
                update_output_files = true;
            } else {
                fprintf(stderr, "ERROR: unsupported argument: \"%s\".\n", argv[1]);
                exit(EXIT_FAILURE);
            }
        }
    
        struct xkb_context *ctx = test_get_context(CONTEXT_NO_FLAG);
        assert(ctx);
    
        /* Make sure we can't (falsely claim to) compile an empty string. */
        struct xkb_keymap *keymap = test_compile_buffer(ctx, "", 0);
        assert(!keymap);
    
        test_encodings(ctx);
        test_floats(ctx);
        test_component_syntax_error(ctx);
        test_recursive(ctx);
        test_alloc_limits(ctx);
        test_integers(ctx, update_output_files);
        test_keycodes(ctx, update_output_files);
        test_masks(ctx, update_output_files);
        test_multi_keysyms_actions(ctx);
        test_invalid_symbols_fields(ctx);
        test_modifier_maps(ctx, update_output_files);
        test_prebuilt_keymap_roundtrip(ctx, update_output_files);
        test_keymap_from_rules(ctx);
    
        xkb_context_unref(ctx);
    
        return EXIT_SUCCESS;
    }