mirror of
https://github.com/mofeng-git/One-KVM.git
synced 2025-12-12 01:00:29 +08:00
fixed hid ensuring
This commit is contained in:
parent
301e58148e
commit
635cfc7d9e
@ -75,25 +75,19 @@ class BaseDeviceProcess(multiprocessing.Process): # pylint: disable=too-many-in
|
|||||||
while not self.__stop_event.is_set():
|
while not self.__stop_event.is_set():
|
||||||
try:
|
try:
|
||||||
while not self.__stop_event.is_set():
|
while not self.__stop_event.is_set():
|
||||||
passed = 0
|
|
||||||
try:
|
try:
|
||||||
event: BaseEvent = self.__events_queue.get(timeout=0.05)
|
event: BaseEvent = self.__events_queue.get(timeout=1)
|
||||||
except queue.Empty:
|
except queue.Empty:
|
||||||
if passed >= 20: # 20 * 0.05 = 1 sec
|
self.__ensure_device() # Check device
|
||||||
self._ensure_device() # Check device
|
|
||||||
passed = 0
|
|
||||||
else:
|
|
||||||
passed += 1
|
|
||||||
else:
|
else:
|
||||||
self._process_event(event)
|
self._process_event(event)
|
||||||
passed = 0
|
|
||||||
except Exception:
|
except Exception:
|
||||||
logger.exception("Unexpected HID-%s error", self.__name)
|
logger.exception("Unexpected HID-%s error", self.__name)
|
||||||
self._close_device()
|
self.__close_device()
|
||||||
finally:
|
finally:
|
||||||
time.sleep(1)
|
time.sleep(1)
|
||||||
|
|
||||||
self._close_device()
|
self.__close_device()
|
||||||
|
|
||||||
def is_online(self) -> bool:
|
def is_online(self) -> bool:
|
||||||
return bool(self.__online_shared.value and self.is_alive())
|
return bool(self.__online_shared.value and self.is_alive())
|
||||||
@ -111,7 +105,20 @@ class BaseDeviceProcess(multiprocessing.Process): # pylint: disable=too-many-in
|
|||||||
def _queue_event(self, event: BaseEvent) -> None:
|
def _queue_event(self, event: BaseEvent) -> None:
|
||||||
self.__events_queue.put(event)
|
self.__events_queue.put(event)
|
||||||
|
|
||||||
def _write_report(self, report: bytes) -> bool:
|
def _ensure_write(self, report: bytes, reopen: bool=False, close: bool=False) -> bool:
|
||||||
|
if reopen:
|
||||||
|
self.__close_device()
|
||||||
|
try:
|
||||||
|
if self.__ensure_device():
|
||||||
|
return self.__write_report(report)
|
||||||
|
return False
|
||||||
|
finally:
|
||||||
|
if close:
|
||||||
|
self.__close_device()
|
||||||
|
|
||||||
|
# =====
|
||||||
|
|
||||||
|
def __write_report(self, report: bytes) -> bool:
|
||||||
if self.__noop:
|
if self.__noop:
|
||||||
return True
|
return True
|
||||||
|
|
||||||
@ -141,10 +148,10 @@ class BaseDeviceProcess(multiprocessing.Process): # pylint: disable=too-many-in
|
|||||||
logger.debug("HID-%s write retries left: %d", self.__name, retries)
|
logger.debug("HID-%s write retries left: %d", self.__name, retries)
|
||||||
time.sleep(self.__write_retries_delay)
|
time.sleep(self.__write_retries_delay)
|
||||||
|
|
||||||
self._close_device()
|
self.__close_device()
|
||||||
return False
|
return False
|
||||||
|
|
||||||
def _ensure_device(self) -> bool:
|
def __ensure_device(self) -> bool:
|
||||||
if self.__noop:
|
if self.__noop:
|
||||||
return True
|
return True
|
||||||
|
|
||||||
@ -168,12 +175,12 @@ class BaseDeviceProcess(multiprocessing.Process): # pylint: disable=too-many-in
|
|||||||
logger.debug("HID-%s is busy/unplugged (select)", self.__name)
|
logger.debug("HID-%s is busy/unplugged (select)", self.__name)
|
||||||
except Exception as err:
|
except Exception as err:
|
||||||
logger.error("Can't select() HID-%s: %s: %s", self.__name, type(err).__name__, err)
|
logger.error("Can't select() HID-%s: %s: %s", self.__name, type(err).__name__, err)
|
||||||
self._close_device()
|
self.__close_device()
|
||||||
|
|
||||||
self.__online_shared.value = 0
|
self.__online_shared.value = 0
|
||||||
return False
|
return False
|
||||||
|
|
||||||
def _close_device(self) -> None:
|
def __close_device(self) -> None:
|
||||||
if self.__fd >= 0:
|
if self.__fd >= 0:
|
||||||
try:
|
try:
|
||||||
os.close(self.__fd)
|
os.close(self.__fd)
|
||||||
|
|||||||
@ -73,11 +73,7 @@ class KeyboardProcess(BaseDeviceProcess):
|
|||||||
def cleanup(self) -> None:
|
def cleanup(self) -> None:
|
||||||
self._stop()
|
self._stop()
|
||||||
get_logger().info("Clearing HID-keyboard events ...")
|
get_logger().info("Clearing HID-keyboard events ...")
|
||||||
if self._ensure_device():
|
self._ensure_write(b"\x00" * 8, close=True) # Release all keys and modifiers
|
||||||
try:
|
|
||||||
self._write_report(b"\x00" * 8) # Release all keys and modifiers
|
|
||||||
finally:
|
|
||||||
self._close_device()
|
|
||||||
|
|
||||||
def send_clear_event(self) -> None:
|
def send_clear_event(self) -> None:
|
||||||
self._queue_event(_ClearEvent())
|
self._queue_event(_ClearEvent())
|
||||||
@ -107,9 +103,7 @@ class KeyboardProcess(BaseDeviceProcess):
|
|||||||
def __process_clear_event(self, reopen: bool=False) -> None:
|
def __process_clear_event(self, reopen: bool=False) -> None:
|
||||||
self.__clear_modifiers()
|
self.__clear_modifiers()
|
||||||
self.__clear_keys()
|
self.__clear_keys()
|
||||||
if reopen:
|
self.__send_current_state(reopen=reopen)
|
||||||
self._close_device()
|
|
||||||
self.__send_current_state()
|
|
||||||
|
|
||||||
def __process_modifier_event(self, event: _ModifierEvent) -> None:
|
def __process_modifier_event(self, event: _ModifierEvent) -> None:
|
||||||
if event.modifier in self.__pressed_modifiers:
|
if event.modifier in self.__pressed_modifiers:
|
||||||
@ -140,28 +134,28 @@ class KeyboardProcess(BaseDeviceProcess):
|
|||||||
|
|
||||||
# =====
|
# =====
|
||||||
|
|
||||||
def __send_current_state(self) -> bool:
|
def __send_current_state(self, reopen: bool=False) -> bool:
|
||||||
ok = False
|
if not self._ensure_write(self.__make_report(), reopen=reopen):
|
||||||
if self._ensure_device():
|
|
||||||
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
|
|
||||||
]
|
|
||||||
|
|
||||||
ok = self._write_report(bytes([modifiers, 0] + keys))
|
|
||||||
|
|
||||||
if not ok:
|
|
||||||
self.__clear_modifiers()
|
self.__clear_modifiers()
|
||||||
self.__clear_keys()
|
self.__clear_keys()
|
||||||
return ok
|
return False
|
||||||
|
return True
|
||||||
|
|
||||||
def __clear_modifiers(self) -> None:
|
def __clear_modifiers(self) -> None:
|
||||||
self.__pressed_modifiers.clear()
|
self.__pressed_modifiers.clear()
|
||||||
|
|
||||||
def __clear_keys(self) -> None:
|
def __clear_keys(self) -> None:
|
||||||
self.__pressed_keys = [None] * 6
|
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)
|
||||||
|
|||||||
@ -70,11 +70,8 @@ class MouseProcess(BaseDeviceProcess):
|
|||||||
def cleanup(self) -> None:
|
def cleanup(self) -> None:
|
||||||
self._stop()
|
self._stop()
|
||||||
get_logger().info("Clearing HID-mouse events ...")
|
get_logger().info("Clearing HID-mouse events ...")
|
||||||
if self._ensure_device():
|
report = self.__make_report(0, self.__x, self.__y, 0, 0)
|
||||||
try:
|
self._ensure_write(report, close=True) # Release all buttons
|
||||||
self._write_report(self.__make_report(0, self.__x, self.__y, 0, 0)) # Release all buttons
|
|
||||||
finally:
|
|
||||||
self._close_device()
|
|
||||||
|
|
||||||
def send_clear_event(self) -> None:
|
def send_clear_event(self) -> None:
|
||||||
self._queue_event(_ClearEvent())
|
self._queue_event(_ClearEvent())
|
||||||
@ -118,9 +115,7 @@ class MouseProcess(BaseDeviceProcess):
|
|||||||
|
|
||||||
def __process_clear_event(self, reopen: bool=False) -> None:
|
def __process_clear_event(self, reopen: bool=False) -> None:
|
||||||
self.__clear_state()
|
self.__clear_state()
|
||||||
if reopen:
|
self.__send_current_state(0, 0, reopen=reopen)
|
||||||
self._close_device()
|
|
||||||
self.__send_current_state(0, 0)
|
|
||||||
|
|
||||||
def __process_button_event(self, event: _ButtonEvent) -> None:
|
def __process_button_event(self, event: _ButtonEvent) -> None:
|
||||||
if event.code & self.__pressed_buttons:
|
if event.code & self.__pressed_buttons:
|
||||||
@ -143,19 +138,18 @@ class MouseProcess(BaseDeviceProcess):
|
|||||||
|
|
||||||
# =====
|
# =====
|
||||||
|
|
||||||
def __send_current_state(self, delta_x: int, delta_y: int) -> bool:
|
def __send_current_state(self, delta_x: int, delta_y: int, reopen: bool=False) -> bool:
|
||||||
ok = False
|
report = self.__make_report(
|
||||||
if self._ensure_device():
|
buttons=self.__pressed_buttons,
|
||||||
ok = self._write_report(self.__make_report(
|
to_x=self.__x,
|
||||||
buttons=self.__pressed_buttons,
|
to_y=self.__y,
|
||||||
to_x=self.__x,
|
delta_x=delta_x,
|
||||||
to_y=self.__y,
|
delta_y=delta_y,
|
||||||
delta_x=delta_x,
|
)
|
||||||
delta_y=delta_y,
|
if not self._ensure_write(report, reopen=reopen):
|
||||||
))
|
|
||||||
if not ok:
|
|
||||||
self.__clear_state()
|
self.__clear_state()
|
||||||
return ok
|
return False
|
||||||
|
return True
|
||||||
|
|
||||||
def __clear_state(self) -> None:
|
def __clear_state(self) -> None:
|
||||||
self.__pressed_buttons = 0
|
self.__pressed_buttons = 0
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user