X87: [fullcode] Switch passing of new.target to register.
port 440a42b741
(r32548)
original commit message:
This passes the new.target value in a register instead of through a
side-channel via the construct stub. Note that this marks the last
consumer of said side-channel and the special slot in the construct
stub frame can be removed as a follow-up.
BUG=
Review URL: https://codereview.chromium.org/1503923002
Cr-Commit-Position: refs/heads/master@{#32635}
This commit is contained in:
parent
d5a52b6636
commit
e3b1cf1726
@ -83,6 +83,7 @@ class JumpPatchSite BASE_EMBEDDED {
|
|||||||
//
|
//
|
||||||
// The live registers are:
|
// The live registers are:
|
||||||
// o edi: the JS function object being called (i.e. ourselves)
|
// o edi: the JS function object being called (i.e. ourselves)
|
||||||
|
// o edx: the new target value
|
||||||
// o esi: our context
|
// o esi: our context
|
||||||
// o ebp: our caller's frame pointer
|
// o ebp: our caller's frame pointer
|
||||||
// o esp: stack pointer (pointing to return address)
|
// o esp: stack pointer (pointing to return address)
|
||||||
@ -174,7 +175,13 @@ void FullCodeGenerator::Generate() {
|
|||||||
__ Push(info->scope()->GetScopeInfo(info->isolate()));
|
__ Push(info->scope()->GetScopeInfo(info->isolate()));
|
||||||
__ CallRuntime(Runtime::kNewScriptContext, 2);
|
__ CallRuntime(Runtime::kNewScriptContext, 2);
|
||||||
PrepareForBailoutForId(BailoutId::ScriptContext(), TOS_REG);
|
PrepareForBailoutForId(BailoutId::ScriptContext(), TOS_REG);
|
||||||
} else if (slots <= FastNewContextStub::kMaximumSlots) {
|
// The new target value is not used, clobbering is safe.
|
||||||
|
DCHECK_NULL(info->scope()->new_target_var());
|
||||||
|
} else {
|
||||||
|
if (info->scope()->new_target_var() != nullptr) {
|
||||||
|
__ push(edx); // Preserve new target.
|
||||||
|
}
|
||||||
|
if (slots <= FastNewContextStub::kMaximumSlots) {
|
||||||
FastNewContextStub stub(isolate(), slots);
|
FastNewContextStub stub(isolate(), slots);
|
||||||
__ CallStub(&stub);
|
__ CallStub(&stub);
|
||||||
// Result of FastNewContextStub is always in new space.
|
// Result of FastNewContextStub is always in new space.
|
||||||
@ -183,6 +190,10 @@ void FullCodeGenerator::Generate() {
|
|||||||
__ push(edi);
|
__ push(edi);
|
||||||
__ CallRuntime(Runtime::kNewFunctionContext, 1);
|
__ CallRuntime(Runtime::kNewFunctionContext, 1);
|
||||||
}
|
}
|
||||||
|
if (info->scope()->new_target_var() != nullptr) {
|
||||||
|
__ pop(edx); // Restore new target.
|
||||||
|
}
|
||||||
|
}
|
||||||
function_in_register = false;
|
function_in_register = false;
|
||||||
// Context is returned in eax. It replaces the context passed to us.
|
// Context is returned in eax. It replaces the context passed to us.
|
||||||
// It's saved in the stack and kept live in esi.
|
// It's saved in the stack and kept live in esi.
|
||||||
@ -215,11 +226,11 @@ void FullCodeGenerator::Generate() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
PrepareForBailoutForId(BailoutId::FunctionContext(), NO_REGISTERS);
|
|
||||||
|
|
||||||
// Function register is trashed in case we bailout here. But since that
|
// Register holding this function and new target are both trashed in case we
|
||||||
// could happen only when we allocate a context the value of
|
// bailout here. But since that can happen only when new target is not used
|
||||||
// |function_in_register| is correct.
|
// and we allocate a context, the value of |function_in_register| is correct.
|
||||||
|
PrepareForBailoutForId(BailoutId::FunctionContext(), NO_REGISTERS);
|
||||||
|
|
||||||
// Possibly set up a local binding to the this function which is used in
|
// Possibly set up a local binding to the this function which is used in
|
||||||
// derived constructors with super calls.
|
// derived constructors with super calls.
|
||||||
@ -228,38 +239,16 @@ void FullCodeGenerator::Generate() {
|
|||||||
Comment cmnt(masm_, "[ This function");
|
Comment cmnt(masm_, "[ This function");
|
||||||
if (!function_in_register) {
|
if (!function_in_register) {
|
||||||
__ mov(edi, Operand(ebp, JavaScriptFrameConstants::kFunctionOffset));
|
__ mov(edi, Operand(ebp, JavaScriptFrameConstants::kFunctionOffset));
|
||||||
// The write barrier clobbers register again, keep is marked as such.
|
// The write barrier clobbers register again, keep it marked as such.
|
||||||
}
|
}
|
||||||
SetVar(this_function_var, edi, ebx, edx);
|
SetVar(this_function_var, edi, ebx, ecx);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Possibly set up a local binding to the new target value.
|
||||||
Variable* new_target_var = scope()->new_target_var();
|
Variable* new_target_var = scope()->new_target_var();
|
||||||
if (new_target_var != nullptr) {
|
if (new_target_var != nullptr) {
|
||||||
Comment cmnt(masm_, "[ new.target");
|
Comment cmnt(masm_, "[ new.target");
|
||||||
__ mov(eax, Operand(ebp, StandardFrameConstants::kCallerFPOffset));
|
SetVar(new_target_var, edx, ebx, ecx);
|
||||||
Label non_adaptor_frame;
|
|
||||||
__ cmp(Operand(eax, StandardFrameConstants::kContextOffset),
|
|
||||||
Immediate(Smi::FromInt(StackFrame::ARGUMENTS_ADAPTOR)));
|
|
||||||
__ j(not_equal, &non_adaptor_frame);
|
|
||||||
__ mov(eax, Operand(eax, StandardFrameConstants::kCallerFPOffset));
|
|
||||||
|
|
||||||
__ bind(&non_adaptor_frame);
|
|
||||||
__ cmp(Operand(eax, StandardFrameConstants::kMarkerOffset),
|
|
||||||
Immediate(Smi::FromInt(StackFrame::CONSTRUCT)));
|
|
||||||
|
|
||||||
Label non_construct_frame, done;
|
|
||||||
__ j(not_equal, &non_construct_frame);
|
|
||||||
|
|
||||||
// Construct frame
|
|
||||||
__ mov(eax, Operand(eax, ConstructFrameConstants::kNewTargetOffset));
|
|
||||||
__ jmp(&done);
|
|
||||||
|
|
||||||
// Non-construct frame
|
|
||||||
__ bind(&non_construct_frame);
|
|
||||||
__ mov(eax, Immediate(isolate()->factory()->undefined_value()));
|
|
||||||
|
|
||||||
__ bind(&done);
|
|
||||||
SetVar(new_target_var, eax, ebx, edx);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Variable* arguments = scope()->arguments();
|
Variable* arguments = scope()->arguments();
|
||||||
|
Loading…
Reference in New Issue
Block a user