272 lines
7.3 KiB
C++
272 lines
7.3 KiB
C++
|
// Copyright (c) 2011 The Chromium Authors. All rights reserved.
|
||
|
// Use of this source code is governed by a BSD-style license that can be
|
||
|
// found in the LICENSE file.
|
||
|
|
||
|
#include "base/basictypes.h"
|
||
|
#include "base/containers/mru_cache.h"
|
||
|
#include "testing/gtest/include/gtest/gtest.h"
|
||
|
|
||
|
namespace {
|
||
|
|
||
|
int cached_item_live_count = 0;
|
||
|
|
||
|
struct CachedItem {
|
||
|
CachedItem() : value(0) {
|
||
|
cached_item_live_count++;
|
||
|
}
|
||
|
|
||
|
explicit CachedItem(int new_value) : value(new_value) {
|
||
|
cached_item_live_count++;
|
||
|
}
|
||
|
|
||
|
explicit CachedItem(const CachedItem& other) : value(other.value) {
|
||
|
cached_item_live_count++;
|
||
|
}
|
||
|
|
||
|
~CachedItem() {
|
||
|
cached_item_live_count--;
|
||
|
}
|
||
|
|
||
|
int value;
|
||
|
};
|
||
|
|
||
|
} // namespace
|
||
|
|
||
|
TEST(MRUCacheTest, Basic) {
|
||
|
typedef base::MRUCache<int, CachedItem> Cache;
|
||
|
Cache cache(Cache::NO_AUTO_EVICT);
|
||
|
|
||
|
// Check failure conditions
|
||
|
{
|
||
|
CachedItem test_item;
|
||
|
EXPECT_TRUE(cache.Get(0) == cache.end());
|
||
|
EXPECT_TRUE(cache.Peek(0) == cache.end());
|
||
|
}
|
||
|
|
||
|
static const int kItem1Key = 5;
|
||
|
CachedItem item1(10);
|
||
|
Cache::iterator inserted_item = cache.Put(kItem1Key, item1);
|
||
|
EXPECT_EQ(1U, cache.size());
|
||
|
|
||
|
// Check that item1 was properly inserted.
|
||
|
{
|
||
|
Cache::iterator found = cache.Get(kItem1Key);
|
||
|
EXPECT_TRUE(inserted_item == cache.begin());
|
||
|
EXPECT_TRUE(found != cache.end());
|
||
|
|
||
|
found = cache.Peek(kItem1Key);
|
||
|
EXPECT_TRUE(found != cache.end());
|
||
|
|
||
|
EXPECT_EQ(kItem1Key, found->first);
|
||
|
EXPECT_EQ(item1.value, found->second.value);
|
||
|
}
|
||
|
|
||
|
static const int kItem2Key = 7;
|
||
|
CachedItem item2(12);
|
||
|
cache.Put(kItem2Key, item2);
|
||
|
EXPECT_EQ(2U, cache.size());
|
||
|
|
||
|
// Check that item1 is the oldest since item2 was added afterwards.
|
||
|
{
|
||
|
Cache::reverse_iterator oldest = cache.rbegin();
|
||
|
ASSERT_TRUE(oldest != cache.rend());
|
||
|
EXPECT_EQ(kItem1Key, oldest->first);
|
||
|
EXPECT_EQ(item1.value, oldest->second.value);
|
||
|
}
|
||
|
|
||
|
// Check that item1 is still accessible by key.
|
||
|
{
|
||
|
Cache::iterator test_item = cache.Get(kItem1Key);
|
||
|
ASSERT_TRUE(test_item != cache.end());
|
||
|
EXPECT_EQ(kItem1Key, test_item->first);
|
||
|
EXPECT_EQ(item1.value, test_item->second.value);
|
||
|
}
|
||
|
|
||
|
// Check that retrieving item1 pushed item2 to oldest.
|
||
|
{
|
||
|
Cache::reverse_iterator oldest = cache.rbegin();
|
||
|
ASSERT_TRUE(oldest != cache.rend());
|
||
|
EXPECT_EQ(kItem2Key, oldest->first);
|
||
|
EXPECT_EQ(item2.value, oldest->second.value);
|
||
|
}
|
||
|
|
||
|
// Remove the oldest item and check that item1 is now the only member.
|
||
|
{
|
||
|
Cache::reverse_iterator next = cache.Erase(cache.rbegin());
|
||
|
|
||
|
EXPECT_EQ(1U, cache.size());
|
||
|
|
||
|
EXPECT_TRUE(next == cache.rbegin());
|
||
|
EXPECT_EQ(kItem1Key, next->first);
|
||
|
EXPECT_EQ(item1.value, next->second.value);
|
||
|
|
||
|
cache.Erase(cache.begin());
|
||
|
EXPECT_EQ(0U, cache.size());
|
||
|
}
|
||
|
|
||
|
// Check that Clear() works properly.
|
||
|
cache.Put(kItem1Key, item1);
|
||
|
cache.Put(kItem2Key, item2);
|
||
|
EXPECT_EQ(2U, cache.size());
|
||
|
cache.Clear();
|
||
|
EXPECT_EQ(0U, cache.size());
|
||
|
}
|
||
|
|
||
|
TEST(MRUCacheTest, GetVsPeek) {
|
||
|
typedef base::MRUCache<int, CachedItem> Cache;
|
||
|
Cache cache(Cache::NO_AUTO_EVICT);
|
||
|
|
||
|
static const int kItem1Key = 1;
|
||
|
CachedItem item1(10);
|
||
|
cache.Put(kItem1Key, item1);
|
||
|
|
||
|
static const int kItem2Key = 2;
|
||
|
CachedItem item2(20);
|
||
|
cache.Put(kItem2Key, item2);
|
||
|
|
||
|
// This should do nothing since the size is bigger than the number of items.
|
||
|
cache.ShrinkToSize(100);
|
||
|
|
||
|
// Check that item1 starts out as oldest
|
||
|
{
|
||
|
Cache::reverse_iterator iter = cache.rbegin();
|
||
|
ASSERT_TRUE(iter != cache.rend());
|
||
|
EXPECT_EQ(kItem1Key, iter->first);
|
||
|
EXPECT_EQ(item1.value, iter->second.value);
|
||
|
}
|
||
|
|
||
|
// Check that Peek doesn't change ordering
|
||
|
{
|
||
|
Cache::iterator peekiter = cache.Peek(kItem1Key);
|
||
|
ASSERT_TRUE(peekiter != cache.end());
|
||
|
|
||
|
Cache::reverse_iterator iter = cache.rbegin();
|
||
|
ASSERT_TRUE(iter != cache.rend());
|
||
|
EXPECT_EQ(kItem1Key, iter->first);
|
||
|
EXPECT_EQ(item1.value, iter->second.value);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
TEST(MRUCacheTest, KeyReplacement) {
|
||
|
typedef base::MRUCache<int, CachedItem> Cache;
|
||
|
Cache cache(Cache::NO_AUTO_EVICT);
|
||
|
|
||
|
static const int kItem1Key = 1;
|
||
|
CachedItem item1(10);
|
||
|
cache.Put(kItem1Key, item1);
|
||
|
|
||
|
static const int kItem2Key = 2;
|
||
|
CachedItem item2(20);
|
||
|
cache.Put(kItem2Key, item2);
|
||
|
|
||
|
static const int kItem3Key = 3;
|
||
|
CachedItem item3(30);
|
||
|
cache.Put(kItem3Key, item3);
|
||
|
|
||
|
static const int kItem4Key = 4;
|
||
|
CachedItem item4(40);
|
||
|
cache.Put(kItem4Key, item4);
|
||
|
|
||
|
CachedItem item5(50);
|
||
|
cache.Put(kItem3Key, item5);
|
||
|
|
||
|
EXPECT_EQ(4U, cache.size());
|
||
|
for (int i = 0; i < 3; ++i) {
|
||
|
Cache::reverse_iterator iter = cache.rbegin();
|
||
|
ASSERT_TRUE(iter != cache.rend());
|
||
|
}
|
||
|
|
||
|
// Make it so only the most important element is there.
|
||
|
cache.ShrinkToSize(1);
|
||
|
|
||
|
Cache::iterator iter = cache.begin();
|
||
|
EXPECT_EQ(kItem3Key, iter->first);
|
||
|
EXPECT_EQ(item5.value, iter->second.value);
|
||
|
}
|
||
|
|
||
|
// Make sure that the owning version release its pointers properly.
|
||
|
TEST(MRUCacheTest, Owning) {
|
||
|
typedef base::OwningMRUCache<int, CachedItem*> Cache;
|
||
|
Cache cache(Cache::NO_AUTO_EVICT);
|
||
|
|
||
|
int initial_count = cached_item_live_count;
|
||
|
|
||
|
// First insert and item and then overwrite it.
|
||
|
static const int kItem1Key = 1;
|
||
|
cache.Put(kItem1Key, new CachedItem(20));
|
||
|
cache.Put(kItem1Key, new CachedItem(22));
|
||
|
|
||
|
// There should still be one item, and one extra live item.
|
||
|
Cache::iterator iter = cache.Get(kItem1Key);
|
||
|
EXPECT_EQ(1U, cache.size());
|
||
|
EXPECT_TRUE(iter != cache.end());
|
||
|
EXPECT_EQ(initial_count + 1, cached_item_live_count);
|
||
|
|
||
|
// Now remove it.
|
||
|
cache.Erase(cache.begin());
|
||
|
EXPECT_EQ(initial_count, cached_item_live_count);
|
||
|
|
||
|
// Now try another cache that goes out of scope to make sure its pointers
|
||
|
// go away.
|
||
|
{
|
||
|
Cache cache2(Cache::NO_AUTO_EVICT);
|
||
|
cache2.Put(1, new CachedItem(20));
|
||
|
cache2.Put(2, new CachedItem(20));
|
||
|
}
|
||
|
|
||
|
// There should be no objects leaked.
|
||
|
EXPECT_EQ(initial_count, cached_item_live_count);
|
||
|
|
||
|
// Check that Clear() also frees things correctly.
|
||
|
{
|
||
|
Cache cache2(Cache::NO_AUTO_EVICT);
|
||
|
cache2.Put(1, new CachedItem(20));
|
||
|
cache2.Put(2, new CachedItem(20));
|
||
|
EXPECT_EQ(initial_count + 2, cached_item_live_count);
|
||
|
cache2.Clear();
|
||
|
EXPECT_EQ(initial_count, cached_item_live_count);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
TEST(MRUCacheTest, AutoEvict) {
|
||
|
typedef base::OwningMRUCache<int, CachedItem*> Cache;
|
||
|
static const Cache::size_type kMaxSize = 3;
|
||
|
|
||
|
int initial_count = cached_item_live_count;
|
||
|
|
||
|
{
|
||
|
Cache cache(kMaxSize);
|
||
|
|
||
|
static const int kItem1Key = 1, kItem2Key = 2, kItem3Key = 3, kItem4Key = 4;
|
||
|
cache.Put(kItem1Key, new CachedItem(20));
|
||
|
cache.Put(kItem2Key, new CachedItem(21));
|
||
|
cache.Put(kItem3Key, new CachedItem(22));
|
||
|
cache.Put(kItem4Key, new CachedItem(23));
|
||
|
|
||
|
// The cache should only have kMaxSize items in it even though we inserted
|
||
|
// more.
|
||
|
EXPECT_EQ(kMaxSize, cache.size());
|
||
|
}
|
||
|
|
||
|
// There should be no objects leaked.
|
||
|
EXPECT_EQ(initial_count, cached_item_live_count);
|
||
|
}
|
||
|
|
||
|
TEST(MRUCacheTest, HashingMRUCache) {
|
||
|
// Very simple test to make sure that the hashing cache works correctly.
|
||
|
typedef base::HashingMRUCache<std::string, CachedItem> Cache;
|
||
|
Cache cache(Cache::NO_AUTO_EVICT);
|
||
|
|
||
|
CachedItem one(1);
|
||
|
cache.Put("First", one);
|
||
|
|
||
|
CachedItem two(2);
|
||
|
cache.Put("Second", two);
|
||
|
|
||
|
EXPECT_EQ(one.value, cache.Get("First")->second.value);
|
||
|
EXPECT_EQ(two.value, cache.Get("Second")->second.value);
|
||
|
cache.ShrinkToSize(1);
|
||
|
EXPECT_EQ(two.value, cache.Get("Second")->second.value);
|
||
|
EXPECT_TRUE(cache.Get("First") == cache.end());
|
||
|
}
|