xref: /openbmc/qemu/hw/input/virtio-input-hid.c (revision 8a49b300)
1 /*
2  * This work is licensed under the terms of the GNU GPL, version 2 or
3  * (at your option) any later version.  See the COPYING file in the
4  * top-level directory.
5  */
6 
7 #include "qemu/osdep.h"
8 #include "qemu/iov.h"
9 #include "qemu/module.h"
10 
11 #include "hw/virtio/virtio.h"
12 #include "hw/qdev-properties.h"
13 #include "hw/virtio/virtio-input.h"
14 
15 #undef CONFIG_CURSES
16 #include "ui/console.h"
17 
18 #include "standard-headers/linux/input.h"
19 
20 #define VIRTIO_ID_NAME_KEYBOARD "QEMU Virtio Keyboard"
21 #define VIRTIO_ID_NAME_MOUSE    "QEMU Virtio Mouse"
22 #define VIRTIO_ID_NAME_TABLET   "QEMU Virtio Tablet"
23 
24 /* ----------------------------------------------------------------- */
25 
26 static const unsigned short keymap_button[INPUT_BUTTON__MAX] = {
27     [INPUT_BUTTON_LEFT]              = BTN_LEFT,
28     [INPUT_BUTTON_RIGHT]             = BTN_RIGHT,
29     [INPUT_BUTTON_MIDDLE]            = BTN_MIDDLE,
30     [INPUT_BUTTON_WHEEL_UP]          = BTN_GEAR_UP,
31     [INPUT_BUTTON_WHEEL_DOWN]        = BTN_GEAR_DOWN,
32     [INPUT_BUTTON_SIDE]              = BTN_SIDE,
33     [INPUT_BUTTON_EXTRA]             = BTN_EXTRA,
34 };
35 
36 static const unsigned short axismap_rel[INPUT_AXIS__MAX] = {
37     [INPUT_AXIS_X]                   = REL_X,
38     [INPUT_AXIS_Y]                   = REL_Y,
39 };
40 
41 static const unsigned short axismap_abs[INPUT_AXIS__MAX] = {
42     [INPUT_AXIS_X]                   = ABS_X,
43     [INPUT_AXIS_Y]                   = ABS_Y,
44 };
45 
46 /* ----------------------------------------------------------------- */
47 
48 static void virtio_input_key_config(VirtIOInput *vinput,
49                                     const unsigned short *keymap,
50                                     size_t mapsize)
51 {
52     virtio_input_config keys;
53     int i, bit, byte, bmax = 0;
54 
55     memset(&keys, 0, sizeof(keys));
56     for (i = 0; i < mapsize; i++) {
57         bit = keymap[i];
58         if (!bit) {
59             continue;
60         }
61         byte = bit / 8;
62         bit  = bit % 8;
63         keys.u.bitmap[byte] |= (1 << bit);
64         if (bmax < byte+1) {
65             bmax = byte+1;
66         }
67     }
68     keys.select = VIRTIO_INPUT_CFG_EV_BITS;
69     keys.subsel = EV_KEY;
70     keys.size   = bmax;
71     virtio_input_add_config(vinput, &keys);
72 }
73 
74 static void virtio_input_handle_event(DeviceState *dev, QemuConsole *src,
75                                       InputEvent *evt)
76 {
77     VirtIOInputHID *vhid = VIRTIO_INPUT_HID(dev);
78     VirtIOInput *vinput = VIRTIO_INPUT(dev);
79     virtio_input_event event;
80     int qcode;
81     InputKeyEvent *key;
82     InputMoveEvent *move;
83     InputBtnEvent *btn;
84 
85     switch (evt->type) {
86     case INPUT_EVENT_KIND_KEY:
87         key = evt->u.key.data;
88         qcode = qemu_input_key_value_to_qcode(key->key);
89         if (qcode < qemu_input_map_qcode_to_linux_len &&
90             qemu_input_map_qcode_to_linux[qcode]) {
91             event.type  = cpu_to_le16(EV_KEY);
92             event.code  = cpu_to_le16(qemu_input_map_qcode_to_linux[qcode]);
93             event.value = cpu_to_le32(key->down ? 1 : 0);
94             virtio_input_send(vinput, &event);
95         } else {
96             if (key->down) {
97                 fprintf(stderr, "%s: unmapped key: %d [%s]\n", __func__,
98                         qcode, QKeyCode_str(qcode));
99             }
100         }
101         break;
102     case INPUT_EVENT_KIND_BTN:
103         btn = evt->u.btn.data;
104         if (vhid->wheel_axis &&
105             (btn->button == INPUT_BUTTON_WHEEL_UP ||
106              btn->button == INPUT_BUTTON_WHEEL_DOWN) &&
107             btn->down) {
108             event.type  = cpu_to_le16(EV_REL);
109             event.code  = cpu_to_le16(REL_WHEEL);
110             event.value = cpu_to_le32(btn->button == INPUT_BUTTON_WHEEL_UP
111                                       ? 1 : -1);
112             virtio_input_send(vinput, &event);
113         } else if (keymap_button[btn->button]) {
114             event.type  = cpu_to_le16(EV_KEY);
115             event.code  = cpu_to_le16(keymap_button[btn->button]);
116             event.value = cpu_to_le32(btn->down ? 1 : 0);
117             virtio_input_send(vinput, &event);
118         } else {
119             if (btn->down) {
120                 fprintf(stderr, "%s: unmapped button: %d [%s]\n", __func__,
121                         btn->button,
122                         InputButton_str(btn->button));
123             }
124         }
125         break;
126     case INPUT_EVENT_KIND_REL:
127         move = evt->u.rel.data;
128         event.type  = cpu_to_le16(EV_REL);
129         event.code  = cpu_to_le16(axismap_rel[move->axis]);
130         event.value = cpu_to_le32(move->value);
131         virtio_input_send(vinput, &event);
132         break;
133     case INPUT_EVENT_KIND_ABS:
134         move = evt->u.abs.data;
135         event.type  = cpu_to_le16(EV_ABS);
136         event.code  = cpu_to_le16(axismap_abs[move->axis]);
137         event.value = cpu_to_le32(move->value);
138         virtio_input_send(vinput, &event);
139         break;
140     default:
141         /* keep gcc happy */
142         break;
143     }
144 }
145 
146 static void virtio_input_handle_sync(DeviceState *dev)
147 {
148     VirtIOInput *vinput = VIRTIO_INPUT(dev);
149     virtio_input_event event = {
150         .type  = cpu_to_le16(EV_SYN),
151         .code  = cpu_to_le16(SYN_REPORT),
152         .value = 0,
153     };
154 
155     virtio_input_send(vinput, &event);
156 }
157 
158 static void virtio_input_hid_realize(DeviceState *dev, Error **errp)
159 {
160     VirtIOInputHID *vhid = VIRTIO_INPUT_HID(dev);
161 
162     vhid->hs = qemu_input_handler_register(dev, vhid->handler);
163     if (vhid->display && vhid->hs) {
164         qemu_input_handler_bind(vhid->hs, vhid->display, vhid->head, NULL);
165     }
166 }
167 
168 static void virtio_input_hid_unrealize(DeviceState *dev)
169 {
170     VirtIOInputHID *vhid = VIRTIO_INPUT_HID(dev);
171     qemu_input_handler_unregister(vhid->hs);
172 }
173 
174 static void virtio_input_hid_change_active(VirtIOInput *vinput)
175 {
176     VirtIOInputHID *vhid = VIRTIO_INPUT_HID(vinput);
177 
178     if (vinput->active) {
179         qemu_input_handler_activate(vhid->hs);
180     } else {
181         qemu_input_handler_deactivate(vhid->hs);
182     }
183 }
184 
185 static void virtio_input_hid_handle_status(VirtIOInput *vinput,
186                                            virtio_input_event *event)
187 {
188     VirtIOInputHID *vhid = VIRTIO_INPUT_HID(vinput);
189     int ledbit = 0;
190 
191     switch (le16_to_cpu(event->type)) {
192     case EV_LED:
193         if (event->code == LED_NUML) {
194             ledbit = QEMU_NUM_LOCK_LED;
195         } else if (event->code == LED_CAPSL) {
196             ledbit = QEMU_CAPS_LOCK_LED;
197         } else if (event->code == LED_SCROLLL) {
198             ledbit = QEMU_SCROLL_LOCK_LED;
199         }
200         if (event->value) {
201             vhid->ledstate |= ledbit;
202         } else {
203             vhid->ledstate &= ~ledbit;
204         }
205         kbd_put_ledstate(vhid->ledstate);
206         break;
207     default:
208         fprintf(stderr, "%s: unknown type %d\n", __func__,
209                 le16_to_cpu(event->type));
210         break;
211     }
212 }
213 
214 static Property virtio_input_hid_properties[] = {
215     DEFINE_PROP_STRING("display", VirtIOInputHID, display),
216     DEFINE_PROP_UINT32("head", VirtIOInputHID, head, 0),
217     DEFINE_PROP_END_OF_LIST(),
218 };
219 
220 static void virtio_input_hid_class_init(ObjectClass *klass, void *data)
221 {
222     DeviceClass *dc = DEVICE_CLASS(klass);
223     VirtIOInputClass *vic = VIRTIO_INPUT_CLASS(klass);
224 
225     device_class_set_props(dc, virtio_input_hid_properties);
226     vic->realize       = virtio_input_hid_realize;
227     vic->unrealize     = virtio_input_hid_unrealize;
228     vic->change_active = virtio_input_hid_change_active;
229     vic->handle_status = virtio_input_hid_handle_status;
230 }
231 
232 static const TypeInfo virtio_input_hid_info = {
233     .name          = TYPE_VIRTIO_INPUT_HID,
234     .parent        = TYPE_VIRTIO_INPUT,
235     .instance_size = sizeof(VirtIOInputHID),
236     .class_init    = virtio_input_hid_class_init,
237     .abstract      = true,
238 };
239 
240 /* ----------------------------------------------------------------- */
241 
242 static QemuInputHandler virtio_keyboard_handler = {
243     .name  = VIRTIO_ID_NAME_KEYBOARD,
244     .mask  = INPUT_EVENT_MASK_KEY,
245     .event = virtio_input_handle_event,
246     .sync  = virtio_input_handle_sync,
247 };
248 
249 static struct virtio_input_config virtio_keyboard_config[] = {
250     {
251         .select    = VIRTIO_INPUT_CFG_ID_NAME,
252         .size      = sizeof(VIRTIO_ID_NAME_KEYBOARD),
253         .u.string  = VIRTIO_ID_NAME_KEYBOARD,
254     },{
255         .select    = VIRTIO_INPUT_CFG_ID_DEVIDS,
256         .size      = sizeof(struct virtio_input_devids),
257         .u.ids     = {
258             .bustype = const_le16(BUS_VIRTUAL),
259             .vendor  = const_le16(0x0627), /* same we use for usb hid devices */
260             .product = const_le16(0x0001),
261             .version = const_le16(0x0001),
262         },
263     },{
264         .select    = VIRTIO_INPUT_CFG_EV_BITS,
265         .subsel    = EV_REP,
266         .size      = 1,
267     },{
268         .select    = VIRTIO_INPUT_CFG_EV_BITS,
269         .subsel    = EV_LED,
270         .size      = 1,
271         .u.bitmap  = {
272             (1 << LED_NUML) | (1 << LED_CAPSL) | (1 << LED_SCROLLL),
273         },
274     },
275     { /* end of list */ },
276 };
277 
278 static void virtio_keyboard_init(Object *obj)
279 {
280     VirtIOInputHID *vhid = VIRTIO_INPUT_HID(obj);
281     VirtIOInput *vinput = VIRTIO_INPUT(obj);
282 
283     vhid->handler = &virtio_keyboard_handler;
284     virtio_input_init_config(vinput, virtio_keyboard_config);
285     virtio_input_key_config(vinput, qemu_input_map_qcode_to_linux,
286                             qemu_input_map_qcode_to_linux_len);
287 }
288 
289 static const TypeInfo virtio_keyboard_info = {
290     .name          = TYPE_VIRTIO_KEYBOARD,
291     .parent        = TYPE_VIRTIO_INPUT_HID,
292     .instance_size = sizeof(VirtIOInputHID),
293     .instance_init = virtio_keyboard_init,
294 };
295 
296 /* ----------------------------------------------------------------- */
297 
298 static QemuInputHandler virtio_mouse_handler = {
299     .name  = VIRTIO_ID_NAME_MOUSE,
300     .mask  = INPUT_EVENT_MASK_BTN | INPUT_EVENT_MASK_REL,
301     .event = virtio_input_handle_event,
302     .sync  = virtio_input_handle_sync,
303 };
304 
305 static struct virtio_input_config virtio_mouse_config_v1[] = {
306     {
307         .select    = VIRTIO_INPUT_CFG_ID_NAME,
308         .size      = sizeof(VIRTIO_ID_NAME_MOUSE),
309         .u.string  = VIRTIO_ID_NAME_MOUSE,
310     },{
311         .select    = VIRTIO_INPUT_CFG_ID_DEVIDS,
312         .size      = sizeof(struct virtio_input_devids),
313         .u.ids     = {
314             .bustype = const_le16(BUS_VIRTUAL),
315             .vendor  = const_le16(0x0627), /* same we use for usb hid devices */
316             .product = const_le16(0x0002),
317             .version = const_le16(0x0001),
318         },
319     },{
320         .select    = VIRTIO_INPUT_CFG_EV_BITS,
321         .subsel    = EV_REL,
322         .size      = 1,
323         .u.bitmap  = {
324             (1 << REL_X) | (1 << REL_Y),
325         },
326     },
327     { /* end of list */ },
328 };
329 
330 static struct virtio_input_config virtio_mouse_config_v2[] = {
331     {
332         .select    = VIRTIO_INPUT_CFG_ID_NAME,
333         .size      = sizeof(VIRTIO_ID_NAME_MOUSE),
334         .u.string  = VIRTIO_ID_NAME_MOUSE,
335     },{
336         .select    = VIRTIO_INPUT_CFG_ID_DEVIDS,
337         .size      = sizeof(struct virtio_input_devids),
338         .u.ids     = {
339             .bustype = const_le16(BUS_VIRTUAL),
340             .vendor  = const_le16(0x0627), /* same we use for usb hid devices */
341             .product = const_le16(0x0002),
342             .version = const_le16(0x0002),
343         },
344     },{
345         .select    = VIRTIO_INPUT_CFG_EV_BITS,
346         .subsel    = EV_REL,
347         .size      = 2,
348         .u.bitmap  = {
349             (1 << REL_X) | (1 << REL_Y),
350             (1 << (REL_WHEEL - 8))
351         },
352     },
353     { /* end of list */ },
354 };
355 
356 static Property virtio_mouse_properties[] = {
357     DEFINE_PROP_BOOL("wheel-axis", VirtIOInputHID, wheel_axis, true),
358     DEFINE_PROP_END_OF_LIST(),
359 };
360 
361 static void virtio_mouse_class_init(ObjectClass *klass, void *data)
362 {
363     DeviceClass *dc = DEVICE_CLASS(klass);
364 
365     device_class_set_props(dc, virtio_mouse_properties);
366 }
367 
368 static void virtio_mouse_init(Object *obj)
369 {
370     VirtIOInputHID *vhid = VIRTIO_INPUT_HID(obj);
371     VirtIOInput *vinput = VIRTIO_INPUT(obj);
372 
373     vhid->handler = &virtio_mouse_handler;
374     virtio_input_init_config(vinput, vhid->wheel_axis
375                              ? virtio_mouse_config_v2
376                              : virtio_mouse_config_v1);
377     virtio_input_key_config(vinput, keymap_button,
378                             ARRAY_SIZE(keymap_button));
379 }
380 
381 static const TypeInfo virtio_mouse_info = {
382     .name          = TYPE_VIRTIO_MOUSE,
383     .parent        = TYPE_VIRTIO_INPUT_HID,
384     .instance_size = sizeof(VirtIOInputHID),
385     .instance_init = virtio_mouse_init,
386     .class_init    = virtio_mouse_class_init,
387 };
388 
389 /* ----------------------------------------------------------------- */
390 
391 static QemuInputHandler virtio_tablet_handler = {
392     .name  = VIRTIO_ID_NAME_TABLET,
393     .mask  = INPUT_EVENT_MASK_BTN | INPUT_EVENT_MASK_ABS,
394     .event = virtio_input_handle_event,
395     .sync  = virtio_input_handle_sync,
396 };
397 
398 static struct virtio_input_config virtio_tablet_config_v1[] = {
399     {
400         .select    = VIRTIO_INPUT_CFG_ID_NAME,
401         .size      = sizeof(VIRTIO_ID_NAME_TABLET),
402         .u.string  = VIRTIO_ID_NAME_TABLET,
403     },{
404         .select    = VIRTIO_INPUT_CFG_ID_DEVIDS,
405         .size      = sizeof(struct virtio_input_devids),
406         .u.ids     = {
407             .bustype = const_le16(BUS_VIRTUAL),
408             .vendor  = const_le16(0x0627), /* same we use for usb hid devices */
409             .product = const_le16(0x0003),
410             .version = const_le16(0x0001),
411         },
412     },{
413         .select    = VIRTIO_INPUT_CFG_EV_BITS,
414         .subsel    = EV_ABS,
415         .size      = 1,
416         .u.bitmap  = {
417             (1 << ABS_X) | (1 << ABS_Y),
418         },
419     },{
420         .select    = VIRTIO_INPUT_CFG_ABS_INFO,
421         .subsel    = ABS_X,
422         .size      = sizeof(virtio_input_absinfo),
423         .u.abs.min = const_le32(INPUT_EVENT_ABS_MIN),
424         .u.abs.max = const_le32(INPUT_EVENT_ABS_MAX),
425     },{
426         .select    = VIRTIO_INPUT_CFG_ABS_INFO,
427         .subsel    = ABS_Y,
428         .size      = sizeof(virtio_input_absinfo),
429         .u.abs.min = const_le32(INPUT_EVENT_ABS_MIN),
430         .u.abs.max = const_le32(INPUT_EVENT_ABS_MAX),
431     },
432     { /* end of list */ },
433 };
434 
435 static struct virtio_input_config virtio_tablet_config_v2[] = {
436     {
437         .select    = VIRTIO_INPUT_CFG_ID_NAME,
438         .size      = sizeof(VIRTIO_ID_NAME_TABLET),
439         .u.string  = VIRTIO_ID_NAME_TABLET,
440     },{
441         .select    = VIRTIO_INPUT_CFG_ID_DEVIDS,
442         .size      = sizeof(struct virtio_input_devids),
443         .u.ids     = {
444             .bustype = const_le16(BUS_VIRTUAL),
445             .vendor  = const_le16(0x0627), /* same we use for usb hid devices */
446             .product = const_le16(0x0003),
447             .version = const_le16(0x0002),
448         },
449     },{
450         .select    = VIRTIO_INPUT_CFG_EV_BITS,
451         .subsel    = EV_ABS,
452         .size      = 1,
453         .u.bitmap  = {
454             (1 << ABS_X) | (1 << ABS_Y),
455         },
456     },{
457         .select    = VIRTIO_INPUT_CFG_EV_BITS,
458         .subsel    = EV_REL,
459         .size      = 2,
460         .u.bitmap  = {
461             0,
462             (1 << (REL_WHEEL - 8))
463         },
464     },{
465         .select    = VIRTIO_INPUT_CFG_ABS_INFO,
466         .subsel    = ABS_X,
467         .size      = sizeof(virtio_input_absinfo),
468         .u.abs.min = const_le32(INPUT_EVENT_ABS_MIN),
469         .u.abs.max = const_le32(INPUT_EVENT_ABS_MAX),
470     },{
471         .select    = VIRTIO_INPUT_CFG_ABS_INFO,
472         .subsel    = ABS_Y,
473         .size      = sizeof(virtio_input_absinfo),
474         .u.abs.min = const_le32(INPUT_EVENT_ABS_MIN),
475         .u.abs.max = const_le32(INPUT_EVENT_ABS_MAX),
476     },
477     { /* end of list */ },
478 };
479 
480 static Property virtio_tablet_properties[] = {
481     DEFINE_PROP_BOOL("wheel-axis", VirtIOInputHID, wheel_axis, true),
482     DEFINE_PROP_END_OF_LIST(),
483 };
484 
485 static void virtio_tablet_class_init(ObjectClass *klass, void *data)
486 {
487     DeviceClass *dc = DEVICE_CLASS(klass);
488 
489     device_class_set_props(dc, virtio_tablet_properties);
490 }
491 
492 static void virtio_tablet_init(Object *obj)
493 {
494     VirtIOInputHID *vhid = VIRTIO_INPUT_HID(obj);
495     VirtIOInput *vinput = VIRTIO_INPUT(obj);
496 
497     vhid->handler = &virtio_tablet_handler;
498     virtio_input_init_config(vinput, vhid->wheel_axis
499                              ? virtio_tablet_config_v2
500                              : virtio_tablet_config_v1);
501     virtio_input_key_config(vinput, keymap_button,
502                             ARRAY_SIZE(keymap_button));
503 }
504 
505 static const TypeInfo virtio_tablet_info = {
506     .name          = TYPE_VIRTIO_TABLET,
507     .parent        = TYPE_VIRTIO_INPUT_HID,
508     .instance_size = sizeof(VirtIOInputHID),
509     .instance_init = virtio_tablet_init,
510     .class_init    = virtio_tablet_class_init,
511 };
512 
513 /* ----------------------------------------------------------------- */
514 
515 static void virtio_register_types(void)
516 {
517     type_register_static(&virtio_input_hid_info);
518     type_register_static(&virtio_keyboard_info);
519     type_register_static(&virtio_mouse_info);
520     type_register_static(&virtio_tablet_info);
521 }
522 
523 type_init(virtio_register_types)
524