function(_get_common_test_compile_options output_var c_test flags) _get_compile_options_from_flags(compile_flags ${flags}) # Remove -fno-math-errno if it was added. if(LIBC_ADD_FNO_MATH_ERRNO) list(REMOVE_ITEM compile_options "-fno-math-errno") endif() set(compile_options ${LIBC_COMPILE_OPTIONS_DEFAULT} ${LIBC_TEST_COMPILE_OPTIONS_DEFAULT} ${compile_flags}) if(LLVM_LIBC_COMPILER_IS_GCC_COMPATIBLE) list(APPEND compile_options "-fpie") if(LLVM_LIBC_FULL_BUILD) list(APPEND compile_options "-DLIBC_FULL_BUILD") # Only add -ffreestanding flag in full build mode. list(APPEND compile_options "-ffreestanding") list(APPEND compile_options "-fno-exceptions") list(APPEND compile_options "-fno-unwind-tables") list(APPEND compile_options "-fno-asynchronous-unwind-tables") if(NOT c_test) list(APPEND compile_options "-fno-rtti") endif() endif() if(LIBC_COMPILER_HAS_FIXED_POINT) list(APPEND compile_options "-ffixed-point") endif() # list(APPEND compile_options "-Wall") # list(APPEND compile_options "-Wextra") # -DLIBC_WNO_ERROR=ON if you can't build cleanly with -Werror. if(NOT LIBC_WNO_ERROR) # list(APPEND compile_options "-Werror") endif() # list(APPEND compile_options "-Wconversion") # list(APPEND compile_options "-Wno-sign-conversion") # list(APPEND compile_options "-Wimplicit-fallthrough") # list(APPEND compile_options "-Wwrite-strings") list(APPEND compile_options "-Wextra-semi") # Silence this warning because _Complex is a part of C99. if(CMAKE_CXX_COMPILER_ID STREQUAL "GNU") if(NOT c_test) list(APPEND compile_options "-fext-numeric-literals") endif() else() list(APPEND compile_options "-Wno-c99-extensions") list(APPEND compile_options "-Wno-gnu-imaginary-constant") endif() list(APPEND compile_options "-Wno-pedantic") # if(NOT CMAKE_COMPILER_IS_GNUCXX) # list(APPEND compile_options "-Wnewline-eof") # list(APPEND compile_options "-Wnonportable-system-include-path") # list(APPEND compile_options "-Wstrict-prototypes") # list(APPEND compile_options "-Wthread-safety") # list(APPEND compile_options "-Wglobal-constructors") # endif() endif() set(${output_var} ${compile_options} PARENT_SCOPE) endfunction() function(_get_hermetic_test_compile_options output_var) _get_common_test_compile_options(compile_options "" "") # The GPU build requires overriding the default CMake triple and architecture. if(LIBC_TARGET_ARCHITECTURE_IS_AMDGPU) list(APPEND compile_options -Wno-multi-gpu -nogpulib -mcpu=${LIBC_GPU_TARGET_ARCHITECTURE} -flto -mcode-object-version=${LIBC_GPU_CODE_OBJECT_VERSION}) elseif(LIBC_TARGET_ARCHITECTURE_IS_NVPTX) list(APPEND compile_options "SHELL:-mllvm -nvptx-emit-init-fini-kernel=false" -Wno-multi-gpu --cuda-path=${LIBC_CUDA_ROOT} -nogpulib -march=${LIBC_GPU_TARGET_ARCHITECTURE} -fno-use-cxa-atexit) endif() set(${output_var} ${compile_options} PARENT_SCOPE) endfunction() # This is a helper function and not a build rule. It is to be used by the # various test rules to generate the full list of object files # recursively produced by "add_entrypoint_object" and "add_object_library" # targets. # Usage: # get_object_files_for_test( # # [ ...]) # # The list of object files is collected in . # If skipped entrypoints were found, then is # set to a true value. # targetN is either an "add_entrypoint_target" target or an # "add_object_library" target. function(get_object_files_for_test result skipped_entrypoints_list) set(object_files "") set(skipped_list "") set(checked_list "") set(unchecked_list "${ARGN}") list(REMOVE_DUPLICATES unchecked_list) foreach(dep IN LISTS unchecked_list) if (NOT TARGET ${dep}) # Skip tests with undefined dependencies. # Compiler-RT targets are added only if they are enabled. However, such targets may not be defined # at the time of the libc build. We should skip checking such targets. if (NOT ${dep} MATCHES "^RTScudo.*|^RTGwp.*") list(APPEND skipped_list ${dep}) endif() continue() endif() get_target_property(aliased_target ${dep} "ALIASED_TARGET") if(aliased_target) # If the target is just an alias, switch to the real target. set(dep ${aliased_target}) endif() get_target_property(dep_type ${dep} "TARGET_TYPE") if(NOT dep_type) # Skip tests with no object dependencies. continue() endif() get_target_property(dep_checked ${dep} "CHECK_OBJ_FOR_TESTS") if(dep_checked) # Target full dependency has already been checked. Just use the results. get_target_property(dep_obj ${dep} "OBJECT_FILES_FOR_TESTS") get_target_property(dep_skip ${dep} "SKIPPED_LIST_FOR_TESTS") else() # Target full dependency hasn't been checked. Recursively check its DEPS. set(dep_obj "${dep}") set(dep_skip "") get_target_property(indirect_deps ${dep} "DEPS") get_object_files_for_test(dep_obj dep_skip ${indirect_deps}) if(${dep_type} STREQUAL ${OBJECT_LIBRARY_TARGET_TYPE}) get_target_property(dep_object_files ${dep} "OBJECT_FILES") if(dep_object_files) list(APPEND dep_obj ${dep_object_files}) endif() elseif(${dep_type} STREQUAL ${ENTRYPOINT_OBJ_TARGET_TYPE}) get_target_property(is_skipped ${dep} "SKIPPED") if(is_skipped) list(APPEND dep_skip ${dep}) list(REMOVE_ITEM dep_obj ${dep}) endif() get_target_property(object_file_raw ${dep} "OBJECT_FILE_RAW") if(object_file_raw) # TODO: Remove this once we stop suffixing the target with ".__internal__" if(fq_target_name STREQUAL "libc.test.include.issignaling_c_test.__unit__" OR fq_target_name STREQUAL "libc.test.include.iscanonical_c_test.__unit__") string(REPLACE ".__internal__" "" object_file_raw ${object_file_raw}) endif() list(APPEND dep_obj ${object_file_raw}) endif() elseif(${dep_type} STREQUAL ${ENTRYPOINT_OBJ_VENDOR_TARGET_TYPE}) # Skip tests for externally implemented entrypoints. list(APPEND dep_skip ${dep}) list(REMOVE_ITEM dep_obj ${dep}) endif() set_target_properties(${dep} PROPERTIES OBJECT_FILES_FOR_TESTS "${dep_obj}" SKIPPED_LIST_FOR_TESTS "${dep_skip}" CHECK_OBJ_FOR_TESTS "YES" ) endif() list(APPEND object_files ${dep_obj}) list(APPEND skipped_list ${dep_skip}) endforeach(dep) list(REMOVE_DUPLICATES object_files) set(${result} ${object_files} PARENT_SCOPE) list(REMOVE_DUPLICATES skipped_list) set(${skipped_entrypoints_list} ${skipped_list} PARENT_SCOPE) endfunction(get_object_files_for_test) # Rule to add a libc unittest. # Usage # add_libc_unittest( # # SUITE # SRCS # HDRS # DEPENDS # COMPILE_OPTIONS # LINK_LIBRARIES # ) function(create_libc_unittest fq_target_name) if(NOT LLVM_INCLUDE_TESTS) return() endif() cmake_parse_arguments( "LIBC_UNITTEST" "NO_RUN_POSTBUILD;C_TEST" # Optional arguments "SUITE;CXX_STANDARD" # Single value arguments "SRCS;HDRS;DEPENDS;COMPILE_OPTIONS;LINK_LIBRARIES;FLAGS" # Multi-value arguments ${ARGN} ) if(NOT LIBC_UNITTEST_SRCS) message(FATAL_ERROR "'add_libc_unittest' target requires a SRCS list of .cpp " "files.") endif() if(NOT LIBC_UNITTEST_DEPENDS) message(FATAL_ERROR "'add_libc_unittest' target requires a DEPENDS list of " "'add_entrypoint_object' targets.") endif() get_fq_deps_list(fq_deps_list ${LIBC_UNITTEST_DEPENDS}) if(NOT LIBC_UNITTEST_C_TEST) list(APPEND fq_deps_list libc.src.__support.StringUtil.error_to_string libc.test.UnitTest.ErrnoSetterMatcher) endif() list(REMOVE_DUPLICATES fq_deps_list) _get_common_test_compile_options(compile_options "${LIBC_UNITTEST_C_TEST}" "${LIBC_UNITTEST_FLAGS}") list(APPEND compile_options ${LIBC_UNITTEST_COMPILE_OPTIONS}) if(SHOW_INTERMEDIATE_OBJECTS) message(STATUS "Adding unit test ${fq_target_name}") if(${SHOW_INTERMEDIATE_OBJECTS} STREQUAL "DEPS") foreach(dep IN LISTS LIBC_UNITTEST_DEPENDS) message(STATUS " ${fq_target_name} depends on ${dep}") endforeach() endif() endif() get_object_files_for_test( link_object_files skipped_entrypoints_list ${fq_deps_list}) if(skipped_entrypoints_list) # If a test is OS/target machine independent, it has to be skipped if the # OS/target machine combination does not provide any dependent entrypoints. # If a test is OS/target machine specific, then such a test will live is a # OS/target machine specific directory and will be skipped at the directory # level if required. # # There can potentially be a setup like this: A unittest is setup for a # OS/target machine independent object library, which in turn depends on a # machine specific object library. Such a test would be testing internals of # the libc and it is assumed that they will be rare in practice. So, they # can be skipped in the corresponding CMake files using platform specific # logic. This pattern is followed in the startup tests for example. # # Another pattern that is present currently is to detect machine # capabilities and add entrypoints and tests accordingly. That approach is # much lower level approach and is independent of the kind of skipping that # is happening here at the entrypoint level. if(LIBC_CMAKE_VERBOSE_LOGGING) set(msg "Skipping unittest ${fq_target_name} as it has missing deps: " "${skipped_entrypoints_list}.") message(STATUS ${msg}) endif() return() endif() if(LIBC_UNITTEST_NO_RUN_POSTBUILD) set(fq_build_target_name ${fq_target_name}) else() set(fq_build_target_name ${fq_target_name}.__build__) endif() add_executable( ${fq_build_target_name} EXCLUDE_FROM_ALL ${LIBC_UNITTEST_SRCS} ${LIBC_UNITTEST_HDRS} ) target_include_directories(${fq_build_target_name} SYSTEM PRIVATE ${LIBC_INCLUDE_DIR}) target_include_directories(${fq_build_target_name} PRIVATE ${LIBC_SOURCE_DIR}) target_compile_options(${fq_build_target_name} PRIVATE ${compile_options}) target_link_options(${fq_build_target_name} PRIVATE ${compile_options}) if(NOT LIBC_UNITTEST_CXX_STANDARD) set(LIBC_UNITTEST_CXX_STANDARD ${CMAKE_CXX_STANDARD}) endif() set_target_properties( ${fq_build_target_name} PROPERTIES CXX_STANDARD ${LIBC_UNITTEST_CXX_STANDARD} ) set(link_libraries ${link_object_files}) # Test object files will depend on LINK_LIBRARIES passed down from `add_fp_unittest` foreach(lib IN LISTS LIBC_UNITTEST_LINK_LIBRARIES) if(TARGET ${lib}.unit) list(APPEND link_libraries ${lib}.unit) else() list(APPEND link_libraries ${lib}) endif() endforeach() set_target_properties(${fq_build_target_name} PROPERTIES RUNTIME_OUTPUT_DIRECTORY ${CMAKE_CURRENT_BINARY_DIR}) add_dependencies( ${fq_build_target_name} ${fq_deps_list} ) # LibcUnitTest should not depend on anything in LINK_LIBRARIES. if(NOT LIBC_UNITTEST_C_TEST) list(APPEND link_libraries LibcDeathTestExecutors.unit LibcTest.unit) endif() target_link_libraries(${fq_build_target_name} PRIVATE ${link_libraries}) if(NOT LIBC_UNITTEST_NO_RUN_POSTBUILD) add_custom_target( ${fq_target_name} COMMAND ${fq_build_target_name} COMMENT "Running unit test ${fq_target_name}" ) endif() if(LIBC_UNITTEST_SUITE) add_dependencies( ${LIBC_UNITTEST_SUITE} ${fq_target_name} ) endif() add_dependencies(libc-unit-tests ${fq_target_name}) endfunction(create_libc_unittest) function(add_libc_unittest target_name) add_target_with_flags( ${target_name} CREATE_TARGET create_libc_unittest ${ARGN} ) endfunction(add_libc_unittest) function(add_libc_exhaustive_testsuite suite_name) add_custom_target(${suite_name}) add_dependencies(exhaustive-check-libc ${suite_name}) endfunction(add_libc_exhaustive_testsuite) function(add_libc_long_running_testsuite suite_name) add_custom_target(${suite_name}) add_dependencies(libc-long-running-tests ${suite_name}) endfunction(add_libc_long_running_testsuite) # Rule to add a fuzzer test. # Usage # add_libc_fuzzer( # # SRCS # HDRS # DEPENDS # ) function(add_libc_fuzzer target_name) cmake_parse_arguments( "LIBC_FUZZER" "NEED_MPFR" # Optional arguments "" # Single value arguments "SRCS;HDRS;DEPENDS;COMPILE_OPTIONS" # Multi-value arguments ${ARGN} ) if(NOT LIBC_FUZZER_SRCS) message(FATAL_ERROR "'add_libc_fuzzer' target requires a SRCS list of .cpp " "files.") endif() if(NOT LIBC_FUZZER_DEPENDS) message(FATAL_ERROR "'add_libc_fuzzer' target requires a DEPENDS list of " "'add_entrypoint_object' targets.") endif() list(APPEND LIBC_FUZZER_LINK_LIBRARIES "") if(LIBC_FUZZER_NEED_MPFR) if(NOT LIBC_TESTS_CAN_USE_MPFR) message(VERBOSE "Fuzz test ${name} will be skipped as MPFR library is not available.") return() endif() list(APPEND LIBC_FUZZER_LINK_LIBRARIES mpfr gmp) endif() get_fq_target_name(${target_name} fq_target_name) get_fq_deps_list(fq_deps_list ${LIBC_FUZZER_DEPENDS}) get_object_files_for_test( link_object_files skipped_entrypoints_list ${fq_deps_list}) if(skipped_entrypoints_list) if(LIBC_CMAKE_VERBOSE_LOGGING) set(msg "Skipping fuzzer target ${fq_target_name} as it has missing deps: " "${skipped_entrypoints_list}.") message(STATUS ${msg}) endif() add_custom_target(${fq_target_name}) # A post build custom command is used to avoid running the command always. add_custom_command( TARGET ${fq_target_name} POST_BUILD COMMAND ${CMAKE_COMMAND} -E echo ${msg} ) return() endif() add_executable( ${fq_target_name} EXCLUDE_FROM_ALL ${LIBC_FUZZER_SRCS} ${LIBC_FUZZER_HDRS} ) target_include_directories(${fq_target_name} SYSTEM PRIVATE ${LIBC_INCLUDE_DIR}) target_include_directories(${fq_target_name} PRIVATE ${LIBC_SOURCE_DIR}) target_link_libraries(${fq_target_name} PRIVATE ${link_object_files} ${LIBC_FUZZER_LINK_LIBRARIES} ) set_target_properties(${fq_target_name} PROPERTIES RUNTIME_OUTPUT_DIRECTORY ${CMAKE_CURRENT_BINARY_DIR}) add_dependencies( ${fq_target_name} ${fq_deps_list} ) add_dependencies(libc-fuzzer ${fq_target_name}) target_compile_options(${fq_target_name} PRIVATE ${LIBC_FUZZER_COMPILE_OPTIONS}) endfunction(add_libc_fuzzer) # Get libgcc_s to be used in hermetic and integration tests. if(NOT MSVC AND NOT LIBC_CC_SUPPORTS_NOSTDLIBPP) execute_process(COMMAND ${CMAKE_CXX_COMPILER} -print-file-name=libgcc_s.so.1 OUTPUT_VARIABLE LIBGCC_S_LOCATION) string(STRIP ${LIBGCC_S_LOCATION} LIBGCC_S_LOCATION) endif() # DEPRECATED: Use add_hermetic_test instead. # # Rule to add an integration test. An integration test is like a unit test # but does not use the system libc. Not even the startup objects from the # system libc are linked in to the final executable. The final exe is fully # statically linked. The libc that the final exe links to consists of only # the object files of the DEPENDS targets. # # Usage: # add_integration_test( # # SUITE # SRCS [src2.cpp ...] # HDRS [hdr1.cpp ...] # DEPENDS # ARGS # ENV # COMPILE_OPTIONS # ) # # The DEPENDS list can be empty. If not empty, it should be a list of # targets added with add_entrypoint_object or add_object_library. function(add_integration_test test_name) get_fq_target_name(${test_name} fq_target_name) set(supported_targets gpu linux) if(NOT (${LIBC_TARGET_OS} IN_LIST supported_targets)) message(STATUS "Skipping ${fq_target_name} as it is not available on ${LIBC_TARGET_OS}.") return() endif() cmake_parse_arguments( "INTEGRATION_TEST" "" # No optional arguments "SUITE" # Single value arguments "SRCS;HDRS;DEPENDS;ARGS;ENV;COMPILE_OPTIONS;LOADER_ARGS" # Multi-value arguments ${ARGN} ) if(NOT INTEGRATION_TEST_SUITE) message(FATAL_ERROR "SUITE not specified for ${fq_target_name}") endif() if(NOT INTEGRATION_TEST_SRCS) message(FATAL_ERROR "The SRCS list for add_integration_test is missing.") endif() if(NOT LLVM_LIBC_FULL_BUILD AND NOT TARGET libc.startup.${LIBC_TARGET_OS}.crt1) message(FATAL_ERROR "The 'crt1' target for the integration test is missing.") endif() get_fq_target_name(${test_name}.libc fq_libc_target_name) get_fq_deps_list(fq_deps_list ${INTEGRATION_TEST_DEPENDS}) list(APPEND fq_deps_list # All integration tests use the operating system's startup object with the # integration test object and need to inherit the same dependencies. libc.startup.${LIBC_TARGET_OS}.crt1 libc.test.IntegrationTest.test # We always add the memory functions objects. This is because the # compiler's codegen can emit calls to the C memory functions. libc.src.string.memcmp libc.src.string.memcpy libc.src.string.memmove libc.src.string.memset libc.src.strings.bcmp libc.src.strings.bzero ) if(libc.src.compiler.__stack_chk_fail IN_LIST TARGET_LLVMLIBC_ENTRYPOINTS) # __stack_chk_fail should always be included if supported to allow building # libc with the stack protector enabled. list(APPEND fq_deps_list libc.src.compiler.__stack_chk_fail) endif() list(REMOVE_DUPLICATES fq_deps_list) # TODO: Instead of gathering internal object files from entrypoints, # collect the object files with public names of entrypoints. get_object_files_for_test( link_object_files skipped_entrypoints_list ${fq_deps_list}) if(skipped_entrypoints_list) if(LIBC_CMAKE_VERBOSE_LOGGING) set(msg "Skipping integration test ${fq_target_name} as it has missing deps: " "${skipped_entrypoints_list}.") message(STATUS ${msg}) endif() return() endif() list(REMOVE_DUPLICATES link_object_files) # Make a library of all deps add_library( ${fq_target_name}.__libc__ STATIC EXCLUDE_FROM_ALL ${link_object_files} ) set_target_properties(${fq_target_name}.__libc__ PROPERTIES ARCHIVE_OUTPUT_DIRECTORY ${CMAKE_CURRENT_BINARY_DIR}) set_target_properties(${fq_target_name}.__libc__ PROPERTIES ARCHIVE_OUTPUT_NAME ${fq_target_name}.libc) set(fq_build_target_name ${fq_target_name}.__build__) add_executable( ${fq_build_target_name} EXCLUDE_FROM_ALL ${INTEGRATION_TEST_SRCS} ${INTEGRATION_TEST_HDRS} ) set_target_properties(${fq_build_target_name} PROPERTIES RUNTIME_OUTPUT_DIRECTORY ${CMAKE_CURRENT_BINARY_DIR}) target_include_directories(${fq_build_target_name} SYSTEM PRIVATE ${LIBC_INCLUDE_DIR}) target_include_directories(${fq_build_target_name} PRIVATE ${LIBC_SOURCE_DIR}) _get_hermetic_test_compile_options(compile_options "") target_compile_options(${fq_build_target_name} PRIVATE ${compile_options} ${INTEGRATION_TEST_COMPILE_OPTIONS}) if(LIBC_TARGET_ARCHITECTURE_IS_AMDGPU) target_link_options(${fq_build_target_name} PRIVATE ${LIBC_COMPILE_OPTIONS_DEFAULT} ${INTEGRATION_TEST_COMPILE_OPTIONS} -Wno-multi-gpu -mcpu=${LIBC_GPU_TARGET_ARCHITECTURE} -flto "-Wl,-mllvm,-amdgpu-lower-global-ctor-dtor=0" -nostdlib -static "-Wl,-mllvm,-amdhsa-code-object-version=${LIBC_GPU_CODE_OBJECT_VERSION}") elseif(LIBC_TARGET_ARCHITECTURE_IS_NVPTX) target_link_options(${fq_build_target_name} PRIVATE ${LIBC_COMPILE_OPTIONS_DEFAULT} ${INTEGRATION_TEST_COMPILE_OPTIONS} "-Wl,--suppress-stack-size-warning" -Wno-multi-gpu "-Wl,-mllvm,-nvptx-lower-global-ctor-dtor=1" "-Wl,-mllvm,-nvptx-emit-init-fini-kernel" -march=${LIBC_GPU_TARGET_ARCHITECTURE} -nostdlib -static "--cuda-path=${LIBC_CUDA_ROOT}") elseif(LIBC_CC_SUPPORTS_NOSTDLIBPP) target_link_options(${fq_build_target_name} PRIVATE -nolibc -nostartfiles -nostdlib++ -static) else() # Older version of gcc does not support `nostdlib++` flag. We use # `nostdlib` and link against libgcc_s, which cannot be linked statically. target_link_options(${fq_build_target_name} PRIVATE -nolibc -nostartfiles -nostdlib) list(APPEND link_libraries ${LIBGCC_S_LOCATION}) endif() target_link_libraries( ${fq_build_target_name} ${fq_target_name}.__libc__ libc.startup.${LIBC_TARGET_OS}.crt1 libc.test.IntegrationTest.test ) add_dependencies(${fq_build_target_name} libc.test.IntegrationTest.test ${INTEGRATION_TEST_DEPENDS}) # Tests on the GPU require an external loader utility to launch the kernel. if(TARGET libc.utils.gpu.loader) add_dependencies(${fq_build_target_name} libc.utils.gpu.loader) get_target_property(gpu_loader_exe libc.utils.gpu.loader "EXECUTABLE") endif() # We have to use a separate var to store the command as a list because # the COMMAND option of `add_custom_target` cannot handle empty vars in the # command. For example, if INTEGRATION_TEST_ENV is empty, the actual # command also will not run. So, we use this list and tell `add_custom_target` # to expand the list (by including the option COMMAND_EXPAND_LISTS). This # makes `add_custom_target` construct the correct command and execute it. set(test_cmd ${INTEGRATION_TEST_ENV} $<$:${gpu_loader_exe}> ${CMAKE_CROSSCOMPILING_EMULATOR} ${INTEGRATION_TEST_LOADER_ARGS} $ ${INTEGRATION_TEST_ARGS}) add_custom_target( ${fq_target_name} COMMAND ${test_cmd} COMMAND_EXPAND_LISTS COMMENT "Running integration test ${fq_target_name}" ) add_dependencies(${INTEGRATION_TEST_SUITE} ${fq_target_name}) endfunction(add_integration_test) # Rule to add a hermetic program. A hermetic program is one whose executable is fully # statically linked and consists of pieces drawn only from LLVM's libc. Nothing, # including the startup objects, come from the system libc. # # For the GPU, these can be either tests or benchmarks, depending on the value # of the LINK_LIBRARIES arg. # # Usage: # add_libc_hermetic( # # SUITE # SRCS [src2.cpp ...] # HDRS [hdr1.cpp ...] # DEPENDS # ARGS # ENV # COMPILE_OPTIONS # LINK_LIBRARIES # LOADER_ARGS # ) function(add_libc_hermetic test_name) if(NOT TARGET libc.startup.${LIBC_TARGET_OS}.crt1) message(VERBOSE "Skipping ${fq_target_name} as it is not available on ${LIBC_TARGET_OS}.") return() endif() cmake_parse_arguments( "HERMETIC_TEST" "IS_GPU_BENCHMARK" # Optional arguments "SUITE" # Single value arguments "SRCS;HDRS;DEPENDS;ARGS;ENV;COMPILE_OPTIONS;LINK_LIBRARIES;LOADER_ARGS" # Multi-value arguments ${ARGN} ) if(NOT HERMETIC_TEST_SUITE) message(FATAL_ERROR "SUITE not specified for ${fq_target_name}") endif() if(NOT HERMETIC_TEST_SRCS) message(FATAL_ERROR "The SRCS list for add_integration_test is missing.") endif() get_fq_target_name(${test_name} fq_target_name) get_fq_target_name(${test_name}.libc fq_libc_target_name) get_fq_deps_list(fq_deps_list ${HERMETIC_TEST_DEPENDS}) list(APPEND fq_deps_list # Hermetic tests use the platform's startup object. So, their deps also # have to be collected. libc.startup.${LIBC_TARGET_OS}.crt1 # We always add the memory functions objects. This is because the # compiler's codegen can emit calls to the C memory functions. libc.src.__support.StringUtil.error_to_string libc.src.string.memcmp libc.src.string.memcpy libc.src.string.memmove libc.src.string.memset libc.src.strings.bcmp libc.src.strings.bzero ) if(libc.src.compiler.__stack_chk_fail IN_LIST TARGET_LLVMLIBC_ENTRYPOINTS) # __stack_chk_fail should always be included if supported to allow building # libc with the stack protector enabled. list(APPEND fq_deps_list libc.src.compiler.__stack_chk_fail) endif() if(libc.src.time.clock IN_LIST TARGET_LLVMLIBC_ENTRYPOINTS) # We will link in the 'clock' implementation if it exists for test timing. list(APPEND fq_deps_list libc.src.time.clock) endif() list(REMOVE_DUPLICATES fq_deps_list) # TODO: Instead of gathering internal object files from entrypoints, # collect the object files with public names of entrypoints. get_object_files_for_test( link_object_files skipped_entrypoints_list ${fq_deps_list}) if(skipped_entrypoints_list) if(LIBC_CMAKE_VERBOSE_LOGGING) set(msg "Skipping hermetic test ${fq_target_name} as it has missing deps: " "${skipped_entrypoints_list}.") message(STATUS ${msg}) endif() return() endif() list(REMOVE_DUPLICATES link_object_files) # Make a library of all deps add_library( ${fq_target_name}.__libc__ STATIC EXCLUDE_FROM_ALL ${link_object_files} ) set_target_properties(${fq_target_name}.__libc__ PROPERTIES ARCHIVE_OUTPUT_DIRECTORY ${CMAKE_CURRENT_BINARY_DIR}) set_target_properties(${fq_target_name}.__libc__ PROPERTIES ARCHIVE_OUTPUT_NAME ${fq_target_name}.libc) set(fq_build_target_name ${fq_target_name}.__build__) add_executable( ${fq_build_target_name} EXCLUDE_FROM_ALL ${HERMETIC_TEST_SRCS} ${HERMETIC_TEST_HDRS} ) set_target_properties(${fq_build_target_name} PROPERTIES RUNTIME_OUTPUT_DIRECTORY ${CMAKE_CURRENT_BINARY_DIR} #OUTPUT_NAME ${fq_target_name} ) target_include_directories(${fq_build_target_name} SYSTEM PRIVATE ${LIBC_INCLUDE_DIR}) target_include_directories(${fq_build_target_name} PRIVATE ${LIBC_SOURCE_DIR}) _get_hermetic_test_compile_options(compile_options "") target_compile_options(${fq_build_target_name} PRIVATE ${compile_options} ${HERMETIC_TEST_COMPILE_OPTIONS}) set(link_libraries "") foreach(lib IN LISTS HERMETIC_TEST_LINK_LIBRARIES) if(TARGET ${lib}.hermetic) list(APPEND link_libraries ${lib}.hermetic) else() list(APPEND link_libraries ${lib}) endif() endforeach() if(LIBC_TARGET_ARCHITECTURE_IS_AMDGPU) target_link_options(${fq_build_target_name} PRIVATE ${LIBC_COMPILE_OPTIONS_DEFAULT} -Wno-multi-gpu -mcpu=${LIBC_GPU_TARGET_ARCHITECTURE} -flto "-Wl,-mllvm,-amdgpu-lower-global-ctor-dtor=0" -nostdlib -static "-Wl,-mllvm,-amdhsa-code-object-version=${LIBC_GPU_CODE_OBJECT_VERSION}") elseif(LIBC_TARGET_ARCHITECTURE_IS_NVPTX) target_link_options(${fq_build_target_name} PRIVATE ${LIBC_COMPILE_OPTIONS_DEFAULT} -Wno-multi-gpu "-Wl,--suppress-stack-size-warning" "-Wl,-mllvm,-nvptx-lower-global-ctor-dtor=1" "-Wl,-mllvm,-nvptx-emit-init-fini-kernel" -march=${LIBC_GPU_TARGET_ARCHITECTURE} -nostdlib -static "--cuda-path=${LIBC_CUDA_ROOT}") elseif(LIBC_CC_SUPPORTS_NOSTDLIBPP) target_link_options(${fq_build_target_name} PRIVATE -nolibc -nostartfiles -nostdlib++ -static) else() # Older version of gcc does not support `nostdlib++` flag. We use # `nostdlib` and link against libgcc_s, which cannot be linked statically. target_link_options(${fq_build_target_name} PRIVATE -nolibc -nostartfiles -nostdlib) list(APPEND link_libraries ${LIBGCC_S_LOCATION}) endif() target_link_libraries( ${fq_build_target_name} PRIVATE libc.startup.${LIBC_TARGET_OS}.crt1 ${link_libraries} LibcHermeticTestSupport.hermetic ${fq_target_name}.__libc__) add_dependencies(${fq_build_target_name} LibcTest.hermetic libc.test.UnitTest.ErrnoSetterMatcher ${fq_deps_list}) # TODO: currently the dependency chain is broken such that getauxval cannot properly # propagate to hermetic tests. This is a temporary workaround. if (LIBC_TARGET_ARCHITECTURE_IS_AARCH64) target_link_libraries( ${fq_build_target_name} PRIVATE libc.src.sys.auxv.getauxval ) endif() # Tests on the GPU require an external loader utility to launch the kernel. if(TARGET libc.utils.gpu.loader) add_dependencies(${fq_build_target_name} libc.utils.gpu.loader) get_target_property(gpu_loader_exe libc.utils.gpu.loader "EXECUTABLE") endif() set(test_cmd ${HERMETIC_TEST_ENV} $<$:${gpu_loader_exe}> ${CMAKE_CROSSCOMPILING_EMULATOR} ${HERMETIC_TEST_LOADER_ARGS} $ ${HERMETIC_TEST_ARGS}) add_custom_target( ${fq_target_name} DEPENDS ${fq_target_name}-cmd ) add_custom_command( OUTPUT ${fq_target_name}-cmd COMMAND ${test_cmd} COMMAND_EXPAND_LISTS COMMENT "Running hermetic test ${fq_target_name}" ${LIBC_HERMETIC_TEST_JOB_POOL} ) set_source_files_properties(${fq_target_name}-cmd PROPERTIES SYMBOLIC "TRUE" ) add_dependencies(${HERMETIC_TEST_SUITE} ${fq_target_name}) if(NOT ${HERMETIC_TEST_IS_GPU_BENCHMARK}) # If it is a benchmark, it will already have been added to the # gpu-benchmark target add_dependencies(libc-hermetic-tests ${fq_target_name}) endif() endfunction(add_libc_hermetic) # A convenience function to add both a unit test as well as a hermetic test. function(add_libc_test test_name) cmake_parse_arguments( "LIBC_TEST" "UNIT_TEST_ONLY;HERMETIC_TEST_ONLY" # Optional arguments "" # Single value arguments "" # Multi-value arguments ${ARGN} ) if(LIBC_ENABLE_UNITTESTS AND NOT LIBC_TEST_HERMETIC_TEST_ONLY) add_libc_unittest(${test_name}.__unit__ ${LIBC_TEST_UNPARSED_ARGUMENTS}) endif() if(LIBC_ENABLE_HERMETIC_TESTS AND NOT LIBC_TEST_UNIT_TEST_ONLY) add_libc_hermetic( ${test_name}.__hermetic__ LINK_LIBRARIES LibcTest.hermetic ${LIBC_TEST_UNPARSED_ARGUMENTS} ) get_fq_target_name(${test_name} fq_test_name) if(TARGET ${fq_test_name}.__hermetic__ AND TARGET ${fq_test_name}.__unit__) # Tests like the file tests perform file operations on disk file. If we # don't chain up the unit test and hermetic test, then those tests will # step on each other's files. add_dependencies(${fq_test_name}.__hermetic__ ${fq_test_name}.__unit__) endif() endif() endfunction(add_libc_test) # Tests all implementations that can run on the target CPU. function(add_libc_multi_impl_test name suite) get_property(fq_implementations GLOBAL PROPERTY ${name}_implementations) foreach(fq_config_name IN LISTS fq_implementations) get_target_property(required_cpu_features ${fq_config_name} REQUIRE_CPU_FEATURES) cpu_supports(can_run "${required_cpu_features}") if(can_run) string(FIND ${fq_config_name} "." last_dot_loc REVERSE) math(EXPR name_loc "${last_dot_loc} + 1") string(SUBSTRING ${fq_config_name} ${name_loc} -1 target_name) add_libc_test( ${target_name}_test SUITE ${suite} COMPILE_OPTIONS ${LIBC_COMPILE_OPTIONS_NATIVE} LINK_LIBRARIES LibcMemoryHelpers ${ARGN} DEPENDS ${fq_config_name} libc.src.__support.macros.sanitizer ) get_fq_target_name(${fq_config_name}_test fq_target_name) else() message(STATUS "Skipping test for '${fq_config_name}' insufficient host cpu features '${required_cpu_features}'") endif() endforeach() endfunction()