From b018e6161533278974b44edfcf8e8257e03f32fd Mon Sep 17 00:00:00 2001 From: Alexey Andreev Date: Sun, 25 Nov 2018 18:46:45 +0300 Subject: [PATCH] Fix converting method to state machine in some cases. Fix #366 --- .../model/util/AsyncProgramSplitter.java | 38 ++++++++++++++++++- 1 file changed, 36 insertions(+), 2 deletions(-) diff --git a/core/src/main/java/org/teavm/model/util/AsyncProgramSplitter.java b/core/src/main/java/org/teavm/model/util/AsyncProgramSplitter.java index 0b3f00402..2d00ad362 100644 --- a/core/src/main/java/org/teavm/model/util/AsyncProgramSplitter.java +++ b/core/src/main/java/org/teavm/model/util/AsyncProgramSplitter.java @@ -40,6 +40,7 @@ import org.teavm.model.TryCatchBlock; import org.teavm.model.ValueType; import org.teavm.model.Variable; import org.teavm.model.instructions.InitClassInstruction; +import org.teavm.model.instructions.InvocationType; import org.teavm.model.instructions.InvokeInstruction; import org.teavm.model.instructions.JumpInstruction; import org.teavm.model.instructions.MonitorEnterInstruction; @@ -82,8 +83,14 @@ public class AsyncProgramSplitter { for (Instruction insn : sourceBlock) { if (insn instanceof InvokeInstruction) { InvokeInstruction invoke = (InvokeInstruction) insn; - if (!asyncMethods.contains(findRealMethod(invoke.getMethod()))) { - continue; + if (invoke.getType() == InvocationType.VIRTUAL) { + if (!isAsyncMethod(findRealMethod(invoke.getMethod()))) { + continue; + } + } else { + if (!asyncMethods.contains(findRealMethod(invoke.getMethod()))) { + continue; + } } } else if (insn instanceof InitClassInstruction) { if (!isSplittingClassInitializer(((InitClassInstruction) insn).getClassName())) { @@ -227,6 +234,33 @@ public class AsyncProgramSplitter { return method; } + private boolean isAsyncMethod(MethodReference method) { + if (asyncMethods.isEmpty()) { + return false; + } + if (asyncMethods.contains(method)) { + return true; + } + + ClassReader cls = classSource.get(method.getClassName()); + if (cls == null) { + return false; + } + + if (cls.getParent() != null) { + if (isAsyncMethod(new MethodReference(cls.getParent(), method.getDescriptor()))) { + return true; + } + } + for (String itf : cls.getInterfaces()) { + if (isAsyncMethod(new MethodReference(itf, method.getDescriptor()))) { + return true; + } + } + + return false; + } + private Program createStubCopy(Program program) { Program copy = new Program(); for (int i = 0; i < program.basicBlockCount(); ++i) {