mirror of
https://github.com/mirror/tinycc.git
synced 2025-01-19 05:30:07 +08:00
fixed bound checking for structures - fixed ?: type handling - fixed '--' parsing
This commit is contained in:
parent
f1418e836f
commit
405c88106d
129
tcc.c
129
tcc.c
@ -271,9 +271,11 @@ int nb_library_paths;
|
|||||||
anon_sym: anonymous symbol index
|
anon_sym: anonymous symbol index
|
||||||
*/
|
*/
|
||||||
int rsym, anon_sym,
|
int rsym, anon_sym,
|
||||||
prog, ind, loc, const_wanted;
|
prog, ind, loc;
|
||||||
int global_expr; /* true if compound literals must be allocated
|
/* expression generation modifiers */
|
||||||
globally (used during initializers parsing */
|
int const_wanted; /* true if constant wanted */
|
||||||
|
int global_expr; /* true if compound literals must be allocated
|
||||||
|
globally (used during initializers parsing */
|
||||||
int func_vt, func_vc; /* current function return type (used by
|
int func_vt, func_vc; /* current function return type (used by
|
||||||
return instruction) */
|
return instruction) */
|
||||||
int last_line_num, last_ind, func_ind; /* debug last line number and pc */
|
int last_line_num, last_ind, func_ind; /* debug last line number and pc */
|
||||||
@ -921,11 +923,14 @@ static void put_extern_sym(Sym *sym, Section *section,
|
|||||||
/* if bound checking is activated, we change some function
|
/* if bound checking is activated, we change some function
|
||||||
names by adding the "__bound" prefix */
|
names by adding the "__bound" prefix */
|
||||||
switch(sym->v) {
|
switch(sym->v) {
|
||||||
|
#if 0
|
||||||
|
/* XXX: we rely only on malloc hooks */
|
||||||
case TOK_malloc:
|
case TOK_malloc:
|
||||||
case TOK_free:
|
case TOK_free:
|
||||||
case TOK_realloc:
|
case TOK_realloc:
|
||||||
case TOK_memalign:
|
case TOK_memalign:
|
||||||
case TOK_calloc:
|
case TOK_calloc:
|
||||||
|
#endif
|
||||||
case TOK_memcpy:
|
case TOK_memcpy:
|
||||||
case TOK_memmove:
|
case TOK_memmove:
|
||||||
case TOK_memset:
|
case TOK_memset:
|
||||||
@ -1901,7 +1906,7 @@ void preprocess(void)
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (include_stack_ptr >= include_stack + INCLUDE_STACK_SIZE)
|
if (include_stack_ptr >= include_stack + INCLUDE_STACK_SIZE)
|
||||||
error("memory full");
|
error("#include recursion too deep");
|
||||||
if (c == '\"') {
|
if (c == '\"') {
|
||||||
/* first search in current dir if "header.h" */
|
/* first search in current dir if "header.h" */
|
||||||
size = 0;
|
size = 0;
|
||||||
@ -2857,12 +2862,18 @@ void swap(int *p, int *q)
|
|||||||
|
|
||||||
void vsetc(int t, int r, CValue *vc)
|
void vsetc(int t, int r, CValue *vc)
|
||||||
{
|
{
|
||||||
|
int v;
|
||||||
|
|
||||||
if (vtop >= vstack + VSTACK_SIZE)
|
if (vtop >= vstack + VSTACK_SIZE)
|
||||||
error("memory full");
|
error("memory full");
|
||||||
/* cannot let cpu flags if other instruction are generated */
|
/* cannot let cpu flags if other instruction are generated. Also
|
||||||
/* XXX: VT_JMP test too ? */
|
avoid leaving VT_JMP anywhere except on the top of the stack
|
||||||
if ((vtop->r & VT_VALMASK) == VT_CMP)
|
because it would complicate the code generator. */
|
||||||
gv(RC_INT);
|
if (vtop >= vstack) {
|
||||||
|
v = vtop->r & VT_VALMASK;
|
||||||
|
if (v == VT_CMP || (v & ~1) == VT_JMP)
|
||||||
|
gv(RC_INT);
|
||||||
|
}
|
||||||
vtop++;
|
vtop++;
|
||||||
vtop->t = t;
|
vtop->t = t;
|
||||||
vtop->r = r;
|
vtop->r = r;
|
||||||
@ -3251,8 +3262,13 @@ int gv(int rc)
|
|||||||
/* generate vtop[-1] and vtop[0] in resp. classes rc1 and rc2 */
|
/* generate vtop[-1] and vtop[0] in resp. classes rc1 and rc2 */
|
||||||
void gv2(int rc1, int rc2)
|
void gv2(int rc1, int rc2)
|
||||||
{
|
{
|
||||||
/* generate more generic register first */
|
int v;
|
||||||
if (rc1 <= rc2) {
|
|
||||||
|
/* generate more generic register first. But VT_JMP or VT_CMP
|
||||||
|
values must be generated first in all cases to avoid possible
|
||||||
|
reload errors */
|
||||||
|
v = vtop[0].r & VT_VALMASK;
|
||||||
|
if (v != VT_CMP && (v & ~1) != VT_JMP && rc1 <= rc2) {
|
||||||
vswap();
|
vswap();
|
||||||
gv(rc1);
|
gv(rc1);
|
||||||
vswap();
|
vswap();
|
||||||
@ -5239,8 +5255,12 @@ void unary(void)
|
|||||||
/* change type to field type, and set to lvalue */
|
/* change type to field type, and set to lvalue */
|
||||||
vtop->t = s->t;
|
vtop->t = s->t;
|
||||||
/* an array is never an lvalue */
|
/* an array is never an lvalue */
|
||||||
if (!(vtop->t & VT_ARRAY))
|
if (!(vtop->t & VT_ARRAY)) {
|
||||||
vtop->r |= lvalue_type(vtop->t);
|
vtop->r |= lvalue_type(vtop->t);
|
||||||
|
/* if bound checking, the referenced pointer must be checked */
|
||||||
|
if (do_bounds_check)
|
||||||
|
vtop->r |= VT_MUSTBOUND;
|
||||||
|
}
|
||||||
next();
|
next();
|
||||||
} else if (tok == '[') {
|
} else if (tok == '[') {
|
||||||
next();
|
next();
|
||||||
@ -5468,21 +5488,23 @@ void eor(void)
|
|||||||
/* XXX: better constant handling */
|
/* XXX: better constant handling */
|
||||||
void expr_eq(void)
|
void expr_eq(void)
|
||||||
{
|
{
|
||||||
int t, u, c, r1, r2, rc;
|
int tt, u, r1, r2, rc, t1, t2, t, bt1, bt2;
|
||||||
|
SValue sv;
|
||||||
|
|
||||||
if (const_wanted) {
|
if (const_wanted) {
|
||||||
|
int c1, c;
|
||||||
sum(10);
|
sum(10);
|
||||||
if (tok == '?') {
|
if (tok == '?') {
|
||||||
c = vtop->c.i;
|
c = vtop->c.i;
|
||||||
vpop();
|
vpop();
|
||||||
next();
|
next();
|
||||||
gexpr();
|
gexpr();
|
||||||
t = vtop->c.i;
|
c1 = vtop->c.i;
|
||||||
vpop();
|
vpop();
|
||||||
skip(':');
|
skip(':');
|
||||||
expr_eq();
|
expr_eq();
|
||||||
if (c)
|
if (c)
|
||||||
vtop->c.i = t;
|
vtop->c.i = c1;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
eor();
|
eor();
|
||||||
@ -5490,23 +5512,76 @@ void expr_eq(void)
|
|||||||
next();
|
next();
|
||||||
save_regs(1); /* we need to save all registers here except
|
save_regs(1); /* we need to save all registers here except
|
||||||
at the top because it is a branch point */
|
at the top because it is a branch point */
|
||||||
t = gtst(1, 0);
|
tt = gtst(1, 0);
|
||||||
gexpr();
|
gexpr();
|
||||||
/* XXX: long long handling ? */
|
t1 = vtop->t;
|
||||||
rc = RC_INT;
|
bt1 = t1 & VT_BTYPE;
|
||||||
if (is_float(vtop->t))
|
sv = *vtop; /* save value to handle it later */
|
||||||
rc = RC_FLOAT;
|
|
||||||
r1 = gv(rc);
|
|
||||||
vtop--; /* no vpop so that FP stack is not flushed */
|
vtop--; /* no vpop so that FP stack is not flushed */
|
||||||
skip(':');
|
skip(':');
|
||||||
u = gjmp(0);
|
u = gjmp(0);
|
||||||
|
|
||||||
gsym(t);
|
gsym(tt);
|
||||||
expr_eq();
|
expr_eq();
|
||||||
|
t2 = vtop->t;
|
||||||
|
|
||||||
|
bt2 = t2 & VT_BTYPE;
|
||||||
|
/* cast operands to correct type according to ISOC rules */
|
||||||
|
if (is_float(bt1) || is_float(bt2)) {
|
||||||
|
if (bt1 == VT_LDOUBLE || bt2 == VT_LDOUBLE) {
|
||||||
|
t = VT_LDOUBLE;
|
||||||
|
} else if (bt1 == VT_DOUBLE || bt2 == VT_DOUBLE) {
|
||||||
|
t = VT_DOUBLE;
|
||||||
|
} else {
|
||||||
|
t = VT_FLOAT;
|
||||||
|
}
|
||||||
|
} else if (bt1 == VT_LLONG || bt2 == VT_LLONG) {
|
||||||
|
/* cast to biggest op */
|
||||||
|
t = VT_LLONG;
|
||||||
|
/* convert to unsigned if it does not fit in a long long */
|
||||||
|
if ((t1 & (VT_BTYPE | VT_UNSIGNED)) == (VT_LLONG | VT_UNSIGNED) ||
|
||||||
|
(t2 & (VT_BTYPE | VT_UNSIGNED)) == (VT_LLONG | VT_UNSIGNED))
|
||||||
|
t |= VT_UNSIGNED;
|
||||||
|
} else if (bt1 == VT_PTR || bt2 == VT_PTR) {
|
||||||
|
/* XXX: test pointer compatibility */
|
||||||
|
t = t1;
|
||||||
|
} else if (bt1 == VT_STRUCT || bt2 == VT_STRUCT) {
|
||||||
|
/* XXX: test structure compatibility */
|
||||||
|
t = t1;
|
||||||
|
} else if (bt1 == VT_VOID || bt2 == VT_VOID) {
|
||||||
|
/* NOTE: as an extension, we accept void on only one side */
|
||||||
|
t = VT_VOID;
|
||||||
|
} else {
|
||||||
|
/* integer operations */
|
||||||
|
t = VT_INT;
|
||||||
|
/* convert to unsigned if it does not fit in an integer */
|
||||||
|
if ((t1 & (VT_BTYPE | VT_UNSIGNED)) == (VT_INT | VT_UNSIGNED) ||
|
||||||
|
(t2 & (VT_BTYPE | VT_UNSIGNED)) == (VT_INT | VT_UNSIGNED))
|
||||||
|
t |= VT_UNSIGNED;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* now we convert second operand */
|
||||||
|
gen_cast(t);
|
||||||
|
rc = RC_INT;
|
||||||
|
if (is_float(t)) {
|
||||||
|
rc = RC_FLOAT;
|
||||||
|
} else if ((t & VT_BTYPE) == VT_LLONG) {
|
||||||
|
/* for long longs, we use fixed registers to avoid having
|
||||||
|
to handle a complicated move */
|
||||||
|
rc = RC_IRET;
|
||||||
|
}
|
||||||
r2 = gv(rc);
|
r2 = gv(rc);
|
||||||
move_reg(r1, r2);
|
/* this is horrible, but we must also convert first
|
||||||
vtop->r = r1;
|
operand */
|
||||||
|
tt = gjmp(0);
|
||||||
gsym(u);
|
gsym(u);
|
||||||
|
/* put again first value and cast it */
|
||||||
|
*vtop = sv;
|
||||||
|
gen_cast(t);
|
||||||
|
r1 = gv(rc);
|
||||||
|
move_reg(r2, r1);
|
||||||
|
vtop->r = r2;
|
||||||
|
gsym(tt);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -7196,7 +7271,7 @@ int tcc_set_output_type(TCCState *s, int output_type)
|
|||||||
|
|
||||||
void help(void)
|
void help(void)
|
||||||
{
|
{
|
||||||
printf("tcc version 0.9.11 - Tiny C Compiler - Copyright (C) 2001, 2002 Fabrice Bellard\n"
|
printf("tcc version 0.9.12 - Tiny C Compiler - Copyright (C) 2001, 2002 Fabrice Bellard\n"
|
||||||
"usage: tcc [-c] [-o outfile] [-Bdir] [-bench] [-Idir] [-Dsym[=val]] [-Usym]\n"
|
"usage: tcc [-c] [-o outfile] [-Bdir] [-bench] [-Idir] [-Dsym[=val]] [-Usym]\n"
|
||||||
" [-g] [-b] [-Ldir] [-llib] [-shared] [-static]\n"
|
" [-g] [-b] [-Ldir] [-llib] [-shared] [-static]\n"
|
||||||
" [--] infile1 [infile2... --] [infile_args...]\n"
|
" [--] infile1 [infile2... --] [infile_args...]\n"
|
||||||
@ -7232,7 +7307,7 @@ int main(int argc, char **argv)
|
|||||||
int optind, output_type, multiple_files, i, reloc_output;
|
int optind, output_type, multiple_files, i, reloc_output;
|
||||||
TCCState *s;
|
TCCState *s;
|
||||||
char **files;
|
char **files;
|
||||||
int nb_files, nb_libraries, nb_objfiles;
|
int nb_files, nb_libraries, nb_objfiles, dminus;
|
||||||
char objfilename[1024];
|
char objfilename[1024];
|
||||||
|
|
||||||
s = tcc_new();
|
s = tcc_new();
|
||||||
@ -7241,6 +7316,7 @@ int main(int argc, char **argv)
|
|||||||
optind = 1;
|
optind = 1;
|
||||||
outfile = NULL;
|
outfile = NULL;
|
||||||
multiple_files = 0;
|
multiple_files = 0;
|
||||||
|
dminus = 0;
|
||||||
files = NULL;
|
files = NULL;
|
||||||
nb_files = 0;
|
nb_files = 0;
|
||||||
nb_libraries = 0;
|
nb_libraries = 0;
|
||||||
@ -7263,10 +7339,11 @@ int main(int argc, char **argv)
|
|||||||
}
|
}
|
||||||
} else if (r[1] == '-') {
|
} else if (r[1] == '-') {
|
||||||
/* '--' enables multiple files input and also ends several file input */
|
/* '--' enables multiple files input and also ends several file input */
|
||||||
if (multiple_files) {
|
if (dminus && multiple_files) {
|
||||||
optind--; /* argv[0] will be '--' */
|
optind--; /* argv[0] will be '--' */
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
dminus = 1;
|
||||||
multiple_files = 1;
|
multiple_files = 1;
|
||||||
} else if (r[1] == 'h' || r[1] == '?') {
|
} else if (r[1] == 'h' || r[1] == '?') {
|
||||||
show_help:
|
show_help:
|
||||||
|
Loading…
Reference in New Issue
Block a user