Пример #1
0
        protected override TypeNode GetTypeFromToken(CompilerNode owner, string token)
        {
            if (token.StartsWith("int"))
            {
                int size = GetNumericSize(token.Substring(3));
                if (size > 0)
                {
                    return(new TypeNode(owner, TypeKind.Integer));
                }
                else
                {
                    return(null);
                }
            }

            if (token.StartsWith("uint"))
            {
                int size = GetNumericSize(token.Substring(4));
                if (size > 0)
                {
                    return(new TypeNode(owner, TypeKind.Integer));
                }
                else
                {
                    return(null);
                }
            }

            if (token.StartsWith("byte"))
            {
                var size = token.Substring(4);
                if (string.IsNullOrEmpty(size) || size == "s")
                {
                    return(new TypeNode(owner, TypeKind.ByteArray));
                }

                int val;
                if (int.TryParse(size, out val))
                {
                    if (val >= 1 && val <= 32)
                    {
                        return(new TypeNode(owner, TypeKind.ByteArray));
                    }
                }

                return(null);
            }

            switch (token)
            {
            case "bool": return(new TypeNode(owner, TypeKind.Boolean));

            case "string": return(new TypeNode(owner, TypeKind.String));

            case "address": return(new TypeNode(owner, TypeKind.Struct));    // TODO fixme

            default: return(null);
            }
        }
Пример #2
0
        protected StatementNode GenerateEntryPoint(CompilerNode owner, IEnumerable <MethodNode> methods)
        {
            var result = new BlockNode(owner);

            var opDecl = new DeclarationNode(result);

            opDecl.identifier = "operation";
            opDecl.type       = new TypeNode(opDecl, TypeKind.String);
            result.declarations.Add(opDecl);

            var assSt = new StackAssignmentNode(result);

            assSt.declaration = opDecl;
            result.statements.Add(assSt);

            foreach (var method in methods)
            {
                if (method.visibility != Visibility.Public)
                {
                    continue;
                }

                var ifExpr   = new IfNode(result);
                var cmpExpr  = new BinaryExpressionNode(ifExpr);
                var leftExpr = new VariableExpressionNode(cmpExpr);
                leftExpr.declaration = opDecl;

                var rightExpr = new LiteralExpressionNode(cmpExpr);
                rightExpr.kind  = LiteralKind.String;
                rightExpr.value = method.name;

                cmpExpr.left      = leftExpr;
                cmpExpr.right     = rightExpr;
                cmpExpr.@operator = OperatorKind.Equals;

                var callSt = new CallNode(ifExpr);
                callSt.method = method;

                ifExpr.expr       = cmpExpr;
                ifExpr.trueBranch = callSt;

                result.statements.Add(ifExpr);
            }

            result.statements.Add(new ExitNode(result));
            return(result);
        }
Пример #3
0
        protected override TypeNode GetTypeFromToken(CompilerNode owner, string token)
        {
            switch (token)
            {
            case "bool": return(new TypeNode(owner, TypeKind.Boolean));

            case "string": return(new TypeNode(owner, TypeKind.String));

            case "int": return(new TypeNode(owner, TypeKind.Integer));

            case "uint": return(new TypeNode(owner, TypeKind.Integer));

            case "var": return(new TypeNode(owner, TypeKind.Integer));    // TODO fixme

            default: return(null);
            }
        }
