Index: third_party/WebKit/Source/platform/image-decoders/gif/GIFImageDecoderTest.cpp |
diff --git a/third_party/WebKit/Source/platform/image-decoders/gif/GIFImageDecoderTest.cpp b/third_party/WebKit/Source/platform/image-decoders/gif/GIFImageDecoderTest.cpp |
index a252a41ee1fde222e4a7d2ec4fa3f8a7a163b898..ec1a4ec4f3c15a03e0bc004d34453d1ccda7cbd7 100644 |
--- a/third_party/WebKit/Source/platform/image-decoders/gif/GIFImageDecoderTest.cpp |
+++ b/third_party/WebKit/Source/platform/image-decoders/gif/GIFImageDecoderTest.cpp |
@@ -31,6 +31,8 @@ |
#include "platform/image-decoders/gif/GIFImageDecoder.h" |
#include "platform/SharedBuffer.h" |
+#include "platform/graphics/ImageDecodingStore.h" |
+#include "platform/graphics/ImageFrameGenerator.h" |
#include "platform/image-decoders/ImageDecoderTestHelpers.h" |
#include "public/platform/WebData.h" |
#include "public/platform/WebSize.h" |
@@ -47,19 +49,66 @@ const char decodersTestingDir[] = "Source/platform/image-decoders/testing"; |
const char layoutTestResourcesDir[] = "LayoutTests/fast/images/resources"; |
const char webTestsDataDir[] = "Source/web/tests/data"; |
-PassOwnPtr<ImageDecoder> createDecoder() |
+#define LOOP_DECODERS_BEGIN \ |
+ DecoderCreator decoderCreators[2] = { &createDecoderN32, &createDecoderIndex8 }; \ |
+ for (DecoderCreator createDecoder : decoderCreators) { |
+ |
+#define LOOP_DECODERS_END \ |
+ } |
+ |
+PassOwnPtr<ImageDecoder> createDecoderN32() |
+{ |
+ return adoptPtr(new GIFImageDecoder(ImageDecoder::AlphaNotPremultiplied, ImageDecoder::GammaAndColorProfileApplied, ImageDecoder::noDecodedImageByteLimit, ImageFrame::N32)); |
+} |
+ |
+PassOwnPtr<ImageDecoder> createDecoderIndex8() |
+{ |
+ return adoptPtr(new GIFImageDecoder(ImageDecoder::AlphaNotPremultiplied, ImageDecoder::GammaAndColorProfileApplied, ImageDecoder::noDecodedImageByteLimit, ImageFrame::Index8)); |
+} |
+ |
+static bool areBitmapsColorsEqual(const SkBitmap& bitmap1, const SkBitmap& bitmap2) |
+{ |
+ int width = bitmap1.width(); |
+ int height = bitmap1.height(); |
+ int width2 = bitmap2.width(); |
+ int height2 = bitmap2.height(); |
+ if (bitmap1.alphaType() != bitmap2.alphaType()) |
+ return false; |
+ if (bitmap1.profileType() != bitmap2.profileType()) |
+ return false; |
+ if (width != width2 || height != height2) |
+ return false; |
+ |
+ for (int j = 0; j < height; ++j) { |
+ for (int i = 0; i < width; ++i) { |
+ uint32_t c1 = bitmap1.getColor(i, j); |
+ uint32_t c2 = bitmap2.getColor(i, j); |
+ if (c1 != c2) { |
+ return false; |
+ } |
+ } |
+ } |
+ |
+ return true; |
+} |
+ |
+static bool areFramesColorsEqual(const ImageFrame& b1, const ImageFrame& b2) |
{ |
- return adoptPtr(new GIFImageDecoder(ImageDecoder::AlphaNotPremultiplied, ImageDecoder::GammaAndColorProfileApplied, ImageDecoder::noDecodedImageByteLimit)); |
+ // The same premultiply alpha should be set. |
+ if (b1.premultiplyAlpha() != b2.premultiplyAlpha()) |
+ return false; |
+ return areBitmapsColorsEqual(b1.getSkBitmap(), b2.getSkBitmap()); |
} |
void testRandomFrameDecode(const char* dir, const char* gifFile) |
{ |
+ LOOP_DECODERS_BEGIN |
SCOPED_TRACE(gifFile); |
RefPtr<SharedBuffer> fullData = readFile(dir, gifFile); |
ASSERT_TRUE(fullData.get()); |
Vector<unsigned> baselineHashes; |
- createDecodingBaseline(&createDecoder, fullData.get(), &baselineHashes); |
+ createDecodingBaseline(createDecoder, fullData.get(), &baselineHashes); |
size_t frameCount = baselineHashes.size(); |
// Random decoding should get the same results as sequential decoding. |
@@ -82,19 +131,25 @@ void testRandomFrameDecode(const char* dir, const char* gifFile) |
ImageFrame* frame = decoder->frameBufferAtIndex(i - 1); |
EXPECT_EQ(baselineHashes[i - 1], hashBitmap(frame->getSkBitmap())); |
} |
+ LOOP_DECODERS_END |
} |
-void testRandomDecodeAfterClearFrameBufferCache(const char* dir, const char* gifFile) |
+void testRandomDecodeAfterClearFrameBufferCache(const char* dir, const char* gifFile, ImageFrame::ColorType targetType = ImageFrame::N32) |
{ |
SCOPED_TRACE(gifFile); |
RefPtr<SharedBuffer> data = readFile(dir, gifFile); |
ASSERT_TRUE(data.get()); |
Vector<unsigned> baselineHashes; |
- createDecodingBaseline(&createDecoder, data.get(), &baselineHashes); |
+ DecoderCreator decoderCreator = &createDecoderIndex8; |
+ if (targetType == ImageFrame::N32) { |
+ decoderCreator = &createDecoderN32; |
+ } |
+ createDecodingBaseline(decoderCreator, data.get(), &baselineHashes); |
+ |
size_t frameCount = baselineHashes.size(); |
- OwnPtr<ImageDecoder> decoder = createDecoder(); |
+ OwnPtr<ImageDecoder> decoder = decoderCreator(); |
decoder->setData(data.get(), true); |
for (size_t clearExceptFrame = 0; clearExceptFrame < frameCount; ++clearExceptFrame) { |
decoder->clearCacheExceptFrame(clearExceptFrame); |
@@ -102,8 +157,86 @@ void testRandomDecodeAfterClearFrameBufferCache(const char* dir, const char* gif |
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); |
+ bool canDecodeToIndex8 = decoder->canDecodeTo(j, ImageFrame::Index8); |
ImageFrame* frame = decoder->frameBufferAtIndex(j); |
EXPECT_EQ(baselineHashes[j], hashBitmap(frame->getSkBitmap())); |
+ EXPECT_EQ(canDecodeToIndex8, frame->getSkBitmap().colorType() == kIndex_8_SkColorType); |
+ } |
+ } |
+ } |
+} |
+ |
+void testRandomFrameDecodeCompare(const char* dir, const char* gifFile) |
+{ |
+ SCOPED_TRACE(gifFile); |
+ |
+ RefPtr<SharedBuffer> fullData = readFile(dir, gifFile); |
+ ASSERT_TRUE(fullData.get()); |
+ Vector<unsigned> baselineHashes; |
+ createDecodingBaseline(&createDecoderN32, fullData.get(), &baselineHashes); |
+ size_t frameCount = baselineHashes.size(); |
+ |
+ // Random decoding should get the same results as sequential decoding. |
+ OwnPtr<ImageDecoder> decoder = createDecoderN32(); |
+ // Test Index8 and RGBA through the same scenario. |
+ OwnPtr<ImageDecoder> decoderI8 = createDecoderIndex8(); |
+ |
+ decoder->setData(fullData.get(), true); |
+ decoderI8->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], hashBitmap(frame->getSkBitmap())); |
+ ImageFrame* frameI8 = decoderI8->frameBufferAtIndex(j); |
+ ASSERT_TRUE(areFramesColorsEqual(*frame, *frameI8)); |
+ } |
+ } |
+ |
+ // Decoding in reverse order. |
+ decoder = createDecoderN32(); |
+ decoderI8 = createDecoderIndex8(); |
+ decoder->setData(fullData.get(), true); |
+ decoderI8->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], hashBitmap(frame->getSkBitmap())); |
+ ImageFrame* frameI8 = decoderI8->frameBufferAtIndex(i - 1); |
+ ASSERT_TRUE(areFramesColorsEqual(*frame, *frameI8)); |
+ } |
+} |
+ |
+void testRandomDecodeAfterClearFrameBufferCacheCompare(const char* dir, const char* gifFile) |
+{ |
+ SCOPED_TRACE(gifFile); |
+ |
+ RefPtr<SharedBuffer> data = readFile(dir, gifFile); |
+ ASSERT_TRUE(data.get()); |
+ Vector<unsigned> baselineHashes; |
+ createDecodingBaseline(&createDecoderN32, data.get(), &baselineHashes); |
+ size_t frameCount = baselineHashes.size(); |
+ |
+ OwnPtr<ImageDecoder> decoder = createDecoderN32(); |
+ // Test Index8 and RGBA through the same scenario and compare pixels. |
+ OwnPtr<ImageDecoder> decoderI8 = createDecoderIndex8(); |
+ |
+ decoder->setData(data.get(), true); |
+ decoderI8->setData(data.get(), true); |
+ for (size_t clearExceptFrame = 0; clearExceptFrame < frameCount; ++clearExceptFrame) { |
+ decoder->clearCacheExceptFrame(clearExceptFrame); |
+ decoderI8->clearCacheExceptFrame(clearExceptFrame); |
+ const size_t skippingStep = 5; |
+ for (size_t i = 0; i < skippingStep; ++i) { |
+ for (size_t j = 0; j < frameCount; j += (i + 2)) { |
+ SCOPED_TRACE(testing::Message() << "Random i:" << i << " j:" << j); |
+ ImageFrame* frame = decoder->frameBufferAtIndex(j); |
+ EXPECT_EQ(baselineHashes[j], hashBitmap(frame->getSkBitmap())); |
+ bool canDecodeToIndex8 = decoderI8->canDecodeTo(j, ImageFrame::Index8); |
+ ImageFrame* frameI8 = decoderI8->frameBufferAtIndex(j); |
+ EXPECT_EQ(canDecodeToIndex8, frameI8->getSkBitmap().colorType() == kIndex_8_SkColorType); |
+ ASSERT_TRUE(areFramesColorsEqual(*frame, *frameI8)); |
} |
} |
} |
@@ -113,11 +246,13 @@ void testRandomDecodeAfterClearFrameBufferCache(const char* dir, const char* gif |
TEST(GIFImageDecoderTest, decodeTwoFrames) |
{ |
- OwnPtr<ImageDecoder> decoder = createDecoder(); |
+ OwnPtr<ImageDecoder> decoder = createDecoderN32(); |
+ OwnPtr<ImageDecoder> decoderI8 = createDecoderIndex8(); |
RefPtr<SharedBuffer> data = readFile(layoutTestResourcesDir, "animated.gif"); |
ASSERT_TRUE(data.get()); |
decoder->setData(data.get(), true); |
+ decoderI8->setData(data.get(), true); |
EXPECT_EQ(cAnimationLoopOnce, decoder->repetitionCount()); |
ImageFrame* frame = decoder->frameBufferAtIndex(0); |
@@ -126,6 +261,14 @@ TEST(GIFImageDecoderTest, decodeTwoFrames) |
EXPECT_EQ(16, frame->getSkBitmap().width()); |
EXPECT_EQ(16, frame->getSkBitmap().height()); |
+ ImageFrame* frameI8 = decoderI8->frameBufferAtIndex(0); |
+ generationID0 = frameI8->getSkBitmap().getGenerationID(); |
+ EXPECT_EQ(ImageFrame::FrameComplete, frameI8->status()); |
+ EXPECT_EQ(16, frameI8->getSkBitmap().width()); |
+ EXPECT_EQ(16, frameI8->getSkBitmap().height()); |
+ |
+ ASSERT_TRUE(areFramesColorsEqual(*frame, *frameI8)); |
+ |
frame = decoder->frameBufferAtIndex(1); |
uint32_t generationID1 = frame->getSkBitmap().getGenerationID(); |
EXPECT_EQ(ImageFrame::FrameComplete, frame->status()); |
@@ -135,10 +278,16 @@ TEST(GIFImageDecoderTest, decodeTwoFrames) |
EXPECT_EQ(2u, decoder->frameCount()); |
EXPECT_EQ(cAnimationLoopInfinite, decoder->repetitionCount()); |
+ |
+ frameI8 = decoderI8->frameBufferAtIndex(1); |
+ ASSERT_TRUE(areFramesColorsEqual(*frame, *frameI8)); |
+ EXPECT_EQ(2u, decoderI8->frameCount()); |
+ EXPECT_EQ(cAnimationLoopInfinite, decoderI8->repetitionCount()); |
} |
TEST(GIFImageDecoderTest, parseAndDecode) |
{ |
+ LOOP_DECODERS_BEGIN |
OwnPtr<ImageDecoder> decoder = createDecoder(); |
RefPtr<SharedBuffer> data = readFile(layoutTestResourcesDir, "animated.gif"); |
@@ -159,11 +308,13 @@ TEST(GIFImageDecoderTest, parseAndDecode) |
EXPECT_EQ(16, frame->getSkBitmap().width()); |
EXPECT_EQ(16, frame->getSkBitmap().height()); |
EXPECT_EQ(cAnimationLoopInfinite, decoder->repetitionCount()); |
+ LOOP_DECODERS_END |
} |
TEST(GIFImageDecoderTest, parseByteByByte) |
{ |
- OwnPtr<ImageDecoder> decoder = createDecoder(); |
+ OwnPtr<ImageDecoder> decoder = createDecoderN32(); |
+ OwnPtr<ImageDecoder> decoderI8 = createDecoderIndex8(); |
RefPtr<SharedBuffer> data = readFile(layoutTestResourcesDir, "animated.gif"); |
ASSERT_TRUE(data.get()); |
@@ -174,8 +325,11 @@ TEST(GIFImageDecoderTest, parseByteByByte) |
for (size_t length = 1; length <= data->size(); ++length) { |
RefPtr<SharedBuffer> tempData = SharedBuffer::create(data->data(), length); |
decoder->setData(tempData.get(), length == data->size()); |
+ decoderI8->setData(tempData.get(), length == data->size()); |
EXPECT_LE(frameCount, decoder->frameCount()); |
+ EXPECT_LE(frameCount, decoderI8->frameCount()); |
+ EXPECT_EQ(decoder->frameCount(), decoderI8->frameCount()); |
frameCount = decoder->frameCount(); |
} |
@@ -188,7 +342,8 @@ TEST(GIFImageDecoderTest, parseByteByByte) |
TEST(GIFImageDecoderTest, parseAndDecodeByteByByte) |
{ |
- OwnPtr<ImageDecoder> decoder = createDecoder(); |
+ OwnPtr<ImageDecoder> decoder = createDecoderN32(); |
+ OwnPtr<ImageDecoder> decoderI8 = createDecoderIndex8(); |
RefPtr<SharedBuffer> data = readFile(layoutTestResourcesDir, "animated-gif-with-offsets.gif"); |
ASSERT_TRUE(data.get()); |
@@ -200,13 +355,18 @@ TEST(GIFImageDecoderTest, parseAndDecodeByteByByte) |
for (size_t length = 1; length <= data->size(); ++length) { |
RefPtr<SharedBuffer> tempData = SharedBuffer::create(data->data(), length); |
decoder->setData(tempData.get(), length == data->size()); |
+ decoderI8->setData(tempData.get(), length == data->size()); |
EXPECT_LE(frameCount, decoder->frameCount()); |
frameCount = decoder->frameCount(); |
+ EXPECT_EQ(frameCount, decoderI8->frameCount()); |
ImageFrame* frame = decoder->frameBufferAtIndex(frameCount - 1); |
if (frame && frame->status() == ImageFrame::FrameComplete && framesDecoded < frameCount) |
++framesDecoded; |
+ |
+ ImageFrame* frameI8 = decoderI8->frameBufferAtIndex(frameCount - 1); |
+ ASSERT_TRUE(!(frame || frameI8) || areFramesColorsEqual(*frame, *frameI8)); |
} |
EXPECT_EQ(5u, decoder->frameCount()); |
@@ -216,6 +376,7 @@ TEST(GIFImageDecoderTest, parseAndDecodeByteByByte) |
TEST(GIFImageDecoderTest, brokenSecondFrame) |
{ |
+ LOOP_DECODERS_BEGIN |
OwnPtr<ImageDecoder> decoder = createDecoder(); |
RefPtr<SharedBuffer> data = readFile(webTestsDataDir, "broken.gif"); |
@@ -226,10 +387,12 @@ TEST(GIFImageDecoderTest, brokenSecondFrame) |
EXPECT_EQ(1u, decoder->frameCount()); |
ImageFrame* frame = decoder->frameBufferAtIndex(1); |
EXPECT_FALSE(frame); |
+ LOOP_DECODERS_END |
} |
TEST(GIFImageDecoderTest, progressiveDecode) |
{ |
+ LOOP_DECODERS_BEGIN |
RefPtr<SharedBuffer> fullData = readFile(webTestsDataDir, "radient.gif"); |
ASSERT_TRUE(fullData.get()); |
const size_t fullLength = fullData->size(); |
@@ -277,10 +440,12 @@ TEST(GIFImageDecoderTest, progressiveDecode) |
} |
} |
EXPECT_TRUE(match); |
+ LOOP_DECODERS_END |
} |
TEST(GIFImageDecoderTest, allDataReceivedTruncation) |
{ |
+ LOOP_DECODERS_BEGIN |
OwnPtr<ImageDecoder> decoder = createDecoder(); |
RefPtr<SharedBuffer> data = readFile(layoutTestResourcesDir, "animated.gif"); |
@@ -297,10 +462,12 @@ TEST(GIFImageDecoderTest, allDataReceivedTruncation) |
EXPECT_FALSE(decoder->failed()); |
decoder->frameBufferAtIndex(1); |
EXPECT_TRUE(decoder->failed()); |
+ LOOP_DECODERS_END |
} |
TEST(GIFImageDecoderTest, frameIsComplete) |
{ |
+ LOOP_DECODERS_BEGIN |
OwnPtr<ImageDecoder> decoder = createDecoder(); |
RefPtr<SharedBuffer> data = readFile(layoutTestResourcesDir, "animated.gif"); |
@@ -312,10 +479,12 @@ TEST(GIFImageDecoderTest, frameIsComplete) |
EXPECT_TRUE(decoder->frameIsCompleteAtIndex(0)); |
EXPECT_TRUE(decoder->frameIsCompleteAtIndex(1)); |
EXPECT_EQ(cAnimationLoopInfinite, decoder->repetitionCount()); |
+ LOOP_DECODERS_END |
} |
TEST(GIFImageDecoderTest, frameIsCompleteLoading) |
{ |
+ LOOP_DECODERS_BEGIN |
OwnPtr<ImageDecoder> decoder = createDecoder(); |
RefPtr<SharedBuffer> data = readFile(layoutTestResourcesDir, "animated.gif"); |
@@ -334,10 +503,12 @@ TEST(GIFImageDecoderTest, frameIsCompleteLoading) |
EXPECT_EQ(2u, decoder->frameCount()); |
EXPECT_TRUE(decoder->frameIsCompleteAtIndex(0)); |
EXPECT_TRUE(decoder->frameIsCompleteAtIndex(1)); |
+ LOOP_DECODERS_END |
} |
TEST(GIFImageDecoderTest, badTerminator) |
{ |
+ LOOP_DECODERS_BEGIN |
RefPtr<SharedBuffer> referenceData = readFile(webTestsDataDir, "radient.gif"); |
RefPtr<SharedBuffer> testData = readFile(webTestsDataDir, "radient-bad-terminator.gif"); |
ASSERT_TRUE(referenceData.get()); |
@@ -356,10 +527,12 @@ TEST(GIFImageDecoderTest, badTerminator) |
ASSERT(testFrame); |
EXPECT_EQ(hashBitmap(referenceFrame->getSkBitmap()), hashBitmap(testFrame->getSkBitmap())); |
+ LOOP_DECODERS_END |
} |
TEST(GIFImageDecoderTest, updateRequiredPreviousFrameAfterFirstDecode) |
{ |
+ LOOP_DECODERS_BEGIN |
OwnPtr<ImageDecoder> decoder = createDecoder(); |
RefPtr<SharedBuffer> fullData = readFile(layoutTestResourcesDir, "animated-10color.gif"); |
@@ -382,6 +555,7 @@ TEST(GIFImageDecoderTest, updateRequiredPreviousFrameAfterFirstDecode) |
decoder->setData(fullData.get(), true); |
for (size_t i = 0; i < frameCount; ++i) |
EXPECT_EQ(kNotFound, decoder->frameBufferAtIndex(i)->requiredPreviousFrameIndex()); |
+ LOOP_DECODERS_END |
} |
TEST(GIFImageDecoderTest, randomFrameDecode) |
@@ -393,6 +567,15 @@ TEST(GIFImageDecoderTest, randomFrameDecode) |
testRandomFrameDecode(layoutTestResourcesDir, "animated-10color.gif"); |
} |
+TEST(GIFImageDecoderTest, randomFrameDecodeCompare) |
+{ |
+ // Single frame image. |
+ testRandomFrameDecodeCompare(webTestsDataDir, "radient.gif"); |
+ // Multiple frame images. |
+ testRandomFrameDecodeCompare(layoutTestResourcesDir, "animated-gif-with-offsets.gif"); |
+ testRandomFrameDecodeCompare(layoutTestResourcesDir, "animated-10color.gif"); |
+} |
+ |
TEST(GIFImageDecoderTest, randomDecodeAfterClearFrameBufferCache) |
{ |
// Single frame image. |
@@ -402,12 +585,23 @@ TEST(GIFImageDecoderTest, randomDecodeAfterClearFrameBufferCache) |
testRandomDecodeAfterClearFrameBufferCache(layoutTestResourcesDir, "animated-10color.gif"); |
} |
+TEST(GIFImageDecoderTest, randomDecodeAfterClearFrameBufferCacheCompare) |
+{ |
+ // Single frame image. |
+ testRandomDecodeAfterClearFrameBufferCacheCompare(webTestsDataDir, "radient.gif"); |
+ // Multiple frame images with offset. All frames depend to frame 0. |
+ testRandomDecodeAfterClearFrameBufferCacheCompare(layoutTestResourcesDir, "animated-gif-with-offsets.gif"); |
+ testRandomDecodeAfterClearFrameBufferCacheCompare(layoutTestResourcesDir, "animated-10color.gif"); |
+} |
+ |
TEST(GIFImageDecoderTest, resumePartialDecodeAfterClearFrameBufferCache) |
{ |
RefPtr<SharedBuffer> fullData = readFile(layoutTestResourcesDir, "animated-10color.gif"); |
ASSERT_TRUE(fullData.get()); |
+ |
+ LOOP_DECODERS_BEGIN |
Vector<unsigned> baselineHashes; |
- createDecodingBaseline(&createDecoder, fullData.get(), &baselineHashes); |
+ createDecodingBaseline(createDecoder, fullData.get(), &baselineHashes); |
size_t frameCount = baselineHashes.size(); |
OwnPtr<ImageDecoder> decoder = createDecoder(); |
@@ -431,6 +625,7 @@ TEST(GIFImageDecoderTest, resumePartialDecodeAfterClearFrameBufferCache) |
ImageFrame* firstFrame = decoder->frameBufferAtIndex(0); |
EXPECT_EQ(ImageFrame::FrameComplete, firstFrame->status()); |
EXPECT_EQ(baselineHashes[0], hashBitmap(firstFrame->getSkBitmap())); |
+ LOOP_DECODERS_END |
} |
// The first LZW codes in the image are invalid values that try to create a loop |
@@ -440,11 +635,13 @@ TEST(GIFImageDecoderTest, badInitialCode) |
RefPtr<SharedBuffer> testData = readFile(decodersTestingDir, "bad-initial-code.gif"); |
ASSERT_TRUE(testData.get()); |
+ LOOP_DECODERS_BEGIN |
OwnPtr<ImageDecoder> testDecoder = createDecoder(); |
testDecoder->setData(testData.get(), true); |
EXPECT_EQ(1u, testDecoder->frameCount()); |
ASSERT_TRUE(testDecoder->frameBufferAtIndex(0)); |
EXPECT_TRUE(testDecoder->failed()); |
+ LOOP_DECODERS_END |
} |
// The image has an invalid LZW code that exceeds dictionary size. Decoding should fail. |
@@ -453,15 +650,18 @@ TEST(GIFImageDecoderTest, badCode) |
RefPtr<SharedBuffer> testData = readFile(decodersTestingDir, "bad-code.gif"); |
ASSERT_TRUE(testData.get()); |
+ LOOP_DECODERS_BEGIN |
OwnPtr<ImageDecoder> testDecoder = createDecoder(); |
testDecoder->setData(testData.get(), true); |
EXPECT_EQ(1u, testDecoder->frameCount()); |
ASSERT_TRUE(testDecoder->frameBufferAtIndex(0)); |
EXPECT_TRUE(testDecoder->failed()); |
+ LOOP_DECODERS_END |
} |
TEST(GIFImageDecoderTest, invalidDisposalMethod) |
{ |
+ LOOP_DECODERS_BEGIN |
OwnPtr<ImageDecoder> decoder = createDecoder(); |
// The image has 2 frames, with disposal method 4 and 5, respectively. |
@@ -474,13 +674,14 @@ TEST(GIFImageDecoderTest, invalidDisposalMethod) |
EXPECT_EQ(ImageFrame::DisposeOverwritePrevious, decoder->frameBufferAtIndex(0)->disposalMethod()); |
// Disposal method 5 is ignored. |
EXPECT_EQ(ImageFrame::DisposeNotSpecified, decoder->frameBufferAtIndex(1)->disposalMethod()); |
+ LOOP_DECODERS_END |
} |
TEST(GIFImageDecoderTest, firstFrameHasGreaterSizeThanScreenSize) |
{ |
RefPtr<SharedBuffer> fullData = readFile(decodersTestingDir, "first-frame-has-greater-size-than-screen-size.gif"); |
ASSERT_TRUE(fullData.get()); |
- |
+ LOOP_DECODERS_BEGIN |
OwnPtr<ImageDecoder> decoder; |
IntSize frameSize; |
@@ -498,6 +699,55 @@ TEST(GIFImageDecoderTest, firstFrameHasGreaterSizeThanScreenSize) |
ASSERT_EQ(frameSize.width(), decoder->decodedSize().width()); |
ASSERT_EQ(frameSize.height(), decoder->decodedSize().height()); |
} |
+ LOOP_DECODERS_END |
+} |
+ |
+TEST(GIFImageDecoderTest, convertInFrameGenerator) |
+{ |
+ RefPtr<SharedBuffer> fullData = readFile(layoutTestResourcesDir, "animated-10color.gif"); |
+ ASSERT_TRUE(fullData.get()); |
+ OwnPtr<ImageDecoder> decoder = createDecoderIndex8(); |
+ decoder->setData(fullData.get(), true); |
+ ImageFrame* firstFrame = decoder->frameBufferAtIndex(0); |
+ ASSERT_TRUE(firstFrame->bitmap().colorType() == kIndex_8_SkColorType); |
+ const SkBitmap index8Bitmap = firstFrame->bitmap(); |
+ SkBitmap target; |
+ target.setInfo(SkImageInfo::MakeN32(firstFrame->bitmap().width(), firstFrame->bitmap().height(), index8Bitmap.alphaType())); |
+ target.allocPixels(); |
+ |
+ RefPtr<ImageFrameGenerator> m_generator = ImageFrameGenerator::create(firstFrame->bitmap().info().dimensions(), fullData, true); |
+ ImageDecodingStore::instance().setCacheLimitInBytes(1024 * 1024); |
+ ImageDecodingStore::instance().insertDecoder(m_generator.get(), decoder.release()); |
+ SkAutoLockPixels lock(target); |
+ m_generator->decodeAndScale(0, target.info(), target.getPixels(), target.rowBytes()); |
+ ImageDecodingStore::instance().clear(); |
+ SkAutoLockPixels lockI8(index8Bitmap); |
+ ASSERT_TRUE(areBitmapsColorsEqual(target, index8Bitmap)); |
} |
+// FIXME following benchmarks need to be removed before submitting. |
+#define BENCHMARK_DECODE(dir, file, title, count)\ |
+TEST(GIFImageDecoderTest, tempBench_decodeIndex8##title##warmuprun)\ |
+{\ |
+ testRandomDecodeAfterClearFrameBufferCache(dir, file);\ |
+}\ |
+TEST(GIFImageDecoderTest, tempBench_decodeN32##title##_x##count) \ |
+{\ |
+ for (int i = 0; i < count; ++i)\ |
+ testRandomDecodeAfterClearFrameBufferCache(dir, file);\ |
+}\ |
+\ |
+TEST(GIFImageDecoderTest, tempBench_decodeIndex8##title##_x##count)\ |
+{\ |
+ for (int i = 0; i < count; ++i)\ |
+ testRandomDecodeAfterClearFrameBufferCache(dir, file, ImageFrame::Index8);\ |
+}\ |
+ |
+BENCHMARK_DECODE(layoutTestResourcesDir, "animated-10color.gif", animated_10color_gif, 5) |
+BENCHMARK_DECODE(webTestsDataDir, "radient.gif", radient_gif, 10) |
+BENCHMARK_DECODE(layoutTestResourcesDir, "animated-gif-with-offsets.gif", animated_gif_with_offsets_gif, 5) |
+BENCHMARK_DECODE(layoutTestResourcesDir, "animated.gif", animated_gif, 10) |
+BENCHMARK_DECODE(layoutTestResourcesDir, "quicksort.gif", quicksort_gif, 5) |
+BENCHMARK_DECODE(layoutTestResourcesDir, "large-gif-checkerboard.gif", large_gif_checkerboard_gif, 5) |
+BENCHMARK_DECODE(layoutTestResourcesDir, "3dolph.gif", 3dolph_gif, 5) |
} // namespace blink |