263

我正在使用arm-linux-androideabi-g++编译器。当我尝试编译一个简单的“Hello, World!” 程序编译得很好。当我通过在该代码中添加一个简单的异常处理来测试它时,它也可以工作(在添加之后-fexceptions.​​. 我猜它默认是禁用的)。

这是针对 Android 设备的,我只想使用 CMake,而不是ndk-build.

例如 -first.cpp

#include <iostream>

using namespace std;

int main()
{
   try
   {
   }
   catch (...)
   {
   }
   return 0;
}

./arm-linux-androideadi-g++ -o first-test first.cpp -fexceptions

它工作没有问题...

问题...我正在尝试使用 CMake 文件编译该文件。

我想将其添加-fexceptions为标志。我试过了

set (CMAKE_EXE_LINKER_FLAGS -fexceptions ) or set (CMAKE_EXE_LINKER_FLAGS "fexceptions" )

set ( CMAKE_C_FLAGS "fexceptions")

它仍然显示错误。

4

6 回答 6

283

注意:鉴于编写此答案后 CMake 的演变,这里的大多数建议现在都已过时/已弃用,并且有更好的选择


假设您要添加这些标志(最好在常量中声明它们):

SET(GCC_COVERAGE_COMPILE_FLAGS "-fprofile-arcs -ftest-coverage")
SET(GCC_COVERAGE_LINK_FLAGS    "-lgcov")

有几种方法可以添加它们:

  1. 最简单的一个(不干净,但简单方便,仅适用于编译标志、C 和 C++):

     add_definitions(${GCC_COVERAGE_COMPILE_FLAGS})
    
  2. 附加到相应的 CMake 变量:

     SET(CMAKE_CXX_FLAGS  "${CMAKE_CXX_FLAGS} ${GCC_COVERAGE_COMPILE_FLAGS}")
     SET(CMAKE_EXE_LINKER_FLAGS  "${CMAKE_EXE_LINKER_FLAGS} ${GCC_COVERAGE_LINK_FLAGS}")
    
  3. 使用目标属性,参见。doc CMake 编译标志目标属性,需要知道目标名称。

     get_target_property(TEMP ${THE_TARGET} COMPILE_FLAGS)
     if(TEMP STREQUAL "TEMP-NOTFOUND")
       SET(TEMP "") # Set to empty string
     else()
       SET(TEMP "${TEMP} ") # A space to cleanly separate from existing content
     endif()
     # Append our values
     SET(TEMP "${TEMP}${GCC_COVERAGE_COMPILE_FLAGS}" )
     set_target_properties(${THE_TARGET} PROPERTIES COMPILE_FLAGS ${TEMP} )
    

现在我使用方法2。

于 2012-08-03T14:02:03.757 回答
216

target_compile_options在较新版本的 CMake 中,您可以分别使用和为单个目标设置编译器和链接器标志target_link_libraries(是的,后者也设置链接器选项):

target_compile_options(first-test PRIVATE -fexceptions)

PUBLIC此方法的优点是您可以通过和控制选项传播到依赖此方法的其他目标PRIVATE

从 CMake 3.13 开始,您还可以使用target_link_options添加链接器选项,使意图更加清晰。

于 2015-02-27T20:07:24.550 回答
53

尝试设置变量CMAKE_CXX_FLAGS而不是CMAKE_C_FLAGS

set (CMAKE_CXX_FLAGS "-fexceptions")

该变量CMAKE_C_FLAGS仅影响 C 编译器,但您正在编译 C++ 代码。

添加标志CMAKE_EXE_LINKER_FLAGS是多余的。

于 2012-08-02T19:27:56.100 回答
5

指定工具链特定选项的首选方法是使用 CMake 的工具链工具。这确保了以下之间的清晰划分:

  • 关于如何将源文件组织成目标的说明——以CMakeLists.txt文件表示,完全与工具链无关;和
  • 应如何配置某些工具链的详细信息 - 分为 CMake 脚本文件,可由项目的未来用户扩展,可扩展。

理想情况下,您的CMakeLists.txt文件中不应有编译器/链接器标志——即使在if/endif块中也是如此。并且您的程序应该使用默认工具链(例如 GNU/Linux 上的 GCC 或 Windows 上的 MSVC)为本地平台构建,而不需要任何额外的标志。

添加工具链的步骤:

  1. 使用全局工具链设置创建一个文件,例如arm-linux-androideadi-gcc.cmake :

    set(CMAKE_CXX_COMPILER arm-linux-gnueabihf-g++)
    set(CMAKE_CXX_FLAGS_INIT "-fexceptions")
    

    (您可以在此处找到示例 Linux 交叉编译工具链文件。)

  2. 当您想使用此工具链生成构建系统时,请CMAKE_TOOLCHAIN_FILE在命令行中指定参数:

    mkdir android-arm-build && cd android-arm-build
    cmake -DCMAKE_TOOLCHAIN_FILE=$(pwd)/../arm-linux-androideadi-gcc.cmake ..
    

    (注意:您不能使用相对路径。)

  3. 正常构建:

    cmake --build .
    

工具链文件使交叉编译更容易,但它们还有其他用途:

  • 为您的单元测试提供强化诊断。

    set(CMAKE_CXX_FLAGS_INIT "-Werror -Wall -Wextra -Wpedantic")
    
  • 难以配置的开发工具。

    # toolchain file for use with gcov
    set(CMAKE_CXX_FLAGS_INIT "--coverage -fno-exceptions -g")
    
  • 加强安全检查。

    # toolchain file for use with gdb
    set(CMAKE_CXX_FLAGS_DEBUG_INIT "-fsanitize=address,undefined -fsanitize-undefined-trap-on-error")
    set(CMAKE_EXE_LINKER_FLAGS_INIT "-fsanitize=address,undefined -static-libasan")
    
于 2020-08-21T17:30:48.877 回答
4

您还可以使用以下属性将链接器标志添加到特定目标LINK_FLAGS

set_property(TARGET ${target} APPEND_STRING PROPERTY LINK_FLAGS " ${flag}")

如果您想将此更改传播到其他目标,您可以创建一个虚拟目标以链接到。

于 2018-04-11T14:29:07.673 回答
2

当我需要一个名为“NO_DEBUG”的预编译定义时,这对我有用:

set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=c++14 -DNO_DEBUG")

然后从代码

#ifdef NO_DEBUG
.....
于 2021-04-18T23:48:10.427 回答