芯片型号:
ESP32-S2-MINI-1
开发环境:VS Code + ESP-IDF插件
1 工程创建
-
在VS Code中点击查看 - 命令面板(或
Ctrl+Shift+P
) -
输入
show examples projects
-
然后点击
Use current ESP-ID
-
选择一个例程(如
blink
),点击Create project using example blink
-
选择一个存放工程的文件夹(不能有中文路径和空格)
-
以 SDK 中的
blink
为模板,自动创建一个新工程 -
修改工程名称,如
my_project
,然后修改blink_example_main.c
修改,如app_main
-
修改项目
CMakeLists.txt
,项目名称改为my_project
-
修改
main
组件CMakeLists.txt
,包含C文件名称修改为app_main
(这个现在会自动修改) -
Makefile
可不修改,为旧版GNU Make编译,Cmake由CMakeLists.txt
控制 -
删除
example_test.py
、sdkconfig.defaults
、sdkconfig.defaults
不必要的文件 -
选择开发芯片型号,我的是
ESP32-S2
2 工程分析
以官方示例项目分析:
- myProject/
- CMakeLists.txt
- sdkconfig
- components/ - component1/ - CMakeLists.txt
- Kconfig
- src1.c
- component2/ - CMakeLists.txt
- Kconfig
- src1.c
- include/ - component2.h
- main/ - CMakeLists.txt
- src1.c
- src2.c
- build/
-
CMakeLists.txt
顶层项目CMakeLists.txt
是 CMake 用于学习如何构建项目的主要文件,可以在这个文件中设置项目全局的 CMake 变量。顶层项目CMakeLists.txt
文件会导入/tools/cmake/project.cmake
文件,由它负责实现构建系统的其余部分。该文件最后会设置项目的名称,并定义该项目。 -
sdkconfig
可以用mecuconfig
配置该文件,该文件保存了项目所有组件(包括 ESP-IDF 本身)的配置信息,使用idf插件可以可视化配置,点击设置按钮即可打开:
-
components
可选组件,包含项目的部分自定义组件,如果项目源文件较多,不要全部放在main
中,可以放在该自定义组件中。该组件名称不可随意改变(文件夹名),起别名需要在顶层CMakeLists.txt
中设置EXTRA_COMPONENT_DIRS
变量以查找其他指定位置处的组件。 -
main
main
是一个特殊的组件,它包含项目本身的源代码。main
是默认名称,CMake 变量COMPONENT_DIRS
默认包含此组件,但可以修改此变量。(一般不改) -
build
该目录编译时会由idf
自动创建,CMake 会配置项目,并在此目录下生成临时的构建文件。在主构建进程的运行期间,该目录还会保存临时目标文件、库文件以及最终输出的二进制文件。此目录通常不会添加到项目的源码管理系统中,也不会随项目源码一同发布。(即调试完可删除该目录发布项目)
注意:
- 每个组件目录都包含一个
CMakeLists.txt
文件,里面会定义一些变量以控制该组件的构建过程,以及其与整个项目的集成 - 每个组件还可以包含一个
Kconfig
文件,它用于定义menuconfig
时展示的 组件配置 选项。某些组件可能还会包含Kconfig.projbuild
和project_include.cmake
特殊文件,它们用于 覆盖项目的部分设置。
3 项目CMakeLists
文件
每个项目都有一个顶层 CMakeLists.txt
文件,包含整个项目的构建设置。默认情况下,项目 CMakeLists 文件会非常小。
3.1 必要部分
最小项目CMakeLists
文件必须包含下面三行代码:
cmake_minimum_required(VERSION 3.5)
include($ENV{IDF_PATH}/tools/cmake/project.cmake)
project(myProject)
-
cmake_minimum_required(VERSION 3.5)
:必须放在 CMakeLists.txt 文件的第一行,它会告诉 CMake 构建该项目所需要的最小版本号。ESP-IDF 支持 CMake 3.5 或更高的版本。 -
include($ENV{IDF_PATH}/tools/cmake/project.cmake)
:会导入 CMake 的其余功能来完成配置项目、检索组件等任务。 -
project(myProject)
:myProject
为工程名(如hello_world),该命令会创建项目本身,并指定项目名称。该名称会作为最终输出的二进制文件的名字,即myProject.elf
和myProject.bin
。每个 CMakeLists 文件只能定义一个项目。
3.2 可选的项目变量
-
COMPONENT_DIRS
:组件的搜索目录,默认为IDF_PATH/components
、PROJECT_DIR/components
、和EXTRA_COMPONENT_DIRS
。如果您不想在这些位置搜索组件,请覆盖此变量。 -
EXTRA_COMPONENT_DIRS
:用于搜索组件的其它可选目录列表。 -
COMPONENTS
:要构建进项目中的组件名称列表,默认为COMPONENT_DIRS
目录下检索到的所有组件。使用此变量可以“精简”项目以缩短构建时间。请注意,如果一个组件通过COMPONENT_REQUIRES
指定了它依赖的另一个组件,则会自动将其添加到COMPONENTS
中,所以COMPONENTS
列表可能会非常短。
注意:
- 以上变量中的路径可以是绝对路径,或者是相对于项目目录的相对路径。
- 使用 cmake 中的 set 命令 来设置这些变量,如
set(VARIABLE "VALUE")
。请注意,set()
命令需放在include(...)
之前,cmake_minimum(...)
之后。如:
cmake_minimum_required(VERSION 3.5)
set(EXTRA_COMPONENT_DIRS $ENV{IDF_PATH}/examples/common_components/led_strip)
include($ENV{IDF_PATH}/tools/cmake/project.cmake)
3.3 覆盖默认的构建规范
构建系统设置了一些全局的构建规范(编译标志、定义等),这些规范可用于编译来自所有组件的所有源文件。
例如,其中一个默认的构建规范是编译选项 Wextra
。假设一个用户想用 Wno-extra
来覆盖这个选项, 应在 project()
之后进行:
cmake_minimum_required(VERSION 3.5)
include($ENV{IDF_PATH}/tools/cmake/project.cmake)
project(myProject)
idf_build_set_property(COMPILE_OPTIONS "-Wno-error" APPEND)
4 组件CMakeLists
文件
组件 Components
是 COMPONENT_DIRS
列表中包含 CMakeLists.txt
文件的任何目录。是模块化且独立的代码,会被编译成静态库(.a 文件)并链接到应用程序。每个项目都包含一个或多个组件,这些组件可以是 ESP-IDF 的一部分,可以是项目自身组件目录的一部分,也可以从自定义组件目录添加
最小组件CMakeLists
文件
最小组件 CMakeLists.txt
文件通过使用 idf_component_register
将组件添加到构建系统中。
idf_component_register( SRCS "app_main.c"
INCLUDE_DIRS "include"
REQUIRES mbedtls)
-
SRCS
是源文件列表(*.c
、*.cpp
、*.cc
、*.S
),里面所有的源文件都将会编译进组件库中。 -
INCLUDE_DIRS
是目录列表,里面的路径会被添加到所有需要该组件的组件(包括 main 组件)全局 include 搜索路径中。(若只有一级目录,可以INCLUDE_DIRS "."
或不填) -
REQUIRES
实际上并不是必需的,但通常需要它来声明该组件需要使用哪些其它组件
也可以用CMake的set
语法:
set(srcs
"app_main.c" "src/oled.c")
idf_component_register(SRCS ${srcs}
INCLUDE_DIRS "inc")
4.1 同名组件
ESP-IDF 在搜索所有待构建的组件时,会按照COMPONENT_DIRS
指定的顺序依次进行,即 ESP-IDF 内部组件(IDF_PATH/components
)–> EXTRA_COMPONENT_DIRS
中的组件 --> 项目组件(PROJECT_DIR/components
)。如果这些目录中的两个或者多个包含具有相同名字的组件,则使用搜索到的最后一个位置的组件。这就允许将组件复制到项目目录中再修改以覆盖 ESP-IDF 组件,如果使用这种方式,ESP-IDF 目录本身可以保持不变。
如果在现有项目中通过将组件移动到一个新位置来覆盖它,项目不会自动看到新组件的路径。请运行
idf.py reconfigure
命令后(或删除项目构建文件夹)再重新构建。
4.2 重命名 main
组件
构建系统会对 main
组件进行特殊处理。假如 main
组件位于预期的位置(即${PROJECT_PATH}/main
),那么它会被自动添加到构建系统中。其他组件也会作为其依赖项被添加到构建系统中,这使用户免于处理依赖关系,并提供即时可用的构建功能。重命名 main
组件会减轻上述这些幕后工作量,但要求用户指定重命名后的组件位置,并手动为其添加依赖项。重命名 main
组件的步骤如下:
- 重命名
main
目录。 - 在项目
CMakeLists.txt
文件中设置EXTRA_COMPONENT_DIRS
,并添加重命名后的main
目录。 - 在组件的
CMakeLists.txt
文件中设置COMPONENT_REQUIRES
或COMPONENT_PRIV_REQUIRES
以指定依赖项。
4.3 预设的组件变量
以下专用于组件的变量可以在组件 CMakeLists 中使用,但不建议修改:
-
COMPONENT_DIR
:组件目录,即包含CMakeLists.txt
文件的绝对路径,它与CMAKE_CURRENT_SOURCE_DIR
变量一样,路径中不能包含空格。 -
COMPONENT_NAME
:组件名,与组件目录名相同。 -
COMPONENT_ALIAS
:库别名,由构建系统在内部为组件创建。 -
COMPONENT_LIB
:库名,由构建系统在内部为组件创建。
以下变量在项目级别中被设置,但可在组件 CMakeLists 中使用:
-
CONFIG_*
:项目配置中的每个值在 cmake 中都对应一个以CONFIG_
开头的变量。(参考Kconfig) -
ESP_PLATFORM
:ESP-IDF 构建系统处理 CMake 文件时,其值设为1。
4.4 构建/项目变量
以下是可作为构建属性的构建/项目变量,可通过组件 CMakeLists.txt 中的 idf_build_get_property
查询其变量值。
-
PROJECT_NAME
:项目名,在项目 CMakeLists.txt 文件中设置。 -
PROJECT_DIR
:项目目录(包含项目 CMakeLists 文件)的绝对路径,与CMAKE_SOURCE_DIR
变量相同。 -
COMPONENTS
:此次构建中包含的所有组件的名称,具体格式为用分号隔开的 CMake 列表。 -
IDF_VER
:ESP-IDF 的 git 版本号,由git describe
命令生成。 -
IDF_VERSION_MAJOR
、IDF_VERSION_MINOR
、IDF_VERSION_PATCH
: ESP-IDF 的组件版本,可用于条件表达式。请注意这些信息的精确度不如IDF_VER
变量,版本号v4.0-dev-*
,v4.0-beta1
,v4.0-rc1
和v4.0
对应的IDF_VERSION_*
变量值是相同的,但是IDF_VER
的值是不同的。 -
IDF_TARGET
:项目的硬件目标名称。 -
PROJECT_VER
:项目版本号。
4.5 组件配置
每个组件都可以包含一个 Kconfig
文件,和 CMakeLists.txt
放在同一目录下。Kconfig
文件中包含要添加到该组件配置菜单中的一些配置设置信息。
5 组件依赖
每个组件都需要声明它所依赖的组件,即requires
。
idf_component_register(...
REQUIRES mbedtls
PRIV_REQUIRES console spiffs)
-
REQUIRES
需要包含所有在当前组件的公共头文件里 #include 的头文件所在的组件。(如a组件REQUIRES b
b组件,若c组件包含了a组件的公共接口a.h,其可以递归包含b组件) -
PRIV_REQUIRES
需要包含被当前组件的源文件#include 的头文件所在的组件(除非已经被设置在了REQUIRES
中)。以及是当前组件正常工作必须要链接的组件。 -
REQUIRES
和PRIV_REQUIRES
的值不能依赖于任何配置选项 (CONFIG_xxx
宏)。这是因为在配置加载之前,依赖关系就已经被展开。其它组件变量(比如包含路径或源文件)可以依赖配置选择。 - 如果当前组件除了通用组件依赖项中设置的通用组件(比如 RTOS、libc 等)外,并不依赖其它组件,那么对于上述两个
REQUIRES
变量,可以选择其中一个或是两个都不设置。
如果组件仅支持某些硬件目标(IDF_TARGET
的值),则可以在 idf_component_register
中指定 REQUIRED_IDF_TARGETS
来声明这个需求。在这种情况下,如果构建系统导入了不支持当前硬件目标的组件时就会报错。
在 CMake 中,
REQUIRES
和PRIV_REQUIRES
是 CMake 函数target_link_libraries(... PUBLIC ...)
和target_link_libraries(... PRIVATE ...)
的近似包装。
5.1 组件依赖示例
假设现在有一个 car
组件,它需要使用 engine
组件,而 engine
组件需要使用 spark_plug
组件:
- autoProject/
- CMakeLists.txt
- components/ - car/ - CMakeLists.txt
- car.c
- car.h
- engine/ - CMakeLists.txt
- engine.c
- include/ - engine.h
- spark_plug/ - CMakeLists.txt
- plug.c
- plug.h
5.1.1 Car 组件
car.h
头文件
car.h
头文件是 car
组件的公共接口。该头文件直接包含了 engine.h
,这是因为它需要使用 engine.h
中的一些声明:
/* car.h */
#include "engine.h"
#ifdef ENGINE_IS_HYBRID
#define CAR_MODEL "Hybrid"
#endif
同时 car.c 也包含了 car.h
:
/* car.c */
#include "car.h"
这代表文件 car/CMakeLists.txt
需要声明 car
需要 engine
:
idf_component_register(SRCS "car.c"
INCLUDE_DIRS "."
REQUIRES engine)
-
SRCS
提供car
组件中源文件列表。 -
INCLUDE_DIRS
提供该组件公共头文件目录列表,由于car.h
是公共接口,所以这里列出了所有包含了car.h
的目录。 -
REQUIRES
给出该组件的公共接口所需的组件列表。由于car.h
是一个公共头文件并且包含了来自engine
的头文件,所以我们这里包含engine
。这样可以确保任何包含car.h
的其他组件也能递归地包含所需的engine.h
。
5.1.2 Engine 组件
engine
组件也有一个公共头文件 include/engine.h
,但这个头文件更为简单:
/* engine.h */
#define ENGINE_IS_HYBRID
void engine_start(void);
在 engine.c
中执行:
/* engine.c */
#include "engine.h"
#include "spark_plug.h"
...
在该组件中,engine
依赖于 spark_plug
,但这是私有依赖关系。编译 engine.c
需要 spark_plug.h
但不需要包含 engine.h
。
这代表文件 engine/CMakeLists.txt
可以使用 PRIV_REQUIRES
:
idf_component_register(SRCS "engine.c"
INCLUDE_DIRS "include"
PRIV_REQUIRES spark_plug)
因此,car
组件中的源文件不需要在编译器搜索路径中添加 spark_plug
include 目录。这可以加快编译速度,避免编译器命令行过于的冗长。
5.1.3 Spark Plug 组件
spark_plug
组件没有依赖项,它有一个公共头文件 spark_plug.h
,但不包含其他组件的头文件。
这代表 spark_plug/CMakeLists.txt
文件不需要任何 REQUIRES
或 PRIV_REQUIRES
:
idf_component_register(SRCS "spark_plug.c"
INCLUDE_DIRS ".")
5.2 源文件 Include 目录
每个组件的源文件都是用这些 Include 路径目录编译的,这些路径在传递给 idf_component_register
的参数中指定:
idf_component_register(..
INCLUDE_DIRS "include" # include文件夹下的所有.h文件
PRIV_INCLUDE_DIRS "other")
- 当前组件
INCLUDE_DIRS
指定include
文件夹下的所有.h文件 -
当前组件
PRIV_INCLUDE_DIRS
包含其他组件的.h文件 -
REQUIRES
和PRIV_REQUIRES
参数指定的所有其他组件(即当前组件的所有公共和私有依赖项)所设置的INCLUDE_DIRS
。 - 递归列出所有组件
REQUIRES
列表中INCLUDE_DIRS
目录(如递归展开这个组件的所有公共依赖项)。
5.3 main
组件依赖项
main
组件比较特别,因为它在构建过程中自动依赖所有其他组件。所以不需要向这个组件传递 REQUIRES
或 PRIV_REQUIRES
。
5.4 通用组件依赖项
为避免重复性工作,各组件都用自动依赖一些“通用” IDF 组件,即使它们没有被明确提及。这些组件的头文件会一直包含在构建系统中。
通用组件包括:cxx、newlib、freertos、esp_hw_support、heap、log、lwip、soc、hal、esp_rom、esp_common、esp_system。
注意:要避免循环依赖,关于其定义和解决方法参考:循环依赖。
参考:
- ESP-IDF编程指南-构建系统
- ESP32开发 2.添加.c.h并修改CMakeLists,来定制自己的工程
- ESPIDF添加自定义组件(创建C和H文件)
- ESP32学习笔记(21)——构建自己的工程和组件库
END