| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | /* 
 | 
					
						
							|  |  |  |  * The MIT License (MIT) | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright (c) 2019 Ha Thach (tinyusb.org) | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Permission is hereby granted, free of charge, to any person obtaining a copy | 
					
						
							|  |  |  |  * of this software and associated documentation files (the "Software"), to deal | 
					
						
							|  |  |  |  * in the Software without restriction, including without limitation the rights | 
					
						
							|  |  |  |  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | 
					
						
							|  |  |  |  * copies of the Software, and to permit persons to whom the Software is | 
					
						
							|  |  |  |  * furnished to do so, subject to the following conditions: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The above copyright notice and this permission notice shall be included in | 
					
						
							|  |  |  |  * all copies or substantial portions of the Software. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | 
					
						
							|  |  |  |  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | 
					
						
							|  |  |  |  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | 
					
						
							|  |  |  |  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | 
					
						
							|  |  |  |  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | 
					
						
							|  |  |  |  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN | 
					
						
							|  |  |  |  * THE SOFTWARE. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This file is part of the TinyUSB stack. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "tusb_option.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-12 20:04:19 +07:00
										 |  |  | #if (TUSB_OPT_HOST_ENABLED && CFG_TUH_HID)
 | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-27 18:18:30 +07:00
										 |  |  | #include "host/usbh.h"
 | 
					
						
							| 
									
										
										
										
											2021-06-10 16:48:20 +07:00
										 |  |  | #include "host/usbh_classdriver.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | #include "hid_host.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // MACRO CONSTANT TYPEDEF
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-12 15:29:17 +07:00
										 |  |  | typedef struct | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2021-02-24 13:54:18 +07:00
										 |  |  |   uint8_t itf_num; | 
					
						
							|  |  |  |   uint8_t ep_in; | 
					
						
							|  |  |  |   uint8_t ep_out; | 
					
						
							| 
									
										
										
										
											2020-09-03 17:07:29 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-22 16:27:28 +07:00
										 |  |  |   uint8_t itf_protocol;   // None, Keyboard, Mouse
 | 
					
						
							|  |  |  |   uint8_t protocol_mode;  // Boot (0) or Report protocol (1)
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-12 19:36:52 +07:00
										 |  |  |   uint8_t  report_desc_type; | 
					
						
							| 
									
										
										
										
											2021-05-12 15:29:17 +07:00
										 |  |  |   uint16_t report_desc_len; | 
					
						
							| 
									
										
										
										
											2021-05-14 17:47:02 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   uint16_t epin_size; | 
					
						
							|  |  |  |   uint16_t epout_size; | 
					
						
							| 
									
										
										
										
											2021-02-24 13:54:18 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-14 17:47:02 +07:00
										 |  |  |   uint8_t epin_buf[CFG_TUH_HID_EP_BUFSIZE]; | 
					
						
							|  |  |  |   uint8_t epout_buf[CFG_TUH_HID_EP_BUFSIZE]; | 
					
						
							| 
									
										
										
										
											2021-05-14 01:02:47 +07:00
										 |  |  | } hidh_interface_t; | 
					
						
							| 
									
										
										
										
											2020-09-03 17:07:29 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-12 15:29:17 +07:00
										 |  |  | typedef struct | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2021-05-12 20:04:19 +07:00
										 |  |  |   uint8_t inst_count; | 
					
						
							|  |  |  |   hidh_interface_t instances[CFG_TUH_HID]; | 
					
						
							| 
									
										
										
										
											2021-05-12 15:29:17 +07:00
										 |  |  | } hidh_device_t; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static hidh_device_t _hidh_dev[CFG_TUSB_HOST_DEVICE_MAX-1]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-13 14:09:33 +07:00
										 |  |  | //------------- Internal prototypes -------------//
 | 
					
						
							| 
									
										
										
										
											2021-05-22 21:48:42 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  | // Get HID device & interface
 | 
					
						
							| 
									
										
										
										
											2021-05-12 20:19:00 +07:00
										 |  |  | TU_ATTR_ALWAYS_INLINE static inline hidh_device_t* get_dev(uint8_t dev_addr); | 
					
						
							|  |  |  | TU_ATTR_ALWAYS_INLINE static inline hidh_interface_t* get_instance(uint8_t dev_addr, uint8_t instance); | 
					
						
							| 
									
										
										
										
											2021-05-13 14:09:33 +07:00
										 |  |  | static uint8_t get_instance_id_by_itfnum(uint8_t dev_addr, uint8_t itf); | 
					
						
							|  |  |  | static uint8_t get_instance_id_by_epaddr(uint8_t dev_addr, uint8_t ep_addr); | 
					
						
							| 
									
										
										
										
											2021-05-12 15:29:17 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-14 17:47:02 +07:00
										 |  |  | TU_ATTR_ALWAYS_INLINE static inline bool hidh_get_report(uint8_t dev_addr, hidh_interface_t* hid_itf) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return usbh_edpt_xfer(dev_addr, hid_itf->ep_in, hid_itf->epin_buf, hid_itf->epin_size); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							| 
									
										
										
										
											2021-05-12 19:36:52 +07:00
										 |  |  | // Application API
 | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							| 
									
										
										
										
											2021-05-13 14:09:33 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-22 18:17:32 +07:00
										 |  |  | uint8_t tuh_hid_instance_count(uint8_t dev_addr) | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2021-05-13 14:09:33 +07:00
										 |  |  |   return get_dev(dev_addr)->inst_count; | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-22 18:17:32 +07:00
										 |  |  | bool tuh_hid_mounted(uint8_t dev_addr, uint8_t instance) | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2021-05-13 14:09:33 +07:00
										 |  |  |   hidh_interface_t* hid_itf = get_instance(dev_addr, instance); | 
					
						
							| 
									
										
										
										
											2021-05-13 13:42:52 +07:00
										 |  |  |   return (hid_itf->ep_in != 0) || (hid_itf->ep_out != 0); | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-22 18:17:32 +07:00
										 |  |  | uint8_t tuh_hid_interface_protocol(uint8_t dev_addr, uint8_t instance) | 
					
						
							| 
									
										
										
										
											2021-05-14 01:02:47 +07:00
										 |  |  | { | 
					
						
							|  |  |  |   hidh_interface_t* hid_itf = get_instance(dev_addr, instance); | 
					
						
							| 
									
										
										
										
											2021-05-18 13:12:33 +07:00
										 |  |  |   return hid_itf->itf_protocol; | 
					
						
							| 
									
										
										
										
											2021-05-14 01:02:47 +07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-17 12:35:51 +07:00
										 |  |  | uint8_t tuh_hid_get_protocol(uint8_t dev_addr, uint8_t instance) | 
					
						
							| 
									
										
										
										
											2021-05-14 01:02:47 +07:00
										 |  |  | { | 
					
						
							|  |  |  |   hidh_interface_t* hid_itf = get_instance(dev_addr, instance); | 
					
						
							| 
									
										
										
										
											2021-05-22 16:27:28 +07:00
										 |  |  |   return hid_itf->protocol_mode; | 
					
						
							| 
									
										
										
										
											2021-05-14 01:02:47 +07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-22 16:27:28 +07:00
										 |  |  | static bool set_protocol_complete(uint8_t dev_addr, tusb_control_request_t const * request, xfer_result_t result) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   uint8_t const itf_num     = (uint8_t) request->wIndex; | 
					
						
							|  |  |  |   uint8_t const instance    = get_instance_id_by_itfnum(dev_addr, itf_num); | 
					
						
							|  |  |  |   hidh_interface_t* hid_itf = get_instance(dev_addr, instance); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (XFER_RESULT_SUCCESS == result) hid_itf->protocol_mode = (uint8_t) request->wValue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (tuh_hid_set_protocol_complete_cb) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     tuh_hid_set_protocol_complete_cb(dev_addr, instance, hid_itf->protocol_mode); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-22 18:17:32 +07:00
										 |  |  | bool tuh_hid_set_protocol(uint8_t dev_addr, uint8_t instance, uint8_t protocol) | 
					
						
							| 
									
										
										
										
											2021-05-22 16:27:28 +07:00
										 |  |  | { | 
					
						
							|  |  |  |   hidh_interface_t* hid_itf = get_instance(dev_addr, instance); | 
					
						
							|  |  |  |   TU_VERIFY(hid_itf->itf_protocol != HID_ITF_PROTOCOL_NONE); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-22 21:48:42 +07:00
										 |  |  |   TU_LOG2("HID Set Protocol = %d\r\n", protocol); | 
					
						
							| 
									
										
										
										
											2021-05-22 16:27:28 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   tusb_control_request_t const request = | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     .bmRequestType_bit = | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       .recipient = TUSB_REQ_RCPT_INTERFACE, | 
					
						
							|  |  |  |       .type      = TUSB_REQ_TYPE_CLASS, | 
					
						
							|  |  |  |       .direction = TUSB_DIR_OUT | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     .bRequest = HID_REQ_CONTROL_SET_PROTOCOL, | 
					
						
							|  |  |  |     .wValue   = protocol, | 
					
						
							|  |  |  |     .wIndex   = hid_itf->itf_num, | 
					
						
							|  |  |  |     .wLength  = 0 | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   TU_ASSERT( tuh_control_xfer(dev_addr, &request, NULL, set_protocol_complete) ); | 
					
						
							|  |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-22 21:48:42 +07:00
										 |  |  | static bool set_report_complete(uint8_t dev_addr, tusb_control_request_t const * request, xfer_result_t result) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   TU_LOG2("HID Set Report complete\r\n"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (tuh_hid_set_report_complete_cb) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     uint8_t const itf_num     = (uint8_t) request->wIndex; | 
					
						
							|  |  |  |     uint8_t const instance    = get_instance_id_by_itfnum(dev_addr, itf_num); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     uint8_t const report_type = tu_u16_high(request->wValue); | 
					
						
							|  |  |  |     uint8_t const report_id   = tu_u16_low(request->wValue); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     tuh_hid_set_report_complete_cb(dev_addr, instance, report_id, report_type, (result == XFER_RESULT_SUCCESS) ? request->wLength : 0); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool tuh_hid_set_report(uint8_t dev_addr, uint8_t instance, uint8_t report_id, uint8_t report_type, void* report, uint16_t len) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   hidh_interface_t* hid_itf = get_instance(dev_addr, instance); | 
					
						
							|  |  |  |   TU_LOG2("HID Set Report: id = %u, type = %u, len = %u\r\n", report_id, report_type, len); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   tusb_control_request_t const request = | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     .bmRequestType_bit = | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       .recipient = TUSB_REQ_RCPT_INTERFACE, | 
					
						
							|  |  |  |       .type      = TUSB_REQ_TYPE_CLASS, | 
					
						
							|  |  |  |       .direction = TUSB_DIR_OUT | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     .bRequest = HID_REQ_CONTROL_SET_REPORT, | 
					
						
							|  |  |  |     .wValue   = tu_u16(report_type, report_id), | 
					
						
							|  |  |  |     .wIndex   = hid_itf->itf_num, | 
					
						
							|  |  |  |     .wLength  = len | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-22 22:43:55 +07:00
										 |  |  |   TU_ASSERT( tuh_control_xfer(dev_addr, &request, report, set_report_complete) ); | 
					
						
							| 
									
										
										
										
											2021-05-22 21:48:42 +07:00
										 |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-17 13:54:39 +07:00
										 |  |  | //bool tuh_n_hid_n_ready(uint8_t dev_addr, uint8_t instance)
 | 
					
						
							|  |  |  | //{
 | 
					
						
							|  |  |  | //  TU_VERIFY(tuh_n_hid_n_mounted(dev_addr, instance));
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | //  hidh_interface_t* hid_itf = get_instance(dev_addr, instance);
 | 
					
						
							| 
									
										
										
										
											2021-06-10 17:19:21 +07:00
										 |  |  | //  return !usbh_edpt_busy(dev_addr, hid_itf->ep_in);
 | 
					
						
							| 
									
										
										
										
											2021-05-17 13:54:39 +07:00
										 |  |  | //}
 | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-22 22:03:21 +07:00
										 |  |  | //void tuh_hid_send_report(uint8_t dev_addr, uint8_t instance, uint8_t report_id, uint8_t const* report, uint16_t len);
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							| 
									
										
										
										
											2021-05-12 19:36:52 +07:00
										 |  |  | // USBH API
 | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | void hidh_init(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2021-05-12 15:29:17 +07:00
										 |  |  |   tu_memclr(_hidh_dev, sizeof(_hidh_dev)); | 
					
						
							| 
									
										
										
										
											2021-05-12 19:36:52 +07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-18 13:08:30 +07:00
										 |  |  | bool hidh_xfer_cb(uint8_t dev_addr, uint8_t ep_addr, xfer_result_t result, uint32_t xferred_bytes) | 
					
						
							| 
									
										
										
										
											2021-05-12 19:36:52 +07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2021-05-18 13:08:30 +07:00
										 |  |  |   (void) result; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-13 14:09:33 +07:00
										 |  |  |   uint8_t const dir = tu_edpt_dir(ep_addr); | 
					
						
							|  |  |  |   uint8_t const instance = get_instance_id_by_epaddr(dev_addr, ep_addr); | 
					
						
							| 
									
										
										
										
											2021-05-14 17:47:02 +07:00
										 |  |  |   hidh_interface_t* hid_itf = get_instance(dev_addr, instance); | 
					
						
							| 
									
										
										
										
											2021-05-12 19:36:52 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-13 14:09:33 +07:00
										 |  |  |   if ( dir == TUSB_DIR_IN ) | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2021-05-17 13:54:39 +07:00
										 |  |  |     TU_LOG2("  Get Report callback (%u, %u)\r\n", dev_addr, instance); | 
					
						
							|  |  |  |     TU_LOG1_MEM(hid_itf->epin_buf, 8, 2); | 
					
						
							| 
									
										
										
										
											2021-05-22 21:51:30 +07:00
										 |  |  |     tuh_hid_report_received_cb(dev_addr, instance, hid_itf->epin_buf, xferred_bytes); | 
					
						
							| 
									
										
										
										
											2021-05-14 17:47:02 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // queue next report
 | 
					
						
							|  |  |  |     hidh_get_report(dev_addr, hid_itf); | 
					
						
							| 
									
										
										
										
											2021-05-13 14:09:33 +07:00
										 |  |  |   }else | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2021-05-22 22:03:21 +07:00
										 |  |  |     if (tuh_hid_report_sent_cb) tuh_hid_report_sent_cb(dev_addr, instance, hid_itf->epout_buf, xferred_bytes); | 
					
						
							| 
									
										
										
										
											2021-05-13 14:09:33 +07:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-05-12 19:36:52 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return true; | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-12 19:36:52 +07:00
										 |  |  | void hidh_close(uint8_t dev_addr) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2021-05-12 20:04:19 +07:00
										 |  |  |   hidh_device_t* hid_dev = get_dev(dev_addr); | 
					
						
							| 
									
										
										
										
											2021-05-22 21:48:42 +07:00
										 |  |  |   if (tuh_hid_umount_cb) | 
					
						
							| 
									
										
										
										
											2021-05-12 19:36:52 +07:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2021-05-23 14:11:12 +07:00
										 |  |  |     for ( uint8_t inst = 0; inst < hid_dev->inst_count; inst++ ) tuh_hid_umount_cb(dev_addr, inst); | 
					
						
							| 
									
										
										
										
											2021-05-12 19:36:52 +07:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-05-12 20:04:19 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   tu_memclr(hid_dev, sizeof(hidh_device_t)); | 
					
						
							| 
									
										
										
										
											2021-05-12 19:36:52 +07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // Enumeration
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-17 12:35:51 +07:00
										 |  |  | static bool config_set_protocol             (uint8_t dev_addr, tusb_control_request_t const * request, xfer_result_t result); | 
					
						
							| 
									
										
										
										
											2021-05-22 23:30:41 +07:00
										 |  |  | static bool config_get_report_desc          (uint8_t dev_addr, tusb_control_request_t const * request, xfer_result_t result); | 
					
						
							|  |  |  | static bool config_get_report_desc_complete (uint8_t dev_addr, tusb_control_request_t const * request, xfer_result_t result); | 
					
						
							| 
									
										
										
										
											2021-05-12 19:36:52 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-17 11:58:34 +07:00
										 |  |  | uint16_t hidh_open(uint8_t rhport, uint8_t dev_addr, tusb_desc_interface_t const *desc_itf, uint16_t max_len) | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2021-06-17 11:58:34 +07:00
										 |  |  |   (void) max_len; | 
					
						
							| 
									
										
										
										
											2021-02-24 13:19:32 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-17 11:58:34 +07:00
										 |  |  |   TU_VERIFY(TUSB_CLASS_HID == desc_itf->bInterfaceClass, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   uint16_t drv_len = sizeof(tusb_desc_interface_t); | 
					
						
							| 
									
										
										
										
											2021-02-24 13:19:32 +07:00
										 |  |  |   uint8_t const *p_desc = (uint8_t const *) desc_itf; | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |   //------------- HID descriptor -------------//
 | 
					
						
							| 
									
										
										
										
											2021-02-24 13:19:32 +07:00
										 |  |  |   p_desc = tu_desc_next(p_desc); | 
					
						
							|  |  |  |   tusb_hid_descriptor_hid_t const *desc_hid = (tusb_hid_descriptor_hid_t const *) p_desc; | 
					
						
							| 
									
										
										
										
											2021-06-17 11:58:34 +07:00
										 |  |  |   TU_ASSERT(HID_DESC_TYPE_HID == desc_hid->bDescriptorType, 0); | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-12 15:29:17 +07:00
										 |  |  |   // not enough interface, try to increase CFG_TUH_HID
 | 
					
						
							|  |  |  |   // TODO multiple devices
 | 
					
						
							|  |  |  |   hidh_device_t* hid_dev = get_dev(dev_addr); | 
					
						
							| 
									
										
										
										
											2021-06-17 11:58:34 +07:00
										 |  |  |   TU_ASSERT(hid_dev->inst_count < CFG_TUH_HID, 0); | 
					
						
							| 
									
										
										
										
											2021-05-12 15:29:17 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  |   //------------- Endpoint Descriptor -------------//
 | 
					
						
							| 
									
										
										
										
											2021-06-17 11:58:34 +07:00
										 |  |  |   drv_len += tu_desc_len(p_desc); | 
					
						
							| 
									
										
										
										
											2021-02-24 13:19:32 +07:00
										 |  |  |   p_desc = tu_desc_next(p_desc); | 
					
						
							|  |  |  |   tusb_desc_endpoint_t const * desc_ep = (tusb_desc_endpoint_t const *) p_desc; | 
					
						
							| 
									
										
										
										
											2021-06-17 11:58:34 +07:00
										 |  |  |   TU_ASSERT(TUSB_DESC_ENDPOINT == desc_ep->bDescriptorType, 0); | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-12 19:36:52 +07:00
										 |  |  |   // TODO also open endpoint OUT
 | 
					
						
							| 
									
										
										
										
											2021-06-17 11:58:34 +07:00
										 |  |  |   TU_ASSERT( usbh_edpt_open(rhport, dev_addr, desc_ep), 0 ); | 
					
						
							| 
									
										
										
										
											2021-05-12 19:36:52 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-12 20:04:19 +07:00
										 |  |  |   hidh_interface_t* hid_itf = get_instance(dev_addr, hid_dev->inst_count); | 
					
						
							|  |  |  |   hid_dev->inst_count++; | 
					
						
							| 
									
										
										
										
											2021-05-12 19:36:52 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-17 13:54:39 +07:00
										 |  |  |   hid_itf->itf_num   = desc_itf->bInterfaceNumber; | 
					
						
							|  |  |  |   hid_itf->ep_in     = desc_ep->bEndpointAddress; | 
					
						
							| 
									
										
										
										
											2021-05-14 17:47:02 +07:00
										 |  |  |   hid_itf->epin_size = desc_ep->wMaxPacketSize.size; | 
					
						
							| 
									
										
										
										
											2021-05-12 19:36:52 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // Assume bNumDescriptors = 1
 | 
					
						
							|  |  |  |   hid_itf->report_desc_type = desc_hid->bReportType; | 
					
						
							| 
									
										
										
										
											2021-05-17 13:54:39 +07:00
										 |  |  |   hid_itf->report_desc_len  = tu_unaligned_read16(&desc_hid->wReportLength); | 
					
						
							| 
									
										
										
										
											2021-05-12 19:36:52 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-17 12:35:51 +07:00
										 |  |  |   // Per HID Specs: default is Report protocol, though we will force Boot protocol when set_config
 | 
					
						
							|  |  |  |   hid_itf->protocol_mode = HID_PROTOCOL_BOOT; | 
					
						
							| 
									
										
										
										
											2021-05-22 16:27:28 +07:00
										 |  |  |   if ( HID_SUBCLASS_BOOT == desc_itf->bInterfaceSubClass ) hid_itf->itf_protocol = desc_itf->bInterfaceProtocol; | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-17 11:58:34 +07:00
										 |  |  |   drv_len += desc_itf->bNumEndpoints*sizeof(tusb_desc_endpoint_t); | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-17 11:58:34 +07:00
										 |  |  |   return drv_len; | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-01 17:46:46 +07:00
										 |  |  | bool hidh_set_config(uint8_t dev_addr, uint8_t itf_num) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2021-05-22 23:30:41 +07:00
										 |  |  |   uint8_t const instance    = get_instance_id_by_itfnum(dev_addr, itf_num); | 
					
						
							|  |  |  |   hidh_interface_t* hid_itf = get_instance(dev_addr, instance); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-12 19:36:52 +07:00
										 |  |  |   // Idle rate = 0 mean only report when there is changes
 | 
					
						
							|  |  |  |   uint16_t const idle_rate = 0; | 
					
						
							| 
									
										
										
										
											2020-11-01 17:46:46 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-12 19:36:52 +07:00
										 |  |  |   // SET IDLE request, device can stall if not support this request
 | 
					
						
							| 
									
										
										
										
											2021-05-22 23:30:41 +07:00
										 |  |  |   TU_LOG2("HID Set Idle \r\n"); | 
					
						
							| 
									
										
										
										
											2020-11-01 17:46:46 +07:00
										 |  |  |   tusb_control_request_t const request = | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     .bmRequestType_bit = | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       .recipient = TUSB_REQ_RCPT_INTERFACE, | 
					
						
							|  |  |  |       .type      = TUSB_REQ_TYPE_CLASS, | 
					
						
							|  |  |  |       .direction = TUSB_DIR_OUT | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     .bRequest = HID_REQ_CONTROL_SET_IDLE, | 
					
						
							| 
									
										
										
										
											2021-05-12 19:36:52 +07:00
										 |  |  |     .wValue   = idle_rate, | 
					
						
							|  |  |  |     .wIndex   = itf_num, | 
					
						
							| 
									
										
										
										
											2020-11-01 17:46:46 +07:00
										 |  |  |     .wLength  = 0 | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-17 12:35:51 +07:00
										 |  |  |   TU_ASSERT( tuh_control_xfer(dev_addr, &request, NULL, (hid_itf->itf_protocol != HID_ITF_PROTOCOL_NONE) ? config_set_protocol : config_get_report_desc) ); | 
					
						
							| 
									
										
										
										
											2020-11-01 17:46:46 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-17 12:35:51 +07:00
										 |  |  | // Force device to work in BOOT protocol
 | 
					
						
							|  |  |  | static bool config_set_protocol(uint8_t dev_addr, tusb_control_request_t const * request, xfer_result_t result) | 
					
						
							| 
									
										
										
										
											2021-05-22 23:30:41 +07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2021-06-28 23:57:57 +07:00
										 |  |  |   // Stall is a valid response for SET_IDLE, therefore we could ignore its result
 | 
					
						
							| 
									
										
										
										
											2021-05-22 23:30:41 +07:00
										 |  |  |   (void) result; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   uint8_t const itf_num     = (uint8_t) request->wIndex; | 
					
						
							|  |  |  |   uint8_t const instance    = get_instance_id_by_itfnum(dev_addr, itf_num); | 
					
						
							|  |  |  |   hidh_interface_t* hid_itf = get_instance(dev_addr, instance); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-17 12:35:51 +07:00
										 |  |  |   TU_LOG2("HID Set Protocol\r\n"); | 
					
						
							|  |  |  |   hid_itf->protocol_mode = HID_PROTOCOL_BOOT; | 
					
						
							| 
									
										
										
										
											2021-05-22 23:30:41 +07:00
										 |  |  |   tusb_control_request_t const new_request = | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     .bmRequestType_bit = | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       .recipient = TUSB_REQ_RCPT_INTERFACE, | 
					
						
							|  |  |  |       .type      = TUSB_REQ_TYPE_CLASS, | 
					
						
							| 
									
										
										
										
											2021-06-17 12:35:51 +07:00
										 |  |  |       .direction = TUSB_DIR_OUT | 
					
						
							| 
									
										
										
										
											2021-05-22 23:30:41 +07:00
										 |  |  |     }, | 
					
						
							| 
									
										
										
										
											2021-06-17 12:35:51 +07:00
										 |  |  |     .bRequest = HID_REQ_CONTROL_SET_PROTOCOL, | 
					
						
							|  |  |  |     .wValue   = HID_PROTOCOL_BOOT, | 
					
						
							| 
									
										
										
										
											2021-05-22 23:30:41 +07:00
										 |  |  |     .wIndex   = hid_itf->itf_num, | 
					
						
							| 
									
										
										
										
											2021-06-28 23:57:57 +07:00
										 |  |  |     .wLength  = 0 | 
					
						
							| 
									
										
										
										
											2021-05-22 23:30:41 +07:00
										 |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-17 12:35:51 +07:00
										 |  |  |   TU_ASSERT( tuh_control_xfer(dev_addr, &new_request, NULL, config_get_report_desc) ); | 
					
						
							| 
									
										
										
										
											2021-05-22 23:30:41 +07:00
										 |  |  |   return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static bool config_get_report_desc(uint8_t dev_addr, tusb_control_request_t const * request, xfer_result_t result) | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2021-06-28 23:57:57 +07:00
										 |  |  |   // We can be here after SET_IDLE or SET_PROTOCOL (boot device)
 | 
					
						
							|  |  |  |   // Trigger assert if result is not successful with set protocol
 | 
					
						
							|  |  |  |   if ( request->bRequest != HID_REQ_CONTROL_SET_IDLE ) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     TU_ASSERT(result == XFER_RESULT_SUCCESS); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-22 16:27:28 +07:00
										 |  |  |   uint8_t const itf_num     = (uint8_t) request->wIndex; | 
					
						
							|  |  |  |   uint8_t const instance    = get_instance_id_by_itfnum(dev_addr, itf_num); | 
					
						
							|  |  |  |   hidh_interface_t* hid_itf = get_instance(dev_addr, instance); | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-12 19:36:52 +07:00
										 |  |  |   // Get Report Descriptor
 | 
					
						
							|  |  |  |   // using usbh enumeration buffer since report descriptor can be very long
 | 
					
						
							| 
									
										
										
										
											2021-06-21 05:32:43 +02:00
										 |  |  |   TU_ASSERT( hid_itf->report_desc_len <= CFG_TUH_ENUMERATION_BUFSIZE ); | 
					
						
							| 
									
										
										
										
											2021-05-12 19:36:52 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   TU_LOG2("HID Get Report Descriptor\r\n"); | 
					
						
							|  |  |  |   tusb_control_request_t const new_request = | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2021-05-12 19:36:52 +07:00
										 |  |  |     .bmRequestType_bit = | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       .recipient = TUSB_REQ_RCPT_INTERFACE, | 
					
						
							|  |  |  |       .type      = TUSB_REQ_TYPE_STANDARD, | 
					
						
							|  |  |  |       .direction = TUSB_DIR_IN | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     .bRequest = TUSB_REQ_GET_DESCRIPTOR, | 
					
						
							|  |  |  |     .wValue   = tu_u16(hid_itf->report_desc_type, 0), | 
					
						
							|  |  |  |     .wIndex   = itf_num, | 
					
						
							|  |  |  |     .wLength  = hid_itf->report_desc_len | 
					
						
							|  |  |  |   }; | 
					
						
							| 
									
										
										
										
											2020-09-06 12:11:07 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-12 19:36:52 +07:00
										 |  |  |   TU_ASSERT(tuh_control_xfer(dev_addr, &new_request, usbh_get_enum_buf(), config_get_report_desc_complete)); | 
					
						
							| 
									
										
										
										
											2020-09-06 12:11:07 +07:00
										 |  |  |   return true; | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-22 23:30:41 +07:00
										 |  |  | static bool config_get_report_desc_complete(uint8_t dev_addr, tusb_control_request_t const * request, xfer_result_t result) | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2021-05-12 19:36:52 +07:00
										 |  |  |   TU_ASSERT(XFER_RESULT_SUCCESS == result); | 
					
						
							| 
									
										
										
										
											2021-05-14 01:02:47 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   uint8_t const itf_num     = (uint8_t) request->wIndex; | 
					
						
							|  |  |  |   uint8_t const instance    = get_instance_id_by_itfnum(dev_addr, itf_num); | 
					
						
							|  |  |  |   hidh_interface_t* hid_itf = get_instance(dev_addr, instance); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   uint8_t const* desc_report = usbh_get_enum_buf(); | 
					
						
							|  |  |  |   uint16_t const desc_len    = request->wLength; | 
					
						
							| 
									
										
										
										
											2021-05-12 15:29:17 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-12 19:36:52 +07:00
										 |  |  |   // enumeration is complete
 | 
					
						
							| 
									
										
										
										
											2021-05-22 21:48:42 +07:00
										 |  |  |   tuh_hid_mount_cb(dev_addr, instance, desc_report, desc_len); | 
					
						
							| 
									
										
										
										
											2021-05-12 19:36:52 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-14 17:47:02 +07:00
										 |  |  |   // queue transfer for IN endpoint
 | 
					
						
							|  |  |  |   hidh_get_report(dev_addr, hid_itf); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-12 19:36:52 +07:00
										 |  |  |   // notify usbh that driver enumeration is complete
 | 
					
						
							|  |  |  |   usbh_driver_set_config_complete(dev_addr, itf_num); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return true; | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-17 13:54:39 +07:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // Report Descriptor Parser
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-22 20:54:59 +07:00
										 |  |  | uint8_t tuh_hid_parse_report_descriptor(tuh_hid_report_info_t* report_info_arr, uint8_t arr_count, uint8_t const* desc_report, uint16_t desc_len) | 
					
						
							| 
									
										
										
										
											2021-05-14 01:02:47 +07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2021-05-14 12:23:09 +07:00
										 |  |  |   // Report Item 6.2.2.2 USB HID 1.11
 | 
					
						
							| 
									
										
										
										
											2021-05-14 01:02:47 +07:00
										 |  |  |   union TU_ATTR_PACKED | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     uint8_t byte; | 
					
						
							|  |  |  |     struct TU_ATTR_PACKED | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         uint8_t size : 2; | 
					
						
							|  |  |  |         uint8_t type : 2; | 
					
						
							|  |  |  |         uint8_t tag  : 4; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |   } header; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-22 20:54:59 +07:00
										 |  |  |   tu_memclr(report_info_arr, arr_count*sizeof(tuh_hid_report_info_t)); | 
					
						
							| 
									
										
										
										
											2021-05-17 13:54:39 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-22 20:54:59 +07:00
										 |  |  |   uint8_t report_num = 0; | 
					
						
							|  |  |  |   tuh_hid_report_info_t* info = report_info_arr; | 
					
						
							| 
									
										
										
										
											2021-05-17 13:54:39 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // current parsed report count & size from descriptor
 | 
					
						
							|  |  |  | //  uint8_t ri_report_count = 0;
 | 
					
						
							|  |  |  | //  uint8_t ri_report_size = 0;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   uint8_t ri_collection_depth = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   while(desc_len && report_num < arr_count) | 
					
						
							| 
									
										
										
										
											2021-05-14 01:02:47 +07:00
										 |  |  |   { | 
					
						
							|  |  |  |     header.byte = *desc_report++; | 
					
						
							| 
									
										
										
										
											2021-05-14 17:47:02 +07:00
										 |  |  |     desc_len--; | 
					
						
							| 
									
										
										
										
											2021-05-14 01:02:47 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-14 17:47:02 +07:00
										 |  |  |     uint8_t const tag  = header.tag; | 
					
						
							| 
									
										
										
										
											2021-05-14 01:02:47 +07:00
										 |  |  |     uint8_t const type = header.type; | 
					
						
							|  |  |  |     uint8_t const size = header.size; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-17 13:54:39 +07:00
										 |  |  |     uint8_t const data8 = desc_report[0]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-11 18:53:56 +07:00
										 |  |  |     TU_LOG(3, "tag = %d, type = %d, size = %d, data = ", tag, type, size); | 
					
						
							|  |  |  |     for(uint32_t i=0; i<size; i++) TU_LOG(3, "%02X ", desc_report[i]); | 
					
						
							|  |  |  |     TU_LOG(3, "\r\n"); | 
					
						
							| 
									
										
										
										
											2021-05-14 01:02:47 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     switch(type) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       case RI_TYPE_MAIN: | 
					
						
							| 
									
										
										
										
											2021-05-14 12:23:09 +07:00
										 |  |  |         switch (tag) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           case RI_MAIN_INPUT: break; | 
					
						
							|  |  |  |           case RI_MAIN_OUTPUT: break; | 
					
						
							|  |  |  |           case RI_MAIN_FEATURE: break; | 
					
						
							| 
									
										
										
										
											2021-05-17 13:54:39 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |           case RI_MAIN_COLLECTION: | 
					
						
							|  |  |  |             ri_collection_depth++; | 
					
						
							|  |  |  |           break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           case RI_MAIN_COLLECTION_END: | 
					
						
							|  |  |  |             ri_collection_depth--; | 
					
						
							| 
									
										
										
										
											2021-05-22 20:54:59 +07:00
										 |  |  |             if (ri_collection_depth == 0) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |               info++; | 
					
						
							|  |  |  |               report_num++; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2021-05-17 13:54:39 +07:00
										 |  |  |           break; | 
					
						
							| 
									
										
										
										
											2021-05-14 12:23:09 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |           default: break; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-05-14 01:02:47 +07:00
										 |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       case RI_TYPE_GLOBAL: | 
					
						
							| 
									
										
										
										
											2021-05-14 12:23:09 +07:00
										 |  |  |         switch(tag) | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2021-05-17 13:54:39 +07:00
										 |  |  |           case RI_GLOBAL_USAGE_PAGE: | 
					
						
							| 
									
										
										
										
											2021-05-22 20:54:59 +07:00
										 |  |  |             // only take in account the "usage page" before REPORT ID
 | 
					
						
							|  |  |  |             if ( ri_collection_depth == 0 ) memcpy(&info->usage_page, desc_report, size); | 
					
						
							| 
									
										
										
										
											2021-05-17 13:54:39 +07:00
										 |  |  |           break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-14 12:23:09 +07:00
										 |  |  |           case RI_GLOBAL_LOGICAL_MIN   : break; | 
					
						
							|  |  |  |           case RI_GLOBAL_LOGICAL_MAX   : break; | 
					
						
							|  |  |  |           case RI_GLOBAL_PHYSICAL_MIN  : break; | 
					
						
							|  |  |  |           case RI_GLOBAL_PHYSICAL_MAX  : break; | 
					
						
							| 
									
										
										
										
											2021-05-17 13:54:39 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |           case RI_GLOBAL_REPORT_ID: | 
					
						
							| 
									
										
										
										
											2021-05-22 20:54:59 +07:00
										 |  |  |             info->report_id = data8; | 
					
						
							| 
									
										
										
										
											2021-05-17 13:54:39 +07:00
										 |  |  |           break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           case RI_GLOBAL_REPORT_SIZE: | 
					
						
							|  |  |  | //            ri_report_size = data8;
 | 
					
						
							|  |  |  |           break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           case RI_GLOBAL_REPORT_COUNT: | 
					
						
							|  |  |  | //            ri_report_count = data8;
 | 
					
						
							|  |  |  |           break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-14 12:23:09 +07:00
										 |  |  |           case RI_GLOBAL_UNIT_EXPONENT : break; | 
					
						
							|  |  |  |           case RI_GLOBAL_UNIT          : break; | 
					
						
							|  |  |  |           case RI_GLOBAL_PUSH          : break; | 
					
						
							|  |  |  |           case RI_GLOBAL_POP           : break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           default: break; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-05-14 01:02:47 +07:00
										 |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       case RI_TYPE_LOCAL: | 
					
						
							| 
									
										
										
										
											2021-05-14 12:23:09 +07:00
										 |  |  |         switch(tag) | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2021-05-17 13:54:39 +07:00
										 |  |  |           case RI_LOCAL_USAGE: | 
					
						
							| 
									
										
										
										
											2021-05-22 20:54:59 +07:00
										 |  |  |             // only take in account the "usage" before starting REPORT ID
 | 
					
						
							|  |  |  |             if ( ri_collection_depth == 0 ) info->usage = data8; | 
					
						
							| 
									
										
										
										
											2021-05-17 13:54:39 +07:00
										 |  |  |           break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-14 12:23:09 +07:00
										 |  |  |           case RI_LOCAL_USAGE_MIN        : break; | 
					
						
							|  |  |  |           case RI_LOCAL_USAGE_MAX        : break; | 
					
						
							|  |  |  |           case RI_LOCAL_DESIGNATOR_INDEX : break; | 
					
						
							|  |  |  |           case RI_LOCAL_DESIGNATOR_MIN   : break; | 
					
						
							|  |  |  |           case RI_LOCAL_DESIGNATOR_MAX   : break; | 
					
						
							|  |  |  |           case RI_LOCAL_STRING_INDEX     : break; | 
					
						
							|  |  |  |           case RI_LOCAL_STRING_MIN       : break; | 
					
						
							|  |  |  |           case RI_LOCAL_STRING_MAX       : break; | 
					
						
							|  |  |  |           case RI_LOCAL_DELIMITER        : break; | 
					
						
							|  |  |  |           default: break; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-05-14 01:02:47 +07:00
										 |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // error
 | 
					
						
							|  |  |  |       default: break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     desc_report += size; | 
					
						
							| 
									
										
										
										
											2021-05-17 13:54:39 +07:00
										 |  |  |     desc_len    -= size; | 
					
						
							| 
									
										
										
										
											2021-05-14 01:02:47 +07:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-05-17 13:54:39 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-22 20:54:59 +07:00
										 |  |  |   for ( uint8_t i = 0; i < report_num; i++ ) | 
					
						
							| 
									
										
										
										
											2021-05-17 13:54:39 +07:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2021-05-22 20:54:59 +07:00
										 |  |  |     info = report_info_arr+i; | 
					
						
							|  |  |  |     TU_LOG2("%u: id = %u, usage_page = %u, usage = %u\r\n", i, info->report_id, info->usage_page, info->usage); | 
					
						
							| 
									
										
										
										
											2021-05-17 13:54:39 +07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return report_num; | 
					
						
							| 
									
										
										
										
											2021-05-14 01:02:47 +07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-12 20:19:00 +07:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							| 
									
										
										
										
											2021-05-14 01:02:47 +07:00
										 |  |  | // Helper
 | 
					
						
							| 
									
										
										
										
											2021-05-12 20:19:00 +07:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Get Device by address
 | 
					
						
							|  |  |  | TU_ATTR_ALWAYS_INLINE static inline hidh_device_t* get_dev(uint8_t dev_addr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return &_hidh_dev[dev_addr-1]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Get Interface by instance number
 | 
					
						
							|  |  |  | TU_ATTR_ALWAYS_INLINE static inline hidh_interface_t* get_instance(uint8_t dev_addr, uint8_t instance) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return &_hidh_dev[dev_addr-1].instances[instance]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Get instance ID by interface number
 | 
					
						
							| 
									
										
										
										
											2021-05-13 14:09:33 +07:00
										 |  |  | static uint8_t get_instance_id_by_itfnum(uint8_t dev_addr, uint8_t itf) | 
					
						
							| 
									
										
										
										
											2021-05-12 20:19:00 +07:00
										 |  |  | { | 
					
						
							|  |  |  |   for ( uint8_t inst = 0; inst < CFG_TUH_HID; inst++ ) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     hidh_interface_t *hid = get_instance(dev_addr, inst); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-13 14:09:33 +07:00
										 |  |  |     if ( (hid->itf_num == itf) && (hid->ep_in || hid->ep_out) ) return inst; | 
					
						
							| 
									
										
										
										
											2021-05-12 20:19:00 +07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return 0xff; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-13 14:09:33 +07:00
										 |  |  | // Get instance ID by endpoint address
 | 
					
						
							|  |  |  | static uint8_t get_instance_id_by_epaddr(uint8_t dev_addr, uint8_t ep_addr) | 
					
						
							| 
									
										
										
										
											2021-05-12 20:19:00 +07:00
										 |  |  | { | 
					
						
							|  |  |  |   for ( uint8_t inst = 0; inst < CFG_TUH_HID; inst++ ) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     hidh_interface_t *hid = get_instance(dev_addr, inst); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-13 14:09:33 +07:00
										 |  |  |     if ( (ep_addr == hid->ep_in) || ( ep_addr == hid->ep_out) ) return inst; | 
					
						
							| 
									
										
										
										
											2021-05-12 20:19:00 +07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-13 14:09:33 +07:00
										 |  |  |   return 0xff; | 
					
						
							| 
									
										
										
										
											2021-05-12 20:19:00 +07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | #endif
 |