mirror of
https://github.com/Eaglercraft-TeaVM-Fork/eagler-teavm.git
synced 2025-01-08 16:04:10 -08:00
Further work on a new thread emulator
This commit is contained in:
parent
93e43456d1
commit
32deaf2716
|
@ -80,6 +80,7 @@ public class TThread extends TObject implements TRunnable {
|
||||||
currentThread.timeSliceStart = System.currentTimeMillis();
|
currentThread.timeSliceStart = System.currentTimeMillis();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static TThread getMainThread(){
|
static TThread getMainThread(){
|
||||||
return mainThread;
|
return mainThread;
|
||||||
}
|
}
|
||||||
|
|
|
@ -305,10 +305,31 @@ public class AstIO {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void visit(RestoreAsyncStatement statement) {
|
public void visit(GotoPartStatement statement) {
|
||||||
try {
|
try {
|
||||||
output.writeByte(17);
|
output.writeByte(17);
|
||||||
output.writeShort(statement.getReceiver() != null ? statement.getReceiver() : -1);
|
output.writeShort(statement.getPart());
|
||||||
|
} catch (IOException e) {
|
||||||
|
throw new IOExceptionWrapper(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void visit(MonitorEnterStatement statement) {
|
||||||
|
try {
|
||||||
|
output.writeByte(18);
|
||||||
|
output.writeShort(statement.getAsyncTarget() != null ? 0 : statement.getAsyncTarget());
|
||||||
|
writeExpr(statement.getObjectRef());
|
||||||
|
} catch (IOException e) {
|
||||||
|
throw new IOExceptionWrapper(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void visit(MonitorExitStatement statement) {
|
||||||
|
try {
|
||||||
|
output.writeByte(19);
|
||||||
|
writeExpr(statement.getObjectRef());
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
throw new IOExceptionWrapper(e);
|
throw new IOExceptionWrapper(e);
|
||||||
}
|
}
|
||||||
|
@ -506,16 +527,6 @@ public class AstIO {
|
||||||
throw new IOExceptionWrapper(e);
|
throw new IOExceptionWrapper(e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public void visit(MonitorEnterStatement statement) {
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void visit(MonitorExitStatement statement) {
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private NodeLocation readLocation(DataInput input) throws IOException {
|
private NodeLocation readLocation(DataInput input) throws IOException {
|
||||||
|
@ -670,11 +681,11 @@ public class AstIO {
|
||||||
return stmt;
|
return stmt;
|
||||||
}
|
}
|
||||||
case 17: {
|
case 17: {
|
||||||
short var = input.readShort();
|
GotoPartStatement stmt = new GotoPartStatement();
|
||||||
RestoreAsyncStatement stmt = new RestoreAsyncStatement();
|
stmt.setPart(input.readShort());
|
||||||
stmt.setReceiver(var >= 0 ? (int)var : null);
|
|
||||||
return stmt;
|
return stmt;
|
||||||
}
|
}
|
||||||
|
// TODO: MonitorEnter/MonitorExit
|
||||||
default:
|
default:
|
||||||
throw new RuntimeException("Unexpected statement type: " + type);
|
throw new RuntimeException("Unexpected statement type: " + type);
|
||||||
}
|
}
|
||||||
|
|
|
@ -259,7 +259,7 @@ public class DiskRegularMethodNodeCache implements RegularMethodNodeCache {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void visit(RestoreAsyncStatement statement) {
|
public void visit(GotoPartStatement statement) {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -269,7 +269,6 @@ public class DiskRegularMethodNodeCache implements RegularMethodNodeCache {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void visit(MonitorExitStatement statement) {
|
public void visit(MonitorExitStatement statement) {
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -114,16 +114,14 @@ class BreakToContinueReplacer implements StatementVisitor {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void visit(RestoreAsyncStatement statement) {
|
public void visit(GotoPartStatement statement) {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void visit(MonitorEnterStatement statement) {
|
public void visit(MonitorEnterStatement statement) {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void visit(MonitorExitStatement statement) {
|
public void visit(MonitorExitStatement statement) {
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -109,16 +109,14 @@ class CertainBlockCountVisitor implements StatementVisitor {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void visit(RestoreAsyncStatement statement) {
|
public void visit(GotoPartStatement statement) {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void visit(MonitorEnterStatement statement) {
|
public void visit(MonitorEnterStatement statement) {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void visit(MonitorExitStatement statement) {
|
public void visit(MonitorExitStatement statement) {
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -203,15 +203,7 @@ public class Decompiler {
|
||||||
AsyncProgramSplitter splitter = new AsyncProgramSplitter(classSource, splitMethods);
|
AsyncProgramSplitter splitter = new AsyncProgramSplitter(classSource, splitMethods);
|
||||||
splitter.split(method.getProgram());
|
splitter.split(method.getProgram());
|
||||||
for (int i = 0; i < splitter.size(); ++i) {
|
for (int i = 0; i < splitter.size(); ++i) {
|
||||||
Integer input = null;
|
AsyncMethodPart part = getRegularMethodStatement(splitter.getProgram(i), splitter.getBlockSuccessors(i));
|
||||||
if (i > 0) {
|
|
||||||
input = splitter.getInput(i);
|
|
||||||
if (input == null) {
|
|
||||||
input = -1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
AsyncMethodPart part = getRegularMethodStatement(splitter.getProgram(i), splitter.getBlockSuccessors(i),
|
|
||||||
input);
|
|
||||||
node.getBody().add(part);
|
node.getBody().add(part);
|
||||||
}
|
}
|
||||||
Program program = method.getProgram();
|
Program program = method.getProgram();
|
||||||
|
@ -219,7 +211,7 @@ public class Decompiler {
|
||||||
node.getVariables().add(program.variableAt(i).getRegister());
|
node.getVariables().add(program.variableAt(i).getRegister());
|
||||||
}
|
}
|
||||||
Optimizer optimizer = new Optimizer();
|
Optimizer optimizer = new Optimizer();
|
||||||
optimizer.optimize(node, program, splitter);
|
optimizer.optimize(node, program);
|
||||||
node.getModifiers().addAll(mapModifiers(method.getModifiers()));
|
node.getModifiers().addAll(mapModifiers(method.getModifiers()));
|
||||||
int paramCount = Math.min(method.getSignature().length, program.variableCount());
|
int paramCount = Math.min(method.getSignature().length, program.variableCount());
|
||||||
for (int i = 0; i < paramCount; ++i) {
|
for (int i = 0; i < paramCount; ++i) {
|
||||||
|
@ -234,7 +226,7 @@ public class Decompiler {
|
||||||
Program program = method.getProgram();
|
Program program = method.getProgram();
|
||||||
int[] targetBlocks = new int[program.basicBlockCount()];
|
int[] targetBlocks = new int[program.basicBlockCount()];
|
||||||
Arrays.fill(targetBlocks, -1);
|
Arrays.fill(targetBlocks, -1);
|
||||||
methodNode.setBody(getRegularMethodStatement(program, targetBlocks, null).getStatement());
|
methodNode.setBody(getRegularMethodStatement(program, targetBlocks).getStatement());
|
||||||
for (int i = 0; i < program.variableCount(); ++i) {
|
for (int i = 0; i < program.variableCount(); ++i) {
|
||||||
methodNode.getVariables().add(program.variableAt(i).getRegister());
|
methodNode.getVariables().add(program.variableAt(i).getRegister());
|
||||||
}
|
}
|
||||||
|
@ -249,7 +241,7 @@ public class Decompiler {
|
||||||
return methodNode;
|
return methodNode;
|
||||||
}
|
}
|
||||||
|
|
||||||
private AsyncMethodPart getRegularMethodStatement(Program program, int[] targetBlocks, Integer inputVar) {
|
private AsyncMethodPart getRegularMethodStatement(Program program, int[] targetBlocks) {
|
||||||
AsyncMethodPart result = new AsyncMethodPart();
|
AsyncMethodPart result = new AsyncMethodPart();
|
||||||
lastBlockId = 1;
|
lastBlockId = 1;
|
||||||
graph = ProgramUtils.buildControlFlowGraph(program);
|
graph = ProgramUtils.buildControlFlowGraph(program);
|
||||||
|
@ -302,11 +294,6 @@ public class Decompiler {
|
||||||
int tmp = indexer.nodeAt(next);
|
int tmp = indexer.nodeAt(next);
|
||||||
generator.nextBlock = tmp >= 0 && next < indexer.size() ? program.basicBlockAt(tmp) : null;
|
generator.nextBlock = tmp >= 0 && next < indexer.size() ? program.basicBlockAt(tmp) : null;
|
||||||
generator.statements.clear();
|
generator.statements.clear();
|
||||||
if (node == 0 && inputVar != null) {
|
|
||||||
RestoreAsyncStatement restoreStmt = new RestoreAsyncStatement();
|
|
||||||
restoreStmt.setReceiver(inputVar >= 0 ? inputVar : null);
|
|
||||||
generator.statements.add(restoreStmt);
|
|
||||||
}
|
|
||||||
generator.asyncTarget = null;
|
generator.asyncTarget = null;
|
||||||
InstructionLocation lastLocation = null;
|
InstructionLocation lastLocation = null;
|
||||||
NodeLocation nodeLocation = null;
|
NodeLocation nodeLocation = null;
|
||||||
|
|
|
@ -188,7 +188,7 @@ public class NameFrequencyEstimator implements StatementVisitor, ExprVisitor, Me
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void visit(RestoreAsyncStatement statement) {
|
public void visit(GotoPartStatement statement) {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -19,7 +19,6 @@ import org.teavm.javascript.ast.AsyncMethodNode;
|
||||||
import org.teavm.javascript.ast.AsyncMethodPart;
|
import org.teavm.javascript.ast.AsyncMethodPart;
|
||||||
import org.teavm.javascript.ast.RegularMethodNode;
|
import org.teavm.javascript.ast.RegularMethodNode;
|
||||||
import org.teavm.model.Program;
|
import org.teavm.model.Program;
|
||||||
import org.teavm.model.util.AsyncProgramSplitter;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
|
@ -43,15 +42,9 @@ public class Optimizer {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void optimize(AsyncMethodNode method, Program program, AsyncProgramSplitter splitter) {
|
public void optimize(AsyncMethodNode method, Program program) {
|
||||||
ReadWriteStatsBuilder stats = new ReadWriteStatsBuilder(method.getVariables().size());
|
ReadWriteStatsBuilder stats = new ReadWriteStatsBuilder(method.getVariables().size());
|
||||||
stats.analyze(program);
|
stats.analyze(program);
|
||||||
for (int i = 0; i < splitter.size(); ++i) {
|
|
||||||
Integer var = splitter.getInput(i);
|
|
||||||
if (var != null) {
|
|
||||||
stats.reads[var]++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
for (AsyncMethodPart part : method.getBody()) {
|
for (AsyncMethodPart part : method.getBody()) {
|
||||||
OptimizingVisitor optimizer = new OptimizingVisitor(stats.copy());
|
OptimizingVisitor optimizer = new OptimizingVisitor(stats.copy());
|
||||||
part.getStatement().acceptVisitor(optimizer);
|
part.getStatement().acceptVisitor(optimizer);
|
||||||
|
|
|
@ -180,7 +180,7 @@ class OptimizingVisitor implements StatementVisitor, ExprVisitor {
|
||||||
}
|
}
|
||||||
|
|
||||||
private boolean tryApplyConstructor(InvocationExpr expr) {
|
private boolean tryApplyConstructor(InvocationExpr expr) {
|
||||||
if (expr.getAsyncTarget() != null || !expr.getMethod().getName().equals("<init>")) {
|
if (!expr.getMethod().getName().equals("<init>")) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if (resultSequence == null || resultSequence.isEmpty()) {
|
if (resultSequence == null || resultSequence.isEmpty()) {
|
||||||
|
@ -615,17 +615,21 @@ class OptimizingVisitor implements StatementVisitor, ExprVisitor {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void visit(RestoreAsyncStatement statement) {
|
public void visit(GotoPartStatement statement) {
|
||||||
resultStmt = statement;
|
resultStmt = statement;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void visit(MonitorEnterStatement statement) {
|
public void visit(MonitorEnterStatement statement) {
|
||||||
|
statement.getObjectRef().acceptVisitor(this);
|
||||||
|
statement.setObjectRef(resultExpr);
|
||||||
resultStmt = statement;
|
resultStmt = statement;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void visit(MonitorExitStatement statement) {
|
public void visit(MonitorExitStatement statement) {
|
||||||
|
statement.getObjectRef().acceptVisitor(this);
|
||||||
|
statement.setObjectRef(resultExpr);
|
||||||
resultStmt = statement;
|
resultStmt = statement;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -119,16 +119,14 @@ class RedundantLabelEliminator implements StatementVisitor {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void visit(RestoreAsyncStatement statement) {
|
public void visit(GotoPartStatement statement) {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void visit(MonitorEnterStatement statement) {
|
public void visit(MonitorEnterStatement statement) {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void visit(MonitorExitStatement statement) {
|
public void visit(MonitorExitStatement statement) {
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -113,16 +113,14 @@ class ReferenceCountingVisitor implements StatementVisitor {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void visit(RestoreAsyncStatement statement) {
|
public void visit(GotoPartStatement statement) {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void visit(MonitorEnterStatement statement) {
|
public void visit(MonitorEnterStatement statement) {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void visit(MonitorExitStatement statement) {
|
public void visit(MonitorExitStatement statement) {
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -260,8 +260,8 @@ public class Renderer implements ExprVisitor, StatementVisitor, RenderingContext
|
||||||
}
|
}
|
||||||
|
|
||||||
private void renderRuntimeAliases() throws IOException {
|
private void renderRuntimeAliases() throws IOException {
|
||||||
String[] names = { "$rt_asyncResult", "$rt_asyncError", "$rt_continue", "$rt_guardAsync", "$rt_throw",
|
String[] names = { "$rt_throw", "$rt_compare", "$rt_nullCheck", "$rt_cls", "$rt_createArray",
|
||||||
"$rt_compare", "$rt_nullCheck", "$rt_cls", "$rt_createArray", "$rt_isInstance" };
|
"$rt_isInstance" };
|
||||||
boolean first = true;
|
boolean first = true;
|
||||||
for (String name : names) {
|
for (String name : names) {
|
||||||
if (!first) {
|
if (!first) {
|
||||||
|
@ -698,20 +698,20 @@ public class Renderer implements ExprVisitor, StatementVisitor, RenderingContext
|
||||||
}
|
}
|
||||||
|
|
||||||
writer.append("function $save()").ws().append("{").indent().softNewLine();
|
writer.append("function $save()").ws().append("{").indent().softNewLine();
|
||||||
writer.append("var $").ws().append('=').ws().append("$rt_stack();").softNewLine();
|
writer.append("$rt_nativeThread()");
|
||||||
for (int i = ref.parameterCount() + 1; i < variableCount; ++i) {
|
for (int i = ref.parameterCount() + 1; i < variableCount; ++i) {
|
||||||
writer.append("$.push(").append(variableName(i)).append(");");
|
writer.append(".push(").append(variableName(i)).append(")");
|
||||||
}
|
}
|
||||||
writer.append("$.push($ptr)");
|
writer.append(".push($ptr);");
|
||||||
writer.softNewLine();
|
writer.softNewLine();
|
||||||
writer.outdent().append("}").softNewLine();
|
writer.outdent().append("}").softNewLine();
|
||||||
|
|
||||||
writer.append("$ptr").ws().append('=').ws().append("0;").softNewLine();
|
writer.append("$ptr").ws().append('=').ws().append("0;").softNewLine();
|
||||||
writer.append("if").ws().append("($rt_isRestoring())").ws().append("{").indent().softNewLine();
|
writer.append("if").ws().append("($rt_resuming())").ws().append("{").indent().softNewLine();
|
||||||
writer.append("var $s").ws().append('=').ws().append("$rt_stack();").softNewLine();
|
writer.append("var $T").ws().append('=').ws().append("$rt_nativeThread();").softNewLine();
|
||||||
writer.append("var $ptr").ws().append('=').append("$s.pop();");
|
writer.append("$ptr").ws().append('=').ws().append("$T.pop();");
|
||||||
for (int i = variableCount - 1; i > ref.parameterCount(); --i) {
|
for (int i = variableCount - 1; i > ref.parameterCount(); --i) {
|
||||||
writer.append(variableName(i)).ws().append('=').ws().append("$.pop();");
|
writer.append(variableName(i)).ws().append('=').ws().append("T.pop();");
|
||||||
}
|
}
|
||||||
writer.softNewLine();
|
writer.softNewLine();
|
||||||
writer.outdent().append("}").softNewLine();
|
writer.outdent().append("}").softNewLine();
|
||||||
|
@ -1681,15 +1681,6 @@ public class Renderer implements ExprVisitor, StatementVisitor, RenderingContext
|
||||||
}
|
}
|
||||||
writer.append(')');
|
writer.append(')');
|
||||||
exitPriority();
|
exitPriority();
|
||||||
if (expr.getAsyncTarget() != null) {
|
|
||||||
writer.append(';').softNewLine();
|
|
||||||
writer.append("$ptr").ws().append("=").ws().append(expr.getAsyncTarget()).append(";")
|
|
||||||
.softNewLine();
|
|
||||||
writer.append("if").ws().append("($rt_suspending())").ws().append("{").indent();
|
|
||||||
writer.append("return $save();").softNewLine();
|
|
||||||
writer.outdent().append("}").softNewLine();
|
|
||||||
writer.append("break $main;").softNewLine();
|
|
||||||
}
|
|
||||||
if (lastCallSite != null) {
|
if (lastCallSite != null) {
|
||||||
if (virtual) {
|
if (virtual) {
|
||||||
lastCallSite.setVirtualMethod(expr.getMethod());
|
lastCallSite.setVirtualMethod(expr.getMethod());
|
||||||
|
@ -1965,18 +1956,25 @@ public class Renderer implements ExprVisitor, StatementVisitor, RenderingContext
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void visit(RestoreAsyncStatement statement) {
|
public void visit(GotoPartStatement statement) {
|
||||||
try {
|
try {
|
||||||
if (statement.getReceiver() != null) {
|
gotoPart(statement.getPart());
|
||||||
writer.append(variableName(statement.getReceiver())).ws().append('=').ws();
|
} catch (IOException ex){
|
||||||
}
|
throw new RenderingException("IO error occured", ex);
|
||||||
writer.append("$restore();").softNewLine();
|
|
||||||
} catch (IOException e) {
|
|
||||||
throw new RenderingException("IO error occured", e);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void gotoPart(int part) throws IOException {
|
||||||
|
writer.append("$ptr").ws().append("=").ws().append(part).append(";")
|
||||||
|
.softNewLine();
|
||||||
|
writer.append("if").ws().append("($rt_suspending())").ws().append("{").indent().softNewLine();
|
||||||
|
writer.append("return $save();").softNewLine();
|
||||||
|
writer.outdent().append("}").softNewLine();
|
||||||
|
writer.append("break $main;").softNewLine();
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void visit(MonitorEnterStatement statement) {
|
public void visit(MonitorEnterStatement statement) {
|
||||||
try {
|
try {
|
||||||
|
@ -1986,6 +1984,9 @@ public class Renderer implements ExprVisitor, StatementVisitor, RenderingContext
|
||||||
writer.appendMethodBody(monitorEnterRef).append("(");
|
writer.appendMethodBody(monitorEnterRef).append("(");
|
||||||
statement.getObjectRef().acceptVisitor(this);
|
statement.getObjectRef().acceptVisitor(this);
|
||||||
writer.append(");").softNewLine();
|
writer.append(");").softNewLine();
|
||||||
|
if (statement.getAsyncTarget() != null) {
|
||||||
|
gotoPart(statement.getAsyncTarget());
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
MethodReference monitorEnterRef = new MethodReference(
|
MethodReference monitorEnterRef = new MethodReference(
|
||||||
Object.class, "monitorEnterSync", Object.class, void.class);
|
Object.class, "monitorEnterSync", Object.class, void.class);
|
||||||
|
|
|
@ -550,20 +550,18 @@ class StatementGenerator implements InstructionVisitor {
|
||||||
} else {
|
} else {
|
||||||
invocationExpr = Expr.invokeStatic(insn.getMethod(), exprArgs);
|
invocationExpr = Expr.invokeStatic(insn.getMethod(), exprArgs);
|
||||||
}
|
}
|
||||||
invocationExpr.setAsyncTarget(asyncTarget);
|
if (insn.getReceiver() != null) {
|
||||||
if (asyncTarget == null) {
|
assign(invocationExpr, insn.getReceiver());
|
||||||
if (insn.getReceiver() != null) {
|
|
||||||
assign(invocationExpr, insn.getReceiver());
|
|
||||||
} else {
|
|
||||||
AssignmentStatement stmt = Statement.assign(null, invocationExpr);
|
|
||||||
stmt.setLocation(currentLocation);
|
|
||||||
statements.add(stmt);
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
AssignmentStatement stmt = Statement.assign(null, invocationExpr);
|
AssignmentStatement stmt = Statement.assign(null, invocationExpr);
|
||||||
stmt.setLocation(currentLocation);
|
stmt.setLocation(currentLocation);
|
||||||
statements.add(stmt);
|
statements.add(stmt);
|
||||||
}
|
}
|
||||||
|
if (asyncTarget != null) {
|
||||||
|
GotoPartStatement gotoStmt = new GotoPartStatement();
|
||||||
|
gotoStmt.setPart(asyncTarget);
|
||||||
|
statements.add(gotoStmt);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -112,16 +112,14 @@ class TryCatchFinder implements StatementVisitor {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void visit(RestoreAsyncStatement statement) {
|
public void visit(GotoPartStatement statement) {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void visit(MonitorEnterStatement statement) {
|
public void visit(MonitorEnterStatement statement) {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void visit(MonitorExitStatement statement) {
|
public void visit(MonitorExitStatement statement) {
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -226,10 +226,7 @@ class UnusedVariableEliminator implements ExprVisitor, StatementVisitor {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void visit(RestoreAsyncStatement statement) {
|
public void visit(GotoPartStatement statement) {
|
||||||
if (statement.getReceiver() != null) {
|
|
||||||
statement.setReceiver(renumber(statement.getReceiver()));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -19,15 +19,15 @@ package org.teavm.javascript.ast;
|
||||||
*
|
*
|
||||||
* @author Alexey Andreev
|
* @author Alexey Andreev
|
||||||
*/
|
*/
|
||||||
public class RestoreAsyncStatement extends Statement {
|
public class GotoPartStatement extends Statement {
|
||||||
private Integer receiver;
|
private int part;
|
||||||
|
|
||||||
public Integer getReceiver() {
|
public int getPart() {
|
||||||
return receiver;
|
return part;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void setReceiver(Integer receiver) {
|
public void setPart(int part) {
|
||||||
this.receiver = receiver;
|
this.part = part;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
|
@ -28,7 +28,6 @@ public class InvocationExpr extends Expr {
|
||||||
private MethodReference method;
|
private MethodReference method;
|
||||||
private InvocationType type;
|
private InvocationType type;
|
||||||
private List<Expr> arguments = new ArrayList<>();
|
private List<Expr> arguments = new ArrayList<>();
|
||||||
private Integer asyncTarget;
|
|
||||||
|
|
||||||
public MethodReference getMethod() {
|
public MethodReference getMethod() {
|
||||||
return method;
|
return method;
|
||||||
|
@ -50,14 +49,6 @@ public class InvocationExpr extends Expr {
|
||||||
return arguments;
|
return arguments;
|
||||||
}
|
}
|
||||||
|
|
||||||
public Integer getAsyncTarget() {
|
|
||||||
return asyncTarget;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setAsyncTarget(Integer asyncTarget) {
|
|
||||||
this.asyncTarget = asyncTarget;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void acceptVisitor(ExprVisitor visitor) {
|
public void acceptVisitor(ExprVisitor visitor) {
|
||||||
visitor.visit(this);
|
visitor.visit(this);
|
||||||
|
@ -72,7 +63,6 @@ public class InvocationExpr extends Expr {
|
||||||
InvocationExpr copy = new InvocationExpr();
|
InvocationExpr copy = new InvocationExpr();
|
||||||
cache.put(this, copy);
|
cache.put(this, copy);
|
||||||
copy.setMethod(method);
|
copy.setMethod(method);
|
||||||
copy.setAsyncTarget(asyncTarget);
|
|
||||||
for (Expr arg : arguments) {
|
for (Expr arg : arguments) {
|
||||||
copy.getArguments().add(arg.clone(cache));
|
copy.getArguments().add(arg.clone(cache));
|
||||||
}
|
}
|
||||||
|
|
|
@ -193,10 +193,7 @@ public class RenamingVisitor implements StatementVisitor, ExprVisitor {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void visit(RestoreAsyncStatement statement) {
|
public void visit(GotoPartStatement statement) {
|
||||||
if (statement.getReceiver() != null) {
|
|
||||||
statement.setReceiver(varNames[statement.getReceiver()]);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -44,7 +44,7 @@ public interface StatementVisitor {
|
||||||
|
|
||||||
void visit(TryCatchStatement statement);
|
void visit(TryCatchStatement statement);
|
||||||
|
|
||||||
void visit(RestoreAsyncStatement statement);
|
void visit(GotoPartStatement statement);
|
||||||
|
|
||||||
void visit(MonitorEnterStatement statement);
|
void visit(MonitorEnterStatement statement);
|
||||||
|
|
||||||
|
|
|
@ -62,16 +62,12 @@ public class AsyncProgramSplitter {
|
||||||
int last = 0;
|
int last = 0;
|
||||||
for (int i = 0; i < sourceBlock.getInstructions().size(); ++i) {
|
for (int i = 0; i < sourceBlock.getInstructions().size(); ++i) {
|
||||||
Instruction insn = sourceBlock.getInstructions().get(i);
|
Instruction insn = sourceBlock.getInstructions().get(i);
|
||||||
Integer receiver;
|
|
||||||
if (insn instanceof InvokeInstruction) {
|
if (insn instanceof InvokeInstruction) {
|
||||||
InvokeInstruction invoke = (InvokeInstruction)insn;
|
InvokeInstruction invoke = (InvokeInstruction)insn;
|
||||||
if (!asyncMethods.contains(findRealMethod(invoke.getMethod()))) {
|
if (!asyncMethods.contains(findRealMethod(invoke.getMethod()))) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
receiver = invoke.getReceiver() != null ? invoke.getReceiver().getIndex() : null;
|
} else if (!(insn instanceof MonitorEnterInstruction)) {
|
||||||
} else if (insn instanceof MonitorEnterInstruction) {
|
|
||||||
receiver = null;
|
|
||||||
} else {
|
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -102,7 +98,6 @@ public class AsyncProgramSplitter {
|
||||||
// Create a new part
|
// Create a new part
|
||||||
Program nextProgram = createStubCopy(program);
|
Program nextProgram = createStubCopy(program);
|
||||||
Part part = new Part();
|
Part part = new Part();
|
||||||
part.input = receiver;
|
|
||||||
part.program = nextProgram;
|
part.program = nextProgram;
|
||||||
int partId = parts.size();
|
int partId = parts.size();
|
||||||
parts.add(part);
|
parts.add(part);
|
||||||
|
@ -201,10 +196,6 @@ public class AsyncProgramSplitter {
|
||||||
return parts.get(index).program;
|
return parts.get(index).program;
|
||||||
}
|
}
|
||||||
|
|
||||||
public Integer getInput(int index) {
|
|
||||||
return parts.get(index).input;
|
|
||||||
}
|
|
||||||
|
|
||||||
public int[] getBlockSuccessors(int index) {
|
public int[] getBlockSuccessors(int index) {
|
||||||
int[] result = parts.get(index).blockSuccessors;
|
int[] result = parts.get(index).blockSuccessors;
|
||||||
return Arrays.copyOf(result, result.length);
|
return Arrays.copyOf(result, result.length);
|
||||||
|
@ -212,7 +203,6 @@ public class AsyncProgramSplitter {
|
||||||
|
|
||||||
private static class Part {
|
private static class Part {
|
||||||
Program program;
|
Program program;
|
||||||
Integer input;
|
|
||||||
int[] blockSuccessors;
|
int[] blockSuccessors;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -303,7 +303,7 @@ public class TeaVMTool {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (mainClass != null) {
|
if (mainClass != null) {
|
||||||
writer.append("main = $rt_mainWrapper(main);\n");
|
writer.append("main = $rt_mainStarter(main);\n");
|
||||||
}
|
}
|
||||||
ProblemProvider problemProvider = vm.getProblemProvider();
|
ProblemProvider problemProvider = vm.getProblemProvider();
|
||||||
if (problemProvider.getProblems().isEmpty()) {
|
if (problemProvider.getProblems().isEmpty()) {
|
||||||
|
|
|
@ -223,13 +223,16 @@ function $rt_init(cls, constructor, args) {
|
||||||
return obj;
|
return obj;
|
||||||
}
|
}
|
||||||
function $rt_throw(ex) {
|
function $rt_throw(ex) {
|
||||||
|
throw $rt_exception(ex);
|
||||||
|
}
|
||||||
|
function $rt_exception(ex) {
|
||||||
var err = ex.$jsException;
|
var err = ex.$jsException;
|
||||||
if (!err) {
|
if (!err) {
|
||||||
var err = new Error("Java exception thrown");
|
var err = new Error("Java exception thrown");
|
||||||
err.$javaException = ex;
|
err.$javaException = ex;
|
||||||
ex.$jsException = err;
|
ex.$jsException = err;
|
||||||
}
|
}
|
||||||
throw err;
|
return err;
|
||||||
}
|
}
|
||||||
function $rt_createMultiArray(cls, dimensions) {
|
function $rt_createMultiArray(cls, dimensions) {
|
||||||
var arrays = new Array($rt_primitiveArrayCount(dimensions));
|
var arrays = new Array($rt_primitiveArrayCount(dimensions));
|
||||||
|
@ -403,71 +406,15 @@ function $rt_metadata(data) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
function $rt_asyncResult(value) {
|
function $rt_threadStarter(f) {
|
||||||
return function() {
|
|
||||||
return value;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
function $rt_asyncError(e) {
|
|
||||||
return function() {
|
|
||||||
throw new TeaVMAsyncError(e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
function $rt_staticAsyncAdapter(f) {
|
|
||||||
return function() {
|
|
||||||
var result;
|
|
||||||
var args = Array.prototype.slice.apply(arguments);
|
|
||||||
var $return = args.pop();
|
|
||||||
try {
|
|
||||||
result = f.apply(this, args);
|
|
||||||
} catch (e) {
|
|
||||||
return $return($rt_asyncError(e));
|
|
||||||
}
|
|
||||||
return $return($rt_asyncResult(result));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
function $rt_asyncAdapter(f) {
|
|
||||||
return function() {
|
|
||||||
var result;
|
|
||||||
var args = Array.prototype.slice.apply(arguments);
|
|
||||||
var $return = args.pop();
|
|
||||||
args.unshift(this);
|
|
||||||
try {
|
|
||||||
result = f.apply(null, args);
|
|
||||||
} catch (e) {
|
|
||||||
return $return($rt_asyncError(e));
|
|
||||||
}
|
|
||||||
return $return($rt_asyncResult(result));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
function $rt_rootInvocationAdapter(f) {
|
|
||||||
return function() {
|
return function() {
|
||||||
var args = Array.prototype.slice.apply(arguments);
|
var args = Array.prototype.slice.apply(arguments);
|
||||||
args.push(function(result) {
|
$rt_startThread(function() {
|
||||||
try {
|
f.apply(this, args);
|
||||||
result();
|
|
||||||
} catch (e) {
|
|
||||||
var prefix = "Exception occured %s at %o";
|
|
||||||
var hasWrappers = false;
|
|
||||||
while (e instanceof TeaVMAsyncError) {
|
|
||||||
console.error(prefix, e.message, e.stack);
|
|
||||||
e = e.cause;
|
|
||||||
prefix = "Caused by %s at %o";
|
|
||||||
hasWrappers = true;
|
|
||||||
}
|
|
||||||
console.error(!hasWrappers ? prefix : "Root cause is %s at %o", e.message, e.stack);
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
f.apply(this, args);
|
|
||||||
var thread = $rt_getThread();
|
|
||||||
if (thread.hasOwnProperty("postponed")) {
|
|
||||||
while (thread.postponed.length > 0) {
|
|
||||||
thread.postponed.shift()();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
function $rt_mainWrapper(f) {
|
function $rt_mainStarter(f) {
|
||||||
return function(args) {
|
return function(args) {
|
||||||
if (!args) {
|
if (!args) {
|
||||||
args = [];
|
args = [];
|
||||||
|
@ -476,7 +423,7 @@ function $rt_mainWrapper(f) {
|
||||||
for (var i = 0; i < args.length; ++i) {
|
for (var i = 0; i < args.length; ++i) {
|
||||||
javaArgs.data[i] = $rt_str(args[i]);
|
javaArgs.data[i] = $rt_str(args[i]);
|
||||||
}
|
}
|
||||||
$rt_rootInvocationAdapter(f)(javaArgs);
|
$rt_threadStarter(f)(javaArgs);
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
var $rt_stringPool_instance;
|
var $rt_stringPool_instance;
|
||||||
|
@ -489,66 +436,51 @@ function $rt_stringPool(strings) {
|
||||||
function $rt_s(index) {
|
function $rt_s(index) {
|
||||||
return $rt_stringPool_instance[index];
|
return $rt_stringPool_instance[index];
|
||||||
}
|
}
|
||||||
var $rt_continueCounter = 0;
|
|
||||||
function $rt_continue(f) {
|
|
||||||
if ($rt_continueCounter++ == 5) {
|
|
||||||
$rt_continueCounter = 0;
|
|
||||||
return function() {
|
|
||||||
var self = this;
|
|
||||||
var args = arguments;
|
|
||||||
var thread = $rt_getThread();
|
|
||||||
if (!thread.hasOwnProperty("postponed")) {
|
|
||||||
thread.postponed = [];
|
|
||||||
}
|
|
||||||
thread.postponed.push(function() {
|
|
||||||
f.apply(self, args);
|
|
||||||
});
|
|
||||||
};
|
|
||||||
} else {
|
|
||||||
return f;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
function $rt_guardAsync(f, continuation) {
|
|
||||||
return function() {
|
|
||||||
try {
|
|
||||||
return f.apply(this, arguments);
|
|
||||||
} catch (e) {
|
|
||||||
return continuation($rt_asyncError(e));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
function TeaVMThread(runner) {
|
function TeaVMThread(runner) {
|
||||||
this.status = 3;
|
this.status = 3;
|
||||||
this.stack = [];
|
this.stack = [];
|
||||||
this.suspendCallback = null;
|
this.suspendCallback = null;
|
||||||
this.runner = runner;
|
this.runner = runner;
|
||||||
|
this.attribute = null;
|
||||||
}
|
}
|
||||||
TeaVMThread.push = function(value) {
|
TeaVMThread.prototype.push = function(value) {
|
||||||
this.stack.push[value];
|
this.stack.push[value];
|
||||||
|
return this;
|
||||||
}
|
}
|
||||||
TeaVMThread.isResuming = function() {
|
TeaVMThread.prototype.pop = function() {
|
||||||
|
return this.stack.pop();
|
||||||
|
}
|
||||||
|
TeaVMThread.prototype.isResuming = function() {
|
||||||
return this.status == 1;
|
return this.status == 1;
|
||||||
}
|
}
|
||||||
TeaVMThread.isSuspending = function() {
|
TeaVMThread.prototype.isSuspending = function() {
|
||||||
return this.status == 2;
|
return this.status == 2;
|
||||||
}
|
}
|
||||||
TeaVMThread.suspend(callback) {
|
TeaVMThread.prototype.suspend = function(callback) {
|
||||||
this.suspendCallback = callback;
|
this.suspendCallback = callback;
|
||||||
this.status = 1;
|
this.status = 1;
|
||||||
}
|
}
|
||||||
TeaVMThread.start = function() {
|
TeaVMThread.prototype.start = function() {
|
||||||
if (this.status != 3) {
|
if (this.status != 3) {
|
||||||
throw new Error("Thread already started");
|
throw new Error("Thread already started");
|
||||||
}
|
}
|
||||||
|
if ($rt_currentNativeThread !== null) {
|
||||||
|
throw new Error("Another thread is running");
|
||||||
|
}
|
||||||
this.status = 0;
|
this.status = 0;
|
||||||
this.run();
|
this.run();
|
||||||
}
|
}
|
||||||
TeaVMThread.resume = function() {
|
TeaVMThread.prototype.resume = function() {
|
||||||
|
if ($rt_currentNativeThread !== null) {
|
||||||
|
throw new Error("Another thread is running");
|
||||||
|
}
|
||||||
this.status = 2;
|
this.status = 2;
|
||||||
this.run();
|
this.run();
|
||||||
}
|
}
|
||||||
TeaVMThread.run = function() {
|
TeaVMThread.prototype.run = function() {
|
||||||
|
$rt_currentNativeThread = this;
|
||||||
this.runner();
|
this.runner();
|
||||||
|
$rt_currentNativeThread = null;
|
||||||
if (this.suspendCallback !== null) {
|
if (this.suspendCallback !== null) {
|
||||||
var self = this;
|
var self = this;
|
||||||
this.suspendCallback(function() {
|
this.suspendCallback(function() {
|
||||||
|
@ -556,27 +488,22 @@ TeaVMThread.run = function() {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
function $rt_nativeThread(thread) {
|
|
||||||
if (!thread.hasNativeProperty("$teavm_thread")) {
|
|
||||||
thread.$teavm_thread = new TeaVMThread();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
function $rt_suspending() {
|
function $rt_suspending() {
|
||||||
return $rt_nativeThread($rt_getThread()).isSuspending();
|
return $rt_nativeThread().isSuspending();
|
||||||
}
|
}
|
||||||
function $rt_resuming() {
|
function $rt_resuming() {
|
||||||
return $rt_nativeThread($rt_getThread()).isResuming();
|
return $rt_nativeThread().isResuming();
|
||||||
}
|
}
|
||||||
|
function $rt_suspend(callback) {
|
||||||
function TeaVMAsyncError(cause) {
|
return $rt_nativeThread().suspend(callback);
|
||||||
this.message = "Async error occured";
|
}
|
||||||
this.cause = cause;
|
function $rt_startThread(runner) {
|
||||||
if (cause) {
|
new TeaVMThread(runner).start();
|
||||||
this.$javaException = cause.$javaException;
|
}
|
||||||
}
|
var $rt_currentNativeThread = null;
|
||||||
|
function $rt_nativeThread() {
|
||||||
|
return $rt_currentNativeThread;
|
||||||
}
|
}
|
||||||
TeaVMAsyncError.prototype = new Error();
|
|
||||||
TeaVMAsyncError.prototype.constructor = TeaVMAsyncError;
|
|
||||||
|
|
||||||
function $dbg_repr(obj) {
|
function $dbg_repr(obj) {
|
||||||
return obj.toString ? obj.toString() : "";
|
return obj.toString ? obj.toString() : "";
|
||||||
|
|
|
@ -73,9 +73,17 @@ public final class Platform {
|
||||||
return ((PlatformHelper)JS.getGlobal()).nextId();
|
return ((PlatformHelper)JS.getGlobal()).nextId();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static <T> T newInstance(PlatformClass cls) {
|
||||||
|
prepareNewInstance();
|
||||||
|
return newInstanceImpl(cls);
|
||||||
|
}
|
||||||
|
|
||||||
|
@GeneratedBy(PlatformGenerator.class)
|
||||||
|
private static native void prepareNewInstance();
|
||||||
|
|
||||||
@GeneratedBy(PlatformGenerator.class)
|
@GeneratedBy(PlatformGenerator.class)
|
||||||
@PluggableDependency(PlatformGenerator.class)
|
@PluggableDependency(PlatformGenerator.class)
|
||||||
public static native <T> T newInstance(PlatformClass cls);
|
private static native <T> T newInstanceImpl(PlatformClass cls);
|
||||||
|
|
||||||
@GeneratedBy(PlatformGenerator.class)
|
@GeneratedBy(PlatformGenerator.class)
|
||||||
@PluggableDependency(PlatformGenerator.class)
|
@PluggableDependency(PlatformGenerator.class)
|
||||||
|
|
|
@ -38,19 +38,29 @@ public class AsyncMethodGenerator implements Generator, DependencyPlugin {
|
||||||
@Override
|
@Override
|
||||||
public void generate(GeneratorContext context, SourceWriter writer, MethodReference methodRef) throws IOException {
|
public void generate(GeneratorContext context, SourceWriter writer, MethodReference methodRef) throws IOException {
|
||||||
MethodReference asyncRef = getAsyncReference(methodRef);
|
MethodReference asyncRef = getAsyncReference(methodRef);
|
||||||
|
writer.append("var thread").ws().append('=').ws().append("$rt_nativeThread();").softNewLine();
|
||||||
|
writer.append("if").ws().append("(thread.isResuming())").ws().append("{").indent().softNewLine();
|
||||||
|
writer.append("var result").ws().append("=").ws().append("thread.attribute;").softNewLine();
|
||||||
|
writer.append("if").ws().append("(result instanceof Error)").ws().append("{").indent().softNewLine();
|
||||||
|
writer.append("throw result;").softNewLine();
|
||||||
|
writer.outdent().append("}").softNewLine();
|
||||||
|
writer.append("return result;").softNewLine();
|
||||||
|
writer.outdent().append("}").softNewLine();
|
||||||
|
|
||||||
writer.append("var callback").ws().append("=").ws().append("function()").ws().append("{};").softNewLine();
|
writer.append("var callback").ws().append("=").ws().append("function()").ws().append("{};").softNewLine();
|
||||||
writer.append("callback.").appendMethod(completeMethod.getDescriptor()).ws().append("=").ws()
|
writer.append("callback.").appendMethod(completeMethod.getDescriptor()).ws().append("=").ws()
|
||||||
.append("function(val)").ws().append("{").indent().softNewLine();
|
.append("function(val)").ws().append("{").indent().softNewLine();
|
||||||
writer.append("return ").append(context.getCompleteContinuation()).append("($rt_asyncResult(val));")
|
writer.append("thread.attribute").ws().append('=').ws().append("val;").softNewLine();
|
||||||
.softNewLine();
|
writer.append("thread.resume();").softNewLine();
|
||||||
writer.outdent().append("};").softNewLine();
|
writer.outdent().append("};").softNewLine();
|
||||||
writer.append("callback.").appendMethod(errorMethod.getDescriptor()).ws().append("=").ws()
|
writer.append("callback.").appendMethod(errorMethod.getDescriptor()).ws().append("=").ws()
|
||||||
.append("function(e)").ws().append("{").indent().softNewLine();
|
.append("function(e)").ws().append("{").indent().softNewLine();
|
||||||
writer.append("return ").append(context.getCompleteContinuation()).append("($rt_asyncError(e));")
|
writer.append("thread.attribute").ws().append('=').ws().append("$rt_exception(e);").softNewLine();
|
||||||
.softNewLine();
|
writer.append("thread.resume();").softNewLine();
|
||||||
writer.outdent().append("};").softNewLine();
|
writer.outdent().append("};").softNewLine();
|
||||||
|
writer.append("return thread.suspend(function()").ws().append("{").indent().softNewLine();
|
||||||
writer.append("try").ws().append("{").indent().softNewLine();
|
writer.append("try").ws().append("{").indent().softNewLine();
|
||||||
writer.append("return ").appendMethodBody(asyncRef).append('(');
|
writer.appendMethodBody(asyncRef).append('(');
|
||||||
ClassReader cls = context.getClassSource().get(methodRef.getClassName());
|
ClassReader cls = context.getClassSource().get(methodRef.getClassName());
|
||||||
MethodReader method = cls.getMethod(methodRef.getDescriptor());
|
MethodReader method = cls.getMethod(methodRef.getDescriptor());
|
||||||
int start = method.hasModifier(ElementModifier.STATIC) ? 1 : 0;
|
int start = method.hasModifier(ElementModifier.STATIC) ? 1 : 0;
|
||||||
|
@ -60,9 +70,10 @@ public class AsyncMethodGenerator implements Generator, DependencyPlugin {
|
||||||
}
|
}
|
||||||
writer.append("callback);").softNewLine();
|
writer.append("callback);").softNewLine();
|
||||||
writer.outdent().append("}").ws().append("catch($e)").ws().append("{").indent().softNewLine();
|
writer.outdent().append("}").ws().append("catch($e)").ws().append("{").indent().softNewLine();
|
||||||
writer.append("return ").append(context.getCompleteContinuation()).append("($rt_asyncError($e));")
|
writer.append("callback.").appendMethod(errorMethod.getDescriptor()).append("($rt_exception($e));")
|
||||||
.softNewLine();
|
.softNewLine();
|
||||||
writer.outdent().append("}").softNewLine();
|
writer.outdent().append("}").softNewLine();
|
||||||
|
writer.outdent().append("});").softNewLine();
|
||||||
}
|
}
|
||||||
|
|
||||||
private MethodReference getAsyncReference(MethodReference methodRef) {
|
private MethodReference getAsyncReference(MethodReference methodRef) {
|
||||||
|
|
|
@ -49,7 +49,7 @@ public class NewInstanceDependencySupport implements DependencyListener {
|
||||||
@Override
|
@Override
|
||||||
public void methodAchieved(final DependencyAgent agent, MethodDependency method, CallLocation location) {
|
public void methodAchieved(final DependencyAgent agent, MethodDependency method, CallLocation location) {
|
||||||
MethodReader reader = method.getMethod();
|
MethodReader reader = method.getMethod();
|
||||||
if (reader.getOwnerName().equals(Platform.class.getName()) && reader.getName().equals("newInstance")) {
|
if (reader.getOwnerName().equals(Platform.class.getName()) && reader.getName().equals("newInstanceImpl")) {
|
||||||
allClassesNode.connect(method.getResult());
|
allClassesNode.connect(method.getResult());
|
||||||
final MethodReference methodRef = reader.getReference();
|
final MethodReference methodRef = reader.getReference();
|
||||||
method.getResult().addConsumer(new DependencyConsumer() {
|
method.getResult().addConsumer(new DependencyConsumer() {
|
||||||
|
|
|
@ -68,8 +68,11 @@ public class PlatformGenerator implements Generator, Injector, DependencyPlugin
|
||||||
@Override
|
@Override
|
||||||
public void generate(GeneratorContext context, SourceWriter writer, MethodReference methodRef) throws IOException {
|
public void generate(GeneratorContext context, SourceWriter writer, MethodReference methodRef) throws IOException {
|
||||||
switch (methodRef.getName()) {
|
switch (methodRef.getName()) {
|
||||||
case "newInstance":
|
case "newInstanceImpl":
|
||||||
generateNewInstance(context, writer, methodRef);
|
generateNewInstance(writer);
|
||||||
|
break;
|
||||||
|
case "prepareNewInstance":
|
||||||
|
generatePrepareNewInstance(context, writer);
|
||||||
break;
|
break;
|
||||||
case "lookupClass":
|
case "lookupClass":
|
||||||
generateLookup(context, writer);
|
generateLookup(context, writer);
|
||||||
|
@ -89,76 +92,37 @@ public class PlatformGenerator implements Generator, Injector, DependencyPlugin
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void generateNewInstance(GeneratorContext context, SourceWriter writer, MethodReference methodRef)
|
private void generatePrepareNewInstance(GeneratorContext context, SourceWriter writer)
|
||||||
throws IOException {
|
throws IOException {
|
||||||
writer.append("var c").ws().append("=").ws().append("'$$constructor$$';").softNewLine();
|
writer.append("var c").ws().append("=").ws().append("'$$constructor$$';").softNewLine();
|
||||||
if (context.isAsync()) {
|
|
||||||
writer.append("function async(cls, init) {").indent().softNewLine();
|
|
||||||
writer.append("return function($return) {").indent().softNewLine();
|
|
||||||
writer.append("var r = new cls;").softNewLine();
|
|
||||||
writer.append("init(r, $rt_guardAsync(function($restore) {").indent().softNewLine();
|
|
||||||
writer.append("$restore();").softNewLine();
|
|
||||||
writer.append("$return($rt_asyncResult(r))").softNewLine();
|
|
||||||
writer.outdent().append("}));").softNewLine();
|
|
||||||
writer.outdent().append("};").softNewLine();
|
|
||||||
writer.outdent().append("}").softNewLine();
|
|
||||||
|
|
||||||
writer.append("function sync(cls, init) {").indent().softNewLine();
|
|
||||||
writer.append("return function($return) {").indent().softNewLine();
|
|
||||||
writer.append("var r = new cls;").softNewLine();
|
|
||||||
writer.append("try {").indent().softNewLine();
|
|
||||||
writer.append("init(r);").softNewLine();
|
|
||||||
writer.append("$return($rt_asyncResult(r));").softNewLine();
|
|
||||||
writer.outdent().append("} catch (e) {").indent().softNewLine();
|
|
||||||
writer.append("$return($rt_asyncError(e));").softNewLine();
|
|
||||||
writer.outdent().append("}").softNewLine();
|
|
||||||
writer.outdent().append("};").softNewLine();
|
|
||||||
writer.outdent().append("}").softNewLine();
|
|
||||||
}
|
|
||||||
for (String clsName : context.getClassSource().getClassNames()) {
|
for (String clsName : context.getClassSource().getClassNames()) {
|
||||||
ClassReader cls = context.getClassSource().get(clsName);
|
ClassReader cls = context.getClassSource().get(clsName);
|
||||||
MethodReader method = cls.getMethod(new MethodDescriptor("<init>", void.class));
|
MethodReader method = cls.getMethod(new MethodDescriptor("<init>", void.class));
|
||||||
if (method != null) {
|
if (method != null) {
|
||||||
writer.appendClass(clsName).append("[c]").ws().append("=").ws();
|
writer.appendClass(clsName).append("[c]").ws().append("=").ws()
|
||||||
if (!context.isAsync()) {
|
.appendMethodBody(method.getReference()).append(")").append(";").softNewLine();
|
||||||
writer.append(writer.getNaming().getNameForInit(method.getReference()));
|
|
||||||
} else {
|
|
||||||
String function = context.isAsync(method.getReference()) ? "async" : "sync";
|
|
||||||
String methodName = context.isAsync(method.getReference()) ?
|
|
||||||
writer.getNaming().getFullNameForAsync(method.getReference()) :
|
|
||||||
writer.getNaming().getFullNameFor(method.getReference());
|
|
||||||
writer.append(function).append("(").appendClass(clsName).append(',').ws()
|
|
||||||
.append(methodName).append(")");
|
|
||||||
}
|
|
||||||
writer.append(";").softNewLine();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
String selfName = context.isAsync() ? writer.getNaming().getFullNameForAsync(methodRef) :
|
writer.appendMethodBody(Platform.class, "newInstance", PlatformClass.class, Object.class).ws().append('=').ws()
|
||||||
writer.getNaming().getFullNameFor(methodRef);
|
.appendMethodBody(Platform.class, "newInstanceImpl", PlatformClass.class, Object.class)
|
||||||
writer.append(selfName).ws().append("=").ws().append("function(cls");
|
.append(";").softNewLine();
|
||||||
if (context.isAsync()) {
|
}
|
||||||
writer.append(',').ws().append("$return");
|
|
||||||
}
|
|
||||||
writer.append(")").ws().append("{").softNewLine().indent();
|
|
||||||
writer.append("if").ws().append("(!cls.hasOwnProperty(c))").ws().append("{").indent().softNewLine();
|
|
||||||
if (!context.isAsync()) {
|
|
||||||
writer.append("return null;").softNewLine();
|
|
||||||
} else {
|
|
||||||
writer.append("return $return($rt_asyncResult(null));").softNewLine();
|
|
||||||
}
|
|
||||||
writer.outdent().append("}").softNewLine();
|
|
||||||
if (!context.isAsync()) {
|
|
||||||
writer.append("return cls[c]();").softNewLine();
|
|
||||||
} else {
|
|
||||||
writer.append("return cls[c]($return);").softNewLine();
|
|
||||||
}
|
|
||||||
writer.outdent().append("};").softNewLine();
|
|
||||||
|
|
||||||
writer.append("return ").append(selfName).append("(").append(context.getParameterName(1));
|
private void generateNewInstance(SourceWriter writer) throws IOException {
|
||||||
if (context.isAsync()) {
|
writer.append("if").ws().append("($rt_resuming())").ws().append("{").indent().softNewLine();
|
||||||
writer.append(',').ws().append(context.getCompleteContinuation());
|
writer.append("return $rt_nativeThread().pop();").softNewLine();
|
||||||
}
|
writer.outdent().append("}").softNewLine();
|
||||||
writer.append(");").softNewLine();
|
|
||||||
|
writer.append("if").ws().append("(!cls.hasOwnProperty('$$constructor$$'))").ws().append("{")
|
||||||
|
.indent().softNewLine();
|
||||||
|
writer.append("return null;").softNewLine();
|
||||||
|
writer.outdent().append("}").softNewLine();
|
||||||
|
|
||||||
|
writer.append("var $r").ws().append('=').ws().append("cls.$$constructor$$();");
|
||||||
|
writer.append("if").ws().append("($rt_suspending())").ws().append("{").indent().softNewLine();
|
||||||
|
writer.append("return $rt_nativeThread().push($r);").softNewLine();
|
||||||
|
writer.outdent().append("}").softNewLine();
|
||||||
|
writer.append("return $r;").softNewLine();
|
||||||
}
|
}
|
||||||
|
|
||||||
private void generateLookup(GeneratorContext context, SourceWriter writer) throws IOException {
|
private void generateLookup(GeneratorContext context, SourceWriter writer) throws IOException {
|
||||||
|
@ -189,18 +153,8 @@ public class PlatformGenerator implements Generator, Injector, DependencyPlugin
|
||||||
PlatformRunnable.class, void.class);
|
PlatformRunnable.class, void.class);
|
||||||
String runnable = context.getParameterName(1);
|
String runnable = context.getParameterName(1);
|
||||||
writer.append("return window.setTimeout(function()").ws().append("{").indent().softNewLine();
|
writer.append("return window.setTimeout(function()").ws().append("{").indent().softNewLine();
|
||||||
boolean async = context.isAsyncFamily(launchRef);
|
writer.append("$rt_threadStarter(").appendMethodBody(launchRef).append(")");
|
||||||
String methodName = async ? writer.getNaming().getFullNameForAsync(launchRef) :
|
writer.append("(").append(runnable).append(");").softNewLine();
|
||||||
writer.getNaming().getFullNameFor(launchRef);
|
|
||||||
if (async) {
|
|
||||||
writer.append("$rt_rootInvocationAdapter(");
|
|
||||||
}
|
|
||||||
writer.append(methodName);
|
|
||||||
if (async) {
|
|
||||||
writer.append(")");
|
|
||||||
}
|
|
||||||
writer.append("(").append(runnable).append(");")
|
|
||||||
.softNewLine();
|
|
||||||
writer.outdent().append("},").ws().append(timeout ? context.getParameterName(2) : "0")
|
writer.outdent().append("},").ws().append(timeout ? context.getParameterName(2) : "0")
|
||||||
.append(");").softNewLine();
|
.append(");").softNewLine();
|
||||||
}
|
}
|
||||||
|
|
|
@ -65,10 +65,12 @@
|
||||||
<targetDirectory>${project.build.directory}/generated/js/teavm</targetDirectory>
|
<targetDirectory>${project.build.directory}/generated/js/teavm</targetDirectory>
|
||||||
<mainClass>org.teavm.samples.async.AsyncProgram</mainClass>
|
<mainClass>org.teavm.samples.async.AsyncProgram</mainClass>
|
||||||
<runtime>SEPARATE</runtime>
|
<runtime>SEPARATE</runtime>
|
||||||
<minifying>true</minifying>
|
<minifying>false</minifying>
|
||||||
<debugInformationGenerated>true</debugInformationGenerated>
|
<debugInformationGenerated>true</debugInformationGenerated>
|
||||||
|
<!--
|
||||||
<sourceMapsGenerated>true</sourceMapsGenerated>
|
<sourceMapsGenerated>true</sourceMapsGenerated>
|
||||||
<sourceFilesCopied>true</sourceFilesCopied>
|
<sourceFilesCopied>true</sourceFilesCopied>
|
||||||
|
-->
|
||||||
</configuration>
|
</configuration>
|
||||||
</execution>
|
</execution>
|
||||||
</executions>
|
</executions>
|
||||||
|
|
Loading…
Reference in New Issue
Block a user