Linker.java :  » Code-Analyzer » pmd-4.2rc1 » net » sourceforge » pmd » dfa » Java Open Source

Java Open Source » Code Analyzer » pmd 4.2rc1 
pmd 4.2rc1 » net » sourceforge » pmd » dfa » Linker.java
/*
 * Created on 12.07.2004
 */
package net.sourceforge.pmd.dfa;

import java.util.List;

import net.sourceforge.pmd.ast.ASTLabeledStatement;
import net.sourceforge.pmd.ast.SimpleNode;

/**
 * @author raik
 *         Links data flow nodes to each other.
 */
public class Linker {

    private List braceStack;
    private List continueBreakReturnStack;

    public Linker(List braceStack, List continueBreakReturnStack) {
        this.braceStack = braceStack;
        this.continueBreakReturnStack = continueBreakReturnStack;
    }

    /**
     * Creates all the links between the data flow nodes.
     */
    public void computePaths() throws LinkerException, SequenceException {
        // Returns true if there are more sequences, computes the first and
        // the last index of the sequence.
        SequenceChecker sc = new SequenceChecker(braceStack);
        while (!sc.run()) {
            if (sc.getFirstIndex() < 0 || sc.getLastIndex() < 0) {
                throw new SequenceException("computePaths(): return index <  0");
            }

            StackObject firstStackObject = (StackObject) braceStack.get(sc.getFirstIndex());

            switch (firstStackObject.getType()) {
                case NodeType.IF_EXPR:
                    int x = sc.getLastIndex() - sc.getFirstIndex();
                    if (x == 2) {
                        this.computeIf(sc.getFirstIndex(), sc.getFirstIndex() + 1, sc.getLastIndex());
                    } else if (x == 1) {
                        this.computeIf(sc.getFirstIndex(), sc.getLastIndex());
                    } else {
                        System.out.println("Error - computePaths 1");
                    }
                    break;

                case NodeType.WHILE_EXPR:
                    this.computeWhile(sc.getFirstIndex(), sc.getLastIndex());
                    break;

                case NodeType.SWITCH_START:
                    this.computeSwitch(sc.getFirstIndex(), sc.getLastIndex());
                    break;

                case NodeType.FOR_INIT:
                case NodeType.FOR_EXPR:
                case NodeType.FOR_UPDATE:
                case NodeType.FOR_BEFORE_FIRST_STATEMENT:
                    this.computeFor(sc.getFirstIndex(), sc.getLastIndex());
                    break;

                case NodeType.DO_BEFORE_FIRST_STATEMENT:
                    this.computeDo(sc.getFirstIndex(), sc.getLastIndex());
                    break;
                                      
                default:
            }

            for (int y = sc.getLastIndex(); y >= sc.getFirstIndex(); y--) {
                braceStack.remove(y);
            }
        }

        while (!continueBreakReturnStack.isEmpty()) {
            StackObject stackObject = (StackObject) continueBreakReturnStack.get(0);
            IDataFlowNode node = stackObject.getDataFlowNode();

            switch (stackObject.getType()) {
              case NodeType.THROW_STATEMENT:
                // do the same like a return
                case NodeType.RETURN_STATEMENT:
                    // remove all children (should contain only one child)
                    node.removePathToChild(node.getChildren().get(0));
                    IDataFlowNode lastNode = node.getFlow().get(node.getFlow().size() - 1);
                    node.addPathToChild(lastNode);
                    continueBreakReturnStack.remove(0);
                    break;                
                case NodeType.BREAK_STATEMENT:
                    IDataFlowNode last = getNodeToBreakStatement(node);
                    node.removePathToChild(node.getChildren().get(0));
                    node.addPathToChild(last);                    
                    continueBreakReturnStack.remove(0);
                    break;

                case NodeType.CONTINUE_STATEMENT:
                    //List cList = node.getFlow();
                    /* traverse up the tree and find the first loop start node
                     */
/*
                               for(int i = cList.indexOf(node)-1;i>=0;i--) {
                                   IDataFlowNode n = (IDataFlowNode)cList.get(i);

                                   if(n.isType(NodeType.FOR_UPDATE) ||
                                               n.isType(NodeType.FOR_EXPR) ||
                                               n.isType(NodeType.WHILE_EXPR)) {
*/
                    /*
                     * while(..) {
                     *              while(...) {
                     *                      ...
                     *              }
                     *              continue;
                     * }
                     *
                     * Without this Expression he continues the second
                     * WHILE loop. The continue statement and the while loop
                     * have to be in different scopes.
                     *
                     * TODO An error occurs if "continue" is even nested in
                     * scopes other than local loop scopes, like "if".
                     * The result is, that the data flow isn't build right
                     * and the pathfinder runs in invinity loop.
                     * */
/*                                     if(n.getSimpleNode().getScope().equals(node.getSimpleNode().getScope())) {
                                               System.err.println("equals");
                                               continue;
                                       }
                                       else {
                                               System.err.println("don't equals");
                                       }

                                               //remove all children (should contain only one child)
                                       node.removePathToChild((IDataFlowNode)node.getChildren().get(0));

                                       node.addPathToChild(n);
                                       cbrStack.remove(0);
                                       break;

                                   }else if(n.isType(NodeType.DO_BEFOR_FIRST_STATEMENT)) {

                                       IDataFlowNode inode = (IDataFlowNode)n.getFlow().get(n.getIndex()1);

                                       for(int j=0;j<inode.getParents().size();j) {
                                               IDataFlowNode parent = (IDataFlowNode)inode.getParents().get(j);

                                               if(parent.isType(NodeType.DO_EXPR)) {
                                                       node.removePathToChild((IDataFlowNode)node.getChildren().get(0));
                                               node.addPathToChild(parent);

                                               cbrStack.remove(0);
                                                       break;
                                               }
                                       }
                                       break;
                                   }
                               }
*/continueBreakReturnStack.remove(0); // delete this statement if you uncomment the stuff above
            }
        }
    }

