• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright 2020 Google LLC
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/v1/StencilMaskHelper.h"
9 
10 #include "include/core/SkMatrix.h"
11 #include "include/core/SkPath.h"
12 #include "src/gpu/GrRecordingContextPriv.h"
13 #include "src/gpu/GrStencilSettings.h"
14 #include "src/gpu/effects/GrDisableColorXP.h"
15 #include "src/gpu/geometry/GrShape.h"
16 #include "src/gpu/geometry/GrStyledShape.h"
17 #include "src/gpu/v1/SurfaceDrawContext_v1.h"
18 
19 namespace {
20 
21 ////////////////////////////////////////////////////////////////////////////////
22 // Stencil Rules for Merging user stencil space into clip
23 //
24 
25 ///////
26 // Replace
27 static constexpr GrUserStencilSettings gUserToClipReplace(
28     GrUserStencilSettings::StaticInit<
29         0x0000,
30         GrUserStencilTest::kNotEqual,
31         0xffff,
32         GrUserStencilOp::kSetClipAndReplaceUserBits,
33         GrUserStencilOp::kZeroClipAndUserBits,
34         0xffff>()
35 );
36 
37 static constexpr GrUserStencilSettings gInvUserToClipReplace(
38     GrUserStencilSettings::StaticInit<
39         0x0000,
40         GrUserStencilTest::kEqual,
41         0xffff,
42         GrUserStencilOp::kSetClipAndReplaceUserBits,
43         GrUserStencilOp::kZeroClipAndUserBits,
44         0xffff>()
45 );
46 
47 ///////
48 // Intersect
49 static constexpr GrUserStencilSettings gUserToClipIsect(
50     GrUserStencilSettings::StaticInit<
51         0x0000,
52         GrUserStencilTest::kLessIfInClip, // "0 < userBits" is equivalent to "0 != userBits".
53         0xffff,
54         GrUserStencilOp::kSetClipAndReplaceUserBits,
55         GrUserStencilOp::kZeroClipAndUserBits,
56         0xffff>()
57 );
58 
59 ///////
60 // Difference
61 static constexpr GrUserStencilSettings gUserToClipDiff(
62     GrUserStencilSettings::StaticInit<
63         0x0000,
64         GrUserStencilTest::kEqualIfInClip,
65         0xffff,
66         GrUserStencilOp::kSetClipAndReplaceUserBits,
67         GrUserStencilOp::kZeroClipAndUserBits,
68         0xffff>()
69 );
70 
71 ///////
72 // Union
73 static constexpr GrUserStencilSettings gUserToClipUnion(
74     GrUserStencilSettings::StaticInit<
75         0x0000,
76         GrUserStencilTest::kNotEqual,
77         0xffff,
78         GrUserStencilOp::kSetClipAndReplaceUserBits,
79         GrUserStencilOp::kKeep,
80         0xffff>()
81 );
82 
83 static constexpr GrUserStencilSettings gInvUserToClipUnionPass0( // Does not zero user bits.
84     GrUserStencilSettings::StaticInit<
85         0x0000,
86         GrUserStencilTest::kEqual,
87         0xffff,
88         GrUserStencilOp::kSetClipBit,
89         GrUserStencilOp::kKeep,
90         0x0000>()
91 );
92 
93 ///////
94 // Xor
95 static constexpr GrUserStencilSettings gUserToClipXorPass0( // Does not zero user bits.
96     GrUserStencilSettings::StaticInit<
97         0x0000,
98         GrUserStencilTest::kNotEqual,
99         0xffff,
100         GrUserStencilOp::kInvertClipBit,
101         GrUserStencilOp::kKeep,
102         0x0000>()
103 );
104 
105 static constexpr GrUserStencilSettings gInvUserToClipXorPass0( // Does not zero user bits.
106     GrUserStencilSettings::StaticInit<
107         0x0000,
108         GrUserStencilTest::kEqual,
109         0xffff,
110         GrUserStencilOp::kInvertClipBit,
111         GrUserStencilOp::kKeep,
112         0x0000>()
113 );
114 
115 ///////
116 // Reverse Diff
117 static constexpr GrUserStencilSettings gUserToClipRDiffPass0( // Does not zero user bits.
118     GrUserStencilSettings::StaticInit<
119         0x0000,
120         GrUserStencilTest::kNotEqual,
121         0xffff,
122         GrUserStencilOp::kInvertClipBit,
123         GrUserStencilOp::kZeroClipBit,
124         0x0000>()
125 );
126 
127 static constexpr GrUserStencilSettings gInvUserToClipRDiffPass0( // Does not zero user bits.
128     GrUserStencilSettings::StaticInit<
129         0x0000,
130         GrUserStencilTest::kEqual,
131         0xffff,
132         GrUserStencilOp::kInvertClipBit,
133         GrUserStencilOp::kZeroClipBit,
134         0x0000>()
135 );
136 
137 ///////
138 // Second pass to clear user bits (only needed sometimes)
139 static constexpr GrUserStencilSettings gZeroUserBits(
140     GrUserStencilSettings::StaticInit<
141         0x0000,
142         GrUserStencilTest::kNotEqual,
143         0xffff,
144         GrUserStencilOp::kZero,
145         GrUserStencilOp::kKeep,
146         0xffff>()
147 );
148 
149 static constexpr const GrUserStencilSettings* gUserToClipTable[2][1 + SkRegion::kLastOp][3] = {
150     {  /* Normal fill. */
151         {&gUserToClipDiff,           nullptr,         nullptr},  // kDifference_Op.
152         {&gUserToClipIsect,          nullptr,         nullptr},  // kIntersect_Op.
153         {&gUserToClipUnion,          nullptr,         nullptr},  // kUnion_Op.
154         {&gUserToClipXorPass0,       &gZeroUserBits,  nullptr},  // kXOR_Op.
155         {&gUserToClipRDiffPass0,     &gZeroUserBits,  nullptr},  // kReverseDifference_Op.
156         {&gUserToClipReplace,        nullptr,         nullptr}   // kReplace_Op.
157 
158     }, /* Inverse fill. */ {
159         {&gUserToClipIsect,          nullptr,         nullptr},  // ~diff (aka isect).
160         {&gUserToClipDiff,           nullptr,         nullptr},  // ~isect (aka diff).
161         {&gInvUserToClipUnionPass0,  &gZeroUserBits,  nullptr},  // ~union.
162         {&gInvUserToClipXorPass0,    &gZeroUserBits,  nullptr},  // ~xor.
163         {&gInvUserToClipRDiffPass0,  &gZeroUserBits,  nullptr},  // ~reverse diff.
164         {&gInvUserToClipReplace,     nullptr,         nullptr}   // ~replace.
165     }
166 };
167 
168 ///////
169 // Direct to Stencil
170 
171 // We can render a clip element directly without first writing to the client
172 // portion of the clip when the fill is not inverse and the set operation will
173 // only modify the in/out status of samples covered by the clip element.
174 
175 // this one only works if used right after stencil clip was cleared.
176 // Our clip mask creation code doesn't allow midstream replace ops.
177 static constexpr GrUserStencilSettings gReplaceClip(
178     GrUserStencilSettings::StaticInit<
179         0x0000,
180         GrUserStencilTest::kAlways,
181         0xffff,
182         GrUserStencilOp::kSetClipBit,
183         GrUserStencilOp::kSetClipBit,
184         0x0000>()
185 );
186 
187 static constexpr GrUserStencilSettings gUnionClip(
188     GrUserStencilSettings::StaticInit<
189         0x0000,
190         GrUserStencilTest::kAlwaysIfInClip,
191         0xffff,
192         GrUserStencilOp::kKeep,
193         GrUserStencilOp::kSetClipBit,
194         0x0000>()
195 );
196 
197 static constexpr GrUserStencilSettings gXorClip(
198     GrUserStencilSettings::StaticInit<
199         0x0000,
200         GrUserStencilTest::kAlways,
201         0xffff,
202         GrUserStencilOp::kInvertClipBit,
203         GrUserStencilOp::kInvertClipBit,
204         0x0000>()
205 );
206 
207 static constexpr GrUserStencilSettings gDiffClip(
208     GrUserStencilSettings::StaticInit<
209         0x0000,
210         GrUserStencilTest::kAlwaysIfInClip,
211         0xffff,
212         GrUserStencilOp::kZeroClipBit,
213         GrUserStencilOp::kKeep,
214         0x0000>()
215 );
216 
217 static constexpr const GrUserStencilSettings* gDirectDrawTable[1 + SkRegion::kLastOp][2] = {
218     {&gDiffClip,     nullptr},  // kDifference_Op.
219     {nullptr,        nullptr},  // kIntersect_Op.
220     {&gUnionClip,    nullptr},  // kUnion_Op.
221     {&gXorClip,      nullptr},  // kXOR_Op.
222     {nullptr,        nullptr},  // kReverseDifference_Op.
223     {&gReplaceClip,  nullptr}   // kReplace_Op.
224 };
225 
226 static_assert(0 == SkRegion::kDifference_Op);
227 static_assert(1 == SkRegion::kIntersect_Op);
228 static_assert(2 == SkRegion::kUnion_Op);
229 static_assert(3 == SkRegion::kXOR_Op);
230 static_assert(4 == SkRegion::kReverseDifference_Op);
231 static_assert(5 == SkRegion::kReplace_Op);
232 
233 // Settings used to when not allowed to draw directly to the clip to fill the user stencil bits
234 // before applying the covering clip stencil passes.
235 static constexpr GrUserStencilSettings gDrawToStencil(
236     GrUserStencilSettings::StaticInit<
237         0x0000,
238         GrUserStencilTest::kAlways,
239         0xffff,
240         GrUserStencilOp::kIncMaybeClamp,
241         GrUserStencilOp::kIncMaybeClamp,
242         0xffff>()
243 );
244 
245 // Get the stencil settings per-pass to achieve the given fill+region op effect on the
246 // stencil buffer.
247 //
248 // If drawDirectToClip comes back false, the caller must first draw the element into the user
249 // stencil bits, and then cover the clip area with multiple passes using the returned
250 // stencil settings.
251 
252 // If drawDirectToClip is true, the returned array will only have one pass and the
253 // caller should use those stencil settings while drawing the element directly.
254 //
255 // This returns a null-terminated list of const GrUserStencilSettings*
get_stencil_passes(SkRegion::Op op,skgpu::v1::PathRenderer::StencilSupport stencilSupport,bool fillInverted,bool * drawDirectToClip)256 GrUserStencilSettings const* const* get_stencil_passes(
257         SkRegion::Op op,
258         skgpu::v1::PathRenderer::StencilSupport stencilSupport,
259         bool fillInverted,
260         bool* drawDirectToClip) {
261     bool canRenderDirectToStencil =
262             skgpu::v1::PathRenderer::kNoRestriction_StencilSupport == stencilSupport;
263 
264     // TODO: inverse fill + intersect op can be direct.
265     // TODO: this can be greatly simplified when we only need intersect and difference ops and
266     //       none of the paths will be inverse-filled (just toggle the op instead).
267     SkASSERT((unsigned)op <= SkRegion::kLastOp);
268     if (canRenderDirectToStencil && !fillInverted) {
269         GrUserStencilSettings const* const* directPass = gDirectDrawTable[op];
270         if (directPass[0]) {
271             *drawDirectToClip = true;
272             return directPass;
273         }
274     }
275     *drawDirectToClip = false;
276     return gUserToClipTable[fillInverted][op];
277 }
278 
draw_stencil_rect(skgpu::v1::SurfaceDrawContext * sdc,const GrHardClip & clip,const GrUserStencilSettings * ss,const SkMatrix & matrix,const SkRect & rect,GrAA aa)279 void draw_stencil_rect(skgpu::v1::SurfaceDrawContext* sdc,
280                        const GrHardClip& clip,
281                        const GrUserStencilSettings* ss,
282                        const SkMatrix& matrix,
283                        const SkRect& rect, GrAA aa) {
284     GrPaint paint;
285     paint.setXPFactory(GrDisableColorXPFactory::Get());
286     sdc->stencilRect(&clip, ss, std::move(paint), aa, matrix, rect);
287 }
288 
draw_path(GrRecordingContext * rContext,skgpu::v1::SurfaceDrawContext * sdc,skgpu::v1::PathRenderer * pr,const GrHardClip & clip,const SkIRect & bounds,const GrUserStencilSettings * ss,const SkMatrix & matrix,const GrStyledShape & shape,GrAA aa)289 void draw_path(GrRecordingContext* rContext,
290                skgpu::v1::SurfaceDrawContext* sdc,
291                skgpu::v1::PathRenderer* pr,
292                const GrHardClip& clip,
293                const SkIRect& bounds,
294                const GrUserStencilSettings* ss,
295                const SkMatrix& matrix,
296                const GrStyledShape& shape,
297                GrAA aa) {
298     GrPaint paint;
299     paint.setXPFactory(GrDisableColorXPFactory::Get());
300 
301     // kMSAA is the only type of AA that's possible on a stencil buffer.
302     GrAAType pathAAType = aa == GrAA::kYes ? GrAAType::kMSAA : GrAAType::kNone;
303 
304     skgpu::v1::PathRenderer::DrawPathArgs args{rContext,
305                                                std::move(paint),
306                                                ss,
307                                                sdc,
308                                                &clip,
309                                                &bounds,
310                                                &matrix,
311                                                &shape,
312                                                pathAAType,
313                                                false};
314     pr->drawPath(args);
315 }
316 
stencil_path(GrRecordingContext * rContext,skgpu::v1::SurfaceDrawContext * sdc,skgpu::v1::PathRenderer * pr,const GrFixedClip & clip,const SkMatrix & matrix,const GrStyledShape & shape,GrAA aa)317 void stencil_path(GrRecordingContext* rContext,
318                   skgpu::v1::SurfaceDrawContext* sdc,
319                   skgpu::v1::PathRenderer* pr,
320                   const GrFixedClip& clip,
321                   const SkMatrix& matrix,
322                   const GrStyledShape& shape,
323                   GrAA aa) {
324     skgpu::v1::PathRenderer::StencilPathArgs args;
325     args.fContext = rContext;
326     args.fSurfaceDrawContext = sdc;
327     args.fClip = &clip;
328     args.fClipConservativeBounds = &clip.scissorRect();
329     args.fViewMatrix = &matrix;
330     args.fShape = &shape;
331     args.fDoStencilMSAA = aa;
332 
333     pr->stencilPath(args);
334 }
335 
supported_aa(skgpu::v1::SurfaceDrawContext * sdc,GrAA aa)336 GrAA supported_aa(skgpu::v1::SurfaceDrawContext* sdc, GrAA aa) {
337     return GrAA(sdc->numSamples() > 1 || sdc->canUseDynamicMSAA());
338 }
339 
340 }  // namespace
341 
342 namespace skgpu::v1 {
343 
StencilMaskHelper(GrRecordingContext * rContext,SurfaceDrawContext * sdc)344 StencilMaskHelper::StencilMaskHelper(GrRecordingContext* rContext,
345                                      SurfaceDrawContext* sdc)
346         : fContext(rContext)
347         , fSDC(sdc)
348         , fClip(sdc->dimensions()) {
349 }
350 
init(const SkIRect & bounds,uint32_t genID,const GrWindowRectangles & windowRects,int numFPs)351 bool StencilMaskHelper::init(const SkIRect& bounds, uint32_t genID,
352                              const GrWindowRectangles& windowRects, int numFPs) {
353     if (!fSDC->mustRenderClip(genID, bounds, numFPs)) {
354         return false;
355     }
356 
357     fClip.setStencilClip(genID);
358     // Should have caught bounds not intersecting the render target much earlier in clip application
359     SkAssertResult(fClip.fixedClip().setScissor(bounds));
360     if (!windowRects.empty()) {
361         fClip.fixedClip().setWindowRectangles(
362                 windowRects, GrWindowRectsState::Mode::kExclusive);
363     }
364     fNumFPs = numFPs;
365     return true;
366 }
367 
drawRect(const SkRect & rect,const SkMatrix & matrix,SkRegion::Op op,GrAA aa)368 void StencilMaskHelper::drawRect(const SkRect& rect,
369                                  const SkMatrix& matrix,
370                                  SkRegion::Op op,
371                                  GrAA aa) {
372     if (rect.isEmpty()) {
373         return;
374     }
375 
376     bool drawDirectToClip;
377     auto passes = get_stencil_passes(op, PathRenderer::kNoRestriction_StencilSupport,
378                                      false, &drawDirectToClip);
379     aa = supported_aa(fSDC, aa);
380 
381     if (!drawDirectToClip) {
382         // Draw to client stencil bits first
383         draw_stencil_rect(fSDC, fClip.fixedClip(), &gDrawToStencil, matrix, rect, aa);
384     }
385 
386     // Now modify the clip bit (either by rendering directly), or by covering the bounding box
387     // of the clip
388     for (GrUserStencilSettings const* const* pass = passes; *pass; ++pass) {
389         if (drawDirectToClip) {
390             draw_stencil_rect(fSDC, fClip, *pass, matrix, rect, aa);
391         } else {
392             draw_stencil_rect(fSDC, fClip, *pass, SkMatrix::I(),
393                               SkRect::Make(fClip.fixedClip().scissorRect()), aa);
394         }
395     }
396 }
397 
drawPath(const SkPath & path,const SkMatrix & matrix,SkRegion::Op op,GrAA aa)398 bool StencilMaskHelper::drawPath(const SkPath& path,
399                                  const SkMatrix& matrix,
400                                  SkRegion::Op op,
401                                  GrAA aa) {
402     if (path.isEmpty()) {
403         return true;
404     }
405 
406     // drawPath follows a similar approach to drawRect(), where we either draw directly to the clip
407     // bit or first draw to client bits and then apply a cover pass. The complicating factor is that
408     // we rely on path rendering and how the chosen path renderer uses the stencil buffer.
409     aa = supported_aa(fSDC, aa);
410 
411     GrAAType pathAAType = aa == GrAA::kYes ? GrAAType::kMSAA : GrAAType::kNone;
412 
413     // This will be used to determine whether the clip shape can be rendered into the
414     // stencil with arbitrary stencil settings.
415     PathRenderer::StencilSupport stencilSupport;
416 
417     // Make path canonical with regards to fill type (inverse handled by stencil settings).
418     bool fillInverted = path.isInverseFillType();
419     SkTCopyOnFirstWrite<SkPath> clipPath(path);
420     if (fillInverted) {
421         clipPath.writable()->toggleInverseFillType();
422     }
423 
424     GrStyledShape shape(*clipPath, GrStyle::SimpleFill());
425     SkASSERT(!shape.inverseFilled());
426 
427     PathRenderer::CanDrawPathArgs canDrawArgs;
428     canDrawArgs.fCaps = fContext->priv().caps();
429     canDrawArgs.fProxy = fSDC->asRenderTargetProxy();
430     canDrawArgs.fClipConservativeBounds = &fClip.fixedClip().scissorRect();
431     canDrawArgs.fViewMatrix = &matrix;
432     canDrawArgs.fShape = &shape;
433     canDrawArgs.fPaint = nullptr;
434     canDrawArgs.fSurfaceProps = &fSDC->surfaceProps();
435     canDrawArgs.fAAType = pathAAType;
436     canDrawArgs.fHasUserStencilSettings = false;
437 
438     auto pr =  fContext->priv().drawingManager()->getPathRenderer(
439             canDrawArgs, false, PathRendererChain::DrawType::kStencil, &stencilSupport);
440     if (!pr) {
441         return false;
442     }
443 
444     bool drawDirectToClip;
445     auto passes = get_stencil_passes(op, stencilSupport, fillInverted, &drawDirectToClip);
446 
447     // Write to client bits if necessary
448     if (!drawDirectToClip) {
449         if (stencilSupport == PathRenderer::kNoRestriction_StencilSupport) {
450             draw_path(fContext, fSDC, pr, fClip.fixedClip(), fClip.fixedClip().scissorRect(),
451                       &gDrawToStencil, matrix, shape, aa);
452         } else {
453             stencil_path(fContext, fSDC, pr, fClip.fixedClip(), matrix, shape, aa);
454         }
455     }
456 
457     // Now modify the clip bit (either by rendering directly), or by covering the bounding box
458     // of the clip
459     for (GrUserStencilSettings const* const* pass = passes; *pass; ++pass) {
460         if (drawDirectToClip) {
461             draw_path(fContext, fSDC, pr, fClip, fClip.fixedClip().scissorRect(),
462                       *pass, matrix, shape, aa);
463         } else {
464             draw_stencil_rect(fSDC, fClip, *pass, SkMatrix::I(),
465                               SkRect::Make(fClip.fixedClip().scissorRect()), aa);
466         }
467     }
468 
469     return true;
470 }
471 
drawShape(const GrShape & shape,const SkMatrix & matrix,SkRegion::Op op,GrAA aa)472 bool StencilMaskHelper::drawShape(const GrShape& shape,
473                                   const SkMatrix& matrix,
474                                   SkRegion::Op op,
475                                   GrAA aa) {
476     if (shape.isRect() && !shape.inverted()) {
477         this->drawRect(shape.rect(), matrix, op, aa);
478         return true;
479     } else {
480         SkPath p;
481         shape.asPath(&p);
482         return this->drawPath(p, matrix, op, aa);
483     }
484 }
485 
clear(bool insideStencil)486 void StencilMaskHelper::clear(bool insideStencil) {
487     if (fClip.fixedClip().hasWindowRectangles()) {
488         // Use a draw to benefit from window rectangles when resetting the stencil buffer; for
489         // large buffers with MSAA this can be significant.
490         draw_stencil_rect(fSDC, fClip.fixedClip(),
491                           GrStencilSettings::SetClipBitSettings(insideStencil), SkMatrix::I(),
492                           SkRect::Make(fClip.fixedClip().scissorRect()), GrAA::kNo);
493     } else {
494         fSDC->clearStencilClip(fClip.fixedClip().scissorRect(), insideStencil);
495     }
496 }
497 
finish()498 void StencilMaskHelper::finish() {
499     fSDC->setLastClip(fClip.stencilStackID(), fClip.fixedClip().scissorRect(), fNumFPs);
500 }
501 
502 } // namespace skgpu::v1
503