Fix bug in escape analysis

This commit is contained in:
Alexey Andreev 2017-10-29 19:08:25 +03:00
parent ffbfd7df79
commit c54b2b9e9e
6 changed files with 50 additions and 10 deletions

View File

@ -63,6 +63,7 @@ import org.teavm.model.instructions.PutFieldInstruction;
import org.teavm.model.instructions.RaiseInstruction; import org.teavm.model.instructions.RaiseInstruction;
import org.teavm.model.instructions.StringConstantInstruction; import org.teavm.model.instructions.StringConstantInstruction;
import org.teavm.model.instructions.UnwrapArrayInstruction; import org.teavm.model.instructions.UnwrapArrayInstruction;
import org.teavm.model.util.DefinitionExtractor;
import org.teavm.model.util.InstructionTransitionExtractor; import org.teavm.model.util.InstructionTransitionExtractor;
import org.teavm.model.util.LivenessAnalyzer; import org.teavm.model.util.LivenessAnalyzer;
import org.teavm.model.util.UsageExtractor; import org.teavm.model.util.UsageExtractor;
@ -122,8 +123,31 @@ public class EscapeAnalysis {
GraphBuilder graphBuilder = new GraphBuilder(program.variableCount()); GraphBuilder graphBuilder = new GraphBuilder(program.variableCount());
IntDeque queue = new IntArrayDeque(); IntDeque queue = new IntArrayDeque();
for (BasicBlock block : program.getBasicBlocks()) { for (BasicBlock block : program.getBasicBlocks()) {
IntSet sharedIncomingVars = new IntOpenHashSet();
BitSet usedVars = getUsedVarsInBlock(livenessAnalyzer, block); BitSet usedVars = getUsedVarsInBlock(livenessAnalyzer, block);
// For instructions like A = B if B lives after instruction, mark both A and B as escaping
UsageExtractor useExtractor = new UsageExtractor();
DefinitionExtractor defExtractor = new DefinitionExtractor();
for (Instruction insn = block.getLastInstruction(); insn != null; insn = insn.getPrevious()) {
if (insn instanceof AssignInstruction) {
AssignInstruction assign = (AssignInstruction) insn;
if (usedVars.get(assign.getAssignee().getIndex())) {
queue.addLast(definitionClasses[assign.getAssignee().getIndex()]);
}
}
insn.acceptVisitor(useExtractor);
insn.acceptVisitor(defExtractor);
for (Variable var : useExtractor.getUsedVariables()) {
usedVars.set(var.getIndex());
}
for (Variable var : defExtractor.getDefinedVariables()) {
usedVars.clear(var.getIndex());
}
}
// If incoming variables of phi functions live after phi, mark them as escaping
IntSet sharedIncomingVars = new IntOpenHashSet();
for (Phi phi : block.getPhis()) { for (Phi phi : block.getPhis()) {
if (escapes(phi.getReceiver().getIndex())) { if (escapes(phi.getReceiver().getIndex())) {
queue.addLast(definitionClasses[phi.getReceiver().getIndex()]); queue.addLast(definitionClasses[phi.getReceiver().getIndex()]);
@ -165,13 +189,6 @@ public class EscapeAnalysis {
usedVars.or(liveness.liveIn(tryCatch.getHandler().getIndex())); usedVars.or(liveness.liveIn(tryCatch.getHandler().getIndex()));
} }
UsageExtractor useExtractor = new UsageExtractor();
for (Instruction instruction : block) {
instruction.acceptVisitor(useExtractor);
for (Variable variable : useExtractor.getUsedVariables()) {
usedVars.set(variable.getIndex());
}
}
return usedVars; return usedVars;
} }

View File

@ -67,6 +67,11 @@ public class ScalarReplacementTest {
doTest(); doTest();
} }
@Test
public void escapingLivingAssignmentSource() {
doTest();
}
@Test @Test
public void copy() { public void copy() {
doTest(); doTest();

View File

@ -11,5 +11,4 @@ $start
@b$bar_1 := @tmp2 @b$bar_1 := @tmp2
@r := @b$foo @r := @b$foo
@s := @b$bar_1 @s := @b$bar_1
@t := @a$bar
return @r return @r

View File

@ -9,5 +9,4 @@ $start
field A.bar @b := @tmp2 as I field A.bar @b := @tmp2 as I
@r := field A.foo @b as I @r := field A.foo @b as I
@s := field A.bar @b as I @s := field A.bar @b as I
@t := field A.bar @a as I
return @r return @r

View File

@ -0,0 +1,10 @@
var @this as this
$start
@first := new A
field A.foo @first := @this as `Ljava/lang/Object;`
@second := @first
@t := null
field A.foo @first := @t as `Ljava/lang/Object;`
@s := field A.foo @second as `Ljava/lang/Object;`
return @s

View File

@ -0,0 +1,10 @@
var @this as this
$start
@first := new A
field A.foo @first := @this as `Ljava/lang/Object;`
@second := @first
@t := null
field A.foo @first := @t as `Ljava/lang/Object;`
@s := field A.foo @second as `Ljava/lang/Object;`
return @s