2019-03-20 16:11:42 +07:00
|
|
|
/*
|
|
|
|
* The MIT License (MIT)
|
|
|
|
*
|
2019-05-14 11:48:05 +07:00
|
|
|
* Copyright (c) 2019 Ha Thach (tinyusb.org)
|
2019-03-20 16:11:42 +07:00
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
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
|
|
|
|
|
|
|
#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"
|
2019-06-10 18:46:00 +07:00
|
|
|
#include "dcd.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-04-16 13:46:28 +07:00
|
|
|
//--------------------------------------------------------------------+
|
2018-07-13 15:08:38 +07:00
|
|
|
// Device Data
|
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
typedef struct {
|
2019-06-06 10:39:37 +07:00
|
|
|
struct TU_ATTR_PACKED
|
2019-03-30 02:26:15 +07:00
|
|
|
{
|
2019-07-19 16:49:04 +07:00
|
|
|
volatile uint8_t connected : 1;
|
|
|
|
volatile uint8_t configured : 1;
|
|
|
|
volatile uint8_t suspended : 1;
|
2019-03-30 13:48:15 +07:00
|
|
|
|
2019-07-19 16:49:04 +07:00
|
|
|
uint8_t remote_wakeup_en : 1; // enable/disable by host
|
|
|
|
uint8_t remote_wakeup_support : 1; // configuration descriptor's attribute
|
|
|
|
uint8_t self_powered : 1; // configuration descriptor's attribute
|
2019-03-30 02:26:15 +07:00
|
|
|
};
|
2018-11-16 21:56:39 +07:00
|
|
|
|
2019-07-19 16:49:04 +07:00
|
|
|
uint8_t itf2drv[16]; // map interface number to driver (0xff is invalid)
|
|
|
|
uint8_t ep2drv[8][2]; // map endpoint to driver ( 0xff is invalid )
|
2019-10-01 22:19:04 +07:00
|
|
|
|
|
|
|
struct TU_ATTR_PACKED
|
|
|
|
{
|
|
|
|
volatile bool busy : 1;
|
|
|
|
volatile bool stalled : 1;
|
|
|
|
|
|
|
|
// TODO merge ep2drv here, 4-bit should be sufficient
|
|
|
|
}ep_status[8][2];
|
2018-07-13 15:08:38 +07:00
|
|
|
}usbd_device_t;
|
|
|
|
|
2019-10-01 14:43:34 +07:00
|
|
|
static usbd_device_t _usbd_dev;
|
2018-07-13 15:08:38 +07:00
|
|
|
|
2019-09-13 22:16:24 +07:00
|
|
|
// Invalid driver ID in itf2drv[] ep2drv[][] mapping
|
2019-09-16 11:13:17 +07:00
|
|
|
enum { DRVID_INVALID = 0xFFu };
|
2019-09-13 22:16:24 +07:00
|
|
|
|
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;
|
|
|
|
|
2019-07-16 18:14:47 +07:00
|
|
|
void (* init ) (void);
|
2019-07-19 20:20:13 +07:00
|
|
|
void (* reset ) (uint8_t rhport);
|
2019-07-16 18:14:47 +07:00
|
|
|
bool (* open ) (uint8_t rhport, tusb_desc_interface_t const * desc_intf, uint16_t* p_length);
|
|
|
|
bool (* control_request ) (uint8_t rhport, tusb_control_request_t const * request);
|
2019-07-12 22:03:40 +07:00
|
|
|
bool (* control_complete ) (uint8_t rhport, tusb_control_request_t const * request);
|
2019-07-19 16:49:04 +07:00
|
|
|
bool (* xfer_cb ) (uint8_t rhport, uint8_t ep_addr, xfer_result_t event, uint32_t xferred_bytes);
|
2019-07-16 18:14:47 +07:00
|
|
|
void (* sof ) (uint8_t rhport);
|
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
|
2019-07-19 16:49:04 +07:00
|
|
|
{
|
|
|
|
.class_code = TUSB_CLASS_CDC,
|
|
|
|
.init = cdcd_init,
|
2019-07-19 20:20:13 +07:00
|
|
|
.reset = cdcd_reset,
|
2019-07-19 16:49:04 +07:00
|
|
|
.open = cdcd_open,
|
|
|
|
.control_request = cdcd_control_request,
|
|
|
|
.control_complete = cdcd_control_complete,
|
|
|
|
.xfer_cb = cdcd_xfer_cb,
|
2019-07-19 20:20:13 +07:00
|
|
|
.sof = NULL
|
2019-07-19 16:49:04 +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
|
2019-07-19 16:49:04 +07:00
|
|
|
{
|
|
|
|
.class_code = TUSB_CLASS_MSC,
|
|
|
|
.init = mscd_init,
|
2019-07-19 20:20:13 +07:00
|
|
|
.reset = mscd_reset,
|
2019-07-19 16:49:04 +07:00
|
|
|
.open = mscd_open,
|
|
|
|
.control_request = mscd_control_request,
|
|
|
|
.control_complete = mscd_control_complete,
|
|
|
|
.xfer_cb = mscd_xfer_cb,
|
2019-07-19 20:20:13 +07:00
|
|
|
.sof = NULL
|
2019-07-19 16:49:04 +07:00
|
|
|
},
|
2018-06-16 12:30:10 +07:00
|
|
|
#endif
|
|
|
|
|
2018-07-28 12:38:45 +07:00
|
|
|
#if CFG_TUD_HID
|
2019-07-19 16:49:04 +07:00
|
|
|
{
|
|
|
|
.class_code = TUSB_CLASS_HID,
|
|
|
|
.init = hidd_init,
|
2019-07-19 20:20:13 +07:00
|
|
|
.reset = hidd_reset,
|
2019-07-19 16:49:04 +07:00
|
|
|
.open = hidd_open,
|
|
|
|
.control_request = hidd_control_request,
|
|
|
|
.control_complete = hidd_control_complete,
|
|
|
|
.xfer_cb = hidd_xfer_cb,
|
2019-07-19 20:20:13 +07:00
|
|
|
.sof = NULL
|
2019-07-19 16:49:04 +07:00
|
|
|
},
|
2018-07-13 17:48:26 +07:00
|
|
|
#endif
|
|
|
|
|
2018-11-16 17:03:33 -08:00
|
|
|
#if CFG_TUD_MIDI
|
2019-07-19 16:49:04 +07:00
|
|
|
{
|
|
|
|
.class_code = TUSB_CLASS_AUDIO,
|
|
|
|
.init = midid_init,
|
|
|
|
.open = midid_open,
|
2019-07-19 20:20:13 +07:00
|
|
|
.reset = midid_reset,
|
2019-07-19 16:49:04 +07:00
|
|
|
.control_request = midid_control_request,
|
|
|
|
.control_complete = midid_control_complete,
|
|
|
|
.xfer_cb = midid_xfer_cb,
|
2019-07-19 20:20:13 +07:00
|
|
|
.sof = NULL
|
2019-07-19 16:49:04 +07:00
|
|
|
},
|
2018-11-16 17:03:33 -08:00
|
|
|
#endif
|
|
|
|
|
2019-07-19 16:23:56 +07:00
|
|
|
#if CFG_TUD_VENDOR
|
2019-07-19 16:49:04 +07:00
|
|
|
{
|
|
|
|
.class_code = TUSB_CLASS_VENDOR_SPECIFIC,
|
2019-07-19 20:20:13 +07:00
|
|
|
.init = vendord_init,
|
|
|
|
.reset = vendord_reset,
|
|
|
|
.open = vendord_open,
|
2019-07-26 11:36:30 +07:00
|
|
|
.control_request = tud_vendor_control_request_cb,
|
|
|
|
.control_complete = tud_vendor_control_complete_cb,
|
2019-07-19 20:20:13 +07:00
|
|
|
.xfer_cb = vendord_xfer_cb,
|
|
|
|
.sof = NULL
|
2019-09-14 12:13:11 -04:00
|
|
|
},
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if CFG_TUD_USBTMC
|
|
|
|
// Presently USBTMC is the only defined class with the APP_SPECIFIC class code.
|
|
|
|
// We maybe need to add subclass codes here, or a callback to ask if a driver can
|
|
|
|
// handle a particular interface.
|
|
|
|
{
|
2019-09-19 19:51:37 -04:00
|
|
|
.class_code = TUD_USBTMC_APP_CLASS,
|
|
|
|
//.subclass_code = TUD_USBTMC_APP_SUBCLASS
|
2019-09-19 20:08:45 -04:00
|
|
|
.init = usbtmcd_init_cb,
|
|
|
|
.reset = usbtmcd_reset_cb,
|
|
|
|
.open = usbtmcd_open_cb,
|
|
|
|
.control_request = usbtmcd_control_request_cb,
|
|
|
|
.control_complete = usbtmcd_control_complete_cb,
|
2019-09-14 12:13:11 -04:00
|
|
|
.xfer_cb = usbtmcd_xfer_cb,
|
|
|
|
.sof = NULL
|
2019-07-19 16:49:04 +07:00
|
|
|
},
|
2014-03-14 17:18:05 +07:00
|
|
|
#endif
|
2019-10-24 00:42:42 +02:00
|
|
|
|
|
|
|
#if CFG_TUD_DFU_RT
|
|
|
|
{
|
|
|
|
.class_code = TUD_DFU_APP_CLASS,
|
|
|
|
//.subclass_code = TUD_DFU_APP_SUBCLASS
|
|
|
|
.init = dfu_rtd_init,
|
|
|
|
.reset = dfu_rtd_reset,
|
|
|
|
.open = dfu_rtd_open,
|
|
|
|
.control_request = dfu_rtd_control_request,
|
|
|
|
.control_complete = dfu_rtd_control_complete,
|
|
|
|
.xfer_cb = dfu_rtd_xfer_cb,
|
|
|
|
.sof = NULL
|
|
|
|
},
|
|
|
|
#endif
|
2014-03-12 14:01:38 +07:00
|
|
|
};
|
|
|
|
|
2019-09-13 22:16:24 +07:00
|
|
|
enum { USBD_CLASS_DRIVER_COUNT = TU_ARRAY_SIZE(usbd_class_drivers) };
|
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-05-17 20:04:52 +07:00
|
|
|
|
2018-12-05 17:01:19 +07:00
|
|
|
// Event queue
|
2018-12-13 13:49:09 +07:00
|
|
|
// OPT_MODE_DEVICE is used by OS NONE for mutex (disable usb isr)
|
2018-12-05 17:01:19 +07:00
|
|
|
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);
|
2019-05-12 19:55:15 +07:00
|
|
|
static bool process_set_config(uint8_t rhport, uint8_t cfg_num);
|
2019-05-11 16:31:08 +07:00
|
|
|
static bool process_get_descriptor(uint8_t rhport, tusb_control_request_t const * p_request);
|
2018-11-16 21:56:39 +07:00
|
|
|
|
|
|
|
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
|
|
|
|
2019-10-23 21:18:46 +07:00
|
|
|
|
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
// Debugging
|
|
|
|
//--------------------------------------------------------------------+
|
2019-10-24 12:00:06 +07:00
|
|
|
#if CFG_TUSB_DEBUG > 1
|
2019-10-23 21:18:46 +07:00
|
|
|
static char const* const _usbd_event_str[DCD_EVENT_COUNT] =
|
|
|
|
{
|
|
|
|
"INVALID" ,
|
|
|
|
"BUS_RESET" ,
|
|
|
|
"UNPLUGGED" ,
|
|
|
|
"SOF" ,
|
|
|
|
"SUSPEND" ,
|
|
|
|
"RESUME" ,
|
|
|
|
"SETUP_RECEIVED" ,
|
|
|
|
"XFER_COMPLETE" ,
|
|
|
|
"FUNC_CALL"
|
|
|
|
};
|
2019-10-24 12:00:06 +07:00
|
|
|
|
|
|
|
// must be same driver order as usbd_class_drivers[]
|
|
|
|
static char const* const _usbd_driver_str[USBD_CLASS_DRIVER_COUNT] =
|
|
|
|
{
|
|
|
|
#if CFG_TUD_CDC
|
|
|
|
"CDC",
|
|
|
|
#endif
|
|
|
|
#if CFG_TUD_MSC
|
|
|
|
"MSC",
|
|
|
|
#endif
|
|
|
|
#if CFG_TUD_HID
|
|
|
|
"HID",
|
|
|
|
#endif
|
|
|
|
#if CFG_TUD_MIDI
|
|
|
|
"MIDI",
|
|
|
|
#endif
|
|
|
|
#if CFG_TUD_VENDOR
|
|
|
|
"Vendor",
|
|
|
|
#endif
|
|
|
|
#if CFG_TUD_USBTMC
|
|
|
|
"USBTMC"
|
|
|
|
#endif
|
|
|
|
};
|
|
|
|
|
|
|
|
static char const* const _tusb_std_request_str[] =
|
|
|
|
{
|
|
|
|
"Get Status" ,
|
|
|
|
"Clear Feature" ,
|
|
|
|
"Reserved" ,
|
|
|
|
"Set Feature" ,
|
|
|
|
"Reserved" ,
|
|
|
|
"Set Address" ,
|
|
|
|
"Get Descriptor" ,
|
|
|
|
"Set Descriptor" ,
|
|
|
|
"Get Configuration" ,
|
|
|
|
"Set Configuration" ,
|
|
|
|
"Get Interface" ,
|
|
|
|
"Set Interface" ,
|
|
|
|
"Synch Frame"
|
|
|
|
};
|
|
|
|
|
2019-10-23 21:18:46 +07:00
|
|
|
#endif
|
|
|
|
|
2018-05-17 19:19:55 +07:00
|
|
|
//--------------------------------------------------------------------+
|
2019-03-29 16:37:08 +07:00
|
|
|
// Application API
|
2018-05-17 19:19:55 +07:00
|
|
|
//--------------------------------------------------------------------+
|
2018-07-12 22:25:06 +07:00
|
|
|
bool tud_mounted(void)
|
2018-05-17 19:19:55 +07:00
|
|
|
{
|
2019-03-30 13:48:15 +07:00
|
|
|
return _usbd_dev.configured;
|
2018-05-17 19:19:55 +07:00
|
|
|
}
|
|
|
|
|
2019-03-30 23:01:23 +07:00
|
|
|
bool tud_suspended(void)
|
|
|
|
{
|
|
|
|
return _usbd_dev.suspended;
|
|
|
|
}
|
|
|
|
|
2019-03-29 16:37:08 +07:00
|
|
|
bool tud_remote_wakeup(void)
|
|
|
|
{
|
2019-03-30 17:38:00 +07:00
|
|
|
// only wake up host if this feature is supported and enabled and we are suspended
|
|
|
|
TU_VERIFY (_usbd_dev.suspended && _usbd_dev.remote_wakeup_support && _usbd_dev.remote_wakeup_en );
|
|
|
|
dcd_remote_wakeup(TUD_OPT_RHPORT);
|
|
|
|
return true;
|
2019-03-29 16:37:08 +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
|
|
|
//--------------------------------------------------------------------+
|
2019-10-30 23:26:34 +07:00
|
|
|
bool tud_init (void)
|
2014-03-31 13:12:51 +07:00
|
|
|
{
|
2019-10-24 12:00:06 +07:00
|
|
|
TU_LOG2("USBD init\r\n");
|
2019-10-23 21:18:46 +07:00
|
|
|
|
2019-03-30 14:34:38 +07:00
|
|
|
tu_varclr(&_usbd_dev);
|
|
|
|
|
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
|
|
|
|
|
|
|
// Init class drivers
|
2019-10-24 12:00:06 +07:00
|
|
|
for (uint8_t i = 0; i < USBD_CLASS_DRIVER_COUNT; i++)
|
|
|
|
{
|
|
|
|
TU_LOG2("%s init\r\n", _usbd_driver_str[i]);
|
|
|
|
usbd_class_drivers[i].init();
|
|
|
|
}
|
2018-12-05 13:20:25 +07:00
|
|
|
|
|
|
|
// Init device controller driver
|
2019-03-29 16:23:00 +07:00
|
|
|
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);
|
2019-03-30 14:34:38 +07:00
|
|
|
|
2019-09-13 22:16:24 +07:00
|
|
|
memset(_usbd_dev.itf2drv, DRVID_INVALID, sizeof(_usbd_dev.itf2drv)); // invalid mapping
|
|
|
|
memset(_usbd_dev.ep2drv , DRVID_INVALID, 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-12-13 13:49:09 +07:00
|
|
|
/* USB Device Driver task
|
|
|
|
* This top level thread manages all device controller event and delegates events to class-specific drivers.
|
2018-12-13 14:51:37 +07:00
|
|
|
* This should be called periodically within the mainloop or rtos thread.
|
|
|
|
*
|
|
|
|
@code
|
|
|
|
int main(void)
|
|
|
|
{
|
|
|
|
application_init();
|
|
|
|
tusb_init();
|
|
|
|
|
|
|
|
while(1) // the mainloop
|
|
|
|
{
|
|
|
|
application_code();
|
|
|
|
tud_task(); // tinyusb device task
|
|
|
|
}
|
|
|
|
}
|
|
|
|
@endcode
|
2018-12-13 13:49:09 +07:00
|
|
|
*/
|
2018-12-13 14:51:37 +07:00
|
|
|
void tud_task (void)
|
2014-03-31 13:12:51 +07:00
|
|
|
{
|
2019-01-29 19:05:07 +07:00
|
|
|
// Skip if stack is not initialized
|
|
|
|
if ( !tusb_inited() ) return;
|
|
|
|
|
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
|
|
|
|
2019-10-23 21:18:46 +07:00
|
|
|
TU_LOG2("USBD: event %s\r\n", event.event_id < DCD_EVENT_COUNT ? _usbd_event_str[event.event_id] : "CORRUPTED");
|
|
|
|
|
2018-11-14 17:40:29 +07:00
|
|
|
switch ( event.event_id )
|
2018-03-06 16:50:50 +07:00
|
|
|
{
|
2019-03-30 02:26:15 +07:00
|
|
|
case DCD_EVENT_BUS_RESET:
|
|
|
|
usbd_reset(event.rhport);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DCD_EVENT_UNPLUGGED:
|
|
|
|
usbd_reset(event.rhport);
|
|
|
|
|
|
|
|
// invoke callback
|
|
|
|
if (tud_umount_cb) tud_umount_cb();
|
|
|
|
break;
|
|
|
|
|
2018-11-14 17:40:29 +07:00
|
|
|
case DCD_EVENT_SETUP_RECEIVED:
|
2019-10-24 12:00:06 +07:00
|
|
|
TU_LOG2(" ");
|
2019-10-23 21:18:46 +07:00
|
|
|
TU_LOG2_MEM(&event.setup_received, 1, 8);
|
|
|
|
|
2019-04-02 01:30:01 +07:00
|
|
|
// Mark as connected after receiving 1st setup packet.
|
|
|
|
// But it is easier to set it every time instead of wasting time to check then set
|
|
|
|
_usbd_dev.connected = 1;
|
|
|
|
|
2019-03-27 16:26:52 +07:00
|
|
|
// Process control request
|
2018-11-16 21:56:39 +07:00
|
|
|
if ( !process_control_request(event.rhport, &event.setup_received) )
|
|
|
|
{
|
2019-03-27 16:26:52 +07:00
|
|
|
// Failed -> stall both control endpoint IN and OUT
|
|
|
|
dcd_edpt_stall(event.rhport, 0);
|
|
|
|
dcd_edpt_stall(event.rhport, 0 | TUSB_DIR_IN_MASK);
|
2018-11-16 21:56:39 +07:00
|
|
|
}
|
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:
|
2019-10-23 21:18:46 +07:00
|
|
|
{
|
|
|
|
// Invoke the class callback associated with the endpoint address
|
|
|
|
uint8_t const ep_addr = event.xfer_complete.ep_addr;
|
|
|
|
uint8_t const epnum = tu_edpt_number(ep_addr);
|
|
|
|
uint8_t const ep_dir = tu_edpt_dir(ep_addr);
|
|
|
|
|
2019-10-24 12:00:06 +07:00
|
|
|
TU_LOG2(" Endpoint: 0x%02X, Bytes: %ld\r\n", ep_addr, event.xfer_complete.len);
|
2019-10-23 21:18:46 +07:00
|
|
|
|
|
|
|
_usbd_dev.ep_status[epnum][ep_dir].busy = false;
|
|
|
|
|
|
|
|
if ( 0 == epnum )
|
2018-08-28 15:56:43 +07:00
|
|
|
{
|
2019-10-23 21:18:46 +07:00
|
|
|
usbd_control_xfer_cb(event.rhport, ep_addr, event.xfer_complete.result, event.xfer_complete.len);
|
2018-08-28 15:56:43 +07:00
|
|
|
}
|
2019-10-23 21:18:46 +07:00
|
|
|
else
|
|
|
|
{
|
|
|
|
uint8_t const drv_id = _usbd_dev.ep2drv[epnum][ep_dir];
|
|
|
|
TU_ASSERT(drv_id < USBD_CLASS_DRIVER_COUNT,);
|
|
|
|
|
2019-10-24 12:00:06 +07:00
|
|
|
TU_LOG2(" %s xfer callback\r\n", _usbd_driver_str[drv_id]);
|
2019-10-23 21:18:46 +07:00
|
|
|
usbd_class_drivers[drv_id].xfer_cb(event.rhport, ep_addr, event.xfer_complete.result, event.xfer_complete.len);
|
|
|
|
}
|
|
|
|
}
|
2018-11-14 17:40:29 +07:00
|
|
|
break;
|
|
|
|
|
2019-03-30 02:26:15 +07:00
|
|
|
case DCD_EVENT_SUSPEND:
|
2019-04-02 01:30:01 +07:00
|
|
|
if (tud_suspend_cb) tud_suspend_cb(_usbd_dev.remote_wakeup_en);
|
2018-11-14 17:40:29 +07:00
|
|
|
break;
|
|
|
|
|
2019-03-30 02:26:15 +07:00
|
|
|
case DCD_EVENT_RESUME:
|
|
|
|
if (tud_resume_cb) tud_resume_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;
|
|
|
|
|
2019-03-30 02:26:15 +07:00
|
|
|
case USBD_EVENT_FUNC_CALL:
|
2018-11-14 17:40:29 +07:00
|
|
|
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-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);
|
|
|
|
|
2019-09-16 11:13:17 +07:00
|
|
|
TU_ASSERT(p_request->bmRequestType_bit.type < TUSB_REQ_TYPE_INVALID);
|
|
|
|
|
2019-07-12 22:03:40 +07:00
|
|
|
// Vendor request
|
2019-07-16 18:14:47 +07:00
|
|
|
if ( p_request->bmRequestType_bit.type == TUSB_REQ_TYPE_VENDOR )
|
|
|
|
{
|
2019-07-24 09:45:32 +07:00
|
|
|
TU_VERIFY(tud_vendor_control_request_cb);
|
2019-07-16 18:14:47 +07:00
|
|
|
|
2019-07-24 09:45:32 +07:00
|
|
|
if (tud_vendor_control_complete_cb) usbd_control_set_complete_callback(tud_vendor_control_complete_cb);
|
|
|
|
return tud_vendor_control_request_cb(rhport, p_request);
|
2019-07-16 18:14:47 +07:00
|
|
|
}
|
2019-07-12 22:03:40 +07:00
|
|
|
|
2019-10-24 12:00:06 +07:00
|
|
|
#if CFG_TUSB_DEBUG > 1
|
|
|
|
if (TUSB_REQ_TYPE_STANDARD == p_request->bmRequestType_bit.type && p_request->bRequest <= TUSB_REQ_SYNCH_FRAME)
|
|
|
|
{
|
|
|
|
TU_LOG2(" %s\r\n", _tusb_std_request_str[p_request->bRequest]);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2019-03-29 16:55:58 +07:00
|
|
|
switch ( p_request->bmRequestType_bit.recipient )
|
2018-11-16 21:56:39 +07:00
|
|
|
{
|
2019-03-29 16:55:58 +07:00
|
|
|
//------------- Device Requests e.g in enumeration -------------//
|
|
|
|
case TUSB_REQ_RCPT_DEVICE:
|
|
|
|
if ( TUSB_REQ_TYPE_STANDARD != p_request->bmRequestType_bit.type )
|
|
|
|
{
|
|
|
|
// Non standard request is not supported
|
|
|
|
TU_BREAKPOINT();
|
|
|
|
return false;
|
|
|
|
}
|
2018-11-16 21:56:39 +07:00
|
|
|
|
2019-03-29 16:55:58 +07:00
|
|
|
switch ( p_request->bRequest )
|
2018-11-16 21:56:39 +07:00
|
|
|
{
|
2019-03-29 16:55:58 +07:00
|
|
|
case TUSB_REQ_SET_ADDRESS:
|
2019-03-30 14:34:38 +07:00
|
|
|
// Depending on mcu, status phase could be sent either before or after changing device address
|
|
|
|
// Therefore DCD must include zero-length status response
|
2019-03-29 16:55:58 +07:00
|
|
|
dcd_set_address(rhport, (uint8_t) p_request->wValue);
|
2019-03-30 14:34:38 +07:00
|
|
|
return true; // skip status
|
2019-03-29 16:55:58 +07:00
|
|
|
break;
|
|
|
|
|
|
|
|
case TUSB_REQ_GET_CONFIGURATION:
|
2019-03-30 14:34:38 +07:00
|
|
|
{
|
|
|
|
uint8_t cfgnum = _usbd_dev.configured ? 1 : 0;
|
2019-07-16 18:14:47 +07:00
|
|
|
tud_control_xfer(rhport, p_request, &cfgnum, 1);
|
2019-03-30 14:34:38 +07:00
|
|
|
}
|
2019-03-29 16:55:58 +07:00
|
|
|
break;
|
|
|
|
|
|
|
|
case TUSB_REQ_SET_CONFIGURATION:
|
|
|
|
{
|
2019-03-30 13:48:15 +07:00
|
|
|
uint8_t const cfg_num = (uint8_t) p_request->wValue;
|
2018-11-16 21:56:39 +07:00
|
|
|
|
2019-03-30 13:48:15 +07:00
|
|
|
dcd_set_config(rhport, cfg_num);
|
|
|
|
_usbd_dev.configured = cfg_num ? 1 : 0;
|
2018-11-16 21:56:39 +07:00
|
|
|
|
2019-05-12 19:55:15 +07:00
|
|
|
if ( cfg_num ) TU_ASSERT( process_set_config(rhport, cfg_num) );
|
2019-07-16 18:14:47 +07:00
|
|
|
tud_control_status(rhport, p_request);
|
2019-03-29 16:55:58 +07:00
|
|
|
}
|
|
|
|
break;
|
2018-11-16 21:56:39 +07:00
|
|
|
|
2019-03-29 16:55:58 +07:00
|
|
|
case TUSB_REQ_GET_DESCRIPTOR:
|
2019-05-15 12:57:31 +07:00
|
|
|
TU_VERIFY( process_get_descriptor(rhport, p_request) );
|
2019-03-29 16:55:58 +07:00
|
|
|
break;
|
2018-11-16 21:56:39 +07:00
|
|
|
|
2019-03-29 16:55:58 +07:00
|
|
|
case TUSB_REQ_SET_FEATURE:
|
2019-03-30 14:34:38 +07:00
|
|
|
// Only support remote wakeup for device feature
|
|
|
|
TU_VERIFY(TUSB_REQ_FEATURE_REMOTE_WAKEUP == p_request->wValue);
|
|
|
|
|
|
|
|
// Host may enable remote wake up before suspending especially HID device
|
|
|
|
_usbd_dev.remote_wakeup_en = true;
|
2019-07-16 18:14:47 +07:00
|
|
|
tud_control_status(rhport, p_request);
|
2019-03-29 16:55:58 +07:00
|
|
|
break;
|
2019-03-29 16:23:00 +07:00
|
|
|
|
2019-03-29 16:55:58 +07:00
|
|
|
case TUSB_REQ_CLEAR_FEATURE:
|
2019-03-30 14:34:38 +07:00
|
|
|
// Only support remote wakeup for device feature
|
|
|
|
TU_VERIFY(TUSB_REQ_FEATURE_REMOTE_WAKEUP == p_request->wValue);
|
|
|
|
|
|
|
|
// Host may disable remote wake up after resuming
|
|
|
|
_usbd_dev.remote_wakeup_en = false;
|
2019-07-16 18:14:47 +07:00
|
|
|
tud_control_status(rhport, p_request);
|
2019-03-29 16:55:58 +07:00
|
|
|
break;
|
2019-03-29 16:23:00 +07:00
|
|
|
|
2019-03-30 13:48:15 +07:00
|
|
|
case TUSB_REQ_GET_STATUS:
|
2019-03-30 14:34:38 +07:00
|
|
|
{
|
|
|
|
// Device status bit mask
|
|
|
|
// - Bit 0: Self Powered
|
|
|
|
// - Bit 1: Remote Wakeup enabled
|
|
|
|
uint16_t status = (_usbd_dev.self_powered ? 1 : 0) | (_usbd_dev.remote_wakeup_en ? 2 : 0);
|
2019-07-16 18:14:47 +07:00
|
|
|
tud_control_xfer(rhport, p_request, &status, 2);
|
2019-03-30 14:34:38 +07:00
|
|
|
}
|
2019-03-30 13:48:15 +07:00
|
|
|
break;
|
|
|
|
|
2019-03-29 16:55:58 +07:00
|
|
|
// Unknown/Unsupported request
|
|
|
|
default: TU_BREAKPOINT(); return false;
|
|
|
|
}
|
|
|
|
break;
|
2018-11-08 13:45:30 -08:00
|
|
|
|
2018-11-16 21:56:39 +07:00
|
|
|
//------------- Class/Interface Specific Request -------------//
|
2019-03-29 16:55:58 +07:00
|
|
|
case TUSB_REQ_RCPT_INTERFACE:
|
|
|
|
{
|
|
|
|
uint8_t const itf = tu_u16_low(p_request->wIndex);
|
2019-09-13 22:16:24 +07:00
|
|
|
TU_VERIFY(itf < TU_ARRAY_SIZE(_usbd_dev.itf2drv));
|
2018-11-16 21:56:39 +07:00
|
|
|
|
2019-09-13 14:03:59 +07:00
|
|
|
uint8_t const drvid = _usbd_dev.itf2drv[itf];
|
2019-03-29 16:55:58 +07:00
|
|
|
TU_VERIFY(drvid < USBD_CLASS_DRIVER_COUNT);
|
2018-11-16 21:56:39 +07:00
|
|
|
|
2019-09-13 13:03:11 +07:00
|
|
|
if (p_request->bmRequestType_bit.type == TUSB_REQ_TYPE_STANDARD)
|
2019-07-26 20:14:56 +07:00
|
|
|
{
|
2019-09-13 13:03:11 +07:00
|
|
|
switch ( p_request->bRequest )
|
2019-07-26 20:14:56 +07:00
|
|
|
{
|
2019-09-13 13:03:11 +07:00
|
|
|
case TUSB_REQ_GET_INTERFACE:
|
|
|
|
{
|
|
|
|
// TODO not support alternate interface yet
|
|
|
|
uint8_t alternate = 0;
|
|
|
|
tud_control_xfer(rhport, p_request, &alternate, 1);
|
|
|
|
}
|
|
|
|
break;
|
2019-07-26 20:14:56 +07:00
|
|
|
|
2019-09-13 13:03:11 +07:00
|
|
|
case TUSB_REQ_SET_INTERFACE:
|
|
|
|
{
|
|
|
|
uint8_t const alternate = (uint8_t) p_request->wValue;
|
2019-03-29 16:55:58 +07:00
|
|
|
|
2019-09-13 13:03:11 +07:00
|
|
|
// TODO not support alternate interface yet
|
|
|
|
TU_ASSERT(alternate == 0);
|
|
|
|
tud_control_status(rhport, p_request);
|
|
|
|
}
|
|
|
|
break;
|
2019-07-26 20:14:56 +07:00
|
|
|
|
2019-09-13 13:03:11 +07:00
|
|
|
default:
|
|
|
|
// forward to class driver: "STD request to Interface"
|
|
|
|
// GET HID REPORT DESCRIPTOR falls into this case
|
|
|
|
// stall control endpoint if driver return false
|
2019-09-13 22:16:24 +07:00
|
|
|
usbd_control_set_complete_callback(usbd_class_drivers[drvid].control_complete);
|
2019-10-24 12:00:06 +07:00
|
|
|
TU_LOG2(" %s control request\r\n", _usbd_driver_str[drvid]);
|
2019-09-16 11:13:17 +07:00
|
|
|
TU_ASSERT(usbd_class_drivers[drvid].control_request != NULL &&
|
|
|
|
usbd_class_drivers[drvid].control_request(rhport, p_request));
|
2019-09-13 13:03:11 +07:00
|
|
|
break;
|
2019-07-26 20:14:56 +07:00
|
|
|
}
|
2019-09-13 13:03:11 +07:00
|
|
|
}else
|
|
|
|
{
|
|
|
|
// forward to class driver: "non-STD request to Interface"
|
|
|
|
// stall control endpoint if driver return false
|
2019-09-13 22:16:24 +07:00
|
|
|
usbd_control_set_complete_callback(usbd_class_drivers[drvid].control_complete);
|
2019-10-24 12:00:06 +07:00
|
|
|
TU_LOG2(" %s control request\r\n", _usbd_driver_str[drvid]);
|
2019-09-16 11:13:17 +07:00
|
|
|
TU_ASSERT(usbd_class_drivers[drvid].control_request != NULL &&
|
|
|
|
usbd_class_drivers[drvid].control_request(rhport, p_request));
|
2019-07-26 20:14:56 +07:00
|
|
|
}
|
2019-03-29 16:55:58 +07:00
|
|
|
}
|
|
|
|
break;
|
2018-11-16 21:56:39 +07:00
|
|
|
|
|
|
|
//------------- Endpoint Request -------------//
|
2019-03-29 16:55:58 +07:00
|
|
|
case TUSB_REQ_RCPT_ENDPOINT:
|
2019-09-13 22:16:24 +07:00
|
|
|
{
|
|
|
|
uint8_t const ep_addr = tu_u16_low(p_request->wIndex);
|
|
|
|
uint8_t const ep_num = tu_edpt_number(ep_addr);
|
|
|
|
uint8_t const ep_dir = tu_edpt_dir(ep_addr);
|
2018-11-16 21:56:39 +07:00
|
|
|
|
2019-09-13 22:16:24 +07:00
|
|
|
TU_ASSERT(ep_num < TU_ARRAY_SIZE(_usbd_dev.ep2drv) );
|
|
|
|
|
|
|
|
uint8_t const drv_id = _usbd_dev.ep2drv[ep_num][ep_dir];
|
|
|
|
TU_ASSERT(drv_id < USBD_CLASS_DRIVER_COUNT);
|
|
|
|
|
2019-09-16 11:13:17 +07:00
|
|
|
bool ret = false;
|
2019-09-13 22:16:24 +07:00
|
|
|
|
|
|
|
if ( TUSB_REQ_TYPE_STANDARD != p_request->bmRequestType_bit.type )
|
2019-03-29 16:55:58 +07:00
|
|
|
{
|
2019-09-13 22:16:24 +07:00
|
|
|
// complete callback is also capable of stalling/acking the request
|
|
|
|
usbd_control_set_complete_callback(usbd_class_drivers[drv_id].control_complete);
|
|
|
|
}
|
2018-11-16 21:56:39 +07:00
|
|
|
|
2019-09-13 22:16:24 +07:00
|
|
|
// Then handle if it is standard request
|
|
|
|
if ( TUSB_REQ_TYPE_STANDARD == p_request->bmRequestType_bit.type )
|
|
|
|
{
|
|
|
|
// force return true for standard request
|
|
|
|
ret = true;
|
|
|
|
|
|
|
|
switch ( p_request->bRequest )
|
|
|
|
{
|
|
|
|
case TUSB_REQ_GET_STATUS:
|
2019-03-29 16:55:58 +07:00
|
|
|
{
|
2019-09-13 22:16:24 +07:00
|
|
|
uint16_t status = usbd_edpt_stalled(rhport, ep_addr) ? 0x0001 : 0x0000;
|
|
|
|
tud_control_xfer(rhport, p_request, &status, 2);
|
2019-03-29 16:55:58 +07:00
|
|
|
}
|
2019-09-13 22:16:24 +07:00
|
|
|
break;
|
2019-03-29 16:55:58 +07:00
|
|
|
|
2019-09-13 22:16:24 +07:00
|
|
|
case TUSB_REQ_CLEAR_FEATURE:
|
|
|
|
if ( TUSB_REQ_FEATURE_EDPT_HALT == p_request->wValue )
|
|
|
|
{
|
|
|
|
usbd_edpt_clear_stall(rhport, ep_addr);
|
|
|
|
}
|
|
|
|
tud_control_status(rhport, p_request);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case TUSB_REQ_SET_FEATURE:
|
|
|
|
if ( TUSB_REQ_FEATURE_EDPT_HALT == p_request->wValue )
|
|
|
|
{
|
|
|
|
usbd_edpt_stall(rhport, ep_addr);
|
|
|
|
}
|
|
|
|
tud_control_status(rhport, p_request);
|
|
|
|
break;
|
|
|
|
|
|
|
|
// Unknown/Unsupported request
|
|
|
|
default: TU_BREAKPOINT(); return false;
|
|
|
|
}
|
2019-03-29 16:55:58 +07:00
|
|
|
}
|
2019-09-21 12:02:06 -04:00
|
|
|
// Some classes such as TMC needs to clear/re-init its buffer when receiving CLEAR_FEATURE request
|
|
|
|
// We will forward all request targeted endpoint to its class driver
|
|
|
|
// For class-type requests: must (call tud_control_status(); return true) or (return false)
|
2019-09-21 12:17:17 -04:00
|
|
|
// For std-type requests: non-std request codes are already discarded.
|
2019-09-21 12:02:06 -04:00
|
|
|
// must not call tud_control_status(), and return value will have no effect
|
2019-09-21 12:17:17 -04:00
|
|
|
// class driver is invoked last, so that EP already has EP stall cleared (in event of clear feature EP halt)
|
2019-10-24 12:00:06 +07:00
|
|
|
TU_LOG2(" %s control request\r\n", _usbd_driver_str[drv_id]);
|
2019-09-21 19:29:57 -04:00
|
|
|
if ( usbd_class_drivers[drv_id].control_request &&
|
|
|
|
usbd_class_drivers[drv_id].control_request(rhport, p_request))
|
2019-09-21 12:02:06 -04:00
|
|
|
{
|
2019-09-21 19:29:57 -04:00
|
|
|
ret = true;
|
2019-09-21 12:02:06 -04:00
|
|
|
}
|
2019-09-13 22:16:24 +07:00
|
|
|
return ret;
|
|
|
|
}
|
2019-03-29 16:55:58 +07:00
|
|
|
break;
|
|
|
|
|
|
|
|
// Unknown recipient
|
|
|
|
default: TU_BREAKPOINT(); return false;
|
2018-11-16 21:56:39 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
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
|
2019-05-12 19:55:15 +07:00
|
|
|
static bool process_set_config(uint8_t rhport, uint8_t cfg_num)
|
2014-03-12 14:01:38 +07:00
|
|
|
{
|
2019-05-12 19:55:15 +07:00
|
|
|
tusb_desc_configuration_t const * desc_cfg = (tusb_desc_configuration_t const *) tud_descriptor_configuration_cb(cfg_num-1); // index is cfg_num-1
|
2019-03-30 17:38:00 +07:00
|
|
|
TU_ASSERT(desc_cfg != NULL && desc_cfg->bDescriptorType == TUSB_DESC_CONFIGURATION);
|
|
|
|
|
|
|
|
// Parse configuration descriptor
|
|
|
|
_usbd_dev.remote_wakeup_support = (desc_cfg->bmAttributes & TUSB_DESC_CONFIG_ATT_REMOTE_WAKEUP) ? 1 : 0;
|
|
|
|
_usbd_dev.self_powered = (desc_cfg->bmAttributes & TUSB_DESC_CONFIG_ATT_SELF_POWERED) ? 1 : 0;
|
2018-11-16 21:56:39 +07:00
|
|
|
|
2019-03-30 17:38:00 +07:00
|
|
|
// Parse interface descriptor
|
|
|
|
uint8_t const * p_desc = ((uint8_t const*) desc_cfg) + sizeof(tusb_desc_configuration_t);
|
|
|
|
uint8_t const * desc_end = ((uint8_t const*) desc_cfg) + desc_cfg->wTotalLength;
|
2014-03-23 15:39:55 +07:00
|
|
|
|
2019-03-30 17:38:00 +07:00
|
|
|
while( p_desc < desc_end )
|
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-12-12 12:01:15 +07:00
|
|
|
if ( TUSB_DESC_INTERFACE_ASSOCIATION == tu_desc_type(p_desc) )
|
2014-03-12 14:01:38 +07:00
|
|
|
{
|
2018-12-12 12:01:15 +07:00
|
|
|
p_desc = tu_desc_next(p_desc); // ignore Interface Association
|
2014-03-12 14:01:38 +07:00
|
|
|
}else
|
|
|
|
{
|
2018-12-12 12:01:15 +07:00
|
|
|
TU_ASSERT( TUSB_DESC_INTERFACE == tu_desc_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
|
2019-09-13 22:16:24 +07:00
|
|
|
TU_ASSERT( DRVID_INVALID == _usbd_dev.itf2drv[desc_itf->bInterfaceNumber] );
|
2018-11-16 21:56:39 +07:00
|
|
|
_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;
|
2019-10-24 12:00:06 +07:00
|
|
|
TU_LOG2(" %s open\r\n", _usbd_driver_str[drv_id]);
|
2019-07-24 23:07:30 +07:00
|
|
|
TU_ASSERT( usbd_class_drivers[drv_id].open(rhport, desc_itf, &itf_len) );
|
2018-12-10 05:07:22 +07:00
|
|
|
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
|
|
|
|
2019-03-29 16:55:58 +07:00
|
|
|
return true;
|
2014-03-12 14:01:38 +07:00
|
|
|
}
|
|
|
|
|
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 )
|
|
|
|
{
|
2018-12-12 12:01:15 +07:00
|
|
|
if ( TUSB_DESC_ENDPOINT == tu_desc_type(p_desc) )
|
2018-07-13 17:48:26 +07:00
|
|
|
{
|
|
|
|
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
|
|
|
}
|
|
|
|
|
2019-09-21 12:03:36 -04:00
|
|
|
len = (uint16_t)(len + tu_desc_len(p_desc));
|
2018-12-12 12:01:15 +07:00
|
|
|
p_desc = tu_desc_next(p_desc);
|
2018-07-13 17:48:26 +07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-16 21:56:39 +07:00
|
|
|
// return descriptor's buffer and update desc_len
|
2019-05-11 16:31:08 +07:00
|
|
|
static bool process_get_descriptor(uint8_t rhport, tusb_control_request_t const * p_request)
|
2018-11-16 21:56:39 +07:00
|
|
|
{
|
|
|
|
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 );
|
|
|
|
|
|
|
|
switch(desc_type)
|
|
|
|
{
|
|
|
|
case TUSB_DESC_DEVICE:
|
2019-07-16 18:14:47 +07:00
|
|
|
return tud_control_xfer(rhport, p_request, (void*) tud_descriptor_device_cb(), sizeof(tusb_desc_device_t));
|
2018-11-16 21:56:39 +07:00
|
|
|
break;
|
|
|
|
|
2019-07-12 00:12:14 +07:00
|
|
|
case TUSB_DESC_BOS:
|
|
|
|
{
|
|
|
|
// requested by host if USB > 2.0 ( i.e 2.1 or 3.x )
|
|
|
|
if (!tud_descriptor_bos_cb) return false;
|
|
|
|
|
|
|
|
tusb_desc_bos_t const* desc_bos = (tusb_desc_bos_t const*) tud_descriptor_bos_cb();
|
|
|
|
uint16_t total_len;
|
|
|
|
memcpy(&total_len, &desc_bos->wTotalLength, 2); // possibly mis-aligned memory
|
|
|
|
|
2019-07-16 18:14:47 +07:00
|
|
|
return tud_control_xfer(rhport, p_request, (void*) desc_bos, total_len);
|
2019-07-12 00:12:14 +07:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2018-11-16 21:56:39 +07:00
|
|
|
case TUSB_DESC_CONFIGURATION:
|
2019-05-12 14:09:35 +07:00
|
|
|
{
|
2019-05-12 19:55:15 +07:00
|
|
|
tusb_desc_configuration_t const* desc_config = (tusb_desc_configuration_t const*) tud_descriptor_configuration_cb(desc_index);
|
2019-10-31 13:06:57 +07:00
|
|
|
TU_ASSERT(desc_config);
|
|
|
|
|
2019-07-12 00:12:14 +07:00
|
|
|
uint16_t total_len;
|
|
|
|
memcpy(&total_len, &desc_config->wTotalLength, 2); // possibly mis-aligned memory
|
|
|
|
|
2019-07-16 18:14:47 +07:00
|
|
|
return tud_control_xfer(rhport, p_request, (void*) desc_config, total_len);
|
2019-05-12 14:09:35 +07:00
|
|
|
}
|
2018-11-16 21:56:39 +07:00
|
|
|
break;
|
|
|
|
|
|
|
|
case TUSB_DESC_STRING:
|
|
|
|
// String Descriptor always uses the desc set from user
|
2019-05-11 16:31:08 +07:00
|
|
|
if ( desc_index == 0xEE )
|
2018-11-16 21:56:39 +07:00
|
|
|
{
|
2019-07-12 14:53:11 +07:00
|
|
|
// The 0xEE index string is a Microsoft OS Descriptors.
|
|
|
|
// https://docs.microsoft.com/en-us/windows-hardware/drivers/usbcon/microsoft-defined-usb-descriptors
|
2019-05-11 16:31:08 +07:00
|
|
|
return false;
|
|
|
|
}else
|
|
|
|
{
|
2019-05-12 15:38:15 +07:00
|
|
|
uint8_t const* desc_str = (uint8_t const*) tud_descriptor_string_cb(desc_index);
|
|
|
|
TU_ASSERT(desc_str);
|
2019-05-11 16:31:08 +07:00
|
|
|
|
2019-05-12 15:38:15 +07:00
|
|
|
// first byte of descriptor is its size
|
2019-07-16 18:14:47 +07:00
|
|
|
return tud_control_xfer(rhport, p_request, (void*) desc_str, desc_str[0]);
|
2018-11-16 21:56:39 +07:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case TUSB_DESC_DEVICE_QUALIFIER:
|
|
|
|
// TODO If not highspeed capable stall this request otherwise
|
|
|
|
// return the descriptor that could work in highspeed
|
2019-05-11 16:31:08 +07:00
|
|
|
return false;
|
2018-11-16 21:56:39 +07:00
|
|
|
break;
|
|
|
|
|
2019-05-11 16:31:08 +07:00
|
|
|
default: return false;
|
2018-11-16 21:56:39 +07:00
|
|
|
}
|
|
|
|
|
2019-05-11 16:31:08 +07:00
|
|
|
return true;
|
2018-11-16 21:56:39 +07:00
|
|
|
}
|
|
|
|
|
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:
|
2019-03-27 23:58:24 +07:00
|
|
|
osal_queue_send(_usbd_q, event, in_isr);
|
|
|
|
break;
|
|
|
|
|
2018-10-23 15:12:30 +07:00
|
|
|
case DCD_EVENT_UNPLUGGED:
|
2019-03-30 02:26:15 +07:00
|
|
|
_usbd_dev.connected = 0;
|
2019-03-30 13:48:15 +07:00
|
|
|
_usbd_dev.configured = 0;
|
2019-03-30 23:01:23 +07:00
|
|
|
_usbd_dev.suspended = 0;
|
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;
|
|
|
|
|
2019-03-29 01:34:53 +07:00
|
|
|
case DCD_EVENT_SUSPEND:
|
2019-04-02 01:30:01 +07:00
|
|
|
// NOTE: When plugging/unplugging device, the D+/D- state are unstable and can accidentally meet the
|
2019-05-29 16:55:15 +07:00
|
|
|
// SUSPEND condition ( Idle for 3ms ). Some MCUs such as SAMD doesn't distinguish suspend vs disconnect as well.
|
2019-04-02 01:30:01 +07:00
|
|
|
// We will skip handling SUSPEND/RESUME event if not currently connected
|
|
|
|
if ( _usbd_dev.connected )
|
|
|
|
{
|
|
|
|
_usbd_dev.suspended = 1;
|
|
|
|
osal_queue_send(_usbd_q, event, in_isr);
|
|
|
|
}
|
2019-03-30 02:26:15 +07:00
|
|
|
break;
|
|
|
|
|
2018-10-23 15:12:30 +07:00
|
|
|
case DCD_EVENT_RESUME:
|
2019-04-02 01:30:01 +07:00
|
|
|
if ( _usbd_dev.connected )
|
|
|
|
{
|
|
|
|
_usbd_dev.suspended = 0;
|
|
|
|
osal_queue_send(_usbd_q, event, in_isr);
|
|
|
|
}
|
2018-10-23 15:08:31 +07:00
|
|
|
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-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;
|
|
|
|
|
2019-03-30 02:26:15 +07:00
|
|
|
// Not an DCD event, just a convenient way to defer ISR function should we need to
|
|
|
|
case USBD_EVENT_FUNC_CALL:
|
2018-12-05 17:01:19 +07:00
|
|
|
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
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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
|
|
|
|
2019-05-01 19:29:56 +07:00
|
|
|
// Parse consecutive endpoint descriptors (IN & OUT)
|
|
|
|
bool usbd_open_edpt_pair(uint8_t rhport, uint8_t const* p_desc, uint8_t ep_count, uint8_t xfer_type, uint8_t* ep_out, uint8_t* ep_in)
|
2018-06-16 12:30:10 +07:00
|
|
|
{
|
2019-05-01 19:29:56 +07:00
|
|
|
for(int i=0; i<ep_count; i++)
|
2018-06-16 12:30:10 +07:00
|
|
|
{
|
2019-05-01 19:29:56 +07:00
|
|
|
tusb_desc_endpoint_t const * desc_ep = (tusb_desc_endpoint_t const *) p_desc;
|
2018-06-16 12:30:10 +07:00
|
|
|
|
2019-09-21 12:02:52 -04:00
|
|
|
TU_ASSERT(TUSB_DESC_ENDPOINT == desc_ep->bDescriptorType && xfer_type == desc_ep->bmAttributes.xfer);
|
2019-05-01 19:29:56 +07:00
|
|
|
TU_ASSERT(dcd_edpt_open(rhport, desc_ep));
|
2018-06-16 12:30:10 +07:00
|
|
|
|
2019-05-01 19:29:56 +07:00
|
|
|
if ( tu_edpt_dir(desc_ep->bEndpointAddress) == TUSB_DIR_IN )
|
2018-06-16 12:30:10 +07:00
|
|
|
{
|
2019-05-01 19:29:56 +07:00
|
|
|
(*ep_in) = desc_ep->bEndpointAddress;
|
2018-06-16 12:30:10 +07:00
|
|
|
}else
|
|
|
|
{
|
2019-05-01 19:29:56 +07:00
|
|
|
(*ep_out) = desc_ep->bEndpointAddress;
|
2018-06-16 12:30:10 +07:00
|
|
|
}
|
2014-03-12 14:01:38 +07:00
|
|
|
|
2019-05-01 19:29:56 +07:00
|
|
|
p_desc = tu_desc_next(p_desc);
|
2018-06-16 12:30:10 +07:00
|
|
|
}
|
|
|
|
|
2018-12-12 13:00:59 +07:00
|
|
|
return true;
|
2018-06-16 12:30:10 +07:00
|
|
|
}
|
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,
|
2019-03-30 02:26:15 +07:00
|
|
|
.event_id = USBD_EVENT_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
|
|
|
}
|
|
|
|
|
2019-03-27 16:09:49 +07:00
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
// USBD Endpoint API
|
|
|
|
//--------------------------------------------------------------------+
|
2019-05-29 16:55:15 +07:00
|
|
|
|
|
|
|
bool usbd_edpt_xfer(uint8_t rhport, uint8_t ep_addr, uint8_t * buffer, uint16_t total_bytes)
|
|
|
|
{
|
|
|
|
uint8_t const epnum = tu_edpt_number(ep_addr);
|
|
|
|
uint8_t const dir = tu_edpt_dir(ep_addr);
|
|
|
|
|
|
|
|
TU_VERIFY( dcd_edpt_xfer(rhport, ep_addr, buffer, total_bytes) );
|
2019-10-01 22:19:04 +07:00
|
|
|
_usbd_dev.ep_status[epnum][dir].busy = true;
|
2019-05-29 16:55:15 +07:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool usbd_edpt_busy(uint8_t rhport, uint8_t ep_addr)
|
|
|
|
{
|
|
|
|
(void) rhport;
|
|
|
|
|
|
|
|
uint8_t const epnum = tu_edpt_number(ep_addr);
|
|
|
|
uint8_t const dir = tu_edpt_dir(ep_addr);
|
|
|
|
|
2019-10-01 22:19:04 +07:00
|
|
|
return _usbd_dev.ep_status[epnum][dir].busy;
|
2019-05-29 16:55:15 +07:00
|
|
|
}
|
|
|
|
|
2019-03-27 16:09:49 +07:00
|
|
|
void usbd_edpt_stall(uint8_t rhport, uint8_t ep_addr)
|
|
|
|
{
|
|
|
|
uint8_t const epnum = tu_edpt_number(ep_addr);
|
|
|
|
uint8_t const dir = tu_edpt_dir(ep_addr);
|
|
|
|
|
|
|
|
dcd_edpt_stall(rhport, ep_addr);
|
2019-10-01 22:19:04 +07:00
|
|
|
_usbd_dev.ep_status[epnum][dir].stalled = true;
|
|
|
|
_usbd_dev.ep_status[epnum][dir].busy = true;
|
2019-03-27 16:09:49 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
void usbd_edpt_clear_stall(uint8_t rhport, uint8_t ep_addr)
|
|
|
|
{
|
|
|
|
uint8_t const epnum = tu_edpt_number(ep_addr);
|
|
|
|
uint8_t const dir = tu_edpt_dir(ep_addr);
|
|
|
|
|
|
|
|
dcd_edpt_clear_stall(rhport, ep_addr);
|
2019-10-01 22:19:04 +07:00
|
|
|
_usbd_dev.ep_status[epnum][dir].stalled = false;
|
|
|
|
_usbd_dev.ep_status[epnum][dir].busy = false;
|
2019-03-27 16:09:49 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
bool usbd_edpt_stalled(uint8_t rhport, uint8_t ep_addr)
|
|
|
|
{
|
|
|
|
(void) rhport;
|
|
|
|
|
|
|
|
uint8_t const epnum = tu_edpt_number(ep_addr);
|
|
|
|
uint8_t const dir = tu_edpt_dir(ep_addr);
|
|
|
|
|
2019-10-01 22:19:04 +07:00
|
|
|
return _usbd_dev.ep_status[epnum][dir].stalled;
|
2019-03-27 16:09:49 +07:00
|
|
|
}
|
|
|
|
|
2014-03-12 14:01:38 +07:00
|
|
|
#endif
|