alinta-ffi 1.9.19

Sign up to get free protection for your applications and to get access to all the features.
Files changed (195) hide show
  1. checksums.yaml +7 -0
  2. data/COPYING +49 -0
  3. data/LICENSE +24 -0
  4. data/README.md +112 -0
  5. data/Rakefile +243 -0
  6. data/ext/ffi_c/AbstractMemory.c +1109 -0
  7. data/ext/ffi_c/AbstractMemory.h +175 -0
  8. data/ext/ffi_c/ArrayType.c +162 -0
  9. data/ext/ffi_c/ArrayType.h +59 -0
  10. data/ext/ffi_c/Buffer.c +365 -0
  11. data/ext/ffi_c/Call.c +517 -0
  12. data/ext/ffi_c/Call.h +110 -0
  13. data/ext/ffi_c/ClosurePool.c +283 -0
  14. data/ext/ffi_c/ClosurePool.h +57 -0
  15. data/ext/ffi_c/DataConverter.c +91 -0
  16. data/ext/ffi_c/DynamicLibrary.c +339 -0
  17. data/ext/ffi_c/DynamicLibrary.h +98 -0
  18. data/ext/ffi_c/Function.c +998 -0
  19. data/ext/ffi_c/Function.h +87 -0
  20. data/ext/ffi_c/FunctionInfo.c +271 -0
  21. data/ext/ffi_c/LastError.c +184 -0
  22. data/ext/ffi_c/LastError.h +47 -0
  23. data/ext/ffi_c/LongDouble.c +63 -0
  24. data/ext/ffi_c/LongDouble.h +51 -0
  25. data/ext/ffi_c/MappedType.c +168 -0
  26. data/ext/ffi_c/MappedType.h +59 -0
  27. data/ext/ffi_c/MemoryPointer.c +197 -0
  28. data/ext/ffi_c/MemoryPointer.h +53 -0
  29. data/ext/ffi_c/MethodHandle.c +358 -0
  30. data/ext/ffi_c/MethodHandle.h +55 -0
  31. data/ext/ffi_c/Platform.c +129 -0
  32. data/ext/ffi_c/Platform.h +45 -0
  33. data/ext/ffi_c/Pointer.c +508 -0
  34. data/ext/ffi_c/Pointer.h +63 -0
  35. data/ext/ffi_c/Struct.c +829 -0
  36. data/ext/ffi_c/Struct.h +106 -0
  37. data/ext/ffi_c/StructByReference.c +190 -0
  38. data/ext/ffi_c/StructByReference.h +50 -0
  39. data/ext/ffi_c/StructByValue.c +150 -0
  40. data/ext/ffi_c/StructByValue.h +55 -0
  41. data/ext/ffi_c/StructLayout.c +698 -0
  42. data/ext/ffi_c/Thread.c +352 -0
  43. data/ext/ffi_c/Thread.h +95 -0
  44. data/ext/ffi_c/Type.c +397 -0
  45. data/ext/ffi_c/Type.h +62 -0
  46. data/ext/ffi_c/Types.c +139 -0
  47. data/ext/ffi_c/Types.h +89 -0
  48. data/ext/ffi_c/Variadic.c +304 -0
  49. data/ext/ffi_c/compat.h +78 -0
  50. data/ext/ffi_c/extconf.rb +71 -0
  51. data/ext/ffi_c/ffi.c +98 -0
  52. data/ext/ffi_c/libffi.bsd.mk +40 -0
  53. data/ext/ffi_c/libffi.darwin.mk +105 -0
  54. data/ext/ffi_c/libffi.gnu.mk +32 -0
  55. data/ext/ffi_c/libffi.mk +18 -0
  56. data/ext/ffi_c/libffi.vc.mk +26 -0
  57. data/ext/ffi_c/libffi.vc64.mk +26 -0
  58. data/ext/ffi_c/rbffi.h +57 -0
  59. data/ext/ffi_c/rbffi_endian.h +59 -0
  60. data/ext/ffi_c/win32/stdbool.h +8 -0
  61. data/ext/ffi_c/win32/stdint.h +201 -0
  62. data/ffi.gemspec +23 -0
  63. data/gen/Rakefile +30 -0
  64. data/lib/ffi.rb +20 -0
  65. data/lib/ffi/autopointer.rb +203 -0
  66. data/lib/ffi/buffer.rb +4 -0
  67. data/lib/ffi/callback.rb +4 -0
  68. data/lib/ffi/enum.rb +296 -0
  69. data/lib/ffi/errno.rb +43 -0
  70. data/lib/ffi/ffi.rb +44 -0
  71. data/lib/ffi/io.rb +62 -0
  72. data/lib/ffi/library.rb +590 -0
  73. data/lib/ffi/managedstruct.rb +84 -0
  74. data/lib/ffi/memorypointer.rb +1 -0
  75. data/lib/ffi/platform.rb +164 -0
  76. data/lib/ffi/platform/aarch64-linux/types.conf +104 -0
  77. data/lib/ffi/platform/arm-linux/types.conf +104 -0
  78. data/lib/ffi/platform/i386-cygwin/types.conf +3 -0
  79. data/lib/ffi/platform/i386-darwin/types.conf +100 -0
  80. data/lib/ffi/platform/i386-freebsd/types.conf +152 -0
  81. data/lib/ffi/platform/i386-gnu/types.conf +107 -0
  82. data/lib/ffi/platform/i386-linux/types.conf +103 -0
  83. data/lib/ffi/platform/i386-netbsd/types.conf +126 -0
  84. data/lib/ffi/platform/i386-openbsd/types.conf +128 -0
  85. data/lib/ffi/platform/i386-solaris/types.conf +122 -0
  86. data/lib/ffi/platform/i386-windows/types.conf +105 -0
  87. data/lib/ffi/platform/ia64-linux/types.conf +104 -0
  88. data/lib/ffi/platform/mips-linux/types.conf +102 -0
  89. data/lib/ffi/platform/mips64el-linux/types.conf +104 -0
  90. data/lib/ffi/platform/mipsel-linux/types.conf +102 -0
  91. data/lib/ffi/platform/powerpc-aix/types.conf +180 -0
  92. data/lib/ffi/platform/powerpc-darwin/types.conf +100 -0
  93. data/lib/ffi/platform/powerpc-linux/types.conf +100 -0
  94. data/lib/ffi/platform/powerpc64-linux/types.conf +104 -0
  95. data/lib/ffi/platform/s390-linux/types.conf +102 -0
  96. data/lib/ffi/platform/s390x-linux/types.conf +102 -0
  97. data/lib/ffi/platform/sparc-linux/types.conf +102 -0
  98. data/lib/ffi/platform/sparc-solaris/types.conf +128 -0
  99. data/lib/ffi/platform/sparc64-linux/types.conf +102 -0
  100. data/lib/ffi/platform/sparcv9-solaris/types.conf +128 -0
  101. data/lib/ffi/platform/x86_64-cygwin/types.conf +3 -0
  102. data/lib/ffi/platform/x86_64-darwin/types.conf +126 -0
  103. data/lib/ffi/platform/x86_64-freebsd/types.conf +128 -0
  104. data/lib/ffi/platform/x86_64-linux/types.conf +102 -0
  105. data/lib/ffi/platform/x86_64-netbsd/types.conf +128 -0
  106. data/lib/ffi/platform/x86_64-openbsd/types.conf +134 -0
  107. data/lib/ffi/platform/x86_64-solaris/types.conf +122 -0
  108. data/lib/ffi/platform/x86_64-windows/types.conf +120 -0
  109. data/lib/ffi/pointer.rb +161 -0
  110. data/lib/ffi/struct.rb +371 -0
  111. data/lib/ffi/struct_layout_builder.rb +227 -0
  112. data/lib/ffi/tools/const_generator.rb +229 -0
  113. data/lib/ffi/tools/generator.rb +60 -0
  114. data/lib/ffi/tools/generator_task.rb +36 -0
  115. data/lib/ffi/tools/struct_generator.rb +194 -0
  116. data/lib/ffi/tools/types_generator.rb +134 -0
  117. data/lib/ffi/types.rb +194 -0
  118. data/lib/ffi/union.rb +43 -0
  119. data/lib/ffi/variadic.rb +78 -0
  120. data/lib/ffi/version.rb +4 -0
  121. data/libtest/Benchmark.c +52 -0
  122. data/libtest/BoolTest.c +34 -0
  123. data/libtest/BufferTest.c +31 -0
  124. data/libtest/ClosureTest.c +205 -0
  125. data/libtest/EnumTest.c +51 -0
  126. data/libtest/FunctionTest.c +70 -0
  127. data/libtest/GNUmakefile +149 -0
  128. data/libtest/GlobalVariable.c +62 -0
  129. data/libtest/LastErrorTest.c +21 -0
  130. data/libtest/NumberTest.c +132 -0
  131. data/libtest/PointerTest.c +63 -0
  132. data/libtest/ReferenceTest.c +23 -0
  133. data/libtest/StringTest.c +34 -0
  134. data/libtest/StructTest.c +243 -0
  135. data/libtest/UnionTest.c +43 -0
  136. data/libtest/VariadicTest.c +99 -0
  137. data/spec/ffi/LICENSE.SPECS +22 -0
  138. data/spec/ffi/async_callback_spec.rb +35 -0
  139. data/spec/ffi/bitmask_spec.rb +575 -0
  140. data/spec/ffi/bool_spec.rb +32 -0
  141. data/spec/ffi/buffer_spec.rb +279 -0
  142. data/spec/ffi/callback_spec.rb +773 -0
  143. data/spec/ffi/custom_param_type.rb +37 -0
  144. data/spec/ffi/custom_type_spec.rb +74 -0
  145. data/spec/ffi/dup_spec.rb +52 -0
  146. data/spec/ffi/enum_spec.rb +423 -0
  147. data/spec/ffi/errno_spec.rb +20 -0
  148. data/spec/ffi/ffi_spec.rb +28 -0
  149. data/spec/ffi/fixtures/Benchmark.c +52 -0
  150. data/spec/ffi/fixtures/BitmaskTest.c +51 -0
  151. data/spec/ffi/fixtures/BoolTest.c +34 -0
  152. data/spec/ffi/fixtures/BufferTest.c +31 -0
  153. data/spec/ffi/fixtures/ClosureTest.c +205 -0
  154. data/spec/ffi/fixtures/EnumTest.c +51 -0
  155. data/spec/ffi/fixtures/FunctionTest.c +142 -0
  156. data/spec/ffi/fixtures/GNUmakefile +149 -0
  157. data/spec/ffi/fixtures/GlobalVariable.c +62 -0
  158. data/spec/ffi/fixtures/LastErrorTest.c +21 -0
  159. data/spec/ffi/fixtures/NumberTest.c +132 -0
  160. data/spec/ffi/fixtures/PipeHelper.h +21 -0
  161. data/spec/ffi/fixtures/PipeHelperPosix.c +41 -0
  162. data/spec/ffi/fixtures/PipeHelperWindows.c +72 -0
  163. data/spec/ffi/fixtures/PointerTest.c +63 -0
  164. data/spec/ffi/fixtures/ReferenceTest.c +23 -0
  165. data/spec/ffi/fixtures/StringTest.c +34 -0
  166. data/spec/ffi/fixtures/StructTest.c +243 -0
  167. data/spec/ffi/fixtures/UnionTest.c +43 -0
  168. data/spec/ffi/fixtures/VariadicTest.c +99 -0
  169. data/spec/ffi/fixtures/classes.rb +438 -0
  170. data/spec/ffi/function_spec.rb +97 -0
  171. data/spec/ffi/io_spec.rb +16 -0
  172. data/spec/ffi/library_spec.rb +286 -0
  173. data/spec/ffi/long_double.rb +30 -0
  174. data/spec/ffi/managed_struct_spec.rb +68 -0
  175. data/spec/ffi/memorypointer_spec.rb +78 -0
  176. data/spec/ffi/number_spec.rb +247 -0
  177. data/spec/ffi/platform_spec.rb +114 -0
  178. data/spec/ffi/pointer_spec.rb +285 -0
  179. data/spec/ffi/rbx/attach_function_spec.rb +34 -0
  180. data/spec/ffi/rbx/memory_pointer_spec.rb +198 -0
  181. data/spec/ffi/rbx/spec_helper.rb +6 -0
  182. data/spec/ffi/rbx/struct_spec.rb +18 -0
  183. data/spec/ffi/spec_helper.rb +93 -0
  184. data/spec/ffi/string_spec.rb +118 -0
  185. data/spec/ffi/strptr_spec.rb +50 -0
  186. data/spec/ffi/struct_by_ref_spec.rb +43 -0
  187. data/spec/ffi/struct_callback_spec.rb +69 -0
  188. data/spec/ffi/struct_initialize_spec.rb +35 -0
  189. data/spec/ffi/struct_packed_spec.rb +50 -0
  190. data/spec/ffi/struct_spec.rb +882 -0
  191. data/spec/ffi/typedef_spec.rb +91 -0
  192. data/spec/ffi/union_spec.rb +67 -0
  193. data/spec/ffi/variadic_spec.rb +132 -0
  194. data/spec/spec.opts +4 -0
  195. metadata +309 -0
