2019-02-18 16:42:15 +00:00
|
|
|
/*
|
|
|
|
* (c) copyright 1987 by the Vrije Universiteit, Amsterdam, The Netherlands.
|
|
|
|
* See the copyright notice in the ACK home directory, in the file "Copyright".
|
|
|
|
*/
|
|
|
|
/* $Id$ */
|
|
|
|
/* STATEMENT SYNTAX PARSER */
|
|
|
|
|
|
|
|
{
|
|
|
|
#include "parameters.h"
|
|
|
|
#ifndef LINT
|
|
|
|
#include <em.h>
|
|
|
|
#else
|
|
|
|
#include "l_em.h"
|
|
|
|
#include "l_lint.h"
|
|
|
|
#endif /* LINT */
|
|
|
|
|
|
|
|
#include <flt_arith.h>
|
|
|
|
#include "idf.h"
|
|
|
|
#include "arith.h"
|
|
|
|
#include "LLlex.h"
|
|
|
|
#include "type.h"
|
|
|
|
#include "label.h"
|
|
|
|
#include "expr.h"
|
|
|
|
#include "code.h"
|
|
|
|
#include "stack.h"
|
|
|
|
#include "def.h"
|
|
|
|
#include "switch.h"
|
|
|
|
#ifdef DBSYMTAB
|
|
|
|
#include <stb.h>
|
|
|
|
#endif /* DBSYMTAB */
|
|
|
|
|
|
|
|
extern int level;
|
|
|
|
extern char options[];
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Each statement construction is stacked in order to trace a
|
|
|
|
* statement to such a construction. Example: a case statement should
|
|
|
|
* be recognized as a piece of the most enclosing switch statement.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* 3.6 */
|
|
|
|
statement
|
|
|
|
{
|
|
|
|
#ifdef LINT
|
|
|
|
lint_statement();
|
|
|
|
#endif /* LINT */
|
|
|
|
}
|
|
|
|
:
|
|
|
|
%if (AHEAD != ':')
|
|
|
|
expression_statement
|
|
|
|
|
|
|
|
|
label ':' statement
|
|
|
|
|
|
|
|
|
compound_statement
|
|
|
|
|
|
|
|
|
if_statement
|
|
|
|
|
|
|
|
|
while_statement
|
|
|
|
|
|
|
|
|
do_statement
|
|
|
|
|
|
|
|
|
for_statement
|
|
|
|
|
|
|
|
|
switch_statement
|
|
|
|
|
|
|
|
|
case_statement
|
|
|
|
|
|
|
|
|
default_statement
|
|
|
|
|
|
|
|
|
BREAK
|
|
|
|
{
|
|
|
|
code_break();
|
|
|
|
#ifdef LINT
|
|
|
|
lint_break_stmt();
|
|
|
|
#endif /* LINT */
|
|
|
|
}
|
|
|
|
';'
|
|
|
|
|
|
|
|
|
CONTINUE
|
|
|
|
{
|
|
|
|
code_continue();
|
|
|
|
#ifdef LINT
|
|
|
|
lint_continue_stmt();
|
|
|
|
#endif /* LINT */
|
|
|
|
}
|
|
|
|
';'
|
|
|
|
|
|
|
|
|
return_statement
|
|
|
|
|
|
|
|
|
jump
|
|
|
|
|
|
|
|
|
';'
|
|
|
|
;
|
|
|
|
|
|
|
|
|
|
|
|
expression_statement
|
|
|
|
{ struct expr *expr;
|
|
|
|
}
|
|
|
|
:
|
|
|
|
expression(&expr)
|
|
|
|
';'
|
|
|
|
{
|
|
|
|
#ifdef DEBUG
|
|
|
|
print_expr("expression_statement", expr);
|
|
|
|
#endif /* DEBUG */
|
|
|
|
code_expr(expr, RVAL, FALSE, NO_LABEL, NO_LABEL);
|
|
|
|
free_expression(expr);
|
|
|
|
}
|
|
|
|
;
|
|
|
|
|
|
|
|
/* 3.6.1 (partially) */
|
|
|
|
label
|
|
|
|
{ struct idf *idf; }
|
|
|
|
:
|
|
|
|
identifier(&idf)
|
|
|
|
{
|
|
|
|
/* This allows the following absurd case:
|
|
|
|
|
|
|
|
typedef int grz;
|
|
|
|
main() {
|
|
|
|
grz: printf("A labelled statement\n");
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
#ifdef LINT
|
|
|
|
lint_label();
|
|
|
|
#endif /* LINT */
|
|
|
|
define_label(idf);
|
|
|
|
C_df_ilb((label)idf->id_label->df_address);
|
|
|
|
}
|
|
|
|
;
|
|
|
|
|
|
|
|
/* 3.6.4.1 */
|
|
|
|
if_statement
|
|
|
|
{
|
|
|
|
struct expr *expr;
|
|
|
|
label l_true = text_label();
|
|
|
|
label l_false = text_label();
|
|
|
|
label l_end = text_label();
|
|
|
|
}
|
|
|
|
:
|
|
|
|
IF
|
|
|
|
'('
|
|
|
|
expression(&expr)
|
|
|
|
{
|
|
|
|
opnd2test(&expr, IF);
|
|
|
|
if (is_cp_cst(expr)) {
|
|
|
|
/* The comparison has been optimized
|
|
|
|
to a 0 or 1.
|
|
|
|
*/
|
Add long long literals like 123LL to ACK C.
For now, a long long literal must have the 'LL' or 'll' suffix. A
literal without 'LL' or 'll' acts as before: it may become unsigned
long but not long long. (For targets where int and long have the same
size, some literals change from unsigned int to unsigned long.)
Type `arith` may be too narrow for long long values. Add a second
type `writh` for wide arithmetic, and change some variables from arith
to writh. This may cause bugs if I forget to use writh, or if a
conversion from writh to arith overflows. I mark some conversions
with (arith) or (writh) casts.
- BigPars, SmallPars: Remove SPECIAL_ARITHMETICS. This feature
would change arith to a different type, but can't work, because it
would conflict with definitions of arith in both <em_arith.h> and
<flt_arith.h>.
- LLlex.c: Understand 'LL' or 'll' suffix. Cut size of constant when
it overflows writh, not only when it overflows the target machine's
types. (This cut might not be necessary, because we might cut it
again later.) When picking signed long or unsigned long, check the
target's long type, not the compiler's arith type; the old check
for `val >= 0` was broken where sizeof(arith) > 4.
- LLlex.h: Change struct token's tok_ival to writh, so it can hold a
long long literal.
- arith.c: Adjust to VL_VALUE being writh. Don't convert between
float and integer at compile-time if the integer might be too wide
for <flt_arith.h>. Add writh2str(), because writh might be too
wide for long2str().
- arith.h: Remove SPECIAL_ARITHMETICS. Declare full_mask[] here,
not in several *.c files. Declare writh2str().
- ch3.c, ch3bin.c, ch3mon.c, declarator.c, statement.g: Remove
obsolete casts. Adjust to VL_VALUE being writh.
- conversion.c, stab.c: Don't declare full_mask[].
- cstoper.c: Use writh for constant operations on VL_VALUE, and for
full_mask[].
- declar., field.c, ival.g: Add casts.
- dumpidf.c: Need to #include "parameters.h" before checking DEBUG.
Use writh2str, because "%ld" might not work.
- eval.c, eval.h: Add casts. Use writh when writing a wide constant
in EM.
- expr.c: Add and remove casts. In fill_int_expr(), make expression
from long long literal. In chk_cst_expr(), allow long long as
constant expression, so the compiler may accept `case 123LL:` in a
switch statement.
- expr.str: Change struct value's vl_value and struct expr's VL_VALUE
to writh, so an expression may have a long long value at compile
time.
- statement.g: Remove obsolete casts.
- switch.c, switch.str: Use writh in case entries for switch
statements, so `switch (ll) {...}` with long long ll works.
- tokenname.c: Add ULNGLNG so LLlex.c can use it for literals.
2019-09-05 02:14:38 +00:00
|
|
|
if (expr->VL_VALUE == 0) {
|
2019-02-18 16:42:15 +00:00
|
|
|
C_bra(l_false);
|
|
|
|
}
|
|
|
|
/* else fall through */
|
|
|
|
#ifdef LINT
|
|
|
|
start_if_part(1);
|
|
|
|
#endif /* LINT */
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
code_expr(expr, RVAL, TRUE, l_true, l_false);
|
|
|
|
C_df_ilb(l_true);
|
|
|
|
#ifdef LINT
|
|
|
|
start_if_part(0);
|
|
|
|
#endif /* LINT */
|
|
|
|
}
|
|
|
|
free_expression(expr);
|
|
|
|
}
|
|
|
|
')'
|
|
|
|
statement
|
|
|
|
[%prefer
|
|
|
|
ELSE
|
|
|
|
{
|
|
|
|
#ifdef LINT
|
|
|
|
start_else_part();
|
|
|
|
#endif /* LINT */
|
|
|
|
C_bra(l_end);
|
|
|
|
C_df_ilb(l_false);
|
|
|
|
}
|
|
|
|
statement
|
|
|
|
{ C_df_ilb(l_end);
|
|
|
|
#ifdef LINT
|
|
|
|
end_if_else_stmt();
|
|
|
|
#endif /* LINT */
|
|
|
|
}
|
|
|
|
|
|
|
|
|
empty
|
|
|
|
{ C_df_ilb(l_false);
|
|
|
|
#ifdef LINT
|
|
|
|
end_if_stmt();
|
|
|
|
#endif /* LINT */
|
|
|
|
}
|
|
|
|
]
|
|
|
|
;
|
|
|
|
|
|
|
|
/* 3.6.5.3 */
|
|
|
|
while_statement
|
|
|
|
{
|
|
|
|
struct expr *expr;
|
|
|
|
label l_break = text_label();
|
|
|
|
label l_continue = text_label();
|
|
|
|
label l_body = text_label();
|
|
|
|
}
|
|
|
|
:
|
|
|
|
WHILE
|
|
|
|
{
|
|
|
|
stack_stmt(l_break, l_continue);
|
|
|
|
C_df_ilb(l_continue);
|
|
|
|
}
|
|
|
|
'('
|
|
|
|
expression(&expr)
|
|
|
|
{
|
|
|
|
opnd2test(&expr, WHILE);
|
|
|
|
if (is_cp_cst(expr)) {
|
Add long long literals like 123LL to ACK C.
For now, a long long literal must have the 'LL' or 'll' suffix. A
literal without 'LL' or 'll' acts as before: it may become unsigned
long but not long long. (For targets where int and long have the same
size, some literals change from unsigned int to unsigned long.)
Type `arith` may be too narrow for long long values. Add a second
type `writh` for wide arithmetic, and change some variables from arith
to writh. This may cause bugs if I forget to use writh, or if a
conversion from writh to arith overflows. I mark some conversions
with (arith) or (writh) casts.
- BigPars, SmallPars: Remove SPECIAL_ARITHMETICS. This feature
would change arith to a different type, but can't work, because it
would conflict with definitions of arith in both <em_arith.h> and
<flt_arith.h>.
- LLlex.c: Understand 'LL' or 'll' suffix. Cut size of constant when
it overflows writh, not only when it overflows the target machine's
types. (This cut might not be necessary, because we might cut it
again later.) When picking signed long or unsigned long, check the
target's long type, not the compiler's arith type; the old check
for `val >= 0` was broken where sizeof(arith) > 4.
- LLlex.h: Change struct token's tok_ival to writh, so it can hold a
long long literal.
- arith.c: Adjust to VL_VALUE being writh. Don't convert between
float and integer at compile-time if the integer might be too wide
for <flt_arith.h>. Add writh2str(), because writh might be too
wide for long2str().
- arith.h: Remove SPECIAL_ARITHMETICS. Declare full_mask[] here,
not in several *.c files. Declare writh2str().
- ch3.c, ch3bin.c, ch3mon.c, declarator.c, statement.g: Remove
obsolete casts. Adjust to VL_VALUE being writh.
- conversion.c, stab.c: Don't declare full_mask[].
- cstoper.c: Use writh for constant operations on VL_VALUE, and for
full_mask[].
- declar., field.c, ival.g: Add casts.
- dumpidf.c: Need to #include "parameters.h" before checking DEBUG.
Use writh2str, because "%ld" might not work.
- eval.c, eval.h: Add casts. Use writh when writing a wide constant
in EM.
- expr.c: Add and remove casts. In fill_int_expr(), make expression
from long long literal. In chk_cst_expr(), allow long long as
constant expression, so the compiler may accept `case 123LL:` in a
switch statement.
- expr.str: Change struct value's vl_value and struct expr's VL_VALUE
to writh, so an expression may have a long long value at compile
time.
- statement.g: Remove obsolete casts.
- switch.c, switch.str: Use writh in case entries for switch
statements, so `switch (ll) {...}` with long long ll works.
- tokenname.c: Add ULNGLNG so LLlex.c can use it for literals.
2019-09-05 02:14:38 +00:00
|
|
|
if (expr->VL_VALUE == 0) {
|
2019-02-18 16:42:15 +00:00
|
|
|
C_bra(l_break);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
code_expr(expr, RVAL, TRUE, l_body, l_break);
|
|
|
|
C_df_ilb(l_body);
|
|
|
|
}
|
|
|
|
#ifdef LINT
|
|
|
|
start_while_stmt(expr);
|
|
|
|
#endif /* LINT */
|
|
|
|
}
|
|
|
|
')'
|
|
|
|
statement
|
|
|
|
{
|
|
|
|
C_bra(l_continue);
|
|
|
|
C_df_ilb(l_break);
|
|
|
|
unstack_stmt();
|
|
|
|
free_expression(expr);
|
|
|
|
#ifdef LINT
|
|
|
|
end_loop_body();
|
|
|
|
end_loop_stmt();
|
|
|
|
#endif /* LINT */
|
|
|
|
}
|
|
|
|
;
|
|
|
|
|
|
|
|
/* 3.6.5.2 */
|
|
|
|
do_statement
|
|
|
|
{ struct expr *expr;
|
|
|
|
label l_break = text_label();
|
|
|
|
label l_continue = text_label();
|
|
|
|
label l_body = text_label();
|
|
|
|
}
|
|
|
|
:
|
|
|
|
DO
|
|
|
|
{ C_df_ilb(l_body);
|
|
|
|
stack_stmt(l_break, l_continue);
|
|
|
|
#ifdef LINT
|
|
|
|
start_do_stmt();
|
|
|
|
#endif /* LINT */
|
|
|
|
}
|
|
|
|
statement
|
|
|
|
WHILE
|
|
|
|
'('
|
|
|
|
{
|
|
|
|
#ifdef LINT
|
|
|
|
end_loop_body();
|
|
|
|
#endif /* LINT */
|
|
|
|
C_df_ilb(l_continue);
|
|
|
|
}
|
|
|
|
expression(&expr)
|
|
|
|
{
|
|
|
|
opnd2test(&expr, WHILE);
|
|
|
|
if (is_cp_cst(expr)) {
|
Add long long literals like 123LL to ACK C.
For now, a long long literal must have the 'LL' or 'll' suffix. A
literal without 'LL' or 'll' acts as before: it may become unsigned
long but not long long. (For targets where int and long have the same
size, some literals change from unsigned int to unsigned long.)
Type `arith` may be too narrow for long long values. Add a second
type `writh` for wide arithmetic, and change some variables from arith
to writh. This may cause bugs if I forget to use writh, or if a
conversion from writh to arith overflows. I mark some conversions
with (arith) or (writh) casts.
- BigPars, SmallPars: Remove SPECIAL_ARITHMETICS. This feature
would change arith to a different type, but can't work, because it
would conflict with definitions of arith in both <em_arith.h> and
<flt_arith.h>.
- LLlex.c: Understand 'LL' or 'll' suffix. Cut size of constant when
it overflows writh, not only when it overflows the target machine's
types. (This cut might not be necessary, because we might cut it
again later.) When picking signed long or unsigned long, check the
target's long type, not the compiler's arith type; the old check
for `val >= 0` was broken where sizeof(arith) > 4.
- LLlex.h: Change struct token's tok_ival to writh, so it can hold a
long long literal.
- arith.c: Adjust to VL_VALUE being writh. Don't convert between
float and integer at compile-time if the integer might be too wide
for <flt_arith.h>. Add writh2str(), because writh might be too
wide for long2str().
- arith.h: Remove SPECIAL_ARITHMETICS. Declare full_mask[] here,
not in several *.c files. Declare writh2str().
- ch3.c, ch3bin.c, ch3mon.c, declarator.c, statement.g: Remove
obsolete casts. Adjust to VL_VALUE being writh.
- conversion.c, stab.c: Don't declare full_mask[].
- cstoper.c: Use writh for constant operations on VL_VALUE, and for
full_mask[].
- declar., field.c, ival.g: Add casts.
- dumpidf.c: Need to #include "parameters.h" before checking DEBUG.
Use writh2str, because "%ld" might not work.
- eval.c, eval.h: Add casts. Use writh when writing a wide constant
in EM.
- expr.c: Add and remove casts. In fill_int_expr(), make expression
from long long literal. In chk_cst_expr(), allow long long as
constant expression, so the compiler may accept `case 123LL:` in a
switch statement.
- expr.str: Change struct value's vl_value and struct expr's VL_VALUE
to writh, so an expression may have a long long value at compile
time.
- statement.g: Remove obsolete casts.
- switch.c, switch.str: Use writh in case entries for switch
statements, so `switch (ll) {...}` with long long ll works.
- tokenname.c: Add ULNGLNG so LLlex.c can use it for literals.
2019-09-05 02:14:38 +00:00
|
|
|
if (expr->VL_VALUE == 1) {
|
2019-02-18 16:42:15 +00:00
|
|
|
C_bra(l_body);
|
|
|
|
}
|
|
|
|
#ifdef LINT
|
Add long long literals like 123LL to ACK C.
For now, a long long literal must have the 'LL' or 'll' suffix. A
literal without 'LL' or 'll' acts as before: it may become unsigned
long but not long long. (For targets where int and long have the same
size, some literals change from unsigned int to unsigned long.)
Type `arith` may be too narrow for long long values. Add a second
type `writh` for wide arithmetic, and change some variables from arith
to writh. This may cause bugs if I forget to use writh, or if a
conversion from writh to arith overflows. I mark some conversions
with (arith) or (writh) casts.
- BigPars, SmallPars: Remove SPECIAL_ARITHMETICS. This feature
would change arith to a different type, but can't work, because it
would conflict with definitions of arith in both <em_arith.h> and
<flt_arith.h>.
- LLlex.c: Understand 'LL' or 'll' suffix. Cut size of constant when
it overflows writh, not only when it overflows the target machine's
types. (This cut might not be necessary, because we might cut it
again later.) When picking signed long or unsigned long, check the
target's long type, not the compiler's arith type; the old check
for `val >= 0` was broken where sizeof(arith) > 4.
- LLlex.h: Change struct token's tok_ival to writh, so it can hold a
long long literal.
- arith.c: Adjust to VL_VALUE being writh. Don't convert between
float and integer at compile-time if the integer might be too wide
for <flt_arith.h>. Add writh2str(), because writh might be too
wide for long2str().
- arith.h: Remove SPECIAL_ARITHMETICS. Declare full_mask[] here,
not in several *.c files. Declare writh2str().
- ch3.c, ch3bin.c, ch3mon.c, declarator.c, statement.g: Remove
obsolete casts. Adjust to VL_VALUE being writh.
- conversion.c, stab.c: Don't declare full_mask[].
- cstoper.c: Use writh for constant operations on VL_VALUE, and for
full_mask[].
- declar., field.c, ival.g: Add casts.
- dumpidf.c: Need to #include "parameters.h" before checking DEBUG.
Use writh2str, because "%ld" might not work.
- eval.c, eval.h: Add casts. Use writh when writing a wide constant
in EM.
- expr.c: Add and remove casts. In fill_int_expr(), make expression
from long long literal. In chk_cst_expr(), allow long long as
constant expression, so the compiler may accept `case 123LL:` in a
switch statement.
- expr.str: Change struct value's vl_value and struct expr's VL_VALUE
to writh, so an expression may have a long long value at compile
time.
- statement.g: Remove obsolete casts.
- switch.c, switch.str: Use writh in case entries for switch
statements, so `switch (ll) {...}` with long long ll works.
- tokenname.c: Add ULNGLNG so LLlex.c can use it for literals.
2019-09-05 02:14:38 +00:00
|
|
|
end_do_stmt(1, expr->VL_VALUE != 0);
|
2019-02-18 16:42:15 +00:00
|
|
|
#endif /* LINT */
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
code_expr(expr, RVAL, TRUE, l_body, l_break);
|
|
|
|
#ifdef LINT
|
|
|
|
end_do_stmt(0, 0);
|
|
|
|
#endif /* LINT */
|
|
|
|
}
|
|
|
|
C_df_ilb(l_break);
|
|
|
|
}
|
|
|
|
')'
|
|
|
|
';'
|
|
|
|
{
|
|
|
|
unstack_stmt();
|
|
|
|
free_expression(expr);
|
|
|
|
}
|
|
|
|
;
|
|
|
|
|
|
|
|
/* 3.6.5.3 */
|
|
|
|
for_statement
|
|
|
|
{ struct expr *e_init = 0, *e_test = 0, *e_incr = 0;
|
|
|
|
label l_break = text_label();
|
|
|
|
label l_continue = text_label();
|
|
|
|
label l_body = text_label();
|
|
|
|
label l_test = text_label();
|
|
|
|
}
|
|
|
|
:
|
|
|
|
FOR
|
|
|
|
{ stack_stmt(l_break, l_continue);
|
|
|
|
}
|
|
|
|
'('
|
|
|
|
[
|
|
|
|
expression(&e_init)
|
|
|
|
{ code_expr(e_init, RVAL, FALSE, NO_LABEL, NO_LABEL);
|
|
|
|
}
|
|
|
|
]?
|
|
|
|
';'
|
|
|
|
{ C_df_ilb(l_test);
|
|
|
|
}
|
|
|
|
[
|
|
|
|
expression(&e_test)
|
|
|
|
{
|
|
|
|
opnd2test(&e_test, FOR);
|
|
|
|
if (is_cp_cst(e_test)) {
|
Add long long literals like 123LL to ACK C.
For now, a long long literal must have the 'LL' or 'll' suffix. A
literal without 'LL' or 'll' acts as before: it may become unsigned
long but not long long. (For targets where int and long have the same
size, some literals change from unsigned int to unsigned long.)
Type `arith` may be too narrow for long long values. Add a second
type `writh` for wide arithmetic, and change some variables from arith
to writh. This may cause bugs if I forget to use writh, or if a
conversion from writh to arith overflows. I mark some conversions
with (arith) or (writh) casts.
- BigPars, SmallPars: Remove SPECIAL_ARITHMETICS. This feature
would change arith to a different type, but can't work, because it
would conflict with definitions of arith in both <em_arith.h> and
<flt_arith.h>.
- LLlex.c: Understand 'LL' or 'll' suffix. Cut size of constant when
it overflows writh, not only when it overflows the target machine's
types. (This cut might not be necessary, because we might cut it
again later.) When picking signed long or unsigned long, check the
target's long type, not the compiler's arith type; the old check
for `val >= 0` was broken where sizeof(arith) > 4.
- LLlex.h: Change struct token's tok_ival to writh, so it can hold a
long long literal.
- arith.c: Adjust to VL_VALUE being writh. Don't convert between
float and integer at compile-time if the integer might be too wide
for <flt_arith.h>. Add writh2str(), because writh might be too
wide for long2str().
- arith.h: Remove SPECIAL_ARITHMETICS. Declare full_mask[] here,
not in several *.c files. Declare writh2str().
- ch3.c, ch3bin.c, ch3mon.c, declarator.c, statement.g: Remove
obsolete casts. Adjust to VL_VALUE being writh.
- conversion.c, stab.c: Don't declare full_mask[].
- cstoper.c: Use writh for constant operations on VL_VALUE, and for
full_mask[].
- declar., field.c, ival.g: Add casts.
- dumpidf.c: Need to #include "parameters.h" before checking DEBUG.
Use writh2str, because "%ld" might not work.
- eval.c, eval.h: Add casts. Use writh when writing a wide constant
in EM.
- expr.c: Add and remove casts. In fill_int_expr(), make expression
from long long literal. In chk_cst_expr(), allow long long as
constant expression, so the compiler may accept `case 123LL:` in a
switch statement.
- expr.str: Change struct value's vl_value and struct expr's VL_VALUE
to writh, so an expression may have a long long value at compile
time.
- statement.g: Remove obsolete casts.
- switch.c, switch.str: Use writh in case entries for switch
statements, so `switch (ll) {...}` with long long ll works.
- tokenname.c: Add ULNGLNG so LLlex.c can use it for literals.
2019-09-05 02:14:38 +00:00
|
|
|
if (e_test->VL_VALUE == 0) {
|
2019-02-18 16:42:15 +00:00
|
|
|
C_bra(l_break);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
code_expr(e_test, RVAL, TRUE, l_body, l_break);
|
|
|
|
C_df_ilb(l_body);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
]?
|
|
|
|
';'
|
|
|
|
expression(&e_incr)?
|
|
|
|
')'
|
|
|
|
{
|
|
|
|
#ifdef LINT
|
|
|
|
start_for_stmt(e_test);
|
|
|
|
#endif /* LINT */
|
|
|
|
}
|
|
|
|
statement
|
|
|
|
{
|
|
|
|
#ifdef LINT
|
|
|
|
end_loop_body();
|
|
|
|
#endif /* LINT */
|
|
|
|
C_df_ilb(l_continue);
|
|
|
|
if (e_incr)
|
|
|
|
code_expr(e_incr, RVAL, FALSE,
|
|
|
|
NO_LABEL, NO_LABEL);
|
|
|
|
C_bra(l_test);
|
|
|
|
C_df_ilb(l_break);
|
|
|
|
unstack_stmt();
|
|
|
|
free_expression(e_init);
|
|
|
|
free_expression(e_test);
|
|
|
|
free_expression(e_incr);
|
|
|
|
#ifdef LINT
|
|
|
|
end_loop_stmt();
|
|
|
|
#endif /* LINT */
|
|
|
|
}
|
|
|
|
;
|
|
|
|
|
|
|
|
/* 3.6.4.2 */
|
|
|
|
switch_statement
|
|
|
|
{
|
|
|
|
struct expr *expr;
|
|
|
|
}
|
|
|
|
:
|
|
|
|
SWITCH
|
|
|
|
'('
|
|
|
|
expression(&expr)
|
|
|
|
{
|
|
|
|
code_startswitch(&expr);
|
|
|
|
#ifdef LINT
|
|
|
|
start_switch_part(is_cp_cst(expr));
|
|
|
|
#endif /* LINT */
|
|
|
|
}
|
|
|
|
')'
|
|
|
|
statement
|
|
|
|
{
|
|
|
|
#ifdef LINT
|
|
|
|
end_switch_stmt();
|
|
|
|
#endif /* LINT */
|
|
|
|
code_endswitch();
|
|
|
|
free_expression(expr);
|
|
|
|
}
|
|
|
|
;
|
|
|
|
|
|
|
|
/* 3.6.1 (partially) */
|
|
|
|
case_statement
|
|
|
|
{
|
|
|
|
struct expr *expr;
|
|
|
|
}
|
|
|
|
:
|
|
|
|
CASE
|
|
|
|
constant_expression(&expr)
|
|
|
|
{
|
|
|
|
#ifdef LINT
|
|
|
|
lint_case_stmt(0);
|
|
|
|
#endif /* LINT */
|
|
|
|
code_case(expr);
|
|
|
|
free_expression(expr);
|
|
|
|
}
|
|
|
|
':'
|
|
|
|
statement
|
|
|
|
;
|
|
|
|
|
|
|
|
/* 3.6.1 (partially) */
|
|
|
|
default_statement
|
|
|
|
:
|
|
|
|
DEFAULT
|
|
|
|
{
|
|
|
|
#ifdef LINT
|
|
|
|
lint_case_stmt(1);
|
|
|
|
#endif /* LINT */
|
|
|
|
code_default();
|
|
|
|
}
|
|
|
|
':'
|
|
|
|
statement
|
|
|
|
;
|
|
|
|
|
|
|
|
/* 3.6.6.4 */
|
|
|
|
return_statement
|
|
|
|
{ struct expr *expr = 0;
|
|
|
|
}
|
|
|
|
:
|
|
|
|
RETURN
|
|
|
|
[
|
|
|
|
expression(&expr)
|
|
|
|
{
|
|
|
|
#ifdef LINT
|
|
|
|
lint_ret_conv(expr);
|
|
|
|
#endif /* LINT */
|
|
|
|
|
|
|
|
do_return_expr(expr);
|
|
|
|
free_expression(expr);
|
|
|
|
#ifdef LINT
|
|
|
|
lint_return_stmt(VALRETURNED);
|
|
|
|
#endif /* LINT */
|
|
|
|
}
|
|
|
|
|
|
|
|
|
empty
|
|
|
|
{
|
|
|
|
do_return();
|
|
|
|
#ifdef LINT
|
|
|
|
lint_return_stmt(NOVALRETURNED);
|
|
|
|
#endif /* LINT */
|
|
|
|
}
|
|
|
|
]
|
|
|
|
';'
|
|
|
|
;
|
|
|
|
|
|
|
|
/* 3.6.6.1 (partially) */
|
|
|
|
jump
|
|
|
|
{ struct idf *idf;
|
|
|
|
}
|
|
|
|
:
|
|
|
|
GOTO
|
|
|
|
identifier(&idf)
|
|
|
|
';'
|
|
|
|
{
|
|
|
|
apply_label(idf);
|
|
|
|
C_bra((label)idf->id_label->df_address);
|
|
|
|
#ifdef LINT
|
|
|
|
lint_jump_stmt(idf);
|
|
|
|
#endif /* LINT */
|
|
|
|
}
|
|
|
|
;
|
|
|
|
|
|
|
|
/* 3.6.2 */
|
|
|
|
compound_statement
|
|
|
|
{
|
|
|
|
#ifdef DBSYMTAB
|
|
|
|
static int brc_level = 1;
|
|
|
|
int decl_seen = brc_level == 1;
|
|
|
|
#endif /* DBSYMTAB */
|
|
|
|
}
|
|
|
|
:
|
|
|
|
'{'
|
|
|
|
{
|
|
|
|
stack_level();
|
|
|
|
}
|
|
|
|
[%while ((DOT != IDENTIFIER && AHEAD != ':') ||
|
|
|
|
(DOT == IDENTIFIER && AHEAD == IDENTIFIER))
|
|
|
|
/* >>> conflict on TYPE_IDENTIFIER, IDENTIFIER */
|
|
|
|
declaration
|
|
|
|
{
|
|
|
|
#ifdef DBSYMTAB
|
|
|
|
decl_seen++;
|
|
|
|
#endif /* DBSYMTAB */
|
|
|
|
}
|
|
|
|
]*
|
|
|
|
{
|
|
|
|
#ifdef DBSYMTAB
|
|
|
|
++brc_level;
|
|
|
|
if (options['g'] && decl_seen) {
|
|
|
|
C_ms_std((char *) 0, N_LBRAC, brc_level);
|
|
|
|
}
|
|
|
|
#endif /* DBSYMTAB */
|
|
|
|
}
|
|
|
|
[%persistent
|
|
|
|
statement
|
|
|
|
]*
|
|
|
|
'}'
|
|
|
|
{
|
|
|
|
unstack_level();
|
|
|
|
#ifdef DBSYMTAB
|
|
|
|
if (options['g'] && decl_seen) {
|
|
|
|
C_ms_std((char *) 0, N_RBRAC, brc_level);
|
|
|
|
}
|
|
|
|
brc_level--;
|
|
|
|
#endif /* DBSYMTAB */
|
|
|
|
}
|
|
|
|
;
|