代码改变世界

cmake 笔记

2024-06-20 18:55  清晨、午后  阅读(34)  评论(0编辑  收藏  举报

一、 一个完整的工程

  给工程起个名字

加上这句:project(hello)

命令:project(<PROJECT-NAME> [LANGUAGES] [<language-name>...])

作用:定义工程名称, 设置几个变量的名字: PROJECT_NAME, PROJECT_SOURCE_DIR, <PROJECT-NAME>_SOURCE_DIR, PROJECT_BINARY_DIR, <PROJECT-NAME>_BINARY_DIR, 高级用法请见参考链接2:CMake命令

  告诉CMake我的构建目标

add_executable(${PROJECT_NAME} ${hello_src})

  让CMake找到我的头文件

target_include_directories(${project_name} PRIVATE
    ${CMAKE_SOURCE_DIR}/include
)

  让CMake找到我的源文件

target_sources(${project_name} PRIVATE
    ${SRC}
    ${QRC}
)

  让CMake找到我的库文件

target_link_directories(${project_name} PRIVATE
    ${Qt_LIBSDIR}
)

  告诉CMake我要链接哪个库文件

set(Qt_LIBS
    Qt5::Core
    Qt5::Gui
    Qt5::Widgets
)

target_link_libraries(${project_name} PRIVATE
    ${Qt_LIBS}
)

  设置编译、连接参数

target_compile_features(${project_name} PRIVATE cxx_std_11)
target_compile_options(${project_name} PRIVATE -g -Wall)
target_link_options(${project_name} PRIVATE -Wl, --warn-unresolved-symbol)

  开始构建

通过以上步骤, 最后,在文件头部添加CMake版本检查,以我的环境为例,我的CMake版本是3.0,那么我在脚本最开始加上:

cmake_minimum_required ( VERSION 3.0)

完整的CMakeLists.txt如下所示:

cmake_minimum_required ( VERSION 3.0)

project(hello)

include_directories(${CMAKE_CURRENT_LIST_DIR}/include)

link_directories(${CMAKE_CURRENT_LIST_DIR}/lib)

aux_source_directory(${CMAKE_CURRENT_LIST_DIR}/src ${hello_src})

add_executable(${PROJECT_NAME} ${hello_src})

target_link_libraries(${PROJECT_NAME} util)

set(CMAKE_CXX_COMPILER      "clang++" )         # 显示指定使用的C++编译器

set(CMAKE_CXX_FLAGS   "-std=c++11")             # c++11
set(CMAKE_CXX_FLAGS   "-g")                     # 调试信息
set(CMAKE_CXX_FLAGS   "-Wall")                  # 开启所有警告

set(CMAKE_CXX_FLAGS_DEBUG   "-O0" )             # 调试包不优化
set(CMAKE_CXX_FLAGS_RELEASE "-O2 -DNDEBUG " )   # release包优化

  构建动态库、静态库

如果要构建动态库,将 add_executable(${PROJECT_NAME} ${hello_src}) 替换为 ADD_LIBRARY(${PROJECT_NAME} SHARED ${hello_src})  即可

如果要构建静态库,则为 ADD_LIBRARY(${PROJECT_NAME} STATIC ${hello_src}) 

可以将目标名称修改如下

  SET_TARGET_PROPERTIES (${PROJECT_NAME}_static PROPERTIES OUTPUT_NAME "hello")

也可以指定动态库版本

  SET_TARGET_PROPERTIES (${PROJECT_NAME}PROPERTIES VERSION 1.2 SOVERSION 1)

二、 其他

1. cmake help

CMake Reference Documentation — CMake 3.27.9 Documentation
CMAKE_SOURCE_DIR:根cmaek文件所在路径
CMAKE_CURRENT_SOURCE_DIR :当前cmake文件所在路径
PROJECT_SOURCE_DIR :当前cmake文件所在路径

现代CMake工具的设计理念和使用

  https://mp.weixin.qq.com/s/qIEsImvOWa3-nRx7Ik_R8g

全面的例子

  https://gitee.com/li736069609/qt_demo/blob/master/gui/CMakeLists.txt

2. cmake 中定义宏,代码中使用

# 新的语法
add_compile_definitions(OPENCV_VERSION="1.0.0") // 代码中可以直接使用 OPENCV_VERSION,值为 1.0.0
add_compile_definitions(WITH_OPENCV2) // 代码中可以直接使用 WITH_OPENCV2 进行条件编译
或者
add_compile_definitions(OPENCV_VERSION="1.0.0" WITH_OPENCV2)

# 旧的语法
add_definitions(-DSOURCE_DIR="aaa")  // 代码中的宏 SOURCE_DIR 就是字符串aaa
add_definitions(-DWITH_TEST)  // 代码中可以使用 WITH_TEST 进行条件编译

3. 静态库的连接顺序

工程在链接阶段静态库是有顺序要求的,如果懒得理这些顺序,就用  -Wl,--start-group  和 -Wl,--end-group 括起来即可

target_link_libraries(
    -Wl,--start-group
    libX3.a
    libX2.a
    libX1.a
    -Wl,--end-group
)

4. 大型工程中有用的几个设置

target_link_options(${project_name} PRIVATE -Wl, --no-undefined)
QMAKE_LFLAGS += -Wl,--no-undefined (qmake)

告诉链接器在链接过程中不允许有未定义的符号 (对动态库 so)

target_link_options(${project_name} PRIVATE -Wl, --warn-unresolved-symbol)

告诉链接器在链接过程中可以有未定义的符号 (对可执行程序)

target_link_options(${project_name} PRIVATE -Wl, --copy-dt-needed-entries)

当链接可执行文件时,依赖于libA.so,而libA.so又依赖于libB.so,而且可执行文件中还直接调用了libB.so中的函数,如果可执行文件没有显示声明连接libB.so,则会报错 DSO missing from command line,因为自从binutils 2.22版本以后,如果你在程序中使用了你依赖的动态库所依赖的动态库中的函数时,你就必须显式的指定你依赖的动态库所依赖的动态库,

5 Debug 与 Release 版本控制 

cmake .. -DCMAKE_BUILD_TYPE=Debug/Release

CMake 会根据构建类型自动设置一些编译选项
Debug 模式: CMake 会默认添加 -g(生成调试信息)并禁用优化(通常是 -O0)
Release 模式: CMake 会默认添加优化选项(例如 -O2 或 -O3)并禁用调试信息
也可以手动修改 CMake 配置来修改这些选项,例如:
set(CMAKE_CXX_FLAGS_DEBUG "-g -O0")
set(CMAKE_CXX_FLAGS_RELEASE "-O3 -DNDEBUG")

 

也可以根据需要做一个差异化的处理

if (CMAKE_BUILD_TYPE)
    if (${CMAKE_BUILD_TYPE} STREQUAL "Debug")
        ...
    elseif (${CMAKE_BUILD_TYPE} STREQUAL "Release")
        ...
    else ()
        ...
    endif ()
else ()
    ...
endif ()    

6. find_package 使用

find_package( <name> [version] [EXACT] [QUIET] [NO_MODULE] [ [ REQUIRED | COMPONENTS ] [ componets... ] ] )

用来调用预定义在 CMAKE_MODULE_PATH 下的 Find<name>.cmake 模块。

(1)查找引用使用第三方库,如使用 curl 库

在工程根目录中建立 src 文件夹,并在其下创建 main.c 文件,内容如下:

#include <curl/curl.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
FILE *fp;
int write_data(void *ptr, size_t size, size_t nmemb, void *stream)
{
int written = fwrite(ptr, size, nmemb, (FILE *)fp);
return written;
}
int main()
{
const char * path = “/tmp/curl-test”;
const char * mode = “w”;
fp = fopen(path,mode);
curl_global_init(CURL_GLOBAL_ALL);
CURLcode res;
CURL *curl = curl_easy_init();
curl_easy_setopt(curl, CURLOPT_URL, “http://www.linux-ren.org”);
curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, write_data);
curl_easy_setopt(curl, CURLOPT_VERBOSE, 1);
res = curl_easy_perform(curl);
curl_easy_cleanup(curl);
}

在工程根目录下建立主工程文件 CMakeLists.txt,内容如下

PROJECT(CURLTEST)
ADD_SUBDIRECTORY(src)

在 src 下创建 CMakeLists.txt

ADD_EXECUTABLE(curltest main.c)

 现在自然是没办法编译的,我们需要添加 curl 的头文件路径和库文件。

方法 1:
直接通过 INCLUDE_DIRECTORIES 和 TARGET_LINK_LIBRARIES 指令添加:
我们可以直接在 src/CMakeLists.txt 中添加:
INCLUDE_DIRECTORIES(/usr/include)
TARGET_LINK_LIBRARIES(curltest curl)
然后建立 build 目录进行外部构建即可。

方法 2
使用 FindCURL 模块。
向src/CMakeLists.txt 中添加:
FIND_PACKAGE(CURL)
IF(CURL_FOUND)
  INCLUDE_DIRECTORIES(${CURL_INCLUDE_DIR})
  TARGET_LINK_LIBRARIES(curltest ${CURL_LIBRARY})
ELSE(CURL_FOUND)
    MESSAGE(FATAL_ERROR ”CURL library not found”)
ENDIF(CURL_FOUND)
对于系统预定义的 Find<name>.cmake 模块,使用方法一般如上例所示:
每一个模块都会定义以下几个变量
    <name>_FOUND
    <name>_INCLUDE_DIR or <name>_INCLUDES
    <name>_LIBRARY or <name>_LIBRARIES
你可以通过<name>_FOUND 来判断模块是否被找到,如果没有找到,按照工程的需要关闭某些特性、给出提醒或者中止编译,上面的例子就是报出致命错误并终止构建。
如果<name>_FOUND 为真,则将<name>_INCLUDE_DIR 加入 INCLUDE_DIRECTORIES,将<name>_LIBRARY 加入 TARGET_LINK_LIBRARIES 中。

(2)自定义 cmake 模块,如编写属于自己的 FindHello 模块。

定义 cmake/FindHELLO.cmake 模块

FIND_PATH(HELLO_INCLUDE_DIR hello.h /usr/include/hello /usr/local/include/hello)
FIND_LIBRARY(HELLO_LIBRARY NAMES hello PATH /usr/lib /usr/local/lib)
IF (HELLO_INCLUDE_DIR AND HELLO_LIBRARY)
  SET(HELLO_FOUND TRUE)
ENDIF (HELLO_INCLUDE_DIR AND HELLO_LIBRARY)
IF (HELLO_FOUND) IF (NOT HELLO_FIND_QUIETLY) MESSAGE(STATUS "Found Hello: ${HELLO_LIBRARY}") ENDIF (NOT HELLO_FIND_QUIETLY) ELSE (HELLO_FOUND) IF (HELLO_FIND_REQUIRED) MESSAGE(FATAL_ERROR "Could not find hello library") ENDIF (HELLO_FIND_REQUIRED) ENDIF (HELLO_FOUND)

可以看到在该模块中定义了 HELLO_FOUND、HELLO_INCLUDE_DIR、HELLO_LIBRARY变量

前面的 curl 例子中我们使用了最简单的 FIND_PACKAGE 指令,其实他可以使用多种参数

QUIET 参数,对应与我们编写的 FindHELLO 中的 HELLO_FIND_QUIETLY ,如果不指定这个参数,就会执行:
MESSAGE(STATUS "Found Hello: ${HELLO_LIBRARY}")

REQUIRED 参数,对应于 FindHELLO.cmake 模块中的 HELLO_FIND_REQUIRED ,其含义是指这个共享库是否是工程必须的,如果使用了这个参数,说明这个链接库是必备库,如果找不到这个链接库,则工程不能编译。

使用自定义的 Hello 与使用 curl 的方法一致。

 设置动态库的版本号:

set_target_properties(${project_name} PROPERTIES VERSION 1.0.0 SOVERSION 1) 

 

------