Merge pull request #2750 from hathach/fix-ch32v203-setup
rework fsdev driver, fix ch32v203 race condition and stability issue
This commit is contained in:
		
							
								
								
									
										59
									
								
								.github/workflows/hil_test.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										59
									
								
								.github/workflows/hil_test.yml
									
									
									
									
										vendored
									
									
								
							| @@ -44,11 +44,17 @@ jobs: | |||||||
|           echo "BOARDS_LIST=$BOARDS_LIST" >> $GITHUB_ENV |           echo "BOARDS_LIST=$BOARDS_LIST" >> $GITHUB_ENV | ||||||
|           echo "BOARDS_LIST=$BOARDS_LIST" >> $GITHUB_OUTPUT |           echo "BOARDS_LIST=$BOARDS_LIST" >> $GITHUB_OUTPUT | ||||||
|  |  | ||||||
|       - name: Setup Toolchain |       - name: Setup arm-gcc toolchain | ||||||
|         uses: ./.github/actions/setup_toolchain |         uses: ./.github/actions/setup_toolchain | ||||||
|         with: |         with: | ||||||
|           toolchain: 'arm-gcc' |           toolchain: 'arm-gcc' | ||||||
|  |  | ||||||
|  |       - name: Setup risv-gcc toolchain | ||||||
|  |         uses: ./.github/actions/setup_toolchain | ||||||
|  |         with: | ||||||
|  |           toolchain: 'riscv-gcc' | ||||||
|  |           toolchain_url: 'https://github.com/xpack-dev-tools/riscv-none-elf-gcc-xpack/releases/download/v13.2.0-2/xpack-riscv-none-elf-gcc-13.2.0-2-linux-x64.tar.gz' | ||||||
|  |  | ||||||
|       - name: Get Dependencies |       - name: Get Dependencies | ||||||
|         uses: ./.github/actions/get_deps |         uses: ./.github/actions/get_deps | ||||||
|         with: |         with: | ||||||
| @@ -65,56 +71,6 @@ jobs: | |||||||
|             cmake-build/cmake-build-*/*/*/*.elf |             cmake-build/cmake-build-*/*/*/*.elf | ||||||
|             cmake-build/cmake-build-*/*/*/*.bin |             cmake-build/cmake-build-*/*/*/*.bin | ||||||
|  |  | ||||||
|   # --------------------------------------- |  | ||||||
|   # Build Espressif (skipped since CP210x cause USB bus issue) |  | ||||||
|   # cp210x ttyUSB0: usb_serial_generic_write_bulk_callback - nonzero urb status: -71 |  | ||||||
|   # --------------------------------------- |  | ||||||
|   build-esp: |  | ||||||
|     if: false |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     outputs: |  | ||||||
|       BOARDS_LIST: ${{ steps.parse_hil_json.outputs.BOARDS_LIST }} |  | ||||||
|     steps: |  | ||||||
|       - name: Checkout TinyUSB |  | ||||||
|         uses: actions/checkout@v4 |  | ||||||
|  |  | ||||||
|       - name: Parse HIL json |  | ||||||
|         id: parse_hil_json |  | ||||||
|         run: | |  | ||||||
|           sudo apt install -y jq |  | ||||||
|           # Espressif boards |  | ||||||
|           BOARDS_LIST=$(jq -r '.boards[] | select(.flasher == "esptool") | "-b " + .name' ${{ env.HIL_JSON }} | tr '\n' ' ') |  | ||||||
|           echo "BOARDS_LIST=$BOARDS_LIST" |  | ||||||
|           echo "BOARDS_LIST=$BOARDS_LIST" >> $GITHUB_ENV |  | ||||||
|           echo "BOARDS_LIST=$BOARDS_LIST" >> $GITHUB_OUTPUT |  | ||||||
|  |  | ||||||
|       - name: Setup ESP-IDF |  | ||||||
|         if: env.BOARDS_LIST != '' |  | ||||||
|         uses: ./.github/actions/setup_toolchain |  | ||||||
|         with: |  | ||||||
|           toolchain: 'esp-idf' |  | ||||||
|           toolchain_url: 'v5.1.1' |  | ||||||
|  |  | ||||||
|       - name: Get Dependencies |  | ||||||
|         uses: ./.github/actions/get_deps |  | ||||||
|         with: |  | ||||||
|           arg: ${{ env.BOARDS_LIST }} |  | ||||||
|  |  | ||||||
|       - name: Build Espressif |  | ||||||
|         if: env.BOARDS_LIST != '' |  | ||||||
|         run: docker run --rm -v $PWD:/project -w /project espressif/idf:v5.1.1 python3 tools/build.py $BOARDS_LIST |  | ||||||
|  |  | ||||||
|       - name: Upload Artifacts for Hardware Testing |  | ||||||
|         uses: actions/upload-artifact@v4 |  | ||||||
|         with: |  | ||||||
|           name: hil_rpi_esp |  | ||||||
|           path: | |  | ||||||
|             cmake-build/cmake-build-*/*/*/*.bin |  | ||||||
|             cmake-build/cmake-build-*/*/*/bootloader/bootloader.bin |  | ||||||
|             cmake-build/cmake-build-*/*/*/partition_table/partition-table.bin |  | ||||||
|             cmake-build/cmake-build-*/*/*/config.env |  | ||||||
|             cmake-build/cmake-build-*/*/*/flash_args |  | ||||||
|  |  | ||||||
|   # --------------------------------------- |   # --------------------------------------- | ||||||
|   # Hardware in the loop (HIL) |   # Hardware in the loop (HIL) | ||||||
|   # self-hosted running on an RPI. For attached hardware checkout test/hil/rpi.json |   # self-hosted running on an RPI. For attached hardware checkout test/hil/rpi.json | ||||||
| @@ -123,7 +79,6 @@ jobs: | |||||||
|     if: github.repository_owner == 'hathach' |     if: github.repository_owner == 'hathach' | ||||||
|     needs: |     needs: | ||||||
|       - build |       - build | ||||||
|       #- build-esp |  | ||||||
|     runs-on: [self-hosted, ARM64, rpi, hardware-in-the-loop] |     runs-on: [self-hosted, ARM64, rpi, hardware-in-the-loop] | ||||||
|     env: |     env: | ||||||
|       BOARDS_LIST: "${{ needs.build-esp.outputs.BOARDS_LIST }} ${{ needs.build.outputs.BOARDS_LIST }}" |       BOARDS_LIST: "${{ needs.build-esp.outputs.BOARDS_LIST }} ${{ needs.build.outputs.BOARDS_LIST }}" | ||||||
|   | |||||||
							
								
								
									
										15
									
								
								.idea/cmake.xml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										15
									
								
								.idea/cmake.xml
									
									
									
										generated
									
									
									
								
							| @@ -89,19 +89,20 @@ | |||||||
