remogatto-ffi 0.5.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (324) hide show
  1. data/LICENSE +51 -0
  2. data/README.rdoc +69 -0
  3. data/Rakefile +144 -0
  4. data/ext/ffi_c/AbstractMemory.c +489 -0
  5. data/ext/ffi_c/AbstractMemory.h +160 -0
  6. data/ext/ffi_c/ArrayType.c +129 -0
  7. data/ext/ffi_c/ArrayType.h +58 -0
  8. data/ext/ffi_c/AutoPointer.c +61 -0
  9. data/ext/ffi_c/AutoPointer.h +18 -0
  10. data/ext/ffi_c/Buffer.c +187 -0
  11. data/ext/ffi_c/Call.c +431 -0
  12. data/ext/ffi_c/Call.h +77 -0
  13. data/ext/ffi_c/DynamicLibrary.c +216 -0
  14. data/ext/ffi_c/DynamicLibrary.h +22 -0
  15. data/ext/ffi_c/Function.c +498 -0
  16. data/ext/ffi_c/Function.h +73 -0
  17. data/ext/ffi_c/FunctionInfo.c +213 -0
  18. data/ext/ffi_c/LastError.c +159 -0
  19. data/ext/ffi_c/LastError.h +18 -0
  20. data/ext/ffi_c/MemoryPointer.c +178 -0
  21. data/ext/ffi_c/MemoryPointer.h +20 -0
  22. data/ext/ffi_c/MethodHandle.c +356 -0
  23. data/ext/ffi_c/MethodHandle.h +53 -0
  24. data/ext/ffi_c/Platform.c +59 -0
  25. data/ext/ffi_c/Platform.h +16 -0
  26. data/ext/ffi_c/Pointer.c +217 -0
  27. data/ext/ffi_c/Pointer.h +49 -0
  28. data/ext/ffi_c/Struct.c +770 -0
  29. data/ext/ffi_c/Struct.h +80 -0
  30. data/ext/ffi_c/StructByValue.c +140 -0
  31. data/ext/ffi_c/StructByValue.h +53 -0
  32. data/ext/ffi_c/StructLayout.c +450 -0
  33. data/ext/ffi_c/Type.c +329 -0
  34. data/ext/ffi_c/Type.h +57 -0
  35. data/ext/ffi_c/Types.c +103 -0
  36. data/ext/ffi_c/Types.h +85 -0
  37. data/ext/ffi_c/Variadic.c +260 -0
  38. data/ext/ffi_c/compat.h +72 -0
  39. data/ext/ffi_c/endian.h +40 -0
  40. data/ext/ffi_c/extconf.rb +34 -0
  41. data/ext/ffi_c/ffi.c +80 -0
  42. data/ext/ffi_c/libffi.bsd.mk +34 -0
  43. data/ext/ffi_c/libffi.darwin.mk +75 -0
  44. data/ext/ffi_c/libffi.gnu.mk +29 -0
  45. data/ext/ffi_c/libffi.mk +13 -0
  46. data/ext/ffi_c/libffi/ChangeLog +3243 -0
  47. data/ext/ffi_c/libffi/ChangeLog.libffi +347 -0
  48. data/ext/ffi_c/libffi/ChangeLog.libgcj +40 -0
  49. data/ext/ffi_c/libffi/ChangeLog.v1 +764 -0
  50. data/ext/ffi_c/libffi/LICENSE +21 -0
  51. data/ext/ffi_c/libffi/Makefile.am +177 -0
  52. data/ext/ffi_c/libffi/Makefile.in +1640 -0
  53. data/ext/ffi_c/libffi/README +328 -0
  54. data/ext/ffi_c/libffi/TODO +1 -0
  55. data/ext/ffi_c/libffi/acinclude.m4 +92 -0
  56. data/ext/ffi_c/libffi/aclocal.m4 +7516 -0
  57. data/ext/ffi_c/libffi/compile +142 -0
  58. data/ext/ffi_c/libffi/config.guess +1516 -0
  59. data/ext/ffi_c/libffi/config.sub +1626 -0
  60. data/ext/ffi_c/libffi/configure +24414 -0
  61. data/ext/ffi_c/libffi/configure.ac +365 -0
  62. data/ext/ffi_c/libffi/configure.host +11 -0
  63. data/ext/ffi_c/libffi/depcomp +584 -0
  64. data/ext/ffi_c/libffi/doc/libffi.info +533 -0
  65. data/ext/ffi_c/libffi/doc/libffi.texi +541 -0
  66. data/ext/ffi_c/libffi/doc/stamp-vti +4 -0
  67. data/ext/ffi_c/libffi/doc/version.texi +4 -0
  68. data/ext/ffi_c/libffi/fficonfig.h.in +160 -0
  69. data/ext/ffi_c/libffi/include/Makefile.am +9 -0
  70. data/ext/ffi_c/libffi/include/Makefile.in +422 -0
  71. data/ext/ffi_c/libffi/include/ffi.h.in +393 -0
  72. data/ext/ffi_c/libffi/include/ffi_common.h +98 -0
  73. data/ext/ffi_c/libffi/install-sh +323 -0
  74. data/ext/ffi_c/libffi/libffi.pc.in +10 -0
  75. data/ext/ffi_c/libffi/libtool-version +29 -0
  76. data/ext/ffi_c/libffi/ltcf-c.sh +861 -0
  77. data/ext/ffi_c/libffi/ltcf-cxx.sh +1069 -0
  78. data/ext/ffi_c/libffi/ltcf-gcj.sh +700 -0
  79. data/ext/ffi_c/libffi/ltconfig +2862 -0
  80. data/ext/ffi_c/libffi/ltmain.sh +6930 -0
  81. data/ext/ffi_c/libffi/man/Makefile.am +8 -0
  82. data/ext/ffi_c/libffi/man/Makefile.in +395 -0
  83. data/ext/ffi_c/libffi/man/ffi.3 +31 -0
  84. data/ext/ffi_c/libffi/man/ffi_call.3 +103 -0
  85. data/ext/ffi_c/libffi/man/ffi_prep_cif.3 +66 -0
  86. data/ext/ffi_c/libffi/mdate-sh +201 -0
  87. data/ext/ffi_c/libffi/missing +353 -0
  88. data/ext/ffi_c/libffi/mkinstalldirs +158 -0
  89. data/ext/ffi_c/libffi/src/alpha/ffi.c +284 -0
  90. data/ext/ffi_c/libffi/src/alpha/ffitarget.h +48 -0
  91. data/ext/ffi_c/libffi/src/alpha/osf.S +366 -0
  92. data/ext/ffi_c/libffi/src/arm/ffi.c +309 -0
  93. data/ext/ffi_c/libffi/src/arm/ffitarget.h +49 -0
  94. data/ext/ffi_c/libffi/src/arm/sysv.S +299 -0
  95. data/ext/ffi_c/libffi/src/closures.c +596 -0
  96. data/ext/ffi_c/libffi/src/cris/ffi.c +383 -0
  97. data/ext/ffi_c/libffi/src/cris/ffitarget.h +51 -0
  98. data/ext/ffi_c/libffi/src/cris/sysv.S +215 -0
  99. data/ext/ffi_c/libffi/src/debug.c +59 -0
  100. data/ext/ffi_c/libffi/src/dlmalloc.c +5099 -0
  101. data/ext/ffi_c/libffi/src/frv/eabi.S +128 -0
  102. data/ext/ffi_c/libffi/src/frv/ffi.c +292 -0
  103. data/ext/ffi_c/libffi/src/frv/ffitarget.h +61 -0
  104. data/ext/ffi_c/libffi/src/ia64/ffi.c +580 -0
  105. data/ext/ffi_c/libffi/src/ia64/ffitarget.h +50 -0
  106. data/ext/ffi_c/libffi/src/ia64/ia64_flags.h +40 -0
  107. data/ext/ffi_c/libffi/src/ia64/unix.S +560 -0
  108. data/ext/ffi_c/libffi/src/java_raw_api.c +359 -0
  109. data/ext/ffi_c/libffi/src/m32r/ffi.c +232 -0
  110. data/ext/ffi_c/libffi/src/m32r/ffitarget.h +48 -0
  111. data/ext/ffi_c/libffi/src/m32r/sysv.S +121 -0
  112. data/ext/ffi_c/libffi/src/m68k/ffi.c +278 -0
  113. data/ext/ffi_c/libffi/src/m68k/ffitarget.h +49 -0
  114. data/ext/ffi_c/libffi/src/m68k/sysv.S +234 -0
  115. data/ext/ffi_c/libffi/src/mips/ffi.c +926 -0
  116. data/ext/ffi_c/libffi/src/mips/ffitarget.h +202 -0
  117. data/ext/ffi_c/libffi/src/mips/n32.S +534 -0
  118. data/ext/ffi_c/libffi/src/mips/o32.S +381 -0
  119. data/ext/ffi_c/libffi/src/pa/ffi.c +709 -0
  120. data/ext/ffi_c/libffi/src/pa/ffitarget.h +77 -0
  121. data/ext/ffi_c/libffi/src/pa/hpux32.S +368 -0
  122. data/ext/ffi_c/libffi/src/pa/linux.S +357 -0
  123. data/ext/ffi_c/libffi/src/powerpc/aix.S +225 -0
  124. data/ext/ffi_c/libffi/src/powerpc/aix_closure.S +247 -0
  125. data/ext/ffi_c/libffi/src/powerpc/asm.h +125 -0
  126. data/ext/ffi_c/libffi/src/powerpc/darwin.S +245 -0
  127. data/ext/ffi_c/libffi/src/powerpc/darwin_closure.S +317 -0
  128. data/ext/ffi_c/libffi/src/powerpc/ffi.c +1429 -0
  129. data/ext/ffi_c/libffi/src/powerpc/ffi_darwin.c +800 -0
  130. data/ext/ffi_c/libffi/src/powerpc/ffitarget.h +118 -0
  131. data/ext/ffi_c/libffi/src/powerpc/linux64.S +187 -0
  132. data/ext/ffi_c/libffi/src/powerpc/linux64_closure.S +236 -0
  133. data/ext/ffi_c/libffi/src/powerpc/ppc_closure.S +327 -0
  134. data/ext/ffi_c/libffi/src/powerpc/sysv.S +230 -0
  135. data/ext/ffi_c/libffi/src/prep_cif.c +174 -0
  136. data/ext/ffi_c/libffi/src/raw_api.c +254 -0
  137. data/ext/ffi_c/libffi/src/s390/ffi.c +780 -0
  138. data/ext/ffi_c/libffi/src/s390/ffitarget.h +60 -0
  139. data/ext/ffi_c/libffi/src/s390/sysv.S +434 -0
  140. data/ext/ffi_c/libffi/src/sh/ffi.c +716 -0
  141. data/ext/ffi_c/libffi/src/sh/ffitarget.h +49 -0
  142. data/ext/ffi_c/libffi/src/sh/sysv.S +850 -0
  143. data/ext/ffi_c/libffi/src/sh64/ffi.c +453 -0
  144. data/ext/ffi_c/libffi/src/sh64/ffitarget.h +53 -0
  145. data/ext/ffi_c/libffi/src/sh64/sysv.S +530 -0
  146. data/ext/ffi_c/libffi/src/sparc/ffi.c +610 -0
  147. data/ext/ffi_c/libffi/src/sparc/ffitarget.h +66 -0
  148. data/ext/ffi_c/libffi/src/sparc/v8.S +272 -0
  149. data/ext/ffi_c/libffi/src/sparc/v9.S +307 -0
  150. data/ext/ffi_c/libffi/src/types.c +77 -0
  151. data/ext/ffi_c/libffi/src/x86/darwin.S +443 -0
  152. data/ext/ffi_c/libffi/src/x86/darwin64.S +416 -0
  153. data/ext/ffi_c/libffi/src/x86/ffi.c +475 -0
  154. data/ext/ffi_c/libffi/src/x86/ffi64.c +572 -0
  155. data/ext/ffi_c/libffi/src/x86/ffitarget.h +90 -0
  156. data/ext/ffi_c/libffi/src/x86/freebsd.S +458 -0
  157. data/ext/ffi_c/libffi/src/x86/sysv.S +437 -0
  158. data/ext/ffi_c/libffi/src/x86/unix64.S +418 -0
  159. data/ext/ffi_c/libffi/src/x86/win32.S +391 -0
  160. data/ext/ffi_c/libffi/testsuite/Makefile.am +71 -0
  161. data/ext/ffi_c/libffi/testsuite/Makefile.in +447 -0
  162. data/ext/ffi_c/libffi/testsuite/config/default.exp +1 -0
  163. data/ext/ffi_c/libffi/testsuite/lib/libffi-dg.exp +289 -0
  164. data/ext/ffi_c/libffi/testsuite/lib/target-libpath.exp +263 -0
  165. data/ext/ffi_c/libffi/testsuite/lib/wrapper.exp +45 -0
  166. data/ext/ffi_c/libffi/testsuite/libffi.call/call.exp +36 -0
  167. data/ext/ffi_c/libffi/testsuite/libffi.call/closure_fn0.c +97 -0
  168. data/ext/ffi_c/libffi/testsuite/libffi.call/closure_fn1.c +89 -0
  169. data/ext/ffi_c/libffi/testsuite/libffi.call/closure_fn2.c +89 -0
  170. data/ext/ffi_c/libffi/testsuite/libffi.call/closure_fn3.c +90 -0
  171. data/ext/ffi_c/libffi/testsuite/libffi.call/closure_fn4.c +97 -0
  172. data/ext/ffi_c/libffi/testsuite/libffi.call/closure_fn5.c +99 -0
  173. data/ext/ffi_c/libffi/testsuite/libffi.call/closure_fn6.c +98 -0
  174. data/ext/ffi_c/libffi/testsuite/libffi.call/closure_stdcall.c +72 -0
  175. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_12byte.c +102 -0
  176. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_16byte.c +103 -0
  177. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_18byte.c +104 -0
  178. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_19byte.c +110 -0
  179. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_1_1byte.c +97 -0
  180. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_20byte.c +99 -0
  181. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_20byte1.c +101 -0
  182. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_24byte.c +121 -0
  183. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_2byte.c +98 -0
  184. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_3_1byte.c +103 -0
  185. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_3byte1.c +98 -0
  186. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_3byte2.c +98 -0
  187. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_4_1byte.c +106 -0
  188. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_4byte.c +98 -0
  189. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_5_1_byte.c +117 -0
  190. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_5byte.c +106 -0
  191. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_64byte.c +132 -0
  192. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_6_1_byte.c +121 -0
  193. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_6byte.c +107 -0
  194. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_7_1_byte.c +125 -0
  195. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_7byte.c +105 -0
  196. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_8byte.c +96 -0
  197. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_9byte1.c +98 -0
  198. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_9byte2.c +99 -0
  199. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_double.c +101 -0
  200. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_float.c +99 -0
  201. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_longdouble.c +100 -0
  202. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_pointer.c +101 -0
  203. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_sint16.c +99 -0
  204. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_sint32.c +99 -0
  205. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_sint64.c +99 -0
  206. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_uint16.c +99 -0
  207. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_uint32.c +99 -0
  208. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_uint64.c +100 -0
  209. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_double.c +51 -0
  210. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_float.c +51 -0
  211. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_multi_schar.c +82 -0
  212. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_multi_sshort.c +82 -0
  213. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_multi_sshortchar.c +94 -0
  214. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_multi_uchar.c +99 -0
  215. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_multi_ushort.c +82 -0
  216. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_multi_ushortchar.c +94 -0
  217. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_schar.c +52 -0
  218. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_sint.c +50 -0
  219. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_sshort.c +50 -0
  220. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_uchar.c +50 -0
  221. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_uint.c +51 -0
  222. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_ulonglong.c +54 -0
  223. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_ushort.c +51 -0
  224. data/ext/ffi_c/libffi/testsuite/libffi.call/ffitest.h +86 -0
  225. data/ext/ffi_c/libffi/testsuite/libffi.call/float.c +59 -0
  226. data/ext/ffi_c/libffi/testsuite/libffi.call/float1.c +58 -0
  227. data/ext/ffi_c/libffi/testsuite/libffi.call/float2.c +57 -0
  228. data/ext/ffi_c/libffi/testsuite/libffi.call/float3.c +72 -0
  229. data/ext/ffi_c/libffi/testsuite/libffi.call/float4.c +62 -0
  230. data/ext/ffi_c/libffi/testsuite/libffi.call/many.c +69 -0
  231. data/ext/ffi_c/libffi/testsuite/libffi.call/many_win32.c +63 -0
  232. data/ext/ffi_c/libffi/testsuite/libffi.call/negint.c +53 -0
  233. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct.c +160 -0
  234. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct1.c +169 -0
  235. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct10.c +141 -0
  236. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct2.c +118 -0
  237. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct3.c +119 -0
  238. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct4.c +119 -0
  239. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct5.c +120 -0
  240. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct6.c +139 -0
  241. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct7.c +119 -0
  242. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct8.c +139 -0
  243. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct9.c +139 -0
  244. data/ext/ffi_c/libffi/testsuite/libffi.call/problem1.c +98 -0
  245. data/ext/ffi_c/libffi/testsuite/libffi.call/promotion.c +59 -0
  246. data/ext/ffi_c/libffi/testsuite/libffi.call/pyobjc-tc.c +114 -0
  247. data/ext/ffi_c/libffi/testsuite/libffi.call/return_dbl.c +35 -0
  248. data/ext/ffi_c/libffi/testsuite/libffi.call/return_dbl1.c +43 -0
  249. data/ext/ffi_c/libffi/testsuite/libffi.call/return_dbl2.c +42 -0
  250. data/ext/ffi_c/libffi/testsuite/libffi.call/return_fl.c +35 -0
  251. data/ext/ffi_c/libffi/testsuite/libffi.call/return_fl1.c +36 -0
  252. data/ext/ffi_c/libffi/testsuite/libffi.call/return_fl2.c +49 -0
  253. data/ext/ffi_c/libffi/testsuite/libffi.call/return_fl3.c +42 -0
  254. data/ext/ffi_c/libffi/testsuite/libffi.call/return_ldl.c +34 -0
  255. data/ext/ffi_c/libffi/testsuite/libffi.call/return_ll.c +41 -0
  256. data/ext/ffi_c/libffi/testsuite/libffi.call/return_ll1.c +42 -0
  257. data/ext/ffi_c/libffi/testsuite/libffi.call/return_sc.c +36 -0
  258. data/ext/ffi_c/libffi/testsuite/libffi.call/return_sl.c +38 -0
  259. data/ext/ffi_c/libffi/testsuite/libffi.call/return_uc.c +38 -0
  260. data/ext/ffi_c/libffi/testsuite/libffi.call/return_ul.c +38 -0
  261. data/ext/ffi_c/libffi/testsuite/libffi.call/strlen.c +44 -0
  262. data/ext/ffi_c/libffi/testsuite/libffi.call/strlen_win32.c +44 -0
  263. data/ext/ffi_c/libffi/testsuite/libffi.call/struct1.c +65 -0
  264. data/ext/ffi_c/libffi/testsuite/libffi.call/struct2.c +67 -0
  265. data/ext/ffi_c/libffi/testsuite/libffi.call/struct3.c +59 -0
  266. data/ext/ffi_c/libffi/testsuite/libffi.call/struct4.c +63 -0
  267. data/ext/ffi_c/libffi/testsuite/libffi.call/struct5.c +65 -0
  268. data/ext/ffi_c/libffi/testsuite/libffi.call/struct6.c +64 -0
  269. data/ext/ffi_c/libffi/testsuite/libffi.call/struct7.c +74 -0
  270. data/ext/ffi_c/libffi/testsuite/libffi.call/struct8.c +80 -0
  271. data/ext/ffi_c/libffi/testsuite/libffi.call/struct9.c +67 -0
  272. data/ext/ffi_c/libffi/testsuite/libffi.special/ffitestcxx.h +86 -0
  273. data/ext/ffi_c/libffi/testsuite/libffi.special/special.exp +38 -0
  274. data/ext/ffi_c/libffi/testsuite/libffi.special/unwindtest.cc +123 -0
  275. data/ext/ffi_c/libffi/testsuite/libffi.special/unwindtest_ffi_call.cc +53 -0
  276. data/ext/ffi_c/libffi/texinfo.tex +7482 -0
  277. data/ext/ffi_c/rbffi.h +26 -0
  278. data/gen/Rakefile +12 -0
  279. data/lib/ffi.rb +11 -0
  280. data/lib/ffi/autopointer.rb +61 -0
  281. data/lib/ffi/buffer.rb +0 -0
  282. data/lib/ffi/callback.rb +10 -0
  283. data/lib/ffi/enum.rb +78 -0
  284. data/lib/ffi/errno.rb +8 -0
  285. data/lib/ffi/ffi.rb +99 -0
  286. data/lib/ffi/io.rb +21 -0
  287. data/lib/ffi/library.rb +218 -0
  288. data/lib/ffi/managedstruct.rb +55 -0
  289. data/lib/ffi/memorypointer.rb +73 -0
  290. data/lib/ffi/platform.rb +78 -0
  291. data/lib/ffi/pointer.rb +119 -0
  292. data/lib/ffi/struct.rb +164 -0
  293. data/lib/ffi/tools/const_generator.rb +177 -0
  294. data/lib/ffi/tools/generator.rb +58 -0
  295. data/lib/ffi/tools/generator_task.rb +35 -0
  296. data/lib/ffi/tools/struct_generator.rb +194 -0
  297. data/lib/ffi/tools/types_generator.rb +123 -0
  298. data/lib/ffi/types.rb +153 -0
  299. data/lib/ffi/union.rb +12 -0
  300. data/lib/ffi/variadic.rb +25 -0
  301. data/spec/ffi/bool_spec.rb +24 -0
  302. data/spec/ffi/buffer_spec.rb +202 -0
  303. data/spec/ffi/callback_spec.rb +591 -0
  304. data/spec/ffi/enum_spec.rb +164 -0
  305. data/spec/ffi/errno_spec.rb +13 -0
  306. data/spec/ffi/function_spec.rb +47 -0
  307. data/spec/ffi/library_spec.rb +144 -0
  308. data/spec/ffi/managed_struct_spec.rb +56 -0
  309. data/spec/ffi/number_spec.rb +231 -0
  310. data/spec/ffi/pointer_spec.rb +195 -0
  311. data/spec/ffi/rbx/attach_function_spec.rb +27 -0
  312. data/spec/ffi/rbx/memory_pointer_spec.rb +102 -0
  313. data/spec/ffi/rbx/spec_helper.rb +1 -0
  314. data/spec/ffi/rbx/struct_spec.rb +13 -0
  315. data/spec/ffi/spec_helper.rb +17 -0
  316. data/spec/ffi/string_spec.rb +103 -0
  317. data/spec/ffi/struct_callback_spec.rb +62 -0
  318. data/spec/ffi/struct_initialize_spec.rb +30 -0
  319. data/spec/ffi/struct_spec.rb +529 -0
  320. data/spec/ffi/typedef_spec.rb +48 -0
  321. data/spec/ffi/union_spec.rb +60 -0
  322. data/spec/ffi/variadic_spec.rb +84 -0
  323. data/spec/spec.opts +4 -0
  324. metadata +432 -0
