alinta-ffi 1.9.19
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/COPYING +49 -0
- data/LICENSE +24 -0
- data/README.md +112 -0
- data/Rakefile +243 -0
- data/ext/ffi_c/AbstractMemory.c +1109 -0
- data/ext/ffi_c/AbstractMemory.h +175 -0
- data/ext/ffi_c/ArrayType.c +162 -0
- data/ext/ffi_c/ArrayType.h +59 -0
- data/ext/ffi_c/Buffer.c +365 -0
- data/ext/ffi_c/Call.c +517 -0
- data/ext/ffi_c/Call.h +110 -0
- data/ext/ffi_c/ClosurePool.c +283 -0
- data/ext/ffi_c/ClosurePool.h +57 -0
- data/ext/ffi_c/DataConverter.c +91 -0
- data/ext/ffi_c/DynamicLibrary.c +339 -0
- data/ext/ffi_c/DynamicLibrary.h +98 -0
- data/ext/ffi_c/Function.c +998 -0
- data/ext/ffi_c/Function.h +87 -0
- data/ext/ffi_c/FunctionInfo.c +271 -0
- data/ext/ffi_c/LastError.c +184 -0
- data/ext/ffi_c/LastError.h +47 -0
- data/ext/ffi_c/LongDouble.c +63 -0
- data/ext/ffi_c/LongDouble.h +51 -0
- data/ext/ffi_c/MappedType.c +168 -0
- data/ext/ffi_c/MappedType.h +59 -0
- data/ext/ffi_c/MemoryPointer.c +197 -0
- data/ext/ffi_c/MemoryPointer.h +53 -0
- data/ext/ffi_c/MethodHandle.c +358 -0
- data/ext/ffi_c/MethodHandle.h +55 -0
- data/ext/ffi_c/Platform.c +129 -0
- data/ext/ffi_c/Platform.h +45 -0
- data/ext/ffi_c/Pointer.c +508 -0
- data/ext/ffi_c/Pointer.h +63 -0
- data/ext/ffi_c/Struct.c +829 -0
- data/ext/ffi_c/Struct.h +106 -0
- data/ext/ffi_c/StructByReference.c +190 -0
- data/ext/ffi_c/StructByReference.h +50 -0
- data/ext/ffi_c/StructByValue.c +150 -0
- data/ext/ffi_c/StructByValue.h +55 -0
- data/ext/ffi_c/StructLayout.c +698 -0
- data/ext/ffi_c/Thread.c +352 -0
- data/ext/ffi_c/Thread.h +95 -0
- data/ext/ffi_c/Type.c +397 -0
- data/ext/ffi_c/Type.h +62 -0
- data/ext/ffi_c/Types.c +139 -0
- data/ext/ffi_c/Types.h +89 -0
- data/ext/ffi_c/Variadic.c +304 -0
- data/ext/ffi_c/compat.h +78 -0
- data/ext/ffi_c/extconf.rb +71 -0
- data/ext/ffi_c/ffi.c +98 -0
- data/ext/ffi_c/libffi.bsd.mk +40 -0
- data/ext/ffi_c/libffi.darwin.mk +105 -0
- data/ext/ffi_c/libffi.gnu.mk +32 -0
- data/ext/ffi_c/libffi.mk +18 -0
- data/ext/ffi_c/libffi.vc.mk +26 -0
- data/ext/ffi_c/libffi.vc64.mk +26 -0
- data/ext/ffi_c/rbffi.h +57 -0
- data/ext/ffi_c/rbffi_endian.h +59 -0
- data/ext/ffi_c/win32/stdbool.h +8 -0
- data/ext/ffi_c/win32/stdint.h +201 -0
- data/ffi.gemspec +23 -0
- data/gen/Rakefile +30 -0
- data/lib/ffi.rb +20 -0
- data/lib/ffi/autopointer.rb +203 -0
- data/lib/ffi/buffer.rb +4 -0
- data/lib/ffi/callback.rb +4 -0
- data/lib/ffi/enum.rb +296 -0
- data/lib/ffi/errno.rb +43 -0
- data/lib/ffi/ffi.rb +44 -0
- data/lib/ffi/io.rb +62 -0
- data/lib/ffi/library.rb +590 -0
- data/lib/ffi/managedstruct.rb +84 -0
- data/lib/ffi/memorypointer.rb +1 -0
- data/lib/ffi/platform.rb +164 -0
- data/lib/ffi/platform/aarch64-linux/types.conf +104 -0
- data/lib/ffi/platform/arm-linux/types.conf +104 -0
- data/lib/ffi/platform/i386-cygwin/types.conf +3 -0
- data/lib/ffi/platform/i386-darwin/types.conf +100 -0
- data/lib/ffi/platform/i386-freebsd/types.conf +152 -0
- data/lib/ffi/platform/i386-gnu/types.conf +107 -0
- data/lib/ffi/platform/i386-linux/types.conf +103 -0
- data/lib/ffi/platform/i386-netbsd/types.conf +126 -0
- data/lib/ffi/platform/i386-openbsd/types.conf +128 -0
- data/lib/ffi/platform/i386-solaris/types.conf +122 -0
- data/lib/ffi/platform/i386-windows/types.conf +105 -0
- data/lib/ffi/platform/ia64-linux/types.conf +104 -0
- data/lib/ffi/platform/mips-linux/types.conf +102 -0
- data/lib/ffi/platform/mips64el-linux/types.conf +104 -0
- data/lib/ffi/platform/mipsel-linux/types.conf +102 -0
- data/lib/ffi/platform/powerpc-aix/types.conf +180 -0
- data/lib/ffi/platform/powerpc-darwin/types.conf +100 -0
- data/lib/ffi/platform/powerpc-linux/types.conf +100 -0
- data/lib/ffi/platform/powerpc64-linux/types.conf +104 -0
- data/lib/ffi/platform/s390-linux/types.conf +102 -0
- data/lib/ffi/platform/s390x-linux/types.conf +102 -0
- data/lib/ffi/platform/sparc-linux/types.conf +102 -0
- data/lib/ffi/platform/sparc-solaris/types.conf +128 -0
- data/lib/ffi/platform/sparc64-linux/types.conf +102 -0
- data/lib/ffi/platform/sparcv9-solaris/types.conf +128 -0
- data/lib/ffi/platform/x86_64-cygwin/types.conf +3 -0
- data/lib/ffi/platform/x86_64-darwin/types.conf +126 -0
- data/lib/ffi/platform/x86_64-freebsd/types.conf +128 -0
- data/lib/ffi/platform/x86_64-linux/types.conf +102 -0
- data/lib/ffi/platform/x86_64-netbsd/types.conf +128 -0
- data/lib/ffi/platform/x86_64-openbsd/types.conf +134 -0
- data/lib/ffi/platform/x86_64-solaris/types.conf +122 -0
- data/lib/ffi/platform/x86_64-windows/types.conf +120 -0
- data/lib/ffi/pointer.rb +161 -0
- data/lib/ffi/struct.rb +371 -0
- data/lib/ffi/struct_layout_builder.rb +227 -0
- data/lib/ffi/tools/const_generator.rb +229 -0
- data/lib/ffi/tools/generator.rb +60 -0
- data/lib/ffi/tools/generator_task.rb +36 -0
- data/lib/ffi/tools/struct_generator.rb +194 -0
- data/lib/ffi/tools/types_generator.rb +134 -0
- data/lib/ffi/types.rb +194 -0
- data/lib/ffi/union.rb +43 -0
- data/lib/ffi/variadic.rb +78 -0
- data/lib/ffi/version.rb +4 -0
- data/libtest/Benchmark.c +52 -0
- data/libtest/BoolTest.c +34 -0
- data/libtest/BufferTest.c +31 -0
- data/libtest/ClosureTest.c +205 -0
- data/libtest/EnumTest.c +51 -0
- data/libtest/FunctionTest.c +70 -0
- data/libtest/GNUmakefile +149 -0
- data/libtest/GlobalVariable.c +62 -0
- data/libtest/LastErrorTest.c +21 -0
- data/libtest/NumberTest.c +132 -0
- data/libtest/PointerTest.c +63 -0
- data/libtest/ReferenceTest.c +23 -0
- data/libtest/StringTest.c +34 -0
- data/libtest/StructTest.c +243 -0
- data/libtest/UnionTest.c +43 -0
- data/libtest/VariadicTest.c +99 -0
- data/spec/ffi/LICENSE.SPECS +22 -0
- data/spec/ffi/async_callback_spec.rb +35 -0
- data/spec/ffi/bitmask_spec.rb +575 -0
- data/spec/ffi/bool_spec.rb +32 -0
- data/spec/ffi/buffer_spec.rb +279 -0
- data/spec/ffi/callback_spec.rb +773 -0
- data/spec/ffi/custom_param_type.rb +37 -0
- data/spec/ffi/custom_type_spec.rb +74 -0
- data/spec/ffi/dup_spec.rb +52 -0
- data/spec/ffi/enum_spec.rb +423 -0
- data/spec/ffi/errno_spec.rb +20 -0
- data/spec/ffi/ffi_spec.rb +28 -0
- data/spec/ffi/fixtures/Benchmark.c +52 -0
- data/spec/ffi/fixtures/BitmaskTest.c +51 -0
- data/spec/ffi/fixtures/BoolTest.c +34 -0
- data/spec/ffi/fixtures/BufferTest.c +31 -0
- data/spec/ffi/fixtures/ClosureTest.c +205 -0
- data/spec/ffi/fixtures/EnumTest.c +51 -0
- data/spec/ffi/fixtures/FunctionTest.c +142 -0
- data/spec/ffi/fixtures/GNUmakefile +149 -0
- data/spec/ffi/fixtures/GlobalVariable.c +62 -0
- data/spec/ffi/fixtures/LastErrorTest.c +21 -0
- data/spec/ffi/fixtures/NumberTest.c +132 -0
- data/spec/ffi/fixtures/PipeHelper.h +21 -0
- data/spec/ffi/fixtures/PipeHelperPosix.c +41 -0
- data/spec/ffi/fixtures/PipeHelperWindows.c +72 -0
- data/spec/ffi/fixtures/PointerTest.c +63 -0
- data/spec/ffi/fixtures/ReferenceTest.c +23 -0
- data/spec/ffi/fixtures/StringTest.c +34 -0
- data/spec/ffi/fixtures/StructTest.c +243 -0
- data/spec/ffi/fixtures/UnionTest.c +43 -0
- data/spec/ffi/fixtures/VariadicTest.c +99 -0
- data/spec/ffi/fixtures/classes.rb +438 -0
- data/spec/ffi/function_spec.rb +97 -0
- data/spec/ffi/io_spec.rb +16 -0
- data/spec/ffi/library_spec.rb +286 -0
- data/spec/ffi/long_double.rb +30 -0
- data/spec/ffi/managed_struct_spec.rb +68 -0
- data/spec/ffi/memorypointer_spec.rb +78 -0
- data/spec/ffi/number_spec.rb +247 -0
- data/spec/ffi/platform_spec.rb +114 -0
- data/spec/ffi/pointer_spec.rb +285 -0
- data/spec/ffi/rbx/attach_function_spec.rb +34 -0
- data/spec/ffi/rbx/memory_pointer_spec.rb +198 -0
- data/spec/ffi/rbx/spec_helper.rb +6 -0
- data/spec/ffi/rbx/struct_spec.rb +18 -0
- data/spec/ffi/spec_helper.rb +93 -0
- data/spec/ffi/string_spec.rb +118 -0
- data/spec/ffi/strptr_spec.rb +50 -0
- data/spec/ffi/struct_by_ref_spec.rb +43 -0
- data/spec/ffi/struct_callback_spec.rb +69 -0
- data/spec/ffi/struct_initialize_spec.rb +35 -0
- data/spec/ffi/struct_packed_spec.rb +50 -0
- data/spec/ffi/struct_spec.rb +882 -0
- data/spec/ffi/typedef_spec.rb +91 -0
- data/spec/ffi/union_spec.rb +67 -0
- data/spec/ffi/variadic_spec.rb +132 -0
- data/spec/spec.opts +4 -0
- metadata +309 -0
data/ext/ffi_c/Call.c
ADDED
@@ -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 = ¶mStorage[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
|
+
|