use of org.candle.decompiler.intermediate.code.AbstractIntermediate in project candle-decompiler by bradsdavis.
the class IfLowerRangeVisitor method visitIfIntermediate.
@Override
public void visitIfIntermediate(IfIntermediate line) {
AbstractIntermediate l = igc.getTrueTarget(line);
InstructionHandle lower = l.getInstruction();
line.getBlockRange().setStart(lower);
// upper range...
AbstractIntermediate u = igc.getFalseTarget(line);
NullIntermediate nullIntermediate = new NullIntermediate(u.getInstruction().getPrev());
AbstractIntermediate ai = igc.getOrderedIntermediate().floor(nullIntermediate);
if (ai != null) {
line.getBlockRange().setEnd(ai.getInstruction());
}
}
use of org.candle.decompiler.intermediate.code.AbstractIntermediate in project candle-decompiler by bradsdavis.
the class WhileRangeVisitor method visitWhileIntermediate.
@Override
public void visitWhileIntermediate(WhileIntermediate line) {
AbstractIntermediate falseTarget = igc.getFalseTarget(line);
AbstractIntermediate trueTarget = igc.getTrueTarget(line);
line.getBlockRange().setStart(trueTarget.getInstruction());
line.getBlockRange().setEnd(falseTarget.getInstruction().getPrev());
super.visitWhileIntermediate(line);
}
use of org.candle.decompiler.intermediate.code.AbstractIntermediate in project candle-decompiler by bradsdavis.
the class WhileToForLoopIncrement method visitWhileIntermediate.
@Override
public void visitWhileIntermediate(WhileIntermediate line) {
List<AbstractIntermediate> predecessors = Graphs.predecessorListOf(igc.getGraph(), line);
// look at the predecessor lines; validate there are only 2 predecessors.
if (predecessors.size() == 2) {
StatementIntermediate declaration = null;
StatementIntermediate iteration = null;
for (AbstractIntermediate predecessor : predecessors) {
if (comparator.before(predecessor, line)) {
if (predecessor instanceof StatementIntermediate) {
declaration = (StatementIntermediate) predecessor;
continue;
} else {
// a for loop.
return;
}
}
// if the
if (comparator.after(predecessor, line)) {
if (predecessor instanceof StatementIntermediate) {
iteration = (StatementIntermediate) predecessor;
} else {
return;
}
}
}
// at this point, both should be set.
if (declaration != null && iteration != null) {
if (declaration.getExpression() instanceof Declaration) {
Declaration declarationExpression = (Declaration) declaration.getExpression();
if (iteration.getExpression() instanceof Increment) {
Increment incrementExpression = (Increment) iteration.getExpression();
if (incrementExpression.getVariable().getType().equals(declarationExpression.getVariable().getType())) {
// now check names.
if (incrementExpression.getVariable().getName().equals(declarationExpression.getVariable().getName())) {
// we can actually convert this to a for loop.
ForIntermediate forIntermediate = new ForIntermediate(line, declarationExpression, incrementExpression);
// forIntermediate.setTrueBranch(line.getTrueBranch());
// forIntermediate.setFalseBranch(line.getFalseBranch());
igc.getGraph().addVertex(forIntermediate);
igc.redirectSuccessors(line, forIntermediate);
igc.redirectPredecessors(iteration, forIntermediate);
igc.redirectPredecessors(declaration, forIntermediate);
// remove the while loop, increment, and declaration.
igc.getGraph().removeVertex(line);
igc.getGraph().removeVertex(declaration);
igc.getGraph().removeVertex(iteration);
}
}
}
}
}
}
}
use of org.candle.decompiler.intermediate.code.AbstractIntermediate in project candle-decompiler by bradsdavis.
the class ConstantArrayCompressor method visitStatementIntermediate.
@Override
public void visitStatementIntermediate(StatementIntermediate line) {
// first, look for the
Assignment assignment = extractConstantArrayAssignment(line.getExpression());
if (assignment == null) {
return;
}
// at this point, we know both the statement is an assignment, and that the left assignment is to a constant array value.
// find the one that is right before the array assignment.
Declaration declaration = extractNextDeclaration(line);
// if we didn't find the declaration, this must not be the constant array assignment proceeding the declaration.
if (declaration == null) {
return;
}
// check the right hand of the declaration...
if (!(declaration.getAssignment().getRightHandSide() instanceof NewConstantArrayInstance)) {
return;
}
NewConstantArrayInstance ncai = (NewConstantArrayInstance) declaration.getAssignment().getRightHandSide();
Expression countExpression = ncai.getCount();
AbstractIntermediate current = line;
Map<Integer, Expression> values = new HashMap<Integer, Expression>();
collectConstantAssignments(current, values);
// create a new array...
Integer count = toInteger(countExpression);
List<Expression> expressions = new ArrayList<Expression>(count);
for (int i = 0, j = count; i < j; i++) {
Expression exp = null;
if (values.containsKey(i)) {
exp = values.get(i);
} else {
exp = new Resolved(ncai.getInstructionHandle(), Type.NULL, "null");
}
expressions.add(i, exp);
}
// ok, we have the stack... now we need to just create a new expression.
// create the contant...
ConstantArray constantArray = new ConstantArray(declaration.getAssignment().getRightHandSide().getInstructionHandle(), expressions);
declaration.getAssignment().setRightHandSide(constantArray);
// excellent. we have reordered the statements into the appropriate ContantArray assignment. Now, we need to remove the dead nodes and heal the graph.
AbstractIntermediate next = Graphs.successorListOf(igc.getGraph(), line).get(0);
// loop through the dead elements...
// we know the number of dead items is equal to the number of values we found.
healGraph(line, next, values.size());
}
use of org.candle.decompiler.intermediate.code.AbstractIntermediate in project candle-decompiler by bradsdavis.
the class ConstantArrayCompressor method collectConstantAssignments.
public void collectConstantAssignments(AbstractIntermediate current, Map<Integer, Expression> assignments) {
StatementIntermediate si = (StatementIntermediate) current;
// get the assignment...
Assignment assignment = extractConstantArrayAssignment(si.getExpression());
if (assignment == null) {
return;
}
Expression right = assignment.getRightHandSide();
ArrayAccess apr = (ArrayAccess) assignment.getLeftHandSide();
assignments.put(toInteger(apr.getIndex()), right);
List<AbstractIntermediate> predecessor = Graphs.predecessorListOf(igc.getGraph(), current);
if (predecessor.size() != 1) {
return;
}
if (!(predecessor.get(0) instanceof StatementIntermediate)) {
return;
}
for (AbstractIntermediate a : predecessor) {
collectConstantAssignments(a, assignments);
}
}
Aggregations