mirror of
https://github.com/mofeng-git/One-KVM.git
synced 2025-12-12 01:00:29 +08:00
168 lines
5.3 KiB
Python
Executable File
168 lines
5.3 KiB
Python
Executable File
#!/usr/bin/env python3
|
|
# ========================================================================== #
|
|
# #
|
|
# KVMD - The main PiKVM daemon. #
|
|
# #
|
|
# Copyright (C) 2018-2022 Maxim Devaev <mdevaev@gmail.com> #
|
|
# #
|
|
# This program is free software: you can redistribute it and/or modify #
|
|
# it under the terms of the GNU General Public License as published by #
|
|
# the Free Software Foundation, either version 3 of the License, or #
|
|
# (at your option) any later version. #
|
|
# #
|
|
# This program 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 General Public License for more details. #
|
|
# #
|
|
# You should have received a copy of the GNU General Public License #
|
|
# along with this program. If not, see <https://www.gnu.org/licenses/>. #
|
|
# #
|
|
# ========================================================================== #
|
|
|
|
|
|
import sys
|
|
import csv
|
|
import textwrap
|
|
import dataclasses
|
|
|
|
import Xlib.keysymdef.latin1
|
|
import Xlib.keysymdef.miscellany
|
|
import Xlib.keysymdef.xf86
|
|
import Xlib.keysymdef.xkb
|
|
|
|
import mako.template
|
|
|
|
|
|
# =====
|
|
@dataclasses.dataclass(frozen=True)
|
|
class _UsbKey:
|
|
code: int
|
|
is_modifier: bool
|
|
|
|
@property
|
|
def arduino_modifier_code(self) -> int:
|
|
# https://github.com/NicoHood/HID/blob/4bf6cd6/src/HID-APIs/DefaultKeyboardAPI.hpp#L31
|
|
assert self.is_modifier
|
|
code = self.code
|
|
offset = 0
|
|
while not (code & 0x1):
|
|
code >>= 1
|
|
offset += 1
|
|
assert offset < 8
|
|
return ((0xE << 4) | offset)
|
|
|
|
|
|
@dataclasses.dataclass(frozen=True)
|
|
class _Ps2Key:
|
|
code: int
|
|
type: str
|
|
|
|
|
|
@dataclasses.dataclass(frozen=True)
|
|
class _X11Key:
|
|
name: str
|
|
code: int
|
|
shift: bool
|
|
|
|
|
|
@dataclasses.dataclass(frozen=True)
|
|
class _KeyMapping:
|
|
web_name: str
|
|
mcu_code: int
|
|
usb_key: _UsbKey
|
|
ps2_key: _Ps2Key
|
|
at1_code: int
|
|
x11_keys: set[_X11Key]
|
|
|
|
|
|
def _resolve_keysym(name: str) -> int:
|
|
code: (int | None) = None
|
|
for module in [
|
|
Xlib.keysymdef.latin1,
|
|
Xlib.keysymdef.miscellany,
|
|
Xlib.keysymdef.xf86,
|
|
Xlib.keysymdef.xkb,
|
|
]:
|
|
code = getattr(module, name, None)
|
|
if code is not None:
|
|
break
|
|
assert code is not None, name
|
|
return code
|
|
|
|
|
|
def _parse_x11_names(names: str) -> set[_X11Key]:
|
|
keys: set[_X11Key] = set()
|
|
for name in filter(None, names.split(",")):
|
|
shift = name.startswith("^")
|
|
name = (name[1:] if shift else name)
|
|
code = _resolve_keysym(name)
|
|
keys.add(_X11Key(name, code, shift))
|
|
return keys
|
|
|
|
|
|
def _parse_usb_key(key: str) -> _UsbKey:
|
|
is_modifier = key.startswith("^")
|
|
code = int((key[1:] if is_modifier else key), 16)
|
|
return _UsbKey(code, is_modifier)
|
|
|
|
|
|
def _parse_ps2_key(key: str) -> _Ps2Key:
|
|
(code_type, raw_code) = key.split(":")
|
|
return _Ps2Key(
|
|
code=int(raw_code, 16),
|
|
type=code_type,
|
|
)
|
|
|
|
|
|
def _read_keymap_csv(path: str) -> list[_KeyMapping]:
|
|
keymap: list[_KeyMapping] = []
|
|
with open(path) as file:
|
|
for row in csv.DictReader(file):
|
|
if len(row) >= 6:
|
|
keymap.append(_KeyMapping(
|
|
web_name=row["web_name"],
|
|
mcu_code=int(row["mcu_code"]),
|
|
usb_key=_parse_usb_key(row["usb_key"]),
|
|
ps2_key=_parse_ps2_key(row["ps2_key"]),
|
|
at1_code=int(row["at1_code"], 16),
|
|
x11_keys=_parse_x11_names(row["x11_names"] or ""),
|
|
))
|
|
return keymap
|
|
|
|
|
|
def _render_keymap(keymap: list[_KeyMapping], template_path: str, out_path: str) -> None:
|
|
with open(template_path) as template_file:
|
|
with open(out_path, "w") as out_file:
|
|
template = textwrap.dedent(template_file.read())
|
|
rendered = mako.template.Template(template).render(keymap=keymap)
|
|
out_file.write(rendered)
|
|
|
|
|
|
# =====
|
|
def main() -> None:
|
|
# https://developer.mozilla.org/en-US/docs/Web/API/KeyboardEvent/code/code_values
|
|
# https://github.com/NicoHood/HID/blob/master/src/KeyboardLayouts/ImprovedKeylayouts.h
|
|
# https://github.com/Harvie/ps2dev/blob/master/src/ps2dev.h
|
|
# https://gist.github.com/MightyPork/6da26e382a7ad91b5496ee55fdc73db2
|
|
# https://github.com/qemu/keycodemapdb/blob/master/data/keymaps.csv
|
|
# Hut1_12v2.pdf
|
|
|
|
# Fields list:
|
|
# - Web
|
|
# - MCU code
|
|
# - USB code (^ for the modifier mask)
|
|
# - PS/2 key
|
|
# - AT set1
|
|
# - X11 keysyms (^ for shift)
|
|
|
|
assert len(sys.argv) == 4, f"{sys.argv[0]} <keymap.csv> <template> <out>"
|
|
|
|
keymap = _read_keymap_csv(sys.argv[1])
|
|
_render_keymap(keymap, sys.argv[2], sys.argv[3])
|
|
|
|
|
|
# =====
|
|
if __name__ == "__main__":
|
|
main()
|