| /* |
| * Copyright (C) 2012 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: |
| * |
| * 1. Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * 2. 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. |
| * |
| * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS 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 APPLE OR ITS 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/graphics/ImageDecodingStore.h" |
| |
| #include "platform/graphics/ImageFrameGenerator.h" |
| #include "platform/graphics/test/MockImageDecoder.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| #include <memory> |
| |
| namespace blink { |
| |
| class ImageDecodingStoreTest : public ::testing::Test, |
| public MockImageDecoderClient { |
| public: |
| void SetUp() override { |
| ImageDecodingStore::Instance().SetCacheLimitInBytes(1024 * 1024); |
| generator_ = ImageFrameGenerator::Create(SkISize::Make(100, 100), true, |
| ColorBehavior::Ignore(), {}); |
| decoders_destroyed_ = 0; |
| } |
| |
| void TearDown() override { ImageDecodingStore::Instance().Clear(); } |
| |
| void DecoderBeingDestroyed() override { ++decoders_destroyed_; } |
| |
| void DecodeRequested() override { |
| // Decoder is never used by ImageDecodingStore. |
| ASSERT_TRUE(false); |
| } |
| |
| ImageFrame::Status GetStatus(size_t index) override { |
| return ImageFrame::kFramePartial; |
| } |
| |
| size_t FrameCount() override { return 1; } |
| int RepetitionCount() const override { return kAnimationNone; } |
| TimeDelta FrameDuration() const override { return TimeDelta(); } |
| |
| protected: |
| void EvictOneCache() { |
| size_t memory_usage_in_bytes = |
| ImageDecodingStore::Instance().MemoryUsageInBytes(); |
| if (memory_usage_in_bytes) |
| ImageDecodingStore::Instance().SetCacheLimitInBytes( |
| memory_usage_in_bytes - 1); |
| else |
| ImageDecodingStore::Instance().SetCacheLimitInBytes(0); |
| } |
| |
| scoped_refptr<ImageFrameGenerator> generator_; |
| int decoders_destroyed_; |
| }; |
| |
| TEST_F(ImageDecodingStoreTest, insertDecoder) { |
| const SkISize size = SkISize::Make(1, 1); |
| std::unique_ptr<ImageDecoder> decoder = MockImageDecoder::Create(this); |
| decoder->SetSize(1, 1); |
| const ImageDecoder* ref_decoder = decoder.get(); |
| ImageDecodingStore::Instance().InsertDecoder(generator_.get(), |
| std::move(decoder)); |
| EXPECT_EQ(1, ImageDecodingStore::Instance().CacheEntries()); |
| EXPECT_EQ(4u, ImageDecodingStore::Instance().MemoryUsageInBytes()); |
| |
| ImageDecoder* test_decoder; |
| EXPECT_TRUE(ImageDecodingStore::Instance().LockDecoder( |
| generator_.get(), size, ImageDecoder::kAlphaPremultiplied, |
| &test_decoder)); |
| EXPECT_TRUE(test_decoder); |
| EXPECT_EQ(ref_decoder, test_decoder); |
| ImageDecodingStore::Instance().UnlockDecoder(generator_.get(), test_decoder); |
| EXPECT_EQ(1, ImageDecodingStore::Instance().CacheEntries()); |
| } |
| |
| TEST_F(ImageDecodingStoreTest, evictDecoder) { |
| std::unique_ptr<ImageDecoder> decoder1 = MockImageDecoder::Create(this); |
| std::unique_ptr<ImageDecoder> decoder2 = MockImageDecoder::Create(this); |
| std::unique_ptr<ImageDecoder> decoder3 = MockImageDecoder::Create(this); |
| decoder1->SetSize(1, 1); |
| decoder2->SetSize(2, 2); |
| decoder3->SetSize(3, 3); |
| ImageDecodingStore::Instance().InsertDecoder(generator_.get(), |
| std::move(decoder1)); |
| ImageDecodingStore::Instance().InsertDecoder(generator_.get(), |
| std::move(decoder2)); |
| ImageDecodingStore::Instance().InsertDecoder(generator_.get(), |
| std::move(decoder3)); |
| EXPECT_EQ(3, ImageDecodingStore::Instance().CacheEntries()); |
| EXPECT_EQ(56u, ImageDecodingStore::Instance().MemoryUsageInBytes()); |
| |
| EvictOneCache(); |
| EXPECT_EQ(2, ImageDecodingStore::Instance().CacheEntries()); |
| EXPECT_EQ(52u, ImageDecodingStore::Instance().MemoryUsageInBytes()); |
| |
| EvictOneCache(); |
| EXPECT_EQ(1, ImageDecodingStore::Instance().CacheEntries()); |
| EXPECT_EQ(36u, ImageDecodingStore::Instance().MemoryUsageInBytes()); |
| |
| EvictOneCache(); |
| EXPECT_FALSE(ImageDecodingStore::Instance().CacheEntries()); |
| EXPECT_FALSE(ImageDecodingStore::Instance().MemoryUsageInBytes()); |
| } |
| |
| TEST_F(ImageDecodingStoreTest, decoderInUseNotEvicted) { |
| std::unique_ptr<ImageDecoder> decoder1 = MockImageDecoder::Create(this); |
| std::unique_ptr<ImageDecoder> decoder2 = MockImageDecoder::Create(this); |
| std::unique_ptr<ImageDecoder> decoder3 = MockImageDecoder::Create(this); |
| decoder1->SetSize(1, 1); |
| decoder2->SetSize(2, 2); |
| decoder3->SetSize(3, 3); |
| ImageDecodingStore::Instance().InsertDecoder(generator_.get(), |
| std::move(decoder1)); |
| ImageDecodingStore::Instance().InsertDecoder(generator_.get(), |
| std::move(decoder2)); |
| ImageDecodingStore::Instance().InsertDecoder(generator_.get(), |
| std::move(decoder3)); |
| EXPECT_EQ(3, ImageDecodingStore::Instance().CacheEntries()); |
| |
| ImageDecoder* test_decoder; |
| EXPECT_TRUE(ImageDecodingStore::Instance().LockDecoder( |
| generator_.get(), SkISize::Make(2, 2), ImageDecoder::kAlphaPremultiplied, |
| &test_decoder)); |
| |
| EvictOneCache(); |
| EvictOneCache(); |
| EvictOneCache(); |
| EXPECT_EQ(1, ImageDecodingStore::Instance().CacheEntries()); |
| EXPECT_EQ(16u, ImageDecodingStore::Instance().MemoryUsageInBytes()); |
| |
| ImageDecodingStore::Instance().UnlockDecoder(generator_.get(), test_decoder); |
| EvictOneCache(); |
| EXPECT_FALSE(ImageDecodingStore::Instance().CacheEntries()); |
| EXPECT_FALSE(ImageDecodingStore::Instance().MemoryUsageInBytes()); |
| } |
| |
| TEST_F(ImageDecodingStoreTest, removeDecoder) { |
| const SkISize size = SkISize::Make(1, 1); |
| std::unique_ptr<ImageDecoder> decoder = MockImageDecoder::Create(this); |
| decoder->SetSize(1, 1); |
| const ImageDecoder* ref_decoder = decoder.get(); |
| ImageDecodingStore::Instance().InsertDecoder(generator_.get(), |
| std::move(decoder)); |
| EXPECT_EQ(1, ImageDecodingStore::Instance().CacheEntries()); |
| EXPECT_EQ(4u, ImageDecodingStore::Instance().MemoryUsageInBytes()); |
| |
| ImageDecoder* test_decoder; |
| EXPECT_TRUE(ImageDecodingStore::Instance().LockDecoder( |
| generator_.get(), size, ImageDecoder::kAlphaPremultiplied, |
| &test_decoder)); |
| EXPECT_TRUE(test_decoder); |
| EXPECT_EQ(ref_decoder, test_decoder); |
| ImageDecodingStore::Instance().RemoveDecoder(generator_.get(), test_decoder); |
| EXPECT_FALSE(ImageDecodingStore::Instance().CacheEntries()); |
| |
| EXPECT_FALSE(ImageDecodingStore::Instance().LockDecoder( |
| generator_.get(), size, ImageDecoder::kAlphaPremultiplied, |
| &test_decoder)); |
| } |
| |
| } // namespace blink |