| 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/elements.h" | 5 #include "src/elements.h" |
| 6 | 6 |
| 7 #include "src/arguments.h" | 7 #include "src/arguments.h" |
| 8 #include "src/conversions.h" | 8 #include "src/conversions.h" |
| 9 #include "src/factory.h" | 9 #include "src/factory.h" |
| 10 #include "src/messages.h" | 10 #include "src/messages.h" |
| (...skipping 79 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 90 #define ELEMENTS_TRAITS(Class, KindParam, Store) \ | 90 #define ELEMENTS_TRAITS(Class, KindParam, Store) \ |
| 91 template<> class ElementsKindTraits<KindParam> { \ | 91 template<> class ElementsKindTraits<KindParam> { \ |
| 92 public: /* NOLINT */ \ | 92 public: /* NOLINT */ \ |
| 93 static const ElementsKind Kind = KindParam; \ | 93 static const ElementsKind Kind = KindParam; \ |
| 94 typedef Store BackingStore; \ | 94 typedef Store BackingStore; \ |
| 95 }; | 95 }; |
| 96 ELEMENTS_LIST(ELEMENTS_TRAITS) | 96 ELEMENTS_LIST(ELEMENTS_TRAITS) |
| 97 #undef ELEMENTS_TRAITS | 97 #undef ELEMENTS_TRAITS |
| 98 | 98 |
| 99 | 99 |
| 100 static bool HasIndex(Handle<FixedArray> array, Handle<Object> index_handle) { | |
| 101 DisallowHeapAllocation no_gc; | |
| 102 Object* index = *index_handle; | |
| 103 int len0 = array->length(); | |
| 104 for (int i = 0; i < len0; i++) { | |
| 105 Object* element = array->get(i); | |
| 106 if (index->KeyEquals(element)) return true; | |
| 107 } | |
| 108 return false; | |
| 109 } | |
| 110 | |
| 111 | |
| 112 MUST_USE_RESULT | 100 MUST_USE_RESULT |
| 113 MaybeHandle<Object> ThrowArrayLengthRangeError(Isolate* isolate) { | 101 MaybeHandle<Object> ThrowArrayLengthRangeError(Isolate* isolate) { |
| 114 THROW_NEW_ERROR(isolate, NewRangeError(MessageTemplate::kInvalidArrayLength), | 102 THROW_NEW_ERROR(isolate, NewRangeError(MessageTemplate::kInvalidArrayLength), |
| 115 Object); | 103 Object); |
| 116 } | 104 } |
| 117 | 105 |
| 118 | 106 |
| 119 void CopyObjectToObjectElements(FixedArrayBase* from_base, | 107 void CopyObjectToObjectElements(FixedArrayBase* from_base, |
| 120 ElementsKind from_kind, uint32_t from_start, | 108 ElementsKind from_kind, uint32_t from_start, |
| 121 FixedArrayBase* to_base, ElementsKind to_kind, | 109 FixedArrayBase* to_base, ElementsKind to_kind, |
| (...skipping 730 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 852 // intentionally to avoid ArrayConcat() builtin performance degradation. | 840 // intentionally to avoid ArrayConcat() builtin performance degradation. |
| 853 // | 841 // |
| 854 // Details: The idea is that allocations actually happen only in case of | 842 // Details: The idea is that allocations actually happen only in case of |
| 855 // copying from object with fast double elements to object with object | 843 // copying from object with fast double elements to object with object |
| 856 // elements. In all the other cases there are no allocations performed and | 844 // elements. In all the other cases there are no allocations performed and |
| 857 // handle creation causes noticeable performance degradation of the builtin. | 845 // handle creation causes noticeable performance degradation of the builtin. |
| 858 ElementsAccessorSubclass::CopyElementsImpl( | 846 ElementsAccessorSubclass::CopyElementsImpl( |
| 859 from, from_start, *to, from_kind, to_start, packed_size, copy_size); | 847 from, from_start, *to, from_kind, to_start, packed_size, copy_size); |
| 860 } | 848 } |
| 861 | 849 |
| 862 virtual Handle<FixedArray> AddElementsToFixedArray( | 850 virtual void AddElementsToFixedArrayWithAccumulator( |
| 863 Handle<JSObject> receiver, Handle<FixedArray> to, | 851 Handle<JSObject> receiver, KeyAccumulator* accumulator, |
| 864 FixedArray::KeyFilter filter) final { | 852 FixedArray::KeyFilter filter) final { |
| 865 Handle<FixedArrayBase> from(receiver->elements()); | 853 Handle<FixedArrayBase> from(receiver->elements()); |
| 866 | 854 uint32_t add_length = |
| 867 int len0 = to->length(); | 855 ElementsAccessorSubclass::GetCapacityImpl(*receiver, *from); |
| 868 #ifdef ENABLE_SLOW_DCHECKS | 856 if (add_length == 0) return; |
| 869 if (FLAG_enable_slow_asserts) { | 857 accumulator->PrepareForComparisons(add_length); |
| 870 for (int i = 0; i < len0; i++) { | 858 int prev_key_count = accumulator->GetLength(); |
| 871 DCHECK(!to->get(i)->IsTheHole()); | 859 for (uint32_t i = 0; i < add_length; i++) { |
| 860 if (!ElementsAccessorSubclass::HasEntryImpl(*from, i)) continue; |
| 861 Handle<Object> value = ElementsAccessorSubclass::GetImpl(from, i); |
| 862 DCHECK(!value->IsAccessorPair()); |
| 863 DCHECK(!value->IsExecutableAccessorInfo()); |
| 864 if (filter == FixedArray::NON_SYMBOL_KEYS && value->IsSymbol()) { |
| 865 continue; |
| 866 } |
| 867 if (!value->IsTheHole() && !accumulator->HasKey(value, prev_key_count)) { |
| 868 accumulator->AddKey(value); |
| 872 } | 869 } |
| 873 } | 870 } |
| 874 #endif | |
| 875 | |
| 876 // Optimize if 'other' is empty. | |
| 877 // We cannot optimize if 'this' is empty, as other may have holes. | |
| 878 uint32_t len1 = ElementsAccessorSubclass::GetCapacityImpl(*receiver, *from); | |
| 879 if (len1 == 0) return to; | |
| 880 | |
| 881 Isolate* isolate = from->GetIsolate(); | |
| 882 | |
| 883 // Compute how many elements are not in other. | |
| 884 uint32_t extra = 0; | |
| 885 for (uint32_t y = 0; y < len1; y++) { | |
| 886 if (ElementsAccessorSubclass::HasEntryImpl(*from, y)) { | |
| 887 Handle<Object> value = ElementsAccessorSubclass::GetImpl(from, y); | |
| 888 | |
| 889 DCHECK(!value->IsTheHole()); | |
| 890 DCHECK(!value->IsAccessorPair()); | |
| 891 DCHECK(!value->IsExecutableAccessorInfo()); | |
| 892 if (filter == FixedArray::NON_SYMBOL_KEYS && value->IsSymbol()) { | |
| 893 continue; | |
| 894 } | |
| 895 if (!HasIndex(to, value)) { | |
| 896 extra++; | |
| 897 } | |
| 898 } | |
| 899 } | |
| 900 | |
| 901 if (extra == 0) return to; | |
| 902 | |
| 903 // Allocate the result | |
| 904 Handle<FixedArray> result = isolate->factory()->NewFixedArray(len0 + extra); | |
| 905 | |
| 906 // Fill in the content | |
| 907 { | |
| 908 DisallowHeapAllocation no_gc; | |
| 909 WriteBarrierMode mode = result->GetWriteBarrierMode(no_gc); | |
| 910 for (int i = 0; i < len0; i++) { | |
| 911 Object* e = to->get(i); | |
| 912 DCHECK(e->IsString() || e->IsNumber()); | |
| 913 result->set(i, e, mode); | |
| 914 } | |
| 915 } | |
| 916 // Fill in the extra values. | |
| 917 uint32_t entry = 0; | |
| 918 for (uint32_t y = 0; y < len1; y++) { | |
| 919 if (ElementsAccessorSubclass::HasEntryImpl(*from, y)) { | |
| 920 Handle<Object> value = ElementsAccessorSubclass::GetImpl(from, y); | |
| 921 DCHECK(!value->IsAccessorPair()); | |
| 922 DCHECK(!value->IsExecutableAccessorInfo()); | |
| 923 if (filter == FixedArray::NON_SYMBOL_KEYS && value->IsSymbol()) { | |
| 924 continue; | |
| 925 } | |
| 926 if (!value->IsTheHole() && !HasIndex(to, value)) { | |
| 927 result->set(len0 + entry, *value); | |
| 928 entry++; | |
| 929 } | |
| 930 } | |
| 931 } | |
| 932 DCHECK(extra == entry); | |
| 933 return result; | |
| 934 } | 871 } |
| 935 | 872 |
| 936 static uint32_t GetCapacityImpl(JSObject* holder, | 873 static uint32_t GetCapacityImpl(JSObject* holder, |
| 937 FixedArrayBase* backing_store) { | 874 FixedArrayBase* backing_store) { |
| 938 return backing_store->length(); | 875 return backing_store->length(); |
| 939 } | 876 } |
| 940 | 877 |
| 941 uint32_t GetCapacity(JSObject* holder, FixedArrayBase* backing_store) final { | 878 uint32_t GetCapacity(JSObject* holder, FixedArrayBase* backing_store) final { |
| 942 return ElementsAccessorSubclass::GetCapacityImpl(holder, backing_store); | 879 return ElementsAccessorSubclass::GetCapacityImpl(holder, backing_store); |
| 943 } | 880 } |
| (...skipping 1465 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 2409 } | 2346 } |
| 2410 } | 2347 } |
| 2411 | 2348 |
| 2412 DCHECK(j == result_len); | 2349 DCHECK(j == result_len); |
| 2413 return result_array; | 2350 return result_array; |
| 2414 } | 2351 } |
| 2415 | 2352 |
| 2416 ElementsAccessor** ElementsAccessor::elements_accessors_ = NULL; | 2353 ElementsAccessor** ElementsAccessor::elements_accessors_ = NULL; |
| 2417 } // namespace internal | 2354 } // namespace internal |
| 2418 } // namespace v8 | 2355 } // namespace v8 |
| OLD | NEW |