|       <configuration PROFILE_NAME="frdm_kl25z" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=frdm_kl25z -DLOG=3 -DLOGGER=RTT" /> |       <configuration PROFILE_NAME="frdm_kl25z" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=frdm_kl25z -DLOG=3 -DLOGGER=RTT" /> | ||||||
|       <configuration PROFILE_NAME="frdm_k32l2a4s" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=frdm_k32l2a4s" /> |       <configuration PROFILE_NAME="frdm_k32l2a4s" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=frdm_k32l2a4s" /> | ||||||
|       <configuration PROFILE_NAME="frdm_k64f" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=frdm_k64f -DLOG=2 -DLOGGER=RTT" /> |       <configuration PROFILE_NAME="frdm_k64f" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=frdm_k64f -DLOG=2 -DLOGGER=RTT" /> | ||||||
|       <configuration PROFILE_NAME="stm32f072disco" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=stm32f072disco" /> |       <configuration PROFILE_NAME="stm32f072disco" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=stm32f072disco -DLOG=0 -DLOGGER=RTT" /> | ||||||
|       <configuration PROFILE_NAME="stm32f103_mini_2" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=stm32f103_mini_2" /> |       <configuration PROFILE_NAME="stm32f103_mini_2" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=stm32f103_mini_2 -DLOG=1 -DLOGGGER=RTT" /> | ||||||
|  |       <configuration PROFILE_NAME="stm32f103ze_iar" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=stm32f103ze_iar -DLOG=1 -DLOGGGER=RTT" /> | ||||||
|       <configuration PROFILE_NAME="stm32f207nucleo" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=stm32f207nucleo -DLOG=2 -DLOGGER=RTT" /> |       <configuration PROFILE_NAME="stm32f207nucleo" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=stm32f207nucleo -DLOG=2 -DLOGGER=RTT" /> | ||||||
|       <configuration PROFILE_NAME="stm32f303disco" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=stm32f303disco -DLOG=2 -DLOGGER=RTT" /> |       <configuration PROFILE_NAME="stm32f303disco" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=stm32f303disco -DLOG=0 -DLOGGER=RTT" /> | ||||||
|       <configuration PROFILE_NAME="stm32f411disco" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=stm32f411disco -DLOG=2 -DLOGGER=RTT" /> |       <configuration PROFILE_NAME="stm32f411disco" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=stm32f411disco -DLOG=2 -DLOGGER=RTT" /> | ||||||
|       <configuration PROFILE_NAME="stm32f769disco" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=stm32f769disco -DLOG=3 -DLOGGER=RTT" /> |       <configuration PROFILE_NAME="stm32f769disco" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=stm32f769disco -DLOG=3 -DLOGGER=RTT" /> | ||||||
|       <configuration PROFILE_NAME="stm32g0b1nucleo" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=stm32g0b1nucleo" /> |       <configuration PROFILE_NAME="stm32g0b1nucleo" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=stm32g0b1nucleo" /> | ||||||
|       <configuration PROFILE_NAME="stm32g474nucleo" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=stm32g474nucleo" /> |       <configuration PROFILE_NAME="stm32g474nucleo" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=stm32g474nucleo" /> | ||||||
|       <configuration PROFILE_NAME="b_g474e_dpow1" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=b_g474e_dpow1 -DLOG=3 -DLOGGER=RTT" /> |       <configuration PROFILE_NAME="b_g474e_dpow1" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=b_g474e_dpow1 -DLOG=3 -DLOGGER=RTT" /> | ||||||
|       <configuration PROFILE_NAME="stm32h563nucleo" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=stm32h563nucleo -DLOG=3 -DLOGGER=RTT -DTRACE_ETM=1" /> |       <configuration PROFILE_NAME="stm32h563nucleo" ENABLED="false" GENERATION_OPTIONS="-DBOARD=stm32h563nucleo -DLOG=1 -DLOGGER=RTT -DTRACE_ETM=1" /> | ||||||
|       <configuration PROFILE_NAME="stm32h743eval" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=stm32h743eval -DLOG=3 -DLOGGER=RTT -DTRACE_ETM=1" /> |       <configuration PROFILE_NAME="stm32h743eval" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=stm32h743eval -DLOG=3 -DLOGGER=RTT -DTRACE_ETM=1" /> | ||||||
|       <configuration PROFILE_NAME="stm32h743nucleo" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=stm32h743nucleo -DLOG=3" /> |       <configuration PROFILE_NAME="stm32h743nucleo" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=stm32h743nucleo -DLOG=3" /> | ||||||
|       <configuration PROFILE_NAME="stm32l0538disco" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=stm32l0538disco" /> |       <configuration PROFILE_NAME="stm32l0538disco" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=stm32l0538disco -DLOG=0 -DLOGGER=RTT" /> | ||||||
|       <configuration PROFILE_NAME="stm32l476disco" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=stm32l476disco" /> |       <configuration PROFILE_NAME="stm32l476disco" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=stm32l476disco" /> | ||||||
|       <configuration PROFILE_NAME="stm32u575nucleo" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=stm32u575nucleo -DLOG=2 -DLOGGER=RTT" /> |       <configuration PROFILE_NAME="stm32u575nucleo" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=stm32u575nucleo -DLOG=2 -DLOGGER=RTT" /> | ||||||
|       <configuration PROFILE_NAME="stm32u5a5nucleo" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=stm32u5a5nucleo -DLOG=3 -DLOGGER=RTT" /> |       <configuration PROFILE_NAME="stm32u5a5nucleo" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=stm32u5a5nucleo -DLOG=3 -DLOGGER=RTT" /> | ||||||
| @@ -133,8 +134,8 @@ | |||||||
|       <configuration PROFILE_NAME="fomu" ENABLED="false" GENERATION_OPTIONS="-DBOARD=fomu" /> |       <configuration PROFILE_NAME="fomu" ENABLED="false" GENERATION_OPTIONS="-DBOARD=fomu" /> | ||||||
|       <configuration PROFILE_NAME="sipeed_longan_nano" ENABLED="false" GENERATION_OPTIONS="-DBOARD=sipeed_longan_nano" /> |       <configuration PROFILE_NAME="sipeed_longan_nano" ENABLED="false" GENERATION_OPTIONS="-DBOARD=sipeed_longan_nano" /> | ||||||
|       <configuration PROFILE_NAME="nanoch32v203" ENABLED="false" GENERATION_OPTIONS="-DBOARD=nanoch32v203" /> |       <configuration PROFILE_NAME="nanoch32v203" ENABLED="false" GENERATION_OPTIONS="-DBOARD=nanoch32v203" /> | ||||||
|       <configuration PROFILE_NAME="ch32v203c_r0_1v0" ENABLED="false" CONFIG_NAME="MinSizeRel" GENERATION_OPTIONS="-DBOARD=ch32v203c_r0_1v0" /> |       <configuration PROFILE_NAME="ch32v203c_r0_1v0" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=ch32v203c_r0_1v0 -DLOG=0" /> | ||||||
|       <configuration PROFILE_NAME="ch32v203c_r0_1v0 USBFS" ENABLED="false" CONFIG_NAME="MinSizeRel" GENERATION_OPTIONS="-DBOARD=ch32v203c_r0_1v0 -DPORT=1" /> |       <configuration PROFILE_NAME="ch32v203c_r0_1v0 USBFS" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=ch32v203c_r0_1v0 -DPORT=1" /> | ||||||
|       <configuration PROFILE_NAME="ch32v203g_r0_1v0" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=ch32v203g_r0_1v0" /> |       <configuration PROFILE_NAME="ch32v203g_r0_1v0" ENABLED="false" CONFIG_NAME="Debug" GENERATION_OPTIONS="-DBOARD=ch32v203g_r0_1v0" /> | ||||||
|       <configuration PROFILE_NAME="ch32v307v_r1_1v0" ENABLED="false" GENERATION_OPTIONS="-DBOARD=ch32v307v_r1_1v0 -DLOG=2" /> |       <configuration PROFILE_NAME="ch32v307v_r1_1v0" ENABLED="false" GENERATION_OPTIONS="-DBOARD=ch32v307v_r1_1v0 -DLOG=2" /> | ||||||
|       <configuration PROFILE_NAME="ch32v307v_r1_1v0 USBFS" ENABLED="false" GENERATION_OPTIONS="-DBOARD=ch32v307v_r1_1v0 -DSPEED=full" /> |       <configuration PROFILE_NAME="ch32v307v_r1_1v0 USBFS" ENABLED="false" GENERATION_OPTIONS="-DBOARD=ch32v307v_r1_1v0 -DSPEED=full" /> | ||||||
|   | |||||||
| @@ -399,6 +399,11 @@ function(family_flash_jlink TARGET) | |||||||
|     set(JLINK_IF swd) |     set(JLINK_IF swd) | ||||||
|   endif () |   endif () | ||||||
|  |  | ||||||
|  |   if (NOT DEFINED JLINK_OPTION) | ||||||
|  |     set(JLINK_OPTION "") | ||||||
|  |   endif () | ||||||
|  |   separate_arguments(OPTION_LIST UNIX_COMMAND ${JLINK_OPTION}) | ||||||
|  |  | ||||||
|   file(GENERATE |   file(GENERATE | ||||||
|     OUTPUT $<TARGET_FILE_DIR:${TARGET}>/${TARGET}.jlink |     OUTPUT $<TARGET_FILE_DIR:${TARGET}>/${TARGET}.jlink | ||||||
|     CONTENT "halt |     CONTENT "halt | ||||||
| @@ -410,7 +415,8 @@ exit" | |||||||
|  |  | ||||||
|   add_custom_target(${TARGET}-jlink |   add_custom_target(${TARGET}-jlink | ||||||
|     DEPENDS ${TARGET} |     DEPENDS ${TARGET} | ||||||
|     COMMAND ${JLINKEXE} -device ${JLINK_DEVICE} -if ${JLINK_IF} -JTAGConf -1,-1 -speed auto -CommandFile $<TARGET_FILE_DIR:${TARGET}>/${TARGET}.jlink |     COMMAND ${JLINKEXE} -device ${JLINK_DEVICE} ${OPTION_LIST} -if ${JLINK_IF} -JTAGConf -1,-1 -speed auto -CommandFile $<TARGET_FILE_DIR:${TARGET}>/${TARGET}.jlink | ||||||
|  |     VERBATIM | ||||||
|     ) |     ) | ||||||
| endfunction() | endfunction() | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										14
									
								
								hw/bsp/stm32f1/boards/stm32f103ze_iar/board.cmake
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								hw/bsp/stm32f1/boards/stm32f103ze_iar/board.cmake
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | |||||||
|  | set(MCU_VARIANT stm32f103xe) | ||||||
|  | set(JLINK_DEVICE stm32f103ze) | ||||||
|  |  | ||||||
|  | string(TOUPPER ${MCU_VARIANT} MCU_VARIANT_UPPER) | ||||||
|  |  | ||||||
|  | set(LD_FILE_GNU ${ST_CMSIS}/Source/Templates/gcc/linker/${MCU_VARIANT_UPPER}_FLASH.ld) | ||||||
|  | set(LD_FILE_IAR ${ST_CMSIS}/Source/Templates/iar/linker/${MCU_VARIANT}_flash.icf) | ||||||
|  |  | ||||||
|  | function(update_board TARGET) | ||||||
|  |   target_compile_definitions(${TARGET} PUBLIC | ||||||
|  |     STM32F103xE | ||||||
|  |     HSE_VALUE=8000000U | ||||||
|  |     ) | ||||||
|  | endfunction() | ||||||
							
								
								
									
										97
									
								
								hw/bsp/stm32f1/boards/stm32f103ze_iar/board.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										97
									
								
								hw/bsp/stm32f1/boards/stm32f103ze_iar/board.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,97 @@ | |||||||
|  | /* | ||||||
|  |  * 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. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | #ifndef BOARD_H_ | ||||||
|  | #define BOARD_H_ | ||||||
|  |  | ||||||
|  | #ifdef __cplusplus | ||||||
|  |  extern "C" { | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | // LED | ||||||
|  | #define LED_PORT              GPIOF | ||||||
|  | #define LED_PIN               GPIO_PIN_6 | ||||||
|  | #define LED_STATE_ON          1 | ||||||
|  |  | ||||||
|  | // Button | ||||||
|  | #define BUTTON_PORT           GPIOG | ||||||
|  | #define BUTTON_PIN            GPIO_PIN_8 | ||||||
|  | #define BUTTON_STATE_ACTIVE   0 | ||||||
|  |  | ||||||
|  | // USB Connect | ||||||
|  | #define USB_CONNECT_PORT      GPIOG | ||||||
|  | #define USB_CONNECT_PIN       GPIO_PIN_11 | ||||||
|  | #define USB_CONNECT_STATE     0 | ||||||
|  |  | ||||||
|  | // UART | ||||||
|  | //#define UART_DEV              USART1 | ||||||
|  | //#define UART_CLK_EN           __HAL_RCC_USART1_CLK_ENABLE | ||||||
|  | //#define UART_GPIO_PORT        GPIOA | ||||||
|  | //#define UART_GPIO_AF          GPIO_AF1_USART1 | ||||||
|  | //#define UART_TX_PIN           GPIO_PIN_9 | ||||||
|  | //#define UART_RX_PIN           GPIO_PIN_10 | ||||||
|  |  | ||||||
|  | //--------------------------------------------------------------------+ | ||||||
|  | // RCC Clock | ||||||
|  | //--------------------------------------------------------------------+ | ||||||
|  | static inline void board_stm32f1_clock_init(void) | ||||||
|  | { | ||||||
|  |   RCC_ClkInitTypeDef clkinitstruct = {0}; | ||||||
|  |   RCC_OscInitTypeDef oscinitstruct = {0}; | ||||||
|  |   RCC_PeriphCLKInitTypeDef rccperiphclkinit = {0}; | ||||||
|  |  | ||||||
|  |   /* Enable HSE Oscillator and activate PLL with HSE as source */ | ||||||
|  |   oscinitstruct.OscillatorType  = RCC_OSCILLATORTYPE_HSE; | ||||||
|  |   oscinitstruct.HSEState        = RCC_HSE_ON; | ||||||
|  |   oscinitstruct.HSEPredivValue  = RCC_HSE_PREDIV_DIV1; | ||||||
|  |   oscinitstruct.PLL.PLLMUL      = RCC_PLL_MUL9; | ||||||
|  |   oscinitstruct.PLL.PLLState    = RCC_PLL_ON; | ||||||
|  |   oscinitstruct.PLL.PLLSource   = RCC_PLLSOURCE_HSE; | ||||||
|  |   HAL_RCC_OscConfig(&oscinitstruct); | ||||||
|  |  | ||||||
|  |   /* USB clock selection */ | ||||||
|  |   rccperiphclkinit.PeriphClockSelection = RCC_PERIPHCLK_USB; | ||||||
|  |   rccperiphclkinit.UsbClockSelection = RCC_USBCLKSOURCE_PLL_DIV1_5; | ||||||
|  |   HAL_RCCEx_PeriphCLKConfig(&rccperiphclkinit); | ||||||
|  |  | ||||||
|  |   /* Select PLL as system clock source and configure the HCLK, PCLK1 and PCLK2 clocks dividers */ | ||||||
|  |   clkinitstruct.ClockType = (RCC_CLOCKTYPE_SYSCLK | RCC_CLOCKTYPE_HCLK | RCC_CLOCKTYPE_PCLK1 | RCC_CLOCKTYPE_PCLK2); | ||||||
|  |   clkinitstruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK; | ||||||
|  |   clkinitstruct.AHBCLKDivider = RCC_SYSCLK_DIV1; | ||||||
|  |   clkinitstruct.APB1CLKDivider = RCC_HCLK_DIV2; | ||||||
|  |   clkinitstruct.APB2CLKDivider = RCC_HCLK_DIV1; | ||||||
|  |   HAL_RCC_ClockConfig(&clkinitstruct, FLASH_LATENCY_2); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static inline void board_vbus_sense_init(void) | ||||||
|  | { | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #ifdef __cplusplus | ||||||
|  |  } | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | #endif /* BOARD_H_ */ | ||||||
							
								
								
									
										13
									
								
								hw/bsp/stm32f1/boards/stm32f103ze_iar/board.mk
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								hw/bsp/stm32f1/boards/stm32f103ze_iar/board.mk
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | |||||||
|  | MCU_VARIANT = stm32f103xe | ||||||
|  |  | ||||||
|  | CFLAGS += -DSTM32F103xE -DHSE_VALUE=8000000U | ||||||
|  |  | ||||||
|  | # Linker | ||||||
|  | LD_FILE_GCC = ${ST_CMSIS}/Source/Templates/gcc/linker/STM32F103XE_FLASH.ld | ||||||
|  | LD_FILE_IAR = ${ST_CMSIS}/Source/Templates/iar/linker/${MCU_VARIANT}_flash.icf | ||||||
|  |  | ||||||
|  | # For flash-jlink target | ||||||
|  | JLINK_DEVICE = stm32f103ze | ||||||
|  |  | ||||||
|  | # flash target ROM bootloader | ||||||
|  | flash: flash-jlink | ||||||
| @@ -57,6 +57,19 @@ void board_init(void) { | |||||||
|   __HAL_RCC_GPIOC_CLK_ENABLE(); |   __HAL_RCC_GPIOC_CLK_ENABLE(); | ||||||
|   __HAL_RCC_GPIOD_CLK_ENABLE(); |   __HAL_RCC_GPIOD_CLK_ENABLE(); | ||||||
|  |  | ||||||
|  | #ifdef __HAL_RCC_GPIOE_CLK_ENABLE | ||||||
|  |   __HAL_RCC_GPIOE_CLK_ENABLE(); | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | #ifdef __HAL_RCC_GPIOF_CLK_ENABLE | ||||||
|  |   __HAL_RCC_GPIOF_CLK_ENABLE(); | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | #ifdef __HAL_RCC_GPIOG_CLK_ENABLE | ||||||
|  |   __HAL_RCC_GPIOG_CLK_ENABLE(); | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  |  | ||||||
| #if CFG_TUSB_OS == OPT_OS_NONE | #if CFG_TUSB_OS == OPT_OS_NONE | ||||||
|   // 1ms tick timer |   // 1ms tick timer | ||||||
|   SysTick_Config(SystemCoreClock / 1000); |   SysTick_Config(SystemCoreClock / 1000); | ||||||
| @@ -107,10 +120,18 @@ void board_init(void) { | |||||||
|   HAL_UART_Init(&UartHandle); |   HAL_UART_Init(&UartHandle); | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
|  | #ifdef USB_CONNECT_PIN | ||||||
|  |   GPIO_InitStruct.Pin = USB_CONNECT_PIN; | ||||||
|  |   GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP; | ||||||
|  |   GPIO_InitStruct.Pull = GPIO_NOPULL; | ||||||
|  |   GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH; | ||||||
|  |   HAL_GPIO_Init(USB_CONNECT_PORT, &GPIO_InitStruct); | ||||||
|  | #endif | ||||||
|  |  | ||||||
|   // USB Pins |   // USB Pins | ||||||
|   // Configure USB DM and DP pins. |   // Configure USB DM and DP pins. | ||||||
|   GPIO_InitStruct.Pin = (GPIO_PIN_11 | GPIO_PIN_12); |   GPIO_InitStruct.Pin = (GPIO_PIN_11 | GPIO_PIN_12); | ||||||
|   GPIO_InitStruct.Mode = GPIO_MODE_INPUT; |   GPIO_InitStruct.Mode = GPIO_MODE_AF_PP; | ||||||
|   GPIO_InitStruct.Pull = GPIO_NOPULL; |   GPIO_InitStruct.Pull = GPIO_NOPULL; | ||||||
|   GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH; |   GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH; | ||||||
|   HAL_GPIO_Init(GPIOA, &GPIO_InitStruct); |   HAL_GPIO_Init(GPIOA, &GPIO_InitStruct); | ||||||
| @@ -119,6 +140,18 @@ void board_init(void) { | |||||||
|   __HAL_RCC_USB_CLK_ENABLE(); |   __HAL_RCC_USB_CLK_ENABLE(); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | #ifdef USB_CONNECT_PIN | ||||||
|  | void dcd_disconnect(uint8_t rhport) { | ||||||
|  |   (void)rhport; | ||||||
|  |   HAL_GPIO_WritePin(USB_CONNECT_PORT, USB_CONNECT_PIN, 1-USB_CONNECT_STATE); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void dcd_connect(uint8_t rhport) { | ||||||
|  |   (void)rhport; | ||||||
|  |   HAL_GPIO_WritePin(USB_CONNECT_PORT, USB_CONNECT_PIN, USB_CONNECT_STATE); | ||||||
|  | } | ||||||
|  | #endif | ||||||
|  |  | ||||||
| //--------------------------------------------------------------------+ | //--------------------------------------------------------------------+ | ||||||
| // Board porting API | // Board porting API | ||||||
| //--------------------------------------------------------------------+ | //--------------------------------------------------------------------+ | ||||||
|   | |||||||
| @@ -1,10 +1,10 @@ | |||||||
| ST_FAMILY = f1 | ST_FAMILY = f1 | ||||||
| DEPS_SUBMODULES += lib/CMSIS_5 hw/mcu/st/cmsis_device_$(ST_FAMILY) hw/mcu/st/stm32$(ST_FAMILY)xx_hal_driver | DEPS_SUBMODULES += lib/CMSIS_5 hw/mcu/st/cmsis_device_${ST_FAMILY} hw/mcu/st/stm32${ST_FAMILY}xx_hal_driver | ||||||
|  |  | ||||||
| ST_CMSIS = hw/mcu/st/cmsis_device_$(ST_FAMILY) | ST_CMSIS = hw/mcu/st/cmsis_device_${ST_FAMILY} | ||||||
| ST_HAL_DRIVER = hw/mcu/st/stm32$(ST_FAMILY)xx_hal_driver | ST_HAL_DRIVER = hw/mcu/st/stm32${ST_FAMILY}xx_hal_driver | ||||||
|  |  | ||||||
| include $(TOP)/$(BOARD_PATH)/board.mk | include ${TOP}/${BOARD_PATH}/board.mk | ||||||
| CPU_CORE ?= cortex-m3 | CPU_CORE ?= cortex-m3 | ||||||
|  |  | ||||||
| # -------------- | # -------------- | ||||||
| @@ -29,23 +29,23 @@ LDFLAGS_GCC += \ | |||||||
| # ------------------------ | # ------------------------ | ||||||
| SRC_C += \ | SRC_C += \ | ||||||
|   src/portable/st/stm32_fsdev/dcd_stm32_fsdev.c \ |   src/portable/st/stm32_fsdev/dcd_stm32_fsdev.c \ | ||||||
|   $(ST_CMSIS)/Source/Templates/system_stm32$(ST_FAMILY)xx.c \ |   ${ST_CMSIS}/Source/Templates/system_stm32${ST_FAMILY}xx.c \ | ||||||
|   $(ST_HAL_DRIVER)/Src/stm32$(ST_FAMILY)xx_hal.c \ |   ${ST_HAL_DRIVER}/Src/stm32${ST_FAMILY}xx_hal.c \ | ||||||
|   $(ST_HAL_DRIVER)/Src/stm32$(ST_FAMILY)xx_hal_cortex.c \ |   ${ST_HAL_DRIVER}/Src/stm32${ST_FAMILY}xx_hal_cortex.c \ | ||||||
|   $(ST_HAL_DRIVER)/Src/stm32$(ST_FAMILY)xx_hal_rcc.c \ |   ${ST_HAL_DRIVER}/Src/stm32${ST_FAMILY}xx_hal_rcc.c \ | ||||||
|   $(ST_HAL_DRIVER)/Src/stm32$(ST_FAMILY)xx_hal_rcc_ex.c \ |   ${ST_HAL_DRIVER}/Src/stm32${ST_FAMILY}xx_hal_rcc_ex.c \ | ||||||
|   $(ST_HAL_DRIVER)/Src/stm32$(ST_FAMILY)xx_hal_gpio.c \ |   ${ST_HAL_DRIVER}/Src/stm32${ST_FAMILY}xx_hal_gpio.c \ | ||||||
|   $(ST_HAL_DRIVER)/Src/stm32$(ST_FAMILY)xx_hal_uart.c |   ${ST_HAL_DRIVER}/Src/stm32${ST_FAMILY}xx_hal_uart.c | ||||||
|  |  | ||||||
| INC += \ | INC += \ | ||||||
|   $(TOP)/$(BOARD_PATH) \ |   ${TOP}/${BOARD_PATH} \ | ||||||
|   $(TOP)/lib/CMSIS_5/CMSIS/Core/Include \ |   ${TOP}/lib/CMSIS_5/CMSIS/Core/Include \ | ||||||
|   $(TOP)/$(ST_CMSIS)/Include \ |   ${TOP}/${ST_CMSIS}/Include \ | ||||||
|   $(TOP)/$(ST_HAL_DRIVER)/Inc |   ${TOP}/${ST_HAL_DRIVER}/Inc | ||||||
|  |  | ||||||
| # Startup | # Startup | ||||||
| SRC_S_GCC += $(ST_CMSIS)/Source/Templates/gcc/startup_$(MCU_VARIANT).s | SRC_S_GCC += ${ST_CMSIS}/Source/Templates/gcc/startup_${MCU_VARIANT}.s | ||||||
| SRC_S_IAR += $(ST_CMSIS)/Source/Templates/iar/startup_$(MCU_VARIANT).s | SRC_S_IAR += ${ST_CMSIS}/Source/Templates/iar/startup_${MCU_VARIANT}.s | ||||||
|  |  | ||||||
| # flash target ROM bootloader: flash-dfu-util | # flash target ROM bootloader: flash-dfu-util | ||||||
| DFU_UTIL_OPTION = -a 0 --dfuse-address 0x08000000 | DFU_UTIL_OPTION = -a 0 --dfuse-address 0x08000000 | ||||||
|   | |||||||
| @@ -1201,7 +1201,6 @@ TU_ATTR_FAST_FUNC void dcd_event_handler(dcd_event_t const* event, bool in_isr) | |||||||
|       break; |       break; | ||||||
|  |  | ||||||
|     case DCD_EVENT_SETUP_RECEIVED: |     case DCD_EVENT_SETUP_RECEIVED: | ||||||
|       // TU_ASSERT(event->setup_received.bRequest != 0,); // for catching issue with ch32v203 and windows with -O0/-Og |  | ||||||
|       _usbd_queued_setup++; |       _usbd_queued_setup++; | ||||||
|       send = true; |       send = true; | ||||||
|       break; |       break; | ||||||
|   | |||||||
| @@ -4,7 +4,6 @@ | |||||||
|  * Copyright (c) 2019 Nathan Conrad |  * Copyright (c) 2019 Nathan Conrad | ||||||
|  * |  * | ||||||
|  * Portions: |  * Portions: | ||||||
|  * Copyright (c) 2016 STMicroelectronics |  | ||||||
|  * Copyright (c) 2019 Ha Thach (tinyusb.org) |  * Copyright (c) 2019 Ha Thach (tinyusb.org) | ||||||
|  * Copyright (c) 2022 Simon Küppers (skuep) |  * Copyright (c) 2022 Simon Küppers (skuep) | ||||||
|  * Copyright (c) 2022 HiFiPhile |  * Copyright (c) 2022 HiFiPhile | ||||||
| @@ -114,8 +113,6 @@ | |||||||
| #include "device/dcd.h" | #include "device/dcd.h" | ||||||
|  |  | ||||||
| #if defined(TUP_USBIP_FSDEV_STM32) | #if defined(TUP_USBIP_FSDEV_STM32) | ||||||
|   // Undefine to reduce the dependence on HAL |  | ||||||
|   #undef USE_HAL_DRIVER |  | ||||||
|   #include "fsdev_stm32.h" |   #include "fsdev_stm32.h" | ||||||
| #elif defined(TUP_USBIP_FSDEV_CH32) | #elif defined(TUP_USBIP_FSDEV_CH32) | ||||||
|   #include "fsdev_ch32.h" |   #include "fsdev_ch32.h" | ||||||
| @@ -125,12 +122,6 @@ | |||||||
|  |  | ||||||
| #include "fsdev_type.h" | #include "fsdev_type.h" | ||||||
|  |  | ||||||
| //--------------------------------------------------------------------+ |  | ||||||
| // Configuration |  | ||||||
| //--------------------------------------------------------------------+ |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| //--------------------------------------------------------------------+ | //--------------------------------------------------------------------+ | ||||||
| // MACRO CONSTANT TYPEDEF | // MACRO CONSTANT TYPEDEF | ||||||
| //--------------------------------------------------------------------+ | //--------------------------------------------------------------------+ | ||||||
| @@ -155,7 +146,6 @@ typedef struct { | |||||||
|  |  | ||||||
| static xfer_ctl_t xfer_status[CFG_TUD_ENDPPOINT_MAX][2]; | static xfer_ctl_t xfer_status[CFG_TUD_ENDPPOINT_MAX][2]; | ||||||
| static ep_alloc_t ep_alloc_status[FSDEV_EP_COUNT]; | static ep_alloc_t ep_alloc_status[FSDEV_EP_COUNT]; | ||||||
|  |  | ||||||
| static uint8_t remoteWakeCountdown; // When wake is requested | static uint8_t remoteWakeCountdown; // When wake is requested | ||||||
|  |  | ||||||
| //--------------------------------------------------------------------+ | //--------------------------------------------------------------------+ | ||||||
| @@ -179,6 +169,10 @@ static bool dcd_read_packet_memory_ff(tu_fifo_t *ff, uint16_t src, uint16_t wNBy | |||||||
|  |  | ||||||
| static void edpt0_open(uint8_t rhport); | static void edpt0_open(uint8_t rhport); | ||||||
|  |  | ||||||
|  | TU_ATTR_ALWAYS_INLINE static inline void edpt0_prepare_setup(void) { | ||||||
|  |    btable_set_rx_bufsize(0, BTABLE_BUF_RX, 8); | ||||||
|  | } | ||||||
|  |  | ||||||
| //--------------------------------------------------------------------+ | //--------------------------------------------------------------------+ | ||||||
| // Inline helper | // Inline helper | ||||||
| //--------------------------------------------------------------------+ | //--------------------------------------------------------------------+ | ||||||
| @@ -220,10 +214,11 @@ void dcd_init(uint8_t rhport) { | |||||||
|   // Reset endpoints to disabled |   // Reset endpoints to disabled | ||||||
|   for (uint32_t i = 0; i < FSDEV_EP_COUNT; i++) { |   for (uint32_t i = 0; i < FSDEV_EP_COUNT; i++) { | ||||||
|     // This doesn't clear all bits since some bits are "toggle", but does set the type to DISABLED. |     // This doesn't clear all bits since some bits are "toggle", but does set the type to DISABLED. | ||||||
|     ep_write(i, 0u); |     ep_write(i, 0u, false); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   FSDEV_REG->CNTR |= USB_CNTR_RESETM | USB_CNTR_ESOFM | USB_CNTR_CTRM | USB_CNTR_SUSPM | USB_CNTR_WKUPM; |   FSDEV_REG->CNTR |= USB_CNTR_RESETM | USB_CNTR_ESOFM | USB_CNTR_CTRM | | ||||||
|  |       USB_CNTR_SUSPM | USB_CNTR_WKUPM | USB_CNTR_PMAOVRM; | ||||||
|   handle_bus_reset(rhport); |   handle_bus_reset(rhport); | ||||||
|  |  | ||||||
|   // Enable pull-up if supported |   // Enable pull-up if supported | ||||||
| @@ -279,13 +274,10 @@ static void handle_bus_reset(uint8_t rhport) { | |||||||
|  |  | ||||||
| // Handle CTR interrupt for the TX/IN direction | // Handle CTR interrupt for the TX/IN direction | ||||||
| static void handle_ctr_tx(uint32_t ep_id) { | static void handle_ctr_tx(uint32_t ep_id) { | ||||||
|   uint32_t ep_reg = ep_read(ep_id) & USB_EPREG_MASK; |   uint32_t ep_reg = ep_read(ep_id) | USB_EP_CTR_TX | USB_EP_CTR_RX; | ||||||
|  |   ep_reg &= USB_EPREG_MASK; | ||||||
|   // Verify the CTR bit is set. This was in the ST Micro code, but I'm not sure it's actually necessary? |  | ||||||
|   TU_VERIFY(ep_reg & USB_EP_CTR_TX, ); |  | ||||||
|  |  | ||||||
|   uint8_t const ep_num = ep_reg & USB_EPADDR_FIELD; |   uint8_t const ep_num = ep_reg & USB_EPADDR_FIELD; | ||||||
|   uint8_t ep_addr = (ep_reg & USB_EPADDR_FIELD) | TUSB_DIR_IN_MASK; |  | ||||||
|   xfer_ctl_t *xfer = xfer_ctl_ptr(ep_num, TUSB_DIR_IN); |   xfer_ctl_t *xfer = xfer_ctl_ptr(ep_num, TUSB_DIR_IN); | ||||||
|  |  | ||||||
|   if (ep_is_iso(ep_reg)) { |   if (ep_is_iso(ep_reg)) { | ||||||
| @@ -301,123 +293,79 @@ static void handle_ctr_tx(uint32_t ep_id) { | |||||||
|   } |   } | ||||||
|  |  | ||||||
|   if (xfer->total_len != xfer->queued_len) { |   if (xfer->total_len != xfer->queued_len) { | ||||||
|     dcd_transmit_packet(xfer, ep_id); // also clear CTR bit |     dcd_transmit_packet(xfer, ep_id); | ||||||
|   } else { |   } else { | ||||||
|     dcd_event_xfer_complete(0, ep_addr, xfer->total_len, XFER_RESULT_SUCCESS, true); |     dcd_event_xfer_complete(0, ep_num | TUSB_DIR_IN_MASK, xfer->queued_len, XFER_RESULT_SUCCESS, true); | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|     // Clear CTR TX and reserved CTR RX | static void handle_ctr_setup(uint32_t ep_id) { | ||||||
|     ep_reg = (ep_reg & ~USB_EP_CTR_TX) | USB_EP_CTR_RX; |   uint16_t rx_count = btable_get_count(ep_id, BTABLE_BUF_RX); | ||||||
|  |   uint16_t rx_addr = btable_get_addr(ep_id, BTABLE_BUF_RX); | ||||||
|  |   uint8_t setup_packet[8] TU_ATTR_ALIGNED(4); | ||||||
|  |  | ||||||
|     ep_write(ep_id, ep_reg); |   dcd_read_packet_memory(setup_packet, rx_addr, rx_count); | ||||||
|  |  | ||||||
|  |   // Clear CTR RX if another setup packet arrived before this, it will be discarded | ||||||
|  |   ep_write_clear_ctr(ep_id, TUSB_DIR_OUT); | ||||||
|  |  | ||||||
|  |   // Setup packet should always be 8 bytes. If not, we probably missed the packet | ||||||
|  |   if (rx_count == 8) { | ||||||
|  |     dcd_event_setup_received(0, (uint8_t*) setup_packet, true); | ||||||
|  |     // Hardware should reset EP0 RX/TX to NAK and both toggle to 1 | ||||||
|  |   } else { | ||||||
|  |     // Missed setup packet !!! | ||||||
|  |     TU_BREAKPOINT(); | ||||||
|  |     edpt0_prepare_setup(); | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| // Handle CTR interrupt for the RX/OUT direction | // Handle CTR interrupt for the RX/OUT direction | ||||||
| static void handle_ctr_rx(uint32_t ep_id) { | static void handle_ctr_rx(uint32_t ep_id) { | ||||||
| #ifdef FSDEV_BUS_32BIT |   uint32_t ep_reg = ep_read(ep_id) | USB_EP_CTR_TX | USB_EP_CTR_RX; | ||||||
|   /* https://www.st.com/resource/en/errata_sheet/es0561-stm32h503cbebkbrb-device-errata-stmicroelectronics.pdf |  | ||||||
|    * From STM32H503 errata 2.15.1: Buffer description table update completes after CTR interrupt triggers |  | ||||||
|    * Description: |  | ||||||
|    * - During OUT transfers, the correct transfer interrupt (CTR) is triggered a little before the last USB SRAM accesses |  | ||||||
|    * have completed. If the software responds quickly to the interrupt, the full buffer contents may not be correct. |  | ||||||
|    * Workaround: |  | ||||||
|    * - Software should ensure that a small delay is included before accessing the SRAM contents. This delay |  | ||||||
|    * should be 800 ns in Full Speed mode and 6.4 μs in Low Speed mode |  | ||||||
|    * - Since H5 can run up to 250Mhz -> 1 cycle = 4ns. Per errata, we need to wait 200 cycles. Though executing code |  | ||||||
|    * also takes time, so we'll wait 60 cycles (count = 20). |  | ||||||
|    * - Since Low Speed mode is not supported/popular, we will ignore it for now. |  | ||||||
|    * |  | ||||||
|    * Note: this errata also seems to apply to G0, U5, H5 etc. |  | ||||||
|    */ |  | ||||||
|   volatile uint32_t cycle_count = 20; // defined as PCD_RX_PMA_CNT in stm32 hal_driver |  | ||||||
|   while (cycle_count > 0U) { |  | ||||||
|     cycle_count--; // each count take 3 cycles (1 for sub, jump, and compare) |  | ||||||
|   } |  | ||||||
| #endif |  | ||||||
|  |  | ||||||
|   uint32_t ep_reg = ep_read(ep_id); |  | ||||||
|  |  | ||||||
|   // Verify the CTR bit is set. This was in the ST Micro code, but I'm not sure it's actually necessary? |  | ||||||
|   TU_VERIFY(ep_reg & USB_EP_CTR_RX, ); |  | ||||||
|   ep_reg = (ep_reg & ~USB_EP_CTR_RX) | USB_EP_CTR_TX; // Clear CTR RX and reserved CTR TX |  | ||||||
|  |  | ||||||
|   uint8_t const ep_num = ep_reg & USB_EPADDR_FIELD; |   uint8_t const ep_num = ep_reg & USB_EPADDR_FIELD; | ||||||
|  |   bool const is_iso = ep_is_iso(ep_reg); | ||||||
|  |   xfer_ctl_t* xfer = xfer_ctl_ptr(ep_num, TUSB_DIR_OUT); | ||||||
|  |  | ||||||
|   if (ep_reg & USB_EP_SETUP) { |   uint8_t buf_id; | ||||||
|     uint32_t count = btable_get_count(ep_id, BTABLE_BUF_RX); |   if (is_iso) { | ||||||
|     // Setup packet should always be 8 bytes. If not, ignore it, and try again. |     buf_id = (ep_reg & USB_EP_DTOG_RX) ? 0 : 1; // ISO are double buffered | ||||||
|     if (count == 8) { |  | ||||||
|       uint16_t rx_addr = btable_get_addr(ep_id, BTABLE_BUF_RX); |  | ||||||
|       uint32_t setup_packet[2]; |  | ||||||
|       dcd_read_packet_memory(setup_packet, rx_addr, 8); |  | ||||||
|       dcd_event_setup_received(0, (uint8_t*) setup_packet, true); |  | ||||||
|  |  | ||||||
|       // Reset EP to NAK (in case it had been stalling) |  | ||||||
|       ep_reg = ep_add_status(ep_reg, TUSB_DIR_IN, EP_STAT_NAK); |  | ||||||
|       ep_reg = ep_add_status(ep_reg, TUSB_DIR_OUT, EP_STAT_NAK); |  | ||||||
|  |  | ||||||
|       ep_reg = ep_add_dtog(ep_reg, TUSB_DIR_IN, 1); |  | ||||||
|       ep_reg = ep_add_dtog(ep_reg, TUSB_DIR_OUT, 1); |  | ||||||
|     } else { |  | ||||||
|       ep_reg &= USB_EPREG_MASK; // reversed all toggle |  | ||||||
|     } |  | ||||||
|   } else { |   } else { | ||||||
|     ep_reg &= USB_EPRX_STAT | USB_EPREG_MASK; // reversed all toggle except RX Status |     buf_id = BTABLE_BUF_RX; | ||||||
|  |  | ||||||
|     bool const is_iso = ep_is_iso(ep_reg); |  | ||||||
|     xfer_ctl_t *xfer = xfer_ctl_ptr(ep_num, TUSB_DIR_OUT); |  | ||||||
|  |  | ||||||
|     uint8_t buf_id; |  | ||||||
|     if (is_iso) { |  | ||||||
|       buf_id = (ep_reg & USB_EP_DTOG_RX) ? 0 : 1; // ISO are double buffered |  | ||||||
|     } else { |  | ||||||
|       buf_id = BTABLE_BUF_RX; |  | ||||||
|     } |  | ||||||
|     uint32_t rx_count = btable_get_count(ep_id, buf_id); |  | ||||||
|     uint16_t pma_addr = (uint16_t) btable_get_addr(ep_id, buf_id); |  | ||||||
|  |  | ||||||
|     if (rx_count != 0) { |  | ||||||
|       if (xfer->ff) { |  | ||||||
|         dcd_read_packet_memory_ff(xfer->ff, pma_addr, rx_count); |  | ||||||
|       } else { |  | ||||||
|         dcd_read_packet_memory(xfer->buffer + xfer->queued_len, pma_addr, rx_count); |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       xfer->queued_len = (uint16_t)(xfer->queued_len + rx_count); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     if ((rx_count < xfer->max_packet_size) || (xfer->queued_len == xfer->total_len)) { |  | ||||||
|       uint8_t const ep_addr = ep_num; |  | ||||||
|       // all bytes received or short packet |  | ||||||
|       dcd_event_xfer_complete(0, ep_addr, xfer->queued_len, XFER_RESULT_SUCCESS, true); |  | ||||||
|  |  | ||||||
|       if (ep_num == 0) { |  | ||||||
|         // prepared for status packet |  | ||||||
|         btable_set_rx_bufsize(ep_id, BTABLE_BUF_RX, CFG_TUD_ENDPOINT0_SIZE); |  | ||||||
|       } |  | ||||||
|       ep_reg = ep_add_status(ep_reg, TUSB_DIR_OUT, EP_STAT_NAK); |  | ||||||
|     } else { |  | ||||||
|       // Set endpoint active again for receiving more data. Note that isochronous endpoints stay active always |  | ||||||
|       if (!is_iso) { |  | ||||||
|         uint16_t const cnt = tu_min16(xfer->total_len - xfer->queued_len, xfer->max_packet_size); |  | ||||||
|         btable_set_rx_bufsize(ep_id, BTABLE_BUF_RX, cnt); |  | ||||||
|       } |  | ||||||
|       ep_reg = ep_add_status(ep_reg, TUSB_DIR_OUT, EP_STAT_VALID); |  | ||||||
|     } |  | ||||||
|   } |   } | ||||||
|  |   uint16_t const rx_count = btable_get_count(ep_id, buf_id); | ||||||
|  |   uint16_t pma_addr = (uint16_t) btable_get_addr(ep_id, buf_id); | ||||||
|  |  | ||||||
|   ep_write(ep_id, ep_reg); |   if (xfer->ff) { | ||||||
|  |     dcd_read_packet_memory_ff(xfer->ff, pma_addr, rx_count); | ||||||
|  |   } else { | ||||||
|  |     dcd_read_packet_memory(xfer->buffer + xfer->queued_len, pma_addr, rx_count); | ||||||
|  |   } | ||||||
|  |   xfer->queued_len += rx_count; | ||||||
|  |  | ||||||
|  |   if ((rx_count < xfer->max_packet_size) || (xfer->queued_len >= xfer->total_len)) { | ||||||
|  |     // all bytes received or short packet | ||||||
|  |     dcd_event_xfer_complete(0, ep_num, xfer->queued_len, XFER_RESULT_SUCCESS, true); | ||||||
|  |  | ||||||
|  |     // For ch32v203: reset rx bufsize to mps to prevent race condition to cause PMAOVR (occurs with msc write10) | ||||||
|  |     // also ch32 seems to unconditionally accept ZLP on EP0 OUT, which can incorrectly use queued_len of previous | ||||||
|  |     // transfer. So reset total_len and queued_len to 0. | ||||||
|  |     btable_set_rx_bufsize(ep_id, BTABLE_BUF_RX, xfer->max_packet_size); | ||||||
|  |     xfer->total_len = xfer->queued_len = 0; | ||||||
|  |   } else { | ||||||
|  |     // Set endpoint active again for receiving more data. Note that isochronous endpoints stay active always | ||||||
|  |     if (!is_iso) { | ||||||
|  |       uint16_t const cnt = tu_min16(xfer->total_len - xfer->queued_len, xfer->max_packet_size); | ||||||
|  |       btable_set_rx_bufsize(ep_id, BTABLE_BUF_RX, cnt); | ||||||
|  |     } | ||||||
|  |     ep_reg &= USB_EPREG_MASK | EP_STAT_MASK(TUSB_DIR_OUT); // will change RX Status, reserved other toggle bits | ||||||
|  |     ep_change_status(&ep_reg, TUSB_DIR_OUT, EP_STAT_VALID); | ||||||
|  |     ep_write(ep_id, ep_reg, false); | ||||||
|  |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| void dcd_int_handler(uint8_t rhport) { | void dcd_int_handler(uint8_t rhport) { | ||||||
|   uint32_t int_status = FSDEV_REG->ISTR; |   uint32_t int_status = FSDEV_REG->ISTR; | ||||||
|   // const uint32_t handled_ints = USB_ISTR_CTR | USB_ISTR_RESET | USB_ISTR_WKUP |  | ||||||
|   //     | USB_ISTR_SUSP | USB_ISTR_SOF | USB_ISTR_ESOF; |  | ||||||
|   //  unused IRQs: (USB_ISTR_PMAOVR | USB_ISTR_ERR | USB_ISTR_L1REQ ) |  | ||||||
|  |  | ||||||
|   // The ST driver loops here on the CTR bit, but that loop has been moved into the |  | ||||||
|   // dcd_ep_ctr_handler(), so less need to loop here. The other interrupts shouldn't |  | ||||||
|   // be triggered repeatedly. |  | ||||||
|  |  | ||||||
|   /* Put SOF flag at the beginning of ISR in case to get least amount of jitter if it is used for timing purposes */ |   /* Put SOF flag at the beginning of ISR in case to get least amount of jitter if it is used for timing purposes */ | ||||||
|   if (int_status & USB_ISTR_SOF) { |   if (int_status & USB_ISTR_SOF) { | ||||||
| @@ -464,17 +412,52 @@ void dcd_int_handler(uint8_t rhport) { | |||||||
|     FSDEV_REG->ISTR = (fsdev_bus_t)~USB_ISTR_ESOF; |     FSDEV_REG->ISTR = (fsdev_bus_t)~USB_ISTR_ESOF; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   // loop to handle all pending CTR interrupts |   if (int_status & USB_ISTR_PMAOVR) { | ||||||
|   while (int_status & USB_ISTR_CTR) { |     TU_BREAKPOINT(); | ||||||
|     uint32_t const ep_id = int_status & USB_ISTR_EP_ID; |     FSDEV_REG->ISTR = (fsdev_bus_t)~USB_ISTR_PMAOVR; | ||||||
|  |   } | ||||||
|  |  | ||||||
|     if ((int_status & USB_ISTR_DIR) == 0U) { |   // loop to handle all pending CTR interrupts | ||||||
|       handle_ctr_tx(ep_id); // TX/IN |   while (FSDEV_REG->ISTR & USB_ISTR_CTR) { | ||||||
|     } else { |     // skip DIR bit, and use CTR TX/RX instead, since there is chance we have both TX/RX completed in one interrupt | ||||||
|       handle_ctr_rx(ep_id); // RX/OUT or both (RX/TX !!) |     uint32_t const ep_id = FSDEV_REG->ISTR & USB_ISTR_EP_ID; | ||||||
|  |     uint32_t const ep_reg = ep_read(ep_id); | ||||||
|  |  | ||||||
|  |     if (ep_reg & USB_EP_CTR_RX) { | ||||||
|  |       #ifdef FSDEV_BUS_32BIT | ||||||
|  |       /* https://www.st.com/resource/en/errata_sheet/es0561-stm32h503cbebkbrb-device-errata-stmicroelectronics.pdf | ||||||
|  |        * https://www.st.com/resource/en/errata_sheet/es0587-stm32u535xx-and-stm32u545xx-device-errata-stmicroelectronics.pdf | ||||||
|  |        * From H503/U535 errata: Buffer description table update completes after CTR interrupt triggers | ||||||
|  |        * Description: | ||||||
|  |        * - During OUT transfers, the correct transfer interrupt (CTR) is triggered a little before the last USB SRAM accesses | ||||||
|  |        * have completed. If the software responds quickly to the interrupt, the full buffer contents may not be correct. | ||||||
|  |        * Workaround: | ||||||
|  |        * - Software should ensure that a small delay is included before accessing the SRAM contents. This delay | ||||||
|  |        * should be 800 ns in Full Speed mode and 6.4 μs in Low Speed mode | ||||||
|  |        * - Since H5 can run up to 250Mhz -> 1 cycle = 4ns. Per errata, we need to wait 200 cycles. Though executing code | ||||||
|  |        * also takes time, so we'll wait 60 cycles (count = 20). | ||||||
|  |        * - Since Low Speed mode is not supported/popular, we will ignore it for now. | ||||||
|  |        * | ||||||
|  |        * Note: this errata may also apply to G0, U5, H5 etc. | ||||||
|  |        */ | ||||||
|  |       volatile uint32_t cycle_count = 20; // defined as PCD_RX_PMA_CNT in stm32 hal_driver | ||||||
|  |       while (cycle_count > 0U) { | ||||||
|  |         cycle_count--; // each count take 3 cycles (1 for sub, jump, and compare) | ||||||
|  |       } | ||||||
|  |       #endif | ||||||
|  |  | ||||||
|  |       if (ep_reg & USB_EP_SETUP) { | ||||||
|  |         handle_ctr_setup(ep_id); // CTR will be clear after copied setup packet | ||||||
|  |       } else { | ||||||
|  |         ep_write_clear_ctr(ep_id, TUSB_DIR_OUT); | ||||||
|  |         handle_ctr_rx(ep_id); | ||||||
|  |       } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     int_status = FSDEV_REG->ISTR; |     if (ep_reg & USB_EP_CTR_TX) { | ||||||
|  |       ep_write_clear_ctr(ep_id, TUSB_DIR_IN); | ||||||
|  |       handle_ctr_tx(ep_id); | ||||||
|  |     } | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -493,6 +476,8 @@ void dcd_edpt0_status_complete(uint8_t rhport, tusb_control_request_t const *req | |||||||
|     uint8_t const dev_addr = (uint8_t)request->wValue; |     uint8_t const dev_addr = (uint8_t)request->wValue; | ||||||
|     FSDEV_REG->DADDR = (USB_DADDR_EF | dev_addr); |     FSDEV_REG->DADDR = (USB_DADDR_EF | dev_addr); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   edpt0_prepare_setup(); | ||||||
| } | } | ||||||
|  |  | ||||||
| /*** | /*** | ||||||
| @@ -577,16 +562,14 @@ void edpt0_open(uint8_t rhport) { | |||||||
|   btable_set_addr(0, BTABLE_BUF_RX, pma_addr0); |   btable_set_addr(0, BTABLE_BUF_RX, pma_addr0); | ||||||
|   btable_set_addr(0, BTABLE_BUF_TX, pma_addr1); |   btable_set_addr(0, BTABLE_BUF_TX, pma_addr1); | ||||||
|  |  | ||||||
|   uint32_t ep_reg = FSDEV_REG->ep[0].reg & ~USB_EPREG_MASK; |   uint32_t ep_reg = ep_read(0) & ~USB_EPREG_MASK; // only get toggle bits | ||||||
|   ep_reg |= USB_EP_CONTROL; |   ep_reg |= USB_EP_CONTROL; | ||||||
|   ep_reg = ep_add_status(ep_reg, TUSB_DIR_IN, EP_STAT_NAK); |   ep_change_status(&ep_reg, TUSB_DIR_IN, EP_STAT_NAK); | ||||||
|   ep_reg = ep_add_status(ep_reg, TUSB_DIR_OUT, EP_STAT_NAK); |   ep_change_status(&ep_reg, TUSB_DIR_OUT, EP_STAT_NAK); | ||||||
|   // no need to explicitly set DTOG bits since we aren't masked DTOG bit |   // no need to explicitly set DTOG bits since we aren't masked DTOG bit | ||||||
|  |  | ||||||
|   // prepare for setup packet |   edpt0_prepare_setup(); // prepare for setup packet | ||||||
|   btable_set_rx_bufsize(0, BTABLE_BUF_RX, CFG_TUD_ENDPOINT0_SIZE); |   ep_write(0, ep_reg, false); | ||||||
|  |  | ||||||
|   ep_write(0, ep_reg); |  | ||||||
| } | } | ||||||
|  |  | ||||||
| bool dcd_edpt_open(uint8_t rhport, tusb_desc_endpoint_t const *desc_ep) { | bool dcd_edpt_open(uint8_t rhport, tusb_desc_endpoint_t const *desc_ep) { | ||||||
| @@ -598,8 +581,8 @@ bool dcd_edpt_open(uint8_t rhport, tusb_desc_endpoint_t const *desc_ep) { | |||||||
|   uint8_t const ep_idx = dcd_ep_alloc(ep_addr, desc_ep->bmAttributes.xfer); |   uint8_t const ep_idx = dcd_ep_alloc(ep_addr, desc_ep->bmAttributes.xfer); | ||||||
|   TU_ASSERT(ep_idx < FSDEV_EP_COUNT); |   TU_ASSERT(ep_idx < FSDEV_EP_COUNT); | ||||||
|  |  | ||||||
|   uint32_t ep_reg = FSDEV_REG->ep[ep_idx].reg & ~USB_EPREG_MASK; |   uint32_t ep_reg = ep_read(ep_idx) & ~USB_EPREG_MASK; | ||||||
|   ep_reg |= tu_edpt_number(ep_addr) | USB_EP_CTR_RX | USB_EP_CTR_TX; |   ep_reg |= tu_edpt_number(ep_addr) | USB_EP_CTR_TX | USB_EP_CTR_RX; | ||||||
|  |  | ||||||
|   // Set type |   // Set type | ||||||
|   switch (desc_ep->bmAttributes.xfer) { |   switch (desc_ep->bmAttributes.xfer) { | ||||||
| @@ -623,8 +606,8 @@ bool dcd_edpt_open(uint8_t rhport, tusb_desc_endpoint_t const *desc_ep) { | |||||||
|   xfer->max_packet_size = packet_size; |   xfer->max_packet_size = packet_size; | ||||||
|   xfer->ep_idx = ep_idx; |   xfer->ep_idx = ep_idx; | ||||||
|  |  | ||||||
|   ep_reg = ep_add_status(ep_reg, dir, EP_STAT_NAK); |   ep_change_status(&ep_reg, dir, EP_STAT_NAK); | ||||||
|   ep_reg = ep_add_dtog(ep_reg, dir, 0); |   ep_change_dtog(&ep_reg, dir, 0); | ||||||
|  |  | ||||||
|   // reserve other direction toggle bits |   // reserve other direction toggle bits | ||||||
|   if (dir == TUSB_DIR_IN) { |   if (dir == TUSB_DIR_IN) { | ||||||
| @@ -633,18 +616,17 @@ bool dcd_edpt_open(uint8_t rhport, tusb_desc_endpoint_t const *desc_ep) { | |||||||
|     ep_reg &= ~(USB_EPTX_STAT | USB_EP_DTOG_TX); |     ep_reg &= ~(USB_EPTX_STAT | USB_EP_DTOG_TX); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   ep_write(ep_idx, ep_reg); |   ep_write(ep_idx, ep_reg, true); | ||||||
|  |  | ||||||
|   return true; |   return true; | ||||||
| } | } | ||||||
|  |  | ||||||
| void dcd_edpt_close_all(uint8_t rhport) | void dcd_edpt_close_all(uint8_t rhport) { | ||||||
| { |   dcd_int_disable(rhport); | ||||||
|   (void)rhport; |  | ||||||
|  |  | ||||||
|   for (uint32_t i = 1; i < FSDEV_EP_COUNT; i++) { |   for (uint32_t i = 1; i < FSDEV_EP_COUNT; i++) { | ||||||
|     // Reset endpoint |     // Reset endpoint | ||||||
|     ep_write(i, 0); |     ep_write(i, 0, false); | ||||||
|     // Clear EP allocation status |     // Clear EP allocation status | ||||||
|     ep_alloc_status[i].ep_num = 0xFF; |     ep_alloc_status[i].ep_num = 0xFF; | ||||||
|     ep_alloc_status[i].ep_type = 0xFF; |     ep_alloc_status[i].ep_type = 0xFF; | ||||||
| @@ -652,6 +634,8 @@ void dcd_edpt_close_all(uint8_t rhport) | |||||||
|     ep_alloc_status[i].allocated[1] = false; |     ep_alloc_status[i].allocated[1] = false; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   dcd_int_enable(rhport); | ||||||
|  |  | ||||||
|   // Reset PMA allocation |   // Reset PMA allocation | ||||||
|   ep_buf_ptr = FSDEV_BTABLE_BASE + 8 * CFG_TUD_ENDPPOINT_MAX + 2 * CFG_TUD_ENDPOINT0_SIZE; |   ep_buf_ptr = FSDEV_BTABLE_BASE + 8 * CFG_TUD_ENDPPOINT_MAX + 2 * CFG_TUD_ENDPOINT0_SIZE; | ||||||
| } | } | ||||||
| @@ -692,14 +676,14 @@ bool dcd_edpt_iso_activate(uint8_t rhport, tusb_desc_endpoint_t const *desc_ep) | |||||||
|  |  | ||||||
|   xfer->max_packet_size = tu_edpt_packet_size(desc_ep); |   xfer->max_packet_size = tu_edpt_packet_size(desc_ep); | ||||||
|  |  | ||||||
|   uint32_t ep_reg = FSDEV_REG->ep[0].reg & ~USB_EPREG_MASK; |   uint32_t ep_reg = ep_read(ep_idx) & ~USB_EPREG_MASK; | ||||||
|   ep_reg |= tu_edpt_number(ep_addr) | USB_EP_ISOCHRONOUS | USB_EP_CTR_RX | USB_EP_CTR_TX; |   ep_reg |= tu_edpt_number(ep_addr) | USB_EP_ISOCHRONOUS | USB_EP_CTR_TX | USB_EP_CTR_RX; | ||||||
|   ep_reg = ep_add_status(ep_reg, TUSB_DIR_IN, EP_STAT_DISABLED); |   ep_change_status(&ep_reg, TUSB_DIR_IN, EP_STAT_DISABLED); | ||||||
|   ep_reg = ep_add_status(ep_reg, TUSB_DIR_OUT, EP_STAT_DISABLED); |   ep_change_status(&ep_reg, TUSB_DIR_OUT, EP_STAT_DISABLED); | ||||||
|   ep_reg = ep_add_dtog(ep_reg, dir, 0); |   ep_change_dtog(&ep_reg, dir, 0); | ||||||
|   ep_reg = ep_add_dtog(ep_reg, 1-dir, 1); |   ep_change_dtog(&ep_reg, 1 - dir, 1); | ||||||
|  |  | ||||||
|   ep_write(ep_idx, ep_reg); |   ep_write(ep_idx, ep_reg, true); | ||||||
|  |  | ||||||
|   return true; |   return true; | ||||||
| } | } | ||||||
| @@ -707,7 +691,9 @@ bool dcd_edpt_iso_activate(uint8_t rhport, tusb_desc_endpoint_t const *desc_ep) | |||||||
| // Currently, single-buffered, and only 64 bytes at a time (max) | // Currently, single-buffered, and only 64 bytes at a time (max) | ||||||
| static void dcd_transmit_packet(xfer_ctl_t *xfer, uint16_t ep_ix) { | static void dcd_transmit_packet(xfer_ctl_t *xfer, uint16_t ep_ix) { | ||||||
|   uint16_t len = tu_min16(xfer->total_len - xfer->queued_len, xfer->max_packet_size); |   uint16_t len = tu_min16(xfer->total_len - xfer->queued_len, xfer->max_packet_size); | ||||||
|   uint16_t ep_reg = ep_read(ep_ix) | EP_CTR_TXRX; |   uint32_t ep_reg = ep_read(ep_ix) | USB_EP_CTR_TX | USB_EP_CTR_RX; // reserve CTR | ||||||
|  |   ep_reg &= USB_EPREG_MASK | EP_STAT_MASK(TUSB_DIR_IN); // only change TX Status, reserve other toggle bits | ||||||
|  |  | ||||||
|   bool const is_iso = ep_is_iso(ep_reg); |   bool const is_iso = ep_is_iso(ep_reg); | ||||||
|  |  | ||||||
|   uint8_t buf_id; |   uint8_t buf_id; | ||||||
| @@ -717,25 +703,21 @@ static void dcd_transmit_packet(xfer_ctl_t *xfer, uint16_t ep_ix) { | |||||||
|     buf_id = BTABLE_BUF_TX; |     buf_id = BTABLE_BUF_TX; | ||||||
|   } |   } | ||||||
|   uint16_t addr_ptr = (uint16_t) btable_get_addr(ep_ix, buf_id); |   uint16_t addr_ptr = (uint16_t) btable_get_addr(ep_ix, buf_id); | ||||||
|   btable_set_count(ep_ix, buf_id, len); |  | ||||||
|  |  | ||||||
|   if (xfer->ff) { |   if (xfer->ff) { | ||||||
|     dcd_write_packet_memory_ff(xfer->ff, addr_ptr, len); |     dcd_write_packet_memory_ff(xfer->ff, addr_ptr, len); | ||||||
|   } else { |   } else { | ||||||
|     dcd_write_packet_memory(addr_ptr, &(xfer->buffer[xfer->queued_len]), len); |     dcd_write_packet_memory(addr_ptr, &(xfer->buffer[xfer->queued_len]), len); | ||||||
|   } |   } | ||||||
|   xfer->queued_len = (uint16_t)(xfer->queued_len + len); |   xfer->queued_len += len; | ||||||
|  |  | ||||||
|   ep_reg &= USB_EPREG_MASK | EP_STAT_MASK(TUSB_DIR_IN); |   btable_set_count(ep_ix, buf_id, len); | ||||||
|   ep_reg = ep_add_status(ep_reg, TUSB_DIR_IN, EP_STAT_VALID); |   ep_change_status(&ep_reg, TUSB_DIR_IN, EP_STAT_VALID); | ||||||
|   ep_reg = ep_clear_ctr(ep_reg, TUSB_DIR_IN); |  | ||||||
|  |  | ||||||
|   dcd_int_disable(0); |  | ||||||
|   ep_write(ep_ix, ep_reg); |  | ||||||
|   if (is_iso) { |   if (is_iso) { | ||||||
|     xfer->iso_in_sending = true; |     xfer->iso_in_sending = true; | ||||||
|   } |   } | ||||||
|   dcd_int_enable(0); |   ep_write(ep_ix, ep_reg, true); | ||||||
| } | } | ||||||
|  |  | ||||||
| static bool edpt_xfer(uint8_t rhport, uint8_t ep_num, uint8_t dir) { | static bool edpt_xfer(uint8_t rhport, uint8_t ep_num, uint8_t dir) { | ||||||
| @@ -747,10 +729,10 @@ static bool edpt_xfer(uint8_t rhport, uint8_t ep_num, uint8_t dir) { | |||||||
|   if (dir == TUSB_DIR_IN) { |   if (dir == TUSB_DIR_IN) { | ||||||
|     dcd_transmit_packet(xfer, ep_idx); |     dcd_transmit_packet(xfer, ep_idx); | ||||||
|   } else { |   } else { | ||||||
|     uint32_t cnt = (uint32_t) tu_min16(xfer->total_len, xfer->max_packet_size); |     uint32_t ep_reg = ep_read(ep_idx) | USB_EP_CTR_TX | USB_EP_CTR_RX; // reserve CTR | ||||||
|     uint16_t ep_reg = ep_read(ep_idx) | USB_EP_CTR_TX; |     ep_reg &= USB_EPREG_MASK | EP_STAT_MASK(dir); | ||||||
|     ep_reg &= USB_EPREG_MASK | EP_STAT_MASK(dir); // keep CTR TX, clear CTR RX |  | ||||||
|     ep_reg = ep_add_status(ep_reg, dir, EP_STAT_VALID); |     uint16_t cnt = tu_min16(xfer->total_len, xfer->max_packet_size); | ||||||
|  |  | ||||||
|     if (ep_is_iso(ep_reg)) { |     if (ep_is_iso(ep_reg)) { | ||||||
|       btable_set_rx_bufsize(ep_idx, 0, cnt); |       btable_set_rx_bufsize(ep_idx, 0, cnt); | ||||||
| @@ -759,7 +741,8 @@ static bool edpt_xfer(uint8_t rhport, uint8_t ep_num, uint8_t dir) { | |||||||
|       btable_set_rx_bufsize(ep_idx, BTABLE_BUF_RX, cnt); |       btable_set_rx_bufsize(ep_idx, BTABLE_BUF_RX, cnt); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     ep_write(ep_idx, ep_reg); |     ep_change_status(&ep_reg, dir, EP_STAT_VALID); | ||||||
|  |     ep_write(ep_idx, ep_reg, true); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   return true; |   return true; | ||||||
| @@ -798,12 +781,11 @@ void dcd_edpt_stall(uint8_t rhport, uint8_t ep_addr) { | |||||||
|   xfer_ctl_t *xfer = xfer_ctl_ptr(ep_num, dir); |   xfer_ctl_t *xfer = xfer_ctl_ptr(ep_num, dir); | ||||||
|   uint8_t const ep_idx = xfer->ep_idx; |   uint8_t const ep_idx = xfer->ep_idx; | ||||||
|  |  | ||||||
|   uint32_t ep_reg = ep_read(ep_idx); |   uint32_t ep_reg = ep_read(ep_idx) | USB_EP_CTR_TX | USB_EP_CTR_RX; // reserve CTR bits | ||||||
|   ep_reg |= USB_EP_CTR_RX | USB_EP_CTR_TX; // reserve CTR bits |  | ||||||
|   ep_reg &= USB_EPREG_MASK | EP_STAT_MASK(dir); |   ep_reg &= USB_EPREG_MASK | EP_STAT_MASK(dir); | ||||||
|   ep_reg = ep_add_status(ep_reg, dir, EP_STAT_STALL); |   ep_change_status(&ep_reg, dir, EP_STAT_STALL); | ||||||
|  |  | ||||||
|   ep_write(ep_idx, ep_reg); |   ep_write(ep_idx, ep_reg, true); | ||||||
| } | } | ||||||
|  |  | ||||||
| void dcd_edpt_clear_stall(uint8_t rhport, uint8_t ep_addr) { | void dcd_edpt_clear_stall(uint8_t rhport, uint8_t ep_addr) { | ||||||
| @@ -814,282 +796,173 @@ void dcd_edpt_clear_stall(uint8_t rhport, uint8_t ep_addr) { | |||||||
|   xfer_ctl_t *xfer = xfer_ctl_ptr(ep_num, dir); |   xfer_ctl_t *xfer = xfer_ctl_ptr(ep_num, dir); | ||||||
|   uint8_t const ep_idx = xfer->ep_idx; |   uint8_t const ep_idx = xfer->ep_idx; | ||||||
|  |  | ||||||
|   uint32_t ep_reg = ep_read(ep_idx) | EP_CTR_TXRX; |   uint32_t ep_reg = ep_read(ep_idx) | USB_EP_CTR_TX | USB_EP_CTR_RX; // reserve CTR bits | ||||||
|   ep_reg &= USB_EPREG_MASK | EP_STAT_MASK(dir) | EP_DTOG_MASK(dir); |   ep_reg &= USB_EPREG_MASK | EP_STAT_MASK(dir) | EP_DTOG_MASK(dir); | ||||||
|  |  | ||||||
|   if (!ep_is_iso(ep_reg)) { |   if (!ep_is_iso(ep_reg)) { | ||||||
|     ep_reg = ep_add_status(ep_reg, dir, EP_STAT_NAK); |     ep_change_status(&ep_reg, dir, EP_STAT_NAK); | ||||||
|   } |   } | ||||||
|   ep_reg = ep_add_dtog(ep_reg, dir, 0); // Reset to DATA0 |   ep_change_dtog(&ep_reg, dir, 0); // Reset to DATA0 | ||||||
|  |   ep_write(ep_idx, ep_reg, true); | ||||||
|   ep_write(ep_idx, ep_reg); |  | ||||||
| } | } | ||||||
|  |  | ||||||
| #ifdef FSDEV_BUS_32BIT | // Write to packet memory area (PMA) from user memory | ||||||
| static bool dcd_write_packet_memory(uint16_t dst, const void *__restrict src, uint16_t wNBytes) { | // - Packet memory must be either strictly 16-bit or 32-bit depending on FSDEV_BUS_32BIT | ||||||
|   const uint8_t *src8 = src; | // - Uses unaligned for RAM (since M0 cannot access unaligned address) | ||||||
|   volatile uint32_t *pma32 = (volatile uint32_t *)(USB_PMAADDR + dst); |  | ||||||
|  |  | ||||||
|   for (uint32_t n = wNBytes / 4; n > 0; --n) { |  | ||||||
|     *pma32++ = tu_unaligned_read32(src8); |  | ||||||
|     src8 += 4; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   uint16_t odd = wNBytes & 0x03; |  | ||||||
|   if (odd) { |  | ||||||
|     uint32_t wrVal = *src8; |  | ||||||
|     odd--; |  | ||||||
|  |  | ||||||
|     if (odd) { |  | ||||||
|       wrVal |= *++src8 << 8; |  | ||||||
|       odd--; |  | ||||||
|  |  | ||||||
|       if (odd) { |  | ||||||
|         wrVal |= *++src8 << 16; |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     *pma32 = wrVal; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   return true; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| static bool dcd_read_packet_memory(void *__restrict dst, uint16_t src, uint16_t wNBytes) { |  | ||||||
|   uint8_t *dst8 = dst; |  | ||||||
|   volatile uint32_t *src32 = (volatile uint32_t *)(USB_PMAADDR + src); |  | ||||||
|  |  | ||||||
|   for (uint32_t n = wNBytes / 4; n > 0; --n) { |  | ||||||
|     tu_unaligned_write32(dst8, *src32++); |  | ||||||
|     dst8 += 4; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   uint16_t odd = wNBytes & 0x03; |  | ||||||
|   if (odd) { |  | ||||||
|     uint32_t rdVal = *src32; |  | ||||||
|  |  | ||||||
|     *dst8 = tu_u32_byte0(rdVal); |  | ||||||
|     odd--; |  | ||||||
|  |  | ||||||
|     if (odd) { |  | ||||||
|       *++dst8 = tu_u32_byte1(rdVal); |  | ||||||
|       odd--; |  | ||||||
|  |  | ||||||
|       if (odd) { |  | ||||||
|         *++dst8 = tu_u32_byte2(rdVal); |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   return true; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| #else |  | ||||||
| // Packet buffer access can only be 8- or 16-bit. |  | ||||||
| /** |  | ||||||
|  * @brief Copy a buffer from user memory area to packet memory area (PMA). |  | ||||||
|  *        This uses un-aligned for user memory and 16-bit access for packet memory. |  | ||||||
|  * @param   dst, byte address in PMA; must be 16-bit aligned |  | ||||||
|  * @param   src pointer to user memory area. |  | ||||||
|  * @param   wPMABufAddr address into PMA. |  | ||||||
|  * @param   nbytes no. of bytes to be copied. |  | ||||||
|  * @retval None |  | ||||||
|  */ |  | ||||||
| static bool dcd_write_packet_memory(uint16_t dst, const void *__restrict src, uint16_t nbytes) { | static bool dcd_write_packet_memory(uint16_t dst, const void *__restrict src, uint16_t nbytes) { | ||||||
|   uint32_t n16 = (uint32_t)nbytes >> 1U; |   if (nbytes == 0) return true; | ||||||
|   const uint8_t *src8 = src; |   uint32_t n_write = nbytes / FSDEV_BUS_SIZE; | ||||||
|   fsdev_pma16_t* pma16 = (fsdev_pma16_t*) (USB_PMAADDR + FSDEV_PMA_STRIDE * dst); |  | ||||||
|  |  | ||||||
|   while (n16--) { |   fsdev_pma_buf_t* pma_buf = PMA_BUF_AT(dst); | ||||||
|     pma16->u16 = tu_unaligned_read16(src8); |   const uint8_t *src8 = src; | ||||||
|     src8 += 2; |  | ||||||
|     pma16++; |   while (n_write--) { | ||||||
|  |     pma_buf->value = fsdevbus_unaligned_read(src8); | ||||||
|  |     src8 += FSDEV_BUS_SIZE; | ||||||
|  |     pma_buf++; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   if (nbytes & 0x01) { |   // odd bytes e.g 1 for 16-bit or 1-3 for 32-bit | ||||||
|     pma16->u16 = (uint16_t) *src8; |   uint16_t odd = nbytes & (FSDEV_BUS_SIZE - 1); | ||||||
|  |   if (odd) { | ||||||
|  |     fsdev_bus_t temp = 0; | ||||||
|  |     for(uint16_t i = 0; i < odd; i++) { | ||||||
|  |       temp |= *src8++ << (i * 8); | ||||||
|  |     } | ||||||
|  |     pma_buf->value = temp; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   return true; |   return true; | ||||||
| } | } | ||||||
|  |  | ||||||
| /** | // Read from packet memory area (PMA) to user memory. | ||||||
|  * @brief Copy a buffer from packet memory area (PMA) to user memory area. | // - Packet memory must be either strictly 16-bit or 32-bit depending on FSDEV_BUS_32BIT | ||||||
|  *        Uses unaligned for system memory and 16-bit access of packet memory | // - Uses unaligned for RAM (since M0 cannot access unaligned address) | ||||||
|  * @param   nbytes no. of bytes to be copied. |  | ||||||
|  * @retval None |  | ||||||
|  */ |  | ||||||
| static bool dcd_read_packet_memory(void *__restrict dst, uint16_t src, uint16_t nbytes) { | static bool dcd_read_packet_memory(void *__restrict dst, uint16_t src, uint16_t nbytes) { | ||||||
|   uint32_t n16 = (uint32_t)nbytes >> 1U; |   if (nbytes == 0) return true; | ||||||
|   fsdev_pma16_t* pma16 = (fsdev_pma16_t*) (USB_PMAADDR + FSDEV_PMA_STRIDE * src); |   uint32_t n_read = nbytes / FSDEV_BUS_SIZE; | ||||||
|  |  | ||||||
|  |   fsdev_pma_buf_t* pma_buf = PMA_BUF_AT(src); | ||||||
|   uint8_t *dst8 = (uint8_t *)dst; |   uint8_t *dst8 = (uint8_t *)dst; | ||||||
|  |  | ||||||
|   while (n16--) { |   while (n_read--) { | ||||||
|     uint16_t temp16 = pma16->u16; |     fsdevbus_unaligned_write(dst8, (fsdev_bus_t ) pma_buf->value); | ||||||
|     tu_unaligned_write16(dst8, temp16); |     dst8 += FSDEV_BUS_SIZE; | ||||||
|     dst8 += 2; |     pma_buf++; | ||||||
|     pma16++; |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   if (nbytes & 0x01) { |   // odd bytes e.g 1 for 16-bit or 1-3 for 32-bit | ||||||
|     *dst8++ = tu_u16_low(pma16->u16); |   uint16_t odd = nbytes & (FSDEV_BUS_SIZE - 1); | ||||||
|  |   if (odd) { | ||||||
|  |     fsdev_bus_t temp = pma_buf->value; | ||||||
|  |     while (odd--) { | ||||||
|  |       *dst8++ = (uint8_t) (temp & 0xfful); | ||||||
|  |       temp >>= 8; | ||||||
|  |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   return true; |   return true; | ||||||
| } | } | ||||||
|  |  | ||||||
| #endif | // Write to PMA from FIFO | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * @brief Copy from FIFO to packet memory area (PMA). |  | ||||||
|  *        Uses byte-access of system memory and 16-bit access of packet memory |  | ||||||
|  * @param   wNBytes no. of bytes to be copied. |  | ||||||
|  * @retval None |  | ||||||
|  */ |  | ||||||
| static bool dcd_write_packet_memory_ff(tu_fifo_t *ff, uint16_t dst, uint16_t wNBytes) { | static bool dcd_write_packet_memory_ff(tu_fifo_t *ff, uint16_t dst, uint16_t wNBytes) { | ||||||
|  |   if (wNBytes == 0) return true; | ||||||
|  |  | ||||||
|   // Since we copy from a ring buffer FIFO, a wrap might occur making it necessary to conduct two copies |   // Since we copy from a ring buffer FIFO, a wrap might occur making it necessary to conduct two copies | ||||||
|   tu_fifo_buffer_info_t info; |   tu_fifo_buffer_info_t info; | ||||||
|   tu_fifo_get_read_info(ff, &info); |   tu_fifo_get_read_info(ff, &info); | ||||||
|  |  | ||||||
|   uint16_t cnt_lin = TU_MIN(wNBytes, info.len_lin); |   uint16_t cnt_lin = tu_min16(wNBytes, info.len_lin); | ||||||
|   uint16_t cnt_wrap = TU_MIN(wNBytes - cnt_lin, info.len_wrap); |   uint16_t cnt_wrap = tu_min16(wNBytes - cnt_lin, info.len_wrap); | ||||||
|  |   uint16_t const cnt_total = cnt_lin + cnt_wrap; | ||||||
|  |  | ||||||
|   // We want to read from the FIFO and write it into the PMA, if LIN part is ODD and has WRAPPED part, |   // We want to read from the FIFO and write it into the PMA, if LIN part is ODD and has WRAPPED part, | ||||||
|   // last lin byte will be combined with wrapped part |   // last lin byte will be combined with wrapped part To ensure PMA is always access aligned | ||||||
|   // To ensure PMA is always access aligned (dst aligned to 16 or 32 bit) |   uint16_t lin_even = cnt_lin & ~(FSDEV_BUS_SIZE - 1); | ||||||
| #ifdef FSDEV_BUS_32BIT |   uint16_t lin_odd = cnt_lin & (FSDEV_BUS_SIZE - 1); | ||||||
|   if ((cnt_lin & 0x03) && cnt_wrap) { |   uint8_t const *src8 = (uint8_t const*) info.ptr_lin; | ||||||
|     // Copy first linear part |  | ||||||
|     dcd_write_packet_memory(dst, info.ptr_lin, cnt_lin & ~0x03); |  | ||||||
|     dst += cnt_lin & ~0x03; |  | ||||||
|  |  | ||||||
|     // Copy last linear bytes & first wrapped bytes to buffer |   // write even linear part | ||||||
|     uint32_t i; |   dcd_write_packet_memory(dst, src8, lin_even); | ||||||
|     uint8_t tmp[4]; |   dst += lin_even; | ||||||
|     for (i = 0; i < (cnt_lin & 0x03); i++) { |   src8 += lin_even; | ||||||
|       tmp[i] = ((uint8_t *)info.ptr_lin)[(cnt_lin & ~0x03) + i]; |  | ||||||
|     } |   if (lin_odd == 0) { | ||||||
|     uint32_t wCnt = cnt_wrap; |     src8 = (uint8_t const*) info.ptr_wrap; | ||||||
|     for (; i < 4 && wCnt > 0; i++, wCnt--) { |   } else { | ||||||
|       tmp[i] = *(uint8_t *)info.ptr_wrap; |     // Combine last linear bytes + first wrapped bytes to form fsdev bus width data | ||||||
|       info.ptr_wrap = (uint8_t *)info.ptr_wrap + 1; |     fsdev_bus_t temp = 0; | ||||||
|  |     uint16_t i; | ||||||
|  |     for(i = 0; i < lin_odd; i++) { | ||||||
|  |       temp |= *src8++ << (i * 8); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     // Write unaligned buffer |     src8 = (uint8_t const*) info.ptr_wrap; | ||||||
|     dcd_write_packet_memory(dst, &tmp, 4); |     for(; i < FSDEV_BUS_SIZE && cnt_wrap > 0; i++, cnt_wrap--) { | ||||||
|     dst += 4; |       temp |= *src8++ << (i * 8); | ||||||
|  |  | ||||||
|     // Copy rest of wrapped byte |  | ||||||
|     if (wCnt) { |  | ||||||
|       dcd_write_packet_memory(dst, info.ptr_wrap, wCnt); |  | ||||||
|     } |     } | ||||||
|   } |  | ||||||
| #else |  | ||||||
|   if ((cnt_lin & 0x01) && cnt_wrap) { |  | ||||||
|     // Copy first linear part |  | ||||||
|     dcd_write_packet_memory(dst, info.ptr_lin, cnt_lin & ~0x01); |  | ||||||
|     dst += cnt_lin & ~0x01; |  | ||||||
|  |  | ||||||
|     // Copy last linear byte & first wrapped byte |     dcd_write_packet_memory(dst, &temp, FSDEV_BUS_SIZE); | ||||||
|     uint16_t tmp = ((uint8_t *)info.ptr_lin)[cnt_lin - 1] | ((uint16_t)(((uint8_t *)info.ptr_wrap)[0]) << 8U); |     dst += FSDEV_BUS_SIZE; | ||||||
|     dcd_write_packet_memory(dst, &tmp, 2); |  | ||||||
|     dst += 2; |  | ||||||
|  |  | ||||||
|     // Copy rest of wrapped byte |  | ||||||
|     dcd_write_packet_memory(dst, ((uint8_t *)info.ptr_wrap) + 1, cnt_wrap - 1); |  | ||||||
|   } |  | ||||||
| #endif |  | ||||||
|   else { |  | ||||||
|     // Copy linear part |  | ||||||
|     dcd_write_packet_memory(dst, info.ptr_lin, cnt_lin); |  | ||||||
|     dst += info.len_lin; |  | ||||||
|  |  | ||||||
|     if (info.len_wrap) { |  | ||||||
|       // Copy wrapped byte |  | ||||||
|       dcd_write_packet_memory(dst, info.ptr_wrap, cnt_wrap); |  | ||||||
|     } |  | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   tu_fifo_advance_read_pointer(ff, cnt_lin + cnt_wrap); |   // write the rest of the wrapped part | ||||||
|  |   dcd_write_packet_memory(dst, src8, cnt_wrap); | ||||||
|  |  | ||||||
|  |   tu_fifo_advance_read_pointer(ff, cnt_total); | ||||||
|   return true; |   return true; | ||||||
| } | } | ||||||
|  |  | ||||||
| /** | // Read from PMA to FIFO | ||||||
|  * @brief Copy a buffer from user packet memory area (PMA) to FIFO. |  | ||||||
|  *        Uses byte-access of system memory and 16-bit access of packet memory |  | ||||||
|  * @param   wNBytes no. of bytes to be copied. |  | ||||||
|  * @retval None |  | ||||||
|  */ |  | ||||||
| static bool dcd_read_packet_memory_ff(tu_fifo_t *ff, uint16_t src, uint16_t wNBytes) { | static bool dcd_read_packet_memory_ff(tu_fifo_t *ff, uint16_t src, uint16_t wNBytes) { | ||||||
|  |   if (wNBytes == 0) return true; | ||||||
|  |  | ||||||
|   // Since we copy into a ring buffer FIFO, a wrap might occur making it necessary to conduct two copies |   // Since we copy into a ring buffer FIFO, a wrap might occur making it necessary to conduct two copies | ||||||
|   // Check for first linear part |   // Check for first linear part | ||||||
|   tu_fifo_buffer_info_t info; |   tu_fifo_buffer_info_t info; | ||||||
|   tu_fifo_get_write_info(ff, &info); // We want to read from the FIFO |   tu_fifo_get_write_info(ff, &info); // We want to read from the FIFO | ||||||
|  |  | ||||||
|   uint16_t cnt_lin = TU_MIN(wNBytes, info.len_lin); |   uint16_t cnt_lin = tu_min16(wNBytes, info.len_lin); | ||||||
|   uint16_t cnt_wrap = TU_MIN(wNBytes - cnt_lin, info.len_wrap); |   uint16_t cnt_wrap = tu_min16(wNBytes - cnt_lin, info.len_wrap); | ||||||
|  |   uint16_t cnt_total = cnt_lin + cnt_wrap; | ||||||
|  |  | ||||||
|   // We want to read from PMA and write it into the FIFO, if LIN part is ODD and has WRAPPED part, |   // We want to read from the FIFO and write it into the PMA, if LIN part is ODD and has WRAPPED part, | ||||||
|   // last lin byte will be combined with wrapped part |   // last lin byte will be combined with wrapped part To ensure PMA is always access aligned | ||||||
|   // To ensure PMA is always access aligned (src aligned to 16 or 32 bit) |  | ||||||
| #ifdef FSDEV_BUS_32BIT |  | ||||||
|   if ((cnt_lin & 0x03) && cnt_wrap) { |  | ||||||
|     // Copy first linear part |  | ||||||
|     dcd_read_packet_memory(info.ptr_lin, src, cnt_lin & ~0x03); |  | ||||||
|     src += cnt_lin & ~0x03; |  | ||||||
|  |  | ||||||
|     // Copy last linear bytes & first wrapped bytes |   uint16_t lin_even = cnt_lin & ~(FSDEV_BUS_SIZE - 1); | ||||||
|     uint8_t tmp[4]; |   uint16_t lin_odd = cnt_lin & (FSDEV_BUS_SIZE - 1); | ||||||
|     dcd_read_packet_memory(tmp, src, 4); |   uint8_t *dst8 = (uint8_t *) info.ptr_lin; | ||||||
|     src += 4; |  | ||||||
|  |  | ||||||
|     uint32_t i; |   // read even linear part | ||||||
|     for (i = 0; i < (cnt_lin & 0x03); i++) { |   dcd_read_packet_memory(dst8, src, lin_even); | ||||||
|       ((uint8_t *)info.ptr_lin)[(cnt_lin & ~0x03) + i] = tmp[i]; |   dst8 += lin_even; | ||||||
|     } |   src += lin_even; | ||||||
|     uint32_t wCnt = cnt_wrap; |  | ||||||
|     for (; i < 4 && wCnt > 0; i++, wCnt--) { |   if (lin_odd == 0) { | ||||||
|       *(uint8_t *)info.ptr_wrap = tmp[i]; |     dst8 = (uint8_t *) info.ptr_wrap; | ||||||
|       info.ptr_wrap = (uint8_t *)info.ptr_wrap + 1; |   } else { | ||||||
|  |     // Combine last linear bytes + first wrapped bytes to form fsdev bus width data | ||||||
|  |     fsdev_bus_t temp; | ||||||
|  |     dcd_read_packet_memory(&temp, src, FSDEV_BUS_SIZE); | ||||||
|  |     src += FSDEV_BUS_SIZE; | ||||||
|  |  | ||||||
|  |     uint16_t i; | ||||||
|  |     for (i = 0; i < lin_odd; i++) { | ||||||
|  |       *dst8++ = (uint8_t) (temp & 0xfful); | ||||||
|  |       temp >>= 8; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     // Copy rest of wrapped byte |     dst8 = (uint8_t *) info.ptr_wrap; | ||||||
|     if (wCnt) { |     for (; i < FSDEV_BUS_SIZE && cnt_wrap > 0; i++, cnt_wrap--) { | ||||||
|       dcd_read_packet_memory(info.ptr_wrap, src, wCnt); |       *dst8++ = (uint8_t) (temp & 0xfful); | ||||||
|     } |       temp >>= 8; | ||||||
|   } |  | ||||||
| #else |  | ||||||
|   if ((cnt_lin & 0x01) && cnt_wrap) { |  | ||||||
|     // Copy first linear part |  | ||||||
|     dcd_read_packet_memory(info.ptr_lin, src, cnt_lin & ~0x01); |  | ||||||
|     src += cnt_lin & ~0x01; |  | ||||||
|  |  | ||||||
|     // Copy last linear byte & first wrapped byte |  | ||||||
|     uint8_t tmp[2]; |  | ||||||
|     dcd_read_packet_memory(tmp, src, 2); |  | ||||||
|     src += 2; |  | ||||||
|  |  | ||||||
|     ((uint8_t *)info.ptr_lin)[cnt_lin - 1] = tmp[0]; |  | ||||||
|     ((uint8_t *)info.ptr_wrap)[0] = tmp[1]; |  | ||||||
|  |  | ||||||
|     // Copy rest of wrapped byte |  | ||||||
|     dcd_read_packet_memory(((uint8_t *)info.ptr_wrap) + 1, src, cnt_wrap - 1); |  | ||||||
|   } |  | ||||||
| #endif |  | ||||||
|   else { |  | ||||||
|     // Copy linear part |  | ||||||
|     dcd_read_packet_memory(info.ptr_lin, src, cnt_lin); |  | ||||||
|     src += cnt_lin; |  | ||||||
|  |  | ||||||
|     if (info.len_wrap) { |  | ||||||
|       // Copy wrapped byte |  | ||||||
|       dcd_read_packet_memory(info.ptr_wrap, src, cnt_wrap); |  | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   tu_fifo_advance_write_pointer(ff, cnt_lin + cnt_wrap); |   // read the rest of the wrapped part | ||||||
|  |   dcd_read_packet_memory(dst8, src, cnt_wrap); | ||||||
|  |  | ||||||
|  |   tu_fifo_advance_write_pointer(ff, cnt_total); | ||||||
|   return true; |   return true; | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -54,22 +54,8 @@ | |||||||
| #endif | #endif | ||||||
|  |  | ||||||
| #define FSDEV_PMA_SIZE (512u) | #define FSDEV_PMA_SIZE (512u) | ||||||
| #define FSDEV_REG_BASE  0x40005C00UL | #define FSDEV_REG_BASE  (APB1PERIPH_BASE + 0x00005C00UL) | ||||||
|  | #define FSDEV_PMA_BASE  (APB1PERIPH_BASE + 0x00006000UL) | ||||||
| #define USB_BASE            (APB1PERIPH_BASE + 0x00005C00UL) /*!< USB_IP Peripheral Registers base address */ |  | ||||||
| #define USB_PMAADDR         (APB1PERIPH_BASE + 0x00006000UL) /*!< USB_IP Packet Memory Area base address */ |  | ||||||
| #define USB                 ((USB_TypeDef *)USB_BASE) |  | ||||||
|  |  | ||||||
| /******************************************************************************/ |  | ||||||
| /*                                                                            */ |  | ||||||
| /*                         USB Device General registers                       */ |  | ||||||
| /*                                                                            */ |  | ||||||
| /******************************************************************************/ |  | ||||||
| #define USB_CNTR                             (USB_BASE + 0x40U)             /*!< Control register */ |  | ||||||
| #define USB_ISTR                             (USB_BASE + 0x44U)             /*!< Interrupt status register */ |  | ||||||
| #define USB_FNR                              (USB_BASE + 0x48U)             /*!< Frame number register */ |  | ||||||
| #define USB_DADDR                            (USB_BASE + 0x4CU)             /*!< Device address register */ |  | ||||||
| #define USB_BTABLE                           (USB_BASE + 0x50U)             /*!< Buffer Table address register */ |  | ||||||
|  |  | ||||||
| /****************************  ISTR interrupt events  *************************/ | /****************************  ISTR interrupt events  *************************/ | ||||||
| #define USB_ISTR_CTR                         ((uint16_t)0x8000U)               /*!< Correct TRansfer (clear-only bit) */ | #define USB_ISTR_CTR                         ((uint16_t)0x8000U)               /*!< Correct TRansfer (clear-only bit) */ | ||||||
|   | |||||||
| @@ -82,12 +82,9 @@ | |||||||
|  |  | ||||||
| #elif CFG_TUSB_MCU == OPT_MCU_STM32G0 | #elif CFG_TUSB_MCU == OPT_MCU_STM32G0 | ||||||
|   #include "stm32g0xx.h" |   #include "stm32g0xx.h" | ||||||
|   #define FSDEV_BUS_32BIT |  | ||||||
|   #define FSDEV_PMA_SIZE (2048u) |   #define FSDEV_PMA_SIZE (2048u) | ||||||
|   #undef USB_PMAADDR |   #define USB USB_DRD_FS | ||||||
|   #define USB_PMAADDR USB_DRD_PMAADDR |  | ||||||
|   #define USB_TypeDef USB_DRD_TypeDef |  | ||||||
|   #define EP0R CHEP0R |  | ||||||
|   #define USB_EP_CTR_RX USB_EP_VTRX |   #define USB_EP_CTR_RX USB_EP_VTRX | ||||||
|   #define USB_EP_CTR_TX USB_EP_VTTX |   #define USB_EP_CTR_TX USB_EP_VTTX | ||||||
|   #define USB_EP_T_FIELD USB_CHEP_UTYPE |   #define USB_EP_T_FIELD USB_CHEP_UTYPE | ||||||
| @@ -100,7 +97,6 @@ | |||||||
|   #define USB_EPRX_DTOG2 USB_CHEP_RX_DTOG2 |   #define USB_EPRX_DTOG2 USB_CHEP_RX_DTOG2 | ||||||
|   #define USB_EPRX_STAT USB_CH_RX_VALID |   #define USB_EPRX_STAT USB_CH_RX_VALID | ||||||
|   #define USB_EPKIND_MASK USB_EP_KIND_MASK |   #define USB_EPKIND_MASK USB_EP_KIND_MASK | ||||||
|   #define USB USB_DRD_FS |  | ||||||
|   #define USB_CNTR_FRES USB_CNTR_USBRST |   #define USB_CNTR_FRES USB_CNTR_USBRST | ||||||
|   #define USB_CNTR_RESUME USB_CNTR_L2RES |   #define USB_CNTR_RESUME USB_CNTR_L2RES | ||||||
|   #define USB_ISTR_EP_ID USB_ISTR_IDN |   #define USB_ISTR_EP_ID USB_ISTR_IDN | ||||||
| @@ -110,17 +106,9 @@ | |||||||
|  |  | ||||||
| #elif CFG_TUSB_MCU == OPT_MCU_STM32H5 | #elif CFG_TUSB_MCU == OPT_MCU_STM32H5 | ||||||
|   #include "stm32h5xx.h" |   #include "stm32h5xx.h" | ||||||
|   #define FSDEV_BUS_32BIT |  | ||||||
|  |  | ||||||
|   #if !defined(USB_DRD_BASE) && defined(USB_DRD_FS_BASE) |  | ||||||
|   #define USB_DRD_BASE USB_DRD_FS_BASE |  | ||||||
|   #endif |  | ||||||
|  |  | ||||||
|   #define FSDEV_PMA_SIZE (2048u) |   #define FSDEV_PMA_SIZE (2048u) | ||||||
|   #undef USB_PMAADDR |   #define USB USB_DRD_FS | ||||||
|   #define USB_PMAADDR USB_DRD_PMAADDR |  | ||||||
|   #define USB_TypeDef USB_DRD_TypeDef |  | ||||||
|   #define EP0R CHEP0R |  | ||||||
|   #define USB_EP_CTR_RX USB_EP_VTRX |   #define USB_EP_CTR_RX USB_EP_VTRX | ||||||
|   #define USB_EP_CTR_TX USB_EP_VTTX |   #define USB_EP_CTR_TX USB_EP_VTTX | ||||||
|   #define USB_EP_T_FIELD USB_CHEP_UTYPE |   #define USB_EP_T_FIELD USB_CHEP_UTYPE | ||||||
| @@ -133,7 +121,6 @@ | |||||||
|   #define USB_EPRX_DTOG2 USB_CHEP_RX_DTOG2 |   #define USB_EPRX_DTOG2 USB_CHEP_RX_DTOG2 | ||||||
|   #define USB_EPRX_STAT USB_CH_RX_VALID |   #define USB_EPRX_STAT USB_CH_RX_VALID | ||||||
|   #define USB_EPKIND_MASK USB_EP_KIND_MASK |   #define USB_EPKIND_MASK USB_EP_KIND_MASK | ||||||
|   #define USB USB_DRD_FS |  | ||||||
|   #define USB_CNTR_FRES USB_CNTR_USBRST |   #define USB_CNTR_FRES USB_CNTR_USBRST | ||||||
|   #define USB_CNTR_RESUME USB_CNTR_L2RES |   #define USB_CNTR_RESUME USB_CNTR_L2RES | ||||||
|   #define USB_ISTR_EP_ID USB_ISTR_IDN |   #define USB_ISTR_EP_ID USB_ISTR_IDN | ||||||
| @@ -144,9 +131,8 @@ | |||||||
| #elif CFG_TUSB_MCU == OPT_MCU_STM32WB | #elif CFG_TUSB_MCU == OPT_MCU_STM32WB | ||||||
|   #include "stm32wbxx.h" |   #include "stm32wbxx.h" | ||||||
|   #define FSDEV_PMA_SIZE (1024u) |   #define FSDEV_PMA_SIZE (1024u) | ||||||
|   /* ST provided header has incorrect value */ |   /* ST provided header has incorrect value of USB_PMAADDR */ | ||||||
|   #undef USB_PMAADDR |   #define FSDEV_PMA_BASE USB1_PMAADDR | ||||||
|   #define USB_PMAADDR USB1_PMAADDR |  | ||||||
|  |  | ||||||
| #elif CFG_TUSB_MCU == OPT_MCU_STM32L4 | #elif CFG_TUSB_MCU == OPT_MCU_STM32L4 | ||||||
|   #include "stm32l4xx.h" |   #include "stm32l4xx.h" | ||||||
| @@ -162,13 +148,9 @@ | |||||||
|  |  | ||||||
| #elif CFG_TUSB_MCU == OPT_MCU_STM32U5 | #elif CFG_TUSB_MCU == OPT_MCU_STM32U5 | ||||||
|   #include "stm32u5xx.h" |   #include "stm32u5xx.h" | ||||||
|   #define FSDEV_BUS_32BIT |  | ||||||
|  |  | ||||||
|   #define FSDEV_PMA_SIZE (2048u) |   #define FSDEV_PMA_SIZE (2048u) | ||||||
|   #undef USB_PMAADDR |   #define USB USB_DRD_FS | ||||||
|   #define USB_PMAADDR USB_DRD_PMAADDR |  | ||||||
|   #define USB_TypeDef USB_DRD_TypeDef |  | ||||||
|   #define EP0R CHEP0R |  | ||||||
|   #define USB_EP_CTR_RX USB_EP_VTRX |   #define USB_EP_CTR_RX USB_EP_VTRX | ||||||
|   #define USB_EP_CTR_TX USB_EP_VTTX |   #define USB_EP_CTR_TX USB_EP_VTTX | ||||||
|   #define USB_EP_T_FIELD USB_CHEP_UTYPE |   #define USB_EP_T_FIELD USB_CHEP_UTYPE | ||||||
| @@ -181,7 +163,6 @@ | |||||||
|   #define USB_EPRX_DTOG2 USB_CHEP_RX_DTOG2 |   #define USB_EPRX_DTOG2 USB_CHEP_RX_DTOG2 | ||||||
|   #define USB_EPRX_STAT USB_CH_RX_VALID |   #define USB_EPRX_STAT USB_CH_RX_VALID | ||||||
|   #define USB_EPKIND_MASK USB_EP_KIND_MASK |   #define USB_EPKIND_MASK USB_EP_KIND_MASK | ||||||
|   #define USB USB_DRD_FS |  | ||||||
|   #define USB_CNTR_FRES USB_CNTR_USBRST |   #define USB_CNTR_FRES USB_CNTR_USBRST | ||||||
|   #define USB_CNTR_RESUME USB_CNTR_L2RES |   #define USB_CNTR_RESUME USB_CNTR_L2RES | ||||||
|   #define USB_ISTR_EP_ID USB_ISTR_IDN |   #define USB_ISTR_EP_ID USB_ISTR_IDN | ||||||
| @@ -194,6 +175,10 @@ | |||||||
|   // This includes U0 |   // This includes U0 | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
|  | //--------------------------------------------------------------------+ | ||||||
|  | // Register and PMA Base Address | ||||||
|  | //--------------------------------------------------------------------+ | ||||||
|  | #ifndef FSDEV_REG_BASE | ||||||
| #if defined(USB_BASE) | #if defined(USB_BASE) | ||||||
|   #define FSDEV_REG_BASE USB_BASE |   #define FSDEV_REG_BASE USB_BASE | ||||||
| #elif defined(USB_DRD_BASE) | #elif defined(USB_DRD_BASE) | ||||||
| @@ -203,6 +188,17 @@ | |||||||
| #else | #else | ||||||
|   #error "FSDEV_REG_BASE not defined" |   #error "FSDEV_REG_BASE not defined" | ||||||
| #endif | #endif | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | #ifndef FSDEV_PMA_BASE | ||||||
|  | #if defined(USB_PMAADDR) | ||||||
|  |   #define FSDEV_PMA_BASE USB_PMAADDR | ||||||
|  | #elif defined(USB_DRD_PMAADDR) | ||||||
|  |   #define FSDEV_PMA_BASE USB_DRD_PMAADDR | ||||||
|  | #else | ||||||
|  |   #error "FSDEV_PMA_BASE not defined" | ||||||
|  | #endif | ||||||
|  | #endif | ||||||
|  |  | ||||||
| // This checks if the device has "LPM" | // This checks if the device has "LPM" | ||||||
| #if defined(USB_ISTR_L1REQ) | #if defined(USB_ISTR_L1REQ) | ||||||
|   | |||||||
| @@ -1,9 +1,8 @@ | |||||||
| /* | /* | ||||||
|  * The MIT License (MIT) |  * The MIT License (MIT) | ||||||
|  * |  * | ||||||
|  * Copyright(c) 2016 STMicroelectronics |  | ||||||
|  * Copyright(c) N Conrad |  * Copyright(c) N Conrad | ||||||
|  * Copyright (c) 2024, hathach (tinyusb.org) |  * Copyright(c) 2024, hathach (tinyusb.org) | ||||||
|  * |  * | ||||||
|  * Permission is hereby granted, free of charge, to any person obtaining a copy |  * Permission is hereby granted, free of charge, to any person obtaining a copy | ||||||
|  * of this software and associated documentation files (the "Software"), to deal |  * of this software and associated documentation files (the "Software"), to deal | ||||||
| @@ -23,6 +22,7 @@ | |||||||
|  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN |  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN | ||||||
|  * THE SOFTWARE. |  * THE SOFTWARE. | ||||||
|  * |  * | ||||||
|  |  * This file is part of the TinyUSB stack. | ||||||
|  */ |  */ | ||||||
|  |  | ||||||
| #ifndef TUSB_FSDEV_TYPE_H | #ifndef TUSB_FSDEV_TYPE_H | ||||||
| @@ -49,19 +49,35 @@ TU_VERIFY_STATIC(FSDEV_BTABLE_BASE % 8 == 0, "BTABLE base must be aligned to 8 b | |||||||
|  |  | ||||||
| // For purposes of accessing the packet | // For purposes of accessing the packet | ||||||
| #if FSDEV_PMA_SIZE == 512 | #if FSDEV_PMA_SIZE == 512 | ||||||
|   #define FSDEV_PMA_STRIDE  (2u) // 1x16 bit access scheme |   // 1x16 bit / word access scheme | ||||||
|   #define pma_aligned TU_ATTR_ALIGNED(4) |   #define FSDEV_PMA_STRIDE  2 | ||||||
|  |   #define pma_access_scheme TU_ATTR_ALIGNED(4) | ||||||
| #elif FSDEV_PMA_SIZE == 1024 | #elif FSDEV_PMA_SIZE == 1024 | ||||||
|   #define FSDEV_PMA_STRIDE  (1u) // 2x16 bit access scheme |   // 2x16 bit / word access scheme | ||||||
|   #define pma_aligned |   #define FSDEV_PMA_STRIDE  1 | ||||||
|  |   #define pma_access_scheme | ||||||
| #elif FSDEV_PMA_SIZE == 2048 | #elif FSDEV_PMA_SIZE == 2048 | ||||||
|   #ifndef FSDEV_BUS_32BIT |   // 32 bit access scheme | ||||||
|     #warning "FSDEV_PMA_SIZE is 2048, but FSDEV_BUS_32BIT is not defined" |   #define FSDEV_BUS_32BIT | ||||||
|   #endif |   #define FSDEV_PMA_STRIDE  1 | ||||||
|   #define FSDEV_PMA_STRIDE  (1u) // 32 bit access scheme |   #define pma_access_scheme | ||||||
|   #define pma_aligned |  | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
|  | // The fsdev_bus_t type can be used for both register and PMA access necessities | ||||||
|  | #ifdef FSDEV_BUS_32BIT | ||||||
|  |   typedef uint32_t fsdev_bus_t; | ||||||
|  |   #define fsdevbus_unaligned_read(_addr)           tu_unaligned_read32(_addr) | ||||||
|  |   #define fsdevbus_unaligned_write(_addr, _value)  tu_unaligned_write32(_addr, _value) | ||||||
|  | #else | ||||||
|  |   typedef uint16_t fsdev_bus_t; | ||||||
|  |   #define fsdevbus_unaligned_read(_addr)           tu_unaligned_read16(_addr) | ||||||
|  |   #define fsdevbus_unaligned_write(_addr, _value)  tu_unaligned_write16(_addr, _value) | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | enum { | ||||||
|  |   FSDEV_BUS_SIZE = sizeof(fsdev_bus_t), | ||||||
|  | }; | ||||||
|  |  | ||||||
| //--------------------------------------------------------------------+ | //--------------------------------------------------------------------+ | ||||||
| // BTable Typedef | // BTable Typedef | ||||||
| //--------------------------------------------------------------------+ | //--------------------------------------------------------------------+ | ||||||
| @@ -75,13 +91,12 @@ enum { | |||||||
|  |  | ||||||
| // Buffer Table is located in Packet Memory Area (PMA) and therefore its address access is forced to either | // Buffer Table is located in Packet Memory Area (PMA) and therefore its address access is forced to either | ||||||
| // 16-bit or 32-bit depending on FSDEV_BUS_32BIT. | // 16-bit or 32-bit depending on FSDEV_BUS_32BIT. | ||||||
|  | // 0: TX (IN), 1: RX (OUT) | ||||||
| typedef union { | typedef union { | ||||||
|   // 0: TX (IN), 1: RX (OUT) |   // data is strictly 16-bit access (address could be 32-bit aligned) | ||||||
|  |  | ||||||
|   // strictly 16-bit access (could be 32-bit aligned) |  | ||||||
|   struct { |   struct { | ||||||
|     volatile pma_aligned uint16_t addr; |     volatile pma_access_scheme uint16_t addr; | ||||||
|     volatile pma_aligned uint16_t count; |     volatile pma_access_scheme uint16_t count; | ||||||
|   } ep16[FSDEV_EP_COUNT][2]; |   } ep16[FSDEV_EP_COUNT][2]; | ||||||
|  |  | ||||||
|   // strictly 32-bit access |   // strictly 32-bit access | ||||||
| @@ -93,11 +108,13 @@ typedef union { | |||||||
| TU_VERIFY_STATIC(sizeof(fsdev_btable_t) == FSDEV_EP_COUNT*8*FSDEV_PMA_STRIDE, "size is not correct"); | TU_VERIFY_STATIC(sizeof(fsdev_btable_t) == FSDEV_EP_COUNT*8*FSDEV_PMA_STRIDE, "size is not correct"); | ||||||
| TU_VERIFY_STATIC(FSDEV_BTABLE_BASE + FSDEV_EP_COUNT*8 <= FSDEV_PMA_SIZE, "BTABLE does not fit in PMA RAM"); | TU_VERIFY_STATIC(FSDEV_BTABLE_BASE + FSDEV_EP_COUNT*8 <= FSDEV_PMA_SIZE, "BTABLE does not fit in PMA RAM"); | ||||||
|  |  | ||||||
| #define FSDEV_BTABLE ((volatile fsdev_btable_t*) (USB_PMAADDR+FSDEV_BTABLE_BASE)) | #define FSDEV_BTABLE ((volatile fsdev_btable_t*) (FSDEV_PMA_BASE + FSDEV_PMA_STRIDE*(FSDEV_BTABLE_BASE))) | ||||||
|  |  | ||||||
| typedef struct { | typedef struct { | ||||||
|   volatile pma_aligned uint16_t u16; |   volatile pma_access_scheme fsdev_bus_t value; | ||||||
| } fsdev_pma16_t; | } fsdev_pma_buf_t; | ||||||
|  |  | ||||||
|  | #define PMA_BUF_AT(_addr) ((fsdev_pma_buf_t*) (FSDEV_PMA_BASE + FSDEV_PMA_STRIDE*(_addr))) | ||||||
|  |  | ||||||
| //--------------------------------------------------------------------+ | //--------------------------------------------------------------------+ | ||||||
| // Registers Typedef | // Registers Typedef | ||||||
| @@ -106,13 +123,6 @@ typedef struct { | |||||||
| // volatile 32-bit aligned | // volatile 32-bit aligned | ||||||
| #define _va32     volatile TU_ATTR_ALIGNED(4) | #define _va32     volatile TU_ATTR_ALIGNED(4) | ||||||
|  |  | ||||||
| // The fsdev_bus_t type can be used for both register and PMA access necessities |  | ||||||
| #ifdef FSDEV_BUS_32BIT |  | ||||||
| typedef uint32_t fsdev_bus_t; |  | ||||||
| #else |  | ||||||
| typedef uint16_t fsdev_bus_t; |  | ||||||
| #endif |  | ||||||
|  |  | ||||||
| typedef struct { | typedef struct { | ||||||
|   struct { |   struct { | ||||||
|     _va32 fsdev_bus_t reg; |     _va32 fsdev_bus_t reg; | ||||||
| @@ -154,9 +164,6 @@ TU_VERIFY_STATIC(sizeof(fsdev_regs_t) == 0x5C, "Size is not correct"); | |||||||
| #define USB_EP_CTR_TX_Pos    7u | #define USB_EP_CTR_TX_Pos    7u | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
|  |  | ||||||
| #define EP_CTR_TXRX (USB_EP_CTR_TX | USB_EP_CTR_RX) |  | ||||||
|  |  | ||||||
| typedef enum { | typedef enum { | ||||||
|   EP_STAT_DISABLED = 0, |   EP_STAT_DISABLED = 0, | ||||||
|   EP_STAT_STALL = 1, |   EP_STAT_STALL = 1, | ||||||
| @@ -173,24 +180,36 @@ typedef enum { | |||||||
| // - DTOG and STAT are write 1 to toggle | // - DTOG and STAT are write 1 to toggle | ||||||
| //--------------------------------------------------------------------+ | //--------------------------------------------------------------------+ | ||||||
|  |  | ||||||
| TU_ATTR_ALWAYS_INLINE static inline void ep_write(uint32_t ep_id, uint32_t value) { |  | ||||||
|   FSDEV_REG->ep[ep_id].reg = (fsdev_bus_t) value; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| TU_ATTR_ALWAYS_INLINE static inline uint32_t ep_read(uint32_t ep_id) { | TU_ATTR_ALWAYS_INLINE static inline uint32_t ep_read(uint32_t ep_id) { | ||||||
|   return FSDEV_REG->ep[ep_id].reg; |   return FSDEV_REG->ep[ep_id].reg; | ||||||
| } | } | ||||||
|  |  | ||||||
| TU_ATTR_ALWAYS_INLINE static inline uint32_t ep_add_status(uint32_t reg, tusb_dir_t dir, ep_stat_t state) { | TU_ATTR_ALWAYS_INLINE static inline void ep_write(uint32_t ep_id, uint32_t value, bool need_exclusive) { | ||||||
|   return reg ^ (state << (USB_EPTX_STAT_Pos + (dir == TUSB_DIR_IN ? 0 : 8))); |   if (need_exclusive) { | ||||||
|  |     dcd_int_disable(0); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   FSDEV_REG->ep[ep_id].reg = (fsdev_bus_t) value; | ||||||
|  |  | ||||||
|  |   if (need_exclusive) { | ||||||
|  |     dcd_int_enable(0); | ||||||
|  |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| TU_ATTR_ALWAYS_INLINE static inline uint32_t ep_add_dtog(uint32_t reg, tusb_dir_t dir, uint8_t state) { | TU_ATTR_ALWAYS_INLINE static inline void ep_write_clear_ctr(uint32_t ep_id, tusb_dir_t dir) { | ||||||
|   return reg ^ (state << (USB_EP_DTOG_TX_Pos + (dir == TUSB_DIR_IN ? 0 : 8))); |   uint32_t reg = FSDEV_REG->ep[ep_id].reg; | ||||||
|  |   reg |= USB_EP_CTR_TX | USB_EP_CTR_RX; | ||||||
|  |   reg &= USB_EPREG_MASK; | ||||||
|  |   reg &= ~(1 << (USB_EP_CTR_TX_Pos + (dir == TUSB_DIR_IN ? 0 : 8))); | ||||||
|  |   ep_write(ep_id, reg, false); | ||||||
| } | } | ||||||
|  |  | ||||||
| TU_ATTR_ALWAYS_INLINE static inline uint32_t ep_clear_ctr(uint32_t reg, tusb_dir_t dir) { | TU_ATTR_ALWAYS_INLINE static inline void ep_change_status(uint32_t* reg, tusb_dir_t dir, ep_stat_t state) { | ||||||
|   return reg & ~(1 << (USB_EP_CTR_TX_Pos + (dir == TUSB_DIR_IN ? 0 : 8))); |   *reg ^= (state << (USB_EPTX_STAT_Pos + (dir == TUSB_DIR_IN ? 0 : 8))); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | TU_ATTR_ALWAYS_INLINE static inline void ep_change_dtog(uint32_t* reg, tusb_dir_t dir, uint8_t state) { | ||||||
|  |   *reg ^= (state << (USB_EP_DTOG_TX_Pos + (dir == TUSB_DIR_IN ? 0 : 8))); | ||||||
| } | } | ||||||
|  |  | ||||||
| TU_ATTR_ALWAYS_INLINE static inline bool ep_is_iso(uint32_t reg) { | TU_ATTR_ALWAYS_INLINE static inline bool ep_is_iso(uint32_t reg) { | ||||||
| @@ -219,7 +238,7 @@ TU_ATTR_ALWAYS_INLINE static inline void btable_set_addr(uint32_t ep_id, uint8_t | |||||||
| #endif | #endif | ||||||
| } | } | ||||||
|  |  | ||||||
| TU_ATTR_ALWAYS_INLINE static inline uint32_t btable_get_count(uint32_t ep_id, uint8_t buf_id) { | TU_ATTR_ALWAYS_INLINE static inline uint16_t btable_get_count(uint32_t ep_id, uint8_t buf_id) { | ||||||
|   uint16_t count; |   uint16_t count; | ||||||
| #ifdef FSDEV_BUS_32BIT | #ifdef FSDEV_BUS_32BIT | ||||||
|   count = (FSDEV_BTABLE->ep32[ep_id][buf_id].count_addr >> 16); |   count = (FSDEV_BTABLE->ep32[ep_id][buf_id].count_addr >> 16); | ||||||
| @@ -249,22 +268,26 @@ TU_ATTR_ALWAYS_INLINE static inline uint16_t pma_align_buffer_size(uint16_t size | |||||||
|   if (size > 62) { |   if (size > 62) { | ||||||
|     block_in_bytes = 32; |     block_in_bytes = 32; | ||||||
|     *blsize = 1; |     *blsize = 1; | ||||||
|  |     *num_block = tu_div_ceil(size, 32); | ||||||
|   } else { |   } else { | ||||||
|     block_in_bytes = 2; |     block_in_bytes = 2; | ||||||
|     *blsize = 0; |     *blsize = 0; | ||||||
|  |     *num_block = tu_div_ceil(size, 2); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   *num_block = tu_div_ceil(size, block_in_bytes); |  | ||||||
|  |  | ||||||
|   return (*num_block) * block_in_bytes; |   return (*num_block) * block_in_bytes; | ||||||
| } | } | ||||||
|  |  | ||||||
| TU_ATTR_ALWAYS_INLINE static inline void btable_set_rx_bufsize(uint32_t ep_id, uint8_t buf_id, uint32_t wCount) { | TU_ATTR_ALWAYS_INLINE static inline void btable_set_rx_bufsize(uint32_t ep_id, uint8_t buf_id, uint16_t wCount) { | ||||||
|   uint8_t blsize, num_block; |   uint8_t blsize, num_block; | ||||||
|   (void) pma_align_buffer_size(wCount, &blsize, &num_block); |   (void) pma_align_buffer_size(wCount, &blsize, &num_block); | ||||||
|  |  | ||||||
|   /* Encode into register. When BLSIZE==1, we need to subtract 1 block count */ |   /* Encode into register. When BLSIZE==1, we need to subtract 1 block count */ | ||||||
|   uint16_t bl_nb = (blsize << 15) | ((num_block - blsize) << 10); |   uint16_t bl_nb = (blsize << 15) | ((num_block - blsize) << 10); | ||||||
|  |   if (bl_nb == 0) { | ||||||
|  |     // zlp but 0 is invalid value, set num_block to 1 (2 bytes) | ||||||
|  |     bl_nb = 1 << 10; | ||||||
|  |   } | ||||||
|  |  | ||||||
| #ifdef FSDEV_BUS_32BIT | #ifdef FSDEV_BUS_32BIT | ||||||
|   uint32_t count_addr = FSDEV_BTABLE->ep32[ep_id][buf_id].count_addr; |   uint32_t count_addr = FSDEV_BTABLE->ep32[ep_id][buf_id].count_addr; | ||||||
| @@ -273,6 +296,7 @@ TU_ATTR_ALWAYS_INLINE static inline void btable_set_rx_bufsize(uint32_t ep_id, u | |||||||
| #else | #else | ||||||
|   FSDEV_BTABLE->ep16[ep_id][buf_id].count = bl_nb; |   FSDEV_BTABLE->ep16[ep_id][buf_id].count = bl_nb; | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
| } | } | ||||||
|  |  | ||||||
| #ifdef __cplusplus | #ifdef __cplusplus | ||||||
|   | |||||||
| @@ -34,14 +34,7 @@ import subprocess | |||||||
| import json | import json | ||||||
| import glob | import glob | ||||||
| import platform | import platform | ||||||
|  | from multiprocessing import Pool | ||||||
| # for RPI double reset |  | ||||||
| if platform.machine() == 'aarch64': |  | ||||||
|     try: |  | ||||||
|         import gpiozero |  | ||||||
|     except ImportError: |  | ||||||
|         pass |  | ||||||
|  |  | ||||||
|  |  | ||||||
| ENUM_TIMEOUT = 30 | ENUM_TIMEOUT = 30 | ||||||
|  |  | ||||||
| @@ -130,10 +123,11 @@ def run_cmd(cmd): | |||||||
|  |  | ||||||
| def flash_jlink(board, firmware): | def flash_jlink(board, firmware): | ||||||
|     script = ['halt', 'r', f'loadfile {firmware}.elf', 'r', 'go', 'exit'] |     script = ['halt', 'r', f'loadfile {firmware}.elf', 'r', 'go', 'exit'] | ||||||
|     with open('flash.jlink', 'w') as f: |     f_jlink = f'{board["name"]}_{os.path.basename(firmware)}.jlink' | ||||||
|  |     with open(f_jlink, 'w') as f: | ||||||
|         f.writelines(f'{s}\n' for s in script) |         f.writelines(f'{s}\n' for s in script) | ||||||
|     ret = run_cmd(f'JLinkExe -USB {board["flasher_sn"]} {board["flasher_args"]} -if swd -JTAGConf -1,-1 -speed auto -NoGui 1 -ExitOnError 1 -CommandFile flash.jlink') |     ret = run_cmd(f'JLinkExe -USB {board["flasher_sn"]} {board["flasher_args"]} -if swd -JTAGConf -1,-1 -speed auto -NoGui 1 -ExitOnError 1 -CommandFile {f_jlink}') | ||||||
|     os.remove('flash.jlink') |     os.remove(f_jlink) | ||||||
|     return ret |     return ret | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -152,6 +146,42 @@ def flash_openocd(board, firmware): | |||||||
|     return ret |     return ret | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def flash_openocd_wch(board, firmware): | ||||||
|  |     # Content of the wch-riscv.cfg file | ||||||
|  |     cfg_content = """ | ||||||
|  | adapter driver wlinke | ||||||
|  | adapter speed 6000 | ||||||
|  | transport select sdi | ||||||
|  |  | ||||||
|  | wlink_set_address 0x00000000 | ||||||
|  | set _CHIPNAME wch_riscv | ||||||
|  | sdi newtap $_CHIPNAME cpu -irlen 5 -expected-id 0x00001 | ||||||
|  |  | ||||||
|  | set _TARGETNAME $_CHIPNAME.cpu | ||||||
|  |  | ||||||
|  | target create $_TARGETNAME.0 wch_riscv -chain-position $_TARGETNAME | ||||||
|  | $_TARGETNAME.0 configure  -work-area-phys 0x20000000 -work-area-size 10000 -work-area-backup 1 | ||||||
|  | set _FLASHNAME $_CHIPNAME.flash | ||||||
|  |  | ||||||
|  | flash bank $_FLASHNAME wch_riscv 0x00000000 0 0 0 $_TARGETNAME.0 | ||||||
|  |  | ||||||
|  | echo "Ready for Remote Connections" | ||||||
|  | """ | ||||||
|  |     f_wch = f"wch-riscv_{board['uid']}.cfg" | ||||||
|  |     if not os.path.exists(f_wch): | ||||||
|  |         with open(f_wch, 'w') as file: | ||||||
|  |             file.write(cfg_content) | ||||||
|  |  | ||||||
|  |     ret = run_cmd(f'openocd_wch -c "adapter serial {board["flasher_sn"]}" -f {f_wch} -c "program {firmware}.elf reset exit"') | ||||||
|  |     return ret | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def flash_wlink_rs(board, firmware): | ||||||
|  |     # wlink use index for probe selection and lacking usb serial support | ||||||
|  |     ret = run_cmd(f'wlink flash {firmware}.elf') | ||||||
|  |     return ret | ||||||
|  |  | ||||||
|  |  | ||||||
| def flash_esptool(board, firmware): | def flash_esptool(board, firmware): | ||||||
|     port = get_serial_dev(board["flasher_sn"], None, None, 0) |     port = get_serial_dev(board["flasher_sn"], None, None, 0) | ||||||
|     dir = os.path.dirname(f'{firmware}.bin') |     dir = os.path.dirname(f'{firmware}.bin') | ||||||
| @@ -165,38 +195,6 @@ def flash_esptool(board, firmware): | |||||||
|     return ret |     return ret | ||||||
|  |  | ||||||
|  |  | ||||||
| def doublereset_with_rpi_gpio(pin): |  | ||||||
|     # Off = 0 = Reset |  | ||||||
|     nrst = gpiozero.LED(pin) |  | ||||||
|  |  | ||||||
|     nrst.off() |  | ||||||
|     time.sleep(0.1) |  | ||||||
|     nrst.on() |  | ||||||
|     time.sleep(0.1) |  | ||||||
|     nrst.off() |  | ||||||
|     time.sleep(0.1) |  | ||||||
|     nrst.on() |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def flash_bossac(board, firmware): |  | ||||||
|     # double reset to enter bootloader |  | ||||||
|     if platform.machine() == 'aarch64': |  | ||||||
|         doublereset_with_rpi_gpio(board["flasher_reset_pin"]) |  | ||||||
|  |  | ||||||
|     port = get_serial_dev(board["uid"], board["flashser_vendor"], board["flasher_product"], 0) |  | ||||||
|     timeout = ENUM_TIMEOUT |  | ||||||
|     while timeout: |  | ||||||
|         if os.path.exists(port): |  | ||||||
|             break |  | ||||||
|         else: |  | ||||||
|             time.sleep(0.5) |  | ||||||
|             timeout = timeout - 0.5 |  | ||||||
|     assert timeout, 'bossac bootloader is not available' |  | ||||||
|     # sleep a bit more for bootloader to be ready |  | ||||||
|     time.sleep(0.5) |  | ||||||
|     ret = run_cmd(f'bossac --port {port} {board["flasher_args"]} -U -i -R -e -w {firmware}.bin') |  | ||||||
|     return ret |  | ||||||
|  |  | ||||||
| # ------------------------------------------------------------- | # ------------------------------------------------------------- | ||||||
| # Tests | # Tests | ||||||
| # ------------------------------------------------------------- | # ------------------------------------------------------------- | ||||||
| @@ -264,29 +262,30 @@ def test_dfu(id): | |||||||
|  |  | ||||||
|     assert timeout, 'Device not available' |     assert timeout, 'Device not available' | ||||||
|  |  | ||||||
|  |     f_dfu0 = f'dfu0_{id}' | ||||||
|  |     f_dfu1 = f'dfu1_{id}' | ||||||
|  |  | ||||||
|     # Test upload |     # Test upload | ||||||
|     try: |     try: | ||||||
|         os.remove('dfu0') |         os.remove(f_dfu0) | ||||||
|         os.remove('dfu1') |         os.remove(f_dfu1) | ||||||
|     except OSError: |     except OSError: | ||||||
|         pass |         pass | ||||||
|  |  | ||||||
|     ret = subprocess.run(f'dfu-util -S {id} -a 0 -U dfu0', |     ret = run_cmd(f'dfu-util -S {id} -a 0 -U {f_dfu0}') | ||||||
|                          shell=True, stdout=subprocess.PIPE, stderr=subprocess.STDOUT) |  | ||||||
|     assert ret.returncode == 0, 'Upload failed' |     assert ret.returncode == 0, 'Upload failed' | ||||||
|  |  | ||||||
|     ret = subprocess.run(f'dfu-util -S {id} -a 1 -U dfu1', |     ret = run_cmd(f'dfu-util -S {id} -a 1 -U {f_dfu1}') | ||||||
|                          shell=True, stdout=subprocess.PIPE, stderr=subprocess.STDOUT) |  | ||||||
|     assert ret.returncode == 0, 'Upload failed' |     assert ret.returncode == 0, 'Upload failed' | ||||||
|  |  | ||||||
|     with open('dfu0') as f: |     with open(f_dfu0) as f: | ||||||
|         assert 'Hello world from TinyUSB DFU! - Partition 0' in f.read(), 'Wrong uploaded data' |         assert 'Hello world from TinyUSB DFU! - Partition 0' in f.read(), 'Wrong uploaded data' | ||||||
|  |  | ||||||
|     with open('dfu1') as f: |     with open(f_dfu1) as f: | ||||||
|         assert 'Hello world from TinyUSB DFU! - Partition 1' in f.read(), 'Wrong uploaded data' |         assert 'Hello world from TinyUSB DFU! - Partition 1' in f.read(), 'Wrong uploaded data' | ||||||
|  |  | ||||||
|     os.remove('dfu0') |     os.remove(f_dfu0) | ||||||
|     os.remove('dfu1') |     os.remove(f_dfu1) | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_dfu_runtime(id): | def test_dfu_runtime(id): | ||||||
| @@ -327,6 +326,61 @@ def test_hid_composite_freertos(id): | |||||||
| # ------------------------------------------------------------- | # ------------------------------------------------------------- | ||||||
| # Main | # Main | ||||||
| # ------------------------------------------------------------- | # ------------------------------------------------------------- | ||||||
|  | # all possible tests: board_test is added last to disable board's usb | ||||||
|  | all_tests = [ | ||||||
|  |     'cdc_dual_ports', | ||||||
|  |     'cdc_msc', | ||||||
|  |     'cdc_msc_freertos', | ||||||
|  |     'dfu', | ||||||
|  |     'dfu_runtime', | ||||||
|  |     'hid_boot_interface', | ||||||
|  |     'board_test' | ||||||
|  | ] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_board(item): | ||||||
|  |     name = item['name'] | ||||||
|  |     flasher = item['flasher'].lower() | ||||||
|  |  | ||||||
|  |     # default to all tests | ||||||
|  |     if 'tests' in item: | ||||||
|  |         test_list = item['tests'] + ['board_test'] | ||||||
|  |     else: | ||||||
|  |         test_list = list(all_tests) | ||||||
|  |  | ||||||
|  |     # remove skip_tests | ||||||
|  |     if 'tests_skip' in item: | ||||||
|  |         for skip in item['tests_skip']: | ||||||
|  |             if skip in test_list: | ||||||
|  |                 test_list.remove(skip) | ||||||
|  |  | ||||||
|  |     for test in test_list: | ||||||
|  |         fw_dir = f'cmake-build/cmake-build-{name}/device/{test}' | ||||||
|  |         if not os.path.exists(fw_dir): | ||||||
|  |             fw_dir = f'examples/cmake-build-{name}/device/{test}' | ||||||
|  |         fw_name = f'{fw_dir}/{test}' | ||||||
|  |         print(f'{name:30} {test:20} ... ', end='') | ||||||
|  |  | ||||||
|  |         if not os.path.exists(fw_dir): | ||||||
|  |             print('Skip') | ||||||
|  |             continue | ||||||
|  |  | ||||||
|  |         # flash firmware. It may fail randomly, retry a few times | ||||||
|  |         for i in range(3): | ||||||
|  |             ret = globals()[f'flash_{flasher}'](item, fw_name) | ||||||
|  |             if ret.returncode == 0: | ||||||
|  |                 break | ||||||
|  |             else: | ||||||
|  |                 print(f'Flashing failed, retry {i+1}') | ||||||
|  |                 time.sleep(1) | ||||||
|  |  | ||||||
|  |         assert ret.returncode == 0, 'Flash failed\n' + ret.stdout.decode() | ||||||
|  |  | ||||||
|  |         # run test | ||||||
|  |         globals()[f'test_{test}'](item['uid']) | ||||||
|  |         print('OK') | ||||||
|  |  | ||||||
|  |  | ||||||
| def main(): | def main(): | ||||||
|     """ |     """ | ||||||
|     Hardware test on specified boards |     Hardware test on specified boards | ||||||
| @@ -345,66 +399,13 @@ def main(): | |||||||
|     with open(config_file) as f: |     with open(config_file) as f: | ||||||
|         config = json.load(f) |         config = json.load(f) | ||||||
|  |  | ||||||
|     # all possible tests: board_test is added last to disable board's usb |  | ||||||
|     all_tests = [ |  | ||||||
|         'cdc_dual_ports', |  | ||||||
|         'cdc_msc', |  | ||||||
|         'cdc_msc_freertos', |  | ||||||
|         'dfu', |  | ||||||
|         'dfu_runtime', |  | ||||||
|         'hid_boot_interface', |  | ||||||
|         'board_test' |  | ||||||
|     ] |  | ||||||
|  |  | ||||||
|     if len(boards) == 0: |     if len(boards) == 0: | ||||||
|         config_boards = config['boards'] |         config_boards = config['boards'] | ||||||
|     else: |     else: | ||||||
|         config_boards = [e for e in config['boards'] if e['name'] in boards] |         config_boards = [e for e in config['boards'] if e['name'] in boards] | ||||||
|  |  | ||||||
|     for item in config_boards: |     with Pool(processes=os.cpu_count()) as pool: | ||||||
|         name = item['name'] |         pool.map(test_board, config_boards) | ||||||
|         print(f'Testing board:{name}') |  | ||||||
|         flasher = item['flasher'].lower() |  | ||||||
|  |  | ||||||
|         # default to all tests |  | ||||||
|         if 'tests' in item: |  | ||||||
|             test_list = item['tests'] + ['board_test'] |  | ||||||
|         else: |  | ||||||
|             test_list = list(all_tests) |  | ||||||
|  |  | ||||||
|         # remove skip_tests |  | ||||||
|         if 'tests_skip' in item: |  | ||||||
|             for skip in item['tests_skip']: |  | ||||||
|                 if skip in test_list: |  | ||||||
|                     test_list.remove(skip) |  | ||||||
|  |  | ||||||
|         for test in test_list: |  | ||||||
|             fw_dir = f'cmake-build/cmake-build-{name}/device/{test}' |  | ||||||
|             if not os.path.exists(fw_dir): |  | ||||||
|                 fw_dir = f'examples/cmake-build-{name}/device/{test}' |  | ||||||
|             fw_name = f'{fw_dir}/{test}' |  | ||||||
|             print(f'  {test} ... ', end='') |  | ||||||
|             sys.stdout.flush() |  | ||||||
|  |  | ||||||
|             if not os.path.exists(fw_dir): |  | ||||||
|                 print('Skip') |  | ||||||
|                 continue |  | ||||||
|  |  | ||||||
|             # flash firmware. It may fail randomly, retry a few times |  | ||||||
|             for i in range(3): |  | ||||||
|                 ret = globals()[f'flash_{flasher}'](item, fw_name) |  | ||||||
|                 if ret.returncode == 0: |  | ||||||
|                     break |  | ||||||
|                 else: |  | ||||||
|                     print(f'Flashing failed, retry {i+1}') |  | ||||||
|                     time.sleep(1) |  | ||||||
|  |  | ||||||
|             assert ret.returncode == 0, 'Flash failed\n' + ret.stdout.decode() |  | ||||||
|  |  | ||||||
|             # run test |  | ||||||
|             globals()[f'test_{test}'](item['uid']) |  | ||||||
|  |  | ||||||
|             print('OK') |  | ||||||
|  |  | ||||||
|  |  | ||||||
| if __name__ == '__main__': | if __name__ == '__main__': | ||||||
|   | |||||||
| @@ -57,6 +57,13 @@ | |||||||
|             "flasher": "openocd", |             "flasher": "openocd", | ||||||
|             "flasher_sn": "066FFF495087534867063844", |             "flasher_sn": "066FFF495087534867063844", | ||||||
|             "flasher_args": "-f interface/stlink.cfg -f target/stm32g0x.cfg" |             "flasher_args": "-f interface/stlink.cfg -f target/stm32g0x.cfg" | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |             "name": "nanoch32v203", | ||||||
|  |             "uid": "CDAB277B0FBC03E339E339E3", | ||||||
|  |             "flasher": "openocd_wch", | ||||||
|  |             "flasher_sn": "EBCA8F0670AF", | ||||||
|  |             "flasher_args": "" | ||||||
|         } |         } | ||||||
|     ], |     ], | ||||||
|     "boards-skip": [ |     "boards-skip": [ | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 Ha Thach
					Ha Thach