summaryrefslogtreecommitdiffstats
path: root/layout/style/SharedStyleSheetCache.h
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--layout/style/SharedStyleSheetCache.h145
1 files changed, 145 insertions, 0 deletions
diff --git a/layout/style/SharedStyleSheetCache.h b/layout/style/SharedStyleSheetCache.h
new file mode 100644
index 0000000000..8d134caf66
--- /dev/null
+++ b/layout/style/SharedStyleSheetCache.h
@@ -0,0 +1,145 @@
+/* -*- 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 mozilla_SharedStyleSheetCache_h__
+#define mozilla_SharedStyleSheetCache_h__
+
+// The shared style sheet cache is a cache that allows us to share sheets across
+// documents.
+//
+// It's generally a singleton, but it is different from GlobalStyleSheetCache in
+// the sense that:
+//
+// * It needs to be cycle-collectable, as it can keep alive style sheets from
+// various documents.
+//
+// * It is conceptually a singleton, but given its cycle-collectable nature, we
+// might re-create it.
+
+#include "mozilla/PrincipalHashKey.h"
+#include "mozilla/WeakPtr.h"
+#include "mozilla/css/Loader.h"
+#include "nsDataHashtable.h"
+#include "nsIMemoryReporter.h"
+#include "nsRefPtrHashtable.h"
+#include "nsURIHashKey.h"
+
+namespace mozilla {
+
+namespace css {
+class SheetLoadData;
+}
+
+class SharedStyleSheetCache final : public nsIMemoryReporter {
+ public:
+ NS_DECL_ISUPPORTS
+ NS_DECL_NSIMEMORYREPORTER
+
+ SharedStyleSheetCache(const SharedStyleSheetCache&) = delete;
+ SharedStyleSheetCache(SharedStyleSheetCache&&) = delete;
+
+ static already_AddRefed<SharedStyleSheetCache> Get() {
+ if (sInstance) {
+ return do_AddRef(sInstance);
+ }
+ return Create();
+ }
+
+ public:
+ // TODO(emilio): Maybe cache inline sheets here rather than in the loader?
+ // But that is a bit dubious, because for inline sheets, the sheet URI is the
+ // document URI, so it is basically document-specific... Maybe we can cache
+ // the RawServoStyleSheetContents or something, but...
+
+ // A cache hit or miss. It is a miss if the `StyleSheet` is null.
+ using CacheResult = std::tuple<RefPtr<StyleSheet>, css::Loader::SheetState>;
+ CacheResult Lookup(css::Loader&, const SheetLoadDataHashKey&, bool aSyncLoad);
+
+ // Tries to coalesce with an already existing load. The sheet state must be
+ // the one that Lookup returned, if it returned a sheet.
+ //
+ // TODO(emilio): Maybe try to merge this with the lookup? Most consumers could
+ // have a data there already.
+ MOZ_MUST_USE bool CoalesceLoad(const SheetLoadDataHashKey&,
+ css::SheetLoadData& aNewLoad,
+ css::Loader::SheetState aExistingLoadState);
+
+ size_t SizeOfIncludingThis(MallocSizeOf) const;
+
+ // Puts the load into the "loading" set.
+ void LoadStarted(const SheetLoadDataHashKey&, css::SheetLoadData&);
+ // Puts a load into the "pending" set.
+ void DeferSheetLoad(const SheetLoadDataHashKey&, css::SheetLoadData&);
+
+ enum class StartLoads {
+ Always,
+ IfNonAlternate,
+ };
+
+ void StartDeferredLoadsForLoader(css::Loader&, StartLoads);
+ void CancelLoadsForLoader(css::Loader&);
+
+ // This has to be static because it's also called for loaders that don't have
+ // a sheet cache (loaders that are not owned by a document).
+ static void LoadCompleted(SharedStyleSheetCache*, css::SheetLoadData&,
+ nsresult);
+
+ // Register a loader into the cache. This has the effect of keeping alive all
+ // stylesheets for the origin of the loader's document until UnregisterLoader
+ // is called.
+ void RegisterLoader(css::Loader&);
+
+ // Unregister a loader from the cache.
+ //
+ // If this is the loader for the last document of a given origin, then all the
+ // stylesheets for that document will be removed from the cache. This needs to
+ // be called when the document goes away, or when its principal changes.
+ void UnregisterLoader(css::Loader&);
+
+ static void Clear(nsIPrincipal* aForPrincipal = nullptr);
+
+ private:
+ static already_AddRefed<SharedStyleSheetCache> Create();
+ void CancelDeferredLoadsForLoader(css::Loader&);
+
+ static void LoadCompletedInternal(SharedStyleSheetCache*, css::SheetLoadData&,
+ nsTArray<RefPtr<css::SheetLoadData>>&);
+ void InsertIntoCompleteCacheIfNeeded(css::SheetLoadData&);
+
+ SharedStyleSheetCache() = default;
+
+ ~SharedStyleSheetCache();
+
+ struct CompleteSheet {
+ uint32_t mExpirationTime = 0;
+ UniquePtr<StyleUseCounters> mUseCounters;
+ RefPtr<StyleSheet> mSheet;
+
+ bool Expired() const;
+ };
+
+ void WillStartPendingLoad(css::SheetLoadData&);
+
+ nsDataHashtable<SheetLoadDataHashKey, CompleteSheet> mCompleteSheets;
+ nsRefPtrHashtable<SheetLoadDataHashKey, css::SheetLoadData> mPendingDatas;
+ // The SheetLoadData pointers in mLoadingDatas below are weak references that
+ // get cleaned up when StreamLoader::OnStopRequest gets called.
+ //
+ // Note that we hold on to all sheet loads, even if in the end they happen not
+ // to be cacheable.
+ nsDataHashtable<SheetLoadDataHashKey, WeakPtr<css::SheetLoadData>>
+ mLoadingDatas;
+
+ // An origin-to-number-of-registered-documents count, in order to manage cache
+ // eviction as described in RegisterLoader / UnregisterLoader.
+ nsDataHashtable<PrincipalHashKey, uint32_t> mLoaderPrincipalRefCnt;
+
+ static SharedStyleSheetCache* sInstance;
+};
+
+} // namespace mozilla
+
+#endif