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