ffi 1.13.1 → 1.14.2

Sign up to get free protection for your applications and to get access to all the features.
Files changed (101) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +41 -0
  3. data/README.md +10 -2
  4. data/Rakefile +7 -0
  5. data/ext/ffi_c/AbstractMemory.c +24 -25
  6. data/ext/ffi_c/Buffer.c +2 -7
  7. data/ext/ffi_c/Call.c +2 -7
  8. data/ext/ffi_c/ClosurePool.c +64 -11
  9. data/ext/ffi_c/ClosurePool.h +3 -1
  10. data/ext/ffi_c/DynamicLibrary.c +1 -6
  11. data/ext/ffi_c/Function.c +8 -13
  12. data/ext/ffi_c/FunctionInfo.c +2 -6
  13. data/ext/ffi_c/LastError.c +2 -6
  14. data/ext/ffi_c/MemoryPointer.c +2 -7
  15. data/ext/ffi_c/MethodHandle.c +4 -8
  16. data/ext/ffi_c/Platform.c +2 -7
  17. data/ext/ffi_c/Pointer.c +24 -25
  18. data/ext/ffi_c/Struct.c +3 -6
  19. data/ext/ffi_c/StructByValue.c +2 -7
  20. data/ext/ffi_c/StructLayout.c +2 -5
  21. data/ext/ffi_c/Thread.c +0 -5
  22. data/ext/ffi_c/Thread.h +1 -6
  23. data/ext/ffi_c/Type.c +1 -1
  24. data/ext/ffi_c/Variadic.c +2 -7
  25. data/ext/ffi_c/extconf.rb +17 -4
  26. data/ext/ffi_c/libffi/.travis/bfin-sim.exp +1 -1
  27. data/ext/ffi_c/libffi/.travis/m32r-sim.exp +1 -1
  28. data/ext/ffi_c/libffi/.travis/moxie-sim.exp +1 -1
  29. data/ext/ffi_c/libffi/.travis/or1k-sim.exp +1 -1
  30. data/ext/ffi_c/libffi/.travis/powerpc-eabisim.exp +1 -1
  31. data/ext/ffi_c/libffi/.travis/wine-sim.exp +1 -1
  32. data/ext/ffi_c/libffi/Makefile.am +48 -58
  33. data/ext/ffi_c/libffi/README.md +4 -0
  34. data/ext/ffi_c/libffi/config.guess +552 -331
  35. data/ext/ffi_c/libffi/config.sub +1321 -1306
  36. data/ext/ffi_c/libffi/configure.ac +6 -1
  37. data/ext/ffi_c/libffi/configure.host +32 -20
  38. data/ext/ffi_c/libffi/doc/Makefile.am +3 -0
  39. data/ext/ffi_c/libffi/doc/libffi.texi +997 -0
  40. data/ext/ffi_c/libffi/doc/version.texi +4 -0
  41. data/ext/ffi_c/libffi/generate-darwin-source-and-headers.py +1 -1
  42. data/ext/ffi_c/libffi/msvcc.sh +11 -11
  43. data/ext/ffi_c/libffi/src/aarch64/ffi.c +45 -35
  44. data/ext/ffi_c/libffi/src/aarch64/ffitarget.h +10 -5
  45. data/ext/ffi_c/libffi/src/aarch64/internal.h +1 -0
  46. data/ext/ffi_c/libffi/src/aarch64/sysv.S +1 -1
  47. data/ext/ffi_c/libffi/src/aarch64/win64_armasm.S +1 -1
  48. data/ext/ffi_c/libffi/src/arm/ffi.c +22 -0
  49. data/ext/ffi_c/libffi/src/arm/sysv.S +4 -4
  50. data/ext/ffi_c/libffi/src/closures.c +23 -6
  51. data/ext/ffi_c/libffi/src/csky/ffi.c +395 -0
  52. data/ext/ffi_c/libffi/src/csky/ffitarget.h +63 -0
  53. data/ext/ffi_c/libffi/src/csky/sysv.S +371 -0
  54. data/ext/ffi_c/libffi/src/dlmalloc.c +1 -1
  55. data/ext/ffi_c/libffi/src/kvx/asm.h +5 -0
  56. data/ext/ffi_c/libffi/src/kvx/ffi.c +273 -0
  57. data/ext/ffi_c/libffi/src/kvx/ffitarget.h +75 -0
  58. data/ext/ffi_c/libffi/src/kvx/sysv.S +127 -0
  59. data/ext/ffi_c/libffi/src/mips/ffi.c +5 -1
  60. data/ext/ffi_c/libffi/src/mips/ffitarget.h +1 -1
  61. data/ext/ffi_c/libffi/src/powerpc/ffi_darwin.c +13 -1
  62. data/ext/ffi_c/libffi/src/powerpc/ffi_powerpc.h +1 -1
  63. data/ext/ffi_c/libffi/src/powerpc/linux64.S +8 -0
  64. data/ext/ffi_c/libffi/src/powerpc/linux64_closure.S +13 -1
  65. data/ext/ffi_c/libffi/src/prep_cif.c +1 -1
  66. data/ext/ffi_c/libffi/src/x86/ffi.c +8 -2
  67. data/ext/ffi_c/libffi/src/x86/ffi64.c +7 -0
  68. data/ext/ffi_c/libffi/src/x86/ffiw64.c +5 -0
  69. data/ext/ffi_c/libffi/src/x86/sysv.S +2 -2
  70. data/ext/ffi_c/libffi/src/x86/unix64.S +1 -2
  71. data/ext/ffi_c/libffi/src/x86/win64.S +3 -2
  72. data/ext/ffi_c/libffi/src/x86/win64_intel.S +3 -2
  73. data/ext/ffi_c/libffi/testsuite/lib/libffi.exp +22 -2
  74. data/ext/ffi_c/libffi/testsuite/libffi.bhaible/test-call.c +4 -4
  75. data/ext/ffi_c/libffi/testsuite/libffi.bhaible/test-callback.c +2 -2
  76. data/ext/ffi_c/libffi/testsuite/libffi.closures/huge_struct.c +2 -0
  77. data/ffi.gemspec +1 -1
  78. data/lib/ffi.rb +2 -2
  79. data/lib/ffi/abstract_memory.rb +44 -0
  80. data/lib/ffi/autopointer.rb +1 -1
  81. data/lib/ffi/ffi.rb +1 -0
  82. data/lib/ffi/io.rb +3 -3
  83. data/lib/ffi/library.rb +1 -1
  84. data/lib/ffi/managedstruct.rb +2 -2
  85. data/lib/ffi/platform.rb +15 -6
  86. data/lib/ffi/platform/aarch64-darwin/types.conf +130 -0
  87. data/lib/ffi/platform/aarch64-openbsd/types.conf +134 -0
  88. data/lib/ffi/platform/x86_64-haiku/types.conf +117 -0
  89. data/lib/ffi/platform/x86_64-msys/types.conf +119 -0
  90. data/lib/ffi/pointer.rb +2 -2
  91. data/lib/ffi/variadic.rb +1 -1
  92. data/lib/ffi/version.rb +1 -1
  93. metadata +15 -11
  94. data/.appveyor.yml +0 -30
  95. data/.github/workflows/ci.yml +0 -64
  96. data/.gitignore +0 -25
  97. data/.gitmodules +0 -4
  98. data/.travis.yml +0 -58
  99. data/.yardopts +0 -5
  100. data/ext/ffi_c/win32/stdbool.h +0 -8
  101. data/ext/ffi_c/win32/stdint.h +0 -201
