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,78 @@
1
+ #
2
+ # This file is part of ruby-ffi.
3
+ # For licensing, see LICENSE.SPECS
4
+ #
5
+
6
+ require File.expand_path(File.join(File.dirname(__FILE__), "spec_helper"))
7
+
8
+ MemoryPointer = FFI::MemoryPointer
9
+
10
+ describe "MemoryPointer#total" do
11
+ it "MemoryPointer.new(:char, 1).total == 1" do
12
+ expect(MemoryPointer.new(:char, 1).total).to eq 1
13
+ end
14
+
15
+ it "MemoryPointer.new(:short, 1).total == 2" do
16
+ expect(MemoryPointer.new(:short, 1).total).to eq 2
17
+ end
18
+
19
+ it "MemoryPointer.new(:int, 1).total == 4" do
20
+ expect(MemoryPointer.new(:int, 1).total).to eq 4
21
+ end
22
+
23
+ it "MemoryPointer.new(:long_long, 1).total == 8" do
24
+ expect(MemoryPointer.new(:long_long, 1).total).to eq 8
25
+ end
26
+
27
+ it "MemoryPointer.new(1024).total == 1024" do
28
+ expect(MemoryPointer.new(1024).total).to eq 1024
29
+ end
30
+ end
31
+ describe "MemoryPointer#read_array_of_long" do
32
+ it "foo" do
33
+ ptr = MemoryPointer.new(:long, 1024)
34
+ ptr[0].write_long 1234
35
+ ptr[1].write_long 5678
36
+ l = ptr.read_array_of_long(2)
37
+ expect(l[0]).to eq 1234
38
+ expect(l[1]).to eq 5678
39
+ end
40
+ end
41
+ describe "MemoryPointer argument" do
42
+ module Ptr
43
+ extend FFI::Library
44
+ ffi_lib FFI::Platform::LIBC
45
+ attach_function :memset, [ :pointer, :int, :ulong ], :pointer
46
+ attach_function :memcpy, [ :pointer, :pointer, :ulong ], :pointer
47
+ end
48
+
49
+ it "Pointer passed correctly" do
50
+ p = MemoryPointer.new :int, 1
51
+ ret = Ptr.memset(p, 0, p.total)
52
+ expect(ret).to eq p
53
+ end
54
+
55
+ it "Data passed to native function" do
56
+ p = MemoryPointer.new :int, 1
57
+ p2 = MemoryPointer.new :int, 1
58
+ p2.put_int(0, 0x5eadbeef)
59
+ Ptr.memcpy(p, p2, p.total)
60
+ expect(p.get_int(0)).to eq p2.get_int(0)
61
+ expect(p2.get_int(0)).not_to eql 0
62
+ end
63
+ end
64
+ describe "MemoryPointer return value" do
65
+ module Stdio
66
+ extend FFI::Library
67
+ ffi_lib FFI::Platform::LIBC
68
+ attach_function :fopen, [ :string, :string ], :pointer
69
+ attach_function :fclose, [ :pointer ], :int
70
+ attach_function :fwrite, [ :pointer, :ulong, :ulong, :string ], :ulong
71
+ end
72
+
73
+ it "fopen returns non-nil" do
74
+ fp = Stdio.fopen("/dev/null", "w")
75
+ expect(fp).to_not be_nil
76
+ expect(Stdio.fclose(fp)).to eq 0 unless fp.nil? or fp.null?
77
+ end
78
+ end
@@ -0,0 +1,247 @@
1
+ #
2
+ # This file is part of ruby-ffi.
3
+ # For licensing, see LICENSE.SPECS
4
+ #
5
+
6
+ require File.expand_path(File.join(File.dirname(__FILE__), "spec_helper"))
7
+
8
+ describe "Function with primitive integer arguments" do
9
+ module LibTest
10
+ extend FFI::Library
11
+ ffi_lib TestLibrary::PATH
12
+ attach_function :ret_s8, [ :char ], :char
13
+ attach_function :ret_u8, [ :uchar ], :uchar
14
+ attach_function :ret_s16, [ :short ], :short
15
+ attach_function :ret_u16, [ :ushort ], :ushort
16
+ attach_function :ret_s32, [ :int ], :int
17
+ attach_function :ret_u32, [ :uint ], :uint
18
+ attach_function :ret_s64, [ :long_long ], :long_long
19
+ attach_function :ret_u64, [ :ulong_long ], :ulong_long
20
+ attach_function :ret_long, [ :long ], :long
21
+ attach_function :ret_ulong, [ :ulong ], :ulong
22
+ attach_function :set_s8, [ :char ], :void
23
+ attach_function :get_s8, [ ], :char
24
+ attach_function :set_float, [ :float ], :void
25
+ attach_function :get_float, [ ], :float
26
+ attach_function :set_double, [ :double ], :void
27
+ attach_function :get_double, [ ], :double
28
+ end
29
+
30
+ it "int8.size" do
31
+ expect(FFI::TYPE_INT8.size).to eq(1)
32
+ end
33
+
34
+ it "uint8.size" do
35
+ expect(FFI::TYPE_UINT8.size).to eq(1)
36
+ end
37
+
38
+ it "int16.size" do
39
+ expect(FFI::TYPE_INT16.size).to eq(2)
40
+ end
41
+
42
+ it "uint16.size" do
43
+ expect(FFI::TYPE_UINT16.size).to eq(2)
44
+ end
45
+
46
+ it "int32.size" do
47
+ expect(FFI::TYPE_INT32.size).to eq(4)
48
+ end
49
+
50
+ it "uint32.size" do
51
+ expect(FFI::TYPE_UINT32.size).to eq(4)
52
+ end
53
+
54
+ it "int64.size" do
55
+ expect(FFI::TYPE_INT64.size).to eq(8)
56
+ end
57
+
58
+ it "uint64.size" do
59
+ expect(FFI::TYPE_UINT64.size).to eq(8)
60
+ end
61
+
62
+ it "float.size" do
63
+ expect(FFI::TYPE_FLOAT32.size).to eq(4)
64
+ end
65
+
66
+ it "double.size" do
67
+ expect(FFI::TYPE_FLOAT64.size).to eq(8)
68
+ end
69
+ [ 0, 127, -128, -1 ].each do |i|
70
+ it ":char call(:char (#{i}))" do
71
+ expect(LibTest.ret_s8(i)).to eq(i)
72
+ end
73
+ end
74
+ [ 0, 0x7f, 0x80, 0xff ].each do |i|
75
+ it ":uchar call(:uchar (#{i}))" do
76
+ expect(LibTest.ret_u8(i)).to eq(i)
77
+ end
78
+ end
79
+ [ 0, 0x7fff, -0x8000, -1 ].each do |i|
80
+ it ":short call(:short (#{i}))" do
81
+ expect(LibTest.ret_s16(i)).to eq(i)
82
+ end
83
+ end
84
+ [ 0, 0x7fff, 0x8000, 0xffff ].each do |i|
85
+ it ":ushort call(:ushort (#{i}))" do
86
+ expect(LibTest.ret_u16(i)).to eq(i)
87
+ end
88
+ end
89
+ [ 0, 0x7fffffff, -0x80000000, -1 ].each do |i|
90
+ it ":int call(:int (#{i}))" do
91
+ expect(LibTest.ret_s32(i)).to eq(i)
92
+ end
93
+ end
94
+ [ 0, 0x7fffffff, 0x80000000, 0xffffffff ].each do |i|
95
+ it ":uint call(:uint (#{i}))" do
96
+ expect(LibTest.ret_u32(i)).to eq(i)
97
+ end
98
+ end
99
+ [ 0, 0x7fffffffffffffff, -0x8000000000000000, -1 ].each do |i|
100
+ it ":long_long call(:long_long (#{i}))" do
101
+ expect(LibTest.ret_s64(i)).to eq(i)
102
+ end
103
+ end
104
+ [ 0, 0x7fffffffffffffff, 0x8000000000000000, 0xffffffffffffffff ].each do |i|
105
+ it ":ulong_long call(:ulong_long (#{i}))" do
106
+ expect(LibTest.ret_u64(i)).to eq(i)
107
+ end
108
+ end
109
+ if FFI::Platform::LONG_SIZE == 32
110
+ [ 0, 0x7fffffff, -0x80000000, -1 ].each do |i|
111
+ it ":long call(:long (#{i}))" do
112
+ expect(LibTest.ret_long(i)).to eq(i)
113
+ end
114
+ end
115
+ [ 0, 0x7fffffff, 0x80000000, 0xffffffff ].each do |i|
116
+ it ":ulong call(:ulong (#{i}))" do
117
+ expect(LibTest.ret_ulong(i)).to eq(i)
118
+ end
119
+ end
120
+ else
121
+ [ 0, 0x7fffffffffffffff, -0x8000000000000000, -1 ].each do |i|
122
+ it ":long call(:long (#{i}))" do
123
+ expect(LibTest.ret_long(i)).to eq(i)
124
+ end
125
+ end
126
+ [ 0, 0x7fffffffffffffff, 0x8000000000000000, 0xffffffffffffffff ].each do |i|
127
+ it ":ulong call(:ulong (#{i}))" do
128
+ expect(LibTest.ret_ulong(i)).to eq(i)
129
+ end
130
+ end
131
+ [ 0.0, 0.1, 1.1, 1.23 ].each do |f|
132
+ it ":float call(:double (#{f}))" do
133
+ LibTest.set_float(f)
134
+ expect((LibTest.get_float - f).abs).to be < 0.001
135
+ end
136
+ end
137
+ [ 0.0, 0.1, 1.1, 1.23 ].each do |f|
138
+ it ":double call(:double (#{f}))" do
139
+ LibTest.set_double(f)
140
+ expect((LibTest.get_double - f).abs).to be < 0.001
141
+ end
142
+ end
143
+ end
144
+ end
145
+ describe "Integer parameter range checking" do
146
+ [ 128, -129 ].each do |i|
147
+ it ":char call(:char (#{i}))" do
148
+ expect { expect(LibTest.ret_int8_t(i)).to eq(i) }.to raise_error
149
+ end
150
+ end
151
+ [ -1, 256 ].each do |i|
152
+ it ":uchar call(:uchar (#{i}))" do
153
+ expect { expect(LibTest.ret_u_int8_t(i)).to eq(i) }.to raise_error
154
+ end
155
+ end
156
+ [ 0x8000, -0x8001 ].each do |i|
157
+ it ":short call(:short (#{i}))" do
158
+ expect { expect(LibTest.ret_int16_t(i)).to eq(i) }.to raise_error
159
+ end
160
+ end
161
+ [ -1, 0x10000 ].each do |i|
162
+ it ":ushort call(:ushort (#{i}))" do
163
+ expect { expect(LibTest.ret_u_int16_t(i)).to eq(i) }.to raise_error
164
+ end
165
+ end
166
+ [ 0x80000000, -0x80000001 ].each do |i|
167
+ it ":int call(:int (#{i}))" do
168
+ expect { expect(LibTest.ret_int32_t(i)).to eq(i) }.to raise_error
169
+ end
170
+ end
171
+ [ -1, 0x100000000 ].each do |i|
172
+ it ":ushort call(:ushort (#{i}))" do
173
+ expect { expect(LibTest.ret_u_int32_t(i)).to eq(i) }.to raise_error
174
+ end
175
+ end
176
+ end
177
+ describe "Three different size Integer arguments" do
178
+ TYPE_MAP = {
179
+ 's8' => :char, 'u8' => :uchar, 's16' => :short, 'u16' => :ushort,
180
+ 's32' => :int, 'u32' => :uint, 's64' => :long_long, 'u64' => :ulong_long,
181
+ 'sL' => :long, 'uL' => :ulong, 'f32' => :float, 'f64' => :double
182
+ }
183
+ TYPES = TYPE_MAP.keys
184
+ module LibTest
185
+ extend FFI::Library
186
+ ffi_lib TestLibrary::PATH
187
+
188
+
189
+ [ 's32', 'u32', 's64', 'u64' ].each do |rt|
190
+ TYPES.each do |t1|
191
+ TYPES.each do |t2|
192
+ TYPES.each do |t3|
193
+ begin
194
+ attach_function "pack_#{t1}#{t2}#{t3}_#{rt}",
195
+ [ TYPE_MAP[t1], TYPE_MAP[t2], TYPE_MAP[t3], :buffer_out ], :void
196
+ rescue FFI::NotFoundError
197
+ end
198
+ end
199
+ end
200
+ end
201
+ end
202
+ end
203
+
204
+ PACK_VALUES = {
205
+ 's8' => [ 0x12 ],
206
+ 'u8' => [ 0x34 ],
207
+ 's16' => [ 0x5678 ],
208
+ 'u16' => [ 0x9abc ],
209
+ 's32' => [ 0x7654321f ],
210
+ 'u32' => [ 0xfee1babe ],
211
+ 'sL' => [ 0x1f2e3d4c ],
212
+ 'uL' => [ 0xf7e8d9ca ],
213
+ 's64' => [ 0x1eafdeadbeefa1b2 ],
214
+ # 'f32' => [ 1.234567 ],
215
+ 'f64' => [ 9.87654321 ]
216
+ }
217
+
218
+ def verify(p, off, t, v)
219
+ if t == 'f32'
220
+ expect(p.get_float32(off)).to eq(v)
221
+ elsif t == 'f64'
222
+ expect(p.get_float64(off)).to eq(v)
223
+ else
224
+ expect(p.get_int64(off)).to eq(v)
225
+ end
226
+ end
227
+
228
+ PACK_VALUES.keys.each do |t1|
229
+ PACK_VALUES.keys.each do |t2|
230
+ PACK_VALUES.keys.each do |t3|
231
+ PACK_VALUES[t1].each do |v1|
232
+ PACK_VALUES[t2].each do |v2|
233
+ PACK_VALUES[t3].each do |v3|
234
+ it "call(#{TYPE_MAP[t1]} (#{v1}), #{TYPE_MAP[t2]} (#{v2}), #{TYPE_MAP[t3]} (#{v3}))" do
235
+ p = FFI::Buffer.new :long_long, 3
236
+ LibTest.send("pack_#{t1}#{t2}#{t3}_s64", v1, v2, v3, p)
237
+ verify(p, 0, t1, v1)
238
+ verify(p, 8, t2, v2)
239
+ verify(p, 16, t3, v3)
240
+ end
241
+ end
242
+ end
243
+ end
244
+ end
245
+ end
246
+ end
247
+ end
@@ -0,0 +1,114 @@
1
+ #
2
+ # This file is part of ruby-ffi.
3
+ # For licensing, see LICENSE.SPECS
4
+ #
5
+
6
+ require File.expand_path(File.join(File.dirname(__FILE__), "spec_helper"))
7
+
8
+ describe "FFI::Platform::LIBSUFFIX" do
9
+ case OS
10
+ when "linux"
11
+ it "returns 'so'" do
12
+ expect(FFI::Platform::LIBSUFFIX).to eq('so')
13
+ end
14
+ when "windows"
15
+ it "returns 'dll'" do
16
+ expect(FFI::Platform::LIBSUFFIX).to eq('dll')
17
+ end
18
+ when "darwin"
19
+ it "returns 'dylib'" do
20
+ expect(FFI::Platform::LIBSUFFIX).to eq('dylib')
21
+ end
22
+ end
23
+ end
24
+
25
+ describe "FFI::Platform::IS_WINDOWS" do
26
+ case OS
27
+ when "linux"
28
+ it "returns false" do
29
+ expect(FFI::Platform::IS_WINDOWS).to be false
30
+ end
31
+ when "windows"
32
+ it "returns true" do
33
+ expect(FFI::Platform::IS_WINDOWS).to be true
34
+ end
35
+ when "darwin"
36
+ it "returns false" do
37
+ expect(FFI::Platform::IS_WINDOWS).to be false
38
+ end
39
+ end
40
+ end
41
+
42
+ describe "FFI::Platform::ARCH" do
43
+ it "returns the architecture type" do
44
+ expect(FFI::Platform::ARCH).to eq(CPU)
45
+ end
46
+ end
47
+
48
+ describe "FFI::Platform::OS" do
49
+ case OS
50
+ when "linux"
51
+ it "returns 'linux' as a string" do
52
+ expect(FFI::Platform::OS).to eq('linux')
53
+ end
54
+ when "windows"
55
+ it "returns 'windows' as a string" do
56
+ expect(FFI::Platform::OS).to eq('windows')
57
+ end
58
+ when "darwin"
59
+ it "returns 'darwin' as a string" do
60
+ expect(FFI::Platform::OS).to eq('darwin')
61
+ end
62
+ end
63
+ end
64
+
65
+ describe "FFI::Platform.windows?" do
66
+ case OS
67
+ when "linux"
68
+ it "returns false" do
69
+ expect(FFI::Platform.windows?).to be false
70
+ end
71
+ when "windows"
72
+ it "returns true" do
73
+ expect(FFI::Platform.windows?).to be true
74
+ end
75
+ when "darwin"
76
+ it "returns false" do
77
+ expect(FFI::Platform.windows?).to be false
78
+ end
79
+ end
80
+ end
81
+
82
+ describe "FFI::Platform.mac?" do
83
+ case OS
84
+ when "linux"
85
+ it "returns false" do
86
+ expect(FFI::Platform.mac?).to be false
87
+ end
88
+ when "windows"
89
+ it "returns false" do
90
+ expect(FFI::Platform.mac?).to be false
91
+ end
92
+ when "darwin"
93
+ it "returns true" do
94
+ expect(FFI::Platform.mac?).to be true
95
+ end
96
+ end
97
+ end
98
+
99
+ describe "FFI::Platform.unix?" do
100
+ case OS
101
+ when "linux"
102
+ it "returns true" do
103
+ expect(FFI::Platform.unix?).to be true
104
+ end
105
+ when "windows"
106
+ it "returns false" do
107
+ expect(FFI::Platform.unix?).to be false
108
+ end
109
+ when "darwin"
110
+ it "returns true" do
111
+ expect(FFI::Platform.unix?).to be true
112
+ end
113
+ end
114
+ end
@@ -0,0 +1,285 @@
1
+ #
2
+ # This file is part of ruby-ffi.
3
+ # For licensing, see LICENSE.SPECS
4
+ #
5
+
6
+ require File.expand_path(File.join(File.dirname(__FILE__), "spec_helper"))
7
+ require 'delegate'
8
+
9
+ module PointerTestLib
10
+ extend FFI::Library
11
+ ffi_lib TestLibrary::PATH
12
+ begin
13
+ attach_function :ptr_ret_int32_t, [ :pointer, :int ], :int
14
+ rescue FFI::NotFoundError
15
+ # NetBSD uses #define instead of typedef for these
16
+ attach_function :ptr_ret_int32_t, :ptr_ret___int32_t, [ :pointer, :int ], :int
17
+ end
18
+ attach_function :ptr_from_address, [ FFI::Platform::ADDRESS_SIZE == 32 ? :uint : :ulong_long ], :pointer
19
+ attach_function :ptr_set_pointer, [ :pointer, :int, :pointer ], :void
20
+ attach_function :ptr_ret_pointer, [ :pointer, :int ], :pointer
21
+ end
22
+ describe "Pointer" do
23
+ include FFI
24
+ class ToPtrTest
25
+ def initialize(ptr)
26
+ @ptr = ptr
27
+ end
28
+ def to_ptr
29
+ @ptr
30
+ end
31
+ end
32
+
33
+ it "Any object implementing #to_ptr can be passed as a :pointer parameter" do
34
+ memory = FFI::MemoryPointer.new :long_long
35
+ magic = 0x12345678
36
+ memory.put_int32(0, magic)
37
+ tp = ToPtrTest.new(memory)
38
+ expect(PointerTestLib.ptr_ret_int32_t(tp, 0)).to eq(magic)
39
+ end
40
+ class PointerDelegate < DelegateClass(FFI::Pointer)
41
+ def initialize(ptr)
42
+ @ptr = ptr
43
+ end
44
+ def to_ptr
45
+ @ptr
46
+ end
47
+ end
48
+
49
+ it "A DelegateClass(Pointer) can be passed as a :pointer parameter" do
50
+ memory = FFI::MemoryPointer.new :long_long
51
+ magic = 0x12345678
52
+ memory.put_int32(0, magic)
53
+ ptr = PointerDelegate.new(memory)
54
+ expect(PointerTestLib.ptr_ret_int32_t(ptr, 0)).to eq(magic)
55
+ end
56
+
57
+ it "Fixnum cannot be used as a Pointer argument" do
58
+ expect { PointerTestLib.ptr_ret_int32(0, 0) }.to raise_error
59
+ end
60
+
61
+ it "Bignum cannot be used as a Pointer argument" do
62
+ expect { PointerTestLib.ptr_ret_int32(0xfee1deadbeefcafebabe, 0) }.to raise_error
63
+ end
64
+
65
+ it "#to_ptr" do
66
+ memory = FFI::MemoryPointer.new :pointer
67
+ expect(memory.to_ptr).to eq(memory)
68
+
69
+ expect(FFI::Pointer::NULL.to_ptr).to eq(FFI::Pointer::NULL)
70
+ end
71
+
72
+ describe "pointer type methods" do
73
+
74
+ it "#read_pointer" do
75
+ memory = FFI::MemoryPointer.new :pointer
76
+ PointerTestLib.ptr_set_pointer(memory, 0, PointerTestLib.ptr_from_address(0xdeadbeef))
77
+ expect(memory.read_pointer.address).to eq(0xdeadbeef)
78
+ end
79
+
80
+ it "#write_pointer" do
81
+ memory = FFI::MemoryPointer.new :pointer
82
+ memory.write_pointer(PointerTestLib.ptr_from_address(0xdeadbeef))
83
+ expect(PointerTestLib.ptr_ret_pointer(memory, 0).address).to eq(0xdeadbeef)
84
+ end
85
+
86
+ it "#read_array_of_pointer" do
87
+ values = [0x12345678, 0xfeedf00d, 0xdeadbeef]
88
+ memory = FFI::MemoryPointer.new :pointer, values.size
89
+ values.each_with_index do |address, j|
90
+ PointerTestLib.ptr_set_pointer(memory, j * FFI.type_size(:pointer), PointerTestLib.ptr_from_address(address))
91
+ end
92
+ array = memory.read_array_of_pointer(values.size)
93
+ values.each_with_index do |address, j|
94
+ expect(array[j].address).to eq(address)
95
+ end
96
+ end
97
+
98
+ end
99
+
100
+ describe 'NULL' do
101
+ it 'should be obtained using Pointer::NULL constant' do
102
+ null_ptr = FFI::Pointer::NULL
103
+ expect(null_ptr).to be_null
104
+ end
105
+ it 'should be obtained passing address 0 to constructor' do
106
+ expect(FFI::Pointer.new(0)).to be_null
107
+ end
108
+ it 'should raise an error when attempting read/write operations on it' do
109
+ null_ptr = FFI::Pointer::NULL
110
+ expect { null_ptr.read_int }.to raise_error(FFI::NullPointerError)
111
+ expect { null_ptr.write_int(0xff1) }.to raise_error(FFI::NullPointerError)
112
+ end
113
+ it 'returns true when compared with nil' do
114
+ expect((FFI::Pointer::NULL == nil)).to be true
115
+ end
116
+ end
117
+
118
+ it "Pointer.size returns sizeof pointer on platform" do
119
+ expect(FFI::Pointer.size).to eq((FFI::Platform::ADDRESS_SIZE / 8))
120
+ end
121
+
122
+ describe "#slice" do
123
+ before(:each) do
124
+ @mptr = FFI::MemoryPointer.new(:char, 12)
125
+ @mptr.put_uint(0, 0x12345678)
126
+ @mptr.put_uint(4, 0xdeadbeef)
127
+ end
128
+
129
+ it "contents of sliced pointer matches original pointer at offset" do
130
+ expect(@mptr.slice(4, 4).get_uint(0)).to eq(0xdeadbeef)
131
+ end
132
+
133
+ it "modifying sliced pointer is reflected in original pointer" do
134
+ @mptr.slice(4, 4).put_uint(0, 0xfee1dead)
135
+ expect(@mptr.get_uint(4)).to eq(0xfee1dead)
136
+ end
137
+
138
+ it "access beyond bounds should raise IndexError" do
139
+ expect { @mptr.slice(4, 4).get_int(4) }.to raise_error(IndexError)
140
+ end
141
+ end
142
+
143
+ describe "#type_size" do
144
+ it "should be same as FFI.type_size(type)" do
145
+ expect(FFI::MemoryPointer.new(:int, 1).type_size).to eq(FFI.type_size(:int))
146
+ end
147
+ end
148
+ end
149
+
150
+ describe "AutoPointer" do
151
+ loop_count = 30
152
+ wiggle_room = 5 # GC rarely cleans up all objects. we can get most of them, and that's enough to determine if the basic functionality is working.
153
+ magic = 0x12345678
154
+
155
+ class AutoPointerTestHelper
156
+ @@count = 0
157
+ def self.release
158
+ @@count += 1 if @@count > 0
159
+ end
160
+ def self.reset
161
+ @@count = 0
162
+ end
163
+ def self.gc_everything(count)
164
+ loop = 5
165
+ while @@count < count && loop > 0
166
+ loop -= 1
167
+ TestLibrary.force_gc
168
+ sleep 0.05 unless @@count == count
169
+ end
170
+ @@count = 0
171
+ end
172
+ def self.finalizer
173
+ self.method(:release).to_proc
174
+ end
175
+ end
176
+ class AutoPointerSubclass < FFI::AutoPointer
177
+ def self.release(ptr); end
178
+ end
179
+
180
+ # see #427
181
+ it "cleanup via default release method", :broken => true do
182
+ expect(AutoPointerSubclass).to receive(:release).at_least(loop_count-wiggle_room).times
183
+ AutoPointerTestHelper.reset
184
+ loop_count.times do
185
+ # note that if we called
186
+ # AutoPointerTestHelper.method(:release).to_proc inline, we'd
187
+ # have a reference to the pointer and it would never get GC'd.
188
+ AutoPointerSubclass.new(PointerTestLib.ptr_from_address(magic))
189
+ end
190
+ AutoPointerTestHelper.gc_everything loop_count
191
+ end
192
+
193
+ # see #427
194
+ it "cleanup when passed a proc", :broken => true do
195
+ # NOTE: passing a proc is touchy, because it's so easy to create a memory leak.
196
+ #
197
+ # specifically, if we made an inline call to
198
+ #
199
+ # AutoPointerTestHelper.method(:release).to_proc
200
+ #
201
+ # we'd have a reference to the pointer and it would
202
+ # never get GC'd.
203
+ expect(AutoPointerTestHelper).to receive(:release).at_least(loop_count-wiggle_room).times
204
+ AutoPointerTestHelper.reset
205
+ loop_count.times do
206
+ FFI::AutoPointer.new(PointerTestLib.ptr_from_address(magic),
207
+ AutoPointerTestHelper.finalizer)
208
+ end
209
+ AutoPointerTestHelper.gc_everything loop_count
210
+ end
211
+
212
+ # see #427
213
+ it "cleanup when passed a method", :broken => true do
214
+ expect(AutoPointerTestHelper).to receive(:release).at_least(loop_count-wiggle_room).times
215
+ AutoPointerTestHelper.reset
216
+ loop_count.times do
217
+ FFI::AutoPointer.new(PointerTestLib.ptr_from_address(magic),
218
+ AutoPointerTestHelper.method(:release))
219
+ end
220
+ AutoPointerTestHelper.gc_everything loop_count
221
+ end
222
+
223
+ it "can be used as the return type of a function" do
224
+ expect do
225
+ Module.new do
226
+ extend FFI::Library
227
+ ffi_lib TestLibrary::PATH
228
+ class CustomAutoPointer < FFI::AutoPointer
229
+ def self.release(ptr); end
230
+ end
231
+ attach_function :ptr_from_address, [ FFI::Platform::ADDRESS_SIZE == 32 ? :uint : :ulong_long ], CustomAutoPointer
232
+ end
233
+ end.not_to raise_error
234
+ end
235
+
236
+ describe "#new" do
237
+ it "MemoryPointer argument raises TypeError" do
238
+ expect { FFI::AutoPointer.new(FFI::MemoryPointer.new(:int))}.to raise_error(::TypeError)
239
+ end
240
+ it "AutoPointer argument raises TypeError" do
241
+ expect { AutoPointerSubclass.new(AutoPointerSubclass.new(PointerTestLib.ptr_from_address(0))) }.to raise_error(::TypeError)
242
+ end
243
+ it "Buffer argument raises TypeError" do
244
+ expect { FFI::AutoPointer.new(FFI::Buffer.new(:int))}.to raise_error(::TypeError)
245
+ end
246
+
247
+ end
248
+
249
+ describe "#autorelease?" do
250
+ ptr_class = Class.new(FFI::AutoPointer) do
251
+ def self.release(ptr); end
252
+ end
253
+
254
+ it "should be true by default" do
255
+ expect(ptr_class.new(FFI::Pointer.new(0xdeadbeef)).autorelease?).to be true
256
+ end
257
+
258
+ it "should return false when autorelease=(false)" do
259
+ ptr = ptr_class.new(FFI::Pointer.new(0xdeadbeef))
260
+ ptr.autorelease = false
261
+ expect(ptr.autorelease?).to be false
262
+ end
263
+ end
264
+
265
+ describe "#type_size" do
266
+ ptr_class = Class.new(FFI::AutoPointer) do
267
+ def self.release(ptr); end
268
+ end
269
+
270
+ it "type_size of AutoPointer should match wrapped Pointer" do
271
+ aptr = ptr_class.new(FFI::Pointer.new(:int, 0xdeadbeef))
272
+ expect(aptr.type_size).to eq(FFI.type_size(:int))
273
+ end
274
+
275
+ it "[] offset should match wrapped Pointer" do
276
+ mptr = FFI::MemoryPointer.new(:int, 1024)
277
+ aptr = ptr_class.new(FFI::Pointer.new(:int, mptr))
278
+ aptr[0].write_uint(0xfee1dead)
279
+ aptr[1].write_uint(0xcafebabe)
280
+ expect(mptr[0].read_uint).to eq(0xfee1dead)
281
+ expect(mptr[1].read_uint).to eq(0xcafebabe)
282
+ end
283
+ end
284
+ end
285
+