2021-03-26 15:30:43 -04:00
|
|
|
/*
|
2020-01-14 23:30:39 -05:00
|
|
|
* The MIT License (MIT)
|
|
|
|
*
|
|
|
|
* Copyright (c) 2019 Ha Thach (tinyusb.org)
|
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
|
|
* in the Software without restriction, including without limitation the rights
|
|
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
|
|
* furnished to do so, subject to the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be included in
|
|
|
|
* all copies or substantial portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
* THE SOFTWARE.
|
|
|
|
*
|
|
|
|
* This file is part of the TinyUSB stack.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "tusb_option.h"
|
|
|
|
|
2022-02-23 21:46:40 +07:00
|
|
|
#if CFG_TUD_ENABLED
|
2020-01-14 23:30:39 -05:00
|
|
|
|
2022-06-24 22:22:52 +07:00
|
|
|
#include "device/dcd.h"
|
2020-01-14 23:30:39 -05:00
|
|
|
#include "tusb.h"
|
2022-03-09 17:17:27 +07:00
|
|
|
#include "common/tusb_private.h"
|
|
|
|
|
2021-01-19 17:10:08 +01:00
|
|
|
#include "device/usbd.h"
|
2020-01-14 23:30:39 -05:00
|
|
|
#include "device/usbd_pvt.h"
|
|
|
|
|
2021-07-21 13:21:58 +07:00
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
// USBD Configuration
|
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
|
2020-01-14 23:30:39 -05:00
|
|
|
#ifndef CFG_TUD_TASK_QUEUE_SZ
|
2021-07-21 13:21:58 +07:00
|
|
|
#define CFG_TUD_TASK_QUEUE_SZ 16
|
2020-01-14 23:30:39 -05:00
|
|
|
#endif
|
2020-09-21 15:23:56 +02:00
|
|
|
|
2022-12-21 12:29:51 +07:00
|
|
|
// Debug level of USBD
|
|
|
|
#define USBD_DBG 2
|
|
|
|
|
2020-01-14 23:30:39 -05:00
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
// Device Data
|
|
|
|
//--------------------------------------------------------------------+
|
2021-05-02 01:46:22 +07:00
|
|
|
|
|
|
|
// Invalid driver ID in itf2drv[] ep2drv[][] mapping
|
|
|
|
enum { DRVID_INVALID = 0xFFu };
|
|
|
|
|
2020-06-01 13:40:18 +07:00
|
|
|
typedef struct
|
|
|
|
{
|
2020-01-14 23:30:39 -05:00
|
|
|
struct TU_ATTR_PACKED
|
|
|
|
{
|
|
|
|
volatile uint8_t connected : 1;
|
2020-03-22 17:19:57 +07:00
|
|
|
volatile uint8_t addressed : 1;
|
2020-01-14 23:30:39 -05:00
|
|
|
volatile uint8_t suspended : 1;
|
|
|
|
|
|
|
|
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
|
|
|
|
};
|
|
|
|
|
2020-07-27 21:03:20 +02:00
|
|
|
volatile uint8_t cfg_num; // current active configuration (0x00 is not configured)
|
2020-06-01 13:40:18 +07:00
|
|
|
uint8_t speed;
|
|
|
|
|
2022-05-13 22:54:47 +07:00
|
|
|
uint8_t itf2drv[CFG_TUD_INTERFACE_MAX]; // map interface number to driver (0xff is invalid)
|
2022-03-09 17:34:12 +07:00
|
|
|
uint8_t ep2drv[CFG_TUD_ENDPPOINT_MAX][2]; // map endpoint to driver ( 0xff is invalid ), can use only 4-bit each
|
2020-01-14 23:30:39 -05:00
|
|
|
|
2022-03-09 17:34:12 +07:00
|
|
|
tu_edpt_state_t ep_status[CFG_TUD_ENDPPOINT_MAX][2];
|
2020-09-09 15:48:11 +07:00
|
|
|
|
2020-01-14 23:30:39 -05:00
|
|
|
}usbd_device_t;
|
|
|
|
|
2023-01-20 15:40:32 -08:00
|
|
|
TU_STATIC usbd_device_t _usbd_dev;
|
2020-01-14 23:30:39 -05:00
|
|
|
|
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
// Class Driver
|
|
|
|
//--------------------------------------------------------------------+
|
2020-04-15 11:06:35 +07:00
|
|
|
#if CFG_TUSB_DEBUG >= 2
|
|
|
|
#define DRIVER_NAME(_name) .name = _name,
|
|
|
|
#else
|
|
|
|
#define DRIVER_NAME(_name)
|
|
|
|
#endif
|
|
|
|
|
2020-04-29 10:49:58 +07:00
|
|
|
// Built-in class drivers
|
2023-01-20 15:40:32 -08:00
|
|
|
TU_STATIC usbd_class_driver_t const _usbd_driver[] =
|
2020-01-14 23:30:39 -05:00
|
|
|
{
|
|
|
|
#if CFG_TUD_CDC
|
|
|
|
{
|
2020-11-20 15:32:16 +07:00
|
|
|
DRIVER_NAME("CDC")
|
|
|
|
.init = cdcd_init,
|
|
|
|
.reset = cdcd_reset,
|
|
|
|
.open = cdcd_open,
|
|
|
|
.control_xfer_cb = cdcd_control_xfer_cb,
|
|
|
|
.xfer_cb = cdcd_xfer_cb,
|
2022-06-10 01:45:48 +07:00
|
|
|
.sof = NULL
|
2020-01-14 23:30:39 -05:00
|
|
|
},
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if CFG_TUD_MSC
|
|
|
|
{
|
2020-11-20 15:32:16 +07:00
|
|
|
DRIVER_NAME("MSC")
|
|
|
|
.init = mscd_init,
|
|
|
|
.reset = mscd_reset,
|
|
|
|
.open = mscd_open,
|
|
|
|
.control_xfer_cb = mscd_control_xfer_cb,
|
|
|
|
.xfer_cb = mscd_xfer_cb,
|
2022-06-10 01:45:48 +07:00
|
|
|
.sof = NULL
|
2020-01-14 23:30:39 -05:00
|
|
|
},
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if CFG_TUD_HID
|
|
|
|
{
|
2020-11-20 15:32:16 +07:00
|
|
|
DRIVER_NAME("HID")
|
|
|
|
.init = hidd_init,
|
|
|
|
.reset = hidd_reset,
|
|
|
|
.open = hidd_open,
|
|
|
|
.control_xfer_cb = hidd_control_xfer_cb,
|
|
|
|
.xfer_cb = hidd_xfer_cb,
|
2022-06-10 01:45:48 +07:00
|
|
|
.sof = NULL
|
2020-01-14 23:30:39 -05:00
|
|
|
},
|
|
|
|
#endif
|
|
|
|
|
2020-11-20 15:32:16 +07:00
|
|
|
#if CFG_TUD_AUDIO
|
|
|
|
{
|
|
|
|
DRIVER_NAME("AUDIO")
|
2020-05-22 12:09:34 +02:00
|
|
|
.init = audiod_init,
|
2020-11-19 21:01:33 +07:00
|
|
|
.reset = audiod_reset,
|
2020-05-22 12:09:34 +02:00
|
|
|
.open = audiod_open,
|
2020-11-20 17:20:05 +07:00
|
|
|
.control_xfer_cb = audiod_control_xfer_cb,
|
2020-05-22 12:09:34 +02:00
|
|
|
.xfer_cb = audiod_xfer_cb,
|
2022-06-10 01:45:48 +07:00
|
|
|
.sof = audiod_sof_isr
|
2020-11-20 15:32:16 +07:00
|
|
|
},
|
|
|
|
#endif
|
2020-05-22 12:09:34 +02:00
|
|
|
|
2021-07-17 18:51:33 +09:00
|
|
|
#if CFG_TUD_VIDEO
|
|
|
|
{
|
|
|
|
DRIVER_NAME("VIDEO")
|
|
|
|
.init = videod_init,
|
|
|
|
.reset = videod_reset,
|
|
|
|
.open = videod_open,
|
|
|
|
.control_xfer_cb = videod_control_xfer_cb,
|
|
|
|
.xfer_cb = videod_xfer_cb,
|
2022-06-10 01:45:48 +07:00
|
|
|
.sof = NULL
|
2021-07-17 18:51:33 +09:00
|
|
|
},
|
|
|
|
#endif
|
|
|
|
|
2020-01-14 23:30:39 -05:00
|
|
|
#if CFG_TUD_MIDI
|
|
|
|
{
|
2020-11-20 15:32:16 +07:00
|
|
|
DRIVER_NAME("MIDI")
|
|
|
|
.init = midid_init,
|
|
|
|
.open = midid_open,
|
|
|
|
.reset = midid_reset,
|
|
|
|
.control_xfer_cb = midid_control_xfer_cb,
|
|
|
|
.xfer_cb = midid_xfer_cb,
|
2022-06-10 01:45:48 +07:00
|
|
|
.sof = NULL
|
2020-01-14 23:30:39 -05:00
|
|
|
},
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if CFG_TUD_VENDOR
|
|
|
|
{
|
2020-11-20 15:32:16 +07:00
|
|
|
DRIVER_NAME("VENDOR")
|
|
|
|
.init = vendord_init,
|
|
|
|
.reset = vendord_reset,
|
|
|
|
.open = vendord_open,
|
2020-11-20 15:50:11 +07:00
|
|
|
.control_xfer_cb = tud_vendor_control_xfer_cb,
|
2020-11-20 15:32:16 +07:00
|
|
|
.xfer_cb = vendord_xfer_cb,
|
2022-06-10 01:45:48 +07:00
|
|
|
.sof = NULL
|
2020-01-14 23:30:39 -05:00
|
|
|
},
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if CFG_TUD_USBTMC
|
|
|
|
{
|
2020-11-20 15:32:16 +07:00
|
|
|
DRIVER_NAME("TMC")
|
|
|
|
.init = usbtmcd_init_cb,
|
|
|
|
.reset = usbtmcd_reset_cb,
|
|
|
|
.open = usbtmcd_open_cb,
|
2020-11-20 15:38:56 +07:00
|
|
|
.control_xfer_cb = usbtmcd_control_xfer_cb,
|
2020-11-20 15:32:16 +07:00
|
|
|
.xfer_cb = usbtmcd_xfer_cb,
|
2022-06-10 01:45:48 +07:00
|
|
|
.sof = NULL
|
2020-01-14 23:30:39 -05:00
|
|
|
},
|
|
|
|
#endif
|
|
|
|
|
2021-02-11 12:05:22 +07:00
|
|
|
#if CFG_TUD_DFU_RUNTIME
|
2020-01-14 23:30:39 -05:00
|
|
|
{
|
2021-02-11 12:05:22 +07:00
|
|
|
DRIVER_NAME("DFU-RUNTIME")
|
2020-11-20 15:32:16 +07:00
|
|
|
.init = dfu_rtd_init,
|
|
|
|
.reset = dfu_rtd_reset,
|
|
|
|
.open = dfu_rtd_open,
|
2020-11-20 15:42:32 +07:00
|
|
|
.control_xfer_cb = dfu_rtd_control_xfer_cb,
|
2021-03-26 15:30:43 -04:00
|
|
|
.xfer_cb = NULL,
|
2022-06-10 01:45:48 +07:00
|
|
|
.sof = NULL
|
2020-01-14 23:30:39 -05:00
|
|
|
},
|
|
|
|
#endif
|
2020-03-02 21:15:01 -06:00
|
|
|
|
2021-07-15 20:52:58 +07:00
|
|
|
#if CFG_TUD_DFU
|
2021-04-05 16:32:58 -04:00
|
|
|
{
|
2021-07-15 20:47:50 +07:00
|
|
|
DRIVER_NAME("DFU")
|
2021-04-07 17:05:04 -04:00
|
|
|
.init = dfu_moded_init,
|
|
|
|
.reset = dfu_moded_reset,
|
|
|
|
.open = dfu_moded_open,
|
|
|
|
.control_xfer_cb = dfu_moded_control_xfer_cb,
|
2021-04-05 17:06:27 -04:00
|
|
|
.xfer_cb = NULL,
|
2022-06-10 01:45:48 +07:00
|
|
|
.sof = NULL
|
2020-01-14 23:30:39 -05:00
|
|
|
},
|
|
|
|
#endif
|
2020-03-02 21:15:01 -06:00
|
|
|
|
2021-10-05 12:32:07 +07:00
|
|
|
#if CFG_TUD_ECM_RNDIS || CFG_TUD_NCM
|
2020-03-02 21:15:01 -06:00
|
|
|
{
|
2020-11-20 15:32:16 +07:00
|
|
|
DRIVER_NAME("NET")
|
|
|
|
.init = netd_init,
|
|
|
|
.reset = netd_reset,
|
|
|
|
.open = netd_open,
|
2020-11-20 16:59:33 +07:00
|
|
|
.control_xfer_cb = netd_control_xfer_cb,
|
2020-11-20 15:32:16 +07:00
|
|
|
.xfer_cb = netd_xfer_cb,
|
2022-06-10 01:45:48 +07:00
|
|
|
.sof = NULL,
|
2020-03-02 21:15:01 -06:00
|
|
|
},
|
|
|
|
#endif
|
2020-05-13 17:04:10 +02:00
|
|
|
|
|
|
|
#if CFG_TUD_BTH
|
|
|
|
{
|
2020-11-20 15:32:16 +07:00
|
|
|
DRIVER_NAME("BTH")
|
|
|
|
.init = btd_init,
|
|
|
|
.reset = btd_reset,
|
|
|
|
.open = btd_open,
|
2020-11-20 16:13:58 +07:00
|
|
|
.control_xfer_cb = btd_control_xfer_cb,
|
2020-11-20 15:32:16 +07:00
|
|
|
.xfer_cb = btd_xfer_cb,
|
2022-06-10 01:45:48 +07:00
|
|
|
.sof = NULL
|
2020-05-13 17:04:10 +02:00
|
|
|
},
|
|
|
|
#endif
|
2020-01-14 23:30:39 -05:00
|
|
|
};
|
|
|
|
|
2020-08-07 14:47:32 +07:00
|
|
|
enum { BUILTIN_DRIVER_COUNT = TU_ARRAY_SIZE(_usbd_driver) };
|
2020-01-14 23:30:39 -05:00
|
|
|
|
2020-04-29 10:49:58 +07:00
|
|
|
// Additional class drivers implemented by application
|
2023-01-20 15:40:32 -08:00
|
|
|
TU_STATIC usbd_class_driver_t const * _app_driver = NULL;
|
|
|
|
TU_STATIC uint8_t _app_driver_count = 0;
|
2020-04-29 10:49:58 +07:00
|
|
|
|
2020-08-07 14:47:32 +07:00
|
|
|
// virtually joins built-in and application drivers together.
|
|
|
|
// Application is positioned first to allow overwriting built-in ones.
|
2020-04-29 10:49:58 +07:00
|
|
|
static inline usbd_class_driver_t const * get_driver(uint8_t drvid)
|
|
|
|
{
|
2020-08-07 14:47:32 +07:00
|
|
|
// Application drivers
|
2020-04-29 10:49:58 +07:00
|
|
|
if ( usbd_app_driver_get_cb )
|
|
|
|
{
|
|
|
|
if ( drvid < _app_driver_count ) return &_app_driver[drvid];
|
2020-08-07 14:47:32 +07:00
|
|
|
drvid -= _app_driver_count;
|
2020-04-29 10:49:58 +07:00
|
|
|
}
|
|
|
|
|
2020-08-07 14:47:32 +07:00
|
|
|
// Built-in drivers
|
|
|
|
if (drvid < BUILTIN_DRIVER_COUNT) return &_usbd_driver[drvid];
|
|
|
|
|
2020-04-29 10:49:58 +07:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2020-08-07 14:47:32 +07:00
|
|
|
#define TOTAL_DRIVER_COUNT (_app_driver_count + BUILTIN_DRIVER_COUNT)
|
2020-04-29 10:49:58 +07:00
|
|
|
|
2020-01-14 23:30:39 -05:00
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
// DCD Event
|
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
|
2022-02-23 18:42:32 +07:00
|
|
|
enum { RHPORT_INVALID = 0xFFu };
|
2023-01-20 15:40:32 -08:00
|
|
|
TU_STATIC uint8_t _usbd_rhport = RHPORT_INVALID;
|
2021-05-02 01:46:22 +07:00
|
|
|
|
2020-01-14 23:30:39 -05:00
|
|
|
// Event queue
|
2022-03-09 12:26:56 +07:00
|
|
|
// usbd_int_set() is used as mutex in OS NONE config
|
2022-02-23 18:42:32 +07:00
|
|
|
OSAL_QUEUE_DEF(usbd_int_set, _usbd_qdef, CFG_TUD_TASK_QUEUE_SZ, dcd_event_t);
|
2023-01-20 15:40:32 -08:00
|
|
|
TU_STATIC osal_queue_t _usbd_q;
|
2020-01-14 23:30:39 -05:00
|
|
|
|
2022-12-15 18:03:01 +07:00
|
|
|
// Mutex for claiming endpoint
|
|
|
|
#if OSAL_MUTEX_REQUIRED
|
2023-01-20 15:40:32 -08:00
|
|
|
TU_STATIC osal_mutex_def_t _ubsd_mutexdef;
|
|
|
|
TU_STATIC osal_mutex_t _usbd_mutex;
|
2022-12-15 18:03:01 +07:00
|
|
|
#else
|
|
|
|
#define _usbd_mutex NULL
|
2020-09-09 15:48:11 +07:00
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2020-01-14 23:30:39 -05:00
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
// Prototypes
|
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
static bool process_control_request(uint8_t rhport, tusb_control_request_t const * p_request);
|
|
|
|
static bool process_set_config(uint8_t rhport, uint8_t cfg_num);
|
|
|
|
static bool process_get_descriptor(uint8_t rhport, tusb_control_request_t const * p_request);
|
|
|
|
|
2020-08-07 14:47:32 +07:00
|
|
|
// from usbd_control.c
|
2020-01-14 23:30:39 -05:00
|
|
|
void usbd_control_reset(void);
|
|
|
|
void usbd_control_set_request(tusb_control_request_t const *request);
|
2020-11-19 21:01:33 +07:00
|
|
|
void usbd_control_set_complete_callback( usbd_control_xfer_cb_t fp );
|
2020-01-14 23:30:39 -05:00
|
|
|
bool usbd_control_xfer_cb (uint8_t rhport, uint8_t ep_addr, xfer_result_t event, uint32_t xferred_bytes);
|
|
|
|
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------+
|
2020-08-07 14:47:32 +07:00
|
|
|
// Debug
|
2020-01-14 23:30:39 -05:00
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
#if CFG_TUSB_DEBUG >= 2
|
2023-01-20 15:40:32 -08:00
|
|
|
TU_STATIC char const* const _usbd_event_str[DCD_EVENT_COUNT] =
|
2020-01-14 23:30:39 -05:00
|
|
|
{
|
2020-04-26 22:02:41 +07:00
|
|
|
"Invalid" ,
|
|
|
|
"Bus Reset" ,
|
|
|
|
"Unplugged" ,
|
2020-01-14 23:30:39 -05:00
|
|
|
"SOF" ,
|
2020-04-26 22:02:41 +07:00
|
|
|
"Suspend" ,
|
|
|
|
"Resume" ,
|
|
|
|
"Setup Received" ,
|
|
|
|
"Xfer Complete" ,
|
|
|
|
"Func Call"
|
2020-01-14 23:30:39 -05:00
|
|
|
};
|
|
|
|
|
2020-04-26 22:02:41 +07:00
|
|
|
// for usbd_control to print the name of control complete driver
|
2020-11-19 21:01:33 +07:00
|
|
|
void usbd_driver_print_control_complete_name(usbd_control_xfer_cb_t callback)
|
2020-04-26 22:02:41 +07:00
|
|
|
{
|
2020-04-29 10:49:58 +07:00
|
|
|
for (uint8_t i = 0; i < TOTAL_DRIVER_COUNT; i++)
|
2020-04-26 22:02:41 +07:00
|
|
|
{
|
2020-04-29 10:49:58 +07:00
|
|
|
usbd_class_driver_t const * driver = get_driver(i);
|
2022-06-30 15:45:04 +07:00
|
|
|
if ( driver && driver->control_xfer_cb == callback )
|
2020-04-26 22:02:41 +07:00
|
|
|
{
|
2022-06-16 12:01:19 +07:00
|
|
|
TU_LOG(USBD_DBG, " %s control complete\r\n", driver->name);
|
2020-04-26 22:02:41 +07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-14 23:30:39 -05:00
|
|
|
#endif
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
// Application API
|
|
|
|
//--------------------------------------------------------------------+
|
2020-07-16 00:44:09 +07:00
|
|
|
tusb_speed_t tud_speed_get(void)
|
|
|
|
{
|
|
|
|
return (tusb_speed_t) _usbd_dev.speed;
|
|
|
|
}
|
|
|
|
|
2020-11-24 21:49:12 +07:00
|
|
|
bool tud_connected(void)
|
|
|
|
{
|
|
|
|
return _usbd_dev.connected;
|
|
|
|
}
|
|
|
|
|
2020-01-14 23:30:39 -05:00
|
|
|
bool tud_mounted(void)
|
|
|
|
{
|
2020-11-24 21:49:12 +07:00
|
|
|
return _usbd_dev.cfg_num ? true : false;
|
2020-01-14 23:30:39 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
bool tud_suspended(void)
|
|
|
|
{
|
|
|
|
return _usbd_dev.suspended;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool tud_remote_wakeup(void)
|
|
|
|
{
|
|
|
|
// 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 );
|
2022-02-23 18:42:32 +07:00
|
|
|
dcd_remote_wakeup(_usbd_rhport);
|
2020-01-14 23:30:39 -05:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-08-07 14:47:32 +07:00
|
|
|
bool tud_disconnect(void)
|
|
|
|
{
|
|
|
|
TU_VERIFY(dcd_disconnect);
|
2022-02-23 18:42:32 +07:00
|
|
|
dcd_disconnect(_usbd_rhport);
|
2020-08-07 14:47:32 +07:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool tud_connect(void)
|
|
|
|
{
|
|
|
|
TU_VERIFY(dcd_connect);
|
2022-02-23 18:42:32 +07:00
|
|
|
dcd_connect(_usbd_rhport);
|
2020-08-07 14:47:32 +07:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-01-14 23:30:39 -05:00
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
// USBD Task
|
|
|
|
//--------------------------------------------------------------------+
|
2021-05-02 01:46:22 +07:00
|
|
|
bool tud_inited(void)
|
|
|
|
{
|
2022-02-23 18:42:32 +07:00
|
|
|
return _usbd_rhport != RHPORT_INVALID;
|
2021-05-02 01:46:22 +07:00
|
|
|
}
|
|
|
|
|
2021-05-11 17:32:52 +07:00
|
|
|
bool tud_init (uint8_t rhport)
|
2020-01-14 23:30:39 -05:00
|
|
|
{
|
2021-05-02 01:46:22 +07:00
|
|
|
// skip if already initialized
|
2022-02-23 18:42:32 +07:00
|
|
|
if ( tud_inited() ) return true;
|
2021-05-02 01:46:22 +07:00
|
|
|
|
2022-06-16 12:01:19 +07:00
|
|
|
TU_LOG(USBD_DBG, "USBD init on controller %u\r\n", rhport);
|
|
|
|
TU_LOG_INT(USBD_DBG, sizeof(usbd_device_t));
|
2023-01-06 11:51:17 +07:00
|
|
|
TU_LOG_INT(USBD_DBG, sizeof(tu_fifo_t));
|
|
|
|
TU_LOG_INT(USBD_DBG, sizeof(tu_edpt_stream_t));
|
2020-01-14 23:30:39 -05:00
|
|
|
|
|
|
|
tu_varclr(&_usbd_dev);
|
|
|
|
|
2022-12-15 18:03:01 +07:00
|
|
|
#if OSAL_MUTEX_REQUIRED
|
2020-09-09 15:48:11 +07:00
|
|
|
// Init device mutex
|
|
|
|
_usbd_mutex = osal_mutex_create(&_ubsd_mutexdef);
|
|
|
|
TU_ASSERT(_usbd_mutex);
|
|
|
|
#endif
|
|
|
|
|
2020-01-14 23:30:39 -05:00
|
|
|
// Init device queue & task
|
|
|
|
_usbd_q = osal_queue_create(&_usbd_qdef);
|
2020-09-09 15:48:11 +07:00
|
|
|
TU_ASSERT(_usbd_q);
|
2020-01-14 23:30:39 -05:00
|
|
|
|
2020-04-29 10:49:58 +07:00
|
|
|
// Get application driver if available
|
|
|
|
if ( usbd_app_driver_get_cb )
|
|
|
|
{
|
|
|
|
_app_driver = usbd_app_driver_get_cb(&_app_driver_count);
|
|
|
|
}
|
|
|
|
|
2020-01-14 23:30:39 -05:00
|
|
|
// Init class drivers
|
2020-04-29 10:49:58 +07:00
|
|
|
for (uint8_t i = 0; i < TOTAL_DRIVER_COUNT; i++)
|
2020-01-14 23:30:39 -05:00
|
|
|
{
|
2020-04-29 10:49:58 +07:00
|
|
|
usbd_class_driver_t const * driver = get_driver(i);
|
2022-06-24 19:45:49 +07:00
|
|
|
TU_ASSERT(driver);
|
2022-06-16 12:01:19 +07:00
|
|
|
TU_LOG(USBD_DBG, "%s init\r\n", driver->name);
|
2020-04-29 10:49:58 +07:00
|
|
|
driver->init();
|
2020-01-14 23:30:39 -05:00
|
|
|
}
|
|
|
|
|
2022-03-07 23:03:37 +07:00
|
|
|
_usbd_rhport = rhport;
|
|
|
|
|
2020-01-14 23:30:39 -05:00
|
|
|
// Init device controller driver
|
2021-05-11 17:32:52 +07:00
|
|
|
dcd_init(rhport);
|
|
|
|
dcd_int_enable(rhport);
|
2020-01-14 23:30:39 -05:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-08-26 17:02:24 +07:00
|
|
|
static void configuration_reset(uint8_t rhport)
|
2020-01-14 23:30:39 -05:00
|
|
|
{
|
2021-08-26 17:02:24 +07:00
|
|
|
for ( uint8_t i = 0; i < TOTAL_DRIVER_COUNT; i++ )
|
|
|
|
{
|
2022-06-24 19:45:49 +07:00
|
|
|
usbd_class_driver_t const * driver = get_driver(i);
|
|
|
|
TU_ASSERT(driver, );
|
|
|
|
driver->reset(rhport);
|
2021-08-26 17:02:24 +07:00
|
|
|
}
|
2020-01-14 23:30:39 -05:00
|
|
|
|
2021-08-26 17:02:24 +07:00
|
|
|
tu_varclr(&_usbd_dev);
|
2020-01-14 23:30:39 -05: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
|
2021-08-26 17:02:24 +07:00
|
|
|
}
|
2020-01-14 23:30:39 -05:00
|
|
|
|
2021-08-26 17:02:24 +07:00
|
|
|
static void usbd_reset(uint8_t rhport)
|
|
|
|
{
|
|
|
|
configuration_reset(rhport);
|
2020-01-14 23:30:39 -05:00
|
|
|
usbd_control_reset();
|
|
|
|
}
|
|
|
|
|
2020-05-20 14:31:45 +07:00
|
|
|
bool tud_task_event_ready(void)
|
|
|
|
{
|
|
|
|
// Skip if stack is not initialized
|
|
|
|
if ( !tusb_inited() ) return false;
|
|
|
|
|
|
|
|
return !osal_queue_empty(_usbd_q);
|
|
|
|
}
|
|
|
|
|
2020-01-14 23:30:39 -05:00
|
|
|
/* USB Device Driver task
|
|
|
|
* This top level thread manages all device controller event and delegates events to class-specific drivers.
|
|
|
|
* 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
|
|
|
|
*/
|
2022-04-20 18:29:41 +07:00
|
|
|
void tud_task_ext(uint32_t timeout_ms, bool in_isr)
|
2020-01-14 23:30:39 -05:00
|
|
|
{
|
2022-04-20 18:29:41 +07:00
|
|
|
(void) in_isr; // not implemented yet
|
|
|
|
|
2020-01-14 23:30:39 -05:00
|
|
|
// Skip if stack is not initialized
|
|
|
|
if ( !tusb_inited() ) return;
|
|
|
|
|
|
|
|
// Loop until there is no more events in the queue
|
|
|
|
while (1)
|
|
|
|
{
|
|
|
|
dcd_event_t event;
|
2022-04-20 18:29:41 +07:00
|
|
|
if ( !osal_queue_receive(_usbd_q, &event, timeout_ms) ) return;
|
2020-01-14 23:30:39 -05:00
|
|
|
|
2020-06-14 18:29:02 +07:00
|
|
|
#if CFG_TUSB_DEBUG >= 2
|
2022-06-16 12:01:19 +07:00
|
|
|
if (event.event_id == DCD_EVENT_SETUP_RECEIVED) TU_LOG(USBD_DBG, "\r\n"); // extra line for setup
|
|
|
|
TU_LOG(USBD_DBG, "USBD %s ", event.event_id < DCD_EVENT_COUNT ? _usbd_event_str[event.event_id] : "CORRUPTED");
|
2020-06-14 18:29:02 +07:00
|
|
|
#endif
|
2020-01-14 23:30:39 -05:00
|
|
|
|
|
|
|
switch ( event.event_id )
|
|
|
|
{
|
|
|
|
case DCD_EVENT_BUS_RESET:
|
2022-06-16 12:01:19 +07:00
|
|
|
TU_LOG(USBD_DBG, ": %s Speed\r\n", tu_str_speed[event.bus_reset.speed]);
|
2020-01-14 23:30:39 -05:00
|
|
|
usbd_reset(event.rhport);
|
2020-06-01 13:40:18 +07:00
|
|
|
_usbd_dev.speed = event.bus_reset.speed;
|
2020-01-14 23:30:39 -05:00
|
|
|
break;
|
|
|
|
|
|
|
|
case DCD_EVENT_UNPLUGGED:
|
2022-06-16 12:01:19 +07:00
|
|
|
TU_LOG(USBD_DBG, "\r\n");
|
2020-01-14 23:30:39 -05:00
|
|
|
usbd_reset(event.rhport);
|
|
|
|
|
|
|
|
// invoke callback
|
|
|
|
if (tud_umount_cb) tud_umount_cb();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DCD_EVENT_SETUP_RECEIVED:
|
2022-12-21 12:29:51 +07:00
|
|
|
TU_LOG_PTR(USBD_DBG, &event.setup_received);
|
2022-06-16 12:01:19 +07:00
|
|
|
TU_LOG(USBD_DBG, "\r\n");
|
2020-01-14 23:30:39 -05: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;
|
|
|
|
|
2021-02-11 11:22:02 +07:00
|
|
|
// mark both in & out control as free
|
|
|
|
_usbd_dev.ep_status[0][TUSB_DIR_OUT].busy = false;
|
|
|
|
_usbd_dev.ep_status[0][TUSB_DIR_OUT].claimed = 0;
|
|
|
|
_usbd_dev.ep_status[0][TUSB_DIR_IN ].busy = false;
|
|
|
|
_usbd_dev.ep_status[0][TUSB_DIR_IN ].claimed = 0;
|
|
|
|
|
2020-01-14 23:30:39 -05:00
|
|
|
// Process control request
|
|
|
|
if ( !process_control_request(event.rhport, &event.setup_received) )
|
|
|
|
{
|
2022-06-16 12:01:19 +07:00
|
|
|
TU_LOG(USBD_DBG, " Stall EP0\r\n");
|
2020-01-14 23:30:39 -05: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);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DCD_EVENT_XFER_COMPLETE:
|
|
|
|
{
|
|
|
|
// 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);
|
|
|
|
|
2022-06-16 12:01:19 +07:00
|
|
|
TU_LOG(USBD_DBG, "on EP %02X with %u bytes\r\n", ep_addr, (unsigned int) event.xfer_complete.len);
|
2020-01-14 23:30:39 -05:00
|
|
|
|
|
|
|
_usbd_dev.ep_status[epnum][ep_dir].busy = false;
|
2020-09-09 15:48:11 +07:00
|
|
|
_usbd_dev.ep_status[epnum][ep_dir].claimed = 0;
|
2020-01-14 23:30:39 -05:00
|
|
|
|
|
|
|
if ( 0 == epnum )
|
|
|
|
{
|
2020-06-16 14:13:30 +02:00
|
|
|
usbd_control_xfer_cb(event.rhport, ep_addr, (xfer_result_t)event.xfer_complete.result, event.xfer_complete.len);
|
2020-01-14 23:30:39 -05:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-04-29 10:49:58 +07:00
|
|
|
usbd_class_driver_t const * driver = get_driver( _usbd_dev.ep2drv[epnum][ep_dir] );
|
|
|
|
TU_ASSERT(driver, );
|
2020-01-14 23:30:39 -05:00
|
|
|
|
2022-06-16 12:01:19 +07:00
|
|
|
TU_LOG(USBD_DBG, " %s xfer callback\r\n", driver->name);
|
2020-12-18 11:48:48 +01:00
|
|
|
driver->xfer_cb(event.rhport, ep_addr, (xfer_result_t)event.xfer_complete.result, event.xfer_complete.len);
|
2020-01-14 23:30:39 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DCD_EVENT_SUSPEND:
|
2021-08-17 17:00:53 +07:00
|
|
|
// NOTE: When plugging/unplugging device, the D+/D- state are unstable and
|
|
|
|
// can accidentally meet the SUSPEND condition ( Bus Idle for 3ms ), which result in a series of event
|
|
|
|
// e.g suspend -> resume -> unplug/plug. Skip suspend/resume if not connected
|
|
|
|
if ( _usbd_dev.connected )
|
|
|
|
{
|
2022-06-16 12:01:19 +07:00
|
|
|
TU_LOG(USBD_DBG, ": Remote Wakeup = %u\r\n", _usbd_dev.remote_wakeup_en);
|
2021-08-17 17:00:53 +07:00
|
|
|
if (tud_suspend_cb) tud_suspend_cb(_usbd_dev.remote_wakeup_en);
|
|
|
|
}else
|
|
|
|
{
|
2022-06-16 12:01:19 +07:00
|
|
|
TU_LOG(USBD_DBG, " Skipped\r\n");
|
2021-08-17 17:00:53 +07:00
|
|
|
}
|
2020-01-14 23:30:39 -05:00
|
|
|
break;
|
|
|
|
|
|
|
|
case DCD_EVENT_RESUME:
|
2021-08-17 17:00:53 +07:00
|
|
|
if ( _usbd_dev.connected )
|
|
|
|
{
|
2022-06-16 12:01:19 +07:00
|
|
|
TU_LOG(USBD_DBG, "\r\n");
|
2021-08-17 17:00:53 +07:00
|
|
|
if (tud_resume_cb) tud_resume_cb();
|
|
|
|
}else
|
|
|
|
{
|
2022-06-16 12:01:19 +07:00
|
|
|
TU_LOG(USBD_DBG, " Skipped\r\n");
|
2021-08-17 17:00:53 +07:00
|
|
|
}
|
2020-01-14 23:30:39 -05:00
|
|
|
break;
|
|
|
|
|
|
|
|
case USBD_EVENT_FUNC_CALL:
|
2022-06-16 12:01:19 +07:00
|
|
|
TU_LOG(USBD_DBG, "\r\n");
|
2020-01-14 23:30:39 -05:00
|
|
|
if ( event.func_call.func ) event.func_call.func(event.func_call.param);
|
|
|
|
break;
|
|
|
|
|
2022-03-29 16:53:45 +07:00
|
|
|
case DCD_EVENT_SOF:
|
2020-01-14 23:30:39 -05:00
|
|
|
default:
|
|
|
|
TU_BREAKPOINT();
|
|
|
|
break;
|
|
|
|
}
|
2022-04-20 18:29:41 +07:00
|
|
|
|
|
|
|
#if CFG_TUSB_OS != OPT_OS_NONE && CFG_TUSB_OS != OPT_OS_PICO
|
|
|
|
// return if there is no more events, for application to run other background
|
|
|
|
if (osal_queue_empty(_usbd_q)) return;
|
|
|
|
#endif
|
2020-01-14 23:30:39 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
// Control Request Parser & Handling
|
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
|
|
|
|
// Helper to invoke class driver control request handler
|
2020-04-29 10:49:58 +07:00
|
|
|
static bool invoke_class_control(uint8_t rhport, usbd_class_driver_t const * driver, tusb_control_request_t const * request)
|
2020-01-14 23:30:39 -05:00
|
|
|
{
|
2020-11-19 21:01:33 +07:00
|
|
|
usbd_control_set_complete_callback(driver->control_xfer_cb);
|
2022-06-16 12:01:19 +07:00
|
|
|
TU_LOG(USBD_DBG, " %s control request\r\n", driver->name);
|
2020-11-19 21:01:33 +07:00
|
|
|
return driver->control_xfer_cb(rhport, CONTROL_STAGE_SETUP, request);
|
2020-01-14 23:30:39 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// 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);
|
|
|
|
|
2020-04-21 10:06:17 -04:00
|
|
|
TU_ASSERT(p_request->bmRequestType_bit.type < TUSB_REQ_TYPE_INVALID);
|
2020-01-14 23:30:39 -05:00
|
|
|
|
|
|
|
// Vendor request
|
|
|
|
if ( p_request->bmRequestType_bit.type == TUSB_REQ_TYPE_VENDOR )
|
|
|
|
{
|
2020-11-19 21:01:33 +07:00
|
|
|
TU_VERIFY(tud_vendor_control_xfer_cb);
|
2020-01-14 23:30:39 -05:00
|
|
|
|
2020-11-19 21:01:33 +07:00
|
|
|
usbd_control_set_complete_callback(tud_vendor_control_xfer_cb);
|
|
|
|
return tud_vendor_control_xfer_cb(rhport, CONTROL_STAGE_SETUP, p_request);
|
2020-01-14 23:30:39 -05:00
|
|
|
}
|
|
|
|
|
2020-04-26 14:51:44 +07:00
|
|
|
#if CFG_TUSB_DEBUG >= 2
|
2020-01-14 23:30:39 -05:00
|
|
|
if (TUSB_REQ_TYPE_STANDARD == p_request->bmRequestType_bit.type && p_request->bRequest <= TUSB_REQ_SYNCH_FRAME)
|
|
|
|
{
|
2022-06-16 12:01:19 +07:00
|
|
|
TU_LOG(USBD_DBG, " %s", tu_str_std_request[p_request->bRequest]);
|
|
|
|
if (TUSB_REQ_GET_DESCRIPTOR != p_request->bRequest) TU_LOG(USBD_DBG, "\r\n");
|
2020-01-14 23:30:39 -05:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
switch ( p_request->bmRequestType_bit.recipient )
|
|
|
|
{
|
|
|
|
//------------- Device Requests e.g in enumeration -------------//
|
|
|
|
case TUSB_REQ_RCPT_DEVICE:
|
2020-05-13 16:45:01 +02:00
|
|
|
if ( TUSB_REQ_TYPE_CLASS == p_request->bmRequestType_bit.type )
|
|
|
|
{
|
2020-08-07 12:13:13 +07:00
|
|
|
uint8_t const itf = tu_u16_low(p_request->wIndex);
|
|
|
|
TU_VERIFY(itf < TU_ARRAY_SIZE(_usbd_dev.itf2drv));
|
2020-05-13 16:45:01 +02:00
|
|
|
|
2020-08-07 12:13:13 +07:00
|
|
|
usbd_class_driver_t const * driver = get_driver(_usbd_dev.itf2drv[itf]);
|
|
|
|
TU_VERIFY(driver);
|
2020-05-13 16:45:01 +02:00
|
|
|
|
2020-08-07 12:13:13 +07:00
|
|
|
// forward to class driver: "non-STD request to Interface"
|
|
|
|
return invoke_class_control(rhport, driver, p_request);
|
2020-05-13 16:45:01 +02:00
|
|
|
}
|
2021-02-11 11:22:02 +07:00
|
|
|
|
2020-01-14 23:30:39 -05:00
|
|
|
if ( TUSB_REQ_TYPE_STANDARD != p_request->bmRequestType_bit.type )
|
|
|
|
{
|
|
|
|
// Non standard request is not supported
|
|
|
|
TU_BREAKPOINT();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch ( p_request->bRequest )
|
|
|
|
{
|
|
|
|
case TUSB_REQ_SET_ADDRESS:
|
|
|
|
// Depending on mcu, status phase could be sent either before or after changing device address,
|
|
|
|
// or even require stack to not response with status at all
|
|
|
|
// Therefore DCD must take full responsibility to response and include zlp status packet if needed.
|
|
|
|
usbd_control_set_request(p_request); // set request since DCD has no access to tud_control_status() API
|
|
|
|
dcd_set_address(rhport, (uint8_t) p_request->wValue);
|
|
|
|
// skip tud_control_status()
|
2020-03-22 17:19:57 +07:00
|
|
|
_usbd_dev.addressed = 1;
|
2020-01-14 23:30:39 -05:00
|
|
|
break;
|
|
|
|
|
|
|
|
case TUSB_REQ_GET_CONFIGURATION:
|
|
|
|
{
|
2020-07-27 21:03:20 +02:00
|
|
|
uint8_t cfg_num = _usbd_dev.cfg_num;
|
|
|
|
tud_control_xfer(rhport, p_request, &cfg_num, 1);
|
2020-01-14 23:30:39 -05:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case TUSB_REQ_SET_CONFIGURATION:
|
|
|
|
{
|
|
|
|
uint8_t const cfg_num = (uint8_t) p_request->wValue;
|
|
|
|
|
2021-08-26 17:02:24 +07:00
|
|
|
// Only process if new configure is different
|
|
|
|
if (_usbd_dev.cfg_num != cfg_num)
|
|
|
|
{
|
|
|
|
if ( _usbd_dev.cfg_num )
|
|
|
|
{
|
|
|
|
// already configured: need to clear all endpoints and driver first
|
2021-08-27 12:38:41 +07:00
|
|
|
TU_LOG(USBD_DBG, " Clear current Configuration (%u) before switching\r\n", _usbd_dev.cfg_num);
|
2021-08-26 17:02:24 +07:00
|
|
|
|
|
|
|
// close all non-control endpoints, cancel all pending transfers if any
|
|
|
|
dcd_edpt_close_all(rhport);
|
2020-01-14 23:30:39 -05:00
|
|
|
|
2021-08-26 17:02:24 +07:00
|
|
|
// close all drivers and current configured state except bus speed
|
|
|
|
uint8_t const speed = _usbd_dev.speed;
|
|
|
|
configuration_reset(rhport);
|
|
|
|
|
|
|
|
_usbd_dev.speed = speed; // restore speed
|
|
|
|
}
|
|
|
|
|
|
|
|
// switch to new configuration if not zero
|
|
|
|
if ( cfg_num ) TU_ASSERT( process_set_config(rhport, cfg_num) );
|
|
|
|
}
|
|
|
|
|
|
|
|
_usbd_dev.cfg_num = cfg_num;
|
2020-01-14 23:30:39 -05:00
|
|
|
tud_control_status(rhport, p_request);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case TUSB_REQ_GET_DESCRIPTOR:
|
|
|
|
TU_VERIFY( process_get_descriptor(rhport, p_request) );
|
|
|
|
break;
|
|
|
|
|
|
|
|
case TUSB_REQ_SET_FEATURE:
|
|
|
|
// Only support remote wakeup for device feature
|
|
|
|
TU_VERIFY(TUSB_REQ_FEATURE_REMOTE_WAKEUP == p_request->wValue);
|
|
|
|
|
2021-08-26 17:02:24 +07:00
|
|
|
TU_LOG(USBD_DBG, " Enable Remote Wakeup\r\n");
|
2021-08-16 20:22:14 +07:00
|
|
|
|
2020-01-14 23:30:39 -05:00
|
|
|
// Host may enable remote wake up before suspending especially HID device
|
|
|
|
_usbd_dev.remote_wakeup_en = true;
|
|
|
|
tud_control_status(rhport, p_request);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case TUSB_REQ_CLEAR_FEATURE:
|
|
|
|
// Only support remote wakeup for device feature
|
|
|
|
TU_VERIFY(TUSB_REQ_FEATURE_REMOTE_WAKEUP == p_request->wValue);
|
|
|
|
|
2021-08-26 17:02:24 +07:00
|
|
|
TU_LOG(USBD_DBG, " Disable Remote Wakeup\r\n");
|
2021-08-16 20:22:14 +07:00
|
|
|
|
2020-01-14 23:30:39 -05:00
|
|
|
// Host may disable remote wake up after resuming
|
|
|
|
_usbd_dev.remote_wakeup_en = false;
|
|
|
|
tud_control_status(rhport, p_request);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case TUSB_REQ_GET_STATUS:
|
|
|
|
{
|
|
|
|
// Device status bit mask
|
|
|
|
// - Bit 0: Self Powered
|
|
|
|
// - Bit 1: Remote Wakeup enabled
|
2022-06-24 19:45:49 +07:00
|
|
|
uint16_t status = (uint16_t) ((_usbd_dev.self_powered ? 1u : 0u) | (_usbd_dev.remote_wakeup_en ? 2u : 0u));
|
2020-01-14 23:30:39 -05:00
|
|
|
tud_control_xfer(rhport, p_request, &status, 2);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
// Unknown/Unsupported request
|
|
|
|
default: TU_BREAKPOINT(); return false;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
//------------- Class/Interface Specific Request -------------//
|
|
|
|
case TUSB_REQ_RCPT_INTERFACE:
|
|
|
|
{
|
|
|
|
uint8_t const itf = tu_u16_low(p_request->wIndex);
|
|
|
|
TU_VERIFY(itf < TU_ARRAY_SIZE(_usbd_dev.itf2drv));
|
|
|
|
|
2020-08-07 12:13:13 +07:00
|
|
|
usbd_class_driver_t const * driver = get_driver(_usbd_dev.itf2drv[itf]);
|
2020-04-29 10:49:58 +07:00
|
|
|
TU_VERIFY(driver);
|
2020-01-14 23:30:39 -05:00
|
|
|
|
2020-04-15 17:51:02 +07:00
|
|
|
// all requests to Interface (STD or Class) is forwarded to class driver.
|
|
|
|
// notable requests are: GET HID REPORT DESCRIPTOR, SET_INTERFACE, GET_INTERFACE
|
2020-04-29 10:49:58 +07:00
|
|
|
if ( !invoke_class_control(rhport, driver, p_request) )
|
2020-01-14 23:30:39 -05:00
|
|
|
{
|
2021-03-10 11:33:03 +07:00
|
|
|
// For GET_INTERFACE and SET_INTERFACE, it is mandatory to respond even if the class
|
|
|
|
// driver doesn't use alternate settings or implement this
|
|
|
|
TU_VERIFY(TUSB_REQ_TYPE_STANDARD == p_request->bmRequestType_bit.type);
|
2020-01-14 23:30:39 -05:00
|
|
|
|
2021-08-26 13:07:10 +07:00
|
|
|
switch(p_request->bRequest)
|
2021-03-10 11:33:03 +07:00
|
|
|
{
|
2021-08-26 13:07:10 +07:00
|
|
|
case TUSB_REQ_GET_INTERFACE:
|
|
|
|
case TUSB_REQ_SET_INTERFACE:
|
|
|
|
// Clear complete callback if driver set since it can also stall the request.
|
|
|
|
usbd_control_set_complete_callback(NULL);
|
|
|
|
|
|
|
|
if (TUSB_REQ_GET_INTERFACE == p_request->bRequest)
|
|
|
|
{
|
|
|
|
uint8_t alternate = 0;
|
|
|
|
tud_control_xfer(rhport, p_request, &alternate, 1);
|
|
|
|
}else
|
|
|
|
{
|
|
|
|
tud_control_status(rhport, p_request);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default: return false;
|
2021-03-10 11:33:03 +07:00
|
|
|
}
|
2020-01-14 23:30:39 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
//------------- Endpoint Request -------------//
|
|
|
|
case TUSB_REQ_RCPT_ENDPOINT:
|
|
|
|
{
|
|
|
|
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);
|
|
|
|
|
|
|
|
TU_ASSERT(ep_num < TU_ARRAY_SIZE(_usbd_dev.ep2drv) );
|
|
|
|
|
2021-02-11 11:22:02 +07:00
|
|
|
usbd_class_driver_t const * driver = get_driver(_usbd_dev.ep2drv[ep_num][ep_dir]);
|
2020-01-14 23:30:39 -05:00
|
|
|
|
2021-02-11 11:22:02 +07:00
|
|
|
if ( TUSB_REQ_TYPE_STANDARD != p_request->bmRequestType_bit.type )
|
2020-01-14 23:30:39 -05:00
|
|
|
{
|
2021-02-11 11:22:02 +07:00
|
|
|
// Forward class request to its driver
|
|
|
|
TU_VERIFY(driver);
|
|
|
|
return invoke_class_control(rhport, driver, p_request);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// Handle STD request to endpoint
|
2020-01-14 23:30:39 -05:00
|
|
|
switch ( p_request->bRequest )
|
|
|
|
{
|
|
|
|
case TUSB_REQ_GET_STATUS:
|
|
|
|
{
|
|
|
|
uint16_t status = usbd_edpt_stalled(rhport, ep_addr) ? 0x0001 : 0x0000;
|
|
|
|
tud_control_xfer(rhport, p_request, &status, 2);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case TUSB_REQ_CLEAR_FEATURE:
|
|
|
|
case TUSB_REQ_SET_FEATURE:
|
2021-02-11 11:22:02 +07:00
|
|
|
{
|
|
|
|
if ( TUSB_REQ_FEATURE_EDPT_HALT == p_request->wValue )
|
|
|
|
{
|
|
|
|
if ( TUSB_REQ_CLEAR_FEATURE == p_request->bRequest )
|
|
|
|
{
|
|
|
|
usbd_edpt_clear_stall(rhport, ep_addr);
|
|
|
|
}else
|
|
|
|
{
|
|
|
|
usbd_edpt_stall(rhport, ep_addr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (driver)
|
|
|
|
{
|
|
|
|
// Some classes such as USBTMC needs to clear/re-init its buffer when receiving CLEAR_FEATURE request
|
|
|
|
// We will also forward std request targeted endpoint to class drivers as well
|
|
|
|
|
|
|
|
// STD request must always be ACKed regardless of driver returned value
|
|
|
|
// Also clear complete callback if driver set since it can also stall the request.
|
|
|
|
(void) invoke_class_control(rhport, driver, p_request);
|
|
|
|
usbd_control_set_complete_callback(NULL);
|
|
|
|
|
|
|
|
// skip ZLP status if driver already did that
|
|
|
|
if ( !_usbd_dev.ep_status[0][TUSB_DIR_IN].busy ) tud_control_status(rhport, p_request);
|
|
|
|
}
|
|
|
|
}
|
2020-01-14 23:30:39 -05:00
|
|
|
break;
|
|
|
|
|
|
|
|
// Unknown/Unsupported request
|
|
|
|
default: TU_BREAKPOINT(); return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
// Unknown recipient
|
|
|
|
default: TU_BREAKPOINT(); return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Process Set Configure Request
|
|
|
|
// This function parse configuration descriptor & open drivers accordingly
|
|
|
|
static bool process_set_config(uint8_t rhport, uint8_t cfg_num)
|
|
|
|
{
|
2021-08-26 17:02:24 +07:00
|
|
|
// index is cfg_num-1
|
|
|
|
tusb_desc_configuration_t const * desc_cfg = (tusb_desc_configuration_t const *) tud_descriptor_configuration_cb(cfg_num-1);
|
2020-01-14 23:30:39 -05:00
|
|
|
TU_ASSERT(desc_cfg != NULL && desc_cfg->bDescriptorType == TUSB_DESC_CONFIGURATION);
|
|
|
|
|
|
|
|
// Parse configuration descriptor
|
2022-06-24 19:45:49 +07:00
|
|
|
_usbd_dev.remote_wakeup_support = (desc_cfg->bmAttributes & TUSB_DESC_CONFIG_ATT_REMOTE_WAKEUP) ? 1u : 0u;
|
|
|
|
_usbd_dev.self_powered = (desc_cfg->bmAttributes & TUSB_DESC_CONFIG_ATT_SELF_POWERED ) ? 1u : 0u;
|
2020-01-14 23:30:39 -05:00
|
|
|
|
|
|
|
// Parse interface descriptor
|
|
|
|
uint8_t const * p_desc = ((uint8_t const*) desc_cfg) + sizeof(tusb_desc_configuration_t);
|
2021-05-29 21:23:39 +02:00
|
|
|
uint8_t const * desc_end = ((uint8_t const*) desc_cfg) + tu_le16toh(desc_cfg->wTotalLength);
|
2020-01-14 23:30:39 -05:00
|
|
|
|
|
|
|
while( p_desc < desc_end )
|
|
|
|
{
|
2021-09-14 21:02:38 +07:00
|
|
|
uint8_t assoc_itf_count = 1;
|
2020-04-15 16:18:24 +07:00
|
|
|
|
|
|
|
// Class will always starts with Interface Association (if any) and then Interface descriptor
|
2020-01-14 23:30:39 -05:00
|
|
|
if ( TUSB_DESC_INTERFACE_ASSOCIATION == tu_desc_type(p_desc) )
|
|
|
|
{
|
2021-09-14 21:02:38 +07:00
|
|
|
tusb_desc_interface_assoc_t const * desc_iad = (tusb_desc_interface_assoc_t const *) p_desc;
|
|
|
|
assoc_itf_count = desc_iad->bInterfaceCount;
|
|
|
|
|
2020-04-15 16:18:24 +07:00
|
|
|
p_desc = tu_desc_next(p_desc); // next to Interface
|
2021-09-14 21:02:38 +07:00
|
|
|
|
|
|
|
// IAD's first interface number and class should match with opened interface
|
|
|
|
//TU_ASSERT(desc_iad->bFirstInterface == desc_itf->bInterfaceNumber &&
|
|
|
|
// desc_iad->bFunctionClass == desc_itf->bInterfaceClass);
|
2020-04-15 16:18:24 +07:00
|
|
|
}
|
2020-01-14 23:30:39 -05:00
|
|
|
|
2020-04-15 16:18:24 +07:00
|
|
|
TU_ASSERT( TUSB_DESC_INTERFACE == tu_desc_type(p_desc) );
|
|
|
|
tusb_desc_interface_t const * desc_itf = (tusb_desc_interface_t const*) p_desc;
|
2021-08-20 19:39:33 +07:00
|
|
|
|
|
|
|
// Find driver for this interface
|
2022-06-24 19:45:49 +07:00
|
|
|
uint16_t const remaining_len = (uint16_t) (desc_end-p_desc);
|
2020-04-15 16:18:24 +07:00
|
|
|
uint8_t drv_id;
|
2020-04-29 10:49:58 +07:00
|
|
|
for (drv_id = 0; drv_id < TOTAL_DRIVER_COUNT; drv_id++)
|
2020-04-15 16:18:24 +07:00
|
|
|
{
|
2020-04-29 10:49:58 +07:00
|
|
|
usbd_class_driver_t const *driver = get_driver(drv_id);
|
2022-06-24 19:45:49 +07:00
|
|
|
TU_ASSERT(driver);
|
2020-08-07 12:13:13 +07:00
|
|
|
uint16_t const drv_len = driver->open(rhport, desc_itf, remaining_len);
|
2020-05-27 19:01:59 +07:00
|
|
|
|
2021-10-01 21:52:29 +07:00
|
|
|
if ( (sizeof(tusb_desc_interface_t) <= drv_len) && (drv_len <= remaining_len) )
|
2020-01-14 23:30:39 -05:00
|
|
|
{
|
2021-09-14 21:02:38 +07:00
|
|
|
// Open successfully
|
2022-06-16 12:01:19 +07:00
|
|
|
TU_LOG(USBD_DBG, " %s opened\r\n", driver->name);
|
2021-08-20 19:39:33 +07:00
|
|
|
|
2021-10-01 21:52:29 +07:00
|
|
|
// Some drivers use 2 or more interfaces but may not have IAD e.g MIDI (always) or
|
|
|
|
// BTH (even CDC) with class in device descriptor (single interface)
|
|
|
|
if ( assoc_itf_count == 1)
|
|
|
|
{
|
|
|
|
#if CFG_TUD_CDC
|
|
|
|
if ( driver->open == cdcd_open ) assoc_itf_count = 2;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if CFG_TUD_MIDI
|
|
|
|
if ( driver->open == midid_open ) assoc_itf_count = 2;
|
|
|
|
#endif
|
2020-01-14 23:30:39 -05:00
|
|
|
|
2021-10-01 21:52:29 +07:00
|
|
|
#if CFG_TUD_BTH && CFG_TUD_BTH_ISO_ALT_COUNT
|
|
|
|
if ( driver->open == btd_open ) assoc_itf_count = 2;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2021-09-14 21:02:38 +07:00
|
|
|
// bind (associated) interfaces to found driver
|
|
|
|
for(uint8_t i=0; i<assoc_itf_count; i++)
|
2020-04-15 00:59:56 +07:00
|
|
|
{
|
2021-10-01 21:52:29 +07:00
|
|
|
uint8_t const itf_num = desc_itf->bInterfaceNumber+i;
|
2020-01-14 23:30:39 -05:00
|
|
|
|
2021-10-01 21:52:29 +07:00
|
|
|
// Interface number must not be used already
|
|
|
|
TU_ASSERT(DRVID_INVALID == _usbd_dev.itf2drv[itf_num]);
|
|
|
|
_usbd_dev.itf2drv[itf_num] = drv_id;
|
2020-04-15 00:59:56 +07:00
|
|
|
}
|
2020-01-14 23:30:39 -05:00
|
|
|
|
2021-08-20 19:39:33 +07:00
|
|
|
// bind all endpoints to found driver
|
2021-08-20 18:01:10 +07:00
|
|
|
tu_edpt_bind_driver(_usbd_dev.ep2drv, desc_itf, drv_len, drv_id);
|
2020-08-07 12:13:13 +07:00
|
|
|
|
2021-10-01 21:52:29 +07:00
|
|
|
// next Interface
|
|
|
|
p_desc += drv_len;
|
2020-08-07 12:13:13 +07:00
|
|
|
|
2021-08-20 19:39:33 +07:00
|
|
|
break; // exit driver find loop
|
2020-04-15 00:59:56 +07:00
|
|
|
}
|
2020-01-14 23:30:39 -05:00
|
|
|
}
|
2020-04-15 00:59:56 +07:00
|
|
|
|
2021-09-14 12:47:20 +07:00
|
|
|
// Failed if there is no supported drivers
|
2020-08-07 12:13:13 +07:00
|
|
|
TU_ASSERT(drv_id < TOTAL_DRIVER_COUNT);
|
2020-01-14 23:30:39 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// invoke callback
|
|
|
|
if (tud_mount_cb) tud_mount_cb();
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// return descriptor's buffer and update desc_len
|
|
|
|
static bool process_get_descriptor(uint8_t rhport, tusb_control_request_t const * p_request)
|
|
|
|
{
|
|
|
|
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:
|
2020-03-22 17:19:57 +07:00
|
|
|
{
|
2022-06-16 12:01:19 +07:00
|
|
|
TU_LOG(USBD_DBG, " Device\r\n");
|
2020-04-26 14:51:44 +07:00
|
|
|
|
2021-10-15 17:35:05 +07:00
|
|
|
void* desc_device = (void*) (uintptr_t) tud_descriptor_device_cb();
|
2020-03-22 17:19:57 +07:00
|
|
|
|
2021-10-15 17:35:05 +07:00
|
|
|
// Only response with exactly 1 Packet if: not addressed and host requested more data than device descriptor has.
|
2020-03-22 17:19:57 +07:00
|
|
|
// This only happens with the very first get device descriptor and EP0 size = 8 or 16.
|
2021-09-23 14:03:22 +02:00
|
|
|
if ((CFG_TUD_ENDPOINT0_SIZE < sizeof(tusb_desc_device_t)) && !_usbd_dev.addressed &&
|
2021-10-15 17:35:05 +07:00
|
|
|
((tusb_control_request_t const*) p_request)->wLength > sizeof(tusb_desc_device_t))
|
2020-03-22 17:19:57 +07:00
|
|
|
{
|
|
|
|
// Hack here: we modify the request length to prevent usbd_control response with zlp
|
2021-10-15 17:35:05 +07:00
|
|
|
// since we are responding with 1 packet & less data than wLength.
|
|
|
|
tusb_control_request_t mod_request = *p_request;
|
|
|
|
mod_request.wLength = CFG_TUD_ENDPOINT0_SIZE;
|
2020-03-22 17:19:57 +07:00
|
|
|
|
2021-10-17 16:26:27 +07:00
|
|
|
return tud_control_xfer(rhport, &mod_request, desc_device, CFG_TUD_ENDPOINT0_SIZE);
|
2021-10-15 17:35:05 +07:00
|
|
|
}else
|
|
|
|
{
|
|
|
|
return tud_control_xfer(rhport, p_request, desc_device, sizeof(tusb_desc_device_t));
|
|
|
|
}
|
2020-03-22 17:19:57 +07:00
|
|
|
}
|
2021-12-30 20:59:53 +07:00
|
|
|
// break; // unreachable
|
2020-01-14 23:30:39 -05:00
|
|
|
|
|
|
|
case TUSB_DESC_BOS:
|
|
|
|
{
|
2022-06-16 12:01:19 +07:00
|
|
|
TU_LOG(USBD_DBG, " BOS\r\n");
|
2020-04-26 14:51:44 +07:00
|
|
|
|
2020-01-14 23:30:39 -05:00
|
|
|
// requested by host if USB > 2.0 ( i.e 2.1 or 3.x )
|
|
|
|
if (!tud_descriptor_bos_cb) return false;
|
|
|
|
|
2021-10-15 17:35:05 +07:00
|
|
|
uintptr_t desc_bos = (uintptr_t) tud_descriptor_bos_cb();
|
|
|
|
TU_ASSERT(desc_bos);
|
2020-06-16 14:13:30 +02:00
|
|
|
|
|
|
|
// Use offsetof to avoid pointer to the odd/misaligned address
|
2021-10-15 17:35:05 +07:00
|
|
|
uint16_t const total_len = tu_le16toh( tu_unaligned_read16((const void*) (desc_bos + offsetof(tusb_desc_bos_t, wTotalLength))) );
|
2020-01-14 23:30:39 -05:00
|
|
|
|
2021-07-22 17:45:24 +07:00
|
|
|
return tud_control_xfer(rhport, p_request, (void*) desc_bos, total_len);
|
2020-01-14 23:30:39 -05:00
|
|
|
}
|
2021-12-30 20:59:53 +07:00
|
|
|
// break; // unreachable
|
2020-01-14 23:30:39 -05:00
|
|
|
|
|
|
|
case TUSB_DESC_CONFIGURATION:
|
2021-09-09 14:34:03 +07:00
|
|
|
case TUSB_DESC_OTHER_SPEED_CONFIG:
|
2020-01-14 23:30:39 -05:00
|
|
|
{
|
2021-10-15 17:35:05 +07:00
|
|
|
uintptr_t desc_config;
|
2021-09-09 14:34:03 +07:00
|
|
|
|
|
|
|
if ( desc_type == TUSB_DESC_CONFIGURATION )
|
|
|
|
{
|
2022-06-16 12:01:19 +07:00
|
|
|
TU_LOG(USBD_DBG, " Configuration[%u]\r\n", desc_index);
|
2021-10-15 17:35:05 +07:00
|
|
|
desc_config = (uintptr_t) tud_descriptor_configuration_cb(desc_index);
|
2021-09-09 14:34:03 +07:00
|
|
|
}else
|
|
|
|
{
|
|
|
|
// Host only request this after getting Device Qualifier descriptor
|
2022-06-16 12:01:19 +07:00
|
|
|
TU_LOG(USBD_DBG, " Other Speed Configuration\r\n");
|
2021-09-09 14:34:03 +07:00
|
|
|
TU_VERIFY( tud_descriptor_other_speed_configuration_cb );
|
2021-10-15 17:35:05 +07:00
|
|
|
desc_config = (uintptr_t) tud_descriptor_other_speed_configuration_cb(desc_index);
|
2021-09-09 14:34:03 +07:00
|
|
|
}
|
2020-04-26 14:51:44 +07:00
|
|
|
|
2020-04-21 10:06:17 -04:00
|
|
|
TU_ASSERT(desc_config);
|
2020-01-14 23:30:39 -05:00
|
|
|
|
2020-06-16 14:13:30 +02:00
|
|
|
// Use offsetof to avoid pointer to the odd/misaligned address
|
2021-10-15 17:35:05 +07:00
|
|
|
uint16_t const total_len = tu_le16toh( tu_unaligned_read16((const void*) (desc_config + offsetof(tusb_desc_configuration_t, wTotalLength))) );
|
2020-01-14 23:30:39 -05:00
|
|
|
|
2021-07-22 17:45:24 +07:00
|
|
|
return tud_control_xfer(rhport, p_request, (void*) desc_config, total_len);
|
2020-01-14 23:30:39 -05:00
|
|
|
}
|
2021-12-30 20:59:53 +07:00
|
|
|
// break; // unreachable
|
2020-01-14 23:30:39 -05:00
|
|
|
|
|
|
|
case TUSB_DESC_STRING:
|
2020-12-18 11:48:48 +01:00
|
|
|
{
|
2022-06-16 12:01:19 +07:00
|
|
|
TU_LOG(USBD_DBG, " String[%u]\r\n", desc_index);
|
2020-04-26 14:51:44 +07:00
|
|
|
|
2020-01-14 23:30:39 -05:00
|
|
|
// String Descriptor always uses the desc set from user
|
2021-10-15 17:35:05 +07:00
|
|
|
uint8_t const* desc_str = (uint8_t const*) tud_descriptor_string_cb(desc_index, tu_le16toh(p_request->wIndex));
|
2020-07-19 19:13:17 +08:00
|
|
|
TU_VERIFY(desc_str);
|
2020-01-14 23:30:39 -05:00
|
|
|
|
2020-07-19 19:13:17 +08:00
|
|
|
// first byte of descriptor is its size
|
2021-10-15 17:35:05 +07:00
|
|
|
return tud_control_xfer(rhport, p_request, (void*) (uintptr_t) desc_str, tu_desc_len(desc_str));
|
2020-12-18 11:48:48 +01:00
|
|
|
}
|
2021-12-30 20:59:53 +07:00
|
|
|
// break; // unreachable
|
2020-01-14 23:30:39 -05:00
|
|
|
|
|
|
|
case TUSB_DESC_DEVICE_QUALIFIER:
|
2021-09-14 11:58:22 +02:00
|
|
|
{
|
2022-06-16 12:01:19 +07:00
|
|
|
TU_LOG(USBD_DBG, " Device Qualifier\r\n");
|
2020-04-26 14:51:44 +07:00
|
|
|
|
2021-09-09 14:34:03 +07:00
|
|
|
TU_VERIFY( tud_descriptor_device_qualifier_cb );
|
2020-01-14 23:30:39 -05:00
|
|
|
|
2021-09-09 14:34:03 +07:00
|
|
|
uint8_t const* desc_qualifier = tud_descriptor_device_qualifier_cb();
|
|
|
|
TU_VERIFY(desc_qualifier);
|
2020-01-14 23:30:39 -05:00
|
|
|
|
2021-09-09 14:34:03 +07:00
|
|
|
// first byte of descriptor is its size
|
2021-10-15 17:35:05 +07:00
|
|
|
return tud_control_xfer(rhport, p_request, (void*) (uintptr_t) desc_qualifier, tu_desc_len(desc_qualifier));
|
2021-09-14 11:58:22 +02:00
|
|
|
}
|
2021-12-30 20:59:53 +07:00
|
|
|
// break; // unreachable
|
2020-01-14 23:30:39 -05:00
|
|
|
|
|
|
|
default: return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
// DCD Event Handler
|
|
|
|
//--------------------------------------------------------------------+
|
2022-04-28 18:00:30 +07:00
|
|
|
TU_ATTR_FAST_FUNC void dcd_event_handler(dcd_event_t const * event, bool in_isr)
|
2020-01-14 23:30:39 -05:00
|
|
|
{
|
|
|
|
switch (event->event_id)
|
|
|
|
{
|
|
|
|
case DCD_EVENT_UNPLUGGED:
|
2021-09-09 16:01:05 +07:00
|
|
|
_usbd_dev.connected = 0;
|
|
|
|
_usbd_dev.addressed = 0;
|
|
|
|
_usbd_dev.cfg_num = 0;
|
|
|
|
_usbd_dev.suspended = 0;
|
|
|
|
osal_queue_send(_usbd_q, event, in_isr);
|
2020-01-14 23:30:39 -05:00
|
|
|
break;
|
|
|
|
|
|
|
|
case DCD_EVENT_SUSPEND:
|
2021-03-02 23:24:36 +07:00
|
|
|
// NOTE: When plugging/unplugging device, the D+/D- state are unstable and
|
|
|
|
// can accidentally meet the SUSPEND condition ( Bus Idle for 3ms ).
|
|
|
|
// In addition, some MCUs such as SAMD or boards that haven no VBUS detection cannot distinguish
|
|
|
|
// suspended vs disconnected. We will skip handling SUSPEND/RESUME event if not currently connected
|
2020-01-14 23:30:39 -05:00
|
|
|
if ( _usbd_dev.connected )
|
|
|
|
{
|
|
|
|
_usbd_dev.suspended = 1;
|
|
|
|
osal_queue_send(_usbd_q, event, in_isr);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case DCD_EVENT_RESUME:
|
|
|
|
// skip event if not connected (especially required for SAMD)
|
|
|
|
if ( _usbd_dev.connected )
|
|
|
|
{
|
|
|
|
_usbd_dev.suspended = 0;
|
|
|
|
osal_queue_send(_usbd_q, event, in_isr);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2021-08-16 20:22:14 +07:00
|
|
|
case DCD_EVENT_SOF:
|
2022-03-14 20:40:33 +01:00
|
|
|
// SOF driver handler in ISR context
|
|
|
|
for (uint8_t i = 0; i < TOTAL_DRIVER_COUNT; i++)
|
|
|
|
{
|
|
|
|
usbd_class_driver_t const * driver = get_driver(i);
|
2022-06-24 19:45:49 +07:00
|
|
|
if (driver && driver->sof)
|
2022-03-14 20:40:33 +01:00
|
|
|
{
|
2022-06-10 01:45:48 +07:00
|
|
|
driver->sof(event->rhport, event->sof.frame_count);
|
2022-03-14 20:40:33 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-08-16 20:22:14 +07:00
|
|
|
// Some MCUs after running dcd_remote_wakeup() does not have way to detect the end of remote wakeup
|
|
|
|
// which last 1-15 ms. DCD can use SOF as a clear indicator that bus is back to operational
|
|
|
|
if ( _usbd_dev.suspended )
|
|
|
|
{
|
|
|
|
_usbd_dev.suspended = 0;
|
2022-03-07 23:03:37 +07:00
|
|
|
|
2021-08-16 20:22:14 +07:00
|
|
|
dcd_event_t const event_resume = { .rhport = event->rhport, .event_id = DCD_EVENT_RESUME };
|
|
|
|
osal_queue_send(_usbd_q, &event_resume, in_isr);
|
|
|
|
}
|
2022-03-29 16:53:45 +07:00
|
|
|
|
|
|
|
// skip osal queue for SOF in usbd task
|
2021-08-16 20:22:14 +07:00
|
|
|
break;
|
|
|
|
|
2020-01-14 23:30:39 -05:00
|
|
|
default:
|
|
|
|
osal_queue_send(_usbd_q, event, in_isr);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------+
|
2021-05-13 12:21:12 +07:00
|
|
|
// USBD API For Class Driver
|
2020-01-14 23:30:39 -05:00
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
|
2022-02-23 18:42:32 +07:00
|
|
|
void usbd_int_set(bool enabled)
|
|
|
|
{
|
|
|
|
if (enabled)
|
|
|
|
{
|
|
|
|
dcd_int_enable(_usbd_rhport);
|
|
|
|
}else
|
|
|
|
{
|
|
|
|
dcd_int_disable(_usbd_rhport);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-14 23:30:39 -05: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)
|
|
|
|
{
|
|
|
|
for(int i=0; i<ep_count; i++)
|
|
|
|
{
|
|
|
|
tusb_desc_endpoint_t const * desc_ep = (tusb_desc_endpoint_t const *) p_desc;
|
|
|
|
|
|
|
|
TU_ASSERT(TUSB_DESC_ENDPOINT == desc_ep->bDescriptorType && xfer_type == desc_ep->bmAttributes.xfer);
|
2020-04-26 14:51:44 +07:00
|
|
|
TU_ASSERT(usbd_edpt_open(rhport, desc_ep));
|
2020-01-14 23:30:39 -05:00
|
|
|
|
|
|
|
if ( tu_edpt_dir(desc_ep->bEndpointAddress) == TUSB_DIR_IN )
|
|
|
|
{
|
|
|
|
(*ep_in) = desc_ep->bEndpointAddress;
|
|
|
|
}else
|
|
|
|
{
|
|
|
|
(*ep_out) = desc_ep->bEndpointAddress;
|
|
|
|
}
|
|
|
|
|
|
|
|
p_desc = tu_desc_next(p_desc);
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Helper to defer an isr function
|
|
|
|
void usbd_defer_func(osal_task_func_t func, void* param, bool in_isr)
|
|
|
|
{
|
|
|
|
dcd_event_t event =
|
|
|
|
{
|
|
|
|
.rhport = 0,
|
|
|
|
.event_id = USBD_EVENT_FUNC_CALL,
|
|
|
|
};
|
|
|
|
|
|
|
|
event.func_call.func = func;
|
|
|
|
event.func_call.param = param;
|
|
|
|
|
|
|
|
dcd_event_handler(&event, in_isr);
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
// USBD Endpoint API
|
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
|
2020-04-26 14:51:44 +07:00
|
|
|
bool usbd_edpt_open(uint8_t rhport, tusb_desc_endpoint_t const * desc_ep)
|
|
|
|
{
|
2022-06-02 16:51:17 +07:00
|
|
|
rhport = _usbd_rhport;
|
|
|
|
|
2021-07-22 17:04:55 +07:00
|
|
|
TU_ASSERT(tu_edpt_number(desc_ep->bEndpointAddress) < CFG_TUD_ENDPPOINT_MAX);
|
2021-08-20 18:01:10 +07:00
|
|
|
TU_ASSERT(tu_edpt_validate(desc_ep, (tusb_speed_t) _usbd_dev.speed));
|
2021-01-08 22:34:36 +07:00
|
|
|
|
2020-04-26 14:51:44 +07:00
|
|
|
return dcd_edpt_open(rhport, desc_ep);
|
|
|
|
}
|
|
|
|
|
2020-09-09 15:48:11 +07:00
|
|
|
bool usbd_edpt_claim(uint8_t rhport, uint8_t ep_addr)
|
|
|
|
{
|
|
|
|
(void) rhport;
|
|
|
|
|
2021-08-12 17:07:39 +07:00
|
|
|
// TODO add this check later, also make sure we don't starve an out endpoint while suspending
|
|
|
|
// TU_VERIFY(tud_ready());
|
2021-08-12 15:46:33 +07:00
|
|
|
|
2022-03-09 17:34:12 +07:00
|
|
|
uint8_t const epnum = tu_edpt_number(ep_addr);
|
|
|
|
uint8_t const dir = tu_edpt_dir(ep_addr);
|
|
|
|
tu_edpt_state_t* ep_state = &_usbd_dev.ep_status[epnum][dir];
|
2020-09-09 15:48:11 +07:00
|
|
|
|
2022-03-09 17:34:12 +07:00
|
|
|
return tu_edpt_claim(ep_state, _usbd_mutex);
|
2020-09-09 15:48:11 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
bool usbd_edpt_release(uint8_t rhport, uint8_t ep_addr)
|
|
|
|
{
|
|
|
|
(void) rhport;
|
|
|
|
|
2022-03-09 17:34:12 +07:00
|
|
|
uint8_t const epnum = tu_edpt_number(ep_addr);
|
|
|
|
uint8_t const dir = tu_edpt_dir(ep_addr);
|
|
|
|
tu_edpt_state_t* ep_state = &_usbd_dev.ep_status[epnum][dir];
|
2020-09-09 15:48:11 +07:00
|
|
|
|
2022-03-09 17:34:12 +07:00
|
|
|
return tu_edpt_release(ep_state, _usbd_mutex);
|
2020-09-09 15:48:11 +07:00
|
|
|
}
|
|
|
|
|
2020-01-14 23:30:39 -05:00
|
|
|
bool usbd_edpt_xfer(uint8_t rhport, uint8_t ep_addr, uint8_t * buffer, uint16_t total_bytes)
|
|
|
|
{
|
2022-06-02 16:51:17 +07:00
|
|
|
rhport = _usbd_rhport;
|
|
|
|
|
2020-01-14 23:30:39 -05:00
|
|
|
uint8_t const epnum = tu_edpt_number(ep_addr);
|
|
|
|
uint8_t const dir = tu_edpt_dir(ep_addr);
|
|
|
|
|
2021-08-12 15:46:33 +07:00
|
|
|
// TODO skip ready() check for now since enumeration also use this API
|
|
|
|
// TU_VERIFY(tud_ready());
|
|
|
|
|
2022-06-16 12:01:19 +07:00
|
|
|
TU_LOG(USBD_DBG, " Queue EP %02X with %u bytes ...\r\n", ep_addr, total_bytes);
|
2020-04-26 14:51:44 +07:00
|
|
|
|
2020-09-11 00:14:07 +07:00
|
|
|
// Attempt to transfer on a busy endpoint, sound like an race condition !
|
|
|
|
TU_ASSERT(_usbd_dev.ep_status[epnum][dir].busy == 0);
|
2020-09-09 15:48:11 +07:00
|
|
|
|
2021-06-10 17:19:21 +07:00
|
|
|
// Set busy first since the actual transfer can be complete before dcd_edpt_xfer()
|
|
|
|
// could return and USBD task can preempt and clear the busy
|
2020-01-14 23:30:39 -05:00
|
|
|
_usbd_dev.ep_status[epnum][dir].busy = true;
|
|
|
|
|
2020-04-29 11:31:27 +07:00
|
|
|
if ( dcd_edpt_xfer(rhport, ep_addr, buffer, total_bytes) )
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}else
|
|
|
|
{
|
2020-09-09 16:25:31 +07:00
|
|
|
// DCD error, mark endpoint as ready to allow next transfer
|
2020-04-29 11:31:27 +07:00
|
|
|
_usbd_dev.ep_status[epnum][dir].busy = false;
|
2020-09-09 16:25:31 +07:00
|
|
|
_usbd_dev.ep_status[epnum][dir].claimed = 0;
|
2022-06-16 12:01:19 +07:00
|
|
|
TU_LOG(USBD_DBG, "FAILED\r\n");
|
2020-04-29 11:31:27 +07:00
|
|
|
TU_BREAKPOINT();
|
|
|
|
return false;
|
|
|
|
}
|
2020-01-14 23:30:39 -05:00
|
|
|
}
|
|
|
|
|
2021-01-31 19:08:23 +01:00
|
|
|
// The number of bytes has to be given explicitly to allow more flexible control of how many
|
|
|
|
// bytes should be written and second to keep the return value free to give back a boolean
|
|
|
|
// success message. If total_bytes is too big, the FIFO will copy only what is available
|
|
|
|
// into the USB buffer!
|
2021-05-26 20:25:44 +07:00
|
|
|
bool usbd_edpt_xfer_fifo(uint8_t rhport, uint8_t ep_addr, tu_fifo_t * ff, uint16_t total_bytes)
|
2021-01-04 12:02:08 +01:00
|
|
|
{
|
2022-06-02 16:51:17 +07:00
|
|
|
rhport = _usbd_rhport;
|
|
|
|
|
2021-01-04 12:02:08 +01:00
|
|
|
uint8_t const epnum = tu_edpt_number(ep_addr);
|
|
|
|
uint8_t const dir = tu_edpt_dir(ep_addr);
|
|
|
|
|
2022-06-16 12:01:19 +07:00
|
|
|
TU_LOG(USBD_DBG, " Queue ISO EP %02X with %u bytes ... ", ep_addr, total_bytes);
|
2021-01-04 12:02:08 +01:00
|
|
|
|
|
|
|
// Attempt to transfer on a busy endpoint, sound like an race condition !
|
|
|
|
TU_ASSERT(_usbd_dev.ep_status[epnum][dir].busy == 0);
|
|
|
|
|
|
|
|
// Set busy first since the actual transfer can be complete before dcd_edpt_xfer() could return
|
|
|
|
// and usbd task can preempt and clear the busy
|
|
|
|
_usbd_dev.ep_status[epnum][dir].busy = true;
|
|
|
|
|
2021-03-02 20:00:39 +01:00
|
|
|
if (dcd_edpt_xfer_fifo(rhport, ep_addr, ff, total_bytes))
|
2021-01-04 12:02:08 +01:00
|
|
|
{
|
2022-06-16 12:01:19 +07:00
|
|
|
TU_LOG(USBD_DBG, "OK\r\n");
|
2021-01-04 12:02:08 +01:00
|
|
|
return true;
|
|
|
|
}else
|
|
|
|
{
|
|
|
|
// DCD error, mark endpoint as ready to allow next transfer
|
|
|
|
_usbd_dev.ep_status[epnum][dir].busy = false;
|
|
|
|
_usbd_dev.ep_status[epnum][dir].claimed = 0;
|
2022-06-16 12:01:19 +07:00
|
|
|
TU_LOG(USBD_DBG, "failed\r\n");
|
2021-01-04 12:02:08 +01:00
|
|
|
TU_BREAKPOINT();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-14 23:30:39 -05:00
|
|
|
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);
|
|
|
|
|
|
|
|
return _usbd_dev.ep_status[epnum][dir].busy;
|
|
|
|
}
|
|
|
|
|
|
|
|
void usbd_edpt_stall(uint8_t rhport, uint8_t ep_addr)
|
|
|
|
{
|
2022-06-02 16:51:17 +07:00
|
|
|
rhport = _usbd_rhport;
|
2021-08-16 20:22:14 +07:00
|
|
|
|
2020-01-14 23:30:39 -05:00
|
|
|
uint8_t const epnum = tu_edpt_number(ep_addr);
|
|
|
|
uint8_t const dir = tu_edpt_dir(ep_addr);
|
|
|
|
|
2021-09-01 16:54:03 +07:00
|
|
|
// only stalled if currently cleared
|
|
|
|
if ( !_usbd_dev.ep_status[epnum][dir].stalled )
|
|
|
|
{
|
|
|
|
TU_LOG(USBD_DBG, " Stall EP %02X\r\n", ep_addr);
|
|
|
|
dcd_edpt_stall(rhport, ep_addr);
|
|
|
|
_usbd_dev.ep_status[epnum][dir].stalled = true;
|
|
|
|
_usbd_dev.ep_status[epnum][dir].busy = true;
|
|
|
|
}
|
2020-01-14 23:30:39 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
void usbd_edpt_clear_stall(uint8_t rhport, uint8_t ep_addr)
|
|
|
|
{
|
2022-06-02 16:51:17 +07:00
|
|
|
rhport = _usbd_rhport;
|
|
|
|
|
2020-01-14 23:30:39 -05:00
|
|
|
uint8_t const epnum = tu_edpt_number(ep_addr);
|
|
|
|
uint8_t const dir = tu_edpt_dir(ep_addr);
|
|
|
|
|
2021-09-01 16:54:03 +07:00
|
|
|
// only clear if currently stalled
|
|
|
|
if ( _usbd_dev.ep_status[epnum][dir].stalled )
|
|
|
|
{
|
|
|
|
TU_LOG(USBD_DBG, " Clear Stall EP %02X\r\n", ep_addr);
|
|
|
|
dcd_edpt_clear_stall(rhport, ep_addr);
|
|
|
|
_usbd_dev.ep_status[epnum][dir].stalled = false;
|
|
|
|
_usbd_dev.ep_status[epnum][dir].busy = false;
|
|
|
|
}
|
2020-01-14 23:30:39 -05: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);
|
|
|
|
|
|
|
|
return _usbd_dev.ep_status[epnum][dir].stalled;
|
|
|
|
}
|
|
|
|
|
2020-04-12 21:07:17 -04:00
|
|
|
/**
|
2020-04-16 09:57:56 -04:00
|
|
|
* usbd_edpt_close will disable an endpoint.
|
2021-03-26 15:30:43 -04:00
|
|
|
*
|
2020-04-16 09:57:56 -04:00
|
|
|
* In progress transfers on this EP may be delivered after this call.
|
2021-03-26 15:30:43 -04:00
|
|
|
*
|
2020-04-12 21:07:17 -04:00
|
|
|
*/
|
2020-04-13 09:25:16 -04:00
|
|
|
void usbd_edpt_close(uint8_t rhport, uint8_t ep_addr)
|
2020-04-12 21:07:17 -04:00
|
|
|
{
|
2022-06-02 16:51:17 +07:00
|
|
|
rhport = _usbd_rhport;
|
|
|
|
|
2020-04-12 21:07:17 -04:00
|
|
|
TU_ASSERT(dcd_edpt_close, /**/);
|
2022-06-16 12:01:19 +07:00
|
|
|
TU_LOG(USBD_DBG, " CLOSING Endpoint: 0x%02X\r\n", ep_addr);
|
2020-04-12 21:07:17 -04:00
|
|
|
|
2021-09-14 21:08:12 +02:00
|
|
|
uint8_t const epnum = tu_edpt_number(ep_addr);
|
|
|
|
uint8_t const dir = tu_edpt_dir(ep_addr);
|
|
|
|
|
2020-04-12 21:07:17 -04:00
|
|
|
dcd_edpt_close(rhport, ep_addr);
|
2021-09-14 21:08:12 +02:00
|
|
|
_usbd_dev.ep_status[epnum][dir].stalled = false;
|
|
|
|
_usbd_dev.ep_status[epnum][dir].busy = false;
|
|
|
|
_usbd_dev.ep_status[epnum][dir].claimed = false;
|
2020-04-12 21:07:17 -04:00
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-03-14 20:40:33 +01:00
|
|
|
void usbd_sof_enable(uint8_t rhport, bool en)
|
|
|
|
{
|
2022-06-02 16:51:17 +07:00
|
|
|
rhport = _usbd_rhport;
|
|
|
|
|
2022-03-29 16:53:45 +07:00
|
|
|
// TODO: Check needed if all drivers including the user sof_cb does not need an active SOF ISR any more.
|
|
|
|
// Only if all drivers switched off SOF calls the SOF interrupt may be disabled
|
2022-03-14 20:40:33 +01:00
|
|
|
dcd_sof_enable(rhport, en);
|
|
|
|
}
|
|
|
|
|
2020-01-14 23:30:39 -05:00
|
|
|
#endif
|