string table in the object file that (currently) holds all identifiers from the import, export and debug info sections. The plan is to put all strings into this table, so we have them in a central place and don't waste memory. Apart from that, the indices are unique, so comparing strings should be a lot easier than before (as soon as the programs take advantage of this fact, which is currently not the case). git-svn-id: svn://svn.cc65.org/cc65/trunk@2169 b7a2c559-68d2-44c3-8de9-860c34a00d81
559 lines
14 KiB
C
559 lines
14 KiB
C
/*****************************************************************************/
|
|
/* */
|
|
/* expr.c */
|
|
/* */
|
|
/* Expression evaluation for the ld65 linker */
|
|
/* */
|
|
/* */
|
|
/* */
|
|
/* (C) 1998-2000 Ullrich von Bassewitz */
|
|
/* Wacholderweg 14 */
|
|
/* D-70597 Stuttgart */
|
|
/* EMail: uz@musoftware.de */
|
|
/* */
|
|
/* */
|
|
/* This software is provided 'as-is', without any expressed or implied */
|
|
/* warranty. In no event will the authors be held liable for any damages */
|
|
/* arising from the use of this software. */
|
|
/* */
|
|
/* Permission is granted to anyone to use this software for any purpose, */
|
|
/* including commercial applications, and to alter it and redistribute it */
|
|
/* freely, subject to the following restrictions: */
|
|
/* */
|
|
/* 1. The origin of this software must not be misrepresented; you must not */
|
|
/* claim that you wrote the original software. If you use this software */
|
|
/* in a product, an acknowledgment in the product documentation would be */
|
|
/* appreciated but is not required. */
|
|
/* 2. Altered source versions must be plainly marked as such, and must not */
|
|
/* be misrepresented as being the original software. */
|
|
/* 3. This notice may not be removed or altered from any source */
|
|
/* distribution. */
|
|
/* */
|
|
/*****************************************************************************/
|
|
|
|
|
|
|
|
/* common */
|
|
#include "check.h"
|
|
#include "exprdefs.h"
|
|
#include "xmalloc.h"
|
|
|
|
/* ld65 */
|
|
#include "global.h"
|
|
#include "error.h"
|
|
#include "fileio.h"
|
|
#include "segments.h"
|
|
#include "expr.h"
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
|
/* Helpers */
|
|
/*****************************************************************************/
|
|
|
|
|
|
|
|
static ExprNode* NewExprNode (ObjData* O)
|
|
/* Create a new expression node */
|
|
{
|
|
/* Allocate fresh memory */
|
|
ExprNode* N = xmalloc (sizeof (ExprNode));
|
|
N->Op = EXPR_NULL;
|
|
N->Left = 0;
|
|
N->Right = 0;
|
|
N->Obj = O;
|
|
N->V.Val = 0;
|
|
|
|
return N;
|
|
}
|
|
|
|
|
|
|
|
static void FreeExprNode (ExprNode* E)
|
|
/* Free a node */
|
|
{
|
|
/* Free the memory */
|
|
xfree (E);
|
|
}
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
|
/* Code */
|
|
/*****************************************************************************/
|
|
|
|
|
|
|
|
void FreeExpr (ExprNode* Root)
|
|
/* Free the expression, Root is pointing to. */
|
|
{
|
|
if (Root) {
|
|
FreeExpr (Root->Left);
|
|
FreeExpr (Root->Right);
|
|
FreeExprNode (Root);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
int IsConstExpr (ExprNode* Root)
|
|
/* Return true if the given expression is a constant expression, that is, one
|
|
* with no references to external symbols.
|
|
*/
|
|
{
|
|
int Const;
|
|
Export* E;
|
|
|
|
if (EXPR_IS_LEAF (Root->Op)) {
|
|
switch (Root->Op) {
|
|
|
|
case EXPR_LITERAL:
|
|
return 1;
|
|
|
|
case EXPR_SYMBOL:
|
|
/* Get the referenced export */
|
|
E = GetExprExport (Root);
|
|
/* If this export has a mark set, we've already encountered it.
|
|
* This means that the export is used to define it's own value,
|
|
* which in turn means, that we have a circular reference.
|
|
*/
|
|
if (ExportHasMark (E)) {
|
|
Error ("Circular reference for symbol `%s', %s(%lu)",
|
|
E->Name, GetSourceFileName (E->Obj, E->Pos.Name),
|
|
E->Pos.Line);
|
|
Const = 0;
|
|
} else {
|
|
MarkExport (E);
|
|
Const = IsConstExport (E);
|
|
UnmarkExport (E);
|
|
}
|
|
return Const;
|
|
|
|
default:
|
|
return 0;
|
|
|
|
}
|
|
} else if (EXPR_IS_UNARY (Root->Op)) {
|
|
|
|
return IsConstExpr (Root->Left);
|
|
|
|
} else {
|
|
|
|
/* We must handle shortcut boolean expressions here */
|
|
switch (Root->Op) {
|
|
|
|
case EXPR_BAND:
|
|
if (IsConstExpr (Root->Left)) {
|
|
/* lhs is const, if it is zero, don't eval right */
|
|
if (GetExprVal (Root->Left) == 0) {
|
|
return 1;
|
|
} else {
|
|
return IsConstExpr (Root->Right);
|
|
}
|
|
} else {
|
|
/* lhs not const --> tree not const */
|
|
return 0;
|
|
}
|
|
break;
|
|
|
|
case EXPR_BOR:
|
|
if (IsConstExpr (Root->Left)) {
|
|
/* lhs is const, if it is not zero, don't eval right */
|
|
if (GetExprVal (Root->Left) != 0) {
|
|
return 1;
|
|
} else {
|
|
return IsConstExpr (Root->Right);
|
|
}
|
|
} else {
|
|
/* lhs not const --> tree not const */
|
|
return 0;
|
|
}
|
|
break;
|
|
|
|
default:
|
|
/* All others are handled normal */
|
|
return IsConstExpr (Root->Left) && IsConstExpr (Root->Right);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
Import* GetExprImport (ExprNode* Expr)
|
|
/* Get the import data structure for a symbol expression node */
|
|
{
|
|
/* Check that this is really a symbol */
|
|
PRECONDITION (Expr->Op == EXPR_SYMBOL);
|
|
|
|
/* Return the import */
|
|
return Expr->Obj->Imports [Expr->V.ImpNum];
|
|
}
|
|
|
|
|
|
|
|
Export* GetExprExport (ExprNode* Expr)
|
|
/* Get the exported symbol for a symbol expression node */
|
|
{
|
|
/* Check that this is really a symbol */
|
|
PRECONDITION (Expr->Op == EXPR_SYMBOL);
|
|
|
|
/* Return the export */
|
|
return Expr->Obj->Imports [Expr->V.ImpNum]->V.Exp;
|
|
}
|
|
|
|
|
|
|
|
Section* GetExprSection (ExprNode* Expr)
|
|
/* Get the segment for a section expression node */
|
|
{
|
|
/* Check that this is really a section node */
|
|
PRECONDITION (Expr->Op == EXPR_SECTION);
|
|
|
|
/* If we have an object file, get the section from it, otherwise
|
|
* (internally generated expressions), get the section from the
|
|
* section pointer.
|
|
*/
|
|
if (Expr->Obj) {
|
|
/* Return the export */
|
|
return Expr->Obj->Sections [Expr->V.SegNum];
|
|
} else {
|
|
return Expr->V.Sec;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
long GetExprVal (ExprNode* Expr)
|
|
/* Get the value of a constant expression */
|
|
{
|
|
long Right, Left, Val;
|
|
Section* S;
|
|
Export* E;
|
|
|
|
switch (Expr->Op) {
|
|
|
|
case EXPR_LITERAL:
|
|
return Expr->V.Val;
|
|
|
|
case EXPR_SYMBOL:
|
|
/* Get the referenced export */
|
|
E = GetExprExport (Expr);
|
|
/* If this export has a mark set, we've already encountered it.
|
|
* This means that the export is used to define it's own value,
|
|
* which in turn means, that we have a circular reference.
|
|
*/
|
|
if (ExportHasMark (E)) {
|
|
CircularRefError (E);
|
|
Val = 0;
|
|
} else {
|
|
MarkExport (E);
|
|
Val = GetExportVal (E);
|
|
UnmarkExport (E);
|
|
}
|
|
return Val;
|
|
|
|
case EXPR_SECTION:
|
|
S = GetExprSection (Expr);
|
|
return S->Offs + S->Seg->PC;
|
|
|
|
case EXPR_SEGMENT:
|
|
return Expr->V.Seg->PC;
|
|
|
|
case EXPR_MEMAREA:
|
|
return Expr->V.Mem->Start;
|
|
|
|
case EXPR_PLUS:
|
|
return GetExprVal (Expr->Left) + GetExprVal (Expr->Right);
|
|
|
|
case EXPR_MINUS:
|
|
return GetExprVal (Expr->Left) - GetExprVal (Expr->Right);
|
|
|
|
case EXPR_MUL:
|
|
return GetExprVal (Expr->Left) * GetExprVal (Expr->Right);
|
|
|
|
case EXPR_DIV:
|
|
Left = GetExprVal (Expr->Left);
|
|
Right = GetExprVal (Expr->Right);
|
|
if (Right == 0) {
|
|
Error ("Division by zero");
|
|
}
|
|
return Left / Right;
|
|
|
|
case EXPR_MOD:
|
|
Left = GetExprVal (Expr->Left);
|
|
Right = GetExprVal (Expr->Right);
|
|
if (Right == 0) {
|
|
Error ("Modulo operation with zero");
|
|
}
|
|
return Left % Right;
|
|
|
|
case EXPR_OR:
|
|
return GetExprVal (Expr->Left) | GetExprVal (Expr->Right);
|
|
|
|
case EXPR_XOR:
|
|
return GetExprVal (Expr->Left) ^ GetExprVal (Expr->Right);
|
|
|
|
case EXPR_AND:
|
|
return GetExprVal (Expr->Left) & GetExprVal (Expr->Right);
|
|
|
|
case EXPR_SHL:
|
|
return GetExprVal (Expr->Left) << GetExprVal (Expr->Right);
|
|
|
|
case EXPR_SHR:
|
|
return GetExprVal (Expr->Left) >> GetExprVal (Expr->Right);
|
|
|
|
case EXPR_EQ:
|
|
return (GetExprVal (Expr->Left) == GetExprVal (Expr->Right));
|
|
|
|
case EXPR_NE:
|
|
return (GetExprVal (Expr->Left) != GetExprVal (Expr->Right));
|
|
|
|
case EXPR_LT:
|
|
return (GetExprVal (Expr->Left) < GetExprVal (Expr->Right));
|
|
|
|
case EXPR_GT:
|
|
return (GetExprVal (Expr->Left) > GetExprVal (Expr->Right));
|
|
|
|
case EXPR_LE:
|
|
return (GetExprVal (Expr->Left) <= GetExprVal (Expr->Right));
|
|
|
|
case EXPR_GE:
|
|
return (GetExprVal (Expr->Left) >= GetExprVal (Expr->Right));
|
|
|
|
case EXPR_UNARY_MINUS:
|
|
return -GetExprVal (Expr->Left);
|
|
|
|
case EXPR_NOT:
|
|
return ~GetExprVal (Expr->Left);
|
|
|
|
case EXPR_BYTE0:
|
|
return GetExprVal (Expr->Left) & 0xFF;
|
|
|
|
case EXPR_BYTE1:
|
|
return (GetExprVal (Expr->Left) >> 8) & 0xFF;
|
|
|
|
case EXPR_BYTE2:
|
|
return (GetExprVal (Expr->Left) >> 16) & 0xFF;
|
|
|
|
case EXPR_BYTE3:
|
|
return (GetExprVal (Expr->Left) >> 24) & 0xFF;
|
|
|
|
case EXPR_SWAP:
|
|
Left = GetExprVal (Expr->Left);
|
|
return ((Left >> 8) & 0x00FF) | ((Left << 8) & 0xFF00);
|
|
|
|
case EXPR_BAND:
|
|
return GetExprVal (Expr->Left) && GetExprVal (Expr->Right);
|
|
|
|
case EXPR_BOR:
|
|
return GetExprVal (Expr->Left) || GetExprVal (Expr->Right);
|
|
|
|
case EXPR_BXOR:
|
|
return (GetExprVal (Expr->Left) != 0) ^ (GetExprVal (Expr->Right) != 0);
|
|
|
|
case EXPR_BNOT:
|
|
return !GetExprVal (Expr->Left);
|
|
|
|
default:
|
|
Internal ("Unknown expression Op type: %u", Expr->Op);
|
|
/* NOTREACHED */
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
ExprNode* LiteralExpr (long Val, ObjData* O)
|
|
/* Return an expression tree that encodes the given literal value */
|
|
{
|
|
ExprNode* Expr = NewExprNode (O);
|
|
Expr->Op = EXPR_LITERAL;
|
|
Expr->V.Val = Val;
|
|
return Expr;
|
|
}
|
|
|
|
|
|
|
|
ExprNode* MemoryExpr (Memory* Mem, long Offs, ObjData* O)
|
|
/* Return an expression tree that encodes an offset into a memory area */
|
|
{
|
|
ExprNode* Root;
|
|
|
|
ExprNode* Expr = NewExprNode (O);
|
|
Expr->Op = EXPR_MEMAREA;
|
|
Expr->V.Mem = Mem;
|
|
|
|
if (Offs != 0) {
|
|
Root = NewExprNode (O);
|
|
Root->Op = EXPR_PLUS;
|
|
Root->Left = Expr;
|
|
Root->Right = LiteralExpr (Offs, O);
|
|
} else {
|
|
Root = Expr;
|
|
}
|
|
|
|
return Root;
|
|
}
|
|
|
|
|
|
|
|
ExprNode* SegmentExpr (Segment* Seg, long Offs, ObjData* O)
|
|
/* Return an expression tree that encodes an offset into a segment */
|
|
{
|
|
ExprNode* Root;
|
|
|
|
ExprNode* Expr = NewExprNode (O);
|
|
Expr->Op = EXPR_SEGMENT;
|
|
Expr->V.Seg = Seg;
|
|
|
|
if (Offs != 0) {
|
|
Root = NewExprNode (O);
|
|
Root->Op = EXPR_PLUS;
|
|
Root->Left = Expr;
|
|
Root->Right = LiteralExpr (Offs, O);
|
|
} else {
|
|
Root = Expr;
|
|
}
|
|
|
|
return Root;
|
|
}
|
|
|
|
|
|
|
|
ExprNode* SectionExpr (Section* Sec, long Offs, ObjData* O)
|
|
/* Return an expression tree that encodes an offset into a section */
|
|
{
|
|
ExprNode* Root;
|
|
|
|
ExprNode* Expr = NewExprNode (O);
|
|
Expr->Op = EXPR_SECTION;
|
|
Expr->V.Sec = Sec;
|
|
|
|
if (Offs != 0) {
|
|
Root = NewExprNode (O);
|
|
Root->Op = EXPR_PLUS;
|
|
Root->Left = Expr;
|
|
Root->Right = LiteralExpr (Offs, O);
|
|
} else {
|
|
Root = Expr;
|
|
}
|
|
|
|
return Root;
|
|
}
|
|
|
|
|
|
|
|
ExprNode* ReadExpr (FILE* F, ObjData* O)
|
|
/* Read an expression from the given file */
|
|
{
|
|
ExprNode* Expr;
|
|
|
|
/* Read the node tag and handle NULL nodes */
|
|
unsigned char Op = Read8 (F);
|
|
if (Op == EXPR_NULL) {
|
|
return 0;
|
|
}
|
|
|
|
/* Create a new node */
|
|
Expr = NewExprNode (O);
|
|
Expr->Op = Op;
|
|
|
|
/* Check the tag and handle the different expression types */
|
|
if (EXPR_IS_LEAF (Op)) {
|
|
switch (Op) {
|
|
|
|
case EXPR_LITERAL:
|
|
Expr->V.Val = Read32Signed (F);
|
|
break;
|
|
|
|
case EXPR_SYMBOL:
|
|
/* Read the import number */
|
|
Expr->V.ImpNum = ReadVar (F);
|
|
break;
|
|
|
|
case EXPR_SECTION:
|
|
/* Read the segment number */
|
|
Expr->V.SegNum = Read8 (F);
|
|
break;
|
|
|
|
default:
|
|
Error ("Invalid expression op: %02X", Op);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/* Not a leaf node */
|
|
Expr->Left = ReadExpr (F, O);
|
|
Expr->Right = ReadExpr (F, O);
|
|
|
|
}
|
|
|
|
/* Return the tree */
|
|
return Expr;
|
|
}
|
|
|
|
|
|
|
|
int EqualExpr (ExprNode* E1, ExprNode* E2)
|
|
/* Check if two expressions are identical. */
|
|
{
|
|
/* If one pointer is NULL, both must be NULL */
|
|
if ((E1 == 0) ^ (E2 == 0)) {
|
|
return 0;
|
|
}
|
|
if (E1 == 0) {
|
|
return 1;
|
|
}
|
|
|
|
/* Both pointers not NULL, check OP */
|
|
if (E1->Op != E2->Op) {
|
|
return 0;
|
|
}
|
|
|
|
/* OPs are identical, check data for leafs, or subtrees */
|
|
switch (E1->Op) {
|
|
|
|
case EXPR_LITERAL:
|
|
/* Value must be identical */
|
|
return (E1->V.Val == E2->V.Val);
|
|
|
|
case EXPR_SYMBOL:
|
|
/* Import number must be identical */
|
|
return (E1->V.ImpNum == E2->V.ImpNum);
|
|
|
|
case EXPR_SECTION:
|
|
/* Section must be identical */
|
|
return (GetExprSection (E1) == GetExprSection (E2));
|
|
|
|
case EXPR_SEGMENT:
|
|
/* Segment must be identical */
|
|
return (E1->V.Seg == E2->V.Seg);
|
|
|
|
case EXPR_MEMAREA:
|
|
/* Memory area must be identical */
|
|
return (E1->V.Mem == E2->V.Mem );
|
|
|
|
default:
|
|
/* Not a leaf node */
|
|
return EqualExpr (E1->Left, E2->Left) && EqualExpr (E1->Right, E2->Right);
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|