/* Copyright 2016 The Chromium OS Authors. All rights reserved. * 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 "util.h" #include "usb_api.h" #include "usb_descriptor.h" #include "usb_hw.h" #include "usb_hid.h" #include "usb_hid_hw.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 struct mutex 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_current_buf; 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((int *)&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); 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); #ifdef CONFIG_USB_HID_KEYBOARD_VIVALDI 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))) 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) { const struct ec_response_keybd_config *config = board_vivaldi_keybd_config(); /* zero-based function key index (e.g. F1 -> 0) */ int index; if (!IS_ENABLED(CONFIG_USB_HID_KEYBOARD_VIVALDI) || !config) return 0; if (IN_RANGE(keycode, HID_F1, HID_F12 + 1)) index = keycode - HID_F1; else if (IN_RANGE(keycode, HID_F13, HID_F15 + 1)) 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; } 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