1864401c2SGerd Hoffmann /* 2864401c2SGerd Hoffmann * Copyright (C) 2010 Red Hat, Inc. 3864401c2SGerd Hoffmann * 4864401c2SGerd Hoffmann * This program is free software; you can redistribute it and/or 5864401c2SGerd Hoffmann * modify it under the terms of the GNU General Public License as 6864401c2SGerd Hoffmann * published by the Free Software Foundation; either version 2 or 7864401c2SGerd Hoffmann * (at your option) version 3 of the License. 8864401c2SGerd Hoffmann * 9864401c2SGerd Hoffmann * This program is distributed in the hope that it will be useful, 10864401c2SGerd Hoffmann * but WITHOUT ANY WARRANTY; without even the implied warranty of 11864401c2SGerd Hoffmann * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 12864401c2SGerd Hoffmann * GNU General Public License for more details. 13864401c2SGerd Hoffmann * 14864401c2SGerd Hoffmann * You should have received a copy of the GNU General Public License 15864401c2SGerd Hoffmann * along with this program; if not, see <http://www.gnu.org/licenses/>. 16864401c2SGerd Hoffmann */ 17864401c2SGerd Hoffmann 18864401c2SGerd Hoffmann #include <stdlib.h> 19864401c2SGerd Hoffmann #include <stdio.h> 20864401c2SGerd Hoffmann #include <string.h> 21864401c2SGerd Hoffmann 22864401c2SGerd Hoffmann #include <spice.h> 23864401c2SGerd Hoffmann #include <spice/enums.h> 24864401c2SGerd Hoffmann 25864401c2SGerd Hoffmann #include "qemu-common.h" 26864401c2SGerd Hoffmann #include "qemu-spice.h" 27864401c2SGerd Hoffmann #include "console.h" 28864401c2SGerd Hoffmann 29864401c2SGerd Hoffmann /* keyboard bits */ 30864401c2SGerd Hoffmann 31864401c2SGerd Hoffmann typedef struct QemuSpiceKbd { 32864401c2SGerd Hoffmann SpiceKbdInstance sin; 33864401c2SGerd Hoffmann int ledstate; 34864401c2SGerd Hoffmann } QemuSpiceKbd; 35864401c2SGerd Hoffmann 36864401c2SGerd Hoffmann static void kbd_push_key(SpiceKbdInstance *sin, uint8_t frag); 37864401c2SGerd Hoffmann static uint8_t kbd_get_leds(SpiceKbdInstance *sin); 38864401c2SGerd Hoffmann static void kbd_leds(void *opaque, int l); 39864401c2SGerd Hoffmann 40864401c2SGerd Hoffmann static const SpiceKbdInterface kbd_interface = { 41864401c2SGerd Hoffmann .base.type = SPICE_INTERFACE_KEYBOARD, 42864401c2SGerd Hoffmann .base.description = "qemu keyboard", 43864401c2SGerd Hoffmann .base.major_version = SPICE_INTERFACE_KEYBOARD_MAJOR, 44864401c2SGerd Hoffmann .base.minor_version = SPICE_INTERFACE_KEYBOARD_MINOR, 45864401c2SGerd Hoffmann .push_scan_freg = kbd_push_key, 46864401c2SGerd Hoffmann .get_leds = kbd_get_leds, 47864401c2SGerd Hoffmann }; 48864401c2SGerd Hoffmann 49864401c2SGerd Hoffmann static void kbd_push_key(SpiceKbdInstance *sin, uint8_t frag) 50864401c2SGerd Hoffmann { 51864401c2SGerd Hoffmann kbd_put_keycode(frag); 52864401c2SGerd Hoffmann } 53864401c2SGerd Hoffmann 54864401c2SGerd Hoffmann static uint8_t kbd_get_leds(SpiceKbdInstance *sin) 55864401c2SGerd Hoffmann { 56864401c2SGerd Hoffmann QemuSpiceKbd *kbd = container_of(sin, QemuSpiceKbd, sin); 57864401c2SGerd Hoffmann return kbd->ledstate; 58864401c2SGerd Hoffmann } 59864401c2SGerd Hoffmann 60864401c2SGerd Hoffmann static void kbd_leds(void *opaque, int ledstate) 61864401c2SGerd Hoffmann { 62864401c2SGerd Hoffmann QemuSpiceKbd *kbd = opaque; 63864401c2SGerd Hoffmann 64864401c2SGerd Hoffmann kbd->ledstate = 0; 65864401c2SGerd Hoffmann if (ledstate & QEMU_SCROLL_LOCK_LED) { 66864401c2SGerd Hoffmann kbd->ledstate |= SPICE_KEYBOARD_MODIFIER_FLAGS_SCROLL_LOCK; 67864401c2SGerd Hoffmann } 68864401c2SGerd Hoffmann if (ledstate & QEMU_NUM_LOCK_LED) { 69864401c2SGerd Hoffmann kbd->ledstate |= SPICE_KEYBOARD_MODIFIER_FLAGS_NUM_LOCK; 70864401c2SGerd Hoffmann } 71864401c2SGerd Hoffmann if (ledstate & QEMU_CAPS_LOCK_LED) { 72864401c2SGerd Hoffmann kbd->ledstate |= SPICE_KEYBOARD_MODIFIER_FLAGS_CAPS_LOCK; 73864401c2SGerd Hoffmann } 74864401c2SGerd Hoffmann spice_server_kbd_leds(&kbd->sin, ledstate); 75864401c2SGerd Hoffmann } 76864401c2SGerd Hoffmann 77*78dd9ac1SGerd Hoffmann /* mouse bits */ 78*78dd9ac1SGerd Hoffmann 79*78dd9ac1SGerd Hoffmann typedef struct QemuSpiceMouse { 80*78dd9ac1SGerd Hoffmann SpiceMouseInstance sin; 81*78dd9ac1SGerd Hoffmann } QemuSpiceMouse; 82*78dd9ac1SGerd Hoffmann 83*78dd9ac1SGerd Hoffmann static int map_buttons(int spice_buttons) 84*78dd9ac1SGerd Hoffmann { 85*78dd9ac1SGerd Hoffmann int qemu_buttons = 0; 86*78dd9ac1SGerd Hoffmann 87*78dd9ac1SGerd Hoffmann /* 88*78dd9ac1SGerd Hoffmann * Note: SPICE_MOUSE_BUTTON_* specifies the wire protocol but this 89*78dd9ac1SGerd Hoffmann * isn't what we get passed in via interface callbacks for the 90*78dd9ac1SGerd Hoffmann * middle and right button ... 91*78dd9ac1SGerd Hoffmann */ 92*78dd9ac1SGerd Hoffmann if (spice_buttons & SPICE_MOUSE_BUTTON_MASK_LEFT) { 93*78dd9ac1SGerd Hoffmann qemu_buttons |= MOUSE_EVENT_LBUTTON; 94*78dd9ac1SGerd Hoffmann } 95*78dd9ac1SGerd Hoffmann if (spice_buttons & 0x04 /* SPICE_MOUSE_BUTTON_MASK_MIDDLE */) { 96*78dd9ac1SGerd Hoffmann qemu_buttons |= MOUSE_EVENT_MBUTTON; 97*78dd9ac1SGerd Hoffmann } 98*78dd9ac1SGerd Hoffmann if (spice_buttons & 0x02 /* SPICE_MOUSE_BUTTON_MASK_RIGHT */) { 99*78dd9ac1SGerd Hoffmann qemu_buttons |= MOUSE_EVENT_RBUTTON; 100*78dd9ac1SGerd Hoffmann } 101*78dd9ac1SGerd Hoffmann return qemu_buttons; 102*78dd9ac1SGerd Hoffmann } 103*78dd9ac1SGerd Hoffmann 104*78dd9ac1SGerd Hoffmann static void mouse_motion(SpiceMouseInstance *sin, int dx, int dy, int dz, 105*78dd9ac1SGerd Hoffmann uint32_t buttons_state) 106*78dd9ac1SGerd Hoffmann { 107*78dd9ac1SGerd Hoffmann kbd_mouse_event(dx, dy, dz, map_buttons(buttons_state)); 108*78dd9ac1SGerd Hoffmann } 109*78dd9ac1SGerd Hoffmann 110*78dd9ac1SGerd Hoffmann static void mouse_buttons(SpiceMouseInstance *sin, uint32_t buttons_state) 111*78dd9ac1SGerd Hoffmann { 112*78dd9ac1SGerd Hoffmann kbd_mouse_event(0, 0, 0, map_buttons(buttons_state)); 113*78dd9ac1SGerd Hoffmann } 114*78dd9ac1SGerd Hoffmann 115*78dd9ac1SGerd Hoffmann static const SpiceMouseInterface mouse_interface = { 116*78dd9ac1SGerd Hoffmann .base.type = SPICE_INTERFACE_MOUSE, 117*78dd9ac1SGerd Hoffmann .base.description = "mouse", 118*78dd9ac1SGerd Hoffmann .base.major_version = SPICE_INTERFACE_MOUSE_MAJOR, 119*78dd9ac1SGerd Hoffmann .base.minor_version = SPICE_INTERFACE_MOUSE_MINOR, 120*78dd9ac1SGerd Hoffmann .motion = mouse_motion, 121*78dd9ac1SGerd Hoffmann .buttons = mouse_buttons, 122*78dd9ac1SGerd Hoffmann }; 123*78dd9ac1SGerd Hoffmann 124864401c2SGerd Hoffmann void qemu_spice_input_init(void) 125864401c2SGerd Hoffmann { 126864401c2SGerd Hoffmann QemuSpiceKbd *kbd; 127*78dd9ac1SGerd Hoffmann QemuSpiceMouse *mouse; 128864401c2SGerd Hoffmann 129864401c2SGerd Hoffmann kbd = qemu_mallocz(sizeof(*kbd)); 130864401c2SGerd Hoffmann kbd->sin.base.sif = &kbd_interface.base; 131864401c2SGerd Hoffmann qemu_spice_add_interface(&kbd->sin.base); 132864401c2SGerd Hoffmann qemu_add_led_event_handler(kbd_leds, kbd); 133*78dd9ac1SGerd Hoffmann 134*78dd9ac1SGerd Hoffmann mouse = qemu_mallocz(sizeof(*mouse)); 135*78dd9ac1SGerd Hoffmann mouse->sin.base.sif = &mouse_interface.base; 136*78dd9ac1SGerd Hoffmann qemu_spice_add_interface(&mouse->sin.base); 137864401c2SGerd Hoffmann } 138