using System.Collections; using System.Collections.Generic; using System.Linq; namespace arookas { class sunExpression : sunNode { public sunExpression(sunSourceLocation location) : base(location) { } public override void Compile(sunContext context) { Stack operatorStack = new Stack(32); AnalyzeExpression(context, this, operatorStack); } void AnalyzeExpression(sunContext context, sunExpression expression, Stack operatorStack) { // this implementation assumes that the expression production child list alternates between operand and operator // we can safely assume this as the grammar "operand {binary_operator operand}" enforces it int stackCount = operatorStack.Count; foreach (var node in expression) { if (node is sunOperand) { var operand = node as sunOperand; // term var term = operand.Term; if (term is sunExpression) { AnalyzeExpression(context, term as sunExpression, operatorStack); } else { term.Compile(context); } var unaryOperators = operand.UnaryOperators; if (unaryOperators != null) { unaryOperators.Compile(context); } } else if (node is sunOperator) { var operatorNode = node as sunOperator; while (operatorStack.Count > stackCount && (operatorNode.IsLeftAssociative && operatorNode.Precedence <= operatorStack.Peek().Precedence) || (operatorNode.IsRightAssociative && operatorNode.Precedence < operatorStack.Peek().Precedence)) { operatorStack.Pop().Compile(context); } operatorStack.Push(operatorNode); } } while (operatorStack.Count > stackCount) { operatorStack.Pop().Compile(context); } } } class sunOperand : sunNode { public sunNode UnaryOperators { get { return Count > 1 ? this[0] : null; } } public sunNode Term { get { return this[Count - 1]; } } public sunOperand(sunSourceLocation location) : base(location) { } // operands are compiled in sunExpression.Compile } class sunUnaryOperatorList : sunNode { public sunUnaryOperatorList(sunSourceLocation location) : base(location) { } public override void Compile(sunContext context) { foreach (var child in this.Reverse()) { // compile unary operators in reverse order child.Compile(context); } } } class sunTernaryOperator : sunNode { public sunExpression Condition { get { return this[0] as sunExpression; } } public sunExpression TrueBody { get { return this[1] as sunExpression; } } public sunExpression FalseBody { get { return this[2] as sunExpression; } } public sunTernaryOperator(sunSourceLocation location) : base(location) { } public override void Compile(sunContext context) { Condition.Compile(context); var falsePrologue = context.Text.GotoIfZero(); TrueBody.Compile(context); var trueEpilogue = context.Text.Goto(); context.Text.ClosePoint(falsePrologue); FalseBody.Compile(context); context.Text.ClosePoint(trueEpilogue); } } // increment/decrement class sunPostfixAugment : sunOperand { public sunIdentifier Variable { get { return this[0] as sunIdentifier; } } public sunAugment Operator { get { return this[1] as sunAugment; } } public sunPostfixAugment(sunSourceLocation location) : base(location) { } public override void Compile(sunContext context) { var variableInfo = context.ResolveVariable(Variable); if (Parent is sunOperand) { context.Text.PushVariable(variableInfo); } Operator.Compile(context, variableInfo); context.Text.StoreVariable(variableInfo); } } class sunPrefixAugment : sunOperand { public sunAugment Operator { get { return this[0] as sunAugment; } } public sunIdentifier Variable { get { return this[1] as sunIdentifier; } } public sunPrefixAugment(sunSourceLocation location) : base(location) { } public override void Compile(sunContext context) { var variableInfo = context.ResolveVariable(Variable); Operator.Compile(context, variableInfo); context.Text.StoreVariable(variableInfo); if (Parent is sunOperand) { context.Text.PushVariable(variableInfo); } } } abstract class sunAugment : sunNode { protected sunAugment(sunSourceLocation location) : base(location) { } public abstract void Compile(sunContext context, sunVariableInfo variable); } class sunIncrement : sunAugment { public sunIncrement(sunSourceLocation location) : base(location) { } public override void Compile(sunContext context, sunVariableInfo variable) { context.Text.IncVariable(variable); } } class sunDecrement : sunAugment { public sunDecrement(sunSourceLocation location) : base(location) { } public override void Compile(sunContext context, sunVariableInfo variable) { context.Text.DecVariable(variable); } } }