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:
Michael Matz 2020-01-13 01:06:25 +01:00
parent 23a8bac7b5
commit aeac24de98
3 changed files with 120 additions and 99 deletions

162
tccgen.c
View File

@ -67,7 +67,7 @@ static int gind(void) { CODE_ON(); return ind; }
/* Set 'nocode_wanted' after unconditional jumps */
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 */
#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
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);
}
@ -1043,7 +1043,11 @@ static void vset_VT_JMP(void)
if (vtop->jtrue || vtop->jfalse) {
/* we need to jump to 'mov $0,%R' or 'mov $1,%R' */
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 {
/* otherwise convert flags (rsp. 0/1) to register */
vtop->c.i = op;
@ -1061,9 +1065,11 @@ static void gvtst_set(int inv, int t)
gen_op(TOK_NE);
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);
else
} else
tcc_error("ICE");
}
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) */
static int gvtst(int inv, int t)
{
int op, u, x;
int op, x, u;
gvtst_set(inv, t);
if (vtop->r != VT_CMP) {
t = u = x = 0;
op = vtop->c.i;
} else {
t = vtop->jtrue, u = vtop->jfalse;
if (inv)
x = u, u = t, t = x;
op = vtop->cmp_op;
}
/* jump to the wanted target */
if (op > 1)
t = gjmp_cond(op ^ inv, t);
t = gjmp_cond(op ^ inv, t), op = inv;//, op = 1;
else if (op != inv)
t = gjmp(t);
/* 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)
{
switch (tok) {
case '|': return 1;
case '^': return 2;
case '&': return 3;
case TOK_EQ: case TOK_NE: return 4;
relat: case TOK_ULT: case TOK_UGE: return 5;
case TOK_SHL: case TOK_SAR: return 6;
case '+': case '-': return 7;
case '*': case '/': case '%': return 8;
case TOK_LOR: return 1;
case TOK_LAND: return 2;
case '|': return 3;
case '^': return 4;
case '&': return 5;
case TOK_EQ: case TOK_NE: return 6;
relat: case TOK_ULT: case TOK_UGE: return 7;
case TOK_SHL: case TOK_SAR: return 8;
case '+': case '-': return 9;
case '*': case '/': case '%': return 10;
default:
if (tok >= TOK_ULE && tok <= TOK_GT)
goto relat;
@ -5709,86 +5736,42 @@ ST_FUNC void init_prec(void)
#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;
while ((p2 = precedence(t)) >= p) {
int t = 0, f = 0;
save_regs(1);
do {
if (!f && condition_3way() == !i)
nocode_wanted++, f = 1;
t = gvtst(i, t);
next();
unary();
while ((p3 = precedence(tok)) > p2) {
expr_infix(p3);
}
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) {
expr_infix(p);
} while (tok == e_op);
if (f) {
vpop();
vpushi(i ^ f);
gsym(t);
nocode_wanted -= f;
} else {
}
gvtst_set(i, t);
}
break;
}
if (c < 0)
t = gvtst(i, t);
else
vpop();
nocode_wanted -= f;
}
static void expr_infix(int p)
{
int t = tok, p2;
while ((p2 = precedence(t)) >= p) {
if (t == TOK_LOR || t == TOK_LAND) {
expr_landor(p2 + 1, t, t == TOK_LAND);
} else {
next();
e_fn();
unary();
if (precedence(tok) > p2)
expr_infix(p2 + 1);
gen_op(t);
}
}
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();
t = tok;
}
return c;
}
static int is_cond_bool(SValue *sv)
@ -5808,7 +5791,8 @@ static void expr_cond(void)
CType type, type1, type2;
int ncw_prev;
expr_lor();
unary();
expr_infix(1);
if (tok == '?') {
next();
c = condition_3way();

View File

@ -84,6 +84,7 @@ void scope_test();
void scope_test2();
void forward_test();
void funcptr_test();
void if_test();
void loop_test();
void switch_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()
{
int i;
@ -728,6 +763,7 @@ int main(int argc, char **argv)
scope_test2();
forward_test();
funcptr_test();
if_test();
loop_test();
switch_test();
goto_test();

View File

@ -1,4 +1,5 @@
#include <stdio.h>
//#include <stdio.h>
extern int printf(const char *, ...);
int main()
{