学校老师留了个作业,让用剩下一半的寒假学学ESP32,做蓝牙透传+STA&AP模式下工作的http服务器,但是不准用Arduino
当场就傻了:ESP32我刚刚好就会一手Arduino;乐鑫那套ESPIDF太难啃,之前点了个灯就去快乐stm32了;micropython......刷完固件发现蓝牙支持跟【数据删除】一样,还不如用c写——一咬牙一跺脚,回头肝ESPIDF吧
总体思路:资源少,跟着官方走准没错,硬啃就完事了
这个系列笔记可以供只接触过单片机开发(STM32、51基础)和硬件相关知识但没有接触过网络相关知识的同学翻阅学习
项目文件夹构建
ESP-IDF项目由各种“组件”构成,你需要什么功能就要往里扔进去什么组件
如果你的代码里用了一堆WiFi的库函数,但是没把WiFi组件加入进去,你是没办法用WiFi功能的
项目保存在项目文件夹下,它的根目录如下所示:
├── CMakeLists.txt Cmake使用的文件
├── other_documents 其他文件
├── main 存储主程序
│ ├── CMakeLists.txt
│ ├── component.mk 组件的make file
│ └── main.c
└── Makefile 由传统的GNU make程序使用的Makefile
需要注意的是:ESP-IDF并不是项目文件夹的一部分,它更像是一个自助编译器,项目文件夹通过idf.py esptools等工具和${IDF_PATH}与ESP-IDF目录建立联系;同样,esp的开发工具链也独立于项目存在,通过${PATH}对项目进行作用
项目建立前,esp-idf会通过idf.py menuconfig配置出Makefile,这些配置保存在sdkconfig中。sdkconfig会被保存在项目文件夹的根目录
CMakeLists.txt通过idf_component_register将项目文件夹下面的组件进行注册,如下所示
idf_component_register(SRCS "foo.c" "bar.c"
INCLUDE_DIRS "include"
REQUIRES mbedtls)
SRCS给出了源文件清单,能支持的源文件后缀名为.c .cpp .cc .S
INCLUDE_DIRS给出了组件中文件的搜索路径
REQUIRES不是必须的,它声明了其他需要加入的组件
通过这个txt文档,esp-idf就能知道你往里扔了什么组件,然后在编译的时候就会把这些组件编译链接进去(可以理解成操作系统的静态链接)
当编译完成后,文件根目录下会多出build文件夹和sdkconfig文件,build文件夹用来存放编译过程中的文件和生成的文件,sdkconfig文件是在menuconfig的过程中产生的,如果曾经多次重新设置过menuconfig,还会发现多出了以.old结尾的config文件
另外组件也可以自制,详细内容参考官方教程;而idf.py 的底层是用Cmake、make工具实现的,所以也可以直接用这些工具进行编译(不过应该没人这么干)
CMake与component组件
【摘自官方文档】一个ESP-IDF项目可以看作是多个不同组件(component)的集合,组件是模块化且独立的代码,会被编译成静态库并链接到应用程序。ESP-IDF自带一些组件,也可以去找开源项目已有的组件来用
ESP-IDF的组件其实是对CMake的封装,如果使用纯CMake风格的构建方式也可行(说到底还是交叉编译的那套流程,只是乐鑫针对ESP32进行了优化),如下所示
cmake_minimum_required(VERSION 3.5)
project(my_custom_app C)
# 源文件 main.c 包含有 app_main() 函数的定义
add_executable(${CMAKE_PROJECT_NAME}.elf main.c)
# 提供 idf_import_components 及 idf_link_components 函数
include($ENV{IDF_PATH}/tools/cmake/idf_functions.cmake)
# 为 idf_import_components 做一些配置
# 使能创建构件(不是每个项目都必须)
set(IDF_BUILD_ARTIFACTS ON)
set(IDF_PROJECT_EXECUTABLE ${CMAKE_PROJECT_NAME}.elf)
set(IDF_BUILD_ARTIFACTS_DIR ${CMAKE_BINARY_DIR})
# idf_import_components 封装了 add_subdirectory(),为组件创建库目标,然后使用给定的变量接收“返回”的库目标。
# 在本例中,返回的库目标被保存在“component”变量中。
idf_import_components(components $ENV{IDF_PATH} esp-idf)
# idf_link_components 封装了 target_link_libraries(),将被 idf_import_components 处理过的组件链接到目标
idf_link_components(${CMAKE_PROJECT_NAME}.elf "${components}")
示例项目的目录树结构可能如下所示:
- myProject/ #主目录
- CMakeLists.txt #全局CMAke文档,用于配置项目CMake
- sdkconfig #项目配置文件,可用menuconfig生成
- components/ - component1/ - CMakeLists.txt #组件的CMake文档,用于配置组件CMake
- Kconfig #用于定义menuconfig时展示的组件配置选项
- src1.c
- component2/ - CMakeLists.txt
- Kconfig
- src1.c
- include/ - component2.h
- main/ #可以将main目录看作特殊的伪组件 - src1.c
- src2.c
- build/ #用于存放输出文件
main
目录是一个特殊的“伪组件”,包含项目本身的源代码。main
是默认名称,CMake 变量 COMPONENT_DIRS
默认包含此组件,但您可以修改此变量。或者,您也可以在顶层 CMakeLists.txt 中设置 EXTRA_COMPONENT_DIRS
变量以查找其他指定位置处的组件。如果项目中源文件较多,建议将其归于组件中,而不是全部放在 main
中。
全局CMake编写
全局CMake文档应该至少包含如下三个部分:
cmake_minimum_required(VERSION 3.5) #必须放在第一行,设置构建该项目所需CMake的最小版本号
include($ENV{IDF_PATH}/tools/cmake/project.cmake) #用于导入CMake的其余功能来完成配置项目、检索组件等任务
project(myProject) #指定项目名称并创建项目,改名成蕙作为最终输出的bin文件或elf文件的名字
每个 CMakeLists 文件只能定义一个项目
还可以包含以下可选部分
COMPONENT_DIRS #组件搜索目录,默认为${IDF_PATH}/components、${PROJECT_PATH}/components和EXTRA_COMPONENT_DIRS
COMPONENTS #要构建进项目中的组件名称列表,默认为COMPONENT_DIRS目录下检索到的所有组件
EXTRA_COMPONENT_DIRS #用于搜索组件的其它可选目录列表,可以是绝对路径也可以是相对路径
COMPONENT_REQUIRES_COMMON #每个组件都需要的通用组件列表,这些通用组件会自动添加到每个组件的COMPONENT_PRIV_REQUIRES列表和项目的COMPONENTS列表中
使用set命令来设置以上变量,如下所示
set(COMPONENTS "COMPONENTx")
注意:set命令需要放在include之前,cmake_minimum_required之后
特别地,可以重命名main组件,分为两种情况
- main组件处于正常位置
${PROJECT_PATH}/main
,则会被自动添加到构建系统中,其他组件自动成为main的依赖项,方便处理依赖关系 - main组件被重命名为xxx,需要在全局CMake设定中设置EXTRA_COMPONENT_DIRS=${PROJECT_PATH}/xxx,并在组件CMake目录中设置COMPONENT_REQUIRES或COMPONENT_PRIV_REQUIRES以指定依赖项
组件CMake编写
每个项目都包含一个或多个组件,这些组件可以是 ESP-IDF 的一部分,可以是项目自身组件目录的一部分,也可以从自定义组件目录添加
组件是COMPONENT_DIRS列表中包含CMakeLists.txt文件的任何目录
ESP-IDF会搜索COMPONENT_DIRS中的目录列表来查找项目的组件此列表中的目录可以是组件自身(即包含CMakeLists.txt文件的目录),也可以是子目录为组件的顶级目录;搜索顺序:【ESP-IDF内部组件】-【项目组件】-【EXTRA_COMPONENT_DIRS】中的组件,如果这些目录中的两个或者多个包含具有相同名字的组件,则使用搜索到的最后一个位置的组件,允许将组件复制到项目目录中再修改以覆盖ESP-IDF组件
最小的组件CMakeLists如下
set(COMPONENT_SRCS "foo.c" "k.c") #用空格分隔的源文件列表
set(COMPONENT_ADD_INCLUDEDIRS "include") #用空格分隔的目录列表,里面的路径会被添加到所有需要该组件的组件(包括 main 组件)全局 include 搜索路径中
register_component() #构建生成与组件同名的库,并最终被链接到应用程序中
有以下预设变量,不建议修改
COMPONENT_PATH #组件目录,是包含CMakeLists.txt文件的绝对路径,注意路径中不能包含空格
COMPONENT_NAME #组件名,等同于组件目录名
COMPONENT_TARGET #库目标名,由CMake在内部自动创建
有以下项目级别的变量,不建议修改,但可以在组件CMake文档中使用
PROJECT_NAME #项目名,在全局CMake文档中设置
PROJECT_PATH #项目目录(包含项目 CMakeLists 文件)的绝对路径,与CMAKE_SOURCE_DIR相同
COMPONENTS #此次构建中包含的所有组件的名称
CONFIG_* #项目配置中的每个值在cmake中都对应一个以CONFIG_开头的变量
IDF_VER #ESP-IDF的git版本号,由git describe命令生成
IDF_TARGET #项目的硬件目标名称,一般是ESP32
PROJECT_VER #项目版本号
COMPONENT_ADD_INCLUDEDIRS #相对于组件目录的相对路径,会被添加到所有需要该组件的其他组件的全局include搜索路径中
COMPONENT_REQUIRES #用空格分隔的组件列表,列出了当前组件依赖的其他组件
【摘自官网】如果一个组件仅需要额外组件的头文件来编译其源文件(而不是全局引入它们的头文件),则这些被依赖的组件需要在 COMPONENT_PRIV_REQUIRES
中指出
有以下可选的组件特定变量,用于控制某组件的行为
COMPONENT_SRCS #要编译进当前组件的源文件的路径,推荐使用此方法向构建系统中添加源文件
COMPONENT_PRIV_INCLUDEDIRS #相对于组件目录的相对路径,仅会被添加到该组件的include搜索路径中
COMPONENT_PRIV_REQUIRES #以空格分隔的组件列表,用于编译或链接当前组件的源文件
COMPONENT_SRCDIRS #相对于组件目录的源文件目录路径,用于搜索源文件,匹配成功的源文件会替代COMPONENT_SRCS中指定的源文件
COMPONENT_SRCEXCLUDE #需要从组件中 剔除 的源文件路径
COMPONENT_ADD_LDFRAGMENTS #组件使用的链接片段文件的路径,用于自动生成链接器脚本文件
组件配置文件Kconfig
每个组件都可以包含一个Kconfig文件,和CMakeLists.txt放在同一目录下
Kconfig文件中包含要添加到该组件配置菜单中的一些配置设置信息,运行menuconfig时,可以在Component Settings菜单栏下找到这些设置
有手就行的入门
#include <stdio.h>
#include "freertos/FreeRTOS.h"
#include "freertos/task.h"
#include "sdkconfig.h"
#include "esp_log.h"
//固定需要include的头文件
//用于freertos支持和输出调试信息
以下内容头文件包含部分会省略这些
一般程序的入口是app_main()函数
void app_main(void)
点灯
#include "driver/gpio.h"
#define BLINK_GPIO CONFIG_BLINK_GPIO
/*Kconfig.projbuild文件内容如下
menu "Example Configuration"
config BLINK_GPIO
int "Blink GPIO number"
range 0 34
default 5
help
GPIO number (IOxx) to blink on and off.
Some GPIOs are used for other purposes (flash connections, etc.) and cannot be used to blink.
GPIOs 35-39 are input-only so cannot be used as outputs.
endmenu
这个文件的内容是在c预编译器之前进行的替换,会把CONFIG_BLINK_GPIO变成default的值(5)
*/
void app_main(void)
{
gpio_pad_select_gpio(BLINK_GPIO);//选择的引脚
gpio_set_direction(BLINK_GPIO,GPIO_MODE_OUTPUT);//设置输入输出方向
while(1)
{
printf("Turning off the LED\n");//串口打印信息
gpio_set_level(BLINK_GPIO, 0);//GPIO寄存器清零
vTaskDelay(1000 / portTICK_PERIOD_MS);//vTaskDelay()用于任务中的延时,下面会提到这其实是将任务转入阻塞态
printf("Turning on the LED\n");
gpio_set_level(BLINK_GPIO, 1);//GPIO寄存器置位
vTaskDelay(1000 / portTICK_PERIOD_MS);
}
}
UART
官方给出的配置步骤为:
- 设置uart_config_t配置结构体
- 通过ESP_ERROR_CHECK(uart_param_config(uart_num, &uart_config));应用设置
- 设置引脚
- 安装驱动,设置buffer和事件处理函数等
- 配置FSM并运行UART
#include "string.h"
#include "esp_system.h"
#include "driver/uart.h"
#include "driver/gpio.h"
//include uart库和gpio库来实现相应功能
void UART_init(void)//uart初始化函数
{
const uart_config_t uart_config =
{
.baud_rate = 115200,
.data_bits = UART_DATA_8_BITS,
.parity = UART_PARITY_DISABLE,
.stop_bits = UART_STOP_BITS_1,
.flow_ctrl = UART_HW_FLOWCTRL_DISABLE,
.source_clk = UART_SCLK_APB,
};//配置uart设置
ESP_ERROR_CHECK(uart_param_config(UART_NUM_1, &uart_config));//应用设置
//设置uart引脚
ESP_ERROR_CHECK(uart_set_pin(UART_NUM_1, TXD_PIN, RXD_PIN, UART_PIN_NO_CHANGE, UART_PIN_NO_CHANGE));
//使用buffer的情况,使用freertos提供的设备驱动
ESP_ERROR_CHECK(uart_driver_install(UART_NUM_1, RX_BUF_SIZE * 2,0,0, NULL, 0));
}
int UART_send_data(const char* TAG, const char* data)//发送数据函数
{
const int length = uart_write_bytes(UART_NUM_1, data, strlen(data));
ESP_LOGI(TAG, "Wrote %d bytes", length);
return length;
}
int UART_read_data(const char* TAG, const char* buffer)//收取数据函数
{
int length = 0;
ESP_ERROR_CHECK(uart_get_buffered_data_len(UART_NUM_1, (size_t*)&length));
length = uart_read_bytes(UART_NUM_1,buffer,length,100);
ESP_LOGI(TAG, "Read %d bytes", length);
return length;
}
void app_main(void)
{
UART_init();//初始化
//分别配置发送和接收串口信息的任务
xTaskCreate(rx_task, "uart_rx_task", 1024*2, NULL, configMAX_PRIORITIES, NULL);
xTaskCreate(tx_task, "uart_tx_task", 1024*2, NULL, configMAX_PRIORITIES-1, NULL);
}
console控制台
ESP提供了一个console用于串口调试,可以实现类似shell的操作
在固件中加入console相关组件后烧录,在串口中打出help就可以查看相关帮助
void register_system(void)//系统相关指令
{
register_free();
register_heap();
register_version();
register_restart();
register_deep_sleep();
register_light_sleep();
#if WITH_TASKS_INFO
register_tasks();
#endif
}
组件中两个目录 :cmd_nvs用于指令的识别;cmd_system用于系统指令的实现(这部分功能需要与RTOS配合才行)
NVS FLASH
NVS即Non-volatile storage非易失性存储
它相当于把ESP32的关键数据以键值格式存储在FLASH里,NVS通过spi_flash_{read|write|erase}
三个API进行操作,NVS使用主flash的一部分。管理方式类似数据库的表,在NVS里面可以存储很多个不同的表,每个表下面有不同的键值,每个键值可以存储8位、16位、32位等等不同的数据类型,但不能是浮点数
- 使用接口函数nvs_flash_init();进行初始化,如果失败可以使用nvs_flash_erase();先擦除再初始化
- 应用程序可以使用nvs_open();选用NVS表中的分区或通过nvs_open_from_part()指定其名称后使用其他分区
注意:NVS分区被截断时,其内容应该被擦除
读写操作
nvs_get_i8(my_handle,//表的句柄
"nvs_i8",//键值
&nvs_i8);//对应变量的指针
//使用这个API来读取8位数据,同理还有i16、u32等版本的API可用
nvs_set_i8(my_handle,//表的句柄
"nvs_i8",//键值
nvs_i8);//对应的变量
//使用这个API来写入8位数据,同理还有i16、u32等版本的API可用
表操作
nvs_open("List",//表名
NVS_READWRITE,//读写模式,可选读写模式或只读模式
&my_handle);//表的句柄
//打开表
nvs_commit(my_handle);//提交表
nvs_close(my_handle);//关闭表
NVS初始化示例程序
官方给出的示例程序中一般以以下形式初始化NVS
//Initialize NVS
esp_err_t ret = nvs_flash_init();//初始化
if (ret == ESP_ERR_NVS_NO_FREE_PAGES || ret == ESP_ERR_NVS_NEW_VERSION_FOUND)//如果初始化未成功
{
ESP_ERROR_CHECK(nvs_flash_erase());//擦除NVS并查错
ret = nvs_flash_init();//再次初始化
}
ESP_ERROR_CHECK(ret);//查错
ESPIDF提供的常用库函数
ESP_LOG打印系统日志到串口
#include "esp_err.h"
//用于打印错误信息
- ESP_LOGE - 错误日志 (最高优先级)
- ESP_LOGW - 警告日志
- ESP_LOGI - 信息级别的日志
- ESP_LOGD - 用于调试的日志
- ESP_LOGV - 仅仅用于提示的日志{最低优先级)
这些日志可以在menuconfig设置中打开或关闭,也可以在代码中手动设置关闭
RTOS操作
- vTaskDelay将任务置为阻塞状态,期间CPU继续运行其它任务
持续时间由参数xTicksToDelay指定,单位是系统节拍时钟周期
void vTaskDelay(portTickTypexTicksToDelay)
常量portTickTypexTicksToDelay用来辅助计算真实时间,此值是系统节拍时钟中断的周期,单位是ms
在文件FreeRTOSConfig.h中,宏INCLUDE_vTaskDelay 必须设置成1,此函数才能有效
- xTaskCreate创建新的任务并添加到任务队列
注意:所有任务应当为死循环且永远不会返回,即嵌套在while(1)内
xTaskCreate(pdTASK_CODE pvTaskCode,//指向任务的入口函数
const portCHAR * const pcName,//任务名
unsigned portSHORT usStackDepth,//任务堆栈大小
void *pvParameters,//任务参数指针
unsigned portBASE_TYPE uxPriority,//任务优先级
xTaskHandle *pvCreatedTask)//任务句柄,用于引用创建的任务
注意,任务优先级0为最低,数字越大优先级越高
- FreeRTOS的神奇之处
一句话概论:RTOS就是彳亍,FreeRTOS可以实现任务之间的时间片轮转调度,两个任务可以你执行一会我执行一会,高优先级任务还能抢占低优先级任务,让它马上爪巴,高优先级任务先运行
- FreeRTOS的底层实现还没看明白,过一阵子再学,反正效果和RTThread差不多,先把作业肝完再说 =)
这种神奇的操作靠的就是上面两个API
需要注意的是:所有任务应当为死循环且永远不会返回(两次强调)
不过如果实在不想写死循环,可以在任务末尾加上
vTaskDelete();//用于删除执行结束的任务
不过只执行一次的任务大多是在初始化阶段完成的,用的时候尽量小心些
- 事件event
事件是一种实现任务间通信的机制,主要用于实现多任务间的同步,但事件通信只能是事件类型的通信,无数据传输。事件可以实现一对多和多对多的传输:一个任务可以等待多个事件的发生:可以是任意一个事件发生时唤醒任务进行事件处理;也可以是几个事件都发生后才唤醒任务进行事件处理
#include "esp_event.h"
//include 这个文件才能使用event
事件使用事件循环来管理,事件循环分别为默认事件循环和自定义事件循环
默认事件循环不需要传入事件循环句柄;但自定义循环需要
esp_event_loop_create(const esp_event_loop_args_t *event_loop_args,//事件循环参数
esp_event_loop_handle_t *event_loop)//事件循环句柄
//用于创建一个事件循环
esp_event_loop_delete(esp_event_loop_handle_t event_loop)//删除事件循环
事件需要注册到事件循环
/* 注册事件到事件循环 */
esp_event_handler_instance_register(esp_event_base_t event_base,//事件基本ID
int32_t event_id,//事件ID
esp_event_handler_t event_handler,//事件回调函数指针(句柄)
void *event_handler_arg,//事件回调函数参数
esp_event_handler_instance_t *instance)
//如果事件回调函数在事件删除之前还没有被注册,需要在这里注册来进行调用
esp_event_handler_instance_register_with(esp_event_loop_handle_t event_loop,//事件循环句柄
esp_event_base_t event_base,//事件基本ID
int32_t event_id,//事件ID
esp_event_handler_t event_handler,//事件回调函数指针(句柄)
void *event_handler_arg,//事件回调函数参数
esp_event_handler_instance_t *instance)
//如果事件回调函数在事件删除之前还没有被注册,需要在这里注册来进行调用
esp_event_handler_register(esp_event_base_t event_base,//事件基本ID
int32_t event_id,//事件ID
esp_event_handler_t event_handler,//事件句柄
void *event_handler_arg)//事件参数
esp_event_handler_register_with(esp_event_loop_handle_t event_loop,//事件循环句柄
esp_event_base_t event_base,//事件基本ID
int32_t event_id,//事件ID
esp_event_handler_t event_handler,//事件回调函数指针(句柄)
void *event_handler_arg)//事件回调函数的参数
/* 取消注册 */
esp_event_handler_unregister(esp_event_base_t event_base,
int32_t event_id,
esp_event_handler_t event_handler)
esp_event_handler_unregister_with(esp_event_loop_handle_t event_loop,
esp_event_base_t event_base,
int32_t event_id,
esp_event_handler_t event_handler)
默认事件循环default event loop是系统的基础事件循环,用于传递系统事件(如WiFi等),但是也可以注册用户事件,一般的蓝牙+WiFi用这一个循环就足够了
esp_event_loop_create_default(void)//创建默认事件循环
esp_event_loop_delete_default(void)//删除默认事件循环
esp_event_loop_run(esp_event_loop_handle_t event_loop,//事件循环句柄
TickType_t ticks_to_run)//运行时间
默认事件和自定义事件之间可以进行发送操作
esp_event_post(esp_event_base_t event_base, int32_t event_id, void *event_data, size_t event_data_size, TickType_t ticks_to_wait)
使用宏
ESP_EVENT_DECLARE_BASE()
ESP_EVENT_DEFINE_BASE()
来声明和定义事件,同时事件的ID应该用enum枚举变量来指出,如下所示
/* 头文件 */
// Declarations for event source 1: periodic timer
#define TIMER_EXPIRIES_COUNT// number of times the periodic timer expires before being stopped
#define TIMER_PERIOD 1000000 // period of the timer event source in microseconds
extern esp_timer_handle_t g_timer; // the periodic timer object
// Declare an event base
ESP_EVENT_DECLARE_BASE(TIMER_EVENTS); // declaration of the timer events family
enum {// declaration of the specific events under the timer event family
TIMER_EVENT_STARTED, // raised when the timer is first started
TIMER_EVENT_EXPIRY, // raised when a period of the timer has elapsed
TIMER_EVENT_STOPPED // raised when the timer has been stopped
};
// Declarations for event source 2: task
#define TASK_ITERATIONS_COUNT 5 // number of times the task iterates
#define TASK_ITERATIONS_UNREGISTER 3 // count at which the task event handler is unregistered
#define TASK_PERIOD 500 // period of the task loop in milliseconds
ESP_EVENT_DECLARE_BASE(TASK_EVENTS); // declaration of the task events family
enum {
TASK_ITERATION_EVENT, // raised during an iteration of the loop within the task
};
/* 头文件 */
/* 源文件 */
ESP_EVENT_DEFINE_BASE(TIMER_EVENTS);
/* 源文件 */
/* 枚举定义的事件名应该放在头文件,宏函数应该放在源文件 */
可使用API esp_event_loop_create_default()来创建事件
esp_event_loop_create_default()