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:
parent
ea43c68b9b
commit
9e83b00152
@ -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;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -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 */
|
||||||
|
@ -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;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -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 */
|
||||||
|
Loading…
x
Reference in New Issue
Block a user