mirror of
https://github.com/flutter/flutter.git
synced 2026-02-20 02:29:02 +08:00
This CL generated by |sed -i '/sky\/engine\/config.h/d'| and a manual sweep to catch some oddballs. TBR=eseidel@chromium.org Review URL: https://codereview.chromium.org/1206763002.
501 lines
19 KiB
C++
501 lines
19 KiB
C++
/*
|
|
* Copyright (C) 2013 Google Inc. All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions are
|
|
* met:
|
|
*
|
|
* * Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* * Redistributions in binary form must reproduce the above
|
|
* copyright notice, this list of conditions and the following disclaimer
|
|
* in the documentation and/or other materials provided with the
|
|
* distribution.
|
|
* * Neither the name of Google Inc. nor the names of its
|
|
* contributors may be used to endorse or promote products derived from
|
|
* this software without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
|
|
#include "platform/image-decoders/gif/GIFImageDecoder.h"
|
|
|
|
#include <gtest/gtest.h>
|
|
#include "sky/engine/platform/SharedBuffer.h"
|
|
#include "sky/engine/public/platform/Platform.h"
|
|
#include "sky/engine/public/platform/WebData.h"
|
|
#include "sky/engine/public/platform/WebSize.h"
|
|
#include "sky/engine/public/platform/WebUnitTestSupport.h"
|
|
#include "sky/engine/wtf/OwnPtr.h"
|
|
#include "sky/engine/wtf/PassOwnPtr.h"
|
|
#include "sky/engine/wtf/StringHasher.h"
|
|
#include "sky/engine/wtf/Vector.h"
|
|
|
|
using namespace blink;
|
|
|
|
namespace {
|
|
|
|
PassRefPtr<SharedBuffer> readFile(const char* fileName)
|
|
{
|
|
String filePath = Platform::current()->unitTestSupport()->webKitRootDir();
|
|
filePath.append(fileName);
|
|
|
|
return Platform::current()->unitTestSupport()->readFromFile(filePath);
|
|
}
|
|
|
|
PassOwnPtr<GIFImageDecoder> createDecoder()
|
|
{
|
|
return adoptPtr(new GIFImageDecoder(ImageSource::AlphaNotPremultiplied, ImageSource::GammaAndColorProfileApplied, ImageDecoder::noDecodedImageByteLimit));
|
|
}
|
|
|
|
unsigned hashSkBitmap(const SkBitmap& bitmap)
|
|
{
|
|
return StringHasher::hashMemory(bitmap.getPixels(), bitmap.getSize());
|
|
}
|
|
|
|
void createDecodingBaseline(SharedBuffer* data, Vector<unsigned>* baselineHashes)
|
|
{
|
|
OwnPtr<GIFImageDecoder> decoder = createDecoder();
|
|
decoder->setData(data, true);
|
|
size_t frameCount = decoder->frameCount();
|
|
for (size_t i = 0; i < frameCount; ++i) {
|
|
ImageFrame* frame = decoder->frameBufferAtIndex(i);
|
|
baselineHashes->append(hashSkBitmap(frame->getSkBitmap()));
|
|
}
|
|
}
|
|
|
|
void testRandomFrameDecode(const char* gifFile)
|
|
{
|
|
SCOPED_TRACE(gifFile);
|
|
|
|
RefPtr<SharedBuffer> fullData = readFile(gifFile);
|
|
ASSERT_TRUE(fullData.get());
|
|
Vector<unsigned> baselineHashes;
|
|
createDecodingBaseline(fullData.get(), &baselineHashes);
|
|
size_t frameCount = baselineHashes.size();
|
|
|
|
// Random decoding should get the same results as sequential decoding.
|
|
OwnPtr<GIFImageDecoder> decoder = createDecoder();
|
|
decoder->setData(fullData.get(), true);
|
|
const size_t skippingStep = 5;
|
|
for (size_t i = 0; i < skippingStep; ++i) {
|
|
for (size_t j = i; j < frameCount; j += skippingStep) {
|
|
SCOPED_TRACE(testing::Message() << "Random i:" << i << " j:" << j);
|
|
ImageFrame* frame = decoder->frameBufferAtIndex(j);
|
|
EXPECT_EQ(baselineHashes[j], hashSkBitmap(frame->getSkBitmap()));
|
|
}
|
|
}
|
|
|
|
// Decoding in reverse order.
|
|
decoder = createDecoder();
|
|
decoder->setData(fullData.get(), true);
|
|
for (size_t i = frameCount; i; --i) {
|
|
SCOPED_TRACE(testing::Message() << "Reverse i:" << i);
|
|
ImageFrame* frame = decoder->frameBufferAtIndex(i - 1);
|
|
EXPECT_EQ(baselineHashes[i - 1], hashSkBitmap(frame->getSkBitmap()));
|
|
}
|
|
}
|
|
|
|
void testRandomDecodeAfterClearFrameBufferCache(const char* gifFile)
|
|
{
|
|
SCOPED_TRACE(gifFile);
|
|
|
|
RefPtr<SharedBuffer> data = readFile(gifFile);
|
|
ASSERT_TRUE(data.get());
|
|
Vector<unsigned> baselineHashes;
|
|
createDecodingBaseline(data.get(), &baselineHashes);
|
|
size_t frameCount = baselineHashes.size();
|
|
|
|
OwnPtr<GIFImageDecoder> decoder = createDecoder();
|
|
decoder->setData(data.get(), true);
|
|
for (size_t clearExceptFrame = 0; clearExceptFrame < frameCount; ++clearExceptFrame) {
|
|
decoder->clearCacheExceptFrame(clearExceptFrame);
|
|
const size_t skippingStep = 5;
|
|
for (size_t i = 0; i < skippingStep; ++i) {
|
|
for (size_t j = 0; j < frameCount; j += skippingStep) {
|
|
SCOPED_TRACE(testing::Message() << "Random i:" << i << " j:" << j);
|
|
ImageFrame* frame = decoder->frameBufferAtIndex(j);
|
|
EXPECT_EQ(baselineHashes[j], hashSkBitmap(frame->getSkBitmap()));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
} // namespace
|
|
|
|
TEST(GIFImageDecoderTest, decodeTwoFrames)
|
|
{
|
|
OwnPtr<GIFImageDecoder> decoder = createDecoder();
|
|
|
|
RefPtr<SharedBuffer> data = readFile("/tests/fast/images/resources/animated.gif");
|
|
ASSERT_TRUE(data.get());
|
|
decoder->setData(data.get(), true);
|
|
EXPECT_EQ(cAnimationLoopOnce, decoder->repetitionCount());
|
|
|
|
ImageFrame* frame = decoder->frameBufferAtIndex(0);
|
|
uint32_t generationID0 = frame->getSkBitmap().getGenerationID();
|
|
EXPECT_EQ(ImageFrame::FrameComplete, frame->status());
|
|
EXPECT_EQ(16, frame->getSkBitmap().width());
|
|
EXPECT_EQ(16, frame->getSkBitmap().height());
|
|
|
|
frame = decoder->frameBufferAtIndex(1);
|
|
uint32_t generationID1 = frame->getSkBitmap().getGenerationID();
|
|
EXPECT_EQ(ImageFrame::FrameComplete, frame->status());
|
|
EXPECT_EQ(16, frame->getSkBitmap().width());
|
|
EXPECT_EQ(16, frame->getSkBitmap().height());
|
|
EXPECT_TRUE(generationID0 != generationID1);
|
|
|
|
EXPECT_EQ(2u, decoder->frameCount());
|
|
EXPECT_EQ(cAnimationLoopInfinite, decoder->repetitionCount());
|
|
}
|
|
|
|
TEST(GIFImageDecoderTest, parseAndDecode)
|
|
{
|
|
OwnPtr<GIFImageDecoder> decoder = createDecoder();
|
|
|
|
RefPtr<SharedBuffer> data = readFile("/tests/fast/images/resources/animated.gif");
|
|
ASSERT_TRUE(data.get());
|
|
decoder->setData(data.get(), true);
|
|
EXPECT_EQ(cAnimationLoopOnce, decoder->repetitionCount());
|
|
|
|
// This call will parse the entire file.
|
|
EXPECT_EQ(2u, decoder->frameCount());
|
|
|
|
ImageFrame* frame = decoder->frameBufferAtIndex(0);
|
|
EXPECT_EQ(ImageFrame::FrameComplete, frame->status());
|
|
EXPECT_EQ(16, frame->getSkBitmap().width());
|
|
EXPECT_EQ(16, frame->getSkBitmap().height());
|
|
|
|
frame = decoder->frameBufferAtIndex(1);
|
|
EXPECT_EQ(ImageFrame::FrameComplete, frame->status());
|
|
EXPECT_EQ(16, frame->getSkBitmap().width());
|
|
EXPECT_EQ(16, frame->getSkBitmap().height());
|
|
EXPECT_EQ(cAnimationLoopInfinite, decoder->repetitionCount());
|
|
}
|
|
|
|
TEST(GIFImageDecoderTest, parseByteByByte)
|
|
{
|
|
OwnPtr<GIFImageDecoder> decoder = createDecoder();
|
|
|
|
RefPtr<SharedBuffer> data = readFile("/tests/fast/images/resources/animated.gif");
|
|
ASSERT_TRUE(data.get());
|
|
|
|
size_t frameCount = 0;
|
|
|
|
// Pass data to decoder byte by byte.
|
|
for (size_t length = 1; length <= data->size(); ++length) {
|
|
RefPtr<SharedBuffer> tempData = SharedBuffer::create(data->data(), length);
|
|
decoder->setData(tempData.get(), length == data->size());
|
|
|
|
EXPECT_LE(frameCount, decoder->frameCount());
|
|
frameCount = decoder->frameCount();
|
|
}
|
|
|
|
EXPECT_EQ(2u, decoder->frameCount());
|
|
|
|
decoder->frameBufferAtIndex(0);
|
|
decoder->frameBufferAtIndex(1);
|
|
EXPECT_EQ(cAnimationLoopInfinite, decoder->repetitionCount());
|
|
}
|
|
|
|
TEST(GIFImageDecoderTest, parseAndDecodeByteByByte)
|
|
{
|
|
OwnPtr<GIFImageDecoder> decoder = createDecoder();
|
|
|
|
RefPtr<SharedBuffer> data = readFile("/tests/fast/images/resources/animated-gif-with-offsets.gif");
|
|
ASSERT_TRUE(data.get());
|
|
|
|
size_t frameCount = 0;
|
|
size_t framesDecoded = 0;
|
|
|
|
// Pass data to decoder byte by byte.
|
|
for (size_t length = 1; length <= data->size(); ++length) {
|
|
RefPtr<SharedBuffer> tempData = SharedBuffer::create(data->data(), length);
|
|
decoder->setData(tempData.get(), length == data->size());
|
|
|
|
EXPECT_LE(frameCount, decoder->frameCount());
|
|
frameCount = decoder->frameCount();
|
|
|
|
ImageFrame* frame = decoder->frameBufferAtIndex(frameCount - 1);
|
|
if (frame && frame->status() == ImageFrame::FrameComplete && framesDecoded < frameCount)
|
|
++framesDecoded;
|
|
}
|
|
|
|
EXPECT_EQ(5u, decoder->frameCount());
|
|
EXPECT_EQ(5u, framesDecoded);
|
|
EXPECT_EQ(cAnimationLoopInfinite, decoder->repetitionCount());
|
|
}
|
|
|
|
TEST(GIFImageDecoderTest, brokenSecondFrame)
|
|
{
|
|
OwnPtr<GIFImageDecoder> decoder = createDecoder();
|
|
|
|
RefPtr<SharedBuffer> data = readFile("/engine/web/tests/data/broken.gif");
|
|
ASSERT_TRUE(data.get());
|
|
decoder->setData(data.get(), true);
|
|
|
|
// One frame is detected but cannot be decoded.
|
|
EXPECT_EQ(1u, decoder->frameCount());
|
|
ImageFrame* frame = decoder->frameBufferAtIndex(1);
|
|
EXPECT_FALSE(frame);
|
|
}
|
|
|
|
TEST(GIFImageDecoderTest, progressiveDecode)
|
|
{
|
|
RefPtr<SharedBuffer> fullData = readFile("/engine/web/tests/data/radient.gif");
|
|
ASSERT_TRUE(fullData.get());
|
|
const size_t fullLength = fullData->size();
|
|
|
|
OwnPtr<GIFImageDecoder> decoder;
|
|
ImageFrame* frame;
|
|
|
|
Vector<unsigned> truncatedHashes;
|
|
Vector<unsigned> progressiveHashes;
|
|
|
|
// Compute hashes when the file is truncated.
|
|
const size_t increment = 1;
|
|
for (size_t i = 1; i <= fullLength; i += increment) {
|
|
decoder = createDecoder();
|
|
RefPtr<SharedBuffer> data = SharedBuffer::create(fullData->data(), i);
|
|
decoder->setData(data.get(), i == fullLength);
|
|
frame = decoder->frameBufferAtIndex(0);
|
|
if (!frame) {
|
|
truncatedHashes.append(0);
|
|
continue;
|
|
}
|
|
truncatedHashes.append(hashSkBitmap(frame->getSkBitmap()));
|
|
}
|
|
|
|
// Compute hashes when the file is progressively decoded.
|
|
decoder = createDecoder();
|
|
EXPECT_EQ(cAnimationLoopOnce, decoder->repetitionCount());
|
|
for (size_t i = 1; i <= fullLength; i += increment) {
|
|
RefPtr<SharedBuffer> data = SharedBuffer::create(fullData->data(), i);
|
|
decoder->setData(data.get(), i == fullLength);
|
|
frame = decoder->frameBufferAtIndex(0);
|
|
if (!frame) {
|
|
progressiveHashes.append(0);
|
|
continue;
|
|
}
|
|
progressiveHashes.append(hashSkBitmap(frame->getSkBitmap()));
|
|
}
|
|
EXPECT_EQ(cAnimationNone, decoder->repetitionCount());
|
|
|
|
bool match = true;
|
|
for (size_t i = 0; i < truncatedHashes.size(); ++i) {
|
|
if (truncatedHashes[i] != progressiveHashes[i]) {
|
|
match = false;
|
|
break;
|
|
}
|
|
}
|
|
EXPECT_TRUE(match);
|
|
}
|
|
|
|
TEST(GIFImageDecoderTest, allDataReceivedTruncation)
|
|
{
|
|
OwnPtr<GIFImageDecoder> decoder = createDecoder();
|
|
|
|
RefPtr<SharedBuffer> data = readFile("/tests/fast/images/resources/animated.gif");
|
|
ASSERT_TRUE(data.get());
|
|
|
|
ASSERT_GE(data->size(), 10u);
|
|
RefPtr<SharedBuffer> tempData = SharedBuffer::create(data->data(), data->size() - 10);
|
|
decoder->setData(tempData.get(), true);
|
|
|
|
EXPECT_EQ(2u, decoder->frameCount());
|
|
EXPECT_FALSE(decoder->failed());
|
|
|
|
decoder->frameBufferAtIndex(0);
|
|
EXPECT_FALSE(decoder->failed());
|
|
decoder->frameBufferAtIndex(1);
|
|
EXPECT_TRUE(decoder->failed());
|
|
}
|
|
|
|
TEST(GIFImageDecoderTest, frameIsComplete)
|
|
{
|
|
OwnPtr<GIFImageDecoder> decoder = createDecoder();
|
|
|
|
RefPtr<SharedBuffer> data = readFile("/tests/fast/images/resources/animated.gif");
|
|
ASSERT_TRUE(data.get());
|
|
decoder->setData(data.get(), true);
|
|
|
|
EXPECT_EQ(2u, decoder->frameCount());
|
|
EXPECT_FALSE(decoder->failed());
|
|
EXPECT_TRUE(decoder->frameIsCompleteAtIndex(0));
|
|
EXPECT_TRUE(decoder->frameIsCompleteAtIndex(1));
|
|
EXPECT_EQ(cAnimationLoopInfinite, decoder->repetitionCount());
|
|
}
|
|
|
|
TEST(GIFImageDecoderTest, frameIsCompleteLoading)
|
|
{
|
|
OwnPtr<GIFImageDecoder> decoder = createDecoder();
|
|
|
|
RefPtr<SharedBuffer> data = readFile("/tests/fast/images/resources/animated.gif");
|
|
ASSERT_TRUE(data.get());
|
|
|
|
ASSERT_GE(data->size(), 10u);
|
|
RefPtr<SharedBuffer> tempData = SharedBuffer::create(data->data(), data->size() - 10);
|
|
decoder->setData(tempData.get(), false);
|
|
|
|
EXPECT_EQ(2u, decoder->frameCount());
|
|
EXPECT_FALSE(decoder->failed());
|
|
EXPECT_TRUE(decoder->frameIsCompleteAtIndex(0));
|
|
EXPECT_FALSE(decoder->frameIsCompleteAtIndex(1));
|
|
|
|
decoder->setData(data.get(), true);
|
|
EXPECT_EQ(2u, decoder->frameCount());
|
|
EXPECT_TRUE(decoder->frameIsCompleteAtIndex(0));
|
|
EXPECT_TRUE(decoder->frameIsCompleteAtIndex(1));
|
|
}
|
|
|
|
TEST(GIFImageDecoderTest, badTerminator)
|
|
{
|
|
RefPtr<SharedBuffer> referenceData = readFile("/engine/web/tests/data/radient.gif");
|
|
RefPtr<SharedBuffer> testData = readFile("/engine/web/tests/data/radient-bad-terminator.gif");
|
|
ASSERT_TRUE(referenceData.get());
|
|
ASSERT_TRUE(testData.get());
|
|
|
|
OwnPtr<GIFImageDecoder> referenceDecoder(createDecoder());
|
|
referenceDecoder->setData(referenceData.get(), true);
|
|
EXPECT_EQ(1u, referenceDecoder->frameCount());
|
|
ImageFrame* referenceFrame = referenceDecoder->frameBufferAtIndex(0);
|
|
ASSERT(referenceFrame);
|
|
|
|
OwnPtr<GIFImageDecoder> testDecoder(createDecoder());
|
|
testDecoder->setData(testData.get(), true);
|
|
EXPECT_EQ(1u, testDecoder->frameCount());
|
|
ImageFrame* testFrame = testDecoder->frameBufferAtIndex(0);
|
|
ASSERT(testFrame);
|
|
|
|
EXPECT_EQ(hashSkBitmap(referenceFrame->getSkBitmap()), hashSkBitmap(testFrame->getSkBitmap()));
|
|
}
|
|
|
|
TEST(GIFImageDecoderTest, updateRequiredPreviousFrameAfterFirstDecode)
|
|
{
|
|
OwnPtr<GIFImageDecoder> decoder = createDecoder();
|
|
|
|
RefPtr<SharedBuffer> fullData = readFile("/tests/fast/images/resources/animated-10color.gif");
|
|
ASSERT_TRUE(fullData.get());
|
|
|
|
// Give it data that is enough to parse but not decode in order to check the status
|
|
// of requiredPreviousFrameIndex before decoding.
|
|
size_t partialSize = 1;
|
|
do {
|
|
RefPtr<SharedBuffer> data = SharedBuffer::create(fullData->data(), partialSize);
|
|
decoder->setData(data.get(), false);
|
|
++partialSize;
|
|
} while (!decoder->frameCount() || decoder->frameBufferAtIndex(0)->status() == ImageFrame::FrameEmpty);
|
|
|
|
EXPECT_EQ(kNotFound, decoder->frameBufferAtIndex(0)->requiredPreviousFrameIndex());
|
|
unsigned frameCount = decoder->frameCount();
|
|
for (size_t i = 1; i < frameCount; ++i)
|
|
EXPECT_EQ(i - 1, decoder->frameBufferAtIndex(i)->requiredPreviousFrameIndex());
|
|
|
|
decoder->setData(fullData.get(), true);
|
|
for (size_t i = 0; i < frameCount; ++i)
|
|
EXPECT_EQ(kNotFound, decoder->frameBufferAtIndex(i)->requiredPreviousFrameIndex());
|
|
}
|
|
|
|
TEST(GIFImageDecoderTest, randomFrameDecode)
|
|
{
|
|
// Single frame image.
|
|
testRandomFrameDecode("/engine/web/tests/data/radient.gif");
|
|
// Multiple frame images.
|
|
testRandomFrameDecode("/tests/fast/images/resources/animated-gif-with-offsets.gif");
|
|
testRandomFrameDecode("/tests/fast/images/resources/animated-10color.gif");
|
|
}
|
|
|
|
TEST(GIFImageDecoderTest, randomDecodeAfterClearFrameBufferCache)
|
|
{
|
|
// Single frame image.
|
|
testRandomDecodeAfterClearFrameBufferCache("/engine/web/tests/data/radient.gif");
|
|
// Multiple frame images.
|
|
testRandomDecodeAfterClearFrameBufferCache("/tests/fast/images/resources/animated-gif-with-offsets.gif");
|
|
testRandomDecodeAfterClearFrameBufferCache("/tests/fast/images/resources/animated-10color.gif");
|
|
}
|
|
|
|
TEST(GIFImageDecoderTest, resumePartialDecodeAfterClearFrameBufferCache)
|
|
{
|
|
RefPtr<SharedBuffer> fullData = readFile("/tests/fast/images/resources/animated-10color.gif");
|
|
ASSERT_TRUE(fullData.get());
|
|
Vector<unsigned> baselineHashes;
|
|
createDecodingBaseline(fullData.get(), &baselineHashes);
|
|
size_t frameCount = baselineHashes.size();
|
|
|
|
OwnPtr<GIFImageDecoder> decoder = createDecoder();
|
|
|
|
// Let frame 0 be partially decoded.
|
|
size_t partialSize = 1;
|
|
do {
|
|
RefPtr<SharedBuffer> data = SharedBuffer::create(fullData->data(), partialSize);
|
|
decoder->setData(data.get(), false);
|
|
++partialSize;
|
|
} while (!decoder->frameCount() || decoder->frameBufferAtIndex(0)->status() == ImageFrame::FrameEmpty);
|
|
|
|
// Skip to the last frame and clear.
|
|
decoder->setData(fullData.get(), true);
|
|
EXPECT_EQ(frameCount, decoder->frameCount());
|
|
ImageFrame* lastFrame = decoder->frameBufferAtIndex(frameCount - 1);
|
|
EXPECT_EQ(baselineHashes[frameCount - 1], hashSkBitmap(lastFrame->getSkBitmap()));
|
|
decoder->clearCacheExceptFrame(kNotFound);
|
|
|
|
// Resume decoding of the first frame.
|
|
ImageFrame* firstFrame = decoder->frameBufferAtIndex(0);
|
|
EXPECT_EQ(ImageFrame::FrameComplete, firstFrame->status());
|
|
EXPECT_EQ(baselineHashes[0], hashSkBitmap(firstFrame->getSkBitmap()));
|
|
}
|
|
|
|
// The first LZW codes in the image are invalid values that try to create a loop
|
|
// in the dictionary. Decoding should fail, but not infinitely loop or corrupt memory.
|
|
TEST(GIFImageDecoderTest, badInitialCode)
|
|
{
|
|
RefPtr<SharedBuffer> testData = readFile("/engine/platform/image-decoders/testing/bad-initial-code.gif");
|
|
ASSERT_TRUE(testData.get());
|
|
|
|
OwnPtr<GIFImageDecoder> testDecoder(createDecoder());
|
|
testDecoder->setData(testData.get(), true);
|
|
EXPECT_EQ(1u, testDecoder->frameCount());
|
|
ASSERT_TRUE(testDecoder->frameBufferAtIndex(0));
|
|
EXPECT_TRUE(testDecoder->failed());
|
|
}
|
|
|
|
// The image has an invalid LZW code that exceeds dictionary size. Decoding should fail.
|
|
TEST(GIFImageDecoderTest, badCode)
|
|
{
|
|
RefPtr<SharedBuffer> testData = readFile("/engine/platform/image-decoders/testing/bad-code.gif");
|
|
ASSERT_TRUE(testData.get());
|
|
|
|
OwnPtr<GIFImageDecoder> testDecoder(createDecoder());
|
|
testDecoder->setData(testData.get(), true);
|
|
EXPECT_EQ(1u, testDecoder->frameCount());
|
|
ASSERT_TRUE(testDecoder->frameBufferAtIndex(0));
|
|
EXPECT_TRUE(testDecoder->failed());
|
|
}
|
|
|
|
TEST(GIFImageDecoderTest, invalidDisposalMethod)
|
|
{
|
|
OwnPtr<GIFImageDecoder> decoder = createDecoder();
|
|
|
|
// The image has 2 frames, with disposal method 4 and 5, respectively.
|
|
RefPtr<SharedBuffer> data = readFile("/engine/web/tests/data/invalid-disposal-method.gif");
|
|
ASSERT_TRUE(data.get());
|
|
decoder->setData(data.get(), true);
|
|
|
|
EXPECT_EQ(2u, decoder->frameCount());
|
|
// Disposal method 4 is converted to ImageFrame::DisposeOverwritePrevious.
|
|
EXPECT_EQ(ImageFrame::DisposeOverwritePrevious, decoder->frameBufferAtIndex(0)->disposalMethod());
|
|
// Disposal method 5 is ignored.
|
|
EXPECT_EQ(ImageFrame::DisposeNotSpecified, decoder->frameBufferAtIndex(1)->disposalMethod());
|
|
}
|