/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* 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 nsXULPrototypeCache_h__ #define nsXULPrototypeCache_h__ #include "nsBaseHashtable.h" #include "nsCOMPtr.h" #include "nsIObserver.h" #include "nsInterfaceHashtable.h" #include "nsRefPtrHashtable.h" #include "nsURIHashKey.h" #include "nsXULPrototypeDocument.h" #include "nsIStorageStream.h" #include "mozilla/scache/StartupCache.h" #include "js/experimental/JSStencil.h" #include "mozilla/RefPtr.h" class nsIHandleReportCallback; namespace mozilla { class StyleSheet; } // namespace mozilla /** * The XUL prototype cache can be used to store and retrieve shared data for * XUL documents, style sheets, XBL, and scripts. * * The cache has two levels: * 1. In-memory hashtables * 2. The on-disk cache file. */ class nsXULPrototypeCache : public nsIObserver { public: // nsISupports NS_DECL_THREADSAFE_ISUPPORTS NS_DECL_NSIOBSERVER bool IsCached(nsIURI* aURI) { return GetPrototype(aURI) != nullptr; } void AbortCaching(); /** * Whether the prototype cache is enabled. */ bool IsEnabled(); /** * Flush the cache; remove all XUL prototype documents, style * sheets, and scripts. */ void Flush(); // The following methods are used to put and retrive various items into and // from the cache. nsXULPrototypeDocument* GetPrototype(nsIURI* aURI); nsresult PutPrototype(nsXULPrototypeDocument* aDocument); JS::Stencil* GetStencil(nsIURI* aURI); nsresult PutStencil(nsIURI* aURI, JS::Stencil* aStencil); /** * Write the XUL prototype document to a cache file. The proto must be * fully loaded. */ nsresult WritePrototype(nsXULPrototypeDocument* aPrototypeDocument); /** * This interface allows partial reads and writes from the buffers in the * startupCache. */ nsresult GetInputStream(nsIURI* aURI, nsIObjectInputStream** objectInput); nsresult FinishInputStream(nsIURI* aURI); nsresult GetOutputStream(nsIURI* aURI, nsIObjectOutputStream** objectOutput); nsresult FinishOutputStream(nsIURI* aURI); nsresult HasData(nsIURI* aURI, bool* exists); static nsXULPrototypeCache* GetInstance(); static nsXULPrototypeCache* MaybeGetInstance() { return sInstance; } static void ReleaseGlobals() { NS_IF_RELEASE(sInstance); } void MarkInCCGeneration(uint32_t aGeneration); static void CollectMemoryReports(nsIHandleReportCallback* aHandleReport, nsISupports* aData); protected: friend nsresult NS_NewXULPrototypeCache(nsISupports* aOuter, REFNSIID aIID, void** aResult); nsXULPrototypeCache(); virtual ~nsXULPrototypeCache() = default; static nsXULPrototypeCache* sInstance; nsRefPtrHashtable mPrototypeTable; // owns the prototypes class StencilHashKey : public nsURIHashKey { public: explicit StencilHashKey(const nsIURI* aKey) : nsURIHashKey(aKey) {} StencilHashKey(StencilHashKey&&) = default; RefPtr mStencil; }; nsTHashtable mStencilTable; // URIs already written to the startup cache, to prevent double-caching. nsTHashtable mStartupCacheURITable; nsInterfaceHashtable mOutputStreamTable; nsInterfaceHashtable mInputStreamTable; // Bootstrap caching service nsresult BeginCaching(nsIURI* aDocumentURI); }; #endif // nsXULPrototypeCache_h__