summaryrefslogtreecommitdiff
path: root/kvmd/plugins/hid
diff options
context:
space:
mode:
Diffstat (limited to 'kvmd/plugins/hid')
-rw-r--r--kvmd/plugins/hid/otg/events.py52
-rw-r--r--kvmd/plugins/hid/otg/keyboard.py27
-rw-r--r--kvmd/plugins/hid/otg/mouse.py32
3 files changed, 78 insertions, 33 deletions
diff --git a/kvmd/plugins/hid/otg/events.py b/kvmd/plugins/hid/otg/events.py
index 351a99fd..6ae05f90 100644
--- a/kvmd/plugins/hid/otg/events.py
+++ b/kvmd/plugins/hid/otg/events.py
@@ -20,8 +20,12 @@
# ========================================================================== #
+import struct
import dataclasses
+from typing import List
+from typing import Set
+from typing import Optional
from typing import Union
from ....keyboard.mappings import OtgKey
@@ -67,6 +71,36 @@ def make_keyboard_event(key: str, state: bool) -> Union[KeyEvent, ModifierEvent]
return KeyEvent(otg_key, state)
+def get_led_caps(flags: int) -> bool:
+ # https://wiki.osdev.org/USB_Human_Interface_Devices#LED_lamps
+ return bool(flags & 2)
+
+
+def get_led_scroll(flags: int) -> bool:
+ return bool(flags & 4)
+
+
+def get_led_num(flags: int) -> bool:
+ return bool(flags & 1)
+
+
+def make_keyboard_report(
+ pressed_modifiers: Set[OtgKey],
+ pressed_keys: List[Optional[OtgKey]],
+) -> bytes:
+
+ modifiers = 0
+ for modifier in pressed_modifiers:
+ modifiers |= modifier.code
+
+ assert len(pressed_keys) == 6
+ keys = [
+ (0 if key is None else key.code)
+ for key in pressed_keys
+ ]
+ return bytes([modifiers, 0] + keys)
+
+
# =====
@dataclasses.dataclass(frozen=True)
class MouseButtonEvent(BaseEvent):
@@ -116,3 +150,21 @@ class MouseWheelEvent(BaseEvent):
def __post_init__(self) -> None:
assert -127 <= self.delta_x <= 127
assert -127 <= self.delta_y <= 127
+
+
+def make_mouse_report(
+ absolute: bool,
+ buttons: int,
+ move_x: int,
+ move_y: int,
+ wheel_x: Optional[int],
+ wheel_y: int,
+) -> bytes:
+
+ # XXX: Wheel Y before X: it's ok.
+ # See /kvmd/apps/otg/hid/mouse.py for details
+
+ if wheel_x is not None:
+ return struct.pack(("<BHHbb" if absolute else "<Bbbbb"), buttons, move_x, move_y, wheel_y, wheel_x)
+ else:
+ return struct.pack(("<BHHb" if absolute else "<Bbbb"), buttons, move_x, move_y, wheel_y)
diff --git a/kvmd/plugins/hid/otg/keyboard.py b/kvmd/plugins/hid/otg/keyboard.py
index 384f8fd2..2629aaeb 100644
--- a/kvmd/plugins/hid/otg/keyboard.py
+++ b/kvmd/plugins/hid/otg/keyboard.py
@@ -39,6 +39,10 @@ from .events import ResetEvent
from .events import KeyEvent
from .events import ModifierEvent
from .events import make_keyboard_event
+from .events import get_led_caps
+from .events import get_led_scroll
+from .events import get_led_num
+from .events import make_keyboard_report
# =====
@@ -74,12 +78,11 @@ class KeyboardProcess(BaseDeviceProcess):
# =====
def _process_read_report(self, report: bytes) -> None:
- # https://wiki.osdev.org/USB_Human_Interface_Devices#LED_lamps
assert len(report) == 1, report
self._update_state(
- caps=bool(report[0] & 2),
- scroll=bool(report[0] & 4),
- num=bool(report[0] & 1),
+ caps=get_led_caps(report[0]),
+ scroll=get_led_scroll(report[0]),
+ num=get_led_num(report[0]),
)
# =====
@@ -132,7 +135,8 @@ class KeyboardProcess(BaseDeviceProcess):
# =====
def __send_current_state(self, reopen: bool=False) -> bool:
- if not self._ensure_write(self.__make_report(), reopen=reopen):
+ report = make_keyboard_report(self.__pressed_modifiers, self.__pressed_keys)
+ if not self._ensure_write(report, reopen=reopen):
self.__clear_modifiers()
self.__clear_keys()
return False
@@ -143,16 +147,3 @@ class KeyboardProcess(BaseDeviceProcess):
def __clear_keys(self) -> None:
self.__pressed_keys = [None] * 6
-
- def __make_report(self) -> bytes:
- modifiers = 0
- for modifier in self.__pressed_modifiers:
- modifiers |= modifier.code
-
- assert len(self.__pressed_keys) == 6
- keys = [
- (0 if key is None else key.code)
- for key in self.__pressed_keys
- ]
-
- return bytes([modifiers, 0] + keys)
diff --git a/kvmd/plugins/hid/otg/mouse.py b/kvmd/plugins/hid/otg/mouse.py
index 723fc5f8..df06d739 100644
--- a/kvmd/plugins/hid/otg/mouse.py
+++ b/kvmd/plugins/hid/otg/mouse.py
@@ -20,8 +20,6 @@
# ========================================================================== #
-import struct
-
from typing import Optional
from typing import Any
@@ -36,6 +34,7 @@ from .events import MouseButtonEvent
from .events import MouseMoveEvent
from .events import MouseRelativeEvent
from .events import MouseWheelEvent
+from .events import make_mouse_report
# =====
@@ -58,10 +57,14 @@ class MouseProcess(BaseDeviceProcess):
def cleanup(self) -> None:
self._stop()
get_logger().info("Clearing HID-mouse events ...")
- if self.__absolute:
- report = self.__make_report(0, self.__x, self.__y, 0, 0)
- else:
- report = self.__make_report(0, 0, 0, 0, 0)
+ report = make_mouse_report(
+ absolute=self.__absolute,
+ buttons=0,
+ move_x=(self.__x if self.__absolute else 0),
+ move_y=(self.__y if self.__absolute else 0),
+ wheel_x=(0 if self.__horizontal_wheel else None),
+ wheel_y=0,
+ )
self._ensure_write(report, close=True) # Release all buttons
def send_clear_event(self) -> None:
@@ -157,7 +160,14 @@ class MouseProcess(BaseDeviceProcess):
else:
wheel_x = wheel_y = 0
- report = self.__make_report(self.__pressed_buttons, move_x, move_y, wheel_x, wheel_y)
+ report = make_mouse_report(
+ absolute=self.__absolute,
+ buttons=self.__pressed_buttons,
+ move_x=move_x,
+ move_y=move_y,
+ wheel_x=(wheel_x if self.__horizontal_wheel else None),
+ wheel_y=wheel_y,
+ )
if not self._ensure_write(report, reopen=reopen):
self.__clear_state()
return False
@@ -167,11 +177,3 @@ class MouseProcess(BaseDeviceProcess):
self.__pressed_buttons = 0
self.__x = 0
self.__y = 0
-
- def __make_report(self, buttons: int, move_x: int, move_y: int, wheel_x: int, wheel_y: int) -> bytes:
- # XXX: Wheel Y before X: it's ok.
- # See /kvmd/apps/otg/hid/mouse.py for details
- if self.__horizontal_wheel:
- return struct.pack(("<BHHbb" if self.__absolute else "<Bbbbb"), buttons, move_x, move_y, wheel_y, wheel_x)
- else:
- return struct.pack(("<BHHb" if self.__absolute else "<Bbbb"), buttons, move_x, move_y, wheel_y)