if(DRACO_CMAKE_DRACO_FLAGS_CMAKE_) return() endif() # DRACO_CMAKE_DRACO_FLAGS_CMAKE_ set(DRACO_CMAKE_DRACO_FLAGS_CMAKE_ 1) include(CheckCXXCompilerFlag) include(CheckCXXSourceCompiles) # Adds compiler flags specified by FLAGS to the sources specified by SOURCES: # # draco_set_compiler_flags_for_sources(SOURCES FLAGS ) macro(draco_set_compiler_flags_for_sources) unset(compiler_SOURCES) unset(compiler_FLAGS) unset(optional_args) unset(single_value_args) set(multi_value_args SOURCES FLAGS) cmake_parse_arguments(compiler "${optional_args}" "${single_value_args}" "${multi_value_args}" ${ARGN}) if(NOT (compiler_SOURCES AND compiler_FLAGS)) draco_die("draco_set_compiler_flags_for_sources: SOURCES and " "FLAGS required.") endif() set_source_files_properties(${compiler_SOURCES} PROPERTIES COMPILE_FLAGS ${compiler_FLAGS}) if(DRACO_VERBOSE GREATER 1) foreach(source ${compiler_SOURCES}) foreach(flag ${compiler_FLAGS}) message("draco_set_compiler_flags_for_sources: source:${source} " "flag:${flag}") endforeach() endforeach() endif() endmacro() # Tests compiler flags stored in list(s) specified by FLAG_LIST_VAR_NAMES, adds # flags to $DRACO_CXX_FLAGS when tests pass. Terminates configuration if # FLAG_REQUIRED is specified and any flag check fails. # # ~~~ # draco_test_cxx_flag(> # [FLAG_REQUIRED]) # ~~~ macro(draco_test_cxx_flag) unset(cxx_test_FLAG_LIST_VAR_NAMES) unset(cxx_test_FLAG_REQUIRED) unset(single_value_args) set(optional_args FLAG_REQUIRED) set(multi_value_args FLAG_LIST_VAR_NAMES) cmake_parse_arguments(cxx_test "${optional_args}" "${single_value_args}" "${multi_value_args}" ${ARGN}) if(NOT cxx_test_FLAG_LIST_VAR_NAMES) draco_die("draco_test_cxx_flag: FLAG_LIST_VAR_NAMES required") endif() unset(cxx_flags) foreach(list_var ${cxx_test_FLAG_LIST_VAR_NAMES}) if(DRACO_VERBOSE) message("draco_test_cxx_flag: adding ${list_var} to cxx_flags") endif() list(APPEND cxx_flags ${${list_var}}) endforeach() if(DRACO_VERBOSE) message("CXX test: all flags: ${cxx_flags}") endif() unset(all_cxx_flags) list(APPEND all_cxx_flags ${DRACO_CXX_FLAGS} ${cxx_flags}) # Turn off output from check_cxx_source_compiles. Print status directly # instead since the logging messages from check_cxx_source_compiles can be # quite confusing. set(CMAKE_REQUIRED_QUIET TRUE) # Run the actual compile test. unset(draco_all_cxx_flags_pass CACHE) message("--- Running combined CXX flags test, flags: ${all_cxx_flags}") # check_cxx_compiler_flag() requires that the flags are a string. When flags # are passed as a list it will remove the list separators, and attempt to run # a compile command using list entries concatenated together as a single # argument. Avoid the problem by forcing the argument to be a string. draco_set_and_stringify(SOURCE_VARS all_cxx_flags DEST all_cxx_flags) check_cxx_compiler_flag("${all_cxx_flags}" draco_all_cxx_flags_pass) if(cxx_test_FLAG_REQUIRED AND NOT draco_all_cxx_flags_pass) draco_die("Flag test failed for required flag(s): " "${all_cxx_flags} and FLAG_REQUIRED specified.") endif() if(draco_all_cxx_flags_pass) # Test passed: update the global flag list used by the draco target creation # wrappers. set(DRACO_CXX_FLAGS ${cxx_flags}) list(REMOVE_DUPLICATES DRACO_CXX_FLAGS) if(DRACO_VERBOSE) message("DRACO_CXX_FLAGS=${DRACO_CXX_FLAGS}") endif() message("--- Passed combined CXX flags test") else() message("--- Failed combined CXX flags test, testing flags individually.") if(cxx_flags) message("--- Testing flags from $cxx_flags: " "${cxx_flags}") foreach(cxx_flag ${cxx_flags}) # Since 3.17.0 check_cxx_compiler_flag() sets a normal variable at # parent scope while check_cxx_source_compiles() continues to set an # internal cache variable, so we unset both to avoid the failure / # success state persisting between checks. This has been fixed in newer # CMake releases, but 3.17 is pretty common: we will need this to avoid # weird build breakages while the fix propagates. unset(cxx_flag_test_passed) unset(cxx_flag_test_passed CACHE) message("--- Testing flag: ${cxx_flag}") check_cxx_compiler_flag("${cxx_flag}" cxx_flag_test_passed) if(cxx_flag_test_passed) message("--- Passed test for ${cxx_flag}") else() list(REMOVE_ITEM cxx_flags ${cxx_flag}) message("--- Failed test for ${cxx_flag}, flag removed.") endif() endforeach() set(DRACO_CXX_FLAGS ${cxx_flags}) endif() endif() if(DRACO_CXX_FLAGS) list(REMOVE_DUPLICATES DRACO_CXX_FLAGS) endif() endmacro() # Tests executable linker flags stored in list specified by FLAG_LIST_VAR_NAME, # adds flags to $DRACO_EXE_LINKER_FLAGS when test passes. Terminates # configuration when flag check fails. draco_set_cxx_flags() must be called # before calling this macro because it assumes $DRACO_CXX_FLAGS contains only # valid CXX flags. # # draco_test_exe_linker_flag() macro(draco_test_exe_linker_flag) unset(link_FLAG_LIST_VAR_NAME) unset(optional_args) unset(multi_value_args) set(single_value_args FLAG_LIST_VAR_NAME) cmake_parse_arguments(link "${optional_args}" "${single_value_args}" "${multi_value_args}" ${ARGN}) if(NOT link_FLAG_LIST_VAR_NAME) draco_die("draco_test_link_flag: FLAG_LIST_VAR_NAME required") endif() draco_set_and_stringify(DEST linker_flags SOURCE_VARS ${link_FLAG_LIST_VAR_NAME}) if(DRACO_VERBOSE) message("EXE LINKER test: all flags: ${linker_flags}") endif() # Tests of $DRACO_CXX_FLAGS have already passed. Include them with the linker # test. draco_set_and_stringify(DEST CMAKE_REQUIRED_FLAGS SOURCE_VARS DRACO_CXX_FLAGS) # Cache the global exe linker flags. if(CMAKE_EXE_LINKER_FLAGS) set(cached_CMAKE_EXE_LINKER_FLAGS ${CMAKE_EXE_LINKER_FLAGS}) draco_set_and_stringify(DEST CMAKE_EXE_LINKER_FLAGS SOURCE ${linker_flags}) endif() draco_set_and_stringify(DEST CMAKE_EXE_LINKER_FLAGS SOURCE ${linker_flags} ${CMAKE_EXE_LINKER_FLAGS}) # Turn off output from check_cxx_source_compiles. Print status directly # instead since the logging messages from check_cxx_source_compiles can be # quite confusing. set(CMAKE_REQUIRED_QUIET TRUE) message("--- Running EXE LINKER test for flags: ${linker_flags}") unset(linker_flag_test_passed CACHE) set(draco_cxx_main "\nint main() { return 0; }") check_cxx_source_compiles("${draco_cxx_main}" linker_flag_test_passed) if(NOT linker_flag_test_passed) draco_die("EXE LINKER test failed.") endif() message("--- Passed EXE LINKER flag test.") # Restore cached global exe linker flags. if(cached_CMAKE_EXE_LINKER_FLAGS) set(CMAKE_EXE_LINKER_FLAGS ${cached_CMAKE_EXE_LINKER_FLAGS}) else() unset(CMAKE_EXE_LINKER_FLAGS) endif() list(APPEND DRACO_EXE_LINKER_FLAGS ${${link_FLAG_LIST_VAR_NAME}}) list(REMOVE_DUPLICATES DRACO_EXE_LINKER_FLAGS) endmacro() # Runs the draco compiler tests. This macro builds up the list of list var(s) # that is passed to draco_test_cxx_flag(). # # Note: draco_set_build_definitions() must be called before this macro. macro(draco_set_cxx_flags) unset(cxx_flag_lists) if(CMAKE_CXX_COMPILER_ID MATCHES "Clang|GNU") list(APPEND cxx_flag_lists draco_base_cxx_flags) endif() # Append clang flags after the base set to allow -Wno* overrides to take # effect. Some of the base flags may enable a large set of warnings, e.g., # -Wall. if(CMAKE_CXX_COMPILER_ID MATCHES "Clang") list(APPEND cxx_flag_lists draco_clang_cxx_flags) endif() if(MSVC) list(APPEND cxx_flag_lists draco_msvc_cxx_flags) endif() draco_set_and_stringify(DEST cxx_flags SOURCE_VARS ${cxx_flag_lists}) if(DRACO_VERBOSE) message("draco_set_cxx_flags: internal CXX flags: ${cxx_flags}") endif() if(DRACO_CXX_FLAGS) list(APPEND cxx_flag_lists DRACO_CXX_FLAGS) if(DRACO_VERBOSE) message("draco_set_cxx_flags: user CXX flags: ${DRACO_CXX_FLAGS}") endif() endif() draco_set_and_stringify(DEST cxx_flags SOURCE_VARS ${cxx_flag_lists}) if(cxx_flags) draco_test_cxx_flag(FLAG_LIST_VAR_NAMES ${cxx_flag_lists}) endif() endmacro()