xref: /aosp_15_r20/external/skia/src/codec/SkJpegCodec.cpp (revision c8dee2aa9b3f27cf6c858bd81872bdeb2c07ed17)
1*c8dee2aaSAndroid Build Coastguard Worker /*
2*c8dee2aaSAndroid Build Coastguard Worker  * Copyright 2015 Google Inc.
3*c8dee2aaSAndroid Build Coastguard Worker  *
4*c8dee2aaSAndroid Build Coastguard Worker  * Use of this source code is governed by a BSD-style license that can be
5*c8dee2aaSAndroid Build Coastguard Worker  * found in the LICENSE file.
6*c8dee2aaSAndroid Build Coastguard Worker  */
7*c8dee2aaSAndroid Build Coastguard Worker 
8*c8dee2aaSAndroid Build Coastguard Worker #include "src/codec/SkJpegCodec.h"
9*c8dee2aaSAndroid Build Coastguard Worker 
10*c8dee2aaSAndroid Build Coastguard Worker #include "include/codec/SkCodec.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "include/codec/SkJpegDecoder.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkAlphaType.h"
13*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkColorType.h"
14*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkData.h"
15*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkImageInfo.h"
16*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkPixmap.h"
17*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkRefCnt.h"
18*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkStream.h"
19*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkTypes.h"
20*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkYUVAInfo.h"
21*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/base/SkAlign.h"
22*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/base/SkTemplates.h"
23*c8dee2aaSAndroid Build Coastguard Worker #include "modules/skcms/skcms.h"
24*c8dee2aaSAndroid Build Coastguard Worker #include "src/codec/SkCodecPriv.h"
25*c8dee2aaSAndroid Build Coastguard Worker #include "src/codec/SkJpegConstants.h"
26*c8dee2aaSAndroid Build Coastguard Worker #include "src/codec/SkJpegDecoderMgr.h"
27*c8dee2aaSAndroid Build Coastguard Worker #include "src/codec/SkJpegMetadataDecoderImpl.h"
28*c8dee2aaSAndroid Build Coastguard Worker #include "src/codec/SkJpegPriv.h"
29*c8dee2aaSAndroid Build Coastguard Worker #include "src/codec/SkParseEncodedOrigin.h"
30*c8dee2aaSAndroid Build Coastguard Worker #include "src/codec/SkSwizzler.h"
31*c8dee2aaSAndroid Build Coastguard Worker 
32*c8dee2aaSAndroid Build Coastguard Worker #ifdef SK_CODEC_DECODES_JPEG_GAINMAPS
33*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/SkGainmapInfo.h"
34*c8dee2aaSAndroid Build Coastguard Worker #endif  // SK_CODEC_DECODES_JPEG_GAINMAPS
35*c8dee2aaSAndroid Build Coastguard Worker 
36*c8dee2aaSAndroid Build Coastguard Worker #include <array>
37*c8dee2aaSAndroid Build Coastguard Worker #include <csetjmp>
38*c8dee2aaSAndroid Build Coastguard Worker #include <cstring>
39*c8dee2aaSAndroid Build Coastguard Worker #include <utility>
40*c8dee2aaSAndroid Build Coastguard Worker 
41*c8dee2aaSAndroid Build Coastguard Worker using namespace skia_private;
42*c8dee2aaSAndroid Build Coastguard Worker 
43*c8dee2aaSAndroid Build Coastguard Worker class SkSampler;
44*c8dee2aaSAndroid Build Coastguard Worker struct SkGainmapInfo;
45*c8dee2aaSAndroid Build Coastguard Worker 
46*c8dee2aaSAndroid Build Coastguard Worker // This warning triggers false postives way too often in here.
47*c8dee2aaSAndroid Build Coastguard Worker #if defined(__GNUC__) && !defined(__clang__)
48*c8dee2aaSAndroid Build Coastguard Worker     #pragma GCC diagnostic ignored "-Wclobbered"
49*c8dee2aaSAndroid Build Coastguard Worker #endif
50*c8dee2aaSAndroid Build Coastguard Worker 
51*c8dee2aaSAndroid Build Coastguard Worker extern "C" {
52*c8dee2aaSAndroid Build Coastguard Worker     #include "jpeglib.h"  // NO_G3_REWRITE
53*c8dee2aaSAndroid Build Coastguard Worker }
54*c8dee2aaSAndroid Build Coastguard Worker 
IsJpeg(const void * buffer,size_t bytesRead)55*c8dee2aaSAndroid Build Coastguard Worker bool SkJpegCodec::IsJpeg(const void* buffer, size_t bytesRead) {
56*c8dee2aaSAndroid Build Coastguard Worker     return bytesRead >= sizeof(kJpegSig) && !memcmp(buffer, kJpegSig, sizeof(kJpegSig));
57*c8dee2aaSAndroid Build Coastguard Worker }
58*c8dee2aaSAndroid Build Coastguard Worker 
get_sk_marker_list(jpeg_decompress_struct * dinfo)59*c8dee2aaSAndroid Build Coastguard Worker SkJpegMarkerList get_sk_marker_list(jpeg_decompress_struct* dinfo) {
60*c8dee2aaSAndroid Build Coastguard Worker     SkJpegMarkerList markerList;
61*c8dee2aaSAndroid Build Coastguard Worker     for (auto* marker = dinfo->marker_list; marker; marker = marker->next) {
62*c8dee2aaSAndroid Build Coastguard Worker         markerList.emplace_back(marker->marker,
63*c8dee2aaSAndroid Build Coastguard Worker                                 SkData::MakeWithoutCopy(marker->data, marker->data_length));
64*c8dee2aaSAndroid Build Coastguard Worker     }
65*c8dee2aaSAndroid Build Coastguard Worker     return markerList;
66*c8dee2aaSAndroid Build Coastguard Worker }
67*c8dee2aaSAndroid Build Coastguard Worker 
get_exif_orientation(sk_sp<SkData> exifData)68*c8dee2aaSAndroid Build Coastguard Worker static SkEncodedOrigin get_exif_orientation(sk_sp<SkData> exifData) {
69*c8dee2aaSAndroid Build Coastguard Worker     SkEncodedOrigin origin = kDefault_SkEncodedOrigin;
70*c8dee2aaSAndroid Build Coastguard Worker     if (exifData && SkParseEncodedOrigin(exifData->bytes(), exifData->size(), &origin)) {
71*c8dee2aaSAndroid Build Coastguard Worker         return origin;
72*c8dee2aaSAndroid Build Coastguard Worker     }
73*c8dee2aaSAndroid Build Coastguard Worker     return kDefault_SkEncodedOrigin;
74*c8dee2aaSAndroid Build Coastguard Worker }
75*c8dee2aaSAndroid Build Coastguard Worker 
ReadHeader(SkStream * stream,SkCodec ** codecOut,JpegDecoderMgr ** decoderMgrOut,std::unique_ptr<SkEncodedInfo::ICCProfile> defaultColorProfile)76*c8dee2aaSAndroid Build Coastguard Worker SkCodec::Result SkJpegCodec::ReadHeader(
77*c8dee2aaSAndroid Build Coastguard Worker         SkStream* stream,
78*c8dee2aaSAndroid Build Coastguard Worker         SkCodec** codecOut,
79*c8dee2aaSAndroid Build Coastguard Worker         JpegDecoderMgr** decoderMgrOut,
80*c8dee2aaSAndroid Build Coastguard Worker         std::unique_ptr<SkEncodedInfo::ICCProfile> defaultColorProfile) {
81*c8dee2aaSAndroid Build Coastguard Worker     // Create a JpegDecoderMgr to own all of the decompress information
82*c8dee2aaSAndroid Build Coastguard Worker     std::unique_ptr<JpegDecoderMgr> decoderMgr(new JpegDecoderMgr(stream));
83*c8dee2aaSAndroid Build Coastguard Worker 
84*c8dee2aaSAndroid Build Coastguard Worker     // libjpeg errors will be caught and reported here
85*c8dee2aaSAndroid Build Coastguard Worker     skjpeg_error_mgr::AutoPushJmpBuf jmp(decoderMgr->errorMgr());
86*c8dee2aaSAndroid Build Coastguard Worker     if (setjmp(jmp)) {
87*c8dee2aaSAndroid Build Coastguard Worker         return decoderMgr->returnFailure("ReadHeader", kInvalidInput);
88*c8dee2aaSAndroid Build Coastguard Worker     }
89*c8dee2aaSAndroid Build Coastguard Worker 
90*c8dee2aaSAndroid Build Coastguard Worker     // Initialize the decompress info and the source manager
91*c8dee2aaSAndroid Build Coastguard Worker     decoderMgr->init();
92*c8dee2aaSAndroid Build Coastguard Worker     auto* dinfo = decoderMgr->dinfo();
93*c8dee2aaSAndroid Build Coastguard Worker 
94*c8dee2aaSAndroid Build Coastguard Worker     // Instruct jpeg library to save the markers that we care about.  Since
95*c8dee2aaSAndroid Build Coastguard Worker     // the orientation and color profile will not change, we can skip this
96*c8dee2aaSAndroid Build Coastguard Worker     // step on rewinds.
97*c8dee2aaSAndroid Build Coastguard Worker     if (codecOut) {
98*c8dee2aaSAndroid Build Coastguard Worker         jpeg_save_markers(dinfo, kExifMarker, 0xFFFF);
99*c8dee2aaSAndroid Build Coastguard Worker         jpeg_save_markers(dinfo, kICCMarker, 0xFFFF);
100*c8dee2aaSAndroid Build Coastguard Worker         jpeg_save_markers(dinfo, kMpfMarker, 0xFFFF);
101*c8dee2aaSAndroid Build Coastguard Worker     }
102*c8dee2aaSAndroid Build Coastguard Worker 
103*c8dee2aaSAndroid Build Coastguard Worker     // Read the jpeg header
104*c8dee2aaSAndroid Build Coastguard Worker     switch (jpeg_read_header(dinfo, TRUE)) {
105*c8dee2aaSAndroid Build Coastguard Worker         case JPEG_HEADER_OK:
106*c8dee2aaSAndroid Build Coastguard Worker             break;
107*c8dee2aaSAndroid Build Coastguard Worker         case JPEG_SUSPENDED:
108*c8dee2aaSAndroid Build Coastguard Worker             return decoderMgr->returnFailure("ReadHeader", kIncompleteInput);
109*c8dee2aaSAndroid Build Coastguard Worker         default:
110*c8dee2aaSAndroid Build Coastguard Worker             return decoderMgr->returnFailure("ReadHeader", kInvalidInput);
111*c8dee2aaSAndroid Build Coastguard Worker     }
112*c8dee2aaSAndroid Build Coastguard Worker 
113*c8dee2aaSAndroid Build Coastguard Worker     if (codecOut) {
114*c8dee2aaSAndroid Build Coastguard Worker         // Get the encoded color type
115*c8dee2aaSAndroid Build Coastguard Worker         SkEncodedInfo::Color color;
116*c8dee2aaSAndroid Build Coastguard Worker         if (!decoderMgr->getEncodedColor(&color)) {
117*c8dee2aaSAndroid Build Coastguard Worker             return kInvalidInput;
118*c8dee2aaSAndroid Build Coastguard Worker         }
119*c8dee2aaSAndroid Build Coastguard Worker 
120*c8dee2aaSAndroid Build Coastguard Worker         auto metadataDecoder =
121*c8dee2aaSAndroid Build Coastguard Worker                 std::make_unique<SkJpegMetadataDecoderImpl>(get_sk_marker_list(dinfo));
122*c8dee2aaSAndroid Build Coastguard Worker 
123*c8dee2aaSAndroid Build Coastguard Worker         SkEncodedOrigin orientation =
124*c8dee2aaSAndroid Build Coastguard Worker                 get_exif_orientation(metadataDecoder->getExifMetadata(/*copyData=*/false));
125*c8dee2aaSAndroid Build Coastguard Worker 
126*c8dee2aaSAndroid Build Coastguard Worker         std::unique_ptr<SkEncodedInfo::ICCProfile> profile;
127*c8dee2aaSAndroid Build Coastguard Worker         if (auto iccProfileData = metadataDecoder->getICCProfileData(/*copyData=*/true)) {
128*c8dee2aaSAndroid Build Coastguard Worker             profile = SkEncodedInfo::ICCProfile::Make(std::move(iccProfileData));
129*c8dee2aaSAndroid Build Coastguard Worker         }
130*c8dee2aaSAndroid Build Coastguard Worker         if (profile) {
131*c8dee2aaSAndroid Build Coastguard Worker             auto type = profile->profile()->data_color_space;
132*c8dee2aaSAndroid Build Coastguard Worker             switch (decoderMgr->dinfo()->jpeg_color_space) {
133*c8dee2aaSAndroid Build Coastguard Worker                 case JCS_CMYK:
134*c8dee2aaSAndroid Build Coastguard Worker                 case JCS_YCCK:
135*c8dee2aaSAndroid Build Coastguard Worker                     if (type != skcms_Signature_CMYK) {
136*c8dee2aaSAndroid Build Coastguard Worker                         profile = nullptr;
137*c8dee2aaSAndroid Build Coastguard Worker                     }
138*c8dee2aaSAndroid Build Coastguard Worker                     break;
139*c8dee2aaSAndroid Build Coastguard Worker                 case JCS_GRAYSCALE:
140*c8dee2aaSAndroid Build Coastguard Worker                     if (type != skcms_Signature_Gray &&
141*c8dee2aaSAndroid Build Coastguard Worker                         type != skcms_Signature_RGB)
142*c8dee2aaSAndroid Build Coastguard Worker                     {
143*c8dee2aaSAndroid Build Coastguard Worker                         profile = nullptr;
144*c8dee2aaSAndroid Build Coastguard Worker                     }
145*c8dee2aaSAndroid Build Coastguard Worker                     break;
146*c8dee2aaSAndroid Build Coastguard Worker                 default:
147*c8dee2aaSAndroid Build Coastguard Worker                     if (type != skcms_Signature_RGB) {
148*c8dee2aaSAndroid Build Coastguard Worker                         profile = nullptr;
149*c8dee2aaSAndroid Build Coastguard Worker                     }
150*c8dee2aaSAndroid Build Coastguard Worker                     break;
151*c8dee2aaSAndroid Build Coastguard Worker             }
152*c8dee2aaSAndroid Build Coastguard Worker         }
153*c8dee2aaSAndroid Build Coastguard Worker         if (!profile) {
154*c8dee2aaSAndroid Build Coastguard Worker             profile = std::move(defaultColorProfile);
155*c8dee2aaSAndroid Build Coastguard Worker         }
156*c8dee2aaSAndroid Build Coastguard Worker 
157*c8dee2aaSAndroid Build Coastguard Worker         SkEncodedInfo info = SkEncodedInfo::Make(dinfo->image_width, dinfo->image_height,
158*c8dee2aaSAndroid Build Coastguard Worker                                                  color, SkEncodedInfo::kOpaque_Alpha, 8,
159*c8dee2aaSAndroid Build Coastguard Worker                                                  std::move(profile));
160*c8dee2aaSAndroid Build Coastguard Worker 
161*c8dee2aaSAndroid Build Coastguard Worker         SkJpegCodec* codec = new SkJpegCodec(std::move(info),
162*c8dee2aaSAndroid Build Coastguard Worker                                              std::unique_ptr<SkStream>(stream),
163*c8dee2aaSAndroid Build Coastguard Worker                                              decoderMgr.release(),
164*c8dee2aaSAndroid Build Coastguard Worker                                              orientation);
165*c8dee2aaSAndroid Build Coastguard Worker         *codecOut = codec;
166*c8dee2aaSAndroid Build Coastguard Worker     } else {
167*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(nullptr != decoderMgrOut);
168*c8dee2aaSAndroid Build Coastguard Worker         *decoderMgrOut = decoderMgr.release();
169*c8dee2aaSAndroid Build Coastguard Worker     }
170*c8dee2aaSAndroid Build Coastguard Worker     return kSuccess;
171*c8dee2aaSAndroid Build Coastguard Worker }
172*c8dee2aaSAndroid Build Coastguard Worker 
MakeFromStream(std::unique_ptr<SkStream> stream,Result * result)173*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<SkCodec> SkJpegCodec::MakeFromStream(std::unique_ptr<SkStream> stream,
174*c8dee2aaSAndroid Build Coastguard Worker                                                      Result* result) {
175*c8dee2aaSAndroid Build Coastguard Worker     return SkJpegCodec::MakeFromStream(std::move(stream), result, nullptr);
176*c8dee2aaSAndroid Build Coastguard Worker }
177*c8dee2aaSAndroid Build Coastguard Worker 
MakeFromStream(std::unique_ptr<SkStream> stream,Result * result,std::unique_ptr<SkEncodedInfo::ICCProfile> defaultColorProfile)178*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<SkCodec> SkJpegCodec::MakeFromStream(std::unique_ptr<SkStream> stream,
179*c8dee2aaSAndroid Build Coastguard Worker         Result* result, std::unique_ptr<SkEncodedInfo::ICCProfile> defaultColorProfile) {
180*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(result);
181*c8dee2aaSAndroid Build Coastguard Worker     if (!stream) {
182*c8dee2aaSAndroid Build Coastguard Worker         *result = SkCodec::kInvalidInput;
183*c8dee2aaSAndroid Build Coastguard Worker         return nullptr;
184*c8dee2aaSAndroid Build Coastguard Worker     }
185*c8dee2aaSAndroid Build Coastguard Worker     SkCodec* codec = nullptr;
186*c8dee2aaSAndroid Build Coastguard Worker     *result = ReadHeader(stream.get(), &codec, nullptr, std::move(defaultColorProfile));
187*c8dee2aaSAndroid Build Coastguard Worker     if (kSuccess == *result) {
188*c8dee2aaSAndroid Build Coastguard Worker         // Codec has taken ownership of the stream, we do not need to delete it
189*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(codec);
190*c8dee2aaSAndroid Build Coastguard Worker         stream.release();
191*c8dee2aaSAndroid Build Coastguard Worker         return std::unique_ptr<SkCodec>(codec);
192*c8dee2aaSAndroid Build Coastguard Worker     }
193*c8dee2aaSAndroid Build Coastguard Worker     return nullptr;
194*c8dee2aaSAndroid Build Coastguard Worker }
195*c8dee2aaSAndroid Build Coastguard Worker 
SkJpegCodec(SkEncodedInfo && info,std::unique_ptr<SkStream> stream,JpegDecoderMgr * decoderMgr,SkEncodedOrigin origin)196*c8dee2aaSAndroid Build Coastguard Worker SkJpegCodec::SkJpegCodec(SkEncodedInfo&& info,
197*c8dee2aaSAndroid Build Coastguard Worker                          std::unique_ptr<SkStream> stream,
198*c8dee2aaSAndroid Build Coastguard Worker                          JpegDecoderMgr* decoderMgr,
199*c8dee2aaSAndroid Build Coastguard Worker                          SkEncodedOrigin origin)
200*c8dee2aaSAndroid Build Coastguard Worker         : INHERITED(std::move(info), skcms_PixelFormat_RGBA_8888, std::move(stream), origin)
201*c8dee2aaSAndroid Build Coastguard Worker         , fDecoderMgr(decoderMgr)
202*c8dee2aaSAndroid Build Coastguard Worker         , fReadyState(decoderMgr->dinfo()->global_state) {}
203*c8dee2aaSAndroid Build Coastguard Worker SkJpegCodec::~SkJpegCodec() = default;
204*c8dee2aaSAndroid Build Coastguard Worker 
205*c8dee2aaSAndroid Build Coastguard Worker /*
206*c8dee2aaSAndroid Build Coastguard Worker  * Return the row bytes of a particular image type and width
207*c8dee2aaSAndroid Build Coastguard Worker  */
get_row_bytes(const j_decompress_ptr dinfo)208*c8dee2aaSAndroid Build Coastguard Worker static size_t get_row_bytes(const j_decompress_ptr dinfo) {
209*c8dee2aaSAndroid Build Coastguard Worker     const size_t colorBytes = (dinfo->out_color_space == JCS_RGB565) ? 2 :
210*c8dee2aaSAndroid Build Coastguard Worker             dinfo->out_color_components;
211*c8dee2aaSAndroid Build Coastguard Worker     return dinfo->output_width * colorBytes;
212*c8dee2aaSAndroid Build Coastguard Worker 
213*c8dee2aaSAndroid Build Coastguard Worker }
214*c8dee2aaSAndroid Build Coastguard Worker 
215*c8dee2aaSAndroid Build Coastguard Worker /*
216*c8dee2aaSAndroid Build Coastguard Worker  *  Calculate output dimensions based on the provided factors.
217*c8dee2aaSAndroid Build Coastguard Worker  *
218*c8dee2aaSAndroid Build Coastguard Worker  *  Not to be used on the actual jpeg_decompress_struct used for decoding, since it will
219*c8dee2aaSAndroid Build Coastguard Worker  *  incorrectly modify num_components.
220*c8dee2aaSAndroid Build Coastguard Worker  */
calc_output_dimensions(jpeg_decompress_struct * dinfo,unsigned int num,unsigned int denom)221*c8dee2aaSAndroid Build Coastguard Worker void calc_output_dimensions(jpeg_decompress_struct* dinfo, unsigned int num, unsigned int denom) {
222*c8dee2aaSAndroid Build Coastguard Worker     dinfo->num_components = 0;
223*c8dee2aaSAndroid Build Coastguard Worker     dinfo->scale_num = num;
224*c8dee2aaSAndroid Build Coastguard Worker     dinfo->scale_denom = denom;
225*c8dee2aaSAndroid Build Coastguard Worker     jpeg_calc_output_dimensions(dinfo);
226*c8dee2aaSAndroid Build Coastguard Worker }
227*c8dee2aaSAndroid Build Coastguard Worker 
228*c8dee2aaSAndroid Build Coastguard Worker /*
229*c8dee2aaSAndroid Build Coastguard Worker  * Return a valid set of output dimensions for this decoder, given an input scale
230*c8dee2aaSAndroid Build Coastguard Worker  */
onGetScaledDimensions(float desiredScale) const231*c8dee2aaSAndroid Build Coastguard Worker SkISize SkJpegCodec::onGetScaledDimensions(float desiredScale) const {
232*c8dee2aaSAndroid Build Coastguard Worker     // libjpeg-turbo supports scaling by 1/8, 1/4, 3/8, 1/2, 5/8, 3/4, 7/8, and 1/1, so we will
233*c8dee2aaSAndroid Build Coastguard Worker     // support these as well
234*c8dee2aaSAndroid Build Coastguard Worker     unsigned int num;
235*c8dee2aaSAndroid Build Coastguard Worker     unsigned int denom = 8;
236*c8dee2aaSAndroid Build Coastguard Worker     if (desiredScale >= 0.9375) {
237*c8dee2aaSAndroid Build Coastguard Worker         num = 8;
238*c8dee2aaSAndroid Build Coastguard Worker     } else if (desiredScale >= 0.8125) {
239*c8dee2aaSAndroid Build Coastguard Worker         num = 7;
240*c8dee2aaSAndroid Build Coastguard Worker     } else if (desiredScale >= 0.6875f) {
241*c8dee2aaSAndroid Build Coastguard Worker         num = 6;
242*c8dee2aaSAndroid Build Coastguard Worker     } else if (desiredScale >= 0.5625f) {
243*c8dee2aaSAndroid Build Coastguard Worker         num = 5;
244*c8dee2aaSAndroid Build Coastguard Worker     } else if (desiredScale >= 0.4375f) {
245*c8dee2aaSAndroid Build Coastguard Worker         num = 4;
246*c8dee2aaSAndroid Build Coastguard Worker     } else if (desiredScale >= 0.3125f) {
247*c8dee2aaSAndroid Build Coastguard Worker         num = 3;
248*c8dee2aaSAndroid Build Coastguard Worker     } else if (desiredScale >= 0.1875f) {
249*c8dee2aaSAndroid Build Coastguard Worker         num = 2;
250*c8dee2aaSAndroid Build Coastguard Worker     } else {
251*c8dee2aaSAndroid Build Coastguard Worker         num = 1;
252*c8dee2aaSAndroid Build Coastguard Worker     }
253*c8dee2aaSAndroid Build Coastguard Worker 
254*c8dee2aaSAndroid Build Coastguard Worker     // Set up a fake decompress struct in order to use libjpeg to calculate output dimensions.
255*c8dee2aaSAndroid Build Coastguard Worker     // This isn't conventional use of libjpeg-turbo but initializing the decompress struct with
256*c8dee2aaSAndroid Build Coastguard Worker     // jpeg_create_decompress allows for less violation of the API regardless of the version.
257*c8dee2aaSAndroid Build Coastguard Worker     jpeg_decompress_struct dinfo;
258*c8dee2aaSAndroid Build Coastguard Worker     jpeg_create_decompress(&dinfo);
259*c8dee2aaSAndroid Build Coastguard Worker     dinfo.image_width = this->dimensions().width();
260*c8dee2aaSAndroid Build Coastguard Worker     dinfo.image_height = this->dimensions().height();
261*c8dee2aaSAndroid Build Coastguard Worker     dinfo.global_state = fReadyState;
262*c8dee2aaSAndroid Build Coastguard Worker     calc_output_dimensions(&dinfo, num, denom);
263*c8dee2aaSAndroid Build Coastguard Worker     SkISize outputDimensions = SkISize::Make(dinfo.output_width, dinfo.output_height);
264*c8dee2aaSAndroid Build Coastguard Worker     jpeg_destroy_decompress(&dinfo);
265*c8dee2aaSAndroid Build Coastguard Worker 
266*c8dee2aaSAndroid Build Coastguard Worker     return outputDimensions;
267*c8dee2aaSAndroid Build Coastguard Worker }
268*c8dee2aaSAndroid Build Coastguard Worker 
onRewind()269*c8dee2aaSAndroid Build Coastguard Worker bool SkJpegCodec::onRewind() {
270*c8dee2aaSAndroid Build Coastguard Worker     JpegDecoderMgr* decoderMgr = nullptr;
271*c8dee2aaSAndroid Build Coastguard Worker     if (kSuccess != ReadHeader(this->stream(), nullptr, &decoderMgr, nullptr)) {
272*c8dee2aaSAndroid Build Coastguard Worker         return fDecoderMgr->returnFalse("onRewind");
273*c8dee2aaSAndroid Build Coastguard Worker     }
274*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(nullptr != decoderMgr);
275*c8dee2aaSAndroid Build Coastguard Worker     fDecoderMgr.reset(decoderMgr);
276*c8dee2aaSAndroid Build Coastguard Worker 
277*c8dee2aaSAndroid Build Coastguard Worker     fSwizzler.reset(nullptr);
278*c8dee2aaSAndroid Build Coastguard Worker     fSwizzleSrcRow = nullptr;
279*c8dee2aaSAndroid Build Coastguard Worker     fColorXformSrcRow = nullptr;
280*c8dee2aaSAndroid Build Coastguard Worker     fStorage.reset();
281*c8dee2aaSAndroid Build Coastguard Worker 
282*c8dee2aaSAndroid Build Coastguard Worker     return true;
283*c8dee2aaSAndroid Build Coastguard Worker }
284*c8dee2aaSAndroid Build Coastguard Worker 
conversionSupported(const SkImageInfo & dstInfo,bool srcIsOpaque,bool needsColorXform)285*c8dee2aaSAndroid Build Coastguard Worker bool SkJpegCodec::conversionSupported(const SkImageInfo& dstInfo, bool srcIsOpaque,
286*c8dee2aaSAndroid Build Coastguard Worker                                       bool needsColorXform) {
287*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(srcIsOpaque);
288*c8dee2aaSAndroid Build Coastguard Worker 
289*c8dee2aaSAndroid Build Coastguard Worker     if (kUnknown_SkAlphaType == dstInfo.alphaType()) {
290*c8dee2aaSAndroid Build Coastguard Worker         return false;
291*c8dee2aaSAndroid Build Coastguard Worker     }
292*c8dee2aaSAndroid Build Coastguard Worker 
293*c8dee2aaSAndroid Build Coastguard Worker     if (kOpaque_SkAlphaType != dstInfo.alphaType()) {
294*c8dee2aaSAndroid Build Coastguard Worker         SkCodecPrintf("Warning: an opaque image should be decoded as opaque "
295*c8dee2aaSAndroid Build Coastguard Worker                       "- it is being decoded as non-opaque, which will draw slower\n");
296*c8dee2aaSAndroid Build Coastguard Worker     }
297*c8dee2aaSAndroid Build Coastguard Worker 
298*c8dee2aaSAndroid Build Coastguard Worker     J_COLOR_SPACE encodedColorType = fDecoderMgr->dinfo()->jpeg_color_space;
299*c8dee2aaSAndroid Build Coastguard Worker 
300*c8dee2aaSAndroid Build Coastguard Worker     // Check for valid color types and set the output color space
301*c8dee2aaSAndroid Build Coastguard Worker     switch (dstInfo.colorType()) {
302*c8dee2aaSAndroid Build Coastguard Worker         case kRGBA_8888_SkColorType:
303*c8dee2aaSAndroid Build Coastguard Worker             fDecoderMgr->dinfo()->out_color_space = JCS_EXT_RGBA;
304*c8dee2aaSAndroid Build Coastguard Worker             break;
305*c8dee2aaSAndroid Build Coastguard Worker         case kBGRA_8888_SkColorType:
306*c8dee2aaSAndroid Build Coastguard Worker             if (needsColorXform) {
307*c8dee2aaSAndroid Build Coastguard Worker                 // Always using RGBA as the input format for color xforms makes the
308*c8dee2aaSAndroid Build Coastguard Worker                 // implementation a little simpler.
309*c8dee2aaSAndroid Build Coastguard Worker                 fDecoderMgr->dinfo()->out_color_space = JCS_EXT_RGBA;
310*c8dee2aaSAndroid Build Coastguard Worker             } else {
311*c8dee2aaSAndroid Build Coastguard Worker                 fDecoderMgr->dinfo()->out_color_space = JCS_EXT_BGRA;
312*c8dee2aaSAndroid Build Coastguard Worker             }
313*c8dee2aaSAndroid Build Coastguard Worker             break;
314*c8dee2aaSAndroid Build Coastguard Worker         case kRGB_565_SkColorType:
315*c8dee2aaSAndroid Build Coastguard Worker             if (needsColorXform) {
316*c8dee2aaSAndroid Build Coastguard Worker                 fDecoderMgr->dinfo()->out_color_space = JCS_EXT_RGBA;
317*c8dee2aaSAndroid Build Coastguard Worker             } else {
318*c8dee2aaSAndroid Build Coastguard Worker                 fDecoderMgr->dinfo()->dither_mode = JDITHER_NONE;
319*c8dee2aaSAndroid Build Coastguard Worker                 fDecoderMgr->dinfo()->out_color_space = JCS_RGB565;
320*c8dee2aaSAndroid Build Coastguard Worker             }
321*c8dee2aaSAndroid Build Coastguard Worker             break;
322*c8dee2aaSAndroid Build Coastguard Worker         case kGray_8_SkColorType:
323*c8dee2aaSAndroid Build Coastguard Worker             if (JCS_GRAYSCALE != encodedColorType) {
324*c8dee2aaSAndroid Build Coastguard Worker                 return false;
325*c8dee2aaSAndroid Build Coastguard Worker             }
326*c8dee2aaSAndroid Build Coastguard Worker 
327*c8dee2aaSAndroid Build Coastguard Worker             if (needsColorXform) {
328*c8dee2aaSAndroid Build Coastguard Worker                 fDecoderMgr->dinfo()->out_color_space = JCS_EXT_RGBA;
329*c8dee2aaSAndroid Build Coastguard Worker             } else {
330*c8dee2aaSAndroid Build Coastguard Worker                 fDecoderMgr->dinfo()->out_color_space = JCS_GRAYSCALE;
331*c8dee2aaSAndroid Build Coastguard Worker             }
332*c8dee2aaSAndroid Build Coastguard Worker             break;
333*c8dee2aaSAndroid Build Coastguard Worker         case kBGRA_10101010_XR_SkColorType:
334*c8dee2aaSAndroid Build Coastguard Worker         case kBGR_101010x_XR_SkColorType:
335*c8dee2aaSAndroid Build Coastguard Worker         case kRGBA_F16_SkColorType:
336*c8dee2aaSAndroid Build Coastguard Worker             SkASSERT(needsColorXform);
337*c8dee2aaSAndroid Build Coastguard Worker             fDecoderMgr->dinfo()->out_color_space = JCS_EXT_RGBA;
338*c8dee2aaSAndroid Build Coastguard Worker             break;
339*c8dee2aaSAndroid Build Coastguard Worker         default:
340*c8dee2aaSAndroid Build Coastguard Worker             return false;
341*c8dee2aaSAndroid Build Coastguard Worker     }
342*c8dee2aaSAndroid Build Coastguard Worker 
343*c8dee2aaSAndroid Build Coastguard Worker     // Check if we will decode to CMYK.  libjpeg-turbo does not convert CMYK to RGBA, so
344*c8dee2aaSAndroid Build Coastguard Worker     // we must do it ourselves.
345*c8dee2aaSAndroid Build Coastguard Worker     if (JCS_CMYK == encodedColorType || JCS_YCCK == encodedColorType) {
346*c8dee2aaSAndroid Build Coastguard Worker         fDecoderMgr->dinfo()->out_color_space = JCS_CMYK;
347*c8dee2aaSAndroid Build Coastguard Worker     }
348*c8dee2aaSAndroid Build Coastguard Worker 
349*c8dee2aaSAndroid Build Coastguard Worker     return true;
350*c8dee2aaSAndroid Build Coastguard Worker }
351*c8dee2aaSAndroid Build Coastguard Worker 
352*c8dee2aaSAndroid Build Coastguard Worker /*
353*c8dee2aaSAndroid Build Coastguard Worker  * Checks if we can natively scale to the requested dimensions and natively scales the
354*c8dee2aaSAndroid Build Coastguard Worker  * dimensions if possible
355*c8dee2aaSAndroid Build Coastguard Worker  */
onDimensionsSupported(const SkISize & size)356*c8dee2aaSAndroid Build Coastguard Worker bool SkJpegCodec::onDimensionsSupported(const SkISize& size) {
357*c8dee2aaSAndroid Build Coastguard Worker     skjpeg_error_mgr::AutoPushJmpBuf jmp(fDecoderMgr->errorMgr());
358*c8dee2aaSAndroid Build Coastguard Worker     if (setjmp(jmp)) {
359*c8dee2aaSAndroid Build Coastguard Worker         return fDecoderMgr->returnFalse("onDimensionsSupported");
360*c8dee2aaSAndroid Build Coastguard Worker     }
361*c8dee2aaSAndroid Build Coastguard Worker 
362*c8dee2aaSAndroid Build Coastguard Worker     const unsigned int dstWidth = size.width();
363*c8dee2aaSAndroid Build Coastguard Worker     const unsigned int dstHeight = size.height();
364*c8dee2aaSAndroid Build Coastguard Worker 
365*c8dee2aaSAndroid Build Coastguard Worker     // Set up a fake decompress struct in order to use libjpeg to calculate output dimensions
366*c8dee2aaSAndroid Build Coastguard Worker     // This isn't conventional use of libjpeg-turbo but initializing the decompress struct with
367*c8dee2aaSAndroid Build Coastguard Worker     // jpeg_create_decompress allows for less violation of the API regardless of the version.
368*c8dee2aaSAndroid Build Coastguard Worker     // FIXME: Why is this necessary?
369*c8dee2aaSAndroid Build Coastguard Worker     jpeg_decompress_struct dinfo;
370*c8dee2aaSAndroid Build Coastguard Worker     jpeg_create_decompress(&dinfo);
371*c8dee2aaSAndroid Build Coastguard Worker     dinfo.image_width = this->dimensions().width();
372*c8dee2aaSAndroid Build Coastguard Worker     dinfo.image_height = this->dimensions().height();
373*c8dee2aaSAndroid Build Coastguard Worker     dinfo.global_state = fReadyState;
374*c8dee2aaSAndroid Build Coastguard Worker 
375*c8dee2aaSAndroid Build Coastguard Worker     // libjpeg-turbo can scale to 1/8, 1/4, 3/8, 1/2, 5/8, 3/4, 7/8, and 1/1
376*c8dee2aaSAndroid Build Coastguard Worker     unsigned int num = 8;
377*c8dee2aaSAndroid Build Coastguard Worker     const unsigned int denom = 8;
378*c8dee2aaSAndroid Build Coastguard Worker     calc_output_dimensions(&dinfo, num, denom);
379*c8dee2aaSAndroid Build Coastguard Worker     while (dinfo.output_width != dstWidth || dinfo.output_height != dstHeight) {
380*c8dee2aaSAndroid Build Coastguard Worker 
381*c8dee2aaSAndroid Build Coastguard Worker         // Return a failure if we have tried all of the possible scales
382*c8dee2aaSAndroid Build Coastguard Worker         if (1 == num || dstWidth > dinfo.output_width || dstHeight > dinfo.output_height) {
383*c8dee2aaSAndroid Build Coastguard Worker             jpeg_destroy_decompress(&dinfo);
384*c8dee2aaSAndroid Build Coastguard Worker             return false;
385*c8dee2aaSAndroid Build Coastguard Worker         }
386*c8dee2aaSAndroid Build Coastguard Worker 
387*c8dee2aaSAndroid Build Coastguard Worker         // Try the next scale
388*c8dee2aaSAndroid Build Coastguard Worker         num -= 1;
389*c8dee2aaSAndroid Build Coastguard Worker         calc_output_dimensions(&dinfo, num, denom);
390*c8dee2aaSAndroid Build Coastguard Worker     }
391*c8dee2aaSAndroid Build Coastguard Worker     jpeg_destroy_decompress(&dinfo);
392*c8dee2aaSAndroid Build Coastguard Worker 
393*c8dee2aaSAndroid Build Coastguard Worker     fDecoderMgr->dinfo()->scale_num = num;
394*c8dee2aaSAndroid Build Coastguard Worker     fDecoderMgr->dinfo()->scale_denom = denom;
395*c8dee2aaSAndroid Build Coastguard Worker     return true;
396*c8dee2aaSAndroid Build Coastguard Worker }
397*c8dee2aaSAndroid Build Coastguard Worker 
readRows(const SkImageInfo & dstInfo,void * dst,size_t rowBytes,int count,const Options & opts)398*c8dee2aaSAndroid Build Coastguard Worker int SkJpegCodec::readRows(const SkImageInfo& dstInfo, void* dst, size_t rowBytes, int count,
399*c8dee2aaSAndroid Build Coastguard Worker                           const Options& opts) {
400*c8dee2aaSAndroid Build Coastguard Worker     // Set the jump location for libjpeg-turbo errors
401*c8dee2aaSAndroid Build Coastguard Worker     skjpeg_error_mgr::AutoPushJmpBuf jmp(fDecoderMgr->errorMgr());
402*c8dee2aaSAndroid Build Coastguard Worker     if (setjmp(jmp)) {
403*c8dee2aaSAndroid Build Coastguard Worker         return 0;
404*c8dee2aaSAndroid Build Coastguard Worker     }
405*c8dee2aaSAndroid Build Coastguard Worker 
406*c8dee2aaSAndroid Build Coastguard Worker     // When fSwizzleSrcRow is non-null, it means that we need to swizzle.  In this case,
407*c8dee2aaSAndroid Build Coastguard Worker     // we will always decode into fSwizzlerSrcRow before swizzling into the next buffer.
408*c8dee2aaSAndroid Build Coastguard Worker     // We can never swizzle "in place" because the swizzler may perform sampling and/or
409*c8dee2aaSAndroid Build Coastguard Worker     // subsetting.
410*c8dee2aaSAndroid Build Coastguard Worker     // When fColorXformSrcRow is non-null, it means that we need to color xform and that
411*c8dee2aaSAndroid Build Coastguard Worker     // we cannot color xform "in place" (many times we can, but not when the src and dst
412*c8dee2aaSAndroid Build Coastguard Worker     // are different sizes).
413*c8dee2aaSAndroid Build Coastguard Worker     // In this case, we will color xform from fColorXformSrcRow into the dst.
414*c8dee2aaSAndroid Build Coastguard Worker     JSAMPLE* decodeDst = (JSAMPLE*) dst;
415*c8dee2aaSAndroid Build Coastguard Worker     uint32_t* swizzleDst = (uint32_t*) dst;
416*c8dee2aaSAndroid Build Coastguard Worker     size_t decodeDstRowBytes = rowBytes;
417*c8dee2aaSAndroid Build Coastguard Worker     size_t swizzleDstRowBytes = rowBytes;
418*c8dee2aaSAndroid Build Coastguard Worker     int dstWidth = opts.fSubset ? opts.fSubset->width() : dstInfo.width();
419*c8dee2aaSAndroid Build Coastguard Worker     if (fSwizzleSrcRow && fColorXformSrcRow) {
420*c8dee2aaSAndroid Build Coastguard Worker         decodeDst = (JSAMPLE*) fSwizzleSrcRow;
421*c8dee2aaSAndroid Build Coastguard Worker         swizzleDst = fColorXformSrcRow;
422*c8dee2aaSAndroid Build Coastguard Worker         decodeDstRowBytes = 0;
423*c8dee2aaSAndroid Build Coastguard Worker         swizzleDstRowBytes = 0;
424*c8dee2aaSAndroid Build Coastguard Worker         dstWidth = fSwizzler->swizzleWidth();
425*c8dee2aaSAndroid Build Coastguard Worker     } else if (fColorXformSrcRow) {
426*c8dee2aaSAndroid Build Coastguard Worker         decodeDst = (JSAMPLE*) fColorXformSrcRow;
427*c8dee2aaSAndroid Build Coastguard Worker         swizzleDst = fColorXformSrcRow;
428*c8dee2aaSAndroid Build Coastguard Worker         decodeDstRowBytes = 0;
429*c8dee2aaSAndroid Build Coastguard Worker         swizzleDstRowBytes = 0;
430*c8dee2aaSAndroid Build Coastguard Worker     } else if (fSwizzleSrcRow) {
431*c8dee2aaSAndroid Build Coastguard Worker         decodeDst = (JSAMPLE*) fSwizzleSrcRow;
432*c8dee2aaSAndroid Build Coastguard Worker         decodeDstRowBytes = 0;
433*c8dee2aaSAndroid Build Coastguard Worker         dstWidth = fSwizzler->swizzleWidth();
434*c8dee2aaSAndroid Build Coastguard Worker     }
435*c8dee2aaSAndroid Build Coastguard Worker 
436*c8dee2aaSAndroid Build Coastguard Worker     for (int y = 0; y < count; y++) {
437*c8dee2aaSAndroid Build Coastguard Worker         uint32_t lines = jpeg_read_scanlines(fDecoderMgr->dinfo(), &decodeDst, 1);
438*c8dee2aaSAndroid Build Coastguard Worker         if (0 == lines) {
439*c8dee2aaSAndroid Build Coastguard Worker             return y;
440*c8dee2aaSAndroid Build Coastguard Worker         }
441*c8dee2aaSAndroid Build Coastguard Worker 
442*c8dee2aaSAndroid Build Coastguard Worker         if (fSwizzler) {
443*c8dee2aaSAndroid Build Coastguard Worker             fSwizzler->swizzle(swizzleDst, decodeDst);
444*c8dee2aaSAndroid Build Coastguard Worker         }
445*c8dee2aaSAndroid Build Coastguard Worker 
446*c8dee2aaSAndroid Build Coastguard Worker         if (this->colorXform()) {
447*c8dee2aaSAndroid Build Coastguard Worker             this->applyColorXform(dst, swizzleDst, dstWidth);
448*c8dee2aaSAndroid Build Coastguard Worker             dst = SkTAddOffset<void>(dst, rowBytes);
449*c8dee2aaSAndroid Build Coastguard Worker         }
450*c8dee2aaSAndroid Build Coastguard Worker 
451*c8dee2aaSAndroid Build Coastguard Worker         decodeDst = SkTAddOffset<JSAMPLE>(decodeDst, decodeDstRowBytes);
452*c8dee2aaSAndroid Build Coastguard Worker         swizzleDst = SkTAddOffset<uint32_t>(swizzleDst, swizzleDstRowBytes);
453*c8dee2aaSAndroid Build Coastguard Worker     }
454*c8dee2aaSAndroid Build Coastguard Worker 
455*c8dee2aaSAndroid Build Coastguard Worker     return count;
456*c8dee2aaSAndroid Build Coastguard Worker }
457*c8dee2aaSAndroid Build Coastguard Worker 
458*c8dee2aaSAndroid Build Coastguard Worker /*
459*c8dee2aaSAndroid Build Coastguard Worker  * This is a bit tricky.  We only need the swizzler to do format conversion if the jpeg is
460*c8dee2aaSAndroid Build Coastguard Worker  * encoded as CMYK.
461*c8dee2aaSAndroid Build Coastguard Worker  * And even then we still may not need it.  If the jpeg has a CMYK color profile and a color
462*c8dee2aaSAndroid Build Coastguard Worker  * xform, the color xform will handle the CMYK->RGB conversion.
463*c8dee2aaSAndroid Build Coastguard Worker  */
needs_swizzler_to_convert_from_cmyk(J_COLOR_SPACE jpegColorType,const skcms_ICCProfile * srcProfile,bool hasColorSpaceXform)464*c8dee2aaSAndroid Build Coastguard Worker static inline bool needs_swizzler_to_convert_from_cmyk(J_COLOR_SPACE jpegColorType,
465*c8dee2aaSAndroid Build Coastguard Worker                                                        const skcms_ICCProfile* srcProfile,
466*c8dee2aaSAndroid Build Coastguard Worker                                                        bool hasColorSpaceXform) {
467*c8dee2aaSAndroid Build Coastguard Worker     if (JCS_CMYK != jpegColorType) {
468*c8dee2aaSAndroid Build Coastguard Worker         return false;
469*c8dee2aaSAndroid Build Coastguard Worker     }
470*c8dee2aaSAndroid Build Coastguard Worker 
471*c8dee2aaSAndroid Build Coastguard Worker     bool hasCMYKColorSpace = srcProfile && srcProfile->data_color_space == skcms_Signature_CMYK;
472*c8dee2aaSAndroid Build Coastguard Worker     return !hasCMYKColorSpace || !hasColorSpaceXform;
473*c8dee2aaSAndroid Build Coastguard Worker }
474*c8dee2aaSAndroid Build Coastguard Worker 
475*c8dee2aaSAndroid Build Coastguard Worker /*
476*c8dee2aaSAndroid Build Coastguard Worker  * Performs the jpeg decode
477*c8dee2aaSAndroid Build Coastguard Worker  */
onGetPixels(const SkImageInfo & dstInfo,void * dst,size_t dstRowBytes,const Options & options,int * rowsDecoded)478*c8dee2aaSAndroid Build Coastguard Worker SkCodec::Result SkJpegCodec::onGetPixels(const SkImageInfo& dstInfo,
479*c8dee2aaSAndroid Build Coastguard Worker                                          void* dst, size_t dstRowBytes,
480*c8dee2aaSAndroid Build Coastguard Worker                                          const Options& options,
481*c8dee2aaSAndroid Build Coastguard Worker                                          int* rowsDecoded) {
482*c8dee2aaSAndroid Build Coastguard Worker     if (options.fSubset) {
483*c8dee2aaSAndroid Build Coastguard Worker         // Subsets are not supported.
484*c8dee2aaSAndroid Build Coastguard Worker         return kUnimplemented;
485*c8dee2aaSAndroid Build Coastguard Worker     }
486*c8dee2aaSAndroid Build Coastguard Worker 
487*c8dee2aaSAndroid Build Coastguard Worker     // Get a pointer to the decompress info since we will use it quite frequently
488*c8dee2aaSAndroid Build Coastguard Worker     jpeg_decompress_struct* dinfo = fDecoderMgr->dinfo();
489*c8dee2aaSAndroid Build Coastguard Worker 
490*c8dee2aaSAndroid Build Coastguard Worker     // Set the jump location for libjpeg errors
491*c8dee2aaSAndroid Build Coastguard Worker     skjpeg_error_mgr::AutoPushJmpBuf jmp(fDecoderMgr->errorMgr());
492*c8dee2aaSAndroid Build Coastguard Worker     if (setjmp(jmp)) {
493*c8dee2aaSAndroid Build Coastguard Worker         return fDecoderMgr->returnFailure("setjmp", kInvalidInput);
494*c8dee2aaSAndroid Build Coastguard Worker     }
495*c8dee2aaSAndroid Build Coastguard Worker 
496*c8dee2aaSAndroid Build Coastguard Worker     if (!jpeg_start_decompress(dinfo)) {
497*c8dee2aaSAndroid Build Coastguard Worker         return fDecoderMgr->returnFailure("startDecompress", kInvalidInput);
498*c8dee2aaSAndroid Build Coastguard Worker     }
499*c8dee2aaSAndroid Build Coastguard Worker 
500*c8dee2aaSAndroid Build Coastguard Worker     // The recommended output buffer height should always be 1 in high quality modes.
501*c8dee2aaSAndroid Build Coastguard Worker     // If it's not, we want to know because it means our strategy is not optimal.
502*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(1 == dinfo->rec_outbuf_height);
503*c8dee2aaSAndroid Build Coastguard Worker 
504*c8dee2aaSAndroid Build Coastguard Worker     if (needs_swizzler_to_convert_from_cmyk(dinfo->out_color_space,
505*c8dee2aaSAndroid Build Coastguard Worker                                             this->getEncodedInfo().profile(), this->colorXform())) {
506*c8dee2aaSAndroid Build Coastguard Worker         this->initializeSwizzler(dstInfo, options, true);
507*c8dee2aaSAndroid Build Coastguard Worker     }
508*c8dee2aaSAndroid Build Coastguard Worker 
509*c8dee2aaSAndroid Build Coastguard Worker     if (!this->allocateStorage(dstInfo)) {
510*c8dee2aaSAndroid Build Coastguard Worker         return kInternalError;
511*c8dee2aaSAndroid Build Coastguard Worker     }
512*c8dee2aaSAndroid Build Coastguard Worker 
513*c8dee2aaSAndroid Build Coastguard Worker     int rows = this->readRows(dstInfo, dst, dstRowBytes, dstInfo.height(), options);
514*c8dee2aaSAndroid Build Coastguard Worker     if (rows < dstInfo.height()) {
515*c8dee2aaSAndroid Build Coastguard Worker         *rowsDecoded = rows;
516*c8dee2aaSAndroid Build Coastguard Worker         return fDecoderMgr->returnFailure("Incomplete image data", kIncompleteInput);
517*c8dee2aaSAndroid Build Coastguard Worker     }
518*c8dee2aaSAndroid Build Coastguard Worker 
519*c8dee2aaSAndroid Build Coastguard Worker     return kSuccess;
520*c8dee2aaSAndroid Build Coastguard Worker }
521*c8dee2aaSAndroid Build Coastguard Worker 
allocateStorage(const SkImageInfo & dstInfo)522*c8dee2aaSAndroid Build Coastguard Worker bool SkJpegCodec::allocateStorage(const SkImageInfo& dstInfo) {
523*c8dee2aaSAndroid Build Coastguard Worker     int dstWidth = dstInfo.width();
524*c8dee2aaSAndroid Build Coastguard Worker 
525*c8dee2aaSAndroid Build Coastguard Worker     size_t swizzleBytes = 0;
526*c8dee2aaSAndroid Build Coastguard Worker     if (fSwizzler) {
527*c8dee2aaSAndroid Build Coastguard Worker         swizzleBytes = get_row_bytes(fDecoderMgr->dinfo());
528*c8dee2aaSAndroid Build Coastguard Worker         dstWidth = fSwizzler->swizzleWidth();
529*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(!this->colorXform() || SkIsAlign4(swizzleBytes));
530*c8dee2aaSAndroid Build Coastguard Worker     }
531*c8dee2aaSAndroid Build Coastguard Worker 
532*c8dee2aaSAndroid Build Coastguard Worker     size_t xformBytes = 0;
533*c8dee2aaSAndroid Build Coastguard Worker 
534*c8dee2aaSAndroid Build Coastguard Worker     if (this->colorXform() && sizeof(uint32_t) != dstInfo.bytesPerPixel()) {
535*c8dee2aaSAndroid Build Coastguard Worker         xformBytes = dstWidth * sizeof(uint32_t);
536*c8dee2aaSAndroid Build Coastguard Worker     }
537*c8dee2aaSAndroid Build Coastguard Worker 
538*c8dee2aaSAndroid Build Coastguard Worker     size_t totalBytes = swizzleBytes + xformBytes;
539*c8dee2aaSAndroid Build Coastguard Worker     if (totalBytes > 0) {
540*c8dee2aaSAndroid Build Coastguard Worker         if (!fStorage.reset(totalBytes)) {
541*c8dee2aaSAndroid Build Coastguard Worker             return false;
542*c8dee2aaSAndroid Build Coastguard Worker         }
543*c8dee2aaSAndroid Build Coastguard Worker         fSwizzleSrcRow = (swizzleBytes > 0) ? fStorage.get() : nullptr;
544*c8dee2aaSAndroid Build Coastguard Worker         fColorXformSrcRow = (xformBytes > 0) ?
545*c8dee2aaSAndroid Build Coastguard Worker                 SkTAddOffset<uint32_t>(fStorage.get(), swizzleBytes) : nullptr;
546*c8dee2aaSAndroid Build Coastguard Worker     }
547*c8dee2aaSAndroid Build Coastguard Worker     return true;
548*c8dee2aaSAndroid Build Coastguard Worker }
549*c8dee2aaSAndroid Build Coastguard Worker 
initializeSwizzler(const SkImageInfo & dstInfo,const Options & options,bool needsCMYKToRGB)550*c8dee2aaSAndroid Build Coastguard Worker void SkJpegCodec::initializeSwizzler(const SkImageInfo& dstInfo, const Options& options,
551*c8dee2aaSAndroid Build Coastguard Worker         bool needsCMYKToRGB) {
552*c8dee2aaSAndroid Build Coastguard Worker     Options swizzlerOptions = options;
553*c8dee2aaSAndroid Build Coastguard Worker     if (options.fSubset) {
554*c8dee2aaSAndroid Build Coastguard Worker         // Use fSwizzlerSubset if this is a subset decode.  This is necessary in the case
555*c8dee2aaSAndroid Build Coastguard Worker         // where libjpeg-turbo provides a subset and then we need to subset it further.
556*c8dee2aaSAndroid Build Coastguard Worker         // Also, verify that fSwizzlerSubset is initialized and valid.
557*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(!fSwizzlerSubset.isEmpty() && fSwizzlerSubset.x() <= options.fSubset->x() &&
558*c8dee2aaSAndroid Build Coastguard Worker                 fSwizzlerSubset.width() == options.fSubset->width());
559*c8dee2aaSAndroid Build Coastguard Worker         swizzlerOptions.fSubset = &fSwizzlerSubset;
560*c8dee2aaSAndroid Build Coastguard Worker     }
561*c8dee2aaSAndroid Build Coastguard Worker 
562*c8dee2aaSAndroid Build Coastguard Worker     SkImageInfo swizzlerDstInfo = dstInfo;
563*c8dee2aaSAndroid Build Coastguard Worker     if (this->colorXform()) {
564*c8dee2aaSAndroid Build Coastguard Worker         // The color xform will be expecting RGBA 8888 input.
565*c8dee2aaSAndroid Build Coastguard Worker         swizzlerDstInfo = swizzlerDstInfo.makeColorType(kRGBA_8888_SkColorType);
566*c8dee2aaSAndroid Build Coastguard Worker     }
567*c8dee2aaSAndroid Build Coastguard Worker 
568*c8dee2aaSAndroid Build Coastguard Worker     if (needsCMYKToRGB) {
569*c8dee2aaSAndroid Build Coastguard Worker         // The swizzler is used to convert to from CMYK.
570*c8dee2aaSAndroid Build Coastguard Worker         // The swizzler does not use the width or height on SkEncodedInfo.
571*c8dee2aaSAndroid Build Coastguard Worker         auto swizzlerInfo = SkEncodedInfo::Make(0, 0, SkEncodedInfo::kInvertedCMYK_Color,
572*c8dee2aaSAndroid Build Coastguard Worker                                                 SkEncodedInfo::kOpaque_Alpha, 8);
573*c8dee2aaSAndroid Build Coastguard Worker         fSwizzler = SkSwizzler::Make(swizzlerInfo, nullptr, swizzlerDstInfo, swizzlerOptions);
574*c8dee2aaSAndroid Build Coastguard Worker     } else {
575*c8dee2aaSAndroid Build Coastguard Worker         int srcBPP = 0;
576*c8dee2aaSAndroid Build Coastguard Worker         switch (fDecoderMgr->dinfo()->out_color_space) {
577*c8dee2aaSAndroid Build Coastguard Worker             case JCS_EXT_RGBA:
578*c8dee2aaSAndroid Build Coastguard Worker             case JCS_EXT_BGRA:
579*c8dee2aaSAndroid Build Coastguard Worker             case JCS_CMYK:
580*c8dee2aaSAndroid Build Coastguard Worker                 srcBPP = 4;
581*c8dee2aaSAndroid Build Coastguard Worker                 break;
582*c8dee2aaSAndroid Build Coastguard Worker             case JCS_RGB565:
583*c8dee2aaSAndroid Build Coastguard Worker                 srcBPP = 2;
584*c8dee2aaSAndroid Build Coastguard Worker                 break;
585*c8dee2aaSAndroid Build Coastguard Worker             case JCS_GRAYSCALE:
586*c8dee2aaSAndroid Build Coastguard Worker                 srcBPP = 1;
587*c8dee2aaSAndroid Build Coastguard Worker                 break;
588*c8dee2aaSAndroid Build Coastguard Worker             default:
589*c8dee2aaSAndroid Build Coastguard Worker                 SkASSERT(false);
590*c8dee2aaSAndroid Build Coastguard Worker                 break;
591*c8dee2aaSAndroid Build Coastguard Worker         }
592*c8dee2aaSAndroid Build Coastguard Worker         fSwizzler = SkSwizzler::MakeSimple(srcBPP, swizzlerDstInfo, swizzlerOptions);
593*c8dee2aaSAndroid Build Coastguard Worker     }
594*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(fSwizzler);
595*c8dee2aaSAndroid Build Coastguard Worker }
596*c8dee2aaSAndroid Build Coastguard Worker 
getSampler(bool createIfNecessary)597*c8dee2aaSAndroid Build Coastguard Worker SkSampler* SkJpegCodec::getSampler(bool createIfNecessary) {
598*c8dee2aaSAndroid Build Coastguard Worker     if (!createIfNecessary || fSwizzler) {
599*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(!fSwizzler || (fSwizzleSrcRow && fStorage.get() == fSwizzleSrcRow));
600*c8dee2aaSAndroid Build Coastguard Worker         return fSwizzler.get();
601*c8dee2aaSAndroid Build Coastguard Worker     }
602*c8dee2aaSAndroid Build Coastguard Worker 
603*c8dee2aaSAndroid Build Coastguard Worker     bool needsCMYKToRGB = needs_swizzler_to_convert_from_cmyk(
604*c8dee2aaSAndroid Build Coastguard Worker             fDecoderMgr->dinfo()->out_color_space, this->getEncodedInfo().profile(),
605*c8dee2aaSAndroid Build Coastguard Worker             this->colorXform());
606*c8dee2aaSAndroid Build Coastguard Worker     this->initializeSwizzler(this->dstInfo(), this->options(), needsCMYKToRGB);
607*c8dee2aaSAndroid Build Coastguard Worker     if (!this->allocateStorage(this->dstInfo())) {
608*c8dee2aaSAndroid Build Coastguard Worker         return nullptr;
609*c8dee2aaSAndroid Build Coastguard Worker     }
610*c8dee2aaSAndroid Build Coastguard Worker     return fSwizzler.get();
611*c8dee2aaSAndroid Build Coastguard Worker }
612*c8dee2aaSAndroid Build Coastguard Worker 
onStartScanlineDecode(const SkImageInfo & dstInfo,const Options & options)613*c8dee2aaSAndroid Build Coastguard Worker SkCodec::Result SkJpegCodec::onStartScanlineDecode(const SkImageInfo& dstInfo,
614*c8dee2aaSAndroid Build Coastguard Worker         const Options& options) {
615*c8dee2aaSAndroid Build Coastguard Worker     // Set the jump location for libjpeg errors
616*c8dee2aaSAndroid Build Coastguard Worker     skjpeg_error_mgr::AutoPushJmpBuf jmp(fDecoderMgr->errorMgr());
617*c8dee2aaSAndroid Build Coastguard Worker     if (setjmp(jmp)) {
618*c8dee2aaSAndroid Build Coastguard Worker         SkCodecPrintf("setjmp: Error from libjpeg\n");
619*c8dee2aaSAndroid Build Coastguard Worker         return kInvalidInput;
620*c8dee2aaSAndroid Build Coastguard Worker     }
621*c8dee2aaSAndroid Build Coastguard Worker 
622*c8dee2aaSAndroid Build Coastguard Worker     if (!jpeg_start_decompress(fDecoderMgr->dinfo())) {
623*c8dee2aaSAndroid Build Coastguard Worker         SkCodecPrintf("start decompress failed\n");
624*c8dee2aaSAndroid Build Coastguard Worker         return kInvalidInput;
625*c8dee2aaSAndroid Build Coastguard Worker     }
626*c8dee2aaSAndroid Build Coastguard Worker 
627*c8dee2aaSAndroid Build Coastguard Worker     bool needsCMYKToRGB = needs_swizzler_to_convert_from_cmyk(
628*c8dee2aaSAndroid Build Coastguard Worker             fDecoderMgr->dinfo()->out_color_space, this->getEncodedInfo().profile(),
629*c8dee2aaSAndroid Build Coastguard Worker             this->colorXform());
630*c8dee2aaSAndroid Build Coastguard Worker     if (options.fSubset) {
631*c8dee2aaSAndroid Build Coastguard Worker         uint32_t startX = options.fSubset->x();
632*c8dee2aaSAndroid Build Coastguard Worker         uint32_t width = options.fSubset->width();
633*c8dee2aaSAndroid Build Coastguard Worker 
634*c8dee2aaSAndroid Build Coastguard Worker         // libjpeg-turbo may need to align startX to a multiple of the IDCT
635*c8dee2aaSAndroid Build Coastguard Worker         // block size.  If this is the case, it will decrease the value of
636*c8dee2aaSAndroid Build Coastguard Worker         // startX to the appropriate alignment and also increase the value
637*c8dee2aaSAndroid Build Coastguard Worker         // of width so that the right edge of the requested subset remains
638*c8dee2aaSAndroid Build Coastguard Worker         // the same.
639*c8dee2aaSAndroid Build Coastguard Worker         jpeg_crop_scanline(fDecoderMgr->dinfo(), &startX, &width);
640*c8dee2aaSAndroid Build Coastguard Worker 
641*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(startX <= (uint32_t) options.fSubset->x());
642*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(width >= (uint32_t) options.fSubset->width());
643*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(startX + width >= (uint32_t) options.fSubset->right());
644*c8dee2aaSAndroid Build Coastguard Worker 
645*c8dee2aaSAndroid Build Coastguard Worker         // Instruct the swizzler (if it is necessary) to further subset the
646*c8dee2aaSAndroid Build Coastguard Worker         // output provided by libjpeg-turbo.
647*c8dee2aaSAndroid Build Coastguard Worker         //
648*c8dee2aaSAndroid Build Coastguard Worker         // We set this here (rather than in the if statement below), so that
649*c8dee2aaSAndroid Build Coastguard Worker         // if (1) we don't need a swizzler for the subset, and (2) we need a
650*c8dee2aaSAndroid Build Coastguard Worker         // swizzler for CMYK, the swizzler will still use the proper subset
651*c8dee2aaSAndroid Build Coastguard Worker         // dimensions.
652*c8dee2aaSAndroid Build Coastguard Worker         //
653*c8dee2aaSAndroid Build Coastguard Worker         // Note that the swizzler will ignore the y and height parameters of
654*c8dee2aaSAndroid Build Coastguard Worker         // the subset.  Since the scanline decoder (and the swizzler) handle
655*c8dee2aaSAndroid Build Coastguard Worker         // one row at a time, only the subsetting in the x-dimension matters.
656*c8dee2aaSAndroid Build Coastguard Worker         fSwizzlerSubset.setXYWH(options.fSubset->x() - startX, 0,
657*c8dee2aaSAndroid Build Coastguard Worker                 options.fSubset->width(), options.fSubset->height());
658*c8dee2aaSAndroid Build Coastguard Worker 
659*c8dee2aaSAndroid Build Coastguard Worker         // We will need a swizzler if libjpeg-turbo cannot provide the exact
660*c8dee2aaSAndroid Build Coastguard Worker         // subset that we request.
661*c8dee2aaSAndroid Build Coastguard Worker         if (startX != (uint32_t) options.fSubset->x() ||
662*c8dee2aaSAndroid Build Coastguard Worker                 width != (uint32_t) options.fSubset->width()) {
663*c8dee2aaSAndroid Build Coastguard Worker             this->initializeSwizzler(dstInfo, options, needsCMYKToRGB);
664*c8dee2aaSAndroid Build Coastguard Worker         }
665*c8dee2aaSAndroid Build Coastguard Worker     }
666*c8dee2aaSAndroid Build Coastguard Worker 
667*c8dee2aaSAndroid Build Coastguard Worker     // Make sure we have a swizzler if we are converting from CMYK.
668*c8dee2aaSAndroid Build Coastguard Worker     if (!fSwizzler && needsCMYKToRGB) {
669*c8dee2aaSAndroid Build Coastguard Worker         this->initializeSwizzler(dstInfo, options, true);
670*c8dee2aaSAndroid Build Coastguard Worker     }
671*c8dee2aaSAndroid Build Coastguard Worker 
672*c8dee2aaSAndroid Build Coastguard Worker     if (!this->allocateStorage(dstInfo)) {
673*c8dee2aaSAndroid Build Coastguard Worker         return kInternalError;
674*c8dee2aaSAndroid Build Coastguard Worker     }
675*c8dee2aaSAndroid Build Coastguard Worker 
676*c8dee2aaSAndroid Build Coastguard Worker     return kSuccess;
677*c8dee2aaSAndroid Build Coastguard Worker }
678*c8dee2aaSAndroid Build Coastguard Worker 
onGetScanlines(void * dst,int count,size_t dstRowBytes)679*c8dee2aaSAndroid Build Coastguard Worker int SkJpegCodec::onGetScanlines(void* dst, int count, size_t dstRowBytes) {
680*c8dee2aaSAndroid Build Coastguard Worker     int rows = this->readRows(this->dstInfo(), dst, dstRowBytes, count, this->options());
681*c8dee2aaSAndroid Build Coastguard Worker     if (rows < count) {
682*c8dee2aaSAndroid Build Coastguard Worker         // This allows us to skip calling jpeg_finish_decompress().
683*c8dee2aaSAndroid Build Coastguard Worker         fDecoderMgr->dinfo()->output_scanline = this->dstInfo().height();
684*c8dee2aaSAndroid Build Coastguard Worker     }
685*c8dee2aaSAndroid Build Coastguard Worker 
686*c8dee2aaSAndroid Build Coastguard Worker     return rows;
687*c8dee2aaSAndroid Build Coastguard Worker }
688*c8dee2aaSAndroid Build Coastguard Worker 
onSkipScanlines(int count)689*c8dee2aaSAndroid Build Coastguard Worker bool SkJpegCodec::onSkipScanlines(int count) {
690*c8dee2aaSAndroid Build Coastguard Worker     // Set the jump location for libjpeg errors
691*c8dee2aaSAndroid Build Coastguard Worker     skjpeg_error_mgr::AutoPushJmpBuf jmp(fDecoderMgr->errorMgr());
692*c8dee2aaSAndroid Build Coastguard Worker     if (setjmp(jmp)) {
693*c8dee2aaSAndroid Build Coastguard Worker         return fDecoderMgr->returnFalse("onSkipScanlines");
694*c8dee2aaSAndroid Build Coastguard Worker     }
695*c8dee2aaSAndroid Build Coastguard Worker 
696*c8dee2aaSAndroid Build Coastguard Worker     return (uint32_t) count == jpeg_skip_scanlines(fDecoderMgr->dinfo(), count);
697*c8dee2aaSAndroid Build Coastguard Worker }
698*c8dee2aaSAndroid Build Coastguard Worker 
is_yuv_supported(const jpeg_decompress_struct * dinfo,const SkJpegCodec & codec,const SkYUVAPixmapInfo::SupportedDataTypes * supportedDataTypes,SkYUVAPixmapInfo * yuvaPixmapInfo)699*c8dee2aaSAndroid Build Coastguard Worker static bool is_yuv_supported(const jpeg_decompress_struct* dinfo,
700*c8dee2aaSAndroid Build Coastguard Worker                              const SkJpegCodec& codec,
701*c8dee2aaSAndroid Build Coastguard Worker                              const SkYUVAPixmapInfo::SupportedDataTypes* supportedDataTypes,
702*c8dee2aaSAndroid Build Coastguard Worker                              SkYUVAPixmapInfo* yuvaPixmapInfo) {
703*c8dee2aaSAndroid Build Coastguard Worker     // Scaling is not supported in raw data mode.
704*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(dinfo->scale_num == dinfo->scale_denom);
705*c8dee2aaSAndroid Build Coastguard Worker 
706*c8dee2aaSAndroid Build Coastguard Worker     // I can't imagine that this would ever change, but we do depend on it.
707*c8dee2aaSAndroid Build Coastguard Worker     static_assert(8 == DCTSIZE, "DCTSIZE (defined in jpeg library) should always be 8.");
708*c8dee2aaSAndroid Build Coastguard Worker 
709*c8dee2aaSAndroid Build Coastguard Worker     if (JCS_YCbCr != dinfo->jpeg_color_space) {
710*c8dee2aaSAndroid Build Coastguard Worker         return false;
711*c8dee2aaSAndroid Build Coastguard Worker     }
712*c8dee2aaSAndroid Build Coastguard Worker 
713*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(3 == dinfo->num_components);
714*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(dinfo->comp_info);
715*c8dee2aaSAndroid Build Coastguard Worker 
716*c8dee2aaSAndroid Build Coastguard Worker     // It is possible to perform a YUV decode for any combination of
717*c8dee2aaSAndroid Build Coastguard Worker     // horizontal and vertical sampling that is supported by
718*c8dee2aaSAndroid Build Coastguard Worker     // libjpeg/libjpeg-turbo.  However, we will start by supporting only the
719*c8dee2aaSAndroid Build Coastguard Worker     // common cases (where U and V have samp_factors of one).
720*c8dee2aaSAndroid Build Coastguard Worker     //
721*c8dee2aaSAndroid Build Coastguard Worker     // The definition of samp_factor is kind of the opposite of what SkCodec
722*c8dee2aaSAndroid Build Coastguard Worker     // thinks of as a sampling factor.  samp_factor is essentially a
723*c8dee2aaSAndroid Build Coastguard Worker     // multiplier, and the larger the samp_factor is, the more samples that
724*c8dee2aaSAndroid Build Coastguard Worker     // there will be.  Ex:
725*c8dee2aaSAndroid Build Coastguard Worker     //     U_plane_width = image_width * (U_h_samp_factor / max_h_samp_factor)
726*c8dee2aaSAndroid Build Coastguard Worker     //
727*c8dee2aaSAndroid Build Coastguard Worker     // Supporting cases where the samp_factors for U or V were larger than
728*c8dee2aaSAndroid Build Coastguard Worker     // that of Y would be an extremely difficult change, given that clients
729*c8dee2aaSAndroid Build Coastguard Worker     // allocate memory as if the size of the Y plane is always the size of the
730*c8dee2aaSAndroid Build Coastguard Worker     // image.  However, this case is very, very rare.
731*c8dee2aaSAndroid Build Coastguard Worker     if  ((1 != dinfo->comp_info[1].h_samp_factor) ||
732*c8dee2aaSAndroid Build Coastguard Worker          (1 != dinfo->comp_info[1].v_samp_factor) ||
733*c8dee2aaSAndroid Build Coastguard Worker          (1 != dinfo->comp_info[2].h_samp_factor) ||
734*c8dee2aaSAndroid Build Coastguard Worker          (1 != dinfo->comp_info[2].v_samp_factor))
735*c8dee2aaSAndroid Build Coastguard Worker     {
736*c8dee2aaSAndroid Build Coastguard Worker         return false;
737*c8dee2aaSAndroid Build Coastguard Worker     }
738*c8dee2aaSAndroid Build Coastguard Worker 
739*c8dee2aaSAndroid Build Coastguard Worker     // Support all common cases of Y samp_factors.
740*c8dee2aaSAndroid Build Coastguard Worker     // TODO (msarett): As mentioned above, it would be possible to support
741*c8dee2aaSAndroid Build Coastguard Worker     //                 more combinations of samp_factors.  The issues are:
742*c8dee2aaSAndroid Build Coastguard Worker     //                 (1) Are there actually any images that are not covered
743*c8dee2aaSAndroid Build Coastguard Worker     //                     by these cases?
744*c8dee2aaSAndroid Build Coastguard Worker     //                 (2) How much complexity would be added to the
745*c8dee2aaSAndroid Build Coastguard Worker     //                     implementation in order to support these rare
746*c8dee2aaSAndroid Build Coastguard Worker     //                     cases?
747*c8dee2aaSAndroid Build Coastguard Worker     int hSampY = dinfo->comp_info[0].h_samp_factor;
748*c8dee2aaSAndroid Build Coastguard Worker     int vSampY = dinfo->comp_info[0].v_samp_factor;
749*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(hSampY == dinfo->max_h_samp_factor);
750*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(vSampY == dinfo->max_v_samp_factor);
751*c8dee2aaSAndroid Build Coastguard Worker 
752*c8dee2aaSAndroid Build Coastguard Worker     SkYUVAInfo::Subsampling tempSubsampling;
753*c8dee2aaSAndroid Build Coastguard Worker     if        (1 == hSampY && 1 == vSampY) {
754*c8dee2aaSAndroid Build Coastguard Worker         tempSubsampling = SkYUVAInfo::Subsampling::k444;
755*c8dee2aaSAndroid Build Coastguard Worker     } else if (2 == hSampY && 1 == vSampY) {
756*c8dee2aaSAndroid Build Coastguard Worker         tempSubsampling = SkYUVAInfo::Subsampling::k422;
757*c8dee2aaSAndroid Build Coastguard Worker     } else if (2 == hSampY && 2 == vSampY) {
758*c8dee2aaSAndroid Build Coastguard Worker         tempSubsampling = SkYUVAInfo::Subsampling::k420;
759*c8dee2aaSAndroid Build Coastguard Worker     } else if (1 == hSampY && 2 == vSampY) {
760*c8dee2aaSAndroid Build Coastguard Worker         tempSubsampling = SkYUVAInfo::Subsampling::k440;
761*c8dee2aaSAndroid Build Coastguard Worker     } else if (4 == hSampY && 1 == vSampY) {
762*c8dee2aaSAndroid Build Coastguard Worker         tempSubsampling = SkYUVAInfo::Subsampling::k411;
763*c8dee2aaSAndroid Build Coastguard Worker     } else if (4 == hSampY && 2 == vSampY) {
764*c8dee2aaSAndroid Build Coastguard Worker         tempSubsampling = SkYUVAInfo::Subsampling::k410;
765*c8dee2aaSAndroid Build Coastguard Worker     } else {
766*c8dee2aaSAndroid Build Coastguard Worker         return false;
767*c8dee2aaSAndroid Build Coastguard Worker     }
768*c8dee2aaSAndroid Build Coastguard Worker     if (supportedDataTypes &&
769*c8dee2aaSAndroid Build Coastguard Worker         !supportedDataTypes->supported(SkYUVAInfo::PlaneConfig::kY_U_V,
770*c8dee2aaSAndroid Build Coastguard Worker                                        SkYUVAPixmapInfo::DataType::kUnorm8)) {
771*c8dee2aaSAndroid Build Coastguard Worker         return false;
772*c8dee2aaSAndroid Build Coastguard Worker     }
773*c8dee2aaSAndroid Build Coastguard Worker     if (yuvaPixmapInfo) {
774*c8dee2aaSAndroid Build Coastguard Worker         SkColorType colorTypes[SkYUVAPixmapInfo::kMaxPlanes];
775*c8dee2aaSAndroid Build Coastguard Worker         size_t rowBytes[SkYUVAPixmapInfo::kMaxPlanes];
776*c8dee2aaSAndroid Build Coastguard Worker         for (int i = 0; i < 3; ++i) {
777*c8dee2aaSAndroid Build Coastguard Worker             colorTypes[i] = kAlpha_8_SkColorType;
778*c8dee2aaSAndroid Build Coastguard Worker             rowBytes[i] = dinfo->comp_info[i].width_in_blocks * DCTSIZE;
779*c8dee2aaSAndroid Build Coastguard Worker         }
780*c8dee2aaSAndroid Build Coastguard Worker         SkYUVAInfo yuvaInfo(codec.dimensions(),
781*c8dee2aaSAndroid Build Coastguard Worker                             SkYUVAInfo::PlaneConfig::kY_U_V,
782*c8dee2aaSAndroid Build Coastguard Worker                             tempSubsampling,
783*c8dee2aaSAndroid Build Coastguard Worker                             kJPEG_Full_SkYUVColorSpace,
784*c8dee2aaSAndroid Build Coastguard Worker                             codec.getOrigin(),
785*c8dee2aaSAndroid Build Coastguard Worker                             SkYUVAInfo::Siting::kCentered,
786*c8dee2aaSAndroid Build Coastguard Worker                             SkYUVAInfo::Siting::kCentered);
787*c8dee2aaSAndroid Build Coastguard Worker         *yuvaPixmapInfo = SkYUVAPixmapInfo(yuvaInfo, colorTypes, rowBytes);
788*c8dee2aaSAndroid Build Coastguard Worker     }
789*c8dee2aaSAndroid Build Coastguard Worker     return true;
790*c8dee2aaSAndroid Build Coastguard Worker }
791*c8dee2aaSAndroid Build Coastguard Worker 
onQueryYUVAInfo(const SkYUVAPixmapInfo::SupportedDataTypes & supportedDataTypes,SkYUVAPixmapInfo * yuvaPixmapInfo) const792*c8dee2aaSAndroid Build Coastguard Worker bool SkJpegCodec::onQueryYUVAInfo(const SkYUVAPixmapInfo::SupportedDataTypes& supportedDataTypes,
793*c8dee2aaSAndroid Build Coastguard Worker                                   SkYUVAPixmapInfo* yuvaPixmapInfo) const {
794*c8dee2aaSAndroid Build Coastguard Worker     jpeg_decompress_struct* dinfo = fDecoderMgr->dinfo();
795*c8dee2aaSAndroid Build Coastguard Worker     return is_yuv_supported(dinfo, *this, &supportedDataTypes, yuvaPixmapInfo);
796*c8dee2aaSAndroid Build Coastguard Worker }
797*c8dee2aaSAndroid Build Coastguard Worker 
onGetYUVAPlanes(const SkYUVAPixmaps & yuvaPixmaps)798*c8dee2aaSAndroid Build Coastguard Worker SkCodec::Result SkJpegCodec::onGetYUVAPlanes(const SkYUVAPixmaps& yuvaPixmaps) {
799*c8dee2aaSAndroid Build Coastguard Worker     // Get a pointer to the decompress info since we will use it quite frequently
800*c8dee2aaSAndroid Build Coastguard Worker     jpeg_decompress_struct* dinfo = fDecoderMgr->dinfo();
801*c8dee2aaSAndroid Build Coastguard Worker     if (!is_yuv_supported(dinfo, *this, nullptr, nullptr)) {
802*c8dee2aaSAndroid Build Coastguard Worker         return fDecoderMgr->returnFailure("onGetYUVAPlanes", kInvalidInput);
803*c8dee2aaSAndroid Build Coastguard Worker     }
804*c8dee2aaSAndroid Build Coastguard Worker     // Set the jump location for libjpeg errors
805*c8dee2aaSAndroid Build Coastguard Worker     skjpeg_error_mgr::AutoPushJmpBuf jmp(fDecoderMgr->errorMgr());
806*c8dee2aaSAndroid Build Coastguard Worker     if (setjmp(jmp)) {
807*c8dee2aaSAndroid Build Coastguard Worker         return fDecoderMgr->returnFailure("setjmp", kInvalidInput);
808*c8dee2aaSAndroid Build Coastguard Worker     }
809*c8dee2aaSAndroid Build Coastguard Worker 
810*c8dee2aaSAndroid Build Coastguard Worker     dinfo->raw_data_out = TRUE;
811*c8dee2aaSAndroid Build Coastguard Worker     if (!jpeg_start_decompress(dinfo)) {
812*c8dee2aaSAndroid Build Coastguard Worker         return fDecoderMgr->returnFailure("startDecompress", kInvalidInput);
813*c8dee2aaSAndroid Build Coastguard Worker     }
814*c8dee2aaSAndroid Build Coastguard Worker 
815*c8dee2aaSAndroid Build Coastguard Worker     const std::array<SkPixmap, SkYUVAPixmaps::kMaxPlanes>& planes = yuvaPixmaps.planes();
816*c8dee2aaSAndroid Build Coastguard Worker 
817*c8dee2aaSAndroid Build Coastguard Worker #ifdef SK_DEBUG
818*c8dee2aaSAndroid Build Coastguard Worker     {
819*c8dee2aaSAndroid Build Coastguard Worker         // A previous implementation claims that the return value of is_yuv_supported()
820*c8dee2aaSAndroid Build Coastguard Worker         // may change after calling jpeg_start_decompress().  It looks to me like this
821*c8dee2aaSAndroid Build Coastguard Worker         // was caused by a bug in the old code, but we'll be safe and check here.
822*c8dee2aaSAndroid Build Coastguard Worker         // Also check that pixmap properties agree with expectations.
823*c8dee2aaSAndroid Build Coastguard Worker         SkYUVAPixmapInfo info;
824*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(is_yuv_supported(dinfo, *this, nullptr, &info));
825*c8dee2aaSAndroid Build Coastguard Worker         SkASSERT(info.yuvaInfo() == yuvaPixmaps.yuvaInfo());
826*c8dee2aaSAndroid Build Coastguard Worker         for (int i = 0; i < info.numPlanes(); ++i) {
827*c8dee2aaSAndroid Build Coastguard Worker             SkASSERT(planes[i].colorType() == kAlpha_8_SkColorType);
828*c8dee2aaSAndroid Build Coastguard Worker             SkASSERT(info.planeInfo(i) == planes[i].info());
829*c8dee2aaSAndroid Build Coastguard Worker         }
830*c8dee2aaSAndroid Build Coastguard Worker     }
831*c8dee2aaSAndroid Build Coastguard Worker #endif
832*c8dee2aaSAndroid Build Coastguard Worker 
833*c8dee2aaSAndroid Build Coastguard Worker     // Build a JSAMPIMAGE to handle output from libjpeg-turbo.  A JSAMPIMAGE has
834*c8dee2aaSAndroid Build Coastguard Worker     // a 2-D array of pixels for each of the components (Y, U, V) in the image.
835*c8dee2aaSAndroid Build Coastguard Worker     // Cheat Sheet:
836*c8dee2aaSAndroid Build Coastguard Worker     //     JSAMPIMAGE == JSAMPLEARRAY* == JSAMPROW** == JSAMPLE***
837*c8dee2aaSAndroid Build Coastguard Worker     JSAMPARRAY yuv[3];
838*c8dee2aaSAndroid Build Coastguard Worker 
839*c8dee2aaSAndroid Build Coastguard Worker     // Set aside enough space for pointers to rows of Y, U, and V.
840*c8dee2aaSAndroid Build Coastguard Worker     JSAMPROW rowptrs[2 * DCTSIZE + DCTSIZE + DCTSIZE];
841*c8dee2aaSAndroid Build Coastguard Worker     yuv[0] = &rowptrs[0];            // Y rows (DCTSIZE or 2 * DCTSIZE)
842*c8dee2aaSAndroid Build Coastguard Worker     yuv[1] = &rowptrs[2 * DCTSIZE];  // U rows (DCTSIZE)
843*c8dee2aaSAndroid Build Coastguard Worker     yuv[2] = &rowptrs[3 * DCTSIZE];  // V rows (DCTSIZE)
844*c8dee2aaSAndroid Build Coastguard Worker 
845*c8dee2aaSAndroid Build Coastguard Worker     // Initialize rowptrs.
846*c8dee2aaSAndroid Build Coastguard Worker     int numYRowsPerBlock = DCTSIZE * dinfo->comp_info[0].v_samp_factor;
847*c8dee2aaSAndroid Build Coastguard Worker     static_assert(sizeof(JSAMPLE) == 1);
848*c8dee2aaSAndroid Build Coastguard Worker     for (int i = 0; i < numYRowsPerBlock; i++) {
849*c8dee2aaSAndroid Build Coastguard Worker         rowptrs[i] = static_cast<JSAMPLE*>(planes[0].writable_addr()) + i* planes[0].rowBytes();
850*c8dee2aaSAndroid Build Coastguard Worker     }
851*c8dee2aaSAndroid Build Coastguard Worker     for (int i = 0; i < DCTSIZE; i++) {
852*c8dee2aaSAndroid Build Coastguard Worker         rowptrs[i + 2 * DCTSIZE] =
853*c8dee2aaSAndroid Build Coastguard Worker                 static_cast<JSAMPLE*>(planes[1].writable_addr()) + i* planes[1].rowBytes();
854*c8dee2aaSAndroid Build Coastguard Worker         rowptrs[i + 3 * DCTSIZE] =
855*c8dee2aaSAndroid Build Coastguard Worker                 static_cast<JSAMPLE*>(planes[2].writable_addr()) + i* planes[2].rowBytes();
856*c8dee2aaSAndroid Build Coastguard Worker     }
857*c8dee2aaSAndroid Build Coastguard Worker 
858*c8dee2aaSAndroid Build Coastguard Worker     // After each loop iteration, we will increment pointers to Y, U, and V.
859*c8dee2aaSAndroid Build Coastguard Worker     size_t blockIncrementY = numYRowsPerBlock * planes[0].rowBytes();
860*c8dee2aaSAndroid Build Coastguard Worker     size_t blockIncrementU = DCTSIZE * planes[1].rowBytes();
861*c8dee2aaSAndroid Build Coastguard Worker     size_t blockIncrementV = DCTSIZE * planes[2].rowBytes();
862*c8dee2aaSAndroid Build Coastguard Worker 
863*c8dee2aaSAndroid Build Coastguard Worker     uint32_t numRowsPerBlock = numYRowsPerBlock;
864*c8dee2aaSAndroid Build Coastguard Worker 
865*c8dee2aaSAndroid Build Coastguard Worker     // We intentionally round down here, as this first loop will only handle
866*c8dee2aaSAndroid Build Coastguard Worker     // full block rows.  As a special case at the end, we will handle any
867*c8dee2aaSAndroid Build Coastguard Worker     // remaining rows that do not make up a full block.
868*c8dee2aaSAndroid Build Coastguard Worker     const int numIters = dinfo->output_height / numRowsPerBlock;
869*c8dee2aaSAndroid Build Coastguard Worker     for (int i = 0; i < numIters; i++) {
870*c8dee2aaSAndroid Build Coastguard Worker         JDIMENSION linesRead = jpeg_read_raw_data(dinfo, yuv, numRowsPerBlock);
871*c8dee2aaSAndroid Build Coastguard Worker         if (linesRead < numRowsPerBlock) {
872*c8dee2aaSAndroid Build Coastguard Worker             // FIXME: Handle incomplete YUV decodes without signalling an error.
873*c8dee2aaSAndroid Build Coastguard Worker             return kInvalidInput;
874*c8dee2aaSAndroid Build Coastguard Worker         }
875*c8dee2aaSAndroid Build Coastguard Worker 
876*c8dee2aaSAndroid Build Coastguard Worker         // Update rowptrs.
877*c8dee2aaSAndroid Build Coastguard Worker         for (int j = 0; j < numYRowsPerBlock; j++) {
878*c8dee2aaSAndroid Build Coastguard Worker             rowptrs[j] += blockIncrementY;
879*c8dee2aaSAndroid Build Coastguard Worker         }
880*c8dee2aaSAndroid Build Coastguard Worker         for (int j = 0; j < DCTSIZE; j++) {
881*c8dee2aaSAndroid Build Coastguard Worker             rowptrs[j + 2 * DCTSIZE] += blockIncrementU;
882*c8dee2aaSAndroid Build Coastguard Worker             rowptrs[j + 3 * DCTSIZE] += blockIncrementV;
883*c8dee2aaSAndroid Build Coastguard Worker         }
884*c8dee2aaSAndroid Build Coastguard Worker     }
885*c8dee2aaSAndroid Build Coastguard Worker 
886*c8dee2aaSAndroid Build Coastguard Worker     uint32_t remainingRows = dinfo->output_height - dinfo->output_scanline;
887*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(remainingRows == dinfo->output_height % numRowsPerBlock);
888*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(dinfo->output_scanline == numIters * numRowsPerBlock);
889*c8dee2aaSAndroid Build Coastguard Worker     if (remainingRows > 0) {
890*c8dee2aaSAndroid Build Coastguard Worker         // libjpeg-turbo needs memory to be padded by the block sizes.  We will fulfill
891*c8dee2aaSAndroid Build Coastguard Worker         // this requirement using an extra row buffer.
892*c8dee2aaSAndroid Build Coastguard Worker         // FIXME: Should SkCodec have an extra memory buffer that can be shared among
893*c8dee2aaSAndroid Build Coastguard Worker         //        all of the implementations that use temporary/garbage memory?
894*c8dee2aaSAndroid Build Coastguard Worker         AutoTMalloc<JSAMPLE> extraRow(planes[0].rowBytes());
895*c8dee2aaSAndroid Build Coastguard Worker         for (int i = remainingRows; i < numYRowsPerBlock; i++) {
896*c8dee2aaSAndroid Build Coastguard Worker             rowptrs[i] = extraRow.get();
897*c8dee2aaSAndroid Build Coastguard Worker         }
898*c8dee2aaSAndroid Build Coastguard Worker         int remainingUVRows = dinfo->comp_info[1].downsampled_height - DCTSIZE * numIters;
899*c8dee2aaSAndroid Build Coastguard Worker         for (int i = remainingUVRows; i < DCTSIZE; i++) {
900*c8dee2aaSAndroid Build Coastguard Worker             rowptrs[i + 2 * DCTSIZE] = extraRow.get();
901*c8dee2aaSAndroid Build Coastguard Worker             rowptrs[i + 3 * DCTSIZE] = extraRow.get();
902*c8dee2aaSAndroid Build Coastguard Worker         }
903*c8dee2aaSAndroid Build Coastguard Worker 
904*c8dee2aaSAndroid Build Coastguard Worker         JDIMENSION linesRead = jpeg_read_raw_data(dinfo, yuv, numRowsPerBlock);
905*c8dee2aaSAndroid Build Coastguard Worker         if (linesRead < remainingRows) {
906*c8dee2aaSAndroid Build Coastguard Worker             // FIXME: Handle incomplete YUV decodes without signalling an error.
907*c8dee2aaSAndroid Build Coastguard Worker             return kInvalidInput;
908*c8dee2aaSAndroid Build Coastguard Worker         }
909*c8dee2aaSAndroid Build Coastguard Worker     }
910*c8dee2aaSAndroid Build Coastguard Worker 
911*c8dee2aaSAndroid Build Coastguard Worker     return kSuccess;
912*c8dee2aaSAndroid Build Coastguard Worker }
913*c8dee2aaSAndroid Build Coastguard Worker 
onGetGainmapCodec(SkGainmapInfo * info,std::unique_ptr<SkCodec> * gainmapCodec)914*c8dee2aaSAndroid Build Coastguard Worker bool SkJpegCodec::onGetGainmapCodec(SkGainmapInfo* info, std::unique_ptr<SkCodec>* gainmapCodec) {
915*c8dee2aaSAndroid Build Coastguard Worker     std::unique_ptr<SkStream> stream;
916*c8dee2aaSAndroid Build Coastguard Worker     if (!this->onGetGainmapInfo(info, &stream)) {
917*c8dee2aaSAndroid Build Coastguard Worker         return false;
918*c8dee2aaSAndroid Build Coastguard Worker     }
919*c8dee2aaSAndroid Build Coastguard Worker     if (gainmapCodec) {
920*c8dee2aaSAndroid Build Coastguard Worker         Result result;
921*c8dee2aaSAndroid Build Coastguard Worker         *gainmapCodec = MakeFromStream(std::move(stream), &result);
922*c8dee2aaSAndroid Build Coastguard Worker         if (!*gainmapCodec) {
923*c8dee2aaSAndroid Build Coastguard Worker             return false;
924*c8dee2aaSAndroid Build Coastguard Worker         }
925*c8dee2aaSAndroid Build Coastguard Worker     }
926*c8dee2aaSAndroid Build Coastguard Worker     return true;
927*c8dee2aaSAndroid Build Coastguard Worker }
928*c8dee2aaSAndroid Build Coastguard Worker 
onGetGainmapInfo(SkGainmapInfo * info,std::unique_ptr<SkStream> * gainmapImageStream)929*c8dee2aaSAndroid Build Coastguard Worker bool SkJpegCodec::onGetGainmapInfo(SkGainmapInfo* info,
930*c8dee2aaSAndroid Build Coastguard Worker                                    std::unique_ptr<SkStream>* gainmapImageStream) {
931*c8dee2aaSAndroid Build Coastguard Worker #ifdef SK_CODEC_DECODES_JPEG_GAINMAPS
932*c8dee2aaSAndroid Build Coastguard Worker     sk_sp<SkData> gainmap_data;
933*c8dee2aaSAndroid Build Coastguard Worker     SkGainmapInfo gainmap_info;
934*c8dee2aaSAndroid Build Coastguard Worker 
935*c8dee2aaSAndroid Build Coastguard Worker     auto metadataDecoder =
936*c8dee2aaSAndroid Build Coastguard Worker             std::make_unique<SkJpegMetadataDecoderImpl>(get_sk_marker_list(fDecoderMgr->dinfo()));
937*c8dee2aaSAndroid Build Coastguard Worker     if (!metadataDecoder->findGainmapImage(
938*c8dee2aaSAndroid Build Coastguard Worker                 fDecoderMgr->getSourceMgr(), gainmap_data, gainmap_info)) {
939*c8dee2aaSAndroid Build Coastguard Worker         return false;
940*c8dee2aaSAndroid Build Coastguard Worker     }
941*c8dee2aaSAndroid Build Coastguard Worker 
942*c8dee2aaSAndroid Build Coastguard Worker     *info = gainmap_info;
943*c8dee2aaSAndroid Build Coastguard Worker     *gainmapImageStream = SkMemoryStream::Make(gainmap_data);
944*c8dee2aaSAndroid Build Coastguard Worker     return true;
945*c8dee2aaSAndroid Build Coastguard Worker #else
946*c8dee2aaSAndroid Build Coastguard Worker     return false;
947*c8dee2aaSAndroid Build Coastguard Worker #endif  // SK_CODEC_DECODES_JPEG_GAINMAPS
948*c8dee2aaSAndroid Build Coastguard Worker }
949*c8dee2aaSAndroid Build Coastguard Worker 
950*c8dee2aaSAndroid Build Coastguard Worker namespace SkJpegDecoder {
IsJpeg(const void * data,size_t len)951*c8dee2aaSAndroid Build Coastguard Worker bool IsJpeg(const void* data, size_t len) {
952*c8dee2aaSAndroid Build Coastguard Worker     return SkJpegCodec::IsJpeg(data, len);
953*c8dee2aaSAndroid Build Coastguard Worker }
954*c8dee2aaSAndroid Build Coastguard Worker 
Decode(std::unique_ptr<SkStream> stream,SkCodec::Result * outResult,SkCodecs::DecodeContext)955*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<SkCodec> Decode(std::unique_ptr<SkStream> stream,
956*c8dee2aaSAndroid Build Coastguard Worker                                 SkCodec::Result* outResult,
957*c8dee2aaSAndroid Build Coastguard Worker                                 SkCodecs::DecodeContext) {
958*c8dee2aaSAndroid Build Coastguard Worker     SkCodec::Result resultStorage;
959*c8dee2aaSAndroid Build Coastguard Worker     if (!outResult) {
960*c8dee2aaSAndroid Build Coastguard Worker         outResult = &resultStorage;
961*c8dee2aaSAndroid Build Coastguard Worker     }
962*c8dee2aaSAndroid Build Coastguard Worker     return SkJpegCodec::MakeFromStream(std::move(stream), outResult);
963*c8dee2aaSAndroid Build Coastguard Worker }
964*c8dee2aaSAndroid Build Coastguard Worker 
Decode(sk_sp<SkData> data,SkCodec::Result * outResult,SkCodecs::DecodeContext)965*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<SkCodec> Decode(sk_sp<SkData> data,
966*c8dee2aaSAndroid Build Coastguard Worker                                 SkCodec::Result* outResult,
967*c8dee2aaSAndroid Build Coastguard Worker                                 SkCodecs::DecodeContext) {
968*c8dee2aaSAndroid Build Coastguard Worker     if (!data) {
969*c8dee2aaSAndroid Build Coastguard Worker         if (outResult) {
970*c8dee2aaSAndroid Build Coastguard Worker             *outResult = SkCodec::kInvalidInput;
971*c8dee2aaSAndroid Build Coastguard Worker         }
972*c8dee2aaSAndroid Build Coastguard Worker         return nullptr;
973*c8dee2aaSAndroid Build Coastguard Worker     }
974*c8dee2aaSAndroid Build Coastguard Worker     return Decode(SkMemoryStream::Make(std::move(data)), outResult, nullptr);
975*c8dee2aaSAndroid Build Coastguard Worker }
976*c8dee2aaSAndroid Build Coastguard Worker 
977*c8dee2aaSAndroid Build Coastguard Worker }  // namespace SkJpegDecoder
978