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

...
author Shinji KONO <kono@ie.u-ryukyu.ac.jp>
date Mon, 25 May 2020 18:13:55 +0900
parents 04ced10e8804
children
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
111
kono
parents:
diff changeset
1 /* -----------------------------------------------------------------------
kono
parents:
diff changeset
2 raw_api.c - Copyright (c) 1999, 2008 Red Hat, Inc.
kono
parents:
diff changeset
3
kono
parents:
diff changeset
4 Author: Kresten Krab Thorup <krab@gnu.org>
kono
parents:
diff changeset
5
kono
parents:
diff changeset
6 Permission is hereby granted, free of charge, to any person obtaining
kono
parents:
diff changeset
7 a copy of this software and associated documentation files (the
kono
parents:
diff changeset
8 ``Software''), to deal in the Software without restriction, including
kono
parents:
diff changeset
9 without limitation the rights to use, copy, modify, merge, publish,
kono
parents:
diff changeset
10 distribute, sublicense, and/or sell copies of the Software, and to
kono
parents:
diff changeset
11 permit persons to whom the Software is furnished to do so, subject to
kono
parents:
diff changeset
12 the following conditions:
kono
parents:
diff changeset
13
kono
parents:
diff changeset
14 The above copyright notice and this permission notice shall be included
kono
parents:
diff changeset
15 in all copies or substantial portions of the Software.
kono
parents:
diff changeset
16
kono
parents:
diff changeset
17 THE SOFTWARE IS PROVIDED ``AS IS'', WITHOUT WARRANTY OF ANY KIND,
kono
parents:
diff changeset
18 EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
kono
parents:
diff changeset
19 MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
kono
parents:
diff changeset
20 NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
kono
parents:
diff changeset
21 HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
kono
parents:
diff changeset
22 WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
kono
parents:
diff changeset
23 OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
kono
parents:
diff changeset
24 DEALINGS IN THE SOFTWARE.
kono
parents:
diff changeset
25 ----------------------------------------------------------------------- */
kono
parents:
diff changeset
26
kono
parents:
diff changeset
27 /* This file defines generic functions for use with the raw api. */
kono
parents:
diff changeset
28
kono
parents:
diff changeset
29 #include <ffi.h>
kono
parents:
diff changeset
30 #include <ffi_common.h>
kono
parents:
diff changeset
31
kono
parents:
diff changeset
32 #if !FFI_NO_RAW_API
kono
parents:
diff changeset
33
kono
parents:
diff changeset
34 size_t
kono
parents:
diff changeset
35 ffi_raw_size (ffi_cif *cif)
kono
parents:
diff changeset
36 {
kono
parents:
diff changeset
37 size_t result = 0;
kono
parents:
diff changeset
38 int i;
kono
parents:
diff changeset
39
kono
parents:
diff changeset
40 ffi_type **at = cif->arg_types;
kono
parents:
diff changeset
41
kono
parents:
diff changeset
42 for (i = cif->nargs-1; i >= 0; i--, at++)
kono
parents:
diff changeset
43 {
kono
parents:
diff changeset
44 #if !FFI_NO_STRUCTS
kono
parents:
diff changeset
45 if ((*at)->type == FFI_TYPE_STRUCT)
kono
parents:
diff changeset
46 result += ALIGN (sizeof (void*), FFI_SIZEOF_ARG);
kono
parents:
diff changeset
47 else
kono
parents:
diff changeset
48 #endif
kono
parents:
diff changeset
49 result += ALIGN ((*at)->size, FFI_SIZEOF_ARG);
kono
parents:
diff changeset
50 }
kono
parents:
diff changeset
51
kono
parents:
diff changeset
52 return result;
kono
parents:
diff changeset
53 }
kono
parents:
diff changeset
54
kono
parents:
diff changeset
55
kono
parents:
diff changeset
56 void
kono
parents:
diff changeset
57 ffi_raw_to_ptrarray (ffi_cif *cif, ffi_raw *raw, void **args)
kono
parents:
diff changeset
58 {
kono
parents:
diff changeset
59 unsigned i;
kono
parents:
diff changeset
60 ffi_type **tp = cif->arg_types;
kono
parents:
diff changeset
61
kono
parents:
diff changeset
62 #if WORDS_BIGENDIAN
kono
parents:
diff changeset
63
kono
parents:
diff changeset
64 for (i = 0; i < cif->nargs; i++, tp++, args++)
kono
parents:
diff changeset
65 {
kono
parents:
diff changeset
66 switch ((*tp)->type)
kono
parents:
diff changeset
67 {
kono
parents:
diff changeset
68 case FFI_TYPE_UINT8:
kono
parents:
diff changeset
69 case FFI_TYPE_SINT8:
kono
parents:
diff changeset
70 *args = (void*) ((char*)(raw++) + FFI_SIZEOF_ARG - 1);
kono
parents:
diff changeset
71 break;
kono
parents:
diff changeset
72
kono
parents:
diff changeset
73 case FFI_TYPE_UINT16:
kono
parents:
diff changeset
74 case FFI_TYPE_SINT16:
kono
parents:
diff changeset
75 *args = (void*) ((char*)(raw++) + FFI_SIZEOF_ARG - 2);
kono
parents:
diff changeset
76 break;
kono
parents:
diff changeset
77
kono
parents:
diff changeset
78 #if FFI_SIZEOF_ARG >= 4
kono
parents:
diff changeset
79 case FFI_TYPE_UINT32:
kono
parents:
diff changeset
80 case FFI_TYPE_SINT32:
kono
parents:
diff changeset
81 *args = (void*) ((char*)(raw++) + FFI_SIZEOF_ARG - 4);
kono
parents:
diff changeset
82 break;
kono
parents:
diff changeset
83 #endif
kono
parents:
diff changeset
84
kono
parents:
diff changeset
85 #if !FFI_NO_STRUCTS
kono
parents:
diff changeset
86 case FFI_TYPE_STRUCT:
kono
parents:
diff changeset
87 *args = (raw++)->ptr;
kono
parents:
diff changeset
88 break;
kono
parents:
diff changeset
89 #endif
kono
parents:
diff changeset
90
kono
parents:
diff changeset
91 case FFI_TYPE_COMPLEX:
kono
parents:
diff changeset
92 *args = (raw++)->ptr;
kono
parents:
diff changeset
93 break;
kono
parents:
diff changeset
94
kono
parents:
diff changeset
95 case FFI_TYPE_POINTER:
kono
parents:
diff changeset
96 *args = (void*) &(raw++)->ptr;
kono
parents:
diff changeset
97 break;
kono
parents:
diff changeset
98
kono
parents:
diff changeset
99 default:
kono
parents:
diff changeset
100 *args = raw;
kono
parents:
diff changeset
101 raw += ALIGN ((*tp)->size, FFI_SIZEOF_ARG) / FFI_SIZEOF_ARG;
kono
parents:
diff changeset
102 }
kono
parents:
diff changeset
103 }
kono
parents:
diff changeset
104
kono
parents:
diff changeset
105 #else /* WORDS_BIGENDIAN */
kono
parents:
diff changeset
106
kono
parents:
diff changeset
107 #if !PDP
kono
parents:
diff changeset
108
kono
parents:
diff changeset
109 /* then assume little endian */
kono
parents:
diff changeset
110 for (i = 0; i < cif->nargs; i++, tp++, args++)
kono
parents:
diff changeset
111 {
kono
parents:
diff changeset
112 #if !FFI_NO_STRUCTS
kono
parents:
diff changeset
113 if ((*tp)->type == FFI_TYPE_STRUCT)
kono
parents:
diff changeset
114 {
kono
parents:
diff changeset
115 *args = (raw++)->ptr;
kono
parents:
diff changeset
116 }
kono
parents:
diff changeset
117 else
kono
parents:
diff changeset
118 #endif
kono
parents:
diff changeset
119 if ((*tp)->type == FFI_TYPE_COMPLEX)
kono
parents:
diff changeset
120 {
kono
parents:
diff changeset
121 *args = (raw++)->ptr;
kono
parents:
diff changeset
122 }
kono
parents:
diff changeset
123 else
kono
parents:
diff changeset
124 {
kono
parents:
diff changeset
125 *args = (void*) raw;
kono
parents:
diff changeset
126 raw += ALIGN ((*tp)->size, sizeof (void*)) / sizeof (void*);
kono
parents:
diff changeset
127 }
kono
parents:
diff changeset
128 }
kono
parents:
diff changeset
129
kono
parents:
diff changeset
130 #else
kono
parents:
diff changeset
131 #error "pdp endian not supported"
kono
parents:
diff changeset
132 #endif /* ! PDP */
kono
parents:
diff changeset
133
kono
parents:
diff changeset
134 #endif /* WORDS_BIGENDIAN */
kono
parents:
diff changeset
135 }
kono
parents:
diff changeset
136
kono
parents:
diff changeset
137 void
kono
parents:
diff changeset
138 ffi_ptrarray_to_raw (ffi_cif *cif, void **args, ffi_raw *raw)
kono
parents:
diff changeset
139 {
kono
parents:
diff changeset
140 unsigned i;
kono
parents:
diff changeset
141 ffi_type **tp = cif->arg_types;
kono
parents:
diff changeset
142
kono
parents:
diff changeset
143 for (i = 0; i < cif->nargs; i++, tp++, args++)
kono
parents:
diff changeset
144 {
kono
parents:
diff changeset
145 switch ((*tp)->type)
kono
parents:
diff changeset
146 {
kono
parents:
diff changeset
147 case FFI_TYPE_UINT8:
kono
parents:
diff changeset
148 (raw++)->uint = *(UINT8*) (*args);
kono
parents:
diff changeset
149 break;
kono
parents:
diff changeset
150
kono
parents:
diff changeset
151 case FFI_TYPE_SINT8:
kono
parents:
diff changeset
152 (raw++)->sint = *(SINT8*) (*args);
kono
parents:
diff changeset
153 break;
kono
parents:
diff changeset
154
kono
parents:
diff changeset
155 case FFI_TYPE_UINT16:
kono
parents:
diff changeset
156 (raw++)->uint = *(UINT16*) (*args);
kono
parents:
diff changeset
157 break;
kono
parents:
diff changeset
158
kono
parents:
diff changeset
159 case FFI_TYPE_SINT16:
kono
parents:
diff changeset
160 (raw++)->sint = *(SINT16*) (*args);
kono
parents:
diff changeset
161 break;
kono
parents:
diff changeset
162
kono
parents:
diff changeset
163 #if FFI_SIZEOF_ARG >= 4
kono
parents:
diff changeset
164 case FFI_TYPE_UINT32:
kono
parents:
diff changeset
165 (raw++)->uint = *(UINT32*) (*args);
kono
parents:
diff changeset
166 break;
kono
parents:
diff changeset
167
kono
parents:
diff changeset
168 case FFI_TYPE_SINT32:
kono
parents:
diff changeset
169 (raw++)->sint = *(SINT32*) (*args);
kono
parents:
diff changeset
170 break;
kono
parents:
diff changeset
171 #endif
kono
parents:
diff changeset
172
kono
parents:
diff changeset
173 #if !FFI_NO_STRUCTS
kono
parents:
diff changeset
174 case FFI_TYPE_STRUCT:
kono
parents:
diff changeset
175 (raw++)->ptr = *args;
kono
parents:
diff changeset
176 break;
kono
parents:
diff changeset
177 #endif
kono
parents:
diff changeset
178
kono
parents:
diff changeset
179 case FFI_TYPE_COMPLEX:
kono
parents:
diff changeset
180 (raw++)->ptr = *args;
kono
parents:
diff changeset
181 break;
kono
parents:
diff changeset
182
kono
parents:
diff changeset
183 case FFI_TYPE_POINTER:
kono
parents:
diff changeset
184 (raw++)->ptr = **(void***) args;
kono
parents:
diff changeset
185 break;
kono
parents:
diff changeset
186
kono
parents:
diff changeset
187 default:
kono
parents:
diff changeset
188 memcpy ((void*) raw->data, (void*)*args, (*tp)->size);
kono
parents:
diff changeset
189 raw += ALIGN ((*tp)->size, FFI_SIZEOF_ARG) / FFI_SIZEOF_ARG;
kono
parents:
diff changeset
190 }
kono
parents:
diff changeset
191 }
kono
parents:
diff changeset
192 }
kono
parents:
diff changeset
193
kono
parents:
diff changeset
194 #if !FFI_NATIVE_RAW_API
kono
parents:
diff changeset
195
kono
parents:
diff changeset
196
kono
parents:
diff changeset
197 /* This is a generic definition of ffi_raw_call, to be used if the
kono
parents:
diff changeset
198 * native system does not provide a machine-specific implementation.
kono
parents:
diff changeset
199 * Having this, allows code to be written for the raw API, without
kono
parents:
diff changeset
200 * the need for system-specific code to handle input in that format;
kono
parents:
diff changeset
201 * these following couple of functions will handle the translation forth
kono
parents:
diff changeset
202 * and back automatically. */
kono
parents:
diff changeset
203
kono
parents:
diff changeset
204 void ffi_raw_call (ffi_cif *cif, void (*fn)(void), void *rvalue, ffi_raw *raw)
kono
parents:
diff changeset
205 {
kono
parents:
diff changeset
206 void **avalue = (void**) alloca (cif->nargs * sizeof (void*));
kono
parents:
diff changeset
207 ffi_raw_to_ptrarray (cif, raw, avalue);
kono
parents:
diff changeset
208 ffi_call (cif, fn, rvalue, avalue);
kono
parents:
diff changeset
209 }
kono
parents:
diff changeset
210
kono
parents:
diff changeset
211 #if FFI_CLOSURES /* base system provides closures */
kono
parents:
diff changeset
212
kono
parents:
diff changeset
213 static void
kono
parents:
diff changeset
214 ffi_translate_args (ffi_cif *cif, void *rvalue,
kono
parents:
diff changeset
215 void **avalue, void *user_data)
kono
parents:
diff changeset
216 {
kono
parents:
diff changeset
217 ffi_raw *raw = (ffi_raw*)alloca (ffi_raw_size (cif));
kono
parents:
diff changeset
218 ffi_raw_closure *cl = (ffi_raw_closure*)user_data;
kono
parents:
diff changeset
219
kono
parents:
diff changeset
220 ffi_ptrarray_to_raw (cif, avalue, raw);
kono
parents:
diff changeset
221 (*cl->fun) (cif, rvalue, raw, cl->user_data);
kono
parents:
diff changeset
222 }
kono
parents:
diff changeset
223
kono
parents:
diff changeset
224 ffi_status
kono
parents:
diff changeset
225 ffi_prep_raw_closure_loc (ffi_raw_closure* cl,
kono
parents:
diff changeset
226 ffi_cif *cif,
kono
parents:
diff changeset
227 void (*fun)(ffi_cif*,void*,ffi_raw*,void*),
kono
parents:
diff changeset
228 void *user_data,
kono
parents:
diff changeset
229 void *codeloc)
kono
parents:
diff changeset
230 {
kono
parents:
diff changeset
231 ffi_status status;
kono
parents:
diff changeset
232
kono
parents:
diff changeset
233 status = ffi_prep_closure_loc ((ffi_closure*) cl,
kono
parents:
diff changeset
234 cif,
kono
parents:
diff changeset
235 &ffi_translate_args,
kono
parents:
diff changeset
236 codeloc,
kono
parents:
diff changeset
237 codeloc);
kono
parents:
diff changeset
238 if (status == FFI_OK)
kono
parents:
diff changeset
239 {
kono
parents:
diff changeset
240 cl->fun = fun;
kono
parents:
diff changeset
241 cl->user_data = user_data;
kono
parents:
diff changeset
242 }
kono
parents:
diff changeset
243
kono
parents:
diff changeset
244 return status;
kono
parents:
diff changeset
245 }
kono
parents:
diff changeset
246
kono
parents:
diff changeset
247 #endif /* FFI_CLOSURES */
kono
parents:
diff changeset
248 #endif /* !FFI_NATIVE_RAW_API */
kono
parents:
diff changeset
249
kono
parents:
diff changeset
250 #if FFI_CLOSURES
kono
parents:
diff changeset
251
kono
parents:
diff changeset
252 /* Again, here is the generic version of ffi_prep_raw_closure, which
kono
parents:
diff changeset
253 * will install an intermediate "hub" for translation of arguments from
kono
parents:
diff changeset
254 * the pointer-array format, to the raw format */
kono
parents:
diff changeset
255
kono
parents:
diff changeset
256 ffi_status
kono
parents:
diff changeset
257 ffi_prep_raw_closure (ffi_raw_closure* cl,
kono
parents:
diff changeset
258 ffi_cif *cif,
kono
parents:
diff changeset
259 void (*fun)(ffi_cif*,void*,ffi_raw*,void*),
kono
parents:
diff changeset
260 void *user_data)
kono
parents:
diff changeset
261 {
kono
parents:
diff changeset
262 return ffi_prep_raw_closure_loc (cl, cif, fun, user_data, cl);
kono
parents:
diff changeset
263 }
kono
parents:
diff changeset
264
kono
parents:
diff changeset
265 #endif /* FFI_CLOSURES */
kono
parents:
diff changeset
266
kono
parents:
diff changeset
267 #endif /* !FFI_NO_RAW_API */