@@ -0,0 +1,371 @@
1
+ /* -----------------------------------------------------------------------
2
+ sysv.S
3
+
4
+ CSKY Foreign Function Interface
5
+
6
+ Permission is hereby granted, free of charge, to any person obtaining
7
+ a copy of this software and associated documentation files (the
8
+ ``Software''), to deal in the Software without restriction, including
9
+ without limitation the rights to use, copy, modify, merge, publish,
10
+ distribute, sublicense, and/or sell copies of the Software, and to
11
+ permit persons to whom the Software is furnished to do so, subject to
12
+ the following conditions:
13
+
14
+ The above copyright notice and this permission notice shall be included
15
+ in all copies or substantial portions of the Software.
16
+
17
+ THE SOFTWARE IS PROVIDED ``AS IS'', WITHOUT WARRANTY OF ANY KIND,
18
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
19
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
20
+ NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
21
+ HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
22
+ WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
23
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
24
+ DEALINGS IN THE SOFTWARE.
25
+ ----------------------------------------------------------------------- */
26
+
27
+ #define LIBFFI_ASM
28
+ #include <fficonfig.h>
29
+ #include <ffi.h>
30
+
31
+ .macro CSKY_FUNC_START name
32
+ .text
33
+ .align 2
34
+ .globl \name
35
+ .type \name, @function
36
+ \name:
37
+ .endm
38
+
39
+ #ifdef __CSKYABIV2__
40
+
41
+ /*
42
+ * a0: fn
43
+ * a1: &ecif
44
+ * a2: cif->bytes
45
+ * a3: fig->flags
46
+ * sp+0: ecif.rvalue
47
+ */
48
+ CSKY_FUNC_START ffi_call_SYSV
49
+ /* Save registers */
50
+ .cfi_startproc
51
+ subi sp, 28
52
+ .cfi_def_cfa_offset 28
53
+ stw a0, (sp, 0x0)
54
+ .cfi_offset 0, -28
55
+ stw a1, (sp, 0x4)
56
+ .cfi_offset 1, -24
57
+ stw a2, (sp, 0x8)
58
+ .cfi_offset 2, -20
59
+ stw a3, (sp, 0xC)
60
+ .cfi_offset 3, -16
61
+ stw l0, (sp, 0x10)
62
+ .cfi_offset 4, -12
63
+ stw l1, (sp, 0x14)
64
+ .cfi_offset 5, -8
65
+ stw lr, (sp, 0x18)
66
+ .cfi_offset 15, -4
67
+
68
+ mov l0, sp
69
+ .cfi_def_cfa_register 4
70
+
71
+ /* Make room for all of the new args. */
72
+ subu sp, sp, a2
73
+
74
+ /* Place all of the ffi_prep_args in position */
75
+ mov a0, sp
76
+ /* a1 already set */
77
+
78
+ /* Call ffi_prep_args(stack, &ecif) */
79
+ jsri ffi_prep_args
80
+
81
+ /* move first 4 parameters in registers */
82
+ ldw a0, (sp, 0x0)
83
+ ldw a1, (sp, 0x4)
84
+ ldw a2, (sp, 0x8)
85
+ ldw a3, (sp, 0xC)
86
+
87
+ /* and adjust stack */
88
+ subu lr, l0, sp /* cif->bytes == l0 - sp */
89
+ cmphsi lr, 16
90
+ movi l1, 16
91
+ movt lr, l1
92
+ addu sp, sp, lr
93
+
94
+ ldw l1, (l0, 0) /* load fn() in advance */
95
+
96
+ /* call (fn) (...) */
97
+ jsr l1
98
+
99
+ /* Remove the space we pushed for the args */
100
+ mov sp, l0
101
+
102
+ /* Load r2 with the pointer to storage for the return value */
103
+ ldw a2, (sp, 0x1C)
104
+
105
+ /* Load r3 with the return type code */
106
+ ldw a3, (sp, 0xC)
107
+
108
+ /* If the return value pointer is NULL, assume no return value. */
109
+ cmpnei a2, 0
110
+ bf .Lepilogue
111
+
112
+ cmpnei a3, FFI_TYPE_STRUCT
113
+ bf .Lepilogue
114
+
115
+ /* return INT64 */
116
+ cmpnei a3, FFI_TYPE_SINT64
117
+ bt .Lretint
118
+ /* stw a0, (a2, 0x0) at .Lretint */
119
+ stw a1, (a2, 0x4)
120
+
121
+ .Lretint:
122
+ /* return INT */
123
+ stw a0, (a2, 0x0)
124
+
125
+ .Lepilogue:
126
+ ldw a0, (sp, 0x0)
127
+ ldw a1, (sp, 0x4)
128
+ ldw a2, (sp, 0x8)
129
+ ldw a3, (sp, 0xC)
130
+ ldw l0, (sp, 0x10)
131
+ ldw l1, (sp, 0x14)
132
+ ldw lr, (sp, 0x18)
133
+ addi sp, sp, 28
134
+ rts
135
+ .cfi_endproc
136
+ .size ffi_call_SYSV, .-ffi_call_SYSV
137
+
138
+
139
+ /*
140
+ * unsigned int FFI_HIDDEN
141
+ * ffi_closure_SYSV_inner (closure, respp, args)
142
+ * ffi_closure *closure;
143
+ * void **respp;
144
+ * void *args;
145
+ */
146
+ CSKY_FUNC_START ffi_closure_SYSV
147
+ .cfi_startproc
148
+ mov a2, sp
149
+ addi a1, sp, 16
150
+ subi sp, sp, 24
151
+ .cfi_def_cfa_offset 40
152
+ stw a1, (sp, 0x10)
153
+ .cfi_offset 1, -24
154
+ stw lr, (sp, 0x14)
155
+ .cfi_offset 15, -20
156
+ stw sp, (sp, 0x8)
157
+ addi a1, sp, 8
158
+ jsri ffi_closure_SYSV_inner
159
+ ldw a0, (sp, 0x0)
160
+ /*
161
+ * if FFI_TYPE_SINT64, need a1.
162
+ * if FFI_TYPE_INT, ignore a1.
163
+ */
164
+ ldw a1, (sp, 0x4)
165
+
166
+ ldw lr, (sp, 0x14)
167
+ addi sp, sp, 40
168
+ rts
169
+ .cfi_endproc
170
+ .size ffi_closure_SYSV, .-ffi_closure_SYSV
171
+
172
+ CSKY_FUNC_START ffi_csky_trampoline
173
+ subi sp, sp, 16
174
+ stw a0, (sp, 0x0)
175
+ stw a1, (sp, 0x4)
176
+ stw a2, (sp, 0x8)
177
+ stw a3, (sp, 0xC)
178
+ lrw a0, [.Lctx]
179
+ lrw a1, [.Lfun]
180
+ jmp a1
181
+ .Lctx:
182
+ mov a0, a0
183
+ mov a0, a0
184
+ .Lfun:
185
+
186
+ .size ffi_csky_trampoline, .-ffi_csky_trampoline
187
+
188
+ CSKY_FUNC_START ffi_csky_cacheflush
189
+ mov t0, r7
190
+ movi r7, 123
191
+ trap 0
192
+ mov r7, t0
193
+ rts
194
+
195
+ .size ffi_csky_cacheflush, .-ffi_csky_cacheflush
196
+
197
+ #else /* !__CSKYABIV2__ */
198
+
199
+ /*
200
+ * a0: fn
201
+ * a1: &ecif
202
+ * a2: cif->bytes
203
+ * a3: fig->flags
204
+ * a4: ecif.rvalue
205
+ */
206
+ CSKY_FUNC_START ffi_call_SYSV
207
+ /* Save registers */
208
+ .cfi_startproc
209
+ subi sp, 32
210
+ subi sp, 8
211
+ .cfi_def_cfa_offset 40
212
+ stw a0, (sp, 0x0)
213
+ .cfi_offset 2, -40
214
+ stw a1, (sp, 0x4)
215
+ .cfi_offset 3, -36
216
+ stw a2, (sp, 0x8)
217
+ .cfi_offset 4, -32
218
+ stw a3, (sp, 0xC)
219
+ .cfi_offset 5, -28
220
+ stw a4, (sp, 0x10)
221
+ .cfi_offset 6, -24
222
+ stw a5, (sp, 0x14)
223
+ .cfi_offset 7, -20
224
+ stw l0, (sp, 0x18)
225
+ .cfi_offset 8, -16
226
+ stw l1, (sp, 0x1C)
227
+ .cfi_offset 9, -12
228
+ stw lr, (sp, 0x20)
229
+ .cfi_offset 15, -8
230
+
231
+ mov l0, sp
232
+ .cfi_def_cfa_register 8
233
+
234
+ /* Make room for all of the new args. */
235
+ subu sp, sp, a2
236
+
237
+ /* Place all of the ffi_prep_args in position */
238
+ mov a0, sp
239
+ /* a1 already set */
240
+
241
+ /* Call ffi_prep_args(stack, &ecif) */
242
+ jsri ffi_prep_args
243
+
244
+ /* move first 4 parameters in registers */
245
+ ldw a0, (sp, 0x0)
246
+ ldw a1, (sp, 0x4)
247
+ ldw a2, (sp, 0x8)
248
+ ldw a3, (sp, 0xC)
249
+ ldw a4, (sp, 0x10)
250
+ ldw a5, (sp, 0x14)
251
+
252
+ /* and adjust stack */
253
+ mov lr, l0
254
+ subu lr, sp /* cif->bytes == l0 - sp */
255
+ movi l1, 24
256
+ cmphs lr, l1
257
+ movt lr, l1
258
+ addu sp, sp, lr
259
+
260
+ ldw l1, (l0, 0) /* load fn() in advance */
261
+
262
+ /* call (fn) (...) */
263
+ jsr l1
264
+
265
+ /* Remove the space we pushed for the args */
266
+ mov sp, l0
267
+
268
+ /* Load r2 with the pointer to storage for the return value */
269
+ ldw a2, (sp, 0x10)
270
+
271
+ /* Load r3 with the return type code */
272
+ ldw a3, (sp, 0xC)
273
+
274
+ /* If the return value pointer is NULL, assume no return value. */
275
+ cmpnei a2, 0
276
+ bf .Lepilogue
277
+
278
+ cmpnei a3, FFI_TYPE_STRUCT
279
+ bf .Lepilogue
280
+
281
+ /* return INT64 */
282
+ cmpnei a3, FFI_TYPE_SINT64
283
+ bt .Lretint
284
+ /* stw a0, (a2, 0x0) at .Lretint */
285
+ stw a1, (a2, 0x4)
286
+
287
+ .Lretint:
288
+ /* return INT */
289
+ stw a0, (a2, 0x0)
290
+
291
+ .Lepilogue:
292
+ ldw a0, (sp, 0x0)
293
+ ldw a1, (sp, 0x4)
294
+ ldw a2, (sp, 0x8)
295
+ ldw a3, (sp, 0xC)
296
+ ldw a4, (sp, 0x10)
297
+ ldw a5, (sp, 0x14)
298
+ ldw l0, (sp, 0x18)
299
+ ldw l1, (sp, 0x1C)
300
+ ldw lr, (sp, 0x20)
301
+ addi sp, sp, 32
302
+ addi sp, sp, 8
303
+ rts
304
+ .cfi_endproc
305
+
306
+ .size ffi_call_SYSV, .-ffi_call_SYSV
307
+
308
+
309
+ /*
310
+ * unsigned int FFI_HIDDEN
311
+ * ffi_closure_SYSV_inner (closure, respp, args)
312
+ * ffi_closure *closure;
313
+ * void **respp;
314
+ * void *args;
315
+ */
316
+ CSKY_FUNC_START ffi_closure_SYSV
317
+ .cfi_startproc
318
+ mov a2, sp
319
+ mov a1, sp
320
+ addi a1, 24
321
+ subi sp, sp, 24
322
+ .cfi_def_cfa_offset 48
323
+ stw a1, (sp, 0x10)
324
+ .cfi_offset 3, -32
325
+ stw lr, (sp, 0x14)
326
+ .cfi_offset 15, -28
327
+ stw sp, (sp, 0x8)
328
+ mov a1, sp
329
+ addi a1, 8
330
+ jsri ffi_closure_SYSV_inner
331
+ ldw a0, (sp, 0x0)
332
+ /*
333
+ * if FFI_TYPE_SINT64, need a1.
334
+ * if FFI_TYPE_INT, ignore a1.
335
+ */
336
+ ldw a1, (sp, 0x4)
337
+
338
+ ldw lr, (sp, 0x14)
339
+ addi sp, sp, 24
340
+ addi sp, sp, 24
341
+ rts
342
+ .cfi_endproc
343
+
344
+ .size ffi_closure_SYSV, .-ffi_closure_SYSV
345
+
346
+ CSKY_FUNC_START ffi_csky_trampoline
347
+ subi sp, 24
348
+ stw a0, (sp, 0x0)
349
+ stw a1, (sp, 0x4)
350
+ stw a2, (sp, 0x8)
351
+ stw a3, (sp, 0xC)
352
+ stw a4, (sp, 0x10)
353
+ stw a5, (sp, 0x14)
354
+ lrw a0, [.Lctx]
355
+ lrw a1, [.Lfun]
356
+ jmp a1
357
+ .Lctx:
358
+ mov a0, a0
359
+ mov a0, a0
360
+ .Lfun:
361
+
362
+ .size ffi_csky_trampoline, .-ffi_csky_trampoline
363
+
364
+ CSKY_FUNC_START ffi_csky_cacheflush
365
+ lrw r1, 123
366
+ trap 0
367
+ rts
368
+
369
+ .size ffi_csky_cacheflush, .-ffi_csky_cacheflush
370
+
371
+ #endif /* __CSKYABIV2__ */
@@ -2371,7 +2371,7 @@ static size_t traverse_and_check(mstate m);
2371
2371
 
