blob: ff4412ccd776b20502dd120f6fba9ef6036174f7 [file] [log] [blame]
// Copyright (c) 2018 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 "gpu/command_buffer/service/service_transfer_cache.h"
#include "cc/paint/raw_memory_transfer_cache_entry.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace gpu {
namespace {
constexpr int kDecoderId = 2;
constexpr auto kEntryType = cc::TransferCacheEntryType::kRawMemory;
std::unique_ptr<cc::ServiceTransferCacheEntry> CreateEntry(size_t size) {
auto entry = std::make_unique<cc::ServiceRawMemoryTransferCacheEntry>();
std::vector<uint8_t> data(size, 0u);
entry->Deserialize(nullptr, data);
return entry;
}
TEST(ServiceTransferCacheTest, EnforcesOnPurgeMemory) {
ServiceTransferCache cache;
uint32_t entry_id = 0u;
size_t entry_size = 1024u;
uint32_t number_of_entry = 4u;
cache.CreateLocalEntry(
ServiceTransferCache::EntryKey(kDecoderId, kEntryType, ++entry_id),
CreateEntry(entry_size));
EXPECT_EQ(cache.cache_size_for_testing(), entry_size);
cache.PurgeMemory(
base::MemoryPressureListener::MEMORY_PRESSURE_LEVEL_CRITICAL);
EXPECT_EQ(cache.cache_size_for_testing(), 0u);
cache.SetCacheSizeLimitForTesting(entry_size * number_of_entry);
// Create 4 entries, all in the cache.
for (uint32_t i = 0; i < number_of_entry; i++) {
cache.CreateLocalEntry(
ServiceTransferCache::EntryKey(kDecoderId, kEntryType, ++entry_id),
CreateEntry(entry_size));
EXPECT_EQ(cache.cache_size_for_testing(), entry_size * (i + 1));
}
// The 5th entry creates successfully. But the 1st entry will be purged due to
// cache limits.
cache.CreateLocalEntry(
ServiceTransferCache::EntryKey(kDecoderId, kEntryType, ++entry_id),
CreateEntry(entry_size));
EXPECT_EQ(cache.cache_size_for_testing(), entry_size * 4);
cache.PurgeMemory(
base::MemoryPressureListener::MEMORY_PRESSURE_LEVEL_MODERATE);
// Only 1/4 of cache limits remains.
EXPECT_EQ(cache.cache_size_for_testing(), entry_size);
}
TEST(ServiceTransferCache, MultipleDecoderUse) {
ServiceTransferCache cache;
const uint32_t entry_id = 0u;
const size_t entry_size = 1024u;
// Decoder 1 entry.
int decoder1 = 1;
auto decoder_1_entry = CreateEntry(entry_size);
auto* decoder_1_entry_ptr = decoder_1_entry.get();
cache.CreateLocalEntry(
ServiceTransferCache::EntryKey(decoder1, kEntryType, entry_id),
std::move(decoder_1_entry));
// Decoder 2 entry.
int decoder2 = 2;
auto decoder_2_entry = CreateEntry(entry_size);
auto* decoder_2_entry_ptr = decoder_2_entry.get();
cache.CreateLocalEntry(
ServiceTransferCache::EntryKey(decoder2, kEntryType, entry_id),
std::move(decoder_2_entry));
EXPECT_EQ(decoder_1_entry_ptr, cache.GetEntry(ServiceTransferCache::EntryKey(
decoder1, kEntryType, entry_id)));
EXPECT_EQ(decoder_2_entry_ptr, cache.GetEntry(ServiceTransferCache::EntryKey(
decoder2, kEntryType, entry_id)));
}
TEST(ServiceTransferCache, DeleteEntriesForDecoder) {
ServiceTransferCache cache;
const size_t entry_size = 1024u;
const size_t cache_size = 4 * entry_size;
cache.SetCacheSizeLimitForTesting(cache_size);
// Add 2 entries for decoder 1.
cache.CreateLocalEntry(ServiceTransferCache::EntryKey(1, kEntryType, 1),
CreateEntry(entry_size));
cache.CreateLocalEntry(ServiceTransferCache::EntryKey(1, kEntryType, 2),
CreateEntry(entry_size));
// Add 2 entries for decoder 2.
cache.CreateLocalEntry(ServiceTransferCache::EntryKey(2, kEntryType, 1),
CreateEntry(entry_size));
cache.CreateLocalEntry(ServiceTransferCache::EntryKey(2, kEntryType, 2),
CreateEntry(entry_size));
// Erase all entries for decoder 1.
EXPECT_EQ(cache.entries_count_for_testing(), 4u);
cache.DeleteAllEntriesForDecoder(1);
EXPECT_EQ(cache.entries_count_for_testing(), 2u);
EXPECT_NE(cache.GetEntry(ServiceTransferCache::EntryKey(2, kEntryType, 1)),
nullptr);
EXPECT_NE(cache.GetEntry(ServiceTransferCache::EntryKey(2, kEntryType, 2)),
nullptr);
}
} // namespace
} // namespace gpu