1
0
mirror of https://github.com/cc65/cc65.git synced 2025-01-10 19:29:45 +00:00

Rewrote parsing of locals.

Removed non working code for register variables.


git-svn-id: svn://svn.cc65.org/cc65/trunk@96 b7a2c559-68d2-44c3-8de9-860c34a00d81
This commit is contained in:
cuz 2000-06-22 11:28:39 +00:00
parent ea43c68b9b
commit 9e83b00152
4 changed files with 235 additions and 282 deletions

View File

@ -61,8 +61,7 @@ struct Function {
type* ReturnType; /* Function return type */ type* ReturnType; /* Function return type */
struct FuncDesc* Desc; /* Function descriptor */ struct FuncDesc* Desc; /* Function descriptor */
CodeMark EntryCode; /* Backpatch addr for entry code */ CodeMark EntryCode; /* Backpatch addr for entry code */
int LocalMax; /* Total space for locals */ int Reserved; /* Reserved local space */
int LocalSize; /* Current space for locals */
unsigned RetLab; /* Return code label */ unsigned RetLab; /* Return code label */
}; };
@ -88,8 +87,7 @@ static Function* NewFunction (struct SymEntry* Sym)
F->ReturnType = Sym->Type + 1 + DECODE_SIZE; F->ReturnType = Sym->Type + 1 + DECODE_SIZE;
F->Desc = DecodePtr (Sym->Type + 1); F->Desc = DecodePtr (Sym->Type + 1);
F->EntryCode = 0; F->EntryCode = 0;
F->LocalMax = 0; F->Reserved = 0;
F->LocalSize = 0;
F->RetLab = GetLabel (); F->RetLab = GetLabel ();
/* Return the new structure */ /* Return the new structure */
@ -154,33 +152,36 @@ unsigned GetRetLab (const Function* F)
int AllocLocalSpace (Function* F, unsigned Size) int ReserveLocalSpace (Function* F, unsigned Size)
/* Allocate space for the function locals, return stack offset */ /* Reserve (but don't allocate) the given local space and return the stack
* offset.
*/
{ {
/* Add the size */ F->Reserved += Size;
F->LocalSize += Size; return oursp - F->Reserved;
if (F->LocalSize > F->LocalMax) { }
F->LocalMax = F->LocalSize;
void AllocLocalSpace (Function* F)
/* Allocate any local space previously reserved. The function will do
* nothing if there is no reserved local space.
*/
{
if (F->Reserved > 0) {
/* Switch to the code segment */
g_usecode ();
/* Create space on the stack */
g_space (F->Reserved);
/* Correct the stack pointer */
oursp -= F->Reserved;
/* Nothing more reserved */
F->Reserved = 0;
} }
/* Return the offset, it is below the initial stack pointer */
return -F->LocalSize;;
}
void FreeLocalSpace (Function* F, unsigned Size)
/* Free space allocated for function locals */
{
F->LocalSize -= Size;
}
unsigned GetLocalSpace (const Function* F)
/* Get the local variable space needed for the function */
{
return F->LocalMax;
} }

View File

@ -49,18 +49,19 @@ void RememberEntry (Function* F);
unsigned GetRetLab (const Function* F); unsigned GetRetLab (const Function* F);
/* Return the return jump label */ /* Return the return jump label */
int AllocLocalSpace (Function* F, unsigned Size); int ReserveLocalSpace (Function* F, unsigned Size);
/* Allocate space for the function locals, return stack offset */ /* Reserve (but don't allocate) the given local space and return the stack
* offset.
*/
void FreeLocalSpace (Function* F, unsigned Size); void AllocLocalSpace (Function* F);
/* Free space allocated for function locals */ /* Allocate any local space previously reserved. The function will do
* nothing if there is no reserved local space.
*/
void NewFunc (struct SymEntry* Func); void NewFunc (struct SymEntry* Func);
/* Parse argument declarations and function body. */ /* Parse argument declarations and function body. */
unsigned GetLocalSpace (const Function* F);
/* Get the local variable space needed for the function */
/* End of function.h */ /* End of function.h */

