mirror of
https://github.com/llvm/llvm-project.git
synced 2025-04-17 17:26:41 +00:00

This is an attempt to move away from C++ headers to be able to run the test without `libcxx`. closes #129838 cc @lntue @RossComputerGuy
892 lines
33 KiB
CMake
892 lines
33 KiB
CMake
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")
|
|
# FIXME: convert to -Wsign-conversion
|
|
list(APPEND compile_options "-Wno-sign-conversion")
|
|
list(APPEND compile_options "-Wimplicit-fallthrough")
|
|
list(APPEND compile_options "-Wwrite-strings")
|
|
# 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(CMAKE_CXX_COMPILER_ID STREQUAL "Clang")
|
|
list(APPEND compile_options "-Wstrict-prototypes")
|
|
list(APPEND compile_options "-Wextra-semi")
|
|
list(APPEND compile_options "-Wnewline-eof")
|
|
list(APPEND compile_options "-Wnonportable-system-include-path")
|
|
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(<result var>
|
|
# <skipped_entrypoints_var>
|
|
# <target0> [<target1> ...])
|
|
#
|
|
# The list of object files is collected in <result_var>.
|
|
# If skipped entrypoints were found, then <skipped_entrypoints_var> 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(
|
|
# <target name>
|
|
# SUITE <name of the suite this test belongs to>
|
|
# SRCS <list of .cpp files for the test>
|
|
# HDRS <list of .h files for the test>
|
|
# DEPENDS <list of dependencies>
|
|
# ENV <list of environment variables to set before running the test>
|
|
# COMPILE_OPTIONS <list of special compile options for this target>
|
|
# LINK_LIBRARIES <list of linking libraries for this target>
|
|
# )
|
|
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;ENV;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}")
|
|
# TODO: Ideally we would have a separate function for link options.
|
|
set(link_options ${compile_options})
|
|
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 ${link_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 ${LIBC_UNITTEST_ENV} ${CMAKE_CROSSCOMPILING_EMULATOR} ${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(
|
|
# <target name>
|
|
# SRCS <list of .cpp files for the test>
|
|
# HDRS <list of .h files for the test>
|
|
# DEPENDS <list of dependencies>
|
|
# )
|
|
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(
|
|
# <target name>
|
|
# SUITE <the suite to which the test should belong>
|
|
# SRCS <src1.cpp> [src2.cpp ...]
|
|
# HDRS [hdr1.cpp ...]
|
|
# DEPENDS <list of entrypoint or other object targets>
|
|
# ARGS <list of command line arguments to be passed to the test>
|
|
# ENV <list of environment variables to set before running the test>
|
|
# COMPILE_OPTIONS <list of special compile options for this target>
|
|
# )
|
|
#
|
|
# 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 -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-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}
|
|
$<$<BOOL:${LIBC_TARGET_OS_IS_GPU}>:${gpu_loader_exe}>
|
|
${CMAKE_CROSSCOMPILING_EMULATOR}
|
|
${INTEGRATION_TEST_LOADER_ARGS}
|
|
$<TARGET_FILE:${fq_build_target_name}> ${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(
|
|
# <target name>
|
|
# SUITE <the suite to which the test should belong>
|
|
# SRCS <src1.cpp> [src2.cpp ...]
|
|
# HDRS [hdr1.cpp ...]
|
|
# DEPENDS <list of entrypoint or other object targets>
|
|
# ARGS <list of command line arguments to be passed to the test>
|
|
# ENV <list of environment variables to set before running the test>
|
|
# COMPILE_OPTIONS <list of special compile options for the test>
|
|
# LINK_LIBRARIES <list of linking libraries for this target>
|
|
# LOADER_ARGS <list of special args to loaders (like the GPU loader)>
|
|
# )
|
|
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;NO_RUN_POSTBUILD" # Optional arguments
|
|
"SUITE;CXX_STANDARD" # 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)
|
|
|
|
if(HERMETIC_TEST_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
|
|
${HERMETIC_TEST_SRCS}
|
|
${HERMETIC_TEST_HDRS}
|
|
)
|
|
|
|
if(NOT HERMETIC_TEST_CXX_STANDARD)
|
|
set(HERMETIC_TEST_CXX_STANDARD ${CMAKE_CXX_STANDARD})
|
|
endif()
|
|
set_target_properties(${fq_build_target_name}
|
|
PROPERTIES
|
|
RUNTIME_OUTPUT_DIRECTORY ${CMAKE_CURRENT_BINARY_DIR}
|
|
CXX_STANDARD ${HERMETIC_TEST_CXX_STANDARD}
|
|
)
|
|
|
|
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-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()
|
|
|
|
if(NOT HERMETIC_TEST_NO_RUN_POSTBUILD)
|
|
set(test_cmd ${HERMETIC_TEST_ENV}
|
|
$<$<BOOL:${LIBC_TARGET_OS_IS_GPU}>:${gpu_loader_exe}> ${CMAKE_CROSSCOMPILING_EMULATOR} ${HERMETIC_TEST_LOADER_ARGS}
|
|
$<TARGET_FILE:${fq_build_target_name}> ${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"
|
|
)
|
|
endif()
|
|
|
|
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()
|