summaryrefslogtreecommitdiffstats
path: root/js/src/vm/Caches.h
diff options
context:
space:
mode:
authorDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-07 17:32:43 +0000
committerDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-07 17:32:43 +0000
commit6bf0a5cb5034a7e684dcc3500e841785237ce2dd (patch)
treea68f146d7fa01f0134297619fbe7e33db084e0aa /js/src/vm/Caches.h
parentInitial commit. (diff)
downloadthunderbird-6bf0a5cb5034a7e684dcc3500e841785237ce2dd.tar.xz
thunderbird-6bf0a5cb5034a7e684dcc3500e841785237ce2dd.zip
Adding upstream version 1:115.7.0.upstream/1%115.7.0upstream
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'js/src/vm/Caches.h')
-rw-r--r--js/src/vm/Caches.h568
1 files changed, 568 insertions, 0 deletions
diff --git a/js/src/vm/Caches.h b/js/src/vm/Caches.h
new file mode 100644
index 0000000000..c1d9caefdb
--- /dev/null
+++ b/js/src/vm/Caches.h
@@ -0,0 +1,568 @@
+/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*-
+ * vim: set ts=8 sts=2 et sw=2 tw=80:
+ * This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+#ifndef vm_Caches_h
+#define vm_Caches_h
+
+#include "mozilla/Array.h"
+#include "mozilla/MathAlgorithms.h"
+#include "mozilla/Maybe.h"
+#include "mozilla/MruCache.h"
+#include "mozilla/TemplateLib.h"
+#include "mozilla/UniquePtr.h"
+
+#include "frontend/ScopeBindingCache.h"
+#include "gc/Tracer.h"
+#include "js/RootingAPI.h"
+#include "js/TypeDecls.h"
+#include "vm/JSScript.h"
+#include "vm/Shape.h"
+#include "vm/StencilCache.h" // js::StencilCache
+#include "vm/StringType.h"
+
+namespace js {
+
+class SrcNote;
+
+/*
+ * GetSrcNote cache to avoid O(n^2) growth in finding a source note for a
+ * given pc in a script. We use the script->code pointer to tag the cache,
+ * instead of the script address itself, so that source notes are always found
+ * by offset from the bytecode with which they were generated.
+ */
+struct GSNCache {
+ typedef HashMap<jsbytecode*, const SrcNote*, PointerHasher<jsbytecode*>,
+ SystemAllocPolicy>
+ Map;
+
+ jsbytecode* code;
+ Map map;
+
+ GSNCache() : code(nullptr) {}
+
+ void purge();
+};
+
+struct EvalCacheEntry {
+ JSLinearString* str;
+ JSScript* script;
+ JSScript* callerScript;
+ jsbytecode* pc;
+
+ // We sweep this cache after a nursery collection to update entries with
+ // string keys that have been tenured.
+ //
+ // The entire cache is purged on a major GC, so we don't need to sweep it
+ // then.
+ bool traceWeak(JSTracer* trc) {
+ MOZ_ASSERT(trc->kind() == JS::TracerKind::MinorSweeping);
+ return TraceManuallyBarrieredWeakEdge(trc, &str, "EvalCacheEntry::str");
+ }
+};
+
+struct EvalCacheLookup {
+ explicit EvalCacheLookup(JSContext* cx) : str(cx), callerScript(cx) {}
+ Rooted<JSLinearString*> str;
+ RootedScript callerScript;
+ MOZ_INIT_OUTSIDE_CTOR jsbytecode* pc;
+};
+
+struct EvalCacheHashPolicy {
+ using Lookup = EvalCacheLookup;
+
+ static HashNumber hash(const Lookup& l);
+ static bool match(const EvalCacheEntry& entry, const EvalCacheLookup& l);
+};
+
+using EvalCache =
+ GCHashSet<EvalCacheEntry, EvalCacheHashPolicy, SystemAllocPolicy>;
+
+class MegamorphicCacheEntry {
+ // Receiver object's shape.
+ Shape* shape_ = nullptr;
+
+ // The atom or symbol property being accessed.
+ PropertyKey key_;
+
+ // Slot offset and isFixedSlot flag of the data property.
+ TaggedSlotOffset slotOffset_;
+
+ // This entry is valid iff the generation matches the cache's generation.
+ uint16_t generation_ = 0;
+
+ // Number of hops on the proto chain to get to the holder object. If this is
+ // zero, the property exists on the receiver object. It can also be one of
+ // the sentinel values indicating a missing property lookup.
+ uint8_t numHops_ = 0;
+
+ friend class MegamorphicCache;
+
+ public:
+ static constexpr uint8_t MaxHopsForDataProperty = UINT8_MAX - 2;
+ static constexpr uint8_t NumHopsForMissingProperty = UINT8_MAX - 1;
+ static constexpr uint8_t NumHopsForMissingOwnProperty = UINT8_MAX;
+
+ void init(Shape* shape, PropertyKey key, uint16_t generation, uint8_t numHops,
+ TaggedSlotOffset slotOffset) {
+ shape_ = shape;
+ key_ = key;
+ slotOffset_ = slotOffset;
+ generation_ = generation;
+ numHops_ = numHops;
+ MOZ_ASSERT(numHops_ == numHops, "numHops must fit in numHops_");
+ }
+ bool isMissingProperty() const {
+ return numHops_ == NumHopsForMissingProperty;
+ }
+ bool isMissingOwnProperty() const {
+ return numHops_ == NumHopsForMissingOwnProperty;
+ }
+ bool isDataProperty() const { return numHops_ <= MaxHopsForDataProperty; }
+ uint16_t numHops() const {
+ MOZ_ASSERT(isDataProperty());
+ return numHops_;
+ }
+ TaggedSlotOffset slotOffset() const {
+ MOZ_ASSERT(isDataProperty());
+ return slotOffset_;
+ }
+
+ static constexpr size_t offsetOfShape() {
+ return offsetof(MegamorphicCacheEntry, shape_);
+ }
+
+ static constexpr size_t offsetOfKey() {
+ return offsetof(MegamorphicCacheEntry, key_);
+ }
+
+ static constexpr size_t offsetOfGeneration() {
+ return offsetof(MegamorphicCacheEntry, generation_);
+ }
+
+ static constexpr size_t offsetOfSlotOffset() {
+ return offsetof(MegamorphicCacheEntry, slotOffset_);
+ }
+
+ static constexpr size_t offsetOfNumHops() {
+ return offsetof(MegamorphicCacheEntry, numHops_);
+ }
+};
+
+// [SMDOC] Megamorphic Property Lookup Cache (MegamorphicCache)
+//
+// MegamorphicCache is a data structure used to speed up megamorphic property
+// lookups from JIT code. The same cache is currently used for both GetProp and
+// HasProp (in, hasOwnProperty) operations.
+//
+// This is implemented as a fixed-size array of entries. Lookups are performed
+// based on the receiver object's Shape + PropertyKey. If found in the cache,
+// the result of a lookup represents either:
+//
+// * A data property on the receiver or on its proto chain (stored as number of
+// 'hops' up the proto chain + the slot of the data property).
+//
+// * A missing property on the receiver or its proto chain.
+//
+// * A missing property on the receiver, but it might exist on the proto chain.
+// This lets us optimize hasOwnProperty better.
+//
+// Collisions are handled by simply overwriting the previous entry stored in the
+// slot. This is sufficient to achieve a high hit rate on typical web workloads
+// while ensuring cache lookups are always fast and simple.
+//
+// Lookups always check the receiver object's shape (ensuring the properties and
+// prototype are unchanged). Because the cache also caches lookups on the proto
+// chain, Watchtower is used to invalidate the cache when prototype objects are
+// mutated. This is done by incrementing the cache's generation counter to
+// invalidate all entries.
+//
+// The cache is also invalidated on each major GC.
+class MegamorphicCache {
+ public:
+ using Entry = MegamorphicCacheEntry;
+
+ static constexpr size_t NumEntries = 1024;
+ static constexpr uint8_t ShapeHashShift1 =
+ mozilla::tl::FloorLog2<alignof(Shape)>::value;
+ static constexpr uint8_t ShapeHashShift2 =
+ ShapeHashShift1 + mozilla::tl::FloorLog2<NumEntries>::value;
+
+ static_assert(mozilla::IsPowerOfTwo(alignof(Shape)) &&
+ mozilla::IsPowerOfTwo(NumEntries),
+ "FloorLog2 is exact because alignof(Shape) and NumEntries are "
+ "both powers of two");
+
+ private:
+ mozilla::Array<Entry, NumEntries> entries_;
+
+ // Generation counter used to invalidate all entries.
+ uint16_t generation_ = 0;
+
+ // NOTE: this logic is mirrored in MacroAssembler::emitMegamorphicCacheLookup
+ Entry& getEntry(Shape* shape, PropertyKey key) {
+ static_assert(mozilla::IsPowerOfTwo(NumEntries),
+ "NumEntries must be a power-of-two for fast modulo");
+ uintptr_t hash = uintptr_t(shape) >> ShapeHashShift1;
+ hash ^= uintptr_t(shape) >> ShapeHashShift2;
+ hash += HashAtomOrSymbolPropertyKey(key);
+ return entries_[hash % NumEntries];
+ }
+
+ public:
+ void bumpGeneration() {
+ generation_++;
+ if (generation_ == 0) {
+ // Generation overflowed. Invalidate the whole cache.
+ for (size_t i = 0; i < NumEntries; i++) {
+ entries_[i].shape_ = nullptr;
+ }
+ }
+ }
+ bool lookup(Shape* shape, PropertyKey key, Entry** entryp) {
+ Entry& entry = getEntry(shape, key);
+ *entryp = &entry;
+ return (entry.shape_ == shape && entry.key_ == key &&
+ entry.generation_ == generation_);
+ }
+ void initEntryForMissingProperty(Entry* entry, Shape* shape,
+ PropertyKey key) {
+ entry->init(shape, key, generation_, Entry::NumHopsForMissingProperty,
+ TaggedSlotOffset());
+ }
+ void initEntryForMissingOwnProperty(Entry* entry, Shape* shape,
+ PropertyKey key) {
+ entry->init(shape, key, generation_, Entry::NumHopsForMissingOwnProperty,
+ TaggedSlotOffset());
+ }
+ void initEntryForDataProperty(Entry* entry, Shape* shape, PropertyKey key,
+ size_t numHops, TaggedSlotOffset slotOffset) {
+ if (numHops > Entry::MaxHopsForDataProperty) {
+ return;
+ }
+ entry->init(shape, key, generation_, numHops, slotOffset);
+ }
+
+ static constexpr size_t offsetOfEntries() {
+ return offsetof(MegamorphicCache, entries_);
+ }
+
+ static constexpr size_t offsetOfGeneration() {
+ return offsetof(MegamorphicCache, generation_);
+ }
+};
+
+class MegamorphicSetPropCacheEntry {
+ Shape* beforeShape_ = nullptr;
+ Shape* afterShape_ = nullptr;
+
+ // The atom or symbol property being accessed.
+ PropertyKey key_;
+
+ // Slot offset and isFixedSlot flag of the data property.
+ TaggedSlotOffset slotOffset_;
+
+ // If slots need to be grown, this is the new capacity we need.
+ uint16_t newCapacity_ = 0;
+
+ // This entry is valid iff the generation matches the cache's generation.
+ uint16_t generation_ = 0;
+
+ friend class MegamorphicSetPropCache;
+
+ public:
+ void init(Shape* beforeShape, Shape* afterShape, PropertyKey key,
+ uint16_t generation, TaggedSlotOffset slotOffset,
+ uint16_t newCapacity) {
+ beforeShape_ = beforeShape;
+ afterShape_ = afterShape;
+ key_ = key;
+ slotOffset_ = slotOffset;
+ newCapacity_ = newCapacity;
+ generation_ = generation;
+ }
+ TaggedSlotOffset slotOffset() const { return slotOffset_; }
+ Shape* afterShape() const { return afterShape_; }
+
+ static constexpr size_t offsetOfShape() {
+ return offsetof(MegamorphicSetPropCacheEntry, beforeShape_);
+ }
+ static constexpr size_t offsetOfAfterShape() {
+ return offsetof(MegamorphicSetPropCacheEntry, afterShape_);
+ }
+
+ static constexpr size_t offsetOfKey() {
+ return offsetof(MegamorphicSetPropCacheEntry, key_);
+ }
+
+ static constexpr size_t offsetOfNewCapacity() {
+ return offsetof(MegamorphicSetPropCacheEntry, newCapacity_);
+ }
+
+ static constexpr size_t offsetOfGeneration() {
+ return offsetof(MegamorphicSetPropCacheEntry, generation_);
+ }
+
+ static constexpr size_t offsetOfSlotOffset() {
+ return offsetof(MegamorphicSetPropCacheEntry, slotOffset_);
+ }
+};
+
+class MegamorphicSetPropCache {
+ public:
+ using Entry = MegamorphicSetPropCacheEntry;
+ // We can get more hits if we increase this, but this seems to be around
+ // the sweet spot where we are getting most of the hits we would get with
+ // an infinitely sized cache
+ static constexpr size_t NumEntries = 1024;
+ static constexpr uint8_t ShapeHashShift1 =
+ mozilla::tl::FloorLog2<alignof(Shape)>::value;
+ static constexpr uint8_t ShapeHashShift2 =
+ ShapeHashShift1 + mozilla::tl::FloorLog2<NumEntries>::value;
+
+ static_assert(mozilla::IsPowerOfTwo(alignof(Shape)) &&
+ mozilla::IsPowerOfTwo(NumEntries),
+ "FloorLog2 is exact because alignof(Shape) and NumEntries are "
+ "both powers of two");
+
+ private:
+ mozilla::Array<Entry, NumEntries> entries_;
+
+ // Generation counter used to invalidate all entries.
+ uint16_t generation_ = 0;
+
+ Entry& getEntry(Shape* beforeShape, PropertyKey key) {
+ static_assert(mozilla::IsPowerOfTwo(NumEntries),
+ "NumEntries must be a power-of-two for fast modulo");
+ uintptr_t hash = uintptr_t(beforeShape) >> ShapeHashShift1;
+ hash ^= uintptr_t(beforeShape) >> ShapeHashShift2;
+ hash += HashAtomOrSymbolPropertyKey(key);
+ return entries_[hash % NumEntries];
+ }
+
+ public:
+ void bumpGeneration() {
+ generation_++;
+ if (generation_ == 0) {
+ // Generation overflowed. Invalidate the whole cache.
+ for (size_t i = 0; i < NumEntries; i++) {
+ entries_[i].beforeShape_ = nullptr;
+ }
+ }
+ }
+ void set(Shape* beforeShape, Shape* afterShape, PropertyKey key,
+ TaggedSlotOffset slotOffset, uint32_t newCapacity) {
+ uint16_t newSlots = (uint16_t)newCapacity;
+ if (newSlots != newCapacity) {
+ return;
+ }
+ Entry& entry = getEntry(beforeShape, key);
+ entry.init(beforeShape, afterShape, key, generation_, slotOffset, newSlots);
+ }
+
+#ifdef DEBUG
+ bool lookup(Shape* beforeShape, PropertyKey key, Entry** entryp) {
+ Entry& entry = getEntry(beforeShape, key);
+ *entryp = &entry;
+ return (entry.beforeShape_ == beforeShape && entry.key_ == key &&
+ entry.generation_ == generation_);
+ }
+#endif
+
+ static constexpr size_t offsetOfEntries() {
+ return offsetof(MegamorphicSetPropCache, entries_);
+ }
+
+ static constexpr size_t offsetOfGeneration() {
+ return offsetof(MegamorphicSetPropCache, generation_);
+ }
+};
+
+// Cache for AtomizeString, mapping JSString* or JS::Latin1Char* to the
+// corresponding JSAtom*. The cache has three different optimizations:
+//
+// * The two most recent lookups are cached. This has a hit rate of 30-65% on
+// typical web workloads.
+//
+// * MruCache is used for short JS::Latin1Char strings.
+//
+// * For longer strings, there's also a JSLinearString* => JSAtom* HashMap,
+// because hashing the string characters repeatedly can be slow.
+// This map is also used by nursery GC to de-duplicate strings to atoms.
+//
+// This cache is purged on minor and major GC.
+class StringToAtomCache {
+ public:
+ struct LastLookup {
+ JSString* string = nullptr;
+ JSAtom* atom = nullptr;
+
+ static constexpr size_t offsetOfString() {
+ return offsetof(LastLookup, string);
+ }
+
+ static constexpr size_t offsetOfAtom() {
+ return offsetof(LastLookup, atom);
+ }
+ };
+ static constexpr size_t NumLastLookups = 2;
+
+ struct AtomTableKey {
+ explicit AtomTableKey(const JS::Latin1Char* str, size_t len)
+ : string_(str), length_(len) {
+ hash_ = mozilla::HashString(string_, length_);
+ }
+
+ const JS::Latin1Char* string_;
+ size_t length_;
+ uint32_t hash_;
+ };
+
+ private:
+ struct RopeAtomCache
+ : public mozilla::MruCache<AtomTableKey, JSAtom*, RopeAtomCache> {
+ static HashNumber Hash(const AtomTableKey& key) { return key.hash_; }
+ static bool Match(const AtomTableKey& key, const JSAtom* val) {
+ JS::AutoCheckCannotGC nogc;
+ return val->length() == key.length_ &&
+ EqualChars(key.string_, val->latin1Chars(nogc), key.length_);
+ }
+ };
+ using Map =
+ HashMap<JSString*, JSAtom*, PointerHasher<JSString*>, SystemAllocPolicy>;
+ Map map_;
+ mozilla::Array<LastLookup, NumLastLookups> lastLookups_;
+ RopeAtomCache ropeCharCache_;
+
+ public:
+ // Don't use the HashMap for short strings. Hashing them is less expensive.
+ // But the length needs to long enough to cover common identifiers in React.
+ static constexpr size_t MinStringLength = 39;
+
+ JSAtom* lookupInMap(JSString* s) const {
+ MOZ_ASSERT(s->inStringToAtomCache());
+ MOZ_ASSERT(s->length() >= MinStringLength);
+
+ auto p = map_.lookup(s);
+ JSAtom* atom = p ? p->value() : nullptr;
+ return atom;
+ }
+
+ MOZ_ALWAYS_INLINE JSAtom* lookup(JSString* s) const {
+ MOZ_ASSERT(!s->isAtom());
+ for (const LastLookup& entry : lastLookups_) {
+ if (entry.string == s) {
+ return entry.atom;
+ }
+ }
+
+ if (!s->inStringToAtomCache()) {
+ MOZ_ASSERT(!map_.lookup(s));
+ return nullptr;
+ }
+
+ return lookupInMap(s);
+ }
+
+ MOZ_ALWAYS_INLINE JSAtom* lookupWithRopeChars(
+ const JS::Latin1Char* str, size_t len,
+ mozilla::Maybe<AtomTableKey>& key) {
+ MOZ_ASSERT(len < MinStringLength);
+ key.emplace(str, len);
+ if (auto p = ropeCharCache_.Lookup(key.value())) {
+ return p.Data();
+ }
+ return nullptr;
+ }
+
+ static constexpr size_t offsetOfLastLookups() {
+ return offsetof(StringToAtomCache, lastLookups_);
+ }
+
+ void maybePut(JSString* s, JSAtom* atom, mozilla::Maybe<AtomTableKey>& key) {
+ if (key.isSome()) {
+ ropeCharCache_.Put(key.value(), atom);
+ }
+
+ for (size_t i = NumLastLookups - 1; i > 0; i--) {
+ lastLookups_[i] = lastLookups_[i - 1];
+ }
+ lastLookups_[0].string = s;
+ lastLookups_[0].atom = atom;
+
+ if (s->length() < MinStringLength) {
+ return;
+ }
+ if (!map_.putNew(s, atom)) {
+ return;
+ }
+ s->setInStringToAtomCache();
+ }
+
+ void purge() {
+ map_.clearAndCompact();
+ for (LastLookup& entry : lastLookups_) {
+ entry.string = nullptr;
+ entry.atom = nullptr;
+ }
+
+ ropeCharCache_.Clear();
+ }
+};
+
+class RuntimeCaches {
+ public:
+ MegamorphicCache megamorphicCache;
+ UniquePtr<MegamorphicSetPropCache> megamorphicSetPropCache;
+ GSNCache gsnCache;
+ UncompressedSourceCache uncompressedSourceCache;
+ EvalCache evalCache;
+ StringToAtomCache stringToAtomCache;
+
+ // Delazification: Cache binding for runtime objects which are used during
+ // delazification to quickly resolve NameLocation of bindings without linearly
+ // iterating over the list of bindings.
+ frontend::RuntimeScopeBindingCache scopeCache;
+
+ // This cache is used to store the result of delazification compilations which
+ // might be happening off-thread. The main-thread will concurrently read the
+ // content of this cache to avoid delazification, or fallback on running the
+ // delazification on the main-thread.
+ //
+ // Main-thread results are not stored in the StencilCache as there is no other
+ // consumer.
+ StencilCache delazificationCache;
+
+ void sweepAfterMinorGC(JSTracer* trc) { evalCache.traceWeak(trc); }
+#ifdef JSGC_HASH_TABLE_CHECKS
+ void checkEvalCacheAfterMinorGC();
+#endif
+
+ void purgeForCompaction() {
+ evalCache.clear();
+ stringToAtomCache.purge();
+ megamorphicCache.bumpGeneration();
+ if (megamorphicSetPropCache) {
+ // MegamorphicSetPropCache can be null if we failed out of
+ // JSRuntime::init. We will then try to destroy the runtime which will
+ // do a GC and land us here.
+ megamorphicSetPropCache->bumpGeneration();
+ }
+ scopeCache.purge();
+ }
+
+ void purgeStencils() { delazificationCache.clearAndDisable(); }
+
+ void purge() {
+ purgeForCompaction();
+ gsnCache.purge();
+ uncompressedSourceCache.purge();
+ purgeStencils();
+ }
+};
+
+} // namespace js
+
+#endif /* vm_Caches_h */