View File

@ -39,6 +39,7 @@
#include "asmlabel.h" #include "asmlabel.h"
#include "codegen.h" #include "codegen.h"
#include "declare.h" #include "declare.h"
#include "error.h"
#include "expr.h" #include "expr.h"
#include "function.h" #include "function.h"
#include "global.h" #include "global.h"
@ -129,50 +130,30 @@ static int AllocRegVar (const SymEntry* Sym, const type* tarray)
void DeclareLocals (void) static void ParseOneDecl (const DeclSpec* Spec)
/* Declare local variables and types. */ /* Parse one variable declaration */
{ {
int offs = oursp; /* Current stack offset for variable */
int AutoSpace = 0; /* Unallocated space on the stack */
int Size; /* Size of an auto variable */ int Size; /* Size of an auto variable */
int Reg; /* Register variable offset */ int SC; /* Storage class for symbol */
int SymData = 0; /* Symbol data (offset, label name, ...) */
unsigned flags = 0; /* Code generator flags */ unsigned flags = 0; /* Code generator flags */
int SymbolSC; /* Storage class for symbol */ Declaration Decl; /* Declaration data structure */
int ldata = 0; /* Local symbol data temp storage */
/* Loop until we don't find any more variables */
while (1) {
/* Check variable declarations. We need to distinguish between a
* default int type and the end of variable declarations. So we
* will do the following: If there is no explicit storage class
* specifier *and* no explicit type given, it is assume that we
* have reached the end of declarations.
*/
DeclSpec Spec;
ParseDeclSpec (&Spec, SC_AUTO, T_INT);
if ((Spec.Flags & DS_DEF_STORAGE) != 0 && (Spec.Flags & DS_DEF_TYPE) != 0) {
break;
}
/* Accept type only declarations */
if (curtok == TOK_SEMI) {
/* Type declaration only */
CheckEmptyDecl (&Spec);
NextToken ();
continue;
}
/* Parse a comma separated variable list */
while (1) {
Declaration Decl;
/* Remember the storage class for the new symbol */ /* Remember the storage class for the new symbol */
SymbolSC = Spec.StorageClass; SC = Spec->StorageClass;
/* Read the declaration */ /* Read the declaration */
ParseDecl (&Spec, &Decl, DM_NEED_IDENT); ParseDecl (Spec, &Decl, DM_NEED_IDENT);
/* Set the correct storage class for functions */
if (IsFunc (Decl.Type)) {
/* Function prototypes are always external */
if ((SC & SC_EXTERN) == 0) {
Warning (WARN_FUNC_MUST_BE_EXTERN);
}
SC |= SC_FUNC | SC_EXTERN;
}
/* If we don't have a name, this was flagged as an error earlier. /* If we don't have a name, this was flagged as an error earlier.
* To avoid problems later, use an anonymous name here. * To avoid problems later, use an anonymous name here.
@ -181,85 +162,25 @@ void DeclareLocals (void)
AnonName (Decl.Ident, "param"); AnonName (Decl.Ident, "param");
} }
if (!IsFunc (Decl.Type) && (SymbolSC & SC_TYPEDEF) != SC_TYPEDEF) { /* Handle anything that needs storage (no functions, no typdefs) */
if ((SC & SC_FUNC) != SC_FUNC && (SC & SC_TYPEDEF) != SC_TYPEDEF) {
/* Get the size of the variable */ /* Get the size of the variable */
Size = SizeOf (Decl.Type); Size = SizeOf (Decl.Type);
#if 0 if (SC & (SC_AUTO | SC_REGISTER)) {
/* Check the storage class */
if ((SymbolSC & SC_REGISTER) && (Reg = AllocRegVar (psym, tarray)) >= 0) {
/* We will store the current value of the register onto the
* stack, thus making functions with register variables
* reentrant. If we have pending auto variables, emit them
* now.
*/
g_usecode ();
g_space (AutoSpace);
oursp -= AutoSpace;
AutoSpace = 0;
/* Remember the register bank offset */
ldata = Reg;
/* Save the current register value onto the stack */
g_save_regvars (Reg, Size);
/* Allow variable initialization */
if (curtok == TOK_ASSIGN) {
struct expent lval;
/* Skip the '=' */
NextToken ();
/* Get the expression into the primary */
expression1 (&lval);
/* Make type adjustments if needed */
assignadjust (tarray, &lval);
/* Setup the type flags for the assignment */
flags = TypeOf (tarray) | CF_REGVAR;
if (Size == 1) {
flags |= CF_FORCECHAR;
}
/* Store the value into the register */
g_putstatic (flags, Reg, 0);
/* Mark the variable as referenced */
SymbolSC |= SC_REF;
}
/* Account for the stack space needed and remember the
* stack offset of the save area.
*/
offs -= Size;
psym->h_lattr = offs;
} else if (SymbolSC & (SC_AUTO | SC_REGISTER)) {
#endif
if (SymbolSC & (SC_AUTO | SC_REGISTER)) {
/* Auto variable */ /* Auto variable */
if (StaticLocals == 0) { if (StaticLocals == 0) {
/* Change SC in case it was register */ /* Change SC in case it was register */
SymbolSC = (SymbolSC & ~SC_REGISTER) | SC_AUTO; SC = (SC & ~SC_REGISTER) | SC_AUTO;
if (curtok == TOK_ASSIGN) { if (curtok == TOK_ASSIGN) {
struct expent lval; struct expent lval;
/* Switch to the code segment, allocate space for /* Allocate previously reserved local space */
* uninitialized variables. AllocLocalSpace (CurrentFunc);
*/
g_usecode ();
g_space (AutoSpace);
oursp -= AutoSpace;
AutoSpace = 0;
/* Skip the '=' */ /* Skip the '=' */
NextToken (); NextToken ();
@ -281,29 +202,29 @@ void DeclareLocals (void)
g_push (flags | TypeOf (Decl.Type), lval.e_const); g_push (flags | TypeOf (Decl.Type), lval.e_const);
/* Mark the variable as referenced */ /* Mark the variable as referenced */
SymbolSC |= SC_REF; SC |= SC_REF;
/* Variable is located at the current SP */
SymData = oursp;
} else { } else {
/* Non-initialized local variable. Just keep track of /* Non-initialized local variable. Just keep track of
* the space needed. * the space needed.
*/ */
AutoSpace += Size; SymData = ReserveLocalSpace (CurrentFunc, Size);
} }
/* Allocate space on the stack, assign the offset */
offs -= Size;
ldata = offs;
} else { } else {
/* Static local variables. */ /* Static local variables. */
SymbolSC = (SymbolSC & ~(SC_REGISTER | SC_AUTO)) | SC_STATIC; SC = (SC & ~(SC_REGISTER | SC_AUTO)) | SC_STATIC;
/* Put them into the BSS */ /* Put them into the BSS */
g_usebss (); g_usebss ();
/* Define the variable label */ /* Define the variable label */
g_defloclabel (ldata = GetLabel ()); SymData = GetLabel ();
g_defloclabel (SymData);
/* Reserve space for the data */ /* Reserve space for the data */
g_res (Size); g_res (Size);
@ -332,14 +253,14 @@ void DeclareLocals (void)
} }
/* Store the value into the variable */ /* Store the value into the variable */
g_putstatic (flags, ldata, 0); g_putstatic (flags, SymData, 0);
/* Mark the variable as referenced */ /* Mark the variable as referenced */
SymbolSC |= SC_REF; SC |= SC_REF;
} }
} }
} else if ((SymbolSC & SC_STATIC) == SC_STATIC) { } else if ((SC & SC_STATIC) == SC_STATIC) {
/* Static data */ /* Static data */
if (curtok == TOK_ASSIGN) { if (curtok == TOK_ASSIGN) {
@ -348,7 +269,8 @@ void DeclareLocals (void)
g_usedata (); g_usedata ();
/* Define the variable label */ /* Define the variable label */
g_defloclabel (ldata = GetLabel ()); SymData = GetLabel ();
g_defloclabel (SymData);
/* Skip the '=' */ /* Skip the '=' */
NextToken (); NextToken ();
@ -357,7 +279,7 @@ void DeclareLocals (void)
ParseInit (Decl.Type); ParseInit (Decl.Type);
/* Mark the variable as referenced */ /* Mark the variable as referenced */
SymbolSC |= SC_REF; SC |= SC_REF;
} else { } else {
@ -365,7 +287,8 @@ void DeclareLocals (void)
g_usebss (); g_usebss ();
/* Define the variable label */ /* Define the variable label */
g_defloclabel (ldata = GetLabel ()); SymData = GetLabel ();
g_defloclabel (SymData);
/* Reserve space for the data */ /* Reserve space for the data */
g_res (Size); g_res (Size);
@ -376,29 +299,67 @@ void DeclareLocals (void)
} }
/* If the symbol is not marked as external, it will be defined */ /* If the symbol is not marked as external, it will be defined */
if ((SymbolSC & SC_EXTERN) == 0) { if ((SC & SC_EXTERN) == 0) {
SymbolSC |= SC_DEF; SC |= SC_DEF;
} }
/* Add the symbol to the symbol table */ /* Add the symbol to the symbol table */
AddLocalSym (Decl.Ident, Decl.Type, SymbolSC, ldata); AddLocalSym (Decl.Ident, Decl.Type, SC, SymData);
}
if (curtok != TOK_COMMA) {
void DeclareLocals (void)
/* Declare local variables and types. */
{
/* Loop until we don't find any more variables */
while (1) {
/* Check variable declarations. We need to distinguish between a
* default int type and the end of variable declarations. So we
* will do the following: If there is no explicit storage class
* specifier *and* no explicit type given, it is assume that we
* have reached the end of declarations.
*/
DeclSpec Spec;
ParseDeclSpec (&Spec, SC_AUTO, T_INT);
if ((Spec.Flags & DS_DEF_STORAGE) != 0 && (Spec.Flags & DS_DEF_TYPE) != 0) {
break; break;
} }
NextToken ();
} /* Accept type only declarations */
if (curtok == TOK_SEMI) { if (curtok == TOK_SEMI) {
/* Type declaration only */
CheckEmptyDecl (&Spec);
NextToken (); NextToken ();
continue;
}
/* Parse a comma separated variable list */
while (1) {
/* Parse one declaration */
ParseOneDecl (&Spec);
/* Check if there is more */
if (curtok == TOK_COMMA) {
/* More to come */
NextToken ();
} else {
/* Done */
break;
} }
} }
/* A semicolon must follow */
ConsumeSemi ();
}
/* Be sure to allocate any reserved space for locals */
AllocLocalSpace (CurrentFunc);
/* In case we switched away from code segment, switch back now */ /* In case we switched away from code segment, switch back now */
g_usecode (); g_usecode ();
/* Create space for locals */
g_space (AutoSpace);
oursp -= AutoSpace;
} }

View File

@ -631,18 +631,8 @@ SymEntry* AddLabelSym (const char* Name, unsigned Flags)
SymEntry* AddLocalSym (const char* Name, type* Type, unsigned Flags, int Offs) SymEntry* AddLocalSym (const char* Name, type* Type, unsigned Flags, int Offs)
/* Add a local symbol and return the symbol entry */ /* Add a local symbol and return the symbol entry */
{ {
SymEntry* Entry;
/* Functions declared inside of functions do always have external linkage */
if (Type != 0 && IsFunc (Type)) {
if ((Flags & (SC_DEFAULT | SC_EXTERN)) == 0) {
Warning (WARN_FUNC_MUST_BE_EXTERN);
}
Flags = SC_EXTERN;
}
/* Do we have an entry with this name already? */ /* Do we have an entry with this name already? */
Entry = FindSymInTable (SymTab, Name, HashStr (Name)); SymEntry* Entry = FindSymInTable (SymTab, Name, HashStr (Name));
if (Entry) { if (Entry) {
/* We have a symbol with this name already */ /* We have a symbol with this name already */