1
0
mirror of https://github.com/cc65/cc65.git synced 2024-06-01 13:41:34 +00:00
cc65/src/cc65/stmt.c

796 lines
17 KiB
C
Raw Normal View History

/*
* stmt.c
*
* Ullrich von Bassewitz, 06.08.1998
*
* Original by John R. Dunning - see copyleft.jrd
*/
#include <stdio.h>
#include <string.h>
/* common */
#include "xmalloc.h"
/* cc65 */
#include "asmcode.h"
#include "asmlabel.h"
#include "codegen.h"
#include "datatype.h"
#include "error.h"
#include "expr.h"
#include "function.h"
#include "global.h"
#include "goto.h"
#include "litpool.h"
#include "locals.h"
#include "loop.h"
#include "pragma.h"
#include "scanner.h"
#include "symtab.h"
#include "stmt.h"
/*****************************************************************************/
/* Data */
/*****************************************************************************/
/* Maximum count of cases */
#define CASE_MAX 257
/*****************************************************************************/
/* Forwards */
/*****************************************************************************/
/*****************************************************************************/
/* Code */
/*****************************************************************************/
static int IfStatement (void)
/* Handle an 'if' statement */
{
unsigned Label1;
int GotBreak;
/* Skip the if */
NextToken ();
/* Generate a jump label and parse the condition */
Label1 = GetLocalLabel ();
test (Label1, 0);
/* Parse the if body */
GotBreak = Statement ();
/* Else clause present? */
if (CurTok.Tok != TOK_ELSE) {
g_defcodelabel (Label1);
/* Since there's no else clause, we're not sure, if the a break
* statement is really executed.
*/
return 0;
} else {
/* Generate a jump around the else branch */
unsigned Label2 = GetLocalLabel ();
g_jump (Label2);
/* Skip the else */
NextToken ();
/* Define the target for the first test */
g_defcodelabel (Label1);
/* Total break only if both branches had a break. */
GotBreak &= Statement ();
/* Generate the label for the else clause */
g_defcodelabel (Label2);
/* Done */
return GotBreak;
}
}
static void DoStatement (void)
/* Handle the 'do' statement */
{
/* Get the loop control labels */
unsigned loop = GetLocalLabel ();
unsigned lab = GetLocalLabel ();
/* Skip the while token */
NextToken ();
/* Add the loop to the loop stack */
AddLoop (oursp, loop, lab, 0, 0);
/* Define the head label */
g_defcodelabel (loop);
/* Parse the loop body */
Statement ();
/* Parse the end condition */
Consume (TOK_WHILE, "`while' expected");
test (loop, 1);
ConsumeSemi ();
/* Define the break label */
g_defcodelabel (lab);
/* Remove the loop from the loop stack */
DelLoop ();
}
static void WhileStatement (void)
/* Handle the 'while' statement */
{
/* Get the loop control labels */
unsigned loop = GetLocalLabel ();
unsigned lab = GetLocalLabel ();
/* Skip the while token */
NextToken ();
/* Add the loop to the loop stack */
AddLoop (oursp, loop, lab, 0, 0);
/* Define the head label */
g_defcodelabel (loop);
/* Test the loop condition */
test (lab, 0);
/* If the statement following the while loop is empty, that is, we have
* something like "while (1) ;", the test function ommitted the jump as
* an optimization. Since we know, the condition codes are set, we can
* do another small optimization here, and use a conditional jump
* instead an absolute one.
*/
if (CurTok.Tok == TOK_SEMI) {
/* Use a conditional jump */
g_truejump (CF_NONE, loop);
/* Shortcut */
NextToken ();
} else {
/* There is code inside the while loop, parse the body */
Statement ();
g_jump (loop);
g_defcodelabel (lab);
}
/* Remove the loop from the loop stack */
DelLoop ();
}
static void ReturnStatement (void)
/* Handle the 'return' statement */
{
struct expent lval;
NextToken ();
if (CurTok.Tok != TOK_SEMI) {
if (HasVoidReturn (CurrentFunc)) {
Error ("Returning a value in function with return type void");
}
/* Evaluate the return expression. Result will be in primary */
expression (&lval);
/* Convert the return value to the type of the function result */
if (!HasVoidReturn (CurrentFunc)) {
assignadjust (GetReturnType (CurrentFunc), &lval);
}
} else if (!HasVoidReturn (CurrentFunc)) {
Error ("Function `%s' must return a value", GetFuncName (CurrentFunc));
}
/* Cleanup the stack in case we're inside a block with locals */
g_space (oursp - GetTopLevelSP (CurrentFunc));
/* Output a jump to the function exit code */
g_jump (GetRetLab (CurrentFunc));
}
static void BreakStatement (void)
/* Handle the 'break' statement */
{
LoopDesc* L;
/* Skip the break */
NextToken ();
/* Get the current loop descriptor */
L = CurrentLoop ();
/* Check if we are inside a loop */
if (L == 0) {
/* Error: No current loop */
Error ("`break' statement not within loop or switch");
return;
}
/* Correct the stack pointer if needed */
g_space (oursp - L->StackPtr);
/* Jump to the exit label of the loop */
g_jump (L->Label);
}
static void ContinueStatement (void)
/* Handle the 'continue' statement */
{
LoopDesc* L;
/* Skip the continue */
NextToken ();
/* Get the current loop descriptor */
L = CurrentLoop ();
if (L) {
/* Search for the correct loop */
do {
if (L->Loop) {
break;
}
L = L->Next;
} while (L);
}
/* Did we find it? */
if (L == 0) {
Error ("`continue' statement not within a loop");
return;
}
/* Correct the stackpointer if needed */
g_space (oursp - L->StackPtr);
/* Output the loop code */
if (L->linc) {
g_jump (L->linc);
} else {
g_jump (L->Loop);
}
}
static void CascadeSwitch (struct expent* eval)
/* Handle a switch statement for chars with a cmp cascade for the selector */
{
unsigned ExitLab; /* Exit label */
unsigned NextLab; /* Next case label */
unsigned CodeLab; /* Label that starts the actual selector code */
int HaveBreak; /* Remember if we exited with break */
int HaveDefault; /* Remember if we had a default label */
int lcount; /* Label count */
unsigned Flags; /* Code generator flags */
struct expent lval; /* Case label expression */
long Val; /* Case label value */
/* Create a loop so we may break out, init labels */
ExitLab = GetLocalLabel ();
AddLoop (oursp, 0, ExitLab, 0, 0);
/* Setup some variables needed in the loop below */
Flags = TypeOf (eval->e_tptr) | CF_CONST | CF_FORCECHAR;
CodeLab = NextLab = 0;
HaveBreak = 1;
HaveDefault = 0;
/* Parse the labels */
lcount = 0;
while (CurTok.Tok != TOK_RCURLY) {
if (CurTok.Tok == TOK_CASE || CurTok.Tok == TOK_DEFAULT) {
/* If the code for the previous selector did not end with a
* break statement, we must jump over the next selector test.
*/
if (!HaveBreak) {
/* Define a label for the code */
if (CodeLab == 0) {
CodeLab = GetLocalLabel ();
}
g_jump (CodeLab);
}
/* If we have a cascade label, emit it */
if (NextLab) {
g_defcodelabel (NextLab);
NextLab = 0;
}
while (CurTok.Tok == TOK_CASE || CurTok.Tok == TOK_DEFAULT) {
/* Parse the selector */
if (CurTok.Tok == TOK_CASE) {
/* Count labels */
++lcount;
/* Skip the "case" token */
NextToken ();
/* Read the selector expression */
constexpr (&lval);
if (!IsClassInt (lval.e_tptr)) {
Error ("Switch quantity not an integer");
}
/* Check the range of the expression */
Val = lval.e_const;
switch (*eval->e_tptr) {
case T_SCHAR:
/* Signed char */
if (Val < -128 || Val > 127) {
Error ("Range error");
}
break;
case T_UCHAR:
if (Val < 0 || Val > 255) {
Error ("Range error");
}
break;
case T_INT:
if (Val < -32768 || Val > 32767) {
Error ("Range error");
}
break;
case T_UINT:
if (Val < 0 || Val > 65535) {
Error ("Range error");
}
break;
default:
Internal ("Invalid type: %02X", *eval->e_tptr & 0xFF);
}
/* Emit a compare */
g_cmp (Flags, Val);
/* If another case follows, we will jump to the code if
* the condition is true.
*/
if (CurTok.Tok == TOK_CASE) {
/* Create a code label if needed */
if (CodeLab == 0) {
CodeLab = GetLocalLabel ();
}
g_falsejump (CF_NONE, CodeLab);
} else if (CurTok.Tok != TOK_DEFAULT) {
/* No case follows, jump to next selector */
if (NextLab == 0) {
NextLab = GetLocalLabel ();
}
g_truejump (CF_NONE, NextLab);
}
/* Skip the colon */
ConsumeColon ();
} else {
/* Default case */
NextToken ();
/* Handle the pathologic case: DEFAULT followed by CASE */
if (CurTok.Tok == TOK_CASE) {
if (CodeLab == 0) {
CodeLab = GetLocalLabel ();
}
g_jump (CodeLab);
}
/* Skip the colon */
ConsumeColon ();
/* Remember that we had a default label */
HaveDefault = 1;
}
}
}
/* Emit a code label if we have one */
if (CodeLab) {
g_defcodelabel (CodeLab);
CodeLab = 0;
}
/* Parse statements */
if (CurTok.Tok != TOK_RCURLY) {
HaveBreak = Statement ();
}
}
/* Check if we have any labels */
if (lcount == 0 && !HaveDefault) {
Warning ("No case labels");
}
/* Define the exit label and, if there's a next label left, create this
* one, too.
*/
if (NextLab) {
g_defcodelabel (NextLab);
}
g_defcodelabel (ExitLab);
/* Eat the closing curly brace */
NextToken ();
/* End the loop */
DelLoop ();
}
static void TableSwitch (struct expent* eval)
/* Handle a switch statement via table based selector */
{
/* Entry for one case in a switch statement */
struct swent {
long sw_const; /* selector value */
unsigned sw_lab; /* label for this selector */
};
int dlabel; /* for default */
int lab; /* exit label */
int label; /* label for case */
int lcase; /* label for compares */
int lcount; /* Label count */
int HaveBreak; /* Last statement has a break */
int HaveDefault; /* Remember if we had a default label */
unsigned Flags; /* Code generator flags */
struct expent lval; /* Case label expression */
struct swent *p;
struct swent *swtab;
/* Allocate memory for the switch table */
swtab = xmalloc (CASE_MAX * sizeof (struct swent));
/* Create a look so we may break out, init labels */
HaveBreak = 0; /* Keep gcc silent */
HaveDefault = 0; /* No default case until now */
dlabel = 0; /* init */
lab = GetLocalLabel (); /* get exit */
p = swtab;
AddLoop (oursp, 0, lab, 0, 0);
/* Jump behind the code for the CASE labels */
g_jump (lcase = GetLocalLabel ());
lcount = 0;
while (CurTok.Tok != TOK_RCURLY) {
if (CurTok.Tok == TOK_CASE || CurTok.Tok == TOK_DEFAULT) {
if (lcount >= CASE_MAX) {
Fatal ("Too many case labels");
}
label = GetLocalLabel ();
do {
if (CurTok.Tok == TOK_CASE) {
NextToken ();
constexpr (&lval);
if (!IsClassInt (lval.e_tptr)) {
Error ("Switch quantity not an integer");
}
p->sw_const = lval.e_const;
p->sw_lab = label;
++p;
++lcount;
} else {
NextToken ();
dlabel = label;
HaveDefault = 1;
}
ConsumeColon ();
} while (CurTok.Tok == TOK_CASE || CurTok.Tok == TOK_DEFAULT);
g_defcodelabel (label);
HaveBreak = 0;
}
if (CurTok.Tok != TOK_RCURLY) {
HaveBreak = Statement ();
}
}
/* Check if we have any labels */
if (lcount == 0 && !HaveDefault) {
Warning ("No case labels");
}
/* Eat the closing curly brace */
NextToken ();
/* If the last statement doesn't have a break or return, add one */
if (!HaveBreak) {
g_jump (lab);
}
/* Actual selector code goes here */
g_defcodelabel (lcase);
/* Create the call to the switch subroutine */
Flags = TypeOf (eval->e_tptr);
g_switch (Flags);
/* First entry is negative of label count */
g_defdata (CF_INT | CF_CONST, -((int)lcount)-1, 0);
/* Create the case selector table */
p = swtab;
while (lcount) {
g_case (Flags, p->sw_lab, p->sw_const); /* Create one label */
--lcount;
++p;
}
if (dlabel) {
g_jump (dlabel);
}
g_defcodelabel (lab);
DelLoop ();
/* Free the allocated space for the labels */
xfree (swtab);
}
static void SwitchStatement (void)
/* Handle a 'switch' statement */
{
struct expent eval; /* Switch statement expression */
/* Eat the "switch" */
NextToken ();
/* Read the switch expression */
ConsumeLParen ();
intexpr (&eval);
ConsumeRParen ();
/* result of expr is in P */
ConsumeLCurly ();
/* Now decide which sort of switch we will create: */
if (IsTypeChar (eval.e_tptr) || (CodeSizeFactor >= 200 && IsClassInt (eval.e_tptr))) {
CascadeSwitch (&eval);
} else {
TableSwitch (&eval);
}
}
static void ForStatement (void)
/* Handle a 'for' statement */
{
struct expent lval1;
struct expent lval2;
struct expent lval3;
/* Get several local labels needed later */
unsigned TestLabel = GetLocalLabel ();
unsigned lab = GetLocalLabel ();
unsigned IncLabel = GetLocalLabel ();
unsigned lstat = GetLocalLabel ();
/* Skip the FOR token */
NextToken ();
/* Add the loop to the loop stack */
AddLoop (oursp, TestLabel, lab, IncLabel, lstat);
/* Skip the opening paren */
ConsumeLParen ();
/* Parse the initializer expression */
if (CurTok.Tok != TOK_SEMI) {
expression (&lval1);
}
ConsumeSemi ();
/* Label for the test expressions */
g_defcodelabel (TestLabel);
/* Parse the text expression */
if (CurTok.Tok != TOK_SEMI) {
boolexpr (&lval2);
g_truejump (CF_NONE, lstat);
g_jump (lab);
} else {
g_jump (lstat);
}
ConsumeSemi ();
/* Label for the increment expression */
g_defcodelabel (IncLabel);
/* Parse the increment expression */
if (CurTok.Tok != TOK_RPAREN) {
expression (&lval3);
}
/* Jump to the test */
g_jump (TestLabel);
/* Skip the closing paren */
ConsumeRParen ();
/* Loop body */
g_defcodelabel (lstat);
Statement ();
/* Jump back to the increment expression */
g_jump (IncLabel);
/* Declare the break label */
g_defcodelabel (lab);
/* Remove the loop from the loop stack */
DelLoop ();
}
static int CompoundStatement (void)
/* Compound statement. Allow any number of statements inside braces. The
* function returns true if the last statement was a break or return.
*/
{
int GotBreak;
/* Remember the stack at block entry */
int OldStack = oursp;
/* Enter a new lexical level */
EnterBlockLevel ();
/* Skip the rcurly */
NextToken ();
/* Parse local variable declarations if any */
DeclareLocals ();
/* Now process statements in this block */
GotBreak = 0;
while (CurTok.Tok != TOK_RCURLY) {
if (CurTok.Tok != TOK_CEOF) {
GotBreak = Statement ();
} else {
break;
}
}
/* Clean up the stack. */
if (!GotBreak) {
g_space (oursp - OldStack);
}
oursp = OldStack;
/* Skip the closing brace */
ConsumeRCurly ();
/* Emit references to imports/exports for this block */
EmitExternals ();
/* Leave the lexical level */
LeaveBlockLevel ();
return GotBreak;
}
int Statement (void)
/* Statement parser. Returns 1 if the statement does a return/break, returns
* 0 otherwise
*/
{
struct expent lval;
/* Check for a label */
if (CurTok.Tok == TOK_IDENT && NextTok.Tok == TOK_COLON) {
/* Special handling for a label */
DoLabel ();
} else {
switch (CurTok.Tok) {
case TOK_LCURLY:
return CompoundStatement ();
case TOK_IF:
return IfStatement ();
case TOK_WHILE:
WhileStatement ();
break;
case TOK_DO:
DoStatement ();
break;
case TOK_SWITCH:
SwitchStatement ();
break;
case TOK_RETURN:
ReturnStatement ();
ConsumeSemi ();
return 1;
case TOK_BREAK:
BreakStatement ();
ConsumeSemi ();
return 1;
case TOK_CONTINUE:
ContinueStatement ();
ConsumeSemi ();
return 1;
case TOK_FOR:
ForStatement ();
break;
case TOK_GOTO:
GotoStatement ();
ConsumeSemi ();
return 1;
case TOK_SEMI:
/* Ignore it */
NextToken ();
break;
case TOK_PRAGMA:
DoPragma ();
break;
default:
/* Actual statement */
expression (&lval);
ConsumeSemi ();
}
}
return 0;
}