report.h 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277
  1. /*
  2. Copyright 2011,2012 Jun Wako <wakojun@gmail.com>
  3. This program is free software: you can redistribute it and/or modify
  4. it under the terms of the GNU General Public License as published by
  5. the Free Software Foundation, either version 2 of the License, or
  6. (at your option) any later version.
  7. This program is distributed in the hope that it will be useful,
  8. but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  10. GNU General Public License for more details.
  11. You should have received a copy of the GNU General Public License
  12. along with this program. If not, see <http://www.gnu.org/licenses/>.
  13. */
  14. #pragma once
  15. #include <stdint.h>
  16. #include <stdbool.h>
  17. #include "keycode.h"
  18. // clang-format off
  19. /* HID report IDs */
  20. enum hid_report_ids {
  21. REPORT_ID_KEYBOARD = 1,
  22. REPORT_ID_MOUSE,
  23. REPORT_ID_SYSTEM,
  24. REPORT_ID_CONSUMER,
  25. REPORT_ID_NKRO
  26. };
  27. /* Mouse buttons */
  28. enum mouse_buttons {
  29. MOUSE_BTN1 = (1 << 0),
  30. MOUSE_BTN2 = (1 << 1),
  31. MOUSE_BTN3 = (1 << 2),
  32. MOUSE_BTN4 = (1 << 3),
  33. MOUSE_BTN5 = (1 << 4)
  34. };
  35. /* Consumer Page (0x0C)
  36. *
  37. * See https://www.usb.org/sites/default/files/documents/hut1_12v2.pdf#page=75
  38. */
  39. enum consumer_usages {
  40. // 15.5 Display Controls
  41. SNAPSHOT = 0x065,
  42. BRIGHTNESS_UP = 0x06F, // https://www.usb.org/sites/default/files/hutrr41_0.pdf
  43. BRIGHTNESS_DOWN = 0x070,
  44. // 15.7 Transport Controls
  45. TRANSPORT_RECORD = 0x0B2,
  46. TRANSPORT_FAST_FORWARD = 0x0B3,
  47. TRANSPORT_REWIND = 0x0B4,
  48. TRANSPORT_NEXT_TRACK = 0x0B5,
  49. TRANSPORT_PREV_TRACK = 0x0B6,
  50. TRANSPORT_STOP = 0x0B7,
  51. TRANSPORT_EJECT = 0x0B8,
  52. TRANSPORT_RANDOM_PLAY = 0x0B9,
  53. TRANSPORT_STOP_EJECT = 0x0CC,
  54. TRANSPORT_PLAY_PAUSE = 0x0CD,
  55. // 15.9.1 Audio Controls - Volume
  56. AUDIO_MUTE = 0x0E2,
  57. AUDIO_VOL_UP = 0x0E9,
  58. AUDIO_VOL_DOWN = 0x0EA,
  59. // 15.15 Application Launch Buttons
  60. AL_CC_CONFIG = 0x183,
  61. AL_EMAIL = 0x18A,
  62. AL_CALCULATOR = 0x192,
  63. AL_LOCAL_BROWSER = 0x194,
  64. AL_LOCK = 0x19E,
  65. AL_CONTROL_PANEL = 0x19F,
  66. AL_ASSISTANT = 0x1CB,
  67. AL_KEYBOARD_LAYOUT = 0x1AE,
  68. // 15.16 Generic GUI Application Controls
  69. AC_MINIMIZE = 0x206,
  70. AC_SEARCH = 0x221,
  71. AC_HOME = 0x223,
  72. AC_BACK = 0x224,
  73. AC_FORWARD = 0x225,
  74. AC_STOP = 0x226,
  75. AC_REFRESH = 0x227,
  76. AC_BOOKMARKS = 0x22A
  77. };
  78. /* Generic Desktop Page (0x01)
  79. *
  80. * See https://www.usb.org/sites/default/files/documents/hut1_12v2.pdf#page=26
  81. */
  82. enum desktop_usages {
  83. // 4.5.1 System Controls - Power Controls
  84. SYSTEM_POWER_DOWN = 0x81,
  85. SYSTEM_SLEEP = 0x82,
  86. SYSTEM_WAKE_UP = 0x83
  87. };
  88. // clang-format on
  89. #define NKRO_SHARED_EP
  90. /* key report size(NKRO or boot mode) */
  91. #if defined(NKRO_ENABLE)
  92. # if defined(PROTOCOL_LUFA) || defined(PROTOCOL_CHIBIOS)
  93. # include "protocol/usb_descriptor.h"
  94. # define KEYBOARD_REPORT_BITS (SHARED_EPSIZE - 2)
  95. # elif defined(PROTOCOL_ARM_ATSAM)
  96. # include "protocol/arm_atsam/usb/udi_device_epsize.h"
  97. # define KEYBOARD_REPORT_BITS (NKRO_EPSIZE - 1)
  98. # undef NKRO_SHARED_EP
  99. # undef MOUSE_SHARED_EP
  100. # else
  101. # error "NKRO not supported with this protocol"
  102. # endif
  103. #endif
  104. #ifdef KEYBOARD_SHARED_EP
  105. # define KEYBOARD_REPORT_SIZE 9
  106. #else
  107. # define KEYBOARD_REPORT_SIZE 8
  108. #endif
  109. #define KEYBOARD_REPORT_KEYS 6
  110. /* VUSB hardcodes keyboard and mouse+extrakey only */
  111. #if defined(PROTOCOL_VUSB)
  112. # undef KEYBOARD_SHARED_EP
  113. # undef MOUSE_SHARED_EP
  114. #endif
  115. #ifdef __cplusplus
  116. extern "C" {
  117. #endif
  118. /*
  119. * keyboard report is 8-byte array retains state of 8 modifiers and 6 keys.
  120. *
  121. * byte |0 |1 |2 |3 |4 |5 |6 |7
  122. * -----+--------+--------+--------+--------+--------+--------+--------+--------
  123. * desc |mods |reserved|keys[0] |keys[1] |keys[2] |keys[3] |keys[4] |keys[5]
  124. *
  125. * It is exended to 16 bytes to retain 120keys+8mods when NKRO mode.
  126. *
  127. * byte |0 |1 |2 |3 |4 |5 |6 |7 ... |15
  128. * -----+--------+--------+--------+--------+--------+--------+--------+-------- +--------
  129. * desc |mods |bits[0] |bits[1] |bits[2] |bits[3] |bits[4] |bits[5] |bits[6] ... |bit[14]
  130. *
  131. * mods retains state of 8 modifiers.
  132. *
  133. * bit |0 |1 |2 |3 |4 |5 |6 |7
  134. * -----+--------+--------+--------+--------+--------+--------+--------+--------
  135. * desc |Lcontrol|Lshift |Lalt |Lgui |Rcontrol|Rshift |Ralt |Rgui
  136. *
  137. */
  138. typedef union {
  139. uint8_t raw[KEYBOARD_REPORT_SIZE];
  140. struct {
  141. #ifdef KEYBOARD_SHARED_EP
  142. uint8_t report_id;
  143. #endif
  144. uint8_t mods;
  145. uint8_t reserved;
  146. uint8_t keys[KEYBOARD_REPORT_KEYS];
  147. };
  148. #ifdef NKRO_ENABLE
  149. struct nkro_report {
  150. # ifdef NKRO_SHARED_EP
  151. uint8_t report_id;
  152. # endif
  153. uint8_t mods;
  154. uint8_t bits[KEYBOARD_REPORT_BITS];
  155. } nkro;
  156. #endif
  157. } __attribute__((packed)) report_keyboard_t;
  158. typedef struct {
  159. uint8_t report_id;
  160. uint16_t usage;
  161. } __attribute__((packed)) report_extra_t;
  162. typedef struct {
  163. #ifdef MOUSE_SHARED_EP
  164. uint8_t report_id;
  165. #endif
  166. uint8_t buttons;
  167. int8_t x;
  168. int8_t y;
  169. int8_t v;
  170. int8_t h;
  171. } __attribute__((packed)) report_mouse_t;
  172. /* keycode to system usage */
  173. static inline uint16_t KEYCODE2SYSTEM(uint8_t key) {
  174. switch (key) {
  175. case KC_SYSTEM_POWER:
  176. return SYSTEM_POWER_DOWN;
  177. case KC_SYSTEM_SLEEP:
  178. return SYSTEM_SLEEP;
  179. case KC_SYSTEM_WAKE:
  180. return SYSTEM_WAKE_UP;
  181. default:
  182. return 0;
  183. }
  184. }
  185. /* keycode to consumer usage */
  186. static inline uint16_t KEYCODE2CONSUMER(uint8_t key) {
  187. switch (key) {
  188. case KC_AUDIO_MUTE:
  189. return AUDIO_MUTE;
  190. case KC_AUDIO_VOL_UP:
  191. return AUDIO_VOL_UP;
  192. case KC_AUDIO_VOL_DOWN:
  193. return AUDIO_VOL_DOWN;
  194. case KC_MEDIA_NEXT_TRACK:
  195. return TRANSPORT_NEXT_TRACK;
  196. case KC_MEDIA_PREV_TRACK:
  197. return TRANSPORT_PREV_TRACK;
  198. case KC_MEDIA_FAST_FORWARD:
  199. return TRANSPORT_FAST_FORWARD;
  200. case KC_MEDIA_REWIND:
  201. return TRANSPORT_REWIND;
  202. case KC_MEDIA_STOP:
  203. return TRANSPORT_STOP;
  204. case KC_MEDIA_EJECT:
  205. return TRANSPORT_STOP_EJECT;
  206. case KC_MEDIA_PLAY_PAUSE:
  207. return TRANSPORT_PLAY_PAUSE;
  208. case KC_MEDIA_SELECT:
  209. return AL_CC_CONFIG;
  210. case KC_MAIL:
  211. return AL_EMAIL;
  212. case KC_CALCULATOR:
  213. return AL_CALCULATOR;
  214. case KC_MY_COMPUTER:
  215. return AL_LOCAL_BROWSER;
  216. case KC_WWW_SEARCH:
  217. return AC_SEARCH;
  218. case KC_WWW_HOME:
  219. return AC_HOME;
  220. case KC_WWW_BACK:
  221. return AC_BACK;
  222. case KC_WWW_FORWARD:
  223. return AC_FORWARD;
  224. case KC_WWW_STOP:
  225. return AC_STOP;
  226. case KC_WWW_REFRESH:
  227. return AC_REFRESH;
  228. case KC_BRIGHTNESS_UP:
  229. return BRIGHTNESS_UP;
  230. case KC_BRIGHTNESS_DOWN:
  231. return BRIGHTNESS_DOWN;
  232. case KC_WWW_FAVORITES:
  233. return AC_BOOKMARKS;
  234. default:
  235. return 0;
  236. }
  237. }
  238. uint8_t has_anykey(report_keyboard_t* keyboard_report);
  239. uint8_t get_first_key(report_keyboard_t* keyboard_report);
  240. bool is_key_pressed(report_keyboard_t* keyboard_report, uint8_t key);
  241. void add_key_byte(report_keyboard_t* keyboard_report, uint8_t code);
  242. void del_key_byte(report_keyboard_t* keyboard_report, uint8_t code);
  243. #ifdef NKRO_ENABLE
  244. void add_key_bit(report_keyboard_t* keyboard_report, uint8_t code);
  245. void del_key_bit(report_keyboard_t* keyboard_report, uint8_t code);
  246. #endif
  247. void add_key_to_report(report_keyboard_t* keyboard_report, uint8_t key);
  248. void del_key_from_report(report_keyboard_t* keyboard_report, uint8_t key);
  249. void clear_keys_from_report(report_keyboard_t* keyboard_report);
  250. #ifdef __cplusplus
  251. }
  252. #endif