mirror of
				https://github.com/LadybirdBrowser/ladybird.git
				synced 2025-10-24 17:09:43 +00:00 
			
		
		
		
	Before this change, PropertyNameIterator (used by for..in) and `Object::enumerable_own_property_names()` (used by `Object.keys()`, `Object.values()`, and `Object.entries()`) enumerated an object's own enumerable properties exactly as the spec prescribes: - Call `internal_own_property_keys()`, allocating a list of JS::Value keys. - For each key, call internal_get_own_property() to obtain a descriptor and check `[[Enumerable]]`. While that is required in the general case (e.g. for Proxy objects or platform/exotic objects that override `[[OwnPropertyKeys]]`), it's overkill for ordinary JS objects that store their own properties in the shape table and indexed-properties storage. This change introduces `for_each_own_property_with_enumerability()`, which, for objects where `eligible_for_own_property_enumeration_fast_path()` is `true`, lets us read the enumerability directly from shape metadata (and from indexed-properties storage) without a per-property descriptor lookup. When we cannot avoid `internal_get_own_property()`, we still benefit by skipping the temporary `Vector<Value>` of keys and avoiding the unnecessary round-trip between PropertyKey and Value.
		
			
				
	
	
		
			206 lines
		
	
	
	
		
			6.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			206 lines
		
	
	
	
		
			6.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2020, Matthew Olsson <mattco@serenityos.org>
 | |
|  *
 | |
|  * SPDX-License-Identifier: BSD-2-Clause
 | |
|  */
 | |
| 
 | |
| #pragma once
 | |
| 
 | |
| #include <AK/NonnullOwnPtr.h>
 | |
| #include <LibJS/Export.h>
 | |
| #include <LibJS/Runtime/Shape.h>
 | |
| #include <LibJS/Runtime/Value.h>
 | |
| 
 | |
