mirror of
https://github.com/ZDoom/qzdoom.git
synced 2025-01-18 15:11:46 +00:00
aadd4e4de6
- If the parsing is a complete failure, then TopNode won't ever be set to anything during the parsing process.
500 lines
8 KiB
C
500 lines
8 KiB
C
struct ZCCParseState
|
|
{
|
|
ZCCParseState(FScanner &scanner) : sc(scanner)
|
|
{
|
|
TopNode = NULL;
|
|
}
|
|
|
|
FScanner ≻
|
|
FSharedStringArena Strings;
|
|
FMemArena SyntaxArena;
|
|
struct ZCC_TreeNode *TopNode;
|
|
};
|
|
|
|
union ZCCToken
|
|
{
|
|
int Int;
|
|
double Float;
|
|
FString *String;
|
|
|
|
ENamedName Name() { return ENamedName(Int); }
|
|
};
|
|
|
|
// Variable / Function modifiers
|
|
enum
|
|
{
|
|
ZCC_Native = 1 << 0,
|
|
ZCC_Static = 1 << 1,
|
|
ZCC_Private = 1 << 2,
|
|
ZCC_Protected = 1 << 3,
|
|
ZCC_Latent = 1 << 4,
|
|
ZCC_Final = 1 << 5,
|
|
ZCC_Meta = 1 << 6,
|
|
ZCC_Action = 1 << 7,
|
|
ZCC_Deprecated = 1 << 8,
|
|
ZCC_ReadOnly = 1 << 9,
|
|
ZCC_FuncConst = 1 << 10,
|
|
};
|
|
|
|
// Function parameter modifiers
|
|
enum
|
|
{
|
|
ZCC_In = 1 << 0,
|
|
ZCC_Out = 1 << 1,
|
|
ZCC_Optional = 1 << 2,
|
|
};
|
|
|
|
|
|
|
|
// Syntax tree structures.
|
|
enum EZCCTreeNodeType
|
|
{
|
|
AST_Identifier,
|
|
AST_Class,
|
|
AST_Struct,
|
|
AST_Enum,
|
|
AST_EnumNode,
|
|
AST_States,
|
|
AST_StatePart,
|
|
AST_StateLabel,
|
|
AST_StateStop,
|
|
AST_StateWait,
|
|
AST_StateFail,
|
|
AST_StateLoop,
|
|
AST_StateGoto,
|
|
AST_StateLine,
|
|
AST_VarName,
|
|
AST_Type,
|
|
AST_BasicType,
|
|
AST_MapType,
|
|
AST_DynArrayType,
|
|
AST_ClassType,
|
|
AST_Expression,
|
|
AST_ExprID,
|
|
AST_ExprString,
|
|
AST_ExprInt,
|
|
AST_ExprFloat,
|
|
AST_ExprFuncCall,
|
|
AST_ExprMemberAccess,
|
|
AST_ExprUnary,
|
|
AST_ExprBinary,
|
|
AST_ExprTrinary,
|
|
AST_FuncParm,
|
|
AST_Statement,
|
|
AST_CompoundStmt,
|
|
AST_ContinueStmt,
|
|
AST_BreakStmt,
|
|
AST_ReturnStmt,
|
|
AST_ExpressionStmt,
|
|
AST_IterationStmt,
|
|
AST_IfStmt,
|
|
AST_SwitchStmt,
|
|
AST_CaseStmt,
|
|
AST_AssignStmt,
|
|
AST_LocalVarStmt,
|
|
AST_FuncParamDecl,
|
|
AST_ConstantDef,
|
|
AST_Declarator,
|
|
AST_VarDeclarator,
|
|
AST_FuncDeclarator,
|
|
|
|
NUM_AST_NODE_TYPES
|
|
};
|
|
|
|
enum EZCCBuiltinType
|
|
{
|
|
ZCC_SInt8,
|
|
ZCC_UInt8,
|
|
ZCC_SInt16,
|
|
ZCC_UInt16,
|
|
ZCC_SInt32,
|
|
ZCC_UInt32,
|
|
ZCC_IntAuto, // for enums, autoselect appropriately sized int
|
|
|
|
ZCC_Bool,
|
|
ZCC_Float32,
|
|
ZCC_Float64,
|
|
ZCC_FloatAuto, // 32-bit in structs/classes, 64-bit everywhere else
|
|
ZCC_String,
|
|
ZCC_Vector2,
|
|
ZCC_Vector3,
|
|
ZCC_Vector4,
|
|
ZCC_Name,
|
|
ZCC_UserType,
|
|
|
|
ZCC_NUM_BUILT_IN_TYPES
|
|
};
|
|
|
|
enum EZCCExprType
|
|
{
|
|
PEX_Nil,
|
|
|
|
PEX_ID,
|
|
PEX_Super,
|
|
PEX_Self,
|
|
PEX_StringConst,
|
|
PEX_IntConst,
|
|
PEX_UIntConst,
|
|
PEX_FloatConst,
|
|
PEX_FuncCall,
|
|
PEX_ArrayAccess,
|
|
PEX_MemberAccess,
|
|
PEX_PostInc,
|
|
PEX_PostDec,
|
|
|
|
PEX_PreInc,
|
|
PEX_PreDec,
|
|
PEX_Negate,
|
|
PEX_AntiNegate,
|
|
PEX_BitNot,
|
|
PEX_BoolNot,
|
|
PEX_SizeOf,
|
|
PEX_AlignOf,
|
|
|
|
PEX_Add,
|
|
PEX_Sub,
|
|
PEX_Mul,
|
|
PEX_Div,
|
|
PEX_Mod,
|
|
PEX_Pow,
|
|
PEX_CrossProduct,
|
|
PEX_DotProduct,
|
|
PEX_LeftShift,
|
|
PEX_RightShift,
|
|
PEX_Concat,
|
|
|
|
PEX_LT,
|
|
PEX_GT,
|
|
PEX_LTEQ,
|
|
PEX_GTEQ,
|
|
PEX_LTGTEQ,
|
|
PEX_Is,
|
|
|
|
PEX_EQEQ,
|
|
PEX_NEQ,
|
|
PEX_APREQ,
|
|
|
|
PEX_BitAnd,
|
|
PEX_BitOr,
|
|
PEX_BitXor,
|
|
PEX_BoolAnd,
|
|
PEX_BoolOr,
|
|
|
|
PEX_Scope,
|
|
|
|
PEX_Trinary,
|
|
|
|
PEX_COUNT_OF
|
|
};
|
|
|
|
struct ZCC_TreeNode
|
|
{
|
|
// This tree node's siblings are stored in a circular linked list.
|
|
// When you get back to this node, you know you've been through
|
|
// the whole list.
|
|
ZCC_TreeNode *SiblingNext;
|
|
ZCC_TreeNode *SiblingPrev;
|
|
|
|
// Node type is one of the node types above, which corresponds with
|
|
// one of the structures below.
|
|
EZCCTreeNodeType NodeType;
|
|
|
|
// Appends a sibling to this node's sibling list.
|
|
void AppendSibling(ZCC_TreeNode *sibling)
|
|
{
|
|
if (sibling == NULL)
|
|
{
|
|
return;
|
|
}
|
|
|
|
// The new sibling node should only be in a list with itself.
|
|
assert(sibling->SiblingNext == sibling && sibling->SiblingPrev == sibling);
|
|
|
|
// Check integrity of our sibling list.
|
|
assert(SiblingPrev->SiblingNext == this);
|
|
assert(SiblingNext->SiblingPrev == this);
|
|
|
|
SiblingPrev->SiblingNext = sibling;
|
|
sibling->SiblingPrev = SiblingPrev;
|
|
SiblingPrev = sibling;
|
|
sibling->SiblingNext = this;
|
|
}
|
|
};
|
|
|
|
struct ZCC_Identifier : ZCC_TreeNode
|
|
{
|
|
ENamedName Id;
|
|
};
|
|
|
|
struct ZCC_Class : ZCC_TreeNode
|
|
{
|
|
ZCC_Identifier *ClassName;
|
|
ZCC_Identifier *ParentName;
|
|
ZCC_Identifier *Replaces;
|
|
VM_UWORD Flags;
|
|
ZCC_TreeNode *Body;
|
|
};
|
|
|
|
struct ZCC_Struct : ZCC_TreeNode
|
|
{
|
|
ENamedName StructName;
|
|
ZCC_TreeNode *Body;
|
|
};
|
|
|
|
struct ZCC_Enum : ZCC_TreeNode
|
|
{
|
|
ENamedName EnumName;
|
|
EZCCBuiltinType EnumType;
|
|
struct ZCC_EnumNode *Elements;
|
|
};
|
|
|
|
struct ZCC_EnumNode : ZCC_TreeNode
|
|
{
|
|
ENamedName ElemName;
|
|
ZCC_TreeNode *ElemValue;
|
|
};
|
|
|
|
struct ZCC_States : ZCC_TreeNode
|
|
{
|
|
struct ZCC_StatePart *Body;
|
|
};
|
|
|
|
struct ZCC_StatePart : ZCC_TreeNode
|
|
{
|
|
};
|
|
|
|
struct ZCC_StateLabel : ZCC_StatePart
|
|
{
|
|
ENamedName Label;
|
|
};
|
|
|
|
struct ZCC_StateStop : ZCC_StatePart
|
|
{
|
|
};
|
|
|
|
struct ZCC_StateWait : ZCC_StatePart
|
|
{
|
|
};
|
|
|
|
struct ZCC_StateFail : ZCC_StatePart
|
|
{
|
|
};
|
|
|
|
struct ZCC_StateLoop : ZCC_StatePart
|
|
{
|
|
};
|
|
|
|
struct ZCC_Expression : ZCC_TreeNode
|
|
{
|
|
EZCCExprType Operation;
|
|
};
|
|
|
|
struct ZCC_StateGoto : ZCC_StatePart
|
|
{
|
|
ZCC_Identifier *Label;
|
|
ZCC_Expression *Offset;
|
|
};
|
|
|
|
struct ZCC_StateLine : ZCC_StatePart
|
|
{
|
|
char Sprite[4];
|
|
BITFIELD bBright:1;
|
|
FString *Frames;
|
|
ZCC_Expression *Offset;
|
|
ZCC_TreeNode *Action;
|
|
};
|
|
|
|
struct ZCC_VarName : ZCC_TreeNode
|
|
{
|
|
ENamedName Name;
|
|
};
|
|
|
|
struct ZCC_Type : ZCC_TreeNode
|
|
{
|
|
ZCC_Expression *ArraySize; // NULL if not an array
|
|
};
|
|
|
|
struct ZCC_BasicType : ZCC_Type
|
|
{
|
|
EZCCBuiltinType Type;
|
|
ZCC_Identifier *UserType;
|
|
};
|
|
|
|
struct ZCC_MapType : ZCC_Type
|
|
{
|
|
ZCC_Type *KeyType;
|
|
ZCC_Type *ValueType;
|
|
};
|
|
|
|
struct ZCC_DynArrayType : ZCC_Type
|
|
{
|
|
ZCC_Type *ElementType;
|
|
};
|
|
|
|
struct ZCC_ClassType : ZCC_Type
|
|
{
|
|
ZCC_Identifier *Restriction;
|
|
};
|
|
|
|
struct ZCC_ExprID : ZCC_Expression
|
|
{
|
|
ENamedName Identifier;
|
|
};
|
|
|
|
struct ZCC_ExprString : ZCC_Expression
|
|
{
|
|
FString *Value;
|
|
};
|
|
|
|
struct ZCC_ExprInt : ZCC_Expression
|
|
{
|
|
int Value;
|
|
};
|
|
|
|
struct ZCC_ExprFloat : ZCC_Expression
|
|
{
|
|
double Value;
|
|
};
|
|
|
|
struct ZCC_FuncParm : ZCC_TreeNode
|
|
{
|
|
ZCC_Expression *Value;
|
|
ENamedName Label;
|
|
};
|
|
|
|
struct ZCC_ExprFuncCall : ZCC_Expression
|
|
{
|
|
ZCC_Expression *Function;
|
|
ZCC_FuncParm *Parameters;
|
|
};
|
|
|
|
struct ZCC_ExprMemberAccess : ZCC_Expression
|
|
{
|
|
ZCC_Expression *Left;
|
|
ENamedName Right;
|
|
};
|
|
|
|
struct ZCC_ExprUnary : ZCC_Expression
|
|
{
|
|
ZCC_Expression *Operand;
|
|
};
|
|
|
|
struct ZCC_ExprBinary : ZCC_Expression
|
|
{
|
|
ZCC_Expression *Left;
|
|
ZCC_Expression *Right;
|
|
};
|
|
|
|
struct ZCC_ExprTrinary : ZCC_Expression
|
|
{
|
|
ZCC_Expression *Test;
|
|
ZCC_Expression *Left;
|
|
ZCC_Expression *Right;
|
|
};
|
|
|
|
struct ZCC_Statement : ZCC_TreeNode
|
|
{
|
|
};
|
|
|
|
struct ZCC_CompoundStmt : ZCC_Statement
|
|
{
|
|
ZCC_Statement *Content;
|
|
};
|
|
|
|
struct ZCC_ContinueStmt : ZCC_Statement
|
|
{
|
|
};
|
|
|
|
struct ZCC_BreakStmt : ZCC_Statement
|
|
{
|
|
};
|
|
|
|
struct ZCC_ReturnStmt : ZCC_Statement
|
|
{
|
|
ZCC_Expression *Values;
|
|
};
|
|
|
|
struct ZCC_ExpressionStmt : ZCC_Statement
|
|
{
|
|
ZCC_Expression *Expression;
|
|
};
|
|
|
|
struct ZCC_IterationStmt : ZCC_Statement
|
|
{
|
|
ZCC_Expression *LoopCondition;
|
|
ZCC_Statement *LoopStatement;
|
|
ZCC_Statement *LoopBumper;
|
|
|
|
// Should the loop condition be checked at the
|
|
// start of the loop (before the LoopStatement)
|
|
// or at the end (after the LoopStatement)?
|
|
enum { Start, End } CheckAt;
|
|
};
|
|
|
|
struct ZCC_IfStmt : ZCC_Statement
|
|
{
|
|
ZCC_Expression *Condition;
|
|
ZCC_Statement *TruePath;
|
|
ZCC_Statement *FalsePath;
|
|
};
|
|
|
|
struct ZCC_SwitchStmt : ZCC_Statement
|
|
{
|
|
ZCC_Expression *Condition;
|
|
ZCC_Statement *Content;
|
|
};
|
|
|
|
struct ZCC_CaseStmt : ZCC_Statement
|
|
{
|
|
// A NULL Condition represents the default branch
|
|
ZCC_Expression *Condition;
|
|
};
|
|
|
|
struct ZCC_AssignStmt : ZCC_Statement
|
|
{
|
|
ZCC_Expression *Dests;
|
|
ZCC_Expression *Sources;
|
|
int AssignOp;
|
|
};
|
|
|
|
struct ZCC_LocalVarStmt : ZCC_Statement
|
|
{
|
|
ZCC_Type *Type;
|
|
ZCC_VarName *Vars;
|
|
ZCC_Expression *Inits;
|
|
};
|
|
|
|
struct ZCC_FuncParamDecl : ZCC_TreeNode
|
|
{
|
|
ZCC_Type *Type;
|
|
ENamedName Name;
|
|
int Flags;
|
|
};
|
|
|
|
struct ZCC_ConstantDef : ZCC_TreeNode
|
|
{
|
|
ENamedName Name;
|
|
ZCC_Expression *Value;
|
|
};
|
|
|
|
struct ZCC_Declarator : ZCC_TreeNode
|
|
{
|
|
ZCC_Type *Type;
|
|
int Flags;
|
|
};
|
|
|
|
// A variable in a class or struct.
|
|
struct ZCC_VarDeclarator : ZCC_Declarator
|
|
{
|
|
ZCC_VarName *Names;
|
|
};
|
|
|
|
// A function in a class.
|
|
struct ZCC_FuncDeclarator : ZCC_Declarator
|
|
{
|
|
ZCC_FuncParamDecl *Params;
|
|
ENamedName Name;
|
|
ZCC_Statement *Body;
|
|
};
|
|
|
|
FString ZCC_PrintAST(ZCC_TreeNode *root);
|