/* * Copyright (C) 2010-2011 Daiki Ueno * Copyright (C) 2010-2011 Red Hat, Inc. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public License * as published by the Free Software Foundation; either version 2 of * the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA * 02110-1301 USA */ #include "config.h" #define _XOPEN_SOURCE 500 #include #include #include #include #include // TODO: this is Linux-specific #include #include "eek-keyboard.h" static void eek_key_map_deinit(struct keymap *self) { if (self->fd < 0) { g_error("Deinit called multiple times on KeyMap"); } else { close(self->fd); } self->fd = -1; } static struct keymap eek_key_map_from_str(const char *keymap_str) { struct xkb_context *context = xkb_context_new(XKB_CONTEXT_NO_FLAGS); if (!context) { g_error("No context created"); } struct xkb_keymap *keymap = xkb_keymap_new_from_string(context, keymap_str, XKB_KEYMAP_FORMAT_TEXT_V1, XKB_KEYMAP_COMPILE_NO_FLAGS); if (!keymap) g_error("Bad keymap:\n%s", keymap_str); xkb_context_unref(context); char *xkb_keymap_str = xkb_keymap_get_as_string(keymap, XKB_KEYMAP_FORMAT_TEXT_V1); size_t keymap_len = strlen(xkb_keymap_str) + 1; g_autofree char *path = strdup("/eek_keymap-XXXXXX"); char *r = &path[strlen(path) - 6]; if (getrandom(r, 6, GRND_NONBLOCK) < 0) g_error("Failed to get random numbers: %s", strerror(errno)); for (unsigned i = 0; i < 6; i++) { r[i] = (r[i] & 0b1111111) | 0b1000000; // A-z r[i] = r[i] > 'z' ? '?' : r[i]; // The randomizer doesn't need to be good... } int keymap_fd = shm_open(path, O_RDWR | O_CREAT | O_EXCL, 0600); if (keymap_fd < 0) { g_error("Failed to set up keymap fd"); } shm_unlink(path); if (ftruncate(keymap_fd, (off_t)keymap_len)) { g_error("Failed to increase keymap fd size"); } char *ptr = mmap(NULL, keymap_len, PROT_WRITE, MAP_SHARED, keymap_fd, 0); if ((void*)ptr == (void*)-1) { g_error("Failed to set up mmap"); } strncpy(ptr, xkb_keymap_str, keymap_len); munmap(ptr, keymap_len); free(xkb_keymap_str); xkb_keymap_unref(keymap); struct keymap km = { .fd = keymap_fd, .fd_len = keymap_len, }; return km; } void level_keyboard_free(LevelKeyboard *self) { eek_key_map_deinit(&self->keymap); squeek_layout_free(self->layout); g_free(self); } LevelKeyboard* level_keyboard_new (struct squeek_layout *layout) { LevelKeyboard *keyboard = g_new0(LevelKeyboard, 1); if (!keyboard) { g_error("Failed to create a keyboard"); } keyboard->layout = layout; const gchar *keymap_str = squeek_layout_get_keymap(keyboard->layout); keyboard->keymap = eek_key_map_from_str(keymap_str); return keyboard; }