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,87 @@
1
+ /*
2
+ * Copyright (c) 2009, Wayne Meissner
3
+ *
4
+ * Copyright (c) 2008-2013, Ruby FFI project contributors
5
+ * All rights reserved.
6
+ *
7
+ * Redistribution and use in source and binary forms, with or without
8
+ * modification, are permitted provided that the following conditions are met:
9
+ * * Redistributions of source code must retain the above copyright
10
+ * notice, this list of conditions and the following disclaimer.
11
+ * * Redistributions in binary form must reproduce the above copyright
12
+ * notice, this list of conditions and the following disclaimer in the
13
+ * documentation and/or other materials provided with the distribution.
14
+ * * Neither the name of the Ruby FFI project nor the
15
+ * names of its contributors may be used to endorse or promote products
16
+ * derived from this software without specific prior written permission.
17
+ *
18
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
19
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
20
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
21
+ * DISCLAIMED. IN NO EVENT SHALL <COPYRIGHT HOLDER> BE LIABLE FOR ANY
22
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
23
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
24
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
25
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
26
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
27
+ * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28
+ */
29
+
30
+ #ifndef RBFFI_FUNCTION_H
31
+ #define RBFFI_FUNCTION_H
32
+
33
+ #ifdef __cplusplus
34
+ extern "C" {
35
+ #endif
36
+
37
+ #ifndef _MSC_VER
38
+ # include <stdbool.h>
39
+ #else
40
+ # include "win32/stdbool.h"
41
+ #endif
42
+
43
+ #include <ffi.h>
44
+
45
+ typedef struct FunctionType_ FunctionType;
46
+
47
+ #include "Type.h"
48
+ #include "Call.h"
49
+ #include "ClosurePool.h"
50
+
51
+ struct FunctionType_ {
52
+ Type type; /* The native type of a FunctionInfo object */
53
+ VALUE rbReturnType;
54
+ VALUE rbParameterTypes;
55
+
56
+ Type* returnType;
57
+ Type** parameterTypes;
58
+ NativeType* nativeParameterTypes;
59
+ ffi_type* ffiReturnType;
60
+ ffi_type** ffiParameterTypes;
61
+ ffi_cif ffi_cif;
62
+ Invoker invoke;
63
+ ClosurePool* closurePool;
64
+ int parameterCount;
65
+ int flags;
66
+ ffi_abi abi;
67
+ int callbackCount;
68
+ VALUE* callbackParameters;
69
+ VALUE rbEnums;
70
+ bool ignoreErrno;
71
+ bool blocking;
72
+ bool hasStruct;
73
+ };
74
+
75
+ extern VALUE rbffi_FunctionTypeClass, rbffi_FunctionClass;
76
+
77
+ void rbffi_Function_Init(VALUE moduleFFI);
78
+ VALUE rbffi_Function_NewInstance(VALUE functionInfo, VALUE proc);
79
+ VALUE rbffi_Function_ForProc(VALUE cbInfo, VALUE proc);
80
+ void rbffi_FunctionInfo_Init(VALUE moduleFFI);
81
+
82
+ #ifdef __cplusplus
83
+ }
84
+ #endif
85
+
86
+ #endif /* RBFFI_FUNCTION_H */
87
+
@@ -0,0 +1,271 @@
1
+ /*
2
+ * Copyright (c) 2009, Wayne Meissner
3
+ * Copyright (C) 2009 Andrea Fazzi <andrea.fazzi@alcacoop.it>
4
+ * Copyright (c) 2008-2013, Ruby FFI project contributors
5
+ * All rights reserved.
6
+ *
7
+ * Redistribution and use in source and binary forms, with or without
8
+ * modification, are permitted provided that the following conditions are met:
9
+ * * Redistributions of source code must retain the above copyright
10
+ * notice, this list of conditions and the following disclaimer.
11
+ * * Redistributions in binary form must reproduce the above copyright
12
+ * notice, this list of conditions and the following disclaimer in the
13
+ * documentation and/or other materials provided with the distribution.
14
+ * * Neither the name of the Ruby FFI project nor the
15
+ * names of its contributors may be used to endorse or promote products
16
+ * derived from this software without specific prior written permission.
17
+ *
18
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
19
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
20
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
21
+ * DISCLAIMED. IN NO EVENT SHALL <COPYRIGHT HOLDER> BE LIABLE FOR ANY
22
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
23
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
24
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
25
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
26
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
27
+ * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28
+ */
29
+
30
+ #ifndef _MSC_VER
31
+ # include <sys/param.h>
32
+ #endif
33
+ #include <sys/types.h>
34
+ #include <stdio.h>
35
+
36
+ #ifndef _MSC_VER
37
+ # include <stdint.h>
38
+ # include <stdbool.h>
39
+ #else
40
+ # include "win32/stdbool.h"
41
+ #endif
42
+
43
+ #include <errno.h>
44
+ #include <ruby.h>
45
+
46
+ #include <ffi.h>
47
+ #include "rbffi.h"
48
+ #include "compat.h"
49
+
50
+ #include "AbstractMemory.h"
51
+ #include "Types.h"
52
+ #include "Type.h"
53
+ #include "StructByValue.h"
54
+ #include "Function.h"
55
+
56
+ static VALUE fntype_allocate(VALUE klass);
57
+ static VALUE fntype_initialize(int argc, VALUE* argv, VALUE self);
58
+ static void fntype_mark(FunctionType*);
59
+ static void fntype_free(FunctionType *);
60
+
61
+ VALUE rbffi_FunctionTypeClass = Qnil;
62
+
63
+ static VALUE
64
+ fntype_allocate(VALUE klass)
65
+ {
66
+ FunctionType* fnInfo;
67
+ VALUE obj = Data_Make_Struct(klass, FunctionType, fntype_mark, fntype_free, fnInfo);
68
+
69
+ fnInfo->type.ffiType = &ffi_type_pointer;
70
+ fnInfo->type.nativeType = NATIVE_FUNCTION;
71
+ fnInfo->rbReturnType = Qnil;
72
+ fnInfo->rbParameterTypes = Qnil;
73
+ fnInfo->rbEnums = Qnil;
74
+ fnInfo->invoke = rbffi_CallFunction;
75
+ fnInfo->closurePool = NULL;
76
+
77
+ return obj;
78
+ }
79
+
80
+ static void
81
+ fntype_mark(FunctionType* fnInfo)
82
+ {
83
+ rb_gc_mark(fnInfo->rbReturnType);
84
+ rb_gc_mark(fnInfo->rbParameterTypes);
85
+ rb_gc_mark(fnInfo->rbEnums);
86
+ if (fnInfo->callbackCount > 0 && fnInfo->callbackParameters != NULL) {
87
+ rb_gc_mark_locations(&fnInfo->callbackParameters[0], &fnInfo->callbackParameters[fnInfo->callbackCount]);
88
+ }
89
+ }
90
+
91
+ static void
92
+ fntype_free(FunctionType* fnInfo)
93
+ {
94
+ xfree(fnInfo->parameterTypes);
95
+ xfree(fnInfo->ffiParameterTypes);
96
+ xfree(fnInfo->nativeParameterTypes);
97
+ xfree(fnInfo->callbackParameters);
98
+ if (fnInfo->closurePool != NULL) {
99
+ rbffi_ClosurePool_Free(fnInfo->closurePool);
100
+ }
101
+ xfree(fnInfo);
102
+ }
103
+
104
+ /*
105
+ * call-seq: initialize(return_type, param_types, options={})
106
+ * @param [Type, Symbol] return_type return type for the function
107
+ * @param [Array<Type, Symbol>] param_types array of parameters types
108
+ * @param [Hash] options
109
+ * @option options [Boolean] :blocking set to true if the C function is a blocking call
110
+ * @option options [Symbol] :convention calling convention see {FFI::Library#calling_convention}
111
+ * @option options [FFI::Enums] :enums
112
+ * @return [self]
113
+ * A new FunctionType instance.
114
+ */
115
+ static VALUE
116
+ fntype_initialize(int argc, VALUE* argv, VALUE self)
117
+ {
118
+ FunctionType *fnInfo;
119
+ ffi_status status;
120
+ VALUE rbReturnType = Qnil, rbParamTypes = Qnil, rbOptions = Qnil;
121
+ VALUE rbEnums = Qnil, rbConvention = Qnil, rbBlocking = Qnil;
122
+ #if defined(X86_WIN32)
123
+ VALUE rbConventionStr;
124
+ #endif
125
+ int i, nargs;
126
+
127
+ nargs = rb_scan_args(argc, argv, "21", &rbReturnType, &rbParamTypes, &rbOptions);
128
+ if (nargs >= 3 && rbOptions != Qnil) {
129
+ rbConvention = rb_hash_aref(rbOptions, ID2SYM(rb_intern("convention")));
130
+ rbEnums = rb_hash_aref(rbOptions, ID2SYM(rb_intern("enums")));
131
+ rbBlocking = rb_hash_aref(rbOptions, ID2SYM(rb_intern("blocking")));
132
+ }
133
+
134
+ Check_Type(rbParamTypes, T_ARRAY);
135
+
136
+ Data_Get_Struct(self, FunctionType, fnInfo);
137
+ fnInfo->parameterCount = (int) RARRAY_LEN(rbParamTypes);
138
+ fnInfo->parameterTypes = xcalloc(fnInfo->parameterCount, sizeof(*fnInfo->parameterTypes));
139
+ fnInfo->ffiParameterTypes = xcalloc(fnInfo->parameterCount, sizeof(ffi_type *));
140
+ fnInfo->nativeParameterTypes = xcalloc(fnInfo->parameterCount, sizeof(*fnInfo->nativeParameterTypes));
141
+ fnInfo->rbParameterTypes = rb_ary_new2(fnInfo->parameterCount);
142
+ fnInfo->rbEnums = rbEnums;
143
+ fnInfo->blocking = RTEST(rbBlocking);
144
+ fnInfo->hasStruct = false;
145
+
146
+ for (i = 0; i < fnInfo->parameterCount; ++i) {
147
+ VALUE entry = rb_ary_entry(rbParamTypes, i);
148
+ VALUE type = rbffi_Type_Lookup(entry);
149
+
150
+ if (!RTEST(type)) {
151
+ VALUE typeName = rb_funcall2(entry, rb_intern("inspect"), 0, NULL);
152
+ rb_raise(rb_eTypeError, "Invalid parameter type (%s)", RSTRING_PTR(typeName));
153
+ }
154
+
155
+ if (rb_obj_is_kind_of(type, rbffi_FunctionTypeClass)) {
156
+ REALLOC_N(fnInfo->callbackParameters, VALUE, fnInfo->callbackCount + 1);
157
+ fnInfo->callbackParameters[fnInfo->callbackCount++] = type;
158
+ }
159
+
160
+ if (rb_obj_is_kind_of(type, rbffi_StructByValueClass)) {
161
+ fnInfo->hasStruct = true;
162
+ }
163
+
164
+ rb_ary_push(fnInfo->rbParameterTypes, type);
165
+ Data_Get_Struct(type, Type, fnInfo->parameterTypes[i]);
166
+ fnInfo->ffiParameterTypes[i] = fnInfo->parameterTypes[i]->ffiType;
167
+ fnInfo->nativeParameterTypes[i] = fnInfo->parameterTypes[i]->nativeType;
168
+ }
169
+
170
+ fnInfo->rbReturnType = rbffi_Type_Lookup(rbReturnType);
171
+ if (!RTEST(fnInfo->rbReturnType)) {
172
+ VALUE typeName = rb_funcall2(rbReturnType, rb_intern("inspect"), 0, NULL);
173
+ rb_raise(rb_eTypeError, "Invalid return type (%s)", RSTRING_PTR(typeName));
174
+ }
175
+
176
+ if (rb_obj_is_kind_of(fnInfo->rbReturnType, rbffi_StructByValueClass)) {
177
+ fnInfo->hasStruct = true;
178
+ }
179
+
180
+ Data_Get_Struct(fnInfo->rbReturnType, Type, fnInfo->returnType);
181
+ fnInfo->ffiReturnType = fnInfo->returnType->ffiType;
182
+
183
+
184
+ #if defined(X86_WIN32)
185
+ rbConventionStr = (rbConvention != Qnil) ? rb_funcall2(rbConvention, rb_intern("to_s"), 0, NULL) : Qnil;
186
+ fnInfo->abi = (rbConventionStr != Qnil && strcmp(StringValueCStr(rbConventionStr), "stdcall") == 0)
187
+ ? FFI_STDCALL : FFI_DEFAULT_ABI;
188
+ #else
189
+ fnInfo->abi = FFI_DEFAULT_ABI;
190
+ #endif
191
+
192
+ status = ffi_prep_cif(&fnInfo->ffi_cif, fnInfo->abi, fnInfo->parameterCount,
193
+ fnInfo->ffiReturnType, fnInfo->ffiParameterTypes);
194
+ switch (status) {
195
+ case FFI_BAD_ABI:
196
+ rb_raise(rb_eArgError, "Invalid ABI specified");
197
+ case FFI_BAD_TYPEDEF:
198
+ rb_raise(rb_eArgError, "Invalid argument type specified");
199
+ case FFI_OK:
200
+ break;
201
+ default:
202
+ rb_raise(rb_eArgError, "Unknown FFI error");
203
+ }
204
+
205
+ fnInfo->invoke = rbffi_GetInvoker(fnInfo);
206
+
207
+ return self;
208
+ }
209
+
210
+ /*
211
+ * call-seq: result_type
212
+ * @return [Type]
213
+ * Get the return type of the function type
214
+ */
215
+ static VALUE
216
+ fntype_result_type(VALUE self)
217
+ {
218
+ FunctionType* ft;
219
+
220
+ Data_Get_Struct(self, FunctionType, ft);
221
+
222
+ return ft->rbReturnType;
223
+ }
224
+
225
+ /*
226
+ * call-seq: param_types
227
+ * @return [Array<Type>]
228
+ * Get parameters types.
229
+ */
230
+ static VALUE
231
+ fntype_param_types(VALUE self)
232
+ {
233
+ FunctionType* ft;
234
+
235
+ Data_Get_Struct(self, FunctionType, ft);
236
+
237
+ return rb_ary_dup(ft->rbParameterTypes);
238
+ }
239
+
240
+ void
241
+ rbffi_FunctionInfo_Init(VALUE moduleFFI)
242
+ {
243
+ VALUE ffi_Type;
244
+
245
+ ffi_Type = rbffi_TypeClass;
246
+
247
+ /*
248
+ * Document-class: FFI::FunctionType < FFI::Type
249
+ */
250
+ rbffi_FunctionTypeClass = rb_define_class_under(moduleFFI, "FunctionType",ffi_Type);
251
+ rb_global_variable(&rbffi_FunctionTypeClass);
252
+ /*
253
+ * Document-const: FFI::CallbackInfo = FFI::FunctionType
254
+ */
255
+ rb_define_const(moduleFFI, "CallbackInfo", rbffi_FunctionTypeClass);
256
+ /*
257
+ * Document-const: FFI::FunctionInfo = FFI::FunctionType
258
+ */
259
+ rb_define_const(moduleFFI, "FunctionInfo", rbffi_FunctionTypeClass);
260
+ /*
261
+ * Document-const: FFI::Type::Function = FFI::FunctionType
262
+ */
263
+ rb_define_const(ffi_Type, "Function", rbffi_FunctionTypeClass);
264
+
265
+ rb_define_alloc_func(rbffi_FunctionTypeClass, fntype_allocate);
266
+ rb_define_method(rbffi_FunctionTypeClass, "initialize", fntype_initialize, -1);
267
+ rb_define_method(rbffi_FunctionTypeClass, "result_type", fntype_result_type, 0);
268
+ rb_define_method(rbffi_FunctionTypeClass, "param_types", fntype_param_types, 0);
269
+
270
+ }
271
+
@@ -0,0 +1,184 @@
1
+ /*
2
+ * Copyright (c) 2008, 2009, Wayne Meissner
3
+ * Copyright (C) 2009 Aman Gupta <aman@tmm1.net>
4
+ *
5
+ * Copyright (c) 2008-2013, Ruby FFI project contributors
6
+ * All rights reserved.
7
+ *
8
+ * Redistribution and use in source and binary forms, with or without
9
+ * modification, are permitted provided that the following conditions are met:
10
+ * * Redistributions of source code must retain the above copyright
11
+ * notice, this list of conditions and the following disclaimer.
12
+ * * Redistributions in binary form must reproduce the above copyright
13
+ * notice, this list of conditions and the following disclaimer in the
14
+ * documentation and/or other materials provided with the distribution.
15
+ * * Neither the name of the Ruby FFI project nor the
16
+ * names of its contributors may be used to endorse or promote products
17
+ * derived from this software without specific prior written permission.
18
+ *
19
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
20
+ * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
21
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
22
+ * DISCLAIMED. IN NO EVENT SHALL <COPYRIGHT HOLDER> BE LIABLE FOR ANY
23
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
24
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
25
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
26
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
27
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
28
+ * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29
+ */
30
+
31
+ #ifndef _MSC_VER
32
+ # include <sys/param.h>
33
+ #endif
34
+ #include <sys/types.h>
35
+ #include <stdio.h>
36
+ #ifndef _MSC_VER
37
+ # include <stdint.h>
38
+ # include <stdbool.h>
39
+ #else
40
+ # include "win32/stdbool.h"
41
+ #endif
42
+ #include <errno.h>
43
+ #include <ruby.h>
44
+
45
+ #include "LastError.h"
46
+
47
+ #if defined(HAVE_NATIVETHREAD) && !defined(_WIN32) && !defined(__WIN32__)
48
+ # include <pthread.h>
49
+ # define USE_PTHREAD_LOCAL
50
+ #endif
51
+
52
+ typedef struct ThreadData {
53
+ int td_errno;
54
+ } ThreadData;
55
+
56
+ #if defined(USE_PTHREAD_LOCAL)
57
+ static pthread_key_t threadDataKey;
58
+ #endif
59
+
60
+ static inline ThreadData* thread_data_get(void);
61
+
62
+ #if defined(USE_PTHREAD_LOCAL)
63
+
64
+ static ThreadData*
65
+ thread_data_init(void)
66
+ {
67
+ ThreadData* td = xcalloc(1, sizeof(ThreadData));
68
+
69
+ pthread_setspecific(threadDataKey, td);
70
+
71
+ return td;
72
+ }
73
+
74
+
75
+ static inline ThreadData*
76
+ thread_data_get(void)
77
+ {
78
+ ThreadData* td = pthread_getspecific(threadDataKey);
79
+ return td != NULL ? td : thread_data_init();
80
+ }
81
+
82
+ static void
83
+ thread_data_free(void *ptr)
84
+ {
85
+ xfree(ptr);
86
+ }
87
+
88
+ #else
89
+ static ID id_thread_data;
90
+
91
+ static ThreadData*
92
+ thread_data_init(void)
93
+ {
94
+ ThreadData* td;
95
+ VALUE obj;
96
+
97
+ obj = Data_Make_Struct(rb_cObject, ThreadData, NULL, -1, td);
98
+ rb_thread_local_aset(rb_thread_current(), id_thread_data, obj);
99
+
100
+ return td;
101
+ }
102
+
103
+ static inline ThreadData*
104
+ thread_data_get()
105
+ {
106
+ VALUE obj = rb_thread_local_aref(rb_thread_current(), id_thread_data);
107
+
108
+ if (obj != Qnil && TYPE(obj) == T_DATA) {
109
+ return (ThreadData *) DATA_PTR(obj);
110
+ }
111
+
112
+ return thread_data_init();
113
+ }
114
+
115
+ #endif
116
+
117
+
118
+ /*
119
+ * call-seq: error
120
+ * @return [Numeric]
121
+ * Get +errno+ value.
122
+ */
123
+ static VALUE
124
+ get_last_error(VALUE self)
125
+ {
126
+ return INT2NUM(thread_data_get()->td_errno);
127
+ }
128
+
129
+
130
+ /*
131
+ * call-seq: error(error)
132
+ * @param [Numeric] error
133
+ * @return [nil]
134
+ * Set +errno+ value.
135
+ */
136
+ static VALUE
137
+ set_last_error(VALUE self, VALUE error)
138
+ {
139
+
140
+ #ifdef _WIN32
141
+ SetLastError(NUM2INT(error));
142
+ #else
143
+ errno = NUM2INT(error);
144
+ #endif
145
+
146
+ return Qnil;
147
+ }
148
+
149
+
150
+ void
151
+ rbffi_save_errno(void)
152
+ {
153
+ int error = 0;
154
+
155
+ #ifdef _WIN32
156
+ error = GetLastError();
157
+ #else
158
+ error = errno;
159
+ #endif
160
+
161
+ thread_data_get()->td_errno = error;
162
+ }
163
+
164
+
165
+ void
166
+ rbffi_LastError_Init(VALUE moduleFFI)
167
+ {
168
+ /*
169
+ * Document-module: FFI::LastError
170
+ * This module defines a couple of method to set and get +errno+
171
+ * for current thread.
172
+ */
173
+ VALUE moduleError = rb_define_module_under(moduleFFI, "LastError");
174
+
175
+ rb_define_module_function(moduleError, "error", get_last_error, 0);
176
+ rb_define_module_function(moduleError, "error=", set_last_error, 1);
177
+
178
+ #if defined(USE_PTHREAD_LOCAL)
179
+ pthread_key_create(&threadDataKey, thread_data_free);
180
+ #else
181
+ id_thread_data = rb_intern("ffi_thread_local_data");
182
+ #endif /* USE_PTHREAD_LOCAL */
183
+ }
184
+