blob: 4876338e92dbd7b6bde527b421b1e71f7f0a6971 [file] [log] [blame]
/*
* Copyright (C) 2017 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include "host/frontend/vnc_server/virtual_inputs.h"
#include <android-base/logging.h>
#include <android-base/strings.h>
#include <gflags/gflags.h>
#include <linux/input.h>
#include <linux/uinput.h>
#include <cstdint>
#include <mutex>
#include <thread>
#include "keysyms.h"
#include "common/libs/confui/confui.h"
#include "common/libs/fs/shared_select.h"
#include "host/libs/config/cuttlefish_config.h"
#include "host/libs/config/logging.h"
using cuttlefish::vnc::VirtualInputs;
DEFINE_string(touch_fds, "",
"A list of fds for sockets where to accept touch connections");
DEFINE_int32(keyboard_fd, -1,
"A fd for a socket where to accept keyboard connections");
DEFINE_bool(write_virtio_input, false,
"Whether to write the virtio_input struct over the socket");
namespace {
// Necessary to define here as the virtio_input.h header is not available
// in the host glibc.
struct virtio_input_event {
std::uint16_t type;
std::uint16_t code;
std::int32_t value;
};
void AddKeyMappings(std::map<uint32_t, uint16_t>* key_mapping) {
(*key_mapping)[cuttlefish::xk::AltLeft] = KEY_LEFTALT;
(*key_mapping)[cuttlefish::xk::ControlLeft] = KEY_LEFTCTRL;
(*key_mapping)[cuttlefish::xk::ShiftLeft] = KEY_LEFTSHIFT;
(*key_mapping)[cuttlefish::xk::AltRight] = KEY_RIGHTALT;
(*key_mapping)[cuttlefish::xk::ControlRight] = KEY_RIGHTCTRL;
(*key_mapping)[cuttlefish::xk::ShiftRight] = KEY_RIGHTSHIFT;
(*key_mapping)[cuttlefish::xk::MetaLeft] = KEY_LEFTMETA;
(*key_mapping)[cuttlefish::xk::MetaRight] = KEY_RIGHTMETA;
(*key_mapping)[cuttlefish::xk::MultiKey] = KEY_COMPOSE;
(*key_mapping)[cuttlefish::xk::CapsLock] = KEY_CAPSLOCK;
(*key_mapping)[cuttlefish::xk::NumLock] = KEY_NUMLOCK;
(*key_mapping)[cuttlefish::xk::ScrollLock] = KEY_SCROLLLOCK;
(*key_mapping)[cuttlefish::xk::BackSpace] = KEY_BACKSPACE;
(*key_mapping)[cuttlefish::xk::Tab] = KEY_TAB;
(*key_mapping)[cuttlefish::xk::Return] = KEY_ENTER;
(*key_mapping)[cuttlefish::xk::Escape] = KEY_ESC;
(*key_mapping)[' '] = KEY_SPACE;
(*key_mapping)['!'] = KEY_1;
(*key_mapping)['"'] = KEY_APOSTROPHE;
(*key_mapping)['#'] = KEY_3;
(*key_mapping)['$'] = KEY_4;
(*key_mapping)['%'] = KEY_5;
(*key_mapping)['^'] = KEY_6;
(*key_mapping)['&'] = KEY_7;
(*key_mapping)['\''] = KEY_APOSTROPHE;
(*key_mapping)['('] = KEY_9;
(*key_mapping)[')'] = KEY_0;
(*key_mapping)['*'] = KEY_8;
(*key_mapping)['+'] = KEY_EQUAL;
(*key_mapping)[','] = KEY_COMMA;
(*key_mapping)['-'] = KEY_MINUS;
(*key_mapping)['.'] = KEY_DOT;
(*key_mapping)['/'] = KEY_SLASH;
(*key_mapping)['0'] = KEY_0;
(*key_mapping)['1'] = KEY_1;
(*key_mapping)['2'] = KEY_2;
(*key_mapping)['3'] = KEY_3;
(*key_mapping)['4'] = KEY_4;
(*key_mapping)['5'] = KEY_5;
(*key_mapping)['6'] = KEY_6;
(*key_mapping)['7'] = KEY_7;
(*key_mapping)['8'] = KEY_8;
(*key_mapping)['9'] = KEY_9;
(*key_mapping)[':'] = KEY_SEMICOLON;
(*key_mapping)[';'] = KEY_SEMICOLON;
(*key_mapping)['<'] = KEY_COMMA;
(*key_mapping)['='] = KEY_EQUAL;
(*key_mapping)['>'] = KEY_DOT;
(*key_mapping)['?'] = KEY_SLASH;
(*key_mapping)['@'] = KEY_2;
(*key_mapping)['A'] = KEY_A;
(*key_mapping)['B'] = KEY_B;
(*key_mapping)['C'] = KEY_C;
(*key_mapping)['D'] = KEY_D;
(*key_mapping)['E'] = KEY_E;
(*key_mapping)['F'] = KEY_F;
(*key_mapping)['G'] = KEY_G;
(*key_mapping)['H'] = KEY_H;
(*key_mapping)['I'] = KEY_I;
(*key_mapping)['J'] = KEY_J;
(*key_mapping)['K'] = KEY_K;
(*key_mapping)['L'] = KEY_L;
(*key_mapping)['M'] = KEY_M;
(*key_mapping)['N'] = KEY_N;
(*key_mapping)['O'] = KEY_O;
(*key_mapping)['P'] = KEY_P;
(*key_mapping)['Q'] = KEY_Q;
(*key_mapping)['R'] = KEY_R;
(*key_mapping)['S'] = KEY_S;
(*key_mapping)['T'] = KEY_T;
(*key_mapping)['U'] = KEY_U;
(*key_mapping)['V'] = KEY_V;
(*key_mapping)['W'] = KEY_W;
(*key_mapping)['X'] = KEY_X;
(*key_mapping)['Y'] = KEY_Y;
(*key_mapping)['Z'] = KEY_Z;
(*key_mapping)['['] = KEY_LEFTBRACE;
(*key_mapping)['\\'] = KEY_BACKSLASH;
(*key_mapping)[']'] = KEY_RIGHTBRACE;
(*key_mapping)['-'] = KEY_MINUS;
(*key_mapping)['_'] = KEY_MINUS;
(*key_mapping)['`'] = KEY_GRAVE;
(*key_mapping)['a'] = KEY_A;
(*key_mapping)['b'] = KEY_B;
(*key_mapping)['c'] = KEY_C;
(*key_mapping)['d'] = KEY_D;
(*key_mapping)['e'] = KEY_E;
(*key_mapping)['f'] = KEY_F;
(*key_mapping)['g'] = KEY_G;
(*key_mapping)['h'] = KEY_H;
(*key_mapping)['i'] = KEY_I;
(*key_mapping)['j'] = KEY_J;
(*key_mapping)['k'] = KEY_K;
(*key_mapping)['l'] = KEY_L;
(*key_mapping)['m'] = KEY_M;
(*key_mapping)['n'] = KEY_N;
(*key_mapping)['o'] = KEY_O;
(*key_mapping)['p'] = KEY_P;
(*key_mapping)['q'] = KEY_Q;
(*key_mapping)['r'] = KEY_R;
(*key_mapping)['s'] = KEY_S;
(*key_mapping)['t'] = KEY_T;
(*key_mapping)['u'] = KEY_U;
(*key_mapping)['v'] = KEY_V;
(*key_mapping)['w'] = KEY_W;
(*key_mapping)['x'] = KEY_X;
(*key_mapping)['y'] = KEY_Y;
(*key_mapping)['z'] = KEY_Z;
(*key_mapping)['{'] = KEY_LEFTBRACE;
(*key_mapping)['\\'] = KEY_BACKSLASH;
(*key_mapping)['|'] = KEY_BACKSLASH;
(*key_mapping)['}'] = KEY_RIGHTBRACE;
(*key_mapping)['~'] = KEY_GRAVE;
(*key_mapping)[cuttlefish::xk::F1] = KEY_F1;
(*key_mapping)[cuttlefish::xk::F2] = KEY_F2;
(*key_mapping)[cuttlefish::xk::F3] = KEY_F3;
(*key_mapping)[cuttlefish::xk::F4] = KEY_F4;
(*key_mapping)[cuttlefish::xk::F5] = KEY_F5;
(*key_mapping)[cuttlefish::xk::F6] = KEY_F6;
(*key_mapping)[cuttlefish::xk::F7] = KEY_F7;
(*key_mapping)[cuttlefish::xk::F8] = KEY_F8;
(*key_mapping)[cuttlefish::xk::F9] = KEY_F9;
(*key_mapping)[cuttlefish::xk::F10] = KEY_F10;
(*key_mapping)[cuttlefish::xk::F11] = KEY_F11;
(*key_mapping)[cuttlefish::xk::F12] = KEY_F12;
(*key_mapping)[cuttlefish::xk::F13] = KEY_F13;
(*key_mapping)[cuttlefish::xk::F14] = KEY_F14;
(*key_mapping)[cuttlefish::xk::F15] = KEY_F15;
(*key_mapping)[cuttlefish::xk::F16] = KEY_F16;
(*key_mapping)[cuttlefish::xk::F17] = KEY_F17;
(*key_mapping)[cuttlefish::xk::F18] = KEY_F18;
(*key_mapping)[cuttlefish::xk::F19] = KEY_F19;
(*key_mapping)[cuttlefish::xk::F20] = KEY_F20;
(*key_mapping)[cuttlefish::xk::F21] = KEY_F21;
(*key_mapping)[cuttlefish::xk::F22] = KEY_F22;
(*key_mapping)[cuttlefish::xk::F23] = KEY_F23;
(*key_mapping)[cuttlefish::xk::F24] = KEY_F24;
(*key_mapping)[cuttlefish::xk::Keypad0] = KEY_KP0;
(*key_mapping)[cuttlefish::xk::Keypad1] = KEY_KP1;
(*key_mapping)[cuttlefish::xk::Keypad2] = KEY_KP2;
(*key_mapping)[cuttlefish::xk::Keypad3] = KEY_KP3;
(*key_mapping)[cuttlefish::xk::Keypad4] = KEY_KP4;
(*key_mapping)[cuttlefish::xk::Keypad5] = KEY_KP5;
(*key_mapping)[cuttlefish::xk::Keypad6] = KEY_KP6;
(*key_mapping)[cuttlefish::xk::Keypad7] = KEY_KP7;
(*key_mapping)[cuttlefish::xk::Keypad8] = KEY_KP8;
(*key_mapping)[cuttlefish::xk::Keypad9] = KEY_KP9;
(*key_mapping)[cuttlefish::xk::KeypadMultiply] = KEY_KPASTERISK;
(*key_mapping)[cuttlefish::xk::KeypadSubtract] = KEY_KPMINUS;
(*key_mapping)[cuttlefish::xk::KeypadAdd] = KEY_KPPLUS;
(*key_mapping)[cuttlefish::xk::KeypadDecimal] = KEY_KPDOT;
(*key_mapping)[cuttlefish::xk::KeypadEnter] = KEY_KPENTER;
(*key_mapping)[cuttlefish::xk::KeypadDivide] = KEY_KPSLASH;
(*key_mapping)[cuttlefish::xk::KeypadEqual] = KEY_KPEQUAL;
(*key_mapping)[cuttlefish::xk::PlusMinus] = KEY_KPPLUSMINUS;
(*key_mapping)[cuttlefish::xk::SysReq] = KEY_SYSRQ;
(*key_mapping)[cuttlefish::xk::LineFeed] = KEY_LINEFEED;
(*key_mapping)[cuttlefish::xk::Home] = KEY_HOME;
(*key_mapping)[cuttlefish::xk::Up] = KEY_UP;
(*key_mapping)[cuttlefish::xk::PageUp] = KEY_PAGEUP;
(*key_mapping)[cuttlefish::xk::Left] = KEY_LEFT;
(*key_mapping)[cuttlefish::xk::Right] = KEY_RIGHT;
(*key_mapping)[cuttlefish::xk::End] = KEY_END;
(*key_mapping)[cuttlefish::xk::Down] = KEY_DOWN;
(*key_mapping)[cuttlefish::xk::PageDown] = KEY_PAGEDOWN;
(*key_mapping)[cuttlefish::xk::Insert] = KEY_INSERT;
(*key_mapping)[cuttlefish::xk::Delete] = KEY_DELETE;
(*key_mapping)[cuttlefish::xk::Pause] = KEY_PAUSE;
(*key_mapping)[cuttlefish::xk::KeypadSeparator] = KEY_KPCOMMA;
(*key_mapping)[cuttlefish::xk::Yen] = KEY_YEN;
(*key_mapping)[cuttlefish::xk::Cancel] = KEY_STOP;
(*key_mapping)[cuttlefish::xk::Redo] = KEY_AGAIN;
(*key_mapping)[cuttlefish::xk::Undo] = KEY_UNDO;
(*key_mapping)[cuttlefish::xk::Find] = KEY_FIND;
(*key_mapping)[cuttlefish::xk::Print] = KEY_PRINT;
(*key_mapping)[cuttlefish::xk::VolumeDown] = KEY_VOLUMEDOWN;
(*key_mapping)[cuttlefish::xk::Mute] = KEY_MUTE;
(*key_mapping)[cuttlefish::xk::VolumeUp] = KEY_VOLUMEUP;
(*key_mapping)[cuttlefish::xk::Menu] = KEY_MENU;
(*key_mapping)[cuttlefish::xk::VNCMenu] = KEY_MENU;
}
void InitInputEvent(struct input_event* evt, uint16_t type, uint16_t code,
int32_t value) {
evt->type = type;
evt->code = code;
evt->value = value;
}
} // namespace
class SocketVirtualInputs : public VirtualInputs {
public:
SocketVirtualInputs()
: client_connector_([this]() { ClientConnectorLoop(); }) {}
void GenerateKeyPressEvent(int key_code, bool down) override {
struct input_event events[2];
InitInputEvent(&events[0], EV_KEY, keymapping_[key_code], down);
InitInputEvent(&events[1], EV_SYN, 0, 0);
SendEvents(keyboard_socket_, events);
}
void PressPowerButton(bool down) override {
struct input_event events[2];
InitInputEvent(&events[0], EV_KEY, KEY_POWER, down);
InitInputEvent(&events[1], EV_SYN, 0, 0);
SendEvents(keyboard_socket_, events);
}
void HandlePointerEvent(bool touch_down, int x, int y) override {
// TODO(b/124121375): Use multitouch when available
struct input_event events[4];
InitInputEvent(&events[0], EV_ABS, ABS_X, x);
InitInputEvent(&events[1], EV_ABS, ABS_Y, y);
InitInputEvent(&events[2], EV_KEY, BTN_TOUCH, touch_down);
InitInputEvent(&events[3], EV_SYN, 0, 0);
SendEvents(touch_socket_, events);
}
private:
template<size_t num_events>
void SendEvents(cuttlefish::SharedFD socket, struct input_event (&event_buffer)[num_events]) {
std::lock_guard<std::mutex> lock(socket_mutex_);
if (!socket->IsOpen()) {
// This is unlikely as it would only happen between the start of the vnc
// server and the connection of the VMM to the socket.
// If it happens, just drop the events as the VM is not yet ready to
// handle it.
return;
}
if (FLAGS_write_virtio_input) {
struct virtio_input_event virtio_events[num_events];
for (size_t i = 0; i < num_events; i++) {
virtio_events[i] = (struct virtio_input_event) {
.type = event_buffer[i].type,
.code = event_buffer[i].code,
.value = event_buffer[i].value,
};
}
auto ret = socket->Write(virtio_events, sizeof(virtio_events));
if (ret < 0) {
LOG(ERROR) << "Error sending input events: " << socket->StrError();
}
} else {
auto ret = socket->Write(event_buffer, sizeof(event_buffer));
if (ret < 0) {
LOG(ERROR) << "Error sending input events: " << socket->StrError();
}
}
}
void ClientConnectorLoop() {
auto touch_fd =
std::stoi(android::base::Split(FLAGS_touch_fds, ",").front());
auto touch_server = cuttlefish::SharedFD::Dup(touch_fd);
close(touch_fd);
auto keyboard_server = cuttlefish::SharedFD::Dup(FLAGS_keyboard_fd);
close(FLAGS_keyboard_fd);
FLAGS_keyboard_fd = -1;
LOG(DEBUG) << "Input socket host accepting connections...";
while (1) {
cuttlefish::SharedFDSet read_set;
read_set.Set(touch_server);
read_set.Set(keyboard_server);
cuttlefish::Select(&read_set, nullptr, nullptr, nullptr);
{
std::lock_guard<std::mutex> lock(socket_mutex_);
if (read_set.IsSet(touch_server)) {
touch_socket_ = cuttlefish::SharedFD::Accept(*touch_server);
LOG(DEBUG) << "connected to touch";
}
if (read_set.IsSet(keyboard_server)) {
keyboard_socket_ = cuttlefish::SharedFD::Accept(*keyboard_server);
LOG(DEBUG) << "connected to keyboard";
}
}
}
}
cuttlefish::SharedFD touch_socket_;
cuttlefish::SharedFD keyboard_socket_;
std::thread client_connector_;
std::mutex socket_mutex_;
};
VirtualInputs::VirtualInputs() { AddKeyMappings(&keymapping_); }
/**
* Depending on the host mode (e.g. android, confirmation ui(tee), etc)
* deliver the inputs to the right input implementation
* e.g. ConfUI's input or regular socket based input
*/
class VirtualInputDemux : public VirtualInputs {
public:
VirtualInputDemux(cuttlefish::confui::HostVirtualInput& confui_input)
: confui_input_{confui_input} {}
virtual ~VirtualInputDemux() = default;
virtual void GenerateKeyPressEvent(int code, bool down) override;
virtual void PressPowerButton(bool down) override;
virtual void HandlePointerEvent(bool touch_down, int x, int y) override;
private:
SocketVirtualInputs socket_virtual_input_;
cuttlefish::confui::HostVirtualInput& confui_input_;
};
void VirtualInputDemux::GenerateKeyPressEvent(int code, bool down) {
// confui input is active only in the confirmation UI
// also, socket virtual input should be inactive in the confirmation
// UI session
if (confui_input_.IsConfUiActive()) {
if (code == cuttlefish::xk::Menu) {
// release menu button in confirmation UI means for now cancel
confui_input_.PressCancelButton(down);
}
ConfUiLog(DEBUG) << "the key" << code << "ignored."
<< "currently confirmation UI handles"
<< "menu and power only.";
return;
}
socket_virtual_input_.GenerateKeyPressEvent(code, down);
}
void VirtualInputDemux::PressPowerButton(bool down) {
if (confui_input_.IsConfUiActive()) {
confui_input_.PressConfirmButton(down);
return;
}
socket_virtual_input_.PressPowerButton(down);
}
void VirtualInputDemux::HandlePointerEvent(bool touch_down, int x, int y) {
if (confui_input_.IsConfUiActive()) {
ConfUiLog(DEBUG) << "currently confirmation UI ignores pointer events at ("
<< x << ", " << y << ")";
return;
}
socket_virtual_input_.HandlePointerEvent(touch_down, x, y);
}
std::shared_ptr<VirtualInputs> VirtualInputs::Get(
cuttlefish::confui::HostVirtualInput& confui_input) {
return std::make_shared<VirtualInputDemux>(confui_input);
}