2013-08-08 22:55:21 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2013 Google Inc.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#if SK_SUPPORT_GPU
|
2014-01-10 22:08:27 +00:00
|
|
|
|
2013-08-08 22:55:21 +00:00
|
|
|
#include "GrContextFactory.h"
|
2014-01-15 23:09:01 +00:00
|
|
|
#include "GrResourceCache.h"
|
2013-08-08 22:55:21 +00:00
|
|
|
#include "SkGpuDevice.h"
|
2014-01-10 22:08:27 +00:00
|
|
|
#include "Test.h"
|
2013-08-08 22:55:21 +00:00
|
|
|
|
|
|
|
static const int gWidth = 640;
|
|
|
|
static const int gHeight = 480;
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
2013-08-09 07:01:22 +00:00
|
|
|
static void test_cache(skiatest::Reporter* reporter,
|
2013-08-08 22:55:21 +00:00
|
|
|
GrContext* context,
|
|
|
|
SkCanvas* canvas) {
|
|
|
|
const SkIRect size = SkIRect::MakeWH(gWidth, gHeight);
|
|
|
|
|
|
|
|
SkBitmap src;
|
2014-02-13 14:41:43 +00:00
|
|
|
src.allocN32Pixels(size.width(), size.height());
|
2013-08-08 22:55:21 +00:00
|
|
|
src.eraseColor(SK_ColorBLACK);
|
|
|
|
size_t srcSize = src.getSize();
|
|
|
|
|
|
|
|
size_t initialCacheSize = context->getGpuTextureCacheBytes();
|
|
|
|
|
|
|
|
int oldMaxNum;
|
|
|
|
size_t oldMaxBytes;
|
|
|
|
context->getTextureCacheLimits(&oldMaxNum, &oldMaxBytes);
|
2013-08-09 07:01:22 +00:00
|
|
|
|
2013-08-08 22:55:21 +00:00
|
|
|
// Set the cache limits so we can fit 10 "src" images and the
|
|
|
|
// max number of textures doesn't matter
|
|
|
|
size_t maxCacheSize = initialCacheSize + 10*srcSize;
|
|
|
|
context->setTextureCacheLimits(1000, maxCacheSize);
|
|
|
|
|
|
|
|
SkBitmap readback;
|
2014-02-13 14:41:43 +00:00
|
|
|
readback.allocN32Pixels(size.width(), size.height());
|
2013-08-08 22:55:21 +00:00
|
|
|
|
|
|
|
for (int i = 0; i < 100; ++i) {
|
|
|
|
canvas->drawBitmap(src, 0, 0);
|
|
|
|
canvas->readPixels(size, &readback);
|
|
|
|
|
|
|
|
// "modify" the src texture
|
|
|
|
src.notifyPixelsChanged();
|
|
|
|
|
|
|
|
size_t curCacheSize = context->getGpuTextureCacheBytes();
|
|
|
|
|
|
|
|
// we should never go over the size limit
|
|
|
|
REPORTER_ASSERT(reporter, curCacheSize <= maxCacheSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
context->setTextureCacheLimits(oldMaxNum, oldMaxBytes);
|
|
|
|
}
|
|
|
|
|
2014-05-02 21:38:22 +00:00
|
|
|
class TestResource : public GrCacheable {
|
2014-05-05 19:09:13 +00:00
|
|
|
static const size_t kDefaultSize = 100;
|
|
|
|
|
2014-01-15 23:09:01 +00:00
|
|
|
public:
|
|
|
|
SK_DECLARE_INST_COUNT(TestResource);
|
2014-05-05 19:09:13 +00:00
|
|
|
TestResource(size_t size = kDefaultSize)
|
2014-05-02 21:38:22 +00:00
|
|
|
: fCache(NULL)
|
2014-05-05 19:09:13 +00:00
|
|
|
, fToDelete(NULL)
|
|
|
|
, fSize(size) {
|
2014-01-15 23:09:01 +00:00
|
|
|
++fAlive;
|
|
|
|
}
|
|
|
|
|
|
|
|
~TestResource() {
|
|
|
|
--fAlive;
|
|
|
|
if (NULL != fToDelete) {
|
|
|
|
// Breaks our little 2-element cycle below.
|
|
|
|
fToDelete->setDeleteWhenDestroyed(NULL, NULL);
|
|
|
|
fCache->deleteResource(fToDelete->getCacheEntry());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-05 19:09:13 +00:00
|
|
|
void setSize(size_t size) {
|
|
|
|
fSize = size;
|
|
|
|
this->didChangeGpuMemorySize();
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t gpuMemorySize() const SK_OVERRIDE { return fSize; }
|
2014-05-02 21:38:22 +00:00
|
|
|
|
|
|
|
bool isValidOnGpu() const SK_OVERRIDE { return true; }
|
2014-01-15 23:09:01 +00:00
|
|
|
|
|
|
|
static int alive() { return fAlive; }
|
|
|
|
|
|
|
|
void setDeleteWhenDestroyed(GrResourceCache* cache, TestResource* resource) {
|
|
|
|
fCache = cache;
|
|
|
|
fToDelete = resource;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
GrResourceCache* fCache;
|
|
|
|
TestResource* fToDelete;
|
2014-05-05 19:09:13 +00:00
|
|
|
size_t fSize;
|
2014-01-15 23:09:01 +00:00
|
|
|
static int fAlive;
|
|
|
|
|
2014-05-02 21:38:22 +00:00
|
|
|
typedef GrCacheable INHERITED;
|
2014-01-15 23:09:01 +00:00
|
|
|
};
|
|
|
|
int TestResource::fAlive = 0;
|
|
|
|
|
|
|
|
static void test_purge_invalidated(skiatest::Reporter* reporter, GrContext* context) {
|
|
|
|
GrCacheID::Domain domain = GrCacheID::GenerateDomain();
|
|
|
|
GrCacheID::Key keyData;
|
|
|
|
keyData.fData64[0] = 5;
|
|
|
|
keyData.fData64[1] = 18;
|
|
|
|
GrResourceKey::ResourceType t = GrResourceKey::GenerateResourceType();
|
|
|
|
GrResourceKey key(GrCacheID(domain, keyData), t, 0);
|
|
|
|
|
|
|
|
GrResourceCache cache(5, 30000);
|
|
|
|
|
|
|
|
// Add two resources with the same key that delete each other from the cache when destroyed.
|
2014-05-02 21:38:22 +00:00
|
|
|
TestResource* a = new TestResource();
|
|
|
|
TestResource* b = new TestResource();
|
2014-01-15 23:09:01 +00:00
|
|
|
cache.addResource(key, a);
|
|
|
|
cache.addResource(key, b);
|
|
|
|
// Circle back.
|
|
|
|
a->setDeleteWhenDestroyed(&cache, b);
|
|
|
|
b->setDeleteWhenDestroyed(&cache, a);
|
|
|
|
a->unref();
|
|
|
|
b->unref();
|
|
|
|
|
|
|
|
// Add a third independent resource also with the same key.
|
2014-05-02 21:38:22 +00:00
|
|
|
GrCacheable* r = new TestResource();
|
2014-01-15 23:09:01 +00:00
|
|
|
cache.addResource(key, r);
|
|
|
|
r->unref();
|
|
|
|
|
|
|
|
// Invalidate all three, all three should be purged and destroyed.
|
|
|
|
REPORTER_ASSERT(reporter, 3 == TestResource::alive());
|
|
|
|
const GrResourceInvalidatedMessage msg = { key };
|
|
|
|
SkMessageBus<GrResourceInvalidatedMessage>::Post(msg);
|
|
|
|
cache.purgeAsNeeded();
|
|
|
|
REPORTER_ASSERT(reporter, 0 == TestResource::alive());
|
|
|
|
}
|
|
|
|
|
2014-01-17 17:56:21 +00:00
|
|
|
static void test_cache_delete_on_destruction(skiatest::Reporter* reporter,
|
|
|
|
GrContext* context) {
|
|
|
|
GrCacheID::Domain domain = GrCacheID::GenerateDomain();
|
|
|
|
GrCacheID::Key keyData;
|
|
|
|
keyData.fData64[0] = 5;
|
|
|
|
keyData.fData64[1] = 0;
|
|
|
|
GrResourceKey::ResourceType t = GrResourceKey::GenerateResourceType();
|
|
|
|
|
|
|
|
GrResourceKey key(GrCacheID(domain, keyData), t, 0);
|
|
|
|
|
|
|
|
{
|
|
|
|
{
|
|
|
|
GrResourceCache cache(3, 30000);
|
2014-05-02 21:38:22 +00:00
|
|
|
TestResource* a = new TestResource();
|
|
|
|
TestResource* b = new TestResource();
|
2014-01-17 17:56:21 +00:00
|
|
|
cache.addResource(key, a);
|
|
|
|
cache.addResource(key, b);
|
|
|
|
|
|
|
|
a->setDeleteWhenDestroyed(&cache, b);
|
|
|
|
b->setDeleteWhenDestroyed(&cache, a);
|
|
|
|
|
|
|
|
a->unref();
|
|
|
|
b->unref();
|
|
|
|
REPORTER_ASSERT(reporter, 2 == TestResource::alive());
|
|
|
|
}
|
|
|
|
REPORTER_ASSERT(reporter, 0 == TestResource::alive());
|
|
|
|
}
|
|
|
|
{
|
|
|
|
GrResourceCache cache(3, 30000);
|
2014-05-02 21:38:22 +00:00
|
|
|
TestResource* a = new TestResource();
|
|
|
|
TestResource* b = new TestResource();
|
2014-01-17 17:56:21 +00:00
|
|
|
cache.addResource(key, a);
|
|
|
|
cache.addResource(key, b);
|
|
|
|
|
|
|
|
a->setDeleteWhenDestroyed(&cache, b);
|
|
|
|
b->setDeleteWhenDestroyed(&cache, a);
|
|
|
|
|
|
|
|
a->unref();
|
|
|
|
b->unref();
|
|
|
|
|
|
|
|
cache.deleteResource(a->getCacheEntry());
|
|
|
|
|
|
|
|
REPORTER_ASSERT(reporter, 0 == TestResource::alive());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-05 19:09:13 +00:00
|
|
|
static void test_resource_size_changed(skiatest::Reporter* reporter,
|
|
|
|
GrContext* context) {
|
|
|
|
GrCacheID::Domain domain = GrCacheID::GenerateDomain();
|
|
|
|
GrResourceKey::ResourceType t = GrResourceKey::GenerateResourceType();
|
|
|
|
|
|
|
|
GrCacheID::Key key1Data;
|
|
|
|
key1Data.fData64[0] = 0;
|
|
|
|
key1Data.fData64[1] = 0;
|
|
|
|
GrResourceKey key1(GrCacheID(domain, key1Data), t, 0);
|
|
|
|
|
|
|
|
GrCacheID::Key key2Data;
|
|
|
|
key2Data.fData64[0] = 1;
|
|
|
|
key2Data.fData64[1] = 0;
|
|
|
|
GrResourceKey key2(GrCacheID(domain, key2Data), t, 0);
|
|
|
|
|
|
|
|
// Test changing resources sizes (both increase & decrease).
|
|
|
|
{
|
|
|
|
GrResourceCache cache(2, 300);
|
|
|
|
|
|
|
|
TestResource* a = new TestResource(0);
|
|
|
|
a->setSize(100); // Test didChangeGpuMemorySize() when not in the cache.
|
|
|
|
cache.addResource(key1, a);
|
|
|
|
a->unref();
|
|
|
|
|
|
|
|
TestResource* b = new TestResource(0);
|
|
|
|
b->setSize(100);
|
|
|
|
cache.addResource(key2, b);
|
|
|
|
b->unref();
|
|
|
|
|
|
|
|
REPORTER_ASSERT(reporter, 200 == cache.getCachedResourceBytes());
|
|
|
|
REPORTER_ASSERT(reporter, 2 == cache.getCachedResourceCount());
|
|
|
|
|
|
|
|
static_cast<TestResource*>(cache.find(key2))->setSize(200);
|
|
|
|
static_cast<TestResource*>(cache.find(key1))->setSize(50);
|
|
|
|
|
|
|
|
REPORTER_ASSERT(reporter, 250 == cache.getCachedResourceBytes());
|
|
|
|
REPORTER_ASSERT(reporter, 2 == cache.getCachedResourceCount());
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test increasing a resources size beyond the cache budget.
|
|
|
|
{
|
|
|
|
GrResourceCache cache(2, 300);
|
|
|
|
|
|
|
|
TestResource* a = new TestResource(100);
|
|
|
|
cache.addResource(key1, a);
|
|
|
|
a->unref();
|
|
|
|
|
|
|
|
TestResource* b = new TestResource(100);
|
|
|
|
cache.addResource(key2, b);
|
|
|
|
b->unref();
|
|
|
|
|
|
|
|
REPORTER_ASSERT(reporter, 200 == cache.getCachedResourceBytes());
|
|
|
|
REPORTER_ASSERT(reporter, 2 == cache.getCachedResourceCount());
|
|
|
|
|
|
|
|
static_cast<TestResource*>(cache.find(key2))->setSize(201);
|
|
|
|
REPORTER_ASSERT(reporter, NULL == cache.find(key1));
|
|
|
|
|
|
|
|
REPORTER_ASSERT(reporter, 201 == cache.getCachedResourceBytes());
|
|
|
|
REPORTER_ASSERT(reporter, 1 == cache.getCachedResourceCount());
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test changing the size of an exclusively-held resource.
|
|
|
|
{
|
|
|
|
GrResourceCache cache(2, 300);
|
|
|
|
|
|
|
|
TestResource* a = new TestResource(100);
|
|
|
|
cache.addResource(key1, a);
|
|
|
|
cache.makeExclusive(a->getCacheEntry());
|
|
|
|
|
|
|
|
TestResource* b = new TestResource(100);
|
|
|
|
cache.addResource(key2, b);
|
|
|
|
b->unref();
|
|
|
|
|
|
|
|
REPORTER_ASSERT(reporter, 200 == cache.getCachedResourceBytes());
|
|
|
|
REPORTER_ASSERT(reporter, 2 == cache.getCachedResourceCount());
|
|
|
|
REPORTER_ASSERT(reporter, NULL == cache.find(key1));
|
|
|
|
|
|
|
|
a->setSize(200);
|
|
|
|
|
|
|
|
REPORTER_ASSERT(reporter, 300 == cache.getCachedResourceBytes());
|
|
|
|
REPORTER_ASSERT(reporter, 2 == cache.getCachedResourceCount());
|
|
|
|
// Internal resource cache validation will test the detached size (debug mode only).
|
|
|
|
|
|
|
|
cache.makeNonExclusive(a->getCacheEntry());
|
|
|
|
a->unref();
|
|
|
|
|
|
|
|
REPORTER_ASSERT(reporter, 300 == cache.getCachedResourceBytes());
|
|
|
|
REPORTER_ASSERT(reporter, 2 == cache.getCachedResourceCount());
|
|
|
|
REPORTER_ASSERT(reporter, NULL != cache.find(key1));
|
|
|
|
// Internal resource cache validation will test the detached size (debug mode only).
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-01-15 23:09:01 +00:00
|
|
|
////////////////////////////////////////////////////////////////////////////////
|
2014-01-10 22:08:27 +00:00
|
|
|
DEF_GPUTEST(ResourceCache, reporter, factory) {
|
2013-08-08 22:55:21 +00:00
|
|
|
for (int type = 0; type < GrContextFactory::kLastGLContextType; ++type) {
|
|
|
|
GrContextFactory::GLContextType glType = static_cast<GrContextFactory::GLContextType>(type);
|
|
|
|
if (!GrContextFactory::IsRenderingGLContext(glType)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
GrContext* context = factory->get(glType);
|
|
|
|
if (NULL == context) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
GrTextureDesc desc;
|
|
|
|
desc.fConfig = kSkia8888_GrPixelConfig;
|
|
|
|
desc.fFlags = kRenderTarget_GrTextureFlagBit;
|
|
|
|
desc.fWidth = gWidth;
|
|
|
|
desc.fHeight = gHeight;
|
|
|
|
|
|
|
|
SkAutoTUnref<GrTexture> texture(context->createUncachedTexture(desc, NULL, 0));
|
|
|
|
SkAutoTUnref<SkGpuDevice> device(SkNEW_ARGS(SkGpuDevice, (context, texture.get())));
|
|
|
|
SkCanvas canvas(device.get());
|
|
|
|
|
|
|
|
test_cache(reporter, context, &canvas);
|
2014-01-15 23:09:01 +00:00
|
|
|
test_purge_invalidated(reporter, context);
|
2014-01-17 17:56:21 +00:00
|
|
|
test_cache_delete_on_destruction(reporter, context);
|
2014-05-05 19:09:13 +00:00
|
|
|
test_resource_size_changed(reporter, context);
|
2013-08-08 22:55:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|