xref: /aosp_15_r20/external/skia/tests/ImageCacheTest.cpp (revision c8dee2aa9b3f27cf6c858bd81872bdeb2c07ed17)
1*c8dee2aaSAndroid Build Coastguard Worker  /*
2*c8dee2aaSAndroid Build Coastguard Worker  * Copyright 2013 Google Inc.
3*c8dee2aaSAndroid Build Coastguard Worker  *
4*c8dee2aaSAndroid Build Coastguard Worker  * Use of this source code is governed by a BSD-style license that can be
5*c8dee2aaSAndroid Build Coastguard Worker  * found in the LICENSE file.
6*c8dee2aaSAndroid Build Coastguard Worker  */
7*c8dee2aaSAndroid Build Coastguard Worker 
8*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkRefCnt.h"
9*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkTypes.h"
10*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/chromium/SkDiscardableMemory.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "src/core/SkResourceCache.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "src/lazy/SkDiscardableMemoryPool.h"
13*c8dee2aaSAndroid Build Coastguard Worker #include "tests/Test.h"
14*c8dee2aaSAndroid Build Coastguard Worker 
15*c8dee2aaSAndroid Build Coastguard Worker #include <cstddef>
16*c8dee2aaSAndroid Build Coastguard Worker #include <cstdint>
17*c8dee2aaSAndroid Build Coastguard Worker 
18*c8dee2aaSAndroid Build Coastguard Worker namespace {
19*c8dee2aaSAndroid Build Coastguard Worker static void* gGlobalAddress;
20*c8dee2aaSAndroid Build Coastguard Worker struct TestingKey : public SkResourceCache::Key {
21*c8dee2aaSAndroid Build Coastguard Worker     intptr_t    fValue;
22*c8dee2aaSAndroid Build Coastguard Worker 
TestingKey__anonbb93fe480111::TestingKey23*c8dee2aaSAndroid Build Coastguard Worker     TestingKey(intptr_t value, uint64_t sharedID = 0) : fValue(value) {
24*c8dee2aaSAndroid Build Coastguard Worker         this->init(&gGlobalAddress, sharedID, sizeof(fValue));
25*c8dee2aaSAndroid Build Coastguard Worker     }
26*c8dee2aaSAndroid Build Coastguard Worker };
27*c8dee2aaSAndroid Build Coastguard Worker struct TestingRec : public SkResourceCache::Rec {
TestingRec__anonbb93fe480111::TestingRec28*c8dee2aaSAndroid Build Coastguard Worker     TestingRec(const TestingKey& key, uint32_t value) : fKey(key), fValue(value) {}
29*c8dee2aaSAndroid Build Coastguard Worker 
30*c8dee2aaSAndroid Build Coastguard Worker     TestingKey  fKey;
31*c8dee2aaSAndroid Build Coastguard Worker     intptr_t    fValue;
32*c8dee2aaSAndroid Build Coastguard Worker 
getKey__anonbb93fe480111::TestingRec33*c8dee2aaSAndroid Build Coastguard Worker     const Key& getKey() const override { return fKey; }
bytesUsed__anonbb93fe480111::TestingRec34*c8dee2aaSAndroid Build Coastguard Worker     size_t bytesUsed() const override { return sizeof(fKey) + sizeof(fValue); }
getCategory__anonbb93fe480111::TestingRec35*c8dee2aaSAndroid Build Coastguard Worker     const char* getCategory() const override { return "test_cache"; }
diagnostic_only_getDiscardable__anonbb93fe480111::TestingRec36*c8dee2aaSAndroid Build Coastguard Worker     SkDiscardableMemory* diagnostic_only_getDiscardable() const override { return nullptr; }
37*c8dee2aaSAndroid Build Coastguard Worker 
Visitor__anonbb93fe480111::TestingRec38*c8dee2aaSAndroid Build Coastguard Worker     static bool Visitor(const SkResourceCache::Rec& baseRec, void* context) {
39*c8dee2aaSAndroid Build Coastguard Worker         const TestingRec& rec = static_cast<const TestingRec&>(baseRec);
40*c8dee2aaSAndroid Build Coastguard Worker         intptr_t* result = (intptr_t*)context;
41*c8dee2aaSAndroid Build Coastguard Worker 
42*c8dee2aaSAndroid Build Coastguard Worker         *result = rec.fValue;
43*c8dee2aaSAndroid Build Coastguard Worker         return true;
44*c8dee2aaSAndroid Build Coastguard Worker     }
45*c8dee2aaSAndroid Build Coastguard Worker };
46*c8dee2aaSAndroid Build Coastguard Worker }  // namespace
47*c8dee2aaSAndroid Build Coastguard Worker 
48*c8dee2aaSAndroid Build Coastguard Worker static const int COUNT = 10;
49*c8dee2aaSAndroid Build Coastguard Worker static const int DIM = 256;
50*c8dee2aaSAndroid Build Coastguard Worker 
test_cache(skiatest::Reporter * reporter,SkResourceCache & cache,bool testPurge)51*c8dee2aaSAndroid Build Coastguard Worker static void test_cache(skiatest::Reporter* reporter, SkResourceCache& cache, bool testPurge) {
52*c8dee2aaSAndroid Build Coastguard Worker     for (int i = 0; i < COUNT; ++i) {
53*c8dee2aaSAndroid Build Coastguard Worker         TestingKey key(i);
54*c8dee2aaSAndroid Build Coastguard Worker         intptr_t value = -1;
55*c8dee2aaSAndroid Build Coastguard Worker 
56*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, !cache.find(key, TestingRec::Visitor, &value));
57*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, -1 == value);
58*c8dee2aaSAndroid Build Coastguard Worker 
59*c8dee2aaSAndroid Build Coastguard Worker         cache.add(new TestingRec(key, i));
60*c8dee2aaSAndroid Build Coastguard Worker 
61*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, cache.find(key, TestingRec::Visitor, &value));
62*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, i == value);
63*c8dee2aaSAndroid Build Coastguard Worker     }
64*c8dee2aaSAndroid Build Coastguard Worker 
65*c8dee2aaSAndroid Build Coastguard Worker     if (testPurge) {
66*c8dee2aaSAndroid Build Coastguard Worker         // stress test, should trigger purges
67*c8dee2aaSAndroid Build Coastguard Worker         for (int i = 0; i < COUNT * 100; ++i) {
68*c8dee2aaSAndroid Build Coastguard Worker             TestingKey key(i);
69*c8dee2aaSAndroid Build Coastguard Worker             cache.add(new TestingRec(key, i));
70*c8dee2aaSAndroid Build Coastguard Worker         }
71*c8dee2aaSAndroid Build Coastguard Worker     }
72*c8dee2aaSAndroid Build Coastguard Worker 
73*c8dee2aaSAndroid Build Coastguard Worker     // test the originals after all that purging
74*c8dee2aaSAndroid Build Coastguard Worker     for (int i = 0; i < COUNT; ++i) {
75*c8dee2aaSAndroid Build Coastguard Worker         intptr_t value;
76*c8dee2aaSAndroid Build Coastguard Worker         (void)cache.find(TestingKey(i), TestingRec::Visitor, &value);
77*c8dee2aaSAndroid Build Coastguard Worker     }
78*c8dee2aaSAndroid Build Coastguard Worker 
79*c8dee2aaSAndroid Build Coastguard Worker     cache.setTotalByteLimit(0);
80*c8dee2aaSAndroid Build Coastguard Worker }
81*c8dee2aaSAndroid Build Coastguard Worker 
test_cache_purge_shared_id(skiatest::Reporter * reporter,SkResourceCache & cache)82*c8dee2aaSAndroid Build Coastguard Worker static void test_cache_purge_shared_id(skiatest::Reporter* reporter, SkResourceCache& cache) {
83*c8dee2aaSAndroid Build Coastguard Worker     for (int i = 0; i < COUNT; ++i) {
84*c8dee2aaSAndroid Build Coastguard Worker         TestingKey key(i, i & 1);   // every other key will have a 1 for its sharedID
85*c8dee2aaSAndroid Build Coastguard Worker         cache.add(new TestingRec(key, i));
86*c8dee2aaSAndroid Build Coastguard Worker     }
87*c8dee2aaSAndroid Build Coastguard Worker 
88*c8dee2aaSAndroid Build Coastguard Worker     // Ensure that everyone is present
89*c8dee2aaSAndroid Build Coastguard Worker     for (int i = 0; i < COUNT; ++i) {
90*c8dee2aaSAndroid Build Coastguard Worker         TestingKey key(i, i & 1);   // every other key will have a 1 for its sharedID
91*c8dee2aaSAndroid Build Coastguard Worker         intptr_t value = -1;
92*c8dee2aaSAndroid Build Coastguard Worker 
93*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, cache.find(key, TestingRec::Visitor, &value));
94*c8dee2aaSAndroid Build Coastguard Worker         REPORTER_ASSERT(reporter, value == i);
95*c8dee2aaSAndroid Build Coastguard Worker     }
96*c8dee2aaSAndroid Build Coastguard Worker 
97*c8dee2aaSAndroid Build Coastguard Worker     // Now purge the ones that had a non-zero sharedID (the odd-indexed ones)
98*c8dee2aaSAndroid Build Coastguard Worker     cache.purgeSharedID(1);
99*c8dee2aaSAndroid Build Coastguard Worker 
100*c8dee2aaSAndroid Build Coastguard Worker     // Ensure that only the even ones are still present
101*c8dee2aaSAndroid Build Coastguard Worker     for (int i = 0; i < COUNT; ++i) {
102*c8dee2aaSAndroid Build Coastguard Worker         TestingKey key(i, i & 1);   // every other key will have a 1 for its sharedID
103*c8dee2aaSAndroid Build Coastguard Worker         intptr_t value = -1;
104*c8dee2aaSAndroid Build Coastguard Worker 
105*c8dee2aaSAndroid Build Coastguard Worker         if (i & 1) {
106*c8dee2aaSAndroid Build Coastguard Worker             REPORTER_ASSERT(reporter, !cache.find(key, TestingRec::Visitor, &value));
107*c8dee2aaSAndroid Build Coastguard Worker         } else {
108*c8dee2aaSAndroid Build Coastguard Worker             REPORTER_ASSERT(reporter, cache.find(key, TestingRec::Visitor, &value));
109*c8dee2aaSAndroid Build Coastguard Worker             REPORTER_ASSERT(reporter, value == i);
110*c8dee2aaSAndroid Build Coastguard Worker         }
111*c8dee2aaSAndroid Build Coastguard Worker     }
112*c8dee2aaSAndroid Build Coastguard Worker }
113*c8dee2aaSAndroid Build Coastguard Worker 
114*c8dee2aaSAndroid Build Coastguard Worker static SkDiscardableMemoryPool* gPool;
pool_factory(size_t bytes)115*c8dee2aaSAndroid Build Coastguard Worker static SkDiscardableMemory* pool_factory(size_t bytes) {
116*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(gPool);
117*c8dee2aaSAndroid Build Coastguard Worker     return gPool->create(bytes);
118*c8dee2aaSAndroid Build Coastguard Worker }
119*c8dee2aaSAndroid Build Coastguard Worker 
DEF_TEST(ImageCache,reporter)120*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(ImageCache, reporter) {
121*c8dee2aaSAndroid Build Coastguard Worker     static const size_t defLimit = DIM * DIM * 4 * COUNT + 1024;    // 1K slop
122*c8dee2aaSAndroid Build Coastguard Worker 
123*c8dee2aaSAndroid Build Coastguard Worker     {
124*c8dee2aaSAndroid Build Coastguard Worker         SkResourceCache cache(defLimit);
125*c8dee2aaSAndroid Build Coastguard Worker         test_cache(reporter, cache, true);
126*c8dee2aaSAndroid Build Coastguard Worker     }
127*c8dee2aaSAndroid Build Coastguard Worker     {
128*c8dee2aaSAndroid Build Coastguard Worker         sk_sp<SkDiscardableMemoryPool> pool(SkDiscardableMemoryPool::Make(defLimit));
129*c8dee2aaSAndroid Build Coastguard Worker         gPool = pool.get();
130*c8dee2aaSAndroid Build Coastguard Worker         SkResourceCache cache(pool_factory);
131*c8dee2aaSAndroid Build Coastguard Worker         test_cache(reporter, cache, true);
132*c8dee2aaSAndroid Build Coastguard Worker     }
133*c8dee2aaSAndroid Build Coastguard Worker     {
134*c8dee2aaSAndroid Build Coastguard Worker         SkResourceCache cache(SkDiscardableMemory::Create);
135*c8dee2aaSAndroid Build Coastguard Worker         test_cache(reporter, cache, false);
136*c8dee2aaSAndroid Build Coastguard Worker     }
137*c8dee2aaSAndroid Build Coastguard Worker     {
138*c8dee2aaSAndroid Build Coastguard Worker         SkResourceCache cache(defLimit);
139*c8dee2aaSAndroid Build Coastguard Worker         test_cache_purge_shared_id(reporter, cache);
140*c8dee2aaSAndroid Build Coastguard Worker     }
141*c8dee2aaSAndroid Build Coastguard Worker }
142*c8dee2aaSAndroid Build Coastguard Worker 
DEF_TEST(ImageCache_doubleAdd,r)143*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(ImageCache_doubleAdd, r) {
144*c8dee2aaSAndroid Build Coastguard Worker     // Adding the same key twice should be safe.
145*c8dee2aaSAndroid Build Coastguard Worker     SkResourceCache cache(4096);
146*c8dee2aaSAndroid Build Coastguard Worker 
147*c8dee2aaSAndroid Build Coastguard Worker     TestingKey key(1);
148*c8dee2aaSAndroid Build Coastguard Worker 
149*c8dee2aaSAndroid Build Coastguard Worker     cache.add(new TestingRec(key, 2));
150*c8dee2aaSAndroid Build Coastguard Worker     cache.add(new TestingRec(key, 3));
151*c8dee2aaSAndroid Build Coastguard Worker 
152*c8dee2aaSAndroid Build Coastguard Worker     // Lookup can return either value.
153*c8dee2aaSAndroid Build Coastguard Worker     intptr_t value = -1;
154*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(r, cache.find(key, TestingRec::Visitor, &value));
155*c8dee2aaSAndroid Build Coastguard Worker     REPORTER_ASSERT(r, 2 == value || 3 == value);
156*c8dee2aaSAndroid Build Coastguard Worker }
157