2372
2372
  #else /* GNUC */
2373
2373
  #if USE_BUILTIN_FFS
2374
- #define compute_bit2idx(X, I) I = ffs(X)-1
2374
+ #define compute_bit2idx(X, I) I = __builtin_ffs(X)-1
2375
2375
 
2376
2376
  #else /* USE_BUILTIN_FFS */
2377
2377
  #define compute_bit2idx(X, I)\
@@ -0,0 +1,5 @@
1
+ /* args are passed on registers from r0 up to r11 => 12*8 bytes */
2
+ #define REG_ARGS_SIZE (12*8)
3
+ #define KVX_REGISTER_SIZE (8)
4
+ #define KVX_ABI_SLOT_SIZE (KVX_REGISTER_SIZE)
5
+ #define KVX_ABI_MAX_AGGREGATE_IN_REG_SIZE (4*KVX_ABI_SLOT_SIZE)
@@ -0,0 +1,273 @@
1
+ /* Copyright (c) 2020 Kalray
2
+
3
+ Permission is hereby granted, free of charge, to any person obtaining
4
+ a copy of this software and associated documentation files (the
5
+ ``Software''), to deal in the Software without restriction, including
6
+ without limitation the rights to use, copy, modify, merge, publish,
7
+ distribute, sublicense, and/or sell copies of the Software, and to
8
+ permit persons to whom the Software is furnished to do so, subject to
9
+ the following conditions:
10
+
11
+ The above copyright notice and this permission notice shall be
12
+ included in all copies or substantial portions of the Software.
13
+
14
+ THE SOFTWARE IS PROVIDED ``AS IS'', WITHOUT WARRANTY OF ANY KIND,
15
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
16
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
17
+ IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
18
+ CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
19
+ TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
20
+ SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
21
+
22
+ #if defined(__kvx__)
23
+ #include <stdio.h>
24
+ #include <stdlib.h>
25
+ #include <stdint.h>
26
+ #include <fficonfig.h>
27
+ #include <ffi.h>
28
+ #include "ffi_common.h"
29
+ #include "asm.h"
30
+
31
+ #define ALIGN(x, a) ALIGN_MASK(x, (typeof(x))(a) - 1)
32
+ #define ALIGN_MASK(x, mask) (((x) + (mask)) & ~(mask))
33
+ #define KVX_ABI_STACK_ALIGNMENT (32)
34
+ #define KVX_ABI_STACK_ARG_ALIGNMENT (8)
35
+ #define max(a,b) ((a) > (b) ? (a) : (b))
36
+
37
+ #ifdef FFI_DEBUG
38
+ #define DEBUG_PRINT(...) do{ fprintf( stderr, __VA_ARGS__ ); } while(0)
39
+ #else
40
+ #define DEBUG_PRINT(...)
41
+ #endif
42
+
43
+ struct ret_value {
44
+ unsigned long int r0;
45
+ unsigned long int r1;
46
+ unsigned long int r2;
47
+ unsigned long int r3;
48
+ };
49
+
50
+ extern struct ret_value ffi_call_SYSV(unsigned total_size,
51
+ unsigned size,
52
+ extended_cif *ecif,
53
+ unsigned *rvalue_addr,
54
+ void *fn,
55
+ unsigned int_ext_method);
56
+
57
+ /* Perform machine dependent cif processing */
58
+ ffi_status ffi_prep_cif_machdep(ffi_cif *cif)
59
+ {
60
+ cif->flags = cif->rtype->size;
61
+ return FFI_OK;
62
+ }
63
+
64
+ /* ffi_prep_args is called by the assembly routine once stack space
65
+ has been allocated for the function's arguments */
66
+
67
+ void *ffi_prep_args(char *stack, unsigned int arg_slots_size, extended_cif *ecif)
68
+ {
69
+ char *stacktemp = stack;
70
+ char *current_arg_passed_by_value = stack + arg_slots_size;
71
+ int i, s;
72
+ ffi_type **arg;
73
+ int count = 0;
74
+ ffi_cif *cif = ecif->cif;
75
+ void **argv = ecif->avalue;
76
+
77
+ arg = cif->arg_types;
78
+
79
+ DEBUG_PRINT("stack: %p\n", stack);
80
+ DEBUG_PRINT("arg_slots_size: %u\n", arg_slots_size);
81
+ DEBUG_PRINT("current_arg_passed_by_value: %p\n", current_arg_passed_by_value);
82
+ DEBUG_PRINT("ecif: %p\n", ecif);
83
+ DEBUG_PRINT("ecif->avalue: %p\n", ecif->avalue);
84
+
85
+ for (i = 0; i < cif->nargs; i++) {
86
+
87
+ s = KVX_ABI_SLOT_SIZE;
88
+ switch((*arg)->type) {
89
+ case FFI_TYPE_SINT8:
90
+ case FFI_TYPE_UINT8:
91
+ case FFI_TYPE_SINT16:
92
+ case FFI_TYPE_UINT16:
93
+ case FFI_TYPE_SINT32:
94
+ case FFI_TYPE_UINT32:
95
+ case FFI_TYPE_FLOAT:
96
+ case FFI_TYPE_DOUBLE:
97
+ case FFI_TYPE_UINT64:
98
+ case FFI_TYPE_SINT64:
99
+ case FFI_TYPE_POINTER:
100
+ DEBUG_PRINT("INT64/32/16/8/FLOAT/DOUBLE or POINTER @%p\n", stack);
101
+ *(uint64_t *) stack = *(uint64_t *)(* argv);
102
+ break;
103
+
104
+ case FFI_TYPE_COMPLEX:
105
+ if ((*arg)->size == 8)
106
+ *(_Complex float *) stack = *(_Complex float *)(* argv);
107
+ else if ((*arg)->size == 16) {
108
+ *(_Complex double *) stack = *(_Complex double *)(* argv);
109
+ s = 16;
110
+ } else
111
+ abort();
112
+ break;
113
+ case FFI_TYPE_STRUCT: {
114
+ char *value;
115
+ unsigned int written_size = 0;
116
+ DEBUG_PRINT("struct by value @%p\n", stack);
117
+ if ((*arg)->size > KVX_ABI_MAX_AGGREGATE_IN_REG_SIZE) {
118
+ DEBUG_PRINT("big struct\n");
119
+ *(uint64_t *) stack = (uintptr_t)current_arg_passed_by_value;
120
+ value = current_arg_passed_by_value;
121
+ current_arg_passed_by_value += (*arg)->size;
122
+ written_size = KVX_ABI_SLOT_SIZE;
123
+ } else {
124
+ value = stack;
125
+ written_size = (*arg)->size;
126
+ }
127
+ memcpy(value, *argv, (*arg)->size);
128
+ s = ALIGN(written_size, KVX_ABI_STACK_ARG_ALIGNMENT);
129
+ break;
130
+ }
131
+ default:
132
+ printf("Error: unsupported arg type %d\n", (*arg)->type);
133
+ abort();
134
+ break;
135
+
136
+ }
137
+ stack += s;
138
+ count += s;
139
+ argv++;
140
+ arg++;
141
+ }
142
+ #ifdef FFI_DEBUG
143
+ FFI_ASSERT(((intptr_t)(stacktemp + REG_ARGS_SIZE) & (KVX_ABI_STACK_ALIGNMENT-1)) == 0);
144
+ #endif
145
+ return stacktemp + REG_ARGS_SIZE;
146
+ }
147
+
148
+ /* Perform machine dependent cif processing when we have a variadic function */
149
+
150
+ ffi_status ffi_prep_cif_machdep_var(ffi_cif *cif, unsigned int nfixedargs,
151
+ unsigned int ntotalargs)
152
+ {
153
+ cif->flags = cif->rtype->size;
154
+ return FFI_OK;
155
+ }
156
+
157
+ static unsigned long handle_small_int_ext(kvx_intext_method *int_ext_method,
158
+ const ffi_type *rtype)
159
+ {
160
+ switch (rtype->type) {
161
+ case FFI_TYPE_SINT8:
162
+ *int_ext_method = KVX_RET_SXBD;
163
+ return KVX_REGISTER_SIZE;
164
+
165
+ case FFI_TYPE_SINT16:
166
+ *int_ext_method = KVX_RET_SXHD;
167
+ return KVX_REGISTER_SIZE;
168
+
169
+ case FFI_TYPE_SINT32:
170
+ *int_ext_method = KVX_RET_SXWD;
171
+ return KVX_REGISTER_SIZE;
172
+
173
+ case FFI_TYPE_UINT8:
174
+ *int_ext_method = KVX_RET_ZXBD;
175
+ return KVX_REGISTER_SIZE;
176
+
177
+ case FFI_TYPE_UINT16:
178
+ *int_ext_method = KVX_RET_ZXHD;
179
+ return KVX_REGISTER_SIZE;
180
+
181
+ case FFI_TYPE_UINT32:
182
+ *int_ext_method = KVX_RET_ZXWD;
183
+ return KVX_REGISTER_SIZE;
184
+
185
+ default:
186
+ *int_ext_method = KVX_RET_NONE;
187
+ return rtype->size;
188
+ }
189
+ }
190
+
191
+ void ffi_call(ffi_cif *cif, void (*fn)(void), void *rvalue, void **avalue)
192
+ {
193
+ int i;
194
+ unsigned long int slot_fitting_args_size = 0;
195
+ unsigned long int total_size = 0;
196
+ unsigned long int big_struct_size = 0;
197
+ kvx_intext_method int_extension_method;
198
+ ffi_type **arg;
199
+ struct ret_value local_rvalue = {0};
200
+ size_t wb_size;
201
+
202
+
203
+ /* Calculate size to allocate on stack */
204
+ for (i = 0, arg = cif->arg_types; i < cif->nargs; i++, arg++) {
205
+ DEBUG_PRINT("argument %d, type %d, size %lu\n", i, (*arg)->type, (*arg)->size);
206
+ if (((*arg)->type == FFI_TYPE_STRUCT) || ((*arg)->type == FFI_TYPE_COMPLEX)) {
207
+ if ((*arg)->size <= KVX_ABI_MAX_AGGREGATE_IN_REG_SIZE) {
208
+ slot_fitting_args_size += ALIGN((*arg)->size, KVX_ABI_SLOT_SIZE);
209
+ } else {
210
+ slot_fitting_args_size += KVX_ABI_SLOT_SIZE; /* aggregate passed by reference */
211
+ big_struct_size += ALIGN((*arg)->size, KVX_ABI_SLOT_SIZE);
212
+ }
213
+ } else if ((*arg)->size <= KVX_ABI_SLOT_SIZE) {
214
+ slot_fitting_args_size += KVX_ABI_SLOT_SIZE;
215
+ } else {
216
+ printf("Error: unsupported arg size %ld arg type %d\n", (*arg)->size, (*arg)->type);
217
+ abort(); /* should never happen? */
218
+ }
219
+ }
220
+
221
+ extended_cif ecif;
222
+ ecif.cif = cif;
223
+ ecif.avalue = avalue;
224
+ ecif.rvalue = rvalue;
225
+
226
+ /* This implementation allocates anyway for all register based args */
227
+ slot_fitting_args_size = max(slot_fitting_args_size, REG_ARGS_SIZE);
228
+ total_size = slot_fitting_args_size + big_struct_size;
229
+ total_size = ALIGN(total_size, KVX_ABI_STACK_ALIGNMENT);
230
+
231
+ /* wb_size: write back size, the size we will need to write back to user
232
+ * provided buffer. In theory it should always be cif->flags which is
233
+ * cif->rtype->size. But libffi API mandates that for integral types
234
+ * of size <= system register size, then we *MUST* write back
235
+ * the size of system register size.
236
+ * in our case, if size <= 8 bytes we must write back 8 bytes.
237
+ * floats, complex and structs are not affected, only integrals.
238
+ */
239
+ wb_size = handle_small_int_ext(&int_extension_method, cif->rtype);
240
+
241
+ switch (cif->abi) {
242
+ case FFI_SYSV:
243
+ DEBUG_PRINT("total_size: %lu\n", total_size);
244
+ DEBUG_PRINT("slot fitting args size: %lu\n", slot_fitting_args_size);
245
+ DEBUG_PRINT("rvalue: %p\n", rvalue);
246
+ DEBUG_PRINT("fn: %p\n", fn);
247
+ DEBUG_PRINT("rsize: %u\n", cif->flags);
248
+ DEBUG_PRINT("wb_size: %u\n", wb_size);
249
+ DEBUG_PRINT("int_extension_method: %u\n", int_extension_method);
250
+ local_rvalue = ffi_call_SYSV(total_size, slot_fitting_args_size,
251
+ &ecif, rvalue, fn, int_extension_method);
252
+ if ((cif->flags <= KVX_ABI_MAX_AGGREGATE_IN_REG_SIZE)
253
+ && (cif->rtype->type != FFI_TYPE_VOID))
254
+ memcpy(rvalue, &local_rvalue, wb_size);
255
+ break;
256
+ default:
257
+ abort();
258
+ break;
259
+ }
260
+ }
261
+
262
+ /* Closures not supported yet */
263
+ ffi_status
264
+ ffi_prep_closure_loc (ffi_closure* closure,
265
+ ffi_cif* cif,
266
+ void (*fun)(ffi_cif*,void*,void**,void*),
267
+ void *user_data,
268
+ void *codeloc)
269
+ {
270
+ return FFI_BAD_ABI;
271
+ }
272
+
273
+ #endif /* (__kvx__) */