ResourceCache.cpp revision e7d2295c06ef9b9df6336cbff23007a13fb3f6e4
15c13d89c1332fcc499379b9064b891187b75ca32Chet Haase/* 25c13d89c1332fcc499379b9064b891187b75ca32Chet Haase * Copyright (C) 2010 The Android Open Source Project 35c13d89c1332fcc499379b9064b891187b75ca32Chet Haase * 45c13d89c1332fcc499379b9064b891187b75ca32Chet Haase * Licensed under the Apache License, Version 2.0 (the "License"); 55c13d89c1332fcc499379b9064b891187b75ca32Chet Haase * you may not use this file except in compliance with the License. 65c13d89c1332fcc499379b9064b891187b75ca32Chet Haase * You may obtain a copy of the License at 75c13d89c1332fcc499379b9064b891187b75ca32Chet Haase * 85c13d89c1332fcc499379b9064b891187b75ca32Chet Haase * http://www.apache.org/licenses/LICENSE-2.0 95c13d89c1332fcc499379b9064b891187b75ca32Chet Haase * 105c13d89c1332fcc499379b9064b891187b75ca32Chet Haase * Unless required by applicable law or agreed to in writing, software 115c13d89c1332fcc499379b9064b891187b75ca32Chet Haase * distributed under the License is distributed on an "AS IS" BASIS, 125c13d89c1332fcc499379b9064b891187b75ca32Chet Haase * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 135c13d89c1332fcc499379b9064b891187b75ca32Chet Haase * See the License for the specific language governing permissions and 145c13d89c1332fcc499379b9064b891187b75ca32Chet Haase * limitations under the License. 155c13d89c1332fcc499379b9064b891187b75ca32Chet Haase */ 165c13d89c1332fcc499379b9064b891187b75ca32Chet Haase 175c13d89c1332fcc499379b9064b891187b75ca32Chet Haase#include <SkPixelRef.h> 185c13d89c1332fcc499379b9064b891187b75ca32Chet Haase#include "ResourceCache.h" 195c13d89c1332fcc499379b9064b891187b75ca32Chet Haase#include "Caches.h" 205c13d89c1332fcc499379b9064b891187b75ca32Chet Haase 215c13d89c1332fcc499379b9064b891187b75ca32Chet Haasenamespace android { 225c13d89c1332fcc499379b9064b891187b75ca32Chet Haasenamespace uirenderer { 235c13d89c1332fcc499379b9064b891187b75ca32Chet Haase 245c13d89c1332fcc499379b9064b891187b75ca32Chet Haase/////////////////////////////////////////////////////////////////////////////// 255c13d89c1332fcc499379b9064b891187b75ca32Chet Haase// Resource cache 265c13d89c1332fcc499379b9064b891187b75ca32Chet Haase/////////////////////////////////////////////////////////////////////////////// 275c13d89c1332fcc499379b9064b891187b75ca32Chet Haase 285c13d89c1332fcc499379b9064b891187b75ca32Chet Haasevoid ResourceCache::logCache() { 295c13d89c1332fcc499379b9064b891187b75ca32Chet Haase LOGD("ResourceCache: cacheReport:"); 305c13d89c1332fcc499379b9064b891187b75ca32Chet Haase for (size_t i = 0; i < mCache->size(); ++i) { 315c13d89c1332fcc499379b9064b891187b75ca32Chet Haase ResourceReference* ref = mCache->valueAt(i); 325c13d89c1332fcc499379b9064b891187b75ca32Chet Haase LOGD(" ResourceCache: mCache(%d): resource, ref = 0x%p, 0x%p", 335c13d89c1332fcc499379b9064b891187b75ca32Chet Haase i, mCache->keyAt(i), mCache->valueAt(i)); 345c13d89c1332fcc499379b9064b891187b75ca32Chet Haase LOGD(" ResourceCache: mCache(%d): refCount, recycled, destroyed, type = %d, %d, %d, %d", 355c13d89c1332fcc499379b9064b891187b75ca32Chet Haase i, ref->refCount, ref->recycled, ref->destroyed, ref->resourceType); 365c13d89c1332fcc499379b9064b891187b75ca32Chet Haase } 375c13d89c1332fcc499379b9064b891187b75ca32Chet Haase} 385c13d89c1332fcc499379b9064b891187b75ca32Chet Haase 395c13d89c1332fcc499379b9064b891187b75ca32Chet HaaseResourceCache::ResourceCache() { 40e7d2295c06ef9b9df6336cbff23007a13fb3f6e4Chet Haase Mutex::Autolock _l(mLock); 415c13d89c1332fcc499379b9064b891187b75ca32Chet Haase mCache = new KeyedVector<void *, ResourceReference *>(); 425c13d89c1332fcc499379b9064b891187b75ca32Chet Haase} 435c13d89c1332fcc499379b9064b891187b75ca32Chet Haase 445c13d89c1332fcc499379b9064b891187b75ca32Chet HaaseResourceCache::~ResourceCache() { 45e7d2295c06ef9b9df6336cbff23007a13fb3f6e4Chet Haase Mutex::Autolock _l(mLock); 465c13d89c1332fcc499379b9064b891187b75ca32Chet Haase delete mCache; 475c13d89c1332fcc499379b9064b891187b75ca32Chet Haase} 485c13d89c1332fcc499379b9064b891187b75ca32Chet Haase 495c13d89c1332fcc499379b9064b891187b75ca32Chet Haasevoid ResourceCache::incrementRefcount(void* resource, ResourceType resourceType) { 50e7d2295c06ef9b9df6336cbff23007a13fb3f6e4Chet Haase Mutex::Autolock _l(mLock); 515c13d89c1332fcc499379b9064b891187b75ca32Chet Haase for (size_t i = 0; i < mCache->size(); ++i) { 525c13d89c1332fcc499379b9064b891187b75ca32Chet Haase void* ref = mCache->valueAt(i); 535c13d89c1332fcc499379b9064b891187b75ca32Chet Haase } 545c13d89c1332fcc499379b9064b891187b75ca32Chet Haase ResourceReference* ref = mCache->indexOfKey(resource) >= 0 ? mCache->valueFor(resource) : NULL; 555c13d89c1332fcc499379b9064b891187b75ca32Chet Haase if (ref == NULL || mCache->size() == 0) { 565c13d89c1332fcc499379b9064b891187b75ca32Chet Haase ref = new ResourceReference(resourceType); 575c13d89c1332fcc499379b9064b891187b75ca32Chet Haase mCache->add(resource, ref); 585c13d89c1332fcc499379b9064b891187b75ca32Chet Haase } 595c13d89c1332fcc499379b9064b891187b75ca32Chet Haase ref->refCount++; 605c13d89c1332fcc499379b9064b891187b75ca32Chet Haase} 615c13d89c1332fcc499379b9064b891187b75ca32Chet Haase 625c13d89c1332fcc499379b9064b891187b75ca32Chet Haasevoid ResourceCache::incrementRefcount(SkBitmap* bitmapResource) { 635c13d89c1332fcc499379b9064b891187b75ca32Chet Haase bitmapResource->pixelRef()->safeRef(); 645c13d89c1332fcc499379b9064b891187b75ca32Chet Haase bitmapResource->getColorTable()->safeRef(); 655c13d89c1332fcc499379b9064b891187b75ca32Chet Haase incrementRefcount((void*)bitmapResource, kBitmap); 665c13d89c1332fcc499379b9064b891187b75ca32Chet Haase} 675c13d89c1332fcc499379b9064b891187b75ca32Chet Haase 685c13d89c1332fcc499379b9064b891187b75ca32Chet Haasevoid ResourceCache::incrementRefcount(SkiaShader* shaderResource) { 695c13d89c1332fcc499379b9064b891187b75ca32Chet Haase shaderResource->getSkShader()->safeRef(); 705c13d89c1332fcc499379b9064b891187b75ca32Chet Haase incrementRefcount((void*)shaderResource, kShader); 715c13d89c1332fcc499379b9064b891187b75ca32Chet Haase} 725c13d89c1332fcc499379b9064b891187b75ca32Chet Haase 73ad93c2bb63dfc813b2eefa1043aa63afbddce655Chet Haasevoid ResourceCache::incrementRefcount(SkiaColorFilter* filterResource) { 74ad93c2bb63dfc813b2eefa1043aa63afbddce655Chet Haase filterResource->getSkColorFilter()->safeRef(); 75ad93c2bb63dfc813b2eefa1043aa63afbddce655Chet Haase incrementRefcount((void*)filterResource, kColorFilter); 76ad93c2bb63dfc813b2eefa1043aa63afbddce655Chet Haase} 77ad93c2bb63dfc813b2eefa1043aa63afbddce655Chet Haase 785c13d89c1332fcc499379b9064b891187b75ca32Chet Haasevoid ResourceCache::decrementRefcount(void* resource) { 79e7d2295c06ef9b9df6336cbff23007a13fb3f6e4Chet Haase Mutex::Autolock _l(mLock); 805c13d89c1332fcc499379b9064b891187b75ca32Chet Haase ResourceReference* ref = mCache->indexOfKey(resource) >= 0 ? mCache->valueFor(resource) : NULL; 815c13d89c1332fcc499379b9064b891187b75ca32Chet Haase if (ref == NULL) { 825c13d89c1332fcc499379b9064b891187b75ca32Chet Haase // Should not get here - shouldn't get a call to decrement if we're not yet tracking it 835c13d89c1332fcc499379b9064b891187b75ca32Chet Haase return; 845c13d89c1332fcc499379b9064b891187b75ca32Chet Haase } 855c13d89c1332fcc499379b9064b891187b75ca32Chet Haase ref->refCount--; 865c13d89c1332fcc499379b9064b891187b75ca32Chet Haase if (ref->refCount == 0) { 875c13d89c1332fcc499379b9064b891187b75ca32Chet Haase deleteResourceReference(resource, ref); 885c13d89c1332fcc499379b9064b891187b75ca32Chet Haase } 895c13d89c1332fcc499379b9064b891187b75ca32Chet Haase} 905c13d89c1332fcc499379b9064b891187b75ca32Chet Haase 915c13d89c1332fcc499379b9064b891187b75ca32Chet Haasevoid ResourceCache::decrementRefcount(SkBitmap* bitmapResource) { 925c13d89c1332fcc499379b9064b891187b75ca32Chet Haase bitmapResource->pixelRef()->safeUnref(); 935c13d89c1332fcc499379b9064b891187b75ca32Chet Haase bitmapResource->getColorTable()->safeUnref(); 945c13d89c1332fcc499379b9064b891187b75ca32Chet Haase decrementRefcount((void*)bitmapResource); 955c13d89c1332fcc499379b9064b891187b75ca32Chet Haase} 965c13d89c1332fcc499379b9064b891187b75ca32Chet Haase 975c13d89c1332fcc499379b9064b891187b75ca32Chet Haasevoid ResourceCache::decrementRefcount(SkiaShader* shaderResource) { 985c13d89c1332fcc499379b9064b891187b75ca32Chet Haase shaderResource->getSkShader()->safeUnref(); 995c13d89c1332fcc499379b9064b891187b75ca32Chet Haase decrementRefcount((void*)shaderResource); 1005c13d89c1332fcc499379b9064b891187b75ca32Chet Haase} 1015c13d89c1332fcc499379b9064b891187b75ca32Chet Haase 102ad93c2bb63dfc813b2eefa1043aa63afbddce655Chet Haasevoid ResourceCache::decrementRefcount(SkiaColorFilter* filterResource) { 103ad93c2bb63dfc813b2eefa1043aa63afbddce655Chet Haase filterResource->getSkColorFilter()->safeUnref(); 104ad93c2bb63dfc813b2eefa1043aa63afbddce655Chet Haase decrementRefcount((void*)filterResource); 105ad93c2bb63dfc813b2eefa1043aa63afbddce655Chet Haase} 106ad93c2bb63dfc813b2eefa1043aa63afbddce655Chet Haase 1075c13d89c1332fcc499379b9064b891187b75ca32Chet Haasevoid ResourceCache::recycle(SkBitmap* resource) { 108e7d2295c06ef9b9df6336cbff23007a13fb3f6e4Chet Haase Mutex::Autolock _l(mLock); 1095c13d89c1332fcc499379b9064b891187b75ca32Chet Haase if (mCache->indexOfKey(resource) < 0) { 1105c13d89c1332fcc499379b9064b891187b75ca32Chet Haase // not tracking this resource; just recycle the pixel data 1115c13d89c1332fcc499379b9064b891187b75ca32Chet Haase resource->setPixels(NULL, NULL); 1125c13d89c1332fcc499379b9064b891187b75ca32Chet Haase return; 1135c13d89c1332fcc499379b9064b891187b75ca32Chet Haase } 1145c13d89c1332fcc499379b9064b891187b75ca32Chet Haase recycle((void*) resource); 1155c13d89c1332fcc499379b9064b891187b75ca32Chet Haase} 1165c13d89c1332fcc499379b9064b891187b75ca32Chet Haase 1175c13d89c1332fcc499379b9064b891187b75ca32Chet Haasevoid ResourceCache::recycle(void* resource) { 118e7d2295c06ef9b9df6336cbff23007a13fb3f6e4Chet Haase Mutex::Autolock _l(mLock); 1195c13d89c1332fcc499379b9064b891187b75ca32Chet Haase ResourceReference* ref = mCache->indexOfKey(resource) >= 0 ? mCache->valueFor(resource) : NULL; 1205c13d89c1332fcc499379b9064b891187b75ca32Chet Haase if (ref == NULL) { 1215c13d89c1332fcc499379b9064b891187b75ca32Chet Haase // Should not get here - shouldn't get a call to recycle if we're not yet tracking it 1225c13d89c1332fcc499379b9064b891187b75ca32Chet Haase return; 1235c13d89c1332fcc499379b9064b891187b75ca32Chet Haase } 1245c13d89c1332fcc499379b9064b891187b75ca32Chet Haase ref->recycled = true; 1255c13d89c1332fcc499379b9064b891187b75ca32Chet Haase if (ref->refCount == 0) { 1265c13d89c1332fcc499379b9064b891187b75ca32Chet Haase deleteResourceReference(resource, ref); 1275c13d89c1332fcc499379b9064b891187b75ca32Chet Haase } 1285c13d89c1332fcc499379b9064b891187b75ca32Chet Haase} 1295c13d89c1332fcc499379b9064b891187b75ca32Chet Haase 1305c13d89c1332fcc499379b9064b891187b75ca32Chet Haasevoid ResourceCache::destructor(SkBitmap* resource) { 131e7d2295c06ef9b9df6336cbff23007a13fb3f6e4Chet Haase Mutex::Autolock _l(mLock); 1325c13d89c1332fcc499379b9064b891187b75ca32Chet Haase ResourceReference* ref = mCache->indexOfKey(resource) >= 0 ? mCache->valueFor(resource) : NULL; 1335c13d89c1332fcc499379b9064b891187b75ca32Chet Haase if (ref == NULL) { 1345c13d89c1332fcc499379b9064b891187b75ca32Chet Haase // If we're not tracking this resource, just delete it 1355c13d89c1332fcc499379b9064b891187b75ca32Chet Haase if (Caches::hasInstance()) { 136fe48f65922d4a3cc4aefe058cee5acec51504a20Romain Guy Caches::getInstance().textureCache.removeDeferred(resource); 1375c13d89c1332fcc499379b9064b891187b75ca32Chet Haase } 1385c13d89c1332fcc499379b9064b891187b75ca32Chet Haase delete resource; 1395c13d89c1332fcc499379b9064b891187b75ca32Chet Haase return; 1405c13d89c1332fcc499379b9064b891187b75ca32Chet Haase } 1415c13d89c1332fcc499379b9064b891187b75ca32Chet Haase ref->destroyed = true; 1425c13d89c1332fcc499379b9064b891187b75ca32Chet Haase if (ref->refCount == 0) { 1435c13d89c1332fcc499379b9064b891187b75ca32Chet Haase deleteResourceReference(resource, ref); 1445c13d89c1332fcc499379b9064b891187b75ca32Chet Haase return; 1455c13d89c1332fcc499379b9064b891187b75ca32Chet Haase } 1465c13d89c1332fcc499379b9064b891187b75ca32Chet Haase} 1475c13d89c1332fcc499379b9064b891187b75ca32Chet Haase 1485c13d89c1332fcc499379b9064b891187b75ca32Chet Haasevoid ResourceCache::destructor(SkiaShader* resource) { 149e7d2295c06ef9b9df6336cbff23007a13fb3f6e4Chet Haase Mutex::Autolock _l(mLock); 1505c13d89c1332fcc499379b9064b891187b75ca32Chet Haase ResourceReference* ref = mCache->indexOfKey(resource) >= 0 ? mCache->valueFor(resource) : NULL; 1515c13d89c1332fcc499379b9064b891187b75ca32Chet Haase if (ref == NULL) { 1525c13d89c1332fcc499379b9064b891187b75ca32Chet Haase // If we're not tracking this resource, just delete it 1535c13d89c1332fcc499379b9064b891187b75ca32Chet Haase if (Caches::hasInstance()) { 154fe48f65922d4a3cc4aefe058cee5acec51504a20Romain Guy Caches::getInstance().gradientCache.removeDeferred(resource->getSkShader()); 1555c13d89c1332fcc499379b9064b891187b75ca32Chet Haase } 1565c13d89c1332fcc499379b9064b891187b75ca32Chet Haase delete resource; 1575c13d89c1332fcc499379b9064b891187b75ca32Chet Haase return; 1585c13d89c1332fcc499379b9064b891187b75ca32Chet Haase } 1595c13d89c1332fcc499379b9064b891187b75ca32Chet Haase ref->destroyed = true; 1605c13d89c1332fcc499379b9064b891187b75ca32Chet Haase if (ref->refCount == 0) { 1615c13d89c1332fcc499379b9064b891187b75ca32Chet Haase deleteResourceReference(resource, ref); 1625c13d89c1332fcc499379b9064b891187b75ca32Chet Haase return; 1635c13d89c1332fcc499379b9064b891187b75ca32Chet Haase } 1645c13d89c1332fcc499379b9064b891187b75ca32Chet Haase} 1655c13d89c1332fcc499379b9064b891187b75ca32Chet Haase 166ad93c2bb63dfc813b2eefa1043aa63afbddce655Chet Haasevoid ResourceCache::destructor(SkiaColorFilter* resource) { 167e7d2295c06ef9b9df6336cbff23007a13fb3f6e4Chet Haase Mutex::Autolock _l(mLock); 168ad93c2bb63dfc813b2eefa1043aa63afbddce655Chet Haase ResourceReference* ref = mCache->indexOfKey(resource) >= 0 ? mCache->valueFor(resource) : NULL; 169ad93c2bb63dfc813b2eefa1043aa63afbddce655Chet Haase if (ref == NULL) { 170ad93c2bb63dfc813b2eefa1043aa63afbddce655Chet Haase // If we're not tracking this resource, just delete it 171ad93c2bb63dfc813b2eefa1043aa63afbddce655Chet Haase delete resource; 172ad93c2bb63dfc813b2eefa1043aa63afbddce655Chet Haase return; 173ad93c2bb63dfc813b2eefa1043aa63afbddce655Chet Haase } 174ad93c2bb63dfc813b2eefa1043aa63afbddce655Chet Haase ref->destroyed = true; 175ad93c2bb63dfc813b2eefa1043aa63afbddce655Chet Haase if (ref->refCount == 0) { 176ad93c2bb63dfc813b2eefa1043aa63afbddce655Chet Haase deleteResourceReference(resource, ref); 177ad93c2bb63dfc813b2eefa1043aa63afbddce655Chet Haase return; 178ad93c2bb63dfc813b2eefa1043aa63afbddce655Chet Haase } 179ad93c2bb63dfc813b2eefa1043aa63afbddce655Chet Haase} 180ad93c2bb63dfc813b2eefa1043aa63afbddce655Chet Haase 181e7d2295c06ef9b9df6336cbff23007a13fb3f6e4Chet Haase/** 182e7d2295c06ef9b9df6336cbff23007a13fb3f6e4Chet Haase * This method should only be called while the mLock mutex is held (that mutex is grabbed 183e7d2295c06ef9b9df6336cbff23007a13fb3f6e4Chet Haase * by the various destructor() and recycle() methods which call this method). 184e7d2295c06ef9b9df6336cbff23007a13fb3f6e4Chet Haase */ 1855c13d89c1332fcc499379b9064b891187b75ca32Chet Haasevoid ResourceCache::deleteResourceReference(void* resource, ResourceReference* ref) { 1865c13d89c1332fcc499379b9064b891187b75ca32Chet Haase if (ref->recycled && ref->resourceType == kBitmap) { 1875c13d89c1332fcc499379b9064b891187b75ca32Chet Haase ((SkBitmap*) resource)->setPixels(NULL, NULL); 1885c13d89c1332fcc499379b9064b891187b75ca32Chet Haase } 1895c13d89c1332fcc499379b9064b891187b75ca32Chet Haase if (ref->destroyed) { 1905c13d89c1332fcc499379b9064b891187b75ca32Chet Haase switch (ref->resourceType) { 1915c13d89c1332fcc499379b9064b891187b75ca32Chet Haase case kBitmap: 1925c13d89c1332fcc499379b9064b891187b75ca32Chet Haase { 1935c13d89c1332fcc499379b9064b891187b75ca32Chet Haase SkBitmap* bitmap = (SkBitmap*)resource; 1945c13d89c1332fcc499379b9064b891187b75ca32Chet Haase if (Caches::hasInstance()) { 195fe48f65922d4a3cc4aefe058cee5acec51504a20Romain Guy Caches::getInstance().textureCache.removeDeferred(bitmap); 1965c13d89c1332fcc499379b9064b891187b75ca32Chet Haase } 1975c13d89c1332fcc499379b9064b891187b75ca32Chet Haase delete bitmap; 1985c13d89c1332fcc499379b9064b891187b75ca32Chet Haase } 1995c13d89c1332fcc499379b9064b891187b75ca32Chet Haase break; 2005c13d89c1332fcc499379b9064b891187b75ca32Chet Haase case kShader: 201ad93c2bb63dfc813b2eefa1043aa63afbddce655Chet Haase { 2025c13d89c1332fcc499379b9064b891187b75ca32Chet Haase SkiaShader* shader = (SkiaShader*)resource; 2035c13d89c1332fcc499379b9064b891187b75ca32Chet Haase if (Caches::hasInstance()) { 204fe48f65922d4a3cc4aefe058cee5acec51504a20Romain Guy Caches::getInstance().gradientCache.removeDeferred(shader->getSkShader()); 2055c13d89c1332fcc499379b9064b891187b75ca32Chet Haase } 2065c13d89c1332fcc499379b9064b891187b75ca32Chet Haase delete shader; 207ad93c2bb63dfc813b2eefa1043aa63afbddce655Chet Haase } 208ad93c2bb63dfc813b2eefa1043aa63afbddce655Chet Haase break; 209ad93c2bb63dfc813b2eefa1043aa63afbddce655Chet Haase case kColorFilter: 210ad93c2bb63dfc813b2eefa1043aa63afbddce655Chet Haase { 211ad93c2bb63dfc813b2eefa1043aa63afbddce655Chet Haase SkiaColorFilter* filter = (SkiaColorFilter*)resource; 212ad93c2bb63dfc813b2eefa1043aa63afbddce655Chet Haase delete filter; 213ad93c2bb63dfc813b2eefa1043aa63afbddce655Chet Haase } 214ad93c2bb63dfc813b2eefa1043aa63afbddce655Chet Haase break; 2155c13d89c1332fcc499379b9064b891187b75ca32Chet Haase } 2165c13d89c1332fcc499379b9064b891187b75ca32Chet Haase } 2175c13d89c1332fcc499379b9064b891187b75ca32Chet Haase mCache->removeItem(resource); 2185c13d89c1332fcc499379b9064b891187b75ca32Chet Haase delete ref; 2195c13d89c1332fcc499379b9064b891187b75ca32Chet Haase} 2205c13d89c1332fcc499379b9064b891187b75ca32Chet Haase 2215c13d89c1332fcc499379b9064b891187b75ca32Chet Haase}; // namespace uirenderer 2225c13d89c1332fcc499379b9064b891187b75ca32Chet Haase}; // namespace android 223