blob: e58857cb6959a944204ded400cf16bfcff8be0da [file] [log] [blame]
Chet Haase5c13d892010-10-08 08:37:55 -07001/*
2 * Copyright (C) 2010 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
Romain Guye3b0a012013-06-26 15:45:41 -070017#define LOG_TAG "OpenGLRenderer"
18
Chet Haase5c13d892010-10-08 08:37:55 -070019#include <SkPixelRef.h>
20#include "ResourceCache.h"
21#include "Caches.h"
22
23namespace android {
24namespace uirenderer {
25
26///////////////////////////////////////////////////////////////////////////////
27// Resource cache
28///////////////////////////////////////////////////////////////////////////////
29
30void ResourceCache::logCache() {
Steve Block5baa3a62011-12-20 16:23:08 +000031 ALOGD("ResourceCache: cacheReport:");
Chet Haase5c13d892010-10-08 08:37:55 -070032 for (size_t i = 0; i < mCache->size(); ++i) {
33 ResourceReference* ref = mCache->valueAt(i);
Steve Block5baa3a62011-12-20 16:23:08 +000034 ALOGD(" ResourceCache: mCache(%d): resource, ref = 0x%p, 0x%p",
Chet Haase5c13d892010-10-08 08:37:55 -070035 i, mCache->keyAt(i), mCache->valueAt(i));
Steve Block5baa3a62011-12-20 16:23:08 +000036 ALOGD(" ResourceCache: mCache(%d): refCount, recycled, destroyed, type = %d, %d, %d, %d",
Chet Haase5c13d892010-10-08 08:37:55 -070037 i, ref->refCount, ref->recycled, ref->destroyed, ref->resourceType);
38 }
39}
40
41ResourceCache::ResourceCache() {
Chet Haasee7d22952010-11-11 16:30:16 -080042 Mutex::Autolock _l(mLock);
Chris Craikd218a922014-01-02 17:13:34 -080043 mCache = new KeyedVector<const void*, ResourceReference*>();
Chet Haase5c13d892010-10-08 08:37:55 -070044}
45
46ResourceCache::~ResourceCache() {
Chet Haasee7d22952010-11-11 16:30:16 -080047 Mutex::Autolock _l(mLock);
Chet Haase5c13d892010-10-08 08:37:55 -070048 delete mCache;
49}
50
Romain Guy58ecc202012-09-07 11:58:36 -070051void ResourceCache::lock() {
52 mLock.lock();
53}
54
55void ResourceCache::unlock() {
56 mLock.unlock();
57}
58
Chet Haase5c13d892010-10-08 08:37:55 -070059void ResourceCache::incrementRefcount(void* resource, ResourceType resourceType) {
Chet Haasee7d22952010-11-11 16:30:16 -080060 Mutex::Autolock _l(mLock);
Romain Guy58ecc202012-09-07 11:58:36 -070061 incrementRefcountLocked(resource, resourceType);
Chet Haase5c13d892010-10-08 08:37:55 -070062}
63
Chris Craikd218a922014-01-02 17:13:34 -080064void ResourceCache::incrementRefcount(const SkBitmap* bitmapResource) {
Chris Craik0c20c382013-07-02 10:48:54 -070065 bitmapResource->pixelRef()->globalRef();
Derek Sollenberger6062c592011-02-22 13:55:04 -050066 SkSafeRef(bitmapResource->getColorTable());
Romain Guy49c5fc02012-05-15 11:10:01 -070067 incrementRefcount((void*) bitmapResource, kBitmap);
Chet Haase5c13d892010-10-08 08:37:55 -070068}
69
Chris Craikd218a922014-01-02 17:13:34 -080070void ResourceCache::incrementRefcount(const SkPath* pathResource) {
Romain Guy49c5fc02012-05-15 11:10:01 -070071 incrementRefcount((void*) pathResource, kPath);
Chet Haase5a7e8282011-02-04 12:50:55 -080072}
73
Chet Haase5c13d892010-10-08 08:37:55 -070074void ResourceCache::incrementRefcount(SkiaShader* shaderResource) {
Derek Sollenberger6062c592011-02-22 13:55:04 -050075 SkSafeRef(shaderResource->getSkShader());
Romain Guy43ccf462011-01-14 18:51:01 -080076 incrementRefcount((void*) shaderResource, kShader);
Chet Haase5c13d892010-10-08 08:37:55 -070077}
78
Chet Haasead93c2b2010-10-22 16:17:12 -070079void ResourceCache::incrementRefcount(SkiaColorFilter* filterResource) {
Derek Sollenberger6062c592011-02-22 13:55:04 -050080 SkSafeRef(filterResource->getSkColorFilter());
Romain Guy43ccf462011-01-14 18:51:01 -080081 incrementRefcount((void*) filterResource, kColorFilter);
Chet Haasead93c2b2010-10-22 16:17:12 -070082}
83
Chris Craikd218a922014-01-02 17:13:34 -080084void ResourceCache::incrementRefcount(const Res_png_9patch* patchResource) {
Romain Guye3b0a012013-06-26 15:45:41 -070085 incrementRefcount((void*) patchResource, kNinePatch);
86}
87
Chet Haase603f6de2012-09-14 15:31:25 -070088void ResourceCache::incrementRefcount(Layer* layerResource) {
89 incrementRefcount((void*) layerResource, kLayer);
90}
91
Romain Guy58ecc202012-09-07 11:58:36 -070092void ResourceCache::incrementRefcountLocked(void* resource, ResourceType resourceType) {
Romain Guy8dcfd5e2012-07-20 11:36:03 -070093 ssize_t index = mCache->indexOfKey(resource);
94 ResourceReference* ref = index >= 0 ? mCache->valueAt(index) : NULL;
Romain Guy58ecc202012-09-07 11:58:36 -070095 if (ref == NULL || mCache->size() == 0) {
96 ref = new ResourceReference(resourceType);
97 mCache->add(resource, ref);
Chet Haase5c13d892010-10-08 08:37:55 -070098 }
Romain Guy58ecc202012-09-07 11:58:36 -070099 ref->refCount++;
100}
101
Chris Craikd218a922014-01-02 17:13:34 -0800102void ResourceCache::incrementRefcountLocked(const SkBitmap* bitmapResource) {
Chris Craik0c20c382013-07-02 10:48:54 -0700103 bitmapResource->pixelRef()->globalRef();
Romain Guy58ecc202012-09-07 11:58:36 -0700104 SkSafeRef(bitmapResource->getColorTable());
105 incrementRefcountLocked((void*) bitmapResource, kBitmap);
106}
107
Chris Craikd218a922014-01-02 17:13:34 -0800108void ResourceCache::incrementRefcountLocked(const SkPath* pathResource) {
Romain Guy58ecc202012-09-07 11:58:36 -0700109 incrementRefcountLocked((void*) pathResource, kPath);
110}
111
112void ResourceCache::incrementRefcountLocked(SkiaShader* shaderResource) {
113 SkSafeRef(shaderResource->getSkShader());
114 incrementRefcountLocked((void*) shaderResource, kShader);
115}
116
117void ResourceCache::incrementRefcountLocked(SkiaColorFilter* filterResource) {
118 SkSafeRef(filterResource->getSkColorFilter());
119 incrementRefcountLocked((void*) filterResource, kColorFilter);
120}
121
Chris Craikd218a922014-01-02 17:13:34 -0800122void ResourceCache::incrementRefcountLocked(const Res_png_9patch* patchResource) {
Romain Guye3b0a012013-06-26 15:45:41 -0700123 incrementRefcountLocked((void*) patchResource, kNinePatch);
124}
125
Chet Haase603f6de2012-09-14 15:31:25 -0700126void ResourceCache::incrementRefcountLocked(Layer* layerResource) {
127 incrementRefcountLocked((void*) layerResource, kLayer);
128}
129
Romain Guy58ecc202012-09-07 11:58:36 -0700130void ResourceCache::decrementRefcount(void* resource) {
131 Mutex::Autolock _l(mLock);
132 decrementRefcountLocked(resource);
Chet Haase5c13d892010-10-08 08:37:55 -0700133}
134
Chris Craikd218a922014-01-02 17:13:34 -0800135void ResourceCache::decrementRefcount(const SkBitmap* bitmapResource) {
Chris Craik0c20c382013-07-02 10:48:54 -0700136 bitmapResource->pixelRef()->globalUnref();
Derek Sollenberger6062c592011-02-22 13:55:04 -0500137 SkSafeUnref(bitmapResource->getColorTable());
Romain Guy43ccf462011-01-14 18:51:01 -0800138 decrementRefcount((void*) bitmapResource);
Chet Haase5c13d892010-10-08 08:37:55 -0700139}
140
Chris Craikd218a922014-01-02 17:13:34 -0800141void ResourceCache::decrementRefcount(const SkPath* pathResource) {
Chet Haase5a7e8282011-02-04 12:50:55 -0800142 decrementRefcount((void*) pathResource);
143}
144
Chet Haase5c13d892010-10-08 08:37:55 -0700145void ResourceCache::decrementRefcount(SkiaShader* shaderResource) {
Derek Sollenberger6062c592011-02-22 13:55:04 -0500146 SkSafeUnref(shaderResource->getSkShader());
Romain Guy43ccf462011-01-14 18:51:01 -0800147 decrementRefcount((void*) shaderResource);
Chet Haase5c13d892010-10-08 08:37:55 -0700148}
149
Chet Haasead93c2b2010-10-22 16:17:12 -0700150void ResourceCache::decrementRefcount(SkiaColorFilter* filterResource) {
Derek Sollenberger6062c592011-02-22 13:55:04 -0500151 SkSafeUnref(filterResource->getSkColorFilter());
Romain Guy43ccf462011-01-14 18:51:01 -0800152 decrementRefcount((void*) filterResource);
Chet Haasead93c2b2010-10-22 16:17:12 -0700153}
154
Chris Craikd218a922014-01-02 17:13:34 -0800155void ResourceCache::decrementRefcount(const Res_png_9patch* patchResource) {
Romain Guye3b0a012013-06-26 15:45:41 -0700156 decrementRefcount((void*) patchResource);
157}
158
Chet Haase603f6de2012-09-14 15:31:25 -0700159void ResourceCache::decrementRefcount(Layer* layerResource) {
160 decrementRefcount((void*) layerResource);
161}
162
Romain Guy58ecc202012-09-07 11:58:36 -0700163void ResourceCache::decrementRefcountLocked(void* resource) {
Romain Guy8dcfd5e2012-07-20 11:36:03 -0700164 ssize_t index = mCache->indexOfKey(resource);
Romain Guy58ecc202012-09-07 11:58:36 -0700165 ResourceReference* ref = index >= 0 ? mCache->valueAt(index) : NULL;
Chet Haase5c13d892010-10-08 08:37:55 -0700166 if (ref == NULL) {
Romain Guy58ecc202012-09-07 11:58:36 -0700167 // Should not get here - shouldn't get a call to decrement if we're not yet tracking it
Chet Haase5c13d892010-10-08 08:37:55 -0700168 return;
169 }
Romain Guy58ecc202012-09-07 11:58:36 -0700170 ref->refCount--;
Chet Haase5c13d892010-10-08 08:37:55 -0700171 if (ref->refCount == 0) {
Romain Guy97dc9172012-09-23 17:46:45 -0700172 deleteResourceReferenceLocked(resource, ref);
Chet Haase5c13d892010-10-08 08:37:55 -0700173 }
174}
175
Chris Craikd218a922014-01-02 17:13:34 -0800176void ResourceCache::decrementRefcountLocked(const SkBitmap* bitmapResource) {
Chris Craik0c20c382013-07-02 10:48:54 -0700177 bitmapResource->pixelRef()->globalUnref();
Romain Guy58ecc202012-09-07 11:58:36 -0700178 SkSafeUnref(bitmapResource->getColorTable());
179 decrementRefcountLocked((void*) bitmapResource);
180}
181
Chris Craikd218a922014-01-02 17:13:34 -0800182void ResourceCache::decrementRefcountLocked(const SkPath* pathResource) {
Romain Guy58ecc202012-09-07 11:58:36 -0700183 decrementRefcountLocked((void*) pathResource);
184}
185
186void ResourceCache::decrementRefcountLocked(SkiaShader* shaderResource) {
187 SkSafeUnref(shaderResource->getSkShader());
188 decrementRefcountLocked((void*) shaderResource);
189}
190
191void ResourceCache::decrementRefcountLocked(SkiaColorFilter* filterResource) {
192 SkSafeUnref(filterResource->getSkColorFilter());
193 decrementRefcountLocked((void*) filterResource);
194}
195
Chris Craikd218a922014-01-02 17:13:34 -0800196void ResourceCache::decrementRefcountLocked(const Res_png_9patch* patchResource) {
Romain Guye3b0a012013-06-26 15:45:41 -0700197 decrementRefcountLocked((void*) patchResource);
198}
199
Chet Haase603f6de2012-09-14 15:31:25 -0700200void ResourceCache::decrementRefcountLocked(Layer* layerResource) {
201 decrementRefcountLocked((void*) layerResource);
202}
203
Chet Haase5a7e8282011-02-04 12:50:55 -0800204void ResourceCache::destructor(SkPath* resource) {
205 Mutex::Autolock _l(mLock);
Romain Guy58ecc202012-09-07 11:58:36 -0700206 destructorLocked(resource);
207}
208
209void ResourceCache::destructorLocked(SkPath* resource) {
Romain Guy8dcfd5e2012-07-20 11:36:03 -0700210 ssize_t index = mCache->indexOfKey(resource);
211 ResourceReference* ref = index >= 0 ? mCache->valueAt(index) : NULL;
Chet Haase5a7e8282011-02-04 12:50:55 -0800212 if (ref == NULL) {
213 // If we're not tracking this resource, just delete it
214 if (Caches::hasInstance()) {
215 Caches::getInstance().pathCache.removeDeferred(resource);
216 }
217 delete resource;
218 return;
219 }
220 ref->destroyed = true;
221 if (ref->refCount == 0) {
Romain Guy97dc9172012-09-23 17:46:45 -0700222 deleteResourceReferenceLocked(resource, ref);
Chet Haase5a7e8282011-02-04 12:50:55 -0800223 }
224}
225
Chris Craikd218a922014-01-02 17:13:34 -0800226void ResourceCache::destructor(const SkBitmap* resource) {
Chet Haasee7d22952010-11-11 16:30:16 -0800227 Mutex::Autolock _l(mLock);
Romain Guy58ecc202012-09-07 11:58:36 -0700228 destructorLocked(resource);
229}
230
Chris Craikd218a922014-01-02 17:13:34 -0800231void ResourceCache::destructorLocked(const SkBitmap* resource) {
Romain Guy8dcfd5e2012-07-20 11:36:03 -0700232 ssize_t index = mCache->indexOfKey(resource);
233 ResourceReference* ref = index >= 0 ? mCache->valueAt(index) : NULL;
Chet Haase5c13d892010-10-08 08:37:55 -0700234 if (ref == NULL) {
235 // If we're not tracking this resource, just delete it
236 if (Caches::hasInstance()) {
Romain Guyfe48f652010-11-11 15:36:56 -0800237 Caches::getInstance().textureCache.removeDeferred(resource);
Chet Haase5c13d892010-10-08 08:37:55 -0700238 }
239 delete resource;
240 return;
241 }
242 ref->destroyed = true;
243 if (ref->refCount == 0) {
Romain Guy97dc9172012-09-23 17:46:45 -0700244 deleteResourceReferenceLocked(resource, ref);
Chet Haase5c13d892010-10-08 08:37:55 -0700245 }
246}
247
Chet Haase5c13d892010-10-08 08:37:55 -0700248void ResourceCache::destructor(SkiaShader* resource) {
Chet Haasee7d22952010-11-11 16:30:16 -0800249 Mutex::Autolock _l(mLock);
Romain Guy58ecc202012-09-07 11:58:36 -0700250 destructorLocked(resource);
251}
252
253void ResourceCache::destructorLocked(SkiaShader* resource) {
Romain Guy8dcfd5e2012-07-20 11:36:03 -0700254 ssize_t index = mCache->indexOfKey(resource);
255 ResourceReference* ref = index >= 0 ? mCache->valueAt(index) : NULL;
Chet Haase5c13d892010-10-08 08:37:55 -0700256 if (ref == NULL) {
257 // If we're not tracking this resource, just delete it
Chet Haase5c13d892010-10-08 08:37:55 -0700258 delete resource;
259 return;
260 }
261 ref->destroyed = true;
262 if (ref->refCount == 0) {
Romain Guy97dc9172012-09-23 17:46:45 -0700263 deleteResourceReferenceLocked(resource, ref);
Chet Haase5c13d892010-10-08 08:37:55 -0700264 }
265}
266
Chet Haasead93c2b2010-10-22 16:17:12 -0700267void ResourceCache::destructor(SkiaColorFilter* resource) {
Chet Haasee7d22952010-11-11 16:30:16 -0800268 Mutex::Autolock _l(mLock);
Romain Guy58ecc202012-09-07 11:58:36 -0700269 destructorLocked(resource);
270}
271
272void ResourceCache::destructorLocked(SkiaColorFilter* resource) {
Romain Guy8dcfd5e2012-07-20 11:36:03 -0700273 ssize_t index = mCache->indexOfKey(resource);
274 ResourceReference* ref = index >= 0 ? mCache->valueAt(index) : NULL;
Chet Haasead93c2b2010-10-22 16:17:12 -0700275 if (ref == NULL) {
276 // If we're not tracking this resource, just delete it
277 delete resource;
278 return;
279 }
280 ref->destroyed = true;
281 if (ref->refCount == 0) {
Romain Guy97dc9172012-09-23 17:46:45 -0700282 deleteResourceReferenceLocked(resource, ref);
Chet Haasead93c2b2010-10-22 16:17:12 -0700283 }
284}
285
Romain Guye3b0a012013-06-26 15:45:41 -0700286void ResourceCache::destructor(Res_png_9patch* resource) {
287 Mutex::Autolock _l(mLock);
288 destructorLocked(resource);
289}
290
291void ResourceCache::destructorLocked(Res_png_9patch* resource) {
292 ssize_t index = mCache->indexOfKey(resource);
293 ResourceReference* ref = index >= 0 ? mCache->valueAt(index) : NULL;
294 if (ref == NULL) {
295 if (Caches::hasInstance()) {
296 Caches::getInstance().patchCache.removeDeferred(resource);
297 }
298 // If we're not tracking this resource, just delete it
299 // A Res_png_9patch is actually an array of byte that's larger
300 // than sizeof(Res_png_9patch). It must be freed as an array.
301 delete[] (int8_t*) resource;
302 return;
303 }
304 ref->destroyed = true;
305 if (ref->refCount == 0) {
306 deleteResourceReferenceLocked(resource, ref);
307 }
308}
309
Chet Haase547e6652012-10-22 15:07:26 -0700310/**
311 * Return value indicates whether resource was actually recycled, which happens when RefCnt
312 * reaches 0.
313 */
314bool ResourceCache::recycle(SkBitmap* resource) {
Romain Guy58ecc202012-09-07 11:58:36 -0700315 Mutex::Autolock _l(mLock);
Chet Haase547e6652012-10-22 15:07:26 -0700316 return recycleLocked(resource);
Romain Guy58ecc202012-09-07 11:58:36 -0700317}
318
Chet Haase547e6652012-10-22 15:07:26 -0700319/**
320 * Return value indicates whether resource was actually recycled, which happens when RefCnt
321 * reaches 0.
322 */
323bool ResourceCache::recycleLocked(SkBitmap* resource) {
Romain Guy58ecc202012-09-07 11:58:36 -0700324 ssize_t index = mCache->indexOfKey(resource);
325 if (index < 0) {
326 // not tracking this resource; just recycle the pixel data
327 resource->setPixels(NULL, NULL);
Chet Haase547e6652012-10-22 15:07:26 -0700328 return true;
Romain Guy58ecc202012-09-07 11:58:36 -0700329 }
330 ResourceReference* ref = mCache->valueAt(index);
331 if (ref == NULL) {
332 // Should not get here - shouldn't get a call to recycle if we're not yet tracking it
Chet Haase547e6652012-10-22 15:07:26 -0700333 return true;
Romain Guy58ecc202012-09-07 11:58:36 -0700334 }
335 ref->recycled = true;
336 if (ref->refCount == 0) {
Romain Guy97dc9172012-09-23 17:46:45 -0700337 deleteResourceReferenceLocked(resource, ref);
Chet Haase547e6652012-10-22 15:07:26 -0700338 return true;
Romain Guy58ecc202012-09-07 11:58:36 -0700339 }
Chet Haase547e6652012-10-22 15:07:26 -0700340 // Still referring to resource, don't recycle yet
341 return false;
Romain Guy58ecc202012-09-07 11:58:36 -0700342}
343
Chet Haasee7d22952010-11-11 16:30:16 -0800344/**
345 * This method should only be called while the mLock mutex is held (that mutex is grabbed
346 * by the various destructor() and recycle() methods which call this method).
347 */
Chris Craikd218a922014-01-02 17:13:34 -0800348void ResourceCache::deleteResourceReferenceLocked(const void* resource, ResourceReference* ref) {
Chet Haase5c13d892010-10-08 08:37:55 -0700349 if (ref->recycled && ref->resourceType == kBitmap) {
350 ((SkBitmap*) resource)->setPixels(NULL, NULL);
351 }
Chet Haase603f6de2012-09-14 15:31:25 -0700352 if (ref->destroyed || ref->resourceType == kLayer) {
Chet Haase5c13d892010-10-08 08:37:55 -0700353 switch (ref->resourceType) {
Romain Guyd586ad92011-06-22 16:14:36 -0700354 case kBitmap: {
355 SkBitmap* bitmap = (SkBitmap*) resource;
Chet Haase5c13d892010-10-08 08:37:55 -0700356 if (Caches::hasInstance()) {
Romain Guyfe48f652010-11-11 15:36:56 -0800357 Caches::getInstance().textureCache.removeDeferred(bitmap);
Chet Haase5c13d892010-10-08 08:37:55 -0700358 }
359 delete bitmap;
360 }
361 break;
Romain Guyd586ad92011-06-22 16:14:36 -0700362 case kPath: {
363 SkPath* path = (SkPath*) resource;
Chet Haase5a7e8282011-02-04 12:50:55 -0800364 if (Caches::hasInstance()) {
365 Caches::getInstance().pathCache.removeDeferred(path);
366 }
367 delete path;
368 }
369 break;
Romain Guyd586ad92011-06-22 16:14:36 -0700370 case kShader: {
371 SkiaShader* shader = (SkiaShader*) resource;
Chet Haase5c13d892010-10-08 08:37:55 -0700372 delete shader;
Chet Haasead93c2b2010-10-22 16:17:12 -0700373 }
374 break;
Romain Guyd586ad92011-06-22 16:14:36 -0700375 case kColorFilter: {
376 SkiaColorFilter* filter = (SkiaColorFilter*) resource;
Chet Haasead93c2b2010-10-22 16:17:12 -0700377 delete filter;
378 }
379 break;
Romain Guye3b0a012013-06-26 15:45:41 -0700380 case kNinePatch: {
381 if (Caches::hasInstance()) {
382 Caches::getInstance().patchCache.removeDeferred((Res_png_9patch*) resource);
383 }
384 // A Res_png_9patch is actually an array of byte that's larger
385 // than sizeof(Res_png_9patch). It must be freed as an array.
386 int8_t* patch = (int8_t*) resource;
387 delete[] patch;
388 }
389 break;
Chet Haase603f6de2012-09-14 15:31:25 -0700390 case kLayer: {
Dave Burke56257af2012-09-25 20:30:09 -0700391 Layer* layer = (Layer*) resource;
Mathias Agopian54643d72012-09-25 21:30:22 -0700392 Caches::getInstance().deleteLayerDeferred(layer);
Chet Haase603f6de2012-09-14 15:31:25 -0700393 }
394 break;
Chet Haase5c13d892010-10-08 08:37:55 -0700395 }
396 }
397 mCache->removeItem(resource);
398 delete ref;
399}
400
401}; // namespace uirenderer
402}; // namespace android