mirror of
https://github.com/frida/tinycc
synced 2025-01-22 03:02:04 +03:00
Rework expr_landor
so that it also is called from the precedence parser. This is complicated by the fact that something needs to be done before the second operand is parsed in a single pass compiler, so it doesn't quite fit into expr_infix itself. It turns out the smallest code changes result when expr_landor remains separate. But it can be tidied a bit.
This commit is contained in:
parent
23a8bac7b5
commit
aeac24de98
160
tccgen.c
160
tccgen.c
@ -67,7 +67,7 @@ static int gind(void) { CODE_ON(); return ind; }
|
|||||||
|
|
||||||
/* Set 'nocode_wanted' after unconditional jumps */
|
/* Set 'nocode_wanted' after unconditional jumps */
|
||||||
static void gjmp_addr_acs(int t) { gjmp_addr(t); CODE_OFF(); }
|
static void gjmp_addr_acs(int t) { gjmp_addr(t); CODE_OFF(); }
|
||||||
static int gjmp_acs(int t) { t = gjmp(t); CODE_OFF(); return t; }
|
static int gjmp_acs(int t) { if (!nocode_wanted) {t = gjmp(t); CODE_OFF(); } return t; }
|
||||||
|
|
||||||
/* These are #undef'd at the end of this file */
|
/* These are #undef'd at the end of this file */
|
||||||
#define gjmp_addr gjmp_addr_acs
|
#define gjmp_addr gjmp_addr_acs
|
||||||
@ -877,7 +877,7 @@ static void vcheck_cmp(void)
|
|||||||
again, so that the VT_CMP/VT_JMP value will be in vtop
|
again, so that the VT_CMP/VT_JMP value will be in vtop
|
||||||
when code is unsuppressed again. */
|
when code is unsuppressed again. */
|
||||||
|
|
||||||
if (vtop->r == VT_CMP && !nocode_wanted)
|
if (vtop->r == VT_CMP && !nocode_wanted && vtop->cmp_op > 1)
|
||||||
gv(RC_INT);
|
gv(RC_INT);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1043,7 +1043,11 @@ static void vset_VT_JMP(void)
|
|||||||
if (vtop->jtrue || vtop->jfalse) {
|
if (vtop->jtrue || vtop->jfalse) {
|
||||||
/* we need to jump to 'mov $0,%R' or 'mov $1,%R' */
|
/* we need to jump to 'mov $0,%R' or 'mov $1,%R' */
|
||||||
int inv = op & (op < 2); /* small optimization */
|
int inv = op & (op < 2); /* small optimization */
|
||||||
vseti(VT_JMP+inv, gvtst(inv, 0));
|
int t = gvtst(inv, 0);
|
||||||
|
if (t) /* in case gvtst only needed to do a gsym */
|
||||||
|
vseti(VT_JMP+inv, t);
|
||||||
|
else
|
||||||
|
vpushi(inv);
|
||||||
} else {
|
} else {
|
||||||
/* otherwise convert flags (rsp. 0/1) to register */
|
/* otherwise convert flags (rsp. 0/1) to register */
|
||||||
vtop->c.i = op;
|
vtop->c.i = op;
|
||||||
@ -1061,9 +1065,11 @@ static void gvtst_set(int inv, int t)
|
|||||||
gen_op(TOK_NE);
|
gen_op(TOK_NE);
|
||||||
if (vtop->r == VT_CMP) /* must be VT_CONST otherwise */
|
if (vtop->r == VT_CMP) /* must be VT_CONST otherwise */
|
||||||
;
|
;
|
||||||
else if (vtop->r == VT_CONST)
|
else if (vtop->r == VT_CONST) {
|
||||||
|
if (!t)
|
||||||
|
return;
|
||||||
vset_VT_CMP(vtop->c.i != 0);
|
vset_VT_CMP(vtop->c.i != 0);
|
||||||
else
|
} else
|
||||||
tcc_error("ICE");
|
tcc_error("ICE");
|
||||||
}
|
}
|
||||||
p = inv ? &vtop->jfalse : &vtop->jtrue;
|
p = inv ? &vtop->jfalse : &vtop->jtrue;
|
||||||
@ -1075,18 +1081,21 @@ static void gvtst_set(int inv, int t)
|
|||||||
* Generate a test for any value (jump, comparison and integers) */
|
* Generate a test for any value (jump, comparison and integers) */
|
||||||
static int gvtst(int inv, int t)
|
static int gvtst(int inv, int t)
|
||||||
{
|
{
|
||||||
int op, u, x;
|
int op, x, u;
|
||||||
|
|
||||||
gvtst_set(inv, t);
|
gvtst_set(inv, t);
|
||||||
|
if (vtop->r != VT_CMP) {
|
||||||
|
t = u = x = 0;
|
||||||
|
op = vtop->c.i;
|
||||||
|
} else {
|
||||||
t = vtop->jtrue, u = vtop->jfalse;
|
t = vtop->jtrue, u = vtop->jfalse;
|
||||||
if (inv)
|
if (inv)
|
||||||
x = u, u = t, t = x;
|
x = u, u = t, t = x;
|
||||||
op = vtop->cmp_op;
|
op = vtop->cmp_op;
|
||||||
|
}
|
||||||
|
|
||||||
/* jump to the wanted target */
|
/* jump to the wanted target */
|
||||||
if (op > 1)
|
if (op > 1)
|
||||||
t = gjmp_cond(op ^ inv, t);
|
t = gjmp_cond(op ^ inv, t), op = inv;//, op = 1;
|
||||||
else if (op != inv)
|
else if (op != inv)
|
||||||
t = gjmp(t);
|
t = gjmp(t);
|
||||||
/* resolve complementary jumps to here */
|
/* resolve complementary jumps to here */
|
||||||
@ -5681,17 +5690,35 @@ special_math_val:
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Assuming vtop is a value used in a conditional context
|
||||||
|
(i.e. compared with zero) return 0 if it's false, 1 if
|
||||||
|
true and -1 if it can't be statically determined. */
|
||||||
|
static int condition_3way(void)
|
||||||
|
{
|
||||||
|
int c = -1;
|
||||||
|
if ((vtop->r & (VT_VALMASK | VT_LVAL)) == VT_CONST &&
|
||||||
|
(!(vtop->r & VT_SYM) || !vtop->sym->a.weak)) {
|
||||||
|
vdup();
|
||||||
|
gen_cast_s(VT_BOOL);
|
||||||
|
c = vtop->c.i;
|
||||||
|
vpop();
|
||||||
|
}
|
||||||
|
return c;
|
||||||
|
}
|
||||||
|
|
||||||
static int precedence(int tok)
|
static int precedence(int tok)
|
||||||
{
|
{
|
||||||
switch (tok) {
|
switch (tok) {
|
||||||
case '|': return 1;
|
case TOK_LOR: return 1;
|
||||||
case '^': return 2;
|
case TOK_LAND: return 2;
|
||||||
case '&': return 3;
|
case '|': return 3;
|
||||||
case TOK_EQ: case TOK_NE: return 4;
|
case '^': return 4;
|
||||||
relat: case TOK_ULT: case TOK_UGE: return 5;
|
case '&': return 5;
|
||||||
case TOK_SHL: case TOK_SAR: return 6;
|
case TOK_EQ: case TOK_NE: return 6;
|
||||||
case '+': case '-': return 7;
|
relat: case TOK_ULT: case TOK_UGE: return 7;
|
||||||
case '*': case '/': case '%': return 8;
|
case TOK_SHL: case TOK_SAR: return 8;
|
||||||
|
case '+': case '-': return 9;
|
||||||
|
case '*': case '/': case '%': return 10;
|
||||||
default:
|
default:
|
||||||
if (tok >= TOK_ULE && tok <= TOK_GT)
|
if (tok >= TOK_ULE && tok <= TOK_GT)
|
||||||
goto relat;
|
goto relat;
|
||||||
@ -5709,86 +5736,42 @@ ST_FUNC void init_prec(void)
|
|||||||
|
|
||||||
#define precedence(i) ((unsigned)i < 256 ? prec[i] : 0)
|
#define precedence(i) ((unsigned)i < 256 ? prec[i] : 0)
|
||||||
|
|
||||||
static void expr_infix(int p)
|
static void expr_infix(int p);
|
||||||
|
static void expr_landor(int p, int e_op, int i)
|
||||||
{
|
{
|
||||||
int t = tok, p2, p3;
|
int t = 0, f = 0;
|
||||||
while ((p2 = precedence(t)) >= p) {
|
save_regs(1);
|
||||||
|
do {
|
||||||
|
if (!f && condition_3way() == !i)
|
||||||
|
nocode_wanted++, f = 1;
|
||||||
|
t = gvtst(i, t);
|
||||||
next();
|
next();
|
||||||
unary();
|
unary();
|
||||||
while ((p3 = precedence(tok)) > p2) {
|
expr_infix(p);
|
||||||
expr_infix(p3);
|
} while (tok == e_op);
|
||||||
}
|
if (f) {
|
||||||
gen_op(t);
|
|
||||||
t = tok;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static void expr_or(void)
|
|
||||||
{
|
|
||||||
unary();
|
|
||||||
expr_infix(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
static int condition_3way(void);
|
|
||||||
|
|
||||||
static void expr_landor(void(*e_fn)(void), int e_op, int i)
|
|
||||||
{
|
|
||||||
int t = 0, cc = 1, f = 0, c;
|
|
||||||
for(;;) {
|
|
||||||
c = f ? i : condition_3way();
|
|
||||||
if (c < 0) {
|
|
||||||
save_regs(1), cc = 0;
|
|
||||||
} else if (c != i) {
|
|
||||||
nocode_wanted++, f = 1;
|
|
||||||
}
|
|
||||||
if (tok != e_op) {
|
|
||||||
if (cc || f) {
|
|
||||||
vpop();
|
vpop();
|
||||||
vpushi(i ^ f);
|
vpushi(i ^ f);
|
||||||
gsym(t);
|
}
|
||||||
nocode_wanted -= f;
|
|
||||||
} else {
|
|
||||||
gvtst_set(i, t);
|
gvtst_set(i, t);
|
||||||
|
nocode_wanted -= f;
|
||||||
}
|
}
|
||||||
break;
|
|
||||||
}
|
static void expr_infix(int p)
|
||||||
if (c < 0)
|
{
|
||||||
t = gvtst(i, t);
|
int t = tok, p2;
|
||||||
else
|
while ((p2 = precedence(t)) >= p) {
|
||||||
vpop();
|
if (t == TOK_LOR || t == TOK_LAND) {
|
||||||
|
expr_landor(p2 + 1, t, t == TOK_LAND);
|
||||||
|
} else {
|
||||||
next();
|
next();
|
||||||
e_fn();
|
unary();
|
||||||
|
if (precedence(tok) > p2)
|
||||||
|
expr_infix(p2 + 1);
|
||||||
|
gen_op(t);
|
||||||
}
|
}
|
||||||
|
t = tok;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void expr_land(void)
|
|
||||||
{
|
|
||||||
expr_or();
|
|
||||||
if (tok == TOK_LAND)
|
|
||||||
expr_landor(expr_or, TOK_LAND, 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void expr_lor(void)
|
|
||||||
{
|
|
||||||
expr_land();
|
|
||||||
if (tok == TOK_LOR)
|
|
||||||
expr_landor(expr_land, TOK_LOR, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Assuming vtop is a value used in a conditional context
|
|
||||||
(i.e. compared with zero) return 0 if it's false, 1 if
|
|
||||||
true and -1 if it can't be statically determined. */
|
|
||||||
static int condition_3way(void)
|
|
||||||
{
|
|
||||||
int c = -1;
|
|
||||||
if ((vtop->r & (VT_VALMASK | VT_LVAL)) == VT_CONST &&
|
|
||||||
(!(vtop->r & VT_SYM) || !vtop->sym->a.weak)) {
|
|
||||||
vdup();
|
|
||||||
gen_cast_s(VT_BOOL);
|
|
||||||
c = vtop->c.i;
|
|
||||||
vpop();
|
|
||||||
}
|
|
||||||
return c;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int is_cond_bool(SValue *sv)
|
static int is_cond_bool(SValue *sv)
|
||||||
@ -5808,7 +5791,8 @@ static void expr_cond(void)
|
|||||||
CType type, type1, type2;
|
CType type, type1, type2;
|
||||||
int ncw_prev;
|
int ncw_prev;
|
||||||
|
|
||||||
expr_lor();
|
unary();
|
||||||
|
expr_infix(1);
|
||||||
if (tok == '?') {
|
if (tok == '?') {
|
||||||
next();
|
next();
|
||||||
c = condition_3way();
|
c = condition_3way();
|
||||||
|
@ -84,6 +84,7 @@ void scope_test();
|
|||||||
void scope_test2();
|
void scope_test2();
|
||||||
void forward_test();
|
void forward_test();
|
||||||
void funcptr_test();
|
void funcptr_test();
|
||||||
|
void if_test();
|
||||||
void loop_test();
|
void loop_test();
|
||||||
void switch_test();
|
void switch_test();
|
||||||
void goto_test();
|
void goto_test();
|
||||||
@ -513,6 +514,40 @@ void string_test()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void if1t(int n, int a, int b, int c)
|
||||||
|
{
|
||||||
|
if (a && b) printf("if1t: %d 1 %d %d\n", n, a, b);
|
||||||
|
if (a && !b) printf("if1t: %d 2 %d %d\n", n, a, b);
|
||||||
|
if (!a && b) printf("if1t: %d 3 %d %d\n", n, a, b);
|
||||||
|
if (!a && !b) printf("if1t: %d 4 %d %d\n", n, a, b);
|
||||||
|
if (a || b) printf("if1t: %d 5 %d %d\n", n, a, b);
|
||||||
|
if (a || !b) printf("if1t: %d 6 %d %d\n", n, a, b);
|
||||||
|
if (!a || b) printf("if1t: %d 7 %d %d\n", n, a, b);
|
||||||
|
if (!a || !b) printf("if1t: %d 8 %d %d\n", n, a, b);
|
||||||
|
if (a && b || c) printf("if1t: %d 9 %d %d %d\n", n, a, b, c);
|
||||||
|
if (a || b && c) printf("if1t: %d 10 %d %d %d\n", n, a, b, c);
|
||||||
|
if (a > b - 1 && c) printf("if1t: %d 11 %d %d %d\n", n, a, b, c);
|
||||||
|
if (a > b - 1 || c) printf("if1t: %d 12 %d %d %d\n", n, a, b, c);
|
||||||
|
if (a > 0 && 1) printf("if1t: %d 13 %d %d %d\n", n, a, b, c);
|
||||||
|
if (a > 0 || 0) printf("if1t: %d 14 %d %d %d\n", n, a, b, c);
|
||||||
|
}
|
||||||
|
|
||||||
|
void if2t(void)
|
||||||
|
{
|
||||||
|
if (0 && 1 || printf("if2t:ok\n") || 1)
|
||||||
|
printf("if2t:ok2\n");
|
||||||
|
printf("if2t:ok3\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
void if_test(void)
|
||||||
|
{
|
||||||
|
if1t(1, 0, 0, 0);
|
||||||
|
if1t(2, 0, 3, 0);
|
||||||
|
if1t(3, 2, 0, 0);
|
||||||
|
if1t(4, 2, 3, 0);
|
||||||
|
if2t();
|
||||||
|
}
|
||||||
|
|
||||||
void loop_test()
|
void loop_test()
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
@ -728,6 +763,7 @@ int main(int argc, char **argv)
|
|||||||
scope_test2();
|
scope_test2();
|
||||||
forward_test();
|
forward_test();
|
||||||
funcptr_test();
|
funcptr_test();
|
||||||
|
if_test();
|
||||||
loop_test();
|
loop_test();
|
||||||
switch_test();
|
switch_test();
|
||||||
goto_test();
|
goto_test();
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
#include <stdio.h>
|
//#include <stdio.h>
|
||||||
|
extern int printf(const char *, ...);
|
||||||
|
|
||||||
int main()
|
int main()
|
||||||
{
|
{
|
||||||
|
Loading…
Reference in New Issue
Block a user