| # 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} ${CMAKE_BUILD_TYPE}) |
| 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} ${CMAKE_BUILD_TYPE}) |
| 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) |
| |