|  | // 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/render/cpdf_docrenderdata.h" | 
|  |  | 
|  | #include <stdint.h> | 
|  |  | 
|  | #include <algorithm> | 
|  | #include <array> | 
|  | #include <memory> | 
|  | #include <utility> | 
|  |  | 
|  | #include "core/fpdfapi/font/cpdf_type3font.h" | 
|  | #include "core/fpdfapi/page/cpdf_dib.h" | 
|  | #include "core/fpdfapi/page/cpdf_function.h" | 
|  | #include "core/fpdfapi/page/cpdf_transferfunc.h" | 
|  | #include "core/fpdfapi/parser/cpdf_array.h" | 
|  | #include "core/fpdfapi/parser/cpdf_document.h" | 
|  | #include "core/fpdfapi/render/cpdf_type3cache.h" | 
|  | #include "core/fxcrt/compiler_specific.h" | 
|  | #include "core/fxcrt/fixed_size_data_vector.h" | 
|  |  | 
|  | #if BUILDFLAG(IS_WIN) | 
|  | #include "core/fxge/win32/cfx_psfonttracker.h" | 
|  | #endif | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | const int kMaxOutputs = 16; | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | // static | 
|  | CPDF_DocRenderData* CPDF_DocRenderData::FromDocument( | 
|  | const CPDF_Document* pDoc) { | 
|  | return static_cast<CPDF_DocRenderData*>(pDoc->GetRenderData()); | 
|  | } | 
|  |  | 
|  | CPDF_DocRenderData::CPDF_DocRenderData() = default; | 
|  |  | 
|  | CPDF_DocRenderData::~CPDF_DocRenderData() = default; | 
|  |  | 
|  | RetainPtr<CPDF_Type3Cache> CPDF_DocRenderData::GetCachedType3( | 
|  | CPDF_Type3Font* font) { | 
|  | CHECK(font); | 
|  | auto it = type3_face_map_.find(font); | 
|  | if (it != type3_face_map_.end() && it->second) { | 
|  | return pdfium::WrapRetain(it->second.Get()); | 
|  | } | 
|  |  | 
|  | auto cache = pdfium::MakeRetain<CPDF_Type3Cache>(font); | 
|  | type3_face_map_[font].Reset(cache.Get()); | 
|  | return cache; | 
|  | } | 
|  |  | 
|  | RetainPtr<CPDF_TransferFunc> CPDF_DocRenderData::GetTransferFunc( | 
|  | RetainPtr<const CPDF_Object> obj) { | 
|  | CHECK(obj); | 
|  | auto it = transfer_func_map_.find(obj); | 
|  | if (it != transfer_func_map_.end() && it->second) { | 
|  | return pdfium::WrapRetain(it->second.Get()); | 
|  | } | 
|  |  | 
|  | auto func = CreateTransferFunc(obj); | 
|  | transfer_func_map_[obj].Reset(func.Get()); | 
|  | return func; | 
|  | } | 
|  |  | 
|  | #if BUILDFLAG(IS_WIN) | 
|  | CFX_PSFontTracker* CPDF_DocRenderData::GetPSFontTracker() { | 
|  | if (!psfont_tracker_) { | 
|  | psfont_tracker_ = std::make_unique<CFX_PSFontTracker>(); | 
|  | } | 
|  | return psfont_tracker_.get(); | 
|  | } | 
|  | #endif | 
|  |  | 
|  | RetainPtr<CPDF_TransferFunc> CPDF_DocRenderData::CreateTransferFunc( | 
|  | RetainPtr<const CPDF_Object> pObj) const { | 
|  | std::array<std::unique_ptr<CPDF_Function>, 3> pFuncs; | 
|  | const CPDF_Array* pArray = pObj->AsArray(); | 
|  | if (pArray) { | 
|  | if (pArray->size() < 3) { | 
|  | return nullptr; | 
|  | } | 
|  |  | 
|  | for (uint32_t i = 0; i < 3; ++i) { | 
|  | pFuncs[2 - i] = CPDF_Function::Load(pArray->GetDirectObjectAt(i)); | 
|  | if (!pFuncs[2 - i]) { | 
|  | return nullptr; | 
|  | } | 
|  | } | 
|  | } else { | 
|  | pFuncs[0] = CPDF_Function::Load(pObj); | 
|  | if (!pFuncs[0]) { | 
|  | return nullptr; | 
|  | } | 
|  | } | 
|  |  | 
|  | float output[kMaxOutputs]; | 
|  | std::fill(std::begin(output), std::end(output), 0.0f); | 
|  |  | 
|  | bool bIdentity = true; | 
|  | auto samples_r = FixedSizeDataVector<uint8_t>::Uninit( | 
|  | CPDF_TransferFunc::kChannelSampleSize); | 
|  | auto samples_g = FixedSizeDataVector<uint8_t>::Uninit( | 
|  | CPDF_TransferFunc::kChannelSampleSize); | 
|  | auto samples_b = FixedSizeDataVector<uint8_t>::Uninit( | 
|  | CPDF_TransferFunc::kChannelSampleSize); | 
|  |  | 
|  | std::array<pdfium::span<uint8_t>, 3> samples = { | 
|  | samples_r.span(), samples_g.span(), samples_b.span()}; | 
|  | if (pArray) { | 
|  | for (size_t v = 0; v < CPDF_TransferFunc::kChannelSampleSize; ++v) { | 
|  | float input = static_cast<float>(v) / 255.0f; | 
|  | for (int i = 0; i < 3; ++i) { | 
|  | if (pFuncs[i]->OutputCount() > kMaxOutputs) { | 
|  | samples[i][v] = v; | 
|  | continue; | 
|  | } | 
|  | pFuncs[i]->Call(pdfium::span_from_ref(input), output); | 
|  | size_t o = FXSYS_roundf(output[0] * 255); | 
|  | if (o != v) { | 
|  | bIdentity = false; | 
|  | } | 
|  | samples[i][v] = o; | 
|  | } | 
|  | } | 
|  | } else { | 
|  | for (size_t v = 0; v < CPDF_TransferFunc::kChannelSampleSize; ++v) { | 
|  | float input = static_cast<float>(v) / 255.0f; | 
|  | if (pFuncs[0]->OutputCount() <= kMaxOutputs) { | 
|  | pFuncs[0]->Call(pdfium::span_from_ref(input), output); | 
|  | } | 
|  | size_t o = FXSYS_roundf(output[0] * 255); | 
|  | if (o != v) { | 
|  | bIdentity = false; | 
|  | } | 
|  | for (auto& channel : samples) { | 
|  | channel[v] = o; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | return pdfium::MakeRetain<CPDF_TransferFunc>(bIdentity, std::move(samples_r), | 
|  | std::move(samples_g), | 
|  | std::move(samples_b)); | 
|  | } |