1 // CodeGen/RuntimeLibcallSignatures.cpp - R.T. Lib. Call Signatures -*- C++ -*--
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 ///
9 /// \file
10 /// This file contains signature information for runtime libcalls.
11 ///
12 /// CodeGen uses external symbols, which it refers to by name. The WebAssembly
13 /// target needs type information for all functions. This file contains a big
14 /// table providing type signatures for all runtime library functions that LLVM
15 /// uses.
16 ///
17 /// This is currently a fairly heavy-handed solution.
18 ///
19 //===----------------------------------------------------------------------===//
20 
21 #include "WebAssemblyRuntimeLibcallSignatures.h"
22 #include "WebAssemblySubtarget.h"
23 #include "llvm/CodeGen/RuntimeLibcalls.h"
24 #include "llvm/Support/ManagedStatic.h"
25 
26 using namespace llvm;
27 
28 namespace {
29 
30 enum RuntimeLibcallSignature {
31   func,
32   f32_func_f32,
33   f32_func_f64,
34   f32_func_i32,
35   f32_func_i64,
36   f32_func_i16,
37   f64_func_f32,
38   f64_func_f64,
39   f64_func_i32,
40   f64_func_i64,
41   i32_func_f32,
42   i32_func_f64,
43   i32_func_i32,
44   i64_func_f32,
45   i64_func_f64,
46   i64_func_i64,
47   f32_func_f32_f32,
48   f32_func_f32_i32,
49   f32_func_i64_i64,
50   f64_func_f64_f64,
51   f64_func_f64_i32,
52   f64_func_i64_i64,
53   i16_func_f32,
54   i8_func_i8_i8,
55   func_f32_iPTR_iPTR,
56   func_f64_iPTR_iPTR,
57   i16_func_i16_i16,
58   i32_func_f32_f32,
59   i32_func_f64_f64,
60   i32_func_i32_i32,
61   i32_func_i32_i32_iPTR,
62   i64_func_i64_i64,
63   i64_func_i64_i64_iPTR,
64   i64_i64_func_f32,
65   i64_i64_func_f64,
66   i16_i16_func_i16_i16,
67   i32_i32_func_i32_i32,
68   i64_i64_func_i64_i64,
69   i64_i64_func_i64_i64_i64_i64,
70   i64_i64_func_i64_i64_i64_i64_iPTR,
71   i64_i64_i64_i64_func_i64_i64_i64_i64,
72   i64_i64_func_i64_i64_i32,
73   iPTR_func_iPTR_i32_iPTR,
74   iPTR_func_iPTR_iPTR_iPTR,
75   f32_func_f32_f32_f32,
76   f64_func_f64_f64_f64,
77   func_i64_i64_iPTR_iPTR,
78   func_iPTR_f32,
79   func_iPTR_f64,
80   func_iPTR_i32,
81   func_iPTR_i64,
82   func_iPTR_i64_i64,
83   func_iPTR_i64_i64_i64_i64,
84   func_iPTR_i64_i64_i64_i64_i64_i64,
85   i32_func_i64_i64,
86   i32_func_i64_i64_i64_i64,
87   unsupported
88 };
89 
90 struct RuntimeLibcallSignatureTable {
91   std::vector<RuntimeLibcallSignature> Table;
92 
93   // Any newly-added libcalls will be unsupported by default.
94   RuntimeLibcallSignatureTable() : Table(RTLIB::UNKNOWN_LIBCALL, unsupported) {
95     // Integer
96     Table[RTLIB::SHL_I16] = i16_func_i16_i16;
97     Table[RTLIB::SHL_I32] = i32_func_i32_i32;
98     Table[RTLIB::SHL_I64] = i64_func_i64_i64;
99     Table[RTLIB::SHL_I128] = i64_i64_func_i64_i64_i32;
100     Table[RTLIB::SRL_I16] = i16_func_i16_i16;
101     Table[RTLIB::SRL_I32] = i32_func_i32_i32;
102     Table[RTLIB::SRL_I64] = i64_func_i64_i64;
103     Table[RTLIB::SRL_I128] = i64_i64_func_i64_i64_i32;
104     Table[RTLIB::SRA_I16] = i16_func_i16_i16;
105     Table[RTLIB::SRA_I32] = i32_func_i32_i32;
106     Table[RTLIB::SRA_I64] = i64_func_i64_i64;
107     Table[RTLIB::SRA_I128] = i64_i64_func_i64_i64_i32;
108     Table[RTLIB::MUL_I8] = i8_func_i8_i8;
109     Table[RTLIB::MUL_I16] = i16_func_i16_i16;
110     Table[RTLIB::MUL_I32] = i32_func_i32_i32;
111     Table[RTLIB::MUL_I64] = i64_func_i64_i64;
112     Table[RTLIB::MUL_I128] = i64_i64_func_i64_i64_i64_i64;
113     Table[RTLIB::MULO_I32] = i32_func_i32_i32_iPTR;
114     Table[RTLIB::MULO_I64] = i64_func_i64_i64_iPTR;
115     Table[RTLIB::MULO_I128] = i64_i64_func_i64_i64_i64_i64_iPTR;
116     Table[RTLIB::SDIV_I8] = i8_func_i8_i8;
117     Table[RTLIB::SDIV_I16] = i16_func_i16_i16;
118     Table[RTLIB::SDIV_I32] = i32_func_i32_i32;
119     Table[RTLIB::SDIV_I64] = i64_func_i64_i64;
120     Table[RTLIB::SDIV_I128] = i64_i64_func_i64_i64_i64_i64;
121     Table[RTLIB::UDIV_I8] = i8_func_i8_i8;
122     Table[RTLIB::UDIV_I16] = i16_func_i16_i16;
123     Table[RTLIB::UDIV_I32] = i32_func_i32_i32;
124     Table[RTLIB::UDIV_I64] = i64_func_i64_i64;
125     Table[RTLIB::UDIV_I128] = i64_i64_func_i64_i64_i64_i64;
126     Table[RTLIB::SREM_I8] = i8_func_i8_i8;
127     Table[RTLIB::SREM_I16] = i16_func_i16_i16;
128     Table[RTLIB::SREM_I32] = i32_func_i32_i32;
129     Table[RTLIB::SREM_I64] = i64_func_i64_i64;
130     Table[RTLIB::SREM_I128] = i64_i64_func_i64_i64_i64_i64;
131     Table[RTLIB::UREM_I8] = i8_func_i8_i8;
132     Table[RTLIB::UREM_I16] = i16_func_i16_i16;
133     Table[RTLIB::UREM_I32] = i32_func_i32_i32;
134     Table[RTLIB::UREM_I64] = i64_func_i64_i64;
135     Table[RTLIB::UREM_I128] = i64_i64_func_i64_i64_i64_i64;
136     Table[RTLIB::SDIVREM_I8] = i8_func_i8_i8;
137     Table[RTLIB::SDIVREM_I16] = i16_i16_func_i16_i16;
138     Table[RTLIB::SDIVREM_I32] = i32_i32_func_i32_i32;
139     Table[RTLIB::SDIVREM_I64] = i64_func_i64_i64;
140     Table[RTLIB::SDIVREM_I128] = i64_i64_i64_i64_func_i64_i64_i64_i64;
141     Table[RTLIB::UDIVREM_I8] = i8_func_i8_i8;
142     Table[RTLIB::UDIVREM_I16] = i16_i16_func_i16_i16;
143     Table[RTLIB::UDIVREM_I32] = i32_i32_func_i32_i32;
144     Table[RTLIB::UDIVREM_I64] = i64_i64_func_i64_i64;
145     Table[RTLIB::UDIVREM_I128] = i64_i64_i64_i64_func_i64_i64_i64_i64;
146     Table[RTLIB::NEG_I32] = i32_func_i32;
147     Table[RTLIB::NEG_I64] = i64_func_i64;
148 
149     // Floating-point.
150     // All F80 and PPCF128 routines are unsupported.
151     Table[RTLIB::ADD_F32] = f32_func_f32_f32;
152     Table[RTLIB::ADD_F64] = f64_func_f64_f64;
153     Table[RTLIB::ADD_F128] = func_iPTR_i64_i64_i64_i64;
154     Table[RTLIB::SUB_F32] = f32_func_f32_f32;
155     Table[RTLIB::SUB_F64] = f64_func_f64_f64;
156     Table[RTLIB::SUB_F128] = func_iPTR_i64_i64_i64_i64;
157     Table[RTLIB::MUL_F32] = f32_func_f32_f32;
158     Table[RTLIB::MUL_F64] = f64_func_f64_f64;
159     Table[RTLIB::MUL_F128] = func_iPTR_i64_i64_i64_i64;
160     Table[RTLIB::DIV_F32] = f32_func_f32_f32;
161     Table[RTLIB::DIV_F64] = f64_func_f64_f64;
162     Table[RTLIB::DIV_F128] = func_iPTR_i64_i64_i64_i64;
163     Table[RTLIB::REM_F32] = f32_func_f32_f32;
164     Table[RTLIB::REM_F64] = f64_func_f64_f64;
165     Table[RTLIB::REM_F128] = func_iPTR_i64_i64_i64_i64;
166     Table[RTLIB::FMA_F32] = f32_func_f32_f32_f32;
167     Table[RTLIB::FMA_F64] = f64_func_f64_f64_f64;
168     Table[RTLIB::FMA_F128] = func_iPTR_i64_i64_i64_i64_i64_i64;
169     Table[RTLIB::POWI_F32] = f32_func_f32_i32;
170     Table[RTLIB::POWI_F64] = f64_func_f64_i32;
171     Table[RTLIB::POWI_F128] = func_iPTR_i64_i64_i64_i64;
172     Table[RTLIB::SQRT_F32] = f32_func_f32;
173     Table[RTLIB::SQRT_F64] = f64_func_f64;
174     Table[RTLIB::SQRT_F128] = func_iPTR_i64_i64;
175     Table[RTLIB::LOG_F32] = f32_func_f32;
176     Table[RTLIB::LOG_F64] = f64_func_f64;
177     Table[RTLIB::LOG_F128] = func_iPTR_i64_i64;
178     Table[RTLIB::LOG2_F32] = f32_func_f32;
179     Table[RTLIB::LOG2_F64] = f64_func_f64;
180     Table[RTLIB::LOG2_F128] = func_iPTR_i64_i64;
181     Table[RTLIB::LOG10_F32] = f32_func_f32;
182     Table[RTLIB::LOG10_F64] = f64_func_f64;
183     Table[RTLIB::LOG10_F128] = func_iPTR_i64_i64;
184     Table[RTLIB::EXP_F32] = f32_func_f32;
185     Table[RTLIB::EXP_F64] = f64_func_f64;
186     Table[RTLIB::EXP_F128] = func_iPTR_i64_i64;
187     Table[RTLIB::EXP2_F32] = f32_func_f32;
188     Table[RTLIB::EXP2_F64] = f64_func_f64;
189     Table[RTLIB::EXP2_F128] = func_iPTR_i64_i64;
190     Table[RTLIB::SIN_F32] = f32_func_f32;
191     Table[RTLIB::SIN_F64] = f64_func_f64;
192     Table[RTLIB::SIN_F128] = func_iPTR_i64_i64;
193     Table[RTLIB::COS_F32] = f32_func_f32;
194     Table[RTLIB::COS_F64] = f64_func_f64;
195     Table[RTLIB::COS_F128] = func_iPTR_i64_i64;
196     Table[RTLIB::SINCOS_F32] = func_f32_iPTR_iPTR;
197     Table[RTLIB::SINCOS_F64] = func_f64_iPTR_iPTR;
198     Table[RTLIB::SINCOS_F128] = func_i64_i64_iPTR_iPTR;
199     Table[RTLIB::POW_F32] = f32_func_f32_f32;
200     Table[RTLIB::POW_F64] = f64_func_f64_f64;
201     Table[RTLIB::POW_F128] = func_iPTR_i64_i64_i64_i64;
202     Table[RTLIB::CEIL_F32] = f32_func_f32;
203     Table[RTLIB::CEIL_F64] = f64_func_f64;
204     Table[RTLIB::CEIL_F128] = func_iPTR_i64_i64;
205     Table[RTLIB::TRUNC_F32] = f32_func_f32;
206     Table[RTLIB::TRUNC_F64] = f64_func_f64;
207     Table[RTLIB::TRUNC_F128] = func_iPTR_i64_i64;
208     Table[RTLIB::RINT_F32] = f32_func_f32;
209     Table[RTLIB::RINT_F64] = f64_func_f64;
210     Table[RTLIB::RINT_F128] = func_iPTR_i64_i64;
211     Table[RTLIB::NEARBYINT_F32] = f32_func_f32;
212     Table[RTLIB::NEARBYINT_F64] = f64_func_f64;
213     Table[RTLIB::NEARBYINT_F128] = func_iPTR_i64_i64;
214     Table[RTLIB::ROUND_F32] = f32_func_f32;
215     Table[RTLIB::ROUND_F64] = f64_func_f64;
216     Table[RTLIB::ROUND_F128] = func_iPTR_i64_i64;
217     Table[RTLIB::FLOOR_F32] = f32_func_f32;
218     Table[RTLIB::FLOOR_F64] = f64_func_f64;
219     Table[RTLIB::FLOOR_F128] = func_iPTR_i64_i64;
220     Table[RTLIB::COPYSIGN_F32] = f32_func_f32_f32;
221     Table[RTLIB::COPYSIGN_F64] = f64_func_f64_f64;
222     Table[RTLIB::COPYSIGN_F128] = func_iPTR_i64_i64_i64_i64;
223     Table[RTLIB::FMIN_F32] = f32_func_f32_f32;
224     Table[RTLIB::FMIN_F64] = f64_func_f64_f64;
225     Table[RTLIB::FMIN_F128] = func_iPTR_i64_i64_i64_i64;
226     Table[RTLIB::FMAX_F32] = f32_func_f32_f32;
227     Table[RTLIB::FMAX_F64] = f64_func_f64_f64;
228     Table[RTLIB::FMAX_F128] = func_iPTR_i64_i64_i64_i64;
229 
230     // Conversion
231     // All F80 and PPCF128 routines are unspported.
232     Table[RTLIB::FPEXT_F64_F128] = func_iPTR_f64;
233     Table[RTLIB::FPEXT_F32_F128] = func_iPTR_f32;
234     Table[RTLIB::FPEXT_F32_F64] = f64_func_f32;
235     Table[RTLIB::FPEXT_F16_F32] = f32_func_i16;
236     Table[RTLIB::FPROUND_F32_F16] = i16_func_f32;
237     Table[RTLIB::FPROUND_F64_F32] = f32_func_f64;
238     Table[RTLIB::FPROUND_F128_F32] = f32_func_i64_i64;
239     Table[RTLIB::FPROUND_F128_F64] = f64_func_i64_i64;
240     Table[RTLIB::FPTOSINT_F32_I32] = i32_func_f32;
241     Table[RTLIB::FPTOSINT_F32_I64] = i64_func_f32;
242     Table[RTLIB::FPTOSINT_F32_I128] = i64_i64_func_f32;
243     Table[RTLIB::FPTOSINT_F64_I32] = i32_func_f64;
244     Table[RTLIB::FPTOSINT_F64_I64] = i64_func_f64;
245     Table[RTLIB::FPTOSINT_F64_I128] = i64_i64_func_f64;
246     Table[RTLIB::FPTOSINT_F128_I32] = i32_func_i64_i64;
247     Table[RTLIB::FPTOSINT_F128_I64] = i64_func_i64_i64;
248     Table[RTLIB::FPTOSINT_F128_I128] = i64_i64_func_i64_i64;
249     Table[RTLIB::FPTOUINT_F32_I32] = i32_func_f32;
250     Table[RTLIB::FPTOUINT_F32_I64] = i64_func_f32;
251     Table[RTLIB::FPTOUINT_F32_I128] = i64_i64_func_f32;
252     Table[RTLIB::FPTOUINT_F64_I32] = i32_func_f64;
253     Table[RTLIB::FPTOUINT_F64_I64] = i64_func_f64;
254     Table[RTLIB::FPTOUINT_F64_I128] = i64_i64_func_f64;
255     Table[RTLIB::FPTOUINT_F128_I32] = i32_func_i64_i64;
256     Table[RTLIB::FPTOUINT_F128_I64] = i64_func_i64_i64;
257     Table[RTLIB::FPTOUINT_F128_I128] = i64_i64_func_i64_i64;
258     Table[RTLIB::SINTTOFP_I32_F32] = f32_func_i32;
259     Table[RTLIB::SINTTOFP_I32_F64] = f64_func_i32;
260     Table[RTLIB::SINTTOFP_I32_F128] = func_iPTR_i32;
261     Table[RTLIB::SINTTOFP_I64_F32] = f32_func_i64;
262     Table[RTLIB::SINTTOFP_I64_F64] = f64_func_i64;
263     Table[RTLIB::SINTTOFP_I64_F128] = func_iPTR_i64;
264     Table[RTLIB::SINTTOFP_I128_F32] = f32_func_i64_i64;
265     Table[RTLIB::SINTTOFP_I128_F64] = f64_func_i64_i64;
266     Table[RTLIB::SINTTOFP_I128_F128] = func_iPTR_i64_i64;
267     Table[RTLIB::UINTTOFP_I32_F32] = f32_func_i32;
268     Table[RTLIB::UINTTOFP_I32_F64] = f64_func_i64;
269     Table[RTLIB::UINTTOFP_I32_F128] = func_iPTR_i32;
270     Table[RTLIB::UINTTOFP_I64_F32] = f32_func_i64;
271     Table[RTLIB::UINTTOFP_I64_F64] = f64_func_i64;
272     Table[RTLIB::UINTTOFP_I64_F128] = func_iPTR_i64;
273     Table[RTLIB::UINTTOFP_I128_F32] = f32_func_i64_i64;
274     Table[RTLIB::UINTTOFP_I128_F64] = f64_func_i64_i64;
275     Table[RTLIB::UINTTOFP_I128_F128] = func_iPTR_i64_i64;
276 
277     // Comparison
278     // ALl F80 and PPCF128 routines are unsupported.
279     Table[RTLIB::OEQ_F32] = i32_func_f32_f32;
280     Table[RTLIB::OEQ_F64] = i32_func_f64_f64;
281     Table[RTLIB::OEQ_F128] = i32_func_i64_i64_i64_i64;
282     Table[RTLIB::UNE_F32] = i32_func_f32_f32;
283     Table[RTLIB::UNE_F64] = i32_func_f64_f64;
284     Table[RTLIB::UNE_F128] = i32_func_i64_i64_i64_i64;
285     Table[RTLIB::OGE_F32] = i32_func_f32_f32;
286     Table[RTLIB::OGE_F64] = i32_func_f64_f64;
287     Table[RTLIB::OGE_F128] = i32_func_i64_i64_i64_i64;
288     Table[RTLIB::OLT_F32] = i32_func_f32_f32;
289     Table[RTLIB::OLT_F64] = i32_func_f64_f64;
290     Table[RTLIB::OLT_F128] = i32_func_i64_i64_i64_i64;
291     Table[RTLIB::OLE_F32] = i32_func_f32_f32;
292     Table[RTLIB::OLE_F64] = i32_func_f64_f64;
293     Table[RTLIB::OLE_F128] = i32_func_i64_i64_i64_i64;
294     Table[RTLIB::OGT_F32] = i32_func_f32_f32;
295     Table[RTLIB::OGT_F64] = i32_func_f64_f64;
296     Table[RTLIB::OGT_F128] = i32_func_i64_i64_i64_i64;
297     Table[RTLIB::UO_F32] = i32_func_f32_f32;
298     Table[RTLIB::UO_F64] = i32_func_f64_f64;
299     Table[RTLIB::UO_F128] = i32_func_i64_i64_i64_i64;
300     // O_FXX has the weird property that it uses the same libcall name as UO_FXX
301     // This breaks our name-based lookup. Fortunately only the UO family of
302     // libcalls appears to be actually used.
303     Table[RTLIB::O_F32] = unsupported;
304     Table[RTLIB::O_F64] = unsupported;
305     Table[RTLIB::O_F128] = unsupported;
306 
307     // Memory
308     Table[RTLIB::MEMCPY] = iPTR_func_iPTR_iPTR_iPTR;
309     Table[RTLIB::MEMSET] = iPTR_func_iPTR_i32_iPTR;
310     Table[RTLIB::MEMMOVE] = iPTR_func_iPTR_iPTR_iPTR;
311 
312     // __stack_chk_fail
313     Table[RTLIB::STACKPROTECTOR_CHECK_FAIL] = func;
314 
315     // Element-wise Atomic memory
316     // TODO: Fix these when we implement atomic support
317     Table[RTLIB::MEMCPY_ELEMENT_UNORDERED_ATOMIC_1] = unsupported;
318     Table[RTLIB::MEMCPY_ELEMENT_UNORDERED_ATOMIC_2] = unsupported;
319     Table[RTLIB::MEMCPY_ELEMENT_UNORDERED_ATOMIC_4] = unsupported;
320     Table[RTLIB::MEMCPY_ELEMENT_UNORDERED_ATOMIC_8] = unsupported;
321     Table[RTLIB::MEMCPY_ELEMENT_UNORDERED_ATOMIC_16] = unsupported;
322     Table[RTLIB::MEMMOVE_ELEMENT_UNORDERED_ATOMIC_1] = unsupported;
323     Table[RTLIB::MEMMOVE_ELEMENT_UNORDERED_ATOMIC_2] = unsupported;
324     Table[RTLIB::MEMMOVE_ELEMENT_UNORDERED_ATOMIC_4] = unsupported;
325     Table[RTLIB::MEMMOVE_ELEMENT_UNORDERED_ATOMIC_8] = unsupported;
326     Table[RTLIB::MEMMOVE_ELEMENT_UNORDERED_ATOMIC_16] = unsupported;
327 
328     Table[RTLIB::MEMSET_ELEMENT_UNORDERED_ATOMIC_1] = unsupported;
329     Table[RTLIB::MEMSET_ELEMENT_UNORDERED_ATOMIC_2] = unsupported;
330     Table[RTLIB::MEMSET_ELEMENT_UNORDERED_ATOMIC_4] = unsupported;
331     Table[RTLIB::MEMSET_ELEMENT_UNORDERED_ATOMIC_8] = unsupported;
332     Table[RTLIB::MEMSET_ELEMENT_UNORDERED_ATOMIC_16] = unsupported;
333 
334     // Atomic '__sync_*' libcalls.
335     // TODO: Fix these when we implement atomic support
336     Table[RTLIB::SYNC_VAL_COMPARE_AND_SWAP_1] = unsupported;
337     Table[RTLIB::SYNC_VAL_COMPARE_AND_SWAP_2] = unsupported;
338     Table[RTLIB::SYNC_VAL_COMPARE_AND_SWAP_4] = unsupported;
339     Table[RTLIB::SYNC_VAL_COMPARE_AND_SWAP_8] = unsupported;
340     Table[RTLIB::SYNC_VAL_COMPARE_AND_SWAP_16] = unsupported;
341     Table[RTLIB::SYNC_LOCK_TEST_AND_SET_1] = unsupported;
342     Table[RTLIB::SYNC_LOCK_TEST_AND_SET_2] = unsupported;
343     Table[RTLIB::SYNC_LOCK_TEST_AND_SET_4] = unsupported;
344     Table[RTLIB::SYNC_LOCK_TEST_AND_SET_8] = unsupported;
345     Table[RTLIB::SYNC_LOCK_TEST_AND_SET_16] = unsupported;
346     Table[RTLIB::SYNC_FETCH_AND_ADD_1] = unsupported;
347     Table[RTLIB::SYNC_FETCH_AND_ADD_2] = unsupported;
348     Table[RTLIB::SYNC_FETCH_AND_ADD_4] = unsupported;
349     Table[RTLIB::SYNC_FETCH_AND_ADD_8] = unsupported;
350     Table[RTLIB::SYNC_FETCH_AND_ADD_16] = unsupported;
351     Table[RTLIB::SYNC_FETCH_AND_SUB_1] = unsupported;
352     Table[RTLIB::SYNC_FETCH_AND_SUB_2] = unsupported;
353     Table[RTLIB::SYNC_FETCH_AND_SUB_4] = unsupported;
354     Table[RTLIB::SYNC_FETCH_AND_SUB_8] = unsupported;
355     Table[RTLIB::SYNC_FETCH_AND_SUB_16] = unsupported;
356     Table[RTLIB::SYNC_FETCH_AND_AND_1] = unsupported;
357     Table[RTLIB::SYNC_FETCH_AND_AND_2] = unsupported;
358     Table[RTLIB::SYNC_FETCH_AND_AND_4] = unsupported;
359     Table[RTLIB::SYNC_FETCH_AND_AND_8] = unsupported;
360     Table[RTLIB::SYNC_FETCH_AND_AND_16] = unsupported;
361     Table[RTLIB::SYNC_FETCH_AND_OR_1] = unsupported;
362     Table[RTLIB::SYNC_FETCH_AND_OR_2] = unsupported;
363     Table[RTLIB::SYNC_FETCH_AND_OR_4] = unsupported;
364     Table[RTLIB::SYNC_FETCH_AND_OR_8] = unsupported;
365     Table[RTLIB::SYNC_FETCH_AND_OR_16] = unsupported;
366     Table[RTLIB::SYNC_FETCH_AND_XOR_1] = unsupported;
367     Table[RTLIB::SYNC_FETCH_AND_XOR_2] = unsupported;
368     Table[RTLIB::SYNC_FETCH_AND_XOR_4] = unsupported;
369     Table[RTLIB::SYNC_FETCH_AND_XOR_8] = unsupported;
370     Table[RTLIB::SYNC_FETCH_AND_XOR_16] = unsupported;
371     Table[RTLIB::SYNC_FETCH_AND_NAND_1] = unsupported;
372     Table[RTLIB::SYNC_FETCH_AND_NAND_2] = unsupported;
373     Table[RTLIB::SYNC_FETCH_AND_NAND_4] = unsupported;
374     Table[RTLIB::SYNC_FETCH_AND_NAND_8] = unsupported;
375     Table[RTLIB::SYNC_FETCH_AND_NAND_16] = unsupported;
376     Table[RTLIB::SYNC_FETCH_AND_MAX_1] = unsupported;
377     Table[RTLIB::SYNC_FETCH_AND_MAX_2] = unsupported;
378     Table[RTLIB::SYNC_FETCH_AND_MAX_4] = unsupported;
379     Table[RTLIB::SYNC_FETCH_AND_MAX_8] = unsupported;
380     Table[RTLIB::SYNC_FETCH_AND_MAX_16] = unsupported;
381     Table[RTLIB::SYNC_FETCH_AND_UMAX_1] = unsupported;
382     Table[RTLIB::SYNC_FETCH_AND_UMAX_2] = unsupported;
383     Table[RTLIB::SYNC_FETCH_AND_UMAX_4] = unsupported;
384     Table[RTLIB::SYNC_FETCH_AND_UMAX_8] = unsupported;
385     Table[RTLIB::SYNC_FETCH_AND_UMAX_16] = unsupported;
386     Table[RTLIB::SYNC_FETCH_AND_MIN_1] = unsupported;
387     Table[RTLIB::SYNC_FETCH_AND_MIN_2] = unsupported;
388     Table[RTLIB::SYNC_FETCH_AND_MIN_4] = unsupported;
389     Table[RTLIB::SYNC_FETCH_AND_MIN_8] = unsupported;
390     Table[RTLIB::SYNC_FETCH_AND_MIN_16] = unsupported;
391     Table[RTLIB::SYNC_FETCH_AND_UMIN_1] = unsupported;
392     Table[RTLIB::SYNC_FETCH_AND_UMIN_2] = unsupported;
393     Table[RTLIB::SYNC_FETCH_AND_UMIN_4] = unsupported;
394     Table[RTLIB::SYNC_FETCH_AND_UMIN_8] = unsupported;
395     Table[RTLIB::SYNC_FETCH_AND_UMIN_16] = unsupported;
396 
397     // Atomic '__atomic_*' libcalls.
398     // TODO: Fix these when we implement atomic support
399     Table[RTLIB::ATOMIC_LOAD] = unsupported;
400     Table[RTLIB::ATOMIC_LOAD_1] = unsupported;
401     Table[RTLIB::ATOMIC_LOAD_2] = unsupported;
402     Table[RTLIB::ATOMIC_LOAD_4] = unsupported;
403     Table[RTLIB::ATOMIC_LOAD_8] = unsupported;
404     Table[RTLIB::ATOMIC_LOAD_16] = unsupported;
405 
406     Table[RTLIB::ATOMIC_STORE] = unsupported;
407     Table[RTLIB::ATOMIC_STORE_1] = unsupported;
408     Table[RTLIB::ATOMIC_STORE_2] = unsupported;
409     Table[RTLIB::ATOMIC_STORE_4] = unsupported;
410     Table[RTLIB::ATOMIC_STORE_8] = unsupported;
411     Table[RTLIB::ATOMIC_STORE_16] = unsupported;
412 
413     Table[RTLIB::ATOMIC_EXCHANGE] = unsupported;
414     Table[RTLIB::ATOMIC_EXCHANGE_1] = unsupported;
415     Table[RTLIB::ATOMIC_EXCHANGE_2] = unsupported;
416     Table[RTLIB::ATOMIC_EXCHANGE_4] = unsupported;
417     Table[RTLIB::ATOMIC_EXCHANGE_8] = unsupported;
418     Table[RTLIB::ATOMIC_EXCHANGE_16] = unsupported;
419 
420     Table[RTLIB::ATOMIC_COMPARE_EXCHANGE] = unsupported;
421     Table[RTLIB::ATOMIC_COMPARE_EXCHANGE_1] = unsupported;
422     Table[RTLIB::ATOMIC_COMPARE_EXCHANGE_2] = unsupported;
423     Table[RTLIB::ATOMIC_COMPARE_EXCHANGE_4] = unsupported;
424     Table[RTLIB::ATOMIC_COMPARE_EXCHANGE_8] = unsupported;
425     Table[RTLIB::ATOMIC_COMPARE_EXCHANGE_16] = unsupported;
426 
427     Table[RTLIB::ATOMIC_FETCH_ADD_1] = unsupported;
428     Table[RTLIB::ATOMIC_FETCH_ADD_2] = unsupported;
429     Table[RTLIB::ATOMIC_FETCH_ADD_4] = unsupported;
430     Table[RTLIB::ATOMIC_FETCH_ADD_8] = unsupported;
431     Table[RTLIB::ATOMIC_FETCH_ADD_16] = unsupported;
432 
433     Table[RTLIB::ATOMIC_FETCH_SUB_1] = unsupported;
434     Table[RTLIB::ATOMIC_FETCH_SUB_2] = unsupported;
435     Table[RTLIB::ATOMIC_FETCH_SUB_4] = unsupported;
436     Table[RTLIB::ATOMIC_FETCH_SUB_8] = unsupported;
437     Table[RTLIB::ATOMIC_FETCH_SUB_16] = unsupported;
438 
439     Table[RTLIB::ATOMIC_FETCH_AND_1] = unsupported;
440     Table[RTLIB::ATOMIC_FETCH_AND_2] = unsupported;
441     Table[RTLIB::ATOMIC_FETCH_AND_4] = unsupported;
442     Table[RTLIB::ATOMIC_FETCH_AND_8] = unsupported;
443     Table[RTLIB::ATOMIC_FETCH_AND_16] = unsupported;
444 
445     Table[RTLIB::ATOMIC_FETCH_OR_1] = unsupported;
446     Table[RTLIB::ATOMIC_FETCH_OR_2] = unsupported;
447     Table[RTLIB::ATOMIC_FETCH_OR_4] = unsupported;
448     Table[RTLIB::ATOMIC_FETCH_OR_8] = unsupported;
449     Table[RTLIB::ATOMIC_FETCH_OR_16] = unsupported;
450 
451     Table[RTLIB::ATOMIC_FETCH_XOR_1] = unsupported;
452     Table[RTLIB::ATOMIC_FETCH_XOR_2] = unsupported;
453     Table[RTLIB::ATOMIC_FETCH_XOR_4] = unsupported;
454     Table[RTLIB::ATOMIC_FETCH_XOR_8] = unsupported;
455     Table[RTLIB::ATOMIC_FETCH_XOR_16] = unsupported;
456 
457     Table[RTLIB::ATOMIC_FETCH_NAND_1] = unsupported;
458     Table[RTLIB::ATOMIC_FETCH_NAND_2] = unsupported;
459     Table[RTLIB::ATOMIC_FETCH_NAND_4] = unsupported;
460     Table[RTLIB::ATOMIC_FETCH_NAND_8] = unsupported;
461     Table[RTLIB::ATOMIC_FETCH_NAND_16] = unsupported;
462   }
463 };
464 
465 ManagedStatic<RuntimeLibcallSignatureTable> RuntimeLibcallSignatures;
466 
467 // Maps libcall names to their RTLIB::Libcall number. Builds the map in a
468 // constructor for use with ManagedStatic
469 struct StaticLibcallNameMap {
470   StringMap<RTLIB::Libcall> Map;
471   StaticLibcallNameMap() {
472     static const std::pair<const char *, RTLIB::Libcall> NameLibcalls[] = {
473 #define HANDLE_LIBCALL(code, name) {(const char *)name, RTLIB::code},
474 #include "llvm/IR/RuntimeLibcalls.def"
475 #undef HANDLE_LIBCALL
476     };
477     for (const auto &NameLibcall : NameLibcalls) {
478       if (NameLibcall.first != nullptr &&
479           RuntimeLibcallSignatures->Table[NameLibcall.second] != unsupported) {
480         assert(Map.find(NameLibcall.first) == Map.end() &&
481                "duplicate libcall names in name map");
482         Map[NameLibcall.first] = NameLibcall.second;
483       }
484     }
485   }
486 };
487 
488 } // end anonymous namespace
489 
490 void llvm::getLibcallSignature(const WebAssemblySubtarget &Subtarget,
491                                RTLIB::Libcall LC,
492                                SmallVectorImpl<wasm::ValType> &Rets,
493                                SmallVectorImpl<wasm::ValType> &Params) {
494   assert(Rets.empty());
495   assert(Params.empty());
496 
497   wasm::ValType PtrTy =
498       Subtarget.hasAddr64() ? wasm::ValType::I64 : wasm::ValType::I32;
499 
500   auto &Table = RuntimeLibcallSignatures->Table;
501   switch (Table[LC]) {
502   case func:
503     break;
504   case f32_func_f32:
505     Rets.push_back(wasm::ValType::F32);
506     Params.push_back(wasm::ValType::F32);
507     break;
508   case f32_func_f64:
509     Rets.push_back(wasm::ValType::F32);
510     Params.push_back(wasm::ValType::F64);
511     break;
512   case f32_func_i32:
513     Rets.push_back(wasm::ValType::F32);
514     Params.push_back(wasm::ValType::I32);
515     break;
516   case f32_func_i64:
517     Rets.push_back(wasm::ValType::F32);
518     Params.push_back(wasm::ValType::I64);
519     break;
520   case f32_func_i16:
521     Rets.push_back(wasm::ValType::F32);
522     Params.push_back(wasm::ValType::I32);
523     break;
524   case f64_func_f32:
525     Rets.push_back(wasm::ValType::F64);
526     Params.push_back(wasm::ValType::F32);
527     break;
528   case f64_func_f64:
529     Rets.push_back(wasm::ValType::F64);
530     Params.push_back(wasm::ValType::F64);
531     break;
532   case f64_func_i32:
533     Rets.push_back(wasm::ValType::F64);
534     Params.push_back(wasm::ValType::I32);
535     break;
536   case f64_func_i64:
537     Rets.push_back(wasm::ValType::F64);
538     Params.push_back(wasm::ValType::I64);
539     break;
540   case i32_func_f32:
541     Rets.push_back(wasm::ValType::I32);
542     Params.push_back(wasm::ValType::F32);
543     break;
544   case i32_func_f64:
545     Rets.push_back(wasm::ValType::I32);
546     Params.push_back(wasm::ValType::F64);
547     break;
548   case i32_func_i32:
549     Rets.push_back(wasm::ValType::I32);
550     Params.push_back(wasm::ValType::I32);
551     break;
552   case i64_func_f32:
553     Rets.push_back(wasm::ValType::I64);
554     Params.push_back(wasm::ValType::F32);
555     break;
556   case i64_func_f64:
557     Rets.push_back(wasm::ValType::I64);
558     Params.push_back(wasm::ValType::F64);
559     break;
560   case i64_func_i64:
561     Rets.push_back(wasm::ValType::I64);
562     Params.push_back(wasm::ValType::I64);
563     break;
564   case f32_func_f32_f32:
565     Rets.push_back(wasm::ValType::F32);
566     Params.push_back(wasm::ValType::F32);
567     Params.push_back(wasm::ValType::F32);
568     break;
569   case f32_func_f32_i32:
570     Rets.push_back(wasm::ValType::F32);
571     Params.push_back(wasm::ValType::F32);
572     Params.push_back(wasm::ValType::I32);
573     break;
574   case f32_func_i64_i64:
575     Rets.push_back(wasm::ValType::F32);
576     Params.push_back(wasm::ValType::I64);
577     Params.push_back(wasm::ValType::I64);
578     break;
579   case f64_func_f64_f64:
580     Rets.push_back(wasm::ValType::F64);
581     Params.push_back(wasm::ValType::F64);
582     Params.push_back(wasm::ValType::F64);
583     break;
584   case f64_func_f64_i32:
585     Rets.push_back(wasm::ValType::F64);
586     Params.push_back(wasm::ValType::F64);
587     Params.push_back(wasm::ValType::I32);
588     break;
589   case f64_func_i64_i64:
590     Rets.push_back(wasm::ValType::F64);
591     Params.push_back(wasm::ValType::I64);
592     Params.push_back(wasm::ValType::I64);
593     break;
594   case i16_func_f32:
595     Rets.push_back(wasm::ValType::I32);
596     Params.push_back(wasm::ValType::F32);
597     break;
598   case i8_func_i8_i8:
599     Rets.push_back(wasm::ValType::I32);
600     Params.push_back(wasm::ValType::I32);
601     Params.push_back(wasm::ValType::I32);
602     break;
603   case func_f32_iPTR_iPTR:
604     Params.push_back(wasm::ValType::F32);
605     Params.push_back(PtrTy);
606     Params.push_back(PtrTy);
607     break;
608   case func_f64_iPTR_iPTR:
609     Params.push_back(wasm::ValType::F64);
610     Params.push_back(PtrTy);
611     Params.push_back(PtrTy);
612     break;
613   case i16_func_i16_i16:
614     Rets.push_back(wasm::ValType::I32);
615     Params.push_back(wasm::ValType::I32);
616     Params.push_back(wasm::ValType::I32);
617     break;
618   case i32_func_f32_f32:
619     Rets.push_back(wasm::ValType::I32);
620     Params.push_back(wasm::ValType::F32);
621     Params.push_back(wasm::ValType::F32);
622     break;
623   case i32_func_f64_f64:
624     Rets.push_back(wasm::ValType::I32);
625     Params.push_back(wasm::ValType::F64);
626     Params.push_back(wasm::ValType::F64);
627     break;
628   case i32_func_i32_i32:
629     Rets.push_back(wasm::ValType::I32);
630     Params.push_back(wasm::ValType::I32);
631     Params.push_back(wasm::ValType::I32);
632     break;
633   case i32_func_i32_i32_iPTR:
634     Rets.push_back(wasm::ValType::I32);
635     Params.push_back(wasm::ValType::I32);
636     Params.push_back(wasm::ValType::I32);
637     Params.push_back(PtrTy);
638     break;
639   case i64_func_i64_i64:
640     Rets.push_back(wasm::ValType::I64);
641     Params.push_back(wasm::ValType::I64);
642     Params.push_back(wasm::ValType::I64);
643     break;
644   case i64_func_i64_i64_iPTR:
645     Rets.push_back(wasm::ValType::I64);
646     Params.push_back(wasm::ValType::I64);
647     Params.push_back(wasm::ValType::I64);
648     Params.push_back(PtrTy);
649     break;
650   case i64_i64_func_f32:
651 #if 0 // TODO: Enable this when wasm gets multiple-return-value support.
652     Rets.push_back(wasm::ValType::I64);
653     Rets.push_back(wasm::ValType::I64);
654 #else
655     Params.push_back(PtrTy);
656 #endif
657     Params.push_back(wasm::ValType::F32);
658     break;
659   case i64_i64_func_f64:
660 #if 0 // TODO: Enable this when wasm gets multiple-return-value support.
661     Rets.push_back(wasm::ValType::I64);
662     Rets.push_back(wasm::ValType::I64);
663 #else
664     Params.push_back(PtrTy);
665 #endif
666     Params.push_back(wasm::ValType::F64);
667     break;
668   case i16_i16_func_i16_i16:
669 #if 0 // TODO: Enable this when wasm gets multiple-return-value support.
670     Rets.push_back(wasm::ValType::I32);
671     Rets.push_back(wasm::ValType::I32);
672 #else
673     Params.push_back(PtrTy);
674 #endif
675     Params.push_back(wasm::ValType::I32);
676     Params.push_back(wasm::ValType::I32);
677     break;
678   case i32_i32_func_i32_i32:
679 #if 0 // TODO: Enable this when wasm gets multiple-return-value support.
680     Rets.push_back(wasm::ValType::I32);
681     Rets.push_back(wasm::ValType::I32);
682 #else
683     Params.push_back(PtrTy);
684 #endif
685     Params.push_back(wasm::ValType::I32);
686     Params.push_back(wasm::ValType::I32);
687     break;
688   case i64_i64_func_i64_i64:
689 #if 0 // TODO: Enable this when wasm gets multiple-return-value support.
690     Rets.push_back(wasm::ValType::I64);
691     Rets.push_back(wasm::ValType::I64);
692 #else
693     Params.push_back(PtrTy);
694 #endif
695     Params.push_back(wasm::ValType::I64);
696     Params.push_back(wasm::ValType::I64);
697     break;
698   case i64_i64_func_i64_i64_i64_i64:
699 #if 0 // TODO: Enable this when wasm gets multiple-return-value support.
700     Rets.push_back(wasm::ValType::I64);
701     Rets.push_back(wasm::ValType::I64);
702 #else
703     Params.push_back(PtrTy);
704 #endif
705     Params.push_back(wasm::ValType::I64);
706     Params.push_back(wasm::ValType::I64);
707     Params.push_back(wasm::ValType::I64);
708     Params.push_back(wasm::ValType::I64);
709     break;
710   case i64_i64_func_i64_i64_i64_i64_iPTR:
711 #if 0 // TODO: Enable this when wasm gets multiple-return-value support.
712     Rets.push_back(wasm::ValType::I64);
713     Rets.push_back(wasm::ValType::I64);
714 #else
715     Params.push_back(PtrTy);
716 #endif
717     Params.push_back(wasm::ValType::I64);
718     Params.push_back(wasm::ValType::I64);
719     Params.push_back(wasm::ValType::I64);
720     Params.push_back(wasm::ValType::I64);
721     Params.push_back(PtrTy);
722     break;
723   case i64_i64_i64_i64_func_i64_i64_i64_i64:
724 #if 0 // TODO: Enable this when wasm gets multiple-return-value support.
725     Rets.push_back(wasm::ValType::I64);
726     Rets.push_back(wasm::ValType::I64);
727     Rets.push_back(wasm::ValType::I64);
728     Rets.push_back(wasm::ValType::I64);
729 #else
730     Params.push_back(PtrTy);
731 #endif
732     Params.push_back(wasm::ValType::I64);
733     Params.push_back(wasm::ValType::I64);
734     Params.push_back(wasm::ValType::I64);
735     Params.push_back(wasm::ValType::I64);
736     break;
737   case i64_i64_func_i64_i64_i32:
738 #if 0 // TODO: Enable this when wasm gets multiple-return-value support.
739     Rets.push_back(wasm::ValType::I64);
740     Rets.push_back(wasm::ValType::I64);
741     Rets.push_back(wasm::ValType::I64);
742     Rets.push_back(wasm::ValType::I64);
743 #else
744     Params.push_back(PtrTy);
745 #endif
746     Params.push_back(wasm::ValType::I64);
747     Params.push_back(wasm::ValType::I64);
748     Params.push_back(wasm::ValType::I32);
749     break;
750   case iPTR_func_iPTR_i32_iPTR:
751     Rets.push_back(PtrTy);
752     Params.push_back(PtrTy);
753     Params.push_back(wasm::ValType::I32);
754     Params.push_back(PtrTy);
755     break;
756   case iPTR_func_iPTR_iPTR_iPTR:
757     Rets.push_back(PtrTy);
758     Params.push_back(PtrTy);
759     Params.push_back(PtrTy);
760     Params.push_back(PtrTy);
761     break;
762   case f32_func_f32_f32_f32:
763     Rets.push_back(wasm::ValType::F32);
764     Params.push_back(wasm::ValType::F32);
765     Params.push_back(wasm::ValType::F32);
766     Params.push_back(wasm::ValType::F32);
767     break;
768   case f64_func_f64_f64_f64:
769     Rets.push_back(wasm::ValType::F64);
770     Params.push_back(wasm::ValType::F64);
771     Params.push_back(wasm::ValType::F64);
772     Params.push_back(wasm::ValType::F64);
773     break;
774   case func_i64_i64_iPTR_iPTR:
775     Params.push_back(wasm::ValType::I64);
776     Params.push_back(wasm::ValType::I64);
777     Params.push_back(PtrTy);
778     Params.push_back(PtrTy);
779     break;
780   case func_iPTR_f32:
781     Params.push_back(PtrTy);
782     Params.push_back(wasm::ValType::F32);
783     break;
784   case func_iPTR_f64:
785     Params.push_back(PtrTy);
786     Params.push_back(wasm::ValType::F64);
787     break;
788   case func_iPTR_i32:
789     Params.push_back(PtrTy);
790     Params.push_back(wasm::ValType::I32);
791     break;
792   case func_iPTR_i64:
793     Params.push_back(PtrTy);
794     Params.push_back(wasm::ValType::I64);
795     break;
796   case func_iPTR_i64_i64:
797     Params.push_back(PtrTy);
798     Params.push_back(wasm::ValType::I64);
799     Params.push_back(wasm::ValType::I64);
800     break;
801   case func_iPTR_i64_i64_i64_i64:
802     Params.push_back(PtrTy);
803     Params.push_back(wasm::ValType::I64);
804     Params.push_back(wasm::ValType::I64);
805     Params.push_back(wasm::ValType::I64);
806     Params.push_back(wasm::ValType::I64);
807     break;
808   case func_iPTR_i64_i64_i64_i64_i64_i64:
809     Params.push_back(PtrTy);
810     Params.push_back(wasm::ValType::I64);
811     Params.push_back(wasm::ValType::I64);
812     Params.push_back(wasm::ValType::I64);
813     Params.push_back(wasm::ValType::I64);
814     Params.push_back(wasm::ValType::I64);
815     Params.push_back(wasm::ValType::I64);
816     break;
817   case i32_func_i64_i64:
818     Rets.push_back(wasm::ValType::I32);
819     Params.push_back(wasm::ValType::I64);
820     Params.push_back(wasm::ValType::I64);
821     break;
822   case i32_func_i64_i64_i64_i64:
823     Rets.push_back(wasm::ValType::I32);
824     Params.push_back(wasm::ValType::I64);
825     Params.push_back(wasm::ValType::I64);
826     Params.push_back(wasm::ValType::I64);
827     Params.push_back(wasm::ValType::I64);
828     break;
829   case unsupported:
830     llvm_unreachable("unsupported runtime library signature");
831   }
832 }
833 
834 static ManagedStatic<StaticLibcallNameMap> LibcallNameMap;
835 // TODO: If the RTLIB::Libcall-taking flavor of GetSignature remains unsed
836 // other than here, just roll its logic into this version.
837 void llvm::getLibcallSignature(const WebAssemblySubtarget &Subtarget,
838                                const char *Name,
839                                SmallVectorImpl<wasm::ValType> &Rets,
840                                SmallVectorImpl<wasm::ValType> &Params) {
841   auto &Map = LibcallNameMap->Map;
842   auto Val = Map.find(Name);
843 #ifndef NDEBUG
844   if (Val == Map.end()) {
845     errs() << "runtime library name: " << Name << "\n";
846     llvm_unreachable("unexpected runtime library name");
847   }
848 #endif
849   return getLibcallSignature(Subtarget, Val->second, Rets, Params);
850 }
851