1 /* 2 * Copyright 2015 Google Inc. 3 * 4 * Use of this source code is governed by a BSD-style license that can be 5 * found in the LICENSE file. 6 */ 7 8 #include "GrCaps.h" 9 10 #include "GrBackendSurface.h" 11 #include "GrContextOptions.h" 12 #include "GrWindowRectangles.h" 13 #include "SkJSONWriter.h" 14 15 static const char* pixel_config_name(GrPixelConfig config) { 16 switch (config) { 17 case kUnknown_GrPixelConfig: return "Unknown"; 18 case kAlpha_8_GrPixelConfig: return "Alpha8"; 19 case kAlpha_8_as_Alpha_GrPixelConfig: return "Alpha8_asAlpha"; 20 case kAlpha_8_as_Red_GrPixelConfig: return "Alpha8_asRed"; 21 case kGray_8_GrPixelConfig: return "Gray8"; 22 case kGray_8_as_Lum_GrPixelConfig: return "Gray8_asLum"; 23 case kGray_8_as_Red_GrPixelConfig: return "Gray8_asRed"; 24 case kRGB_565_GrPixelConfig: return "RGB565"; 25 case kRGBA_4444_GrPixelConfig: return "RGBA444"; 26 case kRGBA_8888_GrPixelConfig: return "RGBA8888"; 27 case kBGRA_8888_GrPixelConfig: return "BGRA8888"; 28 case kSRGBA_8888_GrPixelConfig: return "SRGBA8888"; 29 case kSBGRA_8888_GrPixelConfig: return "SBGRA8888"; 30 case kRGBA_1010102_GrPixelConfig: return "RGBA1010102"; 31 case kRGBA_float_GrPixelConfig: return "RGBAFloat"; 32 case kRG_float_GrPixelConfig: return "RGFloat"; 33 case kAlpha_half_GrPixelConfig: return "AlphaHalf"; 34 case kAlpha_half_as_Red_GrPixelConfig: return "AlphaHalf_asRed"; 35 case kRGBA_half_GrPixelConfig: return "RGBAHalf"; 36 } 37 SK_ABORT("Invalid pixel config"); 38 return "<invalid>"; 39 } 40 41 GrCaps::GrCaps(const GrContextOptions& options) { 42 fMipMapSupport = false; 43 fNPOTTextureTileSupport = false; 44 fSRGBSupport = false; 45 fSRGBWriteControl = false; 46 fSRGBDecodeDisableSupport = false; 47 fDiscardRenderTargetSupport = false; 48 fReuseScratchTextures = true; 49 fReuseScratchBuffers = true; 50 fGpuTracingSupport = false; 51 fOversizedStencilSupport = false; 52 fTextureBarrierSupport = false; 53 fSampleLocationsSupport = false; 54 fMultisampleDisableSupport = false; 55 fInstanceAttribSupport = false; 56 fUsesMixedSamples = false; 57 fUsePrimitiveRestart = false; 58 fPreferClientSideDynamicBuffers = false; 59 fPreferFullscreenClears = false; 60 fMustClearUploadedBufferData = false; 61 fSampleShadingSupport = false; 62 fFenceSyncSupport = false; 63 fCrossContextTextureSupport = false; 64 65 fBlendEquationSupport = kBasic_BlendEquationSupport; 66 fAdvBlendEqBlacklist = 0; 67 68 fMapBufferFlags = kNone_MapFlags; 69 70 fMaxVertexAttributes = 0; 71 fMaxRenderTargetSize = 1; 72 fMaxPreferredRenderTargetSize = 1; 73 fMaxTextureSize = 1; 74 fMaxRasterSamples = 0; 75 fMaxWindowRectangles = 0; 76 77 // An default count of 4 was chosen because of the common pattern in Blink of: 78 // isect RR 79 // diff RR 80 // isect convex_poly 81 // isect convex_poly 82 // when drawing rounded div borders. 83 fMaxClipAnalyticFPs = 4; 84 85 fSuppressPrints = options.fSuppressPrints; 86 #if GR_TEST_UTILS 87 fWireframeMode = options.fWireframeMode; 88 #else 89 fWireframeMode = false; 90 #endif 91 fBufferMapThreshold = options.fBufferMapThreshold; 92 fBlacklistCoverageCounting = false; 93 fAvoidStencilBuffers = false; 94 95 fPreferVRAMUseOverFlushes = true; 96 } 97 98 void GrCaps::applyOptionsOverrides(const GrContextOptions& options) { 99 this->onApplyOptionsOverrides(options); 100 if (options.fDisableDriverCorrectnessWorkarounds) { 101 // We always blacklist coverage counting on Vulkan currently. TODO: Either stop doing that 102 // or disambiguate blacklisting from incomplete implementation. 103 // SkASSERT(!fBlacklistCoverageCounting); 104 SkASSERT(!fAvoidStencilBuffers); 105 SkASSERT(!fAdvBlendEqBlacklist); 106 } 107 108 fMaxTextureSize = SkTMin(fMaxTextureSize, options.fMaxTextureSizeOverride); 109 fMaxTileSize = fMaxTextureSize; 110 #if GR_TEST_UTILS 111 // If the max tile override is zero, it means we should use the max texture size. 112 if (options.fMaxTileSizeOverride && options.fMaxTileSizeOverride < fMaxTextureSize) { 113 fMaxTileSize = options.fMaxTileSizeOverride; 114 } 115 if (options.fSuppressGeometryShaders) { 116 fShaderCaps->fGeometryShaderSupport = false; 117 } 118 #endif 119 if (fMaxWindowRectangles > GrWindowRectangles::kMaxWindows) { 120 SkDebugf("WARNING: capping window rectangles at %i. HW advertises support for %i.\n", 121 GrWindowRectangles::kMaxWindows, fMaxWindowRectangles); 122 fMaxWindowRectangles = GrWindowRectangles::kMaxWindows; 123 } 124 fAvoidStencilBuffers = options.fAvoidStencilBuffers; 125 } 126 127 static SkString map_flags_to_string(uint32_t flags) { 128 SkString str; 129 if (GrCaps::kNone_MapFlags == flags) { 130 str = "none"; 131 } else { 132 SkASSERT(GrCaps::kCanMap_MapFlag & flags); 133 SkDEBUGCODE(flags &= ~GrCaps::kCanMap_MapFlag); 134 str = "can_map"; 135 136 if (GrCaps::kSubset_MapFlag & flags) { 137 str.append(" partial"); 138 } else { 139 str.append(" full"); 140 } 141 SkDEBUGCODE(flags &= ~GrCaps::kSubset_MapFlag); 142 } 143 SkASSERT(0 == flags); // Make sure we handled all the flags. 144 return str; 145 } 146 147 void GrCaps::dumpJSON(SkJSONWriter* writer) const { 148 writer->beginObject(); 149 150 writer->appendBool("MIP Map Support", fMipMapSupport); 151 writer->appendBool("NPOT Texture Tile Support", fNPOTTextureTileSupport); 152 writer->appendBool("sRGB Support", fSRGBSupport); 153 writer->appendBool("sRGB Write Control", fSRGBWriteControl); 154 writer->appendBool("sRGB Decode Disable", fSRGBDecodeDisableSupport); 155 writer->appendBool("Discard Render Target Support", fDiscardRenderTargetSupport); 156 writer->appendBool("Reuse Scratch Textures", fReuseScratchTextures); 157 writer->appendBool("Reuse Scratch Buffers", fReuseScratchBuffers); 158 writer->appendBool("Gpu Tracing Support", fGpuTracingSupport); 159 writer->appendBool("Oversized Stencil Support", fOversizedStencilSupport); 160 writer->appendBool("Texture Barrier Support", fTextureBarrierSupport); 161 writer->appendBool("Sample Locations Support", fSampleLocationsSupport); 162 writer->appendBool("Multisample disable support", fMultisampleDisableSupport); 163 writer->appendBool("Instance Attrib Support", fInstanceAttribSupport); 164 writer->appendBool("Uses Mixed Samples", fUsesMixedSamples); 165 writer->appendBool("Use primitive restart", fUsePrimitiveRestart); 166 writer->appendBool("Prefer client-side dynamic buffers", fPreferClientSideDynamicBuffers); 167 writer->appendBool("Prefer fullscreen clears", fPreferFullscreenClears); 168 writer->appendBool("Must clear buffer memory", fMustClearUploadedBufferData); 169 writer->appendBool("Sample shading support", fSampleShadingSupport); 170 writer->appendBool("Fence sync support", fFenceSyncSupport); 171 writer->appendBool("Cross context texture support", fCrossContextTextureSupport); 172 173 writer->appendBool("Blacklist Coverage Counting Path Renderer [workaround]", 174 fBlacklistCoverageCounting); 175 writer->appendBool("Prefer VRAM Use over flushes [workaround]", fPreferVRAMUseOverFlushes); 176 writer->appendBool("Avoid stencil buffers [workaround]", fAvoidStencilBuffers); 177 178 if (this->advancedBlendEquationSupport()) { 179 writer->appendHexU32("Advanced Blend Equation Blacklist", fAdvBlendEqBlacklist); 180 } 181 182 writer->appendS32("Max Vertex Attributes", fMaxVertexAttributes); 183 writer->appendS32("Max Texture Size", fMaxTextureSize); 184 writer->appendS32("Max Render Target Size", fMaxRenderTargetSize); 185 writer->appendS32("Max Preferred Render Target Size", fMaxPreferredRenderTargetSize); 186 writer->appendS32("Max Raster Samples", fMaxRasterSamples); 187 writer->appendS32("Max Window Rectangles", fMaxWindowRectangles); 188 writer->appendS32("Max Clip Analytic Fragment Processors", fMaxClipAnalyticFPs); 189 190 static const char* kBlendEquationSupportNames[] = { 191 "Basic", 192 "Advanced", 193 "Advanced Coherent", 194 }; 195 GR_STATIC_ASSERT(0 == kBasic_BlendEquationSupport); 196 GR_STATIC_ASSERT(1 == kAdvanced_BlendEquationSupport); 197 GR_STATIC_ASSERT(2 == kAdvancedCoherent_BlendEquationSupport); 198 GR_STATIC_ASSERT(SK_ARRAY_COUNT(kBlendEquationSupportNames) == kLast_BlendEquationSupport + 1); 199 200 writer->appendString("Blend Equation Support", 201 kBlendEquationSupportNames[fBlendEquationSupport]); 202 writer->appendString("Map Buffer Support", map_flags_to_string(fMapBufferFlags).c_str()); 203 204 SkASSERT(!this->isConfigRenderable(kUnknown_GrPixelConfig)); 205 SkASSERT(!this->isConfigTexturable(kUnknown_GrPixelConfig)); 206 207 writer->beginArray("configs"); 208 209 for (size_t i = 1; i < kGrPixelConfigCnt; ++i) { 210 GrPixelConfig config = static_cast<GrPixelConfig>(i); 211 writer->beginObject(nullptr, false); 212 writer->appendString("name", pixel_config_name(config)); 213 writer->appendS32("max sample count", this->maxRenderTargetSampleCount(config)); 214 writer->appendBool("texturable", this->isConfigTexturable(config)); 215 writer->endObject(); 216 } 217 218 writer->endArray(); 219 220 this->onDumpJSON(writer); 221 222 writer->appendName("shaderCaps"); 223 this->shaderCaps()->dumpJSON(writer); 224 225 writer->endObject(); 226 } 227 228 bool GrCaps::validateSurfaceDesc(const GrSurfaceDesc& desc, GrMipMapped mipped) const { 229 if (!this->isConfigTexturable(desc.fConfig)) { 230 return false; 231 } 232 233 if (GrMipMapped::kYes == mipped && !this->mipMapSupport()) { 234 return false; 235 } 236 237 if (desc.fWidth < 1 || desc.fHeight < 1) { 238 return false; 239 } 240 241 if (SkToBool(desc.fFlags & kRenderTarget_GrSurfaceFlag)) { 242 if (0 == this->getRenderTargetSampleCount(desc.fSampleCnt, desc.fConfig)) { 243 return false; 244 } 245 int maxRTSize = this->maxRenderTargetSize(); 246 if (desc.fWidth > maxRTSize || desc.fHeight > maxRTSize) { 247 return false; 248 } 249 } else { 250 // We currently do not support multisampled textures 251 if (desc.fSampleCnt > 1) { 252 return false; 253 } 254 int maxSize = this->maxTextureSize(); 255 if (desc.fWidth > maxSize || desc.fHeight > maxSize) { 256 return false; 257 } 258 } 259 260 return true; 261 } 262