ffi 1.1.2 → 1.17.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (706) hide show
  1. checksums.yaml +7 -0
  2. checksums.yaml.gz.sig +0 -0
  3. data/CHANGELOG.md +456 -0
  4. data/COPYING +49 -0
  5. data/Gemfile +21 -0
  6. data/LICENSE +21 -11
  7. data/LICENSE.SPECS +22 -0
  8. data/README.md +137 -0
  9. data/Rakefile +166 -156
  10. data/ext/ffi_c/AbstractMemory.c +240 -140
  11. data/ext/ffi_c/AbstractMemory.h +25 -15
  12. data/ext/ffi_c/ArrayType.c +78 -29
  13. data/ext/ffi_c/ArrayType.h +22 -12
  14. data/ext/ffi_c/Buffer.c +121 -60
  15. data/ext/ffi_c/Call.c +184 -137
  16. data/ext/ffi_c/Call.h +46 -22
  17. data/ext/ffi_c/ClosurePool.c +101 -43
  18. data/ext/ffi_c/ClosurePool.h +24 -13
  19. data/ext/ffi_c/DynamicLibrary.c +139 -57
  20. data/ext/ffi_c/DynamicLibrary.h +70 -12
  21. data/ext/ffi_c/Function.c +372 -273
  22. data/ext/ffi_c/Function.h +25 -20
  23. data/ext/ffi_c/FunctionInfo.c +135 -49
  24. data/ext/ffi_c/LastError.c +97 -30
  25. data/ext/ffi_c/LastError.h +21 -12
  26. data/ext/ffi_c/LongDouble.c +15 -9
  27. data/ext/ffi_c/LongDouble.h +21 -16
  28. data/ext/ffi_c/MappedType.c +88 -38
  29. data/ext/ffi_c/MappedType.h +21 -14
  30. data/ext/ffi_c/MemoryPointer.c +91 -38
  31. data/ext/ffi_c/MemoryPointer.h +22 -19
  32. data/ext/ffi_c/MethodHandle.c +52 -51
  33. data/ext/ffi_c/MethodHandle.h +24 -14
  34. data/ext/ffi_c/Platform.c +28 -62
  35. data/ext/ffi_c/Platform.h +21 -12
  36. data/ext/ffi_c/Pointer.c +145 -86
  37. data/ext/ffi_c/Pointer.h +23 -19
  38. data/ext/ffi_c/Struct.c +363 -163
  39. data/ext/ffi_c/Struct.h +39 -21
  40. data/ext/ffi_c/StructByValue.c +72 -37
  41. data/ext/ffi_c/StructByValue.h +21 -12
  42. data/ext/ffi_c/StructLayout.c +308 -94
  43. data/ext/ffi_c/Thread.c +73 -270
  44. data/ext/ffi_c/Thread.h +32 -35
  45. data/ext/ffi_c/Type.c +147 -84
  46. data/ext/ffi_c/Type.h +27 -11
  47. data/ext/ffi_c/Types.c +40 -24
  48. data/ext/ffi_c/Types.h +24 -21
  49. data/ext/ffi_c/Variadic.c +148 -64
  50. data/ext/ffi_c/compat.h +45 -37
  51. data/ext/ffi_c/extconf.rb +86 -32
  52. data/ext/ffi_c/ffi.c +32 -21
  53. data/ext/ffi_c/libffi/.allow-ai-service +0 -0
  54. data/ext/ffi_c/libffi/.appveyor/site.exp +16 -0
  55. data/ext/ffi_c/libffi/.appveyor/unix-noexec.exp +7 -0
  56. data/ext/ffi_c/libffi/.appveyor.yml +84 -0
  57. data/ext/ffi_c/libffi/.ci/ar-lib +270 -0
  58. data/ext/ffi_c/libffi/.ci/bfin-sim.exp +58 -0
  59. data/ext/ffi_c/libffi/.ci/build-cross-in-container.sh +18 -0
  60. data/ext/ffi_c/libffi/.ci/build-in-container.sh +10 -0
  61. data/ext/ffi_c/libffi/.ci/build.sh +124 -0
  62. data/ext/ffi_c/libffi/.ci/compile +351 -0
  63. data/ext/ffi_c/libffi/.ci/install.sh +78 -0
  64. data/ext/ffi_c/libffi/.ci/m32r-sim.exp +58 -0
  65. data/ext/ffi_c/libffi/.ci/moxie-sim.exp +60 -0
  66. data/ext/ffi_c/libffi/.ci/msvs-detect +1103 -0
  67. data/ext/ffi_c/libffi/.ci/or1k-sim.exp +58 -0
  68. data/ext/ffi_c/libffi/.ci/powerpc-eabisim.exp +58 -0
  69. data/ext/ffi_c/libffi/.ci/site.exp +29 -0
  70. data/ext/ffi_c/libffi/.ci/wine-sim.exp +55 -0
  71. data/ext/ffi_c/libffi/.circleci/config.yml +156 -0
  72. data/ext/ffi_c/libffi/.gitattributes +4 -0
  73. data/ext/ffi_c/libffi/.github/issue_template.md +10 -0
  74. data/ext/ffi_c/libffi/.github/workflows/build.yml +479 -0
  75. data/ext/ffi_c/libffi/.github/workflows/emscripten.yml +172 -0
  76. data/ext/ffi_c/libffi/.gitignore +46 -0
  77. data/ext/ffi_c/libffi/{ChangeLog → ChangeLog.old} +6366 -3559
  78. data/ext/ffi_c/libffi/LICENSE +1 -1
  79. data/ext/ffi_c/libffi/LICENSE-BUILDTOOLS +353 -0
  80. data/ext/ffi_c/libffi/Makefile.am +131 -170
  81. data/ext/ffi_c/libffi/Makefile.in +1317 -1010
  82. data/ext/ffi_c/libffi/README.md +531 -0
  83. data/ext/ffi_c/libffi/acinclude.m4 +289 -4
  84. data/ext/ffi_c/libffi/autogen.sh +2 -0
  85. data/ext/ffi_c/libffi/compile +222 -17
  86. data/ext/ffi_c/libffi/config.guess +1033 -780
  87. data/ext/ffi_c/libffi/config.sub +1393 -1206
  88. data/ext/ffi_c/libffi/configure +16340 -10740
  89. data/ext/ffi_c/libffi/configure.ac +226 -271
  90. data/ext/ffi_c/libffi/configure.host +330 -4
  91. data/ext/ffi_c/libffi/doc/Makefile.am +3 -0
  92. data/ext/ffi_c/libffi/doc/Makefile.in +818 -0
  93. data/ext/ffi_c/libffi/doc/libffi.texi +525 -79
  94. data/ext/ffi_c/libffi/doc/version.texi +4 -4
  95. data/ext/ffi_c/libffi/fficonfig.h.in +41 -52
  96. data/ext/ffi_c/libffi/generate-darwin-source-and-headers.py +312 -0
  97. data/ext/ffi_c/libffi/include/Makefile.am +3 -3
  98. data/ext/ffi_c/libffi/include/Makefile.in +201 -79
  99. data/ext/ffi_c/libffi/include/ffi.h.in +190 -99
  100. data/ext/ffi_c/libffi/include/ffi_cfi.h +76 -0
  101. data/ext/ffi_c/libffi/include/ffi_common.h +72 -14
  102. data/ext/ffi_c/libffi/include/tramp.h +45 -0
  103. data/ext/ffi_c/libffi/install-sh +231 -210
  104. data/ext/ffi_c/libffi/libffi.map.in +81 -0
  105. data/ext/ffi_c/libffi/libffi.pc.in +3 -2
  106. data/ext/ffi_c/libffi/libffi.xcodeproj/project.pbxproj +997 -0
  107. data/ext/ffi_c/libffi/libtool-ldflags +106 -0
  108. data/ext/ffi_c/libffi/libtool-version +2 -2
  109. data/ext/ffi_c/libffi/ltmain.sh +3868 -2068
  110. data/ext/ffi_c/libffi/m4/asmcfi.m4 +13 -0
  111. data/ext/ffi_c/libffi/m4/ax_append_flag.m4 +50 -0
  112. data/ext/ffi_c/libffi/m4/ax_cc_maxopt.m4 +47 -25
  113. data/ext/ffi_c/libffi/m4/ax_cflags_warn_all.m4 +110 -147
  114. data/ext/ffi_c/libffi/m4/ax_check_compile_flag.m4 +53 -0
  115. data/ext/ffi_c/libffi/m4/ax_compiler_vendor.m4 +75 -19
  116. data/ext/ffi_c/libffi/m4/ax_configure_args.m4 +8 -29
  117. data/ext/ffi_c/libffi/m4/ax_enable_builddir.m4 +11 -9
  118. data/ext/ffi_c/libffi/m4/ax_gcc_archflag.m4 +104 -52
  119. data/ext/ffi_c/libffi/m4/ax_gcc_x86_cpuid.m4 +18 -8
  120. data/ext/ffi_c/libffi/m4/ax_prepend_flag.m4 +51 -0
  121. data/ext/ffi_c/libffi/m4/ax_require_defined.m4 +37 -0
  122. data/ext/ffi_c/libffi/make_sunver.pl +333 -0
  123. data/ext/ffi_c/libffi/man/Makefile.am +2 -2
  124. data/ext/ffi_c/libffi/man/Makefile.in +156 -57
  125. data/ext/ffi_c/libffi/man/ffi.3 +10 -0
  126. data/ext/ffi_c/libffi/man/ffi_prep_cif.3 +6 -4
  127. data/ext/ffi_c/libffi/man/ffi_prep_cif_var.3 +73 -0
  128. data/ext/ffi_c/libffi/missing +153 -314
  129. data/ext/ffi_c/libffi/msvc_build/aarch64/Ffi_staticLib.sln +33 -0
  130. data/ext/ffi_c/libffi/msvc_build/aarch64/Ffi_staticLib.vcxproj +130 -0
  131. data/ext/ffi_c/libffi/msvc_build/aarch64/Ffi_staticLib.vcxproj.filters +57 -0
  132. data/ext/ffi_c/libffi/msvc_build/aarch64/Ffi_staticLib.vcxproj.user +4 -0
  133. data/ext/ffi_c/libffi/{include/ffi.h.vc64 → msvc_build/aarch64/aarch64_include/ffi.h} +151 -67
  134. data/ext/ffi_c/libffi/msvc_build/aarch64/aarch64_include/fficonfig.h +219 -0
  135. data/ext/ffi_c/libffi/msvcc.sh +176 -20
  136. data/ext/ffi_c/libffi/src/aarch64/ffi.c +1142 -0
  137. data/ext/ffi_c/libffi/src/aarch64/ffitarget.h +97 -0
  138. data/ext/ffi_c/libffi/src/aarch64/internal.h +100 -0
  139. data/ext/ffi_c/libffi/src/aarch64/sysv.S +695 -0
  140. data/ext/ffi_c/libffi/src/aarch64/win64_armasm.S +506 -0
  141. data/ext/ffi_c/libffi/src/alpha/ffi.c +335 -98
  142. data/ext/ffi_c/libffi/src/alpha/ffitarget.h +10 -1
  143. data/ext/ffi_c/libffi/src/alpha/internal.h +23 -0
  144. data/ext/ffi_c/libffi/src/alpha/osf.S +161 -266
  145. data/ext/ffi_c/libffi/src/arc/arcompact.S +210 -0
  146. data/ext/ffi_c/libffi/src/arc/ffi.c +443 -0
  147. data/ext/ffi_c/libffi/src/arc/ffitarget.h +67 -0
  148. data/ext/ffi_c/libffi/src/arm/ffi.c +706 -513
  149. data/ext/ffi_c/libffi/src/arm/ffitarget.h +32 -8
  150. data/ext/ffi_c/libffi/src/arm/internal.h +17 -0
  151. data/ext/ffi_c/libffi/src/arm/sysv.S +372 -413
  152. data/ext/ffi_c/libffi/src/arm/sysv_msvc_arm32.S +311 -0
  153. data/ext/ffi_c/libffi/src/avr32/ffitarget.h +6 -1
  154. data/ext/ffi_c/libffi/src/bfin/ffi.c +196 -0
  155. data/ext/ffi_c/libffi/src/bfin/ffitarget.h +43 -0
  156. data/ext/ffi_c/libffi/src/bfin/sysv.S +179 -0
  157. data/ext/ffi_c/libffi/src/closures.c +565 -73
  158. data/ext/ffi_c/libffi/src/cris/ffi.c +10 -7
  159. data/ext/ffi_c/libffi/src/cris/ffitarget.h +6 -1
  160. data/ext/ffi_c/libffi/src/csky/ffi.c +395 -0
  161. data/ext/ffi_c/libffi/src/csky/ffitarget.h +63 -0
  162. data/ext/ffi_c/libffi/src/csky/sysv.S +371 -0
  163. data/ext/ffi_c/libffi/src/debug.c +8 -3
  164. data/ext/ffi_c/libffi/src/dlmalloc.c +23 -13
  165. data/ext/ffi_c/libffi/src/frv/ffi.c +2 -2
  166. data/ext/ffi_c/libffi/src/frv/ffitarget.h +6 -1
  167. data/ext/ffi_c/libffi/src/ia64/ffi.c +47 -13
  168. data/ext/ffi_c/libffi/src/ia64/ffitarget.h +8 -2
  169. data/ext/ffi_c/libffi/src/ia64/unix.S +28 -3
  170. data/ext/ffi_c/libffi/src/java_raw_api.c +23 -5
  171. data/ext/ffi_c/libffi/src/kvx/asm.h +5 -0
  172. data/ext/ffi_c/libffi/src/kvx/ffi.c +273 -0
  173. data/ext/ffi_c/libffi/src/kvx/ffitarget.h +75 -0
  174. data/ext/ffi_c/libffi/src/kvx/sysv.S +127 -0
  175. data/ext/ffi_c/libffi/src/loongarch64/ffi.c +624 -0
  176. data/ext/ffi_c/libffi/src/loongarch64/ffitarget.h +82 -0
  177. data/ext/ffi_c/libffi/src/loongarch64/sysv.S +327 -0
  178. data/ext/ffi_c/libffi/src/m32r/ffi.c +32 -15
  179. data/ext/ffi_c/libffi/src/m32r/ffitarget.h +6 -1
  180. data/ext/ffi_c/libffi/src/m68k/ffi.c +87 -13
  181. data/ext/ffi_c/libffi/src/m68k/ffitarget.h +6 -1
  182. data/ext/ffi_c/libffi/src/m68k/sysv.S +119 -32
  183. data/ext/ffi_c/libffi/src/m88k/ffi.c +400 -0
  184. data/ext/ffi_c/libffi/src/m88k/ffitarget.h +49 -0
  185. data/ext/ffi_c/libffi/src/m88k/obsd.S +209 -0
  186. data/ext/ffi_c/libffi/src/metag/ffi.c +330 -0
  187. data/ext/ffi_c/libffi/{fficonfig.hw → src/metag/ffitarget.h} +22 -26
  188. data/ext/ffi_c/libffi/src/metag/sysv.S +311 -0
  189. data/ext/ffi_c/libffi/src/microblaze/ffi.c +321 -0
  190. data/ext/ffi_c/libffi/src/microblaze/ffitarget.h +53 -0
  191. data/ext/ffi_c/libffi/src/microblaze/sysv.S +302 -0
  192. data/ext/ffi_c/libffi/src/mips/ffi.c +382 -101
  193. data/ext/ffi_c/libffi/src/mips/ffitarget.h +23 -14
  194. data/ext/ffi_c/libffi/src/mips/n32.S +321 -89
  195. data/ext/ffi_c/libffi/src/mips/o32.S +211 -31
  196. data/ext/ffi_c/libffi/src/moxie/eabi.S +55 -82
  197. data/ext/ffi_c/libffi/src/moxie/ffi.c +100 -66
  198. data/ext/ffi_c/libffi/src/moxie/ffitarget.h +52 -0
  199. data/ext/ffi_c/libffi/src/nios2/ffi.c +304 -0
  200. data/ext/ffi_c/libffi/src/nios2/ffitarget.h +52 -0
  201. data/ext/ffi_c/libffi/src/nios2/sysv.S +136 -0
  202. data/ext/ffi_c/libffi/src/or1k/ffi.c +341 -0
  203. data/ext/ffi_c/libffi/src/or1k/ffitarget.h +58 -0
  204. data/ext/ffi_c/libffi/src/or1k/sysv.S +107 -0
  205. data/ext/ffi_c/libffi/src/pa/ffi.c +62 -108
  206. data/ext/ffi_c/libffi/src/pa/ffi64.c +614 -0
  207. data/ext/ffi_c/libffi/src/pa/ffitarget.h +28 -12
  208. data/ext/ffi_c/libffi/src/pa/hpux32.S +87 -38
  209. data/ext/ffi_c/libffi/src/pa/hpux64.S +681 -0
  210. data/ext/ffi_c/libffi/src/pa/linux.S +109 -39
  211. data/ext/ffi_c/libffi/src/powerpc/aix.S +245 -7
  212. data/ext/ffi_c/libffi/src/powerpc/aix_closure.S +253 -4
  213. data/ext/ffi_c/libffi/src/powerpc/asm.h +2 -2
  214. data/ext/ffi_c/libffi/src/powerpc/darwin.S +2 -7
  215. data/ext/ffi_c/libffi/src/powerpc/darwin_closure.S +22 -26
  216. data/ext/ffi_c/libffi/src/powerpc/ffi.c +105 -1378
  217. data/ext/ffi_c/libffi/src/powerpc/ffi_darwin.c +157 -52
  218. data/ext/ffi_c/libffi/src/powerpc/ffi_linux64.c +1153 -0
  219. data/ext/ffi_c/libffi/src/powerpc/ffi_powerpc.h +105 -0
  220. data/ext/ffi_c/libffi/src/powerpc/ffi_sysv.c +923 -0
  221. data/ext/ffi_c/libffi/src/powerpc/ffitarget.h +110 -45
  222. data/ext/ffi_c/libffi/src/powerpc/linux64.S +191 -85
  223. data/ext/ffi_c/libffi/src/powerpc/linux64_closure.S +438 -108
  224. data/ext/ffi_c/libffi/src/powerpc/ppc_closure.S +138 -68
  225. data/ext/ffi_c/libffi/src/powerpc/sysv.S +73 -119
  226. data/ext/ffi_c/libffi/src/powerpc/t-aix +5 -0
  227. data/ext/ffi_c/libffi/src/prep_cif.c +131 -25
  228. data/ext/ffi_c/libffi/src/raw_api.c +18 -5
  229. data/ext/ffi_c/libffi/src/riscv/ffi.c +514 -0
  230. data/ext/ffi_c/libffi/src/riscv/ffitarget.h +69 -0
  231. data/ext/ffi_c/libffi/src/riscv/sysv.S +293 -0
  232. data/ext/ffi_c/libffi/src/s390/ffi.c +294 -318
  233. data/ext/ffi_c/libffi/src/s390/ffitarget.h +9 -1
  234. data/ext/ffi_c/libffi/src/s390/internal.h +11 -0
  235. data/ext/ffi_c/libffi/src/s390/sysv.S +257 -366
  236. data/ext/ffi_c/libffi/src/sh/ffi.c +4 -3
  237. data/ext/ffi_c/libffi/src/sh/ffitarget.h +6 -1
  238. data/ext/ffi_c/libffi/src/sh64/ffi.c +3 -2
  239. data/ext/ffi_c/libffi/src/sh64/ffitarget.h +6 -1
  240. data/ext/ffi_c/libffi/src/sparc/ffi.c +326 -527
  241. data/ext/ffi_c/libffi/src/sparc/ffi64.c +630 -0
  242. data/ext/ffi_c/libffi/src/sparc/ffitarget.h +20 -7
  243. data/ext/ffi_c/libffi/src/sparc/internal.h +26 -0
  244. data/ext/ffi_c/libffi/src/sparc/v8.S +364 -234
  245. data/ext/ffi_c/libffi/src/sparc/v9.S +340 -207
  246. data/ext/ffi_c/libffi/src/tile/ffi.c +355 -0
  247. data/ext/ffi_c/libffi/src/tile/ffitarget.h +65 -0
  248. data/ext/ffi_c/libffi/src/tile/tile.S +360 -0
  249. data/ext/ffi_c/libffi/src/tramp.c +716 -0
  250. data/ext/ffi_c/libffi/src/types.c +48 -19
  251. data/ext/ffi_c/libffi/src/vax/elfbsd.S +195 -0
  252. data/ext/ffi_c/libffi/src/vax/ffi.c +276 -0
  253. data/ext/ffi_c/libffi/src/vax/ffitarget.h +49 -0
  254. data/ext/ffi_c/libffi/src/wasm32/ffi.c +947 -0
  255. data/ext/ffi_c/libffi/src/wasm32/ffitarget.h +62 -0
  256. data/ext/ffi_c/libffi/src/x86/asmnames.h +30 -0
  257. data/ext/ffi_c/libffi/src/x86/ffi.c +686 -498
  258. data/ext/ffi_c/libffi/src/x86/ffi64.c +444 -129
  259. data/ext/ffi_c/libffi/src/x86/ffitarget.h +79 -36
  260. data/ext/ffi_c/libffi/src/x86/ffiw64.c +361 -0
  261. data/ext/ffi_c/libffi/src/x86/internal.h +43 -0
  262. data/ext/ffi_c/libffi/src/x86/internal64.h +36 -0
  263. data/ext/ffi_c/libffi/src/x86/sysv.S +1198 -394
  264. data/ext/ffi_c/libffi/src/x86/sysv_intel.S +998 -0
  265. data/ext/ffi_c/libffi/src/x86/unix64.S +585 -300
  266. data/ext/ffi_c/libffi/src/x86/win64.S +244 -458
  267. data/ext/ffi_c/libffi/src/x86/win64_intel.S +238 -0
  268. data/ext/ffi_c/libffi/src/xtensa/ffi.c +306 -0
  269. data/ext/ffi_c/libffi/src/xtensa/ffitarget.h +57 -0
  270. data/ext/ffi_c/libffi/src/xtensa/sysv.S +268 -0
  271. data/ext/ffi_c/libffi/stamp-h.in +1 -0
  272. data/ext/ffi_c/libffi/testsuite/Makefile.am +80 -73
  273. data/ext/ffi_c/libffi/testsuite/Makefile.in +233 -117
  274. data/ext/ffi_c/libffi/testsuite/emscripten/build-tests.sh +54 -0
  275. data/ext/ffi_c/libffi/testsuite/emscripten/build.sh +63 -0
  276. data/ext/ffi_c/libffi/testsuite/emscripten/conftest.py +86 -0
  277. data/ext/ffi_c/libffi/testsuite/emscripten/node-tests.sh +48 -0
  278. data/ext/ffi_c/libffi/testsuite/emscripten/test.html +7 -0
  279. data/ext/ffi_c/libffi/testsuite/emscripten/test_libffi.py +51 -0
  280. data/ext/ffi_c/libffi/testsuite/lib/libffi.exp +360 -28
  281. data/ext/ffi_c/libffi/testsuite/lib/target-libpath.exp +22 -2
  282. data/ext/ffi_c/libffi/testsuite/libffi.bhaible/Makefile +28 -0
  283. data/ext/ffi_c/libffi/testsuite/libffi.bhaible/README +78 -0
  284. data/ext/ffi_c/libffi/testsuite/libffi.bhaible/alignof.h +50 -0
  285. data/ext/ffi_c/libffi/testsuite/libffi.bhaible/bhaible.exp +63 -0
  286. data/ext/ffi_c/libffi/testsuite/libffi.bhaible/test-call.c +1746 -0
  287. data/ext/ffi_c/libffi/testsuite/libffi.bhaible/test-callback.c +2885 -0
  288. data/ext/ffi_c/libffi/testsuite/libffi.bhaible/testcases.c +747 -0
  289. data/ext/ffi_c/libffi/testsuite/libffi.call/align_mixed.c +46 -0
  290. data/ext/ffi_c/libffi/testsuite/libffi.call/align_stdcall.c +46 -0
  291. data/ext/ffi_c/libffi/testsuite/libffi.call/bpo_38748.c +41 -0
  292. data/ext/ffi_c/libffi/testsuite/libffi.call/call.exp +28 -6
  293. data/ext/ffi_c/libffi/testsuite/libffi.call/callback.c +99 -0
  294. data/ext/ffi_c/libffi/testsuite/libffi.call/callback2.c +108 -0
  295. data/ext/ffi_c/libffi/testsuite/libffi.call/callback3.c +114 -0
  296. data/ext/ffi_c/libffi/testsuite/libffi.call/callback4.c +119 -0
  297. data/ext/ffi_c/libffi/testsuite/libffi.call/err_bad_typedef.c +2 -2
  298. data/ext/ffi_c/libffi/testsuite/libffi.call/ffitest.h +52 -42
  299. data/ext/ffi_c/libffi/testsuite/libffi.call/float1.c +3 -1
  300. data/ext/ffi_c/libffi/testsuite/libffi.call/float2.c +15 -12
  301. data/ext/ffi_c/libffi/testsuite/libffi.call/float3.c +4 -2
  302. data/ext/ffi_c/libffi/testsuite/libffi.call/float_va.c +110 -0
  303. data/ext/ffi_c/libffi/testsuite/libffi.call/many.c +6 -16
  304. data/ext/ffi_c/libffi/testsuite/libffi.call/many2.c +57 -0
  305. data/ext/ffi_c/libffi/testsuite/libffi.call/many_double.c +70 -0
  306. data/ext/ffi_c/libffi/testsuite/libffi.call/many_mixed.c +78 -0
  307. data/ext/ffi_c/libffi/testsuite/libffi.call/negint.c +0 -1
  308. data/ext/ffi_c/libffi/testsuite/libffi.call/offsets.c +46 -0
  309. data/ext/ffi_c/libffi/testsuite/libffi.call/pr1172638.c +127 -0
  310. data/ext/ffi_c/libffi/testsuite/libffi.call/return_dbl.c +1 -0
  311. data/ext/ffi_c/libffi/testsuite/libffi.call/return_ldl.c +1 -1
  312. data/ext/ffi_c/libffi/testsuite/libffi.call/return_ll1.c +5 -0
  313. data/ext/ffi_c/libffi/testsuite/libffi.call/return_sc.c +1 -1
  314. data/ext/ffi_c/libffi/testsuite/libffi.call/return_sl.c +4 -0
  315. data/ext/ffi_c/libffi/testsuite/libffi.call/return_uc.c +1 -1
  316. data/ext/ffi_c/libffi/testsuite/libffi.call/return_ul.c +4 -0
  317. data/ext/ffi_c/libffi/testsuite/libffi.call/s55.c +60 -0
  318. data/ext/ffi_c/libffi/testsuite/libffi.call/strlen.c +9 -10
  319. data/ext/ffi_c/libffi/testsuite/libffi.call/strlen2.c +49 -0
  320. data/ext/ffi_c/libffi/testsuite/libffi.call/strlen3.c +49 -0
  321. data/ext/ffi_c/libffi/testsuite/libffi.call/strlen4.c +55 -0
  322. data/ext/ffi_c/libffi/testsuite/libffi.call/struct1.c +9 -7
  323. data/ext/ffi_c/libffi/testsuite/libffi.call/struct10.c +57 -0
  324. data/ext/ffi_c/libffi/testsuite/libffi.call/struct2.c +7 -7
  325. data/ext/ffi_c/libffi/testsuite/libffi.call/struct3.c +7 -6
  326. data/ext/ffi_c/libffi/testsuite/libffi.call/struct4.c +9 -8
  327. data/ext/ffi_c/libffi/testsuite/libffi.call/struct5.c +9 -8
  328. data/ext/ffi_c/libffi/testsuite/libffi.call/struct6.c +9 -9
  329. data/ext/ffi_c/libffi/testsuite/libffi.call/struct7.c +9 -9
  330. data/ext/ffi_c/libffi/testsuite/libffi.call/struct8.c +9 -8
  331. data/ext/ffi_c/libffi/testsuite/libffi.call/struct9.c +9 -8
  332. data/ext/ffi_c/libffi/testsuite/libffi.call/struct_by_value_2.c +63 -0
  333. data/ext/ffi_c/libffi/testsuite/libffi.call/struct_by_value_3.c +65 -0
  334. data/ext/ffi_c/libffi/testsuite/libffi.call/struct_by_value_3f.c +65 -0
  335. data/ext/ffi_c/libffi/testsuite/libffi.call/struct_by_value_4.c +67 -0
  336. data/ext/ffi_c/libffi/testsuite/libffi.call/struct_by_value_4f.c +67 -0
  337. data/ext/ffi_c/libffi/testsuite/libffi.call/struct_by_value_big.c +93 -0
  338. data/ext/ffi_c/libffi/testsuite/libffi.call/struct_by_value_small.c +61 -0
  339. data/ext/ffi_c/libffi/testsuite/libffi.call/struct_return_2H.c +63 -0
  340. data/ext/ffi_c/libffi/testsuite/libffi.call/struct_return_8H.c +90 -0
  341. data/ext/ffi_c/libffi/testsuite/libffi.call/uninitialized.c +61 -0
  342. data/ext/ffi_c/libffi/testsuite/libffi.call/va_1.c +78 -0
  343. data/ext/ffi_c/libffi/testsuite/libffi.call/va_2.c +220 -0
  344. data/ext/ffi_c/libffi/testsuite/libffi.call/va_3.c +154 -0
  345. data/ext/ffi_c/libffi/testsuite/libffi.call/va_struct1.c +134 -0
  346. data/ext/ffi_c/libffi/testsuite/libffi.call/va_struct2.c +134 -0
  347. data/ext/ffi_c/libffi/testsuite/libffi.call/va_struct3.c +140 -0
  348. data/ext/ffi_c/libffi/testsuite/libffi.closures/closure.exp +67 -0
  349. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/closure_fn0.c +3 -2
  350. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/closure_fn1.c +2 -0
  351. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/closure_fn2.c +2 -0
  352. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/closure_fn3.c +21 -1
  353. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/closure_fn4.c +2 -0
  354. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/closure_fn5.c +2 -0
  355. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/closure_fn6.c +2 -0
  356. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/closure_loc_fn0.c +7 -6
  357. data/ext/ffi_c/libffi/testsuite/{libffi.call/closure_stdcall.c → libffi.closures/closure_simple.c} +13 -16
  358. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_12byte.c +22 -4
  359. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_16byte.c +26 -4
  360. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_18byte.c +28 -4
  361. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_19byte.c +33 -4
  362. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_1_1byte.c +8 -4
  363. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_20byte.c +23 -5
  364. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_20byte1.c +25 -5
  365. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_24byte.c +40 -8
  366. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_2byte.c +17 -4
  367. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_3_1byte.c +23 -4
  368. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_3byte1.c +17 -4
  369. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_3byte2.c +17 -4
  370. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_3float.c +113 -0
  371. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_4_1byte.c +26 -4
  372. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_4byte.c +17 -4
  373. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_5_1_byte.c +33 -5
  374. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_5byte.c +23 -5
  375. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_64byte.c +29 -5
  376. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_6_1_byte.c +32 -5
  377. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_6byte.c +28 -6
  378. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_7_1_byte.c +43 -5
  379. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_7byte.c +29 -5
  380. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_8byte.c +18 -4
  381. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_9byte1.c +18 -5
  382. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_9byte2.c +18 -6
  383. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_align_double.c +23 -5
  384. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_align_float.c +23 -5
  385. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_align_longdouble.c +24 -5
  386. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_align_longdouble_split.c +44 -31
  387. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_align_longdouble_split2.c +44 -9
  388. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_align_pointer.c +23 -5
  389. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_align_sint16.c +22 -5
  390. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_align_sint32.c +22 -7
  391. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_align_sint64.c +22 -5
  392. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_align_uint16.c +22 -5
  393. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_align_uint32.c +23 -5
  394. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_align_uint64.c +23 -5
  395. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_dbls_struct.c +7 -5
  396. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_double_va.c +19 -10
  397. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_float.c +4 -0
  398. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_longdouble.c +14 -4
  399. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_longdouble_va.c +32 -12
  400. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_many_mixed_args.c +70 -0
  401. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_many_mixed_float_double.c +55 -0
  402. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_multi_schar.c +5 -1
  403. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_multi_sshort.c +6 -1
  404. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_multi_sshortchar.c +9 -1
  405. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_multi_uchar.c +9 -1
  406. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_multi_ushort.c +6 -2
  407. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_multi_ushortchar.c +9 -1
  408. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_pointer.c +6 -1
  409. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_pointer_stack.c +21 -9
  410. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_schar.c +3 -0
  411. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_sint.c +2 -0
  412. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_sshort.c +3 -0
  413. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_struct_va1.c +125 -0
  414. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_uchar.c +3 -0
  415. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_uint.c +4 -0
  416. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_uint_va.c +49 -0
  417. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_ulong_va.c +49 -0
  418. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_ulonglong.c +7 -5
  419. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_ushort.c +3 -0
  420. data/ext/ffi_c/libffi/testsuite/libffi.closures/ffitest.h +1 -0
  421. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/huge_struct.c +41 -40
  422. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/nested_struct.c +38 -15
  423. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/nested_struct1.c +9 -9
  424. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/nested_struct10.c +19 -6
  425. data/ext/ffi_c/libffi/testsuite/libffi.closures/nested_struct11.c +137 -0
  426. data/ext/ffi_c/libffi/testsuite/libffi.closures/nested_struct12.c +86 -0
  427. data/ext/ffi_c/libffi/testsuite/libffi.closures/nested_struct13.c +115 -0
  428. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/nested_struct2.c +15 -6
  429. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/nested_struct3.c +15 -6
  430. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/nested_struct4.c +14 -5
  431. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/nested_struct5.c +14 -5
  432. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/nested_struct6.c +17 -6
  433. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/nested_struct7.c +14 -5
  434. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/nested_struct8.c +17 -6
  435. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/nested_struct9.c +17 -6
  436. data/ext/ffi_c/libffi/testsuite/libffi.closures/single_entry_structs1.c +86 -0
  437. data/ext/ffi_c/libffi/testsuite/libffi.closures/single_entry_structs2.c +102 -0
  438. data/ext/ffi_c/libffi/testsuite/libffi.closures/single_entry_structs3.c +101 -0
  439. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/stret_large.c +7 -7
  440. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/stret_large2.c +7 -7
  441. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/stret_medium.c +6 -6
  442. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/stret_medium2.c +6 -6
  443. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/testclosure.c +8 -3
  444. data/ext/ffi_c/libffi/testsuite/{libffi.special → libffi.closures}/unwindtest.cc +4 -10
  445. data/ext/ffi_c/libffi/testsuite/{libffi.special → libffi.closures}/unwindtest_ffi_call.cc +4 -2
  446. data/ext/ffi_c/libffi/testsuite/libffi.complex/cls_align_complex.inc +91 -0
  447. data/ext/ffi_c/libffi/testsuite/libffi.complex/cls_align_complex_double.c +10 -0
  448. data/ext/ffi_c/libffi/testsuite/libffi.complex/cls_align_complex_float.c +10 -0
  449. data/ext/ffi_c/libffi/testsuite/libffi.complex/cls_align_complex_longdouble.c +10 -0
  450. data/ext/ffi_c/libffi/testsuite/libffi.complex/cls_complex.inc +42 -0
  451. data/ext/ffi_c/libffi/testsuite/libffi.complex/cls_complex_double.c +10 -0
  452. data/ext/ffi_c/libffi/testsuite/libffi.complex/cls_complex_float.c +10 -0
  453. data/ext/ffi_c/libffi/testsuite/libffi.complex/cls_complex_longdouble.c +10 -0
  454. data/ext/ffi_c/libffi/testsuite/libffi.complex/cls_complex_struct.inc +71 -0
  455. data/ext/ffi_c/libffi/testsuite/libffi.complex/cls_complex_struct_double.c +10 -0
  456. data/ext/ffi_c/libffi/testsuite/libffi.complex/cls_complex_struct_float.c +10 -0
  457. data/ext/ffi_c/libffi/testsuite/libffi.complex/cls_complex_struct_longdouble.c +10 -0
  458. data/ext/ffi_c/libffi/testsuite/libffi.complex/cls_complex_va.inc +80 -0
  459. data/ext/ffi_c/libffi/testsuite/libffi.complex/cls_complex_va_double.c +10 -0
  460. data/ext/ffi_c/libffi/testsuite/libffi.complex/cls_complex_va_float.c +16 -0
  461. data/ext/ffi_c/libffi/testsuite/libffi.complex/cls_complex_va_longdouble.c +10 -0
  462. data/ext/ffi_c/libffi/testsuite/{libffi.special/special.exp → libffi.complex/complex.exp} +9 -8
  463. data/ext/ffi_c/libffi/testsuite/libffi.complex/complex.inc +51 -0
  464. data/ext/ffi_c/libffi/testsuite/libffi.complex/complex_defs_double.inc +7 -0
  465. data/ext/ffi_c/libffi/testsuite/libffi.complex/complex_defs_float.inc +7 -0
  466. data/ext/ffi_c/libffi/testsuite/libffi.complex/complex_defs_longdouble.inc +7 -0
  467. data/ext/ffi_c/libffi/testsuite/libffi.complex/complex_double.c +10 -0
  468. data/ext/ffi_c/libffi/testsuite/libffi.complex/complex_float.c +10 -0
  469. data/ext/ffi_c/libffi/testsuite/libffi.complex/complex_int.c +86 -0
  470. data/ext/ffi_c/libffi/testsuite/libffi.complex/complex_longdouble.c +10 -0
  471. data/ext/ffi_c/libffi/testsuite/libffi.complex/ffitest.h +1 -0
  472. data/ext/ffi_c/libffi/testsuite/libffi.complex/many_complex.inc +78 -0
  473. data/ext/ffi_c/libffi/testsuite/libffi.complex/many_complex_double.c +10 -0
  474. data/ext/ffi_c/libffi/testsuite/libffi.complex/many_complex_float.c +10 -0
  475. data/ext/ffi_c/libffi/testsuite/libffi.complex/many_complex_longdouble.c +10 -0
  476. data/ext/ffi_c/libffi/testsuite/libffi.complex/return_complex.inc +37 -0
  477. data/ext/ffi_c/libffi/testsuite/libffi.complex/return_complex1.inc +41 -0
  478. data/ext/ffi_c/libffi/testsuite/libffi.complex/return_complex1_double.c +10 -0
  479. data/ext/ffi_c/libffi/testsuite/libffi.complex/return_complex1_float.c +10 -0
  480. data/ext/ffi_c/libffi/testsuite/libffi.complex/return_complex1_longdouble.c +10 -0
  481. data/ext/ffi_c/libffi/testsuite/libffi.complex/return_complex2.inc +44 -0
  482. data/ext/ffi_c/libffi/testsuite/libffi.complex/return_complex2_double.c +10 -0
  483. data/ext/ffi_c/libffi/testsuite/libffi.complex/return_complex2_float.c +10 -0
  484. data/ext/ffi_c/libffi/testsuite/libffi.complex/return_complex2_longdouble.c +10 -0
  485. data/ext/ffi_c/libffi/testsuite/libffi.complex/return_complex_double.c +10 -0
  486. data/ext/ffi_c/libffi/testsuite/libffi.complex/return_complex_float.c +10 -0
  487. data/ext/ffi_c/libffi/testsuite/libffi.complex/return_complex_longdouble.c +10 -0
  488. data/ext/ffi_c/libffi/testsuite/libffi.go/aa-direct.c +34 -0
  489. data/ext/ffi_c/libffi/testsuite/libffi.go/closure1.c +28 -0
  490. data/ext/ffi_c/libffi/testsuite/libffi.go/ffitest.h +1 -0
  491. data/ext/ffi_c/libffi/testsuite/libffi.go/go.exp +36 -0
  492. data/ext/ffi_c/libffi/testsuite/libffi.go/static-chain.h +19 -0
  493. data/ext/ffi_c/libffi.bsd.mk +13 -7
  494. data/ext/ffi_c/libffi.darwin.mk +25 -13
  495. data/ext/ffi_c/libffi.gnu.mk +4 -3
  496. data/ext/ffi_c/libffi.mk +11 -6
  497. data/ext/ffi_c/libffi.vc.mk +1 -1
  498. data/ext/ffi_c/libffi.vc64.mk +1 -1
  499. data/ext/ffi_c/rbffi.h +23 -15
  500. data/ext/ffi_c/rbffi_endian.h +11 -2
  501. data/ffi.gemspec +42 -0
  502. data/lib/ffi/abstract_memory.rb +44 -0
  503. data/lib/ffi/autopointer.rb +65 -68
  504. data/lib/ffi/compat.rb +43 -0
  505. data/lib/ffi/data_converter.rb +67 -0
  506. data/lib/ffi/dynamic_library.rb +118 -0
  507. data/lib/ffi/enum.rb +166 -25
  508. data/lib/ffi/errno.rb +21 -11
  509. data/lib/ffi/ffi.rb +28 -11
  510. data/lib/ffi/function.rb +71 -0
  511. data/lib/ffi/io.rb +25 -15
  512. data/lib/ffi/library.rb +227 -136
  513. data/lib/ffi/library_path.rb +109 -0
  514. data/lib/ffi/managedstruct.rb +64 -35
  515. data/lib/ffi/platform/aarch64-darwin/types.conf +130 -0
  516. data/lib/ffi/platform/aarch64-freebsd/types.conf +128 -0
  517. data/lib/ffi/platform/aarch64-freebsd12/types.conf +181 -0
  518. data/lib/ffi/platform/aarch64-linux/types.conf +175 -0
  519. data/lib/ffi/platform/aarch64-openbsd/types.conf +134 -0
  520. data/lib/ffi/platform/aarch64-windows/types.conf +52 -0
  521. data/lib/ffi/platform/arm-freebsd/types.conf +152 -0
  522. data/lib/ffi/platform/arm-freebsd12/types.conf +152 -0
  523. data/lib/ffi/platform/arm-linux/types.conf +110 -82
  524. data/lib/ffi/platform/hppa1.1-linux/types.conf +178 -0
  525. data/lib/ffi/platform/hppa2.0-linux/types.conf +178 -0
  526. data/lib/ffi/platform/i386-cygwin/types.conf +3 -0
  527. data/lib/ffi/platform/i386-darwin/types.conf +63 -63
  528. data/lib/ffi/platform/i386-freebsd/types.conf +89 -89
  529. data/lib/ffi/platform/i386-freebsd12/types.conf +152 -0
  530. data/lib/ffi/platform/i386-linux/types.conf +77 -77
  531. data/lib/ffi/platform/i386-netbsd/types.conf +87 -87
  532. data/lib/ffi/platform/i386-openbsd/types.conf +89 -87
  533. data/lib/ffi/platform/i386-solaris/types.conf +96 -96
  534. data/lib/ffi/platform/i386-windows/types.conf +43 -96
  535. data/lib/ffi/platform/ia64-linux/types.conf +79 -79
  536. data/lib/ffi/platform/loongarch64-linux/types.conf +141 -0
  537. data/lib/ffi/platform/mips-linux/types.conf +79 -79
  538. data/lib/ffi/platform/mips64-linux/types.conf +104 -0
  539. data/lib/ffi/platform/mips64el-linux/types.conf +104 -0
  540. data/lib/ffi/platform/mipsel-linux/types.conf +79 -79
  541. data/lib/ffi/platform/mipsisa32r6-linux/types.conf +102 -0
  542. data/lib/ffi/platform/mipsisa32r6el-linux/types.conf +102 -0
  543. data/lib/ffi/platform/mipsisa64r6-linux/types.conf +104 -0
  544. data/lib/ffi/platform/mipsisa64r6el-linux/types.conf +104 -0
  545. data/lib/ffi/platform/powerpc-aix/types.conf +155 -155
  546. data/lib/ffi/platform/powerpc-darwin/types.conf +63 -63
  547. data/lib/ffi/platform/powerpc-linux/types.conf +108 -78
  548. data/lib/ffi/platform/powerpc-openbsd/types.conf +156 -0
  549. data/lib/ffi/platform/powerpc64-linux/types.conf +104 -0
  550. data/lib/ffi/platform/powerpc64le-linux/types.conf +100 -0
  551. data/lib/ffi/platform/riscv64-linux/types.conf +104 -0
  552. data/lib/ffi/platform/s390-linux/types.conf +79 -79
  553. data/lib/ffi/platform/s390x-linux/types.conf +79 -79
  554. data/lib/ffi/platform/sparc-linux/types.conf +79 -79
  555. data/lib/ffi/platform/sparc-solaris/types.conf +103 -103
  556. data/lib/ffi/platform/sparcv9-linux/types.conf +102 -0
  557. data/lib/ffi/platform/sparcv9-openbsd/types.conf +156 -0
  558. data/lib/ffi/platform/sparcv9-solaris/types.conf +103 -103
  559. data/lib/ffi/platform/sw_64-linux/types.conf +141 -0
  560. data/lib/ffi/platform/x86_64-cygwin/types.conf +3 -0
  561. data/lib/ffi/platform/x86_64-darwin/types.conf +93 -63
  562. data/lib/ffi/platform/x86_64-dragonflybsd/types.conf +130 -0
  563. data/lib/ffi/platform/x86_64-freebsd/types.conf +90 -88
  564. data/lib/ffi/platform/x86_64-freebsd12/types.conf +158 -0
  565. data/lib/ffi/platform/x86_64-haiku/types.conf +117 -0
  566. data/lib/ffi/platform/x86_64-linux/types.conf +107 -77
  567. data/lib/ffi/platform/x86_64-msys/types.conf +119 -0
  568. data/lib/ffi/platform/x86_64-netbsd/types.conf +89 -87
  569. data/lib/ffi/platform/x86_64-openbsd/types.conf +100 -92
  570. data/lib/ffi/platform/x86_64-solaris/types.conf +96 -96
  571. data/lib/ffi/platform/x86_64-windows/types.conf +52 -0
  572. data/lib/ffi/platform.rb +91 -34
  573. data/lib/ffi/pointer.rb +78 -33
  574. data/lib/ffi/struct.rb +100 -66
  575. data/lib/ffi/struct_by_reference.rb +72 -0
  576. data/lib/ffi/struct_layout.rb +96 -0
  577. data/lib/ffi/struct_layout_builder.rb +84 -19
  578. data/lib/ffi/tools/const_generator.rb +11 -8
  579. data/lib/ffi/tools/generator.rb +48 -1
  580. data/lib/ffi/tools/generator_task.rb +13 -16
  581. data/lib/ffi/tools/struct_generator.rb +6 -5
  582. data/lib/ffi/tools/types_generator.rb +14 -10
  583. data/lib/ffi/types.rb +96 -51
  584. data/lib/ffi/union.rb +23 -12
  585. data/lib/ffi/variadic.rb +44 -29
  586. data/lib/ffi/version.rb +3 -0
  587. data/lib/ffi.rb +20 -8
  588. data/rakelib/ffi_gem_helper.rb +65 -0
  589. data/samples/getlogin.rb +8 -0
  590. data/samples/getpid.rb +8 -0
  591. data/samples/gettimeofday.rb +18 -0
  592. data/samples/hello.rb +8 -0
  593. data/samples/hello_ractor.rb +11 -0
  594. data/samples/inotify.rb +60 -0
  595. data/samples/pty.rb +75 -0
  596. data/samples/qsort.rb +20 -0
  597. data/samples/qsort_ractor.rb +28 -0
  598. data/sig/ffi/abstract_memory.rbs +165 -0
  599. data/sig/ffi/auto_pointer.rbs +27 -0
  600. data/sig/ffi/buffer.rbs +18 -0
  601. data/sig/ffi/data_converter.rbs +10 -0
  602. data/sig/ffi/dynamic_library.rbs +9 -0
  603. data/sig/ffi/enum.rbs +38 -0
  604. data/sig/ffi/function.rbs +39 -0
  605. data/sig/ffi/library.rbs +42 -0
  606. data/sig/ffi/native_type.rbs +86 -0
  607. data/sig/ffi/pointer.rbs +42 -0
  608. data/sig/ffi/struct.rbs +76 -0
  609. data/sig/ffi/struct_by_reference.rbs +11 -0
  610. data/sig/ffi/struct_by_value.rbs +7 -0
  611. data/sig/ffi/struct_layout.rbs +9 -0
  612. data/sig/ffi/struct_layout_builder.rbs +5 -0
  613. data/sig/ffi/type.rbs +39 -0
  614. data/sig/ffi.rbs +26 -0
  615. data.tar.gz.sig +0 -0
  616. metadata +508 -268
  617. metadata.gz.sig +1 -0
  618. data/History.txt +0 -1
  619. data/README.rdoc +0 -102
  620. data/ext/ffi_c/DataConverter.c +0 -91
  621. data/ext/ffi_c/Ffi_c.iml +0 -12
  622. data/ext/ffi_c/StructByReference.c +0 -158
  623. data/ext/ffi_c/StructByReference.h +0 -50
  624. data/ext/ffi_c/libffi/ChangeLog.libffi +0 -584
  625. data/ext/ffi_c/libffi/ChangeLog.libgcj +0 -40
  626. data/ext/ffi_c/libffi/ChangeLog.v1 +0 -764
  627. data/ext/ffi_c/libffi/Makefile.vc +0 -141
  628. data/ext/ffi_c/libffi/Makefile.vc64 +0 -141
  629. data/ext/ffi_c/libffi/README +0 -342
  630. data/ext/ffi_c/libffi/aclocal.m4 +0 -1873
  631. data/ext/ffi_c/libffi/build-ios.sh +0 -67
  632. data/ext/ffi_c/libffi/depcomp +0 -630
  633. data/ext/ffi_c/libffi/doc/libffi.info +0 -593
  634. data/ext/ffi_c/libffi/doc/stamp-vti +0 -4
  635. data/ext/ffi_c/libffi/include/ffi.h.vc +0 -427
  636. data/ext/ffi_c/libffi/m4/ax_check_compiler_flags.m4 +0 -76
  637. data/ext/ffi_c/libffi/m4/libtool.m4 +0 -7831
  638. data/ext/ffi_c/libffi/m4/ltoptions.m4 +0 -369
  639. data/ext/ffi_c/libffi/m4/ltsugar.m4 +0 -123
  640. data/ext/ffi_c/libffi/m4/ltversion.m4 +0 -23
  641. data/ext/ffi_c/libffi/m4/lt~obsolete.m4 +0 -98
  642. data/ext/ffi_c/libffi/mdate-sh +0 -201
  643. data/ext/ffi_c/libffi/src/arm/gentramp.sh +0 -118
  644. data/ext/ffi_c/libffi/src/arm/trampoline.S +0 -4450
  645. data/ext/ffi_c/libffi/src/x86/darwin.S +0 -444
  646. data/ext/ffi_c/libffi/src/x86/darwin64.S +0 -416
  647. data/ext/ffi_c/libffi/src/x86/freebsd.S +0 -458
  648. data/ext/ffi_c/libffi/src/x86/win32.S +0 -1065
  649. data/ext/ffi_c/libffi/testsuite/lib/libffi-dg.exp +0 -300
  650. data/ext/ffi_c/libffi/testsuite/libffi.call/many_win32.c +0 -63
  651. data/ext/ffi_c/libffi/testsuite/libffi.call/strlen_win32.c +0 -44
  652. data/ext/ffi_c/libffi/testsuite/libffi.special/ffitestcxx.h +0 -96
  653. data/ext/ffi_c/libffi/texinfo.tex +0 -7210
  654. data/ext/ffi_c/win32/stdint.h +0 -199
  655. data/gen/Rakefile +0 -30
  656. data/lib/Lib.iml +0 -13
  657. data/spec/ffi/Ffi.iml +0 -12
  658. data/spec/ffi/async_callback_spec.rb +0 -46
  659. data/spec/ffi/bool_spec.rb +0 -40
  660. data/spec/ffi/buffer_spec.rb +0 -215
  661. data/spec/ffi/callback_spec.rb +0 -668
  662. data/spec/ffi/custom_param_type.rb +0 -47
  663. data/spec/ffi/custom_type_spec.rb +0 -85
  664. data/spec/ffi/dup_spec.rb +0 -65
  665. data/spec/ffi/enum_spec.rb +0 -227
  666. data/spec/ffi/errno_spec.rb +0 -29
  667. data/spec/ffi/ffi_spec.rb +0 -40
  668. data/spec/ffi/function_spec.rb +0 -87
  669. data/spec/ffi/library_spec.rb +0 -208
  670. data/spec/ffi/long_double.rb +0 -41
  671. data/spec/ffi/managed_struct_spec.rb +0 -72
  672. data/spec/ffi/number_spec.rb +0 -247
  673. data/spec/ffi/pointer_spec.rb +0 -214
  674. data/spec/ffi/rbx/attach_function_spec.rb +0 -28
  675. data/spec/ffi/rbx/memory_pointer_spec.rb +0 -111
  676. data/spec/ffi/rbx/spec_helper.rb +0 -1
  677. data/spec/ffi/rbx/struct_spec.rb +0 -13
  678. data/spec/ffi/spec_helper.rb +0 -35
  679. data/spec/ffi/string_spec.rb +0 -119
  680. data/spec/ffi/strptr_spec.rb +0 -61
  681. data/spec/ffi/struct_callback_spec.rb +0 -80
  682. data/spec/ffi/struct_initialize_spec.rb +0 -46
  683. data/spec/ffi/struct_packed_spec.rb +0 -62
  684. data/spec/ffi/struct_spec.rb +0 -719
  685. data/spec/ffi/typedef_spec.rb +0 -89
  686. data/spec/ffi/union_spec.rb +0 -76
  687. data/spec/ffi/variadic_spec.rb +0 -103
  688. data/spec/spec.opts +0 -4
  689. data/tasks/ann.rake +0 -80
  690. data/tasks/extension.rake +0 -32
  691. data/tasks/gem.rake +0 -199
  692. data/tasks/git.rake +0 -41
  693. data/tasks/notes.rake +0 -27
  694. data/tasks/post_load.rake +0 -34
  695. data/tasks/rdoc.rake +0 -50
  696. data/tasks/rubyforge.rake +0 -55
  697. data/tasks/setup.rb +0 -301
  698. data/tasks/spec.rake +0 -54
  699. data/tasks/svn.rake +0 -47
  700. data/tasks/test.rake +0 -40
  701. data/tasks/yard.rake +0 -11
  702. data/ext/ffi_c/libffi/testsuite/libffi.call/{pyobjc-tc.c → pyobjc_tc.c} +0 -0
  703. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/cls_double.c +0 -0
  704. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/err_bad_abi.c +0 -0
  705. data/ext/ffi_c/libffi/testsuite/{libffi.call → libffi.closures}/problem1.c +0 -0
  706. data/lib/ffi/platform/{i486-gnu → i386-gnu}/types.conf +84 -84
