我编写了一个
CMakeLists.txt
来使用 g++
或 clang++
构建项目。
为了捕获尽可能多的错误,我将
libc++
与-D_LIBCPP_DEBUG2=2
(对于clang++
)和libstdc++
与-D_GLIBCXX_DEBUG
(对于g++
和clang++
)一起使用。
set(CMAKE_CXX_FLAGS_DEBUG "-ggdb -fno-inline -DDEBUG=1 -march=x86-64 -mtune=generic")
#[[
if("${CMAKE_CXX_COMPILER_ID}" MATCHES "Clang")
set(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} -D_LIBCPP_DEBUG2=2")
elseif("${CMAKE_CXX_COMPILER_ID}" MATCHES "GNU")
set(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} -D_GLIBCXX_DEBUG")
endif()
]]
if(CMAKE_BUILD_TYPE STREQUAL "Debug")
elseif(CMAKE_BUILD_TYPE STREQUAL "Release")
set(CMAKE_CXX_FLAGS_RELEASE "-O3 -DNDEBUG=1 -march=native")
elseif(CMAKE_BUILD_TYPE STREQUAL "RelWithDebInfo")
set(CMAKE_CXX_FLAGS_RELWITHDEBINFO "-fno-omit-frame-pointer -DNDEBUG=1 -march=native")
if("${CMAKE_CXX_COMPILER_ID}" MATCHES "Clang")
set(CMAKE_CXX_FLAGS_RELWITHDEBINFO "${CMAKE_CXX_FLAGS_RELWITHDEBINFO} -O3 -gline-tables-only")
elseif("${CMAKE_CXX_COMPILER_ID}" MATCHES "GNU")
set(CMAKE_CXX_FLAGS_RELWITHDEBINFO "${CMAKE_CXX_FLAGS_RELWITHDEBINFO} -Og -ggdb")
endif()
elseif(CMAKE_BUILD_TYPE STREQUAL "MinSizeRel")
set(CMAKE_CXX_FLAGS_MINSIZEREL "-Os -DNDEBUG=1 -march=native")
else()
message(STATUS "Wrong build type selected, defaulted to Debug.")
set(CMAKE_BUILD_TYPE "Debug")
endif()
注释掉的代码是我应该知道当前编译器当前将使用哪个库的地方。
如何实现这一目标?我知道,
libstdc++
定义了__GLIBCXX__
并且libc++
定义了_LIBCPP_VERSION
,但是如何检测它们呢?
我认为您可以安全地传递每个库的两个定义。
但是如果您确实想有条件地执行此操作,我建议使用 CheckCXXSourceCompiles 模块和以下代码:
#include <iostream>
int a =
#ifdef __GLIBCXX__
1;
#else
fgsfds;
#endif
int main(int argc, char* argv[])
{
return 0;
}
如果该代码可以编译,那么您正在使用 libstdc++。
在源代码中您可以:
#define STR_HELPER(x) #x
#define STR(x) STR_HELPER(x)
#if defined(__cplusplus) && defined(_LIBCPP_VERSION)
// This indicates that libc++ is being used
# pragma message("libc++ version: " STR(_LIBCPP_VERSION))
#endif
#ifdef __GLIBCXX__
// This indicates that libstdc++ is being used
# pragma message("libstdc++ version: " STR(__GLIBCXX__))
#endif
构建时您应该看到:
main.cpp:38:10: warning: libstdc++ version: 20230801 [-W#pragma-messages]