blob: c452a712ac778bf1ed837112ab5f9d81a684d6ed [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"
Mike Klein509ccb02018-11-02 12:54:01 -040019#include <vector>
djsollenb2a6fe72015-04-03 12:35:27 -070020
reed@android.com8a1c16f2008-12-17 15:59:43 +000021#include "png.h"
reed@android.com8a1c16f2008-12-17 15:59:43 +000022
Matt Sarettbe4c9b02017-05-08 12:11:44 -040023static_assert(PNG_FILTER_NONE == (int)SkPngEncoder::FilterFlag::kNone, "Skia libpng filter err.");
24static_assert(PNG_FILTER_SUB == (int)SkPngEncoder::FilterFlag::kSub, "Skia libpng filter err.");
25static_assert(PNG_FILTER_UP == (int)SkPngEncoder::FilterFlag::kUp, "Skia libpng filter err.");
26static_assert(PNG_FILTER_AVG == (int)SkPngEncoder::FilterFlag::kAvg, "Skia libpng filter err.");
27static_assert(PNG_FILTER_PAETH == (int)SkPngEncoder::FilterFlag::kPaeth, "Skia libpng filter err.");
28static_assert(PNG_ALL_FILTERS == (int)SkPngEncoder::FilterFlag::kAll, "Skia libpng filter err.");
29
Matt Sarettc367d032017-05-05 11:13:26 -040030static constexpr bool kSuppressPngEncodeWarnings = true;
commit-bot@chromium.orga936e372013-03-14 14:42:18 +000031
reed@android.com8a1c16f2008-12-17 15:59:43 +000032static void sk_error_fn(png_structp png_ptr, png_const_charp msg) {
Matt Sarettc367d032017-05-05 11:13:26 -040033 if (!kSuppressPngEncodeWarnings) {
34 SkDebugf("libpng encode error: %s\n", msg);
scroggo04a0d3f2015-12-10 08:54:36 -080035 }
Matt Sarettc367d032017-05-05 11:13:26 -040036
reed@android.com8a1c16f2008-12-17 15:59:43 +000037 longjmp(png_jmpbuf(png_ptr), 1);
38}
39
reed@android.com8a1c16f2008-12-17 15:59:43 +000040static void sk_write_fn(png_structp png_ptr, png_bytep data, png_size_t len) {
Matt Sarettc367d032017-05-05 11:13:26 -040041 SkWStream* stream = (SkWStream*)png_get_io_ptr(png_ptr);
42 if (!stream->write(data, len)) {
43 png_error(png_ptr, "sk_write_fn cannot write to stream");
reed@android.com8a1c16f2008-12-17 15:59:43 +000044 }
45}
46
Matt Sarettc367d032017-05-05 11:13:26 -040047class SkPngEncoderMgr final : SkNoncopyable {
48public:
49
50 /*
51 * Create the decode manager
52 * Does not take ownership of stream
53 */
54 static std::unique_ptr<SkPngEncoderMgr> Make(SkWStream* stream);
55
Matt Sarettbe4c9b02017-05-08 12:11:44 -040056 bool setHeader(const SkImageInfo& srcInfo, const SkPngEncoder::Options& options);
Matt Sarett1950e0a2017-06-12 16:17:30 -040057 bool setColorSpace(const SkImageInfo& info);
Matt Sarettc367d032017-05-05 11:13:26 -040058 bool writeInfo(const SkImageInfo& srcInfo);
Brian Osmanb62f50c2018-07-12 14:44:27 -040059 void chooseProc(const SkImageInfo& srcInfo);
Matt Sarettc367d032017-05-05 11:13:26 -040060
61 png_structp pngPtr() { return fPngPtr; }
62 png_infop infoPtr() { return fInfoPtr; }
63 int pngBytesPerPixel() const { return fPngBytesPerPixel; }
64 transform_scanline_proc proc() const { return fProc; }
65
66 ~SkPngEncoderMgr() {
67 png_destroy_write_struct(&fPngPtr, &fInfoPtr);
68 }
69
70private:
71
72 SkPngEncoderMgr(png_structp pngPtr, png_infop infoPtr)
73 : fPngPtr(pngPtr)
74 , fInfoPtr(infoPtr)
75 {}
76
77 png_structp fPngPtr;
78 png_infop fInfoPtr;
79 int fPngBytesPerPixel;
80 transform_scanline_proc fProc;
81};
82
83std::unique_ptr<SkPngEncoderMgr> SkPngEncoderMgr::Make(SkWStream* stream) {
84 png_structp pngPtr =
85 png_create_write_struct(PNG_LIBPNG_VER_STRING, nullptr, sk_error_fn, nullptr);
86 if (!pngPtr) {
87 return nullptr;
88 }
89
90 png_infop infoPtr = png_create_info_struct(pngPtr);
91 if (!infoPtr) {
92 png_destroy_write_struct(&pngPtr, nullptr);
93 return nullptr;
94 }
95
96 png_set_write_fn(pngPtr, (void*)stream, sk_write_fn, nullptr);
97 return std::unique_ptr<SkPngEncoderMgr>(new SkPngEncoderMgr(pngPtr, infoPtr));
98}
99
Matt Sarettbe4c9b02017-05-08 12:11:44 -0400100bool SkPngEncoderMgr::setHeader(const SkImageInfo& srcInfo, const SkPngEncoder::Options& options) {
Matt Sarettc367d032017-05-05 11:13:26 -0400101 if (setjmp(png_jmpbuf(fPngPtr))) {
102 return false;
103 }
104
105 int pngColorType;
106 png_color_8 sigBit;
107 int bitDepth = 8;
108 switch (srcInfo.colorType()) {
109 case kRGBA_F16_SkColorType:
Mike Klein37854712018-06-26 11:43:06 -0400110 case kRGBA_F32_SkColorType:
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:
Mike Klein08883cd2018-11-02 13:03:57 -0400262 return transform_scanline_RGBX;
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:
Matt Sarett84014f02017-01-10 11:28:54 -0500268 return transform_scanline_4444;
269 default:
270 SkASSERT(false);
271 return nullptr;
272 }
Matt Sarett84014f02017-01-10 11:28:54 -0500273 case kGray_8_SkColorType:
274 return transform_scanline_memcpy;
Matt Sarett1da27ef2017-01-19 17:14:07 -0500275 case kRGBA_F16_SkColorType:
276 switch (info.alphaType()) {
277 case kOpaque_SkAlphaType:
278 case kUnpremul_SkAlphaType:
279 return transform_scanline_F16;
280 case kPremul_SkAlphaType:
281 return transform_scanline_F16_premul;
282 default:
283 SkASSERT(false);
284 return nullptr;
285 }
Mike Klein37854712018-06-26 11:43:06 -0400286 case kRGBA_F32_SkColorType:
287 switch (info.alphaType()) {
288 case kOpaque_SkAlphaType:
289 case kUnpremul_SkAlphaType:
290 return transform_scanline_F32;
291 case kPremul_SkAlphaType:
292 return transform_scanline_F32_premul;
293 default:
294 SkASSERT(false);
295 return nullptr;
296 }
Mike Kleinac568a92018-01-25 09:09:32 -0500297 case kRGBA_1010102_SkColorType:
298 switch (info.alphaType()) {
299 case kOpaque_SkAlphaType:
300 case kUnpremul_SkAlphaType:
301 return transform_scanline_1010102;
302 case kPremul_SkAlphaType:
303 return transform_scanline_1010102_premul;
304 default:
305 SkASSERT(false);
306 return nullptr;
307 }
308 case kRGB_101010x_SkColorType:
309 return transform_scanline_101010x;
Mike Reedd6cb11e2017-11-30 15:33:04 -0500310 case kAlpha_8_SkColorType:
311 return transform_scanline_A8_to_GrayAlpha;
Matt Sarett84014f02017-01-10 11:28:54 -0500312 default:
313 SkASSERT(false);
314 return nullptr;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000315 }
reed@android.com8a1c16f2008-12-17 15:59:43 +0000316}
317
Matt Sarett1950e0a2017-06-12 16:17:30 -0400318static void set_icc(png_structp png_ptr, png_infop info_ptr, const SkImageInfo& info) {
319 sk_sp<SkData> icc = icc_from_color_space(info);
Matt Sarettc367d032017-05-05 11:13:26 -0400320 if (!icc) {
321 return;
322 }
323
324#if PNG_LIBPNG_VER_MAJOR > 1 || (PNG_LIBPNG_VER_MAJOR == 1 && PNG_LIBPNG_VER_MINOR >= 5)
325 const char* name = "Skia";
326 png_const_bytep iccPtr = icc->bytes();
327#else
328 SkString str("Skia");
329 char* name = str.writable_str();
330 png_charp iccPtr = (png_charp) icc->writable_data();
331#endif
332 png_set_iCCP(png_ptr, info_ptr, name, 0, iccPtr, icc->size());
333}
334
Matt Sarett1950e0a2017-06-12 16:17:30 -0400335bool SkPngEncoderMgr::setColorSpace(const SkImageInfo& info) {
Matt Sarettc367d032017-05-05 11:13:26 -0400336 if (setjmp(png_jmpbuf(fPngPtr))) {
337 return false;
338 }
339
Matt Sarett1950e0a2017-06-12 16:17:30 -0400340 if (info.colorSpace() && info.colorSpace()->isSRGB()) {
341 png_set_sRGB(fPngPtr, fInfoPtr, PNG_sRGB_INTENT_PERCEPTUAL);
342 } else {
343 set_icc(fPngPtr, fInfoPtr, info);
Matt Sarett0e032be2017-03-15 17:50:08 -0400344 }
345
Matt Sarettc367d032017-05-05 11:13:26 -0400346 return true;
347}
348
349bool SkPngEncoderMgr::writeInfo(const SkImageInfo& srcInfo) {
350 if (setjmp(png_jmpbuf(fPngPtr))) {
351 return false;
352 }
353
354 png_write_info(fPngPtr, fInfoPtr);
355 if (kRGBA_F16_SkColorType == srcInfo.colorType() &&
356 kOpaque_SkAlphaType == srcInfo.alphaType())
357 {
Matt Sarett1da27ef2017-01-19 17:14:07 -0500358 // For kOpaque, kRGBA_F16, we will keep the row as RGBA and tell libpng
359 // to skip the alpha channel.
Matt Sarettc367d032017-05-05 11:13:26 -0400360 png_set_filler(fPngPtr, 0, PNG_FILLER_AFTER);
Matt Sarett1da27ef2017-01-19 17:14:07 -0500361 }
reed@android.com8a1c16f2008-12-17 15:59:43 +0000362
reed@android.com8a1c16f2008-12-17 15:59:43 +0000363 return true;
364}
365
Brian Osmanb62f50c2018-07-12 14:44:27 -0400366void SkPngEncoderMgr::chooseProc(const SkImageInfo& srcInfo) {
367 fProc = choose_proc(srcInfo);
Matt Sarettc367d032017-05-05 11:13:26 -0400368}
369
Matt Sarett6a4dc662017-05-11 09:32:59 -0400370std::unique_ptr<SkEncoder> SkPngEncoder::Make(SkWStream* dst, const SkPixmap& src,
371 const Options& options) {
Brian Osmane1adc3a2018-06-04 09:21:17 -0400372 if (!SkPixmapIsValid(src)) {
Matt Sarettc367d032017-05-05 11:13:26 -0400373 return nullptr;
374 }
375
376 std::unique_ptr<SkPngEncoderMgr> encoderMgr = SkPngEncoderMgr::Make(dst);
377 if (!encoderMgr) {
378 return nullptr;
379 }
380
Matt Sarettbe4c9b02017-05-08 12:11:44 -0400381 if (!encoderMgr->setHeader(src.info(), options)) {
Matt Sarettc367d032017-05-05 11:13:26 -0400382 return nullptr;
383 }
384
Matt Sarett1950e0a2017-06-12 16:17:30 -0400385 if (!encoderMgr->setColorSpace(src.info())) {
Matt Sarettc367d032017-05-05 11:13:26 -0400386 return nullptr;
387 }
388
389 if (!encoderMgr->writeInfo(src.info())) {
390 return nullptr;
391 }
392
Brian Osmanb62f50c2018-07-12 14:44:27 -0400393 encoderMgr->chooseProc(src.info());
Matt Sarettc367d032017-05-05 11:13:26 -0400394
395 return std::unique_ptr<SkPngEncoder>(new SkPngEncoder(std::move(encoderMgr), src));
396}
397
398SkPngEncoder::SkPngEncoder(std::unique_ptr<SkPngEncoderMgr> encoderMgr, const SkPixmap& src)
399 : INHERITED(src, encoderMgr->pngBytesPerPixel() * src.width())
400 , fEncoderMgr(std::move(encoderMgr))
401{}
402
403SkPngEncoder::~SkPngEncoder() {}
404
405bool SkPngEncoder::onEncodeRows(int numRows) {
406 if (setjmp(png_jmpbuf(fEncoderMgr->pngPtr()))) {
407 return false;
408 }
409
410 const void* srcRow = fSrc.addr(0, fCurrRow);
411 for (int y = 0; y < numRows; y++) {
Mike Klein0c904fa2018-11-02 12:24:15 -0400412 fEncoderMgr->proc()((char*)fStorage.get(),
413 (const char*)srcRow,
414 fSrc.width(),
415 SkColorTypeBytesPerPixel(fSrc.colorType()));
Matt Sarettc367d032017-05-05 11:13:26 -0400416
417 png_bytep rowPtr = (png_bytep) fStorage.get();
418 png_write_rows(fEncoderMgr->pngPtr(), &rowPtr, 1);
419 srcRow = SkTAddOffset<const void>(srcRow, fSrc.rowBytes());
420 }
421
422 fCurrRow += numRows;
423 if (fCurrRow == fSrc.height()) {
424 png_write_end(fEncoderMgr->pngPtr(), fEncoderMgr->infoPtr());
425 }
426
427 return true;
428}
429
430bool SkPngEncoder::Encode(SkWStream* dst, const SkPixmap& src, const Options& options) {
431 auto encoder = SkPngEncoder::Make(dst, src, options);
432 return encoder.get() && encoder->encodeRows(src.height());
433}
434
Hal Canary1fcc4042016-11-30 17:07:59 -0500435#endif