data/ext/ffi_c/Function.c CHANGED
@@ -1,44 +1,49 @@
1
1
  /*
2
2
  * Copyright (c) 2009-2011 Wayne Meissner
3
3
  *
4
+ * Copyright (c) 2008-2013, Ruby FFI project contributors
4
5
  * All rights reserved.
5
6
  *
6
- * This file is part of ruby-ffi.
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.
7
17
  *
8
- * This code is free software: you can redistribute it and/or modify it under
9
- * the terms of the GNU Lesser General Public License version 3 only, as
10
- * published by the Free Software Foundation.
11
- *
12
- * This code is distributed in the hope that it will be useful, but WITHOUT
13
- * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
14
- * FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License
15
- * version 3 for more details.
16
- *
17
- * You should have received a copy of the GNU Lesser General Public License
18
- * version 3 along with this work. If not, see <http://www.gnu.org/licenses/>.
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.
19
28
  */
20
29
 
21
- #include "MethodHandle.h"
22
-
23
30
  #ifndef _MSC_VER
24
31
  #include <sys/param.h>
25
32
  #endif
26
33
  #include <sys/types.h>
27
34
  #ifndef _WIN32
28
35
  # include <sys/mman.h>
36
+ # include <unistd.h>
29
37
  #endif
38
+
30
39
  #include <stdio.h>