    private IDataFlowNode getNodeToBreakStatement(IDataFlowNode node) {
        // What about breaks to labels above if statements?
        List bList = node.getFlow();
        int findEnds = 1; // ignore ends of other for's while's etc.
       
        
        // find out index of the node where the path goes to after the break
        int index = bList.indexOf(node); 
        for (; index < bList.size()-2; index++) {
            IDataFlowNode n = (IDataFlowNode) bList.get(index);
            if (n.isType(NodeType.DO_EXPR) || 
                    n.isType(NodeType.FOR_INIT) ||
                    n.isType(NodeType.WHILE_EXPR) ||
                    n.isType(NodeType.SWITCH_START)) {
                findEnds++;
            }
            if (n.isType(NodeType.WHILE_LAST_STATEMENT) ||
                    n.isType(NodeType.SWITCH_END) ||
                    n.isType(NodeType.FOR_END) ||
                    n.isType(NodeType.DO_EXPR)) {                            
                if (findEnds > 1) {
                    // thats not the right node
                    findEnds--;
                } else {
                    break;
                }
            }
            
            if (n.isType(NodeType.LABEL_LAST_STATEMENT)) {
                SimpleNode parentNode = n.getSimpleNode().getFirstParentOfType(ASTLabeledStatement.class);
                if (parentNode == null) {
                    break;                                
                } else {
                    String label = node.getSimpleNode().getImage();
                    if (label == null || label.equals(parentNode.getImage())) {
                        node.removePathToChild(node.getChildren().get(0));
                        IDataFlowNode last = (IDataFlowNode) bList.get(index + 1);
                        node.addPathToChild(last);
                        break;
                    }
                }
            }
        }
        return node.getFlow().get(index+1);
    }

    private void computeDo(int first, int last) {
        IDataFlowNode doSt = ((StackObject) this.braceStack.get(first)).getDataFlowNode();
        IDataFlowNode doExpr = ((StackObject) this.braceStack.get(last)).getDataFlowNode();
        IDataFlowNode doFirst = doSt.getFlow().get(doSt.getIndex() + 1);
        if (doFirst.getIndex() != doExpr.getIndex()) {
            doExpr.addPathToChild(doFirst);
        }
    }

