Search in sources :

Example 11 with LocalVarNode

use of kalang.ast.LocalVarNode in project kalang by kasonyang.

the class AstBuilder method visitForEachStat.

@Override
public Object visitForEachStat(KalangParser.ForEachStatContext ctx) {
    BlockStmt block = newBlock();
    ExprNode expr = this.visitExpression(ctx.expression());
    Type exprType = expr.getType();
    List<TerminalNode> idsCtx = ctx.Identifier();
    VarExpr indexVarExpr = null;
    TerminalNode varId;
    if (idsCtx.size() == 1) {
        varId = idsCtx.get(0);
    } else {
        TerminalNode indexId = idsCtx.get(0);
        LocalVarNode indexVar = this.declareLocalVar(indexId.getText(), Types.INT_TYPE, Modifier.FINAL, ctx);
        if (indexVar == null)
            return null;
        block.statements.add(new VarDeclStmt(indexVar));
        indexVarExpr = new VarExpr(indexVar);
        varId = idsCtx.get(1);
    }
    LoopStmt loopStmt;
    if (exprType instanceof ArrayType) {
        LocalVarNode localVarNode = this.declareLocalVar(varId.getText(), ((ArrayType) exprType).getComponentType(), Modifier.FINAL, ctx);
        if (localVarNode == null)
            return null;
        VarExpr localVariable = new VarExpr(localVarNode);
        block.statements.add(new VarDeclStmt(localVarNode));
        LocalVarNode lenVar = this.declareTempLocalVar(Types.INT_TYPE);
        LocalVarNode counterVar = this.declareTempLocalVar(Types.INT_TYPE);
        // var len
        block.statements.add(new VarDeclStmt(lenVar));
        // var i
        block.statements.add(new VarDeclStmt(counterVar));
        VarExpr counterVarExpr = new VarExpr(counterVar);
        VarExpr lenVarExpr = new VarExpr(lenVar);
        block.statements.add(new ExprStmt(new AssignExpr(lenVarExpr, new ArrayLengthExpr(expr))));
        // l = array.length
        block.statements.add(new ExprStmt(new AssignExpr(counterVarExpr, new ConstExpr(0))));
        // i=0
        ExprNode cnd = new CompareExpr(counterVarExpr, lenVarExpr, CompareExpr.OP_LT);
        BlockStmt loopBody = this.newBlock();
        loopBody.statements.add(new ExprStmt(new AssignExpr(localVariable, new ElementExpr(expr, counterVarExpr))));
        if (indexVarExpr != null) {
            loopBody.statements.add(new ExprStmt(new AssignExpr(indexVarExpr, counterVarExpr)));
        }
        loopBody.statements.add(visitStat(ctx.stat()));
        popBlock();
        BlockStmt updateBs = newBlock();
        // increment counter
        updateBs.statements.add(new ExprStmt(new AssignExpr(counterVarExpr, new MathExpr(counterVarExpr, new ConstExpr(1), MathExpr.OP_ADD))));
        popBlock();
        loopStmt = new LoopStmt(cnd, null, loopBody, updateBs);
    } else {
        ObjectType iterType = Types.getIterableClassType();
        if (iterType.isAssignableFrom(exprType)) {
            ObjectInvokeExpr getIterableExpr;
            try {
                getIterableExpr = ObjectInvokeExpr.create(expr, "iterator", null);
            } catch (MethodNotFoundException | AmbiguousMethodException ex) {
                throw Exceptions.unexceptedException(ex);
            }
            LocalVarNode iterableVarNode = this.declareTempLocalVar(getIterableExpr.getType());
            block.statements.add(new VarDeclStmt(iterableVarNode));
            VarExpr iterableVarExpr = new VarExpr(iterableVarNode);
            block.statements.add(new ExprStmt(new AssignExpr(iterableVarExpr, getIterableExpr)));
            // set index = 0
            if (indexVarExpr != null) {
                block.statements.add(new ExprStmt(new AssignExpr(indexVarExpr, new ConstExpr(0))));
            }
            ObjectInvokeExpr cnd;
            try {
                cnd = ObjectInvokeExpr.create(iterableVarExpr, "hasNext", null);
            } catch (MethodNotFoundException | AmbiguousMethodException ex) {
                throw Exceptions.unexceptedException(ex);
            }
            BlockStmt loopBody = this.newBlock();
            ObjectInvokeExpr nextInvokeExpr;
            try {
                nextInvokeExpr = ObjectInvokeExpr.create(iterableVarExpr, "next", null);
            } catch (MethodNotFoundException | AmbiguousMethodException ex) {
                throw Exceptions.unexceptedException(ex);
            }
            LocalVarNode localVarNode = this.declareLocalVar(varId.getText(), nextInvokeExpr.getType(), Modifier.FINAL, ctx);
            if (localVarNode == null)
                return null;
            VarExpr localVariable = new VarExpr(localVarNode);
            loopBody.statements.add(new VarDeclStmt(localVarNode));
            loopBody.statements.add(new ExprStmt(new AssignExpr(localVariable, new CastExpr(localVariable.getType(), nextInvokeExpr))));
            loopBody.statements.add(visitStat(ctx.stat()));
            popBlock();
            BlockStmt updateBs = newBlock();
            if (indexVarExpr != null) {
                // do index++
                updateBs.statements.add(new ExprStmt(new AssignExpr(indexVarExpr, new MathExpr(indexVarExpr, new ConstExpr(1), BinaryExpr.OP_ADD))));
            }
            popBlock();
            loopStmt = new LoopStmt(cnd, null, loopBody, updateBs);
        } else {
            this.handleSyntaxError("require array type or iterable type", ctx.expression());
            loopStmt = null;
        }
    }
    popBlock();
    if (loopStmt != null)
        block.statements.add(loopStmt);
    return block;
}
Also used : ConstExpr(kalang.ast.ConstExpr) LoopStmt(kalang.ast.LoopStmt) BlockStmt(kalang.ast.BlockStmt) ArrayLengthExpr(kalang.ast.ArrayLengthExpr) AssignExpr(kalang.ast.AssignExpr) ExprNode(kalang.ast.ExprNode) ArrayType(kalang.core.ArrayType) ObjectType(kalang.core.ObjectType) WildcardType(kalang.core.WildcardType) ArrayType(kalang.core.ArrayType) ClassType(kalang.core.ClassType) PrimitiveType(kalang.core.PrimitiveType) Type(kalang.core.Type) GenericType(kalang.core.GenericType) ObjectType(kalang.core.ObjectType) ExprStmt(kalang.ast.ExprStmt) CompareExpr(kalang.ast.CompareExpr) VarDeclStmt(kalang.ast.VarDeclStmt) ObjectInvokeExpr(kalang.ast.ObjectInvokeExpr) PrimitiveCastExpr(kalang.ast.PrimitiveCastExpr) CastExpr(kalang.ast.CastExpr) VarExpr(kalang.ast.VarExpr) TerminalNode(org.antlr.v4.runtime.tree.TerminalNode) LocalVarNode(kalang.ast.LocalVarNode) MethodNotFoundException(kalang.MethodNotFoundException) MathExpr(kalang.ast.MathExpr) ElementExpr(kalang.ast.ElementExpr) AmbiguousMethodException(kalang.AmbiguousMethodException)

