| // Copyright 2016 PDFium Authors. All rights reserved. |
| // 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/render/cpdf_imagecacheentry.h" |
| |
| #include <memory> |
| #include <utility> |
| |
| #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/fpdfapi/render/cpdf_pagerendercache.h" |
| #include "core/fpdfapi/render/cpdf_rendercontext.h" |
| #include "core/fpdfapi/render/cpdf_renderstatus.h" |
| #include "core/fxge/dib/cfx_dibitmap.h" |
| #include "third_party/base/numerics/safe_conversions.h" |
| |
| namespace { |
| |
| uint32_t GetEstimatedImageSize(const RetainPtr<CFX_DIBBase>& pDIB) { |
| if (!pDIB || !pDIB->GetBuffer()) |
| return 0; |
| |
| int height = pDIB->GetHeight(); |
| ASSERT(pdfium::base::IsValueInRangeForNumericType<uint32_t>(height)); |
| return static_cast<uint32_t>(height) * pDIB->GetPitch() + |
| pDIB->GetPaletteSize() * 4; |
| } |
| |
| } // namespace |
| |
| CPDF_ImageCacheEntry::CPDF_ImageCacheEntry(CPDF_Document* pDoc, |
| const RetainPtr<CPDF_Image>& pImage) |
| : m_pDocument(pDoc), m_pImage(pImage) {} |
| |
| CPDF_ImageCacheEntry::~CPDF_ImageCacheEntry() = default; |
| |
| void CPDF_ImageCacheEntry::Reset() { |
| m_pCachedBitmap.Reset(); |
| CalcSize(); |
| } |
| |
| RetainPtr<CFX_DIBBase> CPDF_ImageCacheEntry::DetachBitmap() { |
| return std::move(m_pCurBitmap); |
| } |
| |
| RetainPtr<CFX_DIBBase> CPDF_ImageCacheEntry::DetachMask() { |
| return std::move(m_pCurMask); |
| } |
| |
| CPDF_DIB::LoadState CPDF_ImageCacheEntry::StartGetCachedBitmap( |
| const CPDF_Dictionary* pPageResources, |
| const CPDF_RenderStatus* pRenderStatus, |
| bool bStdCS) { |
| if (m_pCachedBitmap) { |
| m_pCurBitmap = m_pCachedBitmap; |
| m_pCurMask = m_pCachedMask; |
| return CPDF_DIB::LoadState::kSuccess; |
| } |
| |
| m_pCurBitmap = pdfium::MakeRetain<CPDF_DIB>(); |
| CPDF_DIB::LoadState ret = m_pCurBitmap.As<CPDF_DIB>()->StartLoadDIBBase( |
| m_pDocument.Get(), m_pImage->GetStream(), true, |
| pRenderStatus->GetFormResource(), pPageResources, bStdCS, |
| pRenderStatus->GetGroupFamily(), pRenderStatus->GetLoadMask()); |
| if (ret == CPDF_DIB::LoadState::kContinue) |
| return CPDF_DIB::LoadState::kContinue; |
| |
| if (ret == CPDF_DIB::LoadState::kSuccess) |
| ContinueGetCachedBitmap(pRenderStatus); |
| else |
| m_pCurBitmap.Reset(); |
| return CPDF_DIB::LoadState::kFail; |
| } |
| |
| bool CPDF_ImageCacheEntry::Continue(PauseIndicatorIface* pPause, |
| CPDF_RenderStatus* pRenderStatus) { |
| CPDF_DIB::LoadState ret = |
| m_pCurBitmap.As<CPDF_DIB>()->ContinueLoadDIBBase(pPause); |
| if (ret == CPDF_DIB::LoadState::kContinue) |
| return true; |
| |
| if (ret == CPDF_DIB::LoadState::kSuccess) |
| ContinueGetCachedBitmap(pRenderStatus); |
| else |
| m_pCurBitmap.Reset(); |
| return false; |
| } |
| |
| void CPDF_ImageCacheEntry::ContinueGetCachedBitmap( |
| const CPDF_RenderStatus* pRenderStatus) { |
| m_MatteColor = m_pCurBitmap.As<CPDF_DIB>()->GetMatteColor(); |
| m_pCurMask = m_pCurBitmap.As<CPDF_DIB>()->DetachMask(); |
| CPDF_RenderContext* pContext = pRenderStatus->GetContext(); |
| CPDF_PageRenderCache* pPageRenderCache = pContext->GetPageCache(); |
| m_dwTimeCount = pPageRenderCache->GetTimeCount(); |
| if (m_pCurBitmap->GetPitch() * m_pCurBitmap->GetHeight() < kHugeImageSize) { |
| m_pCachedBitmap = m_pCurBitmap->Clone(nullptr); |
| m_pCurBitmap.Reset(); |
| } else { |
| m_pCachedBitmap = m_pCurBitmap; |
| } |
| if (m_pCurMask) { |
| m_pCachedMask = m_pCurMask->Clone(nullptr); |
| m_pCurMask.Reset(); |
| } |
| m_pCurBitmap = m_pCachedBitmap; |
| m_pCurMask = m_pCachedMask; |
| CalcSize(); |
| } |
| |
| void CPDF_ImageCacheEntry::CalcSize() { |
| m_dwCacheSize = GetEstimatedImageSize(m_pCachedBitmap) + |
| GetEstimatedImageSize(m_pCachedMask); |
| } |