CMake搭建编译环境总结

前言

  交叉编译算是每个嵌入式开发者都会经历的一道坎吧,通俗的描述就是搭建Arm板代码编译环境,让代码能够在Arm板子上跑起来。常用到的编译工具为Makefile和CMake,本篇记录下CMake的常用技巧。

入门案例:单个源文件

代码路径:
https://gitee.com/LinuxTaoist/DesignMode/tree/master/FactoryMode

工程结构

. 
├── CMakeLists.txt
├── abstract_factory.cc
├── factory_method.cc   
└── simple_factory.cc

CMakeList

# 指定最低版本
cmake_minimum_required(VERSION 2.8)

## 指定为C++11 版本
set(CMAKE_CXX_STANDARD 11)

## 指定项目名称
project(FactoryMode)

## 为当前路径以及子目录的源文件加入由-D预编译定义
## add_definitions(-DFOO -DDEBUG ...)

## 编译工具
set(CMAKE_C_COMPILER "gcc")
set(CMAKE_CXX_COMPILER "g++")

## 设置C++编译参数(CMAKE_CXX_FLAGS是全局变量)
set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -Wall -g3")
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -Wall -std=c++11 -g3 -fpermissive")

## 生成bin文件 AbFactory
add_executable(AbFactory  abstract_factory.cc)

## 生成bin文件 FacMethod
add_executable(FacMethod  factory_method.cc)

## 生成bin文件 SmpFactory
add_executable(SmpFactory simple_factory.cc)

工程编译

  CMakeList编写完以后,先执行cmake [CMakeList路径],然后make即可。

$ cmake .
-- The C compiler identification is GNU 7.5.0
-- The CXX compiler identification is GNU 7.5.0
-- Check for working C compiler: /usr/bin/cc
-- Check for working C compiler: /usr/bin/cc -- works
-- Detecting C compiler ABI info
-- Detecting C compiler ABI info - done
-- Detecting C compile features
-- Detecting C compile features - done
-- Check for working CXX compiler: /usr/bin/c++
-- Check for working CXX compiler: /usr/bin/c++ -- works
-- Detecting CXX compiler ABI info
-- Detecting CXX compiler ABI info - done
-- Detecting CXX compile features
-- Detecting CXX compile features - done
-- Configuring done
-- Generating done
-- Build files have been written to: /work/src/DesignMode/FactoryMode

$ make
Scanning dependencies of target SmpFactory
[ 16%] Building CXX object CMakeFiles/SmpFactory.dir/simple_factory.cc.o
[ 33%] Linking CXX executable SmpFactory
[ 33%] Built target SmpFactory
Scanning dependencies of target AbFactory
[ 50%] Building CXX object CMakeFiles/AbFactory.dir/abstract_factory.cc.o
[ 66%] Linking CXX executable AbFactory
[ 66%] Built target AbFactory
Scanning dependencies of target FacMethod
[ 83%] Building CXX object CMakeFiles/FacMethod.dir/factory_method.cc.o
[100%] Linking CXX executable FacMethod
[100%] Built target FacMethod

多个源文件

代码路径:
https://gitee.com/LinuxTaoist/DesignMode/tree/master/Proxy

工程结构

  对于工程中存在大量的文件夹和文件时,一个CMakeLst虽然可以将其全部编译,但是维护起来非常麻烦。

  对于庞大的代码架构场景,通常会按模块划分,将一个模块的代码放到一个CMakeList中配置编译,若模块代码还是很多,将此模块再细分成多个小模块用多个CMakeList管理编译。然后将这些CMakeList按照路径层层嵌套。

  如此工程中各个CMakeList树状层层嵌套,最终都会被嵌套至最顶层CMakeList。类似如下结构:

Proxy/
├── CMakeLists.txt
├── Client
│   ├── CMakeLists.txt
│   └── main_client.cc
├── Ipc
│   ├── CMakeLists.txt
│   ├── msg_manager.cc
│   └── msg_manager.h
├── Server
│   ├── Api
│   │   ├── common_type.h
│   │   ├── led_manager_proxy.cc
│   │   └── led_manager_proxy.h
│   ├── CMakeLists.txt
│   ├── Led
│   │   ├── led_manager.cc
│   │   └── led_manager.h
│   └── main_server.cc
└── build
    └── build.sh

  其中Proxy下的CMakeList会包含Client、Ipc、Server中的CMakeList。假设Server子路径还有子文件夹,Server的CMakeList就继续向下包含。

这么做的优点如下:

  • 每个CMakeList职责清晰。只需专注于当前模块或者当前路径的源码编译。
  • 方便模块化编译管理。当不需要编译哪个模块时,只需在顶层CMakeList屏蔽包含指定路径CMakeList即可。
  • 便于维护。每个CMakeList的代码量都比较少,且功能明确,维护者一眼就能看懂。

