/* * Copyright 2018 Google Inc. * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #include "GrCCDrawPathsOp.h" #include "GrContext.h" #include "GrContextPriv.h" #include "GrMemoryPool.h" #include "GrOpFlushState.h" #include "ccpr/GrCCPathCache.h" #include "ccpr/GrCCPerFlushResources.h" #include "ccpr/GrCoverageCountingPathRenderer.h" static bool has_coord_transforms(const GrPaint& paint) { GrFragmentProcessor::Iter iter(paint); while (const GrFragmentProcessor* fp = iter.next()) { if (!fp->coordTransforms().empty()) { return true; } } return false; } static int64_t area(const SkIRect& r) { return sk_64_mul(r.height(), r.width()); } std::unique_ptr GrCCDrawPathsOp::Make(GrContext* context, const SkIRect& clipIBounds, const SkMatrix& m, const GrShape& shape, const SkRect& devBounds, GrPaint&& paint) { SkIRect shapeDevIBounds; devBounds.roundOut(&shapeDevIBounds); // GrCCPathParser might find slightly tighter bounds. SkIRect maskDevIBounds; Visibility maskVisibility; if (clipIBounds.contains(shapeDevIBounds)) { maskDevIBounds = shapeDevIBounds; maskVisibility = Visibility::kComplete; } else { if (!maskDevIBounds.intersect(clipIBounds, shapeDevIBounds)) { return nullptr; } int64_t unclippedArea = area(shapeDevIBounds); int64_t clippedArea = area(maskDevIBounds); maskVisibility = (clippedArea >= unclippedArea/2 || unclippedArea < 100*100) ? Visibility::kMostlyComplete // i.e., visible enough to justify rendering the // whole thing if we think we can cache it. : Visibility::kPartial; } GrOpMemoryPool* pool = context->contextPriv().opMemoryPool(); return pool->allocate(m, shape, shapeDevIBounds, maskDevIBounds, maskVisibility, devBounds, std::move(paint)); } GrCCDrawPathsOp::GrCCDrawPathsOp(const SkMatrix& m, const GrShape& shape, const SkIRect& shapeDevIBounds, const SkIRect& maskDevIBounds, Visibility maskVisibility, const SkRect& devBounds, GrPaint&& paint) : GrDrawOp(ClassID()) , fViewMatrixIfUsingLocalCoords(has_coord_transforms(paint) ? m : SkMatrix::I()) , fDraws(m, shape, shapeDevIBounds, maskDevIBounds, maskVisibility, paint.getColor()) , fProcessors(std::move(paint)) { // Paint must be moved after fetching its color above. SkDEBUGCODE(fBaseInstance = -1); // FIXME: intersect with clip bounds to (hopefully) improve batching. // (This is nontrivial due to assumptions in generating the octagon cover geometry.) this->setBounds(devBounds, GrOp::HasAABloat::kYes, GrOp::IsZeroArea::kNo); } GrCCDrawPathsOp::~GrCCDrawPathsOp() { if (fOwningPerOpListPaths) { // Remove CCPR's dangling pointer to this Op before deleting it. fOwningPerOpListPaths->fDrawOps.remove(this); } } GrCCDrawPathsOp::SingleDraw::SingleDraw(const SkMatrix& m, const GrShape& shape, const SkIRect& shapeDevIBounds, const SkIRect& maskDevIBounds, Visibility maskVisibility, GrColor color) : fMatrix(m) , fShape(shape) , fShapeDevIBounds(shapeDevIBounds) , fMaskDevIBounds(maskDevIBounds) , fMaskVisibility(maskVisibility) , fColor(color) { #ifdef SK_BUILD_FOR_ANDROID_FRAMEWORK if (fShape.hasUnstyledKey()) { // On AOSP we round view matrix translates to integer values for cachable paths. We do this // to match HWUI's cache hit ratio, which doesn't consider the matrix when caching paths. fMatrix.setTranslateX(SkScalarRoundToScalar(fMatrix.getTranslateX())); fMatrix.setTranslateY(SkScalarRoundToScalar(fMatrix.getTranslateY())); } #endif } GrCCDrawPathsOp::SingleDraw::~SingleDraw() { if (fCacheEntry) { // All currFlushAtlas references must be reset back to null before the flush is finished. fCacheEntry->setCurrFlushAtlas(nullptr); } } GrDrawOp::RequiresDstTexture GrCCDrawPathsOp::finalize(const GrCaps& caps, const GrAppliedClip* clip) { SkASSERT(1 == fNumDraws); // There should only be one single path draw in this Op right now. GrProcessorSet::Analysis analysis = fProcessors.finalize(fDraws.head().fColor, GrProcessorAnalysisCoverage::kSingleChannel, clip, false, caps, &fDraws.head().fColor); return RequiresDstTexture(analysis.requiresDstTexture()); } bool GrCCDrawPathsOp::onCombineIfPossible(GrOp* op, const GrCaps&) { GrCCDrawPathsOp* that = op->cast(); SkASSERT(fOwningPerOpListPaths); SkASSERT(fNumDraws); SkASSERT(!that->fOwningPerOpListPaths || that->fOwningPerOpListPaths == fOwningPerOpListPaths); SkASSERT(that->fNumDraws); if (fProcessors != that->fProcessors || fViewMatrixIfUsingLocalCoords != that->fViewMatrixIfUsingLocalCoords) { return false; } fDraws.append(std::move(that->fDraws), &fOwningPerOpListPaths->fAllocator); this->joinBounds(*that); SkDEBUGCODE(fNumDraws += that->fNumDraws); SkDEBUGCODE(that->fNumDraws = 0); return true; } void GrCCDrawPathsOp::wasRecorded(GrCCPerOpListPaths* owningPerOpListPaths) { SkASSERT(1 == fNumDraws); SkASSERT(!fOwningPerOpListPaths); owningPerOpListPaths->fDrawOps.addToTail(this); fOwningPerOpListPaths = owningPerOpListPaths; } void GrCCDrawPathsOp::accountForOwnPaths(GrCCPathCache* pathCache, GrOnFlushResourceProvider* onFlushRP, const GrUniqueKey& stashedAtlasKey, GrCCPerFlushResourceSpecs* specs) { using CreateIfAbsent = GrCCPathCache::CreateIfAbsent; using MaskTransform = GrCCPathCache::MaskTransform; for (SingleDraw& draw : fDraws) { SkPath path; draw.fShape.asPath(&path); SkASSERT(!draw.fCacheEntry); if (pathCache) { MaskTransform m(draw.fMatrix, &draw.fCachedMaskShift); bool canStashPathMask = draw.fMaskVisibility >= Visibility::kMostlyComplete; draw.fCacheEntry = pathCache->find(draw.fShape, m, CreateIfAbsent(canStashPathMask)); } if (auto cacheEntry = draw.fCacheEntry.get()) { SkASSERT(!cacheEntry->currFlushAtlas()); // Shouldn't be set until setupResources(). if (cacheEntry->atlasKey().isValid()) { // Does the path already exist in a cached atlas? if (cacheEntry->hasCachedAtlas() && (draw.fCachedAtlasProxy = onFlushRP->findOrCreateProxyByUniqueKey( cacheEntry->atlasKey(), GrCCAtlas::kTextureOrigin))) { ++specs->fNumCachedPaths; continue; } // Does the path exist in the atlas that we stashed away from last flush? If so we // can copy it into a new 8-bit atlas and keep it in the resource cache. if (stashedAtlasKey.isValid() && stashedAtlasKey == cacheEntry->atlasKey()) { SkASSERT(!cacheEntry->hasCachedAtlas()); ++specs->fNumCopiedPaths; specs->fCopyPathStats.statPath(path); specs->fCopyAtlasSpecs.accountForSpace(cacheEntry->width(), cacheEntry->height()); continue; } // Whatever atlas the path used to reside in, it no longer exists. cacheEntry->resetAtlasKeyAndInfo(); } if (Visibility::kMostlyComplete == draw.fMaskVisibility && cacheEntry->hitCount() > 1 && SkTMax(draw.fShapeDevIBounds.height(), draw.fShapeDevIBounds.width()) <= onFlushRP->caps()->maxRenderTargetSize()) { // We've seen this path before with a compatible matrix, and it's mostly visible. // Just render the whole mask so we can try to cache it. draw.fMaskDevIBounds = draw.fShapeDevIBounds; draw.fMaskVisibility = Visibility::kComplete; } } ++specs->fNumRenderedPaths; specs->fRenderedPathStats.statPath(path); specs->fRenderedAtlasSpecs.accountForSpace(draw.fMaskDevIBounds.width(), draw.fMaskDevIBounds.height()); } } void GrCCDrawPathsOp::setupResources(GrOnFlushResourceProvider* onFlushRP, GrCCPerFlushResources* resources, DoCopiesToCache doCopies) { using DoEvenOddFill = GrCCPathProcessor::DoEvenOddFill; SkASSERT(fNumDraws > 0); SkASSERT(-1 == fBaseInstance); fBaseInstance = resources->nextPathInstanceIdx(); for (SingleDraw& draw : fDraws) { SkPath path; draw.fShape.asPath(&path); auto doEvenOddFill = DoEvenOddFill(SkPath::kEvenOdd_FillType == path.getFillType()); SkASSERT(SkPath::kEvenOdd_FillType == path.getFillType() || SkPath::kWinding_FillType == path.getFillType()); if (auto cacheEntry = draw.fCacheEntry.get()) { // Does the path already exist in a cached atlas texture? if (auto proxy = draw.fCachedAtlasProxy.get()) { SkASSERT(!cacheEntry->currFlushAtlas()); this->recordInstance(proxy, resources->nextPathInstanceIdx()); resources->appendDrawPathInstance().set(*cacheEntry, draw.fCachedMaskShift, draw.fColor); continue; } // Have we already encountered this path during the flush? (i.e. was the same SkPath // drawn more than once during the same flush, with a compatible matrix?) if (auto atlas = cacheEntry->currFlushAtlas()) { this->recordInstance(atlas->textureProxy(), resources->nextPathInstanceIdx()); resources->appendDrawPathInstance().set( *cacheEntry, draw.fCachedMaskShift, draw.fColor, cacheEntry->hasCachedAtlas() ? DoEvenOddFill::kNo : doEvenOddFill); continue; } // If the cache entry still has a valid atlas key at this point, it means the path // exists in the atlas that we stashed away from last flush. Copy it into a permanent // 8-bit atlas in the resource cache. if (DoCopiesToCache::kYes == doCopies && cacheEntry->atlasKey().isValid()) { SkIVector newOffset; GrCCAtlas* atlas = resources->copyPathToCachedAtlas(*cacheEntry, doEvenOddFill, &newOffset); cacheEntry->updateToCachedAtlas(atlas->getOrAssignUniqueKey(onFlushRP), onFlushRP->contextUniqueID(), newOffset, atlas->refOrMakeCachedAtlasInfo()); this->recordInstance(atlas->textureProxy(), resources->nextPathInstanceIdx()); resources->appendDrawPathInstance().set(*cacheEntry, draw.fCachedMaskShift, draw.fColor); // Remember this atlas in case we encounter the path again during the same flush. cacheEntry->setCurrFlushAtlas(atlas); continue; } } // Render the raw path into a coverage count atlas. renderPathInAtlas() gives us two tight // bounding boxes: One in device space, as well as a second one rotated an additional 45 // degrees. The path vertex shader uses these two bounding boxes to generate an octagon that // circumscribes the path. SkASSERT(!draw.fCachedAtlasProxy); SkRect devBounds, devBounds45; SkIRect devIBounds; SkIVector devToAtlasOffset; if (auto atlas = resources->renderPathInAtlas(draw.fMaskDevIBounds, draw.fMatrix, path, &devBounds, &devBounds45, &devIBounds, &devToAtlasOffset)) { this->recordInstance(atlas->textureProxy(), resources->nextPathInstanceIdx()); resources->appendDrawPathInstance().set(devBounds, devBounds45, devToAtlasOffset, draw.fColor, doEvenOddFill); // If we have a spot in the path cache, try to make a note of where this mask is so we // can reuse it in the future. if (auto cacheEntry = draw.fCacheEntry.get()) { SkASSERT(!cacheEntry->hasCachedAtlas()); if (Visibility::kComplete != draw.fMaskVisibility || cacheEntry->hitCount() <= 1) { // Don't cache a path mask unless it's completely visible with a hit count > 1. // // NOTE: mostly-visible paths with a hit count > 1 should have been promoted to // fully visible during accountForOwnPaths(). continue; } if (resources->nextAtlasToStash() != atlas) { // This mask does not belong to the atlas that will be stashed for next flush. continue; } const GrUniqueKey& atlasKey = resources->nextAtlasToStash()->getOrAssignUniqueKey(onFlushRP); cacheEntry->initAsStashedAtlas(atlasKey, onFlushRP->contextUniqueID(), devToAtlasOffset, devBounds, devBounds45, devIBounds, draw.fCachedMaskShift); // Remember this atlas in case we encounter the path again during the same flush. cacheEntry->setCurrFlushAtlas(atlas); } continue; } } if (!fInstanceRanges.empty()) { fInstanceRanges.back().fEndInstanceIdx = resources->nextPathInstanceIdx(); } } inline void GrCCDrawPathsOp::recordInstance(const GrTextureProxy* atlasProxy, int instanceIdx) { if (fInstanceRanges.empty()) { fInstanceRanges.push_back({atlasProxy, instanceIdx}); return; } if (fInstanceRanges.back().fAtlasProxy != atlasProxy) { fInstanceRanges.back().fEndInstanceIdx = instanceIdx; fInstanceRanges.push_back({atlasProxy, instanceIdx}); return; } } void GrCCDrawPathsOp::onExecute(GrOpFlushState* flushState) { SkASSERT(fOwningPerOpListPaths); const GrCCPerFlushResources* resources = fOwningPerOpListPaths->fFlushResources.get(); if (!resources) { return; // Setup failed. } GrPipeline::InitArgs initArgs; initArgs.fProxy = flushState->drawOpArgs().fProxy; initArgs.fCaps = &flushState->caps(); initArgs.fResourceProvider = flushState->resourceProvider(); initArgs.fDstProxy = flushState->drawOpArgs().fDstProxy; auto clip = flushState->detachAppliedClip(); GrPipeline::FixedDynamicState fixedDynamicState(clip.scissorState().rect()); GrPipeline pipeline(initArgs, std::move(fProcessors), std::move(clip)); int baseInstance = fBaseInstance; SkASSERT(baseInstance >= 0); // Make sure setupResources() has been called. for (const InstanceRange& range : fInstanceRanges) { SkASSERT(range.fEndInstanceIdx > baseInstance); GrCCPathProcessor pathProc(flushState->resourceProvider(), sk_ref_sp(range.fAtlasProxy), fViewMatrixIfUsingLocalCoords); pathProc.drawPaths(flushState, pipeline, &fixedDynamicState, *resources, baseInstance, range.fEndInstanceIdx, this->bounds()); baseInstance = range.fEndInstanceIdx; } }