| // Copyright 2014 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/fpdf_render/cpdf_pagerendercache.h" |
| |
| #include "core/fpdfapi/fpdf_page/include/cpdf_page.h" |
| #include "core/fpdfapi/fpdf_page/pageint.h" |
| #include "core/fpdfapi/fpdf_parser/include/cpdf_document.h" |
| #include "core/fpdfapi/fpdf_render/include/cpdf_rendercontext.h" |
| #include "core/fpdfapi/fpdf_render/render_int.h" |
| #include "core/fxge/include/fx_ge.h" |
| |
| struct CACHEINFO { |
| uint32_t time; |
| CPDF_Stream* pStream; |
| }; |
| |
| extern "C" { |
| static int compare(const void* data1, const void* data2) { |
| return ((CACHEINFO*)data1)->time - ((CACHEINFO*)data2)->time; |
| } |
| } // extern "C" |
| |
| CPDF_PageRenderCache::CPDF_PageRenderCache(CPDF_Page* pPage) |
| : m_pPage(pPage), |
| m_pCurImageCacheEntry(nullptr), |
| m_nTimeCount(0), |
| m_nCacheSize(0), |
| m_bCurFindCache(FALSE) {} |
| |
| CPDF_PageRenderCache::~CPDF_PageRenderCache() { |
| for (const auto& it : m_ImageCache) |
| delete it.second; |
| } |
| void CPDF_PageRenderCache::CacheOptimization(int32_t dwLimitCacheSize) { |
| if (m_nCacheSize <= (uint32_t)dwLimitCacheSize) |
| return; |
| |
| size_t nCount = m_ImageCache.size(); |
| CACHEINFO* pCACHEINFO = FX_Alloc(CACHEINFO, nCount); |
| size_t i = 0; |
| for (const auto& it : m_ImageCache) { |
| pCACHEINFO[i].time = it.second->GetTimeCount(); |
| pCACHEINFO[i++].pStream = it.second->GetStream(); |
| } |
| FXSYS_qsort(pCACHEINFO, nCount, sizeof(CACHEINFO), compare); |
| uint32_t nTimeCount = m_nTimeCount; |
| |
| // Check if time value is about to roll over and reset all entries. |
| // The comparision is legal because uint32_t is an unsigned type. |
| if (nTimeCount + 1 < nTimeCount) { |
| for (i = 0; i < nCount; i++) |
| m_ImageCache[pCACHEINFO[i].pStream]->m_dwTimeCount = i; |
| m_nTimeCount = nCount; |
| } |
| |
| i = 0; |
| while (i + 15 < nCount) |
| ClearImageCacheEntry(pCACHEINFO[i++].pStream); |
| |
| while (i < nCount && m_nCacheSize > (uint32_t)dwLimitCacheSize) |
| ClearImageCacheEntry(pCACHEINFO[i++].pStream); |
| |
| FX_Free(pCACHEINFO); |
| } |
| void CPDF_PageRenderCache::ClearImageCacheEntry(CPDF_Stream* pStream) { |
| auto it = m_ImageCache.find(pStream); |
| if (it == m_ImageCache.end()) |
| return; |
| |
| m_nCacheSize -= it->second->EstimateSize(); |
| delete it->second; |
| m_ImageCache.erase(it); |
| } |
| uint32_t CPDF_PageRenderCache::EstimateSize() { |
| uint32_t dwSize = 0; |
| for (const auto& it : m_ImageCache) |
| dwSize += it.second->EstimateSize(); |
| |
| m_nCacheSize = dwSize; |
| return dwSize; |
| } |
| void CPDF_PageRenderCache::GetCachedBitmap(CPDF_Stream* pStream, |
| CFX_DIBSource*& pBitmap, |
| CFX_DIBSource*& pMask, |
| uint32_t& MatteColor, |
| FX_BOOL bStdCS, |
| uint32_t GroupFamily, |
| FX_BOOL bLoadMask, |
| CPDF_RenderStatus* pRenderStatus, |
| int32_t downsampleWidth, |
| int32_t downsampleHeight) { |
| CPDF_ImageCacheEntry* pEntry; |
| const auto it = m_ImageCache.find(pStream); |
| FX_BOOL bFound = it != m_ImageCache.end(); |
| if (bFound) |
| pEntry = it->second; |
| else |
| pEntry = new CPDF_ImageCacheEntry(m_pPage->m_pDocument, pStream); |
| |
| m_nTimeCount++; |
| FX_BOOL bAlreadyCached = pEntry->GetCachedBitmap( |
| pBitmap, pMask, MatteColor, m_pPage->m_pPageResources, bStdCS, |
| GroupFamily, bLoadMask, pRenderStatus, downsampleWidth, downsampleHeight); |
| |
| if (!bFound) |
| m_ImageCache[pStream] = pEntry; |
| |
| if (!bAlreadyCached) |
| m_nCacheSize += pEntry->EstimateSize(); |
| } |
| FX_BOOL CPDF_PageRenderCache::StartGetCachedBitmap( |
| CPDF_Stream* pStream, |
| FX_BOOL bStdCS, |
| uint32_t GroupFamily, |
| FX_BOOL bLoadMask, |
| CPDF_RenderStatus* pRenderStatus, |
| int32_t downsampleWidth, |
| int32_t downsampleHeight) { |
| const auto it = m_ImageCache.find(pStream); |
| m_bCurFindCache = it != m_ImageCache.end(); |
| if (m_bCurFindCache) { |
| m_pCurImageCacheEntry = it->second; |
| } else { |
| m_pCurImageCacheEntry = |
| new CPDF_ImageCacheEntry(m_pPage->m_pDocument, pStream); |
| } |
| int ret = m_pCurImageCacheEntry->StartGetCachedBitmap( |
| pRenderStatus->m_pFormResource, m_pPage->m_pPageResources, bStdCS, |
| GroupFamily, bLoadMask, pRenderStatus, downsampleWidth, downsampleHeight); |
| if (ret == 2) |
| return TRUE; |
| |
| m_nTimeCount++; |
| if (!m_bCurFindCache) |
| m_ImageCache[pStream] = m_pCurImageCacheEntry; |
| |
| if (!ret) |
| m_nCacheSize += m_pCurImageCacheEntry->EstimateSize(); |
| |
| return FALSE; |
| } |
| FX_BOOL CPDF_PageRenderCache::Continue(IFX_Pause* pPause) { |
| int ret = m_pCurImageCacheEntry->Continue(pPause); |
| if (ret == 2) |
| return TRUE; |
| m_nTimeCount++; |
| if (!m_bCurFindCache) |
| m_ImageCache[m_pCurImageCacheEntry->GetStream()] = m_pCurImageCacheEntry; |
| if (!ret) |
| m_nCacheSize += m_pCurImageCacheEntry->EstimateSize(); |
| return FALSE; |
| } |
| void CPDF_PageRenderCache::ResetBitmap(CPDF_Stream* pStream, |
| const CFX_DIBitmap* pBitmap) { |
| CPDF_ImageCacheEntry* pEntry; |
| const auto it = m_ImageCache.find(pStream); |
| if (it == m_ImageCache.end()) { |
| if (!pBitmap) |
| return; |
| pEntry = new CPDF_ImageCacheEntry(m_pPage->m_pDocument, pStream); |
| m_ImageCache[pStream] = pEntry; |
| } else { |
| pEntry = it->second; |
| } |
| m_nCacheSize -= pEntry->EstimateSize(); |
| pEntry->Reset(pBitmap); |
| m_nCacheSize += pEntry->EstimateSize(); |
| } |
| CPDF_ImageCacheEntry::CPDF_ImageCacheEntry(CPDF_Document* pDoc, |
| CPDF_Stream* pStream) |
| : m_dwTimeCount(0), |
| m_pCurBitmap(nullptr), |
| m_pCurMask(nullptr), |
| m_MatteColor(0), |
| m_pRenderStatus(nullptr), |
| m_pDocument(pDoc), |
| m_pStream(pStream), |
| m_pCachedBitmap(nullptr), |
| m_pCachedMask(nullptr), |
| m_dwCacheSize(0) {} |
| CPDF_ImageCacheEntry::~CPDF_ImageCacheEntry() { |
| delete m_pCachedBitmap; |
| delete m_pCachedMask; |
| } |
| void CPDF_ImageCacheEntry::Reset(const CFX_DIBitmap* pBitmap) { |
| delete m_pCachedBitmap; |
| m_pCachedBitmap = nullptr; |
| if (pBitmap) { |
| m_pCachedBitmap = pBitmap->Clone(); |
| } |
| CalcSize(); |
| } |
| |
| static uint32_t FPDF_ImageCache_EstimateImageSize(const CFX_DIBSource* pDIB) { |
| return pDIB && pDIB->GetBuffer() |
| ? (uint32_t)pDIB->GetHeight() * pDIB->GetPitch() + |
| (uint32_t)pDIB->GetPaletteSize() * 4 |
| : 0; |
| } |
| FX_BOOL CPDF_ImageCacheEntry::GetCachedBitmap(CFX_DIBSource*& pBitmap, |
| CFX_DIBSource*& pMask, |
| uint32_t& MatteColor, |
| CPDF_Dictionary* pPageResources, |
| FX_BOOL bStdCS, |
| uint32_t GroupFamily, |
| FX_BOOL bLoadMask, |
| CPDF_RenderStatus* pRenderStatus, |
| int32_t downsampleWidth, |
| int32_t downsampleHeight) { |
| if (m_pCachedBitmap) { |
| pBitmap = m_pCachedBitmap; |
| pMask = m_pCachedMask; |
| MatteColor = m_MatteColor; |
| return TRUE; |
| } |
| if (!pRenderStatus) { |
| return FALSE; |
| } |
| CPDF_RenderContext* pContext = pRenderStatus->GetContext(); |
| CPDF_PageRenderCache* pPageRenderCache = pContext->GetPageCache(); |
| m_dwTimeCount = pPageRenderCache->GetTimeCount(); |
| CPDF_DIBSource* pSrc = new CPDF_DIBSource; |
| CPDF_DIBSource* pMaskSrc = nullptr; |
| if (!pSrc->Load(m_pDocument, m_pStream, &pMaskSrc, &MatteColor, |
| pRenderStatus->m_pFormResource, pPageResources, bStdCS, |
| GroupFamily, bLoadMask)) { |
| delete pSrc; |
| pBitmap = nullptr; |
| return FALSE; |
| } |
| m_MatteColor = MatteColor; |
| if (pSrc->GetPitch() * pSrc->GetHeight() < FPDF_HUGE_IMAGE_SIZE) { |
| m_pCachedBitmap = pSrc->Clone(); |
| delete pSrc; |
| } else { |
| m_pCachedBitmap = pSrc; |
| } |
| if (pMaskSrc) { |
| m_pCachedMask = pMaskSrc->Clone(); |
| delete pMaskSrc; |
| } |
| |
| pBitmap = m_pCachedBitmap; |
| pMask = m_pCachedMask; |
| CalcSize(); |
| return FALSE; |
| } |
| CFX_DIBSource* CPDF_ImageCacheEntry::DetachBitmap() { |
| CFX_DIBSource* pDIBSource = m_pCurBitmap; |
| m_pCurBitmap = nullptr; |
| return pDIBSource; |
| } |
| CFX_DIBSource* CPDF_ImageCacheEntry::DetachMask() { |
| CFX_DIBSource* pDIBSource = m_pCurMask; |
| m_pCurMask = nullptr; |
| return pDIBSource; |
| } |
| int CPDF_ImageCacheEntry::StartGetCachedBitmap(CPDF_Dictionary* pFormResources, |
| CPDF_Dictionary* pPageResources, |
| FX_BOOL bStdCS, |
| uint32_t GroupFamily, |
| FX_BOOL bLoadMask, |
| CPDF_RenderStatus* pRenderStatus, |
| int32_t downsampleWidth, |
| int32_t downsampleHeight) { |
| if (m_pCachedBitmap) { |
| m_pCurBitmap = m_pCachedBitmap; |
| m_pCurMask = m_pCachedMask; |
| return 1; |
| } |
| if (!pRenderStatus) { |
| return 0; |
| } |
| m_pRenderStatus = pRenderStatus; |
| m_pCurBitmap = new CPDF_DIBSource; |
| int ret = |
| ((CPDF_DIBSource*)m_pCurBitmap) |
| ->StartLoadDIBSource(m_pDocument, m_pStream, TRUE, pFormResources, |
| pPageResources, bStdCS, GroupFamily, bLoadMask); |
| if (ret == 2) { |
| return ret; |
| } |
| if (!ret) { |
| delete m_pCurBitmap; |
| m_pCurBitmap = nullptr; |
| return 0; |
| } |
| ContinueGetCachedBitmap(); |
| return 0; |
| } |
| void CPDF_ImageCacheEntry::ContinueGetCachedBitmap() { |
| m_MatteColor = ((CPDF_DIBSource*)m_pCurBitmap)->GetMatteColor(); |
| m_pCurMask = ((CPDF_DIBSource*)m_pCurBitmap)->DetachMask(); |
| CPDF_RenderContext* pContext = m_pRenderStatus->GetContext(); |
| CPDF_PageRenderCache* pPageRenderCache = pContext->GetPageCache(); |
| m_dwTimeCount = pPageRenderCache->GetTimeCount(); |
| if (m_pCurBitmap->GetPitch() * m_pCurBitmap->GetHeight() < |
| FPDF_HUGE_IMAGE_SIZE) { |
| m_pCachedBitmap = m_pCurBitmap->Clone(); |
| delete m_pCurBitmap; |
| m_pCurBitmap = nullptr; |
| } else { |
| m_pCachedBitmap = m_pCurBitmap; |
| } |
| if (m_pCurMask) { |
| m_pCachedMask = m_pCurMask->Clone(); |
| delete m_pCurMask; |
| m_pCurMask = nullptr; |
| } |
| m_pCurBitmap = m_pCachedBitmap; |
| m_pCurMask = m_pCachedMask; |
| CalcSize(); |
| } |
| int CPDF_ImageCacheEntry::Continue(IFX_Pause* pPause) { |
| int ret = ((CPDF_DIBSource*)m_pCurBitmap)->ContinueLoadDIBSource(pPause); |
| if (ret == 2) { |
| return ret; |
| } |
| if (!ret) { |
| delete m_pCurBitmap; |
| m_pCurBitmap = nullptr; |
| return 0; |
| } |
| ContinueGetCachedBitmap(); |
| return 0; |
| } |
| void CPDF_ImageCacheEntry::CalcSize() { |
| m_dwCacheSize = FPDF_ImageCache_EstimateImageSize(m_pCachedBitmap) + |
| FPDF_ImageCache_EstimateImageSize(m_pCachedMask); |
| } |