espressif_tinyusb/src/class/hid/hid_host.c

565 lines
18 KiB
C
Raw Normal View History

2020-01-15 05:30:39 +01: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"
#if (TUSB_OPT_HOST_ENABLED && CFG_TUH_HID)
2020-01-15 05:30:39 +01:00
#include "common/tusb_common.h"
#include "hid_host.h"
//--------------------------------------------------------------------+
// MACRO CONSTANT TYPEDEF
//--------------------------------------------------------------------+
2021-05-13 20:02:47 +02:00
/*
"KEYBOARD" : in_len=8 , out_len=1, usage_page=0x01, usage=0x06 # Generic Desktop, Keyboard
"MOUSE" : in_len=4 , out_len=0, usage_page=0x01, usage=0x02 # Generic Desktop, Mouse
"CONSUMER" : in_len=2 , out_len=0, usage_page=0x0C, usage=0x01 # Consumer, Consumer Control
"SYS_CONTROL" : in_len=1 , out_len=0, usage_page=0x01, usage=0x80 # Generic Desktop, Sys Control
"GAMEPAD" : in_len=6 , out_len=0, usage_page=0x01, usage=0x05 # Generic Desktop, Game Pad
"DIGITIZER" : in_len=5 , out_len=0, usage_page=0x0D, usage=0x02 # Digitizers, Pen
"XAC_COMPATIBLE_GAMEPAD" : in_len=3 , out_len=0, usage_page=0x01, usage=0x05 # Generic Desktop, Game Pad
"RAW" : in_len=64, out_len=0, usage_page=0xFFAF, usage=0xAF # Vendor 0xFFAF "Adafruit", 0xAF
2021-02-24 07:54:18 +01:00
*/
2021-05-12 10:29:17 +02:00
typedef struct
{
2021-02-24 07:54:18 +01:00
uint8_t itf_num;
uint8_t ep_in;
uint8_t ep_out;
uint8_t itf_protocol; // None, Keyboard, Mouse
uint8_t protocol_mode; // Boot (0) or Report protocol (1)
2021-05-12 14:36:52 +02:00
uint8_t report_desc_type;
2021-05-12 10:29:17 +02:00
uint16_t report_desc_len;
uint16_t epin_size;
uint16_t epout_size;
2021-02-24 07:54:18 +01:00
uint8_t epin_buf[CFG_TUH_HID_EP_BUFSIZE];
uint8_t epout_buf[CFG_TUH_HID_EP_BUFSIZE];
2021-05-13 20:02:47 +02:00
} hidh_interface_t;
2021-05-12 10:29:17 +02:00
typedef struct
{
uint8_t inst_count;
hidh_interface_t instances[CFG_TUH_HID];
2021-05-12 10:29:17 +02:00
} hidh_device_t;
static hidh_device_t _hidh_dev[CFG_TUSB_HOST_DEVICE_MAX-1];
2021-05-13 09:09:33 +02:00
//------------- Internal prototypes -------------//
// Get HID device & interface
2021-05-12 15:19:00 +02: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 09:09:33 +02: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 10:29:17 +02: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-15 05:30:39 +01:00
//--------------------------------------------------------------------+
2021-05-12 14:36:52 +02:00
// Application API
2020-01-15 05:30:39 +01:00
//--------------------------------------------------------------------+
2021-05-13 09:09:33 +02:00
2021-05-22 13:17:32 +02:00
uint8_t tuh_hid_instance_count(uint8_t dev_addr)
2020-01-15 05:30:39 +01:00
{
2021-05-13 09:09:33 +02:00
return get_dev(dev_addr)->inst_count;
2020-01-15 05:30:39 +01:00
}
2021-05-22 13:17:32 +02:00
bool tuh_hid_mounted(uint8_t dev_addr, uint8_t instance)
2020-01-15 05:30:39 +01:00
{
2021-05-13 09:09:33 +02:00
hidh_interface_t* hid_itf = get_instance(dev_addr, instance);
2021-05-13 08:42:52 +02:00
return (hid_itf->ep_in != 0) || (hid_itf->ep_out != 0);
2020-01-15 05:30:39 +01:00
}
2021-05-22 13:17:32 +02:00
uint8_t tuh_hid_interface_protocol(uint8_t dev_addr, uint8_t instance)
2021-05-13 20:02:47 +02:00
{
hidh_interface_t* hid_itf = get_instance(dev_addr, instance);
return hid_itf->itf_protocol;
2021-05-13 20:02:47 +02:00
}
2021-05-22 13:17:32 +02:00
bool tuh_hid_get_protocol(uint8_t dev_addr, uint8_t instance)
2021-05-13 20:02:47 +02:00
{
hidh_interface_t* hid_itf = get_instance(dev_addr, instance);
return hid_itf->protocol_mode;
2021-05-13 20:02:47 +02: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 13:17:32 +02:00
bool tuh_hid_set_protocol(uint8_t dev_addr, uint8_t instance, uint8_t protocol)
{
hidh_interface_t* hid_itf = get_instance(dev_addr, instance);
TU_VERIFY(hid_itf->itf_protocol != HID_ITF_PROTOCOL_NONE);
TU_LOG2("HID Set Protocol = %d\r\n", protocol);
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-15 05:30:39 +01: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
};
TU_ASSERT( tuh_control_xfer(dev_addr, &request, NULL, set_report_complete) );
return true;
}
//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);
// return !hcd_edpt_busy(dev_addr, hid_itf->ep_in);
//}
2020-01-15 05:30:39 +01:00
//--------------------------------------------------------------------+
2021-05-12 14:36:52 +02:00
// USBH API
2020-01-15 05:30:39 +01:00
//--------------------------------------------------------------------+
void hidh_init(void)
{
2021-05-12 10:29:17 +02:00
tu_memclr(_hidh_dev, sizeof(_hidh_dev));
2021-05-12 14:36:52 +02:00
}
2021-05-18 08:08:30 +02:00
bool hidh_xfer_cb(uint8_t dev_addr, uint8_t ep_addr, xfer_result_t result, uint32_t xferred_bytes)
2021-05-12 14:36:52 +02:00
{
2021-05-18 08:08:30 +02:00
(void) result;
2021-05-13 09:09:33 +02:00
uint8_t const dir = tu_edpt_dir(ep_addr);
uint8_t const instance = get_instance_id_by_epaddr(dev_addr, ep_addr);
hidh_interface_t* hid_itf = get_instance(dev_addr, instance);
2021-05-12 14:36:52 +02:00
2021-05-13 09:09:33 +02:00
if ( dir == TUSB_DIR_IN )
{
TU_LOG2(" Get Report callback (%u, %u)\r\n", dev_addr, instance);
TU_LOG1_MEM(hid_itf->epin_buf, 8, 2);
tuh_hid_report_received_cb(dev_addr, instance, hid_itf->epin_buf, xferred_bytes);
// queue next report
hidh_get_report(dev_addr, hid_itf);
2021-05-13 09:09:33 +02:00
}else
{
// if (tuh_hid_set_report_complete_cb) tuh_hid_set_report_complete_cb(dev_addr, instance, xferred_bytes);
2021-05-13 09:09:33 +02:00
}
2021-05-12 14:36:52 +02:00
return true;
2020-01-15 05:30:39 +01:00
}
2021-05-12 14:36:52 +02:00
void hidh_close(uint8_t dev_addr)
{
hidh_device_t* hid_dev = get_dev(dev_addr);
if (tuh_hid_umount_cb)
2021-05-12 14:36:52 +02:00
{
for ( uint8_t inst = 0; inst < hid_dev->inst_count; inst++) tuh_hid_umount_cb(dev_addr, inst);
2021-05-12 14:36:52 +02:00
}
tu_memclr(hid_dev, sizeof(hidh_device_t));
2021-05-12 14:36:52 +02:00
}
//--------------------------------------------------------------------+
// Enumeration
//--------------------------------------------------------------------+
static bool config_set_idle_complete(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-02-24 07:19:32 +01:00
bool hidh_open(uint8_t rhport, uint8_t dev_addr, tusb_desc_interface_t const *desc_itf, uint16_t *p_length)
2020-01-15 05:30:39 +01:00
{
2021-02-24 07:19:32 +01:00
TU_VERIFY(TUSB_CLASS_HID == desc_itf->bInterfaceClass);
uint8_t const *p_desc = (uint8_t const *) desc_itf;
2020-01-15 05:30:39 +01:00
//------------- HID descriptor -------------//
2021-02-24 07:19:32 +01: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-02-24 07:54:18 +01:00
TU_ASSERT(HID_DESC_TYPE_HID == desc_hid->bDescriptorType);
2020-01-15 05:30:39 +01:00
2021-05-12 10:29:17 +02:00
// not enough interface, try to increase CFG_TUH_HID
// TODO multiple devices
hidh_device_t* hid_dev = get_dev(dev_addr);
TU_ASSERT(hid_dev->inst_count < CFG_TUH_HID);
2021-05-12 10:29:17 +02:00
2020-01-15 05:30:39 +01:00
//------------- Endpoint Descriptor -------------//
2021-02-24 07:19:32 +01:00
p_desc = tu_desc_next(p_desc);
tusb_desc_endpoint_t const * desc_ep = (tusb_desc_endpoint_t const *) p_desc;
2021-02-24 07:54:18 +01:00
TU_ASSERT(TUSB_DESC_ENDPOINT == desc_ep->bDescriptorType);
2020-01-15 05:30:39 +01:00
2021-05-12 14:36:52 +02:00
// TODO also open endpoint OUT
TU_ASSERT( usbh_edpt_open(rhport, dev_addr, desc_ep) );
hidh_interface_t* hid_itf = get_instance(dev_addr, hid_dev->inst_count);
hid_dev->inst_count++;
2021-05-12 14:36:52 +02:00
hid_itf->itf_num = desc_itf->bInterfaceNumber;
hid_itf->ep_in = desc_ep->bEndpointAddress;
hid_itf->epin_size = desc_ep->wMaxPacketSize.size;
2021-05-12 14:36:52 +02:00
// Assume bNumDescriptors = 1
hid_itf->report_desc_type = desc_hid->bReportType;
hid_itf->report_desc_len = tu_unaligned_read16(&desc_hid->wReportLength);
2021-05-12 14:36:52 +02:00
hid_itf->protocol_mode = HID_PROTOCOL_REPORT; // Per Specs: default is report mode
if ( HID_SUBCLASS_BOOT == desc_itf->bInterfaceSubClass ) hid_itf->itf_protocol = desc_itf->bInterfaceProtocol;
2020-01-15 05:30:39 +01:00
2021-05-12 10:29:17 +02:00
*p_length = sizeof(tusb_desc_interface_t) + sizeof(tusb_hid_descriptor_hid_t) + desc_itf->bNumEndpoints*sizeof(tusb_desc_endpoint_t);
2020-01-15 05:30:39 +01:00
return true;
}
bool hidh_set_config(uint8_t dev_addr, uint8_t itf_num)
{
2021-05-12 14:36:52 +02:00
// Idle rate = 0 mean only report when there is changes
uint16_t const idle_rate = 0;
2021-05-12 14:36:52 +02:00
// SET IDLE request, device can stall if not support this request
TU_LOG2("Set Idle \r\n");
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 14:36:52 +02:00
.wValue = idle_rate,
.wIndex = itf_num,
.wLength = 0
};
2021-05-12 14:36:52 +02:00
TU_ASSERT( tuh_control_xfer(dev_addr, &request, NULL, config_set_idle_complete) );
return true;
}
2021-05-12 14:36:52 +02:00
bool config_set_idle_complete(uint8_t dev_addr, tusb_control_request_t const * request, xfer_result_t result)
2020-01-15 05:30:39 +01:00
{
2021-05-12 14:36:52 +02:00
// Stall is a valid response for SET_IDLE, therefore we could ignore its result
(void) result;
2020-01-15 05:30:39 +01: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-15 05:30:39 +01:00
2021-05-12 14:36:52 +02:00
// Get Report Descriptor
// using usbh enumeration buffer since report descriptor can be very long
TU_ASSERT( hid_itf->report_desc_len <= CFG_TUH_ENUMERATION_BUFSZIE );
TU_LOG2("HID Get Report Descriptor\r\n");
tusb_control_request_t const new_request =
2020-01-15 05:30:39 +01:00
{
2021-05-12 14:36:52 +02: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
};
2021-05-12 14:36:52 +02:00
TU_ASSERT(tuh_control_xfer(dev_addr, &new_request, usbh_get_enum_buf(), config_get_report_desc_complete));
return true;
2020-01-15 05:30:39 +01:00
}
2021-05-12 14:36:52 +02:00
bool config_get_report_desc_complete(uint8_t dev_addr, tusb_control_request_t const * request, xfer_result_t result)
2020-01-15 05:30:39 +01:00
{
2021-05-12 14:36:52 +02:00
TU_ASSERT(XFER_RESULT_SUCCESS == result);
2021-05-13 20:02:47 +02: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 10:29:17 +02:00
2021-05-12 14:36:52 +02:00
// enumeration is complete
tuh_hid_mount_cb(dev_addr, instance, desc_report, desc_len);
2021-05-12 14:36:52 +02:00
// queue transfer for IN endpoint
hidh_get_report(dev_addr, hid_itf);
2021-05-12 14:36:52 +02:00
// notify usbh that driver enumeration is complete
usbh_driver_set_config_complete(dev_addr, itf_num);
return true;
2020-01-15 05:30:39 +01:00
}
//--------------------------------------------------------------------+
// Report Descriptor Parser
//--------------------------------------------------------------------+
2021-05-22 15:54:59 +02: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-13 20:02:47 +02:00
{
2021-05-14 07:23:09 +02:00
// Report Item 6.2.2.2 USB HID 1.11
2021-05-13 20:02:47 +02: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 15:54:59 +02:00
tu_memclr(report_info_arr, arr_count*sizeof(tuh_hid_report_info_t));
2021-05-22 15:54:59 +02:00
uint8_t report_num = 0;
tuh_hid_report_info_t* info = report_info_arr;
// 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-13 20:02:47 +02:00
{
header.byte = *desc_report++;
desc_len--;
2021-05-13 20:02:47 +02:00
uint8_t const tag = header.tag;
2021-05-13 20:02:47 +02:00
uint8_t const type = header.type;
uint8_t const size = header.size;
uint8_t const data8 = desc_report[0];
2021-05-13 20:02:47 +02:00
TU_LOG2("tag = %d, type = %d, size = %d, data = ", tag, type, size);
for(uint32_t i=0; i<size; i++) TU_LOG2("%02X ", desc_report[i]);
TU_LOG2("\r\n");
switch(type)
{
case RI_TYPE_MAIN:
2021-05-14 07:23:09 +02:00
switch (tag)
{
case RI_MAIN_INPUT: break;
case RI_MAIN_OUTPUT: break;
case RI_MAIN_FEATURE: break;
case RI_MAIN_COLLECTION:
ri_collection_depth++;
break;
case RI_MAIN_COLLECTION_END:
ri_collection_depth--;
2021-05-22 15:54:59 +02:00
if (ri_collection_depth == 0)
{
info++;
report_num++;
}
break;
2021-05-14 07:23:09 +02:00
default: break;
}
2021-05-13 20:02:47 +02:00
break;
case RI_TYPE_GLOBAL:
2021-05-14 07:23:09 +02:00
switch(tag)
{
case RI_GLOBAL_USAGE_PAGE:
2021-05-22 15:54:59 +02:00
// only take in account the "usage page" before REPORT ID
if ( ri_collection_depth == 0 ) memcpy(&info->usage_page, desc_report, size);
break;
2021-05-14 07:23:09 +02: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;
case RI_GLOBAL_REPORT_ID:
2021-05-22 15:54:59 +02:00
info->report_id = data8;
break;
case RI_GLOBAL_REPORT_SIZE:
// ri_report_size = data8;
break;
case RI_GLOBAL_REPORT_COUNT:
// ri_report_count = data8;
break;
2021-05-14 07:23:09 +02: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-13 20:02:47 +02:00
break;
case RI_TYPE_LOCAL:
2021-05-14 07:23:09 +02:00
switch(tag)
{
case RI_LOCAL_USAGE:
2021-05-22 15:54:59 +02:00
// only take in account the "usage" before starting REPORT ID
if ( ri_collection_depth == 0 ) info->usage = data8;
break;
2021-05-14 07:23:09 +02: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-13 20:02:47 +02:00
break;
// error
default: break;
}
desc_report += size;
desc_len -= size;
2021-05-13 20:02:47 +02:00
}
2021-05-22 15:54:59 +02:00
for ( uint8_t i = 0; i < report_num; i++ )
{
2021-05-22 15:54:59 +02: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);
}
return report_num;
2021-05-13 20:02:47 +02:00
}
2021-05-12 15:19:00 +02:00
//--------------------------------------------------------------------+
2021-05-13 20:02:47 +02:00
// Helper
2021-05-12 15:19:00 +02: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 09:09:33 +02:00
static uint8_t get_instance_id_by_itfnum(uint8_t dev_addr, uint8_t itf)
2021-05-12 15:19:00 +02:00
{
for ( uint8_t inst = 0; inst < CFG_TUH_HID; inst++ )
{
hidh_interface_t *hid = get_instance(dev_addr, inst);
2021-05-13 09:09:33 +02:00
if ( (hid->itf_num == itf) && (hid->ep_in || hid->ep_out) ) return inst;
2021-05-12 15:19:00 +02:00
}
return 0xff;
}
2021-05-13 09:09:33 +02: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 15:19:00 +02:00
{
for ( uint8_t inst = 0; inst < CFG_TUH_HID; inst++ )
{
hidh_interface_t *hid = get_instance(dev_addr, inst);
2021-05-13 09:09:33 +02:00
if ( (ep_addr == hid->ep_in) || ( ep_addr == hid->ep_out) ) return inst;
2021-05-12 15:19:00 +02:00
}
2021-05-13 09:09:33 +02:00
return 0xff;
2021-05-12 15:19:00 +02:00
}
2020-01-15 05:30:39 +01:00
#endif