Пример #4
0
        protected StatementNode ParseStatement(List <Token> tokens, ref int index, CompilerNode owner)
        {
            BlockNode block = null;

            do
            {
                if (index >= tokens.Count)
                {
                    throw new ParserException(tokens.Last(), ParserException.Kind.EndOfStream);
                }

                var token = tokens[index];

                StatementNode statement = null;

                var type = GetTypeFromToken(owner, token.text);
                if (type != null)
                {
                    var decl = new DeclarationNode(block);
                    decl.type = type;
                    index++;
                    decl.identifier = ExpectIdentifier(tokens, ref index, false);

                    if (ExpectOptional(tokens, ref index, "="))
                    {
                        var node = new AssignmentNode(owner);
                        node.identifier = decl.identifier;
                        node.expr       = ParseExpression(tokens, ref index, block);
                        statement       = node;
                    }

                    ExpectDelimiter(tokens, ref index, ";");
                }
                else
                {
                    switch (token.text)
                    {
                    case "{":
                    {
                        index++;
                        block = new BlockNode(owner);
                        owner = block;
                        break;
                    }

                    case "return":
                    {
                        index++;

                        var node = new ReturnNode(owner);
                        node.expr = ParseExpression(tokens, ref index, node);

                        ExpectDelimiter(tokens, ref index, ";");

                        statement = node;
                        break;
                    }

                    case "if":
                    {
                        index++;

                        ExpectDelimiter(tokens, ref index, "(");

                        BlockNode parent = (block != null) ? block : owner as BlockNode;
                        if (parent == null)
                        {
                            throw new ParserException(token, ParserException.Kind.UnexpectedToken);
                        }

                        var node = new IfNode(parent);
                        node.expr = ParseExpression(tokens, ref index, parent);

                        ExpectDelimiter(tokens, ref index, ")");

                        node.trueBranch = ParseStatement(tokens, ref index, parent);

                        if (ExpectOptional(tokens, ref index, "else"))
                        {
                            node.falseBranch = ParseStatement(tokens, ref index, parent);
                        }

                        statement = node;
                        break;
                    }

                    case "switch":
                    {
                        index++;

                        var node = new SwitchNode(block);
                        statement = node;

                        ExpectDelimiter(tokens, ref index, "(");
                        node.expr = ParseExpression(tokens, ref index, node);
                        ExpectDelimiter(tokens, ref index, ")");

                        ExpectDelimiter(tokens, ref index, "{");

                        var keys = new HashSet <string>();
                        do
                        {
                            if (tokens[index].text == "}")
                            {
                                break;
                            }


                            if (ExpectOptional(tokens, ref index, "default"))
                            {
                                ExpectDelimiter(tokens, ref index, ":");
                                var st = ParseStatement(tokens, ref index, node);
                                node.defaultBranch = st;
                            }
                            else
                            {
                                ExpectKeyword(tokens, ref index, "case");

                                LiteralKind litKind;
                                var         val = ExpectLiteral(tokens, ref index, out litKind);

                                var key = val.ToString();
                                if (keys.Contains(key))
                                {
                                    throw new ParserException(tokens[index], ParserException.Kind.DuplicatedLabel);
                                }

                                var lit = new LiteralExpressionNode(node);
                                lit.kind  = litKind;
                                lit.value = val;

                                ExpectDelimiter(tokens, ref index, ":");
                                var st = ParseStatement(tokens, ref index, node);
                                node.cases[lit] = st;
                                keys.Add(key);
                            }
                        }while (true);

                        ExpectDelimiter(tokens, ref index, "}");
                        break;
                    }

                    default:
                    {
                        throw new ParserException(tokens[index], ParserException.Kind.UnexpectedToken);
                    }
                    }
                }

                if (block == null)
                {
                    return(statement);
                }
                else
                if (statement != null)
                {
                    block.statements.Add(statement);
                }
            } while (tokens[index].text != "}");

            index++;

            return(block);
        }
Пример #5
0
 protected abstract TypeNode GetTypeFromToken(CompilerNode owner, string token);
Пример #6
0
        protected ExpressionNode ParseExpression(List <Token> tokens, ref int index, CompilerNode owner, int precedence = -1)
        {
            if (index >= tokens.Count)
            {
                throw new ParserException(tokens.Last(), ParserException.Kind.EndOfStream);
            }

            ExpressionNode term;

            if (tokens[index].text == "(")
            {
                index++;
                term = ParseExpression(tokens, ref index, owner);
                ExpectDelimiter(tokens, ref index, ")");
            }
            else
            if (Lexer.IsLiteral(tokens[index].kind))
            {
                LiteralKind litKind;

                var node = new LiteralExpressionNode(owner);
                node.value = ExpectLiteral(tokens, ref index, out litKind);
                node.kind  = litKind;
                term       = node;
            }
            else
            if (tokens[index].kind == Token.Kind.Operator)
            {
                var node = new UnaryExpressionNode(owner);
                node.op = tokens[index].text;
                index++;

                node.term = ParseExpression(tokens, ref index, node);
                term      = node;
            }
            else
            {
                var node = new VariableExpressionNode(owner);
                node.identifier = ExpectIdentifier(tokens, ref index, false);
                term            = node;
            }

            while (tokens[index].kind == Token.Kind.Operator)
            {
                var p = GetOperatorPrecedence(tokens[index].text);

                if (precedence < 0 || p > precedence)
                {
                    var expr = new BinaryExpressionNode(owner);

                    expr.left = term;

                    expr.@operator = ParseOperator(ExpectOperator(tokens, ref index));

                    expr.right = ParseExpression(tokens, ref index, expr, p);

                    term = expr;
                }
                else
                {
                    break;
                }
            }

            return(term);
        }