xref: /aosp_15_r20/external/compiler-rt/cmake/Modules/CompilerRTUtils.cmake (revision 7c3d14c8b49c529e04be81a3ce6f5cc23712e4c6)
1include(CMakePushCheckState)
2include(CheckSymbolExists)
3
4# Because compiler-rt spends a lot of time setting up custom compile flags,
5# define a handy helper function for it. The compile flags setting in CMake
6# has serious issues that make its syntax challenging at best.
7function(set_target_compile_flags target)
8  set(argstring "")
9  foreach(arg ${ARGN})
10    set(argstring "${argstring} ${arg}")
11  endforeach()
12  set_property(TARGET ${target} PROPERTY COMPILE_FLAGS "${argstring}")
13endfunction()
14
15function(set_target_link_flags target)
16  set(argstring "")
17  foreach(arg ${ARGN})
18    set(argstring "${argstring} ${arg}")
19  endforeach()
20  set_property(TARGET ${target} PROPERTY LINK_FLAGS "${argstring}")
21endfunction()
22
23# Set the variable var_PYBOOL to True if var holds a true-ish string,
24# otherwise set it to False.
25macro(pythonize_bool var)
26  if (${var})
27    set(${var}_PYBOOL True)
28  else()
29    set(${var}_PYBOOL False)
30  endif()
31endmacro()
32
33# Appends value to all lists in ARGN, if the condition is true.
34macro(append_list_if condition value)
35  if(${condition})
36    foreach(list ${ARGN})
37      list(APPEND ${list} ${value})
38    endforeach()
39  endif()
40endmacro()
41
42# Appends value to all strings in ARGN, if the condition is true.
43macro(append_string_if condition value)
44  if(${condition})
45    foreach(str ${ARGN})
46      set(${str} "${${str}} ${value}")
47    endforeach()
48  endif()
49endmacro()
50
51macro(append_rtti_flag polarity list)
52  if(polarity)
53    append_list_if(COMPILER_RT_HAS_FRTTI_FLAG -frtti ${list})
54    append_list_if(COMPILER_RT_HAS_GR_FLAG /GR ${list})
55  else()
56    append_list_if(COMPILER_RT_HAS_FNO_RTTI_FLAG -fno-rtti ${list})
57    append_list_if(COMPILER_RT_HAS_GR_FLAG /GR- ${list})
58  endif()
59endmacro()
60
61macro(append_have_file_definition filename varname list)
62  check_include_file("${filename}" "${varname}")
63  if (NOT ${varname})
64    set("${varname}" 0)
65  endif()
66  list(APPEND ${list} "${varname}=${${varname}}")
67endmacro()
68
69macro(list_intersect output input1 input2)
70  set(${output})
71  foreach(it ${${input1}})
72    list(FIND ${input2} ${it} index)
73    if( NOT (index EQUAL -1))
74      list(APPEND ${output} ${it})
75    endif()
76  endforeach()
77endmacro()
78
79# Takes ${ARGN} and puts only supported architectures in @out_var list.
80function(filter_available_targets out_var)
81  set(archs ${${out_var}})
82  foreach(arch ${ARGN})
83    list(FIND COMPILER_RT_SUPPORTED_ARCH ${arch} ARCH_INDEX)
84    if(NOT (ARCH_INDEX EQUAL -1) AND CAN_TARGET_${arch})
85      list(APPEND archs ${arch})
86    endif()
87  endforeach()
88  set(${out_var} ${archs} PARENT_SCOPE)
89endfunction()
90
91function(check_compile_definition def argstring out_var)
92  if("${def}" STREQUAL "")
93    set(${out_var} TRUE PARENT_SCOPE)
94    return()
95  endif()
96  cmake_push_check_state()
97  set(CMAKE_REQUIRED_FLAGS "${CMAKE_REQUIRED_FLAGS} ${argstring}")
98  check_symbol_exists(${def} "" ${out_var})
99  cmake_pop_check_state()
100endfunction()
101
102# test_target_arch(<arch> <def> <target flags...>)
103# Checks if architecture is supported: runs host compiler with provided
104# flags to verify that:
105#   1) <def> is defined (if non-empty)
106#   2) simple file can be successfully built.
107# If successful, saves target flags for this architecture.
108macro(test_target_arch arch def)
109  set(TARGET_${arch}_CFLAGS ${ARGN})
110  set(TARGET_${arch}_LINKFLAGS ${ARGN})
111  set(argstring "")
112  foreach(arg ${ARGN})
113    set(argstring "${argstring} ${arg}")
114  endforeach()
115  check_compile_definition("${def}" "${argstring}" HAS_${arch}_DEF)
116  if(NOT HAS_${arch}_DEF)
117    set(CAN_TARGET_${arch} FALSE)
118  elseif(TEST_COMPILE_ONLY)
119    try_compile_only(CAN_TARGET_${arch} ${TARGET_${arch}_CFLAGS})
120  else()
121    set(argstring "${CMAKE_EXE_LINKER_FLAGS} ${argstring}")
122    try_compile(CAN_TARGET_${arch} ${CMAKE_BINARY_DIR} ${SIMPLE_SOURCE}
123                COMPILE_DEFINITIONS "${TARGET_${arch}_CFLAGS}"
124                OUTPUT_VARIABLE TARGET_${arch}_OUTPUT
125                CMAKE_FLAGS "-DCMAKE_EXE_LINKER_FLAGS:STRING=${argstring}")
126  endif()
127  if(${CAN_TARGET_${arch}})
128    list(APPEND COMPILER_RT_SUPPORTED_ARCH ${arch})
129  elseif("${COMPILER_RT_DEFAULT_TARGET_ARCH}" MATCHES "${arch}" AND
130         COMPILER_RT_HAS_EXPLICIT_DEFAULT_TARGET_TRIPLE)
131    # Bail out if we cannot target the architecture we plan to test.
132    message(FATAL_ERROR "Cannot compile for ${arch}:\n${TARGET_${arch}_OUTPUT}")
133  endif()
134endmacro()
135
136macro(detect_target_arch)
137  check_symbol_exists(__arm__ "" __ARM)
138  check_symbol_exists(__aarch64__ "" __AARCH64)
139  check_symbol_exists(__x86_64__ "" __X86_64)
140  check_symbol_exists(__i686__ "" __I686)
141  check_symbol_exists(__i386__ "" __I386)
142  check_symbol_exists(__mips__ "" __MIPS)
143  check_symbol_exists(__mips64__ "" __MIPS64)
144  check_symbol_exists(__s390x__ "" __S390X)
145  check_symbol_exists(__wasm32__ "" __WEBASSEMBLY32)
146  check_symbol_exists(__wasm64__ "" __WEBASSEMBLY64)
147  if(__ARM)
148    add_default_target_arch(arm)
149  elseif(__AARCH64)
150    add_default_target_arch(aarch64)
151  elseif(__X86_64)
152    add_default_target_arch(x86_64)
153  elseif(__I686)
154    add_default_target_arch(i686)
155  elseif(__I386)
156    add_default_target_arch(i386)
157  elseif(__MIPS64) # must be checked before __MIPS
158    add_default_target_arch(mips64)
159  elseif(__MIPS)
160    add_default_target_arch(mips)
161  elseif(__S390X)
162    add_default_target_arch(s390x)
163  elseif(__WEBASSEMBLY32)
164    add_default_target_arch(wasm32)
165  elseif(__WEBASSEMBLY64)
166    add_default_target_arch(wasm64)
167  endif()
168endmacro()
169