Import more changesets from Rob Landley's fork (part 2)

This commit is contained in:
grischka 2007-11-21 17:16:31 +00:00
parent 54bf8c0556
commit d778bde7f9
10 changed files with 157 additions and 79 deletions

View File

@ -1,5 +1,21 @@
version 0.9.24: version 0.9.24:
- Import more changesets from Rob Landley's fork (part 2):
487: Handle long long constants in gen_opic() (Rob Landley)
484: Handle parentheses within __attribute__((...)) (Rob Landley)
480: Remove a goto in decl_initializer_alloc (Rob Landley)
475: Fix dereferences in inline assembly output (Joshua Phillips)
474: Cast ptrs to ints of different sizes correctly (Joshua Phillips)
473: Fix size of structs with empty array member (Joshua Phillips)
470: No warning for && and || with mixed pointers/integers (Rob Landley)
469: Fix symbol visibility problems in the linker (Vincent Pit)
468: Allow && and || involving pointer arguments (Rob Landley)
455: Optimize case labels with no code in between (Zdenek Pavlas)
450: Implement alloca for x86 (grischka)
415: Parse unicode escape sequences (Axel Liljencrantz)
407: Add a simple va_copy() in stdarg.h (Hasso Tepper)
400: Allow typedef names as symbols (Dave Dodge)
- Import some changesets from Rob Landley's fork (part 1): - Import some changesets from Rob Landley's fork (part 1):
462: Use LGPL with bcheck.c and il-gen.c 462: Use LGPL with bcheck.c and il-gen.c
458: Fix global compound literals (in unary: case '&':) (Andrew Johnson) 458: Fix global compound literals (in unary: case '&':) (Andrew Johnson)

View File

@ -176,6 +176,9 @@ else
LIBTCC1_OBJS=libtcc1.o LIBTCC1_OBJS=libtcc1.o
LIBTCC1_CC=$(CC) LIBTCC1_CC=$(CC)
endif endif
ifeq ($(ARCH),i386)
LIBTCC1_OBJS+=alloca86.o alloca86-bt.o
endif
%.o: %.c %.o: %.c
$(LIBTCC1_CC) -O2 -Wall -c -o $@ $< $(LIBTCC1_CC) -O2 -Wall -c -o $@ $<
@ -237,7 +240,11 @@ libinstall: libtcc.a
$(INSTALL) -m644 libtcc.h "$(includedir)" $(INSTALL) -m644 libtcc.h "$(includedir)"
libtcc.o: tcc.c i386-gen.c Makefile libtcc.o: tcc.c i386-gen.c Makefile
ifdef CONFIG_WIN32
$(CC) $(CFLAGS) -DTCC_TARGET_PE -DLIBTCC -c -o $@ $<
else
$(CC) $(CFLAGS) -DLIBTCC -c -o $@ $< $(CC) $(CFLAGS) -DLIBTCC -c -o $@ $<
endif
libtcc.a: libtcc.o libtcc.a: libtcc.o
$(AR) rcs $@ $^ $(AR) rcs $@ $^
@ -282,11 +289,11 @@ cache: tcc_g
# documentation and man page # documentation and man page
tcc-doc.html: tcc-doc.texi tcc-doc.html: tcc-doc.texi
texi2html -monolithic -number $< -texi2html -monolithic -number $<
tcc.1: tcc-doc.texi tcc.1: tcc-doc.texi
./texi2pod.pl $< tcc.pod -./texi2pod.pl $< tcc.pod
pod2man --section=1 --center=" " --release=" " tcc.pod > $@ -pod2man --section=1 --center=" " --release=" " tcc.pod > $@
FILE=tcc-$(shell cat VERSION) FILE=tcc-$(shell cat VERSION)

View File

@ -18,4 +18,9 @@ typedef int int32_t;
typedef long long int int64_t; typedef long long int int64_t;
#endif #endif
#ifdef __i386__
void *_alloca(size_t);
#define alloca _alloca
#endif
#endif #endif

141
tcc.c
View File

