• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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 #include "SkCodec.h"
9 #include "SkCodecPriv.h"
10 #include "SkMath.h"
11 #include "SkSampledCodec.h"
12 #include "SkSampler.h"
13 #include "SkTemplates.h"
14 
SkSampledCodec(SkCodec * codec)15 SkSampledCodec::SkSampledCodec(SkCodec* codec)
16     : INHERITED(codec)
17 {}
18 
accountForNativeScaling(int * sampleSizePtr,int * nativeSampleSize) const19 SkISize SkSampledCodec::accountForNativeScaling(int* sampleSizePtr, int* nativeSampleSize) const {
20     SkISize preSampledSize = this->codec()->getInfo().dimensions();
21     int sampleSize = *sampleSizePtr;
22     SkASSERT(sampleSize > 1);
23 
24     if (nativeSampleSize) {
25         *nativeSampleSize = 1;
26     }
27 
28     // Only JPEG supports native downsampling.
29     if (this->codec()->getEncodedFormat() == SkEncodedImageFormat::kJPEG) {
30         // See if libjpeg supports this scale directly
31         switch (sampleSize) {
32             case 2:
33             case 4:
34             case 8:
35                 // This class does not need to do any sampling.
36                 *sampleSizePtr = 1;
37                 return this->codec()->getScaledDimensions(get_scale_from_sample_size(sampleSize));
38             default:
39                 break;
40         }
41 
42         // Check if sampleSize is a multiple of something libjpeg can support.
43         int remainder;
44         const int sampleSizes[] = { 8, 4, 2 };
45         for (int supportedSampleSize : sampleSizes) {
46             int actualSampleSize;
47             SkTDivMod(sampleSize, supportedSampleSize, &actualSampleSize, &remainder);
48             if (0 == remainder) {
49                 float scale = get_scale_from_sample_size(supportedSampleSize);
50 
51                 // this->codec() will scale to this size.
52                 preSampledSize = this->codec()->getScaledDimensions(scale);
53 
54                 // And then this class will sample it.
55                 *sampleSizePtr = actualSampleSize;
56                 if (nativeSampleSize) {
57                     *nativeSampleSize = supportedSampleSize;
58                 }
59                 break;
60             }
61         }
62     }
63 
64     return preSampledSize;
65 }
66 
onGetSampledDimensions(int sampleSize) const67 SkISize SkSampledCodec::onGetSampledDimensions(int sampleSize) const {
68     const SkISize size = this->accountForNativeScaling(&sampleSize);
69     return SkISize::Make(get_scaled_dimension(size.width(), sampleSize),
70                          get_scaled_dimension(size.height(), sampleSize));
71 }
72 
onGetAndroidPixels(const SkImageInfo & info,void * pixels,size_t rowBytes,const AndroidOptions & options)73 SkCodec::Result SkSampledCodec::onGetAndroidPixels(const SkImageInfo& info, void* pixels,
74         size_t rowBytes, const AndroidOptions& options) {
75     // Create an Options struct for the codec.
76     SkCodec::Options codecOptions;
77     codecOptions.fZeroInitialized = options.fZeroInitialized;
78     codecOptions.fPremulBehavior = SkTransferFunctionBehavior::kIgnore;
79 
80     SkIRect* subset = options.fSubset;
81     if (!subset || subset->size() == this->codec()->getInfo().dimensions()) {
82         if (this->codec()->dimensionsSupported(info.dimensions())) {
83             return this->codec()->getPixels(info, pixels, rowBytes, &codecOptions,
84                     options.fColorPtr, options.fColorCount);
85         }
86 
87         // If the native codec does not support the requested scale, scale by sampling.
88         return this->sampledDecode(info, pixels, rowBytes, options);
89     }
90 
91     // We are performing a subset decode.
92     int sampleSize = options.fSampleSize;
93     SkISize scaledSize = this->getSampledDimensions(sampleSize);
94     if (!this->codec()->dimensionsSupported(scaledSize)) {
95         // If the native codec does not support the requested scale, scale by sampling.
96         return this->sampledDecode(info, pixels, rowBytes, options);
97     }
98 
99     // Calculate the scaled subset bounds.
100     int scaledSubsetX = subset->x() / sampleSize;
101     int scaledSubsetY = subset->y() / sampleSize;
102     int scaledSubsetWidth = info.width();
103     int scaledSubsetHeight = info.height();
104 
105     const SkImageInfo scaledInfo = info.makeWH(scaledSize.width(), scaledSize.height());
106 
107     {
108         // Although startScanlineDecode expects the bottom and top to match the
109         // SkImageInfo, startIncrementalDecode uses them to determine which rows to
110         // decode.
111         SkIRect incrementalSubset = SkIRect::MakeXYWH(scaledSubsetX, scaledSubsetY,
112                                                       scaledSubsetWidth, scaledSubsetHeight);
113         codecOptions.fSubset = &incrementalSubset;
114         const SkCodec::Result startResult = this->codec()->startIncrementalDecode(
115                 scaledInfo, pixels, rowBytes, &codecOptions,
116                 options.fColorPtr, options.fColorCount);
117         if (SkCodec::kSuccess == startResult) {
118             int rowsDecoded;
119             const SkCodec::Result incResult = this->codec()->incrementalDecode(&rowsDecoded);
120             if (incResult == SkCodec::kSuccess) {
121                 return SkCodec::kSuccess;
122             }
123             SkASSERT(SkCodec::kIncompleteInput == incResult);
124 
125             // FIXME: Can zero initialized be read from SkCodec::fOptions?
126             this->codec()->fillIncompleteImage(scaledInfo, pixels, rowBytes,
127                     options.fZeroInitialized, scaledSubsetHeight, rowsDecoded);
128             return SkCodec::kIncompleteInput;
129         } else if (startResult != SkCodec::kUnimplemented) {
130             return startResult;
131         }
132         // Otherwise fall down to use the old scanline decoder.
133         // codecOptions.fSubset will be reset below, so it will not continue to
134         // point to the object that is no longer on the stack.
135     }
136 
137     // Start the scanline decode.
138     SkIRect scanlineSubset = SkIRect::MakeXYWH(scaledSubsetX, 0, scaledSubsetWidth,
139             scaledSize.height());
140     codecOptions.fSubset = &scanlineSubset;
141 
142     SkCodec::Result result = this->codec()->startScanlineDecode(scaledInfo,
143             &codecOptions, options.fColorPtr, options.fColorCount);
144     if (SkCodec::kSuccess != result) {
145         return result;
146     }
147 
148     // At this point, we are only concerned with subsetting.  Either no scale was
149     // requested, or the this->codec() is handling the scale.
150     // Note that subsetting is only supported for kTopDown, so this code will not be
151     // reached for other orders.
152     SkASSERT(this->codec()->getScanlineOrder() == SkCodec::kTopDown_SkScanlineOrder);
153     if (!this->codec()->skipScanlines(scaledSubsetY)) {
154         this->codec()->fillIncompleteImage(info, pixels, rowBytes, options.fZeroInitialized,
155                 scaledSubsetHeight, 0);
156         return SkCodec::kIncompleteInput;
157     }
158 
159     int decodedLines = this->codec()->getScanlines(pixels, scaledSubsetHeight, rowBytes);
160     if (decodedLines != scaledSubsetHeight) {
161         return SkCodec::kIncompleteInput;
162     }
163     return SkCodec::kSuccess;
164 }
165 
166 
sampledDecode(const SkImageInfo & info,void * pixels,size_t rowBytes,const AndroidOptions & options)167 SkCodec::Result SkSampledCodec::sampledDecode(const SkImageInfo& info, void* pixels,
168         size_t rowBytes, const AndroidOptions& options) {
169     // We should only call this function when sampling.
170     SkASSERT(options.fSampleSize > 1);
171 
172     // Create options struct for the codec.
173     SkCodec::Options sampledOptions;
174     sampledOptions.fZeroInitialized = options.fZeroInitialized;
175     sampledOptions.fPremulBehavior = SkTransferFunctionBehavior::kIgnore;
176 
177     // FIXME: This was already called by onGetAndroidPixels. Can we reduce that?
178     int sampleSize = options.fSampleSize;
179     int nativeSampleSize;
180     SkISize nativeSize = this->accountForNativeScaling(&sampleSize, &nativeSampleSize);
181 
182     // Check if there is a subset.
183     SkIRect subset;
184     int subsetY = 0;
185     int subsetWidth = nativeSize.width();
186     int subsetHeight = nativeSize.height();
187     if (options.fSubset) {
188         // We will need to know about subsetting in the y-dimension in order to use the
189         // scanline decoder.
190         // Update the subset to account for scaling done by this->codec().
191         const SkIRect* subsetPtr = options.fSubset;
192 
193         // Do the divide ourselves, instead of calling get_scaled_dimension. If
194         // X and Y are 0, they should remain 0, rather than being upgraded to 1
195         // due to being smaller than the sampleSize.
196         const int subsetX = subsetPtr->x() / nativeSampleSize;
197         subsetY = subsetPtr->y() / nativeSampleSize;
198 
199         subsetWidth = get_scaled_dimension(subsetPtr->width(), nativeSampleSize);
200         subsetHeight = get_scaled_dimension(subsetPtr->height(), nativeSampleSize);
201 
202         // The scanline decoder only needs to be aware of subsetting in the x-dimension.
203         subset.setXYWH(subsetX, 0, subsetWidth, nativeSize.height());
204         sampledOptions.fSubset = ⊂
205     }
206 
207     // Since we guarantee that output dimensions are always at least one (even if the sampleSize
208     // is greater than a given dimension), the input sampleSize is not always the sampleSize that
209     // we use in practice.
210     const int sampleX = subsetWidth / info.width();
211     const int sampleY = subsetHeight / info.height();
212 
213     const int samplingOffsetY = get_start_coord(sampleY);
214     const int startY = samplingOffsetY + subsetY;
215     const int dstHeight = info.height();
216 
217     const SkImageInfo nativeInfo = info.makeWH(nativeSize.width(), nativeSize.height());
218 
219     {
220         // Although startScanlineDecode expects the bottom and top to match the
221         // SkImageInfo, startIncrementalDecode uses them to determine which rows to
222         // decode.
223         SkCodec::Options incrementalOptions = sampledOptions;
224         SkIRect incrementalSubset;
225         if (sampledOptions.fSubset) {
226             incrementalSubset.fTop = subsetY;
227             incrementalSubset.fBottom = subsetY + subsetHeight;
228             incrementalSubset.fLeft = sampledOptions.fSubset->fLeft;
229             incrementalSubset.fRight = sampledOptions.fSubset->fRight;
230             incrementalOptions.fSubset = &incrementalSubset;
231         }
232         const SkCodec::Result startResult = this->codec()->startIncrementalDecode(nativeInfo,
233                 pixels, rowBytes, &incrementalOptions, options.fColorPtr, options.fColorCount);
234         if (SkCodec::kSuccess == startResult) {
235             SkSampler* sampler = this->codec()->getSampler(true);
236             if (!sampler) {
237                 return SkCodec::kUnimplemented;
238             }
239 
240             if (sampler->setSampleX(sampleX) != info.width()) {
241                 return SkCodec::kInvalidScale;
242             }
243             if (get_scaled_dimension(subsetHeight, sampleY) != info.height()) {
244                 return SkCodec::kInvalidScale;
245             }
246 
247             sampler->setSampleY(sampleY);
248 
249             int rowsDecoded;
250             const SkCodec::Result incResult = this->codec()->incrementalDecode(&rowsDecoded);
251             if (incResult == SkCodec::kSuccess) {
252                 return SkCodec::kSuccess;
253             }
254             SkASSERT(incResult == SkCodec::kIncompleteInput);
255 
256             SkASSERT(rowsDecoded <= info.height());
257             this->codec()->fillIncompleteImage(info, pixels, rowBytes, options.fZeroInitialized,
258                                                info.height(), rowsDecoded);
259             return SkCodec::kIncompleteInput;
260         } else if (startResult != SkCodec::kUnimplemented) {
261             return startResult;
262         } // kUnimplemented means use the old method.
263     }
264 
265     // Start the scanline decode.
266     SkCodec::Result result = this->codec()->startScanlineDecode(nativeInfo,
267             &sampledOptions, options.fColorPtr, options.fColorCount);
268     if (SkCodec::kSuccess != result) {
269         return result;
270     }
271 
272     SkSampler* sampler = this->codec()->getSampler(true);
273     if (!sampler) {
274         return SkCodec::kUnimplemented;
275     }
276 
277     if (sampler->setSampleX(sampleX) != info.width()) {
278         return SkCodec::kInvalidScale;
279     }
280     if (get_scaled_dimension(subsetHeight, sampleY) != info.height()) {
281         return SkCodec::kInvalidScale;
282     }
283 
284     switch(this->codec()->getScanlineOrder()) {
285         case SkCodec::kTopDown_SkScanlineOrder: {
286             if (!this->codec()->skipScanlines(startY)) {
287                 this->codec()->fillIncompleteImage(info, pixels, rowBytes, options.fZeroInitialized,
288                         dstHeight, 0);
289                 return SkCodec::kIncompleteInput;
290             }
291             void* pixelPtr = pixels;
292             for (int y = 0; y < dstHeight; y++) {
293                 if (1 != this->codec()->getScanlines(pixelPtr, 1, rowBytes)) {
294                     this->codec()->fillIncompleteImage(info, pixels, rowBytes,
295                             options.fZeroInitialized, dstHeight, y + 1);
296                     return SkCodec::kIncompleteInput;
297                 }
298                 if (y < dstHeight - 1) {
299                     if (!this->codec()->skipScanlines(sampleY - 1)) {
300                         this->codec()->fillIncompleteImage(info, pixels, rowBytes,
301                                 options.fZeroInitialized, dstHeight, y + 1);
302                         return SkCodec::kIncompleteInput;
303                     }
304                 }
305                 pixelPtr = SkTAddOffset<void>(pixelPtr, rowBytes);
306             }
307             return SkCodec::kSuccess;
308         }
309         case SkCodec::kBottomUp_SkScanlineOrder: {
310             // Note that these modes do not support subsetting.
311             SkASSERT(0 == subsetY && nativeSize.height() == subsetHeight);
312             int y;
313             for (y = 0; y < nativeSize.height(); y++) {
314                 int srcY = this->codec()->nextScanline();
315                 if (is_coord_necessary(srcY, sampleY, dstHeight)) {
316                     void* pixelPtr = SkTAddOffset<void>(pixels,
317                             rowBytes * get_dst_coord(srcY, sampleY));
318                     if (1 != this->codec()->getScanlines(pixelPtr, 1, rowBytes)) {
319                         break;
320                     }
321                 } else {
322                     if (!this->codec()->skipScanlines(1)) {
323                         break;
324                     }
325                 }
326             }
327 
328             if (nativeSize.height() == y) {
329                 return SkCodec::kSuccess;
330             }
331 
332             // We handle filling uninitialized memory here instead of using this->codec().
333             // this->codec() does not know that we are sampling.
334             const uint64_t fillValue = this->codec()->getFillValue(info);
335             const SkImageInfo fillInfo = info.makeWH(info.width(), 1);
336             for (; y < nativeSize.height(); y++) {
337                 int srcY = this->codec()->outputScanline(y);
338                 if (!is_coord_necessary(srcY, sampleY, dstHeight)) {
339                     continue;
340                 }
341 
342                 void* rowPtr = SkTAddOffset<void>(pixels, rowBytes * get_dst_coord(srcY, sampleY));
343                 SkSampler::Fill(fillInfo, rowPtr, rowBytes, fillValue, options.fZeroInitialized);
344             }
345             return SkCodec::kIncompleteInput;
346         }
347         default:
348             SkASSERT(false);
349             return SkCodec::kUnimplemented;
350     }
351 }
352