| /* |
| * 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); |
| } |