diff options
author | Devaev Maxim <[email protected]> | 2020-11-20 05:05:20 +0300 |
---|---|---|
committer | Devaev Maxim <[email protected]> | 2020-11-20 05:05:20 +0300 |
commit | 649a57e842928f5e1da100ea62545445ee51eb88 (patch) | |
tree | 7c07745db633aa41fcb25773b476ed38ce84522f /kvmd/plugins/hid/_mcu | |
parent | 7f43440cae47b69d27358ad168f53b647bd23b21 (diff) |
next
Diffstat (limited to 'kvmd/plugins/hid/_mcu')
-rw-r--r-- | kvmd/plugins/hid/_mcu/__init__.py | 36 | ||||
-rw-r--r-- | kvmd/plugins/hid/_mcu/proto.py | 38 |
2 files changed, 62 insertions, 12 deletions
diff --git a/kvmd/plugins/hid/_mcu/__init__.py b/kvmd/plugins/hid/_mcu/__init__.py index 62c5870a..39f9f03d 100644 --- a/kvmd/plugins/hid/_mcu/__init__.py +++ b/kvmd/plugins/hid/_mcu/__init__.py @@ -55,7 +55,11 @@ from .gpio import Gpio from .proto import REQUEST_PING from .proto import REQUEST_REPEAT from .proto import RESPONSE_LEGACY_OK +from .proto import KEYBOARD_CODES_TO_NAMES +from .proto import MOUSE_CODES_TO_NAMES from .proto import BaseEvent +from .proto import SetKeyboardOutputEvent +from .proto import SetMouseOutputEvent from .proto import ClearEvent from .proto import KeyEvent from .proto import MouseButtonEvent @@ -164,24 +168,24 @@ class BaseMcuHid(BaseHid, multiprocessing.Process): # pylint: disable=too-many- keyboard_outputs: Dict = {"available": {}, "active": ""} mouse_outputs: Dict = {"available": {}, "active": ""} + if outputs & 0b10000000: # Dynamic if features & 0b00000001: # USB keyboard_outputs["available"]["usb"] = {"name": "USB"} - mouse_outputs["available"]["usb_abs"] = {"name": "USB", "absolute": True} + mouse_outputs["available"]["usb"] = {"name": "USB", "absolute": True} mouse_outputs["available"]["usb_rel"] = {"name": "USB Relative", "absolute": False} + if features & 0b00000010: # PS/2 keyboard_outputs["available"]["ps2"] = {"name": "PS/2"} mouse_outputs["available"]["ps2"] = {"name": "PS/2"} - keyboard_outputs["active"] = { - 0b00000001: "usb", - 0b00000011: "ps2", - }.get(outputs & 0b00000111, "") - mouse_outputs["active"] = { - 0b00001000: "usb_abs", - 0b00010000: "usb_rel", - 0b00011000: "ps2", - }.get(outputs & 0b00111000, "") + active = KEYBOARD_CODES_TO_NAMES.get(outputs & 0b00000111, "") + if active in keyboard_outputs["available"]: + keyboard_outputs["active"] = active + + active = MOUSE_CODES_TO_NAMES.get(outputs & 0b00111000, "") + if active in mouse_outputs["available"]: + mouse_outputs["active"] = active return { "online": online, @@ -251,11 +255,19 @@ class BaseMcuHid(BaseHid, multiprocessing.Process): # pylint: disable=too-many- def send_mouse_wheel_event(self, delta_x: int, delta_y: int) -> None: self.__queue_event(MouseWheelEvent(delta_x, delta_y)) - def clear_events(self) -> None: + def set_keyboard_output(self, output: str) -> None: # FIXME: Если очистка производится со стороны процесса хида, то возможна гонка между - # очисткой и добавлением события _ClearEvent. Неприятно, но не смертельно. + # очисткой и добавлением нового события. Неприятно, но не смертельно. # Починить блокировкой после перехода на асинхронные очереди. tools.clear_queue(self.__events_queue) + self.__queue_event(SetKeyboardOutputEvent(output)) + + def set_mouse_output(self, output: str) -> None: + tools.clear_queue(self.__events_queue) + self.__queue_event(SetMouseOutputEvent(output)) + + def clear_events(self) -> None: + tools.clear_queue(self.__events_queue) self.__queue_event(ClearEvent()) def __queue_event(self, event: BaseEvent) -> None: diff --git a/kvmd/plugins/hid/_mcu/proto.py b/kvmd/plugins/hid/_mcu/proto.py index fa76932a..6c8b0735 100644 --- a/kvmd/plugins/hid/_mcu/proto.py +++ b/kvmd/plugins/hid/_mcu/proto.py @@ -32,6 +32,44 @@ class BaseEvent: raise NotImplementedError +KEYBOARD_NAMES_TO_CODES = { + "usb": 0b00000001, + "ps2": 0b00000011, +} +KEYBOARD_CODES_TO_NAMES = {value: key for (key, value) in KEYBOARD_NAMES_TO_CODES.items()} + + [email protected](frozen=True) +class SetKeyboardOutputEvent(BaseEvent): + keyboard: str + + def __post_init__(self) -> None: + assert not self.keyboard or self.keyboard in KEYBOARD_NAMES_TO_CODES + + def make_request(self) -> bytes: + code = KEYBOARD_NAMES_TO_CODES.get(self.keyboard, 0) + return _make_request(struct.pack(">BBxxx", 0x03, code)) + + +MOUSE_NAMES_TO_CODES = { + "usb": 0b00001000, + "usb_rel": 0b00010000, + "ps2": 0b00011000, +} +MOUSE_CODES_TO_NAMES = {value: key for (key, value) in MOUSE_NAMES_TO_CODES.items()} + + [email protected](frozen=True) +class SetMouseOutputEvent(BaseEvent): + mouse: str + + def __post_init__(self) -> None: + assert not self.mouse or self.mouse in MOUSE_NAMES_TO_CODES + + def make_request(self) -> bytes: + return _make_request(struct.pack(">BBxxx", 0x04, MOUSE_NAMES_TO_CODES.get(self.mouse, 0))) + + class ClearEvent(BaseEvent): def make_request(self) -> bytes: return _make_request(b"\x10\x00\x00\x00\x00") |