From 9d550f3d02f6378700b9e79ada46c922b93ab604 Mon Sep 17 00:00:00 2001 From: Speakpig <90046247+Speakpig@users.noreply.github.com> Date: Tue, 27 Aug 2024 15:22:00 +0800 Subject: [PATCH 1/2] Update spi_full_duplex.md Fixed title indication error --- docs/spi_full_duplex.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/spi_full_duplex.md b/docs/spi_full_duplex.md index 55bf86c964..140b69373b 100644 --- a/docs/spi_full_duplex.md +++ b/docs/spi_full_duplex.md @@ -1,4 +1,4 @@ -# ESP-Hosted SPI HD (Full Duplex) Operation +# ESP-Hosted SPI FD (Full Duplex) Operation **Table of Contents** From 2ed13d36e25072106c3aae649926112e76cf63c1 Mon Sep 17 00:00:00 2001 From: Yogesh Mantri Date: Fri, 13 Sep 2024 18:37:49 +0800 Subject: [PATCH 2/2] fix(doc): Updated unform documentation --- README.md | 577 +++++-------- ..._implementation.md => bluetooth_design.md} | 0 docs/design_consideration.md | 27 +- docs/esp32_p4_function_ev_board.md | 200 ++++- docs/images/ESP-Hosted-FG-MCU_design.svg | 4 + docs/sdio.md | 602 +++++++++++--- ...p_esp_idf__latest_stable__linux_macos.fish | 65 ++ ...tup_esp_idf__latest_stable__linux_macos.sh | 62 ++ docs/spi_full_duplex.md | 622 ++++++++++++-- docs/spi_half_duplex.md | 782 +++++++++++++----- docs/troubleshooting.md | 111 ++- docs/wifi_design.md | 83 ++ 12 files changed, 2296 insertions(+), 839 deletions(-) rename docs/{bluetooth_implementation.md => bluetooth_design.md} (100%) create mode 100644 docs/images/ESP-Hosted-FG-MCU_design.svg create mode 100644 docs/setup_esp_idf__latest_stable__linux_macos.fish create mode 100644 docs/setup_esp_idf__latest_stable__linux_macos.sh create mode 100644 docs/wifi_design.md diff --git a/README.md b/README.md index 4825c977ec..ee9033b640 100644 --- a/README.md +++ b/README.md @@ -1,418 +1,267 @@ -# ESP-Hosted - -**Table of Contents** - -- [1. Introduction](#1-introduction) - - [1.1. ESP as MCU Host branch](#11-esp-as-mcu-host-branch) - - [1.2. ESP-Hosted Architecture](#12-esp-hosted-architecture) - - [1.2.1. ESP-Hosted Features](#121-esp-hosted-features) - - [1.2.2. ESP-Hosted Bluetooth Support](#122-esp-hosted-bluetooth-support) - - [1.3. Hosted on ESP32s with Native Wi-Fi](#13-hosted-on-esp32s-with-native-wi-fi) -- [2. Getting Started](#2-getting-started) - - [2.1. DevKit Specific Information](#21-devkit-specific-information) -- [3. Preparing the Host to use ESP-Hosted](#3-preparing-the-host-to-use-esp-hosted) - - [3.1. Adding required components to your ESP-IDF Project](#31-adding-required-components-to-your-esp-idf-project) - - [3.2. Disabling Native Wi-Fi Support](#32-disabling-native-wi-fi-support) -- [4. Getting the ESP-Hosted Slave Project](#4-getting-the-esp-hosted-slave-project) -- [5. Selecting the Hardware Interface for Host and Slave](#5-selecting-the-hardware-interface-for-host-and-slave) - - [5.1. Evaluating ESP-Hosted Hardware Interface](#51-evaluating-esp-hosted-hardware-interface) -- [6. Configuring the ESP-Hosted Components](#6-configuring-the-esp-hosted-components) - - [6.1. Configuring the Hosted Host](#61-configuring-the-hosted-host) - - [6.2. Configuring the Hosted Slave](#62-configuring-the-hosted-slave) - - [6.3. Flashing and Running ESP-Hosted](#63-flashing-and-running-esp-hosted) -- [7. Verifying that ESP-Hosted is Running](#7-verifying-that-esp-hosted-is-running) - - [7.1. Checking the Console Output](#71-checking-the-console-output) -- [8. Troubleshooting](#8-troubleshooting) -- [9. References](#9-references) - -## 1. Introduction - -ESP-Hosted is an open source solution that provides a way to use -Espressif SoCs and modules as a communication (slave) -co-processor. This solution provides wireless connectivity (Wi-Fi and -Bluetooth) to the host microprocessor or microcontroller, allowing it -to communicate with other devices. - -This high-level block diagram shows ESP-Hosted's relationship with the -host MCU and slave co-processor. - -![ESP-Hosted Block Dragram](docs/images/hosted_diagram-ditaa.svg) - -*ESP-Hosted Block Dragram* - -### 1.1. ESP as MCU Host branch - -This branch uses ESP chipsets as the host processor. This allows the -ESP Wi-Fi api calls to be used on ESP chipsets that don't have native -Wi-Fi. ESP-IDF applications like the ESP-IDF Iperf Example can be -built and run on the ESP32-P4 without changes, using Hosted and a -ESP32 co-processor to provide the Wi-Fi connection. - -### 1.2. ESP-Hosted Architecture - -There are two parts in the ESP-Hosted solution, the Hosted slave, -which is an ESP chip, and a host MCU, which can be a generic MCU. The -ESP-Hosted Slave provides the Wi-Fi, Bluetooth and other capabilities, -which the host MCU uses through Hosted. - -ESP-Hosted is an open-source and modular code, and uses RPC (Remote -Procedure Calls) for passing commands from the host to the slave. This -RPC mechanism allows the slave capabilities to be provided to the -host. These RPC calls are sent through a reliable bus communication, -such as SPI or SDIO or UART. - -Data (network or Bluetooth) is encapsulated at the transport layer by -Hosted and passed through the interface, minimising overhead and -delays. - -#### 1.2.1. ESP-Hosted Features - -- **any MCU can be set-up as the host**. An ESP chipset is used here - as an example. For other MCUs, you can evaluate ESP-Hosted using an - ESP chip as the host first, and then follow-up by porting ESP-Hosted - to your desired MCU. -- **any ESP chip with Wi-Fi and/or Bluetooth capabilities can be used - as the Hosted slave**. Pick the desired slave device, depending on - your product requirements. The ESP Product Selector can guide you on - the proper ESP slave selection. -- **the RCP calls used by ESP-Hosted can be extended to provide any - function required by the Host**. As long as the slave can support - it. At present, the essential ESP-IDF Wi-Fi functions have been - implemented. -- **example of ESP-Hosted in action**. It runs on the - ESP32-P4-Function-EV-Board (see [DevKit Specific - Information](#21-devkit-specific-information)), running the standard - ESP-IDF Iperf example. - -#### 1.2.2. ESP-Hosted Bluetooth Support - -See [Bluetooth Implementation](docs/bluetooth_implementation.md) for -details on Bluetooth support by ESP-Hosted. - -
- -Sequence Diagrams for Hosted System - -**Sequence Diagram for Wi-Fi on a Hosted System** - -On a ESP chipset with native Wi-Fi, a Wi-Fi api call or network data -from the application is processed internally on the chip and a Wi-Fi -response is returned to the application. - -```mermaid -sequenceDiagram - box Grey Host With Native WI-Fi - participant app as Application - participant api as ESP-IDF Wi-Fi Library - participant wifi as Wi-Fi Hardware - end - - app ->> api : esp_wifi_xxx() or Network Data - api ->> wifi : - Note over wifi : Do Wi-Fi action - wifi -->> api : Wi-Fi response or Data - api -->> app : Response or Network Data -``` - -*Native Wi-Fi Call* - -Using Wi-Remote and ESP-Hosted, the Wi-Fi api call from the -application is converted into a Hosted Call and transported to the -slave. The slave converts the Hosted Call back into an Wi-Fi api -call. The response (optionally with data) is converted into a Hosted -Response and transported back to the host. On the host, the Hosted -Response is converted back into a Wi-Fi response (optionally with -data) is returned to the application. - -For Network Data, Hosted does not do data conversion and only -encapsulates the data for transport. - -```mermaid -sequenceDiagram - box Grey Host with ESP-Hosted - participant app as Application - participant remote as Wi-Fi Remote - participant hostedh as ESP-Hosted - participant transporth as Host Transport - end - - box SlateGrey Slave ESP-Hosted - participant transports as Slave Transport - participant hosteds as Slave Hosted - participant api as ESP-IDF Wi-Fi Library - participant wifi as Wi-Fi Hardware - end - - app ->> remote : esp_wifi_xxx() - remote ->> hostedh : esp_wifi_remote_xxx() - app ->> hostedh : Network Data - Note over hostedh : add Hosted header - hostedh ->> transporth : - - transporth ->> transports : SPI/SDIO - - transports ->> hosteds : - Note over hosteds : remove Hosted header - hosteds ->> api : esp_wifi_xxx() - api ->> wifi : Wi-Fi command - hosteds ->> wifi : Network Data - Note over wifi: Do Wi-Fi action - wifi -->> hosteds : Network Data - wifi -->> api : Wi-Fi response - api -->> hosteds : Response - Note over hosteds : add Hosted header - hosteds -->> transports : - - transports -->> transporth : SPI/SDIO - - transporth -->> hostedh : - Note over hostedh : remove Hosted header - hostedh -->> app : Network Data - hostedh -->> remote : Wi-Fi Command response - remote -->> app : Response -``` - -*Hosted Wi-Fi Call* +# ESP-Hosted-MCU: Espressif SoCs as Communication Co-Processors -
+## 1 Introduction -
+ESP-Hosted-MCU is an open-source solution that allows you to use Espressif Chipsets and modules as a communication co-processor. This solution provides wireless connectivity (Wi-Fi and Bluetooth) to the host microprocessor or microcontroller, enabling it to communicate with other devices. Additionally, the user has complete control over the co-processor's resources. -Hosted on ESP32s with Native Wi-Fi +This high-level block diagram shows ESP-Hosted's relationship with the host MCU and slave co-processor. -### 1.3. Hosted on ESP32s with Native Wi-Fi +ESP-Hosted -You can also use Hosted on ESP chipsets that have native Wi-Fi. This -is useful for evaluating Hosted using current ESP chipsets before -migrating to ESP chipsets without native Wi-Fi as a host. This is -covered in more detail in the section [Disabling Native Wi-Fi -Support](#32-disabling-native-wi-fi-support). +For detailed design diagrams in Wi-Fi and Bluetooth, refer to the following design documents: -
+- [WiFi Design](docs/wifi_design.md) +- [Bluetooth Design](docs/bluetooth_design.md) -## 2. Getting Started +This branch, `feature/esp_as_mcu_host` is dedicated for any host as MCU support. If you are interested in Linux as host, please refer to [`master`](https://github.com/espressif/esp-hosted/blob/master) branch. -> [!NOTE] -> See [References](#9-references) for the links to ESP-IDF and -> ESP Registry Components +## 2 Architecture -**Set up ESP-IDF before trying Hosted**: see the *ESP-IDF Get Started -Guide* on getting and installing ESP-IDF on your operating system -(Linux, macOS, Windows). +##### Hosted Slave +This is an ESP chip that provides Wi-Fi, Bluetooth, and other capabilities. -**Getting extra IDF components**: extra ESP Components are required to -integrate ESP-Hosted into your ESP-IDF project. Both are available in -the ESP Registry: +##### Host MCU +This can be any generic microcontroller (MCU). It uses the capabilities of the Hosted Slave through Remote Procedure Calls (RPCs). The Host MCU sends these RPC commands to the Hosted Slave using a reliable communication bus, like SPI, SDIO, or UART. The Hosted Slave then handles the RPC and provides the requested functionality to the Host MCU. -- `esp_wifi_remote` (Wi-Fi Remote) -- `esp_hosted` (ESP-Hosted) +The data (network or Bluetooth) is packaged efficiently at the transport layer to minimize overhead and delays when passing between the Host and Slave. -Wi-Fi Remote is an API layer that provides the standard ESP-IDF Wi-Fi -calls to the application (`esp_wifi_init()`, etc.). Wi-Fi Remote -forwards the Wi-Fi calls to ESP-Hosted, which transports the calls as -remote requests to the slave. +This modular design allows any MCU to be used as the Host, and any ESP chip with Wi-Fi and/or Bluetooth to be used as the Hosted Slave. The RPC calls can also be extended to provide any function required by the Host, as long as the Slave can support it. -Responses are received from the slave by ESP-Hosted and returned to -Wi-Fi Remote, which returns the reponses to the calling app. To the -app, it is as if it made a standard ESP-IDF Wi-Fi API call. +## 3 Solution Flexibility -Wi-Fi events are received by ESP-Hosted from the slave and sent to the -standard ESP-IDF event loop on the host. +- **Any MCU can be the host** + - You can evaluate ESP as an example host and then port ESP-Hosted to your desired MCU. +- **Any ESP chip can be the co-processor** + - Any Wi-Fi and/or Bluetooth capable ESP chipset can be chosen as co-processor + - Choose the co-processor device based on your product requirements. The [ESP Product Selector](https://www.espressif.com/en/products/socs) can help in this. +- **Flexible transport layer (SDIO, SPI, UART)** + - ESP-Hosted supports various communication interfaces between the host and the co-processor, allowing you to choose the most suitable one for your application. + - Any other new transport also could be added to the open source code +- **Complete control over co-processor's resources** + - The user is not limited to just using the co-processor for wireless connectivity. They have complete control over the co-processor's resources, allowing for a more flexible and powerful system. +- **Extensible RPC library** + - The Remote Procedure Call (RPC) used by ESP-Hosted can be extended to provide any function required by the Host, as long as the co-processor can support it. Currently, the essential [ESP-IDF](https://github.com/espressif/esp-idf) Wi-Fi functions have been implemented. -Steps to integrate these components into your ESP-IDF project can be -found in the section [Preparing the Host to use -ESP-Hosted](#3-preparing-the-host-to-use-esp-hosted). +## 4 Quick Demo with ESP32-P4-Function-EV-Board -### 2.1. DevKit Specific Information +Impatient to test? We've got you covered! +The [ESP32-P4-Function-EV-Board](https://www.espressif.com/en/products/socs/esp32-p4) can be used as a host MCU with an on-board [ESP32-C6](https://www.espressif.com/en/products/socs/esp32-c6) as co-processor, already connected via SDIO as transport. +Prerequisite: You need to have an ESP32-P4-Function-EV-Board` -ESP-Hosted comes with a default configuration that can be directly -used with the following ESP DevKits: +### 4.1 Set-Up ESP-IDF -- [ESP32-P4-Function-EV-Board](docs/esp32_p4_function_ev_board.md) - -## 3. Preparing the Host to use ESP-Hosted - -### 3.1. Adding required components to your ESP-IDF Project - -Check your project's `idf_component.yml` file. If it does not contain -`espressif/esp_wifi_remote` or `espressif/esp_hosted` as dependencies, -you can add them to your project: +- Windows + - Install and setup ESP-IDF on Windows as documented in the [Standard Setup of Toolchain for Windows](https://docs.espressif.com/projects/esp-idf/en/latest/esp32/get-started/windows-setup.html). + - Use the ESP-IDF [Powershell Command Prompt](https://docs.espressif.com/projects/esp-idf/en/latest/esp32/get-started/windows-setup.html#using-the-command-prompt) to move to expected +- Linux or MacOS + - bash +```bash +bash docs/setup_esp_idf__latest_stable__linux_macos.sh ``` -idf.py add-dependency "espressif/esp_wifi_remote" -idf.py add-dependency "espressif/esp_hosted" + - fish +```fish +fish docs/setup_esp_idf__latest_stable__linux_macos.fish ``` -
- -3.2. Disabling Native Wi-Fi Support +### 4.2 Set-Up P4 with C6 +The host, ESP32-P4, lacks native Wi-Fi/Bluetooth support. Our [Quick Demo](docs/esp32_p4_function_ev_board.md) will help you run iperf over P4--SDIO--C6. -### 3.2. Disabling Native Wi-Fi Support +### 4.3 Don't Have ESP32-P4-Function-EV-Board? -For ESP Chipsets with native Wi-Fi support (the ESP32 series, for -example), you also have to disable native Wi-Fi before ESP-Hosted can used. +No worries if you don't have an ESP32-P4. In fact, most users don't. You can choose and use any two ESP chipsets/SoCs/Modules/DevKits. DevKits are convenient to use as they have GPIO headers already in place. From these two ESP chipsets, one would act as host and another as slave/co-processor. However, as these are not connected directly, you would need to manually connect some transport, which is explained later in the section [`Detailed Setup`](#7-detailed-setup). -To do this, edit the ESP-IDF -`components/soc//include/soc/Kconfig.soc_caps.in` file and change -all `WIFI` related configs to `n`. For example: +## 5 Source Code and Dependencies -``` -config SOC_WIFI_SUPPORTED - bool - # default y # original configuration - default n -``` - -This should be done for all `SOC_WIFI_xxx` configs found in the file. - -For ESP Chipsets without native Wi-FI, `SOC_WIFI_xxx` configs will be -`n` by default. +### 5.1 ESP-Hosted-MCU Source Code -
- -## 4. Getting the ESP-Hosted Slave Project - -The ESP-Hosted slave project can be checked out from the ESP-Hosted -Component's example project: +- ESP-Hosted-MCU code can be found at Espressif Registry Component [`esp_hosted` (ESP-Hosted)](https://components.espressif.com/components/espressif/esp_hosted) or GitHub repo at [`ESP-Hosted`](https://github.com/espressif/esp-hosted/tree/feature/esp_as_mcu_host) +- ESP-Hosted repo clone is **not** required if you have ESP as host. + - Reason: [ESP component manager](https://docs.espressif.com/projects/esp-idf/en/stable/esp32/api-guides/tools/idf-component-manager.html) automatically clones esp-hosted component while building. +- However, For non-ESP host development, you can clone the repo using command: +```bash +git clone --recurse-submodules \ +--branch feature/esp_as_mcu_host --depth 1 \ +https://github.com/espressif/esp-hosted \ esp_hosted_mcu ``` -idf.py create-project-from-example "espressif/esp_hosted:slave" -``` - -## 5. Selecting the Hardware Interface for Host and Slave -ESP-Hosted can currently use the SPI or SDIO (on selected ESP chips) -interfaces. See the following pages for more information on these -interfaces: +### 5.2 Dependencies + +ESP-Hosted-MCU Solution is dependent on `ESP-IDF`, `esp_wifi_remote` and `protobuf-c` + +###### ESP-IDF + - [`ESP-IDF`](https://github.com/espressif/esp-idf) is the development framework for Espressif SoCs supported on Windows, Linux and macOS + - ESP-Hosted-MCU solution is based on ESP-IDF as base software. ESP chipsets as host and slave always tried to design such a way that ESP-IDF components are re-used. + - Although, We totally understand, host MCUs in case of non-ESP chipset may not desire to be dependent on ESP-IDF. The port layer is written to avoid suc dependencies. Some crucial ESP-IDF components could also be just copy-pasted to fast-track the non-ESP host development. + +###### Wi-Fi Remote + - [`esp_wifi_remote`](https://components.espressif.com/components/espressif/esp_wifi_remote) i.e. 'Wi-Fi Remote' is very thin interface made up of ESP-IDF Wi-Fi APIs with empty weak definitions. Real definitions for these APIs are provided by ESP-Hosted-MCU + - Wi-Fi Remote Code can be found at either [GitHub Repo](https://github.com/espressif/esp-protocols/tree/master/components/esp_wifi_remote) or [Espressif Registry Component](https://components.espressif.com/components/espressif/esp_wifi_remote) + + +###### Protobuf + - [`protobuf-c`](https://github.com/protobuf-c/protobuf-c) is data serialization framework provided by Google. RPC messages communicated in host and slave are protobuf encoded. + - It helps to avoid manual serialization or endien-ness conversion. + - Provides Flexibility for users to port the ESP-Hosted-MCU RPC framework in any protobuf supported programming language + - Code is checked-out as submodule at `common/protobuf-c` + +##### 5.2.1 How Dependencies Work Together (short explanation) +- RPC Request - Response + - Wi-Fi Remote is an API layer or interface that provides the standard ESP-IDF Wi-Fi calls to the application (`esp_wifi_init()`, etc.) + - Wi-Fi Remote forwards the Wi-Fi calls to ESP-Hosted, as ESP-Hosted 'implements' tha APIs provided by Wi-Fi Remote interface. + - ESP-Hosted host MCU creates RPC requests which are protobuf encoded and sends over the transport (SPI/SDIO etc) to the slave. + - Slave de-serialize the protobuf RPC request and response send back to host over transport, again with protobuf serialised. + - Responses received at transport returned to Wi-Fi Remote, which returns the reponses to the calling app at host + - To the app, it is as if it made a standard ESP-IDF Wi-Fi API call. +- RPC Event + - Asynchronous Wi-Fi events when subscribed, are sent by slave to host. + - These events terminate in standard ESP-IDF event loop on the host +- Please note, Only RPC i.e. control packets are serialised. Data Packets are never serialised as they do not need endien conversion. + + +## 6. Decide the communication bus in between host and slave + +The communication bus is required to be setup correctly between host and slave. +We refer this as `transport medium` or simply `transport`. + +ESP-Hosted-MCU supports SPI/SDIO/UART transports. User can choose which transport to use. Choosing specific transport depends on factors: high performance, easy and quick to test, number of GPIOs used, or simply co-processor preference + +Below is chart for the transport medium comparison. + +Legends: + +- `FD` : Full duplex communication +- `HD` : Half duplex communication +- `BT` : Bluetooth +- `+2` in column `Num of GPIOs` + - There are two GPIOs additional applicable for all the transports + - (1) Co-Processor reset: Host needs one additional pin to connect to `RST`/`EN` pin of co-processor, to reset on bootup + - (2) Ground: Grounds of both chipsets need to be connected. + - If you use jumper cable connections, connect as many grounds as possible in between two boards for better noise cancellation. +- `Any_Slave` + - Co-processor suppored: ESP32, ESP32-C2, ESP32-C3, ESP32-C5, ESP32-C6, ESP32-S2, ESP32-S3 + - Classic ESP32 supports 'Classic BT', 'BLE 4.2' & 'BTDM' + - Rest all chipsets support BLE only. BLE version supported is 5.0+. Exact bluetooth versions could be refered from [ESP Product Selector Page](https://products.espressif.com/#/product-selector) +- `Dedicated platforms` + - Bluetooth uses dedicated platform, UART and Wi-Fi uses any other base transport + - In other platforms, Bluetooth and Wi-Fi re-use same platform and hence use less GPIOs and less complicated + - This transport combination allows Bluetooth to use dedicated uart transportt with additional 2 or 4 depending on hardware flow control. +- (S) : Sheild box reading +- (O) : Over the air reading +- TBD : To be determined +- iperf : iperf2 with test resukts in mbps -- [SPI Full Duplex interface](docs/spi_full_duplex.md) -- [SDIO interface](docs/sdio.md) -- [SPI Half Duplex interface](docs/spi_half_duplex.md) +**Host can be any ESP chipset or any non-ESP MCU.** -### 5.1. Evaluating ESP-Hosted Hardware Interface +###### Hosted Transports table -For evaluating ESP-Hosted, it is recommended to use the SPI interface -as it is easier to prototype using jumper cables. SDIO provides the -highest throughput but has tighter hardware requirements and requires -using the Host and Slave on a PCB to work correctly. +| Transport | Type | Num of GPIOs | Setup with | Co-processor supported | Host Tx iperf | Host Rx iperf | Remarks | +|:---------------:|:-----:|:------------:|:----------------:|:--------------:|:------------:|:-----------:|:--------------------------:| +| Standard SPI | FD | 6 | jumper or PCB | Any_Slave | udp: 24 tcp: 22 | udp: 25 tcp: 22| Simplest solution for quick test | +| Dual SPI | HD | 5 | jumper or PCB | Any_Slave | udp: 32 tcp: 26 (O) | udp: 33 tcp: 25 (O) | Better throughput, but half duplex | +| Quad SPI | HD | 7 | PCB only | Any_Slave | udp: 41 tcp: 29 (O) | udp: 42 tcp: 28 (O) | Due to signal integrity, PCB is mandatory | +| SDIO 1-Bit | HD | 4 | jumper or PCB | ESP32, ESP32-C6 | TBD | TBD | Stepping stone for PCB based SDIO 4-bit | +| SDIO 4-Bit | HD | 6 | PCB only | ESP32, ESP32-C6 | udp: 79.5 tcp: 53.4 (S) | udp: 68.1 tcp: 44 (S) | Highest performance | +| Only BT over UART | FD | 2 or 4 | jumper or PCB | Any_Slave | NA | NA | Dedicated Bluetooth over UART pins | +| Dedicated platforms | FD | Extra 2 or 4 | jumper or PCB | Any_Slave | NA | NA | UART dedicated for BT & Wi-Fi on any other transport | -## 6. Configuring the ESP-Hosted Components +With jumper cables, 'Standard SPI' and 'Dual SPI' solutions are easiest to evaluate, without much of hardware dependencies. SDIO 1-Bit can be tested with jumper cables, but it needs some additional hardware config, such as installation of external pull-up registers. -Ensure that both Host and Slave are configured to use the same -interface (SPI, SDIO) and GPIOs have been selected to match the -hardware connections. +In case case of dedicated platforms, Blutooth uses standard HCI over UART. In rest of cases, Bluetooth and Wi-Fi uses same transport and hence less GPIOs and less complicated. In shared mode, bluetooth runs as vHCI (multiplexed mode) -> [!NOTE] -> You can use a lower clock speed to verify the connections. For SPI, -> you can try 10 MHz. For SDIO you can use a clock speed between 400 -> kHz to 20 MHz. The actual clock used is determined by the -> hardware. Use an oscilloscope to check the actual clock frequency -> used. +## 7 ESP-Hosted-MCU Header -> [!NOTE] -> For SDIO testing, you can set the SDIO Bus Width to 1-Bit. In 1-Bit -> mode, only `DAT0` and `DAT1` signals are used for data and are less -> affected by noise on the signal lines. +### 7.1 ESP Hosted header -### 6.1. Configuring the Hosted Host +Host and slave always populate below header at the start of every frame, irrespective of actual or dummy data in payload. -To configure the host, go to the directory where your ESP-IDF project -is located and execute: +| Field | Type | Bits | Mandatory? | Description | +|----------------|----------|------|------------|-----------------------------------------------------------------------------| +| if_type | uint8_t | 4 | M | Interface type | +| if_num | uint8_t | 4 | M | Interface number | +| flags | uint8_t | 8 | M | Flags for additional information | +| len | uint16_t | 16 | M | Length of the payload | +| offset | uint16_t | 16 | M | Offset for the payload | +| checksum | uint16_t | 16 | M | Checksum for error detection (0 if checksum disabled) | +| seq_num | uint16_t | 16 | O | Sequence number for tracking packets (Useful in debugging) | +| throttle_cmd | uint8_t | 0 or 2 | O | Flow control command | +| reserved2 | uint8_t | 6 or 8 | M | Reserved bits | +| reserved3 | uint8_t | 8 | M | Reserved byte (union field) | +| hci\_pkt\_type or priv\_pkt\_type | uint8_t | 8 | M | Packet type for HCI interface (union field) | + +### 7.2 Interface Types + +Start of header states which type of frame is being carried. + +| Interface Type | Value | Description | +|----------------------|-------|--------------------------------------------------| +| ESP\_INVALID\_IF | 0 | Invalid interface | +| ESP\_STA\_IF | 1 | Station frame | +| ESP\_AP\_IF | 2 | SoftAP frame | +| ESP\_SERIAL\_IF | 3 | Control frame | +| ESP\_HCI\_IF | 4 | Bluetooth vHCI frame | +| ESP\_PRIV\_IF | 5 | Private communication between slave and host | +| ESP\_TEST\_IF | 6 | Transport throughput test | +| ESP\_ETH\_IF | 7 | Invalid | +| ESP\_MAX\_IF | 8 | type mentioned in dummy or empty frame | + +## 8 Detailed Setup + +Once you decided the transport to use, this section should guide how to set this transport, with hardware connections, configurations and verification. Users can evaluate one transport first and then move to other. + +> [!IMPORTANT] +> +> [Design Considerations](docs/design_consideration.md) that could be reffered to, before you stick to any transport option. Referring to these consideration would help to get you faster to solution, make your design stable and less error-prone. -```sh -idf.py set-target -idf.py menuconfig -``` - -The configuration options for Hosted Host can be found under **Component -config** ---> **ESP-Hosted config**. - -### 6.2. Configuring the Hosted Slave - -To configure the slave, go to the directory where you have checked out -the Hosted slave example project and execute: -```sh -idf.py set-target -idf.py menuconfig -``` -The configuration options for the Hosted Slave can be found under **Example Configuration**. - -### 6.3. Flashing and Running ESP-Hosted +Irrespective of transport chosen, following steps are needed, which are step-wise explained in each transport. -Use the standard ESP-IDF `idf.py` to build, flash and (optionally) -monitor the debug output from the console on both the host and slave: - -```sh -idf.py build -idf.py -p flash monitor -``` +1. Set-up the hosted-transport +2. Slave Flashing + - Slave project creation + - Slave configuration + - Slave flashing + - Slave logs +3. Host flashing + - Host project integration with ESP-IDF example + - Host configuration + - Host flashing + - Host logs + +- [**Standard SPI (Full duplex)**](docs/spi_full_duplex.md) + +- [**SPI - Dual / Quad Half Duplex**](docs/spi_half_duplex.md) + +- [**SDIO (1-Bit / 4-Bit)**](docs/sdio.md) + +- [**UART**](docs/uart.md) + +## 9 Examples +Check [examples](./examples) directory for sample applications using ESP-Hosted. + - `examples/bleprph_host_only_vhci` + - Bluetooth without needing extra GPIOs -## 7. Verifying that ESP-Hosted is Running +## 10 Troubleshooting -> [!NOTE] -> If you are building and connecting the ESP-Hosted Host and Slave -> on the same development system, make sure to verify which -> Serial Port refers to the Host and Slave. +If you encounter issues with using ESP-Hosted, see the following guide: -### 7.1. Checking the Console Output +- [Troubleshooting Guide](docs/troubleshooting.md) -Once both systems have been flashed and running, you should see output -similar to the following output on the console after start-up. For -example, if you are using the SPI Interface: +## 11 References -For ESP-Hosted Master: +- [ESP Product Selector Page](https://products.espressif.com) +- [ESP-IDF Get Started Guide](https://docs.espressif.com/projects/esp-idf/en/latest/esp32/get-started) +- [ESP-IDF Wi-Fi API](https://docs.espressif.com/projects/esp-idf/en/latest/esp32/api-reference/network/esp_wifi.html) +- [ESP-IDF Iperf Example](https://github.com/espressif/esp-idf/tree/master/examples/wifi/iperf) +- [ESP-IDF NimBLE](https://github.com/espressif/esp-nimble) +- [ESP Component Registry](https://components.espressif.com) +- [Registry Component: esp\_wifi\_remote](https://components.espressif.com/components/espressif/esp_wifi_remote) +- [Registry Component: esp\_hosted](https://components.espressif.com/components/espressif/esp_hosted) -``` -I (522) transport: Attempt connection with slave: retry[0] -I (525) transport: Reset slave using GPIO[54] -I (530) os_wrapper_esp: GPIO [54] configured -I (535) gpio: GPIO[54]| InputEn: 0| OutputEn: 1| OpenDrain: 0| Pullup: 0| Pulldown: 0| Intr:0 -I (1712) transport: Received INIT event from ESP32 peripheral -I (1712) transport: EVENT: 12 -I (1712) transport: EVENT: 11 -I (1715) transport: capabilities: 0xe8 -I (1719) transport: Features supported are: -I (1724) transport: - HCI over SPI -I (1728) transport: - BLE only -I (1732) transport: EVENT: 13 -I (1736) transport: ESP board type is : 13 - -I (1741) transport: Base transport is set-up -``` -For ESP-Hosted Slave: -``` -I (492) fg_mcu_slave: ********************************************************************* -I (501) fg_mcu_slave: ESP-Hosted-MCU Slave FW version :: 0.0.6 - -I (511) fg_mcu_slave: Transport used :: SPI only -I (520) fg_mcu_slave: ********************************************************************* -I (529) fg_mcu_slave: Supported features are: -I (534) fg_mcu_slave: - WLAN over SPI -I (538) h_bt: - BT/BLE -I (541) h_bt: - HCI Over SPI -I (545) h_bt: - BLE only -``` -## 8. Troubleshooting -If you encounter issues with using ESP-Hosted, see the following guide: -- [Troubleshooting Guide](docs/troubleshooting.md) -## 9. References - -- ESP Product Selector: https://products.espressif.com/ -- ESP-IDF Get Started Guide: -https://docs.espressif.com/projects/esp-idf/en/latest/esp32/get-started/ -- ESP-IDF Wi-Fi API: - https://docs.espressif.com/projects/esp-idf/en/latest/esp32/api-reference/network/esp_wifi.html -- ESP-IDF Iperf Example: - https://github.com/espressif/esp-idf/tree/master/examples/wifi/iperf -- ESP-IDF NimBLE: https://github.com/espressif/esp-nimble -- ESP Registry: https://components.espressif.com/ -- `esp_wifi_remote` Component: - https://components.espressif.com/components/espressif/esp_wifi_remote -- esp_hosted` Component: https://components.espressif.com/components/espressif/esp_hosted diff --git a/docs/bluetooth_implementation.md b/docs/bluetooth_design.md similarity index 100% rename from docs/bluetooth_implementation.md rename to docs/bluetooth_design.md diff --git a/docs/design_consideration.md b/docs/design_consideration.md index ab128ba354..5633d5b9a1 100644 --- a/docs/design_consideration.md +++ b/docs/design_consideration.md @@ -58,14 +58,27 @@ connected together. Verify that the correct GPIOs are set-up in > under Hosted. Check the ESP datasheet to verify the GPIOs you select > can be used as a Hosted interface. -### 2.2. Using SPI insted of SDIO +### 2.2. Evaluate with jumpers first -In general, SPI imposes fewer hardware requirements compared to +It is flexible to evaluate with jumper cables or bread board than full-fledged PCB. +In general, SPI (Standard & Dual SPI) imposes fewer hardware requirements compared to SDIO. SPI is easier to prototype, and available on more ESP chips and MCUs compared to SDIO. - -However, if you need a high-speed interface, SDIO should be -considered. +Before going to SDIO 4 bit mode PCB, it's better to evaluate SDIO 1-Bit mode. + +Once you evaluate the solution on jumper cables, you can move to PCB solutions with same or high performance transport. + +###### Jumper cable considerations +- Use high quality jumper cables +- Use jumper cables as small as possible. you can cut and solder the joints if need be. +- Use equal length jumper cables for all the connections +- Grounds: Connect as many grounds as possible, this lowers the interference. +- Jumper cable lengths + - Standard SPI: At max 10cm, lower the better + - Dual SPI: At max 10cm, lower the better + - SDIO 1 Bit: At max 5cm, lower the better + - Quad SPI : jumpers not supported, only PCB + - SDIO 4 Bit: Jumpers not supported, only PCB ### 2.3. Whenever possible, Use `IO_MUX` GPIOs. @@ -102,7 +115,7 @@ For jumper cables, you can try surrounding the signals, especially the > [!NOTE] > For SDIO, external pull-up resistors (recommended value: 51 kOhms) -> are required. Using jumper cable are not recommended for SDIO. You +> are required. Using jumper cable are **not** recommended for SDIO. You > may be able to get SDIO working with jumper cables by using a lower > `CLK` frequency and using 1-bit SDIO mode. @@ -130,7 +143,7 @@ be used to light a LED or trigger a capture on an oscilloscope or logic analyzer, for example. This is useful for capturing rare or intermittent conditions while testing Hosted. -In the future, Hosted may also offer more features, like controlling +In the future, Hosted may also offer newer transport options or more features, like controlling power modes on the Host and Slave. These may require additional GPIOs for control, so it would be good to keep some additional GPIOs available and accesable for future use. diff --git a/docs/esp32_p4_function_ev_board.md b/docs/esp32_p4_function_ev_board.md index 1c62679f12..0e89e9ec4b 100644 --- a/docs/esp32_p4_function_ev_board.md +++ b/docs/esp32_p4_function_ev_board.md @@ -1,51 +1,116 @@ # ESP-Hosted on the ESP32-P4-Function-EV-Board DevKit -**Table of Contents** - -- [1. Introduction](#1-introduction) -- [2. Building ESP-Hosted as Host for the P4](#2-building-esp-hosted-as-host-for-the-p4) - - [2.1. Adding Wifi Remote and Hosted Components](#21-adding-wifi-remote-and-hosted-components) - - [2.2. Building the Firmware](#22-building-the-firmware) -- [3. Checking that ESP-Hosted is Running](#3-checking-that-esp-hosted-is-running) -- [4. Flashing the On-board ESP32-C6 using ESP-Prog](#4-flashing-the-on-board-esp32-c6-using-esp-prog) -- [5. Troubleshooting](#5-troubleshooting) -- [6. Flashing the On-board ESP32-P4 through the Serial Interface](#6-flashing-the-on-board-esp32-p4-through-the-serial-interface) -- [7. References](#7-references) +
+**Table of Contents** + +- [Introduction](#1-introduction) +- [Set-Up ESP-IDF](#2-set-up-esp-idf) +- [Building Host for the P4](#3-building-host-for-the-p4) + - [Adding Components](#31-adding-components) + - [Configuring Defaults](#32-configuring-defaults) + - [Building Firmware](#33-building-firmware) +- [Checking ESP-Hosted](#4-checking-esp-hosted) +- [Flashing ESP32-C6](#5-flashing-esp32-c6) + - [Using ESP-Prog](#51-using-esp-prog) + - [OTA Updates](#52-ota-updates) +- [Troubleshooting](#6-troubleshooting) +- [Flashing the On-board ESP32-P4 through the ESP-Prog](#7-flashing-esp32-p4) +- [References](#8-references) + +
## 1. Introduction -This page documents using ESP-Hosted on the -ESP32-P4-Function-EV-Board. The board comes with an on-board ESP32-C6 -module, pre-flashed with ESP-Hosted slave code (v0.0.6). The board -provides a Wi-Fi connection to the on-board ESP32-P4, which acts as -the host. +This page documents using ESP-Hosted-MCU on the ESP32-P4-Function-EV-Board. The board comes with an on-board ESP32-C6 module, pre-flashed with ESP-Hosted-MCU slave code (v0.0.6). The board provides a Wi-Fi connection to the on-board ESP32-P4, which acts as the host. The image below shows the board. - +ESP32-P4-Function-EV-Board *ESP32-P4-Function-EV-Board* The ESP32-P4 communicates with the ESP32-C6 module using SDIO. -## 2. Building ESP-Hosted as Host for the P4 +## 2. Set-Up ESP-IDF + +As you have reached here, it is highly likely that you have already setup ESP-IDF. + +If not done, Please set up ESP-IDF: + +#### Option 1: Installer Way + +- **Windows** + - Install and setup ESP-IDF on Windows as documented in the [Standard Setup of Toolchain for Windows](https://docs.espressif.com/projects/esp-idf/en/latest/esp32/get-started/windows-setup.html). + - Use the ESP-IDF [Powershell Command Prompt](https://docs.espressif.com/projects/esp-idf/en/latest/esp32/get-started/windows-setup.html#using-the-command-prompt) for subsequent commands. + +- **Linux or MacOS** + - For bash: + ```bash + bash docs/setup_esp_idf__latest_stable__linux_macos.sh + ``` + - For fish: + ```fish + fish docs/setup_esp_idf__latest_stable__linux_macos.fish + ``` + +#### Option 2: Manual Way -### 2.1. Adding Wifi Remote and Hosted Components +Please follow the [ESP-IDF Get Started Guide](https://docs.espressif.com/projects/esp-idf/en/latest/esp32/get-started/index.html) for manual installation. -The Wi-Fi service is provided to the ESP32-P4 using the -`esp_wifi_remote` component. Check your project's `idf_component.yml` -file) to see if it is already present. If not, you can add this -component to your project: +## 3. Building Host for the P4 + +### 3.1. Adding Components + +Add `esp_wifi_remote` and `esp_hosted` components to the project: ``` idf.py add-dependency "espressif/esp_wifi_remote" +idf.py add-dependency "espressif/esp_hosted" +``` + +Remove 'esp-extconn' if present in `main/idf_component.yml`, as esp-extconn and esp-hosted cannot work together. +Open the `main/idf_component.yml` file and remove/comment the following block if present: + +``` +# ------- Delete or comment this block --------- +espressif/esp-extconn: + version: "~0.1.0" + rules: + - if: "target in [esp32p4]" +# ----------------------------------- ``` -The `esp_wifi_remote` component has a dependency on ESP-Hosted and -will also add the `esp_hosted` component to your build. +It is always good to use `esp_wifi_remote` as it provides all the Wi-Fi config and a wrapper abstraction layer. +But you can also evaluate without using it. -### 2.2. Building the Firmware +### 3.2. Configuring Defaults + +Edit the `sdkconfig.defaults.esp32p4` file such that, it would have following content: + +``` +#### Comment below two lines if present: +# CONFIG_ESP_HOST_WIFI_ENABLED=y +# CONFIG_PARTITION_TABLE_SINGLE_APP_LARGE=y + +#### Add Wi-Fi Remote config for better performance: +CONFIG_ESP_WIFI_STATIC_RX_BUFFER_NUM=16 +CONFIG_ESP_WIFI_DYNAMIC_RX_BUFFER_NUM=64 +CONFIG_ESP_WIFI_DYNAMIC_TX_BUFFER_NUM=64 +CONFIG_ESP_WIFI_AMPDU_TX_ENABLED=y +CONFIG_ESP_WIFI_TX_BA_WIN=32 +CONFIG_ESP_WIFI_AMPDU_RX_ENABLED=y +CONFIG_ESP_WIFI_RX_BA_WIN=32 + +CONFIG_LWIP_TCP_SND_BUF_DEFAULT=65534 +CONFIG_LWIP_TCP_WND_DEFAULT=65534 +CONFIG_LWIP_TCP_RECVMBOX_SIZE=64 +CONFIG_LWIP_UDP_RECVMBOX_SIZE=64 +CONFIG_LWIP_TCPIP_RECVMBOX_SIZE=64 + +CONFIG_LWIP_TCP_SACK_OUT=y +``` + +### 3.3. Building Firmware Set the ESP32-P4 as the target, build, flash the firmware and (optionally) monitor ESP32-P4 console output: @@ -53,10 +118,10 @@ Set the ESP32-P4 as the target, build, flash the firmware and ```sh idf.py set-target esp32p4 idf.py build -idf.py -p flash monitor` +idf.py -p flash monitor ``` -## 3. Checking that ESP-Hosted is Running +## 4. Checking ESP-Hosted When the P4 is running with Hosted, you should see console output similar to this after start-up: @@ -114,17 +179,68 @@ I (1848) H_SDIO_DRV: Received INIT event I (1868) rpc_wrap: Received Slave ESP Init ``` -## 4. Flashing the On-board ESP32-C6 using ESP-Prog +## 5. Flashing ESP32-C6 +ESP32-C6 flashing is totally **optional**, as C6 is expected to be pre-flashed with ESP-Hosted slave firmware, 0.0.6. If you wish to get updated ESP-Hosted slave firmware, you can flash it using two ways, Either with ESP-Prog on ESP32-C6, or using OTA update configured using web server. + +### 5.1 OTA Updates + +To update the ESP32-C6 slave module using Over-The-Air (OTA) updates, follow these steps: + +1. Build the ESP-Hosted slave firmware for the ESP32-C6 module: + +``` +idf.py create-project-from-example "espressif/esp_hosted:slave" +``` + +2. Set the target and start `Menuconfig`: + +```sh +idf.py set-target esp32c6 +idf.py menuconfig +``` + +3. Under **Example Configuration**, ensure that the Hosted transport + selected is `SDIO`. + +4. Build the firmware: + +```sh +idf.py build +``` + +5. Upload the firmware (the build/network_adapter.bin file) to a server or a local directory accessible via HTTP. + +6. On the ESP32-P4 host, add the following code to your application to initiate the OTA update: + +``` +#include "esp_hosted.h" + +esp_err_t esp_hosted_ota_start(const char *url); +``` + +7. Call the `esp_hosted_ota_start` function with the URL of the firmware binary: + +``` +esp_err_t err = esp_hosted_ota_start("http://example.com/path/to/network_adapter.bin"); +if (err != ESP_OK) { + ESP_LOGE(TAG, "Failed to start OTA update: %s", esp_err_to_name(err)); +} +``` + +8. Monitor the console output to see the OTA update progress. + +### 5.2 Using ESP-Prog > [!NOTE] > ESP-Prog is only required if you want to flash firmware to the > ESP32-C6 module using the standard ESP Tools. +This step is optional, as C6 is expected to be pre-flashed with ESP-Hosted slave firmware, 0.0.6. + The image below shows the board with an ESP-Prog connected to the header to communicate with the on-board ESP32-C6.. - +ESP32-P4-Function-EV-Board with ESP-Prog Connected to ESP32-C6 *ESP32-P4-Function-EV-Board with ESP-Prog Connected to ESP32-C6* @@ -133,7 +249,6 @@ ESP32-C6 module using ESP-Prog, follow these steps: 1. Check out the ESP-Hosted slave example project: - ``` idf.py create-project-from-example "espressif/esp_hosted:slave" ``` @@ -174,10 +289,17 @@ prevent the P4 interfering with the C6 while flashing (by asserting the C6 Reset signal during the firmware download), set the P4 into Bootloader mode before flashing the firmware to the C6: +###### Manual Way 1. hold down the `BOOT` button on the board 2. press and release the `RST` button on the board 3. release the `BOOT` button +###### Script Way + +```sh +esptool.py -p --before default_reset --after no_reset run +``` + You can now flash the firmware to the C6 (and monitor the console output): @@ -185,17 +307,18 @@ output): idf.py -p flash monitor ``` -## 5. Troubleshooting + + +## 6. Troubleshooting If you encounter issues with using ESP-Hosted, see the following guide: - [Troubleshooting Guide](troubleshooting.md)
- Flashing the On-board ESP32-P4 through the Serial Interface -## 6. Flashing the On-board ESP32-P4 through the Serial Interface +## 7. Flashing the On-board ESP32-P4 through the ESP-Prog The USB connector on the board is the standard method for flashing the firmware to the P4. An alternative method is to flash the P4 through @@ -204,8 +327,7 @@ its serial interface using a ESP-Prog. The image below shows the connection between the ESP-Prog and the serial port pins on the P4 header for programming. - +ESP32-P4 Serial Connection with ESP-Prog *ESP32-P4 Serial Connection with ESP-Prog* @@ -236,7 +358,7 @@ on the board.
-## 7. References +## 8. References - ESP32-P4-Function-EV-Board: https://docs.espressif.com/projects/esp-dev-kits/en/latest/esp32p4/esp32-p4-function-ev-board/ - ESP-Prog: https://docs.espressif.com/projects/esp-iot-solution/en/latest/hw-reference/ESP-Prog_guide.html diff --git a/docs/images/ESP-Hosted-FG-MCU_design.svg b/docs/images/ESP-Hosted-FG-MCU_design.svg new file mode 100644 index 0000000000..f09bfbd1c4 --- /dev/null +++ b/docs/images/ESP-Hosted-FG-MCU_design.svg @@ -0,0 +1,4 @@ + + + +
SPI or SDIO
SPI or SDIO
Socket APIs
Socket APIs
Bluetooth App
Bluetooth A...
RPC APIs
(Req / Resp / Event)
RPC APIs...
User App
User App
Standard HCI over UART (Optional)

Standard HCI over UART (Optional)...
Bluetooth
Packets
Bluetooth...
Network
Packets
Network...
Control
Path
Control...
Transport host driver (SPI/SDIO)
Transport host driver (SPI/SDIO)
Protobuf
Protobuf
STA_IF or AP_IF
STA_IF or AP_IF
TCP/IP &
TLS stack
TCP/IP &...
Net If
Net If
SERIAL_IF
SERIAL_IF
vHCI
vHCI
Bluetooth Host Stack
Bluetooth Ho...
RPC Lib
RPC Lib
HCI_IF  (vHCI driver)
HCI_IF  (vHCI driver)
Host Driver
Host Driver
MCU SPI / SDIO driver
MCU SPI / SDIO driver
MCU UART driver (O)
MCU UART driver (...
IDF LWIP Dr.(O)
IDF LWIP Dr.(O)
IDF WiFi Dr.
IDF WiFi Dr.
MCU UART Ctrl (O)
MCU UART Ctrl (O)
MCU SPI/SDIO Ctrl
MCU SPI/SDIO Ctrl
User Code
User Code
Middle layer
Middle layer
ESP-Hosted-MCU driver
ESP-Hosted-MCU driver
HAL driver
HAL driver
Hardware
Hardware
Network
Driver
Network...
RPC server
RPC server
Transport slave driver (SPI/SDIO)
Transport slave driver (SPI/SDIO)
Slave Driver
Slave Driver
Host MCU
Host...
ESP Slave
ESP Slave
ESP UART Ctrl
ESP UART Ctrl
ESP BT Ctrl
ESP BT Ctrl
ESP WiFi Ctrl
ESP WiFi Ctrl
IDF SPI/SDIO Dr.
IDF SPI/SDIO Dr.
Protocomm Dr.
Protocomm Dr.
ESP SPI/SDIO Ctrl
ESP SPI/SDIO Ctrl
vHCI Driver
vHCI Driver
ESP-Hosted-MCU components
ESP-Hosted-MCU compon...
Ready-Made drivers or APIs
Ready-Made drivers...
ESP IDF components
ESP IDF components
Hardware Components
Hardware Componen...
(O) : Optional
(O) : Optional
Control
Path
Control...
Data
Path
Data...
Bluetooth:
Bluetooth:
HCI
HCI
vHCI
vHCI
SPI / SDIO
SPI / SDIO
Legends
Legends
Reset
Reset
UART (O)
UART (O)
\ No newline at end of file diff --git a/docs/sdio.md b/docs/sdio.md index 7224a2a9ec..63ccb3978e 100644 --- a/docs/sdio.md +++ b/docs/sdio.md @@ -1,153 +1,543 @@ # ESP-Hosted SDIO Operation -**Table of Contents** +Sections 3 below covers the hardware requirements like external pull-up requirement, possible efuse burning for co-processor and other hardware aspects to consider for SDIO. -- [1. Introduction](#1-introduction) -- [2. SDIO Configuration](#2-sdio-configuration) - - [2.1. Extra GPIO Signals Required](#21-extra-gpio-signals-required) -- [3. Hardware Considerations](#3-hardware-considerations) - - [3.1. Pull-up Resistors](#31-pull-up-resistors) - - [3.2 PBC Traces](#32-pbc-traces) - - [3.3. Conflicts between SDIO and ESP hardware.](#33-conflicts-between-sdio-and-esp-hardware) - - [3.4. Testing the SDIO Connection](#34-testing-the-sdio-connection) - - [3.5. Summary of SDIO Hardware Requirements](#35-summary-of-sdio-hardware-requirements) - - [4. References](#4-references) +Section 4 to 8 covers the complete step-wise setup co-processor and host with SDIO, for 1-bit and 4-bit SDIO. -## 1. Introduction +If you wish to skip the theory, you can refer the [Quick Start Guide](#1-quick-start-guide) below. For quick navigation, please unfold the Table of Contents below. -SDIO is a high-speed bus. It uses the same SDMMC hardware protocol -used for SD Cards, but with its own set of commands for communicating -with SDIO aware peripherals. +
+**Table of Contents** +1. [Quick Start Guide](#1-quick-start-guide) + +2. [Introduction](#2-introduction) + +3. [Hardware Considerations](#3-hardware-considerations) || [3.1 General Considerations](#31-general-considerations) || [3.2 Pull-up Resistors](#32-pull-up-resistors) || [3.3 Voltage Levels & eFuse burning](#33-voltage-levels--efuse-burning) || [3.4 Jumper Wires](#34-jumper-wires) || [3.5 PCB Design](#35-pcb-design) || [3.6 Advanced Considerations](#36-advanced-considerations) || [3.7 Testing Connections](#37-testing-connections) + +4. [Hardware Setup](#4-hardware-setup) + +5. [Set-Up ESP-IDF](#5-set-up-esp-idf) + +6. [Flashing the Co-processor](#6-flashing-the-co-processor) || [6.1 Create Co-processor Project](#61-create-co-processor-project) || [6.2 Co-processor Config](#62-co-processor-config) || [6.3 Co-processor Build](#63-co-processor-build) || [6.4 Co-processor Flashing](#64-co-processor-flashing) + +7. [Flashing the Host](#7-flashing-the-host) || [7.1 Select Example to Run in Hosted Mode](#71-select-example-to-run-in-hosted-mode) || [7.2 Host Project Component Configuration](#72-host-project-component-configuration) || [7.3 Menuconfig, Build and Flash Host](#73-menuconfig-build-and-flash-host) + +8. [Testing and Troubleshooting](#8-testing-and-troubleshooting) + +9. [References](#9-references) + +
+ +## 1. Quick Start Guide + +This section provides a brief overview of how to get started with ESP-Hosted using SDIO mode. For detailed instructions on each step, please refer to the following sections: + +- [4. Hardware Setup](#4-hardware-setup) +- [5. Set-Up ESP-IDF](#5-set-up-esp-idf) +- [6. Flashing the Co-processor](#6-flashing-the-co-processor) +- [7. Flashing the Host](#7-flashing-the-host) +- [8. Testing and Troubleshooting](#8-testing-and-troubleshooting) + +These sections will guide you through the process of flashing both the slave and host devices, setting up the hardware connections, and verifying successful communication. + +## 2. Introduction + +SDIO is a high-speed bus that uses the same SDMMC hardware protocol used for SD Cards, but with its own set of commands for communicating with SDIO aware peripherals. > [!NOTE] > Only some ESP32 chips support the SDIO Protocol: > -> - SDIO as Slave: ESP32-C6 +> - SDIO as Slave: ESP32, ESP32-C6 > - SDIO as Master: ESP32, ESP32-S3, ESP32-P4 -## 2. SDIO Configuration -For the ESP32 and ESP32-C6, the pin assignments for SDIO are fixed: +## 3. Hardware Considerations + +### 3.1 GPIO Configuration for SDIO + +The SDIO interface can use almost any GPIO pins. For maximum speed and minimal delays, it is recommended to select the SDIO pin configuration that uses the dedicated `IO_MUX` pins. Hardware connections in later sections use `IO_MUX` pins, as much as possible. +ESP32 only supports `IO_MUX` pins for SDIO. other chips may support other flexible pins using GPIO_Matrix, with small performance penalty. + +### 3.2 Extra GPIO Signals Required -| Signal | ESP32 GPIO | ESP32-C6 GPIO | -| :---- | :---: | :---: | -| CLK | 14 | 19 | -| CMD | 15 | 18 | -| DAT0 | 2 | 20 | -| DAT1 | 4 | 21 | -| DAT2 | 12 | 22 | -| DAT3 | 13 | 23 | +Extra GPIO signals are required for SDIO on Hosted and can be assigned to any free GPIO pins: -For the ESP32-S3 and ESP32-P4, any GPIO pin can be configured for use -in SDIO. +- `Reset` signal: an output signal from the host to the slave. When asserted, the host resets the slave. This is done when ESP-Hosted is started on the host, to synchronise the state of the host and slave. > [!NOTE] -> Check the ESP chip documentation for GPIO limitations that may -> prevent some GPIOs from being used for SDIO. +> The `Reset` signal suggested to connect to the `EN` or `RST` pin on the slave, It is however configurable to use another GPIO pin. +> +> To configure this, use `idf.py menuconfig` on the Slave: **Example configuration** ---> **SDIO Configuration** ---> **Host SDIO GPIOs** and set **Slave GPIO pin to reset itself**. -To enable SDIO on the Host and Slave using `Menuconfig`: -1. On Host: **Component config** ---> **ESP-Hosted config** ---> - **Transport layer** and choose **SDIO**. -2. On Slave: **Example configuration** ---> **Transport layer** and - choose **SDIO**. +### 3.3 General Hardware Considerations -### 2.1. Extra GPIO Signals Required +- For SDIO, signal integrity is crucial, hence jumper wires are not recommended. +- Jumper wires are only suitable for initial testing and prototyping. +- If you wish, you can test SDIO 1-Bit mode using jumper cables, only for initial testing and prototyping. Pull-Ups are still mandatory for all, [CMD, DAT0, DAT1, DAT2, DAT3] irrespective how do you connect, using jumpers or PCB. +- Ensure equal trace lengths for all SDIO connections, whether using jumper wires or PCB traces. +- Very strict requirement, to keep wires as short as possible, under 5 cm. Smaller the better. +- Use the lower clock frequency like 5 MHz for evaluation. Once solution verified, optimise the clock frequency in increasing steps to max possible value. Max SDIO slave clock frequency for all SDIO slaves is 50 MHz. +- Provide proper power supply for both host and co-processor devices. Lower or incorrect power supplies can cause communication issues & suboptimal performance. -Extra GPIO signal are required for SDIO on Hosted. These can be -assigned to any free GPIO pins: +### 3.4 Pull-up Resistors +- SDIO requires external pull-up resistor (51 kOhm recommended) and clean signals for proper operation. +- For this reason, it is not recommended to use jumper cables. Use PCB traces to connect between a Hosted Master and Slave. +- For full requirements, refer to ESP-IDF SDIO pull-up resistor requirements at [Pull-Up Requirements](https://docs.espressif.com/projects/esp-idf/en/latest/esp32/api-reference/peripherals/sd_pullup_requirements.html). -- `Reset` signal: an output signal from the host to the slave. When - asserted, the host resets the slave. This is done when ESP-Hosted is - started on the host, to synchronise the state of the host and slave. +### 3.5 Voltage Levels & eFuse burning +- SDIO expects all signals to be at 3.3V level. If you are using level shifter, ensure that the level shifter output is set to 3.3V. +- If you use classic ESP32, there is good chance that you would need to burn the eFuse. +- eFuse burning is one time and **non reversible process**. You may brick your device, if burn the eFuse incorrectly. +- Please check full documentation at [eFuse burning](https://docs.espressif.com/projects/esp-idf/en/latest/esp32/api-reference/peripherals/sd_pullup_requirements.html) If your chip is listed explicitly, not to burn eFuse, you can ignore this. +- This document covers below issues and their solutions: + - External pull-ups to be used on: CMD, DAT0, DAT1, DAT2, DAT3, with 51K Ohm recommended, irrespective of jumpers or PCB. + - Bootstrapping pin and DAT2 voltage issues and solution of eFuse burning, with complete procedure. -> [!NOTE] -> The `Reset` signal can be configured to connect to the `EN` or `RST` -> pin on the slave, or assigned to a GPIO pin on the slave. -> -> To configure this, use `Menuconfig` on the Slave: **Example -> configuration** ---> **SDIO Configuration** and set **Slave GPIO pin -> to reset itself**. -## 3. Hardware Considerations +### 3.6 Jumper Wires (only for SDIO 1-Bit mode) + +- External Pull-ups mandatory for CMD, DAT0, DAT1, DAT2, DAT3 of 51 kOhm. +- Smaller the better, strictly under 5 cm. All equal length. +- Use high-quality, low-capacitance jumper wires. +- Arrange wires to minimize crosstalk, especially for clock and data lines. +- Possibly, use twisted pairs for clock and data lines to reduce electromagnetic interference. +- If possible, use a ground wire between every signal wire to improve signal integrity. +- Connect as many grounds as possible to improve common ground reference and reduce ground noise. + +### 3.7 PCB Design + +For optimal performance and reliability in production designs: -SDIO has several hardware requirements that must be met for proper -operation. +- Ensure equal trace lengths for all SDIO signals (CLK, CMD, DAT0, DAT1, DAT2, DAT3) as much as possible. This practice, known as length matching, is crucial for maintaining signal integrity and reducing timing skew, especially at higher frequencies. +- If perfect length matching is not possible, prioritize matching the clock (CLK) trace length with the data lines. +- Use controlled impedance traces for high-speed signals. +- Place bypass capacitors close to the power pins of both the host and slave devices. +- Consider using series termination resistors on the clock and data lines to reduce reflections. +- For high-speed designs, use a 4-layer PCB with dedicated power and ground planes. -### 3.1. Pull-up Resistors +### 3.8 Advanced Considerations -SDIO requires external pull-up resistor (51 kOhm recommended) and -clean signals for proper operation. For this reason, it is not -recommended to use jumper cables. Use PCB traces to connect between a -Hosted Master and Slave. +- Calculate the maximum allowed trace length based on your clock frequency and PCB material. +- Consider the capacitive load on the SDIO bus, especially for longer traces or when using multiple slave devices. +- For very high-speed designs, consider using differential signaling techniques. +- Implement proper EMI/EMC design techniques to minimize electromagnetic interference. -### 3.2 PBC Traces +## 4. Hardware Setup -The PCB traces for SDIO should be equal length and kept as short as -possible. The SDIO signals, especially the `CLK` signal, should be -isolated from other signals using a ground plane to minimise -crosstalk. +Setting up the hardware involves connecting the master and slave devices via the SDIO pins and ensuring all extra GPIO signals are properly connected. Below is the table of connections for the SDIO setup between a host ESP chipset and another ESP chipset as co-processor: -If you must test SDIO using jumper cables, and you provide pull-up -resistors to the SDIO lines, you may be able to get SDIO working by -using a lower frequency and operating in 1-Bit SDIO mode. See -[3.4. Testing the SDIO Connection](#34-testing-the-sdio-connection) -for more information. -### 3.3. Conflicts between SDIO and ESP hardware. -SDIO requires pull-ups on signal lines which may conflict with the -pull state of hardware pins required for bootstrapping. For example -there is a conflict between the SDIO `DAT2` line and bootstrap -requirement for the EPS32 with the 3.3 V flash chip. This can be fixed -by burning the flash voltage selection eFuses. +### Host connections +| Signal | ESP32 | ESP32-S3 | ESP32-P4-Function-EV-Board | +|-------------|-------|----------|----------| +| CLK | 14 | 19 | 18 | +| Reset Out | 5 | 42 | 54 | +| CMD | 15+[ext-pull-up](#34-pull-up-resistors) | 47+[ext-pull-up](#34-pull-up-resistors) | 19+[ext-pull-up](#34-pull-up-resistors) | +| DAT0 | 2+[ext-pull-up](#34-pull-up-resistors) | 13+[ext-pull-up](#34-pull-up-resistors) | 14+[ext-pull-up](#34-pull-up-resistors) | +| DAT1 | 4+[ext-pull-up](#34-pull-up-resistors) | 35+[ext-pull-up](#34-pull-up-resistors) | 15+[ext-pull-up](#34-pull-up-resistors) | +| DAT2 | 12+[ext-pull-up](#34-pull-up-resistors) | 20+[ext-pull-up](#34-pull-up-resistors) | 16+[ext-pull-up](#34-pull-up-resistors) | +| DAT3 | 13+[ext-pull-up](#34-pull-up-resistors) | 9+[ext-pull-up](#34-pull-up-resistors) | 17+[ext-pull-up](#34-pull-up-resistors) | -> [!WARNING] -> eFuse burning is irreversible and may cause your hardware to stop -> functioning. Check the documentation in -> [4. References](#4-references) carefully to make sure that burning -> the eFuses is the correct option. -### 3.4. Testing the SDIO Connection -**Using a Lower Clock Speed** +### Slave connections -You can use a lower clock speed to verify the connections. Start with -a clock speed between 400 kHz to 20 MHz. +| Signal | ESP32 | ESP32-C6 | +|-------------|-------|----------| +| CLK | 14 | 19 | +| CMD | 15 | 18 | +| DAT0 | 2 | 20 | +| DAT1 | 4 | 21 | +| DAT2 | 12 | 22 | +| DAT3 | 13 | 23 | +| Reset In | EN | EN/RST | -To configure this, use `Menuconfig` on the Host: **Component -config** ---> **ESP-Hosted config** ---> **Hosted SDIO Configuration** -and set **SDIO Clock Freq (in kHz)**. > [!NOTE] -> The actual clock frequency used is determined by the hardware. Use -> an oscilloscope or logic analyser to check the clock frequency. +> +> - 1. Try to use IO_MUX pins from the datasheet for optimal performance on both sides. +> - 2. These GPIO assignments are based on default Kconfig configurations. You can modify these in the menuconfig for both host and slave if needed. +> - 3. Once ported, any other host with standard SDIO can be used. +> - 4. ESP32, ESP32-S3, and ESP32-P4 can be used as hosts; ESP32 and ESP32-C6 can be used as slaves in SDIO mode. +> - 5. External pull-ups are mandatory + +## 5. Set-Up ESP-IDF + +Before setting up the ESP-Hosted co-processor & host for SDIO mode, ensure that ESP-IDF is properly installed and set up on your system. + +### 5.1 Installer Way -**3.4.2. Using 1-bit SDIO Mode** +- **Windows** + - Install and setup ESP-IDF on Windows as documented in the [Standard Setup of Toolchain for Windows](https://docs.espressif.com/projects/esp-idf/en/latest/esp32/get-started/windows-setup.html). + - Use the ESP-IDF [Powershell Command Prompt](https://docs.espressif.com/projects/esp-idf/en/latest/esp32/get-started/windows-setup.html#using-the-command-prompt) for subsequent commands. -You can set the SDIO Bus Width to 1-Bit. In 1-Bit mode, only `DAT0` -and `DAT1` signals are used for data and are less affected by noise on -the signal lines. This can help you verify that the SDIO protocol is -working at the logical level, if you have issues getting 4-Bit SDIO to -work on your prototype board. +- **Linux or MacOS** + - For bash: + ```bash + bash docs/setup_esp_idf__latest_stable__linux_macos.sh + ``` + - For fish: + ```fish + fish docs/setup_esp_idf__latest_stable__linux_macos.fish + ``` -To configure this, use `Menuconfig` on the Host: **Component config** ----> **ESP-Hosted config** ---> **Hosted SDIO Configuration** ---> -**SDIO Bus Width** to **1 Bit**. +### 5.2 Manual Way + +Please follow the [ESP-IDF Get Started Guide](https://docs.espressif.com/projects/esp-idf/en/latest/esp32/get-started/index.html) for manual installation. + +## 6. Flashing the Co-processor + +| Supported Co-processor Targets | ESP32 | ESP32-C6 | +| ------------------------------ | ----- | -------- | + +There are four steps to flash the ESP-Hosted slave firmware: + +### 6.1 Create Co-processor Project + +1. Navigate to the directory where you want to create the co-processor project. +2. Use the following command to create a new project: + ```bash + idf.py create-project + ``` + Replace `` with your desired project name. + +### 6.2 Co-processor Config + +1. Navigate to the project directory: + ```bash + cd + ``` +2. Configure the project: + ```bash + idf.py menuconfig + ``` + +#### 6.2.1 Transport config + - Navigate to "Example configuration" -> "Transport layer" + - Select "SDIO" + +#### 6.2.2 Any other config + - Optionally, Configure any additional SDIO-specific settings like co-processor GPIOs, SDIO Mode, SDIO timing,etc. + +###### Generated files +- Generated config files are (1) `sdkconfig` file and (2) internal `sdkconfig.h` file. +- Please note, any manually changes done to these generated files, would not take effect. + +###### Defaulting specific config (Optional) +- This is advanced option, so please be careful. +- To mark some config options as default, you can add specific config line in file, `sdkconfig.defaults.`. So whenever next time building, you do not need to re-configure. + +### 6.3 Co-processor Build + +1. Build the project: + ```bash + idf.py build + ``` + +### 6.4 Co-processor Flashing + +There are two methods to flash the ESP-Hosted slave firmware: + +#### 6.4.1 Serial Flashing (Initial Setup) + +For the initial setup or when OTA is not available, use serial flashing. + +Flash the co-processor firmware using +``` +idf.py -p flash +``` > [!NOTE] -> Pull-ups are still required on `DAT2` and `DAT3` lines to prevent -> the SDIO slave from going into SPI mode upon startup. +> +> If you are not able to flash the co-processor, there might be a chance that host is not allowing to to do so. +> +> Put host in bootloader mode using following command and then retry flashing the co-processor +> +> ```bash +> esptool.py -p **** --before default_reset --after no_reset run +> ``` + +Monitor the output (optional): +``` +idf.py -p monitor +``` + +##### 6.4.2. Co-processor OTA Flashing (Subsequent Updates) + +For subsequent updates, you can re-use ESP-Hosted-MCU transport, as it should be already working. While doing OTA, Complete co-processor firmware image is not needed and only co-processor application partition, 'network_adapter.bin' need to be re-flashed remotely from host. + +1. Ensure your co-processor device is connected and communicating with the host with existing ESP-Hosted-MCU. + +2. Create a web server +You can re-use your existing web server or create a new locally for testing. Below is example to do it. + - Make a new directory so that web server can be run into it and navigate into it + - Create simple local web server using python3 + + ```bash + python3 -m http.server 8080 + ``` +3. Copy the co-processor app partition `network_adapter.bin` in the directory where you created the web server. + - The `network_adapter.bin` can be found in your co-processor project build at `/build/network_adapter.bin` + +4. Verify if web server is set-up correctly + - Open link `http://127.0.0.1:8080` in the browser and check if network_adapter.bin is available. + - Right click and copy the complete URL of this network_adapter.bin and note somewhere. + +5. On the **host side**, use the `esp_hosted_ota` function to initiate the OTA update: + + ```c + #include "esp_hosted_api.h" + + const char* image_url = "http://example.com/path/to/network_adapter.bin"; //web server full url + esp_err_t ret = esp_hosted_ota(image_url); + if (ret == ESP_OK) { + printf("co-processor OTA update failed[%d]\n", ret); + } + ``` + + This function will download the firmware in chunk by chunk as http client from the specified URL and flash it to the co-processor device through the established transport. + In above web server example, You can paste the copied url earlier. + + +6. Monitor the OTA progress through the console output on both the host and co-processor devices. -### 3.5. Summary of SDIO Hardware Requirements +> [!NOTE] +> +> - The `esp_hosted_ota` function is part of the ESP-Hosted-MCU API and handles the OTA process through the transport layer. +> - Ensure that your host application has web server connectivity to download the firmware file. +> - The co-processor device doesn't need to be connected to the web server for this OTA method. + +## 7. Flashing the Host + +| Supported Host Targets | Any ESP chipset | Any Non-ESP chipset | +| ----------------------- | --------------- | ------------------- | + +Any host having SDIO master can be used as host. Please make sure the hardware configurations, like external pull-ups are installed correctly. Tthe voltage at SDIO pins is expected to be 3v3 volts. +- ESP chipsets as SDIO master + - ESP as host could be one of ESP32, ESP32-S3, ESP32-P4. + - For ESP32 as host, may need additional **eFuse burning** for voltage correction on one of data pin. ESP32-S3 and ESP32-P4 does **not** need this. +- Non ESP SDIO Master + - Any other host having SDIO master can be used as host. Please make sure the hardware configurations, like ([external Pull-up Resistors](#42-pull-up-resistors)) are installed correctly. Tthe voltage at SDIO pins is expected to be 3v3 volts. +- Pull-ups required for CMD, DAT0, DAT1, DAT2, DAT3 lines (for both 1-Bit and 4-Bit SDIO) +- eFuse burning may be required for classic ESP32. +- Pull-Up and eFuse burning is detailed in [(3) Hardware Considerations](#3-hardware-considerations) + +### 7.1 Select Example to Run in Hosted Mode + +Select an example from the [ESP-IDF examples directory](https://github.com/espressif/esp-idf/tree/master/examples) that you wish to run in ESP-Hosted mode. All Wi-Fi and Bluetooth examples are supported. For simplicity and demonstration purposes, we will use the [ESP-IDF iperf example](https://github.com/espressif/esp-idf/tree/master/examples/wifi/iperf). + +### 7.2 Host Project Component Configuration + +Now that ESP-IDF is set up, follow these steps to prepare the host: + +###### 1. Navigate to the iperf example in your ESP-IDF directory: + ``` + cd $IDF_PATH/examples/wifi/iperf + ``` + +###### 2. Dependency components + Add the required components to the project's `idf_component.yml` file: + ``` + idf.py add-dependency "espressif/esp_wifi_remote" + idf.py add-dependency "espressif/esp_hosted" + ``` + +###### 3. Remove conflicting configuration + Open the `main/idf_component.yml` file and remove/comment the following block if present: + ``` + # ------- Delete or comment this block --------- + espressif/esp-extconn: + version: "~0.1.0" + rules: + - if: "target in [esp32p4]" + # ----------------------------------- + ``` + This step is necessary because esp-extconn and esp-hosted cannot work together. + +###### 4. Disable native Wi-Fi if available + If your host ESP chip already has native Wi-Fi support, disable it by editing the `components/soc//include/soc/Kconfig.soc_caps.in` file and changing all `WIFI` related configs to `n`. + + If you happen to have both, host and slave as same ESP chipset type (for example two ESP32-C2), note an [additional step](docs/troubleshooting/#1-esp-host-to-evaluate-already-has-native-wi-fi) + + +### 7.3 Menuconfig, Build and Flash Host + +##### 1. High performance configurations + This is optional step, suggested for high performance applications. + + If using ESP32-P4 as host: + - Remove the default `sdkconfig.defaults.esp32p4` file. + - Create a new `sdkconfig.defaults.esp32p4` file with the following content: + ``` + CONFIG_ESP_WIFI_STATIC_RX_BUFFER_NUM=16 + CONFIG_ESP_WIFI_DYNAMIC_RX_BUFFER_NUM=64 + CONFIG_ESP_WIFI_DYNAMIC_TX_BUFFER_NUM=64 + CONFIG_ESP_WIFI_AMPDU_TX_ENABLED=y + CONFIG_ESP_WIFI_TX_BA_WIN=32 + CONFIG_ESP_WIFI_AMPDU_RX_ENABLED=y + CONFIG_ESP_WIFI_RX_BA_WIN=32 + + CONFIG_LWIP_TCP_SND_BUF_DEFAULT=65534 + CONFIG_LWIP_TCP_WND_DEFAULT=65534 + CONFIG_LWIP_TCP_RECVMBOX_SIZE=64 + CONFIG_LWIP_UDP_RECVMBOX_SIZE=64 + CONFIG_LWIP_TCPIP_RECVMBOX_SIZE=64 + + CONFIG_LWIP_TCP_SACK_OUT=y + ``` + + For other hosts also, you can merge above configs in corresponding `sdkconfig.defaults.esp32XX` file. + +###### 2. Set environment for your host ESP chip: + + ``` + idf.py set-target + ``` + Replace `` with your specific ESP chip (e.g., esp32, esp32s3, esp32p4). + +###### 3. Flexible Menuconfig configurations + + ``` + idf.py menuconfig + ``` + ESP-Hosted-MCU host configurations are available under "Component config" -> "ESP-Hosted config" + 1. Select "SDIO" as the transport layer + 2. Change Slave chipset to connect to under "Slave chipset to be used" + 3. Change SDIO Bus Width to 1-bit or 4-bit based on the co-processor using "Hosted SDIO Configuration" -> "SDIO Bus Width" + 4. Optionally, configure SDIO-specific settings like: + + - SDIO Host GPIO Pins + + - Lower SDIO Clock Speed + You can use a lower clock speed to verify the connections. Start with a clock speed between 400 kHz to 20 MHz. + To configure this, use `Menuconfig` on the Host: **Component config** ---> **ESP-Hosted config** ---> **Hosted SDIO Configuration** and set **SDIO Clock Freq (in kHz)**. + > [!NOTE] + > + > The actual clock frequency used is determined by the hardware. Use an oscilloscope or logic analyzer to check the clock frequency. + + - Using 1-bit SDIO Mode + By default, SDIO operates in 4-Bit mode. + You can set the SDIO Bus Width to 1-Bit. In 1-Bit mode, only `DAT0` and `DAT1` signals are used for data and are less affected by noise on the signal lines. This can help you verify that the SDIO protocol is working at the logical level, if you have issues getting 4-Bit SDIO to work on your prototype board. + + To configure this, use `Menuconfig` on the Host: **Component config** ---> **ESP-Hosted config** ---> **Hosted SDIO Configuration** ---> **SDIO Bus Width** to **1 Bit**. + + - SDIO Mode + Packet or Streaming mode could be used, but slave has to use same SDIO mode used. -> [!IMPORTANT] -> - using jumper cables is not recommended -> - external pull-up are required. 10 kOhm resistors are recommended -> - check for conflicts between SDIO pull-up and GPIO requirements -> - for ESP32, check if eFuse burning is required +> [!NOTE] + +> Pull-ups are still required on `DAT2` and `DAT3` lines to prevent +> the SDIO slave from going into SPI mode upon startup. -### 4. References +After confirming the functionality of the 1-Bit SDIO mode, you can revert to the 4-Bit mode with PCB to benefit from increased data transfer rates. Using the previous configuration, switch back to `4 Bit`. + + +###### 4. Build the project: + ``` + idf.py build + ``` + +###### 5. Flash the firmware: + ``` + idf.py -p flash + ``` + +###### 6. Monitor the output: + ``` + idf.py -p monitor + ``` + - If host was put into bootloader mode earlier, it may need manual reset + +## 8. Testing and Troubleshooting + +After flashing both the co-processor and host devices, follow these steps to connect and test your ESP-Hosted SDIO setup: + +1. Connect the hardware: + - Follow the pin assignments for SDIO as specified in [Hardware Setup](#4-hardware-setup). + - Ensure all necessary connections are made, including power, ground, and the extra GPIO signals (Data_Ready and Reset). + +2. Power on both devices. Apply correct input rating power for both chipsets. + +3. Verify the connection: + - Check the serial output of both devices for successful initialization messages. + - Look for messages indicating that the SDIO transport layer has been established. + +4. Logs at both sides: + - Host: + + ``` + I (522) transport: Attempt connection with slave: retry[0] + I (525) transport: Reset slave using GPIO[54] + I (530) os_wrapper_esp: GPIO [54] configured + I (535) gpio: GPIO[54]| InputEn: 0| OutputEn: 1| OpenDrain: 0| Pullup: 0| Pulldown: 0| Intr:0 + I (1712) transport: Received INIT event from ESP32 peripheral + I (1712) transport: EVENT: 12 + I (1712) transport: EVENT: 11 + I (1715) transport: capabilities: 0xe8 + I (1719) transport: Features supported are: + I (1724) transport: - HCI over SDIO + I (1728) transport: - BLE only + I (1732) transport: EVENT: 13 + I (1736) transport: ESP board type is : 13 + + I (1741) transport: Base transport is set-up + ``` + + - Co-processor: + + ``` + I (492) fg_mcu_slave: ********************************************************************* + I (501) fg_mcu_slave: ESP-Hosted-MCU Slave FW version :: X.Y.Z + + I (511) fg_mcu_slave: Transport used :: SDIO + I (520) fg_mcu_slave: ********************************************************************* + I (529) fg_mcu_slave: Supported features are: + I (534) fg_mcu_slave: - WLAN over SDIO + I (538) h_bt: - BT/BLE + I (541) h_bt: - HCI Over SDIO + I (545) h_bt: - BLE only + ``` + +5. Test basic functionality: + - The iperf example automatically attempts to connect to the configured Wi-Fi network. Watch the serial output for connection status. + - If the automatic connection fails, you can manually initiate a Wi-Fi scan and connection: + ``` + sta_scan + sta_connect + ``` +6. Additional commands to test: + - Get IP address: `sta_ip` + - Disconnect from Wi-Fi: `sta_disconnect` + - Set Wi-Fi mode: `wifi_mode ` (where mode can be 'sta', 'ap', or 'apsta') + +7. Advanced iperf testing: + Once connected, you can run iperf tests: + + | Test Case | Host Command | External STA Command | + |-----------|--------------|----------------------| + | UDP Host TX | `iperf -u -c -t 60 -i 3` | `iperf -u -s -i 3` | + | UDP Host RX | `iperf -u -s -i 3` | `iperf -u -c -t 60 -i 3` | + | TCP Host TX | `iperf -c -t 60 -i 3` | `iperf -s -i 3` | + | TCP Host RX | `iperf -s -i 3` | `iperf -c -t 60 -i 3` | + + Note: Replace `` with the IP address of the external STA, and `` with the IP address of the ESP-Hosted device. + +8. Troubleshooting: + - Consider using a lower clock speed or checking your [hardware setup](docs/sdio.md#7-hardware-setup) if you experience communication problems. + - ESP-Hosted-MCU troubleshooting guide: [docs/troubleshooting.md](docs/troubleshooting.md) + +9. Monitoring and debugging: + - Use the serial monitor on both devices to observe the communication between the host and co-processor. + - For more detailed debugging, consider using a logic analyzer to examine the SDIO signals. + - Use a logic analyzer or oscilloscope to verify the SDIO signals. + - Ensure that the power supply to both devices is stable and within the required voltage levels. + +## 9. References + +- [ESP-IDF Programming Guide](https://docs.espressif.com/projects/esp-idf/en/latest/esp32/) +- [ESP32 Hardware Design Guidelines](https://www.espressif.com/en/products/hardware/esp32/resources) +- [SDIO Protocol Basics](https://en.wikipedia.org/wiki/Serial_Peripheral_Interface) +- [ESP SDIO Slave Communication](https://docs.espressif.com/projects/esp-idf/en/stable/esp32/api-reference/protocols/esp_sdio_slave_protocol.html) -- ESP-IDF SD Pull-up Requirements: -https://docs.espressif.com/projects/esp-idf/en/latest/esp32/api-reference/peripherals/sd_pullup_requirements.html diff --git a/docs/setup_esp_idf__latest_stable__linux_macos.fish b/docs/setup_esp_idf__latest_stable__linux_macos.fish new file mode 100644 index 0000000000..8502d4ad6e --- /dev/null +++ b/docs/setup_esp_idf__latest_stable__linux_macos.fish @@ -0,0 +1,65 @@ +#!/usr/bin/env fish + +echo "Setting up ESP-IDF using fish" +set SHELL_RC "$HOME/.config/fish/config.fish" + +# Step 1: Check if curl and git are installed +echo "============== Step 1: Checking if dependencies ==========================" +if not type -q curl + echo "curl is not installed. Please install curl." + exit 1 +end + +if not type -q git + echo "git is not installed. Please install git." + exit 1 +end + +echo "Dependencies for current script are installed." + +# Step 2: Fetch the branches from the GitHub API and find the latest stable release branch +echo "============== Step 2: Fetching branch list from ESP-IDF GitHub API ==============" +set LATEST_BRANCH (curl -s https://api.github.com/repos/espressif/esp-idf/branches | grep -o '"name": "release/[^"]*' | awk -F'"' '{print $4}' | sort -V | tail -n 1) + +# Log the latest branch found +echo "Latest stable branch found: $LATEST_BRANCH" + +# Step 3: Clone or update the ESP-IDF repository +if not test -d "$HOME/esp-idf" + echo "============== Step 3: Cloning the ESP-IDF repository ================" + git clone -b "$LATEST_BRANCH" --recursive --depth 1 https://github.com/espressif/esp-idf.git "$HOME/esp-idf" +else + echo "ESP-IDF repository already exists." + cd "$HOME/esp-idf" || exit + git checkout "$LATEST_BRANCH" + git pull --recurse-submodules +end + +cd "$HOME/esp-idf" || exit +set IDF_COMMIT (git rev-parse HEAD) +echo "ESP-IDF is set to commit: $IDF_COMMIT" + +# Step 4: Set up the ESP-IDF environment +echo "============== Step 4: Setting up the ESP-IDF environment ================" +"$HOME/esp-idf/install.fish" +source "$HOME/esp-idf/export.fish" | source + +# Step 5: Add alias to Fish config +echo "============== Step 5: Adding alias to Fish configuration ===============" +if not grep -q "alias get-idf" "$SHELL_RC" + read -P "Do you want to add the alias 'get-idf' to $SHELL_RC? [yes/no] " response + if string match -i -r '^yes$' $response + echo "alias get-idf='source $HOME/esp-idf/export.fish | source'" >> "$SHELL_RC" + echo "ESP-IDF setup alias added to $SHELL_RC. Run 'get-idf' to configure your environment." + else + echo "Alias not added. You can manually add 'alias get-idf=\". $HOME/esp-idf/export.fish | source\"' to $SHELL_RC." + end +else + echo "ESP-IDF setup alias already exists in $SHELL_RC." +end + +# Step 6: Inform user to reload shell +echo "============== Step 6: Informing user to reload shell ===============" +echo "Please run 'source $SHELL_RC' to reload the shell with the new alias." +echo "In new shell, run 'get-idf' to enable ESP-IDF environment." + diff --git a/docs/setup_esp_idf__latest_stable__linux_macos.sh b/docs/setup_esp_idf__latest_stable__linux_macos.sh new file mode 100644 index 0000000000..7bc13d5673 --- /dev/null +++ b/docs/setup_esp_idf__latest_stable__linux_macos.sh @@ -0,0 +1,62 @@ +#!/usr/bin/env bash + +SHELL_RC="$HOME/.bashrc" + +# Step 1: Check if curl and git are installed +echo "============== Step 1: Checking if dependencies are installed ==============" +if ! command -v curl &> /dev/null; then + echo "curl is not installed. Please install curl." + exit 1 +fi + +if ! command -v git &> /dev/null; then + echo "git is not installed. Please install git." + exit 1 +fi + +echo "All required dependencies are installed." + +# Step 2: Fetch the branches from the GitHub API and find the latest stable release branch +echo "============== Step 2: Fetching branch list from ESP-IDF GitHub API ==============" +LATEST_BRANCH=$(curl -s https://api.github.com/repos/espressif/esp-idf/branches | grep -o '"name": "release/[^"]*' | awk -F'"' '{print $4}' | sort -V | tail -n 1) +echo "Latest stable branch found: $LATEST_BRANCH" + +# Step 3: Clone or update the ESP-IDF repository +if [ ! -d "$HOME/esp-idf" ]; then + echo "========= Step 3: Cloning the ESP-IDF repository (takes 3-4 mins) =============" + git clone -b "$LATEST_BRANCH" --recursive --depth 1 https://github.com/espressif/esp-idf.git "$HOME/esp-idf" +else + echo "ESP-IDF repository already exists. Updating..." + cd "$HOME/esp-idf" || exit + git checkout "$LATEST_BRANCH" + git pull --recurse-submodules +fi + +# Log the current commit hash +cd "$HOME/esp-idf" || exit +IDF_COMMIT=$(git rev-parse HEAD) +echo "<< ESP-IDF is set to commit: $IDF_COMMIT >>" + +# Step 4: Set up the ESP-IDF environment +echo "============== Step 4: Setting up the ESP-IDF environment ================" +"$HOME/esp-idf/install.sh" +source "$HOME/esp-idf/export.sh" + +# Step 5: Optionally add an alias to shell configuration for easy setup +echo "============== Step 5: Adding alias to shell configuration ===============" +if ! grep -q "alias get-idf" "$SHELL_RC"; then + read -p "Do you want to add the alias 'get-idf' to $SHELL_RC? [yes/no] " -r + if [[ $REPLY =~ ^[Yy][Ee][Ss]$ ]]; then + echo "alias get-idf='source $HOME/esp-idf/export.sh'" >> "$SHELL_RC" + echo "ESP-IDF setup alias added to $SHELL_RC. Run 'get-idf' to configure your environment." + else + echo "Alias not added. You can manually add 'alias get-idf=\"source $HOME/esp-idf/export.sh\"' to $SHELL_RC." + fi +else + echo "ESP-IDF setup alias already exists in $SHELL_RC." +fi + +# Step 6: Inform the user to reload the shell +echo "============== Step 6: Informing user to reload shell ===============" +echo "\nPlease run 'source $SHELL_RC' to reload the shell with the new alias." +echo "\nIn a new shell, run 'get-idf' to enable the ESP-IDF environment." diff --git a/docs/spi_full_duplex.md b/docs/spi_full_duplex.md index 140b69373b..2e13537032 100644 --- a/docs/spi_full_duplex.md +++ b/docs/spi_full_duplex.md @@ -1,119 +1,565 @@ # ESP-Hosted SPI FD (Full Duplex) Operation -**Table of Contents** +Sections 2 to 4 below covers the theoretical part where the SPI Full duplex design and implementation details are explained. -- [1. Introduction](#1-introduction) -- [2. SPI FD Configuration](#2-spi-fd-configuration) - - [2.1. Clock and Phase](#21-clock-and-phase) - - [2.2. GPIO configuration for SPI FD](#22-gpio-configuration-for-spi-fd) - - [2.3. Extra GPIO Signals Required](#23-extra-gpio-signals-required) -- [3. Hardware Considerations](#3-hardware-considerations) - - [3.1. Jumper Wires](#31-jumper-wires) - - [3.2 PCB Design](#32-pcb-design) - - [3.3 Testing the SPI Connection](#33-testing-the-spi-connection) - - [4. References](#4-references) +Section 5 to 9 covers the complete step-wise setup co-processor and host with SPI Full Duplex, using 2 or 4 data lines. -## 1. Introduction +If you wish to skip the theory, you can refer the [Quick Start Guide](#1-quick-start-guide) below. For quick navigation, please unfold the Table of Contents below. -The ESP32 family of chips support the standard SPI FD (Full Duplex) -Mode Protocol. +
+**Table of Contents** -In this mode of operation, SPI uses 2 data lines to transfer data to -the slave and from the slave at the same time (full duplex) during an -SPI transaction. +1. [Quick Start Guide](#1-quick-start-guide) -## 2. SPI FD Configuration +2. [Introduction](#2-introduction) -To enable SPI FD on the Host and Slave using `Menuconfig`: +3. [High Level Design and Implementation](#3-high-level-design-and-implementation) || [3.1 Number of Pins Required](#31-number-of-pins-required) || [3.2 SPI Full Duplex Mode Implementation](#32-spi-full-duplex-mode-implementation) || [3.3 Code Reference](#35-code-reference) -1. On Host: **Component config** ---> **ESP-Hosted config** ---> - **Transport layer** and choose **SPI Full-duplex**. -2. On Slave: **Example configuration** ---> **Transport layer** and - choose **SPI Full-duplex**. +4. [Hardware Considerations](#4-hardware-considerations) || [4.1 General Considerations](#41-general-considerations) || [4.2 Jumper Wires](#42-jumper-wires) || [4.3 PCB Design](#43-pcb-design) || [4.4 Advanced Considerations](#44-advanced-considerations) -### 2.1. Clock and Phase +5. [Hardware Setup](#5-hardware-setup) -> [!NOTE] -> The standard SPI CPOL clock and CPHA phase must be the same on both -> the host and slave for the protocol to work. +6. [Set-Up ESP-IDF](#6-set-up-esp-idf) -### 2.2. GPIO configuration for SPI FD +7. [Flashing the Co-processor](#7-flashing-the-co-processor) || [7.1 Create Co-processor Project](#71-create-co-processor-project) || [7.2 Co-processor Config](#72-co-processor-config) || [7.3 Co-processor Build](#73-co-processor-build) || [7.4 Co-processor Flashing](#74-co-processor-flashing) || [7.4.1 Serial Flashing (Initial Setup)](#741-serial-flashing-initial-setup) || [7.4.2 Co-processor OTA Flashing (Subsequent Updates)](#742-co-processor-ota-flashing-subsequent-updates) -The SPI interface can use almost any GPIO pins. For maximum speed and -minimal delays, it is recommended to select the SPI pin configuration -that uses the dedicated `IO_MUX` pins. See -[4. References](#4-references) for more information. +8. [Preparing the Host](#8-preparing-the-host) || [8.1 Select Example to Run in Hosted Mode](#81-select-example-to-run-in-hosted-mode) || [8.2 Host Project Component Configuration](#82-host-project-component-configuration) || [8.3 Menuconfig, Build and Flash Host](#83-menuconfig-build-and-flash-host) -This table summarises the recommended SPI GPIO pins for various ESP SoCs: +9. [Testing and Troubleshooting](#9-testing-and-troubleshooting) -| GPIO | ESP32 | ESP32-C2/C3/C6 | ESP32-S2/S3 | -| :--- | :--: | :--: | :--: | -| MOSI | 13 | 7 | 11 | -| MISO | 12 | 2 | 13 | -| CLK | 14 | 6 | 12 | -| CS | 15 | 10 | 10 | +10. [References](#10-references) -> [!NOTE] -> Check the ESP chip documentation for GPIO limitations that may -> prevent some GPIOs from being used for SPI. - -### 2.3. Extra GPIO Signals Required - -Extra GPIO signal are required for SPI FD on Hosted and can be -assigned to any free GPIO pins: - -- `Handshake` signal: an output signal from the slave to the - host. When asserted, it acts like the UART CTS (Clear to Send), - telling the host that the slave is ready for a SPI transaction. The - host should not perform a SPI transaction if the `Handshake` signal - is deasserted. -- `Data Ready` signal: an output signal from the slave to the - host. When asserted, the slave is telling the host that it has data - to send. The host should perform a SPI transaction to fetch the data - from the slave. -- `Reset` signal: an output signal from the host to the slave. When - asserted, the host resets the slave. This is done when ESP-Hosted is - started on the host, to synchronise the state of the host and slave. +
-> [!NOTE] -> The `Reset` signal can be configured to connect to the `EN` or `RST` -> pin on the slave, or assigned to a GPIO pin on the slave. -> -> To configure this, use `Menuconfig` on the Slave: **Example -> configuration** ---> **SPI FUll-duplex Configuration** ---> **Host -> SPI GPIOs** and set **Slave GPIO pin to reset itself**. +## 1. Quick Start Guide + +This section provides a brief overview of how to get started with ESP-Hosted using SPI FD mode. For detailed instructions on each step, please refer to the following sections: + +- [5. Hardware Setup](#5-hardware-setup) +- [6. Set-Up ESP-IDF](#6-set-up-esp-idf) +- [7. Flashing the Co-processor](#7-flashing-the-co-processor) +- [8. Preparing the Host](#8-preparing-the-host) +- [9. Testing and Troubleshooting](#9-testing-and-troubleshooting) + +These sections will guide you through the process of flashing both the co-processor and host devices, setting up the hardware connections, and verifying successful communication. + +## 2. Introduction + +The ESP32 family of chips supports the standard SPI FD (Full Duplex) Mode Protocol. In this mode, SPI uses two data lines (MISO and MOSI) to transfer data to and from the co-processor simultaneously during an SPI transaction. + +## 3. High Level Design and Implementation + +SPI Full duplex mode is the simplest mode of operation in ESP-Hosted. It can be easily tested with jumper wires. It doesn't require much complex hardware setup. For any non ESP chipsets as host also can prefer this mode for testing. This can also served as stepping stone before moving on to more complex modes of operations, like Dual SPI, Quad SPI and SDIO. + +### 3.1 Number of Pins Required + +In SPI Full Duplex mode, the following pins are required: + +1. **MISO (Master In Slave Out)**: Data line for the co-processor to send data to the host. +2. **MOSI (Master Out Slave In)**: Data line for the host to send data to the co-processor. +3. **SCLK (Serial Clock)**: Clock signal generated by the host to synchronize data transmission. +4. **CS (Chip Select)**: Signal used by the host to select the co-processor for communication. +5. **Reset Pin**: An output signal from the host to the co-processor. When asserted, the host resets the co-processor to synchronize the state of the host and co-processor. +6. **Handshake Pin**: An output signal from the co-processor to the host. When asserted, it tells the host that the co-processor is ready for an SPI transaction. +7. **Data Ready Pin**: An output signal from the co-processor to the host. When asserted, the co-processor is telling the host that it has data to send. + +The SPI used is full duplex. Handshake, Data Ready and Reset are additional GPIOs used in addition to MISO, MOSI, SCLK and CS. All pins are mandatory. + +### 3.2 SPI Full Duplex Mode Implementation + +- This solution uses SPI full duplex communication mode: + - Read and write operations occur simultaneously in the same SPI transaction + +- Handshake and Data ready are configured as interrupts at host. On loading host, it should automatically reset the co-processor using reset pin. + +- Protocol rules: + - Host must not start a transaction before ESP SPI peripheral is ready + - ESP peripheral indicates readiness via Handshake pin + +- ESP peripheral preparation: + - Always ready for data reception from host + - Queues next SPI transaction immediately after completing previous one + +- SPI transaction structure: + - Each transaction has a TX buffer and an RX buffer + - TX buffer: Contains data ESP peripheral wants to send to host + - RX buffer: Empty space to hold data received from host + +- Buffer initialization: + - ESP peripheral sets TX and RX buffers to 1600 bytes (maximum size) + - Host can send/receive up to 1600 bytes per transaction + +- TX buffer scenarios: + 1. No data to transfer: + - Allocate 1600-byte dummy TX buffer + - Set packet length field in payload header to 0 + 2. Valid data to send: + - TX buffer points to that data + +- SPI transaction setup: + - Set transaction length to 1600 bytes regardless of TX buffer size + - Submit transaction to SPI driver on ESP peripheral + - Pull Handshake pin high to signal readiness + - If TX buffer has valid data, also pull Data ready pin high + +- Host response to Handshake / Data ready interrupt: + - Decide whether to perform SPI transaction (if Handshake is high) + - Perform transfer if Data ready pin is high or host has data to transfer + - If either condition is false, do not perform transfer, just ignore the interrupt + +- During SPI transaction: + - Exchange TX and RX buffers on SPI data lines + +- Post-transaction processing: + - Both ESP peripheral and host process received buffer based on payload header + +- Transaction completion: + - ESP peripheral pulls Handshake pin low + - If transaction had valid co-processor TX buffer, also pulls Data ready pin low + +### 3.3 Code Reference +For a detailed implementation of SPI full duplex communication using the ESP-Hosted framework, refer to the following code files in the ESP-Hosted repository: + +- **Master SPI Communication Code**: + - [spi_drv.c](https://github.com/espressif/esp-hosted/blob/feature/esp_as_mcu_host/host/drivers/transport/spi/spi_drv.c): Contains the implementation for configuring and handling SPI transactions on the master side. + - [spi_wrapper.c](https://github.com/espressif/esp-hosted/blob/feature/esp_as_mcu_host/host/port/esp_idf/spi_wrapper.c): Provides an OS abstraction layer for SPI operations, making it easier to handle SPI communication in a platform-independent manner. + +- **Co-processor SPI Communication Code**: + - [spi_slave_apis.c](https://github.com/espressif/esp-hosted/blob/feature/esp_as_mcu_host/firmware/components/esp_slave/spi_slave_apis.c): Includes the setup and transaction handling for the SPI co-processor, detailing how the co-processor should configure its SPI interface and handle incoming and outgoing data. + +## 4. Hardware Considerations + +### 4.1 General Considerations + +- Ensure equal trace lengths for all SPI connections, whether using jumper wires or PCB traces. +- Use the lower clock frequency like 5 MHz for evaluation. Once solution verified, optimise the clock frequency in increasing steps to max possible value. To find out practical maximum SPI slave frequency for your co-processor, check `IDF_PERFORMANCE_MAX_SPI_CLK_FREQ` in [ESP-IDF SPI slave benchmark](https://github.com/espressif/esp-idf/blob/master/components/esp_driver_spi/test_apps/components/spi_bench_mark/include/spi_performance.h) +- Verify voltage compatibility between host and co-processor devices. +- Provide proper power supply decoupling for both host and co-processor devices. + +### 4.2 Jumper Wires + +- Jumper wires are suitable for initial testing and prototyping. +- Use high-quality, low-capacitance jumper wires. +- Keep wires as short as possible, ideally under 10 cm. +- Arrange wires to minimize crosstalk, especially for clock and data lines. +- Possibly, use twisted pairs for clock and data lines to reduce electromagnetic interference. +- If possible, use a ground wire between every signal wire to improve signal integrity. +- Connect as many grounds as possible to improve common ground reference and reduce ground noise. + +### 4.3 PCB Design +For optimal performance and reliability in production designs: -## 3. Hardware Considerations +- Ensure equal trace lengths for all SPI signals (CLK, MOSI, MISO, CS) as much as possible. This practice, known as length matching, is crucial for maintaining signal integrity and reducing timing skew, especially at higher frequencies. +- If perfect length matching is not possible, prioritize matching the clock (CLK) trace length with the data lines. +- Use controlled impedance traces for high-speed signals. +- Place bypass capacitors close to the power pins of both the host and co-processor devices. +- Consider using series termination resistors on the clock and data lines to reduce reflections. +- For high-speed designs, use a 4-layer PCB with dedicated power and ground planes. -### 3.1. Jumper Wires +### 4.4 Advanced Considerations -While jumper wires can be used to test SPI, it is recommended to use -short wires (5 to 10 cm in length, shorter is better) to -minimise propagation delay and noise on the signals. +- Calculate the maximum allowed trace length based on your clock frequency and PCB material. +- Consider the capacitive load on the SPI bus, especially for longer traces or when using multiple co-processor devices. +- For very high-speed designs, consider using differential signaling techniques. +- Implement proper EMI/EMC design techniques to minimize electromagnetic interference. -### 3.2 PCB Design -The PCB traces for SPI should be equal length and kept as short as -possible. The signals, especially the `CLK` signal, should be isolated -from other signals using a ground plane to minimise crosstalk. +**Debugging Tips** -### 3.3 Testing the SPI Connection +- Use an oscilloscope or logic analyzer to verify signal integrity and timing. +- Start with a lower clock frequency and gradually increase it while monitoring performance. +- Ensure proper grounding between the host and co-processor devices. +- If using multiple power supplies, ensure they share a common ground. +- Consider using level shifters if the host and co-processor operate at different voltage levels. -**Using a Lower Clock Speed** +## 5. Hardware Setup -You can use a lower clock speed to verify the connections. For SPI, -you can start with 10 MHz or lower. +Setting up the hardware involves connecting the master and co-processor devices via the SPI pins and ensuring all extra GPIO signals are properly connected. Below is the table of connections for the SPI full duplex setup between an host ESP chipset and another ESP chipset as co-processor: + + +### Host connections + +| Signal | ESP32 | ESP32-S2/S3 | ESP32-C2/C3/C5/C6 | ESP32-P4 (ESP32-P4-Function-EV-Board) | +|-------------|-------|-------------|-------------------|---------------------------------------| +| CLK | 14 | 12 | 6 | 18 | +| MOSI | 13 | 11 | 7 | 14 | +| MISO | 12 | 13 | 2 | 15 | +| CS | 15 | 10 | 10 | 19 | +| Handshake | 26 | 17 | 3 | 16 | +| Data Ready | 4 | 4 | 4 | 17 | +| Reset Out | 5 | 5 | 5 | 54 | + + +### Co-processor connections + +| Signal | ESP32 | ESP32-C2/C3/C5/C6 | ESP32-S2/S3 | ESP32-C6 on ESP32-P4-Function-EV-Board | +|-------------|-------|-------------------|-------------|---------------------------------------| +| CLK | 14 | 6 | 12 | 19 | +| MOSI | 13 | 7 | 11 | 20 | +| MISO | 12 | 2 | 13 | 21 | +| CS | 15 | 10 | 10 | 18 | +| Handshake | 26 | 3 | 17 | 22 | +| Data Ready | 4 | 4 | 5 | 23 | +| Reset In | EN | EN/RST | EN/RST | EN/RST | -To configure this, use `Menuconfig` on the Host: **Component -config** ---> **ESP-Hosted config** ---> **SPI Configuration** -and set **SPI Clock Freq (MHz)**. > [!NOTE] -> The actual clock frequency used is determined by the hardware. Use -> an oscilloscope to check the clock frequency. +> - Always try to use IO_MUX pins from the datasheet for optimal performance on both sides. +> - These GPIO assignments are based on default Kconfig configurations & are configurable. +> - Once ported, any other non ESP host with standard SPI can be used. +> - All ESP chipsets support SPI Full Duplex mode. Chipsets with Wi-Fi/Bluetooth can be used as co-processor. + +> [!IMPORTANT] +> - Ensure proper grounding between host and co-processor devices. +> - Use short, high-quality cables for connections. +> - For production designs, consider using a properly designed PCB with controlled impedance traces. + +## 6. Set-Up ESP-IDF + +Before setting up the ESP-Hosted host & co-processor for SPI Full Duplex mode, ensure that ESP-IDF is properly installed and set up on your system. + +#### Option 1: Installer Way + +- **Windows** + - Install and setup ESP-IDF on Windows as documented in the [Standard Setup of Toolchain for Windows](https://docs.espressif.com/projects/esp-idf/en/latest/esp32/get-started/windows-setup.html). + - Use the ESP-IDF [Powershell Command Prompt](https://docs.espressif.com/projects/esp-idf/en/latest/esp32/get-started/windows-setup.html#using-the-command-prompt) for subsequent commands. + +- **Linux or MacOS** + - For bash: + ```bash + bash docs/setup_esp_idf__latest_stable__linux_macos.sh + ``` + - For fish: + ```fish + fish docs/setup_esp_idf__latest_stable__linux_macos.fish + ``` + +#### Option 2: Manual Way + +Please follow the [ESP-IDF Get Started Guide](https://docs.espressif.com/projects/esp-idf/en/latest/esp32/get-started/index.html) for manual installation. + +## 7. Flashing the Co-processor + +| Supported Co-processor Targets | ESP32 | ESP32-C2 | ESP32-C3 | ESP32-C5 | ESP32-C6 | ESP32-C61 | ESP32-S2 | ESP32-S3 | +| ------------------------------ | ----- | -------- | -------- | -------- | -------- | --------- | -------- | -------- | + +There are two methods to flash the ESP-Hosted co-processor firmware: + +### 7.1 Create Co-processor Project +1. Create co-processor project possibly outside of ESP-IDF project directory using + + ```bash + idf.py create-project-from-example "espressif/esp_hosted:slave" + ``` -### 4. References +2. Navigate to the created project directory. -- GPIO Matrix and IO_MUX considerations for SPI Master: https://docs.espressif.com/projects/esp-idf/en/latest/esp32/api-reference/peripherals/spi_master.html#gpio-matrix-and-io-mux -- GPIO Matrix and IO_MUX considerations for SPI Slave: https://docs.espressif.com/projects/esp-idf/en/latest/esp32/api-reference/peripherals/spi_slave.html#gpio-matrix-and-io-mux +3. Configure the project for your target ESP chip: + + ```bash + idf.py set-target + ``` + Replace `` with your specific ESP chip (e.g., esp32c3, esp32s3). + +### 7.2 Co-processor Config +Configure the co-processor project using +``` +idf.py menuconfig +``` + +#### 7.2.1 Transport config + - Navigate to "Example configuration" -> "Transport layer" + - Select "SPI Full-duplex" + +#### 7.2.2 Any other config + Optionally, configure any additional SPI-specific settings under "SPI Full-duplex" + - Set the GPIO pins for SPI signals (MOSI, MISO, CLK, CS), Handshake, Data Ready, Reset + - Configure SPI mode (0, 1, 2, or 3) + - Set the SPI clock frequency + - Checksum enable/disable (Checksum is recommended to be enabled as spi hardware doesn't have any error detection) + +###### Generated files +- Generated config files are (1) `sdkconfig` file and (2) internal `sdkconfig.h` file. +- Please note, any manually changes done to these generated files, would not take effect. + +###### Defaulting specific config (Optional) +- This is advanced option, so please be careful. +- To mark some config options as default, you can add specific config line in file, `sdkconfig.defaults.`. So whenever next time building, you do not need to re-configure. + +### 7.3 Co-processor Build +Build the co-processor project + +``` +idf.py build +``` + +### 7.4 Co-processor Flashing + +##### 7.4.1 Serial Flashing (Initial Setup) + +For the initial setup or when OTA is not available, use serial flashing. + +Flash the co-processor firmware using +``` +idf.py -p flash +``` + +> [!NOTE] +> +> If you are not able to flash the co-processor, there might be a chance that host is not allowing to to do so. +> +> Put host in bootloader mode using following command and then retry flashing the co-processor +> +> ```bash +> esptool.py -p **** --before default_reset --after no_reset run +> ``` + +Monitor the output (optional): +``` +idf.py -p monitor +``` + +##### 7.4.2. Co-processor OTA Flashing (Subsequent Updates) + +For subsequent updates, you can re-use ESP-Hosted-MCU transport, as it should be already working. While doing OTA, Complete co-processor firmware image is not needed and only co-processor application partition, 'network_adapter.bin' need to be re-flashed remotely from host. + +1. Ensure your co-processor device is connected and communicating with the host with existing ESP-Hosted-MCU. + +2. Create a web server +You can re-use your existing web server or create a new locally for testing. Below is example to do it. + - Make a new directory so that web server can be run into it and navigate into it + - Create simple local web server using python3 + + ```bash + python3 -m http.server 8080 + ``` +3. Copy the co-processor app partition `network_adapter.bin` in the directory where you created the web server. + - The `network_adapter.bin` can be found in your co-processor project build at `/build/network_adapter.bin` + +4. Verify if web server is set-up correctly + - Open link `http://127.0.0.1:8080` in the browser and check if network_adapter.bin is available. + - Right click and copy the complete URL of this network_adapter.bin and note somewhere. + +5. On the **host side**, use the `esp_hosted_ota` function to initiate the OTA update: + + ```c + #include "esp_hosted_api.h" + + const char* image_url = "http://example.com/path/to/network_adapter.bin"; //web server full url + esp_err_t ret = esp_hosted_ota(image_url); + if (ret == ESP_OK) { + printf("co-processor OTA update failed[%d]\n", ret); + } + ``` + + This function will download the firmware in chunk by chunk as http client from the specified URL and flash it to the co-processor device through the established transport. + In above web server example, You can paste the copied url earlier. + + +6. Monitor the OTA progress through the console output on both the host and co-processor devices. + +> [!NOTE] +> +> - The `esp_hosted_ota` function is part of the ESP-Hosted-MCU API and handles the OTA process through the transport layer. +> - Ensure that your host application has web server connectivity to download the firmware file. +> - The co-processor device doesn't need to be connected to the web server for this OTA method. + +## 8. Flashing the Host + +| Supported Host Targets | Any ESP chipset | Any Non-ESP chipset | +| ----------------------- | --------------- | ------------------- | + +### 8.1 Select Example to Run in Hosted Mode + +Select an example from the [ESP-IDF examples directory](https://github.com/espressif/esp-idf/tree/master/examples) that you wish to run in ESP-Hosted mode. All Wi-Fi and Bluetooth examples are supported. For simplicity and demonstration purposes, we will use the [ESP-IDF iperf example](https://github.com/espressif/esp-idf/tree/master/examples/wifi/iperf). + +### 8.2 Host Project Component Configuration + +Now that ESP-IDF is set up, follow these steps to prepare the host: + +###### 1. Navigate to the iperf example in your ESP-IDF directory: + ``` + cd $IDF_PATH/examples/wifi/iperf + ``` + +###### 2. Dependency components + Add the required components to the project's `idf_component.yml` file: + ``` + idf.py add-dependency "espressif/esp_wifi_remote" + idf.py add-dependency "espressif/esp_hosted" + ``` + +###### 3. Remove conflicting configuration + Open the `main/idf_component.yml` file and remove/comment the following block if present: + ``` + # ------- Delete or comment this block --------- + espressif/esp-extconn: + version: "~0.1.0" + rules: + - if: "target in [esp32p4]" + # ----------------------------------- + ``` + This step is necessary because esp-extconn and esp-hosted cannot work together. + +###### 4. Disable native Wi-Fi if available + If your host ESP chip already has native Wi-Fi support, disable it by editing the `components/soc//include/soc/Kconfig.soc_caps.in` file and changing all `WIFI` related configs to `n`. + + If you happen to have both, host and co-processor as same ESP chipset type (for example two ESP32-C2), note an [additional step](docs/troubleshooting/#1-esp-host-to-evaluate-already-has-native-wi-fi) + + +### 8.3 Menuconfig, Build and Flash Host + +##### 1. High performance configurations + This is optional step, suggested for high performance applications. + + If using ESP32-P4 as host: + - Remove the default `sdkconfig.defaults.esp32p4` file. + - Create a new `sdkconfig.defaults.esp32p4` file with the following content: + ``` + CONFIG_ESP_WIFI_STATIC_RX_BUFFER_NUM=16 + CONFIG_ESP_WIFI_DYNAMIC_RX_BUFFER_NUM=64 + CONFIG_ESP_WIFI_DYNAMIC_TX_BUFFER_NUM=64 + CONFIG_ESP_WIFI_AMPDU_TX_ENABLED=y + CONFIG_ESP_WIFI_TX_BA_WIN=32 + CONFIG_ESP_WIFI_AMPDU_RX_ENABLED=y + CONFIG_ESP_WIFI_RX_BA_WIN=32 + + CONFIG_LWIP_TCP_SND_BUF_DEFAULT=65534 + CONFIG_LWIP_TCP_WND_DEFAULT=65534 + CONFIG_LWIP_TCP_RECVMBOX_SIZE=64 + CONFIG_LWIP_UDP_RECVMBOX_SIZE=64 + CONFIG_LWIP_TCPIP_RECVMBOX_SIZE=64 + + CONFIG_LWIP_TCP_SACK_OUT=y + ``` + + For other hosts also, you can merge above configs in corresponding `sdkconfig.defaults.esp32XX` file. + +###### 2. Set environment for your host ESP chip: + + ``` + idf.py set-target + ``` + +###### 3. Flexible Menuconfig configurations + + ``` + idf.py menuconfig + ``` + ESP-Hosted-MCU host configurations are available under "Component config" -> "ESP-Hosted config" + 1. Select "SPI Full-duplex" as the transport layer + 2. Change co chipset to connect to under "slave chipset to be used" + 3. Optionally, configure SPI-specific settings like + - SPI Clock Freq (MHz) + - SPI Mode + - SPI Pins + - SPI Checksum Enable/Disable (Checksum is recommended to be enabled as spi hardware doesn't have any error detection) + + > [!NOTE] + > + > The actual clock frequency used is determined by the hardware. Use an oscilloscope or logic analyzer to check the clock frequency. + +###### 4. Build the project: + ``` + idf.py build + ``` + +###### 5. Flash the firmware: + ``` + idf.py -p flash + ``` + +###### 6. Monitor the output: + ``` + idf.py -p monitor + ``` + - If host was put into bootloader mode earlier, it may need manual reset + +## 9. Testing and Troubleshooting + +After flashing both the co-processor and host devices, follow these steps to connect and test your ESP-Hosted SPI Full Duplex setup: + +1. Connect the hardware: + - Follow the pin assignments for SPI Full Duplex as specified in [Hardware Setup](docs/spi_full_duplex.md#5-hardware-setup). + - Ensure all necessary connections are made, including power, ground, and the extra GPIO signals (Data_Ready and Reset). + +2. Power on both devices. + +3. Verify the connection: + - Check the serial output of both devices for successful initialization messages. + - Look for messages indicating that the SPI Full Duplex transport layer has been established + +4. Logs at both sides: + - Host: + + ``` + I (522) transport: Attempt connection with slave: retry[0] + I (525) transport: Reset slave using GPIO[54] + I (530) os_wrapper_esp: GPIO [54] configured + I (535) gpio: GPIO[54]| InputEn: 0| OutputEn: 1| OpenDrain: 0| Pullup: 0| Pulldown: 0| Intr:0 + I (1712) transport: Received INIT event from ESP32 peripheral + I (1712) transport: EVENT: 12 + I (1712) transport: EVENT: 11 + I (1715) transport: capabilities: 0xe8 + I (1719) transport: Features supported are: + I (1724) transport: - HCI over SPI + I (1728) transport: - BLE only + I (1732) transport: EVENT: 13 + I (1736) transport: ESP board type is : 13 + + I (1741) transport: Base transport is set-up + ``` + + - Co-processor: + + ``` + I (492) fg_mcu_slave: ********************************************************************* + I (501) fg_mcu_slave: ESP-Hosted-MCU Slave FW version :: X.Y.Z + + I (511) fg_mcu_slave: Transport used :: SPI + I (520) fg_mcu_slave: ********************************************************************* + I (529) fg_mcu_slave: Supported features are: + I (534) fg_mcu_slave: - WLAN over SPI + I (538) h_bt: - BT/BLE + I (541) h_bt: - HCI Over SPI + I (545) h_bt: - BLE only + ``` + +5. Test basic functionality: + - The iperf example automatically attempts to connect to the configured Wi-Fi network. Watch the serial output for connection status. + - If the automatic connection fails, you can manually initiate a Wi-Fi scan and connection: + ``` + sta_scan + sta_connect + ``` +6. Additional commands to test: + - Get IP address: `sta_ip` + - Disconnect from Wi-Fi: `sta_disconnect` + - Set Wi-Fi mode: `wifi_mode ` (where mode can be 'sta', 'ap', or 'apsta') + +7. Advanced iperf testing: + Once connected, you can run iperf tests: + + | Test Case | Host Command | External STA Command | + |-----------|--------------|----------------------| + | UDP Host TX | `iperf -u -c -t 60 -i 3` | `iperf -u -s -i 3` | + | UDP Host RX | `iperf -u -s -i 3` | `iperf -u -c -t 60 -i 3` | + | TCP Host TX | `iperf -c -t 60 -i 3` | `iperf -s -i 3` | + | TCP Host RX | `iperf -s -i 3` | `iperf -c -t 60 -i 3` | + + Note: Replace `` with the IP address of the external STA, and `` with the IP address of the ESP-Hosted device. + +8. Troubleshooting: + - If you encounter issues, refer to section 6.3 for testing the SPI connection. + - Consider using a lower clock speed or checking your [hardware connections](#5-hardware-setup) if you experience communication problems. + - ESP-Hosted-MCU troubleshooting guide: [docs/troubleshooting.md](docs/troubleshooting.md) + +9. Monitoring and debugging: + - Use the serial monitor on both devices to observe the communication between the host and co-processor. + - For more detailed debugging, consider using a logic analyzer to examine the SPI signals. + +## 10. References +- [ESP-IDF Programming Guide](https://docs.espressif.com/projects/esp-idf/en/latest/esp32/) +- [ESP32 Hardware Design Guidelines](https://www.espressif.com/en/products/hardware/esp32/resources) +- [SPI Protocol Basics](https://en.wikipedia.org/wiki/Serial_Peripheral_Interface) diff --git a/docs/spi_half_duplex.md b/docs/spi_half_duplex.md index d7aa029085..501b0c3393 100644 --- a/docs/spi_half_duplex.md +++ b/docs/spi_half_duplex.md @@ -1,102 +1,107 @@ # ESP-Hosted SPI HD (Half Duplex) Operation -**Table of Contents** - -- [1. Introduction](#1-introduction) -- [2. SPI HD Configuration](#2-spi-hd-configuration) - - [2.1. Clock and Phase](#21-clock-and-phase) - - [2.2. Number of Data Lines](#22-number-of-data-lines) - - [2.3. Extra GPIO Signal Required](#23-extra-gpio-signal-required) - - [2.4. Recommended Pin Assignments](#24-recommended-pin-assignments) -- [3. SPI HD protocol used by Hosted](#3-spi-hd-protocol-used-by-hosted) - - [3.1. Data IO Modes](#31-data-io-modes) - - [3.2. Supported Commands](#32-supported-commands) - - [3.2.1 Command Mask](#321-command-mask) - - [3.3. Registers Used](#33-registers-used) - - [3.4. Timing Diagrams](#34-timing-diagrams) -- [4. SPI HD Operation in Hosted](#4-spi-hd-operation-in-hosted) - - [4.1 Transport Initialization](#41-transport-initialization) - - [4.1.1. Slave and Host Initialization](#411-slave-and-host-initialization) - - [4.1.2. Number of Data Lines Used](#412-number-of-data-lines-used) - - [4.2. Slave Transfers Data to Host](#42-slave-transfers-data-to-host) - - [4.3. Host Transfers Data to the Slave](#43-host-transfers-data-to-the-slave) - - [4.4. Code Reference](#44-code-reference) -- [5. Hardware Considerations](#5-hardware-considerations) - - [5.1. Jumper Wires](#51-jumper-wires) - - [5.2 PCB Design](#52-pcb-design) - - [5.3 Testing the SPI Connection](#53-testing-the-spi-connection) -- [6. References](#6-references) - -## 1. Introduction - -The ESP32 family of chips (except the ESP32) support the SPI Slave HD -(Half Duplex) Mode Protocol. - -In this mode of operation, SPI supports 2 to 4 data lines to transfer -data to the slave or from the slave (half duplex) during an SPI -transaction. This is different from 'standard' SPI mode which -transfers data bidirectionally (full duplex) over two data lines (one -for host to slave data [MOSI], one for slave to host data [MISO]) during an SPI -transaction. +Sections 2 to 6 below covers the theoretical part where the SPI Half duplex protocol and expected framing structure is explained. This frame structure is flexible. Host and Co-processor follow the same frame structure. + +Section 7 to 11 covers the complete step-wise setup co-processor and host with SPI Half Duplex, using 2 or 4 data lines. + +If you wish to skip the theory, you can refer the [Quick Start Guide](#1-quick-start-guide) below. For quick navigation, please unfold the Table of Contents below. + +
+Table of Contents + + +- [1. Quick Start Guide](#1-quick-start-guide) + +- [2. Introduction](#2-introduction) + +- [3. SPI HD Configuration](#3-spi-hd-configuration) => [1 Clock and Phase](#31-clock-and-phase) || [2 Data Lines](#32-data-lines) || [3 Extra GPIO Signals](#33-extra-gpio-signals) || [4 Pin Assignments](#34-pin-assignments) + +- [4. SPI HD Protocol](#4-spi-hd-protocol) => [1 Data IO Modes](#41-data-io-modes) || [2 Supported Commands](#42-supported-commands) || [3 Registers Used](#43-registers-used) || [4 Timing Diagrams](#44-timing-diagrams) + +- [5. SPI HD Operation](#5-spi-hd-operation) => [1 Initialization](#51-initialization) || [2 Co-processor to Host Transfer](#52-co-processor-to-host-transfer) || [3 Host to Co-processor Transfer](#53-host-to-co-processor-transfer) || [4 Code Reference](#54-code-reference) + +- [6. Hardware Considerations](#6-hardware-considerations) => [1 General Considerations](#61-general-considerations) || [2 Jumper Wires](#62-jumper-wires) || [3 PCB Design](#63-pcb-design) || [4 Advanced Considerations](#64-advanced-considerations) + +- [7. Hardware Setup](#7-hardware-setup) + +- [8. Set-Up ESP-IDF](#8-set-up-esp-idf) => [Option 1: Installer Way](#option-1-installer-way) || [Option 2: Manual Way](#option-2-manual-way) + +- [9. Flashing the Co-processor](#9-flashing-the-co-processor) => [1 Create Co-processor Project](#91-create-co-processor-project) || [2 Co-processor Config](#92-co-processor-config) || [3 Co-processor Build](#93-co-processor-build) || [4 Co-processor Flashing](#94-co-processor-flashing) + +- [10. Flashing the Host](#10-flashing-the-host) => [1 Select Example to Run in Hosted Mode](#101-select-example-to-run-in-hosted-mode) || [2 Set Up the Host for SPI Half Duplex Mode](#102-set-up-the-host-for-spi-half-duplex-mode) + +- [11. Testing and Troubleshooting](#11-testing-and-troubleshooting) + +- [12. References](#12-references) + +
+ +## 1. Quick Start Guide + +This section provides a brief overview of how to get started with ESP-Hosted using SPI HD mode, bypassing the theory and explanation. Please refer to the following sections to quickly set-up demo. + +- [7. Hardware Setup](#7-hardware-setup) +- [8. Set-Up ESP-IDF](#8-set-up-esp-idf) +- [9. Flashing the Co-processor](#9-flashing-the-co-processor) +- [10. Flashing the Host](#10-flashing-the-host) +- [11. Testing and Troubleshooting](#11-testing-and-troubleshooting) + +These sections will guide you through the process of configuring and flashing both the co-processor and host devices, setting up the hardware connections, and verifying successful communication. + +## 2. Introduction + +The ESP32 family of chips (except the ESP32) support the SPI co-processor HD (Half Duplex) Mode Protocol. + +In this mode of operation, SPI supports 2 to 4 data lines to transfer data to the co-processor or from the co-processor (half duplex) during an SPI transaction. This is different from 'standard' SPI mode which transfers data bidirectionally (full duplex) over two data lines (one for host to co-processor data [MOSI], one for co-processor to host data [MISO]) during an SPI transaction. > [!NOTE] -> SPI Half Duplex mode is not supported on the ESP32 +> +> SPI Half Duplex mode is not supported on the classic ESP32 > [!IMPORTANT] +> > SPI Half Duplex is not an industry standard and has multiple > implementations. Make sure your host processor supports the SPI HD -> protocol implemented by the Hosted slave before proceeding. See [SPI -> HD protocol used by Hosted](#3-spi-hd-protocol-used-by-hosted). +> protocol implemented by the Hosted co-processor before proceeding. See [SPI +> HD protocol used by Hosted](#4-spi-hd-protocol). -## 2. SPI HD Configuration +## 3. SPI HD Configuration -To enable SPI HD on the Host and Slave using `Menuconfig`: +To enable SPI HD on the Host and co-processor using `idf.py menuconfig`: 1. On Host: **Component config** ---> **ESP-Hosted config** ---> **Transport layer** and choose **SPI Half-duplex**. -2. On Slave: **Example configuration** ---> **Transport layer** and +2. On Co-processor: **Example configuration** ---> **Transport layer** and choose **SPI Half-duplex**. -### 2.1. Clock and Phase +### 3.1. Clock and Phase The standard SPI CPOL clock and CPHA phase must be configured -correctly on both the host and slave for the protocol to work. +correctly on both the host and co-processor for the protocol to work. -### 2.2. Number of Data Lines +### 3.2. Data Lines -Both the host and slave can support two or four data lines. Four data -lines will be used to transfer data if configured on both the host and -slave. If the host is configured to use two data lines, only two lines -will be used to transfer data even if the slave is configured to use -four data lines. +Both the host and co-processor can support two or four data lines. Four data lines will be used to transfer data if configured on both the host and co-processor. If the host is configured to use two data lines, only two lines will be used to transfer data even if the co-processor is configured to use four data lines. -### 2.3. Extra GPIO Signal Required +### 3.3. Extra GPIO Signals -Extra GPIO signal are required for SPI HD on Hosted and can be +Extra GPIO signals are required for SPI HD on Hosted and can be assigned to any free GPIO pins: -- `Data_Ready` signal: an output signal from the slave to the - host. When asserted, the slave is telling the host that it has data - to send. The host should perform a data read SPI transaction to - fetch data from the slave. -- `Reset` signal: an output signal from the host to the slave. When - asserted, the host resets the slave. This is done when ESP-Hosted is - started on the host, to synchronise the state of the host and slave. +- `Data_Ready` signal: an output signal from the co-processor to the host. When asserted, the co-processor is telling the host that it has data to send. The host should perform a data read SPI transaction to fetch data from the co-processor. +- `Reset` signal: an output signal from the host to the co-processor. When asserted, the host resets the co-processor. This is done when ESP-Hosted is started on the host, to synchronise the state of the host and co-processor. > [!NOTE] > The `Reset` signal can be configured to connect to the `EN` or `RST` -> pin on the slave, or assigned to a GPIO pin on the slave. +> pin on the co-processor, or assigned to a GPIO pin on the co-processor. > -> To configure this, use `Menuconfig` on the Slave: **Example +> To configure this, use `idf.py menuconfig` on the co-processor: **Example > configuration** ---> **SPI Half-duplex Configuration** ---> > **GPIOs** and set **Slave GPIO pin to reset itself**. -### 2.4. Recommended Pin Assignments +### 3.4. Pin Assignments -Using the pins already assigned to SPI signals (dedicated `IO_MUX` -pins) is recommended to minimise propagation delays. Using other GPIO -pins for SPI signals will route the signals through the GPIO matrix -which may limit the maximum clock frequency that can be used. +Using the pins already assigned to SPI signals (dedicated `IO_MUX` pins) is recommended to minimise propagation delays. Using other GPIO pins for SPI signals will route the signals through the GPIO matrix which may limit the maximum clock frequency that can be used. The following table shows the mapping between the SPI bus signals and their SPI HD Function: @@ -109,19 +114,15 @@ their SPI HD Function: | SPIHD | Data Bit 3 | | SPICLK | Clk | -The SPI HD CS signal and `Data_Ready` can be assigned to any GPIO pin on the host and -slave. +The SPI HD CS signal and `Data_Ready` can be assigned to any GPIO pin on the host and co-processor. -## 3. SPI HD protocol used by Hosted +## 4. SPI HD Protocol -Hosted uses the ESP SPI Slave HD (Half Duplex) Mode Protocol (see -[References](#6-references)) with some modifications. +Hosted uses the ESP SPI co-processor HD (Half Duplex) Mode Protocol (see [References](#11-references)) with some modifications. -### 3.1. Data IO Modes +### 4.1 Data IO Modes -When communicating with the slave, the master uses the Command, -Address, Dummy and Data phases during an SPI transaction. The number -of bits and number of data lines used in each phase are: +When communicating with the co-processor, the master uses the Command, Address, Dummy and Data phases during an SPI transaction. The number of bits and number of data lines used in each phase are: - **Command**: 8 bits, 1 data line - **Address**: 8 bits, 2 or 4 data lines @@ -129,66 +130,56 @@ of bits and number of data lines used in each phase are: - **Data**: variable length, 2 or 4 data lines > [!NOTE] +> > The number of data lines used in the Address and Data phase depends -> on the Command Mask in the Command sent by the host. See [Command -> Mask](#321-command-mask). +> on the Command Mask in the Command sent by the host. See [Supported Commands](#44-supported-commands). -### 3.2. Supported Commands +### 4.2 Supported Commands -Hosted uses the following SPI HD commands when communicating with the -slave: +Hosted uses the following SPI HD commands when communicating with the co-processor: | Command | OpCode | Purpose | | :--- | :--- | :--- | -| WRBUF | 0x01 | Write to a 32-bit buffer register on the slave | -| RDBUF | 0x02 | Read from a 32-bit buffer register on the slave | -| WRDMA | 0x03 | Write data to the slave using DMA | -| RDDMA | 0x04 | Read data from the slave during DMA | +| WRBUF | 0x01 | Write to a 32-bit buffer register on the co-processor | +| RDBUF | 0x02 | Read from a 32-bit buffer register on the co-processor | +| WRDMA | 0x03 | Write data to the co-processor using DMA | +| RDDMA | 0x04 | Read data from the co-processor during DMA | | WR_DONE | 0x07 | End of DMA write | | CMD8 | 0x08 | End of DMA read | | CMD9 | 0x09 | End of register read | -#### 3.2.1 Command Mask +#### 4.2.1 Command Mask -The Commands are masked with a command mask to tell the slave the -correct number of data lines to use during the transaction (2 or 4 -data lines). Hosted uses the following masks, which are bit ORed with -the command during a SPI transactions: +The Commands are masked with a command mask to tell the co-processor the correct number of data lines to use during the transaction (2 or 4 data lines). Hosted uses the following masks, which are bit ORed with the command during a SPI transactions: | Mode | Mask | | :--- | :--- | | 2-bits | 0x50 | | 4-bits | 0xA0 | -For example, if the host sends command `0x51` (2-bit mask + WRBUF), -the host and slave will use 2 data lines to send the address and -data. If the host sends command `0xA1` (4-bit mask + WRBUF), the host -and slave will use 4 data lines to send the address and data. +For example, if the host sends command `0x51` (2-bit mask + WRBUF), the host and co-processor will use 2 data lines to send the address and data. If the host sends command `0xA1` (4-bit mask + WRBUF), the host and co-processor will use 4 data lines to send the address and data. -The Command Mask determines the number of data lines used for the -transaction. Even if there are four data lines between the host and -slave, the host can tell the slave to use only two data lines by -applying the 0x50 command mask. +The Command Mask determines the number of data lines used for the transaction. Even if there are four data lines between the host and co-processor, the host can tell the co-processor to use only two data lines by applying the 0x50 command mask. > [!WARNING] +> > It is an error to apply the 4-bit data mask (0xA0) when there are -> only two data lines connecting the host and slave. +> only two data lines connecting the host and co-processor. -### 3.3. Registers Used +### 4.3 Registers Used -The ESP SPI Slave HD Mode Protocol defines a number of registers on -the slave. These registers are used in Hosted as follows: +The ESP SPI Co-processor HD Mode Protocol defines a number of registers on the co-processor. These registers are used in Hosted as follows: | Register | Name | Purpose | | :--- | :--- | :--- | -| 0x00 | SLAVE\_READY | Indicates if slave is ready | -| 0x04 | MAX\_TX\_BUF\_LEN | Maximum length of DMA data slave can transmit | -| 0x08 | MAX\_RX\_BUF\_LEN | Maximum length of DMA data slave can receive | -| 0x0C | TX\_BUF\_LEN | Updated whenever slave wants to transmit data | -| 0x10 | RX\_BUF\_LEN | Updated whenever slave can receive data | -| 0x14 | SLAVE\_CONTROL | Controls slave operation | +| 0x00 | COPROCESSOR\_READY | Indicates if co-processor is ready | +| 0x04 | MAX\_TX\_BUF\_LEN | Maximum length of DMA data co-processor can transmit | +| 0x08 | MAX\_RX\_BUF\_LEN | Maximum length of DMA data co-processor can receive | +| 0x0C | TX\_BUF\_LEN | Updated whenever co-processor wants to transmit data | +| 0x10 | RX\_BUF\_LEN | Updated whenever co-processor can receive data | +| 0x14 | COPROCESSOR\_CONTROL | Controls co-processor operation | -### 3.4. Timing Diagrams +### 4.4 Timing Diagrams The following diagrams summarize the SPI transactions as used by Hosted: @@ -201,52 +192,45 @@ The following diagrams summarize the SPI transactions as used by Hosted: *SPI Transaction using 2 data lines* -## 4. SPI HD Operation in Hosted +## 5. SPI HD Operation -### 4.1 Transport Initialization +### 5.1. Initialization -#### 4.1.1. Slave and Host Initialization +#### 5.1.1. Co-processor and Host Initialization -The slave starts up and initialises the SPI HD transport. When the -slave is ready it writes the value `SLAVE_IS_READY` (0xEE) to the -SLAVE\_READY register. +The co-processor starts up and initialises the SPI HD transport. When the co-processor is ready it writes the value `COPROCESSOR_IS_READY` (0xEE) to the COPROCESSOR\_READY register. -The Host starts up and initialises the SPI HD transport. When ready, it -polls the SLAVE\_READY register on the slave until it reads the value -`SLAVE_IS_READY`. +The Host starts up and initialises the SPI HD transport. When ready, it polls the COPROCESSOR\_READY register on the co-processor until it reads the value +`COPROCESSOR_IS_READY`. -Once slave is ready, host prepare for interrupts triggered by -`Data_Ready`, and sets bit 0 on the SLAVE\_CONTROL register to 1. This -opens the data path to the slave. +Once co-processor is ready, host prepare for interrupts triggered by `Data_Ready`, and sets bit 0 on the COPROCESSOR\_CONTROL register to 1. This opens the data path to the co-processor. -Both host and slave are now ready to communicate. +Both host and co-processor are now ready to communicate. -The first packet the slave transfers to the host is a Capabilities -Packet, stating what the slave is capable of supporting: +The first packet the co-processor transfers to the host is a Capabilities Packet, stating what the co-processor is capable of supporting: - WLAN, Bluetooth, etc. - the number of data lines supported for SPI HD -The host uses this packet to determine what the slave is capable of -supporting. +The host uses this packet to determine what the co-processor is capable of supporting. ```mermaid sequenceDiagram participant h as Host - participant s as Slave + participant s as Co-processor note over s,h : Init loop Executed periodically - h ->> s : Read SLAVE_READY reg + h ->> s : Read COPROCESSOR_READY reg s -->> h : Not Ready (!0xEE) end - note over s : Ready: SLAVE_READY = 0xEE - h ->> s : Read SLAVE_READY reg + note over s : Ready: COPROCESSOR_READY = 0xEE + h ->> s : Read COPROCESSOR_READY reg s -->> h : Ready (0xEE) note over h : Enable Data_Ready interrupt - h ->> s : Set SLAVE_CONTROL reg = 1 + h ->> s : Set COPROCESSOR_CONTROL reg = 1 note over s : Open Data Path note over s : Prepare Capability Data @@ -261,20 +245,16 @@ sequenceDiagram *SPI HD Initialization Sequence* -#### 4.1.2. Number of Data Lines Used +#### 5.1.2. Number of Data Lines Used -After initialization, the host initially communicates with the slave -using two data lines. If the slave is capable of supporting four data -lines (from the Capabilities Packet sent by the slave), and the host -is configured to also use four data lines, then four data lines will -be used for subsequent data transfers. +After initialization, the host initially communicates with the co-processor using two data lines. If the co-processor is capable of supporting four data +lines (from the Capabilities Packet sent by the co-processor), and the host is configured to also use four data lines, then four data lines will be used for subsequent data transfers. -If neither the host or slave is capable of transferring data using -four data lines, then only two data lines will be used. +If neither the host or co-processor is capable of transferring data using four data lines, then only two data lines will be used. -### 4.2. Slave Transfers Data to Host +### 5.2. Co-processor to Host Transfer -Slave asserts `Data_Ready` to tell the host it has data to send. +Co-processor asserts `Data_Ready` to tell the host it has data to send. Host reads the TX\_BUF\_LEN register. @@ -284,20 +264,14 @@ Host reads the TX\_BUF\_LEN register. > TX\_BUF\_LEN register. The host subtracts the read length from its cached read length -(initial value is zero) to discover how much more data the slave wants -to transfer to the host. The host can now read the data using the -RDDMA command, ending the transfer with CMD8. The host now updates its -cached read length with the slave's read length. +(initial value is zero) to discover how much more data the co-processor wants to transfer to the host. The host can now read the data using the RDDMA command, ending the transfer with CMD8. The host now updates its cached read length with the co-processor's read length. -After reading TX\_BUF\_LEN register, host sends CMD9. This tells the -slave that the host has read the register and it is safe for the slave -to update the register (if required) and deassert the `Data_Ready` -signal. +After reading TX\_BUF\_LEN register, host sends CMD9. This tells the co-processor that the host has read the register and it is safe for the co-processor to update the register (if required) and deassert the `Data_Ready` signal. ```mermaid sequenceDiagram participant h as Host - participant s as Slave + participant s as Co-processor note over s : Prepare data to send, update TX_BUF_LEN @@ -319,10 +293,10 @@ sequenceDiagram *SPI HD Read Sequence* -### 4.3. Host Transfers Data to the Slave +### 5.3. Host to Co-processor Transfer Host reads the RX\_BUF\_LEN register to discover how many buffers are -available on the slave (each buffer is of size MAX\_RX\_BUF\_LEN). If +available on the co-processor (each buffer is of size MAX\_RX\_BUF\_LEN). If there are not enough buffers to store the data to be sent, the host should wait and re-read the register until there are enough buffers. @@ -332,7 +306,7 @@ ending each buffer transfer with WR_DONE. ```mermaid sequenceDiagram participant h as Host - participant s as Slave + participant s as Co-processor note over h : Prepare data to send @@ -350,60 +324,482 @@ sequenceDiagram *SPI HD Write Sequence* -### 4.4. Code Reference +### 5.4. Code Reference + +- [`slave/main/spi_hd_slave_api.c`](https://github.com/espressif/esp-hosted/blob/feature/esp_as_mcu_host/slave/main/spi_hd_slave_api.c) implements the code to run the SPI HD driver on the co-processor +- [`host/drivers/transport/spi_hd/spi_hd_drv.c`](https://github.com/espressif/esp-hosted/blob/feature/esp_as_mcu_host/host/drivers/transport/spi_hd/spi_hd_drv.c) implements the generic code to run the SPI HD driver on the host +- [`host/port/spi_hd_wrapper.c`](https://github.com/espressif/esp-hosted/blob/feature/esp_as_mcu_host/host/port/spi_hd_wrapper.c) implements the ESP-IDF specific code used by the generic SPI HD driver on the host -- `slave/main/spi_hd_slave_api.c` implements the code to run the SPI - HD driver on the slave -- `host/drivers/transport/spi_hd/spi_hd_drv.c` implements the generic - code to run the SPI HD driver on the host -- `host/port/spi_hd_wrapper.c` implements the ESP-IDF specific code - used by the generic SPI HD driver on the host +## 6. Hardware Considerations -## 5. Hardware Considerations +### 6.1. General Considerations + +- Ensure equal trace lengths for all SPI connections, whether using jumper wires or PCB traces. +- Use the lower clock frequency like 5 MHz for evaluation. Once solution verified, optimise the clock frequency in increasing steps to max possible value. To find out practical maximum SPI co-processor frequency for your co-processor, check `IDF_PERFORMANCE_MAX_SPI_CLK_FREQ` in [ESP-IDF co-processor SPI clock benchmark](https://github.com/espressif/esp-idf/blob/master/components/esp_driver_spi/test_apps/components/spi_bench_mark/include/spi_performance.h) +- Verify voltage compatibility between host and co-processor devices. +- Provide proper power supply decoupling for both host and co-processor devices. + +### 6.2. Jumper Wires + +- Jumper wires are suitable for initial testing and prototyping. +- Use high-quality, low-capacitance jumper wires. +- Keep wires as short as possible, ideally under 10 cm. +- Arrange wires to minimize crosstalk, especially for clock and data lines. +- Possibly, use twisted pairs for clock and data lines to reduce electromagnetic interference. +- If possible, use a ground wire between every signal wire to improve signal integrity. +- Connect as many grounds as possible to improve common ground reference and reduce ground noise. + +> [!IMPORTANT] +> +> Quad SPI (QSPI) should not be used with jumper cables due to signal integrity issues. Use Dual SPI for evaluation with jumper cables. -### 5.1. Jumper Wires +### 6.3. PCB Design -While jumper wires can be used to test SPI, it is recommended to use -short wires (5 to 10 cm in length, shorter is better) to -minimise propagation delay and noise on the signals. +For optimal performance and reliability in production designs: -### 5.2 PCB Design +- Ensure equal trace lengths for all SPI signals (CLK, MOSI, MISO, CS) as much as possible. This practice, known as length matching, is crucial for maintaining signal integrity and reducing timing skew, especially at higher frequencies. +- If perfect length matching is not possible, prioritize matching the clock (CLK) trace length with the data lines. +- Use controlled impedance traces for high-speed signals. +- Place bypass capacitors close to the power pins of both the host and co-processor devices. +- Consider using series termination resistors on the clock and data lines to reduce reflections. +- For high-speed designs, use a 4-layer PCB with dedicated power and ground planes. +- Quad SPI (QSPI) should only be implemented on a properly designed PCB. -The PCB traces for SPI should be equal length and kept as short as -possible. The signals, especially the `CLK` signal, should be isolated -from other signals using a ground plane to minimise crosstalk. -### 5.3 Testing the SPI Connection +### 6.4. Advanced Considerations -**Using a Lower Clock Speed** +- Calculate the maximum allowed trace length based on your clock frequency and PCB material. +- Consider the capacitive load on the SPI bus, especially for longer traces or when using multiple co-processor devices. +- For very high-speed designs, consider using differential signaling techniques. +- Implement proper EMI/EMC design techniques to minimize electromagnetic interference. +- Use an oscilloscope or logic analyzer to verify signal integrity and timing. +- Start with a lower clock frequency and gradually increase it while monitoring performance. +- Ensure proper grounding between the host and co-processor devices. +- If using multiple power supplies, ensure they share a common ground. +- Consider using level shifters if the host and co-processor operate at different voltage levels. -You can use a lower clock speed to verify the connections. For SPI, -you can start with 10 MHz or lower. +## 7. Hardware Setup -To configure this, use `Menuconfig` on the Host: **Component -config** ---> **ESP-Hosted config** ---> **SPI Half-duplex Configuration** -and set **SPI HD Clock Freq (MHz)**. +> [!IMPORTANT] +> +> Remember that Quad SPI (using D2 and D3) should only be used with a properly designed PCB, not with jumper wires. + +Before flashing the co-processor and host, ensure that you have made the correct hardware connections. The following tables show the recommended connections for SPI Half Duplex mode: + + + +### Host connections +| Signal | ESP32-S3 | ESP32-P4-Function-EV-Board | +|------------|-------------|----------| +| CLK | 19 | 18 | +| D0 | 13 | 14 | +| D1 | 35 | 15 | +| D2 | 20 | 16 | +| D3 | 9 | 17 | +| CS | 47 | 19 | +| Handshake | 17 | 16 | +| Data Ready | 12 | 6 | +| Reset Out | 42 | 54 | +| GND | GND | GND | + +- GPIOs can be re-configured to any other GPIOs, while co-processor configuration is done. + - Make sure the configuration and hardware connections match. +- Classic ESP32 + - Not supported as host + - Rest all chipsets are supported as host +- ESP32-S2/C2/C3/C5/C6 + - Pins for SPI Half Duplex Host need to be figured out yet. +- ESP32-P4 + - For ESP32-P4-Function-EV-Board, the SDIO onboard pins are re-used for SPI Half Duplex Host. + - For Non ESP32-P4-Function-EV-Board, pins for SPI Half Duplex Host need to be figured out yet. + +### Co-processor connections + +| Signal | ESP32-C6 on ESP32-P4-Function-EV-Board | ESP32-C2/C3/C6 | ESP32-C5 | +|-------------|---------------------------------------|----------|----------| +| CLK | 19 | 6 | 6 | +| D0 | 20 | 7 | 7 | +| D1 | 21 | 2 | 2 | +| D2 | 22 | 5 | 5 | +| D3 | 23 | 4 | 4 | +| CS | 18 | 10 | 10 | +| Data Ready | 2 | 0 | 13 | +| Reset In | EN/RST | EN/RST | EN/RST | +| GND | GND | GND | GND | + +- GPIOs can be re-configured to any other GPIOs, while co-processor configuration is done. + - Make sure the configuration and hardware connections match. +- ESP32, ESP32-C2/C3/C5/C6/S2/S3 + - All supported as SPI Half Duplex co-processor + - Pins for SPI Half Duplex co-processor need to be figured out yet for other boards > [!NOTE] -> The actual clock frequency used is determined by the hardware. Use -> an oscilloscope or logic analyzer to check the clock frequency. +> +> **QSPI Testing:** +> - Tested on ESP32-P4-Function-EV-Board +> - ESP32-P4 as host, ESP32-C6/C3 as QSPI co-processor +> - Reused existing SDIO connections for QSPI on C6 and P4 +> +> **Dual SPI Testing:** +> - ESP32-S3 host with ESP32-C5 co-processor +> - Tested using jumper cables +> +> **Performance Optimization:** +> - Always prefer to use IO_MUX pins from datasheet for optimal performance on both sides +> +> **Portability:** +> - Once ported, any other non ESP host with Dual SPI or QSPI can be used + +## 8. Set-Up ESP-IDF + +Before setting up the ESP-Hosted co-processor & host for SPI Half Duplex mode, ensure that ESP-IDF is properly installed and set up on your system. + +#### Option 1: Installer Way + +- **Windows** + - Install and setup ESP-IDF on Windows as documented in the [Standard Setup of Toolchain for Windows](https://docs.espressif.com/projects/esp-idf/en/latest/esp32/get-started/windows-setup.html). + - Use the ESP-IDF [Powershell Command Prompt](https://docs.espressif.com/projects/esp-idf/en/latest/esp32/get-started/windows-setup.html#using-the-command-prompt) for subsequent commands. + +- **Linux or MacOS** + - For bash: + ```bash + bash docs/setup_esp_idf__latest_stable__linux_macos.sh + ``` + - For fish: + ```fish + fish docs/setup_esp_idf__latest_stable__linux_macos.fish + ``` + +#### Option 2: Manual Way -**3.4.2. Using Two Data Lines** +Please follow the [ESP-IDF Get Started Guide](https://docs.espressif.com/projects/esp-idf/en/latest/esp32/get-started/index.html) for manual installation. -You can configure SPI HD to use two data line only instead of -four. This will reduce noise on the signal lines and help you verify -that the SPI HD protocol is working at the logical level. +## 9. Flashing the Co-processor -To configure this, use `Menuconfig` on the Host: **Component -config** ---> **ESP-Hosted config** ---> **SPI Half-duplex Configuration** -and set **Num Data Lines to use** to **2 data lines**. +| Supported Co-processor Targets | ESP32 | ESP32-C2 | ESP32-C3 | ESP32-C5 | ESP32-C6 | ESP32-C61 | ESP32-S2 | ESP32-S3 | +| ------------------------------ | ----- | -------- | -------- | -------- | -------- | --------- | -------- | -------- | -You can leave the SPI HD configuration on the Hosted Slave as four -data lines, as the host determines the number of data lines to use. If -you want to change this, then on the slave use `Menuconfig`: **Example -Configuration** ---> **SPI Half-duplex Configuration** and set **Num -Data Lines to use** to **2 data lines**. +### 9.1 Create Co-processor Project +1. Create co-processor project possibly outside of ESP-IDF project directory using -## 6. References + ```bash + idf.py create-project-from-example "espressif/esp_hosted:slave" + ``` + +2. Navigate to the created project directory. + +3. Configure the project for your target ESP chip: + + ```bash + idf.py set-target + ``` + Replace `` with your specific co-processor ESP chip (e.g., esp32c3, esp32s3). + +### 9.2 Co-processor Config +Configure the co-processor project using +``` +idf.py menuconfig +``` + +#### 9.2.1 Transport config + - Navigate to "Example configuration" -> "Transport layer" + - Select "SPI Half-duplex" + +#### 9.2.2 Any other config + - Optionally, Configure any additional SPI-specific settings like co-processor GPIOs, SPI mode, etc. + +###### Generated files +- Generated config files are (1) `sdkconfig` file and (2) internal `sdkconfig.h` file. +- Please note, any manually changes done to these generated files, would not take effect. + +###### Defaulting specific config (Optional) +- This is advanced option, so please be careful. +- To mark some config options as default, you can add specific config line in file, `sdkconfig.defaults.`. So whenever next time building, you do not need to re-configure. + +### 9.3 Co-processor Build +Build the co-processor project + +``` +idf.py build +``` + +### 9.4 Co-processor Flashing + +There are two methods to flash the ESP-Hosted co-processor firmware: + +##### 9.4.1 Serial Flashing (Initial Setup) + +For the initial setup or when OTA is not available, use serial flashing. + +Flash the co-processor firmware using +``` +idf.py -p flash +``` + +> [!NOTE] +> +> If you are not able to flash the co-processor, there might be a chance that host is not allowing to to do so. +> +> Put host in bootloader mode using following command and then retry flashing the co-processor +> +> ```bash +> esptool.py -p **** --before default_reset --after no_reset run +> ``` + +Monitor the output (optional): +``` +idf.py -p monitor +``` + +##### 9.4.2. Co-processor OTA Flashing (Subsequent Updates) + +For subsequent updates, you can re-use ESP-Hosted-MCU transport, as it should be already working. While doing OTA, Complete co-processor firmware image is not needed and only co-processor application partition, 'network_adapter.bin' need to be re-flashed remotely from host. + +1. Ensure your co-processor device is connected and communicating with the host with existing ESP-Hosted-MCU. + +2. Create a web server +You can re-use your existing web server or create a new locally for testing. Below is example to do it. + - Make a new directory so that web server can be run into it and navigate into it + - Create simple local web server using python3 + + ```bash + python3 -m http.server 8080 + ``` +3. Copy the co-processor app partition `network_adapter.bin` in the directory where you created the web server. + - The `network_adapter.bin` can be found in your co-processor project build at `/build/network_adapter.bin` + +4. Verify if web server is set-up correctly + - Open link `http://127.0.0.1:8080` in the browser and check if network_adapter.bin is available. + - Right click and copy the complete URL of this network_adapter.bin and note somewhere. + +5. On the **host side**, use the `esp_hosted_ota` function to initiate the OTA update: + + ```c + #include "esp_hosted_api.h" + + const char* image_url = "http://example.com/path/to/network_adapter.bin"; //web server full url + esp_err_t ret = esp_hosted_ota(image_url); + if (ret == ESP_OK) { + printf("co-processor OTA update failed[%d]\n", ret); + } + ``` + + This function will download the firmware in chunk by chunk as http client from the specified URL and flash it to the co-processor device through the established transport. + In above web server example, You can paste the copied url earlier. + + +6. Monitor the OTA progress through the console output on both the host and co-processor devices. + +> [!NOTE] +> +> - The `esp_hosted_ota` function is part of the ESP-Hosted-MCU API and handles the OTA process through the transport layer. +> - Ensure that your host application has web server connectivity to download the firmware file. +> - The co-processor device doesn't need to be connected to the web server for this OTA method. + +## 10. Flashing the Host + +Host are required to support 2 data line SPI (dual SPI) or 4 line SPI (quad SPI or QSPI) in their hardware. All ESP chipsets hardware support dual, quad SPI. + +| Supported Host Targets | Any ESP chipset | Any Non-ESP chipset | +| ----------------------- | --------------- | ------------------- | + +Non ESP chipset may need to port the porting layer. It is strongly recommanded to evaluate the solution using ESP chipset as host before porting to any non-esp chipset. +For Quad SPI, PCB is only supported. Dual SPI could be evaluted using jumper cables. + +Non-ESP Hosts, while porting, need to ensure that the Half duplex protocol and framing is exactly same as that of co-processor. + +### 10.1 Select Example to Run in Hosted Mode + +Select an example from the [ESP-IDF examples directory](https://github.com/espressif/esp-idf/tree/master/examples) that you wish to run in ESP-Hosted mode. All Wi-Fi and Bluetooth examples are supported. For simplicity and demonstration purposes, we will use the [ESP-IDF iperf example](https://github.com/espressif/esp-idf/tree/master/examples/wifi/iperf). + +### 10.2 Host Project Component Configuration + +Now that ESP-IDF is set up, follow these steps to prepare the host: + +###### 1. Navigate to the iperf example in your ESP-IDF directory: + ``` + cd $IDF_PATH/examples/wifi/iperf + ``` + +###### 2. Dependency components + Add the required components to the project's `idf_component.yml` file: + ``` + idf.py add-dependency "espressif/esp_wifi_remote" + idf.py add-dependency "espressif/esp_hosted" + ``` + +###### 3. Remove conflicting configuration + Open the `main/idf_component.yml` file and remove/comment the following block if present: + ``` + # ------- Delete or comment this block --------- + espressif/esp-extconn: + version: "~0.1.0" + rules: + - if: "target in [esp32p4]" + # ----------------------------------- + ``` + This step is necessary because esp-extconn and esp-hosted cannot work together. + +###### 4. Disable native Wi-Fi if available + If your host ESP chip already has native Wi-Fi support, disable it by editing the `components/soc//include/soc/Kconfig.soc_caps.in` file and changing all `WIFI` related configs to `n`. + + If you happen to have both, host and co-processor as same ESP chipset type (for example two ESP32-C2), note an [additional step](docs/troubleshooting/#1-esp-host-to-evaluate-already-has-native-wi-fi) + + +### 8.3 Menuconfig, Build and Flash Host + +##### 1. High performance configurations + This is optional step, suggested for high performance applications. + + If using ESP32-P4 as host: + - Remove the default `sdkconfig.defaults.esp32p4` file. + - Create a new `sdkconfig.defaults.esp32p4` file with the following content: + ``` + CONFIG_ESP_WIFI_STATIC_RX_BUFFER_NUM=16 + CONFIG_ESP_WIFI_DYNAMIC_RX_BUFFER_NUM=64 + CONFIG_ESP_WIFI_DYNAMIC_TX_BUFFER_NUM=64 + CONFIG_ESP_WIFI_AMPDU_TX_ENABLED=y + CONFIG_ESP_WIFI_TX_BA_WIN=32 + CONFIG_ESP_WIFI_AMPDU_RX_ENABLED=y + CONFIG_ESP_WIFI_RX_BA_WIN=32 + + CONFIG_LWIP_TCP_SND_BUF_DEFAULT=65534 + CONFIG_LWIP_TCP_WND_DEFAULT=65534 + CONFIG_LWIP_TCP_RECVMBOX_SIZE=64 + CONFIG_LWIP_UDP_RECVMBOX_SIZE=64 + CONFIG_LWIP_TCPIP_RECVMBOX_SIZE=64 + + CONFIG_LWIP_TCP_SACK_OUT=y + ``` + + For other hosts also, you can merge above configs in corresponding `sdkconfig.defaults.esp32XX` file. + +###### 2. Set environment for your host ESP chip: + + ``` + idf.py set-target + ``` + Replace `` with your specific ESP chip (one of esp32, esp32c2, esp32c3, esp32c5, esp32c6, esp32s2, esp32s3, esp32p4). + +###### 3. Flexible Menuconfig configurations + + ``` + idf.py menuconfig + ``` + ESP-Hosted-MCU host configurations are available under "Component config" -> "ESP-Hosted config" + 1. Select "SPI Half-duplex" as the transport layer + 2. Change co-processor chipset to connect to under "Slave chipset to be used" + 3. Change Number of data lines to 2 or 4 based on the co-processor using "SPI Half-duplex Configuration" -> "Num Data Lines to use" + 3. Opttionally, Configure SPI-specific settings like + - SPI Clock Freq (MHz) + - SPI Mode + - SPI Host GPIO Pins + - SPI Checksum Enable/Disable (Checksum is recommended to be enabled as spi hardware doesn't have any error detection) + + > [!NOTE] + > + > The actual clock frequency used is determined by the hardware. Use an oscilloscope or logic analyzer to check the clock frequency. + +###### 4. Build the project: + ``` + idf.py build + ``` + +###### 5. Flash the firmware: + ``` + idf.py -p flash + ``` + +###### 6. Monitor the output: + ``` + idf.py -p monitor + ``` + - If host was put into bootloader mode earlier, it may need manual reset + +## 11. Testing and Troubleshooting + +After flashing both the co-processor and host devices, follow these steps to connect and test your ESP-Hosted SPI Half Duplex setup: + +1. Connect the hardware: + - Follow the pin assignments for SPI Half Duplex as specified in [Hardware Setup](#7-hardware-setup). + - Ensure all necessary connections are made, including power, ground, and the extra GPIO signals (Data_Ready and Reset). + +2. Power on both devices. + +3. Verify the connection: + - Check the serial output of both devices for successful initialization messages. + - Look for messages indicating that the SPI Half Duplex transport layer has been established. + +4. Logs at both sides: + - Host: + + ``` + I (522) transport: Attempt connection with slave: retry[0] + I (525) transport: Reset slave using GPIO[54] + I (530) os_wrapper_esp: GPIO [54] configured + I (535) gpio: GPIO[54]| InputEn: 0| OutputEn: 1| OpenDrain: 0| Pullup: 0| Pulldown: 0| Intr:0 + I (1712) transport: Received INIT event from ESP32 peripheral + I (1712) transport: EVENT: 12 + I (1712) transport: EVENT: 11 + I (1715) transport: capabilities: 0xe8 + I (1719) transport: Features supported are: + I (1724) transport: - HCI over SPI + I (1728) transport: - BLE only + I (1732) transport: EVENT: 13 + I (1736) transport: ESP board type is : 13 + + I (1741) transport: Base transport is set-up + ``` + + - Co-processor: + + ``` + I (492) fg_mcu_slave: ********************************************************************* + I (501) fg_mcu_slave: ESP-Hosted-MCU Slave FW version :: X.Y.Z + + I (511) fg_mcu_slave: Transport used :: + I (520) fg_mcu_slave: ********************************************************************* + I (529) fg_mcu_slave: Supported features are: + I (534) fg_mcu_slave: - WLAN over SPI + I (538) h_bt: - BT/BLE + I (541) h_bt: - HCI Over SPI + I (545) h_bt: - BLE only + ``` + +5. Test basic functionality: + - The iperf example automatically attempts to connect to the configured Wi-Fi network. Watch the serial output for connection status. + - If the automatic connection fails, you can manually initiate a Wi-Fi scan and connection: + ``` + sta_scan + sta_connect + ``` +6. Additional commands to test: + - Get IP address: `sta_ip` + - Disconnect from Wi-Fi: `sta_disconnect` + - Set Wi-Fi mode: `wifi_mode ` (where mode can be 'sta', 'ap', or 'apsta') + +7. Advanced iperf testing: + Once connected, you can run iperf tests: + + | Test Case | Host Command | External STA Command | + |-----------|--------------|----------------------| + | UDP Host TX | `iperf -u -c -t 60 -i 3` | `iperf -u -s -i 3` | + | UDP Host RX | `iperf -u -s -i 3` | `iperf -u -c -t 60 -i 3` | + | TCP Host TX | `iperf -c -t 60 -i 3` | `iperf -s -i 3` | + | TCP Host RX | `iperf -s -i 3` | `iperf -c -t 60 -i 3` | + + Note: Replace `` with the IP address of the external STA, and `` with the IP address of the ESP-Hosted device. + +8. Troubleshooting: + - If you encounter issues, refer to section 6.3 for testing the SPI connection. + - Consider using a lower clock speed or checking your [hardware setup](#7-hardware-setup) if you experience communication problems. + - ESP-Hosted-MCU troubleshooting guide: [docs/troubleshooting.md](docs/troubleshooting.md) + +9. Monitoring and debugging: + - Use the serial monitor on both devices to observe the communication between the host and co-processor. + - For more detailed debugging, consider using a logic analyzer to examine the SPI signals. + +## 12. References + +- ESP SPI co-processor HD (Half Duplex) Mode Protocol: https://docs.espressif.com/projects/esp-idf/en/latest/esp32/api-reference/protocols/esp_spi_slave_protocol.html -- ESP SPI Slave HD (Half Duplex) Mode Protocol: https://docs.espressif.com/projects/esp-idf/en/latest/esp32/api-reference/protocols/esp_spi_slave_protocol.html diff --git a/docs/troubleshooting.md b/docs/troubleshooting.md index 0e1c0454b9..4e6668cfa2 100644 --- a/docs/troubleshooting.md +++ b/docs/troubleshooting.md @@ -2,21 +2,54 @@ **Table of Contents** -- [1. Raw Throughput Testing Option](#1-raw-throughput-testing-option) -- [2. Make sure Hosted code is in sync for Master and Slave](#2-make-sure-hosted-code-is-in-sync-for-master-and-slave) -- [3. Make sure GPIOs match on both the Host and Slave](#3-make-sure-gpios-match-on-both-the-host-and-slave) -- [4. ESP-Hosted Master Not Connecting to Slave](#4-esp-hosted-master-not-connecting-to-slave) -- [5. Getting `Drop Packet` Errors](#5-getting-drop-packet-errors) -- [6. References](#6-references) +- [1. ESP host to evaluate already has Native Wi-Fi](#1-esp-host-to-evaluate-already-has-native-wi-fi) +- [2. Raw Throughput Testing](#2-raw-throughput-testing) +- [3. Make sure Hosted code is in sync for Master and Slave](#3-make-sure-hosted-code-is-in-sync-for-master-and-slave) +- [4. Make sure GPIOs match on both the Host and Slave](#4-make-sure-gpios-match-on-both-the-host-and-slave) +- [5. ESP-Hosted Master Not Connecting to Slave](#5-esp-hosted-master-not-connecting-to-slave) +- [6. Getting `Drop Packet` Errors](#6-getting-drop-packet-errors) +- [7. References](#7-references) -## 1. Raw Throughput Testing Option +## 1 ESP host to evaluate already has Native Wi-Fi -ESP-Hosted has a Raw Throughput Option to test sending of data between -the Host and Slave. This can be used to verify the hardware for signal -errors and to check the achievable throughput of Hosted. +Sometimes users have two ESPs, but both having Wi-Fi native capability. +This section explains how to run ESP-Hosted-MCU on ESP host chipsets that already have native Wi-Fi support. To run ESP-Hosted-MCU on such hosts, native Wi-Fi support needs to be disabled from base ESP-IDF in use. There are alternatives to do this: + +##### 1.1 Different ESP chipset types for host and slave +If host and slave not the same ESP chipset types, Wi-Fi capability can be disabled for host ESP chipset alone. Edit the ESP-IDF file +`components/soc//include/soc/Kconfig.soc_caps.in` and change +all `WIFI` related configs to `n`. For example: + +``` +config SOC_WIFI_SUPPORTED + bool + # default y # original configuration + default n +``` + +This should be done for all `SOC_WIFI_xxx` configs found in the file. + +For ESP Chipsets without native Wi-FI, `SOC_WIFI_xxx` configs will be +`n` by default. + + +##### 1.2 Same ESP chipset types for host and slave +There is possibility that you have two chipsets to evaluate, but both are exactly same chipset type. For example, two ESP32-C3. In this case, it is a two step build, first for host and second for slave. +While building for host ESP chipset, follow above (1) and flash, monitor. Once host is flashed fine, revert all the changes and flash the slave ESP chipset. + +## 2 Raw Throughput Testing + +While aiming the high performance and even while assessing the solution correctness, It is crucial to understand the bottlenecks in the system. +'Raw throughput testing' is simple transport level testing, which would showcase the maximum throughput that the transport is able to achieve, right now in current set-up. +In this test, dummy data is sent from one transport end to other continously, without involving Wi-Fi, Bluetooth or any other code legs. This test can be performed in following ways: +- Host to slave (Half duplex) : dummy data to be sent from host to slave continously +- Slave to Host (Half duplex) : dummy data to be sent from slave to host continously +- Full duplex bi-directional : dummy data to be sent from both the directions simulataneously + +This can verify hardware signal integrity and address porting issues. It also helps to assess the achievable throughput of the Hosted solution. It can be further optionally used for transport throughput fine-tuning. > [!IMPORTANT] -> Use this option to verify that Hosted hardware and software are +> Use Raw throughput test to verify that Hosted hardware and software are > working as expected before involving other software layers like > networking. @@ -30,7 +63,7 @@ on Host, enter `Menuconfig` and enable **Component config** ---> the data transfer direction: **Host to Slave**, **Slave to Host** or **Bidirectional**. -## 2. Make sure Hosted code is in sync for Master and Slave +## 3 Make sure Hosted code is in sync for Master and Slave The [README](../README.md) instructions will always fetch the latest version of ESP-Hosted from the Component Registry. Generally, this @@ -50,27 +83,23 @@ idf.py create-project-from-example "espressif/esp_hosted^0.0.9:slave" ``` This will ensure that both the Master and Slave code are fixed and in -sync for your project. +sync for your project. Please ensure you use latest versions for bug-fixes -> ![NOTE] +> [!NOTE] > When you switch Hosted versions, make sure you use the same version > of the Master and Slave code. There may be changes to the Hosted > implementation that may make different versions of Hosted Master and > Slave incompatible. -## 3. Make sure GPIOs match on both the Host and Slave +## 4 Make sure GPIOs match on both the Host and Slave -- check that the GPIOs you use on the Host and Slave are correct and - are connected together as expected -- verify that the GPIO values you set in `menuconfig` match the - hardware GPIOs you are actually using -- make sure you are not using incompatible GPIOs: - - on the ESP32, some GPIOs are input only and cannot be used for - output - - on the ESP32 and ESP32-C6, the GPIOs used for SDIO are fixed and - cannot be changed +- Check that the GPIOs you use on the Host and Slave are correct and are connected together as expected +- Verify that the GPIO values you set in `menuconfig` match the hardware GPIOs you are actually using +- Ensure that you are not using incompatible GPIOs: + - on the ESP32, some GPIOs are input only and cannot be used for output + - on the ESP32 and ESP32-C6, the GPIOs used for SDIO are fixed and cannot be changed -## 4. ESP-Hosted Master Not Connecting to Slave +## 5 ESP-Hosted Master Not Connecting to Slave If you see the following error on the ESP-Hosted Master console using the SPI Interface: @@ -87,20 +116,19 @@ E (1735) sdmmc_common: sdmmc_init_ocr: send_op_cond (1) returned 0x107 It means that something is wrong with the SPI or SDIO connection and the Host cannot communicate with the slave. -- check your physical GPIO signals and verify that they are connected -- make sure you have selected the same transports for the slave and - host (both are using the same SPI or SDIO interface) -- verify that the physical GPIO signals is the same as those assigned - to the system using `Menuconfig` on both the Host and Slave -- if you selected SDIO as the interface and your host is a ESP32, - there may be conflict with the GPIO used to bootstrap the ESP32 and - used in SDIO. See "Conflicts Between Bootstrap and SDIO on DAT2" in - [References](#6-references) for more information -- for SDIO, verify that pull-ups and other signalling requirments - (short, shielded connections) are also met. See the [SDIO - interface](sdio.md) page for more information on SDIO requirements - -## 5. Getting `Drop Packet` Errors +- Check your physical GPIO signals and verify that they are connected +- Ensure that you have selected the same transports for the slave and + host (both are using the same SPI or SDIO interface). + - It is expected that slave and host uses exact same codebase (git commit) + - Transport configured at slave matches to that of host + - Firmware configured with incompatible configurations also would result in issues. +- Verify that the physical GPIO signals is the same as those assigned to the system using `Menuconfig` on both the Host and Slave +- If you selected SDIO as the interface and your host is a classic ESP32, there may be conflict with the GPIO used to bootstrap the ESP32 and used in SDIO. See "Conflicts Between Bootstrap and SDIO on DAT2" in + [References](#7-references) for more information +- for SDIO, verify that pull-ups and other signalling requirments (short, shielded connections) are also met. See the [SDIO interface](sdio.md) page for more information on SDIO requirements +- If your transport allows on jumper cables, cross-check max length of jumper cables allowed + +## 6 Getting `Drop Packet` Errors For the SPI interface, if you see an error similar to this: @@ -116,7 +144,6 @@ Your SPI interface is facing signal integrity errors. - use an oscilloscope to check the physical signals on the SPI interface for noise, ringing, etc. that may affect the signals -## 6. References +## 7 References -- Conflicts Between Bootstrap and SDIO on DAT2: - https://docs.espressif.com/projects/esp-idf/en/latest/esp32/api-reference/peripherals/sd_pullup_requirements.html#conflicts-between-bootstrap-and-sdio-on-dat2) +- [Conflicts Between Bootstrap and SDIO on DAT2](https://docs.espressif.com/projects/esp-idf/en/latest/esp32/api-reference/peripherals/sd_pullup_requirements.html#conflicts-between-bootstrap-and-sdio-on-dat2) diff --git a/docs/wifi_design.md b/docs/wifi_design.md new file mode 100644 index 0000000000..6dd8a65d45 --- /dev/null +++ b/docs/wifi_design.md @@ -0,0 +1,83 @@ +# ESP-Hosted-MCU Wi-Fi Design & Implementation details + + +## Sequence Diagram for Wi-Fi communication + +On a ESP chipset with native Wi-Fi, a Wi-Fi api call or network data +from the application is processed internally on the chip and a Wi-Fi +response is returned to the application. + +```mermaid +sequenceDiagram + box Grey Host With Native WI-Fi + participant app as Application + participant api as ESP-IDF Wi-Fi Library + participant wifi as Wi-Fi Hardware + end + + app ->> api : esp_wifi_xxx() or Network Data + api ->> wifi : + Note over wifi : Do Wi-Fi action + wifi -->> api : Wi-Fi response or Data + api -->> app : Response or Network Data +``` + +Using Wi-Remote and ESP-Hosted, the Wi-Fi api call from the +application is converted into a Hosted Call and transported to the +slave. The slave converts the Hosted Call back into an Wi-Fi api +call. The response (optionally with data) is converted into a Hosted +Response and transported back to the host. On the host, the Hosted +Response is converted back into a Wi-Fi response (optionally with +data) is returned to the application. + +For Network Data, Hosted does not do data conversion and only +encapsulates the data for transport. + +```mermaid +sequenceDiagram + box Grey Host with ESP-Hosted + participant app as Application + participant remote as Wi-Fi Remote + participant hostedh as ESP-Hosted + participant transporth as Host Transport + end + + box SlateGrey Slave ESP-Hosted + participant transports as Slave Transport + participant hosteds as Slave Hosted + participant api as ESP-IDF Wi-Fi Library + participant wifi as Wi-Fi Hardware + end + + app ->> remote : esp_wifi_xxx() + remote ->> hostedh : esp_wifi_remote_xxx() + app ->> hostedh : Network Data + Note over hostedh : add Hosted header + hostedh ->> transporth : + + transporth ->> transports : SPI/SDIO + + transports ->> hosteds : + Note over hosteds : remove Hosted header + hosteds ->> api : esp_wifi_xxx() + api ->> wifi : Wi-Fi command + hosteds ->> wifi : Network Data + Note over wifi: Do Wi-Fi action + wifi -->> hosteds : Network Data + wifi -->> api : Wi-Fi response + api -->> hosteds : Response + Note over hosteds : add Hosted header + hosteds -->> transports : + + transports -->> transporth : SPI/SDIO + + transporth -->> hostedh : + Note over hostedh : remove Hosted header + hostedh -->> app : Network Data + hostedh -->> remote : Wi-Fi Command response + remote -->> app : Response +``` + + + +