2013-08-21 19:27:48 +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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "GrTest.h"
|
2015-07-28 16:58:39 +00:00
|
|
|
|
2015-08-21 18:08:00 +00:00
|
|
|
#include "GrBatchAtlas.h"
|
2015-08-19 15:16:43 +00:00
|
|
|
#include "GrBatchFontCache.h"
|
2015-05-22 21:01:46 +00:00
|
|
|
#include "GrContextOptions.h"
|
2015-11-08 16:07:24 +00:00
|
|
|
#include "GrDrawContext.h"
|
2015-10-17 14:43:27 +00:00
|
|
|
#include "GrDrawingManager.h"
|
2015-02-13 22:20:05 +00:00
|
|
|
#include "GrGpuResourceCacheAccess.h"
|
2015-02-11 18:49:59 +00:00
|
|
|
#include "GrResourceCache.h"
|
2015-08-03 17:17:44 +00:00
|
|
|
#include "GrTextBlobCache.h"
|
2015-11-08 16:07:24 +00:00
|
|
|
#include "SkGrPriv.h"
|
2015-02-03 02:26:03 +00:00
|
|
|
#include "SkString.h"
|
2013-08-21 19:27:48 +00:00
|
|
|
|
2015-08-21 18:08:00 +00:00
|
|
|
namespace GrTest {
|
|
|
|
void SetupAlwaysEvictAtlas(GrContext* context) {
|
|
|
|
// These sizes were selected because they allow each atlas to hold a single plot and will thus
|
|
|
|
// stress the atlas
|
|
|
|
int dim = GrBatchAtlas::kGlyphMaxDim;
|
|
|
|
GrBatchAtlasConfig configs[3];
|
|
|
|
configs[kA8_GrMaskFormat].fWidth = dim;
|
|
|
|
configs[kA8_GrMaskFormat].fHeight = dim;
|
|
|
|
configs[kA8_GrMaskFormat].fPlotWidth = dim;
|
|
|
|
configs[kA8_GrMaskFormat].fPlotHeight = dim;
|
|
|
|
|
|
|
|
configs[kA565_GrMaskFormat].fWidth = dim;
|
|
|
|
configs[kA565_GrMaskFormat].fHeight = dim;
|
|
|
|
configs[kA565_GrMaskFormat].fPlotWidth = dim;
|
|
|
|
configs[kA565_GrMaskFormat].fPlotHeight = dim;
|
|
|
|
|
|
|
|
configs[kARGB_GrMaskFormat].fWidth = dim;
|
|
|
|
configs[kARGB_GrMaskFormat].fHeight = dim;
|
|
|
|
configs[kARGB_GrMaskFormat].fPlotWidth = dim;
|
|
|
|
configs[kARGB_GrMaskFormat].fPlotHeight = dim;
|
|
|
|
|
|
|
|
context->setTextContextAtlasSizes_ForTesting(configs);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2015-11-16 14:23:31 +00:00
|
|
|
void GrTestTarget::init(GrContext* ctx, GrDrawTarget* target, GrRenderTarget* rt) {
|
2013-08-21 19:27:48 +00:00
|
|
|
SkASSERT(!fContext);
|
|
|
|
|
|
|
|
fContext.reset(SkRef(ctx));
|
|
|
|
fDrawTarget.reset(SkRef(target));
|
2015-11-16 14:23:31 +00:00
|
|
|
fRenderTarget.reset(SkRef(rt));
|
2013-08-21 19:27:48 +00:00
|
|
|
}
|
|
|
|
|
2015-11-16 19:02:05 +00:00
|
|
|
void GrContext::getTestTarget(GrTestTarget* tar, GrRenderTarget* rt) {
|
2013-08-21 19:27:48 +00:00
|
|
|
this->flush();
|
|
|
|
// We could create a proxy GrDrawTarget that passes through to fGpu until ~GrTextTarget() and
|
|
|
|
// then disconnects. This would help prevent test writers from mixing using the returned
|
|
|
|
// GrDrawTarget and regular drawing. We could also assert or fail in GrContext drawing methods
|
|
|
|
// until ~GrTestTarget().
|
2015-11-16 19:02:05 +00:00
|
|
|
if (!rt) {
|
|
|
|
GrSurfaceDesc desc;
|
|
|
|
desc.fFlags = kRenderTarget_GrSurfaceFlag;
|
|
|
|
desc.fWidth = 32;
|
|
|
|
desc.fHeight = 32;
|
|
|
|
desc.fConfig = kRGBA_8888_GrPixelConfig;
|
|
|
|
desc.fSampleCnt = 0;
|
|
|
|
|
|
|
|
SkAutoTUnref<GrTexture> texture(this->textureProvider()->createTexture(desc, false,
|
|
|
|
nullptr, 0));
|
|
|
|
if (nullptr == texture) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
SkASSERT(nullptr != texture->asRenderTarget());
|
|
|
|
rt = texture->asRenderTarget();
|
2015-11-16 14:23:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SkAutoTUnref<GrDrawTarget> dt(fDrawingManager->newDrawTarget(rt));
|
|
|
|
tar->init(this, dt, rt);
|
2013-08-21 19:27:48 +00:00
|
|
|
}
|
|
|
|
|
2015-08-03 17:17:44 +00:00
|
|
|
void GrContext::setTextBlobCacheLimit_ForTesting(size_t bytes) {
|
|
|
|
fTextBlobCache->setBudget(bytes);
|
|
|
|
}
|
|
|
|
|
2015-08-19 15:16:43 +00:00
|
|
|
void GrContext::setTextContextAtlasSizes_ForTesting(const GrBatchAtlasConfig* configs) {
|
|
|
|
fBatchFontCache->setAtlasSizes_ForTesting(configs);
|
|
|
|
}
|
|
|
|
|
2013-08-21 19:27:48 +00:00
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
2013-12-18 17:25:33 +00:00
|
|
|
void GrContext::purgeAllUnlockedResources() {
|
2015-02-11 18:49:59 +00:00
|
|
|
fResourceCache->purgeAllUnlocked();
|
2013-12-18 17:25:33 +00:00
|
|
|
}
|
2014-11-05 22:47:41 +00:00
|
|
|
|
2015-02-03 02:26:03 +00:00
|
|
|
void GrContext::dumpCacheStats(SkString* out) const {
|
|
|
|
#if GR_CACHE_STATS
|
2015-02-11 18:49:59 +00:00
|
|
|
fResourceCache->dumpStats(out);
|
2015-02-03 02:26:03 +00:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void GrContext::printCacheStats() const {
|
|
|
|
SkString out;
|
|
|
|
this->dumpCacheStats(&out);
|
2015-02-19 14:32:12 +00:00
|
|
|
SkDebugf("%s", out.c_str());
|
2015-02-03 02:26:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GrContext::dumpGpuStats(SkString* out) const {
|
|
|
|
#if GR_GPU_STATS
|
|
|
|
return fGpu->stats()->dump(out);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void GrContext::printGpuStats() const {
|
|
|
|
SkString out;
|
|
|
|
this->dumpGpuStats(&out);
|
2015-02-19 14:32:12 +00:00
|
|
|
SkDebugf("%s", out.c_str());
|
2015-02-03 02:26:03 +00:00
|
|
|
}
|
|
|
|
|
2015-11-08 16:07:24 +00:00
|
|
|
void GrContext::drawFontCache(const SkRect& rect, GrMaskFormat format, const SkPaint& paint,
|
|
|
|
GrRenderTarget* target) {
|
|
|
|
GrBatchFontCache* cache = this->getBatchFontCache();
|
|
|
|
|
|
|
|
GrTexture* atlas = cache->getTexture(format);
|
|
|
|
|
|
|
|
SkAutoTUnref<GrDrawContext> drawContext(this->drawContext(target));
|
|
|
|
// TODO: add drawContext method to encapsulate this.
|
|
|
|
|
|
|
|
GrPaint grPaint;
|
|
|
|
SkMatrix mat;
|
|
|
|
mat.reset();
|
|
|
|
if (!SkPaintToGrPaint(this, paint, mat, &grPaint)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
SkMatrix textureMat;
|
|
|
|
textureMat.reset();
|
|
|
|
// TODO: use setScaleTranslate()
|
|
|
|
textureMat[SkMatrix::kMScaleX] = 1.0f/rect.width();
|
|
|
|
textureMat[SkMatrix::kMScaleY] = 1.0f/rect.height();
|
|
|
|
textureMat[SkMatrix::kMTransX] = -rect.fLeft/rect.width();
|
|
|
|
textureMat[SkMatrix::kMTransY] = -rect.fTop/rect.height();
|
|
|
|
|
|
|
|
grPaint.addColorTextureProcessor(atlas, textureMat);
|
|
|
|
|
|
|
|
GrClip clip;
|
|
|
|
drawContext->drawRect(clip, grPaint, mat, rect);
|
|
|
|
}
|
|
|
|
|
2015-02-03 02:26:03 +00:00
|
|
|
#if GR_GPU_STATS
|
|
|
|
void GrGpu::Stats::dump(SkString* out) {
|
|
|
|
out->appendf("Render Target Binds: %d\n", fRenderTargetBinds);
|
|
|
|
out->appendf("Shader Compilations: %d\n", fShaderCompilations);
|
2015-02-03 05:19:50 +00:00
|
|
|
out->appendf("Textures Created: %d\n", fTextureCreates);
|
|
|
|
out->appendf("Texture Uploads: %d\n", fTextureUploads);
|
2015-04-16 18:22:42 +00:00
|
|
|
out->appendf("Stencil Buffer Creates: %d\n", fStencilAttachmentCreates);
|
2015-09-08 20:42:05 +00:00
|
|
|
out->appendf("Number of draws: %d\n", fNumDraws);
|
2015-02-03 02:26:03 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if GR_CACHE_STATS
|
2015-11-09 21:51:06 +00:00
|
|
|
void GrResourceCache::getStats(Stats* stats) const {
|
|
|
|
stats->reset();
|
2015-02-03 02:26:03 +00:00
|
|
|
|
2015-11-09 21:51:06 +00:00
|
|
|
stats->fTotal = this->getResourceCount();
|
|
|
|
stats->fNumNonPurgeable = fNonpurgeableResources.count();
|
|
|
|
stats->fNumPurgeable = fPurgeableQueue.count();
|
2015-02-03 02:26:03 +00:00
|
|
|
|
2015-02-17 23:09:34 +00:00
|
|
|
for (int i = 0; i < fNonpurgeableResources.count(); ++i) {
|
2015-11-09 21:51:06 +00:00
|
|
|
stats->update(fNonpurgeableResources[i]);
|
2015-02-17 23:09:34 +00:00
|
|
|
}
|
|
|
|
for (int i = 0; i < fPurgeableQueue.count(); ++i) {
|
2015-11-09 21:51:06 +00:00
|
|
|
stats->update(fPurgeableQueue.at(i));
|
2015-02-03 02:26:03 +00:00
|
|
|
}
|
2015-11-09 21:51:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GrResourceCache::dumpStats(SkString* out) const {
|
|
|
|
this->validate();
|
|
|
|
|
|
|
|
Stats stats;
|
|
|
|
|
|
|
|
this->getStats(&stats);
|
2015-02-03 02:26:03 +00:00
|
|
|
|
|
|
|
float countUtilization = (100.f * fBudgetedCount) / fMaxCount;
|
|
|
|
float byteUtilization = (100.f * fBudgetedBytes) / fMaxBytes;
|
|
|
|
|
|
|
|
out->appendf("Budget: %d items %d bytes\n", fMaxCount, (int)fMaxBytes);
|
|
|
|
out->appendf("\t\tEntry Count: current %d"
|
2015-06-18 16:12:16 +00:00
|
|
|
" (%d budgeted, %d external(%d borrowed, %d adopted), %d locked, %d scratch %.2g%% full), high %d\n",
|
2015-11-09 21:51:06 +00:00
|
|
|
stats.fTotal, fBudgetedCount, stats.fExternal, stats.fBorrowed,
|
|
|
|
stats.fAdopted, stats.fNumNonPurgeable, stats.fScratch, countUtilization,
|
|
|
|
fHighWaterCount);
|
2015-02-03 02:26:03 +00:00
|
|
|
out->appendf("\t\tEntry Bytes: current %d (budgeted %d, %.2g%% full, %d unbudgeted) high %d\n",
|
2015-02-17 23:09:34 +00:00
|
|
|
SkToInt(fBytes), SkToInt(fBudgetedBytes), byteUtilization,
|
|
|
|
SkToInt(stats.fUnbudgetedSize), SkToInt(fHighWaterBytes));
|
2015-02-03 02:26:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
2015-02-19 19:38:44 +00:00
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
void GrResourceCache::changeTimestamp(uint32_t newTimestamp) { fTimestamp = newTimestamp; }
|
2015-02-03 02:26:03 +00:00
|
|
|
|
2014-11-05 22:47:41 +00:00
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
// Code for the mock context. It's built on a mock GrGpu class that does nothing.
|
|
|
|
////
|
|
|
|
|
|
|
|
#include "GrGpu.h"
|
|
|
|
|
2015-01-22 18:16:09 +00:00
|
|
|
class GrPipeline;
|
2014-11-11 00:03:14 +00:00
|
|
|
|
2014-11-05 22:47:41 +00:00
|
|
|
class MockGpu : public GrGpu {
|
|
|
|
public:
|
2015-05-22 21:01:46 +00:00
|
|
|
MockGpu(GrContext* context, const GrContextOptions& options) : INHERITED(context) {
|
2015-08-26 20:07:48 +00:00
|
|
|
fCaps.reset(new GrCaps(options));
|
2015-05-22 21:01:46 +00:00
|
|
|
}
|
2015-03-26 01:17:31 +00:00
|
|
|
~MockGpu() override {}
|
2014-11-05 22:47:41 +00:00
|
|
|
|
2015-07-28 20:26:15 +00:00
|
|
|
bool onGetReadPixelsInfo(GrSurface* srcSurface, int readWidth, int readHeight, size_t rowBytes,
|
|
|
|
GrPixelConfig readConfig, DrawPreference*,
|
|
|
|
ReadPixelTempDrawInfo*) override { return false; }
|
|
|
|
|
|
|
|
bool onGetWritePixelsInfo(GrSurface* dstSurface, int width, int height, size_t rowBytes,
|
|
|
|
GrPixelConfig srcConfig, DrawPreference*,
|
|
|
|
WritePixelTempDrawInfo*) override { return false; }
|
2015-07-23 15:07:21 +00:00
|
|
|
|
2015-09-11 15:19:35 +00:00
|
|
|
void buildProgramDesc(GrProgramDesc*, const GrPrimitiveProcessor&,
|
|
|
|
const GrPipeline&) const override {}
|
2014-11-26 20:28:00 +00:00
|
|
|
|
2015-03-26 01:17:31 +00:00
|
|
|
void discard(GrRenderTarget*) override {}
|
2014-11-26 20:28:00 +00:00
|
|
|
|
2015-08-21 18:53:29 +00:00
|
|
|
bool onCopySurface(GrSurface* dst,
|
|
|
|
GrSurface* src,
|
|
|
|
const SkIRect& srcRect,
|
|
|
|
const SkIPoint& dstPoint) override { return false; };
|
2014-11-26 20:28:00 +00:00
|
|
|
|
2015-07-13 15:08:25 +00:00
|
|
|
bool initCopySurfaceDstDesc(const GrSurface* src, GrSurfaceDesc* desc) const override {
|
2014-11-26 20:28:00 +00:00
|
|
|
return false;
|
|
|
|
}
|
2014-11-05 22:47:41 +00:00
|
|
|
|
2014-11-26 18:20:45 +00:00
|
|
|
private:
|
2015-03-26 01:17:31 +00:00
|
|
|
void onResetContext(uint32_t resetBits) override {}
|
2014-11-05 22:47:41 +00:00
|
|
|
|
2015-08-12 18:14:50 +00:00
|
|
|
void xferBarrier(GrRenderTarget*, GrXferBarrierType) override {}
|
|
|
|
|
2015-04-22 20:27:39 +00:00
|
|
|
GrTexture* onCreateTexture(const GrSurfaceDesc& desc, GrGpuResource::LifeCycle lifeCycle,
|
|
|
|
const void* srcData, size_t rowBytes) override {
|
2015-08-27 14:41:13 +00:00
|
|
|
return nullptr;
|
2014-11-05 22:47:41 +00:00
|
|
|
}
|
|
|
|
|
2015-04-22 20:27:39 +00:00
|
|
|
GrTexture* onCreateCompressedTexture(const GrSurfaceDesc& desc, GrGpuResource::LifeCycle,
|
2015-03-26 01:17:31 +00:00
|
|
|
const void* srcData) override {
|
2015-08-27 14:41:13 +00:00
|
|
|
return nullptr;
|
2014-11-05 22:47:41 +00:00
|
|
|
}
|
|
|
|
|
2015-06-18 16:12:16 +00:00
|
|
|
GrTexture* onWrapBackendTexture(const GrBackendTextureDesc&,
|
2015-08-27 14:41:13 +00:00
|
|
|
GrWrapOwnership) override { return nullptr; }
|
2014-11-05 22:47:41 +00:00
|
|
|
|
2015-06-18 16:12:16 +00:00
|
|
|
GrRenderTarget* onWrapBackendRenderTarget(const GrBackendRenderTargetDesc&,
|
|
|
|
GrWrapOwnership) override {
|
2015-08-27 14:41:13 +00:00
|
|
|
return nullptr;
|
2014-11-26 18:20:45 +00:00
|
|
|
}
|
2014-11-05 22:47:41 +00:00
|
|
|
|
2015-08-27 14:41:13 +00:00
|
|
|
GrVertexBuffer* onCreateVertexBuffer(size_t size, bool dynamic) override { return nullptr; }
|
2014-11-05 22:47:41 +00:00
|
|
|
|
2015-08-27 14:41:13 +00:00
|
|
|
GrIndexBuffer* onCreateIndexBuffer(size_t size, bool dynamic) override { return nullptr; }
|
2014-11-26 16:45:30 +00:00
|
|
|
|
2015-08-06 17:54:13 +00:00
|
|
|
void onClear(GrRenderTarget*, const SkIRect& rect, GrColor color) override {}
|
2014-11-26 16:45:30 +00:00
|
|
|
|
2015-03-26 01:17:31 +00:00
|
|
|
void onClearStencilClip(GrRenderTarget*, const SkIRect& rect, bool insideClip) override {}
|
2014-11-26 20:28:00 +00:00
|
|
|
|
2015-05-07 18:35:55 +00:00
|
|
|
void onDraw(const DrawArgs&, const GrNonInstancedVertices&) override {}
|
2014-11-26 20:28:00 +00:00
|
|
|
|
2015-07-30 14:34:27 +00:00
|
|
|
bool onReadPixels(GrSurface* surface,
|
2014-11-26 20:28:00 +00:00
|
|
|
int left, int top, int width, int height,
|
|
|
|
GrPixelConfig,
|
|
|
|
void* buffer,
|
2015-03-26 01:17:31 +00:00
|
|
|
size_t rowBytes) override {
|
2014-11-05 22:47:41 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-07-30 14:34:27 +00:00
|
|
|
bool onWritePixels(GrSurface* surface,
|
|
|
|
int left, int top, int width, int height,
|
|
|
|
GrPixelConfig config, const void* buffer,
|
|
|
|
size_t rowBytes) override {
|
2014-11-05 22:47:41 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-03-26 01:17:31 +00:00
|
|
|
void onResolveRenderTarget(GrRenderTarget* target) override { return; }
|
2014-11-05 22:47:41 +00:00
|
|
|
|
2015-09-14 19:56:10 +00:00
|
|
|
GrStencilAttachment* createStencilAttachmentForRenderTarget(const GrRenderTarget*,
|
|
|
|
int width,
|
|
|
|
int height) override {
|
|
|
|
return nullptr;
|
2014-11-05 22:47:41 +00:00
|
|
|
}
|
|
|
|
|
2015-03-26 01:17:31 +00:00
|
|
|
void clearStencil(GrRenderTarget* target) override {}
|
2014-11-05 22:47:41 +00:00
|
|
|
|
2015-07-14 18:02:52 +00:00
|
|
|
GrBackendObject createTestingOnlyBackendTexture(void* pixels, int w, int h,
|
|
|
|
GrPixelConfig config) const override {
|
|
|
|
return 0;
|
|
|
|
}
|
2015-11-11 20:40:42 +00:00
|
|
|
bool isTestingOnlyBackendTexture(GrBackendObject ) const override { return false; }
|
|
|
|
void deleteTestingOnlyBackendTexture(GrBackendObject, bool abandonTexture) const override {}
|
2015-07-13 14:19:57 +00:00
|
|
|
|
2014-11-05 22:47:41 +00:00
|
|
|
typedef GrGpu INHERITED;
|
|
|
|
};
|
|
|
|
|
|
|
|
GrContext* GrContext::CreateMockContext() {
|
2015-08-26 20:07:48 +00:00
|
|
|
GrContext* context = new GrContext;
|
2014-11-05 22:47:41 +00:00
|
|
|
|
|
|
|
context->initMockContext();
|
|
|
|
return context;
|
|
|
|
}
|
|
|
|
|
|
|
|
void GrContext::initMockContext() {
|
2015-05-22 21:01:46 +00:00
|
|
|
GrContextOptions options;
|
2015-06-01 23:17:03 +00:00
|
|
|
options.fGeometryBufferMapThreshold = 0;
|
2015-08-27 14:41:13 +00:00
|
|
|
SkASSERT(nullptr == fGpu);
|
2015-08-26 20:07:48 +00:00
|
|
|
fGpu = new MockGpu(this, options);
|
2014-11-05 22:47:41 +00:00
|
|
|
SkASSERT(fGpu);
|
2015-11-11 21:18:11 +00:00
|
|
|
this->initCommon();
|
2014-11-05 22:47:41 +00:00
|
|
|
|
|
|
|
// We delete these because we want to test the cache starting with zero resources. Also, none of
|
|
|
|
// these objects are required for any of tests that use this context. TODO: make stop allocating
|
|
|
|
// resources in the buffer pools.
|
2015-10-17 14:43:27 +00:00
|
|
|
fDrawingManager->abandon();
|
2014-11-05 22:47:41 +00:00
|
|
|
}
|