1/* 2 * Copyright 2018 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 8layout(key) in GrClipEdgeType edgeType; 9layout(ctype=SkRect) in float4 rect; 10layout(ctype=SkRect) float4 prevRect = float4(-1); 11uniform float4 rectUniform; 12 13@optimizationFlags { kCompatibleWithCoverageAsAlpha_OptimizationFlag } 14 15void main() { 16 half alpha; 17 @switch (edgeType) { 18 case GrClipEdgeType::kFillBW: // fall through 19 case GrClipEdgeType::kInverseFillBW: 20 // non-AA 21 alpha = all(greaterThan(float4(sk_FragCoord.xy, rectUniform.zw), 22 float4(rectUniform.xy, sk_FragCoord.xy))) ? 1 : 0; 23 break; 24 default: 25 // The amount of coverage removed in x and y by the edges is computed as a pair of 26 // negative numbers, xSub and ySub. 27 half xSub, ySub; 28 xSub = min(half(sk_FragCoord.x - rectUniform.x), 0.0); 29 xSub += min(half(rectUniform.z - sk_FragCoord.x), 0.0); 30 ySub = min(half(sk_FragCoord.y - rectUniform.y), 0.0); 31 ySub += min(half(rectUniform.w - sk_FragCoord.y), 0.0); 32 // Now compute coverage in x and y and multiply them to get the fraction of the pixel 33 // covered. 34 alpha = (1.0 + max(xSub, -1.0)) * (1.0 + max(ySub, -1.0)); 35 } 36 37 @if (edgeType == GrClipEdgeType::kInverseFillBW || edgeType == GrClipEdgeType::kInverseFillAA) { 38 alpha = 1.0 - alpha; 39 } 40 sk_OutColor = sk_InColor * alpha; 41} 42 43@setData(pdman) { 44 const SkRect& newRect = GrProcessorEdgeTypeIsAA(edgeType) ? 45 rect.makeInset(.5f, .5f) : rect; 46 if (newRect != prevRect) { 47 pdman.set4f(rectUniform, newRect.fLeft, newRect.fTop, newRect.fRight, newRect.fBottom); 48 prevRect = newRect; 49 } 50} 51 52@test(d) { 53 SkRect rect = SkRect::MakeLTRB(d->fRandom->nextSScalar1(), 54 d->fRandom->nextSScalar1(), 55 d->fRandom->nextSScalar1(), 56 d->fRandom->nextSScalar1()); 57 std::unique_ptr<GrFragmentProcessor> fp; 58 do { 59 GrClipEdgeType edgeType = static_cast<GrClipEdgeType>( 60 d->fRandom->nextULessThan(kGrClipEdgeTypeCnt)); 61 62 fp = GrAARectEffect::Make(edgeType, rect); 63 } while (nullptr == fp); 64 return fp; 65} 66