mirror of
https://github.com/frida/tinycc
synced 2024-12-25 22:46:49 +03:00
0a9873aa22
Most change was done in #ifdef TCC_TARGET_X86_64. So, nothing should be broken by this change. Summary of current status of x86-64 support: - produces x86-64 object files and executables. - the x86-64 code generator is based on x86's. -- for long long integers, we use 64bit registers instead of tcc's generic implementation. -- for float or double, we use SSE. SSE registers are not utilized well (we only use xmm0 and xmm1). -- for long double, we use x87 FPU. - passes make test. - passes ./libtcc_test. - can compile tcc.c. The compiled tcc can compile tcc.c, too. (there should be some bugs since the binary size of tcc2 and tcc3 is differ where tcc tcc.c -o tcc2 and tcc2 tcc.c -o tcc3) - can compile links browser. It seems working. - not tested well. I tested this work only on my linux box with few programs. - calling convention of long-double-integer or struct is not exactly the same as GCC's x86-64 ABI. - implementation of tcc -run is naive (tcc -run tcctest.c works, but tcc -run tcc.c doesn't work). Relocating 64bit addresses seems to be not as simple as 32bit environments. - shared object support isn't unimplemented - no bounds checker support - some builtin functions such as __divdi3 aren't supported
88 lines
3.0 KiB
C
88 lines
3.0 KiB
C
#ifndef _STDARG_H
|
|
#define _STDARG_H
|
|
|
|
#ifdef __x86_64__
|
|
|
|
#ifdef __TINYC__
|
|
|
|
#include <stdlib.h>
|
|
|
|
/* GCC compatible definition of va_list. */
|
|
struct __va_list_struct {
|
|
unsigned int gp_offset;
|
|
unsigned int fp_offset;
|
|
union {
|
|
unsigned int overflow_offset;
|
|
char *overflow_arg_area;
|
|
};
|
|
char *reg_save_area;
|
|
};
|
|
|
|
typedef struct __va_list_struct *va_list;
|
|
|
|
/* avoid #define malloc tcc_malloc.
|
|
XXX: add __malloc or something into libtcc? */
|
|
inline void *__va_list_malloc(size_t size) { return malloc(size); }
|
|
inline void __va_list_free(void *ptr) { free(ptr); }
|
|
|
|
/* XXX: this lacks the support of aggregated types. */
|
|
#define va_start(ap, last) \
|
|
(ap = (va_list)__va_list_malloc(sizeof(struct __va_list_struct)), \
|
|
*ap = *(struct __va_list_struct*)( \
|
|
(char*)__builtin_frame_address(0) - 16), \
|
|
ap->overflow_arg_area = ((char *)__builtin_frame_address(0) + \
|
|
ap->overflow_offset), \
|
|
ap->reg_save_area = (char *)__builtin_frame_address(0) - 176 - 16 \
|
|
)
|
|
#define va_arg(ap, type) \
|
|
(*(type*)(__builtin_types_compatible_p(type, long double) \
|
|
? (ap->overflow_arg_area += 16, \
|
|
ap->overflow_arg_area - 16) \
|
|
: __builtin_types_compatible_p(type, double) \
|
|
? (ap->fp_offset < 128 + 48 \
|
|
? (ap->fp_offset += 16, \
|
|
ap->reg_save_area + ap->fp_offset - 16) \
|
|
: (ap->overflow_arg_area += 8, \
|
|
ap->overflow_arg_area - 8)) \
|
|
: (ap->gp_offset < 48 \
|
|
? (ap->gp_offset += 8, \
|
|
ap->reg_save_area + ap->gp_offset - 8) \
|
|
: (ap->overflow_arg_area += 8, \
|
|
ap->overflow_arg_area - 8)) \
|
|
))
|
|
#define va_copy(dest, src) \
|
|
((dest) = (va_list)malloc(sizeof(struct __va_list_struct)), \
|
|
*(dest) = *(src))
|
|
#define va_end(ap) __va_list_free(ap)
|
|
|
|
#else
|
|
|
|
/* for GNU C */
|
|
|
|
typedef __builtin_va_list va_list;
|
|
|
|
#define va_start(ap, last) __builtin_va_start(ap, last)
|
|
#define va_arg(ap, type) __builtin_va_arg(ap, type)
|
|
#define va_copy(dest, src) __builtin_va_copy(dest, src)
|
|
#define va_end(ap) __builtin_va_end(ap)
|
|
|
|
#endif
|
|
|
|
#else
|
|
|
|
typedef char *va_list;
|
|
|
|
/* only correct for i386 */
|
|
#define va_start(ap,last) ap = ((char *)&(last)) + ((sizeof(last)+3)&~3)
|
|
#define va_arg(ap,type) (ap += (sizeof(type)+3)&~3, *(type *)(ap - ((sizeof(type)+3)&~3)))
|
|
#define va_copy(dest, src) (dest) = (src)
|
|
#define va_end(ap)
|
|
|
|
#endif
|
|
|
|
/* fix a buggy dependency on GCC in libio.h */
|
|
typedef va_list __gnuc_va_list;
|
|
#define _VA_LIST_DEFINED
|
|
|
|
#endif /* _STDARG_H */
|