| #!/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 |
| |
| import libevdev |
| import logging |
| |
| logger = logging.getLogger("hidtools.test.apple-keyboard") |
| |
| KERNEL_MODULE = ("apple", "hid-apple") |
| |
| |
| class KbdData(object): |
| pass |
| |
| |
| class AppleKeyboard(ArrayKeyboard): |
| # fmt: off |
| report_descriptor = [ |
| 0x05, 0x01, # Usage Page (Generic Desktop) |
| 0x09, 0x06, # Usage (Keyboard) |
| 0xa1, 0x01, # Collection (Application) |
| 0x85, 0x01, # .Report ID (1) |
| 0x05, 0x07, # .Usage Page (Keyboard) |
| 0x19, 0xe0, # .Usage Minimum (224) |
| 0x29, 0xe7, # .Usage Maximum (231) |
| 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) |
| 0x75, 0x08, # .Report Size (8) |
| 0x95, 0x01, # .Report Count (1) |
| 0x81, 0x01, # .Input (Cnst,Arr,Abs) |
| 0x75, 0x01, # .Report Size (1) |
| 0x95, 0x05, # .Report Count (5) |
| 0x05, 0x08, # .Usage Page (LEDs) |
| 0x19, 0x01, # .Usage Minimum (1) |
| 0x29, 0x05, # .Usage Maximum (5) |
| 0x91, 0x02, # .Output (Data,Var,Abs) |
| 0x75, 0x03, # .Report Size (3) |
| 0x95, 0x01, # .Report Count (1) |
| 0x91, 0x01, # .Output (Cnst,Arr,Abs) |
| 0x75, 0x08, # .Report Size (8) |
| 0x95, 0x06, # .Report Count (6) |
| 0x15, 0x00, # .Logical Minimum (0) |
| 0x26, 0xff, 0x00, # .Logical Maximum (255) |
| 0x05, 0x07, # .Usage Page (Keyboard) |
| 0x19, 0x00, # .Usage Minimum (0) |
| 0x2a, 0xff, 0x00, # .Usage Maximum (255) |
| 0x81, 0x00, # .Input (Data,Arr,Abs) |
| 0xc0, # End Collection |
| 0x05, 0x0c, # Usage Page (Consumer Devices) |
| 0x09, 0x01, # Usage (Consumer Control) |
| 0xa1, 0x01, # Collection (Application) |
| 0x85, 0x47, # .Report ID (71) |
| 0x05, 0x01, # .Usage Page (Generic Desktop) |
| 0x09, 0x06, # .Usage (Keyboard) |
| 0xa1, 0x02, # .Collection (Logical) |
| 0x05, 0x06, # ..Usage Page (Generic Device Controls) |
| 0x09, 0x20, # ..Usage (Battery Strength) |
| 0x15, 0x00, # ..Logical Minimum (0) |
| 0x26, 0xff, 0x00, # ..Logical Maximum (255) |
| 0x75, 0x08, # ..Report Size (8) |
| 0x95, 0x01, # ..Report Count (1) |
| 0x81, 0x02, # ..Input (Data,Var,Abs) |
| 0xc0, # .End Collection |
| 0xc0, # End Collection |
| 0x05, 0x0c, # Usage Page (Consumer Devices) |
| 0x09, 0x01, # Usage (Consumer Control) |
| 0xa1, 0x01, # Collection (Application) |
| 0x85, 0x11, # .Report ID (17) |
| 0x15, 0x00, # .Logical Minimum (0) |
| 0x25, 0x01, # .Logical Maximum (1) |
| 0x75, 0x01, # .Report Size (1) |
| 0x95, 0x03, # .Report Count (3) |
| 0x81, 0x01, # .Input (Cnst,Arr,Abs) |
| 0x75, 0x01, # .Report Size (1) |
| 0x95, 0x01, # .Report Count (1) |
| 0x05, 0x0c, # .Usage Page (Consumer Devices) |
| 0x09, 0xb8, # .Usage (Eject) |
| 0x81, 0x02, # .Input (Data,Var,Abs) |
| 0x06, 0xff, 0x00, # .Usage Page (Vendor Usage Page 0xff) |
| 0x09, 0x03, # .Usage (Vendor Usage 0x03) |
| 0x81, 0x02, # .Input (Data,Var,Abs) |
| 0x75, 0x01, # .Report Size (1) |
| 0x95, 0x03, # .Report Count (3) |
| 0x81, 0x01, # .Input (Cnst,Arr,Abs) |
| 0x05, 0x0c, # .Usage Page (Consumer Devices) |
| 0x85, 0x12, # .Report ID (18) |
| 0x15, 0x00, # .Logical Minimum (0) |
| 0x25, 0x01, # .Logical Maximum (1) |
| 0x75, 0x01, # .Report Size (1) |
| 0x95, 0x01, # .Report Count (1) |
| 0x09, 0xcd, # .Usage (Play/Pause) |
| 0x81, 0x02, # .Input (Data,Var,Abs) |
| 0x09, 0xb3, # .Usage (Fast Forward) |
| 0x81, 0x02, # .Input (Data,Var,Abs) |
| 0x09, 0xb4, # .Usage (Rewind) |
| 0x81, 0x02, # .Input (Data,Var,Abs) |
| 0x09, 0xb5, # .Usage (Scan Next Track) |
| 0x81, 0x02, # .Input (Data,Var,Abs) |
| 0x09, 0xb6, # .Usage (Scan Previous Track) |
| 0x81, 0x02, # .Input (Data,Var,Abs) |
| 0x81, 0x01, # .Input (Cnst,Arr,Abs) |
| 0x81, 0x01, # .Input (Cnst,Arr,Abs) |
| 0x81, 0x01, # .Input (Cnst,Arr,Abs) |
| 0x85, 0x13, # .Report ID (19) |
| 0x15, 0x00, # .Logical Minimum (0) |
| 0x25, 0x01, # .Logical Maximum (1) |
| 0x75, 0x01, # .Report Size (1) |
| 0x95, 0x01, # .Report Count (1) |
| 0x06, 0x01, 0xff, # .Usage Page (Vendor Usage Page 0xff01) |
| 0x09, 0x0a, # .Usage (Vendor Usage 0x0a) |
| 0x81, 0x02, # .Input (Data,Var,Abs) |
| 0x06, 0x01, 0xff, # .Usage Page (Vendor Usage Page 0xff01) |
| 0x09, 0x0c, # .Usage (Vendor Usage 0x0c) |
| 0x81, 0x22, # .Input (Data,Var,Abs,NoPref) |
| 0x75, 0x01, # .Report Size (1) |
| 0x95, 0x06, # .Report Count (6) |
| 0x81, 0x01, # .Input (Cnst,Arr,Abs) |
| 0x85, 0x09, # .Report ID (9) |
| 0x09, 0x0b, # .Usage (Vendor Usage 0x0b) |
| 0x75, 0x08, # .Report Size (8) |
| 0x95, 0x01, # .Report Count (1) |
| 0xb1, 0x02, # .Feature (Data,Var,Abs) |
| 0x75, 0x08, # .Report Size (8) |
| 0x95, 0x02, # .Report Count (2) |
| 0xb1, 0x01, # .Feature (Cnst,Arr,Abs) |
| 0xc0, # End Collection |
| ] |
| # fmt: on |
| |
| def __init__( |
| self, |
| rdesc=report_descriptor, |
| name="Apple Wireless Keyboard", |
| input_info=(BusType.BLUETOOTH, 0x05AC, 0x0256), |
| ): |
| 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 |