blob: 50ebcd7e698b2f22056e7afeef07faeead71d262 [file] [log] [blame]
robertphillips76948d42016-05-04 12:47:41 -07001/*
2 * Copyright 2016 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 GrRenderTargetProxy_DEFINED
9#define GrRenderTargetProxy_DEFINED
10
Mike Kleinc0bd9f92019-04-23 12:05:21 -050011#include "include/private/GrTypesPriv.h"
Chris Daltoneffee202019-07-01 22:28:03 -060012#include "src/gpu/GrCaps.h"
Greg Danielf91aeb22019-06-18 09:58:02 -040013#include "src/gpu/GrSurfaceProxy.h"
Greg Daniel2c19e7f2019-06-18 13:29:21 -040014#include "src/gpu/GrSwizzle.h"
robertphillips76948d42016-05-04 12:47:41 -070015
Brian Osman32342f02017-03-04 08:12:46 -050016class GrResourceProvider;
Greg Daniela070ed72018-04-26 16:31:38 -040017class GrRenderTargetProxyPriv;
robertphillips76948d42016-05-04 12:47:41 -070018
19// This class delays the acquisition of RenderTargets until they are actually
20// required
21// Beware: the uniqueID of the RenderTargetProxy will usually be different than
22// the uniqueID of the RenderTarget it represents!
Robert Phillips84a81202016-11-04 11:59:10 -040023class GrRenderTargetProxy : virtual public GrSurfaceProxy {
robertphillips76948d42016-05-04 12:47:41 -070024public:
robertphillips76948d42016-05-04 12:47:41 -070025 GrRenderTargetProxy* asRenderTargetProxy() override { return this; }
26 const GrRenderTargetProxy* asRenderTargetProxy() const override { return this; }
27
28 // Actually instantiate the backing rendertarget, if necessary.
Robert Phillips10d17212019-04-24 14:09:10 -040029 bool instantiate(GrResourceProvider*) override;
robertphillips76948d42016-05-04 12:47:41 -070030
Chris Daltoneffee202019-07-01 22:28:03 -060031 bool canUseMixedSamples(const GrCaps& caps) const {
Chris Daltoneffee202019-07-01 22:28:03 -060032 return caps.mixedSamplesSupport() && !this->glRTFBOIDIs0() &&
Greg Danieleadfac92019-08-02 09:03:53 -040033 caps.internalMultisampleCount(this->backendFormat()) > 0 &&
Chris Dalton215ff332019-07-02 09:38:22 -060034 this->canChangeStencilAttachment();
Chris Daltoneffee202019-07-01 22:28:03 -060035 }
36
Robert Phillips65048132017-08-10 08:44:49 -040037 /*
Chris Daltoneffee202019-07-01 22:28:03 -060038 * Indicate that a draw to this proxy requires stencil, and how many stencil samples it needs.
39 * The number of stencil samples on this proxy will be equal to the largest sample count passed
40 * to this method.
Robert Phillips65048132017-08-10 08:44:49 -040041 */
Chris Daltoneffee202019-07-01 22:28:03 -060042 void setNeedsStencil(int8_t numStencilSamples) {
43 SkASSERT(numStencilSamples >= fSampleCnt);
44 fNumStencilSamples = SkTMax(numStencilSamples, fNumStencilSamples);
45 }
46
47 /**
48 * Returns the number of stencil samples required by this proxy.
49 * NOTE: Once instantiated, the actual render target may have more samples, but it is guaranteed
50 * to have at least this many. (After a multisample stencil buffer has been attached to a render
51 * target, we never "downgrade" it to one with fewer samples.)
52 */
53 int numStencilSamples() const { return fNumStencilSamples; }
Robert Phillips65048132017-08-10 08:44:49 -040054
Brian Salomonbb5711a2017-05-17 13:49:59 -040055 /**
Chris Dalton6ce447a2019-06-23 18:07:38 -060056 * Returns the number of samples/pixel in the color buffer (One if non-MSAA).
Brian Salomonbb5711a2017-05-17 13:49:59 -040057 */
Chris Dalton6ce447a2019-06-23 18:07:38 -060058 int numSamples() const { return fSampleCnt; }
robertphillips76948d42016-05-04 12:47:41 -070059
Robert Phillipsec2249f2016-11-09 08:54:35 -050060 int maxWindowRectangles(const GrCaps& caps) const;
61
Greg Daniel2c19e7f2019-06-18 13:29:21 -040062 const GrSwizzle& outputSwizzle() const { return fOutputSwizzle; }
63
Greg Danielb46add82019-01-02 14:51:29 -050064 bool wrapsVkSecondaryCB() const { return fWrapsVkSecondaryCB == WrapsVkSecondaryCB::kYes; }
65
Chris Dalton4ece96d2019-08-30 11:26:39 -060066 void markMSAADirty() {
67 SkASSERT(this->requiresManualMSAAResolve());
68 fIsMSAADirty = true;
69 }
70 void markMSAAResolved() {
71 SkASSERT(this->requiresManualMSAAResolve());
72 fIsMSAADirty = false;
73 }
74 bool isMSAADirty() const {
75 SkASSERT(!fIsMSAADirty || this->requiresManualMSAAResolve());
76 return fIsMSAADirty;
77 }
78
Robert Phillipse2f7d182016-12-15 09:23:05 -050079 // TODO: move this to a priv class!
80 bool refsWrappedObjects() const;
81
Greg Daniela070ed72018-04-26 16:31:38 -040082 // Provides access to special purpose functions.
83 GrRenderTargetProxyPriv rtPriv();
84 const GrRenderTargetProxyPriv rtPriv() const;
85
Robert Phillips84a81202016-11-04 11:59:10 -040086protected:
Robert Phillips0bd24dc2018-01-16 08:06:32 -050087 friend class GrProxyProvider; // for ctors
Greg Daniela070ed72018-04-26 16:31:38 -040088 friend class GrRenderTargetProxyPriv;
Robert Phillips37430132016-11-09 06:50:43 -050089
csmartdaltonf9635992016-08-10 11:09:07 -070090 // Deferred version
Brian Salomonbeb7f522019-08-30 16:19:42 -040091 GrRenderTargetProxy(const GrCaps&,
92 const GrBackendFormat&,
93 const GrSurfaceDesc&,
94 int sampleCount,
95 GrSurfaceOrigin,
96 const GrSwizzle& textureSwizzle,
97 const GrSwizzle& outputSwizzle,
98 SkBackingFit,
99 SkBudgeted,
100 GrProtected,
101 GrInternalSurfaceFlags,
102 UseAllocator);
robertphillips76948d42016-05-04 12:47:41 -0700103
Greg Danielb085fa92019-03-05 16:55:12 -0500104 enum class WrapsVkSecondaryCB : bool { kNo = false, kYes = true };
105
Chris Dalton706a6ff2017-11-29 22:01:06 -0700106 // Lazy-callback version
Greg Daniel65fa8ca2018-01-10 17:06:31 -0500107 // There are two main use cases for lazily-instantiated proxies:
108 // basic knowledge - width, height, config, samples, origin are known
109 // minimal knowledge - only config is known.
110 //
111 // The basic knowledge version is used for DDL where we know the type of proxy we are going to
112 // use, but we don't have access to the GPU yet to instantiate it.
113 //
114 // The minimal knowledge version is used for CCPR where we are generating an atlas but we do not
115 // know the final size until flush time.
Brian Salomonbeb7f522019-08-30 16:19:42 -0400116 GrRenderTargetProxy(LazyInstantiateCallback&&,
117 const GrBackendFormat&,
118 const GrSurfaceDesc&,
119 int sampleCount,
120 GrSurfaceOrigin,
121 const GrSwizzle& textureSwizzle,
122 const GrSwizzle& outputSwizzle,
123 SkBackingFit,
124 SkBudgeted,
125 GrProtected,
126 GrInternalSurfaceFlags,
127 UseAllocator,
128 WrapsVkSecondaryCB);
Chris Dalton706a6ff2017-11-29 22:01:06 -0700129
robertphillips76948d42016-05-04 12:47:41 -0700130 // Wrapped version
Brian Salomonbeb7f522019-08-30 16:19:42 -0400131 GrRenderTargetProxy(sk_sp<GrSurface>,
132 GrSurfaceOrigin,
133 const GrSwizzle& textureSwizzle,
Greg Daniel2c19e7f2019-06-18 13:29:21 -0400134 const GrSwizzle& outputSwizzle,
Brian Salomonbeb7f522019-08-30 16:19:42 -0400135 UseAllocator,
136 WrapsVkSecondaryCB = WrapsVkSecondaryCB::kNo);
robertphillips76948d42016-05-04 12:47:41 -0700137
Robert Phillips5af44de2017-07-18 14:49:38 -0400138 sk_sp<GrSurface> createSurface(GrResourceProvider*) const override;
139
Robert Phillips84a81202016-11-04 11:59:10 -0400140private:
Greg Daniela070ed72018-04-26 16:31:38 -0400141 void setGLRTFBOIDIs0() {
142 fSurfaceFlags |= GrInternalSurfaceFlags::kGLRTFBOIDIs0;
143 }
144 bool glRTFBOIDIs0() const {
145 return fSurfaceFlags & GrInternalSurfaceFlags::kGLRTFBOIDIs0;
146 }
Chris Dalton215ff332019-07-02 09:38:22 -0600147 bool canChangeStencilAttachment() const;
Greg Daniela070ed72018-04-26 16:31:38 -0400148
Brian Salomonbb5711a2017-05-17 13:49:59 -0400149 size_t onUninstantiatedGpuMemorySize() const override;
Greg Daniel849dce12018-04-24 14:32:53 -0400150 SkDEBUGCODE(void onValidateSurface(const GrSurface*) override;)
Robert Phillips8bc06d02016-11-01 17:28:40 -0400151
Brian Osman086679b2018-09-10 16:26:51 -0400152 // WARNING: Be careful when adding or removing fields here. ASAN is likely to trigger warnings
153 // when instantiating GrTextureRenderTargetProxy. The std::function in GrSurfaceProxy makes
154 // each class in the diamond require 16 byte alignment. Clang appears to layout the fields for
155 // each class to achieve the necessary alignment. However, ASAN checks the alignment of 'this'
156 // in the constructors, and always looks for the full 16 byte alignment, even if the fields in
157 // that particular class don't require it. Changing the size of this object can move the start
158 // address of other types, leading to this problem.
Chris Daltoneffee202019-07-01 22:28:03 -0600159 int8_t fSampleCnt;
160 int8_t fNumStencilSamples = 0;
Greg Danielb46add82019-01-02 14:51:29 -0500161 WrapsVkSecondaryCB fWrapsVkSecondaryCB;
Chris Daltoneffee202019-07-01 22:28:03 -0600162 GrSwizzle fOutputSwizzle;
Chris Dalton4ece96d2019-08-30 11:26:39 -0600163 // Indicates whether some sub-rectangle of the render target requires MSAA resolve. We currently
164 // rely on the GrRenderTarget itself to track the actual dirty rect.
165 // TODO: In the future, convert the flag to a dirty rect and quit tracking it in GrRenderTarget.
166 bool fIsMSAADirty = false;
Greg Daniel2c19e7f2019-06-18 13:29:21 -0400167 // This is to fix issue in large comment above. Without the padding we end 6 bytes into a 16
168 // byte range, so the GrTextureProxy ends up starting 8 byte aligned by not 16. We add the
169 // padding here to get us right up to the 16 byte alignment (technically any padding of 3-10
170 // bytes would work since it always goes up to 8 byte alignment, but we use 10 to more explicit
171 // about what we're doing).
172 char fDummyPadding[10];
Robert Phillips65048132017-08-10 08:44:49 -0400173
robertphillips76948d42016-05-04 12:47:41 -0700174 typedef GrSurfaceProxy INHERITED;
175};
176
177#endif