122 lines
4.3 KiB
Plaintext
122 lines
4.3 KiB
Plaintext
|
package slangc.model.statements;
|
||
|
|
||
|
import slangc.api.BytecodeInstructionWriter;
|
||
|
import slangc.api.Reporter;
|
||
|
import slangc.model.ClauseModel;
|
||
|
import slangc.model.InnerTypeScope;
|
||
|
import slangc.model.MemberModel;
|
||
|
import slangc.model.StatementModel;
|
||
|
import slangc.model.StatementOwner;
|
||
|
import slangc.model.TypeModel;
|
||
|
import slangc.model.clauses.CatchClause;
|
||
|
import slangc.model.clauses.FinallyClause;
|
||
|
import slangc.parser.Branch;
|
||
|
import slangc.parser.Node;
|
||
|
|
||
|
public class TryStatement extends StatementModel implements StatementOwner {
|
||
|
private StatementModel innerStatement;
|
||
|
private CatchClause[] catchClauses;
|
||
|
private FinallyClause finallyClause;
|
||
|
|
||
|
public TryStatement(StatementOwner owner, Branch source) {
|
||
|
super(owner, source);
|
||
|
innerStatement = StatementModel.construct(this, source.getSubnode(2));
|
||
|
ClauseModel[] catches = ClauseModel.constructMultiple(this, (Branch)source.getSubnode(3));
|
||
|
catchClauses = new CatchClause[catches.length];
|
||
|
for (int i = 0; i < catches.length; i++) {
|
||
|
catchClauses[i] = (CatchClause)catches[i];
|
||
|
}
|
||
|
finallyClause = (FinallyClause)ClauseModel.construct(this, source.getSubnode(4));
|
||
|
}
|
||
|
|
||
|
//@Override
|
||
|
public void dump(Reporter reporter, String indent, String incr) {
|
||
|
reporter.note("DUMP", indent + "> Try:");
|
||
|
innerStatement.dump(reporter, indent + incr, incr);
|
||
|
for (int i = 0; i < catchClauses.length; i++) {
|
||
|
catchClauses[i].dump(reporter, indent + incr, incr);
|
||
|
}
|
||
|
if (finallyClause != null) {
|
||
|
finallyClause.dump(reporter, indent + incr, incr);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//@Override
|
||
|
public int resolveExpressions() {
|
||
|
int nresolved = 0;
|
||
|
nresolved += innerStatement.resolveExpressions();
|
||
|
for (int i = 0; i < catchClauses.length; i++) {
|
||
|
nresolved += catchClauses[i].resolveExpressions();
|
||
|
}
|
||
|
if (finallyClause != null) {
|
||
|
nresolved += finallyClause.resolveExpressions();
|
||
|
}
|
||
|
return nresolved;
|
||
|
}
|
||
|
|
||
|
//@Override
|
||
|
public TypeModel resolveType(Node subnode) {
|
||
|
return getOwner().resolveType(subnode);
|
||
|
}
|
||
|
|
||
|
//@Override
|
||
|
public MemberModel getMethodOrField() {
|
||
|
return getOwner().getMethodOrField();
|
||
|
}
|
||
|
|
||
|
//@Override
|
||
|
public InnerTypeScope getTypeScope() {
|
||
|
return getOwner().getTypeScope();
|
||
|
}
|
||
|
|
||
|
public MethodModel[] lookupSimpleMethod(String name) {
|
||
|
// TODO: Better interface support
|
||
|
return super.lookupSimpleMethod(name);
|
||
|
}
|
||
|
|
||
|
public slangc.model.Named lookupSimpleName(String name) {
|
||
|
// TODO: Better interface support
|
||
|
return super.lookupSimpleName(name);
|
||
|
}
|
||
|
|
||
|
@Override
|
||
|
public void innerGenerate(BytecodeInstructionWriter w) {
|
||
|
BytecodeInstructionWriter.Label begin = w.newLabel("try_begin");
|
||
|
BytecodeInstructionWriter.Label end = w.newLabel("try_end");
|
||
|
BytecodeInstructionWriter.Label catchEnd = w.newLabel("try_catch_end");
|
||
|
BytecodeInstructionWriter.CatchLabel[] catches = new BytecodeInstructionWriter.CatchLabel[catchClauses.length];
|
||
|
BytecodeInstructionWriter.FinallyLabel fin = finallyClause == null ? null : w.newFinallyLabel("try_finally", begin, catchEnd);
|
||
|
|
||
|
for (int i = 0; i < catches.length; i++) {
|
||
|
catches[i] = w.newCatchLabel("try_catch", begin, end, catchClauses[i].getCatchType().getTypeSignature());
|
||
|
}
|
||
|
|
||
|
w.labelHere(begin);
|
||
|
getInnerStatement().generate(w);
|
||
|
w.genJump(BytecodeInstructionWriter.JumpOp.GOTO, catchEnd);
|
||
|
w.labelHere(end);
|
||
|
for (int i = 0; i < catches.length; i++) {
|
||
|
w.labelHere(catches[i]);
|
||
|
// Manually notify the writer that a typeName (the exception) has been pushed
|
||
|
w.vpush(catchClauses[i].getCatchType().getTypeSignature());
|
||
|
// Now we just need to store it to our variable so it can be accessed normally
|
||
|
// It needs to be declared first, but in practice a backend may just ignore this declaration
|
||
|
w.genDeclareLocal(catchClauses[i].getCatchVariable().getStorageType().getTypeSignature(), catchClauses[i].getCatchVariable().getIndex());
|
||
|
w.genStoreLocalOrParam(catchClauses[i].getCatchVariable().getStorageType().getTypeSignature(), catchClauses[i].getCatchVariable().getIndex());
|
||
|
catchClauses[i].getInnerStatement().generate(w);
|
||
|
w.genJump(BytecodeInstructionWriter.JumpOp.GOTO, catchEnd);
|
||
|
}
|
||
|
w.labelHere(catchEnd);
|
||
|
|
||
|
if (finallyClause != null) {
|
||
|
w.labelHere(fin);
|
||
|
finallyClause.getInnerStatement().generate(w);
|
||
|
w.genCheckRethrow();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public StatementModel getInnerStatement() {
|
||
|
return innerStatement;
|
||
|
}
|
||
|
}
|