slcom/slangc/model/statements/TryStatement.sauce

122 lines
4.3 KiB
Plaintext
Raw Permalink Normal View History

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;
}
}