1 // Copyright 2014 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 #include "base/basictypes.h"
6 #include "base/logging.h"
7 #include "net/disk_cache/blockfile/addr.h"
8 #include "net/disk_cache/blockfile/disk_format_v3.h"
9 #include "net/disk_cache/blockfile/index_table_v3.h"
10 #include "testing/gtest/include/gtest/gtest.h"
11
12 using disk_cache::EntryCell;
13 using disk_cache::IndexCell;
14 using disk_cache::IndexTable;
15 using disk_cache::IndexTableInitData;
16
17 namespace {
18
GetChecksum(const IndexCell & source)19 int GetChecksum(const IndexCell& source) {
20 // Only the cell pointer is relevant.
21 disk_cache::Addr addr;
22 IndexCell* cell = const_cast<IndexCell*>(&source);
23 EntryCell entry = EntryCell::GetEntryCellForTest(0, 0, addr, cell, false);
24
25 IndexCell result;
26 entry.SerializaForTest(&result);
27 return result.last_part >> 6;
28 }
29
30 class MockIndexBackend : public disk_cache::IndexTableBackend {
31 public:
MockIndexBackend()32 MockIndexBackend() : grow_called_(false), buffer_len_(-1) {}
~MockIndexBackend()33 virtual ~MockIndexBackend() {}
34
grow_called() const35 bool grow_called() const { return grow_called_; }
buffer_len() const36 int buffer_len() const { return buffer_len_; }
37
GrowIndex()38 virtual void GrowIndex() OVERRIDE { grow_called_ = true; }
SaveIndex(net::IOBuffer * buffer,int buffer_len)39 virtual void SaveIndex(net::IOBuffer* buffer, int buffer_len) OVERRIDE {
40 buffer_len_ = buffer_len;
41 }
DeleteCell(EntryCell cell)42 virtual void DeleteCell(EntryCell cell) OVERRIDE {}
FixCell(EntryCell cell)43 virtual void FixCell(EntryCell cell) OVERRIDE {}
44
45 private:
46 bool grow_called_;
47 int buffer_len_;
48 };
49
50 class TestCacheTables {
51 public:
52 // |num_entries| is the capacity of the main table. The extra table is half
53 // the size of the main table.
54 explicit TestCacheTables(int num_entries);
~TestCacheTables()55 ~TestCacheTables() {}
56
57 void GetInitData(IndexTableInitData* result);
58 void CopyFrom(const TestCacheTables& other);
start_time() const59 base::Time start_time() const { return start_time_; }
60
61 private:
62 scoped_ptr<uint64[]> main_bitmap_;
63 scoped_ptr<disk_cache::IndexBucket[]> main_table_;
64 scoped_ptr<disk_cache::IndexBucket[]> extra_table_;
65 base::Time start_time_;
66 int num_bitmap_bytes_;
67
68 DISALLOW_COPY_AND_ASSIGN(TestCacheTables);
69 };
70
TestCacheTables(int num_entries)71 TestCacheTables::TestCacheTables(int num_entries) {
72 DCHECK_GE(num_entries, 1024);
73 DCHECK_EQ(num_entries, num_entries / 1024 * 1024);
74 main_table_.reset(new disk_cache::IndexBucket[num_entries]);
75 extra_table_.reset(new disk_cache::IndexBucket[num_entries / 2]);
76 memset(main_table_.get(), 0, num_entries * sizeof(*main_table_.get()));
77 memset(extra_table_.get(), 0, num_entries / 2 * sizeof(*extra_table_.get()));
78
79 // We allow IndexBitmap smaller than a page because the code should not really
80 // depend on that.
81 num_bitmap_bytes_ = (num_entries + num_entries / 2) / 8;
82 size_t required_size = sizeof(disk_cache::IndexHeaderV3) + num_bitmap_bytes_;
83 main_bitmap_.reset(new uint64[required_size / sizeof(uint64)]);
84 memset(main_bitmap_.get(), 0, required_size);
85
86 disk_cache::IndexHeaderV3* header =
87 reinterpret_cast<disk_cache::IndexHeaderV3*>(main_bitmap_.get());
88
89 header->magic = disk_cache::kIndexMagicV3;
90 header->version = disk_cache::kVersion3;
91 header->table_len = num_entries + num_entries / 2;
92 header->max_bucket = num_entries / 4 - 1;
93
94 start_time_ = base::Time::Now();
95 header->create_time = start_time_.ToInternalValue();
96 header->base_time =
97 (start_time_ - base::TimeDelta::FromDays(20)).ToInternalValue();
98
99 if (num_entries < 64 * 1024)
100 header->flags = disk_cache::SMALL_CACHE;
101 }
102
GetInitData(IndexTableInitData * result)103 void TestCacheTables::GetInitData(IndexTableInitData* result) {
104 result->index_bitmap =
105 reinterpret_cast<disk_cache::IndexBitmap*>(main_bitmap_.get());
106
107 result->main_table = main_table_.get();
108 result->extra_table = extra_table_.get();
109
110 result->backup_header.reset(new disk_cache::IndexHeaderV3);
111 memcpy(result->backup_header.get(), result->index_bitmap,
112 sizeof(result->index_bitmap->header));
113
114 result->backup_bitmap.reset(new uint32[num_bitmap_bytes_ / sizeof(uint32)]);
115 memcpy(result->backup_bitmap.get(), result->index_bitmap->bitmap,
116 num_bitmap_bytes_);
117 }
118
CopyFrom(const TestCacheTables & other)119 void TestCacheTables::CopyFrom(const TestCacheTables& other) {
120 disk_cache::IndexBitmap* this_bitmap =
121 reinterpret_cast<disk_cache::IndexBitmap*>(main_bitmap_.get());
122 disk_cache::IndexBitmap* other_bitmap =
123 reinterpret_cast<disk_cache::IndexBitmap*>(other.main_bitmap_.get());
124
125 DCHECK_GE(this_bitmap->header.table_len, other_bitmap->header.table_len);
126 DCHECK_GE(num_bitmap_bytes_, other.num_bitmap_bytes_);
127
128 memcpy(this_bitmap->bitmap, other_bitmap->bitmap, other.num_bitmap_bytes_);
129
130 int main_table_buckets = (other_bitmap->header.table_len * 2 / 3) / 4;
131 int extra_table_buckets = (other_bitmap->header.table_len * 1 / 3) / 4;
132 memcpy(main_table_.get(), other.main_table_.get(),
133 main_table_buckets * sizeof(disk_cache::IndexBucket));
134 memcpy(extra_table_.get(), other.extra_table_.get(),
135 extra_table_buckets * sizeof(disk_cache::IndexBucket));
136
137 this_bitmap->header.num_entries = other_bitmap->header.num_entries;
138 this_bitmap->header.used_cells = other_bitmap->header.used_cells;
139 this_bitmap->header.max_bucket = other_bitmap->header.max_bucket;
140 this_bitmap->header.create_time = other_bitmap->header.create_time;
141 this_bitmap->header.base_time = other_bitmap->header.base_time;
142 this_bitmap->header.flags = other_bitmap->header.flags;
143 start_time_ = other.start_time_;
144 }
145
146 } // namespace
147
TEST(DiskCacheIndexTable,EntryCell)148 TEST(DiskCacheIndexTable, EntryCell) {
149 uint32 hash = 0x55aa6699;
150 disk_cache::Addr addr(disk_cache::BLOCK_ENTRIES, 1, 5, 0x4531);
151 bool small_table = true;
152 int cell_num = 88;
153 int reuse = 6;
154 int timestamp = 123456;
155 disk_cache::EntryState state = disk_cache::ENTRY_MODIFIED;
156 disk_cache::EntryGroup group = disk_cache::ENTRY_HIGH_USE;
157
158 for (int i = 0; i < 4; i++) {
159 SCOPED_TRACE(i);
160 EntryCell entry = EntryCell::GetEntryCellForTest(cell_num, hash, addr, NULL,
161 small_table);
162 EXPECT_EQ(disk_cache::ENTRY_NO_USE, entry.GetGroup());
163 EXPECT_EQ(disk_cache::ENTRY_NEW, entry.GetState());
164
165 entry.SetGroup(group);
166 entry.SetState(state);
167 entry.SetReuse(reuse);
168 entry.SetTimestamp(timestamp);
169
170 EXPECT_TRUE(entry.IsValid());
171 EXPECT_EQ(hash, entry.hash());
172 EXPECT_EQ(cell_num, entry.cell_num());
173 EXPECT_EQ(addr.value(), entry.GetAddress().value());
174
175 EXPECT_EQ(group, entry.GetGroup());
176 EXPECT_EQ(state, entry.GetState());
177 EXPECT_EQ(reuse, entry.GetReuse());
178 EXPECT_EQ(timestamp, entry.GetTimestamp());
179
180 // Store the data and read it again.
181 IndexCell cell;
182 entry.SerializaForTest(&cell);
183
184 EntryCell entry2 = EntryCell::GetEntryCellForTest(cell_num, hash, addr,
185 &cell, small_table);
186
187 EXPECT_EQ(addr.value(), entry2.GetAddress().value());
188
189 EXPECT_EQ(group, entry2.GetGroup());
190 EXPECT_EQ(state, entry2.GetState());
191 EXPECT_EQ(reuse, entry2.GetReuse());
192 EXPECT_EQ(timestamp, entry2.GetTimestamp());
193
194 small_table = !small_table;
195 if (i == 1) {
196 hash = ~hash;
197 cell_num *= 5;
198 state = disk_cache::ENTRY_USED;
199 group = disk_cache::ENTRY_EVICTED;
200 addr = disk_cache::Addr(disk_cache::BLOCK_EVICTED, 1, 6, 0x18a5);
201 reuse = 15; // 4 bits
202 timestamp = 0xfffff; // 20 bits.
203 }
204 }
205 }
206
207 // Goes over some significant values for a cell's sum.
TEST(DiskCacheIndexTable,EntryCellSum)208 TEST(DiskCacheIndexTable, EntryCellSum) {
209 IndexCell source;
210 source.Clear();
211 EXPECT_EQ(0, GetChecksum(source));
212
213 source.first_part++;
214 EXPECT_EQ(1, GetChecksum(source));
215
216 source.Clear();
217 source.last_part = 0x80;
218 EXPECT_EQ(0, GetChecksum(source));
219
220 source.last_part = 0x55;
221 EXPECT_EQ(3, GetChecksum(source));
222
223 source.first_part = 0x555555;
224 EXPECT_EQ(2, GetChecksum(source));
225
226 source.last_part = 0;
227 EXPECT_EQ(1, GetChecksum(source));
228
229 source.first_part = GG_UINT64_C(0x8000000080000000);
230 EXPECT_EQ(0, GetChecksum(source));
231
232 source.first_part = GG_UINT64_C(0x4000000040000000);
233 EXPECT_EQ(2, GetChecksum(source));
234
235 source.first_part = GG_UINT64_C(0x200000020000000);
236 EXPECT_EQ(1, GetChecksum(source));
237
238 source.first_part = GG_UINT64_C(0x100000010010000);
239 EXPECT_EQ(3, GetChecksum(source));
240
241 source.first_part = 0x80008000;
242 EXPECT_EQ(0, GetChecksum(source));
243
244 source.first_part = GG_UINT64_C(0x800000008000);
245 EXPECT_EQ(1, GetChecksum(source));
246
247 source.first_part = 0x8080;
248 EXPECT_EQ(0, GetChecksum(source));
249
250 source.first_part = 0x800080;
251 EXPECT_EQ(1, GetChecksum(source));
252
253 source.first_part = 0x88;
254 EXPECT_EQ(0, GetChecksum(source));
255
256 source.first_part = 0x808;
257 EXPECT_EQ(1, GetChecksum(source));
258
259 source.first_part = 0xA;
260 EXPECT_EQ(0, GetChecksum(source));
261
262 source.first_part = 0x22;
263 EXPECT_EQ(1, GetChecksum(source));
264 }
265
TEST(DiskCacheIndexTable,Basics)266 TEST(DiskCacheIndexTable, Basics) {
267 TestCacheTables cache(1024);
268 IndexTableInitData init_data;
269 cache.GetInitData(&init_data);
270
271 IndexTable index(NULL);
272 index.Init(&init_data);
273
274 // Write some entries.
275 disk_cache::CellList entries;
276 for (int i = 0; i < 250; i++) {
277 SCOPED_TRACE(i);
278 uint32 hash = i * i * 1111 + i * 11;
279 disk_cache::Addr addr(disk_cache::BLOCK_ENTRIES, 1, 5, i * 13 + 1);
280 EntryCell entry = index.CreateEntryCell(hash, addr);
281 EXPECT_TRUE(entry.IsValid());
282
283 disk_cache::CellInfo info = { hash, addr };
284 entries.push_back(info);
285 }
286
287 // Read them back.
288 for (size_t i = 0; i < entries.size(); i++) {
289 SCOPED_TRACE(i);
290 uint32 hash = entries[i].hash;
291 disk_cache::Addr addr = entries[i].address;
292
293 disk_cache::EntrySet found_entries = index.LookupEntries(hash);
294 ASSERT_EQ(1u, found_entries.cells.size());
295 EXPECT_TRUE(found_entries.cells[0].IsValid());
296 EXPECT_EQ(hash, found_entries.cells[0].hash());
297 EXPECT_EQ(addr.value(), found_entries.cells[0].GetAddress().value());
298
299 EntryCell entry = index.FindEntryCell(hash, addr);
300 EXPECT_TRUE(entry.IsValid());
301 EXPECT_EQ(hash, entry.hash());
302 EXPECT_EQ(addr.value(), entry.GetAddress().value());
303
304 // Delete the first 100 entries.
305 if (i < 100)
306 index.SetSate(hash, addr, disk_cache::ENTRY_DELETED);
307 }
308
309 // See what we have now.
310 for (size_t i = 0; i < entries.size(); i++) {
311 SCOPED_TRACE(i);
312 uint32 hash = entries[i].hash;
313 disk_cache::Addr addr = entries[i].address;
314
315 disk_cache::EntrySet found_entries = index.LookupEntries(hash);
316 if (i < 100) {
317 EXPECT_EQ(0u, found_entries.cells.size());
318 } else {
319 ASSERT_EQ(1u, found_entries.cells.size());
320 EXPECT_TRUE(found_entries.cells[0].IsValid());
321 EXPECT_EQ(hash, found_entries.cells[0].hash());
322 EXPECT_EQ(addr.value(), found_entries.cells[0].GetAddress().value());
323 }
324 }
325 }
326
327 // Tests handling of multiple entries with the same hash.
TEST(DiskCacheIndexTable,SameHash)328 TEST(DiskCacheIndexTable, SameHash) {
329 TestCacheTables cache(1024);
330 IndexTableInitData init_data;
331 cache.GetInitData(&init_data);
332
333 IndexTable index(NULL);
334 index.Init(&init_data);
335
336 disk_cache::CellList entries;
337 uint32 hash = 0x55aa55bb;
338 for (int i = 0; i < 6; i++) {
339 SCOPED_TRACE(i);
340 disk_cache::Addr addr(disk_cache::BLOCK_ENTRIES, 1, 5, i * 13 + 1);
341 EntryCell entry = index.CreateEntryCell(hash, addr);
342 EXPECT_TRUE(entry.IsValid());
343
344 disk_cache::CellInfo info = { hash, addr };
345 entries.push_back(info);
346 }
347
348 disk_cache::EntrySet found_entries = index.LookupEntries(hash);
349 EXPECT_EQ(0, found_entries.evicted_count);
350 ASSERT_EQ(6u, found_entries.cells.size());
351
352 for (size_t i = 0; i < found_entries.cells.size(); i++) {
353 SCOPED_TRACE(i);
354 EXPECT_EQ(entries[i].address, found_entries.cells[i].GetAddress());
355 }
356
357 // Now verify handling of entries on different states.
358 index.SetSate(hash, entries[0].address, disk_cache::ENTRY_DELETED);
359 index.SetSate(hash, entries[1].address, disk_cache::ENTRY_DELETED);
360 index.SetSate(hash, entries[2].address, disk_cache::ENTRY_USED);
361 index.SetSate(hash, entries[3].address, disk_cache::ENTRY_USED);
362 index.SetSate(hash, entries[4].address, disk_cache::ENTRY_USED);
363
364 found_entries = index.LookupEntries(hash);
365 EXPECT_EQ(0, found_entries.evicted_count);
366 ASSERT_EQ(4u, found_entries.cells.size());
367
368 index.SetSate(hash, entries[3].address, disk_cache::ENTRY_OPEN);
369 index.SetSate(hash, entries[4].address, disk_cache::ENTRY_OPEN);
370
371 found_entries = index.LookupEntries(hash);
372 EXPECT_EQ(0, found_entries.evicted_count);
373 ASSERT_EQ(4u, found_entries.cells.size());
374
375 index.SetSate(hash, entries[4].address, disk_cache::ENTRY_MODIFIED);
376
377 found_entries = index.LookupEntries(hash);
378 EXPECT_EQ(0, found_entries.evicted_count);
379 ASSERT_EQ(4u, found_entries.cells.size());
380
381 index.SetSate(hash, entries[1].address, disk_cache::ENTRY_FREE);
382
383 found_entries = index.LookupEntries(hash);
384 EXPECT_EQ(0, found_entries.evicted_count);
385 ASSERT_EQ(4u, found_entries.cells.size());
386
387 // FindEntryCell should not see deleted entries.
388 EntryCell entry = index.FindEntryCell(hash, entries[0].address);
389 EXPECT_FALSE(entry.IsValid());
390
391 // A free entry is gone.
392 entry = index.FindEntryCell(hash, entries[1].address);
393 EXPECT_FALSE(entry.IsValid());
394
395 // Locate a used entry, and evict it. This is not really a correct operation
396 // in that an existing cell doesn't transition to evicted; instead a new cell
397 // for the evicted entry (on a different block file) should be created. Still,
398 // at least evicted_count would be valid.
399 entry = index.FindEntryCell(hash, entries[2].address);
400 EXPECT_TRUE(entry.IsValid());
401 entry.SetGroup(disk_cache::ENTRY_EVICTED);
402 index.Save(&entry);
403
404 found_entries = index.LookupEntries(hash);
405 EXPECT_EQ(1, found_entries.evicted_count);
406 ASSERT_EQ(4u, found_entries.cells.size());
407
408 // Now use the proper way to get an evicted entry.
409 disk_cache::Addr addr2(disk_cache::BLOCK_EVICTED, 1, 6, 6); // Any address.
410 entry = index.CreateEntryCell(hash, addr2);
411 EXPECT_TRUE(entry.IsValid());
412 EXPECT_EQ(disk_cache::ENTRY_EVICTED, entry.GetGroup());
413
414 found_entries = index.LookupEntries(hash);
415 EXPECT_EQ(2, found_entries.evicted_count);
416 ASSERT_EQ(5u, found_entries.cells.size());
417 }
418
TEST(DiskCacheIndexTable,Timestamps)419 TEST(DiskCacheIndexTable, Timestamps) {
420 TestCacheTables cache(1024);
421 IndexTableInitData init_data;
422 cache.GetInitData(&init_data);
423
424 IndexTable index(NULL);
425 index.Init(&init_data);
426
427 // The granularity should be 1 minute.
428 int timestamp1 = index.CalculateTimestamp(cache.start_time());
429 int timestamp2 = index.CalculateTimestamp(cache.start_time() +
430 base::TimeDelta::FromSeconds(59));
431 EXPECT_EQ(timestamp1, timestamp2);
432
433 int timestamp3 = index.CalculateTimestamp(cache.start_time() +
434 base::TimeDelta::FromSeconds(61));
435 EXPECT_EQ(timestamp1 + 1, timestamp3);
436
437 int timestamp4 = index.CalculateTimestamp(cache.start_time() +
438 base::TimeDelta::FromSeconds(119));
439 EXPECT_EQ(timestamp1 + 1, timestamp4);
440
441 int timestamp5 = index.CalculateTimestamp(cache.start_time() +
442 base::TimeDelta::FromSeconds(121));
443 EXPECT_EQ(timestamp1 + 2, timestamp5);
444
445 int timestamp6 = index.CalculateTimestamp(cache.start_time() -
446 base::TimeDelta::FromSeconds(30));
447 EXPECT_EQ(timestamp1 - 1, timestamp6);
448
449 // The base should be 20 days in the past.
450 int timestamp7 = index.CalculateTimestamp(cache.start_time() -
451 base::TimeDelta::FromDays(20));
452 int timestamp8 = index.CalculateTimestamp(cache.start_time() -
453 base::TimeDelta::FromDays(35));
454 EXPECT_EQ(timestamp7, timestamp8);
455 EXPECT_EQ(0, timestamp8);
456
457 int timestamp9 = index.CalculateTimestamp(cache.start_time() -
458 base::TimeDelta::FromDays(19));
459 EXPECT_NE(0, timestamp9);
460 }
461
462 // Tests GetOldest and GetNextCells.
TEST(DiskCacheIndexTable,Iterations)463 TEST(DiskCacheIndexTable, Iterations) {
464 TestCacheTables cache(1024);
465 IndexTableInitData init_data;
466 cache.GetInitData(&init_data);
467
468 IndexTable index(NULL);
469 index.Init(&init_data);
470
471 base::Time time = cache.start_time();
472
473 // Write some entries.
474 disk_cache::CellList entries;
475 for (int i = 0; i < 44; i++) {
476 SCOPED_TRACE(i);
477 uint32 hash = i; // The entries will be ordered on the table.
478 disk_cache::Addr addr(disk_cache::BLOCK_ENTRIES, 1, 5, i * 13 + 1);
479 if (i < 10 || i == 40)
480 addr = disk_cache::Addr(disk_cache::BLOCK_EVICTED, 1, 6, i * 13 + 1);
481
482 EntryCell entry = index.CreateEntryCell(hash, addr);
483 EXPECT_TRUE(entry.IsValid());
484
485 disk_cache::CellInfo info = { hash, addr };
486 entries.push_back(info);
487
488 if (i < 10 || i == 40) {
489 // Do nothing. These are ENTRY_EVICTED by default.
490 } else if (i < 20 || i == 41) {
491 entry.SetGroup(disk_cache::ENTRY_HIGH_USE);
492 index.Save(&entry);
493 } else if (i < 30 || i == 42) {
494 entry.SetGroup(disk_cache::ENTRY_LOW_USE);
495 index.Save(&entry);
496 }
497
498 // Entries [30,39] and 43 are marked as ENTRY_NO_USE (the default).
499
500 if (!(i % 10))
501 time += base::TimeDelta::FromMinutes(1);
502
503 index.UpdateTime(hash, addr, time);
504 }
505
506 // Get the oldest entries of each group.
507 disk_cache::IndexIterator no_use, low_use, high_use;
508 index.GetOldest(&no_use, &low_use, &high_use);
509 ASSERT_EQ(10u, no_use.cells.size());
510 ASSERT_EQ(10u, low_use.cells.size());
511 ASSERT_EQ(10u, high_use.cells.size());
512
513 EXPECT_EQ(entries[10].hash, high_use.cells[0].hash);
514 EXPECT_EQ(entries[19].hash, high_use.cells[9].hash);
515 EXPECT_EQ(entries[20].hash, low_use.cells[0].hash);
516 EXPECT_EQ(entries[29].hash, low_use.cells[9].hash);
517 EXPECT_EQ(entries[30].hash, no_use.cells[0].hash);
518 EXPECT_EQ(entries[39].hash, no_use.cells[9].hash);
519
520 // Now start an iteration from the head (most recent entry).
521 disk_cache::IndexIterator iterator;
522 iterator.timestamp = index.CalculateTimestamp(time) + 1;
523 iterator.forward = false; // Back in time.
524
525 ASSERT_TRUE(index.GetNextCells(&iterator));
526 ASSERT_EQ(3u, iterator.cells.size());
527 EXPECT_EQ(entries[41].hash, iterator.cells[0].hash);
528 EXPECT_EQ(entries[42].hash, iterator.cells[1].hash);
529 EXPECT_EQ(entries[43].hash, iterator.cells[2].hash);
530
531 ASSERT_TRUE(index.GetNextCells(&iterator));
532 ASSERT_EQ(10u, iterator.cells.size());
533 EXPECT_EQ(entries[30].hash, iterator.cells[0].hash);
534 EXPECT_EQ(entries[39].hash, iterator.cells[9].hash);
535
536 ASSERT_TRUE(index.GetNextCells(&iterator));
537 ASSERT_EQ(10u, iterator.cells.size());
538 EXPECT_EQ(entries[20].hash, iterator.cells[0].hash);
539 EXPECT_EQ(entries[29].hash, iterator.cells[9].hash);
540
541 ASSERT_TRUE(index.GetNextCells(&iterator));
542 ASSERT_EQ(10u, iterator.cells.size());
543 EXPECT_EQ(entries[10].hash, iterator.cells[0].hash);
544 EXPECT_EQ(entries[19].hash, iterator.cells[9].hash);
545
546 ASSERT_FALSE(index.GetNextCells(&iterator));
547
548 // Now start an iteration from the tail (oldest entry).
549 iterator.timestamp = 0;
550 iterator.forward = true;
551
552 ASSERT_TRUE(index.GetNextCells(&iterator));
553 ASSERT_EQ(10u, iterator.cells.size());
554 EXPECT_EQ(entries[10].hash, iterator.cells[0].hash);
555 EXPECT_EQ(entries[19].hash, iterator.cells[9].hash);
556
557 ASSERT_TRUE(index.GetNextCells(&iterator));
558 ASSERT_EQ(10u, iterator.cells.size());
559 EXPECT_EQ(entries[20].hash, iterator.cells[0].hash);
560 EXPECT_EQ(entries[29].hash, iterator.cells[9].hash);
561
562 ASSERT_TRUE(index.GetNextCells(&iterator));
563 ASSERT_EQ(10u, iterator.cells.size());
564 EXPECT_EQ(entries[30].hash, iterator.cells[0].hash);
565 EXPECT_EQ(entries[39].hash, iterator.cells[9].hash);
566
567 ASSERT_TRUE(index.GetNextCells(&iterator));
568 ASSERT_EQ(3u, iterator.cells.size());
569 EXPECT_EQ(entries[41].hash, iterator.cells[0].hash);
570 EXPECT_EQ(entries[42].hash, iterator.cells[1].hash);
571 EXPECT_EQ(entries[43].hash, iterator.cells[2].hash);
572 }
573
574 // Tests doubling of the table.
TEST(DiskCacheIndexTable,Doubling)575 TEST(DiskCacheIndexTable, Doubling) {
576 IndexTable index(NULL);
577 int size = 1024;
578 scoped_ptr<TestCacheTables> cache(new TestCacheTables(size));
579 int entry_id = 0;
580 disk_cache::CellList entries;
581
582 // Go from 1024 to 256k cells.
583 for (int resizes = 0; resizes <= 8; resizes++) {
584 scoped_ptr<TestCacheTables> old_cache(cache.Pass());
585 cache.reset(new TestCacheTables(size));
586 cache.get()->CopyFrom(*old_cache.get());
587
588 IndexTableInitData init_data;
589 cache.get()->GetInitData(&init_data);
590 index.Init(&init_data);
591
592 // Write some entries.
593 for (int i = 0; i < 250; i++, entry_id++) {
594 SCOPED_TRACE(entry_id);
595 uint32 hash = entry_id * i * 321 + entry_id * 13;
596 disk_cache::Addr addr(disk_cache::BLOCK_ENTRIES, 1, 5, entry_id * 17 + 1);
597 EntryCell entry = index.CreateEntryCell(hash, addr);
598 EXPECT_TRUE(entry.IsValid());
599
600 disk_cache::CellInfo info = { hash, addr };
601 entries.push_back(info);
602 }
603 size *= 2;
604 }
605
606 // Access all the entries.
607 for (size_t i = 0; i < entries.size(); i++) {
608 SCOPED_TRACE(i);
609 disk_cache::EntrySet found_entries = index.LookupEntries(entries[i].hash);
610 ASSERT_EQ(1u, found_entries.cells.size());
611 EXPECT_TRUE(found_entries.cells[0].IsValid());
612 }
613 }
614
615 // Tests bucket chaining when growing the index.
TEST(DiskCacheIndexTable,BucketChains)616 TEST(DiskCacheIndexTable, BucketChains) {
617 IndexTable index(NULL);
618 int size = 1024;
619 scoped_ptr<TestCacheTables> cache(new TestCacheTables(size));
620 disk_cache::CellList entries;
621
622 IndexTableInitData init_data;
623 cache.get()->GetInitData(&init_data);
624 index.Init(&init_data);
625
626 // Write some entries.
627 for (int i = 0; i < 8; i++) {
628 SCOPED_TRACE(i);
629 uint32 hash = i * 256;
630 disk_cache::Addr addr(disk_cache::BLOCK_ENTRIES, 1, 5, i * 7 + 1);
631 EntryCell entry = index.CreateEntryCell(hash, addr);
632 EXPECT_TRUE(entry.IsValid());
633
634 disk_cache::CellInfo info = { hash, addr };
635 entries.push_back(info);
636 }
637
638 // Double the size.
639 scoped_ptr<TestCacheTables> old_cache(cache.Pass());
640 cache.reset(new TestCacheTables(size * 2));
641 cache.get()->CopyFrom(*old_cache.get());
642
643 cache.get()->GetInitData(&init_data);
644 index.Init(&init_data);
645
646 // Write more entries, starting with the upper half of the table.
647 for (int i = 9; i < 11; i++) {
648 SCOPED_TRACE(i);
649 uint32 hash = i * 256;
650 disk_cache::Addr addr(disk_cache::BLOCK_ENTRIES, 1, 5, i * 7 + 1);
651 EntryCell entry = index.CreateEntryCell(hash, addr);
652 EXPECT_TRUE(entry.IsValid());
653
654 disk_cache::CellInfo info = { hash, addr };
655 entries.push_back(info);
656 }
657
658 // Access all the entries.
659 for (size_t i = 0; i < entries.size(); i++) {
660 SCOPED_TRACE(i);
661 disk_cache::EntrySet found_entries = index.LookupEntries(entries[i].hash);
662 ASSERT_EQ(1u, found_entries.cells.size());
663 EXPECT_TRUE(found_entries.cells[0].IsValid());
664 }
665 }
666
667 // Tests that GrowIndex is called.
TEST(DiskCacheIndexTable,GrowIndex)668 TEST(DiskCacheIndexTable, GrowIndex) {
669 TestCacheTables cache(1024);
670 IndexTableInitData init_data;
671 cache.GetInitData(&init_data);
672 MockIndexBackend backend;
673
674 IndexTable index(&backend);
675 index.Init(&init_data);
676
677 // Write some entries.
678 for (int i = 0; i < 512; i++) {
679 SCOPED_TRACE(i);
680 uint32 hash = 0;
681 disk_cache::Addr addr(disk_cache::BLOCK_ENTRIES, 1, 5, i + 1);
682 EntryCell entry = index.CreateEntryCell(hash, addr);
683 EXPECT_TRUE(entry.IsValid());
684 }
685
686 EXPECT_TRUE(backend.grow_called());
687 }
688
TEST(DiskCacheIndexTable,SaveIndex)689 TEST(DiskCacheIndexTable, SaveIndex) {
690 TestCacheTables cache(1024);
691 IndexTableInitData init_data;
692 cache.GetInitData(&init_data);
693 MockIndexBackend backend;
694
695 IndexTable index(&backend);
696 index.Init(&init_data);
697
698 uint32 hash = 0;
699 disk_cache::Addr addr(disk_cache::BLOCK_ENTRIES, 1, 5, 6);
700 EntryCell entry = index.CreateEntryCell(hash, addr);
701 EXPECT_TRUE(entry.IsValid());
702
703 index.OnBackupTimer();
704 int expected = (1024 + 512) / 8 + sizeof(disk_cache::IndexHeaderV3);
705 EXPECT_EQ(expected, backend.buffer_len());
706 }
707