mirror of
https://github.com/ZDoom/gzdoom.git
synced 2025-01-18 15:42:34 +00:00
Represent enumerations as constant definitions
- Instead of representating enumeration values with a special node type, use the same ZCC_ConstantDef nodes that const_def produces. These are created at the same scope as the ZCC_Enum, rather than being contained entirely within it. To mark the end of enums for a single instance of ZCC_Enum, a ZCC_EnumTerminator node is now appended to the chain of ZCC_ConstantDefs.
This commit is contained in:
parent
d5fa550118
commit
af8e0f2ba6
3 changed files with 93 additions and 44 deletions
|
@ -301,16 +301,13 @@ static void PrintEnum(FLispString &out, ZCC_TreeNode *node)
|
||||||
out.Open("enum");
|
out.Open("enum");
|
||||||
out.AddName(enode->EnumName);
|
out.AddName(enode->EnumName);
|
||||||
PrintBuiltInType(out, enode->EnumType);
|
PrintBuiltInType(out, enode->EnumType);
|
||||||
PrintNodes(out, enode->Elements, false, true);
|
out.Add(enode->Elements == NULL ? "nil" : "...", 3);
|
||||||
out.Close();
|
out.Close();
|
||||||
}
|
}
|
||||||
|
|
||||||
static void PrintEnumNode(FLispString &out, ZCC_TreeNode *node)
|
static void PrintEnumTerminator(FLispString &out, ZCC_TreeNode *node)
|
||||||
{
|
{
|
||||||
ZCC_EnumNode *enode = (ZCC_EnumNode *)node;
|
out.Open("enum-term");
|
||||||
out.Open("enum-node");
|
|
||||||
out.AddName(enode->ElemName);
|
|
||||||
PrintNodes(out, enode->ElemValue, false);
|
|
||||||
out.Close();
|
out.Close();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -802,7 +799,7 @@ void (* const TreeNodePrinter[NUM_AST_NODE_TYPES])(FLispString &, ZCC_TreeNode *
|
||||||
PrintClass,
|
PrintClass,
|
||||||
PrintStruct,
|
PrintStruct,
|
||||||
PrintEnum,
|
PrintEnum,
|
||||||
PrintEnumNode,
|
PrintEnumTerminator,
|
||||||
PrintStates,
|
PrintStates,
|
||||||
PrintStatePart,
|
PrintStatePart,
|
||||||
PrintStateLabel,
|
PrintStateLabel,
|
||||||
|
|
|
@ -33,6 +33,16 @@ static void SetNodeLine(ZCC_TreeNode *name, int line)
|
||||||
// If a is non-null, appends b to a. Otherwise, sets a to b.
|
// If a is non-null, appends b to a. Otherwise, sets a to b.
|
||||||
#define SAFE_APPEND(a,b) \
|
#define SAFE_APPEND(a,b) \
|
||||||
if (a == NULL) a = b; else a->AppendSibling(b);
|
if (a == NULL) a = b; else a->AppendSibling(b);
|
||||||
|
|
||||||
|
#define UNARY_EXPR(X,T) NEW_AST_NODE(ExprUnary, expr, X); expr->Operation = T; expr->Operand = X; expr->Type = NULL
|
||||||
|
#define BINARY_EXPR(X,Y,T) NEW_AST_NODE(ExprBinary, expr, X); expr->Operation = T; expr->Type = NULL; expr->Left = X; expr->Right = Y
|
||||||
|
|
||||||
|
#define NEW_INTCONST_NODE(name,type,val,tok) \
|
||||||
|
NEW_AST_NODE(ExprConstant, name, tok); \
|
||||||
|
name->Operation = PEX_ConstValue; \
|
||||||
|
name->Type = type; \
|
||||||
|
name->IntVal = val
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
%token_prefix ZCC_
|
%token_prefix ZCC_
|
||||||
|
@ -237,20 +247,81 @@ struct_body(X) ::= struct_member(A) struct_body(B). { X = A; A->AppendSibling(B
|
||||||
struct_member(X) ::= declarator_no_fun(A). { X = A; }
|
struct_member(X) ::= declarator_no_fun(A). { X = A; }
|
||||||
struct_member(X) ::= enum_def(A). { X = A; }
|
struct_member(X) ::= enum_def(A). { X = A; }
|
||||||
|
|
||||||
|
/*----- Constant Definition ------*/
|
||||||
|
/* Like UnrealScript, a constant's type is implied by its value's type. */
|
||||||
|
const_def(X) ::= CONST(T) IDENTIFIER(A) EQ expr(B) SEMICOLON.
|
||||||
|
{
|
||||||
|
NEW_AST_NODE(ConstantDef,def,T);
|
||||||
|
def->Name = A.Name();
|
||||||
|
def->Value = B;
|
||||||
|
def->Symbol = NULL;
|
||||||
|
X = def;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/*----- Enum Definition -----*/
|
/*----- Enum Definition -----*/
|
||||||
/* Enumerators are lists of named integers. */
|
/* Enumerators are lists of named integers. */
|
||||||
|
|
||||||
%type enum_list {ZCC_EnumNode *}
|
%type enum_list {ZCC_ConstantDef *}
|
||||||
%type opt_enum_list {ZCC_EnumNode *}
|
%type opt_enum_list {ZCC_ConstantDef *}
|
||||||
%type enumerator {ZCC_EnumNode *}
|
%type enumerator {ZCC_ConstantDef *}
|
||||||
|
|
||||||
enum_def(X) ::= ENUM(T) IDENTIFIER(A) enum_type(B) LBRACE opt_enum_list(C) RBRACE opt_semicolon.
|
enum_def(X) ::= ENUM(T) IDENTIFIER(A) enum_type(B) LBRACE opt_enum_list(C) RBRACE(U) opt_semicolon.
|
||||||
{
|
{
|
||||||
NEW_AST_NODE(Enum,def,T);
|
NEW_AST_NODE(Enum,def,T);
|
||||||
def->EnumName = A.Name();
|
def->EnumName = A.Name();
|
||||||
def->EnumType = (EZCCBuiltinType)B.Int;
|
def->EnumType = (EZCCBuiltinType)B.Int;
|
||||||
def->Elements = C;
|
def->Elements = C;
|
||||||
|
|
||||||
|
// If the first element does not have an explicit value, make it 0.
|
||||||
|
if (C != NULL)
|
||||||
|
{
|
||||||
|
ZCC_ConstantDef *node = C, *prev = node;
|
||||||
|
ZCC_ExprConstant *one = NULL;
|
||||||
|
|
||||||
|
if (node->Value == NULL)
|
||||||
|
{
|
||||||
|
NEW_INTCONST_NODE(zero, TypeSInt32, 0, C);
|
||||||
|
node->Value = zero;
|
||||||
|
}
|
||||||
|
for (node = static_cast<ZCC_ConstantDef *>(node->SiblingNext);
|
||||||
|
node != C;
|
||||||
|
prev = node, node = static_cast<ZCC_ConstantDef *>(node->SiblingNext))
|
||||||
|
{
|
||||||
|
assert(node->NodeType == AST_ConstantDef);
|
||||||
|
// Leave explicit values alone.
|
||||||
|
if (node->Value != NULL)
|
||||||
|
{
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
// Compute implicit values by adding one to the preceding value.
|
||||||
|
assert(prev->Value != NULL);
|
||||||
|
// If the preceding node is a constant, then we can do this now.
|
||||||
|
if (prev->Value->Operation == PEX_ConstValue && prev->Value->Type->IsA(RUNTIME_CLASS(PInt)))
|
||||||
|
{
|
||||||
|
NEW_INTCONST_NODE(cval, prev->Value->Type, static_cast<ZCC_ExprConstant *>(prev->Value)->IntVal + 1, node);
|
||||||
|
node->Value = cval;
|
||||||
|
}
|
||||||
|
// Otherwise, create a new addition expression to add 1.
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (one == NULL)
|
||||||
|
{ // Use a single 1 node for the right-hand side of the addition.
|
||||||
|
NEW_INTCONST_NODE(cval, TypeSInt32, 1, T);
|
||||||
|
one = cval;
|
||||||
|
}
|
||||||
|
BINARY_EXPR(prev->Value, one, PEX_Add);
|
||||||
|
node->Value = expr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Add a new terminating node, to indicate that the ConstantDefs for this enum are done.
|
||||||
|
NEW_AST_NODE(EnumTerminator,term,U);
|
||||||
|
C->AppendSibling(term);
|
||||||
|
}
|
||||||
|
if (C != NULL)
|
||||||
|
{
|
||||||
|
def->AppendSibling(C);
|
||||||
|
}
|
||||||
X = def;
|
X = def;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -266,16 +337,18 @@ opt_enum_list(X) ::= enum_list(A) opt_comma. { X = A; }
|
||||||
|
|
||||||
enumerator(X) ::= IDENTIFIER(A).
|
enumerator(X) ::= IDENTIFIER(A).
|
||||||
{
|
{
|
||||||
NEW_AST_NODE(EnumNode,node,A);
|
NEW_AST_NODE(ConstantDef,node,A);
|
||||||
node->ElemName = A.Name();
|
node->Name = A.Name();
|
||||||
node->ElemValue = NULL;
|
node->Value = NULL;
|
||||||
|
node->Symbol = NULL;
|
||||||
X = node;
|
X = node;
|
||||||
}
|
}
|
||||||
enumerator(X) ::= IDENTIFIER(A) EQ expr(B). /* Expression must be constant. */
|
enumerator(X) ::= IDENTIFIER(A) EQ expr(B). /* Expression must be constant. */
|
||||||
{
|
{
|
||||||
NEW_AST_NODE(EnumNode,node,A);
|
NEW_AST_NODE(ConstantDef,node,A);
|
||||||
node->ElemName = A.Name();
|
node->Name = A.Name();
|
||||||
node->ElemValue = B;
|
node->Value = B;
|
||||||
|
node->Symbol = NULL;
|
||||||
X = node;
|
X = node;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -711,15 +784,6 @@ func_param_flags(X) ::= func_param_flags(A) IN(T). { X.Int = A.Int | ZCC_In; X
|
||||||
func_param_flags(X) ::= func_param_flags(A) OUT(T). { X.Int = A.Int | ZCC_Out; X.SourceLoc = A.SourceLoc ? A.SourceLoc : T.SourceLoc; }
|
func_param_flags(X) ::= func_param_flags(A) OUT(T). { X.Int = A.Int | ZCC_Out; X.SourceLoc = A.SourceLoc ? A.SourceLoc : T.SourceLoc; }
|
||||||
func_param_flags(X) ::= func_param_flags(A) OPTIONAL(T). { X.Int = A.Int | ZCC_Optional; X.SourceLoc = A.SourceLoc ? A.SourceLoc : T.SourceLoc; }
|
func_param_flags(X) ::= func_param_flags(A) OPTIONAL(T). { X.Int = A.Int | ZCC_Optional; X.SourceLoc = A.SourceLoc ? A.SourceLoc : T.SourceLoc; }
|
||||||
|
|
||||||
/* Like UnrealScript, a constant's type is implied by its value's type. */
|
|
||||||
const_def(X) ::= CONST(T) IDENTIFIER(A) EQ expr(B) SEMICOLON.
|
|
||||||
{
|
|
||||||
NEW_AST_NODE(ConstantDef,def,T);
|
|
||||||
def->Name = A.Name();
|
|
||||||
def->Value = B;
|
|
||||||
X = def;
|
|
||||||
}
|
|
||||||
|
|
||||||
/************ Expressions ************/
|
/************ Expressions ************/
|
||||||
|
|
||||||
/* We use default to access a class's default instance. */
|
/* We use default to access a class's default instance. */
|
||||||
|
@ -731,11 +795,6 @@ const_def(X) ::= CONST(T) IDENTIFIER(A) EQ expr(B) SEMICOLON.
|
||||||
%type unary_expr{ZCC_Expression *}
|
%type unary_expr{ZCC_Expression *}
|
||||||
%type constant{ZCC_ExprConstant *}
|
%type constant{ZCC_ExprConstant *}
|
||||||
|
|
||||||
%include {
|
|
||||||
#define UNARY_EXPR(X,T) NEW_AST_NODE(ExprUnary, expr, X); expr->Operation = T; expr->Operand = X; expr->Type = NULL
|
|
||||||
#define BINARY_EXPR(X,Y,T) NEW_AST_NODE(ExprBinary, expr, X); expr->Operation = T; expr->Type = NULL; expr->Left = X; expr->Right = Y
|
|
||||||
}
|
|
||||||
|
|
||||||
/*----- Primary Expressions -----*/
|
/*----- Primary Expressions -----*/
|
||||||
|
|
||||||
primary(X) ::= IDENTIFIER(A).
|
primary(X) ::= IDENTIFIER(A).
|
||||||
|
@ -1119,18 +1178,12 @@ constant(X) ::= string_constant(A).
|
||||||
}
|
}
|
||||||
constant(X) ::= INTCONST(A).
|
constant(X) ::= INTCONST(A).
|
||||||
{
|
{
|
||||||
NEW_AST_NODE(ExprConstant, intconst, A);
|
NEW_INTCONST_NODE(intconst, TypeSInt32, A.Int, A);
|
||||||
intconst->Operation = PEX_ConstValue;
|
|
||||||
intconst->Type = TypeSInt32;
|
|
||||||
intconst->IntVal = A.Int;
|
|
||||||
X = intconst;
|
X = intconst;
|
||||||
}
|
}
|
||||||
constant(X) ::= UINTCONST(A).
|
constant(X) ::= UINTCONST(A).
|
||||||
{
|
{
|
||||||
NEW_AST_NODE(ExprConstant, intconst, A);
|
NEW_INTCONST_NODE(intconst, TypeUInt32, A.Int, A);
|
||||||
intconst->Operation = PEX_ConstValue;
|
|
||||||
intconst->Type = TypeUInt32;
|
|
||||||
intconst->IntVal = A.Int;
|
|
||||||
X = intconst;
|
X = intconst;
|
||||||
}
|
}
|
||||||
constant(X) ::= FLOATCONST(A).
|
constant(X) ::= FLOATCONST(A).
|
||||||
|
|
|
@ -57,7 +57,7 @@ enum EZCCTreeNodeType
|
||||||
AST_Class,
|
AST_Class,
|
||||||
AST_Struct,
|
AST_Struct,
|
||||||
AST_Enum,
|
AST_Enum,
|
||||||
AST_EnumNode,
|
AST_EnumTerminator,
|
||||||
AST_States,
|
AST_States,
|
||||||
AST_StatePart,
|
AST_StatePart,
|
||||||
AST_StateLabel,
|
AST_StateLabel,
|
||||||
|
@ -251,13 +251,11 @@ struct ZCC_Enum : ZCC_TreeNode
|
||||||
{
|
{
|
||||||
ENamedName EnumName;
|
ENamedName EnumName;
|
||||||
EZCCBuiltinType EnumType;
|
EZCCBuiltinType EnumType;
|
||||||
struct ZCC_EnumNode *Elements;
|
struct ZCC_ConstantDef *Elements;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct ZCC_EnumNode : ZCC_TreeNode
|
struct ZCC_EnumTerminator : ZCC_TreeNode
|
||||||
{
|
{
|
||||||
ENamedName ElemName;
|
|
||||||
ZCC_TreeNode *ElemValue;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
struct ZCC_States : ZCC_TreeNode
|
struct ZCC_States : ZCC_TreeNode
|
||||||
|
@ -478,6 +476,7 @@ struct ZCC_ConstantDef : ZCC_TreeNode
|
||||||
{
|
{
|
||||||
ENamedName Name;
|
ENamedName Name;
|
||||||
ZCC_Expression *Value;
|
ZCC_Expression *Value;
|
||||||
|
PSymbolConst *Symbol;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct ZCC_Declarator : ZCC_TreeNode
|
struct ZCC_Declarator : ZCC_TreeNode
|
||||||
|
|
Loading…
Reference in a new issue