Chromium Code Reviews| Index: Source/core/platform/image-decoders/gif/GIFImageDecoderTest.cpp |
| diff --git a/Source/WebKit/chromium/tests/GIFImageDecoderTest.cpp b/Source/core/platform/image-decoders/gif/GIFImageDecoderTest.cpp |
| similarity index 62% |
| rename from Source/WebKit/chromium/tests/GIFImageDecoderTest.cpp |
| rename to Source/core/platform/image-decoders/gif/GIFImageDecoderTest.cpp |
| index f7209045f9d3bb9384fa51596e290b18aa909ca2..ccc2bac6f08feb215a30a550fe5dc07beed89271 100644 |
| --- a/Source/WebKit/chromium/tests/GIFImageDecoderTest.cpp |
| +++ b/Source/core/platform/image-decoders/gif/GIFImageDecoderTest.cpp |
| @@ -42,7 +42,6 @@ |
| #include "wtf/PassOwnPtr.h" |
| #include "wtf/StringHasher.h" |
| #include "wtf/Vector.h" |
| - |
| #include <gtest/gtest.h> |
| using namespace WebCore; |
| @@ -79,9 +78,20 @@ static unsigned hashSkBitmap(const SkBitmap& bitmap) |
| return StringHasher::hashMemory(bitmap.getPixels(), bitmap.getSize()); |
| } |
| +static 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())); |
| + } |
| +} |
| + |
| TEST(GIFImageDecoderTest, decodeTwoFrames) |
| { |
| - OwnPtr<GIFImageDecoder> decoder(createDecoder()); |
| + OwnPtr<GIFImageDecoder> decoder = createDecoder(); |
| RefPtr<SharedBuffer> data = readFile("/LayoutTests/fast/images/resources/animated.gif"); |
| ASSERT_TRUE(data.get()); |
| @@ -104,7 +114,7 @@ TEST(GIFImageDecoderTest, decodeTwoFrames) |
| TEST(GIFImageDecoderTest, parseAndDecode) |
| { |
| - OwnPtr<GIFImageDecoder> decoder(createDecoder()); |
| + OwnPtr<GIFImageDecoder> decoder = createDecoder(); |
| RefPtr<SharedBuffer> data = readFile("/LayoutTests/fast/images/resources/animated.gif"); |
| ASSERT_TRUE(data.get()); |
| @@ -128,7 +138,7 @@ TEST(GIFImageDecoderTest, parseAndDecode) |
| TEST(GIFImageDecoderTest, parseByteByByte) |
| { |
| - OwnPtr<GIFImageDecoder> decoder(createDecoder()); |
| + OwnPtr<GIFImageDecoder> decoder = createDecoder(); |
| RefPtr<SharedBuffer> data = readFile("/LayoutTests/fast/images/resources/animated.gif"); |
| ASSERT_TRUE(data.get()); |
| @@ -136,7 +146,7 @@ TEST(GIFImageDecoderTest, parseByteByByte) |
| size_t frameCount = 0; |
| // Pass data to decoder byte by byte. |
| - for (unsigned length = 1; length <= data->size(); ++length) { |
| + for (size_t length = 1; length <= data->size(); ++length) { |
| RefPtr<SharedBuffer> tempData = SharedBuffer::create(data->data(), length); |
| decoder->setData(tempData.get(), length == data->size()); |
| @@ -153,7 +163,7 @@ TEST(GIFImageDecoderTest, parseByteByByte) |
| TEST(GIFImageDecoderTest, parseAndDecodeByteByByte) |
| { |
| - OwnPtr<GIFImageDecoder> decoder(createDecoder()); |
| + OwnPtr<GIFImageDecoder> decoder = createDecoder(); |
| RefPtr<SharedBuffer> data = readFile("/LayoutTests/fast/images/resources/animated-gif-with-offsets.gif"); |
| ASSERT_TRUE(data.get()); |
| @@ -162,7 +172,7 @@ TEST(GIFImageDecoderTest, parseAndDecodeByteByByte) |
| size_t framesDecoded = 0; |
| // Pass data to decoder byte by byte. |
| - for (unsigned length = 1; length <= data->size(); ++length) { |
| + for (size_t length = 1; length <= data->size(); ++length) { |
| RefPtr<SharedBuffer> tempData = SharedBuffer::create(data->data(), length); |
| decoder->setData(tempData.get(), length == data->size()); |
| @@ -181,7 +191,7 @@ TEST(GIFImageDecoderTest, parseAndDecodeByteByByte) |
| TEST(GIFImageDecoderTest, brokenSecondFrame) |
| { |
| - OwnPtr<GIFImageDecoder> decoder(createDecoder()); |
| + OwnPtr<GIFImageDecoder> decoder = createDecoder(); |
| RefPtr<SharedBuffer> data = readFile("/Source/WebKit/chromium/tests/data/broken.gif"); |
| ASSERT_TRUE(data.get()); |
| @@ -244,7 +254,7 @@ TEST(GIFImageDecoderTest, progressiveDecode) |
| TEST(GIFImageDecoderTest, allDataReceivedTruncation) |
| { |
| - OwnPtr<GIFImageDecoder> decoder(createDecoder()); |
| + OwnPtr<GIFImageDecoder> decoder = createDecoder(); |
| RefPtr<SharedBuffer> data = readFile("/LayoutTests/fast/images/resources/animated.gif"); |
| ASSERT_TRUE(data.get()); |
| @@ -264,7 +274,7 @@ TEST(GIFImageDecoderTest, allDataReceivedTruncation) |
| TEST(GIFImageDecoderTest, frameIsComplete) |
| { |
| - OwnPtr<GIFImageDecoder> decoder(createDecoder()); |
| + OwnPtr<GIFImageDecoder> decoder = createDecoder(); |
| RefPtr<SharedBuffer> data = readFile("/LayoutTests/fast/images/resources/animated.gif"); |
| ASSERT_TRUE(data.get()); |
| @@ -278,7 +288,7 @@ TEST(GIFImageDecoderTest, frameIsComplete) |
| TEST(GIFImageDecoderTest, frameIsCompleteLoading) |
| { |
| - OwnPtr<GIFImageDecoder> decoder(createDecoder()); |
| + OwnPtr<GIFImageDecoder> decoder = createDecoder(); |
| RefPtr<SharedBuffer> data = readFile("/LayoutTests/fast/images/resources/animated.gif"); |
| ASSERT_TRUE(data.get()); |
| @@ -320,6 +330,137 @@ TEST(GIFImageDecoderTest, badTerminator) |
| EXPECT_EQ(hashSkBitmap(referenceFrame->getSkBitmap()), hashSkBitmap(testFrame->getSkBitmap())); |
| } |
| +TEST(GIFImageDecoderTest, updateRequiredPreviousFrameAfterFirstDecode) |
| +{ |
| + OwnPtr<GIFImageDecoder> decoder = createDecoder(); |
| + |
| + RefPtr<SharedBuffer> fullData = readFile("/LayoutTests/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(notFound, 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(notFound, decoder->frameBufferAtIndex(i)->requiredPreviousFrameIndex()); |
| +} |
| + |
| +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())); |
| + } |
| +} |
| + |
| +TEST(GIFImageDecoderTest, randomFrameDecode) |
| +{ |
| + // Single frame image. |
| + testRandomFrameDecode("/Source/WebKit/chromium/tests/data/radient.gif"); |
| + // Multiple frame image. |
| + testRandomFrameDecode("/LayoutTests/fast/images/resources/animated-gif-with-offsets.gif"); |
| + testRandomFrameDecode("/LayoutTests/fast/images/resources/animated-10color.gif"); |
| +} |
| + |
| +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(); |
|
urvang (Google)
2013/06/06 18:33:51
Need to do "decoder->setData()" after this.
|
| + 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())); |
| + } |
| + } |
| + } |
| +} |
| + |
| +TEST(GIFImageDecoderTest, randomDecodeAfterClearFrameBufferCache) |
| +{ |
| + // Single frame image. |
| + testRandomFrameDecode("/Source/WebKit/chromium/tests/data/radient.gif"); |
|
urvang (Google)
2013/06/06 18:33:51
All these calls should be "testRandomDecodeAfterCl
|
| + // Multiple frame image. |
| + testRandomFrameDecode("/LayoutTests/fast/images/resources/animated-gif-with-offsets.gif"); |
| + testRandomFrameDecode("/LayoutTests/fast/images/resources/animated-10color.gif"); |
| +} |
| + |
| +TEST(GIFImageDecoderTest, resumePartialDecodeAfterClearFrameBufferCache) |
| +{ |
| + RefPtr<SharedBuffer> fullData = readFile("/LayoutTests/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(notFound); |
| + |
| + // Resume decoding of the first frame. |
| + ImageFrame* firstFrame = decoder->frameBufferAtIndex(0); |
| + EXPECT_EQ(ImageFrame::FrameComplete, firstFrame->status()); |
| + EXPECT_EQ(baselineHashes[0], hashSkBitmap(firstFrame->getSkBitmap())); |
| +} |
| + |
| #endif |
| } // namespace |