mirror of
https://github.com/mirror/tinycc.git
synced 2025-01-15 05:20:06 +08:00
New implementation of va_list/va_start/var_copy that do not use dynamic memory, with this when compiling fossil-scm with tcc on linux X86_64 it works fine.
This commit is contained in:
parent
700c2f769b
commit
c025478d7c
@ -4,18 +4,28 @@
|
|||||||
#ifdef __x86_64__
|
#ifdef __x86_64__
|
||||||
#ifndef _WIN64
|
#ifndef _WIN64
|
||||||
|
|
||||||
typedef void *va_list;
|
//This should be in sync with the declaration on our lib/libtcc1.c
|
||||||
|
/* GCC compatible definition of va_list. */
|
||||||
|
typedef struct {
|
||||||
|
unsigned int gp_offset;
|
||||||
|
unsigned int fp_offset;
|
||||||
|
union {
|
||||||
|
unsigned int overflow_offset;
|
||||||
|
char *overflow_arg_area;
|
||||||
|
};
|
||||||
|
char *reg_save_area;
|
||||||
|
} __va_list_struct;
|
||||||
|
|
||||||
va_list __va_start(void *fp);
|
typedef __va_list_struct va_list;
|
||||||
void *__va_arg(va_list ap, int arg_type, int size, int align);
|
|
||||||
va_list __va_copy(va_list src);
|
|
||||||
void __va_end(va_list ap);
|
|
||||||
|
|
||||||
#define va_start(ap, last) ((ap) = __va_start(__builtin_frame_address(0)))
|
void __va_start(__va_list_struct *ap, void *fp);
|
||||||
|
void *__va_arg(__va_list_struct *ap, int arg_type, int size, int align);
|
||||||
|
|
||||||
|
#define va_start(ap, last) __va_start(&ap, __builtin_frame_address(0))
|
||||||
#define va_arg(ap, type) \
|
#define va_arg(ap, type) \
|
||||||
(*(type *)(__va_arg(ap, __builtin_va_arg_types(type), sizeof(type), __alignof__(type))))
|
(*(type *)(__va_arg(&ap, __builtin_va_arg_types(type), sizeof(type), __alignof__(type))))
|
||||||
#define va_copy(dest, src) ((dest) = __va_copy(src))
|
#define va_copy(dest, src) ((dest) = (src))
|
||||||
#define va_end(ap) __va_end(ap)
|
#define va_end(ap)
|
||||||
|
|
||||||
#else /* _WIN64 */
|
#else /* _WIN64 */
|
||||||
typedef char *va_list;
|
typedef char *va_list;
|
||||||
|
@ -626,10 +626,12 @@ long long __fixxfdi (long double a1)
|
|||||||
#ifndef __TINYC__
|
#ifndef __TINYC__
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
|
#include <string.h>
|
||||||
#else
|
#else
|
||||||
/* Avoid including stdlib.h because it is not easily available when
|
/* Avoid including stdlib.h because it is not easily available when
|
||||||
cross compiling */
|
cross compiling */
|
||||||
extern void *malloc(unsigned long long);
|
extern void *malloc(unsigned long long);
|
||||||
|
void *memset(void *s, int c, size_t n);
|
||||||
extern void free(void*);
|
extern void free(void*);
|
||||||
extern void abort(void);
|
extern void abort(void);
|
||||||
#endif
|
#endif
|
||||||
@ -638,8 +640,9 @@ enum __va_arg_type {
|
|||||||
__va_gen_reg, __va_float_reg, __va_stack
|
__va_gen_reg, __va_float_reg, __va_stack
|
||||||
};
|
};
|
||||||
|
|
||||||
|
//This should be in sync with the declaration on our include/stdarg.h
|
||||||
/* GCC compatible definition of va_list. */
|
/* GCC compatible definition of va_list. */
|
||||||
struct __va_list_struct {
|
typedef struct {
|
||||||
unsigned int gp_offset;
|
unsigned int gp_offset;
|
||||||
unsigned int fp_offset;
|
unsigned int fp_offset;
|
||||||
union {
|
union {
|
||||||
@ -647,24 +650,22 @@ struct __va_list_struct {
|
|||||||
char *overflow_arg_area;
|
char *overflow_arg_area;
|
||||||
};
|
};
|
||||||
char *reg_save_area;
|
char *reg_save_area;
|
||||||
};
|
} __va_list_struct;
|
||||||
|
|
||||||
#undef __va_start
|
#undef __va_start
|
||||||
#undef __va_arg
|
#undef __va_arg
|
||||||
#undef __va_copy
|
#undef __va_copy
|
||||||
#undef __va_end
|
#undef __va_end
|
||||||
|
|
||||||
void *__va_start(void *fp)
|
void __va_start(__va_list_struct *ap, void *fp)
|
||||||
{
|
{
|
||||||
struct __va_list_struct *ap =
|
memset(ap, 0, sizeof(__va_list_struct));
|
||||||
(struct __va_list_struct *)malloc(sizeof(struct __va_list_struct));
|
*ap = *(__va_list_struct *)((char *)fp - 16);
|
||||||
*ap = *(struct __va_list_struct *)((char *)fp - 16);
|
|
||||||
ap->overflow_arg_area = (char *)fp + ap->overflow_offset;
|
ap->overflow_arg_area = (char *)fp + ap->overflow_offset;
|
||||||
ap->reg_save_area = (char *)fp - 176 - 16;
|
ap->reg_save_area = (char *)fp - 176 - 16;
|
||||||
return ap;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void *__va_arg(struct __va_list_struct *ap,
|
void *__va_arg(__va_list_struct *ap,
|
||||||
enum __va_arg_type arg_type,
|
enum __va_arg_type arg_type,
|
||||||
int size, int align)
|
int size, int align)
|
||||||
{
|
{
|
||||||
@ -701,19 +702,6 @@ void *__va_arg(struct __va_list_struct *ap,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void *__va_copy(struct __va_list_struct *src)
|
|
||||||
{
|
|
||||||
struct __va_list_struct *dest =
|
|
||||||
(struct __va_list_struct *)malloc(sizeof(struct __va_list_struct));
|
|
||||||
*dest = *src;
|
|
||||||
return dest;
|
|
||||||
}
|
|
||||||
|
|
||||||
void __va_end(struct __va_list_struct *ap)
|
|
||||||
{
|
|
||||||
free(ap);
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif /* __x86_64__ */
|
#endif /* __x86_64__ */
|
||||||
|
|
||||||
/* Flushing for tccrun */
|
/* Flushing for tccrun */
|
||||||
|
Loading…
Reference in New Issue
Block a user