add usbh_spin_lock/unlock() use spinlock instead of atomic flag for hcd max3421
This commit is contained in:
		| @@ -0,0 +1,3 @@ | |||||||
|  | # Apply board specific content here | ||||||
|  | set(IDF_TARGET "esp32c6") | ||||||
|  | set(MAX3421_HOST 1) | ||||||
							
								
								
									
										56
									
								
								hw/bsp/espressif/boards/adafruit_feather_esp32c6/board.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										56
									
								
								hw/bsp/espressif/boards/adafruit_feather_esp32c6/board.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,56 @@ | |||||||
|  | /* | ||||||
|  |  * The MIT License (MIT) | ||||||
|  |  * | ||||||
|  |  * Copyright (c) 2020, Ha Thach (tinyusb.org) | ||||||
|  |  * | ||||||
|  |  * Permission is hereby granted, free of charge, to any person obtaining a copy | ||||||
|  |  * of this software and associated documentation files (the "Software"), to deal | ||||||
|  |  * in the Software without restriction, including without limitation the rights | ||||||
|  |  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | ||||||
|  |  * copies of the Software, and to permit persons to whom the Software is | ||||||
|  |  * furnished to do so, subject to the following conditions: | ||||||
|  |  * | ||||||
|  |  * The above copyright notice and this permission notice shall be included in | ||||||
|  |  * all copies or substantial portions of the Software. | ||||||
|  |  * | ||||||
|  |  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | ||||||
|  |  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | ||||||
|  |  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | ||||||
|  |  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | ||||||
|  |  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | ||||||
|  |  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN | ||||||
|  |  * THE SOFTWARE. | ||||||
|  |  * | ||||||
|  |  * This file is part of the TinyUSB stack. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | /* metadata: | ||||||
|  |    name: Adafruit Feather EPS32-C6 | ||||||
|  |    url: https://www.adafruit.com/product/5933 | ||||||
|  | */ | ||||||
|  |  | ||||||
|  | #ifndef BOARD_H_ | ||||||
|  | #define BOARD_H_ | ||||||
|  |  | ||||||
|  | #ifdef __cplusplus | ||||||
|  |  extern "C" { | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | #define NEOPIXEL_PIN          15 | ||||||
|  |  | ||||||
|  | #define BUTTON_PIN            9 | ||||||
|  | #define BUTTON_STATE_ACTIVE   0 | ||||||
|  |  | ||||||
|  | // SPI for USB host shield | ||||||
|  | #define MAX3421_SPI_HOST SPI2_HOST | ||||||
|  | #define MAX3421_SCK_PIN  21 | ||||||
|  | #define MAX3421_MOSI_PIN 22 | ||||||
|  | #define MAX3421_MISO_PIN 23 | ||||||
|  | #define MAX3421_CS_PIN   8 | ||||||
|  | #define MAX3421_INTR_PIN 7 | ||||||
|  |  | ||||||
|  | #ifdef __cplusplus | ||||||
|  |  } | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | #endif /* BOARD_H_ */ | ||||||
| @@ -49,7 +49,9 @@ static led_strip_handle_t led_strip; | |||||||
| static void max3421_init(void); | static void max3421_init(void); | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
|  | #if TU_CHECK_MCU(OPT_MCU_ESP32S2, OPT_MCU_ESP32S3, OPT_MCU_ESP32P4) | ||||||
| static bool usb_init(void); | static bool usb_init(void); | ||||||
|  | #endif | ||||||
|  |  | ||||||
| //--------------------------------------------------------------------+ | //--------------------------------------------------------------------+ | ||||||
| // Implementation | // Implementation | ||||||
|   | |||||||
| @@ -147,6 +147,9 @@ static osal_mutex_t _usbh_mutex; | |||||||
| #define _usbh_mutex   NULL | #define _usbh_mutex   NULL | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
|  | // Spinlock for interrupt handler | ||||||
|  | static OSAL_SPINLOCK_DEF(_usbh_spin, usbh_int_set); | ||||||
|  |  | ||||||
| // Event queue: usbh_int_set() is used as mutex in OS NONE config | // Event queue: usbh_int_set() is used as mutex in OS NONE config | ||||||
| OSAL_QUEUE_DEF(usbh_int_set, _usbh_qdef, CFG_TUH_TASK_QUEUE_SZ, hcd_event_t); | OSAL_QUEUE_DEF(usbh_int_set, _usbh_qdef, CFG_TUH_TASK_QUEUE_SZ, hcd_event_t); | ||||||
| static osal_queue_t _usbh_q; | static osal_queue_t _usbh_q; | ||||||
| @@ -424,6 +427,8 @@ bool tuh_rhport_init(uint8_t rhport, const tusb_rhport_init_t* rh_init) { | |||||||
|     TU_LOG_INT_USBH(sizeof(tu_fifo_t)); |     TU_LOG_INT_USBH(sizeof(tu_fifo_t)); | ||||||
|     TU_LOG_INT_USBH(sizeof(tu_edpt_stream_t)); |     TU_LOG_INT_USBH(sizeof(tu_edpt_stream_t)); | ||||||
|  |  | ||||||
|  |     osal_spin_init(&_usbh_spin); | ||||||
|  |  | ||||||
|     // Event queue |     // Event queue | ||||||
|     _usbh_q = osal_queue_create(&_usbh_qdef); |     _usbh_q = osal_queue_create(&_usbh_qdef); | ||||||
|     TU_ASSERT(_usbh_q != NULL); |     TU_ASSERT(_usbh_q != NULL); | ||||||
| @@ -895,6 +900,14 @@ void usbh_int_set(bool enabled) { | |||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | void usbh_spin_lock(bool in_isr) { | ||||||
|  |   osal_spin_lock(&_usbh_spin, in_isr); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void usbh_spin_unlock(bool in_isr) { | ||||||
|  |   osal_spin_unlock(&_usbh_spin, in_isr); | ||||||
|  | } | ||||||
|  |  | ||||||
| void usbh_defer_func(osal_task_func_t func, void *param, bool in_isr) { | void usbh_defer_func(osal_task_func_t func, void *param, bool in_isr) { | ||||||
|   hcd_event_t event = { 0 }; |   hcd_event_t event = { 0 }; | ||||||
|   event.event_id = USBH_EVENT_FUNC_CALL; |   event.event_id = USBH_EVENT_FUNC_CALL; | ||||||
|   | |||||||
| @@ -71,6 +71,9 @@ void usbh_int_set(bool enabled); | |||||||
|  |  | ||||||
| void usbh_defer_func(osal_task_func_t func, void *param, bool in_isr); | void usbh_defer_func(osal_task_func_t func, void *param, bool in_isr); | ||||||
|  |  | ||||||
|  | void usbh_spin_lock(bool in_isr); | ||||||
|  | void usbh_spin_unlock(bool in_isr); | ||||||
|  |  | ||||||
| //--------------------------------------------------------------------+ | //--------------------------------------------------------------------+ | ||||||
| // USBH Endpoint API | // USBH Endpoint API | ||||||
| //--------------------------------------------------------------------+ | //--------------------------------------------------------------------+ | ||||||
|   | |||||||
| @@ -28,9 +28,9 @@ | |||||||
|  |  | ||||||
| #if CFG_TUH_ENABLED && defined(CFG_TUH_MAX3421) && CFG_TUH_MAX3421 | #if CFG_TUH_ENABLED && defined(CFG_TUH_MAX3421) && CFG_TUH_MAX3421 | ||||||
|  |  | ||||||
| #include <stdatomic.h> |  | ||||||
| #include "host/hcd.h" | #include "host/hcd.h" | ||||||
| #include "host/usbh.h" | #include "host/usbh.h" | ||||||
|  | #include "host/usbh_pvt.h" | ||||||
|  |  | ||||||
| //--------------------------------------------------------------------+ | //--------------------------------------------------------------------+ | ||||||
| // | // | ||||||
| @@ -233,7 +233,7 @@ typedef struct { | |||||||
|     uint8_t hxfr; |     uint8_t hxfr; | ||||||
|   }sndfifo_owner; |   }sndfifo_owner; | ||||||
|  |  | ||||||
|   atomic_flag busy; // busy transferring |   bool busy_lock; // busy transferring | ||||||
|  |  | ||||||
| #if OSAL_MUTEX_REQUIRED | #if OSAL_MUTEX_REQUIRED | ||||||
|   OSAL_MUTEX_DEF(spi_mutexdef); |   OSAL_MUTEX_DEF(spi_mutexdef); | ||||||
| @@ -327,7 +327,9 @@ TU_ATTR_ALWAYS_INLINE static inline void mode_write(uint8_t rhport, uint8_t data | |||||||
| } | } | ||||||
|  |  | ||||||
| TU_ATTR_ALWAYS_INLINE static inline void peraddr_write(uint8_t rhport, uint8_t data, bool in_isr) { | TU_ATTR_ALWAYS_INLINE static inline void peraddr_write(uint8_t rhport, uint8_t data, bool in_isr) { | ||||||
|   if ( _hcd_data.peraddr == data ) return; // no need to change address |   if (_hcd_data.peraddr == data) { | ||||||
|  |     return; // no need to change address | ||||||
|  |   } | ||||||
|  |  | ||||||
|   _hcd_data.peraddr = data; |   _hcd_data.peraddr = data; | ||||||
|   reg_write(rhport, PERADDR_ADDR, data, in_isr); |   reg_write(rhport, PERADDR_ADDR, data, in_isr); | ||||||
| @@ -373,7 +375,7 @@ TU_ATTR_ALWAYS_INLINE static inline void hwfifo_setup(uint8_t rhport, const uint | |||||||
|  |  | ||||||
| static void hwfifo_receive(uint8_t rhport, uint8_t * buffer, uint16_t len, bool in_isr) { | static void hwfifo_receive(uint8_t rhport, uint8_t * buffer, uint16_t len, bool in_isr) { | ||||||
|   uint8_t hirq; |   uint8_t hirq; | ||||||
|   uint8_t const reg = RCVVFIFO_ADDR; |   const uint8_t reg = RCVVFIFO_ADDR; | ||||||
|  |  | ||||||
|   max3421_spi_lock(rhport, in_isr); |   max3421_spi_lock(rhport, in_isr); | ||||||
|  |  | ||||||
| @@ -389,7 +391,7 @@ static void hwfifo_receive(uint8_t rhport, uint8_t * buffer, uint16_t len, bool | |||||||
| //--------------------------------------------------------------------+ | //--------------------------------------------------------------------+ | ||||||
|  |  | ||||||
| static max3421_ep_t* find_ep_not_addr0(uint8_t daddr, uint8_t ep_num, uint8_t ep_dir) { | static max3421_ep_t* find_ep_not_addr0(uint8_t daddr, uint8_t ep_num, uint8_t ep_dir) { | ||||||
|   uint8_t const is_out = 1-ep_dir; |   const uint8_t is_out = 1-ep_dir; | ||||||
|   for(size_t i=1; i<CFG_TUH_MAX3421_ENDPOINT_TOTAL; i++) { |   for(size_t i=1; i<CFG_TUH_MAX3421_ENDPOINT_TOTAL; i++) { | ||||||
|     max3421_ep_t* ep = &_hcd_data.ep[i]; |     max3421_ep_t* ep = &_hcd_data.ep[i]; | ||||||
|     // control endpoint is bi-direction (skip check) |     // control endpoint is bi-direction (skip check) | ||||||
| @@ -727,8 +729,8 @@ static void xact_generic(uint8_t rhport, max3421_ep_t *ep, bool switch_ep, bool | |||||||
|  |  | ||||||
| // Submit a transfer, when complete hcd_event_xfer_complete() must be invoked | // Submit a transfer, when complete hcd_event_xfer_complete() must be invoked | ||||||
| bool hcd_edpt_xfer(uint8_t rhport, uint8_t daddr, uint8_t ep_addr, uint8_t * buffer, uint16_t buflen) { | bool hcd_edpt_xfer(uint8_t rhport, uint8_t daddr, uint8_t ep_addr, uint8_t * buffer, uint16_t buflen) { | ||||||
|   uint8_t const ep_num = tu_edpt_number(ep_addr); |   const uint8_t ep_num = tu_edpt_number(ep_addr); | ||||||
|   uint8_t const ep_dir = (uint8_t) tu_edpt_dir(ep_addr); |   const uint8_t ep_dir = (uint8_t) tu_edpt_dir(ep_addr); | ||||||
|   max3421_ep_t* ep = find_opened_ep(daddr, ep_num, ep_dir); |   max3421_ep_t* ep = find_opened_ep(daddr, ep_num, ep_dir); | ||||||
|   TU_VERIFY(ep); |   TU_VERIFY(ep); | ||||||
|  |  | ||||||
| @@ -744,8 +746,17 @@ bool hcd_edpt_xfer(uint8_t rhport, uint8_t daddr, uint8_t ep_addr, uint8_t * buf | |||||||
|   ep->xferred_len = 0; |   ep->xferred_len = 0; | ||||||
|   ep->state = EP_STATE_ATTEMPT_1; |   ep->state = EP_STATE_ATTEMPT_1; | ||||||
|  |  | ||||||
|  |   bool has_xfer = false; | ||||||
|  |  | ||||||
|  |   usbh_spin_lock(false); | ||||||
|  |   if (!_hcd_data.busy_lock) { | ||||||
|  |     _hcd_data.busy_lock = true; | ||||||
|  |     has_xfer = true; | ||||||
|  |   } | ||||||
|  |   usbh_spin_unlock(false); | ||||||
|  |  | ||||||
|   // carry out transfer if not busy |   // carry out transfer if not busy | ||||||
|   if (!atomic_flag_test_and_set(&_hcd_data.busy)) { |   if (has_xfer) { | ||||||
|     xact_generic(rhport, ep, true, false); |     xact_generic(rhport, ep, true, false); | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -781,8 +792,17 @@ bool hcd_setup_send(uint8_t rhport, uint8_t daddr, uint8_t const setup_packet[8] | |||||||
|   ep->xferred_len = 0; |   ep->xferred_len = 0; | ||||||
|   ep->state = EP_STATE_ATTEMPT_1; |   ep->state = EP_STATE_ATTEMPT_1; | ||||||
|  |  | ||||||
|  |   bool has_xfer = false; | ||||||
|  |  | ||||||
|  |   usbh_spin_lock(false); | ||||||
|  |   if (!_hcd_data.busy_lock) { | ||||||
|  |     _hcd_data.busy_lock = true; | ||||||
|  |     has_xfer = true; | ||||||
|  |   } | ||||||
|  |   usbh_spin_unlock(false); | ||||||
|  |  | ||||||
|   // carry out transfer if not busy |   // carry out transfer if not busy | ||||||
|   if (!atomic_flag_test_and_set(&_hcd_data.busy)) { |   if (has_xfer) { | ||||||
|     xact_setup(rhport, ep, false); |     xact_setup(rhport, ep, false); | ||||||
|   } |   } | ||||||
|  |  | ||||||
| @@ -848,8 +868,8 @@ static void handle_connect_irq(uint8_t rhport, bool in_isr) { | |||||||
| } | } | ||||||
|  |  | ||||||
| static void xfer_complete_isr(uint8_t rhport, max3421_ep_t *ep, xfer_result_t result, uint8_t hrsl, bool in_isr) { | static void xfer_complete_isr(uint8_t rhport, max3421_ep_t *ep, xfer_result_t result, uint8_t hrsl, bool in_isr) { | ||||||
|   uint8_t const ep_dir = 1-ep->hxfr_bm.is_out; |   const uint8_t ep_dir = 1 - ep->hxfr_bm.is_out; | ||||||
|   uint8_t const ep_addr = tu_edpt_addr(ep->hxfr_bm.ep_num, ep_dir); |   const uint8_t ep_addr = tu_edpt_addr(ep->hxfr_bm.ep_num, ep_dir); | ||||||
|  |  | ||||||
|   // save data toggle |   // save data toggle | ||||||
|   if (ep_dir) { |   if (ep_dir) { | ||||||
| @@ -867,7 +887,9 @@ static void xfer_complete_isr(uint8_t rhport, max3421_ep_t *ep, xfer_result_t re | |||||||
|     xact_generic(rhport, next_ep, true, in_isr); |     xact_generic(rhport, next_ep, true, in_isr); | ||||||
|   }else { |   }else { | ||||||
|     // no more pending |     // no more pending | ||||||
|     atomic_flag_clear(&_hcd_data.busy); |     usbh_spin_lock(in_isr); | ||||||
|  |     _hcd_data.busy_lock = false; | ||||||
|  |     usbh_spin_unlock(in_isr); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -906,7 +928,9 @@ static void handle_xfer_done(uint8_t rhport, bool in_isr) { | |||||||
|         xact_generic(rhport, next_ep, true, in_isr); |         xact_generic(rhport, next_ep, true, in_isr); | ||||||
|       } else { |       } else { | ||||||
|         // no more pending in this frame -> clear busy |         // no more pending in this frame -> clear busy | ||||||
|         atomic_flag_clear(&_hcd_data.busy); |         usbh_spin_lock(in_isr); | ||||||
|  |         _hcd_data.busy_lock = false; | ||||||
|  |         usbh_spin_unlock(in_isr); | ||||||
|       } |       } | ||||||
|       return; |       return; | ||||||
|  |  | ||||||
| @@ -997,8 +1021,8 @@ void print_hirq(uint8_t hirq) { | |||||||
| // Interrupt handler | // Interrupt handler | ||||||
| void hcd_int_handler(uint8_t rhport, bool in_isr) { | void hcd_int_handler(uint8_t rhport, bool in_isr) { | ||||||
|   uint8_t hirq = reg_read(rhport, HIRQ_ADDR, in_isr) & _hcd_data.hien; |   uint8_t hirq = reg_read(rhport, HIRQ_ADDR, in_isr) & _hcd_data.hien; | ||||||
|   if (!hirq) return; |   if (!hirq) { return; } | ||||||
| //  print_hirq(hirq); |   //  print_hirq(hirq); | ||||||
|  |  | ||||||
|   if (hirq & HIRQ_FRAME_IRQ) { |   if (hirq & HIRQ_FRAME_IRQ) { | ||||||
|     _hcd_data.frame_count++; |     _hcd_data.frame_count++; | ||||||
| @@ -1017,10 +1041,21 @@ void hcd_int_handler(uint8_t rhport, bool in_isr) { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     // start usb transfer if not busy |     // start usb transfer if not busy | ||||||
|     if (ep_retry != NULL && !atomic_flag_test_and_set(&_hcd_data.busy)) { |     if (ep_retry != NULL) { | ||||||
|  |       bool has_xfer = false; | ||||||
|  |  | ||||||
|  |       usbh_spin_lock(in_isr); | ||||||
|  |       if (!_hcd_data.busy_lock) { | ||||||
|  |         _hcd_data.busy_lock = true; | ||||||
|  |         has_xfer = true; | ||||||
|  |       } | ||||||
|  |       usbh_spin_unlock(in_isr); | ||||||
|  |  | ||||||
|  |       if (has_xfer) { | ||||||
|         xact_generic(rhport, ep_retry, true, in_isr); |         xact_generic(rhport, ep_retry, true, in_isr); | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|   if (hirq & HIRQ_CONDET_IRQ) { |   if (hirq & HIRQ_CONDET_IRQ) { | ||||||
|     handle_connect_irq(rhport, in_isr); |     handle_connect_irq(rhport, in_isr); | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 hathach
					hathach