/*
* This file is part of the Jikes RVM project (http://jikesrvm.org).
*
* This file is licensed to You under the Eclipse Public License (EPL);
* You may not use this file except in compliance with the License. You
* may obtain a copy of the License at
*
* http://www.opensource.org/licenses/eclipse-1.0.php
*
* See the COPYRIGHT.txt file distributed with this work for information
* regarding copyright ownership.
*/
package org.jikesrvm.compilers.opt.depgraph;
import static org.jikesrvm.compilers.opt.ir.Operators.NULL_CHECK;
import org.jikesrvm.compilers.opt.OptimizingCompilerException;
import org.jikesrvm.compilers.opt.ir.BasicBlock;
import org.jikesrvm.compilers.opt.ir.IR;
import org.jikesrvm.compilers.opt.ir.Instruction;
import org.jikesrvm.compilers.opt.liveness.LiveAnalysis;
/**
* This module provides dependence graph statistics. It will only
* be used for for experimental measurements, so compile-time overhead
* is less of a concern.
*
* @see DepGraph
*/
public class DepGraphStats {
/**
* Create a statistical summary of a dependence graph for a given basic
* block.
*
* @param dg the dependence graph
* @param bbName name of the basic block
*/
DepGraphStats(DepGraph dg, String bbName) {
// First pass -- compute numNodes
int _numNodes = 0;
boolean containsLoadOrStore = false;
for (DepGraphNode n = (DepGraphNode) dg.firstNode(); n != null; n = (DepGraphNode) n.getNext()) {
_numNodes++;
Instruction instr = n.instruction();
if (instr.isImplicitStore() || instr.isImplicitLoad()) {
containsLoadOrStore = true;
}
}
DepGraphNode[] nodes = new DepGraphNode[_numNodes];
int[] ECT = new int[_numNodes]; // Earliest Completion Times
int _totalTime = 0;
int _critPathLength = 0;
// Second pass -- compute times
int i = 0;
for (DepGraphNode n = (DepGraphNode) dg.firstNode(); n != null; n = (DepGraphNode) n.getNext()) {
nodes[i] = n;
ECT[i] = 0;
for (DepGraphEdge e = (DepGraphEdge) n.firstInEdge(); e != null; e = (DepGraphEdge) e.getNextIn()) {
DepGraphNode pred = (DepGraphNode) e.fromNode();
// Look for pred in nodes[]
int j;
for (j = 0; j < i; j++) {
if (nodes[j] == pred) {
break;
}
}
if (j == i) {
// Not found
throw new OptimizingCompilerException("DepGraphStats: dep graph is not topologically sorted ???");
// NOTE: I could not use SortedGraphIterator
// for top sort because DepGraphNode
// is not a subclass of SortedGraphNode
}
// TODO: add edge latency also??
ECT[i] = Math.max(ECT[i], ECT[j]);
} // for ( e = ... )
Instruction instr = n.instruction();
int curTime = estimateExecutionTime(instr);
_totalTime += curTime;
ECT[i] += curTime;
_critPathLength = Math.max(_critPathLength, ECT[i]);
i++;
} // for ( n = ... )
System.out.println("@@@@ BB " +
bbName +
"; totalTime = " +
_totalTime +
"; containsLoadOrStore = " +
containsLoadOrStore +
"; critPathLength = " +
_critPathLength);
}
/**
* Print the dependence graph stats for all basic blocks in an IR.
* @param ir the IR
*/
public static void printBasicBlockStatistics(IR ir) {
final boolean DEBUG = false;
System.out.println();
System.out.println("**** START OF printBasicBlockStatistics() for method " + ir.method + " ****");
if (DEBUG) {
ir.printInstructions();
}
// Performing live analysis may reduce dependences between PEIs and stores
if (ir.options.L2M_HANDLER_LIVENESS) {
new LiveAnalysis(false, false, true).perform(ir);
}
for (BasicBlock bb = ir.firstBasicBlockInCodeOrder(); bb != null; bb = bb.nextBasicBlockInCodeOrder()) {
//DepGraph dg =
new DepGraph(ir, bb.firstRealInstruction(), bb.lastRealInstruction(), bb);
}
System.out.println("**** END OF printBasicBlockStatistics() ****");
}
/**
* Returns an estimate of the number of cycles for a given instruction.
* Currently, this estimate is comically simple (see below).
*
* @param instr the instruction
* @return {@code 0} or {@code 1}
*/
int estimateExecutionTime(Instruction instr) {
if (instr.operator() == NULL_CHECK) {
return 0;
} else {
return 1;
}
}
}