|  | /* | 
|  | * Copyright 2015 Google Inc. | 
|  | * | 
|  | * Use of this source code is governed by a BSD-style license that can be | 
|  | * found in the LICENSE file. | 
|  | */ | 
|  | #include "src/codec/SkJpegDecoderMgr.h" | 
|  |  | 
|  | #include "include/core/SkTypes.h" | 
|  | #include "src/codec/SkCodecPriv.h" | 
|  | #include "src/codec/SkJpegSourceMgr.h" | 
|  | #include "src/codec/SkJpegUtility.h" | 
|  |  | 
|  | #ifdef SK_BUILD_FOR_ANDROID_FRAMEWORK | 
|  | #include "include/android/SkAndroidFrameworkUtils.h" | 
|  | #endif | 
|  |  | 
|  | #include <cstddef> | 
|  | #include <utility> | 
|  |  | 
|  | class SkStream; | 
|  |  | 
|  | /* | 
|  | * Print information, warning, and error messages | 
|  | */ | 
|  | static void print_message(const j_common_ptr info, const char caller[]) { | 
|  | char buffer[JMSG_LENGTH_MAX]; | 
|  | info->err->format_message(info, buffer); | 
|  | SkCodecPrintf("libjpeg error %d <%s> from %s\n", info->err->msg_code, buffer, caller); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Reporting function for error and warning messages. | 
|  | */ | 
|  | static void output_message(j_common_ptr info) { | 
|  | print_message(info, "output_message"); | 
|  | } | 
|  |  | 
|  | static void progress_monitor(j_common_ptr info) { | 
|  | int scan = ((j_decompress_ptr)info)->input_scan_number; | 
|  | // Progressive images with a very large number of scans can cause the | 
|  | // decoder to hang.  Here we use the progress monitor to abort on | 
|  | // a very large number of scans.  100 is arbitrary, but much larger | 
|  | // than the number of scans we might expect in a normal image. | 
|  | if (scan >= 100) { | 
|  | skjpeg_err_exit(info); | 
|  | } | 
|  | } | 
|  |  | 
|  | //////////////////////////////////////////////////////////////////////////////////////////////////// | 
|  | // JpegDecoderMgr | 
|  |  | 
|  | bool JpegDecoderMgr::returnFalse(const char caller[]) { | 
|  | print_message((j_common_ptr) &fDInfo, caller); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | SkCodec::Result JpegDecoderMgr::returnFailure(const char caller[], SkCodec::Result result) { | 
|  | print_message((j_common_ptr) &fDInfo, caller); | 
|  | return result; | 
|  | } | 
|  |  | 
|  | bool JpegDecoderMgr::getEncodedColor(SkEncodedInfo::Color* outColor) { | 
|  | switch (fDInfo.jpeg_color_space) { | 
|  | case JCS_GRAYSCALE: | 
|  | *outColor = SkEncodedInfo::kGray_Color; | 
|  | return true; | 
|  | case JCS_YCbCr: | 
|  | *outColor = SkEncodedInfo::kYUV_Color; | 
|  | return true; | 
|  | case JCS_RGB: | 
|  | #ifdef SK_BUILD_FOR_ANDROID_FRAMEWORK | 
|  | SkAndroidFrameworkUtils::SafetyNetLog("118372692"); | 
|  | #endif | 
|  | *outColor = SkEncodedInfo::kRGB_Color; | 
|  | return true; | 
|  | case JCS_YCCK: | 
|  | *outColor = SkEncodedInfo::kYCCK_Color; | 
|  | return true; | 
|  | case JCS_CMYK: | 
|  | *outColor = SkEncodedInfo::kInvertedCMYK_Color; | 
|  | return true; | 
|  | default: | 
|  | return false; | 
|  | } | 
|  | } | 
|  |  | 
|  | SkJpegSourceMgr* JpegDecoderMgr::getSourceMgr() { | 
|  | return fSrcMgr.fSourceMgr.get(); | 
|  | } | 
|  |  | 
|  | JpegDecoderMgr::JpegDecoderMgr(SkStream* stream) | 
|  | : fSrcMgr(SkJpegSourceMgr::Make(stream)), fInit(false) { | 
|  | // Error manager must be set before any calls to libjeg in order to handle failures | 
|  | fDInfo.err = jpeg_std_error(&fErrorMgr); | 
|  | fErrorMgr.error_exit = skjpeg_err_exit; | 
|  | } | 
|  |  | 
|  | void JpegDecoderMgr::init() { | 
|  | jpeg_create_decompress(&fDInfo); | 
|  | fInit = true; | 
|  | fDInfo.src = &fSrcMgr; | 
|  | fDInfo.err->output_message = &output_message; | 
|  | fDInfo.progress = &fProgressMgr; | 
|  | fProgressMgr.progress_monitor = &progress_monitor; | 
|  | } | 
|  |  | 
|  | JpegDecoderMgr::~JpegDecoderMgr() { | 
|  | if (fInit) { | 
|  | jpeg_destroy_decompress(&fDInfo); | 
|  | } | 
|  | } | 
|  |  | 
|  | //////////////////////////////////////////////////////////////////////////////////////////////////// | 
|  | // JpegDecoderMgr::SourceMgr | 
|  |  | 
|  | // static | 
|  | void JpegDecoderMgr::SourceMgr::InitSource(j_decompress_ptr dinfo) { | 
|  | JpegDecoderMgr::SourceMgr* src = (JpegDecoderMgr::SourceMgr*)dinfo->src; | 
|  | src->fSourceMgr->initSource(src->next_input_byte, src->bytes_in_buffer); | 
|  | } | 
|  |  | 
|  | // static | 
|  | void JpegDecoderMgr::SourceMgr::SkipInputData(j_decompress_ptr dinfo, long num_bytes_long) { | 
|  | JpegDecoderMgr::SourceMgr* src = (JpegDecoderMgr::SourceMgr*)dinfo->src; | 
|  | size_t num_bytes = static_cast<size_t>(num_bytes_long); | 
|  | if (!src->fSourceMgr->skipInputBytes(num_bytes, src->next_input_byte, src->bytes_in_buffer)) { | 
|  | SkCodecPrintf("Failure to skip.\n"); | 
|  | src->next_input_byte = nullptr; | 
|  | src->bytes_in_buffer = 0; | 
|  | dinfo->err->error_exit((j_common_ptr)dinfo); | 
|  | } | 
|  | } | 
|  |  | 
|  | // static | 
|  | boolean JpegDecoderMgr::SourceMgr::FillInputBuffer(j_decompress_ptr dinfo) { | 
|  | JpegDecoderMgr::SourceMgr* src = (JpegDecoderMgr::SourceMgr*)dinfo->src; | 
|  | if (!src->fSourceMgr->fillInputBuffer(src->next_input_byte, src->bytes_in_buffer)) { | 
|  | SkCodecPrintf("Failure to fill input buffer.\n"); | 
|  | src->next_input_byte = nullptr; | 
|  | src->bytes_in_buffer = 0; | 
|  | return false; | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  | // static | 
|  | void JpegDecoderMgr::SourceMgr::TermSource(j_decompress_ptr dinfo) {} | 
|  |  | 
|  | JpegDecoderMgr::SourceMgr::SourceMgr(std::unique_ptr<SkJpegSourceMgr> sourceMgr) | 
|  | : fSourceMgr(std::move(sourceMgr)) { | 
|  | init_source = JpegDecoderMgr::SourceMgr::InitSource; | 
|  | fill_input_buffer = JpegDecoderMgr::SourceMgr::FillInputBuffer; | 
|  | skip_input_data = JpegDecoderMgr::SourceMgr::SkipInputData; | 
|  | resync_to_restart = jpeg_resync_to_restart; | 
|  | term_source = JpegDecoderMgr::SourceMgr::TermSource; | 
|  | } |