blob: 86348efbd4a6bffca4b63a5fdc73182738ddaa3d [file] [log] [blame]
// Copyright 2017 The Fuchsia 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 "input.h"
#include <limits.h>
#include <string.h>
#include <ddk/debug.h>
#include <fbl/algorithm.h>
#include <fbl/auto_call.h>
#include <fbl/auto_lock.h>
#include <zircon/assert.h>
#include <zircon/compiler.h>
#include <zircon/device/input.h>
#include <zircon/status.h>
#include "trace.h"
#define LOCAL_TRACE 0
namespace virtio {
// clang-format off
const uint8_t kEventCodeMap[] = {
0, // KEY_RESERVED (0)
41, // KEY_ESC (1)
30, // KEY_1 (2)
31, // KEY_2 (3)
32, // KEY_3 (4)
33, // KEY_4 (5)
34, // KEY_5 (6)
35, // KEY_6 (7)
36, // KEY_7 (8)
37, // KEY_8 (9)
38, // KEY_9 (10)
39, // KEY_0 (11)
45, // KEY_MINUS (12)
46, // KEY_EQUAL (13)
42, // KEY_BACKSPACE (14)
43, // KEY_TAB (15)
20, // KEY_Q (16)
26, // KEY_W (17)
8, // KEY_E (18)
21, // KEY_R (19)
23, // KEY_T (20)
28, // KEY_Y (21)
24, // KEY_U (22)
12, // KEY_I (23)
18, // KEY_O (24)
19, // KEY_P (25)
47, // KEY_LEFTBRACE (26)
48, // KEY_RIGHTBRACE (27)
40, // KEY_ENTER (28)
224, // KEY_LEFTCTRL (29)
4, // KEY_A (30)
22, // KEY_S (31)
7, // KEY_D (32)
9, // KEY_F (33)
10, // KEY_G (34)
11, // KEY_H (35)
13, // KEY_J (36)
14, // KEY_K (37)
15, // KEY_L (38)
51, // KEY_SEMICOLON (39)
52, // KEY_APOSTROPHE (40)
53, // KEY_GRAVE (41)
225, // KEY_LEFTSHIFT (42)
49, // KEY_BACKSLASH (43)
29, // KEY_Z (44)
27, // KEY_X (45)
6, // KEY_C (46)
25, // KEY_V (47)
5, // KEY_B (48)
17, // KEY_N (49)
16, // KEY_M (50)
54, // KEY_COMMA (51)
55, // KEY_DOT (52)
56, // KEY_SLASH (53)
229, // KEY_RIGHTSHIFT (54)
85, // KEY_KPASTERISK (55)
226, // KEY_LEFTALT (56)
44, // KEY_SPACE (57)
57, // KEY_CAPSLOCK (58)
58, // KEY_F1 (59)
59, // KEY_F2 (60)
60, // KEY_F3 (61)
61, // KEY_F4 (62)
62, // KEY_F5 (63)
63, // KEY_F6 (64)
64, // KEY_F7 (65)
65, // KEY_F8 (66)
66, // KEY_F9 (67)
67, // KEY_F10 (68)
83, // KEY_NUMLOCK (69)
71, // KEY_SCROLLLOCK (70)
95, // KEY_KP7 (71)
96, // KEY_KP8 (72)
97, // KEY_KP9 (73)
86, // KEY_KPMINUS (74)
92, // KEY_KP4 (75)
93, // KEY_KP5 (76)
94, // KEY_KP6 (77)
87, // KEY_KPPLUS (78)
89, // KEY_KP1 (79)
90, // KEY_KP2 (80)
91, // KEY_KP3 (81)
98, // KEY_KP0 (82)
99, // KEY_KPDOT (83)
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, // Unsupported
228, // KEY_RIGHTCTRL (97)
0, 0, // Unsupported
230, // KEY_RIGHTALT (100)
};
static const uint8_t kbd_hid_report_desc[] = {
0x05, 0x01, // Usage Page (Generic Desktop Ctrls)
0x09, 0x06, // Usage (Keyboard)
0xA1, 0x01, // Collection (Application)
0x05, 0x07, // Usage Page (Kbrd/Keypad)
0x19, 0xE0, // Usage Minimum (0xE0)
0x29, 0xE7, // Usage Maximum (0xE7)
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,No Wrap,Linear,Preferred State,No Null Position)
0x95, 0x01, // Report Count (1)
0x75, 0x08, // Report Size (8)
0x81, 0x01, // Input (Const,Array,Abs,No Wrap,Linear,Preferred State,No Null Position)
0x95, 0x05, // Report Count (5)
0x75, 0x01, // Report Size (1)
0x05, 0x08, // Usage Page (LEDs)
0x19, 0x01, // Usage Minimum (Num Lock)
0x29, 0x05, // Usage Maximum (Kana)
0x91, 0x02, // Output (Data,Var,Abs,No Wrap,Linear,Preferred State,No Null Position,
// Non-volatile)
0x95, 0x01, // Report Count (1)
0x75, 0x03, // Report Size (3)
0x91, 0x01, // Output (Const,Array,Abs,No Wrap,Linear,Preferred State,No Null Position,
// Non-volatile)
0x95, 0x06, // Report Count (6)
0x75, 0x08, // Report Size (8)
0x15, 0x00, // Logical Minimum (0)
0x25, 0x65, // Logical Maximum (101)
0x05, 0x07, // Usage Page (Kbrd/Keypad)
0x19, 0x00, // Usage Minimum (0x00)
0x29, 0x65, // Usage Maximum (0x65)
0x81, 0x00, // Input (Data,Array,Abs,No Wrap,Linear,Preferred State,No Null Position)
0xC0, // End Collection
};
// clang-format on
// DDK level ops
void InputDevice::virtio_input_release(void* ctx) {
virtio::InputDevice* inp = static_cast<virtio::InputDevice*>(ctx);
return inp->Release();
}
zx_status_t InputDevice::virtio_input_query(void* ctx, uint32_t options, hid_info_t* info) {
virtio::InputDevice* inp = static_cast<virtio::InputDevice*>(ctx);
return inp->Query(options, info);
}
zx_status_t InputDevice::virtio_input_get_descriptor(void* ctx, uint8_t desc_type,
void** data, size_t* len) {
virtio::InputDevice* inp = static_cast<virtio::InputDevice*>(ctx);
return inp->GetDescriptor(desc_type, data, len);
}
zx_status_t InputDevice::virtio_input_get_report(void* ctx, uint8_t rpt_type, uint8_t rpt_id,
void* data, size_t len, size_t* out_len) {
return ZX_ERR_NOT_SUPPORTED;
}
zx_status_t InputDevice::virtio_input_set_report(void* ctx, uint8_t rpt_type, uint8_t rpt_id,
void* data, size_t len) {
return ZX_ERR_NOT_SUPPORTED;
}
zx_status_t InputDevice::virtio_input_get_idle(void* ctx, uint8_t rpt_type, uint8_t* duration) {
return ZX_ERR_NOT_SUPPORTED;
}
zx_status_t InputDevice::virtio_input_set_idle(void* ctx, uint8_t rpt_type, uint8_t duration) {
return ZX_OK;
}
zx_status_t InputDevice::virtio_input_get_protocol(void* ctx, uint8_t* protocol) {
return ZX_ERR_NOT_SUPPORTED;
}
zx_status_t InputDevice::virtio_input_set_protocol(void* ctx, uint8_t protocol) {
return ZX_OK;
}
zx_status_t InputDevice::virtio_input_start(void* ctx, hidbus_ifc_t* ifc, void* cookie) {
virtio::InputDevice* inp = static_cast<virtio::InputDevice*>(ctx);
return inp->Start(ifc, cookie);
}
void InputDevice::virtio_input_stop(void* ctx) {
virtio::InputDevice* inp = static_cast<virtio::InputDevice*>(ctx);
inp->Stop();
}
InputDevice::InputDevice(zx_device_t* bus_device, zx::bti bti, fbl::unique_ptr<Backend> backend)
: Device(bus_device, fbl::move(bti), fbl::move(backend)) {}
InputDevice::~InputDevice() {}
zx_status_t InputDevice::Init() {
LTRACEF("Device %p\n", this);
fbl::AutoLock lock(&lock_);
// Reset the device and read configuration
DeviceReset();
SelectConfig(VIRTIO_INPUT_CFG_ID_NAME, 0);
LTRACEF_LEVEL(2, "name %s\n", config_.u.string);
SelectConfig(VIRTIO_INPUT_CFG_ID_SERIAL, 0);
LTRACEF_LEVEL(2, "serial %s\n", config_.u.string);
SelectConfig(VIRTIO_INPUT_CFG_ID_DEVIDS, 0);
if (config_.size >= sizeof(virtio_input_devids_t)) {
LTRACEF_LEVEL(2, "bustype %d\n", config_.u.ids.bustype);
LTRACEF_LEVEL(2, "vendor %d\n", config_.u.ids.vendor);
LTRACEF_LEVEL(2, "product %d\n", config_.u.ids.product);
LTRACEF_LEVEL(2, "version %d\n", config_.u.ids.version);
}
SelectConfig(VIRTIO_INPUT_CFG_EV_BITS, VIRTIO_INPUT_EV_KEY);
uint8_t cfg_key_size = config_.size;
SelectConfig(VIRTIO_INPUT_CFG_EV_BITS, VIRTIO_INPUT_EV_REL);
uint8_t cfg_rel_size = config_.size;
SelectConfig(VIRTIO_INPUT_CFG_EV_BITS, VIRTIO_INPUT_EV_ABS);
uint8_t cfg_abs_size = config_.size;
// Assume that if the device supports either relative or absolute events
// that it's a pointer, otherwise as long as it supports key events it's a
// keyboard.
if (cfg_rel_size > 0 || cfg_abs_size > 0) {
// Pointer
dev_class_ = HID_DEV_CLASS_POINTER;
} else if (cfg_key_size > 0) {
// Keyboard
dev_class_ = HID_DEV_CLASS_KBD;
} else {
return ZX_ERR_NOT_SUPPORTED;
}
DriverStatusAck();
// Plan to clean up unless everything succeeds.
auto cleanup = fbl::MakeAutoCall([this]() { Release(); });
// Allocate the main vring
zx_status_t status = vring_.Init(0, kEventCount);
if (status != ZX_OK) {
zxlogf(ERROR, "Failed to allocate vring: %s\n", zx_status_get_string(status));
return status;
}
// Allocate event buffers for the ring.
// TODO: Avoid multiple allocations, allocate enough for all buffers once.
for (uint16_t id = 0; id < kEventCount; ++id) {
static_assert(sizeof(virtio_input_event_t) <= PAGE_SIZE, "");
status = io_buffer_init(&buffers_[id], bti_.get(), sizeof(virtio_input_event_t),
IO_BUFFER_RO | IO_BUFFER_CONTIG);
if (status != ZX_OK) {
zxlogf(ERROR, "Failed to allocate I/O buffers: %s\n", zx_status_get_string(status));
return status;
}
}
// Expose event buffers to the host
vring_desc* desc = nullptr;
uint16_t id;
for (uint16_t i = 0; i < kEventCount; ++i) {
desc = vring_.AllocDescChain(1, &id);
if (desc == nullptr) {
zxlogf(ERROR, "Failed to allocate descriptor chain\n");
return ZX_ERR_NO_RESOURCES;
}
ZX_ASSERT(id < kEventCount);
desc->addr = io_buffer_phys(&buffers_[id]);
desc->len = sizeof(virtio_input_event_t);
desc->flags |= VRING_DESC_F_WRITE;
LTRACE_DO(virtio_dump_desc(desc));
vring_.SubmitChain(id);
}
// Prepare the HID report buffer
memset(&report_, 0, sizeof(report_));
StartIrqThread();
DriverStatusOk();
device_ops_.release = virtio_input_release;
hidbus_ops_.query = virtio_input_query;
hidbus_ops_.start = virtio_input_start;
hidbus_ops_.stop = virtio_input_stop;
hidbus_ops_.get_descriptor = virtio_input_get_descriptor;
hidbus_ops_.get_report = virtio_input_get_report;
hidbus_ops_.set_report = virtio_input_set_report;
hidbus_ops_.get_idle = virtio_input_get_idle;
hidbus_ops_.set_idle = virtio_input_set_idle;
hidbus_ops_.get_protocol = virtio_input_get_protocol;
hidbus_ops_.set_protocol = virtio_input_set_protocol;
hidbus_ifc_ = nullptr;
device_add_args_t args = {};
args.version = DEVICE_ADD_ARGS_VERSION;
args.name = "virtio-input";
args.ctx = this;
args.ops = &device_ops_;
args.proto_id = ZX_PROTOCOL_HIDBUS;
args.proto_ops = &hidbus_ops_;
status = device_add(bus_device_, &args, &device_);
if (status != ZX_OK) {
zxlogf(ERROR, "Failed to add device: %s\n", zx_status_get_string(status));
device_ = nullptr;
return status;
}
vring_.Kick();
cleanup.cancel();
return ZX_OK;
}
zx_status_t InputDevice::Start(hidbus_ifc_t* ifc, void* cookie) {
fbl::AutoLock lock(&lock_);
if (hidbus_ifc_ != nullptr) {
return ZX_ERR_ALREADY_BOUND;
}
hidbus_ifc_ = ifc;
hidbus_cookie_ = cookie;
return ZX_OK;
}
void InputDevice::Stop() {
fbl::AutoLock lock(&lock_);
hidbus_ifc_ = nullptr;
hidbus_cookie_ = nullptr;
}
void InputDevice::Release() {
fbl::AutoLock lock(&lock_);
hidbus_ifc_ = nullptr;
hidbus_cookie_ = nullptr;
for (size_t i = 0; i < kEventCount; ++i) {
if (io_buffer_is_valid(&buffers_[i])) {
io_buffer_release(&buffers_[i]);
}
}
}
zx_status_t InputDevice::Query(uint32_t options, hid_info_t* info) {
info->dev_num = dev_class_; // Use type for dev_num for now.
info->dev_class = dev_class_;
info->boot_device = true;
return ZX_OK;
}
zx_status_t InputDevice::GetDescriptor(uint8_t desc_type, void** data, size_t* len) {
if (data == nullptr || len == nullptr) {
return ZX_ERR_INVALID_ARGS;
}
if (desc_type != HID_DESC_TYPE_REPORT) {
return ZX_ERR_NOT_FOUND;
}
const uint8_t* buf = nullptr;
size_t buflen = 0;
// TODO: Handle devices other than keyboards;
buf = static_cast<const uint8_t*>(kbd_hid_report_desc);
buflen = sizeof(kbd_hid_report_desc);
*data = malloc(buflen);
if (*data == nullptr) {
return ZX_ERR_NO_MEMORY;
}
*len = buflen;
memcpy(*data, buf, buflen);
return ZX_OK;
}
void InputDevice::AddKeypressToReport(uint16_t event_code) {
uint8_t hid_code = kEventCodeMap[event_code];
for (size_t i = 0; i != 6; ++i) {
if (report_.usage[i] == hid_code) {
// The key already exists in the report so we ignore it.
return;
}
if (report_.usage[i] == 0) {
report_.usage[i] = hid_code;
return;
}
}
// There's no free slot in the report.
// TODO: Record a rollover status.
}
void InputDevice::RemoveKeypressFromReport(uint16_t event_code) {
uint8_t hid_code = kEventCodeMap[event_code];
int id = -1;
for (int i = 0; i != 6; ++i) {
if (report_.usage[i] == hid_code) {
id = i;
break;
}
}
if (id == -1) {
// They key is not in the report so we ignore it.
return;
}
for (size_t i = id; i != 5; ++i) {
report_.usage[i] = report_.usage[i + 1];
}
report_.usage[5] = 0;
}
void InputDevice::ReceiveEvent(virtio_input_event_t* event) {
// TODO: Support other event types (once we support more than a fake HID keyboard).
if (event->type == VIRTIO_INPUT_EV_KEY) {
if (event->code == 0) {
return;
}
if (event->code >= fbl::count_of(kEventCodeMap)) {
LTRACEF("unknown key\n");
return;
}
if (event->value == VIRTIO_INPUT_EV_KEY_PRESSED) {
AddKeypressToReport(event->code);
} else if (event->value == VIRTIO_INPUT_EV_KEY_RELEASED) {
RemoveKeypressFromReport(event->code);
}
} else if (event->type == VIRTIO_INPUT_EV_SYN) {
fbl::AutoLock lock(&lock_);
if (hidbus_ifc_) {
hidbus_ifc_->io_queue(hidbus_cookie_,
reinterpret_cast<const uint8_t*>(&report_),
sizeof(report_));
}
}
}
void InputDevice::IrqRingUpdate() {
auto free_chain = [this](vring_used_elem* used_elem) {
uint16_t id = static_cast<uint16_t>(used_elem->id & 0xffff);
vring_desc* desc = vring_.DescFromIndex(id);
ZX_ASSERT(id < kEventCount);
ZX_ASSERT(desc->len == sizeof(virtio_input_event_t));
auto evt = static_cast<virtio_input_event_t*>(io_buffer_virt(&buffers_[id]));
ReceiveEvent(evt);
ZX_ASSERT((desc->flags & VRING_DESC_F_NEXT) == 0);
vring_.FreeDesc(id);
};
vring_.IrqRingUpdate(free_chain);
vring_desc* desc = nullptr;
uint16_t id;
bool need_kick = false;
while ((desc = vring_.AllocDescChain(1, &id))) {
desc->len = sizeof(virtio_input_event_t);
vring_.SubmitChain(id);
need_kick = true;
}
if (need_kick) {
vring_.Kick();
}
}
void InputDevice::IrqConfigChange() {
LTRACEF("IrqConfigChange\n");
}
void InputDevice::SelectConfig(uint8_t select, uint8_t subsel) {
WriteDeviceConfig(offsetof(virtio_input_config_t, select), select);
WriteDeviceConfig(offsetof(virtio_input_config_t, subsel), subsel);
CopyDeviceConfig(&config_, sizeof(config_));
}
} // namespace virtio