1 /*
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 "SkColorSpace.h"
13 #include "SkColorSpaceXform.h"
14 #include "SkImageInfo.h"
15 #include "SkSwizzler.h"
16 #include "SkStream.h"
17 #include "SkTemplates.h"
18 
19 class JpegDecoderMgr;
20 
21 /*
22  *
23  * This class implements the decoding for jpeg images
24  *
25  */
26 class SkJpegCodec : public SkCodec {
27 public:
28     static bool IsJpeg(const void*, size_t);
29 
30     /*
31      * Assumes IsJpeg was called and returned true
32      * Takes ownership of the stream
33      */
34     static std::unique_ptr<SkCodec> MakeFromStream(std::unique_ptr<SkStream>, Result*);
35 
36 protected:
37 
38     /*
39      * Recommend a set of destination dimensions given a requested scale
40      */
41     SkISize onGetScaledDimensions(float desiredScale) const override;
42 
43     /*
44      * Initiates the jpeg decode
45      */
46     Result onGetPixels(const SkImageInfo& dstInfo, void* dst, size_t dstRowBytes, const Options&,
47             int*) override;
48 
49     bool onQueryYUV8(SkYUVSizeInfo* sizeInfo, SkYUVColorSpace* colorSpace) const override;
50 
51     Result onGetYUV8Planes(const SkYUVSizeInfo& sizeInfo, void* planes[3]) override;
52 
53     SkEncodedImageFormat onGetEncodedFormat() const override {
54         return SkEncodedImageFormat::kJPEG;
55     }
56 
57     bool onRewind() override;
58 
59     bool onDimensionsSupported(const SkISize&) override;
60 
61     bool conversionSupported(const SkImageInfo&, SkColorType, bool,
62                              const SkColorSpace*) const override {
63         // This class checks for conversion after creating colorXform.
64         return true;
65     }
66 
67 private:
68 
69     /*
70      * Allows SkRawCodec to communicate the color space from the exif data.
71      */
72     static std::unique_ptr<SkCodec> MakeFromStream(std::unique_ptr<SkStream>, Result*,
73                                                    sk_sp<SkColorSpace> defaultColorSpace);
74 
75     /*
76      * Read enough of the stream to initialize the SkJpegCodec.
77      * Returns a bool representing success or failure.
78      *
79      * @param codecOut
80      * If this returns true, and codecOut was not nullptr,
81      * codecOut will be set to a new SkJpegCodec.
82      *
83      * @param decoderMgrOut
84      * If this returns true, and codecOut was nullptr,
85      * decoderMgrOut must be non-nullptr and decoderMgrOut will be set to a new
86      * JpegDecoderMgr pointer.
87      *
88      * @param stream
89      * Deleted on failure.
90      * codecOut will take ownership of it in the case where we created a codec.
91      * Ownership is unchanged when we set decoderMgrOut.
92      *
93      * @param defaultColorSpace
94      * If the jpeg does not have an embedded color space, the image data should
95      * be tagged with this color space.
96      */
97     static Result ReadHeader(SkStream* stream, SkCodec** codecOut,
98             JpegDecoderMgr** decoderMgrOut, sk_sp<SkColorSpace> defaultColorSpace);
99 
100     /*
101      * Creates an instance of the decoder
102      * Called only by NewFromStream
103      *
104      * @param info contains properties of the encoded data
105      * @param stream the encoded image data
106      * @param decoderMgr holds decompress struct, src manager, and error manager
107      *                   takes ownership
108      */
109     SkJpegCodec(int width, int height, const SkEncodedInfo& info, std::unique_ptr<SkStream> stream,
110             JpegDecoderMgr* decoderMgr, sk_sp<SkColorSpace> colorSpace, SkEncodedOrigin origin);
111 
112     /*
113      * Checks if the conversion between the input image and the requested output
114      * image has been implemented.
115      *
116      * Sets the output color space.
117      */
118     bool setOutputColorSpace(const SkImageInfo& dst);
119 
120     void initializeSwizzler(const SkImageInfo& dstInfo, const Options& options,
121                             bool needsCMYKToRGB);
122     void allocateStorage(const SkImageInfo& dstInfo);
123     int readRows(const SkImageInfo& dstInfo, void* dst, size_t rowBytes, int count, const Options&);
124 
125     /*
126      * Scanline decoding.
127      */
128     SkSampler* getSampler(bool createIfNecessary) override;
129     Result onStartScanlineDecode(const SkImageInfo& dstInfo,
130             const Options& options) override;
131     int onGetScanlines(void* dst, int count, size_t rowBytes) override;
132     bool onSkipScanlines(int count) override;
133 
134     std::unique_ptr<JpegDecoderMgr>    fDecoderMgr;
135 
136     // We will save the state of the decompress struct after reading the header.
137     // This allows us to safely call onGetScaledDimensions() at any time.
138     const int                          fReadyState;
139 
140 
141     SkAutoTMalloc<uint8_t>             fStorage;
142     uint8_t*                           fSwizzleSrcRow;
143     uint32_t*                          fColorXformSrcRow;
144 
145     // libjpeg-turbo provides some subsetting.  In the case that libjpeg-turbo
146     // cannot take the exact the subset that we need, we will use the swizzler
147     // to further subset the output from libjpeg-turbo.
148     SkIRect                            fSwizzlerSubset;
149 
150     std::unique_ptr<SkSwizzler>        fSwizzler;
151 
152     friend class SkRawCodec;
153 
154     typedef SkCodec INHERITED;
155 };
156 
157 #endif
158