1 /*
2 * Copyright 2012 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 #include "src/gpu/gl/GrGLCaps.h"
9
10 #include <memory>
11
12 #include "include/gpu/GrContextOptions.h"
13 #include "src/core/SkCompressedDataUtils.h"
14 #include "src/core/SkMathPriv.h"
15 #include "src/core/SkTSearch.h"
16 #include "src/gpu/GrBackendUtils.h"
17 #include "src/gpu/GrProgramDesc.h"
18 #include "src/gpu/GrShaderCaps.h"
19 #include "src/gpu/GrSurfaceProxyPriv.h"
20 #include "src/gpu/GrTextureProxyPriv.h"
21 #include "src/gpu/SkGr.h"
22 #include "src/gpu/gl/GrGLContext.h"
23 #include "src/gpu/gl/GrGLRenderTarget.h"
24 #include "src/gpu/gl/GrGLTexture.h"
25
26 #if defined(SK_BUILD_FOR_IOS)
27 #include <TargetConditionals.h>
28 #endif
29
GrGLCaps(const GrContextOptions & contextOptions,const GrGLContextInfo & ctxInfo,const GrGLInterface * glInterface)30 GrGLCaps::GrGLCaps(const GrContextOptions& contextOptions,
31 const GrGLContextInfo& ctxInfo,
32 const GrGLInterface* glInterface) : INHERITED(contextOptions) {
33 fStandard = ctxInfo.standard();
34
35 fPackFlipYSupport = false;
36 fTextureUsageSupport = false;
37 fImagingSupport = false;
38 fVertexArrayObjectSupport = false;
39 fDebugSupport = false;
40 fES2CompatibilitySupport = false;
41 fDrawRangeElementsSupport = false;
42 fBaseVertexBaseInstanceSupport = false;
43 fIsCoreProfile = false;
44 fBindFragDataLocationSupport = false;
45 fRectangleTextureSupport = false;
46 fBindUniformLocationSupport = false;
47 fMipmapLevelControlSupport = false;
48 fMipmapLodControlSupport = false;
49 fUseBufferDataNullHint = false;
50 fDoManualMipmapping = false;
51 fClearToBoundaryValuesIsBroken = false;
52 fClearTextureSupport = false;
53 fDrawArraysBaseVertexIsBroken = false;
54 fDisallowTexSubImageForUnormConfigTexturesEverBoundToFBO = false;
55 fUseDrawInsteadOfAllRenderTargetWrites = false;
56 fRequiresCullFaceEnableDisableWhenDrawingLinesAfterNonLines = false;
57 fDontSetBaseOrMaxLevelForExternalTextures = false;
58 fNeverDisableColorWrites = false;
59 fMustSetAnyTexParameterToEnableMipmapping = false;
60 fAllowBGRA8CopyTexSubImage = false;
61 fDisallowDynamicMSAA = false;
62 fMustResetBlendFuncBetweenDualSourceAndDisable = false;
63 fBindTexture0WhenChangingTextureFBOMultisampleCount = false;
64 fRebindColorAttachmentAfterCheckFramebufferStatus = false;
65 fProgramBinarySupport = false;
66 fProgramParameterSupport = false;
67 fSamplerObjectSupport = false;
68 fUseSamplerObjects = false;
69 fTextureSwizzleSupport = false;
70 fTiledRenderingSupport = false;
71 fFBFetchRequiresEnablePerSample = false;
72 fSRGBWriteControl = false;
73 fSkipErrorChecks = false;
74
75 fShaderCaps = std::make_unique<GrShaderCaps>();
76
77 // All of Skia's automated testing of ANGLE and all related tuning of performance and driver
78 // workarounds is oriented around the D3D backends of ANGLE. Chrome has started using Skia
79 // on top of ANGLE's GL backend. In this case ANGLE is still interfacing the same underlying
80 // GL driver that our performance and correctness tuning was performed on. To avoid losing
81 // that we strip the ANGLE info and for the rest of caps setup pretend we're directly on top of
82 // the GL driver. Note that this means that some driver workarounds are likely implemented at
83 // two levels of the stack (Skia and ANGLE) but we haven't determined which.
84 if (ctxInfo.angleBackend() == GrGLANGLEBackend::kOpenGL) {
85 this->init(contextOptions, ctxInfo.makeNonAngle(), glInterface);
86 // A major caveat is that ANGLE does not allow client side arrays.
87 fPreferClientSideDynamicBuffers = false;
88 } else {
89 this->init(contextOptions, ctxInfo, glInterface);
90 }
91 }
92
init(const GrContextOptions & contextOptions,const GrGLContextInfo & ctxInfo,const GrGLInterface * gli)93 void GrGLCaps::init(const GrContextOptions& contextOptions,
94 const GrGLContextInfo& ctxInfo,
95 const GrGLInterface* gli) {
96 GrGLStandard standard = ctxInfo.standard();
97 // standard can be unused (optimized away) if SK_ASSUME_GL_ES is set
98 sk_ignore_unused_variable(standard);
99 GrGLVersion version = ctxInfo.version();
100
101 if (GR_IS_GR_GL(standard)) {
102 GrGLint max;
103 GR_GL_GetIntegerv(gli, GR_GL_MAX_FRAGMENT_UNIFORM_COMPONENTS, &max);
104 fMaxFragmentUniformVectors = max / 4;
105 if (version >= GR_GL_VER(3, 2)) {
106 GrGLint profileMask;
107 GR_GL_GetIntegerv(gli, GR_GL_CONTEXT_PROFILE_MASK, &profileMask);
108 fIsCoreProfile = SkToBool(profileMask & GR_GL_CONTEXT_CORE_PROFILE_BIT);
109 }
110 } else if (GR_IS_GR_GL_ES(standard) || GR_IS_GR_WEBGL(standard)) {
111 GR_GL_GetIntegerv(gli, GR_GL_MAX_FRAGMENT_UNIFORM_VECTORS,
112 &fMaxFragmentUniformVectors);
113 }
114
115 if (fDriverBugWorkarounds.max_fragment_uniform_vectors_32) {
116 fMaxFragmentUniformVectors = std::min(fMaxFragmentUniformVectors, 32);
117 }
118 GR_GL_GetIntegerv(gli, GR_GL_MAX_VERTEX_ATTRIBS, &fMaxVertexAttributes);
119
120 if (GR_IS_GR_GL(standard)) {
121 fWritePixelsRowBytesSupport = true;
122 fReadPixelsRowBytesSupport = true;
123 fPackFlipYSupport = false;
124 } else if (GR_IS_GR_GL_ES(standard)) {
125 fWritePixelsRowBytesSupport =
126 version >= GR_GL_VER(3, 0) || ctxInfo.hasExtension("GL_EXT_unpack_subimage");
127 fReadPixelsRowBytesSupport =
128 version >= GR_GL_VER(3, 0) || ctxInfo.hasExtension("GL_NV_pack_subimage");
129 fPackFlipYSupport =
130 ctxInfo.hasExtension("GL_ANGLE_pack_reverse_row_order");
131 } else if (GR_IS_GR_WEBGL(standard)) {
132 // WebGL 2.0 has these
133 fWritePixelsRowBytesSupport = version >= GR_GL_VER(2, 0);
134 fReadPixelsRowBytesSupport = version >= GR_GL_VER(2, 0);
135 }
136 fTransferPixelsToRowBytesSupport = fWritePixelsRowBytesSupport;
137
138 if (fDriverBugWorkarounds.pack_parameters_workaround_with_pack_buffer) {
139 // In some cases drivers handle copying the last row incorrectly
140 // when using GL_PACK_ROW_LENGTH. Chromium handles this by iterating
141 // through every row and conditionally clobbering that value, but
142 // Skia already has a scratch buffer workaround when pack row length
143 // is not supported, so just use that.
144 fReadPixelsRowBytesSupport = false;
145 }
146
147 fTextureUsageSupport = GR_IS_GR_GL_ES(standard) &&
148 ctxInfo.hasExtension("GL_ANGLE_texture_usage");
149
150 if (GR_IS_GR_GL(standard)) {
151 fTextureBarrierSupport = version >= GR_GL_VER(4,5) ||
152 ctxInfo.hasExtension("GL_ARB_texture_barrier") ||
153 ctxInfo.hasExtension("GL_NV_texture_barrier");
154 } else if (GR_IS_GR_GL_ES(standard)) {
155 fTextureBarrierSupport = ctxInfo.hasExtension("GL_NV_texture_barrier");
156 } else if (GR_IS_GR_WEBGL(standard)) {
157 fTextureBarrierSupport = false;
158 }
159
160 if (GR_IS_GR_GL(standard)) {
161 fSampleLocationsSupport = version >= GR_GL_VER(3,2) ||
162 ctxInfo.hasExtension("GL_ARB_texture_multisample");
163 } else if (GR_IS_GR_GL_ES(standard)) {
164 fSampleLocationsSupport = version >= GR_GL_VER(3,1);
165 } else if (GR_IS_GR_WEBGL(standard)) {
166 fSampleLocationsSupport = false;
167 }
168
169 fImagingSupport = GR_IS_GR_GL(standard) &&
170 ctxInfo.hasExtension("GL_ARB_imaging");
171
172 if (((GR_IS_GR_GL(standard) && version >= GR_GL_VER(4,3)) ||
173 (GR_IS_GR_GL_ES(standard) && version >= GR_GL_VER(3,0)) ||
174 ctxInfo.hasExtension("GL_ARB_invalidate_subdata"))) {
175 fInvalidateFBType = kInvalidate_InvalidateFBType;
176 } else if (ctxInfo.hasExtension("GL_EXT_discard_framebuffer")) {
177 fInvalidateFBType = kDiscard_InvalidateFBType;
178 }
179
180 // For future reference on Desktop GL, GL_PRIMITIVE_RESTART_FIXED_INDEX appears in 4.3, and
181 // GL_PRIMITIVE_RESTART (where the client must call glPrimitiveRestartIndex) appears in 3.1.
182 if (GR_IS_GR_GL_ES(standard)) {
183 // Primitive restart can cause a 3x slowdown on Adreno. Enable conservatively.
184 // FIXME: Primitive restart would likely be a win on iOS if we had an enum value for it.
185 if (ctxInfo.vendor() == GrGLVendor::kARM) {
186 fUsePrimitiveRestart = version >= GR_GL_VER(3,0);
187 }
188 }
189
190 if (ctxInfo.vendor() == GrGLVendor::kARM ||
191 ctxInfo.vendor() == GrGLVendor::kImagination ||
192 ctxInfo.vendor() == GrGLVendor::kQualcomm ) {
193 fPreferFullscreenClears = true;
194 }
195
196 if (GR_IS_GR_GL(standard)) {
197 fVertexArrayObjectSupport = version >= GR_GL_VER(3, 0) ||
198 ctxInfo.hasExtension("GL_ARB_vertex_array_object") ||
199 ctxInfo.hasExtension("GL_APPLE_vertex_array_object");
200 } else if (GR_IS_GR_GL_ES(standard)) {
201 fVertexArrayObjectSupport = version >= GR_GL_VER(3, 0) ||
202 ctxInfo.hasExtension("GL_OES_vertex_array_object");
203 } else if (GR_IS_GR_WEBGL(standard)) {
204 fVertexArrayObjectSupport = version >= GR_GL_VER(2, 0) ||
205 ctxInfo.hasExtension("GL_OES_vertex_array_object") ||
206 ctxInfo.hasExtension("OES_vertex_array_object");
207 }
208
209 if (GR_IS_GR_GL(standard) && version >= GR_GL_VER(4,3)) {
210 fDebugSupport = true;
211 } else if (GR_IS_GR_GL_ES(standard)) {
212 fDebugSupport = ctxInfo.hasExtension("GL_KHR_debug");
213 } else if (GR_IS_GR_WEBGL(standard)) {
214 fDebugSupport = false;
215 }
216
217 if (GR_IS_GR_GL(standard)) {
218 fES2CompatibilitySupport = ctxInfo.hasExtension("GL_ARB_ES2_compatibility");
219 }
220 else if (GR_IS_GR_GL_ES(standard)) {
221 fES2CompatibilitySupport = true;
222 } else if (GR_IS_GR_WEBGL(standard)) {
223 fES2CompatibilitySupport = true;
224 }
225
226 if (GR_IS_GR_GL(standard)) {
227 fClientCanDisableMultisample = true;
228 } else if (GR_IS_GR_GL_ES(standard)) {
229 fClientCanDisableMultisample = ctxInfo.hasExtension("GL_EXT_multisample_compatibility");
230 } else if (GR_IS_GR_WEBGL(standard)) {
231 fClientCanDisableMultisample = false;
232 }
233
234 if (GR_IS_GR_GL(standard)) {
235 // 3.1 has draw_instanced but not instanced_arrays, for the time being we only care about
236 // instanced arrays, but we could make this more granular if we wanted
237 fDrawInstancedSupport =
238 version >= GR_GL_VER(3, 2) ||
239 (ctxInfo.hasExtension("GL_ARB_draw_instanced") &&
240 ctxInfo.hasExtension("GL_ARB_instanced_arrays"));
241 } else if (GR_IS_GR_GL_ES(standard)) {
242 fDrawInstancedSupport =
243 version >= GR_GL_VER(3, 0) ||
244 (ctxInfo.hasExtension("GL_EXT_draw_instanced") &&
245 ctxInfo.hasExtension("GL_EXT_instanced_arrays")) ||
246 ctxInfo.hasExtension("GL_ANGLE_instanced_arrays");
247 } else if (GR_IS_GR_WEBGL(standard)) {
248 // WebGL 2.0 has DrawArraysInstanced and drawElementsInstanced
249 fDrawInstancedSupport = version >= GR_GL_VER(2, 0);
250 }
251
252 #ifdef GR_DISABLE_TESSELLATION_ON_ES2
253 if (GR_IS_GR_GL_ES(standard) && version < GR_GL_VER(3, 0)) {
254 // Temporarily disable the tessellation path renderer on Chrome ES2 while we roll the
255 // necessary Skia changes.
256 fDisableTessellationPathRenderer = true;
257 }
258 #else
259 if (GR_IS_GR_GL_ES(standard) && ctxInfo.isOverCommandBuffer() && version < GR_GL_VER(3, 0)) {
260 // Temporarily disable the tessellation path renderer over the ES2 command buffer. This is
261 // an attempt to lower impact while we roll out tessellation in Chrome.
262 fDisableTessellationPathRenderer = true;
263 }
264 #endif
265
266 if (GR_IS_GR_GL(standard)) {
267 if (version >= GR_GL_VER(3, 0)) {
268 fBindFragDataLocationSupport = true;
269 }
270 } else if (GR_IS_GR_GL_ES(standard)) {
271 if (version >= GR_GL_VER(3, 0) && ctxInfo.hasExtension("GL_EXT_blend_func_extended")) {
272 fBindFragDataLocationSupport = true;
273 }
274 } else if (GR_IS_GR_WEBGL(standard)) {
275 fBindFragDataLocationSupport = false;
276 }
277
278 fBindUniformLocationSupport = ctxInfo.hasExtension("GL_CHROMIUM_bind_uniform_location");
279
280 if (GR_IS_GR_GL(standard)) {
281 if (version >= GR_GL_VER(3, 1) || ctxInfo.hasExtension("GL_ARB_texture_rectangle") ||
282 ctxInfo.hasExtension("GL_ANGLE_texture_rectangle")) {
283 fRectangleTextureSupport = true;
284 }
285 } else if (GR_IS_GR_GL_ES(standard)) {
286 fRectangleTextureSupport = ctxInfo.hasExtension("GL_ARB_texture_rectangle") ||
287 ctxInfo.hasExtension("GL_ANGLE_texture_rectangle");
288 } else if (GR_IS_GR_WEBGL(standard)) {
289 fRectangleTextureSupport = false;
290 }
291
292 // GrCaps defaults fClampToBorderSupport to true, so disable when unsupported
293 if (GR_IS_GR_GL(standard)) {
294 // Clamp to border added in 1.3
295 if (version < GR_GL_VER(1, 3) && !ctxInfo.hasExtension("GL_ARB_texture_border_clamp")) {
296 fClampToBorderSupport = false;
297 }
298 } else if (GR_IS_GR_GL_ES(standard)) {
299 // GLES didn't have clamp to border until 3.2, but provides several alternative extensions
300 if (version < GR_GL_VER(3, 2) && !ctxInfo.hasExtension("GL_EXT_texture_border_clamp") &&
301 !ctxInfo.hasExtension("GL_NV_texture_border_clamp") &&
302 !ctxInfo.hasExtension("GL_OES_texture_border_clamp")) {
303 fClampToBorderSupport = false;
304 }
305 } else if (GR_IS_GR_WEBGL(standard)) {
306 // WebGL appears to only have REPEAT, CLAMP_TO_EDGE and MIRRORED_REPEAT
307 fClampToBorderSupport = false;
308 }
309
310 if (GR_IS_GR_GL(standard)) {
311 if (version >= GR_GL_VER(3,3) || ctxInfo.hasExtension("GL_ARB_texture_swizzle")) {
312 fTextureSwizzleSupport = true;
313 }
314 } else if (GR_IS_GR_GL_ES(standard)) {
315 if (version >= GR_GL_VER(3,0)) {
316 fTextureSwizzleSupport = true;
317 }
318 } else if (GR_IS_GR_WEBGL(standard)) {
319 fTextureSwizzleSupport = false;
320 }
321
322 if (GR_IS_GR_GL(standard)) {
323 fMipmapLevelControlSupport = true;
324 fMipmapLodControlSupport = true;
325 } else if (GR_IS_GR_GL_ES(standard)) {
326 if (version >= GR_GL_VER(3,0)) {
327 fMipmapLevelControlSupport = true;
328 fMipmapLodControlSupport = true;
329 }
330 } else if (GR_IS_GR_WEBGL(standard)) {
331 fMipmapLevelControlSupport = false;
332 fMipmapLodControlSupport = false;
333 }
334
335 // Chrome's command buffer will zero out a buffer if null is passed to glBufferData to avoid
336 // letting an application see uninitialized memory. WebGL spec explicitly disallows null values.
337 fUseBufferDataNullHint = !GR_IS_GR_WEBGL(standard) && !ctxInfo.isOverCommandBuffer();
338
339 if (GR_IS_GR_GL(standard)) {
340 fClearTextureSupport = (version >= GR_GL_VER(4,4) ||
341 ctxInfo.hasExtension("GL_ARB_clear_texture"));
342 } else if (GR_IS_GR_GL_ES(standard)) {
343 fClearTextureSupport = ctxInfo.hasExtension("GL_EXT_clear_texture");
344 } else if (GR_IS_GR_WEBGL(standard)) {
345 fClearTextureSupport = false;
346 }
347
348 #if defined(SK_BUILD_FOR_ANDROID) && __ANDROID_API__ >= 26
349 fSupportsAHardwareBufferImages = true;
350 #endif
351
352 if (GR_IS_GR_GL(standard)) {
353 fSRGBWriteControl = version >= GR_GL_VER(3, 0) ||
354 ctxInfo.hasExtension("GL_ARB_framebuffer_sRGB") ||
355 ctxInfo.hasExtension("GL_EXT_framebuffer_sRGB");
356 } else if (GR_IS_GR_GL_ES(standard)) {
357 // ES through 3.2 requires EXT_srgb_write_control to support toggling
358 // sRGB writing for destinations.
359 fSRGBWriteControl = ctxInfo.hasExtension("GL_EXT_sRGB_write_control");
360 } // No WebGL support
361
362 fSkipErrorChecks = ctxInfo.isOverCommandBuffer();
363 if (GR_IS_GR_WEBGL(standard)) {
364 // Error checks are quite costly in webgl, especially in Chrome.
365 fSkipErrorChecks = true;
366 }
367
368 // When we are abandoning the context we cannot call into GL thus we should skip any sync work.
369 fMustSyncGpuDuringAbandon = false;
370
371 /**************************************************************************
372 * GrShaderCaps fields
373 **************************************************************************/
374
375 // This must be called after fCoreProfile is set on the GrGLCaps
376 this->initGLSL(ctxInfo, gli);
377 GrShaderCaps* shaderCaps = fShaderCaps.get();
378
379 // Enable supported shader-related caps
380 if (GR_IS_GR_GL(standard)) {
381 shaderCaps->fDualSourceBlendingSupport =
382 (version >= GR_GL_VER(3, 3) ||
383 ctxInfo.hasExtension("GL_ARB_blend_func_extended")) &&
384 ctxInfo.glslGeneration() >= k130_GrGLSLGeneration;
385
386 shaderCaps->fShaderDerivativeSupport = true;
387
388 shaderCaps->fIntegerSupport = version >= GR_GL_VER(3, 0) &&
389 ctxInfo.glslGeneration() >= k130_GrGLSLGeneration;
390
391 shaderCaps->fNonsquareMatrixSupport = ctxInfo.glslGeneration() >= k130_GrGLSLGeneration;
392 shaderCaps->fInverseHyperbolicSupport = ctxInfo.glslGeneration() >= k130_GrGLSLGeneration;
393 } else if (GR_IS_GR_GL_ES(standard)) {
394 shaderCaps->fDualSourceBlendingSupport = ctxInfo.hasExtension("GL_EXT_blend_func_extended");
395
396 shaderCaps->fShaderDerivativeSupport = version >= GR_GL_VER(3, 0) ||
397 ctxInfo.hasExtension("GL_OES_standard_derivatives");
398
399 shaderCaps->fIntegerSupport = version >= GR_GL_VER(3, 0) &&
400 ctxInfo.glslGeneration() >= k330_GrGLSLGeneration; // We use this value for GLSL ES 3.0.
401 shaderCaps->fNonsquareMatrixSupport = ctxInfo.glslGeneration() >= k330_GrGLSLGeneration;
402 shaderCaps->fInverseHyperbolicSupport = ctxInfo.glslGeneration() >= k330_GrGLSLGeneration;
403 } else if (GR_IS_GR_WEBGL(standard)) {
404 shaderCaps->fShaderDerivativeSupport = version >= GR_GL_VER(2, 0) ||
405 ctxInfo.hasExtension("GL_OES_standard_derivatives") ||
406 ctxInfo.hasExtension("OES_standard_derivatives");
407 shaderCaps->fIntegerSupport = (version >= GR_GL_VER(2, 0));
408 shaderCaps->fNonsquareMatrixSupport = ctxInfo.glslGeneration() >= k330_GrGLSLGeneration;
409 shaderCaps->fInverseHyperbolicSupport = ctxInfo.glslGeneration() >= k330_GrGLSLGeneration;
410 }
411
412 if (ctxInfo.hasExtension("GL_NV_conservative_raster")) {
413 fConservativeRasterSupport = true;
414 }
415
416 if (GR_IS_GR_GL(standard)) {
417 fWireframeSupport = true;
418 }
419
420 if (GR_IS_GR_GL(standard)) {
421 shaderCaps->fRewriteSwitchStatements =
422 ctxInfo.glslGeneration() < k130_GrGLSLGeneration; // introduced in GLSL 1.3
423 } else if (GR_IS_GR_GL_ES(standard)) {
424 shaderCaps->fRewriteSwitchStatements =
425 ctxInfo.glslGeneration() < k330_GrGLSLGeneration; // introduced in GLSL ES3
426 } else if (GR_IS_GR_WEBGL(standard)) {
427 shaderCaps->fRewriteSwitchStatements = version < GR_GL_VER(2, 0); // introduced in WebGL 2
428 }
429
430 // Protect ourselves against tracking huge amounts of texture state.
431 static const uint8_t kMaxSaneSamplers = 32;
432 GrGLint maxSamplers;
433 GR_GL_GetIntegerv(gli, GR_GL_MAX_TEXTURE_IMAGE_UNITS, &maxSamplers);
434 shaderCaps->fMaxFragmentSamplers = std::min<GrGLint>(kMaxSaneSamplers, maxSamplers);
435
436 // SGX and Mali GPUs have tiled architectures that have trouble with frequently changing VBOs.
437 // We've measured a performance increase using non-VBO vertex data for dynamic content on these
438 // GPUs. Perhaps we should read the renderer string and limit this decision to specific GPU
439 // families rather than basing it on the vendor alone.
440 // The Chrome command buffer blocks the use of client side buffers (but may emulate VBOs with
441 // them). Client side buffers are not allowed in core profiles.
442 if (GR_IS_GR_GL(standard) || GR_IS_GR_GL_ES(standard)) {
443 if (!ctxInfo.isOverCommandBuffer() && !fIsCoreProfile &&
444 (ctxInfo.vendor() == GrGLVendor::kARM ||
445 ctxInfo.vendor() == GrGLVendor::kImagination ||
446 ctxInfo.vendor() == GrGLVendor::kQualcomm)) {
447 fPreferClientSideDynamicBuffers = true;
448 }
449 } // No client side arrays in WebGL https://www.khronos.org/registry/webgl/specs/1.0/#6.2
450
451 if (!contextOptions.fAvoidStencilBuffers) {
452 // To reduce surface area, if we avoid stencil buffers, we also disable MSAA.
453 this->initFSAASupport(contextOptions, ctxInfo, gli);
454 this->initStencilSupport(ctxInfo);
455 }
456
457 // Setup blit framebuffer
458 if (GR_IS_GR_GL(standard)) {
459 if (version >= GR_GL_VER(3,0) ||
460 ctxInfo.hasExtension("GL_ARB_framebuffer_object") ||
461 ctxInfo.hasExtension("GL_EXT_framebuffer_blit")) {
462 fBlitFramebufferFlags = 0;
463 }
464 } else if (GR_IS_GR_GL_ES(standard)) {
465 if (version >= GR_GL_VER(3, 0) ||
466 ctxInfo.hasExtension("GL_NV_framebuffer_blit")) {
467 fBlitFramebufferFlags = kNoFormatConversionForMSAASrc_BlitFramebufferFlag |
468 kNoMSAADst_BlitFramebufferFlag |
469 kRectsMustMatchForMSAASrc_BlitFramebufferFlag;
470 } else if (ctxInfo.hasExtension("GL_CHROMIUM_framebuffer_multisample") ||
471 ctxInfo.hasExtension("GL_ANGLE_framebuffer_blit")) {
472 // The CHROMIUM extension uses the ANGLE version of glBlitFramebuffer and includes its
473 // limitations.
474 fBlitFramebufferFlags = kNoScalingOrMirroring_BlitFramebufferFlag |
475 kResolveMustBeFull_BlitFrambufferFlag |
476 kNoMSAADst_BlitFramebufferFlag |
477 kNoFormatConversion_BlitFramebufferFlag |
478 kRectsMustMatchForMSAASrc_BlitFramebufferFlag;
479 }
480 } // No WebGL 1.0 support for BlitFramebuffer
481
482 this->initBlendEqationSupport(ctxInfo);
483
484 if (GR_IS_GR_GL(standard)) {
485 fMapBufferFlags = kCanMap_MapFlag; // we require VBO support and the desktop VBO
486 // extension includes glMapBuffer.
487 if (version >= GR_GL_VER(3, 0) || ctxInfo.hasExtension("GL_ARB_map_buffer_range")) {
488 fMapBufferFlags |= kSubset_MapFlag;
489 fMapBufferType = kMapBufferRange_MapBufferType;
490 } else {
491 fMapBufferType = kMapBuffer_MapBufferType;
492 }
493 } else if (GR_IS_GR_GL_ES(standard)) {
494 // Unextended GLES2 doesn't have any buffer mapping.
495 fMapBufferFlags = kNone_MapFlags;
496 if (ctxInfo.hasExtension("GL_CHROMIUM_map_sub")) {
497 fMapBufferFlags = kCanMap_MapFlag | kSubset_MapFlag;
498 fMapBufferType = kChromium_MapBufferType;
499 } else if (version >= GR_GL_VER(3, 0) || ctxInfo.hasExtension("GL_EXT_map_buffer_range")) {
500 fMapBufferFlags = kCanMap_MapFlag | kSubset_MapFlag;
501 fMapBufferType = kMapBufferRange_MapBufferType;
502 } else if (ctxInfo.hasExtension("GL_OES_mapbuffer")) {
503 fMapBufferFlags = kCanMap_MapFlag;
504 fMapBufferType = kMapBuffer_MapBufferType;
505 }
506 } else if (GR_IS_GR_WEBGL(standard)) {
507 // explicitly removed https://www.khronos.org/registry/webgl/specs/2.0/#5.14
508 fMapBufferFlags = kNone_MapFlags;
509 }
510
511 if (GR_IS_GR_GL(standard)) {
512 if (version >= GR_GL_VER(2, 1) || ctxInfo.hasExtension("GL_ARB_pixel_buffer_object") ||
513 ctxInfo.hasExtension("GL_EXT_pixel_buffer_object")) {
514 fTransferFromBufferToTextureSupport = true;
515 fTransferFromSurfaceToBufferSupport = true;
516 fTransferBufferType = TransferBufferType::kARB_PBO;
517 }
518 } else if (GR_IS_GR_GL_ES(standard)) {
519 if (version >= GR_GL_VER(3, 0) ||
520 (ctxInfo.hasExtension("GL_NV_pixel_buffer_object") &&
521 // GL_EXT_unpack_subimage needed to support subtexture rectangles
522 ctxInfo.hasExtension("GL_EXT_unpack_subimage"))) {
523 fTransferFromBufferToTextureSupport = true;
524 fTransferFromSurfaceToBufferSupport = true;
525 if (version < GR_GL_VER(3, 0)) {
526 fTransferBufferType = TransferBufferType::kNV_PBO;
527 } else {
528 fTransferBufferType = TransferBufferType::kARB_PBO;
529 }
530 // TODO: get transfer buffers working in Chrome
531 // } else if (ctxInfo.hasExtension("GL_CHROMIUM_pixel_transfer_buffer_object")) {
532 // fTransferFromBufferToTextureSupport = false;
533 // fTransferFromSurfaceToBufferSupport = false;
534 // fTransferBufferType = TransferBufferType::kChromium;
535 }
536 } else if (GR_IS_GR_WEBGL(standard)) {
537 fTransferFromBufferToTextureSupport = false;
538 fTransferFromSurfaceToBufferSupport = false;
539 }
540
541 // On many GPUs, map memory is very expensive, so we effectively disable it here by setting the
542 // threshold to the maximum unless the client gives us a hint that map memory is cheap.
543 if (fBufferMapThreshold < 0) {
544 #if 0
545 // We think mapping on Chromium will be cheaper once we know ahead of time how much space
546 // we will use for all GrMeshDrawOps. Right now we might wind up mapping a large buffer and
547 // using a small subset.
548 fBufferMapThreshold = ctxInfo.isOverCommandBuffer() ? 0 : SK_MaxS32;
549 #else
550 fBufferMapThreshold = SK_MaxS32;
551 #endif
552 }
553
554 if (GR_IS_GR_GL(standard)) {
555 fNPOTTextureTileSupport = true;
556 fMipmapSupport = true;
557 } else if (GR_IS_GR_GL_ES(standard)) {
558 // Unextended ES2 supports NPOT textures with clamp_to_edge and non-mip filters only
559 // ES3 has no limitations.
560 fNPOTTextureTileSupport = version >= GR_GL_VER(3,0) ||
561 ctxInfo.hasExtension("GL_OES_texture_npot");
562 // ES2 supports MIP mapping for POT textures but our caps don't allow for limited MIP
563 // support. The OES extension or ES 3.0 allow for MIPS on NPOT textures. So, apparently,
564 // does the undocumented GL_IMG_texture_npot extension. This extension does not seem to
565 // to alllow arbitrary wrap modes, however.
566 fMipmapSupport = fNPOTTextureTileSupport || ctxInfo.hasExtension("GL_IMG_texture_npot");
567 } else if (GR_IS_GR_WEBGL(standard)) {
568 // Texture access works in the WebGL 2.0 API as in the OpenGL ES 3.0 API
569 fNPOTTextureTileSupport = version >= GR_GL_VER(2,0);
570 // All mipmapping and all wrapping modes are supported for non-power-of-
571 // two images [in WebGL 2.0].
572 fMipmapSupport = fNPOTTextureTileSupport;
573 }
574
575 GR_GL_GetIntegerv(gli, GR_GL_MAX_TEXTURE_SIZE, &fMaxTextureSize);
576
577 if (fDriverBugWorkarounds.max_texture_size_limit_4096) {
578 fMaxTextureSize = std::min(fMaxTextureSize, 4096);
579 }
580
581 GR_GL_GetIntegerv(gli, GR_GL_MAX_RENDERBUFFER_SIZE, &fMaxRenderTargetSize);
582 fMaxPreferredRenderTargetSize = fMaxRenderTargetSize;
583
584 if (ctxInfo.vendor() == GrGLVendor::kARM) {
585 // On Mali G71, RT's above 4k have been observed to incur a performance cost.
586 fMaxPreferredRenderTargetSize = std::min(4096, fMaxPreferredRenderTargetSize);
587 }
588
589 fGpuTracingSupport = ctxInfo.hasExtension("GL_EXT_debug_marker");
590
591 // Disable scratch texture reuse on Mali and Adreno devices
592 fReuseScratchTextures = (ctxInfo.vendor() != GrGLVendor::kARM);
593
594 #if 0
595 fReuseScratchBuffers = ctxInfo.vendor() != GrGLVendor::kARM
596 ctxInfo.vendor() != GrGLVendor::kQualcomm;
597 #endif
598
599 if (ctxInfo.hasExtension("GL_EXT_window_rectangles")) {
600 GR_GL_GetIntegerv(gli, GR_GL_MAX_WINDOW_RECTANGLES, &fMaxWindowRectangles);
601 }
602
603 #ifdef SK_BUILD_FOR_WIN
604 // We're assuming that on Windows Chromium we're using ANGLE.
605 bool isANGLE = ctxInfo.angleBackend() != GrGLANGLEBackend::kUnknown ||
606 ctxInfo.isOverCommandBuffer();
607 // On ANGLE deferring flushes can lead to GPU starvation
608 fPreferVRAMUseOverFlushes = !isANGLE;
609 #endif
610
611 if (ctxInfo.isOverCommandBuffer()) {
612 fMustClearUploadedBufferData = true;
613 }
614
615 // In a WASM build on Firefox, we see warnings like
616 // WebGL warning: texSubImage2D: This operation requires zeroing texture data. This is slow.
617 // WebGL warning: texSubImage2D: Texture has not been initialized prior to a partial upload,
618 // forcing the browser to clear it. This may be slow.
619 // Setting the initial clear seems to make those warnings go away and offers a substantial
620 // boost in performance in Firefox. Chrome sees a more modest increase.
621 if (GR_IS_GR_WEBGL(standard)) {
622 fShouldInitializeTextures = true;
623 }
624
625 if (GR_IS_GR_GL(standard)) {
626 // ARB allows mixed size FBO attachments, EXT does not.
627 if (version >= GR_GL_VER(3, 0) ||
628 ctxInfo.hasExtension("GL_ARB_framebuffer_object")) {
629 fOversizedStencilSupport = true;
630 } else {
631 SkASSERT(ctxInfo.hasExtension("GL_EXT_framebuffer_object"));
632 }
633 } else if (GR_IS_GR_GL_ES(standard)) {
634 // ES 3.0 supports mixed size FBO attachments, 2.0 does not.
635 fOversizedStencilSupport = version >= GR_GL_VER(3, 0);
636 } else if (GR_IS_GR_WEBGL(standard)) {
637 // WebGL 1.0 has some constraints for FBO attachments:
638 // https://www.khronos.org/registry/webgl/specs/1.0/index.html#6.6
639 // These constraints "no longer apply in WebGL 2"
640 fOversizedStencilSupport = version >= GR_GL_VER(2, 0);
641 }
642
643 if (GR_IS_GR_GL(standard)) {
644 fBaseVertexBaseInstanceSupport = version >= GR_GL_VER(4,2) ||
645 ctxInfo.hasExtension("GL_ARB_base_instance");
646 if (fBaseVertexBaseInstanceSupport) {
647 fNativeDrawIndirectSupport = version >= GR_GL_VER(4,0) ||
648 ctxInfo.hasExtension("GL_ARB_draw_indirect");
649 if (version >= GR_GL_VER(4,3) || ctxInfo.hasExtension("GL_ARB_multi_draw_indirect")) {
650 fMultiDrawType = MultiDrawType::kMultiDrawIndirect;
651 }
652 }
653 fDrawRangeElementsSupport = version >= GR_GL_VER(2,0);
654 } else if (GR_IS_GR_GL_ES(standard)) {
655 if (ctxInfo.hasExtension("GL_ANGLE_base_vertex_base_instance")) {
656 fBaseVertexBaseInstanceSupport = true;
657 fNativeDrawIndirectSupport = true;
658 fMultiDrawType = MultiDrawType::kANGLEOrWebGL;
659 // The indirect structs need to reside in CPU memory for the ANGLE version.
660 fUseClientSideIndirectBuffers = true;
661 } else {
662 fBaseVertexBaseInstanceSupport = ctxInfo.hasExtension("GL_EXT_base_instance");
663 if (fBaseVertexBaseInstanceSupport) {
664 fNativeDrawIndirectSupport = (version >= GR_GL_VER(3,1));
665 // Don't use GL_EXT_multi_draw_indirect. It doesn't allow VAO 0.
666 // https://github.com/KhronosGroup/OpenGL-Registry/blob/main/extensions/EXT/EXT_multi_draw_indirect.txt#L142
667 }
668 }
669 fDrawRangeElementsSupport = version >= GR_GL_VER(3,0);
670 } else if (GR_IS_GR_WEBGL(standard)) {
671 fBaseVertexBaseInstanceSupport = ctxInfo.hasExtension(
672 "WEBGL_draw_instanced_base_vertex_base_instance");
673 if (fBaseVertexBaseInstanceSupport && ctxInfo.hasExtension(
674 "GL_WEBGL_multi_draw_instanced_base_vertex_base_instance")) {
675 fNativeDrawIndirectSupport = true;
676 fMultiDrawType = MultiDrawType::kANGLEOrWebGL;
677 }
678 // The indirect structs need to reside in CPU memory for the WebGL version.
679 fUseClientSideIndirectBuffers = true;
680 fDrawRangeElementsSupport = version >= GR_GL_VER(2,0);
681 }
682 // We used to disable this as a correctness workaround (http://anglebug.com/4536). Now it is
683 // disabled because of poor performance (http://skbug.com/11998).
684 if (ctxInfo.angleBackend() == GrGLANGLEBackend::kD3D11) {
685 fBaseVertexBaseInstanceSupport = false;
686 fNativeDrawIndirectSupport = false;
687 fMultiDrawType = MultiDrawType::kNone;
688 }
689
690 // We prefer GL sync objects but also support NV_fence_sync. The former can be
691 // used to implements GrFence and GrSemaphore. The latter only implements GrFence.
692 // TODO: support CHROMIUM_sync_point and maybe KHR_fence_sync
693 if (GR_IS_GR_WEBGL(standard)) {
694 // Only in WebGL 2.0
695 fSemaphoreSupport = fFenceSyncSupport = version >= GR_GL_VER(2, 0);
696 fFenceType = FenceType::kSyncObject;
697 } else if (GR_IS_GR_GL(standard) &&
698 (version >= GR_GL_VER(3, 2) || ctxInfo.hasExtension("GL_ARB_sync"))) {
699 fSemaphoreSupport = fFenceSyncSupport = true;
700 fFenceType = FenceType::kSyncObject;
701 } else if (GR_IS_GR_GL_ES(standard) &&
702 (version >= GR_GL_VER(3, 0) || ctxInfo.hasExtension("GL_APPLE_sync"))) {
703 fSemaphoreSupport = fFenceSyncSupport = true;
704 fFenceType = FenceType::kSyncObject;
705 } else if (ctxInfo.hasExtension("GL_NV_fence")) {
706 // This extension can exist in GL and GL ES. We have it last because we prefer the
707 // standard GLsync object implementation which also supports GPU semaphore semantics.
708 fFenceSyncSupport = true;
709 fFenceType = FenceType::kNVFence;
710 }
711
712 // Safely moving textures between contexts requires semaphores.
713 fCrossContextTextureSupport = fSemaphoreSupport;
714
715 // Half float vertex attributes requires GL3 or ES3
716 // It can also work with OES_VERTEX_HALF_FLOAT, but that requires a different enum.
717 if (GR_IS_GR_GL(standard)) {
718 fHalfFloatVertexAttributeSupport = (version >= GR_GL_VER(3, 0));
719 } else if (GR_IS_GR_GL_ES(standard)) {
720 fHalfFloatVertexAttributeSupport = (version >= GR_GL_VER(3, 0));
721 } else if (GR_IS_GR_WEBGL(standard)) {
722 // This appears to be supported in 2.0, looking at the spec.
723 fHalfFloatVertexAttributeSupport = (version >= GR_GL_VER(2, 0));
724 }
725
726 fDynamicStateArrayGeometryProcessorTextureSupport = true;
727
728 if (GR_IS_GR_GL(standard)) {
729 fProgramBinarySupport = (version >= GR_GL_VER(4, 1));
730 fProgramParameterSupport = (version >= GR_GL_VER(4, 1));
731 } else if (GR_IS_GR_GL_ES(standard)) {
732 fProgramBinarySupport =
733 (version >= GR_GL_VER(3, 0)) || ctxInfo.hasExtension("GL_OES_get_program_binary");
734 fProgramParameterSupport = (version >= GR_GL_VER(3, 0));
735 } // Explicitly not supported in WebGL 2.0
736 // https://www.khronos.org/registry/webgl/specs/2.0/#5.4
737 if (fProgramBinarySupport) {
738 GrGLint count;
739 GR_GL_GetIntegerv(gli, GR_GL_NUM_PROGRAM_BINARY_FORMATS, &count);
740 fProgramBinarySupport = count > 0;
741 }
742 if (GR_IS_GR_GL(standard)) {
743 fSamplerObjectSupport =
744 version >= GR_GL_VER(3,3) || ctxInfo.hasExtension("GL_ARB_sampler_objects");
745 } else if (GR_IS_GR_GL_ES(standard)) {
746 fSamplerObjectSupport = version >= GR_GL_VER(3,0);
747 } else if (GR_IS_GR_WEBGL(standard)) {
748 fSamplerObjectSupport = version >= GR_GL_VER(2,0);
749 }
750 // We currently use sampler objects whenever they are available.
751 fUseSamplerObjects = fSamplerObjectSupport;
752
753 if (GR_IS_GR_GL_ES(standard)) {
754 fTiledRenderingSupport = ctxInfo.hasExtension("GL_QCOM_tiled_rendering");
755 }
756
757 if (ctxInfo.vendor() == GrGLVendor::kARM) {
758 fShouldCollapseSrcOverToSrcWhenAble = true;
759 }
760
761 #ifdef SK_BUILD_FOR_ANDROID_FRAMEWORK
762 if (ctxInfo.renderer() == GrGLRenderer::kPowerVRRogue) {
763 // https://b/195281495
764 // The TecnoSpark 3 Pro with a PowerVR GE8300 seems to have a steep dithering performance
765 // cliff in the Android Framework
766 fAvoidDithering = true;
767 }
768 #endif
769
770 FormatWorkarounds formatWorkarounds;
771
772 if (!contextOptions.fDisableDriverCorrectnessWorkarounds) {
773 this->applyDriverCorrectnessWorkarounds(ctxInfo, contextOptions, gli, shaderCaps,
774 &formatWorkarounds);
775 }
776
777 // Requires msaa support, ES compatibility have already been detected.
778 this->initFormatTable(ctxInfo, gli, formatWorkarounds);
779
780 this->finishInitialization(contextOptions);
781
782 // For now these two are equivalent but we could have dst read in shader via some other method.
783 shaderCaps->fDstReadInShaderSupport = shaderCaps->fFBFetchSupport;
784 }
785
get_glsl_version_decl_string(GrGLStandard standard,GrGLSLGeneration generation,bool isCoreProfile)786 const char* get_glsl_version_decl_string(GrGLStandard standard, GrGLSLGeneration generation,
787 bool isCoreProfile) {
788 if (GR_IS_GR_GL(standard)) {
789 switch (generation) {
790 case k110_GrGLSLGeneration:
791 return "#version 110\n";
792 case k130_GrGLSLGeneration:
793 return "#version 130\n";
794 case k140_GrGLSLGeneration:
795 return "#version 140\n";
796 case k150_GrGLSLGeneration:
797 if (isCoreProfile) {
798 return "#version 150\n";
799 } else {
800 return "#version 150 compatibility\n";
801 }
802 case k330_GrGLSLGeneration:
803 if (isCoreProfile) {
804 return "#version 330\n";
805 } else {
806 return "#version 330 compatibility\n";
807 }
808 case k400_GrGLSLGeneration:
809 if (isCoreProfile) {
810 return "#version 400\n";
811 } else {
812 return "#version 400 compatibility\n";
813 }
814 case k420_GrGLSLGeneration:
815 if (isCoreProfile) {
816 return "#version 420\n";
817 } else {
818 return "#version 420 compatibility\n";
819 }
820 default:
821 break;
822 }
823 } else if (GR_IS_GR_GL_ES(standard) || GR_IS_GR_WEBGL(standard)) {
824 switch (generation) {
825 case k110_GrGLSLGeneration:
826 // ES2s shader language is based on version 1.20 but is version
827 // 1.00 of the ES language.
828 return "#version 100\n";
829 case k330_GrGLSLGeneration:
830 return "#version 300 es\n";
831 case k310es_GrGLSLGeneration:
832 return "#version 310 es\n";
833 case k320es_GrGLSLGeneration:
834 return "#version 320 es\n";
835 default:
836 break;
837 }
838 }
839 return "<no version>";
840 }
841
is_float_fp32(const GrGLContextInfo & ctxInfo,const GrGLInterface * gli,GrGLenum precision)842 bool is_float_fp32(const GrGLContextInfo& ctxInfo, const GrGLInterface* gli, GrGLenum precision) {
843 if (GR_IS_GR_GL(ctxInfo.standard()) &&
844 ctxInfo.version() < GR_GL_VER(4,1) &&
845 !ctxInfo.hasExtension("GL_ARB_ES2_compatibility")) {
846 // We're on a desktop GL that doesn't have precision info. Assume they're all 32bit float.
847 return true;
848 }
849 // glGetShaderPrecisionFormat doesn't accept GL_GEOMETRY_SHADER as a shader type. Hopefully the
850 // geometry shaders don't have lower precision than vertex and fragment.
851 for (GrGLenum shader : {GR_GL_FRAGMENT_SHADER, GR_GL_VERTEX_SHADER}) {
852 GrGLint range[2];
853 GrGLint bits;
854 GR_GL_GetShaderPrecisionFormat(gli, shader, precision, range, &bits);
855 if (range[0] < 127 || range[1] < 127 || bits < 23) {
856 return false;
857 }
858 }
859 return true;
860 }
861
initGLSL(const GrGLContextInfo & ctxInfo,const GrGLInterface * gli)862 void GrGLCaps::initGLSL(const GrGLContextInfo& ctxInfo, const GrGLInterface* gli) {
863 GrGLStandard standard = ctxInfo.standard();
864 GrGLVersion version = ctxInfo.version();
865
866 /**************************************************************************
867 * Caps specific to GrShaderCaps
868 **************************************************************************/
869
870 GrShaderCaps* shaderCaps = fShaderCaps.get();
871 shaderCaps->fGLSLGeneration = ctxInfo.glslGeneration();
872 if (GR_IS_GR_GL_ES(standard)) {
873 // fFBFetchRequiresEnablePerSample is not a shader cap but is initialized below to keep it
874 // with related FB fetch logic.
875 if (ctxInfo.hasExtension("GL_EXT_shader_framebuffer_fetch")) {
876 shaderCaps->fFBFetchNeedsCustomOutput = (version >= GR_GL_VER(3, 0));
877 shaderCaps->fFBFetchSupport = true;
878 shaderCaps->fFBFetchColorName = "gl_LastFragData[0]";
879 shaderCaps->fFBFetchExtensionString = "GL_EXT_shader_framebuffer_fetch";
880 fFBFetchRequiresEnablePerSample = false;
881 } else if (ctxInfo.hasExtension("GL_NV_shader_framebuffer_fetch")) {
882 // Actually, we haven't seen an ES3.0 device with this extension yet, so we don't know.
883 shaderCaps->fFBFetchNeedsCustomOutput = false;
884 shaderCaps->fFBFetchSupport = true;
885 shaderCaps->fFBFetchColorName = "gl_LastFragData[0]";
886 shaderCaps->fFBFetchExtensionString = "GL_NV_shader_framebuffer_fetch";
887 fFBFetchRequiresEnablePerSample = false;
888 } else if (ctxInfo.hasExtension("GL_ARM_shader_framebuffer_fetch")) {
889 // The arm extension also requires an additional flag which we will set onResetContext.
890 shaderCaps->fFBFetchNeedsCustomOutput = false;
891 shaderCaps->fFBFetchSupport = true;
892 shaderCaps->fFBFetchColorName = "gl_LastFragColorARM";
893 shaderCaps->fFBFetchExtensionString = "GL_ARM_shader_framebuffer_fetch";
894 fFBFetchRequiresEnablePerSample = true;
895 }
896 shaderCaps->fUsesPrecisionModifiers = true;
897 } else if (GR_IS_GR_GL(standard)) {
898 if (ctxInfo.hasExtension("GL_EXT_shader_framebuffer_fetch")) {
899 shaderCaps->fFBFetchNeedsCustomOutput = (version >= GR_GL_VER(3, 0));
900 shaderCaps->fFBFetchSupport = true;
901 shaderCaps->fFBFetchColorName = "gl_LastFragData[0]";
902 shaderCaps->fFBFetchExtensionString = "GL_EXT_shader_framebuffer_fetch";
903 fFBFetchRequiresEnablePerSample = false;
904 }
905 } else if (GR_IS_GR_WEBGL(standard)) {
906 shaderCaps->fUsesPrecisionModifiers = true;
907 }
908
909 if (GR_IS_GR_GL(standard)) {
910 shaderCaps->fFlatInterpolationSupport = ctxInfo.glslGeneration() >= k130_GrGLSLGeneration;
911 } else if (GR_IS_GR_GL_ES(standard) || GR_IS_GR_WEBGL(standard)) {
912 shaderCaps->fFlatInterpolationSupport =
913 ctxInfo.glslGeneration() >= k330_GrGLSLGeneration; // This is the value for GLSL ES 3.0.
914 } // not sure for WebGL
915
916 // Flat interpolation appears to be slow on Qualcomm GPUs (tested Adreno 405 and 530).
917 // Avoid on ANGLE too, it inserts a geometry shader into the pipeline to implement flat interp.
918 // Is this only true on ANGLE's D3D backends or also on the GL backend?
919 shaderCaps->fPreferFlatInterpolation = shaderCaps->fFlatInterpolationSupport &&
920 ctxInfo.vendor() != GrGLVendor::kQualcomm &&
921 ctxInfo.angleBackend() == GrGLANGLEBackend::kUnknown;
922 if (GR_IS_GR_GL(standard)) {
923 shaderCaps->fNoPerspectiveInterpolationSupport =
924 ctxInfo.glslGeneration() >= k130_GrGLSLGeneration;
925 } else if (GR_IS_GR_GL_ES(standard)) {
926 if (ctxInfo.hasExtension("GL_NV_shader_noperspective_interpolation") &&
927 ctxInfo.glslGeneration() >= k330_GrGLSLGeneration /* GLSL ES 3.0 */) {
928 shaderCaps->fNoPerspectiveInterpolationSupport = true;
929 shaderCaps->fNoPerspectiveInterpolationExtensionString =
930 "GL_NV_shader_noperspective_interpolation";
931 }
932 } // Not sure for WebGL
933
934 if (GR_IS_GR_GL(standard)) {
935 shaderCaps->fSampleMaskSupport = ctxInfo.glslGeneration() >= k400_GrGLSLGeneration;
936 } else if (GR_IS_GR_GL_ES(standard)) {
937 if (ctxInfo.glslGeneration() >= k320es_GrGLSLGeneration) {
938 shaderCaps->fSampleMaskSupport = true;
939 } else if (ctxInfo.hasExtension("GL_OES_sample_variables")) {
940 shaderCaps->fSampleMaskSupport = true;
941 shaderCaps->fSampleVariablesExtensionString = "GL_OES_sample_variables";
942 }
943 }
944
945 bool hasTessellationSupport = false;
946 if (GR_IS_GR_GL(standard)) {
947 hasTessellationSupport = version >= GR_GL_VER(4,0) ||
948 ctxInfo.hasExtension("GL_ARB_tessellation_shader");
949 } else if (version >= GR_GL_VER(3,2)) {
950 hasTessellationSupport = true;
951 } else if (ctxInfo.hasExtension("GL_OES_tessellation_shader")) {
952 hasTessellationSupport = true;
953 shaderCaps->fTessellationExtensionString = "GL_OES_tessellation_shader";
954 }
955 if (hasTessellationSupport) {
956 GR_GL_GetIntegerv(gli, GR_GL_MAX_TESS_GEN_LEVEL_OES,
957 &shaderCaps->fMaxTessellationSegments);
958 // Just in case a driver returns a negative number?
959 shaderCaps->fMaxTessellationSegments = std::max(0, shaderCaps->fMaxTessellationSegments);
960 }
961
962 shaderCaps->fVersionDeclString = get_glsl_version_decl_string(standard,
963 shaderCaps->fGLSLGeneration,
964 fIsCoreProfile);
965
966 if (GR_IS_GR_GL_ES(standard) || GR_IS_GR_WEBGL(standard)) {
967 if (k110_GrGLSLGeneration == shaderCaps->fGLSLGeneration) {
968 shaderCaps->fShaderDerivativeExtensionString = "GL_OES_standard_derivatives";
969 }
970 } // WebGL might have to check for OES_standard_derivatives
971
972 if (GR_IS_GR_GL_ES(standard)) {
973 shaderCaps->fSecondaryOutputExtensionString = "GL_EXT_blend_func_extended";
974 }
975
976 if (ctxInfo.hasExtension("GL_OES_EGL_image_external")) {
977 if (ctxInfo.glslGeneration() == k110_GrGLSLGeneration) {
978 shaderCaps->fExternalTextureSupport = true;
979 shaderCaps->fExternalTextureExtensionString = "GL_OES_EGL_image_external";
980 } else if (ctxInfo.hasExtension("GL_OES_EGL_image_external_essl3") ||
981 ctxInfo.hasExtension("OES_EGL_image_external_essl3")) {
982 // At least one driver has been found that has this extension without the "GL_" prefix.
983 shaderCaps->fExternalTextureSupport = true;
984 shaderCaps->fExternalTextureExtensionString = "GL_OES_EGL_image_external_essl3";
985 }
986 }
987
988 if (GR_IS_GR_GL(standard)) {
989 shaderCaps->fVertexIDSupport = true;
990 } else if (GR_IS_GR_GL_ES(standard) || GR_IS_GR_WEBGL(standard)) {
991 // Desktop GLSL 3.30 == ES GLSL 3.00.
992 shaderCaps->fVertexIDSupport = ctxInfo.glslGeneration() >= k330_GrGLSLGeneration;
993 }
994
995 if (GR_IS_GR_GL(standard)) {
996 shaderCaps->fInfinitySupport = shaderCaps->fNonconstantArrayIndexSupport = true;
997 } else if (GR_IS_GR_GL_ES(standard) || GR_IS_GR_WEBGL(standard)) {
998 // Desktop GLSL 3.30 == ES GLSL 3.00.
999 shaderCaps->fInfinitySupport = shaderCaps->fNonconstantArrayIndexSupport =
1000 (ctxInfo.glslGeneration() >= k330_GrGLSLGeneration);
1001 }
1002
1003 if (GR_IS_GR_GL(standard)) {
1004 shaderCaps->fBitManipulationSupport = ctxInfo.glslGeneration() >= k400_GrGLSLGeneration;
1005 } else if (GR_IS_GR_GL_ES(standard) || GR_IS_GR_WEBGL(standard)) {
1006 shaderCaps->fBitManipulationSupport = ctxInfo.glslGeneration() >= k310es_GrGLSLGeneration;
1007 }
1008
1009 shaderCaps->fFloatIs32Bits = is_float_fp32(ctxInfo, gli, GR_GL_HIGH_FLOAT);
1010 shaderCaps->fHalfIs32Bits = is_float_fp32(ctxInfo, gli, GR_GL_MEDIUM_FLOAT);
1011 shaderCaps->fHasLowFragmentPrecision = ctxInfo.renderer() == GrGLRenderer::kMali4xx;
1012
1013 if (GR_IS_GR_GL(standard)) {
1014 shaderCaps->fBuiltinFMASupport = ctxInfo.glslGeneration() >= k400_GrGLSLGeneration;
1015 } else if (GR_IS_GR_GL_ES(standard)) {
1016 shaderCaps->fBuiltinFMASupport = ctxInfo.glslGeneration() >= k320es_GrGLSLGeneration;
1017 }
1018
1019 shaderCaps->fBuiltinDeterminantSupport = ctxInfo.glslGeneration() >= k150_GrGLSLGeneration;
1020
1021 if (GR_IS_GR_WEBGL(standard)) {
1022 // WebGL 1.0 doesn't support do-while loops.
1023 shaderCaps->fCanUseDoLoops = version >= GR_GL_VER(2, 0);
1024 }
1025 }
1026
initFSAASupport(const GrContextOptions & contextOptions,const GrGLContextInfo & ctxInfo,const GrGLInterface * gli)1027 void GrGLCaps::initFSAASupport(const GrContextOptions& contextOptions,
1028 const GrGLContextInfo& ctxInfo, const GrGLInterface* gli) {
1029 if (GR_IS_GR_GL(ctxInfo.standard())) {
1030 if (ctxInfo.version() >= GR_GL_VER(3,0) ||
1031 ctxInfo.hasExtension("GL_ARB_framebuffer_object")) {
1032 fMSFBOType = kStandard_MSFBOType;
1033 } else if (ctxInfo.hasExtension("GL_EXT_framebuffer_multisample") &&
1034 ctxInfo.hasExtension("GL_EXT_framebuffer_blit")) {
1035 fMSFBOType = kStandard_MSFBOType;
1036 }
1037 } else if (GR_IS_GR_GL_ES(ctxInfo.standard())) {
1038 // We prefer multisampled-render-to-texture extensions over ES3 MSAA because we've observed
1039 // ES3 driver bugs on at least one device with a tiled GPU (N10).
1040 if (ctxInfo.hasExtension("GL_EXT_multisampled_render_to_texture")) {
1041 fMSFBOType = kES_EXT_MsToTexture_MSFBOType;
1042 fMSAAResolvesAutomatically = true;
1043 } else if (ctxInfo.hasExtension("GL_IMG_multisampled_render_to_texture")) {
1044 fMSFBOType = kES_IMG_MsToTexture_MSFBOType;
1045 fMSAAResolvesAutomatically = true;
1046 } else if (ctxInfo.version() >= GR_GL_VER(3,0)) {
1047 fMSFBOType = kStandard_MSFBOType;
1048 } else if (ctxInfo.hasExtension("GL_CHROMIUM_framebuffer_multisample")) {
1049 fMSFBOType = kStandard_MSFBOType;
1050 } else if (ctxInfo.hasExtension("GL_ANGLE_framebuffer_multisample")) {
1051 fMSFBOType = kStandard_MSFBOType;
1052 } else if (ctxInfo.hasExtension("GL_APPLE_framebuffer_multisample")) {
1053 fMSFBOType = kES_Apple_MSFBOType;
1054 }
1055 } else if (GR_IS_GR_WEBGL(ctxInfo.standard())) {
1056 // No support in WebGL 1, but there is for 2.0
1057 if (ctxInfo.version() >= GR_GL_VER(2,0)) {
1058 fMSFBOType = kStandard_MSFBOType;
1059 } else {
1060 fMSFBOType = kNone_MSFBOType;
1061 }
1062 }
1063 }
1064
initBlendEqationSupport(const GrGLContextInfo & ctxInfo)1065 void GrGLCaps::initBlendEqationSupport(const GrGLContextInfo& ctxInfo) {
1066 GrShaderCaps* shaderCaps = static_cast<GrShaderCaps*>(fShaderCaps.get());
1067
1068 bool layoutQualifierSupport = false;
1069 if ((GR_IS_GR_GL(fStandard) && shaderCaps->generation() >= k140_GrGLSLGeneration) ||
1070 (GR_IS_GR_GL_ES(fStandard) && shaderCaps->generation() >= k330_GrGLSLGeneration)) {
1071 layoutQualifierSupport = true;
1072 } else if (GR_IS_GR_WEBGL(fStandard)) {
1073 return;
1074 }
1075
1076 if (ctxInfo.hasExtension("GL_NV_blend_equation_advanced_coherent")) {
1077 fBlendEquationSupport = kAdvancedCoherent_BlendEquationSupport;
1078 shaderCaps->fAdvBlendEqInteraction = GrShaderCaps::kAutomatic_AdvBlendEqInteraction;
1079 } else if (ctxInfo.hasExtension("GL_KHR_blend_equation_advanced_coherent") &&
1080 layoutQualifierSupport) {
1081 fBlendEquationSupport = kAdvancedCoherent_BlendEquationSupport;
1082 shaderCaps->fAdvBlendEqInteraction = GrShaderCaps::kGeneralEnable_AdvBlendEqInteraction;
1083 } else if (ctxInfo.hasExtension("GL_NV_blend_equation_advanced")) {
1084 fBlendEquationSupport = kAdvanced_BlendEquationSupport;
1085 shaderCaps->fAdvBlendEqInteraction = GrShaderCaps::kAutomatic_AdvBlendEqInteraction;
1086 } else if (ctxInfo.hasExtension("GL_KHR_blend_equation_advanced") && layoutQualifierSupport) {
1087 fBlendEquationSupport = kAdvanced_BlendEquationSupport;
1088 shaderCaps->fAdvBlendEqInteraction = GrShaderCaps::kGeneralEnable_AdvBlendEqInteraction;
1089 }
1090 }
1091
1092
initStencilSupport(const GrGLContextInfo & ctxInfo)1093 void GrGLCaps::initStencilSupport(const GrGLContextInfo& ctxInfo) {
1094
1095 // Build up list of legal stencil formats (though perhaps not supported on
1096 // the particular gpu/driver) from most preferred to least.
1097
1098 // We push back stencil formats onto the fStencilFormats array in order of most preferred to
1099 // least preferred.
1100
1101 if (GR_IS_GR_GL(ctxInfo.standard())) {
1102 bool supportsPackedDS =
1103 ctxInfo.version() >= GR_GL_VER(3,0) ||
1104 ctxInfo.hasExtension("GL_EXT_packed_depth_stencil") ||
1105 ctxInfo.hasExtension("GL_ARB_framebuffer_object");
1106
1107 // S1 thru S16 formats are in GL 3.0+, EXT_FBO, and ARB_FBO since we
1108 // require FBO support we can expect these are legal formats and don't
1109 // check.
1110 fStencilFormats.push_back() = GrGLFormat::kSTENCIL_INDEX8;
1111 fStencilFormats.push_back() = GrGLFormat::kSTENCIL_INDEX16;
1112 if (supportsPackedDS) {
1113 fStencilFormats.push_back() = GrGLFormat::kDEPTH24_STENCIL8;
1114 }
1115 } else if (GR_IS_GR_GL_ES(ctxInfo.standard())) {
1116 // ES2 has STENCIL_INDEX8 without extensions but requires extensions
1117 // for other formats.
1118
1119 fStencilFormats.push_back() = GrGLFormat::kSTENCIL_INDEX8;
1120 if (ctxInfo.version() >= GR_GL_VER(3,0) ||
1121 ctxInfo.hasExtension("GL_OES_packed_depth_stencil")) {
1122 fStencilFormats.push_back() = GrGLFormat::kDEPTH24_STENCIL8;
1123 }
1124 } else if (GR_IS_GR_WEBGL(ctxInfo.standard())) {
1125 fStencilFormats.push_back() = GrGLFormat::kSTENCIL_INDEX8;
1126 if (ctxInfo.version() >= GR_GL_VER(2,0)) {
1127 fStencilFormats.push_back() = GrGLFormat::kDEPTH24_STENCIL8;
1128 }
1129 }
1130 }
1131
1132 #ifdef SK_ENABLE_DUMP_GPU
1133 #include "src/utils/SkJSONWriter.h"
1134
multi_draw_type_name(GrGLCaps::MultiDrawType multiDrawType)1135 static const char* multi_draw_type_name(GrGLCaps::MultiDrawType multiDrawType) {
1136 switch (multiDrawType) {
1137 case GrGLCaps::MultiDrawType::kNone : return "kNone";
1138 case GrGLCaps::MultiDrawType::kMultiDrawIndirect : return "kMultiDrawIndirect";
1139 case GrGLCaps::MultiDrawType::kANGLEOrWebGL : return "kMultiDrawIndirect";
1140 }
1141 SkUNREACHABLE;
1142 }
1143
onDumpJSON(SkJSONWriter * writer) const1144 void GrGLCaps::onDumpJSON(SkJSONWriter* writer) const {
1145
1146 // We are called by the base class, which has already called beginObject(). We choose to nest
1147 // all of our caps information in a named sub-object.
1148 writer->beginObject("GL caps");
1149
1150 writer->beginArray("Stencil Formats");
1151
1152 for (int i = 0; i < fStencilFormats.count(); ++i) {
1153 writer->beginObject(nullptr, false);
1154 writer->appendS32("stencil bits", GrGLFormatStencilBits(fStencilFormats[i]));
1155 writer->appendS32("total bytes", GrGLFormatBytesPerBlock(fStencilFormats[i]));
1156 writer->endObject();
1157 }
1158
1159 writer->endArray();
1160
1161 static const char* kMSFBOExtStr[] = {
1162 "None",
1163 "Standard",
1164 "Apple",
1165 "IMG MS To Texture",
1166 "EXT MS To Texture",
1167 };
1168 static_assert(0 == kNone_MSFBOType);
1169 static_assert(1 == kStandard_MSFBOType);
1170 static_assert(2 == kES_Apple_MSFBOType);
1171 static_assert(3 == kES_IMG_MsToTexture_MSFBOType);
1172 static_assert(4 == kES_EXT_MsToTexture_MSFBOType);
1173 static_assert(SK_ARRAY_COUNT(kMSFBOExtStr) == kLast_MSFBOType + 1);
1174
1175 static const char* kInvalidateFBTypeStr[] = {
1176 "None",
1177 "Discard",
1178 "Invalidate",
1179 };
1180 static_assert(0 == kNone_InvalidateFBType);
1181 static_assert(1 == kDiscard_InvalidateFBType);
1182 static_assert(2 == kInvalidate_InvalidateFBType);
1183 static_assert(SK_ARRAY_COUNT(kInvalidateFBTypeStr) == kLast_InvalidateFBType + 1);
1184
1185 static const char* kMapBufferTypeStr[] = {
1186 "None",
1187 "MapBuffer",
1188 "MapBufferRange",
1189 "Chromium",
1190 };
1191 static_assert(0 == kNone_MapBufferType);
1192 static_assert(1 == kMapBuffer_MapBufferType);
1193 static_assert(2 == kMapBufferRange_MapBufferType);
1194 static_assert(3 == kChromium_MapBufferType);
1195 static_assert(SK_ARRAY_COUNT(kMapBufferTypeStr) == kLast_MapBufferType + 1);
1196
1197 writer->appendBool("Core Profile", fIsCoreProfile);
1198 writer->appendString("MSAA Type", kMSFBOExtStr[fMSFBOType]);
1199 writer->appendString("Invalidate FB Type", kInvalidateFBTypeStr[fInvalidateFBType]);
1200 writer->appendString("Map Buffer Type", kMapBufferTypeStr[fMapBufferType]);
1201 writer->appendString("Multi Draw Type", multi_draw_type_name(fMultiDrawType));
1202 writer->appendS32("Max FS Uniform Vectors", fMaxFragmentUniformVectors);
1203 writer->appendBool("Pack Flip Y support", fPackFlipYSupport);
1204
1205 writer->appendBool("Texture Usage support", fTextureUsageSupport);
1206 writer->appendBool("GL_ARB_imaging support", fImagingSupport);
1207 writer->appendBool("Vertex array object support", fVertexArrayObjectSupport);
1208 writer->appendBool("Debug support", fDebugSupport);
1209 writer->appendBool("ES2 compatibility support", fES2CompatibilitySupport);
1210 writer->appendBool("drawRangeElements support", fDrawRangeElementsSupport);
1211 writer->appendBool("Base (vertex base) instance support", fBaseVertexBaseInstanceSupport);
1212 writer->appendBool("Bind uniform location support", fBindUniformLocationSupport);
1213 writer->appendBool("Rectangle texture support", fRectangleTextureSupport);
1214 writer->appendBool("Mipmap LOD control support", fMipmapLodControlSupport);
1215 writer->appendBool("Mipmap level control support", fMipmapLevelControlSupport);
1216 writer->appendBool("Use buffer data null hint", fUseBufferDataNullHint);
1217 writer->appendBool("Clear texture support", fClearTextureSupport);
1218 writer->appendBool("Program binary support", fProgramBinarySupport);
1219 writer->appendBool("Program parameters support", fProgramParameterSupport);
1220 writer->appendBool("Sampler object support", fSamplerObjectSupport);
1221 writer->appendBool("Using sampler objects", fUseSamplerObjects);
1222 writer->appendBool("Texture swizzle support", fTextureSwizzleSupport);
1223 writer->appendBool("Tiled rendering support", fTiledRenderingSupport);
1224 writer->appendBool("FB fetch requires enable per sample", fFBFetchRequiresEnablePerSample);
1225 writer->appendBool("sRGB Write Control", fSRGBWriteControl);
1226
1227 writer->appendBool("Intermediate texture for partial updates of unorm textures ever bound to FBOs",
1228 fDisallowTexSubImageForUnormConfigTexturesEverBoundToFBO);
1229 writer->appendBool("Intermediate texture for all updates of textures bound to FBOs",
1230 fUseDrawInsteadOfAllRenderTargetWrites);
1231 writer->appendBool("Max instances per draw without crashing (or zero)",
1232 fMaxInstancesPerDrawWithoutCrashing);
1233
1234 writer->beginArray("formats");
1235
1236 for (int i = 0; i < kGrGLColorFormatCount; ++i) {
1237 writer->beginObject(nullptr, false);
1238 writer->appendHexU32("flags", fFormatTable[i].fFlags);
1239 writer->appendHexU32("f_type", (uint32_t)fFormatTable[i].fFormatType);
1240 writer->appendHexU32("c_internal", fFormatTable[i].fCompressedInternalFormat);
1241 writer->appendHexU32("i_for_teximage", fFormatTable[i].fInternalFormatForTexImageOrStorage);
1242 writer->appendHexU32("i_for_renderbuffer", fFormatTable[i].fInternalFormatForRenderbuffer);
1243 writer->appendHexU32("default_ex_format", fFormatTable[i].fDefaultExternalFormat);
1244 writer->appendHexU32("default_ex_type", fFormatTable[i].fDefaultExternalType);
1245 writer->appendHexU32("default_color_type", (uint32_t)fFormatTable[i].fDefaultColorType);
1246
1247 writer->beginArray("surface color types");
1248 for (int j = 0; j < fFormatTable[i].fColorTypeInfoCount; ++j) {
1249 const auto& ctInfo = fFormatTable[i].fColorTypeInfos[j];
1250 writer->beginObject(nullptr, false);
1251 writer->appendHexU32("colorType", (uint32_t)ctInfo.fColorType);
1252 writer->appendHexU32("flags", ctInfo.fFlags);
1253
1254 writer->beginArray("data color types");
1255 for (int k = 0; k < ctInfo.fExternalIOFormatCount; ++k) {
1256 const auto& ioInfo = ctInfo.fExternalIOFormats[k];
1257 writer->beginObject(nullptr, false);
1258 writer->appendHexU32("colorType", (uint32_t)ioInfo.fColorType);
1259 writer->appendHexU32("ex_type", ioInfo.fExternalType);
1260 writer->appendHexU32("ex_teximage", ioInfo.fExternalTexImageFormat);
1261 writer->appendHexU32("ex_read", ioInfo.fExternalReadFormat);
1262 writer->endObject();
1263 }
1264 writer->endArray();
1265 writer->endObject();
1266 }
1267 writer->endArray();
1268 writer->endObject();
1269 }
1270
1271 writer->endArray();
1272 writer->endObject();
1273 }
1274 #else
onDumpJSON(SkJSONWriter * writer) const1275 void GrGLCaps::onDumpJSON(SkJSONWriter* writer) const { }
1276 #endif
1277
getTexImageExternalFormatAndType(GrGLFormat surfaceFormat,GrGLenum * externalFormat,GrGLenum * externalType) const1278 void GrGLCaps::getTexImageExternalFormatAndType(GrGLFormat surfaceFormat, GrGLenum* externalFormat,
1279 GrGLenum* externalType) const {
1280 const auto& info = this->getFormatInfo(surfaceFormat);
1281 *externalType = info.fDefaultExternalType;
1282 *externalFormat = info.fDefaultExternalFormat;
1283 }
1284
getTexSubImageDefaultFormatTypeAndColorType(GrGLFormat format,GrGLenum * externalFormat,GrGLenum * externalType,GrColorType * colorType) const1285 void GrGLCaps::getTexSubImageDefaultFormatTypeAndColorType(GrGLFormat format,
1286 GrGLenum* externalFormat,
1287 GrGLenum* externalType,
1288 GrColorType* colorType) const {
1289 const auto& info = this->getFormatInfo(format);
1290 *externalType = info.fDefaultExternalType;
1291 *externalFormat = info.fDefaultExternalFormat;
1292 *colorType = info.fDefaultColorType;
1293 }
1294
getTexSubImageExternalFormatAndType(GrGLFormat surfaceFormat,GrColorType surfaceColorType,GrColorType memoryColorType,GrGLenum * externalFormat,GrGLenum * externalType) const1295 void GrGLCaps::getTexSubImageExternalFormatAndType(GrGLFormat surfaceFormat,
1296 GrColorType surfaceColorType,
1297 GrColorType memoryColorType,
1298 GrGLenum* externalFormat,
1299 GrGLenum* externalType) const {
1300 this->getExternalFormat(surfaceFormat, surfaceColorType, memoryColorType,
1301 kTexImage_ExternalFormatUsage, externalFormat, externalType);
1302 }
1303
getReadPixelsFormat(GrGLFormat surfaceFormat,GrColorType surfaceColorType,GrColorType memoryColorType,GrGLenum * externalFormat,GrGLenum * externalType) const1304 void GrGLCaps::getReadPixelsFormat(GrGLFormat surfaceFormat, GrColorType surfaceColorType,
1305 GrColorType memoryColorType, GrGLenum* externalFormat,
1306 GrGLenum* externalType) const {
1307 this->getExternalFormat(surfaceFormat, surfaceColorType, memoryColorType,
1308 kReadPixels_ExternalFormatUsage, externalFormat, externalType);
1309 }
1310
getExternalFormat(GrGLFormat surfaceFormat,GrColorType surfaceColorType,GrColorType memoryColorType,ExternalFormatUsage usage,GrGLenum * externalFormat,GrGLenum * externalType) const1311 void GrGLCaps::getExternalFormat(GrGLFormat surfaceFormat, GrColorType surfaceColorType,
1312 GrColorType memoryColorType, ExternalFormatUsage usage,
1313 GrGLenum* externalFormat, GrGLenum* externalType) const {
1314 SkASSERT(externalFormat && externalType);
1315 *externalFormat = this->getFormatInfo(surfaceFormat).externalFormat(
1316 surfaceColorType, memoryColorType, usage);
1317 *externalType = this->getFormatInfo(surfaceFormat).externalType(
1318 surfaceColorType, memoryColorType);
1319 }
1320
setStencilFormatIndexForFormat(GrGLFormat format,int index)1321 void GrGLCaps::setStencilFormatIndexForFormat(GrGLFormat format, int index) {
1322 SkASSERT(!this->hasStencilFormatBeenDeterminedForFormat(format));
1323 this->getFormatInfo(format).fStencilFormatIndex =
1324 index < 0 ? FormatInfo::kUnsupported_StencilFormatIndex : index;
1325 }
1326
setColorTypeFormat(GrColorType colorType,GrGLFormat format)1327 void GrGLCaps::setColorTypeFormat(GrColorType colorType, GrGLFormat format) {
1328 int idx = static_cast<int>(colorType);
1329 SkASSERT(fColorTypeToFormatTable[idx] == GrGLFormat::kUnknown);
1330 fColorTypeToFormatTable[idx] = format;
1331 }
1332
initFormatTable(const GrGLContextInfo & ctxInfo,const GrGLInterface * gli,const FormatWorkarounds & formatWorkarounds)1333 void GrGLCaps::initFormatTable(const GrGLContextInfo& ctxInfo, const GrGLInterface* gli,
1334 const FormatWorkarounds& formatWorkarounds) {
1335 GrGLStandard standard = ctxInfo.standard();
1336 // standard can be unused (optimized away) if SK_ASSUME_GL_ES is set
1337 sk_ignore_unused_variable(standard);
1338 GrGLVersion version = ctxInfo.version();
1339
1340 uint32_t nonMSAARenderFlags = FormatInfo::kFBOColorAttachment_Flag;
1341 uint32_t msaaRenderFlags = nonMSAARenderFlags;
1342 if (kNone_MSFBOType != fMSFBOType) {
1343 msaaRenderFlags |= FormatInfo::kFBOColorAttachmentWithMSAA_Flag;
1344 }
1345
1346 bool texStorageSupported = false;
1347 if (GR_IS_GR_GL(standard)) {
1348 // The EXT version can apply to either GL or GLES.
1349 texStorageSupported = version >= GR_GL_VER(4,2) ||
1350 ctxInfo.hasExtension("GL_ARB_texture_storage") ||
1351 ctxInfo.hasExtension("GL_EXT_texture_storage");
1352 } else if (GR_IS_GR_GL_ES(standard)) {
1353 texStorageSupported = version >= GR_GL_VER(3,0) ||
1354 ctxInfo.hasExtension("GL_EXT_texture_storage");
1355 } else if (GR_IS_GR_WEBGL(standard)) {
1356 texStorageSupported = version >= GR_GL_VER(2,0);
1357 }
1358 if (fDriverBugWorkarounds.disable_texture_storage) {
1359 texStorageSupported = false;
1360 }
1361 #ifdef SK_BUILD_FOR_ANDROID
1362 // crbug.com/945506. Telemetry reported a memory usage regression for Android Go Chrome/WebView
1363 // when using glTexStorage2D. This appears to affect OOP-R (so not just over command buffer).
1364 if (!formatWorkarounds.fDontDisableTexStorageOnAndroid) {
1365 texStorageSupported = false;
1366 }
1367 #endif
1368
1369 // ES 2.0 requires that the internal/external formats match so we can't use sized internal
1370 // formats for glTexImage until ES 3.0. TODO: Support sized internal formats in WebGL2.
1371 bool texImageSupportsSizedInternalFormat =
1372 (GR_IS_GR_GL(standard) || (GR_IS_GR_GL_ES(standard) && version >= GR_GL_VER(3,0)));
1373
1374 // for now we don't support floating point MSAA on ES
1375 uint32_t fpRenderFlags = (GR_IS_GR_GL(standard)) ? msaaRenderFlags : nonMSAARenderFlags;
1376
1377 for (int i = 0; i < kGrColorTypeCnt; ++i) {
1378 fColorTypeToFormatTable[i] = GrGLFormat::kUnknown;
1379 }
1380
1381 ///////////////////////////////////////////////////////////////////////////
1382
1383 GrGLenum halfFloatType = GR_GL_HALF_FLOAT;
1384 if ((GR_IS_GR_GL_ES(standard) && version < GR_GL_VER(3, 0)) ||
1385 (GR_IS_GR_WEBGL(standard) && version < GR_GL_VER(2, 0))) {
1386 halfFloatType = GR_GL_HALF_FLOAT_OES;
1387 }
1388
1389 // Format: RGBA8
1390 {
1391 FormatInfo& info = this->getFormatInfo(GrGLFormat::kRGBA8);
1392 info.fFormatType = FormatType::kNormalizedFixedPoint;
1393 info.fInternalFormatForRenderbuffer = GR_GL_RGBA8;
1394 info.fDefaultExternalFormat = GR_GL_RGBA;
1395 info.fDefaultExternalType = GR_GL_UNSIGNED_BYTE;
1396 info.fDefaultColorType = GrColorType::kRGBA_8888;
1397 info.fFlags = FormatInfo::kTexturable_Flag | FormatInfo::kTransfers_Flag;
1398 if (GR_IS_GR_GL(standard)) {
1399 info.fFlags |= msaaRenderFlags;
1400 } else if (GR_IS_GR_GL_ES(standard)) {
1401 if (version >= GR_GL_VER(3,0) || ctxInfo.hasExtension("GL_OES_rgb8_rgba8") ||
1402 ctxInfo.hasExtension("GL_ARM_rgba8")) {
1403 info.fFlags |= msaaRenderFlags;
1404 }
1405 } else if (GR_IS_GR_WEBGL(standard)) {
1406 info.fFlags |= msaaRenderFlags;
1407 }
1408
1409 if (texStorageSupported) {
1410 info.fFlags |= FormatInfo::kUseTexStorage_Flag;
1411 info.fInternalFormatForTexImageOrStorage = GR_GL_RGBA8;
1412 } else {
1413 info.fInternalFormatForTexImageOrStorage =
1414 texImageSupportsSizedInternalFormat ? GR_GL_RGBA8 : GR_GL_RGBA;
1415 }
1416
1417 bool supportsBGRAColorType = GR_IS_GR_GL(standard) &&
1418 (version >= GR_GL_VER(1, 2) || ctxInfo.hasExtension("GL_EXT_bgra"));
1419 info.fColorTypeInfoCount = supportsBGRAColorType ? 3 : 2;
1420 info.fColorTypeInfos = std::make_unique<ColorTypeInfo[]>(info.fColorTypeInfoCount);
1421 int ctIdx = 0;
1422 // Format: RGBA8, Surface: kRGBA_8888
1423 {
1424 auto& ctInfo = info.fColorTypeInfos[ctIdx++];
1425 ctInfo.fColorType = GrColorType::kRGBA_8888;
1426 ctInfo.fFlags = ColorTypeInfo::kUploadData_Flag | ColorTypeInfo::kRenderable_Flag;
1427 this->setColorTypeFormat(GrColorType::kRGBA_8888, GrGLFormat::kRGBA8);
1428
1429 // External IO ColorTypes:
1430 ctInfo.fExternalIOFormatCount = 2;
1431 ctInfo.fExternalIOFormats = std::make_unique<ColorTypeInfo::ExternalIOFormats[]>(
1432 ctInfo.fExternalIOFormatCount);
1433 int ioIdx = 0;
1434 // Format: RGBA8, Surface: kRGBA_8888, Data: kRGBA_8888
1435 {
1436 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
1437 ioFormat.fColorType = GrColorType::kRGBA_8888;
1438 ioFormat.fExternalType = GR_GL_UNSIGNED_BYTE;
1439 ioFormat.fExternalTexImageFormat = GR_GL_RGBA;
1440 ioFormat.fExternalReadFormat = GR_GL_RGBA;
1441 }
1442 // Format: RGBA8, Surface: kRGBA_8888, Data: kBGRA_8888
1443 {
1444 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
1445 ioFormat.fColorType = GrColorType::kBGRA_8888;
1446 ioFormat.fExternalType = GR_GL_UNSIGNED_BYTE;
1447 ioFormat.fExternalTexImageFormat = 0; // TODO: Enable this on non-ES GL
1448 ioFormat.fExternalReadFormat =
1449 formatWorkarounds.fDisallowBGRA8ReadPixels ? 0 : GR_GL_BGRA;
1450 // Not guaranteed by ES/WebGL.
1451 ioFormat.fRequiresImplementationReadQuery = !GR_IS_GR_GL(standard);
1452 }
1453 }
1454
1455 // Format: RGBA8, Surface: kBGRA_8888
1456 if (supportsBGRAColorType) {
1457 auto& ctInfo = info.fColorTypeInfos[ctIdx++];
1458 ctInfo.fColorType = GrColorType::kBGRA_8888;
1459 ctInfo.fFlags = ColorTypeInfo::kUploadData_Flag | ColorTypeInfo::kRenderable_Flag;
1460 this->setColorTypeFormat(GrColorType::kBGRA_8888, GrGLFormat::kRGBA8);
1461
1462 // External IO ColorTypes:
1463 ctInfo.fExternalIOFormatCount = 2;
1464 ctInfo.fExternalIOFormats = std::make_unique<ColorTypeInfo::ExternalIOFormats[]>(
1465 ctInfo.fExternalIOFormatCount);
1466 int ioIdx = 0;
1467 // Format: RGBA8, Surface: kBGRA_8888, Data: kBGRA_8888
1468 {
1469 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
1470 ioFormat.fColorType = GrColorType::kBGRA_8888;
1471 ioFormat.fExternalType = GR_GL_UNSIGNED_BYTE;
1472 ioFormat.fExternalTexImageFormat = GR_GL_BGRA;
1473 ioFormat.fExternalReadFormat =
1474 formatWorkarounds.fDisallowBGRA8ReadPixels ? 0 : GR_GL_BGRA;
1475 // Not guaranteed by ES/WebGL.
1476 ioFormat.fRequiresImplementationReadQuery = !GR_IS_GR_GL(standard);
1477 }
1478
1479 // Format: RGBA8, Surface: kBGRA_8888, Data: kRGBA_8888
1480 {
1481 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
1482 ioFormat.fColorType = GrColorType::kRGBA_8888;
1483 ioFormat.fExternalType = GR_GL_UNSIGNED_BYTE;
1484 ioFormat.fExternalTexImageFormat = 0;
1485 ioFormat.fExternalReadFormat = GR_GL_RGBA;
1486 }
1487 }
1488
1489 // Format: RGBA8, Surface: kRGB_888x
1490 {
1491 auto& ctInfo = info.fColorTypeInfos[ctIdx++];
1492 ctInfo.fColorType = GrColorType::kRGB_888x;
1493 ctInfo.fFlags = ColorTypeInfo::kUploadData_Flag;
1494 ctInfo.fReadSwizzle = GrSwizzle::RGB1();
1495
1496 // External IO ColorTypes:
1497 ctInfo.fExternalIOFormatCount = 1;
1498 ctInfo.fExternalIOFormats = std::make_unique<ColorTypeInfo::ExternalIOFormats[]>(
1499 ctInfo.fExternalIOFormatCount);
1500 int ioIdx = 0;
1501 // Format: RGBA8, Surface: kRGB_888x, Data: kRGBA_888x
1502 {
1503 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
1504 ioFormat.fColorType = GrColorType::kRGB_888x;
1505 ioFormat.fExternalType = GR_GL_UNSIGNED_BYTE;
1506 ioFormat.fExternalTexImageFormat = GR_GL_RGBA;
1507 ioFormat.fExternalReadFormat = GR_GL_RGBA;
1508 }
1509 }
1510 }
1511
1512 // Format: R8
1513 {
1514 FormatInfo& info = this->getFormatInfo(GrGLFormat::kR8);
1515 info.fFormatType = FormatType::kNormalizedFixedPoint;
1516 info.fInternalFormatForRenderbuffer = GR_GL_R8;
1517 info.fDefaultExternalFormat = GR_GL_RED;
1518 info.fDefaultExternalType = GR_GL_UNSIGNED_BYTE;
1519 info.fDefaultColorType = GrColorType::kR_8;
1520 bool r8Support = false;
1521 if (GR_IS_GR_GL(standard)) {
1522 r8Support = version >= GR_GL_VER(3, 0) || ctxInfo.hasExtension("GL_ARB_texture_rg");
1523 } else if (GR_IS_GR_GL_ES(standard)) {
1524 r8Support = version >= GR_GL_VER(3, 0) || ctxInfo.hasExtension("GL_EXT_texture_rg");
1525 } else if (GR_IS_GR_WEBGL(standard)) {
1526 r8Support = ctxInfo.version() >= GR_GL_VER(2, 0);
1527 }
1528 if (formatWorkarounds.fDisallowR8ForPowerVRSGX54x) {
1529 r8Support = false;
1530 }
1531
1532 if (r8Support) {
1533 info.fFlags |= FormatInfo::kTexturable_Flag
1534 | FormatInfo::kTransfers_Flag
1535 | msaaRenderFlags;
1536 }
1537
1538 if (texStorageSupported) {
1539 info.fFlags |= FormatInfo::kUseTexStorage_Flag;
1540 info.fInternalFormatForTexImageOrStorage = GR_GL_R8;
1541 } else {
1542 info.fInternalFormatForTexImageOrStorage =
1543 texImageSupportsSizedInternalFormat ? GR_GL_R8 : GR_GL_RED;
1544 }
1545
1546 if (r8Support) {
1547 info.fColorTypeInfoCount = 2;
1548 info.fColorTypeInfos = std::make_unique<ColorTypeInfo[]>(info.fColorTypeInfoCount);
1549 int ctIdx = 0;
1550 // Format: R8, Surface: kAlpha_8
1551 {
1552 auto& ctInfo = info.fColorTypeInfos[ctIdx++];
1553 ctInfo.fColorType = GrColorType::kAlpha_8;
1554 ctInfo.fFlags = ColorTypeInfo::kUploadData_Flag | ColorTypeInfo::kRenderable_Flag;
1555 ctInfo.fReadSwizzle = GrSwizzle("000r");
1556 ctInfo.fWriteSwizzle = GrSwizzle("a000");
1557 this->setColorTypeFormat(GrColorType::kAlpha_8, GrGLFormat::kR8);
1558
1559 // External IO ColorTypes:
1560 ctInfo.fExternalIOFormatCount = 2;
1561 ctInfo.fExternalIOFormats = std::make_unique<ColorTypeInfo::ExternalIOFormats[]>(
1562 ctInfo.fExternalIOFormatCount);
1563 int ioIdx = 0;
1564 // Format: R8, Surface: kAlpha_8, Data: kAlpha_8
1565 {
1566 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
1567 ioFormat.fColorType = GrColorType::kAlpha_8;
1568 ioFormat.fExternalType = GR_GL_UNSIGNED_BYTE;
1569 ioFormat.fExternalTexImageFormat = GR_GL_RED;
1570 ioFormat.fExternalReadFormat = GR_GL_RED;
1571 // Not guaranteed by ES/WebGL.
1572 ioFormat.fRequiresImplementationReadQuery = !GR_IS_GR_GL(standard);
1573 }
1574
1575 // Format: R8, Surface: kAlpha_8, Data: kAlpha_8xxx
1576 {
1577 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
1578 ioFormat.fColorType = GrColorType::kAlpha_8xxx;
1579 ioFormat.fExternalType = GR_GL_UNSIGNED_BYTE;
1580 ioFormat.fExternalTexImageFormat = 0;
1581 ioFormat.fExternalReadFormat = GR_GL_RGBA;
1582 }
1583 }
1584
1585 // Format: R8, Surface: kGray_8
1586 {
1587 auto& ctInfo = info.fColorTypeInfos[ctIdx++];
1588 ctInfo.fColorType = GrColorType::kGray_8;
1589 ctInfo.fFlags = ColorTypeInfo::kUploadData_Flag;
1590 ctInfo.fReadSwizzle = GrSwizzle("rrr1");
1591 this->setColorTypeFormat(GrColorType::kGray_8, GrGLFormat::kR8);
1592
1593 // External IO ColorTypes:
1594 ctInfo.fExternalIOFormatCount = 2;
1595 ctInfo.fExternalIOFormats = std::make_unique<ColorTypeInfo::ExternalIOFormats[]>(
1596 ctInfo.fExternalIOFormatCount);
1597 int ioIdx = 0;
1598 // Format: R8, Surface: kGray_8, Data: kGray_8
1599 {
1600 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
1601 ioFormat.fColorType = GrColorType::kGray_8;
1602 ioFormat.fExternalType = GR_GL_UNSIGNED_BYTE;
1603 ioFormat.fExternalTexImageFormat = GR_GL_RED;
1604 ioFormat.fExternalReadFormat = GR_GL_RED;
1605 // Not guaranteed by ES/WebGL.
1606 ioFormat.fRequiresImplementationReadQuery = !GR_IS_GR_GL(standard);
1607 }
1608
1609 // Format: R8, Surface: kGray_8, Data: kGray_8xxx
1610 {
1611 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
1612 ioFormat.fColorType = GrColorType::kGray_8xxx;
1613 ioFormat.fExternalType = GR_GL_UNSIGNED_BYTE;
1614 ioFormat.fExternalTexImageFormat = 0;
1615 ioFormat.fExternalReadFormat = GR_GL_RGBA;
1616 }
1617 }
1618 }
1619 }
1620
1621 // Format: ALPHA8
1622 {
1623 bool alpha8IsValidForGL = GR_IS_GR_GL(standard) &&
1624 (!fIsCoreProfile || version <= GR_GL_VER(3, 0));
1625 bool alpha8IsValidForGLES = GR_IS_GR_GL_ES(standard);
1626 bool alpha8IsValidForWebGL = GR_IS_GR_WEBGL(standard);
1627
1628 FormatInfo& info = this->getFormatInfo(GrGLFormat::kALPHA8);
1629 info.fFormatType = FormatType::kNormalizedFixedPoint;
1630 // GL_EXT_texture_storage adds GL_ALPHA8 for texture storage. However, ES3 has glTexStorage
1631 // but does not have GL_ALPHA8 (and requires a sized internal format for glTexStorage).
1632 // WebGL never has GL_ALPHA8.
1633 bool alpha8SizedEnumSupported =
1634 alpha8IsValidForGL ||
1635 (alpha8IsValidForGLES && ctxInfo.hasExtension("GL_EXT_texture_storage"));
1636 bool alpha8TexStorageSupported = alpha8SizedEnumSupported && texStorageSupported;
1637
1638 bool alpha8IsRenderable = false;
1639 if (alpha8IsValidForGL) {
1640 // Core profile removes ALPHA8 support.
1641 // OpenGL 3.0+ (and GL_ARB_framebuffer_object) supports ALPHA8 as renderable.
1642 alpha8IsRenderable = ctxInfo.version() >= GR_GL_VER(3, 0) ||
1643 ctxInfo.hasExtension("GL_ARB_framebuffer_object");
1644 }
1645 info.fInternalFormatForRenderbuffer = GR_GL_ALPHA8;
1646 info.fDefaultExternalFormat = GR_GL_ALPHA;
1647 info.fDefaultExternalType = GR_GL_UNSIGNED_BYTE;
1648 info.fDefaultColorType = GrColorType::kAlpha_8;
1649 if (alpha8IsValidForGL || alpha8IsValidForGLES || alpha8IsValidForWebGL) {
1650 info.fFlags = FormatInfo::kTexturable_Flag | FormatInfo::kTransfers_Flag;
1651 }
1652 if (alpha8IsRenderable && alpha8IsValidForGL) {
1653 // We will use ALPHA8 to create MSAA renderbuffers.
1654 SkASSERT(alpha8SizedEnumSupported);
1655 info.fFlags |= msaaRenderFlags;
1656 }
1657 if (alpha8TexStorageSupported) {
1658 info.fFlags |= FormatInfo::kUseTexStorage_Flag;
1659 info.fInternalFormatForTexImageOrStorage = GR_GL_ALPHA8;
1660 } else {
1661 // Even if GL_ALPHA8 is added to ES by GL_EXT_texture_storage it doesn't become legal
1662 // for glTexImage2D.
1663 if (!GR_IS_GR_GL_ES(standard) && texImageSupportsSizedInternalFormat &&
1664 alpha8SizedEnumSupported) {
1665 info.fInternalFormatForTexImageOrStorage = GR_GL_ALPHA8;
1666 } else {
1667 info.fInternalFormatForTexImageOrStorage = GR_GL_ALPHA;
1668 }
1669 }
1670
1671 if (alpha8IsValidForGL || alpha8IsValidForGLES || alpha8IsValidForWebGL) {
1672 info.fColorTypeInfoCount = 1;
1673 info.fColorTypeInfos = std::make_unique<ColorTypeInfo[]>(info.fColorTypeInfoCount);
1674 int ctIdx = 0;
1675 // Format: ALPHA8, Surface: kAlpha_8
1676 {
1677 if (alpha8IsValidForGL || alpha8IsValidForGLES || alpha8IsValidForWebGL) {
1678 auto& ctInfo = info.fColorTypeInfos[ctIdx++];
1679 ctInfo.fColorType = GrColorType::kAlpha_8;
1680 ctInfo.fFlags = ColorTypeInfo::kUploadData_Flag |
1681 ColorTypeInfo::kRenderable_Flag;
1682 int idx = static_cast<int>(GrColorType::kAlpha_8);
1683 if (fColorTypeToFormatTable[idx] == GrGLFormat::kUnknown) {
1684 this->setColorTypeFormat(GrColorType::kAlpha_8, GrGLFormat::kALPHA8);
1685 }
1686
1687 // External IO ColorTypes:
1688 ctInfo.fExternalIOFormatCount = 2;
1689 ctInfo.fExternalIOFormats = std::make_unique<ColorTypeInfo::ExternalIOFormats[]>(
1690 ctInfo.fExternalIOFormatCount);
1691 int ioIdx = 0;
1692 // Format: ALPHA8, Surface: kAlpha_8, Data: kAlpha_8
1693 {
1694 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
1695 ioFormat.fColorType = GrColorType::kAlpha_8;
1696 ioFormat.fExternalType = GR_GL_UNSIGNED_BYTE;
1697 ioFormat.fExternalTexImageFormat = GR_GL_ALPHA;
1698 ioFormat.fExternalReadFormat = GR_GL_ALPHA;
1699 // Not guaranteed by ES/WebGL.
1700 ioFormat.fRequiresImplementationReadQuery = !GR_IS_GR_GL(standard);
1701 }
1702
1703 // Format: ALPHA8, Surface: kAlpha_8, Data: kRGBA_8888
1704 {
1705 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
1706 ioFormat.fColorType = GrColorType::kRGBA_8888;
1707 ioFormat.fExternalType = GR_GL_UNSIGNED_BYTE;
1708 ioFormat.fExternalTexImageFormat = 0;
1709 ioFormat.fExternalReadFormat = GR_GL_RGBA;
1710 }
1711 }
1712 }
1713 }
1714 }
1715
1716 // Format: LUMINANCE8
1717 {
1718 FormatInfo& info = this->getFormatInfo(GrGLFormat::kLUMINANCE8);
1719 info.fFormatType = FormatType::kNormalizedFixedPoint;
1720 info.fInternalFormatForRenderbuffer = GR_GL_LUMINANCE8;
1721 info.fDefaultExternalFormat = GR_GL_LUMINANCE;
1722 info.fDefaultExternalType = GR_GL_UNSIGNED_BYTE;
1723 info.fDefaultColorType = GrColorType::kGray_8;
1724 bool lum8Supported = false;
1725 bool lum8SizedFormatSupported = false;
1726 if (GR_IS_GR_GL(standard) && !fIsCoreProfile) {
1727 lum8Supported = true;
1728 lum8SizedFormatSupported = true;
1729 } else if (GR_IS_GR_GL_ES(standard)) {
1730 lum8Supported = true;
1731 // Even on ES3 this extension is required to define LUMINANCE8.
1732 lum8SizedFormatSupported = ctxInfo.hasExtension("GL_EXT_texture_storage");
1733 } else if (GR_IS_GR_WEBGL(standard)) {
1734 lum8Supported = true;
1735 }
1736 if (lum8Supported) {
1737 info.fFlags = FormatInfo::kTexturable_Flag | FormatInfo::kTransfers_Flag;
1738 }
1739 if (texStorageSupported && lum8SizedFormatSupported) {
1740 info.fFlags |= FormatInfo::kUseTexStorage_Flag;
1741 info.fInternalFormatForTexImageOrStorage = GR_GL_LUMINANCE8;
1742 } else if (texImageSupportsSizedInternalFormat && lum8SizedFormatSupported) {
1743 info.fInternalFormatForTexImageOrStorage = GR_GL_LUMINANCE8;
1744 } else {
1745 info.fInternalFormatForTexImageOrStorage = GR_GL_LUMINANCE;
1746 }
1747 // We are not enabling attaching to an FBO for LUMINANCE8 mostly because of confusion in the
1748 // spec. For GLES it does not seem to ever support LUMINANCE8 being color-renderable. For GL
1749 // versions less than 3.0 it is provided by GL_ARB_framebuffer_object. However, the original
1750 // version of that extension did not add LUMINANCE8, but was added in a later revsion. So
1751 // even the presence of that extension does not guarantee support. GL 3.0 and higher (core
1752 // or compatibility) do not list LUMINANCE8 as color-renderable (which is strange since the
1753 // GL_ARB_framebuffer_object extension was meant to bring 3.0 functionality to lower
1754 // versions).
1755
1756 if (lum8Supported) {
1757 info.fColorTypeInfoCount = 1;
1758 info.fColorTypeInfos = std::make_unique<ColorTypeInfo[]>(info.fColorTypeInfoCount);
1759 int ctIdx = 0;
1760 // Format: LUMINANCE8, Surface: kGray_8
1761 {
1762 auto& ctInfo = info.fColorTypeInfos[ctIdx++];
1763 ctInfo.fColorType = GrColorType::kGray_8;
1764 ctInfo.fFlags = ColorTypeInfo::kUploadData_Flag;
1765 int idx = static_cast<int>(GrColorType::kGray_8);
1766 if (fColorTypeToFormatTable[idx] == GrGLFormat::kUnknown) {
1767 this->setColorTypeFormat(GrColorType::kGray_8, GrGLFormat::kLUMINANCE8);
1768 }
1769
1770 // External IO ColorTypes:
1771 ctInfo.fExternalIOFormatCount = 2;
1772 ctInfo.fExternalIOFormats = std::make_unique<ColorTypeInfo::ExternalIOFormats[]>(
1773 ctInfo.fExternalIOFormatCount);
1774 int ioIdx = 0;
1775 // Format: LUMINANCE8, Surface: kGray_8, Data: kGray_8
1776 {
1777 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
1778 ioFormat.fColorType = GrColorType::kGray_8;
1779 ioFormat.fExternalType = GR_GL_UNSIGNED_BYTE;
1780 ioFormat.fExternalTexImageFormat = GR_GL_LUMINANCE;
1781 ioFormat.fExternalReadFormat = 0;
1782 }
1783
1784 // Format: LUMINANCE8, Surface: kGray_8, Data: kRGBA_8888
1785 {
1786 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
1787 ioFormat.fColorType = GrColorType::kRGBA_8888;
1788 ioFormat.fExternalType = GR_GL_UNSIGNED_BYTE;
1789 ioFormat.fExternalTexImageFormat = 0;
1790 ioFormat.fExternalReadFormat = GR_GL_RGBA;
1791 }
1792 }
1793 }
1794 }
1795
1796 // Format: LUMINANCE8_ALPHA8
1797 {
1798 FormatInfo& info = this->getFormatInfo(GrGLFormat::kLUMINANCE8_ALPHA8);
1799 info.fFormatType = FormatType::kNormalizedFixedPoint;
1800 info.fInternalFormatForRenderbuffer = GR_GL_LUMINANCE8_ALPHA8;
1801 info.fDefaultExternalFormat = GR_GL_LUMINANCE_ALPHA;
1802 info.fDefaultExternalType = GR_GL_UNSIGNED_BYTE;
1803 info.fDefaultColorType = GrColorType::kGrayAlpha_88;
1804 bool la8Supported = false;
1805 bool la8SizedFormatSupported = false;
1806 if (GR_IS_GR_GL(standard) && !fIsCoreProfile) {
1807 la8Supported = true;
1808 la8SizedFormatSupported = true;
1809 } else if (GR_IS_GR_GL_ES(standard)) {
1810 la8Supported = true;
1811 // Even on ES3 this extension is required to define LUMINANCE8_ALPHA8.
1812 la8SizedFormatSupported = ctxInfo.hasExtension("GL_EXT_texture_storage");
1813 } else if (GR_IS_GR_WEBGL(standard)) {
1814 la8Supported = true;
1815 }
1816 if (la8Supported) {
1817 info.fFlags = FormatInfo::kTexturable_Flag | FormatInfo::kTransfers_Flag;
1818 }
1819 if (texStorageSupported && la8SizedFormatSupported) {
1820 info.fFlags |= FormatInfo::kUseTexStorage_Flag;
1821 info.fInternalFormatForTexImageOrStorage = GR_GL_LUMINANCE8_ALPHA8;
1822 } else if (texImageSupportsSizedInternalFormat && la8SizedFormatSupported) {
1823 info.fInternalFormatForTexImageOrStorage = GR_GL_LUMINANCE8_ALPHA8;
1824 } else {
1825 info.fInternalFormatForTexImageOrStorage = GR_GL_LUMINANCE_ALPHA;
1826 }
1827 // See note in LUMINANCE8 section about not attaching to framebuffers.
1828
1829 if (la8Supported) {
1830 info.fColorTypeInfoCount = 1;
1831 info.fColorTypeInfos = std::make_unique<ColorTypeInfo[]>(info.fColorTypeInfoCount);
1832 int ctIdx = 0;
1833 // Format: LUMINANCE8_ALPHA8, Surface: kGrayAlpha_88
1834 {
1835 auto& ctInfo = info.fColorTypeInfos[ctIdx++];
1836 ctInfo.fColorType = GrColorType::kGrayAlpha_88;
1837 ctInfo.fFlags = ColorTypeInfo::kUploadData_Flag;
1838 int idx = static_cast<int>(GrColorType::kGrayAlpha_88);
1839 if (fColorTypeToFormatTable[idx] == GrGLFormat::kUnknown) {
1840 this->setColorTypeFormat(GrColorType::kGrayAlpha_88,
1841 GrGLFormat::kLUMINANCE8_ALPHA8);
1842 }
1843
1844 // External IO ColorTypes:
1845 ctInfo.fExternalIOFormatCount = 2;
1846 ctInfo.fExternalIOFormats = std::make_unique<ColorTypeInfo::ExternalIOFormats[]>(
1847 ctInfo.fExternalIOFormatCount);
1848 int ioIdx = 0;
1849 // Format: LUMINANCE8, Surface: kGrayAlpha_88, Data: kGrayAlpha_88
1850 {
1851 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
1852 ioFormat.fColorType = GrColorType::kGrayAlpha_88;
1853 ioFormat.fExternalType = GR_GL_UNSIGNED_BYTE;
1854 ioFormat.fExternalTexImageFormat = GR_GL_LUMINANCE_ALPHA;
1855 ioFormat.fExternalReadFormat = 0;
1856 }
1857
1858 // Format: LUMINANCE8, Surface: kGrayAlpha_88, Data: kRGBA_8888
1859 {
1860 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
1861 ioFormat.fColorType = GrColorType::kRGBA_8888;
1862 ioFormat.fExternalType = GR_GL_UNSIGNED_BYTE;
1863 ioFormat.fExternalTexImageFormat = 0;
1864 ioFormat.fExternalReadFormat = GR_GL_RGBA;
1865 }
1866 }
1867 }
1868 }
1869 // Format: BGRA8
1870 {
1871 FormatInfo& info = this->getFormatInfo(GrGLFormat::kBGRA8);
1872 info.fFormatType = FormatType::kNormalizedFixedPoint;
1873
1874 // We currently only use the renderbuffer format when allocating msaa renderbuffers, so we
1875 // are making decisions here based on that use case. The GL_EXT_texture_format_BGRA8888
1876 // extension adds BGRA color renderbuffer support for ES 2.0, but this does not guarantee
1877 // support for MSAA renderbuffers. Additionally, the renderable support was added in a later
1878 // revision of the extension. So it is possible for older drivers to support the extension
1879 // but only an early revision of it without renderable support. We have no way of
1880 // distinguishing between the two. The GL_APPLE_texture_format_BGRA8888 does not add support
1881 // for BGRA color renderbuffers at all. Ideally, for both cases we would use RGBA8 for our
1882 // format for the MSAA buffer. In the GL_EXT_texture_format_BGRA8888 case we can still
1883 // make the resolve BGRA and which will work for glBlitFramebuffer for resolving which just
1884 // requires the src and dst be bindable to FBOs. However, we can't do this in the current
1885 // world since some devices (e.g. chromium & angle) require the formats in glBlitFramebuffer
1886 // to match. We don't have a way to really check this during resolve since we only actually
1887 // have GrBackendFormat that is shared by the GrGLRenderTarget. We always set the
1888 // renderbuffer format to RGBA8 but disable MSAA unless we have the APPLE extension.
1889 // Once we break those up into different surface we can revisit doing this change.
1890 info.fInternalFormatForRenderbuffer = GR_GL_RGBA8;
1891
1892 info.fDefaultExternalFormat = GR_GL_BGRA;
1893 info.fDefaultExternalType = GR_GL_UNSIGNED_BYTE;
1894 info.fDefaultColorType = GrColorType::kBGRA_8888;
1895
1896 GrGLenum bgraTexImageFormat;
1897 // If BGRA is supported as an internal format it must always be specified to glTex[Sub]Image
1898 // as a base format. Which base format depends on which extension is used.
1899 if (ctxInfo.hasExtension("GL_EXT_texture_format_BGRA8888")) {
1900 // GL_EXT_texture_format_BGRA8888:
1901 // This extension adds GL_BGRA as an unsized internal format. However, it is
1902 // written against ES 2.0 and therefore doesn't define a GL_BGRA8 as ES 2.0 doesn't
1903 // have sized internal formats. See later where we check for tex storage BGRA8
1904 // support.
1905 bgraTexImageFormat = GR_GL_BGRA;
1906 } else {
1907 // GL_APPLE_texture_format_BGRA8888:
1908 // ES 2.0: the extension makes BGRA an external format but not an internal format.
1909 // ES 3.0: the extension explicitly states GL_BGRA8 is not a valid internal format
1910 // for glTexImage (just for glTexStorage).
1911 bgraTexImageFormat = GR_GL_RGBA;
1912 }
1913
1914 // TexStorage requires using a sized internal format and BGRA8 is only supported if we have
1915 // the GL_APPLE_texture_format_BGRA8888 extension or if we have GL_EXT_texture_storage and
1916 // GL_EXT_texture_format_BGRA8888.
1917 bool supportsBGRATexStorage = false;
1918
1919 if (GR_IS_GR_GL_ES(standard)) {
1920 if (ctxInfo.hasExtension("GL_EXT_texture_format_BGRA8888")) {
1921 info.fFlags = FormatInfo::kTexturable_Flag
1922 | FormatInfo::kTransfers_Flag
1923 | nonMSAARenderFlags;
1924 // GL_EXT_texture storage has defined interactions with
1925 // GL_EXT_texture_format_BGRA8888. However, ES3 supports glTexStorage but
1926 // without GL_EXT_texture_storage it does not allow the BGRA8 sized internal format.
1927 if (ctxInfo.hasExtension("GL_EXT_texture_storage") &&
1928 !formatWorkarounds.fDisableBGRATextureStorageForIntelWindowsES) {
1929 supportsBGRATexStorage = true;
1930 }
1931 } else if (ctxInfo.hasExtension("GL_APPLE_texture_format_BGRA8888")) {
1932 // This APPLE extension introduces complexity on ES2. It leaves the internal format
1933 // as RGBA, but allows BGRA as the external format. From testing, it appears that
1934 // the driver remembers the external format when the texture is created (with
1935 // TexImage). If you then try to upload data in the other swizzle (with
1936 // TexSubImage), it fails. We could work around this, but it adds even more state
1937 // tracking to code that is already too tricky. Instead, we opt not to support BGRA
1938 // on ES2 with this extension. This also side-steps some ambiguous interactions with
1939 // the texture storage extension.
1940 if (version >= GR_GL_VER(3,0)) {
1941 // The APPLE extension doesn't explicitly make this renderable, but
1942 // internally it appears to use RGBA8, which we'll patch up below.
1943 info.fFlags = FormatInfo::kTexturable_Flag
1944 | FormatInfo::kTransfers_Flag
1945 | msaaRenderFlags;
1946 supportsBGRATexStorage = true;
1947 }
1948 }
1949 }
1950 if (texStorageSupported && supportsBGRATexStorage) {
1951 info.fFlags |= FormatInfo::kUseTexStorage_Flag;
1952 info.fInternalFormatForTexImageOrStorage = GR_GL_BGRA8;
1953 } else {
1954 info.fInternalFormatForTexImageOrStorage = bgraTexImageFormat;
1955 }
1956
1957 if (SkToBool(info.fFlags & FormatInfo::kTexturable_Flag)) {
1958 info.fColorTypeInfoCount = 1;
1959 info.fColorTypeInfos = std::make_unique<ColorTypeInfo[]>(info.fColorTypeInfoCount);
1960 int ctIdx = 0;
1961 // Format: BGRA8, Surface: kBGRA_8888
1962 {
1963 auto& ctInfo = info.fColorTypeInfos[ctIdx++];
1964 ctInfo.fColorType = GrColorType::kBGRA_8888;
1965 ctInfo.fFlags = ColorTypeInfo::kUploadData_Flag | ColorTypeInfo::kRenderable_Flag;
1966 this->setColorTypeFormat(GrColorType::kBGRA_8888, GrGLFormat::kBGRA8);
1967
1968 // External IO ColorTypes:
1969 ctInfo.fExternalIOFormatCount = 2;
1970 ctInfo.fExternalIOFormats = std::make_unique<ColorTypeInfo::ExternalIOFormats[]>(
1971 ctInfo.fExternalIOFormatCount);
1972 int ioIdx = 0;
1973 // Format: BGRA8, Surface: kBGRA_8888, Data: kBGRA_8888
1974 {
1975 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
1976 ioFormat.fColorType = GrColorType::kBGRA_8888;
1977 ioFormat.fExternalType = GR_GL_UNSIGNED_BYTE;
1978 ioFormat.fExternalTexImageFormat = GR_GL_BGRA;
1979 ioFormat.fExternalReadFormat = 0;
1980 ioFormat.fExternalReadFormat =
1981 formatWorkarounds.fDisallowBGRA8ReadPixels ? 0 : GR_GL_BGRA;
1982 // Not guaranteed by ES/WebGL.
1983 ioFormat.fRequiresImplementationReadQuery = !GR_IS_GR_GL(standard);
1984 }
1985
1986 // Format: BGRA8, Surface: kBGRA_8888, Data: kRGBA_8888
1987 {
1988 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
1989 ioFormat.fColorType = GrColorType::kRGBA_8888;
1990 ioFormat.fExternalType = GR_GL_UNSIGNED_BYTE;
1991 ioFormat.fExternalTexImageFormat = 0;
1992 ioFormat.fExternalReadFormat = GR_GL_RGBA;
1993 }
1994 }
1995 }
1996 }
1997
1998 // Format: RGB565
1999 {
2000 FormatInfo& info = this->getFormatInfo(GrGLFormat::kRGB565);
2001 info.fFormatType = FormatType::kNormalizedFixedPoint;
2002 info.fInternalFormatForRenderbuffer = GR_GL_RGB565;
2003 info.fDefaultExternalFormat = GR_GL_RGB;
2004 info.fDefaultExternalType = GR_GL_UNSIGNED_SHORT_5_6_5;
2005 info.fDefaultColorType = GrColorType::kBGR_565;
2006 if (GR_IS_GR_GL(standard)) {
2007 if (version >= GR_GL_VER(4, 2) || ctxInfo.hasExtension("GL_ARB_ES2_compatibility")) {
2008 info.fFlags = FormatInfo::kTexturable_Flag
2009 | FormatInfo::kTransfers_Flag
2010 | msaaRenderFlags;
2011 }
2012 } else if (GR_IS_GR_GL_ES(standard)) {
2013 info.fFlags = FormatInfo::kTexturable_Flag
2014 | FormatInfo::kTransfers_Flag
2015 | msaaRenderFlags;
2016 } else if (GR_IS_GR_WEBGL(standard)) {
2017 info.fFlags = FormatInfo::kTexturable_Flag
2018 | FormatInfo::kTransfers_Flag
2019 | msaaRenderFlags;
2020 }
2021 // 565 is not a sized internal format on desktop GL. So on desktop with
2022 // 565 we always use an unsized internal format to let the system pick
2023 // the best sized format to convert the 565 data to. Since TexStorage
2024 // only allows sized internal formats we disallow it.
2025 //
2026 // TODO: As of 4.2, regular GL supports 565. This logic is due for an
2027 // update.
2028 if (texStorageSupported && GR_IS_GR_GL_ES(standard)) {
2029 info.fFlags |= FormatInfo::kUseTexStorage_Flag;
2030 info.fInternalFormatForTexImageOrStorage = GR_GL_RGB565;
2031 } else {
2032 info.fInternalFormatForTexImageOrStorage =
2033 texImageSupportsSizedInternalFormat ? GR_GL_RGB565 : GR_GL_RGB;
2034 }
2035
2036 if (SkToBool(info.fFlags &FormatInfo::kTexturable_Flag)) {
2037 info.fColorTypeInfoCount = 1;
2038 info.fColorTypeInfos = std::make_unique<ColorTypeInfo[]>(info.fColorTypeInfoCount);
2039 int ctIdx = 0;
2040 // Format: RGB565, Surface: kBGR_565
2041 {
2042 auto& ctInfo = info.fColorTypeInfos[ctIdx++];
2043 ctInfo.fColorType = GrColorType::kBGR_565;
2044 ctInfo.fFlags = ColorTypeInfo::kUploadData_Flag | ColorTypeInfo::kRenderable_Flag;
2045 this->setColorTypeFormat(GrColorType::kBGR_565, GrGLFormat::kRGB565);
2046
2047 // External IO ColorTypes:
2048 ctInfo.fExternalIOFormatCount = 2;
2049 ctInfo.fExternalIOFormats = std::make_unique<ColorTypeInfo::ExternalIOFormats[]>(
2050 ctInfo.fExternalIOFormatCount);
2051 int ioIdx = 0;
2052 // Format: RGB565, Surface: kBGR_565, Data: kBGR_565
2053 {
2054 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
2055 ioFormat.fColorType = GrColorType::kBGR_565;
2056 ioFormat.fExternalType = GR_GL_UNSIGNED_SHORT_5_6_5;
2057 ioFormat.fExternalTexImageFormat = GR_GL_RGB;
2058 ioFormat.fExternalReadFormat = GR_GL_RGB;
2059 // Not guaranteed by ES/WebGL.
2060 ioFormat.fRequiresImplementationReadQuery = !GR_IS_GR_GL(standard);
2061 }
2062
2063 // Format: RGB565, Surface: kBGR_565, Data: kRGBA_8888
2064 {
2065 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
2066 ioFormat.fColorType = GrColorType::kRGBA_8888;
2067 ioFormat.fExternalType = GR_GL_UNSIGNED_BYTE;
2068 ioFormat.fExternalTexImageFormat = 0;
2069 ioFormat.fExternalReadFormat = GR_GL_RGBA;
2070 }
2071 }
2072 }
2073 }
2074
2075 // Format: RGBA16F
2076 {
2077 FormatInfo& info = this->getFormatInfo(GrGLFormat::kRGBA16F);
2078 info.fFormatType = FormatType::kFloat;
2079 info.fInternalFormatForRenderbuffer = GR_GL_RGBA16F;
2080 info.fDefaultExternalFormat = GR_GL_RGBA;
2081 info.fDefaultExternalType = halfFloatType;
2082 info.fDefaultColorType = GrColorType::kRGBA_F16;
2083 bool rgba16FTextureSupport = false;
2084 bool rgba16FRenderTargetSupport = false;
2085
2086 if (GR_IS_GR_GL(standard)) {
2087 if (version >= GR_GL_VER(3, 0)) {
2088 rgba16FTextureSupport = true;
2089 rgba16FRenderTargetSupport = true;
2090 } else if (ctxInfo.hasExtension("GL_ARB_texture_float")) {
2091 rgba16FTextureSupport = true;
2092 }
2093 } else if (GR_IS_GR_GL_ES(standard)) {
2094 if (version >= GR_GL_VER(3, 0)) {
2095 rgba16FTextureSupport = true;
2096 rgba16FRenderTargetSupport =
2097 version >= GR_GL_VER(3, 2) ||
2098 ctxInfo.hasExtension("GL_EXT_color_buffer_half_float") ||
2099 ctxInfo.hasExtension("GL_EXT_color_buffer_float");
2100 } else if (ctxInfo.hasExtension("GL_OES_texture_half_float") &&
2101 ctxInfo.hasExtension("GL_OES_texture_half_float_linear")) {
2102 rgba16FTextureSupport = true;
2103 rgba16FRenderTargetSupport = ctxInfo.hasExtension("GL_EXT_color_buffer_half_float");
2104 }
2105 } else if (GR_IS_GR_WEBGL(standard)) {
2106 if (version >= GR_GL_VER(2, 0)) {
2107 rgba16FTextureSupport = true;
2108 rgba16FRenderTargetSupport =
2109 ctxInfo.hasExtension("GL_EXT_color_buffer_half_float") ||
2110 ctxInfo.hasExtension("EXT_color_buffer_half_float") ||
2111 ctxInfo.hasExtension("GL_EXT_color_buffer_float") ||
2112 ctxInfo.hasExtension("EXT_color_buffer_float");
2113 } else if ((ctxInfo.hasExtension("GL_OES_texture_half_float") ||
2114 ctxInfo.hasExtension("OES_texture_half_float")) &&
2115 (ctxInfo.hasExtension("GL_OES_texture_half_float_linear") ||
2116 ctxInfo.hasExtension("OES_texture_half_float_linear"))) {
2117 rgba16FTextureSupport = true;
2118 // We don't check for EXT_color_buffer_float as it's only defined for WebGL 2.
2119 rgba16FRenderTargetSupport =
2120 ctxInfo.hasExtension("GL_EXT_color_buffer_half_float") ||
2121 ctxInfo.hasExtension("EXT_color_buffer_half_float");
2122 }
2123 }
2124
2125 if (rgba16FTextureSupport) {
2126 info.fFlags = FormatInfo::kTexturable_Flag | FormatInfo::kTransfers_Flag;
2127 if (rgba16FRenderTargetSupport) {
2128 info.fFlags |= fpRenderFlags;
2129 }
2130 }
2131 if (texStorageSupported && !formatWorkarounds.fDisableRGBA16FTexStorageForCrBug1008003) {
2132 info.fFlags |= FormatInfo::kUseTexStorage_Flag;
2133 info.fInternalFormatForTexImageOrStorage = GR_GL_RGBA16F;
2134 } else {
2135 info.fInternalFormatForTexImageOrStorage =
2136 texImageSupportsSizedInternalFormat ? GR_GL_RGBA16F : GR_GL_RGBA;
2137 }
2138
2139 if (rgba16FTextureSupport) {
2140 uint32_t flags = ColorTypeInfo::kUploadData_Flag | ColorTypeInfo::kRenderable_Flag;
2141
2142 info.fColorTypeInfoCount = 2;
2143 info.fColorTypeInfos = std::make_unique<ColorTypeInfo[]>(info.fColorTypeInfoCount);
2144 int ctIdx = 0;
2145 // Format: RGBA16F, Surface: kRGBA_F16
2146 {
2147 auto& ctInfo = info.fColorTypeInfos[ctIdx++];
2148 ctInfo.fColorType = GrColorType::kRGBA_F16;
2149 ctInfo.fFlags = flags;
2150 this->setColorTypeFormat(GrColorType::kRGBA_F16, GrGLFormat::kRGBA16F);
2151
2152 // External IO ColorTypes:
2153 ctInfo.fExternalIOFormatCount = 2;
2154 ctInfo.fExternalIOFormats = std::make_unique<ColorTypeInfo::ExternalIOFormats[]>(
2155 ctInfo.fExternalIOFormatCount);
2156 int ioIdx = 0;
2157 // Format: RGBA16F, Surface: kRGBA_F16, Data: kRGBA_F16
2158 {
2159 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
2160 ioFormat.fColorType = GrColorType::kRGBA_F16;
2161 ioFormat.fExternalType = halfFloatType;
2162 ioFormat.fExternalTexImageFormat = GR_GL_RGBA;
2163 ioFormat.fExternalReadFormat = GR_GL_RGBA;
2164 // Not guaranteed by ES/WebGL.
2165 ioFormat.fRequiresImplementationReadQuery = !GR_IS_GR_GL(standard);
2166 }
2167
2168 // Format: RGBA16F, Surface: kRGBA_F16, Data: kRGBA_F32
2169 {
2170 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
2171 ioFormat.fColorType = GrColorType::kRGBA_F32;
2172 ioFormat.fExternalType = GR_GL_FLOAT;
2173 ioFormat.fExternalTexImageFormat = 0;
2174 ioFormat.fExternalReadFormat = GR_GL_RGBA;
2175 }
2176 }
2177
2178 // Format: RGBA16F, Surface: kRGBA_F16_Clamped
2179 {
2180 auto& ctInfo = info.fColorTypeInfos[ctIdx++];
2181 ctInfo.fColorType = GrColorType::kRGBA_F16_Clamped;
2182 ctInfo.fFlags = flags;
2183 this->setColorTypeFormat(GrColorType::kRGBA_F16_Clamped, GrGLFormat::kRGBA16F);
2184
2185 // External IO ColorTypes:
2186 ctInfo.fExternalIOFormatCount = 2;
2187 ctInfo.fExternalIOFormats = std::make_unique<ColorTypeInfo::ExternalIOFormats[]>(
2188 ctInfo.fExternalIOFormatCount);
2189 int ioIdx = 0;
2190 // Format: RGBA16F, Surface: kRGBA_F16_Clamped, Data: kRGBA_F16_Clamped
2191 {
2192 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
2193 ioFormat.fColorType = GrColorType::kRGBA_F16_Clamped;
2194 ioFormat.fExternalType = halfFloatType;
2195 ioFormat.fExternalTexImageFormat = GR_GL_RGBA;
2196 ioFormat.fExternalReadFormat = GR_GL_RGBA;
2197 // Not guaranteed by ES/WebGL.
2198 ioFormat.fRequiresImplementationReadQuery = !GR_IS_GR_GL(standard);
2199 }
2200
2201 // Format: RGBA16F, Surface: kRGBA_F16_Clamped, Data: kRGBA_F32
2202 {
2203 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
2204 ioFormat.fColorType = GrColorType::kRGBA_F32;
2205 ioFormat.fExternalType = GR_GL_FLOAT;
2206 ioFormat.fExternalTexImageFormat = 0;
2207 ioFormat.fExternalReadFormat = GR_GL_RGBA;
2208 }
2209 }
2210 }
2211 }
2212
2213 // Format: R16F
2214 {
2215 FormatInfo& info = this->getFormatInfo(GrGLFormat::kR16F);
2216 info.fFormatType = FormatType::kFloat;
2217 info.fInternalFormatForRenderbuffer = GR_GL_R16F;
2218 info.fDefaultExternalFormat = GR_GL_RED;
2219 info.fDefaultExternalType = halfFloatType;
2220 info.fDefaultColorType = GrColorType::kR_F16;
2221 bool r16FTextureSupport = false;
2222 bool r16FRenderTargetSupport = false;
2223
2224 if (GR_IS_GR_GL(standard)) {
2225 if (version >= GR_GL_VER(3, 0) || ctxInfo.hasExtension("GL_ARB_texture_rg")) {
2226 r16FTextureSupport = true;
2227 r16FRenderTargetSupport = true;
2228 }
2229 } else if (GR_IS_GR_GL_ES(standard)) {
2230 // It seems possible that a combination of GL_EXT_texture_rg and
2231 // GL_EXT_color_buffer_half_float might add this format to ES 2.0 but it is not entirely
2232 // clear. The latter mentions interaction but that may only be for renderbuffers as
2233 // neither adds the texture format explicitly.
2234 // GL_OES_texture_format_half_float makes no reference to RED formats.
2235 if (version >= GR_GL_VER(3, 0)) {
2236 r16FTextureSupport = true;
2237 r16FRenderTargetSupport = version >= GR_GL_VER(3, 2) ||
2238 ctxInfo.hasExtension("GL_EXT_color_buffer_float") ||
2239 ctxInfo.hasExtension("GL_EXT_color_buffer_half_float");
2240 }
2241 } else if (GR_IS_GR_WEBGL(standard)) {
2242 if (version >= GR_GL_VER(2, 0)) {
2243 r16FTextureSupport = true;
2244 r16FRenderTargetSupport = ctxInfo.hasExtension("GL_EXT_color_buffer_float") ||
2245 ctxInfo.hasExtension("EXT_color_buffer_float");
2246 }
2247 }
2248
2249 if (r16FTextureSupport) {
2250 info.fFlags = FormatInfo::kTexturable_Flag | FormatInfo::kTransfers_Flag;
2251 if (r16FRenderTargetSupport) {
2252 info.fFlags |= fpRenderFlags;
2253 }
2254 }
2255 if (texStorageSupported) {
2256 info.fFlags |= FormatInfo::kUseTexStorage_Flag;
2257 info.fInternalFormatForTexImageOrStorage = GR_GL_R16F;
2258 } else {
2259 info.fInternalFormatForTexImageOrStorage =
2260 texImageSupportsSizedInternalFormat ? GR_GL_R16F : GR_GL_RED;
2261 }
2262
2263 if (r16FTextureSupport) {
2264 // Format: R16F, Surface: kAlpha_F16
2265 info.fColorTypeInfoCount = 1;
2266 info.fColorTypeInfos = std::make_unique<ColorTypeInfo[]>(info.fColorTypeInfoCount);
2267 int ctIdx = 0;
2268 {
2269 auto& ctInfo = info.fColorTypeInfos[ctIdx++];
2270 ctInfo.fColorType = GrColorType::kAlpha_F16;
2271 ctInfo.fFlags = ColorTypeInfo::kUploadData_Flag | ColorTypeInfo::kRenderable_Flag;
2272 ctInfo.fReadSwizzle = GrSwizzle("000r");
2273 ctInfo.fWriteSwizzle = GrSwizzle("a000");
2274 this->setColorTypeFormat(GrColorType::kAlpha_F16, GrGLFormat::kR16F);
2275
2276 // External IO ColorTypes:
2277 ctInfo.fExternalIOFormatCount = 2;
2278 ctInfo.fExternalIOFormats = std::make_unique<ColorTypeInfo::ExternalIOFormats[]>(
2279 ctInfo.fExternalIOFormatCount);
2280 int ioIdx = 0;
2281 // Format: R16F, Surface: kAlpha_F16, Data: kAlpha_F16
2282 {
2283 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
2284 ioFormat.fColorType = GrColorType::kAlpha_F16;
2285 ioFormat.fExternalType = halfFloatType;
2286 ioFormat.fExternalTexImageFormat = GR_GL_RED;
2287 ioFormat.fExternalReadFormat = GR_GL_RED;
2288 // Not guaranteed by ES/WebGL.
2289 ioFormat.fRequiresImplementationReadQuery = !GR_IS_GR_GL(standard);
2290 }
2291
2292 // Format: R16F, Surface: kAlpha_F16, Data: kAlpha_F32xxx
2293 {
2294 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
2295 ioFormat.fColorType = GrColorType::kAlpha_F32xxx;
2296 ioFormat.fExternalType = GR_GL_FLOAT;
2297 ioFormat.fExternalTexImageFormat = 0;
2298 ioFormat.fExternalReadFormat = GR_GL_RGBA;
2299 }
2300 }
2301 }
2302 }
2303
2304 // Format: LUMINANCE16F
2305 {
2306 // NOTE: We disallow lum16f on ES devices if linear filtering modes are not
2307 // supported. This is for simplicity, but a more granular approach is possible.
2308 bool lum16FSupported = false;
2309 bool lum16FSizedFormatSupported = false;
2310 if (GR_IS_GR_GL(standard)) {
2311 if (!fIsCoreProfile && ctxInfo.hasExtension("GL_ARB_texture_float")) {
2312 lum16FSupported = true;
2313 lum16FSizedFormatSupported = true;
2314 }
2315 } else if (GR_IS_GR_GL_ES(standard)) {
2316 if (ctxInfo.hasExtension("GL_OES_texture_half_float_linear") &&
2317 ctxInfo.hasExtension("GL_OES_texture_half_float")) {
2318 lum16FSupported = true;
2319 // Even on ES3 this extension is required to define LUMINANCE16F.
2320 lum16FSizedFormatSupported = ctxInfo.hasExtension("GL_EXT_texture_storage");
2321 }
2322 } // No WebGL support
2323
2324 if (formatWorkarounds.fDisableLuminance16F) {
2325 lum16FSupported = false;
2326 }
2327
2328 FormatInfo& info = this->getFormatInfo(GrGLFormat::kLUMINANCE16F);
2329 info.fFormatType = FormatType::kFloat;
2330 info.fInternalFormatForRenderbuffer = GR_GL_LUMINANCE16F;
2331 info.fDefaultExternalFormat = GR_GL_LUMINANCE;
2332 info.fDefaultExternalType = halfFloatType;
2333 info.fDefaultColorType = GrColorType::kGray_F16;
2334
2335 if (lum16FSupported) {
2336 info.fFlags = FormatInfo::kTexturable_Flag | FormatInfo::kTransfers_Flag;
2337
2338 if (texStorageSupported && lum16FSizedFormatSupported) {
2339 info.fFlags |= FormatInfo::kUseTexStorage_Flag;
2340 info.fInternalFormatForTexImageOrStorage = GR_GL_LUMINANCE16F;
2341 } else if (texImageSupportsSizedInternalFormat && lum16FSizedFormatSupported) {
2342 info.fInternalFormatForTexImageOrStorage = GR_GL_LUMINANCE16F;
2343 } else {
2344 info.fInternalFormatForTexImageOrStorage = GR_GL_LUMINANCE;
2345 }
2346
2347 info.fColorTypeInfoCount = 1;
2348 info.fColorTypeInfos = std::make_unique<ColorTypeInfo[]>(info.fColorTypeInfoCount);
2349 int ctIdx = 0;
2350 // Format: LUMINANCE16F, Surface: kAlpha_F16
2351 {
2352 auto& ctInfo = info.fColorTypeInfos[ctIdx++];
2353 ctInfo.fColorType = GrColorType::kAlpha_F16;
2354 ctInfo.fFlags = ColorTypeInfo::kUploadData_Flag;
2355 ctInfo.fReadSwizzle = GrSwizzle("000r");
2356 ctInfo.fWriteSwizzle = GrSwizzle("aaa0");
2357
2358 int idx = static_cast<int>(GrColorType::kAlpha_F16);
2359 if (fColorTypeToFormatTable[idx] == GrGLFormat::kUnknown) {
2360 this->setColorTypeFormat(GrColorType::kAlpha_F16, GrGLFormat::kLUMINANCE16F);
2361 }
2362
2363 // External IO ColorTypes:
2364 ctInfo.fExternalIOFormatCount = 2;
2365 ctInfo.fExternalIOFormats = std::make_unique<ColorTypeInfo::ExternalIOFormats[]>(
2366 ctInfo.fExternalIOFormatCount);
2367 int ioIdx = 0;
2368 // Format: LUMINANCE16F, Surface: kAlpha_F16, Data: kAlpha_F16
2369 {
2370 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
2371 ioFormat.fColorType = GrColorType::kAlpha_F16;
2372 ioFormat.fExternalType = halfFloatType;
2373 ioFormat.fExternalTexImageFormat = GR_GL_LUMINANCE;
2374 ioFormat.fExternalReadFormat = 0;
2375 }
2376
2377 // Format: LUMINANCE16F, Surface: kAlpha_F16, Data: kRGBA_F32
2378 {
2379 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
2380 ioFormat.fColorType = GrColorType::kRGBA_F32;
2381 ioFormat.fExternalType = GR_GL_FLOAT;
2382 ioFormat.fExternalTexImageFormat = 0;
2383 ioFormat.fExternalReadFormat = GR_GL_RGBA;
2384 }
2385 }
2386 }
2387 }
2388
2389 // Format: RGB8
2390 {
2391 FormatInfo& info = this->getFormatInfo(GrGLFormat::kRGB8);
2392 info.fFormatType = FormatType::kNormalizedFixedPoint;
2393 info.fInternalFormatForRenderbuffer = GR_GL_RGB8;
2394 info.fDefaultExternalFormat = GR_GL_RGB;
2395 info.fDefaultExternalType = GR_GL_UNSIGNED_BYTE;
2396 info.fDefaultColorType = GrColorType::kRGB_888;
2397 info.fFlags = FormatInfo::kTexturable_Flag | FormatInfo::kTransfers_Flag;
2398 if (GR_IS_GR_GL(standard)) {
2399 // Even in OpenGL 4.6 GL_RGB8 is required to be color renderable but not required to be
2400 // a supported render buffer format. Since we usually use render buffers for MSAA on
2401 // non-ES GL we don't support MSAA for GL_RGB8. On 4.2+ we could check using
2402 // glGetInternalFormativ(GL_RENDERBUFFER, GL_RGB8, GL_INTERNALFORMAT_SUPPORTED, ...) if
2403 // this becomes an issue.
2404 info.fFlags |= nonMSAARenderFlags;
2405 } else if (GR_IS_GR_GL_ES(standard)) {
2406 // 3.0 and the extension support this as a render buffer format.
2407 if (version >= GR_GL_VER(3, 0) || ctxInfo.hasExtension("GL_OES_rgb8_rgba8")) {
2408 info.fFlags |= msaaRenderFlags;
2409 }
2410 } else if (GR_IS_GR_WEBGL(standard)) {
2411 // WebGL seems to support RBG8
2412 info.fFlags |= msaaRenderFlags;
2413 }
2414 if (texStorageSupported) {
2415 info.fFlags |= FormatInfo::kUseTexStorage_Flag;
2416 info.fInternalFormatForTexImageOrStorage = GR_GL_RGB8;
2417 } else {
2418 info.fInternalFormatForTexImageOrStorage =
2419 texImageSupportsSizedInternalFormat ? GR_GL_RGB8 : GR_GL_RGB;
2420 }
2421
2422 info.fColorTypeInfoCount = 1;
2423 info.fColorTypeInfos = std::make_unique<ColorTypeInfo[]>(info.fColorTypeInfoCount);
2424 int ctIdx = 0;
2425 // Format: RGB8, Surface: kRGB_888x
2426 {
2427 auto& ctInfo = info.fColorTypeInfos[ctIdx++];
2428 ctInfo.fColorType = GrColorType::kRGB_888x;
2429 ctInfo.fFlags = ColorTypeInfo::kUploadData_Flag | ColorTypeInfo::kRenderable_Flag;
2430 this->setColorTypeFormat(GrColorType::kRGB_888x, GrGLFormat::kRGB8);
2431
2432 // External IO ColorTypes:
2433 ctInfo.fExternalIOFormatCount = 2;
2434 ctInfo.fExternalIOFormats = std::make_unique<ColorTypeInfo::ExternalIOFormats[]>(
2435 ctInfo.fExternalIOFormatCount);
2436 int ioIdx = 0;
2437 // Format: RGB8, Surface: kRGB_888x, Data: kRGB_888
2438 {
2439 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
2440 ioFormat.fColorType = GrColorType::kRGB_888;
2441 ioFormat.fExternalType = GR_GL_UNSIGNED_BYTE;
2442 ioFormat.fExternalTexImageFormat = GR_GL_RGB;
2443 ioFormat.fExternalReadFormat = 0;
2444 }
2445
2446 // Format: RGB8, Surface: kRGB_888x, Data: kRGBA_8888
2447 {
2448 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
2449 ioFormat.fColorType = GrColorType::kRGBA_8888;
2450 ioFormat.fExternalType = GR_GL_UNSIGNED_BYTE;
2451 ioFormat.fExternalTexImageFormat = 0;
2452 ioFormat.fExternalReadFormat = GR_GL_RGBA;
2453 }
2454 }
2455 }
2456
2457 // Format: RG8
2458 {
2459 FormatInfo& info = this->getFormatInfo(GrGLFormat::kRG8);
2460 info.fFormatType = FormatType::kNormalizedFixedPoint;
2461 info.fInternalFormatForRenderbuffer = GR_GL_RG8;
2462 info.fDefaultExternalFormat = GR_GL_RG;
2463 info.fDefaultExternalType = GR_GL_UNSIGNED_BYTE;
2464 info.fDefaultColorType = GrColorType::kRG_88;
2465 bool rg8Support = false;
2466 if (GR_IS_GR_GL(standard)) {
2467 rg8Support = version >= GR_GL_VER(3, 0) || ctxInfo.hasExtension("GL_ARB_texture_rg");
2468 } else if (GR_IS_GR_GL_ES(standard)) {
2469 rg8Support = version >= GR_GL_VER(3, 0) || ctxInfo.hasExtension("GL_EXT_texture_rg");
2470 } else if (GR_IS_GR_WEBGL(standard)) {
2471 rg8Support = version >= GR_GL_VER(2, 0);
2472 }
2473 if (rg8Support) {
2474 info.fFlags |= FormatInfo::kTexturable_Flag
2475 | FormatInfo::kTransfers_Flag
2476 | msaaRenderFlags;
2477 if (texStorageSupported) {
2478 info.fFlags |= FormatInfo::kUseTexStorage_Flag;
2479 info.fInternalFormatForTexImageOrStorage = GR_GL_RG8;
2480 }
2481 }
2482 if (!(info.fFlags & FormatInfo::kUseTexStorage_Flag)) {
2483 info.fInternalFormatForTexImageOrStorage =
2484 texImageSupportsSizedInternalFormat ? GR_GL_RG8 : GR_GL_RG;
2485 }
2486 if (rg8Support) {
2487 info.fColorTypeInfoCount = 1;
2488 info.fColorTypeInfos = std::make_unique<ColorTypeInfo[]>(info.fColorTypeInfoCount);
2489 int ctIdx = 0;
2490 // Format: RG8, Surface: kRG_88
2491 {
2492 auto& ctInfo = info.fColorTypeInfos[ctIdx++];
2493 ctInfo.fColorType = GrColorType::kRG_88;
2494 ctInfo.fFlags = ColorTypeInfo::kUploadData_Flag | ColorTypeInfo::kRenderable_Flag;
2495 this->setColorTypeFormat(GrColorType::kRG_88, GrGLFormat::kRG8);
2496
2497 // External IO ColorTypes:
2498 ctInfo.fExternalIOFormatCount = 2;
2499 ctInfo.fExternalIOFormats = std::make_unique<ColorTypeInfo::ExternalIOFormats[]>(
2500 ctInfo.fExternalIOFormatCount);
2501 int ioIdx = 0;
2502 // Format: RG8, Surface: kRG_88, Data: kRG_88
2503 {
2504 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
2505 ioFormat.fColorType = GrColorType::kRG_88;
2506 ioFormat.fExternalType = GR_GL_UNSIGNED_BYTE;
2507 ioFormat.fExternalTexImageFormat = GR_GL_RG;
2508 ioFormat.fExternalReadFormat = 0;
2509 if (GR_IS_GR_GL(standard) && !formatWorkarounds.fDisallowDirectRG8ReadPixels) {
2510 ioFormat.fExternalReadFormat = GR_GL_RG;
2511 }
2512 }
2513
2514 // Format: RG8, Surface: kRG_88, Data: kRGBA_8888
2515 {
2516 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
2517 ioFormat.fColorType = GrColorType::kRGBA_8888;
2518 ioFormat.fExternalType = GR_GL_UNSIGNED_BYTE;
2519 ioFormat.fExternalTexImageFormat = 0;
2520 ioFormat.fExternalReadFormat = GR_GL_RGBA;
2521 }
2522 }
2523 }
2524 }
2525
2526 // Format: RGB10_A2
2527 {
2528 FormatInfo& info = this->getFormatInfo(GrGLFormat::kRGB10_A2);
2529 info.fFormatType = FormatType::kNormalizedFixedPoint;
2530 info.fInternalFormatForRenderbuffer = GR_GL_RGB10_A2;
2531 info.fDefaultExternalFormat = GR_GL_RGBA;
2532 info.fDefaultExternalType = GR_GL_UNSIGNED_INT_2_10_10_10_REV;
2533 info.fDefaultColorType = GrColorType::kRGBA_1010102;
2534 if (GR_IS_GR_GL(standard) ||
2535 (GR_IS_GR_GL_ES(standard) && version >= GR_GL_VER(3, 0))) {
2536 info.fFlags = FormatInfo::kTexturable_Flag
2537 | FormatInfo::kTransfers_Flag
2538 | msaaRenderFlags;
2539 } else if (GR_IS_GR_GL_ES(standard) &&
2540 ctxInfo.hasExtension("GL_EXT_texture_type_2_10_10_10_REV")) {
2541 info.fFlags = FormatInfo::kTexturable_Flag | FormatInfo::kTransfers_Flag;
2542 } // No WebGL support
2543
2544 if (texStorageSupported) {
2545 info.fFlags |= FormatInfo::kUseTexStorage_Flag;
2546 info.fInternalFormatForTexImageOrStorage = GR_GL_RGB10_A2;
2547 } else {
2548 info.fInternalFormatForTexImageOrStorage =
2549 texImageSupportsSizedInternalFormat ? GR_GL_RGB10_A2 : GR_GL_RGBA;
2550 }
2551
2552 if (SkToBool(info.fFlags & FormatInfo::kTexturable_Flag)) {
2553 bool supportsBGRAColorType = GR_IS_GR_GL(standard) &&
2554 (version >= GR_GL_VER(1, 2) || ctxInfo.hasExtension("GL_EXT_bgra"));
2555
2556 info.fColorTypeInfoCount = supportsBGRAColorType ? 2 : 1;
2557 info.fColorTypeInfos = std::make_unique<ColorTypeInfo[]>(info.fColorTypeInfoCount);
2558 int ctIdx = 0;
2559 // Format: RGB10_A2, Surface: kRGBA_1010102
2560 {
2561 auto& ctInfo = info.fColorTypeInfos[ctIdx++];
2562 ctInfo.fColorType = GrColorType::kRGBA_1010102;
2563 ctInfo.fFlags = ColorTypeInfo::kUploadData_Flag | ColorTypeInfo::kRenderable_Flag;
2564 this->setColorTypeFormat(GrColorType::kRGBA_1010102, GrGLFormat::kRGB10_A2);
2565
2566 // External IO ColorTypes:
2567 ctInfo.fExternalIOFormatCount = 2;
2568 ctInfo.fExternalIOFormats = std::make_unique<ColorTypeInfo::ExternalIOFormats[]>(
2569 ctInfo.fExternalIOFormatCount);
2570 int ioIdx = 0;
2571 // Format: RGB10_A2, Surface: kRGBA_1010102, Data: kRGBA_1010102
2572 {
2573 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
2574 ioFormat.fColorType = GrColorType::kRGBA_1010102;
2575 ioFormat.fExternalType = GR_GL_UNSIGNED_INT_2_10_10_10_REV;
2576 ioFormat.fExternalTexImageFormat = GR_GL_RGBA;
2577 ioFormat.fExternalReadFormat = GR_GL_RGBA;
2578 // Not guaranteed by ES/WebGL.
2579 ioFormat.fRequiresImplementationReadQuery = !GR_IS_GR_GL(standard);
2580 }
2581
2582 // Format: RGB10_A2, Surface: kRGBA_1010102, Data: kRGBA_8888
2583 {
2584 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
2585 ioFormat.fColorType = GrColorType::kRGBA_8888;
2586 ioFormat.fExternalType = GR_GL_UNSIGNED_BYTE;
2587 ioFormat.fExternalTexImageFormat = 0;
2588 ioFormat.fExternalReadFormat = GR_GL_RGBA;
2589 }
2590 }
2591 //------------------------------------------------------------------
2592 // Format: RGB10_A2, Surface: kBGRA_1010102
2593 if (supportsBGRAColorType) {
2594 auto& ctInfo = info.fColorTypeInfos[ctIdx++];
2595 ctInfo.fColorType = GrColorType::kBGRA_1010102;
2596 ctInfo.fFlags = ColorTypeInfo::kUploadData_Flag | ColorTypeInfo::kRenderable_Flag;
2597 this->setColorTypeFormat(GrColorType::kBGRA_1010102, GrGLFormat::kRGB10_A2);
2598
2599 // External IO ColorTypes:
2600 ctInfo.fExternalIOFormatCount = 2;
2601 ctInfo.fExternalIOFormats = std::make_unique<ColorTypeInfo::ExternalIOFormats[]>(
2602 ctInfo.fExternalIOFormatCount);
2603 int ioIdx = 0;
2604 // Format: RGB10_A2, Surface: kBGRA_1010102, Data: kBGRA_1010102
2605 {
2606 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
2607 ioFormat.fColorType = GrColorType::kBGRA_1010102;
2608 ioFormat.fExternalType = GR_GL_UNSIGNED_INT_2_10_10_10_REV;
2609 ioFormat.fExternalTexImageFormat = GR_GL_BGRA;
2610 ioFormat.fExternalReadFormat =
2611 formatWorkarounds.fDisallowBGRA8ReadPixels ? 0 : GR_GL_BGRA;
2612 // Not guaranteed by ES/WebGL.
2613 ioFormat.fRequiresImplementationReadQuery = !GR_IS_GR_GL(standard);
2614 }
2615
2616 // Format: RGB10_A2, Surface: kBGRA_1010102, Data: kRGBA_8888
2617 {
2618 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
2619 ioFormat.fColorType = GrColorType::kRGBA_8888;
2620 ioFormat.fExternalType = GR_GL_UNSIGNED_BYTE;
2621 ioFormat.fExternalTexImageFormat = 0;
2622 ioFormat.fExternalReadFormat = GR_GL_RGBA;
2623 }
2624 }
2625 }
2626 }
2627
2628 // Format: RGBA4
2629 {
2630 FormatInfo& info = this->getFormatInfo(GrGLFormat::kRGBA4);
2631 info.fFormatType = FormatType::kNormalizedFixedPoint;
2632 info.fInternalFormatForRenderbuffer = GR_GL_RGBA4;
2633 info.fDefaultExternalFormat = GR_GL_RGBA;
2634 info.fDefaultExternalType = GR_GL_UNSIGNED_SHORT_4_4_4_4;
2635 info.fDefaultColorType = GrColorType::kABGR_4444;
2636 info.fFlags = FormatInfo::kTexturable_Flag | FormatInfo::kTransfers_Flag;
2637 if (GR_IS_GR_GL(standard)) {
2638 if (version >= GR_GL_VER(4, 2)) {
2639 info.fFlags |= msaaRenderFlags;
2640 }
2641 } else if (GR_IS_GR_GL_ES(standard)) {
2642 info.fFlags |= msaaRenderFlags;
2643 } else if (GR_IS_GR_WEBGL(standard)) {
2644 info.fFlags |= msaaRenderFlags;
2645 }
2646 if (texStorageSupported) {
2647 info.fFlags |= FormatInfo::kUseTexStorage_Flag;
2648 info.fInternalFormatForTexImageOrStorage = GR_GL_RGBA4;
2649 } else {
2650 info.fInternalFormatForTexImageOrStorage =
2651 texImageSupportsSizedInternalFormat ? GR_GL_RGBA4 : GR_GL_RGBA;
2652 }
2653
2654 info.fColorTypeInfoCount = 1;
2655 info.fColorTypeInfos = std::make_unique<ColorTypeInfo[]>(info.fColorTypeInfoCount);
2656 int ctIdx = 0;
2657 // Format: RGBA4, Surface: kABGR_4444
2658 {
2659 auto& ctInfo = info.fColorTypeInfos[ctIdx++];
2660 ctInfo.fColorType = GrColorType::kABGR_4444;
2661 ctInfo.fFlags = ColorTypeInfo::kUploadData_Flag | ColorTypeInfo::kRenderable_Flag;
2662 this->setColorTypeFormat(GrColorType::kABGR_4444, GrGLFormat::kRGBA4);
2663
2664 // External IO ColorTypes:
2665 ctInfo.fExternalIOFormatCount = 2;
2666 ctInfo.fExternalIOFormats = std::make_unique<ColorTypeInfo::ExternalIOFormats[]>(
2667 ctInfo.fExternalIOFormatCount);
2668 int ioIdx = 0;
2669 // Format: RGBA4, Surface: kABGR_4444, Data: kABGR_4444
2670 {
2671 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
2672 ioFormat.fColorType = GrColorType::kABGR_4444;
2673 ioFormat.fExternalType = GR_GL_UNSIGNED_SHORT_4_4_4_4;
2674 ioFormat.fExternalTexImageFormat = GR_GL_RGBA;
2675 ioFormat.fExternalReadFormat = GR_GL_RGBA;
2676 // Not guaranteed by ES/WebGL.
2677 ioFormat.fRequiresImplementationReadQuery = !GR_IS_GR_GL(standard);
2678 }
2679
2680 // Format: RGBA4, Surface: kABGR_4444, Data: kRGBA_8888
2681 {
2682 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
2683 ioFormat.fColorType = GrColorType::kRGBA_8888;
2684 ioFormat.fExternalType = GR_GL_UNSIGNED_BYTE;
2685 ioFormat.fExternalTexImageFormat = 0;
2686 ioFormat.fExternalReadFormat = GR_GL_RGBA;
2687 }
2688 }
2689 }
2690
2691 // Format: SRGB8_ALPHA8
2692 {
2693 FormatInfo& info = this->getFormatInfo(GrGLFormat::kSRGB8_ALPHA8);
2694 info.fFormatType = FormatType::kNormalizedFixedPoint;
2695 info.fInternalFormatForRenderbuffer = GR_GL_SRGB8_ALPHA8;
2696 info.fDefaultExternalType = GR_GL_UNSIGNED_BYTE;
2697 info.fDefaultColorType = GrColorType::kRGBA_8888_SRGB;
2698
2699 // We may modify the default external format below.
2700 info.fDefaultExternalFormat = GR_GL_RGBA;
2701 bool srgb8Alpha8TexStorageSupported = texStorageSupported;
2702 bool srgb8Alpha8TextureSupport = false;
2703 bool srgb8Alpha8RenderTargetSupport = false;
2704 if (GR_IS_GR_GL(standard)) {
2705 if (version >= GR_GL_VER(3, 0)) {
2706 srgb8Alpha8TextureSupport = true;
2707 srgb8Alpha8RenderTargetSupport = true;
2708 } else if (ctxInfo.hasExtension("GL_EXT_texture_sRGB")) {
2709 srgb8Alpha8TextureSupport = true;
2710 if (ctxInfo.hasExtension("GL_ARB_framebuffer_sRGB") ||
2711 ctxInfo.hasExtension("GL_EXT_framebuffer_sRGB")) {
2712 srgb8Alpha8RenderTargetSupport = true;
2713 }
2714 }
2715 } else if (GR_IS_GR_GL_ES(standard)) {
2716 if (version >= GR_GL_VER(3, 0) || ctxInfo.hasExtension("GL_EXT_sRGB")) {
2717 srgb8Alpha8TextureSupport = true;
2718 srgb8Alpha8RenderTargetSupport = true;
2719 }
2720 if (version < GR_GL_VER(3, 0)) {
2721 // ES 2.0 requires that the external format matches the internal format.
2722 info.fDefaultExternalFormat = GR_GL_SRGB_ALPHA;
2723 // There is no defined interaction between GL_EXT_sRGB and GL_EXT_texture_storage.
2724 srgb8Alpha8TexStorageSupported = false;
2725 }
2726 } else if (GR_IS_GR_WEBGL(standard)) {
2727 // sRGB extension should be on most WebGL 1.0 contexts, although sometimes under 2
2728 // names.
2729 if (version >= GR_GL_VER(2, 0) || ctxInfo.hasExtension("GL_EXT_sRGB") ||
2730 ctxInfo.hasExtension("EXT_sRGB")) {
2731 srgb8Alpha8TextureSupport = true;
2732 srgb8Alpha8RenderTargetSupport = true;
2733 }
2734 if (version < GR_GL_VER(2, 0)) {
2735 // WebGL 1.0 requires that the external format matches the internal format.
2736 info.fDefaultExternalFormat = GR_GL_SRGB_ALPHA;
2737 // There is no extension to WebGL 1 that adds glTexStorage.
2738 SkASSERT(!srgb8Alpha8TexStorageSupported);
2739 }
2740 }
2741
2742 if (srgb8Alpha8TextureSupport) {
2743 info.fFlags = FormatInfo::kTexturable_Flag | FormatInfo::kTransfers_Flag;
2744 if (srgb8Alpha8RenderTargetSupport) {
2745 info.fFlags |= formatWorkarounds.fDisableSRGBRenderWithMSAAForMacAMD
2746 ? nonMSAARenderFlags
2747 : msaaRenderFlags;
2748 }
2749 }
2750 if (srgb8Alpha8TexStorageSupported) {
2751 info.fFlags |= FormatInfo::kUseTexStorage_Flag;
2752 info.fInternalFormatForTexImageOrStorage = GR_GL_SRGB8_ALPHA8;
2753 } else {
2754 info.fInternalFormatForTexImageOrStorage =
2755 texImageSupportsSizedInternalFormat ? GR_GL_SRGB8_ALPHA8 : GR_GL_SRGB_ALPHA;
2756 }
2757
2758 if (srgb8Alpha8TextureSupport) {
2759 info.fColorTypeInfoCount = 1;
2760 info.fColorTypeInfos = std::make_unique<ColorTypeInfo[]>(info.fColorTypeInfoCount);
2761 int ctIdx = 0;
2762 // Format: SRGB8_ALPHA8, Surface: kRGBA_8888_SRGB
2763 {
2764 auto& ctInfo = info.fColorTypeInfos[ctIdx++];
2765 ctInfo.fColorType = GrColorType::kRGBA_8888_SRGB;
2766 ctInfo.fFlags = ColorTypeInfo::kUploadData_Flag | ColorTypeInfo::kRenderable_Flag;
2767 this->setColorTypeFormat(GrColorType::kRGBA_8888_SRGB, GrGLFormat::kSRGB8_ALPHA8);
2768
2769 // External IO ColorTypes:
2770 ctInfo.fExternalIOFormatCount = 1;
2771 ctInfo.fExternalIOFormats = std::make_unique<ColorTypeInfo::ExternalIOFormats[]>(
2772 ctInfo.fExternalIOFormatCount);
2773 int ioIdx = 0;
2774
2775 // Format: SRGB8_ALPHA8, Surface: kRGBA_8888_SRGB, Data: kRGBA_8888_SRGB
2776 {
2777 // GL does not do srgb<->rgb conversions when transferring between cpu and gpu.
2778 // Thus, the external format is GL_RGBA. See below for note about ES2.0 and
2779 // glTex[Sub]Image.
2780 GrGLenum texImageExternalFormat = GR_GL_RGBA;
2781
2782 // OpenGL ES 2.0 + GL_EXT_sRGB allows GL_SRGB_ALPHA to be specified as the
2783 // <format> param to Tex(Sub)Image. ES 2.0 requires the <internalFormat> and
2784 // <format> params to match. Thus, on ES 2.0 we will use GL_SRGB_ALPHA as the
2785 // <format> param. On OpenGL and ES 3.0+ GL_SRGB_ALPHA does not work for the
2786 // <format> param to glTexImage.
2787 if (GR_IS_GR_GL_ES(standard) && version == GR_GL_VER(2,0)) {
2788 texImageExternalFormat = GR_GL_SRGB_ALPHA;
2789 }
2790 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
2791 ioFormat.fColorType = GrColorType::kRGBA_8888_SRGB;
2792 ioFormat.fExternalType = GR_GL_UNSIGNED_BYTE;
2793 ioFormat.fExternalTexImageFormat = texImageExternalFormat;
2794 ioFormat.fExternalReadFormat = GR_GL_RGBA;
2795 }
2796 }
2797 }
2798 }
2799
2800 // Format: COMPRESSED_RGB8_BC1
2801 {
2802 FormatInfo& info = this->getFormatInfo(GrGLFormat::kCOMPRESSED_RGB8_BC1);
2803 info.fFormatType = FormatType::kNormalizedFixedPoint;
2804 info.fInternalFormatForTexImageOrStorage = GR_GL_COMPRESSED_RGB_S3TC_DXT1_EXT;
2805 if (GR_IS_GR_GL(standard) || GR_IS_GR_GL_ES(standard)) {
2806 if (ctxInfo.hasExtension("GL_EXT_texture_compression_s3tc")) {
2807 info.fFlags = FormatInfo::kTexturable_Flag;
2808 }
2809 } // No WebGL support
2810
2811 // There are no support GrColorTypes for this format
2812 }
2813
2814 // Format: COMPRESSED_RGBA8_BC1
2815 {
2816 FormatInfo& info = this->getFormatInfo(GrGLFormat::kCOMPRESSED_RGBA8_BC1);
2817 info.fFormatType = FormatType::kNormalizedFixedPoint;
2818 info.fInternalFormatForTexImageOrStorage = GR_GL_COMPRESSED_RGBA_S3TC_DXT1_EXT;
2819 if (GR_IS_GR_GL(standard) || GR_IS_GR_GL_ES(standard)) {
2820 if (ctxInfo.hasExtension("GL_EXT_texture_compression_s3tc")) {
2821 info.fFlags = FormatInfo::kTexturable_Flag;
2822 }
2823 } // No WebGL support
2824
2825 // There are no support GrColorTypes for this format
2826 }
2827
2828 // Format: COMPRESSED_RGB8_ETC2
2829 {
2830 FormatInfo& info = this->getFormatInfo(GrGLFormat::kCOMPRESSED_RGB8_ETC2);
2831 info.fFormatType = FormatType::kNormalizedFixedPoint;
2832 info.fInternalFormatForTexImageOrStorage = GR_GL_COMPRESSED_RGB8_ETC2;
2833 if (!formatWorkarounds.fDisallowETC2Compression) {
2834 if (GR_IS_GR_GL(standard)) {
2835 if (version >= GR_GL_VER(4, 3) ||
2836 ctxInfo.hasExtension("GL_ARB_ES3_compatibility")) {
2837 info.fFlags = FormatInfo::kTexturable_Flag;
2838 }
2839 } else if (GR_IS_GR_GL_ES(standard)) {
2840 if (version >= GR_GL_VER(3, 0) ||
2841 ctxInfo.hasExtension("GL_OES_compressed_ETC2_RGB8_texture")) {
2842 info.fFlags = FormatInfo::kTexturable_Flag;
2843 }
2844 } // No WebGL support
2845 }
2846
2847 // There are no support GrColorTypes for this format
2848 }
2849
2850 // Format: COMPRESSED_ETC1_RGB8
2851 {
2852 FormatInfo& info = this->getFormatInfo(GrGLFormat::kCOMPRESSED_ETC1_RGB8);
2853 info.fFormatType = FormatType::kNormalizedFixedPoint;
2854 info.fInternalFormatForTexImageOrStorage = GR_GL_COMPRESSED_ETC1_RGB8;
2855 if (GR_IS_GR_GL_ES(standard)) {
2856 if (ctxInfo.hasExtension("GL_OES_compressed_ETC1_RGB8_texture")) {
2857 info.fFlags = FormatInfo::kTexturable_Flag;
2858 }
2859 } // No GL or WebGL support
2860
2861 // There are no support GrColorTypes for this format
2862 }
2863
2864 // Format: R16
2865 {
2866 FormatInfo& info = this->getFormatInfo(GrGLFormat::kR16);
2867 info.fFormatType = FormatType::kNormalizedFixedPoint;
2868 info.fInternalFormatForRenderbuffer = GR_GL_R16;
2869 info.fDefaultExternalFormat = GR_GL_RED;
2870 info.fDefaultExternalType = GR_GL_UNSIGNED_SHORT;
2871 info.fDefaultColorType = GrColorType::kR_16;
2872 bool r16Supported = false;
2873 if (!formatWorkarounds.fDisallowTextureUnorm16) {
2874 if (GR_IS_GR_GL(standard)) {
2875 r16Supported = version >= GR_GL_VER(3, 0) ||
2876 ctxInfo.hasExtension("GL_ARB_texture_rg");
2877 } else if (GR_IS_GR_GL_ES(standard)) {
2878 r16Supported = ctxInfo.hasExtension("GL_EXT_texture_norm16");
2879 } // No WebGL support
2880 }
2881
2882 if (r16Supported) {
2883 info.fFlags = FormatInfo::kTexturable_Flag | msaaRenderFlags;
2884 if (!formatWorkarounds.fDisallowUnorm16Transfers) {
2885 info.fFlags |= FormatInfo::kTransfers_Flag;
2886 }
2887 }
2888
2889 if (texStorageSupported) {
2890 info.fFlags |= FormatInfo::kUseTexStorage_Flag;
2891 info.fInternalFormatForTexImageOrStorage = GR_GL_R16;
2892 } else {
2893 info.fInternalFormatForTexImageOrStorage =
2894 texImageSupportsSizedInternalFormat ? GR_GL_R16 : GR_GL_RED;
2895 }
2896
2897 if (r16Supported) {
2898 info.fColorTypeInfoCount = 1;
2899 info.fColorTypeInfos = std::make_unique<ColorTypeInfo[]>(info.fColorTypeInfoCount);
2900 int ctIdx = 0;
2901 // Format: R16, Surface: kAlpha_16
2902 {
2903 auto& ctInfo = info.fColorTypeInfos[ctIdx++];
2904 ctInfo.fColorType = GrColorType::kAlpha_16;
2905 ctInfo.fFlags = ColorTypeInfo::kUploadData_Flag | ColorTypeInfo::kRenderable_Flag;
2906 ctInfo.fReadSwizzle = GrSwizzle("000r");
2907 ctInfo.fWriteSwizzle = GrSwizzle("a000");
2908 this->setColorTypeFormat(GrColorType::kAlpha_16, GrGLFormat::kR16);
2909
2910 // External IO ColorTypes:
2911 ctInfo.fExternalIOFormatCount = 2;
2912 ctInfo.fExternalIOFormats = std::make_unique<ColorTypeInfo::ExternalIOFormats[]>(
2913 ctInfo.fExternalIOFormatCount);
2914 int ioIdx = 0;
2915 // Format: R16, Surface: kAlpha_16, Data: kAlpha_16
2916 {
2917 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
2918 ioFormat.fColorType = GrColorType::kAlpha_16;
2919 ioFormat.fExternalType = GR_GL_UNSIGNED_SHORT;
2920 ioFormat.fExternalTexImageFormat = GR_GL_RED;
2921 ioFormat.fExternalReadFormat = GR_GL_RED;
2922 // Not guaranteed by ES/WebGL.
2923 ioFormat.fRequiresImplementationReadQuery = !GR_IS_GR_GL(standard);
2924 }
2925
2926 // Format: R16, Surface: kAlpha_16, Data: kAlpha_8xxx
2927 {
2928 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
2929 ioFormat.fColorType = GrColorType::kAlpha_8xxx;
2930 ioFormat.fExternalType = GR_GL_UNSIGNED_BYTE;
2931 ioFormat.fExternalTexImageFormat = 0;
2932 ioFormat.fExternalReadFormat = GR_GL_RGBA;
2933 }
2934 }
2935 }
2936 }
2937
2938 // Format: RG16
2939 {
2940 FormatInfo& info = this->getFormatInfo(GrGLFormat::kRG16);
2941 info.fFormatType = FormatType::kNormalizedFixedPoint;
2942 info.fInternalFormatForTexImageOrStorage =
2943 texImageSupportsSizedInternalFormat ? GR_GL_RG16 : GR_GL_RG;
2944 info.fInternalFormatForRenderbuffer = GR_GL_RG16;
2945 info.fDefaultExternalFormat = GR_GL_RG;
2946 info.fDefaultExternalType = GR_GL_UNSIGNED_SHORT;
2947 info.fDefaultColorType = GrColorType::kRG_1616;
2948 bool rg16Supported = false;
2949 if (!formatWorkarounds.fDisallowTextureUnorm16) {
2950 if (GR_IS_GR_GL(standard)) {
2951 rg16Supported = version >= GR_GL_VER(3, 0) ||
2952 ctxInfo.hasExtension("GL_ARB_texture_rg");
2953 } else if (GR_IS_GR_GL_ES(standard)) {
2954 rg16Supported = ctxInfo.hasExtension("GL_EXT_texture_norm16");
2955 } // No WebGL support
2956 }
2957
2958 if (rg16Supported) {
2959 info.fFlags = FormatInfo::kTexturable_Flag | msaaRenderFlags;
2960 if (!formatWorkarounds.fDisallowUnorm16Transfers) {
2961 info.fFlags |= FormatInfo::kTransfers_Flag;
2962 }
2963 }
2964
2965 if (texStorageSupported) {
2966 info.fFlags |= FormatInfo::kUseTexStorage_Flag;
2967 info.fInternalFormatForTexImageOrStorage = GR_GL_RG16;
2968 } else {
2969 info.fInternalFormatForTexImageOrStorage =
2970 texImageSupportsSizedInternalFormat ? GR_GL_RG16 : GR_GL_RG;
2971 }
2972
2973 if (rg16Supported) {
2974 info.fColorTypeInfoCount = 1;
2975 info.fColorTypeInfos = std::make_unique<ColorTypeInfo[]>(info.fColorTypeInfoCount);
2976 int ctIdx = 0;
2977 // Format: GR_GL_RG16, Surface: kRG_1616
2978 {
2979 auto& ctInfo = info.fColorTypeInfos[ctIdx++];
2980 ctInfo.fColorType = GrColorType::kRG_1616;
2981 ctInfo.fFlags = ColorTypeInfo::kUploadData_Flag | ColorTypeInfo::kRenderable_Flag;
2982 this->setColorTypeFormat(GrColorType::kRG_1616, GrGLFormat::kRG16);
2983
2984 // External IO ColorTypes:
2985 ctInfo.fExternalIOFormatCount = 2;
2986 ctInfo.fExternalIOFormats = std::make_unique<ColorTypeInfo::ExternalIOFormats[]>(
2987 ctInfo.fExternalIOFormatCount);
2988 int ioIdx = 0;
2989 // Format: GR_GL_RG16, Surface: kRG_1616, Data: kRG_1616
2990 {
2991 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
2992 ioFormat.fColorType = GrColorType::kRG_1616;
2993 ioFormat.fExternalType = GR_GL_UNSIGNED_SHORT;
2994 ioFormat.fExternalTexImageFormat = GR_GL_RG;
2995 ioFormat.fExternalReadFormat = GR_GL_RG;
2996 // Not guaranteed by ES/WebGL.
2997 ioFormat.fRequiresImplementationReadQuery = !GR_IS_GR_GL(standard);
2998 }
2999
3000 // Format: GR_GL_RG16, Surface: kRG_1616, Data: kRGBA_8888
3001 {
3002 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
3003 ioFormat.fColorType = GrColorType::kRGBA_8888;
3004 ioFormat.fExternalType = GR_GL_UNSIGNED_BYTE;
3005 ioFormat.fExternalTexImageFormat = 0;
3006 ioFormat.fExternalReadFormat = GR_GL_RGBA;
3007 }
3008 }
3009 }
3010 }
3011
3012 // Format: RGBA16
3013 {
3014 bool rgba16Support = false;
3015 if (!formatWorkarounds.fDisallowTextureUnorm16) {
3016 if (GR_IS_GR_GL(standard)) {
3017 rgba16Support = version >= GR_GL_VER(3, 0);
3018 } else if (GR_IS_GR_GL_ES(standard)) {
3019 rgba16Support = ctxInfo.hasExtension("GL_EXT_texture_norm16");
3020 } // No WebGL support
3021 }
3022
3023 FormatInfo& info = this->getFormatInfo(GrGLFormat::kRGBA16);
3024 info.fFormatType = FormatType::kNormalizedFixedPoint;
3025
3026 info.fInternalFormatForRenderbuffer = GR_GL_RGBA16;
3027 info.fDefaultExternalFormat = GR_GL_RGBA;
3028 info.fDefaultExternalType = GR_GL_UNSIGNED_SHORT;
3029 info.fDefaultColorType = GrColorType::kRGBA_16161616;
3030 if (rgba16Support) {
3031 info.fFlags = FormatInfo::kTexturable_Flag | msaaRenderFlags;
3032 if (!formatWorkarounds.fDisallowUnorm16Transfers) {
3033 info.fFlags |= FormatInfo::kTransfers_Flag;
3034 }
3035 }
3036
3037 if (texStorageSupported) {
3038 info.fFlags |= FormatInfo::kUseTexStorage_Flag;
3039 info.fInternalFormatForTexImageOrStorage = GR_GL_RGBA16;
3040 } else {
3041 info.fInternalFormatForTexImageOrStorage =
3042 texImageSupportsSizedInternalFormat ? GR_GL_RGBA16 : GR_GL_RGBA;
3043 }
3044
3045 if (rgba16Support) {
3046 // Format: GR_GL_RGBA16, Surface: kRGBA_16161616
3047 info.fColorTypeInfoCount = 1;
3048 info.fColorTypeInfos = std::make_unique<ColorTypeInfo[]>(info.fColorTypeInfoCount);
3049 int ctIdx = 0;
3050 {
3051 auto& ctInfo = info.fColorTypeInfos[ctIdx++];
3052 ctInfo.fColorType = GrColorType::kRGBA_16161616;
3053 ctInfo.fFlags = ColorTypeInfo::kUploadData_Flag | ColorTypeInfo::kRenderable_Flag;
3054 this->setColorTypeFormat(GrColorType::kRGBA_16161616, GrGLFormat::kRGBA16);
3055
3056 // External IO ColorTypes:
3057 ctInfo.fExternalIOFormatCount = 2;
3058 ctInfo.fExternalIOFormats = std::make_unique<ColorTypeInfo::ExternalIOFormats[]>(
3059 ctInfo.fExternalIOFormatCount);
3060 int ioIdx = 0;
3061 // Format: GR_GL_RGBA16, Surface: kRGBA_16161616, Data: kRGBA_16161616
3062 {
3063 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
3064 ioFormat.fColorType = GrColorType::kRGBA_16161616;
3065 ioFormat.fExternalType = GR_GL_UNSIGNED_SHORT;
3066 ioFormat.fExternalTexImageFormat = GR_GL_RGBA;
3067 ioFormat.fExternalReadFormat = GR_GL_RGBA;
3068 // Not guaranteed by ES/WebGL.
3069 ioFormat.fRequiresImplementationReadQuery = !GR_IS_GR_GL(standard);
3070 }
3071
3072 // Format: GR_GL_RGBA16, Surface: kRGBA_16161616, Data: kRGBA_8888
3073 {
3074 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
3075 ioFormat.fColorType = GrColorType::kRGBA_8888;
3076 ioFormat.fExternalType = GR_GL_UNSIGNED_BYTE;
3077 ioFormat.fExternalTexImageFormat = 0;
3078 ioFormat.fExternalReadFormat = GR_GL_RGBA;
3079 }
3080 }
3081 }
3082 }
3083
3084 // Format:RG16F
3085 {
3086 bool rg16FTextureSupport = false;
3087 bool rg16FRenderTargetSupport = false;
3088 if (GR_IS_GR_GL(standard)) {
3089 if (version >= GR_GL_VER(3, 0) || ctxInfo.hasExtension("GL_ARB_texture_float")) {
3090 rg16FTextureSupport = true;
3091 rg16FRenderTargetSupport = true;
3092 }
3093 } else if (GR_IS_GR_GL_ES(standard)) {
3094 // It seems possible that a combination of GL_EXT_texture_rg and
3095 // GL_EXT_color_buffer_half_float might add this format to ES 2.0 but it is not entirely
3096 // clear. The latter mentions interaction but that may only be for renderbuffers as
3097 // neither adds the texture format explicitly.
3098 // GL_OES_texture_format_half_float makes no reference to RG formats.
3099 if (version >= GR_GL_VER(3, 0)) {
3100 rg16FTextureSupport = true;
3101 rg16FRenderTargetSupport = version >= GR_GL_VER(3, 2) ||
3102 ctxInfo.hasExtension("GL_EXT_color_buffer_float") ||
3103 ctxInfo.hasExtension("GL_EXT_color_buffer_half_float");
3104 }
3105 } else if (GR_IS_GR_WEBGL(standard)) {
3106 if (version >= GR_GL_VER(2, 0)) {
3107 rg16FTextureSupport = true;
3108 rg16FRenderTargetSupport = ctxInfo.hasExtension("GL_EXT_color_buffer_half_float") ||
3109 ctxInfo.hasExtension("EXT_color_buffer_half_float") ||
3110 ctxInfo.hasExtension("GL_EXT_color_buffer_float") ||
3111 ctxInfo.hasExtension("EXT_color_buffer_float");
3112 }
3113 }
3114
3115 FormatInfo& info = this->getFormatInfo(GrGLFormat::kRG16F);
3116 info.fFormatType = FormatType::kFloat;
3117 info.fInternalFormatForRenderbuffer = GR_GL_RG16F;
3118 info.fDefaultExternalFormat = GR_GL_RG;
3119 info.fDefaultExternalType = halfFloatType;
3120 info.fDefaultColorType = GrColorType::kRG_F16;
3121 if (rg16FTextureSupport) {
3122 info.fFlags |= FormatInfo::kTexturable_Flag | FormatInfo::kTransfers_Flag;
3123 if (rg16FRenderTargetSupport) {
3124 info.fFlags |= fpRenderFlags;
3125 }
3126 }
3127
3128 if (texStorageSupported) {
3129 info.fFlags |= FormatInfo::kUseTexStorage_Flag;
3130 info.fInternalFormatForTexImageOrStorage = GR_GL_RG16F;
3131 } else {
3132 info.fInternalFormatForTexImageOrStorage =
3133 texImageSupportsSizedInternalFormat ? GR_GL_RG16F : GR_GL_RG;
3134 }
3135
3136 if (rg16FTextureSupport) {
3137 info.fColorTypeInfoCount = 1;
3138 info.fColorTypeInfos = std::make_unique<ColorTypeInfo[]>(info.fColorTypeInfoCount);
3139 int ctIdx = 0;
3140 // Format: GR_GL_RG16F, Surface: kRG_F16
3141 {
3142 auto& ctInfo = info.fColorTypeInfos[ctIdx++];
3143 ctInfo.fColorType = GrColorType::kRG_F16;
3144 ctInfo.fFlags = ColorTypeInfo::kUploadData_Flag | ColorTypeInfo::kRenderable_Flag;
3145 this->setColorTypeFormat(GrColorType::kRG_F16, GrGLFormat::kRG16F);
3146
3147 // External IO ColorTypes:
3148 ctInfo.fExternalIOFormatCount = 2;
3149 ctInfo.fExternalIOFormats = std::make_unique<ColorTypeInfo::ExternalIOFormats[]>(
3150 ctInfo.fExternalIOFormatCount);
3151 int ioIdx = 0;
3152 // Format: GR_GL_RG16F, Surface: kRG_F16, Data: kRG_F16
3153 {
3154 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
3155 ioFormat.fColorType = GrColorType::kRG_F16;
3156 ioFormat.fExternalType = halfFloatType;
3157 ioFormat.fExternalTexImageFormat = GR_GL_RG;
3158 ioFormat.fExternalReadFormat = GR_GL_RG;
3159 // Not guaranteed by ES/WebGL.
3160 ioFormat.fRequiresImplementationReadQuery = !GR_IS_GR_GL(standard);
3161 }
3162
3163 // Format: GR_GL_RG16F, Surface: kRG_F16, Data: kRGBA_F32
3164 {
3165 auto& ioFormat = ctInfo.fExternalIOFormats[ioIdx++];
3166 ioFormat.fColorType = GrColorType::kRGBA_F32;
3167 ioFormat.fExternalType = GR_GL_FLOAT;
3168 ioFormat.fExternalTexImageFormat = 0;
3169 ioFormat.fExternalReadFormat = GR_GL_RGBA;
3170 }
3171 }
3172 }
3173 }
3174
3175 this->setupSampleCounts(ctxInfo, gli);
3176
3177 #ifdef SK_DEBUG
3178 for (int i = 0; i < kGrGLColorFormatCount; ++i) {
3179 if (GrGLFormat::kUnknown == static_cast<GrGLFormat>(i)) {
3180 continue;
3181 }
3182 const auto& formatInfo = fFormatTable[i];
3183 // Make sure we didn't set fbo attachable with msaa and not fbo attachable.
3184 SkASSERT(!((formatInfo.fFlags & FormatInfo::kFBOColorAttachmentWithMSAA_Flag) &&
3185 !(formatInfo.fFlags & FormatInfo::kFBOColorAttachment_Flag)));
3186
3187 // Make sure we set all the formats' FormatType
3188 SkASSERT(formatInfo.fFormatType != FormatType::kUnknown);
3189
3190 // Make sure if we added a ColorTypeInfo we filled it out
3191 for (int j = 0; j < formatInfo.fColorTypeInfoCount; ++j) {
3192 const auto& ctInfo = formatInfo.fColorTypeInfos[j];
3193 SkASSERT(ctInfo.fColorType != GrColorType::kUnknown);
3194 // Seems silly to add a color type if we don't support any flags on it.
3195 SkASSERT(ctInfo.fFlags);
3196 // Make sure if we added any ExternalIOFormats we filled it out
3197 for (int k = 0; k < ctInfo.fExternalIOFormatCount; ++k) {
3198 const auto& ioInfo = ctInfo.fExternalIOFormats[k];
3199 SkASSERT(ioInfo.fColorType != GrColorType::kUnknown);
3200 }
3201 }
3202 }
3203 #endif
3204 }
3205
setupSampleCounts(const GrGLContextInfo & ctxInfo,const GrGLInterface * gli)3206 void GrGLCaps::setupSampleCounts(const GrGLContextInfo& ctxInfo, const GrGLInterface* gli) {
3207 GrGLStandard standard = ctxInfo.standard();
3208 // standard can be unused (optimized away) if SK_ASSUME_GL_ES is set
3209 sk_ignore_unused_variable(standard);
3210 GrGLVersion version = ctxInfo.version();
3211
3212 for (int i = 0; i < kGrGLColorFormatCount; ++i) {
3213 if (FormatInfo::kFBOColorAttachmentWithMSAA_Flag & fFormatTable[i].fFlags) {
3214 // We assume that MSAA rendering is supported only if we support non-MSAA rendering.
3215 SkASSERT(FormatInfo::kFBOColorAttachment_Flag & fFormatTable[i].fFlags);
3216 if ((GR_IS_GR_GL(standard) &&
3217 (version >= GR_GL_VER(4,2) ||
3218 ctxInfo.hasExtension("GL_ARB_internalformat_query"))) ||
3219 (GR_IS_GR_GL_ES(standard) && version >= GR_GL_VER(3,0))) {
3220 int count;
3221 GrGLFormat grGLFormat = static_cast<GrGLFormat>(i);
3222 GrGLenum glFormat = this->getRenderbufferInternalFormat(grGLFormat);
3223 GR_GL_GetInternalformativ(gli, GR_GL_RENDERBUFFER, glFormat,
3224 GR_GL_NUM_SAMPLE_COUNTS, 1, &count);
3225 if (count) {
3226 std::unique_ptr<int[]> temp(new int[count]);
3227 GR_GL_GetInternalformativ(gli, GR_GL_RENDERBUFFER, glFormat, GR_GL_SAMPLES,
3228 count, temp.get());
3229 // GL has a concept of MSAA rasterization with a single sample but we do not.
3230 if (count && temp[count - 1] == 1) {
3231 --count;
3232 SkASSERT(!count || temp[count -1] > 1);
3233 }
3234 fFormatTable[i].fColorSampleCounts.setCount(count+1);
3235 // We initialize our supported values with 1 (no msaa) and reverse the order
3236 // returned by GL so that the array is ascending.
3237 fFormatTable[i].fColorSampleCounts[0] = 1;
3238 for (int j = 0; j < count; ++j) {
3239 #if defined(SK_BUILD_FOR_IOS) && TARGET_OS_SIMULATOR
3240 // The iOS simulator is reporting incorrect values for sample counts,
3241 // so force them to be a power of 2.
3242 fFormatTable[i].fColorSampleCounts[j+1] = SkPrevPow2(temp[count - j - 1]);
3243 #else
3244 fFormatTable[i].fColorSampleCounts[j+1] = temp[count - j - 1];
3245 #endif
3246 }
3247 }
3248 } else {
3249 // Fake out the table using some semi-standard counts up to the max allowed sample
3250 // count.
3251 int maxSampleCnt = 1;
3252 if (GrGLCaps::kES_IMG_MsToTexture_MSFBOType == fMSFBOType) {
3253 GR_GL_GetIntegerv(gli, GR_GL_MAX_SAMPLES_IMG, &maxSampleCnt);
3254 } else if (GrGLCaps::kNone_MSFBOType != fMSFBOType) {
3255 GR_GL_GetIntegerv(gli, GR_GL_MAX_SAMPLES, &maxSampleCnt);
3256 }
3257 // Chrome has a mock GL implementation that returns 0.
3258 maxSampleCnt = std::max(1, maxSampleCnt);
3259
3260 static constexpr int kDefaultSamples[] = {1, 2, 4, 8};
3261 int count = SK_ARRAY_COUNT(kDefaultSamples);
3262 for (; count > 0; --count) {
3263 if (kDefaultSamples[count - 1] <= maxSampleCnt) {
3264 break;
3265 }
3266 }
3267 if (count > 0) {
3268 fFormatTable[i].fColorSampleCounts.append(count, kDefaultSamples);
3269 }
3270 }
3271 } else if (FormatInfo::kFBOColorAttachment_Flag & fFormatTable[i].fFlags) {
3272 fFormatTable[i].fColorSampleCounts.setCount(1);
3273 fFormatTable[i].fColorSampleCounts[0] = 1;
3274 }
3275 }
3276 }
3277
canCopyTexSubImage(GrGLFormat dstFormat,bool dstHasMSAARenderBuffer,const GrTextureType * dstTypeIfTexture,GrGLFormat srcFormat,bool srcHasMSAARenderBuffer,const GrTextureType * srcTypeIfTexture) const3278 bool GrGLCaps::canCopyTexSubImage(GrGLFormat dstFormat, bool dstHasMSAARenderBuffer,
3279 const GrTextureType* dstTypeIfTexture,
3280 GrGLFormat srcFormat, bool srcHasMSAARenderBuffer,
3281 const GrTextureType* srcTypeIfTexture) const {
3282 // When it comes to format types and component sizes the gl spec is fairly complex as
3283 // requirements differ depending on many properties (e.g. if the internalFormat was created with
3284 // a sized format or not). These affect the rules about which format types can be copied to
3285 // which other types. For now we are being more restrictive and requiring that the types must
3286 // match exactly.
3287 if (this->getFormatDefaultExternalType(dstFormat) !=
3288 this->getFormatDefaultExternalType(srcFormat)) {
3289 return false;
3290 }
3291
3292 // Either both the src and dst formats need to be SRGB or both need to not be SRGB
3293 if (GrGLFormatIsSRGB(dstFormat) != GrGLFormatIsSRGB(srcFormat)) {
3294 return false;
3295 }
3296
3297 if (GR_IS_GR_GL_ES(fStandard)) {
3298 // Table 3.9 of the ES2 spec indicates the supported formats with CopyTexSubImage
3299 // and BGRA isn't in the spec. There doesn't appear to be any extension that adds it.
3300 // ANGLE, for one, does not allow it. However, we've found it works on some drivers and
3301 // avoids bugs with using glBlitFramebuffer.
3302 if ((dstFormat == GrGLFormat::kBGRA8 || srcFormat == GrGLFormat::kBGRA8) &&
3303 !fAllowBGRA8CopyTexSubImage) {
3304 return false;
3305 }
3306
3307 // Table 3.9 of the ES2 spec and 3.16 of ES3 spec indicates the supported internal base
3308 // formats with CopyTexSubImage. Each base format can be copied to itself or formats with
3309 // less channels.
3310 uint32_t dstChannels = GrGLFormatChannels(dstFormat);
3311 uint32_t srcChannels = GrGLFormatChannels(srcFormat);
3312 if (!dstChannels || !srcChannels) {
3313 // The formats don't represent color channels (i.e. may be depth stencil)
3314 return false;
3315 }
3316 // The dst channels have to be a subset of the srcChannels, except R, RG, or RGB, channels
3317 // can go to LUM. (See expansion of Table 3.9 in EXT_texture_rg).
3318 if ((dstChannels & srcChannels) != srcChannels) {
3319 if (dstChannels == kGray_SkColorChannelFlag ||
3320 dstChannels == kGrayAlpha_SkColorChannelFlags) {
3321 // The dst can't have gray if the src is alpha-only.
3322 if (srcChannels == kAlpha_SkColorChannelFlag) {
3323 return false;
3324 }
3325 } else {
3326 return false;
3327 }
3328 }
3329 }
3330
3331 // CopyTexSubImage is invalid or doesn't copy what we want when we have msaa render buffers.
3332 if (dstHasMSAARenderBuffer || srcHasMSAARenderBuffer) {
3333 return false;
3334 }
3335
3336 // CopyTex(Sub)Image writes to a texture and we have no way of dynamically wrapping a RT in a
3337 // texture.
3338 if (!dstTypeIfTexture) {
3339 return false;
3340 }
3341
3342 // Check that we could wrap the source in an FBO, that the dst is not TEXTURE_EXTERNAL, that no
3343 // mirroring is required
3344 return this->canFormatBeFBOColorAttachment(srcFormat) &&
3345 (!srcTypeIfTexture || *srcTypeIfTexture != GrTextureType::kExternal) &&
3346 *dstTypeIfTexture != GrTextureType::kExternal;
3347 }
3348
canCopyAsBlit(GrGLFormat dstFormat,int dstSampleCnt,const GrTextureType * dstTypeIfTexture,GrGLFormat srcFormat,int srcSampleCnt,const GrTextureType * srcTypeIfTexture,const SkRect & srcBounds,bool srcBoundsExact,const SkIRect & srcRect,const SkIPoint & dstPoint) const3349 bool GrGLCaps::canCopyAsBlit(GrGLFormat dstFormat, int dstSampleCnt,
3350 const GrTextureType* dstTypeIfTexture,
3351 GrGLFormat srcFormat, int srcSampleCnt,
3352 const GrTextureType* srcTypeIfTexture,
3353 const SkRect& srcBounds, bool srcBoundsExact,
3354 const SkIRect& srcRect, const SkIPoint& dstPoint) const {
3355 auto blitFramebufferFlags = fBlitFramebufferFlags;
3356 if (!this->canFormatBeFBOColorAttachment(dstFormat) ||
3357 !this->canFormatBeFBOColorAttachment(srcFormat)) {
3358 return false;
3359 }
3360
3361 if (dstTypeIfTexture && *dstTypeIfTexture == GrTextureType::kExternal) {
3362 return false;
3363 }
3364 if (srcTypeIfTexture && *srcTypeIfTexture == GrTextureType::kExternal) {
3365 return false;
3366 }
3367
3368 if (GrGLCaps::kNoSupport_BlitFramebufferFlag & blitFramebufferFlags) {
3369 return false;
3370 }
3371
3372 if (GrGLCaps::kResolveMustBeFull_BlitFrambufferFlag & blitFramebufferFlags) {
3373 if (srcSampleCnt > 1) {
3374 if (1 == dstSampleCnt) {
3375 return false;
3376 }
3377 if (SkRect::Make(srcRect) != srcBounds || !srcBoundsExact) {
3378 return false;
3379 }
3380 }
3381 }
3382
3383 if (GrGLCaps::kNoMSAADst_BlitFramebufferFlag & blitFramebufferFlags) {
3384 if (dstSampleCnt > 1) {
3385 return false;
3386 }
3387 }
3388
3389 if (GrGLCaps::kNoFormatConversion_BlitFramebufferFlag & blitFramebufferFlags) {
3390 if (srcFormat != dstFormat) {
3391 return false;
3392 }
3393 } else if (GrGLCaps::kNoFormatConversionForMSAASrc_BlitFramebufferFlag & blitFramebufferFlags) {
3394 if (srcSampleCnt > 1 && srcFormat != dstFormat) {
3395 return false;
3396 }
3397 }
3398
3399 if (GrGLCaps::kRectsMustMatchForMSAASrc_BlitFramebufferFlag & blitFramebufferFlags) {
3400 if (srcSampleCnt > 1) {
3401 if (dstPoint.fX != srcRect.fLeft || dstPoint.fY != srcRect.fTop) {
3402 return false;
3403 }
3404 }
3405 }
3406 return true;
3407 }
3408
canCopyAsDraw(GrGLFormat dstFormat,bool srcIsTexturable) const3409 bool GrGLCaps::canCopyAsDraw(GrGLFormat dstFormat, bool srcIsTexturable) const {
3410 return this->isFormatRenderable(dstFormat, 1) && srcIsTexturable;
3411 }
3412
has_msaa_render_buffer(const GrSurfaceProxy * surf,const GrGLCaps & glCaps)3413 static bool has_msaa_render_buffer(const GrSurfaceProxy* surf, const GrGLCaps& glCaps) {
3414 const GrRenderTargetProxy* rt = surf->asRenderTargetProxy();
3415 if (!rt) {
3416 return false;
3417 }
3418 // A RT has a separate MSAA renderbuffer if:
3419 // 1) It's multisampled
3420 // 2) We're using an extension with separate MSAA renderbuffers
3421 // 3) It's not FBO 0, which is special and always auto-resolves
3422 return rt->numSamples() > 1 &&
3423 glCaps.usesMSAARenderBuffers() &&
3424 !rt->glRTFBOIDIs0();
3425 }
3426
onCanCopySurface(const GrSurfaceProxy * dst,const GrSurfaceProxy * src,const SkIRect & srcRect,const SkIPoint & dstPoint) const3427 bool GrGLCaps::onCanCopySurface(const GrSurfaceProxy* dst, const GrSurfaceProxy* src,
3428 const SkIRect& srcRect, const SkIPoint& dstPoint) const {
3429 int dstSampleCnt = 0;
3430 int srcSampleCnt = 0;
3431 if (const GrRenderTargetProxy* rtProxy = dst->asRenderTargetProxy()) {
3432 dstSampleCnt = rtProxy->numSamples();
3433 }
3434 if (const GrRenderTargetProxy* rtProxy = src->asRenderTargetProxy()) {
3435 srcSampleCnt = rtProxy->numSamples();
3436 }
3437 SkASSERT((dstSampleCnt > 0) == SkToBool(dst->asRenderTargetProxy()));
3438 SkASSERT((srcSampleCnt > 0) == SkToBool(src->asRenderTargetProxy()));
3439
3440 const GrTextureProxy* dstTex = dst->asTextureProxy();
3441 const GrTextureProxy* srcTex = src->asTextureProxy();
3442
3443 GrTextureType dstTexType;
3444 GrTextureType* dstTexTypePtr = nullptr;
3445 GrTextureType srcTexType;
3446 GrTextureType* srcTexTypePtr = nullptr;
3447 if (dstTex) {
3448 dstTexType = dstTex->textureType();
3449 dstTexTypePtr = &dstTexType;
3450 }
3451 if (srcTex) {
3452 srcTexType = srcTex->textureType();
3453 srcTexTypePtr = &srcTexType;
3454 }
3455
3456 auto dstFormat = dst->backendFormat().asGLFormat();
3457 auto srcFormat = src->backendFormat().asGLFormat();
3458 return this->canCopyTexSubImage(dstFormat, has_msaa_render_buffer(dst, *this), dstTexTypePtr,
3459 srcFormat, has_msaa_render_buffer(src, *this), srcTexTypePtr) ||
3460 this->canCopyAsBlit(dstFormat, dstSampleCnt, dstTexTypePtr, srcFormat, srcSampleCnt,
3461 srcTexTypePtr, src->getBoundsRect(), src->priv().isExact(), srcRect,
3462 dstPoint) ||
3463 this->canCopyAsDraw(dstFormat, SkToBool(srcTex));
3464 }
3465
getDstCopyRestrictions(const GrRenderTargetProxy * src,GrColorType colorType) const3466 GrCaps::DstCopyRestrictions GrGLCaps::getDstCopyRestrictions(const GrRenderTargetProxy* src,
3467 GrColorType colorType) const {
3468 // If the src is a texture, we can implement the blit as a draw assuming the config is
3469 // renderable.
3470 if (src->asTextureProxy() && !this->isFormatAsColorTypeRenderable(colorType,
3471 src->backendFormat())) {
3472 return {};
3473 }
3474
3475 if (const auto* texProxy = src->asTextureProxy()) {
3476 if (texProxy->textureType() == GrTextureType::kExternal) {
3477 // Not supported for FBO blit or CopyTexSubImage. Caller will have to fall back to a
3478 // draw (if the source is also a texture).
3479 return {};
3480 }
3481 }
3482
3483 // We look for opportunities to use CopyTexSubImage, or fbo blit. If neither are
3484 // possible and we return false to fallback to creating a render target dst for render-to-
3485 // texture. This code prefers CopyTexSubImage to fbo blit and avoids triggering temporary fbo
3486 // creation. It isn't clear that avoiding temporary fbo creation is actually optimal.
3487 DstCopyRestrictions blitFramebufferRestrictions = {};
3488 if (src->numSamples() > 1 &&
3489 (fBlitFramebufferFlags & kResolveMustBeFull_BlitFrambufferFlag)) {
3490 blitFramebufferRestrictions.fRectsMustMatch = GrSurfaceProxy::RectsMustMatch::kYes;
3491 blitFramebufferRestrictions.fMustCopyWholeSrc = true;
3492 // Mirroring causes rects to mismatch later, don't allow it.
3493 } else if (src->numSamples() > 1 && (fBlitFramebufferFlags &
3494 kRectsMustMatchForMSAASrc_BlitFramebufferFlag)) {
3495 blitFramebufferRestrictions.fRectsMustMatch = GrSurfaceProxy::RectsMustMatch::kYes;
3496 }
3497
3498 auto srcFormat = src->backendFormat().asGLFormat();
3499 // Check for format issues with glCopyTexSubImage2D
3500 if (srcFormat == GrGLFormat::kBGRA8) {
3501 // glCopyTexSubImage2D doesn't work with this config. If the bgra can be used with fbo blit
3502 // then we set up for that, otherwise fail.
3503 if (this->canFormatBeFBOColorAttachment(srcFormat)) {
3504 return blitFramebufferRestrictions;
3505 }
3506 // Caller will have to use a draw.
3507 return {};
3508 }
3509
3510 {
3511 bool srcIsMSAARenderbuffer = src->numSamples() > 1 &&
3512 this->usesMSAARenderBuffers();
3513 if (srcIsMSAARenderbuffer) {
3514 // It's illegal to call CopyTexSubImage2D on a MSAA renderbuffer. Set up for FBO
3515 // blit or fail.
3516 if (this->canFormatBeFBOColorAttachment(srcFormat)) {
3517 return blitFramebufferRestrictions;
3518 }
3519 // Caller will have to use a draw.
3520 return {};
3521 }
3522 }
3523
3524 // We'll do a CopyTexSubImage, no restrictions.
3525 return {};
3526 }
3527
applyDriverCorrectnessWorkarounds(const GrGLContextInfo & ctxInfo,const GrContextOptions & contextOptions,const GrGLInterface * glInterface,GrShaderCaps * shaderCaps,FormatWorkarounds * formatWorkarounds)3528 void GrGLCaps::applyDriverCorrectnessWorkarounds(const GrGLContextInfo& ctxInfo,
3529 const GrContextOptions& contextOptions,
3530 const GrGLInterface* glInterface,
3531 GrShaderCaps* shaderCaps,
3532 FormatWorkarounds* formatWorkarounds) {
3533 // A driver bug on the nexus 6 causes incorrect dst copies when invalidate is called beforehand.
3534 // Thus we are disabling this extension for now on Adreno4xx devices.
3535 if (ctxInfo.renderer() == GrGLRenderer::kAdreno430 ||
3536 ctxInfo.renderer() == GrGLRenderer::kAdreno4xx_other ||
3537 fDriverBugWorkarounds.disable_discard_framebuffer) {
3538 fInvalidateFBType = kNone_InvalidateFBType;
3539 }
3540
3541 if (ctxInfo.renderer() == GrGLRenderer::kIntelCherryView) {
3542 // When running DMSAA_dst_read_with_existing_barrier with DMSAA disabled on linux Intel
3543 // HD405, the test fails when using texture barriers. Oddly the gpu doing the draw which
3544 // uses the barrier correctly. It is the next draw, which does not use or need a barrier,
3545 // that is blending with a dst as if the barrier draw didn't happen. Since this GPU is not
3546 // that important to us and this driver bug could probably manifest itself in the wild, we
3547 // are just disabling texture barrier support for the gpu.
3548 fTextureBarrierSupport = false;
3549 }
3550
3551 // glClearTexImage seems to have a bug in NVIDIA drivers that was fixed sometime between
3552 // 340.96 and 367.57.
3553 if (GR_IS_GR_GL(ctxInfo.standard()) && ctxInfo.driver() == GrGLDriver::kNVIDIA &&
3554 ctxInfo.driverVersion() < GR_GL_DRIVER_VER(367, 57, 0)) {
3555 fClearTextureSupport = false;
3556 }
3557
3558 // glBlitFramebuffer seems to produce incorrect results on QC, Mali400, and Tegra3 but
3559 // glCopyTexSubImage2D works (even though there is no extension that specifically allows it).
3560 if (ctxInfo.vendor() == GrGLVendor::kQualcomm ||
3561 ctxInfo.renderer() == GrGLRenderer::kMali4xx ||
3562 ctxInfo.renderer() == GrGLRenderer::kTegra_PreK1) {
3563 fAllowBGRA8CopyTexSubImage = true;
3564 }
3565
3566 // http://anglebug.com/6030
3567 if (fMSFBOType == kES_EXT_MsToTexture_MSFBOType &&
3568 ctxInfo.angleBackend() == GrGLANGLEBackend::kD3D11) {
3569 fDisallowDynamicMSAA = true;
3570 }
3571
3572 // http://skbug.com/12081
3573 if (GR_IS_GR_WEBGL(ctxInfo.standard())) {
3574 fDisallowDynamicMSAA = true;
3575 }
3576
3577 #if defined(__has_feature)
3578 #if defined(SK_BUILD_FOR_MAC) && __has_feature(thread_sanitizer)
3579 // See skbug.com/7058
3580 fMapBufferType = kNone_MapBufferType;
3581 fMapBufferFlags = kNone_MapFlags;
3582 fTransferFromBufferToTextureSupport = false;
3583 fTransferFromSurfaceToBufferSupport = false;
3584 fTransferBufferType = TransferBufferType::kNone;
3585 #endif
3586 #endif
3587
3588 // We found that the Galaxy J5 with an Adreno 306 running 6.0.1 has a bug where
3589 // GL_INVALID_OPERATION thrown by glDrawArrays when using a buffer that was mapped. The same bug
3590 // did not reproduce on a Nexus7 2013 with a 320 running Android M with driver 127.0. It's
3591 // unclear whether this really affects a wide range of devices.
3592 if (ctxInfo.renderer() == GrGLRenderer::kAdreno3xx &&
3593 ctxInfo.driverVersion() > GR_GL_DRIVER_VER(127, 0, 0)) {
3594 fMapBufferType = kNone_MapBufferType;
3595 fMapBufferFlags = kNone_MapFlags;
3596 fTransferFromBufferToTextureSupport = false;
3597 fTransferFromSurfaceToBufferSupport = false;
3598 fTransferBufferType = TransferBufferType::kNone;
3599 }
3600
3601 // The TransferPixelsToTexture test fails on ANGLE D3D9 and D3D11 if this is enabled.
3602 // https://anglebug.com/5542
3603 if (ctxInfo.angleBackend() == GrGLANGLEBackend::kD3D9 ||
3604 ctxInfo.angleBackend() == GrGLANGLEBackend::kD3D11) {
3605 fTransferPixelsToRowBytesSupport = false;
3606 }
3607
3608 // Using MIPs on this GPU seems to be a source of trouble.
3609 if (ctxInfo.renderer() == GrGLRenderer::kPowerVR54x) {
3610 fMipmapSupport = false;
3611 }
3612
3613 #ifdef SK_BUILD_FOR_ANDROID
3614 if (ctxInfo.renderer() == GrGLRenderer::kPowerVR54x) {
3615 // Flutter found glTexSubImage2D for GL_RED is much slower than GL_ALPHA on the
3616 // "MC18 PERSONAL SHOPPER"
3617 formatWorkarounds->fDisallowR8ForPowerVRSGX54x = true;
3618 }
3619 #endif
3620
3621 if (ctxInfo.isOverCommandBuffer() && ctxInfo.version() >= GR_GL_VER(3,0)) {
3622 formatWorkarounds->fDisallowTextureUnorm16 = true; // http://crbug.com/1224108
3623 formatWorkarounds->fDisallowETC2Compression = true; // http://crbug.com/1224111
3624 fTransferFromSurfaceToBufferSupport = false; // http://crbug.com/1224138
3625
3626 // http://crbug.com/1224117
3627 fMapBufferFlags = kNone_MapFlags;
3628 fMapBufferType = kNone_MapBufferType;
3629 }
3630
3631 // https://b.corp.google.com/issues/143074513
3632 // https://skbug.com/11152
3633 if (ctxInfo.renderer() == GrGLRenderer::kAdreno615 ||
3634 ctxInfo.renderer() == GrGLRenderer::kAdreno620) {
3635 fMSFBOType = kNone_MSFBOType;
3636 fMSAAResolvesAutomatically = false;
3637 }
3638
3639 #ifndef SK_BUILD_FOR_IOS
3640 if (ctxInfo.renderer() == GrGLRenderer::kPowerVR54x ||
3641 ctxInfo.renderer() == GrGLRenderer::kPowerVRRogue ||
3642 (ctxInfo.renderer() == GrGLRenderer::kAdreno3xx && !ctxInfo.isOverCommandBuffer())) {
3643 fPerformColorClearsAsDraws = true;
3644 }
3645 #endif
3646
3647 // A lot of GPUs have trouble with full screen clears (skbug.com/7195)
3648 if (ctxInfo.renderer() == GrGLRenderer::kAMDRadeonHD7xxx ||
3649 ctxInfo.renderer() == GrGLRenderer::kAMDRadeonR9M4xx) {
3650 fPerformColorClearsAsDraws = true;
3651 }
3652
3653 #ifdef SK_BUILD_FOR_MAC
3654 // crbug.com/768134 - On MacBook Pros, the Intel Iris Pro doesn't always perform
3655 // full screen clears
3656 // crbug.com/773107 - On MacBook Pros, a wide range of Intel GPUs don't always
3657 // perform full screen clears.
3658 // Update on 4/4/2018 - This appears to be fixed on driver 10.30.12 on a macOS 10.13.2 on a
3659 // Retina MBP Early 2015 with Iris 6100. It is possibly fixed on earlier drivers as well.
3660 // crbug.com/1039912 - Crash rate in glClear spiked after OS update, affecting mostly
3661 // Broadwell on 10.13+
3662 if (ctxInfo.vendor() == GrGLVendor::kIntel &&
3663 (ctxInfo.driverVersion() < GR_GL_DRIVER_VER(10, 30, 12) ||
3664 ctxInfo.renderer() == GrGLRenderer::kIntelBroadwell)) {
3665 fPerformColorClearsAsDraws = true;
3666 }
3667 // crbug.com/969609 - NVIDIA on Mac sometimes segfaults during glClear in chrome. It seems
3668 // mostly concentrated in 10.13/14, GT 650Ms, driver 12+. But there are instances of older
3669 // drivers and GTX 775s, so we'll start with a broader workaround.
3670 if (ctxInfo.vendor() == GrGLVendor::kNVIDIA) {
3671 fPerformColorClearsAsDraws = true;
3672 }
3673 #endif
3674
3675 // See crbug.com/755871. This could probably be narrowed to just partial clears as the driver
3676 // bugs seems to involve clearing too much and not skipping the clear.
3677 // See crbug.com/768134. This is also needed for full clears and was seen on an nVidia K620
3678 // but only for D3D11 ANGLE.
3679 if (ctxInfo.angleBackend() == GrGLANGLEBackend::kD3D11) {
3680 fPerformColorClearsAsDraws = true;
3681 }
3682
3683 if (ctxInfo.renderer() == GrGLRenderer::kAdreno430 ||
3684 ctxInfo.renderer() == GrGLRenderer::kAdreno4xx_other) {
3685 // This is known to be fixed sometime between driver 145.0 and 219.0
3686 if (ctxInfo.driverVersion() <= GR_GL_DRIVER_VER(219, 0, 0)) {
3687 fPerformStencilClearsAsDraws = true;
3688 }
3689 // This is known to be fixed sometime between driver 129.0 and 145.0 on Nexus 6P.
3690 // On driver 129 on Android M it fails the unit tests called WritePixelsPendingIO without
3691 // the workaround. It passes on Android N with driver 145 without the workaround.
3692 // skbug.com/11834
3693 if (ctxInfo.driverVersion() < GR_GL_DRIVER_VER(145, 0, 0)) {
3694 fDisallowTexSubImageForUnormConfigTexturesEverBoundToFBO = true;
3695 }
3696 }
3697
3698 if (fDriverBugWorkarounds.gl_clear_broken) {
3699 fPerformColorClearsAsDraws = true;
3700 fPerformStencilClearsAsDraws = true;
3701 }
3702
3703 if (ctxInfo.vendor() == GrGLVendor::kQualcomm) {
3704 // It appears that all the Adreno GPUs have less than optimal performance when
3705 // drawing w/ large index buffers.
3706 fAvoidLargeIndexBufferDraws = true;
3707 }
3708
3709 // This was reproduced on the following configurations:
3710 // - A Galaxy J5 (Adreno 306) running Android 6 with driver 140.0
3711 // - A Nexus 7 2013 (Adreno 320) running Android 5 with driver 104.0
3712 // - A Nexus 7 2013 (Adreno 320) running Android 6 with driver 127.0
3713 // - A Nexus 5 (Adreno 330) running Android 6 with driver 127.0
3714 // and not produced on:
3715 // - A Nexus 7 2013 (Adreno 320) running Android 4 with driver 53.0
3716 // The particular lines that get dropped from test images varies across different devices.
3717 if (ctxInfo.renderer() == GrGLRenderer::kAdreno3xx &&
3718 ctxInfo.driverVersion() > GR_GL_DRIVER_VER(53, 0, 0)) {
3719 fRequiresCullFaceEnableDisableWhenDrawingLinesAfterNonLines = true;
3720 }
3721
3722 // TODO: Don't apply this on iOS?
3723 if (ctxInfo.renderer() == GrGLRenderer::kPowerVRRogue) {
3724 // Our Chromebook with GrGLRenderer::kPowerVRRogue crashes on large instanced draws. The
3725 // current minimum number of instances observed to crash is somewhere between 2^14 and 2^15.
3726 // Keep the number of instances below 1000, just to be safe.
3727 fMaxInstancesPerDrawWithoutCrashing = 999;
3728 } else if (fDriverBugWorkarounds.disallow_large_instanced_draw) {
3729 fMaxInstancesPerDrawWithoutCrashing = 0x4000000;
3730 }
3731
3732 #ifndef SK_BUILD_FOR_IOS
3733 if (ctxInfo.renderer() == GrGLRenderer::kPowerVRRogue) {
3734 // We saw this bug on a TecnoSpark 3 Pro with a PowerVR GE8300.
3735 // GL_VERSION: "OpenGL ES 3.2 build 1.10@51309121"
3736 // Possibly this could be more limited by driver version or HW generation.
3737 // When using samplers, we are seeing a bug where the gpu is sometimes not sampling the
3738 // correct mip level data. A workaround to this issue is that when binding a texture we also
3739 // set some texture state, and it seems like any inividual state works (e.g. min/mag filter,
3740 // base level, max level, etc.). Currently we just set the min filter level every time we
3741 // bind a texture as the workaround.
3742 fMustSetAnyTexParameterToEnableMipmapping = true;
3743 // ColorTypeBackendAllocationTest failed for kAlpha_8 and kGray_8 when using
3744 // GL_UNPACK_ROW_LENGTH. Perhaps this could be a more limited workaround by applying
3745 // only to single channel 8 bit unorm formats but we only have a monolithic query for this
3746 // support at present.
3747 fWritePixelsRowBytesSupport = false;
3748 // TransferPixelsToTextureTest fails for all color types on
3749 // TecnoSpark 3 Pro with a PowerVR GE8300, GL_VERSION: "OpenGL ES 3.2 build 1.10@51309121"
3750 // if GL_UNPACK_ROW_LENGTH is used.
3751 fTransferPixelsToRowBytesSupport = false;
3752 }
3753 #endif
3754
3755 // Texture uploads sometimes seem to be ignored to textures bound to FBOS on Tegra3.
3756 if (ctxInfo.renderer() == GrGLRenderer::kTegra_PreK1) {
3757 fDisallowTexSubImageForUnormConfigTexturesEverBoundToFBO = true;
3758 fUseDrawInsteadOfAllRenderTargetWrites = true;
3759 }
3760
3761 #ifdef SK_BUILD_FOR_MAC
3762 static constexpr bool isMAC = true;
3763 #else
3764 static constexpr bool isMAC = false;
3765 #endif
3766
3767 #ifdef SK_BUILD_FOR_ANDROID
3768 // Older versions of Android have problems with setting GL_TEXTURE_BASE_LEVEL or
3769 // GL_TEXTURE_MAX_LEVEL on GL_TEXTURE_EXTERTNAL_OES textures. We just leave them as is and hope
3770 // the client never changes them either.
3771 fDontSetBaseOrMaxLevelForExternalTextures = true;
3772 // PowerVR can crash setting the levels on Android up to Q for any texture?
3773 // https://crbug.com/1123874
3774 if (ctxInfo.vendor() == GrGLVendor::kImagination) {
3775 fMipmapLevelControlSupport = false;
3776 }
3777 #endif
3778
3779 // We support manual mip-map generation (via iterative downsampling draw calls). This fixes
3780 // bugs on some cards/drivers that produce incorrect mip-maps for sRGB textures when using
3781 // glGenerateMipmap. Our implementation requires mip-level sampling control. Additionally,
3782 // it can be much slower (especially on mobile GPUs), so we opt-in only when necessary:
3783 if (fMipmapLevelControlSupport &&
3784 !ctxInfo.isOverCommandBuffer() && // http://crbug.com/1224110
3785 (contextOptions.fDoManualMipmapping ||
3786 ctxInfo.vendor() == GrGLVendor::kIntel ||
3787 (ctxInfo.driver() == GrGLDriver::kNVIDIA && isMAC) ||
3788 ctxInfo.vendor() == GrGLVendor::kATI)) {
3789 fDoManualMipmapping = true;
3790 }
3791
3792 // See http://crbug.com/710443
3793 #ifdef SK_BUILD_FOR_MAC
3794 if (ctxInfo.renderer() == GrGLRenderer::kIntelBroadwell) {
3795 fClearToBoundaryValuesIsBroken = true;
3796 }
3797 #endif
3798 if (ctxInfo.vendor() == GrGLVendor::kQualcomm) {
3799 fDrawArraysBaseVertexIsBroken = true;
3800 }
3801
3802 // https://b.corp.google.com/issues/188410972
3803 if (ctxInfo.renderer() == GrGLRenderer::kVirgl) {
3804 fDrawInstancedSupport = false;
3805 }
3806
3807 // http://anglebug.com/4538
3808 if (fBaseVertexBaseInstanceSupport && !fDrawInstancedSupport) {
3809 fBaseVertexBaseInstanceSupport = false;
3810 fNativeDrawIndirectSupport = false;
3811 fMultiDrawType = MultiDrawType::kNone;
3812 }
3813
3814 // Currently the extension is advertised but fb fetch is broken on 500 series Adrenos like the
3815 // Galaxy S7.
3816 // TODO: Once this is fixed we can update the check here to look at a driver version number too.
3817 if (ctxInfo.renderer() == GrGLRenderer::kAdreno530 ||
3818 ctxInfo.renderer() == GrGLRenderer::kAdreno5xx_other) {
3819 shaderCaps->fFBFetchSupport = false;
3820 }
3821
3822 // On the NexusS and GalaxyNexus, the use of 'any' causes the compilation error "Calls to any
3823 // function that may require a gradient calculation inside a conditional block may return
3824 // undefined results". This appears to be an issue with the 'any' call since even the simple
3825 // "result=black; if (any()) result=white;" code fails to compile.
3826 shaderCaps->fCanUseAnyFunctionInShader = (ctxInfo.vendor() != GrGLVendor::kImagination);
3827
3828 if (ctxInfo.renderer() == GrGLRenderer::kTegra_PreK1) {
3829 // The Tegra3 compiler will sometimes never return if we have min(abs(x), 1.0),
3830 // so we must do the abs first in a separate expression.
3831 shaderCaps->fCanUseMinAndAbsTogether = false;
3832
3833 // Tegra3 fract() seems to trigger undefined behavior for negative values, so we
3834 // must avoid this condition.
3835 shaderCaps->fCanUseFractForNegativeValues = false;
3836
3837 // Seeing crashes on Tegra3 with inlined functions that have early returns. Looks like the
3838 // do { ... break; } while (false); construct is causing a crash in the driver.
3839 shaderCaps->fCanUseDoLoops = false;
3840 }
3841
3842 // On Intel GPU there is an issue where it reads the second argument to atan "- %s.x" as an int
3843 // thus must us -1.0 * %s.x to work correctly
3844 if (ctxInfo.vendor() == GrGLVendor::kIntel) {
3845 shaderCaps->fMustForceNegatedAtanParamToFloat = true;
3846 }
3847
3848 #if defined(SK_BUILD_FOR_MAC)
3849 if (ctxInfo.vendor() == GrGLVendor::kATI) {
3850 // The Radeon GLSL compiler on Mac gets confused by ldexp(..., -x).
3851 // Convert to ldexp(..., x * -1).
3852 // http://skbug.com/12076
3853 shaderCaps->fMustForceNegatedLdexpParamToMultiply = true;
3854 }
3855 #endif
3856
3857 // On some Intel GPUs there is an issue where the driver outputs bogus values in the shader
3858 // when floor and abs are called on the same line. Thus we must execute an Op between them to
3859 // make sure the compiler doesn't re-inline them even if we break the calls apart.
3860 if (ctxInfo.vendor() == GrGLVendor::kIntel) {
3861 shaderCaps->fMustDoOpBetweenFloorAndAbs = true;
3862 }
3863
3864 // On Adreno devices with framebuffer fetch support, there is a bug where they always return
3865 // the original dst color when reading the outColor even after being written to. By using a
3866 // local outColor we can work around this bug.
3867 if (shaderCaps->fFBFetchSupport && ctxInfo.vendor() == GrGLVendor::kQualcomm) {
3868 shaderCaps->fRequiresLocalOutputColorForFBFetch = true;
3869 }
3870
3871 // Newer Mali GPUs do incorrect static analysis in specific situations: If there is uniform
3872 // color, and that uniform contains an opaque color, and the output of the shader is only based
3873 // on that uniform plus soemthing un-trackable (like a texture read), the compiler will deduce
3874 // that the shader always outputs opaque values. In that case, it appears to remove the shader
3875 // based blending code it normally injects, turning SrcOver into Src. To fix this, we always
3876 // insert an extra bit of math on the uniform that confuses the compiler just enough...
3877 if (ctxInfo.renderer() == GrGLRenderer::kMaliT) {
3878 shaderCaps->fMustObfuscateUniformColor = true;
3879 }
3880
3881 // On Mali G series GPUs, applying transfer functions in the fragment shader with half-floats
3882 // produces answers that are much less accurate than expected/required. This forces full floats
3883 // for some intermediate values to get acceptable results.
3884 if (ctxInfo.renderer() == GrGLRenderer::kMaliG) {
3885 fShaderCaps->fColorSpaceMathNeedsFloat = true;
3886 }
3887
3888 // On Mali 400 there is a bug using dFd* in the x direction. So we avoid using it when possible.
3889 if (ctxInfo.renderer() == GrGLRenderer::kMali4xx) {
3890 fShaderCaps->fAvoidDfDxForGradientsWhenPossible = true;
3891 }
3892
3893 #ifdef SK_BUILD_FOR_WIN
3894 // Check for ANGLE on Windows, so we can workaround a bug in D3D itself (anglebug.com/2098).
3895 //
3896 // Basically, if a shader has a construct like:
3897 //
3898 // float x = someCondition ? someValue : 0;
3899 // float2 result = (0 == x) ? float2(x, x)
3900 // : float2(2 * x / x, 0);
3901 //
3902 // ... the compiler will produce an error 'NaN and infinity literals not allowed', even though
3903 // we've explicitly guarded the division with a check against zero. This manifests in much
3904 // more complex ways in some of our shaders, so we use this caps bit to add an epsilon value
3905 // to the denominator of divisions, even when we've added checks that the denominator isn't 0.
3906 if (ctxInfo.angleBackend() != GrGLANGLEBackend::kUnknown || ctxInfo.isOverCommandBuffer()) {
3907 shaderCaps->fMustGuardDivisionEvenAfterExplicitZeroCheck = true;
3908 }
3909 #endif
3910
3911 // The Adreno 5xx and 6xx produce incorrect results when comparing a pair of matrices.
3912 if (ctxInfo.renderer() == GrGLRenderer::kAdreno530 ||
3913 ctxInfo.renderer() == GrGLRenderer::kAdreno5xx_other ||
3914 ctxInfo.renderer() == GrGLRenderer::kAdreno615 ||
3915 ctxInfo.renderer() == GrGLRenderer::kAdreno620 ||
3916 ctxInfo.renderer() == GrGLRenderer::kAdreno630 ||
3917 ctxInfo.renderer() == GrGLRenderer::kAdreno640 ||
3918 ctxInfo.renderer() == GrGLRenderer::kAdreno6xx_other) {
3919 shaderCaps->fRewriteMatrixComparisons = true;
3920 }
3921
3922 // We've seen Adreno 3xx devices produce incorrect (flipped) values for gl_FragCoord, in some
3923 // (rare) situations. It's sporadic, and mostly on older drivers. Additionally, old Adreno
3924 // compilers (see crbug.com/skia/4078) crash when accessing .zw of gl_FragCoord, so just bypass
3925 // using gl_FragCoord at all to get around it.
3926 if (ctxInfo.renderer() == GrGLRenderer::kAdreno3xx) {
3927 shaderCaps->fCanUseFragCoord = false;
3928 }
3929
3930 // gl_FragCoord has an incorrect subpixel offset on legacy Tegra hardware.
3931 if (ctxInfo.renderer() == GrGLRenderer::kTegra_PreK1) {
3932 shaderCaps->fCanUseFragCoord = false;
3933 }
3934
3935 // On Mali G71, mediump ints don't appear capable of representing every integer beyond +/-2048.
3936 // (Are they implemented with fp16?)
3937 if (ctxInfo.vendor() == GrGLVendor::kARM) {
3938 shaderCaps->fIncompleteShortIntPrecision = true;
3939 }
3940
3941 if (fDriverBugWorkarounds.add_and_true_to_loop_condition) {
3942 shaderCaps->fAddAndTrueToLoopCondition = true;
3943 }
3944
3945 if (fDriverBugWorkarounds.unfold_short_circuit_as_ternary_operation) {
3946 shaderCaps->fUnfoldShortCircuitAsTernary = true;
3947 }
3948
3949 if (fDriverBugWorkarounds.emulate_abs_int_function) {
3950 shaderCaps->fEmulateAbsIntFunction = true;
3951 }
3952
3953 if (fDriverBugWorkarounds.rewrite_do_while_loops) {
3954 shaderCaps->fRewriteDoWhileLoops = true;
3955 }
3956
3957 if (fDriverBugWorkarounds.remove_pow_with_constant_exponent) {
3958 shaderCaps->fRemovePowWithConstantExponent = true;
3959 }
3960
3961 if (fDriverBugWorkarounds.disable_dual_source_blending_support) {
3962 shaderCaps->fDualSourceBlendingSupport = false;
3963 }
3964
3965 if (ctxInfo.renderer() == GrGLRenderer::kAdreno3xx ||
3966 ctxInfo.renderer() == GrGLRenderer::kAdreno4xx_other) {
3967 shaderCaps->fMustWriteToFragColor = true;
3968 }
3969
3970 // Disabling advanced blend on various platforms with major known issues. We also block Chrome
3971 // command buffer for now until its own denylists can be updated.
3972 if (ctxInfo.renderer() == GrGLRenderer::kAdreno430 ||
3973 ctxInfo.renderer() == GrGLRenderer::kAdreno4xx_other ||
3974 ctxInfo.renderer() == GrGLRenderer::kAdreno530 ||
3975 ctxInfo.renderer() == GrGLRenderer::kAdreno5xx_other ||
3976 ctxInfo.driver() == GrGLDriver::kIntel ||
3977 ctxInfo.isOverCommandBuffer() ||
3978 ctxInfo.vendor() == GrGLVendor::kARM /* http://skbug.com/11906 */) {
3979 fBlendEquationSupport = kBasic_BlendEquationSupport;
3980 shaderCaps->fAdvBlendEqInteraction = GrShaderCaps::kNotSupported_AdvBlendEqInteraction;
3981 }
3982
3983 // Non-coherent advanced blend has an issue on NVIDIA pre 337.00.
3984 if (ctxInfo.driver() == GrGLDriver::kNVIDIA &&
3985 ctxInfo.driverVersion() < GR_GL_DRIVER_VER(337, 00, 0) &&
3986 kAdvanced_BlendEquationSupport == fBlendEquationSupport) {
3987 fBlendEquationSupport = kBasic_BlendEquationSupport;
3988 shaderCaps->fAdvBlendEqInteraction = GrShaderCaps::kNotSupported_AdvBlendEqInteraction;
3989 }
3990
3991 if (fDriverBugWorkarounds.disable_blend_equation_advanced) {
3992 fBlendEquationSupport = kBasic_BlendEquationSupport;
3993 shaderCaps->fAdvBlendEqInteraction = GrShaderCaps::kNotSupported_AdvBlendEqInteraction;
3994 }
3995
3996 if (this->advancedBlendEquationSupport()) {
3997 if (ctxInfo.driver() == GrGLDriver::kNVIDIA &&
3998 ctxInfo.driverVersion() < GR_GL_DRIVER_VER(355, 00, 0)) {
3999 // Disable color-dodge and color-burn on pre-355.00 NVIDIA.
4000 fAdvBlendEqDisableFlags |= (1 << kColorDodge_GrBlendEquation) |
4001 (1 << kColorBurn_GrBlendEquation);
4002 }
4003 if (ctxInfo.vendor() == GrGLVendor::kARM) {
4004 // Disable color-burn on ARM until the fix is released.
4005 fAdvBlendEqDisableFlags |= (1 << kColorBurn_GrBlendEquation);
4006 }
4007 }
4008
4009 // On Adreno 5xx devices, there is a bug where we first draw using dual source blending. Thus
4010 // the dst blend func references the dst. Then the next draw we disable blending. However, on
4011 // the second draw the driver has a bug where it tries to access the second color output again.
4012 // This is fixed by reseting the blend function to anything that does not reference src2 when we
4013 // disable blending.
4014 if (ctxInfo.renderer() == GrGLRenderer::kAdreno530 ||
4015 ctxInfo.renderer() == GrGLRenderer::kAdreno5xx_other ||
4016 ctxInfo.renderer() == GrGLRenderer::kAdreno620 ||
4017 ctxInfo.renderer() == GrGLRenderer::kAdreno640) {
4018 fMustResetBlendFuncBetweenDualSourceAndDisable = true;
4019 }
4020
4021 // Many ES3 drivers only advertise the ES2 image_external extension, but support the _essl3
4022 // extension, and require that it be enabled to work with ESSL3. Other devices require the ES2
4023 // extension to be enabled, even when using ESSL3. Enabling both extensions fixes both cases.
4024 // skbug.com/7713
4025 if (ctxInfo.hasExtension("GL_OES_EGL_image_external") &&
4026 ctxInfo.glslGeneration() >= k330_GrGLSLGeneration &&
4027 !shaderCaps->fExternalTextureSupport) { // i.e. Missing the _essl3 extension
4028 shaderCaps->fExternalTextureSupport = true;
4029 shaderCaps->fExternalTextureExtensionString = "GL_OES_EGL_image_external";
4030 shaderCaps->fSecondExternalTextureExtensionString = "GL_OES_EGL_image_external_essl3";
4031 }
4032
4033 #ifdef SK_BUILD_FOR_IOS
4034 // iOS drivers appear to implement TexSubImage by creating a staging buffer, and copying
4035 // UNPACK_ROW_LENGTH * height bytes. That's unsafe in several scenarios, and the simplest fix
4036 // is to just disable the feature.
4037 // https://github.com/flutter/flutter/issues/16718
4038 // https://bugreport.apple.com/web/?problemID=39948888
4039 fWritePixelsRowBytesSupport = false;
4040 // This affects all iOS devices for transfering from a PBO as well (presumably the issue is in
4041 // the GL->Metal layer).
4042 fTransferPixelsToRowBytesSupport = false;
4043 #endif
4044
4045 if (ctxInfo.vendor() == GrGLVendor::kIntel || // IntelIris640 drops draws completely.
4046 ctxInfo.renderer() == GrGLRenderer::kMaliT || // Some curves appear flat on GalaxyS6.
4047 ctxInfo.renderer() == GrGLRenderer::kAdreno3xx ||
4048 ctxInfo.renderer() == GrGLRenderer::kAdreno430 ||
4049 ctxInfo.renderer() == GrGLRenderer::kAdreno4xx_other || // We get garbage on Adreno405.
4050 ctxInfo.angleBackend() == GrGLANGLEBackend::kD3D9) { // D3D9 conic strokes fail.
4051 fDisableTessellationPathRenderer = true;
4052 }
4053 // We found that on Wembley devices (PowerVR GE8320) that using tessellation path renderer would
4054 // cause lots of rendering errors where it seemed like vertices were in the wrong place. This
4055 // led to lots of GMs drawing nothing (e.g. dashing4) or lots of garbage. The Wembley devices
4056 // were running Android 12 with a driver version of 1.13. We previously had TecnoSpark devices
4057 // with the same GPU running on Android P (driver 1.10) which did not have this issue. We don't
4058 // know when the bug appeared in the driver so for now we disable tessellation path renderer for
4059 // all matching gpus regardless of driver version.
4060 if (ctxInfo.renderer() == GrGLRenderer::kPowerVRRogue) {
4061 fDisableTessellationPathRenderer = true;
4062 }
4063
4064 // http://skbug.com/9739
4065 bool isNVIDIAPascal =
4066 ctxInfo.driver() == GrGLDriver::kNVIDIA &&
4067 ctxInfo.hasExtension("GL_NV_conservative_raster_pre_snap_triangles") && // Pascal+.
4068 !ctxInfo.hasExtension("GL_NV_conservative_raster_underestimation"); // Volta+.
4069 if (isNVIDIAPascal && ctxInfo.driverVersion() < GR_GL_DRIVER_VER(440, 00, 0)) {
4070 if (GR_IS_GR_GL(ctxInfo.standard())) {
4071 // glMemoryBarrier wasn't around until version 4.2.
4072 if (ctxInfo.version() >= GR_GL_VER(4,2)) {
4073 fRequiresManualFBBarrierAfterTessellatedStencilDraw = true;
4074 } else {
4075 shaderCaps->fMaxTessellationSegments = 0;
4076 }
4077 } else {
4078 // glMemoryBarrier wasn't around until es version 3.1.
4079 if (ctxInfo.version() >= GR_GL_VER(3,1)) {
4080 fRequiresManualFBBarrierAfterTessellatedStencilDraw = true;
4081 } else {
4082 shaderCaps->fMaxTessellationSegments = 0;
4083 }
4084 }
4085 }
4086
4087 if (ctxInfo.driver() == GrGLDriver::kQualcomm) {
4088 // Qualcomm fails to link programs with tessellation and does not give an error message.
4089 // http://skbug.com/9740
4090 shaderCaps->fMaxTessellationSegments = 0;
4091 }
4092
4093 #ifdef SK_BUILD_FOR_WIN
4094 // glDrawElementsIndirect fails GrMeshTest on every Win10 Intel bot.
4095 if (ctxInfo.driver() == GrGLDriver::kIntel ||
4096 (ctxInfo.angleVendor() == GrGLVendor::kIntel &&
4097 ctxInfo.angleBackend() == GrGLANGLEBackend::kOpenGL)) {
4098 fNativeDrawIndexedIndirectIsBroken = true;
4099 fUseClientSideIndirectBuffers = true;
4100 }
4101 #endif
4102
4103 // PowerVRGX6250 drops every pixel if we modify the sample mask while color writes are disabled.
4104 if (ctxInfo.renderer() == GrGLRenderer::kPowerVRRogue) {
4105 fNeverDisableColorWrites = true;
4106 shaderCaps->fMustWriteToFragColor = true;
4107 }
4108
4109 // It appears that Qualcomm drivers don't actually support
4110 // GL_NV_shader_noperspective_interpolation in ES 3.00 or 3.10 shaders, only 3.20.
4111 // https://crbug.com/986581
4112 if (ctxInfo.vendor() == GrGLVendor::kQualcomm &&
4113 k320es_GrGLSLGeneration != ctxInfo.glslGeneration()) {
4114 shaderCaps->fNoPerspectiveInterpolationSupport = false;
4115 }
4116
4117 // We disable srgb write control for Adreno4xx devices.
4118 // see: https://bug.skia.org/5329
4119 if (ctxInfo.renderer() == GrGLRenderer::kAdreno430 ||
4120 ctxInfo.renderer() == GrGLRenderer::kAdreno4xx_other) {
4121 fSRGBWriteControl = false;
4122 }
4123
4124 // MacPro devices with AMD cards fail to create MSAA sRGB render buffers.
4125 #if defined(SK_BUILD_FOR_MAC)
4126 if (ctxInfo.vendor() == GrGLVendor::kATI) {
4127 formatWorkarounds->fDisableSRGBRenderWithMSAAForMacAMD = true;
4128 }
4129 #endif
4130
4131 // Command buffer fails glTexSubImage2D with type == GL_HALF_FLOAT_OES if a GL_RGBA16F texture
4132 // is created with glTexStorage2D. See crbug.com/1008003.
4133 formatWorkarounds->fDisableRGBA16FTexStorageForCrBug1008003 =
4134 ctxInfo.isOverCommandBuffer() && ctxInfo.version() < GR_GL_VER(3, 0);
4135
4136 #if defined(SK_BUILD_FOR_WIN)
4137 // On Intel Windows ES contexts it seems that using texture storage with BGRA causes
4138 // problems with cross-context SkImages.
4139 formatWorkarounds->fDisableBGRATextureStorageForIntelWindowsES =
4140 ctxInfo.driver() == GrGLDriver::kIntel && GR_IS_GR_GL_ES(ctxInfo.standard());
4141 #endif
4142
4143 // On the Intel Iris 6100, interacting with LUM16F seems to confuse the driver. After
4144 // writing to/reading from a LUM16F texture reads from/writes to other formats behave
4145 // erratically.
4146 // All Adrenos claim to support LUM16F but don't appear to actually do so.
4147 // The failing devices/gpus were: Nexus5/Adreno330, Nexus5x/Adreno418, Pixel/Adreno530,
4148 // Pixel2XL/Adreno540 and Pixel3/Adreno630
4149 formatWorkarounds->fDisableLuminance16F = ctxInfo.renderer() == GrGLRenderer::kIntelBroadwell ||
4150 ctxInfo.vendor() == GrGLVendor::kQualcomm;
4151
4152 #ifdef SK_BUILD_FOR_MAC
4153 // On a MacBookPro 11.5 running MacOS 10.13 with a Radeon M370X the TransferPixelsFrom test
4154 // fails when transferring out from a GL_RG8 texture using GL_RG/GL_UNSIGNED_BYTE.
4155 // The same error also occurs in MacOS 10.15 with a Radeon Pro 5300M.
4156 formatWorkarounds->fDisallowDirectRG8ReadPixels =
4157 ctxInfo.renderer() == GrGLRenderer::kAMDRadeonR9M3xx ||
4158 ctxInfo.renderer() == GrGLRenderer::kAMDRadeonPro5xxx ||
4159 ctxInfo.renderer() == GrGLRenderer::kAMDRadeonProVegaxx;
4160 #endif
4161
4162 #ifdef SK_BUILD_FOR_ANDROID
4163 // We don't usually use glTexStorage() on Android for performance reasons. (crbug.com/945506).
4164 // On a NVIDIA Shield TV running Android 7.0 creating a texture with glTexImage2D() with
4165 // internal format GL_LUMINANCE8 fails. However, it succeeds with glTexStorage2D().
4166 //
4167 // Additionally, on the Nexus 9 running Android 6.0.1 formats added by GL_EXT_texture_rg and
4168 // GL_EXT_texture_norm16 cause errors if they are created with glTexImage2D() with
4169 // an unsized internal format. We wouldn't normally do that but Chrome can limit us
4170 // artificially to ES2. (crbug.com/1003481)
4171 if (ctxInfo.vendor() == GrGLVendor::kNVIDIA) {
4172 formatWorkarounds->fDontDisableTexStorageOnAndroid = true;
4173 }
4174 #endif
4175
4176 // https://github.com/flutter/flutter/issues/38700
4177 if (ctxInfo.driver() == GrGLDriver::kAndroidEmulator) {
4178 shaderCaps->fNoDefaultPrecisionForExternalSamplers = true;
4179 }
4180
4181 // http://skbug.com/9491: Nexus5 produces rendering artifacts when we use QCOM_tiled_rendering.
4182 if (ctxInfo.renderer() == GrGLRenderer::kAdreno3xx) {
4183 fTiledRenderingSupport = false;
4184 }
4185 // https://github.com/flutter/flutter/issues/47164
4186 // https://github.com/flutter/flutter/issues/47804
4187 if (fTiledRenderingSupport && (!glInterface->fFunctions.fStartTiling ||
4188 !glInterface->fFunctions.fEndTiling)) {
4189 // Some devices expose the QCOM tiled memory extension string but don't actually provide the
4190 // start and end tiling functions (see above flutter bugs). To work around this, the funcs
4191 // are marked optional in the interface generator, but we turn off the tiled rendering cap
4192 // if they aren't provided. This disabling is in driver workarounds so that SKQP will still
4193 // fail on devices that advertise the extension w/o the functions.
4194 fTiledRenderingSupport = false;
4195 }
4196
4197 if (ctxInfo.angleBackend() == GrGLANGLEBackend::kD3D9) {
4198 formatWorkarounds->fDisallowBGRA8ReadPixels = true;
4199 }
4200
4201 // We disable MSAA for all Intel GPUs. Before Gen9, performance was very bad. Even with Gen9,
4202 // we've seen driver crashes in the wild. We don't have data on Gen11 yet.
4203 // (crbug.com/527565, crbug.com/983926)
4204 if (ctxInfo.vendor() == GrGLVendor::kIntel ||
4205 ctxInfo.angleVendor() == GrGLVendor::kIntel) {
4206 fMSFBOType = kNone_MSFBOType;
4207 }
4208
4209 // ANGLE doesn't support do-while loops.
4210 if (ctxInfo.angleBackend() != GrGLANGLEBackend::kUnknown) {
4211 shaderCaps->fCanUseDoLoops = false;
4212 }
4213
4214 // ANGLE's D3D9 backend + AMD GPUs are flaky with program binary caching (skbug.com/10395)
4215 if (ctxInfo.angleBackend() == GrGLANGLEBackend::kD3D9 &&
4216 ctxInfo.angleVendor() == GrGLVendor::kATI) {
4217 fProgramBinarySupport = false;
4218 }
4219
4220 // Two Adreno 530 devices (LG G6 and OnePlus 3T) appear to have driver bugs that are corrupting
4221 // SkSL::Program memory. To get better/different crash reports, disable node-pooling, so that
4222 // program allocations aren't reused. (crbug.com/1147008, crbug.com/1164271)
4223 if (ctxInfo.renderer() == GrGLRenderer::kAdreno530) {
4224 shaderCaps->fUseNodePools = false;
4225 }
4226
4227 // skbug.com/11204. Avoid recursion issue in SurfaceContext::writePixels.
4228 if (fDisallowTexSubImageForUnormConfigTexturesEverBoundToFBO) {
4229 fReuseScratchTextures = false;
4230 }
4231
4232 // skbug.com/11935. Don't reorder on these GPUs in GL on old drivers.
4233 if ((ctxInfo.renderer() == GrGLRenderer::kAdreno620 ||
4234 ctxInfo.renderer() == GrGLRenderer::kAdreno640) &&
4235 ctxInfo.driverVersion() < GR_GL_DRIVER_VER(571, 0, 0)) {
4236 fAvoidReorderingRenderTasks = true;
4237 }
4238
4239 // http://skbug.com/11965
4240 if (ctxInfo.renderer() == GrGLRenderer::kGoogleSwiftShader) {
4241 fShaderCaps->fVertexIDSupport = false;
4242 }
4243
4244 // http://crbug.com/1197152
4245 // http://b/187364475
4246 // We could limit this < 1.13 on ChromeOS but we don't really have a good way to detect
4247 // ChromeOS from here.
4248 if (ctxInfo.renderer() == GrGLRenderer::kPowerVRRogue &&
4249 ctxInfo.driver() == GrGLDriver::kImagination &&
4250 ctxInfo.driverVersion() < GR_GL_DRIVER_VER(1, 16, 0)) {
4251 fShaderCaps->fShaderDerivativeSupport = false;
4252 }
4253
4254 if (ctxInfo.driver() == GrGLDriver::kFreedreno) {
4255 formatWorkarounds->fDisallowUnorm16Transfers = true;
4256 }
4257
4258 // If we keep rebind the same texture to an FBO's color attachment but changing between MSAA and
4259 // non-MSAA we get corruption in the texture contents. Binding texture 0 and then rebinding the
4260 // original texture avoids this.
4261 // This was found on Nexus 5, Android 6.0.1, build M4B30Z
4262 // GL_VENDOR : "Qualcomm"
4263 // GL_RENDERER: "Adreno (TM) 330"
4264 // GL_VERSION : "OpenGL ES 3.0 V@127.0 AU@ (GIT@I96aee987eb)"
4265 //
4266 // We also so alpha blending issues on these GMs skbug_9819, p3_ovals, p3 on Mali-Gxx devices
4267 // The GM issues were observed on a Galaxy S9 running Android 10:
4268 // GL_VERSION : "OpenGL ES 3.2 v1.r19p0-01rel0.###other-sha0123456789ABCDEF0###"
4269 // GL_RENDERER: "Mali-G72"
4270 // GL_VENDOR : "ARM"
4271 // and a P30 running Android 9:
4272 // GL_VERSION : "OpenGL ES 3.2 v1.r16p0-01rel0.4aee637066427cbcd25297324dba15f5"
4273 // GL_RENDERER: "Mali-G76"
4274 // GL_VENDOR : "ARM"
4275 // but *not* a Galaxy S20 running Android 10:
4276 // GL_VERSION : "OpenGL ES 3.2 v1.r20p0-01rel0.###other-sha0123456789ABCDEF0###"
4277 // GL_RENDERER: "Mali-G77"
4278 // GL_VENDOR : "ARM"
4279 // It's unclear if the difference is driver version or Bifrost vs Valhall. The workaround is
4280 // fairly trivial so just applying to all Bifrost and Valhall.
4281 if ((ctxInfo.renderer() == GrGLRenderer::kAdreno3xx &&
4282 ctxInfo.driver() == GrGLDriver::kQualcomm) ||
4283 (ctxInfo.renderer() == GrGLRenderer::kMaliG)) {
4284 fBindTexture0WhenChangingTextureFBOMultisampleCount = true;
4285 }
4286
4287 // skbug.com/12640
4288 // We found that on the Galaxy S7 the TransferPixelsTo test would fail after adding
4289 // glCheckFramebufferStatus() checks when making new FBOs. Note that the returned status was
4290 // GL_FRAMEBUFFER_COMPLETE. Switching the color binding to ID 0 and back to the original
4291 // afterwards works around the issue.
4292 // GL_VENDOR : "ARM"
4293 // GL_RENDERER: "Mali-T880"
4294 // GL_VERSION : "OpenGL ES 3.2 v1.r22p0-01rel0.f294e54ceb2cb2d81039204fa4b0402e"
4295 //
4296 // This *didn't* reproduce on a Kevin ChromeOS device:
4297 // GL_VENDOR : "ARM"
4298 // GL_RENDERER: "Mali-T860"
4299 // GL_VERSION : "OpenGL ES 3.2 v1.r26p0-01rel0.217d2597f6bd19b169343737782e56e3"
4300 if (ctxInfo.renderer() == GrGLRenderer::kMaliT &&
4301 ctxInfo.driver() == GrGLDriver::kARM &&
4302 ctxInfo.driverVersion() < GR_GL_DRIVER_VER(1, 26, 0)) {
4303 fRebindColorAttachmentAfterCheckFramebufferStatus = true;
4304 }
4305 }
4306
onApplyOptionsOverrides(const GrContextOptions & options)4307 void GrGLCaps::onApplyOptionsOverrides(const GrContextOptions& options) {
4308 if (options.fDisableDriverCorrectnessWorkarounds) {
4309 SkASSERT(!fDoManualMipmapping);
4310 SkASSERT(!fClearToBoundaryValuesIsBroken);
4311 SkASSERT(0 == fMaxInstancesPerDrawWithoutCrashing);
4312 SkASSERT(!fDrawArraysBaseVertexIsBroken);
4313 SkASSERT(!fDisallowTexSubImageForUnormConfigTexturesEverBoundToFBO);
4314 SkASSERT(!fUseDrawInsteadOfAllRenderTargetWrites);
4315 SkASSERT(!fRequiresCullFaceEnableDisableWhenDrawingLinesAfterNonLines);
4316 SkASSERT(!fDontSetBaseOrMaxLevelForExternalTextures);
4317 SkASSERT(!fNeverDisableColorWrites);
4318 }
4319 if (options.fShaderCacheStrategy < GrContextOptions::ShaderCacheStrategy::kBackendBinary) {
4320 fProgramBinarySupport = false;
4321 }
4322
4323 switch (options.fSkipGLErrorChecks) {
4324 case GrContextOptions::Enable::kNo:
4325 fSkipErrorChecks = false;
4326 break;
4327 case GrContextOptions::Enable::kYes:
4328 fSkipErrorChecks = true;
4329 break;
4330 case GrContextOptions::Enable::kDefault:
4331 break;
4332 }
4333 }
4334
onSurfaceSupportsWritePixels(const GrSurface * surface) const4335 bool GrGLCaps::onSurfaceSupportsWritePixels(const GrSurface* surface) const {
4336 if (fDisallowTexSubImageForUnormConfigTexturesEverBoundToFBO) {
4337 if (auto tex = static_cast<const GrGLTexture*>(surface->asTexture())) {
4338 if (tex->hasBaseLevelBeenBoundToFBO()) {
4339 return false;
4340 }
4341 }
4342 }
4343 if (auto rt = surface->asRenderTarget()) {
4344 if (fUseDrawInsteadOfAllRenderTargetWrites) {
4345 return false;
4346 }
4347 if (rt->numSamples() > 1 && this->usesMSAARenderBuffers()) {
4348 return false;
4349 }
4350 return SkToBool(surface->asTexture());
4351 }
4352 return true;
4353 }
4354
surfaceSupportsReadPixels(const GrSurface * surface) const4355 GrCaps::SurfaceReadPixelsSupport GrGLCaps::surfaceSupportsReadPixels(
4356 const GrSurface* surface) const {
4357 if (auto tex = static_cast<const GrGLTexture*>(surface->asTexture())) {
4358 // We don't support reading pixels directly from EXTERNAL textures as it would require
4359 // binding the texture to a FBO. For now we also disallow reading back directly
4360 // from compressed textures.
4361 if (tex->target() == GR_GL_TEXTURE_EXTERNAL || GrGLFormatIsCompressed(tex->format())) {
4362 return SurfaceReadPixelsSupport::kCopyToTexture2D;
4363 }
4364 } else if (auto rt = static_cast<const GrGLRenderTarget*>(surface->asRenderTarget())) {
4365 // glReadPixels does not allow reading back from a MSAA framebuffer. If the underlying
4366 // GrSurface doesn't have a second FBO to resolve to then we must make a copy.
4367 if (rt->numSamples() > 1 && !rt->asTexture()) {
4368 return SurfaceReadPixelsSupport::kCopyToTexture2D;
4369 }
4370 }
4371 return SurfaceReadPixelsSupport::kSupported;
4372 }
4373
offset_alignment_for_transfer_buffer(GrGLenum externalType)4374 size_t offset_alignment_for_transfer_buffer(GrGLenum externalType) {
4375 // This switch is derived from a table titled "Pixel data type parameter values and the
4376 // corresponding GL data types" in the OpenGL spec (Table 8.2 in OpenGL 4.5).
4377 switch (externalType) {
4378 case GR_GL_UNSIGNED_BYTE: return sizeof(GrGLubyte);
4379 case GR_GL_BYTE: return sizeof(GrGLbyte);
4380 case GR_GL_UNSIGNED_SHORT: return sizeof(GrGLushort);
4381 case GR_GL_SHORT: return sizeof(GrGLshort);
4382 case GR_GL_UNSIGNED_INT: return sizeof(GrGLuint);
4383 case GR_GL_INT: return sizeof(GrGLint);
4384 case GR_GL_HALF_FLOAT: return sizeof(GrGLhalf);
4385 case GR_GL_HALF_FLOAT_OES: return sizeof(GrGLhalf);
4386 case GR_GL_FLOAT: return sizeof(GrGLfloat);
4387 case GR_GL_UNSIGNED_SHORT_5_6_5: return sizeof(GrGLushort);
4388 case GR_GL_UNSIGNED_SHORT_4_4_4_4: return sizeof(GrGLushort);
4389 case GR_GL_UNSIGNED_SHORT_5_5_5_1: return sizeof(GrGLushort);
4390 case GR_GL_UNSIGNED_INT_2_10_10_10_REV: return sizeof(GrGLuint);
4391 #if 0 // GL types we currently don't use. Here for future reference.
4392 case GR_GL_UNSIGNED_BYTE_3_3_2: return sizeof(GrGLubyte);
4393 case GR_GL_UNSIGNED_BYTE_2_3_3_REV: return sizeof(GrGLubyte);
4394 case GR_GL_UNSIGNED_SHORT_5_6_5_REV: return sizeof(GrGLushort);
4395 case GR_GL_UNSIGNED_SHORT_4_4_4_4_REV: return sizeof(GrGLushort);
4396 case GR_GL_UNSIGNED_SHORT_1_5_5_5_REV: return sizeof(GrGLushort);
4397 case GR_GL_UNSIGNED_INT_8_8_8_8: return sizeof(GrGLuint);
4398 case GR_GL_UNSIGNED_INT_8_8_8_8_REV: return sizeof(GrGLuint);
4399 case GR_GL_UNSIGNED_INT_10_10_10_2: return sizeof(GrGLuint);
4400 case GR_GL_UNSIGNED_INT_24_8: return sizeof(GrGLuint);
4401 case GR_GL_UNSIGNED_INT_10F_11F_11F_REV: return sizeof(GrGLuint);
4402 case GR_GL_UNSIGNED_INT_5_9_9_9_REV: return sizeof(GrGLuint);
4403 // This one is not corresponding to a GL data type and the spec just says it is 4.
4404 case GR_GL_FLOAT_32_UNSIGNED_INT_24_8_REV: return 4;
4405 #endif
4406 default: return 0;
4407 }
4408 }
4409
onSupportedReadPixelsColorType(GrColorType srcColorType,const GrBackendFormat & srcBackendFormat,GrColorType dstColorType) const4410 GrCaps::SupportedRead GrGLCaps::onSupportedReadPixelsColorType(
4411 GrColorType srcColorType, const GrBackendFormat& srcBackendFormat,
4412 GrColorType dstColorType) const {
4413
4414 SkImage::CompressionType compression = GrBackendFormatToCompressionType(srcBackendFormat);
4415 if (compression != SkImage::CompressionType::kNone) {
4416 return {SkCompressionTypeIsOpaque(compression) ? GrColorType::kRGB_888x
4417 : GrColorType::kRGBA_8888,
4418 0};
4419 }
4420
4421 // We first try to find a supported read pixels GrColorType that matches the requested
4422 // dstColorType. If that doesn't exists we will use any valid read pixels GrColorType.
4423 GrCaps::SupportedRead fallbackRead = {GrColorType::kUnknown, 0};
4424 const auto& formatInfo = this->getFormatInfo(srcBackendFormat.asGLFormat());
4425 bool foundSrcCT = false;
4426 for (int i = 0; !foundSrcCT && i < formatInfo.fColorTypeInfoCount; ++i) {
4427 if (formatInfo.fColorTypeInfos[i].fColorType == srcColorType) {
4428 const ColorTypeInfo& ctInfo = formatInfo.fColorTypeInfos[i];
4429 foundSrcCT = true;
4430 for (int j = 0; j < ctInfo.fExternalIOFormatCount; ++j) {
4431 const auto& ioInfo = ctInfo.fExternalIOFormats[j];
4432 if (ioInfo.fExternalReadFormat != 0) {
4433 if (formatInfo.fHaveQueriedImplementationReadSupport ||
4434 !ioInfo.fRequiresImplementationReadQuery) {
4435 GrGLenum transferOffsetAlignment = 0;
4436 if (formatInfo.fFlags & FormatInfo::kTransfers_Flag) {
4437 transferOffsetAlignment =
4438 offset_alignment_for_transfer_buffer(ioInfo.fExternalType);
4439 }
4440 if (ioInfo.fColorType == dstColorType) {
4441 return {dstColorType, transferOffsetAlignment};
4442 }
4443 // Currently we just pick the first supported format that we find as our
4444 // fallback.
4445 if (fallbackRead.fColorType == GrColorType::kUnknown) {
4446 fallbackRead = {ioInfo.fColorType, transferOffsetAlignment};
4447 }
4448 }
4449 }
4450 }
4451 }
4452 }
4453 return fallbackRead;
4454 }
4455
supportedWritePixelsColorType(GrColorType surfaceColorType,const GrBackendFormat & surfaceFormat,GrColorType srcColorType) const4456 GrCaps::SupportedWrite GrGLCaps::supportedWritePixelsColorType(GrColorType surfaceColorType,
4457 const GrBackendFormat& surfaceFormat,
4458 GrColorType srcColorType) const {
4459 // We first try to find a supported write pixels GrColorType that matches the data's
4460 // srcColorType. If that doesn't exists we will use any supported GrColorType.
4461 GrColorType fallbackCT = GrColorType::kUnknown;
4462 const auto& formatInfo = this->getFormatInfo(surfaceFormat.asGLFormat());
4463 bool foundSurfaceCT = false;
4464 size_t transferOffsetAlignment = 0;
4465 if (formatInfo.fFlags & FormatInfo::kTransfers_Flag) {
4466 transferOffsetAlignment = 1;
4467 }
4468 for (int i = 0; !foundSurfaceCT && i < formatInfo.fColorTypeInfoCount; ++i) {
4469 if (formatInfo.fColorTypeInfos[i].fColorType == surfaceColorType) {
4470 const ColorTypeInfo& ctInfo = formatInfo.fColorTypeInfos[i];
4471 foundSurfaceCT = true;
4472 for (int j = 0; j < ctInfo.fExternalIOFormatCount; ++j) {
4473 const auto& ioInfo = ctInfo.fExternalIOFormats[j];
4474 if (ioInfo.fExternalTexImageFormat != 0) {
4475 if (ioInfo.fColorType == srcColorType) {
4476 return {srcColorType, transferOffsetAlignment};
4477 }
4478 // Currently we just pick the first supported format that we find as our
4479 // fallback.
4480 if (fallbackCT == GrColorType::kUnknown) {
4481 fallbackCT = ioInfo.fColorType;
4482 }
4483 }
4484 }
4485 }
4486 }
4487 return {fallbackCT, transferOffsetAlignment};
4488 }
4489
onIsWindowRectanglesSupportedForRT(const GrBackendRenderTarget & backendRT) const4490 bool GrGLCaps::onIsWindowRectanglesSupportedForRT(const GrBackendRenderTarget& backendRT) const {
4491 GrGLFramebufferInfo fbInfo;
4492 SkAssertResult(backendRT.getGLFramebufferInfo(&fbInfo));
4493 // Window Rectangles are not supported for FBO 0;
4494 return fbInfo.fFBOID != 0;
4495 }
4496
isFormatSRGB(const GrBackendFormat & format) const4497 bool GrGLCaps::isFormatSRGB(const GrBackendFormat& format) const {
4498 return format.asGLFormat() == GrGLFormat::kSRGB8_ALPHA8;
4499 }
4500
isFormatTexturable(const GrBackendFormat & format,GrTextureType textureType) const4501 bool GrGLCaps::isFormatTexturable(const GrBackendFormat& format, GrTextureType textureType) const {
4502 if (textureType == GrTextureType::kRectangle && !this->rectangleTextureSupport()) {
4503 return false;
4504 }
4505 return this->isFormatTexturable(format.asGLFormat());
4506 }
4507
isFormatTexturable(GrGLFormat format) const4508 bool GrGLCaps::isFormatTexturable(GrGLFormat format) const {
4509 const FormatInfo& info = this->getFormatInfo(format);
4510 return SkToBool(info.fFlags & FormatInfo::kTexturable_Flag);
4511 }
4512
isFormatAsColorTypeRenderable(GrColorType ct,const GrBackendFormat & format,int sampleCount) const4513 bool GrGLCaps::isFormatAsColorTypeRenderable(GrColorType ct, const GrBackendFormat& format,
4514 int sampleCount) const {
4515 if (format.textureType() == GrTextureType::kRectangle && !this->rectangleTextureSupport()) {
4516 return false;
4517 }
4518 if (format.textureType() == GrTextureType::kExternal) {
4519 return false;
4520 }
4521 auto f = format.asGLFormat();
4522 const FormatInfo& info = this->getFormatInfo(f);
4523 if (!SkToBool(info.colorTypeFlags(ct) & ColorTypeInfo::kRenderable_Flag)) {
4524 return false;
4525 }
4526
4527 return this->isFormatRenderable(f, sampleCount);
4528 }
4529
isFormatRenderable(const GrBackendFormat & format,int sampleCount) const4530 bool GrGLCaps::isFormatRenderable(const GrBackendFormat& format, int sampleCount) const {
4531 if (format.textureType() == GrTextureType::kRectangle && !this->rectangleTextureSupport()) {
4532 return false;
4533 }
4534 if (format.textureType() == GrTextureType::kExternal) {
4535 return false;
4536 }
4537 return this->isFormatRenderable(format.asGLFormat(), sampleCount);
4538 }
4539
getRenderTargetSampleCount(int requestedCount,GrGLFormat format) const4540 int GrGLCaps::getRenderTargetSampleCount(int requestedCount, GrGLFormat format) const {
4541 const FormatInfo& info = this->getFormatInfo(format);
4542
4543 int count = info.fColorSampleCounts.count();
4544 if (!count) {
4545 return 0;
4546 }
4547
4548 requestedCount = std::max(1, requestedCount);
4549 if (1 == requestedCount) {
4550 return info.fColorSampleCounts[0] == 1 ? 1 : 0;
4551 }
4552
4553 for (int sampleCount : info.fColorSampleCounts) {
4554 if (sampleCount >= requestedCount) {
4555 if (fDriverBugWorkarounds.max_msaa_sample_count_4) {
4556 sampleCount = std::min(sampleCount, 4);
4557 }
4558 return sampleCount;
4559 }
4560 }
4561 return 0;
4562 }
4563
maxRenderTargetSampleCount(GrGLFormat format) const4564 int GrGLCaps::maxRenderTargetSampleCount(GrGLFormat format) const {
4565 const FormatInfo& info = this->getFormatInfo(format);
4566 const auto& table = info.fColorSampleCounts;
4567 if (!table.count()) {
4568 return 0;
4569 }
4570 int count = table[table.count() - 1];
4571 if (fDriverBugWorkarounds.max_msaa_sample_count_4) {
4572 count = std::min(count, 4);
4573 }
4574 return count;
4575 }
4576
canFormatBeFBOColorAttachment(GrGLFormat format) const4577 bool GrGLCaps::canFormatBeFBOColorAttachment(GrGLFormat format) const {
4578 return SkToBool(this->getFormatInfo(format).fFlags & FormatInfo::kFBOColorAttachment_Flag);
4579 }
4580
isFormatCopyable(const GrBackendFormat & format) const4581 bool GrGLCaps::isFormatCopyable(const GrBackendFormat& format) const {
4582 // In GL we have three ways to be able to copy. CopyTexImage, blit, and draw. CopyTexImage
4583 // requires the src to be an FBO attachment, blit requires both src and dst to be FBO
4584 // attachments, and draw requires the dst to be an FBO attachment. Thus to copy from and to
4585 // the same config, we need that config to be bindable to an FBO.
4586 return this->canFormatBeFBOColorAttachment(format.asGLFormat());
4587 }
4588
formatSupportsTexStorage(GrGLFormat format) const4589 bool GrGLCaps::formatSupportsTexStorage(GrGLFormat format) const {
4590 return SkToBool(this->getFormatInfo(format).fFlags & FormatInfo::kUseTexStorage_Flag);
4591 }
4592
shouldQueryImplementationReadSupport(GrGLFormat format) const4593 bool GrGLCaps::shouldQueryImplementationReadSupport(GrGLFormat format) const {
4594 const auto& formatInfo = const_cast<GrGLCaps*>(this)->getFormatInfo(format);
4595 if (!formatInfo.fHaveQueriedImplementationReadSupport) {
4596 // Check whether we will actually learn anything useful.
4597 bool needQuery = false;
4598 for (int i = 0; i < formatInfo.fColorTypeInfoCount && !needQuery; ++i) {
4599 const auto& surfCTInfo = formatInfo.fColorTypeInfos[i];
4600 for (int j = 0; j < surfCTInfo.fExternalIOFormatCount; ++j) {
4601 if (surfCTInfo.fExternalIOFormats[j].fRequiresImplementationReadQuery) {
4602 needQuery = true;
4603 break;
4604 }
4605 }
4606 }
4607 if (!needQuery) {
4608 // Pretend we already checked it.
4609 const_cast<FormatInfo&>(formatInfo).fHaveQueriedImplementationReadSupport = true;
4610 }
4611 }
4612 return !formatInfo.fHaveQueriedImplementationReadSupport;
4613 }
4614
didQueryImplementationReadSupport(GrGLFormat format,GrGLenum readFormat,GrGLenum readType) const4615 void GrGLCaps::didQueryImplementationReadSupport(GrGLFormat format,
4616 GrGLenum readFormat,
4617 GrGLenum readType) const {
4618 auto& formatInfo = const_cast<GrGLCaps*>(this)->getFormatInfo(format);
4619 for (int i = 0; i < formatInfo.fColorTypeInfoCount; ++i) {
4620 auto& surfCTInfo = formatInfo.fColorTypeInfos[i];
4621 for (int j = 0; j < surfCTInfo.fExternalIOFormatCount; ++j) {
4622 auto& readCTInfo = surfCTInfo.fExternalIOFormats[j];
4623 if (readCTInfo.fRequiresImplementationReadQuery) {
4624 if (readCTInfo.fExternalReadFormat != readFormat ||
4625 readCTInfo.fExternalType != readType) {
4626 // Don't zero out fExternalType. It's also used for writing data to the texture!
4627 readCTInfo.fExternalReadFormat = 0;
4628 }
4629 }
4630 }
4631 }
4632 formatInfo.fHaveQueriedImplementationReadSupport = true;
4633 }
4634
onAreColorTypeAndFormatCompatible(GrColorType ct,const GrBackendFormat & format) const4635 bool GrGLCaps::onAreColorTypeAndFormatCompatible(GrColorType ct,
4636 const GrBackendFormat& format) const {
4637 GrGLFormat glFormat = format.asGLFormat();
4638 const auto& info = this->getFormatInfo(glFormat);
4639 for (int i = 0; i < info.fColorTypeInfoCount; ++i) {
4640 if (info.fColorTypeInfos[i].fColorType == ct) {
4641 return true;
4642 }
4643 }
4644 return false;
4645 }
4646
onGetDefaultBackendFormat(GrColorType ct) const4647 GrBackendFormat GrGLCaps::onGetDefaultBackendFormat(GrColorType ct) const {
4648 auto format = this->getFormatFromColorType(ct);
4649 if (format == GrGLFormat::kUnknown) {
4650 return {};
4651 }
4652 return GrBackendFormat::MakeGL(GrGLFormatToEnum(format), GR_GL_TEXTURE_2D);
4653 }
4654
getBackendFormatFromCompressionType(SkImage::CompressionType compressionType) const4655 GrBackendFormat GrGLCaps::getBackendFormatFromCompressionType(
4656 SkImage::CompressionType compressionType) const {
4657 switch (compressionType) {
4658 case SkImage::CompressionType::kNone:
4659 return {};
4660 case SkImage::CompressionType::kETC2_RGB8_UNORM:
4661 // if ETC2 is available default to that format
4662 if (this->isFormatTexturable(GrGLFormat::kCOMPRESSED_RGB8_ETC2)) {
4663 return GrBackendFormat::MakeGL(GR_GL_COMPRESSED_RGB8_ETC2, GR_GL_TEXTURE_2D);
4664 }
4665 if (this->isFormatTexturable(GrGLFormat::kCOMPRESSED_ETC1_RGB8)) {
4666 return GrBackendFormat::MakeGL(GR_GL_COMPRESSED_ETC1_RGB8, GR_GL_TEXTURE_2D);
4667 }
4668 return {};
4669 case SkImage::CompressionType::kBC1_RGB8_UNORM:
4670 if (this->isFormatTexturable(GrGLFormat::kCOMPRESSED_RGB8_BC1)) {
4671 return GrBackendFormat::MakeGL(GR_GL_COMPRESSED_RGB_S3TC_DXT1_EXT,
4672 GR_GL_TEXTURE_2D);
4673 }
4674 return {};
4675 case SkImage::CompressionType::kBC1_RGBA8_UNORM:
4676 if (this->isFormatTexturable(GrGLFormat::kCOMPRESSED_RGBA8_BC1)) {
4677 return GrBackendFormat::MakeGL(GR_GL_COMPRESSED_RGBA_S3TC_DXT1_EXT,
4678 GR_GL_TEXTURE_2D);
4679 }
4680 return {};
4681 }
4682
4683 SkUNREACHABLE;
4684 }
4685
onGetReadSwizzle(const GrBackendFormat & format,GrColorType colorType) const4686 GrSwizzle GrGLCaps::onGetReadSwizzle(const GrBackendFormat& format, GrColorType colorType) const {
4687 GrGLFormat glFormat = format.asGLFormat();
4688 const auto& info = this->getFormatInfo(glFormat);
4689 for (int i = 0; i < info.fColorTypeInfoCount; ++i) {
4690 const auto& ctInfo = info.fColorTypeInfos[i];
4691 if (ctInfo.fColorType == colorType) {
4692 return ctInfo.fReadSwizzle;
4693 }
4694 }
4695 SkDEBUGFAILF("Illegal color type (%d) and format (%d) combination.",
4696 (int)colorType, (int)glFormat);
4697 return {};
4698 }
4699
getWriteSwizzle(const GrBackendFormat & format,GrColorType colorType) const4700 GrSwizzle GrGLCaps::getWriteSwizzle(const GrBackendFormat& format, GrColorType colorType) const {
4701 const auto& info = this->getFormatInfo(format.asGLFormat());
4702 for (int i = 0; i < info.fColorTypeInfoCount; ++i) {
4703 const auto& ctInfo = info.fColorTypeInfos[i];
4704 if (ctInfo.fColorType == colorType) {
4705 return ctInfo.fWriteSwizzle;
4706 }
4707 }
4708 SkDEBUGFAILF("Illegal color type (%d) and format (%d) combination.",
4709 (int)colorType, (int)format.asGLFormat());
4710 return {};
4711 }
4712
onGetDstSampleFlagsForProxy(const GrRenderTargetProxy * rt) const4713 GrDstSampleFlags GrGLCaps::onGetDstSampleFlagsForProxy(const GrRenderTargetProxy* rt) const {
4714 if (rt->asTextureProxy()) {
4715 return GrDstSampleFlags::kRequiresTextureBarrier;
4716 }
4717 return GrDstSampleFlags::kNone;
4718 }
4719
onSupportsDynamicMSAA(const GrRenderTargetProxy * rtProxy) const4720 bool GrGLCaps::onSupportsDynamicMSAA(const GrRenderTargetProxy* rtProxy) const {
4721 return !fDisallowDynamicMSAA;
4722 }
4723
computeFormatKey(const GrBackendFormat & format) const4724 uint64_t GrGLCaps::computeFormatKey(const GrBackendFormat& format) const {
4725 auto glFormat = format.asGLFormat();
4726 return (uint64_t)(glFormat);
4727 }
4728
makeDesc(GrRenderTarget *,const GrProgramInfo & programInfo,ProgramDescOverrideFlags overrideFlags) const4729 GrProgramDesc GrGLCaps::makeDesc(GrRenderTarget* /* rt */,
4730 const GrProgramInfo& programInfo,
4731 ProgramDescOverrideFlags overrideFlags) const {
4732 SkASSERT(overrideFlags == ProgramDescOverrideFlags::kNone);
4733 GrProgramDesc desc;
4734 GrProgramDesc::Build(&desc, programInfo, *this);
4735 return desc;
4736 }
4737
4738 #if GR_TEST_UTILS
getTestingCombinations() const4739 std::vector<GrCaps::TestFormatColorTypeCombination> GrGLCaps::getTestingCombinations() const {
4740 std::vector<GrCaps::TestFormatColorTypeCombination> combos = {
4741 { GrColorType::kAlpha_8,
4742 GrBackendFormat::MakeGL(GR_GL_ALPHA8, GR_GL_TEXTURE_2D) },
4743 { GrColorType::kAlpha_8,
4744 GrBackendFormat::MakeGL(GR_GL_R8, GR_GL_TEXTURE_2D) },
4745 { GrColorType::kBGR_565,
4746 GrBackendFormat::MakeGL(GR_GL_RGB565, GR_GL_TEXTURE_2D) },
4747 { GrColorType::kABGR_4444,
4748 GrBackendFormat::MakeGL(GR_GL_RGBA4, GR_GL_TEXTURE_2D) },
4749 { GrColorType::kRGBA_8888,
4750 GrBackendFormat::MakeGL(GR_GL_RGBA8, GR_GL_TEXTURE_2D) },
4751 { GrColorType::kRGBA_8888_SRGB,
4752 GrBackendFormat::MakeGL(GR_GL_SRGB8_ALPHA8, GR_GL_TEXTURE_2D) },
4753 { GrColorType::kRGB_888x,
4754 GrBackendFormat::MakeGL(GR_GL_RGBA8, GR_GL_TEXTURE_2D) },
4755 { GrColorType::kRGB_888x,
4756 GrBackendFormat::MakeGL(GR_GL_RGB8, GR_GL_TEXTURE_2D) },
4757 { GrColorType::kRGB_888x,
4758 GrBackendFormat::MakeGL(GR_GL_COMPRESSED_RGB8_ETC2, GR_GL_TEXTURE_2D) },
4759 { GrColorType::kRGB_888x,
4760 GrBackendFormat::MakeGL(GR_GL_COMPRESSED_ETC1_RGB8, GR_GL_TEXTURE_2D) },
4761 { GrColorType::kRGB_888x,
4762 GrBackendFormat::MakeGL(GR_GL_COMPRESSED_RGB_S3TC_DXT1_EXT, GR_GL_TEXTURE_2D) },
4763 { GrColorType::kRGBA_8888,
4764 GrBackendFormat::MakeGL(GR_GL_COMPRESSED_RGBA_S3TC_DXT1_EXT, GR_GL_TEXTURE_2D) },
4765 { GrColorType::kRG_88,
4766 GrBackendFormat::MakeGL(GR_GL_RG8, GR_GL_TEXTURE_2D) },
4767 { GrColorType::kRGBA_1010102,
4768 GrBackendFormat::MakeGL(GR_GL_RGB10_A2, GR_GL_TEXTURE_2D) },
4769 { GrColorType::kGray_8,
4770 GrBackendFormat::MakeGL(GR_GL_LUMINANCE8, GR_GL_TEXTURE_2D) },
4771 { GrColorType::kGray_8,
4772 GrBackendFormat::MakeGL(GR_GL_R8, GR_GL_TEXTURE_2D) },
4773 { GrColorType::kGrayAlpha_88,
4774 GrBackendFormat::MakeGL(GR_GL_LUMINANCE8_ALPHA8, GR_GL_TEXTURE_2D) },
4775 { GrColorType::kAlpha_F16,
4776 GrBackendFormat::MakeGL(GR_GL_R16F, GR_GL_TEXTURE_2D) },
4777 { GrColorType::kAlpha_F16,
4778 GrBackendFormat::MakeGL(GR_GL_LUMINANCE16F, GR_GL_TEXTURE_2D) },
4779 { GrColorType::kRGBA_F16,
4780 GrBackendFormat::MakeGL(GR_GL_RGBA16F, GR_GL_TEXTURE_2D) },
4781 { GrColorType::kRGBA_F16_Clamped,
4782 GrBackendFormat::MakeGL(GR_GL_RGBA16F, GR_GL_TEXTURE_2D) },
4783 { GrColorType::kAlpha_16,
4784 GrBackendFormat::MakeGL(GR_GL_R16, GR_GL_TEXTURE_2D) },
4785 { GrColorType::kRG_1616,
4786 GrBackendFormat::MakeGL(GR_GL_RG16, GR_GL_TEXTURE_2D) },
4787 { GrColorType::kRGBA_16161616,
4788 GrBackendFormat::MakeGL(GR_GL_RGBA16, GR_GL_TEXTURE_2D) },
4789 { GrColorType::kRG_F16,
4790 GrBackendFormat::MakeGL(GR_GL_RG16F, GR_GL_TEXTURE_2D) },
4791 };
4792
4793 if (GR_IS_GR_GL(fStandard)) {
4794 combos.push_back({ GrColorType::kBGRA_8888,
4795 GrBackendFormat::MakeGL(GR_GL_RGBA8, GR_GL_TEXTURE_2D) });
4796 combos.push_back({ GrColorType::kBGRA_1010102,
4797 GrBackendFormat::MakeGL(GR_GL_RGB10_A2, GR_GL_TEXTURE_2D) });
4798 } else {
4799 SkASSERT(GR_IS_GR_GL_ES(fStandard) || GR_IS_GR_WEBGL(fStandard));
4800
4801 combos.push_back({ GrColorType::kBGRA_8888,
4802 GrBackendFormat::MakeGL(GR_GL_BGRA8, GR_GL_TEXTURE_2D) });
4803 }
4804 if (this->rectangleTextureSupport()) {
4805 size_t count2D = combos.size();
4806 for (size_t i = 0; i < count2D; ++i) {
4807 auto combo2D = combos[i];
4808 GrGLenum formatEnum = GrGLFormatToEnum(combo2D.fFormat.asGLFormat());
4809 combos.push_back({combo2D.fColorType,
4810 GrBackendFormat::MakeGL(formatEnum, GR_GL_TEXTURE_RECTANGLE)});
4811 }
4812 }
4813 return combos;
4814 }
4815 #endif
4816