CMakeList

  • 顶层CMakeList
# 指定最低版本
cmake_minimum_required(VERSION 2.8)

## 指定为C++11 版本
set(CMAKE_CXX_STANDARD 11)

## 指定项目名称
project(ProxyMode)

## 为当前路径以及子目录的源文件加入由-D预编译定义
## add_definitions(-DFOO -DDEBUG ...)

## 编译工具
set(CMAKE_C_COMPILER "gcc")
set(CMAKE_CXX_COMPILER "g++")

## 设置C++编译参数(CMAKE_CXX_FLAGS是全局变量)
set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -Wall -g3")
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -Wall -std=c++11 -g3 -fpermissive")

## 包含子路径
add_subdirectory(Server)
add_subdirectory(Client)
add_subdirectory(Ipc)

顶层CMakeList一般需要做如下事项:
① 配置工程相关的属性:使用CMake版本、工程名
② 配置交叉工具:设置编译器、增加编译参数
③ 包含需要嵌套的子路径CMakeList

  • Server路径 CMakeList
## 指定最低版本
## 指定最低版本
cmake_minimum_required(VERSION 2.8)

## 宏
set(PROJECT_PATH ${CMAKE_CURRENT_SOURCE_DIR}/..)
cmake_policy(SET CMP0046 NEW)

## 输出路径
set(OUTPUT_PATH            ${PROJECT_PATH}/Out)
set(LIBRARY_OUTPUT_PATH    ${OUTPUT_PATH}/lib)
set(EXECUTABLE_OUTPUT_PATH ${OUTPUT_PATH}/bin)

## includes
include_directories(${PROJECT_PATH}/Ipc)
include_directories(${PROJECT_PATH}/Server/Api)

## 库路径
link_directories(${OUTPUT_PATH}/lib)

## main_server.exe
set(SRC_BIN_CLIENT   main_client.cc)
add_executable       (client ${SRC_BIN_CLIENT})
set_target_properties(client PROPERTIES OUTPUT_NAME "mainclient")
target_link_libraries(client c pthread ledapi)
add_dependencies     (client libledapi)

子路径下的CMakeList需要关心编译文件:
① 包含头文件路径
② 设置目标生成路径
③ 设置编译目标,bin或so

然后就是根据预期编译的结果,使用相关的变量即可。例子中,为了方便执行,增加了build.sh编译脚本。这个脚本代替执行编译命令,同时将编译生成的缓存文件放到指定路径管理。

