123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143 |
- /*
- * This work is licensed under the terms of the GNU GPL, version 2 or
- * (at your option) any later version. See the COPYING file in the
- * top-level directory.
- */
- #include "qemu/osdep.h"
- #include "qemu/bitmap.h"
- #include "ui/console.h"
- #include "ui/input.h"
- #include "ui/kbd-state.h"
- struct QKbdState {
- QemuConsole *con;
- int key_delay_ms;
- DECLARE_BITMAP(keys, Q_KEY_CODE__MAX);
- DECLARE_BITMAP(mods, QKBD_MOD__MAX);
- };
- static void qkbd_state_modifier_update(QKbdState *kbd,
- QKeyCode qcode1, QKeyCode qcode2,
- QKbdModifier mod)
- {
- if (test_bit(qcode1, kbd->keys) || test_bit(qcode2, kbd->keys)) {
- set_bit(mod, kbd->mods);
- } else {
- clear_bit(mod, kbd->mods);
- }
- }
- bool qkbd_state_modifier_get(QKbdState *kbd, QKbdModifier mod)
- {
- return test_bit(mod, kbd->mods);
- }
- bool qkbd_state_key_get(QKbdState *kbd, QKeyCode qcode)
- {
- return test_bit(qcode, kbd->keys);
- }
- void qkbd_state_key_event(QKbdState *kbd, QKeyCode qcode, bool down)
- {
- bool state = test_bit(qcode, kbd->keys);
- if (down == false /* got key-up event */ &&
- state == false /* key is not pressed */) {
- /*
- * Filter out suspicious key-up events.
- *
- * This allows simply sending along all key-up events, and
- * this function will filter out everything where the
- * corresponding key-down event wasn't sent to the guest, for
- * example due to being a host hotkey.
- *
- * Note that key-down events on already pressed keys are *not*
- * suspicious, those are keyboard autorepeat events.
- */
- return;
- }
- /* update key and modifier state */
- if (down) {
- set_bit(qcode, kbd->keys);
- } else {
- clear_bit(qcode, kbd->keys);
- }
- switch (qcode) {
- case Q_KEY_CODE_SHIFT:
- case Q_KEY_CODE_SHIFT_R:
- qkbd_state_modifier_update(kbd, Q_KEY_CODE_SHIFT, Q_KEY_CODE_SHIFT_R,
- QKBD_MOD_SHIFT);
- break;
- case Q_KEY_CODE_CTRL:
- case Q_KEY_CODE_CTRL_R:
- qkbd_state_modifier_update(kbd, Q_KEY_CODE_CTRL, Q_KEY_CODE_CTRL_R,
- QKBD_MOD_CTRL);
- break;
- case Q_KEY_CODE_ALT:
- qkbd_state_modifier_update(kbd, Q_KEY_CODE_ALT, Q_KEY_CODE_ALT,
- QKBD_MOD_ALT);
- break;
- case Q_KEY_CODE_ALT_R:
- qkbd_state_modifier_update(kbd, Q_KEY_CODE_ALT_R, Q_KEY_CODE_ALT_R,
- QKBD_MOD_ALTGR);
- break;
- case Q_KEY_CODE_CAPS_LOCK:
- if (down) {
- change_bit(QKBD_MOD_CAPSLOCK, kbd->mods);
- }
- break;
- case Q_KEY_CODE_NUM_LOCK:
- if (down) {
- change_bit(QKBD_MOD_NUMLOCK, kbd->mods);
- }
- break;
- default:
- /* keep gcc happy */
- break;
- }
- /* send to guest */
- if (qemu_console_is_graphic(kbd->con)) {
- qemu_input_event_send_key_qcode(kbd->con, qcode, down);
- if (kbd->key_delay_ms) {
- qemu_input_event_send_key_delay(kbd->key_delay_ms);
- }
- }
- }
- void qkbd_state_lift_all_keys(QKbdState *kbd)
- {
- int qcode;
- for (qcode = 0; qcode < Q_KEY_CODE__MAX; qcode++) {
- if (test_bit(qcode, kbd->keys)) {
- qkbd_state_key_event(kbd, qcode, false);
- }
- }
- }
- void qkbd_state_switch_console(QKbdState *kbd, QemuConsole *con)
- {
- qkbd_state_lift_all_keys(kbd);
- kbd->con = con;
- }
- void qkbd_state_set_delay(QKbdState *kbd, int delay_ms)
- {
- kbd->key_delay_ms = delay_ms;
- }
- void qkbd_state_free(QKbdState *kbd)
- {
- g_free(kbd);
- }
- QKbdState *qkbd_state_init(QemuConsole *con)
- {
- QKbdState *kbd = g_new0(QKbdState, 1);
- kbd->con = con;
- return kbd;
- }
|