4 * Copyright (c) 2003-2008 Fabrice Bellard
6 * Permission is hereby granted, free of charge, to any person obtaining a copy
7 * of this software and associated documentation files (the "Software"), to deal
8 * in the Software without restriction, including without limitation the rights
9 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
10 * copies of the Software, and to permit persons to whom the Software is
11 * furnished to do so, subject to the following conditions:
13 * The above copyright notice and this permission notice shall be included in
14 * all copies or substantial portions of the Software.
16 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
19 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
21 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
25 #include "sysemu/sysemu.h"
26 #include "monitor/monitor.h"
27 #include "ui/console.h"
28 #include "qapi/error.h"
29 #include "qmp-commands.h"
30 #include "qapi-types.h"
31 #include "ui/keymaps.h"
34 struct QEMUPutMouseEntry
{
35 QEMUPutMouseEvent
*qemu_put_mouse_event
;
36 void *qemu_put_mouse_event_opaque
;
37 int qemu_put_mouse_event_absolute
;
41 QemuInputHandlerState
*s
;
42 int axis
[INPUT_AXIS_MAX
];
46 struct QEMUPutKbdEntry
{
47 QEMUPutKBDEvent
*put_kbd
;
49 QemuInputHandlerState
*s
;
52 struct QEMUPutLEDEntry
{
53 QEMUPutLEDEvent
*put_led
;
55 QTAILQ_ENTRY(QEMUPutLEDEntry
) next
;
58 static QTAILQ_HEAD(, QEMUPutLEDEntry
) led_handlers
=
59 QTAILQ_HEAD_INITIALIZER(led_handlers
);
60 static QTAILQ_HEAD(, QEMUPutMouseEntry
) mouse_handlers
=
61 QTAILQ_HEAD_INITIALIZER(mouse_handlers
);
63 int index_from_key(const char *key
)
67 for (i
= 0; QKeyCode_lookup
[i
] != NULL
; i
++) {
68 if (!strcmp(key
, QKeyCode_lookup
[i
])) {
73 /* Return Q_KEY_CODE_MAX if the key is invalid */
77 static KeyValue
**keyvalues
;
78 static int keyvalues_size
;
79 static QEMUTimer
*key_timer
;
81 static void free_keyvalues(void)
88 static void release_keys(void *opaque
)
90 while (keyvalues_size
> 0) {
91 qemu_input_event_send_key(NULL
, keyvalues
[--keyvalues_size
],
98 static KeyValue
*copy_key_value(KeyValue
*src
)
100 KeyValue
*dst
= g_new(KeyValue
, 1);
101 memcpy(dst
, src
, sizeof(*src
));
105 void qmp_send_key(KeyValueList
*keys
, bool has_hold_time
, int64_t hold_time
,
111 key_timer
= timer_new_ns(QEMU_CLOCK_VIRTUAL
, release_keys
, NULL
);
114 if (keyvalues
!= NULL
) {
115 timer_del(key_timer
);
119 if (!has_hold_time
) {
123 for (p
= keys
; p
!= NULL
; p
= p
->next
) {
124 qemu_input_event_send_key(NULL
, copy_key_value(p
->value
), true);
126 keyvalues
= g_realloc(keyvalues
, sizeof(KeyValue
*) *
127 (keyvalues_size
+ 1));
128 keyvalues
[keyvalues_size
++] = copy_key_value(p
->value
);
131 /* delayed key up events */
132 timer_mod(key_timer
, qemu_clock_get_ns(QEMU_CLOCK_VIRTUAL
) +
133 muldiv64(get_ticks_per_sec(), hold_time
, 1000));
136 static void legacy_kbd_event(DeviceState
*dev
, QemuConsole
*src
,
139 QEMUPutKbdEntry
*entry
= (QEMUPutKbdEntry
*)dev
;
140 int scancodes
[3], i
, count
;
142 if (!entry
|| !entry
->put_kbd
) {
145 count
= qemu_input_key_value_to_scancode(evt
->key
->key
,
148 for (i
= 0; i
< count
; i
++) {
149 entry
->put_kbd(entry
->opaque
, scancodes
[i
]);
153 static QemuInputHandler legacy_kbd_handler
= {
154 .name
= "legacy-kbd",
155 .mask
= INPUT_EVENT_MASK_KEY
,
156 .event
= legacy_kbd_event
,
159 QEMUPutKbdEntry
*qemu_add_kbd_event_handler(QEMUPutKBDEvent
*func
, void *opaque
)
161 QEMUPutKbdEntry
*entry
;
163 entry
= g_new0(QEMUPutKbdEntry
, 1);
164 entry
->put_kbd
= func
;
165 entry
->opaque
= opaque
;
166 entry
->s
= qemu_input_handler_register((DeviceState
*)entry
,
167 &legacy_kbd_handler
);
168 qemu_input_handler_activate(entry
->s
);
172 void qemu_remove_kbd_event_handler(QEMUPutKbdEntry
*entry
)
174 qemu_input_handler_unregister(entry
->s
);
178 static void legacy_mouse_event(DeviceState
*dev
, QemuConsole
*src
,
181 static const int bmap
[INPUT_BUTTON_MAX
] = {
182 [INPUT_BUTTON_LEFT
] = MOUSE_EVENT_LBUTTON
,
183 [INPUT_BUTTON_MIDDLE
] = MOUSE_EVENT_MBUTTON
,
184 [INPUT_BUTTON_RIGHT
] = MOUSE_EVENT_RBUTTON
,
186 QEMUPutMouseEntry
*s
= (QEMUPutMouseEntry
*)dev
;
189 case INPUT_EVENT_KIND_BTN
:
190 if (evt
->btn
->down
) {
191 s
->buttons
|= bmap
[evt
->btn
->button
];
193 s
->buttons
&= ~bmap
[evt
->btn
->button
];
195 if (evt
->btn
->down
&& evt
->btn
->button
== INPUT_BUTTON_WHEEL_UP
) {
196 s
->qemu_put_mouse_event(s
->qemu_put_mouse_event_opaque
,
197 s
->axis
[INPUT_AXIS_X
],
198 s
->axis
[INPUT_AXIS_Y
],
202 if (evt
->btn
->down
&& evt
->btn
->button
== INPUT_BUTTON_WHEEL_DOWN
) {
203 s
->qemu_put_mouse_event(s
->qemu_put_mouse_event_opaque
,
204 s
->axis
[INPUT_AXIS_X
],
205 s
->axis
[INPUT_AXIS_Y
],
210 case INPUT_EVENT_KIND_ABS
:
211 s
->axis
[evt
->abs
->axis
] = evt
->abs
->value
;
213 case INPUT_EVENT_KIND_REL
:
214 s
->axis
[evt
->rel
->axis
] += evt
->rel
->value
;
221 static void legacy_mouse_sync(DeviceState
*dev
)
223 QEMUPutMouseEntry
*s
= (QEMUPutMouseEntry
*)dev
;
225 s
->qemu_put_mouse_event(s
->qemu_put_mouse_event_opaque
,
226 s
->axis
[INPUT_AXIS_X
],
227 s
->axis
[INPUT_AXIS_Y
],
231 if (!s
->qemu_put_mouse_event_absolute
) {
232 s
->axis
[INPUT_AXIS_X
] = 0;
233 s
->axis
[INPUT_AXIS_Y
] = 0;
237 QEMUPutMouseEntry
*qemu_add_mouse_event_handler(QEMUPutMouseEvent
*func
,
238 void *opaque
, int absolute
,
241 QEMUPutMouseEntry
*s
;
243 s
= g_malloc0(sizeof(QEMUPutMouseEntry
));
245 s
->qemu_put_mouse_event
= func
;
246 s
->qemu_put_mouse_event_opaque
= opaque
;
247 s
->qemu_put_mouse_event_absolute
= absolute
;
250 s
->h
.mask
= INPUT_EVENT_MASK_BTN
|
251 (absolute
? INPUT_EVENT_MASK_ABS
: INPUT_EVENT_MASK_REL
);
252 s
->h
.event
= legacy_mouse_event
;
253 s
->h
.sync
= legacy_mouse_sync
;
254 s
->s
= qemu_input_handler_register((DeviceState
*)s
,
260 void qemu_activate_mouse_event_handler(QEMUPutMouseEntry
*entry
)
262 qemu_input_handler_activate(entry
->s
);
265 void qemu_remove_mouse_event_handler(QEMUPutMouseEntry
*entry
)
267 qemu_input_handler_unregister(entry
->s
);
272 QEMUPutLEDEntry
*qemu_add_led_event_handler(QEMUPutLEDEvent
*func
,
277 s
= g_malloc0(sizeof(QEMUPutLEDEntry
));
281 QTAILQ_INSERT_TAIL(&led_handlers
, s
, next
);
285 void qemu_remove_led_event_handler(QEMUPutLEDEntry
*entry
)
289 QTAILQ_REMOVE(&led_handlers
, entry
, next
);
293 void kbd_put_ledstate(int ledstate
)
295 QEMUPutLEDEntry
*cursor
;
297 QTAILQ_FOREACH(cursor
, &led_handlers
, next
) {
298 cursor
->put_led(cursor
->opaque
, ledstate
);