blob: 626c8373611607ec94811867967b22fd00f8bf68 [file] [log] [blame]
halcanary805ef152014-07-17 06:58:01 -07001/*
2 * Copyright 2014 Google Inc.
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
reed30ad5302014-09-16 10:39:55 -07007
halcanary805ef152014-07-17 06:58:01 -07008#include "Test.h"
reed04617132014-08-21 09:46:49 -07009#include "SkBitmapCache.h"
reed30ad5302014-09-16 10:39:55 -070010#include "SkCanvas.h"
danakj790ffe32014-09-11 10:49:52 -070011#include "SkDiscardableMemoryPool.h"
reed30ad5302014-09-16 10:39:55 -070012#include "SkGraphics.h"
13#include "SkResourceCache.h"
reed3054be12014-12-10 07:24:28 -080014#include "SkSurface.h"
halcanary805ef152014-07-17 06:58:01 -070015
16static const int kCanvasSize = 1;
17static const int kBitmapSize = 16;
18static const int kScale = 8;
19
halcanary1d1795b2014-07-18 09:18:40 -070020static bool is_in_scaled_image_cache(const SkBitmap& orig,
21 SkScalar xScale,
22 SkScalar yScale) {
23 SkBitmap scaled;
humperd73c1692014-08-28 14:27:42 -070024 float roundedImageWidth = SkScalarRoundToScalar(orig.width() * xScale);
yunchao.he90acb8e2015-01-23 17:06:20 -080025 float roundedImageHeight = SkScalarRoundToScalar(orig.height() * yScale);
humperd73c1692014-08-28 14:27:42 -070026 return SkBitmapCache::Find(orig, roundedImageWidth, roundedImageHeight, &scaled);
halcanary1d1795b2014-07-18 09:18:40 -070027}
28
yunchao.he90acb8e2015-01-23 17:06:20 -080029// Draw a scaled bitmap, then return true if it has been cached.
30static bool test_scaled_image_cache_usage() {
reed3054be12014-12-10 07:24:28 -080031 SkAutoTUnref<SkSurface> surface(SkSurface::NewRasterN32Premul(kCanvasSize, kCanvasSize));
32 SkCanvas* canvas = surface->getCanvas();
halcanary805ef152014-07-17 06:58:01 -070033 SkBitmap bitmap;
reed84825042014-09-02 12:50:45 -070034 bitmap.allocN32Pixels(kBitmapSize, kBitmapSize);
halcanary0db38cc2014-07-17 10:17:28 -070035 bitmap.eraseColor(0xFFFFFFFF);
yunchao.he90acb8e2015-01-23 17:06:20 -080036 SkScalar xScale = SkIntToScalar(kScale);
37 SkScalar yScale = xScale / 2;
38 SkScalar xScaledSize = SkIntToScalar(kBitmapSize) * xScale;
39 SkScalar yScaledSize = SkIntToScalar(kBitmapSize) * yScale;
40 canvas->clipRect(SkRect::MakeLTRB(0, 0, xScaledSize, yScaledSize));
halcanary805ef152014-07-17 06:58:01 -070041 SkPaint paint;
42 paint.setFilterLevel(SkPaint::kHigh_FilterLevel);
halcanary1d1795b2014-07-18 09:18:40 -070043
halcanary805ef152014-07-17 06:58:01 -070044 canvas->drawBitmapRect(bitmap,
yunchao.he90acb8e2015-01-23 17:06:20 -080045 SkRect::MakeLTRB(0, 0, xScaledSize, yScaledSize),
halcanary805ef152014-07-17 06:58:01 -070046 &paint);
halcanary1d1795b2014-07-18 09:18:40 -070047
yunchao.he90acb8e2015-01-23 17:06:20 -080048 return is_in_scaled_image_cache(bitmap, xScale, yScale);
halcanary805ef152014-07-17 06:58:01 -070049}
50
51// http://crbug.com/389439
reed011f39a2014-08-28 13:35:23 -070052DEF_TEST(ResourceCache_SingleAllocationByteLimit, reporter) {
53 size_t originalByteLimit = SkGraphics::GetResourceCacheTotalByteLimit();
halcanary805ef152014-07-17 06:58:01 -070054 size_t originalAllocationLimit =
reed011f39a2014-08-28 13:35:23 -070055 SkGraphics::GetResourceCacheSingleAllocationByteLimit();
halcanary805ef152014-07-17 06:58:01 -070056
57 size_t size = kBitmapSize * kScale * kBitmapSize * kScale
58 * SkColorTypeBytesPerPixel(kN32_SkColorType);
59
reed011f39a2014-08-28 13:35:23 -070060 SkGraphics::SetResourceCacheTotalByteLimit(0); // clear cache
61 SkGraphics::SetResourceCacheTotalByteLimit(2 * size);
62 SkGraphics::SetResourceCacheSingleAllocationByteLimit(0); // No limit
halcanary805ef152014-07-17 06:58:01 -070063
yunchao.he90acb8e2015-01-23 17:06:20 -080064 REPORTER_ASSERT(reporter, test_scaled_image_cache_usage());
halcanary805ef152014-07-17 06:58:01 -070065
reed011f39a2014-08-28 13:35:23 -070066 SkGraphics::SetResourceCacheTotalByteLimit(0); // clear cache
67 SkGraphics::SetResourceCacheTotalByteLimit(2 * size);
68 SkGraphics::SetResourceCacheSingleAllocationByteLimit(size * 2); // big enough
halcanary805ef152014-07-17 06:58:01 -070069
yunchao.he90acb8e2015-01-23 17:06:20 -080070 REPORTER_ASSERT(reporter, test_scaled_image_cache_usage());
halcanary805ef152014-07-17 06:58:01 -070071
reed011f39a2014-08-28 13:35:23 -070072 SkGraphics::SetResourceCacheTotalByteLimit(0); // clear cache
73 SkGraphics::SetResourceCacheTotalByteLimit(2 * size);
74 SkGraphics::SetResourceCacheSingleAllocationByteLimit(size / 2); // too small
halcanary805ef152014-07-17 06:58:01 -070075
yunchao.he90acb8e2015-01-23 17:06:20 -080076 REPORTER_ASSERT(reporter, !test_scaled_image_cache_usage());
halcanary805ef152014-07-17 06:58:01 -070077
reed011f39a2014-08-28 13:35:23 -070078 SkGraphics::SetResourceCacheSingleAllocationByteLimit(originalAllocationLimit);
79 SkGraphics::SetResourceCacheTotalByteLimit(originalByteLimit);
halcanary805ef152014-07-17 06:58:01 -070080}
piotaixr42b0dfe2014-09-03 11:33:13 -070081
reed30ad5302014-09-16 10:39:55 -070082////////////////////////////////////////////////////////////////////////////////////////
mtklein26abcf12014-09-04 10:50:53 -070083
reed30ad5302014-09-16 10:39:55 -070084static void make_bitmap(SkBitmap* bitmap, const SkImageInfo& info, SkBitmap::Allocator* allocator) {
bsalomon49f085d2014-09-05 13:34:00 -070085 if (allocator) {
reed30ad5302014-09-16 10:39:55 -070086 bitmap->setInfo(info);
87 allocator->allocPixelRef(bitmap, 0);
piotaixr42b0dfe2014-09-03 11:33:13 -070088 } else {
reed30ad5302014-09-16 10:39:55 -070089 bitmap->allocPixels(info);
piotaixr42b0dfe2014-09-03 11:33:13 -070090 }
piotaixr42b0dfe2014-09-03 11:33:13 -070091}
92
93// http://skbug.com/2894
94DEF_TEST(BitmapCache_add_rect, reporter) {
reed30ad5302014-09-16 10:39:55 -070095 SkResourceCache::DiscardableFactory factory = SkResourceCache::GetDiscardableFactory();
96 SkBitmap::Allocator* allocator = SkBitmapCache::GetAllocator();
97
98 SkAutoTDelete<SkResourceCache> cache;
99 if (factory) {
100 cache.reset(SkNEW_ARGS(SkResourceCache, (factory)));
101 } else {
102 const size_t byteLimit = 100 * 1024;
103 cache.reset(SkNEW_ARGS(SkResourceCache, (byteLimit)));
104 }
105 SkBitmap cachedBitmap;
106 make_bitmap(&cachedBitmap, SkImageInfo::MakeN32Premul(5, 5), allocator);
piotaixr42b0dfe2014-09-03 11:33:13 -0700107 cachedBitmap.setImmutable();
108
reed30ad5302014-09-16 10:39:55 -0700109 SkBitmap bm;
110 SkIRect rect = SkIRect::MakeWH(5, 5);
111
piotaixr42b0dfe2014-09-03 11:33:13 -0700112 // Wrong subset size
reed30ad5302014-09-16 10:39:55 -0700113 REPORTER_ASSERT(reporter, !SkBitmapCache::Add(cachedBitmap.getGenerationID(), SkIRect::MakeWH(4, 6), cachedBitmap, cache));
114 REPORTER_ASSERT(reporter, !SkBitmapCache::Find(cachedBitmap.getGenerationID(), rect, &bm, cache));
piotaixr42b0dfe2014-09-03 11:33:13 -0700115 // Wrong offset value
reed30ad5302014-09-16 10:39:55 -0700116 REPORTER_ASSERT(reporter, !SkBitmapCache::Add(cachedBitmap.getGenerationID(), SkIRect::MakeXYWH(-1, 0, 5, 5), cachedBitmap, cache));
117 REPORTER_ASSERT(reporter, !SkBitmapCache::Find(cachedBitmap.getGenerationID(), rect, &bm, cache));
piotaixr42b0dfe2014-09-03 11:33:13 -0700118
119 // Should not be in the cache
reed30ad5302014-09-16 10:39:55 -0700120 REPORTER_ASSERT(reporter, !SkBitmapCache::Find(cachedBitmap.getGenerationID(), rect, &bm, cache));
piotaixr42b0dfe2014-09-03 11:33:13 -0700121
reed30ad5302014-09-16 10:39:55 -0700122 REPORTER_ASSERT(reporter, SkBitmapCache::Add(cachedBitmap.getGenerationID(), rect, cachedBitmap, cache));
piotaixr42b0dfe2014-09-03 11:33:13 -0700123 // Should be in the cache, we just added it
reed30ad5302014-09-16 10:39:55 -0700124 REPORTER_ASSERT(reporter, SkBitmapCache::Find(cachedBitmap.getGenerationID(), rect, &bm, cache));
piotaixr42b0dfe2014-09-03 11:33:13 -0700125}
danakj790ffe32014-09-11 10:49:52 -0700126
reed9d93c2e2014-10-08 05:17:12 -0700127#include "SkMipMap.h"
128
129enum LockedState {
130 kNotLocked,
131 kLocked,
132};
133
134enum CachedState {
135 kNotInCache,
136 kInCache,
137};
138
139static void check_data(skiatest::Reporter* reporter, const SkCachedData* data,
140 int refcnt, CachedState cacheState, LockedState lockedState) {
141 REPORTER_ASSERT(reporter, data->testing_only_getRefCnt() == refcnt);
142 REPORTER_ASSERT(reporter, data->testing_only_isInCache() == (kInCache == cacheState));
143 bool isLocked = (data->data() != NULL);
144 REPORTER_ASSERT(reporter, isLocked == (lockedState == kLocked));
145}
146
147static void test_mipmapcache(skiatest::Reporter* reporter, SkResourceCache* cache) {
148 cache->purgeAll();
149
150 SkBitmap src;
151 src.allocN32Pixels(5, 5);
152 src.setImmutable();
153
154 const SkMipMap* mipmap = SkMipMapCache::FindAndRef(src, cache);
155 REPORTER_ASSERT(reporter, NULL == mipmap);
156
157 mipmap = SkMipMapCache::AddAndRef(src, cache);
158 REPORTER_ASSERT(reporter, mipmap);
reed7eeba252015-02-24 13:54:23 -0800159
160 {
161 const SkMipMap* mm = SkMipMapCache::FindAndRef(src, cache);
162 REPORTER_ASSERT(reporter, mm);
163 REPORTER_ASSERT(reporter, mm == mipmap);
164 mm->unref();
165 }
166
reed9d93c2e2014-10-08 05:17:12 -0700167 check_data(reporter, mipmap, 2, kInCache, kLocked);
168
169 mipmap->unref();
170 // tricky, since technically after this I'm no longer an owner, but since the cache is
171 // local, I know it won't get purged behind my back
172 check_data(reporter, mipmap, 1, kInCache, kNotLocked);
173
174 // find us again
175 mipmap = SkMipMapCache::FindAndRef(src, cache);
176 check_data(reporter, mipmap, 2, kInCache, kLocked);
177
178 cache->purgeAll();
179 check_data(reporter, mipmap, 1, kNotInCache, kLocked);
180
181 mipmap->unref();
182}
183
reed7eeba252015-02-24 13:54:23 -0800184static void test_mipmap_notify(skiatest::Reporter* reporter, SkResourceCache* cache) {
185 const int N = 3;
186 SkBitmap src[N];
187 for (int i = 0; i < N; ++i) {
188 src[i].allocN32Pixels(5, 5);
189 src[i].setImmutable();
190 SkMipMapCache::AddAndRef(src[i], cache)->unref();
191 }
192
193 for (int i = 0; i < N; ++i) {
194 const SkMipMap* mipmap = SkMipMapCache::FindAndRef(src[i], cache);
195 if (cache) {
196 // if cache is null, we're working on the global cache, and other threads might purge
197 // it, making this check fragile.
198 REPORTER_ASSERT(reporter, mipmap);
199 }
200 SkSafeUnref(mipmap);
201
202 src[i].reset(); // delete the underlying pixelref, which *should* remove us from the cache
203
204 mipmap = SkMipMapCache::FindAndRef(src[i], cache);
205 REPORTER_ASSERT(reporter, !mipmap);
206 }
207}
208
209static void test_bitmap_notify(skiatest::Reporter* reporter, SkResourceCache* cache) {
210 const SkIRect subset = SkIRect::MakeWH(5, 5);
211 const int N = 3;
212 SkBitmap src[N], dst[N];
213 for (int i = 0; i < N; ++i) {
214 src[i].allocN32Pixels(5, 5);
215 src[i].setImmutable();
216 dst[i].allocN32Pixels(5, 5);
217 dst[i].setImmutable();
218 SkBitmapCache::Add(src[i].getGenerationID(), subset, dst[i], cache);
219 }
220
221 for (int i = 0; i < N; ++i) {
222 const uint32_t genID = src[i].getGenerationID();
223 SkBitmap result;
224 bool found = SkBitmapCache::Find(genID, subset, &result, cache);
225 if (cache) {
226 // if cache is null, we're working on the global cache, and other threads might purge
227 // it, making this check fragile.
228 REPORTER_ASSERT(reporter, found);
229 }
230
231 src[i].reset(); // delete the underlying pixelref, which *should* remove us from the cache
232
233 found = SkBitmapCache::Find(genID, subset, &result, cache);
234 REPORTER_ASSERT(reporter, !found);
235 }
236}
237
danakj790ffe32014-09-11 10:49:52 -0700238DEF_TEST(BitmapCache_discarded_bitmap, reporter) {
reed30ad5302014-09-16 10:39:55 -0700239 SkResourceCache::DiscardableFactory factory = SkResourceCache::GetDiscardableFactory();
240 SkBitmap::Allocator* allocator = SkBitmapCache::GetAllocator();
241
242 SkAutoTDelete<SkResourceCache> cache;
243 if (factory) {
244 cache.reset(SkNEW_ARGS(SkResourceCache, (factory)));
245 } else {
246 const size_t byteLimit = 100 * 1024;
247 cache.reset(SkNEW_ARGS(SkResourceCache, (byteLimit)));
248 }
249 SkBitmap cachedBitmap;
250 make_bitmap(&cachedBitmap, SkImageInfo::MakeN32Premul(5, 5), allocator);
danakj790ffe32014-09-11 10:49:52 -0700251 cachedBitmap.setImmutable();
252 cachedBitmap.unlockPixels();
253
reed30ad5302014-09-16 10:39:55 -0700254 SkBitmap bm;
255 SkIRect rect = SkIRect::MakeWH(5, 5);
256
danakj790ffe32014-09-11 10:49:52 -0700257 // Add a bitmap to the cache.
reed30ad5302014-09-16 10:39:55 -0700258 REPORTER_ASSERT(reporter, SkBitmapCache::Add(cachedBitmap.getGenerationID(), rect, cachedBitmap, cache));
259 REPORTER_ASSERT(reporter, SkBitmapCache::Find(cachedBitmap.getGenerationID(), rect, &bm, cache));
danakj790ffe32014-09-11 10:49:52 -0700260
261 // Finding more than once works fine.
reed30ad5302014-09-16 10:39:55 -0700262 REPORTER_ASSERT(reporter, SkBitmapCache::Find(cachedBitmap.getGenerationID(), rect, &bm, cache));
danakj790ffe32014-09-11 10:49:52 -0700263 bm.unlockPixels();
264
265 // Drop the pixels in the bitmap.
reed30ad5302014-09-16 10:39:55 -0700266 if (factory) {
267 REPORTER_ASSERT(reporter, SkGetGlobalDiscardableMemoryPool()->getRAMUsed() > 0);
268 SkGetGlobalDiscardableMemoryPool()->dumpPool();
269 REPORTER_ASSERT(reporter, SkGetGlobalDiscardableMemoryPool()->getRAMUsed() == 0);
danakj790ffe32014-09-11 10:49:52 -0700270
reed30ad5302014-09-16 10:39:55 -0700271 // The bitmap is not in the cache since it has been dropped.
272 REPORTER_ASSERT(reporter, !SkBitmapCache::Find(cachedBitmap.getGenerationID(), rect, &bm, cache));
273 }
danakj790ffe32014-09-11 10:49:52 -0700274
reed30ad5302014-09-16 10:39:55 -0700275 make_bitmap(&cachedBitmap, SkImageInfo::MakeN32Premul(5, 5), allocator);
danakj790ffe32014-09-11 10:49:52 -0700276 cachedBitmap.setImmutable();
277 cachedBitmap.unlockPixels();
278
279 // We can add the bitmap back to the cache and find it again.
reed30ad5302014-09-16 10:39:55 -0700280 REPORTER_ASSERT(reporter, SkBitmapCache::Add(cachedBitmap.getGenerationID(), rect, cachedBitmap, cache));
281 REPORTER_ASSERT(reporter, SkBitmapCache::Find(cachedBitmap.getGenerationID(), rect, &bm, cache));
reed9d93c2e2014-10-08 05:17:12 -0700282
283 test_mipmapcache(reporter, cache);
reed7eeba252015-02-24 13:54:23 -0800284 test_bitmap_notify(reporter, cache);
285 test_mipmap_notify(reporter, cache);
danakj790ffe32014-09-11 10:49:52 -0700286}