Commit ef85e29a authored by Skylot's avatar Skylot

core: improve 'break' and 'continue' insertion

parent 1daf5d10
...@@ -167,7 +167,7 @@ public class RegionMaker { ...@@ -167,7 +167,7 @@ public class RegionMaker {
LoopRegion loopRegion = makeLoopRegion(curRegion, loop, exitBlocks); LoopRegion loopRegion = makeLoopRegion(curRegion, loop, exitBlocks);
if (loopRegion == null) { if (loopRegion == null) {
BlockNode exit = makeEndlessLoop(curRegion, stack, loop, loopStart); BlockNode exit = makeEndlessLoop(curRegion, stack, loop, loopStart);
insertContinueInsns(loop); insertContinue(loop);
return exit; return exit;
} }
curRegion.getSubBlocks().add(loopRegion); curRegion.getSubBlocks().add(loopRegion);
...@@ -192,7 +192,7 @@ public class RegionMaker { ...@@ -192,7 +192,7 @@ public class RegionMaker {
if (!exitBlocks.contains(exitEdge.getSource())) { if (!exitBlocks.contains(exitEdge.getSource())) {
continue; continue;
} }
tryInsertBreak(stack, loopExit, exitEdge); insertBreak(stack, loopExit, exitEdge);
} }
} }
} }
...@@ -235,7 +235,7 @@ public class RegionMaker { ...@@ -235,7 +235,7 @@ public class RegionMaker {
loopRegion.setBody(body); loopRegion.setBody(body);
} }
stack.pop(); stack.pop();
insertContinueInsns(loop); insertContinue(loop);
return out; return out;
} }
...@@ -305,7 +305,7 @@ public class RegionMaker { ...@@ -305,7 +305,7 @@ public class RegionMaker {
List<Edge> exitEdges = loop.getExitEdges(); List<Edge> exitEdges = loop.getExitEdges();
for (Edge exitEdge : exitEdges) { for (Edge exitEdge : exitEdges) {
BlockNode exit = exitEdge.getTarget(); BlockNode exit = exitEdge.getTarget();
if (tryInsertBreak(stack, exit, exitEdge)) { if (insertBreak(stack, exit, exitEdge)) {
BlockNode nextBlock = getNextBlock(exit); BlockNode nextBlock = getNextBlock(exit);
if (nextBlock != null) { if (nextBlock != null) {
stack.addExit(nextBlock); stack.addExit(nextBlock);
...@@ -351,7 +351,7 @@ public class RegionMaker { ...@@ -351,7 +351,7 @@ public class RegionMaker {
return true; return true;
} }
private boolean tryInsertBreak(RegionStack stack, BlockNode loopExit, Edge exitEdge) { private boolean insertBreak(RegionStack stack, BlockNode loopExit, Edge exitEdge) {
BlockNode exit = exitEdge.getTarget(); BlockNode exit = exitEdge.getTarget();
BlockNode insertBlock = null; BlockNode insertBlock = null;
boolean confirm = false; boolean confirm = false;
...@@ -393,52 +393,85 @@ public class RegionMaker { ...@@ -393,52 +393,85 @@ public class RegionMaker {
insertBlock.getInstructions().add(breakInsn); insertBlock.getInstructions().add(breakInsn);
stack.addExit(exit); stack.addExit(exit);
// add label to 'break' if needed // add label to 'break' if needed
List<LoopInfo> loops = mth.getAllLoopsForBlock(exitEdge.getSource()); addBreakLabel(exitEdge, exit, breakInsn);
if (loops.size() >= 2) { return true;
// find parent loop }
for (LoopInfo loop : loops) {
LoopInfo parentLoop = loop.getParentLoop(); private void addBreakLabel(Edge exitEdge, BlockNode exit, InsnNode breakInsn) {
if (parentLoop == null BlockNode outBlock = BlockUtils.getNextBlock(exitEdge.getTarget());
&& loop.getEnd() != exit if (outBlock == null) {
&& !loop.getExitNodes().contains(exit)) { return;
LoopLabelAttr labelAttr = new LoopLabelAttr(loop); }
breakInsn.addAttr(labelAttr); List<LoopInfo> exitLoop = mth.getAllLoopsForBlock(outBlock);
loop.getStart().addAttr(labelAttr); if (!exitLoop.isEmpty()) {
return;
}
List<LoopInfo> inLoops = mth.getAllLoopsForBlock(exitEdge.getSource());
if (inLoops.size() < 2) {
return;
}
// search for parent loop
LoopInfo parentLoop = null;
for (LoopInfo loop : inLoops) {
if (loop.getParentLoop() == null) {
parentLoop = loop;
break; break;
} }
} }
if (parentLoop == null) {
return;
}
if (parentLoop.getEnd() != exit && !parentLoop.getExitNodes().contains(exit)) {
LoopLabelAttr labelAttr = new LoopLabelAttr(parentLoop);
breakInsn.addAttr(labelAttr);
parentLoop.getStart().addAttr(labelAttr);
} }
return true;
} }
private static void insertContinueInsns(LoopInfo loop) { private static void insertContinue(LoopInfo loop) {
BlockNode loopEnd = loop.getEnd(); BlockNode loopEnd = loop.getEnd();
List<BlockNode> predecessors = loopEnd.getPredecessors(); List<BlockNode> predecessors = loopEnd.getPredecessors();
if (predecessors.size() <= 1) { if (predecessors.size() <= 1) {
return; return;
} }
Set<BlockNode> loopExitNodes = loop.getExitNodes();
for (BlockNode pred : predecessors) { for (BlockNode pred : predecessors) {
if (canInsertContinue(pred, predecessors, loopEnd, loopExitNodes)) {
InsnNode cont = new InsnNode(InsnType.CONTINUE, 0);
pred.getInstructions().add(cont);
}
}
}
private static boolean canInsertContinue(BlockNode pred, List<BlockNode> predecessors, BlockNode loopEnd,
Set<BlockNode> loopExitNodes) {
if (!pred.contains(AFlag.SYNTHETIC) if (!pred.contains(AFlag.SYNTHETIC)
|| BlockUtils.checkLastInsnType(pred, InsnType.CONTINUE)) { || BlockUtils.checkLastInsnType(pred, InsnType.CONTINUE)) {
continue; return false;
} }
List<BlockNode> nodes = pred.getPredecessors(); List<BlockNode> preds = pred.getPredecessors();
if (nodes.isEmpty()) { if (preds.isEmpty()) {
continue; return false;
} }
BlockNode codePred = nodes.get(0); BlockNode codePred = preds.get(0);
if (codePred.contains(AFlag.SKIP)) { if (codePred.contains(AFlag.SKIP)) {
continue; return false;
} }
if (!isDominatedOnBlocks(codePred, predecessors)) { if (loopEnd.isDominator(codePred)
for (BlockNode blockNode : predecessors) { || loopExitNodes.contains(codePred)) {
if (blockNode != pred && BlockUtils.isPathExists(codePred, blockNode)) { return false;
InsnNode cont = new InsnNode(InsnType.CONTINUE, 0);
pred.getInstructions().add(cont);
} }
if (isDominatedOnBlocks(codePred, predecessors)) {
return false;
} }
boolean gotoExit = false;
for (BlockNode exit : loopExitNodes) {
if (BlockUtils.isPathExists(codePred, exit)) {
gotoExit = true;
break;
} }
} }
return gotoExit;
} }
private static boolean isDominatedOnBlocks(BlockNode dom, List<BlockNode> blocks) { private static boolean isDominatedOnBlocks(BlockNode dom, List<BlockNode> blocks) {
...@@ -601,7 +634,7 @@ public class RegionMaker { ...@@ -601,7 +634,7 @@ public class RegionMaker {
} }
Map<BlockNode, List<Object>> blocksMap = new LinkedHashMap<BlockNode, List<Object>>(len); Map<BlockNode, List<Object>> blocksMap = new LinkedHashMap<BlockNode, List<Object>>(len);
for (Entry<Integer, List<Object>> entry : casesMap.entrySet()) { for (Map.Entry<Integer, List<Object>> entry : casesMap.entrySet()) {
BlockNode c = getBlockByOffset(entry.getKey(), block.getSuccessors()); BlockNode c = getBlockByOffset(entry.getKey(), block.getSuccessors());
assert c != null; assert c != null;
blocksMap.put(c, entry.getValue()); blocksMap.put(c, entry.getValue());
......
...@@ -310,29 +310,6 @@ public class BlockUtils { ...@@ -310,29 +310,6 @@ public class BlockUtils {
return traverseSuccessorsUntil(start, end, new BitSet()); return traverseSuccessorsUntil(start, end, new BitSet());
} }
public static boolean isCleanPathExists(BlockNode start, BlockNode end) {
if (start == end || start.getCleanSuccessors().contains(end)) {
return true;
}
return traverseCleanSuccessorsUntil(start, end, new BitSet());
}
private static boolean traverseCleanSuccessorsUntil(BlockNode from, BlockNode until, BitSet visited) {
for (BlockNode s : from.getCleanSuccessors()) {
if (s == until) {
return true;
}
int id = s.getId();
if (!visited.get(id) && !s.contains(AType.EXC_HANDLER)) {
visited.set(id);
if (traverseSuccessorsUntil(s, until, visited)) {
return true;
}
}
}
return false;
}
public static boolean isOnlyOnePathExists(BlockNode start, BlockNode end) { public static boolean isOnlyOnePathExists(BlockNode start, BlockNode end) {
if (start == end) { if (start == end) {
return true; return true;
......
package jadx.tests.integration.loops;
import jadx.core.dex.attributes.AType;
import jadx.core.dex.instructions.InsnType;
import jadx.core.dex.nodes.BlockNode;
import jadx.core.dex.nodes.ClassNode;
import jadx.core.dex.nodes.InsnNode;
import jadx.core.dex.nodes.MethodNode;
import jadx.core.dex.trycatch.CatchAttr;
import jadx.core.dex.trycatch.ExcHandlerAttr;
import jadx.core.dex.trycatch.ExceptionHandler;
import jadx.core.dex.trycatch.TryCatchBlock;
import jadx.core.utils.BlockUtils;
import jadx.core.utils.InstructionRemover;
import jadx.tests.api.IntegrationTest;
import org.junit.Test;
import static jadx.tests.api.utils.JadxMatchers.containsOne;
import static org.hamcrest.CoreMatchers.containsString;
import static org.hamcrest.CoreMatchers.not;
import static org.junit.Assert.assertThat;
public class TestContinueInLoop2 extends IntegrationTest {
public static class TestCls {
private static void test(MethodNode mth, BlockNode block) {
ExcHandlerAttr handlerAttr = block.get(AType.EXC_HANDLER);
if (handlerAttr != null) {
ExceptionHandler excHandler = handlerAttr.getHandler();
excHandler.addBlock(block);
for (BlockNode node : BlockUtils.collectBlocksDominatedBy(block, block)) {
excHandler.addBlock(node);
}
for (BlockNode excBlock : excHandler.getBlocks()) {
InstructionRemover remover = new InstructionRemover(mth, excBlock);
for (InsnNode insn : excBlock.getInstructions()) {
if (insn.getType() == InsnType.MONITOR_ENTER) {
break;
}
if (insn.getType() == InsnType.MONITOR_EXIT) {
remover.add(insn);
}
}
remover.perform();
for (InsnNode insn : excBlock.getInstructions()) {
if (insn.getType() == InsnType.THROW) {
CatchAttr catchAttr = insn.get(AType.CATCH_BLOCK);
if (catchAttr != null) {
TryCatchBlock handlerBlock = handlerAttr.getTryBlock();
TryCatchBlock catchBlock = catchAttr.getTryBlock();
if (handlerBlock != catchBlock) {
handlerBlock.merge(mth, catchBlock);
catchBlock.removeInsn(insn);
}
}
}
}
}
}
}
}
@Test
public void test() {
ClassNode cls = getClassNode(TestCls.class);
String code = cls.getCode().toString();
assertThat(code, containsOne("break;"));
assertThat(code, not(containsString("continue;")));
}
}
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment