mirror of
https://github.com/qmk/qmk_firmware.git
synced 2024-12-15 06:10:53 +00:00
176 lines
5.0 KiB
Python
176 lines
5.0 KiB
Python
"""XAP Device
|
|
"""
|
|
import hid
|
|
import json
|
|
import time
|
|
import gzip
|
|
import random
|
|
import threading
|
|
import functools
|
|
from struct import pack, unpack
|
|
from platform import platform
|
|
|
|
from .types import XAPSecureStatus, XAPFlags, XAPRequest, XAPResponse
|
|
from .routes import XAPRoutes, XAPRouteError
|
|
from .util import u32toBCD
|
|
|
|
|
|
def _gen_token():
|
|
"""Generate XAP token - cannot start with 00xx or 'reserved' (FFFE|FFFF)
|
|
"""
|
|
token = random.randrange(0x0100, 0xFFFD)
|
|
|
|
# swap endianness
|
|
return unpack('<H', pack('>H', token))[0]
|
|
|
|
|
|
class XAPDevice:
|
|
def __init__(self, dev):
|
|
"""Constructor opens hid device and starts dependent services
|
|
"""
|
|
self.responses = {}
|
|
self.do_read = True
|
|
|
|
self.dev = hid.Device(path=dev['path'])
|
|
|
|
self.bg = threading.Thread(target=self._read_loop, daemon=True)
|
|
self.bg.start()
|
|
|
|
def __enter__(self):
|
|
return self
|
|
|
|
def __exit__(self, exc_type, exc_value, exc_traceback):
|
|
self.close()
|
|
|
|
def close(self):
|
|
self.do_read = False
|
|
time.sleep(1)
|
|
self.dev.close()
|
|
|
|
def _read_loop(self):
|
|
"""Background thread to signal waiting transactions
|
|
"""
|
|
while self.do_read:
|
|
data = self.dev.read(XAPResponse.fmt.size, 100)
|
|
if data:
|
|
r = XAPResponse.from_bytes(data)
|
|
event = self.responses.get(r.token)
|
|
if event:
|
|
event._ret = data
|
|
event.set()
|
|
|
|
def _query_device_info(self):
|
|
datalen = int.from_bytes(self.transaction(XAPRoutes.QMK_CONFIG_BLOB_LEN) or bytes(0), 'little')
|
|
if not datalen:
|
|
return {}
|
|
|
|
data = []
|
|
offset = 0
|
|
while offset < datalen:
|
|
chunk = self.transaction(XAPRoutes.QMK_CONFIG_BLOB_CHUNK, offset)
|
|
data += chunk
|
|
offset += len(chunk)
|
|
str_data = gzip.decompress(bytearray(data[:datalen]))
|
|
return json.loads(str_data)
|
|
|
|
def listen(self):
|
|
"""Receive a 'broadcast' message
|
|
"""
|
|
token = 0xFFFF
|
|
event = threading.Event()
|
|
self.responses[token] = event
|
|
|
|
while not hasattr(event, '_ret'):
|
|
event.wait(timeout=0.25)
|
|
|
|
r = XAPResponse.from_bytes(event._ret)
|
|
return (r.flags, r.data[:r.length])
|
|
|
|
def _transaction(self, *args):
|
|
"""Request/Receive
|
|
"""
|
|
# convert args to array of bytes
|
|
data = bytes()
|
|
for arg in args:
|
|
if isinstance(arg, (bytes, bytearray)):
|
|
data += arg
|
|
if isinstance(arg, int): # TODO: remove terrible assumption of u16
|
|
data += arg.to_bytes(2, byteorder='little')
|
|
|
|
token = _gen_token()
|
|
|
|
buffer = XAPRequest(token, len(data), data).to_bytes()
|
|
|
|
event = threading.Event()
|
|
self.responses[token] = event
|
|
|
|
# prepend 0 on windows because reasons...
|
|
if 'windows' in platform().lower():
|
|
buffer = b'\x00' + buffer
|
|
self.dev.write(buffer)
|
|
|
|
event.wait(timeout=1)
|
|
self.responses.pop(token, None)
|
|
if not hasattr(event, '_ret'):
|
|
return None
|
|
|
|
r = XAPResponse.from_bytes(event._ret)
|
|
if r.flags & XAPFlags.SUCCESS == 0:
|
|
return None
|
|
|
|
return r.data[:r.length]
|
|
|
|
@functools.lru_cache
|
|
def capability(self, route):
|
|
cap = int.from_bytes(self._transaction(route) or bytes(0), 'little')
|
|
return cap
|
|
|
|
@functools.lru_cache
|
|
def subsystem(self):
|
|
sub = int.from_bytes(self._transaction(XAPRoutes.XAP_SUBSYSTEM_QUERY) or bytes(0), 'little')
|
|
return sub
|
|
|
|
@functools.lru_cache
|
|
def version(self):
|
|
xap = int.from_bytes(self._transaction(XAPRoutes.XAP_VERSION_QUERY) or bytes(0), 'little')
|
|
qmk = int.from_bytes(self._transaction(XAPRoutes.QMK_VERSION_QUERY) or bytes(0), 'little')
|
|
return {'xap': u32toBCD(xap), 'qmk': u32toBCD(qmk)}
|
|
|
|
def _ensure_route(self, route):
|
|
(sub, rt) = route
|
|
cap = bytes([sub, 1])
|
|
|
|
if self.subsystem() & (1 << sub) == 0:
|
|
raise XAPRouteError("subsystem not available")
|
|
if self.capability(cap) & (1 << rt) == 0:
|
|
raise XAPRouteError("route not available")
|
|
|
|
def transaction(self, route, *args):
|
|
self._ensure_route(route)
|
|
|
|
return self._transaction(route, *args)
|
|
|
|
@functools.lru_cache
|
|
def info(self):
|
|
data = self._query_device_info()
|
|
data['_id'] = self.transaction(XAPRoutes.QMK_HARDWARE_ID)
|
|
data['_version'] = self.version()
|
|
return data
|
|
|
|
def status(self):
|
|
lock = int.from_bytes(self.transaction(XAPRoutes.XAP_SECURE_STATUS) or bytes(0), 'little')
|
|
|
|
data = {}
|
|
data['lock'] = XAPSecureStatus(lock).name
|
|
return data
|
|
|
|
def unlock(self):
|
|
self.transaction(XAPRoutes.XAP_SECURE_UNLOCK)
|
|
|
|
def lock(self):
|
|
self.transaction(XAPRoutes.XAP_SECURE_LOCK)
|
|
|
|
def reset(self):
|
|
status = int.from_bytes(self.transaction(XAPRoutes.QMK_BOOTLOADER_JUMP) or bytes(0), 'little')
|
|
return status == 1
|