• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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