blob: efacbd0280de217d7f69f05b81f30d87f1f68da7 [file] [log] [blame]
reed@google.comac10a2d2010-12-22 21:39:39 +00001/*
epoger@google.comec3ed6a2011-07-28 14:26:00 +00002 * Copyright 2010 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.
reed@google.comac10a2d2010-12-22 21:39:39 +00006 */
7
Robert Phillipsf2361d22016-10-25 14:20:06 -04008#ifndef GrRenderTargetOpList_DEFINED
9#define GrRenderTargetOpList_DEFINED
reed@google.comac10a2d2010-12-22 21:39:39 +000010
Mike Kleinc0bd9f92019-04-23 12:05:21 -050011#include "include/core/SkMatrix.h"
12#include "include/core/SkStrokeRec.h"
13#include "include/core/SkTypes.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050014#include "include/private/SkTArray.h"
Ben Wagner729a23f2019-05-17 16:29:34 -040015#include "src/core/SkArenaAlloc.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050016#include "src/core/SkClipStack.h"
17#include "src/core/SkStringUtils.h"
18#include "src/core/SkTLazy.h"
19#include "src/gpu/GrAppliedClip.h"
Greg Danielf91aeb22019-06-18 09:58:02 -040020#include "src/gpu/GrOpList.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050021#include "src/gpu/GrPathRendering.h"
22#include "src/gpu/GrPrimitiveProcessor.h"
23#include "src/gpu/ops/GrDrawOp.h"
24#include "src/gpu/ops/GrOp.h"
Scroggo97c88c22011-05-11 14:05:25 +000025
joshualitt086cee12016-01-12 06:45:24 -080026class GrAuditTrail;
Brian Salomon7dae46a2016-12-14 16:21:37 -050027class GrClearOp;
bsalomon4b91f762015-05-19 09:29:46 -070028class GrCaps;
Robert Phillipsc7635fa2016-10-28 13:25:24 -040029class GrRenderTargetProxy;
sugoi@google.com12b4e272012-12-06 20:13:11 +000030
Robert Phillipsf2361d22016-10-25 14:20:06 -040031class GrRenderTargetOpList final : public GrOpList {
Brian Salomon54d212e2017-03-21 14:22:38 -040032private:
Robert Phillipsbb581ce2017-05-29 15:05:15 -040033 using DstProxy = GrXferProcessor::DstProxy;
Brian Salomon54d212e2017-03-21 14:22:38 -040034
bsalomon@google.comf6601872012-08-28 21:11:35 +000035public:
Robert Phillips12c46292019-04-23 07:36:17 -040036 GrRenderTargetOpList(sk_sp<GrOpMemoryPool>, sk_sp<GrRenderTargetProxy>, GrAuditTrail*);
bsalomona73239a2015-04-28 13:35:17 -070037
Robert Phillipsf2361d22016-10-25 14:20:06 -040038 ~GrRenderTargetOpList() override;
reed@google.comac10a2d2010-12-22 21:39:39 +000039
Brian Salomon588cec72018-11-14 13:56:37 -050040 bool isEmpty() const { return fOpChains.empty(); }
Robert Phillipsf5442bb2017-04-17 14:18:34 -040041
reed@google.comac10a2d2010-12-22 21:39:39 +000042 /**
bsalomona73239a2015-04-28 13:35:17 -070043 * Empties the draw buffer of any queued up draws.
44 */
Chris Daltona84cacf2017-10-04 10:30:29 -060045 void endFlush() override;
Robert Phillipsf2361d22016-10-25 14:20:06 -040046
bsalomona73239a2015-04-28 13:35:17 -070047 /**
bsalomondc438982016-08-31 11:53:49 -070048 * Together these two functions flush all queued up draws to GrCommandBuffer. The return value
Brian Salomon1e41f4a2016-12-07 15:05:04 -050049 * of executeOps() indicates whether any commands were actually issued to the GPU.
bsalomona73239a2015-04-28 13:35:17 -070050 */
Brian Osman407b3422017-08-22 15:01:32 -040051 void onPrepare(GrOpFlushState* flushState) override;
52 bool onExecute(GrOpFlushState* flushState) override;
bsalomona73239a2015-04-28 13:35:17 -070053
Chris Dalton08755122019-08-05 16:13:47 -060054 void addOp(std::unique_ptr<GrOp> op, GrTextureResolveManager textureResolveManager,
55 const GrCaps& caps) {
56 auto addDependency = [ textureResolveManager, &caps, this ] (
57 GrSurfaceProxy* p, GrMipMapped mipmapped) {
58 this->addDependency(p, mipmapped, textureResolveManager, caps);
Robert Phillipsb493eeb2017-09-13 13:10:52 -040059 };
60
61 op->visitProxies(addDependency);
62
Chris Dalton945ee652019-01-23 09:10:36 -070063 this->recordOp(std::move(op), GrProcessorSet::EmptySetAnalysis(), nullptr, nullptr, caps);
Brian Salomon54d212e2017-03-21 14:22:38 -040064 }
Robert Phillipsb493eeb2017-09-13 13:10:52 -040065
Chris Dalton08755122019-08-05 16:13:47 -060066 void addWaitOp(std::unique_ptr<GrOp> op, GrTextureResolveManager textureResolveManager,
67 const GrCaps& caps) {
68 fHasWaitOp = true;
69 this->addOp(std::move(op), textureResolveManager, caps);
Greg Danielcb324152019-02-25 11:36:53 -050070 }
71
Chris Dalton945ee652019-01-23 09:10:36 -070072 void addDrawOp(std::unique_ptr<GrDrawOp> op, const GrProcessorSet::Analysis& processorAnalysis,
Chris Dalton08755122019-08-05 16:13:47 -060073 GrAppliedClip&& clip, const DstProxy& dstProxy,
74 GrTextureResolveManager textureResolveManager, const GrCaps& caps) {
75 auto addDependency = [ textureResolveManager, &caps, this ] (
76 GrSurfaceProxy* p, GrMipMapped mipmapped) {
77 this->addDependency(p, mipmapped, textureResolveManager, caps);
Robert Phillipsb493eeb2017-09-13 13:10:52 -040078 };
79
80 op->visitProxies(addDependency);
81 clip.visitProxies(addDependency);
Robert Phillips2739ab02018-07-27 07:35:55 -040082 if (dstProxy.proxy()) {
Chris Dalton7eb5c0f2019-05-23 15:15:47 -060083 addDependency(dstProxy.proxy(), GrMipMapped::kNo);
Robert Phillips2739ab02018-07-27 07:35:55 -040084 }
Robert Phillipsb493eeb2017-09-13 13:10:52 -040085
Chris Dalton945ee652019-01-23 09:10:36 -070086 this->recordOp(std::move(op), processorAnalysis, clip.doesClip() ? &clip : nullptr,
87 &dstProxy, caps);
Brian Salomon69868af2016-12-22 15:42:51 -050088 }
robertphillips9199a9f2016-07-13 07:48:43 -070089
Robert Phillips380b90c2017-08-30 07:41:07 -040090 void discard();
91
Robert Phillipsbe9aff22019-02-15 11:33:22 -050092 bool copySurface(GrRecordingContext*,
Robert Phillipsbf25d432017-04-07 10:08:53 -040093 GrSurfaceProxy* src,
bsalomonf90a02b2014-11-26 12:28:00 -080094 const SkIRect& srcRect,
Robert Phillips2de8cfa2017-06-28 10:33:41 -040095 const SkIPoint& dstPoint) override;
robertphillips@google.comff175842012-05-14 19:31:39 +000096
Brian Osman45580d32016-11-23 09:37:01 -050097 GrRenderTargetOpList* asRenderTargetOpList() override { return this; }
98
Robert Phillips27483912018-04-20 12:43:18 -040099 SkDEBUGCODE(void dump(bool printDependencies) const override;)
Robert Phillipsc84c0302017-05-08 15:35:11 -0400100 SkDEBUGCODE(int numClips() const override { return fNumClips; })
Chris Dalton706a6ff2017-11-29 22:01:06 -0700101 SkDEBUGCODE(void visitProxies_debugOnly(const GrOp::VisitProxyFunc&) const;)
Robert Phillipsc84c0302017-05-08 15:35:11 -0400102
Brian Salomon13540482018-07-09 10:31:47 -0400103private:
Brian Salomonc3833b42018-07-09 18:23:58 +0000104 friend class GrRenderTargetContextPriv; // for stencil clip state. TODO: this is invasive
105
Michael Ludwigc39d0c82019-01-15 10:03:43 -0500106 // The RTC and RTOpList have to work together to handle buffer clears. In most cases, buffer
107 // clearing can be done natively, in which case the op list's load ops are sufficient. In other
108 // cases, draw ops must be used, which makes the RTC the best place for those decisions. This,
109 // however, requires that the RTC be able to coordinate with the op list to achieve similar ends
110 friend class GrRenderTargetContext;
111
Chris Daltonaa3cbb82019-08-21 00:01:21 -0600112 bool isNoOp() const {
113 // TODO: GrLoadOp::kDiscard -> [empty opList] -> GrStoreOp::kStore should also be a no-op.
114 // We don't count it as a no-op right now because of Vulkan. There are real cases where we
115 // store a discard, and if we skip that render pass, then the next time we load the render
116 // target, Vulkan detects loading of uninitialized memory and complains. If we don't skip
117 // storing the discard, then we trick Vulkan and it doesn't notice us doing anything wrong.
118 // We should definitely address this issue properly.
119 //
120 // TODO: We should also consider stencil load/store here. We get away with it for now
121 // because we never discard stencil buffers.
122 return fOpChains.empty() && GrLoadOp::kClear != fColorLoadOp &&
123 GrLoadOp::kDiscard != fColorLoadOp;
124 }
125
Robert Phillips9313aa72019-04-09 18:41:27 -0400126 bool onIsUsed(GrSurfaceProxy*) const override;
127
Michael Ludwig6e17f1d2019-05-15 14:00:20 +0000128 // Must only be called if native stencil buffer clearing is enabled
Chris Dalton6b982802019-06-27 13:53:46 -0600129 void setStencilLoadOp(GrLoadOp op) { fStencilLoadOp = op; }
Michael Ludwigc39d0c82019-01-15 10:03:43 -0500130 // Must only be called if native color buffer clearing is enabled.
131 void setColorLoadOp(GrLoadOp op, const SkPMColor4f& color);
132 // Sets the clear color to transparent black
133 void setColorLoadOp(GrLoadOp op) {
134 static const SkPMColor4f kDefaultClearColor = {0.f, 0.f, 0.f, 0.f};
135 this->setColorLoadOp(op, kDefaultClearColor);
136 }
137
Chris Dalton6b982802019-06-27 13:53:46 -0600138 enum class CanDiscardPreviousOps : bool {
139 kYes = true,
140 kNo = false
141 };
142
Michael Ludwigc39d0c82019-01-15 10:03:43 -0500143 // Perform book-keeping for a fullscreen clear, regardless of how the clear is implemented later
144 // (i.e. setColorLoadOp(), adding a ClearOp, or adding a GrFillRectOp that covers the device).
145 // Returns true if the clear can be converted into a load op (barring device caps).
Chris Dalton6b982802019-06-27 13:53:46 -0600146 bool resetForFullscreenClear(CanDiscardPreviousOps);
Michael Ludwigc39d0c82019-01-15 10:03:43 -0500147
Robert Phillipsc994a932018-06-19 13:09:54 -0400148 void deleteOps();
149
Brian Salomon588cec72018-11-14 13:56:37 -0500150 class OpChain {
151 public:
152 OpChain(const OpChain&) = delete;
153 OpChain& operator=(const OpChain&) = delete;
Chris Dalton945ee652019-01-23 09:10:36 -0700154 OpChain(std::unique_ptr<GrOp>, GrProcessorSet::Analysis, GrAppliedClip*, const DstProxy*);
Brian Salomon588cec72018-11-14 13:56:37 -0500155
156 ~OpChain() {
157 // The ops are stored in a GrMemoryPool and must be explicitly deleted via the pool.
158 SkASSERT(fList.empty());
Brian Salomon54d212e2017-03-21 14:22:38 -0400159 }
Chris Dalton8816b932017-11-29 16:48:25 -0700160
Chris Dalton1706cbf2019-05-21 19:35:29 -0600161 void visitProxies(const GrOp::VisitProxyFunc&) const;
Robert Phillipsc994a932018-06-19 13:09:54 -0400162
Brian Salomon588cec72018-11-14 13:56:37 -0500163 GrOp* head() const { return fList.head(); }
Robert Phillips7c525e62018-06-12 10:11:12 -0400164
Brian Salomon588cec72018-11-14 13:56:37 -0500165 GrAppliedClip* appliedClip() const { return fAppliedClip; }
166 const DstProxy& dstProxy() const { return fDstProxy; }
167 const SkRect& bounds() const { return fBounds; }
Chris Dalton8816b932017-11-29 16:48:25 -0700168
Brian Salomon588cec72018-11-14 13:56:37 -0500169 // Deletes all the ops in the chain via the pool.
170 void deleteOps(GrOpMemoryPool* pool);
171
172 // Attempts to move the ops from the passed chain to this chain at the head. Also attempts
173 // to merge ops between the chains. Upon success the passed chain is empty.
174 // Fails when the chains aren't of the same op type, have different clips or dst proxies.
175 bool prependChain(OpChain*, const GrCaps&, GrOpMemoryPool*, GrAuditTrail*);
176
177 // Attempts to add 'op' to this chain either by merging or adding to the tail. Returns
178 // 'op' to the caller upon failure, otherwise null. Fails when the op and chain aren't of
179 // the same op type, have different clips or dst proxies.
Chris Dalton945ee652019-01-23 09:10:36 -0700180 std::unique_ptr<GrOp> appendOp(std::unique_ptr<GrOp> op, GrProcessorSet::Analysis,
181 const DstProxy*, const GrAppliedClip*, const GrCaps&,
182 GrOpMemoryPool*, GrAuditTrail*);
Brian Salomon588cec72018-11-14 13:56:37 -0500183
184 private:
185 class List {
186 public:
187 List() = default;
188 List(std::unique_ptr<GrOp>);
189 List(List&&);
190 List& operator=(List&& that);
191
192 bool empty() const { return !SkToBool(fHead); }
193 GrOp* head() const { return fHead.get(); }
194 GrOp* tail() const { return fTail; }
195
196 std::unique_ptr<GrOp> popHead();
197 std::unique_ptr<GrOp> removeOp(GrOp* op);
198 void pushHead(std::unique_ptr<GrOp> op);
199 void pushTail(std::unique_ptr<GrOp>);
200
201 void validate() const;
202
203 private:
204 std::unique_ptr<GrOp> fHead;
205 GrOp* fTail = nullptr;
206 };
207
208 void validate() const;
209
Chris Dalton945ee652019-01-23 09:10:36 -0700210 bool tryConcat(List*, GrProcessorSet::Analysis, const DstProxy&, const GrAppliedClip*,
211 const SkRect& bounds, const GrCaps&, GrOpMemoryPool*, GrAuditTrail*);
Chris Dalton6f6ae6a2019-01-18 12:10:36 -0700212 static List DoConcat(List, List, const GrCaps&, GrOpMemoryPool*, GrAuditTrail*);
Brian Salomon588cec72018-11-14 13:56:37 -0500213
214 List fList;
Chris Dalton945ee652019-01-23 09:10:36 -0700215 GrProcessorSet::Analysis fProcessorAnalysis;
Brian Salomon588cec72018-11-14 13:56:37 -0500216 DstProxy fDstProxy;
217 GrAppliedClip* fAppliedClip;
218 SkRect fBounds;
Brian Salomon54d212e2017-03-21 14:22:38 -0400219 };
220
Chris Dalton6b498102019-08-01 14:14:52 -0600221 void handleInternalAllocationFailure() override;
Greg Danielaa3dfbe2018-01-29 10:34:25 -0500222
Robert Phillipsd375dbf2017-09-14 12:45:25 -0400223 void gatherProxyIntervals(GrResourceAllocator*) const override;
224
Chris Dalton945ee652019-01-23 09:10:36 -0700225 void recordOp(std::unique_ptr<GrOp>, GrProcessorSet::Analysis, GrAppliedClip*, const DstProxy*,
226 const GrCaps& caps);
Brian Salomon2790c522016-12-09 16:32:23 -0500227
Robert Phillipsee683652017-04-26 11:53:10 -0400228 void forwardCombine(const GrCaps&);
bsalomonad792c12015-09-10 11:10:50 -0700229
Chris Daltonaa3cbb82019-08-21 00:01:21 -0600230 ExpectedOutcome onMakeClosed(const GrCaps& caps) override {
231 this->forwardCombine(caps);
232 return (this->isNoOp()) ? ExpectedOutcome::kTargetUnchanged : ExpectedOutcome::kTargetDirty;
233 }
234
235 GrLoadOp fColorLoadOp = GrLoadOp::kLoad;
236 SkPMColor4f fLoadClearColor = SK_PMColor4fTRANSPARENT;
237 GrLoadOp fStencilLoadOp = GrLoadOp::kLoad;
238
Michael Ludwig6e17f1d2019-05-15 14:00:20 +0000239 uint32_t fLastClipStackGenID;
240 SkIRect fLastDevClipBounds;
241 int fLastClipNumAnalyticFPs;
csmartdalton7cdda992016-11-01 07:03:03 -0700242
Greg Danielcb324152019-02-25 11:36:53 -0500243 // We must track if we have a wait op so that we don't delete the op when we have a full clear.
244 bool fHasWaitOp = false;;
245
Robert Phillips6cdc22c2017-05-11 16:29:14 -0400246 // For ops/opList we have mean: 5 stdDev: 28
Chris Daltonaa3cbb82019-08-21 00:01:21 -0600247 SkSTArray<25, OpChain, true> fOpChains;
Brian Salomon54d212e2017-03-21 14:22:38 -0400248
Robert Phillipsf6d7eb12017-04-26 14:55:34 -0400249 // MDB TODO: 4096 for the first allocation of the clip space will be huge overkill.
250 // Gather statistics to determine the correct size.
Michael Ludwig6e17f1d2019-05-15 14:00:20 +0000251 SkArenaAlloc fClipAllocator{4096};
252 SkDEBUGCODE(int fNumClips;)
Brian Salomon54d212e2017-03-21 14:22:38 -0400253
Robert Phillipsf2361d22016-10-25 14:20:06 -0400254 typedef GrOpList INHERITED;
joshualitt6db519c2014-10-29 08:48:18 -0700255};
256
reed@google.comac10a2d2010-12-22 21:39:39 +0000257#endif