Commit fcb83f20 authored by ishell's avatar ishell Committed by Commit bot

[crankshaft] [ia32] Remove dynamic frame alignment optimization.

This optimization does not give us much (see perf try bot results associated with this CL) but complicates things a lot. The main motivation is to avoid additional complexity in tail call optimization.

There are some pieces left in the deoptimizer, but I'll address this in a separate CL.

Review URL: https://codereview.chromium.org/1731273003

Cr-Commit-Position: refs/heads/master@{#34273}
parent fe38ad57
...@@ -14,7 +14,6 @@ namespace internal { ...@@ -14,7 +14,6 @@ namespace internal {
\ \
V(k32BitValueInRegisterIsNotZeroExtended, \ V(k32BitValueInRegisterIsNotZeroExtended, \
"32 bit value in register is not zero-extended") \ "32 bit value in register is not zero-extended") \
V(kAlignmentMarkerExpected, "Alignment marker expected") \
V(kAllocationIsNotDoubleAligned, "Allocation is not double aligned") \ V(kAllocationIsNotDoubleAligned, "Allocation is not double aligned") \
V(kAPICallReturnedInvalidObject, "API call returned invalid object") \ V(kAPICallReturnedInvalidObject, "API call returned invalid object") \
V(kArgumentsObjectValueInATestContext, \ V(kArgumentsObjectValueInATestContext, \
......
...@@ -3601,7 +3601,6 @@ HGraph::HGraph(CompilationInfo* info, CallInterfaceDescriptor descriptor) ...@@ -3601,7 +3601,6 @@ HGraph::HGraph(CompilationInfo* info, CallInterfaceDescriptor descriptor)
info_(info), info_(info),
descriptor_(descriptor), descriptor_(descriptor),
zone_(info->zone()), zone_(info->zone()),
is_recursive_(false),
use_optimistic_licm_(false), use_optimistic_licm_(false),
depends_on_empty_array_proto_elements_(false), depends_on_empty_array_proto_elements_(false),
type_change_checksum_(0), type_change_checksum_(0),
...@@ -8056,9 +8055,6 @@ HInstruction* HOptimizedGraphBuilder::BuildCallConstantFunction( ...@@ -8056,9 +8055,6 @@ HInstruction* HOptimizedGraphBuilder::BuildCallConstantFunction(
bool can_invoke_directly = bool can_invoke_directly =
dont_adapt_arguments || formal_parameter_count == arity; dont_adapt_arguments || formal_parameter_count == arity;
if (can_invoke_directly) { if (can_invoke_directly) {
if (jsfun.is_identical_to(current_info()->closure())) {
graph()->MarkRecursive();
}
return NewPlainFunctionCall(target, argument_count); return NewPlainFunctionCall(target, argument_count);
} else { } else {
HValue* param_count_value = Add<HConstant>(formal_parameter_count); HValue* param_count_value = Add<HConstant>(formal_parameter_count);
......
...@@ -400,9 +400,6 @@ class HGraph final : public ZoneObject { ...@@ -400,9 +400,6 @@ class HGraph final : public ZoneObject {
use_optimistic_licm_ = value; use_optimistic_licm_ = value;
} }
void MarkRecursive() { is_recursive_ = true; }
bool is_recursive() const { return is_recursive_; }
void MarkDependsOnEmptyArrayProtoElements() { void MarkDependsOnEmptyArrayProtoElements() {
// Add map dependency if not already added. // Add map dependency if not already added.
if (depends_on_empty_array_proto_elements_) return; if (depends_on_empty_array_proto_elements_) return;
...@@ -482,7 +479,6 @@ class HGraph final : public ZoneObject { ...@@ -482,7 +479,6 @@ class HGraph final : public ZoneObject {
CallInterfaceDescriptor descriptor_; CallInterfaceDescriptor descriptor_;
Zone* zone_; Zone* zone_;
bool is_recursive_;
bool use_optimistic_licm_; bool use_optimistic_licm_;
bool depends_on_empty_array_proto_elements_; bool depends_on_empty_array_proto_elements_;
int type_change_checksum_; int type_change_checksum_;
......
...@@ -57,13 +57,6 @@ bool LCodeGen::GenerateCode() { ...@@ -57,13 +57,6 @@ bool LCodeGen::GenerateCode() {
// the frame (that is done in GeneratePrologue). // the frame (that is done in GeneratePrologue).
FrameScope frame_scope(masm_, StackFrame::MANUAL); FrameScope frame_scope(masm_, StackFrame::MANUAL);
support_aligned_spilled_doubles_ = info()->IsOptimizing();
dynamic_frame_alignment_ = info()->IsOptimizing() &&
((chunk()->num_double_slots() > 2 &&
!chunk()->graph()->is_recursive()) ||
!info()->osr_ast_id().IsNone());
return GeneratePrologue() && return GeneratePrologue() &&
GenerateBody() && GenerateBody() &&
GenerateDeferredCode() && GenerateDeferredCode() &&
...@@ -130,31 +123,6 @@ bool LCodeGen::GeneratePrologue() { ...@@ -130,31 +123,6 @@ bool LCodeGen::GeneratePrologue() {
if (info()->IsOptimizing()) { if (info()->IsOptimizing()) {
ProfileEntryHookStub::MaybeCallEntryHook(masm_); ProfileEntryHookStub::MaybeCallEntryHook(masm_);
if (support_aligned_spilled_doubles_ && dynamic_frame_alignment_) {
// Move state of dynamic frame alignment into edx.
__ Move(edx, Immediate(kNoAlignmentPadding));
Label do_not_pad, align_loop;
STATIC_ASSERT(kDoubleSize == 2 * kPointerSize);
// Align esp + 4 to a multiple of 2 * kPointerSize.
__ test(esp, Immediate(kPointerSize));
__ j(not_zero, &do_not_pad, Label::kNear);
__ push(Immediate(0));
__ mov(ebx, esp);
__ mov(edx, Immediate(kAlignmentPaddingPushed));
// Copy arguments, receiver, and return address.
__ mov(ecx, Immediate(scope()->num_parameters() + 2));
__ bind(&align_loop);
__ mov(eax, Operand(ebx, 1 * kPointerSize));
__ mov(Operand(ebx, 0), eax);
__ add(Operand(ebx), Immediate(kPointerSize));
__ dec(ecx);
__ j(not_zero, &align_loop, Label::kNear);
__ mov(Operand(ebx, 0), Immediate(kAlignmentZapValue));
__ bind(&do_not_pad);
}
} }
info()->set_prologue_offset(masm_->pc_offset()); info()->set_prologue_offset(masm_->pc_offset());
...@@ -168,55 +136,23 @@ bool LCodeGen::GeneratePrologue() { ...@@ -168,55 +136,23 @@ bool LCodeGen::GeneratePrologue() {
} }
} }
if (info()->IsOptimizing() &&
dynamic_frame_alignment_ &&
FLAG_debug_code) {
__ test(esp, Immediate(kPointerSize));
__ Assert(zero, kFrameIsExpectedToBeAligned);
}
// Reserve space for the stack slots needed by the code. // Reserve space for the stack slots needed by the code.
int slots = GetStackSlotCount(); int slots = GetStackSlotCount();
DCHECK(slots != 0 || !info()->IsOptimizing()); DCHECK(slots != 0 || !info()->IsOptimizing());
if (slots > 0) { if (slots > 0) {
if (slots == 1) { __ sub(Operand(esp), Immediate(slots * kPointerSize));
if (dynamic_frame_alignment_) {
__ push(edx);
} else {
__ push(Immediate(kNoAlignmentPadding));
}
} else {
if (FLAG_debug_code) {
__ sub(Operand(esp), Immediate(slots * kPointerSize));
#ifdef _MSC_VER
MakeSureStackPagesMapped(slots * kPointerSize);
#endif
__ push(eax);
__ mov(Operand(eax), Immediate(slots));
Label loop;
__ bind(&loop);
__ mov(MemOperand(esp, eax, times_4, 0),
Immediate(kSlotsZapValue));
__ dec(eax);
__ j(not_zero, &loop);
__ pop(eax);
} else {
__ sub(Operand(esp), Immediate(slots * kPointerSize));
#ifdef _MSC_VER #ifdef _MSC_VER
MakeSureStackPagesMapped(slots * kPointerSize); MakeSureStackPagesMapped(slots * kPointerSize);
#endif #endif
} if (FLAG_debug_code) {
__ push(eax);
if (support_aligned_spilled_doubles_) { __ mov(Operand(eax), Immediate(slots));
Comment(";;; Store dynamic frame alignment tag for spilled doubles"); Label loop;
// Store dynamic frame alignment state in the first local. __ bind(&loop);
int offset = JavaScriptFrameConstants::kDynamicAlignmentStateOffset; __ mov(MemOperand(esp, eax, times_4, 0), Immediate(kSlotsZapValue));
if (dynamic_frame_alignment_) { __ dec(eax);
__ mov(Operand(ebp, offset), edx); __ j(not_zero, &loop);
} else { __ pop(eax);
__ mov(Operand(ebp, offset), Immediate(kNoAlignmentPadding));
}
}
} }
if (info()->saves_caller_doubles()) SaveCallerDoubles(); if (info()->saves_caller_doubles()) SaveCallerDoubles();
...@@ -298,47 +234,11 @@ void LCodeGen::GenerateOsrPrologue() { ...@@ -298,47 +234,11 @@ void LCodeGen::GenerateOsrPrologue() {
osr_pc_offset_ = masm()->pc_offset(); osr_pc_offset_ = masm()->pc_offset();
// Move state of dynamic frame alignment into edx.
__ Move(edx, Immediate(kNoAlignmentPadding));
if (support_aligned_spilled_doubles_ && dynamic_frame_alignment_) {
Label do_not_pad, align_loop;
// Align ebp + 4 to a multiple of 2 * kPointerSize.
__ test(ebp, Immediate(kPointerSize));
__ j(zero, &do_not_pad, Label::kNear);
__ push(Immediate(0));
__ mov(ebx, esp);
__ mov(edx, Immediate(kAlignmentPaddingPushed));
// Move all parts of the frame over one word. The frame consists of:
// unoptimized frame slots, alignment state, context, frame pointer, return
// address, receiver, and the arguments.
__ mov(ecx, Immediate(scope()->num_parameters() +
5 + graph()->osr()->UnoptimizedFrameSlots()));
__ bind(&align_loop);
__ mov(eax, Operand(ebx, 1 * kPointerSize));
__ mov(Operand(ebx, 0), eax);
__ add(Operand(ebx), Immediate(kPointerSize));
__ dec(ecx);
__ j(not_zero, &align_loop, Label::kNear);
__ mov(Operand(ebx, 0), Immediate(kAlignmentZapValue));
__ sub(Operand(ebp), Immediate(kPointerSize));
__ bind(&do_not_pad);
}
// Save the first local, which is overwritten by the alignment state.
Operand alignment_loc = MemOperand(ebp, -3 * kPointerSize);
__ push(alignment_loc);
// Set the dynamic frame alignment state.
__ mov(alignment_loc, edx);
// Adjust the frame size, subsuming the unoptimized frame into the // Adjust the frame size, subsuming the unoptimized frame into the
// optimized frame. // optimized frame.
int slots = GetStackSlotCount() - graph()->osr()->UnoptimizedFrameSlots(); int slots = GetStackSlotCount() - graph()->osr()->UnoptimizedFrameSlots();
DCHECK(slots >= 1); DCHECK(slots >= 0);
__ sub(esp, Immediate((slots - 1) * kPointerSize)); __ sub(esp, Immediate(slots * kPointerSize));
} }
...@@ -2462,18 +2362,11 @@ void LCodeGen::DoCmpT(LCmpT* instr) { ...@@ -2462,18 +2362,11 @@ void LCodeGen::DoCmpT(LCmpT* instr) {
__ bind(&done); __ bind(&done);
} }
void LCodeGen::EmitReturn(LReturn* instr) {
void LCodeGen::EmitReturn(LReturn* instr, bool dynamic_frame_alignment) { int extra_value_count = 1;
int extra_value_count = dynamic_frame_alignment ? 2 : 1;
if (instr->has_constant_parameter_count()) { if (instr->has_constant_parameter_count()) {
int parameter_count = ToInteger32(instr->constant_parameter_count()); int parameter_count = ToInteger32(instr->constant_parameter_count());
if (dynamic_frame_alignment && FLAG_debug_code) {
__ cmp(Operand(esp,
(parameter_count + extra_value_count) * kPointerSize),
Immediate(kAlignmentZapValue));
__ Assert(equal, kExpectedAlignmentMarker);
}
__ Ret((parameter_count + extra_value_count) * kPointerSize, ecx); __ Ret((parameter_count + extra_value_count) * kPointerSize, ecx);
} else { } else {
DCHECK(info()->IsStub()); // Functions would need to drop one more value. DCHECK(info()->IsStub()); // Functions would need to drop one more value.
...@@ -2481,20 +2374,9 @@ void LCodeGen::EmitReturn(LReturn* instr, bool dynamic_frame_alignment) { ...@@ -2481,20 +2374,9 @@ void LCodeGen::EmitReturn(LReturn* instr, bool dynamic_frame_alignment) {
// The argument count parameter is a smi // The argument count parameter is a smi
__ SmiUntag(reg); __ SmiUntag(reg);
Register return_addr_reg = reg.is(ecx) ? ebx : ecx; Register return_addr_reg = reg.is(ecx) ? ebx : ecx;
if (dynamic_frame_alignment && FLAG_debug_code) {
DCHECK(extra_value_count == 2);
__ cmp(Operand(esp, reg, times_pointer_size,
extra_value_count * kPointerSize),
Immediate(kAlignmentZapValue));
__ Assert(equal, kExpectedAlignmentMarker);
}
// emit code to restore stack based on instr->parameter_count() // emit code to restore stack based on instr->parameter_count()
__ pop(return_addr_reg); // save return address __ pop(return_addr_reg); // save return address
if (dynamic_frame_alignment) {
__ inc(reg); // 1 more for alignment
}
__ shl(reg, kPointerSizeLog2); __ shl(reg, kPointerSizeLog2);
__ add(esp, reg); __ add(esp, reg);
__ jmp(return_addr_reg); __ jmp(return_addr_reg);
...@@ -2513,25 +2395,12 @@ void LCodeGen::DoReturn(LReturn* instr) { ...@@ -2513,25 +2395,12 @@ void LCodeGen::DoReturn(LReturn* instr) {
__ CallRuntime(Runtime::kTraceExit); __ CallRuntime(Runtime::kTraceExit);
} }
if (info()->saves_caller_doubles()) RestoreCallerDoubles(); if (info()->saves_caller_doubles()) RestoreCallerDoubles();
if (dynamic_frame_alignment_) {
// Fetch the state of the dynamic frame alignment.
__ mov(edx, Operand(ebp,
JavaScriptFrameConstants::kDynamicAlignmentStateOffset));
}
if (NeedsEagerFrame()) { if (NeedsEagerFrame()) {
__ mov(esp, ebp); __ mov(esp, ebp);
__ pop(ebp); __ pop(ebp);
} }
if (dynamic_frame_alignment_) {
Label no_padding;
__ cmp(edx, Immediate(kNoAlignmentPadding));
__ j(equal, &no_padding, Label::kNear);
EmitReturn(instr, true);
__ bind(&no_padding);
}
EmitReturn(instr, false); EmitReturn(instr);
} }
......
...@@ -29,8 +29,6 @@ class LCodeGen: public LCodeGenBase { ...@@ -29,8 +29,6 @@ class LCodeGen: public LCodeGenBase {
jump_table_(4, info->zone()), jump_table_(4, info->zone()),
scope_(info->scope()), scope_(info->scope()),
deferred_(8, info->zone()), deferred_(8, info->zone()),
dynamic_frame_alignment_(false),
support_aligned_spilled_doubles_(false),
frame_is_built_(false), frame_is_built_(false),
safepoints_(info->zone()), safepoints_(info->zone()),
resolver_(this), resolver_(this),
...@@ -300,7 +298,7 @@ class LCodeGen: public LCodeGenBase { ...@@ -300,7 +298,7 @@ class LCodeGen: public LCodeGenBase {
template <class T> template <class T>
void EmitVectorStoreICRegisters(T* instr); void EmitVectorStoreICRegisters(T* instr);
void EmitReturn(LReturn* instr, bool dynamic_frame_alignment); void EmitReturn(LReturn* instr);
// Emits code for pushing either a tagged constant, a (non-double) // Emits code for pushing either a tagged constant, a (non-double)
// register, or a stack slot operand. // register, or a stack slot operand.
...@@ -319,8 +317,6 @@ class LCodeGen: public LCodeGenBase { ...@@ -319,8 +317,6 @@ class LCodeGen: public LCodeGenBase {
ZoneList<Deoptimizer::JumpTableEntry> jump_table_; ZoneList<Deoptimizer::JumpTableEntry> jump_table_;
Scope* const scope_; Scope* const scope_;
ZoneList<LDeferredCode*> deferred_; ZoneList<LDeferredCode*> deferred_;
bool dynamic_frame_alignment_;
bool support_aligned_spilled_doubles_;
bool frame_is_built_; bool frame_is_built_;
// Builder that keeps track of safepoints in the code. The table // Builder that keeps track of safepoints in the code. The table
......
...@@ -420,13 +420,6 @@ LPlatformChunk* LChunkBuilder::Build() { ...@@ -420,13 +420,6 @@ LPlatformChunk* LChunkBuilder::Build() {
LPhase phase("L_Building chunk", chunk_); LPhase phase("L_Building chunk", chunk_);
status_ = BUILDING; status_ = BUILDING;
// Reserve the first spill slot for the state of dynamic alignment.
if (info()->IsOptimizing()) {
int alignment_state_index = chunk_->GetNextSpillIndex(GENERAL_REGISTERS);
DCHECK_EQ(alignment_state_index, 4);
USE(alignment_state_index);
}
// If compiling for OSR, reserve space for the unoptimized frame, // If compiling for OSR, reserve space for the unoptimized frame,
// which will be subsumed into this frame. // which will be subsumed into this frame.
if (graph()->has_osr()) { if (graph()->has_osr()) {
...@@ -2477,11 +2470,6 @@ LInstruction* LChunkBuilder::DoUnknownOSRValue(HUnknownOSRValue* instr) { ...@@ -2477,11 +2470,6 @@ LInstruction* LChunkBuilder::DoUnknownOSRValue(HUnknownOSRValue* instr) {
Retry(kNotEnoughSpillSlotsForOsr); Retry(kNotEnoughSpillSlotsForOsr);
spill_index = 0; spill_index = 0;
} }
if (spill_index == 0) {
// The dynamic frame alignment state overwrites the first local.
// The first local is saved at the end of the unoptimized frame.
spill_index = graph()->osr()->UnoptimizedFrameSlots();
}
spill_index += StandardFrameConstants::kFixedSlotCount; spill_index += StandardFrameConstants::kFixedSlotCount;
} }
return DefineAsSpilled(new(zone()) LUnknownOSRValue, spill_index); return DefineAsSpilled(new(zone()) LUnknownOSRValue, spill_index);
......
...@@ -491,16 +491,7 @@ StackFrame::Type StackFrame::GetCallerState(State* state) const { ...@@ -491,16 +491,7 @@ StackFrame::Type StackFrame::GetCallerState(State* state) const {
Address StackFrame::UnpaddedFP() const { Address StackFrame::UnpaddedFP() const {
#if V8_TARGET_ARCH_IA32 || V8_TARGET_ARCH_X87
if (!is_optimized()) return fp();
int32_t alignment_state = Memory::int32_at(
fp() + JavaScriptFrameConstants::kDynamicAlignmentStateOffset);
return (alignment_state == kAlignmentPaddingPushed) ?
(fp() + kPointerSize) : fp();
#else
return fp(); return fp();
#endif
} }
......
...@@ -187,16 +187,8 @@ void Deoptimizer::CopyDoubleRegisters(FrameDescription* output_frame) { ...@@ -187,16 +187,8 @@ void Deoptimizer::CopyDoubleRegisters(FrameDescription* output_frame) {
} }
bool Deoptimizer::HasAlignmentPadding(SharedFunctionInfo* shared) { bool Deoptimizer::HasAlignmentPadding(SharedFunctionInfo* shared) {
int parameter_count = shared->internal_formal_parameter_count() + 1; // There is no dynamic alignment padding on ia32 in the input frame.
unsigned input_frame_size = input_->GetFrameSize(); return false;
unsigned alignment_state_offset =
input_frame_size - parameter_count * kPointerSize -
StandardFrameConstants::kFixedFrameSize -
kPointerSize;
DCHECK(JavaScriptFrameConstants::kDynamicAlignmentStateOffset ==
JavaScriptFrameConstants::kLocal0Offset);
int32_t alignment_state = input_->GetFrameSlot(alignment_state_offset);
return (alignment_state == kAlignmentPaddingPushed);
} }
...@@ -311,18 +303,6 @@ void Deoptimizer::TableEntryGenerator::Generate() { ...@@ -311,18 +303,6 @@ void Deoptimizer::TableEntryGenerator::Generate() {
} }
__ pop(eax); __ pop(eax);
// If frame was dynamically aligned, pop padding.
Label no_padding;
__ cmp(Operand(eax, Deoptimizer::has_alignment_padding_offset()),
Immediate(0));
__ j(equal, &no_padding);
__ pop(ecx);
if (FLAG_debug_code) {
__ cmp(ecx, Immediate(kAlignmentZapValue));
__ Assert(equal, kAlignmentMarkerExpected);
}
__ bind(&no_padding);
// Replace the current frame with the output frames. // Replace the current frame with the output frames.
Label outer_push_loop, inner_push_loop, Label outer_push_loop, inner_push_loop,
outer_loop_header, inner_loop_header; outer_loop_header, inner_loop_header;
......
...@@ -28,10 +28,6 @@ const int kNumJSCallerSaved = 5; ...@@ -28,10 +28,6 @@ const int kNumJSCallerSaved = 5;
// Number of registers for which space is reserved in safepoints. // Number of registers for which space is reserved in safepoints.
const int kNumSafepointRegisters = 8; const int kNumSafepointRegisters = 8;
const int kNoAlignmentPadding = 0;
const int kAlignmentPaddingPushed = 2;
const int kAlignmentZapValue = 0x12345678; // Not heap object tagged.
// ---------------------------------------------------- // ----------------------------------------------------
...@@ -75,8 +71,6 @@ class JavaScriptFrameConstants : public AllStatic { ...@@ -75,8 +71,6 @@ class JavaScriptFrameConstants : public AllStatic {
// Caller SP-relative. // Caller SP-relative.
static const int kParam0Offset = -2 * kPointerSize; static const int kParam0Offset = -2 * kPointerSize;
static const int kReceiverOffset = -1 * kPointerSize; static const int kReceiverOffset = -1 * kPointerSize;
static const int kDynamicAlignmentStateOffset = kLocal0Offset;
}; };
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment