mirror of
https://review.haiku-os.org/haiku
synced 2025-01-22 22:34:48 +01:00
404a0fea46
builds on Linux with gcc 2 and 4, but the images do even run. :-) Not tested on BeOS. * Removed stddef.h and stdarg.h. They are provided by the compiler. * Adjusted size_t.h, wchar_t.h, and wchar.h accordingly. * Made stdio.h avoid gcc 2.95.3's fixincludes hack stdio_va_list * Added gcc 2.95.3 headers to the repository. They are used instead of the headers of the gcc 2.95.3's we use to compile Haiku with. Should avoid build problems with the BeOS native compiler. For sake of personal recreation you can rebuild the cross gcc 2.95.3, but the only thing that changed is its header directory (lib/gcc-lib/.../include), which isn't used anymore. Replacing it with headers/build/gcc-2.95.3 should have the same effect as rebuilding, BTW. git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@24413 a95241bf-73f2-0310-859d-f6bbb57e9c96
131 lines
4.0 KiB
C
131 lines
4.0 KiB
C
/**
|
||
*
|
||
* 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
|
||
* parameter passing.ct and independent implementation.
|
||
* These (somewhat bizarre) parameter-passing conventions are described
|
||
* 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]
|
||
*/
|
||
|
||
/* Define __gnuc_va_list. */
|
||
|
||
#ifndef __GNUC_VA_LIST
|
||
#define __GNUC_VA_LIST
|
||
|
||
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
|
||
|
||
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)
|
||
|
||
/* 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})
|
||
|
||
|
||
/* Avoid errors if compiling GCC v2 with GCC v1. */
|
||
#if __GNUC__ == 1
|
||
#define __extension__
|
||
#endif
|
||
|
||
/* We cast to void * and then to TYPE * because this avoids
|
||
a warning about increasing the alignment requirement. */
|
||
#define va_arg(_AP, _MODE) \
|
||
__extension__ \
|
||
(*({__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); \
|
||
(( _MODE *) (void *) __param_addr); \
|
||
}))
|
||
|
||
void va_end (__gnuc_va_list); /* Defined in libgcc.a */
|
||
#define va_end(_X) ((void)0)
|
||
|
||
#endif /* defined (_STDARG_H) || defined (_VARARGS_H) */
|