Edit

kc3-lang/libxkbcommon/test/keysym.c

Branch :

  • Show log

    Commit

  • Author : Pierre Le Marre
    Date : 2025-07-07 12:28:24
    Hash : dc63e5f8
    Message : Ensure config.h is always included first While `config.h` may not be necessary in every file, it ensures consistency and makes code refactoring safer.

  • test/keysym.c
  • /*
     * Copyright © 2009 Dan Nicholson
     * SPDX-License-Identifier: MIT
     */
    #include "config.h"
    
    #include <assert.h>
    #include <locale.h>
    #include <stdbool.h>
    #include <stdint.h>
    #include <stdio.h>
    #include <string.h>
    #if HAVE_ICU
    #include <unicode/uchar.h>
    #include <unicode/ustring.h>
    #include <unicode/utf16.h>
    #include "test/keysym-case-mapping.h"
    #endif
    
    #include "xkbcommon/xkbcommon-keysyms.h"
    #include "xkbcommon/xkbcommon.h"
    #include "test.h"
    #include "utils.h"
    #include "utils-numbers.h"
    #include "src/keysym.h" /* For unexported is_lower/upper/keypad() */
    #include "test/keysym.h"
    #include "utils.h"
    
    /* Explicit ordered list of modifier keysyms */
    static const xkb_keysym_t modifier_keysyms[] = {
        XKB_KEY_ISO_Lock,
        XKB_KEY_ISO_Level2_Latch,
        XKB_KEY_ISO_Level3_Shift,
        XKB_KEY_ISO_Level3_Latch,
        XKB_KEY_ISO_Level3_Lock,
        /* XKB_KEY_ISO_Group_Shift == XKB_KEY_Mode_switch */
        XKB_KEY_ISO_Group_Latch,
        XKB_KEY_ISO_Group_Lock,
        XKB_KEY_ISO_Next_Group,
        XKB_KEY_ISO_Next_Group_Lock,
        XKB_KEY_ISO_Prev_Group,
        XKB_KEY_ISO_Prev_Group_Lock,
        XKB_KEY_ISO_First_Group,
        XKB_KEY_ISO_First_Group_Lock,
        XKB_KEY_ISO_Last_Group,
        XKB_KEY_ISO_Last_Group_Lock,
        0xfe10, /* Currently unassigned, but xkb_keysym_is_modifier returns true */
        XKB_KEY_ISO_Level5_Shift,
        XKB_KEY_ISO_Level5_Latch,
        XKB_KEY_ISO_Level5_Lock,
    
        XKB_KEY_Mode_switch,
        XKB_KEY_Num_Lock,
    
        XKB_KEY_Shift_L,
        XKB_KEY_Shift_R,
        XKB_KEY_Control_L,
        XKB_KEY_Control_R,
        XKB_KEY_Caps_Lock,
        XKB_KEY_Shift_Lock,
    
        XKB_KEY_Meta_L,
        XKB_KEY_Meta_R,
        XKB_KEY_Alt_L,
        XKB_KEY_Alt_R,
        XKB_KEY_Super_L,
        XKB_KEY_Super_R,
        XKB_KEY_Hyper_L,
        XKB_KEY_Hyper_R
    };
    
    #define MIN_MODIFIER_KEYSYM modifier_keysyms[0]
    #define MAX_MODIFIER_KEYSYM modifier_keysyms[ARRAY_SIZE(modifier_keysyms) - 1]
    
    static void
    test_modifiers_table(void)
    {
        xkb_keysym_t ks = XKB_KEY_NoSymbol;
    
        /* Ensure ordered array */
        for (size_t k = 0; k < ARRAY_SIZE(modifier_keysyms); k++) {
            assert_printf(ks < modifier_keysyms[k],
                          "modifier_keysyms[] is not ordered: 0x%04"PRIx32">=0x%04"PRIx32"\n",
                          ks, modifier_keysyms[k]);
            ks = modifier_keysyms[k];
        }
    
        /* Unassigned keysym */
        assert(!xkb_keysym_is_assigned(0xfe10));
    }
    
    static bool
    test_modifier(xkb_keysym_t ks)
    {
        if (ks < MIN_MODIFIER_KEYSYM || ks > MAX_MODIFIER_KEYSYM)
            return false;
        for (size_t k = 0; k < ARRAY_SIZE(modifier_keysyms); k++) {
            if (ks == modifier_keysyms[k])
                return true;
        }
        return false;
    }
    
    static bool
    test_keypad(xkb_keysym_t ks, char *name)
    {
        const char prefix[] = "KP_";
        return strncmp(prefix, name, sizeof(prefix) - 1) == 0;
    }
    
    static int
    test_string(const char *string, xkb_keysym_t expected)
    {
        xkb_keysym_t keysym;
    
        keysym = xkb_keysym_from_name(string, XKB_KEYSYM_NO_FLAGS);
    
        fprintf(stderr, "Expected string %s -> %x\n", string, expected);
        fprintf(stderr, "Received string %s -> %x\n\n", string, keysym);
    
        return keysym == expected;
    }
    
    static int
    test_casestring(const char *string, xkb_keysym_t expected)
    {
        xkb_keysym_t keysym;
    
        keysym = xkb_keysym_from_name(string, XKB_KEYSYM_CASE_INSENSITIVE);
    
        fprintf(stderr, "Expected casestring %s -> %x\n", string, expected);
        fprintf(stderr, "Received casestring %s -> %x\n\n", string, keysym);
    
        return keysym == expected;
    }
    
    static void
    test_ambiguous_icase_names(const struct ambiguous_icase_ks_names_entry *entry)
    {
        for (int k = 0; k < entry->count; k++) {
            /* Check expected result */
            assert(test_casestring(entry->names[k], entry->keysym));
            /* If the keysym is cased, then check the resulting keysym is lower-cased */
            xkb_keysym_t keysym = xkb_keysym_from_name(entry->names[k], 0);
            if (xkb_keysym_is_lower(keysym) || xkb_keysym_is_upper_or_title(keysym)) {
                assert(xkb_keysym_is_lower(entry->keysym));
            }
        }
    }
    
    static int
    test_keysym(xkb_keysym_t keysym, const char *expected)
    {
        char s[XKB_KEYSYM_NAME_MAX_SIZE];
    
        xkb_keysym_get_name(keysym, s, sizeof(s));
    
        fprintf(stderr, "Expected keysym %#06"PRIx32" -> %s\n", keysym, expected);
        fprintf(stderr, "Received keysym %#06"PRIx32" -> %s\n\n", keysym, s);
    
        return streq(s, expected);
    }
    
    static bool
    test_deprecated(xkb_keysym_t keysym, const char *name,
                    bool expected_deprecated, const char *expected_reference)
    {
        const char *reference;
        bool deprecated = xkb_keysym_is_deprecated(keysym, name, &reference);
    
        fprintf(stderr, "Expected keysym %#06"PRIx32" -> deprecated: %d, reference: %s\n",
                keysym, expected_deprecated, expected_reference);
        fprintf(stderr, "Received keysym %#06"PRIx32" -> deprecated: %d, reference: %s\n",
                keysym, deprecated, reference);
    
        return deprecated == expected_deprecated &&
               (
                (reference == NULL && expected_reference == NULL) ||
                (reference != NULL && expected_reference != NULL &&
                 strcmp(reference, expected_reference) == 0)
               );
    }
    
    static int
    test_utf8(xkb_keysym_t keysym, const char *expected)
    {
        char s[XKB_KEYSYM_UTF8_MAX_SIZE];
        int ret;
    
        ret = xkb_keysym_to_utf8(keysym, s, sizeof(s));
        if (ret <= 0)
            return ret;
    
        assert(expected != NULL);
    
        fprintf(stderr, "Expected keysym %#06"PRIx32" -> %s (%zu bytes)\n", keysym, expected,
                strlen(expected));
        fprintf(stderr, "Received keysym %#06"PRIx32" -> %s (%zu bytes)\n\n", keysym, s,
                strlen(s));
    
        return streq(s, expected);
    }
    
    #if HAVE_ICU
    
    static UVersionInfo xkb_unicode_version = XKB_KEYSYM_UNICODE_VERSION;
    static UVersionInfo icu_unicode_version;
    
    static inline int
    compare_unicode_version(const UVersionInfo v1, const UVersionInfo v2)
    {
        return memcmp(v1,v2,sizeof(UVersionInfo));
    }
    
    /* Unicode assertion
     * In case the test fails, do not raise an exception if there is
     * an ICU version mismatch with our Unicode version. */
    #define uassert_printf(cp, cond, ...)                                                     \
       if (!(cond)) {                                                                         \
          fprintf(stderr, "Assertion failure: " __VA_ARGS__);                                 \
          UVersionInfo char_age;                                                              \
          u_charAge((UChar32)cp, char_age);                                                   \
          if (compare_unicode_version(char_age, xkb_unicode_version) > 0) {                   \
            fprintf(stderr,                                                                   \
                    "[WARNING] ICU version mismatch: "                                        \
                    "too recent for code point: "CODE_POINT"\n", cp);                         \
          } else if (compare_unicode_version(icu_unicode_version, xkb_unicode_version) < 0) { \
            fprintf(stderr,                                                                   \
                    "[WARNING] ICU version mismatch: "                                        \
                    "too old for code point: "CODE_POINT"\n", cp);                            \
          } else {                                                                            \
            assert(cond);                                                                     \
          }                                                                                   \
       }
    
    #define KEYSYM     "0x%04"PRIx32
    #define CODE_POINT "U+%04"PRIX32
    
    static void
    test_icu_case_mappings(xkb_keysym_t ks)
    {
        uint32_t cp = xkb_keysym_to_utf32(ks);
    
        /* Check predicates */
        bool is_lower = xkb_keysym_is_lower(ks);
        uint32_t expected = !!u_isULowercase((UChar32) cp);
        uassert_printf(cp, is_lower == expected,
                       "Invalid xkb_keysym_is_lower("KEYSYM") ("CODE_POINT"): "
                       "expected %d, got: %d\n",
                       ks, cp, expected, is_lower);
        bool is_upper_or_title = xkb_keysym_is_upper_or_title(ks);
        expected = !!(u_isUUppercase((UChar32) cp) || u_istitle((UChar32) cp));
        uassert_printf(cp, is_upper_or_title == expected,
                       "Invalid xkb_keysym_is_upper_or_title("KEYSYM") ("CODE_POINT"): "
                       "expected %d, got: %d\n",
                       ks, cp, expected, is_upper_or_title);
        assert(is_lower != is_upper_or_title || !is_lower);
    
        /* Check lower case mapping */
        xkb_keysym_t ks_mapped = xkb_keysym_to_lower(ks);
        expected = to_simple_lower(cp);
        if (u_istitle((UChar32) cp)) {
            /* Check that title case letter have simple lower case mappings */
            uassert_printf(cp, ks_mapped != ks && expected != cp,
                           "Invalid title case lower transformation. "
                           "Expected keysym: "KEYSYM" != "KEYSYM" "
                           "and code point "CODE_POINT" != "CODE_POINT"\n",
                           ks_mapped, ks, expected, cp);
        }
        if (ks_mapped && ks_mapped != ks) {
            /* Given keysym has been transformed to lower-case */
            uint32_t cp_mapped = xkb_keysym_to_utf32(ks_mapped);
            uint32_t got = cp_mapped;
            uassert_printf(cp, got == expected,
                           "Invalid xkb_keysym_to_lower("KEYSYM") == "KEYSYM": "
                           "expected "CODE_POINT", got: "CODE_POINT"\n",
                           ks, ks_mapped, expected, got);
            uassert_printf(cp, is_upper_or_title,
                           "Expected upper case for keysym "KEYSYM" ("CODE_POINT")\n",
                           ks, cp);
            got = !!xkb_keysym_is_lower(ks_mapped);
            expected = !!u_isULowercase((UChar32) cp_mapped);
            uassert_printf(cp_mapped, got == expected,
                           "Invalid xkb_keysym_is_lower("KEYSYM") ("CODE_POINT"): "
                           "expected %d, got: %d (tested keysym: "KEYSYM")\n",
                           ks_mapped, cp_mapped, expected, got, ks);
        } else if (expected != cp) {
            /* Missing case mapping; the corresponding predicate must be consistent. */
            fprintf(stderr,
                    "[WARNING] Missing lower case mapping for "KEYSYM": "
                    "expected "CODE_POINT", got: "CODE_POINT"\n",
                    ks, expected, cp);
            uassert_printf(cp, !xkb_keysym_is_upper_or_title(ks),
                           "Invalid xkb_keysym_is_upper_or_title("KEYSYM") ("CODE_POINT"): "
                           "expected false, got: true\n",
                           ks, cp);
        }
    
        /* Check upper case mapping */
        ks_mapped = xkb_keysym_to_upper(ks);
        expected = to_simple_upper(cp);
        if (u_istitle((UChar32) cp)) {
            /* Check title case upper mapping; may be:
             * • simple: 1 code point, or
             * • special: muliple code points */
            UChar cp_string[2] = {0};
            UChar cp_expected_string[8] = {0};
            UBool isError = false;
            int32_t offset = 0;
            /* Convert code point to UTF-16 string */
            U16_APPEND(cp_string, offset, (int32_t) ARRAY_SIZE(cp_string), cp, isError);
            UErrorCode pErrorCode = U_ZERO_ERROR;
            /* Unicode full upper mapping */
            int32_t length = u_strToUpper(cp_expected_string,
                                          ARRAY_SIZE(cp_expected_string),
                                          cp_string, offset, "C", &pErrorCode);
            length = u_countChar32(cp_expected_string, length);
            if (length == 1) {
                /* Simple upper case mapping: one-to-one. */
                uint32_t cp_mapped = xkb_keysym_to_utf32(ks_mapped);
                uassert_printf(cp,
                               !isError && pErrorCode == U_ZERO_ERROR &&
                               ks_mapped != ks && expected != cp &&
                               u_isUUppercase(cp_mapped),
                               "Invalid title case simple upper transformation. "
                               "Expected keysym: "KEYSYM" != "KEYSYM" "
                               "and code point "CODE_POINT" != "CODE_POINT"\n",
                               ks_mapped, ks, expected, cp);
            } else {
                /* Special upper case mapping: maps to multiple code points.
                 * We do not handle those, so our mapping is the original. */
                uassert_printf(cp,
                               !isError && pErrorCode == U_ZERO_ERROR &&
                               ks_mapped == ks && expected == cp && length > 1,
                               "Invalid title case special upper transformation. "
                               "Expected keysym: "KEYSYM" == "KEYSYM" "
                               "and code point "CODE_POINT" == "CODE_POINT"\n",
                               ks_mapped, ks, expected, cp);
            }
        }
        if (ks_mapped && ks_mapped != ks) {
            /* Given keysym has been transformed to upper-case */
            uint32_t cp_mapped = xkb_keysym_to_utf32(ks_mapped);
            uint32_t got = cp_mapped;
            uassert_printf(cp, got == expected,
                           "Invalid xkb_keysym_to_upper("KEYSYM") == "KEYSYM": "
                           "expected "CODE_POINT", got: "CODE_POINT"\n",
                           ks, ks_mapped, expected, got);
            uassert_printf(cp, is_lower || u_istitle(cp),
                           "Expected lower or title case for keysym "KEYSYM" ("CODE_POINT")\n",
                           ks, cp);
            got = !!xkb_keysym_is_upper_or_title(ks_mapped);
            expected = !!(u_isUUppercase((UChar32)cp_mapped) || u_istitle((UChar32)cp_mapped));
            uassert_printf(cp_mapped, got == expected,
                           "Invalid xkb_keysym_is_upper_or_title("KEYSYM") ("CODE_POINT"): "
                           "expected %d, got: %d (tested keysym: "KEYSYM")\n",
                           ks_mapped, cp_mapped, expected, got, ks);
        } else if (expected != cp) {
            /* Missing case mapping; the corresponding predicate must be consistent. */
            fprintf(stderr,
                    "[WARNING] Missing upper case mapping for "KEYSYM": "
                    "expected "CODE_POINT", got: "CODE_POINT"\n",
                    ks, expected, cp);
            uassert_printf(cp, !xkb_keysym_is_lower(ks),
                           "Invalid xkb_keysym_is_lower("KEYSYM") ("CODE_POINT"): "
                           "expected false, got: true\n",
                           ks, cp);
        }
    }
    #endif
    
    static void
    test_github_issue_42(void)
    {
        // Verify we are not dependent on locale, Turkish-i problem in particular.
        if (setlocale(LC_CTYPE, "tr_TR.UTF-8") == NULL) {
            // The locale is not available, probably; skip.
            return;
        }
    
        assert(test_string("i", XKB_KEY_i));
        assert(test_string("I", XKB_KEY_I));
        assert(test_casestring("i", XKB_KEY_i));
        assert(test_casestring("I", XKB_KEY_i));
        assert(xkb_keysym_to_upper(XKB_KEY_i) == XKB_KEY_I);
        assert(xkb_keysym_to_lower(XKB_KEY_I) == XKB_KEY_i);
    
        setlocale(LC_CTYPE, "C");
    }
    
    static void
    get_keysym_name(xkb_keysym_t keysym, char *buffer, size_t size)
    {
        int name_length = xkb_keysym_get_name(keysym, buffer, size);
        if (name_length < 0) {
            snprintf(buffer, size, "(unknown: %#06"PRIx32")", keysym);
        }
    }
    
    static int
    test_utf32_to_keysym(uint32_t ucs, xkb_keysym_t expected)
    {
        char expected_name[XKB_KEYSYM_NAME_MAX_SIZE];
        char actual_name[XKB_KEYSYM_NAME_MAX_SIZE];
        xkb_keysym_t actual = xkb_utf32_to_keysym(ucs);
        get_keysym_name(expected, expected_name, XKB_KEYSYM_NAME_MAX_SIZE);
        get_keysym_name(actual, actual_name, XKB_KEYSYM_NAME_MAX_SIZE);
    
        fprintf(stderr, "Code point %#"PRIx32": expected keysym: %s, actual: %s\n\n",
                ucs, expected_name, actual_name);
        return expected == actual;
    }
    
    int
    main(void)
    {
        test_init();
    #if HAVE_ICU
        u_getUnicodeVersion(icu_unicode_version);
    #endif
    
        /* Bounds */
        assert(XKB_KEYSYM_MIN == 0);
        assert(XKB_KEYSYM_MIN < XKB_KEYSYM_MAX);
        assert(XKB_KEYSYM_MAX <= UINT32_MAX); /* Ensure it fits in xkb_keysym_t */
        assert(XKB_KEYSYM_MAX <= INT32_MAX); /* Ensure correct cast to int32_t */
        assert(XKB_KEYSYM_MIN_ASSIGNED == XKB_KEYSYM_MIN);
        assert(XKB_KEYSYM_MIN_ASSIGNED < XKB_KEYSYM_MAX_ASSIGNED);
        assert(XKB_KEYSYM_MAX_ASSIGNED <= XKB_KEYSYM_MAX);
        assert(XKB_KEYSYM_MIN_EXPLICIT == XKB_KEYSYM_MIN_ASSIGNED);
        assert(XKB_KEYSYM_MIN_EXPLICIT < XKB_KEYSYM_MAX_EXPLICIT);
        assert(XKB_KEYSYM_MAX_EXPLICIT <= XKB_KEYSYM_MAX_ASSIGNED);
        assert(XKB_KEYSYM_COUNT_EXPLICIT <= XKB_KEYSYM_MAX_EXPLICIT - XKB_KEYSYM_MIN_EXPLICIT + 1);
        assert(XKB_KEYSYM_UNICODE_MIN >= XKB_KEYSYM_MIN_EXPLICIT);
        assert(XKB_KEYSYM_UNICODE_MIN < XKB_KEYSYM_UNICODE_MAX);
        assert(XKB_KEYSYM_UNICODE_MAX <= XKB_KEYSYM_MAX_EXPLICIT);
    
        /* Assigned keysyms */
        assert(xkb_keysym_is_assigned(XKB_KEYSYM_MIN));
        assert(xkb_keysym_is_assigned(XKB_KEYSYM_MIN_ASSIGNED));
        assert(xkb_keysym_is_assigned(XKB_KEY_space));
        assert(xkb_keysym_is_assigned(XKB_KEY_nobreakspace));
        assert(xkb_keysym_is_assigned(XKB_KEY_Aogonek));
        assert(xkb_keysym_is_assigned(XKB_KEY_Hstroke));
        assert(xkb_keysym_is_assigned(XKB_KEY_kra));
        assert(xkb_keysym_is_assigned(XKB_KEY_braille_dot_1));
        assert(xkb_keysym_is_assigned(XKB_KEY_XF86KbdLcdMenu5));
        assert(xkb_keysym_is_assigned(XKB_KEY_Shift_L));
        assert(xkb_keysym_is_assigned(XKB_KEY_XF86MonBrightnessUp));
        assert(xkb_keysym_is_assigned(XKB_KEY_VoidSymbol));
        assert(xkb_keysym_is_assigned(XKB_KEYSYM_UNICODE_MIN));
        assert(xkb_keysym_is_assigned((XKB_KEYSYM_UNICODE_MIN + XKB_KEYSYM_UNICODE_MAX) / 2));
        assert(xkb_keysym_is_assigned(XKB_KEYSYM_UNICODE_MAX));
        assert(xkb_keysym_is_assigned(XKB_KEYSYM_MAX_ASSIGNED));
        assert(!xkb_keysym_is_assigned(XKB_KEYSYM_MAX));
    
        test_modifiers_table();
    
        /* Check xkb_keysym_get_explicit_names works */
        const char* aliases[XKB_KEYSYM_EXPLICIT_ALIASES_MAX] = {0};
        int aliases_count =
            xkb_keysym_get_explicit_names(XKB_KEY_a,
                                          aliases, ARRAY_SIZE(aliases));
        assert(aliases_count == 1);
        assert_streq_not_null("", "a", aliases[0]);
    
        aliases_count =
            xkb_keysym_get_explicit_names(XKB_KEY_ISO_Group_Shift,
                                          aliases, ARRAY_SIZE(aliases));
        assert(aliases_count == XKB_KEYSYM_EXPLICIT_ALIASES_MAX);
        assert_streq_not_null("", "Mode_switch", aliases[0]);
        assert_streq_not_null("", "Arabic_switch", aliases[1]);
        assert_streq_not_null("", "Greek_switch", aliases[2]);
        assert_streq_not_null("", "Hangul_switch", aliases[3]);
        assert_streq_not_null("", "Hebrew_switch", aliases[4]);
        assert_streq_not_null("", "ISO_Group_Shift", aliases[5]);
        assert_streq_not_null("", "kana_switch", aliases[6]);
        assert_streq_not_null("", "script_switch", aliases[7]);
        assert_streq_not_null("", "SunAltGraph", aliases[8]);
    
        struct xkb_keysym_iterator *iter = xkb_keysym_iterator_new(false);
        xkb_keysym_t ks_prev = XKB_KEYSYM_MIN;
        uint32_t count = 0;
        uint32_t count_non_unicode = 0;
        while (xkb_keysym_iterator_next(iter)) {
            count++;
            xkb_keysym_t ks = xkb_keysym_iterator_get_keysym(iter);
            if (ks < XKB_KEYSYM_UNICODE_MIN || ks > XKB_KEYSYM_UNICODE_MAX) {
                count_non_unicode++;
            } else {
                const uint32_t cp = ks - XKB_KEYSYM_UNICODE_OFFSET;
                assert(cp);
                const xkb_keysym_t ks2 = xkb_utf32_to_keysym(cp);
                const char *ref;
                assert_printf(!xkb_keysym_is_deprecated(ks2, NULL, &ref),
                              "Unexpected deprecated keysym 0x%04"PRIx32" "
                              "corresponding to code point U+%04"PRIX32"\n",
                              ks2, cp);
                const uint32_t cp2 = xkb_keysym_to_utf32(ks2);
                assert_printf((ks2 == XKB_KEY_NoSymbol && cp2 == 0) ^
                              (ks2 != XKB_KEY_NoSymbol && cp2 == cp),
                              "Unexpected U+%04"PRIX32" != U+%04"PRIX32" "
                              "for keysym 0x%04"PRIx32"\n",
                              cp2, cp, ks2);
            }
            assert(ks > ks_prev || count == 1);
            ks_prev = ks;
            /* Check assigned keysyms bounds */
            assert((int32_t)XKB_KEYSYM_MIN_ASSIGNED <= (int32_t)ks && ks <= XKB_KEYSYM_MAX_ASSIGNED);
            /* Check that we do not convert UTF-32 into a deprecated keysyms */
            const char *ref = NULL;
            if (xkb_keysym_is_deprecated(ks, NULL, &ref)) {
                const uint32_t cp = xkb_keysym_to_utf32(ks);
                if (cp != 0) {
                    assert_printf(xkb_utf32_to_keysym(cp) != ks,
                                  "Unexpected xkb_utf32_to_keysym(0x%04"PRIX32") == 0x%04"PRIx32"\n",
                                  cp, ks);
                }
            }
            /* Check utf8 */
            /* Older implementation required 7 bytes for old UTF-8 (see RFC 2279) */
            char utf8[7];
            int needed = xkb_keysym_to_utf8(ks, utf8, sizeof(utf8));
            assert(0 <= needed && needed <= XKB_KEYSYM_UTF8_MAX_SIZE);
            /* Check maximum name length (`needed` does not include the ending NULL) */
            char name[XKB_KEYSYM_NAME_MAX_SIZE];
            needed = xkb_keysym_iterator_get_name(iter, name, sizeof(name));
            assert(0 < needed && (size_t)needed <= sizeof(name) - 1);
            /* Test modifier keysyms */
            bool expected = test_modifier(ks);
            bool got = xkb_keysym_is_modifier(ks);
            assert_printf(got == expected,
                          "xkb_keysym_is_modifier(0x%04"PRIx32"): expected %d, got: %d\n",
                          ks, expected, got);
            /* Test keypad keysyms */
            expected = test_keypad(ks, name);
            got = xkb_keysym_is_keypad(ks);
            assert_printf(got == expected,
                          "xkb_keysym_is_keypad(0x%04"PRIx32") \"%s\": "
                          "expected %d, got: %d\n",
                          ks, name, expected, got);
            /* Ensure no explicit name would clash with the Unicode notation */
            if (xkb_keysym_iterator_is_explicitly_named(iter)) {
                const int n = xkb_keysym_get_explicit_names(ks, aliases,
                                                            ARRAY_SIZE(aliases));
                assert(n > 0);
                for (int k = 0; k < n; k++) {
                    const char* const alias = aliases[k];
                    if (alias[0] != 'U')
                        continue;
                    uint32_t cp = 0;
                    const int r = parse_hex_to_uint32_t(name + 1, SIZE_MAX, &cp);
                    /* Either none or incomplete parsing */
                    assert((r == 0) ^ (r > 0 && name[r + 1] != '\0'));
                }
            }
    
    #if HAVE_ICU
            /* Check case mappings */
            test_icu_case_mappings(ks);
    #endif
        }
        xkb_keysym_iterator_unref(iter);
        assert(ks_prev == XKB_KEYSYM_MAX_ASSIGNED);
        assert(count == XKB_KEYSYM_UNICODE_MAX - XKB_KEYSYM_UNICODE_MIN + 1 + count_non_unicode);
    
        /* Named keysyms */
        assert(test_string("NoSymbol", XKB_KEY_NoSymbol));
        assert(test_string("Undo", 0xFF65));
        assert(test_string("UNDO", XKB_KEY_NoSymbol)); /* Require XKB_KEYSYM_CASE_INSENSITIVE */
        assert(test_string("ThisKeyShouldNotExist", XKB_KEY_NoSymbol));
        assert(test_string("XF86_Switch_VT_5", 0x1008FE05));
        assert(test_string("VoidSymbol", 0xFFFFFF));
        assert(test_string("0", 0x30));
        assert(test_string("9", 0x39));
        assert(test_string("a", 0x61));
        assert(test_string("A", 0x41));
        assert(test_string("ch", 0xfea0));
        assert(test_string("Ch", 0xfea1));
        assert(test_string("CH", 0xfea2));
        assert(test_string("THORN", 0x00de));
        assert(test_string("Thorn", 0x00de));
        assert(test_string("thorn", 0x00fe));
        assert(test_string(" thorn", XKB_KEY_NoSymbol));
        assert(test_string("thorn ", XKB_KEY_NoSymbol));
    #define LONGEST_NAME STRINGIFY2(XKB_KEYSYM_LONGEST_NAME)
    #define XKB_KEY_LONGEST_NAME CONCAT2(XKB_KEY_, XKB_KEYSYM_LONGEST_NAME)
        assert(test_string(LONGEST_NAME, XKB_KEY_LONGEST_NAME));
    #define LONGEST_CANONICAL_NAME STRINGIFY2(XKB_KEYSYM_LONGEST_CANONICAL_NAME)
    #define XKB_KEY_LONGEST_CANONICAL_NAME CONCAT2(XKB_KEY_, XKB_KEYSYM_LONGEST_CANONICAL_NAME)
        assert(test_string(LONGEST_CANONICAL_NAME, XKB_KEY_LONGEST_CANONICAL_NAME));
    
        /* Decimal keysyms are not supported (digits are special cases) */
        assert(test_string("-1", XKB_KEY_NoSymbol));
        assert(test_string("10", XKB_KEY_NoSymbol));
        assert(test_string("010", XKB_KEY_NoSymbol));
        assert(test_string("4567", XKB_KEY_NoSymbol));
    
        /* Unicode: test various ranges */
        assert(test_string("U0000", XKB_KEY_NoSymbol)); /* Min Unicode, first ASCII ctrl */
        assert(test_string("U0001", 0x01000001)); /* ASCII ctrl without named keysym */
        assert(test_string("U000A", 0x0000ff0a)); /* ASCII ctrl mapping to function keysym */
        assert(test_string("U001F", 0x0100001f)); /* Last ASCII ctrl before first printable */
        assert(test_string("U0020", 0x00000020)); /* First ASCII printable */
        assert(test_string("U007E", 0x0000007e));
        assert(test_string("U007F", 0x0000ffff)); /* Last ASCII code point */
        assert(test_string("U0080", 0x01000080)); /* First non-ASCII code point */
        assert(test_string("U009F", 0x0100009f));
        assert(test_string("U00A0", 0x000000a0));
        assert(test_string("U00FF", 0x000000ff)); /* Last Latin-1 code point */
        assert(test_string("U0100", 0x01000100));
        assert(test_string("U4567", 0x01004567));
        assert(test_string("UD7FF", 0x0100d7ff));
        assert(test_string("UD800", 0x0100d800)); /* Surrogates */
        assert(test_string("UDFFF", 0x0100dfff)); /* Surrogates */
        assert(test_string("UE000", 0x0100e000));
        assert(test_string("UFDCF", 0x0100fdcf));
        assert(test_string("UFDD0", 0x0100fdd0)); /* Noncharacter */
        assert(test_string("UFDEF", 0x0100fdef)); /* Noncharacter */
        assert(test_string("UFDF0", 0x0100fdf0));
        assert(test_string("UFFFD", 0x0100fffd));
        assert(test_string("UFFFE", 0x0100fffe)); /* Noncharacter */
        assert(test_string("UFFFF", 0x0100ffff)); /* Noncharacter */
        assert(test_string("U10000", 0x01010000));
        assert(test_string("U1F4A9", 0x0101F4A9));
        assert(test_string("U10FFFF", XKB_KEYSYM_UNICODE_MAX)); /* Max Unicode */
        assert(test_string("U110000", XKB_KEY_NoSymbol));
        /* Unicode: test syntax */
        assert(test_string("U00004567", 0x01004567));        /* OK:  8 digits */
        assert(test_string("U000004567", XKB_KEY_NoSymbol)); /* ERR: 9 digits */
        assert(test_string("U+4567", XKB_KEY_NoSymbol));     /* ERR: Standard Unicode notation */
        assert(test_string("U+4567ffff", XKB_KEY_NoSymbol));
        assert(test_string("U+4567ffffff", XKB_KEY_NoSymbol));
        assert(test_string("U-456", XKB_KEY_NoSymbol)); /* No negative number */
        assert(test_string("U456w", XKB_KEY_NoSymbol)); /* Not hexadecimal digit */
        assert(test_string("U4567   ", XKB_KEY_NoSymbol));
        assert(test_string("   U4567", XKB_KEY_NoSymbol));
        assert(test_string("U   4567", XKB_KEY_NoSymbol));
        assert(test_string("U  +4567", XKB_KEY_NoSymbol));
        assert(test_string("u4567", XKB_KEY_NoSymbol)); /* Require XKB_KEYSYM_CASE_INSENSITIVE */
    
        /* Hexadecimal: test ranges */
        assert(test_string(STRINGIFY2(XKB_KEYSYM_MIN), XKB_KEY_NoSymbol)); /* Min keysym */
        assert(test_string("0x1", 0x00000001));
        assert(test_string("0x01234567", 0x01234567));
        assert(test_string("0x09abcdef", 0x09abcdef));
        assert(test_string(STRINGIFY2(XKB_KEYSYM_UNICODE_OFFSET),
                           XKB_KEYSYM_UNICODE_OFFSET));
        assert(test_string("0x01000001", 0x01000001));
        assert(test_string("0x0100000a", 0x0100000a));
        assert(test_string("0x0100001f", 0x0100001f));
        assert(test_string("0x01000020", 0x01000020));
        assert(test_string("0x0100007e", 0x0100007e));
        assert(test_string("0x0100007f", 0x0100007f));
        assert(test_string("0x01000080", 0x01000080));
        assert(test_string("0x0100009f", 0x0100009f));
        assert(test_string("0x010000a0", 0x010000a0));
        assert(test_string("0x010000ff", 0x010000ff));
        assert(test_string(STRINGIFY2(XKB_KEYSYM_UNICODE_MIN),
                           XKB_KEYSYM_UNICODE_MIN));
        assert(test_string("0x0100d7ff", 0x0100d7ff));
        assert(test_string(STRINGIFY2(XKB_KEYSYM_UNICODE_SURROGATE_MIN),
                           XKB_KEYSYM_UNICODE_SURROGATE_MIN));
        assert(test_string(STRINGIFY2(XKB_KEYSYM_UNICODE_SURROGATE_MAX),
                           XKB_KEYSYM_UNICODE_SURROGATE_MAX));
        assert(test_string("0x0100e000", 0x0100e000));
        assert(test_string("0x0100fdcf", 0x0100fdcf));
        assert(test_string("0x0100fdd0", 0x0100fdd0)); /* Noncharacter */
        assert(test_string("0x0100fdef", 0x0100fdef)); /* Noncharacter */
        assert(test_string("0x0100fdf0", 0x0100fdf0));
        assert(test_string("0x0100fffd", 0x0100fffd));
        assert(test_string("0x0100fffe", 0x0100fffe)); /* Noncharacter */
        assert(test_string("0x0100ffff", 0x0100ffff)); /* Noncharacter */
        assert(test_string("0x01010000", 0x01010000));
        assert(test_string("0x0101fffe", 0x0101fffe)); /* Noncharacter */
        assert(test_string("0x0101ffff", 0x0101ffff)); /* Noncharacter */
        assert(test_string(STRINGIFY2(XKB_KEYSYM_UNICODE_MAX),
                           XKB_KEYSYM_UNICODE_MAX));
        assert(test_string("0x01110000", 0x1110000)); /* XKB_KEYSYM_UNICODE_MAX + 1 */
        assert(test_string(STRINGIFY2(XKB_KEYSYM_MAX), XKB_KEYSYM_MAX)); /* Max keysym */
        assert(test_string("0x20000000", XKB_KEY_NoSymbol));
        assert(test_string("0xffffffff", XKB_KEY_NoSymbol));
        assert(test_string("0x100000000", XKB_KEY_NoSymbol));
        /* Hexadecimal: test syntax */
        assert(test_string("0x10203040", 0x10203040));        /* OK:  8 digits */
        assert(test_string("0x102030400", XKB_KEY_NoSymbol)); /* ERR: 9 digits */
        assert(test_string("0x01020304", 0x1020304));         /* OK:  8 digits, starts with 0 */
        assert(test_string("0x010203040", XKB_KEY_NoSymbol)); /* ERR: 9 digits, starts with 0 */
        assert(test_string("0x+10203040", XKB_KEY_NoSymbol));
        assert(test_string("0x01020304w", XKB_KEY_NoSymbol)); /* Not hexadecimal digit */
        assert(test_string("0x102030  ", XKB_KEY_NoSymbol));
        assert(test_string("0x  102030", XKB_KEY_NoSymbol));
        assert(test_string("  0x102030", XKB_KEY_NoSymbol));
        assert(test_string("0x  +10203040", XKB_KEY_NoSymbol));
        assert(test_string("0x-10203040", XKB_KEY_NoSymbol));
        assert(test_string("0X10203040", XKB_KEY_NoSymbol)); /* Require XKB_KEYSYM_CASE_INSENSITIVE */
        assert(test_string("10203040", XKB_KEY_NoSymbol)); /* Missing prefix/decimal not implemented */
        assert(test_string("0b0101", XKB_KEY_NoSymbol)); /* Wrong prefix: binary not implemented */
        assert(test_string("0o0701", XKB_KEY_NoSymbol)); /* Wrong prefix: octal not implemented */
    
        assert(test_keysym(0x1008FF56, "XF86Close"));
        assert(test_keysym(0x0, "NoSymbol"));
        assert(test_keysym(0x1008FE20, "XF86Ungrab"));
        assert(test_keysym(XKB_KEYSYM_UNICODE_OFFSET, "0x01000000"));
        /* Longest names */
        assert(test_keysym(XKB_KEY_LONGEST_NAME, LONGEST_NAME));
        assert(test_keysym(XKB_KEY_LONGEST_CANONICAL_NAME, LONGEST_CANONICAL_NAME));
        /* Canonical names */
        assert(test_keysym(XKB_KEY_Henkan, "Henkan_Mode"));
        assert(test_keysym(XKB_KEY_ISO_Group_Shift, "Mode_switch"));
        assert(test_keysym(XKB_KEY_dead_perispomeni, "dead_tilde"));
        assert(test_keysym(XKB_KEY_guillemetleft, "guillemotleft"));
        assert(test_keysym(XKB_KEY_ordmasculine, "masculine"));
        assert(test_keysym(XKB_KEY_Greek_lambda, "Greek_lamda"));
        /* Unicode: ISO-8859-1 (Latin-1 + C0 and C1 control code) */
        assert(test_keysym(XKB_KEYSYM_UNICODE_OFFSET, STRINGIFY2(XKB_KEYSYM_UNICODE_OFFSET)));
        assert(test_keysym(0x01000001, "0x01000001"));
        assert(test_keysym(0x0100000a, "0x0100000a"));
        assert(test_keysym(0x0100001f, "0x0100001f"));
        assert(test_keysym(0x01000020, "0x01000020"));
        assert(test_keysym(0x0100007e, "0x0100007e"));
        assert(test_keysym(0x0100007f, "0x0100007f"));
        assert(test_keysym(0x01000080, "0x01000080"));
        assert(test_keysym(0x0100009f, "0x0100009f"));
        assert(test_keysym(0x010000a0, "0x010000a0"));
        assert(test_keysym(0x010000ff, "0x010000ff"));
        /* Min Unicode */
        assert(test_keysym(XKB_KEYSYM_UNICODE_MIN, "U0100"));
        assert(test_keysym(0x01001234, "U1234"));
        /* 16-bit unicode padded to width 4. */
        assert(test_keysym(0x010002DE, "U02DE"));
        /* 32-bit unicode padded to width 8. */
        assert(test_keysym(0x0101F4A9, "U1F4A9"));
        /* Surrogates */
        assert(test_keysym(0x0100d7ff, "UD7FF"));
        assert(test_keysym(XKB_KEYSYM_UNICODE_SURROGATE_MIN, "UD800"));
        assert(test_keysym(XKB_KEYSYM_UNICODE_SURROGATE_MAX, "UDFFF"));
        assert(test_keysym(0x0100e000, "UE000"));
        /* Misc. */
        assert(test_keysym(0x0100fdcf, "UFDCF"));
        /* Noncharacters */
        assert(test_keysym(0x0100fdd0, "UFDD0"));
        assert(test_keysym(0x0100fdef, "UFDEF"));
        /* Misc */
        assert(test_keysym(0x0100fdf0, "UFDF0"));
        assert(test_keysym(0x0100fffd, "UFFFD"));
        /* Noncharacters */
        assert(test_keysym(0x0100fffe, "UFFFE"));
        assert(test_keysym(0x0100ffff, "UFFFF"));
        /* Misc */
        assert(test_keysym(0x01010000, "U10000"));
        /* Max Unicode */
        assert(test_keysym(XKB_KEYSYM_UNICODE_MAX, "U10FFFF"));
        /* Max Unicode + 1 */
        assert(test_keysym(0x01110000, "0x01110000"));
        /* Min keysym. */
        assert(test_keysym(XKB_KEYSYM_MIN, "NoSymbol"));
        /* Max keysym. */
        assert(test_keysym(XKB_KEYSYM_MAX, STRINGIFY2(XKB_KEYSYM_MAX)));
        /* Outside range. */
        assert(test_keysym(XKB_KEYSYM_MAX + 1, "Invalid"));
        assert(test_keysym(0xffffffff, "Invalid"));
    
        /* Name is assumed to be correct but we provide garbage */
        const char garbage_name[] = "bla bla bla";
        assert(test_deprecated(XKB_KEY_NoSymbol, NULL, false, NULL));
        assert(test_deprecated(XKB_KEY_NoSymbol, "NoSymbol", false, NULL));
        assert(test_deprecated(XKB_KEY_A, "A", false, NULL));
        assert(test_deprecated(XKB_KEY_A, NULL, false, NULL));
        assert(test_deprecated(XKB_KEY_A, garbage_name, false, NULL));
        assert(test_deprecated(XKB_KEY_ETH, "ETH", false, "ETH"));
        assert(test_deprecated(XKB_KEY_ETH, "Eth", true, "ETH"));
        assert(test_deprecated(XKB_KEY_ETH, garbage_name, true, "ETH"));
        assert(test_deprecated(XKB_KEY_topleftradical, NULL, true, NULL));
        assert(test_deprecated(XKB_KEY_topleftradical, "topleftradical", true, NULL));
        assert(test_deprecated(XKB_KEY_topleftradical, garbage_name, true, NULL));
        assert(test_deprecated(XKB_KEY_downcaret, NULL, true, NULL));
        assert(test_deprecated(XKB_KEY_downcaret, "downcaret", true, NULL));
        /* Mixed deprecated and not deprecated aliases */
        assert(test_deprecated(XKB_KEY_Mode_switch, NULL, false, "Mode_switch"));
        assert(test_deprecated(XKB_KEY_Mode_switch, "Mode_switch", false, "Mode_switch"));
        assert(test_deprecated(XKB_KEY_Mode_switch, garbage_name, false, "Mode_switch"));
        assert(test_deprecated(XKB_KEY_ISO_Group_Shift, NULL, false, "Mode_switch"));
        assert(test_deprecated(XKB_KEY_ISO_Group_Shift, "ISO_Group_Shift", false, "Mode_switch"));
        assert(test_deprecated(XKB_KEY_ISO_Group_Shift, garbage_name, false, "Mode_switch"));
        assert(test_deprecated(XKB_KEY_SunAltGraph, NULL, false, "Mode_switch"));
        assert(test_deprecated(XKB_KEY_SunAltGraph, "SunAltGraph", true, "Mode_switch"));
        assert(test_deprecated(XKB_KEY_SunAltGraph, garbage_name, false, "Mode_switch"));
        assert(test_deprecated(XKB_KEY_notapproxeq, "notapproxeq", true, NULL));
        assert(test_deprecated(XKB_KEY_approxeq, "approxeq", true, NULL));
        /* Unicode is never deprecated */
        assert(test_deprecated(0x01002247, "U2247", false, NULL));
        assert(test_deprecated(0x01002248, "U2248", false, NULL));
        assert(test_deprecated(0x0100250C, "U250C", false, NULL));
        assert(test_deprecated(0x0100250C, "0x0100250C", false, NULL));
        assert(test_deprecated(XKB_KEYSYM_MAX, NULL, false, NULL));
        assert(test_deprecated(XKB_KEYSYM_MAX, NULL, false, NULL));
        /* Invalid keysym */
        assert(test_deprecated(0xffffffff, NULL, false, NULL));
        assert(test_deprecated(0xffffffff, NULL, false, NULL));
    
        assert(test_casestring("Undo", 0xFF65));
        assert(test_casestring("UNDO", 0xFF65));
        assert(test_casestring("A", 0x61));
        assert(test_casestring("a", 0x61));
        assert(test_casestring("ThisKeyShouldNotExist", XKB_KEY_NoSymbol));
        assert(test_casestring("XF86_Switch_vT_5", 0x1008FE05));
        assert(test_casestring("xF86_SwitcH_VT_5", 0x1008FE05));
        assert(test_casestring("xF86SwiTch_VT_5", 0x1008FE05));
        assert(test_casestring("xF86Switch_vt_5", 0x1008FE05));
        assert(test_casestring("VoidSymbol", 0xFFFFFF));
        assert(test_casestring("vOIDsymBol", 0xFFFFFF));
        assert(test_casestring("U4567", 0x1004567));
        assert(test_casestring("u4567", 0x1004567));
        assert(test_casestring("0x10203040", 0x10203040));
        assert(test_casestring("0X10203040", 0x10203040));
        assert(test_casestring("THORN", 0x00fe));
        assert(test_casestring("Thorn", 0x00fe));
        assert(test_casestring("thorn", 0x00fe));
    
        for (size_t k = 0; k < ARRAY_SIZE(ambiguous_icase_ks_names); k++) {
            test_ambiguous_icase_names(&ambiguous_icase_ks_names[k]);
        }
    
        assert(test_string("", XKB_KEY_NoSymbol));
        assert(test_casestring("", XKB_KEY_NoSymbol));
    
        /* Latin-1 keysyms (1:1 mapping in UTF-32) */
        assert(test_utf8(0x0020, "\x20"));
        assert(test_utf8(0x007e, "\x7e"));
        assert(test_utf8(0x00a0, "\xc2\xa0"));
        assert(test_utf8(0x00ff, "\xc3\xbf"));
    
        assert(test_utf8(XKB_KEY_y, "y"));
        assert(test_utf8(XKB_KEY_u, "u"));
        assert(test_utf8(XKB_KEY_m, "m"));
        assert(test_utf8(XKB_KEY_Cyrillic_em, "м"));
        assert(test_utf8(XKB_KEY_Cyrillic_u, "у"));
        assert(test_utf8(XKB_KEY_exclam, "!"));
        assert(test_utf8(XKB_KEY_oslash, "ø"));
        assert(test_utf8(XKB_KEY_hebrew_aleph, "א"));
        assert(test_utf8(XKB_KEY_Arabic_sheen, "ش"));
    
        /* Keysyms with special handling */
        assert(test_utf8(XKB_KEY_space, " "));
        assert(test_utf8(XKB_KEY_KP_Space, " "));
        assert(test_utf8(XKB_KEY_BackSpace, "\b"));
        assert(test_utf8(XKB_KEY_Escape, "\033"));
        assert(test_utf8(XKB_KEY_KP_Separator, ","));
        assert(test_utf8(XKB_KEY_KP_Decimal, "."));
        assert(test_utf8(XKB_KEY_Tab, "\t"));
        assert(test_utf8(XKB_KEY_KP_Tab, "\t"));
        assert(test_utf8(XKB_KEY_hyphen, "­"));
        assert(test_utf8(XKB_KEY_Linefeed, "\n"));
        assert(test_utf8(XKB_KEY_Return, "\r"));
        assert(test_utf8(XKB_KEY_KP_Enter, "\r"));
        assert(test_utf8(XKB_KEY_KP_Equal, "="));
        assert(test_utf8(XKB_KEY_9, "9"));
        assert(test_utf8(XKB_KEY_KP_9, "9"));
        assert(test_utf8(XKB_KEY_KP_Multiply, "*"));
        assert(test_utf8(XKB_KEY_KP_Subtract, "-"));
    
        /* Unicode keysyms */
        assert(test_utf8(XKB_KEYSYM_UNICODE_OFFSET, NULL) == 0); /* Min Unicode codepoint */
        assert(test_utf8(0x1000001, "\x01"));     /* Currently accepted, but not intended (< 0x100100) */
        assert(test_utf8(0x1000020, " "));        /* Currently accepted, but not intended (< 0x100100) */
        assert(test_utf8(0x100007f, "\x7f"));     /* Currently accepted, but not intended (< 0x100100) */
        assert(test_utf8(0x10000a0, "\xc2\xa0")); /* Currently accepted, but not intended (< 0x100100) */
        assert(test_utf8(XKB_KEYSYM_UNICODE_MIN, "Ā")); /* Min Unicode keysym */
        assert(test_utf8(0x10005d0, "א"));
        assert(test_utf8(XKB_KEYSYM_UNICODE_MAX, "\xf4\x8f\xbf\xbf")); /* Max Unicode */
        assert(test_utf8(XKB_KEYSYM_UNICODE_SURROGATE_MIN, NULL) == 0);
        assert(test_utf8(XKB_KEYSYM_UNICODE_SURROGATE_MAX, NULL) == 0);
        assert(test_utf8(XKB_KEYSYM_UNICODE_MAX + 1, NULL) == 0);
    
        assert(test_utf32_to_keysym('y', XKB_KEY_y));
        assert(test_utf32_to_keysym('u', XKB_KEY_u));
        assert(test_utf32_to_keysym('m', XKB_KEY_m));
        assert(test_utf32_to_keysym(0x43c, XKB_KEY_Cyrillic_em));
        assert(test_utf32_to_keysym(0x443, XKB_KEY_Cyrillic_u));
        assert(test_utf32_to_keysym('!', XKB_KEY_exclam));
        assert(test_utf32_to_keysym(0xF8, XKB_KEY_oslash));
        assert(test_utf32_to_keysym(0x5D0, XKB_KEY_hebrew_aleph));
        assert(test_utf32_to_keysym(0x634, XKB_KEY_Arabic_sheen));
        assert(test_utf32_to_keysym(0x1F609, 0x0101F609)); // ;) emoji
    
        assert(test_utf32_to_keysym('\b', XKB_KEY_BackSpace));
        assert(test_utf32_to_keysym('\t', XKB_KEY_Tab));
        assert(test_utf32_to_keysym('\n', XKB_KEY_Linefeed));
        assert(test_utf32_to_keysym(0x0b, XKB_KEY_Clear));
        assert(test_utf32_to_keysym('\r', XKB_KEY_Return));
        assert(test_utf32_to_keysym(0x1b, XKB_KEY_Escape));
        assert(test_utf32_to_keysym(0x7f, XKB_KEY_Delete));
    
        assert(test_utf32_to_keysym(' ', XKB_KEY_space));
        assert(test_utf32_to_keysym(',', XKB_KEY_comma));
        assert(test_utf32_to_keysym('.', XKB_KEY_period));
        assert(test_utf32_to_keysym('=', XKB_KEY_equal));
        assert(test_utf32_to_keysym('9', XKB_KEY_9));
        assert(test_utf32_to_keysym('*', XKB_KEY_asterisk));
        assert(test_utf32_to_keysym(0xd7, XKB_KEY_multiply));
        assert(test_utf32_to_keysym('-', XKB_KEY_minus));
        assert(test_utf32_to_keysym(0x10fffd, 0x110fffd));
        assert(test_utf32_to_keysym(0x20ac, XKB_KEY_EuroSign));
    
        // Unicode noncharacters
        assert(test_utf32_to_keysym(0xd800, XKB_KEY_NoSymbol)); // Unicode first surrogate
        assert(test_utf32_to_keysym(0xdfff, XKB_KEY_NoSymbol)); // Unicode last surrogate
        assert(test_utf32_to_keysym(0xfdd0, 0x100fdd0));
        assert(test_utf32_to_keysym(0xfdef, 0x100fdef));
        assert(test_utf32_to_keysym(0xfffe, 0x100fffe));
        assert(test_utf32_to_keysym(0xffff, 0x100ffff));
        assert(test_utf32_to_keysym(0x7fffe, 0x107fffe));
        assert(test_utf32_to_keysym(0x7ffff, 0x107ffff));
        assert(test_utf32_to_keysym(0xafffe, 0x10afffe));
        assert(test_utf32_to_keysym(0xaffff, 0x10affff));
    
        // Codepoints outside the Unicode planes
        assert(test_utf32_to_keysym(0x110000, XKB_KEY_NoSymbol));
        assert(test_utf32_to_keysym(0xdeadbeef, XKB_KEY_NoSymbol));
    
        assert(xkb_keysym_is_lower(XKB_KEY_a));
        assert(xkb_keysym_is_lower(XKB_KEY_Greek_lambda));
        assert(xkb_keysym_is_lower(xkb_keysym_from_name("U03b1", 0))); /* GREEK SMALL LETTER ALPHA */
        assert(xkb_keysym_is_lower(xkb_keysym_from_name("U03af", 0))); /* GREEK SMALL LETTER IOTA WITH TONOS */
    
        assert(xkb_keysym_is_upper_or_title(XKB_KEY_A));
        assert(xkb_keysym_is_upper_or_title(XKB_KEY_Greek_LAMBDA));
        assert(xkb_keysym_is_upper_or_title(xkb_keysym_from_name("U0391", 0))); /* GREEK CAPITAL LETTER ALPHA */
        assert(xkb_keysym_is_upper_or_title(xkb_keysym_from_name("U0388", 0))); /* GREEK CAPITAL LETTER EPSILON WITH TONOS */
    
        assert(!xkb_keysym_is_upper_or_title(XKB_KEY_a));
        assert(!xkb_keysym_is_lower(XKB_KEY_A));
        assert(!xkb_keysym_is_lower(XKB_KEY_Return));
        assert(!xkb_keysym_is_upper_or_title(XKB_KEY_Return));
        assert(!xkb_keysym_is_lower(XKB_KEY_hebrew_aleph));
        assert(!xkb_keysym_is_upper_or_title(XKB_KEY_hebrew_aleph));
        assert(!xkb_keysym_is_upper_or_title(xkb_keysym_from_name("U05D0", 0))); /* HEBREW LETTER ALEF */
        assert(!xkb_keysym_is_lower(xkb_keysym_from_name("U05D0", 0))); /* HEBREW LETTER ALEF */
        assert(!xkb_keysym_is_lower(XKB_KEY_8));
        assert(!xkb_keysym_is_upper_or_title(XKB_KEY_8));
    
        assert(xkb_keysym_is_keypad(XKB_KEY_KP_Enter));
        assert(xkb_keysym_is_keypad(XKB_KEY_KP_6));
        assert(xkb_keysym_is_keypad(XKB_KEY_KP_Add));
        assert(!xkb_keysym_is_keypad(XKB_KEY_Num_Lock));
        assert(!xkb_keysym_is_keypad(XKB_KEY_1));
        assert(!xkb_keysym_is_keypad(XKB_KEY_Return));
    
        assert(xkb_keysym_to_upper(XKB_KEY_a) == XKB_KEY_A);
        assert(xkb_keysym_to_upper(XKB_KEY_A) == XKB_KEY_A);
        assert(xkb_keysym_to_lower(XKB_KEY_a) == XKB_KEY_a);
        assert(xkb_keysym_to_lower(XKB_KEY_A) == XKB_KEY_a);
        assert(xkb_keysym_to_upper(XKB_KEY_Return) == XKB_KEY_Return);
        assert(xkb_keysym_to_lower(XKB_KEY_Return) == XKB_KEY_Return);
        assert(xkb_keysym_to_upper(XKB_KEY_Greek_lambda) == XKB_KEY_Greek_LAMBDA);
        assert(xkb_keysym_to_upper(XKB_KEY_Greek_LAMBDA) == XKB_KEY_Greek_LAMBDA);
        assert(xkb_keysym_to_lower(XKB_KEY_Greek_lambda) == XKB_KEY_Greek_lambda);
        assert(xkb_keysym_to_lower(XKB_KEY_Greek_LAMBDA) == XKB_KEY_Greek_lambda);
        assert(xkb_keysym_to_upper(XKB_KEY_eacute) == XKB_KEY_Eacute);
        assert(xkb_keysym_to_lower(XKB_KEY_Eacute) == XKB_KEY_eacute);
    
        /* S sharp
         * • U+00DF ß: lower case
         * •       SS: upper case (special mapping, not handled by us)
         * • U+1E9E ẞ: upper case, only for capitals
         */
    #ifndef XKB_KEY_Ssharp
    #define XKB_KEY_Ssharp (XKB_KEYSYM_UNICODE_OFFSET + 0x1E9E)
    #endif
        assert(!xkb_keysym_is_upper_or_title(XKB_KEY_ssharp));
        assert(xkb_keysym_is_upper_or_title(XKB_KEY_Ssharp));
        assert(xkb_keysym_is_lower(XKB_KEY_ssharp));
        assert(!xkb_keysym_is_lower(XKB_KEY_Ssharp));
        assert(xkb_keysym_to_upper(XKB_KEY_ssharp) == XKB_KEY_Ssharp);
        assert(xkb_keysym_to_lower(XKB_KEY_ssharp) == XKB_KEY_ssharp);
        assert(xkb_keysym_to_upper(XKB_KEY_Ssharp) == XKB_KEY_Ssharp);
        assert(xkb_keysym_to_lower(XKB_KEY_Ssharp) == XKB_KEY_ssharp);
    
        /* Title case: simple mappings
         * • U+01F1 DZ: upper case
         * • U+01F2 Dz: title case
         * • U+01F3 dz: lower case
         */
    #ifndef XKB_KEY_DZ
    #define XKB_KEY_DZ (XKB_KEYSYM_UNICODE_OFFSET + 0x01F1)
    #endif
    #ifndef XKB_KEY_Dz
    #define XKB_KEY_Dz (XKB_KEYSYM_UNICODE_OFFSET + 0x01F2)
    #endif
    #ifndef XKB_KEY_dz
    #define XKB_KEY_dz (XKB_KEYSYM_UNICODE_OFFSET + 0x01F3)
    #endif
        assert(xkb_keysym_is_upper_or_title(XKB_KEY_DZ));
        assert(xkb_keysym_is_upper_or_title(XKB_KEY_Dz));
        assert(!xkb_keysym_is_upper_or_title(XKB_KEY_dz));
        assert(!xkb_keysym_is_lower(XKB_KEY_DZ));
        assert(!xkb_keysym_is_lower(XKB_KEY_Dz));
        assert(xkb_keysym_is_lower(XKB_KEY_dz));
        assert(xkb_keysym_to_upper(XKB_KEY_DZ) == XKB_KEY_DZ);
        assert(xkb_keysym_to_lower(XKB_KEY_DZ) == XKB_KEY_dz);
        assert(xkb_keysym_to_upper(XKB_KEY_Dz) == XKB_KEY_DZ);
        assert(xkb_keysym_to_lower(XKB_KEY_Dz) == XKB_KEY_dz);
        assert(xkb_keysym_to_upper(XKB_KEY_dz) == XKB_KEY_DZ);
        assert(xkb_keysym_to_lower(XKB_KEY_dz) == XKB_KEY_dz);
    
        /* Title case: special mappings
         * • U+1F80         ᾀ: lower case
         * • U+1F88         ᾈ: title case
         * • U+1F88         ᾈ: upper case (simple)
         * • U+1F08 U+0399 ἈΙ: upper case (full)
         *
         * We do not handle special upper mapping
         */
        assert(!xkb_keysym_is_upper_or_title(XKB_KEYSYM_UNICODE_OFFSET + 0x1F80));
        assert(xkb_keysym_is_upper_or_title(XKB_KEYSYM_UNICODE_OFFSET + 0x1F88));
        assert(xkb_keysym_is_lower(XKB_KEYSYM_UNICODE_OFFSET + 0x1F80));
        assert(!xkb_keysym_is_lower(XKB_KEYSYM_UNICODE_OFFSET + 0x1F88));
        assert(xkb_keysym_to_upper(XKB_KEYSYM_UNICODE_OFFSET + 0x1F80) == XKB_KEYSYM_UNICODE_OFFSET + 0x1F88);
        assert(xkb_keysym_to_lower(XKB_KEYSYM_UNICODE_OFFSET + 0x1F80) == XKB_KEYSYM_UNICODE_OFFSET + 0x1F80);
        assert(xkb_keysym_to_upper(XKB_KEYSYM_UNICODE_OFFSET + 0x1F88) == XKB_KEYSYM_UNICODE_OFFSET + 0x1F88);
        assert(xkb_keysym_to_lower(XKB_KEYSYM_UNICODE_OFFSET + 0x1F88) == XKB_KEYSYM_UNICODE_OFFSET + 0x1F80);
    
        test_github_issue_42();
    
        return 0;
    }