ffi 1.15.5 → 1.16.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (308) hide show
  1. checksums.yaml +4 -4
  2. checksums.yaml.gz.sig +0 -0
  3. data/CHANGELOG.md +54 -0
  4. data/Gemfile +1 -1
  5. data/README.md +3 -2
  6. data/Rakefile +10 -7
  7. data/ext/ffi_c/AbstractMemory.c +56 -34
  8. data/ext/ffi_c/AbstractMemory.h +3 -2
  9. data/ext/ffi_c/ArrayType.c +49 -13
  10. data/ext/ffi_c/ArrayType.h +1 -0
  11. data/ext/ffi_c/Buffer.c +86 -29
  12. data/ext/ffi_c/Call.c +13 -5
  13. data/ext/ffi_c/DynamicLibrary.c +89 -26
  14. data/ext/ffi_c/Function.c +238 -98
  15. data/ext/ffi_c/Function.h +1 -0
  16. data/ext/ffi_c/FunctionInfo.c +78 -24
  17. data/ext/ffi_c/LastError.c +25 -7
  18. data/ext/ffi_c/MappedType.c +66 -23
  19. data/ext/ffi_c/MappedType.h +0 -2
  20. data/ext/ffi_c/MemoryPointer.c +34 -7
  21. data/ext/ffi_c/MethodHandle.c +3 -1
  22. data/ext/ffi_c/Pointer.c +68 -29
  23. data/ext/ffi_c/Pointer.h +1 -0
  24. data/ext/ffi_c/Struct.c +151 -80
  25. data/ext/ffi_c/Struct.h +7 -4
  26. data/ext/ffi_c/StructByValue.c +48 -16
  27. data/ext/ffi_c/StructLayout.c +117 -48
  28. data/ext/ffi_c/Type.c +104 -36
  29. data/ext/ffi_c/Type.h +3 -1
  30. data/ext/ffi_c/Types.c +1 -1
  31. data/ext/ffi_c/Variadic.c +65 -23
  32. data/ext/ffi_c/compat.h +22 -22
  33. data/ext/ffi_c/extconf.rb +19 -2
  34. data/ext/ffi_c/ffi.c +4 -0
  35. data/ext/ffi_c/libffi/.appveyor/site.exp +16 -0
  36. data/ext/ffi_c/libffi/.appveyor/unix-noexec.exp +7 -0
  37. data/ext/ffi_c/libffi/.appveyor.yml +27 -9
  38. data/ext/ffi_c/libffi/.ci/build-cross-in-container.sh +18 -0
  39. data/ext/ffi_c/libffi/{.travis → .ci}/build-in-container.sh +4 -6
  40. data/ext/ffi_c/libffi/.ci/build.sh +124 -0
  41. data/ext/ffi_c/libffi/{.travis → .ci}/install.sh +14 -7
  42. data/ext/ffi_c/libffi/.ci/msvs-detect +1103 -0
  43. data/ext/ffi_c/libffi/{.travis → .ci}/site.exp +5 -3
  44. data/ext/ffi_c/libffi/.circleci/config.yml +156 -0
  45. data/ext/ffi_c/libffi/.github/workflows/build.yml +460 -0
  46. data/ext/ffi_c/libffi/.github/workflows/emscripten.yml +171 -0
  47. data/ext/ffi_c/libffi/.gitignore +10 -2
  48. data/ext/ffi_c/libffi/LICENSE +1 -1
  49. data/ext/ffi_c/libffi/Makefile.am +12 -5
  50. data/ext/ffi_c/libffi/Makefile.in +118 -51
  51. data/ext/ffi_c/libffi/README.md +144 -104
  52. data/ext/ffi_c/libffi/acinclude.m4 +10 -112
  53. data/ext/ffi_c/libffi/config.guess +623 -556
  54. data/ext/ffi_c/libffi/config.sub +75 -34
  55. data/ext/ffi_c/libffi/configure +4571 -3830
  56. data/ext/ffi_c/libffi/configure.ac +64 -28
  57. data/ext/ffi_c/libffi/configure.host +25 -6
  58. data/ext/ffi_c/libffi/doc/Makefile.in +9 -6
  59. data/ext/ffi_c/libffi/doc/libffi.texi +82 -33
  60. data/ext/ffi_c/libffi/doc/version.texi +4 -4
  61. data/ext/ffi_c/libffi/fficonfig.h.in +12 -47
  62. data/ext/ffi_c/libffi/generate-darwin-source-and-headers.py +163 -52
  63. data/ext/ffi_c/libffi/include/Makefile.am +1 -1
  64. data/ext/ffi_c/libffi/include/Makefile.in +10 -9
  65. data/ext/ffi_c/libffi/include/ffi.h.in +54 -50
  66. data/ext/ffi_c/libffi/include/ffi_cfi.h +21 -0
  67. data/ext/ffi_c/libffi/include/ffi_common.h +29 -0
  68. data/ext/ffi_c/libffi/include/tramp.h +45 -0
  69. data/ext/ffi_c/libffi/install-sh +92 -69
  70. data/ext/ffi_c/libffi/libtool-version +2 -2
  71. data/ext/ffi_c/libffi/ltmain.sh +512 -315
  72. data/ext/ffi_c/libffi/m4/ax_cc_maxopt.m4 +18 -14
  73. data/ext/ffi_c/libffi/m4/ax_cflags_warn_all.m4 +108 -72
  74. data/ext/ffi_c/libffi/m4/ax_compiler_vendor.m4 +76 -45
  75. data/ext/ffi_c/libffi/m4/ax_prepend_flag.m4 +51 -0
  76. data/ext/ffi_c/libffi/man/Makefile.in +9 -6
  77. data/ext/ffi_c/libffi/missing +1 -1
  78. data/ext/ffi_c/libffi/msvc_build/aarch64/aarch64_include/ffi.h +1 -1
  79. data/ext/ffi_c/libffi/msvcc.sh +1 -1
  80. data/ext/ffi_c/libffi/src/aarch64/ffi.c +146 -42
  81. data/ext/ffi_c/libffi/src/aarch64/internal.h +32 -0
  82. data/ext/ffi_c/libffi/src/aarch64/sysv.S +134 -42
  83. data/ext/ffi_c/libffi/src/arc/arcompact.S +169 -94
  84. data/ext/ffi_c/libffi/src/arc/ffi.c +325 -148
  85. data/ext/ffi_c/libffi/src/arc/ffitarget.h +14 -0
  86. data/ext/ffi_c/libffi/src/arm/ffi.c +62 -17
  87. data/ext/ffi_c/libffi/src/arm/ffitarget.h +3 -3
  88. data/ext/ffi_c/libffi/src/arm/internal.h +10 -0
  89. data/ext/ffi_c/libffi/src/arm/sysv.S +113 -42
  90. data/ext/ffi_c/libffi/src/closures.c +133 -47
  91. data/ext/ffi_c/libffi/src/dlmalloc.c +5 -0
  92. data/ext/ffi_c/libffi/src/ia64/ffi.c +12 -0
  93. data/ext/ffi_c/libffi/src/ia64/unix.S +20 -2
  94. data/ext/ffi_c/libffi/src/loongarch64/ffi.c +621 -0
  95. data/ext/ffi_c/libffi/src/loongarch64/ffitarget.h +82 -0
  96. data/ext/ffi_c/libffi/src/loongarch64/sysv.S +327 -0
  97. data/ext/ffi_c/libffi/src/m32r/ffi.c +31 -14
  98. data/ext/ffi_c/libffi/src/mips/ffi.c +240 -65
  99. data/ext/ffi_c/libffi/src/mips/ffitarget.h +7 -0
  100. data/ext/ffi_c/libffi/src/mips/n32.S +137 -28
  101. data/ext/ffi_c/libffi/src/mips/o32.S +61 -4
  102. data/ext/ffi_c/libffi/src/moxie/ffi.c +47 -22
  103. data/ext/ffi_c/libffi/src/or1k/ffi.c +25 -12
  104. data/ext/ffi_c/libffi/src/pa/ffi.c +32 -33
  105. data/ext/ffi_c/libffi/src/pa/ffi64.c +614 -0
  106. data/ext/ffi_c/libffi/src/pa/ffitarget.h +22 -8
  107. data/ext/ffi_c/libffi/src/pa/hpux32.S +83 -36
  108. data/ext/ffi_c/libffi/src/pa/hpux64.S +681 -0
  109. data/ext/ffi_c/libffi/src/pa/linux.S +82 -35
  110. data/ext/ffi_c/libffi/src/powerpc/ffi_linux64.c +1 -1
  111. data/ext/ffi_c/libffi/src/powerpc/linux64.S +2 -0
  112. data/ext/ffi_c/libffi/src/powerpc/linux64_closure.S +2 -0
  113. data/ext/ffi_c/libffi/src/powerpc/t-aix +5 -0
  114. data/ext/ffi_c/libffi/src/prep_cif.c +22 -2
  115. data/ext/ffi_c/libffi/src/riscv/ffi.c +37 -4
  116. data/ext/ffi_c/libffi/src/sparc/ffi64.c +16 -0
  117. data/ext/ffi_c/libffi/src/tramp.c +716 -0
  118. data/ext/ffi_c/libffi/src/wasm32/ffi.c +934 -0
  119. data/ext/ffi_c/libffi/src/wasm32/ffitarget.h +62 -0
  120. data/ext/ffi_c/libffi/src/x86/ffi.c +99 -37
  121. data/ext/ffi_c/libffi/src/x86/ffi64.c +67 -12
  122. data/ext/ffi_c/libffi/src/x86/ffitarget.h +9 -5
  123. data/ext/ffi_c/libffi/src/x86/ffiw64.c +43 -0
  124. data/ext/ffi_c/libffi/src/x86/internal.h +14 -0
  125. data/ext/ffi_c/libffi/src/x86/internal64.h +14 -0
  126. data/ext/ffi_c/libffi/src/x86/sysv.S +172 -38
  127. data/ext/ffi_c/libffi/src/x86/sysv_intel.S +91 -88
  128. data/ext/ffi_c/libffi/src/x86/unix64.S +96 -6
  129. data/ext/ffi_c/libffi/src/x86/win64.S +20 -7
  130. data/ext/ffi_c/libffi/src/xtensa/ffi.c +16 -8
  131. data/ext/ffi_c/libffi/src/xtensa/ffitarget.h +4 -0
  132. data/ext/ffi_c/libffi/src/xtensa/sysv.S +26 -16
  133. data/ext/ffi_c/libffi/testsuite/Makefile.am +127 -114
  134. data/ext/ffi_c/libffi/testsuite/Makefile.in +137 -121
  135. data/ext/ffi_c/libffi/testsuite/emscripten/build-tests.sh +54 -0
  136. data/ext/ffi_c/libffi/testsuite/emscripten/build.sh +63 -0
  137. data/ext/ffi_c/libffi/testsuite/emscripten/conftest.py +80 -0
  138. data/ext/ffi_c/libffi/testsuite/emscripten/node-tests.sh +48 -0
  139. data/ext/ffi_c/libffi/testsuite/emscripten/test.html +7 -0
  140. data/ext/ffi_c/libffi/testsuite/emscripten/test_libffi.py +51 -0
  141. data/ext/ffi_c/libffi/testsuite/lib/libffi.exp +43 -21
  142. data/ext/ffi_c/libffi/testsuite/lib/target-libpath.exp +2 -2
  143. data/ext/ffi_c/libffi/testsuite/libffi.bhaible/test-call.c +1 -0
  144. data/ext/ffi_c/libffi/testsuite/libffi.bhaible/testcases.c +5 -1
  145. data/ext/ffi_c/libffi/testsuite/libffi.call/bpo_38748.c +41 -0
  146. data/ext/ffi_c/libffi/testsuite/libffi.call/ffitest.h +28 -3
  147. data/ext/ffi_c/libffi/testsuite/libffi.call/float_va.c +3 -0
  148. data/ext/ffi_c/libffi/testsuite/libffi.call/return_ll1.c +5 -0
  149. data/ext/ffi_c/libffi/testsuite/libffi.call/return_sl.c +4 -0
  150. data/ext/ffi_c/libffi/testsuite/libffi.call/return_ul.c +4 -0
  151. data/ext/ffi_c/libffi/testsuite/libffi.call/s55.c +60 -0
  152. data/ext/ffi_c/libffi/testsuite/libffi.call/strlen.c +8 -9
  153. data/ext/ffi_c/libffi/testsuite/libffi.call/strlen2.c +2 -2
  154. data/ext/ffi_c/libffi/testsuite/libffi.call/strlen3.c +2 -2
  155. data/ext/ffi_c/libffi/testsuite/libffi.call/strlen4.c +2 -2
  156. data/ext/ffi_c/libffi/testsuite/libffi.call/struct10.c +1 -1
  157. data/ext/ffi_c/libffi/testsuite/libffi.call/struct_by_value_2.c +63 -0
  158. data/ext/ffi_c/libffi/testsuite/libffi.call/struct_by_value_3.c +65 -0
  159. data/ext/ffi_c/libffi/testsuite/libffi.call/struct_by_value_3f.c +65 -0
  160. data/ext/ffi_c/libffi/testsuite/libffi.call/struct_by_value_4.c +67 -0
  161. data/ext/ffi_c/libffi/testsuite/libffi.call/struct_by_value_4f.c +67 -0
  162. data/ext/ffi_c/libffi/testsuite/libffi.call/struct_by_value_big.c +93 -0
  163. data/ext/ffi_c/libffi/testsuite/libffi.call/struct_by_value_small.c +61 -0
  164. data/ext/ffi_c/libffi/testsuite/libffi.call/struct_return_2H.c +63 -0
  165. data/ext/ffi_c/libffi/testsuite/libffi.call/struct_return_8H.c +90 -0
  166. data/ext/ffi_c/libffi/testsuite/libffi.call/va_1.c +1 -119
  167. data/ext/ffi_c/libffi/testsuite/libffi.call/va_2.c +220 -0
  168. data/ext/ffi_c/libffi/testsuite/libffi.call/va_3.c +154 -0
  169. data/ext/ffi_c/libffi/testsuite/libffi.call/va_struct1.c +13 -0
  170. data/ext/ffi_c/libffi/testsuite/libffi.call/va_struct2.c +11 -0
  171. data/ext/ffi_c/libffi/testsuite/libffi.call/va_struct3.c +15 -0
  172. data/ext/ffi_c/libffi/testsuite/libffi.closures/closure_fn0.c +3 -2
  173. data/ext/ffi_c/libffi/testsuite/libffi.closures/closure_fn1.c +2 -0
  174. data/ext/ffi_c/libffi/testsuite/libffi.closures/closure_fn2.c +2 -0
  175. data/ext/ffi_c/libffi/testsuite/libffi.closures/closure_fn3.c +21 -1
  176. data/ext/ffi_c/libffi/testsuite/libffi.closures/closure_fn4.c +2 -0
  177. data/ext/ffi_c/libffi/testsuite/libffi.closures/closure_fn5.c +2 -0
  178. data/ext/ffi_c/libffi/testsuite/libffi.closures/closure_fn6.c +2 -0
  179. data/ext/ffi_c/libffi/testsuite/libffi.closures/closure_loc_fn0.c +7 -6
  180. data/ext/ffi_c/libffi/testsuite/libffi.closures/closure_simple.c +6 -0
  181. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_12byte.c +18 -0
  182. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_16byte.c +22 -0
  183. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_18byte.c +24 -0
  184. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_19byte.c +29 -0
  185. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_1_1byte.c +4 -0
  186. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_20byte.c +19 -1
  187. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_20byte1.c +21 -1
  188. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_24byte.c +35 -3
  189. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_2byte.c +13 -0
  190. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_3_1byte.c +19 -0
  191. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_3byte1.c +13 -0
  192. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_3byte2.c +13 -0
  193. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_3float.c +18 -0
  194. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_4_1byte.c +22 -0
  195. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_4byte.c +13 -0
  196. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_5_1_byte.c +29 -1
  197. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_5byte.c +19 -1
  198. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_64byte.c +24 -0
  199. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_6_1_byte.c +28 -1
  200. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_6byte.c +24 -2
  201. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_7_1_byte.c +39 -1
  202. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_7byte.c +25 -1
  203. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_8byte.c +14 -0
  204. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_9byte1.c +14 -1
  205. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_9byte2.c +14 -2
  206. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_align_double.c +19 -1
  207. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_align_float.c +19 -1
  208. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_align_longdouble.c +20 -1
  209. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_align_longdouble_split.c +40 -25
  210. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_align_longdouble_split2.c +40 -3
  211. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_align_pointer.c +19 -1
  212. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_align_sint16.c +18 -1
  213. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_align_sint32.c +18 -3
  214. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_align_sint64.c +18 -1
  215. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_align_uint16.c +18 -1
  216. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_align_uint32.c +19 -1
  217. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_align_uint64.c +19 -1
  218. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_dbls_struct.c +3 -1
  219. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_double_va.c +9 -1
  220. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_float.c +4 -0
  221. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_longdouble.c +11 -1
  222. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_longdouble_va.c +22 -3
  223. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_multi_schar.c +5 -1
  224. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_multi_sshort.c +6 -1
  225. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_multi_sshortchar.c +9 -1
  226. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_multi_uchar.c +9 -1
  227. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_multi_ushort.c +6 -2
  228. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_multi_ushortchar.c +9 -1
  229. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_pointer.c +5 -0
  230. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_pointer_stack.c +10 -0
  231. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_schar.c +3 -0
  232. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_sint.c +2 -0
  233. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_sshort.c +3 -0
  234. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_struct_va1.c +11 -0
  235. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_uchar.c +3 -0
  236. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_uint.c +4 -0
  237. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_uint_va.c +4 -0
  238. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_ulong_va.c +4 -0
  239. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_ulonglong.c +2 -0
  240. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_ushort.c +3 -0
  241. data/ext/ffi_c/libffi/testsuite/libffi.closures/ffitest.h +1 -138
  242. data/ext/ffi_c/libffi/testsuite/libffi.closures/huge_struct.c +21 -21
  243. data/ext/ffi_c/libffi/testsuite/libffi.closures/nested_struct.c +32 -9
  244. data/ext/ffi_c/libffi/testsuite/libffi.closures/nested_struct1.c +1 -1
  245. data/ext/ffi_c/libffi/testsuite/libffi.closures/nested_struct10.c +12 -0
  246. data/ext/ffi_c/libffi/testsuite/libffi.closures/nested_struct11.c +21 -5
  247. data/ext/ffi_c/libffi/testsuite/libffi.closures/nested_struct12.c +86 -0
  248. data/ext/ffi_c/libffi/testsuite/libffi.closures/nested_struct13.c +115 -0
  249. data/ext/ffi_c/libffi/testsuite/libffi.closures/nested_struct2.c +10 -1
  250. data/ext/ffi_c/libffi/testsuite/libffi.closures/nested_struct3.c +10 -1
  251. data/ext/ffi_c/libffi/testsuite/libffi.closures/nested_struct4.c +9 -0
  252. data/ext/ffi_c/libffi/testsuite/libffi.closures/nested_struct5.c +9 -0
  253. data/ext/ffi_c/libffi/testsuite/libffi.closures/nested_struct6.c +11 -0
  254. data/ext/ffi_c/libffi/testsuite/libffi.closures/nested_struct7.c +9 -0
  255. data/ext/ffi_c/libffi/testsuite/libffi.closures/nested_struct8.c +11 -0
  256. data/ext/ffi_c/libffi/testsuite/libffi.closures/nested_struct9.c +11 -0
  257. data/ext/ffi_c/libffi/testsuite/libffi.closures/single_entry_structs1.c +86 -0
  258. data/ext/ffi_c/libffi/testsuite/libffi.closures/single_entry_structs2.c +102 -0
  259. data/ext/ffi_c/libffi/testsuite/libffi.closures/single_entry_structs3.c +101 -0
  260. data/ext/ffi_c/libffi/testsuite/libffi.closures/stret_medium.c +1 -1
  261. data/ext/ffi_c/libffi/testsuite/libffi.closures/stret_medium2.c +1 -1
  262. data/ext/ffi_c/libffi/testsuite/libffi.closures/testclosure.c +6 -1
  263. data/ext/ffi_c/libffi/testsuite/libffi.closures/unwindtest.cc +2 -1
  264. data/ext/ffi_c/libffi/testsuite/libffi.closures/unwindtest_ffi_call.cc +1 -0
  265. data/ext/ffi_c/libffi.bsd.mk +2 -2
  266. data/ext/ffi_c/libffi.gnu.mk +2 -2
  267. data/ext/ffi_c/rbffi.h +1 -1
  268. data/ffi.gemspec +2 -2
  269. data/lib/ffi/autopointer.rb +7 -22
  270. data/lib/ffi/compat.rb +43 -0
  271. data/lib/ffi/data_converter.rb +2 -2
  272. data/lib/ffi/dynamic_library.rb +89 -0
  273. data/lib/ffi/enum.rb +18 -11
  274. data/lib/ffi/ffi.rb +3 -0
  275. data/lib/ffi/function.rb +71 -0
  276. data/lib/ffi/library.rb +55 -71
  277. data/lib/ffi/library_path.rb +109 -0
  278. data/lib/ffi/managedstruct.rb +1 -1
  279. data/lib/ffi/platform/aarch64-windows/types.conf +52 -0
  280. data/lib/ffi/platform/hppa1.1-linux/types.conf +178 -0
  281. data/lib/ffi/platform/hppa2.0-linux/types.conf +178 -0
  282. data/lib/ffi/platform/loongarch64-linux/types.conf +141 -0
  283. data/lib/ffi/platform/sw_64-linux/types.conf +141 -0
  284. data/lib/ffi/platform.rb +15 -13
  285. data/lib/ffi/struct.rb +2 -1
  286. data/lib/ffi/struct_layout.rb +1 -1
  287. data/lib/ffi/struct_layout_builder.rb +1 -1
  288. data/lib/ffi/types.rb +30 -5
  289. data/lib/ffi/variadic.rb +19 -8
  290. data/lib/ffi/version.rb +1 -1
  291. data.tar.gz.sig +0 -0
  292. metadata +97 -25
  293. metadata.gz.sig +1 -0
  294. data/ext/ffi_c/libffi/.travis/build-cross-in-container.sh +0 -14
  295. data/ext/ffi_c/libffi/.travis/build.sh +0 -142
  296. data/ext/ffi_c/libffi/.travis.yml +0 -83
  297. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_uchar_va.c +0 -44
  298. data/ext/ffi_c/libffi/testsuite/libffi.closures/cls_ushort_va.c +0 -44
  299. /data/ext/ffi_c/libffi/{.travis → .ci}/ar-lib +0 -0
  300. /data/ext/ffi_c/libffi/{.travis → .ci}/bfin-sim.exp +0 -0
  301. /data/ext/ffi_c/libffi/{.travis → .ci}/compile +0 -0
  302. /data/ext/ffi_c/libffi/{.travis → .ci}/m32r-sim.exp +0 -0
  303. /data/ext/ffi_c/libffi/{.travis → .ci}/moxie-sim.exp +0 -0
  304. /data/ext/ffi_c/libffi/{.travis → .ci}/or1k-sim.exp +0 -0
  305. /data/ext/ffi_c/libffi/{.travis → .ci}/powerpc-eabisim.exp +0 -0
  306. /data/ext/ffi_c/libffi/{.travis → .ci}/wine-sim.exp +0 -0
  307. /data/ext/ffi_c/libffi/testsuite/libffi.call/{pyobjc-tc.c → pyobjc_tc.c} +0 -0
  308. /data/lib/ffi/platform/{sparc64-linux → sparcv9-linux}/types.conf +0 -0
