| 
									
										
										
										
											2019-03-20 16:11:42 +07:00
										 |  |  | /* 
 | 
					
						
							|  |  |  |  * The MIT License (MIT) | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2019-05-14 11:48:05 +07:00
										 |  |  |  * Copyright (c) 2019 Ha Thach (tinyusb.org) | 
					
						
							| 
									
										
										
										
											2019-03-20 16:11:42 +07:00
										 |  |  |  * | 
					
						
							|  |  |  |  * Permission is hereby granted, free of charge, to any person obtaining a copy | 
					
						
							|  |  |  |  * of this software and associated documentation files (the "Software"), to deal | 
					
						
							|  |  |  |  * in the Software without restriction, including without limitation the rights | 
					
						
							|  |  |  |  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | 
					
						
							|  |  |  |  * copies of the Software, and to permit persons to whom the Software is | 
					
						
							|  |  |  |  * furnished to do so, subject to the following conditions: | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The above copyright notice and this permission notice shall be included in | 
					
						
							|  |  |  |  * all copies or substantial portions of the Software. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | 
					
						
							|  |  |  |  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | 
					
						
							|  |  |  |  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | 
					
						
							|  |  |  |  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | 
					
						
							|  |  |  |  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | 
					
						
							|  |  |  |  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN | 
					
						
							|  |  |  |  * THE SOFTWARE. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This file is part of the TinyUSB stack. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2014-03-12 14:01:38 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "tusb_option.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-23 15:25:45 +07:00
										 |  |  | #if TUSB_OPT_DEVICE_ENABLED
 | 
					
						
							| 
									
										
										
										
											2014-03-12 14:01:38 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "tusb.h"
 | 
					
						
							| 
									
										
										
										
											2018-03-11 12:31:24 +07:00
										 |  |  | #include "usbd.h"
 | 
					
						
							| 
									
										
										
										
											2018-03-22 14:15:16 +07:00
										 |  |  | #include "device/usbd_pvt.h"
 | 
					
						
							| 
									
										
										
										
											2019-06-10 18:46:00 +07:00
										 |  |  | #include "dcd.h"
 | 
					
						
							| 
									
										
										
										
											2014-03-12 14:01:38 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-23 13:19:36 +07:00
										 |  |  | #ifndef CFG_TUD_TASK_QUEUE_SZ
 | 
					
						
							|  |  |  | #define CFG_TUD_TASK_QUEUE_SZ   16
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2018-05-17 19:19:55 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-16 13:46:28 +07:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							| 
									
										
										
										
											2018-07-13 15:08:38 +07:00
										 |  |  | // Device Data
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | typedef struct { | 
					
						
							| 
									
										
										
										
											2019-06-06 10:39:37 +07:00
										 |  |  |   struct TU_ATTR_PACKED | 
					
						
							| 
									
										
										
										
											2019-03-30 02:26:15 +07:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2019-07-19 16:49:04 +07:00
										 |  |  |     volatile uint8_t connected    : 1; | 
					
						
							|  |  |  |     volatile uint8_t configured   : 1; | 
					
						
							|  |  |  |     volatile uint8_t suspended    : 1; | 
					
						
							| 
									
										
										
										
											2019-03-30 13:48:15 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-19 16:49:04 +07:00
										 |  |  |     uint8_t remote_wakeup_en      : 1; // enable/disable by host
 | 
					
						
							|  |  |  |     uint8_t remote_wakeup_support : 1; // configuration descriptor's attribute
 | 
					
						
							|  |  |  |     uint8_t self_powered          : 1; // configuration descriptor's attribute
 | 
					
						
							| 
									
										
										
										
											2019-03-30 02:26:15 +07:00
										 |  |  |   }; | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-19 16:49:04 +07:00
										 |  |  |   uint8_t itf2drv[16];     // map interface number to driver (0xff is invalid)
 | 
					
						
							|  |  |  |   uint8_t ep2drv[8][2];    // map endpoint to driver ( 0xff is invalid )
 | 
					
						
							| 
									
										
										
										
											2019-10-01 22:19:04 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   struct TU_ATTR_PACKED | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     volatile bool busy    : 1; | 
					
						
							|  |  |  |     volatile bool stalled : 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // TODO merge ep2drv here, 4-bit should be sufficient
 | 
					
						
							|  |  |  |   }ep_status[8][2]; | 
					
						
							| 
									
										
										
										
											2018-07-13 15:08:38 +07:00
										 |  |  | }usbd_device_t; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-01 14:43:34 +07:00
										 |  |  | static usbd_device_t _usbd_dev; | 
					
						
							| 
									
										
										
										
											2018-07-13 15:08:38 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-13 22:16:24 +07:00
										 |  |  | // Invalid driver ID in itf2drv[] ep2drv[][] mapping
 | 
					
						
							| 
									
										
										
										
											2019-09-16 11:13:17 +07:00
										 |  |  | enum { DRVID_INVALID = 0xFFu }; | 
					
						
							| 
									
										
										
										
											2019-09-13 22:16:24 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-13 15:08:38 +07:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // Class Driver
 | 
					
						
							| 
									
										
										
										
											2018-04-16 13:46:28 +07:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							| 
									
										
										
										
											2018-03-23 14:57:17 +07:00
										 |  |  | typedef struct { | 
					
						
							| 
									
										
										
										
											2018-05-14 13:02:01 +07:00
										 |  |  |   uint8_t class_code; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-16 18:14:47 +07:00
										 |  |  |   void (* init             ) (void); | 
					
						
							| 
									
										
										
										
											2019-07-19 20:20:13 +07:00
										 |  |  |   void (* reset            ) (uint8_t rhport); | 
					
						
							| 
									
										
										
										
											2019-07-16 18:14:47 +07:00
										 |  |  |   bool (* open             ) (uint8_t rhport, tusb_desc_interface_t const * desc_intf, uint16_t* p_length); | 
					
						
							|  |  |  |   bool (* control_request  ) (uint8_t rhport, tusb_control_request_t const * request); | 
					
						
							| 
									
										
										
										
											2019-07-12 22:03:40 +07:00
										 |  |  |   bool (* control_complete ) (uint8_t rhport, tusb_control_request_t const * request); | 
					
						
							| 
									
										
										
										
											2019-07-19 16:49:04 +07:00
										 |  |  |   bool (* xfer_cb          ) (uint8_t rhport, uint8_t ep_addr, xfer_result_t event, uint32_t xferred_bytes); | 
					
						
							| 
									
										
										
										
											2019-07-16 18:14:47 +07:00
										 |  |  |   void (* sof              ) (uint8_t rhport); | 
					
						
							| 
									
										
										
										
											2018-03-23 14:57:17 +07:00
										 |  |  | } usbd_class_driver_t; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-14 17:18:05 +07:00
										 |  |  | static usbd_class_driver_t const usbd_class_drivers[] = | 
					
						
							| 
									
										
										
										
											2014-03-12 14:01:38 +07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-05-14 13:02:01 +07:00
										 |  |  |   #if CFG_TUD_CDC
 | 
					
						
							| 
									
										
										
										
											2019-07-19 16:49:04 +07:00
										 |  |  |   { | 
					
						
							|  |  |  |       .class_code       = TUSB_CLASS_CDC, | 
					
						
							|  |  |  |       .init             = cdcd_init, | 
					
						
							| 
									
										
										
										
											2019-07-19 20:20:13 +07:00
										 |  |  |       .reset            = cdcd_reset, | 
					
						
							| 
									
										
										
										
											2019-07-19 16:49:04 +07:00
										 |  |  |       .open             = cdcd_open, | 
					
						
							|  |  |  |       .control_request  = cdcd_control_request, | 
					
						
							|  |  |  |       .control_complete = cdcd_control_complete, | 
					
						
							|  |  |  |       .xfer_cb          = cdcd_xfer_cb, | 
					
						
							| 
									
										
										
										
											2019-07-19 20:20:13 +07:00
										 |  |  |       .sof              = NULL | 
					
						
							| 
									
										
										
										
											2019-07-19 16:49:04 +07:00
										 |  |  |   }, | 
					
						
							| 
									
										
										
										
											2014-03-14 17:18:05 +07:00
										 |  |  |   #endif
 | 
					
						
							| 
									
										
										
										
											2014-03-12 14:01:38 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-14 13:02:01 +07:00
										 |  |  |   #if CFG_TUD_MSC
 | 
					
						
							| 
									
										
										
										
											2019-07-19 16:49:04 +07:00
										 |  |  |   { | 
					
						
							|  |  |  |       .class_code       = TUSB_CLASS_MSC, | 
					
						
							|  |  |  |       .init             = mscd_init, | 
					
						
							| 
									
										
										
										
											2019-07-19 20:20:13 +07:00
										 |  |  |       .reset            = mscd_reset, | 
					
						
							| 
									
										
										
										
											2019-07-19 16:49:04 +07:00
										 |  |  |       .open             = mscd_open, | 
					
						
							|  |  |  |       .control_request  = mscd_control_request, | 
					
						
							|  |  |  |       .control_complete = mscd_control_complete, | 
					
						
							|  |  |  |       .xfer_cb          = mscd_xfer_cb, | 
					
						
							| 
									
										
										
										
											2019-07-19 20:20:13 +07:00
										 |  |  |       .sof              = NULL | 
					
						
							| 
									
										
										
										
											2019-07-19 16:49:04 +07:00
										 |  |  |   }, | 
					
						
							| 
									
										
										
										
											2018-06-16 12:30:10 +07:00
										 |  |  |   #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-28 12:38:45 +07:00
										 |  |  |   #if CFG_TUD_HID
 | 
					
						
							| 
									
										
										
										
											2019-07-19 16:49:04 +07:00
										 |  |  |   { | 
					
						
							|  |  |  |       .class_code       = TUSB_CLASS_HID, | 
					
						
							|  |  |  |       .init             = hidd_init, | 
					
						
							| 
									
										
										
										
											2019-07-19 20:20:13 +07:00
										 |  |  |       .reset            = hidd_reset, | 
					
						
							| 
									
										
										
										
											2019-07-19 16:49:04 +07:00
										 |  |  |       .open             = hidd_open, | 
					
						
							|  |  |  |       .control_request  = hidd_control_request, | 
					
						
							|  |  |  |       .control_complete = hidd_control_complete, | 
					
						
							|  |  |  |       .xfer_cb          = hidd_xfer_cb, | 
					
						
							| 
									
										
										
										
											2019-07-19 20:20:13 +07:00
										 |  |  |       .sof              = NULL | 
					
						
							| 
									
										
										
										
											2019-07-19 16:49:04 +07:00
										 |  |  |   }, | 
					
						
							| 
									
										
										
										
											2018-07-13 17:48:26 +07:00
										 |  |  |   #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-16 17:03:33 -08:00
										 |  |  |   #if CFG_TUD_MIDI
 | 
					
						
							| 
									
										
										
										
											2019-07-19 16:49:04 +07:00
										 |  |  |   { | 
					
						
							|  |  |  |       .class_code       = TUSB_CLASS_AUDIO, | 
					
						
							|  |  |  |       .init             = midid_init, | 
					
						
							|  |  |  |       .open             = midid_open, | 
					
						
							| 
									
										
										
										
											2019-07-19 20:20:13 +07:00
										 |  |  |       .reset            = midid_reset, | 
					
						
							| 
									
										
										
										
											2019-07-19 16:49:04 +07:00
										 |  |  |       .control_request  = midid_control_request, | 
					
						
							|  |  |  |       .control_complete = midid_control_complete, | 
					
						
							|  |  |  |       .xfer_cb          = midid_xfer_cb, | 
					
						
							| 
									
										
										
										
											2019-07-19 20:20:13 +07:00
										 |  |  |       .sof              = NULL | 
					
						
							| 
									
										
										
										
											2019-07-19 16:49:04 +07:00
										 |  |  |   }, | 
					
						
							| 
									
										
										
										
											2018-11-16 17:03:33 -08:00
										 |  |  |   #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-19 16:23:56 +07:00
										 |  |  |   #if CFG_TUD_VENDOR
 | 
					
						
							| 
									
										
										
										
											2019-07-19 16:49:04 +07:00
										 |  |  |   { | 
					
						
							|  |  |  |       .class_code       = TUSB_CLASS_VENDOR_SPECIFIC, | 
					
						
							| 
									
										
										
										
											2019-07-19 20:20:13 +07:00
										 |  |  |       .init             = vendord_init, | 
					
						
							|  |  |  |       .reset            = vendord_reset, | 
					
						
							|  |  |  |       .open             = vendord_open, | 
					
						
							| 
									
										
										
										
											2019-07-26 11:36:30 +07:00
										 |  |  |       .control_request  = tud_vendor_control_request_cb, | 
					
						
							|  |  |  |       .control_complete = tud_vendor_control_complete_cb, | 
					
						
							| 
									
										
										
										
											2019-07-19 20:20:13 +07:00
										 |  |  |       .xfer_cb          = vendord_xfer_cb, | 
					
						
							|  |  |  |       .sof              = NULL | 
					
						
							| 
									
										
										
										
											2019-09-14 12:13:11 -04:00
										 |  |  |   }, | 
					
						
							|  |  |  |   #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   #if CFG_TUD_USBTMC
 | 
					
						
							|  |  |  |   // Presently USBTMC is the only defined class with the APP_SPECIFIC class code.
 | 
					
						
							|  |  |  |   // We maybe need to add subclass codes here, or a callback to ask if a driver can
 | 
					
						
							|  |  |  |   // handle a particular interface.
 | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2019-09-19 19:51:37 -04:00
										 |  |  |       .class_code       = TUD_USBTMC_APP_CLASS, | 
					
						
							|  |  |  |     //.subclass_code    = TUD_USBTMC_APP_SUBCLASS
 | 
					
						
							| 
									
										
										
										
											2019-09-19 20:08:45 -04:00
										 |  |  |       .init             = usbtmcd_init_cb, | 
					
						
							|  |  |  |       .reset            = usbtmcd_reset_cb, | 
					
						
							|  |  |  |       .open             = usbtmcd_open_cb, | 
					
						
							|  |  |  |       .control_request  = usbtmcd_control_request_cb, | 
					
						
							|  |  |  |       .control_complete = usbtmcd_control_complete_cb, | 
					
						
							| 
									
										
										
										
											2019-09-14 12:13:11 -04:00
										 |  |  |       .xfer_cb          = usbtmcd_xfer_cb, | 
					
						
							|  |  |  |       .sof              = NULL | 
					
						
							| 
									
										
										
										
											2019-07-19 16:49:04 +07:00
										 |  |  |   }, | 
					
						
							| 
									
										
										
										
											2014-03-14 17:18:05 +07:00
										 |  |  |   #endif
 | 
					
						
							| 
									
										
										
										
											2014-03-12 14:01:38 +07:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-13 22:16:24 +07:00
										 |  |  | enum { USBD_CLASS_DRIVER_COUNT = TU_ARRAY_SIZE(usbd_class_drivers) }; | 
					
						
							| 
									
										
										
										
											2018-04-16 13:46:28 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-12 14:01:38 +07:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							| 
									
										
										
										
											2018-05-17 19:19:55 +07:00
										 |  |  | // DCD Event
 | 
					
						
							| 
									
										
										
										
											2014-03-12 14:01:38 +07:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							| 
									
										
										
										
											2018-05-17 20:04:52 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-05 17:01:19 +07:00
										 |  |  | // Event queue
 | 
					
						
							| 
									
										
										
										
											2018-12-13 13:49:09 +07:00
										 |  |  | // OPT_MODE_DEVICE is used by OS NONE for mutex (disable usb isr)
 | 
					
						
							| 
									
										
										
										
											2018-12-05 17:01:19 +07:00
										 |  |  | OSAL_QUEUE_DEF(OPT_MODE_DEVICE, _usbd_qdef, CFG_TUD_TASK_QUEUE_SZ, dcd_event_t); | 
					
						
							| 
									
										
										
										
											2018-05-17 19:19:55 +07:00
										 |  |  | static osal_queue_t _usbd_q; | 
					
						
							| 
									
										
										
										
											2018-02-28 14:21:31 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-17 19:19:55 +07:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  | // Prototypes
 | 
					
						
							| 
									
										
										
										
											2018-05-17 19:19:55 +07:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							| 
									
										
										
										
											2018-12-10 05:15:49 +07:00
										 |  |  | static void mark_interface_endpoint(uint8_t ep2drv[8][2], uint8_t const* p_desc, uint16_t desc_len, uint8_t driver_id); | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  | static bool process_control_request(uint8_t rhport, tusb_control_request_t const * p_request); | 
					
						
							| 
									
										
										
										
											2019-05-12 19:55:15 +07:00
										 |  |  | static bool process_set_config(uint8_t rhport, uint8_t cfg_num); | 
					
						
							| 
									
										
										
										
											2019-05-11 16:31:08 +07:00
										 |  |  | static bool process_get_descriptor(uint8_t rhport, tusb_control_request_t const * p_request); | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  | void usbd_control_reset (uint8_t rhport); | 
					
						
							| 
									
										
										
										
											2018-11-23 15:14:47 +07:00
										 |  |  | bool usbd_control_xfer_cb (uint8_t rhport, uint8_t ep_addr, xfer_result_t event, uint32_t xferred_bytes); | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  | void usbd_control_set_complete_callback( bool (*fp) (uint8_t, tusb_control_request_t const * ) ); | 
					
						
							| 
									
										
										
										
											2014-03-12 14:01:38 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-23 21:18:46 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // Debugging
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							| 
									
										
										
										
											2019-10-24 12:00:06 +07:00
										 |  |  | #if CFG_TUSB_DEBUG > 1
 | 
					
						
							| 
									
										
										
										
											2019-10-23 21:18:46 +07:00
										 |  |  | static char const* const _usbd_event_str[DCD_EVENT_COUNT] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   "INVALID"        , | 
					
						
							|  |  |  |   "BUS_RESET"      , | 
					
						
							|  |  |  |   "UNPLUGGED"      , | 
					
						
							|  |  |  |   "SOF"            , | 
					
						
							|  |  |  |   "SUSPEND"        , | 
					
						
							|  |  |  |   "RESUME"         , | 
					
						
							|  |  |  |   "SETUP_RECEIVED" , | 
					
						
							|  |  |  |   "XFER_COMPLETE"  , | 
					
						
							|  |  |  |   "FUNC_CALL" | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2019-10-24 12:00:06 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  | // must be same driver order as usbd_class_drivers[]
 | 
					
						
							|  |  |  | static char const* const _usbd_driver_str[USBD_CLASS_DRIVER_COUNT] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   #if CFG_TUD_CDC
 | 
					
						
							|  |  |  |     "CDC", | 
					
						
							|  |  |  |   #endif
 | 
					
						
							|  |  |  |   #if CFG_TUD_MSC
 | 
					
						
							|  |  |  |     "MSC", | 
					
						
							|  |  |  |   #endif
 | 
					
						
							|  |  |  |   #if CFG_TUD_HID
 | 
					
						
							|  |  |  |     "HID", | 
					
						
							|  |  |  |   #endif
 | 
					
						
							|  |  |  |   #if CFG_TUD_MIDI
 | 
					
						
							|  |  |  |     "MIDI", | 
					
						
							|  |  |  |   #endif
 | 
					
						
							|  |  |  |   #if CFG_TUD_VENDOR
 | 
					
						
							|  |  |  |     "Vendor", | 
					
						
							|  |  |  |   #endif
 | 
					
						
							|  |  |  |   #if CFG_TUD_USBTMC
 | 
					
						
							|  |  |  |     "USBTMC" | 
					
						
							|  |  |  |   #endif
 | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char const* const _tusb_std_request_str[] = | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   "Get Status"        , | 
					
						
							|  |  |  |   "Clear Feature"     , | 
					
						
							|  |  |  |   "Reserved"          , | 
					
						
							|  |  |  |   "Set Feature"       , | 
					
						
							|  |  |  |   "Reserved"          , | 
					
						
							|  |  |  |   "Set Address"       , | 
					
						
							|  |  |  |   "Get Descriptor"    , | 
					
						
							|  |  |  |   "Set Descriptor"    , | 
					
						
							|  |  |  |   "Get Configuration" , | 
					
						
							|  |  |  |   "Set Configuration" , | 
					
						
							|  |  |  |   "Get Interface"     , | 
					
						
							|  |  |  |   "Set Interface"     , | 
					
						
							|  |  |  |   "Synch Frame" | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-23 21:18:46 +07:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-17 19:19:55 +07:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							| 
									
										
										
										
											2019-03-29 16:37:08 +07:00
										 |  |  | // Application API
 | 
					
						
							| 
									
										
										
										
											2018-05-17 19:19:55 +07:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							| 
									
										
										
										
											2018-07-12 22:25:06 +07:00
										 |  |  | bool tud_mounted(void) | 
					
						
							| 
									
										
										
										
											2018-05-17 19:19:55 +07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-03-30 13:48:15 +07:00
										 |  |  |   return _usbd_dev.configured; | 
					
						
							| 
									
										
										
										
											2018-05-17 19:19:55 +07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-30 23:01:23 +07:00
										 |  |  | bool tud_suspended(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return _usbd_dev.suspended; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-29 16:37:08 +07:00
										 |  |  | bool tud_remote_wakeup(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2019-03-30 17:38:00 +07:00
										 |  |  |   // only wake up host if this feature is supported and enabled and we are suspended
 | 
					
						
							|  |  |  |   TU_VERIFY (_usbd_dev.suspended && _usbd_dev.remote_wakeup_support && _usbd_dev.remote_wakeup_en ); | 
					
						
							|  |  |  |   dcd_remote_wakeup(TUD_OPT_RHPORT); | 
					
						
							|  |  |  |   return true; | 
					
						
							| 
									
										
										
										
											2019-03-29 16:37:08 +07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-31 13:12:51 +07:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  | // USBD Task
 | 
					
						
							| 
									
										
										
										
											2014-03-31 13:12:51 +07:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							| 
									
										
										
										
											2018-12-05 17:30:04 +07:00
										 |  |  | bool usbd_init (void) | 
					
						
							| 
									
										
										
										
											2014-03-31 13:12:51 +07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-10-24 12:00:06 +07:00
										 |  |  |   TU_LOG2("USBD init\r\n"); | 
					
						
							| 
									
										
										
										
											2019-10-23 21:18:46 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-30 14:34:38 +07:00
										 |  |  |   tu_varclr(&_usbd_dev); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-05 13:20:25 +07:00
										 |  |  |   // Init device queue & task
 | 
					
						
							|  |  |  |   _usbd_q = osal_queue_create(&_usbd_qdef); | 
					
						
							| 
									
										
										
										
											2018-12-05 17:30:04 +07:00
										 |  |  |   TU_ASSERT(_usbd_q != NULL); | 
					
						
							| 
									
										
										
										
											2018-12-05 13:20:25 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // Init class drivers
 | 
					
						
							| 
									
										
										
										
											2019-10-24 12:00:06 +07:00
										 |  |  |   for (uint8_t i = 0; i < USBD_CLASS_DRIVER_COUNT; i++) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     TU_LOG2("%s init\r\n", _usbd_driver_str[i]); | 
					
						
							|  |  |  |     usbd_class_drivers[i].init(); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-12-05 13:20:25 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // Init device controller driver
 | 
					
						
							| 
									
										
										
										
											2019-03-29 16:23:00 +07:00
										 |  |  |   dcd_init(TUD_OPT_RHPORT); | 
					
						
							| 
									
										
										
										
											2018-12-05 17:01:19 +07:00
										 |  |  |   dcd_int_enable(TUD_OPT_RHPORT); | 
					
						
							| 
									
										
										
										
											2018-03-02 22:46:36 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-05 17:30:04 +07:00
										 |  |  |   return true; | 
					
						
							| 
									
										
										
										
											2014-03-31 13:12:51 +07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-02 17:29:49 +07:00
										 |  |  | static void usbd_reset(uint8_t rhport) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   tu_varclr(&_usbd_dev); | 
					
						
							| 
									
										
										
										
											2019-03-30 14:34:38 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-13 22:16:24 +07:00
										 |  |  |   memset(_usbd_dev.itf2drv, DRVID_INVALID, sizeof(_usbd_dev.itf2drv)); // invalid mapping
 | 
					
						
							|  |  |  |   memset(_usbd_dev.ep2drv , DRVID_INVALID, sizeof(_usbd_dev.ep2drv )); // invalid mapping
 | 
					
						
							| 
									
										
										
										
											2018-11-14 23:39:58 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  |   usbd_control_reset(rhport); | 
					
						
							| 
									
										
										
										
											2018-11-02 17:29:49 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   for (uint8_t i = 0; i < USBD_CLASS_DRIVER_COUNT; i++) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     if ( usbd_class_drivers[i].reset ) usbd_class_drivers[i].reset( rhport ); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-13 13:49:09 +07:00
										 |  |  | /* USB Device Driver task
 | 
					
						
							|  |  |  |  * This top level thread manages all device controller event and delegates events to class-specific drivers. | 
					
						
							| 
									
										
										
										
											2018-12-13 14:51:37 +07:00
										 |  |  |  * This should be called periodically within the mainloop or rtos thread. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |    @code | 
					
						
							|  |  |  |     int main(void) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       application_init(); | 
					
						
							|  |  |  |       tusb_init(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       while(1) // the mainloop
 | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  |         application_code(); | 
					
						
							|  |  |  |         tud_task(); // tinyusb device task
 | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     @endcode | 
					
						
							| 
									
										
										
										
											2018-12-13 13:49:09 +07:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2018-12-13 14:51:37 +07:00
										 |  |  | void tud_task (void) | 
					
						
							| 
									
										
										
										
											2014-03-31 13:12:51 +07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-01-29 19:05:07 +07:00
										 |  |  |   // Skip if stack is not initialized
 | 
					
						
							|  |  |  |   if ( !tusb_inited() ) return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-28 15:56:43 +07:00
										 |  |  |   // Loop until there is no more events in the queue
 | 
					
						
							| 
									
										
										
										
											2018-11-14 16:31:28 +07:00
										 |  |  |   while (1) | 
					
						
							| 
									
										
										
										
											2018-08-28 15:56:43 +07:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  |     dcd_event_t event; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-14 17:40:29 +07:00
										 |  |  |     if ( !osal_queue_receive(_usbd_q, &event) ) return; | 
					
						
							| 
									
										
										
										
											2018-07-13 18:01:16 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-23 21:18:46 +07:00
										 |  |  |     TU_LOG2("USBD: event %s\r\n", event.event_id < DCD_EVENT_COUNT ? _usbd_event_str[event.event_id] : "CORRUPTED"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-14 17:40:29 +07:00
										 |  |  |     switch ( event.event_id ) | 
					
						
							| 
									
										
										
										
											2018-03-06 16:50:50 +07:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-03-30 02:26:15 +07:00
										 |  |  |       case DCD_EVENT_BUS_RESET: | 
					
						
							|  |  |  |         usbd_reset(event.rhport); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       case DCD_EVENT_UNPLUGGED: | 
					
						
							|  |  |  |         usbd_reset(event.rhport); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // invoke callback
 | 
					
						
							|  |  |  |         if (tud_umount_cb) tud_umount_cb(); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-14 17:40:29 +07:00
										 |  |  |       case DCD_EVENT_SETUP_RECEIVED: | 
					
						
							| 
									
										
										
										
											2019-10-24 12:00:06 +07:00
										 |  |  |         TU_LOG2("  "); | 
					
						
							| 
									
										
										
										
											2019-10-23 21:18:46 +07:00
										 |  |  |         TU_LOG2_MEM(&event.setup_received, 1, 8); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-02 01:30:01 +07:00
										 |  |  |         // Mark as connected after receiving 1st setup packet.
 | 
					
						
							|  |  |  |         // But it is easier to set it every time instead of wasting time to check then set
 | 
					
						
							|  |  |  |         _usbd_dev.connected = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-27 16:26:52 +07:00
										 |  |  |         // Process control request
 | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  |         if ( !process_control_request(event.rhport, &event.setup_received) ) | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2019-03-27 16:26:52 +07:00
										 |  |  |           // Failed -> stall both control endpoint IN and OUT
 | 
					
						
							|  |  |  |           dcd_edpt_stall(event.rhport, 0); | 
					
						
							|  |  |  |           dcd_edpt_stall(event.rhport, 0 | TUSB_DIR_IN_MASK); | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-11-14 17:40:29 +07:00
										 |  |  |       break; | 
					
						
							| 
									
										
										
										
											2018-08-28 15:56:43 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-14 17:40:29 +07:00
										 |  |  |       case DCD_EVENT_XFER_COMPLETE: | 
					
						
							| 
									
										
										
										
											2019-10-23 21:18:46 +07:00
										 |  |  |       { | 
					
						
							|  |  |  |         // Invoke the class callback associated with the endpoint address
 | 
					
						
							|  |  |  |         uint8_t const ep_addr = event.xfer_complete.ep_addr; | 
					
						
							|  |  |  |         uint8_t const epnum   = tu_edpt_number(ep_addr); | 
					
						
							|  |  |  |         uint8_t const ep_dir  = tu_edpt_dir(ep_addr); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-24 12:00:06 +07:00
										 |  |  |         TU_LOG2("  Endpoint: 0x%02X, Bytes: %ld\r\n", ep_addr, event.xfer_complete.len); | 
					
						
							| 
									
										
										
										
											2019-10-23 21:18:46 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         _usbd_dev.ep_status[epnum][ep_dir].busy = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ( 0 == epnum ) | 
					
						
							| 
									
										
										
										
											2018-08-28 15:56:43 +07:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2019-10-23 21:18:46 +07:00
										 |  |  |           // control transfer DATA stage callback
 | 
					
						
							|  |  |  |           usbd_control_xfer_cb(event.rhport, ep_addr, event.xfer_complete.result, event.xfer_complete.len); | 
					
						
							| 
									
										
										
										
											2018-08-28 15:56:43 +07:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-10-23 21:18:46 +07:00
										 |  |  |         else | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           uint8_t const drv_id = _usbd_dev.ep2drv[epnum][ep_dir]; | 
					
						
							|  |  |  |           TU_ASSERT(drv_id < USBD_CLASS_DRIVER_COUNT,); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-24 12:00:06 +07:00
										 |  |  |           TU_LOG2("  %s xfer callback\r\n", _usbd_driver_str[drv_id]); | 
					
						
							| 
									
										
										
										
											2019-10-23 21:18:46 +07:00
										 |  |  |           usbd_class_drivers[drv_id].xfer_cb(event.rhport, ep_addr, event.xfer_complete.result, event.xfer_complete.len); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2018-11-14 17:40:29 +07:00
										 |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-30 02:26:15 +07:00
										 |  |  |       case DCD_EVENT_SUSPEND: | 
					
						
							| 
									
										
										
										
											2019-04-02 01:30:01 +07:00
										 |  |  |         if (tud_suspend_cb) tud_suspend_cb(_usbd_dev.remote_wakeup_en); | 
					
						
							| 
									
										
										
										
											2018-11-14 17:40:29 +07:00
										 |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-30 02:26:15 +07:00
										 |  |  |       case DCD_EVENT_RESUME: | 
					
						
							|  |  |  |         if (tud_resume_cb) tud_resume_cb(); | 
					
						
							| 
									
										
										
										
											2018-11-14 17:40:29 +07:00
										 |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       case DCD_EVENT_SOF: | 
					
						
							|  |  |  |         for ( uint8_t i = 0; i < USBD_CLASS_DRIVER_COUNT; i++ ) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           if ( usbd_class_drivers[i].sof ) | 
					
						
							|  |  |  |           { | 
					
						
							|  |  |  |             usbd_class_drivers[i].sof(event.rhport); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-30 02:26:15 +07:00
										 |  |  |       case USBD_EVENT_FUNC_CALL: | 
					
						
							| 
									
										
										
										
											2018-11-14 17:40:29 +07:00
										 |  |  |         if ( event.func_call.func ) event.func_call.func(event.func_call.param); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       default: | 
					
						
							|  |  |  |         TU_BREAKPOINT(); | 
					
						
							|  |  |  |       break; | 
					
						
							| 
									
										
										
										
											2018-08-28 15:56:43 +07:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-03-31 13:12:51 +07:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2018-11-14 17:40:29 +07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2014-03-31 13:12:51 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // Control Request Parser & Handling
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // This handles the actual request and its response.
 | 
					
						
							|  |  |  | // return false will cause its caller to stall control endpoint
 | 
					
						
							|  |  |  | static bool process_control_request(uint8_t rhport, tusb_control_request_t const * p_request) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   usbd_control_set_complete_callback(NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-16 11:13:17 +07:00
										 |  |  |   TU_ASSERT(p_request->bmRequestType_bit.type < TUSB_REQ_TYPE_INVALID); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-12 22:03:40 +07:00
										 |  |  |   // Vendor request
 | 
					
						
							| 
									
										
										
										
											2019-07-16 18:14:47 +07:00
										 |  |  |   if ( p_request->bmRequestType_bit.type == TUSB_REQ_TYPE_VENDOR ) | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2019-07-24 09:45:32 +07:00
										 |  |  |     TU_VERIFY(tud_vendor_control_request_cb); | 
					
						
							| 
									
										
										
										
											2019-07-16 18:14:47 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-24 09:45:32 +07:00
										 |  |  |     if (tud_vendor_control_complete_cb) usbd_control_set_complete_callback(tud_vendor_control_complete_cb); | 
					
						
							|  |  |  |     return tud_vendor_control_request_cb(rhport, p_request); | 
					
						
							| 
									
										
										
										
											2019-07-16 18:14:47 +07:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-07-12 22:03:40 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-24 12:00:06 +07:00
										 |  |  | #if CFG_TUSB_DEBUG > 1
 | 
					
						
							|  |  |  |   if (TUSB_REQ_TYPE_STANDARD == p_request->bmRequestType_bit.type && p_request->bRequest <= TUSB_REQ_SYNCH_FRAME) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     TU_LOG2("  %s\r\n", _tusb_std_request_str[p_request->bRequest]); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-29 16:55:58 +07:00
										 |  |  |   switch ( p_request->bmRequestType_bit.recipient ) | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2019-03-29 16:55:58 +07:00
										 |  |  |     //------------- Device Requests e.g in enumeration -------------//
 | 
					
						
							|  |  |  |     case TUSB_REQ_RCPT_DEVICE: | 
					
						
							|  |  |  |       if ( TUSB_REQ_TYPE_STANDARD != p_request->bmRequestType_bit.type ) | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  |         // Non standard request is not supported
 | 
					
						
							|  |  |  |         TU_BREAKPOINT(); | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-29 16:55:58 +07:00
										 |  |  |       switch ( p_request->bRequest ) | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  |       { | 
					
						
							| 
									
										
										
										
											2019-03-29 16:55:58 +07:00
										 |  |  |         case TUSB_REQ_SET_ADDRESS: | 
					
						
							| 
									
										
										
										
											2019-03-30 14:34:38 +07:00
										 |  |  |           // Depending on mcu, status phase could be sent either before or after changing device address
 | 
					
						
							|  |  |  |           // Therefore DCD must include zero-length status response
 | 
					
						
							| 
									
										
										
										
											2019-03-29 16:55:58 +07:00
										 |  |  |           dcd_set_address(rhport, (uint8_t) p_request->wValue); | 
					
						
							| 
									
										
										
										
											2019-03-30 14:34:38 +07:00
										 |  |  |           return true; // skip status
 | 
					
						
							| 
									
										
										
										
											2019-03-29 16:55:58 +07:00
										 |  |  |         break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case TUSB_REQ_GET_CONFIGURATION: | 
					
						
							| 
									
										
										
										
											2019-03-30 14:34:38 +07:00
										 |  |  |         { | 
					
						
							|  |  |  |           uint8_t cfgnum = _usbd_dev.configured ? 1 : 0; | 
					
						
							| 
									
										
										
										
											2019-07-16 18:14:47 +07:00
										 |  |  |           tud_control_xfer(rhport, p_request, &cfgnum, 1); | 
					
						
							| 
									
										
										
										
											2019-03-30 14:34:38 +07:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-03-29 16:55:58 +07:00
										 |  |  |         break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         case TUSB_REQ_SET_CONFIGURATION: | 
					
						
							|  |  |  |         { | 
					
						
							| 
									
										
										
										
											2019-03-30 13:48:15 +07:00
										 |  |  |           uint8_t const cfg_num = (uint8_t) p_request->wValue; | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-30 13:48:15 +07:00
										 |  |  |           dcd_set_config(rhport, cfg_num); | 
					
						
							|  |  |  |           _usbd_dev.configured = cfg_num ? 1 : 0; | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-12 19:55:15 +07:00
										 |  |  |           if ( cfg_num ) TU_ASSERT( process_set_config(rhport, cfg_num) ); | 
					
						
							| 
									
										
										
										
											2019-07-16 18:14:47 +07:00
										 |  |  |           tud_control_status(rhport, p_request); | 
					
						
							| 
									
										
										
										
											2019-03-29 16:55:58 +07:00
										 |  |  |         } | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-29 16:55:58 +07:00
										 |  |  |         case TUSB_REQ_GET_DESCRIPTOR: | 
					
						
							| 
									
										
										
										
											2019-05-15 12:57:31 +07:00
										 |  |  |           TU_VERIFY( process_get_descriptor(rhport, p_request) ); | 
					
						
							| 
									
										
										
										
											2019-03-29 16:55:58 +07:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-29 16:55:58 +07:00
										 |  |  |         case TUSB_REQ_SET_FEATURE: | 
					
						
							| 
									
										
										
										
											2019-03-30 14:34:38 +07:00
										 |  |  |           // Only support remote wakeup for device feature
 | 
					
						
							|  |  |  |           TU_VERIFY(TUSB_REQ_FEATURE_REMOTE_WAKEUP == p_request->wValue); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           // Host may enable remote wake up before suspending especially HID device
 | 
					
						
							|  |  |  |           _usbd_dev.remote_wakeup_en = true; | 
					
						
							| 
									
										
										
										
											2019-07-16 18:14:47 +07:00
										 |  |  |           tud_control_status(rhport, p_request); | 
					
						
							| 
									
										
										
										
											2019-03-29 16:55:58 +07:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2019-03-29 16:23:00 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-29 16:55:58 +07:00
										 |  |  |         case TUSB_REQ_CLEAR_FEATURE: | 
					
						
							| 
									
										
										
										
											2019-03-30 14:34:38 +07:00
										 |  |  |           // Only support remote wakeup for device feature
 | 
					
						
							|  |  |  |           TU_VERIFY(TUSB_REQ_FEATURE_REMOTE_WAKEUP == p_request->wValue); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           // Host may disable remote wake up after resuming
 | 
					
						
							|  |  |  |           _usbd_dev.remote_wakeup_en = false; | 
					
						
							| 
									
										
										
										
											2019-07-16 18:14:47 +07:00
										 |  |  |           tud_control_status(rhport, p_request); | 
					
						
							| 
									
										
										
										
											2019-03-29 16:55:58 +07:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2019-03-29 16:23:00 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-30 13:48:15 +07:00
										 |  |  |         case TUSB_REQ_GET_STATUS: | 
					
						
							| 
									
										
										
										
											2019-03-30 14:34:38 +07:00
										 |  |  |         { | 
					
						
							|  |  |  |           // Device status bit mask
 | 
					
						
							|  |  |  |           // - Bit 0: Self Powered
 | 
					
						
							|  |  |  |           // - Bit 1: Remote Wakeup enabled
 | 
					
						
							|  |  |  |           uint16_t status = (_usbd_dev.self_powered ? 1 : 0) | (_usbd_dev.remote_wakeup_en ? 2 : 0); | 
					
						
							| 
									
										
										
										
											2019-07-16 18:14:47 +07:00
										 |  |  |           tud_control_xfer(rhport, p_request, &status, 2); | 
					
						
							| 
									
										
										
										
											2019-03-30 14:34:38 +07:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-03-30 13:48:15 +07:00
										 |  |  |         break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-29 16:55:58 +07:00
										 |  |  |         // Unknown/Unsupported request
 | 
					
						
							|  |  |  |         default: TU_BREAKPOINT(); return false; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     break; | 
					
						
							| 
									
										
										
										
											2018-11-08 13:45:30 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  |     //------------- Class/Interface Specific Request -------------//
 | 
					
						
							| 
									
										
										
										
											2019-03-29 16:55:58 +07:00
										 |  |  |     case TUSB_REQ_RCPT_INTERFACE: | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       uint8_t const itf = tu_u16_low(p_request->wIndex); | 
					
						
							| 
									
										
										
										
											2019-09-13 22:16:24 +07:00
										 |  |  |       TU_VERIFY(itf < TU_ARRAY_SIZE(_usbd_dev.itf2drv)); | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-13 14:03:59 +07:00
										 |  |  |       uint8_t const drvid = _usbd_dev.itf2drv[itf]; | 
					
						
							| 
									
										
										
										
											2019-03-29 16:55:58 +07:00
										 |  |  |       TU_VERIFY(drvid < USBD_CLASS_DRIVER_COUNT); | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-13 13:03:11 +07:00
										 |  |  |       if (p_request->bmRequestType_bit.type == TUSB_REQ_TYPE_STANDARD) | 
					
						
							| 
									
										
										
										
											2019-07-26 20:14:56 +07:00
										 |  |  |       { | 
					
						
							| 
									
										
										
										
											2019-09-13 13:03:11 +07:00
										 |  |  |         switch ( p_request->bRequest ) | 
					
						
							| 
									
										
										
										
											2019-07-26 20:14:56 +07:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2019-09-13 13:03:11 +07:00
										 |  |  |           case TUSB_REQ_GET_INTERFACE: | 
					
						
							|  |  |  |           { | 
					
						
							|  |  |  |             // TODO not support alternate interface yet
 | 
					
						
							|  |  |  |             uint8_t alternate = 0; | 
					
						
							|  |  |  |             tud_control_xfer(rhport, p_request, &alternate, 1); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           break; | 
					
						
							| 
									
										
										
										
											2019-07-26 20:14:56 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-13 13:03:11 +07:00
										 |  |  |           case TUSB_REQ_SET_INTERFACE: | 
					
						
							|  |  |  |           { | 
					
						
							|  |  |  |             uint8_t const alternate = (uint8_t) p_request->wValue; | 
					
						
							| 
									
										
										
										
											2019-03-29 16:55:58 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-13 13:03:11 +07:00
										 |  |  |             // TODO not support alternate interface yet
 | 
					
						
							|  |  |  |             TU_ASSERT(alternate == 0); | 
					
						
							|  |  |  |             tud_control_status(rhport, p_request); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           break; | 
					
						
							| 
									
										
										
										
											2019-07-26 20:14:56 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-13 13:03:11 +07:00
										 |  |  |           default: | 
					
						
							|  |  |  |             // forward to class driver: "STD request to Interface"
 | 
					
						
							|  |  |  |             // GET HID REPORT DESCRIPTOR falls into this case
 | 
					
						
							|  |  |  |             // stall control endpoint if driver return false
 | 
					
						
							| 
									
										
										
										
											2019-09-13 22:16:24 +07:00
										 |  |  |             usbd_control_set_complete_callback(usbd_class_drivers[drvid].control_complete); | 
					
						
							| 
									
										
										
										
											2019-10-24 12:00:06 +07:00
										 |  |  |             TU_LOG2("  %s control request\r\n", _usbd_driver_str[drvid]); | 
					
						
							| 
									
										
										
										
											2019-09-16 11:13:17 +07:00
										 |  |  |             TU_ASSERT(usbd_class_drivers[drvid].control_request != NULL && | 
					
						
							|  |  |  |                       usbd_class_drivers[drvid].control_request(rhport, p_request)); | 
					
						
							| 
									
										
										
										
											2019-09-13 13:03:11 +07:00
										 |  |  |           break; | 
					
						
							| 
									
										
										
										
											2019-07-26 20:14:56 +07:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-09-13 13:03:11 +07:00
										 |  |  |       }else | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  |         // forward to class driver: "non-STD request to Interface"
 | 
					
						
							|  |  |  |         // stall control endpoint if driver return false
 | 
					
						
							| 
									
										
										
										
											2019-09-13 22:16:24 +07:00
										 |  |  |         usbd_control_set_complete_callback(usbd_class_drivers[drvid].control_complete); | 
					
						
							| 
									
										
										
										
											2019-10-24 12:00:06 +07:00
										 |  |  |         TU_LOG2("  %s control request\r\n", _usbd_driver_str[drvid]); | 
					
						
							| 
									
										
										
										
											2019-09-16 11:13:17 +07:00
										 |  |  |         TU_ASSERT(usbd_class_drivers[drvid].control_request != NULL && | 
					
						
							|  |  |  |                   usbd_class_drivers[drvid].control_request(rhport, p_request)); | 
					
						
							| 
									
										
										
										
											2019-07-26 20:14:56 +07:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-03-29 16:55:58 +07:00
										 |  |  |     } | 
					
						
							|  |  |  |     break; | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     //------------- Endpoint Request -------------//
 | 
					
						
							| 
									
										
										
										
											2019-03-29 16:55:58 +07:00
										 |  |  |     case TUSB_REQ_RCPT_ENDPOINT: | 
					
						
							| 
									
										
										
										
											2019-09-13 22:16:24 +07:00
										 |  |  |     { | 
					
						
							|  |  |  |       uint8_t const ep_addr = tu_u16_low(p_request->wIndex); | 
					
						
							|  |  |  |       uint8_t const ep_num  = tu_edpt_number(ep_addr); | 
					
						
							|  |  |  |       uint8_t const ep_dir  = tu_edpt_dir(ep_addr); | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-13 22:16:24 +07:00
										 |  |  |       TU_ASSERT(ep_num < TU_ARRAY_SIZE(_usbd_dev.ep2drv) ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       uint8_t const drv_id = _usbd_dev.ep2drv[ep_num][ep_dir]; | 
					
						
							|  |  |  |       TU_ASSERT(drv_id < USBD_CLASS_DRIVER_COUNT); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-16 11:13:17 +07:00
										 |  |  |       bool ret = false; | 
					
						
							| 
									
										
										
										
											2019-09-13 22:16:24 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |       if ( TUSB_REQ_TYPE_STANDARD != p_request->bmRequestType_bit.type ) | 
					
						
							| 
									
										
										
										
											2019-03-29 16:55:58 +07:00
										 |  |  |       { | 
					
						
							| 
									
										
										
										
											2019-09-13 22:16:24 +07:00
										 |  |  |         // complete callback is also capable of stalling/acking the request
 | 
					
						
							|  |  |  |         usbd_control_set_complete_callback(usbd_class_drivers[drv_id].control_complete); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-13 22:16:24 +07:00
										 |  |  |       // Then handle if it is standard request
 | 
					
						
							|  |  |  |       if ( TUSB_REQ_TYPE_STANDARD == p_request->bmRequestType_bit.type ) | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  |         // force return true for standard request
 | 
					
						
							|  |  |  |         ret = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         switch ( p_request->bRequest ) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           case TUSB_REQ_GET_STATUS: | 
					
						
							| 
									
										
										
										
											2019-03-29 16:55:58 +07:00
										 |  |  |           { | 
					
						
							| 
									
										
										
										
											2019-09-13 22:16:24 +07:00
										 |  |  |             uint16_t status = usbd_edpt_stalled(rhport, ep_addr) ? 0x0001 : 0x0000; | 
					
						
							|  |  |  |             tud_control_xfer(rhport, p_request, &status, 2); | 
					
						
							| 
									
										
										
										
											2019-03-29 16:55:58 +07:00
										 |  |  |           } | 
					
						
							| 
									
										
										
										
											2019-09-13 22:16:24 +07:00
										 |  |  |           break; | 
					
						
							| 
									
										
										
										
											2019-03-29 16:55:58 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-13 22:16:24 +07:00
										 |  |  |           case TUSB_REQ_CLEAR_FEATURE: | 
					
						
							|  |  |  |             if ( TUSB_REQ_FEATURE_EDPT_HALT == p_request->wValue ) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |               usbd_edpt_clear_stall(rhport, ep_addr); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             tud_control_status(rhport, p_request); | 
					
						
							|  |  |  |           break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           case TUSB_REQ_SET_FEATURE: | 
					
						
							|  |  |  |             if ( TUSB_REQ_FEATURE_EDPT_HALT == p_request->wValue ) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |               usbd_edpt_stall(rhport, ep_addr); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             tud_control_status(rhport, p_request); | 
					
						
							|  |  |  |           break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           // Unknown/Unsupported request
 | 
					
						
							|  |  |  |           default: TU_BREAKPOINT(); return false; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-03-29 16:55:58 +07:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-09-21 12:02:06 -04:00
										 |  |  |       // Some classes such as TMC needs to clear/re-init its buffer when receiving CLEAR_FEATURE request
 | 
					
						
							|  |  |  |       // We will forward all request targeted endpoint to its class driver
 | 
					
						
							|  |  |  |       // For class-type requests:  must (call tud_control_status(); return true) or (return false)
 | 
					
						
							| 
									
										
										
										
											2019-09-21 12:17:17 -04:00
										 |  |  |       // For std-type requests:    non-std request codes are already discarded.
 | 
					
						
							| 
									
										
										
										
											2019-09-21 12:02:06 -04:00
										 |  |  |       //                           must not call tud_control_status(), and return value will have no effect
 | 
					
						
							| 
									
										
										
										
											2019-09-21 12:17:17 -04:00
										 |  |  |       // class driver is invoked last, so that EP already has EP stall cleared (in event of clear feature EP halt)
 | 
					
						
							| 
									
										
										
										
											2019-10-24 12:00:06 +07:00
										 |  |  |       TU_LOG2("  %s control request\r\n", _usbd_driver_str[drv_id]); | 
					
						
							| 
									
										
										
										
											2019-09-21 19:29:57 -04:00
										 |  |  |       if ( usbd_class_drivers[drv_id].control_request && | 
					
						
							|  |  |  |            usbd_class_drivers[drv_id].control_request(rhport, p_request)) | 
					
						
							| 
									
										
										
										
											2019-09-21 12:02:06 -04:00
										 |  |  |       { | 
					
						
							| 
									
										
										
										
											2019-09-21 19:29:57 -04:00
										 |  |  |         ret = true; | 
					
						
							| 
									
										
										
										
											2019-09-21 12:02:06 -04:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-09-13 22:16:24 +07:00
										 |  |  |       return ret; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-03-29 16:55:58 +07:00
										 |  |  |     break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Unknown recipient
 | 
					
						
							|  |  |  |     default: TU_BREAKPOINT(); return false; | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return true; | 
					
						
							| 
									
										
										
										
											2014-03-12 14:01:38 +07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-13 17:48:26 +07:00
										 |  |  | // Process Set Configure Request
 | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  | // This function parse configuration descriptor & open drivers accordingly
 | 
					
						
							| 
									
										
										
										
											2019-05-12 19:55:15 +07:00
										 |  |  | static bool process_set_config(uint8_t rhport, uint8_t cfg_num) | 
					
						
							| 
									
										
										
										
											2014-03-12 14:01:38 +07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-05-12 19:55:15 +07:00
										 |  |  |   tusb_desc_configuration_t const * desc_cfg = (tusb_desc_configuration_t const *) tud_descriptor_configuration_cb(cfg_num-1); // index is cfg_num-1
 | 
					
						
							| 
									
										
										
										
											2019-03-30 17:38:00 +07:00
										 |  |  |   TU_ASSERT(desc_cfg != NULL && desc_cfg->bDescriptorType == TUSB_DESC_CONFIGURATION); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Parse configuration descriptor
 | 
					
						
							|  |  |  |   _usbd_dev.remote_wakeup_support = (desc_cfg->bmAttributes & TUSB_DESC_CONFIG_ATT_REMOTE_WAKEUP) ? 1 : 0; | 
					
						
							|  |  |  |   _usbd_dev.self_powered = (desc_cfg->bmAttributes & TUSB_DESC_CONFIG_ATT_SELF_POWERED) ? 1 : 0; | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-30 17:38:00 +07:00
										 |  |  |   // Parse interface descriptor
 | 
					
						
							|  |  |  |   uint8_t const * p_desc   = ((uint8_t const*) desc_cfg) + sizeof(tusb_desc_configuration_t); | 
					
						
							|  |  |  |   uint8_t const * desc_end = ((uint8_t const*) desc_cfg) + desc_cfg->wTotalLength; | 
					
						
							| 
									
										
										
										
											2014-03-23 15:39:55 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-30 17:38:00 +07:00
										 |  |  |   while( p_desc < desc_end ) | 
					
						
							| 
									
										
										
										
											2014-03-12 14:01:38 +07:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  |     // Each interface always starts with Interface or Association descriptor
 | 
					
						
							| 
									
										
										
										
											2018-12-12 12:01:15 +07:00
										 |  |  |     if ( TUSB_DESC_INTERFACE_ASSOCIATION == tu_desc_type(p_desc) ) | 
					
						
							| 
									
										
										
										
											2014-03-12 14:01:38 +07:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2018-12-12 12:01:15 +07:00
										 |  |  |       p_desc = tu_desc_next(p_desc); // ignore Interface Association
 | 
					
						
							| 
									
										
										
										
											2014-03-12 14:01:38 +07:00
										 |  |  |     }else | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2018-12-12 12:01:15 +07:00
										 |  |  |       TU_ASSERT( TUSB_DESC_INTERFACE == tu_desc_type(p_desc) ); | 
					
						
							| 
									
										
										
										
											2014-03-12 14:01:38 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  |       tusb_desc_interface_t* desc_itf = (tusb_desc_interface_t*) p_desc; | 
					
						
							| 
									
										
										
										
											2014-03-12 14:01:38 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-14 13:02:01 +07:00
										 |  |  |       // Check if class is supported
 | 
					
						
							| 
									
										
										
										
											2018-07-13 17:48:26 +07:00
										 |  |  |       uint8_t drv_id; | 
					
						
							|  |  |  |       for (drv_id = 0; drv_id < USBD_CLASS_DRIVER_COUNT; drv_id++) | 
					
						
							| 
									
										
										
										
											2018-05-14 13:02:01 +07:00
										 |  |  |       { | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  |         if ( usbd_class_drivers[drv_id].class_code == desc_itf->bInterfaceClass ) break; | 
					
						
							| 
									
										
										
										
											2018-05-14 13:02:01 +07:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2018-12-10 05:07:22 +07:00
										 |  |  |       TU_ASSERT( drv_id < USBD_CLASS_DRIVER_COUNT ); | 
					
						
							| 
									
										
										
										
											2014-03-12 14:01:38 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  |       // Interface number must not be used already TODO alternate interface
 | 
					
						
							| 
									
										
										
										
											2019-09-13 22:16:24 +07:00
										 |  |  |       TU_ASSERT( DRVID_INVALID == _usbd_dev.itf2drv[desc_itf->bInterfaceNumber] ); | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  |       _usbd_dev.itf2drv[desc_itf->bInterfaceNumber] = drv_id; | 
					
						
							| 
									
										
										
										
											2018-07-13 17:48:26 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-10 05:07:22 +07:00
										 |  |  |       uint16_t itf_len=0; | 
					
						
							| 
									
										
										
										
											2019-10-24 12:00:06 +07:00
										 |  |  |       TU_LOG2("  %s open\r\n", _usbd_driver_str[drv_id]); | 
					
						
							| 
									
										
										
										
											2019-07-24 23:07:30 +07:00
										 |  |  |       TU_ASSERT( usbd_class_drivers[drv_id].open(rhport, desc_itf, &itf_len) ); | 
					
						
							| 
									
										
										
										
											2018-12-10 05:07:22 +07:00
										 |  |  |       TU_ASSERT( itf_len >= sizeof(tusb_desc_interface_t) ); | 
					
						
							| 
									
										
										
										
											2014-03-12 14:01:38 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-10 05:15:49 +07:00
										 |  |  |       mark_interface_endpoint(_usbd_dev.ep2drv, p_desc, itf_len, drv_id); | 
					
						
							| 
									
										
										
										
											2014-03-12 14:01:38 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-10 05:07:22 +07:00
										 |  |  |       p_desc += itf_len; // next interface
 | 
					
						
							| 
									
										
										
										
											2014-03-12 14:01:38 +07:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-07 16:39:33 +07:00
										 |  |  |   // invoke callback
 | 
					
						
							| 
									
										
										
										
											2018-12-10 19:25:57 +07:00
										 |  |  |   if (tud_mount_cb) tud_mount_cb(); | 
					
						
							| 
									
										
										
										
											2018-03-07 16:39:33 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-29 16:55:58 +07:00
										 |  |  |   return true; | 
					
						
							| 
									
										
										
										
											2014-03-12 14:01:38 +07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-13 17:48:26 +07:00
										 |  |  | // Helper marking endpoint of interface belongs to class driver
 | 
					
						
							| 
									
										
										
										
											2018-12-10 05:15:49 +07:00
										 |  |  | static void mark_interface_endpoint(uint8_t ep2drv[8][2], uint8_t const* p_desc, uint16_t desc_len, uint8_t driver_id) | 
					
						
							| 
									
										
										
										
											2018-07-13 17:48:26 +07:00
										 |  |  | { | 
					
						
							|  |  |  |   uint16_t len = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   while( len < desc_len ) | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2018-12-12 12:01:15 +07:00
										 |  |  |     if ( TUSB_DESC_ENDPOINT == tu_desc_type(p_desc) ) | 
					
						
							| 
									
										
										
										
											2018-07-13 17:48:26 +07:00
										 |  |  |     { | 
					
						
							|  |  |  |       uint8_t const ep_addr = ((tusb_desc_endpoint_t const*) p_desc)->bEndpointAddress; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-12 11:51:31 +07:00
										 |  |  |       ep2drv[tu_edpt_number(ep_addr)][tu_edpt_dir(ep_addr)] = driver_id; | 
					
						
							| 
									
										
										
										
											2018-07-13 17:48:26 +07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-21 12:03:36 -04:00
										 |  |  |     len   = (uint16_t)(len + tu_desc_len(p_desc)); | 
					
						
							| 
									
										
										
										
											2018-12-12 12:01:15 +07:00
										 |  |  |     p_desc = tu_desc_next(p_desc); | 
					
						
							| 
									
										
										
										
											2018-07-13 17:48:26 +07:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  | // return descriptor's buffer and update desc_len
 | 
					
						
							| 
									
										
										
										
											2019-05-11 16:31:08 +07:00
										 |  |  | static bool process_get_descriptor(uint8_t rhport, tusb_control_request_t const * p_request) | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  | { | 
					
						
							|  |  |  |   tusb_desc_type_t const desc_type = (tusb_desc_type_t) tu_u16_high(p_request->wValue); | 
					
						
							|  |  |  |   uint8_t const desc_index = tu_u16_low( p_request->wValue ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   switch(desc_type) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     case TUSB_DESC_DEVICE: | 
					
						
							| 
									
										
										
										
											2019-07-16 18:14:47 +07:00
										 |  |  |       return tud_control_xfer(rhport, p_request, (void*) tud_descriptor_device_cb(), sizeof(tusb_desc_device_t)); | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  |     break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-12 00:12:14 +07:00
										 |  |  |     case TUSB_DESC_BOS: | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       // requested by host if USB > 2.0 ( i.e 2.1 or 3.x )
 | 
					
						
							|  |  |  |       if (!tud_descriptor_bos_cb) return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       tusb_desc_bos_t const* desc_bos = (tusb_desc_bos_t const*) tud_descriptor_bos_cb(); | 
					
						
							|  |  |  |       uint16_t total_len; | 
					
						
							|  |  |  |       memcpy(&total_len, &desc_bos->wTotalLength, 2); // possibly mis-aligned memory
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-16 18:14:47 +07:00
										 |  |  |       return tud_control_xfer(rhport, p_request, (void*) desc_bos, total_len); | 
					
						
							| 
									
										
										
										
											2019-07-12 00:12:14 +07:00
										 |  |  |     } | 
					
						
							|  |  |  |     break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  |     case TUSB_DESC_CONFIGURATION: | 
					
						
							| 
									
										
										
										
											2019-05-12 14:09:35 +07:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-05-12 19:55:15 +07:00
										 |  |  |       tusb_desc_configuration_t const* desc_config = (tusb_desc_configuration_t const*) tud_descriptor_configuration_cb(desc_index); | 
					
						
							| 
									
										
										
										
											2019-07-12 00:12:14 +07:00
										 |  |  |       uint16_t total_len; | 
					
						
							|  |  |  |       memcpy(&total_len, &desc_config->wTotalLength, 2); // possibly mis-aligned memory
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-16 18:14:47 +07:00
										 |  |  |       return tud_control_xfer(rhport, p_request, (void*) desc_config, total_len); | 
					
						
							| 
									
										
										
										
											2019-05-12 14:09:35 +07:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  |     break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case TUSB_DESC_STRING: | 
					
						
							|  |  |  |       // String Descriptor always uses the desc set from user
 | 
					
						
							| 
									
										
										
										
											2019-05-11 16:31:08 +07:00
										 |  |  |       if ( desc_index == 0xEE ) | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  |       { | 
					
						
							| 
									
										
										
										
											2019-07-12 14:53:11 +07:00
										 |  |  |         // The 0xEE index string is a Microsoft OS Descriptors.
 | 
					
						
							|  |  |  |         // https://docs.microsoft.com/en-us/windows-hardware/drivers/usbcon/microsoft-defined-usb-descriptors
 | 
					
						
							| 
									
										
										
										
											2019-05-11 16:31:08 +07:00
										 |  |  |         return false; | 
					
						
							|  |  |  |       }else | 
					
						
							|  |  |  |       { | 
					
						
							| 
									
										
										
										
											2019-05-12 15:38:15 +07:00
										 |  |  |         uint8_t const* desc_str = (uint8_t const*) tud_descriptor_string_cb(desc_index); | 
					
						
							|  |  |  |         TU_ASSERT(desc_str); | 
					
						
							| 
									
										
										
										
											2019-05-11 16:31:08 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-12 15:38:15 +07:00
										 |  |  |         // first byte of descriptor is its size
 | 
					
						
							| 
									
										
										
										
											2019-07-16 18:14:47 +07:00
										 |  |  |         return tud_control_xfer(rhport, p_request, (void*) desc_str, desc_str[0]); | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  |       } | 
					
						
							|  |  |  |     break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case TUSB_DESC_DEVICE_QUALIFIER: | 
					
						
							|  |  |  |       // TODO If not highspeed capable stall this request otherwise
 | 
					
						
							|  |  |  |       // return the descriptor that could work in highspeed
 | 
					
						
							| 
									
										
										
										
											2019-05-11 16:31:08 +07:00
										 |  |  |       return false; | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  |     break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-11 16:31:08 +07:00
										 |  |  |     default: return false; | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-11 16:31:08 +07:00
										 |  |  |   return true; | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-12 14:01:38 +07:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  | // DCD Event Handler
 | 
					
						
							| 
									
										
										
										
											2014-03-12 14:01:38 +07:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							| 
									
										
										
										
											2018-10-23 15:08:31 +07:00
										 |  |  | void dcd_event_handler(dcd_event_t const * event, bool in_isr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   switch (event->event_id) | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2018-10-23 15:12:30 +07:00
										 |  |  |     case DCD_EVENT_BUS_RESET: | 
					
						
							| 
									
										
										
										
											2019-03-27 23:58:24 +07:00
										 |  |  |       osal_queue_send(_usbd_q, event, in_isr); | 
					
						
							|  |  |  |     break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-23 15:12:30 +07:00
										 |  |  |     case DCD_EVENT_UNPLUGGED: | 
					
						
							| 
									
										
										
										
											2019-03-30 02:26:15 +07:00
										 |  |  |       _usbd_dev.connected = 0; | 
					
						
							| 
									
										
										
										
											2019-03-30 13:48:15 +07:00
										 |  |  |       _usbd_dev.configured = 0; | 
					
						
							| 
									
										
										
										
											2019-03-30 23:01:23 +07:00
										 |  |  |       _usbd_dev.suspended = 0; | 
					
						
							| 
									
										
										
										
											2018-11-02 17:29:49 +07:00
										 |  |  |       osal_queue_send(_usbd_q, event, in_isr); | 
					
						
							| 
									
										
										
										
											2018-10-23 15:08:31 +07:00
										 |  |  |     break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-20 17:25:41 +07:00
										 |  |  |     case DCD_EVENT_SOF: | 
					
						
							|  |  |  |       // nothing to do now
 | 
					
						
							|  |  |  |     break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-29 01:34:53 +07:00
										 |  |  |     case DCD_EVENT_SUSPEND: | 
					
						
							| 
									
										
										
										
											2019-04-02 01:30:01 +07:00
										 |  |  |       // NOTE: When plugging/unplugging device, the D+/D- state are unstable and can accidentally meet the
 | 
					
						
							| 
									
										
										
										
											2019-05-29 16:55:15 +07:00
										 |  |  |       // SUSPEND condition ( Idle for 3ms ). Some MCUs such as SAMD doesn't distinguish suspend vs disconnect as well.
 | 
					
						
							| 
									
										
										
										
											2019-04-02 01:30:01 +07:00
										 |  |  |       // We will skip handling SUSPEND/RESUME event if not currently connected
 | 
					
						
							|  |  |  |       if ( _usbd_dev.connected ) | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  |         _usbd_dev.suspended = 1; | 
					
						
							|  |  |  |         osal_queue_send(_usbd_q, event, in_isr); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2019-03-30 02:26:15 +07:00
										 |  |  |     break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-23 15:12:30 +07:00
										 |  |  |     case DCD_EVENT_RESUME: | 
					
						
							| 
									
										
										
										
											2019-04-02 01:30:01 +07:00
										 |  |  |       if ( _usbd_dev.connected ) | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  |         _usbd_dev.suspended = 0; | 
					
						
							|  |  |  |         osal_queue_send(_usbd_q, event, in_isr); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2018-10-23 15:08:31 +07:00
										 |  |  |     break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-23 16:07:48 +07:00
										 |  |  |     case DCD_EVENT_SETUP_RECEIVED: | 
					
						
							|  |  |  |       osal_queue_send(_usbd_q, event, in_isr); | 
					
						
							|  |  |  |     break; | 
					
						
							| 
									
										
										
										
											2018-10-23 15:08:31 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-24 00:44:26 +07:00
										 |  |  |     case DCD_EVENT_XFER_COMPLETE: | 
					
						
							| 
									
										
										
										
											2019-05-29 16:55:15 +07:00
										 |  |  |       // skip zero-length control status complete event, should DCD notify us.
 | 
					
						
							|  |  |  |       // TODO could cause issue with actual zero length data used by class such as DFU
 | 
					
						
							| 
									
										
										
										
											2019-03-30 02:26:15 +07:00
										 |  |  |       if ( (0 == tu_edpt_number(event->xfer_complete.ep_addr)) && (event->xfer_complete.len == 0) ) break; | 
					
						
							| 
									
										
										
										
											2018-11-16 21:56:39 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-07 23:04:34 -08:00
										 |  |  |       osal_queue_send(_usbd_q, event, in_isr); | 
					
						
							| 
									
										
										
										
											2018-11-23 15:22:46 +07:00
										 |  |  |       TU_ASSERT(event->xfer_complete.result == XFER_RESULT_SUCCESS,); | 
					
						
							| 
									
										
										
										
											2018-10-24 00:44:26 +07:00
										 |  |  |     break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-30 02:26:15 +07:00
										 |  |  |     // Not an DCD event, just a convenient way to defer ISR function should we need to
 | 
					
						
							|  |  |  |     case USBD_EVENT_FUNC_CALL: | 
					
						
							| 
									
										
										
										
											2018-12-05 17:01:19 +07:00
										 |  |  |       osal_queue_send(_usbd_q, event, in_isr); | 
					
						
							|  |  |  |     break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-23 16:07:48 +07:00
										 |  |  |     default: break; | 
					
						
							| 
									
										
										
										
											2018-10-23 15:08:31 +07:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-12 14:01:38 +07:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							| 
									
										
										
										
											2018-06-22 12:53:13 +07:00
										 |  |  | // Helper
 | 
					
						
							| 
									
										
										
										
											2014-03-12 14:01:38 +07:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							| 
									
										
										
										
											2018-12-05 17:01:19 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-01 19:29:56 +07:00
										 |  |  | // Parse consecutive endpoint descriptors (IN & OUT)
 | 
					
						
							|  |  |  | bool usbd_open_edpt_pair(uint8_t rhport, uint8_t const* p_desc, uint8_t ep_count, uint8_t xfer_type, uint8_t* ep_out, uint8_t* ep_in) | 
					
						
							| 
									
										
										
										
											2018-06-16 12:30:10 +07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-05-01 19:29:56 +07:00
										 |  |  |   for(int i=0; i<ep_count; i++) | 
					
						
							| 
									
										
										
										
											2018-06-16 12:30:10 +07:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2019-05-01 19:29:56 +07:00
										 |  |  |     tusb_desc_endpoint_t const * desc_ep = (tusb_desc_endpoint_t const *) p_desc; | 
					
						
							| 
									
										
										
										
											2018-06-16 12:30:10 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-21 12:02:52 -04:00
										 |  |  |     TU_ASSERT(TUSB_DESC_ENDPOINT == desc_ep->bDescriptorType && xfer_type == desc_ep->bmAttributes.xfer); | 
					
						
							| 
									
										
										
										
											2019-05-01 19:29:56 +07:00
										 |  |  |     TU_ASSERT(dcd_edpt_open(rhport, desc_ep)); | 
					
						
							| 
									
										
										
										
											2018-06-16 12:30:10 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-01 19:29:56 +07:00
										 |  |  |     if ( tu_edpt_dir(desc_ep->bEndpointAddress) == TUSB_DIR_IN ) | 
					
						
							| 
									
										
										
										
											2018-06-16 12:30:10 +07:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-05-01 19:29:56 +07:00
										 |  |  |       (*ep_in) = desc_ep->bEndpointAddress; | 
					
						
							| 
									
										
										
										
											2018-06-16 12:30:10 +07:00
										 |  |  |     }else | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2019-05-01 19:29:56 +07:00
										 |  |  |       (*ep_out) = desc_ep->bEndpointAddress; | 
					
						
							| 
									
										
										
										
											2018-06-16 12:30:10 +07:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-03-12 14:01:38 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-01 19:29:56 +07:00
										 |  |  |     p_desc = tu_desc_next(p_desc); | 
					
						
							| 
									
										
										
										
											2018-06-16 12:30:10 +07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-12 13:00:59 +07:00
										 |  |  |   return true; | 
					
						
							| 
									
										
										
										
											2018-06-16 12:30:10 +07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-06-22 12:53:13 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-05 17:01:19 +07:00
										 |  |  | // Helper to defer an isr function
 | 
					
						
							|  |  |  | void usbd_defer_func(osal_task_func_t func, void* param, bool in_isr) | 
					
						
							| 
									
										
										
										
											2018-06-22 12:53:13 +07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2018-10-23 15:08:31 +07:00
										 |  |  |   dcd_event_t event = | 
					
						
							| 
									
										
										
										
											2018-06-22 12:53:13 +07:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2018-07-13 14:26:40 +07:00
										 |  |  |       .rhport   = 0, | 
					
						
							| 
									
										
										
										
											2019-03-30 02:26:15 +07:00
										 |  |  |       .event_id = USBD_EVENT_FUNC_CALL, | 
					
						
							| 
									
										
										
										
											2018-06-22 12:53:13 +07:00
										 |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   event.func_call.func  = func; | 
					
						
							|  |  |  |   event.func_call.param = param; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-05 17:01:19 +07:00
										 |  |  |   dcd_event_handler(&event, in_isr); | 
					
						
							| 
									
										
										
										
											2018-06-22 12:53:13 +07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-27 16:09:49 +07:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // USBD Endpoint API
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							| 
									
										
										
										
											2019-05-29 16:55:15 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  | bool usbd_edpt_xfer(uint8_t rhport, uint8_t ep_addr, uint8_t * buffer, uint16_t total_bytes) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   uint8_t const epnum = tu_edpt_number(ep_addr); | 
					
						
							|  |  |  |   uint8_t const dir   = tu_edpt_dir(ep_addr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   TU_VERIFY( dcd_edpt_xfer(rhport, ep_addr, buffer, total_bytes) ); | 
					
						
							| 
									
										
										
										
											2019-10-01 22:19:04 +07:00
										 |  |  |   _usbd_dev.ep_status[epnum][dir].busy = true; | 
					
						
							| 
									
										
										
										
											2019-05-29 16:55:15 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool usbd_edpt_busy(uint8_t rhport, uint8_t ep_addr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   (void) rhport; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   uint8_t const epnum = tu_edpt_number(ep_addr); | 
					
						
							|  |  |  |   uint8_t const dir   = tu_edpt_dir(ep_addr); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-01 22:19:04 +07:00
										 |  |  |   return _usbd_dev.ep_status[epnum][dir].busy; | 
					
						
							| 
									
										
										
										
											2019-05-29 16:55:15 +07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-27 16:09:49 +07:00
										 |  |  | void usbd_edpt_stall(uint8_t rhport, uint8_t ep_addr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   uint8_t const epnum = tu_edpt_number(ep_addr); | 
					
						
							|  |  |  |   uint8_t const dir   = tu_edpt_dir(ep_addr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   dcd_edpt_stall(rhport, ep_addr); | 
					
						
							| 
									
										
										
										
											2019-10-01 22:19:04 +07:00
										 |  |  |   _usbd_dev.ep_status[epnum][dir].stalled = true; | 
					
						
							|  |  |  |   _usbd_dev.ep_status[epnum][dir].busy = true; | 
					
						
							| 
									
										
										
										
											2019-03-27 16:09:49 +07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void usbd_edpt_clear_stall(uint8_t rhport, uint8_t ep_addr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   uint8_t const epnum = tu_edpt_number(ep_addr); | 
					
						
							|  |  |  |   uint8_t const dir   = tu_edpt_dir(ep_addr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   dcd_edpt_clear_stall(rhport, ep_addr); | 
					
						
							| 
									
										
										
										
											2019-10-01 22:19:04 +07:00
										 |  |  |   _usbd_dev.ep_status[epnum][dir].stalled = false; | 
					
						
							|  |  |  |   _usbd_dev.ep_status[epnum][dir].busy = false; | 
					
						
							| 
									
										
										
										
											2019-03-27 16:09:49 +07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool usbd_edpt_stalled(uint8_t rhport, uint8_t ep_addr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   (void) rhport; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   uint8_t const epnum = tu_edpt_number(ep_addr); | 
					
						
							|  |  |  |   uint8_t const dir   = tu_edpt_dir(ep_addr); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-01 22:19:04 +07:00
										 |  |  |   return _usbd_dev.ep_status[epnum][dir].stalled; | 
					
						
							| 
									
										
										
										
											2019-03-27 16:09:49 +07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-12 14:01:38 +07:00
										 |  |  | #endif
 |