【ESP32学习】CMake学习
在之前的博客中提到,ESP-IDF采用的是CMake来构建项目,因此需要学习一下CMake,以对ESP32的开发有更好的把握
参考:
Windows下CMake安装教程
从零开始详细介绍CMake
CMake是一种高级编译配置工具,当多人用不同的语言或编译器开发一个项目,最终要输出一个可执行文件或共享库(dll,so等),就可以使用CMake,CMake的所有操作都是通过编译CMakeLists.txt来完成的
CMake的学习其实最好还是在开发中不断的学习,不过掌握一点基础的知识也是可以的,在这里推荐一个github的项目:
cmake-examples
里面有一些案例,可以一步步学习CMake,而且这个项目是英文的,可以锻炼一下自己的英文水平(~ ̄▽ ̄)~
CMake编译helloworld
// helloworld.cpp
#include <iostream>
int int main() {
std::cout << "hello world" << std::endl;
}
# CMakeLists.txt
PROJECT(helloworld)
SET(SRC_LIST helloworld.cpp)
MESSAGE(STATUS "This is BINARY dir" ${helloworld_BINARY_DIR})
MESSAGE(STATUS "This is SOURCE dir" ${helloworld_SOURCE_DIR})
ADD_EXECUTABLE(hello ${SRC_LIST})
CMake可以搜索平台环境,然后生成平台上用于build的文件,比如在windows上安装了visual studio,能够生成.sln,.vcxproj文件,在Linux上能够生成Makefile,CMakeLists.txt本质上就是和平台无关的配置文件,里面存有要运行的CMake指令,接下来是使用CMake在Ubuntu上构建helloworld的过程:
关于上面的CMakeLists.txt的语法说明:
-
PROJECT关键字:用于指定工程的名字和支持的语言,默认支持所有语言
- PROJECT(helloworld):指定了工程名字,且支持所有语言
- PROJECT(helloworld cxx):指定了工程名字,且支持语言是c++
- PROJECT(helloworld c cxx):指定了工程名字,且支持语言是c和c++
- 该指令隐式定义了两个CMake的变量:
- <project_name>_BINARY_DIR,本例中是helloworld_BINARY_DIR
- <project_name>_SOURCE_DIR,本例中是helloworld_SOURCE_DIR
- MESSAGE关键字可以直接使用这两个变量,都指向当前的工作目录,在外部编译时有所区别
-
SET关键字:用于显式指定变量
- SET(SRC_LIST helloworld.cpp):SRC_LIST变量包含了helloworld.cpp
- 也可以SET(SRC_LIST helloworld.cpp a.cpp b.cpp)
-
MESSAGE关键字:向终端输出用户自定义信息,主要包含三种信息:
- SEND_ERROR:产生错误,生成过程被跳过
- STATUS:输出前缀为--的信息
- FATAL_ERROR:立即终止所有CMake过程
-
ADD_EXECUTABLE关键字:生成可执行文件
- ADD_EXECUTABLE(hello ${SRC_LIST}):生成可执行文件名是hello,源文件读取变量SRC_LIST中的内容
- 可以直接写ADD_EXECUTABLE(hello helloworld.cpp)
因此上述例子可简写为:
PROJECT(helloworld)
ADD_EXECUTABLE(hello helloworld.cpp)
上面的工程名和生成的可执行文件名是没有任何关系的
CMake语法的基本规则
- 变量使用${}方式取值,但是在IF控制语句中直接使用变量名
- 指令(参数1,参数2,...):参数用括弧括起,参数之间使用空格或分号分开
- 指令是大小无关的,参数和变量时大小写相关的,推荐使用大写指令
其他注意事项:
- SET(SRC_LIST helloworld.cpp)可以写为SET(SRC_LIST "helloworld.cpp"):如果源文件名中含有空格,则必须要加双引号
- ADD_EXECUTABLE(hello helloworld):后缀可以不写,会自动去找.c和.cpp,但是因为可能有两个文件因此最好不要这样写
CMake内部构建和外部构建
上述的helloworld的例子使用的是内部构建,生成的临时文件较多,不方便清理
外部构建会把生成的临时文件放在build目录下,不会对源文件产生影响
步骤:
- 在当前目录建立一个build目录,可以在任何地方,这一点便于在任何时候都可以从头开始使用CMake进行构建
- 进入build,运行
cmake..
(..表示上一级目录,可以写CMakeLists.txt所在的绝对路径,生产的文件都在build目录下,当然这个是建立在你创建的build目录是在当前目录下的) - 在build目录下,运行make来构建工程
在外部构建下的两个变量:
- helloworld_BINARY_DIR:仍为工程路径
- helloworld_SOURCE_DIR:编译路径,即用户名/cmake/build