net.sandius.rembulan.compiler.gen.asm.ResumeMethod Maven / Gradle / Ivy
The newest version!
/*
* Copyright 2016 Miroslav Janíček
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package net.sandius.rembulan.compiler.gen.asm;
import net.sandius.rembulan.compiler.gen.asm.helpers.ASMUtils;
import net.sandius.rembulan.impl.DefaultSavedState;
import net.sandius.rembulan.impl.NonsuspendableFunctionException;
import net.sandius.rembulan.runtime.ExecutionContext;
import net.sandius.rembulan.util.Check;
import org.objectweb.asm.Type;
import org.objectweb.asm.tree.InsnList;
import org.objectweb.asm.tree.InsnNode;
import org.objectweb.asm.tree.LabelNode;
import org.objectweb.asm.tree.LocalVariableNode;
import org.objectweb.asm.tree.MethodInsnNode;
import org.objectweb.asm.tree.MethodNode;
import org.objectweb.asm.tree.TypeInsnNode;
import org.objectweb.asm.tree.VarInsnNode;
import java.util.List;
import static org.objectweb.asm.Opcodes.*;
class ResumeMethod {
private final ASMBytecodeEmitter context;
private final RunMethod runMethod;
public ResumeMethod(ASMBytecodeEmitter context, RunMethod runMethod) {
this.context = Check.notNull(context);
this.runMethod = Check.notNull(runMethod);
}
public MethodNode methodNode() {
MethodNode node = new MethodNode(
ACC_PUBLIC,
"resume",
Type.getMethodType(
Type.VOID_TYPE,
Type.getType(ExecutionContext.class),
Type.getType(Object.class)).getDescriptor(),
null,
runMethod.throwsExceptions());
if (runMethod.isResumable()) {
InsnList il = node.instructions;
List locals = node.localVariables;
LabelNode begin = new LabelNode();
LabelNode vars = new LabelNode();
LabelNode end = new LabelNode();
il.add(begin);
il.add(new VarInsnNode(ALOAD, 2));
il.add(new TypeInsnNode(CHECKCAST, Type.getInternalName(DefaultSavedState.class)));
il.add(vars);
il.add(new VarInsnNode(ASTORE, 3));
il.add(new VarInsnNode(ALOAD, 0)); // this
il.add(new VarInsnNode(ALOAD, 1)); // context
il.add(new VarInsnNode(ALOAD, 3)); // saved state
il.add(new MethodInsnNode(
INVOKEVIRTUAL,
Type.getInternalName(DefaultSavedState.class),
"resumptionPoint",
Type.getMethodDescriptor(
Type.INT_TYPE),
false
)); // resumption point
// registers
if (context.isVararg() || runMethod.numOfRegisters() > 0) {
il.add(new VarInsnNode(ALOAD, 3));
il.add(new MethodInsnNode(
INVOKEVIRTUAL,
Type.getInternalName(DefaultSavedState.class),
"registers",
Type.getMethodDescriptor(
ASMUtils.arrayTypeFor(Object.class)),
false
));
// varargs stored as the 0th element
int numRegs = runMethod.numOfRegisters() + (context.isVararg() ? 1 : 0);
for (int i = 0; i < numRegs; i++) {
// Note: it might be more elegant to use a local variable
// to store the array instead of having to perform SWAPs
if (i + 1 < numRegs) {
il.add(new InsnNode(DUP));
}
il.add(ASMUtils.loadInt(i));
il.add(new InsnNode(AALOAD));
if (i == 0 && context.isVararg()) {
il.add(new TypeInsnNode(CHECKCAST, ASMUtils.arrayTypeFor(Object.class).getInternalName()));
}
if (i + 1 < numRegs) {
il.add(new InsnNode(SWAP));
}
}
}
// call run(...)
il.add(runMethod.methodInvokeInsn());
il.add(new InsnNode(RETURN));
il.add(end);
locals.add(new LocalVariableNode("this", context.thisClassType().getDescriptor(), null, begin, end, 0));
locals.add(new LocalVariableNode("context", Type.getDescriptor(ExecutionContext.class), null, begin, end, 1));
locals.add(new LocalVariableNode("suspendedState", context.savedStateClassType().getDescriptor(), null, begin, end, 2));
locals.add(new LocalVariableNode("ss", Type.getDescriptor(DefaultSavedState.class), null, vars, end, 3));
// TODO: maxStack, maxLocals
node.maxStack = 3 + (runMethod.numOfRegisters() > 0 ? 3: 0);
node.maxLocals = 5;
}
else
{
InsnList il = node.instructions;
List locals = node.localVariables;
LabelNode begin = new LabelNode();
LabelNode end = new LabelNode();
il.add(begin);
il.add(new TypeInsnNode(NEW, Type.getInternalName(NonsuspendableFunctionException.class)));
il.add(new InsnNode(DUP));
il.add(ASMUtils.ctor(NonsuspendableFunctionException.class));
il.add(new InsnNode(ATHROW));
il.add(end);
locals.add(new LocalVariableNode("this", context.thisClassType().getDescriptor(), null, begin, end, 0));
locals.add(new LocalVariableNode("context", Type.getDescriptor(ExecutionContext.class), null, begin, end, 1));
locals.add(new LocalVariableNode("suspendedState", context.savedStateClassType().getDescriptor(), null, begin, end, 2));
node.maxStack = 2;
node.maxLocals = 3;
}
return node;
}
}