cmake这个跨平台的make工具功能已经很强大了,但它也有不足的地方,就是本文的标题。
在用cmake生成NMake的Makefile或visual studio的.sln时,如何指定运行库(/MD /MT)?
原本,我以为可以在CMAKE的手册中找到一个command或PROPERTY或VARIABLE,简单的调用或设置一下就可以了,很遗憾,把整个CMAKE manual翻了几遍也没找到,没捷径,只能写代码实现。
其实在翻手册之前我已经有了答案,就是照抄开源项目libjpeg_turbo的CMakeLists.txt中的代码,我只是不甘心,想找到更好的办法而已。
下面是libjpeg_turbo的CMakeLists.txt中的代码片段,这段代码对于MSVC工程强制所有target的所有build类型(DEBUG/RELEASE/MINSIZEREL/RELWITHDEBINFO)使用C静态库。
代码中文注释是博主加的。
#判断编译器类型
if(MSVC)
#添加一个名为WITH_CRT_DLL的开关选项,
#这样就可以在cmake-gui中或在命令行随时修改该开关选项。
option(WITH_CRT_DLL
"Link all libjpeg-turbo libraries and executables with the C run-time DLL (msvcr*.dll) instead of the static C run-time library (libcmt*.lib.) The default is to use the C run-time DLL only with the libraries and executables that need it."
FALSE)
if(NOT WITH_CRT_DLL)
# for循环修改所有CMAKE_<LANG>_FLAGS开关的编译选项变量,用正则表达式将/MD替换成/MT
# Use the static C library for all build types
foreach(var CMAKE_C_FLAGS CMAKE_C_FLAGS_DEBUG CMAKE_C_FLAGS_RELEASE
CMAKE_C_FLAGS_MINSIZEREL CMAKE_C_FLAGS_RELWITHDEBINFO)
if(${var} MATCHES "/MD")
#正则表达式替换
string(REGEX REPLACE "/MD" "/MT" ${var} "${${var}}")
endif()
endforeach()
endif()
add_definitions(-W3 -wd4996)
endif()
我将这段代码原样照抄,到openjpeg的CMakeLists.txt中就实现将我要的功能。
因为我的项目代码是C++写的,所以在自己的CMakeLists.txt要使用这段代码,还要加上对C++编译选项的修改
if(MSVC)
# Use the static C library for all build types
foreach(var
CMAKE_C_FLAGS CMAKE_C_FLAGS_DEBUG CMAKE_C_FLAGS_RELEASE
CMAKE_C_FLAGS_MINSIZEREL CMAKE_C_FLAGS_RELWITHDEBINFO
CMAKE_CXX_FLAGS CMAKE_CXX_FLAGS_DEBUG CMAKE_CXX_FLAGS_RELEASE
CMAKE_CXX_FLAGS_MINSIZEREL CMAKE_CXX_FLAGS_RELWITHDEBINFO
)
if(${var} MATCHES "/MD")
string(REGEX REPLACE "/MD" "/MT" ${var} "${${var}}")
endif()
endforeach()
endif(MSVC)
后来在网上找到两个贴子,提到的解决办法也都是差不多是这样的。看来真的是没有捷径了。
http://cmake.cmake.narkive.com/MhC0rVdG/selecting-runtime-library-on-visual-studio-projects
http://stackoverflow.askbro.ru/questions/16212682/why-does-this-cmake-project-not-set-the-appropriate-msvc-runtime