xref: /openbmc/qemu/ui/kbd-state.c (revision 4921d0a7)
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 #include "qemu/osdep.h"
7 #include "qemu/bitmap.h"
8 #include "ui/console.h"
9 #include "ui/input.h"
10 #include "ui/kbd-state.h"
11 
12 struct QKbdState {
13     QemuConsole *con;
14     int key_delay_ms;
15     DECLARE_BITMAP(keys, Q_KEY_CODE__MAX);
16     DECLARE_BITMAP(mods, QKBD_MOD__MAX);
17 };
18 
19 static void qkbd_state_modifier_update(QKbdState *kbd,
20                                       QKeyCode qcode1, QKeyCode qcode2,
21                                       QKbdModifier mod)
22 {
23     if (test_bit(qcode1, kbd->keys) || test_bit(qcode2, kbd->keys)) {
24         set_bit(mod, kbd->mods);
25     } else {
26         clear_bit(mod, kbd->mods);
27     }
28 }
29 
30 bool qkbd_state_modifier_get(QKbdState *kbd, QKbdModifier mod)
31 {
32     return test_bit(mod, kbd->mods);
33 }
34 
35 bool qkbd_state_key_get(QKbdState *kbd, QKeyCode qcode)
36 {
37     return test_bit(qcode, kbd->keys);
38 }
39 
40 void qkbd_state_key_event(QKbdState *kbd, QKeyCode qcode, bool down)
41 {
42     bool state = test_bit(qcode, kbd->keys);
43 
44     if (down == false  /* got key-up event   */ &&
45         state == false /* key is not pressed */) {
46         /*
47          * Filter out suspicious key-up events.
48          *
49          * This allows simply sending along all key-up events, and
50          * this function will filter out everything where the
51          * corresponding key-down event wasn't sent to the guest, for
52          * example due to being a host hotkey.
53          *
54          * Note that key-down events on already pressed keys are *not*
55          * suspicious, those are keyboard autorepeat events.
56          */
57         return;
58     }
59 
60     /* update key and modifier state */
61     if (down) {
62         set_bit(qcode, kbd->keys);
63     } else {
64         clear_bit(qcode, kbd->keys);
65     }
66     switch (qcode) {
67     case Q_KEY_CODE_SHIFT:
68     case Q_KEY_CODE_SHIFT_R:
69         qkbd_state_modifier_update(kbd, Q_KEY_CODE_SHIFT, Q_KEY_CODE_SHIFT_R,
70                                    QKBD_MOD_SHIFT);
71         break;
72     case Q_KEY_CODE_CTRL:
73     case Q_KEY_CODE_CTRL_R:
74         qkbd_state_modifier_update(kbd, Q_KEY_CODE_CTRL, Q_KEY_CODE_CTRL_R,
75                                    QKBD_MOD_CTRL);
76         break;
77     case Q_KEY_CODE_ALT:
78         qkbd_state_modifier_update(kbd, Q_KEY_CODE_ALT, Q_KEY_CODE_ALT,
79                                    QKBD_MOD_ALT);
80         break;
81     case Q_KEY_CODE_ALT_R:
82         qkbd_state_modifier_update(kbd, Q_KEY_CODE_ALT_R, Q_KEY_CODE_ALT_R,
83                                    QKBD_MOD_ALTGR);
84         break;
85     case Q_KEY_CODE_CAPS_LOCK:
86         if (down) {
87             change_bit(QKBD_MOD_CAPSLOCK, kbd->mods);
88         }
89         break;
90     case Q_KEY_CODE_NUM_LOCK:
91         if (down) {
92             change_bit(QKBD_MOD_NUMLOCK, kbd->mods);
93         }
94         break;
95     default:
96         /* keep gcc happy */
97         break;
98     }
99 
100     /* send to guest */
101     if (qemu_console_is_graphic(kbd->con)) {
102         qemu_input_event_send_key_qcode(kbd->con, qcode, down);
103         if (kbd->key_delay_ms) {
104             qemu_input_event_send_key_delay(kbd->key_delay_ms);
105         }
106     }
107 }
108 
109 void qkbd_state_lift_all_keys(QKbdState *kbd)
110 {
111     int qcode;
112 
113     for (qcode = 0; qcode < Q_KEY_CODE__MAX; qcode++) {
114         if (test_bit(qcode, kbd->keys)) {
115             qkbd_state_key_event(kbd, qcode, false);
116         }
117     }
118 }
119 
120 void qkbd_state_switch_console(QKbdState *kbd, QemuConsole *con)
121 {
122     qkbd_state_lift_all_keys(kbd);
123     kbd->con = con;
124 }
125 
126 void qkbd_state_set_delay(QKbdState *kbd, int delay_ms)
127 {
128     kbd->key_delay_ms = delay_ms;
129 }
130 
131 void qkbd_state_free(QKbdState *kbd)
132 {
133     g_free(kbd);
134 }
135 
136 QKbdState *qkbd_state_init(QemuConsole *con)
137 {
138     QKbdState *kbd = g_new0(QKbdState, 1);
139 
140     kbd->con = con;
141 
142     return kbd;
143 }
144