1991-09-03 22:25:28 -04:00
|
|
|
|
/**
|
|
|
|
|
*
|
|
|
|
|
* Varargs for PYR/GNU CC
|
|
|
|
|
*
|
|
|
|
|
* WARNING -- WARNING -- DANGER
|
|
|
|
|
*
|
|
|
|
|
* The code in this file implements varargs for gcc on a pyr in
|
|
|
|
|
* a way that is compatible with code compiled by the Pyramid Technology
|
|
|
|
|
* C compiler.
|
|
|
|
|
* As such, it depends strongly on the Pyramid conventions for
|
1992-05-07 02:41:23 -04:00
|
|
|
|
* parameter passing.ct and independent implementation.
|
1992-03-14 00:17:01 -05:00
|
|
|
|
* These (somewhat bizarre) parameter-passing conventions are described
|
1991-09-03 22:25:28 -04:00
|
|
|
|
* in the ``OSx Operating System Porting Guide''.
|
|
|
|
|
*
|
|
|
|
|
* A quick summary is useful:
|
|
|
|
|
* 12 of the 48 register-windowed regs available for
|
|
|
|
|
* parameter passing. Parameters of a function call that are eligible
|
|
|
|
|
* to be passed in registers are assigned registers from TR0/PR0 onwards;
|
|
|
|
|
* all other arguments are passed on the stack.
|
|
|
|
|
* Structure and union parameters are *never* passed in registers,
|
|
|
|
|
* even if they are small enough to fit. They are always passed on
|
|
|
|
|
* the stack.
|
|
|
|
|
*
|
|
|
|
|
* Double-sized parameters cannot be passed in TR11, because
|
|
|
|
|
* TR12 is not used for passing parameters. If, in the absence of this
|
|
|
|
|
* rule, a double-sized param would have been passed in TR11,
|
|
|
|
|
* that parameter is passed on the stack and no parameters are
|
|
|
|
|
* passed in TR11.
|
|
|
|
|
*
|
|
|
|
|
* It is only known to work for passing 32-bit integer quantities
|
|
|
|
|
* (ie chars, shorts, ints/enums, longs), doubles, or pointers.
|
|
|
|
|
* Passing structures on a Pyramid via varargs is a loser.
|
|
|
|
|
* Passing an object larger than 8 bytes on a pyramid via varargs may
|
|
|
|
|
* also be a loser.
|
|
|
|
|
*
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* pointer to next stack parameter in __va_buf[0]
|
|
|
|
|
* pointer to next parameter register in __va_buf[1]
|
|
|
|
|
* Count of registers seen at __va_buf[2]
|
|
|
|
|
* saved pr0..pr11 in __va_buf[3..14]
|
|
|
|
|
* # of calls to va_arg (debugging) at __va_buf[15]
|
|
|
|
|
*/
|
|
|
|
|
|
1992-06-26 04:38:37 -04:00
|
|
|
|
/* Define __gnuc_va_list. */
|
|
|
|
|
|
|
|
|
|
#ifndef __GNUC_VA_LIST
|
|
|
|
|
#define __GNUC_VA_LIST
|
|
|
|
|
|
1991-09-03 22:25:28 -04:00
|
|
|
|
typedef void *__voidptr;
|
|
|
|
|
#if 1
|
|
|
|
|
|
|
|
|
|
typedef struct __va_regs {
|
|
|
|
|
__voidptr __stackp,__regp,__count;
|
|
|
|
|
__voidptr __pr0,__pr1,__pr2,__pr3,__pr4,__pr5,__pr6,__pr7,__pr8,__pr9,__pr10,__pr11;
|
|
|
|
|
} __va_regs;
|
|
|
|
|
|
|
|
|
|
typedef __va_regs __va_buf;
|
|
|
|
|
#else
|
|
|
|
|
|
|
|
|
|
/* __va_buf[0] = address of next arg passed on the stack
|
|
|
|
|
__va_buf[1] = address of next arg passed in a register
|
|
|
|
|
__va_buf[2] = register-# of next arg passed in a register
|
|
|
|
|
*/
|
|
|
|
|
typedef __voidptr(*__va_buf);
|
|
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
1992-06-26 04:38:37 -04:00
|
|
|
|
typedef __va_buf __gnuc_va_list;
|
|
|
|
|
|
|
|
|
|
#endif /* not __GNUC_VA_LIST */
|
|
|
|
|
|
|
|
|
|
/* If this is for internal libc use, don't define anything but
|
|
|
|
|
__gnuc_va_list. */
|
|
|
|
|
#if defined (_STDARG_H) || defined (_VARARGS_H)
|
|
|
|
|
|
1991-09-03 22:25:28 -04:00
|
|
|
|
/* In GCC version 2, we want an ellipsis at the end of the declaration
|
|
|
|
|
of the argument list. GCC version 1 can't parse it. */
|
|
|
|
|
|
|
|
|
|
#if __GNUC__ > 1
|
|
|
|
|
#define __va_ellipsis ...
|
|
|
|
|
#else
|
|
|
|
|
#define __va_ellipsis
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#define va_alist \
|
|
|
|
|
__va0,__va1,__va2,__va3,__va4,__va5,__va6,__va7,__va8,__va9,__va10,__va11, \
|
|
|
|
|
__builtin_va_alist
|
|
|
|
|
|
|
|
|
|
/* The ... causes current_function_varargs to be set in cc1. */
|
|
|
|
|
#define va_dcl __voidptr va_alist; __va_ellipsis
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* __asm ("rcsp %0" : "=r" ( _AP [0]));*/
|
|
|
|
|
|
|
|
|
|
#define va_start(_AP) \
|
|
|
|
|
_AP = ((struct __va_regs) { \
|
|
|
|
|
&(_AP.__pr0), (void*)&__builtin_va_alist, (void*)0, \
|
|
|
|
|
__va0,__va1,__va2,__va3,__va4,__va5, \
|
|
|
|
|
__va6,__va7,__va8,__va9,__va10,__va11})
|
1992-03-14 00:17:01 -05:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Avoid errors if compiling GCC v2 with GCC v1. */
|
|
|
|
|
#if __GNUC__ == 1
|
|
|
|
|
#define __extension__
|
|
|
|
|
#endif
|
1991-09-03 22:25:28 -04:00
|
|
|
|
|
1992-09-15 01:08:02 -04:00
|
|
|
|
/* We cast to void * and then to TYPE * because this avoids
|
|
|
|
|
a warning about increasing the alignment requirement. */
|
1991-09-03 22:25:28 -04:00
|
|
|
|
#define va_arg(_AP, _MODE) \
|
1992-01-18 23:13:04 -05:00
|
|
|
|
__extension__ \
|
1991-09-03 22:25:28 -04:00
|
|
|
|
({__voidptr *__ap = (__voidptr*)&_AP; \
|
|
|
|
|
register int __size = sizeof (_MODE); \
|
|
|
|
|
register int __onstack = \
|
|
|
|
|
(__size > 8 || ( (int)(__ap[2]) > 11) || \
|
|
|
|
|
(__size==8 && (int)(__ap[2])==11)); \
|
|
|
|
|
register int* __param_addr = ((int*)((__ap) [__onstack])); \
|
|
|
|
|
\
|
|
|
|
|
((void *)__ap[__onstack])+=__size; \
|
|
|
|
|
if (__onstack==0 || (int)(__ap[2])==11) \
|
|
|
|
|
__ap[2]+= (__size >> 2); \
|
1992-09-15 01:08:02 -04:00
|
|
|
|
*(( _MODE *) (void *) __param_addr); \
|
1991-09-03 22:25:28 -04:00
|
|
|
|
})
|
|
|
|
|
|
1992-06-26 04:38:37 -04:00
|
|
|
|
void va_end (__gnuc_va_list); /* Defined in libgcc.a */
|
1991-09-03 22:25:28 -04:00
|
|
|
|
#define va_end(_X)
|
1992-06-26 04:38:37 -04:00
|
|
|
|
|
|
|
|
|
#endif /* defined (_STDARG_H) || defined (_VARARGS_H) */
|