blob: 58924206ea93139fd85786ce279304a958a48134 [file] [log] [blame]
// Copyright 2017 the V8 project authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef V8_OBJECTS_DICTIONARY_H_
#define V8_OBJECTS_DICTIONARY_H_
#include "src/base/export-template.h"
#include "src/base/optional.h"
#include "src/common/globals.h"
#include "src/objects/hash-table.h"
#include "src/objects/property-array.h"
#include "src/objects/smi.h"
#include "src/roots/roots.h"
// Has to be the last include (doesn't have include guards):
#include "src/objects/object-macros.h"
namespace v8 {
namespace internal {
#ifdef V8_ENABLE_SWISS_NAME_DICTIONARY
class SwissNameDictionary;
using PropertyDictionary = SwissNameDictionary;
#else
using PropertyDictionary = NameDictionary;
#endif
template <typename Derived, typename Shape>
class EXPORT_TEMPLATE_DECLARE(V8_EXPORT_PRIVATE) Dictionary
: public HashTable<Derived, Shape> {
using DerivedHashTable = HashTable<Derived, Shape>;
public:
using TodoShape = Shape;
using Key = typename TodoShape::Key;
inline Tagged<Object> ValueAt(InternalIndex entry);
inline Tagged<Object> ValueAt(PtrComprCageBase cage_base,
InternalIndex entry);
inline Tagged<Object> ValueAt(InternalIndex entry, SeqCstAccessTag);
inline Tagged<Object> ValueAt(PtrComprCageBase cage_base, InternalIndex entry,
SeqCstAccessTag);
// Returns {} if we would be reading out of the bounds of the object.
inline base::Optional<Tagged<Object>> TryValueAt(InternalIndex entry);
// Set the value for entry.
inline void ValueAtPut(InternalIndex entry, Tagged<Object> value);
inline void ValueAtPut(InternalIndex entry, Tagged<Object> value,
SeqCstAccessTag);
// Swap the value for the entry.
inline Tagged<Object> ValueAtSwap(InternalIndex entry, Tagged<Object> value,
SeqCstAccessTag);
// Compare and swap the value for the entry.
inline Tagged<Object> ValueAtCompareAndSwap(InternalIndex entry,
Tagged<Object> expected,
Tagged<Object> value,
SeqCstAccessTag);
// Returns the property details for the property at entry.
inline PropertyDetails DetailsAt(InternalIndex entry);
// Set the details for entry.
inline void DetailsAtPut(InternalIndex entry, PropertyDetails value);
static const bool kIsOrderedDictionaryType = false;
// Delete a property from the dictionary.
V8_WARN_UNUSED_RESULT static Handle<Derived> DeleteEntry(
Isolate* isolate, Handle<Derived> dictionary, InternalIndex entry);
// Attempt to shrink the dictionary after deletion of key.
V8_WARN_UNUSED_RESULT static inline Handle<Derived> Shrink(
Isolate* isolate, Handle<Derived> dictionary) {
return DerivedHashTable::Shrink(isolate, dictionary);
}
int NumberOfEnumerableProperties();
// Returns the key (slow).
Tagged<Object> SlowReverseLookup(Tagged<Object> value);
inline void ClearEntry(InternalIndex entry);
// Sets the entry to (key, value) pair.
inline void SetEntry(InternalIndex entry, Tagged<Object> key,
Tagged<Object> value, PropertyDetails details);
// Garbage collection support.
inline ObjectSlot RawFieldOfValueAt(InternalIndex entry);
template <typename IsolateT, AllocationType key_allocation =
std::is_same<IsolateT, Isolate>::value
? AllocationType::kYoung
: AllocationType::kOld>
V8_WARN_UNUSED_RESULT static Handle<Derived> Add(
IsolateT* isolate, Handle<Derived> dictionary, Key key,
Handle<Object> value, PropertyDetails details,
InternalIndex* entry_out = nullptr);
// This method is only safe to use when it is guaranteed that the dictionary
// doesn't need to grow.
// The number of elements stored is not updated. Use
// |SetInitialNumberOfElements| to update the number in one go.
template <typename IsolateT, AllocationType key_allocation =
std::is_same<IsolateT, Isolate>::value
? AllocationType::kYoung
: AllocationType::kOld>
static void UncheckedAdd(IsolateT* isolate, Handle<Derived> dictionary,
Key key, Handle<Object> value,
PropertyDetails details);
static Handle<Derived> ShallowCopy(
Isolate* isolate, Handle<Derived> dictionary,
AllocationType allocation = AllocationType::kYoung);
protected:
// Generic at put operation.
V8_WARN_UNUSED_RESULT static Handle<Derived> AtPut(Isolate* isolate,
Handle<Derived> dictionary,
Key key,
Handle<Object> value,
PropertyDetails details);
static void UncheckedAtPut(Isolate* isolate, Handle<Derived> dictionary,
Key key, Handle<Object> value,
PropertyDetails details);
OBJECT_CONSTRUCTORS(Dictionary, HashTable<Derived, TodoShape>);
};
#define EXTERN_DECLARE_DICTIONARY(DERIVED, SHAPE) \
EXTERN_DECLARE_HASH_TABLE(DERIVED, SHAPE) \
extern template class EXPORT_TEMPLATE_DECLARE(V8_EXPORT_PRIVATE) \
Dictionary<DERIVED, SHAPE>;
template <typename Key>
class BaseDictionaryShape : public BaseShape<Key> {
public:
static const bool kHasDetails = true;
template <typename Dictionary>
static inline PropertyDetails DetailsAt(Tagged<Dictionary> dict,
InternalIndex entry);
template <typename Dictionary>
static inline void DetailsAtPut(Tagged<Dictionary> dict, InternalIndex entry,
PropertyDetails value);
};
class BaseNameDictionaryShape : public BaseDictionaryShape<Handle<Name>> {
public:
static inline bool IsMatch(Handle<Name> key, Tagged<Object> other);
static inline uint32_t Hash(ReadOnlyRoots roots, Handle<Name> key);
static inline uint32_t HashForObject(ReadOnlyRoots roots,
Tagged<Object> object);
template <AllocationType allocation = AllocationType::kYoung>
static inline Handle<Object> AsHandle(Isolate* isolate, Handle<Name> key);
template <AllocationType allocation = AllocationType::kOld>
static inline Handle<Object> AsHandle(LocalIsolate* isolate,
Handle<Name> key);
static const int kEntryValueIndex = 1;
};
class NameDictionaryShape : public BaseNameDictionaryShape {
public:
static const int kPrefixSize = 3;
static const int kEntrySize = 3;
static const bool kMatchNeedsHoleCheck = false;
};
template <typename Derived, typename Shape>
class EXPORT_TEMPLATE_DECLARE(V8_EXPORT_PRIVATE) BaseNameDictionary
: public Dictionary<Derived, Shape> {
using Key = typename Shape::Key;
public:
static const int kNextEnumerationIndexIndex =
HashTableBase::kPrefixStartIndex;
static const int kObjectHashIndex = kNextEnumerationIndexIndex + 1;
static const int kEntryValueIndex = 1;
inline void SetHash(int hash);
inline int Hash() const;
// Creates a new dictionary.
template <typename IsolateT>
V8_WARN_UNUSED_RESULT static Handle<Derived> New(
IsolateT* isolate, int at_least_space_for,
AllocationType allocation = AllocationType::kYoung,
MinimumCapacity capacity_option = USE_DEFAULT_MINIMUM_CAPACITY);
// Allocate the next enumeration index. Possibly updates all enumeration
// indices in the table.
static int NextEnumerationIndex(Isolate* isolate, Handle<Derived> dictionary);
// Accessors for next enumeration index.
inline int next_enumeration_index();
inline void set_next_enumeration_index(int index);
// Return the key indices sorted by its enumeration index.
static Handle<FixedArray> IterationIndices(Isolate* isolate,
Handle<Derived> dictionary);
template <typename IsolateT>
V8_WARN_UNUSED_RESULT static Handle<Derived> AddNoUpdateNextEnumerationIndex(
IsolateT* isolate, Handle<Derived> dictionary, Key key,
Handle<Object> value, PropertyDetails details,
InternalIndex* entry_out = nullptr);
V8_WARN_UNUSED_RESULT static Handle<Derived> Add(
Isolate* isolate, Handle<Derived> dictionary, Key key,
Handle<Object> value, PropertyDetails details,
InternalIndex* entry_out = nullptr);
// Exposed for NameDictionaryLookupForwardedString slow path for forwarded
// strings.
using Dictionary<Derived, Shape>::FindInsertionEntry;
OBJECT_CONSTRUCTORS(BaseNameDictionary, Dictionary<Derived, Shape>);
};
#define EXTERN_DECLARE_BASE_NAME_DICTIONARY(DERIVED, SHAPE) \
EXTERN_DECLARE_DICTIONARY(DERIVED, SHAPE) \
extern template class EXPORT_TEMPLATE_DECLARE(V8_EXPORT_PRIVATE) \
BaseNameDictionary<DERIVED, SHAPE>;
EXTERN_DECLARE_BASE_NAME_DICTIONARY(NameDictionary, NameDictionaryShape)
class V8_EXPORT_PRIVATE NameDictionary
: public BaseNameDictionary<NameDictionary, NameDictionaryShape> {
public:
static inline Handle<Map> GetMap(ReadOnlyRoots roots);
DECL_CAST(NameDictionary)
DECL_PRINTER(NameDictionary)
static const int kFlagsIndex = kObjectHashIndex + 1;
static const int kEntryValueIndex = 1;
static const int kEntryDetailsIndex = 2;
static const int kInitialCapacity = 2;
inline Tagged<Name> NameAt(InternalIndex entry);
inline Tagged<Name> NameAt(PtrComprCageBase cage_base, InternalIndex entry);
inline void set_hash(int hash);
inline int hash() const;
// Note: Flags are stored as smi, so only 31 bits are usable.
using MayHaveInterestingPropertiesBit = base::BitField<bool, 0, 1, uint32_t>;
DECL_BOOLEAN_ACCESSORS(may_have_interesting_properties)
static constexpr int kFlagsDefault = 0;
inline uint32_t flags() const;
inline void set_flags(uint32_t flags);
// Creates a new NameDictionary.
template <typename IsolateT>
V8_WARN_UNUSED_RESULT static Handle<NameDictionary> New(
IsolateT* isolate, int at_least_space_for,
AllocationType allocation = AllocationType::kYoung,
MinimumCapacity capacity_option = USE_DEFAULT_MINIMUM_CAPACITY);
OBJECT_CONSTRUCTORS(NameDictionary,
BaseNameDictionary<NameDictionary, NameDictionaryShape>);
};
class V8_EXPORT_PRIVATE GlobalDictionaryShape : public BaseNameDictionaryShape {
public:
static inline bool IsMatch(Handle<Name> key, Tagged<Object> other);
static inline uint32_t HashForObject(ReadOnlyRoots roots,
Tagged<Object> object);
static const bool kMatchNeedsHoleCheck = true;
static const int kPrefixSize = 2;
static const int kEntrySize = 1;
template <typename Dictionary>
static inline PropertyDetails DetailsAt(Tagged<Dictionary> dict,
InternalIndex entry);
template <typename Dictionary>
static inline void DetailsAtPut(Tagged<Dictionary> dict, InternalIndex entry,
PropertyDetails value);
static inline Tagged<Object> Unwrap(Tagged<Object> key);
};
EXTERN_DECLARE_BASE_NAME_DICTIONARY(GlobalDictionary, GlobalDictionaryShape)
class V8_EXPORT_PRIVATE GlobalDictionary
: public BaseNameDictionary<GlobalDictionary, GlobalDictionaryShape> {
public:
static inline Handle<Map> GetMap(ReadOnlyRoots roots);
DECL_CAST(GlobalDictionary)
DECL_PRINTER(GlobalDictionary)
inline Tagged<Object> ValueAt(InternalIndex entry);
inline Tagged<Object> ValueAt(PtrComprCageBase cage_base,
InternalIndex entry);
inline Tagged<PropertyCell> CellAt(InternalIndex entry);
inline Tagged<PropertyCell> CellAt(PtrComprCageBase cage_base,
InternalIndex entry);
inline void SetEntry(InternalIndex entry, Tagged<Object> key,
Tagged<Object> value, PropertyDetails details);
inline void ClearEntry(InternalIndex entry);
inline Tagged<Name> NameAt(InternalIndex entry);
inline Tagged<Name> NameAt(PtrComprCageBase cage_base, InternalIndex entry);
inline void ValueAtPut(InternalIndex entry, Tagged<Object> value);
base::Optional<Tagged<PropertyCell>>
TryFindPropertyCellForConcurrentLookupIterator(Isolate* isolate,
Handle<Name> name,
RelaxedLoadTag tag);
OBJECT_CONSTRUCTORS(
GlobalDictionary,
BaseNameDictionary<GlobalDictionary, GlobalDictionaryShape>);
};
class NumberDictionaryBaseShape : public BaseDictionaryShape<uint32_t> {
public:
static inline bool IsMatch(uint32_t key, Tagged<Object> other);
template <AllocationType allocation = AllocationType::kYoung>
static inline Handle<Object> AsHandle(Isolate* isolate, uint32_t key);
template <AllocationType allocation = AllocationType::kOld>
static inline Handle<Object> AsHandle(LocalIsolate* isolate, uint32_t key);
static inline uint32_t Hash(ReadOnlyRoots roots, uint32_t key);
static inline uint32_t HashForObject(ReadOnlyRoots roots,
Tagged<Object> object);
static const bool kMatchNeedsHoleCheck = true;
};
class NumberDictionaryShape : public NumberDictionaryBaseShape {
public:
static const int kPrefixSize = 1;
static const int kEntrySize = 3;
};
class SimpleNumberDictionaryShape : public NumberDictionaryBaseShape {
public:
static const bool kHasDetails = false;
static const int kPrefixSize = 0;
static const int kEntrySize = 2;
template <typename Dictionary>
static inline PropertyDetails DetailsAt(Tagged<Dictionary> dict,
InternalIndex entry) {
UNREACHABLE();
}
template <typename Dictionary>
static inline void DetailsAtPut(Tagged<Dictionary> dict, InternalIndex entry,
PropertyDetails value) {
UNREACHABLE();
}
};
EXTERN_DECLARE_DICTIONARY(SimpleNumberDictionary, SimpleNumberDictionaryShape)
// SimpleNumberDictionary is used to map number to an entry.
class SimpleNumberDictionary
: public Dictionary<SimpleNumberDictionary, SimpleNumberDictionaryShape> {
public:
static inline Handle<Map> GetMap(ReadOnlyRoots roots);
DECL_CAST(SimpleNumberDictionary)
// Type specific at put (default NONE attributes is used when adding).
V8_EXPORT_PRIVATE V8_WARN_UNUSED_RESULT static Handle<SimpleNumberDictionary>
Set(Isolate* isolate, Handle<SimpleNumberDictionary> dictionary, uint32_t key,
Handle<Object> value);
static const int kEntryValueIndex = 1;
OBJECT_CONSTRUCTORS(
SimpleNumberDictionary,
Dictionary<SimpleNumberDictionary, SimpleNumberDictionaryShape>);
};
EXTERN_DECLARE_DICTIONARY(NumberDictionary, NumberDictionaryShape)
// NumberDictionary is used as elements backing store and provides a bitfield
// and stores property details for every entry.
class NumberDictionary
: public Dictionary<NumberDictionary, NumberDictionaryShape> {
public:
static inline Handle<Map> GetMap(ReadOnlyRoots roots);
DECL_CAST(NumberDictionary)
DECL_PRINTER(NumberDictionary)
// Type specific at put (default NONE attributes is used when adding).
V8_WARN_UNUSED_RESULT static Handle<NumberDictionary> Set(
Isolate* isolate, Handle<NumberDictionary> dictionary, uint32_t key,
Handle<Object> value,
Handle<JSObject> dictionary_holder = Handle<JSObject>::null(),
PropertyDetails details = PropertyDetails::Empty());
// This method is only safe to use when it is guaranteed that the dictionary
// doesn't need to grow.
// The number of elements stored and the maximum index is not updated. Use
// |SetInitialNumberOfElements| and |UpdateMaxNumberKey| to update the number
// in one go.
static void UncheckedSet(Isolate* isolate,
Handle<NumberDictionary> dictionary, uint32_t key,
Handle<Object> value);
static const int kMaxNumberKeyIndex = kPrefixStartIndex;
void UpdateMaxNumberKey(uint32_t key, Handle<JSObject> dictionary_holder);
// Sorting support
void CopyValuesTo(Tagged<FixedArray> elements);
// If slow elements are required we will never go back to fast-case
// for the elements kept in this dictionary. We require slow
// elements if an element has been added at an index larger than
// kRequiresSlowElementsLimit or set_requires_slow_elements() has been called
// when defining a getter or setter with a number key.
inline bool requires_slow_elements();
inline void set_requires_slow_elements();
// Get the value of the max number key that has been added to this
// dictionary. max_number_key can only be called if
// requires_slow_elements returns false.
inline uint32_t max_number_key();
static const int kEntryValueIndex = 1;
static const int kEntryDetailsIndex = 2;
// Bit masks.
static const int kRequiresSlowElementsMask = 1;
static const int kRequiresSlowElementsTagSize = 1;
static const uint32_t kRequiresSlowElementsLimit = (1 << 29) - 1;
// JSObjects prefer dictionary elements if the dictionary saves this much
// memory compared to a fast elements backing store.
static const uint32_t kPreferFastElementsSizeFactor = 3;
OBJECT_CONSTRUCTORS(NumberDictionary,
Dictionary<NumberDictionary, NumberDictionaryShape>);
};
// The comparator is passed two indices |a| and |b|, and it returns < 0 when the
// property at index |a| comes before the property at index |b| in the
// enumeration order.
template <typename Dictionary>
struct EnumIndexComparator {
explicit EnumIndexComparator(Tagged<Dictionary> dict) : dict(dict) {}
bool operator()(Tagged_t a, Tagged_t b) {
PropertyDetails da(dict->DetailsAt(
InternalIndex(Tagged<Smi>(static_cast<Address>(a)).value())));
PropertyDetails db(dict->DetailsAt(
InternalIndex(Tagged<Smi>(static_cast<Address>(b)).value())));
return da.dictionary_index() < db.dictionary_index();
}
Tagged<Dictionary> dict;
};
} // namespace internal
} // namespace v8
#include "src/objects/object-macros-undef.h"
#endif // V8_OBJECTS_DICTIONARY_H_