# The macro choose_msvc_crt() takes a list of possible | |
# C runtimes to choose from, in the form of compiler flags, | |
# to present to the user. (MTd for /MTd, etc) | |
# | |
# The macro is invoked at the end of the file. | |
# | |
# CMake already sets CRT flags in the CMAKE_CXX_FLAGS_* and | |
# CMAKE_C_FLAGS_* variables by default. To let the user | |
# override that for each build type: | |
# 1. Detect which CRT is already selected, and reflect this in | |
# LLVM_USE_CRT_* so the user can have a better idea of what | |
# changes they're making. | |
# 2. Replace the flags in both variables with the new flag via a regex. | |
# 3. set() the variables back into the cache so the changes | |
# are user-visible. | |
### Helper macros: ### | |
macro(make_crt_regex regex crts) | |
set(${regex} "") | |
foreach(crt ${${crts}}) | |
# Trying to match the beginning or end of the string with stuff | |
# like [ ^]+ didn't work, so use a bunch of parentheses instead. | |
set(${regex} "${${regex}}|(^| +)/${crt}($| +)") | |
endforeach(crt) | |
string(REGEX REPLACE "^\\|" "" ${regex} "${${regex}}") | |
endmacro(make_crt_regex) | |
macro(get_current_crt crt_current regex flagsvar) | |
# Find the selected-by-CMake CRT for each build type, if any. | |
# Strip off the leading slash and any whitespace. | |
string(REGEX MATCH "${${regex}}" ${crt_current} "${${flagsvar}}") | |
string(REPLACE "/" " " ${crt_current} "${${crt_current}}") | |
string(STRIP "${${crt_current}}" ${crt_current}) | |
endmacro(get_current_crt) | |
# Replaces or adds a flag to a variable. | |
# Expects 'flag' to be padded with spaces. | |
macro(set_flag_in_var flagsvar regex flag) | |
string(REGEX MATCH "${${regex}}" current_flag "${${flagsvar}}") | |
if("${current_flag}" STREQUAL "") | |
set(${flagsvar} "${${flagsvar}}${${flag}}") | |
else() | |
string(REGEX REPLACE "${${regex}}" "${${flag}}" ${flagsvar} "${${flagsvar}}") | |
endif() | |
string(STRIP "${${flagsvar}}" ${flagsvar}) | |
# Make sure this change gets reflected in the cache/gui. | |
# CMake requires the docstring parameter whenever set() touches the cache, | |
# so get the existing docstring and re-use that. | |
get_property(flagsvar_docs CACHE ${flagsvar} PROPERTY HELPSTRING) | |
set(${flagsvar} "${${flagsvar}}" CACHE STRING "${flagsvar_docs}" FORCE) | |
endmacro(set_flag_in_var) | |
macro(choose_msvc_crt MSVC_CRT) | |
if(LLVM_USE_CRT) | |
message(FATAL_ERROR | |
"LLVM_USE_CRT is deprecated. Use the CMAKE_BUILD_TYPE-specific | |
variables (LLVM_USE_CRT_DEBUG, etc) instead.") | |
endif() | |
make_crt_regex(MSVC_CRT_REGEX ${MSVC_CRT}) | |
foreach(build_type ${CMAKE_CONFIGURATION_TYPES}) | |
string(TOUPPER "${build_type}" build) | |
if (NOT LLVM_USE_CRT_${build}) | |
get_current_crt(LLVM_USE_CRT_${build} | |
MSVC_CRT_REGEX | |
CMAKE_CXX_FLAGS_${build}) | |
set(LLVM_USE_CRT_${build} | |
"${LLVM_USE_CRT_${build}}" | |
CACHE STRING "Specify VC++ CRT to use for ${build_type} configurations." | |
FORCE) | |
set_property(CACHE LLVM_USE_CRT_${build} | |
PROPERTY STRINGS "";${${MSVC_CRT}}) | |
endif(NOT LLVM_USE_CRT_${build}) | |
endforeach(build_type) | |
foreach(build_type ${CMAKE_CONFIGURATION_TYPES}) | |
string(TOUPPER "${build_type}" build) | |
if ("${LLVM_USE_CRT_${build}}" STREQUAL "") | |
set(flag_string " ") | |
else() | |
set(flag_string " /${LLVM_USE_CRT_${build}} ") | |
list(FIND ${MSVC_CRT} ${LLVM_USE_CRT_${build}} idx) | |
if (idx LESS 0) | |
message(FATAL_ERROR | |
"Invalid value for LLVM_USE_CRT_${build}: ${LLVM_USE_CRT_${build}}. Valid options are one of: ${${MSVC_CRT}}") | |
endif (idx LESS 0) | |
message(STATUS "Using ${build_type} VC++ CRT: ${LLVM_USE_CRT_${build}}") | |
endif() | |
foreach(lang C CXX) | |
set_flag_in_var(CMAKE_${lang}_FLAGS_${build} MSVC_CRT_REGEX flag_string) | |
endforeach(lang) | |
endforeach(build_type) | |
endmacro(choose_msvc_crt MSVC_CRT) | |
# List of valid CRTs for MSVC | |
set(MSVC_CRT | |
MD | |
MDd | |
MT | |
MTd) | |
choose_msvc_crt(MSVC_CRT) | |