| namespace JS {
 | |
| 
 | |
| struct ValueAndAttributes {
 | |
|     Value value;
 | |
|     PropertyAttributes attributes { default_attributes };
 | |
| 
 | |
|     Optional<u32> property_offset {};
 | |
| };
 | |
| 
 | |
| class IndexedProperties;
 | |
| class IndexedPropertyIterator;
 | |
| class GenericIndexedPropertyStorage;
 | |
| 
 | |
| class IndexedPropertyStorage {
 | |
| public:
 | |
|     virtual ~IndexedPropertyStorage() = default;
 | |
| 
 | |
|     enum class IsSimpleStorage {
 | |
|         No,
 | |
|         Yes,
 | |
|     };
 | |
| 
 | |
|     virtual bool has_index(u32 index) const = 0;
 | |
|     virtual Optional<ValueAndAttributes> get(u32 index) const = 0;
 | |
|     virtual void put(u32 index, Value value, PropertyAttributes attributes = default_attributes) = 0;
 | |
|     virtual void remove(u32 index) = 0;
 | |
| 
 | |
|     virtual ValueAndAttributes take_first() = 0;
 | |
|     virtual ValueAndAttributes take_last() = 0;
 | |
| 
 | |
|     virtual size_t size() const = 0;
 | |
|     size_t array_like_size() const { return m_array_size; }
 | |
|     virtual bool set_array_like_size(size_t new_size) = 0;
 | |
| 
 | |
|     bool is_simple_storage() const { return m_is_simple_storage; }
 | |
| 
 | |
| protected:
 | |
|     explicit IndexedPropertyStorage(IsSimpleStorage is_simple_storage, size_t array_size = 0)
 | |
|         : m_array_size(array_size)
 | |
|         , m_is_simple_storage(is_simple_storage == IsSimpleStorage::Yes)
 | |
|     {
 | |
|     }
 | |
| 
 | |
|     size_t m_array_size { 0 };
 | |
| 
 | |
| private:
 | |
|     bool m_is_simple_storage { false };
 | |
| };
 | |
| 
 | |
| class SimpleIndexedPropertyStorage final : public IndexedPropertyStorage {
 | |
| public:
 | |
|     SimpleIndexedPropertyStorage()
 | |
|         : IndexedPropertyStorage(IsSimpleStorage::Yes)
 | |
|     {
 | |
|     }
 | |
|     explicit SimpleIndexedPropertyStorage(Vector<Value>&& initial_values);
 | |
| 
 | |
|     virtual bool has_index(u32 index) const override;
 | |
|     virtual Optional<ValueAndAttributes> get(u32 index) const override;
 | |
|     virtual void put(u32 index, Value value, PropertyAttributes attributes = default_attributes) override;
 | |
|     virtual void remove(u32 index) override;
 | |
| 
 | |
|     virtual ValueAndAttributes take_first() override;
 | |
|     virtual ValueAndAttributes take_last() override;
 | |
| 
 | |
|     virtual size_t size() const override { return m_packed_elements.size(); }
 | |
|     virtual bool set_array_like_size(size_t new_size) override;
 | |
| 
 | |
|     Vector<Value> const& elements() const { return m_packed_elements; }
 | |
| 
 | |
|     [[nodiscard]] bool inline_has_index(u32 index) const
 | |
|     {
 | |
|         return index < m_array_size && !m_packed_elements.data()[index].is_special_empty_value();
 | |
|     }
 | |
| 
 | |
|     [[nodiscard]] Optional<ValueAndAttributes> inline_get(u32 index) const
 | |
|     {
 | |
|         if (!inline_has_index(index))
 | |
|             return {};
 | |
|         return ValueAndAttributes { m_packed_elements.data()[index], default_attributes };
 | |
|     }
 | |
| 
 | |
|     bool has_empty_elements() const { return m_number_of_empty_elements.value() > 0; }
 | |
| 
 | |
| private:
 | |
|     friend GenericIndexedPropertyStorage;
 | |
| 
 | |
|     void grow_storage_if_needed();
 | |
| 
 | |
|     Checked<size_t> m_number_of_empty_elements { 0 };
 | |
|     Vector<Value> m_packed_elements;
 | |
| };
 | |
| 
 | |
| class GenericIndexedPropertyStorage final : public IndexedPropertyStorage {
 | |
| public:
 | |
|     explicit GenericIndexedPropertyStorage(SimpleIndexedPropertyStorage&&);
 | |
|     explicit GenericIndexedPropertyStorage()
 | |
|         : IndexedPropertyStorage(IsSimpleStorage::No)
 | |
|     {
 | |
|     }
 | |
| 
 | |
|     virtual bool has_index(u32 index) const override;
 | |
|     virtual Optional<ValueAndAttributes> get(u32 index) const override;
 | |
|     virtual void put(u32 index, Value value, PropertyAttributes attributes = default_attributes) override;
 | |
|     virtual void remove(u32 index) override;
 | |
| 
 | |
|     virtual ValueAndAttributes take_first() override;
 | |
|     virtual ValueAndAttributes take_last() override;
 | |
| 
 | |
|     virtual size_t size() const override { return m_sparse_elements.size(); }
 | |
|     virtual bool set_array_like_size(size_t new_size) override;
 | |
| 
 | |
|     HashMap<u32, ValueAndAttributes> const& sparse_elements() const { return m_sparse_elements; }
 | |
| 
 | |
| private:
 | |
|     HashMap<u32, ValueAndAttributes> m_sparse_elements;
 | |
| };
 | |
| 
 | |
| class JS_API IndexedPropertyIterator {
 | |
| public:
 | |
|     IndexedPropertyIterator(IndexedProperties const&, u32 starting_index, bool skip_empty);
 | |
| 
 | |
|     IndexedPropertyIterator& operator++();
 | |
|     IndexedPropertyIterator& operator*();
 | |
|     bool operator!=(IndexedPropertyIterator const&) const;
 | |
| 
 | |
|     u32 index() const { return m_index; }
 | |
|     bool enumerable() const;
 | |
| 
 | |
| private:
 | |
|     void skip_empty_indices();
 | |
| 
 | |
|     IndexedProperties const& m_indexed_properties;
 | |
|     Vector<u32> m_cached_indices;
 | |
|     size_t m_next_cached_index { 0 };
 | |
|     u32 m_index { 0 };
 | |
|     bool m_skip_empty { false };
 | |
|     bool m_all_enumerable { false };
 | |
| };
 | |
| 
 | |
| class JS_API IndexedProperties {
 | |
| public:
 | |
|     IndexedProperties() = default;
 | |
| 
 | |
|     explicit IndexedProperties(Vector<Value> values)
 | |
|     {
 | |
|         if (!values.is_empty())
 | |
|             m_storage = make<SimpleIndexedPropertyStorage>(move(values));
 | |
|     }
 | |
| 
 | |
|     bool has_index(u32 index) const { return m_storage ? m_storage->has_index(index) : false; }
 | |
|     Optional<ValueAndAttributes> get(u32 index) const;
 | |
|     void put(u32 index, Value value, PropertyAttributes attributes = default_attributes);
 | |
|     void remove(u32 index);
 | |
| 
 | |
|     void append(Value value, PropertyAttributes attributes = default_attributes) { put(array_like_size(), value, attributes); }
 | |
| 
 | |
|     IndexedPropertyIterator begin(bool skip_empty = true) const { return IndexedPropertyIterator(*this, 0, skip_empty); }
 | |
|     IndexedPropertyIterator end() const { return IndexedPropertyIterator(*this, array_like_size(), false); }
 | |
| 
 | |
|     bool is_empty() const { return array_like_size() == 0; }
 | |
|     size_t array_like_size() const { return m_storage ? m_storage->array_like_size() : 0; }
 | |
|     bool set_array_like_size(size_t);
 | |
| 
 | |
|     IndexedPropertyStorage* storage() { return m_storage; }
 | |
|     IndexedPropertyStorage const* storage() const { return m_storage; }
 | |
| 
 | |
|     size_t real_size() const;
 | |
| 
 | |
|     Vector<u32> indices() const;
 | |
| 
 | |
|     template<typename Callback>
 | |
|     void for_each_value(Callback callback)
 | |
|     {
 | |
|         if (!m_storage)
 | |
|             return;
 | |
|         if (m_storage->is_simple_storage()) {
 | |
|             for (auto& value : static_cast<SimpleIndexedPropertyStorage&>(*m_storage).elements())
 | |
|                 callback(value);
 | |
|         } else {
 | |
|             for (auto& element : static_cast<GenericIndexedPropertyStorage const&>(*m_storage).sparse_elements())
 | |
|                 callback(element.value.value);
 | |
|         }
 | |
|     }
 | |
| 
 | |
| private:
 | |
|     void switch_to_generic_storage();
 | |
|     void ensure_storage();
 | |
| 
 | |
|     OwnPtr<IndexedPropertyStorage> m_storage;
 | |
| };
 | |
| 
 | |
| }
 |