#!/bin/env python3
# SPDX-License-Identifier: GPL-2.0
# -*- coding: utf-8 -*-
#
# Copyright (c) 2019 Benjamin Tissoires <benjamin.tissoires@gmail.com>
# Copyright (c) 2019 Red Hat, Inc.
#
from .test_keyboard import ArrayKeyboard, TestArrayKeyboard
from hidtools.util import BusType
from . import base
import libevdev
import logging
logger = logging.getLogger("hidtools.test.apple-keyboard" )
KERNEL_MODULE = base.KernelModule("apple" , "hid-apple" )
class KbdData(object):
pass
class AppleKeyboard(ArrayKeyboard):
# fmt: off
report_descriptor = [
0 x05, 0 x01, # Usage Page (Generic Desktop)
0 x09, 0 x06, # Usage (Keyboard)
0 xa1, 0 x01, # Collection (Application)
0 x85, 0 x01, # .Report ID (1)
0 x05, 0 x07, # .Usage Page (Keyboard)
0 x19, 0 xe0, # .Usage Minimum (224)
0 x29, 0 xe7, # .Usage Maximum (231)
0 x15, 0 x00, # .Logical Minimum (0)
0 x25, 0 x01, # .Logical Maximum (1)
0 x75, 0 x01, # .Report Size (1)
0 x95, 0 x08, # .Report Count (8)
0 x81, 0 x02, # .Input (Data,Var,Abs)
0 x75, 0 x08, # .Report Size (8)
0 x95, 0 x01, # .Report Count (1)
0 x81, 0 x01, # .Input (Cnst,Arr,Abs)
0 x75, 0 x01, # .Report Size (1)
0 x95, 0 x05, # .Report Count (5)
0 x05, 0 x08, # .Usage Page (LEDs)
0 x19, 0 x01, # .Usage Minimum (1)
0 x29, 0 x05, # .Usage Maximum (5)
0 x91, 0 x02, # .Output (Data,Var,Abs)
0 x75, 0 x03, # .Report Size (3)
0 x95, 0 x01, # .Report Count (1)
0 x91, 0 x01, # .Output (Cnst,Arr,Abs)
0 x75, 0 x08, # .Report Size (8)
0 x95, 0 x06, # .Report Count (6)
0 x15, 0 x00, # .Logical Minimum (0)
0 x26, 0 xff, 0 x00, # .Logical Maximum (255)
0 x05, 0 x07, # .Usage Page (Keyboard)
0 x19, 0 x00, # .Usage Minimum (0)
0 x2a, 0 xff, 0 x00, # .Usage Maximum (255)
0 x81, 0 x00, # .Input (Data,Arr,Abs)
0 xc0, # End Collection
0 x05, 0 x0c, # Usage Page (Consumer Devices)
0 x09, 0 x01, # Usage (Consumer Control)
0 xa1, 0 x01, # Collection (Application)
0 x85, 0 x47, # .Report ID (71)
0 x05, 0 x01, # .Usage Page (Generic Desktop)
0 x09, 0 x06, # .Usage (Keyboard)
0 xa1, 0 x02, # .Collection (Logical)
0 x05, 0 x06, # ..Usage Page (Generic Device Controls)
0 x09, 0 x20, # ..Usage (Battery Strength)
0 x15, 0 x00, # ..Logical Minimum (0)
0 x26, 0 xff, 0 x00, # ..Logical Maximum (255)
0 x75, 0 x08, # ..Report Size (8)
0 x95, 0 x01, # ..Report Count (1)
0 x81, 0 x02, # ..Input (Data,Var,Abs)
0 xc0, # .End Collection
0 xc0, # End Collection
0 x05, 0 x0c, # Usage Page (Consumer Devices)
0 x09, 0 x01, # Usage (Consumer Control)
0 xa1, 0 x01, # Collection (Application)
0 x85, 0 x11, # .Report ID (17)
0 x15, 0 x00, # .Logical Minimum (0)
0 x25, 0 x01, # .Logical Maximum (1)
0 x75, 0 x01, # .Report Size (1)
0 x95, 0 x03, # .Report Count (3)
0 x81, 0 x01, # .Input (Cnst,Arr,Abs)
0 x75, 0 x01, # .Report Size (1)
0 x95, 0 x01, # .Report Count (1)
0 x05, 0 x0c, # .Usage Page (Consumer Devices)
0 x09, 0 xb8, # .Usage (Eject)
0 x81, 0 x02, # .Input (Data,Var,Abs)
0 x06, 0 xff, 0 x00, # .Usage Page (Vendor Usage Page 0xff)
0 x09, 0 x03, # .Usage (Vendor Usage 0x03)
0 x81, 0 x02, # .Input (Data,Var,Abs)
0 x75, 0 x01, # .Report Size (1)
0 x95, 0 x03, # .Report Count (3)
0 x81, 0 x01, # .Input (Cnst,Arr,Abs)
0 x05, 0 x0c, # .Usage Page (Consumer Devices)
0 x85, 0 x12, # .Report ID (18)
0 x15, 0 x00, # .Logical Minimum (0)
0 x25, 0 x01, # .Logical Maximum (1)
0 x75, 0 x01, # .Report Size (1)
0 x95, 0 x01, # .Report Count (1)
0 x09, 0 xcd, # .Usage (Play/Pause)
0 x81, 0 x02, # .Input (Data,Var,Abs)
0 x09, 0 xb3, # .Usage (Fast Forward)
0 x81, 0 x02, # .Input (Data,Var,Abs)
0 x09, 0 xb4, # .Usage (Rewind)
0 x81, 0 x02, # .Input (Data,Var,Abs)
0 x09, 0 xb5, # .Usage (Scan Next Track)
0 x81, 0 x02, # .Input (Data,Var,Abs)
0 x09, 0 xb6, # .Usage (Scan Previous Track)
0 x81, 0 x02, # .Input (Data,Var,Abs)
0 x81, 0 x01, # .Input (Cnst,Arr,Abs)
0 x81, 0 x01, # .Input (Cnst,Arr,Abs)
0 x81, 0 x01, # .Input (Cnst,Arr,Abs)
0 x85, 0 x13, # .Report ID (19)
0 x15, 0 x00, # .Logical Minimum (0)
0 x25, 0 x01, # .Logical Maximum (1)
0 x75, 0 x01, # .Report Size (1)
0 x95, 0 x01, # .Report Count (1)
0 x06, 0 x01, 0 xff, # .Usage Page (Vendor Usage Page 0xff01)
0 x09, 0 x0a, # .Usage (Vendor Usage 0x0a)
0 x81, 0 x02, # .Input (Data,Var,Abs)
0 x06, 0 x01, 0 xff, # .Usage Page (Vendor Usage Page 0xff01)
0 x09, 0 x0c, # .Usage (Vendor Usage 0x0c)
0 x81, 0 x22, # .Input (Data,Var,Abs,NoPref)
0 x75, 0 x01, # .Report Size (1)
0 x95, 0 x06, # .Report Count (6)
0 x81, 0 x01, # .Input (Cnst,Arr,Abs)
0 x85, 0 x09, # .Report ID (9)
0 x09, 0 x0b, # .Usage (Vendor Usage 0x0b)
0 x75, 0 x08, # .Report Size (8)
0 x95, 0 x01, # .Report Count (1)
0 xb1, 0 x02, # .Feature (Data,Var,Abs)
0 x75, 0 x08, # .Report Size (8)
0 x95, 0 x02, # .Report Count (2)
0 xb1, 0 x01, # .Feature (Cnst,Arr,Abs)
0 xc0, # End Collection
]
# fmt: on
def __init__(
self,
rdesc=report_descriptor,
name="Apple Wireless Keyboard" ,
input_info=(BusType.BLUETOOTH, 0 x05AC, 0 x0256),
):
super().__init__(rdesc, name, input_info)
self.default_reportID = 1
def send_fn_state(self, state):
data = KbdData()
setattr(data, "0xff0003" , state)
r = self.create_report(data, reportID=17 )
self.call_input_event(r)
return [r]
class TestAppleKeyboard(TestArrayKeyboard):
kernel_modules = [KERNEL_MODULE]
def create_device(self):
return AppleKeyboard()
def test_single_function_key(self):
"" "check for function key reliability." ""
uhdev = self.uhdev
evdev = uhdev.get_evdev()
syn_event = self.syn_event
r = uhdev.event(["F4" ])
expected = [syn_event]
expected.append(libevdev.InputEvent(libevdev.EV_KEY.KEY_ALL_APPLICATIONS, 1 ))
events = uhdev.next_sync_events()
self.debug_reports(r, uhdev, events)
self.assertInputEventsIn(expected, events)
assert evdev.value[libevdev.EV_KEY.KEY_ALL_APPLICATIONS] == 1
assert evdev.value[libevdev.EV_KEY.KEY_FN] == 0
r = uhdev.event([])
expected = [syn_event]
expected.append(libevdev.InputEvent(libevdev.EV_KEY.KEY_ALL_APPLICATIONS, 0 ))
events = uhdev.next_sync_events()
self.debug_reports(r, uhdev, events)
self.assertInputEventsIn(expected, events)
assert evdev.value[libevdev.EV_KEY.KEY_ALL_APPLICATIONS] == 0
def test_single_fn_function_key(self):
"" "check for function key reliability with the fn key." ""
uhdev = self.uhdev
evdev = uhdev.get_evdev()
syn_event = self.syn_event
r = uhdev.send_fn_state(1 )
r.extend(uhdev.event(["F4" ]))
expected = [syn_event]
expected.append(libevdev.InputEvent(libevdev.EV_KEY.KEY_F4, 1 ))
expected.append(libevdev.InputEvent(libevdev.EV_KEY.KEY_FN, 1 ))
events = uhdev.next_sync_events()
self.debug_reports(r, uhdev, events)
self.assertInputEventsIn(expected, events)
assert evdev.value[libevdev.EV_KEY.KEY_F4] == 1
r = uhdev.event([])
expected = [syn_event]
expected.append(libevdev.InputEvent(libevdev.EV_KEY.KEY_F4, 0 ))
events = uhdev.next_sync_events()
self.debug_reports(r, uhdev, events)
self.assertInputEventsIn(expected, events)
assert evdev.value[libevdev.EV_KEY.KEY_F4] == 0
assert evdev.value[libevdev.EV_KEY.KEY_FN] == 1
r = uhdev.send_fn_state(0 )
expected = [syn_event]
expected.append(libevdev.InputEvent(libevdev.EV_KEY.KEY_FN, 0 ))
events = uhdev.next_sync_events()
self.debug_reports(r, uhdev, events)
self.assertInputEventsIn(expected, events)
def test_single_fn_function_key_release_first(self):
"" "check for function key reliability with the fn key." ""
uhdev = self.uhdev
evdev = uhdev.get_evdev()
syn_event = self.syn_event
r = uhdev.send_fn_state(1 )
r.extend(uhdev.event(["F4" ]))
expected = [syn_event]
expected.append(libevdev.InputEvent(libevdev.EV_KEY.KEY_F4, 1 ))
expected.append(libevdev.InputEvent(libevdev.EV_KEY.KEY_FN, 1 ))
events = uhdev.next_sync_events()
self.debug_reports(r, uhdev, events)
self.assertInputEventsIn(expected, events)
assert evdev.value[libevdev.EV_KEY.KEY_F4] == 1
r = uhdev.send_fn_state(0 )
expected = [syn_event]
expected.append(libevdev.InputEvent(libevdev.EV_KEY.KEY_FN, 0 ))
events = uhdev.next_sync_events()
self.debug_reports(r, uhdev, events)
self.assertInputEventsIn(expected, events)
r = uhdev.event([])
expected = [syn_event]
expected.append(libevdev.InputEvent(libevdev.EV_KEY.KEY_F4, 0 ))
events = uhdev.next_sync_events()
self.debug_reports(r, uhdev, events)
self.assertInputEventsIn(expected, events)
assert evdev.value[libevdev.EV_KEY.KEY_F4] == 0
def test_single_fn_function_key_inverted(self):
"" "check for function key reliability with the fn key." ""
uhdev = self.uhdev
evdev = uhdev.get_evdev()
syn_event = self.syn_event
r = uhdev.event(["F4" ])
r.extend(uhdev.send_fn_state(1 ))
expected = [syn_event]
expected.append(libevdev.InputEvent(libevdev.EV_KEY.KEY_ALL_APPLICATIONS, 1 ))
expected.append(libevdev.InputEvent(libevdev.EV_KEY.KEY_FN, 1 ))
events = uhdev.next_sync_events()
self.debug_reports(r, uhdev, events)
self.assertInputEventsIn(expected, events)
assert evdev.value[libevdev.EV_KEY.KEY_ALL_APPLICATIONS] == 1
r = uhdev.event([])
expected = [syn_event]
expected.append(libevdev.InputEvent(libevdev.EV_KEY.KEY_ALL_APPLICATIONS, 0 ))
events = uhdev.next_sync_events()
self.debug_reports(r, uhdev, events)
self.assertInputEventsIn(expected, events)
assert evdev.value[libevdev.EV_KEY.KEY_ALL_APPLICATIONS] == 0
assert evdev.value[libevdev.EV_KEY.KEY_FN] == 1
r = uhdev.send_fn_state(0 )
expected = [syn_event]
expected.append(libevdev.InputEvent(libevdev.EV_KEY.KEY_FN, 0 ))
events = uhdev.next_sync_events()
self.debug_reports(r, uhdev, events)
self.assertInputEventsIn(expected, events)
def test_multiple_fn_function_key_release_first(self):
"" "check for function key reliability with the fn key." ""
uhdev = self.uhdev
evdev = uhdev.get_evdev()
syn_event = self.syn_event
r = uhdev.send_fn_state(1 )
r.extend(uhdev.event(["F4" ]))
r.extend(uhdev.event(["F4" , "F6" ]))
expected = [syn_event]
expected.append(libevdev.InputEvent(libevdev.EV_KEY.KEY_F4, 1 ))
expected.append(libevdev.InputEvent(libevdev.EV_KEY.KEY_F6, 1 ))
expected.append(libevdev.InputEvent(libevdev.EV_KEY.KEY_FN, 1 ))
events = uhdev.next_sync_events()
self.debug_reports(r, uhdev, events)
self.assertInputEventsIn(expected, events)
assert evdev.value[libevdev.EV_KEY.KEY_F4] == 1
assert evdev.value[libevdev.EV_KEY.KEY_F6] == 1
assert evdev.value[libevdev.EV_KEY.KEY_FN] == 1
r = uhdev.event(["F6" ])
expected = [syn_event]
expected.append(libevdev.InputEvent(libevdev.EV_KEY.KEY_F4, 0 ))
events = uhdev.next_sync_events()
self.debug_reports(r, uhdev, events)
self.assertInputEventsIn(expected, events)
assert evdev.value[libevdev.EV_KEY.KEY_F4] == 0
assert evdev.value[libevdev.EV_KEY.KEY_F6] == 1
assert evdev.value[libevdev.EV_KEY.KEY_FN] == 1
r = uhdev.send_fn_state(0 )
expected = [syn_event]
expected.append(libevdev.InputEvent(libevdev.EV_KEY.KEY_FN, 0 ))
events = uhdev.next_sync_events()
self.debug_reports(r, uhdev, events)
self.assertInputEventsIn(expected, events)
assert evdev.value[libevdev.EV_KEY.KEY_F4] == 0
assert evdev.value[libevdev.EV_KEY.KEY_F6] == 1
assert evdev.value[libevdev.EV_KEY.KEY_FN] == 0
r = uhdev.event([])
expected = [syn_event]
expected.append(libevdev.InputEvent(libevdev.EV_KEY.KEY_F6, 0 ))
events = uhdev.next_sync_events()
self.debug_reports(r, uhdev, events)
self.assertInputEventsIn(expected, events)
assert evdev.value[libevdev.EV_KEY.KEY_F4] == 0
assert evdev.value[libevdev.EV_KEY.KEY_F6] == 0
assert evdev.value[libevdev.EV_KEY.KEY_FN] == 0
def test_multiple_fn_function_key_release_between(self):
"" "check for function key reliability with the fn key." ""
uhdev = self.uhdev
evdev = uhdev.get_evdev()
syn_event = self.syn_event
# press F4
r = uhdev.event(["F4" ])
expected = [syn_event]
expected.append(libevdev.InputEvent(libevdev.EV_KEY.KEY_ALL_APPLICATIONS, 1 ))
events = uhdev.next_sync_events()
self.debug_reports(r, uhdev, events)
self.assertInputEventsIn(expected, events)
assert evdev.value[libevdev.EV_KEY.KEY_F4] == 0
assert evdev.value[libevdev.EV_KEY.KEY_ALL_APPLICATIONS] == 1
assert evdev.value[libevdev.EV_KEY.KEY_F6] == 0
assert evdev.value[libevdev.EV_KEY.KEY_KBDILLUMUP] == 0
assert evdev.value[libevdev.EV_KEY.KEY_FN] == 0
# press Fn key
r = uhdev.send_fn_state(1 )
expected = [syn_event]
expected.append(libevdev.InputEvent(libevdev.EV_KEY.KEY_FN, 1 ))
events = uhdev.next_sync_events()
self.debug_reports(r, uhdev, events)
self.assertInputEventsIn(expected, events)
assert evdev.value[libevdev.EV_KEY.KEY_F4] == 0
assert evdev.value[libevdev.EV_KEY.KEY_ALL_APPLICATIONS] == 1
assert evdev.value[libevdev.EV_KEY.KEY_F6] == 0
assert evdev.value[libevdev.EV_KEY.KEY_KBDILLUMUP] == 0
assert evdev.value[libevdev.EV_KEY.KEY_FN] == 1
# keep F4 and press F6
r = uhdev.event(["F4" , "F6" ])
expected = [syn_event]
expected.append(libevdev.InputEvent(libevdev.EV_KEY.KEY_F6, 1 ))
events = uhdev.next_sync_events()
self.debug_reports(r, uhdev, events)
self.assertInputEventsIn(expected, events)
assert evdev.value[libevdev.EV_KEY.KEY_F4] == 0
assert evdev.value[libevdev.EV_KEY.KEY_ALL_APPLICATIONS] == 1
assert evdev.value[libevdev.EV_KEY.KEY_F6] == 1
assert evdev.value[libevdev.EV_KEY.KEY_KBDILLUMUP] == 0
assert evdev.value[libevdev.EV_KEY.KEY_FN] == 1
# keep F4 and F6
r = uhdev.event(["F4" , "F6" ])
expected = []
events = uhdev.next_sync_events()
self.debug_reports(r, uhdev, events)
self.assertInputEventsIn(expected, events)
assert evdev.value[libevdev.EV_KEY.KEY_F4] == 0
assert evdev.value[libevdev.EV_KEY.KEY_ALL_APPLICATIONS] == 1
assert evdev.value[libevdev.EV_KEY.KEY_F6] == 1
assert evdev.value[libevdev.EV_KEY.KEY_KBDILLUMUP] == 0
assert evdev.value[libevdev.EV_KEY.KEY_FN] == 1
# release Fn key and all keys
r = uhdev.send_fn_state(0 )
r.extend(uhdev.event([]))
expected = [syn_event]
expected.append(libevdev.InputEvent(libevdev.EV_KEY.KEY_ALL_APPLICATIONS, 0 ))
expected.append(libevdev.InputEvent(libevdev.EV_KEY.KEY_F6, 0 ))
events = uhdev.next_sync_events()
self.debug_reports(r, uhdev, events)
self.assertInputEventsIn(expected, events)
assert evdev.value[libevdev.EV_KEY.KEY_F4] == 0
assert evdev.value[libevdev.EV_KEY.KEY_ALL_APPLICATIONS] == 0
assert evdev.value[libevdev.EV_KEY.KEY_F6] == 0
assert evdev.value[libevdev.EV_KEY.KEY_KBDILLUMUP] == 0
assert evdev.value[libevdev.EV_KEY.KEY_FN] == 0
def test_single_pageup_key_release_first(self):
"" "check for function key reliability with the [page] up key." ""
uhdev = self.uhdev
evdev = uhdev.get_evdev()
syn_event = self.syn_event
r = uhdev.send_fn_state(1 )
r.extend(uhdev.event(["UpArrow" ]))
expected = [syn_event]
expected.append(libevdev.InputEvent(libevdev.EV_KEY.KEY_PAGEUP, 1 ))
expected.append(libevdev.InputEvent(libevdev.EV_KEY.KEY_FN, 1 ))
events = uhdev.next_sync_events()
self.debug_reports(r, uhdev, events)
self.assertInputEventsIn(expected, events)
assert evdev.value[libevdev.EV_KEY.KEY_PAGEUP] == 1
assert evdev.value[libevdev.EV_KEY.KEY_UP] == 0
assert evdev.value[libevdev.EV_KEY.KEY_FN] == 1
r = uhdev.send_fn_state(0 )
expected = [syn_event]
expected.append(libevdev.InputEvent(libevdev.EV_KEY.KEY_FN, 0 ))
events = uhdev.next_sync_events()
self.debug_reports(r, uhdev, events)
self.assertInputEventsIn(expected, events)
assert evdev.value[libevdev.EV_KEY.KEY_PAGEUP] == 1
assert evdev.value[libevdev.EV_KEY.KEY_UP] == 0
assert evdev.value[libevdev.EV_KEY.KEY_FN] == 0
r = uhdev.event([])
expected = [syn_event]
expected.append(libevdev.InputEvent(libevdev.EV_KEY.KEY_PAGEUP, 0 ))
events = uhdev.next_sync_events()
self.debug_reports(r, uhdev, events)
self.assertInputEventsIn(expected, events)
assert evdev.value[libevdev.EV_KEY.KEY_PAGEUP] == 0
assert evdev.value[libevdev.EV_KEY.KEY_UP] == 0
assert evdev.value[libevdev.EV_KEY.KEY_FN] == 0
Messung V0.5 in Prozent C=96 H=90 G=93
¤ Dauer der Verarbeitung: 0.12 Sekunden
(vorverarbeitet am 2026-06-05)
¤
*© Formatika GbR, Deutschland