keyboard_emu.ino 12 KB

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