1*3ac0a46fSAndroid Build Coastguard Worker // Copyright 2016 The PDFium Authors
2*3ac0a46fSAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be
3*3ac0a46fSAndroid Build Coastguard Worker // found in the LICENSE file.
4*3ac0a46fSAndroid Build Coastguard Worker
5*3ac0a46fSAndroid Build Coastguard Worker #include <cstdint>
6*3ac0a46fSAndroid Build Coastguard Worker #include <memory>
7*3ac0a46fSAndroid Build Coastguard Worker
8*3ac0a46fSAndroid Build Coastguard Worker #include "core/fpdfapi/page/cpdf_colorspace.h"
9*3ac0a46fSAndroid Build Coastguard Worker #include "core/fxcodec/jpx/cjpx_decoder.h"
10*3ac0a46fSAndroid Build Coastguard Worker #include "core/fxcrt/fx_safe_types.h"
11*3ac0a46fSAndroid Build Coastguard Worker #include "core/fxge/dib/cfx_dibitmap.h"
12*3ac0a46fSAndroid Build Coastguard Worker #include "core/fxge/dib/fx_dib.h"
13*3ac0a46fSAndroid Build Coastguard Worker
14*3ac0a46fSAndroid Build Coastguard Worker namespace {
15*3ac0a46fSAndroid Build Coastguard Worker
16*3ac0a46fSAndroid Build Coastguard Worker const uint32_t kMaxJPXFuzzSize = 100 * 1024 * 1024; // 100 MB
17*3ac0a46fSAndroid Build Coastguard Worker
CheckImageSize(const CJPX_Decoder::JpxImageInfo & image_info)18*3ac0a46fSAndroid Build Coastguard Worker bool CheckImageSize(const CJPX_Decoder::JpxImageInfo& image_info) {
19*3ac0a46fSAndroid Build Coastguard Worker static constexpr uint32_t kMemLimitBytes = 1024 * 1024 * 1024; // 1 GB.
20*3ac0a46fSAndroid Build Coastguard Worker FX_SAFE_UINT32 mem = image_info.width;
21*3ac0a46fSAndroid Build Coastguard Worker mem *= image_info.height;
22*3ac0a46fSAndroid Build Coastguard Worker mem *= image_info.channels;
23*3ac0a46fSAndroid Build Coastguard Worker return mem.IsValid() && mem.ValueOrDie() <= kMemLimitBytes;
24*3ac0a46fSAndroid Build Coastguard Worker }
25*3ac0a46fSAndroid Build Coastguard Worker
26*3ac0a46fSAndroid Build Coastguard Worker } // namespace
27*3ac0a46fSAndroid Build Coastguard Worker
LLVMFuzzerTestOneInput(const uint8_t * data,size_t size)28*3ac0a46fSAndroid Build Coastguard Worker extern "C" int LLVMFuzzerTestOneInput(const uint8_t* data, size_t size) {
29*3ac0a46fSAndroid Build Coastguard Worker if (size < 2)
30*3ac0a46fSAndroid Build Coastguard Worker return 0;
31*3ac0a46fSAndroid Build Coastguard Worker
32*3ac0a46fSAndroid Build Coastguard Worker std::unique_ptr<CJPX_Decoder> decoder = CJPX_Decoder::Create(
33*3ac0a46fSAndroid Build Coastguard Worker {data + 2, size - 2},
34*3ac0a46fSAndroid Build Coastguard Worker static_cast<CJPX_Decoder::ColorSpaceOption>(data[0] % 3), data[1]);
35*3ac0a46fSAndroid Build Coastguard Worker if (!decoder)
36*3ac0a46fSAndroid Build Coastguard Worker return 0;
37*3ac0a46fSAndroid Build Coastguard Worker
38*3ac0a46fSAndroid Build Coastguard Worker // A call to StartDecode could be too expensive if image size is very big, so
39*3ac0a46fSAndroid Build Coastguard Worker // check size before calling StartDecode().
40*3ac0a46fSAndroid Build Coastguard Worker CJPX_Decoder::JpxImageInfo image_info = decoder->GetInfo();
41*3ac0a46fSAndroid Build Coastguard Worker if (!CheckImageSize(image_info))
42*3ac0a46fSAndroid Build Coastguard Worker return 0;
43*3ac0a46fSAndroid Build Coastguard Worker
44*3ac0a46fSAndroid Build Coastguard Worker if (!decoder->StartDecode())
45*3ac0a46fSAndroid Build Coastguard Worker return 0;
46*3ac0a46fSAndroid Build Coastguard Worker
47*3ac0a46fSAndroid Build Coastguard Worker // StartDecode() could change image size, so check again.
48*3ac0a46fSAndroid Build Coastguard Worker image_info = decoder->GetInfo();
49*3ac0a46fSAndroid Build Coastguard Worker if (!CheckImageSize(image_info))
50*3ac0a46fSAndroid Build Coastguard Worker return 0;
51*3ac0a46fSAndroid Build Coastguard Worker
52*3ac0a46fSAndroid Build Coastguard Worker FXDIB_Format format;
53*3ac0a46fSAndroid Build Coastguard Worker if (image_info.channels == 1) {
54*3ac0a46fSAndroid Build Coastguard Worker format = FXDIB_Format::k8bppRgb;
55*3ac0a46fSAndroid Build Coastguard Worker } else if (image_info.channels <= 3) {
56*3ac0a46fSAndroid Build Coastguard Worker format = FXDIB_Format::kRgb;
57*3ac0a46fSAndroid Build Coastguard Worker } else if (image_info.channels == 4) {
58*3ac0a46fSAndroid Build Coastguard Worker format = FXDIB_Format::kRgb32;
59*3ac0a46fSAndroid Build Coastguard Worker } else {
60*3ac0a46fSAndroid Build Coastguard Worker image_info.width = (image_info.width * image_info.channels + 2) / 3;
61*3ac0a46fSAndroid Build Coastguard Worker format = FXDIB_Format::kRgb;
62*3ac0a46fSAndroid Build Coastguard Worker }
63*3ac0a46fSAndroid Build Coastguard Worker auto bitmap = pdfium::MakeRetain<CFX_DIBitmap>();
64*3ac0a46fSAndroid Build Coastguard Worker if (!bitmap->Create(image_info.width, image_info.height, format))
65*3ac0a46fSAndroid Build Coastguard Worker return 0;
66*3ac0a46fSAndroid Build Coastguard Worker
67*3ac0a46fSAndroid Build Coastguard Worker if (bitmap->GetHeight() <= 0 ||
68*3ac0a46fSAndroid Build Coastguard Worker kMaxJPXFuzzSize / bitmap->GetPitch() <
69*3ac0a46fSAndroid Build Coastguard Worker static_cast<uint32_t>(bitmap->GetHeight()))
70*3ac0a46fSAndroid Build Coastguard Worker return 0;
71*3ac0a46fSAndroid Build Coastguard Worker
72*3ac0a46fSAndroid Build Coastguard Worker decoder->Decode(bitmap->GetWritableBuffer(), bitmap->GetPitch(),
73*3ac0a46fSAndroid Build Coastguard Worker /*swap_rgb=*/false, GetCompsFromFormat(format));
74*3ac0a46fSAndroid Build Coastguard Worker
75*3ac0a46fSAndroid Build Coastguard Worker return 0;
76*3ac0a46fSAndroid Build Coastguard Worker }
77