【问题标题】:What's the idiomatic way of enabling UB sanitization in CMake?在 CMake 中启用 UB 清理的惯用方式是什么?
【发布时间】:2021-10-07 22:42:48
【问题描述】:

我想为我的存储库构建一个测试程序,启用未定义的行为清理(至少使用 GCC,也许还有 clang)。我知道如何手动执行此操作:

  • -fsanitize=undefined 添加到编译标志
  • -lubsan 添加到链接标志
  • 确保安装了适当版本的libubsan

现在,在 CMake 中,我希望以下内容可以工作:

find_package(ubsan)
if (CMAKE_CXX_COMPILER_ID STREQUAL "GNU"   OR
    CMAKE_CXX_COMPILER_ID STREQUAL "Clang" )
    target_compile_options(my_test PRIVATE "-fsanitize=undefined")
endif()

target_link_libraries(ubsan::ubsan)

...但没有这样的版本(从 CMake 3.21.0-rc2 开始)。那么,我该怎么做呢?从某个地方获取 FindUBSan.cmake?也许做点别的?

PS - 这个问题同样适用于 C

【问题讨论】:

  • but there is no such version ... 的版本? Make sure an appropriate version of libubsan is installed. 什么意思? libubsan.so 应该与编译器一起分发,至少在我的分发中它是 gcc-libs 包的一部分。

标签: gcc cmake clang idioms ubsan


【解决方案1】:

所有内容都需要在启用消毒剂的情况下进行编译(通常不可能混合和匹配),并且启用消毒剂是编译器特定的;因此,它是一个工具链选项。

在工具链文件中,包括以下内容:

set(CMAKE_C_FLAGS_INIT "-fsanitize=undefined")
set(CMAKE_CXX_FLAGS_INIT "-fsanitize=undefined")

您还可以在命令行中设置CMAKE_<LANG>_FLAGS 以包含-fsanitize=undefined


完整示例:

alex@alex-ubuntu:~/test$ tree
.
├── CMakeLists.txt
├── main.cpp
└── ubsan.cmake

0 directories, 3 files

alex@alex-ubuntu:~/test$ cat CMakeLists.txt 
cmake_minimum_required(VERSION 3.21)
project(test)

add_executable(main main.cpp)

alex@alex-ubuntu:~/test$ cat main.cpp 
int main () { return 0; }

alex@alex-ubuntu:~/test$ cat ubsan.cmake 
set(CMAKE_C_FLAGS_INIT "-fsanitize=undefined")
set(CMAKE_CXX_FLAGS_INIT "-fsanitize=undefined")

alex@alex-ubuntu:~/test$ cmake -G Ninja -S . -B build --toolchain ubsan.cmake 
-- The C compiler identification is GNU 9.3.0
-- The CXX compiler identification is GNU 9.3.0
-- Detecting C compiler ABI info
-- Detecting C compiler ABI info - done
-- Check for working C compiler: /usr/bin/cc - skipped
-- Detecting C compile features
-- Detecting C compile features - done
-- Detecting CXX compiler ABI info
-- Detecting CXX compiler ABI info - done
-- Check for working CXX compiler: /usr/bin/c++ - skipped
-- Detecting CXX compile features
-- Detecting CXX compile features - done
-- Configuring done
-- Generating done
-- Build files have been written to: /home/alex/test/build

alex@alex-ubuntu:~/test$ cmake --build build/ -- -v
[1/2] /usr/bin/c++   -fsanitize=undefined -MD -MT CMakeFiles/main.dir/main.cpp.o -MF CMakeFiles/main.dir/main.cpp.o.d -o CMakeFiles/main.dir/main.cpp.o -c /home/alex/test/main.cpp
[2/2] : && /usr/bin/c++ -fsanitize=undefined  CMakeFiles/main.dir/main.cpp.o -o main   && :

alex@alex-ubuntu:~/test$ ldd build/main 
    linux-vdso.so.1 (0x00007ffc9dd75000)
    libubsan.so.1 => /lib/x86_64-linux-gnu/libubsan.so.1 (0x00007f79a4b52000)
    libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007f79a4960000)
    libdl.so.2 => /lib/x86_64-linux-gnu/libdl.so.2 (0x00007f79a495a000)
    libpthread.so.0 => /lib/x86_64-linux-gnu/libpthread.so.0 (0x00007f79a4937000)
    libstdc++.so.6 => /lib/x86_64-linux-gnu/libstdc++.so.6 (0x00007f79a4755000)
    libgcc_s.so.1 => /lib/x86_64-linux-gnu/libgcc_s.so.1 (0x00007f79a4738000)
    /lib64/ld-linux-x86-64.so.2 (0x00007f79a550a000)
    libm.so.6 => /lib/x86_64-linux-gnu/libm.so.6 (0x00007f79a45e9000)

【讨论】:

  • 我相信您是在推测...因为上述方法行不通。即 GCC 通常还需要 libubsan 才能正常工作。从理论上讲,该库可以引入编译器标志。另外,其他编译器呢?
  • 这对我来说适用于 GCC 9.3.0。这些标志(-fsanitize=undefined 通过CMAKE_<LANG>_FLAGS)也会在喜欢时传递给编译器,并且基本int main () { return 0; } 的生成二进制文件确实链接到libubsan。
  • 所有让我感动的是来自 libubsan 的未定义符号。但是,我使用的是 GCC 10.2
  • @einpoklum - 我刚刚用 10.3(来自 Ubuntu 存储库)尝试了它,并且可执行文件仍然通过这个答案链接到 libubsan(刚刚添加了 set(CMAKE_CXX_COMPILER g++-10) 和 C equiv 到工具链)。可以提供 MRE 吗?
  • 我会尽快尝试(但可能不会在周末之前)。
【解决方案2】:

在 CMake 中启用 UB 清理的惯用方式是什么?

没有“惯用方式”——它是特定于编译器的。做:

  • 将 -fsanitize=undefined 添加到编译标志
target_compile_options(my_test PUBLIC -fsanitize=undefined)
target_link_options(my_test PUBLIC -fsanitize=undefined)

您可以检查CheckCXXCompilerFlag 是否支持该选项。

【讨论】:

  • 如果你采用这种方法,你还需要打电话给target_link_options...我试过了,它没有链接到libubsan,因为链接行缺少这个标志。
  • 这基本上就是我所说的我正在做的,没有你缺少的库依赖。
猜你喜欢
  • 2016-11-12
  • 1970-01-01
  • 1970-01-01
  • 2021-07-04
  • 2021-12-19
  • 1970-01-01
  • 2013-01-03
  • 1970-01-01
相关资源
最近更新 更多