abschlussarbeit/lib/python3.11/site-packages/usb_hid.py

725 lines
24 KiB
Python

# SPDX-FileCopyrightText: 2021 Melissa LeBlanc-Williams for Adafruit Industries
#
# SPDX-License-Identifier: MIT
"""
`usb_hid` - support for usb hid devices via usb_gadget driver
===========================================================
See `CircuitPython:usb_hid` in CircuitPython for more details.
For now using report ids in the descriptor
# regarding usb_gadget see https://www.kernel.org/doc/Documentation/usb/gadget_configfs.txt
* Author(s): Björn Bösel
"""
from typing import Sequence
from pathlib import Path
import os
import atexit
import sys
for module in ["dwc2", "libcomposite"]:
if Path("/proc/modules").read_text(encoding="utf-8").find(module) == -1:
raise Exception( # pylint: disable=broad-exception-raised
"%s module not present in your kernel. did you insmod it?" % module
)
this = sys.modules[__name__]
this.gadget_root = "/sys/kernel/config/usb_gadget/adafruit-blinka"
this.boot_device = 0
this.devices = []
class Device:
"""
HID Device specification: see
https://github.com/adafruit/circuitpython/blob/main/shared-bindings/usb_hid/Device.c
"""
def __init__(
self,
*,
descriptor: bytes,
usage_page: int,
usage: int,
report_ids: Sequence[int],
in_report_lengths: Sequence[int],
out_report_lengths: Sequence[int],
) -> None:
self.out_report_lengths = out_report_lengths
self.in_report_lengths = in_report_lengths
self.report_ids = report_ids
self.usage = usage
self.usage_page = usage_page
self.descriptor = descriptor
self._last_received_report = None
def send_report(self, report: bytearray, report_id: int = None):
"""Send an HID report. If the device descriptor specifies zero or one report id's,
you can supply `None` (the default) as the value of ``report_id``.
Otherwise you must specify which report id to use when sending the report.
"""
report_id = report_id or self.report_ids[0]
device_path = self.get_device_path(report_id)
with open(device_path, "rb+") as fd:
if report_id > 0:
report = bytearray(report_id.to_bytes(1, "big")) + report
fd.write(report)
@property
def last_received_report(
self,
) -> bytes:
"""The HID OUT report as a `bytes` (read-only). `None` if nothing received.
Same as `get_last_received_report()` with no argument.
Deprecated: will be removed in CircutPython 8.0.0. Use `get_last_received_report()` instead.
"""
return self.get_last_received_report()
def get_last_received_report(self, report_id=None) -> bytes:
"""Get the last received HID OUT or feature report for the given report ID.
The report ID may be omitted if there is no report ID, or only one report ID.
Return `None` if nothing received.
"""
device_path = self.get_device_path(report_id or self.report_ids[0])
with open(device_path, "rb+") as fd:
os.set_blocking(fd.fileno(), False)
report = fd.read(self.out_report_lengths[0])
if report is not None:
self._last_received_report = report
return self._last_received_report
def get_device_path(self, report_id):
"""
translates the /dev/hidg device from the report id
"""
device = (
Path(
"%s/functions/hid.usb%s/dev"
% (this.gadget_root, report_id or self.report_ids[0])
)
.read_text(encoding="utf-8")
.strip()
.split(":")[1]
)
device_path = "/dev/hidg%s" % device
return device_path
KEYBOARD = None
MOUSE = None
CONSUMER_CONTROL = None
Device.KEYBOARD = Device(
descriptor=bytes(
(
0x05,
0x01, # usage page (generic desktop ctrls)
0x09,
0x06, # usage (keyboard)
0xA1,
0x01, # collection (application)
0x85,
0x01, # Report ID (1)
0x05,
0x07, # usage page (kbrd/keypad)
0x19,
0xE0, # usage minimum (0xe0)
0x29,
0xE7, # usage maximum (0xe7)
0x15,
0x00, # logical minimum (0)
0x25,
0x01, # logical maximum (1)
0x75,
0x01, # report size (1)
0x95,
0x08, # report count (8)
0x81,
0x02, # input (data,var,abs,no wrap,linear,preferred state,no null position)
0x95,
0x01, # report count (1)
0x75,
0x08, # report size (8)
0x81,
0x01, # input (const,array,abs,no wrap,linear,preferred state,no null position)
0x95,
0x03, # report count (3)
0x75,
0x01, # report size (1)
0x05,
0x08, # usage page (leds)
0x19,
0x01, # usage minimum (num lock)
0x29,
0x05, # usage maximum (kana)
0x91,
0x02, # output
# (data,var,abs,no wrap,linear,preferred state,no null position,non-volatile)
0x95,
0x01, # report count (1)
0x75,
0x05, # report size (5)
0x91,
0x01, # output
# (const,array,abs,no wrap,linear,preferred state,no null position,non-volatile)
0x95,
0x06, # report count (6)
0x75,
0x08, # report size (8)
0x15,
0x00, # logical minimum (0)
0x26,
0xFF,
0x00, # logical maximum (255)
0x05,
0x07, # usage page (kbrd/keypad)
0x19,
0x00, # usage minimum (0x00)
0x2A,
0xFF,
0x00, # usage maximum (0xff)
0x81,
0x00, # input (data,array,abs,no wrap,linear,preferred state,no null position)
0xC0, # end collection
)
),
usage_page=0x1,
usage=0x6,
report_ids=[0x1],
in_report_lengths=[8],
out_report_lengths=[1],
)
Device.MOUSE = Device(
descriptor=bytes(
(
0x05,
0x01, # Usage Page (Generic Desktop Ctrls)
0x09,
0x02, # Usage (Mouse)
0xA1,
0x01, # Collection (Application)
0x85,
0x02, # Report ID (2)
0x09,
0x01, # Usage (Pointer)
0xA1,
0x00, # Collection (Physical)
0x05,
0x09, # Usage Page (Button)
0x19,
0x01, # Usage Minimum (0x01)
0x29,
0x05, # Usage Maximum (0x05)
0x15,
0x00, # Logical Minimum (0)
0x25,
0x01, # Logical Maximum (1)
0x95,
0x05, # Report Count (5)
0x75,
0x01, # Report Size (1)
0x81,
0x02, # Input (Data,Var,Abs,No Wrap,Linear,Preferred State,No Null Position)
0x95,
0x01, # Report Count (1)
0x75,
0x03, # Report Size (3)
0x81,
0x01, # Input (Const,Array,Abs,No Wrap,Linear,Preferred State,No Null Position)
0x05,
0x01, # Usage Page (Generic Desktop Ctrls)
0x09,
0x30, # Usage (X)
0x09,
0x31, # Usage (Y)
0x15,
0x81, # Logical Minimum (-127)
0x25,
0x7F, # Logical Maximum (127)
0x75,
0x08, # Report Size (8)
0x95,
0x02, # Report Count (2)
0x81,
0x06, # Input (Data,Var,Rel,No Wrap,Linear,Preferred State,No Null Position)
0x09,
0x38, # Usage (Wheel)
0x15,
0x81, # Logical Minimum (-127)
0x25,
0x7F, # Logical Maximum (127)
0x75,
0x08, # Report Size (8)
0x95,
0x01, # Report Count (1)
0x81,
0x06, # Input (Data,Var,Rel,No Wrap,Linear,Preferred State,No Null Position)
0xC0, # End Collection
0xC0, # End Collection
)
),
usage_page=0x1,
usage=0x02,
report_ids=[0x02],
in_report_lengths=[4],
out_report_lengths=[0],
)
Device.CONSUMER_CONTROL = Device(
descriptor=bytes(
(
0x05,
0x0C, # Usage Page (Consumer)
0x09,
0x01, # Usage (Consumer Control)
0xA1,
0x01, # Collection (Application)
0x85,
0x03, # Report ID (3)
0x75,
0x10, # Report Size (16)
0x95,
0x01, # Report Count (1)
0x15,
0x01, # Logical Minimum (1)
0x26,
0x8C,
0x02, # Logical Maximum (652)
0x19,
0x01, # Usage Minimum (Consumer Control)
0x2A,
0x8C,
0x02, # Usage Maximum (AC Send)
0x81,
0x00, # Input (Data,Array,Abs,No Wrap,Linear,Preferred State,No Null Position)
0xC0, # End Collection
)
),
usage_page=0x0C,
usage=0x01,
report_ids=[3],
in_report_lengths=[2],
out_report_lengths=[0],
)
Device.BOOT_KEYBOARD = Device(
descriptor=bytes(
(
0x05,
0x01, # usage page (generic desktop ctrls)
0x09,
0x06, # usage (keyboard)
0xA1,
0x01, # collection (application)
0x05,
0x07, # usage page (kbrd/keypad)
0x19,
0xE0, # usage minimum (0xe0)
0x29,
0xE7, # usage maximum (0xe7)
0x15,
0x00, # logical minimum (0)
0x25,
0x01, # logical maximum (1)
0x75,
0x01, # report size (1)
0x95,
0x08, # report count (8)
0x81,
0x02, # input (data,var,abs,no wrap,linear,preferred state,no null position)
0x95,
0x01, # report count (1)
0x75,
0x08, # report size (8)
0x81,
0x01, # input (const,array,abs,no wrap,linear,preferred state,no null position)
0x95,
0x03, # report count (3)
0x75,
0x01, # report size (1)
0x05,
0x08, # usage page (leds)
0x19,
0x01, # usage minimum (num lock)
0x29,
0x05, # usage maximum (kana)
0x91,
0x02, # output
# (data,var,abs,no wrap,linear,preferred state,no null position,non-volatile)
0x95,
0x01, # report count (1)
0x75,
0x05, # report size (5)
0x91,
0x01, # output
# (const,array,abs,no wrap,linear,preferred state,no null position,non-volatile)
0x95,
0x06, # report count (6)
0x75,
0x08, # report size (8)
0x15,
0x00, # logical minimum (0)
0x26,
0xFF,
0x00, # logical maximum (255)
0x05,
0x07, # usage page (kbrd/keypad)
0x19,
0x00, # usage minimum (0x00)
0x2A,
0xFF,
0x00, # usage maximum (0xff)
0x81,
0x00, # input (data,array,abs,no wrap,linear,preferred state,no null position)
0xC0, # end collection
)
),
usage_page=0x1,
usage=0x6,
report_ids=[0x0],
in_report_lengths=[8],
out_report_lengths=[1],
)
Device.BOOT_MOUSE = Device(
descriptor=bytes(
(
0x05,
0x01, # Usage Page (Generic Desktop Ctrls)
0x09,
0x02, # Usage (Mouse)
0xA1,
0x01, # Collection (Application)
0x09,
0x01, # Usage (Pointer)
0xA1,
0x00, # Collection (Physical)
0x05,
0x09, # Usage Page (Button)
0x19,
0x01, # Usage Minimum (0x01)
0x29,
0x05, # Usage Maximum (0x05)
0x15,
0x00, # Logical Minimum (0)
0x25,
0x01, # Logical Maximum (1)
0x95,
0x05, # Report Count (5)
0x75,
0x01, # Report Size (1)
0x81,
0x02, # Input (Data,Var,Abs,No Wrap,Linear,Preferred State,No Null Position)
0x95,
0x01, # Report Count (1)
0x75,
0x03, # Report Size (3)
0x81,
0x01, # Input (Const,Array,Abs,No Wrap,Linear,Preferred State,No Null Position)
0x05,
0x01, # Usage Page (Generic Desktop Ctrls)
0x09,
0x30, # Usage (X)
0x09,
0x31, # Usage (Y)
0x15,
0x81, # Logical Minimum (-127)
0x25,
0x7F, # Logical Maximum (127)
0x75,
0x08, # Report Size (8)
0x95,
0x02, # Report Count (2)
0x81,
0x06, # Input (Data,Var,Rel,No Wrap,Linear,Preferred State,No Null Position)
0x09,
0x38, # Usage (Wheel)
0x15,
0x81, # Logical Minimum (-127)
0x25,
0x7F, # Logical Maximum (127)
0x75,
0x08, # Report Size (8)
0x95,
0x01, # Report Count (1)
0x81,
0x06, # Input (Data,Var,Rel,No Wrap,Linear,Preferred State,No Null Position)
0xC0, # End Collection
0xC0, # End Collection
)
),
usage_page=0x1,
usage=0x02,
report_ids=[0],
in_report_lengths=[4],
out_report_lengths=[0],
)
def disable() -> None:
"""Do not present any USB HID devices to the host computer.
Can be called in ``boot.py``, before USB is connected.
The HID composite device is normally enabled by default,
but on some boards with limited endpoints, including STM32F4,
it is disabled by default. You must turn off another USB device such
as `usb_cdc` or `storage` to free up endpoints for use by `usb_hid`.
"""
try:
Path("%s/UDC" % this.gadget_root).write_text("", encoding="utf-8")
except FileNotFoundError:
pass
for symlink in Path(this.gadget_root).glob("configs/**/hid.usb*"):
symlink.unlink()
for strings_file in Path(this.gadget_root).rglob("configs/*/strings/*/*"):
if strings_file.is_dir():
strings_file.rmdir()
for strings_file in Path(this.gadget_root).rglob("configs/*/strings/*"):
if strings_file.is_dir():
strings_file.rmdir()
for config_dir in Path(this.gadget_root).rglob("configs/*"):
if config_dir.is_dir():
config_dir.rmdir()
for function_dir in Path(this.gadget_root).rglob("functions/*"):
if function_dir.is_dir():
function_dir.rmdir()
try:
Path(this.gadget_root).rmdir()
except FileNotFoundError:
pass
this.devices = []
atexit.register(disable)
def enable(requested_devices: Sequence[Device], boot_device: int = 0) -> None:
"""Specify which USB HID devices that will be available.
Can be called in ``boot.py``, before USB is connected.
:param Sequence devices: `Device` objects.
If `devices` is empty, HID is disabled. The order of the ``Devices``
may matter to the host. For instance, for MacOS, put the mouse device
before any Gamepad or Digitizer HID device or else it will not work.
:param int boot_device: If non-zero, inform the host that support for a
a boot HID device is available.
If ``boot_device=1``, a boot keyboard is available.
If ``boot_device=2``, a boot mouse is available. No other values are allowed.
See below.
If you enable too many devices at once, you will run out of USB endpoints.
The number of available endpoints varies by microcontroller.
CircuitPython will go into safe mode after running ``boot.py`` to inform you if
not enough endpoints are available.
**Boot Devices**
Boot devices implement a fixed, predefined report descriptor, defined in
https://www.usb.org/sites/default/files/hid1_12.pdf, Appendix B. A USB host
can request to use the boot device if the USB device says it is available.
Usually only a BIOS or other kind of limited-functionality
host needs boot keyboard support.
For example, to make a boot keyboard available, you can use this code::
usb_hid.enable((Device.KEYBOARD), boot_device=1) # 1 for a keyboard
If the host requests the boot keyboard, the report descriptor provided by `Device.KEYBOARD`
will be ignored, and the predefined report descriptor will be used.
But if the host does not request the boot keyboard,
the descriptor provided by `Device.KEYBOARD` will be used.
The HID boot device must usually be the first or only device presented by CircuitPython.
The HID device will be USB interface number 0.
To make sure it is the first device, disable other USB devices, including CDC and MSC
(CIRCUITPY).
If you specify a non-zero ``boot_device``, and it is not the first device, CircuitPython
will enter safe mode to report this error.
"""
this.boot_device = boot_device
if len(requested_devices) == 0:
disable()
return
if boot_device == 1:
requested_devices = [Device.BOOT_KEYBOARD]
if boot_device == 2:
requested_devices = [Device.BOOT_MOUSE]
# """
# 1. Creating the gadgets
# -----------------------
#
# For each gadget to be created its corresponding directory must be created::
#
# $ mkdir $CONFIGFS_HOME/usb_gadget/<gadget name>
#
# e.g.::
#
# $ mkdir $CONFIGFS_HOME/usb_gadget/g1
#
# ...
# ...
# ...
#
# $ cd $CONFIGFS_HOME/usb_gadget/g1
#
# Each gadget needs to have its vendor id <VID> and product id <PID> specified::
#
# $ echo <VID> > idVendor
# $ echo <PID> > idProduct
#
# A gadget also needs its serial number, manufacturer and product strings.
# In order to have a place to store them, a strings subdirectory must be created
# for each language, e.g.::
#
# $ mkdir strings/0x409
#
# Then the strings can be specified::
#
# $ echo <serial number> > strings/0x409/serialnumber
# $ echo <manufacturer> > strings/0x409/manufacturer
# $ echo <product> > strings/0x409/product
# """
Path("%s/functions" % this.gadget_root).mkdir(parents=True, exist_ok=True)
Path("%s/configs" % this.gadget_root).mkdir(parents=True, exist_ok=True)
Path("%s/bcdDevice" % this.gadget_root).write_text(
"%s" % 1, encoding="utf-8"
) # Version 1.0.0
Path("%s/bcdUSB" % this.gadget_root).write_text(
"%s" % 0x0200, encoding="utf-8"
) # USB 2.0
Path("%s/bDeviceClass" % this.gadget_root).write_text(
"%s" % 0x00, encoding="utf-8"
) # multipurpose i guess?
Path("%s/bDeviceProtocol" % this.gadget_root).write_text(
"%s" % 0x00, encoding="utf-8"
)
Path("%s/bDeviceSubClass" % this.gadget_root).write_text(
"%s" % 0x00, encoding="utf-8"
)
Path("%s/bMaxPacketSize0" % this.gadget_root).write_text(
"%s" % 0x08, encoding="utf-8"
)
Path("%s/idProduct" % this.gadget_root).write_text(
"%s" % 0x0104, encoding="utf-8"
) # Multifunction Composite Gadget
Path("%s/idVendor" % this.gadget_root).write_text(
"%s" % 0x1D6B, encoding="utf-8"
) # Linux Foundation
# """
# 2. Creating the configurations
# ------------------------------
#
# Each gadget will consist of a number of configurations, their corresponding
# directories must be created:
#
# $ mkdir configs/<name>.<number>
#
# where <name> can be any string which is legal in a filesystem and the
# <number> is the configuration's number, e.g.::
#
# $ mkdir configs/c.1
#
# ...
# ...
# ...
#
# Each configuration also needs its strings, so a subdirectory must be created
# for each language, e.g.::
#
# $ mkdir configs/c.1/strings/0x409
#
# Then the configuration string can be specified::
#
# $ echo <configuration> > configs/c.1/strings/0x409/configuration
#
# Some attributes can also be set for a configuration, e.g.::
#
# $ echo 120 > configs/c.1/MaxPower
# """
for device in requested_devices:
config_root = "%s/configs/device.1" % this.gadget_root
Path("%s/" % config_root).mkdir(parents=True, exist_ok=True)
Path("%s/strings/0x409" % config_root).mkdir(parents=True, exist_ok=True)
Path("%s/strings/0x409/configuration" % config_root).write_text(
"my configuration", encoding="utf-8"
)
Path("%s/MaxPower" % config_root).write_text("150", encoding="utf-8")
Path("%s/bmAttributes" % config_root).write_text("%s" % 0x080, encoding="utf-8")
this.devices.append(device)
# """
# 3. Creating the functions
# -------------------------
#
# The gadget will provide some functions, for each function its corresponding
# directory must be created::
#
# $ mkdir functions/<name>.<instance name>
#
# where <name> corresponds to one of allowed function names and instance name
# is an arbitrary string allowed in a filesystem, e.g.::
#
# $ mkdir functions/ncm.usb0 # usb_f_ncm.ko gets loaded with request_module()
#
# ...
# ...
# ...
#
# Each function provides its specific set of attributes, with either read-only
# or read-write access. Where applicable they need to be written to as
# appropriate.
# Please refer to Documentation/ABI/*/configfs-usb-gadget* for more information. """
for report_index, report_id in enumerate(device.report_ids):
function_root = "%s/functions/hid.usb%s" % (this.gadget_root, report_id)
try:
Path("%s/" % function_root).mkdir(parents=True)
except FileExistsError:
continue
Path("%s/protocol" % function_root).write_text(
"%s" % report_id, encoding="utf-8"
)
Path("%s/report_length" % function_root).write_text(
"%s" % device.in_report_lengths[report_index], encoding="utf-8"
)
Path("%s/subclass" % function_root).write_text("%s" % 1, encoding="utf-8")
Path("%s/report_desc" % function_root).write_bytes(device.descriptor)
# """
# 4. Associating the functions with their configurations
# ------------------------------------------------------
#
# At this moment a number of gadgets is created, each of which has a number of
# configurations specified and a number of functions available. What remains
# is specifying which function is available in which configuration (the same
# function can be used in multiple configurations). This is achieved with
# creating symbolic links::
#
# $ ln -s functions/<name>.<instance name> configs/<name>.<number>
#
# e.g.::
#
# $ ln -s functions/ncm.usb0 configs/c.1 """
try:
Path("%s/hid.usb%s" % (config_root, report_id)).symlink_to(
function_root
)
except FileNotFoundError:
pass
# """ 5. Enabling the gadget
# ----------------------
# Such a gadget must be finally enabled so that the USB host can enumerate it.
#
# In order to enable the gadget it must be bound to a UDC (USB Device
# Controller)::
#
# $ echo <udc name> > UDC
#
# where <udc name> is one of those found in /sys/class/udc/*
# e.g.::
#
# $ echo s3c-hsotg > UDC """
udc = next(Path("/sys/class/udc/").glob("*"))
Path("%s/UDC" % this.gadget_root).write_text("%s" % udc.name, encoding="utf-8")