forked from mirrors/gecko-dev
		
	
		
			
				
	
	
		
			1967 lines
		
	
	
	
		
			68 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			1967 lines
		
	
	
	
		
			68 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /* -*- Mode: C++; tab-width: 2; 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/. */
 | |
| 
 | |
| /**
 | |
|  * SurfaceCache is a service for caching temporary surfaces in imagelib.
 | |
|  */
 | |
| 
 | |
| #include "SurfaceCache.h"
 | |
| 
 | |
| #include <algorithm>
 | |
| #include <utility>
 | |
| 
 | |
| #include "ISurfaceProvider.h"
 | |
| #include "Image.h"
 | |
| #include "LookupResult.h"
 | |
| #include "ShutdownTracker.h"
 | |
| #include "gfx2DGlue.h"
 | |
| #include "gfxPlatform.h"
 | |
| #include "imgFrame.h"
 | |
| #include "mozilla/AppShutdown.h"
 | |
| #include "mozilla/Assertions.h"
 | |
| #include "mozilla/Attributes.h"
 | |
| #include "mozilla/CheckedInt.h"
 | |
| #include "mozilla/DebugOnly.h"
 | |
| #include "mozilla/Likely.h"
 | |
| #include "mozilla/RefPtr.h"
 | |
| #include "mozilla/StaticMutex.h"
 | |
| #include "mozilla/StaticPrefs_image.h"
 | |
| #include "mozilla/StaticPtr.h"
 | |
| #include "mozilla/Tuple.h"
 | |
| #include "nsExpirationTracker.h"
 | |
| #include "nsHashKeys.h"
 | |
| #include "nsIMemoryReporter.h"
 | |
| #include "nsRefPtrHashtable.h"
 | |
| #include "nsSize.h"
 | |
| #include "nsTArray.h"
 | |
| #include "Orientation.h"
 | |
| #include "prsystem.h"
 | |
| 
 | |
| using std::max;
 | |
| using std::min;
 | |
| 
 | |
