Shaun Reed
esp-idf
While exploring my local network, I stumbled upon several devices named espressif
, and after some searching online I found the ESP32 was the likely culprit. I’ve always been interested in hardware and have experience working with electrical circuits from 120-480 volts, but I have never identified a resistor or even considered purchasing a breadboard.
So, I bought a board on Amazon because it was fast shipping and I had a long weekend coming up. I won’t share an Amazon link here for several reasons, but the exact model that I purchased was the ESP32-DevKitC V4 which comes with the ESP32-WROOM chip soldered on.
When the board first arrived, I tried out the Arduino IDE. It was fine for learning purposes, but it felt like the IDE was doing a lot for me without having the chance to really understand everything. Personally I prefer CLion, and I wanted to get back to the IDE that was familiar to me. More importantly, I wanted to work on the ESP32 without tethering myself to any IDE. I wanted the project to be completely stand-alone, and to document and install any dependencies manually. Using Sketches in the Arduino IDE wasn’t going to get me there.
Enter the Espressif IoT Development Framework, or the ESP-IDF for short. This looks slightly intimidating at first, but if you’re at all familiar with C programming you’ll get along fine. Before we dive into the ESP-IDF, lets take a step back and gain our bearings.
Ecosystems
Typically the ESP32 examples recommended for beginners will be using the arduino-esp32 APIs - Espressif provides one such example called WiFiClient. These examples are also made available in the Arduino IDE’s menus where you can create a new project (AKA a new sketch) based on this WiFiClient example. This example uses higher level Arduino APIs compared to what you’ll find in the ESP-IDF examples.
Why would Espressif, creators of the ESP32 and not the Arduino, want to support Arduino APIs? Simply because Arduinos are a popular point of entry for learning programming on microcontrollers, and supporting these APIs lowers the barrier of entry for ESP32 devices.
The ESP-IDF build system supports using those same Arduino APIs via an IDF component configuration marking arduino-esp32
as a dependency. See the Espressif GitHub example here, paired with the IDF component documentation from espressif.
ESP-IDF
In the sections below, we will use the simpler arduino-esp32 APIs and examples to get familiar with deploying to the ESP32 using the ESP-IDF. This example isn’t going to do very much interesting other than validate our build system. If we are successful, it will simply write some output to the serial monitor. No lights, buttons, sensors, wifi, etc.
In a future post, I’ll look at drawing to an LCD display using the I2C communication protocol with the lower-level ESP APIs instead, and drop the dependency for arduino-esp32 entirely.
Installing
First we need to install the ESP-IDF following GitHub instructions on ubuntu 24.04
WARNING: If you are using arduino-esp32 APIs mentioned above, pay attention to arduino-esp32 releases for the latest supported ESP-IDF version. The latest ESP-IDF version is likely not currently supported by arduino-esp32. At the time of this writing, the latest arduino-esp32 release v3.1.1 is based on ESP-IDF 5.3.2
, so below we checkout the v5.3.2
branch of github.com/espressif/esp-idf.git
.
Install the prerequisites required for the ESP-IDF
1 | sudo apt-get install -y git wget flex bison gperf python3 python3-pip python3-venv cmake ninja-build ccache libffi-dev libssl-dev dfu-util libusb-1.0-0 |
Clone the ESP-IDF using the v5.3.2
branch and install. There will be a lot of output produced from these commands that is not shown below.
1 | git clone -b v5.3.2 git@github.com:espressif/esp-idf.git |
Switching Versions
To switch versions after previously installing the ESP-IDF
1 | cd /path/to/esp-idf |
If everything worked correctly, you should end up with this final output in your terminal -
1 | Done! You can now compile ESP-IDF projects. |
Project Example
First set up your environment to use ESP-IDF tools and to have the correct exports required for building with CMake. Unfortunately the way this is done is by sourcing a script in the ESP-IDF repository, after you’ve installed the ESP-IDf.
1 | source /path/to/esp-idf/export.sh |
You should now have the idf.py
tool available in your shell.
1 | idf.py |
To make initializing my environment for using the ESP-IDF simpler, I threw an alias in my .bash_aliases
file.
1 | echo "alias init-idf='source $HOME/Code/Clones/esp-idf/export.sh'" >> ~/.bash_aliases |
So next time when I want to work on ESP things I can just run init-idf
in a terminal to initialize my environment. From here we can use CMake to create an ESP-IDF project, and even flash to our device or open a serial monitor.
CMake
Setting up the project in CMake is fairly simple. For more information about build system see ESP-IDF - Build System
1 | mkdir esp-example |
1 | cmake_minimum_required(VERSION 3.26) |
And then create our main/
directory where we’ll store our source code and write a quick CMakeLists.txt
1 | mkdir main |
1 | idf_component_register( |
The project structure now looks like the following file tree. We’ll configure and build with cmake later, once we set up our dependencies in the ESP-IDF.
1 | esp-example/ |
Source Code
Now we write the source code for our application. I hope you’re prepared, because it’s a lot.
1 | cd esp-example |
1 |
|
This code simply attaches to the serial monitor at 115200 baud and prints Hello world!
every second. That’s it, the project is completed - we have our cmake lists set up and our source code in place. Next we will configure the project with cmake using the idf.py
tool.
Dependencies
Now that we have the tools to configure and build ESP-IDF projects with cmake, we can look at creating our first project. The example code that I’m going to build here is available at git.shaunreed.com/shaunrd0/klips. The commands below will not clone this repository, but instead set up the project as I did when creating it.
First set the build target to the ESP32 device. This tells the ESP-IDF which microcontroller we are working with.
1 | idf.py set-target esp32 |
Notice that the idf.py
command created the sdkconfig
file in our current directory, and produced a build/
directory -
1 | esp-example/ |
Next we need to add a dependency for the arduino-esp32 APIs. These simpler APIs will be a nice way to test our build system and get familiar with the tools.
IMPORTANT: We set the 3.1.1
version because the release is compatible with ESP-IDF v5.3.2
we installed in the first step.
1 | # https://github.com/espressif/arduino-esp32/releases/tag/3.1.1 |
After the idf.py
commands above, we have a new idf_component.yml
file -
1 | esp-example/ |
The contents should look like this, note the arduino-esp32
dependency is listed on the last line
1 | ## IDF Component Manager Manifest File |
Now we’re ready to configure the project and prepare to flash to our device. These commands will automatically pull in the arduino-esp32 dependency into our project directory so it may take some time to complete.
1 | cmake -B build |
If you’re lucky after some output from cmake you’ll get the following error
1 | -- Project sdkconfig file /home/shaun/Code/test-esp/sdkconfig |
Run the following sed
command to fix it. This just modifies a value that we could have probably tracked down in the idf.py menuconfig
interface, but that doesn’t sound like a good time.
1 | sed -i -e 's/CONFIG_FREERTOS_HZ=100/CONFIG_FREERTOS_HZ=1000/' sdkconfig |
Now rerun the cmake command to configure the project and it should complete normally.
1 | cmake -B build |
Finally, we can build the project
1 | cmake --build build |
We’ll hit this error, which was confusing at first for me because I wasn’t familiar with the differences between app_main()
and loop()
/ setup()
1 | /home/shaun/.espressif/tools/xtensa-esp-elf/esp-13.2.0_20240530/xtensa-esp-elf/bin/../lib/gcc/xtensa-esp-elf/13.2.0/../../../../xtensa-esp-elf/bin/ld: esp-idf/freertos/libfreertos.a(app_startup.c.obj):(.literal.main_task+0x24): undefined reference to `app_main' |
The build failed because we are using the ESP-IDF which expects the app_main()
convention. We can use the Arudino style by setting some configurations in the ESP-IDF.
We need to automatically start Arduino under the ESP-IDF so that we can make use of the Arduino loop()
and setup()
functions in our example. You can also use the ESP-IDF app_main()
function if preferred, see the examples below for differences between the two.
ESP-IDF example using app_main()
ESP-IDF example using Arduino loop() and setup()
You can alternatively do this in the TUI tool idf.py menuconfig
, but here is a bash command to append the same configuration to the sdkconfig
in your project directory that was generated by the ESP-IDF.
1 | echo "CONFIG_AUTOSTART_ARDUINO=y" >> sdkconfig |
We can once again build the project, only this this time it won’t fail to build.
1 | cmake --build build |
If everything completed normally, the build output will end with
1 | Project build complete. To flash, run: |
In the next and final section, we’ll look at flashing to the ESP32 and validating the program is behaving correctly using the serial monitor.
Flashing
Find the device for the ESP32. Mine is /dev/ttyUSB0
in the output below.
1 | $ ls /dev/tty* |
To flash run the following commands.
1 | # Manually selecting port with above output |
To open serial monitor via the commandline -
1 | idf.py monitor -b 115200 |
Or in CLion the Serial Monitor extension works well.
If everything is working correctly, you should see Hello world!
printed to the serial monitor repeatedly.