OLD | NEW |
1 // Copyright 2012 the V8 project authors. All rights reserved. | 1 // Copyright 2012 the V8 project authors. All rights reserved. |
2 // Redistribution and use in source and binary forms, with or without | 2 // Redistribution and use in source and binary forms, with or without |
3 // modification, are permitted provided that the following conditions are | 3 // modification, are permitted provided that the following conditions are |
4 // met: | 4 // met: |
5 // | 5 // |
6 // * Redistributions of source code must retain the above copyright | 6 // * Redistributions of source code must retain the above copyright |
7 // notice, this list of conditions and the following disclaimer. | 7 // notice, this list of conditions and the following disclaimer. |
8 // * Redistributions in binary form must reproduce the above | 8 // * Redistributions in binary form must reproduce the above |
9 // copyright notice, this list of conditions and the following | 9 // copyright notice, this list of conditions and the following |
10 // disclaimer in the documentation and/or other materials provided | 10 // disclaimer in the documentation and/or other materials provided |
(...skipping 712 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
723 __ bind(label); | 723 __ bind(label); |
724 __ mov(this->name(), Immediate(name)); | 724 __ mov(this->name(), Immediate(name)); |
725 } | 725 } |
726 } | 726 } |
727 | 727 |
728 | 728 |
729 // Both name_reg and receiver_reg are preserved on jumps to miss_label, | 729 // Both name_reg and receiver_reg are preserved on jumps to miss_label, |
730 // but may be destroyed if store is successful. | 730 // but may be destroyed if store is successful. |
731 void StubCompiler::GenerateStoreField(MacroAssembler* masm, | 731 void StubCompiler::GenerateStoreField(MacroAssembler* masm, |
732 Handle<JSObject> object, | 732 Handle<JSObject> object, |
733 int index, | 733 LookupResult* lookup, |
734 Handle<Map> transition, | 734 Handle<Map> transition, |
735 Handle<Name> name, | 735 Handle<Name> name, |
736 Register receiver_reg, | 736 Register receiver_reg, |
737 Register name_reg, | 737 Register name_reg, |
738 Register value_reg, | 738 Register value_reg, |
739 Register scratch1, | 739 Register scratch1, |
740 Register scratch2, | 740 Register scratch2, |
741 Label* miss_label, | 741 Label* miss_label, |
742 Label* miss_restore_name) { | 742 Label* miss_restore_name) { |
743 LookupResult lookup(masm->isolate()); | |
744 object->Lookup(*name, &lookup); | |
745 if (lookup.IsFound() && (lookup.IsReadOnly() || !lookup.IsCacheable())) { | |
746 // In sloppy mode, we could just return the value and be done. However, we | |
747 // might be in strict mode, where we have to throw. Since we cannot tell, | |
748 // go into slow case unconditionally. | |
749 __ jmp(miss_label); | |
750 return; | |
751 } | |
752 | |
753 // Check that the map of the object hasn't changed. | 743 // Check that the map of the object hasn't changed. |
754 CompareMapMode mode = transition.is_null() ? ALLOW_ELEMENT_TRANSITION_MAPS | 744 CompareMapMode mode = transition.is_null() ? ALLOW_ELEMENT_TRANSITION_MAPS |
755 : REQUIRE_EXACT_MAP; | 745 : REQUIRE_EXACT_MAP; |
756 __ CheckMap(receiver_reg, Handle<Map>(object->map()), | 746 __ CheckMap(receiver_reg, Handle<Map>(object->map()), |
757 miss_label, DO_SMI_CHECK, mode); | 747 miss_label, DO_SMI_CHECK, mode); |
758 | 748 |
759 // Perform global security token check if needed. | 749 // Perform global security token check if needed. |
760 if (object->IsJSGlobalProxy()) { | 750 if (object->IsJSGlobalProxy()) { |
761 __ CheckAccessGlobalProxy(receiver_reg, scratch1, miss_label); | 751 __ CheckAccessGlobalProxy(receiver_reg, scratch1, miss_label); |
762 } | 752 } |
763 | 753 |
764 // Check that we are allowed to write this. | 754 // Check that we are allowed to write this. |
765 if (!transition.is_null() && object->GetPrototype()->IsJSObject()) { | 755 if (!transition.is_null() && object->GetPrototype()->IsJSObject()) { |
766 JSObject* holder; | 756 JSObject* holder; |
767 if (lookup.IsFound()) { | 757 if (lookup->IsFound()) { |
768 holder = lookup.holder(); | 758 holder = lookup->holder(); |
769 } else { | 759 } else { |
770 // Find the top object. | 760 // Find the top object. |
771 holder = *object; | 761 holder = *object; |
772 do { | 762 do { |
773 holder = JSObject::cast(holder->GetPrototype()); | 763 holder = JSObject::cast(holder->GetPrototype()); |
774 } while (holder->GetPrototype()->IsJSObject()); | 764 } while (holder->GetPrototype()->IsJSObject()); |
775 } | 765 } |
776 // We need an extra register, push | 766 // We need an extra register, push |
777 CheckPrototypes(object, receiver_reg, Handle<JSObject>(holder), name_reg, | 767 CheckPrototypes(object, receiver_reg, Handle<JSObject>(holder), name_reg, |
778 scratch1, scratch2, name, miss_restore_name); | 768 scratch1, scratch2, name, miss_restore_name); |
(...skipping 13 matching lines...) Expand all Loading... |
792 __ push(eax); | 782 __ push(eax); |
793 __ push(scratch1); | 783 __ push(scratch1); |
794 __ TailCallExternalReference( | 784 __ TailCallExternalReference( |
795 ExternalReference(IC_Utility(IC::kSharedStoreIC_ExtendStorage), | 785 ExternalReference(IC_Utility(IC::kSharedStoreIC_ExtendStorage), |
796 masm->isolate()), | 786 masm->isolate()), |
797 3, | 787 3, |
798 1); | 788 1); |
799 return; | 789 return; |
800 } | 790 } |
801 | 791 |
| 792 int index; |
802 if (!transition.is_null()) { | 793 if (!transition.is_null()) { |
803 // Update the map of the object. | 794 // Update the map of the object. |
804 __ mov(scratch1, Immediate(transition)); | 795 __ mov(scratch1, Immediate(transition)); |
805 __ mov(FieldOperand(receiver_reg, HeapObject::kMapOffset), scratch1); | 796 __ mov(FieldOperand(receiver_reg, HeapObject::kMapOffset), scratch1); |
806 | 797 |
807 // Update the write barrier for the map field and pass the now unused | 798 // Update the write barrier for the map field and pass the now unused |
808 // name_reg as scratch register. | 799 // name_reg as scratch register. |
809 __ RecordWriteField(receiver_reg, | 800 __ RecordWriteField(receiver_reg, |
810 HeapObject::kMapOffset, | 801 HeapObject::kMapOffset, |
811 scratch1, | 802 scratch1, |
812 name_reg, | 803 name_reg, |
813 kDontSaveFPRegs, | 804 kDontSaveFPRegs, |
814 OMIT_REMEMBERED_SET, | 805 OMIT_REMEMBERED_SET, |
815 OMIT_SMI_CHECK); | 806 OMIT_SMI_CHECK); |
| 807 index = transition->instance_descriptors()->GetFieldIndex( |
| 808 transition->LastAdded()); |
| 809 } else { |
| 810 index = lookup->GetFieldIndex().field_index(); |
816 } | 811 } |
817 | 812 |
| 813 |
818 // Adjust for the number of properties stored in the object. Even in the | 814 // Adjust for the number of properties stored in the object. Even in the |
819 // face of a transition we can use the old map here because the size of the | 815 // face of a transition we can use the old map here because the size of the |
820 // object and the number of in-object properties is not going to change. | 816 // object and the number of in-object properties is not going to change. |
821 index -= object->map()->inobject_properties(); | 817 index -= object->map()->inobject_properties(); |
822 | 818 |
823 if (index < 0) { | 819 if (index < 0) { |
824 // Set the property straight into the object. | 820 // Set the property straight into the object. |
825 int offset = object->map()->instance_size() + (index * kPointerSize); | 821 int offset = object->map()->instance_size() + (index * kPointerSize); |
826 __ mov(FieldOperand(receiver_reg, offset), value_reg); | 822 __ mov(FieldOperand(receiver_reg, offset), value_reg); |
827 | 823 |
(...skipping 2785 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
3613 TailCallBuiltin(masm, Builtins::kKeyedStoreIC_Slow); | 3609 TailCallBuiltin(masm, Builtins::kKeyedStoreIC_Slow); |
3614 } | 3610 } |
3615 } | 3611 } |
3616 | 3612 |
3617 | 3613 |
3618 #undef __ | 3614 #undef __ |
3619 | 3615 |
3620 } } // namespace v8::internal | 3616 } } // namespace v8::internal |
3621 | 3617 |
3622 #endif // V8_TARGET_ARCH_IA32 | 3618 #endif // V8_TARGET_ARCH_IA32 |
OLD | NEW |