[mips][interpreter] Make FunctionEntry StackCheck bytecodes implicit
port 9d3dc6f
https://crrev.com/c/1914218
Original Commit Message:
FunctionEntry StackChecks is one of the two cases where we generate a
StackCheck bytecode. In these cases, we do stack check against the js
limit (not to be confused with the real js limit). Their purpose is to
be able to interrupt the running code.
We can omit the FunctionEntry StackCheck by embedding its code into
the InterpreterEntryTrampoline builtin. We save one bytecode per
interpreted function.
This change has rippling effects for optimized code, as well as the
deoptimizer.
Change-Id: I2ad985959a3f55dfe80fabeffee3d344c1817c88
Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/2050055
Auto-Submit: Zhao Jiazhong <zhaojiazhong-hf@loongson.cn>
Reviewed-by: Jakob Gruber <jgruber@chromium.org>
Commit-Queue: Jakob Gruber <jgruber@chromium.org>
Cr-Commit-Position: refs/heads/master@{#66281}
This commit is contained in:
parent
7b17b5fb84
commit
ad3c4e3d74
@ -62,10 +62,16 @@ static void GenerateTailCallToReturnedCode(MacroAssembler* masm,
|
||||
|
||||
namespace {
|
||||
|
||||
void LoadRealStackLimit(MacroAssembler* masm, Register destination) {
|
||||
enum StackLimitKind { kInterruptStackLimit, kRealStackLimit };
|
||||
|
||||
void LoadStackLimit(MacroAssembler* masm, Register destination,
|
||||
StackLimitKind kind) {
|
||||
DCHECK(masm->root_array_available());
|
||||
Isolate* isolate = masm->isolate();
|
||||
ExternalReference limit = ExternalReference::address_of_real_jslimit(isolate);
|
||||
ExternalReference limit =
|
||||
kind == StackLimitKind::kRealStackLimit
|
||||
? ExternalReference::address_of_real_jslimit(isolate)
|
||||
: ExternalReference::address_of_jslimit(isolate);
|
||||
DCHECK(TurboAssembler::IsAddressableThroughRootRegister(isolate, limit));
|
||||
|
||||
intptr_t offset =
|
||||
@ -144,7 +150,7 @@ static void Generate_StackOverflowCheck(MacroAssembler* masm, Register num_args,
|
||||
// Check the stack for overflow. We are not trying to catch
|
||||
// interruptions (e.g. debug break and preemption) here, so the "real stack
|
||||
// limit" is checked.
|
||||
LoadRealStackLimit(masm, scratch1);
|
||||
LoadStackLimit(masm, scratch1, StackLimitKind::kRealStackLimit);
|
||||
// Make scratch1 the space we have left. The stack might already be overflowed
|
||||
// here which will cause scratch1 to become negative.
|
||||
__ subu(scratch1, sp, scratch1);
|
||||
@ -355,7 +361,7 @@ static void Generate_CheckStackOverflow(MacroAssembler* masm, Register argc,
|
||||
// interruptions (e.g. debug break and preemption) here, so the "real stack
|
||||
// limit" is checked.
|
||||
Label okay;
|
||||
LoadRealStackLimit(masm, scratch1);
|
||||
LoadStackLimit(masm, scratch1, StackLimitKind::kRealStackLimit);
|
||||
// Make a2 the space we have left. The stack might already be overflowed
|
||||
// here which will cause a2 to become negative.
|
||||
__ Subu(scratch1, sp, scratch1);
|
||||
@ -702,7 +708,7 @@ void Builtins::Generate_ResumeGeneratorTrampoline(MacroAssembler* masm) {
|
||||
// Check the stack for overflow. We are not trying to catch interruptions
|
||||
// (i.e. debug break and preemption) here, so check the "real stack limit".
|
||||
Label stack_overflow;
|
||||
LoadRealStackLimit(masm, kScratchReg);
|
||||
LoadStackLimit(masm, kScratchReg, StackLimitKind::kRealStackLimit);
|
||||
__ Branch(&stack_overflow, lo, sp, Operand(kScratchReg));
|
||||
|
||||
// Push receiver.
|
||||
@ -1059,7 +1065,7 @@ void Builtins::Generate_InterpreterEntryTrampoline(MacroAssembler* masm) {
|
||||
|
||||
// Do a stack check to ensure we don't go over the limit.
|
||||
__ Subu(t1, sp, Operand(t0));
|
||||
LoadRealStackLimit(masm, a2);
|
||||
LoadStackLimit(masm, a2, StackLimitKind::kRealStackLimit);
|
||||
__ Branch(&stack_overflow, lo, t1, Operand(a2));
|
||||
|
||||
// If ok, push undefined as the initial value for all register file entries.
|
||||
@ -1088,6 +1094,13 @@ void Builtins::Generate_InterpreterEntryTrampoline(MacroAssembler* masm) {
|
||||
__ sw(a3, MemOperand(t1));
|
||||
__ bind(&no_incoming_new_target_or_generator_register);
|
||||
|
||||
// Perform interrupt stack check.
|
||||
// TODO(solanes): Merge with the real stack limit check above.
|
||||
Label stack_check_interrupt, after_stack_check_interrupt;
|
||||
LoadStackLimit(masm, a2, StackLimitKind::kInterruptStackLimit);
|
||||
__ Branch(&stack_check_interrupt, lo, sp, Operand(a2));
|
||||
__ bind(&after_stack_check_interrupt);
|
||||
|
||||
// Load accumulator with undefined.
|
||||
__ LoadRoot(kInterpreterAccumulatorRegister, RootIndex::kUndefinedValue);
|
||||
|
||||
@ -1129,6 +1142,30 @@ void Builtins::Generate_InterpreterEntryTrampoline(MacroAssembler* masm) {
|
||||
LeaveInterpreterFrame(masm, t0);
|
||||
__ Jump(ra);
|
||||
|
||||
__ bind(&stack_check_interrupt);
|
||||
// Modify the bytecode offset in the stack to be kFunctionEntryBytecodeOffset
|
||||
// for the call to the StackGuard.
|
||||
__ li(kInterpreterBytecodeOffsetRegister,
|
||||
Operand(Smi::FromInt(BytecodeArray::kHeaderSize - kHeapObjectTag +
|
||||
kFunctionEntryBytecodeOffset)));
|
||||
__ Sw(kInterpreterBytecodeOffsetRegister,
|
||||
MemOperand(fp, InterpreterFrameConstants::kBytecodeOffsetFromFp));
|
||||
__ CallRuntime(Runtime::kStackGuard);
|
||||
|
||||
// After the call, restore the bytecode array, bytecode offset and accumulator
|
||||
// registers again. Also, restore the bytecode offset in the stack to its
|
||||
// previous value.
|
||||
__ Lw(kInterpreterBytecodeArrayRegister,
|
||||
MemOperand(fp, InterpreterFrameConstants::kBytecodeArrayFromFp));
|
||||
__ li(kInterpreterBytecodeOffsetRegister,
|
||||
Operand(BytecodeArray::kHeaderSize - kHeapObjectTag));
|
||||
__ LoadRoot(kInterpreterAccumulatorRegister, RootIndex::kUndefinedValue);
|
||||
|
||||
__ SmiTag(a2, kInterpreterBytecodeOffsetRegister);
|
||||
__ Sw(a2, MemOperand(fp, InterpreterFrameConstants::kBytecodeOffsetFromFp));
|
||||
|
||||
__ jmp(&after_stack_check_interrupt);
|
||||
|
||||
__ bind(&optimized_code_slot_not_empty);
|
||||
Label maybe_has_optimized_code;
|
||||
// Check if optimized code marker is actually a weak reference to the
|
||||
@ -1333,6 +1370,15 @@ static void Generate_InterpreterEnterBytecode(MacroAssembler* masm) {
|
||||
MemOperand(fp, InterpreterFrameConstants::kBytecodeOffsetFromFp));
|
||||
__ SmiUntag(kInterpreterBytecodeOffsetRegister);
|
||||
|
||||
if (FLAG_debug_code) {
|
||||
Label okay;
|
||||
__ Branch(&okay, ge, kInterpreterBytecodeOffsetRegister,
|
||||
Operand(BytecodeArray::kHeaderSize - kHeapObjectTag));
|
||||
// Unreachable code.
|
||||
__ break_(0xCC);
|
||||
__ bind(&okay);
|
||||
}
|
||||
|
||||
// Dispatch to the target bytecode.
|
||||
__ Addu(a1, kInterpreterBytecodeArrayRegister,
|
||||
kInterpreterBytecodeOffsetRegister);
|
||||
@ -1352,6 +1398,11 @@ void Builtins::Generate_InterpreterEnterBytecodeAdvance(MacroAssembler* masm) {
|
||||
MemOperand(fp, InterpreterFrameConstants::kBytecodeOffsetFromFp));
|
||||
__ SmiUntag(kInterpreterBytecodeOffsetRegister);
|
||||
|
||||
Label enter_bytecode, function_entry_bytecode;
|
||||
__ Branch(&function_entry_bytecode, eq, kInterpreterBytecodeOffsetRegister,
|
||||
Operand(BytecodeArray::kHeaderSize - kHeapObjectTag +
|
||||
kFunctionEntryBytecodeOffset));
|
||||
|
||||
// Load the current bytecode.
|
||||
__ Addu(a1, kInterpreterBytecodeArrayRegister,
|
||||
kInterpreterBytecodeOffsetRegister);
|
||||
@ -1363,12 +1414,22 @@ void Builtins::Generate_InterpreterEnterBytecodeAdvance(MacroAssembler* masm) {
|
||||
kInterpreterBytecodeOffsetRegister, a1, a2, a3,
|
||||
&if_return);
|
||||
|
||||
__ bind(&enter_bytecode);
|
||||
// Convert new bytecode offset to a Smi and save in the stackframe.
|
||||
__ SmiTag(a2, kInterpreterBytecodeOffsetRegister);
|
||||
__ sw(a2, MemOperand(fp, InterpreterFrameConstants::kBytecodeOffsetFromFp));
|
||||
|
||||
Generate_InterpreterEnterBytecode(masm);
|
||||
|
||||
__ bind(&function_entry_bytecode);
|
||||
// If the code deoptimizes during the implicit function entry stack interrupt
|
||||
// check, it will have a bailout ID of kFunctionEntryBytecodeOffset, which is
|
||||
// not a valid bytecode offset. Detect this case and advance to the first
|
||||
// actual bytecode.
|
||||
__ li(kInterpreterBytecodeOffsetRegister,
|
||||
Operand(BytecodeArray::kHeaderSize - kHeapObjectTag));
|
||||
__ Branch(&enter_bytecode);
|
||||
|
||||
// We should never take the if_return path.
|
||||
__ bind(&if_return);
|
||||
__ Abort(AbortReason::kInvalidBytecodeAdvance);
|
||||
@ -1983,7 +2044,7 @@ void Builtins::Generate_CallBoundFunctionImpl(MacroAssembler* masm) {
|
||||
__ Subu(sp, sp, Operand(t1));
|
||||
// Check the stack for overflow. We are not trying to catch interruptions
|
||||
// (i.e. debug break and preemption) here, so check the "real stack limit".
|
||||
LoadRealStackLimit(masm, kScratchReg);
|
||||
LoadStackLimit(masm, kScratchReg, StackLimitKind::kRealStackLimit);
|
||||
__ Branch(&done, hs, sp, Operand(kScratchReg));
|
||||
// Restore the stack pointer.
|
||||
__ Addu(sp, sp, Operand(t1));
|
||||
@ -2140,7 +2201,7 @@ void Builtins::Generate_ConstructBoundFunction(MacroAssembler* masm) {
|
||||
__ Subu(sp, sp, Operand(t1));
|
||||
// Check the stack for overflow. We are not trying to catch interruptions
|
||||
// (i.e. debug break and preemption) here, so check the "real stack limit".
|
||||
LoadRealStackLimit(masm, kScratchReg);
|
||||
LoadStackLimit(masm, kScratchReg, StackLimitKind::kRealStackLimit);
|
||||
__ Branch(&done, hs, sp, Operand(kScratchReg));
|
||||
// Restore the stack pointer.
|
||||
__ Addu(sp, sp, Operand(t1));
|
||||
|
@ -61,10 +61,16 @@ static void GenerateTailCallToReturnedCode(MacroAssembler* masm,
|
||||
|
||||
namespace {
|
||||
|
||||
void LoadRealStackLimit(MacroAssembler* masm, Register destination) {
|
||||
enum StackLimitKind { kInterruptStackLimit, kRealStackLimit };
|
||||
|
||||
void LoadStackLimit(MacroAssembler* masm, Register destination,
|
||||
StackLimitKind kind) {
|
||||
DCHECK(masm->root_array_available());
|
||||
Isolate* isolate = masm->isolate();
|
||||
ExternalReference limit = ExternalReference::address_of_real_jslimit(isolate);
|
||||
ExternalReference limit =
|
||||
kind == StackLimitKind::kRealStackLimit
|
||||
? ExternalReference::address_of_real_jslimit(isolate)
|
||||
: ExternalReference::address_of_jslimit(isolate);
|
||||
DCHECK(TurboAssembler::IsAddressableThroughRootRegister(isolate, limit));
|
||||
|
||||
intptr_t offset =
|
||||
@ -145,7 +151,7 @@ static void Generate_StackOverflowCheck(MacroAssembler* masm, Register num_args,
|
||||
// Check the stack for overflow. We are not trying to catch
|
||||
// interruptions (e.g. debug break and preemption) here, so the "real stack
|
||||
// limit" is checked.
|
||||
LoadRealStackLimit(masm, scratch1);
|
||||
LoadStackLimit(masm, scratch1, StackLimitKind::kRealStackLimit);
|
||||
// Make scratch1 the space we have left. The stack might already be overflowed
|
||||
// here which will cause scratch1 to become negative.
|
||||
__ dsubu(scratch1, sp, scratch1);
|
||||
@ -395,7 +401,7 @@ void Builtins::Generate_ResumeGeneratorTrampoline(MacroAssembler* masm) {
|
||||
// Check the stack for overflow. We are not trying to catch interruptions
|
||||
// (i.e. debug break and preemption) here, so check the "real stack limit".
|
||||
Label stack_overflow;
|
||||
LoadRealStackLimit(masm, kScratchReg);
|
||||
LoadStackLimit(masm, kScratchReg, StackLimitKind::kRealStackLimit);
|
||||
__ Branch(&stack_overflow, lo, sp, Operand(kScratchReg));
|
||||
|
||||
// Push receiver.
|
||||
@ -502,7 +508,7 @@ static void Generate_CheckStackOverflow(MacroAssembler* masm, Register argc,
|
||||
// interruptions (e.g. debug break and preemption) here, so the "real stack
|
||||
// limit" is checked.
|
||||
Label okay;
|
||||
LoadRealStackLimit(masm, scratch1);
|
||||
LoadStackLimit(masm, scratch1, StackLimitKind::kRealStackLimit);
|
||||
// Make a2 the space we have left. The stack might already be overflowed
|
||||
// here which will cause r2 to become negative.
|
||||
__ dsubu(scratch1, sp, scratch1);
|
||||
@ -1077,7 +1083,7 @@ void Builtins::Generate_InterpreterEntryTrampoline(MacroAssembler* masm) {
|
||||
|
||||
// Do a stack check to ensure we don't go over the limit.
|
||||
__ Dsubu(a5, sp, Operand(a4));
|
||||
LoadRealStackLimit(masm, a2);
|
||||
LoadStackLimit(masm, a2, StackLimitKind::kRealStackLimit);
|
||||
__ Branch(&stack_overflow, lo, a5, Operand(a2));
|
||||
|
||||
// If ok, push undefined as the initial value for all register file entries.
|
||||
@ -1106,6 +1112,13 @@ void Builtins::Generate_InterpreterEntryTrampoline(MacroAssembler* masm) {
|
||||
__ Sd(a3, MemOperand(a5));
|
||||
__ bind(&no_incoming_new_target_or_generator_register);
|
||||
|
||||
// Perform interrupt stack check.
|
||||
// TODO(solanes): Merge with the real stack limit check above.
|
||||
Label stack_check_interrupt, after_stack_check_interrupt;
|
||||
LoadStackLimit(masm, a5, StackLimitKind::kInterruptStackLimit);
|
||||
__ Branch(&stack_check_interrupt, lo, sp, Operand(a5));
|
||||
__ bind(&after_stack_check_interrupt);
|
||||
|
||||
// Load accumulator as undefined.
|
||||
__ LoadRoot(kInterpreterAccumulatorRegister, RootIndex::kUndefinedValue);
|
||||
|
||||
@ -1148,6 +1161,30 @@ void Builtins::Generate_InterpreterEntryTrampoline(MacroAssembler* masm) {
|
||||
LeaveInterpreterFrame(masm, t0);
|
||||
__ Jump(ra);
|
||||
|
||||
__ bind(&stack_check_interrupt);
|
||||
// Modify the bytecode offset in the stack to be kFunctionEntryBytecodeOffset
|
||||
// for the call to the StackGuard.
|
||||
__ li(kInterpreterBytecodeOffsetRegister,
|
||||
Operand(Smi::FromInt(BytecodeArray::kHeaderSize - kHeapObjectTag +
|
||||
kFunctionEntryBytecodeOffset)));
|
||||
__ Sd(kInterpreterBytecodeOffsetRegister,
|
||||
MemOperand(fp, InterpreterFrameConstants::kBytecodeOffsetFromFp));
|
||||
__ CallRuntime(Runtime::kStackGuard);
|
||||
|
||||
// After the call, restore the bytecode array, bytecode offset and accumulator
|
||||
// registers again. Also, restore the bytecode offset in the stack to its
|
||||
// previous value.
|
||||
__ Ld(kInterpreterBytecodeArrayRegister,
|
||||
MemOperand(fp, InterpreterFrameConstants::kBytecodeArrayFromFp));
|
||||
__ li(kInterpreterBytecodeOffsetRegister,
|
||||
Operand(BytecodeArray::kHeaderSize - kHeapObjectTag));
|
||||
__ LoadRoot(kInterpreterAccumulatorRegister, RootIndex::kUndefinedValue);
|
||||
|
||||
__ SmiTag(a5, kInterpreterBytecodeOffsetRegister);
|
||||
__ Sd(a5, MemOperand(fp, InterpreterFrameConstants::kBytecodeOffsetFromFp));
|
||||
|
||||
__ jmp(&after_stack_check_interrupt);
|
||||
|
||||
__ bind(&optimized_code_slot_not_empty);
|
||||
Label maybe_has_optimized_code;
|
||||
// Check if optimized code marker is actually a weak reference to the
|
||||
@ -1351,6 +1388,15 @@ static void Generate_InterpreterEnterBytecode(MacroAssembler* masm) {
|
||||
__ SmiUntag(kInterpreterBytecodeOffsetRegister,
|
||||
MemOperand(fp, InterpreterFrameConstants::kBytecodeOffsetFromFp));
|
||||
|
||||
if (FLAG_debug_code) {
|
||||
Label okay;
|
||||
__ Branch(&okay, ge, kInterpreterBytecodeOffsetRegister,
|
||||
Operand(BytecodeArray::kHeaderSize - kHeapObjectTag));
|
||||
// Unreachable code.
|
||||
__ break_(0xCC);
|
||||
__ bind(&okay);
|
||||
}
|
||||
|
||||
// Dispatch to the target bytecode.
|
||||
__ Daddu(a1, kInterpreterBytecodeArrayRegister,
|
||||
kInterpreterBytecodeOffsetRegister);
|
||||
@ -1370,6 +1416,11 @@ void Builtins::Generate_InterpreterEnterBytecodeAdvance(MacroAssembler* masm) {
|
||||
MemOperand(fp, InterpreterFrameConstants::kBytecodeOffsetFromFp));
|
||||
__ SmiUntag(kInterpreterBytecodeOffsetRegister);
|
||||
|
||||
Label enter_bytecode, function_entry_bytecode;
|
||||
__ Branch(&function_entry_bytecode, eq, kInterpreterBytecodeOffsetRegister,
|
||||
Operand(BytecodeArray::kHeaderSize - kHeapObjectTag +
|
||||
kFunctionEntryBytecodeOffset));
|
||||
|
||||
// Load the current bytecode.
|
||||
__ Daddu(a1, kInterpreterBytecodeArrayRegister,
|
||||
kInterpreterBytecodeOffsetRegister);
|
||||
@ -1381,12 +1432,22 @@ void Builtins::Generate_InterpreterEnterBytecodeAdvance(MacroAssembler* masm) {
|
||||
kInterpreterBytecodeOffsetRegister, a1, a2, a3,
|
||||
&if_return);
|
||||
|
||||
__ bind(&enter_bytecode);
|
||||
// Convert new bytecode offset to a Smi and save in the stackframe.
|
||||
__ SmiTag(a2, kInterpreterBytecodeOffsetRegister);
|
||||
__ Sd(a2, MemOperand(fp, InterpreterFrameConstants::kBytecodeOffsetFromFp));
|
||||
|
||||
Generate_InterpreterEnterBytecode(masm);
|
||||
|
||||
__ bind(&function_entry_bytecode);
|
||||
// If the code deoptimizes during the implicit function entry stack interrupt
|
||||
// check, it will have a bailout ID of kFunctionEntryBytecodeOffset, which is
|
||||
// not a valid bytecode offset. Detect this case and advance to the first
|
||||
// actual bytecode.
|
||||
__ li(kInterpreterBytecodeOffsetRegister,
|
||||
Operand(BytecodeArray::kHeaderSize - kHeapObjectTag));
|
||||
__ Branch(&enter_bytecode);
|
||||
|
||||
// We should never take the if_return path.
|
||||
__ bind(&if_return);
|
||||
__ Abort(AbortReason::kInvalidBytecodeAdvance);
|
||||
@ -2024,7 +2085,7 @@ void Builtins::Generate_CallBoundFunctionImpl(MacroAssembler* masm) {
|
||||
__ Dsubu(sp, sp, Operand(a5));
|
||||
// Check the stack for overflow. We are not trying to catch interruptions
|
||||
// (i.e. debug break and preemption) here, so check the "real stack limit".
|
||||
LoadRealStackLimit(masm, kScratchReg);
|
||||
LoadStackLimit(masm, kScratchReg, StackLimitKind::kRealStackLimit);
|
||||
__ Branch(&done, hs, sp, Operand(kScratchReg));
|
||||
// Restore the stack pointer.
|
||||
__ Daddu(sp, sp, Operand(a5));
|
||||
@ -2177,7 +2238,7 @@ void Builtins::Generate_ConstructBoundFunction(MacroAssembler* masm) {
|
||||
__ Dsubu(sp, sp, Operand(a5));
|
||||
// Check the stack for overflow. We are not trying to catch interruptions
|
||||
// (i.e. debug break and preemption) here, so check the "real stack limit".
|
||||
LoadRealStackLimit(masm, kScratchReg);
|
||||
LoadStackLimit(masm, kScratchReg, StackLimitKind::kRealStackLimit);
|
||||
__ Branch(&done, hs, sp, Operand(kScratchReg));
|
||||
// Restore the stack pointer.
|
||||
__ Daddu(sp, sp, Operand(a5));
|
||||
|
Loading…
Reference in New Issue
Block a user