## buid.sh
rm -rf ../Out/Cache
rm -rf ../Out/bin/*
rm -rf ../Out/lib/*
mkdir -p ../Out/cache/
cd ../Out/cache/
cmake ../../
make

其他用法

设置局部变量

## 设置局部变量PROJECT_PATH
set(PROJECT_PATH ${CMAKE_CURRENT_SOURCE_DIR}/..)

## 使用局部变量PROJECT_PATH
include_directories(${PROJECT_PATH}/Ipc)

设置自定义全局变量

## Proxy/CMakelists.txt
## 设置自定义全局变量 PROJECT_DESC
set(PROJECT_DESC "This is project")
set_property(GLOBAL PROPERTY source_list_property "${PROJECT_DESC}")

获取自定义全局变量

## Proxy/Ipc/CMakeLists.txt
## 获取自定义全局变量 PROJECT_DESC
get_property(PROJECT_DESC GLOBAL PROPERTY source_list_property)
message("PROJECT_DESC=${PROJECT_DESC}")

指定目标(bin/库)输出路径

## 设置库输出路径
set(LIBRARY_OUTPUT_PATH     xx/Out/lib)

## 设置bin文件输出路径
set(EXECUTABLE_OUTPUT_PATH  xx/Out/bin)

设置环境变量

set(ENV{<variable>} [<value>])   
ENV:环境变量标志性前缀
variable:变量名称
value:变量值

E.g 设置环境 CMAKE_FILE

## 设置环境变量
set(ENV{CMAKE_FILE} "./IPC")

获取环境变量

# 判断CMAKE_FILE环境变量是否定义
if(DEFINED ENV{CMAKE_FILE})
    message("CMAKE_FILE: $ENV{CMAKE_FILE}")
else()
    message("NOT DEFINED CMAKE_FILE VARIABLES")
endif()

设置编译器

## 指定C编译工具
set(CMAKE_C_COMPILER "gcc")

## 指定C++编译工具
set(CMAKE_CXX_COMPILER "g++")

当编译工具链路径被加到环境变量中,可以直接写编译工具的名称。在配交叉编译工具时,此处应写对应交叉编译工具链的绝对路径。

设置依赖库路径

## 括号为依赖库的绝对路径
link_directories(${OUTPUT_PATH}/lib)

包含头文件路径

## 括号为包含头文件的绝对路径
include_directories (${PROJECT_PATH}/Ipc)

添加编译器编译选项

## 针对所有编译器,开启编译警告 (包括C、C++编译器)
add_compile_options("-Wall -Werror")

## 针对C编译器,开启编译警告
set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -Werror")

## 针对C++编译,开启编译警告
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -Werror")

添加打印

## 打印CMAKE_CXX_FLAGS的值
message("${CMAKE_CXX_FLAGS}")

CMakeLists路径嵌套

## 添加当前路径Client文件
add_subdirectory(Client)

控制编译流程

option 语法

 ## option编译流程控制
 option(<variable> "<help_text>" [value])
 variable 选项名
 help_text 描述、解释、备注
 value 选项初始化值(除ON而外全为OFF)

E.g TEST_OPTION宏控制编译流程

option(TEST_OPTION "test opiton" ON)
if (DEFINED TEST_OPTION)
        message(STATUS "TEST_OPTION defined: " ${TEST_OPTION})
else ()     
        message(STATUS "TEST_OPTION un-defined: " ${TEST_OPTION})
endif()
 
if (TEST_OPTION)
        message(STATUS "TEST_OPTION ON.")
        add_definitions(-DTEST_OPTION)
else ()
        message(STATUS "TEST_OPTION OFF.")
endif()
 
if (NOT TEST_OPTION)
        message(STATUS "NOT-TEST_OPTION ON.")
else ()
        message(STATUS "NOT-TEST_OPTION OFF.")
endif()  

Shell脚本传递宏至CMakeList

命令行执行cmake时,跟随-DXXX,即可从命令行传递宏XXX至CMakeList。将此命令行写入脚本,便能实现Shell脚本传递宏至CMakeList。

## 增加TEST宏
cmake . -DTEST

## 增加TEST_OPTION=ON
cmake . -DTEST_OPTION=ON

CMakeLists传递变量至代码工程

## 向代码工程添加TEST宏
add_definitions(-DTEST)

代码判断宏TEST是否有定义,实现宏控

// *.c / *.cpp
#ifdef TEST
...  // code
#endif

#if defined TEST
...  // code
#endif

编译警告

CMake编译警告和报错设置

gcc本身设置了一些编译告警/报错选项,归类如下

  • -Werror:-Werror=xxx,表示将xxx的warning变为error,例如-Werror=select, -Werror=return-type

  • -Wall:激活所有的warnings

  • -Wextra:激活不在-Wall所在的warning的其它warnings

  • -Wpedantic: 对于所有不符合 ISO C/ISO C++ 语言标准的源代码发出警告,等价于-pedantic
    -pedantic-errors参数将这些警告视为错误,等同于-Werror=pedantic

  • -Wconversion: 在隐式转换可能导致值变化的时候发出警告。在隐式转换的时候,如果值发生变化,那么结果可能就不是预料中的,所以最好使用显式转换。

  • -Wshadow:激活遮蔽(如两个嵌套的for循环都用变量i做index)类型的warning,即:
    -Wshadow=global:激活任意类型的遮蔽;
    -Wshadow=local:激活local变量的遮蔽(如两个嵌套的for循环都用变量i做index);
    -Wshadow=compatible-local:激活local变量的遮蔽,考虑变量类型(如上例中的i在内外两层的for循环中是不同的类型);

E.g 打开所有编译告警,并视警告为错误,出现任何警告放弃编译

set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -Wall -Wextra -Werror")

常用警告

GCC编译器支持对代码进行诊断,针对代码本身不是错误但是疑似错误或者可能存在风险的地方发出警告,而警告编译选项就是用于控制需要告警的警告类型的。常见告警如下

  • -Wall
    这是一个非常常用的编译选项,用于启用一批比较常见且易于修改的警告,这些选项都是对代码进行基本的检查,比如下面这些:
选项 作用
-Waddress 检查是否存在可疑的内存地址使用
-Wformat 检查标准库函数的使用格式是否正确,比如printf的格式化字符串中的格式符和对应的参数是否匹配
-Wunused-function 对已声明但是未定义的静态函数和未被使用的非内联静态函数发出警告
-Wswitch 当用switch用于枚举类型时,判断分支是否包含所有枚举值,否则发出警告
-Wunused-variable 对声明但未被使用的变量发出警告
-Wunused-but-set-variable 对声明且被赋值但未被使用的变量发出警告
-Warray-bounds=1 数组越界检查,需启用选项-ftree-vrp

完整列表参考 Warning-Options

:当需要排除某些类型的警告,使用-Wno-xxx。 比如使用-Wall -Wno-unused-variable可以从-Wall中排除-Wunused-variable

  • -Wextra
    单单只有-Wall可能还不够严格,GCC还有-Wextra作为补充,包括另外一些没有被-Wall包含的警告类型,譬如:
选项 作用
-Wcast-function-type 当函数被强转为不兼容的函数指针时发出警告
-Wempty-body 当存在空的if、else或者do while语句时发出警告
-Wunused-parameter 当函数有未被使用的参数时发出警告,需配合-Wall
-Wunused-but-set-parameter 当存在被设置但是未被使用的参数发出警告,需配合-Wall
-Wsign-compare 当比较有符号和无符号值时发出警告

配置交叉编译环境常需要的修改

设置默认库和头文件搜索路径

编译默认会从/usr/include目录中搜索头文件、从/usr/lib中搜索依赖库。当设置了CMAKE_SYSROOT后,则会从xxx/usr/include搜索头文件、从xxx/usr/lib中搜索依赖库。

## 系统库路径:${SDKTARGETSYSROOT}/usr/lib
## 系统头文件:${SDKTARGETSYSROOT}/usr/include
set(CMAKE_SYSROOT "${SDKTARGETSYSROOT}")

设置交叉编译工具链

Linux系统在嵌入式板子上运行,需要与嵌入式板配套的交叉编译工具链编译。

同样的,个人代码也需要与编译Linux配套的交叉工具编译,才能在Linux环境运行。一般在Ubuntu上编译运行,只需要设置为gcc/g++即可。

编译工具链都是由厂商提供,用户只需要在编译脚本配置即可。设置交叉编译工具链方式如下:

## 绝对路径
set(CMAKE_C_COMPILER "xxx/arm-linux-gcc")
set(CMAKE_CXX_COMPILER "xxx/arm-linux-g++")

设置浮点运算处理方式

在某些gcc编译器会检查软浮点和硬浮点设置,报错log如下:

armv7at2hf-neon-poky-linux-gnueabi/usr/include/gnu/stubs-32.h:7:11: fatal error: gnu/stubs-soft.h: No such file or directory
    7 | # include <gnu/stubs-soft.h>

初步看报错log,是因为编译器没有文件stubs-soft.h。猜测可能此编译器不支持软浮点运算?

解决方法是在编译脚本将其设置为硬浮点运算,如下方式:

## 第一种
add_definitions("-mfloat-abi=hard -mfpu=neon")

## 第二种
add_compile_options(-mfpu=neon -mfloat-abi=hard)

## 第三种
set(CMAKE_C_FLAGS   "${CMAKE_C_FLAGS} -mfloat-abi=hard -mfpu=neon")
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -mfloat-abi=hard -mfpu=neon")

常见场景

编译动态库

## 生成libtest.so
### 添加源码路径
aux_source_directory (xxx/src            SRC_LIB_TEST)
### 生成so库
add_library          (libtest SHARED     ${SRC_LIB_TEST})
### 指定生成目标名libtest.so
set_target_properties(libtest PROPERTIES OUTPUT_NAME "test")
### 链接语言C++
set_target_properties(libtest PROPERTIES LINKER_LANGUAGE CXX)
### 链接依赖库
target_link_libraries(libtest stdc++)
### 添加依赖库,会先检查依赖库是否生成
add_dependencies     (libtest libstdc++)

编译静态库

## 生成libtest.a
aux_source_directory (xxx/src            SRC_LIB_TEST)
add_library          (libtest STATIC     ${SRC_LIB_TEST})
set_target_properties(libtest PROPERTIES OUTPUT_NAME "test")
set_target_properties(libtest PROPERTIES LINKER_LANGUAGE CXX)
target_link_libraries(libtest stdc++)
add_dependencies     (libtest libstdc++)

编译可执行文件

## 生成test bin
include_directories  (${PROJECT_SOURCE_DIR}/inc)
set(SRC_BIN_TEST  ${PROJECT_SOURCE_DIR}/src/test.cpp)
add_executable       (test ${SRC_BIN_TEST})
set_target_properties(test PROPERTIES OUTPUT_NAME "test")
set_target_properties(test PROPERTIES LINK_FLAGS "-Wl,-rpath-link=${LIBRARY_OUTPUT_PATH}")
target_link_libraries(test stdc++)
add_dependencies     (test libstdc++)
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,014评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,796评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,484评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,830评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,946评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,114评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,182评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,927评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,369评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,678评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,832评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,533评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,166评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,885评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,128评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,659评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,738评论 2 351

推荐阅读更多精彩内容