@ -1290,6 +1290,7 @@ static void put_extern_sym2(Sym *sym, Section *section,
case TOK_memset: case TOK_memset:
case TOK_strlen: case TOK_strlen:
case TOK_strcpy: case TOK_strcpy:
case TOK__alloca:
strcpy(buf, "__bound_"); strcpy(buf, "__bound_");
strcat(buf, name); strcat(buf, name);
name = buf; name = buf;
@ -3123,6 +3124,8 @@ static void parse_escape_string(CString *outstr, const uint8_t *buf, int is_long
c = n; c = n;
goto add_char_nonext; goto add_char_nonext;
case 'x': case 'x':
case 'u':
case 'U':
p++; p++;
n = 0; n = 0;
for(;;) { for(;;) {
@ -5329,23 +5332,29 @@ void gen_opl(int op)
independent opt */ independent opt */
void gen_opic(int op) void gen_opic(int op)
{ {
int fc, c1, c2, n; int c1, c2, t1, t2, n, c;
SValue *v1, *v2; SValue *v1, *v2;
long long l1, l2, l;
typedef unsigned long long U;
v1 = vtop - 1; v1 = vtop - 1;
v2 = vtop; v2 = vtop;
t1 = v1->type.t & VT_BTYPE;
t2 = v2->type.t & VT_BTYPE;
l1 = (t1 == VT_LLONG) ? v1->c.ll : v1->c.i;
l2 = (t2 == VT_LLONG) ? v2->c.ll : v2->c.i;
/* currently, we cannot do computations with forward symbols */ /* currently, we cannot do computations with forward symbols */
c1 = (v1->r & (VT_VALMASK | VT_LVAL | VT_SYM)) == VT_CONST; c1 = (v1->r & (VT_VALMASK | VT_LVAL | VT_SYM)) == VT_CONST;
c2 = (v2->r & (VT_VALMASK | VT_LVAL | VT_SYM)) == VT_CONST; c2 = (v2->r & (VT_VALMASK | VT_LVAL | VT_SYM)) == VT_CONST;
if (c1 && c2) { if (c1 && c2) {
fc = v2->c.i;
switch(op) { switch(op) {
case '+': v1->c.i += fc; break; case '+': l1 += l2; break;
case '-': v1->c.i -= fc; break; case '-': l1 -= l2; break;
case '&': v1->c.i &= fc; break; case '&': l1 &= l2; break;
case '^': v1->c.i ^= fc; break; case '^': l1 ^= l2; break;
case '|': v1->c.i |= fc; break; case '|': l1 |= l2; break;
case '*': v1->c.i *= fc; break; case '*': l1 *= l2; break;
case TOK_PDIV: case TOK_PDIV:
case '/': case '/':
@ -5353,66 +5362,68 @@ void gen_opic(int op)
case TOK_UDIV: case TOK_UDIV:
case TOK_UMOD: case TOK_UMOD:
/* if division by zero, generate explicit division */ /* if division by zero, generate explicit division */
if (fc == 0) { if (l2 == 0) {
if (const_wanted) if (const_wanted)
error("division by zero in constant"); error("division by zero in constant");
goto general_case; goto general_case;
} }
switch(op) { switch(op) {
default: v1->c.i /= fc; break; default: l1 /= l2; break;
case '%': v1->c.i %= fc; break; case '%': l1 %= l2; break;
case TOK_UDIV: v1->c.i = (unsigned)v1->c.i / fc; break; case TOK_UDIV: l1 = (U)l1 / l2; break;
case TOK_UMOD: v1->c.i = (unsigned)v1->c.i % fc; break; case TOK_UMOD: l1 = (U)l1 % l2; break;
} }
break; break;
case TOK_SHL: v1->c.i <<= fc; break; case TOK_SHL: l1 <<= l2; break;
case TOK_SHR: v1->c.i = (unsigned)v1->c.i >> fc; break; case TOK_SHR: l1 = (U)l1 >> l2; break;
case TOK_SAR: v1->c.i >>= fc; break; case TOK_SAR: l1 >>= l2; break;
/* tests */ /* tests */
case TOK_ULT: v1->c.i = (unsigned)v1->c.i < (unsigned)fc; break; case TOK_ULT: l1 = (U)l1 < (U)l2; break;
case TOK_UGE: v1->c.i = (unsigned)v1->c.i >= (unsigned)fc; break; case TOK_UGE: l1 = (U)l1 >= (U)l2; break;
case TOK_EQ: v1->c.i = v1->c.i == fc; break; case TOK_EQ: l1 = l1 == l2; break;
case TOK_NE: v1->c.i = v1->c.i != fc; break; case TOK_NE: l1 = l1 != l2; break;
case TOK_ULE: v1->c.i = (unsigned)v1->c.i <= (unsigned)fc; break; case TOK_ULE: l1 = (U)l1 <= (U)l2; break;
case TOK_UGT: v1->c.i = (unsigned)v1->c.i > (unsigned)fc; break; case TOK_UGT: l1 = (U)l1 > (U)l2; break;
case TOK_LT: v1->c.i = v1->c.i < fc; break; case TOK_LT: l1 = l1 < l2; break;
case TOK_GE: v1->c.i = v1->c.i >= fc; break; case TOK_GE: l1 = l1 >= l2; break;
case TOK_LE: v1->c.i = v1->c.i <= fc; break; case TOK_LE: l1 = l1 <= l2; break;
case TOK_GT: v1->c.i = v1->c.i > fc; break; case TOK_GT: l1 = l1 > l2; break;
/* logical */ /* logical */
case TOK_LAND: v1->c.i = v1->c.i && fc; break; case TOK_LAND: l1 = l1 && l2; break;
case TOK_LOR: v1->c.i = v1->c.i || fc; break; case TOK_LOR: l1 = l1 || l2; break;
default: default:
goto general_case; goto general_case;
} }
v1->c.ll = l1;
vtop--; vtop--;
} else { } else {
/* if commutative ops, put c2 as constant */ /* if commutative ops, put c2 as constant */
if (c1 && (op == '+' || op == '&' || op == '^' || if (c1 && (op == '+' || op == '&' || op == '^' ||
op == '|' || op == '*')) { op == '|' || op == '*')) {
vswap(); vswap();
swap(&c1, &c2); c = c1, c1 = c2, c2 = c;
l = l1, l1 = l2, l2 = l;
} }
fc = vtop->c.i; /* Filter out NOP operations like x*1, x-0, x&-1... */
if (c2 && (((op == '*' || op == '/' || op == TOK_UDIV || if (c2 && (((op == '*' || op == '/' || op == TOK_UDIV ||
op == TOK_PDIV) && op == TOK_PDIV) &&
fc == 1) || l2 == 1) ||
((op == '+' || op == '-' || op == '|' || op == '^' || ((op == '+' || op == '-' || op == '|' || op == '^' ||
op == TOK_SHL || op == TOK_SHR || op == TOK_SAR) && op == TOK_SHL || op == TOK_SHR || op == TOK_SAR) &&
fc == 0) || l2 == 0) ||
(op == '&' && (op == '&' &&
fc == -1))) { l2 == -1))) {
/* nothing to do */ /* nothing to do */
vtop--; vtop--;
} else if (c2 && (op == '*' || op == TOK_PDIV || op == TOK_UDIV)) { } else if (c2 && (op == '*' || op == TOK_PDIV || op == TOK_UDIV)) {
/* try to use shifts instead of muls or divs */ /* try to use shifts instead of muls or divs */
if (fc > 0 && (fc & (fc - 1)) == 0) { if (l2 > 0 && (l2 & (l2 - 1)) == 0) {
n = -1; n = -1;
while (fc) { while (l2) {
fc >>= 1; l2 >>= 1;
n++; n++;
} }
vtop->c.i = n; vtop->c.ll = n;
if (op == '*') if (op == '*')
op = TOK_SHL; op = TOK_SHL;
else if (op == TOK_PDIV) else if (op == TOK_PDIV)
@ -5426,13 +5437,16 @@ void gen_opic(int op)
(VT_CONST | VT_SYM)) { (VT_CONST | VT_SYM)) {
/* symbol + constant case */ /* symbol + constant case */
if (op == '-') if (op == '-')
fc = -fc; l2 = -l2;
vtop--; vtop--;
vtop->c.i += fc; vtop->c.ll += l2;
} else { } else {
general_case: general_case:
if (!nocode_wanted) { if (!nocode_wanted) {
/* call low level op generator */ /* call low level op generator */
if (t1 == VT_LLONG || t2 == VT_LLONG)
gen_opl(op);
else
gen_opi(op); gen_opi(op);
} else { } else {
vtop--; vtop--;
@ -5540,6 +5554,7 @@ static void check_comparison_pointer_types(SValue *p1, SValue *p2, int op)
bt2 = type2->t & VT_BTYPE; bt2 = type2->t & VT_BTYPE;
/* accept comparison between pointer and integer with a warning */ /* accept comparison between pointer and integer with a warning */
if ((is_integer_btype(bt1) || is_integer_btype(bt2)) && op != '-') { if ((is_integer_btype(bt1) || is_integer_btype(bt2)) && op != '-') {
if (op != TOK_LOR && op != TOK_LAND )
warning("comparison between pointer and integer"); warning("comparison between pointer and integer");
return; return;
} }
@ -5586,7 +5601,7 @@ void gen_op(int op)
if (bt1 == VT_PTR || bt2 == VT_PTR) { if (bt1 == VT_PTR || bt2 == VT_PTR) {
/* at least one operand is a pointer */ /* at least one operand is a pointer */
/* relationnal op: must be both pointers */ /* relationnal op: must be both pointers */
if (op >= TOK_ULT && op <= TOK_GT) { if (op >= TOK_ULT && op <= TOK_LOR) {
check_comparison_pointer_types(vtop - 1, vtop, op); check_comparison_pointer_types(vtop - 1, vtop, op);
/* pointers are handled are unsigned */ /* pointers are handled are unsigned */
t = VT_INT | VT_UNSIGNED; t = VT_INT | VT_UNSIGNED;
@ -5696,8 +5711,6 @@ void gen_op(int op)
gen_cast(&type1); gen_cast(&type1);
if (is_float(t)) if (is_float(t))
gen_opif(op); gen_opif(op);
else if ((t & VT_BTYPE) == VT_LLONG)
gen_opl(op);
else else
gen_opic(op); gen_opic(op);
if (op >= TOK_ULT && op <= TOK_GT) { if (op >= TOK_ULT && op <= TOK_GT) {
@ -5947,6 +5960,11 @@ static void gen_cast(CType *type)
the lvalue already contains the real type size (see the lvalue already contains the real type size (see
VT_LVAL_xxx constants) */ VT_LVAL_xxx constants) */
} }
} else if ((dbt & VT_BTYPE) == VT_PTR && !(vtop->r & VT_LVAL)) {
/* if we are casting between pointer types,
we must update the VT_LVAL_xxx size */
vtop->r = (vtop->r & ~VT_LVAL_TYPE)
| (lvalue_type(type->ref->type.t) & VT_LVAL_TYPE);
} }
vtop->type = *type; vtop->type = *type;
} }
@ -6502,12 +6520,15 @@ static void parse_attribute(AttributeDef *ad)
if (tcc_state->warn_unsupported) if (tcc_state->warn_unsupported)
warning("'%s' attribute ignored", get_tok_str(t, NULL)); warning("'%s' attribute ignored", get_tok_str(t, NULL));
/* skip parameters */ /* skip parameters */
/* XXX: skip parenthesis too */
if (tok == '(') { if (tok == '(') {
int parenthesis = 0;
do {
if (tok == '(')
parenthesis++;
else if (tok == ')')
parenthesis--;
next(); next();
while (tok != ')' && tok != -1) } while (parenthesis && tok != -1);
next();
next();
} }
break; break;
} }
@ -6668,6 +6689,7 @@ static void struct_decl(CType *type, int u)
if (a == TOK_STRUCT) { if (a == TOK_STRUCT) {
c = (c + align - 1) & -align; c = (c + align - 1) & -align;
offset = c; offset = c;
if (size > 0)
c += size; c += size;
} else { } else {
offset = 0; offset = 0;
@ -6719,13 +6741,14 @@ static void struct_decl(CType *type, int u)
*/ */
static int parse_btype(CType *type, AttributeDef *ad) static int parse_btype(CType *type, AttributeDef *ad)
{ {
int t, u, type_found, typespec_found; int t, u, type_found, typespec_found, typedef_found;
Sym *s; Sym *s;
CType type1; CType type1;
memset(ad, 0, sizeof(AttributeDef)); memset(ad, 0, sizeof(AttributeDef));
type_found = 0; type_found = 0;
typespec_found = 0; typespec_found = 0;
typedef_found = 0;
t = 0; t = 0;
while(1) { while(1) {
switch(tok) { switch(tok) {
@ -6858,11 +6881,12 @@ static int parse_btype(CType *type, AttributeDef *ad)
parse_expr_type(&type1); parse_expr_type(&type1);
goto basic_type2; goto basic_type2;
default: default:
if (typespec_found) if (typespec_found || typedef_found)
goto the_end; goto the_end;
s = sym_find(tok); s = sym_find(tok);
if (!s || !(s->type.t & VT_TYPEDEF)) if (!s || !(s->type.t & VT_TYPEDEF))
goto the_end; goto the_end;
typedef_found = 1;
t |= (s->type.t & ~VT_TYPEDEF); t |= (s->type.t & ~VT_TYPEDEF);
type->ref = s->type.ref; type->ref = s->type.ref;
next(); next();
@ -8164,6 +8188,9 @@ static void block(int *bsym, int *csym, int *case_sym, int *def_sym,
int v1, v2; int v1, v2;
if (!case_sym) if (!case_sym)
expect("switch"); expect("switch");
/* since a case is like a label, we must skip it with a jmp */
b = gjmp(0);
next_case:
next(); next();
v1 = expr_const(); v1 = expr_const();
v2 = v1; v2 = v1;
@ -8173,24 +8200,26 @@ static void block(int *bsym, int *csym, int *case_sym, int *def_sym,
if (v2 < v1) if (v2 < v1)
warning("empty case range"); warning("empty case range");
} }
/* since a case is like a label, we must skip it with a jmp */
b = gjmp(0);
gsym(*case_sym); gsym(*case_sym);
vseti(case_reg, 0); vseti(case_reg, 0);
vpushi(v1); vpushi(v1);
if (v1 == v2) { if (v1 == v2) {
gen_op(TOK_EQ); gen_op(TOK_EQ);
*case_sym = gtst(1, 0); *case_sym = 0;
} else { } else {
gen_op(TOK_GE); gen_op(TOK_GE);
*case_sym = gtst(1, 0); *case_sym = gtst(1, 0);
vseti(case_reg, 0); vseti(case_reg, 0);
vpushi(v2); vpushi(v2);
gen_op(TOK_LE); gen_op(TOK_LE);
*case_sym = gtst(1, *case_sym);
} }
gsym(b);
skip(':'); skip(':');
if (tok == TOK_CASE) {
b = gtst(0, b);
goto next_case;
}
*case_sym = gtst(1, *case_sym);
gsym(b);
is_expr = 0; is_expr = 0;
goto block_after_label; goto block_after_label;
} else } else
@ -8873,11 +8902,7 @@ static void decl_initializer_alloc(CType *type, AttributeDef *ad, int r,
} }
if (v) { if (v) {
if (scope == VT_CONST) { if (scope != VT_CONST || !sym) {
if (!sym)
goto do_def;
} else {
do_def:
sym = sym_push(v, type, r | VT_SYM, 0); sym = sym_push(v, type, r | VT_SYM, 0);
} }
/* update symbol definition */ /* update symbol definition */

View File

@ -806,7 +806,7 @@ static void subst_asm_operands(ASMOperand *operands, int nb_operands,
sv = *op->vt; sv = *op->vt;
if (op->reg >= 0) { if (op->reg >= 0) {
sv.r = op->reg; sv.r = op->reg;
if ((op->vt->r & VT_VALMASK) == VT_LLOCAL) if ((op->vt->r & VT_VALMASK) == VT_LLOCAL && op->is_memory)
sv.r |= VT_LVAL; sv.r |= VT_LVAL;
} }
subst_asm_operand(out_str, &sv, modifier); subst_asm_operand(out_str, &sv, modifier);

View File

@ -182,9 +182,11 @@ static int add_elf_sym(Section *s, unsigned long value, unsigned long size,
{ {
Elf32_Sym *esym; Elf32_Sym *esym;
int sym_bind, sym_index, sym_type, esym_bind; int sym_bind, sym_index, sym_type, esym_bind;
unsigned char sym_vis, esym_vis, new_vis;
sym_bind = ELF32_ST_BIND(info); sym_bind = ELF32_ST_BIND(info);
sym_type = ELF32_ST_TYPE(info); sym_type = ELF32_ST_TYPE(info);
sym_vis = ELF32_ST_VISIBILITY(other);
if (sym_bind != STB_LOCAL) { if (sym_bind != STB_LOCAL) {
/* we search global or weak symbols */ /* we search global or weak symbols */
@ -194,6 +196,19 @@ static int add_elf_sym(Section *s, unsigned long value, unsigned long size,
esym = &((Elf32_Sym *)s->data)[sym_index]; esym = &((Elf32_Sym *)s->data)[sym_index];
if (esym->st_shndx != SHN_UNDEF) { if (esym->st_shndx != SHN_UNDEF) {
esym_bind = ELF32_ST_BIND(esym->st_info); esym_bind = ELF32_ST_BIND(esym->st_info);
/* propagate the most constraining visibility */
/* STV_DEFAULT(0)<STV_PROTECTED(3)<STV_HIDDEN(2)<STV_INTERNAL(1) */
esym_vis = ELF32_ST_VISIBILITY(esym->st_other);
if (esym_vis == STV_DEFAULT) {
new_vis = sym_vis;
} else if (sym_vis == STV_DEFAULT) {
new_vis = esym_vis;
} else {
new_vis = (esym_vis < sym_vis) ? esym_vis : sym_vis;
}
esym->st_other = (esym->st_other & ~ELF32_ST_VISIBILITY(-1))
| new_vis;
other = esym->st_other; /* in case we have to patch esym */
if (sh_num == SHN_UNDEF) { if (sh_num == SHN_UNDEF) {
/* ignore adding of undefined symbol if the /* ignore adding of undefined symbol if the
corresponding symbol is already defined */ corresponding symbol is already defined */
@ -202,6 +217,8 @@ static int add_elf_sym(Section *s, unsigned long value, unsigned long size,
goto do_patch; goto do_patch;
} else if (sym_bind == STB_WEAK && esym_bind == STB_GLOBAL) { } else if (sym_bind == STB_WEAK && esym_bind == STB_GLOBAL) {
/* weak is ignored if already global */ /* weak is ignored if already global */
} else if (sym_vis == STV_HIDDEN || sym_vis == STV_INTERNAL) {
/* ignore hidden symbols after */
} else { } else {
#if 0 #if 0
printf("new_bind=%d new_shndx=%d last_bind=%d old_shndx=%d\n", printf("new_bind=%d new_shndx=%d last_bind=%d old_shndx=%d\n",

View File

@ -123,7 +123,6 @@
/* builtin functions or variables */ /* builtin functions or variables */
DEF(TOK_memcpy, "memcpy") DEF(TOK_memcpy, "memcpy")
DEF(TOK_memset, "memset") DEF(TOK_memset, "memset")
DEF(TOK_alloca, "alloca")
DEF(TOK___divdi3, "__divdi3") DEF(TOK___divdi3, "__divdi3")
DEF(TOK___moddi3, "__moddi3") DEF(TOK___moddi3, "__moddi3")
DEF(TOK___udivdi3, "__udivdi3") DEF(TOK___udivdi3, "__udivdi3")
@ -188,6 +187,7 @@
DEF(TOK_memmove, "memmove") DEF(TOK_memmove, "memmove")
DEF(TOK_strlen, "strlen") DEF(TOK_strlen, "strlen")
DEF(TOK_strcpy, "strcpy") DEF(TOK_strcpy, "strcpy")
DEF(TOK__alloca, "_alloca")
#endif #endif
/* Tiny Assembler */ /* Tiny Assembler */

View File

@ -18,6 +18,8 @@
#ifndef __MINGW_H #ifndef __MINGW_H
#define __MINGW_H #define __MINGW_H
#include <stddef.h>
#define __int64 long long #define __int64 long long
#define __int32 long #define __int32 long
#define __int16 short #define __int16 short

View File

@ -6,6 +6,7 @@ typedef char *va_list;
/* only correct for i386 */ /* only correct for i386 */
#define va_start(ap,last) ap = ((char *)&(last)) + ((sizeof(last)+3)&~3) #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_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) #define va_end(ap)
/* fix a buggy dependency on GCC in libio.h */ /* fix a buggy dependency on GCC in libio.h */

View File

@ -18,4 +18,9 @@ typedef int int32_t;
typedef long long int int64_t; typedef long long int int64_t;
#endif #endif
#ifdef __i386__
void *_alloca(size_t);
#define alloca _alloca
#endif
#endif #endif