@@ -0,0 +1,517 @@
1
+ /*
2
+ * Copyright (c) 2009, Wayne Meissner
3
+ * Copyright (c) 2009, Luc Heinrich <luc@honk-honk.com>
4
+ * Copyright (c) 2009, Mike Dalessio <mike.dalessio@gmail.com>
5
+ * Copyright (c) 2009, Aman Gupta.
6
+ * Copyright (c) 2008-2013, Ruby FFI project contributors
7
+ * All rights reserved.
8
+ *
9
+ * Redistribution and use in source and binary forms, with or without
10
+ * modification, are permitted provided that the following conditions are met:
11
+ * * Redistributions of source code must retain the above copyright
12
+ * notice, this list of conditions and the following disclaimer.
13
+ * * Redistributions in binary form must reproduce the above copyright
14
+ * notice, this list of conditions and the following disclaimer in the
15
+ * documentation and/or other materials provided with the distribution.
16
+ * * Neither the name of the Ruby FFI project nor the
17
+ * names of its contributors may be used to endorse or promote products
18
+ * derived from this software without specific prior written permission.
19
+ *
20
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
21
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
22
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
23
+ * DISCLAIMED. IN NO EVENT SHALL <COPYRIGHT HOLDER> BE LIABLE FOR ANY
24
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
25
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
26
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
27
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
28
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
29
+ * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30
+ */
31
+
32
+ #ifndef _MSC_VER
33
+ #include <sys/param.h>
34
+ #endif
35
+ #include <sys/types.h>
36
+ #include <stdio.h>
37
+ #ifndef _MSC_VER
38
+ # include <stdint.h>
39
+ # include <stdbool.h>
40
+ #else
41
+ # include "win32/stdbool.h"
42
+ # include "win32/stdint.h"
43
+ #endif
44
+ #include <errno.h>
45
+ #include <ruby.h>
46
+ #if defined(HAVE_NATIVETHREAD) && (defined(HAVE_RB_THREAD_BLOCKING_REGION) || defined(HAVE_RB_THREAD_CALL_WITHOUT_GVL)) && !defined(_WIN32)
47
+ # include <signal.h>
48
+ # include <pthread.h>
49
+ #endif
50
+ #include <ffi.h>
51
+ #include "extconf.h"
52
+ #include "rbffi.h"
53
+ #include "compat.h"
54
+ #include "AbstractMemory.h"
55
+ #include "Pointer.h"
56
+ #include "Struct.h"
57
+ #include "Function.h"
58
+ #include "Type.h"
59
+ #include "LastError.h"
60
+ #include "Call.h"
61
+ #include "MappedType.h"
62
+ #include "Thread.h"
63
+ #include "LongDouble.h"
64
+
65
+ #ifdef USE_RAW
66
+ # ifndef __i386__
67
+ # error "RAW argument packing only supported on i386"
68
+ # endif
69
+
70
+ #define INT8_ADJ (4)
71
+ #define INT16_ADJ (4)
72
+ #define INT32_ADJ (4)
73
+ #define INT64_ADJ (8)
74
+ #define LONG_ADJ (sizeof(long))
75
+ #define FLOAT32_ADJ (4)
76
+ #define FLOAT64_ADJ (8)
77
+ #define ADDRESS_ADJ (sizeof(void *))
78
+ #define LONGDOUBLE_ADJ (ffi_type_longdouble.alignment)
79
+
80
+ #endif /* USE_RAW */
81
+
82
+ #ifdef USE_RAW
83
+ # define ADJ(p, a) ((p) = (FFIStorage*) (((char *) p) + a##_ADJ))
84
+ #else
85
+ # define ADJ(p, a) (++(p))
86
+ #endif
87
+
88
+ static void* callback_param(VALUE proc, VALUE cbinfo);
89
+ static inline void* getPointer(VALUE value, int type);
90
+
91
+ static ID id_to_ptr, id_map_symbol, id_to_native;
92
+
93
+ void
94
+ rbffi_SetupCallParams(int argc, VALUE* argv, int paramCount, Type** paramTypes,
95
+ FFIStorage* paramStorage, void** ffiValues,
96
+ VALUE* callbackParameters, int callbackCount, VALUE enums)
97
+ {
98
+ VALUE callbackProc = Qnil;
99
+ FFIStorage* param = &paramStorage[0];
100
+ int i, argidx, cbidx, argCount;
101
+
102
+ if (unlikely(paramCount != -1 && paramCount != argc)) {
103
+ if (argc == (paramCount - 1) && callbackCount == 1 && rb_block_given_p()) {
104
+ callbackProc = rb_block_proc();
105
+ } else {
106
+ rb_raise(rb_eArgError, "wrong number of arguments (%d for %d)", argc, paramCount);
107
+ }
108
+ }
109
+
110
+ argCount = paramCount != -1 ? paramCount : argc;
111
+
112
+ for (i = 0, argidx = 0, cbidx = 0; i < argCount; ++i) {
113
+ Type* paramType = paramTypes[i];
114
+ int type;
115
+
116
+
117
+ if (unlikely(paramType->nativeType == NATIVE_MAPPED)) {
118
+ VALUE values[] = { argv[argidx], Qnil };
119
+ argv[argidx] = rb_funcall2(((MappedType *) paramType)->rbConverter, id_to_native, 2, values);
120
+ paramType = ((MappedType *) paramType)->type;
121
+ }
122
+
123
+ type = argidx < argc ? TYPE(argv[argidx]) : T_NONE;
124
+ ffiValues[i] = param;
125
+
126
+ switch (paramType->nativeType) {
127
+
128
+ case NATIVE_INT8:
129
+ if (unlikely(type == T_SYMBOL && enums != Qnil)) {
130
+ VALUE value = rb_funcall(enums, id_map_symbol, 1, argv[argidx]);
131
+ param->s8 = NUM2INT(value);
132
+ } else {
133
+ param->s8 = NUM2INT(argv[argidx]);
134
+ }
135
+
136
+ ++argidx;
137
+ ADJ(param, INT8);
138
+ break;
139
+
140
+ case NATIVE_INT16:
141
+ if (unlikely(type == T_SYMBOL && enums != Qnil)) {
142
+ VALUE value = rb_funcall(enums, id_map_symbol, 1, argv[argidx]);
143
+ param->s16 = NUM2INT(value);
144
+
145
+ } else {
146
+ param->s16 = NUM2INT(argv[argidx]);
147
+ }
148
+
149
+ ++argidx;
150
+ ADJ(param, INT16);
151
+ break;
152
+
153
+ case NATIVE_INT32:
154
+ if (unlikely(type == T_SYMBOL && enums != Qnil)) {
155
+ VALUE value = rb_funcall(enums, id_map_symbol, 1, argv[argidx]);
156
+ param->s32 = NUM2INT(value);
157
+
158
+ } else {
159
+ param->s32 = NUM2INT(argv[argidx]);
160
+ }
161
+
162
+ ++argidx;
163
+ ADJ(param, INT32);
164
+ break;
165
+
166
+ case NATIVE_BOOL:
167
+ if (type != T_TRUE && type != T_FALSE) {
168
+ rb_raise(rb_eTypeError, "wrong argument type (expected a boolean parameter)");
169
+ }
170
+ param->s8 = argv[argidx++] == Qtrue;
171
+ ADJ(param, INT8);
172
+ break;
173
+
174
+ case NATIVE_UINT8:
175
+ if (unlikely(type == T_SYMBOL && enums != Qnil)) {
176
+ VALUE value = rb_funcall(enums, id_map_symbol, 1, argv[argidx]);
177
+ param->u8 = NUM2UINT(value);
178
+ } else {
179
+ param->u8 = NUM2UINT(argv[argidx]);
180
+ }
181
+
182
+ ADJ(param, INT8);
183
+ ++argidx;
184
+ break;
185
+
186
+ case NATIVE_UINT16:
187
+ if (unlikely(type == T_SYMBOL && enums != Qnil)) {
188
+ VALUE value = rb_funcall(enums, id_map_symbol, 1, argv[argidx]);
189
+ param->u16 = NUM2UINT(value);
190
+ } else {
191
+ param->u16 = NUM2UINT(argv[argidx]);
192
+ }
193
+
194
+ ADJ(param, INT16);
195
+ ++argidx;
196
+ break;
197
+
198
+ case NATIVE_UINT32:
199
+ if (unlikely(type == T_SYMBOL && enums != Qnil)) {
200
+ VALUE value = rb_funcall(enums, id_map_symbol, 1, argv[argidx]);
201
+ param->u32 = NUM2UINT(value);
202
+ } else {
203
+ param->u32 = NUM2UINT(argv[argidx]);
204
+ }
205
+
206
+ ADJ(param, INT32);
207
+ ++argidx;
208
+ break;
209
+
210
+ case NATIVE_INT64:
211
+ if (unlikely(type == T_SYMBOL && enums != Qnil)) {
212
+ VALUE value = rb_funcall(enums, id_map_symbol, 1, argv[argidx]);
213
+ param->i64 = NUM2LL(value);
214
+ } else {
215
+ param->i64 = NUM2LL(argv[argidx]);
216
+ }
217
+
218
+ ADJ(param, INT64);
219
+ ++argidx;
220
+ break;
221
+
222
+ case NATIVE_UINT64:
223
+ if (unlikely(type == T_SYMBOL && enums != Qnil)) {
224
+ VALUE value = rb_funcall(enums, id_map_symbol, 1, argv[argidx]);
225
+ param->u64 = NUM2ULL(value);
226
+ } else {
227
+ param->u64 = NUM2ULL(argv[argidx]);
228
+ }
229
+
230
+ ADJ(param, INT64);
231
+ ++argidx;
232
+ break;
233
+
234
+ case NATIVE_LONG:
235
+ if (unlikely(type == T_SYMBOL && enums != Qnil)) {
236
+ VALUE value = rb_funcall(enums, id_map_symbol, 1, argv[argidx]);
237
+ *(ffi_sarg *) param = NUM2LONG(value);
238
+ } else {
239
+ *(ffi_sarg *) param = NUM2LONG(argv[argidx]);
240
+ }
241
+
242
+ ADJ(param, LONG);
243
+ ++argidx;
244
+ break;
245
+
246
+ case NATIVE_ULONG:
247
+ if (unlikely(type == T_SYMBOL && enums != Qnil)) {
248
+ VALUE value = rb_funcall(enums, id_map_symbol, 1, argv[argidx]);
249
+ *(ffi_arg *) param = NUM2ULONG(value);
250
+ } else {
251
+ *(ffi_arg *) param = NUM2ULONG(argv[argidx]);
252
+ }
253
+
254
+ ADJ(param, LONG);
255
+ ++argidx;
256
+ break;
257
+
258
+ case NATIVE_FLOAT32:
259
+ if (unlikely(type == T_SYMBOL && enums != Qnil)) {
260
+ VALUE value = rb_funcall(enums, id_map_symbol, 1, argv[argidx]);
261
+ param->f32 = (float) NUM2DBL(value);
262
+ } else {
263
+ param->f32 = (float) NUM2DBL(argv[argidx]);
264
+ }
265
+
266
+ ADJ(param, FLOAT32);
267
+ ++argidx;
268
+ break;
269
+
270
+ case NATIVE_FLOAT64:
271
+ if (unlikely(type == T_SYMBOL && enums != Qnil)) {
272
+ VALUE value = rb_funcall(enums, id_map_symbol, 1, argv[argidx]);
273
+ param->f64 = NUM2DBL(value);
274
+ } else {
275
+ param->f64 = NUM2DBL(argv[argidx]);
276
+ }
277
+
278
+ ADJ(param, FLOAT64);
279
+ ++argidx;
280
+ break;
281
+
282
+ case NATIVE_LONGDOUBLE:
283
+ if (unlikely(type == T_SYMBOL && enums != Qnil)) {
284
+ VALUE value = rb_funcall(enums, id_map_symbol, 1, argv[argidx]);
285
+ param->ld = rbffi_num2longdouble(value);
286
+ } else {
287
+ param->ld = rbffi_num2longdouble(argv[argidx]);
288
+ }
289
+
290
+ ADJ(param, LONGDOUBLE);
291
+ ++argidx;
292
+ break;
293
+
294
+
295
+ case NATIVE_STRING:
296
+ if (type == T_NIL) {
297
+ param->ptr = NULL;
298
+
299
+ } else {
300
+ if (rb_safe_level() >= 1 && OBJ_TAINTED(argv[argidx])) {
301
+ rb_raise(rb_eSecurityError, "Unsafe string parameter");
302
+ }
303
+
304
+ param->ptr = StringValueCStr(argv[argidx]);
305
+ }
306
+
307
+ ADJ(param, ADDRESS);
308
+ ++argidx;
309
+ break;
310
+
311
+ case NATIVE_POINTER:
312
+ case NATIVE_BUFFER_IN:
313
+ case NATIVE_BUFFER_OUT:
314
+ case NATIVE_BUFFER_INOUT:
315
+ param->ptr = getPointer(argv[argidx++], type);
316
+ ADJ(param, ADDRESS);
317
+ break;
318
+
319
+
320
+ case NATIVE_FUNCTION:
321
+ case NATIVE_CALLBACK:
322
+ if (callbackProc != Qnil) {
323
+ param->ptr = callback_param(callbackProc, callbackParameters[cbidx++]);
324
+ } else {
325
+ param->ptr = callback_param(argv[argidx], callbackParameters[cbidx++]);
326
+ ++argidx;
327
+ }
328
+ ADJ(param, ADDRESS);
329
+ break;
330
+
331
+ case NATIVE_STRUCT:
332
+ ffiValues[i] = getPointer(argv[argidx++], type);
333
+ break;
334
+
335
+ default:
336
+ rb_raise(rb_eArgError, "Invalid parameter type: %d", paramType->nativeType);
337
+ }
338
+ }
339
+ }
340
+
341
+ static VALUE
342
+ call_blocking_function(void* data)
343
+ {
344
+ rbffi_blocking_call_t* b = (rbffi_blocking_call_t *) data;
345
+ b->frame->has_gvl = false;
346
+ ffi_call(&b->cif, FFI_FN(b->function), b->retval, b->ffiValues);
347
+ b->frame->has_gvl = true;
348
+
349
+ return Qnil;
350
+ }
351
+
352
+ VALUE
353
+ rbffi_do_blocking_call(void *data)
354
+ {
355
+ rbffi_thread_blocking_region(call_blocking_function, data, (void *) -1, NULL);
356
+
357
+ return Qnil;
358
+ }
359
+
360
+ VALUE
361
+ rbffi_save_frame_exception(void *data, VALUE exc)
362
+ {
363
+ rbffi_frame_t* frame = (rbffi_frame_t *) data;
364
+ frame->exc = exc;
365
+ return Qnil;
366
+ }
367
+
368
+ VALUE
369
+ rbffi_CallFunction(int argc, VALUE* argv, void* function, FunctionType* fnInfo)
370
+ {
371
+ void* retval;
372
+ void** ffiValues;
373
+ FFIStorage* params;
374
+ VALUE rbReturnValue;
375
+ rbffi_frame_t frame = { 0 };
376
+
377
+ retval = alloca(MAX(fnInfo->ffi_cif.rtype->size, FFI_SIZEOF_ARG));
378
+
379
+ if (unlikely(fnInfo->blocking)) {
380
+ rbffi_blocking_call_t* bc;
381
+
382
+ /*
383
+ * due to the way thread switching works on older ruby variants, we
384
+ * cannot allocate anything passed to the blocking function on the stack
385
+ */
386
+ #if defined(HAVE_RB_THREAD_BLOCKING_REGION) || defined(HAVE_RB_THREAD_CALL_WITHOUT_GVL)
387
+ ffiValues = ALLOCA_N(void *, fnInfo->parameterCount);
388
+ params = ALLOCA_N(FFIStorage, fnInfo->parameterCount);
389
+ bc = ALLOCA_N(rbffi_blocking_call_t, 1);
390
+ bc->retval = retval;
391
+ #else
392
+ ffiValues = ALLOC_N(void *, fnInfo->parameterCount);
393
+ params = ALLOC_N(FFIStorage, fnInfo->parameterCount);
394
+ bc = ALLOC_N(rbffi_blocking_call_t, 1);
395
+ bc->retval = xmalloc(MAX(fnInfo->ffi_cif.rtype->size, FFI_SIZEOF_ARG));
396
+ bc->stkretval = retval;
397
+ #endif
398
+ bc->cif = fnInfo->ffi_cif;
399
+ bc->function = function;
400
+ bc->ffiValues = ffiValues;
401
+ bc->params = params;
402
+ bc->frame = &frame;
403
+
404
+ rbffi_SetupCallParams(argc, argv,
405
+ fnInfo->parameterCount, fnInfo->parameterTypes, params, ffiValues,
406
+ fnInfo->callbackParameters, fnInfo->callbackCount, fnInfo->rbEnums);
407
+
408
+ rbffi_frame_push(&frame);
409
+ rb_rescue2(rbffi_do_blocking_call, (VALUE) bc, rbffi_save_frame_exception, (VALUE) &frame, rb_eException, (VALUE) 0);
410
+ rbffi_frame_pop(&frame);
411
+
412
+ #if !(defined(HAVE_RB_THREAD_BLOCKING_REGION) || defined(HAVE_RB_THREAD_CALL_WITHOUT_GVL))
413
+ memcpy(bc->stkretval, bc->retval, MAX(bc->cif.rtype->size, FFI_SIZEOF_ARG));
414
+ xfree(bc->params);
415
+ xfree(bc->ffiValues);
416
+ xfree(bc->retval);
417
+ xfree(bc);
418
+ #endif
419
+
420
+ } else {
421
+
422
+ ffiValues = ALLOCA_N(void *, fnInfo->parameterCount);
423
+ params = ALLOCA_N(FFIStorage, fnInfo->parameterCount);
424
+
425
+ rbffi_SetupCallParams(argc, argv,
426
+ fnInfo->parameterCount, fnInfo->parameterTypes, params, ffiValues,
427
+ fnInfo->callbackParameters, fnInfo->callbackCount, fnInfo->rbEnums);
428
+
429
+ rbffi_frame_push(&frame);
430
+ ffi_call(&fnInfo->ffi_cif, FFI_FN(function), retval, ffiValues);
431
+ rbffi_frame_pop(&frame);
432
+ }
433
+
434
+ if (unlikely(!fnInfo->ignoreErrno)) {
435
+ rbffi_save_errno();
436
+ }
437
+
438
+ if (RTEST(frame.exc) && frame.exc != Qnil) {
439
+ rb_exc_raise(frame.exc);
440
+ }
441
+
442
+ RB_GC_GUARD(rbReturnValue) = rbffi_NativeValue_ToRuby(fnInfo->returnType, fnInfo->rbReturnType, retval);
443
+ RB_GC_GUARD(fnInfo->rbReturnType);
444
+
445
+ return rbReturnValue;
446
+ }
447
+
448
+ static inline void*
449
+ getPointer(VALUE value, int type)
450
+ {
451
+ if (likely(type == T_DATA && rb_obj_is_kind_of(value, rbffi_AbstractMemoryClass))) {
452
+
453
+ return ((AbstractMemory *) DATA_PTR(value))->address;
454
+
455
+ } else if (type == T_DATA && rb_obj_is_kind_of(value, rbffi_StructClass)) {
456
+
457
+ AbstractMemory* memory = ((Struct *) DATA_PTR(value))->pointer;
458
+ return memory != NULL ? memory->address : NULL;
459
+
460
+ } else if (type == T_STRING) {
461
+
462
+ return StringValuePtr(value);
463
+
464
+ } else if (type == T_NIL) {
465
+
466
+ return NULL;
467
+
468
+ } else if (rb_respond_to(value, id_to_ptr)) {
469
+
470
+ VALUE ptr = rb_funcall2(value, id_to_ptr, 0, NULL);
471
+ if (rb_obj_is_kind_of(ptr, rbffi_AbstractMemoryClass) && TYPE(ptr) == T_DATA) {
472
+ return ((AbstractMemory *) DATA_PTR(ptr))->address;
473
+ }
474
+ rb_raise(rb_eArgError, "to_ptr returned an invalid pointer");
475
+ }
476
+
477
+ rb_raise(rb_eArgError, ":pointer argument is not a valid pointer");
478
+ return NULL;
479
+ }
480
+
481
+ Invoker
482
+ rbffi_GetInvoker(FunctionType *fnInfo)
483
+ {
484
+ return rbffi_CallFunction;
485
+ }
486
+
487
+
488
+ static void*
489
+ callback_param(VALUE proc, VALUE cbInfo)
490
+ {
491
+ VALUE callback ;
492
+ if (unlikely(proc == Qnil)) {
493
+ return NULL ;
494
+ }
495
+
496
+ /* Handle Function pointers here */
497
+ if (rb_obj_is_kind_of(proc, rbffi_FunctionClass)) {
498
+ AbstractMemory* ptr;
499
+ Data_Get_Struct(proc, AbstractMemory, ptr);
500
+ return ptr->address;
501
+ }
502
+
503
+ callback = rbffi_Function_ForProc(cbInfo, proc);
504
+ RB_GC_GUARD(callback);
505
+
506
+ return ((AbstractMemory *) DATA_PTR(callback))->address;
507
+ }
508
+
509
+
510
+ void
511
+ rbffi_Call_Init(VALUE moduleFFI)
512
+ {
513
+ id_to_ptr = rb_intern("to_ptr");
514
+ id_to_native = rb_intern("to_native");
515
+ id_map_symbol = rb_intern("__map_symbol");
516
+ }
517
+