All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.jruby.compiler.ir.dataflow.DataFlowProblem Maven / Gradle / Ivy

package org.jruby.compiler.ir.dataflow;

import java.util.ArrayList;
import java.util.BitSet;
import java.util.HashMap;
import java.util.ListIterator;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Set;

import org.jruby.compiler.ir.IRScope;
import org.jruby.compiler.ir.compiler_pass.CompilerPass;
import org.jruby.compiler.ir.representations.CFG;
import org.jruby.compiler.ir.representations.CFG.CFG_Edge;
import org.jruby.compiler.ir.representations.BasicBlock;

public abstract class DataFlowProblem
{
/* -------------- Public fields and methods below ---------------- */
    public enum DF_Direction { FORWARD, BACKWARD, BIDIRECTIONAL };

    public final DF_Direction _direction;

    public DataFlowProblem(DF_Direction d)
    {
        _direction = d;
        _dfVars = new ArrayList();
        _nextDFVarId = -1;
    }

// ------- Abstract methods without a default implementation -------
    abstract public FlowGraphNode buildFlowGraphNode(BasicBlock bb);
    abstract public String getName();

// ------- Default implementation methods below -------
    /** Are there are available data flow facts to run this problem? SSS FIXME: Silly optimization? */
    public boolean isEmpty() { return false; }

    public DF_Direction getFlowDirection() { return _direction; }

    public void setup(CFG c)
    {
        _cfg = c;
        buildFlowGraph();
    }

    public CFG getCFG() { return _cfg; }

    /* Compute Meet Over All Paths solution for this dataflow problem on the input CFG.
     * This implements a standard worklist algorithm. */
    public void compute_MOP_Solution()
    {
        /** Are there are available data flow facts to run this problem? SSS FIXME: Silly optimization? */
        if (!isEmpty()) {
            for (FlowGraphNode fg: _fgNodes)
                fg.init();

            // 2. Initialize work list based on flow direction to make processing efficient!
            LinkedList workList = getInitialWorkList();

            // 3. Initialize a bitset with a flag set for all basic blocks
            int numNodes = _cfg.getMaxNodeID();
            BitSet bbSet = new BitSet(1+numNodes);
            bbSet.flip(0, numNodes);

            // 4. Iteratively compute data flow info
            while (!workList.isEmpty()) {
                workList.removeFirst().computeDataFlowInfo(workList, bbSet);
            }
        }
    }

    private LinkedList getInitialWorkList()
    {
        LinkedList wl = new LinkedList();
        if (_direction == DF_Direction.FORWARD) {
           ListIterator it = _cfg.getReversePostOrderTraverser();
           while (it.hasPrevious())
              wl.add(getFlowGraphNode(it.previous()));
        }
        else {
           ListIterator it = _cfg.getPostOrderTraverser();
           while (it.hasNext())
              wl.add(getFlowGraphNode(it.next()));
        }

        return wl;
    }

    public int getDFVarsCount() { return _dfVars.size(); }

    public Set incomingEdgesOf(BasicBlock bb) { return _cfg.incomingEdgesOf(bb); }

    public Set outgoingEdgesOf(BasicBlock bb) { return _cfg.outgoingEdgesOf(bb); }

    /* Individual analyses should override this */
    public String getDataFlowVarsForOutput() { return ""; }

    public String toString()
    {
        StringBuffer buf = new StringBuffer();
        buf.append("----").append(getName()).append("----\n");
  
        buf.append("---- Data Flow Vars: ----\n");
        buf.append(getDataFlowVarsForOutput());
        buf.append("-------------------------\n");
  
        for (FlowGraphNode n: _fgNodes)
            buf.append("DF State for BB ").append(n._bb.getID()).append(":\n").append(n.toString());

        return buf.toString();
    }

/* -------------- Packaged/protected fields and methods below ---------------- */
    int addDataFlowVar(DataFlowVar v)
    {
        // We want unique ids for dataflow variables
        _nextDFVarId++;
        _dfVars.add(_nextDFVarId, v);
        return _nextDFVarId;
    }

/* -------------- Protected fields and methods below ---------------- */
    protected CFG                    _cfg;
    protected List    _fgNodes;

    protected FlowGraphNode getFlowGraphNode(BasicBlock b)
    {
        return _bbTofgMap.get(b.getID());
    }

/* -------------- Private fields and methods below ---------------- */
    private int     _nextDFVarId;
    private ArrayList _dfVars;
    private Map _bbTofgMap;

    private void buildFlowGraph()
    {
        _fgNodes = new LinkedList();
        _bbTofgMap = new HashMap();

        for (BasicBlock bb: _cfg.getNodes()) {
            FlowGraphNode fgNode = buildFlowGraphNode(bb);
            fgNode.buildDataFlowVars();
            _fgNodes.add(fgNode);
            _bbTofgMap.put(bb.getID(), fgNode);
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy