Commit 89176e88 authored by erik.corry@gmail.com's avatar erik.corry@gmail.com

MIPS: Add negative lookups to polymorphic loads in Crankshaft.

This is a commit of https://chromiumcodereview.appspot.com/10536130/ for Akos Palfi.


git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@11794 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
parent c1944660
...@@ -2315,12 +2315,12 @@ void LCodeGen::DoLoadNamedField(LLoadNamedField* instr) { ...@@ -2315,12 +2315,12 @@ void LCodeGen::DoLoadNamedField(LLoadNamedField* instr) {
void LCodeGen::EmitLoadFieldOrConstantFunction(Register result, void LCodeGen::EmitLoadFieldOrConstantFunction(Register result,
Register object, Register object,
Handle<Map> type, Handle<Map> type,
Handle<String> name) { Handle<String> name,
LEnvironment* env) {
LookupResult lookup(isolate()); LookupResult lookup(isolate());
type->LookupInDescriptors(NULL, *name, &lookup); type->LookupInDescriptors(NULL, *name, &lookup);
ASSERT(lookup.IsFound() && ASSERT(lookup.IsFound() || lookup.IsCacheable());
(lookup.type() == FIELD || lookup.type() == CONSTANT_FUNCTION)); if (lookup.IsFound() && lookup.type() == FIELD) {
if (lookup.type() == FIELD) {
int index = lookup.GetLocalFieldIndexFromMap(*type); int index = lookup.GetLocalFieldIndexFromMap(*type);
int offset = index * kPointerSize; int offset = index * kPointerSize;
if (index < 0) { if (index < 0) {
...@@ -2332,9 +2332,23 @@ void LCodeGen::EmitLoadFieldOrConstantFunction(Register result, ...@@ -2332,9 +2332,23 @@ void LCodeGen::EmitLoadFieldOrConstantFunction(Register result,
__ lw(result, FieldMemOperand(object, JSObject::kPropertiesOffset)); __ lw(result, FieldMemOperand(object, JSObject::kPropertiesOffset));
__ lw(result, FieldMemOperand(result, offset + FixedArray::kHeaderSize)); __ lw(result, FieldMemOperand(result, offset + FixedArray::kHeaderSize));
} }
} else { } else if (lookup.IsFound() && lookup.type() == CONSTANT_FUNCTION) {
Handle<JSFunction> function(lookup.GetConstantFunctionFromMap(*type)); Handle<JSFunction> function(lookup.GetConstantFunctionFromMap(*type));
__ LoadHeapObject(result, function); __ LoadHeapObject(result, function);
} else {
// Negative lookup.
// Check prototypes.
HeapObject* current = HeapObject::cast((*type)->prototype());
Heap* heap = type->GetHeap();
while (current != heap->null_value()) {
Handle<HeapObject> link(current);
__ LoadHeapObject(result, link);
__ lw(result, FieldMemOperand(result, HeapObject::kMapOffset));
DeoptimizeIf(ne, env,
result, Operand(Handle<Map>(JSObject::cast(current)->map())));
current = HeapObject::cast(current->map()->prototype());
}
__ LoadRoot(result, Heap::kUndefinedValueRootIndex);
} }
} }
...@@ -2342,7 +2356,7 @@ void LCodeGen::EmitLoadFieldOrConstantFunction(Register result, ...@@ -2342,7 +2356,7 @@ void LCodeGen::EmitLoadFieldOrConstantFunction(Register result,
void LCodeGen::DoLoadNamedFieldPolymorphic(LLoadNamedFieldPolymorphic* instr) { void LCodeGen::DoLoadNamedFieldPolymorphic(LLoadNamedFieldPolymorphic* instr) {
Register object = ToRegister(instr->object()); Register object = ToRegister(instr->object());
Register result = ToRegister(instr->result()); Register result = ToRegister(instr->result());
Register scratch = scratch0(); Register object_map = scratch0();
int map_count = instr->hydrogen()->types()->length(); int map_count = instr->hydrogen()->types()->length();
bool need_generic = instr->hydrogen()->need_generic(); bool need_generic = instr->hydrogen()->need_generic();
...@@ -2353,17 +2367,25 @@ void LCodeGen::DoLoadNamedFieldPolymorphic(LLoadNamedFieldPolymorphic* instr) { ...@@ -2353,17 +2367,25 @@ void LCodeGen::DoLoadNamedFieldPolymorphic(LLoadNamedFieldPolymorphic* instr) {
} }
Handle<String> name = instr->hydrogen()->name(); Handle<String> name = instr->hydrogen()->name();
Label done; Label done;
__ lw(scratch, FieldMemOperand(object, HeapObject::kMapOffset)); __ lw(object_map, FieldMemOperand(object, HeapObject::kMapOffset));
for (int i = 0; i < map_count; ++i) { for (int i = 0; i < map_count; ++i) {
bool last = (i == map_count - 1); bool last = (i == map_count - 1);
Handle<Map> map = instr->hydrogen()->types()->at(i); Handle<Map> map = instr->hydrogen()->types()->at(i);
Label check_passed;
__ CompareMapAndBranch(
object_map, map, &check_passed,
eq, &check_passed, ALLOW_ELEMENT_TRANSITION_MAPS);
if (last && !need_generic) { if (last && !need_generic) {
DeoptimizeIf(ne, instr->environment(), scratch, Operand(map)); DeoptimizeIf(al, instr->environment());
EmitLoadFieldOrConstantFunction(result, object, map, name); __ bind(&check_passed);
EmitLoadFieldOrConstantFunction(
result, object, map, name, instr->environment());
} else { } else {
Label next; Label next;
__ Branch(&next, ne, scratch, Operand(map)); __ Branch(&next);
EmitLoadFieldOrConstantFunction(result, object, map, name); __ bind(&check_passed);
EmitLoadFieldOrConstantFunction(
result, object, map, name, instr->environment());
__ Branch(&done); __ Branch(&done);
__ bind(&next); __ bind(&next);
} }
......
...@@ -334,7 +334,8 @@ class LCodeGen BASE_EMBEDDED { ...@@ -334,7 +334,8 @@ class LCodeGen BASE_EMBEDDED {
void EmitLoadFieldOrConstantFunction(Register result, void EmitLoadFieldOrConstantFunction(Register result,
Register object, Register object,
Handle<Map> type, Handle<Map> type,
Handle<String> name); Handle<String> name,
LEnvironment* env);
// Emits optimized code to deep-copy the contents of statically known // Emits optimized code to deep-copy the contents of statically known
// object graphs (e.g. object literal boilerplate). // object graphs (e.g. object literal boilerplate).
......
...@@ -3473,6 +3473,16 @@ void MacroAssembler::CompareMapAndBranch(Register obj, ...@@ -3473,6 +3473,16 @@ void MacroAssembler::CompareMapAndBranch(Register obj,
Label* branch_to, Label* branch_to,
CompareMapMode mode) { CompareMapMode mode) {
lw(scratch, FieldMemOperand(obj, HeapObject::kMapOffset)); lw(scratch, FieldMemOperand(obj, HeapObject::kMapOffset));
CompareMapAndBranch(scratch, map, early_success, cond, branch_to, mode);
}
void MacroAssembler::CompareMapAndBranch(Register obj_map,
Handle<Map> map,
Label* early_success,
Condition cond,
Label* branch_to,
CompareMapMode mode) {
Operand right = Operand(map); Operand right = Operand(map);
if (mode == ALLOW_ELEMENT_TRANSITION_MAPS) { if (mode == ALLOW_ELEMENT_TRANSITION_MAPS) {
ElementsKind kind = map->elements_kind(); ElementsKind kind = map->elements_kind();
...@@ -3483,13 +3493,13 @@ void MacroAssembler::CompareMapAndBranch(Register obj, ...@@ -3483,13 +3493,13 @@ void MacroAssembler::CompareMapAndBranch(Register obj,
kind = GetNextMoreGeneralFastElementsKind(kind, packed); kind = GetNextMoreGeneralFastElementsKind(kind, packed);
current_map = current_map->LookupElementsTransitionMap(kind); current_map = current_map->LookupElementsTransitionMap(kind);
if (!current_map) break; if (!current_map) break;
Branch(early_success, eq, scratch, right); Branch(early_success, eq, obj_map, right);
right = Operand(Handle<Map>(current_map)); right = Operand(Handle<Map>(current_map));
} }
} }
} }
Branch(branch_to, cond, scratch, right); Branch(branch_to, cond, obj_map, right);
} }
......
...@@ -992,6 +992,15 @@ class MacroAssembler: public Assembler { ...@@ -992,6 +992,15 @@ class MacroAssembler: public Assembler {
Label* branch_to, Label* branch_to,
CompareMapMode mode = REQUIRE_EXACT_MAP); CompareMapMode mode = REQUIRE_EXACT_MAP);
// As above, but the map of the object is already loaded into the register
// which is preserved by the code generated.
void CompareMapAndBranch(Register obj_map,
Handle<Map> map,
Label* early_success,
Condition cond,
Label* branch_to,
CompareMapMode mode = REQUIRE_EXACT_MAP);
// Check if the map of an object is equal to a specified map and branch to // Check if the map of an object is equal to a specified map and branch to
// label if not. Skip the smi check if not required (object is known to be a // label if not. Skip the smi check if not required (object is known to be a
// heap object). If mode is ALLOW_ELEMENT_TRANSITION_MAPS, then also match // heap object). If mode is ALLOW_ELEMENT_TRANSITION_MAPS, then also match
......
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