2014-03-12 14:01:38 +07:00
|
|
|
/**************************************************************************/
|
|
|
|
/*!
|
|
|
|
@file usbd.c
|
|
|
|
@author hathach (tinyusb.org)
|
|
|
|
|
|
|
|
@section LICENSE
|
|
|
|
|
|
|
|
Software License Agreement (BSD License)
|
|
|
|
|
|
|
|
Copyright (c) 2013, hathach (tinyusb.org)
|
|
|
|
All rights reserved.
|
|
|
|
|
|
|
|
Redistribution and use in source and binary forms, with or without
|
|
|
|
modification, are permitted provided that the following conditions are met:
|
|
|
|
1. Redistributions of source code must retain the above copyright
|
|
|
|
notice, this list of conditions and the following disclaimer.
|
|
|
|
2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
notice, this list of conditions and the following disclaimer in the
|
|
|
|
documentation and/or other materials provided with the distribution.
|
|
|
|
3. Neither the name of the copyright holders nor the
|
|
|
|
names of its contributors may be used to endorse or promote products
|
|
|
|
derived from this software without specific prior written permission.
|
|
|
|
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS ''AS IS'' AND ANY
|
|
|
|
EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
|
|
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
|
|
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE FOR ANY
|
|
|
|
DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
|
|
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
|
|
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
|
|
ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
|
|
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
|
|
|
|
This file is part of the tinyusb stack.
|
|
|
|
*/
|
|
|
|
/**************************************************************************/
|
|
|
|
|
2018-11-07 23:04:34 -08:00
|
|
|
// This top level class manages the bus state and delegates events to class-specific drivers.
|
|
|
|
|
2014-03-12 14:01:38 +07:00
|
|
|
#include "tusb_option.h"
|
|
|
|
|
2018-07-23 15:25:45 +07:00
|
|
|
#if TUSB_OPT_DEVICE_ENABLED
|
2014-03-12 14:01:38 +07:00
|
|
|
|
|
|
|
#define _TINY_USB_SOURCE_FILE_
|
|
|
|
|
|
|
|
#include "tusb.h"
|
2018-03-11 12:31:24 +07:00
|
|
|
#include "usbd.h"
|
2018-03-22 14:15:16 +07:00
|
|
|
#include "device/usbd_pvt.h"
|
2014-03-12 14:01:38 +07:00
|
|
|
|
2018-06-23 13:19:36 +07:00
|
|
|
#ifndef CFG_TUD_TASK_QUEUE_SZ
|
|
|
|
#define CFG_TUD_TASK_QUEUE_SZ 16
|
|
|
|
#endif
|
2018-05-17 19:19:55 +07:00
|
|
|
|
2018-06-23 13:19:36 +07:00
|
|
|
#ifndef CFG_TUD_TASK_STACK_SZ
|
|
|
|
#define CFG_TUD_TASK_STACK_SZ 150
|
2018-05-17 19:19:55 +07:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef CFG_TUD_TASK_PRIO
|
|
|
|
#define CFG_TUD_TASK_PRIO 0
|
|
|
|
#endif
|
|
|
|
|
2018-07-13 15:08:38 +07:00
|
|
|
|
2018-04-16 13:46:28 +07:00
|
|
|
//--------------------------------------------------------------------+
|
2018-07-13 15:08:38 +07:00
|
|
|
// Device Data
|
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
typedef struct {
|
|
|
|
uint8_t config_num;
|
2018-07-13 16:52:22 +07:00
|
|
|
|
2018-11-16 21:56:39 +07:00
|
|
|
uint8_t itf2drv[16]; // map interface number to driver (0xff is invalid)
|
2018-12-10 05:15:49 +07:00
|
|
|
uint8_t ep2drv[8][2]; // map endpoint to driver ( 0xff is invalid )
|
2018-11-16 21:56:39 +07:00
|
|
|
|
2018-07-13 15:08:38 +07:00
|
|
|
}usbd_device_t;
|
|
|
|
|
|
|
|
static usbd_device_t _usbd_dev;
|
|
|
|
|
2018-08-01 00:50:04 +07:00
|
|
|
// Auto descriptor is enabled, descriptor set point to auto generated one
|
|
|
|
#if CFG_TUD_DESC_AUTO
|
|
|
|
extern tud_desc_set_t const _usbd_auto_desc_set;
|
|
|
|
tud_desc_set_t const* usbd_desc_set = &_usbd_auto_desc_set;
|
|
|
|
#else
|
|
|
|
tud_desc_set_t const* usbd_desc_set = &tud_desc_set;
|
|
|
|
#endif
|
|
|
|
|
2018-07-13 15:08:38 +07:00
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
// Class Driver
|
2018-04-16 13:46:28 +07:00
|
|
|
//--------------------------------------------------------------------+
|
2018-03-23 14:57:17 +07:00
|
|
|
typedef struct {
|
2018-05-14 13:02:01 +07:00
|
|
|
uint8_t class_code;
|
|
|
|
|
2018-06-16 12:30:10 +07:00
|
|
|
void (* init ) (void);
|
|
|
|
tusb_error_t (* open ) (uint8_t rhport, tusb_desc_interface_t const * desc_intf, uint16_t* p_length);
|
2018-11-16 21:56:39 +07:00
|
|
|
bool (* control_request ) (uint8_t rhport, tusb_control_request_t const * request);
|
|
|
|
bool (* control_request_complete ) (uint8_t rhport, tusb_control_request_t const * request);
|
2018-11-23 15:14:47 +07:00
|
|
|
tusb_error_t (* xfer_cb ) (uint8_t rhport, uint8_t ep_addr, xfer_result_t, uint32_t);
|
2018-06-16 12:30:10 +07:00
|
|
|
void (* sof ) (uint8_t rhport);
|
2018-07-13 16:09:26 +07:00
|
|
|
void (* reset ) (uint8_t);
|
2018-03-23 14:57:17 +07:00
|
|
|
} usbd_class_driver_t;
|
|
|
|
|
2014-03-14 17:18:05 +07:00
|
|
|
static usbd_class_driver_t const usbd_class_drivers[] =
|
2014-03-12 14:01:38 +07:00
|
|
|
{
|
2018-05-14 13:02:01 +07:00
|
|
|
#if CFG_TUD_CDC
|
2014-03-12 14:01:38 +07:00
|
|
|
{
|
2018-11-07 23:04:34 -08:00
|
|
|
.class_code = TUSB_CLASS_CDC,
|
|
|
|
.init = cdcd_init,
|
|
|
|
.open = cdcd_open,
|
|
|
|
.control_request = cdcd_control_request,
|
2018-11-08 13:45:30 -08:00
|
|
|
.control_request_complete = cdcd_control_request_complete,
|
2018-11-07 23:04:34 -08:00
|
|
|
.xfer_cb = cdcd_xfer_cb,
|
|
|
|
.sof = NULL,
|
|
|
|
.reset = cdcd_reset
|
2014-03-12 14:01:38 +07:00
|
|
|
},
|
2014-03-14 17:18:05 +07:00
|
|
|
#endif
|
2014-03-12 14:01:38 +07:00
|
|
|
|
2018-05-14 13:02:01 +07:00
|
|
|
#if CFG_TUD_MSC
|
2014-03-12 14:01:38 +07:00
|
|
|
{
|
2018-11-07 23:04:34 -08:00
|
|
|
.class_code = TUSB_CLASS_MSC,
|
|
|
|
.init = mscd_init,
|
|
|
|
.open = mscd_open,
|
|
|
|
.control_request = mscd_control_request,
|
2018-11-08 13:45:30 -08:00
|
|
|
.control_request_complete = mscd_control_request_complete,
|
2018-11-07 23:04:34 -08:00
|
|
|
.xfer_cb = mscd_xfer_cb,
|
|
|
|
.sof = NULL,
|
|
|
|
.reset = mscd_reset
|
2018-06-16 12:30:10 +07:00
|
|
|
},
|
|
|
|
#endif
|
|
|
|
|
2018-07-13 17:48:26 +07:00
|
|
|
|
2018-07-28 12:38:45 +07:00
|
|
|
#if CFG_TUD_HID
|
2018-07-13 17:48:26 +07:00
|
|
|
{
|
2018-11-07 23:04:34 -08:00
|
|
|
.class_code = TUSB_CLASS_HID,
|
|
|
|
.init = hidd_init,
|
|
|
|
.open = hidd_open,
|
|
|
|
.control_request = hidd_control_request,
|
2018-11-08 13:45:30 -08:00
|
|
|
.control_request_complete = hidd_control_request_complete,
|
2018-11-07 23:04:34 -08:00
|
|
|
.xfer_cb = hidd_xfer_cb,
|
|
|
|
.sof = NULL,
|
|
|
|
.reset = hidd_reset
|
2018-07-13 17:48:26 +07:00
|
|
|
},
|
|
|
|
#endif
|
|
|
|
|
2018-06-16 12:30:10 +07:00
|
|
|
#if CFG_TUD_CUSTOM_CLASS
|
|
|
|
{
|
2018-11-07 23:04:34 -08:00
|
|
|
.class_code = TUSB_CLASS_VENDOR_SPECIFIC,
|
|
|
|
.init = cusd_init,
|
|
|
|
.open = cusd_open,
|
|
|
|
.control_request = cusd_control_request,
|
2018-11-08 13:45:30 -08:00
|
|
|
.control_request_complete = cusd_control_request_complete,
|
2018-11-07 23:04:34 -08:00
|
|
|
.xfer_cb = cusd_xfer_cb,
|
|
|
|
.sof = NULL,
|
|
|
|
.reset = cusd_reset
|
2014-03-12 14:01:38 +07:00
|
|
|
},
|
2014-03-14 17:18:05 +07:00
|
|
|
#endif
|
2014-03-12 14:01:38 +07:00
|
|
|
};
|
|
|
|
|
2014-03-18 16:58:24 +07:00
|
|
|
enum { USBD_CLASS_DRIVER_COUNT = sizeof(usbd_class_drivers) / sizeof(usbd_class_driver_t) };
|
2014-03-14 17:18:05 +07:00
|
|
|
|
2018-04-16 13:46:28 +07:00
|
|
|
|
2014-03-12 14:01:38 +07:00
|
|
|
//--------------------------------------------------------------------+
|
2018-05-17 19:19:55 +07:00
|
|
|
// DCD Event
|
2014-03-12 14:01:38 +07:00
|
|
|
//--------------------------------------------------------------------+
|
2018-06-23 13:19:36 +07:00
|
|
|
OSAL_TASK_DEF(_usbd_task_def, "usbd", usbd_task, CFG_TUD_TASK_PRIO, CFG_TUD_TASK_STACK_SZ);
|
2018-05-17 20:04:52 +07:00
|
|
|
|
2018-12-05 17:01:19 +07:00
|
|
|
// Event queue
|
|
|
|
// role device/host is used by OS NONE for mutex (disable usb isr) only
|
|
|
|
OSAL_QUEUE_DEF(OPT_MODE_DEVICE, _usbd_qdef, CFG_TUD_TASK_QUEUE_SZ, dcd_event_t);
|
2018-05-17 19:19:55 +07:00
|
|
|
static osal_queue_t _usbd_q;
|
2018-02-28 14:21:31 +07:00
|
|
|
|
2018-05-17 19:19:55 +07:00
|
|
|
//--------------------------------------------------------------------+
|
2018-11-16 21:56:39 +07:00
|
|
|
// Prototypes
|
2018-05-17 19:19:55 +07:00
|
|
|
//--------------------------------------------------------------------+
|
2018-12-10 05:15:49 +07:00
|
|
|
static void mark_interface_endpoint(uint8_t ep2drv[8][2], uint8_t const* p_desc, uint16_t desc_len, uint8_t driver_id);
|
2018-11-16 21:56:39 +07:00
|
|
|
static bool process_control_request(uint8_t rhport, tusb_control_request_t const * p_request);
|
2018-11-26 12:25:28 +07:00
|
|
|
static bool process_set_config(uint8_t rhport);
|
2018-11-16 21:56:39 +07:00
|
|
|
static void const* get_descriptor(tusb_control_request_t const * p_request, uint16_t* desc_len);
|
|
|
|
|
|
|
|
void usbd_control_reset (uint8_t rhport);
|
2018-11-23 15:14:47 +07:00
|
|
|
bool usbd_control_xfer_cb (uint8_t rhport, uint8_t ep_addr, xfer_result_t event, uint32_t xferred_bytes);
|
2018-11-16 21:56:39 +07:00
|
|
|
void usbd_control_set_complete_callback( bool (*fp) (uint8_t, tusb_control_request_t const * ) );
|
2014-03-12 14:01:38 +07:00
|
|
|
|
2018-05-17 19:19:55 +07:00
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
// APPLICATION API
|
|
|
|
//--------------------------------------------------------------------+
|
2018-07-12 22:25:06 +07:00
|
|
|
bool tud_mounted(void)
|
2018-05-17 19:19:55 +07:00
|
|
|
{
|
2018-07-13 15:08:38 +07:00
|
|
|
return _usbd_dev.config_num > 0;
|
2018-05-17 19:19:55 +07:00
|
|
|
}
|
|
|
|
|
2014-03-31 13:12:51 +07:00
|
|
|
//--------------------------------------------------------------------+
|
2018-11-16 21:56:39 +07:00
|
|
|
// USBD Task
|
2014-03-31 13:12:51 +07:00
|
|
|
//--------------------------------------------------------------------+
|
2018-12-05 17:30:04 +07:00
|
|
|
bool usbd_init (void)
|
2014-03-31 13:12:51 +07:00
|
|
|
{
|
2018-12-05 13:20:25 +07:00
|
|
|
// Init device queue & task
|
|
|
|
_usbd_q = osal_queue_create(&_usbd_qdef);
|
2018-12-05 17:30:04 +07:00
|
|
|
TU_ASSERT(_usbd_q != NULL);
|
2018-12-05 13:20:25 +07:00
|
|
|
|
|
|
|
osal_task_create(&_usbd_task_def);
|
|
|
|
|
|
|
|
// Init class drivers
|
|
|
|
for (uint8_t i = 0; i < USBD_CLASS_DRIVER_COUNT; i++) usbd_class_drivers[i].init();
|
|
|
|
|
|
|
|
// Init device controller driver
|
2018-12-05 17:30:04 +07:00
|
|
|
TU_ASSERT(dcd_init(TUD_OPT_RHPORT));
|
2018-12-05 17:01:19 +07:00
|
|
|
dcd_int_enable(TUD_OPT_RHPORT);
|
2018-03-02 22:46:36 +07:00
|
|
|
|
2018-12-05 17:30:04 +07:00
|
|
|
return true;
|
2014-03-31 13:12:51 +07:00
|
|
|
}
|
|
|
|
|
2018-11-02 17:29:49 +07:00
|
|
|
static void usbd_reset(uint8_t rhport)
|
|
|
|
{
|
|
|
|
tu_varclr(&_usbd_dev);
|
|
|
|
memset(_usbd_dev.itf2drv, 0xff, sizeof(_usbd_dev.itf2drv)); // invalid mapping
|
|
|
|
memset(_usbd_dev.ep2drv , 0xff, sizeof(_usbd_dev.ep2drv )); // invalid mapping
|
2018-11-14 23:39:58 +07:00
|
|
|
|
2018-11-16 21:56:39 +07:00
|
|
|
usbd_control_reset(rhport);
|
2018-11-02 17:29:49 +07:00
|
|
|
|
|
|
|
for (uint8_t i = 0; i < USBD_CLASS_DRIVER_COUNT; i++)
|
|
|
|
{
|
|
|
|
if ( usbd_class_drivers[i].reset ) usbd_class_drivers[i].reset( rhport );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-16 21:56:39 +07:00
|
|
|
// Main device task implementation
|
2018-11-14 17:40:29 +07:00
|
|
|
static void usbd_task_body(void)
|
2014-03-31 13:12:51 +07:00
|
|
|
{
|
2018-08-28 15:56:43 +07:00
|
|
|
// Loop until there is no more events in the queue
|
2018-11-14 16:31:28 +07:00
|
|
|
while (1)
|
2018-08-28 15:56:43 +07:00
|
|
|
{
|
2018-11-16 21:56:39 +07:00
|
|
|
dcd_event_t event;
|
|
|
|
|
2018-11-14 17:40:29 +07:00
|
|
|
if ( !osal_queue_receive(_usbd_q, &event) ) return;
|
2018-07-13 18:01:16 +07:00
|
|
|
|
2018-11-14 17:40:29 +07:00
|
|
|
switch ( event.event_id )
|
2018-03-06 16:50:50 +07:00
|
|
|
{
|
2018-11-14 17:40:29 +07:00
|
|
|
case DCD_EVENT_SETUP_RECEIVED:
|
2018-11-16 21:56:39 +07:00
|
|
|
// Process control request, if failed control endpoint is stalled
|
|
|
|
if ( !process_control_request(event.rhport, &event.setup_received) )
|
|
|
|
{
|
|
|
|
usbd_control_stall(event.rhport);
|
|
|
|
}
|
2018-11-14 17:40:29 +07:00
|
|
|
break;
|
2018-08-28 15:56:43 +07:00
|
|
|
|
2018-11-14 17:40:29 +07:00
|
|
|
case DCD_EVENT_XFER_COMPLETE:
|
2018-03-08 14:38:06 +07:00
|
|
|
{
|
2018-11-14 17:40:29 +07:00
|
|
|
// Invoke the class callback associated with the endpoint address
|
|
|
|
uint8_t const ep_addr = event.xfer_complete.ep_addr;
|
2018-11-02 17:29:49 +07:00
|
|
|
|
2018-12-12 11:51:31 +07:00
|
|
|
if ( 0 == tu_edpt_number(ep_addr) )
|
2018-08-28 15:56:43 +07:00
|
|
|
{
|
2018-11-16 21:56:39 +07:00
|
|
|
// control transfer DATA stage callback
|
|
|
|
usbd_control_xfer_cb(event.rhport, ep_addr, event.xfer_complete.result, event.xfer_complete.len);
|
2018-11-14 23:39:58 +07:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-12-12 11:51:31 +07:00
|
|
|
uint8_t const drv_id = _usbd_dev.ep2drv[tu_edpt_number(ep_addr)][tu_edpt_dir(ep_addr)];
|
2018-11-14 23:39:58 +07:00
|
|
|
TU_ASSERT(drv_id < USBD_CLASS_DRIVER_COUNT,);
|
|
|
|
|
2018-11-14 17:40:29 +07:00
|
|
|
usbd_class_drivers[drv_id].xfer_cb(event.rhport, ep_addr, event.xfer_complete.result, event.xfer_complete.len);
|
2018-08-28 15:56:43 +07:00
|
|
|
}
|
|
|
|
}
|
2018-11-14 17:40:29 +07:00
|
|
|
break;
|
|
|
|
|
|
|
|
case DCD_EVENT_BUS_RESET:
|
|
|
|
usbd_reset(event.rhport);
|
2018-12-05 17:01:19 +07:00
|
|
|
// TODO remove since if task is too slow, we could clear the event of the new attached
|
2018-11-14 17:40:29 +07:00
|
|
|
osal_queue_reset(_usbd_q);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DCD_EVENT_UNPLUGGED:
|
|
|
|
usbd_reset(event.rhport);
|
2018-12-05 17:01:19 +07:00
|
|
|
// TODO remove since if task is too slow, we could clear the event of the new attached
|
2018-11-14 17:40:29 +07:00
|
|
|
osal_queue_reset(_usbd_q);
|
|
|
|
|
2018-12-10 19:25:57 +07:00
|
|
|
// invoke callback
|
|
|
|
if (tud_umount_cb) tud_umount_cb();
|
2018-11-14 17:40:29 +07:00
|
|
|
break;
|
|
|
|
|
|
|
|
case DCD_EVENT_SOF:
|
|
|
|
for ( uint8_t i = 0; i < USBD_CLASS_DRIVER_COUNT; i++ )
|
|
|
|
{
|
|
|
|
if ( usbd_class_drivers[i].sof )
|
|
|
|
{
|
|
|
|
usbd_class_drivers[i].sof(event.rhport);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case USBD_EVT_FUNC_CALL:
|
|
|
|
if ( event.func_call.func ) event.func_call.func(event.func_call.param);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
TU_BREAKPOINT();
|
|
|
|
break;
|
2018-08-28 15:56:43 +07:00
|
|
|
}
|
2014-03-31 13:12:51 +07:00
|
|
|
}
|
2018-11-14 17:40:29 +07:00
|
|
|
}
|
2014-03-31 13:12:51 +07:00
|
|
|
|
2018-11-14 17:40:29 +07:00
|
|
|
/* USB device task
|
|
|
|
* Thread that handles all device events. With an real RTOS, the task must be a forever loop and never return.
|
2018-12-05 18:58:30 +07:00
|
|
|
* For coding convenience with no RTOS, we use wrapped sub-function for processing to easily return at any time.
|
2018-11-14 17:40:29 +07:00
|
|
|
*/
|
|
|
|
void usbd_task( void* param)
|
|
|
|
{
|
|
|
|
(void) param;
|
|
|
|
|
|
|
|
#if CFG_TUSB_OS != OPT_OS_NONE
|
|
|
|
while (1) {
|
|
|
|
#endif
|
|
|
|
|
|
|
|
usbd_task_body();
|
|
|
|
|
|
|
|
#if CFG_TUSB_OS != OPT_OS_NONE
|
|
|
|
}
|
|
|
|
#endif
|
2014-03-31 13:12:51 +07:00
|
|
|
}
|
|
|
|
|
2018-11-16 21:56:39 +07:00
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
// Control Request Parser & Handling
|
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
|
|
|
|
// This handles the actual request and its response.
|
|
|
|
// return false will cause its caller to stall control endpoint
|
|
|
|
static bool process_control_request(uint8_t rhport, tusb_control_request_t const * p_request)
|
|
|
|
{
|
|
|
|
usbd_control_set_complete_callback(NULL);
|
|
|
|
|
|
|
|
if ( TUSB_REQ_RCPT_DEVICE == p_request->bmRequestType_bit.recipient &&
|
|
|
|
TUSB_REQ_TYPE_STANDARD == p_request->bmRequestType_bit.type )
|
|
|
|
{
|
|
|
|
//------------- Standard Device Requests e.g in enumeration -------------//
|
|
|
|
void* data_buf = NULL;
|
|
|
|
uint16_t data_len = 0;
|
|
|
|
|
|
|
|
switch ( p_request->bRequest )
|
2018-11-08 13:45:30 -08:00
|
|
|
{
|
2018-11-16 21:56:39 +07:00
|
|
|
case TUSB_REQ_SET_ADDRESS:
|
2018-11-20 17:25:41 +07:00
|
|
|
// response with status first before changing device address
|
|
|
|
usbd_control_status(rhport, p_request);
|
2018-11-16 21:56:39 +07:00
|
|
|
dcd_set_address(rhport, (uint8_t) p_request->wValue);
|
2018-11-20 17:25:41 +07:00
|
|
|
return true; // skip the rest
|
2018-11-16 21:56:39 +07:00
|
|
|
break;
|
|
|
|
|
|
|
|
case TUSB_REQ_GET_CONFIGURATION:
|
|
|
|
data_buf = &_usbd_dev.config_num;
|
|
|
|
data_len = 1;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case TUSB_REQ_SET_CONFIGURATION:
|
|
|
|
{
|
|
|
|
uint8_t const config = (uint8_t) p_request->wValue;
|
|
|
|
|
|
|
|
dcd_set_config(rhport, config);
|
|
|
|
_usbd_dev.config_num = config;
|
|
|
|
|
2018-11-26 12:25:28 +07:00
|
|
|
TU_ASSERT( TUSB_ERROR_NONE == process_set_config(rhport) );
|
2018-11-08 13:45:30 -08:00
|
|
|
}
|
2018-11-16 21:56:39 +07:00
|
|
|
break;
|
|
|
|
|
|
|
|
case TUSB_REQ_GET_DESCRIPTOR:
|
|
|
|
data_buf = (void*) get_descriptor(p_request, &data_len);
|
|
|
|
if ( data_buf == NULL || data_len == 0 ) return false;
|
|
|
|
break;
|
|
|
|
|
2018-11-20 17:25:41 +07:00
|
|
|
default:
|
|
|
|
TU_BREAKPOINT();
|
|
|
|
return false;
|
2018-11-08 13:45:30 -08:00
|
|
|
}
|
|
|
|
|
2018-11-16 21:56:39 +07:00
|
|
|
usbd_control_xfer(rhport, p_request, data_buf, data_len);
|
|
|
|
}
|
|
|
|
else if ( TUSB_REQ_RCPT_INTERFACE == p_request->bmRequestType_bit.recipient )
|
|
|
|
{
|
|
|
|
//------------- Class/Interface Specific Request -------------//
|
|
|
|
uint8_t const itf = tu_u16_low(p_request->wIndex);
|
|
|
|
uint8_t const drvid = _usbd_dev.itf2drv[ itf ];
|
|
|
|
|
2018-12-10 05:15:49 +07:00
|
|
|
TU_VERIFY(drvid < USBD_CLASS_DRIVER_COUNT);
|
2018-11-16 21:56:39 +07:00
|
|
|
|
|
|
|
usbd_control_set_complete_callback(usbd_class_drivers[drvid].control_request_complete );
|
|
|
|
|
|
|
|
// control endpoint will be stalled if driver return false
|
|
|
|
return usbd_class_drivers[drvid].control_request(rhport, p_request);
|
|
|
|
}
|
|
|
|
else if ( p_request->bmRequestType_bit.recipient == TUSB_REQ_RCPT_ENDPOINT &&
|
|
|
|
p_request->bmRequestType_bit.type == TUSB_REQ_TYPE_STANDARD )
|
|
|
|
{
|
|
|
|
//------------- Endpoint Request -------------//
|
|
|
|
switch ( p_request->bRequest )
|
2018-07-25 21:21:33 +07:00
|
|
|
{
|
2018-11-16 21:56:39 +07:00
|
|
|
case TUSB_REQ_GET_STATUS:
|
|
|
|
{
|
|
|
|
uint16_t status = dcd_edpt_stalled(rhport, tu_u16_low(p_request->wIndex)) ? 0x0001 : 0x0000;
|
|
|
|
usbd_control_xfer(rhport, p_request, &status, 2);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case TUSB_REQ_CLEAR_FEATURE:
|
|
|
|
// only endpoint feature is halted/stalled
|
|
|
|
dcd_edpt_clear_stall(rhport, tu_u16_low(p_request->wIndex));
|
|
|
|
usbd_control_status(rhport, p_request);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case TUSB_REQ_SET_FEATURE:
|
|
|
|
// only endpoint feature is halted/stalled
|
|
|
|
dcd_edpt_stall(rhport, tu_u16_low(p_request->wIndex));
|
|
|
|
usbd_control_status(rhport, p_request);
|
|
|
|
break;
|
|
|
|
|
2018-11-20 17:25:41 +07:00
|
|
|
default:
|
|
|
|
TU_BREAKPOINT();
|
|
|
|
return false;
|
2018-07-25 21:21:33 +07:00
|
|
|
}
|
2018-11-16 21:56:39 +07:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
//------------- Unsupported Request -------------//
|
2018-11-20 17:25:41 +07:00
|
|
|
TU_BREAKPOINT();
|
2018-11-16 21:56:39 +07:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
2014-03-12 14:01:38 +07:00
|
|
|
}
|
|
|
|
|
2018-07-13 17:48:26 +07:00
|
|
|
// Process Set Configure Request
|
2018-11-16 21:56:39 +07:00
|
|
|
// This function parse configuration descriptor & open drivers accordingly
|
2018-11-26 12:25:28 +07:00
|
|
|
static bool process_set_config(uint8_t rhport)
|
2014-03-12 14:01:38 +07:00
|
|
|
{
|
2018-08-01 00:50:04 +07:00
|
|
|
uint8_t const * desc_cfg = (uint8_t const *) usbd_desc_set->config;
|
2018-11-16 21:56:39 +07:00
|
|
|
TU_ASSERT(desc_cfg != NULL);
|
|
|
|
|
2018-07-01 15:11:58 +07:00
|
|
|
uint8_t const * p_desc = desc_cfg + sizeof(tusb_desc_configuration_t);
|
|
|
|
uint16_t const cfg_len = ((tusb_desc_configuration_t*)desc_cfg)->wTotalLength;
|
2014-03-23 15:39:55 +07:00
|
|
|
|
2018-07-01 15:11:58 +07:00
|
|
|
while( p_desc < desc_cfg + cfg_len )
|
2014-03-12 14:01:38 +07:00
|
|
|
{
|
2018-11-16 21:56:39 +07:00
|
|
|
// Each interface always starts with Interface or Association descriptor
|
2018-07-13 17:48:26 +07:00
|
|
|
if ( TUSB_DESC_INTERFACE_ASSOCIATION == descriptor_type(p_desc) )
|
2014-03-12 14:01:38 +07:00
|
|
|
{
|
2018-07-13 17:48:26 +07:00
|
|
|
p_desc = descriptor_next(p_desc); // ignore Interface Association
|
2014-03-12 14:01:38 +07:00
|
|
|
}else
|
|
|
|
{
|
2018-11-16 21:56:39 +07:00
|
|
|
TU_ASSERT( TUSB_DESC_INTERFACE == descriptor_type(p_desc) );
|
2014-03-12 14:01:38 +07:00
|
|
|
|
2018-11-16 21:56:39 +07:00
|
|
|
tusb_desc_interface_t* desc_itf = (tusb_desc_interface_t*) p_desc;
|
2014-03-12 14:01:38 +07:00
|
|
|
|
2018-05-14 13:02:01 +07:00
|
|
|
// Check if class is supported
|
2018-07-13 17:48:26 +07:00
|
|
|
uint8_t drv_id;
|
|
|
|
for (drv_id = 0; drv_id < USBD_CLASS_DRIVER_COUNT; drv_id++)
|
2018-05-14 13:02:01 +07:00
|
|
|
{
|
2018-11-16 21:56:39 +07:00
|
|
|
if ( usbd_class_drivers[drv_id].class_code == desc_itf->bInterfaceClass ) break;
|
2018-05-14 13:02:01 +07:00
|
|
|
}
|
2018-12-10 05:07:22 +07:00
|
|
|
TU_ASSERT( drv_id < USBD_CLASS_DRIVER_COUNT );
|
2014-03-12 14:01:38 +07:00
|
|
|
|
2018-11-16 21:56:39 +07:00
|
|
|
// Interface number must not be used already TODO alternate interface
|
|
|
|
TU_ASSERT( 0xff == _usbd_dev.itf2drv[desc_itf->bInterfaceNumber] );
|
|
|
|
_usbd_dev.itf2drv[desc_itf->bInterfaceNumber] = drv_id;
|
2018-07-13 17:48:26 +07:00
|
|
|
|
2018-12-10 05:07:22 +07:00
|
|
|
uint16_t itf_len=0;
|
|
|
|
TU_ASSERT_ERR( usbd_class_drivers[drv_id].open( rhport, desc_itf, &itf_len ), false );
|
|
|
|
TU_ASSERT( itf_len >= sizeof(tusb_desc_interface_t) );
|
2014-03-12 14:01:38 +07:00
|
|
|
|
2018-12-10 05:15:49 +07:00
|
|
|
mark_interface_endpoint(_usbd_dev.ep2drv, p_desc, itf_len, drv_id);
|
2014-03-12 14:01:38 +07:00
|
|
|
|
2018-12-10 05:07:22 +07:00
|
|
|
p_desc += itf_len; // next interface
|
2014-03-12 14:01:38 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-07 16:39:33 +07:00
|
|
|
// invoke callback
|
2018-12-10 19:25:57 +07:00
|
|
|
if (tud_mount_cb) tud_mount_cb();
|
2018-03-07 16:39:33 +07:00
|
|
|
|
2014-03-12 14:01:38 +07:00
|
|
|
return TUSB_ERROR_NONE;
|
|
|
|
}
|
|
|
|
|
2018-07-13 17:48:26 +07:00
|
|
|
// Helper marking endpoint of interface belongs to class driver
|
2018-12-10 05:15:49 +07:00
|
|
|
static void mark_interface_endpoint(uint8_t ep2drv[8][2], uint8_t const* p_desc, uint16_t desc_len, uint8_t driver_id)
|
2018-07-13 17:48:26 +07:00
|
|
|
{
|
|
|
|
uint16_t len = 0;
|
|
|
|
|
|
|
|
while( len < desc_len )
|
|
|
|
{
|
|
|
|
if ( TUSB_DESC_ENDPOINT == descriptor_type(p_desc) )
|
|
|
|
{
|
|
|
|
uint8_t const ep_addr = ((tusb_desc_endpoint_t const*) p_desc)->bEndpointAddress;
|
|
|
|
|
2018-12-12 11:51:31 +07:00
|
|
|
ep2drv[tu_edpt_number(ep_addr)][tu_edpt_dir(ep_addr)] = driver_id;
|
2018-07-13 17:48:26 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
len += descriptor_len(p_desc);
|
|
|
|
p_desc = descriptor_next(p_desc);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-16 21:56:39 +07:00
|
|
|
// return descriptor's buffer and update desc_len
|
|
|
|
static void const* get_descriptor(tusb_control_request_t const * p_request, uint16_t* desc_len)
|
|
|
|
{
|
|
|
|
tusb_desc_type_t const desc_type = (tusb_desc_type_t) tu_u16_high(p_request->wValue);
|
|
|
|
uint8_t const desc_index = tu_u16_low( p_request->wValue );
|
|
|
|
|
|
|
|
uint8_t const * desc_data = NULL;
|
|
|
|
uint16_t len = 0;
|
|
|
|
|
|
|
|
*desc_len = 0;
|
|
|
|
|
|
|
|
switch(desc_type)
|
|
|
|
{
|
|
|
|
case TUSB_DESC_DEVICE:
|
|
|
|
desc_data = (uint8_t const *) usbd_desc_set->device;
|
|
|
|
len = sizeof(tusb_desc_device_t);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case TUSB_DESC_CONFIGURATION:
|
|
|
|
desc_data = (uint8_t const *) usbd_desc_set->config;
|
|
|
|
len = ((tusb_desc_configuration_t const*) desc_data)->wTotalLength;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case TUSB_DESC_STRING:
|
|
|
|
// String Descriptor always uses the desc set from user
|
|
|
|
if ( desc_index < tud_desc_set.string_count )
|
|
|
|
{
|
|
|
|
desc_data = tud_desc_set.string_arr[desc_index];
|
|
|
|
TU_VERIFY( desc_data != NULL, NULL );
|
|
|
|
|
|
|
|
len = desc_data[0]; // first byte of descriptor is its size
|
|
|
|
}else
|
|
|
|
{
|
|
|
|
// out of range
|
|
|
|
/* The 0xEE index string is a Microsoft USB extension.
|
|
|
|
* It can be used to tell Windows what driver it should use for the device !!!
|
|
|
|
*/
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case TUSB_DESC_DEVICE_QUALIFIER:
|
|
|
|
// TODO If not highspeed capable stall this request otherwise
|
|
|
|
// return the descriptor that could work in highspeed
|
|
|
|
return NULL;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default: return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
*desc_len = len;
|
|
|
|
return desc_data;
|
|
|
|
}
|
|
|
|
|
2014-03-12 14:01:38 +07:00
|
|
|
//--------------------------------------------------------------------+
|
2018-11-16 21:56:39 +07:00
|
|
|
// DCD Event Handler
|
2014-03-12 14:01:38 +07:00
|
|
|
//--------------------------------------------------------------------+
|
2018-10-23 15:08:31 +07:00
|
|
|
void dcd_event_handler(dcd_event_t const * event, bool in_isr)
|
|
|
|
{
|
|
|
|
switch (event->event_id)
|
|
|
|
{
|
2018-10-23 15:12:30 +07:00
|
|
|
case DCD_EVENT_BUS_RESET:
|
|
|
|
case DCD_EVENT_UNPLUGGED:
|
2018-11-02 17:29:49 +07:00
|
|
|
osal_queue_send(_usbd_q, event, in_isr);
|
2018-10-23 15:08:31 +07:00
|
|
|
break;
|
|
|
|
|
2018-11-20 17:25:41 +07:00
|
|
|
case DCD_EVENT_SOF:
|
|
|
|
// nothing to do now
|
|
|
|
break;
|
|
|
|
|
2018-10-23 15:12:30 +07:00
|
|
|
case DCD_EVENT_SUSPENDED:
|
2018-10-23 15:08:31 +07:00
|
|
|
// TODO support suspended
|
|
|
|
break;
|
|
|
|
|
2018-10-23 15:12:30 +07:00
|
|
|
case DCD_EVENT_RESUME:
|
2018-10-23 15:08:31 +07:00
|
|
|
// TODO support resume
|
|
|
|
break;
|
|
|
|
|
2018-10-23 16:07:48 +07:00
|
|
|
case DCD_EVENT_SETUP_RECEIVED:
|
|
|
|
osal_queue_send(_usbd_q, event, in_isr);
|
|
|
|
break;
|
2018-10-23 15:08:31 +07:00
|
|
|
|
2018-10-24 00:44:26 +07:00
|
|
|
case DCD_EVENT_XFER_COMPLETE:
|
2018-11-16 21:56:39 +07:00
|
|
|
// skip zero-length control status complete event, should dcd notifies us.
|
2018-12-12 11:51:31 +07:00
|
|
|
if ( 0 == tu_edpt_number(event->xfer_complete.ep_addr) && event->xfer_complete.len == 0) break;
|
2018-11-16 21:56:39 +07:00
|
|
|
|
2018-11-07 23:04:34 -08:00
|
|
|
osal_queue_send(_usbd_q, event, in_isr);
|
2018-11-23 15:22:46 +07:00
|
|
|
TU_ASSERT(event->xfer_complete.result == XFER_RESULT_SUCCESS,);
|
2018-10-24 00:44:26 +07:00
|
|
|
break;
|
|
|
|
|
2018-12-05 17:01:19 +07:00
|
|
|
// Not an DCD event, just a convenient way to defer ISR function should we need
|
|
|
|
case USBD_EVT_FUNC_CALL:
|
|
|
|
osal_queue_send(_usbd_q, event, in_isr);
|
|
|
|
break;
|
|
|
|
|
2018-10-23 16:07:48 +07:00
|
|
|
default: break;
|
2018-10-23 15:08:31 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-24 23:55:10 -07:00
|
|
|
// helper to send bus signal event
|
|
|
|
void dcd_event_bus_signal (uint8_t rhport, dcd_eventid_t eid, bool in_isr)
|
|
|
|
{
|
2018-11-26 12:25:28 +07:00
|
|
|
dcd_event_t event = { .rhport = rhport, .event_id = eid, };
|
2018-10-24 23:55:10 -07:00
|
|
|
dcd_event_handler(&event, in_isr);
|
|
|
|
}
|
|
|
|
|
|
|
|
// helper to send setup received
|
|
|
|
void dcd_event_setup_received(uint8_t rhport, uint8_t const * setup, bool in_isr)
|
|
|
|
{
|
2018-11-26 12:25:28 +07:00
|
|
|
dcd_event_t event = { .rhport = rhport, .event_id = DCD_EVENT_SETUP_RECEIVED };
|
2018-10-24 23:55:10 -07:00
|
|
|
memcpy(&event.setup_received, setup, 8);
|
|
|
|
|
2018-11-26 12:25:28 +07:00
|
|
|
dcd_event_handler(&event, in_isr);
|
2018-10-24 23:55:10 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// helper to send transfer complete event
|
|
|
|
void dcd_event_xfer_complete (uint8_t rhport, uint8_t ep_addr, uint32_t xferred_bytes, uint8_t result, bool in_isr)
|
|
|
|
{
|
2018-11-26 12:25:28 +07:00
|
|
|
dcd_event_t event = { .rhport = rhport, .event_id = DCD_EVENT_XFER_COMPLETE };
|
2018-10-24 23:55:10 -07:00
|
|
|
|
|
|
|
event.xfer_complete.ep_addr = ep_addr;
|
|
|
|
event.xfer_complete.len = xferred_bytes;
|
|
|
|
event.xfer_complete.result = result;
|
|
|
|
|
|
|
|
dcd_event_handler(&event, in_isr);
|
|
|
|
}
|
|
|
|
|
2014-03-12 14:01:38 +07:00
|
|
|
//--------------------------------------------------------------------+
|
2018-06-22 12:53:13 +07:00
|
|
|
// Helper
|
2014-03-12 14:01:38 +07:00
|
|
|
//--------------------------------------------------------------------+
|
2018-12-05 17:01:19 +07:00
|
|
|
|
|
|
|
// Helper to parse an pair of endpoint descriptors (IN & OUT)
|
2018-12-10 19:01:28 +07:00
|
|
|
tusb_error_t usbd_open_edpt_pair(uint8_t rhport, tusb_desc_endpoint_t const* ep_desc, uint8_t xfer_type, uint8_t* ep_out, uint8_t* ep_in)
|
2018-06-16 12:30:10 +07:00
|
|
|
{
|
|
|
|
for(int i=0; i<2; i++)
|
|
|
|
{
|
2018-12-10 19:01:28 +07:00
|
|
|
TU_ASSERT(TUSB_DESC_ENDPOINT == ep_desc->bDescriptorType &&
|
|
|
|
xfer_type == ep_desc->bmAttributes.xfer, TUSB_ERROR_DESCRIPTOR_CORRUPTED);
|
2018-06-16 12:30:10 +07:00
|
|
|
|
2018-12-10 19:01:28 +07:00
|
|
|
TU_ASSERT( dcd_edpt_open(rhport, ep_desc), TUSB_ERROR_DCD_OPEN_PIPE_FAILED );
|
2018-06-16 12:30:10 +07:00
|
|
|
|
2018-12-12 11:51:31 +07:00
|
|
|
if ( tu_edpt_dir(ep_desc->bEndpointAddress) == TUSB_DIR_IN )
|
2018-06-16 12:30:10 +07:00
|
|
|
{
|
2018-12-10 19:01:28 +07:00
|
|
|
(*ep_in) = ep_desc->bEndpointAddress;
|
2018-06-16 12:30:10 +07:00
|
|
|
}else
|
|
|
|
{
|
2018-12-10 19:01:28 +07:00
|
|
|
(*ep_out) = ep_desc->bEndpointAddress;
|
2018-06-16 12:30:10 +07:00
|
|
|
}
|
2014-03-12 14:01:38 +07:00
|
|
|
|
2018-12-10 19:01:28 +07:00
|
|
|
ep_desc = (tusb_desc_endpoint_t const *) descriptor_next( (uint8_t const*) ep_desc );
|
2018-06-16 12:30:10 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
return TUSB_ERROR_NONE;
|
|
|
|
}
|
2018-06-22 12:53:13 +07:00
|
|
|
|
2018-12-05 17:01:19 +07:00
|
|
|
// Helper to defer an isr function
|
|
|
|
void usbd_defer_func(osal_task_func_t func, void* param, bool in_isr)
|
2018-06-22 12:53:13 +07:00
|
|
|
{
|
2018-10-23 15:08:31 +07:00
|
|
|
dcd_event_t event =
|
2018-06-22 12:53:13 +07:00
|
|
|
{
|
2018-07-13 14:26:40 +07:00
|
|
|
.rhport = 0,
|
|
|
|
.event_id = USBD_EVT_FUNC_CALL,
|
2018-06-22 12:53:13 +07:00
|
|
|
};
|
|
|
|
|
|
|
|
event.func_call.func = func;
|
|
|
|
event.func_call.param = param;
|
|
|
|
|
2018-12-05 17:01:19 +07:00
|
|
|
dcd_event_handler(&event, in_isr);
|
2018-06-22 12:53:13 +07:00
|
|
|
}
|
|
|
|
|
2014-03-12 14:01:38 +07:00
|
|
|
#endif
|