introduce optional usbd_edpt_claim, usbd_edpt_release which can be used to gain exclusive access to usbd_edpt_xfer
This commit is contained in:
		| @@ -63,9 +63,11 @@ typedef struct | ||||
|   { | ||||
|     volatile bool busy    : 1; | ||||
|     volatile bool stalled : 1; | ||||
|     volatile bool claimed : 1; | ||||
|  | ||||
|     // TODO merge ep2drv here, 4-bit should be sufficient | ||||
|   }ep_status[8][2]; | ||||
|  | ||||
| }usbd_device_t; | ||||
|  | ||||
| static usbd_device_t _usbd_dev; | ||||
| @@ -237,6 +239,13 @@ static inline usbd_class_driver_t const * get_driver(uint8_t drvid) | ||||
| OSAL_QUEUE_DEF(OPT_MODE_DEVICE, _usbd_qdef, CFG_TUD_TASK_QUEUE_SZ, dcd_event_t); | ||||
| static osal_queue_t _usbd_q; | ||||
|  | ||||
| // Mutex for claiming endpoint, only needed when using with preempted RTOS | ||||
| #if CFG_TUSB_OS != OPT_OS_NONE | ||||
| static osal_mutex_def_t _ubsd_mutexdef; | ||||
| static osal_mutex_t _usbd_mutex; | ||||
| #endif | ||||
|  | ||||
|  | ||||
| //--------------------------------------------------------------------+ | ||||
| // Prototypes | ||||
| //--------------------------------------------------------------------+ | ||||
| @@ -351,9 +360,15 @@ bool tud_init (void) | ||||
|  | ||||
|   tu_varclr(&_usbd_dev); | ||||
|  | ||||
| #if CFG_TUSB_OS != OPT_OS_NONE | ||||
|   // Init device mutex | ||||
|   _usbd_mutex = osal_mutex_create(&_ubsd_mutexdef); | ||||
|   TU_ASSERT(_usbd_mutex); | ||||
| #endif | ||||
|  | ||||
|   // Init device queue & task | ||||
|   _usbd_q = osal_queue_create(&_usbd_qdef); | ||||
|   TU_ASSERT(_usbd_q != NULL); | ||||
|   TU_ASSERT(_usbd_q); | ||||
|  | ||||
|   // Get application driver if available | ||||
|   if ( usbd_app_driver_get_cb ) | ||||
| @@ -478,6 +493,7 @@ void tud_task (void) | ||||
|         TU_LOG2("on EP %02X with %u bytes\r\n", ep_addr, (unsigned int) event.xfer_complete.len); | ||||
|  | ||||
|         _usbd_dev.ep_status[epnum][ep_dir].busy = false; | ||||
|         _usbd_dev.ep_status[epnum][ep_dir].claimed = 0; | ||||
|  | ||||
|         if ( 0 == epnum ) | ||||
|         { | ||||
| @@ -1076,6 +1092,56 @@ bool usbd_edpt_open(uint8_t rhport, tusb_desc_endpoint_t const * desc_ep) | ||||
|   return dcd_edpt_open(rhport, desc_ep); | ||||
| } | ||||
|  | ||||
| bool usbd_edpt_claim(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); | ||||
|  | ||||
| #if CFG_TUSB_OS != OPT_OS_NONE | ||||
|   osal_mutex_lock(_usbd_mutex, OSAL_TIMEOUT_WAIT_FOREVER); | ||||
| #endif | ||||
|  | ||||
|   // can only claim the endpoint if it is not busy and not claimed yet. | ||||
|   bool ret = (_usbd_dev.ep_status[epnum][dir].busy == 0) && (_usbd_dev.ep_status[epnum][dir].claimed == 0); | ||||
|   if (ret) | ||||
|   { | ||||
|     _usbd_dev.ep_status[epnum][dir].claimed = 1; | ||||
|   } | ||||
|  | ||||
| #if CFG_TUSB_OS != OPT_OS_NONE | ||||
|   osal_mutex_unlock(_usbd_mutex); | ||||
| #endif | ||||
|  | ||||
|   return ret; | ||||
| } | ||||
|  | ||||
| bool usbd_edpt_release(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); | ||||
|  | ||||
| #if CFG_TUSB_OS != OPT_OS_NONE | ||||
|   osal_mutex_lock(_usbd_mutex, OSAL_TIMEOUT_WAIT_FOREVER); | ||||
| #endif | ||||
|  | ||||
|   // can only release the endpoint if it is claimed and not busy | ||||
|   bool ret = (_usbd_dev.ep_status[epnum][dir].busy == 0) && (_usbd_dev.ep_status[epnum][dir].claimed == 1); | ||||
|   if (ret) | ||||
|   { | ||||
|     _usbd_dev.ep_status[epnum][dir].claimed = 0; | ||||
|   } | ||||
|  | ||||
| #if CFG_TUSB_OS != OPT_OS_NONE | ||||
|   osal_mutex_unlock(_usbd_mutex); | ||||
| #endif | ||||
|  | ||||
|   return ret; | ||||
| } | ||||
|  | ||||
| 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); | ||||
| @@ -1083,6 +1149,9 @@ bool usbd_edpt_xfer(uint8_t rhport, uint8_t ep_addr, uint8_t * buffer, uint16_t | ||||
|  | ||||
|   TU_LOG2("  Queue EP %02X with %u bytes ... ", ep_addr, total_bytes); | ||||
|  | ||||
|   // Attempt to transfer on busy or not claimed (skip now) endpoint, sound like an race condition ! | ||||
|   TU_ASSERT(_usbd_dev.ep_status[epnum][dir].busy == 0 /*&& _usbd_dev.ep_status[epnum][dir].claimed == 1*/); | ||||
|  | ||||
|   // Set busy first since the actual transfer can be complete before dcd_edpt_xfer() could return | ||||
|   // and usbd task can preempt and clear the busy | ||||
|   _usbd_dev.ep_status[epnum][dir].busy = true; | ||||
|   | ||||
| @@ -70,6 +70,12 @@ void usbd_edpt_close(uint8_t rhport, uint8_t ep_addr); | ||||
| // Submit a usb transfer | ||||
| bool usbd_edpt_xfer(uint8_t rhport, uint8_t ep_addr, uint8_t * buffer, uint16_t total_bytes); | ||||
|  | ||||
| // Claim an endpoint before submitting a transfer | ||||
| bool usbd_edpt_claim(uint8_t rhport, uint8_t ep_addr); | ||||
|  | ||||
| // Release an endpoint without submitting a transfer | ||||
| bool usbd_edpt_release(uint8_t rhport, uint8_t ep_addr); | ||||
|  | ||||
| // Check if endpoint transferring is complete | ||||
| bool usbd_edpt_busy(uint8_t rhport, uint8_t ep_addr); | ||||
|  | ||||
|   | ||||
		Reference in New Issue
	
	Block a user
	 hathach
					hathach