blob: 95d5984a3fb505ee35d6deb2c0cebac6de926b59 [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.isColorWriteDisabled(),
57 drawState.getStencil().doesWrite(),
joshualitt9b989322014-12-15 14:16:27 -080058 &overrideColor,
egdanielc2304142014-12-11 13:15:13 -080059 caps);
egdaniel95131432014-12-09 11:15:43 -080060 }
bsalomonb03c4a32014-11-20 09:56:11 -080061
62 // When path rendering the stencil settings are not always set on the draw state
63 // so we must check the draw type. In cases where we will skip drawing we simply return a
64 // null GrOptDrawState.
bsalomon3e791242014-12-17 13:43:13 -080065 if (!xferProcessor || (GrXferProcessor::kSkipDraw_OptFlag & optFlags)) {
bsalomonb03c4a32014-11-20 09:56:11 -080066 // Set the fields that don't default init and return. The lack of a render target will
67 // indicate that this can be skipped.
68 fFlags = 0;
bsalomonb03c4a32014-11-20 09:56:11 -080069 fDrawFace = GrDrawState::kInvalid_DrawFace;
bsalomonb03c4a32014-11-20 09:56:11 -080070 fViewMatrix.reset();
71 return;
72 }
73
74 fRenderTarget.reset(drawState.fRenderTarget.get());
75 SkASSERT(fRenderTarget);
joshualitt54e0c122014-11-19 09:38:51 -080076 fScissorState = scissorState;
egdaniel3658f382014-09-15 07:01:59 -070077 fViewMatrix = drawState.getViewMatrix();
egdaniel3658f382014-09-15 07:01:59 -070078 fStencilSettings = drawState.getStencil();
bsalomon04ddf892014-11-19 12:36:22 -080079 fDrawFace = drawState.getDrawFace();
joshualitt9176e2c2014-11-20 07:28:52 -080080 // TODO move this out of optDrawState
81 if (dstCopy) {
82 fDstCopy = *dstCopy;
83 }
84
bsalomon04ddf892014-11-19 12:36:22 -080085 fFlags = 0;
86 if (drawState.isHWAntialias()) {
87 fFlags |= kHWAA_Flag;
88 }
89 if (drawState.isColorWriteDisabled()) {
90 fFlags |= kDisableColorWrite_Flag;
91 }
92 if (drawState.isDither()) {
93 fFlags |= kDither_Flag;
94 }
95
joshualitt9b989322014-12-15 14:16:27 -080096 // TODO move local coords completely into GP
joshualitt56995b52014-12-11 15:44:02 -080097 bool hasLocalCoords = gp && gp->hasLocalCoords();
egdaniel9cf45bf2014-10-08 06:49:10 -070098
egdaniel912b3d22014-11-17 07:45:53 -080099 int firstColorStageIdx = colorPOI.firstEffectiveStageIndex();
egdaniel912b3d22014-11-17 07:45:53 -0800100
101 // TODO: Once we can handle single or four channel input into coverage stages then we can use
102 // drawState's coverageProcInfo (like color above) to set this initial information.
103 int firstCoverageStageIdx = 0;
egdaniel912b3d22014-11-17 07:45:53 -0800104
egdaniel95131432014-12-09 11:15:43 -0800105 GrXferProcessor::BlendInfo blendInfo;
106 fXferProcessor->getBlendInfo(&blendInfo);
egdaniel95131432014-12-09 11:15:43 -0800107
108 this->adjustProgramFromOptimizations(drawState, optFlags, colorPOI, coveragePOI,
109 &firstColorStageIdx, &firstCoverageStageIdx);
110
111 fDescInfo.fRequiresLocalCoordAttrib = hasLocalCoords;
egdaniel9cf45bf2014-10-08 06:49:10 -0700112
joshualitt290c09b2014-12-19 13:45:20 -0800113 bool usesLocalCoords = false;
114
bsalomonae59b772014-11-19 08:23:49 -0800115 // Copy Stages from DS to ODS
bsalomonae59b772014-11-19 08:23:49 -0800116 for (int i = firstColorStageIdx; i < drawState.numColorStages(); ++i) {
117 SkNEW_APPEND_TO_TARRAY(&fFragmentStages,
118 GrPendingFragmentStage,
joshualitt2dd1ae02014-12-03 06:24:10 -0800119 (drawState.fColorStages[i], hasLocalCoords));
joshualitt290c09b2014-12-19 13:45:20 -0800120 usesLocalCoords = usesLocalCoords ||
121 drawState.fColorStages[i].getProcessor()->usesLocalCoords();
bsalomonae59b772014-11-19 08:23:49 -0800122 }
egdaniel95131432014-12-09 11:15:43 -0800123
egdanield9aa2182014-10-09 13:47:05 -0700124 fNumColorStages = fFragmentStages.count();
bsalomonae59b772014-11-19 08:23:49 -0800125 for (int i = firstCoverageStageIdx; i < drawState.numCoverageStages(); ++i) {
126 SkNEW_APPEND_TO_TARRAY(&fFragmentStages,
127 GrPendingFragmentStage,
joshualitt2dd1ae02014-12-03 06:24:10 -0800128 (drawState.fCoverageStages[i], hasLocalCoords));
joshualitt290c09b2014-12-19 13:45:20 -0800129 usesLocalCoords = usesLocalCoords ||
130 drawState.fCoverageStages[i].getProcessor()->usesLocalCoords();
egdaniel9cf45bf2014-10-08 06:49:10 -0700131 }
joshualitt79f8fae2014-10-28 17:59:26 -0700132
joshualitt87f48d92014-12-04 10:41:40 -0800133 // let the GP init the batch tracker
joshualitt9b989322014-12-15 14:16:27 -0800134 GrGeometryProcessor::InitBT init;
135 init.fColorIgnored = SkToBool(optFlags & GrXferProcessor::kIgnoreColor_OptFlag);
136 init.fOverrideColor = init.fColorIgnored ? GrColor_ILLEGAL : overrideColor;
137 init.fCoverageIgnored = SkToBool(optFlags & GrXferProcessor::kIgnoreCoverage_OptFlag);
joshualitt290c09b2014-12-19 13:45:20 -0800138 init.fUsesLocalCoords = usesLocalCoords;
joshualitt9b989322014-12-15 14:16:27 -0800139 fPrimitiveProcessor->initBatchTracker(&fBatchTracker, init);
egdanielc0648242014-09-22 13:17:02 -0700140}
141
egdaniel95131432014-12-09 11:15:43 -0800142void GrOptDrawState::adjustProgramFromOptimizations(const GrDrawState& ds,
143 GrXferProcessor::OptFlags flags,
144 const GrProcOptInfo& colorPOI,
145 const GrProcOptInfo& coveragePOI,
146 int* firstColorStageIdx,
147 int* firstCoverageStageIdx) {
joshualittdafa4d02014-12-04 08:59:10 -0800148 fDescInfo.fReadsDst = false;
149 fDescInfo.fReadsFragPosition = false;
egdaniela7dc0a82014-09-17 08:25:05 -0700150
joshualitt9b989322014-12-15 14:16:27 -0800151 if ((flags & GrXferProcessor::kIgnoreColor_OptFlag) ||
152 (flags & GrXferProcessor::kOverrideColor_OptFlag)) {
egdaniel95131432014-12-09 11:15:43 -0800153 *firstColorStageIdx = ds.numColorStages();
egdaniel95131432014-12-09 11:15:43 -0800154 } else {
155 fDescInfo.fReadsDst = colorPOI.readsDst();
156 fDescInfo.fReadsFragPosition = colorPOI.readsFragPosition();
egdaniela7dc0a82014-09-17 08:25:05 -0700157 }
egdaniel95131432014-12-09 11:15:43 -0800158
joshualitt9b989322014-12-15 14:16:27 -0800159 if (flags & GrXferProcessor::kIgnoreCoverage_OptFlag) {
egdaniel95131432014-12-09 11:15:43 -0800160 *firstCoverageStageIdx = ds.numCoverageStages();
egdaniel95131432014-12-09 11:15:43 -0800161 } else {
162 if (coveragePOI.readsDst()) {
163 fDescInfo.fReadsDst = true;
164 }
165 if (coveragePOI.readsFragPosition()) {
166 fDescInfo.fReadsFragPosition = true;
167 }
egdaniela7dc0a82014-09-17 08:25:05 -0700168 }
169}
170
joshualittdafa4d02014-12-04 08:59:10 -0800171void GrOptDrawState::finalize(GrGpu* gpu) {
172 gpu->buildProgramDesc(*this, fDescInfo, fDrawType, &fDesc);
173 fFinalized = true;
174}
175
egdaniel89af44a2014-09-26 06:15:04 -0700176////////////////////////////////////////////////////////////////////////////////
177
joshualitt9b989322014-12-15 14:16:27 -0800178bool GrOptDrawState::combineIfPossible(const GrOptDrawState& that) {
joshualittdafa4d02014-12-04 08:59:10 -0800179 if (fDescInfo != that.fDescInfo) {
joshualittf78c60c2014-12-04 06:01:45 -0800180 return false;
181 }
joshualittdafa4d02014-12-04 08:59:10 -0800182
egdaniel89af44a2014-09-26 06:15:04 -0700183 if (this->getRenderTarget() != that.getRenderTarget() ||
joshualittdafa4d02014-12-04 08:59:10 -0800184 this->fFragmentStages.count() != that.fFragmentStages.count() ||
185 this->fNumColorStages != that.fNumColorStages ||
joshualitt54e0c122014-11-19 09:38:51 -0800186 this->fScissorState != that.fScissorState ||
egdaniel89af44a2014-09-26 06:15:04 -0700187 !this->fViewMatrix.cheapEqualTo(that.fViewMatrix) ||
joshualittdafa4d02014-12-04 08:59:10 -0800188 this->fDrawType != that.fDrawType ||
bsalomon04ddf892014-11-19 12:36:22 -0800189 this->fFlags != that.fFlags ||
egdaniel89af44a2014-09-26 06:15:04 -0700190 this->fStencilSettings != that.fStencilSettings ||
joshualitt9176e2c2014-11-20 07:28:52 -0800191 this->fDrawFace != that.fDrawFace ||
192 this->fDstCopy.texture() != that.fDstCopy.texture()) {
egdaniel89af44a2014-09-26 06:15:04 -0700193 return false;
194 }
195
joshualitt9b989322014-12-15 14:16:27 -0800196 if (!this->getPrimitiveProcessor()->canMakeEqual(fBatchTracker,
197 *that.getPrimitiveProcessor(),
198 that.getBatchTracker())) {
egdaniel89af44a2014-09-26 06:15:04 -0700199 return false;
200 }
201
egdanielc2304142014-12-11 13:15:13 -0800202 if (!this->getXferProcessor()->isEqual(*that.getXferProcessor())) {
203 return false;
204 }
205
bsalomonae59b772014-11-19 08:23:49 -0800206 // The program desc comparison should have already assured that the stage counts match.
207 SkASSERT(this->numFragmentStages() == that.numFragmentStages());
egdanield9aa2182014-10-09 13:47:05 -0700208 for (int i = 0; i < this->numFragmentStages(); i++) {
bsalomonae59b772014-11-19 08:23:49 -0800209
210 if (this->getFragmentStage(i) != that.getFragmentStage(i)) {
egdaniel89af44a2014-09-26 06:15:04 -0700211 return false;
212 }
213 }
joshualitt9b989322014-12-15 14:16:27 -0800214
215 // Now update the GrPrimitiveProcessor's batch tracker
216 fPrimitiveProcessor->makeEqual(&fBatchTracker, that.getBatchTracker());
egdaniel89af44a2014-09-26 06:15:04 -0700217 return true;
218}
219