blob: 0ca255173569d0fc6579207b0b8e4169bf820014 [file] [log] [blame]
epoger@google.comec3ed6a2011-07-28 14:26:00 +00001/*
2 * Copyright 2006 The Android Open Source Project
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
7
Hal Canarydb683012016-11-23 08:55:18 -07008#include "SkImageEncoderPriv.h"
Hal Canary1fcc4042016-11-30 17:07:59 -05009
10#ifdef SK_HAS_PNG_LIBRARY
11
Mike Reed086a4272017-07-18 10:53:11 -040012#include "SkColorTable.h"
Matt Sarett5df93de2017-03-22 21:52:47 +000013#include "SkImageEncoderFns.h"
Matt Sarettc367d032017-05-05 11:13:26 -040014#include "SkImageInfoPriv.h"
reed@android.com8a1c16f2008-12-17 15:59:43 +000015#include "SkStream.h"
Matt Sarett74de1f62017-03-21 13:03:26 -040016#include "SkString.h"
Matt Sarettc367d032017-05-05 11:13:26 -040017#include "SkPngEncoder.h"
Mike Reedd6cb11e2017-11-30 15:33:04 -050018#include "SkPngPriv.h"
djsollenb2a6fe72015-04-03 12:35:27 -070019
reed@android.com8a1c16f2008-12-17 15:59:43 +000020#include "png.h"
reed@android.com8a1c16f2008-12-17 15:59:43 +000021
Matt Sarettbe4c9b02017-05-08 12:11:44 -040022static_assert(PNG_FILTER_NONE == (int)SkPngEncoder::FilterFlag::kNone, "Skia libpng filter err.");
23static_assert(PNG_FILTER_SUB == (int)SkPngEncoder::FilterFlag::kSub, "Skia libpng filter err.");
24static_assert(PNG_FILTER_UP == (int)SkPngEncoder::FilterFlag::kUp, "Skia libpng filter err.");
25static_assert(PNG_FILTER_AVG == (int)SkPngEncoder::FilterFlag::kAvg, "Skia libpng filter err.");
26static_assert(PNG_FILTER_PAETH == (int)SkPngEncoder::FilterFlag::kPaeth, "Skia libpng filter err.");
27static_assert(PNG_ALL_FILTERS == (int)SkPngEncoder::FilterFlag::kAll, "Skia libpng filter err.");
28
Matt Sarettc367d032017-05-05 11:13:26 -040029static constexpr bool kSuppressPngEncodeWarnings = true;
commit-bot@chromium.orga936e372013-03-14 14:42:18 +000030
reed@android.com8a1c16f2008-12-17 15:59:43 +000031static void sk_error_fn(png_structp png_ptr, png_const_charp msg) {
Matt Sarettc367d032017-05-05 11:13:26 -040032 if (!kSuppressPngEncodeWarnings) {
33 SkDebugf("libpng encode error: %s\n", msg);
scroggo04a0d3f2015-12-10 08:54:36 -080034 }
Matt Sarettc367d032017-05-05 11:13:26 -040035
reed@android.com8a1c16f2008-12-17 15:59:43 +000036 longjmp(png_jmpbuf(png_ptr), 1);
37}
38
reed@android.com8a1c16f2008-12-17 15:59:43 +000039static void sk_write_fn(png_structp png_ptr, png_bytep data, png_size_t len) {
Matt Sarettc367d032017-05-05 11:13:26 -040040 SkWStream* stream = (SkWStream*)png_get_io_ptr(png_ptr);
41 if (!stream->write(data, len)) {
42 png_error(png_ptr, "sk_write_fn cannot write to stream");
reed@android.com8a1c16f2008-12-17 15:59:43 +000043 }
44}
45
Matt Sarettc367d032017-05-05 11:13:26 -040046class SkPngEncoderMgr final : SkNoncopyable {
47public:
48
49 /*
50 * Create the decode manager
51 * Does not take ownership of stream
52 */
53 static std::unique_ptr<SkPngEncoderMgr> Make(SkWStream* stream);
54
Matt Sarettbe4c9b02017-05-08 12:11:44 -040055 bool setHeader(const SkImageInfo& srcInfo, const SkPngEncoder::Options& options);
Matt Sarett1950e0a2017-06-12 16:17:30 -040056 bool setColorSpace(const SkImageInfo& info);
Matt Sarettc367d032017-05-05 11:13:26 -040057 bool writeInfo(const SkImageInfo& srcInfo);
Brian Osmanb62f50c2018-07-12 14:44:27 -040058 void chooseProc(const SkImageInfo& srcInfo);
Matt Sarettc367d032017-05-05 11:13:26 -040059
60 png_structp pngPtr() { return fPngPtr; }
61 png_infop infoPtr() { return fInfoPtr; }
62 int pngBytesPerPixel() const { return fPngBytesPerPixel; }
63 transform_scanline_proc proc() const { return fProc; }
64
65 ~SkPngEncoderMgr() {
66 png_destroy_write_struct(&fPngPtr, &fInfoPtr);
67 }
68
69private:
70
71 SkPngEncoderMgr(png_structp pngPtr, png_infop infoPtr)
72 : fPngPtr(pngPtr)
73 , fInfoPtr(infoPtr)
74 {}
75
76 png_structp fPngPtr;
77 png_infop fInfoPtr;
78 int fPngBytesPerPixel;
79 transform_scanline_proc fProc;
80};
81
82std::unique_ptr<SkPngEncoderMgr> SkPngEncoderMgr::Make(SkWStream* stream) {
83 png_structp pngPtr =
84 png_create_write_struct(PNG_LIBPNG_VER_STRING, nullptr, sk_error_fn, nullptr);
85 if (!pngPtr) {
86 return nullptr;
87 }
88
89 png_infop infoPtr = png_create_info_struct(pngPtr);
90 if (!infoPtr) {
91 png_destroy_write_struct(&pngPtr, nullptr);
92 return nullptr;
93 }
94
95 png_set_write_fn(pngPtr, (void*)stream, sk_write_fn, nullptr);
96 return std::unique_ptr<SkPngEncoderMgr>(new SkPngEncoderMgr(pngPtr, infoPtr));
97}
98
Matt Sarettbe4c9b02017-05-08 12:11:44 -040099bool SkPngEncoderMgr::setHeader(const SkImageInfo& srcInfo, const SkPngEncoder::Options& options) {
Matt Sarettc367d032017-05-05 11:13:26 -0400100 if (setjmp(png_jmpbuf(fPngPtr))) {
101 return false;
102 }
103
104 int pngColorType;
105 png_color_8 sigBit;
106 int bitDepth = 8;
107 switch (srcInfo.colorType()) {
108 case kRGBA_F16_SkColorType:
Mike Klein37854712018-06-26 11:43:06 -0400109 case kRGBA_F32_SkColorType:
Mike Kleince4cf722018-05-10 11:29:15 -0400110 SkASSERT(srcInfo.colorSpace());
Matt Sarettc367d032017-05-05 11:13:26 -0400111 sigBit.red = 16;
112 sigBit.green = 16;
113 sigBit.blue = 16;
114 sigBit.alpha = 16;
115 bitDepth = 16;
116 pngColorType = srcInfo.isOpaque() ? PNG_COLOR_TYPE_RGB : PNG_COLOR_TYPE_RGB_ALPHA;
117 fPngBytesPerPixel = 8;
118 break;
Matt Sarettc367d032017-05-05 11:13:26 -0400119 case kGray_8_SkColorType:
120 sigBit.gray = 8;
121 pngColorType = PNG_COLOR_TYPE_GRAY;
122 fPngBytesPerPixel = 1;
123 SkASSERT(srcInfo.isOpaque());
124 break;
125 case kRGBA_8888_SkColorType:
126 case kBGRA_8888_SkColorType:
127 sigBit.red = 8;
128 sigBit.green = 8;
129 sigBit.blue = 8;
130 sigBit.alpha = 8;
131 pngColorType = srcInfo.isOpaque() ? PNG_COLOR_TYPE_RGB : PNG_COLOR_TYPE_RGB_ALPHA;
132 fPngBytesPerPixel = srcInfo.isOpaque() ? 3 : 4;
133 break;
Mike Kleinac568a92018-01-25 09:09:32 -0500134 case kRGB_888x_SkColorType:
135 sigBit.red = 8;
136 sigBit.green = 8;
137 sigBit.blue = 8;
138 pngColorType = PNG_COLOR_TYPE_RGB;
139 fPngBytesPerPixel = 3;
140 SkASSERT(srcInfo.isOpaque());
141 break;
Matt Sarettc367d032017-05-05 11:13:26 -0400142 case kARGB_4444_SkColorType:
143 if (kUnpremul_SkAlphaType == srcInfo.alphaType()) {
144 return false;
145 }
146
147 sigBit.red = 4;
148 sigBit.green = 4;
149 sigBit.blue = 4;
150 sigBit.alpha = 4;
151 pngColorType = srcInfo.isOpaque() ? PNG_COLOR_TYPE_RGB : PNG_COLOR_TYPE_RGB_ALPHA;
152 fPngBytesPerPixel = srcInfo.isOpaque() ? 3 : 4;
153 break;
154 case kRGB_565_SkColorType:
155 sigBit.red = 5;
156 sigBit.green = 6;
157 sigBit.blue = 5;
158 pngColorType = PNG_COLOR_TYPE_RGB;
159 fPngBytesPerPixel = 3;
160 SkASSERT(srcInfo.isOpaque());
161 break;
Mike Reedd6cb11e2017-11-30 15:33:04 -0500162 case kAlpha_8_SkColorType: // store as gray+alpha, but ignore gray
163 sigBit.gray = kGraySigBit_GrayAlphaIsJustAlpha;
164 sigBit.alpha = 8;
165 pngColorType = PNG_COLOR_TYPE_GRAY_ALPHA;
166 fPngBytesPerPixel = 2;
167 break;
Mike Kleinac568a92018-01-25 09:09:32 -0500168 case kRGBA_1010102_SkColorType:
169 bitDepth = 16;
170 sigBit.red = 10;
171 sigBit.green = 10;
172 sigBit.blue = 10;
173 sigBit.alpha = 2;
174 pngColorType = srcInfo.isOpaque() ? PNG_COLOR_TYPE_RGB : PNG_COLOR_TYPE_RGB_ALPHA;
175 fPngBytesPerPixel = 8;
176 break;
177 case kRGB_101010x_SkColorType:
178 bitDepth = 16;
179 sigBit.red = 10;
180 sigBit.green = 10;
181 sigBit.blue = 10;
182 pngColorType = PNG_COLOR_TYPE_RGB;
183 fPngBytesPerPixel = 6;
184 break;
Matt Sarettc367d032017-05-05 11:13:26 -0400185 default:
186 return false;
187 }
188
189 png_set_IHDR(fPngPtr, fInfoPtr, srcInfo.width(), srcInfo.height(),
190 bitDepth, pngColorType,
191 PNG_INTERLACE_NONE, PNG_COMPRESSION_TYPE_BASE,
192 PNG_FILTER_TYPE_BASE);
193 png_set_sBIT(fPngPtr, fInfoPtr, &sigBit);
194
Matt Sarettbe4c9b02017-05-08 12:11:44 -0400195 int filters = (int)options.fFilterFlags & (int)SkPngEncoder::FilterFlag::kAll;
196 SkASSERT(filters == (int)options.fFilterFlags);
197 png_set_filter(fPngPtr, PNG_FILTER_TYPE_BASE, filters);
198
199 int zlibLevel = SkTMin(SkTMax(0, options.fZLibLevel), 9);
200 SkASSERT(zlibLevel == options.fZLibLevel);
201 png_set_compression_level(fPngPtr, zlibLevel);
Yuqian Lid0dbee62017-06-09 11:35:58 -0400202
203 // Set comments in tEXt chunk
204 const sk_sp<SkDataTable>& comments = options.fComments;
205 if (comments != nullptr) {
206 std::vector<png_text> png_texts(comments->count());
207 std::vector<SkString> clippedKeys;
208 for (int i = 0; i < comments->count() / 2; ++i) {
209 const char* keyword;
210 const char* originalKeyword = comments->atStr(2 * i);
211 const char* text = comments->atStr(2 * i + 1);
212 if (strlen(originalKeyword) <= PNG_KEYWORD_MAX_LENGTH) {
213 keyword = originalKeyword;
214 } else {
215 SkDEBUGFAILF("PNG tEXt keyword should be no longer than %d.",
216 PNG_KEYWORD_MAX_LENGTH);
217 clippedKeys.emplace_back(originalKeyword, PNG_KEYWORD_MAX_LENGTH);
218 keyword = clippedKeys.back().c_str();
219 }
220 // It seems safe to convert png_const_charp to png_charp for key/text,
221 // and we don't have to provide text_length and other fields as we're providing
222 // 0-terminated c_str with PNG_TEXT_COMPRESSION_NONE (no compression, no itxt).
223 png_texts[i].compression = PNG_TEXT_COMPRESSION_NONE;
224 png_texts[i].key = (png_charp)keyword;
225 png_texts[i].text = (png_charp)text;
226 }
227 png_set_text(fPngPtr, fInfoPtr, png_texts.data(), png_texts.size());
228 }
229
Matt Sarettc367d032017-05-05 11:13:26 -0400230 return true;
Matt Sarett687b6562017-03-21 10:06:45 -0400231}
232
Brian Osmanb62f50c2018-07-12 14:44:27 -0400233static transform_scanline_proc choose_proc(const SkImageInfo& info) {
Matt Sarett84014f02017-01-10 11:28:54 -0500234 switch (info.colorType()) {
235 case kRGBA_8888_SkColorType:
236 switch (info.alphaType()) {
237 case kOpaque_SkAlphaType:
238 return transform_scanline_RGBX;
239 case kUnpremul_SkAlphaType:
240 return transform_scanline_memcpy;
241 case kPremul_SkAlphaType:
Brian Osmanb62f50c2018-07-12 14:44:27 -0400242 return transform_scanline_rgbA;
Matt Sarett84014f02017-01-10 11:28:54 -0500243 default:
244 SkASSERT(false);
245 return nullptr;
246 }
247 case kBGRA_8888_SkColorType:
248 switch (info.alphaType()) {
249 case kOpaque_SkAlphaType:
250 return transform_scanline_BGRX;
251 case kUnpremul_SkAlphaType:
252 return transform_scanline_BGRA;
253 case kPremul_SkAlphaType:
Brian Osmanb62f50c2018-07-12 14:44:27 -0400254 return transform_scanline_bgrA;
Matt Sarett84014f02017-01-10 11:28:54 -0500255 default:
256 SkASSERT(false);
257 return nullptr;
258 }
259 case kRGB_565_SkColorType:
260 return transform_scanline_565;
Mike Kleinac568a92018-01-25 09:09:32 -0500261 case kRGB_888x_SkColorType:
262 return transform_scanline_888x;
Matt Sarett84014f02017-01-10 11:28:54 -0500263 case kARGB_4444_SkColorType:
264 switch (info.alphaType()) {
265 case kOpaque_SkAlphaType:
266 return transform_scanline_444;
267 case kPremul_SkAlphaType:
268 // 4444 is assumed to be legacy premul.
269 return transform_scanline_4444;
270 default:
271 SkASSERT(false);
272 return nullptr;
273 }
Matt Sarett84014f02017-01-10 11:28:54 -0500274 case kGray_8_SkColorType:
275 return transform_scanline_memcpy;
Matt Sarett1da27ef2017-01-19 17:14:07 -0500276 case kRGBA_F16_SkColorType:
277 switch (info.alphaType()) {
278 case kOpaque_SkAlphaType:
279 case kUnpremul_SkAlphaType:
280 return transform_scanline_F16;
281 case kPremul_SkAlphaType:
282 return transform_scanline_F16_premul;
283 default:
284 SkASSERT(false);
285 return nullptr;
286 }
Mike Klein37854712018-06-26 11:43:06 -0400287 case kRGBA_F32_SkColorType:
288 switch (info.alphaType()) {
289 case kOpaque_SkAlphaType:
290 case kUnpremul_SkAlphaType:
291 return transform_scanline_F32;
292 case kPremul_SkAlphaType:
293 return transform_scanline_F32_premul;
294 default:
295 SkASSERT(false);
296 return nullptr;
297 }
Mike Kleinac568a92018-01-25 09:09:32 -0500298 case kRGBA_1010102_SkColorType:
299 switch (info.alphaType()) {
300 case kOpaque_SkAlphaType:
301 case kUnpremul_SkAlphaType:
302 return transform_scanline_1010102;
303 case kPremul_SkAlphaType:
304 return transform_scanline_1010102_premul;
305 default:
306 SkASSERT(false);
307 return nullptr;
308 }
309 case kRGB_101010x_SkColorType:
310 return transform_scanline_101010x;
Mike Reedd6cb11e2017-11-30 15:33:04 -0500311 case kAlpha_8_SkColorType:
312 return transform_scanline_A8_to_GrayAlpha;
Matt Sarett84014f02017-01-10 11:28:54 -0500313 default:
314 SkASSERT(false);
315 return nullptr;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000316 }
reed@android.com8a1c16f2008-12-17 15:59:43 +0000317}
318
Matt Sarett1950e0a2017-06-12 16:17:30 -0400319static void set_icc(png_structp png_ptr, png_infop info_ptr, const SkImageInfo& info) {
320 sk_sp<SkData> icc = icc_from_color_space(info);
Matt Sarettc367d032017-05-05 11:13:26 -0400321 if (!icc) {
322 return;
323 }
324
325#if PNG_LIBPNG_VER_MAJOR > 1 || (PNG_LIBPNG_VER_MAJOR == 1 && PNG_LIBPNG_VER_MINOR >= 5)
326 const char* name = "Skia";
327 png_const_bytep iccPtr = icc->bytes();
328#else
329 SkString str("Skia");
330 char* name = str.writable_str();
331 png_charp iccPtr = (png_charp) icc->writable_data();
332#endif
333 png_set_iCCP(png_ptr, info_ptr, name, 0, iccPtr, icc->size());
334}
335
Matt Sarett1950e0a2017-06-12 16:17:30 -0400336bool SkPngEncoderMgr::setColorSpace(const SkImageInfo& info) {
Matt Sarettc367d032017-05-05 11:13:26 -0400337 if (setjmp(png_jmpbuf(fPngPtr))) {
338 return false;
339 }
340
Matt Sarett1950e0a2017-06-12 16:17:30 -0400341 if (info.colorSpace() && info.colorSpace()->isSRGB()) {
342 png_set_sRGB(fPngPtr, fInfoPtr, PNG_sRGB_INTENT_PERCEPTUAL);
343 } else {
344 set_icc(fPngPtr, fInfoPtr, info);
Matt Sarett0e032be2017-03-15 17:50:08 -0400345 }
346
Matt Sarettc367d032017-05-05 11:13:26 -0400347 return true;
348}
349
350bool SkPngEncoderMgr::writeInfo(const SkImageInfo& srcInfo) {
351 if (setjmp(png_jmpbuf(fPngPtr))) {
352 return false;
353 }
354
355 png_write_info(fPngPtr, fInfoPtr);
356 if (kRGBA_F16_SkColorType == srcInfo.colorType() &&
357 kOpaque_SkAlphaType == srcInfo.alphaType())
358 {
Matt Sarett1da27ef2017-01-19 17:14:07 -0500359 // For kOpaque, kRGBA_F16, we will keep the row as RGBA and tell libpng
360 // to skip the alpha channel.
Matt Sarettc367d032017-05-05 11:13:26 -0400361 png_set_filler(fPngPtr, 0, PNG_FILLER_AFTER);
Matt Sarett1da27ef2017-01-19 17:14:07 -0500362 }
reed@android.com8a1c16f2008-12-17 15:59:43 +0000363
reed@android.com8a1c16f2008-12-17 15:59:43 +0000364 return true;
365}
366
Brian Osmanb62f50c2018-07-12 14:44:27 -0400367void SkPngEncoderMgr::chooseProc(const SkImageInfo& srcInfo) {
368 fProc = choose_proc(srcInfo);
Matt Sarettc367d032017-05-05 11:13:26 -0400369}
370
Matt Sarett6a4dc662017-05-11 09:32:59 -0400371std::unique_ptr<SkEncoder> SkPngEncoder::Make(SkWStream* dst, const SkPixmap& src,
372 const Options& options) {
Brian Osmane1adc3a2018-06-04 09:21:17 -0400373 if (!SkPixmapIsValid(src)) {
Matt Sarettc367d032017-05-05 11:13:26 -0400374 return nullptr;
375 }
376
377 std::unique_ptr<SkPngEncoderMgr> encoderMgr = SkPngEncoderMgr::Make(dst);
378 if (!encoderMgr) {
379 return nullptr;
380 }
381
Matt Sarettbe4c9b02017-05-08 12:11:44 -0400382 if (!encoderMgr->setHeader(src.info(), options)) {
Matt Sarettc367d032017-05-05 11:13:26 -0400383 return nullptr;
384 }
385
Matt Sarett1950e0a2017-06-12 16:17:30 -0400386 if (!encoderMgr->setColorSpace(src.info())) {
Matt Sarettc367d032017-05-05 11:13:26 -0400387 return nullptr;
388 }
389
390 if (!encoderMgr->writeInfo(src.info())) {
391 return nullptr;
392 }
393
Brian Osmanb62f50c2018-07-12 14:44:27 -0400394 encoderMgr->chooseProc(src.info());
Matt Sarettc367d032017-05-05 11:13:26 -0400395
396 return std::unique_ptr<SkPngEncoder>(new SkPngEncoder(std::move(encoderMgr), src));
397}
398
399SkPngEncoder::SkPngEncoder(std::unique_ptr<SkPngEncoderMgr> encoderMgr, const SkPixmap& src)
400 : INHERITED(src, encoderMgr->pngBytesPerPixel() * src.width())
401 , fEncoderMgr(std::move(encoderMgr))
402{}
403
404SkPngEncoder::~SkPngEncoder() {}
405
406bool SkPngEncoder::onEncodeRows(int numRows) {
407 if (setjmp(png_jmpbuf(fEncoderMgr->pngPtr()))) {
408 return false;
409 }
410
411 const void* srcRow = fSrc.addr(0, fCurrRow);
412 for (int y = 0; y < numRows; y++) {
413 fEncoderMgr->proc()((char*) fStorage.get(), (const char*) srcRow, fSrc.width(),
414 SkColorTypeBytesPerPixel(fSrc.colorType()), nullptr);
415
416 png_bytep rowPtr = (png_bytep) fStorage.get();
417 png_write_rows(fEncoderMgr->pngPtr(), &rowPtr, 1);
418 srcRow = SkTAddOffset<const void>(srcRow, fSrc.rowBytes());
419 }
420
421 fCurrRow += numRows;
422 if (fCurrRow == fSrc.height()) {
423 png_write_end(fEncoderMgr->pngPtr(), fEncoderMgr->infoPtr());
424 }
425
426 return true;
427}
428
429bool SkPngEncoder::Encode(SkWStream* dst, const SkPixmap& src, const Options& options) {
430 auto encoder = SkPngEncoder::Make(dst, src, options);
431 return encoder.get() && encoder->encodeRows(src.height());
432}
433
Hal Canary1fcc4042016-11-30 17:07:59 -0500434#endif