OLD | NEW |
1 // Copyright 2012 the V8 project authors. All rights reserved. | 1 // Copyright 2012 the V8 project authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "src/v8.h" | 5 #include "src/v8.h" |
6 | 6 |
7 #if V8_TARGET_ARCH_ARM | 7 #if V8_TARGET_ARCH_ARM |
8 | 8 |
9 #include "src/codegen.h" | 9 #include "src/codegen.h" |
10 #include "src/ic/ic.h" | 10 #include "src/ic/ic.h" |
(...skipping 747 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
758 __ LoadTransitionedArrayMapConditional(FAST_DOUBLE_ELEMENTS, FAST_ELEMENTS, | 758 __ LoadTransitionedArrayMapConditional(FAST_DOUBLE_ELEMENTS, FAST_ELEMENTS, |
759 receiver_map, r4, slow); | 759 receiver_map, r4, slow); |
760 mode = AllocationSite::GetMode(FAST_DOUBLE_ELEMENTS, FAST_ELEMENTS); | 760 mode = AllocationSite::GetMode(FAST_DOUBLE_ELEMENTS, FAST_ELEMENTS); |
761 ElementsTransitionGenerator::GenerateDoubleToObject( | 761 ElementsTransitionGenerator::GenerateDoubleToObject( |
762 masm, receiver, key, value, receiver_map, mode, slow); | 762 masm, receiver, key, value, receiver_map, mode, slow); |
763 __ ldr(elements, FieldMemOperand(receiver, JSObject::kElementsOffset)); | 763 __ ldr(elements, FieldMemOperand(receiver, JSObject::kElementsOffset)); |
764 __ jmp(&finish_object_store); | 764 __ jmp(&finish_object_store); |
765 } | 765 } |
766 | 766 |
767 | 767 |
768 void KeyedStoreIC::GenerateGeneric(MacroAssembler* masm, | 768 void KeyedStoreIC::GenerateGeneric( |
769 StrictMode strict_mode) { | 769 MacroAssembler* masm, StrictMode strict_mode, |
| 770 KeyedStoreStubCacheRequirement handler_requirement) { |
770 // ---------- S t a t e -------------- | 771 // ---------- S t a t e -------------- |
771 // -- r0 : value | 772 // -- r0 : value |
772 // -- r1 : key | 773 // -- r1 : key |
773 // -- r2 : receiver | 774 // -- r2 : receiver |
774 // -- lr : return address | 775 // -- lr : return address |
775 // ----------------------------------- | 776 // ----------------------------------- |
776 Label slow, fast_object, fast_object_grow; | 777 Label slow, fast_object, fast_object_grow; |
777 Label fast_double, fast_double_grow; | 778 Label fast_double, fast_double_grow; |
778 Label array, extra, check_if_double_array; | 779 Label array, extra, check_if_double_array, maybe_name_key, miss; |
779 | 780 |
780 // Register usage. | 781 // Register usage. |
781 Register value = StoreDescriptor::ValueRegister(); | 782 Register value = StoreDescriptor::ValueRegister(); |
782 Register key = StoreDescriptor::NameRegister(); | 783 Register key = StoreDescriptor::NameRegister(); |
783 Register receiver = StoreDescriptor::ReceiverRegister(); | 784 Register receiver = StoreDescriptor::ReceiverRegister(); |
784 DCHECK(receiver.is(r1)); | 785 DCHECK(receiver.is(r1)); |
785 DCHECK(key.is(r2)); | 786 DCHECK(key.is(r2)); |
786 DCHECK(value.is(r0)); | 787 DCHECK(value.is(r0)); |
787 Register receiver_map = r3; | 788 Register receiver_map = r3; |
788 Register elements_map = r6; | 789 Register elements_map = r6; |
789 Register elements = r9; // Elements array of the receiver. | 790 Register elements = r9; // Elements array of the receiver. |
790 // r4 and r5 are used as general scratch registers. | 791 // r4 and r5 are used as general scratch registers. |
791 | 792 |
792 // Check that the key is a smi. | 793 // Check that the key is a smi. |
793 __ JumpIfNotSmi(key, &slow); | 794 __ JumpIfNotSmi(key, &maybe_name_key); |
794 // Check that the object isn't a smi. | 795 // Check that the object isn't a smi. |
795 __ JumpIfSmi(receiver, &slow); | 796 __ JumpIfSmi(receiver, &slow); |
796 // Get the map of the object. | 797 // Get the map of the object. |
797 __ ldr(receiver_map, FieldMemOperand(receiver, HeapObject::kMapOffset)); | 798 __ ldr(receiver_map, FieldMemOperand(receiver, HeapObject::kMapOffset)); |
798 // Check that the receiver does not require access checks and is not observed. | 799 // Check that the receiver does not require access checks and is not observed. |
799 // The generic stub does not perform map checks or handle observed objects. | 800 // The generic stub does not perform map checks or handle observed objects. |
800 __ ldrb(ip, FieldMemOperand(receiver_map, Map::kBitFieldOffset)); | 801 __ ldrb(ip, FieldMemOperand(receiver_map, Map::kBitFieldOffset)); |
801 __ tst(ip, Operand(1 << Map::kIsAccessCheckNeeded | 1 << Map::kIsObserved)); | 802 __ tst(ip, Operand(1 << Map::kIsAccessCheckNeeded | 1 << Map::kIsObserved)); |
802 __ b(ne, &slow); | 803 __ b(ne, &slow); |
803 // Check if the object is a JS array or not. | 804 // Check if the object is a JS array or not. |
(...skipping 11 matching lines...) Expand all Loading... |
815 __ cmp(key, Operand(ip)); | 816 __ cmp(key, Operand(ip)); |
816 __ b(lo, &fast_object); | 817 __ b(lo, &fast_object); |
817 | 818 |
818 // Slow case, handle jump to runtime. | 819 // Slow case, handle jump to runtime. |
819 __ bind(&slow); | 820 __ bind(&slow); |
820 // Entry registers are intact. | 821 // Entry registers are intact. |
821 // r0: value. | 822 // r0: value. |
822 // r1: key. | 823 // r1: key. |
823 // r2: receiver. | 824 // r2: receiver. |
824 PropertyICCompiler::GenerateRuntimeSetProperty(masm, strict_mode); | 825 PropertyICCompiler::GenerateRuntimeSetProperty(masm, strict_mode); |
| 826 // Never returns to here. |
| 827 |
| 828 __ bind(&maybe_name_key); |
| 829 __ ldr(r4, FieldMemOperand(key, HeapObject::kMapOffset)); |
| 830 __ ldrb(r4, FieldMemOperand(r4, Map::kInstanceTypeOffset)); |
| 831 __ JumpIfNotUniqueNameInstanceType(r4, &slow); |
| 832 Code::Flags flags = Code::RemoveTypeAndHolderFromFlags( |
| 833 Code::ComputeHandlerFlags(Code::STORE_IC)); |
| 834 masm->isolate()->stub_cache()->GenerateProbe(masm, flags, false, receiver, |
| 835 key, r3, r4, r5, r6); |
| 836 // Cache miss. |
| 837 if (handler_requirement == kCallRuntimeOnMissingHandler) { |
| 838 __ b(&slow); |
| 839 } else { |
| 840 DCHECK(handler_requirement == kMissOnMissingHandler); |
| 841 __ b(&miss); |
| 842 } |
825 | 843 |
826 // Extra capacity case: Check if there is extra capacity to | 844 // Extra capacity case: Check if there is extra capacity to |
827 // perform the store and update the length. Used for adding one | 845 // perform the store and update the length. Used for adding one |
828 // element to the array by writing to array[array.length]. | 846 // element to the array by writing to array[array.length]. |
829 __ bind(&extra); | 847 __ bind(&extra); |
830 // Condition code from comparing key and array length is still available. | 848 // Condition code from comparing key and array length is still available. |
831 __ b(ne, &slow); // Only support writing to writing to array[array.length]. | 849 __ b(ne, &slow); // Only support writing to writing to array[array.length]. |
832 // Check for room in the elements backing store. | 850 // Check for room in the elements backing store. |
833 // Both the key and the length of FixedArray are smis. | 851 // Both the key and the length of FixedArray are smis. |
834 __ ldr(ip, FieldMemOperand(elements, FixedArray::kLengthOffset)); | 852 __ ldr(ip, FieldMemOperand(elements, FixedArray::kLengthOffset)); |
(...skipping 21 matching lines...) Expand all Loading... |
856 __ cmp(key, Operand(ip)); | 874 __ cmp(key, Operand(ip)); |
857 __ b(hs, &extra); | 875 __ b(hs, &extra); |
858 | 876 |
859 KeyedStoreGenerateGenericHelper( | 877 KeyedStoreGenerateGenericHelper( |
860 masm, &fast_object, &fast_double, &slow, kCheckMap, kDontIncrementLength, | 878 masm, &fast_object, &fast_double, &slow, kCheckMap, kDontIncrementLength, |
861 value, key, receiver, receiver_map, elements_map, elements); | 879 value, key, receiver, receiver_map, elements_map, elements); |
862 KeyedStoreGenerateGenericHelper(masm, &fast_object_grow, &fast_double_grow, | 880 KeyedStoreGenerateGenericHelper(masm, &fast_object_grow, &fast_double_grow, |
863 &slow, kDontCheckMap, kIncrementLength, value, | 881 &slow, kDontCheckMap, kIncrementLength, value, |
864 key, receiver, receiver_map, elements_map, | 882 key, receiver, receiver_map, elements_map, |
865 elements); | 883 elements); |
| 884 |
| 885 __ bind(&miss); |
| 886 GenerateMiss(masm); |
866 } | 887 } |
867 | 888 |
868 | 889 |
869 void StoreIC::GenerateMegamorphic(MacroAssembler* masm) { | 890 void StoreIC::GenerateMegamorphic(MacroAssembler* masm) { |
870 Register receiver = StoreDescriptor::ReceiverRegister(); | 891 Register receiver = StoreDescriptor::ReceiverRegister(); |
871 Register name = StoreDescriptor::NameRegister(); | 892 Register name = StoreDescriptor::NameRegister(); |
872 DCHECK(receiver.is(r1)); | 893 DCHECK(receiver.is(r1)); |
873 DCHECK(name.is(r2)); | 894 DCHECK(name.is(r2)); |
874 DCHECK(StoreDescriptor::ValueRegister().is(r0)); | 895 DCHECK(StoreDescriptor::ValueRegister().is(r0)); |
875 | 896 |
(...skipping 134 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
1010 patcher.EmitCondition(ne); | 1031 patcher.EmitCondition(ne); |
1011 } else { | 1032 } else { |
1012 DCHECK(Assembler::GetCondition(branch_instr) == ne); | 1033 DCHECK(Assembler::GetCondition(branch_instr) == ne); |
1013 patcher.EmitCondition(eq); | 1034 patcher.EmitCondition(eq); |
1014 } | 1035 } |
1015 } | 1036 } |
1016 } | 1037 } |
1017 } // namespace v8::internal | 1038 } // namespace v8::internal |
1018 | 1039 |
1019 #endif // V8_TARGET_ARCH_ARM | 1040 #endif // V8_TARGET_ARCH_ARM |
OLD | NEW |