1/* 2 * Copyright 2017 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; 9in half2 center; 10in half radius; 11 12half2 prevCenter; 13half prevRadius = -1; 14// The circle uniform is (center.x, center.y, radius + 0.5, 1 / (radius + 0.5)) for regular 15// fills and (..., radius - 0.5, 1 / (radius - 0.5)) for inverse fills. 16uniform half4 circle; 17 18@make { 19 static std::unique_ptr<GrFragmentProcessor> Make(GrClipEdgeType edgeType, SkPoint center, 20 float radius) { 21 // A radius below half causes the implicit insetting done by this processor to become 22 // inverted. We could handle this case by making the processor code more complicated. 23 if (radius < .5f && GrProcessorEdgeTypeIsInverseFill(edgeType)) { 24 return nullptr; 25 } 26 return std::unique_ptr<GrFragmentProcessor>(new GrCircleEffect(edgeType, center, radius)); 27 } 28} 29 30@optimizationFlags { kCompatibleWithCoverageAsAlpha_OptimizationFlag } 31 32@setData(pdman) { 33 if (radius != prevRadius || center != prevCenter) { 34 SkScalar effectiveRadius = radius; 35 if (GrProcessorEdgeTypeIsInverseFill((GrClipEdgeType) edgeType)) { 36 effectiveRadius -= 0.5f; 37 // When the radius is 0.5 effectiveRadius is 0 which causes an inf * 0 in the shader. 38 effectiveRadius = SkTMax(0.001f, effectiveRadius); 39 } else { 40 effectiveRadius += 0.5f; 41 } 42 pdman.set4f(circle, center.fX, center.fY, effectiveRadius, 43 SkScalarInvert(effectiveRadius)); 44 prevCenter = center; 45 prevRadius = radius; 46 } 47} 48 49void main() { 50 // TODO: Right now the distance to circle caclulation is performed in a space normalized to the 51 // radius and then denormalized. This is to prevent overflow on devices that have a "real" 52 // mediump. It'd be nice to only do this on mediump devices. 53 half d; 54 @if (edgeType == GrClipEdgeType::kInverseFillBW || 55 edgeType == GrClipEdgeType::kInverseFillAA) { 56 d = half((length((circle.xy - sk_FragCoord.xy) * circle.w) - 1.0) * circle.z); 57 } else { 58 d = half((1.0 - length((circle.xy - sk_FragCoord.xy) * circle.w)) * circle.z); 59 } 60 @if (edgeType == GrClipEdgeType::kFillAA || 61 edgeType == GrClipEdgeType::kInverseFillAA || 62 edgeType == GrClipEdgeType::kHairlineAA) { 63 d = saturate(d); 64 } else { 65 d = d > 0.5 ? 1.0 : 0.0; 66 } 67 68 sk_OutColor = sk_InColor * d; 69} 70 71@test(testData) { 72 SkPoint center; 73 center.fX = testData->fRandom->nextRangeScalar(0.f, 1000.f); 74 center.fY = testData->fRandom->nextRangeScalar(0.f, 1000.f); 75 SkScalar radius = testData->fRandom->nextRangeF(1.f, 1000.f); 76 GrClipEdgeType et; 77 do { 78 et = (GrClipEdgeType) testData->fRandom->nextULessThan(kGrClipEdgeTypeCnt); 79 } while (GrClipEdgeType::kHairlineAA == et); 80 return GrCircleEffect::Make(et, center, radius); 81}