1*c8dee2aaSAndroid Build Coastguard Worker /*
2*c8dee2aaSAndroid Build Coastguard Worker * Copyright 2015 Google Inc.
3*c8dee2aaSAndroid Build Coastguard Worker *
4*c8dee2aaSAndroid Build Coastguard Worker * Use of this source code is governed by a BSD-style license that can be
5*c8dee2aaSAndroid Build Coastguard Worker * found in the LICENSE file.
6*c8dee2aaSAndroid Build Coastguard Worker */
7*c8dee2aaSAndroid Build Coastguard Worker #include "src/codec/SkJpegDecoderMgr.h"
8*c8dee2aaSAndroid Build Coastguard Worker
9*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkTypes.h"
10*c8dee2aaSAndroid Build Coastguard Worker #include "src/codec/SkCodecPriv.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "src/codec/SkJpegSourceMgr.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "src/codec/SkJpegUtility.h"
13*c8dee2aaSAndroid Build Coastguard Worker
14*c8dee2aaSAndroid Build Coastguard Worker #ifdef SK_BUILD_FOR_ANDROID_FRAMEWORK
15*c8dee2aaSAndroid Build Coastguard Worker #include "include/android/SkAndroidFrameworkUtils.h"
16*c8dee2aaSAndroid Build Coastguard Worker #endif
17*c8dee2aaSAndroid Build Coastguard Worker
18*c8dee2aaSAndroid Build Coastguard Worker #include <cstddef>
19*c8dee2aaSAndroid Build Coastguard Worker #include <utility>
20*c8dee2aaSAndroid Build Coastguard Worker
21*c8dee2aaSAndroid Build Coastguard Worker class SkStream;
22*c8dee2aaSAndroid Build Coastguard Worker
23*c8dee2aaSAndroid Build Coastguard Worker /*
24*c8dee2aaSAndroid Build Coastguard Worker * Print information, warning, and error messages
25*c8dee2aaSAndroid Build Coastguard Worker */
print_message(const j_common_ptr info,const char caller[])26*c8dee2aaSAndroid Build Coastguard Worker static void print_message(const j_common_ptr info, const char caller[]) {
27*c8dee2aaSAndroid Build Coastguard Worker char buffer[JMSG_LENGTH_MAX];
28*c8dee2aaSAndroid Build Coastguard Worker info->err->format_message(info, buffer);
29*c8dee2aaSAndroid Build Coastguard Worker SkCodecPrintf("libjpeg error %d <%s> from %s\n", info->err->msg_code, buffer, caller);
30*c8dee2aaSAndroid Build Coastguard Worker }
31*c8dee2aaSAndroid Build Coastguard Worker
32*c8dee2aaSAndroid Build Coastguard Worker /*
33*c8dee2aaSAndroid Build Coastguard Worker * Reporting function for error and warning messages.
34*c8dee2aaSAndroid Build Coastguard Worker */
output_message(j_common_ptr info)35*c8dee2aaSAndroid Build Coastguard Worker static void output_message(j_common_ptr info) {
36*c8dee2aaSAndroid Build Coastguard Worker print_message(info, "output_message");
37*c8dee2aaSAndroid Build Coastguard Worker }
38*c8dee2aaSAndroid Build Coastguard Worker
progress_monitor(j_common_ptr info)39*c8dee2aaSAndroid Build Coastguard Worker static void progress_monitor(j_common_ptr info) {
40*c8dee2aaSAndroid Build Coastguard Worker int scan = ((j_decompress_ptr)info)->input_scan_number;
41*c8dee2aaSAndroid Build Coastguard Worker // Progressive images with a very large number of scans can cause the
42*c8dee2aaSAndroid Build Coastguard Worker // decoder to hang. Here we use the progress monitor to abort on
43*c8dee2aaSAndroid Build Coastguard Worker // a very large number of scans. 100 is arbitrary, but much larger
44*c8dee2aaSAndroid Build Coastguard Worker // than the number of scans we might expect in a normal image.
45*c8dee2aaSAndroid Build Coastguard Worker if (scan >= 100) {
46*c8dee2aaSAndroid Build Coastguard Worker skjpeg_err_exit(info);
47*c8dee2aaSAndroid Build Coastguard Worker }
48*c8dee2aaSAndroid Build Coastguard Worker }
49*c8dee2aaSAndroid Build Coastguard Worker
50*c8dee2aaSAndroid Build Coastguard Worker ////////////////////////////////////////////////////////////////////////////////////////////////////
51*c8dee2aaSAndroid Build Coastguard Worker // JpegDecoderMgr
52*c8dee2aaSAndroid Build Coastguard Worker
returnFalse(const char caller[])53*c8dee2aaSAndroid Build Coastguard Worker bool JpegDecoderMgr::returnFalse(const char caller[]) {
54*c8dee2aaSAndroid Build Coastguard Worker print_message((j_common_ptr) &fDInfo, caller);
55*c8dee2aaSAndroid Build Coastguard Worker return false;
56*c8dee2aaSAndroid Build Coastguard Worker }
57*c8dee2aaSAndroid Build Coastguard Worker
returnFailure(const char caller[],SkCodec::Result result)58*c8dee2aaSAndroid Build Coastguard Worker SkCodec::Result JpegDecoderMgr::returnFailure(const char caller[], SkCodec::Result result) {
59*c8dee2aaSAndroid Build Coastguard Worker print_message((j_common_ptr) &fDInfo, caller);
60*c8dee2aaSAndroid Build Coastguard Worker return result;
61*c8dee2aaSAndroid Build Coastguard Worker }
62*c8dee2aaSAndroid Build Coastguard Worker
getEncodedColor(SkEncodedInfo::Color * outColor)63*c8dee2aaSAndroid Build Coastguard Worker bool JpegDecoderMgr::getEncodedColor(SkEncodedInfo::Color* outColor) {
64*c8dee2aaSAndroid Build Coastguard Worker switch (fDInfo.jpeg_color_space) {
65*c8dee2aaSAndroid Build Coastguard Worker case JCS_GRAYSCALE:
66*c8dee2aaSAndroid Build Coastguard Worker *outColor = SkEncodedInfo::kGray_Color;
67*c8dee2aaSAndroid Build Coastguard Worker return true;
68*c8dee2aaSAndroid Build Coastguard Worker case JCS_YCbCr:
69*c8dee2aaSAndroid Build Coastguard Worker *outColor = SkEncodedInfo::kYUV_Color;
70*c8dee2aaSAndroid Build Coastguard Worker return true;
71*c8dee2aaSAndroid Build Coastguard Worker case JCS_RGB:
72*c8dee2aaSAndroid Build Coastguard Worker #ifdef SK_BUILD_FOR_ANDROID_FRAMEWORK
73*c8dee2aaSAndroid Build Coastguard Worker SkAndroidFrameworkUtils::SafetyNetLog("118372692");
74*c8dee2aaSAndroid Build Coastguard Worker #endif
75*c8dee2aaSAndroid Build Coastguard Worker *outColor = SkEncodedInfo::kRGB_Color;
76*c8dee2aaSAndroid Build Coastguard Worker return true;
77*c8dee2aaSAndroid Build Coastguard Worker case JCS_YCCK:
78*c8dee2aaSAndroid Build Coastguard Worker *outColor = SkEncodedInfo::kYCCK_Color;
79*c8dee2aaSAndroid Build Coastguard Worker return true;
80*c8dee2aaSAndroid Build Coastguard Worker case JCS_CMYK:
81*c8dee2aaSAndroid Build Coastguard Worker *outColor = SkEncodedInfo::kInvertedCMYK_Color;
82*c8dee2aaSAndroid Build Coastguard Worker return true;
83*c8dee2aaSAndroid Build Coastguard Worker default:
84*c8dee2aaSAndroid Build Coastguard Worker return false;
85*c8dee2aaSAndroid Build Coastguard Worker }
86*c8dee2aaSAndroid Build Coastguard Worker }
87*c8dee2aaSAndroid Build Coastguard Worker
getSourceMgr()88*c8dee2aaSAndroid Build Coastguard Worker SkJpegSourceMgr* JpegDecoderMgr::getSourceMgr() {
89*c8dee2aaSAndroid Build Coastguard Worker return fSrcMgr.fSourceMgr.get();
90*c8dee2aaSAndroid Build Coastguard Worker }
91*c8dee2aaSAndroid Build Coastguard Worker
JpegDecoderMgr(SkStream * stream)92*c8dee2aaSAndroid Build Coastguard Worker JpegDecoderMgr::JpegDecoderMgr(SkStream* stream)
93*c8dee2aaSAndroid Build Coastguard Worker : fSrcMgr(SkJpegSourceMgr::Make(stream)), fInit(false) {
94*c8dee2aaSAndroid Build Coastguard Worker // An error manager must be set before any calls to libjpeg, in order to handle failures.
95*c8dee2aaSAndroid Build Coastguard Worker fDInfo.err = jpeg_std_error(&fErrorMgr);
96*c8dee2aaSAndroid Build Coastguard Worker fErrorMgr.error_exit = skjpeg_err_exit;
97*c8dee2aaSAndroid Build Coastguard Worker }
98*c8dee2aaSAndroid Build Coastguard Worker
init()99*c8dee2aaSAndroid Build Coastguard Worker void JpegDecoderMgr::init() {
100*c8dee2aaSAndroid Build Coastguard Worker jpeg_create_decompress(&fDInfo);
101*c8dee2aaSAndroid Build Coastguard Worker fInit = true;
102*c8dee2aaSAndroid Build Coastguard Worker fDInfo.src = &fSrcMgr;
103*c8dee2aaSAndroid Build Coastguard Worker fDInfo.err->output_message = &output_message;
104*c8dee2aaSAndroid Build Coastguard Worker fDInfo.progress = &fProgressMgr;
105*c8dee2aaSAndroid Build Coastguard Worker fProgressMgr.progress_monitor = &progress_monitor;
106*c8dee2aaSAndroid Build Coastguard Worker }
107*c8dee2aaSAndroid Build Coastguard Worker
~JpegDecoderMgr()108*c8dee2aaSAndroid Build Coastguard Worker JpegDecoderMgr::~JpegDecoderMgr() {
109*c8dee2aaSAndroid Build Coastguard Worker if (fInit) {
110*c8dee2aaSAndroid Build Coastguard Worker jpeg_destroy_decompress(&fDInfo);
111*c8dee2aaSAndroid Build Coastguard Worker }
112*c8dee2aaSAndroid Build Coastguard Worker }
113*c8dee2aaSAndroid Build Coastguard Worker
114*c8dee2aaSAndroid Build Coastguard Worker ////////////////////////////////////////////////////////////////////////////////////////////////////
115*c8dee2aaSAndroid Build Coastguard Worker // JpegDecoderMgr::SourceMgr
116*c8dee2aaSAndroid Build Coastguard Worker
117*c8dee2aaSAndroid Build Coastguard Worker // static
InitSource(j_decompress_ptr dinfo)118*c8dee2aaSAndroid Build Coastguard Worker void JpegDecoderMgr::SourceMgr::InitSource(j_decompress_ptr dinfo) {
119*c8dee2aaSAndroid Build Coastguard Worker JpegDecoderMgr::SourceMgr* src = (JpegDecoderMgr::SourceMgr*)dinfo->src;
120*c8dee2aaSAndroid Build Coastguard Worker src->fSourceMgr->initSource(src->next_input_byte, src->bytes_in_buffer);
121*c8dee2aaSAndroid Build Coastguard Worker }
122*c8dee2aaSAndroid Build Coastguard Worker
123*c8dee2aaSAndroid Build Coastguard Worker // static
SkipInputData(j_decompress_ptr dinfo,long num_bytes_long)124*c8dee2aaSAndroid Build Coastguard Worker void JpegDecoderMgr::SourceMgr::SkipInputData(j_decompress_ptr dinfo, long num_bytes_long) {
125*c8dee2aaSAndroid Build Coastguard Worker JpegDecoderMgr::SourceMgr* src = (JpegDecoderMgr::SourceMgr*)dinfo->src;
126*c8dee2aaSAndroid Build Coastguard Worker size_t num_bytes = static_cast<size_t>(num_bytes_long);
127*c8dee2aaSAndroid Build Coastguard Worker if (!src->fSourceMgr->skipInputBytes(num_bytes, src->next_input_byte, src->bytes_in_buffer)) {
128*c8dee2aaSAndroid Build Coastguard Worker SkCodecPrintf("Failure to skip.\n");
129*c8dee2aaSAndroid Build Coastguard Worker src->next_input_byte = nullptr;
130*c8dee2aaSAndroid Build Coastguard Worker src->bytes_in_buffer = 0;
131*c8dee2aaSAndroid Build Coastguard Worker dinfo->err->error_exit((j_common_ptr)dinfo);
132*c8dee2aaSAndroid Build Coastguard Worker }
133*c8dee2aaSAndroid Build Coastguard Worker }
134*c8dee2aaSAndroid Build Coastguard Worker
135*c8dee2aaSAndroid Build Coastguard Worker // static
FillInputBuffer(j_decompress_ptr dinfo)136*c8dee2aaSAndroid Build Coastguard Worker boolean JpegDecoderMgr::SourceMgr::FillInputBuffer(j_decompress_ptr dinfo) {
137*c8dee2aaSAndroid Build Coastguard Worker JpegDecoderMgr::SourceMgr* src = (JpegDecoderMgr::SourceMgr*)dinfo->src;
138*c8dee2aaSAndroid Build Coastguard Worker if (!src->fSourceMgr->fillInputBuffer(src->next_input_byte, src->bytes_in_buffer)) {
139*c8dee2aaSAndroid Build Coastguard Worker SkCodecPrintf("Failure to fill input buffer.\n");
140*c8dee2aaSAndroid Build Coastguard Worker src->next_input_byte = nullptr;
141*c8dee2aaSAndroid Build Coastguard Worker src->bytes_in_buffer = 0;
142*c8dee2aaSAndroid Build Coastguard Worker return false;
143*c8dee2aaSAndroid Build Coastguard Worker }
144*c8dee2aaSAndroid Build Coastguard Worker return true;
145*c8dee2aaSAndroid Build Coastguard Worker }
146*c8dee2aaSAndroid Build Coastguard Worker
147*c8dee2aaSAndroid Build Coastguard Worker // static
TermSource(j_decompress_ptr dinfo)148*c8dee2aaSAndroid Build Coastguard Worker void JpegDecoderMgr::SourceMgr::TermSource(j_decompress_ptr dinfo) {}
149*c8dee2aaSAndroid Build Coastguard Worker
SourceMgr(std::unique_ptr<SkJpegSourceMgr> sourceMgr)150*c8dee2aaSAndroid Build Coastguard Worker JpegDecoderMgr::SourceMgr::SourceMgr(std::unique_ptr<SkJpegSourceMgr> sourceMgr)
151*c8dee2aaSAndroid Build Coastguard Worker : fSourceMgr(std::move(sourceMgr)) {
152*c8dee2aaSAndroid Build Coastguard Worker init_source = JpegDecoderMgr::SourceMgr::InitSource;
153*c8dee2aaSAndroid Build Coastguard Worker fill_input_buffer = JpegDecoderMgr::SourceMgr::FillInputBuffer;
154*c8dee2aaSAndroid Build Coastguard Worker skip_input_data = JpegDecoderMgr::SourceMgr::SkipInputData;
155*c8dee2aaSAndroid Build Coastguard Worker resync_to_restart = jpeg_resync_to_restart;
156*c8dee2aaSAndroid Build Coastguard Worker term_source = JpegDecoderMgr::SourceMgr::TermSource;
157*c8dee2aaSAndroid Build Coastguard Worker }
158