keyboard_emu.ino 11 KB

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