Fix jump targets on ARM to merge virtual frames (really this time).
Use the jump targets to move the first deferred code to not flush registers in the inlined case (KeyedLoad). Review URL: http://codereview.chromium.org/2249002 git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@4745 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
This commit is contained in:
parent
67c0ec6f17
commit
73256b05af
@ -66,50 +66,19 @@ static void MultiplyByKnownInt(MacroAssembler* masm,
|
|||||||
static bool IsEasyToMultiplyBy(int x);
|
static bool IsEasyToMultiplyBy(int x);
|
||||||
|
|
||||||
|
|
||||||
#define __ ACCESS_MASM(masm)
|
|
||||||
|
|
||||||
// -------------------------------------------------------------------------
|
|
||||||
// Platform-specific FrameRegisterState functions.
|
|
||||||
|
|
||||||
void FrameRegisterState::Save(MacroAssembler* masm) const {
|
|
||||||
for (int i = 0; i < RegisterAllocator::kNumRegisters; i++) {
|
|
||||||
int action = registers_[i];
|
|
||||||
if (action == kPush) {
|
|
||||||
__ push(RegisterAllocator::ToRegister(i));
|
|
||||||
} else if (action != kIgnore && (action & kSyncedFlag) == 0) {
|
|
||||||
__ str(RegisterAllocator::ToRegister(i), MemOperand(fp, action));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void FrameRegisterState::Restore(MacroAssembler* masm) const {
|
|
||||||
// Restore registers in reverse order due to the stack.
|
|
||||||
for (int i = RegisterAllocator::kNumRegisters - 1; i >= 0; i--) {
|
|
||||||
int action = registers_[i];
|
|
||||||
if (action == kPush) {
|
|
||||||
__ pop(RegisterAllocator::ToRegister(i));
|
|
||||||
} else if (action != kIgnore) {
|
|
||||||
action &= ~kSyncedFlag;
|
|
||||||
__ ldr(RegisterAllocator::ToRegister(i), MemOperand(fp, action));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
#undef __
|
|
||||||
#define __ ACCESS_MASM(masm_)
|
#define __ ACCESS_MASM(masm_)
|
||||||
|
|
||||||
// -------------------------------------------------------------------------
|
// -------------------------------------------------------------------------
|
||||||
// Platform-specific DeferredCode functions.
|
// Platform-specific DeferredCode functions.
|
||||||
|
|
||||||
void DeferredCode::SaveRegisters() {
|
void DeferredCode::SaveRegisters() {
|
||||||
frame_state_.Save(masm_);
|
// On ARM you either have a completely spilled frame or you
|
||||||
|
// handle it yourself, but at the moment there's no automation
|
||||||
|
// of registers and deferred code.
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void DeferredCode::RestoreRegisters() {
|
void DeferredCode::RestoreRegisters() {
|
||||||
frame_state_.Restore(masm_);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -117,12 +86,11 @@ void DeferredCode::RestoreRegisters() {
|
|||||||
// Platform-specific RuntimeCallHelper functions.
|
// Platform-specific RuntimeCallHelper functions.
|
||||||
|
|
||||||
void VirtualFrameRuntimeCallHelper::BeforeCall(MacroAssembler* masm) const {
|
void VirtualFrameRuntimeCallHelper::BeforeCall(MacroAssembler* masm) const {
|
||||||
frame_state_->Save(masm);
|
frame_state_->frame()->AssertIsSpilled();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void VirtualFrameRuntimeCallHelper::AfterCall(MacroAssembler* masm) const {
|
void VirtualFrameRuntimeCallHelper::AfterCall(MacroAssembler* masm) const {
|
||||||
frame_state_->Restore(masm);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -3455,7 +3423,6 @@ void CodeGenerator::EmitNamedPropertyAssignment(Assignment* node) {
|
|||||||
frame_->Dup();
|
frame_->Dup();
|
||||||
}
|
}
|
||||||
EmitNamedLoad(name, var != NULL);
|
EmitNamedLoad(name, var != NULL);
|
||||||
frame_->EmitPush(r0);
|
|
||||||
|
|
||||||
// Perform the binary operation.
|
// Perform the binary operation.
|
||||||
Literal* literal = node->value()->AsLiteral();
|
Literal* literal = node->value()->AsLiteral();
|
||||||
@ -5613,11 +5580,19 @@ class DeferredReferenceGetNamedValue: public DeferredCode {
|
|||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
// Convention for this is that on entry the receiver is in a register that
|
||||||
|
// is not used by the stack. On exit the answer is found in that same
|
||||||
|
// register and the stack has the same height.
|
||||||
void DeferredReferenceGetNamedValue::Generate() {
|
void DeferredReferenceGetNamedValue::Generate() {
|
||||||
ASSERT(receiver_.is(r0) || receiver_.is(r1));
|
#ifdef DEBUG
|
||||||
|
int expected_height = frame_state()->frame()->height();
|
||||||
|
#endif
|
||||||
|
VirtualFrame copied_frame(*frame_state()->frame());
|
||||||
|
copied_frame.SpillAll();
|
||||||
|
|
||||||
Register scratch1 = VirtualFrame::scratch0();
|
Register scratch1 = VirtualFrame::scratch0();
|
||||||
Register scratch2 = VirtualFrame::scratch1();
|
Register scratch2 = VirtualFrame::scratch1();
|
||||||
|
ASSERT(!receiver_.is(scratch1) && !receiver_.is(scratch2));
|
||||||
__ DecrementCounter(&Counters::named_load_inline, 1, scratch1, scratch2);
|
__ DecrementCounter(&Counters::named_load_inline, 1, scratch1, scratch2);
|
||||||
__ IncrementCounter(&Counters::named_load_inline_miss, 1, scratch1, scratch2);
|
__ IncrementCounter(&Counters::named_load_inline_miss, 1, scratch1, scratch2);
|
||||||
|
|
||||||
@ -5633,11 +5608,23 @@ void DeferredReferenceGetNamedValue::Generate() {
|
|||||||
// in-object has been inlined.
|
// in-object has been inlined.
|
||||||
__ nop(PROPERTY_ACCESS_INLINED);
|
__ nop(PROPERTY_ACCESS_INLINED);
|
||||||
|
|
||||||
|
// At this point the answer is in r0. We move it to the expected register
|
||||||
|
// if necessary.
|
||||||
|
__ Move(receiver_, r0);
|
||||||
|
|
||||||
|
// Now go back to the frame that we entered with. This will not overwrite
|
||||||
|
// the receiver register since that register was not in use when we came
|
||||||
|
// in. The instructions emitted by this merge are skipped over by the
|
||||||
|
// inline load patching mechanism when looking for the branch instruction
|
||||||
|
// that tells it where the code to patch is.
|
||||||
|
copied_frame.MergeTo(frame_state()->frame());
|
||||||
|
|
||||||
// Block the constant pool for one more instruction after leaving this
|
// Block the constant pool for one more instruction after leaving this
|
||||||
// constant pool block scope to include the branch instruction ending the
|
// constant pool block scope to include the branch instruction ending the
|
||||||
// deferred code.
|
// deferred code.
|
||||||
__ BlockConstPoolFor(1);
|
__ BlockConstPoolFor(1);
|
||||||
}
|
}
|
||||||
|
ASSERT_EQ(expected_height, frame_state()->frame()->height());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -5738,6 +5725,7 @@ void DeferredReferenceSetKeyedValue::Generate() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// Consumes the top of stack (the receiver) and pushes the result instead.
|
||||||
void CodeGenerator::EmitNamedLoad(Handle<String> name, bool is_contextual) {
|
void CodeGenerator::EmitNamedLoad(Handle<String> name, bool is_contextual) {
|
||||||
if (is_contextual || scope()->is_global_scope() || loop_nesting() == 0) {
|
if (is_contextual || scope()->is_global_scope() || loop_nesting() == 0) {
|
||||||
Comment cmnt(masm(), "[ Load from named Property");
|
Comment cmnt(masm(), "[ Load from named Property");
|
||||||
@ -5746,6 +5734,7 @@ void CodeGenerator::EmitNamedLoad(Handle<String> name, bool is_contextual) {
|
|||||||
is_contextual
|
is_contextual
|
||||||
? RelocInfo::CODE_TARGET_CONTEXT
|
? RelocInfo::CODE_TARGET_CONTEXT
|
||||||
: RelocInfo::CODE_TARGET);
|
: RelocInfo::CODE_TARGET);
|
||||||
|
frame_->EmitPush(r0); // Push answer.
|
||||||
} else {
|
} else {
|
||||||
// Inline the in-object property case.
|
// Inline the in-object property case.
|
||||||
Comment cmnt(masm(), "[ Inlined named property load");
|
Comment cmnt(masm(), "[ Inlined named property load");
|
||||||
@ -5762,7 +5751,6 @@ void CodeGenerator::EmitNamedLoad(Handle<String> name, bool is_contextual) {
|
|||||||
|
|
||||||
// Load the receiver from the stack.
|
// Load the receiver from the stack.
|
||||||
Register receiver = frame_->PopToRegister();
|
Register receiver = frame_->PopToRegister();
|
||||||
VirtualFrame::SpilledScope spilled(frame_);
|
|
||||||
|
|
||||||
DeferredReferenceGetNamedValue* deferred =
|
DeferredReferenceGetNamedValue* deferred =
|
||||||
new DeferredReferenceGetNamedValue(receiver, name);
|
new DeferredReferenceGetNamedValue(receiver, name);
|
||||||
@ -5778,16 +5766,19 @@ void CodeGenerator::EmitNamedLoad(Handle<String> name, bool is_contextual) {
|
|||||||
__ tst(receiver, Operand(kSmiTagMask));
|
__ tst(receiver, Operand(kSmiTagMask));
|
||||||
deferred->Branch(eq);
|
deferred->Branch(eq);
|
||||||
|
|
||||||
|
Register scratch = VirtualFrame::scratch0();
|
||||||
|
Register scratch2 = VirtualFrame::scratch1();
|
||||||
|
|
||||||
// Check the map. The null map used below is patched by the inline cache
|
// Check the map. The null map used below is patched by the inline cache
|
||||||
// code.
|
// code. Therefore we can't use a LoadRoot call.
|
||||||
__ ldr(r2, FieldMemOperand(receiver, HeapObject::kMapOffset));
|
__ ldr(scratch, FieldMemOperand(receiver, HeapObject::kMapOffset));
|
||||||
__ mov(r3, Operand(Factory::null_value()));
|
__ mov(scratch2, Operand(Factory::null_value()));
|
||||||
__ cmp(r2, r3);
|
__ cmp(scratch, scratch2);
|
||||||
deferred->Branch(ne);
|
deferred->Branch(ne);
|
||||||
|
|
||||||
// Initially use an invalid index. The index will be patched by the
|
// Initially use an invalid index. The index will be patched by the
|
||||||
// inline cache code.
|
// inline cache code.
|
||||||
__ ldr(r0, MemOperand(receiver, 0));
|
__ ldr(receiver, MemOperand(receiver, 0));
|
||||||
|
|
||||||
// Make sure that the expected number of instructions are generated.
|
// Make sure that the expected number of instructions are generated.
|
||||||
ASSERT_EQ(kInlinedNamedLoadInstructions,
|
ASSERT_EQ(kInlinedNamedLoadInstructions,
|
||||||
@ -5795,6 +5786,9 @@ void CodeGenerator::EmitNamedLoad(Handle<String> name, bool is_contextual) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
deferred->BindExit();
|
deferred->BindExit();
|
||||||
|
// At this point the receiver register has the result, either from the
|
||||||
|
// deferred code or from the inlined code.
|
||||||
|
frame_->EmitPush(receiver);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -6010,6 +6004,27 @@ Handle<String> Reference::GetName() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void Reference::DupIfPersist() {
|
||||||
|
if (persist_after_get_) {
|
||||||
|
switch (type_) {
|
||||||
|
case KEYED:
|
||||||
|
cgen_->frame()->Dup2();
|
||||||
|
break;
|
||||||
|
case NAMED:
|
||||||
|
cgen_->frame()->Dup();
|
||||||
|
// Fall through.
|
||||||
|
case UNLOADED:
|
||||||
|
case ILLEGAL:
|
||||||
|
case SLOT:
|
||||||
|
// Do nothing.
|
||||||
|
;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
set_unloaded();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
void Reference::GetValue() {
|
void Reference::GetValue() {
|
||||||
ASSERT(cgen_->HasValidEntryRegisters());
|
ASSERT(cgen_->HasValidEntryRegisters());
|
||||||
ASSERT(!is_illegal());
|
ASSERT(!is_illegal());
|
||||||
@ -6025,10 +6040,8 @@ void Reference::GetValue() {
|
|||||||
Comment cmnt(masm, "[ Load from Slot");
|
Comment cmnt(masm, "[ Load from Slot");
|
||||||
Slot* slot = expression_->AsVariableProxy()->AsVariable()->slot();
|
Slot* slot = expression_->AsVariableProxy()->AsVariable()->slot();
|
||||||
ASSERT(slot != NULL);
|
ASSERT(slot != NULL);
|
||||||
|
DupIfPersist();
|
||||||
cgen_->LoadFromSlotCheckForArguments(slot, NOT_INSIDE_TYPEOF);
|
cgen_->LoadFromSlotCheckForArguments(slot, NOT_INSIDE_TYPEOF);
|
||||||
if (!persist_after_get_) {
|
|
||||||
cgen_->UnloadReference(this);
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -6036,23 +6049,17 @@ void Reference::GetValue() {
|
|||||||
Variable* var = expression_->AsVariableProxy()->AsVariable();
|
Variable* var = expression_->AsVariableProxy()->AsVariable();
|
||||||
bool is_global = var != NULL;
|
bool is_global = var != NULL;
|
||||||
ASSERT(!is_global || var->is_global());
|
ASSERT(!is_global || var->is_global());
|
||||||
if (persist_after_get_) {
|
Handle<String> name = GetName();
|
||||||
cgen_->frame()->Dup();
|
DupIfPersist();
|
||||||
}
|
cgen_->EmitNamedLoad(name, is_global);
|
||||||
cgen_->EmitNamedLoad(GetName(), is_global);
|
|
||||||
cgen_->frame()->EmitPush(r0);
|
|
||||||
if (!persist_after_get_) set_unloaded();
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
case KEYED: {
|
case KEYED: {
|
||||||
ASSERT(property != NULL);
|
ASSERT(property != NULL);
|
||||||
if (persist_after_get_) {
|
DupIfPersist();
|
||||||
cgen_->frame()->Dup2();
|
|
||||||
}
|
|
||||||
cgen_->EmitKeyedLoad();
|
cgen_->EmitKeyedLoad();
|
||||||
cgen_->frame()->EmitPush(r0);
|
cgen_->frame()->EmitPush(r0);
|
||||||
if (!persist_after_get_) set_unloaded();
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -101,6 +101,11 @@ class Reference BASE_EMBEDDED {
|
|||||||
// is popped from beneath it (unloaded).
|
// is popped from beneath it (unloaded).
|
||||||
void SetValue(InitState init_state);
|
void SetValue(InitState init_state);
|
||||||
|
|
||||||
|
// This is in preparation for something that uses the reference on the stack.
|
||||||
|
// If we need this reference afterwards get then dup it now. Otherwise mark
|
||||||
|
// it as used.
|
||||||
|
inline void DupIfPersist();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
CodeGenerator* cgen_;
|
CodeGenerator* cgen_;
|
||||||
Expression* expression_;
|
Expression* expression_;
|
||||||
|
@ -579,7 +579,13 @@ static inline bool IsInlinedICSite(Address address,
|
|||||||
}
|
}
|
||||||
Address address_after_nop = address_after_call + Assembler::kInstrSize;
|
Address address_after_nop = address_after_call + Assembler::kInstrSize;
|
||||||
Instr instr_after_nop = Assembler::instr_at(address_after_nop);
|
Instr instr_after_nop = Assembler::instr_at(address_after_nop);
|
||||||
ASSERT(Assembler::IsBranch(instr_after_nop));
|
// There may be some reg-reg move and frame merging code to skip over before
|
||||||
|
// the branch back from the DeferredReferenceGetKeyedValue code to the inlined
|
||||||
|
// code.
|
||||||
|
while (!Assembler::IsBranch(instr_after_nop)) {
|
||||||
|
address_after_nop += Assembler::kInstrSize;
|
||||||
|
instr_after_nop = Assembler::instr_at(address_after_nop);
|
||||||
|
}
|
||||||
|
|
||||||
// Find the end of the inlined code for handling the load.
|
// Find the end of the inlined code for handling the load.
|
||||||
int b_offset =
|
int b_offset =
|
||||||
|
@ -69,18 +69,15 @@ void JumpTarget::DoBranch(Condition cc, Hint ignored) {
|
|||||||
if (entry_frame_set_) {
|
if (entry_frame_set_) {
|
||||||
// Backward branch. We have an expected frame to merge to on the
|
// Backward branch. We have an expected frame to merge to on the
|
||||||
// backward edge.
|
// backward edge.
|
||||||
if (cc == al) {
|
cgen()->frame()->MergeTo(&entry_frame_, cc);
|
||||||
cgen()->frame()->MergeTo(&entry_frame_);
|
|
||||||
} else {
|
|
||||||
// We can't do conditional merges yet so you have to ensure that all
|
|
||||||
// conditional branches to the JumpTarget have the same virtual frame.
|
|
||||||
ASSERT(cgen()->frame()->Equals(&entry_frame_));
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
// Clone the current frame to use as the expected one at the target.
|
// Clone the current frame to use as the expected one at the target.
|
||||||
set_entry_frame(cgen()->frame());
|
set_entry_frame(cgen()->frame());
|
||||||
}
|
}
|
||||||
__ b(cc, &entry_label_);
|
__ b(cc, &entry_label_);
|
||||||
|
if (cc == al) {
|
||||||
|
cgen()->DeleteFrame();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -183,15 +183,18 @@ void MacroAssembler::Drop(int count, Condition cond) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void MacroAssembler::Swap(Register reg1, Register reg2, Register scratch) {
|
void MacroAssembler::Swap(Register reg1,
|
||||||
|
Register reg2,
|
||||||
|
Register scratch,
|
||||||
|
Condition cond) {
|
||||||
if (scratch.is(no_reg)) {
|
if (scratch.is(no_reg)) {
|
||||||
eor(reg1, reg1, Operand(reg2));
|
eor(reg1, reg1, Operand(reg2), LeaveCC, cond);
|
||||||
eor(reg2, reg2, Operand(reg1));
|
eor(reg2, reg2, Operand(reg1), LeaveCC, cond);
|
||||||
eor(reg1, reg1, Operand(reg2));
|
eor(reg1, reg1, Operand(reg2), LeaveCC, cond);
|
||||||
} else {
|
} else {
|
||||||
mov(scratch, reg1);
|
mov(scratch, reg1, LeaveCC, cond);
|
||||||
mov(reg1, reg2);
|
mov(reg1, reg2, LeaveCC, cond);
|
||||||
mov(reg2, scratch);
|
mov(reg2, scratch, LeaveCC, cond);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -88,7 +88,10 @@ class MacroAssembler: public Assembler {
|
|||||||
|
|
||||||
// Swap two registers. If the scratch register is omitted then a slightly
|
// Swap two registers. If the scratch register is omitted then a slightly
|
||||||
// less efficient form using xor instead of mov is emitted.
|
// less efficient form using xor instead of mov is emitted.
|
||||||
void Swap(Register reg1, Register reg2, Register scratch = no_reg);
|
void Swap(Register reg1,
|
||||||
|
Register reg2,
|
||||||
|
Register scratch = no_reg,
|
||||||
|
Condition cond = al);
|
||||||
|
|
||||||
void Call(Label* target);
|
void Call(Label* target);
|
||||||
void Move(Register dst, Handle<Object> value);
|
void Move(Register dst, Handle<Object> value);
|
||||||
|
@ -72,90 +72,90 @@ void VirtualFrame::PopToR0() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void VirtualFrame::MergeTo(VirtualFrame* expected) {
|
void VirtualFrame::MergeTo(const VirtualFrame* expected, Condition cond) {
|
||||||
if (Equals(expected)) return;
|
if (Equals(expected)) return;
|
||||||
MergeTOSTo(expected->top_of_stack_state_);
|
MergeTOSTo(expected->top_of_stack_state_, cond);
|
||||||
ASSERT(register_allocation_map_ == expected->register_allocation_map_);
|
ASSERT(register_allocation_map_ == expected->register_allocation_map_);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void VirtualFrame::MergeTOSTo(
|
void VirtualFrame::MergeTOSTo(
|
||||||
VirtualFrame::TopOfStack expected_top_of_stack_state) {
|
VirtualFrame::TopOfStack expected_top_of_stack_state, Condition cond) {
|
||||||
#define CASE_NUMBER(a, b) ((a) * TOS_STATES + (b))
|
#define CASE_NUMBER(a, b) ((a) * TOS_STATES + (b))
|
||||||
switch (CASE_NUMBER(top_of_stack_state_, expected_top_of_stack_state)) {
|
switch (CASE_NUMBER(top_of_stack_state_, expected_top_of_stack_state)) {
|
||||||
case CASE_NUMBER(NO_TOS_REGISTERS, NO_TOS_REGISTERS):
|
case CASE_NUMBER(NO_TOS_REGISTERS, NO_TOS_REGISTERS):
|
||||||
break;
|
break;
|
||||||
case CASE_NUMBER(NO_TOS_REGISTERS, R0_TOS):
|
case CASE_NUMBER(NO_TOS_REGISTERS, R0_TOS):
|
||||||
__ pop(r0);
|
__ pop(r0, cond);
|
||||||
break;
|
break;
|
||||||
case CASE_NUMBER(NO_TOS_REGISTERS, R1_TOS):
|
case CASE_NUMBER(NO_TOS_REGISTERS, R1_TOS):
|
||||||
__ pop(r1);
|
__ pop(r1, cond);
|
||||||
break;
|
break;
|
||||||
case CASE_NUMBER(NO_TOS_REGISTERS, R0_R1_TOS):
|
case CASE_NUMBER(NO_TOS_REGISTERS, R0_R1_TOS):
|
||||||
__ pop(r0);
|
__ pop(r0, cond);
|
||||||
__ pop(r1);
|
__ pop(r1, cond);
|
||||||
break;
|
break;
|
||||||
case CASE_NUMBER(NO_TOS_REGISTERS, R1_R0_TOS):
|
case CASE_NUMBER(NO_TOS_REGISTERS, R1_R0_TOS):
|
||||||
__ pop(r1);
|
__ pop(r1, cond);
|
||||||
__ pop(r0);
|
__ pop(r0, cond);
|
||||||
break;
|
break;
|
||||||
case CASE_NUMBER(R0_TOS, NO_TOS_REGISTERS):
|
case CASE_NUMBER(R0_TOS, NO_TOS_REGISTERS):
|
||||||
__ push(r0);
|
__ push(r0, cond);
|
||||||
break;
|
break;
|
||||||
case CASE_NUMBER(R0_TOS, R0_TOS):
|
case CASE_NUMBER(R0_TOS, R0_TOS):
|
||||||
break;
|
break;
|
||||||
case CASE_NUMBER(R0_TOS, R1_TOS):
|
case CASE_NUMBER(R0_TOS, R1_TOS):
|
||||||
__ mov(r1, r0);
|
__ mov(r1, r0, LeaveCC, cond);
|
||||||
break;
|
break;
|
||||||
case CASE_NUMBER(R0_TOS, R0_R1_TOS):
|
case CASE_NUMBER(R0_TOS, R0_R1_TOS):
|
||||||
__ pop(r1);
|
__ pop(r1, cond);
|
||||||
break;
|
break;
|
||||||
case CASE_NUMBER(R0_TOS, R1_R0_TOS):
|
case CASE_NUMBER(R0_TOS, R1_R0_TOS):
|
||||||
__ mov(r1, r0);
|
__ mov(r1, r0, LeaveCC, cond);
|
||||||
__ pop(r0);
|
__ pop(r0, cond);
|
||||||
break;
|
break;
|
||||||
case CASE_NUMBER(R1_TOS, NO_TOS_REGISTERS):
|
case CASE_NUMBER(R1_TOS, NO_TOS_REGISTERS):
|
||||||
__ push(r1);
|
__ push(r1, cond);
|
||||||
break;
|
break;
|
||||||
case CASE_NUMBER(R1_TOS, R0_TOS):
|
case CASE_NUMBER(R1_TOS, R0_TOS):
|
||||||
__ mov(r0, r1);
|
__ mov(r0, r1, LeaveCC, cond);
|
||||||
break;
|
break;
|
||||||
case CASE_NUMBER(R1_TOS, R1_TOS):
|
case CASE_NUMBER(R1_TOS, R1_TOS):
|
||||||
break;
|
break;
|
||||||
case CASE_NUMBER(R1_TOS, R0_R1_TOS):
|
case CASE_NUMBER(R1_TOS, R0_R1_TOS):
|
||||||
__ mov(r0, r1);
|
__ mov(r0, r1, LeaveCC, cond);
|
||||||
__ pop(r1);
|
__ pop(r1, cond);
|
||||||
break;
|
break;
|
||||||
case CASE_NUMBER(R1_TOS, R1_R0_TOS):
|
case CASE_NUMBER(R1_TOS, R1_R0_TOS):
|
||||||
__ pop(r0);
|
__ pop(r0, cond);
|
||||||
break;
|
break;
|
||||||
case CASE_NUMBER(R0_R1_TOS, NO_TOS_REGISTERS):
|
case CASE_NUMBER(R0_R1_TOS, NO_TOS_REGISTERS):
|
||||||
__ Push(r1, r0);
|
__ Push(r1, r0, cond);
|
||||||
break;
|
break;
|
||||||
case CASE_NUMBER(R0_R1_TOS, R0_TOS):
|
case CASE_NUMBER(R0_R1_TOS, R0_TOS):
|
||||||
__ push(r1);
|
__ push(r1, cond);
|
||||||
break;
|
break;
|
||||||
case CASE_NUMBER(R0_R1_TOS, R1_TOS):
|
case CASE_NUMBER(R0_R1_TOS, R1_TOS):
|
||||||
__ push(r1);
|
__ push(r1, cond);
|
||||||
__ mov(r1, r0);
|
__ mov(r1, r0, LeaveCC, cond);
|
||||||
break;
|
break;
|
||||||
case CASE_NUMBER(R0_R1_TOS, R0_R1_TOS):
|
case CASE_NUMBER(R0_R1_TOS, R0_R1_TOS):
|
||||||
break;
|
break;
|
||||||
case CASE_NUMBER(R0_R1_TOS, R1_R0_TOS):
|
case CASE_NUMBER(R0_R1_TOS, R1_R0_TOS):
|
||||||
__ Swap(r0, r1, ip);
|
__ Swap(r0, r1, ip, cond);
|
||||||
break;
|
break;
|
||||||
case CASE_NUMBER(R1_R0_TOS, NO_TOS_REGISTERS):
|
case CASE_NUMBER(R1_R0_TOS, NO_TOS_REGISTERS):
|
||||||
__ Push(r0, r1);
|
__ Push(r0, r1, cond);
|
||||||
break;
|
break;
|
||||||
case CASE_NUMBER(R1_R0_TOS, R0_TOS):
|
case CASE_NUMBER(R1_R0_TOS, R0_TOS):
|
||||||
__ push(r0);
|
__ push(r0, cond);
|
||||||
__ mov(r0, r1);
|
__ mov(r0, r1, LeaveCC, cond);
|
||||||
break;
|
break;
|
||||||
case CASE_NUMBER(R1_R0_TOS, R1_TOS):
|
case CASE_NUMBER(R1_R0_TOS, R1_TOS):
|
||||||
__ push(r0);
|
__ push(r0, cond);
|
||||||
break;
|
break;
|
||||||
case CASE_NUMBER(R1_R0_TOS, R0_R1_TOS):
|
case CASE_NUMBER(R1_R0_TOS, R0_R1_TOS):
|
||||||
__ Swap(r0, r1, ip);
|
__ Swap(r0, r1, ip, cond);
|
||||||
break;
|
break;
|
||||||
case CASE_NUMBER(R1_R0_TOS, R1_R0_TOS):
|
case CASE_NUMBER(R1_R0_TOS, R1_R0_TOS):
|
||||||
break;
|
break;
|
||||||
@ -163,7 +163,16 @@ void VirtualFrame::MergeTOSTo(
|
|||||||
UNREACHABLE();
|
UNREACHABLE();
|
||||||
#undef CASE_NUMBER
|
#undef CASE_NUMBER
|
||||||
}
|
}
|
||||||
|
// A conditional merge will be followed by a conditional branch and the
|
||||||
|
// fall-through code will have an unchanged virtual frame state. If the
|
||||||
|
// merge is unconditional ('al'ways) then it might be followed by a fall
|
||||||
|
// through. We need to update the virtual frame state to match the code we
|
||||||
|
// are falling into. The final case is an unconditional merge followed by an
|
||||||
|
// unconditional branch, in which case it doesn't matter what we do to the
|
||||||
|
// virtual frame state, because the virtual frame will be invalidated.
|
||||||
|
if (cond == al) {
|
||||||
top_of_stack_state_ = expected_top_of_stack_state;
|
top_of_stack_state_ = expected_top_of_stack_state;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -107,14 +107,14 @@ class VirtualFrame : public ZoneObject {
|
|||||||
// Construct a virtual frame as a clone of an existing one.
|
// Construct a virtual frame as a clone of an existing one.
|
||||||
explicit inline VirtualFrame(VirtualFrame* original);
|
explicit inline VirtualFrame(VirtualFrame* original);
|
||||||
|
|
||||||
inline CodeGenerator* cgen();
|
inline CodeGenerator* cgen() const;
|
||||||
inline MacroAssembler* masm();
|
inline MacroAssembler* masm();
|
||||||
|
|
||||||
// The number of elements on the virtual frame.
|
// The number of elements on the virtual frame.
|
||||||
int element_count() { return element_count_; }
|
int element_count() const { return element_count_; }
|
||||||
|
|
||||||
// The height of the virtual expression stack.
|
// The height of the virtual expression stack.
|
||||||
inline int height();
|
inline int height() const;
|
||||||
|
|
||||||
bool is_used(int num) {
|
bool is_used(int num) {
|
||||||
switch (num) {
|
switch (num) {
|
||||||
@ -162,7 +162,7 @@ class VirtualFrame : public ZoneObject {
|
|||||||
// Spill all values from the frame to memory.
|
// Spill all values from the frame to memory.
|
||||||
void SpillAll();
|
void SpillAll();
|
||||||
|
|
||||||
void AssertIsSpilled() {
|
void AssertIsSpilled() const {
|
||||||
ASSERT(top_of_stack_state_ == NO_TOS_REGISTERS);
|
ASSERT(top_of_stack_state_ == NO_TOS_REGISTERS);
|
||||||
ASSERT(register_allocation_map_ == 0);
|
ASSERT(register_allocation_map_ == 0);
|
||||||
}
|
}
|
||||||
@ -184,7 +184,7 @@ class VirtualFrame : public ZoneObject {
|
|||||||
// Make this virtual frame have a state identical to an expected virtual
|
// Make this virtual frame have a state identical to an expected virtual
|
||||||
// frame. As a side effect, code may be emitted to make this frame match
|
// frame. As a side effect, code may be emitted to make this frame match
|
||||||
// the expected one.
|
// the expected one.
|
||||||
void MergeTo(VirtualFrame* expected);
|
void MergeTo(const VirtualFrame* expected, Condition cond = al);
|
||||||
|
|
||||||
// Detach a frame from its code generator, perhaps temporarily. This
|
// Detach a frame from its code generator, perhaps temporarily. This
|
||||||
// tells the register allocator that it is free to use frame-internal
|
// tells the register allocator that it is free to use frame-internal
|
||||||
@ -426,13 +426,13 @@ class VirtualFrame : public ZoneObject {
|
|||||||
int stack_pointer() { return element_count_ - 1; }
|
int stack_pointer() { return element_count_ - 1; }
|
||||||
|
|
||||||
// The number of frame-allocated locals and parameters respectively.
|
// The number of frame-allocated locals and parameters respectively.
|
||||||
inline int parameter_count();
|
inline int parameter_count() const;
|
||||||
inline int local_count();
|
inline int local_count() const;
|
||||||
|
|
||||||
// The index of the element that is at the processor's frame pointer
|
// The index of the element that is at the processor's frame pointer
|
||||||
// (the fp register). The parameters, receiver, function, and context
|
// (the fp register). The parameters, receiver, function, and context
|
||||||
// are below the frame pointer.
|
// are below the frame pointer.
|
||||||
inline int frame_pointer();
|
inline int frame_pointer() const;
|
||||||
|
|
||||||
// The index of the first parameter. The receiver lies below the first
|
// The index of the first parameter. The receiver lies below the first
|
||||||
// parameter.
|
// parameter.
|
||||||
@ -448,10 +448,10 @@ class VirtualFrame : public ZoneObject {
|
|||||||
|
|
||||||
// The index of the first local. Between the frame pointer and the
|
// The index of the first local. Between the frame pointer and the
|
||||||
// locals lies the return address.
|
// locals lies the return address.
|
||||||
inline int local0_index();
|
inline int local0_index() const;
|
||||||
|
|
||||||
// The index of the base of the expression stack.
|
// The index of the base of the expression stack.
|
||||||
inline int expression_base_index();
|
inline int expression_base_index() const;
|
||||||
|
|
||||||
// Convert a frame index into a frame pointer relative offset into the
|
// Convert a frame index into a frame pointer relative offset into the
|
||||||
// actual stack.
|
// actual stack.
|
||||||
@ -469,9 +469,9 @@ class VirtualFrame : public ZoneObject {
|
|||||||
|
|
||||||
// Emit instructions to get the top of stack state from where we are to where
|
// Emit instructions to get the top of stack state from where we are to where
|
||||||
// we want to be.
|
// we want to be.
|
||||||
void MergeTOSTo(TopOfStack expected_state);
|
void MergeTOSTo(TopOfStack expected_state, Condition cond);
|
||||||
|
|
||||||
inline bool Equals(VirtualFrame* other);
|
inline bool Equals(const VirtualFrame* other);
|
||||||
|
|
||||||
friend class JumpTarget;
|
friend class JumpTarget;
|
||||||
};
|
};
|
||||||
|
@ -180,6 +180,8 @@ class CodeGeneratorScope BASE_EMBEDDED {
|
|||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
#if V8_TARGET_ARCH_IA32 || V8_TARGET_ARCH_X64
|
||||||
|
|
||||||
// State of used registers in a virtual frame.
|
// State of used registers in a virtual frame.
|
||||||
class FrameRegisterState {
|
class FrameRegisterState {
|
||||||
public:
|
public:
|
||||||
@ -203,14 +205,28 @@ class FrameRegisterState {
|
|||||||
// it should fit in the low zero bits of a valid offset.
|
// it should fit in the low zero bits of a valid offset.
|
||||||
static const int kSyncedFlag = 2;
|
static const int kSyncedFlag = 2;
|
||||||
|
|
||||||
// C++ doesn't allow zero length arrays, so we make the array length 1 even
|
int registers_[RegisterAllocator::kNumRegisters];
|
||||||
// if we don't need it.
|
|
||||||
static const int kRegistersArrayLength =
|
|
||||||
(RegisterAllocator::kNumRegisters == 0) ?
|
|
||||||
1 : RegisterAllocator::kNumRegisters;
|
|
||||||
int registers_[kRegistersArrayLength];
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
#elif V8_TARGET_ARCH_ARM || V8_TARGET_ARCH_MIPS
|
||||||
|
|
||||||
|
|
||||||
|
class FrameRegisterState {
|
||||||
|
public:
|
||||||
|
inline FrameRegisterState(VirtualFrame frame) : frame_(frame) { }
|
||||||
|
|
||||||
|
inline const VirtualFrame* frame() const { return &frame_; }
|
||||||
|
|
||||||
|
private:
|
||||||
|
VirtualFrame frame_;
|
||||||
|
};
|
||||||
|
|
||||||
|
#else
|
||||||
|
|
||||||
|
#error Unsupported target architecture.
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
// Helper interface to prepare to/restore after making runtime calls.
|
// Helper interface to prepare to/restore after making runtime calls.
|
||||||
class RuntimeCallHelper {
|
class RuntimeCallHelper {
|
||||||
|
@ -34,17 +34,11 @@ namespace v8 {
|
|||||||
namespace internal {
|
namespace internal {
|
||||||
|
|
||||||
|
|
||||||
FrameRegisterState::FrameRegisterState(VirtualFrame* frame) {
|
|
||||||
// Nothing to do when register allocation is not supported.
|
|
||||||
ASSERT(RegisterAllocator::kNumRegisters == 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
DeferredCode::DeferredCode()
|
DeferredCode::DeferredCode()
|
||||||
: masm_(CodeGeneratorScope::Current()->masm()),
|
: masm_(CodeGeneratorScope::Current()->masm()),
|
||||||
statement_position_(masm_->current_statement_position()),
|
statement_position_(masm_->current_statement_position()),
|
||||||
position_(masm_->current_position()),
|
position_(masm_->current_position()),
|
||||||
frame_state_(CodeGeneratorScope::Current()->frame()) {
|
frame_state_(*CodeGeneratorScope::Current()->frame()) {
|
||||||
ASSERT(statement_position_ != RelocInfo::kNoPosition);
|
ASSERT(statement_position_ != RelocInfo::kNoPosition);
|
||||||
ASSERT(position_ != RelocInfo::kNoPosition);
|
ASSERT(position_ != RelocInfo::kNoPosition);
|
||||||
|
|
||||||
@ -52,7 +46,6 @@ DeferredCode::DeferredCode()
|
|||||||
|
|
||||||
#ifdef DEBUG
|
#ifdef DEBUG
|
||||||
comment_ = "";
|
comment_ = "";
|
||||||
CodeGeneratorScope::Current()->frame()->AssertIsSpilled();
|
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -60,7 +60,7 @@ VirtualFrame::VirtualFrame(VirtualFrame* original)
|
|||||||
register_allocation_map_(original->register_allocation_map_) { }
|
register_allocation_map_(original->register_allocation_map_) { }
|
||||||
|
|
||||||
|
|
||||||
bool VirtualFrame::Equals(VirtualFrame* other) {
|
bool VirtualFrame::Equals(const VirtualFrame* other) {
|
||||||
ASSERT(element_count() == other->element_count());
|
ASSERT(element_count() == other->element_count());
|
||||||
if (top_of_stack_state_ != other->top_of_stack_state_) return false;
|
if (top_of_stack_state_ != other->top_of_stack_state_) return false;
|
||||||
if (register_allocation_map_ != other->register_allocation_map_) return false;
|
if (register_allocation_map_ != other->register_allocation_map_) return false;
|
||||||
@ -99,7 +99,9 @@ VirtualFrame::RegisterAllocationScope::~RegisterAllocationScope() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
CodeGenerator* VirtualFrame::cgen() { return CodeGeneratorScope::Current(); }
|
CodeGenerator* VirtualFrame::cgen() const {
|
||||||
|
return CodeGeneratorScope::Current();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
MacroAssembler* VirtualFrame::masm() { return cgen()->masm(); }
|
MacroAssembler* VirtualFrame::masm() { return cgen()->masm(); }
|
||||||
@ -112,15 +114,17 @@ void VirtualFrame::CallStub(CodeStub* stub, int arg_count) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
int VirtualFrame::parameter_count() {
|
int VirtualFrame::parameter_count() const {
|
||||||
return cgen()->scope()->num_parameters();
|
return cgen()->scope()->num_parameters();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
int VirtualFrame::local_count() { return cgen()->scope()->num_stack_slots(); }
|
int VirtualFrame::local_count() const {
|
||||||
|
return cgen()->scope()->num_stack_slots();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
int VirtualFrame::frame_pointer() { return parameter_count() + 3; }
|
int VirtualFrame::frame_pointer() const { return parameter_count() + 3; }
|
||||||
|
|
||||||
|
|
||||||
int VirtualFrame::context_index() { return frame_pointer() - 1; }
|
int VirtualFrame::context_index() { return frame_pointer() - 1; }
|
||||||
@ -129,7 +133,7 @@ int VirtualFrame::context_index() { return frame_pointer() - 1; }
|
|||||||
int VirtualFrame::function_index() { return frame_pointer() - 2; }
|
int VirtualFrame::function_index() { return frame_pointer() - 2; }
|
||||||
|
|
||||||
|
|
||||||
int VirtualFrame::local0_index() { return frame_pointer() + 2; }
|
int VirtualFrame::local0_index() const { return frame_pointer() + 2; }
|
||||||
|
|
||||||
|
|
||||||
int VirtualFrame::fp_relative(int index) {
|
int VirtualFrame::fp_relative(int index) {
|
||||||
@ -139,12 +143,12 @@ int VirtualFrame::fp_relative(int index) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
int VirtualFrame::expression_base_index() {
|
int VirtualFrame::expression_base_index() const {
|
||||||
return local0_index() + local_count();
|
return local0_index() + local_count();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
int VirtualFrame::height() {
|
int VirtualFrame::height() const {
|
||||||
return element_count() - expression_base_index();
|
return element_count() - expression_base_index();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user