ffi 0.5.0 → 0.6.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (212) hide show
  1. data/LICENSE +1 -27
  2. data/Rakefile +9 -12
  3. data/ext/ffi_c/AbstractMemory.c +11 -8
  4. data/ext/ffi_c/AbstractMemory.h +23 -21
  5. data/ext/ffi_c/AutoPointer.c +0 -1
  6. data/ext/ffi_c/Buffer.c +24 -8
  7. data/ext/ffi_c/Call.c +28 -0
  8. data/ext/ffi_c/Call.h +8 -4
  9. data/ext/ffi_c/ClosurePool.c +7 -5
  10. data/ext/ffi_c/DynamicLibrary.c +0 -1
  11. data/ext/ffi_c/Function.c +19 -2
  12. data/ext/ffi_c/MemoryPointer.c +3 -3
  13. data/ext/ffi_c/MethodHandle.c +1 -1
  14. data/ext/ffi_c/Pointer.c +23 -9
  15. data/ext/ffi_c/Struct.c +142 -69
  16. data/ext/ffi_c/Struct.h +16 -7
  17. data/ext/ffi_c/StructLayout.c +92 -55
  18. data/ext/ffi_c/Type.c +5 -22
  19. data/ext/ffi_c/Type.h +1 -1
  20. data/ext/ffi_c/Types.c +8 -2
  21. data/ext/ffi_c/Types.h +2 -0
  22. data/ext/ffi_c/Variadic.c +7 -19
  23. data/ext/ffi_c/endian.h +1 -1
  24. data/ext/ffi_c/extconf.rb +20 -11
  25. data/ext/ffi_c/libffi/ChangeLog +900 -84
  26. data/ext/ffi_c/libffi/ChangeLog.libffi +311 -0
  27. data/ext/ffi_c/libffi/LICENSE +1 -1
  28. data/ext/ffi_c/libffi/Makefile.am +14 -4
  29. data/ext/ffi_c/libffi/Makefile.in +362 -211
  30. data/ext/ffi_c/libffi/README +70 -92
  31. data/ext/ffi_c/libffi/aclocal.m4 +6068 -4586
  32. data/ext/ffi_c/libffi/config.guess +125 -143
  33. data/ext/ffi_c/libffi/config.sub +103 -27
  34. data/ext/ffi_c/libffi/configure +11364 -18497
  35. data/ext/ffi_c/libffi/configure.ac +43 -4
  36. data/ext/ffi_c/libffi/doc/libffi.info +15 -15
  37. data/ext/ffi_c/libffi/doc/libffi.texi +1 -1
  38. data/ext/ffi_c/libffi/doc/stamp-vti +4 -4
  39. data/ext/ffi_c/libffi/doc/version.texi +4 -4
  40. data/ext/ffi_c/libffi/fficonfig.h.in +24 -3
  41. data/ext/ffi_c/libffi/include/Makefile.am +1 -1
  42. data/ext/ffi_c/libffi/include/Makefile.in +97 -50
  43. data/ext/ffi_c/libffi/include/ffi.h.in +8 -2
  44. data/ext/ffi_c/libffi/include/ffi_common.h +24 -0
  45. data/ext/ffi_c/libffi/libtool-version +1 -1
  46. data/ext/ffi_c/libffi/ltmain.sh +7346 -5870
  47. data/ext/ffi_c/libffi/m4/libtool.m4 +7360 -0
  48. data/ext/ffi_c/libffi/m4/ltoptions.m4 +368 -0
  49. data/ext/ffi_c/libffi/m4/ltsugar.m4 +123 -0
  50. data/ext/ffi_c/libffi/m4/ltversion.m4 +23 -0
  51. data/ext/ffi_c/libffi/m4/lt~obsolete.m4 +92 -0
  52. data/ext/ffi_c/libffi/man/Makefile.in +115 -62
  53. data/ext/ffi_c/libffi/man/ffi_call.3 +3 -3
  54. data/ext/ffi_c/libffi/missing +15 -8
  55. data/ext/ffi_c/libffi/src/arm/sysv.S +15 -8
  56. data/ext/ffi_c/libffi/src/avr32/ffi.c +421 -0
  57. data/ext/ffi_c/libffi/src/avr32/ffitarget.h +50 -0
  58. data/ext/ffi_c/libffi/src/avr32/sysv.S +208 -0
  59. data/ext/ffi_c/libffi/src/closures.c +47 -10
  60. data/ext/ffi_c/libffi/src/frv/ffi.c +1 -1
  61. data/ext/ffi_c/libffi/src/java_raw_api.c +0 -3
  62. data/ext/ffi_c/libffi/src/mips/ffi.c +135 -32
  63. data/ext/ffi_c/libffi/src/mips/ffitarget.h +37 -4
  64. data/ext/ffi_c/libffi/src/mips/n32.S +67 -10
  65. data/ext/ffi_c/libffi/src/mips/o32.S +8 -8
  66. data/ext/ffi_c/libffi/src/pa/ffi.c +7 -0
  67. data/ext/ffi_c/libffi/src/powerpc/aix.S +163 -64
  68. data/ext/ffi_c/libffi/src/powerpc/aix_closure.S +308 -112
  69. data/ext/ffi_c/libffi/src/powerpc/ffi.c +20 -7
  70. data/ext/ffi_c/libffi/src/powerpc/ffi_darwin.c +208 -80
  71. data/ext/ffi_c/libffi/src/powerpc/ffitarget.h +11 -3
  72. data/ext/ffi_c/libffi/src/powerpc/sysv.S +12 -23
  73. data/ext/ffi_c/libffi/src/s390/sysv.S +1 -1
  74. data/ext/ffi_c/libffi/src/sh/sysv.S +9 -9
  75. data/ext/ffi_c/libffi/src/sh64/ffi.c +37 -22
  76. data/ext/ffi_c/libffi/src/sh64/sysv.S +23 -14
  77. data/ext/ffi_c/libffi/src/sparc/ffi.c +21 -6
  78. data/ext/ffi_c/libffi/src/sparc/v8.S +55 -14
  79. data/ext/ffi_c/libffi/src/x86/darwin.S +10 -9
  80. data/ext/ffi_c/libffi/src/x86/ffi.c +293 -86
  81. data/ext/ffi_c/libffi/src/x86/ffi64.c +73 -19
  82. data/ext/ffi_c/libffi/src/x86/ffitarget.h +30 -0
  83. data/ext/ffi_c/libffi/src/x86/sysv.S +21 -4
  84. data/ext/ffi_c/libffi/src/x86/unix64.S +8 -4
  85. data/ext/ffi_c/libffi/src/x86/win32.S +633 -147
  86. data/ext/ffi_c/libffi/src/x86/win64.S +460 -0
  87. data/ext/ffi_c/libffi/testsuite/Makefile.am +63 -54
  88. data/ext/ffi_c/libffi/testsuite/Makefile.in +112 -77
  89. data/ext/ffi_c/libffi/testsuite/lib/libffi-dg.exp +12 -1
  90. data/ext/ffi_c/libffi/testsuite/libffi.call/call.exp +4 -4
  91. data/ext/ffi_c/libffi/testsuite/libffi.call/closure_fn0.c +7 -15
  92. data/ext/ffi_c/libffi/testsuite/libffi.call/closure_fn1.c +7 -15
  93. data/ext/ffi_c/libffi/testsuite/libffi.call/closure_fn2.c +7 -15
  94. data/ext/ffi_c/libffi/testsuite/libffi.call/closure_fn3.c +7 -15
  95. data/ext/ffi_c/libffi/testsuite/libffi.call/closure_fn4.c +7 -15
  96. data/ext/ffi_c/libffi/testsuite/libffi.call/closure_fn5.c +7 -14
  97. data/ext/ffi_c/libffi/testsuite/libffi.call/closure_fn6.c +7 -15
  98. data/ext/ffi_c/libffi/testsuite/libffi.call/closure_loc_fn0.c +95 -0
  99. data/ext/ffi_c/libffi/testsuite/libffi.call/closure_stdcall.c +6 -14
  100. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_12byte.c +4 -12
  101. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_16byte.c +4 -12
  102. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_18byte.c +4 -12
  103. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_19byte.c +4 -12
  104. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_1_1byte.c +4 -12
  105. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_20byte.c +4 -12
  106. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_20byte1.c +4 -12
  107. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_24byte.c +4 -12
  108. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_2byte.c +4 -12
  109. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_3_1byte.c +4 -12
  110. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_3byte1.c +4 -12
  111. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_3byte2.c +4 -12
  112. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_4_1byte.c +4 -12
  113. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_4byte.c +4 -12
  114. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_5_1_byte.c +4 -12
  115. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_5byte.c +4 -12
  116. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_64byte.c +4 -12
  117. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_6_1_byte.c +4 -12
  118. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_6byte.c +4 -12
  119. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_7_1_byte.c +4 -12
  120. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_7byte.c +4 -12
  121. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_8byte.c +4 -12
  122. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_9byte1.c +4 -12
  123. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_9byte2.c +4 -12
  124. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_double.c +4 -12
  125. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_float.c +4 -12
  126. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_longdouble.c +4 -12
  127. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_longdouble_split.c +134 -0
  128. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_longdouble_split2.c +117 -0
  129. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_pointer.c +11 -17
  130. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_sint16.c +4 -12
  131. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_sint32.c +4 -12
  132. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_sint64.c +7 -15
  133. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_uint16.c +4 -12
  134. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_uint32.c +4 -12
  135. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_align_uint64.c +7 -15
  136. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_dbls_struct.c +66 -0
  137. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_double.c +4 -12
  138. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_double_va.c +57 -0
  139. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_float.c +4 -13
  140. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_longdouble.c +105 -0
  141. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_longdouble_va.c +57 -0
  142. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_multi_schar.c +4 -12
  143. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_multi_sshort.c +4 -12
  144. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_multi_sshortchar.c +4 -12
  145. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_multi_uchar.c +4 -12
  146. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_multi_ushort.c +4 -12
  147. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_multi_ushortchar.c +4 -12
  148. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_pointer.c +74 -0
  149. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_pointer_stack.c +140 -0
  150. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_schar.c +4 -12
  151. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_sint.c +4 -12
  152. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_sshort.c +4 -12
  153. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_uchar.c +4 -12
  154. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_uint.c +4 -12
  155. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_ulonglong.c +8 -16
  156. data/ext/ffi_c/libffi/testsuite/libffi.call/cls_ushort.c +4 -12
  157. data/ext/ffi_c/libffi/testsuite/libffi.call/err_bad_abi.c +37 -0
  158. data/ext/ffi_c/libffi/testsuite/libffi.call/err_bad_typedef.c +25 -0
  159. data/ext/ffi_c/libffi/testsuite/libffi.call/ffitest.h +31 -0
  160. data/ext/ffi_c/libffi/testsuite/libffi.call/float2.c +2 -1
  161. data/ext/ffi_c/libffi/testsuite/libffi.call/huge_struct.c +342 -0
  162. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct.c +4 -12
  163. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct1.c +4 -12
  164. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct10.c +4 -12
  165. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct2.c +4 -12
  166. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct3.c +4 -12
  167. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct4.c +4 -12
  168. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct5.c +4 -12
  169. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct6.c +4 -12
  170. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct7.c +4 -12
  171. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct8.c +4 -12
  172. data/ext/ffi_c/libffi/testsuite/libffi.call/nested_struct9.c +4 -12
  173. data/ext/ffi_c/libffi/testsuite/libffi.call/problem1.c +4 -12
  174. data/ext/ffi_c/libffi/testsuite/libffi.call/return_ldl.c +1 -1
  175. data/ext/ffi_c/libffi/testsuite/libffi.call/return_ll1.c +1 -1
  176. data/ext/ffi_c/libffi/testsuite/libffi.call/stret_large.c +145 -0
  177. data/ext/ffi_c/libffi/testsuite/libffi.call/stret_large2.c +148 -0
  178. data/ext/ffi_c/libffi/testsuite/libffi.call/stret_medium.c +124 -0
  179. data/ext/ffi_c/libffi/testsuite/libffi.call/stret_medium2.c +124 -0
  180. data/ext/ffi_c/libffi/testsuite/libffi.call/testclosure.c +70 -0
  181. data/ext/ffi_c/libffi/testsuite/libffi.special/ffitestcxx.h +10 -0
  182. data/ext/ffi_c/libffi/testsuite/libffi.special/special.exp +4 -5
  183. data/ext/ffi_c/libffi/testsuite/libffi.special/unwindtest.cc +17 -16
  184. data/ext/ffi_c/libffi/texinfo.tex +155 -427
  185. data/ext/ffi_c/libffi.bsd.mk +1 -1
  186. data/lib/ffi/autopointer.rb +79 -20
  187. data/lib/ffi/buffer.rb +4 -0
  188. data/lib/ffi/callback.rb +4 -10
  189. data/lib/ffi/enum.rb +28 -0
  190. data/lib/ffi/ffi.rb +0 -1
  191. data/lib/ffi/io.rb +28 -0
  192. data/lib/ffi/library.rb +237 -182
  193. data/lib/ffi/memorypointer.rb +28 -62
  194. data/lib/ffi/platform.rb +27 -0
  195. data/lib/ffi/pointer.rb +28 -0
  196. data/lib/ffi/struct.rb +55 -1
  197. data/lib/ffi/types.rb +29 -0
  198. data/lib/ffi/variadic.rb +29 -0
  199. data/spec/ffi/library_spec.rb +31 -5
  200. data/spec/ffi/managed_struct_spec.rb +1 -1
  201. data/spec/ffi/pointer_spec.rb +1 -1
  202. data/spec/ffi/rbx/attach_function_spec.rb +2 -1
  203. data/spec/ffi/rbx/memory_pointer_spec.rb +2 -1
  204. data/spec/ffi/spec_helper.rb +5 -1
  205. data/spec/ffi/struct_spec.rb +77 -0
  206. metadata +28 -18
  207. data/ext/ffi_c/libffi/TODO +0 -1
  208. data/ext/ffi_c/libffi/ltcf-c.sh +0 -861
  209. data/ext/ffi_c/libffi/ltcf-cxx.sh +0 -1069
  210. data/ext/ffi_c/libffi/ltcf-gcj.sh +0 -700
  211. data/ext/ffi_c/libffi/ltconfig +0 -2862
  212. data/ext/ffi_c/libffi/mkinstalldirs +0 -158