31
- #ifndef _MSC_VER
32
40
  #include <stdint.h>
33
41
  #include <stdbool.h>
34
- #else
35
- typedef int bool;
36
- #define true 1
37
- #define false 0
38
- #endif
39
42
  #include <ruby.h>
40
- #if defined(_MSC_VER) && !defined(INT8_MIN)
41
- # include "win32/stdint.h"
43
+ #include <ruby/thread.h>
44
+
45
+ #if HAVE_RB_EXT_RACTOR_SAFE
46
+ #include <ruby/ractor.h>
42
47
  #endif
43
48
 
44
49
  #include <ffi.h>
@@ -58,11 +63,15 @@ typedef int bool;
58
63
  #include "LastError.h"
59
64
  #include "Call.h"
60
65
  #include "ClosurePool.h"
61
- #include "Function.h"
62
66
  #include "MappedType.h"
63
67
  #include "Thread.h"
64
68
  #include "LongDouble.h"
69
+ #include "MethodHandle.h"
70
+ #include "Function.h"
71
+
72
+ #define DEFER_ASYNC_CALLBACK 1
65
73
 
74
+ struct async_cb_dispatcher;
66
75
  typedef struct Function_ {
67
76
  Pointer base;
68
77
  FunctionType* info;
@@ -71,33 +80,46 @@ typedef struct Function_ {
71
80
  Closure* closure;
72
81
  VALUE rbProc;
73
82
  VALUE rbFunctionInfo;
83
+ #if defined(DEFER_ASYNC_CALLBACK)
84
+ struct async_cb_dispatcher *dispatcher;
85
+ #endif
74
86
  } Function;
75
87
 
76
- static void function_mark(Function *);
77
- static void function_free(Function *);
88
+ static void function_mark(void *data);
89
+ static void function_compact(void *data);
90
+ static void function_free(void *data);
91
+ static size_t function_memsize(const void *data);
78
92
  static VALUE function_init(VALUE self, VALUE rbFunctionInfo, VALUE rbProc);
79
93
  static void callback_invoke(ffi_cif* cif, void* retval, void** parameters, void* user_data);
80
94
  static bool callback_prep(void* ctx, void* code, Closure* closure, char* errmsg, size_t errmsgsize);
81
95
  static void* callback_with_gvl(void* data);
82
-
83
- #define DEFER_ASYNC_CALLBACK 1
84
-
96
+ static VALUE invoke_callback(VALUE data);
97
+ static VALUE save_callback_exception(VALUE data, VALUE exc);
85
98
 
86
99
  #if defined(DEFER_ASYNC_CALLBACK)
87
100
  static VALUE async_cb_event(void *);
88
101
  static VALUE async_cb_call(void *);
89
102
  #endif
90
103
 
91
- #ifdef HAVE_RB_THREAD_CALL_WITH_GVL
92
- extern void *rb_thread_call_with_gvl(void *(*func)(void *), void *data1);
93
- #endif
104
+ extern int ruby_thread_has_gvl_p(void);
105
+ extern int ruby_native_thread_p(void);
106
+
107
+ static const rb_data_type_t function_data_type = {
108
+ .wrap_struct_name = "FFI::Function",
109
+ .function = {
110
+ .dmark = function_mark,
111
+ .dfree = function_free,
112
+ .dsize = function_memsize,
113
+ ffi_compact_callback( function_compact )
114
+ },
115
+ .parent = &rbffi_pointer_data_type,
116
+ // IMPORTANT: WB_PROTECTED objects must only use the RB_OBJ_WRITE()
117
+ // macro to update VALUE references, as to trigger write barriers.
118
+ .flags = RUBY_TYPED_FREE_IMMEDIATELY | RUBY_TYPED_WB_PROTECTED | FFI_RUBY_TYPED_FROZEN_SHAREABLE
119
+ };
94
120
 
95
121
  VALUE rbffi_FunctionClass = Qnil;
96
122
 
97
- #if defined(DEFER_ASYNC_CALLBACK)
98
- static VALUE async_cb_thread = Qnil;
99
- #endif
100
-
101
123
  static ID id_call = 0, id_to_native = 0, id_from_native = 0, id_cbtable = 0, id_cb_ref = 0;
102
124
 
103
125
  struct gvl_callback {
@@ -105,8 +127,12 @@ struct gvl_callback {
105
127
  void* retval;
106
128
  void** parameters;
107
129
  bool done;
130
+ rbffi_frame_t *frame;
108
131
  #if defined(DEFER_ASYNC_CALLBACK)
132
+ struct async_cb_dispatcher *dispatcher;
109
133
  struct gvl_callback* next;
134
+
135
+ /* Signal when the callback has finished and retval is set */
110
136
  # ifndef _WIN32
111
137
  pthread_cond_t async_cond;
112
138
  pthread_mutex_t async_mutex;
@@ -118,22 +144,78 @@ struct gvl_callback {
118
144
 
119
145
 
120
146
  #if defined(DEFER_ASYNC_CALLBACK)
121
- static struct gvl_callback* async_cb_list = NULL;
147
+ struct async_cb_dispatcher {
148
+ /* the Ractor-local dispatcher thread */
149
+ VALUE thread;
150
+
151
+ /* single linked list of pending callbacks */
152
+ struct gvl_callback* async_cb_list;
153
+
154
+ /* Signal new entries in async_cb_list */
122
155
  # ifndef _WIN32
123
- static pthread_mutex_t async_cb_mutex = PTHREAD_MUTEX_INITIALIZER;
124
- static pthread_cond_t async_cb_cond = PTHREAD_COND_INITIALIZER;
125
- # if !defined(HAVE_RB_THREAD_BLOCKING_REGION)
126
- static int async_cb_pipe[2];
127
- # endif
156
+ pthread_mutex_t async_cb_mutex;
157
+ pthread_cond_t async_cb_cond;
128
158
  # else
129
- static HANDLE async_cb_cond;
130
- static CRITICAL_SECTION async_cb_lock;
131
- # if !defined(HAVE_RB_THREAD_BLOCKING_REGION)
132
- static int async_cb_pipe[2];
133
- # endif
159
+ HANDLE async_cb_cond;
160
+ CRITICAL_SECTION async_cb_lock;
134
161
  # endif
135
- #endif
162
+ };
163
+
164
+ #if HAVE_RB_EXT_RACTOR_SAFE
165
+ static void
166
+ async_cb_dispatcher_mark(void *ptr)
167
+ {
168
+ struct async_cb_dispatcher *ctx = (struct async_cb_dispatcher *)ptr;
169
+ if (ctx) {
170
+ rb_gc_mark(ctx->thread);
171
+ }
172
+ }
173
+
174
+ static void
175
+ async_cb_dispatcher_free(void *ptr)
176
+ {
177
+ struct async_cb_dispatcher *ctx = (struct async_cb_dispatcher *)ptr;
178
+ if (ctx) {
179
+ xfree(ctx);
180
+ }
181
+ }
182
+
183
+ struct rb_ractor_local_storage_type async_cb_dispatcher_key_type = {
184
+ async_cb_dispatcher_mark,
185
+ async_cb_dispatcher_free,
186
+ };
187
+
188
+ static rb_ractor_local_key_t async_cb_dispatcher_key;
136
189
 
190
+ static struct async_cb_dispatcher *
191
+ async_cb_dispatcher_get(void)
192
+ {
193
+ struct async_cb_dispatcher *ctx = (struct async_cb_dispatcher *)rb_ractor_local_storage_ptr(async_cb_dispatcher_key);
194
+ return ctx;
195
+ }
196
+
197
+ static void
198
+ async_cb_dispatcher_set(struct async_cb_dispatcher *ctx)
199
+ {
200
+ rb_ractor_local_storage_ptr_set(async_cb_dispatcher_key, ctx);
201
+ }
202
+ #else
203
+ // for ruby 2.x
204
+ static struct async_cb_dispatcher *async_cb_dispatcher = NULL;
205
+
206
+ static struct async_cb_dispatcher *
207
+ async_cb_dispatcher_get(void)
208
+ {
209
+ return async_cb_dispatcher;
210
+ }
211
+
212
+ static void
213
+ async_cb_dispatcher_set(struct async_cb_dispatcher *ctx)
214
+ {
215
+ async_cb_dispatcher = ctx;
216
+ }
217
+ #endif
218
+ #endif
137
219
 
138
220
  static VALUE
139
221
  function_allocate(VALUE klass)
@@ -141,28 +223,39 @@ function_allocate(VALUE klass)
141
223
  Function *fn;
142
224
  VALUE obj;
143
225
 
144
- obj = Data_Make_Struct(klass, Function, function_mark, function_free, fn);
226
+ obj = TypedData_Make_Struct(klass, Function, &function_data_type, fn);
145
227
 
146
228
  fn->base.memory.flags = MEM_RD;
147
- fn->base.rbParent = Qnil;
148
- fn->rbProc = Qnil;
149
- fn->rbFunctionInfo = Qnil;
229
+ RB_OBJ_WRITE(obj, &fn->base.rbParent, Qnil);
230
+ RB_OBJ_WRITE(obj, &fn->rbProc, Qnil);
231
+ RB_OBJ_WRITE(obj, &fn->rbFunctionInfo, Qnil);
150
232
  fn->autorelease = true;
151
233
 
152
234
  return obj;
153
235
  }
154
236
 
155
237
  static void
156
- function_mark(Function *fn)
238
+ function_mark(void *data)
157
239
  {
158
- rb_gc_mark(fn->base.rbParent);
159
- rb_gc_mark(fn->rbProc);
160
- rb_gc_mark(fn->rbFunctionInfo);
240
+ Function *fn = (Function *)data;
241
+ rb_gc_mark_movable(fn->base.rbParent);
242
+ rb_gc_mark_movable(fn->rbProc);
243
+ rb_gc_mark_movable(fn->rbFunctionInfo);
161
244
  }
162
245
 
163
246
  static void
164
- function_free(Function *fn)
247
+ function_compact(void *data)
165
248
  {
249
+ Function *fn = (Function *)data;
250
+ ffi_gc_location(fn->base.rbParent);
251
+ ffi_gc_location(fn->rbProc);
252
+ ffi_gc_location(fn->rbFunctionInfo);
253
+ }
254
+
255
+ static void
256
+ function_free(void *data)
257
+ {
258
+ Function *fn = (Function *)data;
166
259
  if (fn->methodHandle != NULL) {
167
260
  rbffi_MethodHandle_Free(fn->methodHandle);
168
261
  }
@@ -174,6 +267,20 @@ function_free(Function *fn)
174
267
  xfree(fn);
175
268
  }
176
269
 
270
+ static size_t
271
+ function_memsize(const void *data)
272
+ {
273
+ const Function *fn = (const Function *)data;
274
+ size_t memsize = sizeof(Function);
275
+
276
+ // Would be nice to better account for MethodHandle and Closure too.
277
+ if (fn->closure) {
278
+ memsize += sizeof(Closure);
279
+ }
280
+
281
+ return memsize;
282
+ }
283
+
177
284
  /*
178
285
  * @param [Type, Symbol] return_type return type for the function
179
286
  * @param [Array<Type, Symbol>] param_types array of parameters types
@@ -191,7 +298,7 @@ function_free(Function *fn)
191
298
  static VALUE
192
299
  function_initialize(int argc, VALUE* argv, VALUE self)
193
300
  {
194
-
301
+
195
302
  VALUE rbReturnType = Qnil, rbParamTypes = Qnil, rbProc = Qnil, rbOptions = Qnil;
196
303
  VALUE rbFunctionInfo = Qnil;
197
304
  VALUE infoArgv[3];
@@ -199,11 +306,11 @@ function_initialize(int argc, VALUE* argv, VALUE self)
199
306
 
200
307
  nargs = rb_scan_args(argc, argv, "22", &rbReturnType, &rbParamTypes, &rbProc, &rbOptions);
201
308
 
202
- //
203
- // Callback with block,
204
- // e.g. Function.new(:int, [ :int ]) { |i| blah }
205
- // or Function.new(:int, [ :int ], { :convention => :stdcall }) { |i| blah }
206
- //
309
+ /*
310
+ * Callback with block,
311
+ * e.g. Function.new(:int, [ :int ]) { |i| blah }
312
+ * or Function.new(:int, [ :int ], { :convention => :stdcall }) { |i| blah }
313
+ */
207
314
  if (rb_block_given_p()) {
208
315
  if (nargs > 3) {
209
316
  rb_raise(rb_eArgError, "cannot create function with both proc/address and block");
@@ -211,20 +318,21 @@ function_initialize(int argc, VALUE* argv, VALUE self)
211
318
  rbOptions = rbProc;
212
319
  rbProc = rb_block_proc();
213
320
  } else {
214
- // Callback with proc, or Function with address
215
- // e.g. Function.new(:int, [ :int ], Proc.new { |i| })
216
- // Function.new(:int, [ :int ], Proc.new { |i| }, { :convention => :stdcall })
217
- // Function.new(:int, [ :int ], addr)
218
- // Function.new(:int, [ :int ], addr, { :convention => :stdcall })
321
+ /* Callback with proc, or Function with address
322
+ * e.g. Function.new(:int, [ :int ], Proc.new { |i| })
323
+ * Function.new(:int, [ :int ], Proc.new { |i| }, { :convention => :stdcall })
324
+ * Function.new(:int, [ :int ], addr)
325
+ * Function.new(:int, [ :int ], addr, { :convention => :stdcall })
326
+ */
219
327
  }
220
-
328
+
221
329
  infoArgv[0] = rbReturnType;
222
330
  infoArgv[1] = rbParamTypes;
223
331
  infoArgv[2] = rbOptions;
224
332
  rbFunctionInfo = rb_class_new_instance(rbOptions != Qnil ? 3 : 2, infoArgv, rbffi_FunctionTypeClass);
225
333
 
226
334
  function_init(self, rbFunctionInfo, rbProc);
227
-
335
+
228
336
  return self;
229
337
  }
230
338
 
@@ -250,22 +358,22 @@ VALUE
250
358
  rbffi_Function_ForProc(VALUE rbFunctionInfo, VALUE proc)
251
359
  {
252
360
  VALUE callback, cbref, cbTable;
253
- Function* fp;
254
361
 
255
362
  cbref = RTEST(rb_ivar_defined(proc, id_cb_ref)) ? rb_ivar_get(proc, id_cb_ref) : Qnil;
256
363
  /* If the first callback reference has the same function function signature, use it */
257
364
  if (cbref != Qnil && CLASS_OF(cbref) == rbffi_FunctionClass) {
258
- Data_Get_Struct(cbref, Function, fp);
365
+ Function* fp;
366
+ TypedData_Get_Struct(cbref, Function, &function_data_type, fp);
259
367
  if (fp->rbFunctionInfo == rbFunctionInfo) {
260
368
  return cbref;
261
369
  }
262
370
  }
263
-
371
+
264
372
  cbTable = RTEST(rb_ivar_defined(proc, id_cbtable)) ? rb_ivar_get(proc, id_cbtable) : Qnil;
265
373
  if (cbTable != Qnil && (callback = rb_hash_aref(cbTable, rbFunctionInfo)) != Qnil) {
266
374
  return callback;
267
375
  }
268
-
376
+
269
377
  /* No existing function for the proc with that signature, create a new one and cache it */
270
378
  callback = rbffi_Function_NewInstance(rbFunctionInfo, proc);
271
379
  if (cbref == Qnil) {
@@ -273,30 +381,41 @@ rbffi_Function_ForProc(VALUE rbFunctionInfo, VALUE proc)
273
381
  rb_ivar_set(proc, id_cb_ref, callback);
274
382
  } else {
275
383
  /* The proc instance has been used as more than one type of callback, store extras in a hash */
276
- cbTable = rb_hash_new();
277
- rb_ivar_set(proc, id_cbtable, cbTable);
384
+ if(cbTable == Qnil) {
385
+ cbTable = rb_hash_new();
386
+ rb_ivar_set(proc, id_cbtable, cbTable);
387
+ }
278
388
  rb_hash_aset(cbTable, rbFunctionInfo, callback);
279
389
  }
280
390
 
281
391
  return callback;
282
392
  }
283
393
 
394
+ #if !defined(_WIN32) && defined(DEFER_ASYNC_CALLBACK)
395
+ static void
396
+ after_fork_callback(void)
397
+ {
398
+ /* Ensure that a new dispatcher thread is started in a forked process */
399
+ async_cb_dispatcher_set(NULL);
400
+ }
401
+ #endif
402
+
284
403
  static VALUE
285
404
  function_init(VALUE self, VALUE rbFunctionInfo, VALUE rbProc)
286
405
  {
287
406
  Function* fn = NULL;
288
-
289
- Data_Get_Struct(self, Function, fn);
290
407
 
291
- fn->rbFunctionInfo = rbFunctionInfo;
408
+ TypedData_Get_Struct(self, Function, &function_data_type, fn);
409
+
410
+ RB_OBJ_WRITE(self, &fn->rbFunctionInfo, rbFunctionInfo);
292
411
 
293
- Data_Get_Struct(fn->rbFunctionInfo, FunctionType, fn->info);
412
+ TypedData_Get_Struct(fn->rbFunctionInfo, FunctionType, &rbffi_fntype_data_type, fn->info);
294
413
 
295
414
  if (rb_obj_is_kind_of(rbProc, rbffi_PointerClass)) {
296
415
  Pointer* orig;
297
- Data_Get_Struct(rbProc, Pointer, orig);
416
+ TypedData_Get_Struct(rbProc, Pointer, &rbffi_pointer_data_type, orig);
298
417
  fn->base.memory = orig->memory;
299
- fn->base.rbParent = rbProc;
418
+ RB_OBJ_WRITE(self, &fn->base.rbParent, rbProc);
300
419
 
301
420
  } else if (rb_obj_is_kind_of(rbProc, rb_cProc) || rb_respond_to(rbProc, id_call)) {
302
421
  if (fn->info->closurePool == NULL) {
@@ -307,17 +426,31 @@ function_init(VALUE self, VALUE rbFunctionInfo, VALUE rbProc)
307
426
  }
308
427
 
309
428
  #if defined(DEFER_ASYNC_CALLBACK)
310
- if (async_cb_thread == Qnil) {
311
- #if !defined(HAVE_RB_THREAD_BLOCKING_REGION) && defined(_WIN32)
312
- _pipe(async_cb_pipe, 1024, O_BINARY);
313
- #elif !defined(HAVE_RB_THREAD_BLOCKING_REGION)
314
- pipe(async_cb_pipe);
315
- fcntl(async_cb_pipe[0], F_SETFL, fcntl(async_cb_pipe[0], F_GETFL) | O_NONBLOCK);
316
- fcntl(async_cb_pipe[1], F_SETFL, fcntl(async_cb_pipe[1], F_GETFL) | O_NONBLOCK);
429
+ {
430
+ struct async_cb_dispatcher *ctx = async_cb_dispatcher_get();
431
+ if (ctx == NULL) {
432
+ ctx = (struct async_cb_dispatcher*)ALLOC(struct async_cb_dispatcher);
433
+ ctx->async_cb_list = NULL;
434
+
435
+ #if !defined(_WIN32)
436
+ pthread_mutex_init(&ctx->async_cb_mutex, NULL);
437
+ pthread_cond_init(&ctx->async_cb_cond, NULL);
438
+ if( pthread_atfork(NULL, NULL, after_fork_callback) ){
439
+ rb_warn("FFI: unable to register fork callback");
440
+ }
441
+ #else
442
+ InitializeCriticalSection(&ctx->async_cb_lock);
443
+ ctx->async_cb_cond = CreateEvent(NULL, FALSE, FALSE, NULL);
317
444
  #endif
318
- async_cb_thread = rb_thread_create(async_cb_event, NULL);
319
- }
445
+ ctx->thread = rb_thread_create(async_cb_event, ctx);
446
+
447
+ /* Name thread, for better debugging */
448
+ rb_funcall(ctx->thread, rb_intern("name="), 1, rb_str_new2("FFI Callback Dispatcher"));
320
449
 
450
+ async_cb_dispatcher_set(ctx);
451
+ }
452
+ fn->dispatcher = ctx;
453
+ }
321
454
  #endif
322
455
 
323
456
  fn->closure = rbffi_Closure_Alloc(fn->info->closurePool);
@@ -330,8 +463,8 @@ function_init(VALUE self, VALUE rbFunctionInfo, VALUE rbProc)
330
463
  rb_raise(rb_eTypeError, "wrong argument type %s, expected pointer or proc",
331
464
  rb_obj_classname(rbProc));
332
465
  }
333
-
334
- fn->rbProc = rbProc;
466
+
467
+ RB_OBJ_WRITE(self, &fn->rbProc, rbProc);
335
468
 
336
469
  return self;
337
470
  }
@@ -347,7 +480,7 @@ function_call(int argc, VALUE* argv, VALUE self)
347
480
  {
348
481
  Function* fn;
349
482
 
350
- Data_Get_Struct(self, Function, fn);
483
+ TypedData_Get_Struct(self, Function, &function_data_type, fn);
351
484
 
352
485
  return (*fn->info->invoke)(argc, argv, fn->base.memory.address, fn->info);
353
486
  }
@@ -363,9 +496,9 @@ static VALUE
363
496
  function_attach(VALUE self, VALUE module, VALUE name)
364
497
  {
365
498
  Function* fn;
366
- char var[1024];
367
499
 
368
- Data_Get_Struct(self, Function, fn);
500
+ StringValue(name);
501
+ TypedData_Get_Struct(self, Function, &function_data_type, fn);
369
502
 
370
503
  if (fn->info->parameterCount == -1) {
371
504
  rb_raise(rb_eRuntimeError, "cannot attach variadic functions");
@@ -381,16 +514,10 @@ function_attach(VALUE self, VALUE module, VALUE name)
381
514
  fn->methodHandle = rbffi_MethodHandle_Alloc(fn->info, fn->base.memory.address);
382
515
  }
383
516
 
384
- //
385
- // Stash the Function in a module variable so it does not get garbage collected
386
- //
387
- snprintf(var, sizeof(var), "@@%s", StringValueCStr(name));
388
- rb_cv_set(module, var, self);
389
-
390
517
  rb_define_singleton_method(module, StringValueCStr(name),
391
518
  rbffi_MethodHandle_CodeAddress(fn->methodHandle), -1);
392
519
 
393
-
520
+
394
521
  rb_define_method(module, StringValueCStr(name),
395
522
  rbffi_MethodHandle_CodeAddress(fn->methodHandle), -1);
396
523
 
@@ -408,7 +535,8 @@ function_set_autorelease(VALUE self, VALUE autorelease)
408
535
  {
409
536
  Function* fn;
410
537
 
411
- Data_Get_Struct(self, Function, fn);
538
+ rb_check_frozen(self);
539
+ TypedData_Get_Struct(self, Function, &function_data_type, fn);
412
540
 
413
541
  fn->autorelease = RTEST(autorelease);
414
542
 
@@ -420,11 +548,21 @@ function_autorelease_p(VALUE self)
420
548
  {
421
549
  Function* fn;
422
550
 
423
- Data_Get_Struct(self, Function, fn);
551
+ TypedData_Get_Struct(self, Function, &function_data_type, fn);
424
552
 
425
553
  return fn->autorelease ? Qtrue : Qfalse;
426
554
  }
427
555
 
556
+ static VALUE
557
+ function_type(VALUE self)
558
+ {
559
+ Function* fn;
560
+
561
+ TypedData_Get_Struct(self, Function, &function_data_type, fn);
562
+
563
+ return fn->rbFunctionInfo;
564
+ }
565
+
428
566
  /*
429
567
  * call-seq: free
430
568
  * @return [self]
@@ -435,59 +573,57 @@ function_release(VALUE self)
435
573
  {
436
574
  Function* fn;
437
575
 
438
- Data_Get_Struct(self, Function, fn);
576
+ TypedData_Get_Struct(self, Function, &function_data_type, fn);
439
577
 
440
578
  if (fn->closure == NULL) {
441
579
  rb_raise(rb_eRuntimeError, "cannot free function which was not allocated");
442
580
  }
443
-
581
+
444
582
  rbffi_Closure_Free(fn->closure);
445
583
  fn->closure = NULL;
446
-
584
+
447
585
  return self;
448
586
  }
449
587
 
450
588
  static void
451
589
  callback_invoke(ffi_cif* cif, void* retval, void** parameters, void* user_data)
452
590
  {
453
- struct gvl_callback cb;
591
+ Function* fn;
592
+ struct gvl_callback cb = { 0 };
593
+
454
594
  cb.closure = (Closure *) user_data;
455
595
  cb.retval = retval;
456
596
  cb.parameters = parameters;
457
597
  cb.done = false;
598
+ cb.frame = rbffi_frame_current();
599
+ fn = (Function *) cb.closure->info;
600
+
601
+ if (cb.frame != NULL) cb.frame->exc = Qnil;
458
602
 
459
- if (rbffi_thread_has_gvl_p()) {
603
+ if (ruby_native_thread_p()) {
604
+ if(ruby_thread_has_gvl_p()) {
460
605
  callback_with_gvl(&cb);
461
-
462
- #if defined(HAVE_RUBY_NATIVE_THREAD_P) && defined (HAVE_RB_THREAD_CALL_WITH_GVL)
463
- } else if (ruby_native_thread_p()) {
606
+ } else {
464
607
  rb_thread_call_with_gvl(callback_with_gvl, &cb);
465
- #endif
608
+ }
466
609
  #if defined(DEFER_ASYNC_CALLBACK) && !defined(_WIN32)
467
610
  } else {
468
611
  bool empty = false;
612
+ struct async_cb_dispatcher *ctx = fn->dispatcher;
469
613
 
470
614
  pthread_mutex_init(&cb.async_mutex, NULL);
471
615
  pthread_cond_init(&cb.async_cond, NULL);
472
616
 
473
- // Now signal the async callback thread
474
- pthread_mutex_lock(&async_cb_mutex);
475
- empty = async_cb_list == NULL;
476
- cb.next = async_cb_list;
477
- async_cb_list = &cb;
478
- pthread_mutex_unlock(&async_cb_mutex);
479
-
480
- #if !defined(HAVE_RB_THREAD_BLOCKING_REGION)
481
- // Only signal if the list was empty
482
- if (empty) {
483
- char c;
484
- write(async_cb_pipe[1], &c, 1);
485
- }
486
- #else
487
- pthread_cond_signal(&async_cb_cond);
488
- #endif
617
+ /* Now signal the async callback dispatcher thread */
618
+ pthread_mutex_lock(&ctx->async_cb_mutex);
619
+ empty = ctx->async_cb_list == NULL;
620
+ cb.next = ctx->async_cb_list;
621
+ ctx->async_cb_list = &cb;
622
+
623
+ pthread_cond_signal(&ctx->async_cb_cond);
624
+ pthread_mutex_unlock(&ctx->async_cb_mutex);
489
625
 
490
- // Wait for the thread executing the ruby callback to signal it is done
626
+ /* Wait for the thread executing the ruby callback to signal it is done */
491
627
  pthread_mutex_lock(&cb.async_mutex);
492
628
  while (!cb.done) {
493
629
  pthread_cond_wait(&cb.async_cond, &cb.async_mutex);
@@ -499,27 +635,20 @@ callback_invoke(ffi_cif* cif, void* retval, void** parameters, void* user_data)
499
635
  #elif defined(DEFER_ASYNC_CALLBACK) && defined(_WIN32)
500
636
  } else {
501
637
  bool empty = false;
638
+ struct async_cb_dispatcher *ctx = fn->dispatcher;
502
639
 
503
640
  cb.async_event = CreateEvent(NULL, FALSE, FALSE, NULL);
504
641
 
505
- // Now signal the async callback thread
506
- EnterCriticalSection(&async_cb_lock);
507
- empty = async_cb_list == NULL;
508
- cb.next = async_cb_list;
509
- async_cb_list = &cb;
510
- LeaveCriticalSection(&async_cb_lock);
511
-
512
- #if !defined(HAVE_RB_THREAD_BLOCKING_REGION)
513
- // Only signal if the list was empty
514
- if (empty) {
515
- char c;
516
- write(async_cb_pipe[1], &c, 1);
517
- }
518
- #else
519
- SetEvent(async_cb_cond);
520
- #endif
642
+ /* Now signal the async callback dispatcher thread */
643
+ EnterCriticalSection(&ctx->async_cb_lock);
644
+ empty = ctx->async_cb_list == NULL;
645
+ cb.next = ctx->async_cb_list;
646
+ ctx->async_cb_list = &cb;
647
+ LeaveCriticalSection(&ctx->async_cb_lock);
521
648
 
522
- // Wait for the thread executing the ruby callback to signal it is done
649
+ SetEvent(ctx->async_cb_cond);
650
+
651
+ /* Wait for the thread executing the ruby callback to signal it is done */
523
652
  WaitForSingleObject(cb.async_event, INFINITE);
524
653
  CloseHandle(cb.async_event);
525
654
  #endif
@@ -528,164 +657,108 @@ callback_invoke(ffi_cif* cif, void* retval, void** parameters, void* user_data)
528
657
 
529
658
  #if defined(DEFER_ASYNC_CALLBACK)
530
659
  struct async_wait {
660
+ struct async_cb_dispatcher *dispatcher;
531
661
  void* cb;
532
662
  bool stop;
533
663
  };
534
664
 
535
- static VALUE async_cb_wait(void *);
665
+ static void * async_cb_wait(void *);
536
666
  static void async_cb_stop(void *);
537
667
 
538
- #if defined(HAVE_RB_THREAD_BLOCKING_REGION)
539
668
  static VALUE
540
- async_cb_event(void* unused)
669
+ async_cb_event(void* ptr)
541
670
  {
542
- struct async_wait w = { 0 };
671
+ struct async_cb_dispatcher *ctx = (struct async_cb_dispatcher *)ptr;
672
+ struct async_wait w = { ctx };
543
673
 
544
674
  w.stop = false;
545
675
  while (!w.stop) {
546
- rb_thread_blocking_region(async_cb_wait, &w, async_cb_stop, &w);
676
+ rb_thread_call_without_gvl(async_cb_wait, &w, async_cb_stop, &w);
547
677
  if (w.cb != NULL) {
548
- // Start up a new ruby thread to run the ruby callback
549
- rb_thread_create(async_cb_call, w.cb);
550
- }
551
- }
552
-
553
- return Qnil;
554
- }
555
-
556
- #elif defined(_WIN32)
557
- static VALUE
558
- async_cb_event(void* unused)
559
- {
560
- while (true) {
561
- struct gvl_callback* cb;
562
- char buf[64];
563
- fd_set rfds;
564
-
565
- FD_ZERO(&rfds);
566
- FD_SET(async_cb_pipe[0], &rfds);
567
- rb_thread_select(async_cb_pipe[0] + 1, &rfds, NULL, NULL, NULL);
568
- read(async_cb_pipe[0], buf, sizeof(buf));
569
-
570
- EnterCriticalSection(&async_cb_lock);
571
- cb = async_cb_list;
572
- async_cb_list = NULL;
573
- LeaveCriticalSection(&async_cb_lock);
574
-
575
- while (cb != NULL) {
576
- struct gvl_callback* next = cb->next;
577
- // Start up a new ruby thread to run the ruby callback
578
- rb_thread_create(async_cb_call, cb);
579
- cb = next;
580
- }
581
- }
582
-
583
- return Qnil;
584
- }
585
- #else
586
- static VALUE
587
- async_cb_event(void* unused)
588
- {
589
- while (true) {
590
- struct gvl_callback* cb;
591
- char buf[64];
592
-
593
- if (read(async_cb_pipe[0], buf, sizeof(buf)) < 0) {
594
- rb_thread_wait_fd(async_cb_pipe[0]);
595
- while (read(async_cb_pipe[0], buf, sizeof (buf)) < 0) {
596
- if (rb_io_wait_readable(async_cb_pipe[0]) != Qtrue) {
597
- return Qfalse;
598
- }
599
- }
600
- }
601
-
602
- pthread_mutex_lock(&async_cb_mutex);
603
- cb = async_cb_list;
604
- async_cb_list = NULL;
605
- pthread_mutex_unlock(&async_cb_mutex);
606
-
607
- while (cb != NULL) {
608
- struct gvl_callback* next = cb->next;
609
- // Start up a new ruby thread to run the ruby callback
610
- rb_thread_create(async_cb_call, cb);
611
- cb = next;
678
+ /* Start up a new ruby thread to run the ruby callback */
679
+ VALUE new_thread = rb_thread_create(async_cb_call, w.cb);
680
+ /* Name thread, for better debugging */
681
+ rb_funcall(new_thread, rb_intern("name="), 1, rb_str_new2("FFI Callback Runner"));
612
682
  }
613
683
  }
614
684
 
615
685
  return Qnil;
616
686
  }
617
- #endif
618
687
 
619
688
  #ifdef _WIN32
620
- static VALUE
689
+ static void *
621
690
  async_cb_wait(void *data)
622
691
  {
623
692
  struct async_wait* w = (struct async_wait *) data;
693
+ struct async_cb_dispatcher *ctx = w->dispatcher;
624
694
 
625
695
  w->cb = NULL;
626
696
 
627
- EnterCriticalSection(&async_cb_lock);
697
+ EnterCriticalSection(&ctx->async_cb_lock);
628
698
 
629
- while (!w->stop && async_cb_list == NULL) {
630
- LeaveCriticalSection(&async_cb_lock);
631
- WaitForSingleObject(async_cb_cond, INFINITE);
632
- EnterCriticalSection(&async_cb_lock);
699
+ while (!w->stop && ctx->async_cb_list == NULL) {
700
+ LeaveCriticalSection(&ctx->async_cb_lock);
701
+ WaitForSingleObject(ctx->async_cb_cond, INFINITE);
702
+ EnterCriticalSection(&ctx->async_cb_lock);
633
703
  }
634
-
635
- if (async_cb_list != NULL) {
636
- w->cb = async_cb_list;
637
- async_cb_list = async_cb_list->next;
704
+
705
+ if (ctx->async_cb_list != NULL) {
706
+ w->cb = ctx->async_cb_list;
707
+ ctx->async_cb_list = ctx->async_cb_list->next;
638
708
  }
639
709
 
640
- LeaveCriticalSection(&async_cb_lock);
641
-
642
- return Qnil;
710
+ LeaveCriticalSection(&ctx->async_cb_lock);
711
+
712
+ return NULL;
643
713
  }
644
714
 
645
715
  static void
646
716
  async_cb_stop(void *data)
647
717
  {
648
718
  struct async_wait* w = (struct async_wait *) data;
719
+ struct async_cb_dispatcher *ctx = w->dispatcher;
649
720
 
650
- EnterCriticalSection(&async_cb_lock);
721
+ EnterCriticalSection(&ctx->async_cb_lock);
651
722
  w->stop = true;
652
- LeaveCriticalSection(&async_cb_lock);
653
- SetEvent(async_cb_cond);
723
+ LeaveCriticalSection(&ctx->async_cb_lock);
724
+ SetEvent(ctx->async_cb_cond);
654
725
  }
655
726
 
656
727
  #else
657
- static VALUE
728
+ static void *
658
729
  async_cb_wait(void *data)
659
730
  {
660
731
  struct async_wait* w = (struct async_wait *) data;
732
+ struct async_cb_dispatcher *ctx = w->dispatcher;
661
733
 
662
734
  w->cb = NULL;
663
735
 
664
- pthread_mutex_lock(&async_cb_mutex);
736
+ pthread_mutex_lock(&ctx->async_cb_mutex);
665
737
 
666
- while (!w->stop && async_cb_list == NULL) {
667
- pthread_cond_wait(&async_cb_cond, &async_cb_mutex);
738
+ while (!w->stop && ctx->async_cb_list == NULL) {
739
+ pthread_cond_wait(&ctx->async_cb_cond, &ctx->async_cb_mutex);
668
740
  }
669
-
670
- if (async_cb_list != NULL) {
671
- w->cb = async_cb_list;
672
- async_cb_list = async_cb_list->next;
741
+
742
+ if (ctx->async_cb_list != NULL) {
743
+ w->cb = ctx->async_cb_list;
744
+ ctx->async_cb_list = ctx->async_cb_list->next;
673
745
  }
674
746
 
675
- pthread_mutex_unlock(&async_cb_mutex);
676
-
677
- return Qnil;
747
+ pthread_mutex_unlock(&ctx->async_cb_mutex);
748
+
749
+ return NULL;
678
750
  }
679
751
 
680
752
  static void
681
753
  async_cb_stop(void *data)
682
754
  {
683
755
  struct async_wait* w = (struct async_wait *) data;
756
+ struct async_cb_dispatcher *ctx = w->dispatcher;
684
757
 
685
- pthread_mutex_lock(&async_cb_mutex);
758
+ pthread_mutex_lock(&ctx->async_cb_mutex);
686
759
  w->stop = true;
687
- pthread_cond_signal(&async_cb_cond);
688
- pthread_mutex_unlock(&async_cb_mutex);
760
+ pthread_cond_signal(&ctx->async_cb_cond);
761
+ pthread_mutex_unlock(&ctx->async_cb_mutex);
689
762
  }
690
763
  #endif
691
764
 
@@ -694,9 +767,9 @@ async_cb_call(void *data)
694
767
  {
695
768
  struct gvl_callback* cb = (struct gvl_callback *) data;
696
769
 
697
- callback_with_gvl(cb);
770
+ callback_with_gvl(data);
698
771
 
699
- // Signal the original native thread that the ruby code has completed
772
+ /* Signal the original native thread that the ruby code has completed */
700
773
  #ifdef _WIN32
701
774
  SetEvent(cb->async_event);
702
775
  #else
@@ -711,9 +784,15 @@ async_cb_call(void *data)
711
784
 
712
785
  #endif
713
786
 
714
-
715
- static void*
787
+ static void *
716
788
  callback_with_gvl(void* data)
789
+ {
790
+ rb_rescue2(invoke_callback, (VALUE) data, save_callback_exception, (VALUE) data, rb_eException, (VALUE) 0);
791
+ return NULL;
792
+ }
793
+
794
+ static VALUE
795
+ invoke_callback(VALUE data)
717
796
  {
718
797
  struct gvl_callback* cb = (struct gvl_callback *) data;
719
798
 
@@ -734,8 +813,8 @@ callback_with_gvl(void* data)
734
813
  VALUE rbParamType = rb_ary_entry(cbInfo->rbParameterTypes, i);
735
814
 
736
815
  if (unlikely(paramType->nativeType == NATIVE_MAPPED)) {
737
- paramType = ((MappedType *) paramType)->type;
738
816
  rbParamType = ((MappedType *) paramType)->rbType;
817
+ paramType = ((MappedType *) paramType)->type;
739
818
  }
740
819
 
741
820
  switch (paramType->nativeType) {
@@ -776,10 +855,10 @@ callback_with_gvl(void* data)
776
855
  param = rb_float_new(*(double *) parameters[i]);
777
856
  break;
778
857
  case NATIVE_LONGDOUBLE:
779
- param = rbffi_longdouble_new(*(long double *) parameters[i]);
858
+ param = rbffi_longdouble_new(*(long double *) parameters[i]);
780
859
  break;
781
860
  case NATIVE_STRING:
782
- param = (*(void **) parameters[i] != NULL) ? rb_tainted_str_new2(*(char **) parameters[i]) : Qnil;
861
+ param = (*(void **) parameters[i] != NULL) ? rb_str_new2(*(char **) parameters[i]) : Qnil;
783
862
  break;
784
863
  case NATIVE_POINTER:
785
864
  param = rbffi_Pointer_NewInstance(*(void **) parameters[i]);
@@ -789,7 +868,6 @@ callback_with_gvl(void* data)
789
868
  break;
790
869
 
791
870
  case NATIVE_FUNCTION:
792
- case NATIVE_CALLBACK:
793
871
  case NATIVE_STRUCT:
794
872
  param = rbffi_NativeValue_ToRuby(paramType, rbParamType, parameters[i]);
795
873
  break;
@@ -799,7 +877,7 @@ callback_with_gvl(void* data)
799
877
  break;
800
878
  }
801
879
 
802
- // Convert the native value into a custom ruby value
880
+ /* Convert the native value into a custom ruby value */
803
881
  if (unlikely(cbInfo->parameterTypes[i]->nativeType == NATIVE_MAPPED)) {
804
882
  VALUE values[] = { param, Qnil };
805
883
  param = rb_funcall2(((MappedType *) cbInfo->parameterTypes[i])->rbConverter, id_from_native, 2, values);
@@ -809,8 +887,7 @@ callback_with_gvl(void* data)
809
887
  }
810
888
 
811
889
  rbReturnValue = rb_funcall2(fn->rbProc, id_call, cbInfo->parameterCount, rbParams);
812
- RB_GC_GUARD_PTR(rbParams);
813
-
890
+
814
891
  if (unlikely(returnType->nativeType == NATIVE_MAPPED)) {
815
892
  VALUE values[] = { rbReturnValue, Qnil };
816
893
  rbReturnValue = rb_funcall2(((MappedType *) returnType)->rbConverter, id_to_native, 2, values);
@@ -849,11 +926,16 @@ callback_with_gvl(void* data)
849
926
  case NATIVE_FLOAT64:
850
927
  *((double *) retval) = NUM2DBL(rbReturnValue);
851
928
  break;
929
+ case NATIVE_LONGDOUBLE:
930
+ *((long double *) retval) = rbffi_num2longdouble(rbReturnValue);
931
+ break;
852
932
  case NATIVE_POINTER:
853
933
  if (TYPE(rbReturnValue) == T_DATA && rb_obj_is_kind_of(rbReturnValue, rbffi_PointerClass)) {
854
- *((void **) retval) = ((AbstractMemory *) DATA_PTR(rbReturnValue))->address;
934
+ AbstractMemory* memory;
935
+ TypedData_Get_Struct(rbReturnValue, AbstractMemory, &rbffi_abstract_memory_data_type, memory);
936
+ *((void **) retval) = memory->address;
855
937
  } else {
856
- // Default to returning NULL if not a value pointer object. handles nil case as well
938
+ /* Default to returning NULL if not a value pointer object. handles nil case as well */
857
939
  *((void **) retval) = NULL;
858
940
  }
859
941
  break;
@@ -863,17 +945,21 @@ callback_with_gvl(void* data)
863
945
  break;
864
946
 
865
947
  case NATIVE_FUNCTION:
866
- case NATIVE_CALLBACK:
867
948
  if (TYPE(rbReturnValue) == T_DATA && rb_obj_is_kind_of(rbReturnValue, rbffi_PointerClass)) {
949
+ AbstractMemory* memory;
950
+ TypedData_Get_Struct(rbReturnValue, AbstractMemory, &rbffi_abstract_memory_data_type, memory);
868
951
 
869
- *((void **) retval) = ((AbstractMemory *) DATA_PTR(rbReturnValue))->address;
952
+ *((void **) retval) = memory->address;
870
953
 
871
954
  } else if (rb_obj_is_kind_of(rbReturnValue, rb_cProc) || rb_respond_to(rbReturnValue, id_call)) {
872
955
  VALUE function;
956
+ AbstractMemory* memory;
873
957
 
874
958
  function = rbffi_Function_ForProc(rbReturnType, rbReturnValue);
875
959
 
876
- *((void **) retval) = ((AbstractMemory *) DATA_PTR(function))->address;
960
+ TypedData_Get_Struct(function, AbstractMemory, &rbffi_abstract_memory_data_type, memory);
961
+
962
+ *((void **) retval) = memory->address;
877
963
  } else {
878
964
  *((void **) retval) = NULL;
879
965
  }
@@ -881,7 +967,11 @@ callback_with_gvl(void* data)
881
967
 
882
968
  case NATIVE_STRUCT:
883
969
  if (TYPE(rbReturnValue) == T_DATA && rb_obj_is_kind_of(rbReturnValue, rbffi_StructClass)) {
884
- AbstractMemory* memory = ((Struct *) DATA_PTR(rbReturnValue))->pointer;
970
+ Struct* s;
971
+ AbstractMemory* memory;
972
+
973
+ TypedData_Get_Struct(rbReturnValue, Struct, &rbffi_struct_data_type, s);
974
+ memory = s->pointer;
885
975
 
886
976
  if (memory->address != NULL) {
887
977
  memcpy(retval, memory->address, returnType->ffiType->size);
@@ -889,7 +979,7 @@ callback_with_gvl(void* data)
889
979
  } else {
890
980
  memset(retval, 0, returnType->ffiType->size);
891
981
  }
892
-
982
+
893
983
  } else {
894
984
  memset(retval, 0, returnType->ffiType->size);
895
985
  }
@@ -900,9 +990,19 @@ callback_with_gvl(void* data)
900
990
  break;
901
991
  }
902
992
 
903
- return NULL;
993
+ return Qnil;
904
994
  }
905
995
 
996
+ static VALUE
997
+ save_callback_exception(VALUE data, VALUE exc)
998
+ {
999
+ struct gvl_callback* cb = (struct gvl_callback *) data;
1000
+
1001
+ memset(cb->retval, 0, ((Function *) cb->closure->info)->info->returnType->ffiType->size);
1002
+ if (cb->frame != NULL) cb->frame->exc = exc;
1003
+
1004
+ return Qnil;
1005
+ }
906
1006
 
907
1007
  static bool
908
1008
  callback_prep(void* ctx, void* code, Closure* closure, char* errmsg, size_t errmsgsize)
@@ -910,9 +1010,9 @@ callback_prep(void* ctx, void* code, Closure* closure, char* errmsg, size_t errm
910
1010
  FunctionType* fnInfo = (FunctionType *) ctx;
911
1011
  ffi_status ffiStatus;
912
1012
 
913
- ffiStatus = ffi_prep_closure(code, &fnInfo->ffi_cif, callback_invoke, closure);
1013
+ ffiStatus = ffi_prep_closure_loc(closure->pcl, &fnInfo->ffi_cif, callback_invoke, closure, code);
914
1014
  if (ffiStatus != FFI_OK) {
915
- snprintf(errmsg, errmsgsize, "ffi_prep_closure failed. status=%#x", ffiStatus);
1015
+ snprintf(errmsg, errmsgsize, "ffi_prep_closure_loc failed. status=%#x", ffiStatus);
916
1016
  return false;
917
1017
  }
918
1018
 
@@ -927,7 +1027,7 @@ rbffi_Function_Init(VALUE moduleFFI)
927
1027
  * Document-class: FFI::Function < FFI::Pointer
928
1028
  */
929
1029
  rbffi_FunctionClass = rb_define_class_under(moduleFFI, "Function", rbffi_PointerClass);
930
-
1030
+
931
1031
  rb_global_variable(&rbffi_FunctionClass);
932
1032
  rb_define_alloc_func(rbffi_FunctionClass, function_allocate);
933
1033
 
@@ -937,6 +1037,7 @@ rbffi_Function_Init(VALUE moduleFFI)
937
1037
  rb_define_method(rbffi_FunctionClass, "attach", function_attach, 2);
938
1038
  rb_define_method(rbffi_FunctionClass, "free", function_release, 0);
939
1039
  rb_define_method(rbffi_FunctionClass, "autorelease=", function_set_autorelease, 1);
1040
+ rb_define_private_method(rbffi_FunctionClass, "type", function_type, 0);
940
1041
  /*
941
1042
  * call-seq: autorelease
942
1043
  * @return [Boolean]
@@ -956,9 +1057,7 @@ rbffi_Function_Init(VALUE moduleFFI)
956
1057
  id_cb_ref = rb_intern("@__ffi_callback__");
957
1058
  id_to_native = rb_intern("to_native");
958
1059
  id_from_native = rb_intern("from_native");
959
- #if defined(_WIN32)
960
- InitializeCriticalSection(&async_cb_lock);
961
- async_cb_cond = CreateEvent(NULL, FALSE, FALSE, NULL);
1060
+ #if defined(DEFER_ASYNC_CALLBACK) && defined(HAVE_RB_EXT_RACTOR_SAFE)
1061
+ async_cb_dispatcher_key = rb_ractor_local_storage_ptr_newkey(&async_cb_dispatcher_key_type);
962
1062
  #endif
963
1063
  }
964
-