| dan sinclair | 61b2fc7 | 2016-03-23 19:21:44 -0400 | [diff] [blame] | 1 | // Copyright 2016 PDFium Authors. All rights reserved. |
| John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 2 | // Use of this source code is governed by a BSD-style license that can be |
| 3 | // found in the LICENSE file. |
| Lei Zhang | 95e854f | 2015-06-13 00:58:06 -0700 | [diff] [blame] | 4 | |
| John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 5 | // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com |
| 6 | |
| dsinclair | 41872fa | 2016-10-04 11:29:35 -0700 | [diff] [blame] | 7 | #include "core/fpdfapi/page/cpdf_image.h" |
| dan sinclair | 61b2fc7 | 2016-03-23 19:21:44 -0400 | [diff] [blame] | 8 | |
| thestig | dc359b0 | 2016-08-09 15:46:20 -0700 | [diff] [blame] | 9 | #include <algorithm> |
| tsepez | 9e05ee1 | 2016-11-21 13:19:10 -0800 | [diff] [blame] | 10 | #include <memory> |
| tsepez | 0e606b5 | 2016-11-18 16:22:41 -0800 | [diff] [blame] | 11 | #include <utility> |
| thestig | dc359b0 | 2016-08-09 15:46:20 -0700 | [diff] [blame] | 12 | #include <vector> |
| 13 | |
| Lei Zhang | 2617056 | 2018-04-17 17:01:52 +0000 | [diff] [blame] | 14 | #include "constants/stream_dict_common.h" |
| dsinclair | 39c62fd | 2016-09-29 12:49:17 -0700 | [diff] [blame] | 15 | #include "core/fpdfapi/cpdf_modulemgr.h" |
| dsinclair | 41872fa | 2016-10-04 11:29:35 -0700 | [diff] [blame] | 16 | #include "core/fpdfapi/page/cpdf_page.h" |
| dsinclair | 488b7ad | 2016-10-04 11:55:50 -0700 | [diff] [blame] | 17 | #include "core/fpdfapi/parser/cpdf_array.h" |
| 18 | #include "core/fpdfapi/parser/cpdf_boolean.h" |
| tsepez | 4e4d1a6 | 2016-10-13 15:56:53 -0700 | [diff] [blame] | 19 | #include "core/fpdfapi/parser/cpdf_dictionary.h" |
| dsinclair | 488b7ad | 2016-10-04 11:55:50 -0700 | [diff] [blame] | 20 | #include "core/fpdfapi/parser/cpdf_document.h" |
| tsepez | 8a3aa45 | 2016-11-16 12:26:06 -0800 | [diff] [blame] | 21 | #include "core/fpdfapi/parser/cpdf_name.h" |
| 22 | #include "core/fpdfapi/parser/cpdf_number.h" |
| 23 | #include "core/fpdfapi/parser/cpdf_reference.h" |
| tsepez | 9e05ee1 | 2016-11-21 13:19:10 -0800 | [diff] [blame] | 24 | #include "core/fpdfapi/parser/cpdf_stream.h" |
| dsinclair | 488b7ad | 2016-10-04 11:55:50 -0700 | [diff] [blame] | 25 | #include "core/fpdfapi/parser/cpdf_string.h" |
| Tom Sepez | e6ff2eb | 2018-08-24 21:55:46 +0000 | [diff] [blame] | 26 | #include "core/fpdfapi/render/cpdf_dibbase.h" |
| dsinclair | 69d9c68 | 2016-10-04 12:18:35 -0700 | [diff] [blame] | 27 | #include "core/fpdfapi/render/cpdf_pagerendercache.h" |
| Lei Zhang | 7acd826 | 2017-09-19 14:34:37 -0700 | [diff] [blame] | 28 | #include "core/fxcodec/codec/ccodec_jpegmodule.h" |
| Dan Sinclair | bcd1e70 | 2017-08-31 13:19:18 -0400 | [diff] [blame] | 29 | #include "core/fxcrt/fx_stream.h" |
| Lei Zhang | 6e59fb5 | 2018-01-18 19:03:58 +0000 | [diff] [blame] | 30 | #include "core/fxge/dib/cfx_dibitmap.h" |
| dsinclair | 74a34fc | 2016-09-29 16:41:42 -0700 | [diff] [blame] | 31 | #include "core/fxge/fx_dib.h" |
| tsepez | 06104a8 | 2016-11-21 16:22:10 -0800 | [diff] [blame] | 32 | #include "third_party/base/numerics/safe_conversions.h" |
| tsepez | 9e05ee1 | 2016-11-21 13:19:10 -0800 | [diff] [blame] | 33 | #include "third_party/base/ptr_util.h" |
| dan sinclair | 61b2fc7 | 2016-03-23 19:21:44 -0400 | [diff] [blame] | 34 | |
| Lei Zhang | 8590a98 | 2018-09-21 15:34:45 +0000 | [diff] [blame] | 35 | // static |
| 36 | bool CPDF_Image::IsValidJpegComponent(int32_t comps) { |
| Lei Zhang | 34cdc8f | 2018-09-20 16:23:02 +0000 | [diff] [blame] | 37 | return comps == 1 || comps == 3 || comps == 4; |
| 38 | } |
| 39 | |
| Lei Zhang | 8590a98 | 2018-09-21 15:34:45 +0000 | [diff] [blame] | 40 | // static |
| 41 | bool CPDF_Image::IsValidJpegBitsPerComponent(int32_t bpc) { |
| Lei Zhang | 34cdc8f | 2018-09-20 16:23:02 +0000 | [diff] [blame] | 42 | return bpc == 1 || bpc == 2 || bpc == 4 || bpc == 8 || bpc == 16; |
| 43 | } |
| 44 | |
| tsepez | 4e4d1a6 | 2016-10-13 15:56:53 -0700 | [diff] [blame] | 45 | CPDF_Image::CPDF_Image(CPDF_Document* pDoc) : m_pDocument(pDoc) {} |
| thestig | dc359b0 | 2016-08-09 15:46:20 -0700 | [diff] [blame] | 46 | |
| tsepez | 33fdebc | 2016-11-04 11:38:40 -0700 | [diff] [blame] | 47 | CPDF_Image::CPDF_Image(CPDF_Document* pDoc, |
| 48 | std::unique_ptr<CPDF_Stream> pStream) |
| Tom Sepez | bb1ee53 | 2018-01-30 18:10:01 +0000 | [diff] [blame] | 49 | : m_bIsInline(true), m_pDocument(pDoc), m_pStream(std::move(pStream)) { |
| tsepez | 9fd0c63 | 2016-11-23 14:34:58 -0800 | [diff] [blame] | 50 | ASSERT(m_pStream.IsOwned()); |
| Tom Sepez | bb1ee53 | 2018-01-30 18:10:01 +0000 | [diff] [blame] | 51 | FinishInitialization(m_pStream->GetDict()); |
| thestig | dc359b0 | 2016-08-09 15:46:20 -0700 | [diff] [blame] | 52 | } |
| dan sinclair | 61b2fc7 | 2016-03-23 19:21:44 -0400 | [diff] [blame] | 53 | |
| tsepez | 4e4d1a6 | 2016-10-13 15:56:53 -0700 | [diff] [blame] | 54 | CPDF_Image::CPDF_Image(CPDF_Document* pDoc, uint32_t dwStreamObjNum) |
| 55 | : m_pDocument(pDoc), |
| Tom Sepez | bb1ee53 | 2018-01-30 18:10:01 +0000 | [diff] [blame] | 56 | m_pStream(ToStream(pDoc->GetIndirectObject(dwStreamObjNum))) { |
| tsepez | 9fd0c63 | 2016-11-23 14:34:58 -0800 | [diff] [blame] | 57 | ASSERT(!m_pStream.IsOwned()); |
| Tom Sepez | bb1ee53 | 2018-01-30 18:10:01 +0000 | [diff] [blame] | 58 | FinishInitialization(m_pStream->GetDict()); |
| tsepez | 4e4d1a6 | 2016-10-13 15:56:53 -0700 | [diff] [blame] | 59 | } |
| 60 | |
| 61 | CPDF_Image::~CPDF_Image() {} |
| 62 | |
| Lei Zhang | aca3efc | 2018-03-16 20:27:04 +0000 | [diff] [blame] | 63 | void CPDF_Image::FinishInitialization(CPDF_Dictionary* pStreamDict) { |
| 64 | m_pOC = pStreamDict->GetDictFor("OC"); |
| 65 | m_bIsMask = !pStreamDict->KeyExist("ColorSpace") || |
| 66 | pStreamDict->GetIntegerFor("ImageMask"); |
| 67 | m_bInterpolate = !!pStreamDict->GetIntegerFor("Interpolate"); |
| 68 | m_Height = pStreamDict->GetIntegerFor("Height"); |
| 69 | m_Width = pStreamDict->GetIntegerFor("Width"); |
| dan sinclair | 61b2fc7 | 2016-03-23 19:21:44 -0400 | [diff] [blame] | 70 | } |
| 71 | |
| tsepez | 137a344 | 2016-11-14 15:03:55 -0800 | [diff] [blame] | 72 | void CPDF_Image::ConvertStreamToIndirectObject() { |
| 73 | if (!m_pStream->IsInline()) |
| 74 | return; |
| 75 | |
| tsepez | 9fd0c63 | 2016-11-23 14:34:58 -0800 | [diff] [blame] | 76 | ASSERT(m_pStream.IsOwned()); |
| 77 | m_pDocument->AddIndirectObject(m_pStream.Release()); |
| tsepez | 137a344 | 2016-11-14 15:03:55 -0800 | [diff] [blame] | 78 | } |
| 79 | |
| Lei Zhang | 731526e | 2017-12-11 21:28:38 +0000 | [diff] [blame] | 80 | CPDF_Dictionary* CPDF_Image::GetDict() const { |
| 81 | return m_pStream ? m_pStream->GetDict() : nullptr; |
| 82 | } |
| 83 | |
| Tom Sepez | 126927e | 2018-08-28 22:25:49 +0000 | [diff] [blame] | 84 | std::unique_ptr<CPDF_Dictionary> CPDF_Image::InitJPEG( |
| 85 | pdfium::span<uint8_t> src_span) { |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 86 | int32_t width; |
| 87 | int32_t height; |
| 88 | int32_t num_comps; |
| 89 | int32_t bits; |
| thestig | b0fcfad | 2016-06-06 17:54:29 -0700 | [diff] [blame] | 90 | bool color_trans; |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 91 | if (!CPDF_ModuleMgr::Get()->GetJpegModule()->LoadInfo( |
| Tom Sepez | 126927e | 2018-08-28 22:25:49 +0000 | [diff] [blame] | 92 | src_span, &width, &height, &num_comps, &bits, &color_trans)) { |
| thestig | b0fcfad | 2016-06-06 17:54:29 -0700 | [diff] [blame] | 93 | return nullptr; |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 94 | } |
| Lei Zhang | 34cdc8f | 2018-09-20 16:23:02 +0000 | [diff] [blame] | 95 | if (!IsValidJpegComponent(num_comps) || !IsValidJpegBitsPerComponent(bits)) |
| 96 | return nullptr; |
| thestig | b0fcfad | 2016-06-06 17:54:29 -0700 | [diff] [blame] | 97 | |
| tsepez | 06104a8 | 2016-11-21 16:22:10 -0800 | [diff] [blame] | 98 | auto pDict = |
| 99 | pdfium::MakeUnique<CPDF_Dictionary>(m_pDocument->GetByteStringPool()); |
| tsepez | 0e606b5 | 2016-11-18 16:22:41 -0800 | [diff] [blame] | 100 | pDict->SetNewFor<CPDF_Name>("Type", "XObject"); |
| 101 | pDict->SetNewFor<CPDF_Name>("Subtype", "Image"); |
| 102 | pDict->SetNewFor<CPDF_Number>("Width", width); |
| 103 | pDict->SetNewFor<CPDF_Number>("Height", height); |
| Dan Sinclair | 812e96c | 2017-03-13 16:43:37 -0400 | [diff] [blame] | 104 | const char* csname = nullptr; |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 105 | if (num_comps == 1) { |
| 106 | csname = "DeviceGray"; |
| 107 | } else if (num_comps == 3) { |
| 108 | csname = "DeviceRGB"; |
| 109 | } else if (num_comps == 4) { |
| 110 | csname = "DeviceCMYK"; |
| tsepez | 0e606b5 | 2016-11-18 16:22:41 -0800 | [diff] [blame] | 111 | CPDF_Array* pDecode = pDict->SetNewFor<CPDF_Array>("Decode"); |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 112 | for (int n = 0; n < 4; n++) { |
| tsepez | 8a3aa45 | 2016-11-16 12:26:06 -0800 | [diff] [blame] | 113 | pDecode->AddNew<CPDF_Number>(1); |
| 114 | pDecode->AddNew<CPDF_Number>(0); |
| John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 115 | } |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 116 | } |
| tsepez | 0e606b5 | 2016-11-18 16:22:41 -0800 | [diff] [blame] | 117 | pDict->SetNewFor<CPDF_Name>("ColorSpace", csname); |
| 118 | pDict->SetNewFor<CPDF_Number>("BitsPerComponent", bits); |
| 119 | pDict->SetNewFor<CPDF_Name>("Filter", "DCTDecode"); |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 120 | if (!color_trans) { |
| Lei Zhang | 2617056 | 2018-04-17 17:01:52 +0000 | [diff] [blame] | 121 | CPDF_Dictionary* pParms = |
| 122 | pDict->SetNewFor<CPDF_Dictionary>(pdfium::stream::kDecodeParms); |
| tsepez | 0e606b5 | 2016-11-18 16:22:41 -0800 | [diff] [blame] | 123 | pParms->SetNewFor<CPDF_Number>("ColorTransform", 0); |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 124 | } |
| tsepez | 12f3e4a | 2016-11-02 15:17:29 -0700 | [diff] [blame] | 125 | m_bIsMask = false; |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 126 | m_Width = width; |
| 127 | m_Height = height; |
| tsepez | 9fd0c63 | 2016-11-23 14:34:58 -0800 | [diff] [blame] | 128 | if (!m_pStream) |
| 129 | m_pStream = pdfium::MakeUnique<CPDF_Stream>(); |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 130 | return pDict; |
| John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 131 | } |
| dan sinclair | 61b2fc7 | 2016-03-23 19:21:44 -0400 | [diff] [blame] | 132 | |
| Dan Sinclair | 0b95042 | 2017-09-21 15:49:49 -0400 | [diff] [blame] | 133 | void CPDF_Image::SetJpegImage(const RetainPtr<IFX_SeekableReadStream>& pFile) { |
| tsepez | 06104a8 | 2016-11-21 16:22:10 -0800 | [diff] [blame] | 134 | uint32_t size = pdfium::base::checked_cast<uint32_t>(pFile->GetSize()); |
| thestig | dc359b0 | 2016-08-09 15:46:20 -0700 | [diff] [blame] | 135 | if (!size) |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 136 | return; |
| thestig | dc359b0 | 2016-08-09 15:46:20 -0700 | [diff] [blame] | 137 | |
| 138 | uint32_t dwEstimateSize = std::min(size, 8192U); |
| 139 | std::vector<uint8_t> data(dwEstimateSize); |
| tsepez | 06104a8 | 2016-11-21 16:22:10 -0800 | [diff] [blame] | 140 | if (!pFile->ReadBlock(data.data(), 0, dwEstimateSize)) |
| 141 | return; |
| 142 | |
| Tom Sepez | 126927e | 2018-08-28 22:25:49 +0000 | [diff] [blame] | 143 | std::unique_ptr<CPDF_Dictionary> pDict = InitJPEG(data); |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 144 | if (!pDict && size > dwEstimateSize) { |
| thestig | dc359b0 | 2016-08-09 15:46:20 -0700 | [diff] [blame] | 145 | data.resize(size); |
| 146 | pFile->ReadBlock(data.data(), 0, size); |
| Tom Sepez | 126927e | 2018-08-28 22:25:49 +0000 | [diff] [blame] | 147 | pDict = InitJPEG(data); |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 148 | } |
| thestig | dc359b0 | 2016-08-09 15:46:20 -0700 | [diff] [blame] | 149 | if (!pDict) |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 150 | return; |
| thestig | dc359b0 | 2016-08-09 15:46:20 -0700 | [diff] [blame] | 151 | |
| tsepez | 06104a8 | 2016-11-21 16:22:10 -0800 | [diff] [blame] | 152 | m_pStream->InitStreamFromFile(pFile, std::move(pDict)); |
| John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 153 | } |
| dan sinclair | 61b2fc7 | 2016-03-23 19:21:44 -0400 | [diff] [blame] | 154 | |
| rbpotter | f085db3 | 2016-12-14 11:44:31 -0800 | [diff] [blame] | 155 | void CPDF_Image::SetJpegImageInline( |
| Dan Sinclair | 0b95042 | 2017-09-21 15:49:49 -0400 | [diff] [blame] | 156 | const RetainPtr<IFX_SeekableReadStream>& pFile) { |
| rbpotter | f085db3 | 2016-12-14 11:44:31 -0800 | [diff] [blame] | 157 | uint32_t size = pdfium::base::checked_cast<uint32_t>(pFile->GetSize()); |
| 158 | if (!size) |
| 159 | return; |
| 160 | |
| 161 | std::vector<uint8_t> data(size); |
| 162 | if (!pFile->ReadBlock(data.data(), 0, size)) |
| 163 | return; |
| 164 | |
| Tom Sepez | 126927e | 2018-08-28 22:25:49 +0000 | [diff] [blame] | 165 | std::unique_ptr<CPDF_Dictionary> pDict = InitJPEG(data); |
| rbpotter | f085db3 | 2016-12-14 11:44:31 -0800 | [diff] [blame] | 166 | if (!pDict) |
| 167 | return; |
| 168 | |
| Tom Sepez | 367ed46 | 2018-08-23 23:52:53 +0000 | [diff] [blame] | 169 | m_pStream->InitStream(data, std::move(pDict)); |
| rbpotter | f085db3 | 2016-12-14 11:44:31 -0800 | [diff] [blame] | 170 | } |
| 171 | |
| Dan Sinclair | 0b95042 | 2017-09-21 15:49:49 -0400 | [diff] [blame] | 172 | void CPDF_Image::SetImage(const RetainPtr<CFX_DIBitmap>& pBitmap) { |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 173 | int32_t BitmapWidth = pBitmap->GetWidth(); |
| 174 | int32_t BitmapHeight = pBitmap->GetHeight(); |
| thestig | 4ccdb14 | 2016-11-21 15:09:23 -0800 | [diff] [blame] | 175 | if (BitmapWidth < 1 || BitmapHeight < 1) |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 176 | return; |
| dan sinclair | 61b2fc7 | 2016-03-23 19:21:44 -0400 | [diff] [blame] | 177 | |
| tsepez | 9e05ee1 | 2016-11-21 13:19:10 -0800 | [diff] [blame] | 178 | auto pDict = |
| 179 | pdfium::MakeUnique<CPDF_Dictionary>(m_pDocument->GetByteStringPool()); |
| tsepez | 0e606b5 | 2016-11-18 16:22:41 -0800 | [diff] [blame] | 180 | pDict->SetNewFor<CPDF_Name>("Type", "XObject"); |
| 181 | pDict->SetNewFor<CPDF_Name>("Subtype", "Image"); |
| 182 | pDict->SetNewFor<CPDF_Number>("Width", BitmapWidth); |
| 183 | pDict->SetNewFor<CPDF_Number>("Height", BitmapHeight); |
| thestig | 4ccdb14 | 2016-11-21 15:09:23 -0800 | [diff] [blame] | 184 | |
| 185 | const int32_t bpp = pBitmap->GetBPP(); |
| Ryan Harrison | 875e98c | 2017-09-27 10:53:11 -0400 | [diff] [blame] | 186 | size_t dest_pitch = 0; |
| thestig | 4ccdb14 | 2016-11-21 15:09:23 -0800 | [diff] [blame] | 187 | bool bCopyWithoutAlpha = true; |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 188 | if (bpp == 1) { |
| thestig | 4ccdb14 | 2016-11-21 15:09:23 -0800 | [diff] [blame] | 189 | int32_t reset_a = 0; |
| 190 | int32_t reset_r = 0; |
| 191 | int32_t reset_g = 0; |
| 192 | int32_t reset_b = 0; |
| 193 | int32_t set_a = 0; |
| 194 | int32_t set_r = 0; |
| 195 | int32_t set_g = 0; |
| 196 | int32_t set_b = 0; |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 197 | if (!pBitmap->IsAlphaMask()) { |
| Nicolas Pena | ddfc3dc | 2017-04-20 15:29:25 -0400 | [diff] [blame] | 198 | std::tie(reset_a, reset_r, reset_g, reset_b) = |
| 199 | ArgbDecode(pBitmap->GetPaletteArgb(0)); |
| 200 | std::tie(set_a, set_r, set_g, set_b) = |
| 201 | ArgbDecode(pBitmap->GetPaletteArgb(1)); |
| John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 202 | } |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 203 | if (set_a == 0 || reset_a == 0) { |
| tsepez | 0e606b5 | 2016-11-18 16:22:41 -0800 | [diff] [blame] | 204 | pDict->SetNewFor<CPDF_Boolean>("ImageMask", true); |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 205 | if (reset_a == 0) { |
| tsepez | 0e606b5 | 2016-11-18 16:22:41 -0800 | [diff] [blame] | 206 | CPDF_Array* pArray = pDict->SetNewFor<CPDF_Array>("Decode"); |
| tsepez | 8a3aa45 | 2016-11-16 12:26:06 -0800 | [diff] [blame] | 207 | pArray->AddNew<CPDF_Number>(1); |
| 208 | pArray->AddNew<CPDF_Number>(0); |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 209 | } |
| John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 210 | } else { |
| tsepez | 0e606b5 | 2016-11-18 16:22:41 -0800 | [diff] [blame] | 211 | CPDF_Array* pCS = pDict->SetNewFor<CPDF_Array>("ColorSpace"); |
| tsepez | 8a3aa45 | 2016-11-16 12:26:06 -0800 | [diff] [blame] | 212 | pCS->AddNew<CPDF_Name>("Indexed"); |
| 213 | pCS->AddNew<CPDF_Name>("DeviceRGB"); |
| 214 | pCS->AddNew<CPDF_Number>(1); |
| Ryan Harrison | 275e260 | 2017-09-18 14:23:18 -0400 | [diff] [blame] | 215 | ByteString ct; |
| Tom Sepez | 1dbfe99 | 2018-04-17 17:19:30 +0000 | [diff] [blame] | 216 | { |
| 217 | // Span's lifetime must end before ReleaseBuffer() below. |
| 218 | pdfium::span<char> pBuf = ct.GetBuffer(6); |
| 219 | pBuf[0] = (char)reset_r; |
| 220 | pBuf[1] = (char)reset_g; |
| 221 | pBuf[2] = (char)reset_b; |
| 222 | pBuf[3] = (char)set_r; |
| 223 | pBuf[4] = (char)set_g; |
| 224 | pBuf[5] = (char)set_b; |
| 225 | } |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 226 | ct.ReleaseBuffer(6); |
| tsepez | 8a3aa45 | 2016-11-16 12:26:06 -0800 | [diff] [blame] | 227 | pCS->AddNew<CPDF_String>(ct, true); |
| John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 228 | } |
| tsepez | 0e606b5 | 2016-11-18 16:22:41 -0800 | [diff] [blame] | 229 | pDict->SetNewFor<CPDF_Number>("BitsPerComponent", 1); |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 230 | dest_pitch = (BitmapWidth + 7) / 8; |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 231 | } else if (bpp == 8) { |
| 232 | int32_t iPalette = pBitmap->GetPaletteSize(); |
| 233 | if (iPalette > 0) { |
| tsepez | 70c4afd | 2016-11-15 11:33:44 -0800 | [diff] [blame] | 234 | CPDF_Array* pCS = m_pDocument->NewIndirect<CPDF_Array>(); |
| tsepez | 8a3aa45 | 2016-11-16 12:26:06 -0800 | [diff] [blame] | 235 | pCS->AddNew<CPDF_Name>("Indexed"); |
| 236 | pCS->AddNew<CPDF_Name>("DeviceRGB"); |
| 237 | pCS->AddNew<CPDF_Number>(iPalette - 1); |
| tsepez | 47fb8c0 | 2016-12-15 13:51:34 -0800 | [diff] [blame] | 238 | std::unique_ptr<uint8_t, FxFreeDeleter> pColorTable( |
| 239 | FX_Alloc2D(uint8_t, iPalette, 3)); |
| 240 | uint8_t* ptr = pColorTable.get(); |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 241 | for (int32_t i = 0; i < iPalette; i++) { |
| tsepez | b5e8f14 | 2016-03-25 15:18:35 -0700 | [diff] [blame] | 242 | uint32_t argb = pBitmap->GetPaletteArgb(i); |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 243 | ptr[0] = (uint8_t)(argb >> 16); |
| 244 | ptr[1] = (uint8_t)(argb >> 8); |
| 245 | ptr[2] = (uint8_t)argb; |
| 246 | ptr += 3; |
| 247 | } |
| tsepez | 9e05ee1 | 2016-11-21 13:19:10 -0800 | [diff] [blame] | 248 | auto pNewDict = |
| 249 | pdfium::MakeUnique<CPDF_Dictionary>(m_pDocument->GetByteStringPool()); |
| tsepez | 70c4afd | 2016-11-15 11:33:44 -0800 | [diff] [blame] | 250 | CPDF_Stream* pCTS = m_pDocument->NewIndirect<CPDF_Stream>( |
| tsepez | 47fb8c0 | 2016-12-15 13:51:34 -0800 | [diff] [blame] | 251 | std::move(pColorTable), iPalette * 3, std::move(pNewDict)); |
| Artem Strygin | fb72726 | 2018-06-11 18:19:57 +0000 | [diff] [blame] | 252 | pCS->Add(pCTS->MakeReference(m_pDocument.Get())); |
| 253 | pDict->SetFor("ColorSpace", pCS->MakeReference(m_pDocument.Get())); |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 254 | } else { |
| tsepez | 0e606b5 | 2016-11-18 16:22:41 -0800 | [diff] [blame] | 255 | pDict->SetNewFor<CPDF_Name>("ColorSpace", "DeviceGray"); |
| John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 256 | } |
| tsepez | 0e606b5 | 2016-11-18 16:22:41 -0800 | [diff] [blame] | 257 | pDict->SetNewFor<CPDF_Number>("BitsPerComponent", 8); |
| thestig | 4ccdb14 | 2016-11-21 15:09:23 -0800 | [diff] [blame] | 258 | dest_pitch = BitmapWidth; |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 259 | } else { |
| tsepez | 0e606b5 | 2016-11-18 16:22:41 -0800 | [diff] [blame] | 260 | pDict->SetNewFor<CPDF_Name>("ColorSpace", "DeviceRGB"); |
| 261 | pDict->SetNewFor<CPDF_Number>("BitsPerComponent", 8); |
| thestig | 4ccdb14 | 2016-11-21 15:09:23 -0800 | [diff] [blame] | 262 | dest_pitch = BitmapWidth * 3; |
| 263 | bCopyWithoutAlpha = false; |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 264 | } |
| thestig | 4ccdb14 | 2016-11-21 15:09:23 -0800 | [diff] [blame] | 265 | |
| Dan Sinclair | 0b95042 | 2017-09-21 15:49:49 -0400 | [diff] [blame] | 266 | RetainPtr<CFX_DIBitmap> pMaskBitmap; |
| tsepez | 72c1bda | 2016-12-14 14:15:14 -0800 | [diff] [blame] | 267 | if (pBitmap->HasAlpha()) |
| 268 | pMaskBitmap = pBitmap->CloneAlphaMask(); |
| 269 | |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 270 | if (pMaskBitmap) { |
| 271 | int32_t maskWidth = pMaskBitmap->GetWidth(); |
| 272 | int32_t maskHeight = pMaskBitmap->GetHeight(); |
| tsepez | 47fb8c0 | 2016-12-15 13:51:34 -0800 | [diff] [blame] | 273 | std::unique_ptr<uint8_t, FxFreeDeleter> mask_buf; |
| Ryan Harrison | aa3a9cd | 2017-08-29 16:39:44 -0400 | [diff] [blame] | 274 | int32_t mask_size = 0; |
| tsepez | 9e05ee1 | 2016-11-21 13:19:10 -0800 | [diff] [blame] | 275 | auto pMaskDict = |
| 276 | pdfium::MakeUnique<CPDF_Dictionary>(m_pDocument->GetByteStringPool()); |
| tsepez | 0e606b5 | 2016-11-18 16:22:41 -0800 | [diff] [blame] | 277 | pMaskDict->SetNewFor<CPDF_Name>("Type", "XObject"); |
| 278 | pMaskDict->SetNewFor<CPDF_Name>("Subtype", "Image"); |
| 279 | pMaskDict->SetNewFor<CPDF_Number>("Width", maskWidth); |
| 280 | pMaskDict->SetNewFor<CPDF_Number>("Height", maskHeight); |
| 281 | pMaskDict->SetNewFor<CPDF_Name>("ColorSpace", "DeviceGray"); |
| 282 | pMaskDict->SetNewFor<CPDF_Number>("BitsPerComponent", 8); |
| thestig | 4ccdb14 | 2016-11-21 15:09:23 -0800 | [diff] [blame] | 283 | if (pMaskBitmap->GetFormat() != FXDIB_1bppMask) { |
| tsepez | 47fb8c0 | 2016-12-15 13:51:34 -0800 | [diff] [blame] | 284 | mask_buf.reset(FX_Alloc2D(uint8_t, maskHeight, maskWidth)); |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 285 | mask_size = maskHeight * maskWidth; // Safe since checked alloc returned. |
| 286 | for (int32_t a = 0; a < maskHeight; a++) { |
| Dan Sinclair | 1c5d0b4 | 2017-04-03 15:05:11 -0400 | [diff] [blame] | 287 | memcpy(mask_buf.get() + a * maskWidth, pMaskBitmap->GetScanline(a), |
| 288 | maskWidth); |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 289 | } |
| 290 | } |
| tsepez | 0e606b5 | 2016-11-18 16:22:41 -0800 | [diff] [blame] | 291 | pMaskDict->SetNewFor<CPDF_Number>("Length", mask_size); |
| tsepez | 9e05ee1 | 2016-11-21 13:19:10 -0800 | [diff] [blame] | 292 | CPDF_Stream* pNewStream = m_pDocument->NewIndirect<CPDF_Stream>( |
| tsepez | 47fb8c0 | 2016-12-15 13:51:34 -0800 | [diff] [blame] | 293 | std::move(mask_buf), mask_size, std::move(pMaskDict)); |
| Artem Strygin | fb72726 | 2018-06-11 18:19:57 +0000 | [diff] [blame] | 294 | pDict->SetFor("SMask", pNewStream->MakeReference(m_pDocument.Get())); |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 295 | } |
| dsinclair | 65ea394 | 2016-03-25 09:16:34 -0700 | [diff] [blame] | 296 | |
| thestig | 4ccdb14 | 2016-11-21 15:09:23 -0800 | [diff] [blame] | 297 | uint8_t* src_buf = pBitmap->GetBuffer(); |
| 298 | int32_t src_pitch = pBitmap->GetPitch(); |
| 299 | uint8_t* dest_buf = FX_Alloc2D(uint8_t, dest_pitch, BitmapHeight); |
| 300 | // Safe as checked alloc returned. |
| Ryan Harrison | 875e98c | 2017-09-27 10:53:11 -0400 | [diff] [blame] | 301 | size_t dest_size = dest_pitch * BitmapHeight; |
| thestig | 4ccdb14 | 2016-11-21 15:09:23 -0800 | [diff] [blame] | 302 | uint8_t* pDest = dest_buf; |
| 303 | if (bCopyWithoutAlpha) { |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 304 | for (int32_t i = 0; i < BitmapHeight; i++) { |
| Dan Sinclair | 1c5d0b4 | 2017-04-03 15:05:11 -0400 | [diff] [blame] | 305 | memcpy(pDest, src_buf, dest_pitch); |
| dsinclair | 65ea394 | 2016-03-25 09:16:34 -0700 | [diff] [blame] | 306 | pDest += dest_pitch; |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 307 | src_buf += src_pitch; |
| 308 | } |
| thestig | 4ccdb14 | 2016-11-21 15:09:23 -0800 | [diff] [blame] | 309 | } else { |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 310 | int32_t src_offset = 0; |
| 311 | int32_t dest_offset = 0; |
| 312 | for (int32_t row = 0; row < BitmapHeight; row++) { |
| 313 | src_offset = row * src_pitch; |
| 314 | for (int32_t column = 0; column < BitmapWidth; column++) { |
| Dan Sinclair | 05df075 | 2017-03-14 14:43:42 -0400 | [diff] [blame] | 315 | float alpha = 1; |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 316 | pDest[dest_offset] = (uint8_t)(src_buf[src_offset + 2] * alpha); |
| 317 | pDest[dest_offset + 1] = (uint8_t)(src_buf[src_offset + 1] * alpha); |
| 318 | pDest[dest_offset + 2] = (uint8_t)(src_buf[src_offset] * alpha); |
| 319 | dest_offset += 3; |
| 320 | src_offset += bpp == 24 ? 3 : 4; |
| 321 | } |
| dsinclair | 65ea394 | 2016-03-25 09:16:34 -0700 | [diff] [blame] | 322 | |
| 323 | pDest += dest_pitch; |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 324 | dest_offset = 0; |
| John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 325 | } |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 326 | } |
| tsepez | 9fd0c63 | 2016-11-23 14:34:58 -0800 | [diff] [blame] | 327 | if (!m_pStream) |
| 328 | m_pStream = pdfium::MakeUnique<CPDF_Stream>(); |
| 329 | |
| Tom Sepez | 367ed46 | 2018-08-23 23:52:53 +0000 | [diff] [blame] | 330 | m_pStream->InitStream({dest_buf, dest_size}, std::move(pDict)); |
| Nico Weber | 9d8ec5a | 2015-08-04 13:00:21 -0700 | [diff] [blame] | 331 | m_bIsMask = pBitmap->IsAlphaMask(); |
| 332 | m_Width = BitmapWidth; |
| 333 | m_Height = BitmapHeight; |
| Lei Zhang | da180e9 | 2015-08-14 22:22:13 -0700 | [diff] [blame] | 334 | FX_Free(dest_buf); |
| John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 335 | } |
| dan sinclair | 61b2fc7 | 2016-03-23 19:21:44 -0400 | [diff] [blame] | 336 | |
| Lei Zhang | 66601c4 | 2018-10-02 18:58:16 +0000 | [diff] [blame] | 337 | void CPDF_Image::ResetCache(CPDF_Page* pPage) { |
| Dan Sinclair | 0b95042 | 2017-09-21 15:49:49 -0400 | [diff] [blame] | 338 | RetainPtr<CPDF_Image> pHolder(this); |
| Lei Zhang | 66601c4 | 2018-10-02 18:58:16 +0000 | [diff] [blame] | 339 | pPage->GetRenderCache()->ResetBitmap(pHolder); |
| John Abd-El-Malek | 3f3b45c | 2014-05-23 17:28:10 -0700 | [diff] [blame] | 340 | } |
| dan sinclair | 61b2fc7 | 2016-03-23 19:21:44 -0400 | [diff] [blame] | 341 | |
| Tom Sepez | e6ff2eb | 2018-08-24 21:55:46 +0000 | [diff] [blame] | 342 | RetainPtr<CFX_DIBBase> CPDF_Image::LoadDIBBase() const { |
| 343 | auto source = pdfium::MakeRetain<CPDF_DIBBase>(); |
| Tom Sepez | 4734512 | 2017-05-24 14:07:18 -0700 | [diff] [blame] | 344 | if (!source->Load(m_pDocument.Get(), m_pStream.Get())) |
| tsepez | 5bed98c | 2016-12-14 13:54:33 -0800 | [diff] [blame] | 345 | return nullptr; |
| Nicolas Pena | 48f776f | 2017-01-05 13:22:17 -0500 | [diff] [blame] | 346 | |
| Lei Zhang | 1330ebb | 2018-03-05 15:16:37 +0000 | [diff] [blame] | 347 | if (!source->IsJBigImage()) |
| 348 | return source; |
| 349 | |
| Tom Sepez | e6ff2eb | 2018-08-24 21:55:46 +0000 | [diff] [blame] | 350 | CPDF_DIBBase::LoadState ret = CPDF_DIBBase::LoadState::kContinue; |
| 351 | while (ret == CPDF_DIBBase::LoadState::kContinue) |
| 352 | ret = source->ContinueLoadDIBBase(nullptr); |
| 353 | return ret == CPDF_DIBBase::LoadState::kSuccess ? source : nullptr; |
| dan sinclair | 61b2fc7 | 2016-03-23 19:21:44 -0400 | [diff] [blame] | 354 | } |
| 355 | |
| Tom Sepez | e6ff2eb | 2018-08-24 21:55:46 +0000 | [diff] [blame] | 356 | RetainPtr<CFX_DIBBase> CPDF_Image::DetachBitmap() { |
| 357 | return std::move(m_pDIBBase); |
| dan sinclair | 61b2fc7 | 2016-03-23 19:21:44 -0400 | [diff] [blame] | 358 | } |
| 359 | |
| Tom Sepez | e6ff2eb | 2018-08-24 21:55:46 +0000 | [diff] [blame] | 360 | RetainPtr<CFX_DIBBase> CPDF_Image::DetachMask() { |
| Tom Sepez | f0799fe | 2017-03-28 09:31:32 -0700 | [diff] [blame] | 361 | return std::move(m_pMask); |
| dan sinclair | 61b2fc7 | 2016-03-23 19:21:44 -0400 | [diff] [blame] | 362 | } |
| 363 | |
| Tom Sepez | e6ff2eb | 2018-08-24 21:55:46 +0000 | [diff] [blame] | 364 | bool CPDF_Image::StartLoadDIBBase(const CPDF_Dictionary* pFormResource, |
| 365 | CPDF_Dictionary* pPageResource, |
| 366 | bool bStdCS, |
| 367 | uint32_t GroupFamily, |
| 368 | bool bLoadMask) { |
| 369 | auto source = pdfium::MakeRetain<CPDF_DIBBase>(); |
| 370 | CPDF_DIBBase::LoadState ret = source->StartLoadDIBBase( |
| Lei Zhang | 40482e6 | 2018-03-05 15:03:37 +0000 | [diff] [blame] | 371 | m_pDocument.Get(), m_pStream.Get(), true, pFormResource, pPageResource, |
| 372 | bStdCS, GroupFamily, bLoadMask); |
| Tom Sepez | e6ff2eb | 2018-08-24 21:55:46 +0000 | [diff] [blame] | 373 | if (ret == CPDF_DIBBase::LoadState::kFail) { |
| 374 | m_pDIBBase.Reset(); |
| tsepez | 12f3e4a | 2016-11-02 15:17:29 -0700 | [diff] [blame] | 375 | return false; |
| dan sinclair | 61b2fc7 | 2016-03-23 19:21:44 -0400 | [diff] [blame] | 376 | } |
| Tom Sepez | e6ff2eb | 2018-08-24 21:55:46 +0000 | [diff] [blame] | 377 | m_pDIBBase = source; |
| 378 | if (ret == CPDF_DIBBase::LoadState::kContinue) |
| Tom Sepez | f0799fe | 2017-03-28 09:31:32 -0700 | [diff] [blame] | 379 | return true; |
| 380 | |
| dan sinclair | 61b2fc7 | 2016-03-23 19:21:44 -0400 | [diff] [blame] | 381 | m_pMask = source->DetachMask(); |
| 382 | m_MatteColor = source->GetMatteColor(); |
| tsepez | 12f3e4a | 2016-11-02 15:17:29 -0700 | [diff] [blame] | 383 | return false; |
| dan sinclair | 61b2fc7 | 2016-03-23 19:21:44 -0400 | [diff] [blame] | 384 | } |
| 385 | |
| Dan Sinclair | a32145f | 2018-03-06 18:53:05 +0000 | [diff] [blame] | 386 | bool CPDF_Image::Continue(PauseIndicatorIface* pPause) { |
| Tom Sepez | e6ff2eb | 2018-08-24 21:55:46 +0000 | [diff] [blame] | 387 | RetainPtr<CPDF_DIBBase> pSource = m_pDIBBase.As<CPDF_DIBBase>(); |
| 388 | CPDF_DIBBase::LoadState ret = pSource->ContinueLoadDIBBase(pPause); |
| 389 | if (ret == CPDF_DIBBase::LoadState::kContinue) |
| Tom Sepez | f0799fe | 2017-03-28 09:31:32 -0700 | [diff] [blame] | 390 | return true; |
| 391 | |
| Tom Sepez | e6ff2eb | 2018-08-24 21:55:46 +0000 | [diff] [blame] | 392 | if (ret == CPDF_DIBBase::LoadState::kSuccess) { |
| Lei Zhang | b058365 | 2018-03-05 14:46:57 +0000 | [diff] [blame] | 393 | m_pMask = pSource->DetachMask(); |
| 394 | m_MatteColor = pSource->GetMatteColor(); |
| 395 | } else { |
| Tom Sepez | e6ff2eb | 2018-08-24 21:55:46 +0000 | [diff] [blame] | 396 | m_pDIBBase.Reset(); |
| Lei Zhang | b058365 | 2018-03-05 14:46:57 +0000 | [diff] [blame] | 397 | } |
| tsepez | 12f3e4a | 2016-11-02 15:17:29 -0700 | [diff] [blame] | 398 | return false; |
| dan sinclair | 61b2fc7 | 2016-03-23 19:21:44 -0400 | [diff] [blame] | 399 | } |