// Copyright 2016 The PDFium Authors // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com #include "core/fpdfapi/page/cpdf_pageimagecache.h" #include #include #include #include #include #include "core/fpdfapi/page/cpdf_dib.h" #include "core/fpdfapi/page/cpdf_image.h" #include "core/fpdfapi/page/cpdf_page.h" #include "core/fpdfapi/parser/cpdf_dictionary.h" #include "core/fpdfapi/parser/cpdf_document.h" #include "core/fpdfapi/parser/cpdf_stream.h" #include "core/fxcrt/check.h" #include "core/fxcrt/retain_ptr.h" #include "core/fxcrt/stl_util.h" #include "core/fxge/dib/cfx_dibbase.h" #include "core/fxge/dib/cfx_dibitmap.h" #if defined(PDF_USE_SKIA) #include "core/fxcrt/data_vector.h" #include "core/fxge/cfx_defaultrenderdevice.h" #include "third_party/skia/include/core/SkImage.h" // nogncheck #include "third_party/skia/include/core/SkRefCnt.h" // nogncheck #endif namespace { struct CacheInfo { CacheInfo(uint32_t t, RetainPtr stream) : time(t), pStream(std::move(stream)) {} uint32_t time; RetainPtr pStream; bool operator<(const CacheInfo& other) const { return time < other.time; } }; #if defined(PDF_USE_SKIA) // Wrapper around a `CFX_DIBBase` that memoizes `RealizeSkImage()`. This is only // safe if the underlying `CFX_DIBBase` is not mutable. class CachedImage final : public CFX_DIBBase { public: explicit CachedImage(RetainPtr image) : image_(std::move(image)) { SetFormat(image_->GetFormat()); SetWidth(image_->GetWidth()); SetHeight(image_->GetHeight()); SetPitch(image_->GetPitch()); if (image_->HasPalette()) { pdfium::span palette = image_->GetPaletteSpan(); palette_ = DataVector(palette.begin(), palette.end()); } } pdfium::span GetScanline(int line) const override { // TODO(crbug.com/pdfium/2050): Still needed for `Realize()` call in // `CPDF_ImageRenderer`. return image_->GetScanline(line); } bool SkipToScanline(int line, PauseIndicatorIface* pause) const override { return image_->SkipToScanline(line, pause); } size_t GetEstimatedImageMemoryBurden() const override { // A better estimate would account for realizing the `SkImage`. return image_->GetEstimatedImageMemoryBurden(); } #if BUILDFLAG(IS_WIN) || defined(PDF_USE_SKIA) RetainPtr RealizeIfNeeded() const override { return image_->RealizeIfNeeded(); } #endif sk_sp RealizeSkImage() const override { if (!cached_skia_image_) { cached_skia_image_ = image_->RealizeSkImage(); } return cached_skia_image_; } private: RetainPtr image_; mutable sk_sp cached_skia_image_; }; #endif // defined(PDF_USE_SKIA) // Makes a `CachedImage` backed by `image` if Skia is the default renderer, // otherwise return the image itself. `realize_hint` indicates whether it would // be beneficial to realize `image` before caching. RetainPtr MakeCachedImage(RetainPtr image, bool realize_hint) { #if defined(PDF_USE_SKIA) if (CFX_DefaultRenderDevice::UseSkiaRenderer()) { // Ignore `realize_hint`, as `RealizeSkImage()` doesn't benefit from it. return pdfium::MakeRetain(std::move(image)); } #endif // defined(PDF_USE_SKIA) return realize_hint ? image->Realize() : image; } } // namespace CPDF_PageImageCache::CPDF_PageImageCache(CPDF_Page* pPage) : m_pPage(pPage) {} CPDF_PageImageCache::~CPDF_PageImageCache() = default; void CPDF_PageImageCache::CacheOptimization(int32_t dwLimitCacheSize) { if (m_nCacheSize <= (uint32_t)dwLimitCacheSize) return; uint32_t nCount = fxcrt::CollectionSize(m_ImageCache); std::vector cache_info; cache_info.reserve(nCount); for (const auto& it : m_ImageCache) { cache_info.emplace_back(it.second->GetTimeCount(), it.second->GetImage()->GetStream()); } std::sort(cache_info.begin(), cache_info.end()); // Check if time value is about to roll over and reset all entries. // The comparison is legal because uint32_t is an unsigned type. uint32_t nTimeCount = m_nTimeCount; if (nTimeCount + 1 < nTimeCount) { for (uint32_t i = 0; i < nCount; i++) m_ImageCache[cache_info[i].pStream]->SetTimeCount(i); m_nTimeCount = nCount; } size_t i = 0; while (i + 15 < nCount) ClearImageCacheEntry(cache_info[i++].pStream); while (i < nCount && m_nCacheSize > (uint32_t)dwLimitCacheSize) ClearImageCacheEntry(cache_info[i++].pStream); } void CPDF_PageImageCache::ClearImageCacheEntry(const CPDF_Stream* pStream) { auto it = m_ImageCache.find(pStream); if (it == m_ImageCache.end()) return; m_nCacheSize -= it->second->EstimateSize(); // Avoid leaving `m_pCurImageCacheEntry` as a dangling pointer when `it` is // about to be deleted. if (m_pCurImageCacheEntry.Get() == it->second.get()) { DCHECK(!m_pCurImageCacheEntry.IsOwned()); m_pCurImageCacheEntry.Reset(); } m_ImageCache.erase(it); } bool CPDF_PageImageCache::StartGetCachedBitmap( RetainPtr pImage, const CPDF_Dictionary* pFormResources, const CPDF_Dictionary* pPageResources, bool bStdCS, CPDF_ColorSpace::Family eFamily, bool bLoadMask, const CFX_Size& max_size_required) { // A cross-document image may have come from the embedder. if (m_pPage->GetDocument() != pImage->GetDocument()) return false; RetainPtr pStream = pImage->GetStream(); const auto it = m_ImageCache.find(pStream); m_bCurFindCache = it != m_ImageCache.end(); if (m_bCurFindCache) { m_pCurImageCacheEntry = it->second.get(); } else { m_pCurImageCacheEntry = std::make_unique(std::move(pImage)); } CPDF_DIB::LoadState ret = m_pCurImageCacheEntry->StartGetCachedBitmap( this, pFormResources, pPageResources, bStdCS, eFamily, bLoadMask, max_size_required); if (ret == CPDF_DIB::LoadState::kContinue) return true; m_nTimeCount++; if (!m_bCurFindCache) m_ImageCache[pStream] = m_pCurImageCacheEntry.Release(); if (ret == CPDF_DIB::LoadState::kFail) m_nCacheSize += m_pCurImageCacheEntry->EstimateSize(); return false; } bool CPDF_PageImageCache::Continue(PauseIndicatorIface* pPause) { bool ret = m_pCurImageCacheEntry->Continue(pPause, this); if (ret) return true; m_nTimeCount++; if (!m_bCurFindCache) { m_ImageCache[m_pCurImageCacheEntry->GetImage()->GetStream()] = m_pCurImageCacheEntry.Release(); } m_nCacheSize += m_pCurImageCacheEntry->EstimateSize(); return false; } void CPDF_PageImageCache::ResetBitmapForImage(RetainPtr pImage) { RetainPtr pStream = pImage->GetStream(); const auto it = m_ImageCache.find(pStream); if (it == m_ImageCache.end()) return; Entry* pEntry = it->second.get(); m_nCacheSize -= pEntry->EstimateSize(); pEntry->Reset(); m_nCacheSize += pEntry->EstimateSize(); } uint32_t CPDF_PageImageCache::GetCurMatteColor() const { return m_pCurImageCacheEntry->GetMatteColor(); } RetainPtr CPDF_PageImageCache::DetachCurBitmap() { return m_pCurImageCacheEntry->DetachBitmap(); } RetainPtr CPDF_PageImageCache::DetachCurMask() { return m_pCurImageCacheEntry->DetachMask(); } CPDF_PageImageCache::Entry::Entry(RetainPtr pImage) : m_pImage(std::move(pImage)) {} CPDF_PageImageCache::Entry::~Entry() = default; void CPDF_PageImageCache::Entry::Reset() { m_pCachedBitmap.Reset(); CalcSize(); } RetainPtr CPDF_PageImageCache::Entry::DetachBitmap() { return std::move(m_pCurBitmap); } RetainPtr CPDF_PageImageCache::Entry::DetachMask() { return std::move(m_pCurMask); } CPDF_DIB::LoadState CPDF_PageImageCache::Entry::StartGetCachedBitmap( CPDF_PageImageCache* pPageImageCache, const CPDF_Dictionary* pFormResources, const CPDF_Dictionary* pPageResources, bool bStdCS, CPDF_ColorSpace::Family eFamily, bool bLoadMask, const CFX_Size& max_size_required) { if (m_pCachedBitmap && IsCacheValid(max_size_required)) { m_pCurBitmap = m_pCachedBitmap; m_pCurMask = m_pCachedMask; return CPDF_DIB::LoadState::kSuccess; } m_pCurBitmap = m_pImage->CreateNewDIB(); CPDF_DIB::LoadState ret = m_pCurBitmap.AsRaw()->StartLoadDIBBase( true, pFormResources, pPageResources, bStdCS, eFamily, bLoadMask, max_size_required); m_bCachedSetMaxSizeRequired = (max_size_required.width != 0 && max_size_required.height != 0); if (ret == CPDF_DIB::LoadState::kContinue) return CPDF_DIB::LoadState::kContinue; if (ret == CPDF_DIB::LoadState::kSuccess) ContinueGetCachedBitmap(pPageImageCache); else m_pCurBitmap.Reset(); return CPDF_DIB::LoadState::kFail; } bool CPDF_PageImageCache::Entry::Continue( PauseIndicatorIface* pPause, CPDF_PageImageCache* pPageImageCache) { CPDF_DIB::LoadState ret = m_pCurBitmap.AsRaw()->ContinueLoadDIBBase(pPause); if (ret == CPDF_DIB::LoadState::kContinue) return true; if (ret == CPDF_DIB::LoadState::kSuccess) ContinueGetCachedBitmap(pPageImageCache); else m_pCurBitmap.Reset(); return false; } void CPDF_PageImageCache::Entry::ContinueGetCachedBitmap( CPDF_PageImageCache* pPageImageCache) { m_MatteColor = m_pCurBitmap.AsRaw()->GetMatteColor(); m_pCurMask = m_pCurBitmap.AsRaw()->DetachMask(); m_dwTimeCount = pPageImageCache->GetTimeCount(); if (m_pCurBitmap->GetPitch() * m_pCurBitmap->GetHeight() < kHugeImageSize) { m_pCachedBitmap = MakeCachedImage(m_pCurBitmap, /*realize_hint=*/true); m_pCurBitmap.Reset(); } else { m_pCachedBitmap = MakeCachedImage(m_pCurBitmap, /*realize_hint=*/false); } if (m_pCurMask) { m_pCachedMask = MakeCachedImage(m_pCurMask, /*realize_hint=*/true); m_pCurMask.Reset(); } m_pCurBitmap = m_pCachedBitmap; m_pCurMask = m_pCachedMask; CalcSize(); } void CPDF_PageImageCache::Entry::CalcSize() { m_dwCacheSize = 0; if (m_pCachedBitmap) m_dwCacheSize += m_pCachedBitmap->GetEstimatedImageMemoryBurden(); if (m_pCachedMask) m_dwCacheSize += m_pCachedMask->GetEstimatedImageMemoryBurden(); } bool CPDF_PageImageCache::Entry::IsCacheValid( const CFX_Size& max_size_required) const { if (!m_bCachedSetMaxSizeRequired) { return true; } if (max_size_required.width == 0 && max_size_required.height == 0) { return false; } return (m_pCachedBitmap->GetWidth() >= max_size_required.width) && (m_pCachedBitmap->GetHeight() >= max_size_required.height); }