Reland 10216 - Optimize the equality check case of ICCompare stubs.

Now with arm and x64 support. Additionally, added default unreachable case to switch statement in CompareIC::TargetState to make win and mac compilers happy.

Reviewer guide:
This is an exact copy of 10216 except:
src/arm/*
src/x64/*
src/ic.cc (added default case to swith in CompareIC::TargetState)
Review URL: http://codereview.chromium.org/8872060

git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@10219 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
This commit is contained in:
ricow@chromium.org 2011-12-09 09:26:14 +00:00
parent ef54f5690f
commit ec66c36fbf
16 changed files with 245 additions and 81 deletions

View File

@ -6716,26 +6716,47 @@ void ICCompareStub::GenerateObjects(MacroAssembler* masm) {
} }
void ICCompareStub::GenerateMiss(MacroAssembler* masm) { void ICCompareStub::GenerateKnownObjects(MacroAssembler* masm) {
__ Push(r1, r0); Label miss;
__ push(lr); __ and_(r2, r1, Operand(r0));
__ JumpIfSmi(r2, &miss);
__ ldr(r2, FieldMemOperand(r0, HeapObject::kMapOffset));
__ ldr(r3, FieldMemOperand(r1, HeapObject::kMapOffset));
__ cmp(r2, Operand(known_map_));
__ b(ne, &miss);
__ cmp(r3, Operand(known_map_));
__ b(ne, &miss);
__ sub(r0, r0, Operand(r1));
__ Ret();
__ bind(&miss);
GenerateMiss(masm);
}
void ICCompareStub::GenerateMiss(MacroAssembler* masm) {
{
// Call the runtime system in a fresh internal frame. // Call the runtime system in a fresh internal frame.
ExternalReference miss = ExternalReference miss =
ExternalReference(IC_Utility(IC::kCompareIC_Miss), masm->isolate()); ExternalReference(IC_Utility(IC::kCompareIC_Miss), masm->isolate());
{
FrameScope scope(masm, StackFrame::INTERNAL); FrameScope scope(masm, StackFrame::INTERNAL);
__ Push(r1, r0); __ Push(r1, r0);
__ push(lr);
__ Push(r1, r0);
__ mov(ip, Operand(Smi::FromInt(op_))); __ mov(ip, Operand(Smi::FromInt(op_)));
__ push(ip); __ push(ip);
__ CallExternalReference(miss, 3); __ CallExternalReference(miss, 3);
}
// Compute the entry point of the rewritten stub. // Compute the entry point of the rewritten stub.
__ add(r2, r0, Operand(Code::kHeaderSize - kHeapObjectTag)); __ add(r2, r0, Operand(Code::kHeaderSize - kHeapObjectTag));
// Restore registers. // Restore registers.
__ pop(lr); __ pop(lr);
__ pop(r0); __ pop(r0);
__ pop(r1); __ pop(r1);
}
__ Jump(r2); __ Jump(r2);
} }

View File

@ -1587,6 +1587,9 @@ void CompareIC::UpdateCaches(Handle<Object> x, Handle<Object> y) {
rewritten = stub.GetCode(); rewritten = stub.GetCode();
} else { } else {
ICCompareStub stub(op_, state); ICCompareStub stub(op_, state);
if (state == KNOWN_OBJECTS) {
stub.set_known_map(Handle<Map>(Handle<JSObject>::cast(x)->map()));
}
rewritten = stub.GetCode(); rewritten = stub.GetCode();
} }
set_target(*rewritten); set_target(*rewritten);

View File

@ -101,7 +101,14 @@ Handle<Code> CodeStub::GetCode() {
Factory* factory = isolate->factory(); Factory* factory = isolate->factory();
Heap* heap = isolate->heap(); Heap* heap = isolate->heap();
Code* code; Code* code;
if (!FindCodeInCache(&code)) { if (UseSpecialCache()
? FindCodeInSpecialCache(&code)
: FindCodeInCache(&code)) {
ASSERT(IsPregenerated() == code->is_pregenerated());
return Handle<Code>(code);
}
{
HandleScope scope(isolate); HandleScope scope(isolate);
// Generate the new code. // Generate the new code.
@ -121,6 +128,9 @@ Handle<Code> CodeStub::GetCode() {
RecordCodeGeneration(*new_object, &masm); RecordCodeGeneration(*new_object, &masm);
FinishCode(new_object); FinishCode(new_object);
if (UseSpecialCache()) {
AddToSpecialCache(new_object);
} else {
// Update the dictionary and the root in Heap. // Update the dictionary and the root in Heap.
Handle<NumberDictionary> dict = Handle<NumberDictionary> dict =
factory->DictionaryAtNumberPut( factory->DictionaryAtNumberPut(
@ -128,12 +138,11 @@ Handle<Code> CodeStub::GetCode() {
GetKey(), GetKey(),
new_object); new_object);
heap->public_set_code_stubs(*dict); heap->public_set_code_stubs(*dict);
}
code = *new_object; code = *new_object;
Activate(code);
} else {
CHECK(IsPregenerated() == code->is_pregenerated());
} }
Activate(code);
ASSERT(!NeedsImmovableCode() || heap->lo_space()->Contains(code)); ASSERT(!NeedsImmovableCode() || heap->lo_space()->Contains(code));
return Handle<Code>(code, isolate); return Handle<Code>(code, isolate);
} }
@ -159,6 +168,32 @@ void CodeStub::PrintName(StringStream* stream) {
} }
void ICCompareStub::AddToSpecialCache(Handle<Code> new_object) {
ASSERT(*known_map_ != NULL);
Isolate* isolate = new_object->GetIsolate();
Factory* factory = isolate->factory();
return Map::UpdateCodeCache(known_map_,
factory->compare_ic_symbol(),
new_object);
}
bool ICCompareStub::FindCodeInSpecialCache(Code** code_out) {
Isolate* isolate = known_map_->GetIsolate();
Factory* factory = isolate->factory();
Code::Flags flags = Code::ComputeFlags(
static_cast<Code::Kind>(GetCodeKind()),
UNINITIALIZED);
Handle<Object> probe(
known_map_->FindInCodeCache(*factory->compare_ic_symbol(), flags));
if (probe->IsCode()) {
*code_out = Code::cast(*probe);
return true;
}
return false;
}
int ICCompareStub::MinorKey() { int ICCompareStub::MinorKey() {
return OpField::encode(op_ - Token::EQ) | StateField::encode(state_); return OpField::encode(op_ - Token::EQ) | StateField::encode(state_);
} }
@ -184,6 +219,10 @@ void ICCompareStub::Generate(MacroAssembler* masm) {
case CompareIC::OBJECTS: case CompareIC::OBJECTS:
GenerateObjects(masm); GenerateObjects(masm);
break; break;
case CompareIC::KNOWN_OBJECTS:
ASSERT(*known_map_ != NULL);
GenerateKnownObjects(masm);
break;
default: default:
UNREACHABLE(); UNREACHABLE();
} }

View File

@ -194,6 +194,17 @@ class CodeStub BASE_EMBEDDED {
return UNINITIALIZED; return UNINITIALIZED;
} }
// Add the code to a specialized cache, specific to an individual
// stub type. Please note, this method must add the code object to a
// roots object, otherwise we will remove the code during GC.
virtual void AddToSpecialCache(Handle<Code> new_object) { }
// Find code in a specialized cache, work is delegated to the specific stub.
virtual bool FindCodeInSpecialCache(Code** code_out) { return false; }
// If a stub uses a special cache override this.
virtual bool UseSpecialCache() { return false; }
// Returns a name for logging/debugging purposes. // Returns a name for logging/debugging purposes.
SmartArrayPointer<const char> GetName(); SmartArrayPointer<const char> GetName();
virtual void PrintName(StringStream* stream); virtual void PrintName(StringStream* stream);
@ -465,6 +476,8 @@ class ICCompareStub: public CodeStub {
virtual void Generate(MacroAssembler* masm); virtual void Generate(MacroAssembler* masm);
void set_known_map(Handle<Map> map) { known_map_ = map; }
private: private:
class OpField: public BitField<int, 0, 3> { }; class OpField: public BitField<int, 0, 3> { };
class StateField: public BitField<int, 3, 5> { }; class StateField: public BitField<int, 3, 5> { };
@ -484,12 +497,18 @@ class ICCompareStub: public CodeStub {
void GenerateStrings(MacroAssembler* masm); void GenerateStrings(MacroAssembler* masm);
void GenerateObjects(MacroAssembler* masm); void GenerateObjects(MacroAssembler* masm);
void GenerateMiss(MacroAssembler* masm); void GenerateMiss(MacroAssembler* masm);
void GenerateKnownObjects(MacroAssembler* masm);
bool strict() const { return op_ == Token::EQ_STRICT; } bool strict() const { return op_ == Token::EQ_STRICT; }
Condition GetCondition() const { return CompareIC::ComputeCondition(op_); } Condition GetCondition() const { return CompareIC::ComputeCondition(op_); }
virtual void AddToSpecialCache(Handle<Code> new_object);
virtual bool FindCodeInSpecialCache(Code** code_out);
virtual bool UseSpecialCache() { return state_ == CompareIC::KNOWN_OBJECTS; }
Token::Value op_; Token::Value op_;
CompareIC::State state_; CompareIC::State state_;
Handle<Map> known_map_;
}; };

View File

@ -2417,6 +2417,7 @@ bool Heap::CreateInitialObjects() {
} }
set_code_stubs(NumberDictionary::cast(obj)); set_code_stubs(NumberDictionary::cast(obj));
// Allocate the non_monomorphic_cache used in stub-cache.cc. The initial size // Allocate the non_monomorphic_cache used in stub-cache.cc. The initial size
// is set to avoid expanding the dictionary during bootstrapping. // is set to avoid expanding the dictionary during bootstrapping.
{ MaybeObject* maybe_obj = NumberDictionary::Allocate(64); { MaybeObject* maybe_obj = NumberDictionary::Allocate(64);

View File

@ -245,6 +245,7 @@ inline Heap* _inline_get_heap_();
V(use_strict, "use strict") \ V(use_strict, "use strict") \
V(dot_symbol, ".") \ V(dot_symbol, ".") \
V(anonymous_function_symbol, "(anonymous function)") \ V(anonymous_function_symbol, "(anonymous function)") \
V(compare_ic_symbol, ".compare_ic") \
V(infinity_symbol, "Infinity") \ V(infinity_symbol, "Infinity") \
V(minus_infinity_symbol, "-Infinity") V(minus_infinity_symbol, "-Infinity")

View File

@ -6141,6 +6141,18 @@ void HGraphBuilder::VisitCompareOperation(CompareOperation* expr) {
switch (op) { switch (op) {
case Token::EQ: case Token::EQ:
case Token::EQ_STRICT: { case Token::EQ_STRICT: {
// Can we get away with map check and not instance type check?
Handle<Map> map = oracle()->GetCompareMap(expr);
if (!map.is_null()) {
AddInstruction(new(zone()) HCheckNonSmi(left));
AddInstruction(new(zone()) HCheckMap(left, map));
AddInstruction(new(zone()) HCheckNonSmi(right));
AddInstruction(new(zone()) HCheckMap(right, map));
HCompareObjectEqAndBranch* result =
new(zone()) HCompareObjectEqAndBranch(left, right);
result->set_position(expr->position());
return ast_context()->ReturnControl(result, expr->id());
} else {
AddInstruction(new(zone()) HCheckNonSmi(left)); AddInstruction(new(zone()) HCheckNonSmi(left));
AddInstruction(HCheckInstanceType::NewIsSpecObject(left)); AddInstruction(HCheckInstanceType::NewIsSpecObject(left));
AddInstruction(new(zone()) HCheckNonSmi(right)); AddInstruction(new(zone()) HCheckNonSmi(right));
@ -6150,6 +6162,7 @@ void HGraphBuilder::VisitCompareOperation(CompareOperation* expr) {
result->set_position(expr->position()); result->set_position(expr->position());
return ast_context()->ReturnControl(result, expr->id()); return ast_context()->ReturnControl(result, expr->id());
} }
}
default: default:
return Bailout("Unsupported non-primitive compare"); return Bailout("Unsupported non-primitive compare");
} }

View File

@ -6670,32 +6670,44 @@ void ICCompareStub::GenerateObjects(MacroAssembler* masm) {
} }
void ICCompareStub::GenerateMiss(MacroAssembler* masm) { void ICCompareStub::GenerateKnownObjects(MacroAssembler* masm) {
// Save the registers. Label miss;
__ pop(ecx); __ mov(ecx, edx);
__ push(edx); __ and_(ecx, eax);
__ push(eax); __ JumpIfSmi(ecx, &miss, Label::kNear);
__ push(ecx);
__ mov(ecx, FieldOperand(eax, HeapObject::kMapOffset));
__ mov(ebx, FieldOperand(edx, HeapObject::kMapOffset));
__ cmp(ecx, known_map_);
__ j(not_equal, &miss, Label::kNear);
__ cmp(ebx, known_map_);
__ j(not_equal, &miss, Label::kNear);
__ sub(eax, edx);
__ ret(0);
__ bind(&miss);
GenerateMiss(masm);
}
void ICCompareStub::GenerateMiss(MacroAssembler* masm) {
{ {
// Call the runtime system in a fresh internal frame. // Call the runtime system in a fresh internal frame.
ExternalReference miss = ExternalReference(IC_Utility(IC::kCompareIC_Miss), ExternalReference miss = ExternalReference(IC_Utility(IC::kCompareIC_Miss),
masm->isolate()); masm->isolate());
FrameScope scope(masm, StackFrame::INTERNAL); FrameScope scope(masm, StackFrame::INTERNAL);
__ push(edx); __ push(edx); // Preserve edx and eax.
__ push(eax);
__ push(edx); // And also use them as the arguments.
__ push(eax); __ push(eax);
__ push(Immediate(Smi::FromInt(op_))); __ push(Immediate(Smi::FromInt(op_)));
__ CallExternalReference(miss, 3); __ CallExternalReference(miss, 3);
}
// Compute the entry point of the rewritten stub. // Compute the entry point of the rewritten stub.
__ lea(edi, FieldOperand(eax, Code::kHeaderSize)); __ lea(edi, FieldOperand(eax, Code::kHeaderSize));
// Restore registers.
__ pop(ecx);
__ pop(eax); __ pop(eax);
__ pop(edx); __ pop(edx);
__ push(ecx); }
// Do a tail call to the rewritten stub. // Do a tail call to the rewritten stub.
__ jmp(edi); __ jmp(edi);

View File

@ -1625,6 +1625,9 @@ void CompareIC::UpdateCaches(Handle<Object> x, Handle<Object> y) {
rewritten = stub.GetCode(); rewritten = stub.GetCode();
} else { } else {
ICCompareStub stub(op_, state); ICCompareStub stub(op_, state);
if (state == KNOWN_OBJECTS) {
stub.set_known_map(Handle<Map>(Handle<JSObject>::cast(x)->map()));
}
rewritten = stub.GetCode(); rewritten = stub.GetCode();
} }
set_target(*rewritten); set_target(*rewritten);

View File

@ -2320,6 +2320,7 @@ const char* CompareIC::GetStateName(State state) {
case SMIS: return "SMIS"; case SMIS: return "SMIS";
case HEAP_NUMBERS: return "HEAP_NUMBERS"; case HEAP_NUMBERS: return "HEAP_NUMBERS";
case OBJECTS: return "OBJECTS"; case OBJECTS: return "OBJECTS";
case KNOWN_OBJECTS: return "OBJECTS";
case SYMBOLS: return "SYMBOLS"; case SYMBOLS: return "SYMBOLS";
case STRINGS: return "STRINGS"; case STRINGS: return "STRINGS";
case GENERIC: return "GENERIC"; case GENERIC: return "GENERIC";
@ -2334,19 +2335,38 @@ CompareIC::State CompareIC::TargetState(State state,
bool has_inlined_smi_code, bool has_inlined_smi_code,
Handle<Object> x, Handle<Object> x,
Handle<Object> y) { Handle<Object> y) {
if (!has_inlined_smi_code && state != UNINITIALIZED && state != SYMBOLS) { switch (state) {
case UNINITIALIZED:
if (x->IsSmi() && y->IsSmi()) return SMIS;
if (x->IsNumber() && y->IsNumber()) return HEAP_NUMBERS;
if (!Token::IsEqualityOp(op_)) return GENERIC;
if (x->IsSymbol() && y->IsSymbol()) return SYMBOLS;
if (x->IsString() && y->IsString()) return STRINGS;
if (x->IsJSObject() && y->IsJSObject()) {
if (Handle<JSObject>::cast(x)->map() ==
Handle<JSObject>::cast(y)->map() &&
Token::IsEqualityOp(op_)) {
return KNOWN_OBJECTS;
} else {
return OBJECTS;
}
}
return GENERIC;
case SMIS:
return has_inlined_smi_code && x->IsNumber() && y->IsNumber()
? HEAP_NUMBERS
: GENERIC;
case SYMBOLS:
ASSERT(Token::IsEqualityOp(op_));
return x->IsString() && y->IsString() ? STRINGS : GENERIC;
case HEAP_NUMBERS:
case STRINGS:
case OBJECTS:
case KNOWN_OBJECTS:
case GENERIC:
return GENERIC; return GENERIC;
} }
if (state == UNINITIALIZED && x->IsSmi() && y->IsSmi()) return SMIS; UNREACHABLE();
if ((state == UNINITIALIZED || (state == SMIS && has_inlined_smi_code)) &&
x->IsNumber() && y->IsNumber()) return HEAP_NUMBERS;
if (op_ != Token::EQ && op_ != Token::EQ_STRICT) return GENERIC;
if (state == UNINITIALIZED &&
x->IsSymbol() && y->IsSymbol()) return SYMBOLS;
if ((state == UNINITIALIZED || state == SYMBOLS) &&
x->IsString() && y->IsString()) return STRINGS;
if (state == UNINITIALIZED &&
x->IsJSObject() && y->IsJSObject()) return OBJECTS;
return GENERIC; return GENERIC;
} }

View File

@ -724,6 +724,7 @@ class CompareIC: public IC {
SYMBOLS, SYMBOLS,
STRINGS, STRINGS,
OBJECTS, OBJECTS,
KNOWN_OBJECTS,
GENERIC GENERIC
}; };

View File

@ -883,8 +883,6 @@ class StaticMarkingVisitor : public StaticVisitorBase {
Code* target = Code::GetCodeFromTargetAddress(rinfo->target_address()); Code* target = Code::GetCodeFromTargetAddress(rinfo->target_address());
if (FLAG_cleanup_code_caches_at_gc && target->is_inline_cache_stub()) { if (FLAG_cleanup_code_caches_at_gc && target->is_inline_cache_stub()) {
IC::Clear(rinfo->pc()); IC::Clear(rinfo->pc());
// Please note targets for cleared inline cached do not have to be
// marked since they are contained in HEAP->non_monomorphic_cache().
target = Code::GetCodeFromTargetAddress(rinfo->target_address()); target = Code::GetCodeFromTargetAddress(rinfo->target_address());
} else { } else {
if (FLAG_cleanup_code_caches_at_gc && if (FLAG_cleanup_code_caches_at_gc &&
@ -893,9 +891,10 @@ class StaticMarkingVisitor : public StaticVisitorBase {
target->has_function_cache()) { target->has_function_cache()) {
CallFunctionStub::Clear(heap, rinfo->pc()); CallFunctionStub::Clear(heap, rinfo->pc());
} }
}
MarkBit code_mark = Marking::MarkBitFrom(target); MarkBit code_mark = Marking::MarkBitFrom(target);
heap->mark_compact_collector()->MarkObject(target, code_mark); heap->mark_compact_collector()->MarkObject(target, code_mark);
}
heap->mark_compact_collector()->RecordRelocSlot(rinfo, target); heap->mark_compact_collector()->RecordRelocSlot(rinfo, target);
} }

View File

@ -259,6 +259,7 @@ TypeInfo TypeFeedbackOracle::CompareType(CompareOperation* expr) {
case CompareIC::STRINGS: case CompareIC::STRINGS:
return TypeInfo::String(); return TypeInfo::String();
case CompareIC::OBJECTS: case CompareIC::OBJECTS:
case CompareIC::KNOWN_OBJECTS:
// TODO(kasperl): We really need a type for JS objects here. // TODO(kasperl): We really need a type for JS objects here.
return TypeInfo::NonPrimitive(); return TypeInfo::NonPrimitive();
case CompareIC::GENERIC: case CompareIC::GENERIC:
@ -278,6 +279,19 @@ bool TypeFeedbackOracle::IsSymbolCompare(CompareOperation* expr) {
} }
Handle<Map> TypeFeedbackOracle::GetCompareMap(CompareOperation* expr) {
Handle<Object> object = GetInfo(expr->id());
if (!object->IsCode()) return Handle<Map>::null();
Handle<Code> code = Handle<Code>::cast(object);
if (!code->is_compare_ic_stub()) return Handle<Map>::null();
CompareIC::State state = static_cast<CompareIC::State>(code->compare_state());
if (state != CompareIC::KNOWN_OBJECTS) {
return Handle<Map>::null();
}
return Handle<Map>(code->FindFirstMap());
}
TypeInfo TypeFeedbackOracle::UnaryType(UnaryOperation* expr) { TypeInfo TypeFeedbackOracle::UnaryType(UnaryOperation* expr) {
Handle<Object> object = GetInfo(expr->id()); Handle<Object> object = GetInfo(expr->id());
TypeInfo unknown = TypeInfo::Unknown(); TypeInfo unknown = TypeInfo::Unknown();
@ -367,6 +381,7 @@ TypeInfo TypeFeedbackOracle::SwitchType(CaseClause* clause) {
case CompareIC::HEAP_NUMBERS: case CompareIC::HEAP_NUMBERS:
return TypeInfo::Number(); return TypeInfo::Number();
case CompareIC::OBJECTS: case CompareIC::OBJECTS:
case CompareIC::KNOWN_OBJECTS:
// TODO(kasperl): We really need a type for JS objects here. // TODO(kasperl): We really need a type for JS objects here.
return TypeInfo::NonPrimitive(); return TypeInfo::NonPrimitive();
case CompareIC::GENERIC: case CompareIC::GENERIC:

View File

@ -273,6 +273,7 @@ class TypeFeedbackOracle BASE_EMBEDDED {
TypeInfo BinaryType(BinaryOperation* expr); TypeInfo BinaryType(BinaryOperation* expr);
TypeInfo CompareType(CompareOperation* expr); TypeInfo CompareType(CompareOperation* expr);
bool IsSymbolCompare(CompareOperation* expr); bool IsSymbolCompare(CompareOperation* expr);
Handle<Map> GetCompareMap(CompareOperation* expr);
TypeInfo SwitchType(CaseClause* clause); TypeInfo SwitchType(CaseClause* clause);
TypeInfo IncrementType(CountOperation* expr); TypeInfo IncrementType(CountOperation* expr);

View File

@ -5608,32 +5608,45 @@ void ICCompareStub::GenerateObjects(MacroAssembler* masm) {
} }
void ICCompareStub::GenerateMiss(MacroAssembler* masm) { void ICCompareStub::GenerateKnownObjects(MacroAssembler* masm) {
// Save the registers. Label miss;
__ pop(rcx); Condition either_smi = masm->CheckEitherSmi(rdx, rax);
__ push(rdx); __ j(either_smi, &miss, Label::kNear);
__ push(rax);
__ push(rcx);
__ movq(rcx, FieldOperand(rax, HeapObject::kMapOffset));
__ movq(rbx, FieldOperand(rdx, HeapObject::kMapOffset));
__ Cmp(rcx, known_map_);
__ j(not_equal, &miss, Label::kNear);
__ Cmp(rbx, known_map_);
__ j(not_equal, &miss, Label::kNear);
__ subq(rax, rdx);
__ ret(0);
__ bind(&miss);
GenerateMiss(masm);
}
void ICCompareStub::GenerateMiss(MacroAssembler* masm) {
{
// Call the runtime system in a fresh internal frame. // Call the runtime system in a fresh internal frame.
ExternalReference miss = ExternalReference miss =
ExternalReference(IC_Utility(IC::kCompareIC_Miss), masm->isolate()); ExternalReference(IC_Utility(IC::kCompareIC_Miss), masm->isolate());
{
FrameScope scope(masm, StackFrame::INTERNAL); FrameScope scope(masm, StackFrame::INTERNAL);
__ push(rdx); __ push(rdx);
__ push(rax); __ push(rax);
__ push(rdx);
__ push(rax);
__ Push(Smi::FromInt(op_)); __ Push(Smi::FromInt(op_));
__ CallExternalReference(miss, 3); __ CallExternalReference(miss, 3);
}
// Compute the entry point of the rewritten stub. // Compute the entry point of the rewritten stub.
__ lea(rdi, FieldOperand(rax, Code::kHeaderSize)); __ lea(rdi, FieldOperand(rax, Code::kHeaderSize));
// Restore registers.
__ pop(rcx);
__ pop(rax); __ pop(rax);
__ pop(rdx); __ pop(rdx);
__ push(rcx); }
// Do a tail call to the rewritten stub. // Do a tail call to the rewritten stub.
__ jmp(rdi); __ jmp(rdi);

View File

@ -1641,6 +1641,9 @@ void CompareIC::UpdateCaches(Handle<Object> x, Handle<Object> y) {
rewritten = stub.GetCode(); rewritten = stub.GetCode();
} else { } else {
ICCompareStub stub(op_, state); ICCompareStub stub(op_, state);
if (state == KNOWN_OBJECTS) {
stub.set_known_map(Handle<Map>(Handle<JSObject>::cast(x)->map()));
}
rewritten = stub.GetCode(); rewritten = stub.GetCode();
} }
set_target(*rewritten); set_target(*rewritten);