Index: src/mips/stub-cache-mips.cc |
diff --git a/src/mips/stub-cache-mips.cc b/src/mips/stub-cache-mips.cc |
index cfa43a03ca2d8e5e69f62014f9b9bacd46ed2ef6..995ebf789c3743d6d3783e934640a0a0f31e119e 100644 |
--- a/src/mips/stub-cache-mips.cc |
+++ b/src/mips/stub-cache-mips.cc |
@@ -429,29 +429,27 @@ static void GenerateCheckPropertyCell(MacroAssembler* masm, |
} |
-// Generate StoreField code, value is passed in a0 register. |
+// Generate StoreTransition code, value is passed in a0 register. |
// After executing generated code, the receiver_reg and name_reg |
// may be clobbered. |
-void StubCompiler::GenerateStoreField(MacroAssembler* masm, |
- Handle<JSObject> object, |
- LookupResult* lookup, |
- Handle<Map> transition, |
- Handle<Name> name, |
- Register receiver_reg, |
- Register name_reg, |
- Register value_reg, |
- Register scratch1, |
- Register scratch2, |
- Label* miss_label, |
- Label* miss_restore_name) { |
+void StubCompiler::GenerateStoreTransition(MacroAssembler* masm, |
+ Handle<JSObject> object, |
+ LookupResult* lookup, |
+ Handle<Map> transition, |
+ Handle<Name> name, |
+ Register receiver_reg, |
+ Register name_reg, |
+ Register value_reg, |
+ Register scratch1, |
+ Register scratch2, |
+ Label* miss_label, |
+ Label* miss_restore_name) { |
// a0 : value. |
Label exit; |
// Check that the map of the object hasn't changed. |
- CompareMapMode mode = transition.is_null() ? ALLOW_ELEMENT_TRANSITION_MAPS |
- : REQUIRE_EXACT_MAP; |
__ CheckMap(receiver_reg, scratch1, Handle<Map>(object->map()), miss_label, |
- DO_SMI_CHECK, mode); |
+ DO_SMI_CHECK, REQUIRE_EXACT_MAP); |
// Perform global security token check if needed. |
if (object->IsJSGlobalProxy()) { |
@@ -459,7 +457,7 @@ void StubCompiler::GenerateStoreField(MacroAssembler* masm, |
} |
// Check that we are allowed to write this. |
- if (!transition.is_null() && object->GetPrototype()->IsJSObject()) { |
+ if (object->GetPrototype()->IsJSObject()) { |
JSObject* holder; |
// holder == object indicates that no property was found. |
if (lookup->holder() != *object) { |
@@ -497,7 +495,7 @@ void StubCompiler::GenerateStoreField(MacroAssembler* masm, |
ASSERT(object->IsJSGlobalProxy() || !object->IsAccessCheckNeeded()); |
// Perform map transition for the receiver if necessary. |
- if (!transition.is_null() && (object->map()->unused_property_fields() == 0)) { |
+ if (object->map()->unused_property_fields() == 0) { |
// The properties must be extended before we can store the value. |
// We jump to a runtime call that extends the properties array. |
__ push(receiver_reg); |
@@ -510,33 +508,114 @@ void StubCompiler::GenerateStoreField(MacroAssembler* masm, |
return; |
} |
- int index; |
- if (!transition.is_null()) { |
- // Update the map of the object. |
- __ li(scratch1, Operand(transition)); |
- __ sw(scratch1, FieldMemOperand(receiver_reg, HeapObject::kMapOffset)); |
+ // Update the map of the object. |
+ __ li(scratch1, Operand(transition)); |
+ __ sw(scratch1, FieldMemOperand(receiver_reg, HeapObject::kMapOffset)); |
+ |
+ // Update the write barrier for the map field and pass the now unused |
+ // name_reg as scratch register. |
+ __ RecordWriteField(receiver_reg, |
+ HeapObject::kMapOffset, |
+ scratch1, |
+ name_reg, |
+ kRAHasNotBeenSaved, |
+ kDontSaveFPRegs, |
+ OMIT_REMEMBERED_SET, |
+ OMIT_SMI_CHECK); |
+ |
+ int index = transition->instance_descriptors()->GetFieldIndex( |
+ transition->LastAdded()); |
+ |
+ // Adjust for the number of properties stored in the object. Even in the |
+ // face of a transition we can use the old map here because the size of the |
+ // object and the number of in-object properties is not going to change. |
+ index -= object->map()->inobject_properties(); |
+ |
+ // TODO(verwaest): Share this code as a code stub. |
+ if (index < 0) { |
+ // Set the property straight into the object. |
+ int offset = object->map()->instance_size() + (index * kPointerSize); |
+ __ sw(value_reg, FieldMemOperand(receiver_reg, offset)); |
+ |
+ // Skip updating write barrier if storing a smi. |
+ __ JumpIfSmi(value_reg, &exit); |
- // Update the write barrier for the map field and pass the now unused |
- // name_reg as scratch register. |
+ // Update the write barrier for the array address. |
+ // Pass the now unused name_reg as a scratch register. |
+ __ mov(name_reg, value_reg); |
__ RecordWriteField(receiver_reg, |
- HeapObject::kMapOffset, |
- scratch1, |
+ offset, |
name_reg, |
+ scratch1, |
kRAHasNotBeenSaved, |
- kDontSaveFPRegs, |
- OMIT_REMEMBERED_SET, |
- OMIT_SMI_CHECK); |
- index = transition->instance_descriptors()->GetFieldIndex( |
- transition->LastAdded()); |
+ kDontSaveFPRegs); |
} else { |
- index = lookup->GetFieldIndex().field_index(); |
+ // Write to the properties array. |
+ int offset = index * kPointerSize + FixedArray::kHeaderSize; |
+ // Get the properties array |
+ __ lw(scratch1, |
+ FieldMemOperand(receiver_reg, JSObject::kPropertiesOffset)); |
+ __ sw(value_reg, FieldMemOperand(scratch1, offset)); |
+ |
+ // Skip updating write barrier if storing a smi. |
+ __ JumpIfSmi(value_reg, &exit); |
+ |
+ // Update the write barrier for the array address. |
+ // Ok to clobber receiver_reg and name_reg, since we return. |
+ __ mov(name_reg, value_reg); |
+ __ RecordWriteField(scratch1, |
+ offset, |
+ name_reg, |
+ receiver_reg, |
+ kRAHasNotBeenSaved, |
+ kDontSaveFPRegs); |
} |
+ // Return the value (register v0). |
+ ASSERT(value_reg.is(a0)); |
+ __ bind(&exit); |
+ __ mov(v0, a0); |
+ __ Ret(); |
Paul Lind
2013/04/11 23:09:15
Please note (for the future) that this is a small
|
+} |
+ |
+ |
+// Generate StoreField code, value is passed in a0 register. |
+// When leaving generated code after success, the receiver_reg and name_reg |
+// may be clobbered. Upon branch to miss_label, the receiver and name |
+// registers have their original values. |
+void StubCompiler::GenerateStoreField(MacroAssembler* masm, |
+ Handle<JSObject> object, |
+ LookupResult* lookup, |
+ Register receiver_reg, |
+ Register name_reg, |
+ Register value_reg, |
+ Register scratch1, |
+ Register scratch2, |
+ Label* miss_label) { |
+ // a0 : value |
+ Label exit; |
+ |
+ // Check that the map of the object hasn't changed. |
+ __ CheckMap(receiver_reg, scratch1, Handle<Map>(object->map()), miss_label, |
+ DO_SMI_CHECK, ALLOW_ELEMENT_TRANSITION_MAPS); |
+ |
+ // Perform global security token check if needed. |
+ if (object->IsJSGlobalProxy()) { |
+ __ CheckAccessGlobalProxy(receiver_reg, scratch1, miss_label); |
+ } |
+ |
+ // Stub never generated for non-global objects that require access |
+ // checks. |
+ ASSERT(object->IsJSGlobalProxy() || !object->IsAccessCheckNeeded()); |
+ |
+ int index = lookup->GetFieldIndex().field_index(); |
+ |
// Adjust for the number of properties stored in the object. Even in the |
// face of a transition we can use the old map here because the size of the |
// object and the number of in-object properties is not going to change. |
index -= object->map()->inobject_properties(); |
+ // TODO(verwaest): Share this code as a code stub. |
if (index < 0) { |
// Set the property straight into the object. |
int offset = object->map()->instance_size() + (index * kPointerSize); |