ffi 1.9.18 → 1.9.21

Sign up to get free protection for your applications and to get access to all the features.

Potentially problematic release.


This version of ffi might be problematic. Click here for more details.

Files changed (366) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +3 -2
  3. data/Rakefile +14 -3
  4. data/ext/ffi_c/AbstractMemory.c +6 -1
  5. data/ext/ffi_c/Platform.c +10 -2
  6. data/ext/ffi_c/extconf.rb +7 -2
  7. data/ext/ffi_c/libffi.bsd.mk +9 -3
  8. data/ext/ffi_c/libffi.darwin.mk +14 -4
  9. data/ext/ffi_c/libffi.gnu.mk +2 -1
  10. data/ext/ffi_c/libffi.mk +9 -4
  11. data/ext/ffi_c/libffi/ChangeLog.libffi +2 -2
  12. data/ext/ffi_c/libffi/{ChangeLog → ChangeLog.libffi-3.1} +1402 -2
  13. data/ext/ffi_c/libffi/ChangeLog.v1 +1 -1
  14. data/ext/ffi_c/libffi/LICENSE +1 -1
  15. data/ext/ffi_c/libffi/Makefile.am +166 -157
  16. data/ext/ffi_c/libffi/README +164 -52
  17. data/ext/ffi_c/libffi/acinclude.m4 +381 -0
  18. data/ext/ffi_c/libffi/autogen.sh +2 -0
  19. data/ext/ffi_c/libffi/configure.ac +148 -256
  20. data/ext/ffi_c/libffi/configure.host +265 -4
  21. data/ext/ffi_c/libffi/doc/Makefile.am +3 -0
  22. data/ext/ffi_c/libffi/doc/libffi.texi +430 -45
  23. data/ext/ffi_c/libffi/doc/version.texi +4 -4
  24. data/ext/ffi_c/libffi/generate-darwin-source-and-headers.py +207 -0
  25. data/ext/ffi_c/libffi/include/Makefile.am +3 -3
  26. data/ext/ffi_c/libffi/include/ffi.h.in +107 -50
  27. data/ext/ffi_c/libffi/include/ffi_cfi.h +55 -0
  28. data/ext/ffi_c/libffi/include/ffi_common.h +32 -11
  29. data/ext/ffi_c/libffi/libffi.map.in +80 -0
  30. data/ext/ffi_c/libffi/libffi.pc.in +3 -2
  31. data/ext/ffi_c/libffi/libffi.xcodeproj/project.pbxproj +637 -0
  32. data/ext/ffi_c/libffi/libtool-ldflags +106 -0
  33. data/ext/ffi_c/libffi/libtool-version +1 -1
  34. data/ext/ffi_c/libffi/m4/asmcfi.m4 +13 -0
  35. data/ext/ffi_c/libffi/m4/ax_append_flag.m4 +69 -0
  36. data/ext/ffi_c/libffi/m4/ax_cc_maxopt.m4 +13 -8
  37. data/ext/ffi_c/libffi/m4/ax_cflags_warn_all.m4 +31 -104
  38. data/ext/ffi_c/libffi/m4/{ax_check_compiler_flags.m4 → ax_check_compile_flag.m4} +30 -34
  39. data/ext/ffi_c/libffi/m4/ax_compiler_vendor.m4 +32 -11
  40. data/ext/ffi_c/libffi/m4/ax_enable_builddir.m4 +6 -5
  41. data/ext/ffi_c/libffi/m4/ax_gcc_archflag.m4 +31 -21
  42. data/ext/ffi_c/libffi/man/Makefile.am +2 -2
  43. data/ext/ffi_c/libffi/man/ffi.3 +10 -0
  44. data/ext/ffi_c/libffi/man/ffi_prep_cif.3 +6 -4
  45. data/ext/ffi_c/libffi/man/ffi_prep_cif_var.3 +73 -0
  46. data/ext/ffi_c/libffi/msvcc.sh +72 -9
  47. data/ext/ffi_c/libffi/src/aarch64/ffi.c +941 -0
  48. data/ext/ffi_c/libffi/src/aarch64/ffitarget.h +81 -0
  49. data/ext/ffi_c/libffi/src/aarch64/internal.h +67 -0
  50. data/ext/ffi_c/libffi/src/aarch64/sysv.S +438 -0
  51. data/ext/ffi_c/libffi/src/alpha/ffi.c +335 -98
  52. data/ext/ffi_c/libffi/src/alpha/ffitarget.h +10 -1
  53. data/ext/ffi_c/libffi/src/alpha/internal.h +23 -0
  54. data/ext/ffi_c/libffi/src/alpha/osf.S +161 -266
  55. data/ext/ffi_c/libffi/src/arc/arcompact.S +135 -0
  56. data/ext/ffi_c/libffi/src/arc/ffi.c +266 -0
  57. data/ext/ffi_c/libffi/src/arc/ffitarget.h +53 -0
  58. data/ext/ffi_c/libffi/src/arm/ffi.c +597 -517
  59. data/ext/ffi_c/libffi/src/arm/ffitarget.h +24 -7
  60. data/ext/ffi_c/libffi/src/arm/internal.h +7 -0
  61. data/ext/ffi_c/libffi/src/arm/sysv.S +303 -417
  62. data/ext/ffi_c/libffi/src/avr32/ffitarget.h +6 -1
  63. data/ext/ffi_c/libffi/src/bfin/ffi.c +196 -0
  64. data/ext/ffi_c/libffi/src/bfin/ffitarget.h +43 -0
  65. data/ext/ffi_c/libffi/src/bfin/sysv.S +179 -0
  66. data/ext/ffi_c/libffi/src/closures.c +319 -44
  67. data/ext/ffi_c/libffi/src/cris/ffi.c +10 -7
  68. data/ext/ffi_c/libffi/src/cris/ffitarget.h +6 -1
  69. data/ext/ffi_c/libffi/src/debug.c +6 -1
  70. data/ext/ffi_c/libffi/src/dlmalloc.c +16 -11
  71. data/ext/ffi_c/libffi/src/frv/ffi.c +1 -1
  72. data/ext/ffi_c/libffi/src/frv/ffitarget.h +6 -1
  73. data/ext/ffi_c/libffi/src/ia64/ffi.c +11 -7
  74. data/ext/ffi_c/libffi/src/ia64/ffitarget.h +6 -1
  75. data/ext/ffi_c/libffi/src/java_raw_api.c +23 -5
  76. data/ext/ffi_c/libffi/src/m32r/ffi.c +1 -1
  77. data/ext/ffi_c/libffi/src/m32r/ffitarget.h +6 -1
  78. data/ext/ffi_c/libffi/src/m68k/ffi.c +87 -13
  79. data/ext/ffi_c/libffi/src/m68k/ffitarget.h +6 -1
  80. data/ext/ffi_c/libffi/src/m68k/sysv.S +119 -32
  81. data/ext/ffi_c/libffi/src/m88k/ffi.c +400 -0
  82. data/ext/ffi_c/libffi/src/m88k/ffitarget.h +49 -0
  83. data/ext/ffi_c/libffi/src/m88k/obsd.S +209 -0
  84. data/ext/ffi_c/libffi/src/metag/ffi.c +330 -0
  85. data/ext/ffi_c/libffi/{fficonfig.hw → src/metag/ffitarget.h} +22 -26
  86. data/ext/ffi_c/libffi/src/metag/sysv.S +311 -0
  87. data/ext/ffi_c/libffi/src/microblaze/ffi.c +321 -0
  88. data/ext/ffi_c/libffi/src/microblaze/ffitarget.h +53 -0
  89. data/ext/ffi_c/libffi/src/microblaze/sysv.S +302 -0
  90. data/ext/ffi_c/libffi/src/mips/ffi.c +95 -28
  91. data/ext/ffi_c/libffi/src/mips/ffitarget.h +9 -2
  92. data/ext/ffi_c/libffi/src/mips/n32.S +126 -56
  93. data/ext/ffi_c/libffi/src/mips/o32.S +148 -27
  94. data/ext/ffi_c/libffi/src/moxie/eabi.S +55 -82
  95. data/ext/ffi_c/libffi/src/moxie/ffi.c +40 -44
  96. data/ext/ffi_c/libffi/src/moxie/ffitarget.h +52 -0
  97. data/ext/ffi_c/libffi/src/nios2/ffi.c +304 -0
  98. data/ext/ffi_c/libffi/src/nios2/ffitarget.h +52 -0
  99. data/ext/ffi_c/libffi/src/nios2/sysv.S +136 -0
  100. data/ext/ffi_c/libffi/src/or1k/ffi.c +328 -0
  101. data/ext/ffi_c/libffi/src/or1k/ffitarget.h +58 -0
  102. data/ext/ffi_c/libffi/src/or1k/sysv.S +107 -0
  103. data/ext/ffi_c/libffi/src/pa/ffitarget.h +8 -1
  104. data/ext/ffi_c/libffi/src/powerpc/aix.S +6 -6
  105. data/ext/ffi_c/libffi/src/powerpc/aix_closure.S +3 -1
  106. data/ext/ffi_c/libffi/src/powerpc/asm.h +2 -2
  107. data/ext/ffi_c/libffi/src/powerpc/darwin.S +2 -7
  108. data/ext/ffi_c/libffi/src/powerpc/darwin_closure.S +22 -26
  109. data/ext/ffi_c/libffi/src/powerpc/ffi.c +103 -1378
  110. data/ext/ffi_c/libffi/src/powerpc/ffi_darwin.c +25 -25
  111. data/ext/ffi_c/libffi/src/powerpc/ffi_linux64.c +945 -0
  112. data/ext/ffi_c/libffi/src/powerpc/ffi_powerpc.h +94 -0
  113. data/ext/ffi_c/libffi/src/powerpc/ffi_sysv.c +923 -0
  114. data/ext/ffi_c/libffi/src/powerpc/ffitarget.h +100 -44
  115. data/ext/ffi_c/libffi/src/powerpc/linux64.S +100 -59
  116. data/ext/ffi_c/libffi/src/powerpc/linux64_closure.S +360 -108
  117. data/ext/ffi_c/libffi/src/powerpc/ppc_closure.S +138 -68
  118. data/ext/ffi_c/libffi/src/powerpc/sysv.S +68 -112
  119. data/ext/ffi_c/libffi/src/prep_cif.c +108 -24
  120. data/ext/ffi_c/libffi/src/raw_api.c +18 -5
  121. data/ext/ffi_c/libffi/src/s390/ffi.c +294 -318
  122. data/ext/ffi_c/libffi/src/s390/ffitarget.h +9 -1
  123. data/ext/ffi_c/libffi/src/s390/internal.h +11 -0
  124. data/ext/ffi_c/libffi/src/s390/sysv.S +257 -366
  125. data/ext/ffi_c/libffi/src/sh/ffi.c +4 -3
  126. data/ext/ffi_c/libffi/src/sh/ffitarget.h +6 -1
  127. data/ext/ffi_c/libffi/src/sh64/ffi.c +3 -2
  128. data/ext/ffi_c/libffi/src/sh64/ffitarget.h +6 -1
  129. data/ext/ffi_c/libffi/src/sparc/ffi.c +326 -527
  130. data/ext/ffi_c/libffi/src/sparc/ffi64.c +608 -0
  131. data/ext/ffi_c/libffi/src/sparc/ffitarget.h +20 -7
  132. data/ext/ffi_c/libffi/src/sparc/internal.h +26 -0
  133. data/ext/ffi_c/libffi/src/sparc/v8.S +364 -234
  134. data/ext/ffi_c/libffi/src/sparc/v9.S +340 -207
  135. data/ext/ffi_c/libffi/src/tile/ffi.c +355 -0
  136. data/ext/ffi_c/libffi/src/tile/ffitarget.h +65 -0
  137. data/ext/ffi_c/libffi/src/tile/tile.S +360 -0
  138. data/ext/ffi_c/libffi/src/types.c +43 -14
  139. data/ext/ffi_c/libffi/src/vax/elfbsd.S +195 -0
  140. data/ext/ffi_c/libffi/src/vax/ffi.c +276 -0
  141. data/ext/ffi_c/libffi/src/vax/ffitarget.h +49 -0
  142. data/ext/ffi_c/libffi/src/x86/asmnames.h +30 -0
  143. data/ext/ffi_c/libffi/src/x86/ffi.c +589 -500
  144. data/ext/ffi_c/libffi/src/x86/ffi64.c +338 -116
  145. data/ext/ffi_c/libffi/src/x86/ffitarget.h +55 -35
  146. data/ext/ffi_c/libffi/src/x86/ffiw64.c +287 -0
  147. data/ext/ffi_c/libffi/src/x86/internal.h +29 -0
  148. data/ext/ffi_c/libffi/src/x86/internal64.h +22 -0
  149. data/ext/ffi_c/libffi/src/x86/sysv.S +975 -400
  150. data/ext/ffi_c/libffi/src/x86/unix64.S +398 -299
  151. data/ext/ffi_c/libffi/src/x86/win64.S +222 -458
  152. data/ext/ffi_c/libffi/src/x86/win64_intel.S +237 -0
  153. data/ext/ffi_c/libffi/src/xtensa/ffi.c +298 -0
  154. data/ext/ffi_c/libffi/src/xtensa/ffitarget.h +53 -0
  155. data/ext/ffi_c/libffi/src/xtensa/sysv.S +253 -0
  156. data/ext/ffi_c/libffi/stamp-h.in +1 -0
  157. data/ext/ffi_c/libffi/testsuite/Makefile.am +78 -73
  158. data/ext/ffi_c/libffi/testsuite/lib/libffi.exp +120 -25
  159. data/ext/ffi_c/libffi/testsuite/lib/target-libpath.exp +21 -1
  160. data/ext/ffi_c/libffi/testsuite/libffi.call/align_mixed.c +46 -0
  161. data/ext/ffi_c/libffi/testsuite/libffi.call/call.exp +4 -6
  162. data/ext/ffi_c/libffi/testsuite/libffi.call/{closure_stdcall.c → closure_simple.c} +7 -16
  163. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_12byte.c +4 -4
  164. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_16byte.c +4 -4
  165. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_18byte.c +4 -4
  166. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_19byte.c +4 -4
  167. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_1_1byte.c +4 -4
  168. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_20byte.c +4 -4
  169. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_20byte1.c +4 -4
  170. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_24byte.c +5 -5
  171. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_2byte.c +4 -4
  172. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_3_1byte.c +4 -4
  173. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_3byte1.c +4 -4
  174. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_3byte2.c +4 -4
  175. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_4_1byte.c +4 -4
  176. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_4byte.c +4 -4
  177. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_5_1_byte.c +4 -4
  178. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_5byte.c +4 -4
  179. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_64byte.c +5 -5
  180. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_6_1_byte.c +4 -4
  181. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_6byte.c +4 -4
  182. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_7_1_byte.c +4 -4
  183. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_7byte.c +4 -4
  184. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_8byte.c +4 -4
  185. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_9byte1.c +4 -4
  186. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_9byte2.c +4 -4
  187. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_double.c +4 -4
  188. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_float.c +4 -4
  189. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_longdouble.c +4 -4
  190. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_longdouble_split.c +4 -6
  191. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_longdouble_split2.c +4 -6
  192. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_pointer.c +4 -4
  193. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_sint16.c +4 -4
  194. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_sint32.c +4 -4
  195. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_sint64.c +4 -4
  196. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_uint16.c +4 -4
  197. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_uint32.c +4 -4
  198. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_uint64.c +4 -4
  199. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_dbls_struct.c +4 -4
  200. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_double_va.c +10 -9
  201. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_longdouble.c +3 -3
  202. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_longdouble_va.c +10 -9
  203. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_many_mixed_args.c +70 -0
  204. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_many_mixed_float_double.c +55 -0
  205. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_pointer.c +1 -1
  206. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_pointer_stack.c +11 -9
  207. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_struct_va1.c +114 -0
  208. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_uchar_va.c +44 -0
  209. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_uint_va.c +45 -0
  210. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_ulong_va.c +45 -0
  211. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_ulonglong.c +5 -5
  212. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_ushort_va.c +44 -0
  213. data/ext/ffi_c/libffi/testsuite/libffi.call/err_bad_typedef.c +2 -2
  214. data/ext/ffi_c/libffi/testsuite/libffi.call/ffitest.h +23 -40
  215. data/ext/ffi_c/libffi/testsuite/libffi.call/float1.c +3 -1
  216. data/ext/ffi_c/libffi/testsuite/libffi.call/float2.c +6 -4
  217. data/ext/ffi_c/libffi/testsuite/libffi.call/float3.c +4 -2
  218. data/ext/ffi_c/libffi/testsuite/libffi.call/float_va.c +107 -0
  219. data/ext/ffi_c/libffi/testsuite/libffi.call/huge_struct.c +18 -19
  220. data/ext/ffi_c/libffi/testsuite/libffi.call/many.c +6 -16
  221. data/ext/ffi_c/libffi/testsuite/libffi.call/many2.c +57 -0
  222. data/ext/ffi_c/libffi/testsuite/libffi.call/many_double.c +70 -0
  223. data/ext/ffi_c/libffi/testsuite/libffi.call/many_mixed.c +78 -0
  224. data/ext/ffi_c/libffi/testsuite/libffi.call/negint.c +0 -1
  225. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct.c +6 -6
  226. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct1.c +8 -8
  227. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct10.c +6 -6
  228. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct11.c +121 -0
  229. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct2.c +5 -5
  230. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct3.c +5 -5
  231. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct4.c +5 -5
  232. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct5.c +5 -5
  233. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct6.c +6 -6
  234. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct7.c +5 -5
  235. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct8.c +6 -6
  236. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct9.c +6 -6
  237. data/ext/ffi_c/libffi/testsuite/libffi.call/offsets.c +46 -0
  238. data/ext/ffi_c/libffi/testsuite/libffi.call/pr1172638.c +127 -0
  239. data/ext/ffi_c/libffi/testsuite/libffi.call/return_dbl.c +1 -0
  240. data/ext/ffi_c/libffi/testsuite/libffi.call/return_ldl.c +1 -1
  241. data/ext/ffi_c/libffi/testsuite/libffi.call/return_sc.c +1 -1
  242. data/ext/ffi_c/libffi/testsuite/libffi.call/return_uc.c +1 -1
  243. data/ext/ffi_c/libffi/testsuite/libffi.call/stret_large.c +7 -7
  244. data/ext/ffi_c/libffi/testsuite/libffi.call/stret_large2.c +7 -7
  245. data/ext/ffi_c/libffi/testsuite/libffi.call/stret_medium.c +5 -5
  246. data/ext/ffi_c/libffi/testsuite/libffi.call/stret_medium2.c +5 -5
  247. data/ext/ffi_c/libffi/testsuite/libffi.call/strlen.c +2 -2
  248. data/ext/ffi_c/libffi/testsuite/libffi.call/strlen2.c +49 -0
  249. data/ext/ffi_c/libffi/testsuite/libffi.call/strlen3.c +49 -0
  250. data/ext/ffi_c/libffi/testsuite/libffi.call/strlen4.c +55 -0
  251. data/ext/ffi_c/libffi/testsuite/libffi.call/struct1.c +9 -7
  252. data/ext/ffi_c/libffi/testsuite/libffi.call/struct2.c +7 -7
  253. data/ext/ffi_c/libffi/testsuite/libffi.call/struct3.c +7 -6
  254. data/ext/ffi_c/libffi/testsuite/libffi.call/struct4.c +9 -8
  255. data/ext/ffi_c/libffi/testsuite/libffi.call/struct5.c +9 -8
  256. data/ext/ffi_c/libffi/testsuite/libffi.call/struct6.c +9 -9
  257. data/ext/ffi_c/libffi/testsuite/libffi.call/struct7.c +9 -9
  258. data/ext/ffi_c/libffi/testsuite/libffi.call/struct8.c +9 -8
  259. data/ext/ffi_c/libffi/testsuite/libffi.call/struct9.c +9 -8
  260. data/ext/ffi_c/libffi/testsuite/libffi.call/testclosure.c +2 -2
  261. data/ext/ffi_c/libffi/testsuite/libffi.call/uninitialized.c +61 -0
  262. data/ext/ffi_c/libffi/testsuite/{libffi.special → libffi.call}/unwindtest.cc +3 -10
  263. data/ext/ffi_c/libffi/testsuite/{libffi.special → libffi.call}/unwindtest_ffi_call.cc +2 -1
  264. data/ext/ffi_c/libffi/testsuite/libffi.call/va_1.c +196 -0
  265. data/ext/ffi_c/libffi/testsuite/libffi.call/va_struct1.c +121 -0
  266. data/ext/ffi_c/libffi/testsuite/libffi.call/va_struct2.c +123 -0
  267. data/ext/ffi_c/libffi/testsuite/libffi.call/va_struct3.c +125 -0
  268. data/ext/ffi_c/libffi/testsuite/libffi.complex/cls_align_complex.inc +91 -0
  269. data/ext/ffi_c/libffi/testsuite/libffi.complex/cls_align_complex_double.c +10 -0
  270. data/ext/ffi_c/libffi/testsuite/libffi.complex/cls_align_complex_float.c +10 -0
  271. data/ext/ffi_c/libffi/testsuite/libffi.complex/cls_align_complex_longdouble.c +10 -0
  272. data/ext/ffi_c/libffi/testsuite/libffi.complex/cls_complex.inc +42 -0
  273. data/ext/ffi_c/libffi/testsuite/libffi.complex/cls_complex_double.c +10 -0
  274. data/ext/ffi_c/libffi/testsuite/libffi.complex/cls_complex_float.c +10 -0
  275. data/ext/ffi_c/libffi/testsuite/libffi.complex/cls_complex_longdouble.c +10 -0
  276. data/ext/ffi_c/libffi/testsuite/libffi.complex/cls_complex_struct.inc +71 -0
  277. data/ext/ffi_c/libffi/testsuite/libffi.complex/cls_complex_struct_double.c +10 -0
  278. data/ext/ffi_c/libffi/testsuite/libffi.complex/cls_complex_struct_float.c +10 -0
  279. data/ext/ffi_c/libffi/testsuite/libffi.complex/cls_complex_struct_longdouble.c +10 -0
  280. data/ext/ffi_c/libffi/testsuite/libffi.complex/cls_complex_va.inc +80 -0
  281. data/ext/ffi_c/libffi/testsuite/libffi.complex/cls_complex_va_double.c +10 -0
  282. data/ext/ffi_c/libffi/testsuite/libffi.complex/cls_complex_va_float.c +16 -0
  283. data/ext/ffi_c/libffi/testsuite/libffi.complex/cls_complex_va_longdouble.c +10 -0
  284. data/ext/ffi_c/libffi/testsuite/{libffi.special/special.exp → libffi.complex/complex.exp} +9 -8
  285. data/ext/ffi_c/libffi/testsuite/libffi.complex/complex.inc +51 -0
  286. data/ext/ffi_c/libffi/testsuite/libffi.complex/complex_defs_double.inc +7 -0
  287. data/ext/ffi_c/libffi/testsuite/libffi.complex/complex_defs_float.inc +7 -0
  288. data/ext/ffi_c/libffi/testsuite/libffi.complex/complex_defs_longdouble.inc +7 -0
  289. data/ext/ffi_c/libffi/testsuite/libffi.complex/complex_double.c +10 -0
  290. data/ext/ffi_c/libffi/testsuite/libffi.complex/complex_float.c +10 -0
  291. data/ext/ffi_c/libffi/testsuite/libffi.complex/complex_int.c +86 -0
  292. data/ext/ffi_c/libffi/testsuite/libffi.complex/complex_longdouble.c +10 -0
  293. data/ext/ffi_c/libffi/testsuite/libffi.complex/ffitest.h +1 -0
  294. data/ext/ffi_c/libffi/testsuite/libffi.complex/many_complex.inc +78 -0
  295. data/ext/ffi_c/libffi/testsuite/libffi.complex/many_complex_double.c +10 -0
  296. data/ext/ffi_c/libffi/testsuite/libffi.complex/many_complex_float.c +10 -0
  297. data/ext/ffi_c/libffi/testsuite/libffi.complex/many_complex_longdouble.c +10 -0
  298. data/ext/ffi_c/libffi/testsuite/libffi.complex/return_complex.inc +37 -0
  299. data/ext/ffi_c/libffi/testsuite/libffi.complex/return_complex1.inc +41 -0
  300. data/ext/ffi_c/libffi/testsuite/libffi.complex/return_complex1_double.c +10 -0
  301. data/ext/ffi_c/libffi/testsuite/libffi.complex/return_complex1_float.c +10 -0
  302. data/ext/ffi_c/libffi/testsuite/libffi.complex/return_complex1_longdouble.c +10 -0
  303. data/ext/ffi_c/libffi/testsuite/libffi.complex/return_complex2.inc +44 -0
  304. data/ext/ffi_c/libffi/testsuite/libffi.complex/return_complex2_double.c +10 -0
  305. data/ext/ffi_c/libffi/testsuite/libffi.complex/return_complex2_float.c +10 -0
  306. data/ext/ffi_c/libffi/testsuite/libffi.complex/return_complex2_longdouble.c +10 -0
  307. data/ext/ffi_c/libffi/testsuite/libffi.complex/return_complex_double.c +10 -0
  308. data/ext/ffi_c/libffi/testsuite/libffi.complex/return_complex_float.c +10 -0
  309. data/ext/ffi_c/libffi/testsuite/libffi.complex/return_complex_longdouble.c +10 -0
  310. data/ext/ffi_c/libffi/testsuite/libffi.go/aa-direct.c +34 -0
  311. data/ext/ffi_c/libffi/testsuite/libffi.go/closure1.c +28 -0
  312. data/ext/ffi_c/libffi/testsuite/libffi.go/ffitest.h +1 -0
  313. data/ext/ffi_c/libffi/testsuite/libffi.go/go.exp +36 -0
  314. data/ext/ffi_c/libffi/testsuite/libffi.go/static-chain.h +19 -0
  315. data/ffi.gemspec +2 -2
  316. data/lib/ffi/enum.rb +124 -0
  317. data/lib/ffi/library.rb +65 -13
  318. data/lib/ffi/platform.rb +7 -2
  319. data/lib/ffi/platform/sparc64-linux/types.conf +102 -0
  320. data/lib/ffi/platform/x86_64-windows/types.conf +113 -20
  321. data/lib/ffi/pointer.rb +1 -0
  322. data/lib/ffi/struct.rb +0 -2
  323. data/lib/ffi/version.rb +1 -1
  324. data/spec/ffi/bitmask_spec.rb +575 -0
  325. data/spec/ffi/fixtures/BitmaskTest.c +51 -0
  326. data/spec/ffi/rbx/memory_pointer_spec.rb +4 -0
  327. data/spec/ffi/struct_spec.rb +0 -4
  328. metadata +143 -51
  329. data/ext/ffi_c/libffi/Makefile.in +0 -1820
  330. data/ext/ffi_c/libffi/Makefile.vc +0 -141
  331. data/ext/ffi_c/libffi/Makefile.vc64 +0 -141
  332. data/ext/ffi_c/libffi/aclocal.m4 +0 -1873
  333. data/ext/ffi_c/libffi/build-ios.sh +0 -67
  334. data/ext/ffi_c/libffi/compile +0 -143
  335. data/ext/ffi_c/libffi/config.guess +0 -1501
  336. data/ext/ffi_c/libffi/config.sub +0 -1705
  337. data/ext/ffi_c/libffi/configure +0 -17191
  338. data/ext/ffi_c/libffi/depcomp +0 -630
  339. data/ext/ffi_c/libffi/doc/libffi.info +0 -593
  340. data/ext/ffi_c/libffi/doc/stamp-vti +0 -4
  341. data/ext/ffi_c/libffi/fficonfig.h.in +0 -199
  342. data/ext/ffi_c/libffi/include/Makefile.in +0 -487
  343. data/ext/ffi_c/libffi/include/ffi.h.vc +0 -427
  344. data/ext/ffi_c/libffi/include/ffi.h.vc64 +0 -427
  345. data/ext/ffi_c/libffi/install-sh +0 -520
  346. data/ext/ffi_c/libffi/ltmain.sh +0 -9636
  347. data/ext/ffi_c/libffi/m4/libtool.m4 +0 -7831
  348. data/ext/ffi_c/libffi/m4/ltoptions.m4 +0 -369
  349. data/ext/ffi_c/libffi/m4/ltsugar.m4 +0 -123
  350. data/ext/ffi_c/libffi/m4/ltversion.m4 +0 -23
  351. data/ext/ffi_c/libffi/m4/lt~obsolete.m4 +0 -98
  352. data/ext/ffi_c/libffi/man/Makefile.in +0 -466
  353. data/ext/ffi_c/libffi/mdate-sh +0 -201
  354. data/ext/ffi_c/libffi/missing +0 -376
  355. data/ext/ffi_c/libffi/src/arm/gentramp.sh +0 -118
  356. data/ext/ffi_c/libffi/src/arm/trampoline.S +0 -4450
  357. data/ext/ffi_c/libffi/src/x86/darwin.S +0 -444
  358. data/ext/ffi_c/libffi/src/x86/darwin64.S +0 -416
  359. data/ext/ffi_c/libffi/src/x86/freebsd.S +0 -458
  360. data/ext/ffi_c/libffi/src/x86/win32.S +0 -1065
  361. data/ext/ffi_c/libffi/testsuite/Makefile.in +0 -500
  362. data/ext/ffi_c/libffi/testsuite/lib/libffi-dg.exp +0 -300
  363. data/ext/ffi_c/libffi/testsuite/libffi.call/many_win32.c +0 -63
  364. data/ext/ffi_c/libffi/testsuite/libffi.call/strlen_win32.c +0 -44
  365. data/ext/ffi_c/libffi/testsuite/libffi.special/ffitestcxx.h +0 -96
  366. data/ext/ffi_c/libffi/texinfo.tex +0 -7210
