annotate libffi/src/riscv/ffi.c @ 158:494b0b89df80 default tip

...
author Shinji KONO <kono@ie.u-ryukyu.ac.jp>
date Mon, 25 May 2020 18:13:55 +0900
parents 84e7813d76e9
children
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
131
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
1 /* -----------------------------------------------------------------------
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
2 ffi.c - Copyright (c) 2015 Michael Knyszek <mknyszek@berkeley.edu>
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
3 2015 Andrew Waterman <waterman@cs.berkeley.edu>
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
4 2018 Stef O'Rear <sorear2@gmail.com>
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
5 Based on MIPS N32/64 port
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
6
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
7 RISC-V Foreign Function Interface
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
8
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
9 Permission is hereby granted, free of charge, to any person obtaining
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
10 a copy of this software and associated documentation files (the
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
11 ``Software''), to deal in the Software without restriction, including
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
12 without limitation the rights to use, copy, modify, merge, publish,
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
13 distribute, sublicense, and/or sell copies of the Software, and to
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
14 permit persons to whom the Software is furnished to do so, subject to
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
15 the following conditions:
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
16
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
17 The above copyright notice and this permission notice shall be included
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
18 in all copies or substantial portions of the Software.
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
19
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
20 THE SOFTWARE IS PROVIDED ``AS IS'', WITHOUT WARRANTY OF ANY KIND,
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
21 EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
22 MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
23 NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
24 HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
25 WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
26 OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
27 DEALINGS IN THE SOFTWARE.
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
28 ----------------------------------------------------------------------- */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
29
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
30 #include <ffi.h>
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
31 #include <ffi_common.h>
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
32
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
33 #include <stdlib.h>
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
34 #include <stdint.h>
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
35
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
36 #if __riscv_float_abi_double
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
37 #define ABI_FLEN 64
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
38 #define ABI_FLOAT double
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
39 #elif __riscv_float_abi_single
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
40 #define ABI_FLEN 32
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
41 #define ABI_FLOAT float
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
42 #endif
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
43
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
44 #define NARGREG 8
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
45 #define STKALIGN 16
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
46 #define MAXCOPYARG (2 * sizeof(double))
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
47
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
48 typedef struct call_context
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
49 {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
50 #if ABI_FLEN
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
51 ABI_FLOAT fa[8];
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
52 #endif
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
53 size_t a[8];
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
54 /* used by the assembly code to in-place construct its own stack frame */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
55 char frame[16];
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
56 } call_context;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
57
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
58 typedef struct call_builder
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
59 {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
60 call_context *aregs;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
61 int used_integer;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
62 int used_float;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
63 size_t *used_stack;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
64 } call_builder;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
65
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
66 /* integer (not pointer) less than ABI XLEN */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
67 /* FFI_TYPE_INT does not appear to be used */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
68 #if __SIZEOF_POINTER__ == 8
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
69 #define IS_INT(type) ((type) >= FFI_TYPE_UINT8 && (type) <= FFI_TYPE_SINT64)
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
70 #else
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
71 #define IS_INT(type) ((type) >= FFI_TYPE_UINT8 && (type) <= FFI_TYPE_SINT32)
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
72 #endif
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
73
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
74 #if ABI_FLEN
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
75 typedef struct {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
76 char as_elements, type1, offset2, type2;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
77 } float_struct_info;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
78
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
79 #if ABI_FLEN >= 64
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
80 #define IS_FLOAT(type) ((type) >= FFI_TYPE_FLOAT && (type) <= FFI_TYPE_DOUBLE)
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
81 #else
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
82 #define IS_FLOAT(type) ((type) == FFI_TYPE_FLOAT)
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
83 #endif
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
84
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
85 static ffi_type **flatten_struct(ffi_type *in, ffi_type **out, ffi_type **out_end) {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
86 int i;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
87 if (out == out_end) return out;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
88 if (in->type != FFI_TYPE_STRUCT) {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
89 *(out++) = in;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
90 } else {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
91 for (i = 0; in->elements[i]; i++)
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
92 out = flatten_struct(in->elements[i], out, out_end);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
93 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
94 return out;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
95 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
96
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
97 /* Structs with at most two fields after flattening, one of which is of
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
98 floating point type, are passed in multiple registers if sufficient
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
99 registers are available. */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
100 static float_struct_info struct_passed_as_elements(call_builder *cb, ffi_type *top) {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
101 float_struct_info ret = {0, 0, 0, 0};
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
102 ffi_type *fields[3];
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
103 int num_floats, num_ints;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
104 int num_fields = flatten_struct(top, fields, fields + 3) - fields;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
105
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
106 if (num_fields == 1) {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
107 if (IS_FLOAT(fields[0]->type)) {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
108 ret.as_elements = 1;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
109 ret.type1 = fields[0]->type;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
110 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
111 } else if (num_fields == 2) {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
112 num_floats = IS_FLOAT(fields[0]->type) + IS_FLOAT(fields[1]->type);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
113 num_ints = IS_INT(fields[0]->type) + IS_INT(fields[1]->type);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
114 if (num_floats == 0 || num_floats + num_ints != 2)
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
115 return ret;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
116 if (cb->used_float + num_floats > NARGREG || cb->used_integer + (2 - num_floats) > NARGREG)
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
117 return ret;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
118 if (!IS_FLOAT(fields[0]->type) && !IS_FLOAT(fields[1]->type))
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
119 return ret;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
120
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
121 ret.type1 = fields[0]->type;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
122 ret.type2 = fields[1]->type;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
123 ret.offset2 = ALIGN(fields[0]->size, fields[1]->alignment);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
124 ret.as_elements = 1;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
125 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
126
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
127 return ret;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
128 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
129 #endif
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
130
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
131 /* allocates a single register, float register, or XLEN-sized stack slot to a datum */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
132 static void marshal_atom(call_builder *cb, int type, void *data) {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
133 size_t value = 0;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
134 switch (type) {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
135 case FFI_TYPE_UINT8: value = *(uint8_t *)data; break;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
136 case FFI_TYPE_SINT8: value = *(int8_t *)data; break;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
137 case FFI_TYPE_UINT16: value = *(uint16_t *)data; break;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
138 case FFI_TYPE_SINT16: value = *(int16_t *)data; break;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
139 /* 32-bit quantities are always sign-extended in the ABI */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
140 case FFI_TYPE_UINT32: value = *(int32_t *)data; break;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
141 case FFI_TYPE_SINT32: value = *(int32_t *)data; break;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
142 #if __SIZEOF_POINTER__ == 8
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
143 case FFI_TYPE_UINT64: value = *(uint64_t *)data; break;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
144 case FFI_TYPE_SINT64: value = *(int64_t *)data; break;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
145 #endif
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
146 case FFI_TYPE_POINTER: value = *(size_t *)data; break;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
147
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
148 /* float values may be recoded in an implementation-defined way
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
149 by hardware conforming to 2.1 or earlier, so use asm to
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
150 reinterpret floats as doubles */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
151 #if ABI_FLEN >= 32
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
152 case FFI_TYPE_FLOAT:
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
153 asm("" : "=f"(cb->aregs->fa[cb->used_float++]) : "0"(*(float *)data));
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
154 return;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
155 #endif
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
156 #if ABI_FLEN >= 64
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
157 case FFI_TYPE_DOUBLE:
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
158 asm("" : "=f"(cb->aregs->fa[cb->used_float++]) : "0"(*(double *)data));
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
159 return;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
160 #endif
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
161 default: FFI_ASSERT(0); break;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
162 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
163
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
164 if (cb->used_integer == NARGREG) {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
165 *cb->used_stack++ = value;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
166 } else {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
167 cb->aregs->a[cb->used_integer++] = value;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
168 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
169 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
170
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
171 static void unmarshal_atom(call_builder *cb, int type, void *data) {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
172 size_t value;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
173 switch (type) {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
174 #if ABI_FLEN >= 32
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
175 case FFI_TYPE_FLOAT:
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
176 asm("" : "=f"(*(float *)data) : "0"(cb->aregs->fa[cb->used_float++]));
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
177 return;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
178 #endif
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
179 #if ABI_FLEN >= 64
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
180 case FFI_TYPE_DOUBLE:
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
181 asm("" : "=f"(*(double *)data) : "0"(cb->aregs->fa[cb->used_float++]));
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
182 return;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
183 #endif
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
184 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
185
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
186 if (cb->used_integer == NARGREG) {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
187 value = *cb->used_stack++;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
188 } else {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
189 value = cb->aregs->a[cb->used_integer++];
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
190 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
191
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
192 switch (type) {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
193 case FFI_TYPE_UINT8: *(uint8_t *)data = value; break;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
194 case FFI_TYPE_SINT8: *(uint8_t *)data = value; break;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
195 case FFI_TYPE_UINT16: *(uint16_t *)data = value; break;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
196 case FFI_TYPE_SINT16: *(uint16_t *)data = value; break;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
197 case FFI_TYPE_UINT32: *(uint32_t *)data = value; break;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
198 case FFI_TYPE_SINT32: *(uint32_t *)data = value; break;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
199 #if __SIZEOF_POINTER__ == 8
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
200 case FFI_TYPE_UINT64: *(uint64_t *)data = value; break;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
201 case FFI_TYPE_SINT64: *(uint64_t *)data = value; break;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
202 #endif
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
203 case FFI_TYPE_POINTER: *(size_t *)data = value; break;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
204 default: FFI_ASSERT(0); break;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
205 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
206 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
207
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
208 /* adds an argument to a call, or a not by reference return value */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
209 static void marshal(call_builder *cb, ffi_type *type, int var, void *data) {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
210 size_t realign[2];
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
211
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
212 #if ABI_FLEN
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
213 if (!var && type->type == FFI_TYPE_STRUCT) {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
214 float_struct_info fsi = struct_passed_as_elements(cb, type);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
215 if (fsi.as_elements) {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
216 marshal_atom(cb, fsi.type1, data);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
217 if (fsi.offset2)
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
218 marshal_atom(cb, fsi.type2, ((char*)data) + fsi.offset2);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
219 return;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
220 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
221 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
222
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
223 if (!var && cb->used_float < NARGREG && IS_FLOAT(type->type)) {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
224 marshal_atom(cb, type->type, data);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
225 return;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
226 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
227 #endif
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
228
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
229 if (type->size > 2 * __SIZEOF_POINTER__) {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
230 /* pass by reference */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
231 marshal_atom(cb, FFI_TYPE_POINTER, &data);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
232 } else if (IS_INT(type->type) || type->type == FFI_TYPE_POINTER) {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
233 marshal_atom(cb, type->type, data);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
234 } else {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
235 /* overlong integers, soft-float floats, and structs without special
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
236 float handling are treated identically from this point on */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
237
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
238 /* variadics are aligned even in registers */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
239 if (type->alignment > __SIZEOF_POINTER__) {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
240 if (var)
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
241 cb->used_integer = ALIGN(cb->used_integer, 2);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
242 cb->used_stack = (size_t *)ALIGN(cb->used_stack, 2*__SIZEOF_POINTER__);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
243 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
244
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
245 memcpy(realign, data, type->size);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
246 if (type->size > 0)
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
247 marshal_atom(cb, FFI_TYPE_POINTER, realign);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
248 if (type->size > __SIZEOF_POINTER__)
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
249 marshal_atom(cb, FFI_TYPE_POINTER, realign + 1);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
250 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
251 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
252
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
253 /* for arguments passed by reference returns the pointer, otherwise the arg is copied (up to MAXCOPYARG bytes) */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
254 static void *unmarshal(call_builder *cb, ffi_type *type, int var, void *data) {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
255 size_t realign[2];
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
256 void *pointer;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
257
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
258 #if ABI_FLEN
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
259 if (!var && type->type == FFI_TYPE_STRUCT) {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
260 float_struct_info fsi = struct_passed_as_elements(cb, type);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
261 if (fsi.as_elements) {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
262 unmarshal_atom(cb, fsi.type1, data);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
263 if (fsi.offset2)
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
264 unmarshal_atom(cb, fsi.type2, ((char*)data) + fsi.offset2);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
265 return data;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
266 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
267 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
268
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
269 if (!var && cb->used_float < NARGREG && IS_FLOAT(type->type)) {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
270 unmarshal_atom(cb, type->type, data);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
271 return data;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
272 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
273 #endif
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
274
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
275 if (type->size > 2 * __SIZEOF_POINTER__) {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
276 /* pass by reference */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
277 unmarshal_atom(cb, FFI_TYPE_POINTER, (char*)&pointer);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
278 return pointer;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
279 } else if (IS_INT(type->type) || type->type == FFI_TYPE_POINTER) {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
280 unmarshal_atom(cb, type->type, data);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
281 return data;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
282 } else {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
283 /* overlong integers, soft-float floats, and structs without special
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
284 float handling are treated identically from this point on */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
285
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
286 /* variadics are aligned even in registers */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
287 if (type->alignment > __SIZEOF_POINTER__) {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
288 if (var)
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
289 cb->used_integer = ALIGN(cb->used_integer, 2);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
290 cb->used_stack = (size_t *)ALIGN(cb->used_stack, 2*__SIZEOF_POINTER__);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
291 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
292
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
293 if (type->size > 0)
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
294 unmarshal_atom(cb, FFI_TYPE_POINTER, realign);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
295 if (type->size > __SIZEOF_POINTER__)
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
296 unmarshal_atom(cb, FFI_TYPE_POINTER, realign + 1);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
297 memcpy(data, realign, type->size);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
298 return data;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
299 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
300 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
301
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
302 static int passed_by_ref(call_builder *cb, ffi_type *type, int var) {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
303 #if ABI_FLEN
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
304 if (!var && type->type == FFI_TYPE_STRUCT) {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
305 float_struct_info fsi = struct_passed_as_elements(cb, type);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
306 if (fsi.as_elements) return 0;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
307 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
308 #endif
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
309
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
310 return type->size > 2 * __SIZEOF_POINTER__;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
311 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
312
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
313 /* Perform machine dependent cif processing */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
314 ffi_status ffi_prep_cif_machdep(ffi_cif *cif) {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
315 cif->riscv_nfixedargs = cif->nargs;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
316 return FFI_OK;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
317 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
318
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
319 /* Perform machine dependent cif processing when we have a variadic function */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
320
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
321 ffi_status ffi_prep_cif_machdep_var(ffi_cif *cif, unsigned int nfixedargs, unsigned int ntotalargs) {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
322 cif->riscv_nfixedargs = nfixedargs;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
323 return FFI_OK;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
324 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
325
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
326 /* Low level routine for calling functions */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
327 extern void ffi_call_asm (void *stack, struct call_context *regs,
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
328 void (*fn) (void), void *closure) FFI_HIDDEN;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
329
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
330 static void
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
331 ffi_call_int (ffi_cif *cif, void (*fn) (void), void *rvalue, void **avalue,
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
332 void *closure)
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
333 {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
334 /* this is a conservative estimate, assuming a complex return value and
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
335 that all remaining arguments are long long / __int128 */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
336 size_t arg_bytes = cif->nargs <= 3 ? 0 :
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
337 ALIGN(2 * sizeof(size_t) * (cif->nargs - 3), STKALIGN);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
338 size_t rval_bytes = 0;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
339 if (rvalue == NULL && cif->rtype->size > 2*__SIZEOF_POINTER__)
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
340 rval_bytes = ALIGN(cif->rtype->size, STKALIGN);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
341 size_t alloc_size = arg_bytes + rval_bytes + sizeof(call_context);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
342
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
343 /* the assembly code will deallocate all stack data at lower addresses
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
344 than the argument region, so we need to allocate the frame and the
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
345 return value after the arguments in a single allocation */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
346 size_t alloc_base;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
347 /* Argument region must be 16-byte aligned */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
348 if (_Alignof(max_align_t) >= STKALIGN) {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
349 /* since sizeof long double is normally 16, the compiler will
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
350 guarantee alloca alignment to at least that much */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
351 alloc_base = (size_t)alloca(alloc_size);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
352 } else {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
353 alloc_base = ALIGN(alloca(alloc_size + STKALIGN - 1), STKALIGN);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
354 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
355
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
356 if (rval_bytes)
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
357 rvalue = (void*)(alloc_base + arg_bytes);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
358
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
359 call_builder cb;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
360 cb.used_float = cb.used_integer = 0;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
361 cb.aregs = (call_context*)(alloc_base + arg_bytes + rval_bytes);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
362 cb.used_stack = (void*)alloc_base;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
363
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
364 int return_by_ref = passed_by_ref(&cb, cif->rtype, 0);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
365 if (return_by_ref)
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
366 marshal(&cb, &ffi_type_pointer, 0, &rvalue);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
367
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
368 int i;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
369 for (i = 0; i < cif->nargs; i++)
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
370 marshal(&cb, cif->arg_types[i], i >= cif->riscv_nfixedargs, avalue[i]);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
371
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
372 ffi_call_asm ((void *) alloc_base, cb.aregs, fn, closure);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
373
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
374 cb.used_float = cb.used_integer = 0;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
375 if (!return_by_ref && rvalue)
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
376 unmarshal(&cb, cif->rtype, 0, rvalue);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
377 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
378
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
379 void
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
380 ffi_call (ffi_cif *cif, void (*fn) (void), void *rvalue, void **avalue)
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
381 {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
382 ffi_call_int(cif, fn, rvalue, avalue, NULL);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
383 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
384
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
385 void
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
386 ffi_call_go (ffi_cif *cif, void (*fn) (void), void *rvalue,
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
387 void **avalue, void *closure)
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
388 {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
389 ffi_call_int(cif, fn, rvalue, avalue, closure);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
390 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
391
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
392 extern void ffi_closure_asm(void) FFI_HIDDEN;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
393
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
394 ffi_status ffi_prep_closure_loc(ffi_closure *closure, ffi_cif *cif, void (*fun)(ffi_cif*,void*,void**,void*), void *user_data, void *codeloc)
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
395 {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
396 uint32_t *tramp = (uint32_t *) &closure->tramp[0];
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
397 uint64_t fn = (uint64_t) (uintptr_t) ffi_closure_asm;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
398
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
399 if (cif->abi <= FFI_FIRST_ABI || cif->abi >= FFI_LAST_ABI)
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
400 return FFI_BAD_ABI;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
401
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
402 /* we will call ffi_closure_inner with codeloc, not closure, but as long
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
403 as the memory is readable it should work */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
404
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
405 tramp[0] = 0x00000317; /* auipc t1, 0 (i.e. t0 <- codeloc) */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
406 #if __SIZEOF_POINTER__ == 8
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
407 tramp[1] = 0x01033383; /* ld t2, 16(t1) */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
408 #else
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
409 tramp[1] = 0x01032383; /* lw t2, 16(t1) */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
410 #endif
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
411 tramp[2] = 0x00038067; /* jr t2 */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
412 tramp[3] = 0x00000013; /* nop */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
413 tramp[4] = fn;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
414 tramp[5] = fn >> 32;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
415
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
416 closure->cif = cif;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
417 closure->fun = fun;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
418 closure->user_data = user_data;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
419
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
420 __builtin___clear_cache(codeloc, codeloc + FFI_TRAMPOLINE_SIZE);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
421
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
422 return FFI_OK;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
423 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
424
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
425 extern void ffi_go_closure_asm (void) FFI_HIDDEN;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
426
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
427 ffi_status
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
428 ffi_prep_go_closure (ffi_go_closure *closure, ffi_cif *cif,
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
429 void (*fun) (ffi_cif *, void *, void **, void *))
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
430 {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
431 if (cif->abi <= FFI_FIRST_ABI || cif->abi >= FFI_LAST_ABI)
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
432 return FFI_BAD_ABI;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
433
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
434 closure->tramp = (void *) ffi_go_closure_asm;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
435 closure->cif = cif;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
436 closure->fun = fun;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
437
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
438 return FFI_OK;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
439 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
440
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
441 /* Called by the assembly code with aregs pointing to saved argument registers
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
442 and stack pointing to the stacked arguments. Return values passed in
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
443 registers will be reloaded from aregs. */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
444 void FFI_HIDDEN
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
445 ffi_closure_inner (ffi_cif *cif,
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
446 void (*fun) (ffi_cif *, void *, void **, void *),
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
447 void *user_data,
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
448 size_t *stack, call_context *aregs)
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
449 {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
450 void **avalue = alloca(cif->nargs * sizeof(void*));
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
451 /* storage for arguments which will be copied by unmarshal(). We could
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
452 theoretically avoid the copies in many cases and use at most 128 bytes
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
453 of memory, but allocating disjoint storage for each argument is
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
454 simpler. */
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
455 char *astorage = alloca(cif->nargs * MAXCOPYARG);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
456 void *rvalue;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
457 call_builder cb;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
458 int return_by_ref;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
459 int i;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
460
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
461 cb.aregs = aregs;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
462 cb.used_integer = cb.used_float = 0;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
463 cb.used_stack = stack;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
464
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
465 return_by_ref = passed_by_ref(&cb, cif->rtype, 0);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
466 if (return_by_ref)
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
467 unmarshal(&cb, &ffi_type_pointer, 0, &rvalue);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
468 else
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
469 rvalue = alloca(cif->rtype->size);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
470
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
471 for (i = 0; i < cif->nargs; i++)
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
472 avalue[i] = unmarshal(&cb, cif->arg_types[i],
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
473 i >= cif->riscv_nfixedargs, astorage + i*MAXCOPYARG);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
474
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
475 fun (cif, rvalue, avalue, user_data);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
476
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
477 if (!return_by_ref && cif->rtype->type != FFI_TYPE_VOID) {
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
478 cb.used_integer = cb.used_float = 0;
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
479 marshal(&cb, cif->rtype, 0, rvalue);
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
480 }
84e7813d76e9 gcc-8.2
mir3636
parents:
diff changeset
481 }