blob: 45efba2f175789631e6e48c7e85a04f6d3c6dbcd [file] [log] [blame]
bsalomon@google.comaa5b6732011-07-29 15:13:20 +00001/*
2 * Copyright 2011 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
bsalomon@google.comaa5b6732011-07-29 15:13:20 +00008#include "GrGLRenderTarget.h"
9
jvanverth39edf762014-12-22 11:44:19 -080010#include "GrGLGpu.h"
egdanielec00d942015-09-14 12:56:10 -070011#include "GrGLUtil.h"
kkinnunen2e6055b2016-04-22 01:48:29 -070012#include "GrGpuResourcePriv.h"
13#include "GrRenderTargetPriv.h"
ericrk0a5fa482015-09-15 14:16:10 -070014#include "SkTraceMemoryDump.h"
bsalomon@google.comaa5b6732011-07-29 15:13:20 +000015
egdanield803f272015-03-18 13:01:52 -070016#define GPUGL static_cast<GrGLGpu*>(this->getGpu())
17#define GL_CALL(X) GR_GL_CALL(GPUGL->glInterface(), X)
bsalomon@google.com0b77d682011-08-19 13:28:54 +000018
bsalomon37dd3312014-11-03 08:47:23 -080019// Because this class is virtually derived from GrSurface we must explicitly call its constructor.
kkinnunen2e6055b2016-04-22 01:48:29 -070020// Constructor for wrapped render targets.
egdanielec00d942015-09-14 12:56:10 -070021GrGLRenderTarget::GrGLRenderTarget(GrGLGpu* gpu,
22 const GrSurfaceDesc& desc,
23 const IDDesc& idDesc,
24 GrGLStencilAttachment* stencil)
kkinnunen2e6055b2016-04-22 01:48:29 -070025 : GrSurface(gpu, desc)
csmartdaltonf9635992016-08-10 11:09:07 -070026 , INHERITED(gpu, desc, ComputeFlags(gpu->glCaps(), idDesc), stencil) {
bsalomon37dd3312014-11-03 08:47:23 -080027 this->init(desc, idDesc);
kkinnunen2e6055b2016-04-22 01:48:29 -070028 this->registerWithCacheWrapped();
bsalomon@google.comaa5b6732011-07-29 15:13:20 +000029}
30
kkinnunen2e6055b2016-04-22 01:48:29 -070031GrGLRenderTarget::GrGLRenderTarget(GrGLGpu* gpu, const GrSurfaceDesc& desc,
32 const IDDesc& idDesc)
33 : GrSurface(gpu, desc)
csmartdaltonf9635992016-08-10 11:09:07 -070034 , INHERITED(gpu, desc, ComputeFlags(gpu->glCaps(), idDesc)) {
bsalomon37dd3312014-11-03 08:47:23 -080035 this->init(desc, idDesc);
bsalomon@google.com5bfc2172011-07-29 20:29:05 +000036}
37
csmartdaltonf9635992016-08-10 11:09:07 -070038inline GrRenderTarget::Flags GrGLRenderTarget::ComputeFlags(const GrGLCaps& glCaps,
39 const IDDesc& idDesc) {
40 Flags flags = Flags::kNone;
41 if (idDesc.fIsMixedSampled) {
42 SkASSERT(glCaps.usesMixedSamples() && idDesc.fRTFBOID); // FBO 0 can't be mixed sampled.
43 flags |= Flags::kMixedSampled;
44 }
45 if (glCaps.maxWindowRectangles() > 0 && idDesc.fRTFBOID) {
46 flags |= Flags::kWindowRectsSupport;
47 }
48 return flags;
49}
50
bsalomon37dd3312014-11-03 08:47:23 -080051void GrGLRenderTarget::init(const GrSurfaceDesc& desc, const IDDesc& idDesc) {
cdaltond4727922015-11-10 12:49:06 -080052 fRTFBOID = idDesc.fRTFBOID;
53 fTexFBOID = idDesc.fTexFBOID;
54 fMSColorRenderbufferID = idDesc.fMSColorRenderbufferID;
kkinnunen2e6055b2016-04-22 01:48:29 -070055 fRTFBOOwnership = idDesc.fRTFBOOwnership;
bsalomon37dd3312014-11-03 08:47:23 -080056
57 fViewport.fLeft = 0;
58 fViewport.fBottom = 0;
59 fViewport.fWidth = desc.fWidth;
60 fViewport.fHeight = desc.fHeight;
61
Robert Phillips29e52f12016-11-03 10:19:14 -040062 fNumSamplesOwnedPerPixel = this->totalSamples();
senorblancod2981212015-04-30 12:06:10 -070063}
64
bungeman6bd52842016-10-27 09:30:08 -070065sk_sp<GrGLRenderTarget> GrGLRenderTarget::MakeWrapped(GrGLGpu* gpu,
66 const GrSurfaceDesc& desc,
67 const IDDesc& idDesc,
68 int stencilBits) {
egdanielec00d942015-09-14 12:56:10 -070069 GrGLStencilAttachment* sb = nullptr;
70 if (stencilBits) {
71 GrGLStencilAttachment::IDDesc sbDesc;
72 GrGLStencilAttachment::Format format;
73 format.fInternalFormat = GrGLStencilAttachment::kUnknownInternalFormat;
74 format.fPacked = false;
75 format.fStencilBits = stencilBits;
76 format.fTotalBits = stencilBits;
77 // Owndership of sb is passed to the GrRenderTarget so doesn't need to be deleted
78 sb = new GrGLStencilAttachment(gpu, sbDesc, desc.fWidth, desc.fHeight,
79 desc.fSampleCnt, format);
80 }
bungeman6bd52842016-10-27 09:30:08 -070081 return sk_sp<GrGLRenderTarget>(new GrGLRenderTarget(gpu, desc, idDesc, sb));
egdanielec00d942015-09-14 12:56:10 -070082}
83
senorblancod2981212015-04-30 12:06:10 -070084size_t GrGLRenderTarget::onGpuMemorySize() const {
Robert Phillipsd6214d42016-11-07 08:23:48 -050085 return GrSurface::ComputeSize(fDesc, fNumSamplesOwnedPerPixel, false);
bsalomon@google.com5bfc2172011-07-29 20:29:05 +000086}
87
egdanielec00d942015-09-14 12:56:10 -070088bool GrGLRenderTarget::completeStencilAttachment() {
89 GrGLGpu* gpu = this->getGLGpu();
90 const GrGLInterface* interface = gpu->glInterface();
91 GrStencilAttachment* stencil = this->renderTargetPriv().getStencilAttachment();
92 if (nullptr == stencil) {
egdaniel79bd2ae2015-09-15 08:46:13 -070093 GR_GL_CALL(interface, FramebufferRenderbuffer(GR_GL_FRAMEBUFFER,
94 GR_GL_STENCIL_ATTACHMENT,
95 GR_GL_RENDERBUFFER, 0));
96 GR_GL_CALL(interface, FramebufferRenderbuffer(GR_GL_FRAMEBUFFER,
97 GR_GL_DEPTH_ATTACHMENT,
98 GR_GL_RENDERBUFFER, 0));
egdanielec00d942015-09-14 12:56:10 -070099#ifdef SK_DEBUG
robertphillipsfa7ff472016-04-21 11:27:43 -0700100 if (kChromium_GrGLDriver != gpu->glContext().driver()) {
101 // This check can cause problems in Chromium if the context has been asynchronously
102 // abandoned (see skbug.com/5200)
103 GrGLenum status;
104 GR_GL_CALL_RET(interface, status, CheckFramebufferStatus(GR_GL_FRAMEBUFFER));
105 SkASSERT(GR_GL_FRAMEBUFFER_COMPLETE == status);
106 }
egdanielec00d942015-09-14 12:56:10 -0700107#endif
egdanielec00d942015-09-14 12:56:10 -0700108 return true;
109 } else {
110 const GrGLStencilAttachment* glStencil = static_cast<const GrGLStencilAttachment*>(stencil);
111 GrGLuint rb = glStencil->renderbufferID();
112
113 gpu->invalidateBoundRenderTarget();
114 gpu->stats()->incRenderTargetBinds();
115 GR_GL_CALL(interface, BindFramebuffer(GR_GL_FRAMEBUFFER, this->renderFBOID()));
116 GR_GL_CALL(interface, FramebufferRenderbuffer(GR_GL_FRAMEBUFFER,
117 GR_GL_STENCIL_ATTACHMENT,
118 GR_GL_RENDERBUFFER, rb));
119 if (glStencil->format().fPacked) {
120 GR_GL_CALL(interface, FramebufferRenderbuffer(GR_GL_FRAMEBUFFER,
121 GR_GL_DEPTH_ATTACHMENT,
122 GR_GL_RENDERBUFFER, rb));
123 } else {
124 GR_GL_CALL(interface, FramebufferRenderbuffer(GR_GL_FRAMEBUFFER,
125 GR_GL_DEPTH_ATTACHMENT,
126 GR_GL_RENDERBUFFER, 0));
127 }
128
129#ifdef SK_DEBUG
robertphillipsfa7ff472016-04-21 11:27:43 -0700130 if (kChromium_GrGLDriver != gpu->glContext().driver()) {
131 // This check can cause problems in Chromium if the context has been asynchronously
132 // abandoned (see skbug.com/5200)
133 GrGLenum status;
134 GR_GL_CALL_RET(interface, status, CheckFramebufferStatus(GR_GL_FRAMEBUFFER));
135 SkASSERT(GR_GL_FRAMEBUFFER_COMPLETE == status);
136 }
egdanielec00d942015-09-14 12:56:10 -0700137#endif
138 return true;
139 }
140}
141
bsalomon@google.comaa5b6732011-07-29 15:13:20 +0000142void GrGLRenderTarget::onRelease() {
kkinnunen2e6055b2016-04-22 01:48:29 -0700143 if (GrBackendObjectOwnership::kBorrowed != fRTFBOOwnership) {
egdanield803f272015-03-18 13:01:52 -0700144 if (fTexFBOID) {
145 GL_CALL(DeleteFramebuffers(1, &fTexFBOID));
bsalomon@google.comaa5b6732011-07-29 15:13:20 +0000146 }
egdanield803f272015-03-18 13:01:52 -0700147 if (fRTFBOID && fRTFBOID != fTexFBOID) {
148 GL_CALL(DeleteFramebuffers(1, &fRTFBOID));
bsalomon@google.comaa5b6732011-07-29 15:13:20 +0000149 }
bsalomon@google.comaa5b6732011-07-29 15:13:20 +0000150 if (fMSColorRenderbufferID) {
bsalomon@google.com0b77d682011-08-19 13:28:54 +0000151 GL_CALL(DeleteRenderbuffers(1, &fMSColorRenderbufferID));
bsalomon@google.comaa5b6732011-07-29 15:13:20 +0000152 }
153 }
egdanield803f272015-03-18 13:01:52 -0700154 fRTFBOID = 0;
155 fTexFBOID = 0;
156 fMSColorRenderbufferID = 0;
robertphillips@google.comd6bbbf82012-09-05 15:46:34 +0000157 INHERITED::onRelease();
bsalomon@google.comaa5b6732011-07-29 15:13:20 +0000158}
159
160void GrGLRenderTarget::onAbandon() {
egdanield803f272015-03-18 13:01:52 -0700161 fRTFBOID = 0;
162 fTexFBOID = 0;
bsalomon@google.comaa5b6732011-07-29 15:13:20 +0000163 fMSColorRenderbufferID = 0;
robertphillips@google.comd6bbbf82012-09-05 15:46:34 +0000164 INHERITED::onAbandon();
bsalomon@google.comaa5b6732011-07-29 15:13:20 +0000165}
egdanielec00d942015-09-14 12:56:10 -0700166
167GrGLGpu* GrGLRenderTarget::getGLGpu() const {
168 SkASSERT(!this->wasDestroyed());
169 return static_cast<GrGLGpu*>(this->getGpu());
170}
171
kkinnunen2e6055b2016-04-22 01:48:29 -0700172bool GrGLRenderTarget::canAttemptStencilAttachment() const {
ericrkc4025182016-05-04 12:01:58 -0700173 // Only modify the FBO's attachments if we have created the FBO. Public APIs do not currently
174 // allow for borrowed FBO ownership, so we can safely assume that if an object is owned,
175 // Skia created it.
176 return this->fRTFBOOwnership == GrBackendObjectOwnership::kOwned;
kkinnunen2e6055b2016-04-22 01:48:29 -0700177}
178
ericrk0a5fa482015-09-15 14:16:10 -0700179void GrGLRenderTarget::dumpMemoryStatistics(SkTraceMemoryDump* traceMemoryDump) const {
180 // Don't log the backing texture's contribution to the memory size. This will be handled by the
181 // texture object.
182
183 // Log any renderbuffer's contribution to memory. We only do this if we own the renderbuffer
184 // (have a fMSColorRenderbufferID).
185 if (fMSColorRenderbufferID) {
Robert Phillipsd6214d42016-11-07 08:23:48 -0500186 size_t size = GrSurface::ComputeSize(fDesc, this->msaaSamples(), false);
ericrk0a5fa482015-09-15 14:16:10 -0700187
188 // Due to this resource having both a texture and a renderbuffer component, dump as
189 // skia/gpu_resources/resource_#/renderbuffer
190 SkString dumpName("skia/gpu_resources/resource_");
Robert Phillips294870f2016-11-11 12:38:40 -0500191 dumpName.appendU32(this->uniqueID().asUInt());
ericrk0a5fa482015-09-15 14:16:10 -0700192 dumpName.append("/renderbuffer");
193
194 traceMemoryDump->dumpNumericValue(dumpName.c_str(), "size", "bytes", size);
195
196 if (this->isPurgeable()) {
197 traceMemoryDump->dumpNumericValue(dumpName.c_str(), "purgeable_size", "bytes", size);
198 }
199
200 SkString renderbuffer_id;
201 renderbuffer_id.appendU32(fMSColorRenderbufferID);
202 traceMemoryDump->setMemoryBacking(dumpName.c_str(), "gl_renderbuffer",
203 renderbuffer_id.c_str());
204 }
205}
206
ericrk0a5fa482015-09-15 14:16:10 -0700207int GrGLRenderTarget::msaaSamples() const {
208 if (fTexFBOID == kUnresolvableFBOID || fTexFBOID != fRTFBOID) {
209 // If the render target's FBO is external (fTexFBOID == kUnresolvableFBOID), or if we own
210 // the render target's FBO (fTexFBOID == fRTFBOID) then we use the provided sample count.
211 return SkTMax(1, fDesc.fSampleCnt);
212 }
213
214 // When fTexFBOID == fRTFBOID, we either are not using MSAA, or MSAA is auto resolving, so use
215 // 0 for the sample count.
216 return 0;
217}
218
219int GrGLRenderTarget::totalSamples() const {
220 int total_samples = this->msaaSamples();
221
222 if (fTexFBOID != kUnresolvableFBOID) {
223 // If we own the resolve buffer then that is one more sample per pixel.
224 total_samples += 1;
225 }
226
227 return total_samples;
228}