2
0

kbd-state.c 3.7 KB

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