blob: f01236c0da67fc75cd5bf9b9238929452f163118 [file] [log] [blame]
reed@google.comac10a2d2010-12-22 21:39:39 +00001/*
Greg Danielf41b2bd2019-08-22 16:19:24 -04002 * Copyright 2019 Google Inc.
epoger@google.comec3ed6a2011-07-28 14:26:00 +00003 *
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
Greg Danielf41b2bd2019-08-22 16:19:24 -04008#ifndef GrOpsTask_DEFINED
9#define GrOpsTask_DEFINED
reed@google.comac10a2d2010-12-22 21:39:39 +000010
Mike Kleinc0bd9f92019-04-23 12:05:21 -050011#include "include/core/SkMatrix.h"
Greg Danielf41b2bd2019-08-22 16:19:24 -040012#include "include/core/SkRefCnt.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050013#include "include/core/SkStrokeRec.h"
14#include "include/core/SkTypes.h"
Robert Phillipsb7bfbc22020-07-01 12:55:01 -040015#include "include/gpu/GrRecordingContext.h"
Greg Danielf41b2bd2019-08-22 16:19:24 -040016#include "include/private/SkColorData.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050017#include "include/private/SkTArray.h"
Greg Danielf41b2bd2019-08-22 16:19:24 -040018#include "include/private/SkTDArray.h"
Ben Wagner729a23f2019-05-17 16:29:34 -040019#include "src/core/SkArenaAlloc.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050020#include "src/core/SkClipStack.h"
21#include "src/core/SkStringUtils.h"
22#include "src/core/SkTLazy.h"
23#include "src/gpu/GrAppliedClip.h"
24#include "src/gpu/GrPathRendering.h"
25#include "src/gpu/GrPrimitiveProcessor.h"
Greg Danielf41b2bd2019-08-22 16:19:24 -040026#include "src/gpu/GrRenderTask.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050027#include "src/gpu/ops/GrDrawOp.h"
28#include "src/gpu/ops/GrOp.h"
Scroggo97c88c22011-05-11 14:05:25 +000029
joshualitt086cee12016-01-12 06:45:24 -080030class GrAuditTrail;
Greg Danielf21bf9e2019-08-22 20:12:20 +000031class GrCaps;
Greg Danielf41b2bd2019-08-22 16:19:24 -040032class GrClearOp;
33class GrGpuBuffer;
Robert Phillipsc7635fa2016-10-28 13:25:24 -040034class GrRenderTargetProxy;
sugoi@google.com12b4e272012-12-06 20:13:11 +000035
Brian Salomon3b8486a2020-04-21 12:43:26 -040036/** Observer is notified when a GrOpsTask is closed. */
37class GrOpsTaskClosedObserver {
38public:
39 virtual ~GrOpsTaskClosedObserver() = 0;
Brian Salomonf1c9eae2020-05-01 15:00:34 -040040 /**
41 * Called when the GrOpsTask is closed. Must not add/remove observers to 'task'.
42 * The GrOpsTask will remove all its observers after it finishes calling wasClosed().
43 */
Brian Salomon3b8486a2020-04-21 12:43:26 -040044 virtual void wasClosed(const GrOpsTask& task) = 0;
45};
46
Greg Danielf41b2bd2019-08-22 16:19:24 -040047class GrOpsTask : public GrRenderTask {
Brian Salomon54d212e2017-03-21 14:22:38 -040048private:
Greg Daniel524e28b2019-11-01 11:48:53 -040049 using DstProxyView = GrXferProcessor::DstProxyView;
Brian Salomon54d212e2017-03-21 14:22:38 -040050
bsalomon@google.comf6601872012-08-28 21:11:35 +000051public:
Michael Ludwig28b0c5d2019-12-19 14:51:00 -050052 // The Arenas must outlive the GrOpsTask, either by preserving the context that owns
Michael Ludwigcb10e4d2019-12-12 13:59:20 -050053 // the pool, or by moving the pool to the DDL that takes over the GrOpsTask.
Adlai Hollerd71b7b02020-06-08 15:55:00 -040054 GrOpsTask(GrDrawingManager*, GrRecordingContext::Arenas, GrSurfaceProxyView, GrAuditTrail*);
Greg Danielf41b2bd2019-08-22 16:19:24 -040055 ~GrOpsTask() override;
bsalomona73239a2015-04-28 13:35:17 -070056
Greg Danielf41b2bd2019-08-22 16:19:24 -040057 GrOpsTask* asOpsTask() override { return this; }
reed@google.comac10a2d2010-12-22 21:39:39 +000058
Brian Salomonf1c9eae2020-05-01 15:00:34 -040059 void addClosedObserver(GrOpsTaskClosedObserver* observer) {
60 SkASSERT(observer);
61 fClosedObservers.push_back(observer);
62 }
63
64 void removeClosedObserver(GrOpsTaskClosedObserver* observer);
Brian Salomon3b8486a2020-04-21 12:43:26 -040065
Brian Salomon588cec72018-11-14 13:56:37 -050066 bool isEmpty() const { return fOpChains.empty(); }
Robert Phillipsf5442bb2017-04-17 14:18:34 -040067
reed@google.comac10a2d2010-12-22 21:39:39 +000068 /**
bsalomona73239a2015-04-28 13:35:17 -070069 * Empties the draw buffer of any queued up draws.
70 */
Adlai Hollerd71b7b02020-06-08 15:55:00 -040071 void endFlush(GrDrawingManager*) override;
Robert Phillipsf2361d22016-10-25 14:20:06 -040072
Robert Phillips29f38542019-10-16 09:20:25 -040073 void onPrePrepare(GrRecordingContext*) override;
bsalomona73239a2015-04-28 13:35:17 -070074 /**
bsalomondc438982016-08-31 11:53:49 -070075 * Together these two functions flush all queued up draws to GrCommandBuffer. The return value
Brian Salomon1e41f4a2016-12-07 15:05:04 -050076 * of executeOps() indicates whether any commands were actually issued to the GPU.
bsalomona73239a2015-04-28 13:35:17 -070077 */
Brian Osman407b3422017-08-22 15:01:32 -040078 void onPrepare(GrOpFlushState* flushState) override;
79 bool onExecute(GrOpFlushState* flushState) override;
bsalomona73239a2015-04-28 13:35:17 -070080
Michael Ludwigfcdd0612019-11-25 08:34:31 -050081 void addSampledTexture(GrSurfaceProxy* proxy) {
82 // This function takes a GrSurfaceProxy because all subsequent uses of the proxy do not
83 // require the specifics of GrTextureProxy, so this avoids a number of unnecessary virtual
84 // asTextureProxy() calls. However, sampling the proxy implicitly requires that the proxy
85 // be a texture. Eventually, when proxies are a unified type with flags, this can just
86 // assert that capability.
87 SkASSERT(proxy->asTextureProxy());
Greg Danielb20d7e52019-09-03 13:54:39 -040088 fSampledProxies.push_back(proxy);
89 }
90
Adlai Hollerd71b7b02020-06-08 15:55:00 -040091 void addOp(GrDrawingManager* drawingMgr, std::unique_ptr<GrOp> op,
92 GrTextureResolveManager textureResolveManager, const GrCaps& caps) {
93 auto addDependency = [ drawingMgr, textureResolveManager, &caps, this ] (
Brian Salomon7e67dca2020-07-21 09:27:25 -040094 GrSurfaceProxy* p, GrMipmapped mipmapped) {
Adlai Hollerd71b7b02020-06-08 15:55:00 -040095 this->addDependency(drawingMgr, p, mipmapped, textureResolveManager, caps);
Robert Phillipsb493eeb2017-09-13 13:10:52 -040096 };
97
98 op->visitProxies(addDependency);
99
Chris Dalton945ee652019-01-23 09:10:36 -0700100 this->recordOp(std::move(op), GrProcessorSet::EmptySetAnalysis(), nullptr, nullptr, caps);
Brian Salomon54d212e2017-03-21 14:22:38 -0400101 }
Robert Phillipsb493eeb2017-09-13 13:10:52 -0400102
Adlai Hollerd71b7b02020-06-08 15:55:00 -0400103 void addDrawOp(GrDrawingManager* drawingMgr, std::unique_ptr<GrDrawOp> op,
104 const GrProcessorSet::Analysis& processorAnalysis,
Greg Daniel524e28b2019-11-01 11:48:53 -0400105 GrAppliedClip&& clip, const DstProxyView& dstProxyView,
Chris Dalton08755122019-08-05 16:13:47 -0600106 GrTextureResolveManager textureResolveManager, const GrCaps& caps) {
Adlai Hollerd71b7b02020-06-08 15:55:00 -0400107 auto addDependency = [ drawingMgr, textureResolveManager, &caps, this ] (
Brian Salomon7e67dca2020-07-21 09:27:25 -0400108 GrSurfaceProxy* p, GrMipmapped mipmapped) {
Greg Danielb20d7e52019-09-03 13:54:39 -0400109 this->addSampledTexture(p);
Adlai Hollerd71b7b02020-06-08 15:55:00 -0400110 this->addDependency(drawingMgr, p, mipmapped, textureResolveManager, caps);
Robert Phillipsb493eeb2017-09-13 13:10:52 -0400111 };
112
113 op->visitProxies(addDependency);
114 clip.visitProxies(addDependency);
Greg Daniel524e28b2019-11-01 11:48:53 -0400115 if (dstProxyView.proxy()) {
Greg Danield358cbe2020-09-11 09:33:54 -0400116 if (GrDstSampleTypeUsesTexture(dstProxyView.dstSampleType())) {
117 this->addSampledTexture(dstProxyView.proxy());
118 }
Brian Salomon7e67dca2020-07-21 09:27:25 -0400119 addDependency(dstProxyView.proxy(), GrMipmapped::kNo);
Greg Danield358cbe2020-09-11 09:33:54 -0400120 if (this->target(0).proxy() == dstProxyView.proxy()) {
Greg Daniel9a18b082020-08-14 14:03:50 -0400121 // Since we are sampling and drawing to the same surface we will need to use
122 // texture barriers.
Greg Danield358cbe2020-09-11 09:33:54 -0400123 SkASSERT(GrDstSampleTypeDirectlySamplesDst(dstProxyView.dstSampleType()));
124 fRenderPassXferBarriers |= GrXferBarrierFlags::kTexture;
Greg Daniel9a18b082020-08-14 14:03:50 -0400125 }
Greg Danield358cbe2020-09-11 09:33:54 -0400126 SkASSERT(dstProxyView.dstSampleType() != GrDstSampleType::kAsInputAttachment ||
127 dstProxyView.offset().isZero());
Robert Phillips2739ab02018-07-27 07:35:55 -0400128 }
Robert Phillipsb493eeb2017-09-13 13:10:52 -0400129
Greg Danield358cbe2020-09-11 09:33:54 -0400130 if (processorAnalysis.usesNonCoherentHWBlending()) {
131 fRenderPassXferBarriers |= GrXferBarrierFlags::kBlend;
132 }
Greg Daniel9a18b082020-08-14 14:03:50 -0400133
Chris Dalton945ee652019-01-23 09:10:36 -0700134 this->recordOp(std::move(op), processorAnalysis, clip.doesClip() ? &clip : nullptr,
Greg Daniel524e28b2019-11-01 11:48:53 -0400135 &dstProxyView, caps);
Brian Salomon69868af2016-12-22 15:42:51 -0500136 }
robertphillips9199a9f2016-07-13 07:48:43 -0700137
Robert Phillips380b90c2017-08-30 07:41:07 -0400138 void discard();
139
Robert Phillips44e2c5f2020-04-14 13:00:04 -0400140#ifdef SK_DEBUG
Robert Phillips44e2c5f2020-04-14 13:00:04 -0400141 int numClips() const override { return fNumClips; }
142 void visitProxies_debugOnly(const GrOp::VisitProxyFunc&) const override;
143#endif
Robert Phillipsc84c0302017-05-08 15:35:11 -0400144
Robert Phillips438d9862019-11-14 12:46:05 -0500145#if GR_TEST_UTILS
John Stiles1e0136e2020-08-12 18:44:00 -0400146 void dump(bool printDependencies) const override;
147 const char* name() const final { return "Ops"; }
Robert Phillips438d9862019-11-14 12:46:05 -0500148 int numOpChains() const { return fOpChains.count(); }
149 const GrOp* getChain(int index) const { return fOpChains[index].head(); }
150#endif
151
Brian Salomon13540482018-07-09 10:31:47 -0400152private:
Chris Daltonaa3cbb82019-08-21 00:01:21 -0600153 bool isNoOp() const {
Chris Daltona43e2d22019-08-30 01:14:33 -0600154 // TODO: GrLoadOp::kDiscard (i.e., storing a discard) should also be grounds for skipping
155 // execution. We currently don't because of Vulkan. See http://skbug.com/9373.
Chris Daltonaa3cbb82019-08-21 00:01:21 -0600156 //
157 // TODO: We should also consider stencil load/store here. We get away with it for now
158 // because we never discard stencil buffers.
Chris Daltona43e2d22019-08-30 01:14:33 -0600159 return fOpChains.empty() && GrLoadOp::kLoad == fColorLoadOp;
Chris Daltonaa3cbb82019-08-21 00:01:21 -0600160 }
161
Greg Danielf41b2bd2019-08-22 16:19:24 -0400162 void deleteOps();
Robert Phillips9313aa72019-04-09 18:41:27 -0400163
Chris Dalton674f77a2019-09-30 20:49:39 -0600164 enum class StencilContent {
165 kDontCare,
166 kUserBitsCleared, // User bits: cleared
167 // Clip bit: don't care (Ganesh always pre-clears the clip bit.)
168 kPreserved
169 };
170
171 // Lets the caller specify what the content of the stencil buffer should be at the beginning
172 // of the render pass.
173 //
174 // When requesting kClear: Tilers will load the stencil buffer with a "clear" op; non-tilers
175 // will clear the stencil on first load, and then preserve it on subsequent loads. (Preserving
176 // works because renderTargetContexts are required to leave the user bits in a cleared state
177 // once finished.)
178 //
179 // NOTE: initialContent must not be kClear if caps.performStencilClearsAsDraws() is true.
180 void setInitialStencilContent(StencilContent initialContent) {
181 fInitialStencilContent = initialContent;
182 }
183
184 // If a renderTargetContext splits its opsTask, it uses this method to guarantee stencil values
185 // get preserved across its split tasks.
186 void setMustPreserveStencil() { fMustPreserveStencil = true; }
187
Michael Ludwigc39d0c82019-01-15 10:03:43 -0500188 // Must only be called if native color buffer clearing is enabled.
189 void setColorLoadOp(GrLoadOp op, const SkPMColor4f& color);
190 // Sets the clear color to transparent black
191 void setColorLoadOp(GrLoadOp op) {
192 static const SkPMColor4f kDefaultClearColor = {0.f, 0.f, 0.f, 0.f};
193 this->setColorLoadOp(op, kDefaultClearColor);
194 }
195
Chris Dalton6b982802019-06-27 13:53:46 -0600196 enum class CanDiscardPreviousOps : bool {
197 kYes = true,
198 kNo = false
199 };
200
Michael Ludwigc39d0c82019-01-15 10:03:43 -0500201 // Perform book-keeping for a fullscreen clear, regardless of how the clear is implemented later
202 // (i.e. setColorLoadOp(), adding a ClearOp, or adding a GrFillRectOp that covers the device).
203 // Returns true if the clear can be converted into a load op (barring device caps).
Chris Dalton6b982802019-06-27 13:53:46 -0600204 bool resetForFullscreenClear(CanDiscardPreviousOps);
Michael Ludwigc39d0c82019-01-15 10:03:43 -0500205
Brian Salomon588cec72018-11-14 13:56:37 -0500206 class OpChain {
207 public:
Greg Daniel524e28b2019-11-01 11:48:53 -0400208 OpChain(std::unique_ptr<GrOp>, GrProcessorSet::Analysis, GrAppliedClip*,
209 const DstProxyView*);
Brian Salomon588cec72018-11-14 13:56:37 -0500210 ~OpChain() {
211 // The ops are stored in a GrMemoryPool and must be explicitly deleted via the pool.
212 SkASSERT(fList.empty());
Brian Salomon54d212e2017-03-21 14:22:38 -0400213 }
Chris Dalton8816b932017-11-29 16:48:25 -0700214
John Stiles48025a02020-07-27 10:33:50 -0400215 OpChain(const OpChain&) = delete;
216 OpChain& operator=(const OpChain&) = delete;
217 OpChain(OpChain&&) = default;
218 OpChain& operator=(OpChain&&) = default;
219
Chris Dalton1706cbf2019-05-21 19:35:29 -0600220 void visitProxies(const GrOp::VisitProxyFunc&) const;
Robert Phillipsc994a932018-06-19 13:09:54 -0400221
Brian Salomon588cec72018-11-14 13:56:37 -0500222 GrOp* head() const { return fList.head(); }
Robert Phillips7c525e62018-06-12 10:11:12 -0400223
Brian Salomon588cec72018-11-14 13:56:37 -0500224 GrAppliedClip* appliedClip() const { return fAppliedClip; }
Greg Daniel524e28b2019-11-01 11:48:53 -0400225 const DstProxyView& dstProxyView() const { return fDstProxyView; }
Brian Salomon588cec72018-11-14 13:56:37 -0500226 const SkRect& bounds() const { return fBounds; }
Chris Dalton8816b932017-11-29 16:48:25 -0700227
Brian Salomon588cec72018-11-14 13:56:37 -0500228 // Deletes all the ops in the chain via the pool.
229 void deleteOps(GrOpMemoryPool* pool);
230
231 // Attempts to move the ops from the passed chain to this chain at the head. Also attempts
232 // to merge ops between the chains. Upon success the passed chain is empty.
233 // Fails when the chains aren't of the same op type, have different clips or dst proxies.
Michael Ludwig28b0c5d2019-12-19 14:51:00 -0500234 bool prependChain(OpChain*, const GrCaps&, GrRecordingContext::Arenas*, GrAuditTrail*);
Brian Salomon588cec72018-11-14 13:56:37 -0500235
236 // Attempts to add 'op' to this chain either by merging or adding to the tail. Returns
237 // 'op' to the caller upon failure, otherwise null. Fails when the op and chain aren't of
238 // the same op type, have different clips or dst proxies.
Chris Dalton945ee652019-01-23 09:10:36 -0700239 std::unique_ptr<GrOp> appendOp(std::unique_ptr<GrOp> op, GrProcessorSet::Analysis,
Greg Daniel524e28b2019-11-01 11:48:53 -0400240 const DstProxyView*, const GrAppliedClip*, const GrCaps&,
Michael Ludwig28b0c5d2019-12-19 14:51:00 -0500241 GrRecordingContext::Arenas*, GrAuditTrail*);
Brian Salomon588cec72018-11-14 13:56:37 -0500242
Greg Daniel15ecdf92019-08-30 15:35:23 -0400243 void setSkipExecuteFlag() { fSkipExecute = true; }
244 bool shouldExecute() const {
245 return SkToBool(this->head()) && !fSkipExecute;
246 }
247
Brian Salomon588cec72018-11-14 13:56:37 -0500248 private:
249 class List {
250 public:
251 List() = default;
252 List(std::unique_ptr<GrOp>);
253 List(List&&);
254 List& operator=(List&& that);
255
256 bool empty() const { return !SkToBool(fHead); }
257 GrOp* head() const { return fHead.get(); }
258 GrOp* tail() const { return fTail; }
259
260 std::unique_ptr<GrOp> popHead();
261 std::unique_ptr<GrOp> removeOp(GrOp* op);
262 void pushHead(std::unique_ptr<GrOp> op);
263 void pushTail(std::unique_ptr<GrOp>);
264
265 void validate() const;
266
267 private:
268 std::unique_ptr<GrOp> fHead;
269 GrOp* fTail = nullptr;
270 };
271
272 void validate() const;
273
Greg Daniel524e28b2019-11-01 11:48:53 -0400274 bool tryConcat(List*, GrProcessorSet::Analysis, const DstProxyView&, const GrAppliedClip*,
Michael Ludwig28b0c5d2019-12-19 14:51:00 -0500275 const SkRect& bounds, const GrCaps&, GrRecordingContext::Arenas*,
Michael Ludwigd0840ec2019-12-12 09:48:38 -0500276 GrAuditTrail*);
Michael Ludwig28b0c5d2019-12-19 14:51:00 -0500277 static List DoConcat(List, List, const GrCaps&, GrRecordingContext::Arenas*, GrAuditTrail*);
Brian Salomon588cec72018-11-14 13:56:37 -0500278
279 List fList;
Chris Dalton945ee652019-01-23 09:10:36 -0700280 GrProcessorSet::Analysis fProcessorAnalysis;
Greg Daniel524e28b2019-11-01 11:48:53 -0400281 DstProxyView fDstProxyView;
Brian Salomon588cec72018-11-14 13:56:37 -0500282 GrAppliedClip* fAppliedClip;
283 SkRect fBounds;
Greg Daniel15ecdf92019-08-30 15:35:23 -0400284
285 // We set this flag to true if any of the ops' proxies fail to instantiate so that we know
286 // not to try and draw the op.
287 bool fSkipExecute = false;
Brian Salomon54d212e2017-03-21 14:22:38 -0400288 };
289
Greg Danielf41b2bd2019-08-22 16:19:24 -0400290
291 bool onIsUsed(GrSurfaceProxy*) const override;
292
Chris Dalton6b498102019-08-01 14:14:52 -0600293 void handleInternalAllocationFailure() override;
Greg Danielaa3dfbe2018-01-29 10:34:25 -0500294
Robert Phillipsd375dbf2017-09-14 12:45:25 -0400295 void gatherProxyIntervals(GrResourceAllocator*) const override;
296
Greg Daniel524e28b2019-11-01 11:48:53 -0400297 void recordOp(std::unique_ptr<GrOp>, GrProcessorSet::Analysis, GrAppliedClip*,
298 const DstProxyView*, const GrCaps& caps);
Brian Salomon2790c522016-12-09 16:32:23 -0500299
Robert Phillipsee683652017-04-26 11:53:10 -0400300 void forwardCombine(const GrCaps&);
bsalomonad792c12015-09-10 11:10:50 -0700301
Chris Dalton16a33c62019-09-24 22:19:17 -0600302 ExpectedOutcome onMakeClosed(const GrCaps& caps, SkIRect* targetUpdateBounds) override;
Chris Daltonaa3cbb82019-08-21 00:01:21 -0600303
Robert Phillips089b7c92020-08-12 11:57:07 -0400304 friend class OpsTaskTestingAccess;
Greg Danielf41b2bd2019-08-22 16:19:24 -0400305 friend class GrRenderTargetContextPriv; // for stencil clip state. TODO: this is invasive
Chris Daltonaa3cbb82019-08-21 00:01:21 -0600306
Greg Danielf41b2bd2019-08-22 16:19:24 -0400307 // The RTC and OpsTask have to work together to handle buffer clears. In most cases, buffer
308 // clearing can be done natively, in which case the op list's load ops are sufficient. In other
309 // cases, draw ops must be used, which makes the RTC the best place for those decisions. This,
310 // however, requires that the RTC be able to coordinate with the op list to achieve similar ends
311 friend class GrRenderTargetContext;
312
Michael Ludwig28b0c5d2019-12-19 14:51:00 -0500313 // This is a backpointer to the Arenas that holds the memory for this GrOpsTask's ops. In the
314 // DDL case, the Arenas must have been detached from the original recording context and moved
315 // into the owning DDL.
316 GrRecordingContext::Arenas fArenas;
317 GrAuditTrail* fAuditTrail;
Brian Salomonf1c9eae2020-05-01 15:00:34 -0400318
319 SkSTArray<2, GrOpsTaskClosedObserver*, true> fClosedObservers;
Greg Danielf41b2bd2019-08-22 16:19:24 -0400320
321 GrLoadOp fColorLoadOp = GrLoadOp::kLoad;
322 SkPMColor4f fLoadClearColor = SK_PMColor4fTRANSPARENT;
Chris Dalton674f77a2019-09-30 20:49:39 -0600323 StencilContent fInitialStencilContent = StencilContent::kDontCare;
324 bool fMustPreserveStencil = false;
Greg Danielf41b2bd2019-08-22 16:19:24 -0400325
Brian Salomon3b8486a2020-04-21 12:43:26 -0400326 uint32_t fLastClipStackGenID = SK_InvalidUniqueID;
Greg Danielf41b2bd2019-08-22 16:19:24 -0400327 SkIRect fLastDevClipBounds;
John Stiles9e8f4842020-07-09 11:30:05 -0400328 int fLastClipNumAnalyticElements;
csmartdalton7cdda992016-11-01 07:03:03 -0700329
Greg Danield358cbe2020-09-11 09:33:54 -0400330 GrXferBarrierFlags fRenderPassXferBarriers = GrXferBarrierFlags::kNone;
Greg Daniel9a18b082020-08-14 14:03:50 -0400331
Greg Danielf41b2bd2019-08-22 16:19:24 -0400332 // For ops/opsTask we have mean: 5 stdDev: 28
John Stiles48025a02020-07-27 10:33:50 -0400333 SkSTArray<25, OpChain> fOpChains;
Brian Salomon54d212e2017-03-21 14:22:38 -0400334
Robert Phillipsf6d7eb12017-04-26 14:55:34 -0400335 // MDB TODO: 4096 for the first allocation of the clip space will be huge overkill.
336 // Gather statistics to determine the correct size.
Herb Derby6e2c56f2020-08-01 16:26:04 -0400337 SkArenaAllocWithReset fClipAllocator{4096};
Greg Danielf41b2bd2019-08-22 16:19:24 -0400338 SkDEBUGCODE(int fNumClips;)
Brian Salomon54d212e2017-03-21 14:22:38 -0400339
Greg Danielb20d7e52019-09-03 13:54:39 -0400340 // TODO: We could look into this being a set if we find we're adding a lot of duplicates that is
341 // causing slow downs.
Michael Ludwigfcdd0612019-11-25 08:34:31 -0500342 SkTArray<GrSurfaceProxy*, true> fSampledProxies;
Chris Dalton16a33c62019-09-24 22:19:17 -0600343
344 SkRect fTotalBounds = SkRect::MakeEmpty();
Greg Daniel94ed83f2019-09-27 13:05:43 -0400345 SkIRect fClippedContentBounds = SkIRect::MakeEmpty();
joshualitt6db519c2014-10-29 08:48:18 -0700346};
347
reed@google.comac10a2d2010-12-22 21:39:39 +0000348#endif