/* * 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 "GrAuditTrail.h" #include "GrCaps.h" #include "GrGpu.h" #include "GrGpuCommandBuffer.h" #include "GrRenderTarget.h" #include "GrRenderTargetContext.h" #include "GrResourceProvider.h" #include "ops/GrClearOp.h" #include "ops/GrCopySurfaceOp.h" #include "instanced/InstancedRendering.h" using gr_instanced::InstancedRendering; //////////////////////////////////////////////////////////////////////////////// // Experimentally we have found that most combining occurs within the first 10 comparisons. static const int kDefaultMaxOpLookback = 10; static const int kDefaultMaxOpLookahead = 10; GrRenderTargetOpList::GrRenderTargetOpList(sk_sp proxy, GrGpu* gpu, GrResourceProvider* resourceProvider, GrAuditTrail* auditTrail, const Options& options) : INHERITED(std::move(proxy), auditTrail) , fGpu(SkRef(gpu)) , fResourceProvider(resourceProvider) , fLastClipStackGenID(SK_InvalidUniqueID) , fClipAllocator(fClipAllocatorStorage, sizeof(fClipAllocatorStorage), sizeof(fClipAllocatorStorage)) { fMaxOpLookback = (options.fMaxOpCombineLookback < 0) ? kDefaultMaxOpLookback : options.fMaxOpCombineLookback; fMaxOpLookahead = (options.fMaxOpCombineLookahead < 0) ? kDefaultMaxOpLookahead : options.fMaxOpCombineLookahead; 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("ops (%d):\n", fRecordedOps.count()); for (int i = 0; i < fRecordedOps.count(); ++i) { SkDebugf("*******************************\n"); if (!fRecordedOps[i].fOp) { SkDebugf("%d: \n", i); } else { SkDebugf("%d: %s\n", i, fRecordedOps[i].fOp->name()); SkString str = fRecordedOps[i].fOp->dumpInfo(); SkDebugf("%s\n", str.c_str()); const SkRect& bounds = fRecordedOps[i].fOp->bounds(); SkDebugf("ClippedBounds: [L: %.2f, T: %.2f, R: %.2f, B: %.2f]\n", bounds.fLeft, bounds.fTop, bounds.fRight, bounds.fBottom); } } } void GrRenderTargetOpList::validateTargetsSingleRenderTarget() const { GrRenderTarget* rt = nullptr; for (int i = 0; i < fRecordedOps.count(); ++i) { if (!fRecordedOps[i].fOp) { continue; // combined forward } if (!rt) { rt = fRecordedOps[i].fRenderTarget.get(); } else { SkASSERT(fRecordedOps[i].fRenderTarget.get() == rt); } } } #endif void GrRenderTargetOpList::prepareOps(GrOpFlushState* flushState) { // MDB TODO: add SkASSERT(this->isClosed()); // Loop over the ops that haven't yet been prepared. for (int i = 0; i < fRecordedOps.count(); ++i) { if (fRecordedOps[i].fOp) { GrOpFlushState::DrawOpArgs opArgs; if (fRecordedOps[i].fRenderTarget) { opArgs = { fRecordedOps[i].fRenderTarget.get(), fRecordedOps[i].fAppliedClip, fRecordedOps[i].fDstTexture }; } flushState->setDrawOpArgs(&opArgs); fRecordedOps[i].fOp->prepare(flushState); flushState->setDrawOpArgs(nullptr); } } if (fInstancedRendering) { fInstancedRendering->beginFlush(flushState->resourceProvider()); } } static std::unique_ptr create_command_buffer(GrGpu* gpu) { static const GrGpuCommandBuffer::LoadAndStoreInfo kBasicLoadStoreInfo { GrGpuCommandBuffer::LoadOp::kLoad, GrGpuCommandBuffer::StoreOp::kStore, GrColor_ILLEGAL }; std::unique_ptr buffer( gpu->createCommandBuffer(kBasicLoadStoreInfo, // Color kBasicLoadStoreInfo)); // Stencil return buffer; } static inline void finish_command_buffer(GrGpuCommandBuffer* buffer) { if (!buffer) { return; } buffer->end(); buffer->submit(); } // TODO: this is where GrOp::renderTarget is used (which is fine since it // is at flush time). However, we need to store the RenderTargetProxy in the // Ops and instantiate them here. bool GrRenderTargetOpList::executeOps(GrOpFlushState* flushState) { if (0 == fRecordedOps.count()) { return false; } // Draw all the generated geometry. const GrRenderTarget* currentRenderTarget = fRecordedOps[0].fRenderTarget.get(); SkASSERT(currentRenderTarget); std::unique_ptr commandBuffer; for (int i = 0; i < fRecordedOps.count(); ++i) { if (!fRecordedOps[i].fOp) { continue; } SkASSERT(fRecordedOps[i].fRenderTarget.get()); if (fRecordedOps[i].fOp->needsCommandBufferIsolation()) { // This op is a special snowflake and must occur between command buffers // TODO: make this go through the command buffer finish_command_buffer(commandBuffer.get()); currentRenderTarget = fRecordedOps[i].fRenderTarget.get(); commandBuffer.reset(); flushState->setCommandBuffer(commandBuffer.get()); } else if (fRecordedOps[i].fRenderTarget.get() != currentRenderTarget) { // Changing renderTarget // MDB TODO: this code path goes away finish_command_buffer(commandBuffer.get()); currentRenderTarget = fRecordedOps[i].fRenderTarget.get(); commandBuffer = create_command_buffer(fGpu); flushState->setCommandBuffer(commandBuffer.get()); } else if (!commandBuffer) { commandBuffer = create_command_buffer(fGpu); flushState->setCommandBuffer(commandBuffer.get()); } GrOpFlushState::DrawOpArgs opArgs { fRecordedOps[i].fRenderTarget.get(), fRecordedOps[i].fAppliedClip, fRecordedOps[i].fDstTexture }; flushState->setDrawOpArgs(&opArgs); fRecordedOps[i].fOp->execute(flushState); flushState->setDrawOpArgs(nullptr); } finish_command_buffer(commandBuffer.get()); flushState->setCommandBuffer(nullptr); fGpu->finishOpList(); return true; } void GrRenderTargetOpList::reset() { fLastFullClearOp = nullptr; fLastFullClearResourceID.makeInvalid(); fLastFullClearProxyID.makeInvalid(); fRecordedOps.reset(); if (fInstancedRendering) { fInstancedRendering->endFlush(); } } void GrRenderTargetOpList::abandonGpuResources() { if (GrCaps::InstancedSupport::kNone != this->caps()->instancedSupport()) { InstancedRendering* ir = this->instancedRendering(); ir->resetGpuResources(InstancedRendering::ResetType::kAbandon); } } void GrRenderTargetOpList::freeGpuResources() { if (GrCaps::InstancedSupport::kNone != this->caps()->instancedSupport()) { InstancedRendering* ir = this->instancedRendering(); ir->resetGpuResources(InstancedRendering::ResetType::kDestroy); } } void GrRenderTargetOpList::fullClear(GrRenderTargetContext* renderTargetContext, GrColor color) { // MDB TODO: remove this. Right now we need the renderTargetContext for the // accessRenderTarget call. This method should just take the renderTargetProxy. GrRenderTarget* renderTarget = renderTargetContext->accessRenderTarget(); if (!renderTarget) { return; } // Currently this just inserts or updates the last clear op. However, once in MDB this can // remove all the previously recorded ops and change the load op to clear with supplied // color. // TODO: this needs to be updated to use GrSurfaceProxy::UniqueID SkASSERT((fLastFullClearResourceID == renderTarget->uniqueID()) == (fLastFullClearProxyID == renderTargetContext->asRenderTargetProxy()->uniqueID())); if (fLastFullClearResourceID == renderTarget->uniqueID()) { // As currently implemented, fLastFullClearOp should be the last op because we would // have cleared it when another op was recorded. SkASSERT(fRecordedOps.back().fOp.get() == fLastFullClearOp); GrOP_INFO("opList: %d Fusing clears (opID: %d Color: 0x%08x -> 0x%08x)\n", this->uniqueID(), fLastFullClearOp->uniqueID(), fLastFullClearOp->color(), color); fLastFullClearOp->setColor(color); return; } std::unique_ptr op(GrClearOp::Make(GrFixedClip::Disabled(), color, renderTargetContext)); if (!op) { return; } if (GrOp* clearOp = this->recordOp(std::move(op), renderTargetContext)) { // This is either the clear op we just created or another one that it combined with. fLastFullClearOp = static_cast(clearOp); fLastFullClearResourceID = renderTarget->uniqueID(); fLastFullClearProxyID = renderTargetContext->asRenderTargetProxy()->uniqueID(); } } //////////////////////////////////////////////////////////////////////////////// bool GrRenderTargetOpList::copySurface(GrResourceProvider* resourceProvider, GrRenderTargetContext* dst, GrSurfaceProxy* src, const SkIRect& srcRect, const SkIPoint& dstPoint) { std::unique_ptr op = GrCopySurfaceOp::Make(resourceProvider, dst->asSurfaceProxy(), src, srcRect, dstPoint); if (!op) { return false; } #ifdef ENABLE_MDB this->addDependency(src); #endif this->recordOp(std::move(op), dst); 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; } bool GrRenderTargetOpList::combineIfPossible(const RecordedOp& a, GrOp* b, const GrAppliedClip* bClip, const DstTexture* bDstTexture) { if (a.fAppliedClip) { if (!bClip) { return false; } if (*a.fAppliedClip != *bClip) { return false; } } else if (bClip) { return false; } if (bDstTexture) { if (a.fDstTexture != *bDstTexture) { return false; } } else if (a.fDstTexture.texture()) { return false; } return a.fOp->combineIfPossible(b, *this->caps()); } GrOp* GrRenderTargetOpList::recordOp(std::unique_ptr op, GrRenderTargetContext* renderTargetContext, GrAppliedClip* clip, const DstTexture* dstTexture) { GrRenderTarget* renderTarget = renderTargetContext->accessRenderTarget(); if (!renderTarget) { SkASSERT(false); return nullptr; } // A closed GrOpList should never receive new/more ops SkASSERT(!this->isClosed()); // Check if there is an op we can combine with by linearly searching back until we either // 1) check every op // 2) intersect with something // 3) find a 'blocker' GR_AUDIT_TRAIL_ADD_OP(fAuditTrail, op.get(), renderTarget->uniqueID(), renderTargetContext->asRenderTargetProxy()->uniqueID()); GrOP_INFO("opList: %d Recording (%s, opID: %u)\n" "\tBounds [L: %.2f, T: %.2f R: %.2f B: %.2f]\n", this->uniqueID(), op->name(), op->uniqueID(), op->bounds().fLeft, op->bounds().fTop, op->bounds().fRight, op->bounds().fBottom); GrOP_INFO(SkTabString(op->dumpInfo(), 1).c_str()); GrOP_INFO("\tOutcome:\n"); int maxCandidates = SkTMin(fMaxOpLookback, fRecordedOps.count()); // If we don't have a valid destination render target then we cannot reorder. if (maxCandidates && renderTarget) { int i = 0; while (true) { const RecordedOp& candidate = fRecordedOps.fromBack(i); // We cannot continue to search backwards if the render target changes if (candidate.fRenderTarget.get() != renderTarget) { GrOP_INFO("\t\tBackward: Breaking because of (%s, opID: %u) Rendertarget mismatch\n", candidate.fOp->name(), candidate.fOp->uniqueID()); break; } if (this->combineIfPossible(candidate, op.get(), clip, dstTexture)) { GrOP_INFO("\t\tBackward: Combining with (%s, opID: %u)\n", candidate.fOp->name(), candidate.fOp->uniqueID()); GrOP_INFO("\t\t\tBackward: Combined op info:\n"); GrOP_INFO(SkTabString(candidate.fOp->dumpInfo(), 4).c_str()); GR_AUDIT_TRAIL_OPS_RESULT_COMBINED(fAuditTrail, candidate.fOp.get(), op.get()); return candidate.fOp.get(); } // Stop going backwards if we would cause a painter's order violation. if (!can_reorder(fRecordedOps.fromBack(i).fOp->bounds(), op->bounds())) { GrOP_INFO("\t\tBackward: Intersects with (%s, opID: %u)\n", candidate.fOp->name(), candidate.fOp->uniqueID()); break; } ++i; if (i == maxCandidates) { GrOP_INFO("\t\tBackward: Reached max lookback or beginning of op array %d\n", i); break; } } } else { GrOP_INFO("\t\tBackward: FirstOp\n"); } GR_AUDIT_TRAIL_OP_RESULT_NEW(fAuditTrail, op); if (clip) { clip = fClipAllocator.make(std::move(*clip)); } fRecordedOps.emplace_back(std::move(op), renderTarget, clip, dstTexture); fRecordedOps.back().fOp->wasRecorded(); fLastFullClearOp = nullptr; fLastFullClearResourceID.makeInvalid(); fLastFullClearProxyID.makeInvalid(); return fRecordedOps.back().fOp.get(); } void GrRenderTargetOpList::forwardCombine() { SkASSERT(!this->isClosed()); if (fMaxOpLookahead <= 0) { return; } for (int i = 0; i < fRecordedOps.count() - 1; ++i) { GrOp* op = fRecordedOps[i].fOp.get(); GrRenderTarget* renderTarget = fRecordedOps[i].fRenderTarget.get(); SkASSERT(renderTarget); // If we don't have a valid destination render target ID then we cannot reorder. if (!renderTarget) { continue; } int maxCandidateIdx = SkTMin(i + fMaxOpLookahead, fRecordedOps.count() - 1); int j = i + 1; while (true) { const RecordedOp& candidate = fRecordedOps[j]; // We cannot continue to search if the render target changes if (candidate.fRenderTarget.get() != renderTarget) { GrOP_INFO("\t\tForward: Breaking because of (%s, opID: %u) Rendertarget\n", candidate.fOp->name(), candidate.fOp->uniqueID()); break; } if (this->combineIfPossible(fRecordedOps[i], candidate.fOp.get(), candidate.fAppliedClip, &candidate.fDstTexture)) { GrOP_INFO("\t\tForward: Combining with (%s, opID: %u)\n", candidate.fOp->name(), candidate.fOp->uniqueID()); GR_AUDIT_TRAIL_OPS_RESULT_COMBINED(fAuditTrail, op, candidate.fOp.get()); fRecordedOps[j].fOp = std::move(fRecordedOps[i].fOp); break; } // Stop going traversing if we would cause a painter's order violation. if (!can_reorder(fRecordedOps[j].fOp->bounds(), op->bounds())) { GrOP_INFO("\t\tForward: Intersects with (%s, opID: %u)\n", candidate.fOp->name(), candidate.fOp->uniqueID()); break; } ++j; if (j > maxCandidateIdx) { GrOP_INFO("\t\tForward: Reached max lookahead or end of op array %d\n", i); break; } } } }