Mercurial > hg > truffle
view graal/com.oracle.graal.phases/src/com/oracle/graal/phases/graph/ReentrantBlockIterator.java @ 7530:5e3d1a68664e
applied mx eclipseformat to all Java files
author | Doug Simon <doug.simon@oracle.com> |
---|---|
date | Wed, 23 Jan 2013 16:34:57 +0100 |
parents | ca3e5df0e6cf |
children | 741884454253 |
line wrap: on
line source
/* * Copyright (c) 2011, 2012, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA * or visit www.oracle.com if you need additional information or have any * questions. */ package com.oracle.graal.phases.graph; import java.util.*; import com.oracle.graal.nodes.*; import com.oracle.graal.nodes.cfg.*; public final class ReentrantBlockIterator { public abstract static class MergeableBlockState<T> { public abstract T cloneState(); } public static class LoopInfo<T extends MergeableBlockState<T>> { public final List<T> endStates = new ArrayList<>(); public final List<T> exitStates = new ArrayList<>(); } public abstract static class BlockIteratorClosure<T extends MergeableBlockState<T>> { protected abstract void processBlock(Block block, T currentState); protected abstract T merge(MergeNode merge, List<T> states); protected abstract T afterSplit(FixedNode node, T oldState); protected abstract List<T> processLoop(Loop loop, T initialState); } private ReentrantBlockIterator() { // no instances allowed } public static <T extends MergeableBlockState<T>> LoopInfo<T> processLoop(BlockIteratorClosure<T> closure, Loop loop, T initialState) { IdentityHashMap<FixedNode, T> blockEndStates = apply(closure, loop.header, initialState, new HashSet<>(loop.blocks)); LoopInfo<T> info = new LoopInfo<>(); List<Block> predecessors = loop.header.getPredecessors(); for (int i = 1; i < predecessors.size(); i++) { info.endStates.add(blockEndStates.get(predecessors.get(i).getEndNode())); } for (Block loopExit : loop.exits) { assert loopExit.getPredecessorCount() == 1; T exitState = blockEndStates.get(loopExit.getFirstPredecessor().getEndNode()); assert exitState != null; info.exitStates.add(exitState); } return info; } public static <T extends MergeableBlockState<T>> IdentityHashMap<FixedNode, T> apply(BlockIteratorClosure<T> closure, Block start, T initialState, Set<Block> boundary) { Deque<Block> blockQueue = new ArrayDeque<>(); IdentityHashMap<FixedNode, T> blockEndStates = new IdentityHashMap<>(); T state = initialState; Block current = start; do { if (boundary == null || boundary.contains(current)) { closure.processBlock(current, state); if (current.getSuccessors().isEmpty()) { // nothing to do... } else if (current.getSuccessors().size() == 1) { Block successor = current.getSuccessors().get(0); if (successor.isLoopHeader()) { if (current.isLoopEnd()) { // nothing to do... loop ends only lead to loop begins we've already // visited blockEndStates.put(current.getEndNode(), state); } else { // recurse into the loop Loop loop = successor.getLoop(); LoopBeginNode loopBegin = loop.loopBegin(); assert successor.getBeginNode() == loopBegin; List<T> exitStates = closure.processLoop(loop, state); int i = 0; assert loop.exits.size() == exitStates.size(); for (Block exit : loop.exits) { blockEndStates.put(exit.getFirstPredecessor().getEndNode(), exitStates.get(i++)); blockQueue.addFirst(exit); } } } else { if (successor.getBeginNode() instanceof LoopExitNode) { assert successor.getPredecessors().size() == 1; blockEndStates.put(current.getEndNode(), state); current = successor; continue; } else { if (current.getEndNode() instanceof EndNode) { assert successor.getPredecessors().size() > 1 : "invalid block schedule at " + successor.getBeginNode(); EndNode end = (EndNode) current.getEndNode(); // add the end node and see if the merge is ready for processing assert !blockEndStates.containsKey(end); blockEndStates.put(end, state); MergeNode merge = end.merge(); boolean endsVisited = true; for (int i = 0; i < merge.forwardEndCount(); i++) { if (!blockEndStates.containsKey(merge.forwardEndAt(i))) { endsVisited = false; break; } } if (endsVisited) { blockQueue.addFirst(successor); } } else { assert successor.getPredecessors().size() == 1 : "invalid block schedule at " + successor.getBeginNode(); current = successor; continue; } } } } else { assert current.getSuccessors().size() > 1; blockEndStates.put(current.getEndNode(), state); for (Block block : current.getSuccessors()) { blockQueue.addFirst(block); } } } // get next queued block if (blockQueue.isEmpty()) { current = null; } else { int maxIterations = blockQueue.size(); while (maxIterations-- > 0) { current = blockQueue.removeFirst(); if (current.getPredecessors().size() > 1) { MergeNode merge = (MergeNode) current.getBeginNode(); ArrayList<T> states = new ArrayList<>(merge.forwardEndCount()); for (int i = 0; i < merge.forwardEndCount(); i++) { T other = blockEndStates.get(merge.forwardEndAt(i)); assert other != null; states.add(other); } state = closure.merge(merge, states); if (state != null) { break; } else { blockQueue.addLast(current); current = null; } } else { assert current.getPredecessors().size() == 1; assert current.getBeginNode().predecessor() != null; if (boundary == null || boundary.contains(current)) { state = closure.afterSplit(current.getBeginNode(), blockEndStates.get(current.getBeginNode().predecessor())); break; } } } } } while (current != null); return blockEndStates; } }