blob: 15eeab5addeaf93597f37e0d7d06405e39714ce1 [file] [log] [blame]
bsalomon75398562015-08-17 12:55:38 -07001/*
2 * Copyright 2015 Google Inc.
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
7
Mike Kleinc0bd9f92019-04-23 12:05:21 -05008#include "src/gpu/GrOpFlushState.h"
bsalomon75398562015-08-17 12:55:38 -07009
Mike Kleinc0bd9f92019-04-23 12:05:21 -050010#include "include/gpu/GrTexture.h"
Brian Salomon1047a492019-07-02 12:25:21 -040011#include "src/core/SkConvertPixels.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050012#include "src/gpu/GrContextPriv.h"
13#include "src/gpu/GrDrawOpAtlas.h"
14#include "src/gpu/GrGpu.h"
Brian Salomonf2ebdd92019-09-30 12:15:30 -040015#include "src/gpu/GrImageInfo.h"
Robert Phillips901aff02019-10-08 12:32:56 -040016#include "src/gpu/GrProgramInfo.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050017#include "src/gpu/GrResourceProvider.h"
bsalomon75398562015-08-17 12:55:38 -070018
Brian Salomon7dc6e752017-11-02 11:34:51 -040019//////////////////////////////////////////////////////////////////////////////
20
Brian Salomon58f153c2018-10-18 21:51:15 -040021GrOpFlushState::GrOpFlushState(GrGpu* gpu, GrResourceProvider* resourceProvider,
Robert Phillipse5f73282019-06-18 17:15:04 -040022 GrTokenTracker* tokenTracker,
Brian Salomon601ac802019-02-07 13:37:16 -050023 sk_sp<GrBufferAllocPool::CpuBufferCache> cpuBufferCache)
24 : fVertexPool(gpu, cpuBufferCache)
25 , fIndexPool(gpu, std::move(cpuBufferCache))
Robert Phillips40a29d72018-01-18 12:59:22 -050026 , fGpu(gpu)
27 , fResourceProvider(resourceProvider)
Brian Salomonbeb7f522019-08-30 16:19:42 -040028 , fTokenTracker(tokenTracker) {}
bsalomon75398562015-08-17 12:55:38 -070029
Robert Phillips646e4292017-06-13 12:44:56 -040030const GrCaps& GrOpFlushState::caps() const {
31 return *fGpu->caps();
32}
33
Chris Dalton07cdcfc92019-02-26 11:13:22 -070034void GrOpFlushState::executeDrawsAndUploadsForMeshDrawOp(
35 const GrOp* op, const SkRect& chainBounds, GrProcessorSet&& processorSet,
Chris Daltonbaa1b352019-04-03 12:03:00 -060036 GrPipeline::InputFlags pipelineFlags, const GrUserStencilSettings* stencilSettings) {
Greg Daniel2d41d0d2019-08-26 11:08:51 -040037 SkASSERT(this->opsRenderPass());
Chris Dalton07cdcfc92019-02-26 11:13:22 -070038
39 GrPipeline::InitArgs pipelineArgs;
Chris Daltonbaa1b352019-04-03 12:03:00 -060040 pipelineArgs.fInputFlags = pipelineFlags;
Greg Daniel524e28b2019-11-01 11:48:53 -040041 pipelineArgs.fDstProxyView = this->dstProxyView();
Chris Dalton07cdcfc92019-02-26 11:13:22 -070042 pipelineArgs.fCaps = &this->caps();
Chris Dalton07cdcfc92019-02-26 11:13:22 -070043 pipelineArgs.fUserStencil = stencilSettings;
Robert Phillips405413f2019-10-04 10:39:28 -040044 pipelineArgs.fOutputSwizzle = this->drawOpArgs().outputSwizzle();
Greg Daniel0b002e22019-08-12 13:25:36 -040045 GrPipeline* pipeline = this->allocator()->make<GrPipeline>(pipelineArgs,
46 std::move(processorSet),
47 this->detachAppliedClip());
Chris Dalton07cdcfc92019-02-26 11:13:22 -070048
Brian Osmane8012102018-11-29 14:05:07 -050049 while (fCurrDraw != fDraws.end() && fCurrDraw->fOp == op) {
Robert Phillips40a29d72018-01-18 12:59:22 -050050 GrDeferredUploadToken drawToken = fTokenTracker->nextTokenToFlush();
Brian Salomon7dc6e752017-11-02 11:34:51 -040051 while (fCurrUpload != fInlineUploads.end() &&
52 fCurrUpload->fUploadBeforeToken == drawToken) {
Greg Daniel2d41d0d2019-08-26 11:08:51 -040053 this->opsRenderPass()->inlineUpload(this, fCurrUpload->fUpload);
Brian Salomon7dc6e752017-11-02 11:34:51 -040054 ++fCurrUpload;
55 }
Robert Phillips901aff02019-10-08 12:32:56 -040056
57 GrProgramInfo programInfo(this->proxy()->numSamples(),
58 this->proxy()->origin(),
59 *pipeline,
60 *fCurrDraw->fGeometryProcessor,
61 fCurrDraw->fFixedDynamicState,
Robert Phillips2d8a95e2019-10-10 12:50:22 -040062 fCurrDraw->fDynamicStateArrays,
63 fCurrDraw->fMeshCnt);
Robert Phillips901aff02019-10-08 12:32:56 -040064
65 this->opsRenderPass()->draw(programInfo, fCurrDraw->fMeshes,
66 fCurrDraw->fMeshCnt, chainBounds);
Robert Phillips40a29d72018-01-18 12:59:22 -050067 fTokenTracker->flushToken();
Brian Salomon7dc6e752017-11-02 11:34:51 -040068 ++fCurrDraw;
69 }
70}
71
72void GrOpFlushState::preExecuteDraws() {
73 fVertexPool.unmap();
74 fIndexPool.unmap();
Robert Phillips40a29d72018-01-18 12:59:22 -050075 for (auto& upload : fASAPUploads) {
Brian Salomon7dc6e752017-11-02 11:34:51 -040076 this->doUpload(upload);
77 }
78 // Setup execution iterators.
79 fCurrDraw = fDraws.begin();
80 fCurrUpload = fInlineUploads.begin();
Brian Salomon7dc6e752017-11-02 11:34:51 -040081}
82
83void GrOpFlushState::reset() {
84 SkASSERT(fCurrDraw == fDraws.end());
85 SkASSERT(fCurrUpload == fInlineUploads.end());
86 fVertexPool.reset();
87 fIndexPool.reset();
88 fArena.reset();
Robert Phillips40a29d72018-01-18 12:59:22 -050089 fASAPUploads.reset();
Brian Salomon7dc6e752017-11-02 11:34:51 -040090 fInlineUploads.reset();
91 fDraws.reset();
Brian Salomon7dc6e752017-11-02 11:34:51 -040092 fBaseDrawToken = GrDeferredUploadToken::AlreadyFlushedToken();
93}
94
Greg Danielb20d7e52019-09-03 13:54:39 -040095void GrOpFlushState::doUpload(GrDeferredTextureUploadFn& upload,
96 bool shouldPrepareSurfaceForSampling) {
97 GrDeferredTextureUploadWritePixelsFn wp = [this, shouldPrepareSurfaceForSampling](
98 GrTextureProxy* dstProxy, int left, int top, int width, int height,
99 GrColorType colorType, const void* buffer, size_t rowBytes) {
Brian Salomonfd98c2c2018-07-31 17:25:29 -0400100 GrSurface* dstSurface = dstProxy->peekSurface();
Brian Salomon42be09d2019-07-26 12:12:26 -0400101 if (!fGpu->caps()->surfaceSupportsWritePixels(dstSurface)) {
102 return false;
103 }
Brian Salomon01915c02019-08-02 09:57:21 -0400104 GrCaps::SupportedWrite supportedWrite = fGpu->caps()->supportedWritePixelsColorType(
105 colorType, dstSurface->backendFormat(), colorType);
Brian Salomon77a684f2019-08-01 14:38:04 -0400106 size_t tightRB = width * GrColorTypeBytesPerPixel(supportedWrite.fColorType);
107 SkASSERT(rowBytes >= tightRB);
Brian Salomon1047a492019-07-02 12:25:21 -0400108 std::unique_ptr<char[]> tmpPixels;
Brian Salomonf77c1462019-08-01 15:19:29 -0400109 if (supportedWrite.fColorType != colorType ||
Brian Salomon77a684f2019-08-01 14:38:04 -0400110 (!fGpu->caps()->writePixelsRowBytesSupport() && rowBytes != tightRB)) {
111 tmpPixels.reset(new char[height * tightRB]);
112 // Use kUnpremul to ensure no alpha type conversions or clamping occur.
113 static constexpr auto kAT = kUnpremul_SkAlphaType;
Brian Salomonf2ebdd92019-09-30 12:15:30 -0400114 GrImageInfo srcInfo(colorType, kAT, nullptr, width, height);
115 GrImageInfo tmpInfo(supportedWrite.fColorType, kAT, nullptr, width,
Brian Salomon77a684f2019-08-01 14:38:04 -0400116 height);
117 if (!GrConvertPixels(tmpInfo, tmpPixels.get(), tightRB, srcInfo, buffer, rowBytes)) {
118 return false;
119 }
Brian Salomon1047a492019-07-02 12:25:21 -0400120 rowBytes = tightRB;
Brian Salomon77a684f2019-08-01 14:38:04 -0400121 buffer = tmpPixels.get();
Brian Salomon1047a492019-07-02 12:25:21 -0400122 }
Brian Salomonf77c1462019-08-01 15:19:29 -0400123 return this->fGpu->writePixels(dstSurface, left, top, width, height, colorType,
Greg Danielb20d7e52019-09-03 13:54:39 -0400124 supportedWrite.fColorType, buffer, rowBytes,
125 shouldPrepareSurfaceForSampling);
Brian Salomon9bada542017-06-12 12:09:30 -0400126 };
127 upload(wp);
128}
Brian Salomon29b60c92017-10-31 14:42:10 -0400129
130GrDeferredUploadToken GrOpFlushState::addInlineUpload(GrDeferredTextureUploadFn&& upload) {
Robert Phillips40a29d72018-01-18 12:59:22 -0500131 return fInlineUploads.append(&fArena, std::move(upload), fTokenTracker->nextDrawToken())
Brian Salomon7dc6e752017-11-02 11:34:51 -0400132 .fUploadBeforeToken;
Brian Salomon29b60c92017-10-31 14:42:10 -0400133}
134
135GrDeferredUploadToken GrOpFlushState::addASAPUpload(GrDeferredTextureUploadFn&& upload) {
Robert Phillips40a29d72018-01-18 12:59:22 -0500136 fASAPUploads.append(&fArena, std::move(upload));
137 return fTokenTracker->nextTokenToFlush();
Brian Salomon29b60c92017-10-31 14:42:10 -0400138}
139
Chris Dalton07cdcfc92019-02-26 11:13:22 -0700140void GrOpFlushState::recordDraw(
141 sk_sp<const GrGeometryProcessor> gp, const GrMesh meshes[], int meshCnt,
142 const GrPipeline::FixedDynamicState* fixedDynamicState,
143 const GrPipeline::DynamicStateArrays* dynamicStateArrays) {
Brian Salomon29b60c92017-10-31 14:42:10 -0400144 SkASSERT(fOpArgs);
Robert Phillips405413f2019-10-04 10:39:28 -0400145 SkDEBUGCODE(fOpArgs->validate());
Brian Salomon7dc6e752017-11-02 11:34:51 -0400146 bool firstDraw = fDraws.begin() == fDraws.end();
Brian Salomon7dc6e752017-11-02 11:34:51 -0400147 auto& draw = fDraws.append(&fArena);
Robert Phillips40a29d72018-01-18 12:59:22 -0500148 GrDeferredUploadToken token = fTokenTracker->issueDrawToken();
Brian Salomonf7232642018-09-19 08:58:08 -0400149 if (fixedDynamicState && fixedDynamicState->fPrimitiveProcessorTextures) {
150 for (int i = 0; i < gp->numTextureSamplers(); ++i) {
Robert Phillips3d4cac52019-06-11 08:08:08 -0400151 fixedDynamicState->fPrimitiveProcessorTextures[i]->ref();
Brian Salomonf7232642018-09-19 08:58:08 -0400152 }
153 }
154 if (dynamicStateArrays && dynamicStateArrays->fPrimitiveProcessorTextures) {
155 int n = gp->numTextureSamplers() * meshCnt;
156 for (int i = 0; i < n; ++i) {
Robert Phillips3d4cac52019-06-11 08:08:08 -0400157 dynamicStateArrays->fPrimitiveProcessorTextures[i]->ref();
Brian Salomonf7232642018-09-19 08:58:08 -0400158 }
Brian Salomon7eae3e02018-08-07 14:02:38 +0000159 }
160 draw.fGeometryProcessor = std::move(gp);
Brian Salomon49348902018-06-26 09:12:38 -0400161 draw.fFixedDynamicState = fixedDynamicState;
Brian Salomonf7232642018-09-19 08:58:08 -0400162 draw.fDynamicStateArrays = dynamicStateArrays;
Brian Salomon7eae3e02018-08-07 14:02:38 +0000163 draw.fMeshes = meshes;
164 draw.fMeshCnt = meshCnt;
Robert Phillips405413f2019-10-04 10:39:28 -0400165 draw.fOp = fOpArgs->op();
Brian Salomon7dc6e752017-11-02 11:34:51 -0400166 if (firstDraw) {
167 fBaseDrawToken = token;
Brian Salomon29b60c92017-10-31 14:42:10 -0400168 }
169}
170
Brian Salomon12d22642019-01-29 14:38:50 -0500171void* GrOpFlushState::makeVertexSpace(size_t vertexSize, int vertexCount,
172 sk_sp<const GrBuffer>* buffer, int* startVertex) {
Brian Salomon29b60c92017-10-31 14:42:10 -0400173 return fVertexPool.makeSpace(vertexSize, vertexCount, buffer, startVertex);
174}
175
Brian Salomon12d22642019-01-29 14:38:50 -0500176uint16_t* GrOpFlushState::makeIndexSpace(int indexCount, sk_sp<const GrBuffer>* buffer,
177 int* startIndex) {
Brian Salomon29b60c92017-10-31 14:42:10 -0400178 return reinterpret_cast<uint16_t*>(fIndexPool.makeSpace(indexCount, buffer, startIndex));
179}
180
181void* GrOpFlushState::makeVertexSpaceAtLeast(size_t vertexSize, int minVertexCount,
Brian Salomon12d22642019-01-29 14:38:50 -0500182 int fallbackVertexCount, sk_sp<const GrBuffer>* buffer,
Brian Salomon29b60c92017-10-31 14:42:10 -0400183 int* startVertex, int* actualVertexCount) {
184 return fVertexPool.makeSpaceAtLeast(vertexSize, minVertexCount, fallbackVertexCount, buffer,
185 startVertex, actualVertexCount);
186}
187
188uint16_t* GrOpFlushState::makeIndexSpaceAtLeast(int minIndexCount, int fallbackIndexCount,
Brian Salomon12d22642019-01-29 14:38:50 -0500189 sk_sp<const GrBuffer>* buffer, int* startIndex,
Brian Salomon29b60c92017-10-31 14:42:10 -0400190 int* actualIndexCount) {
191 return reinterpret_cast<uint16_t*>(fIndexPool.makeSpaceAtLeast(
192 minIndexCount, fallbackIndexCount, buffer, startIndex, actualIndexCount));
193}
194
195void GrOpFlushState::putBackIndices(int indexCount) {
196 fIndexPool.putBack(indexCount * sizeof(uint16_t));
197}
198
199void GrOpFlushState::putBackVertices(int vertices, size_t vertexStride) {
200 fVertexPool.putBack(vertices * vertexStride);
201}
202
203GrAppliedClip GrOpFlushState::detachAppliedClip() {
Robert Phillips405413f2019-10-04 10:39:28 -0400204 return fOpArgs->appliedClip() ? std::move(*fOpArgs->appliedClip()) : GrAppliedClip();
Brian Salomon29b60c92017-10-31 14:42:10 -0400205}
Robert Phillipsc4039ea2018-03-01 11:36:45 -0500206
Herb Derby081e6f32019-01-16 13:46:02 -0500207GrStrikeCache* GrOpFlushState::glyphCache() const {
Herb Derbya00da612019-03-04 17:10:01 -0500208 return fGpu->getContext()->priv().getGrStrikeCache();
Robert Phillipsc4039ea2018-03-01 11:36:45 -0500209}
210
Robert Phillips5a66efb2018-03-07 15:13:18 -0500211GrAtlasManager* GrOpFlushState::atlasManager() const {
Robert Phillips9da87e02019-02-04 13:26:26 -0500212 return fGpu->getContext()->priv().getAtlasManager();
Robert Phillipsc4039ea2018-03-01 11:36:45 -0500213}
Brian Salomonf7232642018-09-19 08:58:08 -0400214
215//////////////////////////////////////////////////////////////////////////////
216
217GrOpFlushState::Draw::~Draw() {
218 if (fFixedDynamicState && fFixedDynamicState->fPrimitiveProcessorTextures) {
219 for (int i = 0; i < fGeometryProcessor->numTextureSamplers(); ++i) {
Robert Phillips3d4cac52019-06-11 08:08:08 -0400220 fFixedDynamicState->fPrimitiveProcessorTextures[i]->unref();
Brian Salomonf7232642018-09-19 08:58:08 -0400221 }
222 }
223 if (fDynamicStateArrays && fDynamicStateArrays->fPrimitiveProcessorTextures) {
224 int n = fGeometryProcessor->numTextureSamplers() * fMeshCnt;
225 const auto* textures = fDynamicStateArrays->fPrimitiveProcessorTextures;
226 for (int i = 0; i < n; ++i) {
Robert Phillips3d4cac52019-06-11 08:08:08 -0400227 textures[i]->unref();
Brian Salomonf7232642018-09-19 08:58:08 -0400228 }
229 }
230}