Add interface to compile keyboard description from keymap files
authorDan Nicholson <dbn.lists@gmail.com>
Fri, 10 Apr 2009 19:25:51 +0000 (12:25 -0700)
committerDan Nicholson <dbn.lists@gmail.com>
Fri, 10 Apr 2009 19:25:51 +0000 (12:25 -0700)
We need to support generating a keyboard description from a keymap file
because there are just some cases where RMLVO or ktcsg is not enough.

The map choosing logic has been refactored into its own function and now
supports choosing a named or default keymap.

include/X11/extensions/XKBcommon.h
src/xkbcomp/xkbcomp.c

index 1bc0f35..1a53658 100644 (file)
@@ -28,6 +28,7 @@ authorization from the authors.
 #ifndef _XKBCOMMON_H_
 #define _XKBCOMMON_H_
 
+#include <stdio.h>
 #include <X11/Xfuncproto.h>
 
 /* Action structures used in the server */
@@ -134,6 +135,9 @@ XkbcCompileKeymapFromRules(const XkbRMLVOSet *rmlvo);
 extern XkbcDescPtr
 XkbcCompileKeymapFromComponents(const XkbComponentNamesPtr ktcsg);
 
+extern XkbcDescPtr
+XkbcCompileKeymapFromFile(FILE *inputFile, const char *mapName);
+
 _XFUNCPROTOEND
 
 #endif /* _XKBCOMMON_H_ */
index 89753da..cfd73c5 100644 (file)
@@ -156,6 +156,40 @@ XkbcCompileKeymapFromRules(const XkbRMLVOSet *rmlvo)
     return xkb;
 }
 
+static XkbFile *
+XkbChooseMap(XkbFile *file, const char *name)
+{
+    XkbFile *map = file;
+
+    /* map specified? */
+    if (name) {
+        while (map) {
+            if (map->name && strcmp(map->name, name) == 0)
+                break;
+            map = (XkbFile *) map->common.next;
+        }
+
+        if (!map)
+            ERROR("no map named \"%s\" in input file\n", name);
+    }
+    else if (file->common.next) {
+        /* look for map with XkbLC_Default flag. */
+        for (; map; map = (XkbFile *) map->common.next) {
+            if (map->flags & XkbLC_Default)
+                break;
+        }
+
+        if (!map) {
+            map = file;
+            WARN("no map specified, but components have several\n");
+            WARN("using the first defined map, \"%s\"\n",
+                 map->name ? map->name : "");
+        }
+    }
+
+    return map;
+}
+
 XkbcDescPtr
 XkbcCompileKeymapFromComponents(const XkbComponentNamesPtr ktcsg)
 {
@@ -173,16 +207,58 @@ XkbcCompileKeymapFromComponents(const XkbComponentNamesPtr ktcsg)
     }
 
     /* Find map to use */
-    mapToUse = file;
-    if (file->common.next) {
-        for (; mapToUse; mapToUse = (XkbFile *)mapToUse->common.next) {
-            if (mapToUse->flags & XkbLC_Default)
-                break;
-        }
-        if (!mapToUse) {
-            mapToUse = file;
-            WARN("no map specified, but components have several\n");
-        }
+    if (!(mapToUse = XkbChooseMap(file, NULL)))
+        goto unwind_file;
+
+    /* Compile the keyboard */
+    if (!(xkb = XkbcAllocKeyboard())) {
+        ERROR("could not allocate keyboard description\n");
+        goto unwind_file;
+    }
+
+    if (!CompileKeymap(mapToUse, xkb, MergeReplace)) {
+        ERROR("failed to compile keymap\n");
+        goto unwind_xkb;
+    }
+
+    return xkb;
+unwind_xkb:
+    XkbcFreeKeyboard(xkb, XkbAllComponentsMask, True);
+unwind_file:
+    /* XXX: here's where we would free the XkbFile */
+fail:
+    return NULL;
+}
+
+XkbcDescPtr
+XkbcCompileKeymapFromFile(FILE *inputFile, const char *mapName)
+{
+    XkbFile *file, *mapToUse;
+    XkbcDescPtr xkb;
+
+    if (!inputFile) {
+        ERROR("no file specified to generate XKB keymap\n");
+        goto fail;
+    }
+
+    setScanState("input", 1);
+    if (!XKBParseFile(inputFile, &file) || !file) {
+        ERROR("failed to parse input xkb file\n");
+        goto fail;
+    }
+
+    /* Find map to use */
+    if (!(mapToUse = XkbChooseMap(file, mapName)))
+        goto unwind_file;
+
+    switch (mapToUse->type) {
+    case XkmSemanticsFile:
+    case XkmLayoutFile:
+    case XkmKeymapFile:
+        break;
+    default:
+        ERROR("file type %d not handled\n", mapToUse->type);
+        goto unwind_file;
     }
 
     /* Compile the keyboard */