package org.graalvm.compiler.hotspot.amd64;
import static org.graalvm.compiler.lir.LIRInstruction.OperandFlag.REG;
import static jdk.vm.ci.amd64.AMD64.rbp;
import static jdk.vm.ci.amd64.AMD64.rsp;
import static jdk.vm.ci.code.ValueUtil.asRegister;
import org.graalvm.compiler.asm.amd64.AMD64Address;
import org.graalvm.compiler.asm.amd64.AMD64Assembler.ConditionFlag;
import org.graalvm.compiler.asm.amd64.AMD64MacroAssembler;
import org.graalvm.compiler.lir.LIRInstructionClass;
import org.graalvm.compiler.lir.Opcode;
import org.graalvm.compiler.lir.asm.CompilationResultBuilder;
import org.graalvm.compiler.serviceprovider.GraalServices;
import jdk.vm.ci.code.Register;
import jdk.vm.ci.meta.AllocatableValue;
@Opcode("JUMP_TO_EXCEPTION_HANDLER_IN_CALLER")
final class AMD64HotSpotJumpToExceptionHandlerInCallerOp extends AMD64HotSpotEpilogueBlockEndOp {
public static final LIRInstructionClass<AMD64HotSpotJumpToExceptionHandlerInCallerOp> TYPE = LIRInstructionClass.create(AMD64HotSpotJumpToExceptionHandlerInCallerOp.class);
@Use(REG) AllocatableValue handlerInCallerPc;
@Use(REG) AllocatableValue exception;
@Use(REG) AllocatableValue exceptionPc;
private final Register thread;
private final int isMethodHandleReturnOffset;
AMD64HotSpotJumpToExceptionHandlerInCallerOp(AllocatableValue handlerInCallerPc, AllocatableValue exception, AllocatableValue exceptionPc, int isMethodHandleReturnOffset, Register thread) {
super(TYPE);
this.handlerInCallerPc = handlerInCallerPc;
this.exception = exception;
this.exceptionPc = exceptionPc;
this.isMethodHandleReturnOffset = isMethodHandleReturnOffset;
this.thread = thread;
}
@Override
public void emitCode(CompilationResultBuilder crb, AMD64MacroAssembler masm) {
leaveFrameAndRestoreRbp(crb, masm);
masm.incrementq(rsp, 8);
if (GraalServices.JAVA_SPECIFICATION_VERSION < 8) {
AMD64Address dst = new AMD64Address(thread, isMethodHandleReturnOffset);
masm.cmpl(dst, 0);
masm.cmovq(ConditionFlag.NotEqual, rsp, rbp);
}
masm.jmp(asRegister(handlerInCallerPc));
}
}