    private void computeFor(int firstIndex, int lastIndex) {
        IDataFlowNode fExpr = null;
        IDataFlowNode fUpdate = null;
        IDataFlowNode fSt = null;
        IDataFlowNode fEnd = null;
        boolean isUpdate = false;

        for (int i = firstIndex; i <= lastIndex; i++) {
            StackObject so = (StackObject) this.braceStack.get(i);
            IDataFlowNode node = so.getDataFlowNode();

            if (so.getType() == NodeType.FOR_EXPR) {
                fExpr = node;
            } else if (so.getType() == NodeType.FOR_UPDATE) {
                fUpdate = node;
                isUpdate = true;
            } else if (so.getType() == NodeType.FOR_BEFORE_FIRST_STATEMENT) {
                fSt = node;
            } else if (so.getType() == NodeType.FOR_END) {
                fEnd = node;
            }
        }
        IDataFlowNode end = fEnd.getFlow().get(fEnd.getIndex() + 1);

        IDataFlowNode firstSt = fSt.getChildren().get(0);

        if (isUpdate) {
            if (fSt.getIndex() != fEnd.getIndex()) {
                end.reverseParentPathsTo(fUpdate);
                fExpr.removePathToChild(fUpdate);
                fUpdate.addPathToChild(fExpr);
                fUpdate.removePathToChild(firstSt);
                fExpr.addPathToChild(firstSt);
                fExpr.addPathToChild(end);
            } else {
                fSt.removePathToChild(end);
                fExpr.removePathToChild(fUpdate);
                fUpdate.addPathToChild(fExpr);
                fExpr.addPathToChild(fUpdate);
                fExpr.addPathToChild(end);
            }
        } else {
            if (fSt.getIndex() != fEnd.getIndex()) {
                end.reverseParentPathsTo(fExpr);
                fExpr.addPathToChild(end);
            }
        }
    }

    private void computeSwitch(int firstIndex, int lastIndex) {

        int diff = lastIndex - firstIndex;
        boolean defaultStatement = false;

        IDataFlowNode sStart = ((StackObject) this.braceStack.get(firstIndex)).getDataFlowNode();
        IDataFlowNode sEnd = ((StackObject) this.braceStack.get(lastIndex)).getDataFlowNode();
        IDataFlowNode end = sEnd.getChildren().get(0);

        for (int i = 0; i < diff - 2; i++) {
            StackObject so = (StackObject) this.braceStack.get(firstIndex + 2 + i);
            IDataFlowNode node = so.getDataFlowNode();

            sStart.addPathToChild(node.getChildren().get(0));

            if (so.getType() == NodeType.SWITCH_LAST_DEFAULT_STATEMENT)
                defaultStatement = true;
        }

        if (!defaultStatement)
            sStart.addPathToChild(end);
    }

    private void computeWhile(int first, int last) {
        IDataFlowNode wStart = ((StackObject) this.braceStack.get(first)).getDataFlowNode();
        IDataFlowNode wEnd = ((StackObject) this.braceStack.get(last)).getDataFlowNode();

        IDataFlowNode end = wEnd.getFlow().get(wEnd.getIndex() + 1);

        if (wStart.getIndex() != wEnd.getIndex()) {
            end.reverseParentPathsTo(wStart);
            wStart.addPathToChild(end);
        }
    }

    private void computeIf(int first, int second, int last) {
        IDataFlowNode ifStart = ((StackObject) this.braceStack.get(first)).getDataFlowNode();
        IDataFlowNode ifEnd = ((StackObject) this.braceStack.get(second)).getDataFlowNode();
        IDataFlowNode elseEnd = ((StackObject) this.braceStack.get(last)).getDataFlowNode();

        IDataFlowNode elseStart = ifEnd.getFlow().get(ifEnd.getIndex() + 1);
        IDataFlowNode end = elseEnd.getFlow().get(elseEnd.getIndex() + 1);

        // if if-statement and else-statement contains statements or expressions
        if (ifStart.getIndex() != ifEnd.getIndex() &&
                ifEnd.getIndex() != elseEnd.getIndex()) {
            elseStart.reverseParentPathsTo(end);
            ifStart.addPathToChild(elseStart);
        }
        // if only if-statement contains no expressions
        else if (ifStart.getIndex() == ifEnd.getIndex() &&
                ifEnd.getIndex() != elseEnd.getIndex()) {
            ifStart.addPathToChild(end);
        }
        // if only else-statement contains no expressions
        else if (ifEnd.getIndex() == elseEnd.getIndex() &&
                ifStart.getIndex() != ifEnd.getIndex()) {
            ifStart.addPathToChild(end);
        }
    }

    private void computeIf(int first, int last) {
        IDataFlowNode ifStart = ((StackObject) this.braceStack.get(first)).getDataFlowNode();
        IDataFlowNode ifEnd = ((StackObject) this.braceStack.get(last)).getDataFlowNode();

        // only if the if-statement contains another Statement or Expression
        if (ifStart.getIndex() != ifEnd.getIndex()) {
            IDataFlowNode end = ifEnd.getFlow().get(ifEnd.getIndex() + 1);
            ifStart.addPathToChild(end);
        }
    }
}
java2s.com  | Contact Us | Privacy Policy
Copyright 2009 - 12 Demo Source and Support. All rights reserved.
All other trademarks are property of their respective owners.