blob: 2365f1e5a456ad3b3a72d16ec88e8ddb8b268e34 [file] [log] [blame]
egdaniel3658f382014-09-15 07:01:59 -07001/*
2 * Copyright 2014 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
8#include "GrOptDrawState.h"
9
10#include "GrDrawState.h"
egdanielc0648242014-09-22 13:17:02 -070011#include "GrDrawTargetCaps.h"
joshualitt4973d9d2014-11-08 09:24:25 -080012#include "GrGpu.h"
egdanielb6cbc382014-11-13 11:00:34 -080013#include "GrProcOptInfo.h"
egdaniel95131432014-12-09 11:15:43 -080014#include "GrXferProcessor.h"
egdaniel3658f382014-09-15 07:01:59 -070015
egdaniel170f90b2014-09-16 12:54:40 -070016GrOptDrawState::GrOptDrawState(const GrDrawState& drawState,
joshualitt56995b52014-12-11 15:44:02 -080017 const GrGeometryProcessor* gp,
18 const GrPathProcessor* pathProc,
joshualittdafa4d02014-12-04 08:59:10 -080019 const GrDrawTargetCaps& caps,
bsalomon3e791242014-12-17 13:43:13 -080020 const GrScissorState& scissorState,
joshualitt79f8fae2014-10-28 17:59:26 -070021 const GrDeviceCoordTexture* dstCopy,
joshualittdafa4d02014-12-04 08:59:10 -080022 GrGpu::DrawType drawType)
23 : fFinalized(false) {
24 fDrawType = drawType;
egdaniel95131432014-12-09 11:15:43 -080025
joshualitt56995b52014-12-11 15:44:02 -080026 // Copy GeometryProcesssor from DS or ODS
27 if (gp) {
28 SkASSERT(!pathProc);
bsalomon3e791242014-12-17 13:43:13 -080029 SkASSERT(!GrGpu::IsPathRenderingDrawType(drawType));
joshualitt56995b52014-12-11 15:44:02 -080030 fGeometryProcessor.reset(gp);
31 fPrimitiveProcessor.reset(gp);
32 } else {
bsalomon3e791242014-12-17 13:43:13 -080033 SkASSERT(!gp && pathProc && GrGpu::IsPathRenderingDrawType(drawType));
joshualitt56995b52014-12-11 15:44:02 -080034 fPrimitiveProcessor.reset(pathProc);
35 }
36
37
38 const GrProcOptInfo& colorPOI = drawState.colorProcInfo(fPrimitiveProcessor);
39 const GrProcOptInfo& coveragePOI = drawState.coverageProcInfo(fPrimitiveProcessor);
egdaniel95131432014-12-09 11:15:43 -080040
41 // Create XferProcessor from DS's XPFactory
42 SkAutoTUnref<GrXferProcessor> xferProcessor(
43 drawState.getXPFactory()->createXferProcessor(colorPOI, coveragePOI));
44
joshualitt9b989322014-12-15 14:16:27 -080045 GrColor overrideColor = GrColor_ILLEGAL;
46 if (colorPOI.firstEffectiveStageIndex() != 0) {
47 overrideColor = colorPOI.inputColorToEffectiveStage();
48 }
49
egdaniel95131432014-12-09 11:15:43 -080050 GrXferProcessor::OptFlags optFlags;
51 if (xferProcessor) {
52 fXferProcessor.reset(xferProcessor.get());
53
54 optFlags = xferProcessor->getOptimizations(colorPOI,
55 coveragePOI,
egdaniel95131432014-12-09 11:15:43 -080056 drawState.getStencil().doesWrite(),
joshualitt9b989322014-12-15 14:16:27 -080057 &overrideColor,
egdanielc2304142014-12-11 13:15:13 -080058 caps);
egdaniel95131432014-12-09 11:15:43 -080059 }
bsalomonb03c4a32014-11-20 09:56:11 -080060
61 // When path rendering the stencil settings are not always set on the draw state
62 // so we must check the draw type. In cases where we will skip drawing we simply return a
63 // null GrOptDrawState.
bsalomon3e791242014-12-17 13:43:13 -080064 if (!xferProcessor || (GrXferProcessor::kSkipDraw_OptFlag & optFlags)) {
bsalomonb03c4a32014-11-20 09:56:11 -080065 // Set the fields that don't default init and return. The lack of a render target will
66 // indicate that this can be skipped.
67 fFlags = 0;
bsalomonb03c4a32014-11-20 09:56:11 -080068 fDrawFace = GrDrawState::kInvalid_DrawFace;
bsalomonb03c4a32014-11-20 09:56:11 -080069 return;
70 }
71
72 fRenderTarget.reset(drawState.fRenderTarget.get());
73 SkASSERT(fRenderTarget);
joshualitt54e0c122014-11-19 09:38:51 -080074 fScissorState = scissorState;
egdaniel3658f382014-09-15 07:01:59 -070075 fStencilSettings = drawState.getStencil();
bsalomon04ddf892014-11-19 12:36:22 -080076 fDrawFace = drawState.getDrawFace();
joshualitt9176e2c2014-11-20 07:28:52 -080077 // TODO move this out of optDrawState
78 if (dstCopy) {
79 fDstCopy = *dstCopy;
80 }
81
bsalomon04ddf892014-11-19 12:36:22 -080082 fFlags = 0;
83 if (drawState.isHWAntialias()) {
84 fFlags |= kHWAA_Flag;
85 }
bsalomon04ddf892014-11-19 12:36:22 -080086 if (drawState.isDither()) {
87 fFlags |= kDither_Flag;
88 }
89
joshualitt9b989322014-12-15 14:16:27 -080090 // TODO move local coords completely into GP
joshualitt56995b52014-12-11 15:44:02 -080091 bool hasLocalCoords = gp && gp->hasLocalCoords();
egdaniel9cf45bf2014-10-08 06:49:10 -070092
egdaniel912b3d22014-11-17 07:45:53 -080093 int firstColorStageIdx = colorPOI.firstEffectiveStageIndex();
egdaniel912b3d22014-11-17 07:45:53 -080094
95 // TODO: Once we can handle single or four channel input into coverage stages then we can use
96 // drawState's coverageProcInfo (like color above) to set this initial information.
97 int firstCoverageStageIdx = 0;
egdaniel912b3d22014-11-17 07:45:53 -080098
egdaniel95131432014-12-09 11:15:43 -080099 GrXferProcessor::BlendInfo blendInfo;
100 fXferProcessor->getBlendInfo(&blendInfo);
egdaniel95131432014-12-09 11:15:43 -0800101
102 this->adjustProgramFromOptimizations(drawState, optFlags, colorPOI, coveragePOI,
103 &firstColorStageIdx, &firstCoverageStageIdx);
104
105 fDescInfo.fRequiresLocalCoordAttrib = hasLocalCoords;
egdaniel9cf45bf2014-10-08 06:49:10 -0700106
joshualitt290c09b2014-12-19 13:45:20 -0800107 bool usesLocalCoords = false;
108
bsalomonae59b772014-11-19 08:23:49 -0800109 // Copy Stages from DS to ODS
bsalomonae59b772014-11-19 08:23:49 -0800110 for (int i = firstColorStageIdx; i < drawState.numColorStages(); ++i) {
111 SkNEW_APPEND_TO_TARRAY(&fFragmentStages,
112 GrPendingFragmentStage,
joshualitt40d4bd82014-12-29 09:04:40 -0800113 (drawState.fColorStages[i]));
joshualitt290c09b2014-12-19 13:45:20 -0800114 usesLocalCoords = usesLocalCoords ||
joshualitt40d4bd82014-12-29 09:04:40 -0800115 drawState.fColorStages[i].processor()->usesLocalCoords();
bsalomonae59b772014-11-19 08:23:49 -0800116 }
egdaniel95131432014-12-09 11:15:43 -0800117
egdanield9aa2182014-10-09 13:47:05 -0700118 fNumColorStages = fFragmentStages.count();
bsalomonae59b772014-11-19 08:23:49 -0800119 for (int i = firstCoverageStageIdx; i < drawState.numCoverageStages(); ++i) {
120 SkNEW_APPEND_TO_TARRAY(&fFragmentStages,
121 GrPendingFragmentStage,
joshualitt40d4bd82014-12-29 09:04:40 -0800122 (drawState.fCoverageStages[i]));
joshualitt290c09b2014-12-19 13:45:20 -0800123 usesLocalCoords = usesLocalCoords ||
joshualitt40d4bd82014-12-29 09:04:40 -0800124 drawState.fCoverageStages[i].processor()->usesLocalCoords();
egdaniel9cf45bf2014-10-08 06:49:10 -0700125 }
joshualitt79f8fae2014-10-28 17:59:26 -0700126
joshualitt87f48d92014-12-04 10:41:40 -0800127 // let the GP init the batch tracker
joshualitt9b989322014-12-15 14:16:27 -0800128 GrGeometryProcessor::InitBT init;
129 init.fColorIgnored = SkToBool(optFlags & GrXferProcessor::kIgnoreColor_OptFlag);
130 init.fOverrideColor = init.fColorIgnored ? GrColor_ILLEGAL : overrideColor;
131 init.fCoverageIgnored = SkToBool(optFlags & GrXferProcessor::kIgnoreCoverage_OptFlag);
joshualitt290c09b2014-12-19 13:45:20 -0800132 init.fUsesLocalCoords = usesLocalCoords;
joshualitt9b989322014-12-15 14:16:27 -0800133 fPrimitiveProcessor->initBatchTracker(&fBatchTracker, init);
egdanielc0648242014-09-22 13:17:02 -0700134}
135
egdaniel95131432014-12-09 11:15:43 -0800136void GrOptDrawState::adjustProgramFromOptimizations(const GrDrawState& ds,
137 GrXferProcessor::OptFlags flags,
138 const GrProcOptInfo& colorPOI,
139 const GrProcOptInfo& coveragePOI,
140 int* firstColorStageIdx,
141 int* firstCoverageStageIdx) {
joshualittdafa4d02014-12-04 08:59:10 -0800142 fDescInfo.fReadsDst = false;
143 fDescInfo.fReadsFragPosition = false;
egdaniela7dc0a82014-09-17 08:25:05 -0700144
joshualitt9b989322014-12-15 14:16:27 -0800145 if ((flags & GrXferProcessor::kIgnoreColor_OptFlag) ||
146 (flags & GrXferProcessor::kOverrideColor_OptFlag)) {
egdaniel95131432014-12-09 11:15:43 -0800147 *firstColorStageIdx = ds.numColorStages();
egdaniel95131432014-12-09 11:15:43 -0800148 } else {
149 fDescInfo.fReadsDst = colorPOI.readsDst();
150 fDescInfo.fReadsFragPosition = colorPOI.readsFragPosition();
egdaniela7dc0a82014-09-17 08:25:05 -0700151 }
egdaniel95131432014-12-09 11:15:43 -0800152
joshualitt9b989322014-12-15 14:16:27 -0800153 if (flags & GrXferProcessor::kIgnoreCoverage_OptFlag) {
egdaniel95131432014-12-09 11:15:43 -0800154 *firstCoverageStageIdx = ds.numCoverageStages();
egdaniel95131432014-12-09 11:15:43 -0800155 } else {
156 if (coveragePOI.readsDst()) {
157 fDescInfo.fReadsDst = true;
158 }
159 if (coveragePOI.readsFragPosition()) {
160 fDescInfo.fReadsFragPosition = true;
161 }
egdaniela7dc0a82014-09-17 08:25:05 -0700162 }
163}
164
joshualittdafa4d02014-12-04 08:59:10 -0800165void GrOptDrawState::finalize(GrGpu* gpu) {
166 gpu->buildProgramDesc(*this, fDescInfo, fDrawType, &fDesc);
167 fFinalized = true;
168}
169
egdaniel89af44a2014-09-26 06:15:04 -0700170////////////////////////////////////////////////////////////////////////////////
171
joshualitt9b989322014-12-15 14:16:27 -0800172bool GrOptDrawState::combineIfPossible(const GrOptDrawState& that) {
joshualittdafa4d02014-12-04 08:59:10 -0800173 if (fDescInfo != that.fDescInfo) {
joshualittf78c60c2014-12-04 06:01:45 -0800174 return false;
175 }
joshualittdafa4d02014-12-04 08:59:10 -0800176
egdaniel89af44a2014-09-26 06:15:04 -0700177 if (this->getRenderTarget() != that.getRenderTarget() ||
joshualittdafa4d02014-12-04 08:59:10 -0800178 this->fFragmentStages.count() != that.fFragmentStages.count() ||
179 this->fNumColorStages != that.fNumColorStages ||
joshualitt54e0c122014-11-19 09:38:51 -0800180 this->fScissorState != that.fScissorState ||
joshualittdafa4d02014-12-04 08:59:10 -0800181 this->fDrawType != that.fDrawType ||
bsalomon04ddf892014-11-19 12:36:22 -0800182 this->fFlags != that.fFlags ||
egdaniel89af44a2014-09-26 06:15:04 -0700183 this->fStencilSettings != that.fStencilSettings ||
joshualitt9176e2c2014-11-20 07:28:52 -0800184 this->fDrawFace != that.fDrawFace ||
185 this->fDstCopy.texture() != that.fDstCopy.texture()) {
egdaniel89af44a2014-09-26 06:15:04 -0700186 return false;
187 }
188
joshualitt9b989322014-12-15 14:16:27 -0800189 if (!this->getPrimitiveProcessor()->canMakeEqual(fBatchTracker,
190 *that.getPrimitiveProcessor(),
191 that.getBatchTracker())) {
egdaniel89af44a2014-09-26 06:15:04 -0700192 return false;
193 }
194
egdanielc2304142014-12-11 13:15:13 -0800195 if (!this->getXferProcessor()->isEqual(*that.getXferProcessor())) {
196 return false;
197 }
198
bsalomonae59b772014-11-19 08:23:49 -0800199 // The program desc comparison should have already assured that the stage counts match.
200 SkASSERT(this->numFragmentStages() == that.numFragmentStages());
egdanield9aa2182014-10-09 13:47:05 -0700201 for (int i = 0; i < this->numFragmentStages(); i++) {
bsalomonae59b772014-11-19 08:23:49 -0800202
203 if (this->getFragmentStage(i) != that.getFragmentStage(i)) {
egdaniel89af44a2014-09-26 06:15:04 -0700204 return false;
205 }
206 }
joshualitt9b989322014-12-15 14:16:27 -0800207
208 // Now update the GrPrimitiveProcessor's batch tracker
209 fPrimitiveProcessor->makeEqual(&fBatchTracker, that.getBatchTracker());
egdaniel89af44a2014-09-26 06:15:04 -0700210 return true;
211}
212