Detect empty clip when difference op clips everything

Change-Id: Ifbe0f3ae6e01d65f18351903da8aef63170ce6c7
Reviewed-on: https://skia-review.googlesource.com/c/skia/+/294457
Commit-Queue: Michael Ludwig <michaelludwig@google.com>
Auto-Submit: Michael Ludwig <michaelludwig@google.com>
Reviewed-by: Brian Salomon <bsalomon@google.com>
This commit is contained in:
Michael Ludwig 2020-06-05 10:56:32 -04:00 committed by Skia Commit-Bot
parent b96995d05f
commit de00dc9004
2 changed files with 23 additions and 9 deletions

View File

@ -115,6 +115,13 @@ GrReducedClip::GrReducedClip(const SkClipStack& stack, const SkRect& queryBounds
// Now that we have determined the bounds to use and filtered out the trivial cases, call // Now that we have determined the bounds to use and filtered out the trivial cases, call
// the helper that actually walks the stack. // the helper that actually walks the stack.
this->walkStack(stack, tighterQuery); this->walkStack(stack, tighterQuery);
if (fInitialState == InitialState::kAllOut && fMaskElements.isEmpty()) {
// The clip starts with no coverage and there are no elements to add coverage with
// expanding ops. We ignore the AAClipRectGenID since it is an implied intersection.
this->makeEmpty();
return;
}
} }
if (SK_InvalidGenID != fAAClipRectGenID && // Is there an AA clip rect? if (SK_InvalidGenID != fAAClipRectGenID && // Is there an AA clip rect?

View File

@ -924,6 +924,13 @@ static void add_oval(const SkRect& rect, bool invert, SkClipOp op, SkClipStack*
stack->clipPath(path, SkMatrix::I(), op, doAA); stack->clipPath(path, SkMatrix::I(), op, doAA);
}; };
static void add_noop(const SkRect& rect, bool invert, SkClipOp op, SkClipStack* stack,
bool doAA) {
// FIXME (michaelludwig) - this will be replaced with add_shader, but this lets us tickle a
// bug in GrReducedClip due to the particular set of random clips the test generated, which
// requires an additional clip element type.
};
static void add_elem_to_stack(const SkClipStack::Element& element, SkClipStack* stack) { static void add_elem_to_stack(const SkClipStack::Element& element, SkClipStack* stack) {
switch (element.getDeviceSpaceType()) { switch (element.getDeviceSpaceType()) {
case SkClipStack::Element::DeviceSpaceType::kRect: case SkClipStack::Element::DeviceSpaceType::kRect:
@ -986,6 +993,7 @@ static void test_reduced_clip_stack(skiatest::Reporter* reporter) {
add_rect, add_rect,
add_round_rect, add_round_rect,
add_oval, add_oval,
add_noop,
}; };
SkRandom r; SkRandom r;
@ -1192,8 +1200,8 @@ static void test_reduced_clip_stack_genid(skiatest::Reporter* reporter) {
// Other tests: // Other tests:
{ XYWH(0, 0, 100, 100), 4, genIDD, GrReducedClip::InitialState::kAllOut, stackBounds }, { XYWH(0, 0, 100, 100), 4, genIDD, GrReducedClip::InitialState::kAllOut, stackBounds },
// Rect in the middle, touches none. // Rect in the middle, touches none (so should not be drawn)
{ XYWH(26, 26, 24, 24), 0, SkClipStack::kInvalidGenID, GrReducedClip::InitialState::kAllOut, IXYWH(26, 26, 24, 24) }, { XYWH(26, 26, 24, 24), 0, SkClipStack::kInvalidGenID, GrReducedClip::InitialState::kAllOut, SkIRect::MakeEmpty() },
// Rect in the middle, touches all the rects. GenID is the last rect. // Rect in the middle, touches all the rects. GenID is the last rect.
{ XYWH(24, 24, 27, 27), 4, genIDD, GrReducedClip::InitialState::kAllOut, IXYWH(24, 24, 27, 27) }, { XYWH(24, 24, 27, 27), 4, genIDD, GrReducedClip::InitialState::kAllOut, IXYWH(24, 24, 27, 27) },
@ -1208,17 +1216,16 @@ static void test_reduced_clip_stack_genid(skiatest::Reporter* reporter) {
const GrReducedClip reduced(stack, testCases[i].testBounds, caps); const GrReducedClip reduced(stack, testCases[i].testBounds, caps);
REPORTER_ASSERT(reporter, reduced.maskElements().count() == REPORTER_ASSERT(reporter, reduced.maskElements().count() ==
testCases[i].reducedClipCount); testCases[i].reducedClipCount);
SkASSERT(reduced.maskElements().count() == testCases[i].reducedClipCount);
if (reduced.maskElements().count()) { if (reduced.maskElements().count()) {
REPORTER_ASSERT(reporter, reduced.maskGenID() == testCases[i].reducedGenID); REPORTER_ASSERT(reporter, reduced.maskGenID() == testCases[i].reducedGenID);
SkASSERT(reduced.maskGenID() == testCases[i].reducedGenID);
} }
REPORTER_ASSERT(reporter, reduced.initialState() == testCases[i].initialState); REPORTER_ASSERT(reporter, reduced.initialState() == testCases[i].initialState);
SkASSERT(reduced.initialState() == testCases[i].initialState);
REPORTER_ASSERT(reporter, reduced.hasScissor()); bool expectsScissor = !testCases[i].clipIRect.isEmpty();
SkASSERT(reduced.hasScissor()); REPORTER_ASSERT(reporter, expectsScissor == reduced.hasScissor());
if (expectsScissor) {
REPORTER_ASSERT(reporter, reduced.scissor() == testCases[i].clipIRect); REPORTER_ASSERT(reporter, reduced.scissor() == testCases[i].clipIRect);
SkASSERT(reduced.scissor() == testCases[i].clipIRect); }
} }
} }
} }