blob: bb5ce75375fa899c238dfd0dbeeaafcb31814242 [file] [log] [blame]
msarette16b04a2015-04-15 07:32:19 -07001/*
2 * Copyright 2015 Google Inc.
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
8#ifndef SkJpegCodec_DEFINED
9#define SkJpegCodec_DEFINED
10
11#include "SkCodec.h"
12#include "SkImageInfo.h"
msarett39b2d5a2016-02-17 08:26:31 -080013#include "SkSwizzler.h"
msarette16b04a2015-04-15 07:32:19 -070014#include "SkStream.h"
scroggo565901d2015-12-10 10:44:13 -080015#include "SkTemplates.h"
msarette16b04a2015-04-15 07:32:19 -070016
msarett39b2d5a2016-02-17 08:26:31 -080017class JpegDecoderMgr;
msarette16b04a2015-04-15 07:32:19 -070018
19/*
20 *
21 * This class implements the decoding for jpeg images
22 *
23 */
24class SkJpegCodec : public SkCodec {
25public:
scroggodb30be22015-12-08 18:54:13 -080026 static bool IsJpeg(const void*, size_t);
msarette16b04a2015-04-15 07:32:19 -070027
28 /*
29 * Assumes IsJpeg was called and returned true
30 * Creates a jpeg decoder
31 * Takes ownership of the stream
32 */
33 static SkCodec* NewFromStream(SkStream*);
34
35protected:
36
37 /*
38 * Recommend a set of destination dimensions given a requested scale
39 */
40 SkISize onGetScaledDimensions(float desiredScale) const override;
41
42 /*
43 * Initiates the jpeg decode
44 */
45 Result onGetPixels(const SkImageInfo& dstInfo, void* dst, size_t dstRowBytes, const Options&,
msarette6dd0042015-10-09 11:07:34 -070046 SkPMColor*, int*, int*) override;
msarette16b04a2015-04-15 07:32:19 -070047
msarettb714fb02016-01-22 14:46:42 -080048 bool onQueryYUV8(YUVSizeInfo* sizeInfo, SkYUVColorSpace* colorSpace) const override;
49
50 Result onGetYUV8Planes(const YUVSizeInfo& sizeInfo, void* pixels[3]) override;
51
msarette16b04a2015-04-15 07:32:19 -070052 SkEncodedFormat onGetEncodedFormat() const override {
53 return kJPEG_SkEncodedFormat;
54 }
55
scroggob427db12015-08-12 07:24:13 -070056 bool onRewind() override;
57
scroggoe7fc14b2015-10-02 13:14:46 -070058 bool onDimensionsSupported(const SkISize&) override;
59
msarette16b04a2015-04-15 07:32:19 -070060private:
61
62 /*
63 * Read enough of the stream to initialize the SkJpegCodec.
64 * Returns a bool representing success or failure.
65 *
66 * @param codecOut
halcanary96fcdcc2015-08-27 07:41:13 -070067 * If this returns true, and codecOut was not nullptr,
msarette16b04a2015-04-15 07:32:19 -070068 * codecOut will be set to a new SkJpegCodec.
69 *
70 * @param decoderMgrOut
halcanary96fcdcc2015-08-27 07:41:13 -070071 * If this returns true, and codecOut was nullptr,
72 * decoderMgrOut must be non-nullptr and decoderMgrOut will be set to a new
msarette16b04a2015-04-15 07:32:19 -070073 * JpegDecoderMgr pointer.
74 *
75 * @param stream
76 * Deleted on failure.
77 * codecOut will take ownership of it in the case where we created a codec.
78 * Ownership is unchanged when we set decoderMgrOut.
79 *
80 */
81 static bool ReadHeader(SkStream* stream, SkCodec** codecOut,
82 JpegDecoderMgr** decoderMgrOut);
83
84 /*
85 * Creates an instance of the decoder
86 * Called only by NewFromStream
87 *
88 * @param srcInfo contains the source width and height
89 * @param stream the encoded image data
90 * @param decoderMgr holds decompress struct, src manager, and error manager
91 * takes ownership
92 */
93 SkJpegCodec(const SkImageInfo& srcInfo, SkStream* stream, JpegDecoderMgr* decoderMgr);
94
msarett97fdea62015-04-29 08:17:15 -070095 /*
msarett1c8a5872015-07-07 08:50:01 -070096 * Checks if the conversion between the input image and the requested output
97 * image has been implemented
98 * Sets the output color space
99 */
100 bool setOutputColorSpace(const SkImageInfo& dst);
101
scroggo46c57472015-09-30 08:57:13 -0700102 // scanline decoding
msarettfdb47572015-10-13 12:50:14 -0700103 void initializeSwizzler(const SkImageInfo& dstInfo, const Options& options);
msarette6dd0042015-10-09 11:07:34 -0700104 SkSampler* getSampler(bool createIfNecessary) override;
scroggo46c57472015-09-30 08:57:13 -0700105 Result onStartScanlineDecode(const SkImageInfo& dstInfo, const Options& options,
msarettfdb47572015-10-13 12:50:14 -0700106 SkPMColor ctable[], int* ctableCount) override;
msarette6dd0042015-10-09 11:07:34 -0700107 int onGetScanlines(void* dst, int count, size_t rowBytes) override;
108 bool onSkipScanlines(int count) override;
scroggo46c57472015-09-30 08:57:13 -0700109
msarette16b04a2015-04-15 07:32:19 -0700110 SkAutoTDelete<JpegDecoderMgr> fDecoderMgr;
msarettfbccb592015-09-01 06:43:41 -0700111 // We will save the state of the decompress struct after reading the header.
112 // This allows us to safely call onGetScaledDimensions() at any time.
113 const int fReadyState;
msarette16b04a2015-04-15 07:32:19 -0700114
scroggo46c57472015-09-30 08:57:13 -0700115 // scanline decoding
scroggo565901d2015-12-10 10:44:13 -0800116 SkAutoTMalloc<uint8_t> fStorage; // Only used if sampling is needed
scroggo46c57472015-09-30 08:57:13 -0700117 uint8_t* fSrcRow; // Only used if sampling is needed
msarett91c22b22016-02-22 12:27:46 -0800118 // libjpeg-turbo provides some subsetting. In the case that libjpeg-turbo
119 // cannot take the exact the subset that we need, we will use the swizzler
120 // to further subset the output from libjpeg-turbo.
121 SkIRect fSwizzlerSubset;
scroggo46c57472015-09-30 08:57:13 -0700122 SkAutoTDelete<SkSwizzler> fSwizzler;
123
msarette16b04a2015-04-15 07:32:19 -0700124 typedef SkCodec INHERITED;
125};
126
127#endif