xref: /aosp_15_r20/external/cronet/base/containers/lru_cache_unittest.cc (revision 6777b5387eb2ff775bb5750e3f5d96f37fb7352b)
1*6777b538SAndroid Build Coastguard Worker // Copyright 2011 The Chromium Authors
2*6777b538SAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be
3*6777b538SAndroid Build Coastguard Worker // found in the LICENSE file.
4*6777b538SAndroid Build Coastguard Worker 
5*6777b538SAndroid Build Coastguard Worker #include "base/containers/lru_cache.h"
6*6777b538SAndroid Build Coastguard Worker 
7*6777b538SAndroid Build Coastguard Worker #include <cstddef>
8*6777b538SAndroid Build Coastguard Worker #include <memory>
9*6777b538SAndroid Build Coastguard Worker #include <string>
10*6777b538SAndroid Build Coastguard Worker 
11*6777b538SAndroid Build Coastguard Worker #include "base/memory/ref_counted.h"
12*6777b538SAndroid Build Coastguard Worker #include "base/memory/scoped_refptr.h"
13*6777b538SAndroid Build Coastguard Worker #include "base/tracing_buildflags.h"
14*6777b538SAndroid Build Coastguard Worker #include "testing/gtest/include/gtest/gtest.h"
15*6777b538SAndroid Build Coastguard Worker 
16*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(ENABLE_BASE_TRACING)
17*6777b538SAndroid Build Coastguard Worker #include "base/trace_event/memory_usage_estimator.h"  // no-presubmit-check
18*6777b538SAndroid Build Coastguard Worker #endif  // BUILDFLAG(ENABLE_BASE_TRACING)
19*6777b538SAndroid Build Coastguard Worker 
20*6777b538SAndroid Build Coastguard Worker namespace base {
21*6777b538SAndroid Build Coastguard Worker 
22*6777b538SAndroid Build Coastguard Worker namespace {
23*6777b538SAndroid Build Coastguard Worker 
24*6777b538SAndroid Build Coastguard Worker int cached_item_live_count = 0;
25*6777b538SAndroid Build Coastguard Worker 
26*6777b538SAndroid Build Coastguard Worker struct CachedItem {
CachedItembase::__anon287a6ea90111::CachedItem27*6777b538SAndroid Build Coastguard Worker   CachedItem() : value(0) { cached_item_live_count++; }
28*6777b538SAndroid Build Coastguard Worker 
CachedItembase::__anon287a6ea90111::CachedItem29*6777b538SAndroid Build Coastguard Worker   explicit CachedItem(int new_value) : value(new_value) {
30*6777b538SAndroid Build Coastguard Worker     cached_item_live_count++;
31*6777b538SAndroid Build Coastguard Worker   }
32*6777b538SAndroid Build Coastguard Worker 
CachedItembase::__anon287a6ea90111::CachedItem33*6777b538SAndroid Build Coastguard Worker   CachedItem(const CachedItem& other) : value(other.value) {
34*6777b538SAndroid Build Coastguard Worker     cached_item_live_count++;
35*6777b538SAndroid Build Coastguard Worker   }
36*6777b538SAndroid Build Coastguard Worker 
~CachedItembase::__anon287a6ea90111::CachedItem37*6777b538SAndroid Build Coastguard Worker   ~CachedItem() { cached_item_live_count--; }
38*6777b538SAndroid Build Coastguard Worker 
39*6777b538SAndroid Build Coastguard Worker   int value;
40*6777b538SAndroid Build Coastguard Worker };
41*6777b538SAndroid Build Coastguard Worker 
42*6777b538SAndroid Build Coastguard Worker }  // namespace
43*6777b538SAndroid Build Coastguard Worker 
44*6777b538SAndroid Build Coastguard Worker template <typename LRUCacheTemplate>
45*6777b538SAndroid Build Coastguard Worker class LRUCacheTest : public testing::Test {};
46*6777b538SAndroid Build Coastguard Worker 
47*6777b538SAndroid Build Coastguard Worker struct LRUCacheTemplate {
48*6777b538SAndroid Build Coastguard Worker   template <class Key, class Value, class KeyCompare = std::less<Key>>
49*6777b538SAndroid Build Coastguard Worker   using Type = base::LRUCache<Key, Value, KeyCompare>;
50*6777b538SAndroid Build Coastguard Worker };
51*6777b538SAndroid Build Coastguard Worker 
52*6777b538SAndroid Build Coastguard Worker struct HashingLRUCacheTemplate {
53*6777b538SAndroid Build Coastguard Worker   template <class Key,
54*6777b538SAndroid Build Coastguard Worker             class Value,
55*6777b538SAndroid Build Coastguard Worker             class KeyHash = std::hash<Key>,
56*6777b538SAndroid Build Coastguard Worker             class KeyEqual = std::equal_to<Key>>
57*6777b538SAndroid Build Coastguard Worker   using Type = base::HashingLRUCache<Key, Value, KeyHash, KeyEqual>;
58*6777b538SAndroid Build Coastguard Worker };
59*6777b538SAndroid Build Coastguard Worker 
60*6777b538SAndroid Build Coastguard Worker using LRUCacheTemplates =
61*6777b538SAndroid Build Coastguard Worker     testing::Types<LRUCacheTemplate, HashingLRUCacheTemplate>;
62*6777b538SAndroid Build Coastguard Worker TYPED_TEST_SUITE(LRUCacheTest, LRUCacheTemplates);
63*6777b538SAndroid Build Coastguard Worker 
64*6777b538SAndroid Build Coastguard Worker template <typename LRUCacheSetTemplate>
65*6777b538SAndroid Build Coastguard Worker class LRUCacheSetTest : public testing::Test {};
66*6777b538SAndroid Build Coastguard Worker 
67*6777b538SAndroid Build Coastguard Worker struct LRUCacheSetTemplate {
68*6777b538SAndroid Build Coastguard Worker   template <class Value, class Compare = std::less<Value>>
69*6777b538SAndroid Build Coastguard Worker   using Type = base::LRUCacheSet<Value, Compare>;
70*6777b538SAndroid Build Coastguard Worker };
71*6777b538SAndroid Build Coastguard Worker 
72*6777b538SAndroid Build Coastguard Worker struct HashingLRUCacheSetTemplate {
73*6777b538SAndroid Build Coastguard Worker   template <class Value,
74*6777b538SAndroid Build Coastguard Worker             class Hash = std::hash<Value>,
75*6777b538SAndroid Build Coastguard Worker             class Equal = std::equal_to<Value>>
76*6777b538SAndroid Build Coastguard Worker   using Type = base::HashingLRUCacheSet<Value, Hash, Equal>;
77*6777b538SAndroid Build Coastguard Worker };
78*6777b538SAndroid Build Coastguard Worker 
79*6777b538SAndroid Build Coastguard Worker using LRUCacheSetTemplates =
80*6777b538SAndroid Build Coastguard Worker     testing::Types<LRUCacheSetTemplate, HashingLRUCacheSetTemplate>;
81*6777b538SAndroid Build Coastguard Worker TYPED_TEST_SUITE(LRUCacheSetTest, LRUCacheSetTemplates);
82*6777b538SAndroid Build Coastguard Worker 
TYPED_TEST(LRUCacheTest,Basic)83*6777b538SAndroid Build Coastguard Worker TYPED_TEST(LRUCacheTest, Basic) {
84*6777b538SAndroid Build Coastguard Worker   typedef typename TypeParam::template Type<int, CachedItem> Cache;
85*6777b538SAndroid Build Coastguard Worker   Cache cache(Cache::NO_AUTO_EVICT);
86*6777b538SAndroid Build Coastguard Worker 
87*6777b538SAndroid Build Coastguard Worker   // Check failure conditions
88*6777b538SAndroid Build Coastguard Worker   {
89*6777b538SAndroid Build Coastguard Worker     CachedItem test_item;
90*6777b538SAndroid Build Coastguard Worker     EXPECT_TRUE(cache.Get(0) == cache.end());
91*6777b538SAndroid Build Coastguard Worker     EXPECT_TRUE(cache.Peek(0) == cache.end());
92*6777b538SAndroid Build Coastguard Worker   }
93*6777b538SAndroid Build Coastguard Worker 
94*6777b538SAndroid Build Coastguard Worker   static const int kItem1Key = 5;
95*6777b538SAndroid Build Coastguard Worker   CachedItem item1(10);
96*6777b538SAndroid Build Coastguard Worker   auto inserted_item = cache.Put(kItem1Key, item1);
97*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(1U, cache.size());
98*6777b538SAndroid Build Coastguard Worker 
99*6777b538SAndroid Build Coastguard Worker   // Check that item1 was properly inserted.
100*6777b538SAndroid Build Coastguard Worker   {
101*6777b538SAndroid Build Coastguard Worker     auto found = cache.Get(kItem1Key);
102*6777b538SAndroid Build Coastguard Worker     EXPECT_TRUE(inserted_item == cache.begin());
103*6777b538SAndroid Build Coastguard Worker     EXPECT_TRUE(found != cache.end());
104*6777b538SAndroid Build Coastguard Worker 
105*6777b538SAndroid Build Coastguard Worker     found = cache.Peek(kItem1Key);
106*6777b538SAndroid Build Coastguard Worker     EXPECT_TRUE(found != cache.end());
107*6777b538SAndroid Build Coastguard Worker 
108*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(kItem1Key, found->first);
109*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(item1.value, found->second.value);
110*6777b538SAndroid Build Coastguard Worker   }
111*6777b538SAndroid Build Coastguard Worker 
112*6777b538SAndroid Build Coastguard Worker   static const int kItem2Key = 7;
113*6777b538SAndroid Build Coastguard Worker   CachedItem item2(12);
114*6777b538SAndroid Build Coastguard Worker   cache.Put(kItem2Key, item2);
115*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(2U, cache.size());
116*6777b538SAndroid Build Coastguard Worker 
117*6777b538SAndroid Build Coastguard Worker   // Check that item1 is the oldest since item2 was added afterwards.
118*6777b538SAndroid Build Coastguard Worker   {
119*6777b538SAndroid Build Coastguard Worker     auto oldest = cache.rbegin();
120*6777b538SAndroid Build Coastguard Worker     ASSERT_TRUE(oldest != cache.rend());
121*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(kItem1Key, oldest->first);
122*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(item1.value, oldest->second.value);
123*6777b538SAndroid Build Coastguard Worker   }
124*6777b538SAndroid Build Coastguard Worker 
125*6777b538SAndroid Build Coastguard Worker   // Check that item1 is still accessible by key.
126*6777b538SAndroid Build Coastguard Worker   {
127*6777b538SAndroid Build Coastguard Worker     auto test_item = cache.Get(kItem1Key);
128*6777b538SAndroid Build Coastguard Worker     ASSERT_TRUE(test_item != cache.end());
129*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(kItem1Key, test_item->first);
130*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(item1.value, test_item->second.value);
131*6777b538SAndroid Build Coastguard Worker   }
132*6777b538SAndroid Build Coastguard Worker 
133*6777b538SAndroid Build Coastguard Worker   // Check that retrieving item1 pushed item2 to oldest.
134*6777b538SAndroid Build Coastguard Worker   {
135*6777b538SAndroid Build Coastguard Worker     auto oldest = cache.rbegin();
136*6777b538SAndroid Build Coastguard Worker     ASSERT_TRUE(oldest != cache.rend());
137*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(kItem2Key, oldest->first);
138*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(item2.value, oldest->second.value);
139*6777b538SAndroid Build Coastguard Worker   }
140*6777b538SAndroid Build Coastguard Worker 
141*6777b538SAndroid Build Coastguard Worker   // Remove the oldest item and check that item1 is now the only member.
142*6777b538SAndroid Build Coastguard Worker   {
143*6777b538SAndroid Build Coastguard Worker     auto next = cache.Erase(cache.rbegin());
144*6777b538SAndroid Build Coastguard Worker 
145*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(1U, cache.size());
146*6777b538SAndroid Build Coastguard Worker 
147*6777b538SAndroid Build Coastguard Worker     EXPECT_TRUE(next == cache.rbegin());
148*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(kItem1Key, next->first);
149*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(item1.value, next->second.value);
150*6777b538SAndroid Build Coastguard Worker 
151*6777b538SAndroid Build Coastguard Worker     cache.Erase(cache.begin());
152*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(0U, cache.size());
153*6777b538SAndroid Build Coastguard Worker   }
154*6777b538SAndroid Build Coastguard Worker 
155*6777b538SAndroid Build Coastguard Worker   // Check that Clear() works properly.
156*6777b538SAndroid Build Coastguard Worker   cache.Put(kItem1Key, item1);
157*6777b538SAndroid Build Coastguard Worker   cache.Put(kItem2Key, item2);
158*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(2U, cache.size());
159*6777b538SAndroid Build Coastguard Worker   cache.Clear();
160*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(0U, cache.size());
161*6777b538SAndroid Build Coastguard Worker }
162*6777b538SAndroid Build Coastguard Worker 
TYPED_TEST(LRUCacheTest,GetVsPeek)163*6777b538SAndroid Build Coastguard Worker TYPED_TEST(LRUCacheTest, GetVsPeek) {
164*6777b538SAndroid Build Coastguard Worker   typedef typename TypeParam::template Type<int, CachedItem> Cache;
165*6777b538SAndroid Build Coastguard Worker   Cache cache(Cache::NO_AUTO_EVICT);
166*6777b538SAndroid Build Coastguard Worker 
167*6777b538SAndroid Build Coastguard Worker   static const int kItem1Key = 1;
168*6777b538SAndroid Build Coastguard Worker   CachedItem item1(10);
169*6777b538SAndroid Build Coastguard Worker   cache.Put(kItem1Key, item1);
170*6777b538SAndroid Build Coastguard Worker 
171*6777b538SAndroid Build Coastguard Worker   static const int kItem2Key = 2;
172*6777b538SAndroid Build Coastguard Worker   CachedItem item2(20);
173*6777b538SAndroid Build Coastguard Worker   cache.Put(kItem2Key, item2);
174*6777b538SAndroid Build Coastguard Worker 
175*6777b538SAndroid Build Coastguard Worker   // This should do nothing since the size is bigger than the number of items.
176*6777b538SAndroid Build Coastguard Worker   cache.ShrinkToSize(100);
177*6777b538SAndroid Build Coastguard Worker 
178*6777b538SAndroid Build Coastguard Worker   // Check that item1 starts out as oldest
179*6777b538SAndroid Build Coastguard Worker   {
180*6777b538SAndroid Build Coastguard Worker     auto iter = cache.rbegin();
181*6777b538SAndroid Build Coastguard Worker     ASSERT_TRUE(iter != cache.rend());
182*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(kItem1Key, iter->first);
183*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(item1.value, iter->second.value);
184*6777b538SAndroid Build Coastguard Worker   }
185*6777b538SAndroid Build Coastguard Worker 
186*6777b538SAndroid Build Coastguard Worker   // Check that Peek doesn't change ordering
187*6777b538SAndroid Build Coastguard Worker   {
188*6777b538SAndroid Build Coastguard Worker     auto peekiter = cache.Peek(kItem1Key);
189*6777b538SAndroid Build Coastguard Worker     ASSERT_TRUE(peekiter != cache.end());
190*6777b538SAndroid Build Coastguard Worker 
191*6777b538SAndroid Build Coastguard Worker     auto iter = cache.rbegin();
192*6777b538SAndroid Build Coastguard Worker     ASSERT_TRUE(iter != cache.rend());
193*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(kItem1Key, iter->first);
194*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(item1.value, iter->second.value);
195*6777b538SAndroid Build Coastguard Worker   }
196*6777b538SAndroid Build Coastguard Worker }
197*6777b538SAndroid Build Coastguard Worker 
TYPED_TEST(LRUCacheTest,KeyReplacement)198*6777b538SAndroid Build Coastguard Worker TYPED_TEST(LRUCacheTest, KeyReplacement) {
199*6777b538SAndroid Build Coastguard Worker   typedef typename TypeParam::template Type<int, CachedItem> Cache;
200*6777b538SAndroid Build Coastguard Worker   Cache cache(Cache::NO_AUTO_EVICT);
201*6777b538SAndroid Build Coastguard Worker 
202*6777b538SAndroid Build Coastguard Worker   static const int kItem1Key = 1;
203*6777b538SAndroid Build Coastguard Worker   CachedItem item1(10);
204*6777b538SAndroid Build Coastguard Worker   cache.Put(kItem1Key, item1);
205*6777b538SAndroid Build Coastguard Worker 
206*6777b538SAndroid Build Coastguard Worker   static const int kItem2Key = 2;
207*6777b538SAndroid Build Coastguard Worker   CachedItem item2(20);
208*6777b538SAndroid Build Coastguard Worker   cache.Put(kItem2Key, item2);
209*6777b538SAndroid Build Coastguard Worker 
210*6777b538SAndroid Build Coastguard Worker   static const int kItem3Key = 3;
211*6777b538SAndroid Build Coastguard Worker   CachedItem item3(30);
212*6777b538SAndroid Build Coastguard Worker   cache.Put(kItem3Key, item3);
213*6777b538SAndroid Build Coastguard Worker 
214*6777b538SAndroid Build Coastguard Worker   static const int kItem4Key = 4;
215*6777b538SAndroid Build Coastguard Worker   CachedItem item4(40);
216*6777b538SAndroid Build Coastguard Worker   cache.Put(kItem4Key, item4);
217*6777b538SAndroid Build Coastguard Worker 
218*6777b538SAndroid Build Coastguard Worker   CachedItem item5(50);
219*6777b538SAndroid Build Coastguard Worker   cache.Put(kItem3Key, item5);
220*6777b538SAndroid Build Coastguard Worker 
221*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(4U, cache.size());
222*6777b538SAndroid Build Coastguard Worker   for (int i = 0; i < 3; ++i) {
223*6777b538SAndroid Build Coastguard Worker     auto iter = cache.rbegin();
224*6777b538SAndroid Build Coastguard Worker     ASSERT_TRUE(iter != cache.rend());
225*6777b538SAndroid Build Coastguard Worker   }
226*6777b538SAndroid Build Coastguard Worker 
227*6777b538SAndroid Build Coastguard Worker   // Make it so only the most important element is there.
228*6777b538SAndroid Build Coastguard Worker   cache.ShrinkToSize(1);
229*6777b538SAndroid Build Coastguard Worker 
230*6777b538SAndroid Build Coastguard Worker   auto iter = cache.begin();
231*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(kItem3Key, iter->first);
232*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(item5.value, iter->second.value);
233*6777b538SAndroid Build Coastguard Worker }
234*6777b538SAndroid Build Coastguard Worker 
235*6777b538SAndroid Build Coastguard Worker // Make sure that the cache release its pointers properly.
TYPED_TEST(LRUCacheTest,Owning)236*6777b538SAndroid Build Coastguard Worker TYPED_TEST(LRUCacheTest, Owning) {
237*6777b538SAndroid Build Coastguard Worker   using Cache =
238*6777b538SAndroid Build Coastguard Worker       typename TypeParam::template Type<int, std::unique_ptr<CachedItem>>;
239*6777b538SAndroid Build Coastguard Worker   Cache cache(Cache::NO_AUTO_EVICT);
240*6777b538SAndroid Build Coastguard Worker 
241*6777b538SAndroid Build Coastguard Worker   int initial_count = cached_item_live_count;
242*6777b538SAndroid Build Coastguard Worker 
243*6777b538SAndroid Build Coastguard Worker   // First insert and item and then overwrite it.
244*6777b538SAndroid Build Coastguard Worker   static const int kItem1Key = 1;
245*6777b538SAndroid Build Coastguard Worker   cache.Put(kItem1Key, std::make_unique<CachedItem>(20));
246*6777b538SAndroid Build Coastguard Worker   cache.Put(kItem1Key, std::make_unique<CachedItem>(22));
247*6777b538SAndroid Build Coastguard Worker 
248*6777b538SAndroid Build Coastguard Worker   // There should still be one item, and one extra live item.
249*6777b538SAndroid Build Coastguard Worker   auto iter = cache.Get(kItem1Key);
250*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(1U, cache.size());
251*6777b538SAndroid Build Coastguard Worker   EXPECT_TRUE(iter != cache.end());
252*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(initial_count + 1, cached_item_live_count);
253*6777b538SAndroid Build Coastguard Worker 
254*6777b538SAndroid Build Coastguard Worker   // Now remove it.
255*6777b538SAndroid Build Coastguard Worker   cache.Erase(cache.begin());
256*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(initial_count, cached_item_live_count);
257*6777b538SAndroid Build Coastguard Worker 
258*6777b538SAndroid Build Coastguard Worker   // Now try another cache that goes out of scope to make sure its pointers
259*6777b538SAndroid Build Coastguard Worker   // go away.
260*6777b538SAndroid Build Coastguard Worker   {
261*6777b538SAndroid Build Coastguard Worker     Cache cache2(Cache::NO_AUTO_EVICT);
262*6777b538SAndroid Build Coastguard Worker     cache2.Put(1, std::make_unique<CachedItem>(20));
263*6777b538SAndroid Build Coastguard Worker     cache2.Put(2, std::make_unique<CachedItem>(20));
264*6777b538SAndroid Build Coastguard Worker   }
265*6777b538SAndroid Build Coastguard Worker 
266*6777b538SAndroid Build Coastguard Worker   // There should be no objects leaked.
267*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(initial_count, cached_item_live_count);
268*6777b538SAndroid Build Coastguard Worker 
269*6777b538SAndroid Build Coastguard Worker   // Check that Clear() also frees things correctly.
270*6777b538SAndroid Build Coastguard Worker   {
271*6777b538SAndroid Build Coastguard Worker     Cache cache2(Cache::NO_AUTO_EVICT);
272*6777b538SAndroid Build Coastguard Worker     cache2.Put(1, std::make_unique<CachedItem>(20));
273*6777b538SAndroid Build Coastguard Worker     cache2.Put(2, std::make_unique<CachedItem>(20));
274*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(initial_count + 2, cached_item_live_count);
275*6777b538SAndroid Build Coastguard Worker     cache2.Clear();
276*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(initial_count, cached_item_live_count);
277*6777b538SAndroid Build Coastguard Worker   }
278*6777b538SAndroid Build Coastguard Worker }
279*6777b538SAndroid Build Coastguard Worker 
TYPED_TEST(LRUCacheTest,AutoEvict)280*6777b538SAndroid Build Coastguard Worker TYPED_TEST(LRUCacheTest, AutoEvict) {
281*6777b538SAndroid Build Coastguard Worker   using Cache =
282*6777b538SAndroid Build Coastguard Worker       typename TypeParam::template Type<int, std::unique_ptr<CachedItem>>;
283*6777b538SAndroid Build Coastguard Worker   static const typename Cache::size_type kMaxSize = 3;
284*6777b538SAndroid Build Coastguard Worker 
285*6777b538SAndroid Build Coastguard Worker   int initial_count = cached_item_live_count;
286*6777b538SAndroid Build Coastguard Worker 
287*6777b538SAndroid Build Coastguard Worker   {
288*6777b538SAndroid Build Coastguard Worker     Cache cache(kMaxSize);
289*6777b538SAndroid Build Coastguard Worker 
290*6777b538SAndroid Build Coastguard Worker     static const int kItem1Key = 1, kItem2Key = 2, kItem3Key = 3, kItem4Key = 4;
291*6777b538SAndroid Build Coastguard Worker     cache.Put(kItem1Key, std::make_unique<CachedItem>(20));
292*6777b538SAndroid Build Coastguard Worker     cache.Put(kItem2Key, std::make_unique<CachedItem>(21));
293*6777b538SAndroid Build Coastguard Worker     cache.Put(kItem3Key, std::make_unique<CachedItem>(22));
294*6777b538SAndroid Build Coastguard Worker     cache.Put(kItem4Key, std::make_unique<CachedItem>(23));
295*6777b538SAndroid Build Coastguard Worker 
296*6777b538SAndroid Build Coastguard Worker     // The cache should only have kMaxSize items in it even though we inserted
297*6777b538SAndroid Build Coastguard Worker     // more.
298*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(kMaxSize, cache.size());
299*6777b538SAndroid Build Coastguard Worker   }
300*6777b538SAndroid Build Coastguard Worker 
301*6777b538SAndroid Build Coastguard Worker   // There should be no objects leaked.
302*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(initial_count, cached_item_live_count);
303*6777b538SAndroid Build Coastguard Worker }
304*6777b538SAndroid Build Coastguard Worker 
TYPED_TEST(LRUCacheTest,HashingLRUCache)305*6777b538SAndroid Build Coastguard Worker TYPED_TEST(LRUCacheTest, HashingLRUCache) {
306*6777b538SAndroid Build Coastguard Worker   // Very simple test to make sure that the hashing cache works correctly.
307*6777b538SAndroid Build Coastguard Worker   typedef typename TypeParam::template Type<std::string, CachedItem> Cache;
308*6777b538SAndroid Build Coastguard Worker   Cache cache(Cache::NO_AUTO_EVICT);
309*6777b538SAndroid Build Coastguard Worker 
310*6777b538SAndroid Build Coastguard Worker   CachedItem one(1);
311*6777b538SAndroid Build Coastguard Worker   cache.Put("First", one);
312*6777b538SAndroid Build Coastguard Worker 
313*6777b538SAndroid Build Coastguard Worker   CachedItem two(2);
314*6777b538SAndroid Build Coastguard Worker   cache.Put("Second", two);
315*6777b538SAndroid Build Coastguard Worker 
316*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(one.value, cache.Get("First")->second.value);
317*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(two.value, cache.Get("Second")->second.value);
318*6777b538SAndroid Build Coastguard Worker   cache.ShrinkToSize(1);
319*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(two.value, cache.Get("Second")->second.value);
320*6777b538SAndroid Build Coastguard Worker   EXPECT_TRUE(cache.Get("First") == cache.end());
321*6777b538SAndroid Build Coastguard Worker }
322*6777b538SAndroid Build Coastguard Worker 
TYPED_TEST(LRUCacheTest,Swap)323*6777b538SAndroid Build Coastguard Worker TYPED_TEST(LRUCacheTest, Swap) {
324*6777b538SAndroid Build Coastguard Worker   typedef typename TypeParam::template Type<int, CachedItem> Cache;
325*6777b538SAndroid Build Coastguard Worker   Cache cache1(Cache::NO_AUTO_EVICT);
326*6777b538SAndroid Build Coastguard Worker 
327*6777b538SAndroid Build Coastguard Worker   // Insert two items into cache1.
328*6777b538SAndroid Build Coastguard Worker   static const int kItem1Key = 1;
329*6777b538SAndroid Build Coastguard Worker   CachedItem item1(2);
330*6777b538SAndroid Build Coastguard Worker   auto inserted_item = cache1.Put(kItem1Key, item1);
331*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(1U, cache1.size());
332*6777b538SAndroid Build Coastguard Worker 
333*6777b538SAndroid Build Coastguard Worker   static const int kItem2Key = 3;
334*6777b538SAndroid Build Coastguard Worker   CachedItem item2(4);
335*6777b538SAndroid Build Coastguard Worker   cache1.Put(kItem2Key, item2);
336*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(2U, cache1.size());
337*6777b538SAndroid Build Coastguard Worker 
338*6777b538SAndroid Build Coastguard Worker   // Verify cache1's elements.
339*6777b538SAndroid Build Coastguard Worker   {
340*6777b538SAndroid Build Coastguard Worker     auto iter = cache1.begin();
341*6777b538SAndroid Build Coastguard Worker     ASSERT_TRUE(iter != cache1.end());
342*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(kItem2Key, iter->first);
343*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(item2.value, iter->second.value);
344*6777b538SAndroid Build Coastguard Worker 
345*6777b538SAndroid Build Coastguard Worker     ++iter;
346*6777b538SAndroid Build Coastguard Worker     ASSERT_TRUE(iter != cache1.end());
347*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(kItem1Key, iter->first);
348*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(item1.value, iter->second.value);
349*6777b538SAndroid Build Coastguard Worker   }
350*6777b538SAndroid Build Coastguard Worker 
351*6777b538SAndroid Build Coastguard Worker   // Create another cache2.
352*6777b538SAndroid Build Coastguard Worker   Cache cache2(Cache::NO_AUTO_EVICT);
353*6777b538SAndroid Build Coastguard Worker 
354*6777b538SAndroid Build Coastguard Worker   // Insert three items into cache2.
355*6777b538SAndroid Build Coastguard Worker   static const int kItem3Key = 5;
356*6777b538SAndroid Build Coastguard Worker   CachedItem item3(6);
357*6777b538SAndroid Build Coastguard Worker   inserted_item = cache2.Put(kItem3Key, item3);
358*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(1U, cache2.size());
359*6777b538SAndroid Build Coastguard Worker 
360*6777b538SAndroid Build Coastguard Worker   static const int kItem4Key = 7;
361*6777b538SAndroid Build Coastguard Worker   CachedItem item4(8);
362*6777b538SAndroid Build Coastguard Worker   cache2.Put(kItem4Key, item4);
363*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(2U, cache2.size());
364*6777b538SAndroid Build Coastguard Worker 
365*6777b538SAndroid Build Coastguard Worker   static const int kItem5Key = 9;
366*6777b538SAndroid Build Coastguard Worker   CachedItem item5(10);
367*6777b538SAndroid Build Coastguard Worker   cache2.Put(kItem5Key, item5);
368*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(3U, cache2.size());
369*6777b538SAndroid Build Coastguard Worker 
370*6777b538SAndroid Build Coastguard Worker   // Verify cache2's elements.
371*6777b538SAndroid Build Coastguard Worker   {
372*6777b538SAndroid Build Coastguard Worker     auto iter = cache2.begin();
373*6777b538SAndroid Build Coastguard Worker     ASSERT_TRUE(iter != cache2.end());
374*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(kItem5Key, iter->first);
375*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(item5.value, iter->second.value);
376*6777b538SAndroid Build Coastguard Worker 
377*6777b538SAndroid Build Coastguard Worker     ++iter;
378*6777b538SAndroid Build Coastguard Worker     ASSERT_TRUE(iter != cache2.end());
379*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(kItem4Key, iter->first);
380*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(item4.value, iter->second.value);
381*6777b538SAndroid Build Coastguard Worker 
382*6777b538SAndroid Build Coastguard Worker     ++iter;
383*6777b538SAndroid Build Coastguard Worker     ASSERT_TRUE(iter != cache2.end());
384*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(kItem3Key, iter->first);
385*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(item3.value, iter->second.value);
386*6777b538SAndroid Build Coastguard Worker   }
387*6777b538SAndroid Build Coastguard Worker 
388*6777b538SAndroid Build Coastguard Worker   // Swap cache1 and cache2 and verify cache2 has cache1's elements and cache1
389*6777b538SAndroid Build Coastguard Worker   // has cache2's elements.
390*6777b538SAndroid Build Coastguard Worker   cache2.Swap(cache1);
391*6777b538SAndroid Build Coastguard Worker 
392*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(3U, cache1.size());
393*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(2U, cache2.size());
394*6777b538SAndroid Build Coastguard Worker 
395*6777b538SAndroid Build Coastguard Worker   // Verify cache1's elements.
396*6777b538SAndroid Build Coastguard Worker   {
397*6777b538SAndroid Build Coastguard Worker     auto iter = cache1.begin();
398*6777b538SAndroid Build Coastguard Worker     ASSERT_TRUE(iter != cache1.end());
399*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(kItem5Key, iter->first);
400*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(item5.value, iter->second.value);
401*6777b538SAndroid Build Coastguard Worker 
402*6777b538SAndroid Build Coastguard Worker     ++iter;
403*6777b538SAndroid Build Coastguard Worker     ASSERT_TRUE(iter != cache1.end());
404*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(kItem4Key, iter->first);
405*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(item4.value, iter->second.value);
406*6777b538SAndroid Build Coastguard Worker 
407*6777b538SAndroid Build Coastguard Worker     ++iter;
408*6777b538SAndroid Build Coastguard Worker     ASSERT_TRUE(iter != cache1.end());
409*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(kItem3Key, iter->first);
410*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(item3.value, iter->second.value);
411*6777b538SAndroid Build Coastguard Worker   }
412*6777b538SAndroid Build Coastguard Worker 
413*6777b538SAndroid Build Coastguard Worker   // Verify cache2's elements.
414*6777b538SAndroid Build Coastguard Worker   {
415*6777b538SAndroid Build Coastguard Worker     auto iter = cache2.begin();
416*6777b538SAndroid Build Coastguard Worker     ASSERT_TRUE(iter != cache2.end());
417*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(kItem2Key, iter->first);
418*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(item2.value, iter->second.value);
419*6777b538SAndroid Build Coastguard Worker 
420*6777b538SAndroid Build Coastguard Worker     ++iter;
421*6777b538SAndroid Build Coastguard Worker     ASSERT_TRUE(iter != cache2.end());
422*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(kItem1Key, iter->first);
423*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(item1.value, iter->second.value);
424*6777b538SAndroid Build Coastguard Worker   }
425*6777b538SAndroid Build Coastguard Worker }
426*6777b538SAndroid Build Coastguard Worker 
TYPED_TEST(LRUCacheSetTest,SetTest)427*6777b538SAndroid Build Coastguard Worker TYPED_TEST(LRUCacheSetTest, SetTest) {
428*6777b538SAndroid Build Coastguard Worker   typedef typename TypeParam::template Type<std::string> Cache;
429*6777b538SAndroid Build Coastguard Worker   Cache cache(Cache::NO_AUTO_EVICT);
430*6777b538SAndroid Build Coastguard Worker 
431*6777b538SAndroid Build Coastguard Worker   cache.Put("Hello");
432*6777b538SAndroid Build Coastguard Worker   cache.Put("world");
433*6777b538SAndroid Build Coastguard Worker   cache.Put("foo");
434*6777b538SAndroid Build Coastguard Worker   cache.Put("bar");
435*6777b538SAndroid Build Coastguard Worker 
436*6777b538SAndroid Build Coastguard Worker   // Insert a duplicate element
437*6777b538SAndroid Build Coastguard Worker   cache.Put("foo");
438*6777b538SAndroid Build Coastguard Worker 
439*6777b538SAndroid Build Coastguard Worker   // Iterate from oldest to newest
440*6777b538SAndroid Build Coastguard Worker   auto r_iter = cache.rbegin();
441*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(*r_iter, "Hello");
442*6777b538SAndroid Build Coastguard Worker   ++r_iter;
443*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(*r_iter, "world");
444*6777b538SAndroid Build Coastguard Worker   ++r_iter;
445*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(*r_iter, "bar");
446*6777b538SAndroid Build Coastguard Worker   ++r_iter;
447*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(*r_iter, "foo");
448*6777b538SAndroid Build Coastguard Worker   ++r_iter;
449*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(r_iter, cache.rend());
450*6777b538SAndroid Build Coastguard Worker 
451*6777b538SAndroid Build Coastguard Worker   // Iterate from newest to oldest
452*6777b538SAndroid Build Coastguard Worker   auto iter = cache.begin();
453*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(*iter, "foo");
454*6777b538SAndroid Build Coastguard Worker   ++iter;
455*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(*iter, "bar");
456*6777b538SAndroid Build Coastguard Worker   ++iter;
457*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(*iter, "world");
458*6777b538SAndroid Build Coastguard Worker   ++iter;
459*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(*iter, "Hello");
460*6777b538SAndroid Build Coastguard Worker   ++iter;
461*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(iter, cache.end());
462*6777b538SAndroid Build Coastguard Worker }
463*6777b538SAndroid Build Coastguard Worker 
464*6777b538SAndroid Build Coastguard Worker // Generalized dereference function. For the base case, this is the identity
465*6777b538SAndroid Build Coastguard Worker // function.
466*6777b538SAndroid Build Coastguard Worker template <typename T>
467*6777b538SAndroid Build Coastguard Worker struct Deref {
468*6777b538SAndroid Build Coastguard Worker   using Target = T;
derefbase::Deref469*6777b538SAndroid Build Coastguard Worker   static const Target& deref(const T& x) { return x; }
470*6777b538SAndroid Build Coastguard Worker };
471*6777b538SAndroid Build Coastguard Worker 
472*6777b538SAndroid Build Coastguard Worker // `RefCountedData` wraps a type in an interface that supports refcounting.
473*6777b538SAndroid Build Coastguard Worker // Deref this as the wrapped type.
474*6777b538SAndroid Build Coastguard Worker template <typename T>
475*6777b538SAndroid Build Coastguard Worker struct Deref<RefCountedData<T>> {
476*6777b538SAndroid Build Coastguard Worker   using Target = typename Deref<T>::Target;
derefbase::Deref477*6777b538SAndroid Build Coastguard Worker   static const Target& deref(const RefCountedData<T>& x) {
478*6777b538SAndroid Build Coastguard Worker     return Deref<T>::deref(x.data);
479*6777b538SAndroid Build Coastguard Worker   }
480*6777b538SAndroid Build Coastguard Worker };
481*6777b538SAndroid Build Coastguard Worker 
482*6777b538SAndroid Build Coastguard Worker // `scoped_refptr` is a smart pointer that implements reference counting.
483*6777b538SAndroid Build Coastguard Worker // Deref this as the pointee.
484*6777b538SAndroid Build Coastguard Worker template <typename T>
485*6777b538SAndroid Build Coastguard Worker struct Deref<scoped_refptr<T>> {
486*6777b538SAndroid Build Coastguard Worker   using Target = typename Deref<T>::Target;
derefbase::Deref487*6777b538SAndroid Build Coastguard Worker   static const Target& deref(const scoped_refptr<T>& x) {
488*6777b538SAndroid Build Coastguard Worker     return Deref<T>::deref(*x);
489*6777b538SAndroid Build Coastguard Worker   }
490*6777b538SAndroid Build Coastguard Worker };
491*6777b538SAndroid Build Coastguard Worker 
492*6777b538SAndroid Build Coastguard Worker // Implementation of a `std::less`-like type that dereferences the given values
493*6777b538SAndroid Build Coastguard Worker // before comparison.
494*6777b538SAndroid Build Coastguard Worker template <typename T>
495*6777b538SAndroid Build Coastguard Worker struct DerefCompare {
operator ()base::DerefCompare496*6777b538SAndroid Build Coastguard Worker   bool operator()(const T& lhs, const T& rhs) const {
497*6777b538SAndroid Build Coastguard Worker     return Deref<T>::deref(lhs) < Deref<T>::deref(rhs);
498*6777b538SAndroid Build Coastguard Worker   }
499*6777b538SAndroid Build Coastguard Worker };
500*6777b538SAndroid Build Coastguard Worker 
501*6777b538SAndroid Build Coastguard Worker // Implementation of a `std::equal_to`-like type that dereferences the given
502*6777b538SAndroid Build Coastguard Worker // values before comparison.
503*6777b538SAndroid Build Coastguard Worker template <typename T>
504*6777b538SAndroid Build Coastguard Worker struct DerefEqual {
operator ()base::DerefEqual505*6777b538SAndroid Build Coastguard Worker   bool operator()(const T& lhs, const T& rhs) const {
506*6777b538SAndroid Build Coastguard Worker     return Deref<T>::deref(lhs) == Deref<T>::deref(rhs);
507*6777b538SAndroid Build Coastguard Worker   }
508*6777b538SAndroid Build Coastguard Worker };
509*6777b538SAndroid Build Coastguard Worker 
510*6777b538SAndroid Build Coastguard Worker // Implementation of a `std::hash`-like type that dereferences the given value
511*6777b538SAndroid Build Coastguard Worker // before calculating the hash.
512*6777b538SAndroid Build Coastguard Worker template <typename T, template <class> typename HashT = std::hash>
513*6777b538SAndroid Build Coastguard Worker struct DerefHash {
operator ()base::DerefHash514*6777b538SAndroid Build Coastguard Worker   size_t operator()(const T& x) const {
515*6777b538SAndroid Build Coastguard Worker     return HashT<typename Deref<T>::Target>()(Deref<T>::deref(x));
516*6777b538SAndroid Build Coastguard Worker   }
517*6777b538SAndroid Build Coastguard Worker };
518*6777b538SAndroid Build Coastguard Worker 
519*6777b538SAndroid Build Coastguard Worker // This tests that upon replacing a duplicate element in the cache with `Put`,
520*6777b538SAndroid Build Coastguard Worker // the element's identity is replaced as well.
TYPED_TEST(LRUCacheSetTest,ReplacementIdentity)521*6777b538SAndroid Build Coastguard Worker TYPED_TEST(LRUCacheSetTest, ReplacementIdentity) {
522*6777b538SAndroid Build Coastguard Worker   using Item = RefCountedData<std::string>;
523*6777b538SAndroid Build Coastguard Worker   using Ptr = scoped_refptr<Item>;
524*6777b538SAndroid Build Coastguard Worker 
525*6777b538SAndroid Build Coastguard Worker   // Helper to create the correct type of base::*LRUCacheSet, since they have
526*6777b538SAndroid Build Coastguard Worker   // different template arguments.
527*6777b538SAndroid Build Coastguard Worker   constexpr auto kCreateCache = []() {
528*6777b538SAndroid Build Coastguard Worker     if constexpr (std::is_same_v<TypeParam, LRUCacheSetTemplate>) {
529*6777b538SAndroid Build Coastguard Worker       using Cache = typename TypeParam::template Type<Ptr, DerefCompare<Ptr>>;
530*6777b538SAndroid Build Coastguard Worker       return Cache(Cache::NO_AUTO_EVICT);
531*6777b538SAndroid Build Coastguard Worker     } else if constexpr (std::is_same_v<TypeParam,
532*6777b538SAndroid Build Coastguard Worker                                         HashingLRUCacheSetTemplate>) {
533*6777b538SAndroid Build Coastguard Worker       using Cache = typename TypeParam::template Type<Ptr, DerefHash<Ptr>,
534*6777b538SAndroid Build Coastguard Worker                                                       DerefEqual<Ptr>>;
535*6777b538SAndroid Build Coastguard Worker       return Cache(Cache::NO_AUTO_EVICT);
536*6777b538SAndroid Build Coastguard Worker     } else {
537*6777b538SAndroid Build Coastguard Worker       static_assert(!sizeof(TypeParam),
538*6777b538SAndroid Build Coastguard Worker                     "This test was only written to support "
539*6777b538SAndroid Build Coastguard Worker                     "`LRUCacheSetTemplate` and `HashingLRUCacheSetTemplate`");
540*6777b538SAndroid Build Coastguard Worker     }
541*6777b538SAndroid Build Coastguard Worker   };
542*6777b538SAndroid Build Coastguard Worker 
543*6777b538SAndroid Build Coastguard Worker   auto cache = kCreateCache();
544*6777b538SAndroid Build Coastguard Worker   cache.Put(MakeRefCounted<Item>("Hello"));
545*6777b538SAndroid Build Coastguard Worker   cache.Put(MakeRefCounted<Item>("world"));
546*6777b538SAndroid Build Coastguard Worker   cache.Put(MakeRefCounted<Item>("foo"));
547*6777b538SAndroid Build Coastguard Worker   cache.Put(MakeRefCounted<Item>("bar"));
548*6777b538SAndroid Build Coastguard Worker 
549*6777b538SAndroid Build Coastguard Worker   // Insert a duplicate element
550*6777b538SAndroid Build Coastguard Worker   {
551*6777b538SAndroid Build Coastguard Worker     auto foo = MakeRefCounted<Item>("foo");
552*6777b538SAndroid Build Coastguard Worker     const auto* new_foo_addr = foo.get();
553*6777b538SAndroid Build Coastguard Worker     const auto* old_foo_addr = cache.Peek(foo)->get();
554*6777b538SAndroid Build Coastguard Worker     auto iter = cache.Put(std::move(foo));
555*6777b538SAndroid Build Coastguard Worker     EXPECT_EQ(iter->get(), new_foo_addr);
556*6777b538SAndroid Build Coastguard Worker     EXPECT_NE(iter->get(), old_foo_addr);
557*6777b538SAndroid Build Coastguard Worker   }
558*6777b538SAndroid Build Coastguard Worker 
559*6777b538SAndroid Build Coastguard Worker   // Iterate from oldest to newest
560*6777b538SAndroid Build Coastguard Worker   auto r_iter = cache.rbegin();
561*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ((*r_iter)->data, "Hello");
562*6777b538SAndroid Build Coastguard Worker   ++r_iter;
563*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ((*r_iter)->data, "world");
564*6777b538SAndroid Build Coastguard Worker   ++r_iter;
565*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ((*r_iter)->data, "bar");
566*6777b538SAndroid Build Coastguard Worker   ++r_iter;
567*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ((*r_iter)->data, "foo");
568*6777b538SAndroid Build Coastguard Worker   ++r_iter;
569*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(r_iter, cache.rend());
570*6777b538SAndroid Build Coastguard Worker 
571*6777b538SAndroid Build Coastguard Worker   // Iterate from newest to oldest
572*6777b538SAndroid Build Coastguard Worker   auto iter = cache.begin();
573*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ((*iter)->data, "foo");
574*6777b538SAndroid Build Coastguard Worker   ++iter;
575*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ((*iter)->data, "bar");
576*6777b538SAndroid Build Coastguard Worker   ++iter;
577*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ((*iter)->data, "world");
578*6777b538SAndroid Build Coastguard Worker   ++iter;
579*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ((*iter)->data, "Hello");
580*6777b538SAndroid Build Coastguard Worker   ++iter;
581*6777b538SAndroid Build Coastguard Worker   EXPECT_EQ(iter, cache.end());
582*6777b538SAndroid Build Coastguard Worker }
583*6777b538SAndroid Build Coastguard Worker 
584*6777b538SAndroid Build Coastguard Worker #if BUILDFLAG(ENABLE_BASE_TRACING)
TYPED_TEST(LRUCacheTest,EstimateMemory)585*6777b538SAndroid Build Coastguard Worker TYPED_TEST(LRUCacheTest, EstimateMemory) {
586*6777b538SAndroid Build Coastguard Worker   typedef typename TypeParam::template Type<std::string, int> Cache;
587*6777b538SAndroid Build Coastguard Worker   Cache cache(10);
588*6777b538SAndroid Build Coastguard Worker 
589*6777b538SAndroid Build Coastguard Worker   const std::string key(100u, 'a');
590*6777b538SAndroid Build Coastguard Worker   cache.Put(key, 1);
591*6777b538SAndroid Build Coastguard Worker 
592*6777b538SAndroid Build Coastguard Worker   EXPECT_GT(trace_event::EstimateMemoryUsage(cache),
593*6777b538SAndroid Build Coastguard Worker             trace_event::EstimateMemoryUsage(key));
594*6777b538SAndroid Build Coastguard Worker }
595*6777b538SAndroid Build Coastguard Worker #endif  // BUILDFLAG(ENABLE_BASE_TRACING)
596*6777b538SAndroid Build Coastguard Worker 
597*6777b538SAndroid Build Coastguard Worker }  // namespace base
598