708deb9
From f8958c3495edf6d1563a5309e84bd68931a46213 Mon Sep 17 00:00:00 2001
708deb9
From: David Herrmann <dh.herrmann@gmail.com>
708deb9
Date: Fri, 3 Oct 2014 12:50:41 +0200
708deb9
Subject: [PATCH] terminal/screen: add keyboard mapping
708deb9
708deb9
Implement the feed_keyboard() handling by mapping XKB keys according to
708deb9
DEC-VT behavior.
708deb9
708deb9
Public information on terminal key-mappings is pretty scarce. We only
708deb9
implement the most basic mapping for now. Further improvements welcome!
708deb9
---
708deb9
 src/libsystemd-terminal/term-screen.c | 324 +++++++++++++++++++++++++++++++++-
708deb9
 src/libsystemd-terminal/term.h        |  22 ++-
708deb9
 2 files changed, 342 insertions(+), 4 deletions(-)
708deb9
708deb9
diff --git a/src/libsystemd-terminal/term-screen.c b/src/libsystemd-terminal/term-screen.c
708deb9
index b442b96050..5b0562e4c3 100644
708deb9
--- a/src/libsystemd-terminal/term-screen.c
708deb9
+++ b/src/libsystemd-terminal/term-screen.c
708deb9
@@ -47,6 +47,7 @@
708deb9
 #include <stdbool.h>
708deb9
 #include <stdint.h>
708deb9
 #include <stdlib.h>
708deb9
+#include <xkbcommon/xkbcommon-keysyms.h>
708deb9
 #include "macro.h"
708deb9
 #include "term-internal.h"
708deb9
 #include "util.h"
708deb9
@@ -3784,12 +3785,329 @@ int term_screen_feed_text(term_screen *screen, const uint8_t *in, size_t size) {
708deb9
         return 0;
708deb9
 }
708deb9
 
