From f2361d2d93c200cd4555b5e8ecea4531801abaaa Mon Sep 17 00:00:00 2001 From: Robert Phillips Date: Tue, 25 Oct 2016 14:20:06 -0400 Subject: Add GrOpList and rename GrDrawTarget to GrRenderTargetOpList GOLD_TRYBOT_URL= https://gold.skia.org/search?issue=3910 Change-Id: I026aa26ecc61a0d002e98892dca728536259e8b1 Reviewed-on: https://skia-review.googlesource.com/3910 Reviewed-by: Brian Osman Commit-Queue: Robert Phillips --- gyp/gpu.gypi | 6 +- include/gpu/GrDrawContext.h | 14 +- include/gpu/GrGpuResource.h | 2 +- include/gpu/GrGpuResourceRef.h | 2 +- include/gpu/GrRenderTarget.h | 18 +- include/gpu/GrSurface.h | 19 +- include/private/GrRenderTargetProxy.h | 13 - include/private/GrSurfaceProxy.h | 21 +- src/core/SkTTopoSort.h | 2 +- src/gpu/GrBatchFlushState.h | 2 +- src/gpu/GrContext.cpp | 14 +- src/gpu/GrDrawContext.cpp | 90 ++-- src/gpu/GrDrawContextPriv.h | 4 +- src/gpu/GrDrawTarget.cpp | 609 ------------------------- src/gpu/GrDrawTarget.h | 249 ---------- src/gpu/GrDrawingManager.cpp | 94 ++-- src/gpu/GrDrawingManager.h | 20 +- src/gpu/GrGpu.h | 8 +- src/gpu/GrOpList.cpp | 71 +++ src/gpu/GrOpList.h | 124 +++++ src/gpu/GrPathRenderer.h | 2 +- src/gpu/GrPipeline.cpp | 10 +- src/gpu/GrPipeline.h | 4 +- src/gpu/GrPipelineBuilder.h | 5 +- src/gpu/GrPrimitiveProcessor.h | 2 +- src/gpu/GrRenderTarget.cpp | 28 +- src/gpu/GrRenderTargetOpList.cpp | 580 +++++++++++++++++++++++ src/gpu/GrRenderTargetOpList.h | 175 +++++++ src/gpu/GrRenderTargetProxy.cpp | 28 +- src/gpu/GrResourceProvider.h | 2 +- src/gpu/GrSurface.cpp | 23 + src/gpu/GrSurfaceProxy.cpp | 20 + src/gpu/GrTracing.h | 2 +- src/gpu/GrUserStencilSettings.h | 6 +- src/gpu/batches/GrDrawBatch.h | 2 +- src/gpu/batches/GrTessellatingPathRenderer.cpp | 1 - src/gpu/gl/GrGLGpu.cpp | 4 +- src/gpu/gl/GrGLGpu.h | 2 +- src/gpu/text/GrStencilAndCoverTextContext.h | 2 +- src/gpu/vk/GrVkGpu.cpp | 2 +- src/gpu/vk/GrVkGpu.h | 2 +- tools/gpu/GrTest.cpp | 6 +- tools/gpu/GrTest.h | 4 +- 43 files changed, 1196 insertions(+), 1098 deletions(-) delete mode 100644 src/gpu/GrDrawTarget.cpp delete mode 100644 src/gpu/GrDrawTarget.h create mode 100644 src/gpu/GrOpList.cpp create mode 100644 src/gpu/GrOpList.h create mode 100644 src/gpu/GrRenderTargetOpList.cpp create mode 100644 src/gpu/GrRenderTargetOpList.h diff --git a/gyp/gpu.gypi b/gyp/gpu.gypi index 3a2ad20b54..315de8bd4e 100644 --- a/gyp/gpu.gypi +++ b/gyp/gpu.gypi @@ -101,8 +101,6 @@ '<(skia_src_path)/gpu/GrPathRenderingDrawContext.h', '<(skia_src_path)/gpu/GrDrawingManager.cpp', '<(skia_src_path)/gpu/GrDrawingManager.h', - '<(skia_src_path)/gpu/GrDrawTarget.cpp', - '<(skia_src_path)/gpu/GrDrawTarget.h', '<(skia_src_path)/gpu/GrFixedClip.cpp', '<(skia_src_path)/gpu/GrFixedClip.h', '<(skia_src_path)/gpu/GrFragmentProcessor.cpp', @@ -124,6 +122,8 @@ '<(skia_src_path)/gpu/GrMemoryPool.h', '<(skia_src_path)/gpu/GrMesh.h', '<(skia_src_path)/gpu/GrNonAtomicRef.h', + '<(skia_src_path)/gpu/GrOpList.cpp', + '<(skia_src_path)/gpu/GrOpList.h', '<(skia_src_path)/gpu/GrOvalRenderer.cpp', '<(skia_src_path)/gpu/GrOvalRenderer.h', '<(skia_src_path)/gpu/GrPaint.cpp', @@ -168,6 +168,8 @@ '<(skia_src_path)/gpu/GrRenderTargetProxy.cpp', '<(skia_src_path)/gpu/GrReducedClip.cpp', '<(skia_src_path)/gpu/GrReducedClip.h', + '<(skia_src_path)/gpu/GrRenderTargetOpList.cpp', + '<(skia_src_path)/gpu/GrRenderTargetOpList.h', '<(skia_src_path)/gpu/GrResourceCache.cpp', '<(skia_src_path)/gpu/GrResourceCache.h', '<(skia_src_path)/gpu/GrResourceHandle.h', diff --git a/include/gpu/GrDrawContext.h b/include/gpu/GrDrawContext.h index 72de15deae..6e9056dd02 100644 --- a/include/gpu/GrDrawContext.h +++ b/include/gpu/GrDrawContext.h @@ -24,12 +24,12 @@ class GrDrawBatch; class GrDrawContextPriv; class GrDrawPathBatchBase; class GrDrawingManager; -class GrDrawTarget; class GrFixedClip; class GrPaint; class GrPathProcessor; class GrPipelineBuilder; class GrRenderTarget; +class GrRenderTargetOpList; class GrStyle; class GrSurface; struct GrUserStencilSettings; @@ -366,7 +366,7 @@ private: friend class GrDrawingManager; // for ctor friend class GrDrawContextPriv; - friend class GrTestTarget; // for access to getDrawTarget + friend class GrTestTarget; // for access to getOpList friend class GrSWMaskHelper; // for access to drawBatch // All the path renderers currently make their own batches @@ -412,17 +412,17 @@ private: const GrStyle& style); // This entry point allows the GrTextContext-derived classes to add their batches to - // the drawTarget. + // the GrOpList. void drawBatch(const GrPipelineBuilder& pipelineBuilder, const GrClip&, GrDrawBatch* batch); - GrDrawTarget* getDrawTarget(); + GrRenderTargetOpList* getOpList(); GrDrawingManager* fDrawingManager; sk_sp fRenderTarget; - // In MDB-mode the drawTarget can be closed by some other drawContext that has picked - // it up. For this reason, the drawTarget should only ever be accessed via 'getDrawTarget'. - GrDrawTarget* fDrawTarget; + // In MDB-mode the GrOpList can be closed by some other drawContext that has picked + // it up. For this reason, the GrOpList should only ever be accessed via 'getOpList'. + GrRenderTargetOpList* fOpList; GrContext* fContext; GrInstancedPipelineInfo fInstancedPipelineInfo; diff --git a/include/gpu/GrGpuResource.h b/include/gpu/GrGpuResource.h index 364a886408..d3cfa86086 100644 --- a/include/gpu/GrGpuResource.h +++ b/include/gpu/GrGpuResource.h @@ -23,7 +23,7 @@ class SkTraceMemoryDump; * * Gpu resources can have three types of refs: * 1) Normal ref (+ by ref(), - by unref()): These are used by code that is issuing draw calls - * that read and write the resource via GrDrawTarget and by any object that must own a + * that read and write the resource via GrOpList and by any object that must own a * GrGpuResource and is itself owned (directly or indirectly) by Skia-client code. * 2) Pending read (+ by addPendingRead(), - by completedRead()): GrContext has scheduled a read * of the resource by the GPU as a result of a skia API call but hasn't executed it yet. diff --git a/include/gpu/GrGpuResourceRef.h b/include/gpu/GrGpuResourceRef.h index 4511adce68..13adbee242 100644 --- a/include/gpu/GrGpuResourceRef.h +++ b/include/gpu/GrGpuResourceRef.h @@ -63,7 +63,7 @@ private: execution. It can only be called once. */ void markPendingIO() const; - /** Called when the program element/draw state is no longer owned by GrDrawTarget-client code. + /** Called when the program element/draw state is no longer owned by GrOpList-client code. This lets the cache know that the drawing code will no longer schedule additional reads or writes to the resource using the program element or draw state. It can only be called once. */ diff --git a/include/gpu/GrRenderTarget.h b/include/gpu/GrRenderTarget.h index 1f87787d45..445846fbb1 100644 --- a/include/gpu/GrRenderTarget.h +++ b/include/gpu/GrRenderTarget.h @@ -12,9 +12,9 @@ #include "SkRect.h" class GrCaps; -class GrDrawTarget; -class GrStencilAttachment; +class GrRenderTargetOpList; class GrRenderTargetPriv; +class GrStencilAttachment; /** * GrRenderTarget represents a 2D buffer of pixels that can be rendered to. @@ -115,8 +115,9 @@ public: GrRenderTargetPriv renderTargetPriv(); const GrRenderTargetPriv renderTargetPriv() const; - void setLastDrawTarget(GrDrawTarget* dt); - GrDrawTarget* getLastDrawTarget() { return fLastDrawTarget; } + GrRenderTargetOpList* getLastRenderTargetOpList() { + return (GrRenderTargetOpList*) this->getLastOpList(); + } protected: enum class Flags { @@ -129,7 +130,6 @@ protected: GrRenderTarget(GrGpu*, const GrSurfaceDesc&, Flags = Flags::kNone, GrStencilAttachment* = nullptr); - ~GrRenderTarget() override; // override of GrResource void onAbandon() override; @@ -150,14 +150,6 @@ private: SkIRect fResolveRect; - // The last drawTarget that wrote to or is currently going to write to this renderTarget - // The drawTarget can be closed (e.g., no draw context is currently bound - // to this renderTarget). - // This back-pointer is required so that we can add a dependancy between - // the drawTarget used to create the current contents of this renderTarget - // and the drawTarget of a destination renderTarget to which this one is being drawn. - GrDrawTarget* fLastDrawTarget; - typedef GrSurface INHERITED; }; diff --git a/include/gpu/GrSurface.h b/include/gpu/GrSurface.h index ac5c5fa1bb..0ac4bf3b12 100644 --- a/include/gpu/GrSurface.h +++ b/include/gpu/GrSurface.h @@ -14,6 +14,7 @@ #include "SkImageInfo.h" #include "SkRect.h" +class GrOpList; class GrRenderTarget; class GrSurfacePriv; class GrTexture; @@ -127,6 +128,9 @@ public: static size_t WorstCaseSize(const GrSurfaceDesc& desc); + void setLastOpList(GrOpList* opList); + GrOpList* getLastOpList() { return fLastOpList; } + protected: // Methods made available via GrSurfacePriv bool savePixels(const char* filename); @@ -142,12 +146,9 @@ protected: , fDesc(desc) , fReleaseProc(NULL) , fReleaseCtx(NULL) - {} - - ~GrSurface() override { - // check that invokeReleaseProc has been called (if needed) - SkASSERT(NULL == fReleaseProc); + , fLastOpList(nullptr) { } + ~GrSurface() override; GrSurfaceDesc fDesc; @@ -165,6 +166,14 @@ private: ReleaseProc fReleaseProc; ReleaseCtx fReleaseCtx; + // The last opList that wrote to or is currently going to write to this surface + // The opList can be closed (e.g., no draw or copy context is currently bound + // to this renderTarget or texture). + // This back-pointer is required so that we can add a dependancy between + // the opList used to create the current contents of this surface + // and the opList of a destination surface to which this one is being drawn or copied. + GrOpList* fLastOpList; + typedef GrGpuResource INHERITED; }; diff --git a/include/private/GrRenderTargetProxy.h b/include/private/GrRenderTargetProxy.h index e4bc70f212..59fed598e1 100644 --- a/include/private/GrRenderTargetProxy.h +++ b/include/private/GrRenderTargetProxy.h @@ -28,8 +28,6 @@ public: SkBackingFit, SkBudgeted); static sk_sp Make(const GrCaps&, sk_sp); - ~GrRenderTargetProxy() override; - // TODO: add asTextureProxy variants GrRenderTargetProxy* asRenderTargetProxy() override { return this; } const GrRenderTargetProxy* asRenderTargetProxy() const override { return this; } @@ -60,9 +58,6 @@ public: */ int numColorSamples() const { return this->isMixedSampled() ? 0 : fDesc.fSampleCnt; } - void setLastDrawTarget(GrDrawTarget* dt); - GrDrawTarget* getLastDrawTarget() { return fLastDrawTarget; } - GrRenderTargetPriv::Flags testingOnly_getFlags() const; private: @@ -83,14 +78,6 @@ private: // rendertarget's info here. GrRenderTargetPriv::Flags fFlags; - // The last drawTarget that wrote to or is currently going to write to this renderTarget - // The drawTarget can be closed (e.g., no draw context is currently bound - // to this renderTarget). - // This back-pointer is required so that we can add a dependancy between - // the drawTarget used to create the current contents of this renderTarget - // and the drawTarget of a destination renderTarget to which this one is being drawn. - GrDrawTarget* fLastDrawTarget; - typedef GrSurfaceProxy INHERITED; }; diff --git a/include/private/GrSurfaceProxy.h b/include/private/GrSurfaceProxy.h index 69656fe4f5..4e2ad3bf14 100644 --- a/include/private/GrSurfaceProxy.h +++ b/include/private/GrSurfaceProxy.h @@ -11,6 +11,7 @@ #include "GrGpuResource.h" #include "SkRect.h" +class GrOpList; class GrTextureProxy; class GrRenderTargetProxy; @@ -51,13 +52,17 @@ public: */ SkBudgeted isBudgeted() const { return fBudgeted; } + void setLastOpList(GrOpList* opList); + GrOpList* getLastOpList() { return fLastOpList; } + protected: // Deferred version GrSurfaceProxy(const GrSurfaceDesc& desc, SkBackingFit fit, SkBudgeted budgeted) : fDesc(desc) , fFit(fit) , fBudgeted(budgeted) - , fUniqueID(GrGpuResource::CreateUniqueID()) { + , fUniqueID(GrGpuResource::CreateUniqueID()) + , fLastOpList(nullptr) { } // Wrapped version @@ -66,10 +71,12 @@ protected: : fDesc(desc) , fFit(fit) , fBudgeted(budgeted) - , fUniqueID(uniqueID) { + , fUniqueID(uniqueID) + , fLastOpList(nullptr) { } - virtual ~GrSurfaceProxy() {} + virtual ~GrSurfaceProxy(); + // For wrapped resources, 'fDesc' will always be filled in from the wrapped resource. const GrSurfaceDesc fDesc; @@ -83,6 +90,14 @@ private: void notifyAllCntsAreZero(CntType) const { delete this; } bool notifyRefCountIsZero() const { return true; } + // The last opList that wrote to or is currently going to write to this surface + // The opList can be closed (e.g., no draw context is currently bound + // to this renderTarget). + // This back-pointer is required so that we can add a dependancy between + // the opList used to create the current contents of this surface + // and the opList of a destination surface to which this one is being drawn or copied. + GrOpList* fLastOpList; + typedef GrIORef INHERITED; // to access notifyAllCntsAreZero and notifyRefCntIsZero. diff --git a/src/core/SkTTopoSort.h b/src/core/SkTTopoSort.h index 35b85eeb81..21c80696e7 100644 --- a/src/core/SkTTopoSort.h +++ b/src/core/SkTTopoSort.h @@ -76,7 +76,7 @@ bool SkTTopoSort_Visit(T* node, SkTDArray* result) { // // TODO: potentially add a version that takes a seed node and just outputs that // node and all the nodes on which it depends. This could be used to partially -// flush a drawTarget DAG. +// flush a GrOpList DAG. template bool SkTTopoSort(SkTDArray* graph) { SkTDArray result; diff --git a/src/gpu/GrBatchFlushState.h b/src/gpu/GrBatchFlushState.h index 89c5292eea..794bc60da4 100644 --- a/src/gpu/GrBatchFlushState.h +++ b/src/gpu/GrBatchFlushState.h @@ -15,7 +15,7 @@ class GrGpuCommandBuffer; class GrResourceProvider; -/** Tracks the state across all the GrBatches in a GrDrawTarget flush. */ +/** Tracks the state across all the GrBatches in a GrOpList flush. */ class GrBatchFlushState { public: GrBatchFlushState(GrGpu*, GrResourceProvider*); diff --git a/src/gpu/GrContext.cpp b/src/gpu/GrContext.cpp index 176d5da59b..a0884f9274 100644 --- a/src/gpu/GrContext.cpp +++ b/src/gpu/GrContext.cpp @@ -90,17 +90,17 @@ void GrContext::initCommon(const GrContextOptions& options) { fDidTestPMConversions = false; - GrDrawTarget::Options dtOptions; - dtOptions.fClipBatchToBounds = options.fClipBatchToBounds; - dtOptions.fDrawBatchBounds = options.fDrawBatchBounds; - dtOptions.fMaxBatchLookback = options.fMaxBatchLookback; - dtOptions.fMaxBatchLookahead = options.fMaxBatchLookahead; + GrRenderTargetOpList::Options rtOpListOptions; + rtOpListOptions.fClipBatchToBounds = options.fClipBatchToBounds; + rtOpListOptions.fDrawBatchBounds = options.fDrawBatchBounds; + rtOpListOptions.fMaxBatchLookback = options.fMaxBatchLookback; + rtOpListOptions.fMaxBatchLookahead = options.fMaxBatchLookahead; GrPathRendererChain::Options prcOptions; prcOptions.fDisableDistanceFieldRenderer = options.fDisableDistanceFieldPaths; prcOptions.fAllowPathMaskCaching = options.fAllowPathMaskCaching; prcOptions.fDisableAllPathRenderers = options.fForceSWPathMasks; - fDrawingManager.reset(new GrDrawingManager(this, dtOptions, prcOptions, options.fImmediateMode, - &fSingleOwner)); + fDrawingManager.reset(new GrDrawingManager(this, rtOpListOptions, prcOptions, + options.fImmediateMode, &fSingleOwner)); // GrBatchFontCache will eventually replace GrFontCache fBatchFontCache = new GrBatchFontCache(this); diff --git a/src/gpu/GrDrawContext.cpp b/src/gpu/GrDrawContext.cpp index 3060062653..54de4f0ba2 100644 --- a/src/gpu/GrDrawContext.cpp +++ b/src/gpu/GrDrawContext.cpp @@ -69,10 +69,10 @@ bool GrDrawContext::wasAbandoned() const { return fDrawingManager->wasAbandoned(); } -// In MDB mode the reffing of the 'getLastDrawTarget' call's result allows in-progress -// drawTargets to be picked up and added to by drawContexts lower in the call -// stack. When this occurs with a closed drawTarget, a new one will be allocated -// when the drawContext attempts to use it (via getDrawTarget). +// In MDB mode the reffing of the 'getLastOpList' call's result allows in-progress +// GrOpLists to be picked up and added to by drawContexts lower in the call +// stack. When this occurs with a closed GrOpList, a new one will be allocated +// when the drawContext attempts to use it (via getOpList). GrDrawContext::GrDrawContext(GrContext* context, GrDrawingManager* drawingMgr, sk_sp rt, @@ -82,7 +82,7 @@ GrDrawContext::GrDrawContext(GrContext* context, GrSingleOwner* singleOwner) : fDrawingManager(drawingMgr) , fRenderTarget(std::move(rt)) - , fDrawTarget(SkSafeRef(fRenderTarget->getLastDrawTarget())) + , fOpList(SkSafeRef(fRenderTarget->getLastRenderTargetOpList())) , fContext(context) , fInstancedPipelineInfo(fRenderTarget.get()) , fColorSpace(std::move(colorSpace)) @@ -106,26 +106,26 @@ void GrDrawContext::validate() const { SkASSERT(fRenderTarget); ASSERT_OWNED_RESOURCE(fRenderTarget); - if (fDrawTarget && !fDrawTarget->isClosed()) { - SkASSERT(fRenderTarget->getLastDrawTarget() == fDrawTarget); + if (fOpList && !fOpList->isClosed()) { + SkASSERT(fRenderTarget->getLastOpList() == fOpList); } } #endif GrDrawContext::~GrDrawContext() { ASSERT_SINGLE_OWNER - SkSafeUnref(fDrawTarget); + SkSafeUnref(fOpList); } -GrDrawTarget* GrDrawContext::getDrawTarget() { +GrRenderTargetOpList* GrDrawContext::getOpList() { ASSERT_SINGLE_OWNER SkDEBUGCODE(this->validate();) - if (!fDrawTarget || fDrawTarget->isClosed()) { - fDrawTarget = fDrawingManager->newDrawTarget(fRenderTarget.get()); + if (!fOpList || fOpList->isClosed()) { + fOpList = fDrawingManager->newOpList(fRenderTarget.get()); } - return fDrawTarget; + return fOpList; } bool GrDrawContext::copySurface(GrSurface* src, const SkIRect& srcRect, const SkIPoint& dstPoint) { @@ -134,7 +134,7 @@ bool GrDrawContext::copySurface(GrSurface* src, const SkIRect& srcRect, const Sk SkDEBUGCODE(this->validate();) GR_AUDIT_TRAIL_AUTO_FRAME(fAuditTrail, "GrDrawContext::copySurface"); - return this->getDrawTarget()->copySurface(fRenderTarget.get(), src, srcRect, dstPoint); + return this->getOpList()->copySurface(fRenderTarget.get(), src, srcRect, dstPoint); } void GrDrawContext::drawText(const GrClip& clip, const GrPaint& grPaint, @@ -191,7 +191,7 @@ void GrDrawContext::discard() { GR_AUDIT_TRAIL_AUTO_FRAME(fAuditTrail, "GrDrawContext::discard"); AutoCheckFlush acf(fDrawingManager); - this->getDrawTarget()->discard(fRenderTarget.get()); + this->getOpList()->discard(fRenderTarget.get()); } void GrDrawContext::clear(const SkIRect* rect, @@ -245,13 +245,13 @@ void GrDrawContext::internalClear(const GrFixedClip& clip, this->drawRect(clip, paint, SkMatrix::I(), clearRect); } else if (isFull) { - this->getDrawTarget()->fullClear(this->accessRenderTarget(), color); + this->getOpList()->fullClear(this->accessRenderTarget(), color); } else { sk_sp batch(GrClearBatch::Make(clip, color, this->accessRenderTarget())); if (!batch) { return; } - this->getDrawTarget()->addBatch(std::move(batch)); + this->getOpList()->addBatch(std::move(batch)); } } @@ -396,7 +396,7 @@ bool GrDrawContext::drawFilledRect(const GrClip& clip, bool useHWAA; if (GrCaps::InstancedSupport::kNone != fContext->caps()->instancedSupport()) { - InstancedRendering* ir = this->getDrawTarget()->instancedRendering(); + InstancedRendering* ir = this->getOpList()->instancedRendering(); batch.reset(ir->recordRect(croppedRect, viewMatrix, paint.getColor(), paint.isAntiAlias(), fInstancedPipelineInfo, &useHWAA)); @@ -405,7 +405,7 @@ bool GrDrawContext::drawFilledRect(const GrClip& clip, if (ss) { pipelineBuilder.setUserStencil(ss); } - this->getDrawTarget()->drawBatch(pipelineBuilder, this, clip, batch); + this->getOpList()->drawBatch(pipelineBuilder, this, clip, batch); return true; } } @@ -423,7 +423,7 @@ bool GrDrawContext::drawFilledRect(const GrClip& clip, if (ss) { pipelineBuilder.setUserStencil(ss); } - this->getDrawTarget()->drawBatch(pipelineBuilder, this, clip, batch); + this->getOpList()->drawBatch(pipelineBuilder, this, clip, batch); return true; } } @@ -549,7 +549,7 @@ void GrDrawContext::drawRect(const GrClip& clip, snapToPixelCenters); } - this->getDrawTarget()->drawBatch(pipelineBuilder, this, clip, batch); + this->getOpList()->drawBatch(pipelineBuilder, this, clip, batch); return; } } @@ -567,15 +567,15 @@ void GrDrawContextPriv::clearStencilClip(const GrFixedClip& clip, bool insideSte GR_AUDIT_TRAIL_AUTO_FRAME(fDrawContext->fAuditTrail, "GrDrawContextPriv::clearStencilClip"); AutoCheckFlush acf(fDrawContext->fDrawingManager); - fDrawContext->getDrawTarget()->clearStencilClip(clip, insideStencilMask, - fDrawContext->accessRenderTarget()); + fDrawContext->getOpList()->clearStencilClip(clip, insideStencilMask, + fDrawContext->accessRenderTarget()); } void GrDrawContextPriv::stencilPath(const GrClip& clip, bool useHWAA, const SkMatrix& viewMatrix, const GrPath* path) { - fDrawContext->getDrawTarget()->stencilPath(fDrawContext, clip, useHWAA, viewMatrix, path); + fDrawContext->getOpList()->stencilPath(fDrawContext, clip, useHWAA, viewMatrix, path); } void GrDrawContextPriv::stencilRect(const GrClip& clip, @@ -646,13 +646,13 @@ void GrDrawContext::fillRectToRect(const GrClip& clip, bool useHWAA; if (GrCaps::InstancedSupport::kNone != fContext->caps()->instancedSupport()) { - InstancedRendering* ir = this->getDrawTarget()->instancedRendering(); + InstancedRendering* ir = this->getOpList()->instancedRendering(); SkAutoTUnref batch(ir->recordRect(croppedRect, viewMatrix, paint.getColor(), croppedLocalRect, paint.isAntiAlias(), fInstancedPipelineInfo, &useHWAA)); if (batch) { GrPipelineBuilder pipelineBuilder(paint, useHWAA); - this->getDrawTarget()->drawBatch(pipelineBuilder, this, clip, batch); + this->getOpList()->drawBatch(pipelineBuilder, this, clip, batch); return; } } @@ -705,13 +705,13 @@ void GrDrawContext::fillRectWithLocalMatrix(const GrClip& clip, bool useHWAA; if (GrCaps::InstancedSupport::kNone != fContext->caps()->instancedSupport()) { - InstancedRendering* ir = this->getDrawTarget()->instancedRendering(); + InstancedRendering* ir = this->getOpList()->instancedRendering(); SkAutoTUnref batch(ir->recordRect(croppedRect, viewMatrix, paint.getColor(), localMatrix, paint.isAntiAlias(), fInstancedPipelineInfo, &useHWAA)); if (batch) { GrPipelineBuilder pipelineBuilder(paint, useHWAA); - this->getDrawTarget()->drawBatch(pipelineBuilder, this, clip, batch); + this->getOpList()->drawBatch(pipelineBuilder, this, clip, batch); return; } } @@ -726,7 +726,7 @@ void GrDrawContext::fillRectWithLocalMatrix(const GrClip& clip, SkAutoTUnref batch(GrAAFillRectBatch::Create(paint.getColor(), viewMatrix, localMatrix, croppedRect)); GrPipelineBuilder pipelineBuilder(paint, useHWAA); - this->getDrawTarget()->drawBatch(pipelineBuilder, this, clip, batch); + this->getOpList()->drawBatch(pipelineBuilder, this, clip, batch); return; } @@ -776,7 +776,7 @@ void GrDrawContext::drawVertices(const GrClip& clip, colors, texCoords, bounds)); GrPipelineBuilder pipelineBuilder(paint, this->mustUseHWAA(paint)); - this->getDrawTarget()->drawBatch(pipelineBuilder, this, clip, batch); + this->getOpList()->drawBatch(pipelineBuilder, this, clip, batch); } /////////////////////////////////////////////////////////////////////////////// @@ -799,7 +799,7 @@ void GrDrawContext::drawAtlas(const GrClip& clip, xform, texRect, colors)); GrPipelineBuilder pipelineBuilder(paint, this->mustUseHWAA(paint)); - this->getDrawTarget()->drawBatch(pipelineBuilder, this, clip, batch); + this->getOpList()->drawBatch(pipelineBuilder, this, clip, batch); } /////////////////////////////////////////////////////////////////////////////// @@ -838,13 +838,13 @@ void GrDrawContext::drawRRect(const GrClip& origClip, if (GrCaps::InstancedSupport::kNone != fContext->caps()->instancedSupport() && stroke.isFillStyle()) { - InstancedRendering* ir = this->getDrawTarget()->instancedRendering(); + InstancedRendering* ir = this->getOpList()->instancedRendering(); SkAutoTUnref batch(ir->recordRRect(rrect, viewMatrix, paint.getColor(), paint.isAntiAlias(), fInstancedPipelineInfo, &useHWAA)); if (batch) { GrPipelineBuilder pipelineBuilder(paint, useHWAA); - this->getDrawTarget()->drawBatch(pipelineBuilder, this, *clip, batch); + this->getOpList()->drawBatch(pipelineBuilder, this, *clip, batch); return; } } @@ -860,7 +860,7 @@ void GrDrawContext::drawRRect(const GrClip& origClip, shaderCaps)); if (batch) { GrPipelineBuilder pipelineBuilder(paint, useHWAA); - this->getDrawTarget()->drawBatch(pipelineBuilder, this, *clip, batch); + this->getOpList()->drawBatch(pipelineBuilder, this, *clip, batch); return; } } @@ -881,13 +881,13 @@ bool GrDrawContext::drawFilledDRRect(const GrClip& clip, if (GrCaps::InstancedSupport::kNone != fContext->caps()->instancedSupport()) { bool useHWAA; - InstancedRendering* ir = this->getDrawTarget()->instancedRendering(); + InstancedRendering* ir = this->getOpList()->instancedRendering(); SkAutoTUnref batch(ir->recordDRRect(origOuter, origInner, viewMatrix, paintIn.getColor(), paintIn.isAntiAlias(), fInstancedPipelineInfo, &useHWAA)); if (batch) { GrPipelineBuilder pipelineBuilder(paintIn, useHWAA); - this->getDrawTarget()->drawBatch(pipelineBuilder, this, clip, batch); + this->getOpList()->drawBatch(pipelineBuilder, this, clip, batch); return true; } } @@ -997,7 +997,7 @@ void GrDrawContext::drawRegion(const GrClip& clip, SkAutoTUnref batch(GrRegionBatch::Create(paint.getColor(), viewMatrix, region)); GrPipelineBuilder pipelineBuilder(paint, false); - this->getDrawTarget()->drawBatch(pipelineBuilder, this, clip, batch); + this->getOpList()->drawBatch(pipelineBuilder, this, clip, batch); } void GrDrawContext::drawOval(const GrClip& clip, @@ -1022,13 +1022,13 @@ void GrDrawContext::drawOval(const GrClip& clip, if (GrCaps::InstancedSupport::kNone != fContext->caps()->instancedSupport() && stroke.isFillStyle()) { - InstancedRendering* ir = this->getDrawTarget()->instancedRendering(); + InstancedRendering* ir = this->getOpList()->instancedRendering(); SkAutoTUnref batch(ir->recordOval(oval, viewMatrix, paint.getColor(), paint.isAntiAlias(), fInstancedPipelineInfo, &useHWAA)); if (batch) { GrPipelineBuilder pipelineBuilder(paint, useHWAA); - this->getDrawTarget()->drawBatch(pipelineBuilder, this, clip, batch); + this->getOpList()->drawBatch(pipelineBuilder, this, clip, batch); return; } } @@ -1042,7 +1042,7 @@ void GrDrawContext::drawOval(const GrClip& clip, shaderCaps)); if (batch) { GrPipelineBuilder pipelineBuilder(paint, useHWAA); - this->getDrawTarget()->drawBatch(pipelineBuilder, this, clip, batch); + this->getOpList()->drawBatch(pipelineBuilder, this, clip, batch); return; } } @@ -1074,7 +1074,7 @@ void GrDrawContext::drawArc(const GrClip& clip, shaderCaps)); if (batch) { GrPipelineBuilder pipelineBuilder(paint, useHWAA); - this->getDrawTarget()->drawBatch(pipelineBuilder, this, clip, batch); + this->getOpList()->drawBatch(pipelineBuilder, this, clip, batch); return; } } @@ -1104,7 +1104,7 @@ void GrDrawContext::drawImageLattice(const GrClip& clip, std::move(iter), dst)); GrPipelineBuilder pipelineBuilder(paint, this->mustUseHWAA(paint)); - this->getDrawTarget()->drawBatch(pipelineBuilder, this, clip, batch); + this->getOpList()->drawBatch(pipelineBuilder, this, clip, batch); } void GrDrawContext::prepareForExternalIO() { @@ -1134,7 +1134,7 @@ void GrDrawContext::drawNonAAFilledRect(const GrClip& clip, if (ss) { pipelineBuilder.setUserStencil(ss); } - this->getDrawTarget()->drawBatch(pipelineBuilder, this, clip, batch); + this->getOpList()->drawBatch(pipelineBuilder, this, clip, batch); } bool GrDrawContext::readPixels(const SkImageInfo& dstInfo, void* dstBuffer, size_t dstRowBytes, @@ -1246,7 +1246,7 @@ void GrDrawContext::drawPath(const GrClip& clip, paint.getColor(), viewMatrix, rects)); if (batch) { GrPipelineBuilder pipelineBuilder(paint, useHWAA); - this->getDrawTarget()->drawBatch(pipelineBuilder, this, clip, batch); + this->getOpList()->drawBatch(pipelineBuilder, this, clip, batch); } return; } @@ -1263,7 +1263,7 @@ void GrDrawContext::drawPath(const GrClip& clip, shaderCaps)); if (batch) { GrPipelineBuilder pipelineBuilder(paint, useHWAA); - this->getDrawTarget()->drawBatch(pipelineBuilder, this, clip, batch); + this->getOpList()->drawBatch(pipelineBuilder, this, clip, batch); return; } } @@ -1433,5 +1433,5 @@ void GrDrawContext::drawBatch(const GrPipelineBuilder& pipelineBuilder, const Gr SkDEBUGCODE(this->validate();) GR_AUDIT_TRAIL_AUTO_FRAME(fAuditTrail, "GrDrawContext::drawBatch"); - this->getDrawTarget()->drawBatch(pipelineBuilder, this, clip, batch); + this->getOpList()->drawBatch(pipelineBuilder, this, clip, batch); } diff --git a/src/gpu/GrDrawContextPriv.h b/src/gpu/GrDrawContextPriv.h index 63eae12a06..a48e4681d6 100644 --- a/src/gpu/GrDrawContextPriv.h +++ b/src/gpu/GrDrawContextPriv.h @@ -9,7 +9,7 @@ #define GrDrawContextPriv_DEFINED #include "GrDrawContext.h" -#include "GrDrawTarget.h" +#include "GrRenderTargetOpList.h" #include "GrPathRendering.h" class GrFixedClip; @@ -22,7 +22,7 @@ struct GrUserStencilSettings; class GrDrawContextPriv { public: gr_instanced::InstancedRendering* accessInstancedRendering() const { - return fDrawContext->getDrawTarget()->instancedRendering(); + return fDrawContext->getOpList()->instancedRendering(); } void clear(const GrFixedClip&, const GrColor, bool canIgnoreClip); diff --git a/src/gpu/GrDrawTarget.cpp b/src/gpu/GrDrawTarget.cpp deleted file mode 100644 index 5b2530fa5a..0000000000 --- a/src/gpu/GrDrawTarget.cpp +++ /dev/null @@ -1,609 +0,0 @@ -/* - * Copyright 2010 Google Inc. - * - * Use of this source code is governed by a BSD-style license that can be - * found in the LICENSE file. - */ - -#include "GrDrawTarget.h" - -#include "GrAppliedClip.h" -#include "GrAuditTrail.h" -#include "GrCaps.h" -#include "GrDrawContext.h" -#include "GrGpu.h" -#include "GrGpuCommandBuffer.h" -#include "GrPath.h" -#include "GrPipeline.h" -#include "GrMemoryPool.h" -#include "GrPipelineBuilder.h" -#include "GrRenderTarget.h" -#include "GrResourceProvider.h" -#include "GrRenderTargetPriv.h" -#include "GrStencilAttachment.h" -#include "GrSurfacePriv.h" -#include "GrTexture.h" -#include "gl/GrGLRenderTarget.h" - -#include "SkStrokeRec.h" - -#include "batches/GrClearBatch.h" -#include "batches/GrClearStencilClipBatch.h" -#include "batches/GrCopySurfaceBatch.h" -#include "batches/GrDiscardBatch.h" -#include "batches/GrDrawBatch.h" -#include "batches/GrDrawPathBatch.h" -#include "batches/GrRectBatchFactory.h" -#include "batches/GrStencilPathBatch.h" - -#include "instanced/InstancedRendering.h" - -//////////////////////////////////////////////////////////////////////////////// - -// Experimentally we have found that most batching occurs within the first 10 comparisons. -static const int kDefaultMaxBatchLookback = 10; -static const int kDefaultMaxBatchLookahead = 10; - -GrDrawTarget::GrDrawTarget(GrRenderTarget* rt, GrGpu* gpu, GrResourceProvider* resourceProvider, - GrAuditTrail* auditTrail, const Options& options) - : fLastFullClearBatch(nullptr) - , fGpu(SkRef(gpu)) - , fResourceProvider(resourceProvider) - , fAuditTrail(auditTrail) - , fFlags(0) - , fRenderTarget(rt) { - // TODO: Stop extracting the context (currently needed by GrClip) - fContext = fGpu->getContext(); - - fClipBatchToBounds = options.fClipBatchToBounds; - fDrawBatchBounds = options.fDrawBatchBounds; - fMaxBatchLookback = (options.fMaxBatchLookback < 0) ? kDefaultMaxBatchLookback : - options.fMaxBatchLookback; - fMaxBatchLookahead = (options.fMaxBatchLookahead < 0) ? kDefaultMaxBatchLookahead : - options.fMaxBatchLookahead; - - if (GrCaps::InstancedSupport::kNone != this->caps()->instancedSupport()) { - fInstancedRendering.reset(fGpu->createInstancedRendering()); - } - - rt->setLastDrawTarget(this); - -#ifdef SK_DEBUG - static int debugID = 0; - fDebugID = debugID++; -#endif -} - -GrDrawTarget::~GrDrawTarget() { - if (fRenderTarget && this == fRenderTarget->getLastDrawTarget()) { - fRenderTarget->setLastDrawTarget(nullptr); - } - - fGpu->unref(); -} - -//////////////////////////////////////////////////////////////////////////////// - -// Add a GrDrawTarget-based dependency -void GrDrawTarget::addDependency(GrDrawTarget* dependedOn) { - SkASSERT(!dependedOn->dependsOn(this)); // loops are bad - - if (this->dependsOn(dependedOn)) { - return; // don't add duplicate dependencies - } - - *fDependencies.push() = dependedOn; -} - -// Convert from a GrSurface-based dependency to a GrDrawTarget one -void GrDrawTarget::addDependency(GrSurface* dependedOn) { - if (dependedOn->asRenderTarget() && dependedOn->asRenderTarget()->getLastDrawTarget()) { - // If it is still receiving dependencies, this DT shouldn't be closed - SkASSERT(!this->isClosed()); - - GrDrawTarget* dt = dependedOn->asRenderTarget()->getLastDrawTarget(); - if (dt == this) { - // self-read - presumably for dst reads - } else { - this->addDependency(dt); - - // Can't make it closed in the self-read case - dt->makeClosed(); - } - } -} - -#ifdef SK_DEBUG -void GrDrawTarget::dump() const { - SkDebugf("--------------------------------------------------------------\n"); - SkDebugf("node: %d -> RT: %d\n", fDebugID, fRenderTarget ? fRenderTarget->uniqueID() : -1); - SkDebugf("relies On (%d): ", fDependencies.count()); - for (int i = 0; i < fDependencies.count(); ++i) { - SkDebugf("%d, ", fDependencies[i]->fDebugID); - } - SkDebugf("\n"); - SkDebugf("batches (%d):\n", fRecordedBatches.count()); - for (int i = 0; i < fRecordedBatches.count(); ++i) { - SkDebugf("*******************************\n"); - if (!fRecordedBatches[i].fBatch) { - SkDebugf("%d: \n", i); - } else { - SkDebugf("%d: %s\n", i, fRecordedBatches[i].fBatch->name()); - SkString str = fRecordedBatches[i].fBatch->dumpInfo(); - SkDebugf("%s\n", str.c_str()); - const SkRect& clippedBounds = fRecordedBatches[i].fClippedBounds; - SkDebugf("ClippedBounds: [L: %.2f, T: %.2f, R: %.2f, B: %.2f]\n", - clippedBounds.fLeft, clippedBounds.fTop, clippedBounds.fRight, - clippedBounds.fBottom); - } - } -} -#endif - -bool GrDrawTarget::setupDstReadIfNecessary(const GrPipelineBuilder& pipelineBuilder, - GrRenderTarget* rt, - const GrClip& clip, - const GrPipelineOptimizations& optimizations, - GrXferProcessor::DstTexture* dstTexture, - const SkRect& batchBounds) { - SkRect bounds = batchBounds; - bounds.outset(0.5f, 0.5f); - - if (!pipelineBuilder.willXPNeedDstTexture(*this->caps(), optimizations)) { - return true; - } - - if (this->caps()->textureBarrierSupport()) { - if (GrTexture* rtTex = rt->asTexture()) { - // The render target is a texture, so we can read from it directly in the shader. The XP - // will be responsible to detect this situation and request a texture barrier. - dstTexture->setTexture(rtTex); - dstTexture->setOffset(0, 0); - return true; - } - } - - SkIRect copyRect; - clip.getConservativeBounds(rt->width(), rt->height(), ©Rect); - - SkIRect drawIBounds; - bounds.roundOut(&drawIBounds); - if (!copyRect.intersect(drawIBounds)) { -#ifdef SK_DEBUG - GrCapsDebugf(this->caps(), "Missed an early reject. " - "Bailing on draw from setupDstReadIfNecessary.\n"); -#endif - return false; - } - - // MSAA consideration: When there is support for reading MSAA samples in the shader we could - // have per-sample dst values by making the copy multisampled. - GrSurfaceDesc desc; - if (!fGpu->initDescForDstCopy(rt, &desc)) { - desc.fOrigin = kDefault_GrSurfaceOrigin; - desc.fFlags = kRenderTarget_GrSurfaceFlag; - desc.fConfig = rt->config(); - } - - desc.fWidth = copyRect.width(); - desc.fHeight = copyRect.height(); - - static const uint32_t kFlags = 0; - SkAutoTUnref copy(fResourceProvider->createApproxTexture(desc, kFlags)); - - if (!copy) { - SkDebugf("Failed to create temporary copy of destination texture.\n"); - return false; - } - SkIPoint dstPoint = {0, 0}; - this->copySurface(copy, rt, copyRect, dstPoint); - dstTexture->setTexture(copy); - dstTexture->setOffset(copyRect.fLeft, copyRect.fTop); - return true; -} - -void GrDrawTarget::prepareBatches(GrBatchFlushState* flushState) { - // Semi-usually the drawTargets are already closed at this point, but sometimes Ganesh - // needs to flush mid-draw. In that case, the SkGpuDevice's drawTargets won't be closed - // but need to be flushed anyway. Closing such drawTargets here will mean new - // drawTargets will be created to replace them if the SkGpuDevice(s) write to them again. - this->makeClosed(); - - // Loop over the batches that haven't yet generated their geometry - for (int i = 0; i < fRecordedBatches.count(); ++i) { - if (fRecordedBatches[i].fBatch) { - fRecordedBatches[i].fBatch->prepare(flushState); - } - } - - if (fInstancedRendering) { - fInstancedRendering->beginFlush(flushState->resourceProvider()); - } -} - -bool GrDrawTarget::drawBatches(GrBatchFlushState* flushState) { - if (0 == fRecordedBatches.count()) { - return false; - } - // Draw all the generated geometry. - SkRandom random; - GrRenderTarget* currentRT = nullptr; - SkAutoTDelete commandBuffer; - for (int i = 0; i < fRecordedBatches.count(); ++i) { - if (!fRecordedBatches[i].fBatch) { - continue; - } - if (fRecordedBatches[i].fBatch->renderTarget() != currentRT) { - if (commandBuffer) { - commandBuffer->end(); - commandBuffer->submit(); - commandBuffer.reset(); - } - currentRT = fRecordedBatches[i].fBatch->renderTarget(); - if (currentRT) { - static const GrGpuCommandBuffer::LoadAndStoreInfo kBasicLoadStoreInfo - { GrGpuCommandBuffer::LoadOp::kLoad,GrGpuCommandBuffer::StoreOp::kStore, - GrColor_ILLEGAL }; - commandBuffer.reset(fGpu->createCommandBuffer(currentRT, - kBasicLoadStoreInfo, // Color - kBasicLoadStoreInfo)); // Stencil - } - flushState->setCommandBuffer(commandBuffer); - } - if (fDrawBatchBounds) { - const SkRect& bounds = fRecordedBatches[i].fClippedBounds; - SkIRect ibounds; - bounds.roundOut(&ibounds); - // In multi-draw buffer all the batches use the same render target and we won't need to - // get the batchs bounds. - if (GrRenderTarget* rt = fRecordedBatches[i].fBatch->renderTarget()) { - fGpu->drawDebugWireRect(rt, ibounds, 0xFF000000 | random.nextU()); - } - } - fRecordedBatches[i].fBatch->draw(flushState, fRecordedBatches[i].fClippedBounds); - } - if (commandBuffer) { - commandBuffer->end(); - commandBuffer->submit(); - flushState->setCommandBuffer(nullptr); - } - - fGpu->finishDrawTarget(); - return true; -} - -void GrDrawTarget::reset() { - fLastFullClearBatch = nullptr; - fRecordedBatches.reset(); - if (fInstancedRendering) { - fInstancedRendering->endFlush(); - } -} - -static void batch_bounds(SkRect* bounds, const GrBatch* batch) { - *bounds = batch->bounds(); - if (batch->hasZeroArea()) { - if (batch->hasAABloat()) { - bounds->outset(0.5f, 0.5f); - } else { - // We don't know which way the particular GPU will snap lines or points at integer - // coords. So we ensure that the bounds is large enough for either snap. - SkRect before = *bounds; - bounds->roundOut(bounds); - if (bounds->fLeft == before.fLeft) { - bounds->fLeft -= 1; - } - if (bounds->fTop == before.fTop) { - bounds->fTop -= 1; - } - if (bounds->fRight == before.fRight) { - bounds->fRight += 1; - } - if (bounds->fBottom == before.fBottom) { - bounds->fBottom += 1; - } - } - } -} - -void GrDrawTarget::drawBatch(const GrPipelineBuilder& pipelineBuilder, - GrDrawContext* drawContext, - const GrClip& clip, - GrDrawBatch* batch) { - // Setup clip - SkRect bounds; - batch_bounds(&bounds, batch); - GrAppliedClip appliedClip(bounds); - if (!clip.apply(fContext, drawContext, pipelineBuilder.isHWAntialias(), - pipelineBuilder.hasUserStencilSettings(), &appliedClip)) { - return; - } - - // TODO: this is the only remaining usage of the AutoRestoreFragmentProcessorState - remove it - GrPipelineBuilder::AutoRestoreFragmentProcessorState arfps; - if (appliedClip.clipCoverageFragmentProcessor()) { - arfps.set(&pipelineBuilder); - arfps.addCoverageFragmentProcessor(sk_ref_sp(appliedClip.clipCoverageFragmentProcessor())); - } - - if (pipelineBuilder.hasUserStencilSettings() || appliedClip.hasStencilClip()) { - if (!fResourceProvider->attachStencilAttachment(drawContext->accessRenderTarget())) { - SkDebugf("ERROR creating stencil attachment. Draw skipped.\n"); - return; - } - } - - GrPipeline::CreateArgs args; - args.fPipelineBuilder = &pipelineBuilder; - args.fDrawContext = drawContext; - args.fCaps = this->caps(); - batch->getPipelineOptimizations(&args.fOpts); - if (args.fOpts.fOverrides.fUsePLSDstRead || fClipBatchToBounds) { - GrGLIRect viewport; - viewport.fLeft = 0; - viewport.fBottom = 0; - viewport.fWidth = drawContext->width(); - viewport.fHeight = drawContext->height(); - SkIRect ibounds; - ibounds.fLeft = SkTPin(SkScalarFloorToInt(batch->bounds().fLeft), viewport.fLeft, - viewport.fWidth); - ibounds.fTop = SkTPin(SkScalarFloorToInt(batch->bounds().fTop), viewport.fBottom, - viewport.fHeight); - ibounds.fRight = SkTPin(SkScalarCeilToInt(batch->bounds().fRight), viewport.fLeft, - viewport.fWidth); - ibounds.fBottom = SkTPin(SkScalarCeilToInt(batch->bounds().fBottom), viewport.fBottom, - viewport.fHeight); - if (!appliedClip.addScissor(ibounds)) { - return; - } - } - args.fOpts.fColorPOI.completeCalculations( - sk_sp_address_as_pointer_address(pipelineBuilder.fColorFragmentProcessors.begin()), - pipelineBuilder.numColorFragmentProcessors()); - args.fOpts.fCoveragePOI.completeCalculations( - sk_sp_address_as_pointer_address(pipelineBuilder.fCoverageFragmentProcessors.begin()), - pipelineBuilder.numCoverageFragmentProcessors()); - args.fScissor = &appliedClip.scissorState(); - args.fWindowRectsState = &appliedClip.windowRectsState(); - args.fHasStencilClip = appliedClip.hasStencilClip(); - if (!this->setupDstReadIfNecessary(pipelineBuilder, drawContext->accessRenderTarget(), - clip, args.fOpts, - &args.fDstTexture, batch->bounds())) { - return; - } - - if (!batch->installPipeline(args)) { - return; - } - -#ifdef ENABLE_MDB - SkASSERT(fRenderTarget); - batch->pipeline()->addDependenciesTo(fRenderTarget); -#endif - this->recordBatch(batch, appliedClip.clippedDrawBounds()); -} - -void GrDrawTarget::stencilPath(GrDrawContext* drawContext, - const GrClip& clip, - bool useHWAA, - const SkMatrix& viewMatrix, - const GrPath* path) { - // TODO: extract portions of checkDraw that are relevant to path stenciling. - SkASSERT(path); - SkASSERT(this->caps()->shaderCaps()->pathRenderingSupport()); - - // FIXME: Use path bounds instead of this WAR once - // https://bugs.chromium.org/p/skia/issues/detail?id=5640 is resolved. - SkRect bounds = SkRect::MakeIWH(drawContext->width(), drawContext->height()); - - // Setup clip - GrAppliedClip appliedClip(bounds); - if (!clip.apply(fContext, drawContext, useHWAA, true, &appliedClip)) { - return; - } - // TODO: respect fClipBatchToBounds if we ever start computing bounds here. - - // Coverage AA does not make sense when rendering to the stencil buffer. The caller should never - // attempt this in a situation that would require coverage AA. - SkASSERT(!appliedClip.clipCoverageFragmentProcessor()); - - GrStencilAttachment* stencilAttachment = fResourceProvider->attachStencilAttachment( - drawContext->accessRenderTarget()); - if (!stencilAttachment) { - SkDebugf("ERROR creating stencil attachment. Draw skipped.\n"); - return; - } - - GrBatch* batch = GrStencilPathBatch::Create(viewMatrix, - useHWAA, - path->getFillType(), - appliedClip.hasStencilClip(), - stencilAttachment->bits(), - appliedClip.scissorState(), - drawContext->accessRenderTarget(), - path); - this->recordBatch(batch, appliedClip.clippedDrawBounds()); - batch->unref(); -} - -void GrDrawTarget::addBatch(sk_sp batch) { - this->recordBatch(batch.get(), batch->bounds()); -} - -void GrDrawTarget::fullClear(GrRenderTarget* renderTarget, GrColor color) { - // Currently this just inserts or updates the last clear batch. However, once in MDB this can - // remove all the previously recorded batches and change the load op to clear with supplied - // color. - if (fLastFullClearBatch && - fLastFullClearBatch->renderTargetUniqueID() == renderTarget->uniqueID()) { - // As currently implemented, fLastFullClearBatch should be the last batch because we would - // have cleared it when another batch was recorded. - SkASSERT(fRecordedBatches.back().fBatch.get() == fLastFullClearBatch); - fLastFullClearBatch->setColor(color); - return; - } - sk_sp batch(GrClearBatch::Make(GrFixedClip::Disabled(), color, renderTarget)); - if (batch.get() == this->recordBatch(batch.get(), batch->bounds())) { - fLastFullClearBatch = batch.get(); - } -} - -void GrDrawTarget::discard(GrRenderTarget* renderTarget) { - // Currently this just inserts a discard batch. However, once in MDB this can remove all the - // previously recorded batches and change the load op to discard. - if (this->caps()->discardRenderTargetSupport()) { - GrBatch* batch = new GrDiscardBatch(renderTarget); - this->recordBatch(batch, batch->bounds()); - batch->unref(); - } -} - -//////////////////////////////////////////////////////////////////////////////// - -bool GrDrawTarget::copySurface(GrSurface* dst, - GrSurface* src, - const SkIRect& srcRect, - const SkIPoint& dstPoint) { - GrBatch* batch = GrCopySurfaceBatch::Create(dst, src, srcRect, dstPoint); - if (!batch) { - return false; - } -#ifdef ENABLE_MDB - this->addDependency(src); -#endif - - this->recordBatch(batch, batch->bounds()); - batch->unref(); - return true; -} - -static inline bool can_reorder(const SkRect& a, const SkRect& b) { - return a.fRight <= b.fLeft || a.fBottom <= b.fTop || - b.fRight <= a.fLeft || b.fBottom <= a.fTop; -} - -static void join(SkRect* out, const SkRect& a, const SkRect& b) { - SkASSERT(a.fLeft <= a.fRight && a.fTop <= a.fBottom); - SkASSERT(b.fLeft <= b.fRight && b.fTop <= b.fBottom); - out->fLeft = SkTMin(a.fLeft, b.fLeft); - out->fTop = SkTMin(a.fTop, b.fTop); - out->fRight = SkTMax(a.fRight, b.fRight); - out->fBottom = SkTMax(a.fBottom, b.fBottom); -} - -GrBatch* GrDrawTarget::recordBatch(GrBatch* batch, const SkRect& clippedBounds) { - // A closed drawTarget should never receive new/more batches - SkASSERT(!this->isClosed()); - - // Check if there is a Batch Draw we can batch with by linearly searching back until we either - // 1) check every draw - // 2) intersect with something - // 3) find a 'blocker' - GR_AUDIT_TRAIL_ADDBATCH(fAuditTrail, batch); - GrBATCH_INFO("Re-Recording (%s, B%u)\n" - "\tBounds LRTB (%f, %f, %f, %f)\n", - batch->name(), - batch->uniqueID(), - batch->bounds().fLeft, batch->bounds().fRight, - batch->bounds().fTop, batch->bounds().fBottom); - GrBATCH_INFO(SkTabString(batch->dumpInfo(), 1).c_str()); - GrBATCH_INFO("\tClipped Bounds: [L: %.2f, T: %.2f, R: %.2f, B: %.2f]\n", - clippedBounds.fLeft, clippedBounds.fTop, clippedBounds.fRight, - clippedBounds.fBottom); - GrBATCH_INFO("\tOutcome:\n"); - int maxCandidates = SkTMin(fMaxBatchLookback, fRecordedBatches.count()); - if (maxCandidates) { - int i = 0; - while (true) { - GrBatch* candidate = fRecordedBatches.fromBack(i).fBatch.get(); - // We cannot continue to search backwards if the render target changes - if (candidate->renderTargetUniqueID() != batch->renderTargetUniqueID()) { - GrBATCH_INFO("\t\tBreaking because of (%s, B%u) Rendertarget\n", - candidate->name(), candidate->uniqueID()); - break; - } - if (candidate->combineIfPossible(batch, *this->caps())) { - GrBATCH_INFO("\t\tCombining with (%s, B%u)\n", candidate->name(), - candidate->uniqueID()); - GR_AUDIT_TRAIL_BATCHING_RESULT_COMBINED(fAuditTrail, candidate, batch); - join(&fRecordedBatches.fromBack(i).fClippedBounds, - fRecordedBatches.fromBack(i).fClippedBounds, clippedBounds); - return candidate; - } - // Stop going backwards if we would cause a painter's order violation. - const SkRect& candidateBounds = fRecordedBatches.fromBack(i).fClippedBounds; - if (!can_reorder(candidateBounds, clippedBounds)) { - GrBATCH_INFO("\t\tIntersects with (%s, B%u)\n", candidate->name(), - candidate->uniqueID()); - break; - } - ++i; - if (i == maxCandidates) { - GrBATCH_INFO("\t\tReached max lookback or beginning of batch array %d\n", i); - break; - } - } - } else { - GrBATCH_INFO("\t\tFirstBatch\n"); - } - GR_AUDIT_TRAIL_BATCHING_RESULT_NEW(fAuditTrail, batch); - fRecordedBatches.emplace_back(RecordedBatch{sk_ref_sp(batch), clippedBounds}); - fLastFullClearBatch = nullptr; - return batch; -} - -void GrDrawTarget::forwardCombine() { - if (fMaxBatchLookahead <= 0) { - return; - } - for (int i = 0; i < fRecordedBatches.count() - 2; ++i) { - GrBatch* batch = fRecordedBatches[i].fBatch.get(); - const SkRect& batchBounds = fRecordedBatches[i].fClippedBounds; - int maxCandidateIdx = SkTMin(i + fMaxBatchLookahead, fRecordedBatches.count() - 1); - int j = i + 1; - while (true) { - GrBatch* candidate = fRecordedBatches[j].fBatch.get(); - // We cannot continue to search if the render target changes - if (candidate->renderTargetUniqueID() != batch->renderTargetUniqueID()) { - GrBATCH_INFO("\t\tBreaking because of (%s, B%u) Rendertarget\n", - candidate->name(), candidate->uniqueID()); - break; - } - if (j == i +1) { - // We assume batch would have combined with candidate when the candidate was added - // via backwards combining in recordBatch. - SkASSERT(!batch->combineIfPossible(candidate, *this->caps())); - } else if (batch->combineIfPossible(candidate, *this->caps())) { - GrBATCH_INFO("\t\tCombining with (%s, B%u)\n", candidate->name(), - candidate->uniqueID()); - GR_AUDIT_TRAIL_BATCHING_RESULT_COMBINED(fAuditTrail, batch, candidate); - fRecordedBatches[j].fBatch = std::move(fRecordedBatches[i].fBatch); - join(&fRecordedBatches[j].fClippedBounds, fRecordedBatches[j].fClippedBounds, - batchBounds); - break; - } - // Stop going traversing if we would cause a painter's order violation. - const SkRect& candidateBounds = fRecordedBatches[j].fClippedBounds; - if (!can_reorder(candidateBounds, batchBounds)) { - GrBATCH_INFO("\t\tIntersects with (%s, B%u)\n", candidate->name(), - candidate->uniqueID()); - break; - } - ++j; - if (j > maxCandidateIdx) { - GrBATCH_INFO("\t\tReached max lookahead or end of batch array %d\n", i); - break; - } - } - } -} - -/////////////////////////////////////////////////////////////////////////////// - -void GrDrawTarget::clearStencilClip(const GrFixedClip& clip, - bool insideStencilMask, - GrRenderTarget* rt) { - GrBatch* batch = new GrClearStencilClipBatch(clip, insideStencilMask, rt); - this->recordBatch(batch, batch->bounds()); - batch->unref(); -} diff --git a/src/gpu/GrDrawTarget.h b/src/gpu/GrDrawTarget.h deleted file mode 100644 index 35de239150..0000000000 --- a/src/gpu/GrDrawTarget.h +++ /dev/null @@ -1,249 +0,0 @@ -/* - * Copyright 2010 Google Inc. - * - * Use of this source code is governed by a BSD-style license that can be - * found in the LICENSE file. - */ - -#ifndef GrDrawTarget_DEFINED -#define GrDrawTarget_DEFINED - -#include "GrClip.h" -#include "GrContext.h" -#include "GrPathProcessor.h" -#include "GrPrimitiveProcessor.h" -#include "GrPathRendering.h" -#include "GrXferProcessor.h" - -#include "batches/GrDrawBatch.h" - -#include "SkClipStack.h" -#include "SkMatrix.h" -#include "SkPath.h" -#include "SkStringUtils.h" -#include "SkStrokeRec.h" -#include "SkTArray.h" -#include "SkTLazy.h" -#include "SkTypes.h" -#include "SkXfermode.h" - -//#define ENABLE_MDB 1 - -class GrAuditTrail; -class GrBatch; -class GrClearBatch; -class GrClip; -class GrCaps; -class GrPath; -class GrDrawPathBatchBase; -class GrPipelineBuilder; - -class GrDrawTarget final : public SkRefCnt { -public: - /** Options for GrDrawTarget behavior. */ - struct Options { - Options () - : fClipBatchToBounds(false) - , fDrawBatchBounds(false) - , fMaxBatchLookback(-1) - , fMaxBatchLookahead(-1) {} - bool fClipBatchToBounds; - bool fDrawBatchBounds; - int fMaxBatchLookback; - int fMaxBatchLookahead; - }; - - GrDrawTarget(GrRenderTarget*, GrGpu*, GrResourceProvider*, GrAuditTrail*, const Options&); - - ~GrDrawTarget() override; - - void makeClosed() { - fLastFullClearBatch = nullptr; - // We only close drawTargets When MDB is enabled. When MDB is disabled there is only - // ever one drawTarget and all calls will be funnelled into it. -#ifdef ENABLE_MDB - this->setFlag(kClosed_Flag); -#endif - this->forwardCombine(); - } - - bool isClosed() const { return this->isSetFlag(kClosed_Flag); } - - // TODO: this entry point is only needed in the non-MDB world. Remove when - // we make the switch to MDB - void clearRT() { fRenderTarget = nullptr; } - - /* - * Notify this drawTarget that it relies on the contents of 'dependedOn' - */ - void addDependency(GrSurface* dependedOn); - - /* - * Does this drawTarget depend on 'dependedOn'? - */ - bool dependsOn(GrDrawTarget* dependedOn) const { - return fDependencies.find(dependedOn) >= 0; - } - - /* - * Dump out the drawTarget dependency DAG - */ - SkDEBUGCODE(void dump() const;) - - /** - * Empties the draw buffer of any queued up draws. - */ - void reset(); - - /** - * Together these two functions flush all queued up draws to GrCommandBuffer. The return value - * of drawBatches() indicates whether any commands were actually issued to the GPU. - */ - void prepareBatches(GrBatchFlushState* flushState); - bool drawBatches(GrBatchFlushState* flushState); - - /** - * Gets the capabilities of the draw target. - */ - const GrCaps* caps() const { return fGpu->caps(); } - - void drawBatch(const GrPipelineBuilder&, GrDrawContext*, const GrClip&, GrDrawBatch*); - - void addBatch(sk_sp); - - /** - * Draws the path into user stencil bits. Upon return, all user stencil values - * inside the path will be nonzero. The path's fill must be either even/odd or - * winding (notnverse or hairline).It will respect the HW antialias boolean (if - * possible in the 3D API). Note, we will never have an inverse fill with - * stencil path. - */ - void stencilPath(GrDrawContext*, - const GrClip&, - bool useHWAA, - const SkMatrix& viewMatrix, - const GrPath*); - - /** Clears the entire render target */ - void fullClear(GrRenderTarget*, GrColor color); - - /** Discards the contents render target. */ - void discard(GrRenderTarget*); - - /** - * Copies a pixel rectangle from one surface to another. This call may finalize - * reserved vertex/index data (as though a draw call was made). The src pixels - * copied are specified by srcRect. They are copied to a rect of the same - * size in dst with top left at dstPoint. If the src rect is clipped by the - * src bounds then pixel values in the dst rect corresponding to area clipped - * by the src rect are not overwritten. This method is not guaranteed to succeed - * depending on the type of surface, configs, etc, and the backend-specific - * limitations. - */ - bool copySurface(GrSurface* dst, - GrSurface* src, - const SkIRect& srcRect, - const SkIPoint& dstPoint); - - gr_instanced::InstancedRendering* instancedRendering() const { - SkASSERT(fInstancedRendering); - return fInstancedRendering; - } - -private: - friend class GrDrawingManager; // for resetFlag & TopoSortTraits - friend class GrDrawContextPriv; // for clearStencilClip - - enum Flags { - kClosed_Flag = 0x01, //!< This drawTarget can't accept any more batches - - kWasOutput_Flag = 0x02, //!< Flag for topological sorting - kTempMark_Flag = 0x04, //!< Flag for topological sorting - }; - - void setFlag(uint32_t flag) { - fFlags |= flag; - } - - void resetFlag(uint32_t flag) { - fFlags &= ~flag; - } - - bool isSetFlag(uint32_t flag) const { - return SkToBool(fFlags & flag); - } - - struct TopoSortTraits { - static void Output(GrDrawTarget* dt, int /* index */) { - dt->setFlag(GrDrawTarget::kWasOutput_Flag); - } - static bool WasOutput(const GrDrawTarget* dt) { - return dt->isSetFlag(GrDrawTarget::kWasOutput_Flag); - } - static void SetTempMark(GrDrawTarget* dt) { - dt->setFlag(GrDrawTarget::kTempMark_Flag); - } - static void ResetTempMark(GrDrawTarget* dt) { - dt->resetFlag(GrDrawTarget::kTempMark_Flag); - } - static bool IsTempMarked(const GrDrawTarget* dt) { - return dt->isSetFlag(GrDrawTarget::kTempMark_Flag); - } - static int NumDependencies(const GrDrawTarget* dt) { - return dt->fDependencies.count(); - } - static GrDrawTarget* Dependency(GrDrawTarget* dt, int index) { - return dt->fDependencies[index]; - } - }; - - // Returns the batch that the input batch was combined with or the input batch if it wasn't - // combined. - GrBatch* recordBatch(GrBatch*, const SkRect& clippedBounds); - void forwardCombine(); - - // Makes a copy of the dst if it is necessary for the draw. Returns false if a copy is required - // but couldn't be made. Otherwise, returns true. This method needs to be protected because it - // needs to be accessed by GLPrograms to setup a correct drawstate - bool setupDstReadIfNecessary(const GrPipelineBuilder&, - GrRenderTarget*, - const GrClip&, - const GrPipelineOptimizations& optimizations, - GrXferProcessor::DstTexture*, - const SkRect& batchBounds); - - void addDependency(GrDrawTarget* dependedOn); - - // Used only by drawContextPriv. - void clearStencilClip(const GrFixedClip&, bool insideStencilMask, GrRenderTarget*); - - struct RecordedBatch { - sk_sp fBatch; - SkRect fClippedBounds; - }; - SkSTArray<256, RecordedBatch, true> fRecordedBatches; - GrClearBatch* fLastFullClearBatch; - // The context is only in service of the GrClip, remove once it doesn't need this. - GrContext* fContext; - GrGpu* fGpu; - GrResourceProvider* fResourceProvider; - GrAuditTrail* fAuditTrail; - - SkDEBUGCODE(int fDebugID;) - uint32_t fFlags; - - // 'this' drawTarget relies on the output of the drawTargets in 'fDependencies' - SkTDArray fDependencies; - GrRenderTarget* fRenderTarget; - - bool fClipBatchToBounds; - bool fDrawBatchBounds; - int fMaxBatchLookback; - int fMaxBatchLookahead; - - SkAutoTDelete fInstancedRendering; - - typedef SkRefCnt INHERITED; -}; - -#endif diff --git a/src/gpu/GrDrawingManager.cpp b/src/gpu/GrDrawingManager.cpp index 5e9b68d88f..0fe173dac3 100644 --- a/src/gpu/GrDrawingManager.cpp +++ b/src/gpu/GrDrawingManager.cpp @@ -9,7 +9,6 @@ #include "GrContext.h" #include "GrDrawContext.h" -#include "GrDrawTarget.h" #include "GrPathRenderingDrawContext.h" #include "GrResourceProvider.h" #include "GrSoftwarePathRenderer.h" @@ -17,25 +16,21 @@ #include "SkSurface_Gpu.h" #include "SkTTopoSort.h" -#include "instanced/InstancedRendering.h" - #include "text/GrAtlasTextContext.h" #include "text/GrStencilAndCoverTextContext.h" -using gr_instanced::InstancedRendering; - void GrDrawingManager::cleanup() { - for (int i = 0; i < fDrawTargets.count(); ++i) { - fDrawTargets[i]->makeClosed(); // no drawTarget should receive a new command after this - fDrawTargets[i]->clearRT(); + for (int i = 0; i < fOpLists.count(); ++i) { + fOpLists[i]->makeClosed(); // no opList should receive a new command after this + fOpLists[i]->clearTarget(); - // We shouldn't need to do this, but it turns out some clients still hold onto drawtargets + // We shouldn't need to do this, but it turns out some clients still hold onto opLists // after a cleanup - fDrawTargets[i]->reset(); - fDrawTargets[i]->unref(); + fOpLists[i]->reset(); + fOpLists[i]->unref(); } - fDrawTargets.reset(); + fOpLists.reset(); delete fPathRendererChain; fPathRendererChain = nullptr; @@ -48,11 +43,8 @@ GrDrawingManager::~GrDrawingManager() { void GrDrawingManager::abandon() { fAbandoned = true; - for (int i = 0; i < fDrawTargets.count(); ++i) { - if (GrCaps::InstancedSupport::kNone != fContext->caps()->instancedSupport()) { - InstancedRendering* ir = fDrawTargets[i]->instancedRendering(); - ir->resetGpuResources(InstancedRendering::ResetType::kAbandon); - } + for (int i = 0; i < fOpLists.count(); ++i) { + fOpLists[i]->abandonGpuResources(); } this->cleanup(); } @@ -62,17 +54,14 @@ void GrDrawingManager::freeGpuResources() { delete fPathRendererChain; fPathRendererChain = nullptr; SkSafeSetNull(fSoftwarePathRenderer); - for (int i = 0; i < fDrawTargets.count(); ++i) { - if (GrCaps::InstancedSupport::kNone != fContext->caps()->instancedSupport()) { - InstancedRendering* ir = fDrawTargets[i]->instancedRendering(); - ir->resetGpuResources(InstancedRendering::ResetType::kDestroy); - } + for (int i = 0; i < fOpLists.count(); ++i) { + fOpLists[i]->freeGpuResources(); } } void GrDrawingManager::reset() { - for (int i = 0; i < fDrawTargets.count(); ++i) { - fDrawTargets[i]->reset(); + for (int i = 0; i < fOpLists.count(); ++i) { + fOpLists[i]->reset(); } fFlushState.reset(); } @@ -84,48 +73,48 @@ void GrDrawingManager::internalFlush(GrResourceCache::FlushType type) { fFlushing = true; bool flushed = false; SkDEBUGCODE(bool result =) - SkTTopoSort(&fDrawTargets); + SkTTopoSort(&fOpLists); SkASSERT(result); - for (int i = 0; i < fDrawTargets.count(); ++i) { - fDrawTargets[i]->prepareBatches(&fFlushState); + for (int i = 0; i < fOpLists.count(); ++i) { + fOpLists[i]->prepareBatches(&fFlushState); } // Enable this to print out verbose batching information #if 0 - for (int i = 0; i < fDrawTargets.count(); ++i) { - SkDEBUGCODE(fDrawTargets[i]->dump();) + for (int i = 0; i < fOpLists.count(); ++i) { + SkDEBUGCODE(fOpLists[i]->dump();) } #endif // Upload all data to the GPU fFlushState.preIssueDraws(); - for (int i = 0; i < fDrawTargets.count(); ++i) { - if (fDrawTargets[i]->drawBatches(&fFlushState)) { + for (int i = 0; i < fOpLists.count(); ++i) { + if (fOpLists[i]->drawBatches(&fFlushState)) { flushed = true; } } SkASSERT(fFlushState.nextDrawToken() == fFlushState.nextTokenToFlush()); - for (int i = 0; i < fDrawTargets.count(); ++i) { - fDrawTargets[i]->reset(); + for (int i = 0; i < fOpLists.count(); ++i) { + fOpLists[i]->reset(); #ifdef ENABLE_MDB - fDrawTargets[i]->unref(); + fOpLists[i]->unref(); #endif } #ifndef ENABLE_MDB - // When MDB is disabled we keep reusing the same drawTarget - if (fDrawTargets.count()) { - SkASSERT(fDrawTargets.count() == 1); + // When MDB is disabled we keep reusing the same GrOpList + if (fOpLists.count()) { + SkASSERT(fOpLists.count() == 1); // Clear out this flag so the topological sort's SkTTopoSort_CheckAllUnmarked check // won't bark - fDrawTargets[0]->resetFlag(GrDrawTarget::kWasOutput_Flag); + fOpLists[0]->resetFlag(GrOpList::kWasOutput_Flag); } #else - fDrawTargets.reset(); + fOpLists.reset(); #endif fFlushState.reset(); @@ -153,28 +142,33 @@ void GrDrawingManager::prepareSurfaceForExternalIO(GrSurface* surface) { } } -GrDrawTarget* GrDrawingManager::newDrawTarget(GrRenderTarget* rt) { +GrRenderTargetOpList* GrDrawingManager::newOpList(GrRenderTarget* rt) { SkASSERT(fContext); #ifndef ENABLE_MDB - // When MDB is disabled we always just return the single drawTarget - if (fDrawTargets.count()) { - SkASSERT(fDrawTargets.count() == 1); - // In the non-MDB-world the same drawTarget gets reused for multiple render targets. + // When MDB is disabled we always just return the single GrOpList + if (fOpLists.count()) { + SkASSERT(fOpLists.count() == 1); + // In the non-MDB-world the same GrOpList gets reused for multiple render targets. // Update this pointer so all the asserts are happy - rt->setLastDrawTarget(fDrawTargets[0]); + rt->setLastOpList(fOpLists[0]); // DrawingManager gets the creation ref - this ref is for the caller - return SkRef(fDrawTargets[0]); + + // TODO: although this is true right now it isn't cool + return SkRef((GrRenderTargetOpList*) fOpLists[0]); } #endif - GrDrawTarget* dt = new GrDrawTarget(rt, fContext->getGpu(), fContext->resourceProvider(), - fContext->getAuditTrail(), fOptionsForDrawTargets); + GrRenderTargetOpList* opList = new GrRenderTargetOpList(rt, + fContext->getGpu(), + fContext->resourceProvider(), + fContext->getAuditTrail(), + fOptionsForOpLists); - *fDrawTargets.append() = dt; + *fOpLists.append() = opList; // DrawingManager gets the creation ref - this ref is for the caller - return SkRef(dt); + return SkRef(opList); } GrAtlasTextContext* GrDrawingManager::getAtlasTextContext() { diff --git a/src/gpu/GrDrawingManager.h b/src/gpu/GrDrawingManager.h index 9fced38163..838bb9b75e 100644 --- a/src/gpu/GrDrawingManager.h +++ b/src/gpu/GrDrawingManager.h @@ -9,23 +9,24 @@ #define GrDrawingManager_DEFINED #include "text/GrAtlasTextContext.h" -#include "GrDrawTarget.h" #include "GrBatchFlushState.h" #include "GrPathRendererChain.h" #include "GrPathRenderer.h" +#include "GrRenderTargetOpList.h" #include "GrResourceCache.h" #include "SkTDArray.h" + class GrContext; class GrDrawContext; class GrSingleOWner; class GrSoftwarePathRenderer; // The GrDrawingManager allocates a new GrDrawContext for each GrRenderTarget -// but all of them still land in the same GrDrawTarget! +// but all of them still land in the same GrOpList! // // In the future this class will allocate a new GrDrawContext for -// each GrRenderTarget/GrDrawTarget and manage the DAG. +// each GrRenderTarget/GrOpList and manage the DAG. class GrDrawingManager { public: ~GrDrawingManager(); @@ -37,9 +38,9 @@ public: sk_sp, const SkSurfaceProps*); - // The caller automatically gets a ref on the returned drawTarget. It must + // The caller automatically gets a ref on the returned opList. It must // be balanced by an unref call. - GrDrawTarget* newDrawTarget(GrRenderTarget* rt); + GrRenderTargetOpList* newOpList(GrRenderTarget* rt); GrContext* getContext() { return fContext; } @@ -63,11 +64,12 @@ public: void prepareSurfaceForExternalIO(GrSurface*); private: - GrDrawingManager(GrContext* context, const GrDrawTarget::Options& optionsForDrawTargets, + GrDrawingManager(GrContext* context, + const GrRenderTargetOpList::Options& optionsForOpLists, const GrPathRendererChain::Options& optionsForPathRendererChain, bool isImmediateMode, GrSingleOwner* singleOwner) : fContext(context) - , fOptionsForDrawTargets(optionsForDrawTargets) + , fOptionsForOpLists(optionsForOpLists) , fOptionsForPathRendererChain(optionsForPathRendererChain) , fSingleOwner(singleOwner) , fAbandoned(false) @@ -91,14 +93,14 @@ private: static const int kNumDFTOptions = 2; // DFT or no DFT GrContext* fContext; - GrDrawTarget::Options fOptionsForDrawTargets; + GrRenderTargetOpList::Options fOptionsForOpLists; GrPathRendererChain::Options fOptionsForPathRendererChain; // In debug builds we guard against improper thread handling GrSingleOwner* fSingleOwner; bool fAbandoned; - SkTDArray fDrawTargets; + SkTDArray fOpLists; SkAutoTDelete fAtlasTextContext; diff --git a/src/gpu/GrGpu.h b/src/gpu/GrGpu.h index b8703dc68e..ba7ed6f47e 100644 --- a/src/gpu/GrGpu.h +++ b/src/gpu/GrGpu.h @@ -331,7 +331,7 @@ public: ResetTimestamp getResetTimestamp() const { return fResetTimestamp; } // Called to perform a surface to surface copy. Fallbacks to issuing a draw from the src to dst - // take place at the GrDrawTarget level and this function implement faster copy paths. The rect + // take place at the GrOpList level and this function implement faster copy paths. The rect // and point are pre-clipped. The src rect and implied dst rect are guaranteed to be within the // src/dst bounds and non-empty. bool copySurface(GrSurface* dst, @@ -360,16 +360,16 @@ public: // multisample information itself. const MultisampleSpecs& getMultisampleSpecs(GrRenderTarget*, const GrStencilSettings&); - // Creates a GrGpuCommandBuffer in which the GrDrawTarget can send draw commands to instead of + // Creates a GrGpuCommandBuffer in which the GrOpList can send draw commands to instead of // directly to the Gpu object. virtual GrGpuCommandBuffer* createCommandBuffer( GrRenderTarget* target, const GrGpuCommandBuffer::LoadAndStoreInfo& colorInfo, const GrGpuCommandBuffer::LoadAndStoreInfo& stencilInfo) = 0; - // Called by drawtarget when flushing. + // Called by GrOpList when flushing. // Provides a hook for post-flush actions (e.g. PLS reset and Vulkan command buffer submits). - virtual void finishDrawTarget() {} + virtual void finishOpList() {} virtual GrFence SK_WARN_UNUSED_RESULT insertFence() const = 0; virtual bool waitFence(GrFence, uint64_t timeout = 1000) const = 0; diff --git a/src/gpu/GrOpList.cpp b/src/gpu/GrOpList.cpp new file mode 100644 index 0000000000..ac7f1b274d --- /dev/null +++ b/src/gpu/GrOpList.cpp @@ -0,0 +1,71 @@ +/* + * Copyright 2016 Google Inc. + * + * Use of this source code is governed by a BSD-style license that can be + * found in the LICENSE file. + */ + +#include "GrOpList.h" + +#include "GrRenderTarget.h" +#include "GrRenderTargetOpList.h" +#include "GrSurface.h" + +GrOpList::GrOpList(GrSurface* surface) + : fFlags(0) + , fTarget(surface) { + + surface->setLastOpList(this); + +#ifdef SK_DEBUG + static int debugID = 0; + fDebugID = debugID++; +#endif +} + +GrOpList::~GrOpList() { + if (fTarget && this == fTarget->getLastOpList()) { + fTarget->setLastOpList(nullptr); + } +} + +// Add a GrOpList-based dependency +void GrOpList::addDependency(GrOpList* dependedOn) { + SkASSERT(!dependedOn->dependsOn(this)); // loops are bad + + if (this->dependsOn(dependedOn)) { + return; // don't add duplicate dependencies + } + + *fDependencies.push() = dependedOn; +} + +// Convert from a GrSurface-based dependency to a GrOpList one +void GrOpList::addDependency(GrSurface* dependedOn) { + if (dependedOn->getLastOpList()) { + // If it is still receiving dependencies, this GrOpList shouldn't be closed + SkASSERT(!this->isClosed()); + + GrOpList* opList = dependedOn->getLastOpList(); + if (opList == this) { + // self-read - presumably for dst reads + } else { + this->addDependency(opList); + + // Can't make it closed in the self-read case + opList->makeClosed(); + } + } +} + +#ifdef SK_DEBUG +void GrOpList::dump() const { + SkDebugf("--------------------------------------------------------------\n"); + SkDebugf("node: %d -> RT: %d\n", fDebugID, fTarget ? fTarget->uniqueID() : -1); + SkDebugf("relies On (%d): ", fDependencies.count()); + for (int i = 0; i < fDependencies.count(); ++i) { + SkDebugf("%d, ", fDependencies[i]->fDebugID); + } + SkDebugf("\n"); +} +#endif diff --git a/src/gpu/GrOpList.h b/src/gpu/GrOpList.h new file mode 100644 index 0000000000..4af85128b6 --- /dev/null +++ b/src/gpu/GrOpList.h @@ -0,0 +1,124 @@ +/* + * Copyright 2016 Google Inc. + * + * Use of this source code is governed by a BSD-style license that can be + * found in the LICENSE file. + */ + +#ifndef GrOpList_DEFINED +#define GrOpList_DEFINED + +#include "SkRefCnt.h" +#include "SkTDArray.h" + +//#define ENABLE_MDB 1 + +class GrBatchFlushState; +class GrSurface; + +class GrOpList : public SkRefCnt { +public: + GrOpList(GrSurface* surface); + ~GrOpList() override; + + virtual void prepareBatches(GrBatchFlushState* flushState) = 0; + virtual bool drawBatches(GrBatchFlushState* flushState) = 0; + + virtual void makeClosed() { + // We only close GrOpLists when MDB is enabled. When MDB is disabled there is only + // ever one GrOpLists and all calls will be funnelled into it. +#ifdef ENABLE_MDB + this->setFlag(kClosed_Flag); +#endif + } + + // TODO: it seems a bit odd that GrOpList has nothing to clear on reset + virtual void reset() = 0; + + // TODO: in an MDB world, where the OpLists don't allocate GPU resources, it seems like + // these could go away + virtual void abandonGpuResources() = 0; + virtual void freeGpuResources() = 0; + + // TODO: this entry point is only needed in the non-MDB world. Remove when + // we make the switch to MDB + void clearTarget() { fTarget = nullptr; } + + bool isClosed() const { return this->isSetFlag(kClosed_Flag); } + + /* + * Notify this GrOpList that it relies on the contents of 'dependedOn' + */ + void addDependency(GrSurface* dependedOn); + + /* + * Does this opList depend on 'dependedOn'? + */ + bool dependsOn(GrOpList* dependedOn) const { + return fDependencies.find(dependedOn) >= 0; + } + + /* + * Dump out the GrOpList dependency DAG + */ + SkDEBUGCODE(virtual void dump() const;) + +private: + friend class GrDrawingManager; // for resetFlag & TopoSortTraits + + enum Flags { + kClosed_Flag = 0x01, //!< This GrOpList can't accept any more batches + + kWasOutput_Flag = 0x02, //!< Flag for topological sorting + kTempMark_Flag = 0x04, //!< Flag for topological sorting + }; + + void setFlag(uint32_t flag) { + fFlags |= flag; + } + + void resetFlag(uint32_t flag) { + fFlags &= ~flag; + } + + bool isSetFlag(uint32_t flag) const { + return SkToBool(fFlags & flag); + } + + struct TopoSortTraits { + static void Output(GrOpList* dt, int /* index */) { + dt->setFlag(GrOpList::kWasOutput_Flag); + } + static bool WasOutput(const GrOpList* dt) { + return dt->isSetFlag(GrOpList::kWasOutput_Flag); + } + static void SetTempMark(GrOpList* dt) { + dt->setFlag(GrOpList::kTempMark_Flag); + } + static void ResetTempMark(GrOpList* dt) { + dt->resetFlag(GrOpList::kTempMark_Flag); + } + static bool IsTempMarked(const GrOpList* dt) { + return dt->isSetFlag(GrOpList::kTempMark_Flag); + } + static int NumDependencies(const GrOpList* dt) { + return dt->fDependencies.count(); + } + static GrOpList* Dependency(GrOpList* dt, int index) { + return dt->fDependencies[index]; + } + }; + + void addDependency(GrOpList* dependedOn); + + SkDEBUGCODE(int fDebugID;) + uint32_t fFlags; + GrSurface* fTarget; + + // 'this' GrOpList relies on the output of the GrOpLists in 'fDependencies' + SkTDArray fDependencies; + + typedef SkRefCnt INHERITED; +}; + +#endif diff --git a/src/gpu/GrPathRenderer.h b/src/gpu/GrPathRenderer.h index 37cc3f9863..3ac18144e0 100644 --- a/src/gpu/GrPathRenderer.h +++ b/src/gpu/GrPathRenderer.h @@ -22,7 +22,7 @@ class GrFixedClip; struct GrPoint; /** - * Base class for drawing paths into a GrDrawTarget. + * Base class for drawing paths into a GrOpList. * * Derived classes can use stages GrPaint::kTotalStages through GrPipelineBuilder::kNumStages-1. * The stages before GrPaint::kTotalStages are reserved for setting up the draw (i.e., textures and diff --git a/src/gpu/GrPipeline.cpp b/src/gpu/GrPipeline.cpp index d9ebcf85f9..5e658fcc87 100644 --- a/src/gpu/GrPipeline.cpp +++ b/src/gpu/GrPipeline.cpp @@ -9,10 +9,10 @@ #include "GrCaps.h" #include "GrDrawContext.h" -#include "GrDrawTarget.h" #include "GrGpu.h" #include "GrPipelineBuilder.h" #include "GrProcOptInfo.h" +#include "GrRenderTargetOpList.h" #include "GrRenderTargetPriv.h" #include "GrXferProcessor.h" @@ -178,8 +178,8 @@ GrPipeline* GrPipeline::CreateAt(void* memory, const CreateArgs& args, static void add_dependencies_for_processor(const GrFragmentProcessor* proc, GrRenderTarget* rt) { GrFragmentProcessor::TextureAccessIter iter(proc); while (const GrTextureAccess* access = iter.next()) { - SkASSERT(rt->getLastDrawTarget()); - rt->getLastDrawTarget()->addDependency(access->getTexture()); + SkASSERT(rt->getLastOpList()); + rt->getLastOpList()->addDependency(access->getTexture()); } } @@ -192,8 +192,8 @@ void GrPipeline::addDependenciesTo(GrRenderTarget* rt) const { for (int i = 0; i < xfer.numTextures(); ++i) { GrTexture* texture = xfer.textureAccess(i).getTexture(); - SkASSERT(rt->getLastDrawTarget()); - rt->getLastDrawTarget()->addDependency(texture); + SkASSERT(rt->getLastOpList()); + rt->getLastOpList()->addDependency(texture); } } diff --git a/src/gpu/GrPipeline.h b/src/gpu/GrPipeline.h index 845044f720..6330de18d9 100644 --- a/src/gpu/GrPipeline.h +++ b/src/gpu/GrPipeline.h @@ -105,8 +105,8 @@ public: /////////////////////////////////////////////////////////////////////////// /// @name GrFragmentProcessors - // Make the renderTarget's drawTarget (if it exists) be dependent on any - // drawTargets in this pipeline + // Make the renderTarget's GrOpList (if it exists) be dependent on any + // GrOpLists in this pipeline void addDependenciesTo(GrRenderTarget* rt) const; int numColorFragmentProcessors() const { return fNumColorProcessors; } diff --git a/src/gpu/GrPipelineBuilder.h b/src/gpu/GrPipelineBuilder.h index 0c33eb344f..cb69756b69 100644 --- a/src/gpu/GrPipelineBuilder.h +++ b/src/gpu/GrPipelineBuilder.h @@ -311,7 +311,10 @@ private: FragmentProcessorArray fCoverageFragmentProcessors; friend class GrPipeline; - friend class GrDrawTarget; + // This gives the GrRenderTargetOpList raw access to fColorFragmentProcessors & + // fCoverageFragmentProcessors + // TODO: that access seems a little dodgy + friend class GrRenderTargetOpList; }; #endif diff --git a/src/gpu/GrPrimitiveProcessor.h b/src/gpu/GrPrimitiveProcessor.h index 00b4df01ed..561f54be03 100644 --- a/src/gpu/GrPrimitiveProcessor.h +++ b/src/gpu/GrPrimitiveProcessor.h @@ -171,7 +171,7 @@ public: const Attribute& getAttrib(int index) const { return fAttribs[index]; } // Returns the vertex stride of the GP. A common use case is to request geometry from a - // drawtarget based off of the stride, and to populate this memory using an implicit array of + // GrOpList based off of the stride, and to populate this memory using an implicit array of // structs. In this case, it is best to assert the vertexstride == sizeof(VertexStruct). size_t getVertexStride() const { return fVertexStride; } diff --git a/src/gpu/GrRenderTarget.cpp b/src/gpu/GrRenderTarget.cpp index 2053a166e5..d28b2b4191 100644 --- a/src/gpu/GrRenderTarget.cpp +++ b/src/gpu/GrRenderTarget.cpp @@ -11,8 +11,8 @@ #include "GrContext.h" #include "GrContextPriv.h" #include "GrDrawContext.h" -#include "GrDrawTarget.h" #include "GrGpu.h" +#include "GrRenderTargetOpList.h" #include "GrRenderTargetPriv.h" #include "GrStencilAttachment.h" @@ -21,20 +21,12 @@ GrRenderTarget::GrRenderTarget(GrGpu* gpu, const GrSurfaceDesc& desc, Flags flag : INHERITED(gpu, desc) , fStencilAttachment(stencil) , fMultisampleSpecsID(0) - , fFlags(flags) - , fLastDrawTarget(nullptr) { + , fFlags(flags) { SkASSERT(!(fFlags & Flags::kMixedSampled) || fDesc.fSampleCnt > 0); SkASSERT(!(fFlags & Flags::kWindowRectsSupport) || gpu->caps()->maxWindowRectangles() > 0); fResolveRect.setLargestInverted(); } -GrRenderTarget::~GrRenderTarget() { - if (fLastDrawTarget) { - fLastDrawTarget->clearRT(); - } - SkSafeUnref(fLastDrawTarget); -} - void GrRenderTarget::discard() { // go through context so that all necessary flushing occurs GrContext* context = this->getContext(); @@ -85,24 +77,12 @@ void GrRenderTarget::onAbandon() { SkSafeSetNull(fStencilAttachment); // The contents of this renderTarget are gone/invalid. It isn't useful to point back - // the creating drawTarget. - this->setLastDrawTarget(nullptr); + // the creating opList. + this->setLastOpList(nullptr); INHERITED::onAbandon(); } -void GrRenderTarget::setLastDrawTarget(GrDrawTarget* dt) { - if (fLastDrawTarget) { - // The non-MDB world never closes so we can't check this condition -#ifdef ENABLE_MDB - SkASSERT(fLastDrawTarget->isClosed()); -#endif - fLastDrawTarget->clearRT(); - } - - SkRefCnt_SafeAssign(fLastDrawTarget, dt); -} - /////////////////////////////////////////////////////////////////////////////// bool GrRenderTargetPriv::attachStencilAttachment(GrStencilAttachment* stencil) { diff --git a/src/gpu/GrRenderTargetOpList.cpp b/src/gpu/GrRenderTargetOpList.cpp new file mode 100644 index 0000000000..0dbb7772eb --- /dev/null +++ b/src/gpu/GrRenderTargetOpList.cpp @@ -0,0 +1,580 @@ +/* + * Copyright 2010 Google Inc. + * + * Use of this source code is governed by a BSD-style license that can be + * found in the LICENSE file. + */ + +#include "GrRenderTargetOpList.h" + +#include "GrAppliedClip.h" +#include "GrAuditTrail.h" +#include "GrCaps.h" +#include "GrDrawContext.h" +#include "GrGpu.h" +#include "GrGpuCommandBuffer.h" +#include "GrPath.h" +#include "GrPipeline.h" +#include "GrMemoryPool.h" +#include "GrPipelineBuilder.h" +#include "GrRenderTarget.h" +#include "GrResourceProvider.h" +#include "GrRenderTargetPriv.h" +#include "GrStencilAttachment.h" +#include "GrSurfacePriv.h" +#include "GrTexture.h" +#include "gl/GrGLRenderTarget.h" + +#include "SkStrokeRec.h" + +#include "batches/GrClearBatch.h" +#include "batches/GrClearStencilClipBatch.h" +#include "batches/GrCopySurfaceBatch.h" +#include "batches/GrDiscardBatch.h" +#include "batches/GrDrawBatch.h" +#include "batches/GrDrawPathBatch.h" +#include "batches/GrRectBatchFactory.h" +#include "batches/GrStencilPathBatch.h" + +#include "instanced/InstancedRendering.h" + +using gr_instanced::InstancedRendering; + +//////////////////////////////////////////////////////////////////////////////// + +// Experimentally we have found that most batching occurs within the first 10 comparisons. +static const int kDefaultMaxBatchLookback = 10; +static const int kDefaultMaxBatchLookahead = 10; + +GrRenderTargetOpList::GrRenderTargetOpList(GrRenderTarget* rt, GrGpu* gpu, + GrResourceProvider* resourceProvider, + GrAuditTrail* auditTrail, const Options& options) + : INHERITED(rt) + , fLastFullClearBatch(nullptr) + , fGpu(SkRef(gpu)) + , fResourceProvider(resourceProvider) + , fAuditTrail(auditTrail) { + // TODO: Stop extracting the context (currently needed by GrClip) + fContext = fGpu->getContext(); + + fClipBatchToBounds = options.fClipBatchToBounds; + fDrawBatchBounds = options.fDrawBatchBounds; + fMaxBatchLookback = (options.fMaxBatchLookback < 0) ? kDefaultMaxBatchLookback : + options.fMaxBatchLookback; + fMaxBatchLookahead = (options.fMaxBatchLookahead < 0) ? kDefaultMaxBatchLookahead : + options.fMaxBatchLookahead; + + if (GrCaps::InstancedSupport::kNone != this->caps()->instancedSupport()) { + fInstancedRendering.reset(fGpu->createInstancedRendering()); + } +} + +GrRenderTargetOpList::~GrRenderTargetOpList() { + fGpu->unref(); +} + +//////////////////////////////////////////////////////////////////////////////// + +#ifdef SK_DEBUG +void GrRenderTargetOpList::dump() const { + INHERITED::dump(); + + SkDebugf("batches (%d):\n", fRecordedBatches.count()); + for (int i = 0; i < fRecordedBatches.count(); ++i) { + SkDebugf("*******************************\n"); + if (!fRecordedBatches[i].fBatch) { + SkDebugf("%d: \n", i); + } else { + SkDebugf("%d: %s\n", i, fRecordedBatches[i].fBatch->name()); + SkString str = fRecordedBatches[i].fBatch->dumpInfo(); + SkDebugf("%s\n", str.c_str()); + const SkRect& clippedBounds = fRecordedBatches[i].fClippedBounds; + SkDebugf("ClippedBounds: [L: %.2f, T: %.2f, R: %.2f, B: %.2f]\n", + clippedBounds.fLeft, clippedBounds.fTop, clippedBounds.fRight, + clippedBounds.fBottom); + } + } +} +#endif + +bool GrRenderTargetOpList::setupDstReadIfNecessary(const GrPipelineBuilder& pipelineBuilder, + GrRenderTarget* rt, + const GrClip& clip, + const GrPipelineOptimizations& optimizations, + GrXferProcessor::DstTexture* dstTexture, + const SkRect& batchBounds) { + SkRect bounds = batchBounds; + bounds.outset(0.5f, 0.5f); + + if (!pipelineBuilder.willXPNeedDstTexture(*this->caps(), optimizations)) { + return true; + } + + if (this->caps()->textureBarrierSupport()) { + if (GrTexture* rtTex = rt->asTexture()) { + // The render target is a texture, so we can read from it directly in the shader. The XP + // will be responsible to detect this situation and request a texture barrier. + dstTexture->setTexture(rtTex); + dstTexture->setOffset(0, 0); + return true; + } + } + + SkIRect copyRect; + clip.getConservativeBounds(rt->width(), rt->height(), ©Rect); + + SkIRect drawIBounds; + bounds.roundOut(&drawIBounds); + if (!copyRect.intersect(drawIBounds)) { +#ifdef SK_DEBUG + GrCapsDebugf(this->caps(), "Missed an early reject. " + "Bailing on draw from setupDstReadIfNecessary.\n"); +#endif + return false; + } + + // MSAA consideration: When there is support for reading MSAA samples in the shader we could + // have per-sample dst values by making the copy multisampled. + GrSurfaceDesc desc; + if (!fGpu->initDescForDstCopy(rt, &desc)) { + desc.fOrigin = kDefault_GrSurfaceOrigin; + desc.fFlags = kRenderTarget_GrSurfaceFlag; + desc.fConfig = rt->config(); + } + + desc.fWidth = copyRect.width(); + desc.fHeight = copyRect.height(); + + static const uint32_t kFlags = 0; + SkAutoTUnref copy(fResourceProvider->createApproxTexture(desc, kFlags)); + + if (!copy) { + SkDebugf("Failed to create temporary copy of destination texture.\n"); + return false; + } + SkIPoint dstPoint = {0, 0}; + this->copySurface(copy, rt, copyRect, dstPoint); + dstTexture->setTexture(copy); + dstTexture->setOffset(copyRect.fLeft, copyRect.fTop); + return true; +} + +void GrRenderTargetOpList::prepareBatches(GrBatchFlushState* flushState) { + // Semi-usually the GrOpLists are already closed at this point, but sometimes Ganesh + // needs to flush mid-draw. In that case, the SkGpuDevice's GrOpLists won't be closed + // but need to be flushed anyway. Closing such GrOpLists here will mean new + // GrOpLists will be created to replace them if the SkGpuDevice(s) write to them again. + this->makeClosed(); + + // Loop over the batches that haven't yet generated their geometry + for (int i = 0; i < fRecordedBatches.count(); ++i) { + if (fRecordedBatches[i].fBatch) { + fRecordedBatches[i].fBatch->prepare(flushState); + } + } + + if (fInstancedRendering) { + fInstancedRendering->beginFlush(flushState->resourceProvider()); + } +} + +bool GrRenderTargetOpList::drawBatches(GrBatchFlushState* flushState) { + if (0 == fRecordedBatches.count()) { + return false; + } + // Draw all the generated geometry. + SkRandom random; + GrRenderTarget* currentRT = nullptr; + SkAutoTDelete commandBuffer; + for (int i = 0; i < fRecordedBatches.count(); ++i) { + if (!fRecordedBatches[i].fBatch) { + continue; + } + if (fRecordedBatches[i].fBatch->renderTarget() != currentRT) { + if (commandBuffer) { + commandBuffer->end(); + commandBuffer->submit(); + commandBuffer.reset(); + } + currentRT = fRecordedBatches[i].fBatch->renderTarget(); + if (currentRT) { + static const GrGpuCommandBuffer::LoadAndStoreInfo kBasicLoadStoreInfo + { GrGpuCommandBuffer::LoadOp::kLoad,GrGpuCommandBuffer::StoreOp::kStore, + GrColor_ILLEGAL }; + commandBuffer.reset(fGpu->createCommandBuffer(currentRT, + kBasicLoadStoreInfo, // Color + kBasicLoadStoreInfo)); // Stencil + } + flushState->setCommandBuffer(commandBuffer); + } + if (fDrawBatchBounds) { + const SkRect& bounds = fRecordedBatches[i].fClippedBounds; + SkIRect ibounds; + bounds.roundOut(&ibounds); + // In multi-draw buffer all the batches use the same render target and we won't need to + // get the batchs bounds. + if (GrRenderTarget* rt = fRecordedBatches[i].fBatch->renderTarget()) { + fGpu->drawDebugWireRect(rt, ibounds, 0xFF000000 | random.nextU()); + } + } + fRecordedBatches[i].fBatch->draw(flushState, fRecordedBatches[i].fClippedBounds); + } + if (commandBuffer) { + commandBuffer->end(); + commandBuffer->submit(); + flushState->setCommandBuffer(nullptr); + } + + fGpu->finishOpList(); + return true; +} + +void GrRenderTargetOpList::reset() { + fLastFullClearBatch = nullptr; + fRecordedBatches.reset(); + if (fInstancedRendering) { + fInstancedRendering->endFlush(); + } +} + +void GrRenderTargetOpList::abandonGpuResources() { + if (GrCaps::InstancedSupport::kNone != fContext->caps()->instancedSupport()) { + InstancedRendering* ir = this->instancedRendering(); + ir->resetGpuResources(InstancedRendering::ResetType::kAbandon); + } +} + +void GrRenderTargetOpList::freeGpuResources() { + if (GrCaps::InstancedSupport::kNone != fContext->caps()->instancedSupport()) { + InstancedRendering* ir = this->instancedRendering(); + ir->resetGpuResources(InstancedRendering::ResetType::kDestroy); + } +} + +static void batch_bounds(SkRect* bounds, const GrBatch* batch) { + *bounds = batch->bounds(); + if (batch->hasZeroArea()) { + if (batch->hasAABloat()) { + bounds->outset(0.5f, 0.5f); + } else { + // We don't know which way the particular GPU will snap lines or points at integer + // coords. So we ensure that the bounds is large enough for either snap. + SkRect before = *bounds; + bounds->roundOut(bounds); + if (bounds->fLeft == before.fLeft) { + bounds->fLeft -= 1; + } + if (bounds->fTop == before.fTop) { + bounds->fTop -= 1; + } + if (bounds->fRight == before.fRight) { + bounds->fRight += 1; + } + if (bounds->fBottom == before.fBottom) { + bounds->fBottom += 1; + } + } + } +} + +void GrRenderTargetOpList::drawBatch(const GrPipelineBuilder& pipelineBuilder, + GrDrawContext* drawContext, + const GrClip& clip, + GrDrawBatch* batch) { + // Setup clip + SkRect bounds; + batch_bounds(&bounds, batch); + GrAppliedClip appliedClip(bounds); + if (!clip.apply(fContext, drawContext, pipelineBuilder.isHWAntialias(), + pipelineBuilder.hasUserStencilSettings(), &appliedClip)) { + return; + } + + // TODO: this is the only remaining usage of the AutoRestoreFragmentProcessorState - remove it + GrPipelineBuilder::AutoRestoreFragmentProcessorState arfps; + if (appliedClip.clipCoverageFragmentProcessor()) { + arfps.set(&pipelineBuilder); + arfps.addCoverageFragmentProcessor(sk_ref_sp(appliedClip.clipCoverageFragmentProcessor())); + } + + if (pipelineBuilder.hasUserStencilSettings() || appliedClip.hasStencilClip()) { + if (!fResourceProvider->attachStencilAttachment(drawContext->accessRenderTarget())) { + SkDebugf("ERROR creating stencil attachment. Draw skipped.\n"); + return; + } + } + + GrPipeline::CreateArgs args; + args.fPipelineBuilder = &pipelineBuilder; + args.fDrawContext = drawContext; + args.fCaps = this->caps(); + batch->getPipelineOptimizations(&args.fOpts); + if (args.fOpts.fOverrides.fUsePLSDstRead || fClipBatchToBounds) { + GrGLIRect viewport; + viewport.fLeft = 0; + viewport.fBottom = 0; + viewport.fWidth = drawContext->width(); + viewport.fHeight = drawContext->height(); + SkIRect ibounds; + ibounds.fLeft = SkTPin(SkScalarFloorToInt(batch->bounds().fLeft), viewport.fLeft, + viewport.fWidth); + ibounds.fTop = SkTPin(SkScalarFloorToInt(batch->bounds().fTop), viewport.fBottom, + viewport.fHeight); + ibounds.fRight = SkTPin(SkScalarCeilToInt(batch->bounds().fRight), viewport.fLeft, + viewport.fWidth); + ibounds.fBottom = SkTPin(SkScalarCeilToInt(batch->bounds().fBottom), viewport.fBottom, + viewport.fHeight); + if (!appliedClip.addScissor(ibounds)) { + return; + } + } + args.fOpts.fColorPOI.completeCalculations( + sk_sp_address_as_pointer_address(pipelineBuilder.fColorFragmentProcessors.begin()), + pipelineBuilder.numColorFragmentProcessors()); + args.fOpts.fCoveragePOI.completeCalculations( + sk_sp_address_as_pointer_address(pipelineBuilder.fCoverageFragmentProcessors.begin()), + pipelineBuilder.numCoverageFragmentProcessors()); + args.fScissor = &appliedClip.scissorState(); + args.fWindowRectsState = &appliedClip.windowRectsState(); + args.fHasStencilClip = appliedClip.hasStencilClip(); + if (!this->setupDstReadIfNecessary(pipelineBuilder, drawContext->accessRenderTarget(), + clip, args.fOpts, + &args.fDstTexture, batch->bounds())) { + return; + } + + if (!batch->installPipeline(args)) { + return; + } + +#ifdef ENABLE_MDB + SkASSERT(fSurface); + batch->pipeline()->addDependenciesTo(fSurface); +#endif + this->recordBatch(batch, appliedClip.clippedDrawBounds()); +} + +void GrRenderTargetOpList::stencilPath(GrDrawContext* drawContext, + const GrClip& clip, + bool useHWAA, + const SkMatrix& viewMatrix, + const GrPath* path) { + // TODO: extract portions of checkDraw that are relevant to path stenciling. + SkASSERT(path); + SkASSERT(this->caps()->shaderCaps()->pathRenderingSupport()); + + // FIXME: Use path bounds instead of this WAR once + // https://bugs.chromium.org/p/skia/issues/detail?id=5640 is resolved. + SkRect bounds = SkRect::MakeIWH(drawContext->width(), drawContext->height()); + + // Setup clip + GrAppliedClip appliedClip(bounds); + if (!clip.apply(fContext, drawContext, useHWAA, true, &appliedClip)) { + return; + } + // TODO: respect fClipBatchToBounds if we ever start computing bounds here. + + // Coverage AA does not make sense when rendering to the stencil buffer. The caller should never + // attempt this in a situation that would require coverage AA. + SkASSERT(!appliedClip.clipCoverageFragmentProcessor()); + + GrStencilAttachment* stencilAttachment = fResourceProvider->attachStencilAttachment( + drawContext->accessRenderTarget()); + if (!stencilAttachment) { + SkDebugf("ERROR creating stencil attachment. Draw skipped.\n"); + return; + } + + GrBatch* batch = GrStencilPathBatch::Create(viewMatrix, + useHWAA, + path->getFillType(), + appliedClip.hasStencilClip(), + stencilAttachment->bits(), + appliedClip.scissorState(), + drawContext->accessRenderTarget(), + path); + this->recordBatch(batch, appliedClip.clippedDrawBounds()); + batch->unref(); +} + +void GrRenderTargetOpList::addBatch(sk_sp batch) { + this->recordBatch(batch.get(), batch->bounds()); +} + +void GrRenderTargetOpList::fullClear(GrRenderTarget* renderTarget, GrColor color) { + // Currently this just inserts or updates the last clear batch. However, once in MDB this can + // remove all the previously recorded batches and change the load op to clear with supplied + // color. + if (fLastFullClearBatch && + fLastFullClearBatch->renderTargetUniqueID() == renderTarget->uniqueID()) { + // As currently implemented, fLastFullClearBatch should be the last batch because we would + // have cleared it when another batch was recorded. + SkASSERT(fRecordedBatches.back().fBatch.get() == fLastFullClearBatch); + fLastFullClearBatch->setColor(color); + return; + } + sk_sp batch(GrClearBatch::Make(GrFixedClip::Disabled(), color, renderTarget)); + if (batch.get() == this->recordBatch(batch.get(), batch->bounds())) { + fLastFullClearBatch = batch.get(); + } +} + +void GrRenderTargetOpList::discard(GrRenderTarget* renderTarget) { + // Currently this just inserts a discard batch. However, once in MDB this can remove all the + // previously recorded batches and change the load op to discard. + if (this->caps()->discardRenderTargetSupport()) { + GrBatch* batch = new GrDiscardBatch(renderTarget); + this->recordBatch(batch, batch->bounds()); + batch->unref(); + } +} + +//////////////////////////////////////////////////////////////////////////////// + +bool GrRenderTargetOpList::copySurface(GrSurface* dst, + GrSurface* src, + const SkIRect& srcRect, + const SkIPoint& dstPoint) { + GrBatch* batch = GrCopySurfaceBatch::Create(dst, src, srcRect, dstPoint); + if (!batch) { + return false; + } +#ifdef ENABLE_MDB + this->addDependency(src); +#endif + + this->recordBatch(batch, batch->bounds()); + batch->unref(); + return true; +} + +static inline bool can_reorder(const SkRect& a, const SkRect& b) { + return a.fRight <= b.fLeft || a.fBottom <= b.fTop || + b.fRight <= a.fLeft || b.fBottom <= a.fTop; +} + +static void join(SkRect* out, const SkRect& a, const SkRect& b) { + SkASSERT(a.fLeft <= a.fRight && a.fTop <= a.fBottom); + SkASSERT(b.fLeft <= b.fRight && b.fTop <= b.fBottom); + out->fLeft = SkTMin(a.fLeft, b.fLeft); + out->fTop = SkTMin(a.fTop, b.fTop); + out->fRight = SkTMax(a.fRight, b.fRight); + out->fBottom = SkTMax(a.fBottom, b.fBottom); +} + +GrBatch* GrRenderTargetOpList::recordBatch(GrBatch* batch, const SkRect& clippedBounds) { + // A closed GrOpList should never receive new/more batches + SkASSERT(!this->isClosed()); + + // Check if there is a Batch Draw we can batch with by linearly searching back until we either + // 1) check every draw + // 2) intersect with something + // 3) find a 'blocker' + GR_AUDIT_TRAIL_ADDBATCH(fAuditTrail, batch); + GrBATCH_INFO("Re-Recording (%s, B%u)\n" + "\tBounds LRTB (%f, %f, %f, %f)\n", + batch->name(), + batch->uniqueID(), + batch->bounds().fLeft, batch->bounds().fRight, + batch->bounds().fTop, batch->bounds().fBottom); + GrBATCH_INFO(SkTabString(batch->dumpInfo(), 1).c_str()); + GrBATCH_INFO("\tClipped Bounds: [L: %.2f, T: %.2f, R: %.2f, B: %.2f]\n", + clippedBounds.fLeft, clippedBounds.fTop, clippedBounds.fRight, + clippedBounds.fBottom); + GrBATCH_INFO("\tOutcome:\n"); + int maxCandidates = SkTMin(fMaxBatchLookback, fRecordedBatches.count()); + if (maxCandidates) { + int i = 0; + while (true) { + GrBatch* candidate = fRecordedBatches.fromBack(i).fBatch.get(); + // We cannot continue to search backwards if the render target changes + if (candidate->renderTargetUniqueID() != batch->renderTargetUniqueID()) { + GrBATCH_INFO("\t\tBreaking because of (%s, B%u) Rendertarget\n", + candidate->name(), candidate->uniqueID()); + break; + } + if (candidate->combineIfPossible(batch, *this->caps())) { + GrBATCH_INFO("\t\tCombining with (%s, B%u)\n", candidate->name(), + candidate->uniqueID()); + GR_AUDIT_TRAIL_BATCHING_RESULT_COMBINED(fAuditTrail, candidate, batch); + join(&fRecordedBatches.fromBack(i).fClippedBounds, + fRecordedBatches.fromBack(i).fClippedBounds, clippedBounds); + return candidate; + } + // Stop going backwards if we would cause a painter's order violation. + const SkRect& candidateBounds = fRecordedBatches.fromBack(i).fClippedBounds; + if (!can_reorder(candidateBounds, clippedBounds)) { + GrBATCH_INFO("\t\tIntersects with (%s, B%u)\n", candidate->name(), + candidate->uniqueID()); + break; + } + ++i; + if (i == maxCandidates) { + GrBATCH_INFO("\t\tReached max lookback or beginning of batch array %d\n", i); + break; + } + } + } else { + GrBATCH_INFO("\t\tFirstBatch\n"); + } + GR_AUDIT_TRAIL_BATCHING_RESULT_NEW(fAuditTrail, batch); + fRecordedBatches.emplace_back(RecordedBatch{sk_ref_sp(batch), clippedBounds}); + fLastFullClearBatch = nullptr; + return batch; +} + +void GrRenderTargetOpList::forwardCombine() { + if (fMaxBatchLookahead <= 0) { + return; + } + for (int i = 0; i < fRecordedBatches.count() - 2; ++i) { + GrBatch* batch = fRecordedBatches[i].fBatch.get(); + const SkRect& batchBounds = fRecordedBatches[i].fClippedBounds; + int maxCandidateIdx = SkTMin(i + fMaxBatchLookahead, fRecordedBatches.count() - 1); + int j = i + 1; + while (true) { + GrBatch* candidate = fRecordedBatches[j].fBatch.get(); + // We cannot continue to search if the render target changes + if (candidate->renderTargetUniqueID() != batch->renderTargetUniqueID()) { + GrBATCH_INFO("\t\tBreaking because of (%s, B%u) Rendertarget\n", + candidate->name(), candidate->uniqueID()); + break; + } + if (j == i +1) { + // We assume batch would have combined with candidate when the candidate was added + // via backwards combining in recordBatch. + SkASSERT(!batch->combineIfPossible(candidate, *this->caps())); + } else if (batch->combineIfPossible(candidate, *this->caps())) { + GrBATCH_INFO("\t\tCombining with (%s, B%u)\n", candidate->name(), + candidate->uniqueID()); + GR_AUDIT_TRAIL_BATCHING_RESULT_COMBINED(fAuditTrail, batch, candidate); + fRecordedBatches[j].fBatch = std::move(fRecordedBatches[i].fBatch); + join(&fRecordedBatches[j].fClippedBounds, fRecordedBatches[j].fClippedBounds, + batchBounds); + break; + } + // Stop going traversing if we would cause a painter's order violation. + const SkRect& candidateBounds = fRecordedBatches[j].fClippedBounds; + if (!can_reorder(candidateBounds, batchBounds)) { + GrBATCH_INFO("\t\tIntersects with (%s, B%u)\n", candidate->name(), + candidate->uniqueID()); + break; + } + ++j; + if (j > maxCandidateIdx) { + GrBATCH_INFO("\t\tReached max lookahead or end of batch array %d\n", i); + break; + } + } + } +} + +/////////////////////////////////////////////////////////////////////////////// + +void GrRenderTargetOpList::clearStencilClip(const GrFixedClip& clip, + bool insideStencilMask, + GrRenderTarget* rt) { + GrBatch* batch = new GrClearStencilClipBatch(clip, insideStencilMask, rt); + this->recordBatch(batch, batch->bounds()); + batch->unref(); +} diff --git a/src/gpu/GrRenderTargetOpList.h b/src/gpu/GrRenderTargetOpList.h new file mode 100644 index 0000000000..080ff69ce4 --- /dev/null +++ b/src/gpu/GrRenderTargetOpList.h @@ -0,0 +1,175 @@ +/* + * Copyright 2010 Google Inc. + * + * Use of this source code is governed by a BSD-style license that can be + * found in the LICENSE file. + */ + +#ifndef GrRenderTargetOpList_DEFINED +#define GrRenderTargetOpList_DEFINED + +#include "GrClip.h" +#include "GrContext.h" +#include "GrOpList.h" +#include "GrPathProcessor.h" +#include "GrPrimitiveProcessor.h" +#include "GrPathRendering.h" +#include "GrXferProcessor.h" + +#include "batches/GrDrawBatch.h" + +#include "SkClipStack.h" +#include "SkMatrix.h" +#include "SkPath.h" +#include "SkStringUtils.h" +#include "SkStrokeRec.h" +#include "SkTArray.h" +#include "SkTLazy.h" +#include "SkTypes.h" +#include "SkXfermode.h" + +class GrAuditTrail; +class GrBatch; +class GrClearBatch; +class GrClip; +class GrCaps; +class GrPath; +class GrDrawPathBatchBase; +class GrPipelineBuilder; + +class GrRenderTargetOpList final : public GrOpList { +public: + /** Options for GrRenderTargetOpList behavior. */ + struct Options { + Options () + : fClipBatchToBounds(false) + , fDrawBatchBounds(false) + , fMaxBatchLookback(-1) + , fMaxBatchLookahead(-1) {} + bool fClipBatchToBounds; + bool fDrawBatchBounds; + int fMaxBatchLookback; + int fMaxBatchLookahead; + }; + + GrRenderTargetOpList(GrRenderTarget*, GrGpu*, GrResourceProvider*, + GrAuditTrail*, const Options&); + + ~GrRenderTargetOpList() override; + + void makeClosed() override { + INHERITED::makeClosed(); + + fLastFullClearBatch = nullptr; + this->forwardCombine(); + } + + /** + * Empties the draw buffer of any queued up draws. + */ + void reset() override; + + void abandonGpuResources() override; + void freeGpuResources() override; + + /** + * Together these two functions flush all queued up draws to GrCommandBuffer. The return value + * of drawBatches() indicates whether any commands were actually issued to the GPU. + */ + void prepareBatches(GrBatchFlushState* flushState) override; + bool drawBatches(GrBatchFlushState* flushState) override; + + /** + * Gets the capabilities of the draw target. + */ + const GrCaps* caps() const { return fGpu->caps(); } + + void drawBatch(const GrPipelineBuilder&, GrDrawContext*, const GrClip&, GrDrawBatch*); + + void addBatch(sk_sp); + + /** + * Draws the path into user stencil bits. Upon return, all user stencil values + * inside the path will be nonzero. The path's fill must be either even/odd or + * winding (notnverse or hairline).It will respect the HW antialias boolean (if + * possible in the 3D API). Note, we will never have an inverse fill with + * stencil path. + */ + void stencilPath(GrDrawContext*, + const GrClip&, + bool useHWAA, + const SkMatrix& viewMatrix, + const GrPath*); + + /** Clears the entire render target */ + void fullClear(GrRenderTarget*, GrColor color); + + /** Discards the contents render target. */ + void discard(GrRenderTarget*); + + /** + * Copies a pixel rectangle from one surface to another. This call may finalize + * reserved vertex/index data (as though a draw call was made). The src pixels + * copied are specified by srcRect. They are copied to a rect of the same + * size in dst with top left at dstPoint. If the src rect is clipped by the + * src bounds then pixel values in the dst rect corresponding to area clipped + * by the src rect are not overwritten. This method is not guaranteed to succeed + * depending on the type of surface, configs, etc, and the backend-specific + * limitations. + */ + bool copySurface(GrSurface* dst, + GrSurface* src, + const SkIRect& srcRect, + const SkIPoint& dstPoint); + + gr_instanced::InstancedRendering* instancedRendering() const { + SkASSERT(fInstancedRendering); + return fInstancedRendering; + } + + SkDEBUGCODE(void dump() const override;) + +private: + friend class GrDrawContextPriv; // for clearStencilClip + + // Returns the batch that the input batch was combined with or the input batch if it wasn't + // combined. + GrBatch* recordBatch(GrBatch*, const SkRect& clippedBounds); + void forwardCombine(); + + // Makes a copy of the dst if it is necessary for the draw. Returns false if a copy is required + // but couldn't be made. Otherwise, returns true. This method needs to be protected because it + // needs to be accessed by GLPrograms to setup a correct drawstate + bool setupDstReadIfNecessary(const GrPipelineBuilder&, + GrRenderTarget*, + const GrClip&, + const GrPipelineOptimizations& optimizations, + GrXferProcessor::DstTexture*, + const SkRect& batchBounds); + + // Used only by drawContextPriv. + void clearStencilClip(const GrFixedClip&, bool insideStencilMask, GrRenderTarget*); + + struct RecordedBatch { + sk_sp fBatch; + SkRect fClippedBounds; + }; + SkSTArray<256, RecordedBatch, true> fRecordedBatches; + GrClearBatch* fLastFullClearBatch; + // The context is only in service of the GrClip, remove once it doesn't need this. + GrContext* fContext; + GrGpu* fGpu; + GrResourceProvider* fResourceProvider; + GrAuditTrail* fAuditTrail; + + bool fClipBatchToBounds; + bool fDrawBatchBounds; + int fMaxBatchLookback; + int fMaxBatchLookahead; + + SkAutoTDelete fInstancedRendering; + + typedef GrOpList INHERITED; +}; + +#endif diff --git a/src/gpu/GrRenderTargetProxy.cpp b/src/gpu/GrRenderTargetProxy.cpp index fa6bd26577..5ea43d63d0 100644 --- a/src/gpu/GrRenderTargetProxy.cpp +++ b/src/gpu/GrRenderTargetProxy.cpp @@ -8,8 +8,9 @@ #include "GrRenderTargetProxy.h" #include "GrCaps.h" -#include "GrDrawTarget.h" #include "GrGpuResourcePriv.h" +#include "GrRenderTargetOpList.h" +#include "GrTextureProvider.h" // Deferred version // TODO: we can probably munge the 'desc' in both the wrapped and deferred @@ -18,8 +19,7 @@ GrRenderTargetProxy::GrRenderTargetProxy(const GrCaps& caps, const GrSurfaceDesc SkBackingFit fit, SkBudgeted budgeted) : INHERITED(desc, fit, budgeted) , fTarget(nullptr) - , fFlags(GrRenderTargetPriv::Flags::kNone) - , fLastDrawTarget(nullptr) { + , fFlags(GrRenderTargetPriv::Flags::kNone) { // Since we know the newly created render target will be internal, we are able to precompute // what the flags will ultimately end up being. if (caps.usesMixedSamples() && fDesc.fSampleCnt > 0) { @@ -35,15 +35,7 @@ GrRenderTargetProxy::GrRenderTargetProxy(const GrCaps& caps, sk_spdesc(), SkBackingFit::kExact, rt->resourcePriv().isBudgeted(), rt->uniqueID()) , fTarget(std::move(rt)) - , fFlags(fTarget->renderTargetPriv().flags()) - , fLastDrawTarget(nullptr) { -} - -GrRenderTargetProxy::~GrRenderTargetProxy() { - if (fLastDrawTarget) { - fLastDrawTarget->clearRT(); - } - SkSafeUnref(fLastDrawTarget); + , fFlags(fTarget->renderTargetPriv().flags()) { } GrRenderTarget* GrRenderTargetProxy::instantiate(GrTextureProvider* texProvider) { @@ -73,18 +65,6 @@ GrRenderTarget* GrRenderTargetProxy::instantiate(GrTextureProvider* texProvider) return fTarget.get(); } -void GrRenderTargetProxy::setLastDrawTarget(GrDrawTarget* dt) { - if (fLastDrawTarget) { - // The non-MDB world never closes so we can't check this condition -#ifdef ENABLE_MDB - SkASSERT(fLastDrawTarget->isClosed()); -#endif - fLastDrawTarget->clearRT(); - } - - SkRefCnt_SafeAssign(fLastDrawTarget, dt); -} - sk_sp GrRenderTargetProxy::Make(const GrCaps& caps, const GrSurfaceDesc& desc, SkBackingFit fit, diff --git a/src/gpu/GrResourceProvider.h b/src/gpu/GrResourceProvider.h index b4d72d5338..4c90cddd4d 100644 --- a/src/gpu/GrResourceProvider.h +++ b/src/gpu/GrResourceProvider.h @@ -96,7 +96,7 @@ public: enum Flags { /** If the caller intends to do direct reads/writes to/from the CPU then this flag must be - * set when accessing resources during a GrDrawTarget flush. This includes the execution of + * set when accessing resources during a GrOpList flush. This includes the execution of * GrBatch objects. The reason is that these memory operations are done immediately and * will occur out of order WRT the operations being flushed. * Make this automatic: https://bug.skia.org/4156 diff --git a/src/gpu/GrSurface.cpp b/src/gpu/GrSurface.cpp index 3c9368e67f..9fe00ded5d 100644 --- a/src/gpu/GrSurface.cpp +++ b/src/gpu/GrSurface.cpp @@ -7,6 +7,7 @@ #include "GrSurface.h" #include "GrContext.h" +#include "GrOpList.h" #include "GrSurfacePriv.h" #include "SkBitmap.h" @@ -14,6 +15,16 @@ #include "SkImageEncoder.h" #include +GrSurface::~GrSurface() { + if (fLastOpList) { + fLastOpList->clearTarget(); + } + SkSafeUnref(fLastOpList); + + // check that invokeReleaseProc has been called (if needed) + SkASSERT(NULL == fReleaseProc); +} + size_t GrSurface::WorstCaseSize(const GrSurfaceDesc& desc) { size_t size; @@ -195,3 +206,15 @@ void GrSurface::onAbandon() { this->invokeReleaseProc(); this->INHERITED::onAbandon(); } + +void GrSurface::setLastOpList(GrOpList* opList) { + if (fLastOpList) { + // The non-MDB world never closes so we can't check this condition +#ifdef ENABLE_MDB + SkASSERT(fLastOpList->isClosed()); +#endif + fLastOpList->clearTarget(); + } + + SkRefCnt_SafeAssign(fLastOpList, opList); +} diff --git a/src/gpu/GrSurfaceProxy.cpp b/src/gpu/GrSurfaceProxy.cpp index f5c401f670..9cd84ed4c6 100644 --- a/src/gpu/GrSurfaceProxy.cpp +++ b/src/gpu/GrSurfaceProxy.cpp @@ -7,3 +7,23 @@ #include "GrSurfaceProxy.h" +#include "GrOpList.h" + +GrSurfaceProxy::~GrSurfaceProxy() { + if (fLastOpList) { + fLastOpList->clearTarget(); + } + SkSafeUnref(fLastOpList); +} + +void GrSurfaceProxy::setLastOpList(GrOpList* opList) { + if (fLastOpList) { + // The non-MDB world never closes so we can't check this condition +#ifdef ENABLE_MDB + SkASSERT(fLastOpList->isClosed()); +#endif + fLastOpList->clearTarget(); + } + + SkRefCnt_SafeAssign(fLastOpList, opList); +} diff --git a/src/gpu/GrTracing.h b/src/gpu/GrTracing.h index 273aa65989..1f2a04ace4 100644 --- a/src/gpu/GrTracing.h +++ b/src/gpu/GrTracing.h @@ -57,7 +57,7 @@ private: /** * GR_CREATE_TRACE_MARKER will place begin and end trace markers for both * cpu and gpu (if gpu tracing enabled) for the current scope. - * name is of type const char* and target is of type GrDrawTarget* + * name is of type const char* and target is of type GrOpList* */ #define GR_CREATE_TRACE_MARKER(name, target) \ /* Chromium tracing */ \ diff --git a/src/gpu/GrUserStencilSettings.h b/src/gpu/GrUserStencilSettings.h index 32fb1396f9..18002591ec 100644 --- a/src/gpu/GrUserStencilSettings.h +++ b/src/gpu/GrUserStencilSettings.h @@ -13,12 +13,12 @@ /** * Gr uses the stencil buffer to implement complex clipping inside the - * GrDrawTarget class. The GrDrawTarget makes a subset of the stencil buffer + * GrOpList class. The GrOpList makes a subset of the stencil buffer * bits available for other uses by external code (user bits). Client code can - * modify these bits. GrDrawTarget will ignore ref, mask, and writemask bits + * modify these bits. GrOpList will ignore ref, mask, and writemask bits * provided by clients that fall outside the user range. * - * When code outside the GrDrawTarget class uses the stencil buffer the contract + * When code outside the GrOpList class uses the stencil buffer the contract * is as follows: * * > Normal stencil funcs allow the client to pass / fail regardless of the diff --git a/src/gpu/batches/GrDrawBatch.h b/src/gpu/batches/GrDrawBatch.h index 5f37b7b000..e675f5fc4d 100644 --- a/src/gpu/batches/GrDrawBatch.h +++ b/src/gpu/batches/GrDrawBatch.h @@ -42,7 +42,7 @@ private: }; /** - * Base class for GrBatches that draw. These batches have a GrPipeline installed by GrDrawTarget. + * Base class for GrBatches that draw. These batches have a GrPipeline installed by GrOpList. */ class GrDrawBatch : public GrBatch { public: diff --git a/src/gpu/batches/GrTessellatingPathRenderer.cpp b/src/gpu/batches/GrTessellatingPathRenderer.cpp index 57eac3c4b2..cb4f2c79ed 100644 --- a/src/gpu/batches/GrTessellatingPathRenderer.cpp +++ b/src/gpu/batches/GrTessellatingPathRenderer.cpp @@ -12,7 +12,6 @@ #include "GrBatchTest.h" #include "GrClip.h" #include "GrDefaultGeoProcFactory.h" -#include "GrDrawTarget.h" #include "GrMesh.h" #include "GrPathUtils.h" #include "GrPipelineBuilder.h" diff --git a/src/gpu/gl/GrGLGpu.cpp b/src/gpu/gl/GrGLGpu.cpp index 1e8049727b..3a32e2b67d 100644 --- a/src/gpu/gl/GrGLGpu.cpp +++ b/src/gpu/gl/GrGLGpu.cpp @@ -2238,7 +2238,7 @@ void GrGLGpu::clearStencilClip(const GrFixedClip& clip, #else // we could just clear the clip bit but when we go through // ANGLE a partial stencil mask will cause clears to be - // turned into draws. Our contract on GrDrawTarget says that + // turned into draws. Our contract on GrOpList says that // changing the clip between stencil passes may or may not // zero the client's clip bits. So we just clear the whole thing. static const GrGLint clipStencilMask = ~0; @@ -2616,7 +2616,7 @@ GrGpuCommandBuffer* GrGLGpu::createCommandBuffer( return new GrGLGpuCommandBuffer(this, static_cast(target)); } -void GrGLGpu::finishDrawTarget() { +void GrGLGpu::finishOpList() { if (fPLSHasBeenUsed) { /* There is an ARM driver bug where if we use PLS, and then draw a frame which does not * use PLS, it leaves garbage all over the place. As a workaround, we use PLS in a diff --git a/src/gpu/gl/GrGLGpu.h b/src/gpu/gl/GrGLGpu.h index 7ba79b2eea..0160835c6e 100644 --- a/src/gpu/gl/GrGLGpu.h +++ b/src/gpu/gl/GrGLGpu.h @@ -142,7 +142,7 @@ public: void drawDebugWireRect(GrRenderTarget*, const SkIRect&, GrColor) override; - void finishDrawTarget() override; + void finishOpList() override; GrFence SK_WARN_UNUSED_RESULT insertFence() const override; bool waitFence(GrFence, uint64_t timeout) const override; diff --git a/src/gpu/text/GrStencilAndCoverTextContext.h b/src/gpu/text/GrStencilAndCoverTextContext.h index 0f6735e133..c87041a9e4 100644 --- a/src/gpu/text/GrStencilAndCoverTextContext.h +++ b/src/gpu/text/GrStencilAndCoverTextContext.h @@ -25,7 +25,7 @@ class SkSurfaceProps; /* * This class implements text rendering using stencil and cover path rendering - * (by the means of GrDrawTarget::drawPath). + * (by the means of GrOpList::drawPath). */ class GrStencilAndCoverTextContext { public: diff --git a/src/gpu/vk/GrVkGpu.cpp b/src/gpu/vk/GrVkGpu.cpp index 0eb8241a94..b13e16282e 100644 --- a/src/gpu/vk/GrVkGpu.cpp +++ b/src/gpu/vk/GrVkGpu.cpp @@ -1323,7 +1323,7 @@ void GrVkGpu::addImageMemoryBarrier(VkPipelineStageFlags srcStageMask, barrier); } -void GrVkGpu::finishDrawTarget() { +void GrVkGpu::finishOpList() { // Submit the current command buffer to the Queue this->submitCommandBuffer(kSkip_SyncQueue); } diff --git a/src/gpu/vk/GrVkGpu.h b/src/gpu/vk/GrVkGpu.h index 273f28c5db..77eb24215c 100644 --- a/src/gpu/vk/GrVkGpu.h +++ b/src/gpu/vk/GrVkGpu.h @@ -139,7 +139,7 @@ public: GrVkRenderTarget*, const SkIRect& bounds); - void finishDrawTarget() override; + void finishOpList() override; GrFence SK_WARN_UNUSED_RESULT insertFence() const override; bool waitFence(GrFence, uint64_t timeout) const override; diff --git a/tools/gpu/GrTest.cpp b/tools/gpu/GrTest.cpp index 2e17902519..60c10e8226 100644 --- a/tools/gpu/GrTest.cpp +++ b/tools/gpu/GrTest.cpp @@ -65,9 +65,9 @@ void GrTestTarget::init(GrContext* ctx, sk_sp drawContext) { void GrContext::getTestTarget(GrTestTarget* tar, sk_sp drawContext) { this->flush(); SkASSERT(drawContext); - // We could create a proxy GrDrawTarget that passes through to fGpu until ~GrTextTarget() and + // We could create a proxy GrOpList 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 + // GrOpList and regular drawing. We could also assert or fail in GrContext drawing methods // until ~GrTestTarget(). tar->init(this, std::move(drawContext)); } @@ -252,7 +252,7 @@ void GrDrawContextPriv::testingOnly_drawBatch(const GrPaint& paint, pipelineBuilder.setState(GrPipelineBuilder::kSnapVerticesToPixelCenters_Flag, true); } - fDrawContext->getDrawTarget()->drawBatch(pipelineBuilder, fDrawContext, GrNoClip(), batch); + fDrawContext->getOpList()->drawBatch(pipelineBuilder, fDrawContext, GrNoClip(), batch); } #undef ASSERT_SINGLE_OWNER diff --git a/tools/gpu/GrTest.h b/tools/gpu/GrTest.h index 7b05a05146..86333b4f6c 100644 --- a/tools/gpu/GrTest.h +++ b/tools/gpu/GrTest.h @@ -20,8 +20,8 @@ namespace GrTest { }; /** TODO Please do not use this if you can avoid it. We are in the process of deleting it. - Allows a test to temporarily draw to a GrDrawTarget owned by a GrContext. Tests that use this - should be careful not to mix using the GrDrawTarget directly and drawing via SkCanvas or + Allows a test to temporarily draw to a GrOpList owned by a GrContext. Tests that use this + should be careful not to mix using the GrOpList directly and drawing via SkCanvas or GrContext. In the future this object may provide some guards to prevent this. */ class GrTestTarget { public: -- cgit v1.2.3