blob: a9d3cd371f50d95a70f5dec111ac931205b39a09 [file] [log] [blame]
/* 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