2003-12-26 Guilhem Lavaux <guilhem@kaffe.org>
[official-gcc.git] / libffi / src / alpha / ffi.c
blob00d337901096395f4a39bbe1f59af7ef39650e0e
1 /* -----------------------------------------------------------------------
2 ffi.c - Copyright (c) 1998, 2001 Red Hat, Inc.
4 Alpha Foreign Function Interface
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:
14 The above copyright notice and this permission notice shall be included
15 in all copies or substantial portions of the Software.
17 THE SOFTWARE IS PROVIDED ``AS IS'', WITHOUT WARRANTY OF ANY KIND, EXPRESS
18 OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
19 MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
20 IN NO EVENT SHALL CYGNUS SOLUTIONS BE LIABLE FOR ANY CLAIM, DAMAGES OR
21 OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
22 ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
23 OTHER DEALINGS IN THE SOFTWARE.
24 ----------------------------------------------------------------------- */
26 #include <ffi.h>
27 #include <ffi_common.h>
29 #include <stdlib.h>
31 extern void ffi_call_osf(void *, unsigned long, unsigned, void *, void (*)());
32 extern void ffi_closure_osf(void);
35 ffi_status
36 ffi_prep_cif_machdep(ffi_cif *cif)
38 /* Adjust cif->bytes to represent a minimum 6 words for the temporary
39 register argument loading area. */
40 if (cif->bytes < 6*FFI_SIZEOF_ARG)
41 cif->bytes = 6*FFI_SIZEOF_ARG;
43 /* Set the return type flag */
44 switch (cif->rtype->type)
46 case FFI_TYPE_STRUCT:
47 case FFI_TYPE_FLOAT:
48 case FFI_TYPE_DOUBLE:
49 cif->flags = cif->rtype->type;
50 break;
52 default:
53 cif->flags = FFI_TYPE_INT;
54 break;
57 return FFI_OK;
60 void
61 ffi_call(ffi_cif *cif, void (*fn)(), void *rvalue, void **avalue)
63 unsigned long *stack, *argp;
64 long i, avn;
65 ffi_type **arg_types;
67 FFI_ASSERT (cif->abi == FFI_OSF);
69 /* If the return value is a struct and we don't have a return
70 value address then we need to make one. */
71 if (rvalue == NULL && cif->flags == FFI_TYPE_STRUCT)
72 rvalue = alloca(cif->rtype->size);
74 /* Allocate the space for the arguments, plus 4 words of temp
75 space for ffi_call_osf. */
76 argp = stack = alloca(cif->bytes + 4*FFI_SIZEOF_ARG);
78 if (cif->flags == FFI_TYPE_STRUCT)
79 *(void **) argp++ = rvalue;
81 i = 0;
82 avn = cif->nargs;
83 arg_types = cif->arg_types;
85 while (i < avn)
87 switch ((*arg_types)->type)
89 case FFI_TYPE_SINT8:
90 *(SINT64 *) argp = *(SINT8 *)(* avalue);
91 break;
93 case FFI_TYPE_UINT8:
94 *(SINT64 *) argp = *(UINT8 *)(* avalue);
95 break;
97 case FFI_TYPE_SINT16:
98 *(SINT64 *) argp = *(SINT16 *)(* avalue);
99 break;
101 case FFI_TYPE_UINT16:
102 *(SINT64 *) argp = *(UINT16 *)(* avalue);
103 break;
105 case FFI_TYPE_SINT32:
106 case FFI_TYPE_UINT32:
107 /* Note that unsigned 32-bit quantities are sign extended. */
108 *(SINT64 *) argp = *(SINT32 *)(* avalue);
109 break;
111 case FFI_TYPE_SINT64:
112 case FFI_TYPE_UINT64:
113 case FFI_TYPE_POINTER:
114 *(UINT64 *) argp = *(UINT64 *)(* avalue);
115 break;
117 case FFI_TYPE_FLOAT:
118 if (argp - stack < 6)
120 /* Note the conversion -- all the fp regs are loaded as
121 doubles. The in-register format is the same. */
122 *(double *) argp = *(float *)(* avalue);
124 else
125 *(float *) argp = *(float *)(* avalue);
126 break;
128 case FFI_TYPE_DOUBLE:
129 *(double *) argp = *(double *)(* avalue);
130 break;
132 case FFI_TYPE_STRUCT:
133 memcpy(argp, *avalue, (*arg_types)->size);
134 break;
136 default:
137 FFI_ASSERT(0);
140 argp += ALIGN((*arg_types)->size, FFI_SIZEOF_ARG) / FFI_SIZEOF_ARG;
141 i++, arg_types++, avalue++;
144 ffi_call_osf(stack, cif->bytes, cif->flags, rvalue, fn);
148 ffi_status
149 ffi_prep_closure (ffi_closure* closure,
150 ffi_cif* cif,
151 void (*fun)(ffi_cif*, void*, void**, void*),
152 void *user_data)
154 unsigned int *tramp;
156 FFI_ASSERT (cif->abi == FFI_OSF);
158 tramp = (unsigned int *) &closure->tramp[0];
159 tramp[0] = 0x47fb0401; /* mov $27,$1 */
160 tramp[1] = 0xa77b0010; /* ldq $27,16($27) */
161 tramp[2] = 0x6bfb0000; /* jmp $31,($27),0 */
162 tramp[3] = 0x47ff041f; /* nop */
163 *(void **) &tramp[4] = ffi_closure_osf;
165 closure->cif = cif;
166 closure->fun = fun;
167 closure->user_data = user_data;
169 /* Flush the Icache.
171 Tru64 UNIX as doesn't understand the imb mnemonic, so use call_pal
172 instead, since both Compaq as and gas can handle it.
174 0x86 is PAL_imb in Tru64 UNIX <alpha/pal.h>. */
175 asm volatile ("call_pal 0x86" : : : "memory");
177 return FFI_OK;
181 ffi_closure_osf_inner(ffi_closure *closure, void *rvalue, unsigned long *argp)
183 ffi_cif *cif;
184 void **avalue;
185 ffi_type **arg_types;
186 long i, avn, argn;
188 cif = closure->cif;
189 avalue = alloca(cif->nargs * sizeof(void *));
191 argn = 0;
193 /* Copy the caller's structure return address to that the closure
194 returns the data directly to the caller. */
195 if (cif->flags == FFI_TYPE_STRUCT)
197 rvalue = (void *) argp[0];
198 argn = 1;
201 i = 0;
202 avn = cif->nargs;
203 arg_types = cif->arg_types;
205 /* Grab the addresses of the arguments from the stack frame. */
206 while (i < avn)
208 switch (arg_types[i]->type)
210 case FFI_TYPE_SINT8:
211 case FFI_TYPE_UINT8:
212 case FFI_TYPE_SINT16:
213 case FFI_TYPE_UINT16:
214 case FFI_TYPE_SINT32:
215 case FFI_TYPE_UINT32:
216 case FFI_TYPE_SINT64:
217 case FFI_TYPE_UINT64:
218 case FFI_TYPE_POINTER:
219 case FFI_TYPE_STRUCT:
220 avalue[i] = &argp[argn];
221 break;
223 case FFI_TYPE_FLOAT:
224 if (argn < 6)
226 /* Floats coming from registers need conversion from double
227 back to float format. */
228 *(float *)&argp[argn - 6] = *(double *)&argp[argn - 6];
229 avalue[i] = &argp[argn - 6];
231 else
232 avalue[i] = &argp[argn];
233 break;
235 case FFI_TYPE_DOUBLE:
236 avalue[i] = &argp[argn - (argn < 6 ? 6 : 0)];
237 break;
239 default:
240 FFI_ASSERT(0);
243 argn += ALIGN(arg_types[i]->size, FFI_SIZEOF_ARG) / FFI_SIZEOF_ARG;
244 i++;
247 /* Invoke the closure. */
248 (closure->fun) (cif, rvalue, avalue, closure->user_data);
250 /* Tell ffi_closure_osf how to perform return type promotions. */
251 return cif->rtype->type;