diff options
Diffstat (limited to 'kvmd')
-rw-r--r-- | kvmd/apps/kvmd/__init__.py | 11 | ||||
-rw-r--r-- | kvmd/apps/kvmd/hid.py | 262 | ||||
-rw-r--r-- | kvmd/apps/kvmd/server.py | 14 |
3 files changed, 196 insertions, 91 deletions
diff --git a/kvmd/apps/kvmd/__init__.py b/kvmd/apps/kvmd/__init__.py index e8b74abc..e9daea32 100644 --- a/kvmd/apps/kvmd/__init__.py +++ b/kvmd/apps/kvmd/__init__.py @@ -35,9 +35,18 @@ def main() -> None: hid = Hid( reset=int(config["hid"]["pinout"]["reset"]), + + reset_delay=float(config["hid"]["reset_delay"]), + device_path=str(config["hid"]["device"]), speed=int(config["hid"]["speed"]), - reset_delay=float(config["hid"]["reset_delay"]), + read_timeout=float(config["hid"]["read_timeout"]), + read_retries=int(config["hid"]["read_retries"]), + common_retries=int(config["hid"]["common_retries"]), + retries_delay=float(config["hid"]["retries_delay"]), + noop=bool(config["hid"].get("noop", False)), + + state_poll=float(config["hid"]["state_poll"]), ) atx = Atx( diff --git a/kvmd/apps/kvmd/hid.py b/kvmd/apps/kvmd/hid.py index 26cfa898..3f329c64 100644 --- a/kvmd/apps/kvmd/hid.py +++ b/kvmd/apps/kvmd/hid.py @@ -11,6 +11,7 @@ import time from typing import Dict from typing import Set from typing import NamedTuple +from typing import AsyncGenerator import yaml import serial @@ -33,41 +34,92 @@ class _KeyEvent(NamedTuple): key: str state: bool + @staticmethod + def is_valid(key: str) -> bool: + return (key in _KEYMAP) + + def make_command(self) -> bytes: + code = _KEYMAP[self.key] + key_bytes = bytes([code]) + assert len(key_bytes) == 1, (self, key_bytes, code) + state_bytes = (b"\x01" if self.state else b"\x00") + return b"\x11" + key_bytes + state_bytes + b"\x00\x00" + class _MouseMoveEvent(NamedTuple): to_x: int to_y: int + def make_command(self) -> bytes: + to_x = min(max(-32768, self.to_x), 32767) + to_y = min(max(-32768, self.to_y), 32767) + return b"\x12" + struct.pack(">hh", to_x, to_y) + class _MouseButtonEvent(NamedTuple): button: str state: bool + @staticmethod + def is_valid(button: str) -> bool: + return (button in ["left", "right"]) + + def make_command(self) -> bytes: + code = 0 + if self.button == "left": + code = (0b10000000 | (0b00001000 if self.state else 0)) + elif self.button == "right": + code = (0b01000000 | (0b00000100 if self.state else 0)) + assert code, self + return b"\x13" + bytes([code]) + b"\x00\x00\x00" + class _MouseWheelEvent(NamedTuple): delta_y: int + def make_command(self) -> bytes: + delta_y = min(max(-128, self.delta_y), 127) + return b"\x14\x00" + struct.pack(">b", delta_y) + b"\x00\x00" + class Hid(multiprocessing.Process): # pylint: disable=too-many-instance-attributes - def __init__( + def __init__( # pylint: disable=too-many-arguments self, reset: int, + reset_delay: float, + device_path: str, speed: int, - reset_delay: float, + read_timeout: float, + read_retries: int, + common_retries: int, + retries_delay: float, + noop: bool, + + state_poll: float, ) -> None: super().__init__(daemon=True) self.__reset = gpio.set_output(reset) + self.__reset_delay = reset_delay + self.__device_path = device_path self.__speed = speed - self.__reset_delay = reset_delay + self.__read_timeout = read_timeout + self.__read_retries = read_retries + self.__common_retries = common_retries + self.__retries_delay = retries_delay + self.__noop = noop + + self.__state_poll = state_poll self.__pressed_keys: Set[str] = set() self.__pressed_mouse_buttons: Set[str] = set() self.__lock = asyncio.Lock() - self.__queue: multiprocessing.queues.Queue = multiprocessing.Queue() + self.__events_queue: multiprocessing.queues.Queue = multiprocessing.Queue() + + self.__ok_shared = multiprocessing.Value("i", 1) self.__stop_event = multiprocessing.Event() @@ -75,6 +127,14 @@ class Hid(multiprocessing.Process): # pylint: disable=too-many-instance-attribu get_logger().info("Starting HID daemon ...") super().start() + def get_state(self) -> Dict: + return {"ok": bool(self.__ok_shared.value)} + + async def poll_state(self) -> AsyncGenerator[Dict, None]: + while self.is_alive(): + yield self.get_state() + await asyncio.sleep(self.__state_poll) + async def reset(self) -> None: async with self.__lock: gpio.write(self.__reset, True) @@ -84,32 +144,34 @@ class Hid(multiprocessing.Process): # pylint: disable=too-many-instance-attribu async def send_key_event(self, key: str, state: bool) -> None: if not self.__stop_event.is_set(): async with self.__lock: - if state and key not in self.__pressed_keys: - self.__pressed_keys.add(key) - self.__queue.put(_KeyEvent(key, state)) - elif not state and key in self.__pressed_keys: - self.__pressed_keys.remove(key) - self.__queue.put(_KeyEvent(key, state)) + if _KeyEvent.is_valid(key): + if state and key not in self.__pressed_keys: + self.__pressed_keys.add(key) + self.__events_queue.put(_KeyEvent(key, state)) + elif not state and key in self.__pressed_keys: + self.__pressed_keys.remove(key) + self.__events_queue.put(_KeyEvent(key, state)) async def send_mouse_move_event(self, to_x: int, to_y: int) -> None: if not self.__stop_event.is_set(): async with self.__lock: - self.__queue.put(_MouseMoveEvent(to_x, to_y)) + self.__events_queue.put(_MouseMoveEvent(to_x, to_y)) async def send_mouse_button_event(self, button: str, state: bool) -> None: if not self.__stop_event.is_set(): async with self.__lock: - if state and button not in self.__pressed_mouse_buttons: - self.__pressed_mouse_buttons.add(button) - self.__queue.put(_MouseButtonEvent(button, state)) - elif not state and button in self.__pressed_mouse_buttons: - self.__pressed_mouse_buttons.remove(button) - self.__queue.put(_MouseButtonEvent(button, state)) + if _MouseButtonEvent.is_valid(button): + if state and button not in self.__pressed_mouse_buttons: + self.__pressed_mouse_buttons.add(button) + self.__events_queue.put(_MouseButtonEvent(button, state)) + elif not state and button in self.__pressed_mouse_buttons: + self.__pressed_mouse_buttons.remove(button) + self.__events_queue.put(_MouseButtonEvent(button, state)) async def send_mouse_wheel_event(self, delta_y: int) -> None: if not self.__stop_event.is_set(): async with self.__lock: - self.__queue.put(_MouseWheelEvent(delta_y)) + self.__events_queue.put(_MouseWheelEvent(delta_y)) async def clear_events(self) -> None: if not self.__stop_event.is_set(): @@ -120,7 +182,7 @@ class Hid(multiprocessing.Process): # pylint: disable=too-many-instance-attribu async with self.__lock: if self.is_alive(): self.__unsafe_clear_events() - get_logger().info("Stopping keyboard daemon ...") + get_logger().info("Stopping HID daemon ...") self.__stop_event.set() self.join() else: @@ -130,17 +192,17 @@ class Hid(multiprocessing.Process): # pylint: disable=too-many-instance-attribu def __unsafe_clear_events(self) -> None: for button in self.__pressed_mouse_buttons: - self.__queue.put(_MouseButtonEvent(button, False)) + self.__events_queue.put(_MouseButtonEvent(button, False)) self.__pressed_mouse_buttons.clear() for key in self.__pressed_keys: - self.__queue.put(_KeyEvent(key, False)) + self.__events_queue.put(_KeyEvent(key, False)) self.__pressed_keys.clear() def __emergency_clear_events(self) -> None: if os.path.exists(self.__device_path): try: - with serial.Serial(self.__device_path, self.__speed) as tty: - self.__send_clear_hid(tty) + with self.__get_serial() as tty: + self.__process_request(tty, b"\x10\x00\x00\x00\x00") except Exception: get_logger().exception("Can't execute emergency clear HID events") @@ -148,70 +210,102 @@ class Hid(multiprocessing.Process): # pylint: disable=too-many-instance-attribu signal.signal(signal.SIGINT, signal.SIG_IGN) setproctitle.setproctitle("[hid] " + setproctitle.getproctitle()) try: - with serial.Serial(self.__device_path, self.__speed) as tty: - hid_ready = False - while True: - if hid_ready: - try: - event = self.__queue.get(timeout=0.05) - except queue.Empty: - pass + with self.__get_serial() as tty: + passed = 0 + while not (self.__stop_event.is_set() and self.__events_queue.qsize() == 0): + try: + event = self.__events_queue.get(timeout=0.05) + except queue.Empty: + if passed >= 20: # 20 * 0.05 = 1 sec + self.__process_request(tty, b"\x01\x00\x00\x00\x00") # Ping + passed = 0 else: - if isinstance(event, _KeyEvent): - self.__send_key_event(tty, event) - elif isinstance(event, _MouseMoveEvent): - self.__send_mouse_move_event(tty, event) - elif isinstance(event, _MouseButtonEvent): - self.__send_mouse_button_event(tty, event) - elif isinstance(event, _MouseWheelEvent): - self.__send_mouse_wheel_event(tty, event) - else: - raise RuntimeError("Unknown HID event") - hid_ready = False - - if tty.in_waiting: - while tty.in_waiting: - tty.read(tty.in_waiting) - hid_ready = True + passed += 1 else: - time.sleep(0.05) - - if self.__stop_event.is_set() and self.__queue.qsize() == 0: - break + self.__process_request(tty, event.make_command()) + passed = 0 except Exception: get_logger().exception("Unhandled exception") raise - def __send_key_event(self, tty: serial.Serial, event: _KeyEvent) -> None: - code = _KEYMAP.get(event.key) - if code: - key_bytes = bytes([code]) - assert len(key_bytes) == 1, (event, key_bytes) - tty.write( - b"\01" - + key_bytes - + (b"\01" if event.state else b"\00") - + b"\00\00" - ) - - def __send_mouse_move_event(self, tty: serial.Serial, event: _MouseMoveEvent) -> None: - to_x = min(max(-32768, event.to_x), 32767) - to_y = min(max(-32768, event.to_y), 32767) - tty.write(b"\02" + struct.pack(">hh", to_x, to_y)) - - def __send_mouse_button_event(self, tty: serial.Serial, event: _MouseButtonEvent) -> None: - if event.button == "left": - code = (0b10000000 | (0b00001000 if event.state else 0)) - elif event.button == "right": - code = (0b01000000 | (0b00000100 if event.state else 0)) - else: - code = 0 - if code: - tty.write(b"\03" + bytes([code]) + b"\00\00\00") - - def __send_mouse_wheel_event(self, tty: serial.Serial, event: _MouseWheelEvent) -> None: - delta_y = min(max(-128, event.delta_y), 127) - tty.write(b"\04\00" + struct.pack(">b", delta_y) + b"\00\00") - - def __send_clear_hid(self, tty: serial.Serial) -> None: - tty.write(b"\00\00\00\00\00") + def __get_serial(self) -> serial.Serial: + return serial.Serial(self.__device_path, self.__speed, timeout=self.__read_timeout) + + def __process(self, tty: serial.Serial, command: bytes) -> None: + self.__process_request(tty, self.__make_request(command)) + + def __process_request(self, tty: serial.Serial, request: bytes) -> None: # pylint: disable=too-many-branches + logger = get_logger() + + common_retries = self.__common_retries + read_retries = self.__read_retries + error_occured = False + + while common_retries and read_retries: + if not self.__noop: + if tty.in_waiting: + tty.read(tty.in_waiting) + + assert tty.write(request) == len(request) + response = tty.read(4) + else: + response = b"\x33\x20" # Magic + OK + response += struct.pack(">H", self.__make_crc16(response)) + + if len(response) < 4: + logger.error("No response from HID: request=%r", request) + read_retries -= 1 + else: + assert len(response) == 4, response + if self.__make_crc16(response[-4:-2]) != struct.unpack(">H", response[-2:])[0]: + get_logger().error("Invalid response CRC; requesting response again ...") + request = self.__make_request(b"\x02\x00\x00\x00\x00") # Repeat an answer + else: + code = response[1] + if code == 0x48: # Request timeout + logger.error("Got request timeout from HID: request=%r", request) + elif code == 0x40: # CRC Error + logger.error("Got CRC error of request from HID: request=%r", request) + elif code == 0x45: # Unknown command + logger.error("HID did not recognize the request=%r", request) + self.__ok_shared.value = 1 + return + elif code == 0x24: # Rebooted? + logger.error("No previous command state inside HID, seems it was rebooted") + self.__ok_shared.value = 1 + return + elif code == 0x20: # Done + if error_occured: + logger.info("Success!") + self.__ok_shared.value = 1 + return + else: + logger.error("Invalid response from HID: request=%r; code=0x%x", request, code) + + common_retries -= 1 + error_occured = True + self.__ok_shared.value = 0 + + if common_retries and read_retries: + logger.error("Retries left: common_retries=%d; read_retries=%d", common_retries, read_retries) + time.sleep(self.__retries_delay) + + logger.error("Can't process HID request due many errors: %r", request) + + def __make_request(self, command: bytes) -> bytes: + request = b"\x33" + command + request += struct.pack(">H", self.__make_crc16(request)) + assert len(request) == 8, (request, command) + return request + + def __make_crc16(self, data: bytes) -> int: + crc = 0xFFFF + for byte in data: + crc = crc ^ byte + for _ in range(8): + if crc & 0x0001 == 0: + crc = crc >> 1 + else: + crc = crc >> 1 + crc = crc ^ 0xA001 + return crc diff --git a/kvmd/apps/kvmd/server.py b/kvmd/apps/kvmd/server.py index f3e7ddc8..37e6ca5b 100644 --- a/kvmd/apps/kvmd/server.py +++ b/kvmd/apps/kvmd/server.py @@ -144,6 +144,7 @@ def _system_task(method: Callable) -> Callable: async def wrap(self: "Server") -> None: try: await method(self) + raise RuntimeError("Dead system task: %s" % (method)) except asyncio.CancelledError: pass except Exception: @@ -201,6 +202,7 @@ def _valid_int(name: str, value: Optional[str], min_value: Optional[int]=None, m class _Events(Enum): INFO_STATE = "info_state" + HID_STATE = "hid_state" ATX_STATE = "atx_state" MSD_STATE = "msd_state" STREAMER_STATE = "streamer_state" @@ -357,6 +359,7 @@ class Server: # pylint: disable=too-many-instance-attributes await self.__register_socket(ws) await asyncio.gather(*[ self.__broadcast_event(_Events.INFO_STATE, (await self.__make_info())), + self.__broadcast_event(_Events.HID_STATE, self.__hid.get_state()), self.__broadcast_event(_Events.ATX_STATE, self.__atx.get_state()), self.__broadcast_event(_Events.MSD_STATE, self.__msd.get_state()), self.__broadcast_event(_Events.STREAMER_STATE, (await self.__streamer.get_state())), @@ -568,12 +571,6 @@ class Server: # pylint: disable=too-many-instance-attributes # ===== SYSTEM TASKS @_system_task - async def __hid_watchdog(self) -> None: - while self.__hid.is_alive(): - await asyncio.sleep(0.1) - raise RuntimeError("HID is dead") - - @_system_task async def __stream_controller(self) -> None: prev = 0 shutdown_at = 0.0 @@ -607,6 +604,11 @@ class Server: # pylint: disable=too-many-instance-attributes await asyncio.sleep(0.1) @_system_task + async def __poll_hid_state(self) -> None: + async for state in self.__hid.poll_state(): + await self.__broadcast_event(_Events.HID_STATE, state) + + @_system_task async def __poll_atx_state(self) -> None: async for state in self.__atx.poll_state(): await self.__broadcast_event(_Events.ATX_STATE, state) |