 Chromium Code Reviews
 Chromium Code Reviews Issue 6708085:
  Enable GVN for polymorphic loads by not expanding them at the HIR level.  (Closed) 
  Base URL: http://v8.googlecode.com/svn/branches/bleeding_edge/
    
  
    Issue 6708085:
  Enable GVN for polymorphic loads by not expanding them at the HIR level.  (Closed) 
  Base URL: http://v8.googlecode.com/svn/branches/bleeding_edge/| Index: src/ia32/lithium-codegen-ia32.cc | 
| =================================================================== | 
| --- src/ia32/lithium-codegen-ia32.cc (revision 7299) | 
| +++ src/ia32/lithium-codegen-ia32.cc (working copy) | 
| @@ -2122,6 +2122,58 @@ | 
| } | 
| +void LCodeGen::EmitLoadField(Register result, | 
| + Register object, | 
| 
Kevin Millikin (Chromium)
2011/03/23 08:08:53
Indentation is off here.
 
fschneider
2011/03/23 13:57:56
Done.
 | 
| + Handle<Map> type, | 
| + LookupResult* lookup) { | 
| + ASSERT(lookup->IsProperty() && lookup->type() == FIELD); | 
| + int index = lookup->GetLocalFieldIndexFromMap(*type); | 
| + int offset = index * kPointerSize; | 
| + if (index < 0) { | 
| + // Negative property indices are in-object properties, indexed | 
| + // from the end of the fixed part of the object. | 
| + __ mov(result, FieldOperand(object, offset + type->instance_size())); | 
| + } else { | 
| + // Non-negative property indices are in the properties array. | 
| + __ mov(result, FieldOperand(object, JSObject::kPropertiesOffset)); | 
| + __ mov(result, FieldOperand(result, offset + FixedArray::kHeaderSize)); | 
| + } | 
| +} | 
| + | 
| + | 
| +void LCodeGen::DoLoadNamedFieldPolymorphic(LLoadNamedFieldPolymorphic* instr) { | 
| + Register object = ToRegister(instr->InputAt(0)); | 
| 
Kevin Millikin (Chromium)
2011/03/23 08:08:53
I don't like relying on the order of the operands.
 
fschneider
2011/03/23 13:57:56
Done.
 | 
| + Register result = ToRegister(instr->result()); | 
| + NearLabel done; | 
| + int num_maps = instr->hydrogen()->types()->length(); | 
| 
Kevin Millikin (Chromium)
2011/03/23 08:08:53
map_count is only one character longer and avoids
 
fschneider
2011/03/23 13:57:56
Done.
 | 
| + for (int i = 0; i < num_maps; ++i) { | 
| + Handle<Map> map = instr->hydrogen()->types()->at(i); | 
| + LookupResult lookup; | 
| + map->LookupInDescriptors(NULL, *instr->hydrogen()->name(), &lookup); | 
| + | 
| + NearLabel next; | 
| + __ cmp(FieldOperand(object, HeapObject::kMapOffset), map); | 
| 
Kevin Millikin (Chromium)
2011/03/23 08:08:53
We could avoid multiple loads of the map by using
 
fschneider
2011/03/23 13:57:56
I'll leave it as is in this change. Another way wo
 
Kevin Millikin (Chromium)
2011/03/23 14:17:04
OK, leaving it for later sounds good to me.
 | 
| + if (i == num_maps - 1 && !instr->hydrogen()->need_generic()) { | 
| + DeoptimizeIf(not_equal, instr->environment()); | 
| + } else { | 
| + __ j(not_equal, &next); | 
| + } | 
| + EmitLoadField(result, object, map, &lookup); | 
| + if (i < num_maps - 1 || instr->hydrogen()->need_generic()) { | 
| 
Kevin Millikin (Chromium)
2011/03/23 08:08:53
If you handle all but the last map in the loop, I
 
fschneider
2011/03/23 13:57:56
Done.
 | 
| + __ jmp(&done); | 
| + __ bind(&next); | 
| + } | 
| + } | 
| + if (instr->hydrogen()->need_generic()) { | 
| + __ mov(ecx, instr->hydrogen()->name()); | 
| + Handle<Code> ic( | 
| + isolate()->builtins()->builtin(Builtins::LoadIC_Initialize)); | 
| + CallCode(ic, RelocInfo::CODE_TARGET, instr, false); | 
| + } | 
| + __ bind(&done); | 
| +} | 
| + | 
| + | 
| void LCodeGen::DoLoadNamedGeneric(LLoadNamedGeneric* instr) { | 
| ASSERT(ToRegister(instr->context()).is(esi)); | 
| ASSERT(ToRegister(instr->object()).is(eax)); |