blob: bc1ce7dc930258052749c203fde00296401dd236 [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,
joshualitt71c92602015-01-14 08:12:47 -080017 const GrPrimitiveProcessor* primProc,
joshualittdafa4d02014-12-04 08:59:10 -080018 const GrDrawTargetCaps& caps,
bsalomon3e791242014-12-17 13:43:13 -080019 const GrScissorState& scissorState,
joshualitt79f8fae2014-10-28 17:59:26 -070020 const GrDeviceCoordTexture* dstCopy,
joshualittdafa4d02014-12-04 08:59:10 -080021 GrGpu::DrawType drawType)
22 : fFinalized(false) {
23 fDrawType = drawType;
egdaniel95131432014-12-09 11:15:43 -080024
joshualitt71c92602015-01-14 08:12:47 -080025 fPrimitiveProcessor.reset(primProc);
joshualitt56995b52014-12-11 15:44:02 -080026
27
28 const GrProcOptInfo& colorPOI = drawState.colorProcInfo(fPrimitiveProcessor);
29 const GrProcOptInfo& coveragePOI = drawState.coverageProcInfo(fPrimitiveProcessor);
egdaniel95131432014-12-09 11:15:43 -080030
31 // Create XferProcessor from DS's XPFactory
32 SkAutoTUnref<GrXferProcessor> xferProcessor(
33 drawState.getXPFactory()->createXferProcessor(colorPOI, coveragePOI));
34
joshualitt9b989322014-12-15 14:16:27 -080035 GrColor overrideColor = GrColor_ILLEGAL;
36 if (colorPOI.firstEffectiveStageIndex() != 0) {
37 overrideColor = colorPOI.inputColorToEffectiveStage();
38 }
39
egdaniel95131432014-12-09 11:15:43 -080040 GrXferProcessor::OptFlags optFlags;
41 if (xferProcessor) {
42 fXferProcessor.reset(xferProcessor.get());
43
44 optFlags = xferProcessor->getOptimizations(colorPOI,
45 coveragePOI,
egdaniel95131432014-12-09 11:15:43 -080046 drawState.getStencil().doesWrite(),
joshualitt9b989322014-12-15 14:16:27 -080047 &overrideColor,
egdanielc2304142014-12-11 13:15:13 -080048 caps);
egdaniel95131432014-12-09 11:15:43 -080049 }
bsalomonb03c4a32014-11-20 09:56:11 -080050
51 // When path rendering the stencil settings are not always set on the draw state
52 // so we must check the draw type. In cases where we will skip drawing we simply return a
53 // null GrOptDrawState.
bsalomon3e791242014-12-17 13:43:13 -080054 if (!xferProcessor || (GrXferProcessor::kSkipDraw_OptFlag & optFlags)) {
bsalomonb03c4a32014-11-20 09:56:11 -080055 // Set the fields that don't default init and return. The lack of a render target will
56 // indicate that this can be skipped.
57 fFlags = 0;
bsalomonb03c4a32014-11-20 09:56:11 -080058 fDrawFace = GrDrawState::kInvalid_DrawFace;
bsalomonb03c4a32014-11-20 09:56:11 -080059 return;
60 }
61
62 fRenderTarget.reset(drawState.fRenderTarget.get());
63 SkASSERT(fRenderTarget);
joshualitt54e0c122014-11-19 09:38:51 -080064 fScissorState = scissorState;
egdaniel3658f382014-09-15 07:01:59 -070065 fStencilSettings = drawState.getStencil();
bsalomon04ddf892014-11-19 12:36:22 -080066 fDrawFace = drawState.getDrawFace();
joshualitt9176e2c2014-11-20 07:28:52 -080067 // TODO move this out of optDrawState
68 if (dstCopy) {
69 fDstCopy = *dstCopy;
70 }
71
bsalomon04ddf892014-11-19 12:36:22 -080072 fFlags = 0;
73 if (drawState.isHWAntialias()) {
74 fFlags |= kHWAA_Flag;
75 }
bsalomon04ddf892014-11-19 12:36:22 -080076 if (drawState.isDither()) {
77 fFlags |= kDither_Flag;
78 }
79
egdaniel912b3d22014-11-17 07:45:53 -080080 int firstColorStageIdx = colorPOI.firstEffectiveStageIndex();
egdaniel912b3d22014-11-17 07:45:53 -080081
82 // TODO: Once we can handle single or four channel input into coverage stages then we can use
83 // drawState's coverageProcInfo (like color above) to set this initial information.
84 int firstCoverageStageIdx = 0;
egdaniel912b3d22014-11-17 07:45:53 -080085
egdaniel95131432014-12-09 11:15:43 -080086 GrXferProcessor::BlendInfo blendInfo;
87 fXferProcessor->getBlendInfo(&blendInfo);
egdaniel95131432014-12-09 11:15:43 -080088
89 this->adjustProgramFromOptimizations(drawState, optFlags, colorPOI, coveragePOI,
90 &firstColorStageIdx, &firstCoverageStageIdx);
91
egdaniel71e236c2015-01-20 06:34:51 -080092 fDescInfo.fReadsDst = fXferProcessor->willReadDstColor();
93
joshualitt290c09b2014-12-19 13:45:20 -080094 bool usesLocalCoords = false;
95
bsalomonae59b772014-11-19 08:23:49 -080096 // Copy Stages from DS to ODS
bsalomonae59b772014-11-19 08:23:49 -080097 for (int i = firstColorStageIdx; i < drawState.numColorStages(); ++i) {
98 SkNEW_APPEND_TO_TARRAY(&fFragmentStages,
99 GrPendingFragmentStage,
joshualitt40d4bd82014-12-29 09:04:40 -0800100 (drawState.fColorStages[i]));
joshualitt290c09b2014-12-19 13:45:20 -0800101 usesLocalCoords = usesLocalCoords ||
joshualitt40d4bd82014-12-29 09:04:40 -0800102 drawState.fColorStages[i].processor()->usesLocalCoords();
bsalomonae59b772014-11-19 08:23:49 -0800103 }
egdaniel95131432014-12-09 11:15:43 -0800104
egdanield9aa2182014-10-09 13:47:05 -0700105 fNumColorStages = fFragmentStages.count();
bsalomonae59b772014-11-19 08:23:49 -0800106 for (int i = firstCoverageStageIdx; i < drawState.numCoverageStages(); ++i) {
107 SkNEW_APPEND_TO_TARRAY(&fFragmentStages,
108 GrPendingFragmentStage,
joshualitt40d4bd82014-12-29 09:04:40 -0800109 (drawState.fCoverageStages[i]));
joshualitt290c09b2014-12-19 13:45:20 -0800110 usesLocalCoords = usesLocalCoords ||
joshualitt40d4bd82014-12-29 09:04:40 -0800111 drawState.fCoverageStages[i].processor()->usesLocalCoords();
egdaniel9cf45bf2014-10-08 06:49:10 -0700112 }
joshualitt79f8fae2014-10-28 17:59:26 -0700113
joshualitt87f48d92014-12-04 10:41:40 -0800114 // let the GP init the batch tracker
joshualitt9b989322014-12-15 14:16:27 -0800115 GrGeometryProcessor::InitBT init;
116 init.fColorIgnored = SkToBool(optFlags & GrXferProcessor::kIgnoreColor_OptFlag);
117 init.fOverrideColor = init.fColorIgnored ? GrColor_ILLEGAL : overrideColor;
118 init.fCoverageIgnored = SkToBool(optFlags & GrXferProcessor::kIgnoreCoverage_OptFlag);
joshualitt290c09b2014-12-19 13:45:20 -0800119 init.fUsesLocalCoords = usesLocalCoords;
joshualitt9b989322014-12-15 14:16:27 -0800120 fPrimitiveProcessor->initBatchTracker(&fBatchTracker, init);
egdanielc0648242014-09-22 13:17:02 -0700121}
122
egdaniel95131432014-12-09 11:15:43 -0800123void GrOptDrawState::adjustProgramFromOptimizations(const GrDrawState& ds,
124 GrXferProcessor::OptFlags flags,
125 const GrProcOptInfo& colorPOI,
126 const GrProcOptInfo& coveragePOI,
127 int* firstColorStageIdx,
128 int* firstCoverageStageIdx) {
joshualittdafa4d02014-12-04 08:59:10 -0800129 fDescInfo.fReadsFragPosition = false;
egdaniela7dc0a82014-09-17 08:25:05 -0700130
joshualitt9b989322014-12-15 14:16:27 -0800131 if ((flags & GrXferProcessor::kIgnoreColor_OptFlag) ||
132 (flags & GrXferProcessor::kOverrideColor_OptFlag)) {
egdaniel95131432014-12-09 11:15:43 -0800133 *firstColorStageIdx = ds.numColorStages();
egdaniel95131432014-12-09 11:15:43 -0800134 } else {
egdaniel95131432014-12-09 11:15:43 -0800135 fDescInfo.fReadsFragPosition = colorPOI.readsFragPosition();
egdaniela7dc0a82014-09-17 08:25:05 -0700136 }
egdaniel95131432014-12-09 11:15:43 -0800137
joshualitt9b989322014-12-15 14:16:27 -0800138 if (flags & GrXferProcessor::kIgnoreCoverage_OptFlag) {
egdaniel95131432014-12-09 11:15:43 -0800139 *firstCoverageStageIdx = ds.numCoverageStages();
egdaniel95131432014-12-09 11:15:43 -0800140 } else {
egdaniel95131432014-12-09 11:15:43 -0800141 if (coveragePOI.readsFragPosition()) {
142 fDescInfo.fReadsFragPosition = true;
143 }
egdaniela7dc0a82014-09-17 08:25:05 -0700144 }
145}
146
joshualittdafa4d02014-12-04 08:59:10 -0800147void GrOptDrawState::finalize(GrGpu* gpu) {
148 gpu->buildProgramDesc(*this, fDescInfo, fDrawType, &fDesc);
149 fFinalized = true;
150}
151
egdaniel89af44a2014-09-26 06:15:04 -0700152////////////////////////////////////////////////////////////////////////////////
153
joshualitt9b989322014-12-15 14:16:27 -0800154bool GrOptDrawState::combineIfPossible(const GrOptDrawState& that) {
joshualittdafa4d02014-12-04 08:59:10 -0800155 if (fDescInfo != that.fDescInfo) {
joshualittf78c60c2014-12-04 06:01:45 -0800156 return false;
157 }
joshualittdafa4d02014-12-04 08:59:10 -0800158
egdaniel89af44a2014-09-26 06:15:04 -0700159 if (this->getRenderTarget() != that.getRenderTarget() ||
joshualittdafa4d02014-12-04 08:59:10 -0800160 this->fFragmentStages.count() != that.fFragmentStages.count() ||
161 this->fNumColorStages != that.fNumColorStages ||
joshualitt54e0c122014-11-19 09:38:51 -0800162 this->fScissorState != that.fScissorState ||
joshualittdafa4d02014-12-04 08:59:10 -0800163 this->fDrawType != that.fDrawType ||
bsalomon04ddf892014-11-19 12:36:22 -0800164 this->fFlags != that.fFlags ||
egdaniel89af44a2014-09-26 06:15:04 -0700165 this->fStencilSettings != that.fStencilSettings ||
joshualitt9176e2c2014-11-20 07:28:52 -0800166 this->fDrawFace != that.fDrawFace ||
167 this->fDstCopy.texture() != that.fDstCopy.texture()) {
egdaniel89af44a2014-09-26 06:15:04 -0700168 return false;
169 }
170
joshualitt9b989322014-12-15 14:16:27 -0800171 if (!this->getPrimitiveProcessor()->canMakeEqual(fBatchTracker,
172 *that.getPrimitiveProcessor(),
173 that.getBatchTracker())) {
egdaniel89af44a2014-09-26 06:15:04 -0700174 return false;
175 }
176
egdanielc2304142014-12-11 13:15:13 -0800177 if (!this->getXferProcessor()->isEqual(*that.getXferProcessor())) {
178 return false;
179 }
180
bsalomonae59b772014-11-19 08:23:49 -0800181 // The program desc comparison should have already assured that the stage counts match.
182 SkASSERT(this->numFragmentStages() == that.numFragmentStages());
egdanield9aa2182014-10-09 13:47:05 -0700183 for (int i = 0; i < this->numFragmentStages(); i++) {
bsalomonae59b772014-11-19 08:23:49 -0800184
185 if (this->getFragmentStage(i) != that.getFragmentStage(i)) {
egdaniel89af44a2014-09-26 06:15:04 -0700186 return false;
187 }
188 }
joshualitt9b989322014-12-15 14:16:27 -0800189
190 // Now update the GrPrimitiveProcessor's batch tracker
191 fPrimitiveProcessor->makeEqual(&fBatchTracker, that.getBatchTracker());
egdaniel89af44a2014-09-26 06:15:04 -0700192 return true;
193}
194