1include(ExternalProject)
2include(CompilerRTUtils)
3include(HandleCompilerRT)
4
5function(set_target_output_directories target output_dir)
6  # For RUNTIME_OUTPUT_DIRECTORY variable, Multi-configuration generators
7  # append a per-configuration subdirectory to the specified directory.
8  # To avoid the appended folder, the configuration specific variable must be
9  # set 'RUNTIME_OUTPUT_DIRECTORY_${CONF}':
10  # RUNTIME_OUTPUT_DIRECTORY_DEBUG, RUNTIME_OUTPUT_DIRECTORY_RELEASE, ...
11  if(CMAKE_CONFIGURATION_TYPES)
12    foreach(build_mode ${CMAKE_CONFIGURATION_TYPES})
13      string(TOUPPER "${build_mode}" CONFIG_SUFFIX)
14      set_target_properties("${target}" PROPERTIES
15          "ARCHIVE_OUTPUT_DIRECTORY_${CONFIG_SUFFIX}" ${output_dir}
16          "LIBRARY_OUTPUT_DIRECTORY_${CONFIG_SUFFIX}" ${output_dir}
17          "RUNTIME_OUTPUT_DIRECTORY_${CONFIG_SUFFIX}" ${output_dir})
18    endforeach()
19  else()
20    set_target_properties("${target}" PROPERTIES
21        ARCHIVE_OUTPUT_DIRECTORY ${output_dir}
22        LIBRARY_OUTPUT_DIRECTORY ${output_dir}
23        RUNTIME_OUTPUT_DIRECTORY ${output_dir})
24  endif()
25endfunction()
26
27# Tries to add an "object library" target for a given list of OSs and/or
28# architectures with name "<name>.<arch>" for non-Darwin platforms if
29# architecture can be targeted, and "<name>.<os>" for Darwin platforms.
30# add_compiler_rt_object_libraries(<name>
31#                                  OS <os names>
32#                                  ARCHS <architectures>
33#                                  SOURCES <source files>
34#                                  CFLAGS <compile flags>
35#                                  DEFS <compile definitions>
36#                                  DEPS <dependencies>
37#                                  ADDITIONAL_HEADERS <header files>)
38function(add_compiler_rt_object_libraries name)
39  cmake_parse_arguments(LIB "" "" "OS;ARCHS;SOURCES;CFLAGS;DEFS;DEPS;ADDITIONAL_HEADERS"
40    ${ARGN})
41  set(libnames)
42  if(APPLE)
43    foreach(os ${LIB_OS})
44      set(libname "${name}.${os}")
45      set(libnames ${libnames} ${libname})
46      set(extra_cflags_${libname} ${DARWIN_${os}_CFLAGS})
47      list_intersect(LIB_ARCHS_${libname} DARWIN_${os}_ARCHS LIB_ARCHS)
48    endforeach()
49  else()
50    foreach(arch ${LIB_ARCHS})
51      set(libname "${name}.${arch}")
52      set(libnames ${libnames} ${libname})
53      set(extra_cflags_${libname} ${TARGET_${arch}_CFLAGS})
54      if(NOT CAN_TARGET_${arch})
55        message(FATAL_ERROR "Architecture ${arch} can't be targeted")
56        return()
57      endif()
58    endforeach()
59  endif()
60
61  # Add headers to LIB_SOURCES for IDEs
62  compiler_rt_process_sources(LIB_SOURCES
63    ${LIB_SOURCES}
64    ADDITIONAL_HEADERS
65      ${LIB_ADDITIONAL_HEADERS}
66  )
67
68  foreach(libname ${libnames})
69    add_library(${libname} OBJECT ${LIB_SOURCES})
70    if(LIB_DEPS)
71      add_dependencies(${libname} ${LIB_DEPS})
72    endif()
73
74    # Strip out -msse3 if this isn't macOS.
75    set(target_flags ${LIB_CFLAGS})
76    if(APPLE AND NOT "${libname}" MATCHES ".*\.osx.*")
77      list(REMOVE_ITEM target_flags "-msse3")
78    endif()
79
80    set_target_compile_flags(${libname}
81      ${extra_cflags_${libname}} ${target_flags})
82    set_property(TARGET ${libname} APPEND PROPERTY
83      COMPILE_DEFINITIONS ${LIB_DEFS})
84    set_target_properties(${libname} PROPERTIES FOLDER "Compiler-RT Libraries")
85    if(APPLE)
86      set_target_properties(${libname} PROPERTIES
87        OSX_ARCHITECTURES "${LIB_ARCHS_${libname}}")
88    endif()
89  endforeach()
90endfunction()
91
92# Takes a list of object library targets, and a suffix and appends the proper
93# TARGET_OBJECTS string to the output variable.
94# format_object_libs(<output> <suffix> ...)
95macro(format_object_libs output suffix)
96  foreach(lib ${ARGN})
97    list(APPEND ${output} $<TARGET_OBJECTS:${lib}.${suffix}>)
98  endforeach()
99endmacro()
100
101function(add_compiler_rt_component name)
102  add_custom_target(${name})
103  set_target_properties(${name} PROPERTIES FOLDER "Compiler-RT Misc")
104  if(COMMAND runtime_register_component)
105    runtime_register_component(${name})
106  endif()
107  add_dependencies(compiler-rt ${name})
108endfunction()
109
110function(add_asm_sources output)
111  set(${output} ${ARGN} PARENT_SCOPE)
112  # CMake doesn't pass the correct architecture for Apple prior to CMake 3.19. https://gitlab.kitware.com/cmake/cmake/-/issues/20771
113  # MinGW didn't work correctly with assembly prior to CMake 3.17. https://gitlab.kitware.com/cmake/cmake/-/merge_requests/4287 and https://reviews.llvm.org/rGb780df052dd2b246a760d00e00f7de9ebdab9d09
114  # Workaround these two issues by compiling as C.
115  # Same workaround used in libunwind. Also update there if changed here.
116  if((APPLE AND CMAKE_VERSION VERSION_LESS 3.19) OR (MINGW AND CMAKE_VERSION VERSION_LESS 3.17))
117    set_source_files_properties(${ARGN} PROPERTIES LANGUAGE C)
118  endif()
119endfunction()
120
121macro(set_output_name output name arch)
122  if(LLVM_ENABLE_PER_TARGET_RUNTIME_DIR)
123    set(${output} ${name})
124  else()
125    if(ANDROID AND ${arch} STREQUAL "i386")
126      set(${output} "${name}-i686${COMPILER_RT_OS_SUFFIX}")
127    else()
128      set(${output} "${name}-${arch}${COMPILER_RT_OS_SUFFIX}")
129    endif()
130  endif()
131endmacro()
132
133# Adds static or shared runtime for a list of architectures and operating
134# systems and puts it in the proper directory in the build and install trees.
135# add_compiler_rt_runtime(<name>
136#                         {OBJECT|STATIC|SHARED}
137#                         ARCHS <architectures>
138#                         OS <os list>
139#                         SOURCES <source files>
140#                         CFLAGS <compile flags>
141#                         LINK_FLAGS <linker flags>
142#                         DEFS <compile definitions>
143#                         LINK_LIBS <linked libraries> (only for shared library)
144#                         OBJECT_LIBS <object libraries to use as sources>
145#                         PARENT_TARGET <convenience parent target>
146#                         ADDITIONAL_HEADERS <header files>)
147function(add_compiler_rt_runtime name type)
148  if(NOT type MATCHES "^(OBJECT|STATIC|SHARED)$")
149    message(FATAL_ERROR "type argument must be OBJECT, STATIC or SHARED")
150    return()
151  endif()
152  cmake_parse_arguments(LIB
153    ""
154    "PARENT_TARGET"
155    "OS;ARCHS;SOURCES;CFLAGS;LINK_FLAGS;DEFS;LINK_LIBS;OBJECT_LIBS;ADDITIONAL_HEADERS"
156    ${ARGN})
157  set(libnames)
158  # Until we support this some other way, build compiler-rt runtime without LTO
159  # to allow non-LTO projects to link with it.
160  if(COMPILER_RT_HAS_FNO_LTO_FLAG)
161    set(NO_LTO_FLAGS "-fno-lto")
162  else()
163    set(NO_LTO_FLAGS "")
164  endif()
165
166  # By default do not instrument or use profdata for compiler-rt.
167  set(NO_PGO_FLAGS "")
168  if(NOT COMPILER_RT_ENABLE_PGO)
169    if(LLVM_PROFDATA_FILE AND COMPILER_RT_HAS_FNO_PROFILE_INSTR_USE_FLAG)
170      list(APPEND NO_PGO_FLAGS "-fno-profile-instr-use")
171    endif()
172    if(LLVM_BUILD_INSTRUMENTED MATCHES IR AND COMPILER_RT_HAS_FNO_PROFILE_GENERATE_FLAG)
173      list(APPEND NO_PGO_FLAGS "-fno-profile-generate")
174    elseif(LLVM_BUILD_INSTRUMENTED AND COMPILER_RT_HAS_FNO_PROFILE_INSTR_GENERATE_FLAG)
175      list(APPEND NO_PGO_FLAGS "-fno-profile-instr-generate")
176    endif()
177  endif()
178
179  list(LENGTH LIB_SOURCES LIB_SOURCES_LENGTH)
180  if (${LIB_SOURCES_LENGTH} GREATER 0)
181    # Add headers to LIB_SOURCES for IDEs. It doesn't make sense to
182    # do this for a runtime library that only consists of OBJECT
183    # libraries, so only add the headers when source files are present.
184    compiler_rt_process_sources(LIB_SOURCES
185      ${LIB_SOURCES}
186      ADDITIONAL_HEADERS
187        ${LIB_ADDITIONAL_HEADERS}
188    )
189  endif()
190
191  if(APPLE)
192    foreach(os ${LIB_OS})
193      # Strip out -msse3 if this isn't macOS.
194      list(LENGTH LIB_CFLAGS HAS_EXTRA_CFLAGS)
195      if(HAS_EXTRA_CFLAGS AND NOT "${os}" MATCHES "^(osx)$")
196        list(REMOVE_ITEM LIB_CFLAGS "-msse3")
197      endif()
198      if(type STREQUAL "STATIC")
199        set(libname "${name}_${os}")
200      else()
201        set(libname "${name}_${os}_dynamic")
202        set(extra_link_flags_${libname} ${DARWIN_${os}_LINK_FLAGS} ${LIB_LINK_FLAGS})
203      endif()
204      list_intersect(LIB_ARCHS_${libname} DARWIN_${os}_ARCHS LIB_ARCHS)
205      if(LIB_ARCHS_${libname})
206        list(APPEND libnames ${libname})
207        set(extra_cflags_${libname} ${DARWIN_${os}_CFLAGS} ${NO_LTO_FLAGS} ${NO_PGO_FLAGS} ${LIB_CFLAGS})
208        set(output_name_${libname} ${libname}${COMPILER_RT_OS_SUFFIX})
209        set(sources_${libname} ${LIB_SOURCES})
210        format_object_libs(sources_${libname} ${os} ${LIB_OBJECT_LIBS})
211        get_compiler_rt_output_dir(${COMPILER_RT_DEFAULT_TARGET_ARCH} output_dir_${libname})
212        get_compiler_rt_install_dir(${COMPILER_RT_DEFAULT_TARGET_ARCH} install_dir_${libname})
213      endif()
214    endforeach()
215  else()
216    foreach(arch ${LIB_ARCHS})
217      if(NOT CAN_TARGET_${arch})
218        message(FATAL_ERROR "Architecture ${arch} can't be targeted")
219        return()
220      endif()
221      if(type STREQUAL "OBJECT")
222        set(libname "${name}-${arch}")
223        set_output_name(output_name_${libname} ${name}${COMPILER_RT_OS_SUFFIX} ${arch})
224      elseif(type STREQUAL "STATIC")
225        set(libname "${name}-${arch}")
226        set_output_name(output_name_${libname} ${name} ${arch})
227      else()
228        set(libname "${name}-dynamic-${arch}")
229        set(extra_cflags_${libname} ${TARGET_${arch}_CFLAGS} ${LIB_CFLAGS})
230        set(extra_link_flags_${libname} ${TARGET_${arch}_LINK_FLAGS} ${LIB_LINK_FLAGS})
231        if(WIN32)
232          set_output_name(output_name_${libname} ${name}_dynamic ${arch})
233        else()
234          set_output_name(output_name_${libname} ${name} ${arch})
235        endif()
236      endif()
237      if(COMPILER_RT_USE_BUILTINS_LIBRARY AND NOT type STREQUAL "OBJECT" AND
238         NOT name STREQUAL "clang_rt.builtins")
239        get_compiler_rt_target(${arch} target)
240        find_compiler_rt_library(builtins ${target} builtins_${libname})
241        if(builtins_${libname} STREQUAL "NOTFOUND")
242          message(FATAL_ERROR "Cannot find builtins library for the target architecture")
243        endif()
244      endif()
245      set(sources_${libname} ${LIB_SOURCES})
246      format_object_libs(sources_${libname} ${arch} ${LIB_OBJECT_LIBS})
247      set(libnames ${libnames} ${libname})
248      set(extra_cflags_${libname} ${TARGET_${arch}_CFLAGS} ${NO_LTO_FLAGS} ${NO_PGO_FLAGS} ${LIB_CFLAGS})
249      get_compiler_rt_output_dir(${arch} output_dir_${libname})
250      get_compiler_rt_install_dir(${arch} install_dir_${libname})
251    endforeach()
252  endif()
253
254  if(NOT libnames)
255    return()
256  endif()
257
258  if(LIB_PARENT_TARGET)
259    # If the parent targets aren't created we should create them
260    if(NOT TARGET ${LIB_PARENT_TARGET})
261      add_custom_target(${LIB_PARENT_TARGET})
262      set_target_properties(${LIB_PARENT_TARGET} PROPERTIES
263                            FOLDER "Compiler-RT Misc")
264    endif()
265  endif()
266
267  foreach(libname ${libnames})
268    # If you are using a multi-configuration generator we don't generate
269    # per-library install rules, so we fall back to the parent target COMPONENT
270    if(CMAKE_CONFIGURATION_TYPES AND LIB_PARENT_TARGET)
271      set(COMPONENT_OPTION COMPONENT ${LIB_PARENT_TARGET})
272    else()
273      set(COMPONENT_OPTION COMPONENT ${libname})
274    endif()
275
276    if(type STREQUAL "OBJECT")
277      if(CMAKE_C_COMPILER_ID MATCHES Clang AND CMAKE_C_COMPILER_TARGET)
278        list(APPEND extra_cflags_${libname} "--target=${CMAKE_C_COMPILER_TARGET}")
279      endif()
280      if(CMAKE_SYSROOT)
281        list(APPEND extra_cflags_${libname} "--sysroot=${CMAKE_SYSROOT}")
282      endif()
283      string(REPLACE ";" " " extra_cflags_${libname} "${extra_cflags_${libname}}")
284      string(REGEX MATCHALL "<[A-Za-z0-9_]*>" substitutions
285             ${CMAKE_C_COMPILE_OBJECT})
286      set(compile_command_${libname} "${CMAKE_C_COMPILE_OBJECT}")
287
288      set(output_file_${libname} ${output_name_${libname}}${CMAKE_C_OUTPUT_EXTENSION})
289      foreach(substitution ${substitutions})
290        if(substitution STREQUAL "<CMAKE_C_COMPILER>")
291          string(REPLACE "<CMAKE_C_COMPILER>" "${CMAKE_C_COMPILER} ${CMAKE_C_COMPILER_ARG1}"
292                 compile_command_${libname} ${compile_command_${libname}})
293        elseif(substitution STREQUAL "<OBJECT>")
294          string(REPLACE "<OBJECT>" "${output_dir_${libname}}/${output_file_${libname}}"
295                 compile_command_${libname} ${compile_command_${libname}})
296        elseif(substitution STREQUAL "<SOURCE>")
297          string(REPLACE "<SOURCE>" "${sources_${libname}}"
298                 compile_command_${libname} ${compile_command_${libname}})
299        elseif(substitution STREQUAL "<FLAGS>")
300          string(REPLACE "<FLAGS>" "${CMAKE_C_FLAGS} ${extra_cflags_${libname}}"
301                 compile_command_${libname} ${compile_command_${libname}})
302        else()
303          string(REPLACE "${substitution}" "" compile_command_${libname}
304                 ${compile_command_${libname}})
305        endif()
306      endforeach()
307      separate_arguments(compile_command_${libname})
308      add_custom_command(
309          OUTPUT ${output_dir_${libname}}/${output_file_${libname}}
310          COMMAND ${compile_command_${libname}}
311          DEPENDS ${sources_${libname}}
312          COMMENT "Building C object ${output_file_${libname}}")
313      add_custom_target(${libname} DEPENDS ${output_dir_${libname}}/${output_file_${libname}})
314      install(FILES ${output_dir_${libname}}/${output_file_${libname}}
315        DESTINATION ${install_dir_${libname}}
316        ${COMPONENT_OPTION})
317    else()
318      add_library(${libname} ${type} ${sources_${libname}})
319      set_target_compile_flags(${libname} ${extra_cflags_${libname}})
320      set_target_link_flags(${libname} ${extra_link_flags_${libname}})
321      set_property(TARGET ${libname} APPEND PROPERTY
322                   COMPILE_DEFINITIONS ${LIB_DEFS})
323      set_target_output_directories(${libname} ${output_dir_${libname}})
324      install(TARGETS ${libname}
325        ARCHIVE DESTINATION ${install_dir_${libname}}
326                ${COMPONENT_OPTION}
327        LIBRARY DESTINATION ${install_dir_${libname}}
328                ${COMPONENT_OPTION}
329        RUNTIME DESTINATION ${install_dir_${libname}}
330                ${COMPONENT_OPTION})
331    endif()
332    set_target_properties(${libname} PROPERTIES
333        OUTPUT_NAME ${output_name_${libname}})
334    set_target_properties(${libname} PROPERTIES FOLDER "Compiler-RT Runtime")
335    if(LIB_LINK_LIBS)
336      target_link_libraries(${libname} PRIVATE ${LIB_LINK_LIBS})
337    endif()
338    if(builtins_${libname})
339      target_link_libraries(${libname} PRIVATE ${builtins_${libname}})
340    endif()
341    if(${type} STREQUAL "SHARED")
342      if(COMMAND llvm_setup_rpath)
343        llvm_setup_rpath(${libname})
344      endif()
345      if(WIN32 AND NOT CYGWIN AND NOT MINGW)
346        set_target_properties(${libname} PROPERTIES IMPORT_PREFIX "")
347        set_target_properties(${libname} PROPERTIES IMPORT_SUFFIX ".lib")
348      endif()
349      if(APPLE)
350        # Ad-hoc sign the dylibs
351        add_custom_command(TARGET ${libname}
352          POST_BUILD
353          COMMAND codesign --sign - $<TARGET_FILE:${libname}>
354          WORKING_DIRECTORY ${COMPILER_RT_LIBRARY_OUTPUT_DIR}
355        )
356      endif()
357    endif()
358
359    set(parent_target_arg)
360    if(LIB_PARENT_TARGET)
361      set(parent_target_arg PARENT_TARGET ${LIB_PARENT_TARGET})
362    endif()
363    add_compiler_rt_install_targets(${libname} ${parent_target_arg})
364
365    if(APPLE)
366      set_target_properties(${libname} PROPERTIES
367      OSX_ARCHITECTURES "${LIB_ARCHS_${libname}}")
368    endif()
369
370    if(type STREQUAL "SHARED")
371      rt_externalize_debuginfo(${libname})
372    endif()
373  endforeach()
374  if(LIB_PARENT_TARGET)
375    add_dependencies(${LIB_PARENT_TARGET} ${libnames})
376  endif()
377endfunction()
378
379# when cross compiling, COMPILER_RT_TEST_COMPILER_CFLAGS help
380# in compilation and linking of unittests.
381string(REPLACE " " ";" COMPILER_RT_UNITTEST_CFLAGS "${COMPILER_RT_TEST_COMPILER_CFLAGS}")
382set(COMPILER_RT_UNITTEST_LINK_FLAGS ${COMPILER_RT_UNITTEST_CFLAGS})
383
384# Unittests support.
385set(COMPILER_RT_GTEST_PATH ${LLVM_MAIN_SRC_DIR}/utils/unittest/googletest)
386set(COMPILER_RT_GTEST_SOURCE ${COMPILER_RT_GTEST_PATH}/src/gtest-all.cc)
387set(COMPILER_RT_GTEST_CFLAGS
388  -DGTEST_NO_LLVM_SUPPORT=1
389  -DGTEST_HAS_RTTI=0
390  -I${COMPILER_RT_GTEST_PATH}/include
391  -I${COMPILER_RT_GTEST_PATH}
392)
393
394# Mocking support.
395set(COMPILER_RT_GMOCK_PATH ${LLVM_MAIN_SRC_DIR}/utils/unittest/googlemock)
396set(COMPILER_RT_GMOCK_SOURCE ${COMPILER_RT_GMOCK_PATH}/src/gmock-all.cc)
397set(COMPILER_RT_GMOCK_CFLAGS
398  -DGTEST_NO_LLVM_SUPPORT=1
399  -DGTEST_HAS_RTTI=0
400  -I${COMPILER_RT_GMOCK_PATH}/include
401  -I${COMPILER_RT_GMOCK_PATH}
402)
403
404append_list_if(COMPILER_RT_DEBUG -DSANITIZER_DEBUG=1 COMPILER_RT_UNITTEST_CFLAGS)
405append_list_if(COMPILER_RT_HAS_WCOVERED_SWITCH_DEFAULT_FLAG -Wno-covered-switch-default COMPILER_RT_UNITTEST_CFLAGS)
406
407if(MSVC)
408  # gtest use a lot of stuff marked as deprecated on Windows.
409  list(APPEND COMPILER_RT_GTEST_CFLAGS -Wno-deprecated-declarations)
410endif()
411
412# Compile and register compiler-rt tests.
413# generate_compiler_rt_tests(<output object files> <test_suite> <test_name>
414#                           <test architecture>
415#                           KIND <custom prefix>
416#                           SUBDIR <subdirectory for testing binary>
417#                           SOURCES <sources to compile>
418#                           RUNTIME <tests runtime to link in>
419#                           CFLAGS <compile-time flags>
420#                           COMPILE_DEPS <compile-time dependencies>
421#                           DEPS <dependencies>
422#                           LINK_FLAGS <flags to use during linking>
423# )
424function(generate_compiler_rt_tests test_objects test_suite testname arch)
425  cmake_parse_arguments(TEST "" "KIND;RUNTIME;SUBDIR"
426    "SOURCES;COMPILE_DEPS;DEPS;CFLAGS;LINK_FLAGS" ${ARGN})
427
428  foreach(source ${TEST_SOURCES})
429    sanitizer_test_compile(
430      "${test_objects}" "${source}" "${arch}"
431      KIND ${TEST_KIND}
432      COMPILE_DEPS ${TEST_COMPILE_DEPS}
433      DEPS ${TEST_DEPS}
434      CFLAGS ${TEST_CFLAGS}
435      )
436  endforeach()
437
438  set(TEST_DEPS ${${test_objects}})
439
440  if(NOT "${TEST_RUNTIME}" STREQUAL "")
441    list(APPEND TEST_DEPS ${TEST_RUNTIME})
442    list(APPEND "${test_objects}" $<TARGET_FILE:${TEST_RUNTIME}>)
443  endif()
444
445  add_compiler_rt_test(${test_suite} "${testname}" "${arch}"
446    SUBDIR ${TEST_SUBDIR}
447    OBJECTS ${${test_objects}}
448    DEPS ${TEST_DEPS}
449    LINK_FLAGS ${TEST_LINK_FLAGS}
450    )
451  set("${test_objects}" "${${test_objects}}" PARENT_SCOPE)
452endfunction()
453
454# Link objects into a single executable with COMPILER_RT_TEST_COMPILER,
455# using specified link flags. Make executable a part of provided
456# test_suite.
457# add_compiler_rt_test(<test_suite> <test_name> <arch>
458#                      SUBDIR <subdirectory for binary>
459#                      OBJECTS <object files>
460#                      DEPS <deps (e.g. runtime libs)>
461#                      LINK_FLAGS <link flags>)
462function(add_compiler_rt_test test_suite test_name arch)
463  cmake_parse_arguments(TEST "" "SUBDIR" "OBJECTS;DEPS;LINK_FLAGS" "" ${ARGN})
464  set(output_dir ${CMAKE_CURRENT_BINARY_DIR})
465  if(TEST_SUBDIR)
466    set(output_dir "${output_dir}/${TEST_SUBDIR}")
467  endif()
468  set(output_dir "${output_dir}/${CMAKE_CFG_INTDIR}")
469  file(MAKE_DIRECTORY "${output_dir}")
470  set(output_bin "${output_dir}/${test_name}")
471  if(MSVC)
472    set(output_bin "${output_bin}.exe")
473  endif()
474
475  # Use host compiler in a standalone build, and just-built Clang otherwise.
476  if(NOT COMPILER_RT_STANDALONE_BUILD)
477    list(APPEND TEST_DEPS clang)
478  endif()
479
480  get_target_flags_for_arch(${arch} TARGET_LINK_FLAGS)
481  list(APPEND TEST_LINK_FLAGS ${TARGET_LINK_FLAGS})
482
483  # If we're not on MSVC, include the linker flags from CMAKE but override them
484  # with the provided link flags. This ensures that flags which are required to
485  # link programs at all are included, but the changes needed for the test
486  # trump. With MSVC we can't do that because CMake is set up to run link.exe
487  # when linking, not the compiler. Here, we hack it to use the compiler
488  # because we want to use -fsanitize flags.
489
490  # Only add CMAKE_EXE_LINKER_FLAGS when in a standalone bulid.
491  # Or else CMAKE_EXE_LINKER_FLAGS contains flags for build compiler of Clang/llvm.
492  # This might not be the same as what the COMPILER_RT_TEST_COMPILER supports.
493  # eg: the build compiler use lld linker and we build clang with default ld linker
494  # then to be tested clang will complain about lld options like --color-diagnostics.
495  if(NOT MSVC AND COMPILER_RT_STANDALONE_BUILD)
496    set(TEST_LINK_FLAGS "${CMAKE_EXE_LINKER_FLAGS} ${TEST_LINK_FLAGS}")
497    separate_arguments(TEST_LINK_FLAGS)
498  endif()
499  if(NOT COMPILER_RT_STANDALONE_BUILD AND COMPILER_RT_HAS_LLD AND "lld" IN_LIST LLVM_ENABLE_PROJECTS)
500    # CMAKE_EXE_LINKER_FLAGS may contain -fuse=lld
501    # FIXME: -DLLVM_ENABLE_LLD=ON and -DLLVM_ENABLE_PROJECTS without lld case.
502    list(APPEND TEST_DEPS lld)
503  endif()
504  add_custom_command(
505    OUTPUT "${output_bin}"
506    COMMAND ${COMPILER_RT_TEST_COMPILER} ${TEST_OBJECTS} -o "${output_bin}"
507            ${TEST_LINK_FLAGS}
508    DEPENDS ${TEST_DEPS}
509    )
510  add_custom_target(T${test_name} DEPENDS "${output_bin}")
511  set_target_properties(T${test_name} PROPERTIES FOLDER "Compiler-RT Tests")
512
513  # Make the test suite depend on the binary.
514  add_dependencies(${test_suite} T${test_name})
515endfunction()
516
517macro(add_compiler_rt_resource_file target_name file_name component)
518  set(src_file "${CMAKE_CURRENT_SOURCE_DIR}/${file_name}")
519  set(dst_file "${COMPILER_RT_OUTPUT_DIR}/share/${file_name}")
520  add_custom_command(OUTPUT ${dst_file}
521    DEPENDS ${src_file}
522    COMMAND ${CMAKE_COMMAND} -E copy_if_different ${src_file} ${dst_file}
523    COMMENT "Copying ${file_name}...")
524  add_custom_target(${target_name} DEPENDS ${dst_file})
525  # Install in Clang resource directory.
526  install(FILES ${file_name}
527    DESTINATION ${COMPILER_RT_INSTALL_PATH}/share
528    COMPONENT ${component})
529  add_dependencies(${component} ${target_name})
530
531  set_target_properties(${target_name} PROPERTIES FOLDER "Compiler-RT Misc")
532endmacro()
533
534macro(add_compiler_rt_script name)
535  set(dst ${COMPILER_RT_EXEC_OUTPUT_DIR}/${name})
536  set(src ${CMAKE_CURRENT_SOURCE_DIR}/${name})
537  add_custom_command(OUTPUT ${dst}
538    DEPENDS ${src}
539    COMMAND ${CMAKE_COMMAND} -E copy_if_different ${src} ${dst}
540    COMMENT "Copying ${name}...")
541  add_custom_target(${name} DEPENDS ${dst})
542  install(FILES ${dst}
543    PERMISSIONS OWNER_READ OWNER_WRITE OWNER_EXECUTE GROUP_READ GROUP_EXECUTE WORLD_READ WORLD_EXECUTE
544    DESTINATION ${COMPILER_RT_INSTALL_PATH}/bin)
545endmacro(add_compiler_rt_script src name)
546
547# Builds custom version of libc++ and installs it in <prefix>.
548# Can be used to build sanitized versions of libc++ for running unit tests.
549# add_custom_libcxx(<name> <prefix>
550#                   DEPS <list of build deps>
551#                   CFLAGS <list of compile flags>
552#                   USE_TOOLCHAIN)
553macro(add_custom_libcxx name prefix)
554  if(NOT COMPILER_RT_LIBCXX_PATH)
555    message(FATAL_ERROR "libcxx not found!")
556  endif()
557  if(NOT COMPILER_RT_LIBCXXABI_PATH)
558    message(FATAL_ERROR "libcxxabi not found!")
559  endif()
560
561  cmake_parse_arguments(LIBCXX "USE_TOOLCHAIN" "" "DEPS;CFLAGS;CMAKE_ARGS" ${ARGN})
562
563  if(LIBCXX_USE_TOOLCHAIN)
564    set(compiler_args -DCMAKE_C_COMPILER=${COMPILER_RT_TEST_COMPILER}
565                      -DCMAKE_CXX_COMPILER=${COMPILER_RT_TEST_CXX_COMPILER})
566    if(NOT COMPILER_RT_STANDALONE_BUILD AND NOT RUNTIMES_BUILD)
567      set(toolchain_deps $<TARGET_FILE:clang>)
568      set(force_deps DEPENDS $<TARGET_FILE:clang>)
569    endif()
570  else()
571    set(compiler_args -DCMAKE_C_COMPILER=${CMAKE_C_COMPILER}
572                      -DCMAKE_CXX_COMPILER=${CMAKE_CXX_COMPILER})
573  endif()
574
575  set(STAMP_DIR ${prefix}-stamps/)
576  set(BINARY_DIR ${prefix}-bins/)
577
578  add_custom_target(${name}-clear
579    COMMAND ${CMAKE_COMMAND} -E remove_directory ${BINARY_DIR}
580    COMMAND ${CMAKE_COMMAND} -E remove_directory ${STAMP_DIR}
581    COMMENT "Clobbering ${name} build and stamp directories"
582    USES_TERMINAL
583    )
584  set_target_properties(${name}-clear PROPERTIES FOLDER "Compiler-RT Misc")
585
586  add_custom_command(
587    OUTPUT ${CMAKE_CURRENT_BINARY_DIR}/${name}-clobber-stamp
588    DEPENDS ${LIBCXX_DEPS} ${toolchain_deps}
589    COMMAND ${CMAKE_COMMAND} -E touch ${BINARY_DIR}/CMakeCache.txt
590    COMMAND ${CMAKE_COMMAND} -E touch ${STAMP_DIR}/${name}-mkdir
591    COMMAND ${CMAKE_COMMAND} -E touch ${CMAKE_CURRENT_BINARY_DIR}/${name}-clobber-stamp
592    COMMENT "Clobbering bootstrap build and stamp directories"
593    )
594
595  add_custom_target(${name}-clobber
596    DEPENDS ${CMAKE_CURRENT_BINARY_DIR}/${name}-clobber-stamp)
597  set_target_properties(${name}-clobber PROPERTIES FOLDER "Compiler-RT Misc")
598
599  set(PASSTHROUGH_VARIABLES
600    CMAKE_C_COMPILER_TARGET
601    CMAKE_CXX_COMPILER_TARGET
602    CMAKE_SHARED_LINKER_FLAGS
603    CMAKE_MODULE_LINKER_FLAGS
604    CMAKE_EXE_LINKER_FLAGS
605    CMAKE_INSTALL_PREFIX
606    CMAKE_MAKE_PROGRAM
607    CMAKE_LINKER
608    CMAKE_AR
609    CMAKE_RANLIB
610    CMAKE_NM
611    CMAKE_OBJCOPY
612    CMAKE_OBJDUMP
613    CMAKE_STRIP
614    CMAKE_SYSROOT
615    CMAKE_SYSTEM_NAME)
616  foreach(variable ${PASSTHROUGH_VARIABLES})
617    get_property(is_value_set CACHE ${variable} PROPERTY VALUE SET)
618    if(${is_value_set})
619      get_property(value CACHE ${variable} PROPERTY VALUE)
620      list(APPEND CMAKE_PASSTHROUGH_VARIABLES -D${variable}=${value})
621    endif()
622  endforeach()
623
624  string(REPLACE ";" " " LIBCXX_C_FLAGS "${LIBCXX_CFLAGS}")
625  get_property(C_FLAGS CACHE CMAKE_C_FLAGS PROPERTY VALUE)
626  set(LIBCXX_C_FLAGS "${LIBCXX_C_FLAGS} ${C_FLAGS}")
627
628  string(REPLACE ";" " " LIBCXX_CXX_FLAGS "${LIBCXX_CFLAGS}")
629  get_property(CXX_FLAGS CACHE CMAKE_CXX_FLAGS PROPERTY VALUE)
630  set(LIBCXX_CXX_FLAGS "${LIBCXX_CXX_FLAGS} ${CXX_FLAGS}")
631
632  ExternalProject_Add(${name}
633    DEPENDS ${name}-clobber ${LIBCXX_DEPS}
634    PREFIX ${prefix}
635    SOURCE_DIR ${COMPILER_RT_SOURCE_DIR}/cmake/Modules/CustomLibcxx
636    STAMP_DIR ${STAMP_DIR}
637    BINARY_DIR ${BINARY_DIR}
638    CMAKE_ARGS ${CMAKE_PASSTHROUGH_VARIABLES}
639               ${compiler_args}
640               -DCMAKE_C_FLAGS=${LIBCXX_C_FLAGS}
641               -DCMAKE_CXX_FLAGS=${LIBCXX_CXX_FLAGS}
642               -DCMAKE_BUILD_TYPE=Release
643               -DCMAKE_TRY_COMPILE_TARGET_TYPE=STATIC_LIBRARY
644               -DLLVM_PATH=${LLVM_MAIN_SRC_DIR}
645               -DLLVM_BINARY_DIR=${prefix}
646               -DLLVM_LIBRARY_OUTPUT_INTDIR=${prefix}/lib
647               -DCOMPILER_RT_LIBCXX_PATH=${COMPILER_RT_LIBCXX_PATH}
648               -DCOMPILER_RT_LIBCXXABI_PATH=${COMPILER_RT_LIBCXXABI_PATH}
649               ${LIBCXX_CMAKE_ARGS}
650    INSTALL_COMMAND ""
651    STEP_TARGETS configure build
652    BUILD_ALWAYS 1
653    USES_TERMINAL_CONFIGURE 1
654    USES_TERMINAL_BUILD 1
655    USES_TERMINAL_INSTALL 1
656    EXCLUDE_FROM_ALL TRUE
657    BUILD_BYPRODUCTS "${prefix}/lib/libc++.a" "${prefix}/lib/libc++abi.a"
658    )
659
660  if (CMAKE_GENERATOR MATCHES "Make")
661    set(run_clean "$(MAKE)" "-C" "${BINARY_DIR}" "clean")
662  else()
663    set(run_clean ${CMAKE_COMMAND} --build ${BINARY_DIR} --target clean
664                                   --config "$<CONFIG>")
665  endif()
666
667  ExternalProject_Add_Step(${name} clean
668    COMMAND ${run_clean}
669    COMMENT "Cleaning ${name}..."
670    DEPENDEES configure
671    ${force_deps}
672    WORKING_DIRECTORY ${BINARY_DIR}
673    EXCLUDE_FROM_MAIN 1
674    USES_TERMINAL 1
675    )
676  ExternalProject_Add_StepTargets(${name} clean)
677
678  if(LIBCXX_USE_TOOLCHAIN)
679    add_dependencies(${name}-clean ${name}-clobber)
680    set_target_properties(${name}-clean PROPERTIES
681      SOURCES ${CMAKE_CURRENT_BINARY_DIR}/${name}-clobber-stamp)
682  endif()
683endmacro()
684
685function(rt_externalize_debuginfo name)
686  if(NOT COMPILER_RT_EXTERNALIZE_DEBUGINFO)
687    return()
688  endif()
689
690  if(NOT COMPILER_RT_EXTERNALIZE_DEBUGINFO_SKIP_STRIP)
691    set(strip_command COMMAND xcrun strip -Sl $<TARGET_FILE:${name}>)
692  endif()
693
694  if(APPLE)
695    if(CMAKE_CXX_FLAGS MATCHES "-flto"
696      OR CMAKE_CXX_FLAGS_${uppercase_CMAKE_BUILD_TYPE} MATCHES "-flto")
697
698      set(lto_object ${CMAKE_CURRENT_BINARY_DIR}/${CMAKE_CFG_INTDIR}/${name}-lto.o)
699      set_property(TARGET ${name} APPEND_STRING PROPERTY
700        LINK_FLAGS " -Wl,-object_path_lto -Wl,${lto_object}")
701    endif()
702    add_custom_command(TARGET ${name} POST_BUILD
703      COMMAND xcrun dsymutil $<TARGET_FILE:${name}>
704      ${strip_command})
705  else()
706    message(FATAL_ERROR "COMPILER_RT_EXTERNALIZE_DEBUGINFO isn't implemented for non-darwin platforms!")
707  endif()
708endfunction()
709
710
711# Configure lit configuration files, including compiler-rt specific variables.
712function(configure_compiler_rt_lit_site_cfg input output)
713  set_llvm_build_mode()
714
715  get_compiler_rt_output_dir(${COMPILER_RT_DEFAULT_TARGET_ARCH} output_dir)
716
717  string(REPLACE ${CMAKE_CFG_INTDIR} ${LLVM_BUILD_MODE} COMPILER_RT_RESOLVED_TEST_COMPILER ${COMPILER_RT_TEST_COMPILER})
718  string(REPLACE ${CMAKE_CFG_INTDIR} ${LLVM_BUILD_MODE} COMPILER_RT_RESOLVED_LIBRARY_OUTPUT_DIR ${output_dir})
719
720  configure_lit_site_cfg(${input} ${output})
721endfunction()
722