@@ -0,0 +1,941 @@
1
+ /* Copyright (c) 2009, 2010, 2011, 2012 ARM Ltd.
2
+
3
+ Permission is hereby granted, free of charge, to any person obtaining
4
+ a copy of this software and associated documentation files (the
5
+ ``Software''), to deal in the Software without restriction, including
6
+ without limitation the rights to use, copy, modify, merge, publish,
7
+ distribute, sublicense, and/or sell copies of the Software, and to
8
+ permit persons to whom the Software is furnished to do so, subject to
9
+ the following conditions:
10
+
11
+ The above copyright notice and this permission notice shall be
12
+ included in all copies or substantial portions of the Software.
13
+
14
+ THE SOFTWARE IS PROVIDED ``AS IS'', WITHOUT WARRANTY OF ANY KIND,
15
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
16
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
17
+ IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
18
+ CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
19
+ TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
20
+ SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
21
+
22
+ #include <stdio.h>
23
+ #include <stdlib.h>
24
+ #include <stdint.h>
25
+ #include <fficonfig.h>
26
+ #include <ffi.h>
27
+ #include <ffi_common.h>
28
+ #include "internal.h"
29
+
30
+ /* Force FFI_TYPE_LONGDOUBLE to be different than FFI_TYPE_DOUBLE;
31
+ all further uses in this file will refer to the 128-bit type. */
32
+ #if FFI_TYPE_DOUBLE != FFI_TYPE_LONGDOUBLE
33
+ # if FFI_TYPE_LONGDOUBLE != 4
34
+ # error FFI_TYPE_LONGDOUBLE out of date
35
+ # endif
36
+ #else
37
+ # undef FFI_TYPE_LONGDOUBLE
38
+ # define FFI_TYPE_LONGDOUBLE 4
39
+ #endif
40
+
41
+ union _d
42
+ {
43
+ UINT64 d;
44
+ UINT32 s[2];
45
+ };
46
+
47
+ struct _v
48
+ {
49
+ union _d d[2] __attribute__((aligned(16)));
50
+ };
51
+
52
+ struct call_context
53
+ {
54
+ struct _v v[N_V_ARG_REG];
55
+ UINT64 x[N_X_ARG_REG];
56
+ };
57
+
58
+ #if FFI_EXEC_TRAMPOLINE_TABLE
59
+
60
+ #ifdef __MACH__
61
+ #include <mach/vm_param.h>
62
+ #endif
63
+
64
+ #else
65
+
66
+ #if defined (__clang__) && defined (__APPLE__)
67
+ extern void sys_icache_invalidate (void *start, size_t len);
68
+ #endif
69
+
70
+ static inline void
71
+ ffi_clear_cache (void *start, void *end)
72
+ {
73
+ #if defined (__clang__) && defined (__APPLE__)
74
+ sys_icache_invalidate (start, (char *)end - (char *)start);
75
+ #elif defined (__GNUC__)
76
+ __builtin___clear_cache (start, end);
77
+ #else
78
+ #error "Missing builtin to flush instruction cache"
79
+ #endif
80
+ }
81
+
82
+ #endif
83
+
84
+ /* A subroutine of is_vfp_type. Given a structure type, return the type code
85
+ of the first non-structure element. Recurse for structure elements.
86
+ Return -1 if the structure is in fact empty, i.e. no nested elements. */
87
+
88
+ static int
89
+ is_hfa0 (const ffi_type *ty)
90
+ {
91
+ ffi_type **elements = ty->elements;
92
+ int i, ret = -1;
93
+
94
+ if (elements != NULL)
95
+ for (i = 0; elements[i]; ++i)
96
+ {
97
+ ret = elements[i]->type;
98
+ if (ret == FFI_TYPE_STRUCT || ret == FFI_TYPE_COMPLEX)
99
+ {
100
+ ret = is_hfa0 (elements[i]);
101
+ if (ret < 0)
102
+ continue;
103
+ }
104
+ break;
105
+ }
106
+
107
+ return ret;
108
+ }
109
+
110
+ /* A subroutine of is_vfp_type. Given a structure type, return true if all
111
+ of the non-structure elements are the same as CANDIDATE. */
112
+
113
+ static int
114
+ is_hfa1 (const ffi_type *ty, int candidate)
115
+ {
116
+ ffi_type **elements = ty->elements;
117
+ int i;
118
+
119
+ if (elements != NULL)
120
+ for (i = 0; elements[i]; ++i)
121
+ {
122
+ int t = elements[i]->type;
123
+ if (t == FFI_TYPE_STRUCT || t == FFI_TYPE_COMPLEX)
124
+ {
125
+ if (!is_hfa1 (elements[i], candidate))
126
+ return 0;
127
+ }
128
+ else if (t != candidate)
129
+ return 0;
130
+ }
131
+
132
+ return 1;
133
+ }
134
+
135
+ /* Determine if TY may be allocated to the FP registers. This is both an
136
+ fp scalar type as well as an homogenous floating point aggregate (HFA).
137
+ That is, a structure consisting of 1 to 4 members of all the same type,
138
+ where that type is an fp scalar.
139
+
140
+ Returns non-zero iff TY is an HFA. The result is the AARCH64_RET_*
141
+ constant for the type. */
142
+
143
+ static int
144
+ is_vfp_type (const ffi_type *ty)
145
+ {
146
+ ffi_type **elements;
147
+ int candidate, i;
148
+ size_t size, ele_count;
149
+
150
+ /* Quickest tests first. */
151
+ candidate = ty->type;
152
+ switch (candidate)
153
+ {
154
+ default:
155
+ return 0;
156
+ case FFI_TYPE_FLOAT:
157
+ case FFI_TYPE_DOUBLE:
158
+ case FFI_TYPE_LONGDOUBLE:
159
+ ele_count = 1;
160
+ goto done;
161
+ case FFI_TYPE_COMPLEX:
162
+ candidate = ty->elements[0]->type;
163
+ switch (candidate)
164
+ {
165
+ case FFI_TYPE_FLOAT:
166
+ case FFI_TYPE_DOUBLE:
167
+ case FFI_TYPE_LONGDOUBLE:
168
+ ele_count = 2;
169
+ goto done;
170
+ }
171
+ return 0;
172
+ case FFI_TYPE_STRUCT:
173
+ break;
174
+ }
175
+
176
+ /* No HFA types are smaller than 4 bytes, or larger than 64 bytes. */
177
+ size = ty->size;
178
+ if (size < 4 || size > 64)
179
+ return 0;
180
+
181
+ /* Find the type of the first non-structure member. */
182
+ elements = ty->elements;
183
+ candidate = elements[0]->type;
184
+ if (candidate == FFI_TYPE_STRUCT || candidate == FFI_TYPE_COMPLEX)
185
+ {
186
+ for (i = 0; ; ++i)
187
+ {
188
+ candidate = is_hfa0 (elements[i]);
189
+ if (candidate >= 0)
190
+ break;
191
+ }
192
+ }
193
+
194
+ /* If the first member is not a floating point type, it's not an HFA.
195
+ Also quickly re-check the size of the structure. */
196
+ switch (candidate)
197
+ {
198
+ case FFI_TYPE_FLOAT:
199
+ ele_count = size / sizeof(float);
200
+ if (size != ele_count * sizeof(float))
201
+ return 0;
202
+ break;
203
+ case FFI_TYPE_DOUBLE:
204
+ ele_count = size / sizeof(double);
205
+ if (size != ele_count * sizeof(double))
206
+ return 0;
207
+ break;
208
+ case FFI_TYPE_LONGDOUBLE:
209
+ ele_count = size / sizeof(long double);
210
+ if (size != ele_count * sizeof(long double))
211
+ return 0;
212
+ break;
213
+ default:
214
+ return 0;
215
+ }
216
+ if (ele_count > 4)
217
+ return 0;
218
+
219
+ /* Finally, make sure that all scalar elements are the same type. */
220
+ for (i = 0; elements[i]; ++i)
221
+ {
222
+ int t = elements[i]->type;
223
+ if (t == FFI_TYPE_STRUCT || t == FFI_TYPE_COMPLEX)
224
+ {
225
+ if (!is_hfa1 (elements[i], candidate))
226
+ return 0;
227
+ }
228
+ else if (t != candidate)
229
+ return 0;
230
+ }
231
+
232
+ /* All tests succeeded. Encode the result. */
233
+ done:
234
+ return candidate * 4 + (4 - ele_count);
235
+ }
236
+
237
+ /* Representation of the procedure call argument marshalling
238
+ state.
239
+
240
+ The terse state variable names match the names used in the AARCH64
241
+ PCS. */
242
+
243
+ struct arg_state
244
+ {
245
+ unsigned ngrn; /* Next general-purpose register number. */
246
+ unsigned nsrn; /* Next vector register number. */
247
+ size_t nsaa; /* Next stack offset. */
248
+
249
+ #if defined (__APPLE__)
250
+ unsigned allocating_variadic;
251
+ #endif
252
+ };
253
+
254
+ /* Initialize a procedure call argument marshalling state. */
255
+ static void
256
+ arg_init (struct arg_state *state)
257
+ {
258
+ state->ngrn = 0;
259
+ state->nsrn = 0;
260
+ state->nsaa = 0;
261
+ #if defined (__APPLE__)
262
+ state->allocating_variadic = 0;
263
+ #endif
264
+ }
265
+
266
+ /* Allocate an aligned slot on the stack and return a pointer to it. */
267
+ static void *
268
+ allocate_to_stack (struct arg_state *state, void *stack,
269
+ size_t alignment, size_t size)
270
+ {
271
+ size_t nsaa = state->nsaa;
272
+
273
+ /* Round up the NSAA to the larger of 8 or the natural
274
+ alignment of the argument's type. */
275
+ #if defined (__APPLE__)
276
+ if (state->allocating_variadic && alignment < 8)
277
+ alignment = 8;
278
+ #else
279
+ if (alignment < 8)
280
+ alignment = 8;
281
+ #endif
282
+
283
+ nsaa = FFI_ALIGN (nsaa, alignment);
284
+ state->nsaa = nsaa + size;
285
+
286
+ return (char *)stack + nsaa;
287
+ }
288
+
289
+ static ffi_arg
290
+ extend_integer_type (void *source, int type)
291
+ {
292
+ switch (type)
293
+ {
294
+ case FFI_TYPE_UINT8:
295
+ return *(UINT8 *) source;
296
+ case FFI_TYPE_SINT8:
297
+ return *(SINT8 *) source;
298
+ case FFI_TYPE_UINT16:
299
+ return *(UINT16 *) source;
300
+ case FFI_TYPE_SINT16:
301
+ return *(SINT16 *) source;
302
+ case FFI_TYPE_UINT32:
303
+ return *(UINT32 *) source;
304
+ case FFI_TYPE_INT:
305
+ case FFI_TYPE_SINT32:
306
+ return *(SINT32 *) source;
307
+ case FFI_TYPE_UINT64:
308
+ case FFI_TYPE_SINT64:
309
+ return *(UINT64 *) source;
310
+ break;
311
+ case FFI_TYPE_POINTER:
312
+ return *(uintptr_t *) source;
313
+ default:
314
+ abort();
315
+ }
316
+ }
317
+
318
+ static void
319
+ extend_hfa_type (void *dest, void *src, int h)
320
+ {
321
+ ssize_t f = h - AARCH64_RET_S4;
322
+ void *x0;
323
+
324
+ asm volatile (
325
+ "adr %0, 0f\n"
326
+ " add %0, %0, %1\n"
327
+ " br %0\n"
328
+ "0: ldp s16, s17, [%3]\n" /* S4 */
329
+ " ldp s18, s19, [%3, #8]\n"
330
+ " b 4f\n"
331
+ " ldp s16, s17, [%3]\n" /* S3 */
332
+ " ldr s18, [%3, #8]\n"
333
+ " b 3f\n"
334
+ " ldp s16, s17, [%3]\n" /* S2 */
335
+ " b 2f\n"
336
+ " nop\n"
337
+ " ldr s16, [%3]\n" /* S1 */
338
+ " b 1f\n"
339
+ " nop\n"
340
+ " ldp d16, d17, [%3]\n" /* D4 */
341
+ " ldp d18, d19, [%3, #16]\n"
342
+ " b 4f\n"
343
+ " ldp d16, d17, [%3]\n" /* D3 */
344
+ " ldr d18, [%3, #16]\n"
345
+ " b 3f\n"
346
+ " ldp d16, d17, [%3]\n" /* D2 */
347
+ " b 2f\n"
348
+ " nop\n"
349
+ " ldr d16, [%3]\n" /* D1 */
350
+ " b 1f\n"
351
+ " nop\n"
352
+ " ldp q16, q17, [%3]\n" /* Q4 */
353
+ " ldp q18, q19, [%3, #16]\n"
354
+ " b 4f\n"
355
+ " ldp q16, q17, [%3]\n" /* Q3 */
356
+ " ldr q18, [%3, #16]\n"
357
+ " b 3f\n"
358
+ " ldp q16, q17, [%3]\n" /* Q2 */
359
+ " b 2f\n"
360
+ " nop\n"
361
+ " ldr q16, [%3]\n" /* Q1 */
362
+ " b 1f\n"
363
+ "4: str q19, [%2, #48]\n"
364
+ "3: str q18, [%2, #32]\n"
365
+ "2: str q17, [%2, #16]\n"
366
+ "1: str q16, [%2]"
367
+ : "=&r"(x0)
368
+ : "r"(f * 12), "r"(dest), "r"(src)
369
+ : "memory", "v16", "v17", "v18", "v19");
370
+ }
371
+
372
+ static void *
373
+ compress_hfa_type (void *dest, void *reg, int h)
374
+ {
375
+ switch (h)
376
+ {
377
+ case AARCH64_RET_S1:
378
+ if (dest == reg)
379
+ {
380
+ #ifdef __AARCH64EB__
381
+ dest += 12;
382
+ #endif
383
+ }
384
+ else
385
+ *(float *)dest = *(float *)reg;
386
+ break;
387
+ case AARCH64_RET_S2:
388
+ asm ("ldp q16, q17, [%1]\n\t"
389
+ "st2 { v16.s, v17.s }[0], [%0]"
390
+ : : "r"(dest), "r"(reg) : "memory", "v16", "v17");
391
+ break;
392
+ case AARCH64_RET_S3:
393
+ asm ("ldp q16, q17, [%1]\n\t"
394
+ "ldr q18, [%1, #32]\n\t"
395
+ "st3 { v16.s, v17.s, v18.s }[0], [%0]"
396
+ : : "r"(dest), "r"(reg) : "memory", "v16", "v17", "v18");
397
+ break;
398
+ case AARCH64_RET_S4:
399
+ asm ("ldp q16, q17, [%1]\n\t"
400
+ "ldp q18, q19, [%1, #32]\n\t"
401
+ "st4 { v16.s, v17.s, v18.s, v19.s }[0], [%0]"
402
+ : : "r"(dest), "r"(reg) : "memory", "v16", "v17", "v18", "v19");
403
+ break;
404
+
405
+ case AARCH64_RET_D1:
406
+ if (dest == reg)
407
+ {
408
+ #ifdef __AARCH64EB__
409
+ dest += 8;
410
+ #endif
411
+ }
412
+ else
413
+ *(double *)dest = *(double *)reg;
414
+ break;
415
+ case AARCH64_RET_D2:
416
+ asm ("ldp q16, q17, [%1]\n\t"
417
+ "st2 { v16.d, v17.d }[0], [%0]"
418
+ : : "r"(dest), "r"(reg) : "memory", "v16", "v17");
419
+ break;
420
+ case AARCH64_RET_D3:
421
+ asm ("ldp q16, q17, [%1]\n\t"
422
+ "ldr q18, [%1, #32]\n\t"
423
+ "st3 { v16.d, v17.d, v18.d }[0], [%0]"
424
+ : : "r"(dest), "r"(reg) : "memory", "v16", "v17", "v18");
425
+ break;
426
+ case AARCH64_RET_D4:
427
+ asm ("ldp q16, q17, [%1]\n\t"
428
+ "ldp q18, q19, [%1, #32]\n\t"
429
+ "st4 { v16.d, v17.d, v18.d, v19.d }[0], [%0]"
430
+ : : "r"(dest), "r"(reg) : "memory", "v16", "v17", "v18", "v19");
431
+ break;
432
+
433
+ default:
434
+ if (dest != reg)
435
+ return memcpy (dest, reg, 16 * (4 - (h & 3)));
436
+ break;
437
+ }
438
+ return dest;
439
+ }
440
+
441
+ /* Either allocate an appropriate register for the argument type, or if
442
+ none are available, allocate a stack slot and return a pointer
443
+ to the allocated space. */
444
+
445
+ static void *
446
+ allocate_int_to_reg_or_stack (struct call_context *context,
447
+ struct arg_state *state,
448
+ void *stack, size_t size)
449
+ {
450
+ if (state->ngrn < N_X_ARG_REG)
451
+ return &context->x[state->ngrn++];
452
+
453
+ state->ngrn = N_X_ARG_REG;
454
+ return allocate_to_stack (state, stack, size, size);
455
+ }
456
+
457
+ ffi_status
458
+ ffi_prep_cif_machdep (ffi_cif *cif)
459
+ {
460
+ ffi_type *rtype = cif->rtype;
461
+ size_t bytes = cif->bytes;
462
+ int flags, i, n;
463
+
464
+ switch (rtype->type)
465
+ {
466
+ case FFI_TYPE_VOID:
467
+ flags = AARCH64_RET_VOID;
468
+ break;
469
+ case FFI_TYPE_UINT8:
470
+ flags = AARCH64_RET_UINT8;
471
+ break;
472
+ case FFI_TYPE_UINT16:
473
+ flags = AARCH64_RET_UINT16;
474
+ break;
475
+ case FFI_TYPE_UINT32:
476
+ flags = AARCH64_RET_UINT32;
477
+ break;
478
+ case FFI_TYPE_SINT8:
479
+ flags = AARCH64_RET_SINT8;
480
+ break;
481
+ case FFI_TYPE_SINT16:
482
+ flags = AARCH64_RET_SINT16;
483
+ break;
484
+ case FFI_TYPE_INT:
485
+ case FFI_TYPE_SINT32:
486
+ flags = AARCH64_RET_SINT32;
487
+ break;
488
+ case FFI_TYPE_SINT64:
489
+ case FFI_TYPE_UINT64:
490
+ flags = AARCH64_RET_INT64;
491
+ break;
492
+ case FFI_TYPE_POINTER:
493
+ flags = (sizeof(void *) == 4 ? AARCH64_RET_UINT32 : AARCH64_RET_INT64);
494
+ break;
495
+
496
+ case FFI_TYPE_FLOAT:
497
+ case FFI_TYPE_DOUBLE:
498
+ case FFI_TYPE_LONGDOUBLE:
499
+ case FFI_TYPE_STRUCT:
500
+ case FFI_TYPE_COMPLEX:
501
+ flags = is_vfp_type (rtype);
502
+ if (flags == 0)
503
+ {
504
+ size_t s = rtype->size;
505
+ if (s > 16)
506
+ {
507
+ flags = AARCH64_RET_VOID | AARCH64_RET_IN_MEM;
508
+ bytes += 8;
509
+ }
510
+ else if (s == 16)
511
+ flags = AARCH64_RET_INT128;
512
+ else if (s == 8)
513
+ flags = AARCH64_RET_INT64;
514
+ else
515
+ flags = AARCH64_RET_INT128 | AARCH64_RET_NEED_COPY;
516
+ }
517
+ break;
518
+
519
+ default:
520
+ abort();
521
+ }
522
+
523
+ for (i = 0, n = cif->nargs; i < n; i++)
524
+ if (is_vfp_type (cif->arg_types[i]))
525
+ {
526
+ flags |= AARCH64_FLAG_ARG_V;
527
+ break;
528
+ }
529
+
530
+ /* Round the stack up to a multiple of the stack alignment requirement. */
531
+ cif->bytes = FFI_ALIGN(bytes, 16);
532
+ cif->flags = flags;
533
+ #if defined (__APPLE__)
534
+ cif->aarch64_nfixedargs = 0;
535
+ #endif
536
+
537
+ return FFI_OK;
538
+ }
539
+
540
+ #if defined (__APPLE__)
541
+ /* Perform Apple-specific cif processing for variadic calls */
542
+ ffi_status ffi_prep_cif_machdep_var(ffi_cif *cif,
543
+ unsigned int nfixedargs,
544
+ unsigned int ntotalargs)
545
+ {
546
+ ffi_status status = ffi_prep_cif_machdep (cif);
547
+ cif->aarch64_nfixedargs = nfixedargs;
548
+ return status;
549
+ }
550
+ #endif /* __APPLE__ */
551
+
552
+ extern void ffi_call_SYSV (struct call_context *context, void *frame,
553
+ void (*fn)(void), void *rvalue, int flags,
554
+ void *closure) FFI_HIDDEN;
555
+
556
+ /* Call a function with the provided arguments and capture the return
557
+ value. */
558
+ static void
559
+ ffi_call_int (ffi_cif *cif, void (*fn)(void), void *orig_rvalue,
560
+ void **avalue, void *closure)
561
+ {
562
+ struct call_context *context;
563
+ void *stack, *frame, *rvalue;
564
+ struct arg_state state;
565
+ size_t stack_bytes, rtype_size, rsize;
566
+ int i, nargs, flags;
567
+ ffi_type *rtype;
568
+
569
+ flags = cif->flags;
570
+ rtype = cif->rtype;
571
+ rtype_size = rtype->size;
572
+ stack_bytes = cif->bytes;
573
+
574
+ /* If the target function returns a structure via hidden pointer,
575
+ then we cannot allow a null rvalue. Otherwise, mash a null
576
+ rvalue to void return type. */
577
+ rsize = 0;
578
+ if (flags & AARCH64_RET_IN_MEM)
579
+ {
580
+ if (orig_rvalue == NULL)
581
+ rsize = rtype_size;
582
+ }
583
+ else if (orig_rvalue == NULL)
584
+ flags &= AARCH64_FLAG_ARG_V;
585
+ else if (flags & AARCH64_RET_NEED_COPY)
586
+ rsize = 16;
587
+
588
+ /* Allocate consectutive stack for everything we'll need. */
589
+ context = alloca (sizeof(struct call_context) + stack_bytes + 32 + rsize);
590
+ stack = context + 1;
591
+ frame = stack + stack_bytes;
592
+ rvalue = (rsize ? frame + 32 : orig_rvalue);
593
+
594
+ arg_init (&state);
595
+ for (i = 0, nargs = cif->nargs; i < nargs; i++)
596
+ {
597
+ ffi_type *ty = cif->arg_types[i];
598
+ size_t s = ty->size;
599
+ void *a = avalue[i];
600
+ int h, t;
601
+
602
+ t = ty->type;
603
+ switch (t)
604
+ {
605
+ case FFI_TYPE_VOID:
606
+ FFI_ASSERT (0);
607
+ break;
608
+
609
+ /* If the argument is a basic type the argument is allocated to an
610
+ appropriate register, or if none are available, to the stack. */
611
+ case FFI_TYPE_INT:
612
+ case FFI_TYPE_UINT8:
613
+ case FFI_TYPE_SINT8:
614
+ case FFI_TYPE_UINT16:
615
+ case FFI_TYPE_SINT16:
616
+ case FFI_TYPE_UINT32:
617
+ case FFI_TYPE_SINT32:
618
+ case FFI_TYPE_UINT64:
619
+ case FFI_TYPE_SINT64:
620
+ case FFI_TYPE_POINTER:
621
+ do_pointer:
622
+ {
623
+ ffi_arg ext = extend_integer_type (a, t);
624
+ if (state.ngrn < N_X_ARG_REG)
625
+ context->x[state.ngrn++] = ext;
626
+ else
627
+ {
628
+ void *d = allocate_to_stack (&state, stack, ty->alignment, s);
629
+ state.ngrn = N_X_ARG_REG;
630
+ /* Note that the default abi extends each argument
631
+ to a full 64-bit slot, while the iOS abi allocates
632
+ only enough space. */
633
+ #ifdef __APPLE__
634
+ memcpy(d, a, s);
635
+ #else
636
+ *(ffi_arg *)d = ext;
637
+ #endif
638
+ }
639
+ }
640
+ break;
641
+
642
+ case FFI_TYPE_FLOAT:
643
+ case FFI_TYPE_DOUBLE:
644
+ case FFI_TYPE_LONGDOUBLE:
645
+ case FFI_TYPE_STRUCT:
646
+ case FFI_TYPE_COMPLEX:
647
+ {
648
+ void *dest;
649
+
650
+ h = is_vfp_type (ty);
651
+ if (h)
652
+ {
653
+ int elems = 4 - (h & 3);
654
+ if (state.nsrn + elems <= N_V_ARG_REG)
655
+ {
656
+ dest = &context->v[state.nsrn];
657
+ state.nsrn += elems;
658
+ extend_hfa_type (dest, a, h);
659
+ break;
660
+ }
661
+ state.nsrn = N_V_ARG_REG;
662
+ dest = allocate_to_stack (&state, stack, ty->alignment, s);
663
+ }
664
+ else if (s > 16)
665
+ {
666
+ /* If the argument is a composite type that is larger than 16
667
+ bytes, then the argument has been copied to memory, and
668
+ the argument is replaced by a pointer to the copy. */
669
+ a = &avalue[i];
670
+ t = FFI_TYPE_POINTER;
671
+ s = sizeof (void *);
672
+ goto do_pointer;
673
+ }
674
+ else
675
+ {
676
+ size_t n = (s + 7) / 8;
677
+ if (state.ngrn + n <= N_X_ARG_REG)
678
+ {
679
+ /* If the argument is a composite type and the size in
680
+ double-words is not more than the number of available
681
+ X registers, then the argument is copied into
682
+ consecutive X registers. */
683
+ dest = &context->x[state.ngrn];
684
+ state.ngrn += n;
685
+ }
686
+ else
687
+ {
688
+ /* Otherwise, there are insufficient X registers. Further
689
+ X register allocations are prevented, the NSAA is
690
+ adjusted and the argument is copied to memory at the
691
+ adjusted NSAA. */
692
+ state.ngrn = N_X_ARG_REG;
693
+ dest = allocate_to_stack (&state, stack, ty->alignment, s);
694
+ }
695
+ }
696
+ memcpy (dest, a, s);
697
+ }
698
+ break;
699
+
700
+ default:
701
+ abort();
702
+ }
703
+
704
+ #if defined (__APPLE__)
705
+ if (i + 1 == cif->aarch64_nfixedargs)
706
+ {
707
+ state.ngrn = N_X_ARG_REG;
708
+ state.nsrn = N_V_ARG_REG;
709
+ state.allocating_variadic = 1;
710
+ }
711
+ #endif
712
+ }
713
+
714
+ ffi_call_SYSV (context, frame, fn, rvalue, flags, closure);
715
+
716
+ if (flags & AARCH64_RET_NEED_COPY)
717
+ memcpy (orig_rvalue, rvalue, rtype_size);
718
+ }
719
+
720
+ void
721
+ ffi_call (ffi_cif *cif, void (*fn) (void), void *rvalue, void **avalue)
722
+ {
723
+ ffi_call_int (cif, fn, rvalue, avalue, NULL);
724
+ }
725
+
726
+ #ifdef FFI_GO_CLOSURES
727
+ void
728
+ ffi_call_go (ffi_cif *cif, void (*fn) (void), void *rvalue,
729
+ void **avalue, void *closure)
730
+ {
731
+ ffi_call_int (cif, fn, rvalue, avalue, closure);
732
+ }
733
+ #endif /* FFI_GO_CLOSURES */
734
+
735
+ /* Build a trampoline. */
736
+
737
+ extern void ffi_closure_SYSV (void) FFI_HIDDEN;
738
+ extern void ffi_closure_SYSV_V (void) FFI_HIDDEN;
739
+
740
+ ffi_status
741
+ ffi_prep_closure_loc (ffi_closure *closure,
742
+ ffi_cif* cif,
743
+ void (*fun)(ffi_cif*,void*,void**,void*),
744
+ void *user_data,
745
+ void *codeloc)
746
+ {
747
+ if (cif->abi != FFI_SYSV)
748
+ return FFI_BAD_ABI;
749
+
750
+ void (*start)(void);
751
+
752
+ if (cif->flags & AARCH64_FLAG_ARG_V)
753
+ start = ffi_closure_SYSV_V;
754
+ else
755
+ start = ffi_closure_SYSV;
756
+
757
+ #if FFI_EXEC_TRAMPOLINE_TABLE
758
+ #ifdef __MACH__
759
+ void **config = (void **)((uint8_t *)codeloc - PAGE_MAX_SIZE);
760
+ config[0] = closure;
761
+ config[1] = start;
762
+ #endif
763
+ #else
764
+ static const unsigned char trampoline[16] = {
765
+ 0x90, 0x00, 0x00, 0x58, /* ldr x16, tramp+16 */
766
+ 0xf1, 0xff, 0xff, 0x10, /* adr x17, tramp+0 */
767
+ 0x00, 0x02, 0x1f, 0xd6 /* br x16 */
768
+ };
769
+ char *tramp = closure->tramp;
770
+
771
+ memcpy (tramp, trampoline, sizeof(trampoline));
772
+
773
+ *(UINT64 *)(tramp + 16) = (uintptr_t)start;
774
+
775
+ ffi_clear_cache(tramp, tramp + FFI_TRAMPOLINE_SIZE);
776
+ #endif
777
+
778
+ closure->cif = cif;
779
+ closure->fun = fun;
780
+ closure->user_data = user_data;
781
+
782
+ return FFI_OK;
783
+ }
784
+
785
+ #ifdef FFI_GO_CLOSURES
786
+ extern void ffi_go_closure_SYSV (void) FFI_HIDDEN;
787
+ extern void ffi_go_closure_SYSV_V (void) FFI_HIDDEN;
788
+
789
+ ffi_status
790
+ ffi_prep_go_closure (ffi_go_closure *closure, ffi_cif* cif,
791
+ void (*fun)(ffi_cif*,void*,void**,void*))
792
+ {
793
+ void (*start)(void);
794
+
795
+ if (cif->abi != FFI_SYSV)
796
+ return FFI_BAD_ABI;
797
+
798
+ if (cif->flags & AARCH64_FLAG_ARG_V)
799
+ start = ffi_go_closure_SYSV_V;
800
+ else
801
+ start = ffi_go_closure_SYSV;
802
+
803
+ closure->tramp = start;
804
+ closure->cif = cif;
805
+ closure->fun = fun;
806
+
807
+ return FFI_OK;
808
+ }
809
+ #endif /* FFI_GO_CLOSURES */
810
+
811
+ /* Primary handler to setup and invoke a function within a closure.
812
+
813
+ A closure when invoked enters via the assembler wrapper
814
+ ffi_closure_SYSV(). The wrapper allocates a call context on the
815
+ stack, saves the interesting registers (from the perspective of
816
+ the calling convention) into the context then passes control to
817
+ ffi_closure_SYSV_inner() passing the saved context and a pointer to
818
+ the stack at the point ffi_closure_SYSV() was invoked.
819
+
820
+ On the return path the assembler wrapper will reload call context
821
+ registers.
822
+
823
+ ffi_closure_SYSV_inner() marshalls the call context into ffi value
824
+ descriptors, invokes the wrapped function, then marshalls the return
825
+ value back into the call context. */
826
+
827
+ int FFI_HIDDEN
828
+ ffi_closure_SYSV_inner (ffi_cif *cif,
829
+ void (*fun)(ffi_cif*,void*,void**,void*),
830
+ void *user_data,
831
+ struct call_context *context,
832
+ void *stack, void *rvalue, void *struct_rvalue)
833
+ {
834
+ void **avalue = (void**) alloca (cif->nargs * sizeof (void*));
835
+ int i, h, nargs, flags;
836
+ struct arg_state state;
837
+
838
+ arg_init (&state);
839
+
840
+ for (i = 0, nargs = cif->nargs; i < nargs; i++)
841
+ {
842
+ ffi_type *ty = cif->arg_types[i];
843
+ int t = ty->type;
844
+ size_t n, s = ty->size;
845
+
846
+ switch (t)
847
+ {
848
+ case FFI_TYPE_VOID:
849
+ FFI_ASSERT (0);
850
+ break;
851
+
852
+ case FFI_TYPE_INT:
853
+ case FFI_TYPE_UINT8:
854
+ case FFI_TYPE_SINT8:
855
+ case FFI_TYPE_UINT16:
856
+ case FFI_TYPE_SINT16:
857
+ case FFI_TYPE_UINT32:
858
+ case FFI_TYPE_SINT32:
859
+ case FFI_TYPE_UINT64:
860
+ case FFI_TYPE_SINT64:
861
+ case FFI_TYPE_POINTER:
862
+ avalue[i] = allocate_int_to_reg_or_stack (context, &state, stack, s);
863
+ break;
864
+
865
+ case FFI_TYPE_FLOAT:
866
+ case FFI_TYPE_DOUBLE:
867
+ case FFI_TYPE_LONGDOUBLE:
868
+ case FFI_TYPE_STRUCT:
869
+ case FFI_TYPE_COMPLEX:
870
+ h = is_vfp_type (ty);
871
+ if (h)
872
+ {
873
+ n = 4 - (h & 3);
874
+ if (state.nsrn + n <= N_V_ARG_REG)
875
+ {
876
+ void *reg = &context->v[state.nsrn];
877
+ state.nsrn += n;
878
+
879
+ /* Eeek! We need a pointer to the structure, however the
880
+ homogeneous float elements are being passed in individual
881
+ registers, therefore for float and double the structure
882
+ is not represented as a contiguous sequence of bytes in
883
+ our saved register context. We don't need the original
884
+ contents of the register storage, so we reformat the
885
+ structure into the same memory. */
886
+ avalue[i] = compress_hfa_type (reg, reg, h);
887
+ }
888
+ else
889
+ {
890
+ state.nsrn = N_V_ARG_REG;
891
+ avalue[i] = allocate_to_stack (&state, stack,
892
+ ty->alignment, s);
893
+ }
894
+ }
895
+ else if (s > 16)
896
+ {
897
+ /* Replace Composite type of size greater than 16 with a
898
+ pointer. */
899
+ avalue[i] = *(void **)
900
+ allocate_int_to_reg_or_stack (context, &state, stack,
901
+ sizeof (void *));
902
+ }
903
+ else
904
+ {
905
+ n = (s + 7) / 8;
906
+ if (state.ngrn + n <= N_X_ARG_REG)
907
+ {
908
+ avalue[i] = &context->x[state.ngrn];
909
+ state.ngrn += n;
910
+ }
911
+ else
912
+ {
913
+ state.ngrn = N_X_ARG_REG;
914
+ avalue[i] = allocate_to_stack (&state, stack,
915
+ ty->alignment, s);
916
+ }
917
+ }
918
+ break;
919
+
920
+ default:
921
+ abort();
922
+ }
923
+
924
+ #if defined (__APPLE__)
925
+ if (i + 1 == cif->aarch64_nfixedargs)
926
+ {
927
+ state.ngrn = N_X_ARG_REG;
928
+ state.nsrn = N_V_ARG_REG;
929
+ state.allocating_variadic = 1;
930
+ }
931
+ #endif
932
+ }
933
+
934
+ flags = cif->flags;
935
+ if (flags & AARCH64_RET_IN_MEM)
936
+ rvalue = struct_rvalue;
937
+
938
+ fun (cif, rvalue, avalue, user_data);
939
+
940
+ return flags;
941
+ }