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