1cb93a386Sopenharmony_ci/*
2cb93a386Sopenharmony_ci * Copyright 2016 Google Inc.
3cb93a386Sopenharmony_ci *
4cb93a386Sopenharmony_ci * Use of this source code is governed by a BSD-style license that can be
5cb93a386Sopenharmony_ci * found in the LICENSE file.
6cb93a386Sopenharmony_ci */
7cb93a386Sopenharmony_ci
8cb93a386Sopenharmony_ci#include "include/codec/SkCodec.h"
9cb93a386Sopenharmony_ci#include "include/core/SkBitmap.h"
10cb93a386Sopenharmony_ci#include "include/core/SkData.h"
11cb93a386Sopenharmony_ci#include "include/core/SkImageInfo.h"
12cb93a386Sopenharmony_ci#include "include/core/SkRefCnt.h"
13cb93a386Sopenharmony_ci#include "include/core/SkStream.h"
14cb93a386Sopenharmony_ci#include "include/core/SkString.h"
15cb93a386Sopenharmony_ci#include "include/core/SkTypes.h"
16cb93a386Sopenharmony_ci#include "tests/CodecPriv.h"
17cb93a386Sopenharmony_ci#include "tests/FakeStreams.h"
18cb93a386Sopenharmony_ci#include "tests/Test.h"
19cb93a386Sopenharmony_ci#include "tools/Resources.h"
20cb93a386Sopenharmony_ci
21cb93a386Sopenharmony_ci#include <cstring>
22cb93a386Sopenharmony_ci#include <initializer_list>
23cb93a386Sopenharmony_ci#include <memory>
24cb93a386Sopenharmony_ci#include <utility>
25cb93a386Sopenharmony_ci#include <vector>
26cb93a386Sopenharmony_ci
27cb93a386Sopenharmony_cistatic SkImageInfo standardize_info(SkCodec* codec) {
28cb93a386Sopenharmony_ci    SkImageInfo defaultInfo = codec->getInfo();
29cb93a386Sopenharmony_ci    // Note: This drops the SkColorSpace, allowing the equality check between two
30cb93a386Sopenharmony_ci    // different codecs created from the same file to have the same SkImageInfo.
31cb93a386Sopenharmony_ci    return SkImageInfo::MakeN32Premul(defaultInfo.width(), defaultInfo.height());
32cb93a386Sopenharmony_ci}
33cb93a386Sopenharmony_ci
34cb93a386Sopenharmony_cistatic bool create_truth(sk_sp<SkData> data, SkBitmap* dst) {
35cb93a386Sopenharmony_ci    std::unique_ptr<SkCodec> codec(SkCodec::MakeFromData(std::move(data)));
36cb93a386Sopenharmony_ci    if (!codec) {
37cb93a386Sopenharmony_ci        return false;
38cb93a386Sopenharmony_ci    }
39cb93a386Sopenharmony_ci
40cb93a386Sopenharmony_ci    const SkImageInfo info = standardize_info(codec.get());
41cb93a386Sopenharmony_ci    dst->allocPixels(info);
42cb93a386Sopenharmony_ci    return SkCodec::kSuccess == codec->getPixels(info, dst->getPixels(), dst->rowBytes());
43cb93a386Sopenharmony_ci}
44cb93a386Sopenharmony_ci
45cb93a386Sopenharmony_cistatic bool compare_bitmaps(skiatest::Reporter* r, const SkBitmap& bm1, const SkBitmap& bm2) {
46cb93a386Sopenharmony_ci    const SkImageInfo& info = bm1.info();
47cb93a386Sopenharmony_ci    if (info != bm2.info()) {
48cb93a386Sopenharmony_ci        ERRORF(r, "Bitmaps have different image infos!");
49cb93a386Sopenharmony_ci        return false;
50cb93a386Sopenharmony_ci    }
51cb93a386Sopenharmony_ci    const size_t rowBytes = info.minRowBytes();
52cb93a386Sopenharmony_ci    for (int i = 0; i < info.height(); i++) {
53cb93a386Sopenharmony_ci        if (0 != memcmp(bm1.getAddr(0, i), bm2.getAddr(0, i), rowBytes)) {
54cb93a386Sopenharmony_ci            ERRORF(r, "Bitmaps have different pixels, starting on line %i!", i);
55cb93a386Sopenharmony_ci            return false;
56cb93a386Sopenharmony_ci        }
57cb93a386Sopenharmony_ci    }
58cb93a386Sopenharmony_ci
59cb93a386Sopenharmony_ci    return true;
60cb93a386Sopenharmony_ci}
61cb93a386Sopenharmony_ci
62cb93a386Sopenharmony_cistatic void test_partial(skiatest::Reporter* r, const char* name, const sk_sp<SkData>& file,
63cb93a386Sopenharmony_ci                         size_t minBytes, size_t increment) {
64cb93a386Sopenharmony_ci    SkBitmap truth;
65cb93a386Sopenharmony_ci    if (!create_truth(file, &truth)) {
66cb93a386Sopenharmony_ci        ERRORF(r, "Failed to decode %s\n", name);
67cb93a386Sopenharmony_ci        return;
68cb93a386Sopenharmony_ci    }
69cb93a386Sopenharmony_ci
70cb93a386Sopenharmony_ci    // Now decode part of the file
71cb93a386Sopenharmony_ci    HaltingStream* stream = new HaltingStream(file, minBytes);
72cb93a386Sopenharmony_ci
73cb93a386Sopenharmony_ci    // Note that we cheat and hold on to a pointer to stream, though it is owned by
74cb93a386Sopenharmony_ci    // partialCodec.
75cb93a386Sopenharmony_ci    auto partialCodec = SkCodec::MakeFromStream(std::unique_ptr<SkStream>(stream));
76cb93a386Sopenharmony_ci    if (!partialCodec) {
77cb93a386Sopenharmony_ci        ERRORF(r, "Failed to create codec for %s with %zu bytes", name, minBytes);
78cb93a386Sopenharmony_ci        return;
79cb93a386Sopenharmony_ci    }
80cb93a386Sopenharmony_ci
81cb93a386Sopenharmony_ci    const SkImageInfo info = standardize_info(partialCodec.get());
82cb93a386Sopenharmony_ci    SkASSERT(info == truth.info());
83cb93a386Sopenharmony_ci    SkBitmap incremental;
84cb93a386Sopenharmony_ci    incremental.allocPixels(info);
85cb93a386Sopenharmony_ci
86cb93a386Sopenharmony_ci    while (true) {
87cb93a386Sopenharmony_ci        const SkCodec::Result startResult = partialCodec->startIncrementalDecode(info,
88cb93a386Sopenharmony_ci                incremental.getPixels(), incremental.rowBytes());
89cb93a386Sopenharmony_ci        if (startResult == SkCodec::kSuccess) {
90cb93a386Sopenharmony_ci            break;
91cb93a386Sopenharmony_ci        }
92cb93a386Sopenharmony_ci
93cb93a386Sopenharmony_ci        if (stream->isAllDataReceived()) {
94cb93a386Sopenharmony_ci            ERRORF(r, "Failed to start incremental decode\n");
95cb93a386Sopenharmony_ci            return;
96cb93a386Sopenharmony_ci        }
97cb93a386Sopenharmony_ci
98cb93a386Sopenharmony_ci        stream->addNewData(increment);
99cb93a386Sopenharmony_ci    }
100cb93a386Sopenharmony_ci
101cb93a386Sopenharmony_ci    while (true) {
102cb93a386Sopenharmony_ci        // This imitates how Chromium calls getFrameCount before resuming a decode.
103cb93a386Sopenharmony_ci        partialCodec->getFrameCount();
104cb93a386Sopenharmony_ci
105cb93a386Sopenharmony_ci        const SkCodec::Result result = partialCodec->incrementalDecode();
106cb93a386Sopenharmony_ci
107cb93a386Sopenharmony_ci        if (result == SkCodec::kSuccess) {
108cb93a386Sopenharmony_ci            break;
109cb93a386Sopenharmony_ci        }
110cb93a386Sopenharmony_ci
111cb93a386Sopenharmony_ci        REPORTER_ASSERT(r, result == SkCodec::kIncompleteInput);
112cb93a386Sopenharmony_ci
113cb93a386Sopenharmony_ci        if (stream->isAllDataReceived()) {
114cb93a386Sopenharmony_ci            ERRORF(r, "Failed to completely decode %s", name);
115cb93a386Sopenharmony_ci            return;
116cb93a386Sopenharmony_ci        }
117cb93a386Sopenharmony_ci
118cb93a386Sopenharmony_ci        stream->addNewData(increment);
119cb93a386Sopenharmony_ci    }
120cb93a386Sopenharmony_ci
121cb93a386Sopenharmony_ci    // compare to original
122cb93a386Sopenharmony_ci    compare_bitmaps(r, truth, incremental);
123cb93a386Sopenharmony_ci}
124cb93a386Sopenharmony_ci
125cb93a386Sopenharmony_cistatic void test_partial(skiatest::Reporter* r, const char* name, size_t minBytes = 0) {
126cb93a386Sopenharmony_ci    sk_sp<SkData> file = GetResourceAsData(name);
127cb93a386Sopenharmony_ci    if (!file) {
128cb93a386Sopenharmony_ci        SkDebugf("missing resource %s\n", name);
129cb93a386Sopenharmony_ci        return;
130cb93a386Sopenharmony_ci    }
131cb93a386Sopenharmony_ci
132cb93a386Sopenharmony_ci    // This size is arbitrary, but deliberately different from the buffer size used by SkPngCodec.
133cb93a386Sopenharmony_ci    constexpr size_t kIncrement = 1000;
134cb93a386Sopenharmony_ci    test_partial(r, name, file, std::max(file->size() / 2, minBytes), kIncrement);
135cb93a386Sopenharmony_ci}
136cb93a386Sopenharmony_ci
137cb93a386Sopenharmony_ciDEF_TEST(Codec_partial, r) {
138cb93a386Sopenharmony_ci#if 0
139cb93a386Sopenharmony_ci    // FIXME (scroggo): SkPngCodec needs to use SkStreamBuffer in order to
140cb93a386Sopenharmony_ci    // support incremental decoding.
141cb93a386Sopenharmony_ci    test_partial(r, "images/plane.png");
142cb93a386Sopenharmony_ci    test_partial(r, "images/plane_interlaced.png");
143cb93a386Sopenharmony_ci    test_partial(r, "images/yellow_rose.png");
144cb93a386Sopenharmony_ci    test_partial(r, "images/index8.png");
145cb93a386Sopenharmony_ci    test_partial(r, "images/color_wheel.png");
146cb93a386Sopenharmony_ci    test_partial(r, "images/mandrill_256.png");
147cb93a386Sopenharmony_ci    test_partial(r, "images/mandrill_32.png");
148cb93a386Sopenharmony_ci    test_partial(r, "images/arrow.png");
149cb93a386Sopenharmony_ci    test_partial(r, "images/randPixels.png");
150cb93a386Sopenharmony_ci    test_partial(r, "images/baby_tux.png");
151cb93a386Sopenharmony_ci#endif
152cb93a386Sopenharmony_ci    test_partial(r, "images/box.gif");
153cb93a386Sopenharmony_ci    test_partial(r, "images/randPixels.gif", 215);
154cb93a386Sopenharmony_ci    test_partial(r, "images/color_wheel.gif");
155cb93a386Sopenharmony_ci}
156cb93a386Sopenharmony_ci
157cb93a386Sopenharmony_ciDEF_TEST(Codec_partialWuffs, r) {
158cb93a386Sopenharmony_ci    const char* path = "images/alphabetAnim.gif";
159cb93a386Sopenharmony_ci    auto file = GetResourceAsData(path);
160cb93a386Sopenharmony_ci    if (!file) {
161cb93a386Sopenharmony_ci        ERRORF(r, "missing %s", path);
162cb93a386Sopenharmony_ci    } else {
163cb93a386Sopenharmony_ci        // This is the end of the first frame. SkCodec will treat this as a
164cb93a386Sopenharmony_ci        // single frame gif.
165cb93a386Sopenharmony_ci        file = SkData::MakeSubset(file.get(), 0, 153);
166cb93a386Sopenharmony_ci        // Start with 100 to get a partial decode, then add the rest of the
167cb93a386Sopenharmony_ci        // first frame to decode a full image.
168cb93a386Sopenharmony_ci        test_partial(r, path, file, 100, 53);
169cb93a386Sopenharmony_ci    }
170cb93a386Sopenharmony_ci}
171cb93a386Sopenharmony_ci
172cb93a386Sopenharmony_ci// Verify that when decoding an animated gif byte by byte we report the correct
173cb93a386Sopenharmony_ci// fRequiredFrame as soon as getFrameInfo reports the frame.
174cb93a386Sopenharmony_ciDEF_TEST(Codec_requiredFrame, r) {
175cb93a386Sopenharmony_ci    auto path = "images/colorTables.gif";
176cb93a386Sopenharmony_ci    sk_sp<SkData> file = GetResourceAsData(path);
177cb93a386Sopenharmony_ci    if (!file) {
178cb93a386Sopenharmony_ci        return;
179cb93a386Sopenharmony_ci    }
180cb93a386Sopenharmony_ci
181cb93a386Sopenharmony_ci    std::unique_ptr<SkCodec> codec(SkCodec::MakeFromData(file));
182cb93a386Sopenharmony_ci    if (!codec) {
183cb93a386Sopenharmony_ci        ERRORF(r, "Failed to create codec from %s", path);
184cb93a386Sopenharmony_ci        return;
185cb93a386Sopenharmony_ci    }
186cb93a386Sopenharmony_ci
187cb93a386Sopenharmony_ci    auto frameInfo = codec->getFrameInfo();
188cb93a386Sopenharmony_ci    if (frameInfo.size() <= 1) {
189cb93a386Sopenharmony_ci        ERRORF(r, "Test is uninteresting with 0 or 1 frames");
190cb93a386Sopenharmony_ci        return;
191cb93a386Sopenharmony_ci    }
192cb93a386Sopenharmony_ci
193cb93a386Sopenharmony_ci    HaltingStream* stream(nullptr);
194cb93a386Sopenharmony_ci    std::unique_ptr<SkCodec> partialCodec(nullptr);
195cb93a386Sopenharmony_ci    for (size_t i = 0; !partialCodec; i++) {
196cb93a386Sopenharmony_ci        if (file->size() == i) {
197cb93a386Sopenharmony_ci            ERRORF(r, "Should have created a partial codec for %s", path);
198cb93a386Sopenharmony_ci            return;
199cb93a386Sopenharmony_ci        }
200cb93a386Sopenharmony_ci        stream = new HaltingStream(file, i);
201cb93a386Sopenharmony_ci        partialCodec = SkCodec::MakeFromStream(std::unique_ptr<SkStream>(stream));
202cb93a386Sopenharmony_ci    }
203cb93a386Sopenharmony_ci
204cb93a386Sopenharmony_ci    std::vector<SkCodec::FrameInfo> partialInfo;
205cb93a386Sopenharmony_ci    size_t frameToCompare = 0;
206cb93a386Sopenharmony_ci    while (true) {
207cb93a386Sopenharmony_ci        partialInfo = partialCodec->getFrameInfo();
208cb93a386Sopenharmony_ci        for (; frameToCompare < partialInfo.size(); frameToCompare++) {
209cb93a386Sopenharmony_ci            REPORTER_ASSERT(r, partialInfo[frameToCompare].fRequiredFrame
210cb93a386Sopenharmony_ci                                == frameInfo[frameToCompare].fRequiredFrame);
211cb93a386Sopenharmony_ci        }
212cb93a386Sopenharmony_ci
213cb93a386Sopenharmony_ci        if (frameToCompare == frameInfo.size()) {
214cb93a386Sopenharmony_ci            break;
215cb93a386Sopenharmony_ci        }
216cb93a386Sopenharmony_ci
217cb93a386Sopenharmony_ci        if (stream->getLength() == file->size()) {
218cb93a386Sopenharmony_ci            ERRORF(r, "Should have found all frames for %s", path);
219cb93a386Sopenharmony_ci            return;
220cb93a386Sopenharmony_ci        }
221cb93a386Sopenharmony_ci        stream->addNewData(1);
222cb93a386Sopenharmony_ci    }
223cb93a386Sopenharmony_ci}
224cb93a386Sopenharmony_ci
225cb93a386Sopenharmony_ciDEF_TEST(Codec_partialAnim, r) {
226cb93a386Sopenharmony_ci    auto path = "images/test640x479.gif";
227cb93a386Sopenharmony_ci    sk_sp<SkData> file = GetResourceAsData(path);
228cb93a386Sopenharmony_ci    if (!file) {
229cb93a386Sopenharmony_ci        return;
230cb93a386Sopenharmony_ci    }
231cb93a386Sopenharmony_ci
232cb93a386Sopenharmony_ci    // This stream will be owned by fullCodec, but we hang on to the pointer
233cb93a386Sopenharmony_ci    // to determine frame offsets.
234cb93a386Sopenharmony_ci    std::unique_ptr<SkCodec> fullCodec(SkCodec::MakeFromStream(std::make_unique<SkMemoryStream>(file)));
235cb93a386Sopenharmony_ci    const auto info = standardize_info(fullCodec.get());
236cb93a386Sopenharmony_ci
237cb93a386Sopenharmony_ci    // frameByteCounts stores the number of bytes to decode a particular frame.
238cb93a386Sopenharmony_ci    // - [0] is the number of bytes for the header
239cb93a386Sopenharmony_ci    // - frames[i] requires frameByteCounts[i+1] bytes to decode
240cb93a386Sopenharmony_ci    const std::vector<size_t> frameByteCounts = { 455, 69350, 1344, 1346, 1327 };
241cb93a386Sopenharmony_ci    std::vector<SkBitmap> frames;
242cb93a386Sopenharmony_ci    for (size_t i = 0; true; i++) {
243cb93a386Sopenharmony_ci        SkBitmap frame;
244cb93a386Sopenharmony_ci        frame.allocPixels(info);
245cb93a386Sopenharmony_ci
246cb93a386Sopenharmony_ci        SkCodec::Options opts;
247cb93a386Sopenharmony_ci        opts.fFrameIndex = i;
248cb93a386Sopenharmony_ci        const SkCodec::Result result = fullCodec->getPixels(info, frame.getPixels(),
249cb93a386Sopenharmony_ci                frame.rowBytes(), &opts);
250cb93a386Sopenharmony_ci
251cb93a386Sopenharmony_ci        if (result == SkCodec::kIncompleteInput || result == SkCodec::kInvalidInput) {
252cb93a386Sopenharmony_ci            // We need to distinguish between a partial frame and no more frames.
253cb93a386Sopenharmony_ci            // getFrameInfo lets us do this, since it tells the number of frames
254cb93a386Sopenharmony_ci            // not considering whether they are complete.
255cb93a386Sopenharmony_ci            // FIXME: Should we use a different Result?
256cb93a386Sopenharmony_ci            if (fullCodec->getFrameInfo().size() > i) {
257cb93a386Sopenharmony_ci                // This is a partial frame.
258cb93a386Sopenharmony_ci                frames.push_back(frame);
259cb93a386Sopenharmony_ci            }
260cb93a386Sopenharmony_ci            break;
261cb93a386Sopenharmony_ci        }
262cb93a386Sopenharmony_ci
263cb93a386Sopenharmony_ci        if (result != SkCodec::kSuccess) {
264cb93a386Sopenharmony_ci            ERRORF(r, "Failed to decode frame %zu from %s", i, path);
265cb93a386Sopenharmony_ci            return;
266cb93a386Sopenharmony_ci        }
267cb93a386Sopenharmony_ci
268cb93a386Sopenharmony_ci        frames.push_back(frame);
269cb93a386Sopenharmony_ci    }
270cb93a386Sopenharmony_ci
271cb93a386Sopenharmony_ci    // Now decode frames partially, then completely, and compare to the original.
272cb93a386Sopenharmony_ci    HaltingStream* haltingStream = new HaltingStream(file, frameByteCounts[0]);
273cb93a386Sopenharmony_ci    std::unique_ptr<SkCodec> partialCodec(SkCodec::MakeFromStream(
274cb93a386Sopenharmony_ci                                                      std::unique_ptr<SkStream>(haltingStream)));
275cb93a386Sopenharmony_ci    if (!partialCodec) {
276cb93a386Sopenharmony_ci        ERRORF(r, "Failed to create a partial codec from %s with %zu bytes out of %zu",
277cb93a386Sopenharmony_ci               path, frameByteCounts[0], file->size());
278cb93a386Sopenharmony_ci        return;
279cb93a386Sopenharmony_ci    }
280cb93a386Sopenharmony_ci
281cb93a386Sopenharmony_ci    SkASSERT(frameByteCounts.size() > frames.size());
282cb93a386Sopenharmony_ci    for (size_t i = 0; i < frames.size(); i++) {
283cb93a386Sopenharmony_ci        const size_t fullFrameBytes = frameByteCounts[i + 1];
284cb93a386Sopenharmony_ci        const size_t firstHalf = fullFrameBytes / 2;
285cb93a386Sopenharmony_ci        const size_t secondHalf = fullFrameBytes - firstHalf;
286cb93a386Sopenharmony_ci
287cb93a386Sopenharmony_ci        haltingStream->addNewData(firstHalf);
288cb93a386Sopenharmony_ci        auto frameInfo = partialCodec->getFrameInfo();
289cb93a386Sopenharmony_ci        REPORTER_ASSERT(r, frameInfo.size() == i + 1);
290cb93a386Sopenharmony_ci        REPORTER_ASSERT(r, !frameInfo[i].fFullyReceived);
291cb93a386Sopenharmony_ci
292cb93a386Sopenharmony_ci        SkBitmap frame;
293cb93a386Sopenharmony_ci        frame.allocPixels(info);
294cb93a386Sopenharmony_ci
295cb93a386Sopenharmony_ci        SkCodec::Options opts;
296cb93a386Sopenharmony_ci        opts.fFrameIndex = i;
297cb93a386Sopenharmony_ci        SkCodec::Result result = partialCodec->startIncrementalDecode(info,
298cb93a386Sopenharmony_ci                frame.getPixels(), frame.rowBytes(), &opts);
299cb93a386Sopenharmony_ci        if (result != SkCodec::kSuccess) {
300cb93a386Sopenharmony_ci            ERRORF(r, "Failed to start incremental decode for %s on frame %zu",
301cb93a386Sopenharmony_ci                   path, i);
302cb93a386Sopenharmony_ci            return;
303cb93a386Sopenharmony_ci        }
304cb93a386Sopenharmony_ci
305cb93a386Sopenharmony_ci        result = partialCodec->incrementalDecode();
306cb93a386Sopenharmony_ci        REPORTER_ASSERT(r, SkCodec::kIncompleteInput == result);
307cb93a386Sopenharmony_ci
308cb93a386Sopenharmony_ci        haltingStream->addNewData(secondHalf);
309cb93a386Sopenharmony_ci        result = partialCodec->incrementalDecode();
310cb93a386Sopenharmony_ci        REPORTER_ASSERT(r, SkCodec::kSuccess == result);
311cb93a386Sopenharmony_ci
312cb93a386Sopenharmony_ci        frameInfo = partialCodec->getFrameInfo();
313cb93a386Sopenharmony_ci        REPORTER_ASSERT(r, frameInfo.size() == i + 1);
314cb93a386Sopenharmony_ci        REPORTER_ASSERT(r, frameInfo[i].fFullyReceived);
315cb93a386Sopenharmony_ci        if (!compare_bitmaps(r, frames[i], frame)) {
316cb93a386Sopenharmony_ci            ERRORF(r, "\tfailure was on frame %zu", i);
317cb93a386Sopenharmony_ci            SkString name = SkStringPrintf("expected_%zu", i);
318cb93a386Sopenharmony_ci            write_bm(name.c_str(), frames[i]);
319cb93a386Sopenharmony_ci
320cb93a386Sopenharmony_ci            name = SkStringPrintf("actual_%zu", i);
321cb93a386Sopenharmony_ci            write_bm(name.c_str(), frame);
322cb93a386Sopenharmony_ci        }
323cb93a386Sopenharmony_ci    }
324cb93a386Sopenharmony_ci}
325cb93a386Sopenharmony_ci
326cb93a386Sopenharmony_ci// Test that calling getPixels when an incremental decode has been
327cb93a386Sopenharmony_ci// started (but not finished) makes the next call to incrementalDecode
328cb93a386Sopenharmony_ci// require a call to startIncrementalDecode.
329cb93a386Sopenharmony_cistatic void test_interleaved(skiatest::Reporter* r, const char* name) {
330cb93a386Sopenharmony_ci    sk_sp<SkData> file = GetResourceAsData(name);
331cb93a386Sopenharmony_ci    if (!file) {
332cb93a386Sopenharmony_ci        return;
333cb93a386Sopenharmony_ci    }
334cb93a386Sopenharmony_ci    const size_t halfSize = file->size() / 2;
335cb93a386Sopenharmony_ci    std::unique_ptr<SkCodec> partialCodec(SkCodec::MakeFromStream(
336cb93a386Sopenharmony_ci                                  std::make_unique<HaltingStream>(std::move(file), halfSize)));
337cb93a386Sopenharmony_ci    if (!partialCodec) {
338cb93a386Sopenharmony_ci        ERRORF(r, "Failed to create codec for %s", name);
339cb93a386Sopenharmony_ci        return;
340cb93a386Sopenharmony_ci    }
341cb93a386Sopenharmony_ci
342cb93a386Sopenharmony_ci    const SkImageInfo info = standardize_info(partialCodec.get());
343cb93a386Sopenharmony_ci    SkBitmap incremental;
344cb93a386Sopenharmony_ci    incremental.allocPixels(info);
345cb93a386Sopenharmony_ci
346cb93a386Sopenharmony_ci    const SkCodec::Result startResult = partialCodec->startIncrementalDecode(info,
347cb93a386Sopenharmony_ci            incremental.getPixels(), incremental.rowBytes());
348cb93a386Sopenharmony_ci    if (startResult != SkCodec::kSuccess) {
349cb93a386Sopenharmony_ci        ERRORF(r, "Failed to start incremental decode\n");
350cb93a386Sopenharmony_ci        return;
351cb93a386Sopenharmony_ci    }
352cb93a386Sopenharmony_ci
353cb93a386Sopenharmony_ci    SkCodec::Result result = partialCodec->incrementalDecode();
354cb93a386Sopenharmony_ci    REPORTER_ASSERT(r, result == SkCodec::kIncompleteInput);
355cb93a386Sopenharmony_ci
356cb93a386Sopenharmony_ci    SkBitmap full;
357cb93a386Sopenharmony_ci    full.allocPixels(info);
358cb93a386Sopenharmony_ci    result = partialCodec->getPixels(info, full.getPixels(), full.rowBytes());
359cb93a386Sopenharmony_ci    REPORTER_ASSERT(r, result == SkCodec::kIncompleteInput);
360cb93a386Sopenharmony_ci
361cb93a386Sopenharmony_ci    // Now incremental decode will fail
362cb93a386Sopenharmony_ci    result = partialCodec->incrementalDecode();
363cb93a386Sopenharmony_ci    REPORTER_ASSERT(r, result == SkCodec::kInvalidParameters);
364cb93a386Sopenharmony_ci}
365cb93a386Sopenharmony_ci
366cb93a386Sopenharmony_ciDEF_TEST(Codec_rewind, r) {
367cb93a386Sopenharmony_ci    test_interleaved(r, "images/plane.png");
368cb93a386Sopenharmony_ci    test_interleaved(r, "images/plane_interlaced.png");
369cb93a386Sopenharmony_ci    test_interleaved(r, "images/box.gif");
370cb93a386Sopenharmony_ci}
371cb93a386Sopenharmony_ci
372cb93a386Sopenharmony_ci// Modified version of the giflib logo, from
373cb93a386Sopenharmony_ci// http://giflib.sourceforge.net/whatsinagif/bits_and_bytes.html
374cb93a386Sopenharmony_ci// The global color map has been replaced with a local color map.
375cb93a386Sopenharmony_cistatic unsigned char gNoGlobalColorMap[] = {
376cb93a386Sopenharmony_ci  // Header
377cb93a386Sopenharmony_ci  0x47, 0x49, 0x46, 0x38, 0x39, 0x61,
378cb93a386Sopenharmony_ci
379cb93a386Sopenharmony_ci  // Logical screen descriptor
380cb93a386Sopenharmony_ci  0x0A, 0x00, 0x0A, 0x00, 0x11, 0x00, 0x00,
381cb93a386Sopenharmony_ci
382cb93a386Sopenharmony_ci  // Image descriptor
383cb93a386Sopenharmony_ci  0x2C, 0x00, 0x00, 0x00, 0x00, 0x0A, 0x00, 0x0A, 0x00, 0x81,
384cb93a386Sopenharmony_ci
385cb93a386Sopenharmony_ci  // Local color table
386cb93a386Sopenharmony_ci  0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x00, 0x00, 0x00, 0xFF, 0x00, 0x00, 0x00,
387cb93a386Sopenharmony_ci
388cb93a386Sopenharmony_ci  // Image data
389cb93a386Sopenharmony_ci  0x02, 0x16, 0x8C, 0x2D, 0x99, 0x87, 0x2A, 0x1C, 0xDC, 0x33, 0xA0, 0x02, 0x75,
390cb93a386Sopenharmony_ci  0xEC, 0x95, 0xFA, 0xA8, 0xDE, 0x60, 0x8C, 0x04, 0x91, 0x4C, 0x01, 0x00,
391cb93a386Sopenharmony_ci
392cb93a386Sopenharmony_ci  // Trailer
393cb93a386Sopenharmony_ci  0x3B,
394cb93a386Sopenharmony_ci};
395cb93a386Sopenharmony_ci
396cb93a386Sopenharmony_ci// Test that a gif file truncated before its local color map behaves as expected.
397cb93a386Sopenharmony_ciDEF_TEST(Codec_GifPreMap, r) {
398cb93a386Sopenharmony_ci    sk_sp<SkData> data = SkData::MakeWithoutCopy(gNoGlobalColorMap, sizeof(gNoGlobalColorMap));
399cb93a386Sopenharmony_ci    std::unique_ptr<SkCodec> codec(SkCodec::MakeFromData(data));
400cb93a386Sopenharmony_ci    if (!codec) {
401cb93a386Sopenharmony_ci        ERRORF(r, "failed to create codec");
402cb93a386Sopenharmony_ci        return;
403cb93a386Sopenharmony_ci    }
404cb93a386Sopenharmony_ci
405cb93a386Sopenharmony_ci    SkBitmap truth;
406cb93a386Sopenharmony_ci    auto info = standardize_info(codec.get());
407cb93a386Sopenharmony_ci    truth.allocPixels(info);
408cb93a386Sopenharmony_ci
409cb93a386Sopenharmony_ci    auto result = codec->getPixels(info, truth.getPixels(), truth.rowBytes());
410cb93a386Sopenharmony_ci    REPORTER_ASSERT(r, result == SkCodec::kSuccess);
411cb93a386Sopenharmony_ci
412cb93a386Sopenharmony_ci    // Truncate to 23 bytes, just before the color map. This should fail to decode.
413cb93a386Sopenharmony_ci    //
414cb93a386Sopenharmony_ci    // See also Codec_GifTruncated2 in GifTest.cpp for this magic 23.
415cb93a386Sopenharmony_ci    codec = SkCodec::MakeFromData(SkData::MakeWithoutCopy(gNoGlobalColorMap, 23));
416cb93a386Sopenharmony_ci    REPORTER_ASSERT(r, codec);
417cb93a386Sopenharmony_ci    if (codec) {
418cb93a386Sopenharmony_ci        SkBitmap bm;
419cb93a386Sopenharmony_ci        bm.allocPixels(info);
420cb93a386Sopenharmony_ci        result = codec->getPixels(info, bm.getPixels(), bm.rowBytes());
421cb93a386Sopenharmony_ci
422cb93a386Sopenharmony_ci        // See the comments in Codec_GifTruncated2.
423cb93a386Sopenharmony_ci#ifdef SK_HAS_WUFFS_LIBRARY
424cb93a386Sopenharmony_ci        REPORTER_ASSERT(r, result == SkCodec::kIncompleteInput);
425cb93a386Sopenharmony_ci#else
426cb93a386Sopenharmony_ci        REPORTER_ASSERT(r, result == SkCodec::kInvalidInput);
427cb93a386Sopenharmony_ci#endif
428cb93a386Sopenharmony_ci    }
429cb93a386Sopenharmony_ci
430cb93a386Sopenharmony_ci    // Again, truncate to 23 bytes, this time for an incremental decode. We
431cb93a386Sopenharmony_ci    // cannot start an incremental decode until we have more data. If we did,
432cb93a386Sopenharmony_ci    // we would be using the wrong color table.
433cb93a386Sopenharmony_ci    HaltingStream* stream = new HaltingStream(data, 23);
434cb93a386Sopenharmony_ci    codec = SkCodec::MakeFromStream(std::unique_ptr<SkStream>(stream));
435cb93a386Sopenharmony_ci    REPORTER_ASSERT(r, codec);
436cb93a386Sopenharmony_ci    if (codec) {
437cb93a386Sopenharmony_ci        SkBitmap bm;
438cb93a386Sopenharmony_ci        bm.allocPixels(info);
439cb93a386Sopenharmony_ci        result = codec->startIncrementalDecode(info, bm.getPixels(), bm.rowBytes());
440cb93a386Sopenharmony_ci
441cb93a386Sopenharmony_ci        // See the comments in Codec_GifTruncated2.
442cb93a386Sopenharmony_ci#ifdef SK_HAS_WUFFS_LIBRARY
443cb93a386Sopenharmony_ci        REPORTER_ASSERT(r, result == SkCodec::kSuccess);
444cb93a386Sopenharmony_ci
445cb93a386Sopenharmony_ci        // Note that this is incrementalDecode, not startIncrementalDecode.
446cb93a386Sopenharmony_ci        result = codec->incrementalDecode();
447cb93a386Sopenharmony_ci        REPORTER_ASSERT(r, result == SkCodec::kIncompleteInput);
448cb93a386Sopenharmony_ci
449cb93a386Sopenharmony_ci        stream->addNewData(data->size());
450cb93a386Sopenharmony_ci#else
451cb93a386Sopenharmony_ci        REPORTER_ASSERT(r, result == SkCodec::kIncompleteInput);
452cb93a386Sopenharmony_ci
453cb93a386Sopenharmony_ci        // Note that this is startIncrementalDecode, not incrementalDecode.
454cb93a386Sopenharmony_ci        stream->addNewData(data->size());
455cb93a386Sopenharmony_ci        result = codec->startIncrementalDecode(info, bm.getPixels(), bm.rowBytes());
456cb93a386Sopenharmony_ci        REPORTER_ASSERT(r, result == SkCodec::kSuccess);
457cb93a386Sopenharmony_ci#endif
458cb93a386Sopenharmony_ci
459cb93a386Sopenharmony_ci        result = codec->incrementalDecode();
460cb93a386Sopenharmony_ci        REPORTER_ASSERT(r, result == SkCodec::kSuccess);
461cb93a386Sopenharmony_ci        compare_bitmaps(r, truth, bm);
462cb93a386Sopenharmony_ci    }
463cb93a386Sopenharmony_ci}
464cb93a386Sopenharmony_ci
465cb93a386Sopenharmony_ciDEF_TEST(Codec_emptyIDAT, r) {
466cb93a386Sopenharmony_ci    const char* name = "images/baby_tux.png";
467cb93a386Sopenharmony_ci    sk_sp<SkData> file = GetResourceAsData(name);
468cb93a386Sopenharmony_ci    if (!file) {
469cb93a386Sopenharmony_ci        return;
470cb93a386Sopenharmony_ci    }
471cb93a386Sopenharmony_ci
472cb93a386Sopenharmony_ci    // Truncate to the beginning of the IDAT, immediately after the IDAT tag.
473cb93a386Sopenharmony_ci    file = SkData::MakeSubset(file.get(), 0, 80);
474cb93a386Sopenharmony_ci
475cb93a386Sopenharmony_ci    std::unique_ptr<SkCodec> codec(SkCodec::MakeFromData(std::move(file)));
476cb93a386Sopenharmony_ci    if (!codec) {
477cb93a386Sopenharmony_ci        ERRORF(r, "Failed to create a codec for %s", name);
478cb93a386Sopenharmony_ci        return;
479cb93a386Sopenharmony_ci    }
480cb93a386Sopenharmony_ci
481cb93a386Sopenharmony_ci    SkBitmap bm;
482cb93a386Sopenharmony_ci    const auto info = standardize_info(codec.get());
483cb93a386Sopenharmony_ci    bm.allocPixels(info);
484cb93a386Sopenharmony_ci
485cb93a386Sopenharmony_ci    const auto result = codec->getPixels(info, bm.getPixels(), bm.rowBytes());
486cb93a386Sopenharmony_ci    REPORTER_ASSERT(r, SkCodec::kIncompleteInput == result);
487cb93a386Sopenharmony_ci}
488cb93a386Sopenharmony_ci
489cb93a386Sopenharmony_ciDEF_TEST(Codec_incomplete, r) {
490cb93a386Sopenharmony_ci    for (const char* name : { "images/baby_tux.png",
491cb93a386Sopenharmony_ci                              "images/baby_tux.webp",
492cb93a386Sopenharmony_ci                              "images/CMYK.jpg",
493cb93a386Sopenharmony_ci                              "images/color_wheel.gif",
494cb93a386Sopenharmony_ci                              "images/google_chrome.ico",
495cb93a386Sopenharmony_ci                              "images/rle.bmp",
496cb93a386Sopenharmony_ci                              "images/mandrill.wbmp",
497cb93a386Sopenharmony_ci                              }) {
498cb93a386Sopenharmony_ci        sk_sp<SkData> file = GetResourceAsData(name);
499cb93a386Sopenharmony_ci        if (!file) {
500cb93a386Sopenharmony_ci            continue;
501cb93a386Sopenharmony_ci        }
502cb93a386Sopenharmony_ci
503cb93a386Sopenharmony_ci        for (size_t len = 14; len <= file->size(); len += 5) {
504cb93a386Sopenharmony_ci            SkCodec::Result result;
505cb93a386Sopenharmony_ci            std::unique_ptr<SkCodec> codec(SkCodec::MakeFromStream(
506cb93a386Sopenharmony_ci                                   std::make_unique<SkMemoryStream>(file->data(), len), &result));
507cb93a386Sopenharmony_ci            if (codec) {
508cb93a386Sopenharmony_ci                if (result != SkCodec::kSuccess) {
509cb93a386Sopenharmony_ci                    ERRORF(r, "Created an SkCodec for %s with %zu bytes, but "
510cb93a386Sopenharmony_ci                              "reported an error %i", name, len, (int)result);
511cb93a386Sopenharmony_ci                }
512cb93a386Sopenharmony_ci                break;
513cb93a386Sopenharmony_ci            }
514cb93a386Sopenharmony_ci
515cb93a386Sopenharmony_ci            if (SkCodec::kIncompleteInput != result) {
516cb93a386Sopenharmony_ci                ERRORF(r, "Reported error %i for %s with %zu bytes",
517cb93a386Sopenharmony_ci                       (int)result, name, len);
518cb93a386Sopenharmony_ci                break;
519cb93a386Sopenharmony_ci            }
520cb93a386Sopenharmony_ci        }
521cb93a386Sopenharmony_ci    }
522cb93a386Sopenharmony_ci}
523