fusb_hid.c 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554
  1. /*
  2. * Copyright : (C) 2022 Phytium Information Technology, Inc.
  3. * All Rights Reserved.
  4. *
  5. * This program is OPEN SOURCE software: you can redistribute it and/or modify it
  6. * under the terms of the Phytium Public License as published by the Phytium Technology Co.,Ltd,
  7. * either version 1.0 of the License, or (at your option) any later version.
  8. *
  9. * This program is distributed in the hope that it will be useful,but WITHOUT ANY WARRANTY;
  10. * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
  11. * See the Phytium Public License for more details.
  12. *
  13. *
  14. * FilePath: fusb_hid.c
  15. * Date: 2022-09-28 18:26:42
  16. * LastEditTime: 2022-09-29 14:50:09
  17. * Description:  This files is for
  18. *
  19. * Modify History:
  20. * Ver   Who        Date         Changes
  21. * ----- ------     --------    --------------------------------------
  22. */
  23. #include <string.h>
  24. #include "fkernel.h"
  25. #include "fdebug.h"
  26. #include "fsleep.h"
  27. #include "fusb.h"
  28. #include "fusb_hid.h"
  29. enum FUsbHidRequests
  30. {
  31. GET_REPORT = 0x1,
  32. GET_IDLE = 0x2,
  33. GET_PROTOCOL = 0x3,
  34. SET_REPORT = 0x9,
  35. SET_IDLE = 0xa,
  36. SET_PROTOCOL = 0xb
  37. };
  38. enum FUsbHidKeyboardModifiers
  39. {
  40. KB_MOD_SHIFT = (1 << 0),
  41. KB_MOD_ALT = (1 << 1),
  42. KB_MOD_CTRL = (1 << 2),
  43. KB_MOD_CAPSLOCK = (1 << 3),
  44. };
  45. typedef union
  46. {
  47. struct
  48. {
  49. u8 modifiers;
  50. u8 repeats;
  51. u8 keys[6];
  52. };
  53. u8 buffer[8];
  54. } FUsbHidKeyboardEvent;
  55. typedef struct
  56. {
  57. void *queue;
  58. FUsbHidDescriptor *descriptor;
  59. FUsbHidKeyboardEvent previous;
  60. int lastkeypress;
  61. int repeat_delay;
  62. } FUsbHid;
  63. #define KEYBOARD_REPEAT_MS 30
  64. #define INITIAL_REPEAT_DELAY 10
  65. #define REPEAT_DELAY 2
  66. #define FUSB_HID_INST(dev) ((FUsbHid*)(dev)->data)
  67. #define FUSB_DEBUG_TAG "FUSB_HID"
  68. #define FUSB_ERROR(format, ...) FT_DEBUG_PRINT_E(FUSB_DEBUG_TAG, format, ##__VA_ARGS__)
  69. #define FUSB_WARN(format, ...) FT_DEBUG_PRINT_W(FUSB_DEBUG_TAG, format, ##__VA_ARGS__)
  70. #define FUSB_INFO(format, ...) FT_DEBUG_PRINT_I(FUSB_DEBUG_TAG, format, ##__VA_ARGS__)
  71. #define FUSB_DEBUG(format, ...) FT_DEBUG_PRINT_D(FUSB_DEBUG_TAG, format, ##__VA_ARGS__)
  72. static const char *boot_protos[3] = { "(none)", "keyboard", "mouse" };
  73. static void FUsbHidDestory(FUsbDev *dev)
  74. {
  75. FUsb *instance = dev->controller->usb;
  76. if (FUSB_HID_INST(dev)->queue)
  77. {
  78. int i;
  79. for (i = 0; i <= dev->num_endp; i++)
  80. {
  81. if (dev->endpoints[i].endpoint == 0)
  82. continue;
  83. if (dev->endpoints[i].type != FUSB_INTERRUPT_EP)
  84. continue;
  85. if (dev->endpoints[i].direction != FUSB_IN)
  86. continue;
  87. break;
  88. }
  89. dev->controller->destroy_intr_queue(
  90. &dev->endpoints[i], FUSB_HID_INST(dev)->queue);
  91. FUSB_HID_INST(dev)->queue = NULL;
  92. }
  93. FUSB_FREE(instance, FUSB_HID_INST(dev)->descriptor);
  94. FUSB_HID_INST(dev)->descriptor = NULL;
  95. FUSB_FREE(instance, dev->data);
  96. }
  97. /* keybuffer is global to all USB keyboards */
  98. static int keycount;
  99. #define KEYBOARD_BUFFER_SIZE 16
  100. static short keybuffer[KEYBOARD_BUFFER_SIZE];
  101. static int modifiers;
  102. static const char *countries[36][2] =
  103. {
  104. { "not supported", "us" },
  105. { "Arabic", "ae" },
  106. { "Belgian", "be" },
  107. { "Canadian-Bilingual", "ca" },
  108. { "Canadian-French", "ca" },
  109. { "Czech Republic", "cz" },
  110. { "Danish", "dk" },
  111. { "Finnish", "fi" },
  112. { "French", "fr" },
  113. { "German", "de" },
  114. { "Greek", "gr" },
  115. { "Hebrew", "il" },
  116. { "Hungary", "hu" },
  117. { "International (ISO)", "iso" },
  118. { "Italian", "it" },
  119. { "Japan (Katakana)", "jp" },
  120. { "Korean", "us" },
  121. { "Latin American", "us" },
  122. { "Netherlands/Dutch", "nl" },
  123. { "Norwegian", "no" },
  124. { "Persian (Farsi)", "ir" },
  125. { "Poland", "pl" },
  126. { "Portuguese", "pt" },
  127. { "Russia", "ru" },
  128. { "Slovakia", "sl" },
  129. { "Spanish", "es" },
  130. { "Swedish", "se" },
  131. { "Swiss/French", "ch" },
  132. { "Swiss/German", "ch" },
  133. { "Switzerland", "ch" },
  134. { "Taiwan", "tw" },
  135. { "Turkish-Q", "tr" },
  136. { "UK", "uk" },
  137. { "US", "us" },
  138. { "Yugoslavia", "yu" },
  139. { "Turkish-F", "tr" },
  140. /* 36 - 255: Reserved */
  141. };
  142. struct FUsbHidLayoutMaps
  143. {
  144. const char *country;
  145. const short map[4][0x80];
  146. };
  147. static const struct FUsbHidLayoutMaps *map;
  148. static const struct FUsbHidLayoutMaps keyboard_layouts[] =
  149. {
  150. {
  151. .country = "us",
  152. .map = {
  153. { /* No modifier */
  154. -1, -1, -1, -1, 'a', 'b', 'c', 'd',
  155. 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l',
  156. /* 0x10 */
  157. 'm', 'n', 'o', 'p', 'q', 'r', 's', 't',
  158. 'u', 'v', 'w', 'x', 'y', 'z', '1', '2',
  159. /* 0x20 */
  160. '3', '4', '5', '6', '7', '8', '9', '0',
  161. '\n', '\e', '\b', '\t', ' ', '-', '=', '[',
  162. /* 0x30 */
  163. ']', '\\', -1, ';', '\'', '`', ',', '.',
  164. '/', -1 /* CapsLk */, KEY_F(1), KEY_F(2), KEY_F(3), KEY_F(4), KEY_F(5), KEY_F(6),
  165. /* 0x40 */
  166. KEY_F(7), KEY_F(8), KEY_F(9), KEY_F(10), KEY_F(11), KEY_F(12), KEY_PRINT, -1 /* ScrLk */,
  167. KEY_BREAK, KEY_IC, KEY_HOME, KEY_PPAGE, KEY_DC, KEY_END, KEY_NPAGE, KEY_RIGHT,
  168. /* 50 */
  169. KEY_LEFT, KEY_DOWN, KEY_UP, -1 /*NumLck*/, '/', '*', '-' /* = ? */, '+',
  170. KEY_ENTER, KEY_END, KEY_DOWN, KEY_NPAGE, KEY_LEFT, -1, KEY_RIGHT, KEY_HOME,
  171. /* 60 */
  172. KEY_UP, KEY_PPAGE, -1, KEY_DC, -1 /* < > | */, -1 /* Win Key Right */, -1, -1,
  173. -1, -1, -1, -1, -1, -1, -1, -1,
  174. /* 70 */
  175. -1, -1, -1, -1, -1, -1, -1, -1,
  176. -1, -1, -1, -1, -1, -1, -1, -1,
  177. },
  178. { /* Shift modifier */
  179. -1, -1, -1, -1, 'A', 'B', 'C', 'D',
  180. 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L',
  181. /* 0x10 */
  182. 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T',
  183. 'U', 'V', 'W', 'X', 'Y', 'Z', '!', '@',
  184. /* 0x20 */
  185. '#', '$', '%', '^', '&', '*', '(', ')',
  186. '\n', '\e', '\b', '\t', ' ', '_', '+', '[',
  187. /* 0x30 */
  188. ']', '\\', -1, ':', '\'', '`', ',', '.',
  189. '/', -1 /* CapsLk */, KEY_F(1), KEY_F(2), KEY_F(3), KEY_F(4), KEY_F(5), KEY_F(6),
  190. /* 0x40 */
  191. KEY_F(7), KEY_F(8), KEY_F(9), KEY_F(10), KEY_F(11), KEY_F(12), KEY_PRINT, -1 /* ScrLk */,
  192. KEY_BREAK, KEY_IC, KEY_HOME, KEY_PPAGE, KEY_DC, KEY_END, KEY_NPAGE, KEY_RIGHT,
  193. /* 50 */
  194. KEY_LEFT, KEY_DOWN, KEY_UP, -1 /*NumLck*/, '/', '*', '-' /* = ? */, '+',
  195. KEY_ENTER, KEY_END, KEY_DOWN, KEY_NPAGE, KEY_LEFT, -1, KEY_RIGHT, KEY_HOME,
  196. /* 60 */
  197. KEY_UP, KEY_PPAGE, -1, KEY_DC, -1 /* < > | */, -1 /* Win Key Right */, -1, -1,
  198. -1, -1, -1, -1, -1, -1, -1, -1,
  199. /* 70 */
  200. -1, -1, -1, -1, -1, -1, -1, -1,
  201. -1, -1, -1, -1, -1, -1, -1, -1,
  202. },
  203. { /* Alt */
  204. -1, -1, -1, -1, 'a', 'b', 'c', 'd',
  205. 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l',
  206. /* 0x10 */
  207. 'm', 'n', 'o', 'p', 'q', 'r', 's', 't',
  208. 'u', 'v', 'w', 'x', 'y', 'z', '1', '2',
  209. /* 0x20 */
  210. '3', '4', '5', '6', '7', '8', '9', '0',
  211. '\n', '\e', '\b', '\t', ' ', '-', '=', '[',
  212. /* 0x30 */
  213. ']', '\\', -1, ';', '\'', '`', ',', '.',
  214. '/', -1 /* CapsLk */, KEY_F(1), KEY_F(2), KEY_F(3), KEY_F(4), KEY_F(5), KEY_F(6),
  215. /* 0x40 */
  216. KEY_F(7), KEY_F(8), KEY_F(9), KEY_F(10), KEY_F(11), KEY_F(12), KEY_PRINT, -1 /* ScrLk */,
  217. KEY_BREAK, KEY_IC, KEY_HOME, KEY_PPAGE, KEY_DC, KEY_END, KEY_NPAGE, KEY_RIGHT,
  218. /* 50 */
  219. KEY_LEFT, KEY_DOWN, KEY_UP, -1 /*NumLck*/, '/', '*', '-' /* = ? */, '+',
  220. KEY_ENTER, KEY_END, KEY_DOWN, KEY_NPAGE, KEY_LEFT, -1, KEY_RIGHT, KEY_HOME,
  221. /* 60 */
  222. KEY_UP, KEY_PPAGE, -1, KEY_DC, -1 /* < > | */, -1 /* Win Key Right */, -1, -1,
  223. -1, -1, -1, -1, -1, -1, -1, -1,
  224. /* 70 */
  225. -1, -1, -1, -1, -1, -1, -1, -1,
  226. -1, -1, -1, -1, -1, -1, -1, -1,
  227. },
  228. { /* Shift+Alt modifier */
  229. -1, -1, -1, -1, 'A', 'B', 'C', 'D',
  230. 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L',
  231. /* 0x10 */
  232. 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T',
  233. 'U', 'V', 'W', 'X', 'Y', 'Z', '!', '@',
  234. /* 0x20 */
  235. '#', '$', '%', '^', '&', '*', '(', ')',
  236. '\n', '\e', '\b', '\t', ' ', '-', '=', '[',
  237. /* 0x30 */
  238. ']', '\\', -1, ':', '\'', '`', ',', '.',
  239. '/', -1 /* CapsLk */, KEY_F(1), KEY_F(2), KEY_F(3), KEY_F(4), KEY_F(5), KEY_F(6),
  240. /* 0x40 */
  241. KEY_F(7), KEY_F(8), KEY_F(9), KEY_F(10), KEY_F(11), KEY_F(12), KEY_PRINT, -1 /* ScrLk */,
  242. KEY_BREAK, KEY_IC, KEY_HOME, KEY_PPAGE, KEY_DC, KEY_END, KEY_NPAGE, KEY_RIGHT,
  243. /* 50 */
  244. KEY_LEFT, KEY_DOWN, KEY_UP, -1 /*NumLck*/, '/', '*', '-' /* = ? */, '+',
  245. KEY_ENTER, KEY_END, KEY_DOWN, KEY_NPAGE, KEY_LEFT, -1, KEY_RIGHT, KEY_HOME,
  246. /* 60 */
  247. KEY_UP, KEY_PPAGE, -1, KEY_DC, -1 /* < > | */, -1 /* Win Key Right */, -1, -1,
  248. -1, -1, -1, -1, -1, -1, -1, -1,
  249. /* 70 */
  250. -1, -1, -1, -1, -1, -1, -1, -1,
  251. -1, -1, -1, -1, -1, -1, -1, -1,
  252. }
  253. }
  254. },
  255. };
  256. static void FUsbHidKeyboardQueue(int ch)
  257. {
  258. /* ignore key presses if buffer full */
  259. if (keycount < KEYBOARD_BUFFER_SIZE)
  260. keybuffer[keycount++] = ch;
  261. }
  262. /* handle hid received data */
  263. static void FUsbHidProcessKeyboardEvent(FUsbHid *const inst,
  264. const FUsbHidKeyboardEvent *const current)
  265. {
  266. const FUsbHidKeyboardEvent *const previous = &inst->previous;
  267. int i, keypress = 0;
  268. modifiers = 0;
  269. if (current->modifiers & 0x01) /* Left-Ctrl */
  270. modifiers |= KB_MOD_CTRL;
  271. if (current->modifiers & 0x02) /* Left-Shift */
  272. modifiers |= KB_MOD_SHIFT;
  273. if (current->modifiers & 0x04) /* Left-Alt */
  274. modifiers |= KB_MOD_ALT;
  275. if (current->modifiers & 0x08) /* Left-GUI */
  276. {
  277. }
  278. if (current->modifiers & 0x10) /* Right-Ctrl */
  279. modifiers |= KB_MOD_CTRL;
  280. if (current->modifiers & 0x20) /* Right-Shift */
  281. modifiers |= KB_MOD_SHIFT;
  282. if (current->modifiers & 0x40) /* Right-AltGr */
  283. modifiers |= KB_MOD_ALT;
  284. if (current->modifiers & 0x80) /* Right-GUI */
  285. {
  286. }
  287. if ((current->modifiers & 0x05) && ((current->keys[0] == 0x4c) ||
  288. (current->keys[0] == 0x63)))
  289. {
  290. /* todo, Reboot here */
  291. }
  292. /* Did the event change at all? */
  293. if (inst->lastkeypress &&
  294. !memcmp(current, previous, sizeof(*current)))
  295. {
  296. /* No. Then it's a key repeat event. */
  297. if (inst->repeat_delay)
  298. {
  299. inst->repeat_delay--;
  300. }
  301. else
  302. {
  303. FUsbHidKeyboardQueue(inst->lastkeypress);
  304. inst->repeat_delay = REPEAT_DELAY;
  305. }
  306. return;
  307. }
  308. inst->lastkeypress = 0;
  309. for (i = 0; i < 6; i++)
  310. {
  311. int j;
  312. int skip = 0;
  313. /* No more keys? skip */
  314. if (current->keys[i] == 0)
  315. return;
  316. for (j = 0; j < 6; j++)
  317. {
  318. if (current->keys[i] == previous->keys[j])
  319. {
  320. skip = 1;
  321. break;
  322. }
  323. }
  324. if (skip)
  325. continue;
  326. /* Mask off KB_MOD_CTRL */
  327. keypress = map->map[modifiers & 0x03][current->keys[i]];
  328. if (modifiers & KB_MOD_CTRL)
  329. {
  330. switch (keypress)
  331. {
  332. case 'a' ... 'z':
  333. keypress &= 0x1f;
  334. break;
  335. default:
  336. continue;
  337. }
  338. }
  339. if (keypress == -1)
  340. {
  341. /* Debug: Print unknown keys */
  342. FUSB_INFO("usbhid: <%x> %x [ %x %x %x %x %x %x ] %d\n",
  343. current->modifiers, current->repeats,
  344. current->keys[0], current->keys[1],
  345. current->keys[2], current->keys[3],
  346. current->keys[4], current->keys[5], i);
  347. /* Unknown key? Try next one in the queue */
  348. continue;
  349. }
  350. FUsbHidKeyboardQueue(keypress);
  351. /* Remember for authentic key repeat */
  352. inst->lastkeypress = keypress;
  353. inst->repeat_delay = INITIAL_REPEAT_DELAY;
  354. }
  355. }
  356. static void FUsbHidPoll(FUsbDev *dev)
  357. {
  358. FUsbHidKeyboardEvent current;
  359. const u8 *buf;
  360. while ((buf = dev->controller->poll_intr_queue(FUSB_HID_INST(dev)->queue)))
  361. {
  362. memcpy(&current.buffer, buf, 8);
  363. FUsbHidProcessKeyboardEvent(FUSB_HID_INST(dev), &current);
  364. FUSB_HID_INST(dev)->previous = current;
  365. }
  366. }
  367. static void FUsbHidSetIdle(FUsbDev *dev, FUsbInterfaceDescriptor *interface, u16 duration)
  368. {
  369. FUsbDevReq dr;
  370. dr.data_dir = FUSB_REQ_HOST_TO_DEVICE;
  371. dr.req_type = FUSB_REQ_TYPE_CLASS;
  372. dr.req_recp = FUSB_REQ_RECP_IF;
  373. dr.bRequest = SET_IDLE;
  374. dr.wValue = (duration >> 2) << 8;
  375. dr.wIndex = interface->bInterfaceNumber;
  376. dr.wLength = 0;
  377. dev->controller->control(dev, FUSB_OUT, sizeof(FUsbDevReq), &dr, 0, NULL);
  378. }
  379. static void FUsbHidSetProtocol(FUsbDev *dev, FUsbInterfaceDescriptor *interface, FUsbHidProtocol proto)
  380. {
  381. FUsbDevReq dr;
  382. dr.data_dir = FUSB_REQ_HOST_TO_DEVICE;
  383. dr.req_type = FUSB_REQ_TYPE_CLASS;
  384. dr.req_recp = FUSB_REQ_RECP_IF;
  385. dr.bRequest = SET_PROTOCOL;
  386. dr.wValue = proto;
  387. dr.wIndex = interface->bInterfaceNumber;
  388. dr.wLength = 0;
  389. dev->controller->control(dev, FUSB_OUT, sizeof(FUsbDevReq), &dr, 0, 0);
  390. }
  391. static int FUsbHidSetLayout(const char *country)
  392. {
  393. /* FIXME should be per keyboard */
  394. for (fsize_t i = 0; i < ARRAY_SIZE(keyboard_layouts); i++)
  395. {
  396. if (strncmp(keyboard_layouts[i].country, country,
  397. strlen(keyboard_layouts[i].country)))
  398. continue;
  399. /* Found, changing keyboard layout */
  400. map = &keyboard_layouts[i];
  401. FUSB_DEBUG(" Keyboard layout '%s'\n", map->country);
  402. return 0;
  403. }
  404. FUSB_DEBUG(" Keyboard layout '%s' not found, using '%s'\n",
  405. country, map->country);
  406. /* Nothing found, not changed */
  407. return -1;
  408. }
  409. void FUsbHidInit(FUsbDev *dev)
  410. {
  411. FUsb *instance = dev->controller->usb;
  412. FUsbConfigurationDescriptor *cd = (FUsbConfigurationDescriptor *)dev->configuration;
  413. FUsbInterfaceDescriptor *interface = (FUsbInterfaceDescriptor *)(((char *) cd) + cd->bLength);
  414. if (interface->bInterfaceSubClass == FUSB_HID_SUBCLASS_BOOT)
  415. {
  416. u8 countrycode;
  417. FUSB_DEBUG(" supports boot interface..\n");
  418. FUSB_DEBUG(" it's a %s\n",
  419. boot_protos[interface->bInterfaceProtocol]);
  420. switch (interface->bInterfaceProtocol)
  421. {
  422. case FUSB_HID_BOOT_PROTOCOL_KEYBOARD:
  423. dev->data = FUSB_ALLOCATE(instance, sizeof(FUsbHid), FUSB_DEFAULT_ALIGN);
  424. FUSB_DEBUG(" configuring...\n");
  425. FUsbHidSetProtocol(dev, interface, FUSB_HID_PROTOCOL_BOOT);
  426. FUsbHidSetIdle(dev, interface, KEYBOARD_REPEAT_MS);
  427. FUSB_DEBUG(" activating...\n");
  428. FUsbHidDescriptor *desc = FUSB_ALLOCATE(instance, sizeof(FUsbHidDescriptor), FUSB_DEFAULT_ALIGN);
  429. if (!desc || FUsbGetDescriptor(dev, FUsbGenerateReqType(
  430. FUSB_REQ_DEVICE_TO_HOST, FUSB_REQ_TYPE_STANDARD, FUSB_REQ_RECP_IF),
  431. 0x21, 0, desc, sizeof(*desc)) != sizeof(*desc))
  432. {
  433. FUSB_DEBUG("FUsbGetDescriptor(HID) failed\n");
  434. FUsbDetachDev(dev->controller, dev->address);
  435. return;
  436. }
  437. FUSB_HID_INST(dev)->descriptor = desc;
  438. countrycode = desc->bCountryCode;
  439. /* 35 countries defined: */
  440. if (countrycode >= ARRAY_SIZE(countries))
  441. countrycode = 0;
  442. printf(" Keyboard has %s layout (country code %02x)\n",
  443. countries[countrycode][0], countrycode);
  444. /* Set keyboard layout accordingly */
  445. FUsbHidSetLayout(countries[countrycode][1]);
  446. // only add here, because we only support boot-keyboard HID devices
  447. dev->destroy = FUsbHidDestory;
  448. dev->poll = FUsbHidPoll;
  449. int i;
  450. for (i = 1; i < dev->num_endp; i++)
  451. {
  452. if (dev->endpoints[i].type != FUSB_INTERRUPT_EP)
  453. continue;
  454. if (dev->endpoints[i].direction != FUSB_IN)
  455. continue;
  456. break;
  457. }
  458. if (i >= dev->num_endp)
  459. {
  460. FUSB_DEBUG("Could not find HID endpoint\n");
  461. FUsbDetachDev(dev->controller, dev->address);
  462. return;
  463. }
  464. FUSB_DEBUG(" found endpoint %x for interrupt-in\n", i);
  465. /* 20 buffers of 8 bytes, for every 10 msecs */
  466. FUSB_HID_INST(dev)->queue = dev->controller->create_intr_queue(&dev->endpoints[i], 8, 20, 10);
  467. keycount = 0;
  468. FUSB_DEBUG(" configuration done.\n");
  469. break;
  470. case FUSB_HID_BOOT_PROTOCOL_MOUSE:
  471. FUSB_DEBUG("NOTICE: USB mice are not supported.\n");
  472. break;
  473. }
  474. }
  475. }
  476. int FUsbHidCheckInput(FUsbDev *dev, int times)
  477. {
  478. short ret;
  479. FUsb *instance = dev->controller->usb;
  480. for (int i = 0; i < times; i++)
  481. {
  482. FUsbPoll(instance);
  483. while (keycount != 0)
  484. {
  485. ret = keybuffer[0];
  486. memmove(keybuffer, keybuffer + 1, --keycount);
  487. printf("%c", ret);
  488. }
  489. fsleep_millisec(10);
  490. }
  491. printf("\r\n");
  492. }