Home | History | Annotate | Download | only in codec
      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 "SkMSAN.h"
     10 #include "SkJpegCodec.h"
     11 #include "SkJpegDecoderMgr.h"
     12 #include "SkJpegUtility_codec.h"
     13 #include "SkCodecPriv.h"
     14 #include "SkColorPriv.h"
     15 #include "SkStream.h"
     16 #include "SkTemplates.h"
     17 #include "SkTypes.h"
     18 
     19 // stdio is needed for libjpeg-turbo
     20 #include <stdio.h>
     21 
     22 extern "C" {
     23     #include "jerror.h"
     24     #include "jpeglib.h"
     25 }
     26 
     27 bool SkJpegCodec::IsJpeg(const void* buffer, size_t bytesRead) {
     28     static const uint8_t jpegSig[] = { 0xFF, 0xD8, 0xFF };
     29     return bytesRead >= 3 && !memcmp(buffer, jpegSig, sizeof(jpegSig));
     30 }
     31 
     32 bool SkJpegCodec::ReadHeader(SkStream* stream, SkCodec** codecOut,
     33         JpegDecoderMgr** decoderMgrOut) {
     34 
     35     // Create a JpegDecoderMgr to own all of the decompress information
     36     SkAutoTDelete<JpegDecoderMgr> decoderMgr(new JpegDecoderMgr(stream));
     37 
     38     // libjpeg errors will be caught and reported here
     39     if (setjmp(decoderMgr->getJmpBuf())) {
     40         return decoderMgr->returnFalse("setjmp");
     41     }
     42 
     43     // Initialize the decompress info and the source manager
     44     decoderMgr->init();
     45 
     46     // Read the jpeg header
     47     if (JPEG_HEADER_OK != jpeg_read_header(decoderMgr->dinfo(), true)) {
     48         return decoderMgr->returnFalse("read_header");
     49     }
     50 
     51     if (nullptr != codecOut) {
     52         // Recommend the color type to decode to
     53         const SkColorType colorType = decoderMgr->getColorType();
     54 
     55         // Create image info object and the codec
     56         const SkImageInfo& imageInfo = SkImageInfo::Make(decoderMgr->dinfo()->image_width,
     57                 decoderMgr->dinfo()->image_height, colorType, kOpaque_SkAlphaType);
     58         *codecOut = new SkJpegCodec(imageInfo, stream, decoderMgr.detach());
     59     } else {
     60         SkASSERT(nullptr != decoderMgrOut);
     61         *decoderMgrOut = decoderMgr.detach();
     62     }
     63     return true;
     64 }
     65 
     66 SkCodec* SkJpegCodec::NewFromStream(SkStream* stream) {
     67     SkAutoTDelete<SkStream> streamDeleter(stream);
     68     SkCodec* codec = nullptr;
     69     if (ReadHeader(stream,  &codec, nullptr)) {
     70         // Codec has taken ownership of the stream, we do not need to delete it
     71         SkASSERT(codec);
     72         streamDeleter.detach();
     73         return codec;
     74     }
     75     return nullptr;
     76 }
     77 
     78 SkJpegCodec::SkJpegCodec(const SkImageInfo& srcInfo, SkStream* stream,
     79         JpegDecoderMgr* decoderMgr)
     80     : INHERITED(srcInfo, stream)
     81     , fDecoderMgr(decoderMgr)
     82     , fReadyState(decoderMgr->dinfo()->global_state)
     83     , fSwizzlerSubset(SkIRect::MakeEmpty())
     84 {}
     85 
     86 /*
     87  * Return the row bytes of a particular image type and width
     88  */
     89 static size_t get_row_bytes(const j_decompress_ptr dinfo) {
     90 #ifdef TURBO_HAS_565
     91     const size_t colorBytes = (dinfo->out_color_space == JCS_RGB565) ? 2 :
     92             dinfo->out_color_components;
     93 #else
     94     const size_t colorBytes = dinfo->out_color_components;
     95 #endif
     96     return dinfo->output_width * colorBytes;
     97 
     98 }
     99 
    100 /*
    101  *  Calculate output dimensions based on the provided factors.
    102  *
    103  *  Not to be used on the actual jpeg_decompress_struct used for decoding, since it will
    104  *  incorrectly modify num_components.
    105  */
    106 void calc_output_dimensions(jpeg_decompress_struct* dinfo, unsigned int num, unsigned int denom) {
    107     dinfo->num_components = 0;
    108     dinfo->scale_num = num;
    109     dinfo->scale_denom = denom;
    110     jpeg_calc_output_dimensions(dinfo);
    111 }
    112 
    113 /*
    114  * Return a valid set of output dimensions for this decoder, given an input scale
    115  */
    116 SkISize SkJpegCodec::onGetScaledDimensions(float desiredScale) const {
    117     // 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
    118     // support these as well
    119     unsigned int num;
    120     unsigned int denom = 8;
    121     if (desiredScale >= 0.9375) {
    122         num = 8;
    123     } else if (desiredScale >= 0.8125) {
    124         num = 7;
    125     } else if (desiredScale >= 0.6875f) {
    126         num = 6;
    127     } else if (desiredScale >= 0.5625f) {
    128         num = 5;
    129     } else if (desiredScale >= 0.4375f) {
    130         num = 4;
    131     } else if (desiredScale >= 0.3125f) {
    132         num = 3;
    133     } else if (desiredScale >= 0.1875f) {
    134         num = 2;
    135     } else {
    136         num = 1;
    137     }
    138 
    139     // Set up a fake decompress struct in order to use libjpeg to calculate output dimensions
    140     jpeg_decompress_struct dinfo;
    141     sk_bzero(&dinfo, sizeof(dinfo));
    142     dinfo.image_width = this->getInfo().width();
    143     dinfo.image_height = this->getInfo().height();
    144     dinfo.global_state = fReadyState;
    145     calc_output_dimensions(&dinfo, num, denom);
    146 
    147     // Return the calculated output dimensions for the given scale
    148     return SkISize::Make(dinfo.output_width, dinfo.output_height);
    149 }
    150 
    151 bool SkJpegCodec::onRewind() {
    152     JpegDecoderMgr* decoderMgr = nullptr;
    153     if (!ReadHeader(this->stream(), nullptr, &decoderMgr)) {
    154         return fDecoderMgr->returnFalse("could not rewind");
    155     }
    156     SkASSERT(nullptr != decoderMgr);
    157     fDecoderMgr.reset(decoderMgr);
    158     return true;
    159 }
    160 
    161 /*
    162  * Checks if the conversion between the input image and the requested output
    163  * image has been implemented
    164  * Sets the output color space
    165  */
    166 bool SkJpegCodec::setOutputColorSpace(const SkImageInfo& dst) {
    167     const SkImageInfo& src = this->getInfo();
    168 
    169     // Ensure that the profile type is unchanged
    170     if (dst.profileType() != src.profileType()) {
    171         return false;
    172     }
    173 
    174     if (kUnknown_SkAlphaType == dst.alphaType()) {
    175         return false;
    176     }
    177 
    178     if (kOpaque_SkAlphaType != dst.alphaType()) {
    179         SkCodecPrintf("Warning: an opaque image should be decoded as opaque "
    180                       "- it is being decoded as non-opaque, which will draw slower\n");
    181     }
    182 
    183     // Check if we will decode to CMYK because a conversion to RGBA is not supported
    184     J_COLOR_SPACE colorSpace = fDecoderMgr->dinfo()->jpeg_color_space;
    185     bool isCMYK = JCS_CMYK == colorSpace || JCS_YCCK == colorSpace;
    186 
    187     // Check for valid color types and set the output color space
    188     switch (dst.colorType()) {
    189         case kN32_SkColorType:
    190             if (isCMYK) {
    191                 fDecoderMgr->dinfo()->out_color_space = JCS_CMYK;
    192             } else {
    193 #ifdef LIBJPEG_TURBO_VERSION
    194             // Check the byte ordering of the RGBA color space for the
    195             // current platform
    196     #ifdef SK_PMCOLOR_IS_RGBA
    197             fDecoderMgr->dinfo()->out_color_space = JCS_EXT_RGBA;
    198     #else
    199             fDecoderMgr->dinfo()->out_color_space = JCS_EXT_BGRA;
    200     #endif
    201 #else
    202             fDecoderMgr->dinfo()->out_color_space = JCS_RGB;
    203 #endif
    204             }
    205             return true;
    206         case kRGB_565_SkColorType:
    207             if (isCMYK) {
    208                 fDecoderMgr->dinfo()->out_color_space = JCS_CMYK;
    209             } else {
    210 #ifdef TURBO_HAS_565
    211                 fDecoderMgr->dinfo()->dither_mode = JDITHER_NONE;
    212                 fDecoderMgr->dinfo()->out_color_space = JCS_RGB565;
    213 #else
    214                 fDecoderMgr->dinfo()->out_color_space = JCS_RGB;
    215 #endif
    216             }
    217             return true;
    218         case kGray_8_SkColorType:
    219             if (isCMYK) {
    220                 return false;
    221             } else {
    222                 // We will enable decodes to gray even if the image is color because this is
    223                 // much faster than decoding to color and then converting
    224                 fDecoderMgr->dinfo()->out_color_space = JCS_GRAYSCALE;
    225             }
    226             return true;
    227         default:
    228             return false;
    229     }
    230 }
    231 
    232 /*
    233  * Checks if we can natively scale to the requested dimensions and natively scales the
    234  * dimensions if possible
    235  */
    236 bool SkJpegCodec::onDimensionsSupported(const SkISize& size) {
    237     if (setjmp(fDecoderMgr->getJmpBuf())) {
    238         return fDecoderMgr->returnFalse("onDimensionsSupported/setjmp");
    239     }
    240 
    241     const unsigned int dstWidth = size.width();
    242     const unsigned int dstHeight = size.height();
    243 
    244     // Set up a fake decompress struct in order to use libjpeg to calculate output dimensions
    245     // FIXME: Why is this necessary?
    246     jpeg_decompress_struct dinfo;
    247     sk_bzero(&dinfo, sizeof(dinfo));
    248     dinfo.image_width = this->getInfo().width();
    249     dinfo.image_height = this->getInfo().height();
    250     dinfo.global_state = fReadyState;
    251 
    252     // libjpeg-turbo can scale to 1/8, 1/4, 3/8, 1/2, 5/8, 3/4, 7/8, and 1/1
    253     unsigned int num = 8;
    254     const unsigned int denom = 8;
    255     calc_output_dimensions(&dinfo, num, denom);
    256     while (dinfo.output_width != dstWidth || dinfo.output_height != dstHeight) {
    257 
    258         // Return a failure if we have tried all of the possible scales
    259         if (1 == num || dstWidth > dinfo.output_width || dstHeight > dinfo.output_height) {
    260             return false;
    261         }
    262 
    263         // Try the next scale
    264         num -= 1;
    265         calc_output_dimensions(&dinfo, num, denom);
    266     }
    267 
    268     fDecoderMgr->dinfo()->scale_num = num;
    269     fDecoderMgr->dinfo()->scale_denom = denom;
    270     return true;
    271 }
    272 
    273 /*
    274  * Performs the jpeg decode
    275  */
    276 SkCodec::Result SkJpegCodec::onGetPixels(const SkImageInfo& dstInfo,
    277                                          void* dst, size_t dstRowBytes,
    278                                          const Options& options, SkPMColor*, int*,
    279                                          int* rowsDecoded) {
    280     if (options.fSubset) {
    281         // Subsets are not supported.
    282         return kUnimplemented;
    283     }
    284 
    285     // Get a pointer to the decompress info since we will use it quite frequently
    286     jpeg_decompress_struct* dinfo = fDecoderMgr->dinfo();
    287 
    288     // Set the jump location for libjpeg errors
    289     if (setjmp(fDecoderMgr->getJmpBuf())) {
    290         return fDecoderMgr->returnFailure("setjmp", kInvalidInput);
    291     }
    292 
    293     // Check if we can decode to the requested destination and set the output color space
    294     if (!this->setOutputColorSpace(dstInfo)) {
    295         return fDecoderMgr->returnFailure("conversion_possible", kInvalidConversion);
    296     }
    297 
    298     // Now, given valid output dimensions, we can start the decompress
    299     if (!jpeg_start_decompress(dinfo)) {
    300         return fDecoderMgr->returnFailure("startDecompress", kInvalidInput);
    301     }
    302 
    303     // The recommended output buffer height should always be 1 in high quality modes.
    304     // If it's not, we want to know because it means our strategy is not optimal.
    305     SkASSERT(1 == dinfo->rec_outbuf_height);
    306 
    307     J_COLOR_SPACE colorSpace = dinfo->out_color_space;
    308     if (JCS_CMYK == colorSpace || JCS_RGB == colorSpace) {
    309         this->initializeSwizzler(dstInfo, options);
    310     }
    311 
    312     // Perform the decode a single row at a time
    313     uint32_t dstHeight = dstInfo.height();
    314 
    315     JSAMPLE* dstRow;
    316     if (fSwizzler) {
    317         // write data to storage row, then sample using swizzler
    318         dstRow = fSrcRow;
    319     } else {
    320         // write data directly to dst
    321         dstRow = (JSAMPLE*) dst;
    322     }
    323 
    324     for (uint32_t y = 0; y < dstHeight; y++) {
    325         // Read rows of the image
    326         uint32_t lines = jpeg_read_scanlines(dinfo, &dstRow, 1);
    327         sk_msan_mark_initialized(dstRow, dstRow + dstRowBytes, "skbug.com/4550");
    328 
    329         // If we cannot read enough rows, assume the input is incomplete
    330         if (lines != 1) {
    331             *rowsDecoded = y;
    332 
    333             return fDecoderMgr->returnFailure("Incomplete image data", kIncompleteInput);
    334         }
    335 
    336         if (fSwizzler) {
    337             // use swizzler to sample row
    338             fSwizzler->swizzle(dst, dstRow);
    339             dst = SkTAddOffset<JSAMPLE>(dst, dstRowBytes);
    340         } else {
    341             dstRow = SkTAddOffset<JSAMPLE>(dstRow, dstRowBytes);
    342         }
    343     }
    344 
    345     return kSuccess;
    346 }
    347 
    348 void SkJpegCodec::initializeSwizzler(const SkImageInfo& dstInfo, const Options& options) {
    349     SkSwizzler::SrcConfig srcConfig = SkSwizzler::kUnknown;
    350     if (JCS_CMYK == fDecoderMgr->dinfo()->out_color_space) {
    351         srcConfig = SkSwizzler::kCMYK;
    352     } else {
    353         // If the out_color_space is not CMYK, the only reason we would need a swizzler is
    354         // for sampling and/or subsetting.
    355         switch (dstInfo.colorType()) {
    356             case kGray_8_SkColorType:
    357                 srcConfig = SkSwizzler::kNoOp8;
    358                 break;
    359             case kN32_SkColorType:
    360                 srcConfig = SkSwizzler::kNoOp32;
    361                 break;
    362             case kRGB_565_SkColorType:
    363                 srcConfig = SkSwizzler::kNoOp16;
    364                 break;
    365             default:
    366                 // This function should only be called if the colorType is supported by jpeg
    367                 SkASSERT(false);
    368         }
    369     }
    370 
    371     if (JCS_RGB == fDecoderMgr->dinfo()->out_color_space) {
    372         srcConfig = SkSwizzler::kRGB;
    373     }
    374 
    375     Options swizzlerOptions = options;
    376     if (options.fSubset) {
    377         // Use fSwizzlerSubset if this is a subset decode.  This is necessary in the case
    378         // where libjpeg-turbo provides a subset and then we need to subset it further.
    379         // Also, verify that fSwizzlerSubset is initialized and valid.
    380         SkASSERT(!fSwizzlerSubset.isEmpty() && fSwizzlerSubset.x() <= options.fSubset->x() &&
    381                 fSwizzlerSubset.width() == options.fSubset->width());
    382         swizzlerOptions.fSubset = &fSwizzlerSubset;
    383     }
    384     fSwizzler.reset(SkSwizzler::CreateSwizzler(srcConfig, nullptr, dstInfo, swizzlerOptions));
    385     SkASSERT(fSwizzler);
    386     fStorage.reset(get_row_bytes(fDecoderMgr->dinfo()));
    387     fSrcRow = fStorage.get();
    388 }
    389 
    390 SkSampler* SkJpegCodec::getSampler(bool createIfNecessary) {
    391     if (!createIfNecessary || fSwizzler) {
    392         SkASSERT(!fSwizzler || (fSrcRow && fStorage.get() == fSrcRow));
    393         return fSwizzler;
    394     }
    395 
    396     this->initializeSwizzler(this->dstInfo(), this->options());
    397     return fSwizzler;
    398 }
    399 
    400 SkCodec::Result SkJpegCodec::onStartScanlineDecode(const SkImageInfo& dstInfo,
    401         const Options& options, SkPMColor ctable[], int* ctableCount) {
    402     // Set the jump location for libjpeg errors
    403     if (setjmp(fDecoderMgr->getJmpBuf())) {
    404         SkCodecPrintf("setjmp: Error from libjpeg\n");
    405         return kInvalidInput;
    406     }
    407 
    408     // Check if we can decode to the requested destination and set the output color space
    409     if (!this->setOutputColorSpace(dstInfo)) {
    410         return kInvalidConversion;
    411     }
    412 
    413     // Remove objects used for sampling.
    414     fSwizzler.reset(nullptr);
    415     fSrcRow = nullptr;
    416     fStorage.free();
    417 
    418     // Now, given valid output dimensions, we can start the decompress
    419     if (!jpeg_start_decompress(fDecoderMgr->dinfo())) {
    420         SkCodecPrintf("start decompress failed\n");
    421         return kInvalidInput;
    422     }
    423 
    424     if (options.fSubset) {
    425         fSwizzlerSubset = *options.fSubset;
    426     }
    427 
    428 #ifdef TURBO_HAS_CROP
    429     if (options.fSubset) {
    430         uint32_t startX = options.fSubset->x();
    431         uint32_t width = options.fSubset->width();
    432 
    433         // libjpeg-turbo may need to align startX to a multiple of the IDCT
    434         // block size.  If this is the case, it will decrease the value of
    435         // startX to the appropriate alignment and also increase the value
    436         // of width so that the right edge of the requested subset remains
    437         // the same.
    438         jpeg_crop_scanline(fDecoderMgr->dinfo(), &startX, &width);
    439 
    440         SkASSERT(startX <= (uint32_t) options.fSubset->x());
    441         SkASSERT(width >= (uint32_t) options.fSubset->width());
    442         SkASSERT(startX + width >= (uint32_t) options.fSubset->right());
    443 
    444         // Instruct the swizzler (if it is necessary) to further subset the
    445         // output provided by libjpeg-turbo.
    446         //
    447         // We set this here (rather than in the if statement below), so that
    448         // if (1) we don't need a swizzler for the subset, and (2) we need a
    449         // swizzler for CMYK, the swizzler will still use the proper subset
    450         // dimensions.
    451         //
    452         // Note that the swizzler will ignore the y and height parameters of
    453         // the subset.  Since the scanline decoder (and the swizzler) handle
    454         // one row at a time, only the subsetting in the x-dimension matters.
    455         fSwizzlerSubset.setXYWH(options.fSubset->x() - startX, 0,
    456                 options.fSubset->width(), options.fSubset->height());
    457 
    458         // We will need a swizzler if libjpeg-turbo cannot provide the exact
    459         // subset that we request.
    460         if (startX != (uint32_t) options.fSubset->x() ||
    461                 width != (uint32_t) options.fSubset->width()) {
    462             this->initializeSwizzler(dstInfo, options);
    463         }
    464     }
    465 
    466     // Make sure we have a swizzler if we are converting from CMYK.
    467     if (!fSwizzler && JCS_CMYK == fDecoderMgr->dinfo()->out_color_space) {
    468         this->initializeSwizzler(dstInfo, options);
    469     }
    470 #else
    471     // We will need a swizzler if we are performing a subset decode or
    472     // converting from CMYK.
    473     J_COLOR_SPACE colorSpace = fDecoderMgr->dinfo()->out_color_space;
    474     if (options.fSubset || JCS_CMYK == colorSpace || JCS_RGB == colorSpace) {
    475         this->initializeSwizzler(dstInfo, options);
    476     }
    477 #endif
    478 
    479     return kSuccess;
    480 }
    481 
    482 int SkJpegCodec::onGetScanlines(void* dst, int count, size_t dstRowBytes) {
    483     // Set the jump location for libjpeg errors
    484     if (setjmp(fDecoderMgr->getJmpBuf())) {
    485         return fDecoderMgr->returnFailure("setjmp", kInvalidInput);
    486     }
    487     // Read rows one at a time
    488     JSAMPLE* dstRow;
    489     size_t srcRowBytes = get_row_bytes(fDecoderMgr->dinfo());
    490     if (fSwizzler) {
    491         // write data to storage row, then sample using swizzler
    492         dstRow = fSrcRow;
    493     } else {
    494         // write data directly to dst
    495         SkASSERT(count == 1 || dstRowBytes >= srcRowBytes);
    496         dstRow = (JSAMPLE*) dst;
    497     }
    498 
    499     for (int y = 0; y < count; y++) {
    500         // Read row of the image
    501         uint32_t rowsDecoded = jpeg_read_scanlines(fDecoderMgr->dinfo(), &dstRow, 1);
    502         sk_msan_mark_initialized(dstRow, dstRow + srcRowBytes, "skbug.com/4550");
    503         if (rowsDecoded != 1) {
    504             fDecoderMgr->dinfo()->output_scanline = this->dstInfo().height();
    505             return y;
    506         }
    507 
    508         if (fSwizzler) {
    509             // use swizzler to sample row
    510             fSwizzler->swizzle(dst, dstRow);
    511             dst = SkTAddOffset<JSAMPLE>(dst, dstRowBytes);
    512         } else {
    513             dstRow = SkTAddOffset<JSAMPLE>(dstRow, dstRowBytes);
    514         }
    515     }
    516     return count;
    517 }
    518 
    519 bool SkJpegCodec::onSkipScanlines(int count) {
    520     // Set the jump location for libjpeg errors
    521     if (setjmp(fDecoderMgr->getJmpBuf())) {
    522         return fDecoderMgr->returnFalse("setjmp");
    523     }
    524 
    525 #ifdef TURBO_HAS_SKIP
    526     return (uint32_t) count == jpeg_skip_scanlines(fDecoderMgr->dinfo(), count);
    527 #else
    528     if (!fSrcRow) {
    529         fStorage.reset(get_row_bytes(fDecoderMgr->dinfo()));
    530         fSrcRow = fStorage.get();
    531     }
    532 
    533     for (int y = 0; y < count; y++) {
    534         if (1 != jpeg_read_scanlines(fDecoderMgr->dinfo(), &fSrcRow, 1)) {
    535             return false;
    536         }
    537     }
    538     return true;
    539 #endif
    540 }
    541 
    542 static bool is_yuv_supported(jpeg_decompress_struct* dinfo) {
    543     // Scaling is not supported in raw data mode.
    544     SkASSERT(dinfo->scale_num == dinfo->scale_denom);
    545 
    546     // I can't imagine that this would ever change, but we do depend on it.
    547     static_assert(8 == DCTSIZE, "DCTSIZE (defined in jpeg library) should always be 8.");
    548 
    549     if (JCS_YCbCr != dinfo->jpeg_color_space) {
    550         return false;
    551     }
    552 
    553     SkASSERT(3 == dinfo->num_components);
    554     SkASSERT(dinfo->comp_info);
    555 
    556     // It is possible to perform a YUV decode for any combination of
    557     // horizontal and vertical sampling that is supported by
    558     // libjpeg/libjpeg-turbo.  However, we will start by supporting only the
    559     // common cases (where U and V have samp_factors of one).
    560     //
    561     // The definition of samp_factor is kind of the opposite of what SkCodec
    562     // thinks of as a sampling factor.  samp_factor is essentially a
    563     // multiplier, and the larger the samp_factor is, the more samples that
    564     // there will be.  Ex:
    565     //     U_plane_width = image_width * (U_h_samp_factor / max_h_samp_factor)
    566     //
    567     // Supporting cases where the samp_factors for U or V were larger than
    568     // that of Y would be an extremely difficult change, given that clients
    569     // allocate memory as if the size of the Y plane is always the size of the
    570     // image.  However, this case is very, very rare.
    571     if (!(1 == dinfo->comp_info[1].h_samp_factor) &&
    572          (1 == dinfo->comp_info[1].v_samp_factor) &&
    573          (1 == dinfo->comp_info[2].h_samp_factor) &&
    574          (1 == dinfo->comp_info[2].v_samp_factor)) {
    575         return false;
    576     }
    577 
    578     // Support all common cases of Y samp_factors.
    579     // TODO (msarett): As mentioned above, it would be possible to support
    580     //                 more combinations of samp_factors.  The issues are:
    581     //                 (1) Are there actually any images that are not covered
    582     //                     by these cases?
    583     //                 (2) How much complexity would be added to the
    584     //                     implementation in order to support these rare
    585     //                     cases?
    586     int hSampY = dinfo->comp_info[0].h_samp_factor;
    587     int vSampY = dinfo->comp_info[0].v_samp_factor;
    588     return (1 == hSampY && 1 == vSampY) ||
    589            (2 == hSampY && 1 == vSampY) ||
    590            (2 == hSampY && 2 == vSampY) ||
    591            (1 == hSampY && 2 == vSampY) ||
    592            (4 == hSampY && 1 == vSampY) ||
    593            (4 == hSampY && 2 == vSampY);
    594 }
    595 
    596 bool SkJpegCodec::onQueryYUV8(YUVSizeInfo* sizeInfo, SkYUVColorSpace* colorSpace) const {
    597     jpeg_decompress_struct* dinfo = fDecoderMgr->dinfo();
    598     if (!is_yuv_supported(dinfo)) {
    599         return false;
    600     }
    601 
    602     sizeInfo->fYSize.set(dinfo->comp_info[0].downsampled_width,
    603                          dinfo->comp_info[0].downsampled_height);
    604     sizeInfo->fUSize.set(dinfo->comp_info[1].downsampled_width,
    605                          dinfo->comp_info[1].downsampled_height);
    606     sizeInfo->fVSize.set(dinfo->comp_info[2].downsampled_width,
    607                          dinfo->comp_info[2].downsampled_height);
    608     sizeInfo->fYWidthBytes = dinfo->comp_info[0].width_in_blocks * DCTSIZE;
    609     sizeInfo->fUWidthBytes = dinfo->comp_info[1].width_in_blocks * DCTSIZE;
    610     sizeInfo->fVWidthBytes = dinfo->comp_info[2].width_in_blocks * DCTSIZE;
    611 
    612     if (colorSpace) {
    613         *colorSpace = kJPEG_SkYUVColorSpace;
    614     }
    615 
    616     return true;
    617 }
    618 
    619 SkCodec::Result SkJpegCodec::onGetYUV8Planes(const YUVSizeInfo& sizeInfo, void* pixels[3]) {
    620     YUVSizeInfo defaultInfo;
    621 
    622     // This will check is_yuv_supported(), so we don't need to here.
    623     bool supportsYUV = this->onQueryYUV8(&defaultInfo, nullptr);
    624     if (!supportsYUV || sizeInfo.fYSize != defaultInfo.fYSize ||
    625             sizeInfo.fUSize != defaultInfo.fUSize ||
    626             sizeInfo.fVSize != defaultInfo.fVSize ||
    627             sizeInfo.fYWidthBytes < defaultInfo.fYWidthBytes ||
    628             sizeInfo.fUWidthBytes < defaultInfo.fUWidthBytes ||
    629             sizeInfo.fVWidthBytes < defaultInfo.fVWidthBytes) {
    630         return fDecoderMgr->returnFailure("onGetYUV8Planes", kInvalidInput);
    631     }
    632 
    633     // Set the jump location for libjpeg errors
    634     if (setjmp(fDecoderMgr->getJmpBuf())) {
    635         return fDecoderMgr->returnFailure("setjmp", kInvalidInput);
    636     }
    637 
    638     // Get a pointer to the decompress info since we will use it quite frequently
    639     jpeg_decompress_struct* dinfo = fDecoderMgr->dinfo();
    640 
    641     dinfo->raw_data_out = TRUE;
    642     if (!jpeg_start_decompress(dinfo)) {
    643         return fDecoderMgr->returnFailure("startDecompress", kInvalidInput);
    644     }
    645 
    646     // A previous implementation claims that the return value of is_yuv_supported()
    647     // may change after calling jpeg_start_decompress().  It looks to me like this
    648     // was caused by a bug in the old code, but we'll be safe and check here.
    649     SkASSERT(is_yuv_supported(dinfo));
    650 
    651     // Currently, we require that the Y plane dimensions match the image dimensions
    652     // and that the U and V planes are the same dimensions.
    653     SkASSERT(sizeInfo.fUSize == sizeInfo.fVSize);
    654     SkASSERT((uint32_t) sizeInfo.fYSize.width() == dinfo->output_width &&
    655             (uint32_t) sizeInfo.fYSize.height() == dinfo->output_height);
    656 
    657     // Build a JSAMPIMAGE to handle output from libjpeg-turbo.  A JSAMPIMAGE has
    658     // a 2-D array of pixels for each of the components (Y, U, V) in the image.
    659     // Cheat Sheet:
    660     //     JSAMPIMAGE == JSAMPLEARRAY* == JSAMPROW** == JSAMPLE***
    661     JSAMPARRAY yuv[3];
    662 
    663     // Set aside enough space for pointers to rows of Y, U, and V.
    664     JSAMPROW rowptrs[2 * DCTSIZE + DCTSIZE + DCTSIZE];
    665     yuv[0] = &rowptrs[0];           // Y rows (DCTSIZE or 2 * DCTSIZE)
    666     yuv[1] = &rowptrs[2 * DCTSIZE]; // U rows (DCTSIZE)
    667     yuv[2] = &rowptrs[3 * DCTSIZE]; // V rows (DCTSIZE)
    668 
    669     // Initialize rowptrs.
    670     int numYRowsPerBlock = DCTSIZE * dinfo->comp_info[0].v_samp_factor;
    671     for (int i = 0; i < numYRowsPerBlock; i++) {
    672         rowptrs[i] = SkTAddOffset<JSAMPLE>(pixels[0], i * sizeInfo.fYWidthBytes);
    673     }
    674     for (int i = 0; i < DCTSIZE; i++) {
    675         rowptrs[i + 2 * DCTSIZE] = SkTAddOffset<JSAMPLE>(pixels[1], i * sizeInfo.fUWidthBytes);
    676         rowptrs[i + 3 * DCTSIZE] = SkTAddOffset<JSAMPLE>(pixels[2], i * sizeInfo.fVWidthBytes);
    677     }
    678 
    679     // After each loop iteration, we will increment pointers to Y, U, and V.
    680     size_t blockIncrementY = numYRowsPerBlock * sizeInfo.fYWidthBytes;
    681     size_t blockIncrementU = DCTSIZE * sizeInfo.fUWidthBytes;
    682     size_t blockIncrementV = DCTSIZE * sizeInfo.fVWidthBytes;
    683 
    684     uint32_t numRowsPerBlock = numYRowsPerBlock;
    685 
    686     // We intentionally round down here, as this first loop will only handle
    687     // full block rows.  As a special case at the end, we will handle any
    688     // remaining rows that do not make up a full block.
    689     const int numIters = dinfo->output_height / numRowsPerBlock;
    690     for (int i = 0; i < numIters; i++) {
    691         JDIMENSION linesRead = jpeg_read_raw_data(dinfo, yuv, numRowsPerBlock);
    692         if (linesRead < numRowsPerBlock) {
    693             // FIXME: Handle incomplete YUV decodes without signalling an error.
    694             return kInvalidInput;
    695         }
    696 
    697         // Update rowptrs.
    698         for (int i = 0; i < numYRowsPerBlock; i++) {
    699             rowptrs[i] += blockIncrementY;
    700         }
    701         for (int i = 0; i < DCTSIZE; i++) {
    702             rowptrs[i + 2 * DCTSIZE] += blockIncrementU;
    703             rowptrs[i + 3 * DCTSIZE] += blockIncrementV;
    704         }
    705     }
    706 
    707     uint32_t remainingRows = dinfo->output_height - dinfo->output_scanline;
    708     SkASSERT(remainingRows == dinfo->output_height % numRowsPerBlock);
    709     SkASSERT(dinfo->output_scanline == numIters * numRowsPerBlock);
    710     if (remainingRows > 0) {
    711         // libjpeg-turbo needs memory to be padded by the block sizes.  We will fulfill
    712         // this requirement using a dummy row buffer.
    713         // FIXME: Should SkCodec have an extra memory buffer that can be shared among
    714         //        all of the implementations that use temporary/garbage memory?
    715         SkAutoTMalloc<JSAMPLE> dummyRow(sizeInfo.fYWidthBytes);
    716         for (int i = remainingRows; i < numYRowsPerBlock; i++) {
    717             rowptrs[i] = dummyRow.get();
    718         }
    719         int remainingUVRows = dinfo->comp_info[1].downsampled_height - DCTSIZE * numIters;
    720         for (int i = remainingUVRows; i < DCTSIZE; i++) {
    721             rowptrs[i + 2 * DCTSIZE] = dummyRow.get();
    722             rowptrs[i + 3 * DCTSIZE] = dummyRow.get();
    723         }
    724 
    725         JDIMENSION linesRead = jpeg_read_raw_data(dinfo, yuv, numRowsPerBlock);
    726         if (linesRead < remainingRows) {
    727             // FIXME: Handle incomplete YUV decodes without signalling an error.
    728             return kInvalidInput;
    729         }
    730     }
    731 
    732     return kSuccess;
    733 }
    734