123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561 |
- /************************************************************
- Copyright (c) 1993 by Silicon Graphics Computer Systems, Inc.
- Permission to use, copy, modify, and distribute this
- software and its documentation for any purpose and without
- fee is hereby granted, provided that the above copyright
- notice appear in all copies and that both that copyright
- notice and this permission notice appear in supporting
- documentation, and that the name of Silicon Graphics not be
- used in advertising or publicity pertaining to distribution
- of the software without specific prior written permission.
- Silicon Graphics makes no representation about the suitability
- of this software for any purpose. It is provided "as is"
- without any express or implied warranty.
- SILICON GRAPHICS DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS
- SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
- AND FITNESS FOR A PARTICULAR PURPOSE. IN NO EVENT SHALL SILICON
- GRAPHICS BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL
- DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE,
- DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE
- OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH
- THE USE OR PERFORMANCE OF THIS SOFTWARE.
- ********************************************************/
- #ifdef HAVE_DIX_CONFIG_H
- #include <dix-config.h>
- #endif
- #include <xkb-config.h>
- #include <stdio.h>
- #include <ctype.h>
- #include <X11/X.h>
- #include <X11/Xos.h>
- #include <X11/Xproto.h>
- #include <X11/keysym.h>
- #include <X11/extensions/XKM.h>
- #include "inputstr.h"
- #include "scrnintstr.h"
- #include "windowstr.h"
- #define XKBSRV_NEED_FILE_FUNCS
- #include <xkbsrv.h>
- #include <X11/extensions/XI.h>
- #include "xkb.h"
- /*
- * If XKM_OUTPUT_DIR specifies a path without a leading slash, it is
- * relative to the top-level XKB configuration directory.
- * Making the server write to a subdirectory of that directory
- * requires some work in the general case (install procedure
- * has to create links to /var or somesuch on many machines),
- * so we just compile into /usr/tmp for now.
- */
- #ifndef XKM_OUTPUT_DIR
- #define XKM_OUTPUT_DIR "compiled/"
- #endif
- #define PRE_ERROR_MSG "\"The XKEYBOARD keymap compiler (xkbcomp) reports:\""
- #define ERROR_PREFIX "\"> \""
- #define POST_ERROR_MSG1 "\"Errors from xkbcomp are not fatal to the X server\""
- #define POST_ERROR_MSG2 "\"End of messages from xkbcomp\""
- #if defined(WIN32)
- #define PATHSEPARATOR "\\"
- #else
- #define PATHSEPARATOR "/"
- #endif
- static unsigned
- LoadXKM(unsigned want, unsigned need, const char *keymap, XkbDescPtr *xkbRtrn);
- static void
- OutputDirectory(char *outdir, size_t size)
- {
- #ifndef WIN32
- /* Can we write an xkm and then open it too? */
- if (access(XKM_OUTPUT_DIR, W_OK | X_OK) == 0 &&
- (strlen(XKM_OUTPUT_DIR) < size)) {
- (void) strcpy(outdir, XKM_OUTPUT_DIR);
- }
- else
- #else
- if (strlen(Win32TempDir()) + 1 < size) {
- (void) strcpy(outdir, Win32TempDir());
- (void) strcat(outdir, "\\");
- }
- else
- #endif
- if (strlen("/tmp/") < size) {
- (void) strcpy(outdir, "/tmp/");
- }
- }
- /**
- * Callback invoked by XkbRunXkbComp. Write to out to talk to xkbcomp.
- */
- typedef void (*xkbcomp_buffer_callback)(FILE *out, void *userdata);
- /**
- * Start xkbcomp, let the callback write into xkbcomp's stdin. When done,
- * return a strdup'd copy of the file name we've written to.
- */
- static char *
- RunXkbComp(xkbcomp_buffer_callback callback, void *userdata)
- {
- FILE *out;
- char *buf = NULL, keymap[PATH_MAX], xkm_output_dir[PATH_MAX];
- const char *emptystring = "";
- char *xkbbasedirflag = NULL;
- const char *xkbbindir = emptystring;
- const char *xkbbindirsep = emptystring;
- #ifdef WIN32
- /* WIN32 has no popen. The input must be stored in a file which is
- used as input for xkbcomp. xkbcomp does not read from stdin. */
- char tmpname[PATH_MAX];
- const char *xkmfile = tmpname;
- #else
- const char *xkmfile = "-";
- #endif
- snprintf(keymap, sizeof(keymap), "server-%s", display);
- OutputDirectory(xkm_output_dir, sizeof(xkm_output_dir));
- #ifdef WIN32
- strcpy(tmpname, Win32TempDir());
- strcat(tmpname, "\\xkb_XXXXXX");
- (void) mktemp(tmpname);
- #endif
- if (XkbBaseDirectory != NULL) {
- if (asprintf(&xkbbasedirflag, "\"-R%s\"", XkbBaseDirectory) == -1)
- xkbbasedirflag = NULL;
- }
- if (XkbBinDirectory != NULL) {
- int ld = strlen(XkbBinDirectory);
- int lps = strlen(PATHSEPARATOR);
- xkbbindir = XkbBinDirectory;
- if ((ld >= lps) && (strcmp(xkbbindir + ld - lps, PATHSEPARATOR) != 0)) {
- xkbbindirsep = PATHSEPARATOR;
- }
- }
- if (asprintf(&buf,
- "\"%s%sxkbcomp\" -w %d %s -xkm \"%s\" "
- "-em1 %s -emp %s -eml %s \"%s%s.xkm\"",
- xkbbindir, xkbbindirsep,
- ((xkbDebugFlags < 2) ? 1 :
- ((xkbDebugFlags > 10) ? 10 : (int) xkbDebugFlags)),
- xkbbasedirflag ? xkbbasedirflag : "", xkmfile,
- PRE_ERROR_MSG, ERROR_PREFIX, POST_ERROR_MSG1,
- xkm_output_dir, keymap) == -1)
- buf = NULL;
- free(xkbbasedirflag);
- if (!buf) {
- LogMessage(X_ERROR,
- "XKB: Could not invoke xkbcomp: not enough memory\n");
- return NULL;
- }
- #ifndef WIN32
- out = Popen(buf, "w");
- #else
- out = fopen(tmpname, "w");
- #endif
- if (out != NULL) {
- /* Now write to xkbcomp */
- (*callback)(out, userdata);
- #ifndef WIN32
- if (Pclose(out) == 0)
- #else
- if (fclose(out) == 0 && System(buf) >= 0)
- #endif
- {
- if (xkbDebugFlags)
- DebugF("[xkb] xkb executes: %s\n", buf);
- free(buf);
- #ifdef WIN32
- unlink(tmpname);
- #endif
- return xnfstrdup(keymap);
- }
- else
- LogMessage(X_ERROR, "Error compiling keymap (%s)\n", keymap);
- #ifdef WIN32
- /* remove the temporary file */
- unlink(tmpname);
- #endif
- }
- else {
- #ifndef WIN32
- LogMessage(X_ERROR, "XKB: Could not invoke xkbcomp\n");
- #else
- LogMessage(X_ERROR, "Could not open file %s\n", tmpname);
- #endif
- }
- free(buf);
- return NULL;
- }
- typedef struct {
- XkbDescPtr xkb;
- XkbComponentNamesPtr names;
- unsigned int want;
- unsigned int need;
- } XkbKeymapNamesCtx;
- static void
- xkb_write_keymap_for_names_cb(FILE *out, void *userdata)
- {
- XkbKeymapNamesCtx *ctx = userdata;
- #ifdef DEBUG
- if (xkbDebugFlags) {
- ErrorF("[xkb] XkbDDXCompileKeymapByNames compiling keymap:\n");
- XkbWriteXKBKeymapForNames(stderr, ctx->names, ctx->xkb, ctx->want, ctx->need);
- }
- #endif
- XkbWriteXKBKeymapForNames(out, ctx->names, ctx->xkb, ctx->want, ctx->need);
- }
- static Bool
- XkbDDXCompileKeymapByNames(XkbDescPtr xkb,
- XkbComponentNamesPtr names,
- unsigned want,
- unsigned need, char *nameRtrn, int nameRtrnLen)
- {
- char *keymap;
- Bool rc = FALSE;
- XkbKeymapNamesCtx ctx = {
- .xkb = xkb,
- .names = names,
- .want = want,
- .need = need
- };
- keymap = RunXkbComp(xkb_write_keymap_for_names_cb, &ctx);
- if (keymap) {
- if(nameRtrn)
- strlcpy(nameRtrn, keymap, nameRtrnLen);
- free(keymap);
- rc = TRUE;
- } else if (nameRtrn)
- *nameRtrn = '\0';
- return rc;
- }
- typedef struct {
- const char *keymap;
- size_t len;
- } XkbKeymapString;
- static void
- xkb_write_keymap_string_cb(FILE *out, void *userdata)
- {
- XkbKeymapString *s = userdata;
- fwrite(s->keymap, s->len, 1, out);
- }
- static unsigned int
- XkbDDXLoadKeymapFromString(DeviceIntPtr keybd,
- const char *keymap, int keymap_length,
- unsigned int want,
- unsigned int need,
- XkbDescPtr *xkbRtrn)
- {
- unsigned int have;
- char *map_name;
- XkbKeymapString map = {
- .keymap = keymap,
- .len = keymap_length
- };
- *xkbRtrn = NULL;
- map_name = RunXkbComp(xkb_write_keymap_string_cb, &map);
- if (!map_name) {
- LogMessage(X_ERROR, "XKB: Couldn't compile keymap\n");
- return 0;
- }
- have = LoadXKM(want, need, map_name, xkbRtrn);
- free(map_name);
- return have;
- }
- static FILE *
- XkbDDXOpenConfigFile(const char *mapName, char *fileNameRtrn, int fileNameRtrnLen)
- {
- char buf[PATH_MAX], xkm_output_dir[PATH_MAX];
- FILE *file;
- buf[0] = '\0';
- if (mapName != NULL) {
- OutputDirectory(xkm_output_dir, sizeof(xkm_output_dir));
- if ((XkbBaseDirectory != NULL) && (xkm_output_dir[0] != '/')
- #ifdef WIN32
- && (!isalpha(xkm_output_dir[0]) || xkm_output_dir[1] != ':')
- #endif
- ) {
- if (snprintf(buf, PATH_MAX, "%s/%s%s.xkm", XkbBaseDirectory,
- xkm_output_dir, mapName) >= PATH_MAX)
- buf[0] = '\0';
- }
- else {
- if (snprintf(buf, PATH_MAX, "%s%s.xkm", xkm_output_dir, mapName)
- >= PATH_MAX)
- buf[0] = '\0';
- }
- if (buf[0] != '\0')
- file = fopen(buf, "rb");
- else
- file = NULL;
- }
- else
- file = NULL;
- if ((fileNameRtrn != NULL) && (fileNameRtrnLen > 0)) {
- strlcpy(fileNameRtrn, buf, fileNameRtrnLen);
- }
- return file;
- }
- static unsigned
- LoadXKM(unsigned want, unsigned need, const char *keymap, XkbDescPtr *xkbRtrn)
- {
- FILE *file;
- char fileName[PATH_MAX];
- unsigned missing;
- file = XkbDDXOpenConfigFile(keymap, fileName, PATH_MAX);
- if (file == NULL) {
- LogMessage(X_ERROR, "Couldn't open compiled keymap file %s\n",
- fileName);
- return 0;
- }
- missing = XkmReadFile(file, need, want, xkbRtrn);
- if (*xkbRtrn == NULL) {
- LogMessage(X_ERROR, "Error loading keymap %s\n", fileName);
- fclose(file);
- (void) unlink(fileName);
- return 0;
- }
- else {
- DebugF("Loaded XKB keymap %s, defined=0x%x\n", fileName,
- (*xkbRtrn)->defined);
- }
- fclose(file);
- (void) unlink(fileName);
- return (need | want) & (~missing);
- }
- unsigned
- XkbDDXLoadKeymapByNames(DeviceIntPtr keybd,
- XkbComponentNamesPtr names,
- unsigned want,
- unsigned need,
- XkbDescPtr *xkbRtrn, char *nameRtrn, int nameRtrnLen)
- {
- XkbDescPtr xkb;
- *xkbRtrn = NULL;
- if ((keybd == NULL) || (keybd->key == NULL) ||
- (keybd->key->xkbInfo == NULL))
- xkb = NULL;
- else
- xkb = keybd->key->xkbInfo->desc;
- if ((names->keycodes == NULL) && (names->types == NULL) &&
- (names->compat == NULL) && (names->symbols == NULL) &&
- (names->geometry == NULL)) {
- LogMessage(X_ERROR, "XKB: No components provided for device %s\n",
- keybd->name ? keybd->name : "(unnamed keyboard)");
- return 0;
- }
- else if (!XkbDDXCompileKeymapByNames(xkb, names, want, need,
- nameRtrn, nameRtrnLen)) {
- LogMessage(X_ERROR, "XKB: Couldn't compile keymap\n");
- return 0;
- }
- return LoadXKM(want, need, nameRtrn, xkbRtrn);
- }
- Bool
- XkbDDXNamesFromRules(DeviceIntPtr keybd,
- const char *rules_name,
- XkbRF_VarDefsPtr defs, XkbComponentNamesPtr names)
- {
- char buf[PATH_MAX];
- FILE *file;
- Bool complete;
- XkbRF_RulesPtr rules;
- if (!rules_name)
- return FALSE;
- if (snprintf(buf, PATH_MAX, "%s/rules/%s", XkbBaseDirectory, rules_name)
- >= PATH_MAX) {
- LogMessage(X_ERROR, "XKB: Rules name is too long\n");
- return FALSE;
- }
- file = fopen(buf, "r");
- if (!file) {
- LogMessage(X_ERROR, "XKB: Couldn't open rules file %s\n", buf);
- return FALSE;
- }
- rules = XkbRF_Create();
- if (!rules) {
- LogMessage(X_ERROR, "XKB: Couldn't create rules struct\n");
- fclose(file);
- return FALSE;
- }
- if (!XkbRF_LoadRules(file, rules)) {
- LogMessage(X_ERROR, "XKB: Couldn't parse rules file %s\n", rules_name);
- fclose(file);
- XkbRF_Free(rules, TRUE);
- return FALSE;
- }
- memset(names, 0, sizeof(*names));
- complete = XkbRF_GetComponents(rules, defs, names);
- fclose(file);
- XkbRF_Free(rules, TRUE);
- if (!complete)
- LogMessage(X_ERROR, "XKB: Rules returned no components\n");
- return complete;
- }
- static Bool
- XkbRMLVOtoKcCGST(DeviceIntPtr dev, XkbRMLVOSet * rmlvo,
- XkbComponentNamesPtr kccgst)
- {
- XkbRF_VarDefsRec mlvo;
- mlvo.model = rmlvo->model;
- mlvo.layout = rmlvo->layout;
- mlvo.variant = rmlvo->variant;
- mlvo.options = rmlvo->options;
- return XkbDDXNamesFromRules(dev, rmlvo->rules, &mlvo, kccgst);
- }
- /**
- * Compile the given RMLVO keymap and return it. Returns the XkbDescPtr on
- * success or NULL on failure. If the components compiled are not a superset
- * or equal to need, the compiliation is treated as failure.
- */
- static XkbDescPtr
- XkbCompileKeymapForDevice(DeviceIntPtr dev, XkbRMLVOSet * rmlvo, int need)
- {
- XkbDescPtr xkb = NULL;
- unsigned int provided;
- XkbComponentNamesRec kccgst = { 0 };
- char name[PATH_MAX];
- if (XkbRMLVOtoKcCGST(dev, rmlvo, &kccgst)) {
- provided =
- XkbDDXLoadKeymapByNames(dev, &kccgst, XkmAllIndicesMask, need, &xkb,
- name, PATH_MAX);
- if ((need & provided) != need) {
- if (xkb) {
- XkbFreeKeyboard(xkb, 0, TRUE);
- xkb = NULL;
- }
- }
- }
- XkbFreeComponentNames(&kccgst, FALSE);
- return xkb;
- }
- static XkbDescPtr
- KeymapOrDefaults(DeviceIntPtr dev, XkbDescPtr xkb)
- {
- XkbRMLVOSet dflts;
- if (xkb)
- return xkb;
- /* we didn't get what we really needed. And that will likely leave
- * us with a keyboard that doesn't work. Use the defaults instead */
- LogMessage(X_ERROR, "XKB: Failed to load keymap. Loading default "
- "keymap instead.\n");
- XkbGetRulesDflts(&dflts);
- xkb = XkbCompileKeymapForDevice(dev, &dflts, 0);
- XkbFreeRMLVOSet(&dflts, FALSE);
- return xkb;
- }
- XkbDescPtr
- XkbCompileKeymap(DeviceIntPtr dev, XkbRMLVOSet * rmlvo)
- {
- XkbDescPtr xkb;
- unsigned int need;
- if (!dev || !rmlvo) {
- LogMessage(X_ERROR, "XKB: No device or RMLVO specified\n");
- return NULL;
- }
- /* These are the components we really really need */
- need = XkmSymbolsMask | XkmCompatMapMask | XkmTypesMask |
- XkmKeyNamesMask | XkmVirtualModsMask;
- xkb = XkbCompileKeymapForDevice(dev, rmlvo, need);
- return KeymapOrDefaults(dev, xkb);
- }
- XkbDescPtr
- XkbCompileKeymapFromString(DeviceIntPtr dev,
- const char *keymap, int keymap_length)
- {
- XkbDescPtr xkb;
- unsigned int need, provided;
- if (!dev || !keymap) {
- LogMessage(X_ERROR, "XKB: No device or keymap specified\n");
- return NULL;
- }
- /* These are the components we really really need */
- need = XkmSymbolsMask | XkmCompatMapMask | XkmTypesMask |
- XkmKeyNamesMask | XkmVirtualModsMask;
- provided =
- XkbDDXLoadKeymapFromString(dev, keymap, keymap_length,
- XkmAllIndicesMask, need, &xkb);
- if ((need & provided) != need) {
- if (xkb) {
- XkbFreeKeyboard(xkb, 0, TRUE);
- xkb = NULL;
- }
- }
- return KeymapOrDefaults(dev, xkb);
- }
|