1 /*
2  * Copyright 2016 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 
9 #ifndef GrUserStencilSettings_DEFINED
10 #define GrUserStencilSettings_DEFINED
11 
12 #include "GrTypes.h"
13 
14 /**
15  * Gr uses the stencil buffer to implement complex clipping inside the
16  * GrOpList class. The GrOpList makes a subset of the stencil buffer
17  * bits available for other uses by external code (user bits). Client code can
18  * modify these bits. GrOpList will ignore ref, mask, and writemask bits
19  * provided by clients that fall outside the user range.
20  *
21  * When code outside the GrOpList class uses the stencil buffer the contract
22  * is as follows:
23  *
24  * > Normal stencil funcs allow the client to pass / fail regardless of the
25  *   reserved clip bits.
26  * > Additional functions allow a test against the clip along with a limited
27  *   set of tests against the user bits.
28  * > Client can assume all user bits are zero initially.
29  * > Client must ensure that after all its passes are finished it has only
30  *   written to the color buffer in the region inside the clip. Furthermore, it
31  *   must zero all user bits that were modifed (both inside and outside the
32  *   clip).
33  */
34 
35 enum GrStencilFlags {
36     kDisabled_StencilFlag         = (1 << 0),
37     kTestAlwaysPasses_StencilFlag = (1 << 1),
38     kNoModifyStencil_StencilFlag  = (1 << 2),
39     kNoWrapOps_StencilFlag        = (1 << 3),
40     kSingleSided_StencilFlag      = (1 << 4),
41 
42     kLast_StencilFlag = kSingleSided_StencilFlag,
43     kAll_StencilFlags = kLast_StencilFlag | (kLast_StencilFlag - 1)
44 };
45 
46 template<typename TTest, typename TOp> struct GrTStencilFaceSettings {
47     uint16_t   fRef;        // Reference value for stencil test and ops.
48     TTest      fTest;       // Stencil test function, where fRef is on the left side.
49     uint16_t   fTestMask;   // Bitwise "and" to perform on fRef and stencil values before testing.
50                             // (e.g. (fRef & fTestMask) < (stencil & fTestMask))
51     TOp        fPassOp;     // Op to perform when the test passes.
52     TOp        fFailOp;     // Op to perform when the test fails.
53     uint16_t   fWriteMask;  // Indicates which bits in the stencil buffer should be updated.
54                             // (e.g. stencil = (newValue & fWriteMask) | (stencil & ~fWriteMask))
55 };
56 
57 enum class GrUserStencilTest : uint16_t {
58     // Tests that respect the clip bit. If a stencil clip is not in effect, the "IfInClip" is
59     // ignored and these only act on user bits.
60     kAlwaysIfInClip,
61     kEqualIfInClip,
62     kLessIfInClip,
63     kLEqualIfInClip,
64 
65     // Tests that ignore the clip bit. The client is responsible to ensure no color write occurs
66     // outside the clip if it is in use.
67     kAlways,
68     kNever,
69     kGreater,
70     kGEqual,
71     kLess,
72     kLEqual,
73     kEqual,
74     kNotEqual
75 };
76 constexpr static GrUserStencilTest kLastClippedStencilTest = GrUserStencilTest::kLEqualIfInClip;
77 constexpr static int kGrUserStencilTestCount = 1 + (int)GrUserStencilTest::kNotEqual;
78 
79 enum class GrUserStencilOp : uint8_t {
80     kKeep,
81 
82     // Ops that only modify user bits. These must not be paired with ops that modify the clip bit.
83     kZero,
84     kReplace, // Replace stencil value with fRef (only the bits enabled in fWriteMask).
85     kInvert,
86     kIncWrap,
87     kDecWrap,
88     // These two should only be used if wrap ops are not supported, or if the math is guaranteed
89     // to not overflow. The user bits may or may not clamp, depending on the state of non-user bits.
90     kIncMaybeClamp,
91     kDecMaybeClamp,
92 
93     // Ops that only modify the clip bit. These must not be paired with ops that modify user bits.
94     kZeroClipBit,
95     kSetClipBit,
96     kInvertClipBit,
97 
98     // Ops that modify both clip and user bits. These can only be paired with kKeep or each other.
99     kSetClipAndReplaceUserBits,
100     kZeroClipAndUserBits
101 };
102 constexpr static GrUserStencilOp kLastUserOnlyStencilOp = GrUserStencilOp::kDecMaybeClamp;
103 constexpr static GrUserStencilOp kLastClipOnlyStencilOp = GrUserStencilOp::kInvertClipBit;
104 constexpr static int kGrUserStencilOpCount = 1 + (int)GrUserStencilOp::kZeroClipAndUserBits;
105 
106 /**
107  * This struct is a compile-time constant representation of user stencil settings. It describes in
108  * abstract terms how a draw will use the stencil buffer. It gets ODR-used at runtime to define a
109  * draw's stencil settings, and is later translated into concrete settings when the pipeline is
110  * finalized.
111  */
112 struct GrUserStencilSettings {
113     typedef GrTStencilFaceSettings<GrUserStencilTest, GrUserStencilOp> Face;
114 
115     template<GrUserStencilTest, GrUserStencilOp PassOp, GrUserStencilOp FailOp> struct Attrs;
116 
117     // Unfortunately, this is the only way to pass template arguments to a constructor.
118     template<uint16_t Ref, GrUserStencilTest Test, uint16_t TestMask,
119              GrUserStencilOp PassOp, GrUserStencilOp FailOp, uint16_t WriteMask> struct Init {};
120 
121     template<uint16_t FtRef,            uint16_t BkRef,
122              GrUserStencilTest FtTest,  GrUserStencilTest BkTest,
123              uint16_t FtTestMask,       uint16_t BkTestMask,
124              GrUserStencilOp FtPassOp,  GrUserStencilOp BkPassOp,
125              GrUserStencilOp FtFailOp,  GrUserStencilOp BkFailOp,
126              uint16_t FtWriteMask,      uint16_t BkWriteMask> struct InitSeparate {};
127 
128     template<uint16_t Ref, GrUserStencilTest Test, uint16_t TestMask,
129              GrUserStencilOp PassOp, GrUserStencilOp FailOp, uint16_t WriteMask>
StaticInitGrUserStencilSettings130     constexpr static Init<Ref, Test, TestMask, PassOp, FailOp, WriteMask> StaticInit() {
131         return Init<Ref, Test, TestMask, PassOp, FailOp, WriteMask>();
132     }
133 
134     template<uint16_t FtRef,            uint16_t BkRef,
135              GrUserStencilTest FtTest,  GrUserStencilTest BkTest,
136              uint16_t FtTestMask,       uint16_t BkTestMask,
137              GrUserStencilOp FtPassOp,  GrUserStencilOp BkPassOp,
138              GrUserStencilOp FtFailOp,  GrUserStencilOp BkFailOp,
139              uint16_t FtWriteMask,      uint16_t BkWriteMask>
140     constexpr static InitSeparate<FtRef, BkRef, FtTest, BkTest, FtTestMask, BkTestMask,
141                                   FtPassOp, BkPassOp, FtFailOp, BkFailOp, FtWriteMask,
StaticInitSeparateGrUserStencilSettings142                                   BkWriteMask> StaticInitSeparate() {
143         return InitSeparate<FtRef, BkRef, FtTest, BkTest, FtTestMask, BkTestMask,
144                             FtPassOp, BkPassOp, FtFailOp, BkFailOp, FtWriteMask, BkWriteMask>();
145     }
146 
147     // We construct with template arguments in order to enforce that the struct be compile-time
148     // constant and to make use of static asserts.
149     template<uint16_t Ref, GrUserStencilTest Test, uint16_t TestMask,
150              GrUserStencilOp PassOp, GrUserStencilOp FailOp, uint16_t WriteMask,
151              typename Attrs = Attrs<Test, PassOp, FailOp> >
GrUserStencilSettingsGrUserStencilSettings152     constexpr explicit GrUserStencilSettings(
153             const Init<Ref, Test, TestMask, PassOp, FailOp, WriteMask>&)
154         : fFrontFlags{(uint16_t)(Attrs::Flags(false) | kSingleSided_StencilFlag),
155                       (uint16_t)(Attrs::Flags(true) | kSingleSided_StencilFlag)}
156         , fFront{Ref, Test, Attrs::EffectiveTestMask(TestMask), PassOp, FailOp,
157                  Attrs::EffectiveWriteMask(WriteMask)}
158         , fBackFlags{(uint16_t)(Attrs::Flags(false) | kSingleSided_StencilFlag),
159                      (uint16_t)(Attrs::Flags(true) | kSingleSided_StencilFlag)}
160         , fBack{Ref, Test, Attrs::EffectiveTestMask(TestMask), PassOp, FailOp,
161                 Attrs::EffectiveWriteMask(WriteMask)} {
162     }
163 
164     template<uint16_t FtRef,            uint16_t BkRef,
165              GrUserStencilTest FtTest,  GrUserStencilTest BkTest,
166              uint16_t FtTestMask,       uint16_t BkTestMask,
167              GrUserStencilOp FtPassOp,  GrUserStencilOp BkPassOp,
168              GrUserStencilOp FtFailOp,  GrUserStencilOp BkFailOp,
169              uint16_t FtWriteMask,      uint16_t BkWriteMask,
170              typename FtAttrs = Attrs<FtTest, FtPassOp, FtFailOp>,
171              typename BkAttrs = Attrs<BkTest, BkPassOp, BkFailOp> >
GrUserStencilSettingsGrUserStencilSettings172     constexpr explicit GrUserStencilSettings(
173             const InitSeparate<FtRef, BkRef, FtTest, BkTest, FtTestMask, BkTestMask,
174                                FtPassOp, BkPassOp, FtFailOp, BkFailOp, FtWriteMask, BkWriteMask>&)
175         : fFrontFlags{FtAttrs::Flags(false), FtAttrs::Flags(true)}
176         , fFront{FtRef, FtTest, FtAttrs::EffectiveTestMask(FtTestMask), FtPassOp, FtFailOp,
177                  FtAttrs::EffectiveWriteMask(FtWriteMask)}
178         , fBackFlags{BkAttrs::Flags(false), BkAttrs::Flags(true)}
179         , fBack{BkRef, BkTest, BkAttrs::EffectiveTestMask(BkTestMask), BkPassOp, BkFailOp,
180                 BkAttrs::EffectiveWriteMask(BkWriteMask)} {}
181 
182     // This struct can only be constructed with static initializers.
183     GrUserStencilSettings() = delete;
184     GrUserStencilSettings(const GrUserStencilSettings&) = delete;
185 
flagsGrUserStencilSettings186     uint16_t flags(bool hasStencilClip) const {
187         return fFrontFlags[hasStencilClip] & fBackFlags[hasStencilClip];
188     }
isDisabledGrUserStencilSettings189     bool isDisabled(bool hasStencilClip) const {
190         return this->flags(hasStencilClip) & kDisabled_StencilFlag;
191     }
testAlwaysPassesGrUserStencilSettings192     bool testAlwaysPasses(bool hasStencilClip) const {
193         return this->flags(hasStencilClip) & kTestAlwaysPasses_StencilFlag;
194     }
isTwoSidedGrUserStencilSettings195     bool isTwoSided(bool hasStencilClip) const {
196         return !(this->flags(hasStencilClip) & kSingleSided_StencilFlag);
197     }
usesWrapOpGrUserStencilSettings198     bool usesWrapOp(bool hasStencilClip) const {
199         return !(this->flags(hasStencilClip) & kNoWrapOps_StencilFlag);
200     }
201 
202     const uint16_t   fFrontFlags[2]; // frontFlagsForDraw = fFrontFlags[hasStencilClip].
203     const Face       fFront;
204     const uint16_t   fBackFlags[2]; // backFlagsForDraw = fBackFlags[hasStencilClip].
205     const Face       fBack;
206 
207     static const GrUserStencilSettings& kUnused;
208 
isUnusedGrUserStencilSettings209     bool isUnused() const { return this == &kUnused; }
210 };
211 
212 template<GrUserStencilTest Test, GrUserStencilOp PassOp, GrUserStencilOp FailOp>
213 struct GrUserStencilSettings::Attrs {
214     // Ensure an op that only modifies user bits isn't paired with one that modifies clip bits.
215     GR_STATIC_ASSERT(GrUserStencilOp::kKeep == PassOp || GrUserStencilOp::kKeep == FailOp ||
216                      (PassOp <= kLastUserOnlyStencilOp) == (FailOp <= kLastUserOnlyStencilOp));
217     // Ensure an op that only modifies clip bits isn't paired with one that modifies clip and user.
218     GR_STATIC_ASSERT(GrUserStencilOp::kKeep == PassOp || GrUserStencilOp::kKeep == FailOp ||
219                      (PassOp <= kLastClipOnlyStencilOp) == (FailOp <= kLastClipOnlyStencilOp));
220 
TestAlwaysPassesAttrs221     constexpr static bool TestAlwaysPasses(bool hasStencilClip) {
222         return (!hasStencilClip && GrUserStencilTest::kAlwaysIfInClip == Test) ||
223                 GrUserStencilTest::kAlways == Test;
224     }
DoesNotModifyStencilAttrs225     constexpr static bool DoesNotModifyStencil(bool hasStencilClip) {
226         return (GrUserStencilTest::kNever == Test || GrUserStencilOp::kKeep == PassOp) &&
227                 (TestAlwaysPasses(hasStencilClip) || GrUserStencilOp::kKeep == FailOp);
228     }
IsDisabledAttrs229     constexpr static bool IsDisabled(bool hasStencilClip) {
230         return TestAlwaysPasses(hasStencilClip) && DoesNotModifyStencil(hasStencilClip);
231     }
UsesWrapOpsAttrs232     constexpr static bool UsesWrapOps() {
233         return GrUserStencilOp::kIncWrap == PassOp || GrUserStencilOp::kDecWrap == PassOp ||
234                GrUserStencilOp::kIncWrap == FailOp || GrUserStencilOp::kDecWrap == FailOp;
235     }
TestIgnoresRefAttrs236     constexpr static bool TestIgnoresRef() {
237         return (GrUserStencilTest::kAlwaysIfInClip == Test || GrUserStencilTest::kAlways == Test ||
238                 GrUserStencilTest::kNever == Test);
239     }
FlagsAttrs240     constexpr static uint16_t Flags(bool hasStencilClip) {
241         return (IsDisabled(hasStencilClip) ? kDisabled_StencilFlag : 0) |
242                (TestAlwaysPasses(hasStencilClip) ? kTestAlwaysPasses_StencilFlag : 0) |
243                (DoesNotModifyStencil(hasStencilClip) ? kNoModifyStencil_StencilFlag : 0) |
244                (UsesWrapOps() ? 0 : kNoWrapOps_StencilFlag);
245     }
EffectiveTestMaskAttrs246     constexpr static uint16_t EffectiveTestMask(uint16_t testMask) {
247         return TestIgnoresRef() ? 0 : testMask;
248     }
EffectiveWriteMaskAttrs249     constexpr static uint16_t EffectiveWriteMask(uint16_t writeMask) {
250         // We don't modify the mask differently when hasStencilClip=false because either the entire
251         // face gets disabled in that case (e.g. Test=kAlwaysIfInClip, PassOp=kKeep), or else the
252         // effective mask stays the same either way.
253         return DoesNotModifyStencil(true) ? 0 : writeMask;
254     }
255 };
256 
257 #endif
258