HID: Support Consumer (media) keys
What a short title for such a massive diff. This (heavily squashed) commit adds support for Consumer keys such as volume keys, media play/pause/stop, etc. by exposing four HID devices over a single USB lane (as opposed to just exposing a keyboard). This heavily refactors how HIDHelper works due to the new reporting structure. Many of the media keys were changed (mostly Keycodes.Media section), but many (especially anything regarding Application keys) haven't been touched yet - thus many keycodes may still be wrong. Probably worth updating those soon, but I didn't get around to it yet. The definitive list I refered to was http://www.freebsddiary.org/APC/usb_hid_usages.php, which is basically copy-pasta from the official USB HID spec at https://www.usb.org/sites/default/files/documents/hut1_12v2.pdf (warning: massive PDF, not light reading). The only known regression this introduces is that instead of 6KRO as the USB spec usually supports, we can now only have 5KRO (maybe even 4KRO), for reasons I have yet to fully debug - this seems to be related to the report having to include the device descriptor _and_ not supporting a full 8 bytes as it used to. For now I'm willing to accept this, but it definitely will be great to squash that bug. This adds descriptor support for MOUSE and SYSCONTROL devices, as of yet unimplemented.
This commit is contained in:
@@ -1,34 +1,22 @@
|
||||
import logging
|
||||
import string
|
||||
|
||||
from pyb import USB_HID, delay
|
||||
from pyb import USB_HID, delay, hid_keyboard
|
||||
|
||||
from kmk.common.consts import HID_REPORT_STRUCTURE, HIDReportTypes
|
||||
from kmk.common.event_defs import HID_REPORT_EVENT
|
||||
from kmk.common.keycodes import (FIRST_KMK_INTERNAL_KEYCODE, Keycodes,
|
||||
ModifierKeycode, char_lookup)
|
||||
from kmk.common.keycodes import (FIRST_KMK_INTERNAL_KEYCODE, ConsumerKeycode,
|
||||
Keycodes, ModifierKeycode, char_lookup)
|
||||
|
||||
|
||||
def generate_pyb_hid_descriptor():
|
||||
existing_keyboard = list(hid_keyboard)
|
||||
existing_keyboard[-1] = HID_REPORT_STRUCTURE
|
||||
return tuple(existing_keyboard)
|
||||
|
||||
|
||||
class HIDHelper:
|
||||
'''
|
||||
Wraps a HID reporting event. The structure of such events is (courtesy of
|
||||
http://wiki.micropython.org/USB-HID-Keyboard-mode-example-a-password-dongle):
|
||||
|
||||
>Byte 0 is for a modifier key, or combination thereof. It is used as a
|
||||
>bitmap, each bit mapped to a modifier:
|
||||
> bit 0: left control
|
||||
> bit 1: left shift
|
||||
> bit 2: left alt
|
||||
> bit 3: left GUI (Win/Apple/Meta key)
|
||||
> bit 4: right control
|
||||
> bit 5: right shift
|
||||
> bit 6: right alt
|
||||
> bit 7: right GUI
|
||||
>
|
||||
> Examples: 0x02 for Shift, 0x05 for Control+Alt
|
||||
>
|
||||
>Byte 1 is "reserved" (unused, actually)
|
||||
>Bytes 2-7 are for the actual key scancode(s) - up to 6 at a time ("chording").
|
||||
|
||||
Most methods here return `self` upon completion, allowing chaining:
|
||||
|
||||
```python
|
||||
@@ -46,24 +34,64 @@ class HIDHelper:
|
||||
)
|
||||
|
||||
self._hid = USB_HID()
|
||||
self.clear_all()
|
||||
|
||||
# For some bizarre reason this can no longer be 8, it'll just fail to
|
||||
# send anything. This is almost certainly a bug in the report descriptor
|
||||
# sent over in the boot process. For now the sacrifice is that we only
|
||||
# support 5KRO until I figure this out, rather than the 6KRO HID defines.
|
||||
self._evt = bytearray(7)
|
||||
self.report_device = memoryview(self._evt)[0:1]
|
||||
|
||||
# Landmine alert for HIDReportTypes.KEYBOARD: byte index 1 of this view
|
||||
# is "reserved" and evidently (mostly?) unused. However, other modes (or
|
||||
# at least consumer, so far) will use this byte, which is the main reason
|
||||
# this view exists. For KEYBOARD, use report_mods and report_non_mods
|
||||
self.report_keys = memoryview(self._evt)[1:]
|
||||
|
||||
self.report_mods = memoryview(self._evt)[1:2]
|
||||
self.report_non_mods = memoryview(self._evt)[3:]
|
||||
|
||||
def _subscription(self, state, action):
|
||||
if action['type'] == HID_REPORT_EVENT:
|
||||
self.clear_all()
|
||||
|
||||
consumer_key = None
|
||||
for key in state.keys_pressed:
|
||||
if key.code >= FIRST_KMK_INTERNAL_KEYCODE:
|
||||
continue
|
||||
if isinstance(key, ConsumerKeycode):
|
||||
consumer_key = key
|
||||
break
|
||||
|
||||
if isinstance(key, ModifierKeycode):
|
||||
self.add_modifier(key)
|
||||
else:
|
||||
self.add_key(key)
|
||||
reporting_device = self.report_device[0]
|
||||
needed_reporting_device = HIDReportTypes.KEYBOARD
|
||||
|
||||
if key.has_modifiers:
|
||||
for mod in key.has_modifiers:
|
||||
self.add_modifier(mod)
|
||||
if consumer_key:
|
||||
needed_reporting_device = HIDReportTypes.CONSUMER
|
||||
|
||||
if reporting_device != needed_reporting_device:
|
||||
# If we are about to change reporting devices, release
|
||||
# all keys and close our proverbial tab on the existing
|
||||
# device, or keys will get stuck (mostly when releasing
|
||||
# media/consumer keys)
|
||||
self.send()
|
||||
delay(10)
|
||||
|
||||
self.report_device[0] = needed_reporting_device
|
||||
|
||||
if consumer_key:
|
||||
self.add_key(consumer_key)
|
||||
else:
|
||||
for key in state.keys_pressed:
|
||||
if key.code >= FIRST_KMK_INTERNAL_KEYCODE:
|
||||
continue
|
||||
|
||||
if isinstance(key, ModifierKeycode):
|
||||
self.add_modifier(key)
|
||||
else:
|
||||
self.add_key(key)
|
||||
|
||||
if key.has_modifiers:
|
||||
for mod in key.has_modifiers:
|
||||
self.add_modifier(mod)
|
||||
|
||||
self.send()
|
||||
|
||||
@@ -114,37 +142,45 @@ class HIDHelper:
|
||||
return self
|
||||
|
||||
def clear_all(self):
|
||||
self._evt = bytearray(8)
|
||||
for idx, _ in enumerate(self.report_keys):
|
||||
self.report_keys[idx] = 0x00
|
||||
|
||||
return self
|
||||
|
||||
def clear_non_modifiers(self):
|
||||
for pos in range(2, 8):
|
||||
self._evt[pos] = 0x00
|
||||
for idx, _ in enumerate(self.report_non_mods):
|
||||
self.report_non_mods[idx] = 0x00
|
||||
|
||||
return self
|
||||
|
||||
def add_modifier(self, modifier):
|
||||
if isinstance(modifier, ModifierKeycode):
|
||||
self._evt[0] |= modifier.code
|
||||
self.report_mods[0] |= modifier.code
|
||||
else:
|
||||
self._evt[0] |= modifier
|
||||
self.report_mods[0] |= modifier
|
||||
|
||||
return self
|
||||
|
||||
def remove_modifier(self, modifier):
|
||||
if isinstance(modifier, ModifierKeycode):
|
||||
self._evt[0] ^= modifier.code
|
||||
self.report_mods[0] ^= modifier.code
|
||||
else:
|
||||
self._evt[0] ^= modifier
|
||||
self.report_mods[0] ^= modifier
|
||||
|
||||
return self
|
||||
|
||||
def add_key(self, key):
|
||||
# Try to find the first empty slot in the key report, and fill it
|
||||
placed = False
|
||||
for pos in range(2, 8):
|
||||
if self._evt[pos] == 0x00:
|
||||
self._evt[pos] = key.code
|
||||
|
||||
where_to_place = self.report_non_mods
|
||||
|
||||
if self.report_device[0] == HIDReportTypes.CONSUMER:
|
||||
where_to_place = self.report_keys
|
||||
|
||||
for idx, _ in enumerate(where_to_place):
|
||||
if where_to_place[idx] == 0x00:
|
||||
where_to_place[idx] = key.code
|
||||
placed = True
|
||||
break
|
||||
|
||||
@@ -155,9 +191,15 @@ class HIDHelper:
|
||||
|
||||
def remove_key(self, key):
|
||||
removed = False
|
||||
for pos in range(2, 8):
|
||||
if self._evt[pos] == key.code:
|
||||
self._evt[pos] = 0x00
|
||||
|
||||
where_to_place = self.report_non_mods
|
||||
|
||||
if self.report_device[0] == HIDReportTypes.CONSUMER:
|
||||
where_to_place = self.report_keys
|
||||
|
||||
for idx, _ in enumerate(where_to_place):
|
||||
if where_to_place[idx] == key.code:
|
||||
where_to_place[idx] = 0x00
|
||||
removed = True
|
||||
|
||||
if not removed:
|
||||
|
Reference in New Issue
Block a user