From a694ab1192046595919f901eaca93bfaa6a33534 Mon Sep 17 00:00:00 2001 From: Li Jun Ru Date: Thu, 23 May 2024 16:51:48 +0800 Subject: [PATCH] feat(usb_dongle): add usb dongle example --- .gitlab/ci/build.yml | 22 +- .gitlab/ci/rules.yml | 12 + examples/.build-rules.yml | 4 + examples/usb/device/usb_dongle/CMakeLists.txt | 7 + examples/usb/device/usb_dongle/Commands.md | 269 ++++++++ examples/usb/device/usb_dongle/Commands_EN.md | 269 ++++++++ examples/usb/device/usb_dongle/README.md | 286 +++++++++ examples/usb/device/usb_dongle/README_EN.md | 293 +++++++++ .../usb/device/usb_dongle/_static/ACM.png | Bin 0 -> 43409 bytes .../device/usb_dongle/_static/ESP32-S2.png | Bin 0 -> 61393 bytes .../device/usb_dongle/_static/ESP32-S3.png | Bin 0 -> 13636 bytes .../usb/device/usb_dongle/_static/error.png | Bin 0 -> 4779 bytes .../device/usb_dongle/_static/hciconfig.png | Bin 0 -> 19998 bytes .../device/usb_dongle/_static/ifconfig.png | Bin 0 -> 96678 bytes .../usb_dongle/_static/tinyusb_config.png | Bin 0 -> 44087 bytes .../device/usb_dongle/_static/uart_config.png | Bin 0 -> 30142 bytes .../FreeRTOS-Plus-CLI/CMakeLists.txt | 3 + .../FreeRTOS-Plus-CLI/FreeRTOS_CLI.c | 322 ++++++++++ .../components/FreeRTOS-Plus-CLI/History.txt | 31 + .../FreeRTOS-Plus-CLI/LICENSE_INFORMATION.txt | 19 + .../components/FreeRTOS-Plus-CLI/ReadMe.url | 5 + .../components/FreeRTOS-Plus-CLI/component.mk | 7 + .../FreeRTOS-Plus-CLI/include/FreeRTOS_CLI.h | 101 +++ .../components/FreeRTOS-Plus-CLI/readme.txt | 2 + .../components/tinyusb_dongle/CMakeLists.txt | 65 ++ .../components/tinyusb_dongle/Kconfig | 227 +++++++ .../components/tinyusb_dongle/LICENSE | 202 ++++++ .../components/tinyusb_dongle/cdc.c | 109 ++++ .../tinyusb_dongle/descriptors_control.c | 284 +++++++++ .../tinyusb_dongle/idf_component.yml | 10 + .../tinyusb_dongle/include/tinyusb.h | 75 +++ .../tinyusb_dongle/include/tinyusb_dfu_ota.h | 36 ++ .../tinyusb_dongle/include/tinyusb_net.h | 99 +++ .../tinyusb_dongle/include/tinyusb_types.h | 21 + .../tinyusb_dongle/include/tusb_bth.h | 69 +++ .../tinyusb_dongle/include/tusb_cdc_acm.h | 205 +++++++ .../tinyusb_dongle/include/tusb_config.h | 136 ++++ .../tinyusb_dongle/include/tusb_console.h | 33 + .../tinyusb_dongle/include/tusb_tasks.h | 39 ++ .../tinyusb_dongle/include/vfs_tinyusb.h | 69 +++ .../tinyusb_dongle/include_private/cdc.h | 80 +++ .../include_private/descriptors_control.h | 47 ++ .../include_private/usb_descriptors.h | 62 ++ .../components/tinyusb_dongle/sbom.yml | 2 + .../components/tinyusb_dongle/tinyusb.c | 73 +++ .../components/tinyusb_dongle/tinyusb_net.c | 174 ++++++ .../components/tinyusb_dongle/tusb_bth.c | 266 ++++++++ .../components/tinyusb_dongle/tusb_cdc_acm.c | 345 +++++++++++ .../components/tinyusb_dongle/tusb_console.c | 113 ++++ .../components/tinyusb_dongle/tusb_dfu.c | 91 +++ .../components/tinyusb_dongle/tusb_dfu_ota.c | 207 +++++++ .../components/tinyusb_dongle/tusb_tasks.c | 77 +++ .../tinyusb_dongle/usb_descriptors.c | 307 ++++++++++ .../components/tinyusb_dongle/vfs_tinyusb.c | 298 +++++++++ .../usb/device/usb_dongle/main/CLI_Commands.c | 579 ++++++++++++++++++ .../usb/device/usb_dongle/main/CMakeLists.txt | 17 + .../device/usb_dongle/main/Command_Parse.c | 85 +++ .../device/usb_dongle/main/Kconfig.projbuild | 59 ++ .../usb/device/usb_dongle/main/cmd_wifi.c | 457 ++++++++++++++ .../usb/device/usb_dongle/main/cmd_wifi.h | 42 ++ .../usb/device/usb_dongle/main/data_back.c | 33 + .../usb/device/usb_dongle/main/data_back.h | 13 + .../device/usb_dongle/main/idf_component.yml | 2 + examples/usb/device/usb_dongle/main/uart.c | 128 ++++ .../device/usb_dongle/main/usb_dongle_main.c | 132 ++++ .../usb/device/usb_dongle/sdkconfig.ci.bth | 4 + .../usb/device/usb_dongle/sdkconfig.ci.dfu | 1 + .../usb/device/usb_dongle/sdkconfig.ci.ecm | 1 + .../usb/device/usb_dongle/sdkconfig.ci.ncm | 1 + .../usb/device/usb_dongle/sdkconfig.ci.rndis | 1 + .../usb/device/usb_dongle/sdkconfig.defaults | 16 + tools/ci/check_copyright_config.yaml | 2 + 72 files changed, 7043 insertions(+), 3 deletions(-) create mode 100644 examples/usb/device/usb_dongle/CMakeLists.txt create mode 100644 examples/usb/device/usb_dongle/Commands.md create mode 100644 examples/usb/device/usb_dongle/Commands_EN.md create mode 100644 examples/usb/device/usb_dongle/README.md create mode 100644 examples/usb/device/usb_dongle/README_EN.md create mode 100644 examples/usb/device/usb_dongle/_static/ACM.png create mode 100644 examples/usb/device/usb_dongle/_static/ESP32-S2.png create mode 100644 examples/usb/device/usb_dongle/_static/ESP32-S3.png create mode 100644 examples/usb/device/usb_dongle/_static/error.png create mode 100644 examples/usb/device/usb_dongle/_static/hciconfig.png create mode 100644 examples/usb/device/usb_dongle/_static/ifconfig.png create mode 100644 examples/usb/device/usb_dongle/_static/tinyusb_config.png create mode 100644 examples/usb/device/usb_dongle/_static/uart_config.png create mode 100644 examples/usb/device/usb_dongle/components/FreeRTOS-Plus-CLI/CMakeLists.txt create mode 100644 examples/usb/device/usb_dongle/components/FreeRTOS-Plus-CLI/FreeRTOS_CLI.c create mode 100644 examples/usb/device/usb_dongle/components/FreeRTOS-Plus-CLI/History.txt create mode 100644 examples/usb/device/usb_dongle/components/FreeRTOS-Plus-CLI/LICENSE_INFORMATION.txt create mode 100644 examples/usb/device/usb_dongle/components/FreeRTOS-Plus-CLI/ReadMe.url create mode 100644 examples/usb/device/usb_dongle/components/FreeRTOS-Plus-CLI/component.mk create mode 100644 examples/usb/device/usb_dongle/components/FreeRTOS-Plus-CLI/include/FreeRTOS_CLI.h create mode 100644 examples/usb/device/usb_dongle/components/FreeRTOS-Plus-CLI/readme.txt create mode 100644 examples/usb/device/usb_dongle/components/tinyusb_dongle/CMakeLists.txt create mode 100644 examples/usb/device/usb_dongle/components/tinyusb_dongle/Kconfig create mode 100644 examples/usb/device/usb_dongle/components/tinyusb_dongle/LICENSE create mode 100644 examples/usb/device/usb_dongle/components/tinyusb_dongle/cdc.c create mode 100644 examples/usb/device/usb_dongle/components/tinyusb_dongle/descriptors_control.c create mode 100644 examples/usb/device/usb_dongle/components/tinyusb_dongle/idf_component.yml create mode 100644 examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tinyusb.h create mode 100644 examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tinyusb_dfu_ota.h create mode 100644 examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tinyusb_net.h create mode 100644 examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tinyusb_types.h create mode 100644 examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tusb_bth.h create mode 100644 examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tusb_cdc_acm.h create mode 100644 examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tusb_config.h create mode 100644 examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tusb_console.h create mode 100644 examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tusb_tasks.h create mode 100644 examples/usb/device/usb_dongle/components/tinyusb_dongle/include/vfs_tinyusb.h create mode 100644 examples/usb/device/usb_dongle/components/tinyusb_dongle/include_private/cdc.h create mode 100644 examples/usb/device/usb_dongle/components/tinyusb_dongle/include_private/descriptors_control.h create mode 100644 examples/usb/device/usb_dongle/components/tinyusb_dongle/include_private/usb_descriptors.h create mode 100644 examples/usb/device/usb_dongle/components/tinyusb_dongle/sbom.yml create mode 100644 examples/usb/device/usb_dongle/components/tinyusb_dongle/tinyusb.c create mode 100644 examples/usb/device/usb_dongle/components/tinyusb_dongle/tinyusb_net.c create mode 100644 examples/usb/device/usb_dongle/components/tinyusb_dongle/tusb_bth.c create mode 100644 examples/usb/device/usb_dongle/components/tinyusb_dongle/tusb_cdc_acm.c create mode 100644 examples/usb/device/usb_dongle/components/tinyusb_dongle/tusb_console.c create mode 100644 examples/usb/device/usb_dongle/components/tinyusb_dongle/tusb_dfu.c create mode 100644 examples/usb/device/usb_dongle/components/tinyusb_dongle/tusb_dfu_ota.c create mode 100644 examples/usb/device/usb_dongle/components/tinyusb_dongle/tusb_tasks.c create mode 100644 examples/usb/device/usb_dongle/components/tinyusb_dongle/usb_descriptors.c create mode 100644 examples/usb/device/usb_dongle/components/tinyusb_dongle/vfs_tinyusb.c create mode 100644 examples/usb/device/usb_dongle/main/CLI_Commands.c create mode 100644 examples/usb/device/usb_dongle/main/CMakeLists.txt create mode 100644 examples/usb/device/usb_dongle/main/Command_Parse.c create mode 100644 examples/usb/device/usb_dongle/main/Kconfig.projbuild create mode 100644 examples/usb/device/usb_dongle/main/cmd_wifi.c create mode 100644 examples/usb/device/usb_dongle/main/cmd_wifi.h create mode 100644 examples/usb/device/usb_dongle/main/data_back.c create mode 100644 examples/usb/device/usb_dongle/main/data_back.h create mode 100644 examples/usb/device/usb_dongle/main/idf_component.yml create mode 100644 examples/usb/device/usb_dongle/main/uart.c create mode 100644 examples/usb/device/usb_dongle/main/usb_dongle_main.c create mode 100644 examples/usb/device/usb_dongle/sdkconfig.ci.bth create mode 100644 examples/usb/device/usb_dongle/sdkconfig.ci.dfu create mode 100644 examples/usb/device/usb_dongle/sdkconfig.ci.ecm create mode 100644 examples/usb/device/usb_dongle/sdkconfig.ci.ncm create mode 100644 examples/usb/device/usb_dongle/sdkconfig.ci.rndis create mode 100644 examples/usb/device/usb_dongle/sdkconfig.defaults diff --git a/.gitlab/ci/build.yml b/.gitlab/ci/build.yml index 45c6e7477..030382d23 100644 --- a/.gitlab/ci/build.yml +++ b/.gitlab/ci/build.yml @@ -39,7 +39,7 @@ - pip install idf_build_apps - python tools/build_apps.py ${EXAMPLE_DIR} --config ${EXAMPLE_CONFIG} -t all -vv -.build_idf_release_version: +.build_idf_active_release_version: parallel: matrix: - IMAGE: espressif/idf:release-v4.4 @@ -48,6 +48,14 @@ - IMAGE: espressif/idf:release-v5.2 - IMAGE: espressif/idf:release-v5.3 +.build_idf_version_larger_than_v5_0: + parallel: + matrix: + - IMAGE: espressif/idf:release-v5.0 + - IMAGE: espressif/idf:release-v5.1 + - IMAGE: espressif/idf:release-v5.2 + - IMAGE: espressif/idf:release-v5.3 + build_example_audio_wav_player: extends: - .build_examples_template @@ -565,6 +573,14 @@ build_example_ulp_lp_cpu_lp_environment_sensor: variables: EXAMPLE_DIR: examples/ulp/lp_cpu/lp_environment_sensor +build_example_usb_device_usb_dongle: + extends: + - .build_examples_template + - .rules:build:example_usb_device_usb_dongle + - .build_idf_version_larger_than_v5_0 + variables: + EXAMPLE_DIR: examples/usb/device/usb_dongle + build_example_usb_device_usb_uart_bridge: extends: - .build_examples_template @@ -580,7 +596,7 @@ build_example_usb_device_usb_uac: extends: - .build_examples_template - .rules:build:example_usb_device_usb_uac - - .build_idf_release_version + - .build_idf_active_release_version variables: EXAMPLE_DIR: examples/usb/device/usb_uac @@ -1057,7 +1073,7 @@ build_components_led_lightbulb_driver_test_apps: extends: - .build_examples_template - .rules:build:components_led_lightbulb_driver_test_apps - - .build_idf_release_version + - .build_idf_active_release_version variables: EXAMPLE_DIR: components/led/lightbulb_driver/test_apps diff --git a/.gitlab/ci/rules.yml b/.gitlab/ci/rules.yml index 6f9538974..1ab2136ba 100644 --- a/.gitlab/ci/rules.yml +++ b/.gitlab/ci/rules.yml @@ -417,6 +417,9 @@ .patterns-example_sensors_sensor_hub_monitor: &patterns-example_sensors_sensor_hub_monitor - "examples/sensors/sensor_hub_monitor/**/*" +.patterns-example_usb_device_usb_dongle: &patterns-example_usb_device_usb_dongle + - "examples/usb/device/usb_dongle/**/*" + .patterns-example_ulp_lp_cpu_lp_environment_sensor: &patterns-example_ulp_lp_cpu_lp_environment_sensor - "examples/ulp/lp_cpu/lp_environment_sensor/**/*" @@ -1048,6 +1051,15 @@ - <<: *if-dev-push changes: *patterns-example_usb_device_usb_hid_device +.rules:build:example_usb_device_usb_dongle: + rules: + - <<: *if-protected + - <<: *if-label-build + - <<: *if-dev-push + changes: *patterns-build_system + - <<: *if-dev-push + changes: *patterns-example_usb_device_usb_dongle + .rules:build:example_usb_device_usb_uart_bridge: rules: - <<: *if-protected diff --git a/examples/.build-rules.yml b/examples/.build-rules.yml index 52dfc25fd..91da2137c 100644 --- a/examples/.build-rules.yml +++ b/examples/.build-rules.yml @@ -213,6 +213,10 @@ examples/sensors/sensor_hub_monitor: enable: - if: IDF_TARGET in ["esp32","esp32s2"] and (IDF_VERSION_MAJOR == 4 and IDF_VERSION_MINOR == 4) +examples/usb/device/usb_dongle: + enable: + - if: SOC_USB_OTG_SUPPORTED == 1 + examples/ulp/lp_cpu/lp_environment_sensor: enable: - if: IDF_TARGET in ["esp32c6"] diff --git a/examples/usb/device/usb_dongle/CMakeLists.txt b/examples/usb/device/usb_dongle/CMakeLists.txt new file mode 100644 index 000000000..b5a04addd --- /dev/null +++ b/examples/usb/device/usb_dongle/CMakeLists.txt @@ -0,0 +1,7 @@ +# The following lines of boilerplate have to be in your project's +# CMakeLists in this exact order for cmake to work correctly +cmake_minimum_required(VERSION 3.5) + +add_compile_options(-fdiagnostics-color=always) +include($ENV{IDF_PATH}/tools/cmake/project.cmake) +project(usb_dongle) diff --git a/examples/usb/device/usb_dongle/Commands.md b/examples/usb/device/usb_dongle/Commands.md new file mode 100644 index 000000000..e48fc4f7d --- /dev/null +++ b/examples/usb/device/usb_dongle/Commands.md @@ -0,0 +1,269 @@ +### 命令介绍 + +#### 1.help + +**Function:** + +列出所有注册的命令 + +**Command:** + +``` +help +``` + +**Response:** + +``` +help: + Lists all the registered commands + +ap []: configure ssid and password +sta -s [-p ]: join specified soft-AP +sta -d: disconnect specified soft-AP +mode : station mode; ap mode +smartconfig [op]: op:1, start smartconfig; op:0, stop smartconfig +scan []: SSID of AP want to be scanned +ram: Get the current size of free heap memory and minimum size of free heap memory +restart: Software reset of the chip +version: Get version of chip and SDK +> +``` + +#### 2.ap + +**Function:** + +设置 AP 模式、查询 AP 设置 + +**Set Command:** + +``` +ap Soft_AP espressif +``` + +**Query Command:** + +``` +ap +``` + +**Response:** + +``` +AP mode:Soft_AP,espressif +> +``` + +Note: + +>password 为可选项,若不配置默认不加密 + +#### 3.sta + +**Function:** + +启动 Station 模式、查询所连接 AP 信息 + +**Set Command:** + +``` +sta -s AP_Test -p espressif +``` + +**Query Command:** + +``` +sta +``` + +**Response:** + +``` +,,, +> +``` + +| authmode_value | mode | +| :------------: | :------------------------ | +| 0 | WIFI_AUTH_OPEN | +| 1 | WIFI_AUTH_WEP | +| 2 | WIFI_AUTH_WPA_PSK | +| 3 | WIFI_AUTH_WPA2_PSK | +| 4 | WIFI_AUTH_WPA_WPA2_PSK | +| 5 | WIFI_AUTH_WPA2_ENTERPRISE | +| 6 | WIFI_AUTH_WPA3_PSK | +| 7 | WIFI_AUTH_WPA2_WPA3_PSK | +| 8 | WIFI_AUTH_WAPI_PSK | + +Note: + +>password 为可选项 + +**Function:** + +断开与 AP 的连接 + +**Set Command:** + +``` +sta -d +``` + +**Response:** + +``` +OK +> +``` + +#### 4.mode + +**Function:** + +设置 WiFi 模式 + +**Command:** + +* 设置 Station 模式 + + ``` + mode sta + ``` + +* 设置 AP 模式 + + ``` + mode ap + ``` + +#### 5.smartconfig + +**Function:** + +* 开启 SmartConfig 配网 + + **Command:** + + ``` + smartconfig 1 + ``` + + **Response:** + + ``` + >SSID:FAST_XLZ,PASSWORD:12345678 + OK + > + ``` + +* 关闭 SmartConfig 配网 + + **Command:** + + ``` + smartconfig 0 + ``` + + **Response:** + + ``` + OK + > + ``` + + Note: + + >使用 `smartconfig 1` 命令开启 SmartConfig 配网并成功连接后,不需要再使用 `smartconfig 0` 命令来关闭 SmartConfig 配网 + > + >`smartconfig 0` 命令只需要在 SmartConfig 配网失败时进行调用 + +配网步骤: + +>* 下载 ESPTOUCH APP :[Android source code](https://github.com/EspressifApp/EsptouchForAndroid) [iOS source code](https://github.com/EspressifApp/EsptouchForIOS) +>* 确保你的手机连接至目标 AP(2.4GHz) +>* 打开 ESPTOUCH APP 输入 password 并确认 +>* PC 端通过 USB 端口发送 `smartconfig 1` 命令 + +#### 6.scan + +**Function:** + +扫描 AP 并列出对应 SSID 以及 RSSI + +**Command:** + +* 扫描特定 AP + + ``` + scan + ``` + +* 扫描所有 AP + + ``` + scan + ``` + +**Response:** + +``` +> +[ssid][rssi=-22] +> +``` + +#### 7.ram + +**Function:** + +获取当前剩余内存大小以及系统运行期间最小时内存大小 + +**Command:** + +``` +ram +``` + +**Response:** + +``` +free heap size: 132612, min heap size: 116788 +> +``` + +#### 8.restart + +**Function:** + +重启系统 + +**Command:** + +``` +restart +``` + +#### 9.version + +**Function:** + +获取当前 IDF 版本以及芯片信息 + +**Command:** + +``` +version +``` + +**Response:** + +``` +IDF Version:v4.4-dev-2571-gb1c3ee71c5 +Chip info: + cores:1 + feature:/802.11bgn/External-Flash:2 MB + revision number:0 +> +``` + diff --git a/examples/usb/device/usb_dongle/Commands_EN.md b/examples/usb/device/usb_dongle/Commands_EN.md new file mode 100644 index 000000000..79282cfe7 --- /dev/null +++ b/examples/usb/device/usb_dongle/Commands_EN.md @@ -0,0 +1,269 @@ +### Command Set + +#### 1.help + +**Function:** + +List all registered commands. + +**Command:** + +``` +help +``` + +**Response:** + +``` +help: + Lists all the registered commands + +ap []: configure ssid and password +sta -s [-p ]: join specified soft-AP +sta -d: disconnect specified soft-AP +mode : station mode; ap mode +smartconfig [op]: op:1, start smartconfig; op:0, stop smartconfig +scan []: SSID of AP want to be scanned +ram: Get the current size of free heap memory and minimum size of free heap memory +restart: Software reset of the chip +version: Get version of chip and SDK +> +``` + +#### 2.ap + +**Function:** + +Set/get softAP configuration. + +**Set Command:** + +``` +ap Soft_AP espressif +``` + +**Query Command:** + +``` +ap +``` + +**Response:** + +``` +AP mode:Soft_AP,espressif +> +``` + +Note: + +>password is optional. If you didn't set any password, then the ESP softAP will be set into the open mode. + +#### 3.sta + +**Function:** + +Set station mode, query the information of the AP it connected to. + +**Set Command:** + +``` +sta -s AP_Test -p espressif +``` + +**Query Command:** + +``` +sta +``` + +**Response:** + +``` +,,, +> +``` + +| authmode_value | mode | +| :------------: | :------------------------ | +| 0 | WIFI_AUTH_OPEN | +| 1 | WIFI_AUTH_WEP | +| 2 | WIFI_AUTH_WPA_PSK | +| 3 | WIFI_AUTH_WPA2_PSK | +| 4 | WIFI_AUTH_WPA_WPA2_PSK | +| 5 | WIFI_AUTH_WPA2_ENTERPRISE | +| 6 | WIFI_AUTH_WPA3_PSK | +| 7 | WIFI_AUTH_WPA2_WPA3_PSK | +| 8 | WIFI_AUTH_WAPI_PSK | + +Note: + +>password is optional. + +**Function:** + +Disconnect from AP. + +**Set Command:** + +``` +sta -d +``` + +**Response:** + +``` +OK +> +``` + +#### 4.mode + +**Function:** + +Set Wi-Fi mode. + +**Command:** + +* Set Station mode + + ``` + mode sta + ``` + +* Set AP mode + + ``` + mode ap + ``` + +#### 5.smartconfig + +**Function:** + +* Start SmartConfig to connect to target AP. + + **Command:** + + ``` + smartconfig 1 + ``` + + **Response:** + + ``` + >SSID:FAST_XLZ,PASSWORD:12345678 + OK + > + ``` + +* Stop SmartConfig. + + **Command:** + + ``` + smartconfig 0 + ``` + + **Response:** + + ``` + OK + > + ``` + + Note: + + > After called `smartconfig 1` command, if the ESP device connects to the target AP successfully, then you don't need to call `smartconfig 0` to stop SmartConfig. + > + >`smartconfig 0` command only needs to be called, when the SmartConfig fails. + +SmartConfig steps: + +>* Download ESPTOUCH APP to your phone: [Android source code](https://github.com/EspressifApp/EsptouchForAndroid) [iOS source code](https://github.com/EspressifApp/EsptouchForIOS) +>* Connect your phone to the target AP (2.4GHz). +>* Open ESPTOUCH APP, input AP's password. +>* PC sends command `smartconfig 1` to USB port to start SmartConfig. + +#### 6.scan + +**Function:** + +Scan APs, list their SSID and RSSI. + +**Command:** + +* Scan a specific AP. + + ``` + scan + ``` + +* Scan all APs nearby. + + ``` + scan + ``` + +**Response:** + +``` +> +[ssid][rssi=-22] +> +``` + +#### 7.ram + +**Function:** + +Get the size of available heap, and get the minimum heap that has ever been available. + +**Command:** + +``` +ram +``` + +**Response:** + +``` +free heap size: 132612, min heap size: 116788 +> +``` + +#### 8.restart + +**Function:** + +Restart the system. + +**Command:** + +``` +restart +``` + +#### 9.version + +**Function:** + +Get the ESP-IDF and chip version. + +**Command:** + +``` +version +``` + +**Response:** + +``` +IDF Version:v4.4-dev-2571-gb1c3ee71c5 +Chip info: + cores:1 + feature:/802.11bgn/External-Flash:2 MB + revision number:0 +> +``` + diff --git a/examples/usb/device/usb_dongle/README.md b/examples/usb/device/usb_dongle/README.md new file mode 100644 index 000000000..e51ebee11 --- /dev/null +++ b/examples/usb/device/usb_dongle/README.md @@ -0,0 +1,286 @@ +# ESP32-S 系列 USB 无线适配器方案介绍 + +## 1.概述 + +本示例程序演示 ESP32-S 系列芯片如何实现 USB Dongle 设备功能,支持以下功能: + +* 支持 Host 主机通过 USB-ECM/RNDIS 无线上网 +* 支持 Host 主机通过 USB-BTH 进行 BLE 扫描、广播、配对、连接、绑定以及读写数据 +* 支持 Host 主机通过 USB-DFU 进行设备升级 +* 支持 Host 主机通过 USB-CDC、UART 对 ESP32-S 系列设备进行通信和控制 +* 支持多种 system、Wi-Fi 控制命令,使用 FreeRTOS-Plus-CLI 命令行接口,易拓展更多命令 +* 支持使用 USB webusb / 串口 / smartconfig 等多种配网方式 +* 支持热插拔 + +## 2. 如何使用示例 +### 2.1 硬件准备 + +支持 USB-OTG 的 ESP 开发板。 + +* ESP32-S2 + +* ESP32-S3 + +### 2.2 引脚分配 + +只有具有 USB-OTG 外设的 ESP 芯片才需要引脚分配。 如果您的电路板没有连接到 USB-OTG 专用 GPIO 的 USB 连接器,您可能需要自己动手制作电缆并将 **D+** 和 **D-** 连接到下面列出的引脚 + +``` +ESP BOARD USB CONNECTOR (type A) + -- + | || VCC +[USBPHY_DM_NUM] ------> | || D- +[USBPHY_DP_NUM] ------> | || D+ + | || GND + -- +``` + +| | USB_DP | USB_DM | +| ----------- | ------ | ------ | +| ESP32-S2/S3 | GPIO20 | GPIO19 | + +* ESP32-S2-Saola + +ESP32-S2 + +* ESP32-S3 DevKitC + +ESP32-S3 + +### 2.3 软件准备 + +1. 确认 ESP-IDF 环境成功搭建。 + ``` + >git checkout release/v5.0 + >git pull origin release/v5.0 + >git submodule update --init --recursive + > + ``` + +2. 添加 ESP-IDF 环境变量,Linux 方法如下,其它平台请查阅 [设置环境变量](https://docs.espressif.com/projects/esp-idf/zh_CN/latest/esp32/get-started/index.html#get-started-set-up-env)。 + ``` + >cd esp-idf + >./install.sh + >. ./export.sh + > + ``` + +3. 确认已经完整下载 `ESP-IOT-SOLUTION` 仓库。 + + ``` + >git clone -b usb/add_usb_solutions --recursive https://github.com/espressif/esp-iot-solution + > + ``` + +4. 设置编译目标为 `esp32s2` 或 `esp32s3`。 + + ``` + >idf.py set-target esp32s2 + > + ``` + +### 2.4 软件工程配置 + +![tinyusb_config](./_static/tinyusb_config.png) + +![uart_config](./_static/uart_config.png) + +目前 USB-Dongle 支持如下最大组合选项: + +| USB-ECM/RNDIS | USB-BTH | USB-CDC | UART | USB-DFU | WEBUSB | +| :-----------: | :-----: | :-----: | :--: | :-----: | ------ | +| √ | | | √ | √ | √ | +| √ | | √ | | √ | | +| √ | √ | | √ | | | +| | √ | | √ | √ | | + +* 本软件默认使能 ECM、CDC。 +* UART 默认在 CDC 使能时禁用。 +* UART 可以用于命令通讯,与此同时,你也可以用于蓝牙通讯。 +* 可以通过 `component config -> TinyUSB Stack` 选择 USB 设备。 +* 同时使能 RNDIS/ECM 和 BTH 时,建议禁用 CDC,采用 UART 发送命令,可以通过 `Example Configuration` 进行串口配置。 + +>由于目前硬件限制,EndPoint 不能超过一定数量,故不支持 ECM/RNDIS、BTH、CDC 同时使能。 + +### 2.5 固件编译&烧录 + +您可以通过以下命令编译、下载固件,并查看输出。 + +``` +>idf.py -p (PORT) build flash monitor +> +``` + + * ``(PORT)`` 需要替换为实际的串口名称。 + * 使用命令 `Ctrl-]` 可以退出串口监控状态。 + + +### 2.6 使用说明 + +1. 完成上述[硬件连接](#connect)并成功烧录固件后,将 USB 连接至 PC 端 + +2. PC 端将会新增一个 USB 网卡以及一个蓝牙设备 + + * 显示网络设备 + + ``` + >ifconfig -a + > + ``` + + ifconfig + + * 显示蓝牙设备 + + ``` + >hciconfig + > + ``` + + ![hciconfig](./_static/hciconfig.png) + + * 显示 USB-CDC 设备 + + ``` + >ls /dev/ttyACM* + > + ``` + + ![ACM](./_static/ACM.png) + +3. 通过 USB-CDC 或者 UART 与 ESP 设备进行通信,使用 help 命令来查看目前所支持的所有指令 + + >与 ESP 设备进行通信时命令末尾需加上 LF(\n) + +4. 若使能 USB-ECM/RNDIS,则可通过指令来控制 ESP 设备进行配网操作 + +* [通过 sta 命令来连接至对应路由器](./Commands.md#3sta) +* [通过 startsmart 命令开启 smartconfig 配网](./Commands.md#5smartconfig) + +### 2.7 网络设备常见问题 + +- #### Windows + +Windows 平台只支持 RNDIS,USB ECM 无法识别 + +- #### MAC + +MAC 平台只支持 ECM,USB RNDIS 无法识别 + +- #### Linux + +Linux 平台同时支持 ECM 和 RNDIS,不过使用 RNDIS 时,如果切换不同的路由器,Linux 下的网络设备并不会主动重新获取 IP。 + +如果使用嵌入式 Linux, 没有显示网络设备,可能是在内核中没有使能上述两个模块,在 Linux 内核中使能如下两个配置项,分别用于支持 CDC-ECM 和 RNDIS。 + +``` +Device Drivers ---> + Network Device Support ---> + Usb Network Adapters ---> + Multi-purpose USB Networking Framework ---> + CDC Ethernet Support + Host For RDNIS and ActiveSync Devices +``` + +如果确定使能上述模块依然无法看到网络设备,请通过 `dmesg` 命令查看内核信息, 确定 Linux 内核中是否有探测到 ESP32-S USB 网络设备,以及是否有错误打印信息。 + +## 3. 配置连接 Wi-Fi 网络 + +本示例提供了两种连接 Wi-Fi 网络的方法。 + +### [方法 1. 通过 `sta` 命令连接至 Wi-Fi 路由器](./Commands.md#3sta) + +**命令示例** + +``` +sta -s -p [] +``` + +**说明** + +* `password` 为选填参数。 + +### [方法 2. 通过 smartconfig 连接至 Wi-Fi 路由器](./Commands.md#5smartconfig) + +(1) 硬件准备 + +从手机应用市场下载 ESPTOUCH APP:[Android source code](https://github.com/EspressifApp/EsptouchForAndroid) [iOS source code](https://github.com/EspressifApp/EsptouchForIOS)。 + +(2) 将手机连接到目标 Wi-Fi AP(2.4GHz)。 + +(3) 手机打开 ESPTOUCH app 输入 Wi-Fi 密码。 + +(4) Host 通过 USB ACM port 发送以下命令给 ESP 设备,开始配网。 + +**命令示例** + +``` +smartconfig 1 +``` + +## 4. 命令说明 + +[Command](./Commands.md) + +注意:Wi-Fi 相关命令只有在 USB Network Class 使能时才可以使用. + +## 5. 如何使用 USB-DFU 对设备升级 + +在使用 DFU 对设备升级之前,请确保已经在配置项中使能了 DFU 功能 + +``` +component config -> TinyUSB Stack -> Use TinyUSB Stack -> Firmware Upgrade Class (DFU) -> Enable TinyUSB DFU feature +``` + +#### Ubuntu + +在 Ubuntu 环境下首先需要安装 DFU 工具 + +``` + +``` + +使用如下命令进行升级操作 + +``` +sudo dfu-util -d -a 0 -D +``` + +其中: + +1. VendorID 为 USB vendor ID, 缺省为 0x303A +2. OTA_BIN_PATH 为需要升级的固件 + +使用如下命令进行上传操作,默认会把 OTA_0 分区读取出来 + +``` +sudo dfu-util -d -a 0 -U +``` + +其中: + +1. VendorID 为 USB vendor ID, 缺省为 0x303A +2. OTA_BIN_PATH 从设备读取固件保存的文件名 + +#### Windows + +1. 在 Windows 平台首先需要下载 [dfu-util](http://dfu-util.sourceforge.net/releases/dfu-util-0.9-win64.zip)。 + +2. dfu-util 使用 libusb 访问 USB 设备,这要求我们需要在 Windows 上安装 WinUSB 驱动,可以使用 [Zadig 工具](http://zadig.akeo.ie/) 安装。 + +3. 打开命令行窗口,将 dfu-util.exe 拖进去,然后执行如下命令进行升级操作 + + ``` + dfu-util.exe -d -a 0 -D + ``` + + 其中: + + 1. VendorID 为 USB vendor ID, 缺省为 0xcafe + 2. OTA_BIN_PATH 为需要升级的固件 + +#### 常见问题和错误 + +1. 各平台 dfu-util 工具安装错误的问题请参考[此链接](https://support.particle.io/hc/en-us/articles/360039251394-Installing-DFU-util) 。 +2. dfu-util 执行时打印 “No DFU capable USB device available”, 这说明 dfu-util 没有探测到 ESP32-S 芯片的 DFU 设备,请确保在配置项已经使能 DFU 功能。在 Ubuntu 中,请确保使用了管理员权限操作。 +sudo apt install dfu-util \ No newline at end of file diff --git a/examples/usb/device/usb_dongle/README_EN.md b/examples/usb/device/usb_dongle/README_EN.md new file mode 100644 index 000000000..3b7de4025 --- /dev/null +++ b/examples/usb/device/usb_dongle/README_EN.md @@ -0,0 +1,293 @@ +# ESP32-S USB Dongle Solution + +## 1.Overview + +This example shows how to set up ESP32-S chip to work as a USB Dongle Device. + +Supports the following functions: + +* Support Host to surf the Internet wirelessly via USB-ECM/RNDIS. +* Add BLE devices via USB-BTH, support scan, broadcast, connect and other functions. +* Support Host to communicate and control ESP32-S series devices via USB-CDC or UART. +* Support Host to upgrade device using USB-DFU. +* Support system commands and Wi-Fi control commands. It uses FreeRTOS-Plus-CLI interfaces, so it is easy to add more commands. +* Support hot swap. + +## 2.How to use example + +### 2.1 Hardware Preparation + +Any ESP boards that have USB-OTG supported. + +* ESP32-S2 + +* ESP32-S3 + +### 2.2 Hardware Connection + +Pin assignment is only needed for ESP chips that have an USB-OTG peripheral. If your board doesn't have a USB connector connected to the USB-OTG dedicated GPIOs, you may have to DIY a cable and connect **D+** and **D-** to the pins listed below. + +``` +ESP BOARD USB CONNECTOR (type A) + -- + | || VCC +[USBPHY_DM_NUM] ------> | || D- +[USBPHY_DP_NUM] ------> | || D+ + | || GND + -- +``` + +Refer to `soc/usb_pins.h` to find the real GPIO number of **USBPHY_DP_NUM** and **USBPHY_DM_NUM**. + +| | USB_DP | USB_DM | +| ----------- | ------ | ------ | +| ESP32-S2/S3 | GPIO20 | GPIO19 | + +* ESP32-S2-Saola + +ESP32-S2 + +* ESP32-S3 DevKitC + +ESP32-S3 + +### 2.3 Software Preparation + +* Confirm that the ESP-IDF environment is successfully set up. + + ``` + >git checkout release/v4.4 + >git pull origin release/v4.4 + >git submodule update --init --recursive + > + ``` + +* To add ESP-IDF environment variables. If you are using Linux OS, you can follow below steps. If you are using other OS, please refer to [Set up the environment variables](https://docs.espressif.com/projects/esp-idf/en/latest/esp32/get-started/index.html#step-4-set-up-the-environment-variables). + + ``` + >cd esp-idf + >./install.sh + >. ./export.sh + > + ``` + +* Confirm that the `ESP-IOT-SOLUTION` repository has been completely downloaded, and switch to the `usb/add_usb_solutions` branch. + + ``` + >git clone -b usb/add_usb_solutions --recursive https://github.com/espressif/esp-iot-solution + >cd esp-iot-solution/example/usb/device/usb_dongle + > + ``` + +* Set the compilation target to `esp32s2` or `esp32s3` + + ``` + >idf.py set-target esp32s2 + > + ``` + +### 2.4 Project Configuration + +![tinyusb_config](./_static/tinyusb_config.png) + +![uart_config](./_static/uart_config.png) + +Currently USB-Dongle supports the following four combination options. + +| ECM/RNDIS | BTH | CDC | UART | DFU | +| :-------: | :--: | :--: | :--: | :--: | +| √ | | | √ | √ | +| √ | | √ | | √ | +| √ | √ | | √ | √ | +| | √ | | √ | √ | + +* UART is disabled by default when CDC is enabled. +* UART can be used for command communication, and you can also use Bluetooth for communication. +* The project enables ECM and CDC by default. +* You can select USB Device through `component config -> TinyUSB Stack`. +* When ECM/RNDIS and BTH are enabled at the same time, it is recommended to disable CDC, use UART to send commands, and configure the serial port through `Example Configuration`. + +>Due to current hardware limitations, the number of EndPoints cannot exceed a certain number, so ECM/RNDIS, BTH, and CDC should not be all enabled at the same time. + +### 2.5 build & flash & monitor + +You can use the following command to build and flash the firmware. + +``` +>idf.py -p (PORT) build flash monitor +> +``` + + * Replace ``(PORT)`` with the actual name of your serial port. + + * To exit the serial monitor, type `Ctrl-]`. + +### 2.6 User Guide + +1. After completing above preparation, you can connect PC and your ESP device with USB cable. + +2. You can see a USB network card and a bluetooth device on your PC. + + * Show network devices + + ``` + >ifconfig -a + > + ``` + + ifconfig + + * Show bluetooth device + + ``` + >hciconfig + > + ``` + + ![hciconfig](./_static/hciconfig.png) + + Show USB-CDC device + + ``` + >ls /dev/ttyACM* + > + ``` + + ![ACM](./_static/ACM.png) + +3. The PC can communicate with the ESP board through the USB ACM port or UART. And you can use ``help`` command to view all commands supported. + + > When communicating with ESP device, add **LF(\n)** at the end of the command + +4. If USB-RNDIS is enabled, you can connect ESP device to AP by commands. + * [Connect to target AP by sta command](./Commands_EN.md#3sta) + * [Connect to target AP by startsmart command (SmartConfig)](./Commands_EN.md#5smartconfig) + +### 2.7 Common network device Problems + +#### Windows + +Windows platform only supports RNDIS, USB ECM is not recognized. + +#### MAC + +MAC platform only supports ECM, USB RNDIS is not recognized. + +#### Linux + +Linux platform support both ECM and RNDIS. Howerver, if RNDIS is used, network devices in Linux do not proactively obtain IP addresses when switching between different routers. + +If embedded Linux is used and network devices are not displayed, the preceding two modules may not be enabled in the kernel. The following two configuration items are enabled in the Linux kernel to support CDC-ECM and RNDIS respectively. + +``` +Device Drivers ---> + Network Device Support ---> + Usb Network Adapters ---> + Multi-purpose USB Networking Framework ---> + CDC Ethernet Support + Host For RDNIS and ActiveSync Devices +``` + +If you are sure that the network device cannot be seen after the preceding modules are enabled, run the `dmesg` command to view the kernel information to check whether ESP32-S USB network devices are detected in the Linux kernel and whether error messages are displayed. + +## 3. Connect to a Wi-Fi AP + +The example provides two methods to connect ESP device to a Wi-Fi AP. + +### [Method 1. Connect to target AP by `sta` command](./Commands_EN.md#3sta) + +**Command Example** + +``` +sta -s -p [] +``` + +**Notes** + +* `password` is optional + + +### [Method 2. Connect to target AP by startsmart command (SmartConfig)](./Commands_EN.md#5smartconfig) + +(1) Hardware Required + +Download ESPTOUCH APP from app store: [Android source code](https://github.com/EspressifApp/EsptouchForAndroid) [iOS source code](https://github.com/EspressifApp/EsptouchForIOS) is available. + +(2) Make sure your phone connect to the target AP (2.4GHz). + +(3) Open ESPTOUCH app and input password. + +(4) Send commands via USB ACM port + +**Example** + +``` +smartconfig 1 +``` + +## 4.Command introduction + +[Commands](./Commands_EN.md) + +Note: Wi-Fi commands can only be used when USB Network Class is enabled + +## 5. How to use USB-DFU to upgrade device + +Before using DFU to upgrade ES[32-S device, ensure that the DFU feature has been enabled in the configuration item. + +``` +component config -> TinyUSB Stack -> Use TinyUSB Stack -> Firmware Upgrade Class (DFU) -> Enable TinyUSB DFU feature +``` + +#### Ubuntu + +You need to install the DFU tool first in Ubuntu. + +``` +sudo apt install dfu-util +``` + +Run the following command to upgrade. + +``` +sudo dfu-util -d -a 0 -D +``` + +**Notes** + +- VendorID is USB vendor ID, the default value 0x303A +- OTA_BIN_PATH is the upgrade firmware + +Run the following command to upload. + +``` +sudo dfu-util -d -a 0 -U +``` + +**Notes** + +- VendorID is USB vendor ID, the default value 0x303A +- Read firmware from device into OTA_BIN_PATH + +#### Windows + +1. On Windows you need to download [dfu-util](http://dfu-util.sourceforge.net/releases/dfu-util-0.9-win64.zip) first. + +2. `dfu-util` uses libusb to access the device. You have to register on Windows the device with the WinUSB driver. Installation using [Zadig](http://zadig.akeo.ie/) tool is recommended. + +3. Open the command window, run the following commands using `dfu-util.exe`. + + ``` + dfu-util.exe -d -a 0 -D + ``` + + **Notes** + + - VendorID is USB vendor ID, the default value 0x303A + - OTA_BIN_PATH is the upgrade firmware + +#### Common problems + +1. Please refer to [this link](https://support.particle.io/hc/en-us/articles/360039251394-Installing-DFU-util) for the installation error of `dfu-util` tool on each platform. +2. "No DFU capable USB device available" means `dfu-util` does not detect the DFU device of the ESP32-S chip. Ensure that the DFU feature is enabled in the configuration item. + In Linux platform, make sure you are using administrator rights. diff --git a/examples/usb/device/usb_dongle/_static/ACM.png b/examples/usb/device/usb_dongle/_static/ACM.png new file mode 100644 index 0000000000000000000000000000000000000000..96259f04a7cd25b7db2227d0b56dbc64cff53c0d GIT binary patch literal 43409 zcmce-1yo$ywysNXFWlV|q;Qwu4#Ay5LI@Vz-642zw-DTeL*WuMxCWQt?ski{*S`Cl zwf5Qfwf1hiw1NQ?%vrNWpS}0-{r?DmrzDMrOoR*t1%)OnBcTcf1)mQ2J{}Ps^3^^x zHxDwvI*Q4vBSJpB5Z{MEetzYsDlG<8F-o%g@}#+_q9_zpbqvabF&yM)Bs&=$M<^)t zo|n&8iHzvPkY5ryNoqT(*?w?xHFhwAGXG?2W5(=g>R@K}$?>DD(=klDFcehLsjP&k zx|`uaCX$z$#=^7jHJ#$3)5>*HL1GR zrk(9&J9{;{*T`75{uGD-e|;R`89o9Hfh)FdGF0!L@oUo$JU9^JY{1#W>OC}Y6R?CQL)8nb{PKRU zPAC`=<~Tc?atopAUZ&i`z&5W%IL{d6H2GN5WC7}zGCPi{6vIOpLe<0LEx`xTYlpvG ztg+yNL2oSe@d2GzyBW2Qa9Z?-Vy?#R#m4XF@?M_b-k$W*acWT;BOQ9MBcYp zisA$_9!3~O>_HsLlP%Tek%??U@!`ov$Y-QEvbMndWU;=GRkh5g9CsNhKl>I1Ea;3A zUXW~maYpyBb^DIDq;g^mCGjRe6jg>{E&h+D?pM zFw~I0+t~ouB{jmN{Y@=W6^3?*qg;mw^5n@-I<*e7Zgp1>Fy8LyfE`s$GyX6>SS-Zv zStlGgD}qdZjBq+S!^3J03qT<@Z=RF*YXAmyII8BckmDf&FLQkjiZC8gb!w2@RNZ(Q|op+$3GAzdk^)DD$u4iD}kMp;l zr%;c!-6K3ajf-qNkE)9=t;t1j1%RI6L9N2D4PXzL_Vhd6r`CEzN?YIHcEO`^ei^RN zt97Sx{Jd{yHZ6_RM1CncqD2B#2!*9n`$swX!ixAO(~-MwC)MP!vnhciF(YrgQi`op zL;9EbCmoEH-x5l?Q$=Td3;$Ge@9TML{ zZFf&MN-Eb?aE@=k_EY$w{t2J zK3;}A9z%Nr1fc|Hn2JH4jFU)Ix*PY}b zCX9zwPPd)I&MY<8qg|DJKN1pN!!eh6d3{IpdO)WdJk4AgxAzz|?a!Fzd%A|U;vjEbmG6ezlYvj{pC=yYnrq>yx5vc%=J2kS z2m`l!9(m`VNEaPlbw#z+H|BUeL=0LWdzuX_iHn!d?`82p#euVzxqeA9J za6q|$y~@Ty^nog}DsdYo8!aN-v}%3wi80hTeH_+HqHD7_LrAYMJ3A^4?tn83_3B0X+(DhhfnR zeTBnnDk@F>mf=MS;Ko-)VOGjUE1)*(R+sur8q=Be5u&=@BFNj-SgNbfh|QEo;44;< z2tkeo2r)CH#9$?gyB>)3ION8u>S_bFg@;O2k5yHLWVVQeNGOXgar)vswc-h=EeOFY zL47%uuU>y`e&G2CL$htURIBEV63jArI(LlC+Leb~M}^>qMjk2)hz@z^fPY9+N0Pi$ z+?-eJ>k=834hn^i%D`cP2{F2uIz3@%4l4UR1C!Hnhzi&D;PZN05fEg3fA4mmexVfe zV^WWt>odP+q`@7A34DZ#I&33q6FznTZsQkSkDX6+lxAt?HI!LroOr?xKzQw}zV;-W zPG=7(vcL8K7(ziGf9Ckx&M*MEbCG&e35()9I)+9_$qWP8oO*aAA_<^|&QO=$SpB@0 zX&4<)aoVM05-(hPD#SOsij1>+BOKUp>#2OV)m$Wkwf>En<;+IysrsGIgQy92B4PL9D5u$l zE4|~;BY=YvVM`(o`^yk!sZn==7{N!86h3d>gL7rmm0YH!CPFzGLP@w!I0&n`w8pWI zCt44eJHw-{x_^(66vR)1o7$3VrwW(8daE=_CrAXJF-?!e=b-4r_C`1Efj$zt@y%~R zUHnA8%{N?qKKHEKPq$QQ)1{tKhD8AK?>H+o%Q5k4*e4e329{KRZOQU>R_h)!zT{P- z^6k=Bxvp5?yg44m45zeoFCH^?UQy_2{9y8&#MtJW8$Pafxpwf#Cg{W~rH~dg%Oi-; zko_EO3;Y>O_qVJCd}|CCpoPoN{OnowhwW%UE7wG>G;Xe<-9|H5F-vlysvxz}_T zCXTmSDZssvZkye^T6?^Dn|z)BaFpdJ^VhNjYs{*@1bH7iPZjjbhTc_9ynPR&2gaLL zeEy}y9#D=@{Jcl}`$Ccc*VLUJBm{HI4jf2!7X{y;3#G=#R%ZFH54BQ&w&8B~1THpj zre|}mCS7xcbDu?(?#Va5LxvF)aj)H`T3JQM5#Psub3uq=0s(K# zO^U#_2seRpko??p7#QuJ=O0EnGc$|@o(FiP;&1->5~xu97nGj^Gpy_TCU#Ei*)eIv#8pu)mpF?z0w3D*#>;w`Y{P$$?H#-D#G1P_yu}? zgwm{x5%pNGRxecU?gjO&*AZBSP1CAt%dp+YtntPRqHxXE#yV3#ACK1f7u=b}i-I|S zFC>5bEySB2NzdJ3$#NMQO7%5Rn_3*qPgNt)lTTe`VNUXyeM805kzchTbd4ELKy!~O zU@!&wtRR$(EIsHg_~K_B@SZ}M$&()S9=(t9`Oe9U)O?*@H@;FSECjuCUTH&3kel{9 zT{yHOmfBk||9BnnO4bEk`@s;01(X`KP?kC{M;ap1D;tF|X#jb?laGIGcUq;WnMh6)`1&@w&;Dk^GM=6=MRp{913%otaZ4GisFCji1WSQMCjo5F8O4W3_PB z3dI%o>e@oVcPIL@vv-hp-FzUKPH;)w>vX$WJ425=%s%?AVF!x?tKEA&w6ibl&8yhT zXge0+o6SXx2jga;j1nvRTB%6Njfay_U#cpq#=3MY(=Ot;xr;CO0R?D{UlyQtF_c@+ z%xHwe*baHH2ehPvZdBDZHhyM7TmPDF##s}S9b#B~x8Nt&w5J^Z7R-rhI{f%5MHp&8 zx%{o1u>5cFw(9jE#4?g!cp!shHndH79w8xaB!916sdgZy7+&ZGiNNmNqN;7=^_HDL zExaUJ-@TVjY~|<)I$o9euQaAg)WxGn6*ykCpX^ddL$V4mw|fd&;0`_0wKN&8)33zP z@3~}@Yy!lqc$T$!(2u3kT=2elS#;s=G%TKb_=P;?C!b-NDssZnP%rq+rDe$B36$6A z5Dt!cZ~j3SY%AfnEec35JI=Ss2y$Rg^VO;(aK2h$zJBXha=yy8`&j&uAtP`pX34nR z1E#x+uvB@(m>x^5nohSCX2OuX@D{Vkcv1E)kko31WC>3qul|zQR%a~*h^yJRzKkg| zBL7xRmw~LxVxIe$hal8(t~nBs$;nb@)|C>J={4`U8hG4ivV}@~QpY|}$K}Yi4&__+ z=tOZ;TqLbh>ro&}2}%Rw6g0UVZVkDy_y7dO0TjFkteRP7nK=cdACzlUXk#-<@spUV zf+aQBk`zV!sv2R!v~_}z}(s~uB*A7oFqNG z>CFZW^^DB4_Wq8oRC+XaL<8%7CqC-Frh31C85zOn517PV?NLP~E)y2QR_&vS1sdE+ zjm0R6QGkA3Zeig^j+C~pYN-TVCy@k=RS6NW-7*-8n8-D4%BKUD4WXqC{quyhSe1pO zgl^oho?J@IjIh}IQsK|0NkLwIfUGQQq=X$)ivv*!#a}T+8U;nzi^e}xm3giB)cgFs zEbi714B}Wtp)1${^7N#A-$P?44icf&Yl{eRj}SfMn^>ib9zWKTFVB)9QI>_;Fm(>a z1E@TX?0fnSgRhGuyb`_%r{X(A`{JKN^#y074O$#uGH-qtP&%f2i4=}-wO5XCkVyIL z&dcQtiIhcL4isrY&LJu1q1jQ8yj_1{roi$FMgVObcWi<>Tstylk1G{pO zr%S*nw|fVrH5laWuB5htp0N2zxH(c9awtjY+D|&HA9Rw}4)G`KbLruE&kAV4IE{6W z@L)MhD9?zq_QB=wM@DHM(R|BAJZb}sKK|LX{CSy3WbqFUymXmA`kr(|JEKj%P`aA0dg7AC<>Qjdtl7=gT-jRG{ba(*c7 zS8G=@rK=RIt+3AMG`u5+NDjQsQa!L6G21~xXufPn3H%lj-zO5rUPZi6?rsu{IUmiT z6i9f2aW2sq%8Xu_+JvUqt7 zx+%>bnbIB)T9~P)-XsD27WAv8ETnT>u<}OXJIqqcb;lX$O^utZ*g^gXR*Fh}iGn3n z4m76Y%T5>{l1BXNK z*?T=!DbF_+OwZ-fq|31k33K+KVpNE(3r(3eS8v3&=Hp`_ zdm{ywTHLjY1=@f&d0WHkVS$JnTjbX0i^b^{wU%eFj;xJ1;5JJ=!3)$x zn=$z24DDPhVb|?%0{d79anA}ClM}N!KJ+$Gv&%pu^^024`v>x$ew36UG!+AXzun8M zmPLH|XI?F>o+x#n_K5zDk0MO1EBMO793(4;k$ni*3fJ05X9Xkp2mm1Chp0ebG1mG| z;i7v5#k@~o05t93EEfiAA0f)wv=FG~g6g34wYmLv1R6qKwf6?^!ISyhA=w&Q03$XW z!R^F)DVX56VxTLLPC&5EhrNP4s<(c}Ca(RM9naTroPAfO2+|!jJu-aSS-%w7TKCAu z{Z<*)RhRH%?tEM*MMSkru7L^?5i*1u;!29YzbD_}IV>0y9vUr9WJ!s0O``mwA}QlU}#K^6?kL9?Di;5J|o&AU6x*jBq)63?6$ zH8%)t0D}4WaT`XJCGW;W33_Qxm1c}=(2JNmT^IMqnd&9Gm=LtJ`AtOwaHw>+z>jO( zdwSdjQd-`TvK;7-x_;1np&b^6h)Hu4{27^ZRd)x1@45{Shaf5Zu7-dg@!4zU6>oMl z{;XEMdDej}_~Ndnruz#puXya4>4xCJvznRTD_r>}fjQHFR)aU!QeS$+sLAD# zoH{Qbl}ju2(P_P35GjYdXy)G6#hX}QIhYr-HPCv^_;vqfw>$4FO$g~&w|Ap}$vk@c z*HR97NC}8><*V@Wgs0qyULDuqsV;mQfE0t;-Dkk2oozH?IiC!<>75rCAvZnOTyd7* z=k4``lx#K1m)vhor=*x6;`%f~$A~W|7^tDX?6=~>EE#H=&oZcEXVhkN@?562>r3xnj_ir{!)4{y3BELhc%e20ulSbogoihVt z`qZ}?XlRda*lE$pCDSMhXhF^1Ity#_bt*SZ%S5>SYHatNIMzUG&$f!jSBYifmnX?x z--B$?z8Be-k8eEzPn{O;ww{pCOHF52qATG6R2l5Bz6Cl%u5JWu}SJ^LnT zup2a!oEK;FlVk3cFj0dH%4<`cfn+>T8|O>==e(GlrxHzPW?xN`STORjth76~ym#P3 z%lkFPf*Vy+4-hlwEKly&B6N#`&GY`TFV*jDaNlPj=8lgxp0{j@_4AsFf>F;jvg1N= zW5~NYa#oBuwn^%>91f^)Wkht#JV?SIMMKv+JOsUuen@}Sj*IjhFH2JFz%bH>65%$B z%EbvfB^Q zTxS#-%18RpqDUtUvbl-&5*qX!s?Ot!U(uyF0P)o!y2~+ zk*<=u%Y16tFIN@=Hy@X>mlNpKZy7u8*EEp84Zh*wSzQ8C=`-vS6!rOe?Vk0T4oBo_ z3-uy9?@VB{v!eA6P63N8nw^Nv_p~$fVX7G3rFxwpdd!i>f3-G}2S=gMg=w5kdxOel z`aF!ATWUsw=PLLJhzZXMy7A5Z{h*MJHp4Su$2UmTWwpGGh2#=-AxjbVP5G9}Ka2Kp zDPJM7h4sTasXiOEKB1W)d@x^UD4LmKQ{6!Rjp{fyIAJBazQLwpj0y=_XQfbIO=~{L zL&FV;d?OO>mc2)dQBY9KtJ=eq0q@>17JqA_a{rCunH%j{!LMYu9A;=8{KBHdT&b;Z zggiTERll=ZG=La{?>=~nTeSK`)2EByUTwq12YbtNk&YoxTU_4D|Yh2NYP zh+pp89*fyOigoHgS~l}f_6wga&UR*EG^wSm1Ly_aAcfW0=gMd#7uCzBfG{JUURZ>c zu+TNKUr&X!fGZ2^&*M5R&SKvYvC>b;Xiqgt5-#qupzb9wTXG1KURu4j63no}gt%oxJ_R#}`jO2KWVtY7-k;i!n5}*PnlN#p$Ad zTGp(xtmukqzq&M+j|ct&^P+lk!isof`;!|wsr;IGeRka`y-#&=o+e5_t+{$PhJQNn zg=Sa{cn)d~#xe8#sqYU=Ht4u>G6mRe$`=Ar%mr`5gvxl~_R2F)?{+ZfhL?(yDq%2x zNU=#;kO@EK!n5mXl`V4-?#!8I=x@P$>Y#CZCB{(Y?@(_JVSt$J5Q%nBfiBSR9FMqF z6+Cu?EVw z7q&nKs~stO(}JJAX**-ldA)RXLlhvUS%PX#X#7#d{*hm~Zm$)(loBFNx4*pi0vUaN z`T~je=>+}AZd_!Uu8@tX@s9^LPr_dhh{Hdkz#(66Z3LY)z(4KhUgeiQ+Epyy+}HmY zP)rTN*2}B9PuAAg-v`)gHar9d^OnAuo8|-mWbX^gNL#QNyM`nLib8{J6;6EfPfEeV zxIeuzQ+tcU&vyj+mEf#!0R6HZJhN-FfAnnE5}jNFOq-ee<=lR|eYpU*IY0#|Gh*fnuDG=U6e8C-C*BRz`d z8(z#bOT%tW1ipSh_q>CxePZ#dVoHwHDGavG0`6ut6dsCS|veNDn>@+u@%vDIiBW)l_Uu(np9(7V7FIVH;@1 z{Nc%d($dslefISZNAZNfoMs@|7)*d=Zgn%yNxh-)3hZ-?+Na3t@2Z?m!J{q)cDnnr zW40E4i(j?~c4o;8n9?$>5BQ>k~mJKCY%ovSYo0Vq+yw{unH1KCXe{oF?eBN{&ii zvlMpU?QaTMMGS&j!4b}cb}6`@t*pG)Qk-F`uMN6R4WBgj$SPwBU^K16`}9oT&s|)+ zVp>|@euZHzB=e}X;^)P7al?++ccug;fJFB|@}~j(Krh0errW(A89`ob3}Dm%uiX$Y zwt-~Pl=R9hlJ+~IDIWqMjAOAJ;m)vFwRmwQw7?AZf#M&#F-mBs(=O)N=KhVa0NzvN z_Xh)#s=$c7A9szzXp)0jA2Z}y0k+XwD)tXV%gf6Go%94}l3-K_oC)=c1Byd>HcAlW zh95}w8Nk0~hdnDGhk!`%%lYm=_mqX3Q&X{O-TxRTl}?bIqj(Vj!|5Mi8EOVH>oNY^ zuka`nCVB8KutRd{VVM1(zYzC1G;8#2+SSV09!D(j%nWf;USu=EixB#D0&q(O3V}d5 zaqvJgWi0R!!aL9YK&;OT06yBnZy^~(=>Hg9Led2-^74L544GaacC*ef`$JfPP#~M_ zIgJ9~@67a)G;&J}adXJdV?qlQxS+h^e|PePz<1~yI|&&ck8=;%JpLwOW5C)c|Lpxc zl19v1Byg(^4_st*9rm?m%|cABddBo7{yU%jlc7TJsw4q)UG~3D_;TJXiLqWf^v=qj zJTU94UeJAqY$+1vw^lrVTM>fjXS6!jY9^t}HMcp>d6Ow9Av7r&!rGAk!hZjC>_4$E zX2{#a!hd1t7aCoedv6jyM+WF!6lJpQ$tss*P%E0f#^c}brrew+z@b&{I>IM0&NW=b zq%?yyqU=2o3E5CMW14zDCN|@b*eWmtpR||k30aiEkiis2byR=nD_RtRrIy^Awv-SL zUA0z%eqV2DjTt)D;iI;18^`gSTG_7*qrZB1YSW-h&EOY~ijNB!QXRAn<+X$MdsbD;Q z=^p0ELl3^|6pfk*+m3Wx3Sr*#Q_=G_v?G5rGYr^q(x=0Yx#(Q#B-zB3yxDyh?_qPC zTPe)%nnx-&_**&>$yUN-QXaB|<6f1sg!w8icsiIIcSF%Pr6t)C;$_j*E!2Z=TM^OQ zbm>p9Q{?U08VGkV-&FEKQ(bV4NKCoVcxX^)>#3u0T@SPn)iDlbN^g(DI&g{ z?pb8W1ibp7hA$*wrc)~%+gtWIqL6MULJ%nv2cf1Hq2{-xs82^P=k5ARy-qq2p*%s;wXh5_OPVd)pj;9Ml$*1}cWNsm3Z|&xc z%LIo=v=@x8Xws8Vu%R=F$X^g_MjIm*olJ+uw(;)Q?v$$&%%C<^Kg+AVB0K51^s+G; zQ086tHj!7|N1mn7W66Q<)CBmB(B$73JHiVHxEnXjMC|=UW_=Z zso##{o_t8Rd(lAmPh?5JPa|wf#{Zo+E5;X)KnQUq3nYYob3~y);KC3s3qiV;duL@H z>@hPRVz#j-tI3&4tItlUBEWAYckz z^O|g2b$O~&3CpQv-yrdl<2V^wU=0)(XIvRVbextA^^d3%5ZVJF!G8o&@ve8F7{+EB z?z~N_z%y&t@nUEyNJG3&kHR6NZLfmS;2bMWx&ZF~2#JLR%)^5K{if%o!iad&>leFn z3H()u%z*wGtxRJ$KBa>yxPK)CYmg=XUN}SX~7lu_{j2Fkysvq>6wY3wruF+3T3zPC-%A9rhI8- zh5m)E5@d1wcL+^%|6XArSra7YAR_Ah4N}Z9y~C}m-BQUPBpu!t%yjYyBObS728D*` z{pJln6dll_2en2KxC|QDe90{l)e(y(?w7NVzYPC_B=dL6yt_AsXaaAO~vjI-V^v9+l8e}NsFF;+gfQ_GaS9UT=U z(L$rJ`CX#yyv{Z3`@os>tl2NY@XnpL{P6S5gcFGsjwXCmL-ixZ{)L2Q3W*hft#qu& zj(*O;L0?ck(yKmV=2Md0DlWrtMR3n7{85GKjqG0RkE_6jx*!38#x{s{ejuOfm+{LE z&i9upI(OEtUtnl5uEYW8ImpQR7gpeuyqf)kH+j_!8{y0S>r&-Vs~K#(ML@=d|7kU| z9L;)2E{K;)_yuk4MVuaeX41E)=~mmP-?$ssw)R}cyS-;IJ>9`&4lhKacmZ3te5RTi z0#t!Cv34Y1U9fnI;*sgF{yWOv#*3kY)Xekgx`^dL*#WIy{4ga(JS|bF~%qWA6Mq&@}Ay+g==QLxcMIm1%i)7AhE(v>)lz5ar7n$ z0`$(#NTv!DdsgW6D14=nv+gx@q<@`b>E6MRsW`T-(whWdZ?D^xdvNW;K0wrLB&5ZSd zucK>%5C!Wi#*dQILqLPdO3iDo<~>t|H2dGP&^QBpuuA7*RXyE5c_7&kr0Jr4CdtVyWC6IHPS>ttK$CUeoArP@o3;UEATEp6pHwAm+^q>_SYdTX z@2m+gkyan%6tyTpi~tw6Z|3Ys>V+5Ac=(XWGp$1Q8z~<4HF%~ESMkIV*0RU=p>`%u z*Y7MZhU24uBy)pYH}cFKvChnjfM%KfLTpfLNytlgyYSrjrCRBb2DVKEC+mI}(vSck z)kJ>N7){xw{ta-rkD79kg0u>O16=ktzkoDJ_!tK0oIQIE;mW7ecEq+U{1MYFj4ErJ zVRNU^js1(zST2H_aLxhrZ|_8=e7;|gIq2VBp^Ej9cIKCUGEw%2#U>*RiE1&a67Ec! zK4aM(KbLc`R4r_qXoyt{%-o9-O^e`=E)+^9M)A+EbbS-l5@zNO!*T%Zu)UX$)2;Bk z9#*-zMQcEhVV%xM{=L+$*R;AuSe9pP+Wx5f8NFYMIiz+~eQbLl6@d-AK*EI?TUp2j z83|>PS^298sP0UIFZwwlf|wm$eBB#^cqgW5nPhLw%%qz(Do*T$J6iw`UDV?%2B*lz zI+Fz9dIHEc(Xpw z1HKt~uKl{@fPP$%*qJN&WzU z%T9O(+*$+f%a0Fm|0X*WKCZ15Dy|n1H&SA?#>f?g0$Ijpp8g$CTRS3uC1=(YI{R$&?tXmUCxH_u!FVza zXE?PHFn*kvqe|1;x&_0k+qM$$d-rAX9SVh(A(P*ofgJWS`D45q7wVswU>!<+S=5SM z(ol|m(K1NT>XMaUcJ(g-SxLsa>c9zmuHlfmT@Z-Tfx6bD*aEUqdmdSuD-)g4guY1K zQY;6GJNgLw2+o6NH7PjiKHCJ*UjP4N)&63x-U}a#)P!q-0yASSMiaUIVAajP|76ws zOFi6f4?s}@pfSvBv>;9C{-V<^_vlphodklBO3qo;=|6b$!7ANKU+`7E2gVEArI>sB z!FKVeF^mQgHMVHIsiFqylJwp%|1H?*cAagz_gz_*NpQ2HMHnb0#MEpr*#q+hQ?L8i zBff`8$kPlV(qQ7+=nH3mUu{ZvRoe-LO+&LO9||0WepP1M5)ZV)l8AAAM{95ye92(e zU+_Orb2gt)8SwXeto1w|$g<@x7~I{#&OpU{}U!cXQjfCEd+b?noyHj_R2ub#n{4F}dt}$8@zr0UYcH&1$3_U=D7>*O40> zO8<$g#cBV<)sL%x;A$=*Uqe|X#l!E3S^E(BM5zSVLjlp3lP3)aqJrNl)d|?Jl!=2K z+675j2Mf$Hhhz|L!V@Z|^WScjdOp?$wS6e`0(cteNX)iM?0=XJ!%SaA0vP&d7-VJ( zX*)8&WP>c=cYcSD0TlXnNjFY4)%%G!|KD)!`p1f}JWc@l^E;T-az@SLv`EK;M|!QM zP6$at18wCGz-AglDuw|#VZ>)HpYmom9r58LFJiA=DXtwLtc&lTJ&W2!FeCpN9?*3Z z!4d0Y2aWzr6m>4Ec9Q9kz0Y`h$$Wj5KHmEtUp=&WvAQ4Ba zN#|E@+~3F6*wW-N<1j2G$6!nOMg#u?@Fs4b!OhXn?YPVauE){JrW6`(w>zP{9(*Wj zf(Eg)_+9Syk!=nnyS(HXFG!w2J)MRQ&9NUnF_HBdXM^m91;)A1oIBTXRPRj=r0p>nr5<2 zG!f25n;G|ZNwhjhv>gkf-A-`$AmtRkG%x>z@r9Jg@6D%HN2&J}UqOOH9uiF(-gLJU z*dTTiw|!W$M;h`UW>m0fbpZ#OY%>o}us-d2=QQo16d;d#@$g!%2kW#r`)^R7c31cf zx{QvNLeE|OCLlyjGrwPJ)Frto#?ablW}{ldt-h8sjst}-XJ>Rris0oaXsCDG=H#5q zQHv=l9w&_FzeLgSU!w>o%1rz6MyG69%^xeAOA4(aL{bH={(%|^;H8ajHVxP7peDpp zs7~GRC2;4RbYxRMR?WfU+{3PWk5F=fM_FBkJJP3#ui>B9v2_}?lV64E2?+35S9)2JpKhy)LYk>89@LB<&#$4cE*nnG}D8 z>VIYK+gqiTq7b56NA4h}q=?4HMe>jFuqQJ@F(VLRTk$&-u(UI&9QU`gmXIbn`V&~S z5tiR1!HR?BP^MR0*_=Z0-q!t+@PA|bzn%d?^`92KNgdWMSqg+--JBg-6(P4lU;8%Y zbnsvlv^8BAc~@kX_>HZ5!7gWqlz3M9dB*kI0}stEnHQo)=XkpGTC>6sRndQX7Gt+?D2<%`Y$$PjJ=^)6r>(s0Cr;g!Y?$2<$Lt0 z&~*I1Gg?@O1=k#2E}#dyQaBh;$@)zE`>$^E0E#dOA?+xBrujF|v~+iv!`aypZ%5z8 zp!H#|VKrM%?@+#&w*N-WA!h0biYLz;x8r|E0T42;7QhNCgrCj}+m01+*9D8?dk8~c zK28hSPDpifh|qo7o{L9^{`SAH`Aol9_7DHFQQI4B>-hXT3k%;YR}yi`O&GcLXtdwM zm^0$=w3k8CQrG!0YGC)gaU)SSph2mU*l}DfdkGcQn&oX7FG>aJ^5iE>)xOB2tE1+Y z6ui&Z{}K^s5!LeVdD45X+!&V8P28!kteM4hY@ff!kKWlKx^uxx{)n%-hm+PWnst$o zAQx~&16_0+g_RUI3VoK-6K&)U;H`5T=Tyh6DC z2~}oD9(eJ6zDOxMB%(dC*NSZhqf{P|(UfW+8NN&Zf1~sNYJ*0Lw0LnB|67i4m4sF{ zC7JMtuQ~&EKs#@-zQ_9 zAfT%JuzyCPVV!7m5~x==4@4iO^l`i1|ES+4SX!Vg$n`3H`J@j6$#Z@gpjU(S)6Jdf z!{YbWZete)FG_#~Flz&{ICJTmQ?g(gCCR8S40SZ^Vj90qS6q)1P7z26Y|TRyA+=Pk zJQTW^4&7-6`!Ifu^l@tpy;&b|uI6nJ#@HaapW!k+D+qrFq6%U{@8@%IIBO_aCO95i zbNPVwnMh$j`9H&Z<{u%k?B4>soF!sWmWfbiCDtS8e_>FRiaJdzeE0x>*a$Hbu+`j_ep5x2f(H(C)~3)JoC{v1viB(rC+g&1wU$ z1a{i^r=vV`l*9uFQDK_LA?-(~av5Q$GG0?>do{AOvU=Xa2;uX2zum)42I-K50>bUy z@UIK!X49XX5ObkZVrc~1T9k~n#}Aaw33z>38$YKodc#|GEb^X|tHlAx#Bp!5vHnW! zO+l?9J&qzgIOgVcQv!63mz$niwqkoV!GA57$?58Rb*gq?%xvCgz_d1MlHlsnFMeH} z5m(#s{a$EwcqZJSd^Jzb;2-n%pL77%RHPriL0fI#{4y*lIxKAem_bV;^+Hbk!k(8Y zEQCy<;P|-bIb!BUFHstF!M#IurCeK@v49{H=z$%sSLc+Wh4qzfh=s)*p6al^3lM8; zSi*VK8i!V~!*xgqBgZmQkgiPx7cJN^_C8LaOvBblo7eQRnD3y+_RWlhm8QI>cgD7K zPKXWtF2IGUV0E>g0;{fk<=rrHzgKY;JKlBlTT4Uev`X))ga?QR97fs;Zyi z_kk>TNjNX1*^Avi8YtX>psZXEJ9u8r zc}0=;?MQm6M0G0suMT1xJcMEt8dP$ zoFn6LT{a!gq97qRoai33XRE#!x7|%oNX{+O!&R@VF zMhN(YB}x`y%o~lX8J^_a?5ThTe{}y&C9{i9H2wv^jh$~?_y1uNbG1eycye_AL_lRB zwjLTzK+p;>{YKIw)a`0AkWbi1VY=vCAA~X+xz3LHt*@nPc^>=RlouzNlVc^x&a7qG zb2$pGj(p^*`@Wf)GH9zHiq_5jOjKe_D0;53?Sw2bEM{UVJUaG4+GyFz3&QU*u9o@0 z7z4$xGhbAJbGrq(v7Nr4rY@zr??!83p&#apK5Z=CS|J_vWdGe6&JmJ9izQ4FGg5g+ zp{*Vc7`ojBZ7p2RR$Tzu z0nEp|A|vRPNlSF_jKb>@or=1uXd1Lnc!H~j`zl_a-tQ6)AgjHb{K zYx)xyq)}b0=d&Pt7<9cmCteMSp!6~Av$sBq@w0rSJ8w&bY~1>%oZy@=;}S+lA52T% zwd(n`#9K+eLLzbI{SxAf1r+wu*`PP)H3fv+A`MJ4OEps{AWxBuNSUY;u*Y|^?J96z z&DO*cZ$x6$3qkg|e7_FWWRXds5k0S?^13v@9J=pGVg2ikf0UTwM}`t2 zm99}jUcBqt?WQi!<8VvgXS%8S=y>RJa!U8gamvD9jthE_EkhKA-yG!}?U_@e)TC@T z`-0~4xT2Z=ov|E`=4?q}ID#)|U=GIZlXAZiW(-MC!;W0%*3RnuZDG~#Tbf~gFH(Fn z!H|*bZZ|wT=fbn9eC=0pl?WfuwcxAHEGu{ddXP5{hP&;jd)^Z_@2O?Fp`DY9L~XTC zSFft(oZ0Cv^57(Y+tALhV*l70br`B$NMF;nsqGo2<1G~|(3up!v7T$PS?Zz^T}xcC zTB$eKdDkxt5d*S-qtm$d)b4g9G5+H|+HUQ@EUyo|;6=hH0A0$_DMXk2w;BKu;%ify z7gGNxWbaJ{a+8GhT}5qF-DkgI#!qE(*G^^VfS6}Mnm(f`xI57U6i75A2*sJO=&odj zW1ZbY$W&LAfANPWwr3kb`WtQ#)zbY0L94K)fA?`_e`s==2HSMp@(+kKz~eoQzwrxK zr*YzaXyJ@<-(`OF+*K>{*|kC6`WU0GQja18Z*I%Wfaq!9A($AVb;wkFI{biyVCdg< zpey5LvwHL%X*?{Nmww)ko@$fc~|O*R8}TmLi#p zG3zcYhRF?OAkKe|_*LSo2y0lmIFi@zTq;$F&B zON+4T^vbwW`lOHo{Fh?zWARTb`q~lxM;Kx}pd!9pY!P^~Y(5p(ETS{F_@}Fop=Q>&Hj`Ok|`mZe`wFDLg*u)+>N=(kmlic{e)6Z&ESZb z534TR$=y=l2T_AOymjV{f8dYLU3?YV)Uq|{%;Th)E`|_L8Nz%~2*52}<6hQx!Yg&v z%Io@pS-nAOAJYIQFmKAH_0H8?wF_cM^adkylZ2L!x!fo!bxH4}t?{!UK6^|ndPlTF zCyVgGudG6L9C2aBv?c~`>LPWKlLz_&>&K$moO9z!S$X9qDDnq$HV!w3!UKz_a z7-DS3rw?$AWfo-GYV_d+?4LcQn+2hElHnFKVDL;O;#7-ehN z5fKc@jq9&^6HH2RX+;z|C<(G)92&4u=h4Q$`$&e$ zJU17n(JtDVh-)m(4v_YQgJ`}|sSc#Jujr?POYuBrUeZZ^+v9-!)*=TYp(|`mMRT|i z9F~IHdjjsG&z*O$R+>XC;=apP)&s?zN}d-aAjHk71W3oK6n@Kl*s@b70SZ3G5!GaI zbAz|?I)i_>vU<(`-PKuTGyo=op4kqgc2$k4B`Lg+OdvQY_%MlHwYpOGwuGeLcd!?Y zFqHCCI4tVCU*8t}a)-To?Mol(SMXg?Q!RoA|JVOXJ%}<_zv6qbmA@w1`lyPBKcu3w z5ZTwvPqLsJXy(R5S3o62=iH~wOThTbrR*?0|d3IwyIzIauC-s-HRpk zlF(iI^o9}@^$v~Obkx|BwqID)|02J)q<@!?^laM9wX6zAp#lBJzD->NH~Qse+^g#P^SZ#K@2hArayyfj$IQXxgFZ(x1=V7m}q- zOU|NE!Uezz;oZ`4t^&VIAUUbw|KvFr=9UWf+C2bVUOTtp1VWT)AZ3$>fj!KdEfT?p zf+`D3i4QK+X=McjzxzZDr~RmG5_xdW3WUHt=qZF&XqXX&L1X4(nyhHqimRQcSPGw= z1A@H83|&ZxbnGK?(FwlQ9d8(!mbPBUlG0a@>1l#`KD(-(j8?#(m}s&op$ZUJxc-qS z`0vgC2h;n^g5gE}&-iO1TnEdP$t>YNQbaK%MVO{UO$vMY$X)2TMQ5u2FW%ldtg3B~ z_m=MNF6okPkx-Bnl#(v#Zdi0T3X%&D0g;dn5mL-#)Cy!XEU z>GOc7bF4Au_{Qf~B}2Lt=#GWidwP1ZXL3G%?P$&qIZAsz5J+h>&n zkMin+V{eRv*uKadqsm744?Wx%xhKb1n>3 zxQyB!D!VUrH5)%l#sg`Jk0)qHibj_;5kvK3_b3l8{K7oVtZg0Ad7?LSnki?wBEVsy z1B4DCPv_=v<|Bv6q{_xJ1M8Wne04P}UUWWq&Pf3gM1zNy{hWFp7*Fk4AIS2&n6pSG zzr18Yx>VzC#T$a>@cx@(-Fz-D72ob8N0K4{)`f=YQ?6?3p4R0NQ#&ElgkcNYrdsd& z?;jb#lE4Dh#$$R)^T>OIei2VeET5dEIxI+&l#g-YosS#%p(wqD>c6EMkYWL&IFQ9L z{cR09Xf|Jy#KQR&(BonhafcV%jD4uO%zMhuQTaiD7BC$+RpLb6Ae8Eql%pcK)m+%N z+2vWO*A}-X1B3*p-T_vpe`*rc1T8PQ6d}NO7bH)HGUcdVK!{yV(oxs?U z`?D*2fp?ab_y^+w)eodbWVjW}7Z+~5TNfSwyxtBs=AEwNvKRO_atR=Yc zQD)Pyo%xa7R@A6?Y}2!4r793bnC#NHe;haP$tGEhY@ZFw#d&XUD{Q-s*)DI0+Z7q{ zTOQp6$RqBz*PhQ7W)H??^5TSRXFIGf;#;TMVLxr>?EA)=!cgyGLM|XeVPgd)z9oMnt`f1g-71Hy_^>7@m@I z2_N0veVsTHoqYZk18j=W2B(9aeIXWA&vwxmE2n}fui(v!n6kn!cd#A)9%P7UYm50L zT3?k6CqX}K={`klTG+~%Cr%N2t=_IJ8&JcEaO)c;^PS-*sb!A0HcgV3ClP6mtnjhrp=tFnYeaJPg1RH+^uJEt*R<9A*<1K#Qum*E3oFf6#Ukn9 z*E%*-cgVq^wRAoo7@x(Kh3s?qOiyE$4T|E3x5H#moM&bDgGg58L*VSvo3b+(>1q{% zUS+ArO|f*O&SW<2xLIK$J^;!(KXnhflfEBycl6PJrq!oKQEAI{B`N8HP2u`>*rhn^ zqG!vLh}XVGS#_3|8X3~+34JNwsaL}*rXQ79G>zT8BZFRUE4`nAeqp=qP$W8@8bOM) zpKEr3D~Yvo8+xL4e;j69)^AFMd->MYr7h>2eD0}M3_Dfk{jXH}QR2*UK*WB1526GF zx|Kn|{fU5k^ZFYK+ezgu4NH1rVd;R0N?6eDCloeu>i}>duzuUMs2MtN2*KnQE3B`s zKB9G@bvrP`zQhrs=9lvz$|C=0rDW8?Gg``=Pk0!^Pt4as*yo})=J5*k(E47yoOX== z{=6y~xpYC@Jr7OR#^R~Ln~Qk*UY|pi;O=9Z;t_2l>G!NgJ1<*k8_~7z3WbvRvfWE5 zFdsdQjjJDh3w6_v4xm?Y3oOChTEPlOIczr+A`#{XDvq>dL>t_*cUbX9N6uQfHV}<= z>Vg&;E(-O<^r+;KipFrGYdX40*Vf=P-yd!Z*ZvoVLe$;Gjkuol9r_u=*ZUhSmR&^I zNFWk4<2DNon|a$;N3E-LTcKz6bVd)1i?O?y1LubpAAxZ%)2lbbEAHt8Ac5TxxLp@% z^OFKg7&jKU1@M`@UOTYt-K`f&lxg3ya`*C@Q3R{~FTwgPU#0N7SThsde@gK0jT#@$ z&F-)U%fF&~P|LsYCFY6ffivQlg#OIU*1!EhC4Edg))u;Q-mtq%lcSFaJQ2399VPtj zsO7wz)|$_BMATq1+fR|}#WB>5^o%bOim$T^PJtNmxC%FDv7U zcoEMYJ3yxp%Vmc^|2^)nF_Xn;m(2hGTiT(tmfx%f$+w)w`a0BSj3Iz0u}b!cYWh3G z#8QvAhiE{2cI`;$uOyb&b)QFTeqt-wwmPkge?UU2G2BkR9y1Wd;V`_WJLCV2#_|_$ z0SE9l@OS=APLab7d9~8_$IQQ#bNHR5DrC@kv;BqL+u(4xN6OxUlt+5i#C-RZ8L9sc z&!S=X`huO(J5s*SC3*>w;qE+AS_K{TwG#!b6Q8~YN6qeqNb9VxHL3u*1iC%(Bn$&poFzDdN?i~9Tf*AWPyx=HnFF%@0#w<_z)y6h--(o&K=yd z-tAch)b;^uq>jM(hJV)!P+EpoA!E0zJEitqv3tnM%&c&_J;CfyOD!Cn=I(;*=#V^kFc6$pL@En0_y0Mv0&freo4h6HWP}lraC}#%=Z~L ziQ}YLJ)$S|#OhnTeXm2j3r7aWBx`07D?iB}YIP8K|2s za(Is;c^Hk07>>E!!8{$!Ti6i96L&}gt@+;|X&%R@jX&3%FVO29aT_;Fg%%8SA!Dq+ zRRn(Tb42f^KyfcLq{8rKB1^t|V+o%VZ=*6B;{qG*2g7)KglfrS^Rqkgim$aoj* zGoP?LKVe9yZU;aM4FD0%o4!~f3>jo4#EK?Wt^FsMBA3**!9W8v%Pv##Az;{{3 zfhXWY4eaX6Y{{pidAF7%_J>KPn_-XG6ihr!$6Oha*qTxTs>cxlu^Lz=4>XM9mKT;uVc(dQ;jb6bzkhcPY1Ny9cvH`8Q6@9%jylwq1FC(eU7LTSp1 zcav%|l4o@5H^`F|;f03p`9041c8j-vMAFv*c+Id$4tz}C@RAH~8KtB_YxjS)0Z11; zfXXBbDe4OKF`b)c;dE1}eaHG^C(YLM2)==LfpV-98w$N4JFIW!+~}WOypmQ zQpp=*(ORziNWWvbpFV~upXk}5)sw{y*jV3jFs_JCDpydK^PQTIsFYh~fJ{^t;iHZd zD9+*oc^drU#T#HV*rzXV$D^7FUCdcy>JF391-2nd(okQ~^;@B?p-r|qRMS^Xu$_2$ zS&QnTq^QfUjyjxmrUa3@IaO+g7-5z|CHM030(SR~vv>WZFPGKF%*3u#@Rg`CLyW4U zJzdeiDMxh@p-~v*U9r)e52KV}_#6)^@Rl7fStA>CK5m#^1+G=r;`JpWxRRQ1=09P| zU~blhxp^;?TTtgsIk4Q$%k`*MKZ}KmbnT*GmbLV+$#*n(wLQ@YW8BRXqu2Q%>5YlA zN*Cj`3zrx)nz$Bl4<8b!+C|P~d4zn@qMd8q8@njL3b$OVn~;@1?Jj-R(e&%h<0CDMIpbZb)KL-pw_{LTy3HX~tYDCm zK)cDQ>Q~{^+!mK8Wd!vLv>eCnA;_UDV-ekwpo?jj`n^q@sHf_Fq-yywno-UY#AvHO z>kh2hBGBtk?GM8!1)L`#(YDCNux@ZHd~=TW=4Mzdtl(s_w>rlFy&SrAM7ZRu?a3JY z-9})2{tIA?!bvosmb7bd)?&j#<>lw>gq+_38)ajPhJ;T#kmfE00(L*!qfvwhJg5Gv zW8j>68)w9f6p)D6fr$Tm)BTFyvtrcyo+ZP*8$cPx{WFnB3ciXSa!CCL@HGQ+z7Z4jbyN`^y zJpfmFp`sjw^t8D-BCsgKyP%35Bt$Z~QhjDErW!m+TIYduq-AWR!{r(m^s=O>c0;k0 zWHsXYQk!_1bW~LwSY9}kbQz?9YR6iy)&c(UMV&%vO)4Uw15d=hF2JF;j#~Uz0U?77 zT0jt| zBe`pWBTPl;q;|&VG$U)_MR$ZoD2>!JRpyjRmlof5`&`ns_3}HRlNOUZ{Qepjh z<=`c>azI86ygRIClpKX2o-VYLblXu7&0m&uzbcal?!JI;h^cNnQgEk;yfu5d1*5*A z%HO6IgH_z@Q-qU@NgOd4VPE1uKj!hcZ~hYz8K$)Hd56Q;Wxroyq(Y)-kpjK}^t&-B z_0&I+BK@%v0Y6>qBnidIKvx()|CBCtg(Yo+;P!<<*WC^ZVAos6z@~^{<{#dV(I!}} zzhxol4g+^l)AFo|Uly*KZue9tp&YX^_io&1}16M zPvG@eXH%#W=PpQ?G0$5|_xOl3b?U+qBEQ(r_p#;dW&I=a=JxascPuyVVr8AuWOH>c zJ9O_NJc}PZ`m6&M65N7rUHX;(V%-RTYp4~}Db(vFeF9O4*0hjb)ZBsisv3UArbMP=rcvOkJ)vyTT@kFIQjHzV7=*Lmqie=H?3G3jGoXE!Vvz?c{VoPP+jHQ za0LexSG94|6iCGBLaW%5?c$e3+H8*EWc_Sz5t`e7-Xt#;LqY@)i;~(uOPs7!U^bmP z*T095dhQV5e>NktmYW9|2S7KI`E$y#Vu+F^<1CS4pd$b)r! znrgP;tMu~xzLUf9?2T3fUK4%xy_c}}k!*LeBzN`m^l)>BmVJR8noCGvGD7Bm1H|cj ze>g1Dn|}A6*qC{CA$L`$Qz^YN%$|YWr_A|~F+))k0kpF0DIS)(*TtHrQm#Lj*&jOt zVT4J@cGG%2wVu$eq1n959$v&-(8&LJd1Y&bRran{Js(R{;B9zPd1m@ zX*AbwcEbA57RQ+a5On`3w^uu z-W$*YP;P7BzQOLK*j2tZSFaQ_*>xik=?;gCu%n$`etUx$Tp(J))#{MvdRCof-nM4E z^mfr^?<-O(&c<6kJ5)fNvl0I}B1e6v#%G&07C5Z|K#uNWC2xPns+rPG(^2H|{HIon zSRme@HE5CZ^7o~Kt4$66PeUvtA1fB`em4sg|2wmQB6_TvpG<&P>AlNnUUZ1?Qq3#D zu~XTdi>`86mIvo~LM~cwvhc63{1GC@5M!_Dkr(Jub9TC<94FJGbiaJ>9SF(n+Then7mha9UYMq?UwH@WKa7D!rpzrev1y5y=tD#! z*+cN_K)_DFVHV&H6rMv;oeJVQIyk4B33ysw(tDtzIt5^nb*DO7J(tB2oF42KC9CHZgGm;Dy302hTp z$%_WKw33a0Cu&f?y8*0m6@}lv-L}Fso&CnzJ4-IMck2I9u)$b#I>K^vmN+`kZnjtV zFixI)cmCL+OAZ{yBQqbD$xkrTSEcG>fYv)aw!0&l;FO5Gi<2cphju+SK9EHmxZ||A~%1)kY_&G*!KIlJL_aFjxXvEV3j+P0k@e}&;<{f^e z2m6qRg4JIT)c1w0pQf1wpol?>F|eb)kT99V+lCpQDc4`vbC~+YQ(1SH%F#44`z$$t zA_~XP&^WLjDqPb#FQRtPgtt>1fROV)Kz5ASbD|u8bRP(XEG=mHq5}xC9}J^@JNzmT zZ~yH0HuyZ0NJmhp_$n^AZA4}nMOwv&th2Z)tJJISPi zQMXRHKP913N`;J-7VG(MMv*#u1{DG34A5nL+B|w+%L}u(b&4z0qpm{1DL}w0yu!Na zw86A%=^c*f2nt<#n!}bRp*ovF8>`GvXJ;~SIzT+X0>@F3|KD(So;g~>cO_-)$=`wn zjvSzp@~u5iP`kDN4GD=E>h7VJ*&W*!lzOiT_}VN0+b~FEHymX3(FNOMoKEx;PGAw) z^Lj>L9)%YNBI$;y?BaZCr0Xvd(`HAfJttcNA=*$u2 ziDi)CJ;M^<;(wsx$^A2hU+kMzgz(zlnWnHod!l(g--8fEWvZr@cO97qXw{TCS3P%b zKs?_o0>rC5n$XufVtjo>pq?Z3hZcc!=Dz7%gfBRv?!d!+ef7WfDaI^>$+QjP+k=^gutLE=c2gQc>_dhEwN7 zC*nf5J<1f_-_!u0O!XidpppHDB7oj;v?6(ce)xTFL)eJMBmel3Rb0bvE1Z*~z`2u| zM`(CRuYSip=vx zbQS^O4UYyy0nhrdxj=A{z zw!3ykEdzLf{pr+kT^gTBm01zU>_qhUBDvf48y%wu^EJvBciz`!ujxwunU9Wi{m8E9 zgd3JsrsfRy*WmSxp&xPef4GsLabPfTR9yq?>-4o?b-PMF{>X&6Z^e-qcLUYe!-{sm>(mgSXvHE1OvIL+CQg$keauIM1*2}X0tfR;p9R3J z*fzja(q4t$^p%|f>rG~roO9$eKGRfEq_^7oF#6;i%*lKbGgDuL%y=-0JCA~VYk8c} za=F)Y_h$?RPS3KLEMOtp+y1wPO;BPV?kL}vnj>HIS;Vv&CL<>7)AY8)lt+GJq*%Ee z?tgS9*c*EfV^Zl0)u!e|ce9PeZApZ1?t~uxPcswie;0jVKh=gbK^0M zCt78&kK|wqv>1h1zTzA)Z18+cdc%3&filshAG2Cn}z95SXz)JwDRre_wRH-)aD+8Z&4VAz(m|3h;9F=VmULlbg>PYJ3w zO*vNLva}IoDv;Y-8Rd4U@XoM!0tJ#-9vr#Af>RNp^kX|tf4Y);0~xHzc`;evW$+X> zdw8JIzVRyvPFq6w4S<)G$Sq!q6i_7AB9C2UB#}ktf0Af_s?_1hDZXh1Y?w;tIKneu zL>9P(wqhPjKtFi-YvMqLu$bVReL;>JpeE`JToH@?;MtJPK%wW1SDRn=u0pWc!YloA za!~VY+IQ_o#%}$`K$`$u?AYH?&*@M;t7N_bC)yQ|L;4WOkFG<&lQuItYRZ&33}n*xpF zJK5aH^qvB<)w*^IBBF!|0J#Nj49hy z{cBSLUYK8`&RSaJyGW*!qWvw`+zXHj#q`02tn&}{X9X$YvM)<_ zee^|7KiPw7cu1)f?o1g&pmv1YR)CGX7p4UW`kRQ!TK7xk+V|7u!E3Cq?b%Jak3h~? z%drWiU)T?RoURQ=bk>kuw+_=Adim$^&_U6(7d}hCHvc3!a&Wpk8s*!4dEX{F%NU{y zX*QIvX_2i}>%&E-CkPWwAyu#6pwY`d(RWlcH`)EJ=Vs3TOwVA8uo ze#b7iJ)f7}#VijF%_SejTA9;{Cw3LZI-c7OGev6Wr2v}G8$I2a@{q8OH;h{}jh=XD zgt;{N#Y^;&S%JI#W)yzk*4!Pu`8=1`J}k;-mIPpQ8g5ij_{7KW4%9w-AA{D?sOMr; zX9{jEYg@Sp?YzUgPpqC>3YUY)7A<9ZyDKRRlHfvl+;%HSo5g-3k8Dh*NY4gH1OQZ- z!BG?^OF*hWI#He7XY8CMt4E@H!!ZB6raVrERpMm9cz370`jOaGt_F1E#ZBs6MCt|6 zvxVnzHgT9^fh79suLizg%4Z(?CwY=XAqHtBb&gq#p}cp2?f~fH-vzw+J^61oYOc!5 z)jR>Zun5I<;29DxFH}M|>0j!x275UOm_~#kIg?TDy&>bgxYE+8eQA~> zHCFwX3n#$Zp;@?ddm5?C;VOBfH|t7=fnR6c{ZP{m0L84q9q{h{BIzCD+W&SF+RQn} z#h{2%PONR3vlL!t7mW-WYaqUq#M~DX+uat2<&yKZjsl|BTag$~{>+0h+}I!wpgL&P zv}zJ|7F12>Pi_z;ef)ApDgOOJPl6x_f4o+N2iU7kLCk-lk#DjUL)qqR`FsD&Hs_}K z&Njc@2C&U7_FRs=b(55$fw4s-`oBSJL?kZvDooh4v=K-|!Z88a73(U*MdqIrV6i#* zQnUu49Dq~h`7JsmfZO_Me1H(%G0j;6?vsd|p}93@ zDh_x@013oP)Om+4ui*u7vBj7f;nt4(y~gU8UV(6v$ad@^eVi<=nydH(hcFD_L1Y=g7GdAliSW%^d7 zU4q9J%p^r9^2|t_4E>Ho=)+$iT!xqzUi1hrH4cL4bUmSO-jtUU~V9L&q_vxixn zj8h!N{hjC%QVu1$sDScUQ2>^QbGj5MV6)Ics9XGK_)G0Bm4fO~Mma_ceDs6e65dvi zlij>HD<4nBQhOTeFyw^@Zk^uM0Xi3z(3WsL_~-#pz7crW`!Jfg3?*ZCmY{2V_|)SC zUnuwP>C0#KF&QTpO(x)JE=s%mF6eC?g95wD#hzXCnC^S$f+qAzZ7FmPUCL<@b}LV- zybYZA_%G_Pm%#L}AZhcyt`>1u{&H%I*sHV3CgBiyf`V{@zW~R%3Vs3{ul7}E`n^zE zATtzp;k`TX0r{f-8Ft6pXQ1ynRBV04kL`jwS$?;z1(hhGJ7RWrq`D!`!>S*>ve&<> zM11Md^tb9LSm*kbmQw3QkjJ{|m3&RyYx4NCA9AMcCJ)knZV%DQc3?@2l9bOE>D#u^ z9r^K(OD!OLCvQltH39il-QnkoEeY$lAF*YU#X9f)Lj+)g4ZE(jCq8{Ki5!+fGtD}MO$Tj!z`ExL?V^7q3t4lT+jaY}G}_(Y@4Zz|fA)CX$)b)p&DOshGgAL|^z zDWhA6ck5IYI$dN?0rx2hfT99@znTF9CD9b2tv%o><>o$OSJV6g61s4EHf>{fV`MVj z7r#D>ypU2XoFGlk;jwM-tjdvnya4s3BnoC;&(i>zzagjaR{uHOTp_gv3>&n>%AH$N zjhO1#eaz~iad#@G-7v1E6bQ~BPW`iMtqcrvHRd1p2-t z&wBEK$*RBE<-N%A9Xq!~3#DD#Mq@ExlUzx2B&jx5&=>>h-6tFA(lqUT64Ydp6dgQ^&u-o05#3y;}UJ32PLx{v}^OtJ$0mz z);mJRa%!XAcyN2?W7EzuWoLK%yF3Rm^i%H7N$5z^!u~SiA7vaYx(5BoI=0W zqDMN~8)%(HWMK_ExG~3{ICnR4Iya2h{0OvP?~M?X@QZohLaQMaH@}%-$c?_=4!8(^ zktG3yICmlxN`}~;BoW6s6~`Gua=H%O;tUV&o#VhBqCEIF*Th(0p!Kg%ljhO*Lx&Wy zhy<|#!I$n-)8swb^Cx`u4GnI>Pn!QW4g3q9_kSBZuDrgho$3ESsGrQyQ_}r|b6lQ5 zYh26kUInfF5(&~%e<$%qJcYm=qWlQ(F5ma1s0-h2E(Dy!zhs8P>O%6&_;CM<5T6PT zl@Q11`2R4~5jTit7u;7mhkYT(wvLAP=%NY>TuF%ce^-QD^@+;(lL)!;ABvEc1ms_( z;7UEc(Z38n%NF(~<+$P|ENR!2D&jEYhH8_iMwuone(gK==k{NYzwJ#8kD4+(c?=$E z>ccUt_Vszer2zUWXmG>$(+P=-;oK^QfOqwMO= zziHxphqoR;4w{;M`!sWy&l@&n*&NhLwgF5FfC6`s{5)pz7uI=?(P!1m@{d6Cd$s#! zq~ze*Vd72nIJsQ;w!U!LpD`)xnUs)1a9bB}ZT{s{NBB~r{Z^<3uS8sTxwuNpTkaSREJ=^z;r~UPCJIaS!hsQ2)hHN4wyJ0W7RzZ2kIL4#JfYBqOl# z%9g{gCpeB(h66aL94r0VcV+g9w;}FdFks3_Z|Cth(~ObX6_JIy(J4-9zbJ`8j;4Ud zXg3?7I^m?6-@ElC1%C@30CX8nY`l%tbO3YD#}YwxU+5&%)3`keM^os-aFQP}FAhKq zfq`iHp0uSpoR~Z(yC5sHKai#`m;&#K`=WQo_K!<+3e~Fs=AuJ5N%+j&y;cJ;Gfcpb z#)BfgI~lkb>T$vqIe|#dgdb}@MjYl|8>{|<`9R`9AMF<)mp82X5e1gy6NA-08^yrL z8ViqiS(Pv^9fvcaqz9>>xb`_gH)>d4?e*wA2)V7<$g8(mE|ZduxKOFritp-(Gs++z%QK>}{9d*7^W4v2u4>Cq4% zpwxRKTONE8m0WQz3wd*I4WhGIs9N%D3=^w~s8K<}L0z<)L!l|n@F{6^Fg{PxRAd_u zk4icZ+5LxjyHk!>@NXF;-G7?z9Sfg%QWQQ?l=yqqa?Aqpe~!hy7fhF z1~6~Be*{=92lTwI?p`S59taUfI?eT+%5xIKFM=A=xtCTSK!n>hem%3Xnw}a_@rDud zb6iIxL}f0fH1Dc(1(c72bYi@f2j`%CUo0YZn3WGu+uq;ga!Jz!H8|O{4bUPBjM&nA zZW%ftVV9L76VQ^ed3K{w;_G{NCy5GR^$^t*iwFM*C^Jsvo_w=Pd2dmk0Mv=T{c2F; z>+OS}#oN%8b2jfi^&Gn;#={szF9U=<;1o;8Z_8OPLqDI*>q0wb^We{3;{4c zKA&R62`|7$C@xJbHVvw_Z>{N{kKGM?R%$Z(}V(cg})Oe$?uQY(1(Ul-ewy&A58zWmy0wwJC zq>ceqm!Ul7yd|>m(2@%2z{^IO6;}YR;Fd(A!BDCohjTLqSF^8^$CkJ#lhLSb7I-3zd0a;sE*pO;Y zZv-M;2zJS%?is%>@OorOv=DpBCcF5DF^(AWe>=uet7$mByEVoLK^j7tETluy)LmFR zjf@KfWq7bU)zTGzo|66VX&c3clc`lK=&YT)Z)@%0cC2k49hLz(W|#gXwgzMoyYA-N zGox02bL$yF%(`om&)lV2`=wuqwSnndz+65GVdK!Ba=PK=o**ayuhd&T-TfpL^IvgD zZuCr8%=!=1?r+uqrgj4n@?WQQok8%q zSKziAp!eUQxYyEeM3_$Jtrgod3}b-nzfnnHt*zUa1>ACOcjd1E7d0KU<<&~)$^aSK z$nZpXNcZjqWQL{V@AS6MJw3Ek@8kT8$Y13Cow;4h!t3dJ{KyOlq{4Xqk-5#b>&|FZ z>SX;t38JpF-Rl+K?qPTxa+LjSWaBeus4Dfdfrti-Z`q>R-#&J!2PE;e!(0JZ{9jL3 zRQR?8SVa9A%^?e=hvS%yA0M8q>hS1vBFyQRe5jh@o?d`Td%u)pgDHgZfDPdEkFzGF zCXe@n6S!_PO%$p$LG`2mN@k_SvT`r4Q2RM8B;Q!}VgE9K{{4HoRQ@%BrT80bakoJ` zVk|%^1tvI}1NU%W;PH~TqNkT%AL;-a4NrQfH7j(MBi~&QIG^(>e2R%W;W0rpOty^5 zCW%i0%Hf+&88SRK@o@>OwKHVfA(l&pj^x+OL;@z{A#LZ>{C!kT4MIbiJ#8WKqzw^J zD#jqv>O=AmERVfJk+k>mv9XRiE|_h@C_1EEE3ieLW@547C$@>!@uOz8YiG~9vpf?| z4%iBItBtM$8pMl*cj>TLDaC{QB^sNbeG;-RSsasLiSwz{bU_1XK|pW_1z?q)F(lv* z(?rw5{2?G3^3X>=8RSBukD7vhnzX;0beL;yLUw@e>KAKKeBpv|x?T-9`^_gNDQM)y zZ;Yo~DrCc^A_q7Zsb>b^cCP5Kvxho60F#3K$DGL#KbKYe9Hz`XZx7x#EIaSxerU1G zxR2gAd79l>ls5wboBJ>UtFK^EU}z&gzhZRPCtK+5ILw?8J#)JRP z-bR6iigjnIyxjvi7mwLwOf7gwH`eXVt+>jyT-ZUm>H8SE#$e!I`%5Xc@c6uu(o&qk zM%-=Eju-(zTl{#sJKt3nwU+g-#=r?bvYWc4@&~d7vIdlpoz7MvA^g9QEz)tVwbP=> zNYf*J=%^s5j=Gjk^#boX)1J>;%plj=xwW|s7J9+?w8iOjBCfhn!2Qt+1u*la3nCnzvKeg z*BT%fF@8SX579g>MSTM>n)q+)vAgJ24Sf|}lyiTqamaGfhTg+wG+1Ah=_PL$7dg>X zTPb0G&#F2p5q#0y+`SQ5$=ls;aOc}e8aTHRNfX~`Lp*vn;}$0{H?j0| zcxQb#=7q@Qg~kLs(HN&-Mr!OCMlF!}#cT*X8A55?8% zbX3e%N{nXZ8R^+jBHbskCfKLvv)L{&< z`aVXHj~Ht9|AV-y)*jpjVh2g%b< zW)WPQZGaB}j_>LAwTw`FhER+4iK*z!F zFZ~}UWBq(u-`@mGf28mGm0LOT4$qqW5D_@p|@~qer9}SCz;b@&xvz2BBF#MhSVt6)}psel$5z zqKsrzcBbh_=XKl83|CH7-fm6JzXVcmC+8A?bQ65-Z{|1_ZQjgr6gg500rcyi2RlFH zdq08BVV_L$8=HHVYR_JaNz3UK6e?sC=R)`5*ksntDb{%DqX^xrJ?J>bUk`Q+^PEBG zcr^fBng2|+7F*Du0&pk?^yb#8U8UFkjHR=KusZoK4OX#HEC*3{n|T;ZFG^7yG5-a% z_D`mW1`@H!n~XFGahzh8S-+HKZByRkB*qg13=5H&Sxbl6JV~8*HEm(`8g0|8!S{Vg zszHvJJXe~2+-d`qeK!)k=c`NSDLY~)^XSLHa1G`*=je+97pyh|qJY=?cbw!Ew$w)udBN!(4HmEZyNQ^ZW%Q(W(a#MB2a!tc3-{um-@qZz18=4^ z-bk*9r`*WEslUY+T}ljK2yfBLtz8(Abuvx266D?{6v-QXo^%M{~A(jd}SQCC;dL? zGIn(m7&aO3%K?im*l#Zn#DM^9AoxhUi&=s=AOrXl{+|Xz|H4a;%t zB79d*p+lA6Ft@W{G#l3izDMJNJE#kU28|QjGf%v=`2AA2w7Uf7YPN*f=|T7sq`z zH%B=iDKOeTe_i9o@LA=_8s)WyTVA~Lk+(87u2w5yxj&2~awM{Y^c=J36T$bdPKuQ$D=gd~|K!t7ZkG_wP~ z%c$#J=UC`-e7-)dt*fTBLHm3*-_Dyh%AFVX@!2g?kHaqZ1#C*ZD0=-4<2CyprI7fN zx;c3*G|yG?{hed<&D2Cbkg6hrr&nLSNhn>jo@QEGj+)|n*i*5yu`NDRSi$Q@#1 zNueP^{%qA}6W2Q!`?>NvwS9?PvJmGv2WwnGmIdRS0Y?{IYTZ*4f&n#w2u;$uVI>|kgmI`mg( zMyICnhmewD(oPLMnk3^yFuZ=d?QG%#kz6o|}{5V<(o_uj#pU zKE+FX5y!RzrG&+HYMl`+j{s>^%x9M3)t|@~pCC~xokl|+Jl=WO#aUAbCa9>GIS`lX)`D9Y&eX|hEQ50+Bgc6T3OG6DM# zE4IhJ;&x(#=#%>)_nL&?!n+HG*J?*kDB2CJSVlufvI99gB9fgbwiMi+vRo^P9o=;2 ztvED+EO&e%L248VH($G0l2ajnCV%qMftZijBiF*bKel;CM|gxCc_Z||sCx6M=(1-S z!Zysw$@z0l;bDXTLZO7SWk>wp6U%AJ6Re3)!+0z@!>F%ycJ%!&ljpFQ^BC;@UpCy$mvzHWR5O*LhvrEVhr>G6-@MuZj&l(0X%gdFy`{ZTE zCm-q&Jk=3=OKP>71hhXs&=2aC0IfXzfJeSL7{-?o)t2=cj@S=*Vrhh2r~*Qt8Qh2P z@9lMkg6*&gdmVzMj0p%8h5W{b;KKBuhI?kAEsGHa-HL2~#g}ztoPT6oS1+Ly?a*!Y zsmqwnn4~LT#lezCA4HwkAy}n5lU%CdtnG<}nc)sfja=r2=xP@_h+KWW*qb{u)Qnq& zb#S5Ex4e)1Io*Gr{Q1#GVrw5B_<>fpiayhFArSj!|$hHPbN*A}Rzs7~p zA>r+L=HLn(>9?HMvAcJGK`zG#1h$*+BTBeU4fItSx6HR#=)1coc{i#_7JCHsB{e?j zO1;okR94N{723AK}%+mpaV#7#*7VoIRBk1qTX68okJoc@u>w(C;^95UGo4btI- zGFJAw%_9Pz;hraw0%t%2WOmiRJyXUw)rn&I(iq0^0BpUFZXXo1g!Vq&LEv_8- ztOXM~;H}}q?06`z$YKwZet|j(%lf!P&p5^c?rr)|Qs*N$Lgel%RuhcwTkMS|KE z;oL2DrkZIAf(9rBFqxDHDhmAUE zT&eMdY3Ro(Xw@^b9Q5p$5(M4eqMD{EKAHb?7I`w6sBO?$Pw9aAt)udZ@9hWSG|Pgr zU)y#+BNovE^>2`i|I4&$Q9^kcLJewUMWKqV`pV7+j9AyNcSsE}@VM3&%-xL~AmAL~IITBsj2qxuUe_ouwoHYB0--{S2zTd}1h zs_p~RLcXj-#-d$$%Q{Iyq7H_i1|KKk&l%N`WoRb_u*$qByEN5E*lP4UE!+@cXa&BU z8+4&cw@DVG@ur}p>0MK({;)*e2X!FKdz@Rq-mHE7IDpkgDY@K)mt*CP!A=AcmseQw z%k7+F(CX$2GL}Cofq*tOri*`L3A~{ZGwS#i_Iu$L{nzZ~osZ7Tbk5fz_`V?u@r_sz zjeYPW<@q9-<}FlJ&3QNOF?48vT<_(P@>Ayg@?S(JKk^I7CQc|OR@0BUd- zfyJQc?Cff^kOeOClH$pB=hI5pBDu({vt&F3OdEEY+80ubi`3GQx^eiOh_8$8)?xBi zDDOn?A9l`?mctn`r)WOU^|O=2zdBZI6iHG^CDHkzCsr0E-fTidWLxgeO^tbbN&~DZA@tQq--cBfFnbg0M(V5U zV_y~>O%MF60WU6hRm2QX9`ZQ5QJa_;?93;(U{elhI{mYS9zu>2;BRx(j)FtJ6md(2 zzBng|won&d!2i}&;KlP#uBekPgnpny=L^+Q#<`TcVNOrxI+QiaYUS$2o&f{mAqnqK zrYz9q1*7s3^*g5>xjOC&VUxMWwtnC>Q}6K#G~V+(iq9sFbI0#-8dn-Z%Q@2hX7^u$+=60`l%#)p^rxSc*M&1yoSSA$Ym_LU|jr3V|42usvA3;wR80mu^@xyr(s zhy?SuDz6k-+Ourb-hbr(Nb^K;FE>|!$kLP5iSx9yQ^#xzU`;ceS~jRh&?{55z^ODg znMxD7^4b>#p(4l>@YW3S05PmsPRxPWV8h#F!9tzrlEr5}6GS1#FU}&+%ETd`6x$=( zVv5sVS5XYVR9&o)7w`fg`9lwat(622(uy1+?QnVd_wb)~>-vV=4>i8T> zl!L%v4Ok>HY=h}7B+4`P?Z7!vW0|M)Qh_g|?!AZS*s@`&9}jUgL!sPVcx9cC zr8vex2}EC zPCb86V9~?THbG7gk7KBLX?y2FYkeHD%uu|pU`|rpPhp~Z$o@?5eXy*pc$*FrPSyrN zYD41l;n_5*af8Q;0eRP?!CXcJsIG~$=X%9Ri{#iiG#y)l3vIHb3t11<-x0ifJMfWx z*$+grPu~=#VYE9S%_v*3&*g15+QkoShXJo1=y3>6e{CnnqOZ z>Leql6{{Kcgu=q?I2tQoddGtCCO^EEsj40yoCutc1BuD>fS;n|Mc!-I3Sog4zDiog!lFE?#i$t5^hvhh0f9`cK>F0`Iiz>t>>Eeoikc zgIVl}7CLL+$ULZIJ0Nz z-e;VvOHp#W%6g5m(U{-F-iL!jBgs@uFqc_3p&93#yK?5-gxh-Rso3I3N*2v_bggum z8t|H@bua9U`vJy9H~74fOz>+zs~-9jfG6i~VwH}wXp%e) zEDHVE3TId^`rUQ0ZZjV%@2le4}#qyYlP?LC7*2mt7{wDQ6K+m7N120|Ci%mG#e{Y zT8v$CoLS!GdCZgPcqwcvaN=HZpV6GV;q5|=#*5sm5<9jtP45>=Jb8y_PhrIF`ZJv1 zI)Zbd-K$&0A7r2OOP;y?_IipwOZi=cY{z{*UsR7bmiIkgIW_gfI$@=(2H_{>t2Vsi zKmM`Zb{BiTG4ObqUyr7~UL)K5hhju_l0|OzwaDH~8(deqGZi ztK0v%z2}&7tI#r*e=?QGq-CMa< zQmp!oa{P(u5C5@UZ(_C&eGs%-neF&brOGKRe5s{3HW_gT|Bvaj@(q;x8K8Q6eZ6I$ z<@~9Sle#6Ce|2A+6yMh$Z_#)3rP!NufED3ek%);C<~qMa<^4_r|GZvQuDzmqGMDlT82Zb z)x*y|rf&sa7so&Kfchn2=UP#V{l@(L_Vv6!f6YJtZ8m6?ZLmVTt@FMU(={iop9dOH z7C8}q9<t0zQT1d*0J%skGEhxrX zSxS*?+4;^hX1cod-hc1=-uJudUa4lzbDr~@bAJ1Beov^ruKL;n^rNs5R|iNN4S;&49r9!Y8N zkCc?Cw6xs8#s1dzNEgz8I@qIV491!drY0pS4u;x(0PSLj^>PCrb&bG365^m4jsTy) z1!>8}k4K~>cYu~XuC5s50i=x<8q7mY3N9lGmj=z-wKR-$wE1Al;4=p8garSoBW<0q z&@IXiIIJ^hQImj4ih`ku&os0~S>w=4hJeVRVxByO>% z1XwL0kjaP!4Q3vG(~bR540W9jkLPOe(qQ-#vSeYb)zlT#RX|gLI~1;);Ju5 z1;iDTnow8_X>FtiSAa)f^?(`-f7|3G-43Ab+#MGC%EDw8=K}TCK%(s($WNA#mRr2y zY)$S=gb9G0gEc_P#rwYj)8a0Y_Y#N2g7KC>ap`4pC+8!|NDQ>q2<>Z;afY18`GQxaWRDS+vd(mjl)!641;cKzmwaJjjShY;be;CL=Wt>)~QY zWk(=3I-QBUypaTZFfUvf8vM0h;8H>R#aw|BB_yf8SAGXKMb&-G}z+02QpwLGk zY*_399_rxk4pBZ}Z))Jz*3Lyj6b)z}1vowuCu$3x2vf7Oc87!k)DGB{nhl^PH+O(M zBA!@`2w+EQ5`e|4dAQk#IHPTGSb&(aqOLCXB;@>`Vv~d{2}4QP1Y6|TF!3e01ScWf zQd}ap{XDotTE`#Z(oaXr64E3NMB05h(oQegd|6VDrJFASCpG>&yDv+UKYz6S5WTtn zOQa?VOluo5=OwMt)%rIP@L#e{2^o^?Cv7)*ZL&&T!e za_~jC|FJaqLus~%4_{>2;@2er0g5(JXf4Hmsh+t_2>h>Jn4gEqyK{NSf2Dj zwt{{=fJiP+0+IMj%)9@I%KJMBLjJnUB)Y^8_->Y`NGH3_mK#7wg#GKWEV8+`+yL^+ z8imr!RECheKffM@NH0?@>z6gtWtJ(0`~r6R@^)NF`~sFa2?ER1RVDEY*y3aeEKjlU z*M#N&Tw9L>fo1BBg8c&4HW>oTlm>nQ^O^*KW$JMvi?8KuEs!CwOlja3FrY~gSf)ND za<1erKm*Ivi3IxvOk@%bEK{!$IkWZ`pn+xTCL+h!e*p+AQ@;@G&$fx3v7V$vKPYV) zhjc?9UHTpC4l-sJ69J*?AcG&AT=?xf(hj}!D+X)p^d&?4>oh{>Zfgt%>$QiRdJiS@ z+M#hEvv)Bq*b50A13`e>+^x|r#B&Q!7aTN&#U66d_kWqTxHLWS&ov=Q`Q9WYh{Qx? zq`s5vy&N)5B>!yZ`#&f|Oxg$es!M*M#QSfL43MaI*`v^(tqc8wLSG@zALZ#v{3>kd z&yH~XgF`=#L%#|i`md%hS{DVV3 zj6?8WL?4p+v*S$v;Lwlb(67RW{_M=te{krBaY*u4;X{A6d+4uo*pWEul5FMgona*> zdi^|SCS`s=6#Vx`xc?_UQYldxxcHJE5>YG#0VrfcLb0U>SIMbHKM@pufSLY}psDVDHW~)R3_y4Ql=HwkArFx1 zuRu|UB~@WukvKFc;zEoq*nqktHdq&={%>svS1j7aomi7a3c)9(0&d27xI+Ugf2kn^ zg&0&vM!G1R)UbpgB_u`VNWQqG z1WCe}p9cxZ6*_);B>3;5RR1jsz(mEx;Sz8uDLFU{CJrZ;hgkw}X;Dc@acLPDIcaGL zaZr`&JKXr^1O5+K*#B?jsU{L`lApn5&67c{&eyb!eOaAZ!%GytX8-KNiAS)^-^?Q~;cI7XT5fT!j zvdcm)kR;VVN&(4H@%_95pySC%kl?ul%F?3JB)R%?Gk(&{{{iwJ0vRSD3WrGng-%jV zLJB4&303O?zv90*=l_9Rlz%uszy!$7`yK0-68%n{L!cp$lOum_uJ8lF|DOWzf2}$Q zlM$7XmXei}k(82=hRH}skZk=WVn7ZQ&n8-eB*w8UWWql#q2GrlzAyTLRFEZ${@;wi zMai%P2Y|vSw;bRuLn;gU)9V6%wW1G`0;c}gnm(vt|E(Vw5(^|n;iqQzGXHphevshh zL}kgg8;Rg$zJy$sQapGi0!f7a)PN^fSzki&s)qV-m>~&kmz0PH#onPK1WP>r)})H$ zwxD=D0u+!<|W2h}HV@WBkN8}C~aB5><7f5FZ zGZMS_ThbZ)Tf1~w(Af`oA%0mp`%cBwWdXClvkpSerCDaYQ9}Ihtb~xg9?K4Be`hU( z{6eQ?2eiMl8bbDkEjyt7Kmx?Cs)v9yw{QH}%Rvv3Ao~OUjtW1BM=n#Kkx=XJ05ll< z*YIFTi2p#ffnS9^egm{+>cWyBhbR7)l(tNLSQ6rYBjfrTN?WE*EQud^%hE5)9+#;X zOX5e~#PrJoZMnL!e&lUUzbw#}s~d}aX6A3Hh%8q(7TLc1TLNvly0OTn%HI)a%hru0 z@i+E}EL%4gIS+gZrK#)h)g!-jXGv<$|F^6sDuSn`ohh@+79bQy2H-e&EZRN?j9YCn+@m&<>D^6 zU1AUP2egwC75}GcoiI6&GEYJ)GVaOAeFyhIi50Tt@D=*~boEsv$VUa2vJqXCZ<0KJ zIY}Tvei6eLNkD?l5=lUoh+l#5=i?yXc}tR%@!tW0UsEEG<=>%1@(K5^U`WbM`H7?e zd74Ye@EZssUF?XjPyjsA7Gv$^hPDN4ZwWj9`pg1goH%b2&=(R1#4o1c3$#U~uPV^X z4IqI+yr=@v!Ea|9kaqSEF9vfU?bXtEERdSqt#S59aOwi;_1%(U5{riTf1s`{A;Rw+ zZXmbekQnfW2+yx)`aP)r$P|7ny{-=xg91lxNN<0D$&x~dUs(R{cz_$$180jQ-37v? zAYV<5@4tJb6e&bbekA!=4|#Emd;cFi60Qs;x8wx{e>_c6Z)M<@{67{19YG{^gNdW< z{s^WjpW~`6TjCwz8vdW`bHB7v59KZS0%m{DeOBZLB4*0 z3Ft=3KwL@$-^chxI{xEL{SDE+eNBY8s06t#3kmchBx?KrgX>=(8x$d7AsOqiD3lwZ zc;bf!iz@Q ze}LmZH)ki?Eo4;x^Vzxt`Ms?q;)0PgR>+V1k9j&-4+BP(wKve($pexUMWE&U2)GtU zA&=#Jgrt*yF&?D9_Emy9jsKqh^zV*OE|&VE`q}?6z6AVt^Z#FuFHU;izc+uPmbnPX z9|785uay2sR_gzE?g}t|33HVeC;AE?e)`2#@c*B={unryShO(lpNwTnfg(#J=i(P0 z2Ff^qI=x?he&-kdA8^s%6(oR>gMehF_mgu`@~fl2Y#_<^_wR9uf6b+TKfWYM-Tsrt zm-yW~B4GdP@yX!*_vZhXb5Y_1a9AwxDys5{@A<0i4Vh;{!66+icpvRoeTY}>(AFS! z1Y%Ldn@Kl-coT3CLM=+y#pNg{xG46j>@hlGIhefqyp^i=g8Ct5suP<}G8b&)33<6T z^P&^aX&!ymCIkb6_(gL*8vNRe5B*Q%D!mtNPBpJNX?_N=V#SRWtcAn(`0yKA(loUg z4;D5Y66WG6!_THnd{X>?)jJ+12OpY!lH{iF*4w9Ppl;WYPp8-0GPphJN$txlv*IV8 zlqjee{3&R-Ua7o7(q%Bo2<};I*g!?MBb0)Y4!Zt{%O8LA&J{r#E@A^W-^#@gR2(!6 z2J4R)1%dm)6|U5t#Scn%PEsJ=Q=Yp{9GQYLFnTNW2+-vSgOY~)t9_dmpXrZ}q$YJ) zPeqrpa`$z_;=>sXaO7$2q2M0(SKm*Zhd%{Ea~)~IyTRi)K3!Y2I720k>ctQK{wpa1 zo2WVVb1%-FZin3Bhi?{6$sQkG#@yd`JCK5=iJmLz?htey0t3 zu0A8^tx6T|dEfI;<;T(wbrI?u=tqwpWzuWM!yYEN-lOlkJ^Ca+d$i=K?-SvJ1y!ZD zHzw`$xK3~`?x+iw>2$CdnjL*AbMoZLm%VM9+I>dr>t6?Rx`0#cM&tqfm zsh=J&>7CCiY88wZ>0@j#%}2$@!|fAx4c*SSWixnnHPHn&GgMhwEtCo~ydZnPaQG47hTT!_>7zS=FC z$`n);0ROB`M7&VfU~H%|*x<#b?+rRVM4Z9%q;Aqemr@731K@lfyglV4-3~;8nCj0RW*3E_O=Ho^@2{DR!e|!jfKl!S$n!Ix_Kd6J1@` zF-6aAdym#VWadrP{dA3nl1>(2c=GdyrV`TJzKw~UcAG^G+M%9aa9*JuIF{u0L}H@z zSkC^bf@ctpKauTT19xsMh#g1OCm;dlXR9=+wkOJ;lmC1^UcvXM&D-bdoP+O9iVU4S zwH**;bS&75)@M>)?Vg8*1_!0OySq^sOx+j}?Ruj0bfdi=3RynS^*OVGp<&CF6M78W z6#;h6N+dO%)JGg#p-qS0bXkLwi-D6eP_p9u5$`~J(W7Ae(X`Dm+!O(u0fC%Oqx}}9 zt*iR!1(NVvI5;?pj;bd-4U^U4JT{OUYMU}Ymfq$fJ^lVbWkR%5F}3XAC6OW7&{vyi zxGn-N^y1={h_$qzKhjcncJ%5vF@!`=Ag*tWaaY<83ud8{Wi%;vNj8*+*yl%pm0q4v zjy$QFToT=?0CDcABS8^4hGB#U0lzR1nkE)c@n# zGGFiuNGES!QFJoiWRf|?uj7)o5JN~JH2ajxw0S19IKDG!aHiD@mq&8kDeqg;(ExfB z+!K!E3VXPoFY|d~XpooiaTQMSt$~2|(q(LDG68qZ>fd86k2g8;_>KA)<^4URn?xMB z!jR~F|Mo09ls;Q6j9a0G-#qM_=~}4pes?K2a-}1``ppib5}k_=4^(pr?1vG|w(n?)|kZ zNxSJK@Vn#;8Z}#IEk;*xXQ8PY2Dd|-XYC-3ufBL>?76yt$W~kxH>%CKSjy>T=HB^* zSTh$ihj$NSzI%T|&w|zAhnsY5uFK4i3z zvcbsscW)IY{yVUB%gV}T6uy3K4fmU$4TbUK+1lD#a!6b}`nuM_G^`j~ z;V>1!MzIM<`;&M}$|%l#1g0KHvx-OU2CJpb3CG*mz> zh`E?Y(102p=yxQBYi>T$9z+2*nA`elGEAWnX3{gM#d10D90Q$}KTF02D^WXrDpMHz zAVRRC4{<7T1*aGHrv(TPGM+g$hJJ{_7P+9;^%OZ9eAJ^Um@4ZFdwEsBW;G>U58&E; z_Ad+6$^z&;>hq1b@-mtEImR^RR7vtnqt zj@&%Ag9!kA$H6RGShG+Urel=^zO6_{ZZ0w096|~<^5*0R)V+mv&xSBy^8x#!20b!S z4Uo?$DYBp6f5TWOnuPBE8ejL2&>i=&y#!Sq$%k@qs9Ljb zovx$vks~d2DT;oV&x8!Lw3vvVD~R3vU4oHc#CV2=$Q5h6zA*1Ji?fy6S3G-uX0u@-U@dtx$`7Cs`OM%;&jul%@*Y7BXCQEsaHlyLU2- z&-L^kE(9bj^iNmdDZvPKx)2rAE5CM zJ4h3+i?~P);gsrW)~cOrd#vV1;_Bn=FA#F$fP$DQw=g?eAHBmcGB0oSWWY)eab_!@ z!y~;3kn%Wi^ZvTN@*&@xcp2;k!r{m40_B`OKrqzpS;=nqv2tO4ytNgm$5F?ECc6Cm z+*+)IYCLjGM^Hf=$n!KxRhi#t5c z-ZZv#&3n2{z79^ja(1Vqe3|zQ*IIy|ZhfcNT*MDpc*K}i_>>TEwIcN^6({$&)$B;$ z=8W|Adeg;t+aTcbh+egCozlf7Xxl$+lEQoJJRY$sBczM1^V6x=@|dT^q0g$5bHpkp zFPF@Y)J8q6bS@u%*=Nf+bKd*K4NK(liL$PEGnbNwvwbp^ducvLrOc)hT<@=CO3c7l z%=U`Oa85r9<`;b%mr^}}c-ws1PDI6s+M zj`f>q>jSD6JkIiHhTP<lH9Q(L;R_>KQHNV8b1FIe7I}>&YF6`W?MoHHMsUsVE z0DIA;_7+T4oa@)C&o2Au_@Zqkr;j+}nGUb64q(X~!{74mV zl%H4s{fUZ+a+O=Xywt%QN8bhdYqbd^xgthZK9FiQx3!!oZPqF12(8@Y8iY{oO*!%5 zyuVUEBtW=@9l1V8;>+mzn=9wX?TQGRKD}ep-e2?@*sGkv594cya3Ed*205jbDI#I!QIxwVc z?DdJ&qT*U{qB^M5yfYn>1$oV(l?F#WBn^$`J>1VwF8$^Y~+GLXGG1)=g9c|fk(T;Wn99p5Y`aV)@QNqjl0;(J~-ZNrHC$S z4Xt$E%2r#_qqWbks$}fkPKIl&&=W6{rO10peA_^42Rc2OpFXgTPF37a`P^-$7Q+oiQL<5gQ3q(zp6Rg z<#~=SC%V2O-=kmohUf9g`<(n*9l=|1AyYL6DqwqCb`2MY+a7!(1?1W7)E(<71ijw}MF$04~)=TA`G!BIM_GpCEZBZPJf2zGT4(&C| zmnAU7uDZQ<&e7Ok1v7MeRF^j~vLX2vZjGAfWDVvPffueSx5{n0Kb_6Ha)&x9&vSI_ zk}2DjD_5$YCxlD4Zhy@+WX;B@LpfT_tfOe|cJ3&{H6dtQMhF3Q6Zt0DE`=K!c9(hP zJJ)U)0%U9EHdH>rKO~Iwx=|;LR3Ccv>QLU^YaPSPat6Hc)?}|v0w(w2X7)1v{Ax37 zxs;sXDB!Gb;k0ZMqW~WA(a-GIo#<Je)45ghhF5Cxfb_ii z6Ksw4r~dR5UC&Q7axcds{GvMUSSdt~@c&-x%^J%RZm-_JlzZix%w!m*Gr|nt^XSqT zEHG2Ce|Td&V~xjAtaq(h5-JsNDl6RcfURi}@XO&(ne=u?%$BwLZ?oY~Jj(&hmekwr z_-#YFz2z>SSMAAt-5UI8t>(yaA0WVroOfv! zjGW%)o`ag@kCMxH4)WmhgoX3%;*&!iY8Yd7eZ0j>*uTr^rLbEyU)*`X#TC%asv2pQ zpXbd>oDO^(EuVgWqn=THYXn`1IZjOLRbHDH@FYmBe4^*vgv8Ae3DSnR=*MWQbyWk14 zCHPC8{mQ{YkgC7R?uM6b_JN9$(%$K_PYTORn;ZKE6U17NH_2y@aZ-5$3)0)6a7UX9 zms?3-`wJ0TqXRgXal|Lj>Ntkuw-_%01yvmkbJ`nN6MYSrH=I5g9~$JhZ|cB}Lye|% z1(2bgrnZmTRG7MC!=nb$wNn4r;v|!3vlJf9Ua1dFEccU`;DRHw>&`U+{E)IyupP3s z!K6)gE?(1Q7T3 zI> zZr%|#zx8ZvVj^*qkTID79R^204gnKeYTwnIoTq2boaq_SaN>IUVtbGPodeOlj!q@3 z$Ao~);$j)b+7h?734B_y=l_LeOg5~oJ-EjK-}&u+zzeG)wNWE{(1Por(s8!P2h$>P047Iwda1v&NYdo7T40PBaeRR+=UcHeahGjxO-kP$bxV7V_o;4^+WftL@g; zNQhIoT9*Bk5))*n9NZ{{e+a`1{kup7Q2&o^3#k{?)Nr)s-#o9{YbA0*TY&pn09a5o z573)(*Ys#^(XwB>)=zcn{{L;b{A+8c70|ZaA?z2_G^d*v1+E`sk^Fah@et-ccO+sr z>Vt7eKzosMO@9}rCjo2@fsTZkiEHf?pxxV4JG|P=?}o8%RHYseiu-GYrw#d`Eh@+Z zxNDz_>1fyrjIq@A6f(AD{)TOJ6Uz0#r?IxDXmFnkj8bI20!$(@o?-i0u<8d2yYThr zCGMX&dp2Rk>dohaRx>Be&mYgXdAO>vu`#MZRzbmqmX@~e`u-dIHDlxB@evUb<{V(K z64ta&G~~gee;aK4^Jjh=Ij@iA`F7PC;F6Mdv@2H%6iV;jjp=)uqPR~@#S?=Oj#I}- z*92*m-`wm6Nw62nVSDIIfWtUzU!hnCUNAs7o+Yrrj`F_67v~^a=7R#9Db4?2!t*!v z4cSYL*;Y&-5J|v5j>YE@sw60kPAbtrgwM6ncb0D{MTiXIW`lmCh-m~Z(5dip^$zqI zlC?o=nlpN`ka03Xj;tCrtz`{f4sFo;-i> zQkOWFMIQ@QS_E)p1Zg`eATJAW)}?L+?piCO`4>D@?i;AsA<*wIXj0l-3$77ex!_ui z=porPzuCfBfSG!%hM%P=Eq5xtHtY64S&Gi+G(>k#F4Mxmy!~rX3Q*|XG)qL{XE z1jb}L@9hy%!?2wZQcO)Cp8)k|O&9B{fi&ijch&mh;Sv}$G5sReD)Ss4L)>aOAMY!G z-RNqn@Uj>lAisk02}*r#^d*sI=XWqtLR?#Gql0-C?g5nyRd?Q6O<$t}6%)Yd1RkWG&hsKggpsIjpa0 zFHX;}V7Sl`nqCOxymxY>1)kzp{hoZ#?poVHPD`D7;IwSK(>SOJ!Q);PikA;BC3PQl zC4u&W%YhRRY}J6PCm=I}T5yk25E$_UBlrG$AuQ^Esqo5P-=hR^M2dad;`ki_56&=Ajy$Dj zi{ZKmDK>Wjfm%N}!wUjbj9XuDU4)b{|Ee=a^gO5}AZnhxyGBQe1;fbPi(B)fx3{3x z-sA3AR*Td{1A76Vn}a$iCcXnpH|(oXLy#aTa%hvKYh9xujn_g!oo<%&6VIg zCl0tZ#C2nLv`!wBf+nWIXaG(2B>v@ZfChRkP2N?MN$XO~$&Y~MZ=JtShgz>#2t;ygej7?4@Ql1-QC zP4#TffJiPcKTZbJ^7WAEHmF-{$;Q*dzD1oWj*Jazb+X`Q_PhJmQmq0Ax+28pM1VH^ zHdGQF-cPp$#Qrvc1+G#9njbG}E5L}euddNjt^jRRDyURV28nKM3T_WIbyd0}fRC0l zR@nLfH@ACQwtC1N%IVpR($u&E`Jy+_vc?H1LrdoS-jW}sjRW$;oYMN>wAH87(h&m2 z*1~J0o@MF&6^WSAo zkOG*Hut&q`H#9eG-A`p|2p1k)icU~a1TY=KwW|T?Z84C=5ERiZ1~I3r*B@I|%#Y_J z>Kz}6(z#OF8*7%}RXNjv>hhhCTDjHLpI{Rv;>hY-YT*g(4-E;SjD``Xj5MIw8WgSF z``q_^MtYrvS4Yfj8|23mEq%w6Kl)@>vn=(yqo$DB8PR=O381-*VWAL}A-DX@KtANL z%e8b@)dns+#AvbKOfl*)yO(6Mg;z2Z>8;QA{}@Lku3XQ@>&->E4?uEb+p9>iTn@kf zZK)eBFk2qEn{@1BN$9ir`mWhD`aH~oD{LdP%MPXxsRc~X7)Xqe5=eHBESaDFKy*!_o+O!Sa=+lX7}J>gxeqaS zL7aELW|lss0d75flK_L$JYiKv;M-cO`8>JNI(XDXYF=QXZ_UkuF#Myru+8r$Xn=ip z>>9<6lHmB$&@+4j$Ltkn;=P|9>ui+3fhe6JEGj1XuBv?Qh}dcM%)wKqnE=}vUh;q` zx_%HEsRw=*1xC4&vg!GmA(=#MMG4{f%wRn&8wNi+5R>DRCigbUtqwR8%}ksVwqA*oG^0 z503%%ifE;P^aBP1HqFgQ#EPoR=+jDfda5hIgyje=yjtl{MhWEWKFL_HsR8l{m+VA!Ea#eHl5gcUKB{Z|5c&&p>I@ODdDW+fV#g)o*{xjfuIK{^E=iO3 zFG{et6go}S?#Yb>i33$pK+7s)zG!q?SAM){?;C-WMdng*A75=`j3VfK-n?~(z=G&P zdH>Z!{uAxnQ!^%|PTL6HYN&0peE;~(!?36_o5lc3mv<2hVv5ZB@Vo4g>KQFD@rSOo z*k=UY;`s#P%6n;yP#43WLTMAJ<#;^-Zs*%PpQt$}l}3V}J6Vn<;6+#DvOD^(dxHrN z2lxUvNK19x%$1^571Mq_Dc%@fRd9vJDDRlz6K>#(W&}AKeeq(x{aV$S%zYa9HZ64G zZABpuAhxg1i&KQ$s?7dE|2cODG40l)h>`)yL zwJ4RGS~y^SI0%ZG^J`1iRsyQL>(eV_C9!_@r#nZTt2g`aYr}6 zZTKqa&k5z+&U?lytWYU@iPnKLC~x;fYVpUq znt3#+YV4|KH6vg);-Kb&s)mKfYGq~b|#Sm;}usCJvow}kUi239{`$&_|2{x?HRcCE5L}1bS1d6HS*G#Lk zYU2C1*@d&a>^D~F11^0JSX6ZSVYOCFEWsf^880K+dllX>Fpz)vQL96$S%E}KyRR* zgIN1)g-3c`J&|`r!w3{VVxVbPUvW(6FDPi+9>|kgFU=+uSzR&VZrbWxG?nq9tle0v zOmvC|UJC_jXMxKtM6^YjwOsa{ZqyBi)rL095uH@cNMYyY-UawYxNU+b3 zxRvnn6A1kHm3Goz-VLq79fZ>0inFiRmr*OK3(=w1nQR;cN_c_OwGLJrbaQy3R+jN& zD}#Mb=B@R+c%3&szWvxL&$rrb_Mr#*faQviklJDPw%{}iGlS2)V6mS$^czkoU6YmI zExd%?cdA}(Gk#;3WEr0O*z17$N83_Xvot(Cjx38=RpOJ{`WZo+Rj)R5tZSxjq_?&$ z5|tle!!z$^2=LHX?}M|1Q4&zVZ@-xgg9PsLo%1|e*CF3xBrig7e)ybO!@$eoxY9s) z6NLry@s9WQ*JQadd6t%HIaZd6h$Ep+XYSf&`uQ86#&wa+Yd_j$`xU3vp7Tt2fhiV|}~ zHd)ib(1r}Gi80Nv#I<~Cc3P{IO@nwy&+QSean+K{pR1z$ps-&g%XLiPbe+9|?fl1~ zdC@J&sKk4$Y*Dqd_8{A8o#O1Lo{=?-6EFqu$+%p<{ljv^Rj;An#KkuuZg`kVbV}fa zV2WsztwPSRjt?go95*&VnS3HeEj9+bK#G>hNX%C6o{F1i|-6;kr)!5$9m`Ym35_atec^&y6?34(JhnAXC zYKjlkAfSuzjEyzwCDzMyweLKBiOOz}Vuh0qkf|4rzRwu%v~avz$D4|qng>zp^FA00 z+9Ac6e8ol_?=5!D`Mb%Z4zIfA2fNPrBs}el@M;fZ7Ax9u&?Fw(KeIrivs_ilsUs`} z>wR8Pvr}^lDzUR&rJ&im&=5Oe@%F;8>=MY8o&to_%Gc-BH5%8`R>6iqa1Cpxjy1g? zI6JQhG!L~v!}(plJ*)S3&3*LBx_=Y!Ubx45qYon;qFVqiq7yZ00zBlnDna&VLjIGT zwF9}2IYhLASLqisg@3Nu%cVEp0ixyA4MbgIG0o3vYi1)ITD+k16!P(ySH&nhZ&`6~ zS=9o_qZ)K}>u*Tw1F52)iX-$t z!=9~|Y_3tZ-i;~RZ=7iovDMW3O{hSDj=*QplAcJr_Rw*NObV1A^on`P3A`9keh8bq z;N2Gw<&$Ab*@TbLPqB4PKC;8()oDYa)C#p&L{G_^(?XGh3j}rAE+CX)@q4T6OY+Xm zu9*B_wXln?O;k;|)9Ngn`spY-g*&dWFf0Q(={p ztJ~gKhy~+4*{4!iqjl8txUaluoLpOFc+abZ+ZV2GPF1#)UOTOQ-fzXoE4 zJ%>uy41}Kx3AlYf(lmBA0qcTN0aM% ztNC6l{>$!QkFy0*-TLbGcIKw*-B5G|8V5TUltls35FVwjucwh5SH|ISTvTA4W#$NSo%AAZug! zY@_RWg;o@I9=)OSd@pkTQ?TOsxXRhyk`Epand$0&f#IB?&EQS`9ekku&fo=L6e-azlwYD_E6z zv4gjkz|&4Ru!8#yRwy;^I**_GZjd$6XVx`sJW^(GRiKF?>!DI?4Vw&Q^%=u?y69tP zcgg3qi+1lXQ-7q+;`YF$^DUIZ67hi-7z$&z_q)Zd2YA{9+c=^nI#h?b08o9dT8Zx@ zV*0otEv?+m_DLwjul+z$q_Mf#zVJ+B@(al)mlMiI^M^}>kqPwZ?o%B7j{(C4v*t`K{Ub*ShgoZZnJZa%a(-b%Y{90dsv77B;YmX=ik>|& z_}O8;M1r}!3rPC3z!h3eR~@rA-J9!cS8ZXuGeMr`(FjN9cuQ@t!q}xA`CDiF@=_PF zy~@GKiS}MWu~g*2N7YQ_K#+))D7!m2Y$xn!o~#l(vxXIi_XeqB)gvE4qL}(9tR+k% z{p8_pgY0ej16yO_<~z%hX<>O3FWWpC;jvp;k9)3i9t1oNRdg*;P(z@0<9eg)NaMvpE)Y+mOPGvkr? z-09sJ_^Z-A>uIOXBQC5yjA?5QG0ZQz93Sy$EX+KPm8*rbC~8D z>MH4T!ZEcS5BqtG3#wPFk&v$slpMzbV@SvNLxIBRTc&i+ItGKVGd1}H4V3|Z>UH8! zGE|gBMBcz_(A4hav+{gLVcWaoWdW_16Vn761`wx&*%HI|W}$9*i`h987PL*CE$En1 z2fA&({Zm~_zJi&PMkwDr?tT%hw^oJ?M}EEnbwS+&WNiG~i2AU|p}`AdFZS88lx%MB zWOm{3^QkFfGRPHr>niAEjI*J5GS`FN}QER*`$f^(HeN_EF08J+>P zcq*val)R6RI)@ANyvqwhDkRx>A3zVVxX>lbOe0>|D9 z98M_|1>ViZ77a(`^Ivyemm*ve%&i`+T0y74$%ZedtnaL?7>UUlnEWVpE=3xNL~bkJ zS!g&VbNlt$YR;g%sm9*nP{Xp%3=PumJ*6XchdZkUnH8$|IS7Qd$tpB8W2TFI zvNT&L=Lm=C*|B7Bgh(d`RxtiT09YU*y{B+6Az6?s8QCo)UDggfGIu?$HvtFyd^NKY zl=sK)%`vLljigIoy-;A4r))0$tg0D{;V8Ma=SiBAeE-?urVDkSd!&?#FX$o!<6c(D z3Uf|>^7R4M%FbtTl`rku3I~}4m5K%pvpZ?3Ds8%lY8Sw<5p7X^{NeNt-UH)kKlL#v zi!jG;?n6DW4&U9mu8TW!T|Ts$KETPmpJNWrKUshC?5U;GbLN` z>o#4~rw{6IWDF5m=&=bgUWkBa4IZ~FDjku3(=u6RHvuvw<@%?UrmjB&Ubd!&=#X-s z+NQn;vx1(^Uer0CaE&b>A0|Nh>g}39;Fn0C55vGB3#UMoc1uZI!)N+U7so73r$!J! z^qhR8)!(~n&y<$vvBxf-Bge+p5iltYW$1M6c$+qh8dWteqxlj~`1s?(pS8%hk`8BtcPt{{}6`bP~Mpl;1zU~|#Cb(i3T6gEEhbK_`&h2hn zg-+Lt+LS!&;RJHh5o;N;2W&?l7jhPjUvKC_!0QF-qr zZA@s(fa4v{6RkCVAZnyWwJRAD1e|RRn|qL+#lS5hJ-0Q+C&VOsRdn2Ff1|Jx zh~n4cZWN?E5p@nUinV5qL#OzDCZtx(md&aXKIYpQYojTio&R?%0kd@2)*v3S~ z?75p+2h!$in9+7Tm%J@wSTdaXy<@{97?F+7UmURN$g|ID)Q&ySD|Rb$p(5pB%_?;M z5Z2PG!y{B*`U!ud={4cunj&0a6I`Oqs6)oeW@aw5KINHprj1`OM_>2ZT~FFQnk_mf z84R$2p0qaFDsAhJR=$-ViZ@xcw^IH6%%wh0tt^690wE7p7R}fBFpOCMZj{KfL#xcw zoAr7$w&KC(8Ky+VOK)Dfs0z9#cI1gAp8HrfR1sQ_ke8RQp6jlS70Iu-&cHorP+&9n zFe*N|ZWfc0GTS4d7P#M~cMCha&c~2wETYD`8(7o!kt}i-eSqz!OU|_48xNdZJLxl; zm?>i>ww_a*Kf-fuFxHcRv199S?FDLgQCCXKEIo5coN#kiWrKZ|Tk0j-Mh8WsB*D?M zk}I0(FBUjzovjtNs@OfXWfFwwqH-`*e2qAQdp8_pBlIBp3eL=?$w*5}?@N6#rO&w6 zTfGf~mdQ#I9q#mOa}K=-WZ>D#jVV&)<>l4I2#Zm^CW_aZ{L+5rGuCD~E_13H0`{~B z$C<>)D@k>tJ)75CaqHM$#g%Dv2dyEDUe1a|Hm=^n8f1j3B?uh5t9&-wz$g?%4QcYY z;B@F?p9We74Q!_QFJtG7JEHYgS-90pz1$cpYLQ@mqp(qTm$%5^K^cv^nhb}~#qHK> zq~@a&HGMjRPaF24Qu|Sy;S8-V(ve$BXKbrtq*Fe@QC zA5;#C+O>c{LexYOA-%E^Bu)X175hfLl^OOR)_;=e0tGWH(0=RnAbo20?S3B|Wq{+T zVKd7M;4HUKS5acC#N&;zo45d7y-#@OF;whCMD1=y5^5LEx8g#x%pw1Y4o`Ap05`rNDomuPiv6)ZWz8VnCda*? z;d6dTJ;{B;We53juXJbnkU|r{dw3^k6+ZIV`3%{BmXW8;DFw#~_fFv-u}gH_R?(n- zEMw$PRTe&VngPfNpE-19rRQL)-BspMzgmG!K!skY?MaNIL%-Srd@+ZLC9D{!a(h#9 z6f4Hd*k9&NwK*%~7BCCy3J(gZ%6{=;#rjpuPre-Tyj9zWEuMPNV7cg(Tt}96Uq<-& z+X%oN8K_Klr(vevemIAna|tfP%zGy42@e9!2EtCSzl5F4=BK8+%_`eoizZff^f752 zBRbN4iYctcN6OVH-3e6^KF~C^&ZJ^I`IWbkU}7bQg^0%sqDQiU3OJ-gS;RnGlUdrW_o3hV&S|0-0?y!H z3I|IDxr0=AX>jaqALGW9dEoTeIR$hj>(zOIr||C|QfYkv5pgp+Rl@P%)0!?i-15M0 zRNs)M74}}U(NmG+)~~4sx;Q->;(G8}Ek3?GX9`jS+&s(-)2GsNDJW=rp}GJXDL}J; zgB&b3U`QoE5z`lZF-O^)3nB=yhD+aI}m;*8w`;4_;T!6@XI<1y|+#!Qtl)4PPtn zv8=9eg$7y>OiSpa0H<(0dB=T&o)sULnbu$fnKMoa)WkuXeq>Ot;(5tV)iPi)o(FEW zysMJ@UdQSaq)>ycg-i>#z$u6=CFP+4ij>e(Q@U&&mF^@{Jho~l#N>gaCh&T2BE>?p zt88#A(=@-1-K+iiAWG=@g&clJNL1IBjad9TK6^ccgG(vs=wQEUn|po(@e+s zIy*ECs~zXv`4>;DH+bnk7;AOPkph9u_>!TDnL1!+PwtJ`gne{a?F6 z$7j>laXwMz(xs|EyfahO6+E(kKiHK0yjqo9`dq|qwK~*{E~%xY~(EU zYYA2YIYiMupuR&BJ@g#k@paY=RL&qpDRm_{IIT;ZY zQMz|=lk;;5R*)vKbn>K`c5EY}L>b{g>j@4i2Qh2dh1pZ1VDw03DkBvNk8TJ`8lNrZRT(Vr)ndpy&uz zIs^6>I_RDScCUw)hKqq6+>L*;KiL-)>qYJc=@(Lr9lj+KD^ALrX`hR-~mU_eF3_Thx^7KZ)WT4xRXa8RiFA literal 0 HcmV?d00001 diff --git a/examples/usb/device/usb_dongle/_static/ESP32-S3.png b/examples/usb/device/usb_dongle/_static/ESP32-S3.png new file mode 100644 index 0000000000000000000000000000000000000000..0cb737615ac0a6ac3552a3d2c9391b83476ac523 GIT binary patch literal 13636 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

*e;pYN&Kw-sYP$U-Tccqtnf#oP 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 literal 0 HcmV?d00001 diff --git a/examples/usb/device/usb_dongle/_static/tinyusb_config.png b/examples/usb/device/usb_dongle/_static/tinyusb_config.png new file mode 100644 index 0000000000000000000000000000000000000000..bfdad8b898329f5a2c9120c9c9e0158e0dabc209 GIT binary patch literal 44087 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 literal 0 HcmV?d00001 diff --git a/examples/usb/device/usb_dongle/_static/uart_config.png b/examples/usb/device/usb_dongle/_static/uart_config.png new file mode 100644 index 0000000000000000000000000000000000000000..1673713b226af87690cdbc7ca6a1b456bfb18556 GIT binary patch literal 30142 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 ShWY{P1KIaVQY8{bU;YP%Am~j1 literal 0 HcmV?d00001 diff --git a/examples/usb/device/usb_dongle/components/FreeRTOS-Plus-CLI/CMakeLists.txt b/examples/usb/device/usb_dongle/components/FreeRTOS-Plus-CLI/CMakeLists.txt new file mode 100644 index 000000000..4dae72e53 --- /dev/null +++ b/examples/usb/device/usb_dongle/components/FreeRTOS-Plus-CLI/CMakeLists.txt @@ -0,0 +1,3 @@ +idf_component_register(SRCS "FreeRTOS_CLI.c" + INCLUDE_DIRS include + PRIV_REQUIRES ) diff --git a/examples/usb/device/usb_dongle/components/FreeRTOS-Plus-CLI/FreeRTOS_CLI.c b/examples/usb/device/usb_dongle/components/FreeRTOS-Plus-CLI/FreeRTOS_CLI.c new file mode 100644 index 000000000..02a26f990 --- /dev/null +++ b/examples/usb/device/usb_dongle/components/FreeRTOS-Plus-CLI/FreeRTOS_CLI.c @@ -0,0 +1,322 @@ +/* + * FreeRTOS+CLI V1.0.4 + * Copyright (C) 2017 Amazon.com, Inc. or its affiliates. All Rights Reserved. + * + * 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. + * + * http://www.FreeRTOS.org + * http://aws.amazon.com/freertos + * + * 1 tab == 4 spaces! + */ + +/* Standard includes. */ +#include +#include + +/* FreeRTOS includes. */ +#include "freertos/FreeRTOS.h" +#include "freertos/task.h" + +/* Utils includes. */ +#include "FreeRTOS_CLI.h" + +/* If the application writer needs to place the buffer used by the CLI at a +fixed address then set configAPPLICATION_PROVIDES_cOutputBuffer to 1 in +FreeRTOSConfig.h, then declare an array with the following name and size in +one of the application files: + char cOutputBuffer[ configCOMMAND_INT_MAX_OUTPUT_SIZE ]; +*/ +#ifndef configAPPLICATION_PROVIDES_cOutputBuffer +#define configAPPLICATION_PROVIDES_cOutputBuffer 0 +#endif + +PRIVILEGED_DATA portMUX_TYPE xCLIMux = portMUX_INITIALIZER_UNLOCKED; + +typedef struct xCOMMAND_INPUT_LIST { + const CLI_Command_Definition_t *pxCommandLineDefinition; + struct xCOMMAND_INPUT_LIST *pxNext; +} CLI_Definition_List_Item_t; + +/* + * The callback function that is executed when "help" is entered. This is the + * only default command that is always present. + */ +static BaseType_t prvHelpCommand(char *pcWriteBuffer, size_t xWriteBufferLen, const char *pcCommandString); + +/* + * Return the number of parameters that follow the command name. + */ +static int8_t prvGetNumberOfParameters(const char *pcCommandString); + +/* The definition of the "help" command. This command is always at the front +of the list of registered commands. */ +static const CLI_Command_Definition_t xHelpCommand = { + "help", + "help:\r\n Lists all the registered commands\r\n\r\n", + prvHelpCommand, + 0 +}; + +/* The definition of the list of commands. Commands that are registered are +added to this list. */ +static CLI_Definition_List_Item_t xRegisteredCommands = { + &xHelpCommand, /* The first command in the list is always the help command, defined in this file. */ + NULL /* The next pointer is initialised to NULL, as there are no other registered commands yet. */ +}; + +/* A buffer into which command outputs can be written is declared here, rather +than in the command console implementation, to allow multiple command consoles +to share the same buffer. For example, an application may allow access to the +command interpreter by UART and by Ethernet. Sharing a buffer is done purely +to save RAM. Note, however, that the command console itself is not re-entrant, +so only one command interpreter interface can be used at any one time. For that +reason, no attempt at providing mutual exclusion to the cOutputBuffer array is +attempted. + +configAPPLICATION_PROVIDES_cOutputBuffer is provided to allow the application +writer to provide their own cOutputBuffer declaration in cases where the +buffer needs to be placed at a fixed address (rather than by the linker). */ +#if( configAPPLICATION_PROVIDES_cOutputBuffer == 0 ) +static char cOutputBuffer[ configCOMMAND_INT_MAX_OUTPUT_SIZE ]; +#else +extern char cOutputBuffer[ configCOMMAND_INT_MAX_OUTPUT_SIZE ]; +#endif + +/*-----------------------------------------------------------*/ + +void FreeRTOS_CLICreatMux(void) +{ + spinlock_initialize(&xCLIMux); +} +/*-----------------------------------------------------------*/ + +BaseType_t FreeRTOS_CLIRegisterCommand(const CLI_Command_Definition_t * const pxCommandToRegister) +{ + static CLI_Definition_List_Item_t *pxLastCommandInList = &xRegisteredCommands; + CLI_Definition_List_Item_t *pxNewListItem; + BaseType_t xReturn = pdFAIL; + + /* Check the parameter is not NULL. */ + configASSERT(pxCommandToRegister); + + /* Create a new list item that will reference the command being registered. */ + pxNewListItem = (CLI_Definition_List_Item_t *) pvPortMalloc(sizeof(CLI_Definition_List_Item_t)); + configASSERT(pxNewListItem); + + if (pxNewListItem != NULL) { + taskENTER_CRITICAL(&xCLIMux); + { + /* Reference the command being registered from the newly created + list item. */ + pxNewListItem->pxCommandLineDefinition = pxCommandToRegister; + + /* The new list item will get added to the end of the list, so + pxNext has nowhere to point. */ + pxNewListItem->pxNext = NULL; + + /* Add the newly created list item to the end of the already existing + list. */ + pxLastCommandInList->pxNext = pxNewListItem; + + /* Set the end of list marker to the new list item. */ + pxLastCommandInList = pxNewListItem; + } + taskEXIT_CRITICAL(&xCLIMux); + + xReturn = pdPASS; + } + + return xReturn; +} +/*-----------------------------------------------------------*/ + +BaseType_t FreeRTOS_CLIProcessCommand(const char * const pcCommandInput, char * pcWriteBuffer, size_t xWriteBufferLen) +{ + static const CLI_Definition_List_Item_t *pxCommand = NULL; + BaseType_t xReturn = pdTRUE; + const char *pcRegisteredCommandString; + size_t xCommandStringLength; + + /* Note: This function is not re-entrant. It must not be called from more + thank one task. */ + + if (pxCommand == NULL) { + /* Search for the command string in the list of registered commands. */ + for (pxCommand = &xRegisteredCommands; pxCommand != NULL; pxCommand = pxCommand->pxNext) { + pcRegisteredCommandString = pxCommand->pxCommandLineDefinition->pcCommand; + xCommandStringLength = strlen(pcRegisteredCommandString); + + /* To ensure the string lengths match exactly, so as not to pick up + a sub-string of a longer command, check the byte after the expected + end of the string is either the end of the string or a space before + a parameter. */ + if (strncmp(pcCommandInput, pcRegisteredCommandString, xCommandStringLength) == 0) { + if ((pcCommandInput[ xCommandStringLength ] == ' ') || (pcCommandInput[ xCommandStringLength ] == 0x00)) { + /* The command has been found. Check it has the expected + number of parameters. If cExpectedNumberOfParameters is -1, + then there could be a variable number of parameters and no + check is made. */ + if (pxCommand->pxCommandLineDefinition->cExpectedNumberOfParameters >= 0) { + if (prvGetNumberOfParameters(pcCommandInput) != pxCommand->pxCommandLineDefinition->cExpectedNumberOfParameters) { + xReturn = pdFALSE; + } + } + + break; + } + } + } + } + + if ((pxCommand != NULL) && (xReturn == pdFALSE)) { + /* The command was found, but the number of parameters with the command + was incorrect. */ + strncpy(pcWriteBuffer, "Incorrect command parameter(s). Enter \"help\" to view a list of available commands.\r\n", xWriteBufferLen); + pxCommand = NULL; + } else if (pxCommand != NULL) { + /* Call the callback function that is registered to this command. */ + xReturn = pxCommand->pxCommandLineDefinition->pxCommandInterpreter(pcWriteBuffer, xWriteBufferLen, pcCommandInput); + + /* If xReturn is pdFALSE, then no further strings will be returned + after this one, and pxCommand can be reset to NULL ready to search + for the next entered command. */ + if (xReturn == pdFALSE) { + pxCommand = NULL; + } + } else { + /* pxCommand was NULL, the command was not found. */ + strncpy(pcWriteBuffer, "Command not recognised. Enter 'help' to view a list of available commands.\r\n", xWriteBufferLen); + xReturn = pdFALSE; + } + + return xReturn; +} +/*-----------------------------------------------------------*/ + +char *FreeRTOS_CLIGetOutputBuffer(void) +{ + return cOutputBuffer; +} +/*-----------------------------------------------------------*/ + +const char *FreeRTOS_CLIGetParameter(const char *pcCommandString, UBaseType_t uxWantedParameter, BaseType_t *pxParameterStringLength) +{ + UBaseType_t uxParametersFound = 0; + const char *pcReturn = NULL; + + *pxParameterStringLength = 0; + + while (uxParametersFound < uxWantedParameter) { + /* Index the character pointer past the current word. If this is the start + of the command string then the first word is the command itself. */ + while (((*pcCommandString) != 0x00) && ((*pcCommandString) != ' ')) { + pcCommandString++; + } + + /* Find the start of the next string. */ + while (((*pcCommandString) != 0x00) && ((*pcCommandString) == ' ')) { + pcCommandString++; + } + + /* Was a string found? */ + if (*pcCommandString != 0x00) { + /* Is this the start of the required parameter? */ + uxParametersFound++; + + if (uxParametersFound == uxWantedParameter) { + /* How long is the parameter? */ + pcReturn = pcCommandString; + while (((*pcCommandString) != 0x00) && ((*pcCommandString) != ' ')) { + (*pxParameterStringLength)++; + pcCommandString++; + } + + if (*pxParameterStringLength == 0) { + pcReturn = NULL; + } + + break; + } + } else { + break; + } + } + + return pcReturn; +} +/*-----------------------------------------------------------*/ + +static BaseType_t prvHelpCommand(char *pcWriteBuffer, size_t xWriteBufferLen, const char *pcCommandString) +{ + static const CLI_Definition_List_Item_t * pxCommand = NULL; + BaseType_t xReturn; + + (void) pcCommandString; + + if (pxCommand == NULL) { + /* Reset the pxCommand pointer back to the start of the list. */ + pxCommand = &xRegisteredCommands; + } + + /* Return the next command help string, before moving the pointer on to + the next command in the list. */ + strncpy(pcWriteBuffer, pxCommand->pxCommandLineDefinition->pcHelpString, xWriteBufferLen); + pxCommand = pxCommand->pxNext; + + if (pxCommand == NULL) { + /* There are no more commands in the list, so there will be no more + strings to return after this one and pdFALSE should be returned. */ + xReturn = pdFALSE; + } else { + xReturn = pdTRUE; + } + + return xReturn; +} +/*-----------------------------------------------------------*/ + +static int8_t prvGetNumberOfParameters(const char *pcCommandString) +{ + int8_t cParameters = 0; + BaseType_t xLastCharacterWasSpace = pdFALSE; + + /* Count the number of space delimited words in pcCommandString. */ + while (*pcCommandString != 0x00) { + if ((*pcCommandString) == ' ') { + if (xLastCharacterWasSpace != pdTRUE) { + cParameters++; + xLastCharacterWasSpace = pdTRUE; + } + } else { + xLastCharacterWasSpace = pdFALSE; + } + + pcCommandString++; + } + + /* If the command string ended with spaces, then there will have been too + many parameters counted. */ + if (xLastCharacterWasSpace == pdTRUE) { + cParameters--; + } + + /* The value returned is one less than the number of space delimited words, + as the first word should be the command itself. */ + return cParameters; +} diff --git a/examples/usb/device/usb_dongle/components/FreeRTOS-Plus-CLI/History.txt b/examples/usb/device/usb_dongle/components/FreeRTOS-Plus-CLI/History.txt new file mode 100644 index 000000000..f3d2626ca --- /dev/null +++ b/examples/usb/device/usb_dongle/components/FreeRTOS-Plus-CLI/History.txt @@ -0,0 +1,31 @@ +Changes between V1.0.3 and V1.0.4 released + + + Update to use stdint and the FreeRTOS specific typedefs that were + introduced in FreeRTOS V8.0.0. + +Changes between V1.0.2 and V1.0.3 released + + + Previously, and in line with good software engineering practice, the + FreeRTOS coding standard did not permit the use of char types that were + not explicitly qualified as either signed or unsigned. As a result char + pointers used to reference strings required casts, as did the use of any + standard string handling functions. The casts ensured compiler warnings + were not generated by compilers that defaulted unqualified char types to + be signed or compilers that defaulted unqualified char types to be + unsigned. As it has in later MISRA standards, this rule has now been + relaxed, and unqualified char types are now permitted, but only when: + 1) The char is used to point to a human readable text string. + 2) The char is used to hold a single ASCII character. + +Changes between V1.0.1 and V1.0.2 released 14/10/2013 + + + Changed double quotes (") to single quotes (') in the help string to + allow the strings to be used with JSON in FreeRTOS+Nabto. + +Changes between V1.0.0 and V1.0.1 released 05/07/2012 + + + Change the name of the structure used to map a function that implements + a CLI command to the string used to call the command from + xCommandLineInput to CLI_Command_Definition_t, as it was always intended + to be. A #define was added to map the old name to the new name for + reasons of backward compatibility. diff --git a/examples/usb/device/usb_dongle/components/FreeRTOS-Plus-CLI/LICENSE_INFORMATION.txt b/examples/usb/device/usb_dongle/components/FreeRTOS-Plus-CLI/LICENSE_INFORMATION.txt new file mode 100644 index 000000000..a8b0a6d36 --- /dev/null +++ b/examples/usb/device/usb_dongle/components/FreeRTOS-Plus-CLI/LICENSE_INFORMATION.txt @@ -0,0 +1,19 @@ +FreeRTOS+CLI is released under the following MIT license. + +Copyright (C) 2017 Amazon.com, Inc. or its affiliates. All Rights Reserved. +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. diff --git a/examples/usb/device/usb_dongle/components/FreeRTOS-Plus-CLI/ReadMe.url b/examples/usb/device/usb_dongle/components/FreeRTOS-Plus-CLI/ReadMe.url new file mode 100644 index 000000000..afc826fd2 --- /dev/null +++ b/examples/usb/device/usb_dongle/components/FreeRTOS-Plus-CLI/ReadMe.url @@ -0,0 +1,5 @@ +[InternetShortcut] +URL=http://www.freertos.org/cli +IDList= +[{000214A0-0000-0000-C000-000000000046}] +Prop3=19,2 diff --git a/examples/usb/device/usb_dongle/components/FreeRTOS-Plus-CLI/component.mk b/examples/usb/device/usb_dongle/components/FreeRTOS-Plus-CLI/component.mk new file mode 100644 index 000000000..1b25f971a --- /dev/null +++ b/examples/usb/device/usb_dongle/components/FreeRTOS-Plus-CLI/component.mk @@ -0,0 +1,7 @@ +# +# "main" pseudo-component makefile. +# +# (Uses default behaviour of compiling all source files in directory, adding 'include' to include path.) + +COMPONENT_ADD_INCLUDEDIRS := ./include +COMPONENT_SRCDIRS := . diff --git a/examples/usb/device/usb_dongle/components/FreeRTOS-Plus-CLI/include/FreeRTOS_CLI.h b/examples/usb/device/usb_dongle/components/FreeRTOS-Plus-CLI/include/FreeRTOS_CLI.h new file mode 100644 index 000000000..40049c409 --- /dev/null +++ b/examples/usb/device/usb_dongle/components/FreeRTOS-Plus-CLI/include/FreeRTOS_CLI.h @@ -0,0 +1,101 @@ +/* + * FreeRTOS+CLI V1.0.4 + * Copyright (C) 2017 Amazon.com, Inc. or its affiliates. All Rights Reserved. + * + * 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. + * + * http://www.FreeRTOS.org + * http://aws.amazon.com/freertos + * + * 1 tab == 4 spaces! + */ + +#ifndef COMMAND_INTERPRETER_H +#define COMMAND_INTERPRETER_H + +#include "freertos/FreeRTOS.h" + +/* The prototype to which callback functions used to process command line +commands must comply. pcWriteBuffer is a buffer into which the output from +executing the command can be written, xWriteBufferLen is the length, in bytes of +the pcWriteBuffer buffer, and pcCommandString is the entire string as input by +the user (from which parameters can be extracted).*/ +typedef BaseType_t (*pdCOMMAND_LINE_CALLBACK)(char *pcWriteBuffer, size_t xWriteBufferLen, const char *pcCommandString); + +/* The structure that defines command line commands. A command line command +should be defined by declaring a const structure of this type. */ +typedef struct xCOMMAND_LINE_INPUT { + const char * const pcCommand; /* The command that causes pxCommandInterpreter to be executed. For example "help". Must be all lower case. */ + const char * const pcHelpString; /* String that describes how to use the command. Should start with the command itself, and end with "\r\n". For example "help: Returns a list of all the commands\r\n". */ + const pdCOMMAND_LINE_CALLBACK pxCommandInterpreter; /* A pointer to the callback function that will return the output generated by the command. */ + int8_t cExpectedNumberOfParameters; /* Commands expect a fixed number of parameters, which may be zero. */ +} CLI_Command_Definition_t; + +/* For backward compatibility. */ +#define xCommandLineInput CLI_Command_Definition_t +#define configCOMMAND_INT_MAX_OUTPUT_SIZE 500 + +/* + * Create a Mux. + * + * Note: Must be called before FreeRTOS_CLIProcessCommand. + */ +void FreeRTOS_CLICreatMux(void); + +/* + * Register the command passed in using the pxCommandToRegister parameter. + * Registering a command adds the command to the list of commands that are + * handled by the command interpreter. Once a command has been registered it + * can be executed from the command line. + */ +BaseType_t FreeRTOS_CLIRegisterCommand(const CLI_Command_Definition_t * const pxCommandToRegister); + +/* + * Runs the command interpreter for the command string "pcCommandInput". Any + * output generated by running the command will be placed into pcWriteBuffer. + * xWriteBufferLen must indicate the size, in bytes, of the buffer pointed to + * by pcWriteBuffer. + * + * FreeRTOS_CLIProcessCommand should be called repeatedly until it returns pdFALSE. + * + * pcCmdIntProcessCommand is not reentrant. It must not be called from more + * than one task - or at least - by more than one task at a time. + */ +BaseType_t FreeRTOS_CLIProcessCommand(const char * const pcCommandInput, char * pcWriteBuffer, size_t xWriteBufferLen); + +/*-----------------------------------------------------------*/ + +/* + * A buffer into which command outputs can be written is declared in the + * main command interpreter, rather than in the command console implementation, + * to allow application that provide access to the command console via multiple + * interfaces to share a buffer, and therefore save RAM. Note, however, that + * the command interpreter itself is not re-entrant, so only one command + * console interface can be used at any one time. For that reason, no attempt + * is made to provide any mutual exclusion mechanism on the output buffer. + * + * FreeRTOS_CLIGetOutputBuffer() returns the address of the output buffer. + */ +char *FreeRTOS_CLIGetOutputBuffer(void); + +/* + * Return a pointer to the xParameterNumber'th word in pcCommandString. + */ +const char *FreeRTOS_CLIGetParameter(const char *pcCommandString, UBaseType_t uxWantedParameter, BaseType_t *pxParameterStringLength); + +#endif /* COMMAND_INTERPRETER_H */ diff --git a/examples/usb/device/usb_dongle/components/FreeRTOS-Plus-CLI/readme.txt b/examples/usb/device/usb_dongle/components/FreeRTOS-Plus-CLI/readme.txt new file mode 100644 index 000000000..5f3ed31b2 --- /dev/null +++ b/examples/usb/device/usb_dongle/components/FreeRTOS-Plus-CLI/readme.txt @@ -0,0 +1,2 @@ +Contains source and header files that implement FreeRTOS+CLI. See +http://www.FreeRTOS.org/cli for documentation and license information. diff --git a/examples/usb/device/usb_dongle/components/tinyusb_dongle/CMakeLists.txt b/examples/usb/device/usb_dongle/components/tinyusb_dongle/CMakeLists.txt new file mode 100644 index 000000000..3f7affb97 --- /dev/null +++ b/examples/usb/device/usb_dongle/components/tinyusb_dongle/CMakeLists.txt @@ -0,0 +1,65 @@ +set(srcs + "descriptors_control.c" + "tinyusb.c" + "usb_descriptors.c" + ) + +if(NOT CONFIG_TINYUSB_NO_DEFAULT_TASK) + list(APPEND srcs "tusb_tasks.c") +endif() # CONFIG_TINYUSB_NO_DEFAULT_TASK + +if(CONFIG_TINYUSB_CDC_ENABLED) + list(APPEND srcs + "cdc.c" + "tusb_cdc_acm.c" + ) + if(CONFIG_VFS_SUPPORT_IO) + list(APPEND srcs + "tusb_console.c" + "vfs_tinyusb.c" + ) + endif() # CONFIG_VFS_SUPPORT_IO +endif() # CONFIG_TINYUSB_CDC_ENABLED + +if(CONFIG_TINYUSB_BTH_ENABLED) + list(APPEND srcs + tusb_bth.c + ) +endif() # CONFIG_TINYUSB_BTH_ENABLED + +if(CONFIG_TINYUSB_DFU_ENABLED) + list (APPEND srcs + tusb_dfu.c + tusb_dfu_ota.c + ) +endif() # CONFIG_TINYUSB_DFU_ENABLED + +if(CONFIG_TINYUSB_NET_MODE_NCM + OR CONFIG_TINYUSB_NET_MODE_RNDIS + OR CONFIG_TINYUSB_NET_MODE_ECM) + list(APPEND srcs + tinyusb_net.c + ) +endif() # CONFIG_TINYUSB_NET_MODE_NCM || CONFIG_TINYUSB_NET_MODE_ECM || CONFIG_TINYUSB_NET_MODE_RNDIS + +idf_component_register(SRCS ${srcs} + INCLUDE_DIRS "include" + PRIV_INCLUDE_DIRS "include_private" + PRIV_REQUIRES usb + REQUIRES fatfs vfs bt nvs_flash app_update + ) + +# Determine whether tinyusb is fetched from component registry or from local path +idf_build_get_property(build_components BUILD_COMPONENTS) +if(tinyusb IN_LIST build_components) + set(tinyusb_name tinyusb) # Local component +else() + set(tinyusb_name espressif__tinyusb) # Managed component +endif() + +# Pass tusb_config.h from this component to TinyUSB +idf_component_get_property(tusb_lib ${tinyusb_name} COMPONENT_LIB) +target_include_directories(${tusb_lib} PRIVATE "include") +if(CONFIG_TINYUSB_DFU_ENABLED) + target_sources(${tusb_lib} PUBLIC "tusb_dfu.c") +endif() # CONFIG_TINYUSB_DFU_ENABLED diff --git a/examples/usb/device/usb_dongle/components/tinyusb_dongle/Kconfig b/examples/usb/device/usb_dongle/components/tinyusb_dongle/Kconfig new file mode 100644 index 000000000..e356f377f --- /dev/null +++ b/examples/usb/device/usb_dongle/components/tinyusb_dongle/Kconfig @@ -0,0 +1,227 @@ +menu "TinyUSB Stack" + config TINYUSB_DEBUG_LEVEL + int "TinyUSB log level (0-3)" + default 1 + range 0 3 + help + Specify verbosity of TinyUSB log output. + + choice TINYUSB_RHPORT + depends on IDF_TARGET_ESP32P4 + prompt "TinyUSB PHY" + default TINYUSB_RHPORT_HS + help + Allows set the USB PHY Controller for TinyUSB: HS (USB OTG2.0 PHY for HighSpeed) + + config TINYUSB_RHPORT_HS + bool "HS" + endchoice + + menu "TinyUSB task configuration" + config TINYUSB_NO_DEFAULT_TASK + bool "Do not create a TinyUSB task" + default n + help + This option allows to not create the FreeRTOS task during the driver initialization. + User will have to handle TinyUSB events manually. + + config TINYUSB_TASK_PRIORITY + int "TinyUSB task priority" + default 5 + depends on !TINYUSB_NO_DEFAULT_TASK + help + Set the priority of the default TinyUSB main task. + + config TINYUSB_TASK_STACK_SIZE + int "TinyUSB task stack size (bytes)" + default 4096 + depends on !TINYUSB_NO_DEFAULT_TASK + help + Set the stack size of the default TinyUSB main task. + + choice TINYUSB_TASK_AFFINITY + prompt "TinyUSB task affinity" + default TINYUSB_TASK_AFFINITY_NO_AFFINITY + depends on !TINYUSB_NO_DEFAULT_TASK + help + Allows setting TinyUSB tasks affinity, i.e. whether the task is pinned to + CPU0, pinned to CPU1, or allowed to run on any CPU. + + config TINYUSB_TASK_AFFINITY_NO_AFFINITY + bool "No affinity" + config TINYUSB_TASK_AFFINITY_CPU0 + bool "CPU0" + config TINYUSB_TASK_AFFINITY_CPU1 + bool "CPU1" + depends on !FREERTOS_UNICORE + endchoice + + config TINYUSB_TASK_AFFINITY + hex + default FREERTOS_NO_AFFINITY if TINYUSB_TASK_AFFINITY_NO_AFFINITY + default 0x0 if TINYUSB_TASK_AFFINITY_CPU0 + default 0x1 if TINYUSB_TASK_AFFINITY_CPU1 + + config TINYUSB_INIT_IN_DEFAULT_TASK + bool "Initialize TinyUSB stack within the default TinyUSB task" + default n + depends on !TINYUSB_NO_DEFAULT_TASK + help + Run TinyUSB stack initialization just after starting the default TinyUSB task. + This is especially useful in multicore scenarios, when we need to pin the task + to a specific core and, at the same time initialize TinyUSB stack + (i.e. install interrupts) on the same core. + endmenu + + menu "Descriptor configuration" + comment "You can provide your custom descriptors via tinyusb_driver_install()" + config TINYUSB_DESC_USE_ESPRESSIF_VID + bool "VID: Use Espressif's vendor ID" + default y + help + Enable this option, USB device will use Espressif's vendor ID as its VID. + This is helpful at product develop stage. + + config TINYUSB_DESC_CUSTOM_VID + hex "VID: Custom vendor ID" + default 0xcafe + depends on !TINYUSB_DESC_USE_ESPRESSIF_VID + help + Custom Vendor ID. + + config TINYUSB_DESC_USE_DEFAULT_PID + bool "PID: Use a default PID assigned to TinyUSB" + default y + help + Default TinyUSB PID assigning uses values 0x4000...0x4007. + + config TINYUSB_DESC_CUSTOM_PID + hex "PID: Custom product ID" + default 0x5678 + depends on !TINYUSB_DESC_USE_DEFAULT_PID + help + Custom Product ID. + + config TINYUSB_DESC_BCD_DEVICE + hex "bcdDevice" + default 0x0100 + help + Version of the firmware of the USB device. + + config TINYUSB_DESC_MANUFACTURER_STRING + string "Manufacturer name" + default "Espressif Systems" + help + Name of the manufacturer of the USB device. + + config TINYUSB_DESC_PRODUCT_STRING + string "Product name" + default "Espressif Device" + help + Name of the USB device. + + config TINYUSB_DESC_SERIAL_STRING + string "Serial string" + default "123456" + help + Serial number of the USB device. + + config TINYUSB_DESC_CDC_STRING + depends on TINYUSB_CDC_ENABLED + string "CDC Device String" + default "Espressif CDC Device" + help + Name of the CDC device. + + config TINYUSB_DESC_BTH_STRING + depends on TINYUSB_BTH_ENABLED + string "BTH String" + default "Espressif BTH Device" + help + Name of the BTH device. + + endmenu # "Descriptor configuration" + + menu "Communication Device Class (CDC)" + config TINYUSB_CDC_ENABLED + bool "Enable TinyUSB CDC feature" + default n + help + Enable TinyUSB CDC feature. + + config TINYUSB_CDC_COUNT + int "CDC Channel Count" + default 1 + range 1 2 + depends on TINYUSB_CDC_ENABLED + help + Number of independent serial ports. + + config TINYUSB_CDC_RX_BUFSIZE + depends on TINYUSB_CDC_ENABLED + int "CDC FIFO size of RX channel" + default 512 + range 64 10000 + help + CDC FIFO size of RX channel. + + config TINYUSB_CDC_TX_BUFSIZE + depends on TINYUSB_CDC_ENABLED + int "CDC FIFO size of TX channel" + default 512 + help + CDC FIFO size of TX channel. + endmenu # "Communication Device Class" + + menu "Device Firmware Upgrade (DFU)" + config TINYUSB_DFU_ENABLED + bool "Enable TinyUSB DFU feature" + default n + help + Enable TinyUSB DFU feature. + + config TINYUSB_DFU_BUFSIZE + depends on TINYUSB_DFU_ENABLED + int "DFU XFER BUFFSIZE" + default 512 + help + DFU XFER BUFFSIZE. + endmenu # Device Firmware Upgrade (DFU) + + menu "Bluetooth Host Class (BTH)" + config TINYUSB_BTH_ENABLED + bool "Enable TinyUSB BTH feature" + default n + help + Enable TinyUSB BTH feature. + + config TINYUSB_BTH_ISO_ALT_COUNT + depends on TINYUSB_BTH_ENABLED + int "BTH ISO ALT COUNT" + default 0 + help + BTH ISO ALT COUNT. + endmenu # "Bluetooth Host Device Class" + + menu "Network driver (ECM/NCM/RNDIS)" + choice TINYUSB_NET_MODE + prompt "Network mode" + default TINYUSB_NET_MODE_NONE + help + Select network driver you want to use. + + config TINYUSB_NET_MODE_RNDIS + bool "RNDIS" + + config TINYUSB_NET_MODE_ECM + bool "ECM" + + config TINYUSB_NET_MODE_NCM + bool "NCM" + + config TINYUSB_NET_MODE_NONE + bool "None" + endchoice + endmenu # "Network driver (ECM/NCM/RNDIS)" + +endmenu # "TinyUSB Stack" diff --git a/examples/usb/device/usb_dongle/components/tinyusb_dongle/LICENSE b/examples/usb/device/usb_dongle/components/tinyusb_dongle/LICENSE new file mode 100644 index 000000000..d64569567 --- /dev/null +++ b/examples/usb/device/usb_dongle/components/tinyusb_dongle/LICENSE @@ -0,0 +1,202 @@ + + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS + + APPENDIX: How to apply the Apache License to your work. + + To apply the Apache License to your work, attach the following + boilerplate notice, with the fields enclosed by brackets "[]" + replaced with your own identifying information. (Don't include + the brackets!) The text should be enclosed in the appropriate + comment syntax for the file format. We also recommend that a + file or class name and description of purpose be included on the + same "printed page" as the copyright notice for easier + identification within third-party archives. + + Copyright [yyyy] [name of copyright owner] + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. diff --git a/examples/usb/device/usb_dongle/components/tinyusb_dongle/cdc.c b/examples/usb/device/usb_dongle/components/tinyusb_dongle/cdc.c new file mode 100644 index 000000000..0af2d5c1e --- /dev/null +++ b/examples/usb/device/usb_dongle/components/tinyusb_dongle/cdc.c @@ -0,0 +1,109 @@ +/* + * SPDX-FileCopyrightText: 2020-2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#include +#include "esp_check.h" +#include "esp_err.h" +#include "esp_log.h" +#include "tusb.h" +#include "cdc.h" + +#define CDC_INTF_NUM CFG_TUD_CDC // number of cdc blocks +static esp_tusb_cdc_t *cdc_obj[CDC_INTF_NUM] = {}; +static const char *TAG = "tusb_cdc"; + +esp_tusb_cdc_t *tinyusb_cdc_get_intf(int itf_num) +{ + if (itf_num >= CDC_INTF_NUM || itf_num < 0) { + return NULL; + } + return cdc_obj[itf_num]; +} + +static esp_err_t cdc_obj_check(int itf, bool expected_inited, tusb_class_code_t expected_type) +{ + esp_tusb_cdc_t *this_itf = tinyusb_cdc_get_intf(itf); + + bool inited = (this_itf != NULL); + ESP_RETURN_ON_FALSE(expected_inited == inited, ESP_ERR_INVALID_STATE, TAG, "Wrong state of the interface. Expected state: %s", expected_inited ? "initialized" : "not initialized"); + ESP_RETURN_ON_FALSE(!(inited && (expected_type != -1) && !(this_itf->type == expected_type)), ESP_ERR_INVALID_STATE, TAG, "Wrong type of the interface. Should be : 0x%x (tusb_class_code_t)", expected_type); + return ESP_OK; +} + +static esp_err_t tusb_cdc_comm_init(int itf) +{ + ESP_RETURN_ON_ERROR(cdc_obj_check(itf, false, -1), TAG, "cdc_obj_check failed"); + cdc_obj[itf] = calloc(1, sizeof(esp_tusb_cdc_t)); + if (cdc_obj[itf] != NULL) { + cdc_obj[itf]->type = TUSB_CLASS_CDC; + ESP_LOGD(TAG, "CDC Comm class initialized"); + return ESP_OK; + } else { + ESP_LOGE(TAG, "CDC Comm initialization error"); + return ESP_FAIL; + } +} + +static esp_err_t tusb_cdc_deinit_comm(int itf) +{ + ESP_RETURN_ON_ERROR(cdc_obj_check(itf, true, TUSB_CLASS_CDC), TAG, "cdc_obj_check failed"); + free(cdc_obj[itf]); + cdc_obj[itf] = NULL; + return ESP_OK; +} + +static esp_err_t tusb_cdc_data_init(int itf) +{ + ESP_RETURN_ON_ERROR(cdc_obj_check(itf, false, TUSB_CLASS_CDC_DATA), TAG, "cdc_obj_check failed"); + cdc_obj[itf] = calloc(1, sizeof(esp_tusb_cdc_t)); + if (cdc_obj[itf] != NULL) { + cdc_obj[itf]->type = TUSB_CLASS_CDC_DATA; + ESP_LOGD(TAG, "CDC Data class initialized"); + return ESP_OK; + } else { + ESP_LOGE(TAG, "CDC Data initialization error"); + return ESP_FAIL; + } +} + +static esp_err_t tusb_cdc_deinit_data(int itf) +{ + ESP_RETURN_ON_ERROR(cdc_obj_check(itf, true, TUSB_CLASS_CDC_DATA), TAG, "cdc_obj_check failed"); + free(cdc_obj[itf]); + cdc_obj[itf] = NULL; + return ESP_OK; +} + +esp_err_t tinyusb_cdc_init(int itf, const tinyusb_config_cdc_t *cfg) +{ + ESP_RETURN_ON_ERROR(cdc_obj_check(itf, false, -1), TAG, "cdc_obj_check failed"); + + ESP_LOGD(TAG, "Init CDC %d", itf); + if (cfg->cdc_class == TUSB_CLASS_CDC) { + ESP_RETURN_ON_ERROR(tusb_cdc_comm_init(itf), TAG, "tusb_cdc_comm_init failed"); + cdc_obj[itf]->cdc_subclass.comm_subclass = cfg->cdc_subclass.comm_subclass; + } else { + ESP_RETURN_ON_ERROR(tusb_cdc_data_init(itf), TAG, "tusb_cdc_data_init failed"); + cdc_obj[itf]->cdc_subclass.data_subclass = cfg->cdc_subclass.data_subclass; + } + cdc_obj[itf]->usb_dev = cfg->usb_dev; + return ESP_OK; +} + +esp_err_t tinyusb_cdc_deinit(int itf) +{ + ESP_RETURN_ON_ERROR(cdc_obj_check(itf, true, -1), TAG, "cdc_obj_check failed"); + + ESP_LOGD(TAG, "Deinit CDC %d", itf); + if (cdc_obj[itf]->type == TUSB_CLASS_CDC) { + ESP_RETURN_ON_ERROR(tusb_cdc_deinit_comm(itf), TAG, "tusb_cdc_deinit_comm failed"); + } else if (cdc_obj[itf]->type == TUSB_CLASS_CDC_DATA) { + ESP_RETURN_ON_ERROR(tusb_cdc_deinit_data(itf), TAG, "tusb_cdc_deinit_data failed"); + } else { + return ESP_ERR_INVALID_ARG; + } + return ESP_OK; +} diff --git a/examples/usb/device/usb_dongle/components/tinyusb_dongle/descriptors_control.c b/examples/usb/device/usb_dongle/components/tinyusb_dongle/descriptors_control.c new file mode 100644 index 000000000..ea6e0efdc --- /dev/null +++ b/examples/usb/device/usb_dongle/components/tinyusb_dongle/descriptors_control.c @@ -0,0 +1,284 @@ +/* + * SPDX-FileCopyrightText: 2020-2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#include +#include "esp_log.h" +#include "esp_check.h" +#include "esp_err.h" +#include "descriptors_control.h" +#include "usb_descriptors.h" + +#define MAX_DESC_BUF_SIZE 32 // Max length of string descriptor (can be extended, USB supports lengths up to 255 bytes) + +static const char *TAG = "tusb_desc"; + +// ============================================================================= +// STRUCTS +// ============================================================================= + +/** + * @brief Descriptor pointers for tinyusb descriptor requests callbacks + * + */ +typedef struct { + const tusb_desc_device_t *dev; /*!< Pointer to device descriptor */ + union { + const uint8_t *cfg; /*!< Pointer to FullSpeed configuration descriptor when device one-speed only */ + const uint8_t *fs_cfg; /*!< Pointer to FullSpeed configuration descriptor when device support HighSpeed */ + }; +#if (TUD_OPT_HIGH_SPEED) + const uint8_t *hs_cfg; /*!< Pointer to HighSpeed configuration descriptor */ + const tusb_desc_device_qualifier_t *qualifier; /*!< Pointer to Qualifier descriptor */ + uint8_t *other_speed; /*!< Pointer for other speed configuration descriptor */ +#endif // TUD_OPT_HIGH_SPEED + const char *str[USB_STRING_DESCRIPTOR_ARRAY_SIZE]; /*!< Pointer to array of UTF-8 strings */ + int str_count; /*!< Number of descriptors in str */ +} tinyusb_descriptor_config_t; + +static tinyusb_descriptor_config_t s_desc_cfg; + +// ============================================================================= +// CALLBACKS +// ============================================================================= + +/** + * @brief Invoked when received GET DEVICE DESCRIPTOR. + * Descriptor contents must exist long enough for transfer to complete + * + * @return Pointer to device descriptor + */ +uint8_t const *tud_descriptor_device_cb(void) +{ + assert(s_desc_cfg.dev); + return (uint8_t const *)s_desc_cfg.dev; +} + +/** + * @brief Invoked when received GET CONFIGURATION DESCRIPTOR. + * Descriptor contents must exist long enough for transfer to complete + * + * @param[in] index Index of required configuration + * @return Pointer to configuration descriptor + */ +uint8_t const *tud_descriptor_configuration_cb(uint8_t index) +{ + (void)index; // Unused, this driver supports only 1 configuration + assert(s_desc_cfg.cfg); + +#if (TUD_OPT_HIGH_SPEED) + // HINT: cfg and fs_cfg are union, no need to assert(fs_cfg) + assert(s_desc_cfg.hs_cfg); + // Return configuration descriptor based on Host speed + return (TUSB_SPEED_HIGH == tud_speed_get()) + ? s_desc_cfg.hs_cfg + : s_desc_cfg.fs_cfg; +#else + return s_desc_cfg.cfg; +#endif // TUD_OPT_HIGH_SPEED +} + +#if (TUD_OPT_HIGH_SPEED) +/** + * @brief Invoked when received GET DEVICE QUALIFIER DESCRIPTOR request + * Descriptor contents must exist long enough for transfer to complete + * If not highspeed capable stall this request + */ +uint8_t const *tud_descriptor_device_qualifier_cb(void) +{ + assert(s_desc_cfg.qualifier); + return (uint8_t const *)s_desc_cfg.qualifier; +} + +/** + * @brief Invoked when received GET OTHER SPEED CONFIGURATION DESCRIPTOR request + * Descriptor contents must exist long enough for transfer to complete + * Configuration descriptor in the other speed e.g if high speed then this is for full speed and vice versa + */ +uint8_t const *tud_descriptor_other_speed_configuration_cb(uint8_t index) +{ + assert(s_desc_cfg.other_speed); + + const uint8_t *other_speed = (TUSB_SPEED_HIGH == tud_speed_get()) + ? s_desc_cfg.fs_cfg + : s_desc_cfg.hs_cfg; + + memcpy(s_desc_cfg.other_speed, + other_speed, + ((tusb_desc_configuration_t *)other_speed)->wTotalLength); + + ((tusb_desc_configuration_t *)s_desc_cfg.other_speed)->bDescriptorType = TUSB_DESC_OTHER_SPEED_CONFIG; + return s_desc_cfg.other_speed; +} +#endif // TUD_OPT_HIGH_SPEED + +/** + * @brief Invoked when received GET STRING DESCRIPTOR request + * + * @param[in] index Index of required descriptor + * @param[in] langid Language of the descriptor + * @return Pointer to UTF-16 string descriptor + */ +uint16_t const *tud_descriptor_string_cb(uint8_t index, uint16_t langid) +{ + (void) langid; // Unused, this driver supports only one language in string descriptors + assert(s_desc_cfg.str); + uint8_t chr_count; + static uint16_t _desc_str[MAX_DESC_BUF_SIZE]; + + if (index == 0) { + memcpy(&_desc_str[1], s_desc_cfg.str[0], 2); + chr_count = 1; + } else { + if (index >= USB_STRING_DESCRIPTOR_ARRAY_SIZE) { + ESP_LOGW(TAG, "String index (%u) is out of bounds, check your string descriptor", index); + return NULL; + } + + if (s_desc_cfg.str[index] == NULL) { + ESP_LOGW(TAG, "String index (%u) points to NULL, check your string descriptor", index); + return NULL; + } + + const char *str = s_desc_cfg.str[index]; + chr_count = strnlen(str, MAX_DESC_BUF_SIZE - 1); // Buffer len - header + + // Convert ASCII string into UTF-16 + for (uint8_t i = 0; i < chr_count; i++) { + _desc_str[1 + i] = str[i]; + } + } + + // First byte is length in bytes (including header), second byte is descriptor type (TUSB_DESC_STRING) + _desc_str[0] = (TUSB_DESC_STRING << 8) | (2 * chr_count + 2); + + return _desc_str; +} + +// ============================================================================= +// Driver functions +// ============================================================================= +esp_err_t tinyusb_set_descriptors(const tinyusb_config_t *config) +{ + esp_err_t ret = ESP_FAIL; + assert(config); + const char **pstr_desc; + // Flush descriptors control struct + memset(&s_desc_cfg, 0x00, sizeof(tinyusb_descriptor_config_t)); + // Parse configuration and save descriptors's pointer + // Select Device Descriptor + if (config->device_descriptor == NULL) { + ESP_LOGW(TAG, "No Device descriptor provided, using default."); + s_desc_cfg.dev = &descriptor_dev_default; + } else { + s_desc_cfg.dev = config->device_descriptor; + } + + // Select FullSpeed configuration descriptor + if (config->configuration_descriptor == NULL) { + ESP_LOGW(TAG, "No FullSpeed configuration descriptor provided, using default."); + s_desc_cfg.cfg = descriptor_fs_cfg_default; + } else { + s_desc_cfg.cfg = config->configuration_descriptor; + } + +#if (TUD_OPT_HIGH_SPEED) + // High Speed + if (config->hs_configuration_descriptor == NULL) { + ESP_LOGW(TAG, "No HighSpeed configuration descriptor provided, using default."); + s_desc_cfg.hs_cfg = descriptor_hs_cfg_default; + } else { + s_desc_cfg.hs_cfg = config->hs_configuration_descriptor; + } + + // HS and FS cfg desc should be equal length + ESP_GOTO_ON_FALSE(((tusb_desc_configuration_t *)s_desc_cfg.hs_cfg)->wTotalLength == + ((tusb_desc_configuration_t *)s_desc_cfg.fs_cfg)->wTotalLength, + ESP_ERR_INVALID_ARG, fail, TAG, "HighSpeed and FullSpeed configuration descriptors must be same length"); + + // Qualifier Descriptor + if (config->qualifier_descriptor == NULL) { + ESP_GOTO_ON_FALSE((s_desc_cfg.dev == &descriptor_dev_default), ESP_ERR_INVALID_ARG, fail, TAG, "Qualifier descriptor must be present (Device Descriptor not default)."); + // Get default qualifier if device descriptor is default + ESP_LOGW(TAG, "No Qulifier descriptor provided, using default."); + s_desc_cfg.qualifier = &descriptor_qualifier_default; + } else { + s_desc_cfg.qualifier = config->qualifier_descriptor; + } + + // Other Speed buffer allocate + s_desc_cfg.other_speed = calloc(1, ((tusb_desc_configuration_t *)s_desc_cfg.hs_cfg)->wTotalLength); + ESP_GOTO_ON_FALSE(s_desc_cfg.other_speed, ESP_ERR_NO_MEM, fail, TAG, "Other speed memory allocation error"); +#endif // TUD_OPT_HIGH_SPEED + + // Select String Descriptors and count them + if (config->string_descriptor == NULL) { + ESP_LOGW(TAG, "No String descriptors provided, using default."); + pstr_desc = descriptor_str_default; + while (descriptor_str_default[++s_desc_cfg.str_count] != NULL); + } else { + pstr_desc = config->string_descriptor; + s_desc_cfg.str_count = (config->string_descriptor_count != 0) + ? config->string_descriptor_count + : 8; // '8' is for backward compatibility with esp_tinyusb v1.0.0. Do NOT remove! + } + + ESP_GOTO_ON_FALSE(s_desc_cfg.str_count <= USB_STRING_DESCRIPTOR_ARRAY_SIZE, ESP_ERR_NOT_SUPPORTED, fail, TAG, "String descriptors exceed limit"); + memcpy(s_desc_cfg.str, pstr_desc, s_desc_cfg.str_count * sizeof(pstr_desc[0])); + + ESP_LOGI(TAG, "\n" + "┌─────────────────────────────────┐\n" + "│ USB Device Descriptor Summary │\n" + "├───────────────────┬─────────────┤\n" + "│bDeviceClass │ %-4u │\n" + "├───────────────────┼─────────────┤\n" + "│bDeviceSubClass │ %-4u │\n" + "├───────────────────┼─────────────┤\n" + "│bDeviceProtocol │ %-4u │\n" + "├───────────────────┼─────────────┤\n" + "│bMaxPacketSize0 │ %-4u │\n" + "├───────────────────┼─────────────┤\n" + "│idVendor │ %-#10x │\n" + "├───────────────────┼─────────────┤\n" + "│idProduct │ %-#10x │\n" + "├───────────────────┼─────────────┤\n" + "│bcdDevice │ %-#10x │\n" + "├───────────────────┼─────────────┤\n" + "│iManufacturer │ %-#10x │\n" + "├───────────────────┼─────────────┤\n" + "│iProduct │ %-#10x │\n" + "├───────────────────┼─────────────┤\n" + "│iSerialNumber │ %-#10x │\n" + "├───────────────────┼─────────────┤\n" + "│bNumConfigurations │ %-#10x │\n" + "└───────────────────┴─────────────┘", + s_desc_cfg.dev->bDeviceClass, s_desc_cfg.dev->bDeviceSubClass, + s_desc_cfg.dev->bDeviceProtocol, s_desc_cfg.dev->bMaxPacketSize0, + s_desc_cfg.dev->idVendor, s_desc_cfg.dev->idProduct, s_desc_cfg.dev->bcdDevice, + s_desc_cfg.dev->iManufacturer, s_desc_cfg.dev->iProduct, s_desc_cfg.dev->iSerialNumber, + s_desc_cfg.dev->bNumConfigurations); + + return ESP_OK; + +fail: +#if (TUD_OPT_HIGH_SPEED) + free(s_desc_cfg.other_speed); +#endif // TUD_OPT_HIGH_SPEED + return ret; +} + +void tinyusb_set_str_descriptor(const char *str, int str_idx) +{ + assert(str_idx < USB_STRING_DESCRIPTOR_ARRAY_SIZE); + s_desc_cfg.str[str_idx] = str; +} + +void tinyusb_free_descriptors(void) +{ +#if (TUD_OPT_HIGH_SPEED) + assert(s_desc_cfg.other_speed); + free(s_desc_cfg.other_speed); +#endif // TUD_OPT_HIGH_SPEED +} diff --git a/examples/usb/device/usb_dongle/components/tinyusb_dongle/idf_component.yml b/examples/usb/device/usb_dongle/components/tinyusb_dongle/idf_component.yml new file mode 100644 index 000000000..93c3964b8 --- /dev/null +++ b/examples/usb/device/usb_dongle/components/tinyusb_dongle/idf_component.yml @@ -0,0 +1,10 @@ +version: 0.1.0 +targets: + - esp32s2 + - esp32s3 +dependencies: + idf: + version: '>=5.0' + tinyusb: + public: true + version: '>=0.15.0' diff --git a/examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tinyusb.h b/examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tinyusb.h new file mode 100644 index 000000000..484c86e6a --- /dev/null +++ b/examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tinyusb.h @@ -0,0 +1,75 @@ +/* + * SPDX-FileCopyrightText: 2020-2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#pragma once + +#include +#include "esp_err.h" +#include "tusb.h" +#include "tinyusb_types.h" + +#ifdef __cplusplus +extern "C" { +#endif + +/** + * @brief Configuration structure of the TinyUSB core + * + * USB specification mandates self-powered devices to monitor USB VBUS to detect connection/disconnection events. + * If you want to use this feature, connected VBUS to any free GPIO through a voltage divider or voltage comparator. + * The voltage divider output should be (0.75 * Vdd) if VBUS is 4.4V (lowest valid voltage at device port). + * The comparator thresholds should be set with hysteresis: 4.35V (falling edge) and 4.75V (raising edge). + */ +typedef struct { + union { + const tusb_desc_device_t *device_descriptor; /*!< Pointer to a device descriptor. If set to NULL, the TinyUSB device will use a default device descriptor whose values are set in Kconfig */ + const tusb_desc_device_t *descriptor __attribute__((deprecated)); /*!< Alias to `device_descriptor` for backward compatibility */ + }; + const char **string_descriptor; /*!< Pointer to array of string descriptors. If set to NULL, TinyUSB device will use a default string descriptors whose values are set in Kconfig */ + int string_descriptor_count; /*!< Number of descriptors in above array */ + bool external_phy; /*!< Should USB use an external PHY */ + union { + struct { + const uint8_t *configuration_descriptor; /*!< Pointer to a configuration descriptor. If set to NULL, TinyUSB device will use a default configuration descriptor whose values are set in Kconfig */ + }; +#if (TUD_OPT_HIGH_SPEED) + struct { + const uint8_t *fs_configuration_descriptor; /*!< Pointer to a FullSpeed configuration descriptor. If set to NULL, TinyUSB device will use a default configuration descriptor whose values are set in Kconfig */ + }; + }; + const uint8_t *hs_configuration_descriptor; /*!< Pointer to a HighSpeed configuration descriptor. If set to NULL, TinyUSB device will use a default configuration descriptor whose values are set in Kconfig */ + const tusb_desc_device_qualifier_t *qualifier_descriptor; /*!< Pointer to a qualifier descriptor */ +#else + }; +#endif // TUD_OPT_HIGH_SPEED + bool self_powered; /*!< This is a self-powered USB device. USB VBUS must be monitored. */ + int vbus_monitor_io; /*!< GPIO for VBUS monitoring. Ignored if not self_powered. */ +} tinyusb_config_t; + +/** + * @brief This is an all-in-one helper function, including: + * 1. USB device driver initialization + * 2. Descriptors preparation + * 3. TinyUSB stack initialization + * 4. Creates and start a task to handle usb events + * + * @note Don't change Custom descriptor, but if it has to be done, + * Suggest to define as follows in order to match the Interface Association Descriptor (IAD): + * bDeviceClass = TUSB_CLASS_MISC, + * bDeviceSubClass = MISC_SUBCLASS_COMMON, + * + * @param config tinyusb stack specific configuration + * @retval ESP_ERR_INVALID_ARG Install driver and tinyusb stack failed because of invalid argument + * @retval ESP_FAIL Install driver and tinyusb stack failed because of internal error + * @retval ESP_OK Install driver and tinyusb stack successfully + */ +esp_err_t tinyusb_driver_install(const tinyusb_config_t *config); + +esp_err_t tinyusb_driver_uninstall(void); + +#ifdef __cplusplus +} +#endif diff --git a/examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tinyusb_dfu_ota.h b/examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tinyusb_dfu_ota.h new file mode 100644 index 000000000..199d0ca78 --- /dev/null +++ b/examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tinyusb_dfu_ota.h @@ -0,0 +1,36 @@ +/* + * SPDX-FileCopyrightText: 2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ +#pragma once + +#ifdef __cplusplus +extern "C" { +#endif + +#include "esp_err.h" + +/** + * @brief Starts the OTA (Over-The-Air) update process. + * + * This function is used to start the OTA update process by providing the OTA write data and the amount of data read. + * + * @param ota_write_data Pointer to the OTA write data. + * @param data_read The amount of data read. + * @return `esp_err_t` indicating the status of the OTA start process. + */ +esp_err_t ota_start(uint8_t const *ota_write_data, uint32_t data_read); + +/** + * @brief Completes the OTA (Over-The-Air) update process. + * + * This function is used to complete the OTA update process. + * + * @return `esp_err_t` indicating the status of the OTA completion process. + */ +esp_err_t ota_complete(void); + +#ifdef __cplusplus +} +#endif diff --git a/examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tinyusb_net.h b/examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tinyusb_net.h new file mode 100644 index 000000000..6c57fdf97 --- /dev/null +++ b/examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tinyusb_net.h @@ -0,0 +1,99 @@ +/* + * SPDX-FileCopyrightText: 2023-2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#pragma once + +#include +#include "tinyusb_types.h" +#include "esp_err.h" +#include "sdkconfig.h" + +#if (CONFIG_TINYUSB_NET_MODE_NONE != 1) + +#ifdef __cplusplus +extern "C" { +#endif + +/** + * @brief On receive callback type + */ +typedef esp_err_t (*tusb_net_rx_cb_t)(void *buffer, uint16_t len, void *ctx); + +/** + * @brief Free Tx buffer callback type + */ +typedef void (*tusb_net_free_tx_cb_t)(void *buffer, void *ctx); + +/** + * @brief On init callback type + */ +typedef void (*tusb_net_init_cb_t)(void *ctx); + +/** + * @brief ESP TinyUSB NCM driver configuration structure + */ +typedef struct { + uint8_t mac_addr[6]; /*!< MAC address. Must be 6 bytes long. */ + tusb_net_rx_cb_t on_recv_callback; /*!< TinyUSB receive data callbeck */ + tusb_net_free_tx_cb_t free_tx_buffer; /*!< User function for freeing the Tx buffer. + * - could be NULL, if user app is responsible for freeing the buffer + * - must be used in asynchronous send mode + * - is only called if the used tinyusb_net_send...() function returns ESP_OK + * - in sync mode means that the packet was accepted by TinyUSB + * - in async mode means that the packet was queued to be processed in TinyUSB task + */ + tusb_net_init_cb_t on_init_callback; /*!< TinyUSB init network callback */ + void *user_context; /*!< User context to be passed to any of the callback */ +} tinyusb_net_config_t; + +/** + * @brief Initialize TinyUSB NET driver + * + * @param[in] usb_dev USB device to use + * @param[in] cfg Configuration of the driver + * @return esp_err_t + */ +esp_err_t tinyusb_net_init(tinyusb_usbdev_t usb_dev, const tinyusb_net_config_t *cfg); + +/** + * @brief TinyUSB NET driver send data synchronously + * + * @note It is possible to use sync and async send interchangeably. + * This function needs some synchronization primitives, so using sync mode (even once) uses more heap + * + * @param[in] buffer USB send data + * @param[in] len Send data len + * @param[in] buff_free_arg Pointer to be passed to the free_tx_buffer() callback + * @param[in] timeout Send data len + * @return ESP_OK on success == packet has been consumed by tusb and would be eventually freed + * by free_tx_buffer() callback (if non null) + * ESP_ERR_TIMEOUT on timeout + * ESP_ERR_INVALID_STATE if tusb not initialized, ESP_ERR_NO_MEM on alloc failure + */ +esp_err_t tinyusb_net_send_sync(void *buffer, uint16_t len, void *buff_free_arg, TickType_t timeout); + +/** + * @brief TinyUSB NET driver send data asynchronously + * + * @note If using asynchronous sends, you must free the buffer using free_tx_buffer() callback. + * @note It is possible to use sync and async send interchangeably. + * @note Async flavor of the send is useful when the USB stack runs faster than the caller, + * since we have no control over the transmitted packets, if they get accepted or discarded. + * + * @param[in] buffer USB send data + * @param[in] len Send data len + * @param[in] buff_free_arg Pointer to be passed to the free_tx_buffer() callback + * @return ESP_OK on success == packet has been consumed by tusb and will be freed + * by free_tx_buffer() callback (if non null) + * ESP_ERR_INVALID_STATE if tusb not initialized + */ +esp_err_t tinyusb_net_send_async(void *buffer, uint16_t len, void *buff_free_arg); + +#endif // (CONFIG_TINYUSB_NET_MODE_NONE != 1) + +#ifdef __cplusplus +} +#endif diff --git a/examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tinyusb_types.h b/examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tinyusb_types.h new file mode 100644 index 000000000..9263d108a --- /dev/null +++ b/examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tinyusb_types.h @@ -0,0 +1,21 @@ +/* + * SPDX-FileCopyrightText: 2020-2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#pragma once + +#ifdef __cplusplus +extern "C" { +#endif + +#define USB_ESPRESSIF_VID 0x303A + +typedef enum { + TINYUSB_USBDEV_0, +} tinyusb_usbdev_t; + +#ifdef __cplusplus +} +#endif diff --git a/examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tusb_bth.h b/examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tusb_bth.h new file mode 100644 index 000000000..39dc7d278 --- /dev/null +++ b/examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tusb_bth.h @@ -0,0 +1,69 @@ +/* + * SPDX-FileCopyrightText: 2020-2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#pragma once + +#ifdef __cplusplus +extern "C" { +#endif + +#include "esp_err.h" +#include "tinyusb.h" + +static inline void hci_dump_buffer(const char *prefix, uint8_t *data, uint16_t len) +{ + uint16_t i; + + if (!data || !len) { + return; + } + + if (prefix) { + printf("%s: len %d\r\n", prefix, len); + } + + for (i = 0; i < len; i++) { + if (((i % 8) == 0) && (i != 0)) { + printf("\r\n"); + } + printf("%02x ", *(data + i)); + } + + printf("\r\n"); +} + +#ifdef HCI_DUMP_BUFFER +#define HCI_DUMP_BUFFER(_prefix, _data, _len) hci_dump_buffer(_prefix, _data, _len) +#else +#define HCI_DUMP_BUFFER(_prefix, _data, _len) +#endif + +/* HCI message type definitions (for H4 messages) */ +#define HCIT_TYPE_COMMAND 1 +#define HCIT_TYPE_ACL_DATA 2 +#define HCIT_TYPE_SCO_DATA 3 +#define HCIT_TYPE_EVENT 4 + +typedef struct acl_data { + bool is_new_pkt; + uint16_t pkt_total_len; + uint16_t pkt_cur_offset; + uint8_t * pkt_val; +} acl_data_t; + +/** + * @brief Initialize BTH Device. + */ +void tusb_bth_init(void); + +/** + * @brief Deinitialization BTH Device. + */ +void tusb_bth_deinit(void); + +#ifdef __cplusplus +} +#endif diff --git a/examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tusb_cdc_acm.h b/examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tusb_cdc_acm.h new file mode 100644 index 000000000..ac28db8f3 --- /dev/null +++ b/examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tusb_cdc_acm.h @@ -0,0 +1,205 @@ +/* + * SPDX-FileCopyrightText: 2020-2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#pragma once + +#ifdef __cplusplus +extern "C" { +#endif + +#include +#include "sdkconfig.h" +#include "esp_err.h" + +#include "tinyusb_types.h" +#include "class/cdc/cdc.h" + +#if (CONFIG_TINYUSB_CDC_ENABLED != 1) +#error "TinyUSB CDC driver must be enabled in menuconfig" +#endif + +/** + * @brief CDC ports available to setup + */ +typedef enum { + TINYUSB_CDC_ACM_0 = 0x0, + TINYUSB_CDC_ACM_1, + TINYUSB_CDC_ACM_MAX +} tinyusb_cdcacm_itf_t; + +/* Callbacks and events + ********************************************************************* */ + +/** + * @brief Data provided to the input of the `callback_rx_wanted_char` callback + */ +typedef struct { + char wanted_char; /*!< Wanted character */ +} cdcacm_event_rx_wanted_char_data_t; + +/** + * @brief Data provided to the input of the `callback_line_state_changed` callback + */ +typedef struct { + bool dtr; /*!< Data Terminal Ready (DTR) line state */ + bool rts; /*!< Request To Send (RTS) line state */ +} cdcacm_event_line_state_changed_data_t; + +/** + * @brief Data provided to the input of the `line_coding_changed` callback + */ +typedef struct { + cdc_line_coding_t const *p_line_coding; /*!< New line coding value */ +} cdcacm_event_line_coding_changed_data_t; + +/** + * @brief Types of CDC ACM events + */ +typedef enum { + CDC_EVENT_RX, + CDC_EVENT_RX_WANTED_CHAR, + CDC_EVENT_LINE_STATE_CHANGED, + CDC_EVENT_LINE_CODING_CHANGED +} cdcacm_event_type_t; + +/** + * @brief Describes an event passing to the input of a callbacks + */ +typedef struct { + cdcacm_event_type_t type; /*!< Event type */ + union { + cdcacm_event_rx_wanted_char_data_t rx_wanted_char_data; /*!< Data input of the `callback_rx_wanted_char` callback */ + cdcacm_event_line_state_changed_data_t line_state_changed_data; /*!< Data input of the `callback_line_state_changed` callback */ + cdcacm_event_line_coding_changed_data_t line_coding_changed_data; /*!< Data input of the `line_coding_changed` callback */ + }; +} cdcacm_event_t; + +/** + * @brief CDC-ACM callback type + */ +typedef void(*tusb_cdcacm_callback_t)(int itf, cdcacm_event_t *event); + +/*********************************************************************** Callbacks and events*/ +/* Other structs + ********************************************************************* */ + +/** + * @brief Configuration structure for CDC-ACM + */ +typedef struct { + tinyusb_usbdev_t usb_dev; /*!< Usb device to set up */ + tinyusb_cdcacm_itf_t cdc_port; /*!< CDC port */ + size_t rx_unread_buf_sz __attribute__((deprecated("This parameter is not used any more. Configure RX buffer in menuconfig."))); + tusb_cdcacm_callback_t callback_rx; /*!< Pointer to the function with the `tusb_cdcacm_callback_t` type that will be handled as a callback */ + tusb_cdcacm_callback_t callback_rx_wanted_char; /*!< Pointer to the function with the `tusb_cdcacm_callback_t` type that will be handled as a callback */ + tusb_cdcacm_callback_t callback_line_state_changed; /*!< Pointer to the function with the `tusb_cdcacm_callback_t` type that will be handled as a callback */ + tusb_cdcacm_callback_t callback_line_coding_changed; /*!< Pointer to the function with the `tusb_cdcacm_callback_t` type that will be handled as a callback */ +} tinyusb_config_cdcacm_t; + +/*********************************************************************** Other structs*/ +/* Public functions + ********************************************************************* */ +/** + * @brief Initialize CDC ACM. Initialization will be finished with + * the `tud_cdc_line_state_cb` callback + * + * @param[in] cfg Configuration structure + * @return esp_err_t + */ +esp_err_t tusb_cdc_acm_init(const tinyusb_config_cdcacm_t *cfg); + +/** + * @brief De-initialize CDC ACM. + * + * @param[in] itf Index of CDC interface + * @return esp_err_t + */ +esp_err_t tusb_cdc_acm_deinit(int itf); + +/** + * @brief Register a callback invoking on CDC event. If the callback had been + * already registered, it will be overwritten + * + * @param[in] itf Index of CDC interface + * @param[in] event_type Type of registered event for a callback + * @param[in] callback Callback function + * @return esp_err_t - ESP_OK or ESP_ERR_INVALID_ARG + */ +esp_err_t tinyusb_cdcacm_register_callback(tinyusb_cdcacm_itf_t itf, + cdcacm_event_type_t event_type, + tusb_cdcacm_callback_t callback); + +/** + * @brief Unregister a callback invoking on CDC event + * + * @param[in] itf Index of CDC interface + * @param[in] event_type Type of registered event for a callback + * @return esp_err_t - ESP_OK or ESP_ERR_INVALID_ARG + */ +esp_err_t tinyusb_cdcacm_unregister_callback(tinyusb_cdcacm_itf_t itf, cdcacm_event_type_t event_type); + +/** + * @brief Sent one character to a write buffer + * + * @param[in] itf Index of CDC interface + * @param[in] ch Character to send + * @return size_t - amount of queued bytes + */ +size_t tinyusb_cdcacm_write_queue_char(tinyusb_cdcacm_itf_t itf, char ch); + +/** + * @brief Write data to write buffer + * + * @param[in] itf Index of CDC interface + * @param[in] in_buf Data + * @param[in] in_size Data size in bytes + * @return size_t - amount of queued bytes + */ +size_t tinyusb_cdcacm_write_queue(tinyusb_cdcacm_itf_t itf, const uint8_t *in_buf, size_t in_size); + +/** + * @brief Flush data in write buffer of CDC interface + * + * Use `tinyusb_cdcacm_write_queue` to add data to the buffer + * + * WARNING! TinyUSB can block output Endpoint for several RX callbacks, after will do additional flush + * after the each transfer. That can leads to the situation when you requested a flush, but it will fail until + * one of the next callbacks ends. + * SO USING OF THE FLUSH WITH TIMEOUTS IN CALLBACKS IS NOT RECOMMENDED - YOU CAN GET A LOCK FOR THE TIMEOUT + * + * @param[in] itf Index of CDC interface + * @param[in] timeout_ticks Transfer timeout. Set to zero for non-blocking mode + * @return - ESP_OK All data flushed + * - ESP_ERR_TIMEOUT Time out occurred in blocking mode + * - ESP_NOT_FINISHED The transfer is still in progress in non-blocking mode + */ +esp_err_t tinyusb_cdcacm_write_flush(tinyusb_cdcacm_itf_t itf, uint32_t timeout_ticks); + +/** + * @brief Receive data from CDC interface + * + * @param[in] itf Index of CDC interface + * @param[out] out_buf Data buffer + * @param[in] out_buf_sz Data buffer size in bytes + * @param[out] rx_data_size Number of bytes written to out_buf + * @return esp_err_t ESP_OK, ESP_FAIL or ESP_ERR_INVALID_STATE + */ +esp_err_t tinyusb_cdcacm_read(tinyusb_cdcacm_itf_t itf, uint8_t *out_buf, size_t out_buf_sz, size_t *rx_data_size); + +/** + * @brief Check if the CDC interface is initialized + * + * @param[in] itf Index of CDC interface + * @return - true Initialized + * - false Not Initialized + */ +bool tusb_cdc_acm_initialized(tinyusb_cdcacm_itf_t itf); + +/*********************************************************************** Public functions*/ + +#ifdef __cplusplus +} +#endif diff --git a/examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tusb_config.h b/examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tusb_config.h new file mode 100644 index 000000000..7a0f11326 --- /dev/null +++ b/examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tusb_config.h @@ -0,0 +1,136 @@ +/* + * SPDX-FileCopyrightText: 2019 Ha Thach (tinyusb.org), + * SPDX-FileContributor: 2020-2024 Espressif Systems (Shanghai) CO LTD + * SPDX-License-Identifier: MIT + * + * Copyright (c) 2019 Ha Thach (tinyusb.org), + * Additions Copyright (c) 2020, Espressif Systems (Shanghai) PTE LTD + * + * 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. + * + */ + +#pragma once + +#include "tusb_option.h" +#include "sdkconfig.h" + +#ifdef __cplusplus +extern "C" { +#endif + +#ifndef CONFIG_TINYUSB_CDC_ENABLED +# define CONFIG_TINYUSB_CDC_ENABLED 0 +#endif + +#ifndef CONFIG_TINYUSB_CDC_COUNT +# define CONFIG_TINYUSB_CDC_COUNT 0 +#endif + +#ifndef CONFIG_TINYUSB_NET_MODE_ECM +# define CONFIG_TINYUSB_NET_MODE_ECM 0 +#endif + +#ifndef CONFIG_TINYUSB_NET_MODE_RNDIS +# define CONFIG_TINYUSB_NET_MODE_RNDIS 0 +#endif + +#ifndef CONFIG_TINYUSB_NET_MODE_NCM +# define CONFIG_TINYUSB_NET_MODE_NCM 0 +#endif + +#ifndef CONFIG_TINYUSB_DFU_ENABLED +# define CONFIG_TINYUSB_DFU_ENABLED 0 +# define CONFIG_TINYUSB_DFU_BUFSIZE 512 +#endif + +#ifndef CONFIG_TINYUSB_BTH_ENABLED +# define CONFIG_TINYUSB_BTH_ENABLED 0 +# define CONFIG_TINYUSB_BTH_ISO_ALT_COUNT 0 +#endif + +#ifndef CONFIG_TINYUSB_DEBUG_LEVEL +# define CONFIG_TINYUSB_DEBUG_LEVEL 0 +#endif + +#ifdef CONFIG_TINYUSB_RHPORT_HS +# define CFG_TUSB_RHPORT1_MODE OPT_MODE_DEVICE | OPT_MODE_HIGH_SPEED +#else +# define CFG_TUSB_RHPORT0_MODE OPT_MODE_DEVICE | OPT_MODE_FULL_SPEED +#endif + +#define CFG_TUSB_OS OPT_OS_FREERTOS + +// Espressif IDF requires "freertos/" prefix in include path +#if TU_CHECK_MCU(OPT_MCU_ESP32S2, OPT_MCU_ESP32S3) +#define CFG_TUSB_OS_INC_PATH freertos/ +#endif + +#ifndef ESP_PLATFORM +#define ESP_PLATFORM 1 +#endif + +/* USB DMA on some MCUs can only access a specific SRAM region with restriction on alignment. + * Tinyusb use follows macros to declare transferring memory so that they can be put + * into those specific section. + * e.g + * - CFG_TUSB_MEM SECTION : __attribute__ (( section(".usb_ram") )) + * - CFG_TUSB_MEM_ALIGN : __attribute__ ((aligned(4))) + */ +#ifndef CFG_TUSB_MEM_SECTION +# define CFG_TUSB_MEM_SECTION +#endif + +#ifndef CFG_TUSB_MEM_ALIGN +# define CFG_TUSB_MEM_ALIGN TU_ATTR_ALIGNED(4) +#endif + +#ifndef CFG_TUD_ENDPOINT0_SIZE +#define CFG_TUD_ENDPOINT0_SIZE 64 +#endif + +// Debug Level +#define CFG_TUSB_DEBUG CONFIG_TINYUSB_DEBUG_LEVEL + +// CDC FIFO size of TX and RX +#define CFG_TUD_CDC_RX_BUFSIZE CONFIG_TINYUSB_CDC_RX_BUFSIZE +#define CFG_TUD_CDC_TX_BUFSIZE CONFIG_TINYUSB_CDC_TX_BUFSIZE + +// Vendor FIFO size of TX and RX +// If not configured vendor endpoints will not be buffered +#define CFG_TUD_VENDOR_RX_BUFSIZE 64 +#define CFG_TUD_VENDOR_TX_BUFSIZE 64 + +// DFU macros +#define CFG_TUD_DFU_XFER_BUFSIZE CONFIG_TINYUSB_DFU_BUFSIZE + +// Number of BTH ISO alternatives +#define CFG_TUD_BTH_ISO_ALT_COUNT CONFIG_TINYUSB_BTH_ISO_ALT_COUNT + +// Enabled device class driver +#define CFG_TUD_CDC CONFIG_TINYUSB_CDC_COUNT +#define CFG_TUD_CUSTOM_CLASS CONFIG_TINYUSB_CUSTOM_CLASS_ENABLED +#define CFG_TUD_ECM_RNDIS (CONFIG_TINYUSB_NET_MODE_ECM || CONFIG_TINYUSB_NET_MODE_RNDIS) +#define CFG_TUD_NCM CONFIG_TINYUSB_NET_MODE_NCM +#define CFG_TUD_DFU CONFIG_TINYUSB_DFU_ENABLED +#define CFG_TUD_BTH CONFIG_TINYUSB_BTH_ENABLED + +#ifdef __cplusplus +} +#endif diff --git a/examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tusb_console.h b/examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tusb_console.h new file mode 100644 index 000000000..eaf8fc0af --- /dev/null +++ b/examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tusb_console.h @@ -0,0 +1,33 @@ +/* + * SPDX-FileCopyrightText: 2020-2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#pragma once + +#ifdef __cplusplus +extern "C" { +#endif + +#include "esp_err.h" + +/** + * @brief Redirect output to the USB serial + * @param cdc_intf - interface number of TinyUSB's CDC + * + * @return esp_err_t - ESP_OK, ESP_FAIL or an error code + */ +esp_err_t esp_tusb_init_console(int cdc_intf); + +/** + * @brief Switch log to the default output + * @param cdc_intf - interface number of TinyUSB's CDC + * + * @return esp_err_t + */ +esp_err_t esp_tusb_deinit_console(int cdc_intf); + +#ifdef __cplusplus +} +#endif diff --git a/examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tusb_tasks.h b/examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tusb_tasks.h new file mode 100644 index 000000000..1ad8c20f4 --- /dev/null +++ b/examples/usb/device/usb_dongle/components/tinyusb_dongle/include/tusb_tasks.h @@ -0,0 +1,39 @@ +/* + * SPDX-FileCopyrightText: 2020-2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#pragma once + +#include "esp_err.h" + +#ifdef __cplusplus +extern "C" { +#endif + +/** + * @brief This helper function creates and starts a task which wraps `tud_task()`. + * + * The wrapper function basically wraps tud_task and some log. + * Default parameters: stack size and priority as configured, argument = NULL, not pinned to any core. + * If you have more requirements for this task, you can create your own task which calls tud_task as the last step. + * + * @retval ESP_OK run tinyusb main task successfully + * @retval ESP_FAIL run tinyusb main task failed of internal error or initialization within the task failed when TINYUSB_INIT_IN_DEFAULT_TASK=y + * @retval ESP_FAIL initialization within the task failed if CONFIG_TINYUSB_INIT_IN_DEFAULT_TASK is enabled + * @retval ESP_ERR_INVALID_STATE tinyusb main task has been created before + */ +esp_err_t tusb_run_task(void); + +/** + * @brief This helper function stops and destroys the task created by `tusb_run_task()` + * + * @retval ESP_OK stop and destroy tinyusb main task successfully + * @retval ESP_ERR_INVALID_STATE tinyusb main task hasn't been created yet + */ +esp_err_t tusb_stop_task(void); + +#ifdef __cplusplus +} +#endif diff --git a/examples/usb/device/usb_dongle/components/tinyusb_dongle/include/vfs_tinyusb.h b/examples/usb/device/usb_dongle/components/tinyusb_dongle/include/vfs_tinyusb.h new file mode 100644 index 000000000..640b633a1 --- /dev/null +++ b/examples/usb/device/usb_dongle/components/tinyusb_dongle/include/vfs_tinyusb.h @@ -0,0 +1,69 @@ +/* + * SPDX-FileCopyrightText: 2020-2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#pragma once + +#include "esp_err.h" +#include "esp_vfs_common.h" // For esp_line_endings_t definitions + +#ifdef __cplusplus +extern "C" { +#endif + +#define VFS_TUSB_MAX_PATH 16 +#define VFS_TUSB_PATH_DEFAULT "/dev/tusb_cdc" + +/** + * @brief Register TinyUSB CDC at VFS with path + * + * Know limitation: + * In case there are multiple CDC interfaces in the system, only one of them can be registered to VFS. + * + * @param[in] cdc_intf Interface number of TinyUSB's CDC + * @param[in] path Path where the CDC will be registered, `/dev/tusb_cdc` will be used if left NULL. + * @return esp_err_t ESP_OK or ESP_FAIL + */ +esp_err_t esp_vfs_tusb_cdc_register(int cdc_intf, char const *path); + +/** + * @brief Unregister TinyUSB CDC from VFS + * + * @param[in] path Path where the CDC will be unregistered if NULL will be used `/dev/tusb_cdc` + * @return esp_err_t ESP_OK or ESP_FAIL + */ +esp_err_t esp_vfs_tusb_cdc_unregister(char const *path); + +/** + * @brief Set the line endings to sent + * + * This specifies the conversion between newlines ('\n', LF) on stdout and line + * endings sent: + * + * - ESP_LINE_ENDINGS_CRLF: convert LF to CRLF + * - ESP_LINE_ENDINGS_CR: convert LF to CR + * - ESP_LINE_ENDINGS_LF: no modification + * + * @param[in] mode line endings to send + */ +void esp_vfs_tusb_cdc_set_tx_line_endings(esp_line_endings_t mode); + +/** + * @brief Set the line endings expected to be received + * + * This specifies the conversion between line endings received and + * newlines ('\n', LF) passed into stdin: + * + * - ESP_LINE_ENDINGS_CRLF: convert CRLF to LF + * - ESP_LINE_ENDINGS_CR: convert CR to LF + * - ESP_LINE_ENDINGS_LF: no modification + * + * @param[in] mode line endings expected + */ +void esp_vfs_tusb_cdc_set_rx_line_endings(esp_line_endings_t mode); + +#ifdef __cplusplus +} +#endif diff --git a/examples/usb/device/usb_dongle/components/tinyusb_dongle/include_private/cdc.h b/examples/usb/device/usb_dongle/components/tinyusb_dongle/include_private/cdc.h new file mode 100644 index 000000000..064f54d32 --- /dev/null +++ b/examples/usb/device/usb_dongle/components/tinyusb_dongle/include_private/cdc.h @@ -0,0 +1,80 @@ +/* + * SPDX-FileCopyrightText: 2020-2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#pragma once + +#ifdef __cplusplus +extern "C" { +#endif + +#include +#include "freertos/FreeRTOS.h" +#include "freertos/semphr.h" +#include "freertos/timers.h" +#include "tusb.h" +#include "tinyusb_types.h" + +/* CDC classification + ********************************************************************* */ +typedef enum { + TINYUSB_CDC_DATA = 0x00, +} cdc_data_sublcass_type_t; // CDC120 specification + +/* Note:other classification is represented in the file components\tinyusb\tinyusb\src\class\cdc\cdc.h */ + +/*********************************************************************** CDC classification*/ +/* Structs + ********************************************************************* */ +typedef struct { + tinyusb_usbdev_t usb_dev; /*!< USB device to set up */ + tusb_class_code_t cdc_class; /*!< CDC device class : Communications or Data device */ + union { + cdc_comm_sublcass_type_t comm_subclass; /*!< Communications device subclasses: ACM, ECM, etc. */ + cdc_data_sublcass_type_t data_subclass; /*!< Data device has only one subclass.*/ + } cdc_subclass; /*!< CDC device subclass according to Class Definitions for Communications Devices the CDC v.1.20 */ +} tinyusb_config_cdc_t; /*!< Main configuration structure of a CDC device */ + +typedef struct { + tinyusb_usbdev_t usb_dev; /*!< USB device used for the instance */ + tusb_class_code_t type; + union { + cdc_comm_sublcass_type_t comm_subclass; /*!< Communications device subclasses: ACM, ECM, etc. */ + cdc_data_sublcass_type_t data_subclass; /*!< Data device has only one subclass.*/ + } cdc_subclass; /*!< CDC device subclass according to Class Definitions for Communications Devices the CDC v.1.20 */ + void *subclass_obj; /*!< Dynamically allocated subclass specific object */ +} esp_tusb_cdc_t; +/*********************************************************************** Structs*/ +/* Functions + ********************************************************************* */ +/** + * @brief Initializing CDC basic object + * @param itf - number of a CDC object + * @param cfg - CDC configuration structure + * + * @return esp_err_t ESP_OK or ESP_FAIL + */ +esp_err_t tinyusb_cdc_init(int itf, const tinyusb_config_cdc_t *cfg); + +/** + * @brief De-initializing CDC. Clean its objects + * @param itf - number of a CDC object + * @return esp_err_t ESP_OK, ESP_ERR_INVALID_ARG, ESP_ERR_INVALID_STATE + * + */ +esp_err_t tinyusb_cdc_deinit(int itf); + +/** + * @brief Return interface of a CDC device + * + * @param itf_num + * @return esp_tusb_cdc_t* pointer to the interface or (NULL) on error + */ +esp_tusb_cdc_t *tinyusb_cdc_get_intf(int itf_num); +/*********************************************************************** Functions*/ + +#ifdef __cplusplus +} +#endif diff --git a/examples/usb/device/usb_dongle/components/tinyusb_dongle/include_private/descriptors_control.h b/examples/usb/device/usb_dongle/components/tinyusb_dongle/include_private/descriptors_control.h new file mode 100644 index 000000000..fcf352226 --- /dev/null +++ b/examples/usb/device/usb_dongle/components/tinyusb_dongle/include_private/descriptors_control.h @@ -0,0 +1,47 @@ +/* + * SPDX-FileCopyrightText: 2020-2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#pragma once + +#include "tinyusb.h" + +#ifdef __cplusplus +extern "C" { +#endif + +#define USB_STRING_DESCRIPTOR_ARRAY_SIZE 8 // Max 8 string descriptors for a device. LANGID, Manufacturer, Product, Serial number + 4 user defined + +/** + * @brief Parse tinyusb configuration and prepare the device configuration pointer list to configure tinyusb driver + * + * @attention All descriptors passed to this function must exist for the duration of USB device lifetime + * + * @param[in] config tinyusb stack specific configuration + * @retval ESP_ERR_INVALID_ARG Default configuration descriptor is provided only for CDC, MSC and NCM classes + * @retval ESP_ERR_NO_MEM Memory allocation error + * @retval ESP_OK Descriptors configured without error + */ +esp_err_t tinyusb_set_descriptors(const tinyusb_config_t *config); + +/** + * @brief Set specific string descriptor + * + * @attention The descriptor passed to this function must exist for the duration of USB device lifetime + * + * @param[in] str UTF-8 string + * @param[in] str_idx String descriptor index + */ +void tinyusb_set_str_descriptor(const char *str, int str_idx); + +/** + * @brief Free memory allocated during tinyusb_set_descriptors + * + */ +void tinyusb_free_descriptors(void); + +#ifdef __cplusplus +} +#endif diff --git a/examples/usb/device/usb_dongle/components/tinyusb_dongle/include_private/usb_descriptors.h b/examples/usb/device/usb_dongle/components/tinyusb_dongle/include_private/usb_descriptors.h new file mode 100644 index 000000000..be1979ab7 --- /dev/null +++ b/examples/usb/device/usb_dongle/components/tinyusb_dongle/include_private/usb_descriptors.h @@ -0,0 +1,62 @@ +/* + * SPDX-FileCopyrightText: 2020-2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#pragma once + +#include "tusb.h" + +#ifdef __cplusplus +extern "C" { +#endif + +/** + * @brief Device descriptor generated from Kconfig + * + * This descriptor is used by default. + * The user can provide their own device descriptor via tinyusb_driver_install() call + */ +extern const tusb_desc_device_t descriptor_dev_default; + +#if (TUD_OPT_HIGH_SPEED) +/** + * @brief Qualifier Device descriptor generated from Kconfig + * + * This descriptor is used by default. + * The user can provide their own descriptor via tinyusb_driver_install() call + */ +extern const tusb_desc_device_qualifier_t descriptor_qualifier_default; +#endif // TUD_OPT_HIGH_SPEED + +/** + * @brief Array of string descriptors generated from Kconfig + * + * This descriptor is used by default. + * The user can provide their own descriptor via tinyusb_driver_install() call + */ +extern const char *descriptor_str_default[]; + +/** + * @brief FullSpeed configuration descriptor generated from Kconfig + * This descriptor is used by default. + * The user can provide their own FullSpeed configuration descriptor via tinyusb_driver_install() call + */ +extern const uint8_t descriptor_fs_cfg_default[]; + +#if (TUD_OPT_HIGH_SPEED) +/** + * @brief HighSpeed Configuration descriptor generated from Kconfig + * + * This descriptor is used by default. + * The user can provide their own HighSpeed configuration descriptor via tinyusb_driver_install() call + */ +extern const uint8_t descriptor_hs_cfg_default[]; +#endif // TUD_OPT_HIGH_SPEED + +uint8_t tusb_get_mac_string_id(void); + +#ifdef __cplusplus +} +#endif diff --git a/examples/usb/device/usb_dongle/components/tinyusb_dongle/sbom.yml b/examples/usb/device/usb_dongle/components/tinyusb_dongle/sbom.yml new file mode 100644 index 000000000..f1e805b9c --- /dev/null +++ b/examples/usb/device/usb_dongle/components/tinyusb_dongle/sbom.yml @@ -0,0 +1,2 @@ +supplier: 'Organization: Espressif Systems (Shanghai) CO LTD' +originator: 'Organization: Espressif Systems (Shanghai) CO LTD' diff --git a/examples/usb/device/usb_dongle/components/tinyusb_dongle/tinyusb.c b/examples/usb/device/usb_dongle/components/tinyusb_dongle/tinyusb.c new file mode 100644 index 000000000..fbffe4b7c --- /dev/null +++ b/examples/usb/device/usb_dongle/components/tinyusb_dongle/tinyusb.c @@ -0,0 +1,73 @@ +/* + * SPDX-FileCopyrightText: 2020-2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#include "sdkconfig.h" +#include "esp_log.h" +#include "esp_check.h" +#include "esp_err.h" +#include "esp_private/periph_ctrl.h" +#include "esp_private/usb_phy.h" +#include "soc/usb_pins.h" +#include "tinyusb.h" +#include "descriptors_control.h" +#include "tusb.h" +#include "tusb_tasks.h" + +const static char *TAG = "TinyUSB"; +static usb_phy_handle_t phy_hdl; + +esp_err_t tinyusb_driver_install(const tinyusb_config_t *config) +{ + ESP_RETURN_ON_FALSE(config, ESP_ERR_INVALID_ARG, TAG, "Config can't be NULL"); + + // Configure USB PHY + usb_phy_config_t phy_conf = { + .controller = USB_PHY_CTRL_OTG, + .otg_mode = USB_OTG_MODE_DEVICE, + }; + + // External PHY IOs config + usb_phy_ext_io_conf_t ext_io_conf = { + .vp_io_num = USBPHY_VP_NUM, + .vm_io_num = USBPHY_VM_NUM, + .rcv_io_num = USBPHY_RCV_NUM, + .oen_io_num = USBPHY_OEN_NUM, + .vpo_io_num = USBPHY_VPO_NUM, + .vmo_io_num = USBPHY_VMO_NUM, + }; + if (config->external_phy) { + phy_conf.target = USB_PHY_TARGET_EXT; + phy_conf.ext_io_conf = &ext_io_conf; + } else { + phy_conf.target = USB_PHY_TARGET_INT; + } + + // OTG IOs config + const usb_phy_otg_io_conf_t otg_io_conf = USB_PHY_SELF_POWERED_DEVICE(config->vbus_monitor_io); + if (config->self_powered) { + phy_conf.otg_io_conf = &otg_io_conf; + } + ESP_RETURN_ON_ERROR(usb_new_phy(&phy_conf, &phy_hdl), TAG, "Install USB PHY failed"); + + // Descriptors config + ESP_RETURN_ON_ERROR(tinyusb_set_descriptors(config), TAG, "Descriptors config failed"); + + // Init +#if !CONFIG_TINYUSB_INIT_IN_DEFAULT_TASK + ESP_RETURN_ON_FALSE(tusb_init(), ESP_FAIL, TAG, "Init TinyUSB stack failed"); +#endif +#if !CONFIG_TINYUSB_NO_DEFAULT_TASK + ESP_RETURN_ON_ERROR(tusb_run_task(), TAG, "Run TinyUSB task failed"); +#endif + ESP_LOGI(TAG, "TinyUSB Driver installed"); + return ESP_OK; +} + +esp_err_t tinyusb_driver_uninstall() +{ + tinyusb_free_descriptors(); + return usb_del_phy(phy_hdl); +} diff --git a/examples/usb/device/usb_dongle/components/tinyusb_dongle/tinyusb_net.c b/examples/usb/device/usb_dongle/components/tinyusb_dongle/tinyusb_net.c new file mode 100644 index 000000000..2ccbf24b7 --- /dev/null +++ b/examples/usb/device/usb_dongle/components/tinyusb_dongle/tinyusb_net.c @@ -0,0 +1,174 @@ +/* + * SPDX-FileCopyrightText: 2023-2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ +#include "freertos/FreeRTOS.h" +#include "freertos/event_groups.h" +#include "tinyusb_net.h" +#include "descriptors_control.h" +#include "usb_descriptors.h" +#include "device/usbd_pvt.h" +#include "esp_check.h" + +#define MAC_ADDR_LEN 6 + +typedef struct packet { + void *buffer; + void *buff_free_arg; + uint16_t len; + esp_err_t result; +} packet_t; + +struct tinyusb_net_handle { + bool initialized; + SemaphoreHandle_t buffer_sema; + EventGroupHandle_t tx_flags; + tusb_net_rx_cb_t rx_cb; + tusb_net_free_tx_cb_t tx_buff_free_cb; + tusb_net_init_cb_t init_cb; + char mac_str[2 * MAC_ADDR_LEN + 1]; + void *ctx; + packet_t *packet_to_send; +}; + +const static int TX_FINISHED_BIT = BIT0; +static struct tinyusb_net_handle s_net_obj = { }; +static const char *TAG = "tusb_net"; + +static void do_send_sync(void *ctx) +{ + (void) ctx; + if (xSemaphoreTake(s_net_obj.buffer_sema, 0) != pdTRUE || s_net_obj.packet_to_send == NULL) { + return; + } + + packet_t *packet = s_net_obj.packet_to_send; + if (tud_network_can_xmit(packet->len)) { + tud_network_xmit(packet, packet->len); + packet->result = ESP_OK; + } else { + packet->result = ESP_FAIL; + } + xSemaphoreGive(s_net_obj.buffer_sema); + xEventGroupSetBits(s_net_obj.tx_flags, TX_FINISHED_BIT); +} + +static void do_send_async(void *ctx) +{ + packet_t *packet = ctx; + if (tud_network_can_xmit(packet->len)) { + tud_network_xmit(packet, packet->len); + } else if (s_net_obj.tx_buff_free_cb) { + ESP_LOGW(TAG, "Packet cannot be accepted on USB interface, dropping"); + s_net_obj.tx_buff_free_cb(packet->buff_free_arg, s_net_obj.ctx); + } + free(packet); +} + +esp_err_t tinyusb_net_send_async(void *buffer, uint16_t len, void *buff_free_arg) +{ + if (!tud_ready()) { + return ESP_ERR_INVALID_STATE; + } + + packet_t *packet = calloc(1, sizeof(packet_t)); + packet->len = len; + packet->buffer = buffer; + packet->buff_free_arg = buff_free_arg; + ESP_RETURN_ON_FALSE(packet, ESP_ERR_NO_MEM, TAG, "Failed to allocate packet to send"); + usbd_defer_func(do_send_async, packet, false); + return ESP_OK; +} + +esp_err_t tinyusb_net_send_sync(void *buffer, uint16_t len, void *buff_free_arg, TickType_t timeout) +{ + if (!tud_ready()) { + return ESP_ERR_INVALID_STATE; + } + + // Lazy init the flags and semaphores, as they might not be needed (if async approach is used) + if (!s_net_obj.tx_flags) { + s_net_obj.tx_flags = xEventGroupCreate(); + ESP_RETURN_ON_FALSE(s_net_obj.tx_flags, ESP_ERR_NO_MEM, TAG, "Failed to allocate event flags"); + } + if (!s_net_obj.buffer_sema) { + s_net_obj.buffer_sema = xSemaphoreCreateBinary(); + ESP_RETURN_ON_FALSE(s_net_obj.buffer_sema, ESP_ERR_NO_MEM, TAG, "Failed to allocate buffer semaphore"); + } + + packet_t packet = { + .buffer = buffer, + .len = len, + .buff_free_arg = buff_free_arg + }; + s_net_obj.packet_to_send = &packet; + xSemaphoreGive(s_net_obj.buffer_sema); // now the packet is ready, let's mark it available to tusb send + + // to execute the send function in tinyUSB task context + usbd_defer_func(do_send_sync, NULL, false); // arg=NULL -> sync send, we keep the packet inside the object + + // wait wor completion with defined timeout + EventBits_t bits = xEventGroupWaitBits(s_net_obj.tx_flags, TX_FINISHED_BIT, pdTRUE, pdTRUE, timeout); + xSemaphoreTake(s_net_obj.buffer_sema, portMAX_DELAY); // if tusb sending already started, we have wait before ditching the packet + s_net_obj.packet_to_send = NULL; // invalidate the argument + if (bits & TX_FINISHED_BIT) { // If transaction finished, return error code + return packet.result; + } + return ESP_ERR_TIMEOUT; +} + +esp_err_t tinyusb_net_init(tinyusb_usbdev_t usb_dev, const tinyusb_net_config_t *cfg) +{ + (void) usb_dev; + + ESP_RETURN_ON_FALSE(s_net_obj.initialized == false, ESP_ERR_INVALID_STATE, TAG, "TinyUSB Net class is already initialized"); + + // the semaphore and event flags are initialized only if needed + s_net_obj.rx_cb = cfg->on_recv_callback; + s_net_obj.init_cb = cfg->on_init_callback; + s_net_obj.tx_buff_free_cb = cfg->free_tx_buffer; + s_net_obj.ctx = cfg->user_context; + + const uint8_t *mac = &cfg->mac_addr[0]; + snprintf(s_net_obj.mac_str, sizeof(s_net_obj.mac_str), "%02X%02X%02X%02X%02X%02X", + mac[0], mac[1], mac[2], mac[3], mac[4], mac[5]); + uint8_t mac_id = tusb_get_mac_string_id(); + // Pass it to Descriptor control module + tinyusb_set_str_descriptor(s_net_obj.mac_str, mac_id); + + s_net_obj.initialized = true; + + return ESP_OK; +} + +//--------------------------------------------------------------------+ +// tinyusb callbacks +//--------------------------------------------------------------------+ +bool tud_network_recv_cb(const uint8_t *src, uint16_t size) +{ + if (s_net_obj.rx_cb) { + s_net_obj.rx_cb((void *)src, size, s_net_obj.ctx); + } + tud_network_recv_renew(); + return true; +} + +uint16_t tud_network_xmit_cb(uint8_t *dst, void *ref, uint16_t arg) +{ + packet_t *packet = ref; + uint16_t len = arg; + + memcpy(dst, packet->buffer, packet->len); + if (s_net_obj.tx_buff_free_cb) { + s_net_obj.tx_buff_free_cb(packet->buff_free_arg, s_net_obj.ctx); + } + return len; +} + +void tud_network_init_cb(void) +{ + if (s_net_obj.init_cb) { + s_net_obj.init_cb(s_net_obj.ctx); + } +} diff --git a/examples/usb/device/usb_dongle/components/tinyusb_dongle/tusb_bth.c b/examples/usb/device/usb_dongle/components/tinyusb_dongle/tusb_bth.c new file mode 100644 index 000000000..ec2236fc5 --- /dev/null +++ b/examples/usb/device/usb_dongle/components/tinyusb_dongle/tusb_bth.c @@ -0,0 +1,266 @@ +/* + * SPDX-FileCopyrightText: 2020-2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#include "esp_log.h" +#include "freertos/FreeRTOS.h" +#include "freertos/semphr.h" +#include "esp_bt.h" + +#include "tinyusb.h" +#include "tusb_bth.h" + +static const char *TAG = "tusb_bth"; + +static SemaphoreHandle_t evt_sem = NULL; +static SemaphoreHandle_t acl_sem = NULL; + +static acl_data_t acl_tx_data = { + .is_new_pkt = true, + .pkt_total_len = 0, + .pkt_cur_offset = 0, + .pkt_val = NULL +}; + +/* + * @brief: BT controller callback function, used to notify the upper layer that + * controller is ready to receive command + */ +static void controller_rcv_pkt_ready(void) +{ + // Do nothing, will check by esp_vhci_host_check_send_available(). +} + +/* + * @brief: BT controller callback function, to transfer data packet to upper + * controller is ready to receive command + */ +static int host_rcv_pkt(uint8_t *data, uint16_t len) +{ + uint16_t act_len = len - 1; + uint8_t type = data[0]; + uint8_t * acl_rx_buf = NULL; + uint8_t * evt_rx_buf = NULL; + + HCI_DUMP_BUFFER("Recv Pkt", data, len); + + switch (type) { + case HCIT_TYPE_COMMAND: + break; + case HCIT_TYPE_ACL_DATA: + acl_rx_buf = (uint8_t *)malloc(act_len * sizeof(uint8_t)); + assert(acl_rx_buf); + + memcpy(acl_rx_buf, data + 1, act_len); + tud_bt_acl_data_send(acl_rx_buf, act_len); + + xSemaphoreTake(acl_sem, portMAX_DELAY); + free(acl_rx_buf); + break; + case HCIT_TYPE_SCO_DATA: + break; + case HCIT_TYPE_EVENT: + evt_rx_buf = (uint8_t *)malloc(act_len * sizeof(uint8_t)); + assert(evt_rx_buf); + + memcpy(evt_rx_buf, data + 1, act_len); + tud_bt_event_send(evt_rx_buf, act_len); + + xSemaphoreTake(evt_sem, portMAX_DELAY); + free(evt_rx_buf); + break; + default: + ESP_LOGE(TAG, "Unknow type [0x%02x]", type); + break; + } + + return 0; +} + +//--------------------------------------------------------------------+ +// tinyusb callbacks +//--------------------------------------------------------------------+ + +// Invoked when HCI command was received over USB from Bluetooth host. +// Detailed format is described in Bluetooth core specification Vol 2, +// Part E, 5.4.1. +// Length of the command is from 3 bytes (2 bytes for OpCode, +// 1 byte for parameter total length) to 258. +void tud_bt_hci_cmd_cb(void * hci_cmd, size_t cmd_len) +{ + uint8_t * cmd_tx_buf = (uint8_t *)malloc((cmd_len + 1) * sizeof(uint8_t)); + + assert(cmd_tx_buf); + + cmd_tx_buf[0] = HCIT_TYPE_COMMAND; + memcpy(cmd_tx_buf + 1, hci_cmd, cmd_len); + + HCI_DUMP_BUFFER("Transmit Pkt", cmd_tx_buf, cmd_len + 1); + + while (!esp_vhci_host_check_send_available()) { + vTaskDelay(1); + } + esp_vhci_host_send_packet(cmd_tx_buf, cmd_len + 1); + + free(cmd_tx_buf); + cmd_tx_buf = NULL; +} + +// Invoked when ACL data was received over USB from Bluetooth host. +// Detailed format is described in Bluetooth core specification Vol 2, +// Part E, 5.4.2. +// Length is from 4 bytes, (12 bits for Handle, 4 bits for flags +// and 16 bits for data total length) to endpoint size. +void tud_bt_acl_data_received_cb(void *acl_data, uint16_t data_len) +{ + if (acl_tx_data.is_new_pkt) { + acl_tx_data.pkt_total_len = *(((uint16_t *)acl_data) + 1) + 4; + acl_tx_data.pkt_cur_offset = 0; + + acl_tx_data.pkt_val = (uint8_t *)malloc((acl_tx_data.pkt_total_len + 1) * sizeof(uint8_t)); + assert(acl_tx_data.pkt_val); + memset(acl_tx_data.pkt_val, 0x0, (acl_tx_data.pkt_total_len + 1)); + + acl_tx_data.pkt_val[0] = HCIT_TYPE_ACL_DATA; + acl_tx_data.pkt_cur_offset++; + memcpy(acl_tx_data.pkt_val + acl_tx_data.pkt_cur_offset, acl_data, data_len); + acl_tx_data.pkt_cur_offset += data_len; + + if (data_len < acl_tx_data.pkt_total_len) { + acl_tx_data.is_new_pkt = false; + return; + } + + while (!esp_vhci_host_check_send_available()) { + vTaskDelay(1); + } + + HCI_DUMP_BUFFER("Transmit Pkt", acl_tx_data.pkt_val, data_len + 1); + esp_vhci_host_send_packet(acl_tx_data.pkt_val, data_len + 1); + goto reset_params; + } else { + memcpy(acl_tx_data.pkt_val + acl_tx_data.pkt_cur_offset, acl_data, data_len); + acl_tx_data.pkt_cur_offset += data_len; + + if ((acl_tx_data.pkt_cur_offset - 1) == acl_tx_data.pkt_total_len) { + while (!esp_vhci_host_check_send_available()) { + vTaskDelay(1); + } + + HCI_DUMP_BUFFER("Transmit Pkt", acl_tx_data.pkt_val, acl_tx_data.pkt_total_len + 1); + esp_vhci_host_send_packet(acl_tx_data.pkt_val, acl_tx_data.pkt_total_len + 1); + goto reset_params; + } + } + + return; + +reset_params: + + acl_tx_data.is_new_pkt = true; + acl_tx_data.pkt_total_len = 0; + acl_tx_data.pkt_cur_offset = 0; + + if (acl_tx_data.pkt_val) { + free(acl_tx_data.pkt_val); + acl_tx_data.pkt_val = NULL; + } + + return; +} + +// Called when event sent with tud_bt_event_send() was delivered to BT stack. +// Controller can release/reuse buffer with Event packet at this point. +void tud_bt_event_sent_cb(uint16_t sent_bytes) +{ + if (evt_sem) { + xSemaphoreGive(evt_sem); + } +} + +// Called when ACL data that was sent with tud_bt_acl_data_send() +// was delivered to BT stack. +// Controller can release/reuse buffer with ACL packet at this point. +void tud_bt_acl_data_sent_cb(uint16_t sent_bytes) +{ + if (acl_sem) { + xSemaphoreGive(acl_sem); + } +} + +static esp_vhci_host_callback_t vhci_host_cb = { + controller_rcv_pkt_ready, + host_rcv_pkt +}; + +/** + * @brief Initialize BTH Device. + */ +void tusb_bth_init(void) +{ + esp_bt_controller_config_t bt_cfg = BT_CONTROLLER_INIT_CONFIG_DEFAULT(); + + if (esp_bt_controller_mem_release(ESP_BT_MODE_CLASSIC_BT) != ESP_OK) { + ESP_LOGE(TAG, "Bluetooth controller release classic bt memory failed."); + return; + } + + if (esp_bt_controller_init(&bt_cfg) != ESP_OK) { + ESP_LOGE(TAG, "Bluetooth controller initialize failed."); + return; + } + + if (esp_bt_controller_enable(ESP_BT_MODE_BLE) != ESP_OK) { + ESP_LOGE(TAG, "Bluetooth controller enable failed."); + return; + } + + evt_sem = xSemaphoreCreateBinary(); + assert(evt_sem); + acl_sem = xSemaphoreCreateBinary(); + assert(acl_sem); + + acl_tx_data.is_new_pkt = true; + acl_tx_data.pkt_total_len = 0; + acl_tx_data.pkt_cur_offset = 0; + acl_tx_data.pkt_val = NULL; + + esp_vhci_host_register_callback(&vhci_host_cb); +} + +/** + * @brief Deinitialization BTH Device. + */ +void tusb_bth_deinit(void) +{ + if (esp_bt_controller_disable() != ESP_OK) { + ESP_LOGE(TAG, "Bluetooth controller disable failed."); + return; + } + + if (esp_bt_controller_deinit() != ESP_OK) { + ESP_LOGE(TAG, "Bluetooth controller deinit failed."); + return; + } + + if (evt_sem) { + vSemaphoreDelete(evt_sem); + evt_sem = NULL; + } + + if (acl_sem) { + vSemaphoreDelete(acl_sem); + acl_sem = NULL; + } + + acl_tx_data.is_new_pkt = true; + acl_tx_data.pkt_total_len = 0; + acl_tx_data.pkt_cur_offset = 0; + + if (acl_tx_data.pkt_val) { + free(acl_tx_data.pkt_val); + acl_tx_data.pkt_val = NULL; + } +} diff --git a/examples/usb/device/usb_dongle/components/tinyusb_dongle/tusb_cdc_acm.c b/examples/usb/device/usb_dongle/components/tinyusb_dongle/tusb_cdc_acm.c new file mode 100644 index 000000000..77211e11f --- /dev/null +++ b/examples/usb/device/usb_dongle/components/tinyusb_dongle/tusb_cdc_acm.c @@ -0,0 +1,345 @@ +/* + * SPDX-FileCopyrightText: 2020-2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#include +#include "esp_check.h" +#include "esp_err.h" +#include "esp_log.h" +#include "freertos/FreeRTOS.h" +#include "freertos/task.h" +#include "tusb.h" +#include "tusb_cdc_acm.h" +#include "cdc.h" +#include "sdkconfig.h" + +#ifndef MIN +#define MIN(x, y) (((x) < (y)) ? (x) : (y)) +#endif + +// CDC-ACM spinlock +static portMUX_TYPE cdc_acm_lock = portMUX_INITIALIZER_UNLOCKED; +#define CDC_ACM_ENTER_CRITICAL() portENTER_CRITICAL(&cdc_acm_lock) +#define CDC_ACM_EXIT_CRITICAL() portEXIT_CRITICAL(&cdc_acm_lock) + +typedef struct { + tusb_cdcacm_callback_t callback_rx; + tusb_cdcacm_callback_t callback_rx_wanted_char; + tusb_cdcacm_callback_t callback_line_state_changed; + tusb_cdcacm_callback_t callback_line_coding_changed; +} esp_tusb_cdcacm_t; /*!< CDC_ACM object */ + +static const char *TAG = "tusb_cdc_acm"; + +static inline esp_tusb_cdcacm_t *get_acm(tinyusb_cdcacm_itf_t itf) +{ + esp_tusb_cdc_t *cdc_inst = tinyusb_cdc_get_intf(itf); + if (cdc_inst == NULL) { + return (esp_tusb_cdcacm_t *)NULL; + } + return (esp_tusb_cdcacm_t *)(cdc_inst->subclass_obj); +} + +/* TinyUSB callbacks + ********************************************************************* */ + +/* Invoked by cdc interface when line state changed e.g connected/disconnected */ +void tud_cdc_line_state_cb(uint8_t itf, bool dtr, bool rts) +{ + esp_tusb_cdcacm_t *acm = get_acm(itf); + if (dtr && rts) { // connected + if (acm != NULL) { + ESP_LOGV(TAG, "Host connected to CDC no.%d.", itf); + } else { + ESP_LOGW(TAG, "Host is connected to CDC no.%d, but it is not initialized. Initialize it using `tinyusb_cdc_init`.", itf); + return; + } + } else { // disconnected + if (acm != NULL) { + ESP_LOGV(TAG, "Serial device is ready to connect to CDC no.%d", itf); + } else { + return; + } + } + if (acm) { + CDC_ACM_ENTER_CRITICAL(); + tusb_cdcacm_callback_t cb = acm->callback_line_state_changed; + CDC_ACM_EXIT_CRITICAL(); + if (cb) { + cdcacm_event_t event = { + .type = CDC_EVENT_LINE_STATE_CHANGED, + .line_state_changed_data = { + .dtr = dtr, + .rts = rts + } + }; + cb(itf, &event); + } + } +} + +/* Invoked when CDC interface received data from host */ +void tud_cdc_rx_cb(uint8_t itf) +{ + esp_tusb_cdcacm_t *acm = get_acm(itf); + if (acm) { + CDC_ACM_ENTER_CRITICAL(); + tusb_cdcacm_callback_t cb = acm->callback_rx; + CDC_ACM_EXIT_CRITICAL(); + if (cb) { + cdcacm_event_t event = { + .type = CDC_EVENT_RX + }; + cb(itf, &event); + } + } +} + +// Invoked when line coding is change via SET_LINE_CODING +void tud_cdc_line_coding_cb(uint8_t itf, cdc_line_coding_t const *p_line_coding) +{ + esp_tusb_cdcacm_t *acm = get_acm(itf); + if (acm) { + CDC_ACM_ENTER_CRITICAL(); + tusb_cdcacm_callback_t cb = acm->callback_line_coding_changed; + CDC_ACM_EXIT_CRITICAL(); + if (cb) { + cdcacm_event_t event = { + .type = CDC_EVENT_LINE_CODING_CHANGED, + .line_coding_changed_data = { + .p_line_coding = p_line_coding, + } + }; + cb(itf, &event); + } + } +} + +// Invoked when received `wanted_char` +void tud_cdc_rx_wanted_cb(uint8_t itf, char wanted_char) +{ + esp_tusb_cdcacm_t *acm = get_acm(itf); + if (acm) { + CDC_ACM_ENTER_CRITICAL(); + tusb_cdcacm_callback_t cb = acm->callback_rx_wanted_char; + CDC_ACM_EXIT_CRITICAL(); + if (cb) { + cdcacm_event_t event = { + .type = CDC_EVENT_RX_WANTED_CHAR, + .rx_wanted_char_data = { + .wanted_char = wanted_char, + } + }; + cb(itf, &event); + } + } +} + +esp_err_t tinyusb_cdcacm_register_callback(tinyusb_cdcacm_itf_t itf, + cdcacm_event_type_t event_type, + tusb_cdcacm_callback_t callback) +{ + esp_tusb_cdcacm_t *acm = get_acm(itf); + if (acm) { + switch (event_type) { + case CDC_EVENT_RX: + CDC_ACM_ENTER_CRITICAL(); + acm->callback_rx = callback; + CDC_ACM_EXIT_CRITICAL(); + return ESP_OK; + case CDC_EVENT_RX_WANTED_CHAR: + CDC_ACM_ENTER_CRITICAL(); + acm->callback_rx_wanted_char = callback; + CDC_ACM_EXIT_CRITICAL(); + return ESP_OK; + case CDC_EVENT_LINE_STATE_CHANGED: + CDC_ACM_ENTER_CRITICAL(); + acm->callback_line_state_changed = callback; + CDC_ACM_EXIT_CRITICAL(); + return ESP_OK; + case CDC_EVENT_LINE_CODING_CHANGED: + CDC_ACM_ENTER_CRITICAL(); + acm->callback_line_coding_changed = callback; + CDC_ACM_EXIT_CRITICAL(); + return ESP_OK; + default: + ESP_LOGE(TAG, "Wrong event type"); + return ESP_ERR_INVALID_ARG; + } + } else { + ESP_LOGE(TAG, "CDC-ACM is not initialized"); + return ESP_ERR_INVALID_STATE; + } +} + +esp_err_t tinyusb_cdcacm_unregister_callback(tinyusb_cdcacm_itf_t itf, + cdcacm_event_type_t event_type) +{ + esp_tusb_cdcacm_t *acm = get_acm(itf); + if (!acm) { + ESP_LOGE(TAG, "Interface is not initialized. Use `tinyusb_cdc_init` for initialization"); + return ESP_ERR_INVALID_STATE; + } + switch (event_type) { + case CDC_EVENT_RX: + CDC_ACM_ENTER_CRITICAL(); + acm->callback_rx = NULL; + CDC_ACM_EXIT_CRITICAL(); + return ESP_OK; + case CDC_EVENT_RX_WANTED_CHAR: + CDC_ACM_ENTER_CRITICAL(); + acm->callback_rx_wanted_char = NULL; + CDC_ACM_EXIT_CRITICAL(); + return ESP_OK; + case CDC_EVENT_LINE_STATE_CHANGED: + CDC_ACM_ENTER_CRITICAL(); + acm->callback_line_state_changed = NULL; + CDC_ACM_EXIT_CRITICAL(); + return ESP_OK; + case CDC_EVENT_LINE_CODING_CHANGED: + CDC_ACM_ENTER_CRITICAL(); + acm->callback_line_coding_changed = NULL; + CDC_ACM_EXIT_CRITICAL(); + return ESP_OK; + default: + ESP_LOGE(TAG, "Wrong event type"); + return ESP_ERR_INVALID_ARG; + } +} + +/*********************************************************************** TinyUSB callbacks*/ +/* CDC-ACM + ********************************************************************* */ + +esp_err_t tinyusb_cdcacm_read(tinyusb_cdcacm_itf_t itf, uint8_t *out_buf, size_t out_buf_sz, size_t *rx_data_size) +{ + esp_tusb_cdcacm_t *acm = get_acm(itf); + ESP_RETURN_ON_FALSE(acm, ESP_ERR_INVALID_STATE, TAG, "Interface is not initialized. Use `tinyusb_cdc_init` for initialization"); + + if (tud_cdc_n_available(itf) == 0) { + *rx_data_size = 0; + } else { + *rx_data_size = tud_cdc_n_read(itf, out_buf, out_buf_sz); + } + return ESP_OK; +} + +size_t tinyusb_cdcacm_write_queue_char(tinyusb_cdcacm_itf_t itf, char ch) +{ + if (!get_acm(itf)) { // non-initialized + return 0; + } + return tud_cdc_n_write_char(itf, ch); +} + +size_t tinyusb_cdcacm_write_queue(tinyusb_cdcacm_itf_t itf, const uint8_t *in_buf, size_t in_size) +{ + if (!get_acm(itf)) { // non-initialized + return 0; + } + const uint32_t size_available = tud_cdc_n_write_available(itf); + return tud_cdc_n_write(itf, in_buf, MIN(in_size, size_available)); +} + +static uint32_t tud_cdc_n_write_occupied(tinyusb_cdcacm_itf_t itf) +{ + return CFG_TUD_CDC_TX_BUFSIZE - tud_cdc_n_write_available(itf); +} + +esp_err_t tinyusb_cdcacm_write_flush(tinyusb_cdcacm_itf_t itf, uint32_t timeout_ticks) +{ + if (!get_acm(itf)) { // non-initialized + return ESP_FAIL; + } + + if (!timeout_ticks) { // if no timeout - nonblocking mode + // It might take some time until TinyUSB flushes the endpoint + // Since this call is non-blocking, we don't wait for flush finished, + // We only inform the user by returning ESP_ERR_NOT_FINISHED + tud_cdc_n_write_flush(itf); + if (tud_cdc_n_write_occupied(itf)) { + return ESP_ERR_NOT_FINISHED; + } + } else { // trying during the timeout + uint32_t ticks_start = xTaskGetTickCount(); + uint32_t ticks_now = ticks_start; + while (1) { // loop until success or until the time runs out + ticks_now = xTaskGetTickCount(); + tud_cdc_n_write_flush(itf); + if (tud_cdc_n_write_occupied(itf) == 0) { + break; // All data flushed + } + if ((ticks_now - ticks_start) > timeout_ticks) { // Time is up + ESP_LOGW(TAG, "Flush failed"); + return ESP_ERR_TIMEOUT; + } + vTaskDelay(1); + } + } + return ESP_OK; +} + +static esp_err_t alloc_obj(tinyusb_cdcacm_itf_t itf) +{ + esp_tusb_cdc_t *cdc_inst = tinyusb_cdc_get_intf(itf); + if (cdc_inst == NULL) { + return ESP_FAIL; + } + cdc_inst->subclass_obj = calloc(1, sizeof(esp_tusb_cdcacm_t)); + if (cdc_inst->subclass_obj == NULL) { + return ESP_FAIL; + } + return ESP_OK; +} + +esp_err_t tusb_cdc_acm_init(const tinyusb_config_cdcacm_t *cfg) +{ + esp_err_t ret = ESP_OK; + int itf = (int)cfg->cdc_port; + /* Creating a CDC object */ + const tinyusb_config_cdc_t cdc_cfg = { + .usb_dev = cfg->usb_dev, + .cdc_class = TUSB_CLASS_CDC, + .cdc_subclass.comm_subclass = CDC_COMM_SUBCLASS_ABSTRACT_CONTROL_MODEL + }; + + ESP_RETURN_ON_ERROR(tinyusb_cdc_init(itf, &cdc_cfg), TAG, "tinyusb_cdc_init failed"); + ESP_GOTO_ON_ERROR(alloc_obj(itf), fail, TAG, "alloc_obj failed"); + + /* Callbacks setting up*/ + if (cfg->callback_rx) { + tinyusb_cdcacm_register_callback(itf, CDC_EVENT_RX, cfg->callback_rx); + } + if (cfg->callback_rx_wanted_char) { + tinyusb_cdcacm_register_callback(itf, CDC_EVENT_RX_WANTED_CHAR, cfg->callback_rx_wanted_char); + } + if (cfg->callback_line_state_changed) { + tinyusb_cdcacm_register_callback(itf, CDC_EVENT_LINE_STATE_CHANGED, cfg->callback_line_state_changed); + } + if (cfg->callback_line_coding_changed) { + tinyusb_cdcacm_register_callback(itf, CDC_EVENT_LINE_CODING_CHANGED, cfg->callback_line_coding_changed); + } + + return ESP_OK; +fail: + tinyusb_cdc_deinit(itf); + return ret; +} + +esp_err_t tusb_cdc_acm_deinit(int itf) +{ + return tinyusb_cdc_deinit(itf); +} + +bool tusb_cdc_acm_initialized(tinyusb_cdcacm_itf_t itf) +{ + esp_tusb_cdcacm_t *acm = get_acm(itf); + if (acm) { + return true; + } else { + return false; + } +} +/*********************************************************************** CDC-ACM*/ diff --git a/examples/usb/device/usb_dongle/components/tinyusb_dongle/tusb_console.c b/examples/usb/device/usb_dongle/components/tinyusb_dongle/tusb_console.c new file mode 100644 index 000000000..60a233e23 --- /dev/null +++ b/examples/usb/device/usb_dongle/components/tinyusb_dongle/tusb_console.c @@ -0,0 +1,113 @@ +/* + * SPDX-FileCopyrightText: 2020-2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#include +#include +#include "esp_log.h" +#include "cdc.h" +#include "tusb_console.h" +#include "tinyusb.h" +#include "vfs_tinyusb.h" +#include "esp_check.h" + +#define STRINGIFY(s) STRINGIFY2(s) +#define STRINGIFY2(s) #s + +static const char *TAG = "tusb_console"; + +typedef struct { + FILE *in; + FILE *out; + FILE *err; +} console_handle_t; + +static console_handle_t con; + +/** + * @brief Reopen standard streams using a new path + * + * @param f_in - pointer to a pointer holding a file for in or NULL to don't change stdin + * @param f_out - pointer to a pointer holding a file for out or NULL to don't change stdout + * @param f_err - pointer to a pointer holding a file for err or NULL to don't change stderr + * @param path - mount point + * @return esp_err_t ESP_FAIL or ESP_OK + */ +static esp_err_t redirect_std_streams_to(FILE **f_in, FILE **f_out, FILE **f_err, const char *path) +{ + if (f_in) { + *f_in = freopen(path, "r", stdin); + if (*f_in == NULL) { + ESP_LOGE(TAG, "Failed to reopen in!"); + return ESP_FAIL; + } + } + if (f_out) { + *f_out = freopen(path, "w", stdout); + if (*f_out == NULL) { + ESP_LOGE(TAG, "Failed to reopen out!"); + return ESP_FAIL; + } + } + if (f_err) { + *f_err = freopen(path, "w", stderr); + if (*f_err == NULL) { + ESP_LOGE(TAG, "Failed to reopen err!"); + return ESP_FAIL; + } + } + + return ESP_OK; +} + +/** + * @brief Restore output to default + * + * @param f_in - pointer to a pointer of an in file updated with `redirect_std_streams_to` or NULL to don't change stdin + * @param f_out - pointer to a pointer of an out file updated with `redirect_std_streams_to` or NULL to don't change stdout + * @param f_err - pointer to a pointer of an err file updated with `redirect_std_streams_to` or NULL to don't change stderr + * @return esp_err_t ESP_FAIL or ESP_OK + */ +static esp_err_t restore_std_streams(FILE **f_in, FILE **f_out, FILE **f_err) +{ + const char *default_uart_dev = "/dev/uart/" STRINGIFY(CONFIG_ESP_CONSOLE_UART_NUM); + if (f_in) { + stdin = freopen(default_uart_dev, "r", *f_in); + if (stdin == NULL) { + ESP_LOGE(TAG, "Failed to reopen stdin!"); + return ESP_FAIL; + } + } + if (f_out) { + stdout = freopen(default_uart_dev, "w", *f_out); + if (stdout == NULL) { + ESP_LOGE(TAG, "Failed to reopen stdout!"); + return ESP_FAIL; + } + } + if (f_err) { + stderr = freopen(default_uart_dev, "w", *f_err); + if (stderr == NULL) { + ESP_LOGE(TAG, "Failed to reopen stderr!"); + return ESP_FAIL; + } + } + return ESP_OK; +} + +esp_err_t esp_tusb_init_console(int cdc_intf) +{ + /* Registering TUSB at VFS */ + ESP_RETURN_ON_ERROR(esp_vfs_tusb_cdc_register(cdc_intf, NULL), TAG, ""); + ESP_RETURN_ON_ERROR(redirect_std_streams_to(&con.in, &con.out, &con.err, VFS_TUSB_PATH_DEFAULT), TAG, "Failed to redirect STD streams"); + return ESP_OK; +} + +esp_err_t esp_tusb_deinit_console(int cdc_intf) +{ + ESP_RETURN_ON_ERROR(restore_std_streams(&con.in, &con.out, &con.err), TAG, "Failed to restore STD streams"); + esp_vfs_tusb_cdc_unregister(NULL); + return ESP_OK; +} diff --git a/examples/usb/device/usb_dongle/components/tinyusb_dongle/tusb_dfu.c b/examples/usb/device/usb_dongle/components/tinyusb_dongle/tusb_dfu.c new file mode 100644 index 000000000..99cd1d98f --- /dev/null +++ b/examples/usb/device/usb_dongle/components/tinyusb_dongle/tusb_dfu.c @@ -0,0 +1,91 @@ +/* + * SPDX-FileCopyrightText: 2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +/* + * After device is enumerated in dfu mode run the following commands + * + * To transfer firmware from host to device (best to test with text file) + * + * $ dfu-util -d cafe -a 0 -D [filename] + * $ dfu-util -d cafe -a 1 -D [filename] + * + * To transfer firmware from device to host: + * + * $ dfu-util -d cafe -a 0 -U [filename] + * $ dfu-util -d cafe -a 1 -U [filename] + * + */ + +#include +#include +#include + +#include "esp_log.h" +#include "tusb.h" +#include "tinyusb.h" +#include "tinyusb_dfu_ota.h" + +static char* TAG = "esp_dfu"; + +//--------------------------------------------------------------------+ +// DFU callbacks +// Note: alt is used as the partition number, in order to support multiple partitions like FLASH, EEPROM, etc. +//--------------------------------------------------------------------+ + +// Invoked right before tud_dfu_download_cb() (state=DFU_DNBUSY) or tud_dfu_manifest_cb() (state=DFU_MANIFEST) +// Application return timeout in milliseconds (bwPollTimeout) for the next download/manifest operation. +// During this period, USB host won't try to communicate with us. +uint32_t tud_dfu_get_timeout_cb(uint8_t alt, uint8_t state) +{ + if (state == DFU_DNBUSY) { + // For this example + // - Atl0 Flash is fast : 1 ms + // - Alt1 EEPROM is slow: 100 ms + return (alt == 0) ? 1 : 100; + } else if (state == DFU_MANIFEST) { + // since we don't buffer entire image and do any flashing in manifest stage + return 0; + } + + return 0; +} + +// Invoked when received DFU_DNLOAD (wLength>0) following by DFU_GETSTATUS (state=DFU_DNBUSY) requests +// This callback could be returned before flashing op is complete (async). +// Once finished flashing, application must call tud_dfu_finish_flashing() +void tud_dfu_download_cb(uint8_t alt, uint16_t block_num, uint8_t const* data, uint16_t length) +{ + (void) alt; + (void) block_num; + + ESP_LOGI(TAG, "Received Alt %u BlockNum %u of length %u", alt, block_num, length); + + esp_err_t ret = ota_start(data, length); + if (ret == ESP_OK) { + tud_dfu_finish_flashing(DFU_STATUS_OK); + } else { + tud_dfu_finish_flashing(DFU_STATUS_ERR_WRITE); + } +} + +// Invoked when download process is complete, received DFU_DNLOAD (wLength=0) following by DFU_GETSTATUS (state=Manifest) +// Application can do checksum, or actual flashing if buffered entire image previously. +// Once finished flashing, application must call tud_dfu_finish_flashing() +void tud_dfu_manifest_cb(uint8_t alt) +{ + (void) alt; + ESP_LOGI(TAG, "Download completed, enter manifestation\r\n"); + + esp_err_t ret = ota_complete(); + // flashing op for manifest is complete without error + // Application can perform checksum, should it fail, use appropriate status such as errVERIFY. + + if (ret == ESP_OK) { + tud_dfu_finish_flashing(DFU_STATUS_OK); + } else { + tud_dfu_finish_flashing(DFU_STATUS_ERR_VERIFY); + } +} diff --git a/examples/usb/device/usb_dongle/components/tinyusb_dongle/tusb_dfu_ota.c b/examples/usb/device/usb_dongle/components/tinyusb_dongle/tusb_dfu_ota.c new file mode 100644 index 000000000..bffaa9c25 --- /dev/null +++ b/examples/usb/device/usb_dongle/components/tinyusb_dongle/tusb_dfu_ota.c @@ -0,0 +1,207 @@ +/* + * SPDX-FileCopyrightText: 2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +/* + * After device is enumerated in dfu mode run the following commands + * + * To transfer firmware from host to device (best to test with text file) + * + * $ dfu-util -d cafe -a 0 -D [filename] + * $ dfu-util -d cafe -a 1 -D [filename] + * + * To transfer firmware from device to host: + * + * $ dfu-util -d cafe -a 0 -U [filename] + * $ dfu-util -d cafe -a 1 -U [filename] + * + */ + +#include +#include +#include + +#include "freertos/FreeRTOS.h" +#include "freertos/task.h" + +#include "tusb.h" +#include "tinyusb.h" +#include "nvs_flash.h" + +#include "esp_app_format.h" +#include "esp_system.h" +#include "esp_log.h" +#include "esp_ota_ops.h" +#include "esp_flash_partitions.h" +#include "esp_partition.h" + +static char* TAG = "esp_dfu_ota"; + +#define BUFFSIZE CONFIG_TINYUSB_DFU_BUFSIZE + +static esp_ota_handle_t update_handle = 0; +static esp_partition_t const *update_partition = NULL; +static int binary_file_length = 0; +/*deal with all receive packet*/ +static bool image_header_was_checked = false; + +static const esp_partition_t *ota_partition; +static uint32_t current_index = 0; + +esp_err_t ota_start(uint8_t const *ota_write_data, uint32_t data_read) +{ + esp_err_t err; + /* update handle : set by esp_ota_begin(), must be freed via esp_ota_end() */ + if (image_header_was_checked == false) { + esp_app_desc_t new_app_info; + + ESP_LOGI(TAG, "Starting OTA example"); + + const esp_partition_t *running = esp_ota_get_running_partition(); + + ESP_LOGI(TAG, "Running partition type %d subtype %d (offset 0x%"PRIx32")", + running->type, running->subtype, running->address); + + update_partition = esp_ota_get_next_update_partition(NULL); + assert(update_partition != NULL); + ESP_LOGI(TAG, "Writing to partition subtype %d at offset 0x%"PRIx32"", + update_partition->subtype, update_partition->address); + if (data_read > sizeof(esp_image_header_t) + sizeof(esp_image_segment_header_t) + sizeof(esp_app_desc_t)) { + // check current version with downloading + memcpy(&new_app_info, &ota_write_data[sizeof(esp_image_header_t) + sizeof(esp_image_segment_header_t)], sizeof(esp_app_desc_t)); + ESP_LOGI(TAG, "New firmware version: %s", new_app_info.version); + + esp_app_desc_t running_app_info; + if (esp_ota_get_partition_description(running, &running_app_info) == ESP_OK) { + ESP_LOGI(TAG, "Running firmware version: %s", running_app_info.version); + } + + const esp_partition_t* last_invalid_app = esp_ota_get_last_invalid_partition(); + esp_app_desc_t invalid_app_info; + if (esp_ota_get_partition_description(last_invalid_app, &invalid_app_info) == ESP_OK) { + ESP_LOGI(TAG, "Last invalid firmware version: %s", invalid_app_info.version); + } + + // check current version with last invalid partition + if (last_invalid_app != NULL) { + if (memcmp(invalid_app_info.version, new_app_info.version, sizeof(new_app_info.version)) == 0) { + ESP_LOGW(TAG, "New version is the same as invalid version."); + ESP_LOGW(TAG, "Previously, there was an attempt to launch the firmware with %s version, but it failed.", invalid_app_info.version); + ESP_LOGW(TAG, "The firmware has been rolled back to the previous version."); + } + } + + image_header_was_checked = true; + + err = esp_ota_begin(update_partition, OTA_WITH_SEQUENTIAL_WRITES, &update_handle); + if (err != ESP_OK) { + ESP_LOGE(TAG, "esp_ota_begin failed (%s)", esp_err_to_name(err)); + esp_ota_abort(update_handle); + return err; + } + ESP_LOGI(TAG, "esp_ota_begin succeeded"); + } else { + ESP_LOGE(TAG, "received package is not fit len"); + esp_ota_abort(update_handle); + return err; + } + } + err = esp_ota_write(update_handle, (const void *)ota_write_data, data_read); + if (err != ESP_OK) { + esp_ota_abort(update_handle); + return err; + } + binary_file_length += data_read; + ESP_LOGD(TAG, "Written image length %d", binary_file_length); + return err; +} + +esp_err_t ota_complete(void) +{ + esp_err_t err = ESP_OK; + ESP_LOGI(TAG, "Total Write binary data length: %d", binary_file_length); + + err = esp_ota_end(update_handle); + if (err != ESP_OK) { + if (err == ESP_ERR_OTA_VALIDATE_FAILED) { + ESP_LOGE(TAG, "Image validation failed, image is corrupted"); + } else { + ESP_LOGE(TAG, "esp_ota_end failed (%s)!", esp_err_to_name(err)); + } + return err; + } + + err = esp_ota_set_boot_partition(update_partition); + if (err != ESP_OK) { + ESP_LOGE(TAG, "esp_ota_set_boot_partition failed (%s)!", esp_err_to_name(err)); + return err; + } + ESP_LOGI(TAG, "OTA done, please restart system!"); + return ESP_OK; +} + +static uint16_t upload_bin(uint8_t* data, uint16_t length) +{ + uint16_t read_size = length; + if (ota_partition == NULL) { + ota_partition = esp_partition_find_first(ESP_PARTITION_TYPE_APP, ESP_PARTITION_SUBTYPE_APP_OTA_MIN, NULL); + if (ota_partition == NULL) { + ESP_LOGE(TAG, "not found ota_0"); + return 0; + } + ESP_LOGI(TAG, "Bin size: %"PRIu32", addr: %p", ota_partition->size, (void *)ota_partition->address); + } + if (current_index > ota_partition->size) { + ESP_LOGI(TAG, "read error %"PRIu32",%"PRIu32"\n", current_index, ota_partition->size); + return 0; + } + + if (current_index == ota_partition->size) { // upload done + ESP_LOGI(TAG, "Upload done"); + current_index = 0; + ota_partition = NULL; + return 0; + } else if (current_index + read_size > ota_partition->size) { // the last data + read_size = ota_partition->size - current_index; + } + ESP_ERROR_CHECK(esp_partition_read(ota_partition, current_index, data, read_size)); + + current_index += read_size; + + return read_size; +} + +//--------------------------------------------------------------------+ +// DFU callbacks +// Note: alt is used as the partition number, in order to support multiple partitions like FLASH, EEPROM, etc. +//--------------------------------------------------------------------+ + +// Invoked when received DFU_UPLOAD request +// Application must populate data with up to length bytes and +// Return the number of written bytes +uint16_t tud_dfu_upload_cb(uint8_t alt, uint16_t block_num, uint8_t* data, uint16_t length) +{ + (void) block_num; + (void) length; + ESP_LOGI(TAG, "upload data,block_num: %d,length: %d\n", block_num, length); + + uint16_t xfer_len = upload_bin(data, length); + + return xfer_len; +} + +// Invoked when the Host has terminated a download or upload transfer +void tud_dfu_abort_cb(uint8_t alt) +{ + (void) alt; + ESP_LOGI(TAG, "Host aborted transfer\r\n"); +} + +// Invoked when a DFU_DETACH request is received +void tud_dfu_detach_cb(void) +{ + ESP_LOGI(TAG, "Host detach, upload/download done\r\n"); + //esp_restart(); +} diff --git a/examples/usb/device/usb_dongle/components/tinyusb_dongle/tusb_tasks.c b/examples/usb/device/usb_dongle/components/tinyusb_dongle/tusb_tasks.c new file mode 100644 index 000000000..051dd677d --- /dev/null +++ b/examples/usb/device/usb_dongle/components/tinyusb_dongle/tusb_tasks.c @@ -0,0 +1,77 @@ +/* + * SPDX-FileCopyrightText: 2020-2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#include "sdkconfig.h" +#include "freertos/FreeRTOS.h" +#include "freertos/task.h" +#include "freertos/event_groups.h" +#include "esp_log.h" +#include "esp_check.h" +#include "tinyusb.h" +#include "tusb_tasks.h" + +const static char *TAG = "tusb_tsk"; +static TaskHandle_t s_tusb_tskh; + +#if CONFIG_TINYUSB_INIT_IN_DEFAULT_TASK +const static int INIT_OK = BIT0; +const static int INIT_FAILED = BIT1; +#endif + +/** + * @brief This top level thread processes all usb events and invokes callbacks + */ +static void tusb_device_task(void *arg) +{ + ESP_LOGD(TAG, "tinyusb task started"); +#if CONFIG_TINYUSB_INIT_IN_DEFAULT_TASK + EventGroupHandle_t *init_flags = arg; + if (!tusb_init()) { + ESP_LOGI(TAG, "Init TinyUSB stack failed"); + xEventGroupSetBits(*init_flags, INIT_FAILED); + vTaskDelete(NULL); + } + ESP_LOGD(TAG, "tinyusb task has been initialized"); + xEventGroupSetBits(*init_flags, INIT_OK); +#endif // CONFIG_TINYUSB_INIT_IN_DEFAULT_TASK + while (1) { // RTOS forever loop + tud_task(); + } +} + +esp_err_t tusb_run_task(void) +{ + // This function is not guaranteed to be thread safe, if invoked multiple times without calling `tusb_stop_task`, will cause memory leak + // doing a sanity check anyway + ESP_RETURN_ON_FALSE(!s_tusb_tskh, ESP_ERR_INVALID_STATE, TAG, "TinyUSB main task already started"); + + void *task_arg = NULL; +#if CONFIG_TINYUSB_INIT_IN_DEFAULT_TASK + // need to synchronize to potentially report issue if init failed + EventGroupHandle_t init_flags = xEventGroupCreate(); + ESP_RETURN_ON_FALSE(init_flags, ESP_ERR_NO_MEM, TAG, "Failed to allocate task sync flags"); + task_arg = &init_flags; +#endif + // Create a task for tinyusb device stack: + xTaskCreatePinnedToCore(tusb_device_task, "TinyUSB", CONFIG_TINYUSB_TASK_STACK_SIZE, task_arg, CONFIG_TINYUSB_TASK_PRIORITY, &s_tusb_tskh, CONFIG_TINYUSB_TASK_AFFINITY); + ESP_RETURN_ON_FALSE(s_tusb_tskh, ESP_FAIL, TAG, "create TinyUSB main task failed"); +#if CONFIG_TINYUSB_INIT_IN_DEFAULT_TASK + // wait until tusb initialization has completed + EventBits_t bits = xEventGroupWaitBits(init_flags, INIT_OK | INIT_FAILED, pdFALSE, pdFALSE, portMAX_DELAY); + vEventGroupDelete(init_flags); + ESP_RETURN_ON_FALSE(bits & INIT_OK, ESP_FAIL, TAG, "Init TinyUSB stack failed"); +#endif + + return ESP_OK; +} + +esp_err_t tusb_stop_task(void) +{ + ESP_RETURN_ON_FALSE(s_tusb_tskh, ESP_ERR_INVALID_STATE, TAG, "TinyUSB main task not started yet"); + vTaskDelete(s_tusb_tskh); + s_tusb_tskh = NULL; + return ESP_OK; +} diff --git a/examples/usb/device/usb_dongle/components/tinyusb_dongle/usb_descriptors.c b/examples/usb/device/usb_dongle/components/tinyusb_dongle/usb_descriptors.c new file mode 100644 index 000000000..67546d387 --- /dev/null +++ b/examples/usb/device/usb_dongle/components/tinyusb_dongle/usb_descriptors.c @@ -0,0 +1,307 @@ +/* + * SPDX-FileCopyrightText: 2020-2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#include "usb_descriptors.h" +#include "sdkconfig.h" +#include "tinyusb_types.h" + +/* + * A combination of interfaces must have a unique product id, since PC will save device driver after the first plug. + * Same VID/PID with different interface e.g MSC (first), then CDC (later) will possibly cause system error on PC. + * + * Auto ProductID layout's Bitmap: + * [MSB] HID | MSC | CDC [LSB] + */ +#define _PID_MAP(itf, n) ((CFG_TUD_##itf) << (n)) +#define USB_TUSB_PID (0x4000 | _PID_MAP(CDC, 0) | _PID_MAP(MSC, 1) | _PID_MAP(HID, 2) | \ + _PID_MAP(MIDI, 3) ) //| _PID_MAP(AUDIO, 4) | _PID_MAP(VENDOR, 5) ) + +/**** Kconfig driven Descriptor ****/ + +//------------- Device Descriptor -------------// +const tusb_desc_device_t descriptor_dev_default = { + .bLength = sizeof(descriptor_dev_default), + .bDescriptorType = TUSB_DESC_DEVICE, + .bcdUSB = 0x0200, + +#if CFG_TUD_CDC + // Use Interface Association Descriptor (IAD) for CDC + // As required by USB Specs IAD's subclass must be common class (2) and protocol must be IAD (1) + .bDeviceClass = TUSB_CLASS_MISC, + .bDeviceSubClass = MISC_SUBCLASS_COMMON, + .bDeviceProtocol = MISC_PROTOCOL_IAD, +#else + .bDeviceClass = 0x00, + .bDeviceSubClass = 0x00, + .bDeviceProtocol = 0x00, +#endif + + .bMaxPacketSize0 = CFG_TUD_ENDPOINT0_SIZE, + +#if CONFIG_TINYUSB_DESC_USE_ESPRESSIF_VID + .idVendor = USB_ESPRESSIF_VID, +#else + .idVendor = CONFIG_TINYUSB_DESC_CUSTOM_VID, +#endif + +#if CONFIG_TINYUSB_DESC_USE_DEFAULT_PID + .idProduct = USB_TUSB_PID, +#else + .idProduct = CONFIG_TINYUSB_DESC_CUSTOM_PID, +#endif + + .bcdDevice = CONFIG_TINYUSB_DESC_BCD_DEVICE, + + .iManufacturer = 0x01, + .iProduct = 0x02, + .iSerialNumber = 0x03, + + .bNumConfigurations = 0x01 +}; + +#if (TUD_OPT_HIGH_SPEED) +const tusb_desc_device_qualifier_t descriptor_qualifier_default = { + .bLength = sizeof(tusb_desc_device_qualifier_t), + .bDescriptorType = TUSB_DESC_DEVICE_QUALIFIER, + .bcdUSB = 0x0200, + +#if CFG_TUD_CDC + // Use Interface Association Descriptor (IAD) for CDC + // As required by USB Specs IAD's subclass must be common class (2) and protocol must be IAD (1) + .bDeviceClass = TUSB_CLASS_MISC, + .bDeviceSubClass = MISC_SUBCLASS_COMMON, + .bDeviceProtocol = MISC_PROTOCOL_IAD, +#else + .bDeviceClass = 0x00, + .bDeviceSubClass = 0x00, + .bDeviceProtocol = 0x00, +#endif + + .bMaxPacketSize0 = CFG_TUD_ENDPOINT0_SIZE, + .bNumConfigurations = 0x01, + .bReserved = 0 +}; +#endif // TUD_OPT_HIGH_SPEED + +//------------- Array of String Descriptors -------------// +const char *descriptor_str_default[] = { + // array of pointer to string descriptors + (char[]){0x09, 0x04}, // 0: is supported language is English (0x0409) + CONFIG_TINYUSB_DESC_MANUFACTURER_STRING, // 1: Manufacturer + CONFIG_TINYUSB_DESC_PRODUCT_STRING, // 2: Product + CONFIG_TINYUSB_DESC_SERIAL_STRING, // 3: Serials, should use chip ID + +#if CONFIG_TINYUSB_CDC_ENABLED + CONFIG_TINYUSB_DESC_CDC_STRING, // 4: CDC Interface +#endif + +#if CONFIG_TINYUSB_BTH_ENABLED + CONFIG_TINYUSB_DESC_BTH_STRING, // 6: BTH Interface +#endif + +#if CONFIG_TINYUSB_DFU_ENABLED + "FLASH", // 4: DFU Partition 1 + "EEPROM", // 5: DFU Partition 2 +#endif + +#if CFG_TUD_NCM || CFG_TUD_ECM_RNDIS + "USB net", // 8. NET Interface + "", // 9. MAC +#endif + + NULL // NULL: Must be last. Indicates end of array +}; + +//------------- Interfaces enumeration -------------// +enum { + +#if CFG_TUD_NCM || CFG_TUD_ECM_RNDIS + ITF_NUM_NET = 0, + ITF_NUM_NET_DATA, +#endif + +#if CFG_TUD_CDC + ITF_NUM_CDC, + ITF_NUM_CDC_DATA, +#endif + +#if CFG_TUD_CDC > 1 + ITF_NUM_CDC1, + ITF_NUM_CDC1_DATA, +#endif + +#if CFG_TUD_BTH + ITF_NUM_BTH, +#endif + +#if CFG_TUD_DFU + ITF_NUM_DFU, +#endif + + ITF_NUM_TOTAL +}; + +#define DFU_ALT_COUNT 2 +#define FUNC_ATTRS (DFU_ATTR_CAN_UPLOAD | DFU_ATTR_CAN_DOWNLOAD | DFU_ATTR_MANIFESTATION_TOLERANT) + +enum { + TUSB_DESC_TOTAL_LEN = TUD_CONFIG_DESC_LEN + + CFG_TUD_CDC * TUD_CDC_DESC_LEN + + CFG_TUD_NCM * TUD_CDC_NCM_DESC_LEN + + CFG_TUD_BTH * TUD_BTH_DESC_LEN + + CFG_TUD_DFU * TUD_DFU_DESC_LEN(DFU_ALT_COUNT) + + CONFIG_TINYUSB_NET_MODE_ECM * TUD_CDC_ECM_DESC_LEN + + CONFIG_TINYUSB_NET_MODE_RNDIS * TUD_RNDIS_DESC_LEN +}; + +//------------- USB Endpoint numbers -------------// +enum { + // Available USB Endpoints: 5 IN/OUT EPs and 1 IN EP + EP_EMPTY = 0, + +#if CFG_TUD_NCM || CFG_TUD_ECM_RNDIS + EPNUM_NET_NOTIF, + EPNUM_NET_DATA, +#endif + +#if CFG_TUD_CDC + EPNUM_0_CDC_NOTIF, + EPNUM_0_CDC, +#endif + +#if CFG_TUD_CDC > 1 + EPNUM_1_CDC_NOTIF, + EPNUM_1_CDC, +#endif + +#if CFG_TUD_BTH + EPNUM_BT_EVT, + EPNUM_BT_BULK_OUT, +#endif + +}; + +//------------- STRID -------------// +enum { + STRID_LANGID = 0, + STRID_MANUFACTURER, + STRID_PRODUCT, + STRID_SERIAL, + +#if CFG_TUD_CDC + STRID_CDC_INTERFACE, +#endif + +#if CFG_TUD_BTH + STRID_BTH_INTERFACE, +#endif + +#if CFG_TUD_DFU + STRID_DFU_INTERFACE, +#endif + +#if CFG_TUD_NCM || CFG_TUD_ECM_RNDIS + STRID_NET_INTERFACE, + STRID_MAC, +#endif + +}; + +//------------- Configuration Descriptor -------------// +uint8_t const descriptor_fs_cfg_default[] = { + // Configuration number, interface count, string index, total length, attribute, power in mA + TUD_CONFIG_DESCRIPTOR(1, ITF_NUM_TOTAL, 0, TUSB_DESC_TOTAL_LEN, TUSB_DESC_CONFIG_ATT_REMOTE_WAKEUP, 100), + +#if CFG_TUD_NCM + // Interface number, description string index, MAC address string index, EP notification address and size, EP data address (out, in), and size, max segment size. + TUD_CDC_NCM_DESCRIPTOR(ITF_NUM_NET, STRID_NET_INTERFACE, STRID_MAC, (0x80 | EPNUM_NET_NOTIF), 64, EPNUM_NET_DATA, (0x80 | EPNUM_NET_DATA), 64, CFG_TUD_NET_MTU), +#endif + +#if CFG_TUD_ECM_RNDIS +#if CONFIG_TINYUSB_NET_MODE_RNDIS + // Interface number, string index, EP notification address and size, EP data address (out, in) and size. + TUD_RNDIS_DESCRIPTOR(ITF_NUM_NET, STRID_NET_INTERFACE, (0x80 | EPNUM_NET_NOTIF), 8, EPNUM_NET_DATA, (0x80 | EPNUM_NET_DATA), 64), +#elif CONFIG_TINYUSB_NET_MODE_ECM + // Interface number, description string index, MAC address string index, EP notification address and size, EP data address (out, in), and size, max segment size. + TUD_CDC_ECM_DESCRIPTOR(ITF_NUM_NET, STRID_NET_INTERFACE, STRID_MAC, (0x80 | EPNUM_NET_NOTIF), 64, EPNUM_NET_DATA, (0x80 | EPNUM_NET_DATA), 64, CFG_TUD_NET_MTU), +#endif +#endif + +#if CFG_TUD_CDC + // Interface number, string index, EP notification address and size, EP data address (out, in) and size. + TUD_CDC_DESCRIPTOR(ITF_NUM_CDC, STRID_CDC_INTERFACE, 0x80 | EPNUM_0_CDC_NOTIF, 8, EPNUM_0_CDC, 0x80 | EPNUM_0_CDC, 64), +#endif + +#if CFG_TUD_CDC > 1 + // Interface number, string index, EP notification address and size, EP data address (out, in) and size. + TUD_CDC_DESCRIPTOR(ITF_NUM_CDC1, STRID_CDC_INTERFACE, 0x80 | EPNUM_1_CDC_NOTIF, 8, EPNUM_1_CDC, 0x80 | EPNUM_1_CDC, 64), +#endif + +#if CFG_TUD_BTH + // BT Primary controller descriptor + // Interface number, string index, attributes, event endpoint, event endpoint size, interval, data in, data out, data endpoint size, iso endpoint sizes + TUD_BTH_DESCRIPTOR(ITF_NUM_BTH, STRID_BTH_INTERFACE, (0x80 | EPNUM_BT_EVT), 16, 1, (0x80 | EPNUM_BT_BULK_OUT), EPNUM_BT_BULK_OUT, 64, 64, 64), +#endif + +#if CFG_TUD_DFU + // Interface number, Alternate count, starting string index, attributes, detach timeout, transfer size + TUD_DFU_DESCRIPTOR(ITF_NUM_DFU, DFU_ALT_COUNT, STRID_DFU_INTERFACE, FUNC_ATTRS, 1000, CFG_TUD_DFU_XFER_BUFSIZE), +#endif + +}; + +#if (TUD_OPT_HIGH_SPEED) +uint8_t const descriptor_hs_cfg_default[] = { + // Configuration number, interface count, string index, total length, attribute, power in mA + TUD_CONFIG_DESCRIPTOR(1, ITF_NUM_TOTAL, 0, TUSB_DESC_TOTAL_LEN, TUSB_DESC_CONFIG_ATT_REMOTE_WAKEUP, 100), + +#if CFG_TUD_NCM + // Interface number, description string index, MAC address string index, EP notification address and size, EP data address (out, in), and size, max segment size. + TUD_CDC_NCM_DESCRIPTOR(ITF_NUM_NET, STRID_NET_INTERFACE, STRID_MAC, (0x80 | EPNUM_NET_NOTIF), 512, EPNUM_NET_DATA, (0x80 | EPNUM_NET_DATA), 512, CFG_TUD_NET_MTU), +#endif + +#if CFG_TUD_ECM_RNDIS +#if CONFIG_TINYUSB_NET_MODE_RNDIS + // Interface number, string index, EP notification address and size, EP data address (out, in) and size. + TUD_RNDIS_DESCRIPTOR(ITF_NUM_NET, STRID_NET_INTERFACE, (0x80 | EPNUM_NET_NOTIF), 8, EPNUM_NET_DATA, (0x80 | EPNUM_NET_DATA), 512), +#elif CONFIG_TINYUSB_NET_MODE_ECM + // Interface number, description string index, MAC address string index, EP notification address and size, EP data address (out, in), and size, max segment size. + TUD_CDC_ECM_DESCRIPTOR(ITF_NUM_NET, STRID_NET_INTERFACE, STRID_MAC, (0x80 | EPNUM_NET_NOTIF), 512, EPNUM_NET_DATA, (0x80 | EPNUM_NET_DATA), 512, CFG_TUD_NET_MTU), +#endif +#endif + +#if CFG_TUD_CDC + // Interface number, string index, EP notification address and size, EP data address (out, in) and size. + TUD_CDC_DESCRIPTOR(ITF_NUM_CDC, STRID_CDC_INTERFACE, 0x80 | EPNUM_0_CDC_NOTIF, 8, EPNUM_0_CDC, 0x80 | EPNUM_0_CDC, 512), +#endif + +#if CFG_TUD_CDC > 1 + // Interface number, string index, EP notification address and size, EP data address (out, in) and size. + TUD_CDC_DESCRIPTOR(ITF_NUM_CDC1, STRID_CDC_INTERFACE, 0x80 | EPNUM_1_CDC_NOTIF, 8, EPNUM_1_CDC, 0x80 | EPNUM_1_CDC, 512), +#endif + +#if CFG_TUD_BTH + // BT Primary controller descriptor + // Interface number, string index, attributes, event endpoint, event endpoint size, interval, data in, data out, data endpoint size, iso endpoint sizes + TUD_BTH_DESCRIPTOR(ITF_NUM_BTH, STRID_BTH_INTERFACE, (0x80 | EPNUM_BT_EVT), 16, 1, (0x80 | EPNUM_BT_BULK_OUT), EPNUM_BT_BULK_OUT, 512, 1024, 1024), +#endif + +#if CFG_TUD_DFU + // Interface number, Alternate count, starting string index, attributes, detach timeout, transfer size + TUD_DFU_DESCRIPTOR(ITF_NUM_DFU, DFU_ALT_COUNT, STRID_DFU_INTERFACE, FUNC_ATTRS, 1000, CFG_TUD_DFU_XFER_BUFSIZE), +#endif + +}; +#endif // TUD_OPT_HIGH_SPEED + +#if CFG_TUD_NCM || CFG_TUD_ECM_RNDIS +uint8_t tusb_get_mac_string_id(void) +{ + return STRID_MAC; +} +#endif + +/* End of Kconfig driven Descriptor */ diff --git a/examples/usb/device/usb_dongle/components/tinyusb_dongle/vfs_tinyusb.c b/examples/usb/device/usb_dongle/components/tinyusb_dongle/vfs_tinyusb.c new file mode 100644 index 000000000..b3d431a2b --- /dev/null +++ b/examples/usb/device/usb_dongle/components/tinyusb_dongle/vfs_tinyusb.c @@ -0,0 +1,298 @@ +/* + * SPDX-FileCopyrightText: 2020-2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include "esp_attr.h" +#include "esp_log.h" +#include "esp_vfs.h" +#include "esp_vfs_dev.h" +#include "tinyusb.h" +#include "tusb_cdc_acm.h" +#include "vfs_tinyusb.h" +#include "sdkconfig.h" + +const static char *TAG = "tusb_vfs"; + +// Token signifying that no character is available +#define NONE -1 + +#define FD_CHECK(fd, ret_val) do { \ + if ((fd) != 0) { \ + errno = EBADF; \ + return (ret_val); \ + } \ + } while (0) + +#if CONFIG_NEWLIB_STDOUT_LINE_ENDING_CRLF +# define DEFAULT_TX_MODE ESP_LINE_ENDINGS_CRLF +#elif CONFIG_NEWLIB_STDOUT_LINE_ENDING_CR +# define DEFAULT_TX_MODE ESP_LINE_ENDINGS_CR +#else +# define DEFAULT_TX_MODE ESP_LINE_ENDINGS_LF +#endif + +#if CONFIG_NEWLIB_STDIN_LINE_ENDING_CRLF +# define DEFAULT_RX_MODE ESP_LINE_ENDINGS_CRLF +#elif CONFIG_NEWLIB_STDIN_LINE_ENDING_CR +# define DEFAULT_RX_MODE ESP_LINE_ENDINGS_CR +#else +# define DEFAULT_RX_MODE ESP_LINE_ENDINGS_LF +#endif + +typedef struct { + _lock_t write_lock; + _lock_t read_lock; + esp_line_endings_t tx_mode; // Newline conversion mode when transmitting + esp_line_endings_t rx_mode; // Newline conversion mode when receiving + uint32_t flags; + char vfs_path[VFS_TUSB_MAX_PATH]; + int cdc_intf; +} vfs_tinyusb_t; + +static vfs_tinyusb_t s_vfstusb; + +static esp_err_t apply_path(char const *path) +{ + if (path == NULL) { + path = VFS_TUSB_PATH_DEFAULT; + } + + size_t path_len = strlen(path) + 1; + if (path_len > VFS_TUSB_MAX_PATH) { + ESP_LOGE(TAG, "The path is too long; maximum is %d characters", VFS_TUSB_MAX_PATH); + return ESP_ERR_INVALID_ARG; + } + strncpy(s_vfstusb.vfs_path, path, (VFS_TUSB_MAX_PATH - 1)); + ESP_LOGV(TAG, "Path is set to `%s`", path); + return ESP_OK; +} + +/** + * @brief Fill s_vfstusb + * + * @param cdc_intf - interface of tusb for registration + * @param path - a path where the CDC will be registered + * @return esp_err_t ESP_OK or ESP_ERR_INVALID_ARG + */ +static esp_err_t vfstusb_init(int cdc_intf, char const *path) +{ + s_vfstusb.cdc_intf = cdc_intf; + s_vfstusb.tx_mode = DEFAULT_TX_MODE; + s_vfstusb.rx_mode = DEFAULT_RX_MODE; + + return apply_path(path); +} + +/** + * @brief Clear s_vfstusb to default values + */ +static void vfstusb_deinit(void) +{ + memset(&s_vfstusb, 0, sizeof(s_vfstusb)); +} + +static int tusb_open(const char *path, int flags, int mode) +{ + (void) mode; + (void) path; + s_vfstusb.flags = flags | O_NONBLOCK; // for now only non-blocking mode is implemented + return 0; +} + +static ssize_t tusb_write(int fd, const void *data, size_t size) +{ + FD_CHECK(fd, -1); + size_t written_sz = 0; + const char *data_c = (const char *)data; + _lock_acquire(&(s_vfstusb.write_lock)); + for (size_t i = 0; i < size; i++) { + int c = data_c[i]; + if (c != '\n') { + if (!tinyusb_cdcacm_write_queue_char(s_vfstusb.cdc_intf, c)) { + break; // can't write anymore + } + } else { + if (s_vfstusb.tx_mode == ESP_LINE_ENDINGS_CRLF || s_vfstusb.tx_mode == ESP_LINE_ENDINGS_CR) { + char cr = '\r'; + if (!tinyusb_cdcacm_write_queue_char(s_vfstusb.cdc_intf, cr)) { + break; // can't write anymore + } + } + if (s_vfstusb.tx_mode == ESP_LINE_ENDINGS_CRLF || s_vfstusb.tx_mode == ESP_LINE_ENDINGS_LF) { + char lf = '\n'; + if (!tinyusb_cdcacm_write_queue_char(s_vfstusb.cdc_intf, lf)) { + break; // can't write anymore + } + } + } + written_sz++; + } + tud_cdc_n_write_flush(s_vfstusb.cdc_intf); + _lock_release(&(s_vfstusb.write_lock)); + return written_sz; +} + +static int tusb_close(int fd) +{ + FD_CHECK(fd, -1); + return 0; +} + +static ssize_t tusb_read(int fd, void *data, size_t size) +{ + FD_CHECK(fd, -1); + char *data_c = (char *) data; + size_t received = 0; + _lock_acquire(&(s_vfstusb.read_lock)); + + if (tud_cdc_n_available(s_vfstusb.cdc_intf) == 0) { + goto finish; + } + while (received < size) { + int c = tud_cdc_n_read_char(s_vfstusb.cdc_intf); + if (c == NONE) { // if data ends + break; + } + + // Handle line endings. From configured mode -> LF mode + if (s_vfstusb.rx_mode == ESP_LINE_ENDINGS_CR) { + // Change CRs to newlines + if (c == '\r') { + c = '\n'; + } + } else if (s_vfstusb.rx_mode == ESP_LINE_ENDINGS_CRLF) { + if (c == '\r') { + uint8_t next_char = NONE; + // Check if next char is newline. If yes, we got CRLF sequence + tud_cdc_n_peek(s_vfstusb.cdc_intf, &next_char); + if (next_char == '\n') { + c = tud_cdc_n_read_char(s_vfstusb.cdc_intf); // Remove '\n' from the fifo + } + } + } + + data_c[received] = (char) c; + ++received; + if (c == '\n') { + break; + } + } +finish: + _lock_release(&(s_vfstusb.read_lock)); + if (received > 0) { + return received; + } + errno = EWOULDBLOCK; + return -1; +} + +static int tusb_fstat(int fd, struct stat *st) +{ + FD_CHECK(fd, -1); + memset(st, 0, sizeof(*st)); + st->st_mode = S_IFCHR; + return 0; +} + +static int tusb_fcntl(int fd, int cmd, int arg) +{ + FD_CHECK(fd, -1); + int result = 0; + switch (cmd) { + case F_GETFL: + result = s_vfstusb.flags; + break; + case F_SETFL: + s_vfstusb.flags = arg; + break; + default: + result = -1; + errno = ENOSYS; + break; + } + return result; +} + +esp_err_t esp_vfs_tusb_cdc_unregister(char const *path) +{ + ESP_LOGD(TAG, "Unregistering CDC-VFS driver"); + int res; + + if (path == NULL) { // NULL means using the default path for unregistering: VFS_TUSB_PATH_DEFAULT + path = VFS_TUSB_PATH_DEFAULT; + } + res = strcmp(s_vfstusb.vfs_path, path); + + if (res) { + res = ESP_ERR_INVALID_ARG; + ESP_LOGE(TAG, "There is no CDC-VFS driver registered to path '%s' (err: 0x%x)", path, res); + return res; + } + + res = esp_vfs_unregister(s_vfstusb.vfs_path); + if (res != ESP_OK) { + ESP_LOGE(TAG, "Can't unregister CDC-VFS driver from '%s' (err: 0x%x)", s_vfstusb.vfs_path, res); + } else { + ESP_LOGD(TAG, "Unregistered CDC-VFS driver"); + vfstusb_deinit(); + } + return res; +} + +esp_err_t esp_vfs_tusb_cdc_register(int cdc_intf, char const *path) +{ + ESP_LOGD(TAG, "Registering CDC-VFS driver"); + int res; + if (!tusb_cdc_acm_initialized(cdc_intf)) { + ESP_LOGE(TAG, "TinyUSB CDC#%d is not initialized", cdc_intf); + return ESP_ERR_INVALID_STATE; + } + + res = vfstusb_init(cdc_intf, path); + if (res != ESP_OK) { + return res; + } + + esp_vfs_t vfs = { + .flags = ESP_VFS_FLAG_DEFAULT, + .close = &tusb_close, + .fcntl = &tusb_fcntl, + .fstat = &tusb_fstat, + .open = &tusb_open, + .read = &tusb_read, + .write = &tusb_write, + }; + + res = esp_vfs_register(s_vfstusb.vfs_path, &vfs, NULL); + if (res != ESP_OK) { + ESP_LOGE(TAG, "Can't register CDC-VFS driver (err: %x)", res); + } else { + ESP_LOGD(TAG, "CDC-VFS registered (%s)", s_vfstusb.vfs_path); + } + return res; +} + +void esp_vfs_tusb_cdc_set_rx_line_endings(esp_line_endings_t mode) +{ + _lock_acquire(&(s_vfstusb.read_lock)); + s_vfstusb.rx_mode = mode; + _lock_release(&(s_vfstusb.read_lock)); +} + +void esp_vfs_tusb_cdc_set_tx_line_endings(esp_line_endings_t mode) +{ + _lock_acquire(&(s_vfstusb.write_lock)); + s_vfstusb.tx_mode = mode; + _lock_release(&(s_vfstusb.write_lock)); +} diff --git a/examples/usb/device/usb_dongle/main/CLI_Commands.c b/examples/usb/device/usb_dongle/main/CLI_Commands.c new file mode 100644 index 000000000..dc61c3aeb --- /dev/null +++ b/examples/usb/device/usb_dongle/main/CLI_Commands.c @@ -0,0 +1,579 @@ +/* + * FreeRTOS V202107.00 + * Copyright (C) 2017 Amazon.com, Inc. or its affiliates. All Rights Reserved. + * + * 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. + * + * http://www.FreeRTOS.org + * http://aws.amazon.com/freertos + * + * 1 tab == 4 spaces! + */ + +/* + * SPDX-FileCopyrightText: 2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#include +#include +#include +#include "freertos/FreeRTOS.h" +#include "freertos/task.h" +#include "esp_chip_info.h" +// #include "esp_spi_flash.h" +#include "esp_flash.h" +#include "FreeRTOS_CLI.h" + +#include "tinyusb.h" +#include "cmd_wifi.h" + +#define cliNEW_LINE "\r\n" + +char* null_password = ""; + +#if CONFIG_FREERTOS_USE_STATS_FORMATTING_FUNCTIONS +static BaseType_t prvTaskStatusCommand(char *pcWriteBuffer, size_t xWriteBufferLen, const char *pcCommandString); +#endif + +#if CFG_TUD_NCM || CFG_TUD_ECM_RNDIS +static BaseType_t prvStationCommand(char *pcWriteBuffer, size_t xWriteBufferLen, const char *pcCommandString); + +static BaseType_t prvScanCommand(char *pcWriteBuffer, size_t xWriteBufferLen, const char *pcCommandString); + +static BaseType_t prvAPCommand(char *pcWriteBuffer, size_t xWriteBufferLen, const char *pcCommandString); + +static BaseType_t prvSetWiFiModeCommand(char *pcWriteBuffer, size_t xWriteBufferLen, const char *pcCommandString); + +static BaseType_t prvSmartConfigCommand(char *pcWriteBuffer, size_t xWriteBufferLen, const char *pcCommandString); +#endif /* CFG_TUD_NCM || CFG_TUD_ECM_RNDIS */ + +static BaseType_t prvRamCommand(char *pcWriteBuffer, size_t xWriteBufferLen, const char *pcCommandString); + +static BaseType_t prvRestartCommand(char *pcWriteBuffer, size_t xWriteBufferLen, const char *pcCommandString); + +static BaseType_t prvGetVersionCommand(char *pcWriteBuffer, size_t xWriteBufferLen, const char *pcCommandString); + +#if CONFIG_FREERTOS_USE_STATS_FORMATTING_FUNCTIONS +/* Structure that defines the "task-stats" command line command. */ +static const CLI_Command_Definition_t xTaskStatus = { + "task-status", /* The command string to type. */ + "task-status: Displays the state of each task\r\n", + prvTaskStatusCommand, /* The function to run. */ + 0 /* No parameters are expected. */ +}; +#endif + +#if CFG_TUD_NCM || CFG_TUD_ECM_RNDIS +/* Structure that defines the "sta" command line command. */ +static const CLI_Command_Definition_t xStationCommand = { + "sta", /* The command string to type. */ + "sta -s [-p ]: join specified soft-AP\r\nsta -d: disconnect specified soft-AP\r\n", + prvStationCommand, /* The function to run. */ + -1 /* The user can enter any number of commands. */ +}; + +/* Structure that defines the "scan" command line command. */ +static const CLI_Command_Definition_t xScanCommand = { + "scan", /* The command string to type. */ + "scan []: SSID of AP want to be scanned\r\n", + prvScanCommand, /* The function to run. */ + -1 /* The user can enter any number of commands. */ +}; + +/* Structure that defines the "ap" command line command. */ +static const CLI_Command_Definition_t xAPCommand = { + "ap", /* The command string to type. */ + "ap []: configure ssid and password\r\n", + prvAPCommand, /* The function to run. */ + -1 /* The user can enter any number of commands. */ +}; + +/* Structure that defines the "mode" command line command. */ +static const CLI_Command_Definition_t xSetWiFiModeCommand = { + "mode", /* The command string to type. */ + "mode : station mode; ap mode\r\n", + prvSetWiFiModeCommand, /* The function to run. */ + 1 /* one parameters are expected. */ +}; + +/* Structure that defines the "smartconfig" command line command. */ +static const CLI_Command_Definition_t xSmartConfigCommand = { + "smartconfig", /* The command string to type. */ + "smartconfig [op]: op:1, start smartconfig; op:0, stop smartconfig\r\n", + prvSmartConfigCommand, /* The function to run. */ + 1 /* No parameters are expected. */ +}; +#endif /* CFG_TUD_NCM || CFG_TUD_ECM_RNDIS */ + +/* Structure that defines the "ram" command line command. */ +static const CLI_Command_Definition_t xRamCommand = { + "ram", /* The command string to type. */ + "ram: Get the current size of free heap memory and minimum size of free heap memory\r\n", + prvRamCommand, /* The function to run. */ + 0 /* No parameters are expected. */ +}; + +/* Structure that defines the "restart" command line command. */ +static const CLI_Command_Definition_t xRestartCommand = { + "restart", /* The command string to type. */ + "restart: Software reset of the chip\r\n", + prvRestartCommand, /* The function to run. */ + 0 /* No parameters are expected. */ +}; + +/* Structure that defines the "version" command line command. */ +static const CLI_Command_Definition_t xGetVersionCommand = { + "version", /* The command string to type. */ + "version: Get version of chip and SDK\r\n", + prvGetVersionCommand, /* The function to run. */ + 0 /* No parameters are expected. */ +}; + +/*-----------------------------------------------------------*/ + +void vRegisterCLICommands(void) +{ + FreeRTOS_CLICreatMux(); + + /* Register all the command line commands defined immediately above. */ +#if CONFIG_FREERTOS_USE_STATS_FORMATTING_FUNCTIONS + FreeRTOS_CLIRegisterCommand(&xTaskStatus); +#endif +#if CFG_TUD_NCM || CFG_TUD_ECM_RNDIS + FreeRTOS_CLIRegisterCommand(&xAPCommand); + FreeRTOS_CLIRegisterCommand(&xStationCommand); + FreeRTOS_CLIRegisterCommand(&xSetWiFiModeCommand); + FreeRTOS_CLIRegisterCommand(&xSmartConfigCommand); + FreeRTOS_CLIRegisterCommand(&xScanCommand); +#endif /* CFG_TUD_NCM || CFG_TUD_ECM_RNDIS */ + FreeRTOS_CLIRegisterCommand(&xRamCommand); + FreeRTOS_CLIRegisterCommand(&xRestartCommand); + FreeRTOS_CLIRegisterCommand(&xGetVersionCommand); +} +/*-----------------------------------------------------------*/ + +#if CONFIG_FREERTOS_USE_STATS_FORMATTING_FUNCTIONS +static BaseType_t prvTaskStatusCommand(char *pcWriteBuffer, size_t xWriteBufferLen, const char *pcCommandString) +{ + const char *const pcHeader = "Task State Priority Stack #\r\n************************************************\r\n"; + + /* Remove compile time warnings about unused parameters, and check the + write buffer is not NULL. NOTE - for simplicity, this example assumes the + write buffer length is adequate, so does not check for buffer overflows. */ + (void) pcCommandString; + (void) xWriteBufferLen; + configASSERT(pcWriteBuffer); + memset(pcWriteBuffer, 0x00, xWriteBufferLen); + + char *data = (char *)malloc(512); + /* Generate a table of task stats. */ + strcpy(data, pcHeader); + vTaskList(data + strlen(pcHeader)); + printf("%s", data); + free(data); + + /* There is no more data to return after this single string, so return pdFALSE. */ + return pdFALSE; +} +#endif +/*-----------------------------------------------------------*/ + +#if CFG_TUD_NCM || CFG_TUD_ECM_RNDIS +static BaseType_t prvStationCommand(char *pcWriteBuffer, size_t xWriteBufferLen, const char *pcCommandString) +{ + char *pc1, *pc2, *pc3, *pc4, *pc5, *pc6; + BaseType_t xLength1, xLength2, xLength3, xLength4, xLength5, xLength6; + + /* Remove compile time warnings about unused parameters, and check the + write buffer is not NULL. NOTE - for simplicity, this example assumes the + write buffer length is adequate, so does not check for buffer overflows. */ + (void) pcCommandString; + (void) xWriteBufferLen; + configASSERT(pcWriteBuffer); + memset(pcWriteBuffer, 0x00, xWriteBufferLen); + + /* Obtain the sixth parameter */ + pc6 = (char *) FreeRTOS_CLIGetParameter + ( + pcCommandString, /* The command string itself. */ + 6, /* Return the Sixth parameter. */ + &xLength6 /* Store the parameter string length. */ + ); + + /* Obtain the fifth parameter */ + pc5 = (char *) FreeRTOS_CLIGetParameter + ( + pcCommandString, /* The command string itself. */ + 5, /* Return the fifth parameter. */ + &xLength5 /* Store the parameter string length. */ + ); + + /* Obtain the fourth parameter */ + pc4 = (char *) FreeRTOS_CLIGetParameter + ( + pcCommandString, /* The command string itself. */ + 4, /* Return the fourth parameter. */ + &xLength4 /* Store the parameter string length. */ + ); + + /* Obtain the third parameter. */ + pc3 = (char *) FreeRTOS_CLIGetParameter + ( + pcCommandString, /* The command string itself. */ + 3, /* Return the third parameter. */ + &xLength3 /* Store the parameter string length. */ + ); + + /* Obtain the second parameter */ + pc2 = (char *) FreeRTOS_CLIGetParameter + ( + pcCommandString, /* The command string itself. */ + 2, /* Return the second parameter. */ + &xLength2 /* Store the parameter string length. */ + ); + + /* Obtain the first parameter. */ + pc1 = (char *) FreeRTOS_CLIGetParameter + ( + pcCommandString, /* The command string itself. */ + 1, /* Return the first parameter. */ + &xLength1 /* Store the parameter string length. */ + ); + + if (pc1 == NULL) { + wifi_cmd_query(); + return pdFALSE; + } else { + /* Sanity check something was returned. */ + configASSERT(pc1); + /* Terminate the string. */ + pc1[ xLength1 ] = 0x00; + + if (strncmp(pc1, "-d", strlen("-d")) == 0) { + if (wif_cmd_disconnect_wifi() == ESP_OK) { + sprintf(pcWriteBuffer, "OK\r\n"); + } else { + sprintf(pcWriteBuffer, "FAIL\r\n"); + } + return pdFALSE; + } else if (strncmp(pc1, "-s", strlen("-s")) != 0) { + sprintf(pcWriteBuffer, "Invalid parameter\r\n"); + return pdFALSE; + } + } + + if (pc2 == NULL) { + sprintf(pcWriteBuffer, "Invalid parameter\r\n"); + return pdFALSE; + } else { + if (pc3 == NULL) { + wifi_cmd_sta_join(pc2, null_password); + printf("the ssid is %s.\r\n", pc2); + printf("the ssid len is %d.\r\n", xLength2); + return pdFALSE; + } + } + + if (pc4 == NULL) { + wifi_cmd_sta_join(pc2, null_password); + printf("the ssid is %s.\r\n", pc2); + printf("the ssid len is %d.\r\n", (xLength2 + xLength3 + 1)); + return pdFALSE; + } + + if (pc5 == NULL) { + /* Terminate the string. */ + pc2[ xLength2 ] = 0x00; + + if (strncmp(pc3, "-p", strlen("-p")) != 0) { + sprintf(pcWriteBuffer, "Invalid parameter\r\n"); + return pdFALSE; + } + /* Terminate the string. */ + pc3[ xLength3 ] = 0x00; + wifi_cmd_sta_join(pc2, pc4); + printf("the ssid is %s, the password is %s.\r\n", pc2, pc4); + printf("the ssid len is %d, the password len is %d.\r\n", xLength2, xLength4); + return pdFALSE; + } + + if (pc6 == NULL) { + if (strncmp(pc3, "-p", strlen("-p")) == 0) { + if (strncmp(pc4, "-p", strlen("-p")) != 0) { + /* Terminate the string. */ + pc2[ xLength2 ] = 0x00; + pc3[ xLength3 ] = 0x00; + wifi_cmd_sta_join(pc2, pc4); + printf("the ssid is %s, the password is %s.\r\n", pc2, pc4); + printf("the ssid len is %d, the password len is %d.\r\n", xLength2, (xLength4 + xLength5 + 1)); + return pdFALSE; + } + } + /* Terminate the string. */ + pc3[ xLength3 ] = 0x00; + pc4[ xLength4 ] = 0x00; + wifi_cmd_sta_join(pc2, pc5); + printf("the ssid is %s, the password is %s.\r\n", pc2, pc5); + printf("the ssid len is %d, the password len is %d.\r\n", (xLength2 + xLength3 + 1), xLength5); + } else { + /* Terminate the string. */ + pc3[ xLength3 ] = 0x00; + pc4[ xLength4 ] = 0x00; + wifi_cmd_sta_join(pc2, pc5); + printf("the ssid is %s, the password is %s.\r\n", pc2, pc5); + printf("the ssid len is %d, the password len is %d.\r\n", (xLength2 + xLength3 + 1), (xLength5 + xLength6 + 1)); + } + + return pdFALSE; +} +/*-----------------------------------------------------------*/ + +static BaseType_t prvScanCommand(char *pcWriteBuffer, size_t xWriteBufferLen, const char *pcCommandString) +{ + const char *pcSSID; + BaseType_t SSIDLength; + + /* Remove compile time warnings about unused parameters, and check the + write buffer is not NULL. NOTE - for simplicity, this example assumes the + write buffer length is adequate, so does not check for buffer overflows. */ + (void) pcCommandString; + (void) xWriteBufferLen; + configASSERT(pcWriteBuffer); + memset(pcWriteBuffer, 0x00, xWriteBufferLen); + + /* Obtain the SSID of AP . */ + pcSSID = FreeRTOS_CLIGetParameter + ( + pcCommandString, /* The command string itself. */ + 1, /* Return the first parameter. */ + &SSIDLength /* Store the parameter string length. */ + ); + + if (pcSSID == NULL) { + /* TODO */ + wifi_cmd_sta_scan(NULL); + } else { + /* TODO */ + wifi_cmd_sta_scan(pcSSID); + } + + /* There is no more data to return after this single string, so return + pdFALSE. */ + return pdFALSE; +} +/*-----------------------------------------------------------*/ + +static BaseType_t prvAPCommand(char *pcWriteBuffer, size_t xWriteBufferLen, const char *pcCommandString) +{ + char *pcSSID, *pcPassWord; + BaseType_t xSSIDLength, xPassWordLength; + + /* Remove compile time warnings about unused parameters, and check the + write buffer is not NULL. NOTE - for simplicity, this example assumes the + write buffer length is adequate, so does not check for buffer overflows. */ + (void) pcCommandString; + (void) xWriteBufferLen; + configASSERT(pcWriteBuffer); + memset(pcWriteBuffer, 0x00, xWriteBufferLen); + + /* Obtain the ssid of AP. */ + pcPassWord = (char *) FreeRTOS_CLIGetParameter + ( + pcCommandString, /* The command string itself. */ + 2, /* Return the second parameter. */ + &xPassWordLength /* Store the parameter string length. */ + ); + + /* Obtain the password of AP */ + pcSSID = (char *) FreeRTOS_CLIGetParameter + ( + pcCommandString, /* The command string itself. */ + 1, /* Return the first parameter. */ + &xSSIDLength /* Store the parameter string length. */ + ); + + if (pcSSID == NULL) { + wifi_cmd_query(); + return pdFALSE; + } + + /* Sanity check something was returned. */ + configASSERT(pcSSID); + + /* Terminate the string. */ + pcSSID[ xSSIDLength ] = 0x00; + + if (pcPassWord == NULL) { + /* TODO */ + wifi_cmd_ap_set(pcSSID, null_password); + printf("the ssid is %s.\r\n", pcSSID); + printf("the ssid len is %d.\r\n", xSSIDLength); + } else { + /* TODO */ + wifi_cmd_ap_set(pcSSID, pcPassWord); + printf("the ssid is %s, the password is %s.\r\n", pcSSID, pcPassWord); + printf("the ssid len is %d, the password len is %d.\r\n", xSSIDLength, xPassWordLength); + } + + return pdFALSE; +} +/*-----------------------------------------------------------*/ + +static BaseType_t prvSetWiFiModeCommand(char *pcWriteBuffer, size_t xWriteBufferLen, const char *pcCommandString) +{ + const char *pcWiFiMode; + BaseType_t WiFiModeLength; + + /* Remove compile time warnings about unused parameters, and check the + write buffer is not NULL. NOTE - for simplicity, this example assumes the + write buffer length is adequate, so does not check for buffer overflows. */ + (void) pcCommandString; + (void) xWriteBufferLen; + configASSERT(pcWriteBuffer); + memset(pcWriteBuffer, 0x00, xWriteBufferLen); + + /* Obtain the parameter string. */ + pcWiFiMode = FreeRTOS_CLIGetParameter + ( + pcCommandString, /* The command string itself. */ + 1, /* Return the first parameter. */ + &WiFiModeLength /* Store the parameter string length. */ + ); + + /* Sanity check something was returned. */ + configASSERT(pcWiFiMode); + + if (wifi_cmd_set_mode((char *)pcWiFiMode) == ESP_FAIL) { + sprintf(pcWriteBuffer, "Invalid parameter\r\n"); + } + + /* There is no more data to return after this single string, so return + pdFALSE. */ + return pdFALSE; +} +/*-----------------------------------------------------------*/ + +static BaseType_t prvSmartConfigCommand(char *pcWriteBuffer, size_t xWriteBufferLen, const char *pcCommandString) +{ + const char *pcSmartConfig; + BaseType_t Length; + + /* Remove compile time warnings about unused parameters, and check the + write buffer is not NULL. NOTE - for simplicity, this example assumes the + write buffer length is adequate, so does not check for buffer overflows. */ + (void) pcCommandString; + (void) xWriteBufferLen; + configASSERT(pcWriteBuffer); + memset(pcWriteBuffer, 0x00, xWriteBufferLen); + + /* Obtain the parameter string. */ + pcSmartConfig = FreeRTOS_CLIGetParameter + ( + pcCommandString, /* The command string itself. */ + 1, /* Return the first parameter. */ + &Length /* Store the parameter string length. */ + ); + + /* Sanity check something was returned. */ + configASSERT(pcSmartConfig); + + /* There are only two valid parameter values. */ + if (strncmp(pcSmartConfig, "0", strlen("0")) == 0) { + if (wifi_cmd_stop_smart_config() == ESP_OK) { + sprintf(pcWriteBuffer, "OK\r\n"); + } else { + sprintf(pcWriteBuffer, "FAIL\r\n"); + } + // sprintf(pcWriteBuffer, "Stop SmartConfig\r\n"); + } else if (strncmp(pcSmartConfig, "1", strlen("1")) == 0) { + if (wifi_cmd_start_smart_config() == ESP_FAIL) { + sprintf(pcWriteBuffer, "SmartConfig Task has been created, Don't create repeatedly\r\n"); + } + } else { + sprintf(pcWriteBuffer, "Valid parameters are '0' and '1'\r\n"); + } + + return pdFALSE; +} +/*-----------------------------------------------------------*/ +#endif /* CFG_TUD_NCM || CFG_TUD_ECM_RNDIS */ + +static BaseType_t prvRamCommand(char *pcWriteBuffer, size_t xWriteBufferLen, const char *pcCommandString) +{ + /* Remove compile time warnings about unused parameters, and check the + write buffer is not NULL. NOTE - for simplicity, this example assumes the + write buffer length is adequate, so does not check for buffer overflows. */ + (void) pcCommandString; + (void) xWriteBufferLen; + configASSERT(pcWriteBuffer); + memset(pcWriteBuffer, 0x00, xWriteBufferLen); + + uint32_t heap_size = heap_caps_get_minimum_free_size(MALLOC_CAP_DEFAULT); + uint32_t size = esp_get_free_heap_size(); + + sprintf(pcWriteBuffer, "free heap size: %"PRIu32", min heap size: %"PRIu32"\r\n", size, heap_size); + + return pdFALSE; +} +/*-----------------------------------------------------------*/ + +static BaseType_t prvRestartCommand(char *pcWriteBuffer, size_t xWriteBufferLen, const char *pcCommandString) +{ + /* Remove compile time warnings about unused parameters, and check the + write buffer is not NULL. NOTE - for simplicity, this example assumes the + write buffer length is adequate, so does not check for buffer overflows. */ + (void) pcCommandString; + (void) xWriteBufferLen; + configASSERT(pcWriteBuffer); + memset(pcWriteBuffer, 0x00, xWriteBufferLen); + + esp_restart(); + + return pdFALSE; +} +/*-----------------------------------------------------------*/ + +static BaseType_t prvGetVersionCommand(char *pcWriteBuffer, size_t xWriteBufferLen, const char *pcCommandString) +{ + /* Remove compile time warnings about unused parameters, and check the + write buffer is not NULL. NOTE - for simplicity, this example assumes the + write buffer length is adequate, so does not check for buffer overflows. */ + (void) pcCommandString; + (void) xWriteBufferLen; + configASSERT(pcWriteBuffer); + memset(pcWriteBuffer, 0x00, xWriteBufferLen); + + esp_chip_info_t info; + esp_chip_info(&info); + uint32_t flash_size = 0; + esp_flash_get_physical_size(NULL, &flash_size); + sprintf(pcWriteBuffer, "IDF Version:%s\r\nChip info:\r\n\tcores:%d\r\n\tfeature:%s%s%s%s%"PRIu32"%s\r\n\trevision number:%d\r\n", + esp_get_idf_version(), + info.cores, + info.features & CHIP_FEATURE_WIFI_BGN ? "/802.11bgn" : "", + info.features & CHIP_FEATURE_BLE ? "/BLE" : "", + info.features & CHIP_FEATURE_BT ? "/BT" : "", + info.features & CHIP_FEATURE_EMB_FLASH ? "/Embedded-Flash:" : "/External-Flash:", + flash_size / (1024 * 1024), " MB", + info.revision); + + return pdFALSE; +} +/*-----------------------------------------------------------*/ diff --git a/examples/usb/device/usb_dongle/main/CMakeLists.txt b/examples/usb/device/usb_dongle/main/CMakeLists.txt new file mode 100644 index 000000000..6e41355cf --- /dev/null +++ b/examples/usb/device/usb_dongle/main/CMakeLists.txt @@ -0,0 +1,17 @@ +set(srcs) + +list(APPEND srcs + "usb_dongle_main.c" + "CLI_Commands.c" + "Command_Parse.c" + "cmd_wifi.c" + "data_back.c" + ) + +if(CONFIG_UART_ENABLE) + list(APPEND srcs + "uart.c") +endif() # CONFIG_UART_ENABLE + +idf_component_register(SRCS ${srcs} + INCLUDE_DIRS .) diff --git a/examples/usb/device/usb_dongle/main/Command_Parse.c b/examples/usb/device/usb_dongle/main/Command_Parse.c new file mode 100644 index 000000000..2f0fa34c4 --- /dev/null +++ b/examples/usb/device/usb_dongle/main/Command_Parse.c @@ -0,0 +1,85 @@ +/* + * SPDX-FileCopyrightText: 2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#include +#include +#include "esp_log.h" +#include "esp_err.h" + +#include "FreeRTOS_CLI.h" + +#include "data_back.h" + +/* Dimensions the buffer into which input characters are placed. */ +#define cmdMAX_INPUT_SIZE 80 + +/* Dimensions the buffer into which string outputs can be placed. */ +#define cmdMAX_OUTPUT_SIZE 1024 + +/* Dimensions the buffer passed to the recvfrom() call. */ +#define cmdSOCKET_INPUT_BUFFER_SIZE 80 + +void Command_Parse(char* Cmd) +{ + size_t lBytes, lByte; + signed char cInChar, cInputIndex = 0; + static char cInputString[ cmdMAX_INPUT_SIZE ], cOutputString[ cmdMAX_OUTPUT_SIZE ], cLocalBuffer[ cmdSOCKET_INPUT_BUFFER_SIZE ]; + BaseType_t xMoreDataToFollow; + uint8_t *data = (uint8_t *)heap_caps_malloc(256, MALLOC_CAP_INTERNAL | MALLOC_CAP_8BIT); + char *format_buf = (char *)malloc(64); + + lBytes = strlen(Cmd); + memcpy(cLocalBuffer, Cmd, lBytes); + + /* Process each received byte in turn. */ + lByte = 0; + while (lByte < lBytes) { + /* The next character in the input buffer. */ + cInChar = cLocalBuffer[ lByte ]; + lByte++; + + /* Newline characters are taken as the end of the command string. */ + if (cInChar == '\n') { + size_t lenth = sprintf(format_buf, "\r\n"); + esp_data_back(format_buf, lenth, ENABLE_FLUSH); + /* Process the input string received prior to the newline. */ + do { + memset(data, 0x00, 256); + /* Pass the string to FreeRTOS+CLI. */ + xMoreDataToFollow = FreeRTOS_CLIProcessCommand(cInputString, cOutputString, cmdMAX_OUTPUT_SIZE); + printf("%s%d\r\n", cOutputString, strlen(cOutputString)); + /* Send the output generated by the command's implementation. */ + memcpy(data, (uint8_t*)cOutputString, strlen(cOutputString)); + esp_data_back(data, strlen(cOutputString), DISABLE_FLUSH); + } while (xMoreDataToFollow != pdFALSE); /* Until the command does not generate any more output. */ + + /* All the strings generated by the command processing have been sent. + Clear the input string ready to receive the next command. */ + lenth = sprintf(format_buf, ">"); + esp_data_back(format_buf, lenth, ENABLE_FLUSH); + cInputIndex = 0; + memset(cInputString, 0x00, cmdMAX_INPUT_SIZE); + free(data); + } else { + if (cInChar == '\r') { + /* Ignore the character. Newlines are used to detect the end of the input string. */ + } else if (cInChar == '\b') { + /* Backspace was pressed. Erase the last character in the string - if any. */ + if (cInputIndex > 0) { + cInputIndex--; + cInputString[ cInputIndex ] = '\0'; + } + } else { + /* A character was entered. Add it to the string entered so far. + When a \n is entered the complete string will be passed to the command interpreter.*/ + if (cInputIndex < cmdMAX_INPUT_SIZE) { + cInputString[ cInputIndex ] = cInChar; + cInputIndex++; + } + } + } + } +} diff --git a/examples/usb/device/usb_dongle/main/Kconfig.projbuild b/examples/usb/device/usb_dongle/main/Kconfig.projbuild new file mode 100644 index 000000000..7bfa284b4 --- /dev/null +++ b/examples/usb/device/usb_dongle/main/Kconfig.projbuild @@ -0,0 +1,59 @@ +menu "Example Configuration" + + config UART_ENABLE + depends on !TINYUSB_CDCACM_ENABLED + bool "Use Uart to Communication" + default y + help + Enable Uart. + + config UART_PORT_NUM + depends on UART_ENABLE + int "UART port number" + range 0 2 if IDF_TARGET_ESP32 || IDF_TARGET_ESP32S3 + range 0 1 if IDF_TARGET_ESP32S2 || IDF_TARGET_ESP32C3 + default 2 if IDF_TARGET_ESP32 || IDF_TARGET_ESP32S3 + default 1 if IDF_TARGET_ESP32S2 || IDF_TARGET_ESP32C3 + help + UART communication port number for the example. + See UART documentation for available port numbers. + + config UART_BAUD_RATE + depends on UART_ENABLE + int "UART communication speed" + range 1200 115200 + default 115200 + help + UART communication speed for Modbus example. + + config UART_RXD + depends on UART_ENABLE + int "UART RXD pin number" + range 0 34 if IDF_TARGET_ESP32 + range 0 46 if IDF_TARGET_ESP32S2 || IDF_TARGET_ESP32S3 + range 0 19 if IDF_TARGET_ESP32C3 + default 5 + help + GPIO number for UART RX pin. See UART documentation for more information + about available pin numbers for UART. + + config UART_TXD + depends on UART_ENABLE + int "UART TXD pin number" + range 0 34 if IDF_TARGET_ESP32 + range 0 46 if IDF_TARGET_ESP32S2 || IDF_TARGET_ESP32S3 + range 0 19 if IDF_TARGET_ESP32C3 + default 4 + help + GPIO number for UART TX pin. See UART documentation for more information + about available pin numbers for UART. + + config TASK_STACK_SIZE + depends on UART_ENABLE + int "UART echo example task stack size" + range 1024 16384 + default 4096 + help + Defines stack size for UART echo example. Insufficient stack size can cause crash. + +endmenu # "Example Configuration" diff --git a/examples/usb/device/usb_dongle/main/cmd_wifi.c b/examples/usb/device/usb_dongle/main/cmd_wifi.c new file mode 100644 index 000000000..eeb2036ad --- /dev/null +++ b/examples/usb/device/usb_dongle/main/cmd_wifi.c @@ -0,0 +1,457 @@ +/* + * SPDX-FileCopyrightText: 2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#include "tinyusb.h" + +#if CFG_TUD_NCM || CFG_TUD_ECM_RNDIS + +#include +#include +#include +#include + +#include "freertos/FreeRTOS.h" +#include "freertos/task.h" +#include "freertos/event_groups.h" +#include "lwip/ip_addr.h" +#include "lwip/netif.h" + +#include "esp_log.h" +#include "esp_err.h" +#include "esp_wifi.h" +#include "esp_eap_client.h" +#include "esp_netif.h" +#include "esp_event.h" +#include "esp_system.h" +#include "esp_smartconfig.h" +#include "esp_private/wifi.h" + +#include "tinyusb_net.h" +#include "tinyusb.h" +#include "data_back.h" + +static const char *TAG = "esp_network"; + +static TaskHandle_t Smart_Config_Handle = NULL; + +static bool reconnect = true; +static bool wifi_start = false; +static bool smart_config = false; + +static EventGroupHandle_t wifi_event_group; +const int CONNECTED_BIT = BIT0; +const int DISCONNECTED_BIT = BIT1; +const int ESPTOUCH_DONE_BIT = BIT2; +esp_netif_t *ap_netif; +esp_netif_t *sta_netif; + +DRAM_ATTR uint8_t tud_network_mac_address[6] = {0x02, 0x02, 0x84, 0x6A, 0x96, 0x00}; +bool s_wifi_is_connected = false; + +static esp_err_t pkt_wifi2usb(void *buffer, uint16_t len, void *eb); + +static void scan_done_handler(void* arg, esp_event_base_t event_base, + int32_t event_id, void* event_data) +{ + uint16_t sta_number = 0; + uint8_t i; + wifi_ap_record_t *ap_list_buffer; + char *scan_buf = (char *)malloc(64); + size_t lenth = 0; + + esp_wifi_scan_get_ap_num(&sta_number); + if (!sta_number) { + ESP_LOGE(TAG, "No AP found"); + lenth = sprintf(scan_buf, "\r\nNo AP found\r\n>"); + esp_data_back(scan_buf, lenth, ENABLE_FLUSH); + free(scan_buf); + return; + } + + ap_list_buffer = malloc(sta_number * sizeof(wifi_ap_record_t)); + if (ap_list_buffer == NULL) { + ESP_LOGE(TAG, "Failed to malloc buffer to print scan results"); + lenth = sprintf(scan_buf, "\r\nFailed to malloc\r\n>"); + esp_data_back(scan_buf, lenth, ENABLE_FLUSH); + free(scan_buf); + return; + } + + if (esp_wifi_scan_get_ap_records(&sta_number, (wifi_ap_record_t *)ap_list_buffer) == ESP_OK) { + for (i = 0; i < sta_number; i++) { + ESP_LOGI(TAG, "[%s][rssi=%d]", ap_list_buffer[i].ssid, ap_list_buffer[i].rssi); + lenth = sprintf(scan_buf, "\r\n[%s][rssi=%d]", ap_list_buffer[i].ssid, ap_list_buffer[i].rssi); + esp_data_back(scan_buf, lenth, DISABLE_FLUSH); + } + } + lenth = sprintf(scan_buf, "\r\n>"); + esp_data_back(scan_buf, lenth, ENABLE_FLUSH); + free(scan_buf); + free(ap_list_buffer); + ESP_LOGI(TAG, "sta scan done"); +} + +static void wifi_event_handler(void *arg, esp_event_base_t event_base, + int32_t event_id, void *event_data) +{ + char *wifi_event_buf = (char *)malloc(128); + size_t lenth = 0; + + if (event_base == WIFI_EVENT && event_id == WIFI_EVENT_STA_START) { + uint8_t *tud_network_mac_address_dummy = tud_network_mac_address; + esp_wifi_get_mac(ESP_IF_WIFI_STA, tud_network_mac_address_dummy); + // esp_wifi_connect(); + wifi_start = true; + } else if (event_base == WIFI_EVENT && event_id == WIFI_EVENT_STA_DISCONNECTED) { + ESP_LOGI(TAG, "Wi-Fi STA disconnected"); + s_wifi_is_connected = false; + esp_wifi_internal_reg_rxcb(ESP_IF_WIFI_STA, NULL); + + if (reconnect && tud_ready()) { + ESP_LOGI(TAG, "sta disconnect, reconnect..."); + esp_wifi_connect(); + } else { + ESP_LOGI(TAG, "sta disconnect"); + } + xEventGroupClearBits(wifi_event_group, CONNECTED_BIT); + xEventGroupSetBits(wifi_event_group, DISCONNECTED_BIT); + ESP_LOGI(TAG, "DISCONNECTED_BIT\r\n"); + } else if (event_base == WIFI_EVENT && event_id == WIFI_EVENT_STA_CONNECTED) { + if (smart_config == false) { + ESP_LOGI(TAG, "Wi-Fi STA connected"); + esp_wifi_internal_reg_rxcb(ESP_IF_WIFI_STA, pkt_wifi2usb); + s_wifi_is_connected = true; + xEventGroupClearBits(wifi_event_group, DISCONNECTED_BIT); + xEventGroupSetBits(wifi_event_group, CONNECTED_BIT); + ESP_LOGI(TAG, "CONNECTED_BIT\r\n"); + } + } else if (event_base == SC_EVENT && event_id == SC_EVENT_SCAN_DONE) { + ESP_LOGI(TAG, "Scan done"); + } else if (event_base == SC_EVENT && event_id == SC_EVENT_FOUND_CHANNEL) { + ESP_LOGI(TAG, "Found channel"); + } else if (event_base == SC_EVENT && event_id == SC_EVENT_GOT_SSID_PSWD) { + ESP_LOGI(TAG, "Got SSID and password"); + + smartconfig_event_got_ssid_pswd_t *evt = (smartconfig_event_got_ssid_pswd_t *)event_data; + wifi_config_t wifi_config; + uint8_t ssid[33] = { 0 }; + uint8_t password[65] = { 0 }; + uint8_t rvd_data[33] = { 0 }; + + bzero(&wifi_config, sizeof(wifi_config_t)); + memcpy(wifi_config.sta.ssid, evt->ssid, sizeof(wifi_config.sta.ssid)); + memcpy(wifi_config.sta.password, evt->password, sizeof(wifi_config.sta.password)); + wifi_config.sta.bssid_set = evt->bssid_set; + if (wifi_config.sta.bssid_set == true) { + memcpy(wifi_config.sta.bssid, evt->bssid, sizeof(wifi_config.sta.bssid)); + } + + memcpy(ssid, evt->ssid, sizeof(evt->ssid)); + memcpy(password, evt->password, sizeof(evt->password)); + ESP_LOGI(TAG, "SSID:%s PASSWORD:%s", ssid, password); + lenth = sprintf(wifi_event_buf, "SSID:%s,PASSWORD:%s\r\n", ssid, password); + esp_data_back(wifi_event_buf, lenth, ENABLE_FLUSH); + + if (evt->type == SC_TYPE_ESPTOUCH_V2) { + ESP_ERROR_CHECK(esp_smartconfig_get_rvd_data(rvd_data, sizeof(rvd_data))); + ESP_LOGI(TAG, "RVD_DATA:"); + for (int i = 0; i < 33; i++) { + printf("%02x ", rvd_data[i]); + } + printf("\n"); + } + + ESP_ERROR_CHECK(esp_wifi_disconnect()); + ESP_ERROR_CHECK(esp_wifi_set_mode(WIFI_MODE_STA)); + ESP_ERROR_CHECK(esp_wifi_set_config(WIFI_IF_STA, &wifi_config)); + esp_wifi_connect(); + } else if (event_base == SC_EVENT && event_id == SC_EVENT_SEND_ACK_DONE) { + ESP_LOGI(TAG, "Send ACK done"); + xEventGroupSetBits(wifi_event_group, ESPTOUCH_DONE_BIT); + esp_wifi_internal_reg_rxcb(ESP_IF_WIFI_STA, pkt_wifi2usb); + s_wifi_is_connected = true; + xEventGroupClearBits(wifi_event_group, DISCONNECTED_BIT); + xEventGroupSetBits(wifi_event_group, CONNECTED_BIT); + } + free(wifi_event_buf); +} + +uint32_t wifi_get_local_ip(void) +{ + int bits = xEventGroupWaitBits(wifi_event_group, CONNECTED_BIT, 0, 1, 0); + esp_netif_t * netif = ap_netif; + esp_netif_ip_info_t ip_info; + wifi_mode_t mode; + + esp_wifi_get_mode(&mode); + if (WIFI_MODE_STA == mode) { + bits = xEventGroupWaitBits(wifi_event_group, CONNECTED_BIT, 0, 1, 0); + if (bits & CONNECTED_BIT) { + netif = sta_netif; + } else { + ESP_LOGE(TAG, "sta has no IP"); + return 0; + } + } + + esp_netif_get_ip_info(netif, &ip_info); + return ip_info.ip.addr; +} + +esp_err_t wifi_cmd_set_mode(char* mode) +{ + esp_err_t ret = ESP_FAIL; + if (!strncmp(mode, "sta", strlen("sta"))) { + ESP_ERROR_CHECK(esp_wifi_set_mode(WIFI_MODE_STA)); + ret = ESP_OK; + } else if (!strncmp(mode, "ap", strlen("ap"))) { + ESP_ERROR_CHECK(esp_wifi_set_mode(WIFI_MODE_AP)); + ret = ESP_OK; + } + + return ret; +} + +esp_err_t wifi_cmd_sta_join(const char* ssid, const char* pass) +{ + int bits = xEventGroupWaitBits(wifi_event_group, CONNECTED_BIT, 0, 1, 0); + + wifi_config_t wifi_config = { 0 }; + wifi_config.sta.pmf_cfg.capable = true; + + strlcpy((char*) wifi_config.sta.ssid, ssid, sizeof(wifi_config.sta.ssid)); + if (pass) { + strlcpy((char*) wifi_config.sta.password, pass, sizeof(wifi_config.sta.password)); + } + + if (bits & CONNECTED_BIT) { + reconnect = false; + xEventGroupClearBits(wifi_event_group, CONNECTED_BIT); + ESP_ERROR_CHECK(esp_wifi_disconnect()); + + xEventGroupWaitBits(wifi_event_group, DISCONNECTED_BIT, 0, 1, 1000 / portTICK_PERIOD_MS); + } + + reconnect = true; + ESP_ERROR_CHECK(esp_wifi_set_mode(WIFI_MODE_STA)); + ESP_ERROR_CHECK(esp_wifi_set_config(WIFI_IF_STA, &wifi_config)); + esp_wifi_connect(); + + EventBits_t status = xEventGroupWaitBits(wifi_event_group, CONNECTED_BIT, 0, 1, 5000 / portTICK_PERIOD_MS); + + if (status & CONNECTED_BIT) { + ESP_LOGI(TAG, "connect success\n"); + + return ESP_OK; + } + + ESP_LOGE(TAG, "Connect fail\n"); + reconnect = false; + return ESP_FAIL; +} + +esp_err_t wif_cmd_disconnect_wifi(void) +{ + int bits = xEventGroupWaitBits(wifi_event_group, CONNECTED_BIT, 0, 1, 0); + if (bits & CONNECTED_BIT) { + reconnect = false; + xEventGroupClearBits(wifi_event_group, CONNECTED_BIT); + ESP_ERROR_CHECK(esp_wifi_disconnect()); + + xEventGroupWaitBits(wifi_event_group, DISCONNECTED_BIT, 0, 1, portTICK_PERIOD_MS); + return ESP_OK; + } + return ESP_FAIL; +} + +esp_err_t wifi_cmd_sta_scan(const char* ssid) +{ + wifi_scan_config_t scan_config = { 0 }; + scan_config.ssid = (uint8_t *) ssid; + + ESP_ERROR_CHECK(esp_wifi_set_mode(WIFI_MODE_STA)); + esp_wifi_scan_start(&scan_config, false); + + return ESP_OK; +} + +esp_err_t wifi_cmd_ap_set(const char* ssid, const char* pass) +{ + wifi_config_t wifi_config = { + .ap = { + .ssid = "", + .ssid_len = 0, + .max_connection = 4, + .password = "", + .authmode = WIFI_AUTH_WPA_WPA2_PSK + }, + }; + + reconnect = false; + strlcpy((char*) wifi_config.ap.ssid, ssid, sizeof(wifi_config.ap.ssid)); + if (pass) { + if (strlen(pass) != 0 && strlen(pass) < 8) { + reconnect = true; + ESP_LOGE(TAG, "password less than 8"); + return ESP_FAIL; + } + strlcpy((char*) wifi_config.ap.password, pass, sizeof(wifi_config.ap.password)); + } + + if (strlen(pass) == 0) { + wifi_config.ap.authmode = WIFI_AUTH_OPEN; + } + + ESP_ERROR_CHECK(esp_wifi_set_mode(WIFI_MODE_AP)); + ESP_ERROR_CHECK(esp_wifi_set_config(WIFI_IF_AP, &wifi_config)); + return ESP_OK; +} + +esp_err_t wifi_cmd_query(void) +{ + wifi_config_t cfg = {0}; + wifi_mode_t mode; + char *query_buf = (char *)malloc(128); + + memset(&cfg, 0, sizeof(cfg)); + + esp_wifi_get_mode(&mode); + if (WIFI_MODE_AP == mode) { + esp_wifi_get_config(WIFI_IF_AP, &cfg); + ESP_LOGI(TAG, "AP mode, %s %s", cfg.ap.ssid, cfg.ap.password); + size_t lenth = sprintf(query_buf, "AP mode:%s,%s\r\n", cfg.ap.ssid, cfg.ap.password); + esp_data_back(query_buf, lenth, ENABLE_FLUSH); + } else if (WIFI_MODE_STA == mode) { + int bits = xEventGroupWaitBits(wifi_event_group, CONNECTED_BIT, 0, 1, 0); + if (bits & CONNECTED_BIT) { + esp_wifi_get_config(WIFI_IF_STA, &cfg); + ESP_LOGI(TAG, "STA mode: %s,%d,%d,%d", cfg.sta.ssid, cfg.sta.channel, cfg.sta.listen_interval, cfg.sta.threshold.authmode); + size_t lenth = sprintf(query_buf, "STA mode:%s,%d,%d,%d\r\n", cfg.sta.ssid, cfg.sta.channel, cfg.sta.listen_interval, cfg.sta.threshold.authmode); + esp_data_back(query_buf, lenth, ENABLE_FLUSH); + } else { + ESP_LOGI(TAG, "sta mode, disconnected"); + } + } else { + ESP_LOGI(TAG, "NULL mode"); + return ESP_FAIL; + } + free(query_buf); + return ESP_OK; +} + +static void smartconfig_task(void * parm) +{ + EventBits_t uxBits; + char *sm_buf = (char *)malloc(52); + ESP_ERROR_CHECK(esp_smartconfig_set_type(SC_TYPE_ESPTOUCH)); + smartconfig_start_config_t cfg = SMARTCONFIG_START_CONFIG_DEFAULT(); + wif_cmd_disconnect_wifi(); + + ESP_ERROR_CHECK(esp_smartconfig_start(&cfg)); + while (1) { + uxBits = xEventGroupWaitBits(wifi_event_group, CONNECTED_BIT | ESPTOUCH_DONE_BIT, true, false, portMAX_DELAY); + if (uxBits & CONNECTED_BIT) { + ESP_LOGI(TAG, "WiFi Connected to ap"); + } + + if (uxBits & ESPTOUCH_DONE_BIT) { + ESP_LOGI(TAG, "smartconfig over"); + size_t lenth = sprintf(sm_buf, "OK\r\n>"); + esp_data_back(sm_buf, lenth, ENABLE_FLUSH); + esp_smartconfig_stop(); + smart_config = false; + ESP_LOGI(TAG, "free the buffer taken by smartconfig"); + free(sm_buf); + vTaskDelete(NULL); + } + } +} + +esp_err_t wifi_cmd_start_smart_config(void) +{ + if (wifi_start) { + if (smart_config) { + ESP_LOGE(TAG, "SmartConfig Task is Created\r\n"); + return ESP_FAIL; + } + xTaskCreate(smartconfig_task, "smartconfig_task", 4096, NULL, 3, &Smart_Config_Handle); + ESP_LOGI(TAG, "Smart Config Task Create Success\r\n"); + smart_config = true; + return ESP_OK; + } + return ESP_FAIL; +} + +esp_err_t wifi_cmd_stop_smart_config(void) +{ + if (smart_config) { + ESP_LOGI(TAG, "stop smartconfig"); + esp_smartconfig_stop(); + smart_config = false; + ESP_LOGI(TAG, "free the buffer taken by smartconfig"); + vTaskDelete(Smart_Config_Handle); + ESP_LOGI(TAG, "delete OK\r\n"); + return ESP_OK; + } + return ESP_FAIL; +} + +void wifi_buffer_free(void *buffer, void *ctx) +{ + esp_wifi_internal_free_rx_buffer(buffer); +} + +esp_err_t wifi_recv_callback(void *buffer, uint16_t len, void *ctx) +{ + if (s_wifi_is_connected) { + esp_wifi_internal_tx(ESP_IF_WIFI_STA, buffer, len); + } + return ESP_OK; +} + +static esp_err_t pkt_wifi2usb(void *buffer, uint16_t len, void *eb) +{ + if (tinyusb_net_send_sync(buffer, len, eb, portMAX_DELAY) != ESP_OK) { + esp_wifi_internal_free_rx_buffer(eb); + } + return ESP_OK; +} + +/* Initialize Wi-Fi as sta and set scan method */ +void initialise_wifi(void) +{ + esp_log_level_set("wifi", ESP_LOG_WARN); + static bool initialized = false; + + if (initialized) { + return; + } + + ESP_ERROR_CHECK(esp_netif_init()); + wifi_event_group = xEventGroupCreate(); + ESP_ERROR_CHECK(esp_event_loop_create_default()); + ap_netif = esp_netif_create_default_wifi_ap(); + assert(ap_netif); + sta_netif = esp_netif_create_default_wifi_sta(); + assert(sta_netif); + wifi_init_config_t cfg = WIFI_INIT_CONFIG_DEFAULT(); + ESP_ERROR_CHECK(esp_wifi_init(&cfg)); + + ESP_ERROR_CHECK(esp_event_handler_register(WIFI_EVENT, ESP_EVENT_ANY_ID, wifi_event_handler, NULL)); + ESP_ERROR_CHECK(esp_event_handler_register(SC_EVENT, ESP_EVENT_ANY_ID, wifi_event_handler, NULL)); + ESP_ERROR_CHECK(esp_event_handler_instance_register(WIFI_EVENT, + WIFI_EVENT_SCAN_DONE, + &scan_done_handler, + NULL, + NULL)); + + ESP_ERROR_CHECK(esp_wifi_set_storage(WIFI_STORAGE_RAM)); + ESP_ERROR_CHECK(esp_wifi_set_mode(WIFI_MODE_STA)); + ESP_ERROR_CHECK(esp_wifi_start()); + initialized = true; +} + +#endif /* CFG_TUD_NCM || CFG_TUD_ECM_RNDIS */ diff --git a/examples/usb/device/usb_dongle/main/cmd_wifi.h b/examples/usb/device/usb_dongle/main/cmd_wifi.h new file mode 100644 index 000000000..eac6b657a --- /dev/null +++ b/examples/usb/device/usb_dongle/main/cmd_wifi.h @@ -0,0 +1,42 @@ +/* + * SPDX-FileCopyrightText: 2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#ifndef __CMD_WIFI_H +#define __CMD_WIFI_H + +#ifdef __cplusplus +extern "C" { +#endif + +void initialise_wifi(void); + +uint32_t wifi_get_local_ip(void); + +esp_err_t wifi_cmd_sta_join(const char* ssid, const char* pass); + +esp_err_t wifi_cmd_ap_set(const char* ssid, const char* pass); + +esp_err_t wifi_cmd_sta_scan(const char* ssid); + +esp_err_t wifi_cmd_query(void); + +esp_err_t wifi_cmd_set_mode(char* mode); + +esp_err_t wifi_cmd_start_smart_config(void); + +esp_err_t wifi_cmd_stop_smart_config(void); + +esp_err_t wif_cmd_disconnect_wifi(void); + +void wifi_buffer_free(void *buffer, void *ctx); + +esp_err_t wifi_recv_callback(void *buffer, uint16_t len, void *ctx); + +#ifdef __cplusplus +} +#endif + +#endif // __CMD_WIFI_H diff --git a/examples/usb/device/usb_dongle/main/data_back.c b/examples/usb/device/usb_dongle/main/data_back.c new file mode 100644 index 000000000..15e66b206 --- /dev/null +++ b/examples/usb/device/usb_dongle/main/data_back.c @@ -0,0 +1,33 @@ +/* + * SPDX-FileCopyrightText: 2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#include "data_back.h" +#include "sdkconfig.h" +#include "tinyusb.h" + +#ifndef CONFIG_UART_ENABLE +#define CONFIG_UART_ENABLE 0 +#endif + +#if CONFIG_UART_ENABLE +#include "driver/uart.h" +#define UART_NUM (CONFIG_UART_PORT_NUM) +#elif CFG_TUD_CDCACM +#include "tusb_cdc_acm.h" +#define ITF_NUM_CDC 0 +#endif + +void esp_data_back(void* data_buf, size_t lenth, bool flush) +{ +#if CONFIG_UART_ENABLE + uart_write_bytes(UART_NUM, (char*)data_buf, lenth); +#elif CFG_TUD_CDCACM + tinyusb_cdcacm_write_queue(ITF_NUM_CDC, (uint8_t*)data_buf, lenth); + if (flush) { + tinyusb_cdcacm_write_flush(ITF_NUM_CDC, 0); + } +#endif /* CONFIG_UART_ENABLE */ +} diff --git a/examples/usb/device/usb_dongle/main/data_back.h b/examples/usb/device/usb_dongle/main/data_back.h new file mode 100644 index 000000000..fc9b9ef6b --- /dev/null +++ b/examples/usb/device/usb_dongle/main/data_back.h @@ -0,0 +1,13 @@ +/* + * SPDX-FileCopyrightText: 2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#include +#include + +#define ENABLE_FLUSH 1 +#define DISABLE_FLUSH 0 + +void esp_data_back(void* data_buf, size_t lenth, bool flush); diff --git a/examples/usb/device/usb_dongle/main/idf_component.yml b/examples/usb/device/usb_dongle/main/idf_component.yml new file mode 100644 index 000000000..e6a86d849 --- /dev/null +++ b/examples/usb/device/usb_dongle/main/idf_component.yml @@ -0,0 +1,2 @@ +dependencies: + idf: ">=5.0" diff --git a/examples/usb/device/usb_dongle/main/uart.c b/examples/usb/device/usb_dongle/main/uart.c new file mode 100644 index 000000000..3073f1e6d --- /dev/null +++ b/examples/usb/device/usb_dongle/main/uart.c @@ -0,0 +1,128 @@ +/* + * SPDX-FileCopyrightText: 2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#include +#include +#include "freertos/FreeRTOS.h" +#include "freertos/task.h" +#include "freertos/queue.h" +#include "driver/uart.h" +#include "esp_log.h" + +static const char *TAG = "UART"; + +/** + * - Port: UART0 + * - Receive (Rx) buffer: on + * - Transmit (Tx) buffer: off + * - Flow control: off + * - Event queue: on + * - Pin assignment: TxD (default), RxD (default) + */ + +#define UART_TXD (CONFIG_UART_TXD) +#define UART_RXD (CONFIG_UART_RXD) +#define UART_RTS (UART_PIN_NO_CHANGE) +#define UART_CTS (UART_PIN_NO_CHANGE) + +#define UART_NUM (CONFIG_UART_PORT_NUM) +#define UART_BAUD_RATE (CONFIG_UART_BAUD_RATE) +#define UART_TASK_STACK_SIZE (CONFIG_TASK_STACK_SIZE) + +#define PATTERN_CHR_NUM (3) /*!< Set the number of consecutive and identical characters received by receiver which defines a UART pattern*/ + +#define BUF_SIZE (1024) +#define RD_BUF_SIZE (BUF_SIZE) + +void Command_Parse(char* Cmd); + +static QueueHandle_t uart_queue; + +static void uart_receive_task(void *pvParameters) +{ + uart_event_t event; + uint8_t* dtmp = (uint8_t*) malloc(RD_BUF_SIZE); + for (;;) { + //Waiting for UART event. + if (xQueueReceive(uart_queue, (void *)&event, portMAX_DELAY)) { + bzero(dtmp, RD_BUF_SIZE); + ESP_LOGI(TAG, "uart[%d] event:", UART_NUM); + switch (event.type) { + //Event of UART receving data + /*We'd better handler data event fast, there would be much more data events than + other types of events. If we take too much time on data event, the queue might + be full.*/ + case UART_DATA: + ESP_LOGI(TAG, "[UART DATA]: %d", event.size); + uart_read_bytes(UART_NUM, dtmp, event.size, portMAX_DELAY); + ESP_LOGI(TAG, "[DATA EVT]:"); + Command_Parse((char*)dtmp); + break; + //Event of HW FIFO overflow detected + case UART_FIFO_OVF: + ESP_LOGI(TAG, "hw fifo overflow"); + // If fifo overflow happened, you should consider adding flow control for your application. + // The ISR has already reset the rx FIFO, + // As an example, we directly flush the rx buffer here in order to read more data. + uart_flush_input(UART_NUM); + xQueueReset(uart_queue); + break; + //Event of UART ring buffer full + case UART_BUFFER_FULL: + ESP_LOGI(TAG, "ring buffer full"); + // If buffer full happened, you should consider encreasing your buffer size + // As an example, we directly flush the rx buffer here in order to read more data. + uart_flush_input(UART_NUM); + xQueueReset(uart_queue); + break; + //Event of UART RX break detected + case UART_BREAK: + ESP_LOGI(TAG, "uart rx break"); + break; + //Event of UART parity check error + case UART_PARITY_ERR: + ESP_LOGI(TAG, "uart parity error"); + break; + //Event of UART frame error + case UART_FRAME_ERR: + ESP_LOGI(TAG, "uart frame error"); + break; + //Others + default: + ESP_LOGI(TAG, "uart event type: %d", event.type); + break; + } + } + } + free(dtmp); + dtmp = NULL; + vTaskDelete(NULL); +} + +void initialise_uart(void) +{ + /* Configure parameters of an UART driver, + * communication pins and install the driver */ + uart_config_t uart_config = { + .baud_rate = UART_BAUD_RATE, + .data_bits = UART_DATA_8_BITS, + .parity = UART_PARITY_DISABLE, + .stop_bits = UART_STOP_BITS_1, + .flow_ctrl = UART_HW_FLOWCTRL_DISABLE, + .source_clk = UART_SCLK_APB, + }; + //Install UART driver, and get the queue. + uart_driver_install(UART_NUM, BUF_SIZE, BUF_SIZE * 4, 20, &uart_queue, 0); + uart_param_config(UART_NUM, &uart_config); + + //Set UART log level + esp_log_level_set(TAG, ESP_LOG_INFO); + //Set UART pins (using UART0 default pins ie no changes.) + ESP_ERROR_CHECK(uart_set_pin(UART_NUM, UART_TXD, UART_RXD, UART_RTS, UART_CTS)); + + //Create a task to handler UART event from ISR + xTaskCreate(uart_receive_task, "uart_receive_task", UART_TASK_STACK_SIZE, NULL, 12, NULL); +} diff --git a/examples/usb/device/usb_dongle/main/usb_dongle_main.c b/examples/usb/device/usb_dongle/main/usb_dongle_main.c new file mode 100644 index 000000000..d7a2f9714 --- /dev/null +++ b/examples/usb/device/usb_dongle/main/usb_dongle_main.c @@ -0,0 +1,132 @@ +/* + * SPDX-FileCopyrightText: 2024 Espressif Systems (Shanghai) CO LTD + * + * SPDX-License-Identifier: Apache-2.0 + */ + +#include +#include "nvs_flash.h" +#include "freertos/FreeRTOS.h" +#include "freertos/task.h" + +#include "esp_log.h" +#include "esp_err.h" +#include "esp_mac.h" +#include "sdkconfig.h" + +#include "cmd_wifi.h" + +#include "tinyusb.h" +#include "tinyusb_net.h" +#include "tusb_console.h" +#include "tusb_bth.h" + +#if CFG_TUD_CDC +#include "tusb_cdc_acm.h" +static uint8_t buf[CONFIG_TINYUSB_CDC_RX_BUFSIZE + 1]; +void Command_Parse(char* Cmd); +tinyusb_config_cdcacm_t amc_cfg; +#elif CONFIG_UART_ENABLE +void initialise_uart(void); +#endif + +#ifdef CONFIG_HEAP_TRACING +#include "esp_heap_trace.h" +#define NUM_RECORDS 100 +static heap_trace_record_t trace_record[NUM_RECORDS]; // This buffer must be in internal RAM +#endif /* CONFIG_HEAP_TRACING */ + +static const char *TAG = "USB_Dongle"; + +/* + * Register commands that can be used with FreeRTOS+CLI through the UDP socket. + * The commands are defined in CLI-commands.c. + */ +void vRegisterCLICommands(void); + +#if CFG_TUD_CDC +void tinyusb_cdc_rx_callback(int itf, cdcacm_event_t *event) +{ + /* initialization */ + size_t rx_size = 0; + + /* read */ + esp_err_t ret = tinyusb_cdcacm_read(0, buf, CONFIG_TINYUSB_CDC_RX_BUFSIZE, &rx_size); + if (ret == ESP_OK) { + buf[rx_size] = '\0'; + Command_Parse((char*)buf); + } else { + ESP_LOGE(TAG, "itf %d: itf Read error", itf); + } +} + +void tinyusb_cdc_line_state_changed_callback(int itf, cdcacm_event_t *event) +{ + int dtr = event->line_state_changed_data.dtr; + int rst = event->line_state_changed_data.rts; + ESP_LOGI(TAG, "Line state changed! itf:%d dtr:%d, rst:%d", itf, dtr, rst); +} +#endif /* CFG_TUD_CDC */ + +void app_main(void) +{ + // Initialize NVS + esp_err_t ret = nvs_flash_init(); + if (ret == ESP_ERR_NVS_NO_FREE_PAGES || ret == ESP_ERR_NVS_NEW_VERSION_FOUND) { + ESP_ERROR_CHECK(nvs_flash_erase()); + ret = nvs_flash_init(); + } + ESP_ERROR_CHECK(ret); + + ESP_LOGI(TAG, "USB initialization"); + + tinyusb_config_t tusb_cfg = { + .external_phy = false // In the most cases you need to use a `false` value + }; + + ESP_ERROR_CHECK(tinyusb_driver_install(&tusb_cfg)); + +#if CFG_TUD_NCM || CFG_TUD_ECM_RNDIS + tinyusb_net_config_t tusb_net_cfg = { + .on_recv_callback = wifi_recv_callback, + .free_tx_buffer = wifi_buffer_free, + }; + + esp_read_mac(tusb_net_cfg.mac_addr, ESP_MAC_WIFI_STA); + + tinyusb_net_init(TINYUSB_USBDEV_0, &tusb_net_cfg); + initialise_wifi(); +#endif /* CFG_TUD_NCM || CFG_TUD_ECM_RNDIS */ + +#if CFG_TUD_BTH + // init ble controller + tusb_bth_init(); +#endif /* CFG_TUD_BTH */ + +#if CFG_TUD_CDC + tinyusb_config_cdcacm_t amc_cfg = { + .usb_dev = TINYUSB_USBDEV_0, + .cdc_port = TINYUSB_CDC_ACM_0, + .rx_unread_buf_sz = 128, + .callback_rx = &tinyusb_cdc_rx_callback, // the first way to register a callback + .callback_rx_wanted_char = NULL, + .callback_line_state_changed = &tinyusb_cdc_line_state_changed_callback, + .callback_line_coding_changed = NULL + }; + + ESP_ERROR_CHECK(tusb_cdc_acm_init(&amc_cfg)); + esp_tusb_init_console(TINYUSB_CDC_ACM_0); +#elif CONFIG_UART_ENABLE + initialise_uart(); +#endif /* CFG_TUD_CDC */ + + ESP_LOGI(TAG, "USB initialization DONE"); + +#ifdef CONFIG_HEAP_TRACING + heap_trace_init_standalone(trace_record, NUM_RECORDS); + heap_trace_start(HEAP_TRACE_LEAKS); +#endif + + /* Register commands with the FreeRTOS+CLI command interpreter. */ + vRegisterCLICommands(); +} diff --git a/examples/usb/device/usb_dongle/sdkconfig.ci.bth b/examples/usb/device/usb_dongle/sdkconfig.ci.bth new file mode 100644 index 000000000..509ae896f --- /dev/null +++ b/examples/usb/device/usb_dongle/sdkconfig.ci.bth @@ -0,0 +1,4 @@ +# Only build for esp32s3 +CONFIG_IDF_TARGET="esp32s3" +CONFIG_TINYUSB_NET_MODE_NONE=y +CONFIG_TINYUSB_BTH_ENABLED=y diff --git a/examples/usb/device/usb_dongle/sdkconfig.ci.dfu b/examples/usb/device/usb_dongle/sdkconfig.ci.dfu new file mode 100644 index 000000000..fe0096eae --- /dev/null +++ b/examples/usb/device/usb_dongle/sdkconfig.ci.dfu @@ -0,0 +1 @@ +CONFIG_TINYUSB_DFU_ENABLED=y diff --git a/examples/usb/device/usb_dongle/sdkconfig.ci.ecm b/examples/usb/device/usb_dongle/sdkconfig.ci.ecm new file mode 100644 index 000000000..e55fb3a28 --- /dev/null +++ b/examples/usb/device/usb_dongle/sdkconfig.ci.ecm @@ -0,0 +1 @@ +CONFIG_TINYUSB_NET_MODE_ECM=y diff --git a/examples/usb/device/usb_dongle/sdkconfig.ci.ncm b/examples/usb/device/usb_dongle/sdkconfig.ci.ncm new file mode 100644 index 000000000..65730e3ed --- /dev/null +++ b/examples/usb/device/usb_dongle/sdkconfig.ci.ncm @@ -0,0 +1 @@ +CONFIG_TINYUSB_NET_MODE_NCM=y diff --git a/examples/usb/device/usb_dongle/sdkconfig.ci.rndis b/examples/usb/device/usb_dongle/sdkconfig.ci.rndis new file mode 100644 index 000000000..b897c133a --- /dev/null +++ b/examples/usb/device/usb_dongle/sdkconfig.ci.rndis @@ -0,0 +1 @@ +CONFIG_TINYUSB_NET_MODE_RNDIS=y diff --git a/examples/usb/device/usb_dongle/sdkconfig.defaults b/examples/usb/device/usb_dongle/sdkconfig.defaults new file mode 100644 index 000000000..807571c9a --- /dev/null +++ b/examples/usb/device/usb_dongle/sdkconfig.defaults @@ -0,0 +1,16 @@ +# This file was generated using idf.py save-defconfig. It can be edited manually. +# Espressif IoT Development Framework (ESP-IDF) 5.2.1 Project Minimal Configuration +# +CONFIG_IDF_TARGET="esp32s3" +CONFIG_ESPTOOLPY_FLASHSIZE_4MB=y +CONFIG_PARTITION_TABLE_TWO_OTA=y +CONFIG_BT_ENABLED=y +CONFIG_BT_CONTROLLER_ONLY=y +CONFIG_ESP_DEFAULT_CPU_FREQ_MHZ_240=y +CONFIG_FREERTOS_UNICORE=y +CONFIG_FREERTOS_HZ=1000 +CONFIG_TINYUSB_DEBUG_LEVEL=0 +CONFIG_TINYUSB_CDC_ENABLED=y +CONFIG_TINYUSB_CDC_RX_BUFSIZE=128 +CONFIG_TINYUSB_CDC_TX_BUFSIZE=3072 +CONFIG_TINYUSB_NET_MODE_RNDIS=y diff --git a/tools/ci/check_copyright_config.yaml b/tools/ci/check_copyright_config.yaml index 9a388da2e..b3f6f2e15 100644 --- a/tools/ci/check_copyright_config.yaml +++ b/tools/ci/check_copyright_config.yaml @@ -77,3 +77,5 @@ ignore: # You can also select ignoring files here - 'examples/usb/host/usb_cdc_4g_module/components/json_parse/**/*' - 'tools/ci/check_copyright_ignore.txt' - 'tools/cmake_utilities/test_apps/pytest_cmake_utilities.py' + - 'examples/usb/device/usb_dongle/main/CLI_Commands.c' + - 'examples/usb/device/usb_dongle/components/FreeRTOS-Plus-CLI/**/*'