@@ -0,0 +1,421 @@
1
+ /* -----------------------------------------------------------------------
2
+ ffi.c - Copyright (c) 2009 Bradley Smith <brad@brad-smith.co.uk>
3
+
4
+ AVR32 Foreign Function Interface
5
+
6
+ Permission is hereby granted, free of charge, to any person obtaining
7
+ a copy of this software and associated documentation files (the
8
+ ``Software''), to deal in the Software without restriction, including
9
+ without limitation the rights to use, copy, modify, merge, publish,
10
+ distribute, sublicense, and/or sell copies of the Software, and to
11
+ permit persons to whom the Software is furnished to do so, subject to
12
+ the following conditions:
13
+
14
+ The above copyright notice and this permission notice shall be included
15
+ in all copies or substantial portions of the Software.
16
+
17
+ THE SOFTWARE IS PROVIDED ``AS IS'', WITHOUT WARRANTY OF ANY KIND,
18
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
19
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
20
+ NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
21
+ HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
22
+ WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
23
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
24
+ DEALINGS IN THE SOFTWARE.
25
+ ----------------------------------------------------------------------- */
26
+
27
+ #include <ffi.h>
28
+ #include <ffi_common.h>
29
+
30
+ #include <stdlib.h>
31
+ #include <stdio.h>
32
+ #include <unistd.h>
33
+ #include <asm/unistd.h>
34
+
35
+ /* #define DEBUG */
36
+
37
+ extern void ffi_call_SYSV(void (*)(char *, extended_cif *), extended_cif *,
38
+ unsigned int, unsigned int, unsigned int*, unsigned int,
39
+ void (*fn)(void));
40
+ extern void ffi_closure_SYSV (ffi_closure *);
41
+
42
+ unsigned int pass_struct_on_stack(ffi_type *type)
43
+ {
44
+ if(type->type != FFI_TYPE_STRUCT)
45
+ return 0;
46
+
47
+ if(type->alignment < type->size &&
48
+ !(type->size == 4 || type->size == 8) &&
49
+ !(type->size == 8 && type->alignment >= 4))
50
+ return 1;
51
+
52
+ if(type->size == 3 || type->size == 5 || type->size == 6 ||
53
+ type->size == 7)
54
+ return 1;
55
+
56
+ return 0;
57
+ }
58
+
59
+ /* ffi_prep_args is called by the assembly routine once stack space
60
+ * has been allocated for the function's arguments
61
+ *
62
+ * This is annoyingly complex since we need to keep track of used
63
+ * registers.
64
+ */
65
+
66
+ void ffi_prep_args(char *stack, extended_cif *ecif)
67
+ {
68
+ unsigned int i;
69
+ void **p_argv;
70
+ ffi_type **p_arg;
71
+ char *reg_base = stack;
72
+ char *stack_base = stack + 20;
73
+ unsigned int stack_offset = 0;
74
+ unsigned int reg_mask = 0;
75
+
76
+ p_argv = ecif->avalue;
77
+
78
+ /* If cif->flags is struct then we know it's not passed in registers */
79
+ if(ecif->cif->flags == FFI_TYPE_STRUCT)
80
+ {
81
+ *(void**)reg_base = ecif->rvalue;
82
+ reg_mask |= 1;
83
+ }
84
+
85
+ for(i = 0, p_arg = ecif->cif->arg_types; i < ecif->cif->nargs;
86
+ i++, p_arg++)
87
+ {
88
+ size_t z = (*p_arg)->size;
89
+ int alignment = (*p_arg)->alignment;
90
+ int type = (*p_arg)->type;
91
+ char *addr = 0;
92
+
93
+ if(z % 4 != 0)
94
+ z += (4 - z % 4);
95
+
96
+ if(reg_mask != 0x1f)
97
+ {
98
+ if(pass_struct_on_stack(*p_arg))
99
+ {
100
+ addr = stack_base + stack_offset;
101
+ stack_offset += z;
102
+ }
103
+ else if(z == sizeof(int))
104
+ {
105
+ char index = 0;
106
+
107
+ while((reg_mask >> index) & 1)
108
+ index++;
109
+
110
+ addr = reg_base + (index * 4);
111
+ reg_mask |= (1 << index);
112
+ }
113
+ else if(z == 2 * sizeof(int))
114
+ {
115
+ if(!((reg_mask >> 1) & 1))
116
+ {
117
+ addr = reg_base + 4;
118
+ reg_mask |= (3 << 1);
119
+ }
120
+ else if(!((reg_mask >> 3) & 1))
121
+ {
122
+ addr = reg_base + 12;
123
+ reg_mask |= (3 << 3);
124
+ }
125
+ }
126
+ }
127
+
128
+ if(!addr)
129
+ {
130
+ addr = stack_base + stack_offset;
131
+ stack_offset += z;
132
+ }
133
+
134
+ if(type == FFI_TYPE_STRUCT && (*p_arg)->elements[1] == NULL)
135
+ type = (*p_arg)->elements[0]->type;
136
+
137
+ switch(type)
138
+ {
139
+ case FFI_TYPE_UINT8:
140
+ *(unsigned int *)addr = (unsigned int)*(UINT8 *)(*p_argv);
141
+ break;
142
+ case FFI_TYPE_SINT8:
143
+ *(signed int *)addr = (signed int)*(SINT8 *)(*p_argv);
144
+ break;
145
+ case FFI_TYPE_UINT16:
146
+ *(unsigned int *)addr = (unsigned int)*(UINT16 *)(*p_argv);
147
+ break;
148
+ case FFI_TYPE_SINT16:
149
+ *(signed int *)addr = (signed int)*(SINT16 *)(*p_argv);
150
+ break;
151
+ default:
152
+ memcpy(addr, *p_argv, z);
153
+ }
154
+
155
+ p_argv++;
156
+ }
157
+
158
+ #ifdef DEBUG
159
+ /* Debugging */
160
+ for(i = 0; i < 5; i++)
161
+ {
162
+ if((reg_mask & (1 << i)) == 0)
163
+ printf("r%d: (unused)\n", 12 - i);
164
+ else
165
+ printf("r%d: 0x%08x\n", 12 - i, ((unsigned int*)reg_base)[i]);
166
+ }
167
+
168
+ for(i = 0; i < stack_offset / 4; i++)
169
+ {
170
+ printf("sp+%d: 0x%08x\n", i*4, ((unsigned int*)stack_base)[i]);
171
+ }
172
+ #endif
173
+ }
174
+
175
+ /* Perform machine dependent cif processing */
176
+ ffi_status ffi_prep_cif_machdep(ffi_cif *cif)
177
+ {
178
+ /* Round the stack up to a multiple of 8 bytes. This isn't needed
179
+ * everywhere, but it is on some platforms, and it doesn't harm
180
+ * anything when it isn't needed. */
181
+ cif->bytes = (cif->bytes + 7) & ~7;
182
+
183
+ /* Flag to indicate that he return value is in fact a struct */
184
+ cif->rstruct_flag = 0;
185
+
186
+ /* Set the return type flag */
187
+ switch(cif->rtype->type)
188
+ {
189
+ case FFI_TYPE_SINT8:
190
+ case FFI_TYPE_UINT8:
191
+ cif->flags = (unsigned)FFI_TYPE_UINT8;
192
+ break;
193
+ case FFI_TYPE_SINT16:
194
+ case FFI_TYPE_UINT16:
195
+ cif->flags = (unsigned)FFI_TYPE_UINT16;
196
+ break;
197
+ case FFI_TYPE_FLOAT:
198
+ case FFI_TYPE_SINT32:
199
+ case FFI_TYPE_UINT32:
200
+ case FFI_TYPE_POINTER:
201
+ cif->flags = (unsigned)FFI_TYPE_UINT32;
202
+ break;
203
+ case FFI_TYPE_DOUBLE:
204
+ case FFI_TYPE_SINT64:
205
+ case FFI_TYPE_UINT64:
206
+ cif->flags = (unsigned)FFI_TYPE_UINT64;
207
+ break;
208
+ case FFI_TYPE_STRUCT:
209
+ cif->rstruct_flag = 1;
210
+ if(!pass_struct_on_stack(cif->rtype))
211
+ {
212
+ if(cif->rtype->size <= 1)
213
+ cif->flags = (unsigned)FFI_TYPE_UINT8;
214
+ else if(cif->rtype->size <= 2)
215
+ cif->flags = (unsigned)FFI_TYPE_UINT16;
216
+ else if(cif->rtype->size <= 4)
217
+ cif->flags = (unsigned)FFI_TYPE_UINT32;
218
+ else if(cif->rtype->size <= 8)
219
+ cif->flags = (unsigned)FFI_TYPE_UINT64;
220
+ else
221
+ cif->flags = (unsigned)cif->rtype->type;
222
+ }
223
+ else
224
+ cif->flags = (unsigned)cif->rtype->type;
225
+ break;
226
+ default:
227
+ cif->flags = (unsigned)cif->rtype->type;
228
+ break;
229
+ }
230
+
231
+ return FFI_OK;
232
+ }
233
+
234
+ void ffi_call(ffi_cif *cif, void (*fn)(void), void *rvalue, void **avalue)
235
+ {
236
+ extended_cif ecif;
237
+
238
+ unsigned int size = 0, i = 0;
239
+ ffi_type **p_arg;
240
+
241
+ ecif.cif = cif;
242
+ ecif.avalue = avalue;
243
+
244
+ for(i = 0, p_arg = cif->arg_types; i < cif->nargs; i++, p_arg++)
245
+ size += (*p_arg)->size + (4 - (*p_arg)->size % 4);
246
+
247
+ /* If the return value is a struct and we don't have a return value
248
+ * address then we need to make one */
249
+
250
+ /* If cif->flags is struct then it's not suitable for registers */
251
+ if((rvalue == NULL) && (cif->flags == FFI_TYPE_STRUCT))
252
+ ecif.rvalue = alloca(cif->rtype->size);
253
+ else
254
+ ecif.rvalue = rvalue;
255
+
256
+ switch(cif->abi)
257
+ {
258
+ case FFI_SYSV:
259
+ ffi_call_SYSV(ffi_prep_args, &ecif, size, cif->flags,
260
+ ecif.rvalue, cif->rstruct_flag, fn);
261
+ break;
262
+ default:
263
+ FFI_ASSERT(0);
264
+ break;
265
+ }
266
+ }
267
+
268
+ static void ffi_prep_incoming_args_SYSV(char *stack, void **rvalue,
269
+ void **avalue, ffi_cif *cif)
270
+ {
271
+ register unsigned int i, reg_mask = 0;
272
+ register void **p_argv;
273
+ register ffi_type **p_arg;
274
+ register char *reg_base = stack;
275
+ register char *stack_base = stack + 20;
276
+ register unsigned int stack_offset = 0;
277
+
278
+ #ifdef DEBUG
279
+ /* Debugging */
280
+ for(i = 0; i < cif->nargs + 7; i++)
281
+ {
282
+ printf("sp+%d: 0x%08x\n", i*4, ((unsigned int*)stack)[i]);
283
+ }
284
+ #endif
285
+
286
+ /* If cif->flags is struct then we know it's not passed in registers */
287
+ if(cif->flags == FFI_TYPE_STRUCT)
288
+ {
289
+ *rvalue = *(void **)reg_base;
290
+ reg_mask |= 1;
291
+ }
292
+
293
+ p_argv = avalue;
294
+
295
+ for(i = 0, p_arg = cif->arg_types; i < cif->nargs; i++, p_arg++)
296
+ {
297
+ size_t z = (*p_arg)->size;
298
+ int alignment = (*p_arg)->alignment;
299
+
300
+ *p_argv = 0;
301
+
302
+ if(z % 4 != 0)
303
+ z += (4 - z % 4);
304
+
305
+ if(reg_mask != 0x1f)
306
+ {
307
+ if(pass_struct_on_stack(*p_arg))
308
+ {
309
+ *p_argv = (void*)stack_base + stack_offset;
310
+ stack_offset += z;
311
+ }
312
+ else if(z <= sizeof(int))
313
+ {
314
+ char index = 0;
315
+
316
+ while((reg_mask >> index) & 1)
317
+ index++;
318
+
319
+ *p_argv = (void*)reg_base + (index * 4);
320
+ reg_mask |= (1 << index);
321
+ }
322
+ else if(z == 2 * sizeof(int))
323
+ {
324
+ if(!((reg_mask >> 1) & 1))
325
+ {
326
+ *p_argv = (void*)reg_base + 4;
327
+ reg_mask |= (3 << 1);
328
+ }
329
+ else if(!((reg_mask >> 3) & 1))
330
+ {
331
+ *p_argv = (void*)reg_base + 12;
332
+ reg_mask |= (3 << 3);
333
+ }
334
+ }
335
+ }
336
+
337
+ if(!*p_argv)
338
+ {
339
+ *p_argv = (void*)stack_base + stack_offset;
340
+ stack_offset += z;
341
+ }
342
+
343
+ if((*p_arg)->type != FFI_TYPE_STRUCT ||
344
+ (*p_arg)->elements[1] == NULL)
345
+ {
346
+ if(alignment == 1)
347
+ **(unsigned int**)p_argv <<= 24;
348
+ else if(alignment == 2)
349
+ **(unsigned int**)p_argv <<= 16;
350
+ }
351
+
352
+ p_argv++;
353
+ }
354
+
355
+ #ifdef DEBUG
356
+ /* Debugging */
357
+ for(i = 0; i < cif->nargs; i++)
358
+ {
359
+ printf("sp+%d: 0x%08x\n", i*4, *(((unsigned int**)avalue)[i]));
360
+ }
361
+ #endif
362
+ }
363
+
364
+ /* This function is jumped to by the trampoline */
365
+
366
+ unsigned int ffi_closure_SYSV_inner(ffi_closure *closure, void **respp,
367
+ void *args)
368
+ {
369
+ ffi_cif *cif;
370
+ void **arg_area;
371
+ unsigned int i, size = 0;
372
+ ffi_type **p_arg;
373
+
374
+ cif = closure->cif;
375
+
376
+ for(i = 0, p_arg = cif->arg_types; i < cif->nargs; i++, p_arg++)
377
+ size += (*p_arg)->size + (4 - (*p_arg)->size % 4);
378
+
379
+ arg_area = (void **)alloca(size);
380
+
381
+ /* this call will initialize ARG_AREA, such that each element in that
382
+ * array points to the corresponding value on the stack; and if the
383
+ * function returns a structure, it will re-set RESP to point to the
384
+ * structure return address. */
385
+
386
+ ffi_prep_incoming_args_SYSV(args, respp, arg_area, cif);
387
+
388
+ (closure->fun)(cif, *respp, arg_area, closure->user_data);
389
+
390
+ return cif->flags;
391
+ }
392
+
393
+ ffi_status ffi_prep_closure_loc(ffi_closure* closure, ffi_cif* cif,
394
+ void (*fun)(ffi_cif*, void*, void**, void*), void *user_data,
395
+ void *codeloc)
396
+ {
397
+ FFI_ASSERT(cif->abi == FFI_SYSV);
398
+
399
+ unsigned char *__tramp = (unsigned char*)(&closure->tramp[0]);
400
+ unsigned int __fun = (unsigned int)(&ffi_closure_SYSV);
401
+ unsigned int __ctx = (unsigned int)(codeloc);
402
+ unsigned int __rstruct_flag = (unsigned int)(cif->rstruct_flag);
403
+ unsigned int __inner = (unsigned int)(&ffi_closure_SYSV_inner);
404
+ *(unsigned int*) &__tramp[0] = 0xebcd1f00; /* pushm r8-r12 */
405
+ *(unsigned int*) &__tramp[4] = 0xfefc0010; /* ld.w r12, pc[16] */
406
+ *(unsigned int*) &__tramp[8] = 0xfefb0010; /* ld.w r11, pc[16] */
407
+ *(unsigned int*) &__tramp[12] = 0xfefa0010; /* ld.w r10, pc[16] */
408
+ *(unsigned int*) &__tramp[16] = 0xfeff0010; /* ld.w pc, pc[16] */
409
+ *(unsigned int*) &__tramp[20] = __ctx;
410
+ *(unsigned int*) &__tramp[24] = __rstruct_flag;
411
+ *(unsigned int*) &__tramp[28] = __inner;
412
+ *(unsigned int*) &__tramp[32] = __fun;
413
+ syscall(__NR_cacheflush, 0, (&__tramp[0]), 36);
414
+
415
+ closure->cif = cif;
416
+ closure->user_data = user_data;
417
+ closure->fun = fun;
418
+
419
+ return FFI_OK;
420
+ }
421
+
@@ -0,0 +1,50 @@
1
+ /* -----------------------------------------------------------------*-C-*-
2
+ ffitarget.h - Copyright (c) 2009 Bradley Smith <brad@brad-smith.co.uk>
3
+ Target configuration macros for AVR32.
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining
6
+ a copy of this software and associated documentation files (the
7
+ ``Software''), to deal in the Software without restriction, including
8
+ without limitation the rights to use, copy, modify, merge, publish,
9
+ distribute, sublicense, and/or sell copies of the Software, and to
10
+ permit persons to whom the Software is furnished to do so, subject to
11
+ the following conditions:
12
+
13
+ The above copyright notice and this permission notice shall be included
14
+ in all copies or substantial portions of the Software.
15
+
16
+ THE SOFTWARE IS PROVIDED ``AS IS'', WITHOUT WARRANTY OF ANY KIND,
17
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
18
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
19
+ NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
20
+ HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
21
+ WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
22
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
23
+ DEALINGS IN THE SOFTWARE.
24
+
25
+ ----------------------------------------------------------------------- */
26
+
27
+ #ifndef LIBFFI_TARGET_H
28
+ #define LIBFFI_TARGET_H
29
+
30
+ #ifndef LIBFFI_ASM
31
+ typedef unsigned long ffi_arg;
32
+ typedef signed long ffi_sarg;
33
+
34
+ typedef enum ffi_abi {
35
+ FFI_FIRST_ABI = 0,
36
+ FFI_SYSV,
37
+ FFI_DEFAULT_ABI = FFI_SYSV,
38
+ FFI_LAST_ABI = FFI_DEFAULT_ABI + 1
39
+ } ffi_abi;
40
+ #endif
41
+
42
+ #define FFI_EXTRA_CIF_FIELDS unsigned int rstruct_flag
43
+
44
+ /* Definitions for closures */
45
+
46
+ #define FFI_CLOSURES 1
47
+ #define FFI_TRAMPOLINE_SIZE 36
48
+ #define FFI_NATIVE_RAW_API 0
49
+
50
+ #endif
@@ -0,0 +1,208 @@
1
+ /* -----------------------------------------------------------------------
2
+ sysv.S - Copyright (c) 2009 Bradley Smith <brad@brad-smith.co.uk>
3
+
4
+ AVR32 Foreign Function Interface
5
+
6
+ Permission is hereby granted, free of charge, to any person obtaining
7
+ a copy of this software and associated documentation files (the
8
+ ``Software''), to deal in the Software without restriction, including
9
+ without limitation the rights to use, copy, modify, merge, publish,
10
+ distribute, sublicense, and/or sell copies of the Software, and to
11
+ permit persons to whom the Software is furnished to do so, subject to
12
+ the following conditions:
13
+
14
+ The above copyright notice and this permission notice shall be included
15
+ in all copies or substantial portions of the Software.
16
+
17
+ THE SOFTWARE IS PROVIDED ``AS IS'', WITHOUT WARRANTY OF ANY KIND,
18
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
19
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
20
+ IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
21
+ CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
22
+ TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
23
+ SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
24
+ --------------------------------------------------------------------- */
25
+
26
+ #define LIBFFI_ASM
27
+ #include <fficonfig.h>
28
+ #include <ffi.h>
29
+
30
+ /* r12: ffi_prep_args
31
+ * r11: &ecif
32
+ * r10: size
33
+ * r9: cif->flags
34
+ * r8: ecif.rvalue
35
+ * sp+0: cif->rstruct_flag
36
+ * sp+4: fn */
37
+
38
+ .text
39
+ .align 1
40
+ .globl ffi_call_SYSV
41
+ .type ffi_call_SYSV, @function
42
+ ffi_call_SYSV:
43
+ stm --sp, r0,r1,lr
44
+ stm --sp, r8-r12
45
+ mov r0, sp
46
+
47
+ /* Make room for all of the new args. */
48
+ sub sp, r10
49
+ /* Pad to make way for potential skipped registers */
50
+ sub sp, 20
51
+
52
+ /* Call ffi_prep_args(stack, &ecif). */
53
+ /* r11 already set */
54
+ mov r1, r12
55
+ mov r12, sp
56
+ icall r1
57
+
58
+ /* Save new argument size */
59
+ mov r1, r12
60
+
61
+ /* Move first 5 parameters in registers. */
62
+ ldm sp++, r8-r12
63
+
64
+ /* call (fn) (...). */
65
+ ld.w r1, r0[36]
66
+ icall r1
67
+
68
+ /* Remove the space we pushed for the args. */
69
+ mov sp, r0
70
+
71
+ /* Load r1 with the rstruct flag. */
72
+ ld.w r1, sp[32]
73
+
74
+ /* Load r9 with the return type code. */
75
+ ld.w r9, sp[12]
76
+
77
+ /* Load r8 with the return value pointer. */
78
+ ld.w r8, sp[16]
79
+
80
+ /* If the return value pointer is NULL, assume no return value. */
81
+ cp.w r8, 0
82
+ breq .Lend
83
+
84
+ /* Check if return type is actually a struct */
85
+ cp.w r1, 0
86
+ breq 1f
87
+
88
+ /* Return 8bit */
89
+ cp.w r9, FFI_TYPE_UINT8
90
+ breq .Lstore8
91
+
92
+ /* Return 16bit */
93
+ cp.w r9, FFI_TYPE_UINT16
94
+ breq .Lstore16
95
+
96
+ 1:
97
+ /* Return 32bit */
98
+ cp.w r9, FFI_TYPE_UINT32
99
+ breq .Lstore32
100
+ cp.w r9, FFI_TYPE_UINT16
101
+ breq .Lstore32
102
+ cp.w r9, FFI_TYPE_UINT8
103
+ breq .Lstore32
104
+
105
+ /* Return 64bit */
106
+ cp.w r9, FFI_TYPE_UINT64
107
+ breq .Lstore64
108
+
109
+ /* Didn't match anything */
110
+ bral .Lend
111
+
112
+ .Lstore64:
113
+ st.w r8[0], r11
114
+ st.w r8[4], r10
115
+ bral .Lend
116
+
117
+ .Lstore32:
118
+ st.w r8[0], r12
119
+ bral .Lend
120
+
121
+ .Lstore16:
122
+ st.h r8[0], r12
123
+ bral .Lend
124
+
125
+ .Lstore8:
126
+ st.b r8[0], r12
127
+ bral .Lend
128
+
129
+ .Lend:
130
+ sub sp, -20
131
+ ldm sp++, r0,r1,pc
132
+
133
+ .size ffi_call_SYSV, . - ffi_call_SYSV
134
+
135
+
136
+ /* r12: __ctx
137
+ * r11: __rstruct_flag
138
+ * r10: __inner */
139
+
140
+ .align 1
141
+ .globl ffi_closure_SYSV
142
+ .type ffi_closure_SYSV, @function
143
+ ffi_closure_SYSV:
144
+ stm --sp, r0,lr
145
+ mov r0, r11
146
+ mov r8, r10
147
+ sub r10, sp, -8
148
+ sub sp, 12
149
+ st.w sp[8], sp
150
+ sub r11, sp, -8
151
+ icall r8
152
+
153
+ /* Check if return type is actually a struct */
154
+ cp.w r0, 0
155
+ breq 1f
156
+
157
+ /* Return 8bit */
158
+ cp.w r12, FFI_TYPE_UINT8
159
+ breq .Lget8
160
+
161
+ /* Return 16bit */
162
+ cp.w r12, FFI_TYPE_UINT16
163
+ breq .Lget16
164
+
165
+ 1:
166
+ /* Return 32bit */
167
+ cp.w r12, FFI_TYPE_UINT32
168
+ breq .Lget32
169
+ cp.w r12, FFI_TYPE_UINT16
170
+ breq .Lget32
171
+ cp.w r12, FFI_TYPE_UINT8
172
+ breq .Lget32
173
+
174
+ /* Return 64bit */
175
+ cp.w r12, FFI_TYPE_UINT64
176
+ breq .Lget64
177
+
178
+ /* Didn't match anything */
179
+ bral .Lclend
180
+
181
+ .Lget64:
182
+ ld.w r11, sp[0]
183
+ ld.w r10, sp[4]
184
+ bral .Lclend
185
+
186
+ .Lget32:
187
+ ld.w r12, sp[0]
188
+ bral .Lclend
189
+
190
+ .Lget16:
191
+ ld.uh r12, sp[0]
192
+ bral .Lclend
193
+
194
+ .Lget8:
195
+ ld.ub r12, sp[0]
196
+ bral .Lclend
197
+
198
+ .Lclend:
199
+ sub sp, -12
200
+ ldm sp++, r0,lr
201
+ sub sp, -20
202
+ mov pc, lr
203
+
204
+ .size ffi_closure_SYSV, . - ffi_closure_SYSV
205
+
206
+ #if defined __ELF__ && defined __linux__
207
+ .section .note.GNU-stack,"",@progbits
208
+ #endif