diff options
-rw-r--r-- | hid/Makefile | 4 | ||||
-rwxr-xr-x | hid/genmap.py | 36 | ||||
-rw-r--r-- | hid/keymap.in | 107 | ||||
-rw-r--r-- | hid/platformio.ini | 3 | ||||
-rw-r--r-- | hid/src/inline.h | 1 | ||||
-rw-r--r-- | hid/src/keymap.h | 94 | ||||
-rw-r--r-- | hid/src/main.cpp | 31 | ||||
-rw-r--r-- | kvmd/kvmd/data/keymap.yaml | 84 | ||||
-rw-r--r-- | kvmd/kvmd/hid.py | 61 | ||||
-rwxr-xr-x | kvmd/setup.py | 4 | ||||
-rw-r--r-- | os/platforms/v1/Dockerfile.part | 4 |
11 files changed, 375 insertions, 54 deletions
diff --git a/hid/Makefile b/hid/Makefile index ec8bb999..edbd63f1 100644 --- a/hid/Makefile +++ b/hid/Makefile @@ -7,11 +7,15 @@ build: update: platformio platform update +install: upload upload: platformio run --target upload serial: platformio serialports monitor +regen: + python3 genmap.py + clean: rm -rf .pioenvs .piolibdeps diff --git a/hid/genmap.py b/hid/genmap.py new file mode 100755 index 00000000..fe65d449 --- /dev/null +++ b/hid/genmap.py @@ -0,0 +1,36 @@ +#!/usr/bin/env python3 + + +import operator + +from typing import Tuple +from typing import List + +import yaml + + +# ===== +def main() -> None: + keymap: List[Tuple[int, str, str]] = [] + with open("keymap.in") as keymap_file: + for row in keymap_file: + if not row.startswith("#"): + parts = row.split() + keymap.append((int(parts[0]), parts[1], parts[2])) + + with open("../kvmd/kvmd/data/keymap.yaml", "w") as kvmd_yaml_file: + yaml.dump({ + js_key: code + for (code, _, js_key) in sorted(keymap, key=operator.itemgetter(2)) + }, kvmd_yaml_file, indent=4, default_flow_style=False) + + with open("src/keymap.h", "w") as hid_header_file: + hid_header_file.write("#include <HID-Project.h>\n\n#include \"inline.h\"\n\n\n") + hid_header_file.write("INLINE uint8_t keymap(uint8_t code) {\n\tswitch(code) {\n") + for (code, hid_key, _) in sorted(keymap, key=operator.itemgetter(1)): + hid_header_file.write("\t\tcase %d: return %s;\n" % (code, hid_key)) + hid_header_file.write("\t\tdefault: return 0;\n\t}\n}") + + +if __name__ == "__main__": + main() diff --git a/hid/keymap.in b/hid/keymap.in new file mode 100644 index 00000000..2d356f47 --- /dev/null +++ b/hid/keymap.in @@ -0,0 +1,107 @@ +# https://github.com/NicoHood/HID/blob/master/src/HID-APIs/ImprovedKeylayouts.h +1 KEY_A KeyA +2 KEY_B KeyB +3 KEY_C KeyC +4 KEY_D KeyD +5 KEY_E KeyE +6 KEY_F KeyF +7 KEY_G KeyG +8 KEY_H KeyH +9 KEY_I KeyI +10 KEY_J KeyJ +11 KEY_K KeyK +12 KEY_L KeyL +13 KEY_M KeyM +14 KEY_N KeyN +15 KEY_O KeyO +16 KEY_P KeyP +17 KEY_Q KeyQ +18 KEY_R KeyR +19 KEY_S KeyS +20 KEY_T KeyT +21 KEY_U KeyU +22 KEY_V KeyV +23 KEY_W KeyW +24 KEY_X KeyX +25 KEY_Y KeyY +26 KEY_Z KeyZ +27 KEY_1 Digit1 +28 KEY_2 Digit2 +29 KEY_3 Digit3 +30 KEY_4 Digit4 +31 KEY_5 Digit5 +32 KEY_6 Digit6 +33 KEY_7 Digit7 +34 KEY_8 Digit8 +35 KEY_9 Digit9 +36 KEY_0 Digit0 +37 KEY_ENTER Enter +38 KEY_ESC Escape +39 KEY_BACKSPACE Backspace +40 KEY_TAB Tab +41 KEY_SPACE Space +42 KEY_MINUS Minus +43 KEY_EQUAL Equal +44 KEY_LEFT_BRACE BracketLeft +45 KEY_RIGHT_BRACE BracketRight +46 KEY_BACKSLASH Backslash +47 KEY_SEMICOLON Semicolon +48 KEY_QUOTE Quote +49 KEY_TILDE Backquote +50 KEY_COMMA Comma +51 KEY_PERIOD Period +52 KEY_SLASH Slash +53 KEY_CAPS_LOCK CapsLock +54 KEY_F1 F1 +55 KEY_F2 F2 +56 KEY_F3 F3 +57 KEY_F4 F4 +58 KEY_F5 F5 +59 KEY_F6 F6 +60 KEY_F7 F7 +61 KEY_F8 F8 +62 KEY_F9 F9 +63 KEY_F10 F10 +64 KEY_F11 F11 +65 KEY_F12 F12 +66 KEY_PRINT PrintScreen +# KEY_SCROLL_LOCK +# KEY_PAUSE +67 KEY_INSERT Insert +68 KEY_HOME Home +69 KEY_PAGE_UP PageUp +70 KEY_DELETE Delete +71 KEY_END End +72 KEY_PAGE_DOWN PageDown +73 KEY_RIGHT_ARROW ArrowRight +74 KEY_LEFT_ARROW ArrowLeft +75 KEY_DOWN_ARROW ArrowDown +76 KEY_UP_ARROW ArrowUp +# KEY_NUM_LOCK +# KEYPAD_DIVIDE +# KEYPAD_MULTIPLY +# KEYPAD_SUBTRACT +# KEYPAD_ADD +# KEYPAD_ENTER +# KEYPAD_1 +# KEYPAD_2 +# KEYPAD_3 +# KEYPAD_4 +# KEYPAD_5 +# KEYPAD_6 +# KEYPAD_7 +# KEYPAD_8 +# KEYPAD_9 +# KEYPAD_0 +# KEYPAD_DOT +# KEY_NON_US +# KEY_APPLICATION +# KEY_MENU +77 KEY_LEFT_CTRL ControlLeft +78 KEY_LEFT_SHIFT ShiftLeft +79 KEY_LEFT_ALT AltLeft +80 KEY_LEFT_GUI MetaLeft +81 KEY_RIGHT_CTRL ControlRight +82 KEY_RIGHT_SHIFT ShiftRight +83 KEY_RIGHT_ALT AltRight +84 KEY_RIGHT_GUI MetaRight diff --git a/hid/platformio.ini b/hid/platformio.ini index d56457fc..dbbb9c3e 100644 --- a/hid/platformio.ini +++ b/hid/platformio.ini @@ -14,3 +14,6 @@ board = micro framework = arduino upload_port = /dev/ttyACM0 monitor_baud = 115200 + +lib_deps = + HID-Project diff --git a/hid/src/inline.h b/hid/src/inline.h new file mode 100644 index 00000000..25e50329 --- /dev/null +++ b/hid/src/inline.h @@ -0,0 +1 @@ +#define INLINE inline __attribute__((always_inline)) diff --git a/hid/src/keymap.h b/hid/src/keymap.h new file mode 100644 index 00000000..be393625 --- /dev/null +++ b/hid/src/keymap.h @@ -0,0 +1,94 @@ +#include <HID-Project.h> + +#include "inline.h" + + +INLINE uint8_t keymap(uint8_t code) { + switch(code) { + case 36: return KEY_0; + case 27: return KEY_1; + case 28: return KEY_2; + case 29: return KEY_3; + case 30: return KEY_4; + case 31: return KEY_5; + case 32: return KEY_6; + case 33: return KEY_7; + case 34: return KEY_8; + case 35: return KEY_9; + case 1: return KEY_A; + case 2: return KEY_B; + case 46: return KEY_BACKSLASH; + case 39: return KEY_BACKSPACE; + case 3: return KEY_C; + case 53: return KEY_CAPS_LOCK; + case 50: return KEY_COMMA; + case 4: return KEY_D; + case 70: return KEY_DELETE; + case 75: return KEY_DOWN_ARROW; + case 5: return KEY_E; + case 71: return KEY_END; + case 37: return KEY_ENTER; + case 43: return KEY_EQUAL; + case 38: return KEY_ESC; + case 6: return KEY_F; + case 54: return KEY_F1; + case 63: return KEY_F10; + case 64: return KEY_F11; + case 65: return KEY_F12; + case 55: return KEY_F2; + case 56: return KEY_F3; + case 57: return KEY_F4; + case 58: return KEY_F5; + case 59: return KEY_F6; + case 60: return KEY_F7; + case 61: return KEY_F8; + case 62: return KEY_F9; + case 7: return KEY_G; + case 8: return KEY_H; + case 68: return KEY_HOME; + case 9: return KEY_I; + case 67: return KEY_INSERT; + case 10: return KEY_J; + case 11: return KEY_K; + case 12: return KEY_L; + case 79: return KEY_LEFT_ALT; + case 74: return KEY_LEFT_ARROW; + case 44: return KEY_LEFT_BRACE; + case 77: return KEY_LEFT_CTRL; + case 80: return KEY_LEFT_GUI; + case 78: return KEY_LEFT_SHIFT; + case 13: return KEY_M; + case 42: return KEY_MINUS; + case 14: return KEY_N; + case 15: return KEY_O; + case 16: return KEY_P; + case 72: return KEY_PAGE_DOWN; + case 69: return KEY_PAGE_UP; + case 51: return KEY_PERIOD; + case 66: return KEY_PRINT; + case 17: return KEY_Q; + case 48: return KEY_QUOTE; + case 18: return KEY_R; + case 83: return KEY_RIGHT_ALT; + case 73: return KEY_RIGHT_ARROW; + case 45: return KEY_RIGHT_BRACE; + case 81: return KEY_RIGHT_CTRL; + case 84: return KEY_RIGHT_GUI; + case 82: return KEY_RIGHT_SHIFT; + case 19: return KEY_S; + case 47: return KEY_SEMICOLON; + case 52: return KEY_SLASH; + case 41: return KEY_SPACE; + case 20: return KEY_T; + case 40: return KEY_TAB; + case 49: return KEY_TILDE; + case 21: return KEY_U; + case 76: return KEY_UP_ARROW; + case 22: return KEY_V; + case 23: return KEY_W; + case 24: return KEY_X; + case 25: return KEY_Y; + case 26: return KEY_Z; + default: return 0; + } +}
\ No newline at end of file diff --git a/hid/src/main.cpp b/hid/src/main.cpp index 32c14c14..631925f0 100644 --- a/hid/src/main.cpp +++ b/hid/src/main.cpp @@ -1,35 +1,44 @@ #include <Arduino.h> -#include <Keyboard.h> +#include <HID-Project.h> + +#include "inline.h" +#include "keymap.h" #define CMD_SERIAL Serial1 -#define SERIAL_SPEED 115200 +#define CMD_SERIAL_SPEED 115200 #define INLINE inline __attribute__((always_inline)) INLINE void cmdResetHid() { + CMD_SERIAL.read(); // unused now + CMD_SERIAL.read(); // unused now + CMD_SERIAL.read(); // unused now Keyboard.releaseAll(); } INLINE void cmdKeyEvent() { - uint8_t state = Serial.read(); - uint8_t key = Serial.read(); - if (state) { - Keyboard.press(key); - } else { - Keyboard.release(key); + uint8_t state = CMD_SERIAL.read(); + uint8_t code = keymap(CMD_SERIAL.read()); + CMD_SERIAL.read(); // unused now + if (code) { + if (state) { + Keyboard.press(code); + } else { + Keyboard.release(code); + } } } void setup() { - CMD_SERIAL.begin(SERIAL_SPEED); + CMD_SERIAL.begin(CMD_SERIAL_SPEED); Keyboard.begin(); } void loop() { - while (true) { // fast - switch (Serial.read()) { + if (CMD_SERIAL.available() >= 4) { + switch ((uint8_t)CMD_SERIAL.read()) { case 0: cmdResetHid(); break; case 1: cmdKeyEvent(); break; default: break; diff --git a/kvmd/kvmd/data/keymap.yaml b/kvmd/kvmd/data/keymap.yaml new file mode 100644 index 00000000..1904c639 --- /dev/null +++ b/kvmd/kvmd/data/keymap.yaml @@ -0,0 +1,84 @@ +AltLeft: 79 +AltRight: 83 +ArrowDown: 75 +ArrowLeft: 74 +ArrowRight: 73 +ArrowUp: 76 +Backquote: 49 +Backslash: 46 +Backspace: 39 +BracketLeft: 44 +BracketRight: 45 +CapsLock: 53 +Comma: 50 +ControlLeft: 77 +ControlRight: 81 +Delete: 70 +Digit0: 36 +Digit1: 27 +Digit2: 28 +Digit3: 29 +Digit4: 30 +Digit5: 31 +Digit6: 32 +Digit7: 33 +Digit8: 34 +Digit9: 35 +End: 71 +Enter: 37 +Equal: 43 +Escape: 38 +F1: 54 +F10: 63 +F11: 64 +F12: 65 +F2: 55 +F3: 56 +F4: 57 +F5: 58 +F6: 59 +F7: 60 +F8: 61 +F9: 62 +Home: 68 +Insert: 67 +KeyA: 1 +KeyB: 2 +KeyC: 3 +KeyD: 4 +KeyE: 5 +KeyF: 6 +KeyG: 7 +KeyH: 8 +KeyI: 9 +KeyJ: 10 +KeyK: 11 +KeyL: 12 +KeyM: 13 +KeyN: 14 +KeyO: 15 +KeyP: 16 +KeyQ: 17 +KeyR: 18 +KeyS: 19 +KeyT: 20 +KeyU: 21 +KeyV: 22 +KeyW: 23 +KeyX: 24 +KeyY: 25 +KeyZ: 26 +MetaLeft: 80 +MetaRight: 84 +Minus: 42 +PageDown: 72 +PageUp: 69 +Period: 51 +PrintScreen: 66 +Quote: 48 +Semicolon: 47 +ShiftLeft: 78 +ShiftRight: 82 +Slash: 52 +Space: 41 +Tab: 40 diff --git a/kvmd/kvmd/hid.py b/kvmd/kvmd/hid.py index 9ad5ba62..d600d438 100644 --- a/kvmd/kvmd/hid.py +++ b/kvmd/kvmd/hid.py @@ -1,13 +1,14 @@ -import re import asyncio import multiprocessing import multiprocessing.queues import queue +import pkgutil +from typing import Dict from typing import Set from typing import NamedTuple -from typing import Union +import yaml import serial from .logging import get_logger @@ -16,50 +17,23 @@ from . import gpio # ===== +def _get_keymap() -> Dict[str, int]: + return yaml.load(pkgutil.get_data(__name__, "data/keymap.yaml").decode()) # type: ignore + + +_KEYMAP = _get_keymap() + + +def _keymap(key: str) -> bytes: + code = _KEYMAP.get(key) + return (bytes([code]) if code else b"") # type: ignore + + class _KeyEvent(NamedTuple): key: str state: bool -def _key_to_bytes(key: str) -> bytes: - # https://www.arduino.cc/reference/en/language/functions/usb/keyboard/ - # Also locate Keyboard.h - - match = re.match(r"(Digit|Key)([0-9A-Z])", key) - code: Union[str, int, None] - if match: - code = match.group(2) - else: - code = { # type: ignore - "Escape": 0xB1, "Backspace": 0xB2, - "Tab": 0xB3, "Enter": 0xB0, - "Insert": 0xD1, "Delete": 0xD4, - "Home": 0xD2, "End": 0xD5, - "PageUp": 0xD3, "PageDown": 0xD6, - "ArrowLeft": 0xD8, "ArrowRight": 0xD7, - "ArrowUp": 0xDA, "ArrowDown": 0xD9, - - "CapsLock": 0xC1, - "ShiftLeft": 0x81, "ShiftRight": 0x85, - "ControlLeft": 0x80, "ControlRight": 0x84, - "AltLeft": 0x82, "AltRight": 0x86, - "MetaLeft": 0x83, "MetaRight": 0x87, - - "Backquote": "`", "Minus": "-", "Equal": "=", "Space": " ", - "BracketLeft": "[", "BracketRight": "]", "Semicolon": ";", "Quote": "'", - "Comma": ",", "Period": ".", "Slash": "/", "Backslash": "\\", - - "F1": 0xC2, "F2": 0xC3, "F3": 0xC4, "F4": 0xC5, - "F5": 0xC6, "F6": 0xC7, "F7": 0xC8, "F8": 0xC9, - "F9": 0xCA, "F10": 0xCB, "F11": 0xCC, "F12": 0xCD, - }.get(key) - if isinstance(code, str): - return bytes(code, encoding="ascii") # type: ignore - elif isinstance(code, int): - return bytes([code]) - return b"" - - class Hid(multiprocessing.Process): def __init__( self, @@ -138,14 +112,15 @@ class Hid(multiprocessing.Process): raise def __send_key_event(self, tty: serial.Serial, event: _KeyEvent) -> None: - key_bytes = _key_to_bytes(event.key) + key_bytes = _keymap(event.key) if key_bytes: assert len(key_bytes) == 1, (event, key_bytes) tty.write( b"\01" + (b"\01" if event.state else b"\00") + key_bytes + + b"\00" ) def __send_clear_hid(self, tty: serial.Serial) -> None: - tty.write(b"\00") + tty.write(b"\00\00\00\00") diff --git a/kvmd/setup.py b/kvmd/setup.py index d5319f24..a2d842b9 100755 --- a/kvmd/setup.py +++ b/kvmd/setup.py @@ -26,6 +26,10 @@ def main() -> None: "kvmd.extras.wscli", ], + package_data={ + "kvmd": ["data/*.yaml"], + }, + entry_points={ "console_scripts": [ "kvmd = kvmd:main", diff --git a/os/platforms/v1/Dockerfile.part b/os/platforms/v1/Dockerfile.part index 300ece23..56909ae7 100644 --- a/os/platforms/v1/Dockerfile.part +++ b/os/platforms/v1/Dockerfile.part @@ -11,6 +11,10 @@ COPY stages/pikvm/config.txt /boot/ COPY stages/pikvm/sysctl.conf /etc/sysctl.d/99-pikvm.conf COPY stages/pikvm/udev.rules /etc/udev/rules.d/pikvm.rules +RUN sed -i -e "s/console=ttyAMA0\,115200//g" /boot/cmdline.txt \ + && sed -i -e "s/kgdboc=ttyAMA0\,115200//g" /boo/cmdline.txt +RUN systemctl mask [email protected] + RUN cp /usr/share/kvmd/configs/nginx/nginx.conf /etc/nginx/ RUN cp /usr/share/kvmd/configs/kvmd/v1.yaml /etc/kvmd.yaml |