keyboard_emu.ino 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350
  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(bool isPress, uint8_t key) {
  87. if (isPress) {
  88. switch (key) {
  89. case PAYLOAD_KEY_LEFT_CTRL:
  90. Keyboard_press(KEY_LEFT_CTRL);
  91. return;
  92. case PAYLOAD_KEY_LEFT_SHIFT:
  93. Keyboard_press(KEY_LEFT_SHIFT);
  94. return;
  95. case PAYLOAD_KEY_LEFT_ALT:
  96. Keyboard_press(KEY_LEFT_ALT);
  97. return;
  98. case PAYLOAD_KEY_LEFT_GUI:
  99. Keyboard_press(KEY_LEFT_GUI);
  100. return;
  101. case PAYLOAD_KEY_RIGHT_CTRL:
  102. Keyboard_press(KEY_RIGHT_CTRL);
  103. return;
  104. case PAYLOAD_KEY_RIGHT_SHIFT:
  105. Keyboard_press(KEY_RIGHT_SHIFT);
  106. return;
  107. case PAYLOAD_KEY_RIGHT_ALT:
  108. Keyboard_press(KEY_RIGHT_ALT);
  109. return;
  110. case PAYLOAD_KEY_RIGHT_GUI:
  111. Keyboard_press(KEY_RIGHT_GUI);
  112. return;
  113. }
  114. } else {
  115. switch (key) {
  116. case PAYLOAD_KEY_LEFT_CTRL:
  117. Keyboard_release(KEY_LEFT_CTRL);
  118. return;
  119. case PAYLOAD_KEY_LEFT_SHIFT:
  120. Keyboard_release(KEY_LEFT_SHIFT);
  121. return;
  122. case PAYLOAD_KEY_LEFT_ALT:
  123. Keyboard_release(KEY_LEFT_ALT);
  124. return;
  125. case PAYLOAD_KEY_LEFT_GUI:
  126. Keyboard_release(KEY_LEFT_GUI);
  127. return;
  128. case PAYLOAD_KEY_RIGHT_CTRL:
  129. Keyboard_release(KEY_RIGHT_CTRL);
  130. return;
  131. case PAYLOAD_KEY_RIGHT_SHIFT:
  132. Keyboard_release(KEY_RIGHT_SHIFT);
  133. return;
  134. case PAYLOAD_KEY_RIGHT_ALT:
  135. Keyboard_release(KEY_RIGHT_ALT);
  136. return;
  137. case PAYLOAD_KEY_RIGHT_GUI:
  138. Keyboard_release(KEY_RIGHT_GUI);
  139. return;
  140. }
  141. }
  142. }
  143. //Set a specific numpad key value
  144. void numpad_key_set(bool isPress, uint8_t value) {
  145. if (isPress) {
  146. switch (value) {
  147. case PAYLOAD_NUMPAD_0:
  148. Keyboard_press('\352'); //0
  149. return;
  150. case PAYLOAD_NUMPAD_1:
  151. Keyboard_press('\341'); //1
  152. return;
  153. case PAYLOAD_NUMPAD_2:
  154. Keyboard_press('\342'); //2
  155. return;
  156. case PAYLOAD_NUMPAD_3:
  157. Keyboard_press('\343'); //3
  158. return;
  159. case PAYLOAD_NUMPAD_4:
  160. Keyboard_press('\344'); //4
  161. return;
  162. case PAYLOAD_NUMPAD_5:
  163. Keyboard_press('\345'); //5
  164. return;
  165. case PAYLOAD_NUMPAD_6:
  166. Keyboard_press('\346'); //6
  167. return;
  168. case PAYLOAD_NUMPAD_7:
  169. Keyboard_press('\347'); //7
  170. return;
  171. case PAYLOAD_NUMPAD_8:
  172. Keyboard_press('\350'); //8
  173. return;
  174. case PAYLOAD_NUMPAD_9:
  175. Keyboard_press('\351'); //9
  176. return;
  177. case PAYLOAD_NUMPAD_DOT:
  178. Keyboard_press('\353'); //.
  179. return;
  180. case PAYLOAD_NUMPAD_TIMES:
  181. Keyboard_press('\335'); //*
  182. return;
  183. case PAYLOAD_NUMPAD_DIV:
  184. Keyboard_press('\334'); // /
  185. return;
  186. case PAYLOAD_NUMPAD_PLUS:
  187. Keyboard_press('\337'); // +
  188. return;
  189. case PAYLOAD_NUMPAD_MINUS:
  190. Keyboard_press('\336'); // -
  191. return;
  192. case PAYLOAD_NUMPAD_ENTER:
  193. Keyboard_press('\340'); // Enter
  194. return;
  195. case PAYLOAD_NUMPAD_NUMLOCK:
  196. Keyboard_press('\333'); // NumLock
  197. return;
  198. }
  199. } else {
  200. switch (value) {
  201. case PAYLOAD_NUMPAD_0:
  202. Keyboard_release('\352'); //0
  203. return;
  204. case PAYLOAD_NUMPAD_1:
  205. Keyboard_release('\341'); //1
  206. return;
  207. case PAYLOAD_NUMPAD_2:
  208. Keyboard_release('\342'); //2
  209. return;
  210. case PAYLOAD_NUMPAD_3:
  211. Keyboard_release('\343'); //3
  212. return;
  213. case PAYLOAD_NUMPAD_4:
  214. Keyboard_release('\344'); //4
  215. return;
  216. case PAYLOAD_NUMPAD_5:
  217. Keyboard_release('\345'); //5
  218. return;
  219. case PAYLOAD_NUMPAD_6:
  220. Keyboard_release('\346'); //6
  221. return;
  222. case PAYLOAD_NUMPAD_7:
  223. Keyboard_release('\347'); //7
  224. return;
  225. case PAYLOAD_NUMPAD_8:
  226. Keyboard_release('\350'); //8
  227. return;
  228. case PAYLOAD_NUMPAD_9:
  229. Keyboard_release('\351'); //9
  230. return;
  231. case PAYLOAD_NUMPAD_DOT:
  232. Keyboard_release('\353'); //.
  233. return;
  234. case PAYLOAD_NUMPAD_TIMES:
  235. Keyboard_release('\335'); //*
  236. return;
  237. case PAYLOAD_NUMPAD_DIV:
  238. Keyboard_release('\334'); // /
  239. return;
  240. case PAYLOAD_NUMPAD_PLUS:
  241. Keyboard_release('\337'); // +
  242. return;
  243. case PAYLOAD_NUMPAD_MINUS:
  244. Keyboard_release('\336'); // -
  245. return;
  246. case PAYLOAD_NUMPAD_ENTER:
  247. Keyboard_release('\340'); // Enter
  248. return;
  249. case PAYLOAD_NUMPAD_NUMLOCK:
  250. Keyboard_release('\333'); // NumLock
  251. return;
  252. }
  253. }
  254. }
  255. //Entry point for keyboard emulation
  256. uint8_t keyboard_emulation(uint8_t subtype, uint8_t value) {
  257. //Check if the input is a supported ascii value
  258. switch (subtype) {
  259. case SUBTYPE_KEYBOARD_ASCII_WRITE:
  260. if (!is_ascii(value))
  261. return resp_invalid_key_value;
  262. Keyboard_write(value);
  263. return resp_ok;
  264. case SUBTYPE_KEYBOARD_ASCII_PRESS:
  265. if (!is_ascii(value))
  266. return resp_invalid_key_value;
  267. Keyboard_press(value);
  268. return resp_ok;
  269. case SUBTYPE_KEYBOARD_ASCII_RELEASE:
  270. if (!is_ascii(value))
  271. return resp_invalid_key_value;
  272. Keyboard_release(value);
  273. return resp_ok;
  274. case SUBTYPE_KEYBOARD_MODIFIER_PRESS:
  275. keyboard_modifying_key_set(true, value);
  276. return resp_ok;
  277. case SUBTYPE_KEYBOARD_MODIFIER_RELEASE:
  278. keyboard_modifying_key_set(false, value);
  279. return resp_ok;
  280. case SUBTYPE_KEYBOARD_FUNCTKEY_PRESS:
  281. if (!is_funckey(value))
  282. return resp_invalid_key_value;
  283. Keyboard_press(value);
  284. return resp_ok;
  285. case SUBTYPE_KEYBOARD_FUNCTKEY_RELEASE:
  286. if (!is_funckey(value))
  287. return resp_invalid_key_value;
  288. Keyboard_release(value);
  289. return resp_ok;
  290. case SUBTYPE_KEYBOARD_OTHERKEY_PRESS:
  291. if (!is_validkeys(value))
  292. return resp_invalid_key_value;
  293. Keyboard_press(value);
  294. return resp_ok;
  295. case SUBTYPE_KEYBOARD_OTHERKEY_RELEASE:
  296. if (!is_validkeys(value))
  297. return resp_invalid_key_value;
  298. Keyboard_release(value);
  299. return resp_ok;
  300. case SUBTYPE_KEYBOARD_NUMPAD_PRESS:
  301. if (value > PAYLOAD_NUMPAD_NUMLOCK)
  302. return resp_invalid_key_value;
  303. numpad_key_set(true, value);
  304. return resp_ok;
  305. case SUBTYPE_KEYBOARD_NUMPAD_RELEASE:
  306. if (value > PAYLOAD_NUMPAD_NUMLOCK)
  307. return resp_invalid_key_value;
  308. numpad_key_set(false, value);
  309. return resp_ok;
  310. case SUBTYPE_KEYBOARD_SPECIAL_CTRLALTDEL:
  311. // Press Ctrl + Alt + Del
  312. Keyboard_press(KEY_LEFT_CTRL);
  313. Keyboard_press(KEY_LEFT_ALT);
  314. Keyboard_press(KEY_DELETE);
  315. delay(100); // Give a little longer time for all keys to be registered together
  316. // Release Ctrl + Alt + Del in reverse order
  317. Keyboard_release(KEY_DELETE);
  318. delay(MIN_KEY_EVENTS_DELAY);
  319. Keyboard_release(KEY_LEFT_ALT);
  320. delay(MIN_KEY_EVENTS_DELAY);
  321. Keyboard_release(KEY_LEFT_CTRL);
  322. delay(MIN_KEY_EVENTS_DELAY);
  323. return resp_ok;
  324. case SUBTYPE_KEYBOARD_SPECIAL_RESET:
  325. //Release all pressed keys
  326. Keyboard_releaseAll();
  327. return resp_ok;
  328. default:
  329. return resp_invalid_opr_type;
  330. }
  331. }