aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorDrew DeVault <sir@cmpwn.com>2015-12-10 18:34:12 -0500
committerDrew DeVault <sir@cmpwn.com>2015-12-10 18:34:12 -0500
commita7710c5537cb005acaeb2afe9a53bafa3e022817 (patch)
treeb24327e97548fc528895e82f000b6e668cab93ac
parent266393a705ce2b6db982a4bd048bb536ec556a85 (diff)
Initialize keyboard in registry poll
-rw-r--r--include/client/registry.h55
-rw-r--r--wayland/CMakeLists.txt2
-rw-r--r--wayland/registry.c71
3 files changed, 128 insertions, 0 deletions
diff --git a/include/client/registry.h b/include/client/registry.h
index 5e47b18e..2ab02174 100644
--- a/include/client/registry.h
+++ b/include/client/registry.h
@@ -2,16 +2,70 @@
#define _SWAY_CLIENT_REGISTRY_H
#include <wayland-client.h>
+#include <xkbcommon/xkbcommon.h>
#include "wayland-desktop-shell-client-protocol.h"
#include "wayland-swaylock-client-protocol.h"
#include "list.h"
+enum mod_bit {
+ MOD_SHIFT = 1<<0,
+ MOD_CAPS = 1<<1,
+ MOD_CTRL = 1<<2,
+ MOD_ALT = 1<<3,
+ MOD_MOD2 = 1<<4,
+ MOD_MOD3 = 1<<5,
+ MOD_LOGO = 1<<6,
+ MOD_MOD5 = 1<<7,
+};
+
+enum mask {
+ MASK_SHIFT,
+ MASK_CAPS,
+ MASK_CTRL,
+ MASK_ALT,
+ MASK_MOD2,
+ MASK_MOD3,
+ MASK_LOGO,
+ MASK_MOD5,
+ MASK_LAST
+};
+
struct output_state {
struct wl_output *output;
uint32_t flags;
uint32_t width, height;
};
+struct xkb {
+ struct xkb_state *state;
+ struct xkb_context *context;
+ struct xkb_keymap *keymap;
+ xkb_mod_mask_t masks[MASK_LAST];
+};
+
+struct input {
+ int *repeat_fd;
+
+ struct xkb xkb;
+
+ xkb_keysym_t sym;
+ uint32_t code;
+ uint32_t last_code;
+ uint32_t modifiers;
+
+ xkb_keysym_t repeat_sym;
+ uint32_t repeat_key;
+
+ int32_t repeat_rate_sec;
+ int32_t repeat_rate_nsec;
+ int32_t repeat_delay_sec;
+ int32_t repeat_delay_nsec;
+
+ struct {
+ void (*key)(enum wl_keyboard_key_state state, xkb_keysym_t sym, uint32_t code);
+ } notify;
+};
+
struct registry {
struct wl_compositor *compositor;
struct wl_display *display;
@@ -22,6 +76,7 @@ struct registry {
struct wl_shm *shm;
struct desktop_shell *desktop_shell;
struct lock *swaylock;
+ struct input *input;
list_t *outputs;
};
diff --git a/wayland/CMakeLists.txt b/wayland/CMakeLists.txt
index 8e19fc03..6519cd7d 100644
--- a/wayland/CMakeLists.txt
+++ b/wayland/CMakeLists.txt
@@ -2,6 +2,7 @@ include_directories(
${PROTOCOLS_INCLUDE_DIRS}
${PANGO_INCLUDE_DIRS}
${GDK_PIXBUF_INCLUDE_DIRS}
+ ${XKBCOMMON_INCLUDE_DIRS}
)
add_library(sway-wayland
@@ -17,4 +18,5 @@ target_link_libraries(sway-wayland
sway-protocols
${PANGO_LIBRARIES}
${GDK_PIXBUF_LIBRARIES}
+ ${XKBCOMMON_LIBRARIES}
)
diff --git a/wayland/registry.c b/wayland/registry.c
index 883c69d3..45735ccd 100644
--- a/wayland/registry.c
+++ b/wayland/registry.c
@@ -1,6 +1,10 @@
#include <wayland-client.h>
+#include <xkbcommon/xkbcommon.h>
#include <stdlib.h>
#include <string.h>
+#include <unistd.h>
+#include <sys/mman.h>
+#include <sys/timerfd.h>
#include "wayland-desktop-shell-client-protocol.h"
#include "wayland-swaylock-client-protocol.h"
#include "client/registry.h"
@@ -38,8 +42,73 @@ static const struct wl_output_listener output_listener = {
.scale = display_handle_scale
};
+const char *XKB_MASK_NAMES[MASK_LAST] = {
+ XKB_MOD_NAME_SHIFT,
+ XKB_MOD_NAME_CAPS,
+ XKB_MOD_NAME_CTRL,
+ XKB_MOD_NAME_ALT,
+ "Mod2",
+ "Mod3",
+ XKB_MOD_NAME_LOGO,
+ "Mod5",
+};
+
+const enum mod_bit XKB_MODS[MASK_LAST] = {
+ MOD_SHIFT,
+ MOD_CAPS,
+ MOD_CTRL,
+ MOD_ALT,
+ MOD_MOD2,
+ MOD_MOD3,
+ MOD_LOGO,
+ MOD_MOD5
+};
+
static void keyboard_handle_keymap(void *data, struct wl_keyboard *keyboard,
uint32_t format, int fd, uint32_t size) {
+ // Keyboard errors are abort-worthy because you wouldn't be able to unlock your screen otherwise.
+
+ struct registry *registry = data;
+ if (!data) {
+ close(fd);
+ return;
+ }
+
+ if (format != WL_KEYBOARD_KEYMAP_FORMAT_XKB_V1) {
+ close(fd);
+ sway_abort("Unknown keymap format %d, aborting", format);
+ }
+
+ char *map_str = mmap(NULL, size, PROT_READ, MAP_SHARED, fd, 0);
+ if (map_str == MAP_FAILED) {
+ close(fd);
+ sway_abort("Unable to initialized shared keyboard memory, aborting");
+ }
+
+ struct xkb_keymap *keymap = xkb_keymap_new_from_string(registry->input->xkb.context,
+ map_str, XKB_KEYMAP_FORMAT_TEXT_V1, 0);
+ munmap(map_str, size);
+ close(fd);
+
+ if (!keymap) {
+ sway_abort("Failed to compile keymap, aborting");
+ }
+
+ struct xkb_state *state = xkb_state_new(keymap);
+ if (!state) {
+ xkb_keymap_unref(keymap);
+ sway_abort("Failed to create xkb state, aborting");
+ }
+
+ xkb_keymap_unref(registry->input->xkb.keymap);
+ xkb_state_unref(registry->input->xkb.state);
+ registry->input->xkb.keymap = keymap;
+ registry->input->xkb.state = state;
+
+ int i;
+ for (i = 0; i < MASK_LAST; ++i) {
+ registry->input->xkb.masks[i] = 1 << xkb_keymap_mod_get_index(registry->input->xkb.keymap, XKB_MASK_NAMES[i]);
+ }
}
static void keyboard_handle_enter(void *data, struct wl_keyboard *keyboard,
@@ -115,6 +184,8 @@ struct registry *registry_poll(void) {
struct registry *registry = malloc(sizeof(struct registry));
memset(registry, 0, sizeof(struct registry));
registry->outputs = create_list();
+ registry->input = calloc(sizeof(struct input), 1);
+ registry->input->xkb.context = xkb_context_new(XKB_CONTEXT_NO_FLAGS);
registry->display = wl_display_connect(NULL);
if (!registry->display) {