2020-04-22 01:10:58 +08:00
|
|
|
# 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:
|
2020-06-09 15:31:48 +08:00
|
|
|
# get_object_files_for_test(<result var>
|
|
|
|
# <skipped_entrypoints_var>
|
|
|
|
# <target0> [<target1> ...])
|
2020-04-22 01:10:58 +08:00
|
|
|
#
|
2020-06-09 15:31:48 +08:00
|
|
|
# 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.
|
2020-04-22 01:10:58 +08:00
|
|
|
# targetN is either an "add_entrypoint_target" target or an
|
|
|
|
# "add_object_library" target.
|
2020-06-09 15:31:48 +08:00
|
|
|
function(get_object_files_for_test result skipped_entrypoints_list)
|
2020-04-22 01:10:58 +08:00
|
|
|
set(object_files "")
|
2020-06-09 15:31:48 +08:00
|
|
|
set(skipped_list "")
|
2020-04-22 01:10:58 +08:00
|
|
|
foreach(dep IN LISTS ARGN)
|
|
|
|
get_target_property(dep_type ${dep} "TARGET_TYPE")
|
|
|
|
if(NOT dep_type)
|
|
|
|
# Target for which TARGET_TYPE property is not set do not
|
|
|
|
# provide any object files.
|
|
|
|
continue()
|
|
|
|
endif()
|
|
|
|
|
|
|
|
if(${dep_type} STREQUAL ${OBJECT_LIBRARY_TARGET_TYPE})
|
|
|
|
get_target_property(dep_object_files ${dep} "OBJECT_FILES")
|
|
|
|
if(dep_object_files)
|
|
|
|
list(APPEND object_files ${dep_object_files})
|
|
|
|
endif()
|
|
|
|
elseif(${dep_type} STREQUAL ${ENTRYPOINT_OBJ_TARGET_TYPE})
|
2020-06-09 15:31:48 +08:00
|
|
|
get_target_property(is_skipped ${dep} "SKIPPED")
|
|
|
|
if(is_skipped)
|
|
|
|
list(APPEND skipped_list ${dep})
|
|
|
|
continue()
|
|
|
|
endif()
|
2020-04-22 01:10:58 +08:00
|
|
|
get_target_property(object_file_raw ${dep} "OBJECT_FILE_RAW")
|
|
|
|
if(object_file_raw)
|
|
|
|
list(APPEND object_files ${object_file_raw})
|
|
|
|
endif()
|
|
|
|
endif()
|
|
|
|
|
|
|
|
get_target_property(indirect_deps ${dep} "DEPS")
|
2020-06-09 15:31:48 +08:00
|
|
|
get_object_files_for_test(
|
|
|
|
indirect_objfiles indirect_skipped_list ${indirect_deps})
|
2020-04-22 01:10:58 +08:00
|
|
|
list(APPEND object_files ${indirect_objfiles})
|
2020-06-09 15:31:48 +08:00
|
|
|
if(indirect_skipped_list)
|
|
|
|
list(APPEND skipped_list ${indirect_skipped_list})
|
|
|
|
endif()
|
2020-04-22 01:10:58 +08:00
|
|
|
endforeach(dep)
|
|
|
|
list(REMOVE_DUPLICATES object_files)
|
|
|
|
set(${result} ${object_files} PARENT_SCOPE)
|
2020-06-09 15:31:48 +08:00
|
|
|
list(REMOVE_DUPLICATES skipped_list)
|
|
|
|
set(${skipped_entrypoints_list} ${skipped_list} PARENT_SCOPE)
|
2020-04-22 01:10:58 +08:00
|
|
|
endfunction(get_object_files_for_test)
|
|
|
|
|
2020-04-18 09:42:40 +08:00
|
|
|
# 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>
|
|
|
|
# COMPILE_OPTIONS <list of special compile options for this target>
|
|
|
|
# )
|
|
|
|
function(add_libc_unittest target_name)
|
|
|
|
if(NOT LLVM_INCLUDE_TESTS)
|
|
|
|
return()
|
|
|
|
endif()
|
|
|
|
|
|
|
|
cmake_parse_arguments(
|
|
|
|
"LIBC_UNITTEST"
|
|
|
|
"" # No optional arguments
|
|
|
|
"SUITE" # Single value arguments
|
|
|
|
"SRCS;HDRS;DEPENDS;COMPILE_OPTIONS" # 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_target_name(${target_name} fq_target_name)
|
2020-06-09 15:31:48 +08:00
|
|
|
get_fq_deps_list(fq_deps_list ${LIBC_UNITTEST_DEPENDS})
|
|
|
|
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 loader 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.
|
|
|
|
|
|
|
|
set(msg "Skipping unittest ${fq_target_name} as it has missing deps: "
|
|
|
|
"${skipped_entrypoints_list}.")
|
|
|
|
message(STATUS ${msg})
|
|
|
|
return()
|
|
|
|
endif()
|
|
|
|
|
2020-04-18 09:42:40 +08:00
|
|
|
add_executable(
|
|
|
|
${fq_target_name}
|
|
|
|
EXCLUDE_FROM_ALL
|
|
|
|
${LIBC_UNITTEST_SRCS}
|
|
|
|
${LIBC_UNITTEST_HDRS}
|
|
|
|
)
|
|
|
|
target_include_directories(
|
|
|
|
${fq_target_name}
|
|
|
|
PRIVATE
|
|
|
|
${LIBC_SOURCE_DIR}
|
|
|
|
${LIBC_BUILD_DIR}
|
|
|
|
${LIBC_BUILD_DIR}/include
|
|
|
|
)
|
|
|
|
if(LIBC_UNITTEST_COMPILE_OPTIONS)
|
|
|
|
target_compile_options(
|
2020-12-08 14:46:03 +08:00
|
|
|
${fq_target_name}
|
2020-04-18 09:42:40 +08:00
|
|
|
PRIVATE ${LIBC_UNITTEST_COMPILE_OPTIONS}
|
|
|
|
)
|
|
|
|
endif()
|
|
|
|
|
2020-04-22 01:10:58 +08:00
|
|
|
target_link_libraries(${fq_target_name} PRIVATE ${link_object_files})
|
2020-04-18 09:42:40 +08:00
|
|
|
|
|
|
|
set_target_properties(${fq_target_name}
|
|
|
|
PROPERTIES RUNTIME_OUTPUT_DIRECTORY ${CMAKE_CURRENT_BINARY_DIR})
|
|
|
|
|
|
|
|
add_dependencies(
|
|
|
|
${fq_target_name}
|
|
|
|
${fq_deps_list}
|
|
|
|
)
|
|
|
|
|
|
|
|
target_link_libraries(${fq_target_name} PRIVATE LibcUnitTest libc_test_utils)
|
|
|
|
|
|
|
|
add_custom_command(
|
|
|
|
TARGET ${fq_target_name}
|
|
|
|
POST_BUILD
|
|
|
|
COMMAND $<TARGET_FILE:${fq_target_name}>
|
|
|
|
)
|
|
|
|
if(LIBC_UNITTEST_SUITE)
|
|
|
|
add_dependencies(
|
|
|
|
${LIBC_UNITTEST_SUITE}
|
|
|
|
${fq_target_name}
|
|
|
|
)
|
|
|
|
endif()
|
|
|
|
endfunction(add_libc_unittest)
|
|
|
|
|
|
|
|
function(add_libc_testsuite suite_name)
|
|
|
|
add_custom_target(${suite_name})
|
2021-03-23 00:43:31 +08:00
|
|
|
add_dependencies(check-llvmlibc ${suite_name})
|
2020-04-18 09:42:40 +08:00
|
|
|
endfunction(add_libc_testsuite)
|
|
|
|
|
2021-02-19 04:04:50 +08:00
|
|
|
function(add_libc_exhaustive_testsuite suite_name)
|
|
|
|
add_custom_target(${suite_name})
|
|
|
|
add_dependencies(exhaustive-check-libc ${suite_name})
|
|
|
|
endfunction(add_libc_exhaustive_testsuite)
|
|
|
|
|
2020-04-18 09:42:40 +08:00
|
|
|
# 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"
|
|
|
|
"" # No optional arguments
|
|
|
|
"" # Single value arguments
|
|
|
|
"SRCS;HDRS;DEPENDS" # 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()
|
|
|
|
|
|
|
|
get_fq_target_name(${target_name} fq_target_name)
|
2020-06-10 07:56:50 +08:00
|
|
|
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)
|
|
|
|
set(msg "Skipping fuzzer target ${fq_target_name} as it has missing deps: "
|
|
|
|
"${skipped_entrypoints_list}.")
|
|
|
|
message(STATUS ${msg})
|
|
|
|
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()
|
|
|
|
|
2020-04-18 09:42:40 +08:00
|
|
|
add_executable(
|
|
|
|
${fq_target_name}
|
|
|
|
EXCLUDE_FROM_ALL
|
|
|
|
${LIBC_FUZZER_SRCS}
|
|
|
|
${LIBC_FUZZER_HDRS}
|
|
|
|
)
|
|
|
|
target_include_directories(
|
|
|
|
${fq_target_name}
|
|
|
|
PRIVATE
|
|
|
|
${LIBC_SOURCE_DIR}
|
|
|
|
${LIBC_BUILD_DIR}
|
|
|
|
${LIBC_BUILD_DIR}/include
|
|
|
|
)
|
|
|
|
|
2020-04-22 01:10:58 +08:00
|
|
|
target_link_libraries(${fq_target_name} PRIVATE ${link_object_files})
|
2020-04-18 09:42:40 +08:00
|
|
|
|
|
|
|
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})
|
|
|
|
endfunction(add_libc_fuzzer)
|