@@ -22,7 +22,7 @@ typedef struct struct_72byte {
22
22
  long long i;
23
23
  } struct_72byte;
24
24
 
25
- struct_72byte cls_struct_72byte_fn(
25
+ static struct_72byte cls_struct_72byte_fn(
26
26
  struct_72byte b0,
27
27
  struct_72byte b1,
28
28
  struct_72byte b2,
@@ -14,12 +14,17 @@ typedef struct cls_struct_combined {
14
14
  float d;
15
15
  } cls_struct_combined;
16
16
 
17
- void cls_struct_combined_fn(struct cls_struct_combined arg)
17
+ static void cls_struct_combined_fn(struct cls_struct_combined arg)
18
18
  {
19
19
  printf("%g %g %g %g\n",
20
20
  arg.a, arg.b,
21
21
  arg.c, arg.d);
22
22
  fflush(stdout);
23
+
24
+ CHECK_FLOAT_EQ(arg.a, 4);
25
+ CHECK_FLOAT_EQ(arg.b, 5);
26
+ CHECK_FLOAT_EQ(arg.c, 1);
27
+ CHECK_FLOAT_EQ(arg.d, 8);
23
28
  }
24
29
 
25
30
  static void
@@ -4,7 +4,7 @@
4
4
  PR: none.
5
5
  Originator: Jeff Sturm <jsturm@one-point.com> */
6
6
 
7
- /* { dg-do run { xfail x86_64-apple-darwin* moxie*-*-* } } */
7
+ /* { dg-do run { xfail moxie*-*-* } } */
8
8
 
9
9
  #include "ffitest.h"
10
10
 
@@ -48,6 +48,7 @@ typedef int (*closure_test_type1)(float, float, float, float, signed short,
48
48
  float, float, int, double, int, int, float,
49
49
  int, int, int, int);
50
50
 
51
+ extern "C"
51
52
  int main (void)
52
53
  {
53
54
  ffi_cif cif;
@@ -14,6 +14,7 @@ static int checking(int a __UNUSED__, short b __UNUSED__,
14
14
  throw 9;
15
15
  }
16
16
 
17
+ extern "C"
17
18
  int main (void)
18
19
  {
19
20
  ffi_cif cif;
@@ -18,8 +18,8 @@ INCFLAGS := -I${LIBFFI_BUILD_DIR}/include -I${INCFLAGS}
18
18
 
19
19
  LIBFFI = ${LIBFFI_BUILD_DIR}/.libs/libffi_convenience.a
20
20
  LIBFFI_AUTOGEN = ${LIBFFI_SRC_DIR}/autogen.sh
21
- LIBFFI_CONFIGURE = ${LIBFFI_SRC_DIR}/configure --disable-static \
22
- --with-pic=yes --disable-dependency-tracking --disable-docs
21
+ LIBFFI_CONFIGURE = ${LIBFFI_SRC_DIR}/configure --disable-shared --enable-static \
22
+ --with-pic=yes --disable-dependency-tracking --disable-docs $(LIBFFI_DEBUG)
23
23
 
24
24
  $(OBJS): ${LIBFFI}
25
25
 
@@ -21,8 +21,8 @@ endif
21
21
 
22
22
  LIBFFI = "$(LIBFFI_BUILD_DIR)"/.libs/libffi_convenience.a
23
23
  LIBFFI_AUTOGEN = ${LIBFFI_SRC_DIR}/autogen.sh
24
- LIBFFI_CONFIGURE = "$(LIBFFI_SRC_DIR)"/configure --disable-static \
25
- --with-pic=yes --disable-dependency-tracking --disable-docs
24
+ LIBFFI_CONFIGURE = "$(LIBFFI_SRC_DIR)"/configure --disable-shared --enable-static \
25
+ --with-pic=yes --disable-dependency-tracking --disable-docs $(LIBFFI_DEBUG)
26
26
 
27
27
  $(OBJS): $(LIBFFI)
28
28
 
data/ext/ffi_c/rbffi.h CHANGED
@@ -39,7 +39,7 @@ extern "C" {
39
39
  #define MAX_PARAMETERS (32)
40
40
 
41
41
  extern VALUE rbffi_FFIModule;
42
-
42
+
43
43
  extern void rbffi_Type_Init(VALUE ffiModule);
44
44
  extern void rbffi_Buffer_Init(VALUE ffiModule);
45
45
  extern void rbffi_Invoker_Init(VALUE ffiModule);
data/ffi.gemspec CHANGED
@@ -34,9 +34,9 @@ Gem::Specification.new do |s|
34
34
  s.rdoc_options = %w[--exclude=ext/ffi_c/.*\.o$ --exclude=ffi_c\.(bundle|so)$]
35
35
  s.license = 'BSD-3-Clause'
36
36
  s.require_paths << 'ext/ffi_c'
37
- s.required_ruby_version = '>= 2.3'
37
+ s.required_ruby_version = '>= 2.5'
38
38
  s.add_development_dependency 'rake', '~> 13.0'
39
- s.add_development_dependency 'rake-compiler', '~> 1.0'
39
+ s.add_development_dependency 'rake-compiler', '~> 1.1'
40
40
  s.add_development_dependency 'rake-compiler-dock', '~> 1.0'
41
41
  s.add_development_dependency 'rspec', '~> 2.14.1'
42
42
  end
@@ -76,21 +76,21 @@ module FFI
76
76
  # going to be useful if you subclass {AutoPointer}, and override
77
77
  # #release, which by default does nothing.
78
78
  def initialize(ptr, proc=nil, &block)
79
+ raise TypeError, "Invalid pointer" if ptr.nil? || !ptr.kind_of?(Pointer) ||
80
+ ptr.kind_of?(MemoryPointer) || ptr.kind_of?(AutoPointer)
79
81
  super(ptr.type_size, ptr)
80
- raise TypeError, "Invalid pointer" if ptr.nil? || !ptr.kind_of?(Pointer) \
81
- || ptr.kind_of?(MemoryPointer) || ptr.kind_of?(AutoPointer)
82
82
 
83
83
  @releaser = if proc
84
84
  if not proc.respond_to?(:call)
85
85
  raise RuntimeError.new("proc must be callable")
86
86
  end
87
- CallableReleaser.new(ptr, proc)
87
+ Releaser.new(ptr, proc)
88
88
 
89
89
  else
90
- if not self.class.respond_to?(:release)
90
+ if not self.class.respond_to?(:release, true)
91
91
  raise RuntimeError.new("no release method defined")
92
92
  end
93
- DefaultReleaser.new(ptr, self.class)
93
+ Releaser.new(ptr, self.class.method(:release))
94
94
  end
95
95
 
96
96
  ObjectSpace.define_finalizer(self, @releaser)
@@ -107,6 +107,7 @@ module FFI
107
107
  # @return [Boolean] +autorelease+
108
108
  # Set +autorelease+ property. See {Pointer Autorelease section at Pointer}.
109
109
  def autorelease=(autorelease)
110
+ raise FrozenError.new("can't modify frozen #{self.class}") if frozen?
110
111
  @releaser.autorelease=(autorelease)
111
112
  end
112
113
 
@@ -149,23 +150,7 @@ module FFI
149
150
  def call(*args)
150
151
  release(@ptr) if @autorelease && @ptr
151
152
  end
152
- end
153
-
154
- # DefaultReleaser is a {Releaser} used when an {AutoPointer} is defined
155
- # without Proc or Method. In this case, the pointer to release must be of
156
- # a class derived from AutoPointer with a {release} class method.
157
- class DefaultReleaser < Releaser
158
- # @param [Pointer] ptr
159
- # @return [nil]
160
- # Release +ptr+ using the {release} class method of its class.
161
- def release(ptr)
162
- @proc.release(ptr)
163
- end
164
- end
165
153
 
166
- # CallableReleaser is a {Releaser} used when an {AutoPointer} is defined with a
167
- # Proc or a Method.
168
- class CallableReleaser < Releaser
169
154
  # Release +ptr+ by using Proc or Method defined at +ptr+
170
155
  # {AutoPointer#initialize initialization}.
171
156
  #
@@ -182,7 +167,7 @@ module FFI
182
167
  # @return [Type::POINTER]
183
168
  # @raise {RuntimeError} if class does not implement a +#release+ method
184
169
  def self.native_type
185
- if not self.respond_to?(:release)
170
+ if not self.respond_to?(:release, true)
186
171
  raise RuntimeError.new("no release method defined for #{self.inspect}")
187
172
  end
188
173
  Type::POINTER
data/lib/ffi/compat.rb ADDED
@@ -0,0 +1,43 @@
1
+ #
2
+ # Copyright (C) 2023-2023 Lars Kanis
3
+ #
4
+ # This file is part of ruby-ffi.
5
+ #
6
+ # All rights reserved.
7
+ #
8
+ # Redistribution and use in source and binary forms, with or without
9
+ # modification, are permitted provided that the following conditions are met:
10
+ #
11
+ # * Redistributions of source code must retain the above copyright notice, this
12
+ # list of conditions and the following disclaimer.
13
+ # * Redistributions in binary form must reproduce the above copyright notice
14
+ # this list of conditions and the following disclaimer in the documentation
15
+ # and/or other materials provided with the distribution.
16
+ # * Neither the name of the Ruby FFI project nor the names of its contributors
17
+ # may be used to endorse or promote products derived from this software
18
+ # without specific prior written permission.
19
+ #
20
+ # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
21
+ # AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
22
+ # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
23
+ # DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE
24
+ # FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
25
+ # DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
26
+ # SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
27
+ # CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
28
+ # OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
29
+ # OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30
+ #
31
+
32
+ module FFI
33
+ if defined?(Ractor.make_shareable)
34
+ # This is for FFI internal use only.
35
+ def self.make_shareable(obj)
36
+ Ractor.make_shareable(obj)
37
+ end
38
+ else
39
+ def self.make_shareable(obj)
40
+ obj.freeze
41
+ end
42
+ end
43
+ end
@@ -31,7 +31,7 @@
31
31
  module FFI
32
32
  # This module is used to extend somes classes and give then a common API.
33
33
  #
34
- # Most of methods defined here must be overriden.
34
+ # Most of methods defined here must be overridden.
35
35
  module DataConverter
36
36
  # Get native type.
37
37
  #
@@ -41,7 +41,7 @@ module FFI
41
41
  # Get native type from +type+.
42
42
  #
43
43
  # @overload native_type
44
- # @raise {NotImplementedError} This method must be overriden.
44
+ # @raise {NotImplementedError} This method must be overridden.
45
45
  def native_type(type = nil)
46
46
  if type
47
47
  @native_type = FFI.find_type(type)
@@ -0,0 +1,89 @@
1
+ #
2
+ # Copyright (C) 2008-2010 Wayne Meissner
3
+ #
4
+ # This file is part of ruby-ffi.
5
+ #
6
+ # All rights reserved.
7
+ #
8
+ # Redistribution and use in source and binary forms, with or without
9
+ # modification, are permitted provided that the following conditions are met:
10
+ #
11
+ # * Redistributions of source code must retain the above copyright notice, this
12
+ # list of conditions and the following disclaimer.
13
+ # * Redistributions in binary form must reproduce the above copyright notice
14
+ # this list of conditions and the following disclaimer in the documentation
15
+ # and/or other materials provided with the distribution.
16
+ # * Neither the name of the Ruby FFI project nor the names of its contributors
17
+ # may be used to endorse or promote products derived from this software
18
+ # without specific prior written permission.
19
+ #
20
+ # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
21
+ # AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
22
+ # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
23
+ # DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE
24
+ # FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
25
+ # DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
26
+ # SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
27
+ # CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
28
+ # OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
29
+ # OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.#
30
+
31
+ module FFI
32
+ class DynamicLibrary
33
+ SEARCH_PATH = %w[/usr/lib /usr/local/lib /opt/local/lib]
34
+ if FFI::Platform::ARCH == 'aarch64' && FFI::Platform.mac?
35
+ SEARCH_PATH << '/opt/homebrew/lib'
36
+ end
37
+
38
+ SEARCH_PATH_MESSAGE = "Searched in <system library path>, #{SEARCH_PATH.join(', ')}".freeze
39
+
40
+ def self.load_library(name, flags)
41
+ if name == FFI::CURRENT_PROCESS
42
+ FFI::DynamicLibrary.open(nil, RTLD_LAZY | RTLD_LOCAL)
43
+ else
44
+ flags ||= RTLD_LAZY | RTLD_LOCAL
45
+
46
+ libnames = (name.is_a?(::Array) ? name : [name])
47
+ libnames = libnames.map(&:to_s).map { |n| [n, FFI.map_library_name(n)].uniq }.flatten.compact
48
+ errors = []
49
+
50
+ libnames.each do |libname|
51
+ lib = try_load(libname, flags, errors)
52
+ return lib if lib
53
+
54
+ unless libname.start_with?("/") || FFI::Platform.windows?
55
+ SEARCH_PATH.each do |prefix|
56
+ path = "#{prefix}/#{libname}"
57
+ if File.exist?(path)
58
+ lib = try_load(path, flags, errors)
59
+ return lib if lib
60
+ end
61
+ end
62
+ end
63
+ end
64
+
65
+ raise LoadError, [*errors, SEARCH_PATH_MESSAGE].join(".\n")
66
+ end
67
+ end
68
+ private_class_method :load_library
69
+
70
+ def self.try_load(libname, flags, errors)
71
+ begin
72
+ lib = FFI::DynamicLibrary.open(libname, flags)
73
+ return lib if lib
74
+
75
+ # LoadError for C ext & JRuby, RuntimeError for TruffleRuby
76
+ rescue LoadError, RuntimeError => ex
77
+ if ex.message =~ /(([^ \t()])+\.so([^ \t:()])*):([ \t])*(invalid ELF header|file too short|invalid file format)/
78
+ if File.binread($1) =~ /(?:GROUP|INPUT) *\( *([^ \)]+)/
79
+ return try_load($1, flags, errors)
80
+ end
81
+ end
82
+
83
+ errors << ex
84
+ nil
85
+ end
86
+ end
87
+ private_class_method :try_load
88
+ end
89
+ end
data/lib/ffi/enum.rb CHANGED
@@ -192,6 +192,7 @@ module FFI
192
192
  # @param [nil, Symbol] tag name of new Bitmask
193
193
  def initialize(*args)
194
194
  @native_type = args.first.kind_of?(FFI::Type) ? args.shift : Type::INT
195
+ @signed = [Type::INT8, Type::INT16, Type::INT32, Type::INT64].include?(@native_type)
195
196
  info, @tag = *args
196
197
  @kv_map = Hash.new
197
198
  unless info.nil?
@@ -220,7 +221,7 @@ module FFI
220
221
  # @param [Symbol] query
221
222
  # @return [Integer]
222
223
  # @overload [](query)
223
- # Get bitmaks value from symbol array
224
+ # Get bitmask value from symbol array
224
225
  # @param [Array<Symbol>] query
225
226
  # @return [Integer]
226
227
  # @overload [](*query)
@@ -240,7 +241,7 @@ module FFI
240
241
  when Symbol
241
242
  flat_query.inject(0) do |val, o|
242
243
  v = @kv_map[o]
243
- if v then val |= v else val end
244
+ if v then val | v else val end
244
245
  end
245
246
  when Integer, ->(o) { o.respond_to?(:to_int) }
246
247
  val = flat_query.inject(0) { |mask, o| mask |= o.to_int }
@@ -260,35 +261,41 @@ module FFI
260
261
  def to_native(query, ctx)
261
262
  return 0 if query.nil?
262
263
  flat_query = [query].flatten
263
- flat_query.inject(0) do |val, o|
264
+ res = flat_query.inject(0) do |val, o|
264
265
  case o
265
266
  when Symbol
266
267
  v = @kv_map[o]
267
268
  raise ArgumentError, "invalid bitmask value, #{o.inspect}" unless v
268
- val |= v
269
+ val | v
269
270
  when Integer
270
- val |= o
271
+ val | o
271
272
  when ->(obj) { obj.respond_to?(:to_int) }
272
- val |= o.to_int
273
+ val | o.to_int
273
274
  else
274
275
  raise ArgumentError, "invalid bitmask value, #{o.inspect}"
275
276
  end
276
277
  end
278
+ # Take two's complement of positive values bigger than the max value
279
+ # for the type when native type is signed.
280
+ if @signed && res >= (1 << (@native_type.size * 8 - 1))
281
+ res = -(-res & ((1 << (@native_type.size * 8)) - 1))
282
+ end
283
+ res
277
284
  end
278
285
 
279
286
  # @param [Integer] val
280
287
  # @param ctx unused
281
288
  # @return [Array<Symbol, Integer>] list of symbol names corresponding to val, plus an optional remainder if some bits don't match any constant
282
289
  def from_native(val, ctx)
283
- list = @kv_map.select { |_, v| v & val != 0 }.keys
290
+ flags = @kv_map.select { |_, v| v & val != 0 }
291
+ list = flags.keys
292
+ # force an unsigned value of the correct size
293
+ val &= (1 << (@native_type.size * 8)) - 1 if @signed
284
294
  # If there are unmatch flags,
285
295
  # return them in an integer,
286
296
  # else information can be lost.
287
297
  # Similar to Enum behavior.
288
- remainder = val ^ list.inject(0) do |tmp, o|
289
- v = @kv_map[o]
290
- if v then tmp |= v else tmp end
291
- end
298
+ remainder = val ^ flags.values.reduce(0, :|)
292
299
  list.push remainder unless remainder == 0
293
300
  return list
294
301
  end
data/lib/ffi/ffi.rb CHANGED
@@ -28,9 +28,11 @@
28
28
  # OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
29
29
  # OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30
30
 
31
+ require 'ffi/compat'
31
32
  require 'ffi/platform'
32
33
  require 'ffi/data_converter'
33
34
  require 'ffi/types'
35
+ require 'ffi/library_path'
34
36
  require 'ffi/library'
35
37
  require 'ffi/errno'
36
38
  require 'ffi/abstract_memory'
@@ -45,3 +47,4 @@ require 'ffi/autopointer'
45
47
  require 'ffi/variadic'
46
48
  require 'ffi/enum'
47
49
  require 'ffi/version'
50
+ require 'ffi/function'
@@ -0,0 +1,71 @@
1
+ #
2
+ # Copyright (C) 2008-2010 JRuby project
3
+ #
4
+ # This file is part of ruby-ffi.
5
+ #
6
+ # All rights reserved.
7
+ #
8
+ # Redistribution and use in source and binary forms, with or without
9
+ # modification, are permitted provided that the following conditions are met:
10
+ #
11
+ # * Redistributions of source code must retain the above copyright notice, this
12
+ # list of conditions and the following disclaimer.
13
+ # * Redistributions in binary form must reproduce the above copyright notice
14
+ # this list of conditions and the following disclaimer in the documentation
15
+ # and/or other materials provided with the distribution.
16
+ # * Neither the name of the Ruby FFI project nor the names of its contributors
17
+ # may be used to endorse or promote products derived from this software
18
+ # without specific prior written permission.
19
+ #
20
+ # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
21
+ # AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
22
+ # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
23
+ # DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE
24
+ # FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
25
+ # DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
26
+ # SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
27
+ # CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
28
+ # OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
29
+ # OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30
+
31
+ module FFI
32
+ class Function
33
+ # Only MRI allows function type queries
34
+ if private_method_defined?(:type)
35
+ # Retrieve the return type of the function
36
+ #
37
+ # This method returns FFI type returned by the function.
38
+ #
39
+ # @return [FFI::Type]
40
+ def return_type
41
+ type.return_type
42
+ end
43
+
44
+ # Retrieve Array of parameter types
45
+ #
46
+ # This method returns an Array of FFI types accepted as function parameters.
47
+ #
48
+ # @return [Array<FFI::Type>]
49
+ def param_types
50
+ type.param_types
51
+ end
52
+ end
53
+
54
+ # Stash the Function in a module variable so it can be inspected by attached_functions.
55
+ # On CRuby it also ensures that it does not get garbage collected.
56
+ module RegisterAttach
57
+ def attach(mod, name)
58
+ funcs = mod.instance_variable_get("@ffi_functions")
59
+ unless funcs
60
+ funcs = {}
61
+ mod.instance_variable_set("@ffi_functions", funcs)
62
+ end
63
+ funcs[name.to_sym] = self
64
+ # Jump to the native attach method of CRuby, JRuby or Tuffleruby
65
+ super
66
+ end
67
+ end
68
+ private_constant :RegisterAttach
69
+ prepend RegisterAttach
70
+ end
71
+ end
data/lib/ffi/library.rb CHANGED
@@ -28,10 +28,12 @@
28
28
  # OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
29
29
  # OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.#
30
30
 
31
+ require 'ffi/dynamic_library'
32
+
31
33
  module FFI
32
- CURRENT_PROCESS = USE_THIS_PROCESS_AS_LIBRARY = Object.new
34
+ CURRENT_PROCESS = USE_THIS_PROCESS_AS_LIBRARY = FFI.make_shareable(Object.new)
33
35
 
34
- # @param [#to_s] lib library name
36
+ # @param [String, FFI::LibraryPath] lib library name or LibraryPath object
35
37
  # @return [String] library name formatted for current platform
36
38
  # Transform a generic library name to a platform library name
37
39
  # @example
@@ -43,15 +45,7 @@ module FFI
43
45
  # FFI.map_library_name 'jpeg' # -> "jpeg.dll"
44
46
  def self.map_library_name(lib)
45
47
  # Mangle the library name to reflect the native library naming conventions
46
- lib = Library::LIBC if lib == 'c'
47
-
48
- if lib && File.basename(lib) == lib
49
- lib = Platform::LIBPREFIX + lib unless lib =~ /^#{Platform::LIBPREFIX}/
50
- r = Platform::IS_WINDOWS || Platform::IS_MAC ? "\\.#{Platform::LIBSUFFIX}$" : "\\.so($|\\.[1234567890]+)"
51
- lib += ".#{Platform::LIBSUFFIX}" unless lib =~ /#{r}/
52
- end
53
-
54
- lib
48
+ LibraryPath.wrap(lib).to_s
55
49
  end
56
50
 
57
51
  # Exception raised when a function is not found in libraries
@@ -95,62 +89,11 @@ module FFI
95
89
  def ffi_lib(*names)
96
90
  raise LoadError.new("library names list must not be empty") if names.empty?
97
91
 
98
- lib_flags = defined?(@ffi_lib_flags) ? @ffi_lib_flags : FFI::DynamicLibrary::RTLD_LAZY | FFI::DynamicLibrary::RTLD_LOCAL
99
- ffi_libs = names.map do |name|
92
+ lib_flags = defined?(@ffi_lib_flags) && @ffi_lib_flags
100
93
 
101
- if name == FFI::CURRENT_PROCESS
102
- FFI::DynamicLibrary.open(nil, FFI::DynamicLibrary::RTLD_LAZY | FFI::DynamicLibrary::RTLD_LOCAL)
103
-
104
- else
105
- libnames = (name.is_a?(::Array) ? name : [ name ]).map(&:to_s).map { |n| [ n, FFI.map_library_name(n) ].uniq }.flatten.compact
106
- lib = nil
107
- errors = {}
108
-
109
- libnames.each do |libname|
110
- begin
111
- orig = libname
112
- lib = FFI::DynamicLibrary.open(libname, lib_flags)
113
- break if lib
114
-
115
- rescue Exception => ex
116
- ldscript = false
117
- if ex.message =~ /(([^ \t()])+\.so([^ \t:()])*):([ \t])*(invalid ELF header|file too short|invalid file format)/
118
- if File.binread($1) =~ /(?:GROUP|INPUT) *\( *([^ \)]+)/
119
- libname = $1
120
- ldscript = true
121
- end
122
- end
123
-
124
- if ldscript
125
- retry
126
- else
127
- # TODO better library lookup logic
128
- unless libname.start_with?("/") || FFI::Platform.windows?
129
- path = ['/usr/lib/','/usr/local/lib/','/opt/local/lib/', '/opt/homebrew/lib/'].find do |pth|
130
- File.exist?(pth + libname)
131
- end
132
- if path
133
- libname = path + libname
134
- retry
135
- end
136
- end
137
-
138
- libr = (orig == libname ? orig : "#{orig} #{libname}")
139
- errors[libr] = ex
140
- end
141
- end
142
- end
143
-
144
- if lib.nil?
145
- raise LoadError.new(errors.values.join(".\n"))
146
- end
147
-
148
- # return the found lib
149
- lib
150
- end
94
+ @ffi_libs = names.map do |name|
95
+ FFI::DynamicLibrary.send(:load_library, name, lib_flags)
151
96
  end
152
-
153
- @ffi_libs = ffi_libs
154
97
  end
155
98
 
156
99
  # Set the calling convention for {#attach_function} and {#callback}
@@ -258,7 +201,7 @@ module FFI
258
201
  end
259
202
  raise LoadError unless function
260
203
 
261
- invokers << if arg_types.length > 0 && arg_types[arg_types.length - 1] == FFI::NativeType::VARARGS
204
+ invokers << if arg_types[-1] == FFI::NativeType::VARARGS
262
205
  VariadicInvoker.new(function, arg_types, find_type(ret_type), options)
263
206
 
264
207
  else
@@ -330,6 +273,7 @@ module FFI
330
273
  # Attach C variable +cname+ to this module.
331
274
  def attach_variable(mname, a1, a2 = nil)
332
275
  cname, type = a2 ? [ a1, a2 ] : [ mname.to_s, a1 ]
276
+ mname = mname.to_sym
333
277
  address = nil
334
278
  ffi_libraries.each do |lib|
335
279
  begin
@@ -344,9 +288,10 @@ module FFI
344
288
  # If it is a global struct, just attach directly to the pointer
345
289
  s = s = type.new(address) # Assigning twice to suppress unused variable warning
346
290
  self.module_eval <<-code, __FILE__, __LINE__
347
- @@ffi_gvar_#{mname} = s
291
+ @ffi_gsvars = {} unless defined?(@ffi_gsvars)
292
+ @ffi_gsvars[#{mname.inspect}] = s
348
293
  def self.#{mname}
349
- @@ffi_gvar_#{mname}
294
+ @ffi_gsvars[#{mname.inspect}]
350
295
  end
351
296
  code
352
297
 
@@ -358,12 +303,13 @@ module FFI
358
303
  # Attach to this module as mname/mname=
359
304
  #
360
305
  self.module_eval <<-code, __FILE__, __LINE__
361
- @@ffi_gvar_#{mname} = s
306
+ @ffi_gvars = {} unless defined?(@ffi_gvars)
307
+ @ffi_gvars[#{mname.inspect}] = s
362
308
  def self.#{mname}
363
- @@ffi_gvar_#{mname}[:gvar]
309
+ @ffi_gvars[#{mname.inspect}][:gvar]
364
310
  end
365
311
  def self.#{mname}=(value)
366
- @@ffi_gvar_#{mname}[:gvar] = value
312
+ @ffi_gvars[#{mname.inspect}][:gvar] = value
367
313
  end
368
314
  code
369
315
 
@@ -588,5 +534,43 @@ module FFI
588
534
 
589
535
  end || FFI.find_type(t)
590
536
  end
537
+
538
+ # Retrieve all attached functions and their function signature
539
+ #
540
+ # This method returns a Hash of method names of attached functions connected by #attach_function and the corresponding function type.
541
+ # The function type responds to #return_type and #param_types which return the FFI types of the function signature.
542
+ #
543
+ # @return [Hash< Symbol => [FFI::Function, FFI::VariadicInvoker] >]
544
+ def attached_functions
545
+ @ffi_functions || {}
546
+ end
547
+
548
+ # Retrieve all attached variables and their type
549
+ #
550
+ # This method returns a Hash of variable names and the corresponding type or variables connected by #attach_variable .
551
+ #
552
+ # @return [Hash< Symbol => ffi_type >]
553
+ def attached_variables
554
+ (
555
+ (@ffi_gsvars || {}).map do |name, gvar|
556
+ [name, gvar.class]
557
+ end +
558
+ (@ffi_gvars || {}).map do |name, gvar|
559
+ [name, gvar.layout[:gvar].type]
560
+ end
561
+ ).to_h
562
+ end
563
+
564
+ # Freeze all definitions of the module
565
+ #
566
+ # This freezes the module's definitions, so that it can be used in a Ractor.
567
+ # No further methods or variables can be attached and no further enums or typedefs can be created in this module afterwards.
568
+ def freeze
569
+ instance_variables.each do |name|
570
+ var = instance_variable_get(name)
571
+ FFI.make_shareable(var)
572
+ end
573
+ nil
574
+ end
591
575
  end
592
576
  end