Creating and Referencing Custom ESP-IDF Components

This tutorial shows how to create custom ESP-IDF components and share them between your ESP32 projects. We will create a basic component for running a simple PWM loop and will show how to reference it from 2 separate projects.

Before you begin, install VisualGDB 5.4 or later.

  1. Start Visual Studio and open the VisualGDB ESP-IDF project wizard:01-prj
  2. On the first page select “Create a new project based on a sample project”:02-newprj
  3. Select your ESP32 toolchain and the ESP-IDF checkout you want to use:03-toolchain
  4. Choose a sample project that will be cloned when creating your project. We will demonstrate the custom component based on the “blink” sample:
    04-sample
  5. Finally choose the debug method. In this tutorial we won’t be debugging the code, so we can use a serial port connection to program the firmware into the FLASH memory without configuring JTAG:05-com
  6. Press “Finish” to create the project and wait for it to load: 06-files
  7. All ESP-IDF projects consist of one or more components (i.e. static libraries) that are linked together to build the final executable. Most samples include only one component called “main”. For simplicity VisualGDB doesn’t show it as a separate component, but instead displays its contents directly under the project node. Now we will add another component to the project. Right-click on the “Components” folder and select “Add->New Item”:
    07-newitem
  8. Proceed with the default location, enter the component name and click “Add”:
    08-pwm
  9. VisualGDB will add the component files and will start refreshing the code model from the ESP-IDF. You won’t see the contents of the component until the refresh is complete:
    09-loadingWARNING: if the project is configured with idf_build_set_property(MINIMAL_BUILD ON), the new component will not appear in the list. You can disable that option by editing the main CMakeLists.txt file.
  10. Once the code model is refreshed, the new component will be shown in Solution Explorer similar to other components:10-files
  11. Select the main .cpp file and press F2 to start renaming it. Then change the extension to “.c”:11-rename
  12. Right-click on the component and select “Add->New File->C/C++ Header file”.  Append “\PublicIncludes” to the location field:12-pubincl
  13. Add the following code to the component:
    #pragma once
     
    #include "driver/gpio.h"
     
    #ifdef __cplusplus
    extern "C"
    {
    #endif
     
     void RunGPIOCycle(gpio_num_t gpio, double dutyCycle);
     
    #ifdef __cplusplus
    }
    #endif

    Then select the “RunGPIOCycle” declaration, press Ctrl+. and select “Create implementation”:
    13-makeimplVisualGDB will create an empty implementation for the RunGPIOCycle() function, however it won’t automatically find the “BasicPWM.h” header as the “PublicIncludes” directory is not yet added to the include search path.

  14. To fix that, we could add the header directory to either regular include directories (visible only for files inside the component) or exported directories (visible for all components in the same project). As the file will be included by the code that uses the component, add it to the exported directories:
    14-sharedirNote that the exported directories must have relative paths. For your convenience you can select “<Edit…>” in the “Additional Include Directories” field and let VisualGDB compute the correct relative paths automatically:edit
  15. Now you can the implementation to the RunGPIOCycle() function:
    void RunGPIOCycle(gpio_num_t gpio, double dutyCycle)
    {
        const int TotalCycles = 10000;
        const int position = TotalCycles * dutyCycle;
        gpio_set_level(gpio, 1);
        for (int i = 0; i < position; i++)
            asm("nop");
        gpio_set_level(gpio, 0);
        for (int i = position; i < TotalCycles; i++)
            asm("nop");
    }

    15-pwm

  16. Modify the blink_task() function in the main source file to call the RunGPIOCycle() function:
    #include "BasicPWM.h"
    #include <math.h>
     
    void blink_task(void *pvParameter)
    {
        gpio_pad_select_gpio(BLINK_GPIO);
        /* Set the GPIO as a push/pull output */
        gpio_set_direction(BLINK_GPIO, GPIO_MODE_OUTPUT);
     
        for (double i = 0; ; i += 0.005)
            RunGPIOCycle(BLINK_GPIO, (sin(i) + 1) / 2);
    }

    16-call

  17. Build the project and select “Debug->Program and Start Without Debugging” to program it to the FLASH memory:17-programEnsure that the on-board LED gradually turns on and off as the PWM duty cycle goes from 0 to 100% and back to 0.
  18. You can change the GPIO number of the LED via VisualGDB Project Properties -> ESP-IDF Project -> ESP-IDF Configuration -> Blink GPIO Number:gpio
  19. Now we will show how to import the component from another project. Create another project in a separate solution using the VisualGDB ESP-IDF project wizard:18-prj2
  20. Click on the “Components” node and select “Add->Existing Item”:
    19-addcomp
  21. Then locate the component.mk file of the BasicPWM component:20-locatemak
  22. See how VisualGDB automatically updates the project’s Makefile to include the new component:21-relpath
  23. You can now use the component exactly the same way as you did in the first project – include the <BasicPWM.h> file and call the RunGPIOCycle() function. ESP-IDF will automatically use the correct include paths and will link the component with the final project:22-importedYou can also store your components outside the project directories. Simply pick an arbitrary location when creating a new component and VisualGDB will place it there, updating all necessary Makefiles.