mirror of
https://github.com/RGBCube/serenity
synced 2025-10-24 10:32:31 +00:00

This patch rethinks the way indexed property storage works: Instead of having a cut-off point at 200 elements where we always move to generic property storage, we now allow arrays to stay in simple mode as long as we don't create a gap/hole larger than 200 elements. We also simplify generic storage to only have a hash map (for now) instead of juggling both a vector and a hash map. This is mostly fine since the vast majority of arrays get to stay simple now. This is a huge speedup on anything that uses basic JS arrays with more than 200 elements in them. :^)
185 lines
7.2 KiB
C++
185 lines
7.2 KiB
C++
/*
|
|
* Copyright (c) 2020, Matthew Olsson <matthewcolsson@gmail.com>
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions are met:
|
|
*
|
|
* 1. Redistributions of source code must retain the above copyright notice, this
|
|
* list of conditions and the following disclaimer.
|
|
*
|
|
* 2. Redistributions in binary form must reproduce the above copyright notice,
|
|
* this list of conditions and the following disclaimer in the documentation
|
|
* and/or other materials provided with the distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
|
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
|
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
|
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include <AK/NonnullOwnPtr.h>
|
|
#include <LibJS/Runtime/Shape.h>
|
|
#include <LibJS/Runtime/Value.h>
|
|
|
|
namespace JS {
|
|
|
|
struct ValueAndAttributes {
|
|
Value value;
|
|
PropertyAttributes attributes { default_attributes };
|
|
};
|
|
|
|
class IndexedProperties;
|
|
class IndexedPropertyIterator;
|
|
class GenericIndexedPropertyStorage;
|
|
|
|
class IndexedPropertyStorage {
|
|
public:
|
|
virtual ~IndexedPropertyStorage() {};
|
|
|
|
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 void insert(u32 index, Value value, PropertyAttributes attributes = default_attributes) = 0;
|
|
virtual ValueAndAttributes take_first() = 0;
|
|
virtual ValueAndAttributes take_last() = 0;
|
|
|
|
virtual size_t size() const = 0;
|
|
virtual size_t array_like_size() const = 0;
|
|
virtual void set_array_like_size(size_t new_size) = 0;
|
|
|
|
virtual bool is_simple_storage() const { return false; }
|
|
};
|
|
|
|
class SimpleIndexedPropertyStorage final : public IndexedPropertyStorage {
|
|
public:
|
|
SimpleIndexedPropertyStorage() = default;
|
|
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 void insert(u32 index, Value value, PropertyAttributes attributes = default_attributes) override;
|
|
virtual ValueAndAttributes take_first() override;
|
|
virtual ValueAndAttributes take_last() override;
|
|
|
|
virtual size_t size() const override { return m_packed_elements.size(); }
|
|
virtual size_t array_like_size() const override { return m_array_size; }
|
|
virtual void set_array_like_size(size_t new_size) override;
|
|
|
|
virtual bool is_simple_storage() const override { return true; }
|
|
const Vector<Value>& elements() const { return m_packed_elements; }
|
|
|
|
private:
|
|
friend GenericIndexedPropertyStorage;
|
|
|
|
void grow_storage_if_needed();
|
|
|
|
size_t m_array_size { 0 };
|
|
Vector<Value> m_packed_elements;
|
|
};
|
|
|
|
class GenericIndexedPropertyStorage final : public IndexedPropertyStorage {
|
|
public:
|
|
explicit GenericIndexedPropertyStorage(SimpleIndexedPropertyStorage&&);
|
|
|
|
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 void insert(u32 index, Value value, PropertyAttributes attributes = default_attributes) override;
|
|
virtual ValueAndAttributes take_first() override;
|
|
virtual ValueAndAttributes take_last() override;
|
|
|
|
virtual size_t size() const override { return m_sparse_elements.size(); }
|
|
virtual size_t array_like_size() const override { return m_array_size; }
|
|
virtual void set_array_like_size(size_t new_size) override;
|
|
|
|
const HashMap<u32, ValueAndAttributes>& sparse_elements() const { return m_sparse_elements; }
|
|
|
|
private:
|
|
size_t m_array_size { 0 };
|
|
HashMap<u32, ValueAndAttributes> m_sparse_elements;
|
|
};
|
|
|
|
class IndexedPropertyIterator {
|
|
public:
|
|
IndexedPropertyIterator(const IndexedProperties&, u32 starting_index, bool skip_empty);
|
|
|
|
IndexedPropertyIterator& operator++();
|
|
IndexedPropertyIterator& operator*();
|
|
bool operator!=(const IndexedPropertyIterator&) const;
|
|
|
|
u32 index() const { return m_index; };
|
|
ValueAndAttributes value_and_attributes(Object* this_object, bool evaluate_accessors = true);
|
|
|
|
private:
|
|
void skip_empty_indices();
|
|
|
|
const IndexedProperties& m_indexed_properties;
|
|
u32 m_index;
|
|
bool m_skip_empty;
|
|
};
|
|
|
|
class IndexedProperties {
|
|
public:
|
|
IndexedProperties() = default;
|
|
|
|
explicit IndexedProperties(Vector<Value> values)
|
|
: m_storage(make<SimpleIndexedPropertyStorage>(move(values)))
|
|
{
|
|
}
|
|
|
|
bool has_index(u32 index) const { return m_storage->has_index(index); }
|
|
Optional<ValueAndAttributes> get(Object* this_object, u32 index, bool evaluate_accessors = true) const;
|
|
void put(Object* this_object, u32 index, Value value, PropertyAttributes attributes = default_attributes, bool evaluate_accessors = true);
|
|
bool remove(u32 index);
|
|
|
|
void insert(u32 index, Value value, PropertyAttributes attributes = default_attributes);
|
|
ValueAndAttributes take_first(Object* this_object);
|
|
ValueAndAttributes take_last(Object* this_object);
|
|
|
|
void append(Value value, PropertyAttributes attributes = default_attributes) { put(nullptr, array_like_size(), value, attributes, false); }
|
|
void append_all(Object* this_object, const IndexedProperties& properties, bool evaluate_accessors = true);
|
|
|
|
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->array_like_size(); }
|
|
void set_array_like_size(size_t);
|
|
|
|
Vector<u32> indices() const;
|
|
|
|
template<typename Callback>
|
|
void for_each_value(Callback callback)
|
|
{
|
|
if (m_storage->is_simple_storage()) {
|
|
for (auto& value : static_cast<SimpleIndexedPropertyStorage&>(*m_storage).elements())
|
|
callback(value);
|
|
} else {
|
|
for (auto& element : static_cast<const GenericIndexedPropertyStorage&>(*m_storage).sparse_elements())
|
|
callback(element.value.value);
|
|
}
|
|
}
|
|
|
|
private:
|
|
void switch_to_generic_storage();
|
|
|
|
NonnullOwnPtr<IndexedPropertyStorage> m_storage { make<SimpleIndexedPropertyStorage>() };
|
|
};
|
|
|
|
}
|