blob: 90fa46f2fca88c52216cec52d761b8c5e87aef4c [file] [log] [blame]
robertphillips77a2e522015-10-17 07:43:27 -07001/*
2 * Copyright 2015 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#ifndef GrDrawingManager_DEFINED
9#define GrDrawingManager_DEFINED
10
Robert Phillips15c91422019-05-07 16:54:48 -040011#include <set>
Mike Kleinc0bd9f92019-04-23 12:05:21 -050012#include "include/core/SkSurface.h"
13#include "include/private/SkTArray.h"
Adlai Hollerd71b7b02020-06-08 15:55:00 -040014#include "include/private/SkTHash.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050015#include "src/gpu/GrBufferAllocPool.h"
16#include "src/gpu/GrDeferredUpload.h"
17#include "src/gpu/GrPathRenderer.h"
18#include "src/gpu/GrPathRendererChain.h"
19#include "src/gpu/GrResourceCache.h"
20#include "src/gpu/text/GrTextContext.h"
Robert Phillipsf2361d22016-10-25 14:20:06 -040021
Robert Phillipsd81379d2020-04-21 10:39:02 -040022// Enabling this will print out which path renderers are being chosen
23#define GR_PATH_RENDERER_SPEW 0
24
Chris Daltonfddb6c02017-11-04 15:22:22 -060025class GrCoverageCountingPathRenderer;
Robert Phillipsfbcef6e2017-06-15 12:07:18 -040026class GrOnFlushCallbackObject;
Herb Derbydc214c22018-11-08 13:31:39 -050027class GrOpFlushState;
Greg Danielf41b2bd2019-08-22 16:19:24 -040028class GrOpsTask;
Robert Phillips69893702019-02-22 11:16:30 -050029class GrRecordingContext;
Brian Osman11052242016-10-27 14:47:55 -040030class GrRenderTargetContext;
Robert Phillipsc7635fa2016-10-28 13:25:24 -040031class GrRenderTargetProxy;
robertphillips68737822015-10-29 12:12:21 -070032class GrSoftwarePathRenderer;
Greg Daniel46e366a2019-12-16 14:38:36 -050033class GrSurfaceContext;
Greg Daniel16f5c652019-10-29 11:26:01 -040034class GrSurfaceProxyView;
Chris Daltone2a903e2019-09-18 13:41:50 -060035class GrTextureResolveRenderTask;
Brian Salomon653f42f2018-07-10 10:07:31 -040036class SkDeferredDisplayList;
robertphillips77a2e522015-10-17 07:43:27 -070037
robertphillips77a2e522015-10-17 07:43:27 -070038class GrDrawingManager {
39public:
40 ~GrDrawingManager();
41
robertphillips68737822015-10-29 12:12:21 -070042 void freeGpuResources();
robertphillips77a2e522015-10-17 07:43:27 -070043
Greg Danielf41b2bd2019-08-22 16:19:24 -040044 // A managed opsTask is controlled by the drawing manager (i.e., sorted & flushed with the
Robert Phillips831a2932019-04-12 17:18:39 -040045 // others). An unmanaged one is created and used by the onFlushCallback.
Greg Daniel16f5c652019-10-29 11:26:01 -040046 sk_sp<GrOpsTask> newOpsTask(GrSurfaceProxyView, bool managedOpsTask);
robertphillips77a2e522015-10-17 07:43:27 -070047
Chris Daltone2a903e2019-09-18 13:41:50 -060048 // Create a render task that can resolve MSAA and/or regenerate mipmap levels on proxies. This
49 // method will only add the new render task to the list. It is up to the caller to call
50 // addProxy() on the returned object.
51 GrTextureResolveRenderTask* newTextureResolveRenderTask(const GrCaps&);
Chris Dalton3d770272019-08-14 09:24:37 -060052
Greg Danielc30f1a92019-09-06 15:28:58 -040053 // Create a new render task that will cause the gpu to wait on semaphores before executing any
54 // more RenderTasks that target proxy. It is possible for this wait to also block additional
55 // work (even to other proxies) that has already been recorded or will be recorded later. The
56 // only guarantee is that future work to the passed in proxy will wait on the semaphores to be
57 // signaled.
Greg Daniel301015c2019-11-18 14:06:46 -050058 void newWaitRenderTask(sk_sp<GrSurfaceProxy> proxy,
59 std::unique_ptr<std::unique_ptr<GrSemaphore>[]>,
Greg Danielc30f1a92019-09-06 15:28:58 -040060 int numSemaphores);
61
Greg Danielbbfec9d2019-08-20 10:56:51 -040062 // Create a new render task which copies the pixels from the srcProxy into the dstBuffer. This
63 // is used to support the asynchronous readback API. The srcRect is the region of the srcProxy
64 // to be copied. The surfaceColorType says how we should interpret the data when reading back
65 // from the source. DstColorType describes how the data should be stored in the dstBuffer.
66 // DstOffset is the offset into the dstBuffer where we will start writing data.
67 void newTransferFromRenderTask(sk_sp<GrSurfaceProxy> srcProxy, const SkIRect& srcRect,
68 GrColorType surfaceColorType, GrColorType dstColorType,
69 sk_sp<GrGpuBuffer> dstBuffer, size_t dstOffset);
70
Greg Daniel16f5c652019-10-29 11:26:01 -040071 // Creates a new render task which copies a pixel rectangle from srcView into dstView. The src
Greg Daniele227fe42019-08-21 13:52:24 -040072 // pixels copied are specified by srcRect. They are copied to a rect of the same size in
73 // dstProxy with top left at dstPoint. If the src rect is clipped by the src bounds then pixel
74 // values in the dst rect corresponding to the area clipped by the src rect are not overwritten.
75 // This method is not guaranteed to succeed depending on the type of surface, formats, etc, and
76 // the backend-specific limitations.
Greg Daniel16f5c652019-10-29 11:26:01 -040077 bool newCopyRenderTask(GrSurfaceProxyView srcView, const SkIRect& srcRect,
78 GrSurfaceProxyView dstView, const SkIPoint& dstPoint);
Greg Daniele227fe42019-08-21 13:52:24 -040079
Robert Phillips69893702019-02-22 11:16:30 -050080 GrRecordingContext* getContext() { return fContext; }
robertphillips77a2e522015-10-17 07:43:27 -070081
Herb Derby26cbe512018-05-24 14:39:01 -040082 GrTextContext* getTextContext();
brianosman86e76262016-08-11 12:17:31 -070083
robertphillips68737822015-10-29 12:12:21 -070084 GrPathRenderer* getPathRenderer(const GrPathRenderer::CanDrawPathArgs& args,
85 bool allowSW,
86 GrPathRendererChain::DrawType drawType,
Ben Wagnera93a14a2017-08-28 10:34:05 -040087 GrPathRenderer::StencilSupport* stencilSupport = nullptr);
robertphillips68737822015-10-29 12:12:21 -070088
Brian Salomone7df0bb2018-05-07 14:44:57 -040089 GrPathRenderer* getSoftwarePathRenderer();
90
Chris Daltonfddb6c02017-11-04 15:22:22 -060091 // Returns a direct pointer to the coverage counting path renderer, or null if it is not
92 // supported and turned on.
93 GrCoverageCountingPathRenderer* getCoverageCountingPathRenderer();
94
Brian Salomon653f42f2018-07-10 10:07:31 -040095 void flushIfNecessary();
bsalomonb77a9072016-09-07 10:02:04 -070096
Greg Daniel78325c12017-06-19 16:39:13 -040097 static bool ProgramUnitTest(GrContext* context, int maxStages, int maxLevels);
robertphillipsa13e2022015-11-11 12:01:09 -080098
Brian Salomonf9a1fdf2019-05-09 10:30:12 -040099 GrSemaphoresSubmitted flushSurfaces(GrSurfaceProxy* proxies[],
100 int cnt,
101 SkSurface::BackendSurfaceAccess access,
102 const GrFlushInfo& info);
Greg Daniel55f040b2020-02-13 15:38:32 +0000103 GrSemaphoresSubmitted flushSurface(GrSurfaceProxy* proxy,
Greg Daniel4aa13e72019-04-15 14:42:20 -0400104 SkSurface::BackendSurfaceAccess access,
Brian Salomonf9a1fdf2019-05-09 10:30:12 -0400105 const GrFlushInfo& info) {
Greg Daniel55f040b2020-02-13 15:38:32 +0000106 return this->flushSurfaces(&proxy, 1, access, info);
Brian Salomonf9a1fdf2019-05-09 10:30:12 -0400107 }
bsalomon6a2b1942016-09-08 11:28:59 -0700108
Chris Daltonfe199b72017-05-05 11:26:15 -0400109 void addOnFlushCallbackObject(GrOnFlushCallbackObject*);
Robert Phillipsdbaf3172019-02-06 15:12:53 -0500110
111#if GR_TEST_UTILS
Chris Daltonfe199b72017-05-05 11:26:15 -0400112 void testingOnly_removeOnFlushCallbackObject(GrOnFlushCallbackObject*);
Robert Phillipsdbaf3172019-02-06 15:12:53 -0500113#endif
Robert Phillipseb35f4d2017-03-21 07:56:47 -0400114
Adlai Hollerd71b7b02020-06-08 15:55:00 -0400115 GrRenderTask* getLastRenderTask(const GrSurfaceProxy*) const;
116 GrOpsTask* getLastOpsTask(const GrSurfaceProxy*) const;
117 void setLastRenderTask(const GrSurfaceProxy*, GrRenderTask*);
118
Chris Dalton6b498102019-08-01 14:14:52 -0600119 void moveRenderTasksToDDL(SkDeferredDisplayList* ddl);
120 void copyRenderTasksFromDDL(const SkDeferredDisplayList*, GrRenderTargetProxy* newDest);
Robert Phillips62000362018-02-01 09:10:04 -0500121
robertphillips77a2e522015-10-17 07:43:27 -0700122private:
Chris Dalton6b498102019-08-01 14:14:52 -0600123 // This class encapsulates maintenance and manipulation of the drawing manager's DAG of
124 // renderTasks.
125 class RenderTaskDAG {
Robert Phillips22310d62018-09-05 11:07:21 -0400126 public:
Chris Dalton6b498102019-08-01 14:14:52 -0600127 RenderTaskDAG(bool sortRenderTasks);
128 ~RenderTaskDAG();
Robert Phillips22310d62018-09-05 11:07:21 -0400129
130 // Currently, when explicitly allocating resources, this call will topologically sort the
Greg Danielf41b2bd2019-08-22 16:19:24 -0400131 // GrRenderTasks.
132 // MDB TODO: remove once incremental GrRenderTask sorting is enabled
Robert Phillips22310d62018-09-05 11:07:21 -0400133 void prepForFlush();
134
135 void closeAll(const GrCaps* caps);
136
137 // A yucky combination of closeAll and reset
Adlai Hollerd71b7b02020-06-08 15:55:00 -0400138 void cleanup(GrDrawingManager*, const GrCaps* caps);
Robert Phillips22310d62018-09-05 11:07:21 -0400139
140 void gatherIDs(SkSTArray<8, uint32_t, true>* idArray) const;
141
142 void reset();
143
Greg Danielf41b2bd2019-08-22 16:19:24 -0400144 // These calls forceably remove a GrRenderTask from the DAG. They are problematic bc they
145 // just remove the GrRenderTask but don't cleanup any refering pointers (i.e., dependency
146 // pointers in the DAG). They work right now bc they are only called at flush time, after
147 // the topological sort is complete (so the dangling pointers aren't used).
Chris Dalton6b498102019-08-01 14:14:52 -0600148 void removeRenderTask(int index);
149 void removeRenderTasks(int startIndex, int stopIndex);
Robert Phillips22310d62018-09-05 11:07:21 -0400150
Chris Dalton6b498102019-08-01 14:14:52 -0600151 bool empty() const { return fRenderTasks.empty(); }
152 int numRenderTasks() const { return fRenderTasks.count(); }
Robert Phillips22310d62018-09-05 11:07:21 -0400153
Robert Phillips9313aa72019-04-09 18:41:27 -0400154 bool isUsed(GrSurfaceProxy*) const;
155
Chris Dalton6b498102019-08-01 14:14:52 -0600156 GrRenderTask* renderTask(int index) { return fRenderTasks[index].get(); }
157 const GrRenderTask* renderTask(int index) const { return fRenderTasks[index].get(); }
Robert Phillips22310d62018-09-05 11:07:21 -0400158
Chris Dalton6b498102019-08-01 14:14:52 -0600159 GrRenderTask* back() { return fRenderTasks.back().get(); }
160 const GrRenderTask* back() const { return fRenderTasks.back().get(); }
Robert Phillips22310d62018-09-05 11:07:21 -0400161
Chris Dalton3d770272019-08-14 09:24:37 -0600162 GrRenderTask* add(sk_sp<GrRenderTask>);
163 GrRenderTask* addBeforeLast(sk_sp<GrRenderTask>);
Chris Dalton6b498102019-08-01 14:14:52 -0600164 void add(const SkTArray<sk_sp<GrRenderTask>>&);
Robert Phillips22310d62018-09-05 11:07:21 -0400165
Chris Dalton6b498102019-08-01 14:14:52 -0600166 void swap(SkTArray<sk_sp<GrRenderTask>>* renderTasks);
Robert Phillips22310d62018-09-05 11:07:21 -0400167
Chris Dalton6b498102019-08-01 14:14:52 -0600168 bool sortingRenderTasks() const { return fSortRenderTasks; }
Robert Phillips46acf9d2018-10-09 09:31:40 -0400169
Robert Phillips22310d62018-09-05 11:07:21 -0400170 private:
Chris Dalton6b498102019-08-01 14:14:52 -0600171 SkTArray<sk_sp<GrRenderTask>> fRenderTasks;
172 bool fSortRenderTasks;
Robert Phillips22310d62018-09-05 11:07:21 -0400173 };
174
Robert Phillips69893702019-02-22 11:16:30 -0500175 GrDrawingManager(GrRecordingContext*, const GrPathRendererChain::Options&,
Robert Phillips0d075de2019-03-04 11:08:13 -0500176 const GrTextContext::Options&,
Chris Dalton6b498102019-08-01 14:14:52 -0600177 bool sortRenderTasks,
Greg Danielf41b2bd2019-08-22 16:19:24 -0400178 bool reduceOpsTaskSplitting);
robertphillips77a2e522015-10-17 07:43:27 -0700179
Robert Phillipsa9162df2019-02-11 14:12:03 -0500180 bool wasAbandoned() const;
181
robertphillips77a2e522015-10-17 07:43:27 -0700182 void cleanup();
Robert Phillipseafd48a2017-11-16 07:52:08 -0500183
Greg Danielf41b2bd2019-08-22 16:19:24 -0400184 // Closes the target's dependent render tasks (or, if not in sorting/opsTask-splitting-reduction
185 // mode, closes fActiveOpsTask) in preparation for us opening a new opsTask that will write to
Chris Dalton5fe99772019-08-06 11:57:39 -0600186 // 'target'.
Greg Danielbbfec9d2019-08-20 10:56:51 -0400187 void closeRenderTasksForNewRenderTask(GrSurfaceProxy* target);
Chris Dalton5fe99772019-08-06 11:57:39 -0600188
Greg Danielf41b2bd2019-08-22 16:19:24 -0400189 // return true if any GrRenderTasks were actually executed; false otherwise
Chris Dalton6b498102019-08-01 14:14:52 -0600190 bool executeRenderTasks(int startIndex, int stopIndex, GrOpFlushState*,
191 int* numRenderTasksExecuted);
Robert Phillipseafd48a2017-11-16 07:52:08 -0500192
Adlai Hollerd71b7b02020-06-08 15:55:00 -0400193 void removeRenderTasks(int startIndex, int stopIndex);
194
Greg Danielfe159622020-04-10 17:43:51 +0000195 bool flush(GrSurfaceProxy* proxies[],
196 int numProxies,
197 SkSurface::BackendSurfaceAccess access,
198 const GrFlushInfo&,
199 const GrPrepareForExternalIORequests&);
200
201 bool submitToGpu(bool syncToCpu);
robertphillips77a2e522015-10-17 07:43:27 -0700202
Robert Phillips38d64b02018-09-04 13:23:26 -0400203 SkDEBUGCODE(void validate() const);
204
Robert Phillips69893702019-02-22 11:16:30 -0500205 friend class GrContext; // access to: flush & cleanup
Robert Phillips7ee385e2017-03-30 08:02:11 -0400206 friend class GrContextPriv; // access to: flush
Chris Daltonfe199b72017-05-05 11:26:15 -0400207 friend class GrOnFlushResourceProvider; // this is just a shallow wrapper around this class
Robert Phillips69893702019-02-22 11:16:30 -0500208 friend class GrRecordingContext; // access to: ctor
Greg Danielb6c15ba2019-03-04 13:08:25 -0500209 friend class SkImage; // for access to: flush
robertphillips77a2e522015-10-17 07:43:27 -0700210
211 static const int kNumPixelGeometries = 5; // The different pixel geometries
212 static const int kNumDFTOptions = 2; // DFT or no DFT
213
Robert Phillips69893702019-02-22 11:16:30 -0500214 GrRecordingContext* fContext;
bsalomon6b2552f2016-09-15 13:50:26 -0700215 GrPathRendererChain::Options fOptionsForPathRendererChain;
Herb Derby26cbe512018-05-24 14:39:01 -0400216 GrTextContext::Options fOptionsForTextContext;
Brian Salomon601ac802019-02-07 13:37:16 -0500217 // This cache is used by both the vertex and index pools. It reuses memory across multiple
218 // flushes.
219 sk_sp<GrBufferAllocPool::CpuBufferCache> fCpuBufferCache;
joshualittde8dc7e2016-01-08 10:09:13 -0800220
Chris Dalton6b498102019-08-01 14:14:52 -0600221 RenderTaskDAG fDAG;
Greg Danielf41b2bd2019-08-22 16:19:24 -0400222 GrOpsTask* fActiveOpsTask = nullptr;
223 // These are the IDs of the opsTask currently being flushed (in internalFlush)
Chris Dalton6b498102019-08-01 14:14:52 -0600224 SkSTArray<8, uint32_t, true> fFlushingRenderTaskIDs;
Chris Daltonc4b47352019-08-23 10:10:36 -0600225 // These are the new renderTasks generated by the onFlush CBs
226 SkSTArray<4, sk_sp<GrRenderTask>> fOnFlushRenderTasks;
robertphillips77a2e522015-10-17 07:43:27 -0700227
Herb Derby26cbe512018-05-24 14:39:01 -0400228 std::unique_ptr<GrTextContext> fTextContext;
robertphillipsa13e2022015-11-11 12:01:09 -0800229
Ben Wagner9ec70c62018-07-12 13:30:47 -0400230 std::unique_ptr<GrPathRendererChain> fPathRendererChain;
231 sk_sp<GrSoftwarePathRenderer> fSoftwarePathRenderer;
brianosman86e76262016-08-11 12:17:31 -0700232
Robert Phillips40a29d72018-01-18 12:59:22 -0500233 GrTokenTracker fTokenTracker;
brianosman86e76262016-08-11 12:17:31 -0700234 bool fFlushing;
Greg Danielf41b2bd2019-08-22 16:19:24 -0400235 bool fReduceOpsTaskSplitting;
bsalomonb77a9072016-09-07 10:02:04 -0700236
Chris Daltonfe199b72017-05-05 11:26:15 -0400237 SkTArray<GrOnFlushCallbackObject*> fOnFlushCBObjects;
Robert Phillips15c91422019-05-07 16:54:48 -0400238
239 void addDDLTarget(GrSurfaceProxy* proxy) { fDDLTargets.insert(proxy); }
240 bool isDDLTarget(GrSurfaceProxy* proxy) { return fDDLTargets.find(proxy) != fDDLTargets.end(); }
241 void clearDDLTargets() { fDDLTargets.clear(); }
242
243 // We play a trick with lazy proxies to retarget the base target of a DDL to the SkSurface
244 // it is replayed on. Because of this remapping we need to explicitly store the targets of
245 // DDL replaying.
246 // Note: we do not expect a whole lot of these per flush
247 std::set<GrSurfaceProxy*> fDDLTargets;
Adlai Hollerd71b7b02020-06-08 15:55:00 -0400248
249 // Keys are UniqueID of GrSurfaceProxys.
250 SkTHashMap<uint32_t, GrRenderTask*> fLastRenderTasks;
robertphillips77a2e522015-10-17 07:43:27 -0700251};
252
253#endif