win32: wincon.h: support more console mode flags
[tinycc.git] / lib / va_list.c
blob1fb55127cef60b5d8e17b0a646c6915b46ed1fce
1 /* va_list.c - tinycc support for va_list on X86_64 */
3 #if defined __x86_64__
5 /* Avoid include files, they may not be available when cross compiling */
6 extern void abort(void);
8 /* This should be in sync with our include/stdarg.h */
9 enum __va_arg_type {
10 __va_gen_reg, __va_float_reg, __va_stack
13 /* GCC compatible definition of va_list. */
14 /*predefined by TCC (tcc_predefs.h):
15 typedef struct {
16 unsigned int gp_offset;
17 unsigned int fp_offset;
18 union {
19 unsigned int overflow_offset;
20 char *overflow_arg_area;
22 char *reg_save_area;
23 } __builtin_va_list[1];
26 extern void *memcpy(void *dest, const void *src, unsigned long n);
28 void *__va_arg(__builtin_va_list ap,
29 int arg_type,
30 int size, int align)
32 size = (size + 7) & ~7;
33 align = (align + 7) & ~7;
34 switch ((enum __va_arg_type)arg_type) {
35 case __va_gen_reg:
36 if (ap->gp_offset + size <= 48) {
37 ap->gp_offset += size;
38 return ap->reg_save_area + ap->gp_offset - size;
40 goto use_overflow_area;
42 case __va_float_reg:
43 if (ap->fp_offset < 128 + 48) {
44 ap->fp_offset += 16;
45 if (size == 8)
46 return ap->reg_save_area + ap->fp_offset - 16;
47 if (ap->fp_offset < 128 + 48) {
48 memcpy(ap->reg_save_area + ap->fp_offset - 8,
49 ap->reg_save_area + ap->fp_offset, 8);
50 ap->fp_offset += 16;
51 return ap->reg_save_area + ap->fp_offset - 32;
54 goto use_overflow_area;
56 case __va_stack:
57 use_overflow_area:
58 ap->overflow_arg_area += size;
59 ap->overflow_arg_area = (char*)((long long)(ap->overflow_arg_area + align - 1) & -align);
60 return ap->overflow_arg_area - size;
62 default: /* should never happen */
63 abort();
64 return 0;
67 #endif