keyboard_emu.ino 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342
  1. /*
  2. keyboard_emu.ino
  3. Author: tobychui
  4. This code file handle keyboard emulation related functionality
  5. When opr_type is set to 0x01, the sub-handler will process the
  6. request here.
  7. -- Keyboard Opcode --
  8. 0x00 = Reserved
  9. 0x01 = keyboard write
  10. 0x02 = keyboard press
  11. 0x03 = keyboard release
  12. (ASCII bytes in range of 32 to 127)
  13. 0x04 = Modifier key combination set (bit flags in payload)
  14. Bit 0 (0x01) = KEY_LEFT_CTRL
  15. Bit 1 (0x02) = KEY_LEFT_SHIFT
  16. Bit 2 (0x04) = KEY_LEFT_ALT
  17. Bit 3 (0x08) = KEY_LEFT_GUI
  18. Bit 4 (0x10) = KEY_RIGHT_CTRL
  19. Bit 5 (0x20) = KEY_RIGHT_SHIFT
  20. Bit 6 (0x40) = KEY_RIGHT_ALT
  21. Bit 7 (0x80) = KEY_RIGHT_GUI
  22. 0x05 = Modifier key combination reset
  23. 0x07 = Function key combinartion write
  24. 0xC2 = KEY_F1
  25. 0xC3 = KEY_F2
  26. 0xC4 = KEY_F3
  27. ...
  28. 0xCC = KEY_F11
  29. 0xCD = KEY_F12
  30. 0xF0 = KEY_F13
  31. ...
  32. 0xFB = KEY_F24
  33. 0x08 = Other keys press
  34. 0x09 = Other keys release
  35. 0xDA = KEY_UP_ARROW
  36. 0xD9 = KEY_DOWN_ARROW
  37. 0xD8 = KEY_LEFT_ARROW
  38. 0xD7 = KEY_RIGHT_ARROW
  39. 0xB2 = KEY_BACKSPACE
  40. 0xB3 = KEY_TAB
  41. 0xB0 = KEY_RETURN
  42. 0xB1 = KEY_ESC
  43. 0xD1 = KEY_INSERT
  44. 0xD4 = KEY_DELETE
  45. 0xD3 = KEY_PAGE_UP
  46. 0xD6 = KEY_PAGE_DOWN
  47. 0xD2 = KEY_HOME
  48. 0xD5 = KEY_END
  49. 0xC1 = KEY_CAPS_LOCK
  50. -- Special Opcode --
  51. 0xFE = Ctrl + Alt + Delete
  52. 0xFF = Reset all keys state
  53. */
  54. #include "usbkvm_fw.h"
  55. //Check if the value is a supported ascii code
  56. bool is_ascii(uint8_t value) {
  57. return value >= 32 && value <= 127;
  58. }
  59. //Check if the value is a valid function key value
  60. bool is_funckey(uint8_t value) {
  61. return ((value >= 0xC2 && value <= 0xCD) || (value >= 0xF0 && value <= 0xFB));
  62. }
  63. //Check if the key is valid key that do not belongs in ASCII or function key
  64. bool is_validkeys(uint8_t key) {
  65. if (key >= 0xD7 && key <= 0xDA) {
  66. //Arrow keys
  67. return true;
  68. }
  69. if (key >= 0xB0 && key <= 0xB3) {
  70. //ESC, tabs and other left-most keys
  71. return true;
  72. }
  73. if (key >= 0xD1 && key <= 0xD6) {
  74. //Keys above arrow keys like pageup and down
  75. return true;
  76. }
  77. //CAP_LOCK
  78. return key == 0xC1;
  79. }
  80. //Check if the nth bit is set in the value
  81. bool isBitSet(uint8_t value, uint8_t n) {
  82. if (n > 7) return false; // Ensure n is within 0-7
  83. return (value & (1 << n)) != 0;
  84. }
  85. //Handle modifying key set or unset
  86. void keyboard_modifying_key_set(uint8_t key) {
  87. if (key & 0x01) { // Bit 0 = KEY_LEFT_CTRL
  88. Keyboard_press(KEY_LEFT_CTRL);
  89. } else {
  90. Keyboard_release(KEY_LEFT_CTRL);
  91. }
  92. if (key & 0x02) { // Bit 1 = KEY_LEFT_SHIFT
  93. Keyboard_press(KEY_LEFT_SHIFT);
  94. } else {
  95. Keyboard_release(KEY_LEFT_SHIFT);
  96. }
  97. if (key & 0x04) { // Bit 2 = KEY_LEFT_ALT
  98. Keyboard_press(KEY_LEFT_ALT);
  99. } else {
  100. Keyboard_release(KEY_LEFT_ALT);
  101. }
  102. if (key & 0x08) { // Bit 3 = KEY_LEFT_GUI
  103. Keyboard_press(KEY_LEFT_GUI);
  104. } else {
  105. Keyboard_release(KEY_LEFT_GUI);
  106. }
  107. if (key & 0x10) { // Bit 4 = KEY_RIGHT_CTRL
  108. Keyboard_press(KEY_RIGHT_CTRL);
  109. } else {
  110. Keyboard_release(KEY_RIGHT_CTRL);
  111. }
  112. if (key & 0x20) { // Bit 5 = KEY_RIGHT_SHIFT
  113. Keyboard_press(KEY_RIGHT_SHIFT);
  114. } else {
  115. Keyboard_release(KEY_RIGHT_SHIFT);
  116. }
  117. if (key & 0x40) { // Bit 6 = KEY_RIGHT_ALT
  118. Keyboard_press(KEY_RIGHT_ALT);
  119. } else {
  120. Keyboard_release(KEY_RIGHT_ALT);
  121. }
  122. if (key & 0x80) { // Bit 7 = KEY_RIGHT_GUI
  123. Keyboard_press(KEY_RIGHT_GUI);
  124. } else {
  125. Keyboard_release(KEY_RIGHT_GUI);
  126. }
  127. }
  128. //Set a specific numpad key value
  129. void numpad_key_set(bool isPress, uint8_t value) {
  130. if (isPress) {
  131. switch (value) {
  132. case PAYLOAD_NUMPAD_0:
  133. Keyboard_press('\352'); //0
  134. return;
  135. case PAYLOAD_NUMPAD_1:
  136. Keyboard_press('\341'); //1
  137. return;
  138. case PAYLOAD_NUMPAD_2:
  139. Keyboard_press('\342'); //2
  140. return;
  141. case PAYLOAD_NUMPAD_3:
  142. Keyboard_press('\343'); //3
  143. return;
  144. case PAYLOAD_NUMPAD_4:
  145. Keyboard_press('\344'); //4
  146. return;
  147. case PAYLOAD_NUMPAD_5:
  148. Keyboard_press('\345'); //5
  149. return;
  150. case PAYLOAD_NUMPAD_6:
  151. Keyboard_press('\346'); //6
  152. return;
  153. case PAYLOAD_NUMPAD_7:
  154. Keyboard_press('\347'); //7
  155. return;
  156. case PAYLOAD_NUMPAD_8:
  157. Keyboard_press('\350'); //8
  158. return;
  159. case PAYLOAD_NUMPAD_9:
  160. Keyboard_press('\351'); //9
  161. return;
  162. case PAYLOAD_NUMPAD_DOT:
  163. Keyboard_press('\353'); //.
  164. return;
  165. case PAYLOAD_NUMPAD_TIMES:
  166. Keyboard_press('\335'); //*
  167. return;
  168. case PAYLOAD_NUMPAD_DIV:
  169. Keyboard_press('\334'); // /
  170. return;
  171. case PAYLOAD_NUMPAD_PLUS:
  172. Keyboard_press('\337'); // +
  173. return;
  174. case PAYLOAD_NUMPAD_MINUS:
  175. Keyboard_press('\336'); // -
  176. return;
  177. case PAYLOAD_NUMPAD_ENTER:
  178. Keyboard_press('\340'); // Enter
  179. return;
  180. case PAYLOAD_NUMPAD_NUMLOCK:
  181. Keyboard_press('\333'); // NumLock
  182. return;
  183. }
  184. } else {
  185. switch (value) {
  186. case PAYLOAD_NUMPAD_0:
  187. Keyboard_release('\352'); //0
  188. return;
  189. case PAYLOAD_NUMPAD_1:
  190. Keyboard_release('\341'); //1
  191. return;
  192. case PAYLOAD_NUMPAD_2:
  193. Keyboard_release('\342'); //2
  194. return;
  195. case PAYLOAD_NUMPAD_3:
  196. Keyboard_release('\343'); //3
  197. return;
  198. case PAYLOAD_NUMPAD_4:
  199. Keyboard_release('\344'); //4
  200. return;
  201. case PAYLOAD_NUMPAD_5:
  202. Keyboard_release('\345'); //5
  203. return;
  204. case PAYLOAD_NUMPAD_6:
  205. Keyboard_release('\346'); //6
  206. return;
  207. case PAYLOAD_NUMPAD_7:
  208. Keyboard_release('\347'); //7
  209. return;
  210. case PAYLOAD_NUMPAD_8:
  211. Keyboard_release('\350'); //8
  212. return;
  213. case PAYLOAD_NUMPAD_9:
  214. Keyboard_release('\351'); //9
  215. return;
  216. case PAYLOAD_NUMPAD_DOT:
  217. Keyboard_release('\353'); //.
  218. return;
  219. case PAYLOAD_NUMPAD_TIMES:
  220. Keyboard_release('\335'); //*
  221. return;
  222. case PAYLOAD_NUMPAD_DIV:
  223. Keyboard_release('\334'); // /
  224. return;
  225. case PAYLOAD_NUMPAD_PLUS:
  226. Keyboard_release('\337'); // +
  227. return;
  228. case PAYLOAD_NUMPAD_MINUS:
  229. Keyboard_release('\336'); // -
  230. return;
  231. case PAYLOAD_NUMPAD_ENTER:
  232. Keyboard_release('\340'); // Enter
  233. return;
  234. case PAYLOAD_NUMPAD_NUMLOCK:
  235. Keyboard_release('\333'); // NumLock
  236. return;
  237. }
  238. }
  239. }
  240. //Entry point for keyboard emulation
  241. uint8_t keyboard_emulation(uint8_t subtype, uint8_t value) {
  242. //Check if the input is a supported ascii value
  243. switch (subtype) {
  244. case SUBTYPE_KEYBOARD_ASCII_WRITE:
  245. if (!is_ascii(value))
  246. return resp_invalid_key_value;
  247. Keyboard_write(value);
  248. return resp_ok;
  249. case SUBTYPE_KEYBOARD_ASCII_PRESS:
  250. if (!is_ascii(value))
  251. return resp_invalid_key_value;
  252. Keyboard_press(value);
  253. return resp_ok;
  254. case SUBTYPE_KEYBOARD_ASCII_RELEASE:
  255. if (!is_ascii(value))
  256. return resp_invalid_key_value;
  257. Keyboard_release(value);
  258. return resp_ok;
  259. case SUBTYPE_KEYBOARD_MODIFIER_SET:
  260. keyboard_modifying_key_set(value);
  261. return resp_ok;
  262. case SUBTYPE_KEYBOARD_MODIFIER_CLEAR:
  263. keyboard_modifying_key_set(0x00);
  264. return resp_ok;
  265. case SUBTYPE_KEYBOARD_FUNCTKEY_PRESS:
  266. if (!is_funckey(value))
  267. return resp_invalid_key_value;
  268. Keyboard_press(value);
  269. return resp_ok;
  270. case SUBTYPE_KEYBOARD_FUNCTKEY_RELEASE:
  271. if (!is_funckey(value))
  272. return resp_invalid_key_value;
  273. Keyboard_release(value);
  274. return resp_ok;
  275. case SUBTYPE_KEYBOARD_OTHERKEY_PRESS:
  276. if (!is_validkeys(value))
  277. return resp_invalid_key_value;
  278. Keyboard_press(value);
  279. return resp_ok;
  280. case SUBTYPE_KEYBOARD_OTHERKEY_RELEASE:
  281. if (!is_validkeys(value))
  282. return resp_invalid_key_value;
  283. Keyboard_release(value);
  284. return resp_ok;
  285. case SUBTYPE_KEYBOARD_NUMPAD_PRESS:
  286. if (value > PAYLOAD_NUMPAD_NUMLOCK)
  287. return resp_invalid_key_value;
  288. numpad_key_set(true, value);
  289. return resp_ok;
  290. case SUBTYPE_KEYBOARD_NUMPAD_RELEASE:
  291. if (value > PAYLOAD_NUMPAD_NUMLOCK)
  292. return resp_invalid_key_value;
  293. numpad_key_set(false, value);
  294. return resp_ok;
  295. case SUBTYPE_KEYBOARD_SPECIAL_CTRLALTDEL:
  296. // Press Ctrl + Alt + Del
  297. Keyboard_press(KEY_LEFT_CTRL);
  298. Keyboard_press(KEY_LEFT_ALT);
  299. Keyboard_press(KEY_DELETE);
  300. delay(100); // Give a little longer time for all keys to be registered together
  301. // Release Ctrl + Alt + Del in reverse order
  302. Keyboard_release(KEY_DELETE);
  303. delay(MIN_KEY_EVENTS_DELAY);
  304. Keyboard_release(KEY_LEFT_ALT);
  305. delay(MIN_KEY_EVENTS_DELAY);
  306. Keyboard_release(KEY_LEFT_CTRL);
  307. delay(MIN_KEY_EVENTS_DELAY);
  308. return resp_ok;
  309. case SUBTYPE_KEYBOARD_SPECIAL_RESET:
  310. //Release all pressed keys
  311. Keyboard_releaseAll();
  312. return resp_ok;
  313. default:
  314. return resp_invalid_opr_type;
  315. }
  316. }