| namespace mozilla {
 | |
| 
 | |
| using namespace gfx;
 | |
| 
 | |
| namespace image {
 | |
| 
 | |
| MOZ_DEFINE_MALLOC_SIZE_OF(SurfaceCacheMallocSizeOf)
 | |
| 
 | |
| class CachedSurface;
 | |
| class SurfaceCacheImpl;
 | |
| 
 | |
| ///////////////////////////////////////////////////////////////////////////////
 | |
| // Static Data
 | |
| ///////////////////////////////////////////////////////////////////////////////
 | |
| 
 | |
| // The single surface cache instance.
 | |
| static StaticRefPtr<SurfaceCacheImpl> sInstance;
 | |
| 
 | |
| // The mutex protecting the surface cache.
 | |
| static StaticMutex sInstanceMutex MOZ_UNANNOTATED;
 | |
| 
 | |
| ///////////////////////////////////////////////////////////////////////////////
 | |
| // SurfaceCache Implementation
 | |
| ///////////////////////////////////////////////////////////////////////////////
 | |
| 
 | |
| /**
 | |
|  * Cost models the cost of storing a surface in the cache. Right now, this is
 | |
|  * simply an estimate of the size of the surface in bytes, but in the future it
 | |
|  * may be worth taking into account the cost of rematerializing the surface as
 | |
|  * well.
 | |
|  */
 | |
| typedef size_t Cost;
 | |
| 
 | |
| static Cost ComputeCost(const IntSize& aSize, uint32_t aBytesPerPixel) {
 | |
|   MOZ_ASSERT(aBytesPerPixel == 1 || aBytesPerPixel == 4);
 | |
|   return aSize.width * aSize.height * aBytesPerPixel;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Since we want to be able to make eviction decisions based on cost, we need to
 | |
|  * be able to look up the CachedSurface which has a certain cost as well as the
 | |
|  * cost associated with a certain CachedSurface. To make this possible, in data
 | |
|  * structures we actually store a CostEntry, which contains a weak pointer to
 | |
|  * its associated surface.
 | |
|  *
 | |
|  * To make usage of the weak pointer safe, SurfaceCacheImpl always calls
 | |
|  * StartTracking after a surface is stored in the cache and StopTracking before
 | |
|  * it is removed.
 | |
|  */
 | |
| class CostEntry {
 | |
|  public:
 | |
|   CostEntry(NotNull<CachedSurface*> aSurface, Cost aCost)
 | |
|       : mSurface(aSurface), mCost(aCost) {}
 | |
| 
 | |
|   NotNull<CachedSurface*> Surface() const { return mSurface; }
 | |
|   Cost GetCost() const { return mCost; }
 | |
| 
 | |
|   bool operator==(const CostEntry& aOther) const {
 | |
|     return mSurface == aOther.mSurface && mCost == aOther.mCost;
 | |
|   }
 | |
| 
 | |
|   bool operator<(const CostEntry& aOther) const {
 | |
|     return mCost < aOther.mCost ||
 | |
|            (mCost == aOther.mCost && mSurface < aOther.mSurface);
 | |
|   }
 | |
| 
 | |
|  private:
 | |
|   NotNull<CachedSurface*> mSurface;
 | |
|   Cost mCost;
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * A CachedSurface associates a surface with a key that uniquely identifies that
 | |
|  * surface.
 | |
|  */
 | |
| class CachedSurface {
 | |
|   ~CachedSurface() {}
 | |
| 
 | |
|  public:
 | |
|   MOZ_DECLARE_REFCOUNTED_TYPENAME(CachedSurface)
 | |
|   NS_INLINE_DECL_THREADSAFE_REFCOUNTING(CachedSurface)
 | |
| 
 | |
|   explicit CachedSurface(NotNull<ISurfaceProvider*> aProvider)
 | |
|       : mProvider(aProvider), mIsLocked(false) {}
 | |
| 
 | |
|   DrawableSurface GetDrawableSurface() const {
 | |
|     if (MOZ_UNLIKELY(IsPlaceholder())) {
 | |
|       MOZ_ASSERT_UNREACHABLE("Called GetDrawableSurface() on a placeholder");
 | |
|       return DrawableSurface();
 | |
|     }
 | |
| 
 | |
|     return mProvider->Surface();
 | |
|   }
 | |
| 
 | |
|   DrawableSurface GetDrawableSurfaceEvenIfPlaceholder() const {
 | |
|     return mProvider->Surface();
 | |
|   }
 | |
| 
 | |
|   void SetLocked(bool aLocked) {
 | |
|     if (IsPlaceholder()) {
 | |
|       return;  // Can't lock a placeholder.
 | |
|     }
 | |
| 
 | |
|     // Update both our state and our provider's state. Some surface providers
 | |
|     // are permanently locked; maintaining our own locking state enables us to
 | |
|     // respect SetLocked() even when it's meaningless from the provider's
 | |
|     // perspective.
 | |
|     mIsLocked = aLocked;
 | |
|     mProvider->SetLocked(aLocked);
 | |
|   }
 | |
| 
 | |
|   bool IsLocked() const {
 | |
|     return !IsPlaceholder() && mIsLocked && mProvider->IsLocked();
 | |
|   }
 | |
| 
 | |
|   void SetCannotSubstitute() {
 | |
|     mProvider->Availability().SetCannotSubstitute();
 | |
|   }
 | |
|   bool CannotSubstitute() const {
 | |
|     return mProvider->Availability().CannotSubstitute();
 | |
|   }
 | |
| 
 | |
|   bool IsPlaceholder() const {
 | |
|     return mProvider->Availability().IsPlaceholder();
 | |
|   }
 | |
|   bool IsDecoded() const { return !IsPlaceholder() && mProvider->IsFinished(); }
 | |
| 
 | |
|   ImageKey GetImageKey() const { return mProvider->GetImageKey(); }
 | |
|   const SurfaceKey& GetSurfaceKey() const { return mProvider->GetSurfaceKey(); }
 | |
|   nsExpirationState* GetExpirationState() { return &mExpirationState; }
 | |
| 
 | |
|   CostEntry GetCostEntry() {
 | |
|     return image::CostEntry(WrapNotNull(this), mProvider->LogicalSizeInBytes());
 | |
|   }
 | |
| 
 | |
|   size_t ShallowSizeOfIncludingThis(MallocSizeOf aMallocSizeOf) const {
 | |
|     return aMallocSizeOf(this) + aMallocSizeOf(mProvider.get());
 | |
|   }
 | |
| 
 | |
|   void InvalidateRecording() { mProvider->InvalidateRecording(); }
 | |
| 
 | |
|   // A helper type used by SurfaceCacheImpl::CollectSizeOfSurfaces.
 | |
|   struct MOZ_STACK_CLASS SurfaceMemoryReport {
 | |
|     SurfaceMemoryReport(nsTArray<SurfaceMemoryCounter>& aCounters,
 | |
|                         MallocSizeOf aMallocSizeOf)
 | |
|         : mCounters(aCounters), mMallocSizeOf(aMallocSizeOf) {}
 | |
| 
 | |
|     void Add(NotNull<CachedSurface*> aCachedSurface, bool aIsFactor2) {
 | |
|       if (aCachedSurface->IsPlaceholder()) {
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       // Record the memory used by the ISurfaceProvider. This may not have a
 | |
|       // straightforward relationship to the size of the surface that
 | |
|       // DrawableRef() returns if the surface is generated dynamically. (i.e.,
 | |
|       // for surfaces with PlaybackType::eAnimated.)
 | |
|       aCachedSurface->mProvider->AddSizeOfExcludingThis(
 | |
|           mMallocSizeOf, [&](ISurfaceProvider::AddSizeOfCbData& aMetadata) {
 | |
|             SurfaceMemoryCounter counter(aCachedSurface->GetSurfaceKey(),
 | |
|                                          aCachedSurface->IsLocked(),
 | |
|                                          aCachedSurface->CannotSubstitute(),
 | |
|                                          aIsFactor2, aMetadata.mFinished);
 | |
| 
 | |
|             counter.Values().SetDecodedHeap(aMetadata.mHeapBytes);
 | |
|             counter.Values().SetDecodedNonHeap(aMetadata.mNonHeapBytes);
 | |
|             counter.Values().SetDecodedUnknown(aMetadata.mUnknownBytes);
 | |
|             counter.Values().SetExternalHandles(aMetadata.mExternalHandles);
 | |
|             counter.Values().SetFrameIndex(aMetadata.mIndex);
 | |
|             counter.Values().SetExternalId(aMetadata.mExternalId);
 | |
|             counter.Values().SetSurfaceTypes(aMetadata.mTypes);
 | |
| 
 | |
|             mCounters.AppendElement(counter);
 | |
|           });
 | |
|     }
 | |
| 
 | |
|    private:
 | |
|     nsTArray<SurfaceMemoryCounter>& mCounters;
 | |
|     MallocSizeOf mMallocSizeOf;
 | |
|   };
 | |
| 
 | |
|  private:
 | |
|   nsExpirationState mExpirationState;
 | |
|   NotNull<RefPtr<ISurfaceProvider>> mProvider;
 | |
|   bool mIsLocked;
 | |
| };
 | |
| 
 | |
| static int64_t AreaOfIntSize(const IntSize& aSize) {
 | |
|   return static_cast<int64_t>(aSize.width) * static_cast<int64_t>(aSize.height);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * An ImageSurfaceCache is a per-image surface cache. For correctness we must be
 | |
|  * able to remove all surfaces associated with an image when the image is
 | |
|  * destroyed or invalidated. Since this will happen frequently, it makes sense
 | |
|  * to make it cheap by storing the surfaces for each image separately.
 | |
|  *
 | |
|  * ImageSurfaceCache also keeps track of whether its associated image is locked
 | |
|  * or unlocked.
 | |
|  *
 | |
|  * The cache may also enter "factor of 2" mode which occurs when the number of
 | |
|  * surfaces in the cache exceeds the "image.cache.factor2.threshold-surfaces"
 | |
|  * pref plus the number of native sizes of the image. When in "factor of 2"
 | |
|  * mode, the cache will strongly favour sizes which are a factor of 2 of the
 | |
|  * largest native size. It accomplishes this by suggesting a factor of 2 size
 | |
|  * when lookups fail and substituting the nearest factor of 2 surface to the
 | |
|  * ideal size as the "best" available (as opposed to substitution but not
 | |
|  * found). This allows us to minimize memory consumption and CPU time spent
 | |
|  * decoding when a website requires many variants of the same surface.
 | |
|  */
 | |
| class ImageSurfaceCache {
 | |
|   ~ImageSurfaceCache() {}
 | |
| 
 | |
|  public:
 | |
|   explicit ImageSurfaceCache(const ImageKey aImageKey)
 | |
|       : mLocked(false),
 | |
|         mFactor2Mode(false),
 | |
|         mFactor2Pruned(false),
 | |
|         mIsVectorImage(aImageKey->GetType() == imgIContainer::TYPE_VECTOR) {}
 | |
| 
 | |
|   MOZ_DECLARE_REFCOUNTED_TYPENAME(ImageSurfaceCache)
 | |
|   NS_INLINE_DECL_THREADSAFE_REFCOUNTING(ImageSurfaceCache)
 | |
| 
 | |
|   typedef nsRefPtrHashtable<nsGenericHashKey<SurfaceKey>, CachedSurface>
 | |
|       SurfaceTable;
 | |
| 
 | |
|   auto Values() const { return mSurfaces.Values(); }
 | |
|   uint32_t Count() const { return mSurfaces.Count(); }
 | |
|   bool IsEmpty() const { return mSurfaces.Count() == 0; }
 | |
| 
 | |
|   size_t ShallowSizeOfIncludingThis(MallocSizeOf aMallocSizeOf) const {
 | |
|     size_t bytes = aMallocSizeOf(this) +
 | |
|                    mSurfaces.ShallowSizeOfExcludingThis(aMallocSizeOf);
 | |
|     for (const auto& value : Values()) {
 | |
|       bytes += value->ShallowSizeOfIncludingThis(aMallocSizeOf);
 | |
|     }
 | |
|     return bytes;
 | |
|   }
 | |
| 
 | |
|   [[nodiscard]] bool Insert(NotNull<CachedSurface*> aSurface) {
 | |
|     MOZ_ASSERT(!mLocked || aSurface->IsPlaceholder() || aSurface->IsLocked(),
 | |
|                "Inserting an unlocked surface for a locked image");
 | |
|     const auto& surfaceKey = aSurface->GetSurfaceKey();
 | |
|     if (surfaceKey.Region()) {
 | |
|       // We don't allow substitutes for surfaces with regions, so we don't want
 | |
|       // to allow factor of 2 mode pruning to release these surfaces.
 | |
|       aSurface->SetCannotSubstitute();
 | |
|     }
 | |
|     return mSurfaces.InsertOrUpdate(surfaceKey, RefPtr<CachedSurface>{aSurface},
 | |
|                                     fallible);
 | |
|   }
 | |
| 
 | |
|   already_AddRefed<CachedSurface> Remove(NotNull<CachedSurface*> aSurface) {
 | |
|     MOZ_ASSERT(mSurfaces.GetWeak(aSurface->GetSurfaceKey()),
 | |
|                "Should not be removing a surface we don't have");
 | |
| 
 | |
|     RefPtr<CachedSurface> surface;
 | |
|     mSurfaces.Remove(aSurface->GetSurfaceKey(), getter_AddRefs(surface));
 | |
|     AfterMaybeRemove();
 | |
|     return surface.forget();
 | |
|   }
 | |
| 
 | |
|   already_AddRefed<CachedSurface> Lookup(const SurfaceKey& aSurfaceKey,
 | |
|                                          bool aForAccess) {
 | |
|     RefPtr<CachedSurface> surface;
 | |
|     mSurfaces.Get(aSurfaceKey, getter_AddRefs(surface));
 | |
| 
 | |
|     if (aForAccess) {
 | |
|       if (surface) {
 | |
|         // We don't want to allow factor of 2 mode pruning to release surfaces
 | |
|         // for which the callers will accept no substitute.
 | |
|         surface->SetCannotSubstitute();
 | |
|       } else if (!mFactor2Mode) {
 | |
|         // If no exact match is found, and this is for use rather than internal
 | |
|         // accounting (i.e. insert and removal), we know this will trigger a
 | |
|         // decode. Make sure we switch now to factor of 2 mode if necessary.
 | |
|         MaybeSetFactor2Mode();
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     return surface.forget();
 | |
|   }
 | |
| 
 | |
|   /**
 | |
|    * @returns A tuple containing the best matching CachedSurface if available,
 | |
|    *          a MatchType describing how the CachedSurface was selected, and
 | |
|    *          an IntSize which is the size the caller should choose to decode
 | |
|    *          at should it attempt to do so.
 | |
|    */
 | |
|   Tuple<already_AddRefed<CachedSurface>, MatchType, IntSize> LookupBestMatch(
 | |
|       const SurfaceKey& aIdealKey) {
 | |
|     // Try for an exact match first.
 | |
|     RefPtr<CachedSurface> exactMatch;
 | |
|     mSurfaces.Get(aIdealKey, getter_AddRefs(exactMatch));
 | |
|     if (exactMatch) {
 | |
|       if (exactMatch->IsDecoded()) {
 | |
|         return MakeTuple(exactMatch.forget(), MatchType::EXACT, IntSize());
 | |
|       }
 | |
|     } else if (aIdealKey.Region()) {
 | |
|       // We cannot substitute if we have a region. Allow it to create an exact
 | |
|       // match.
 | |
|       return MakeTuple(exactMatch.forget(), MatchType::NOT_FOUND, IntSize());
 | |
|     } else if (!mFactor2Mode) {
 | |
|       // If no exact match is found, and we are not in factor of 2 mode, then
 | |
|       // we know that we will trigger a decode because at best we will provide
 | |
|       // a substitute. Make sure we switch now to factor of 2 mode if necessary.
 | |
|       MaybeSetFactor2Mode();
 | |
|     }
 | |
| 
 | |
|     // Try for a best match second, if using compact.
 | |
|     IntSize suggestedSize = SuggestedSize(aIdealKey.Size());
 | |
|     if (suggestedSize != aIdealKey.Size()) {
 | |
|       if (!exactMatch) {
 | |
|         SurfaceKey compactKey = aIdealKey.CloneWithSize(suggestedSize);
 | |
|         mSurfaces.Get(compactKey, getter_AddRefs(exactMatch));
 | |
|         if (exactMatch && exactMatch->IsDecoded()) {
 | |
|           MOZ_ASSERT(suggestedSize != aIdealKey.Size());
 | |
|           return MakeTuple(exactMatch.forget(),
 | |
|                            MatchType::SUBSTITUTE_BECAUSE_BEST, suggestedSize);
 | |
|         }
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     // There's no perfect match, so find the best match we can.
 | |
|     RefPtr<CachedSurface> bestMatch;
 | |
|     for (const auto& value : Values()) {
 | |
|       NotNull<CachedSurface*> current = WrapNotNull(value);
 | |
|       const SurfaceKey& currentKey = current->GetSurfaceKey();
 | |
| 
 | |
|       // We never match a placeholder or a surface with a region.
 | |
|       if (current->IsPlaceholder() || currentKey.Region()) {
 | |
|         continue;
 | |
|       }
 | |
|       // Matching the playback type and SVG context is required.
 | |
|       if (currentKey.Playback() != aIdealKey.Playback() ||
 | |
|           currentKey.SVGContext() != aIdealKey.SVGContext()) {
 | |
|         continue;
 | |
|       }
 | |
|       // Matching the flags is required.
 | |
|       if (currentKey.Flags() != aIdealKey.Flags()) {
 | |
|         continue;
 | |
|       }
 | |
|       // Anything is better than nothing! (Within the constraints we just
 | |
|       // checked, of course.)
 | |
|       if (!bestMatch) {
 | |
|         bestMatch = current;
 | |
|         continue;
 | |
|       }
 | |
| 
 | |
|       MOZ_ASSERT(bestMatch, "Should have a current best match");
 | |
| 
 | |
|       // Always prefer completely decoded surfaces.
 | |
|       bool bestMatchIsDecoded = bestMatch->IsDecoded();
 | |
|       if (bestMatchIsDecoded && !current->IsDecoded()) {
 | |
|         continue;
 | |
|       }
 | |
|       if (!bestMatchIsDecoded && current->IsDecoded()) {
 | |
|         bestMatch = current;
 | |
|         continue;
 | |
|       }
 | |
| 
 | |
|       SurfaceKey bestMatchKey = bestMatch->GetSurfaceKey();
 | |
|       if (CompareArea(aIdealKey.Size(), bestMatchKey.Size(),
 | |
|                       currentKey.Size())) {
 | |
|         bestMatch = current;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     MatchType matchType;
 | |
|     if (bestMatch) {
 | |
|       if (!exactMatch) {
 | |
|         // No exact match, neither ideal nor factor of 2.
 | |
|         MOZ_ASSERT(suggestedSize != bestMatch->GetSurfaceKey().Size(),
 | |
|                    "No exact match despite the fact the sizes match!");
 | |
|         matchType = MatchType::SUBSTITUTE_BECAUSE_NOT_FOUND;
 | |
|       } else if (exactMatch != bestMatch) {
 | |
|         // The exact match is still decoding, but we found a substitute.
 | |
|         matchType = MatchType::SUBSTITUTE_BECAUSE_PENDING;
 | |
|       } else if (aIdealKey.Size() != bestMatch->GetSurfaceKey().Size()) {
 | |
|         // The best factor of 2 match is still decoding, but the best we've got.
 | |
|         MOZ_ASSERT(suggestedSize != aIdealKey.Size());
 | |
|         MOZ_ASSERT(mFactor2Mode || mIsVectorImage);
 | |
|         matchType = MatchType::SUBSTITUTE_BECAUSE_BEST;
 | |
|       } else {
 | |
|         // The exact match is still decoding, but it's the best we've got.
 | |
|         matchType = MatchType::EXACT;
 | |
|       }
 | |
|     } else {
 | |
|       if (exactMatch) {
 | |
|         // We found an "exact match"; it must have been a placeholder.
 | |
|         MOZ_ASSERT(exactMatch->IsPlaceholder());
 | |
|         matchType = MatchType::PENDING;
 | |
|       } else {
 | |
|         // We couldn't find an exact match *or* a substitute.
 | |
|         matchType = MatchType::NOT_FOUND;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     return MakeTuple(bestMatch.forget(), matchType, suggestedSize);
 | |
|   }
 | |
| 
 | |
|   void MaybeSetFactor2Mode() {
 | |
|     MOZ_ASSERT(!mFactor2Mode);
 | |
| 
 | |
|     // Typically an image cache will not have too many size-varying surfaces, so
 | |
|     // if we exceed the given threshold, we should consider using a subset.
 | |
|     int32_t thresholdSurfaces =
 | |
|         StaticPrefs::image_cache_factor2_threshold_surfaces();
 | |
|     if (thresholdSurfaces < 0 ||
 | |
|         mSurfaces.Count() <= static_cast<uint32_t>(thresholdSurfaces)) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     // Determine how many native surfaces this image has. If it is zero, and it
 | |
|     // is a vector image, then we should impute a single native size. Otherwise,
 | |
|     // it may be zero because we don't know yet, or the image has an error, or
 | |
|     // it isn't supported.
 | |
|     NotNull<CachedSurface*> current =
 | |
|         WrapNotNull(mSurfaces.ConstIter().UserData());
 | |
|     Image* image = static_cast<Image*>(current->GetImageKey());
 | |
|     size_t nativeSizes = image->GetNativeSizesLength();
 | |
|     if (mIsVectorImage) {
 | |
|       MOZ_ASSERT(nativeSizes == 0);
 | |
|       nativeSizes = 1;
 | |
|     } else if (nativeSizes == 0) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     // Increase the threshold by the number of native sizes. This ensures that
 | |
|     // we do not prevent decoding of the image at all its native sizes. It does
 | |
|     // not guarantee we will provide a surface at that size however (i.e. many
 | |
|     // other sized surfaces are requested, in addition to the native sizes).
 | |
|     thresholdSurfaces += nativeSizes;
 | |
|     if (mSurfaces.Count() <= static_cast<uint32_t>(thresholdSurfaces)) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     // We have a valid size, we can change modes.
 | |
|     mFactor2Mode = true;
 | |
|   }
 | |
| 
 | |
|   template <typename Function>
 | |
|   void Prune(Function&& aRemoveCallback) {
 | |
|     if (!mFactor2Mode || mFactor2Pruned) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     // Attempt to discard any surfaces which are not factor of 2 and the best
 | |
|     // factor of 2 match exists.
 | |
|     bool hasNotFactorSize = false;
 | |
|     for (auto iter = mSurfaces.Iter(); !iter.Done(); iter.Next()) {
 | |
|       NotNull<CachedSurface*> current = WrapNotNull(iter.UserData());
 | |
|       const SurfaceKey& currentKey = current->GetSurfaceKey();
 | |
|       const IntSize& currentSize = currentKey.Size();
 | |
| 
 | |
|       // First we check if someone requested this size and would not accept
 | |
|       // an alternatively sized surface.
 | |
|       if (current->CannotSubstitute()) {
 | |
|         continue;
 | |
|       }
 | |
| 
 | |
|       // Next we find the best factor of 2 size for this surface. If this
 | |
|       // surface is a factor of 2 size, then we want to keep it.
 | |
|       IntSize bestSize = SuggestedSize(currentSize);
 | |
|       if (bestSize == currentSize) {
 | |
|         continue;
 | |
|       }
 | |
| 
 | |
|       // Check the cache for a surface with the same parameters except for the
 | |
|       // size which uses the closest factor of 2 size.
 | |
|       SurfaceKey compactKey = currentKey.CloneWithSize(bestSize);
 | |
|       RefPtr<CachedSurface> compactMatch;
 | |
|       mSurfaces.Get(compactKey, getter_AddRefs(compactMatch));
 | |
|       if (compactMatch && compactMatch->IsDecoded()) {
 | |
|         aRemoveCallback(current);
 | |
|         iter.Remove();
 | |
|       } else {
 | |
|         hasNotFactorSize = true;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     // We have no surfaces that are not factor of 2 sized, so we can stop
 | |
|     // pruning henceforth, because we avoid the insertion of new surfaces that
 | |
|     // don't match our sizing set (unless the caller won't accept a
 | |
|     // substitution.)
 | |
|     if (!hasNotFactorSize) {
 | |
|       mFactor2Pruned = true;
 | |
|     }
 | |
| 
 | |
|     // We should never leave factor of 2 mode due to pruning in of itself, but
 | |
|     // if we discarded surfaces due to the volatile buffers getting released,
 | |
|     // it is possible.
 | |
|     AfterMaybeRemove();
 | |
|   }
 | |
| 
 | |
|   template <typename Function>
 | |
|   bool Invalidate(Function&& aRemoveCallback) {
 | |
|     // Remove all non-blob recordings from the cache. Invalidate any blob
 | |
|     // recordings.
 | |
|     bool foundRecording = false;
 | |
|     for (auto iter = mSurfaces.Iter(); !iter.Done(); iter.Next()) {
 | |
|       NotNull<CachedSurface*> current = WrapNotNull(iter.UserData());
 | |
| 
 | |
|       if (current->GetSurfaceKey().Flags() & SurfaceFlags::RECORD_BLOB) {
 | |
|         foundRecording = true;
 | |
|         current->InvalidateRecording();
 | |
|         continue;
 | |
|       }
 | |
| 
 | |
|       aRemoveCallback(current);
 | |
|       iter.Remove();
 | |
|     }
 | |
| 
 | |
|     AfterMaybeRemove();
 | |
|     return foundRecording;
 | |
|   }
 | |
| 
 | |
|   IntSize SuggestedSize(const IntSize& aSize) const {
 | |
|     IntSize suggestedSize = SuggestedSizeInternal(aSize);
 | |
|     if (mIsVectorImage) {
 | |
|       suggestedSize = SurfaceCache::ClampVectorSize(suggestedSize);
 | |
|     }
 | |
|     return suggestedSize;
 | |
|   }
 | |
| 
 | |
|   IntSize SuggestedSizeInternal(const IntSize& aSize) const {
 | |
|     // When not in factor of 2 mode, we can always decode at the given size.
 | |
|     if (!mFactor2Mode) {
 | |
|       return aSize;
 | |
|     }
 | |
| 
 | |
|     // We cannot enter factor of 2 mode unless we have a minimum number of
 | |
|     // surfaces, and we should have left it if the cache was emptied.
 | |
|     if (MOZ_UNLIKELY(IsEmpty())) {
 | |
|       MOZ_ASSERT_UNREACHABLE("Should not be empty and in factor of 2 mode!");
 | |
|       return aSize;
 | |
|     }
 | |
| 
 | |
|     // This bit of awkwardness gets the largest native size of the image.
 | |
|     NotNull<CachedSurface*> firstSurface =
 | |
|         WrapNotNull(mSurfaces.ConstIter().UserData());
 | |
|     Image* image = static_cast<Image*>(firstSurface->GetImageKey());
 | |
|     IntSize factorSize;
 | |
|     if (NS_FAILED(image->GetWidth(&factorSize.width)) ||
 | |
|         NS_FAILED(image->GetHeight(&factorSize.height)) ||
 | |
|         factorSize.IsEmpty()) {
 | |
|       // Valid vector images may have a default size of 0x0. In that case, just
 | |
|       // assume a default size of 100x100 and apply the intrinsic ratio if
 | |
|       // available. If our guess was too small, don't use factor-of-scaling.
 | |
|       MOZ_ASSERT(mIsVectorImage);
 | |
|       factorSize = IntSize(100, 100);
 | |
|       Maybe<AspectRatio> aspectRatio = image->GetIntrinsicRatio();
 | |
|       if (aspectRatio && *aspectRatio) {
 | |
|         factorSize.width =
 | |
|             NSToIntRound(aspectRatio->ApplyToFloat(float(factorSize.height)));
 | |
|         if (factorSize.IsEmpty()) {
 | |
|           return aSize;
 | |
|         }
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     if (mIsVectorImage) {
 | |
|       // Ensure the aspect ratio matches the native size before forcing the
 | |
|       // caller to accept a factor of 2 size. The difference between the aspect
 | |
|       // ratios is:
 | |
|       //
 | |
|       //     delta = nativeWidth/nativeHeight - desiredWidth/desiredHeight
 | |
|       //
 | |
|       //     delta*nativeHeight*desiredHeight = nativeWidth*desiredHeight
 | |
|       //                                      - desiredWidth*nativeHeight
 | |
|       //
 | |
|       // Using the maximum accepted delta as a constant, we can avoid the
 | |
|       // floating point division and just compare after some integer ops.
 | |
|       int32_t delta =
 | |
|           factorSize.width * aSize.height - aSize.width * factorSize.height;
 | |
|       int32_t maxDelta = (factorSize.height * aSize.height) >> 4;
 | |
|       if (delta > maxDelta || delta < -maxDelta) {
 | |
|         return aSize;
 | |
|       }
 | |
| 
 | |
|       // If the requested size is bigger than the native size, we actually need
 | |
|       // to grow the native size instead of shrinking it.
 | |
|       if (factorSize.width < aSize.width) {
 | |
|         do {
 | |
|           IntSize candidate(factorSize.width * 2, factorSize.height * 2);
 | |
|           if (!SurfaceCache::IsLegalSize(candidate)) {
 | |
|             break;
 | |
|           }
 | |
| 
 | |
|           factorSize = candidate;
 | |
|         } while (factorSize.width < aSize.width);
 | |
| 
 | |
|         return factorSize;
 | |
|       }
 | |
| 
 | |
|       // Otherwise we can find the best fit as normal.
 | |
|     }
 | |
| 
 | |
|     // Start with the native size as the best first guess.
 | |
|     IntSize bestSize = factorSize;
 | |
|     factorSize.width /= 2;
 | |
|     factorSize.height /= 2;
 | |
| 
 | |
|     while (!factorSize.IsEmpty()) {
 | |
|       if (!CompareArea(aSize, bestSize, factorSize)) {
 | |
|         // This size is not better than the last. Since we proceed from largest
 | |
|         // to smallest, we know that the next size will not be better if the
 | |
|         // previous size was rejected. Break early.
 | |
|         break;
 | |
|       }
 | |
| 
 | |
|       // The current factor of 2 size is better than the last selected size.
 | |
|       bestSize = factorSize;
 | |
|       factorSize.width /= 2;
 | |
|       factorSize.height /= 2;
 | |
|     }
 | |
| 
 | |
|     return bestSize;
 | |
|   }
 | |
| 
 | |
|   bool CompareArea(const IntSize& aIdealSize, const IntSize& aBestSize,
 | |
|                    const IntSize& aSize) const {
 | |
|     // Compare sizes. We use an area-based heuristic here instead of computing a
 | |
|     // truly optimal answer, since it seems very unlikely to make a difference
 | |
|     // for realistic sizes.
 | |
|     int64_t idealArea = AreaOfIntSize(aIdealSize);
 | |
|     int64_t currentArea = AreaOfIntSize(aSize);
 | |
|     int64_t bestMatchArea = AreaOfIntSize(aBestSize);
 | |
| 
 | |
|     // If the best match is smaller than the ideal size, prefer bigger sizes.
 | |
|     if (bestMatchArea < idealArea) {
 | |
|       if (currentArea > bestMatchArea) {
 | |
|         return true;
 | |
|       }
 | |
|       return false;
 | |
|     }
 | |
| 
 | |
|     // Other, prefer sizes closer to the ideal size, but still not smaller.
 | |
|     if (idealArea <= currentArea && currentArea < bestMatchArea) {
 | |
|       return true;
 | |
|     }
 | |
| 
 | |
|     // This surface isn't an improvement over the current best match.
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   template <typename Function>
 | |
|   void CollectSizeOfSurfaces(nsTArray<SurfaceMemoryCounter>& aCounters,
 | |
|                              MallocSizeOf aMallocSizeOf,
 | |
|                              Function&& aRemoveCallback) {
 | |
|     CachedSurface::SurfaceMemoryReport report(aCounters, aMallocSizeOf);
 | |
|     for (auto iter = mSurfaces.Iter(); !iter.Done(); iter.Next()) {
 | |
|       NotNull<CachedSurface*> surface = WrapNotNull(iter.UserData());
 | |
| 
 | |
|       // We don't need the drawable surface for ourselves, but adding a surface
 | |
|       // to the report will trigger this indirectly. If the surface was
 | |
|       // discarded by the OS because it was in volatile memory, we should remove
 | |
|       // it from the cache immediately rather than include it in the report.
 | |
|       DrawableSurface drawableSurface;
 | |
|       if (!surface->IsPlaceholder()) {
 | |
|         drawableSurface = surface->GetDrawableSurface();
 | |
|         if (!drawableSurface) {
 | |
|           aRemoveCallback(surface);
 | |
|           iter.Remove();
 | |
|           continue;
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       const IntSize& size = surface->GetSurfaceKey().Size();
 | |
|       bool factor2Size = false;
 | |
|       if (mFactor2Mode) {
 | |
|         factor2Size = (size == SuggestedSize(size));
 | |
|       }
 | |
|       report.Add(surface, factor2Size);
 | |
|     }
 | |
| 
 | |
|     AfterMaybeRemove();
 | |
|   }
 | |
| 
 | |
|   void SetLocked(bool aLocked) { mLocked = aLocked; }
 | |
|   bool IsLocked() const { return mLocked; }
 | |
| 
 | |
|  private:
 | |
|   void AfterMaybeRemove() {
 | |
|     if (IsEmpty() && mFactor2Mode) {
 | |
|       // The last surface for this cache was removed. This can happen if the
 | |
|       // surface was stored in a volatile buffer and got purged, or the surface
 | |
|       // expired from the cache. If the cache itself lingers for some reason
 | |
|       // (e.g. in the process of performing a lookup, the cache itself is
 | |
|       // locked), then we need to reset the factor of 2 state because it
 | |
|       // requires at least one surface present to get the native size
 | |
|       // information from the image.
 | |
|       mFactor2Mode = mFactor2Pruned = false;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   SurfaceTable mSurfaces;
 | |
| 
 | |
|   bool mLocked;
 | |
| 
 | |
|   // True in "factor of 2" mode.
 | |
|   bool mFactor2Mode;
 | |
| 
 | |
|   // True if all non-factor of 2 surfaces have been removed from the cache. Note
 | |
|   // that this excludes unsubstitutable sizes.
 | |
|   bool mFactor2Pruned;
 | |
| 
 | |
|   // True if the surfaces are produced from a vector image. If so, it must match
 | |
|   // the aspect ratio when using factor of 2 mode.
 | |
|   bool mIsVectorImage;
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * SurfaceCacheImpl is responsible for determining which surfaces will be cached
 | |
|  * and managing the surface cache data structures. Rather than interact with
 | |
|  * SurfaceCacheImpl directly, client code interacts with SurfaceCache, which
 | |
|  * maintains high-level invariants and encapsulates the details of the surface
 | |
|  * cache's implementation.
 | |
|  */
 | |
| class SurfaceCacheImpl final : public nsIMemoryReporter {
 | |
|  public:
 | |
|   NS_DECL_ISUPPORTS
 | |
| 
 | |
|   SurfaceCacheImpl(uint32_t aSurfaceCacheExpirationTimeMS,
 | |
|                    uint32_t aSurfaceCacheDiscardFactor,
 | |
|                    uint32_t aSurfaceCacheSize)
 | |
|       : mExpirationTracker(aSurfaceCacheExpirationTimeMS),
 | |
|         mMemoryPressureObserver(new MemoryPressureObserver),
 | |
|         mDiscardFactor(aSurfaceCacheDiscardFactor),
 | |
|         mMaxCost(aSurfaceCacheSize),
 | |
|         mAvailableCost(aSurfaceCacheSize),
 | |
|         mLockedCost(0),
 | |
|         mOverflowCount(0),
 | |
|         mAlreadyPresentCount(0),
 | |
|         mTableFailureCount(0),
 | |
|         mTrackingFailureCount(0) {
 | |
|     nsCOMPtr<nsIObserverService> os = services::GetObserverService();
 | |
|     if (os) {
 | |
|       os->AddObserver(mMemoryPressureObserver, "memory-pressure", false);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|  private:
 | |
|   virtual ~SurfaceCacheImpl() {
 | |
|     nsCOMPtr<nsIObserverService> os = services::GetObserverService();
 | |
|     if (os) {
 | |
|       os->RemoveObserver(mMemoryPressureObserver, "memory-pressure");
 | |
|     }
 | |
| 
 | |
|     UnregisterWeakMemoryReporter(this);
 | |
|   }
 | |
| 
 | |
|  public:
 | |
|   void InitMemoryReporter() { RegisterWeakMemoryReporter(this); }
 | |
| 
 | |
|   InsertOutcome Insert(NotNull<ISurfaceProvider*> aProvider, bool aSetAvailable,
 | |
|                        const StaticMutexAutoLock& aAutoLock) {
 | |
|     // If this is a duplicate surface, refuse to replace the original.
 | |
|     // XXX(seth): Calling Lookup() and then RemoveEntry() does the lookup
 | |
|     // twice. We'll make this more efficient in bug 1185137.
 | |
|     LookupResult result =
 | |
|         Lookup(aProvider->GetImageKey(), aProvider->GetSurfaceKey(), aAutoLock,
 | |
|                /* aMarkUsed = */ false);
 | |
|     if (MOZ_UNLIKELY(result)) {
 | |
|       mAlreadyPresentCount++;
 | |
|       return InsertOutcome::FAILURE_ALREADY_PRESENT;
 | |
|     }
 | |
| 
 | |
|     if (result.Type() == MatchType::PENDING) {
 | |
|       RemoveEntry(aProvider->GetImageKey(), aProvider->GetSurfaceKey(),
 | |
|                   aAutoLock);
 | |
|     }
 | |
| 
 | |
|     MOZ_ASSERT(result.Type() == MatchType::NOT_FOUND ||
 | |
|                    result.Type() == MatchType::PENDING,
 | |
|                "A LookupResult with no surface should be NOT_FOUND or PENDING");
 | |
| 
 | |
|     // If this is bigger than we can hold after discarding everything we can,
 | |
|     // refuse to cache it.
 | |
|     Cost cost = aProvider->LogicalSizeInBytes();
 | |
|     if (MOZ_UNLIKELY(!CanHoldAfterDiscarding(cost))) {
 | |
|       mOverflowCount++;
 | |
|       return InsertOutcome::FAILURE;
 | |
|     }
 | |
| 
 | |
|     // Remove elements in order of cost until we can fit this in the cache. Note
 | |
|     // that locked surfaces aren't in mCosts, so we never remove them here.
 | |
|     while (cost > mAvailableCost) {
 | |
|       MOZ_ASSERT(!mCosts.IsEmpty(),
 | |
|                  "Removed everything and it still won't fit");
 | |
|       Remove(mCosts.LastElement().Surface(), /* aStopTracking */ true,
 | |
|              aAutoLock);
 | |
|     }
 | |
| 
 | |
|     // Locate the appropriate per-image cache. If there's not an existing cache
 | |
|     // for this image, create it.
 | |
|     const ImageKey imageKey = aProvider->GetImageKey();
 | |
|     RefPtr<ImageSurfaceCache> cache = GetImageCache(imageKey);
 | |
|     if (!cache) {
 | |
|       cache = new ImageSurfaceCache(imageKey);
 | |
|       if (!mImageCaches.InsertOrUpdate(aProvider->GetImageKey(), RefPtr{cache},
 | |
|                                        fallible)) {
 | |
|         mTableFailureCount++;
 | |
|         return InsertOutcome::FAILURE;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     // If we were asked to mark the cache entry available, do so.
 | |
|     if (aSetAvailable) {
 | |
|       aProvider->Availability().SetAvailable();
 | |
|     }
 | |
| 
 | |
|     auto surface = MakeNotNull<RefPtr<CachedSurface>>(aProvider);
 | |
| 
 | |
|     // We require that locking succeed if the image is locked and we're not
 | |
|     // inserting a placeholder; the caller may need to know this to handle
 | |
|     // errors correctly.
 | |
|     bool mustLock = cache->IsLocked() && !surface->IsPlaceholder();
 | |
|     if (mustLock) {
 | |
|       surface->SetLocked(true);
 | |
|       if (!surface->IsLocked()) {
 | |
|         return InsertOutcome::FAILURE;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     // Insert.
 | |
|     MOZ_ASSERT(cost <= mAvailableCost, "Inserting despite too large a cost");
 | |
|     if (!cache->Insert(surface)) {
 | |
|       mTableFailureCount++;
 | |
|       if (mustLock) {
 | |
|         surface->SetLocked(false);
 | |
|       }
 | |
|       return InsertOutcome::FAILURE;
 | |
|     }
 | |
| 
 | |
|     if (MOZ_UNLIKELY(!StartTracking(surface, aAutoLock))) {
 | |
|       MOZ_ASSERT(!mustLock);
 | |
|       Remove(surface, /* aStopTracking */ false, aAutoLock);
 | |
|       return InsertOutcome::FAILURE;
 | |
|     }
 | |
| 
 | |
|     return InsertOutcome::SUCCESS;
 | |
|   }
 | |
| 
 | |
|   void Remove(NotNull<CachedSurface*> aSurface, bool aStopTracking,
 | |
|               const StaticMutexAutoLock& aAutoLock) {
 | |
|     ImageKey imageKey = aSurface->GetImageKey();
 | |
| 
 | |
|     RefPtr<ImageSurfaceCache> cache = GetImageCache(imageKey);
 | |
|     MOZ_ASSERT(cache, "Shouldn't try to remove a surface with no image cache");
 | |
| 
 | |
|     // If the surface was not a placeholder, tell its image that we discarded
 | |
|     // it.
 | |
|     if (!aSurface->IsPlaceholder()) {
 | |
|       static_cast<Image*>(imageKey)->OnSurfaceDiscarded(
 | |
|           aSurface->GetSurfaceKey());
 | |
|     }
 | |
| 
 | |
|     // If we failed during StartTracking, we can skip this step.
 | |
|     if (aStopTracking) {
 | |
|       StopTracking(aSurface, /* aIsTracked */ true, aAutoLock);
 | |
|     }
 | |
| 
 | |
|     // Individual surfaces must be freed outside the lock.
 | |
|     mCachedSurfacesDiscard.AppendElement(cache->Remove(aSurface));
 | |
| 
 | |
|     MaybeRemoveEmptyCache(imageKey, cache);
 | |
|   }
 | |
| 
 | |
|   bool StartTracking(NotNull<CachedSurface*> aSurface,
 | |
|                      const StaticMutexAutoLock& aAutoLock) {
 | |
|     CostEntry costEntry = aSurface->GetCostEntry();
 | |
|     MOZ_ASSERT(costEntry.GetCost() <= mAvailableCost,
 | |
|                "Cost too large and the caller didn't catch it");
 | |
| 
 | |
|     if (aSurface->IsLocked()) {
 | |
|       mLockedCost += costEntry.GetCost();
 | |
|       MOZ_ASSERT(mLockedCost <= mMaxCost, "Locked more than we can hold?");
 | |
|     } else {
 | |
|       if (NS_WARN_IF(!mCosts.InsertElementSorted(costEntry, fallible))) {
 | |
|         mTrackingFailureCount++;
 | |
|         return false;
 | |
|       }
 | |
| 
 | |
|       // This may fail during XPCOM shutdown, so we need to ensure the object is
 | |
|       // tracked before calling RemoveObject in StopTracking.
 | |
|       nsresult rv = mExpirationTracker.AddObjectLocked(aSurface, aAutoLock);
 | |
|       if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|         DebugOnly<bool> foundInCosts = mCosts.RemoveElementSorted(costEntry);
 | |
|         MOZ_ASSERT(foundInCosts, "Lost track of costs for this surface");
 | |
|         mTrackingFailureCount++;
 | |
|         return false;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     mAvailableCost -= costEntry.GetCost();
 | |
|     return true;
 | |
|   }
 | |
| 
 | |
|   void StopTracking(NotNull<CachedSurface*> aSurface, bool aIsTracked,
 | |
|                     const StaticMutexAutoLock& aAutoLock) {
 | |
|     CostEntry costEntry = aSurface->GetCostEntry();
 | |
| 
 | |
|     if (aSurface->IsLocked()) {
 | |
|       MOZ_ASSERT(mLockedCost >= costEntry.GetCost(), "Costs don't balance");
 | |
|       mLockedCost -= costEntry.GetCost();
 | |
|       // XXX(seth): It'd be nice to use an O(log n) lookup here. This is O(n).
 | |
|       MOZ_ASSERT(!mCosts.Contains(costEntry),
 | |
|                  "Shouldn't have a cost entry for a locked surface");
 | |
|     } else {
 | |
|       if (MOZ_LIKELY(aSurface->GetExpirationState()->IsTracked())) {
 | |
|         MOZ_ASSERT(aIsTracked, "Expiration-tracking a surface unexpectedly!");
 | |
|         mExpirationTracker.RemoveObjectLocked(aSurface, aAutoLock);
 | |
|       } else {
 | |
|         // Our call to AddObject must have failed in StartTracking; most likely
 | |
|         // we're in XPCOM shutdown right now.
 | |
|         MOZ_ASSERT(!aIsTracked, "Not expiration-tracking an unlocked surface!");
 | |
|       }
 | |
| 
 | |
|       DebugOnly<bool> foundInCosts = mCosts.RemoveElementSorted(costEntry);
 | |
|       MOZ_ASSERT(foundInCosts, "Lost track of costs for this surface");
 | |
|     }
 | |
| 
 | |
|     mAvailableCost += costEntry.GetCost();
 | |
|     MOZ_ASSERT(mAvailableCost <= mMaxCost,
 | |
|                "More available cost than we started with");
 | |
|   }
 | |
| 
 | |
|   LookupResult Lookup(const ImageKey aImageKey, const SurfaceKey& aSurfaceKey,
 | |
|                       const StaticMutexAutoLock& aAutoLock, bool aMarkUsed) {
 | |
|     RefPtr<ImageSurfaceCache> cache = GetImageCache(aImageKey);
 | |
|     if (!cache) {
 | |
|       // No cached surfaces for this image.
 | |
|       return LookupResult(MatchType::NOT_FOUND);
 | |
|     }
 | |
| 
 | |
|     RefPtr<CachedSurface> surface = cache->Lookup(aSurfaceKey, aMarkUsed);
 | |
|     if (!surface) {
 | |
|       // Lookup in the per-image cache missed.
 | |
|       return LookupResult(MatchType::NOT_FOUND);
 | |
|     }
 | |
| 
 | |
|     if (surface->IsPlaceholder()) {
 | |
|       return LookupResult(MatchType::PENDING);
 | |
|     }
 | |
| 
 | |
|     DrawableSurface drawableSurface = surface->GetDrawableSurface();
 | |
|     if (!drawableSurface) {
 | |
|       // The surface was released by the operating system. Remove the cache
 | |
|       // entry as well.
 | |
|       Remove(WrapNotNull(surface), /* aStopTracking */ true, aAutoLock);
 | |
|       return LookupResult(MatchType::NOT_FOUND);
 | |
|     }
 | |
| 
 | |
|     if (aMarkUsed &&
 | |
|         !MarkUsed(WrapNotNull(surface), WrapNotNull(cache), aAutoLock)) {
 | |
|       Remove(WrapNotNull(surface), /* aStopTracking */ false, aAutoLock);
 | |
|       return LookupResult(MatchType::NOT_FOUND);
 | |
|     }
 | |
| 
 | |
|     MOZ_ASSERT(surface->GetSurfaceKey() == aSurfaceKey,
 | |
|                "Lookup() not returning an exact match?");
 | |
|     return LookupResult(std::move(drawableSurface), MatchType::EXACT);
 | |
|   }
 | |
| 
 | |
|   LookupResult LookupBestMatch(const ImageKey aImageKey,
 | |
|                                const SurfaceKey& aSurfaceKey,
 | |
|                                const StaticMutexAutoLock& aAutoLock,
 | |
|                                bool aMarkUsed) {
 | |
|     RefPtr<ImageSurfaceCache> cache = GetImageCache(aImageKey);
 | |
|     if (!cache) {
 | |
|       // No cached surfaces for this image.
 | |
|       return LookupResult(
 | |
|           MatchType::NOT_FOUND,
 | |
|           SurfaceCache::ClampSize(aImageKey, aSurfaceKey.Size()));
 | |
|     }
 | |
| 
 | |
|     // Repeatedly look up the best match, trying again if the resulting surface
 | |
|     // has been freed by the operating system, until we can either lock a
 | |
|     // surface for drawing or there are no matching surfaces left.
 | |
|     // XXX(seth): This is O(N^2), but N is expected to be very small. If we
 | |
|     // encounter a performance problem here we can revisit this.
 | |
| 
 | |
|     RefPtr<CachedSurface> surface;
 | |
|     DrawableSurface drawableSurface;
 | |
|     MatchType matchType = MatchType::NOT_FOUND;
 | |
|     IntSize suggestedSize;
 | |
|     while (true) {
 | |
|       Tie(surface, matchType, suggestedSize) =
 | |
|           cache->LookupBestMatch(aSurfaceKey);
 | |
| 
 | |
|       if (!surface) {
 | |
|         return LookupResult(
 | |
|             matchType, suggestedSize);  // Lookup in the per-image cache missed.
 | |
|       }
 | |
| 
 | |
|       drawableSurface = surface->GetDrawableSurface();
 | |
|       if (drawableSurface) {
 | |
|         break;
 | |
|       }
 | |
| 
 | |
|       // The surface was released by the operating system. Remove the cache
 | |
|       // entry as well.
 | |
|       Remove(WrapNotNull(surface), /* aStopTracking */ true, aAutoLock);
 | |
|     }
 | |
| 
 | |
|     MOZ_ASSERT_IF(matchType == MatchType::EXACT,
 | |
|                   surface->GetSurfaceKey() == aSurfaceKey);
 | |
|     MOZ_ASSERT_IF(
 | |
|         matchType == MatchType::SUBSTITUTE_BECAUSE_NOT_FOUND ||
 | |
|             matchType == MatchType::SUBSTITUTE_BECAUSE_PENDING,
 | |
|         surface->GetSurfaceKey().Region() == aSurfaceKey.Region() &&
 | |
|             surface->GetSurfaceKey().SVGContext() == aSurfaceKey.SVGContext() &&
 | |
|             surface->GetSurfaceKey().Playback() == aSurfaceKey.Playback() &&
 | |
|             surface->GetSurfaceKey().Flags() == aSurfaceKey.Flags());
 | |
| 
 | |
|     if (matchType == MatchType::EXACT ||
 | |
|         matchType == MatchType::SUBSTITUTE_BECAUSE_BEST) {
 | |
|       if (aMarkUsed &&
 | |
|           !MarkUsed(WrapNotNull(surface), WrapNotNull(cache), aAutoLock)) {
 | |
|         Remove(WrapNotNull(surface), /* aStopTracking */ false, aAutoLock);
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     return LookupResult(std::move(drawableSurface), matchType, suggestedSize);
 | |
|   }
 | |
| 
 | |
|   bool CanHold(const Cost aCost) const { return aCost <= mMaxCost; }
 | |
| 
 | |
|   size_t MaximumCapacity() const { return size_t(mMaxCost); }
 | |
| 
 | |
|   void SurfaceAvailable(NotNull<ISurfaceProvider*> aProvider,
 | |
|                         const StaticMutexAutoLock& aAutoLock) {
 | |
|     if (!aProvider->Availability().IsPlaceholder()) {
 | |
|       MOZ_ASSERT_UNREACHABLE("Calling SurfaceAvailable on non-placeholder");
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     // Reinsert the provider, requesting that Insert() mark it available. This
 | |
|     // may or may not succeed, depending on whether some other decoder has
 | |
|     // beaten us to the punch and inserted a non-placeholder version of this
 | |
|     // surface first, but it's fine either way.
 | |
|     // XXX(seth): This could be implemented more efficiently; we should be able
 | |
|     // to just update our data structures without reinserting.
 | |
|     Insert(aProvider, /* aSetAvailable = */ true, aAutoLock);
 | |
|   }
 | |
| 
 | |
|   void LockImage(const ImageKey aImageKey) {
 | |
|     RefPtr<ImageSurfaceCache> cache = GetImageCache(aImageKey);
 | |
|     if (!cache) {
 | |
|       cache = new ImageSurfaceCache(aImageKey);
 | |
|       mImageCaches.InsertOrUpdate(aImageKey, RefPtr{cache});
 | |
|     }
 | |
| 
 | |
|     cache->SetLocked(true);
 | |
| 
 | |
|     // We don't relock this image's existing surfaces right away; instead, the
 | |
|     // image should arrange for Lookup() to touch them if they are still useful.
 | |
|   }
 | |
| 
 | |
|   void UnlockImage(const ImageKey aImageKey,
 | |
|                    const StaticMutexAutoLock& aAutoLock) {
 | |
|     RefPtr<ImageSurfaceCache> cache = GetImageCache(aImageKey);
 | |
|     if (!cache || !cache->IsLocked()) {
 | |
|       return;  // Already unlocked.
 | |
|     }
 | |
| 
 | |
|     cache->SetLocked(false);
 | |
|     DoUnlockSurfaces(WrapNotNull(cache), /* aStaticOnly = */ false, aAutoLock);
 | |
|   }
 | |
| 
 | |
|   void UnlockEntries(const ImageKey aImageKey,
 | |
|                      const StaticMutexAutoLock& aAutoLock) {
 | |
|     RefPtr<ImageSurfaceCache> cache = GetImageCache(aImageKey);
 | |
|     if (!cache || !cache->IsLocked()) {
 | |
|       return;  // Already unlocked.
 | |
|     }
 | |
| 
 | |
|     // (Note that we *don't* unlock the per-image cache here; that's the
 | |
|     // difference between this and UnlockImage.)
 | |
|     DoUnlockSurfaces(WrapNotNull(cache),
 | |
|                      /* aStaticOnly = */
 | |
|                      !StaticPrefs::image_mem_animated_discardable_AtStartup(),
 | |
|                      aAutoLock);
 | |
|   }
 | |
| 
 | |
|   already_AddRefed<ImageSurfaceCache> RemoveImage(
 | |
|       const ImageKey aImageKey, const StaticMutexAutoLock& aAutoLock) {
 | |
|     RefPtr<ImageSurfaceCache> cache = GetImageCache(aImageKey);
 | |
|     if (!cache) {
 | |
|       return nullptr;  // No cached surfaces for this image, so nothing to do.
 | |
|     }
 | |
| 
 | |
|     // Discard all of the cached surfaces for this image.
 | |
|     // XXX(seth): This is O(n^2) since for each item in the cache we are
 | |
|     // removing an element from the costs array. Since n is expected to be
 | |
|     // small, performance should be good, but if usage patterns change we should
 | |
|     // change the data structure used for mCosts.
 | |
|     for (const auto& value : cache->Values()) {
 | |
|       StopTracking(WrapNotNull(value),
 | |
|                    /* aIsTracked */ true, aAutoLock);
 | |
|     }
 | |
| 
 | |
|     // The per-image cache isn't needed anymore, so remove it as well.
 | |
|     // This implicitly unlocks the image if it was locked.
 | |
|     mImageCaches.Remove(aImageKey);
 | |
| 
 | |
|     // Since we did not actually remove any of the surfaces from the cache
 | |
|     // itself, only stopped tracking them, we should free it outside the lock.
 | |
|     return cache.forget();
 | |
|   }
 | |
| 
 | |
|   void PruneImage(const ImageKey aImageKey,
 | |
|                   const StaticMutexAutoLock& aAutoLock) {
 | |
|     RefPtr<ImageSurfaceCache> cache = GetImageCache(aImageKey);
 | |
|     if (!cache) {
 | |
|       return;  // No cached surfaces for this image, so nothing to do.
 | |
|     }
 | |
| 
 | |
|     cache->Prune([this, &aAutoLock](NotNull<CachedSurface*> aSurface) -> void {
 | |
|       StopTracking(aSurface, /* aIsTracked */ true, aAutoLock);
 | |
|       // Individual surfaces must be freed outside the lock.
 | |
|       mCachedSurfacesDiscard.AppendElement(aSurface);
 | |
|     });
 | |
| 
 | |
|     MaybeRemoveEmptyCache(aImageKey, cache);
 | |
|   }
 | |
| 
 | |
|   bool InvalidateImage(const ImageKey aImageKey,
 | |
|                        const StaticMutexAutoLock& aAutoLock) {
 | |
|     RefPtr<ImageSurfaceCache> cache = GetImageCache(aImageKey);
 | |
|     if (!cache) {
 | |
|       return false;  // No cached surfaces for this image, so nothing to do.
 | |
|     }
 | |
| 
 | |
|     bool rv = cache->Invalidate(
 | |
|         [this, &aAutoLock](NotNull<CachedSurface*> aSurface) -> void {
 | |
|           StopTracking(aSurface, /* aIsTracked */ true, aAutoLock);
 | |
|           // Individual surfaces must be freed outside the lock.
 | |
|           mCachedSurfacesDiscard.AppendElement(aSurface);
 | |
|         });
 | |
| 
 | |
|     MaybeRemoveEmptyCache(aImageKey, cache);
 | |
|     return rv;
 | |
|   }
 | |
| 
 | |
|   void DiscardAll(const StaticMutexAutoLock& aAutoLock) {
 | |
|     // Remove in order of cost because mCosts is an array and the other data
 | |
|     // structures are all hash tables. Note that locked surfaces are not
 | |
|     // removed, since they aren't present in mCosts.
 | |
|     while (!mCosts.IsEmpty()) {
 | |
|       Remove(mCosts.LastElement().Surface(), /* aStopTracking */ true,
 | |
|              aAutoLock);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   void DiscardForMemoryPressure(const StaticMutexAutoLock& aAutoLock) {
 | |
|     // Compute our discardable cost. Since locked surfaces aren't discardable,
 | |
|     // we exclude them.
 | |
|     const Cost discardableCost = (mMaxCost - mAvailableCost) - mLockedCost;
 | |
|     MOZ_ASSERT(discardableCost <= mMaxCost, "Discardable cost doesn't add up");
 | |
| 
 | |
|     // Our target is to raise our available cost by (1 / mDiscardFactor) of our
 | |
|     // discardable cost - in other words, we want to end up with about
 | |
|     // (discardableCost / mDiscardFactor) fewer bytes stored in the surface
 | |
|     // cache after we're done.
 | |
|     const Cost targetCost = mAvailableCost + (discardableCost / mDiscardFactor);
 | |
| 
 | |
|     if (targetCost > mMaxCost - mLockedCost) {
 | |
|       MOZ_ASSERT_UNREACHABLE("Target cost is more than we can discard");
 | |
|       DiscardAll(aAutoLock);
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     // Discard surfaces until we've reduced our cost to our target cost.
 | |
|     while (mAvailableCost < targetCost) {
 | |
|       MOZ_ASSERT(!mCosts.IsEmpty(), "Removed everything and still not done");
 | |
|       Remove(mCosts.LastElement().Surface(), /* aStopTracking */ true,
 | |
|              aAutoLock);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   void TakeDiscard(nsTArray<RefPtr<CachedSurface>>& aDiscard,
 | |
|                    const StaticMutexAutoLock& aAutoLock) {
 | |
|     MOZ_ASSERT(aDiscard.IsEmpty());
 | |
|     aDiscard = std::move(mCachedSurfacesDiscard);
 | |
|   }
 | |
| 
 | |
|   already_AddRefed<CachedSurface> GetSurfaceForResetAnimation(
 | |
|       const ImageKey aImageKey, const SurfaceKey& aSurfaceKey,
 | |
|       const StaticMutexAutoLock& aAutoLock) {
 | |
|     RefPtr<CachedSurface> surface;
 | |
| 
 | |
|     RefPtr<ImageSurfaceCache> cache = GetImageCache(aImageKey);
 | |
|     if (!cache) {
 | |
|       // No cached surfaces for this image.
 | |
|       return surface.forget();
 | |
|     }
 | |
| 
 | |
|     surface = cache->Lookup(aSurfaceKey, /* aForAccess = */ false);
 | |
|     return surface.forget();
 | |
|   }
 | |
| 
 | |
|   void LockSurface(NotNull<CachedSurface*> aSurface,
 | |
|                    const StaticMutexAutoLock& aAutoLock) {
 | |
|     if (aSurface->IsPlaceholder() || aSurface->IsLocked()) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     StopTracking(aSurface, /* aIsTracked */ true, aAutoLock);
 | |
| 
 | |
|     // Lock the surface. This can fail.
 | |
|     aSurface->SetLocked(true);
 | |
|     DebugOnly<bool> tracked = StartTracking(aSurface, aAutoLock);
 | |
|     MOZ_ASSERT(tracked);
 | |
|   }
 | |
| 
 | |
|   size_t ShallowSizeOfIncludingThis(
 | |
|       MallocSizeOf aMallocSizeOf, const StaticMutexAutoLock& aAutoLock) const {
 | |
|     size_t bytes =
 | |
|         aMallocSizeOf(this) + mCosts.ShallowSizeOfExcludingThis(aMallocSizeOf) +
 | |
|         mImageCaches.ShallowSizeOfExcludingThis(aMallocSizeOf) +
 | |
|         mCachedSurfacesDiscard.ShallowSizeOfExcludingThis(aMallocSizeOf) +
 | |
|         mExpirationTracker.ShallowSizeOfExcludingThis(aMallocSizeOf);
 | |
|     for (const auto& data : mImageCaches.Values()) {
 | |
|       bytes += data->ShallowSizeOfIncludingThis(aMallocSizeOf);
 | |
|     }
 | |
|     return bytes;
 | |
|   }
 | |
| 
 | |
|   NS_IMETHOD
 | |
|   CollectReports(nsIHandleReportCallback* aHandleReport, nsISupports* aData,
 | |
|                  bool aAnonymize) override {
 | |
|     StaticMutexAutoLock lock(sInstanceMutex);
 | |
| 
 | |
|     uint32_t lockedImageCount = 0;
 | |
|     uint32_t totalSurfaceCount = 0;
 | |
|     uint32_t lockedSurfaceCount = 0;
 | |
|     for (const auto& cache : mImageCaches.Values()) {
 | |
|       totalSurfaceCount += cache->Count();
 | |
|       if (cache->IsLocked()) {
 | |
|         ++lockedImageCount;
 | |
|       }
 | |
|       for (const auto& value : cache->Values()) {
 | |
|         if (value->IsLocked()) {
 | |
|           ++lockedSurfaceCount;
 | |
|         }
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     // clang-format off
 | |
|     // We have explicit memory reporting for the surface cache which is more
 | |
|     // accurate than the cost metrics we report here, but these metrics are
 | |
|     // still useful to report, since they control the cache's behavior.
 | |
|     MOZ_COLLECT_REPORT(
 | |
|       "explicit/images/cache/overhead", KIND_HEAP, UNITS_BYTES,
 | |
|       ShallowSizeOfIncludingThis(SurfaceCacheMallocSizeOf, lock),
 | |
| "Memory used by the surface cache data structures, excluding surface data.");
 | |
| 
 | |
|     MOZ_COLLECT_REPORT(
 | |
|       "imagelib-surface-cache-estimated-total",
 | |
|       KIND_OTHER, UNITS_BYTES, (mMaxCost - mAvailableCost),
 | |
| "Estimated total memory used by the imagelib surface cache.");
 | |
| 
 | |
|     MOZ_COLLECT_REPORT(
 | |
|       "imagelib-surface-cache-estimated-locked",
 | |
|       KIND_OTHER, UNITS_BYTES, mLockedCost,
 | |
| "Estimated memory used by locked surfaces in the imagelib surface cache.");
 | |
| 
 | |
|     MOZ_COLLECT_REPORT(
 | |
|       "imagelib-surface-cache-tracked-cost-count",
 | |
|       KIND_OTHER, UNITS_COUNT, mCosts.Length(),
 | |
| "Total number of surfaces tracked for cost (and expiry) in the imagelib surface cache.");
 | |
| 
 | |
|     MOZ_COLLECT_REPORT(
 | |
|       "imagelib-surface-cache-tracked-expiry-count",
 | |
|       KIND_OTHER, UNITS_COUNT, mExpirationTracker.Length(lock),
 | |
| "Total number of surfaces tracked for expiry (and cost) in the imagelib surface cache.");
 | |
| 
 | |
|     MOZ_COLLECT_REPORT(
 | |
|       "imagelib-surface-cache-image-count",
 | |
|       KIND_OTHER, UNITS_COUNT, mImageCaches.Count(),
 | |
| "Total number of images in the imagelib surface cache.");
 | |
| 
 | |
|     MOZ_COLLECT_REPORT(
 | |
|       "imagelib-surface-cache-locked-image-count",
 | |
|       KIND_OTHER, UNITS_COUNT, lockedImageCount,
 | |
| "Total number of locked images in the imagelib surface cache.");
 | |
| 
 | |
|     MOZ_COLLECT_REPORT(
 | |
|       "imagelib-surface-cache-image-surface-count",
 | |
|       KIND_OTHER, UNITS_COUNT, totalSurfaceCount,
 | |
| "Total number of surfaces in the imagelib surface cache.");
 | |
| 
 | |
|     MOZ_COLLECT_REPORT(
 | |
|       "imagelib-surface-cache-locked-surfaces-count",
 | |
|       KIND_OTHER, UNITS_COUNT, lockedSurfaceCount,
 | |
| "Total number of locked surfaces in the imagelib surface cache.");
 | |
| 
 | |
|     MOZ_COLLECT_REPORT(
 | |
|       "imagelib-surface-cache-overflow-count",
 | |
|       KIND_OTHER, UNITS_COUNT, mOverflowCount,
 | |
| "Count of how many times the surface cache has hit its capacity and been "
 | |
| "unable to insert a new surface.");
 | |
| 
 | |
|     MOZ_COLLECT_REPORT(
 | |
|       "imagelib-surface-cache-tracking-failure-count",
 | |
|       KIND_OTHER, UNITS_COUNT, mTrackingFailureCount,
 | |
| "Count of how many times the surface cache has failed to begin tracking a "
 | |
| "given surface.");
 | |
| 
 | |
|     MOZ_COLLECT_REPORT(
 | |
|       "imagelib-surface-cache-already-present-count",
 | |
|       KIND_OTHER, UNITS_COUNT, mAlreadyPresentCount,
 | |
| "Count of how many times the surface cache has failed to insert a surface "
 | |
| "because it is already present.");
 | |
| 
 | |
|     MOZ_COLLECT_REPORT(
 | |
|       "imagelib-surface-cache-table-failure-count",
 | |
|       KIND_OTHER, UNITS_COUNT, mTableFailureCount,
 | |
| "Count of how many times the surface cache has failed to insert a surface "
 | |
| "because a hash table could not accept an entry.");
 | |
|     // clang-format on
 | |
| 
 | |
|     return NS_OK;
 | |
|   }
 | |
| 
 | |
|   void CollectSizeOfSurfaces(const ImageKey aImageKey,
 | |
|                              nsTArray<SurfaceMemoryCounter>& aCounters,
 | |
|                              MallocSizeOf aMallocSizeOf,
 | |
|                              const StaticMutexAutoLock& aAutoLock) {
 | |
|     RefPtr<ImageSurfaceCache> cache = GetImageCache(aImageKey);
 | |
|     if (!cache) {
 | |
|       return;  // No surfaces for this image.
 | |
|     }
 | |
| 
 | |
|     // Report all surfaces in the per-image cache.
 | |
|     cache->CollectSizeOfSurfaces(
 | |
|         aCounters, aMallocSizeOf,
 | |
|         [this, &aAutoLock](NotNull<CachedSurface*> aSurface) -> void {
 | |
|           StopTracking(aSurface, /* aIsTracked */ true, aAutoLock);
 | |
|           // Individual surfaces must be freed outside the lock.
 | |
|           mCachedSurfacesDiscard.AppendElement(aSurface);
 | |
|         });
 | |
| 
 | |
|     MaybeRemoveEmptyCache(aImageKey, cache);
 | |
|   }
 | |
| 
 | |
|   void ReleaseImageOnMainThread(already_AddRefed<image::Image>&& aImage,
 | |
|                                 const StaticMutexAutoLock& aAutoLock) {
 | |
|     RefPtr<image::Image> image = aImage;
 | |
|     if (!image) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     bool needsDispatch = mReleasingImagesOnMainThread.IsEmpty();
 | |
|     mReleasingImagesOnMainThread.AppendElement(image);
 | |
| 
 | |
|     if (!needsDispatch ||
 | |
|         AppShutdown::IsInOrBeyond(ShutdownPhase::XPCOMShutdownFinal)) {
 | |
|       // Either there is already a ongoing task for ClearReleasingImages() or
 | |
|       // it's too late in shutdown to dispatch.
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     NS_DispatchToMainThread(NS_NewRunnableFunction(
 | |
|         "SurfaceCacheImpl::ReleaseImageOnMainThread",
 | |
|         []() -> void { SurfaceCache::ClearReleasingImages(); }));
 | |
|   }
 | |
| 
 | |
|   void TakeReleasingImages(nsTArray<RefPtr<image::Image>>& aImage,
 | |
|                            const StaticMutexAutoLock& aAutoLock) {
 | |
|     MOZ_ASSERT(NS_IsMainThread());
 | |
|     aImage.SwapElements(mReleasingImagesOnMainThread);
 | |
|   }
 | |
| 
 | |
|  private:
 | |
|   already_AddRefed<ImageSurfaceCache> GetImageCache(const ImageKey aImageKey) {
 | |
|     RefPtr<ImageSurfaceCache> imageCache;
 | |
|     mImageCaches.Get(aImageKey, getter_AddRefs(imageCache));
 | |
|     return imageCache.forget();
 | |
|   }
 | |
| 
 | |
|   void MaybeRemoveEmptyCache(const ImageKey aImageKey,
 | |
|                              ImageSurfaceCache* aCache) {
 | |
|     // Remove the per-image cache if it's unneeded now. Keep it if the image is
 | |
|     // locked, since the per-image cache is where we store that state. Note that
 | |
|     // we don't push it into mImageCachesDiscard because all of its surfaces
 | |
|     // have been removed, so it is safe to free while holding the lock.
 | |
|     if (aCache->IsEmpty() && !aCache->IsLocked()) {
 | |
|       mImageCaches.Remove(aImageKey);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   // This is similar to CanHold() except that it takes into account the costs of
 | |
|   // locked surfaces. It's used internally in Insert(), but it's not exposed
 | |
|   // publicly because we permit multithreaded access to the surface cache, which
 | |
|   // means that the result would be meaningless: another thread could insert a
 | |
|   // surface or lock an image at any time.
 | |
|   bool CanHoldAfterDiscarding(const Cost aCost) const {
 | |
|     return aCost <= mMaxCost - mLockedCost;
 | |
|   }
 | |
| 
 | |
|   bool MarkUsed(NotNull<CachedSurface*> aSurface,
 | |
|                 NotNull<ImageSurfaceCache*> aCache,
 | |
|                 const StaticMutexAutoLock& aAutoLock) {
 | |
|     if (aCache->IsLocked()) {
 | |
|       LockSurface(aSurface, aAutoLock);
 | |
|       return true;
 | |
|     }
 | |
| 
 | |
|     nsresult rv = mExpirationTracker.MarkUsedLocked(aSurface, aAutoLock);
 | |
|     if (NS_WARN_IF(NS_FAILED(rv))) {
 | |
|       // If mark used fails, it is because it failed to reinsert the surface
 | |
|       // after removing it from the tracker. Thus we need to update our
 | |
|       // own accounting but otherwise expect it to be untracked.
 | |
|       StopTracking(aSurface, /* aIsTracked */ false, aAutoLock);
 | |
|       return false;
 | |
|     }
 | |
|     return true;
 | |
|   }
 | |
| 
 | |
|   void DoUnlockSurfaces(NotNull<ImageSurfaceCache*> aCache, bool aStaticOnly,
 | |
|                         const StaticMutexAutoLock& aAutoLock) {
 | |
|     AutoTArray<NotNull<CachedSurface*>, 8> discard;
 | |
| 
 | |
|     // Unlock all the surfaces the per-image cache is holding.
 | |
|     for (const auto& value : aCache->Values()) {
 | |
|       NotNull<CachedSurface*> surface = WrapNotNull(value);
 | |
|       if (surface->IsPlaceholder() || !surface->IsLocked()) {
 | |
|         continue;
 | |
|       }
 | |
|       if (aStaticOnly &&
 | |
|           surface->GetSurfaceKey().Playback() != PlaybackType::eStatic) {
 | |
|         continue;
 | |
|       }
 | |
|       StopTracking(surface, /* aIsTracked */ true, aAutoLock);
 | |
|       surface->SetLocked(false);
 | |
|       if (MOZ_UNLIKELY(!StartTracking(surface, aAutoLock))) {
 | |
|         discard.AppendElement(surface);
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     // Discard any that we failed to track.
 | |
|     for (auto iter = discard.begin(); iter != discard.end(); ++iter) {
 | |
|       Remove(*iter, /* aStopTracking */ false, aAutoLock);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   void RemoveEntry(const ImageKey aImageKey, const SurfaceKey& aSurfaceKey,
 | |
|                    const StaticMutexAutoLock& aAutoLock) {
 | |
|     RefPtr<ImageSurfaceCache> cache = GetImageCache(aImageKey);
 | |
|     if (!cache) {
 | |
|       return;  // No cached surfaces for this image.
 | |
|     }
 | |
| 
 | |
|     RefPtr<CachedSurface> surface =
 | |
|         cache->Lookup(aSurfaceKey, /* aForAccess = */ false);
 | |
|     if (!surface) {
 | |
|       return;  // Lookup in the per-image cache missed.
 | |
|     }
 | |
| 
 | |
|     Remove(WrapNotNull(surface), /* aStopTracking */ true, aAutoLock);
 | |
|   }
 | |
| 
 | |
|   class SurfaceTracker final
 | |
|       : public ExpirationTrackerImpl<CachedSurface, 2, StaticMutex,
 | |
|                                      StaticMutexAutoLock> {
 | |
|    public:
 | |
|     explicit SurfaceTracker(uint32_t aSurfaceCacheExpirationTimeMS)
 | |
|         : ExpirationTrackerImpl<CachedSurface, 2, StaticMutex,
 | |
|                                 StaticMutexAutoLock>(
 | |
|               aSurfaceCacheExpirationTimeMS, "SurfaceTracker") {}
 | |
| 
 | |
|    protected:
 | |
|     void NotifyExpiredLocked(CachedSurface* aSurface,
 | |
|                              const StaticMutexAutoLock& aAutoLock) override {
 | |
|       sInstance->Remove(WrapNotNull(aSurface), /* aStopTracking */ true,
 | |
|                         aAutoLock);
 | |
|     }
 | |
| 
 | |
|     void NotifyHandlerEndLocked(const StaticMutexAutoLock& aAutoLock) override {
 | |
|       sInstance->TakeDiscard(mDiscard, aAutoLock);
 | |
|     }
 | |
| 
 | |
|     void NotifyHandlerEnd() override {
 | |
|       nsTArray<RefPtr<CachedSurface>> discard(std::move(mDiscard));
 | |
|     }
 | |
| 
 | |
|     StaticMutex& GetMutex() override { return sInstanceMutex; }
 | |
| 
 | |
|     nsTArray<RefPtr<CachedSurface>> mDiscard;
 | |
|   };
 | |
| 
 | |
|   class MemoryPressureObserver final : public nsIObserver {
 | |
|    public:
 | |
|     NS_DECL_ISUPPORTS
 | |
| 
 | |
|     NS_IMETHOD Observe(nsISupports*, const char* aTopic,
 | |
|                        const char16_t*) override {
 | |
|       nsTArray<RefPtr<CachedSurface>> discard;
 | |
|       {
 | |
|         StaticMutexAutoLock lock(sInstanceMutex);
 | |
|         if (sInstance && strcmp(aTopic, "memory-pressure") == 0) {
 | |
|           sInstance->DiscardForMemoryPressure(lock);
 | |
|           sInstance->TakeDiscard(discard, lock);
 | |
|         }
 | |
|       }
 | |
|       return NS_OK;
 | |
|     }
 | |
| 
 | |
|    private:
 | |
|     virtual ~MemoryPressureObserver() {}
 | |
|   };
 | |
| 
 | |
|   nsTArray<CostEntry> mCosts;
 | |
|   nsRefPtrHashtable<nsPtrHashKey<Image>, ImageSurfaceCache> mImageCaches;
 | |
|   nsTArray<RefPtr<CachedSurface>> mCachedSurfacesDiscard;
 | |
|   SurfaceTracker mExpirationTracker;
 | |
|   RefPtr<MemoryPressureObserver> mMemoryPressureObserver;
 | |
|   nsTArray<RefPtr<image::Image>> mReleasingImagesOnMainThread;
 | |
|   const uint32_t mDiscardFactor;
 | |
|   const Cost mMaxCost;
 | |
|   Cost mAvailableCost;
 | |
|   Cost mLockedCost;
 | |
|   size_t mOverflowCount;
 | |
|   size_t mAlreadyPresentCount;
 | |
|   size_t mTableFailureCount;
 | |
|   size_t mTrackingFailureCount;
 | |
| };
 | |
| 
 | |
| NS_IMPL_ISUPPORTS(SurfaceCacheImpl, nsIMemoryReporter)
 | |
| NS_IMPL_ISUPPORTS(SurfaceCacheImpl::MemoryPressureObserver, nsIObserver)
 | |
| 
 | |
| ///////////////////////////////////////////////////////////////////////////////
 | |
| // Public API
 | |
| ///////////////////////////////////////////////////////////////////////////////
 | |
| 
 | |
| /* static */
 | |
| void SurfaceCache::Initialize() {
 | |
|   // Initialize preferences.
 | |
|   MOZ_ASSERT(NS_IsMainThread());
 | |
|   MOZ_ASSERT(!sInstance, "Shouldn't initialize more than once");
 | |
| 
 | |
|   // See StaticPrefs for the default values of these preferences.
 | |
| 
 | |
|   // Length of time before an unused surface is removed from the cache, in
 | |
|   // milliseconds.
 | |
|   uint32_t surfaceCacheExpirationTimeMS =
 | |
|       StaticPrefs::image_mem_surfacecache_min_expiration_ms_AtStartup();
 | |
| 
 | |
|   // What fraction of the memory used by the surface cache we should discard
 | |
|   // when we get a memory pressure notification. This value is interpreted as
 | |
|   // 1/N, so 1 means to discard everything, 2 means to discard about half of the
 | |
|   // memory we're using, and so forth. We clamp it to avoid division by zero.
 | |
|   uint32_t surfaceCacheDiscardFactor =
 | |
|       max(StaticPrefs::image_mem_surfacecache_discard_factor_AtStartup(), 1u);
 | |
| 
 | |
|   // Maximum size of the surface cache, in kilobytes.
 | |
|   uint64_t surfaceCacheMaxSizeKB =
 | |
|       StaticPrefs::image_mem_surfacecache_max_size_kb_AtStartup();
 | |
| 
 | |
|   if (sizeof(uintptr_t) <= 4) {
 | |
|     // Limit surface cache to 1 GB if our address space is 32 bit.
 | |
|     surfaceCacheMaxSizeKB = 1024 * 1024;
 | |
|   }
 | |
| 
 | |
|   // A knob determining the actual size of the surface cache. Currently the
 | |
|   // cache is (size of main memory) / (surface cache size factor) KB
 | |
|   // or (surface cache max size) KB, whichever is smaller. The formula
 | |
|   // may change in the future, though.
 | |
|   // For example, a value of 4 would yield a 256MB cache on a 1GB machine.
 | |
|   // The smallest machines we are likely to run this code on have 256MB
 | |
|   // of memory, which would yield a 64MB cache on this setting.
 | |
|   // We clamp this value to avoid division by zero.
 | |
|   uint32_t surfaceCacheSizeFactor =
 | |
|       max(StaticPrefs::image_mem_surfacecache_size_factor_AtStartup(), 1u);
 | |
| 
 | |
|   // Compute the size of the surface cache.
 | |
|   uint64_t memorySize = PR_GetPhysicalMemorySize();
 | |
|   if (memorySize == 0) {
 | |
| #if !defined(__DragonFly__)
 | |
|     MOZ_ASSERT_UNREACHABLE("PR_GetPhysicalMemorySize not implemented here");
 | |
| #endif
 | |
|     memorySize = 256 * 1024 * 1024;  // Fall back to 256MB.
 | |
|   }
 | |
|   uint64_t proposedSize = memorySize / surfaceCacheSizeFactor;
 | |
|   uint64_t surfaceCacheSizeBytes =
 | |
|       min(proposedSize, surfaceCacheMaxSizeKB * 1024);
 | |
|   uint32_t finalSurfaceCacheSizeBytes =
 | |
|       min(surfaceCacheSizeBytes, uint64_t(UINT32_MAX));
 | |
| 
 | |
|   // Create the surface cache singleton with the requested settings.  Note that
 | |
|   // the size is a limit that the cache may not grow beyond, but we do not
 | |
|   // actually allocate any storage for surfaces at this time.
 | |
|   sInstance = new SurfaceCacheImpl(surfaceCacheExpirationTimeMS,
 | |
|                                    surfaceCacheDiscardFactor,
 | |
|                                    finalSurfaceCacheSizeBytes);
 | |
|   sInstance->InitMemoryReporter();
 | |
| }
 | |
| 
 | |
| /* static */
 | |
| void SurfaceCache::Shutdown() {
 | |
|   RefPtr<SurfaceCacheImpl> cache;
 | |
|   {
 | |
|     StaticMutexAutoLock lock(sInstanceMutex);
 | |
|     MOZ_ASSERT(NS_IsMainThread());
 | |
|     MOZ_ASSERT(sInstance, "No singleton - was Shutdown() called twice?");
 | |
|     cache = sInstance.forget();
 | |
|   }
 | |
| }
 | |
| 
 | |
| /* static */
 | |
| LookupResult SurfaceCache::Lookup(const ImageKey aImageKey,
 | |
|                                   const SurfaceKey& aSurfaceKey,
 | |
|                                   bool aMarkUsed) {
 | |
|   nsTArray<RefPtr<CachedSurface>> discard;
 | |
|   LookupResult rv(MatchType::NOT_FOUND);
 | |
| 
 | |
|   {
 | |
|     StaticMutexAutoLock lock(sInstanceMutex);
 | |
|     if (!sInstance) {
 | |
|       return rv;
 | |
|     }
 | |
| 
 | |
|     rv = sInstance->Lookup(aImageKey, aSurfaceKey, lock, aMarkUsed);
 | |
|     sInstance->TakeDiscard(discard, lock);
 | |
|   }
 | |
| 
 | |
|   return rv;
 | |
| }
 | |
| 
 | |
| /* static */
 | |
| LookupResult SurfaceCache::LookupBestMatch(const ImageKey aImageKey,
 | |
|                                            const SurfaceKey& aSurfaceKey,
 | |
|                                            bool aMarkUsed) {
 | |
|   nsTArray<RefPtr<CachedSurface>> discard;
 | |
|   LookupResult rv(MatchType::NOT_FOUND);
 | |
| 
 | |
|   {
 | |
|     StaticMutexAutoLock lock(sInstanceMutex);
 | |
|     if (!sInstance) {
 | |
|       return rv;
 | |
|     }
 | |
| 
 | |
|     rv = sInstance->LookupBestMatch(aImageKey, aSurfaceKey, lock, aMarkUsed);
 | |
|     sInstance->TakeDiscard(discard, lock);
 | |
|   }
 | |
| 
 | |
|   return rv;
 | |
| }
 | |
| 
 | |
| /* static */
 | |
| InsertOutcome SurfaceCache::Insert(NotNull<ISurfaceProvider*> aProvider) {
 | |
|   nsTArray<RefPtr<CachedSurface>> discard;
 | |
|   InsertOutcome rv(InsertOutcome::FAILURE);
 | |
| 
 | |
|   {
 | |
|     StaticMutexAutoLock lock(sInstanceMutex);
 | |
|     if (!sInstance) {
 | |
|       return rv;
 | |
|     }
 | |
| 
 | |
|     rv = sInstance->Insert(aProvider, /* aSetAvailable = */ false, lock);
 | |
|     sInstance->TakeDiscard(discard, lock);
 | |
|   }
 | |
| 
 | |
|   return rv;
 | |
| }
 | |
| 
 | |
| /* static */
 | |
| bool SurfaceCache::CanHold(const IntSize& aSize,
 | |
|                            uint32_t aBytesPerPixel /* = 4 */) {
 | |
|   StaticMutexAutoLock lock(sInstanceMutex);
 | |
|   if (!sInstance) {
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   Cost cost = ComputeCost(aSize, aBytesPerPixel);
 | |
|   return sInstance->CanHold(cost);
 | |
| }
 | |
| 
 | |
| /* static */
 | |
| bool SurfaceCache::CanHold(size_t aSize) {
 | |
|   StaticMutexAutoLock lock(sInstanceMutex);
 | |
|   if (!sInstance) {
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   return sInstance->CanHold(aSize);
 | |
| }
 | |
| 
 | |
| /* static */
 | |
| void SurfaceCache::SurfaceAvailable(NotNull<ISurfaceProvider*> aProvider) {
 | |
|   StaticMutexAutoLock lock(sInstanceMutex);
 | |
|   if (!sInstance) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   sInstance->SurfaceAvailable(aProvider, lock);
 | |
| }
 | |
| 
 | |
| /* static */
 | |
| void SurfaceCache::LockImage(const ImageKey aImageKey) {
 | |
|   StaticMutexAutoLock lock(sInstanceMutex);
 | |
|   if (sInstance) {
 | |
|     return sInstance->LockImage(aImageKey);
 | |
|   }
 | |
| }
 | |
| 
 | |
| /* static */
 | |
| void SurfaceCache::UnlockImage(const ImageKey aImageKey) {
 | |
|   StaticMutexAutoLock lock(sInstanceMutex);
 | |
|   if (sInstance) {
 | |
|     return sInstance->UnlockImage(aImageKey, lock);
 | |
|   }
 | |
| }
 | |
| 
 | |
| /* static */
 | |
| void SurfaceCache::UnlockEntries(const ImageKey aImageKey) {
 | |
|   StaticMutexAutoLock lock(sInstanceMutex);
 | |
|   if (sInstance) {
 | |
|     return sInstance->UnlockEntries(aImageKey, lock);
 | |
|   }
 | |
| }
 | |
| 
 | |
| /* static */
 | |
| void SurfaceCache::RemoveImage(const ImageKey aImageKey) {
 | |
|   RefPtr<ImageSurfaceCache> discard;
 | |
|   {
 | |
|     StaticMutexAutoLock lock(sInstanceMutex);
 | |
|     if (sInstance) {
 | |
|       discard = sInstance->RemoveImage(aImageKey, lock);
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| /* static */
 | |
| void SurfaceCache::PruneImage(const ImageKey aImageKey) {
 | |
|   nsTArray<RefPtr<CachedSurface>> discard;
 | |
|   {
 | |
|     StaticMutexAutoLock lock(sInstanceMutex);
 | |
|     if (sInstance) {
 | |
|       sInstance->PruneImage(aImageKey, lock);
 | |
|       sInstance->TakeDiscard(discard, lock);
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| /* static */
 | |
| bool SurfaceCache::InvalidateImage(const ImageKey aImageKey) {
 | |
|   nsTArray<RefPtr<CachedSurface>> discard;
 | |
|   bool rv = false;
 | |
|   {
 | |
|     StaticMutexAutoLock lock(sInstanceMutex);
 | |
|     if (sInstance) {
 | |
|       rv = sInstance->InvalidateImage(aImageKey, lock);
 | |
|       sInstance->TakeDiscard(discard, lock);
 | |
|     }
 | |
|   }
 | |
|   return rv;
 | |
| }
 | |
| 
 | |
| /* static */
 | |
| void SurfaceCache::DiscardAll() {
 | |
|   nsTArray<RefPtr<CachedSurface>> discard;
 | |
|   {
 | |
|     StaticMutexAutoLock lock(sInstanceMutex);
 | |
|     if (sInstance) {
 | |
|       sInstance->DiscardAll(lock);
 | |
|       sInstance->TakeDiscard(discard, lock);
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| /* static */
 | |
| void SurfaceCache::ResetAnimation(const ImageKey aImageKey,
 | |
|                                   const SurfaceKey& aSurfaceKey) {
 | |
|   RefPtr<CachedSurface> surface;
 | |
|   nsTArray<RefPtr<CachedSurface>> discard;
 | |
|   {
 | |
|     StaticMutexAutoLock lock(sInstanceMutex);
 | |
|     if (!sInstance) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     surface =
 | |
|         sInstance->GetSurfaceForResetAnimation(aImageKey, aSurfaceKey, lock);
 | |
|     sInstance->TakeDiscard(discard, lock);
 | |
|   }
 | |
| 
 | |
|   // Calling Reset will acquire the AnimationSurfaceProvider::mFramesMutex
 | |
|   // mutex. In other places we acquire the mFramesMutex then call into the
 | |
|   // surface cache (acquiring the surface cache mutex), so that determines a
 | |
|   // lock order which we must obey by calling Reset after releasing the surface
 | |
|   // cache mutex.
 | |
|   if (surface) {
 | |
|     DrawableSurface drawableSurface =
 | |
|         surface->GetDrawableSurfaceEvenIfPlaceholder();
 | |
|     if (drawableSurface) {
 | |
|       MOZ_ASSERT(surface->GetSurfaceKey() == aSurfaceKey,
 | |
|                  "ResetAnimation() not returning an exact match?");
 | |
| 
 | |
|       drawableSurface.Reset();
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| /* static */
 | |
| void SurfaceCache::CollectSizeOfSurfaces(
 | |
|     const ImageKey aImageKey, nsTArray<SurfaceMemoryCounter>& aCounters,
 | |
|     MallocSizeOf aMallocSizeOf) {
 | |
|   nsTArray<RefPtr<CachedSurface>> discard;
 | |
|   {
 | |
|     StaticMutexAutoLock lock(sInstanceMutex);
 | |
|     if (!sInstance) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     sInstance->CollectSizeOfSurfaces(aImageKey, aCounters, aMallocSizeOf, lock);
 | |
|     sInstance->TakeDiscard(discard, lock);
 | |
|   }
 | |
| }
 | |
| 
 | |
| /* static */
 | |
| size_t SurfaceCache::MaximumCapacity() {
 | |
|   StaticMutexAutoLock lock(sInstanceMutex);
 | |
|   if (!sInstance) {
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   return sInstance->MaximumCapacity();
 | |
| }
 | |
| 
 | |
| /* static */
 | |
| bool SurfaceCache::IsLegalSize(const IntSize& aSize) {
 | |
|   // reject over-wide or over-tall images
 | |
|   const int32_t k64KLimit = 0x0000FFFF;
 | |
|   if (MOZ_UNLIKELY(aSize.width > k64KLimit || aSize.height > k64KLimit)) {
 | |
|     NS_WARNING("image too big");
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   // protect against invalid sizes
 | |
|   if (MOZ_UNLIKELY(aSize.height <= 0 || aSize.width <= 0)) {
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   // check to make sure we don't overflow a 32-bit
 | |
|   CheckedInt32 requiredBytes =
 | |
|       CheckedInt32(aSize.width) * CheckedInt32(aSize.height) * 4;
 | |
|   if (MOZ_UNLIKELY(!requiredBytes.isValid())) {
 | |
|     NS_WARNING("width or height too large");
 | |
|     return false;
 | |
|   }
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| IntSize SurfaceCache::ClampVectorSize(const IntSize& aSize) {
 | |
|   // If we exceed the maximum, we need to scale the size downwards to fit.
 | |
|   // It shouldn't get here if it is significantly larger because
 | |
|   // VectorImage::UseSurfaceCacheForSize should prevent us from requesting
 | |
|   // a rasterized version of a surface greater than 4x the maximum.
 | |
|   int32_t maxSizeKB =
 | |
|       StaticPrefs::image_cache_max_rasterized_svg_threshold_kb();
 | |
|   if (maxSizeKB <= 0) {
 | |
|     return aSize;
 | |
|   }
 | |
| 
 | |
|   int64_t proposedKB = int64_t(aSize.width) * aSize.height / 256;
 | |
|   if (maxSizeKB >= proposedKB) {
 | |
|     return aSize;
 | |
|   }
 | |
| 
 | |
|   double scale = sqrt(double(maxSizeKB) / proposedKB);
 | |
|   return IntSize(int32_t(scale * aSize.width), int32_t(scale * aSize.height));
 | |
| }
 | |
| 
 | |
| IntSize SurfaceCache::ClampSize(ImageKey aImageKey, const IntSize& aSize) {
 | |
|   if (aImageKey->GetType() != imgIContainer::TYPE_VECTOR) {
 | |
|     return aSize;
 | |
|   }
 | |
| 
 | |
|   return ClampVectorSize(aSize);
 | |
| }
 | |
| 
 | |
| /* static */
 | |
| void SurfaceCache::ReleaseImageOnMainThread(
 | |
|     already_AddRefed<image::Image> aImage, bool aAlwaysProxy) {
 | |
|   if (NS_IsMainThread() && !aAlwaysProxy) {
 | |
|     RefPtr<image::Image> image = std::move(aImage);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   // Don't try to dispatch the release after shutdown, we'll just leak the
 | |
|   // runnable.
 | |
|   if (AppShutdown::IsInOrBeyond(ShutdownPhase::XPCOMShutdownFinal)) {
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   StaticMutexAutoLock lock(sInstanceMutex);
 | |
|   if (sInstance) {
 | |
|     sInstance->ReleaseImageOnMainThread(std::move(aImage), lock);
 | |
|   } else {
 | |
|     NS_ReleaseOnMainThread("SurfaceCache::ReleaseImageOnMainThread",
 | |
|                            std::move(aImage), /* aAlwaysProxy */ true);
 | |
|   }
 | |
| }
 | |
| 
 | |
| /* static */
 | |
| void SurfaceCache::ClearReleasingImages() {
 | |
|   MOZ_ASSERT(NS_IsMainThread());
 | |
| 
 | |
|   nsTArray<RefPtr<image::Image>> images;
 | |
|   {
 | |
|     StaticMutexAutoLock lock(sInstanceMutex);
 | |
|     if (sInstance) {
 | |
|       sInstance->TakeReleasingImages(images, lock);
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| }  // namespace image
 | |
| }  // namespace mozilla
 | 
