| /* Copyright 2016 The ChromiumOS Authors |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| #include "atomic.h" |
| #include "clock.h" |
| #include "common.h" |
| #include "config.h" |
| #include "console.h" |
| #include "gpio.h" |
| #include "hooks.h" |
| #include "hwtimer.h" |
| #include "keyboard_config.h" |
| #include "keyboard_protocol.h" |
| #include "link_defs.h" |
| #include "pwm.h" |
| #include "queue.h" |
| #include "registers.h" |
| #include "tablet_mode.h" |
| #include "task.h" |
| #include "timer.h" |
| #include "usb_api.h" |
| #include "usb_descriptor.h" |
| #include "usb_hid.h" |
| #include "usb_hid_hw.h" |
| #include "usb_hw.h" |
| #include "util.h" |
| |
| /* Console output macro */ |
| #define CPRINTF(format, args...) cprintf(CC_USB, format, ##args) |
| |
| static const int keyboard_debug; |
| |
| struct key_event { |
| uint32_t time; |
| uint8_t keycode; |
| uint8_t pressed; |
| }; |
| |
| static struct queue const key_queue = QUEUE_NULL(16, struct key_event); |
| static mutex_t key_queue_mutex; |
| |
| enum hid_protocol { |
| HID_BOOT_PROTOCOL = 0, |
| HID_REPORT_PROTOCOL = 1, |
| HID_PROTOCOL_COUNT = 2, |
| }; |
| |
| /* Current protocol, behaviour is identical in both modes. */ |
| static enum hid_protocol protocol = HID_REPORT_PROTOCOL; |
| |
| #if defined(CONFIG_KEYBOARD_ASSISTANT_KEY) || \ |
| defined(CONFIG_KEYBOARD_TABLET_MODE_SWITCH) |
| #define HID_KEYBOARD_EXTRA_FIELD |
| #endif |
| |
| /* |
| * Note: This first 8 bytes of this report format cannot be changed, as that |
| * would break HID Boot protocol compatibility (see HID 1.11 "Appendix B: Boot |
| * Interface Descriptors"). |
| */ |
| struct usb_hid_keyboard_report { |
| uint8_t modifiers; /* bitmap of modifiers 224-231 */ |
| uint8_t reserved; /* 0x0 */ |
| uint8_t keys[6]; |
| /* Non-boot protocol fields below */ |
| #ifdef HID_KEYBOARD_EXTRA_FIELD |
| /* Assistant/tablet mode switch bitmask */ |
| uint8_t extra; |
| #endif |
| #ifdef CONFIG_USB_HID_KEYBOARD_VIVALDI |
| uint32_t top_row; /* bitmap of top row action keys */ |
| #endif |
| } __packed; |
| |
| struct usb_hid_keyboard_output_report { |
| uint8_t brightness; |
| } __packed; |
| |
| #define HID_KEYBOARD_BOOT_SIZE 8 |
| |
| #define HID_KEYBOARD_REPORT_SIZE sizeof(struct usb_hid_keyboard_report) |
| #define HID_KEYBOARD_OUTPUT_REPORT_SIZE \ |
| sizeof(struct usb_hid_keyboard_output_report) |
| |
| #define HID_KEYBOARD_EP_INTERVAL_MS 16 /* ms */ |
| |
| /* |
| * Coalesce events happening within some interval. The value must be greater |
| * than EP interval to ensure we cannot have a backlog of keys. |
| * It must also be short enough to ensure that the intended order of key presses |
| * is passed to AP, and that we do not coalesce press and release events (which |
| * would result in lost keys). |
| */ |
| #define COALESCE_INTERVAL (18 * MSEC) |
| |
| /* |
| * Discard key events in the FIFO buffer that are older than this amount of |
| * time. Note that we do not fully drop them, we still update the report, |
| * but we do not send the events individually anymore (so an old key press |
| * and release will be dropped altogether, but a single press/release will |
| * still be reported correctly). |
| */ |
| #define KEY_DISCARD_MAX_TIME (1 * SECOND) |
| |
| /* Modifiers keycode range */ |
| #define HID_KEYBOARD_MODIFIER_LOW 0xe0 |
| #define HID_KEYBOARD_MODIFIER_HIGH 0xe7 |
| |
| /* Supported function key range */ |
| #define HID_F1 0x3a |
| #define HID_F12 0x45 |
| #define HID_F13 0x68 |
| #define HID_F15 0x6a |
| |
| /* Special keys/switches */ |
| #define HID_KEYBOARD_EXTRA_LOW 0xf0 |
| #define HID_KEYBOARD_ASSISTANT_KEY 0xf0 |
| #define HID_KEYBOARD_TABLET_MODE_SWITCH 0xf1 |
| #define HID_KEYBOARD_EXTRA_HIGH 0xf1 |
| |
| /* The standard Chrome OS keyboard matrix table. See HUT 1.12v2 Table 12 and |
| * https://www.w3.org/TR/DOM-Level-3-Events-code . |
| * |
| * Assistant key is mapped as 0xf0, but this key code is never actually send. |
| */ |
| const uint8_t keycodes[KEYBOARD_COLS_MAX][KEYBOARD_ROWS] = { |
| { 0x00, 0x00, 0xe0, 0xe3, 0xe4, HID_KEYBOARD_ASSISTANT_KEY, 0x00, |
| 0x00 }, |
| { 0xe3, 0x29, 0x2b, 0x35, 0x04, 0x1d, 0x1e, 0x14 }, |
| { 0x3a, 0x3d, 0x3c, 0x3b, 0x07, 0x06, 0x20, 0x08 }, |
| { 0x05, 0x0a, 0x17, 0x22, 0x09, 0x19, 0x21, 0x15 }, |
| { 0x43, 0x40, 0x3f, 0x3e, 0x16, 0x1b, 0x1f, 0x1a }, |
| { 0x87, 0x00, 0x30, 0x00, 0x0e, 0x36, 0x25, 0x0c }, |
| { 0x11, 0x0b, 0x1c, 0x23, 0x0d, 0x10, 0x24, 0x18 }, |
| { 0x00, 0x00, 0x64, 0x00, 0x00, 0xe1, 0x00, 0xe5 }, |
| { 0x2e, 0x34, 0x2F, 0x2d, 0x33, 0x38, 0x27, 0x13 }, |
| { 0x00, 0x42, 0x41, 0x68, 0x0f, 0x37, 0x26, 0x12 }, |
| { 0xe6, 0x00, 0x89, 0x00, 0x31, 0x00, 0xe2, 0x00 }, |
| { 0x00, 0x2a, 0x00, 0x31, 0x28, 0x2c, 0x51, 0x52 }, |
| { 0x00, 0x8a, 0x00, 0x8b, 0x00, 0x00, 0x4f, 0x50 }, |
| }; |
| |
| /* HID descriptors */ |
| const struct usb_interface_descriptor USB_IFACE_DESC(USB_IFACE_HID_KEYBOARD) = { |
| .bLength = USB_DT_INTERFACE_SIZE, |
| .bDescriptorType = USB_DT_INTERFACE, |
| .bInterfaceNumber = USB_IFACE_HID_KEYBOARD, |
| .bAlternateSetting = 0, |
| #ifdef CONFIG_USB_HID_KEYBOARD_BACKLIGHT |
| .bNumEndpoints = 2, |
| #else |
| .bNumEndpoints = 1, |
| #endif |
| .bInterfaceClass = USB_CLASS_HID, |
| .bInterfaceSubClass = USB_HID_SUBCLASS_BOOT, |
| .bInterfaceProtocol = USB_HID_PROTOCOL_KEYBOARD, |
| .iInterface = 0, |
| }; |
| const struct usb_endpoint_descriptor USB_EP_DESC(USB_IFACE_HID_KEYBOARD, 81) = { |
| .bLength = USB_DT_ENDPOINT_SIZE, |
| .bDescriptorType = USB_DT_ENDPOINT, |
| .bEndpointAddress = 0x80 | USB_EP_HID_KEYBOARD, |
| .bmAttributes = 0x03 /* Interrupt endpoint */, |
| .wMaxPacketSize = HID_KEYBOARD_REPORT_SIZE, |
| .bInterval = HID_KEYBOARD_EP_INTERVAL_MS /* ms polling interval */ |
| }; |
| |
| #ifdef CONFIG_USB_HID_KEYBOARD_BACKLIGHT |
| const struct usb_endpoint_descriptor USB_EP_DESC(USB_IFACE_HID_KEYBOARD, 02) = { |
| .bLength = USB_DT_ENDPOINT_SIZE, |
| .bDescriptorType = USB_DT_ENDPOINT, |
| .bEndpointAddress = USB_EP_HID_KEYBOARD, |
| .bmAttributes = 0x03 /* Interrupt endpoint */, |
| .wMaxPacketSize = HID_KEYBOARD_OUTPUT_REPORT_SIZE, |
| .bInterval = HID_KEYBOARD_EP_INTERVAL_MS |
| }; |
| #endif |
| |
| #define KEYBOARD_BASE_DESC \ |
| 0x05, 0x01, /* Usage Page (Generic Desktop) */ \ |
| 0x09, 0x06, /* Usage (Keyboard) */ \ |
| 0xA1, 0x01, /* Collection (Application) */ \ |
| \ |
| /* Modifiers */ \ |
| 0x05, 0x07, /* Usage Page (Key Codes) */ \ |
| 0x19, HID_KEYBOARD_MODIFIER_LOW, /* Usage Minimum */ \ |
| 0x29, HID_KEYBOARD_MODIFIER_HIGH, /* Usage Maximum */ \ |
| 0x15, 0x00, /* Logical Minimum (0) */ \ |
| 0x25, 0x01, /* Logical Maximum (1) */ \ |
| 0x75, 0x01, /* Report Size (1) */ \ |
| 0x95, 0x08, /* Report Count (8) */ \ |
| 0x81, 0x02, /* Input (Data, Variable, Absolute), ;Modifier \ |
| byte */ \ |
| \ |
| 0x95, 0x01, /* Report Count (1) */ \ |
| 0x75, 0x08, /* Report Size (8) */ \ |
| 0x81, 0x01, /* Input (Constant), ;Reserved byte */ \ |
| \ |
| /* Normal keys */ \ |
| 0x95, 0x06, /* Report Count (6) */ \ |
| 0x75, 0x08, /* Report Size (8) */ \ |
| 0x15, 0x00, /* Logical Minimum (0) */ \ |
| 0x25, 0xa4, /* Logical Maximum (164) */ \ |
| 0x05, 0x07, /* Usage Page (Key Codes) */ \ |
| 0x19, 0x00, /* Usage Minimum (0) */ \ |
| 0x29, 0xa4, /* Usage Maximum (164) */ \ |
| 0x81, 0x00, /* Input (Data, Array), ;Key arrays (6 bytes) */ |
| |
| #define KEYBOARD_TOP_ROW_DESC \ |
| /* Modifiers */ \ |
| 0x05, 0x0C, /* Consumer Page */ \ |
| 0x0A, 0x24, 0x02, /* AC Back (0x224) */ \ |
| 0x0A, 0x25, 0x02, /* AC Forward (0x225) */ \ |
| 0x0A, 0x27, 0x02, /* AC Refresh (0x227) */ \ |
| 0x0A, 0x32, 0x02, /* AC View Toggle (0x232) */ \ |
| 0x0A, 0x9F, 0x02, /* AC Desktop Show All windows (0x29F) */ \ |
| 0x09, 0x70, /* Display Brightness Decrement (0x70) */ \ |
| 0x09, 0x6F, /* Display Brightness Increment (0x6F) */ \ |
| 0x09, 0xE2, /* Mute (0xE2) */ \ |
| 0x09, 0xEA, /* Volume Decrement (0xEA) */ \ |
| 0x09, 0xE9, /* Volume Increment (0xE9) */ \ |
| 0x0B, 0x46, 0x00, 0x07, 0x00, /* PrintScreen (Page 0x7, Usage \ |
| 0x46) */ \ |
| 0x0A, 0xD0, 0x02, /* Privacy Screen Toggle (0x2D0) */ \ |
| 0x09, 0x7A, /* Keyboard Brightness Decrement (0x7A) */ \ |
| 0x09, 0x79, /* Keyboard Brightness Increment (0x79)*/ \ |
| 0x09, 0xCD, /* Play / Pause (0xCD) */ \ |
| 0x09, 0xB5, /* Scan Next Track (0xB5) */ \ |
| 0x09, 0xB6, /* Scan Previous Track (0xB6) */ \ |
| 0x09, 0x7C, /* Keyboard Backlight OOC (0x7C) */ \ |
| 0x0B, 0x2F, 0x00, 0x0B, 0x00, /* Phone Mute (Page 0xB, Usage \ |
| 0x2F) */ \ |
| 0x09, 0x32, /* Sleep (0x32) */ \ |
| 0x15, 0x00, /* Logical Minimum (0) */ \ |
| 0x25, 0x01, /* Logical Maximum (1) */ \ |
| 0x75, 0x01, /* Report Size (1) */ \ |
| 0x95, 0x14, /* Report Count (20) */ \ |
| 0x81, 0x02, /* Input (Data, Variable, Absolute), ;Modifier \ |
| byte */ \ |
| \ |
| /* 12-bit padding */ \ |
| 0x95, 0x0C, /* Report Count (12) */ \ |
| 0x75, 0x01, /* Report Size (1) */ \ |
| 0x81, 0x01, /* Input (Constant), ;1-bit padding */ |
| |
| #define KEYBOARD_TOP_ROW_FEATURE_DESC \ |
| 0x06, 0xd1, 0xff, /* Usage Page (Google) */ \ |
| 0x09, 0x01, /* Usage (Top Row List) */ \ |
| 0xa1, 0x02, /* Collection (Logical) */ \ |
| 0x05, 0x0a, /* Usage Page (Ordinal) */ \ |
| 0x19, 0x01, /* Usage Minimum (1) */ \ |
| 0x29, CONFIG_USB_HID_KB_NUM_TOP_ROW_KEYS, /* Usage Maximum */ \ |
| 0x95, CONFIG_USB_HID_KB_NUM_TOP_ROW_KEYS, /* Report Count */ \ |
| 0x75, 0x20, /* Report Size (32) */ \ |
| 0xb1, 0x03, /* Feature (Cnst,Var,Abs) */ \ |
| 0xc0, /* End Collection */ |
| |
| /* |
| * Vendor-defined Usage Page 0xffd1: |
| * - 0x18: Assistant key |
| * - 0x19: Tablet mode switch |
| */ |
| #ifdef HID_KEYBOARD_EXTRA_FIELD |
| #ifdef CONFIG_KEYBOARD_ASSISTANT_KEY |
| #define KEYBOARD_ASSISTANT_KEY_DESC \ |
| 0x19, 0x18, /* Usage Minimum */ \ |
| 0x29, 0x18, /* Usage Maximum */ \ |
| 0x15, 0x00, /* Logical Minimum (0) */ \ |
| 0x25, 0x01, /* Logical Maximum (1) */ \ |
| 0x75, 0x01, /* Report Size (1) */ \ |
| 0x95, 0x01, /* Report Count (1) */ \ |
| 0x81, 0x02, /* Input (Data, Variable, Absolute), ;Modifier \ |
| byte */ |
| #else |
| /* No assistant key: just pad 1 bit. */ |
| #define KEYBOARD_ASSISTANT_KEY_DESC \ |
| 0x95, 0x01, /* Report Count (1) */ \ |
| 0x75, 0x01, /* Report Size (1) */ \ |
| 0x81, 0x01, /* Input (Constant), ;1-bit padding */ |
| #endif /* !CONFIG_KEYBOARD_ASSISTANT_KEY */ |
| |
| #ifdef CONFIG_KEYBOARD_TABLET_MODE_SWITCH |
| #define KEYBOARD_TABLET_MODE_SWITCH_DESC \ |
| 0x19, 0x19, /* Usage Minimum */ \ |
| 0x29, 0x19, /* Usage Maximum */ \ |
| 0x15, 0x00, /* Logical Minimum (0) */ \ |
| 0x25, 0x01, /* Logical Maximum (1) */ \ |
| 0x75, 0x01, /* Report Size (1) */ \ |
| 0x95, 0x01, /* Report Count (1) */ \ |
| 0x81, 0x02, /* Input (Data, Variable, Absolute), ;Modifier \ |
| byte */ |
| #else |
| /* No tablet mode swtch: just pad 1 bit. */ |
| #define KEYBOARD_TABLET_MODE_SWITCH_DESC \ |
| 0x95, 0x01, /* Report Count (1) */ \ |
| 0x75, 0x01, /* Report Size (1) */ \ |
| 0x81, 0x01, /* Input (Constant), ;1-bit padding */ |
| #endif /* CONFIG_KEYBOARD_TABLET_MODE_SWITCH */ |
| |
| #define KEYBOARD_VENDOR_DESC \ |
| 0x06, 0xd1, 0xff, /* Usage Page (Vendor-defined 0xffd1) */ \ |
| \ |
| KEYBOARD_ASSISTANT_KEY_DESC KEYBOARD_TABLET_MODE_SWITCH_DESC \ |
| \ |
| 0x95, \ |
| 0x01, /* Report Count (1) */ \ |
| 0x75, 0x06, /* Report Size (6) */ \ |
| 0x81, 0x01, /* Input (Constant), ;6-bit padding */ |
| #endif /* HID_KEYBOARD_EXTRA_FIELD */ |
| |
| #define KEYBOARD_BACKLIGHT_DESC \ |
| 0xA1, 0x02, /* Collection (Logical) */ \ |
| 0x05, 0x14, /* Usage Page (Alphanumeric Display) */ \ |
| 0x09, 0x46, /* Usage (Display Brightness) */ \ |
| 0x95, 0x01, /* Report Count (1) */ \ |
| 0x75, 0x08, /* Report Size (8) */ \ |
| 0x15, 0x00, /* Logical Minimum (0) */ \ |
| 0x25, 0x64, /* Logical Maximum (100) */ \ |
| 0x91, 0x02, /* Output (Data, Variable, Absolute) */ \ |
| 0xC0, /* End Collection */ |
| |
| /* |
| * To allow dynamic detection of keyboard backlights, we define two descriptors. |
| * One has keyboard backlight, and the other one does not. |
| */ |
| |
| /* HID : Report Descriptor */ |
| static const uint8_t report_desc[] = { |
| |
| KEYBOARD_BASE_DESC |
| |
| #ifdef KEYBOARD_VENDOR_DESC |
| KEYBOARD_VENDOR_DESC |
| #endif |
| |
| #ifdef CONFIG_USB_HID_KEYBOARD_VIVALDI |
| KEYBOARD_TOP_ROW_DESC KEYBOARD_TOP_ROW_FEATURE_DESC |
| #endif |
| 0xC0 /* End Collection */ |
| }; |
| |
| #ifdef CONFIG_USB_HID_KEYBOARD_BACKLIGHT |
| |
| /* HID : Report Descriptor with keyboard backlight */ |
| static const uint8_t report_desc_with_backlight[] = { |
| |
| KEYBOARD_BASE_DESC |
| |
| #ifdef KEYBOARD_VENDOR_DESC |
| KEYBOARD_VENDOR_DESC |
| #endif |
| |
| #ifdef CONFIG_USB_HID_KEYBOARD_VIVALDI |
| KEYBOARD_TOP_ROW_DESC KEYBOARD_TOP_ROW_FEATURE_DESC |
| #endif |
| KEYBOARD_BACKLIGHT_DESC |
| |
| 0xC0 /* End Collection */ |
| }; |
| |
| #endif |
| |
| /* HID: HID Descriptor */ |
| const struct usb_hid_descriptor USB_CUSTOM_DESC_VAR(USB_IFACE_HID_KEYBOARD, hid, |
| hid_desc_kb) = { |
| .bLength = 9, |
| .bDescriptorType = USB_HID_DT_HID, |
| .bcdHID = 0x0100, |
| .bCountryCode = 0x00, /* Hardware target country */ |
| .bNumDescriptors = 1, |
| .desc = { { .bDescriptorType = USB_HID_DT_REPORT, |
| .wDescriptorLength = sizeof(report_desc) } } |
| }; |
| |
| #define EP_TX_BUF_SIZE DIV_ROUND_UP(HID_KEYBOARD_REPORT_SIZE, 2) |
| |
| static usb_uint hid_ep_tx_buf[EP_TX_BUF_SIZE] __usb_ram; |
| |
| static volatile int hid_ep_data_ready; |
| |
| #ifdef CONFIG_USB_HID_KEYBOARD_BACKLIGHT |
| #define EP_RX_BUF_SIZE DIV_ROUND_UP(HID_KEYBOARD_OUTPUT_REPORT_SIZE, 2) |
| static usb_uint hid_ep_rx_buf[EP_RX_BUF_SIZE] __usb_ram; |
| #endif |
| |
| static struct usb_hid_keyboard_report report; |
| |
| static void keyboard_process_queue(void); |
| DECLARE_DEFERRED(keyboard_process_queue); |
| |
| static void write_keyboard_report(void) |
| { |
| /* Tell the interrupt handler to send the next buffer. */ |
| hid_ep_data_ready = 1; |
| if ((STM32_USB_EP(USB_EP_HID_KEYBOARD) & EP_TX_MASK) == EP_TX_VALID) { |
| /* Endpoint is busy */ |
| return; |
| } |
| |
| if (atomic_clear((atomic_t *)&hid_ep_data_ready)) { |
| /* |
| * Endpoint is not busy, and interrupt handler did not just |
| * send the buffer: enable TX. |
| */ |
| |
| memcpy_to_usbram((void *)usb_sram_addr(hid_ep_tx_buf), &report, |
| sizeof(report)); |
| STM32_TOGGLE_EP(USB_EP_HID_KEYBOARD, EP_TX_MASK, EP_TX_VALID, |
| 0); |
| } |
| |
| /* |
| * Wake the host. This is required to prevent a race between EP getting |
| * reloaded and host suspending the device, as, ideally, we never want |
| * to have EP loaded during suspend, to avoid reporting stale data. |
| */ |
| usb_wake(); |
| } |
| |
| #ifdef CONFIG_USB_HID_KEYBOARD_BACKLIGHT |
| |
| static void hid_keyboard_rx(void) |
| { |
| struct usb_hid_keyboard_output_report report; |
| memcpy_from_usbram(&report, (void *)usb_sram_addr(hid_ep_rx_buf), |
| HID_KEYBOARD_OUTPUT_REPORT_SIZE); |
| |
| CPRINTF("Keyboard backlight set to %d%%\n", report.brightness); |
| |
| pwm_enable(PWM_CH_KBLIGHT, report.brightness > 0); |
| pwm_set_duty(PWM_CH_KBLIGHT, report.brightness); |
| |
| STM32_TOGGLE_EP(USB_EP_HID_KEYBOARD, EP_TX_RX_MASK, EP_TX_RX_VALID, 0); |
| } |
| |
| #endif |
| |
| static void hid_keyboard_tx(void) |
| { |
| hid_tx(USB_EP_HID_KEYBOARD); |
| if (hid_ep_data_ready) { |
| memcpy_to_usbram((void *)usb_sram_addr(hid_ep_tx_buf), &report, |
| sizeof(report)); |
| STM32_TOGGLE_EP(USB_EP_HID_KEYBOARD, EP_TX_MASK, EP_TX_VALID, |
| 0); |
| hid_ep_data_ready = 0; |
| } |
| |
| if (queue_count(&key_queue) > 0) |
| hook_call_deferred(&keyboard_process_queue_data, 0); |
| } |
| |
| static void hid_keyboard_event(enum usb_ep_event evt) |
| { |
| if (evt == USB_EVENT_RESET) { |
| protocol = HID_REPORT_PROTOCOL; |
| |
| hid_reset(USB_EP_HID_KEYBOARD, hid_ep_tx_buf, |
| HID_KEYBOARD_REPORT_SIZE, |
| #ifdef CONFIG_USB_HID_KEYBOARD_BACKLIGHT |
| hid_ep_rx_buf, HID_KEYBOARD_OUTPUT_REPORT_SIZE |
| #else |
| NULL, 0 |
| #endif |
| ); |
| |
| /* |
| * Reload endpoint on reset, to make sure we report accurate |
| * state to host (this is especially important for tablet mode |
| * switch). |
| */ |
| write_keyboard_report(); |
| return; |
| } |
| |
| if (evt == USB_EVENT_DEVICE_RESUME && queue_count(&key_queue) > 0) |
| hook_call_deferred(&keyboard_process_queue_data, 0); |
| } |
| |
| USB_DECLARE_EP(USB_EP_HID_KEYBOARD, hid_keyboard_tx, |
| #ifdef CONFIG_USB_HID_KEYBOARD_BACKLIGHT |
| hid_keyboard_rx, |
| #else |
| hid_keyboard_tx, |
| #endif |
| hid_keyboard_event); |
| |
| #ifdef CONFIG_USB_HID_KEYBOARD_VIVALDI |
| struct action_key_config { |
| uint32_t mask; /* bit position of usb_hid_keyboard_report.top_row */ |
| uint32_t usage; /*usage ID */ |
| }; |
| |
| static const struct action_key_config action_key[] = { |
| [TK_BACK] = { .mask = BIT(0), .usage = 0x000C0224 }, |
| [TK_FORWARD] = { .mask = BIT(1), .usage = 0x000C0225 }, |
| [TK_REFRESH] = { .mask = BIT(2), .usage = 0x000C0227 }, |
| [TK_FULLSCREEN] = { .mask = BIT(3), .usage = 0x000C0232 }, |
| [TK_OVERVIEW] = { .mask = BIT(4), .usage = 0x000C029F }, |
| [TK_BRIGHTNESS_DOWN] = { .mask = BIT(5), .usage = 0x000C0070 }, |
| [TK_BRIGHTNESS_UP] = { .mask = BIT(6), .usage = 0x000C006F }, |
| [TK_VOL_MUTE] = { .mask = BIT(7), .usage = 0x000C00E2 }, |
| [TK_VOL_DOWN] = { .mask = BIT(8), .usage = 0x000C00EA }, |
| [TK_VOL_UP] = { .mask = BIT(9), .usage = 0x000C00E9 }, |
| [TK_SNAPSHOT] = { .mask = BIT(10), .usage = 0x00070046 }, |
| [TK_PRIVACY_SCRN_TOGGLE] = { .mask = BIT(11), .usage = 0x000C02D0 }, |
| [TK_KBD_BKLIGHT_DOWN] = { .mask = BIT(12), .usage = 0x000C007A }, |
| [TK_KBD_BKLIGHT_UP] = { .mask = BIT(13), .usage = 0x000C0079 }, |
| [TK_PLAY_PAUSE] = { .mask = BIT(14), .usage = 0x000C00CD }, |
| [TK_NEXT_TRACK] = { .mask = BIT(15), .usage = 0x000C00B5 }, |
| [TK_PREV_TRACK] = { .mask = BIT(16), .usage = 0x000C00B6 }, |
| [TK_KBD_BKLIGHT_TOGGLE] = { .mask = BIT(17), .usage = 0x000C007C }, |
| [TK_MICMUTE] = { .mask = BIT(18), .usage = 0x000B002F }, |
| }; |
| |
| /* TK_* is 1-indexed, so the next bit is at ARRAY_SIZE(action_key) - 1 */ |
| static const int SLEEP_KEY_MASK = BIT(ARRAY_SIZE(action_key) - 1); |
| |
| static uint32_t feature_report[CONFIG_USB_HID_KB_NUM_TOP_ROW_KEYS]; |
| |
| static void hid_keyboard_feature_init(void) |
| { |
| const struct ec_response_keybd_config *config = |
| board_vivaldi_keybd_config(); |
| |
| for (int i = 0; i < CONFIG_USB_HID_KB_NUM_TOP_ROW_KEYS; i++) { |
| int key = config->action_keys[i]; |
| |
| if (IN_RANGE(key, 0, ARRAY_SIZE(action_key) - 1)) |
| feature_report[i] = action_key[key].usage; |
| } |
| } |
| DECLARE_HOOK(HOOK_INIT, hid_keyboard_feature_init, HOOK_PRIO_DEFAULT - 1); |
| #endif |
| |
| static int hid_keyboard_get_report(uint8_t report_id, uint8_t report_type, |
| const uint8_t **buffer_ptr, int *buffer_size) |
| { |
| if (report_type == REPORT_TYPE_INPUT) { |
| *buffer_ptr = (uint8_t *)&report; |
| *buffer_size = sizeof(report); |
| return 0; |
| } |
| |
| #ifdef CONFIG_USB_HID_KEYBOARD_VIVALDI |
| if (report_type == REPORT_TYPE_FEATURE) { |
| *buffer_ptr = (uint8_t *)feature_report; |
| *buffer_size = |
| (sizeof(uint32_t) * CONFIG_USB_HID_KB_NUM_TOP_ROW_KEYS); |
| return 0; |
| } |
| #endif |
| |
| return -1; |
| } |
| |
| static struct usb_hid_config_t hid_config_kb = { |
| .report_desc = report_desc, |
| .report_size = sizeof(report_desc), |
| .hid_desc = &hid_desc_kb, |
| .get_report = &hid_keyboard_get_report, |
| }; |
| |
| static int hid_keyboard_iface_request(usb_uint *ep0_buf_rx, |
| usb_uint *ep0_buf_tx) |
| { |
| int ret; |
| |
| ret = hid_iface_request(ep0_buf_rx, ep0_buf_tx, &hid_config_kb); |
| if (ret >= 0) |
| return ret; |
| |
| if (ep0_buf_rx[0] == |
| (USB_DIR_OUT | USB_TYPE_CLASS | USB_RECIP_INTERFACE | |
| (USB_HID_REQ_SET_PROTOCOL << 8))) { |
| uint16_t value = ep0_buf_rx[1]; |
| |
| if (value >= HID_PROTOCOL_COUNT) |
| return -1; |
| |
| protocol = value; |
| |
| /* Reload endpoint with appropriate tx_count. */ |
| btable_ep[USB_EP_HID_KEYBOARD].tx_count = |
| (protocol == HID_BOOT_PROTOCOL) ? |
| HID_KEYBOARD_BOOT_SIZE : |
| HID_KEYBOARD_REPORT_SIZE; |
| STM32_TOGGLE_EP(USB_EP_HID_KEYBOARD, EP_TX_MASK, EP_TX_VALID, |
| 0); |
| |
| btable_ep[0].tx_count = 0; |
| STM32_TOGGLE_EP(0, EP_TX_RX_MASK, EP_TX_RX_VALID, 0); |
| return 0; |
| } else if (ep0_buf_rx[0] == |
| (USB_DIR_IN | USB_TYPE_CLASS | USB_RECIP_INTERFACE | |
| (USB_HID_REQ_GET_PROTOCOL << 8))) { |
| uint8_t value = protocol; |
| |
| memcpy_to_usbram((void *)usb_sram_addr(ep0_buf_tx), &value, |
| sizeof(value)); |
| btable_ep[0].tx_count = 1; |
| STM32_TOGGLE_EP(0, EP_TX_RX_MASK, EP_TX_RX_VALID, 0); |
| return 0; |
| } |
| |
| return -1; |
| } |
| USB_DECLARE_IFACE(USB_IFACE_HID_KEYBOARD, hid_keyboard_iface_request) |
| |
| void keyboard_clear_buffer(void) |
| { |
| mutex_lock(&key_queue_mutex); |
| queue_init(&key_queue); |
| mutex_unlock(&key_queue_mutex); |
| |
| memset(&report, 0, sizeof(report)); |
| #ifdef CONFIG_KEYBOARD_TABLET_MODE_SWITCH |
| if (tablet_get_mode()) |
| report.extra |= 0x01 << (HID_KEYBOARD_TABLET_MODE_SWITCH - |
| HID_KEYBOARD_EXTRA_LOW); |
| #endif |
| write_keyboard_report(); |
| } |
| |
| /* |
| * Convert a function key to the bit mask of corresponding action key. |
| * |
| * Return 0 if no need to map (not a function key or vivaldi not enabled) |
| */ |
| static uint32_t maybe_convert_function_key(int keycode) |
| { |
| #ifndef CONFIG_USB_HID_KEYBOARD_VIVALDI |
| return 0; |
| #else |
| const struct ec_response_keybd_config *config = |
| board_vivaldi_keybd_config(); |
| /* zero-based function key index (e.g. F1 -> 0) */ |
| int index; |
| |
| if (!config) |
| return 0; |
| |
| if (IN_RANGE(keycode, HID_F1, HID_F12)) |
| index = keycode - HID_F1; |
| else if (IN_RANGE(keycode, HID_F13, HID_F15)) |
| index = keycode - HID_F13 + 12; |
| else |
| return 0; /* not a function key */ |
| |
| /* convert F13 to Sleep */ |
| if (index == 12 && (config->capabilities & KEYBD_CAP_SCRNLOCK_KEY)) |
| return SLEEP_KEY_MASK; |
| |
| if (index >= config->num_top_row_keys || |
| config->action_keys[index] == TK_ABSENT) |
| return 0; /* not mapped */ |
| return action_key[config->action_keys[index]].mask; |
| #endif |
| } |
| |
| static void keyboard_process_queue(void) |
| { |
| int i; |
| uint8_t mask; |
| struct key_event ev; |
| int valid = 0; |
| int trimming = 0; |
| uint32_t now = __hw_clock_source_read(); |
| uint32_t first_key_time; |
| |
| if (keyboard_debug) |
| CPRINTF("Q%d (s%d ep%d hw%d)\n", queue_count(&key_queue), |
| usb_is_suspended(), hid_ep_data_ready, |
| (STM32_USB_EP(USB_EP_HID_KEYBOARD) & EP_TX_MASK) == |
| EP_TX_VALID); |
| mutex_lock(&key_queue_mutex); |
| |
| if (queue_count(&key_queue) == 0) { |
| mutex_unlock(&key_queue_mutex); |
| return; |
| } |
| |
| if (usb_is_suspended() || hid_ep_data_ready) { |
| usb_wake(); |
| |
| if (!queue_is_full(&key_queue)) { |
| /* Queue still has space, let's keep gathering keys. */ |
| mutex_unlock(&key_queue_mutex); |
| return; |
| } |
| |
| /* |
| * Queue is full, so we continue, as the code below is |
| * guaranteed to pop at least one key from the queue, but we do |
| * not write the report at the end. |
| */ |
| CPRINTF("Trimming queue (%d %d %d)\n", queue_count(&key_queue), |
| usb_is_suspended(), hid_ep_data_ready); |
| |
| trimming = 1; |
| } |
| |
| /* There is at least one element in the queue. */ |
| queue_peek_units(&key_queue, &ev, 0, 1); |
| first_key_time = ev.time; |
| |
| /* |
| * Pick key events from the queue, coalescing events older than events |
| * within EP interval time to make sure the queue cannot grow, and |
| * dropping keys that are too old. |
| */ |
| while (queue_count(&key_queue) > 0) { |
| uint32_t action_key_mask; |
| |
| queue_peek_units(&key_queue, &ev, 0, 1); |
| if (keyboard_debug) |
| CPRINTF(" =%02x/%d %d %d\n", ev.keycode, ev.keycode, |
| ev.pressed, ev.time - now); |
| |
| if ((now - ev.time) <= KEY_DISCARD_MAX_TIME && |
| (ev.time - first_key_time) >= COALESCE_INTERVAL) |
| break; |
| |
| queue_advance_head(&key_queue, 1); |
| |
| action_key_mask = maybe_convert_function_key(ev.keycode); |
| if (action_key_mask) { |
| #ifdef CONFIG_USB_HID_KEYBOARD_VIVALDI |
| if (ev.pressed) |
| report.top_row |= action_key_mask; |
| else |
| report.top_row &= ~action_key_mask; |
| valid = 1; |
| #endif |
| } else if (ev.keycode >= HID_KEYBOARD_EXTRA_LOW && |
| ev.keycode <= HID_KEYBOARD_EXTRA_HIGH) { |
| #ifdef HID_KEYBOARD_EXTRA_FIELD |
| mask = 0x01 << (ev.keycode - HID_KEYBOARD_EXTRA_LOW); |
| if (ev.pressed) |
| report.extra |= mask; |
| else |
| report.extra &= ~mask; |
| valid = 1; |
| #endif |
| } else if (ev.keycode >= HID_KEYBOARD_MODIFIER_LOW && |
| ev.keycode <= HID_KEYBOARD_MODIFIER_HIGH) { |
| mask = 0x01 << (ev.keycode - HID_KEYBOARD_MODIFIER_LOW); |
| if (ev.pressed) |
| report.modifiers |= mask; |
| else |
| report.modifiers &= ~mask; |
| valid = 1; |
| } else if (ev.pressed) { |
| /* |
| * Add keycode to the list of keys (does nothing if the |
| * array is already full). |
| */ |
| for (i = 0; i < ARRAY_SIZE(report.keys); i++) { |
| /* Is key already pressed? */ |
| if (report.keys[i] == ev.keycode) |
| break; |
| if (report.keys[i] == 0) { |
| report.keys[i] = ev.keycode; |
| valid = 1; |
| break; |
| } |
| } |
| } else { |
| /* |
| * Remove keycode from the list of keys (does nothing |
| * if the key is not in the array). |
| */ |
| for (i = 0; i < ARRAY_SIZE(report.keys); i++) { |
| if (report.keys[i] == ev.keycode) { |
| report.keys[i] = 0; |
| valid = 1; |
| break; |
| } |
| } |
| } |
| } |
| |
| mutex_unlock(&key_queue_mutex); |
| |
| if (valid && !trimming) |
| write_keyboard_report(); |
| } |
| |
| static void queue_keycode_event(uint8_t keycode, int is_pressed) |
| { |
| struct key_event ev = { |
| .time = __hw_clock_source_read(), |
| .keycode = keycode, |
| .pressed = is_pressed, |
| }; |
| |
| mutex_lock(&key_queue_mutex); |
| queue_add_unit(&key_queue, &ev); |
| mutex_unlock(&key_queue_mutex); |
| |
| keyboard_process_queue(); |
| } |
| |
| #ifdef CONFIG_KEYBOARD_TABLET_MODE_SWITCH |
| #include "console.h" |
| |
| static void tablet_mode_change(void) |
| { |
| queue_keycode_event(HID_KEYBOARD_TABLET_MODE_SWITCH, tablet_get_mode()); |
| } |
| DECLARE_HOOK(HOOK_TABLET_MODE_CHANGE, tablet_mode_change, HOOK_PRIO_DEFAULT); |
| /* Run after tablet_mode_init. */ |
| DECLARE_HOOK(HOOK_INIT, tablet_mode_change, HOOK_PRIO_DEFAULT + 1); |
| #endif |
| |
| void keyboard_state_changed(int row, int col, int is_pressed) |
| { |
| uint8_t keycode = keycodes[col][row]; |
| |
| if (!keycode) { |
| CPRINTF("Unknown key at %d/%d\n", row, col); |
| return; |
| } |
| |
| queue_keycode_event(keycode, is_pressed); |
| } |
| |
| void clear_typematic_key(void) |
| { |
| } |
| |
| #ifdef CONFIG_USB_HID_KEYBOARD_BACKLIGHT |
| void usb_hid_keyboard_init(void) |
| { |
| if (board_has_keyboard_backlight()) { |
| hid_config_kb.report_desc = report_desc_with_backlight; |
| hid_config_kb.report_size = sizeof(report_desc_with_backlight); |
| |
| set_descriptor_patch(USB_DESC_KEYBOARD_BACKLIGHT, |
| &hid_desc_kb.desc[0].wDescriptorLength, |
| sizeof(report_desc_with_backlight)); |
| } |
| } |
| /* This needs to happen before usb_init (HOOK_PRIO_DEFAULT) */ |
| DECLARE_HOOK(HOOK_INIT, usb_hid_keyboard_init, HOOK_PRIO_DEFAULT - 1); |
| #endif |