| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							|  |  |  | #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-01-19 17:10:08 +01:00
										 |  |  | #include "device/dcd.h"
 | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-21 13:21:58 +07:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // USBD Configuration
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-16 20:22:14 +07:00
										 |  |  | // Debug level of USBD
 | 
					
						
							| 
									
										
										
										
											2021-08-26 17:02:24 +07:00
										 |  |  | #define USBD_DBG   2
 | 
					
						
							| 
									
										
										
										
											2021-08-16 20:22:14 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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-03-09 17:34:12 +07:00
										 |  |  |   uint8_t itf2drv[16];                      // map interface number to driver (0xff is invalid)
 | 
					
						
							|  |  |  |   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; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static usbd_device_t _usbd_dev; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // 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
 | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | static usbd_class_driver_t const _usbd_driver[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   #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, | 
					
						
							|  |  |  |     .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, | 
					
						
							|  |  |  |     .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, | 
					
						
							|  |  |  |     .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, | 
					
						
							|  |  |  |     .sof              = NULL | 
					
						
							| 
									
										
										
										
											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, | 
					
						
							|  |  |  |     .sof              = NULL | 
					
						
							|  |  |  |   }, | 
					
						
							|  |  |  |   #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, | 
					
						
							|  |  |  |     .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, | 
					
						
							|  |  |  |     .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, | 
					
						
							|  |  |  |     .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, | 
					
						
							| 
									
										
										
										
											2020-11-20 15:32:16 +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, | 
					
						
							| 
									
										
										
										
											2020-11-20 15:32:16 +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, | 
					
						
							|  |  |  |     .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, | 
					
						
							|  |  |  |     .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
 | 
					
						
							|  |  |  | static usbd_class_driver_t const * _app_driver = NULL; | 
					
						
							|  |  |  | static uint8_t _app_driver_count = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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 }; | 
					
						
							|  |  |  | 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); | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | static osal_queue_t _usbd_q; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-09 15:48:11 +07:00
										 |  |  | // Mutex for claiming endpoint, only needed when using with preempted RTOS
 | 
					
						
							|  |  |  | #if CFG_TUSB_OS != OPT_OS_NONE
 | 
					
						
							|  |  |  | static osal_mutex_def_t _ubsd_mutexdef; | 
					
						
							|  |  |  | static osal_mutex_t _usbd_mutex; | 
					
						
							|  |  |  | #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
 | 
					
						
							|  |  |  | static char const* const _usbd_event_str[DCD_EVENT_COUNT] = | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							| 
									
										
										
										
											2020-11-19 21:01:33 +07:00
										 |  |  |     if ( driver->control_xfer_cb == callback ) | 
					
						
							| 
									
										
										
										
											2020-04-26 22:02:41 +07:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2020-04-29 10:49:58 +07:00
										 |  |  |       TU_LOG2("  %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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  |   TU_LOG2("USBD init\r\n"); | 
					
						
							| 
									
										
										
										
											2022-03-04 19:26:54 +07:00
										 |  |  |   TU_LOG2_INT(sizeof(usbd_device_t)); | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |   tu_varclr(&_usbd_dev); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-09 15:48:11 +07:00
										 |  |  | #if CFG_TUSB_OS != OPT_OS_NONE
 | 
					
						
							|  |  |  |   // 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); | 
					
						
							|  |  |  |     TU_LOG2("%s init\r\n", driver->name); | 
					
						
							|  |  |  |     driver->init(); | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-23 18:42:32 +07:00
										 |  |  |   _usbd_rhport = 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++ ) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     get_driver(i)->reset(rhport); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											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 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void tud_task (void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   // 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; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if ( !osal_queue_receive(_usbd_q, &event) ) return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-14 18:29:02 +07:00
										 |  |  | #if CFG_TUSB_DEBUG >= 2
 | 
					
						
							|  |  |  |     if (event.event_id == DCD_EVENT_SETUP_RECEIVED) TU_LOG2("\r\n"); // extra line for setup
 | 
					
						
							|  |  |  |     TU_LOG2("USBD %s ", event.event_id < DCD_EVENT_COUNT ? _usbd_event_str[event.event_id] : "CORRUPTED"); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     switch ( event.event_id ) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       case DCD_EVENT_BUS_RESET: | 
					
						
							| 
									
										
										
										
											2022-03-09 16:42:51 +07:00
										 |  |  |         TU_LOG2(": %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: | 
					
						
							| 
									
										
										
										
											2020-06-14 18:29:02 +07:00
										 |  |  |         TU_LOG2("\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: | 
					
						
							| 
									
										
										
										
											2020-04-26 14:51:44 +07:00
										 |  |  |         TU_LOG2_VAR(&event.setup_received); | 
					
						
							|  |  |  |         TU_LOG2("\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) ) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           TU_LOG2("  Stall EP0\r\n"); | 
					
						
							|  |  |  |           // 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); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-26 14:51:44 +07:00
										 |  |  |         TU_LOG2("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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-29 10:49:58 +07:00
										 |  |  |           TU_LOG2("  %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 ) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           TU_LOG2(": Remote Wakeup = %u\r\n", _usbd_dev.remote_wakeup_en); | 
					
						
							|  |  |  |           if (tud_suspend_cb) tud_suspend_cb(_usbd_dev.remote_wakeup_en); | 
					
						
							|  |  |  |         }else | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           TU_LOG2(" Skipped\r\n"); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											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 ) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           TU_LOG2("\r\n"); | 
					
						
							|  |  |  |           if (tud_resume_cb) tud_resume_cb(); | 
					
						
							|  |  |  |         }else | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           TU_LOG2(" Skipped\r\n"); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       case DCD_EVENT_SOF: | 
					
						
							| 
									
										
										
										
											2020-06-14 18:29:02 +07:00
										 |  |  |         TU_LOG2("\r\n"); | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							|  |  |  |           if ( driver->sof ) driver->sof(event.rhport); | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  |         } | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       case USBD_EVENT_FUNC_CALL: | 
					
						
							| 
									
										
										
										
											2020-06-14 18:29:02 +07:00
										 |  |  |         TU_LOG2("\r\n"); | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  |         if ( event.func_call.func ) event.func_call.func(event.func_call.param); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       default: | 
					
						
							|  |  |  |         TU_BREAKPOINT(); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // 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); | 
					
						
							| 
									
										
										
										
											2020-04-29 10:49:58 +07:00
										 |  |  |   TU_LOG2("  %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-03-09 16:42:51 +07:00
										 |  |  |     TU_LOG2("  %s", tu_str_std_request[p_request->bRequest]); | 
					
						
							| 
									
										
										
										
											2020-04-26 14:51:44 +07:00
										 |  |  |     if (TUSB_REQ_GET_DESCRIPTOR != p_request->bRequest) TU_LOG2("\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
 | 
					
						
							|  |  |  |           uint16_t status = (_usbd_dev.self_powered ? 1 : 0) | (_usbd_dev.remote_wakeup_en ? 2 : 0); | 
					
						
							|  |  |  |           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
 | 
					
						
							|  |  |  |   _usbd_dev.remote_wakeup_support = (desc_cfg->bmAttributes & TUSB_DESC_CONFIG_ATT_REMOTE_WAKEUP) ? 1 : 0; | 
					
						
							| 
									
										
										
										
											2021-09-14 12:47:20 +07:00
										 |  |  |   _usbd_dev.self_powered          = (desc_cfg->bmAttributes & TUSB_DESC_CONFIG_ATT_SELF_POWERED ) ? 1 : 0; | 
					
						
							| 
									
										
										
										
											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
 | 
					
						
							| 
									
										
										
										
											2021-10-01 21:52:29 +07:00
										 |  |  |     uint16_t const remaining_len = 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); | 
					
						
							| 
									
										
										
										
											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
 | 
					
						
							| 
									
										
										
										
											2020-04-29 10:49:58 +07:00
										 |  |  |         TU_LOG2("  %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
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2020-04-26 14:51:44 +07:00
										 |  |  |       TU_LOG2(" Device\r\n"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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: | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2020-04-26 14:51:44 +07:00
										 |  |  |       TU_LOG2(" BOS\r\n"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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 ) | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  |         TU_LOG2(" 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
 | 
					
						
							|  |  |  |         TU_LOG2(" Other Speed Configuration\r\n"); | 
					
						
							|  |  |  |         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
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2020-04-26 14:51:44 +07:00
										 |  |  |       TU_LOG2(" String[%u]\r\n", desc_index); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2020-04-26 14:51:44 +07:00
										 |  |  |       TU_LOG2(" Device Qualifier\r\n"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | void dcd_event_handler(dcd_event_t const * event, bool in_isr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   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: | 
					
						
							|  |  |  |       // 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; | 
					
						
							|  |  |  |         dcd_event_t const event_resume = { .rhport = event->rhport, .event_id = DCD_EVENT_RESUME }; | 
					
						
							|  |  |  |         osal_queue_send(_usbd_q, &event_resume, in_isr); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  |     default: | 
					
						
							|  |  |  |       osal_queue_send(_usbd_q, event, in_isr); | 
					
						
							|  |  |  |     break; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void dcd_event_bus_signal (uint8_t rhport, dcd_eventid_t eid, bool in_isr) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2020-05-31 23:43:29 +07:00
										 |  |  |   dcd_event_t event = { .rhport = rhport, .event_id = eid }; | 
					
						
							|  |  |  |   dcd_event_handler(&event, in_isr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void dcd_event_bus_reset (uint8_t rhport, tusb_speed_t speed, bool in_isr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   dcd_event_t event = { .rhport = rhport, .event_id = DCD_EVENT_BUS_RESET }; | 
					
						
							|  |  |  |   event.bus_reset.speed = speed; | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  |   dcd_event_handler(&event, in_isr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void dcd_event_setup_received(uint8_t rhport, uint8_t const * setup, bool in_isr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   dcd_event_t event = { .rhport = rhport, .event_id = DCD_EVENT_SETUP_RECEIVED }; | 
					
						
							|  |  |  |   memcpy(&event.setup_received, setup, 8); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   dcd_event_handler(&event, in_isr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void dcd_event_xfer_complete (uint8_t rhport, uint8_t ep_addr, uint32_t xferred_bytes, uint8_t result, bool in_isr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   dcd_event_t event = { .rhport = rhport, .event_id = DCD_EVENT_XFER_COMPLETE }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   event.xfer_complete.ep_addr = ep_addr; | 
					
						
							|  |  |  |   event.xfer_complete.len     = xferred_bytes; | 
					
						
							|  |  |  |   event.xfer_complete.result  = result; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   dcd_event_handler(&event, in_isr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							| 
									
										
										
										
											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) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											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
										 |  |  | #if TUSB_OPT_MUTEX
 | 
					
						
							|  |  |  |   return tu_edpt_claim(ep_state, _usbd_mutex); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |   return tu_edpt_claim(ep_state, NULL); | 
					
						
							| 
									
										
										
										
											2020-09-09 15:48:11 +07:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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
										 |  |  | #if TUSB_OPT_MUTEX
 | 
					
						
							|  |  |  |   return tu_edpt_release(ep_state, _usbd_mutex); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |   return tu_edpt_release(ep_state, NULL); | 
					
						
							| 
									
										
										
										
											2020-09-09 15:48:11 +07:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   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());
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-01 22:46:39 +07:00
										 |  |  |   TU_LOG2("  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; | 
					
						
							| 
									
										
										
										
											2021-06-17 01:55:35 +07:00
										 |  |  |     TU_LOG2("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
										 |  |  | { | 
					
						
							|  |  |  |   uint8_t const epnum = tu_edpt_number(ep_addr); | 
					
						
							|  |  |  |   uint8_t const dir   = tu_edpt_dir(ep_addr); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-15 23:47:28 +07:00
										 |  |  |   TU_LOG2("  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
										 |  |  |   { | 
					
						
							|  |  |  |     TU_LOG2("OK\r\n"); | 
					
						
							|  |  |  |     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; | 
					
						
							|  |  |  |     TU_LOG2("failed\r\n"); | 
					
						
							|  |  |  |     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) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											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) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   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
										 |  |  | { | 
					
						
							|  |  |  |   TU_ASSERT(dcd_edpt_close, /**/); | 
					
						
							|  |  |  |   TU_LOG2("  CLOSING Endpoint: 0x%02X\r\n", ep_addr); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-14 23:30:39 -05:00
										 |  |  | #endif
 |