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 #ifndef GrPipeline_DEFINED 9 #define GrPipeline_DEFINED 10 11 #include "include/core/SkMatrix.h" 12 #include "include/core/SkRefCnt.h" 13 #include "src/gpu/GrColor.h" 14 #include "src/gpu/GrFragmentProcessor.h" 15 #include "src/gpu/GrNonAtomicRef.h" 16 #include "src/gpu/GrProcessorSet.h" 17 #include "src/gpu/GrScissorState.h" 18 #include "src/gpu/GrSurfaceProxyView.h" 19 #include "src/gpu/GrUserStencilSettings.h" 20 #include "src/gpu/GrWindowRectsState.h" 21 #include "src/gpu/effects/GrCoverageSetOpXP.h" 22 #include "src/gpu/effects/GrDisableColorXP.h" 23 #include "src/gpu/effects/GrPorterDuffXferProcessor.h" 24 #include "src/gpu/effects/GrTextureEffect.h" 25 #include "src/gpu/geometry/GrRect.h" 26 27 class GrAppliedClip; 28 class GrAppliedHardClip; 29 class GrOp; 30 class GrRenderTargetContext; 31 32 /** 33 * This immutable object contains information needed to set build a shader program and set API 34 * state for a draw. It is used along with a GrPrimitiveProcessor and a source of geometric 35 * data (GrMesh or GrPath) to draw. 36 */ 37 class GrPipeline { 38 public: 39 /////////////////////////////////////////////////////////////////////////// 40 /// @name Creation 41 42 // Pipeline options that the caller may enable. 43 // NOTE: This enum is extended later by GrPipeline::Flags. 44 enum class InputFlags : uint8_t { 45 kNone = 0, 46 /** 47 * Perform HW anti-aliasing. This means either HW FSAA, if supported by the render target, 48 * or smooth-line rendering if a line primitive is drawn and line smoothing is supported by 49 * the 3D API. 50 */ 51 kHWAntialias = (1 << 0), 52 /** 53 * Cause every pixel to be rasterized that is touched by the triangle anywhere (not just at 54 * pixel center). Additionally, if using MSAA, the sample mask will always have 100% 55 * coverage. 56 * NOTE: The primitive type must be a triangle type. 57 */ 58 kConservativeRaster = (1 << 1), 59 /** 60 * Draws triangles as outlines. 61 */ 62 kWireframe = (1 << 2), 63 /** 64 * Modifies the vertex shader so that vertices will be positioned at pixel centers. 65 */ 66 kSnapVerticesToPixelCenters = (1 << 3), // This value must be last. (See kLastInputFlag.) 67 }; 68 69 struct InitArgs { 70 InputFlags fInputFlags = InputFlags::kNone; 71 const GrUserStencilSettings* fUserStencil = &GrUserStencilSettings::kUnused; 72 const GrCaps* fCaps = nullptr; 73 GrXferProcessor::DstProxyView fDstProxyView; 74 GrSwizzle fOutputSwizzle; 75 }; 76 77 /** 78 * Some state can be changed between GrMeshes without changing GrPipelines. This is generally 79 * less expensive then using multiple pipelines. Such state is called "dynamic state". It can 80 * be specified in two ways: 81 * 1) FixedDynamicState - use this to specify state that does not vary between GrMeshes. 82 * 2) DynamicStateArrays - use this to specify per mesh values for dynamic state. 83 **/ 84 struct FixedDynamicState { FixedDynamicStateFixedDynamicState85 explicit FixedDynamicState(const SkIRect& scissorRect) : fScissorRect(scissorRect) {} 86 FixedDynamicState() = default; 87 SkIRect fScissorRect = SkIRect::MakeEmpty(); 88 // Must have GrPrimitiveProcessor::numTextureSamplers() entries. Can be null if no samplers 89 // or textures are passed using DynamicStateArrays. 90 GrSurfaceProxy** fPrimitiveProcessorTextures = nullptr; 91 }; 92 93 /** 94 * Any non-null array overrides the FixedDynamicState on a mesh-by-mesh basis. Arrays must 95 * have one entry for each GrMesh. 96 */ 97 struct DynamicStateArrays { 98 const SkIRect* fScissorRects = nullptr; 99 // Must have GrPrimitiveProcessor::numTextureSamplers() * num_meshes entries. 100 // Can be null if no samplers or to use the same textures for all meshes via' 101 // FixedDynamicState. 102 GrSurfaceProxy** fPrimitiveProcessorTextures = nullptr; 103 }; 104 105 /** 106 * Creates a simple pipeline with default settings and no processors. The provided blend mode 107 * must be "Porter Duff" (<= kLastCoeffMode). If using GrScissorTest::kEnabled, the caller must 108 * specify a scissor rectangle through the DynamicState struct. 109 **/ 110 GrPipeline(GrScissorTest scissor, SkBlendMode blend, const GrSwizzle& outputSwizzle, 111 InputFlags flags = InputFlags::kNone, 112 const GrUserStencilSettings* stencil = &GrUserStencilSettings::kUnused) GrPipeline(scissor,GrPorterDuffXPFactory::MakeNoCoverageXP (blend),outputSwizzle,flags,stencil)113 : GrPipeline(scissor, GrPorterDuffXPFactory::MakeNoCoverageXP(blend), outputSwizzle, 114 flags, stencil) { 115 } 116 117 GrPipeline(GrScissorTest, sk_sp<const GrXferProcessor>, const GrSwizzle& outputSwizzle, 118 InputFlags = InputFlags::kNone, 119 const GrUserStencilSettings* = &GrUserStencilSettings::kUnused); 120 121 GrPipeline(const InitArgs& args, sk_sp<const GrXferProcessor>, const GrAppliedHardClip&); 122 GrPipeline(const InitArgs&, GrProcessorSet&&, GrAppliedClip&&); 123 124 GrPipeline(const GrPipeline&) = delete; 125 GrPipeline& operator=(const GrPipeline&) = delete; 126 127 /// @} 128 129 /////////////////////////////////////////////////////////////////////////// 130 /// @name GrFragmentProcessors 131 numColorFragmentProcessors()132 int numColorFragmentProcessors() const { return fNumColorProcessors; } numCoverageFragmentProcessors()133 int numCoverageFragmentProcessors() const { 134 return fFragmentProcessors.count() - fNumColorProcessors; 135 } numFragmentProcessors()136 int numFragmentProcessors() const { return fFragmentProcessors.count(); } 137 getXferProcessor()138 const GrXferProcessor& getXferProcessor() const { 139 if (fXferProcessor) { 140 return *fXferProcessor.get(); 141 } else { 142 // A null xp member means the common src-over case. GrXferProcessor's ref'ing 143 // mechanism is not thread safe so we do not hold a ref on this global. 144 return GrPorterDuffXPFactory::SimpleSrcOverXP(); 145 } 146 } 147 148 /** 149 * This returns the GrSurfaceProxyView for the texture used to access the dst color. If the 150 * GrXferProcessor does not use the dst color then the proxy on the GrSurfaceProxyView will be 151 * nullptr. 152 */ dstProxyView()153 const GrSurfaceProxyView& dstProxyView() const { 154 return fDstProxyView; 155 } 156 157 /** 158 * If the GrXferProcessor uses a texture to access the dst color, then this returns that 159 * texture and the offset to the dst contents within that texture. 160 */ 161 GrTexture* peekDstTexture(SkIPoint* offset = nullptr) const { 162 if (offset) { 163 *offset = fDstTextureOffset; 164 } 165 166 if (GrTextureProxy* dstProxy = fDstProxyView.asTextureProxy()) { 167 return dstProxy->peekTexture(); 168 } 169 170 return nullptr; 171 } 172 getColorFragmentProcessor(int idx)173 const GrFragmentProcessor& getColorFragmentProcessor(int idx) const { 174 SkASSERT(idx < this->numColorFragmentProcessors()); 175 return *fFragmentProcessors[idx].get(); 176 } 177 getCoverageFragmentProcessor(int idx)178 const GrFragmentProcessor& getCoverageFragmentProcessor(int idx) const { 179 SkASSERT(idx < this->numCoverageFragmentProcessors()); 180 return *fFragmentProcessors[fNumColorProcessors + idx].get(); 181 } 182 getFragmentProcessor(int idx)183 const GrFragmentProcessor& getFragmentProcessor(int idx) const { 184 return *fFragmentProcessors[idx].get(); 185 } 186 187 /// @} 188 getUserStencil()189 const GrUserStencilSettings* getUserStencil() const { return fUserStencilSettings; } setUserStencil(const GrUserStencilSettings * stencil)190 void setUserStencil(const GrUserStencilSettings* stencil) { 191 fUserStencilSettings = stencil; 192 if (!fUserStencilSettings->isDisabled(fFlags & Flags::kHasStencilClip)) { 193 fFlags |= Flags::kStencilEnabled; 194 } 195 } 196 isScissorEnabled()197 bool isScissorEnabled() const { 198 return SkToBool(fFlags & Flags::kScissorEnabled); 199 } 200 getWindowRectsState()201 const GrWindowRectsState& getWindowRectsState() const { return fWindowRectsState; } 202 isHWAntialiasState()203 bool isHWAntialiasState() const { return fFlags & InputFlags::kHWAntialias; } usesConservativeRaster()204 bool usesConservativeRaster() const { return fFlags & InputFlags::kConservativeRaster; } isWireframe()205 bool isWireframe() const { return fFlags & InputFlags::kWireframe; } snapVerticesToPixelCenters()206 bool snapVerticesToPixelCenters() const { 207 return fFlags & InputFlags::kSnapVerticesToPixelCenters; 208 } hasStencilClip()209 bool hasStencilClip() const { 210 return SkToBool(fFlags & Flags::kHasStencilClip); 211 } isStencilEnabled()212 bool isStencilEnabled() const { 213 return SkToBool(fFlags & Flags::kStencilEnabled); 214 } 215 #ifdef SK_DEBUG allProxiesInstantiated()216 bool allProxiesInstantiated() const { 217 for (int i = 0; i < fFragmentProcessors.count(); ++i) { 218 if (!fFragmentProcessors[i]->isInstantiated()) { 219 return false; 220 } 221 } 222 if (fDstProxyView.proxy()) { 223 return fDstProxyView.proxy()->isInstantiated(); 224 } 225 226 return true; 227 } 228 #endif 229 230 GrXferBarrierType xferBarrierType(GrTexture*, const GrCaps&) const; 231 232 // Used by Vulkan and Metal to cache their respective pipeline objects 233 void genKey(GrProcessorKeyBuilder*, const GrCaps&) const; 234 outputSwizzle()235 const GrSwizzle& outputSwizzle() const { return fOutputSwizzle; } 236 237 void visitProxies(const GrOp::VisitProxyFunc&) const; 238 239 private: 240 static constexpr uint8_t kLastInputFlag = (uint8_t)InputFlags::kSnapVerticesToPixelCenters; 241 242 /** This is a continuation of the public "InputFlags" enum. */ 243 enum class Flags : uint8_t { 244 kHasStencilClip = (kLastInputFlag << 1), 245 kStencilEnabled = (kLastInputFlag << 2), 246 kScissorEnabled = (kLastInputFlag << 3), 247 }; 248 249 GR_DECL_BITFIELD_CLASS_OPS_FRIENDS(Flags); 250 251 friend bool operator&(Flags, InputFlags); 252 253 using FragmentProcessorArray = SkAutoSTArray<8, std::unique_ptr<const GrFragmentProcessor>>; 254 255 GrSurfaceProxyView fDstProxyView; 256 SkIPoint fDstTextureOffset; 257 GrWindowRectsState fWindowRectsState; 258 const GrUserStencilSettings* fUserStencilSettings; 259 Flags fFlags; 260 sk_sp<const GrXferProcessor> fXferProcessor; 261 FragmentProcessorArray fFragmentProcessors; 262 263 // This value is also the index in fFragmentProcessors where coverage processors begin. 264 int fNumColorProcessors = 0; 265 266 GrSwizzle fOutputSwizzle; 267 }; 268 269 GR_MAKE_BITFIELD_CLASS_OPS(GrPipeline::InputFlags); 270 GR_MAKE_BITFIELD_CLASS_OPS(GrPipeline::Flags); 271 272 inline bool operator&(GrPipeline::Flags flags, GrPipeline::InputFlags inputFlag) { 273 return (flags & (GrPipeline::Flags)inputFlag); 274 } 275 276 #endif 277