@@ -0,0 +1,216 @@
1
+ #include <sys/types.h>
2
+ #include <stdio.h>
3
+ #include <stdint.h>
4
+ #if defined(_WIN32) || defined(__WIN32__)
5
+ # include <windows.h>
6
+ #else
7
+ # include <dlfcn.h>
8
+ #endif
9
+ #include <ruby.h>
10
+
11
+ #include <ffi.h>
12
+
13
+ #include "rbffi.h"
14
+ #include "compat.h"
15
+ #include "AbstractMemory.h"
16
+ #include "Pointer.h"
17
+ #include "DynamicLibrary.h"
18
+
19
+ typedef struct LibrarySymbol_ {
20
+ AbstractMemory memory;
21
+ VALUE library;
22
+ VALUE name;
23
+ } LibrarySymbol;
24
+
25
+ static VALUE library_initialize(VALUE self, VALUE libname, VALUE libflags);
26
+ static void library_free(Library* lib);
27
+
28
+
29
+ static VALUE symbol_allocate(VALUE klass);
30
+ static VALUE symbol_new(VALUE library, void* address, VALUE name);
31
+ static void symbol_mark(LibrarySymbol* sym);
32
+
33
+ static VALUE LibraryClass = Qnil, SymbolClass = Qnil;
34
+
35
+ #if defined(_WIN32) || defined(__WIN32__)
36
+ static void* dl_open(const char* name, int flags);
37
+ static void dl_error(char* buf, int size);
38
+ #define dl_sym(handle, name) GetProcAddress(handle, name)
39
+ #define dl_close(handle) FreeLibrary(handle)
40
+ enum { RTLD_LAZY=1, RTLD_NOW, RTLD_GLOBAL, RTLD_LOCAL };
41
+ #else
42
+ # define dl_open(name, flags) dlopen(name, flags != 0 ? flags : RTLD_LAZY)
43
+ # define dl_error(buf, size) do { snprintf(buf, size, "%s", dlerror()); } while(0)
44
+ # define dl_sym(handle, name) dlsym(handle, name)
45
+ # define dl_close(handle) dlclose(handle)
46
+ #ifndef RTLD_LOCAL
47
+ # define RTLD_LOCAL 8
48
+ #endif
49
+ #endif
50
+
51
+ static VALUE
52
+ library_allocate(VALUE klass)
53
+ {
54
+ Library* library;
55
+ return Data_Make_Struct(klass, Library, NULL, library_free, library);
56
+ }
57
+
58
+ static VALUE
59
+ library_open(VALUE klass, VALUE libname, VALUE libflags)
60
+ {
61
+ return library_initialize(library_allocate(klass), libname, libflags);
62
+ }
63
+
64
+ static VALUE
65
+ library_initialize(VALUE self, VALUE libname, VALUE libflags)
66
+ {
67
+ Library* library;
68
+ int flags;
69
+
70
+ Check_Type(libflags, T_FIXNUM);
71
+
72
+ Data_Get_Struct(self, Library, library);
73
+ flags = libflags != Qnil ? NUM2UINT(libflags) : 0;
74
+
75
+ library->handle = dl_open(libname != Qnil ? StringValueCStr(libname) : NULL, flags);
76
+ if (library->handle == NULL) {
77
+ char errmsg[1024];
78
+ dl_error(errmsg, sizeof(errmsg));
79
+ rb_raise(rb_eLoadError, "Could not open library '%s': %s",
80
+ libname != Qnil ? StringValueCStr(libname) : "[current process]",
81
+ errmsg);
82
+ }
83
+ rb_iv_set(self, "@name", libname != Qnil ? libname : rb_str_new2("[current process]"));
84
+ return self;
85
+ }
86
+
87
+ static VALUE
88
+ library_dlsym(VALUE self, VALUE name)
89
+ {
90
+ Library* library;
91
+ void* address = NULL;
92
+ Check_Type(name, T_STRING);
93
+
94
+ Data_Get_Struct(self, Library, library);
95
+ address = dl_sym(library->handle, StringValueCStr(name));
96
+
97
+ return address != NULL ? symbol_new(self, address, name) : Qnil;
98
+ }
99
+
100
+ static VALUE
101
+ library_dlerror(VALUE self)
102
+ {
103
+ char errmsg[1024];
104
+ dl_error(errmsg, sizeof(errmsg));
105
+ return rb_tainted_str_new2(errmsg);
106
+ }
107
+
108
+ static void
109
+ library_free(Library* library)
110
+ {
111
+ // dlclose() on MacOS tends to segfault - avoid it
112
+ #ifndef __APPLE__
113
+ if (library->handle != NULL) {
114
+ dl_close(library->handle);
115
+ }
116
+ #endif
117
+ xfree(library);
118
+ }
119
+
120
+ #if defined(_WIN32) || defined(__WIN32__)
121
+ static void*
122
+ dl_open(const char* name, int flags)
123
+ {
124
+ if (name == NULL) {
125
+ return GetModuleHandle(NULL);
126
+ } else {
127
+ return LoadLibraryEx(name, NULL, LOAD_WITH_ALTERED_SEARCH_PATH);
128
+ }
129
+ }
130
+
131
+ static void
132
+ dl_error(char* buf, int size)
133
+ {
134
+ FormatMessageA(FORMAT_MESSAGE_FROM_SYSTEM, NULL, GetLastError(),
135
+ 0, buf, size, NULL);
136
+ }
137
+ #endif
138
+
139
+ static VALUE
140
+ symbol_allocate(VALUE klass)
141
+ {
142
+ LibrarySymbol* sym;
143
+ VALUE obj = Data_Make_Struct(klass, LibrarySymbol, NULL, -1, sym);
144
+ sym->name = Qnil;
145
+ sym->library = Qnil;
146
+
147
+ return obj;
148
+ }
149
+
150
+ static VALUE
151
+ symbol_new(VALUE library, void* address, VALUE name)
152
+ {
153
+ LibrarySymbol* sym;
154
+ VALUE obj = Data_Make_Struct(SymbolClass, LibrarySymbol, symbol_mark, -1, sym);
155
+
156
+ sym->memory.address = address;
157
+ sym->memory.size = LONG_MAX;
158
+ sym->memory.typeSize = 1;
159
+ sym->memory.access = MEM_RD | MEM_WR;
160
+ sym->memory.ops = &rbffi_AbstractMemoryOps;
161
+ sym->library = library;
162
+ sym->name = name;
163
+
164
+ return obj;
165
+ }
166
+
167
+ static void
168
+ symbol_mark(LibrarySymbol* sym)
169
+ {
170
+ rb_gc_mark(sym->library);
171
+ rb_gc_mark(sym->name);
172
+ }
173
+
174
+ static VALUE
175
+ symbol_inspect(VALUE self)
176
+ {
177
+ LibrarySymbol* sym;
178
+ char buf[256];
179
+
180
+ Data_Get_Struct(self, LibrarySymbol, sym);
181
+ snprintf(buf, sizeof(buf), "#<FFI::Library::Symbol name=%s address=%p>",
182
+ StringValueCStr(sym->name), sym->memory.address);
183
+ return rb_str_new2(buf);
184
+ }
185
+
186
+ void
187
+ rbffi_DynamicLibrary_Init(VALUE moduleFFI)
188
+ {
189
+ LibraryClass = rb_define_class_under(moduleFFI, "DynamicLibrary", rb_cObject);
190
+ rb_global_variable(&LibraryClass);
191
+ SymbolClass = rb_define_class_under(LibraryClass, "Symbol", rbffi_PointerClass);
192
+ rb_global_variable(&SymbolClass);
193
+
194
+ rb_define_const(moduleFFI, "NativeLibrary", LibraryClass); // backwards compat library
195
+ rb_define_alloc_func(LibraryClass, library_allocate);
196
+ rb_define_singleton_method(LibraryClass, "open", library_open, 2);
197
+ rb_define_singleton_method(LibraryClass, "last_error", library_dlerror, 0);
198
+ rb_define_method(LibraryClass, "initialize", library_initialize, 2);
199
+ rb_define_method(LibraryClass, "find_symbol", library_dlsym, 1);
200
+ rb_define_method(LibraryClass, "find_function", library_dlsym, 1);
201
+ rb_define_method(LibraryClass, "find_variable", library_dlsym, 1);
202
+ rb_define_method(LibraryClass, "last_error", library_dlerror, 0);
203
+ rb_define_attr(LibraryClass, "name", 1, 0);
204
+
205
+ rb_define_alloc_func(SymbolClass, symbol_allocate);
206
+ rb_undef_method(SymbolClass, "new");
207
+ rb_define_method(SymbolClass, "inspect", symbol_inspect, 0);
208
+
209
+ #define DEF(x) rb_define_const(LibraryClass, "RTLD_" #x, UINT2NUM(RTLD_##x))
210
+ DEF(LAZY);
211
+ DEF(NOW);
212
+ DEF(GLOBAL);
213
+ DEF(LOCAL);
214
+
215
+ }
216
+
@@ -0,0 +1,22 @@
1
+
2
+ #ifndef _LIBRARY_H
3
+ #define _LIBRARY_H
4
+
5
+ #ifdef __cplusplus
6
+ extern "C" {
7
+ #endif
8
+
9
+
10
+
11
+ typedef struct Library {
12
+ void* handle;
13
+ } Library;
14
+
15
+ extern void rbffi_DynamicLibrary_Init(VALUE ffiModule);
16
+
17
+ #ifdef __cplusplus
18
+ }
19
+ #endif
20
+
21
+ #endif /* _LIBRARY_H */
22
+
@@ -0,0 +1,498 @@
1
+ /*
2
+ * Copyright (c) 2009, Wayne Meissner
3
+ * All rights reserved.
4
+ *
5
+ * Redistribution and use in source and binary forms, with or without
6
+ * modification, are permitted provided that the following conditions are met:
7
+ *
8
+ * * Redistributions of source code must retain the above copyright notice, this
9
+ * list of conditions and the following disclaimer.
10
+ * * Redistributions in binary form must reproduce the above copyright notice
11
+ * this list of conditions and the following disclaimer in the documentation
12
+ * and/or other materials provided with the distribution.
13
+ * * The name of the author or authors may not be used to endorse or promote
14
+ * products derived from this software without specific prior written permission.
15
+ *
16
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
17
+ * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
19
+ * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE
20
+ * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
21
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
22
+ * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
23
+ * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
24
+ * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
25
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26
+ */
27
+
28
+ #include "MethodHandle.h"
29
+
30
+
31
+ #include <sys/param.h>
32
+ #include <sys/types.h>
33
+ #ifndef _WIN32
34
+ # include <sys/mman.h>
35
+ #endif
36
+ #include <stdio.h>
37
+ #include <stdint.h>
38
+ #include <stdbool.h>
39
+ #include <ruby.h>
40
+
41
+ #include <ffi.h>
42
+ #include "rbffi.h"
43
+ #include "compat.h"
44
+
45
+ #include "AbstractMemory.h"
46
+ #include "Pointer.h"
47
+ #include "Struct.h"
48
+ #include "Platform.h"
49
+ #include "Type.h"
50
+ #include "LastError.h"
51
+ #include "Call.h"
52
+ #include "Function.h"
53
+
54
+ #if defined(HAVE_LIBFFI) && !defined(HAVE_FFI_CLOSURE_ALLOC)
55
+ static void* ffi_closure_alloc(size_t size, void** code);
56
+ static void ffi_closure_free(void* ptr);
57
+ ffi_status ffi_prep_closure_loc(ffi_closure* closure, ffi_cif* cif,
58
+ void (*fun)(ffi_cif*, void*, void**, void*),
59
+ void* user_data, void* code);
60
+ #endif /* HAVE_FFI_CLOSURE_ALLOC */
61
+
62
+ typedef struct Function_ {
63
+ AbstractMemory memory;
64
+ FunctionType* info;
65
+ MethodHandle* methodHandle;
66
+ bool autorelease;
67
+ ffi_closure* ffiClosure;
68
+ VALUE rbProc;
69
+ VALUE rbFunctionInfo;
70
+ } Function;
71
+
72
+ static void function_mark(Function *);
73
+ static void function_free(Function *);
74
+ static VALUE function_init(VALUE self, VALUE rbFunctionInfo, VALUE rbProc);
75
+ static void callback_invoke(ffi_cif* cif, void* retval, void** parameters, void* user_data);
76
+
77
+ VALUE rbffi_FunctionClass = Qnil;
78
+
79
+ static ID id_call = 0, id_cbtable = 0;
80
+
81
+ static VALUE
82
+ function_allocate(VALUE klass)
83
+ {
84
+ Function *fn;
85
+ VALUE obj;
86
+
87
+ obj = Data_Make_Struct(klass, Function, function_mark, function_free, fn);
88
+
89
+ fn->memory.access = MEM_RD;
90
+ fn->memory.ops = &rbffi_AbstractMemoryOps;
91
+
92
+ fn->rbProc = Qnil;
93
+ fn->rbFunctionInfo = Qnil;
94
+ fn->autorelease = true;
95
+
96
+ return obj;
97
+ }
98
+
99
+ static void
100
+ function_mark(Function *fn)
101
+ {
102
+ rb_gc_mark(fn->rbProc);
103
+ rb_gc_mark(fn->rbFunctionInfo);
104
+ }
105
+
106
+ static void
107
+ function_free(Function *fn)
108
+ {
109
+ if (fn->methodHandle != NULL) {
110
+ rbffi_MethodHandle_Free(fn->methodHandle);
111
+ }
112
+
113
+ if (fn->ffiClosure != NULL && fn->autorelease) {
114
+ ffi_closure_free(fn->ffiClosure);
115
+ }
116
+
117
+ xfree(fn);
118
+ }
119
+
120
+ static VALUE
121
+ function_initialize(int argc, VALUE* argv, VALUE self)
122
+ {
123
+
124
+ VALUE rbReturnType = Qnil, rbParamTypes = Qnil, rbProc = Qnil, rbOptions = Qnil;
125
+ VALUE rbFunctionInfo = Qnil;
126
+ VALUE infoArgv[3];
127
+ int nargs;
128
+
129
+ nargs = rb_scan_args(argc, argv, "22", &rbReturnType, &rbParamTypes, &rbProc, &rbOptions);
130
+
131
+ //
132
+ // Callback with block,
133
+ // e.g. Function.new(:int, [ :int ]) { |i| blah }
134
+ // or Function.new(:int, [ :int ], { :convention => :stdcall }) { |i| blah }
135
+ //
136
+ if (rb_block_given_p()) {
137
+ if (nargs > 3) {
138
+ rb_raise(rb_eArgError, "cannot create function with both proc/address and block");
139
+ }
140
+ rbOptions = rbProc;
141
+ rbProc = rb_block_proc();
142
+ } else {
143
+ // Callback with proc, or Function with address
144
+ // e.g. Function.new(:int, [ :int ], Proc.new { |i| })
145
+ // Function.new(:int, [ :int ], Proc.new { |i| }, { :convention => :stdcall })
146
+ // Function.new(:int, [ :int ], addr)
147
+ // Function.new(:int, [ :int ], addr, { :convention => :stdcall })
148
+ }
149
+
150
+ infoArgv[0] = rbReturnType;
151
+ infoArgv[1] = rbParamTypes;
152
+ infoArgv[2] = rbOptions;
153
+ rbFunctionInfo = rb_class_new_instance(rbOptions != Qnil ? 3 : 2, infoArgv, rbffi_FunctionTypeClass);
154
+
155
+ function_init(self, rbFunctionInfo, rbProc);
156
+
157
+ return self;
158
+ }
159
+
160
+ VALUE
161
+ rbffi_Function_NewInstance(VALUE rbFunctionInfo, VALUE rbProc)
162
+ {
163
+ return function_init(function_allocate(rbffi_FunctionClass), rbFunctionInfo, rbProc);
164
+ }
165
+
166
+ VALUE
167
+ rbffi_Function_ForProc(VALUE rbFunctionInfo, VALUE proc)
168
+ {
169
+ VALUE callback;
170
+ VALUE cbTable = RTEST(rb_ivar_defined(proc, id_cbtable)) ? rb_ivar_get(proc, id_cbtable) : Qnil;
171
+
172
+ if (cbTable == Qnil) {
173
+ cbTable = rb_hash_new();
174
+ rb_ivar_set(proc, id_cbtable, cbTable);
175
+ }
176
+
177
+ callback = rb_hash_aref(cbTable, rbFunctionInfo);
178
+ if (callback != Qnil) {
179
+ return callback;
180
+ }
181
+
182
+ callback = rbffi_Function_NewInstance(rbFunctionInfo, proc);
183
+ rb_hash_aset(cbTable, rbFunctionInfo, callback);
184
+ return callback;
185
+ }
186
+
187
+ static VALUE
188
+ function_init(VALUE self, VALUE rbFunctionInfo, VALUE rbProc)
189
+ {
190
+ Function* fn = NULL;
191
+
192
+ Data_Get_Struct(self, Function, fn);
193
+
194
+ fn->rbFunctionInfo = rbFunctionInfo;
195
+
196
+ Data_Get_Struct(fn->rbFunctionInfo, FunctionType, fn->info);
197
+
198
+ if (rb_obj_is_kind_of(rbProc, rbffi_PointerClass)) {
199
+ AbstractMemory* memory;
200
+ Data_Get_Struct(rbProc, AbstractMemory, memory);
201
+ fn->memory = *memory;
202
+
203
+ } else if (rb_obj_is_kind_of(rbProc, rb_cProc) || rb_respond_to(rbProc, id_call)) {
204
+ void* code;
205
+ ffi_status status;
206
+ fn->ffiClosure = ffi_closure_alloc(sizeof(*fn->ffiClosure), &code);
207
+ if (fn->ffiClosure == NULL) {
208
+ rb_raise(rb_eNoMemError, "Failed to allocate libffi closure");
209
+ }
210
+
211
+ status = ffi_prep_closure_loc(fn->ffiClosure, &fn->info->ffi_cif,
212
+ callback_invoke, fn, code);
213
+ if (status != FFI_OK) {
214
+ rb_raise(rb_eArgError, "ffi_prep_closure_loc failed");
215
+ }
216
+
217
+ fn->memory.address = code;
218
+ fn->memory.size = sizeof(*fn->ffiClosure);
219
+ fn->autorelease = true;
220
+
221
+ } else {
222
+ rb_raise(rb_eTypeError, "wrong argument type. Expected pointer or proc");
223
+ }
224
+
225
+ fn->rbProc = rbProc;
226
+
227
+ return self;
228
+ }
229
+
230
+ static VALUE
231
+ function_call(int argc, VALUE* argv, VALUE self)
232
+ {
233
+ Function* fn;
234
+
235
+ Data_Get_Struct(self, Function, fn);
236
+
237
+ return rbffi_CallFunction(argc, argv, fn->memory.address, fn->info);
238
+ }
239
+
240
+ static VALUE
241
+ function_attach(VALUE self, VALUE module, VALUE name)
242
+ {
243
+ Function* fn;
244
+ char var[1024];
245
+
246
+ Data_Get_Struct(self, Function, fn);
247
+
248
+ if (fn->info->parameterCount == -1) {
249
+ rb_raise(rb_eRuntimeError, "Cannot attach variadic functions");
250
+ }
251
+
252
+ if (fn->methodHandle == NULL) {
253
+ fn->methodHandle = rbffi_MethodHandle_Alloc(fn->info, fn->memory.address);
254
+ }
255
+
256
+ //
257
+ // Stash the Function in a module variable so it does not get garbage collected
258
+ //
259
+ snprintf(var, sizeof(var), "@@%s", StringValueCStr(name));
260
+ rb_cv_set(module, var, self);
261
+
262
+ rb_define_module_function(module, StringValueCStr(name),
263
+ rbffi_MethodHandle_CodeAddress(fn->methodHandle), -1);
264
+
265
+
266
+ rb_define_method(module, StringValueCStr(name),
267
+ rbffi_MethodHandle_CodeAddress(fn->methodHandle), -1);
268
+
269
+ return self;
270
+ }
271
+
272
+ static VALUE
273
+ function_set_autorelease(VALUE self, VALUE autorelease)
274
+ {
275
+ Function* fn;
276
+
277
+ Data_Get_Struct(self, Function, fn);
278
+
279
+ fn->autorelease = RTEST(autorelease);
280
+
281
+ return self;
282
+ }
283
+
284
+ static VALUE
285
+ function_autorelease_p(VALUE self)
286
+ {
287
+ Function* fn;
288
+
289
+ Data_Get_Struct(self, Function, fn);
290
+
291
+ return fn->autorelease ? Qtrue : Qfalse;
292
+ }
293
+
294
+ static VALUE
295
+ function_release(VALUE self)
296
+ {
297
+ Function* fn;
298
+
299
+ Data_Get_Struct(self, Function, fn);
300
+
301
+ if (fn->ffiClosure == NULL) {
302
+ rb_raise(rb_eRuntimeError, "cannot free function which was not allocated");
303
+ }
304
+
305
+ ffi_closure_free(fn->ffiClosure);
306
+ fn->ffiClosure = NULL;
307
+
308
+ return self;
309
+ }
310
+
311
+ static void
312
+ callback_invoke(ffi_cif* cif, void* retval, void** parameters, void* user_data)
313
+ {
314
+ Function* fn = (Function *) user_data;
315
+ FunctionType *cbInfo = fn->info;
316
+ VALUE* rbParams;
317
+ VALUE rbReturnValue;
318
+ int i;
319
+
320
+ rbParams = ALLOCA_N(VALUE, cbInfo->parameterCount);
321
+ for (i = 0; i < cbInfo->parameterCount; ++i) {
322
+ VALUE param;
323
+ switch (cbInfo->parameterTypes[i]->nativeType) {
324
+ case NATIVE_INT8:
325
+ param = INT2NUM(*(int8_t *) parameters[i]);
326
+ break;
327
+ case NATIVE_UINT8:
328
+ param = UINT2NUM(*(uint8_t *) parameters[i]);
329
+ break;
330
+ case NATIVE_INT16:
331
+ param = INT2NUM(*(int16_t *) parameters[i]);
332
+ break;
333
+ case NATIVE_UINT16:
334
+ param = UINT2NUM(*(uint16_t *) parameters[i]);
335
+ break;
336
+ case NATIVE_INT32:
337
+ param = INT2NUM(*(int32_t *) parameters[i]);
338
+ break;
339
+ case NATIVE_UINT32:
340
+ param = UINT2NUM(*(uint32_t *) parameters[i]);
341
+ break;
342
+ case NATIVE_INT64:
343
+ param = LL2NUM(*(int64_t *) parameters[i]);
344
+ break;
345
+ case NATIVE_UINT64:
346
+ param = ULL2NUM(*(uint64_t *) parameters[i]);
347
+ break;
348
+ case NATIVE_FLOAT32:
349
+ param = rb_float_new(*(float *) parameters[i]);
350
+ break;
351
+ case NATIVE_FLOAT64:
352
+ param = rb_float_new(*(double *) parameters[i]);
353
+ break;
354
+ case NATIVE_STRING:
355
+ param = (*(void **) parameters[i] != NULL) ? rb_tainted_str_new2(*(char **) parameters[i]) : Qnil;
356
+ break;
357
+ case NATIVE_POINTER:
358
+ param = rbffi_Pointer_NewInstance(*(void **) parameters[i]);
359
+ break;
360
+ case NATIVE_BOOL:
361
+ param = (*(void **) parameters[i]) ? Qtrue : Qfalse;
362
+ break;
363
+
364
+ case NATIVE_FUNCTION:
365
+ case NATIVE_CALLBACK:
366
+ param = rbffi_NativeValue_ToRuby(cbInfo->parameterTypes[i],
367
+ rb_ary_entry(cbInfo->rbParameterTypes, i), parameters[i], Qnil);
368
+ break;
369
+ default:
370
+ param = Qnil;
371
+ break;
372
+ }
373
+ rbParams[i] = param;
374
+ }
375
+ rbReturnValue = rb_funcall2(fn->rbProc, id_call, cbInfo->parameterCount, rbParams);
376
+ if (rbReturnValue == Qnil || TYPE(rbReturnValue) == T_NIL) {
377
+ memset(retval, 0, cbInfo->ffiReturnType->size);
378
+ } else switch (cbInfo->returnType->nativeType) {
379
+ case NATIVE_INT8:
380
+ case NATIVE_INT16:
381
+ case NATIVE_INT32:
382
+ *((ffi_sarg *) retval) = NUM2INT(rbReturnValue);
383
+ break;
384
+ case NATIVE_UINT8:
385
+ case NATIVE_UINT16:
386
+ case NATIVE_UINT32:
387
+ *((ffi_arg *) retval) = NUM2UINT(rbReturnValue);
388
+ break;
389
+ case NATIVE_INT64:
390
+ *((int64_t *) retval) = NUM2LL(rbReturnValue);
391
+ break;
392
+ case NATIVE_UINT64:
393
+ *((uint64_t *) retval) = NUM2ULL(rbReturnValue);
394
+ break;
395
+ case NATIVE_FLOAT32:
396
+ *((float *) retval) = (float) NUM2DBL(rbReturnValue);
397
+ break;
398
+ case NATIVE_FLOAT64:
399
+ *((double *) retval) = NUM2DBL(rbReturnValue);
400
+ break;
401
+ case NATIVE_POINTER:
402
+ if (TYPE(rbReturnValue) == T_DATA && rb_obj_is_kind_of(rbReturnValue, rbffi_PointerClass)) {
403
+ *((void **) retval) = ((AbstractMemory *) DATA_PTR(rbReturnValue))->address;
404
+ } else {
405
+ // Default to returning NULL if not a value pointer object. handles nil case as well
406
+ *((void **) retval) = NULL;
407
+ }
408
+ break;
409
+ case NATIVE_BOOL:
410
+ *((ffi_sarg *) retval) = TYPE(rbReturnValue) == T_TRUE ? 1 : 0;
411
+ break;
412
+
413
+ case NATIVE_FUNCTION:
414
+ case NATIVE_CALLBACK:
415
+ if (TYPE(rbReturnValue) == T_DATA && rb_obj_is_kind_of(rbReturnValue, rbffi_PointerClass)) {
416
+
417
+ *((void **) retval) = ((AbstractMemory *) DATA_PTR(rbReturnValue))->address;
418
+
419
+ } else if (rb_obj_is_kind_of(rbReturnValue, rb_cProc) || rb_respond_to(rbReturnValue, id_call)) {
420
+ VALUE function;
421
+
422
+ function = rbffi_Function_ForProc(cbInfo->rbReturnType, rbReturnValue);
423
+
424
+ *((void **) retval) = ((AbstractMemory *) DATA_PTR(function))->address;
425
+ } else {
426
+ *((void **) retval) = NULL;
427
+ }
428
+ break;
429
+
430
+ default:
431
+ *((ffi_arg *) retval) = 0;
432
+ break;
433
+ }
434
+ }
435
+
436
+
437
+ #if defined(HAVE_LIBFFI) && !defined(HAVE_FFI_CLOSURE_ALLOC)
438
+ /*
439
+ * versions of ffi_closure_alloc, ffi_closure_free and ffi_prep_closure_loc for older
440
+ * system libffi versions.
441
+ */
442
+ static void*
443
+ ffi_closure_alloc(size_t size, void** code)
444
+ {
445
+ void* closure;
446
+ closure = mmap(NULL, size, PROT_READ | PROT_WRITE,
447
+ MAP_ANON | MAP_PRIVATE, -1, 0);
448
+ if (closure == (void *) -1) {
449
+ return NULL;
450
+ }
451
+ memset(closure, 0, size);
452
+ *code = closure;
453
+ return closure;
454
+ }
455
+
456
+ static void
457
+ ffi_closure_free(void* ptr)
458
+ {
459
+ if (ptr != NULL && ptr != (void *) -1) {
460
+ munmap(ptr, sizeof(ffi_closure));
461
+ }
462
+ }
463
+
464
+ ffi_status
465
+ ffi_prep_closure_loc(ffi_closure* closure, ffi_cif* cif,
466
+ void (*fun)(ffi_cif*, void*, void**, void*),
467
+ void* user_data, void* code)
468
+ {
469
+ ffi_status retval = ffi_prep_closure(closure, cif, fun, user_data);
470
+ if (retval == FFI_OK) {
471
+ mprotect(closure, sizeof(ffi_closure), PROT_READ | PROT_EXEC);
472
+ }
473
+ return retval;
474
+ }
475
+
476
+ #endif /* HAVE_FFI_CLOSURE_ALLOC */
477
+
478
+ void
479
+ rbffi_Function_Init(VALUE moduleFFI)
480
+ {
481
+ rbffi_FunctionInfo_Init(moduleFFI);
482
+ rbffi_FunctionClass = rb_define_class_under(moduleFFI, "Function", rbffi_PointerClass);
483
+
484
+ rb_global_variable(&rbffi_FunctionClass);
485
+ rb_define_alloc_func(rbffi_FunctionClass, function_allocate);
486
+
487
+ rb_define_method(rbffi_FunctionClass, "initialize", function_initialize, -1);
488
+ rb_define_method(rbffi_FunctionClass, "call", function_call, -1);
489
+ rb_define_method(rbffi_FunctionClass, "attach", function_attach, 2);
490
+ rb_define_method(rbffi_FunctionClass, "free", function_release, 0);
491
+ rb_define_method(rbffi_FunctionClass, "autorelease=", function_set_autorelease, 1);
492
+ rb_define_method(rbffi_FunctionClass, "autorelease", function_autorelease_p, 0);
493
+ rb_define_method(rbffi_FunctionClass, "autorelease?", function_autorelease_p, 0);
494
+
495
+ id_call = rb_intern("call");
496
+ id_cbtable = rb_intern("@__ffi_callback_table__");
497
+ }
498
+