Example 12 with LocalVarNode

use of kalang.ast.LocalVarNode in project kalang by kasonyang.

the class AstBuilder method visitTryStat.

@Override
public AstNode visitTryStat(TryStatContext ctx) {
    // TODO handle multi-branched assign
    BlockStmt tryExecStmt = requireBlock(ctx.exec);
    boolean tryReturned = this.returned;
    List<CatchBlock> tryCatchBlocks = new LinkedList<>();
    if (ctx.catchTypes != null) {
        for (int i = 0; i < ctx.catchTypes.size(); i++) {
            this.newFrame();
            this.returned = false;
            String vName = ctx.catchVarNames.get(i).getText();
            String vType = ctx.catchTypes.get(i).getText();
            LocalVarNode vo = this.declareLocalVar(vName, requireClassType(vType, ctx.catchTypes.get(i).start), Modifier.FINAL, ctx);
            if (vo == null)
                return null;
            BlockStmt catchExecStmt = requireBlock(ctx.catchExec.get(i));
            CatchBlock catchStmt = new CatchBlock(vo, catchExecStmt);
            tryCatchBlocks.add(catchStmt);
            this.returned = this.returned && tryReturned;
            this.popFrame();
        }
    }
    BlockStmt tryFinallyStmt = null;
    if (ctx.finallyExec != null) {
        tryFinallyStmt = requireBlock(ctx.finallyExec);
    }
    TryStmt tryStmt = new TryStmt(tryExecStmt, tryCatchBlocks, tryFinallyStmt);
    mapAst(tryStmt, ctx);
    return tryStmt;
}
Also used : CatchBlock(kalang.ast.CatchBlock) BlockStmt(kalang.ast.BlockStmt) TryStmt(kalang.ast.TryStmt) LocalVarNode(kalang.ast.LocalVarNode) LinkedList(java.util.LinkedList)

Example 13 with LocalVarNode

use of kalang.ast.LocalVarNode in project kalang by kasonyang.

the class InitializationAnalyzer method addIntersectedAssignedVar.

protected void addIntersectedAssignedVar(VarTable<LocalVarNode, Void>... assignedVarsList) {
    Set<LocalVarNode>[] assigned = new Set[assignedVarsList.length];
    for (int i = 0; i < assigned.length; i++) {
        assigned[i] = assignedVarsList[i].keySet();
    }
    Set<LocalVarNode> sets = CollectionsUtil.getIntersection(assigned);
    for (LocalVarNode s : sets) {
        assignedVars.put(s, null);
    }
}
Also used : Set(java.util.Set) LocalVarNode(kalang.ast.LocalVarNode)

Aggregations

LocalVarNode (kalang.ast.LocalVarNode)13 VarExpr (kalang.ast.VarExpr)8 AssignExpr (kalang.ast.AssignExpr)6 ExprNode (kalang.ast.ExprNode)6 ExprStmt (kalang.ast.ExprStmt)6 VarDeclStmt (kalang.ast.VarDeclStmt)6 ArrayType (kalang.core.ArrayType)6 ObjectType (kalang.core.ObjectType)6 PrimitiveType (kalang.core.PrimitiveType)6 Type (kalang.core.Type)6 ClassType (kalang.core.ClassType)5 GenericType (kalang.core.GenericType)5 WildcardType (kalang.core.WildcardType)5 LinkedList (java.util.LinkedList)4 ConstExpr (kalang.ast.ConstExpr)4 MultiStmtExpr (kalang.ast.MultiStmtExpr)4 Statement (kalang.ast.Statement)4 BlockStmt (kalang.ast.BlockStmt)3 ElementExpr (kalang.ast.ElementExpr)3 Nullable (javax.annotation.Nullable)2