Edit

kc3-lang/libxkbcommon/src/xkbcomp/xkbcomp.c

Branch :

  • Show log

    Commit

  • Author : Pierre Le Marre
    Date : 2025-03-13 21:26:59
    Hash : 8e92f25e
    Message : rules: Added xkb_components_names_from_rules() This is mainly for debugging purposes and to enable displaying KcCGST values from RMLVO resolution in `xkbcli compile-keymap --kccgst`.

  • src/xkbcomp/xkbcomp.c
  • /*
     * Copyright © 2009 Dan Nicholson
     * Copyright © 2012 Intel Corporation
     * Copyright © 2012 Ran Benita <ran234@gmail.com>
     * SPDX-License-Identifier: MIT
     *
     * Author: Dan Nicholson <dbn.lists@gmail.com>
     * Author: Ran Benita <ran234@gmail.com>
     * Author: Daniel Stone <daniel@fooishbar.org>
     */
    
    #include "config.h"
    
    #include <stdbool.h>
    #include "xkbcomp-priv.h"
    #include "rules.h"
    
    bool
    xkb_components_names_from_rules(struct xkb_context *ctx,
                                    const struct xkb_rule_names *rmlvo_in,
                                    struct xkb_rule_names *rmlvo_out,
                                    struct xkb_component_names *components_out)
    {
        /* Resolve default RMLVO values. We need a mutable copy of the input. */
        struct xkb_rule_names rmlvo = *rmlvo_in;
        xkb_context_sanitize_rule_names(ctx, &rmlvo);
    
        if (rmlvo_out) {
            /* Copy the sanitized RMLVO, if required. */
            *rmlvo_out = rmlvo;
        }
        if (!components_out) {
            /* KcCGST not required.
             * While RMLVO and KcCGST outputs are both optional, at least one must
             * be provided. */
            return !!rmlvo_out;
        }
    
        /* Resolve the RMLVO names to KcCGST components */
        *components_out = (struct xkb_component_names){ 0 };
        return xkb_components_from_rules(ctx, &rmlvo, components_out, NULL);
    }
    
    static bool
    compile_keymap_file(struct xkb_keymap *keymap, XkbFile *file)
    {
        if (file->file_type != FILE_TYPE_KEYMAP) {
            log_err(keymap->ctx, XKB_ERROR_KEYMAP_COMPILATION_FAILED,
                    "Cannot compile a %s file alone into a keymap\n",
                    xkb_file_type_to_string(file->file_type));
            return false;
        }
    
        if (!CompileKeymap(file, keymap, MERGE_OVERRIDE)) {
            log_err(keymap->ctx, XKB_ERROR_KEYMAP_COMPILATION_FAILED,
                    "Failed to compile keymap\n");
            return false;
        }
    
        return true;
    }
    
    static bool
    text_v1_keymap_new_from_names(struct xkb_keymap *keymap,
                                  const struct xkb_rule_names *rmlvo)
    {
        bool ok;
        struct xkb_component_names kccgst;
        XkbFile *file;
    
        log_dbg(keymap->ctx, XKB_LOG_MESSAGE_NO_ID,
                "Compiling from RMLVO: rules '%s', model '%s', layout '%s', "
                "variant '%s', options '%s'\n",
                rmlvo->rules, rmlvo->model, rmlvo->layout, rmlvo->variant,
                rmlvo->options);
    
        /* Resolve the RMLVO components to KcCGST components and get the
         * expected number of layouts */
        ok = xkb_components_from_rules(keymap->ctx, rmlvo, &kccgst,
                                       &keymap->num_groups);
        if (!ok) {
            log_err(keymap->ctx, XKB_ERROR_KEYMAP_COMPILATION_FAILED,
                    "Couldn't look up rules '%s', model '%s', layout '%s', "
                    "variant '%s', options '%s'\n",
                    rmlvo->rules, rmlvo->model, rmlvo->layout, rmlvo->variant,
                    rmlvo->options);
            return false;
        }
    
        log_dbg(keymap->ctx, XKB_LOG_MESSAGE_NO_ID,
                "Compiling from KcCGST: keycodes '%s', types '%s', "
                "compat '%s', symbols '%s'\n",
                kccgst.keycodes, kccgst.types, kccgst.compatibility, kccgst.symbols);
    
        file = XkbFileFromComponents(keymap->ctx, &kccgst);
    
        free(kccgst.keycodes);
        free(kccgst.types);
        free(kccgst.compatibility);
        free(kccgst.symbols);
        free(kccgst.geometry);
    
        if (!file) {
            log_err(keymap->ctx, XKB_ERROR_KEYMAP_COMPILATION_FAILED,
                    "Failed to generate parsed XKB file from components\n");
            return false;
        }
    
        ok = compile_keymap_file(keymap, file);
        FreeXkbFile(file);
        return ok;
    }
    
    static bool
    text_v1_keymap_new_from_string(struct xkb_keymap *keymap,
                                   const char *string, size_t len)
    {
        bool ok;
        XkbFile *xkb_file;
    
        xkb_file = XkbParseString(keymap->ctx, string, len, "(input string)", NULL);
        if (!xkb_file) {
            log_err(keymap->ctx, XKB_ERROR_KEYMAP_COMPILATION_FAILED,
                    "Failed to parse input xkb string\n");
            return false;
        }
    
        ok = compile_keymap_file(keymap, xkb_file);
        FreeXkbFile(xkb_file);
        return ok;
    }
    
    static bool
    text_v1_keymap_new_from_file(struct xkb_keymap *keymap, FILE *file)
    {
        bool ok;
        XkbFile *xkb_file;
    
        xkb_file = XkbParseFile(keymap->ctx, file, "(unknown file)", NULL);
        if (!xkb_file) {
            log_err(keymap->ctx, XKB_ERROR_KEYMAP_COMPILATION_FAILED,
                    "Failed to parse input xkb file\n");
            return false;
        }
    
        ok = compile_keymap_file(keymap, xkb_file);
        FreeXkbFile(xkb_file);
        return ok;
    }
    
    const struct xkb_keymap_format_ops text_v1_keymap_format_ops = {
        .keymap_new_from_names = text_v1_keymap_new_from_names,
        .keymap_new_from_string = text_v1_keymap_new_from_string,
        .keymap_new_from_file = text_v1_keymap_new_from_file,
        .keymap_get_as_string = text_v1_keymap_get_as_string,
    };