708deb9
-int term_screen_feed_keyboard(term_screen *screen, uint32_t keysym, uint32_t ascii, uint32_t ucs4, unsigned int mods) {
708deb9
+static char *screen_map_key(term_screen *screen,
708deb9
+                            char *p,
708deb9
+                            const uint32_t *keysyms,
708deb9
+                            size_t n_syms,
708deb9
+                            uint32_t ascii,
708deb9
+                            const uint32_t *ucs4,
708deb9
+                            unsigned int mods) {
708deb9
+        char ch, ch2, ch_mods;
708deb9
+        uint32_t v;
708deb9
+        size_t i;
708deb9
+
708deb9
+        /* TODO: All these key-mappings need to be verified. Public information
708deb9
+         * on those mappings is pretty scarce and every emulator seems to do it
708deb9
+         * slightly differently.
708deb9
+         * A lot of mappings are also missing. */
708deb9
+
708deb9
+        if (n_syms < 1)
708deb9
+                return p;
708deb9
+
708deb9
+        if (n_syms == 1)
708deb9
+                v = keysyms[0];
708deb9
+        else
708deb9
+                v = XKB_KEY_NoSymbol;
708deb9
+
708deb9
+        /* In some mappings, the modifiers are encoded as CSI parameters. The
708deb9
+         * encoding is rather arbitrary, but seems to work. */
708deb9
+        ch_mods = 0;
708deb9
+        switch (mods & (TERM_KBDMOD_SHIFT | TERM_KBDMOD_ALT | TERM_KBDMOD_CTRL)) {
708deb9
+        case TERM_KBDMOD_SHIFT:
708deb9
+                ch_mods = '2';
708deb9
+                break;
708deb9
+        case TERM_KBDMOD_ALT:
708deb9
+                ch_mods = '3';
708deb9
+                break;
708deb9
+        case TERM_KBDMOD_SHIFT | TERM_KBDMOD_ALT:
708deb9
+                ch_mods = '4';
708deb9
+                break;
708deb9
+        case TERM_KBDMOD_CTRL:
708deb9
+                ch_mods = '5';
708deb9
+                break;
708deb9
+        case TERM_KBDMOD_CTRL | TERM_KBDMOD_SHIFT:
708deb9
+                ch_mods = '6';
708deb9
+                break;
708deb9
+        case TERM_KBDMOD_CTRL | TERM_KBDMOD_ALT:
708deb9
+                ch_mods = '7';
708deb9
+                break;
708deb9
+        case TERM_KBDMOD_CTRL | TERM_KBDMOD_SHIFT | TERM_KBDMOD_ALT:
708deb9
+                ch_mods = '8';
708deb9
+                break;
708deb9
+        }
708deb9
+
708deb9
+        /* A user might actually use multiple layouts for keyboard
708deb9
+         * input. @keysyms[0] contains the actual keysym that the user
708deb9
+         * used. But if this keysym is not in the ascii range, the
708deb9
+         * input handler does check all other layouts that the user
708deb9
+         * specified whether one of them maps the key to some ASCII
708deb9
+         * keysym and provides this via @ascii. We always use the real
708deb9
+         * keysym except when handling CTRL+<XY> shortcuts we use the
708deb9
+         * ascii keysym. This is for compatibility to xterm et. al. so
708deb9
+         * ctrl+c always works regardless of the currently active
708deb9
+         * keyboard layout. But if no ascii-sym is found, we still use
708deb9
+         * the real keysym. */
708deb9
+        if (ascii == XKB_KEY_NoSymbol)
708deb9
+                ascii = v;
708deb9
+
708deb9
+        /* map CTRL+<ascii> */
708deb9
+        if (mods & TERM_KBDMOD_CTRL) {
708deb9
+                switch (ascii) {
708deb9
+                case 0x60 ... 0x7e:
708deb9
+                        /* Right hand side is mapped to the left and then
708deb9
+                         * treated equally. Fall through to left-hand side.. */
708deb9
+                        ascii -= 0x20;
708deb9
+                case 0x20 ... 0x5f:
708deb9
+                        /* Printable ASCII is mapped 1-1 in XKB and in
708deb9
+                         * combination with CTRL bit 7 is flipped. This
708deb9
+                         * is equivalent to the caret-notation. */
708deb9
+                        *p++ = ascii ^ 0x40;
708deb9
+                        return p;
708deb9
+                }
708deb9
+        }
708deb9
+
708deb9
+        /* map cursor keys */
708deb9
+        ch = 0;
708deb9
+        switch (v) {
708deb9
+        case XKB_KEY_Up:
708deb9
+                ch = 'A';
708deb9
+                break;
708deb9
+        case XKB_KEY_Down:
708deb9
+                ch = 'B';
708deb9
+                break;
708deb9
+        case XKB_KEY_Right:
708deb9
+                ch = 'C';
708deb9
+                break;
708deb9
+        case XKB_KEY_Left:
708deb9
+                ch = 'D';
708deb9
+                break;
708deb9
+        case XKB_KEY_Home:
708deb9
+                ch = 'H';
708deb9
+                break;
708deb9
+        case XKB_KEY_End:
708deb9
+                ch = 'F';
708deb9
+                break;
708deb9
+        }
708deb9
+        if (ch) {
708deb9
+                *p++ = 0x1b;
708deb9
+                if (screen->flags & TERM_FLAG_CURSOR_KEYS)
708deb9
+                        *p++ = 'O';
708deb9
+                else
708deb9
+                        *p++ = '[';
708deb9
+                if (ch_mods) {
708deb9
+                        *p++ = '1';
708deb9
+                        *p++ = ';';
708deb9
+                        *p++ = ch_mods;
708deb9
+                }
708deb9
+                *p++ = ch;
708deb9
+                return p;
708deb9
+        }
708deb9
+
708deb9
+        /* map action keys */
708deb9
+        ch = 0;
708deb9
+        switch (v) {
708deb9
+        case XKB_KEY_Find:
708deb9
+                ch = '1';
708deb9
+                break;
708deb9
+        case XKB_KEY_Insert:
708deb9
+                ch = '2';
708deb9
+                break;
708deb9
+        case XKB_KEY_Delete:
708deb9
+                ch = '3';
708deb9
+                break;
708deb9
+        case XKB_KEY_Select:
708deb9
+                ch = '4';
708deb9
+                break;
708deb9
+        case XKB_KEY_Page_Up:
708deb9
+                ch = '5';
708deb9
+                break;
708deb9
+        case XKB_KEY_Page_Down:
708deb9
+                ch = '6';
708deb9
+                break;
708deb9
+        }
708deb9
+        if (ch) {
708deb9
+                *p++ = 0x1b;
708deb9
+                *p++ = '[';
708deb9
+                *p++ = ch;
708deb9
+                if (ch_mods) {
708deb9
+                        *p++ = ';';
708deb9
+                        *p++ = ch_mods;
708deb9
+                }
708deb9
+                *p++ = '~';
708deb9
+                return p;
708deb9
+        }
708deb9
+
708deb9
+        /* map lower function keys */
708deb9
+        ch = 0;
708deb9
+        switch (v) {
708deb9
+        case XKB_KEY_F1:
708deb9
+                ch = 'P';
708deb9
+                break;
708deb9
+        case XKB_KEY_F2:
708deb9
+                ch = 'Q';
708deb9
+                break;
708deb9
+        case XKB_KEY_F3:
708deb9
+                ch = 'R';
708deb9
+                break;
708deb9
+        case XKB_KEY_F4:
708deb9
+                ch = 'S';
708deb9
+                break;
708deb9
+        }
708deb9
+        if (ch) {
708deb9
+                if (ch_mods) {
708deb9
+                        *p++ = 0x1b;
708deb9
+                        *p++ = '[';
708deb9
+                        *p++ = '1';
708deb9
+                        *p++ = ';';
708deb9
+                        *p++ = ch_mods;
708deb9
+                        *p++ = ch;
708deb9
+                } else {
708deb9
+                        *p++ = 0x1b;
708deb9
+                        *p++ = 'O';
708deb9
+                        *p++ = ch;
708deb9
+                }
708deb9
+
708deb9
+                return p;
708deb9
+        }
708deb9
+
708deb9
+        /* map upper function keys */
708deb9
+        ch = 0;
708deb9
+        ch2 = 0;
708deb9
+        switch (v) {
708deb9
+        case XKB_KEY_F5:
708deb9
+                ch = '1';
708deb9
+                ch2 = '5';
708deb9
+                break;
708deb9
+        case XKB_KEY_F6:
708deb9
+                ch = '1';
708deb9
+                ch2 = '7';
708deb9
+                break;
708deb9
+        case XKB_KEY_F7:
708deb9
+                ch = '1';
708deb9
+                ch2 = '8';
708deb9
+                break;
708deb9
+        case XKB_KEY_F8:
708deb9
+                ch = '1';
708deb9
+                ch2 = '9';
708deb9
+                break;
708deb9
+        case XKB_KEY_F9:
708deb9
+                ch = '2';
708deb9
+                ch2 = '0';
708deb9
+                break;
708deb9
+        case XKB_KEY_F10:
708deb9
+                ch = '2';
708deb9
+                ch2 = '1';
708deb9
+                break;
708deb9
+        case XKB_KEY_F11:
708deb9
+                ch = '2';
708deb9
+                ch2 = '2';
708deb9
+                break;
708deb9
+        case XKB_KEY_F12:
708deb9
+                ch = '2';
708deb9
+                ch2 = '3';
708deb9
+                break;
708deb9
+        }
708deb9
+        if (ch) {
708deb9
+                *p++ = 0x1b;
708deb9
+                *p++ = '[';
708deb9
+                *p++ = ch;
708deb9
+                if (ch2)
708deb9
+                        *p++ = ch2;
708deb9
+                if (ch_mods) {
708deb9
+                        *p++ = ';';
708deb9
+                        *p++ = ch_mods;
708deb9
+                }
708deb9
+                *p++ = '~';
708deb9
+                return p;
708deb9
+        }
708deb9
+
708deb9
+        /* map special keys */
708deb9
+        switch (v) {
708deb9
+        case 0xff08: /* XKB_KEY_BackSpace */
708deb9
+        case 0xff09: /* XKB_KEY_Tab */
708deb9
+        case 0xff0a: /* XKB_KEY_Linefeed */
708deb9
+        case 0xff0b: /* XKB_KEY_Clear */
708deb9
+        case 0xff15: /* XKB_KEY_Sys_Req */
708deb9
+        case 0xff1b: /* XKB_KEY_Escape */
708deb9
+        case 0xffff: /* XKB_KEY_Delete */
708deb9
+                *p++ = v - 0xff00;
708deb9
+                return p;
708deb9
+        case 0xff13: /* XKB_KEY_Pause */
708deb9
+                /* TODO: What should we do with this key?
708deb9
+                 * Sending XOFF is awful as there is no simple
708deb9
+                 * way on modern keyboards to send XON again.
708deb9
+                 * If someone wants this, we can re-eanble
708deb9
+                 * optionally. */
708deb9
+                return p;
708deb9
+        case 0xff14: /* XKB_KEY_Scroll_Lock */
708deb9
+                /* TODO: What should we do on scroll-lock?
708deb9
+                 * Sending 0x14 is what the specs say but it is
708deb9
+                 * not used today the way most users would
708deb9
+                 * expect so we disable it. If someone wants
708deb9
+                 * this, we can re-enable it (optionally). */
708deb9
+                return p;
708deb9
+        case XKB_KEY_Return:
708deb9
+                *p++ = 0x0d;
708deb9
+                if (screen->flags & TERM_FLAG_NEWLINE_MODE)
708deb9
+                        *p++ = 0x0a;
708deb9
+                return p;
708deb9
+        case XKB_KEY_ISO_Left_Tab:
708deb9
+                *p++ = 0x09;
708deb9
+                return p;
708deb9
+        }
708deb9
+
708deb9
+        /* map unicode keys */
708deb9
+        for (i = 0; i < n_syms; ++i)
708deb9
+                p += term_utf8_encode(p, ucs4[i]);
708deb9
+
708deb9
+        return p;
708deb9
+}
708deb9
+
708deb9
+int term_screen_feed_keyboard(term_screen *screen,
708deb9
+                              const uint32_t *keysyms,
708deb9
+                              size_t n_syms,
708deb9
+                              uint32_t ascii,
708deb9
+                              const uint32_t *ucs4,
708deb9
+                              unsigned int mods) {
708deb9
+        _cleanup_free_ char *dyn = NULL;
708deb9
+        static const size_t padding = 1;
708deb9
+        char buf[128], *start, *p = buf;
708deb9
+
708deb9
         assert_return(screen, -EINVAL);
708deb9
 
708deb9
-        /* TODO */
708deb9
+        /* allocate buffer if too small */
708deb9
+        start = buf;
708deb9
+        if (4 * n_syms + padding > sizeof(buf)) {
708deb9
+                dyn = malloc(4 * n_syms + padding);
708deb9
+                if (!dyn)
708deb9
+                        return -ENOMEM;
708deb9
 
708deb9
-        return 0;
708deb9
+                start = dyn;
708deb9
+        }
708deb9
+
708deb9
+        /* reserve prefix space */
708deb9
+        start += padding;
708deb9
+        p = start;
708deb9
+
708deb9
+        p = screen_map_key(screen, p, keysyms, n_syms, ascii, ucs4, mods);
708deb9
+        if (!p || p - start < 1)
708deb9
+                return 0;
708deb9
+
708deb9
+        /* The ALT modifier causes ESC to be prepended to any key-stroke. We
708deb9
+         * already accounted for that buffer space above, so simply prepend it
708deb9
+         * here.
708deb9
+         * TODO: is altSendsEscape a suitable default? What are the semantics
708deb9
+         * exactly? Is it used in C0/C1 conversion? Is it prepended if there
708deb9
+         * already is an escape character? */
708deb9
+        if (mods & TERM_KBDMOD_ALT && *start != 0x1b)
708deb9
+                *--start = 0x1b;
708deb9
+
708deb9
+        /* turn C0 into C1 */
708deb9
+        if (!(screen->flags & TERM_FLAG_7BIT_MODE) && p - start >= 2)
708deb9
+                if (start[0] == 0x1b && start[1] >= 0x40 && start[1] <= 0x5f)
708deb9
+                        *++start ^= 0x40;
708deb9
+
708deb9
+        return screen_write(screen, start, p - start);
708deb9
 }
708deb9
 
708deb9
 int term_screen_resize(term_screen *screen, unsigned int x, unsigned int y) {
708deb9
diff --git a/src/libsystemd-terminal/term.h b/src/libsystemd-terminal/term.h
708deb9
index a3ca252e31..5228ce0601 100644
708deb9
--- a/src/libsystemd-terminal/term.h
708deb9
+++ b/src/libsystemd-terminal/term.h
708deb9
@@ -128,6 +128,21 @@ DEFINE_TRIVIAL_CLEANUP_FUNC(term_parser*, term_parser_free);
708deb9
  * Screens
708deb9
  */
708deb9
 
708deb9
+enum {
708deb9
+        TERM_KBDMOD_IDX_SHIFT,
708deb9
+        TERM_KBDMOD_IDX_CTRL,
708deb9
+        TERM_KBDMOD_IDX_ALT,
708deb9
+        TERM_KBDMOD_IDX_LINUX,
708deb9
+        TERM_KBDMOD_IDX_CAPS,
708deb9
+        TERM_KBDMOD_CNT,
708deb9
+
708deb9
+        TERM_KBDMOD_SHIFT               = 1 << TERM_KBDMOD_IDX_SHIFT,
708deb9
+        TERM_KBDMOD_CTRL                = 1 << TERM_KBDMOD_IDX_CTRL,
708deb9
+        TERM_KBDMOD_ALT                 = 1 << TERM_KBDMOD_IDX_ALT,
708deb9
+        TERM_KBDMOD_LINUX               = 1 << TERM_KBDMOD_IDX_LINUX,
708deb9
+        TERM_KBDMOD_CAPS                = 1 << TERM_KBDMOD_IDX_CAPS,
708deb9
+};
708deb9
+
708deb9
 typedef int (*term_screen_write_fn) (term_screen *screen, void *userdata, const void *buf, size_t size);
708deb9
 typedef int (*term_screen_cmd_fn) (term_screen *screen, void *userdata, unsigned int cmd, const term_seq *seq);
708deb9
 
708deb9
@@ -141,7 +156,12 @@ unsigned int term_screen_get_width(term_screen *screen);
708deb9
 unsigned int term_screen_get_height(term_screen *screen);
708deb9
 
708deb9
 int term_screen_feed_text(term_screen *screen, const uint8_t *in, size_t size);
708deb9
-int term_screen_feed_keyboard(term_screen *screen, uint32_t keysym, uint32_t ascii, uint32_t ucs4, unsigned int mods);
708deb9
+int term_screen_feed_keyboard(term_screen *screen,
708deb9
+                              const uint32_t *keysyms,
708deb9
+                              size_t n_syms,
708deb9
+                              uint32_t ascii,
708deb9
+                              const uint32_t *ucs4,
708deb9
+                              unsigned int mods);
708deb9
 int term_screen_resize(term_screen *screen, unsigned int width, unsigned int height);
708deb9
 void term_screen_soft_reset(term_screen *screen);
708deb9
 void term_screen_hard_reset(term_screen *screen);