Index: src/x64/lithium-codegen-x64.cc |
diff --git a/src/x64/lithium-codegen-x64.cc b/src/x64/lithium-codegen-x64.cc |
index 596072617f11ddc0146714dd5dc0aa2448ddae2f..1dac13d2b8f979fb30251032ccc0f91864c7d90c 100644 |
--- a/src/x64/lithium-codegen-x64.cc |
+++ b/src/x64/lithium-codegen-x64.cc |
@@ -1862,10 +1862,12 @@ int LCodeGen::GetNextEmittedBlock() const { |
} |
-void LCodeGen::EmitBranch(int left_block, int right_block, Condition cc) { |
+template<class InstrType> |
+void LCodeGen::EmitBranch(InstrType instr, Condition cc) { |
+ int right_block = instr->FalseDestination(chunk_); |
+ int left_block = instr->TrueDestination(chunk_); |
+ |
int next_block = GetNextEmittedBlock(); |
- right_block = chunk_->LookupDestination(right_block); |
- left_block = chunk_->LookupDestination(left_block); |
if (right_block == left_block) { |
EmitGoto(left_block); |
@@ -1888,26 +1890,23 @@ void LCodeGen::DoDebugBreak(LDebugBreak* instr) { |
void LCodeGen::DoBranch(LBranch* instr) { |
- int true_block = chunk_->LookupDestination(instr->true_block_id()); |
- int false_block = chunk_->LookupDestination(instr->false_block_id()); |
- |
Representation r = instr->hydrogen()->value()->representation(); |
if (r.IsInteger32()) { |
ASSERT(!info()->IsStub()); |
Register reg = ToRegister(instr->value()); |
__ testl(reg, reg); |
- EmitBranch(true_block, false_block, not_zero); |
+ EmitBranch(instr, not_zero); |
} else if (r.IsSmi()) { |
ASSERT(!info()->IsStub()); |
Register reg = ToRegister(instr->value()); |
__ testq(reg, reg); |
- EmitBranch(true_block, false_block, not_zero); |
+ EmitBranch(instr, not_zero); |
} else if (r.IsDouble()) { |
ASSERT(!info()->IsStub()); |
XMMRegister reg = ToDoubleRegister(instr->value()); |
__ xorps(xmm0, xmm0); |
__ ucomisd(reg, xmm0); |
- EmitBranch(true_block, false_block, not_equal); |
+ EmitBranch(instr, not_equal); |
} else { |
ASSERT(r.IsTagged()); |
Register reg = ToRegister(instr->value()); |
@@ -1915,15 +1914,12 @@ void LCodeGen::DoBranch(LBranch* instr) { |
if (type.IsBoolean()) { |
ASSERT(!info()->IsStub()); |
__ CompareRoot(reg, Heap::kTrueValueRootIndex); |
- EmitBranch(true_block, false_block, equal); |
+ EmitBranch(instr, equal); |
} else if (type.IsSmi()) { |
ASSERT(!info()->IsStub()); |
__ SmiCompare(reg, Smi::FromInt(0)); |
- EmitBranch(true_block, false_block, not_equal); |
+ EmitBranch(instr, not_equal); |
} else { |
- Label* true_label = chunk_->GetAssemblyLabel(true_block); |
- Label* false_label = chunk_->GetAssemblyLabel(false_block); |
- |
ToBooleanStub::Types expected = instr->hydrogen()->expected_input_types(); |
// Avoid deopts in the case where we've never executed this path before. |
if (expected.IsEmpty()) expected = ToBooleanStub::all_types(); |
@@ -1931,27 +1927,27 @@ void LCodeGen::DoBranch(LBranch* instr) { |
if (expected.Contains(ToBooleanStub::UNDEFINED)) { |
// undefined -> false. |
__ CompareRoot(reg, Heap::kUndefinedValueRootIndex); |
- __ j(equal, false_label); |
+ __ j(equal, instr->FalseLabel(chunk_)); |
} |
if (expected.Contains(ToBooleanStub::BOOLEAN)) { |
// true -> true. |
__ CompareRoot(reg, Heap::kTrueValueRootIndex); |
- __ j(equal, true_label); |
+ __ j(equal, instr->TrueLabel(chunk_)); |
// false -> false. |
__ CompareRoot(reg, Heap::kFalseValueRootIndex); |
- __ j(equal, false_label); |
+ __ j(equal, instr->FalseLabel(chunk_)); |
} |
if (expected.Contains(ToBooleanStub::NULL_TYPE)) { |
// 'null' -> false. |
__ CompareRoot(reg, Heap::kNullValueRootIndex); |
- __ j(equal, false_label); |
+ __ j(equal, instr->FalseLabel(chunk_)); |
} |
if (expected.Contains(ToBooleanStub::SMI)) { |
// Smis: 0 -> false, all other -> true. |
__ Cmp(reg, Smi::FromInt(0)); |
- __ j(equal, false_label); |
- __ JumpIfSmi(reg, true_label); |
+ __ j(equal, instr->FalseLabel(chunk_)); |
+ __ JumpIfSmi(reg, instr->TrueLabel(chunk_)); |
} else if (expected.NeedsMap()) { |
// If we need a map later and have a Smi -> deopt. |
__ testb(reg, Immediate(kSmiTagMask)); |
@@ -1966,14 +1962,14 @@ void LCodeGen::DoBranch(LBranch* instr) { |
// Undetectable -> false. |
__ testb(FieldOperand(map, Map::kBitFieldOffset), |
Immediate(1 << Map::kIsUndetectable)); |
- __ j(not_zero, false_label); |
+ __ j(not_zero, instr->FalseLabel(chunk_)); |
} |
} |
if (expected.Contains(ToBooleanStub::SPEC_OBJECT)) { |
// spec object -> true. |
__ CmpInstanceType(map, FIRST_SPEC_OBJECT_TYPE); |
- __ j(above_equal, true_label); |
+ __ j(above_equal, instr->TrueLabel(chunk_)); |
} |
if (expected.Contains(ToBooleanStub::STRING)) { |
@@ -1982,8 +1978,8 @@ void LCodeGen::DoBranch(LBranch* instr) { |
__ CmpInstanceType(map, FIRST_NONSTRING_TYPE); |
__ j(above_equal, ¬_string, Label::kNear); |
__ cmpq(FieldOperand(reg, String::kLengthOffset), Immediate(0)); |
- __ j(not_zero, true_label); |
- __ jmp(false_label); |
+ __ j(not_zero, instr->TrueLabel(chunk_)); |
+ __ jmp(instr->FalseLabel(chunk_)); |
__ bind(¬_string); |
} |
@@ -1994,8 +1990,8 @@ void LCodeGen::DoBranch(LBranch* instr) { |
__ j(not_equal, ¬_heap_number, Label::kNear); |
__ xorps(xmm0, xmm0); |
__ ucomisd(xmm0, FieldOperand(reg, HeapNumber::kValueOffset)); |
- __ j(zero, false_label); |
- __ jmp(true_label); |
+ __ j(zero, instr->FalseLabel(chunk_)); |
+ __ jmp(instr->TrueLabel(chunk_)); |
__ bind(¬_heap_number); |
} |
@@ -2049,24 +2045,21 @@ inline Condition LCodeGen::TokenToCondition(Token::Value op, bool is_unsigned) { |
void LCodeGen::DoCmpIDAndBranch(LCmpIDAndBranch* instr) { |
LOperand* left = instr->left(); |
LOperand* right = instr->right(); |
- int false_block = chunk_->LookupDestination(instr->false_block_id()); |
- int true_block = chunk_->LookupDestination(instr->true_block_id()); |
Condition cc = TokenToCondition(instr->op(), instr->is_double()); |
if (left->IsConstantOperand() && right->IsConstantOperand()) { |
// We can statically evaluate the comparison. |
double left_val = ToDouble(LConstantOperand::cast(left)); |
double right_val = ToDouble(LConstantOperand::cast(right)); |
- int next_block = |
- EvalComparison(instr->op(), left_val, right_val) ? true_block |
- : false_block; |
+ int next_block = EvalComparison(instr->op(), left_val, right_val) ? |
+ instr->TrueDestination(chunk_) : instr->FalseDestination(chunk_); |
EmitGoto(next_block); |
} else { |
if (instr->is_double()) { |
// Don't base result on EFLAGS when a NaN is involved. Instead |
// jump to the false block. |
__ ucomisd(ToDoubleRegister(left), ToDoubleRegister(right)); |
- __ j(parity_even, chunk_->GetAssemblyLabel(false_block)); |
+ __ j(parity_even, instr->FalseLabel(chunk_)); |
} else { |
int32_t value; |
if (right->IsConstantOperand()) { |
@@ -2105,15 +2098,13 @@ void LCodeGen::DoCmpIDAndBranch(LCmpIDAndBranch* instr) { |
} |
} |
} |
- EmitBranch(true_block, false_block, cc); |
+ EmitBranch(instr, cc); |
} |
} |
void LCodeGen::DoCmpObjectEqAndBranch(LCmpObjectEqAndBranch* instr) { |
Register left = ToRegister(instr->left()); |
- int false_block = chunk_->LookupDestination(instr->false_block_id()); |
- int true_block = chunk_->LookupDestination(instr->true_block_id()); |
if (instr->right()->IsConstantOperand()) { |
Handle<Object> right = ToHandle(LConstantOperand::cast(instr->right())); |
@@ -2122,17 +2113,15 @@ void LCodeGen::DoCmpObjectEqAndBranch(LCmpObjectEqAndBranch* instr) { |
Register right = ToRegister(instr->right()); |
__ cmpq(left, right); |
} |
- EmitBranch(true_block, false_block, equal); |
+ EmitBranch(instr, equal); |
} |
void LCodeGen::DoCmpConstantEqAndBranch(LCmpConstantEqAndBranch* instr) { |
Register left = ToRegister(instr->left()); |
- int true_block = chunk_->LookupDestination(instr->true_block_id()); |
- int false_block = chunk_->LookupDestination(instr->false_block_id()); |
__ cmpq(left, Immediate(instr->hydrogen()->right())); |
- EmitBranch(true_block, false_block, equal); |
+ EmitBranch(instr, equal); |
} |
@@ -2164,14 +2153,10 @@ Condition LCodeGen::EmitIsObject(Register input, |
void LCodeGen::DoIsObjectAndBranch(LIsObjectAndBranch* instr) { |
Register reg = ToRegister(instr->value()); |
- int true_block = chunk_->LookupDestination(instr->true_block_id()); |
- int false_block = chunk_->LookupDestination(instr->false_block_id()); |
- Label* true_label = chunk_->GetAssemblyLabel(true_block); |
- Label* false_label = chunk_->GetAssemblyLabel(false_block); |
+ Condition true_cond = EmitIsObject( |
+ reg, instr->FalseLabel(chunk_), instr->TrueLabel(chunk_)); |
- Condition true_cond = EmitIsObject(reg, false_label, true_label); |
- |
- EmitBranch(true_block, false_block, true_cond); |
+ EmitBranch(instr, true_cond); |
} |
@@ -2189,20 +2174,13 @@ void LCodeGen::DoIsStringAndBranch(LIsStringAndBranch* instr) { |
Register reg = ToRegister(instr->value()); |
Register temp = ToRegister(instr->temp()); |
- int true_block = chunk_->LookupDestination(instr->true_block_id()); |
- int false_block = chunk_->LookupDestination(instr->false_block_id()); |
- Label* false_label = chunk_->GetAssemblyLabel(false_block); |
- |
- Condition true_cond = EmitIsString(reg, temp, false_label); |
+ Condition true_cond = EmitIsString(reg, temp, instr->FalseLabel(chunk_)); |
- EmitBranch(true_block, false_block, true_cond); |
+ EmitBranch(instr, true_cond); |
} |
void LCodeGen::DoIsSmiAndBranch(LIsSmiAndBranch* instr) { |
- int true_block = chunk_->LookupDestination(instr->true_block_id()); |
- int false_block = chunk_->LookupDestination(instr->false_block_id()); |
- |
Condition is_smi; |
if (instr->value()->IsRegister()) { |
Register input = ToRegister(instr->value()); |
@@ -2211,7 +2189,7 @@ void LCodeGen::DoIsSmiAndBranch(LIsSmiAndBranch* instr) { |
Operand input = ToOperand(instr->value()); |
is_smi = masm()->CheckSmi(input); |
} |
- EmitBranch(true_block, false_block, is_smi); |
+ EmitBranch(instr, is_smi); |
} |
@@ -2219,21 +2197,16 @@ void LCodeGen::DoIsUndetectableAndBranch(LIsUndetectableAndBranch* instr) { |
Register input = ToRegister(instr->value()); |
Register temp = ToRegister(instr->temp()); |
- int true_block = chunk_->LookupDestination(instr->true_block_id()); |
- int false_block = chunk_->LookupDestination(instr->false_block_id()); |
- |
- __ JumpIfSmi(input, chunk_->GetAssemblyLabel(false_block)); |
+ __ JumpIfSmi(input, instr->FalseLabel(chunk_)); |
__ movq(temp, FieldOperand(input, HeapObject::kMapOffset)); |
__ testb(FieldOperand(temp, Map::kBitFieldOffset), |
Immediate(1 << Map::kIsUndetectable)); |
- EmitBranch(true_block, false_block, not_zero); |
+ EmitBranch(instr, not_zero); |
} |
void LCodeGen::DoStringCompareAndBranch(LStringCompareAndBranch* instr) { |
Token::Value op = instr->op(); |
- int true_block = chunk_->LookupDestination(instr->true_block_id()); |
- int false_block = chunk_->LookupDestination(instr->false_block_id()); |
Handle<Code> ic = CompareIC::GetUninitialized(isolate(), op); |
CallCode(ic, RelocInfo::CODE_TARGET, instr); |
@@ -2241,7 +2214,7 @@ void LCodeGen::DoStringCompareAndBranch(LStringCompareAndBranch* instr) { |
Condition condition = TokenToCondition(op, false); |
__ testq(rax, rax); |
- EmitBranch(true_block, false_block, condition); |
+ EmitBranch(instr, condition); |
} |
@@ -2268,15 +2241,10 @@ static Condition BranchCondition(HHasInstanceTypeAndBranch* instr) { |
void LCodeGen::DoHasInstanceTypeAndBranch(LHasInstanceTypeAndBranch* instr) { |
Register input = ToRegister(instr->value()); |
- int true_block = chunk_->LookupDestination(instr->true_block_id()); |
- int false_block = chunk_->LookupDestination(instr->false_block_id()); |
- |
- Label* false_label = chunk_->GetAssemblyLabel(false_block); |
- |
- __ JumpIfSmi(input, false_label); |
+ __ JumpIfSmi(input, instr->FalseLabel(chunk_)); |
__ CmpObjectType(input, TestType(instr->hydrogen()), kScratchRegister); |
- EmitBranch(true_block, false_block, BranchCondition(instr->hydrogen())); |
+ EmitBranch(instr, BranchCondition(instr->hydrogen())); |
} |
@@ -2296,12 +2264,9 @@ void LCodeGen::DoHasCachedArrayIndexAndBranch( |
LHasCachedArrayIndexAndBranch* instr) { |
Register input = ToRegister(instr->value()); |
- int true_block = chunk_->LookupDestination(instr->true_block_id()); |
- int false_block = chunk_->LookupDestination(instr->false_block_id()); |
- |
__ testl(FieldOperand(input, String::kHashFieldOffset), |
Immediate(String::kContainsCachedArrayIndexMask)); |
- EmitBranch(true_block, false_block, equal); |
+ EmitBranch(instr, equal); |
} |
@@ -2379,25 +2344,18 @@ void LCodeGen::DoClassOfTestAndBranch(LClassOfTestAndBranch* instr) { |
Register temp2 = ToRegister(instr->temp2()); |
Handle<String> class_name = instr->hydrogen()->class_name(); |
- int true_block = chunk_->LookupDestination(instr->true_block_id()); |
- int false_block = chunk_->LookupDestination(instr->false_block_id()); |
- |
- Label* true_label = chunk_->GetAssemblyLabel(true_block); |
- Label* false_label = chunk_->GetAssemblyLabel(false_block); |
- |
- EmitClassOfTest(true_label, false_label, class_name, input, temp, temp2); |
+ EmitClassOfTest(instr->TrueLabel(chunk_), instr->FalseLabel(chunk_), |
+ class_name, input, temp, temp2); |
- EmitBranch(true_block, false_block, equal); |
+ EmitBranch(instr, equal); |
} |
void LCodeGen::DoCmpMapAndBranch(LCmpMapAndBranch* instr) { |
Register reg = ToRegister(instr->value()); |
- int true_block = instr->true_block_id(); |
- int false_block = instr->false_block_id(); |
__ Cmp(FieldOperand(reg, HeapObject::kMapOffset), instr->map()); |
- EmitBranch(true_block, false_block, equal); |
+ EmitBranch(instr, equal); |
} |
@@ -5347,15 +5305,12 @@ void LCodeGen::EmitPushTaggedOperand(LOperand* operand) { |
void LCodeGen::DoTypeofIsAndBranch(LTypeofIsAndBranch* instr) { |
Register input = ToRegister(instr->value()); |
- int true_block = chunk_->LookupDestination(instr->true_block_id()); |
- int false_block = chunk_->LookupDestination(instr->false_block_id()); |
- Label* true_label = chunk_->GetAssemblyLabel(true_block); |
- Label* false_label = chunk_->GetAssemblyLabel(false_block); |
Condition final_branch_condition = |
- EmitTypeofIs(true_label, false_label, input, instr->type_literal()); |
+ EmitTypeofIs(instr->TrueLabel(chunk_), |
+ instr->FalseLabel(chunk_), input, instr->type_literal()); |
if (final_branch_condition != no_condition) { |
- EmitBranch(true_block, false_block, final_branch_condition); |
+ EmitBranch(instr, final_branch_condition); |
} |
} |
@@ -5438,11 +5393,9 @@ Condition LCodeGen::EmitTypeofIs(Label* true_label, |
void LCodeGen::DoIsConstructCallAndBranch(LIsConstructCallAndBranch* instr) { |
Register temp = ToRegister(instr->temp()); |
- int true_block = chunk_->LookupDestination(instr->true_block_id()); |
- int false_block = chunk_->LookupDestination(instr->false_block_id()); |
EmitIsConstructCall(temp); |
- EmitBranch(true_block, false_block, equal); |
+ EmitBranch(instr, equal); |
} |