blob: 3a97506396c6cb7ae882d73d9bd7c3974824732b [file] [log] [blame]
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +00001/*
2 * Copyright 2013 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#ifndef GrBezierEffect_DEFINED
skia.committer@gmail.com44a77c82013-08-23 07:01:29 +00009#define GrBezierEffect_DEFINED
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +000010
bsalomoneb1cb5c2015-05-22 08:01:09 -070011#include "GrCaps.h"
joshualittb0a8a372014-09-23 09:50:21 -070012#include "GrProcessor.h"
joshualitt249af152014-09-15 11:41:13 -070013#include "GrGeometryProcessor.h"
commit-bot@chromium.orgcabf4b22014-03-05 18:27:43 +000014#include "GrTypesPriv.h"
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +000015
16/**
17 * Shader is based off of Loop-Blinn Quadratic GPU Rendering
18 * The output of this effect is a hairline edge for conics.
19 * Conics specified by implicit equation K^2 - LM.
20 * K, L, and M, are the first three values of the vertex attribute,
21 * the fourth value is not used. Distance is calculated using a
22 * first order approximation from the taylor series.
23 * Coverage for AA is max(0, 1-distance).
24 *
25 * Test were also run using a second order distance approximation.
26 * There were two versions of the second order approx. The first version
27 * is of roughly the form:
28 * f(q) = |f(p)| - ||f'(p)||*||q-p|| - ||f''(p)||*||q-p||^2.
29 * The second is similar:
30 * f(q) = |f(p)| + ||f'(p)||*||q-p|| + ||f''(p)||*||q-p||^2.
31 * The exact version of the equations can be found in the paper
32 * "Distance Approximations for Rasterizing Implicit Curves" by Gabriel Taubin
33 *
34 * In both versions we solve the quadratic for ||q-p||.
35 * Version 1:
36 * gFM is magnitude of first partials and gFM2 is magnitude of 2nd partials (as derived from paper)
37 * builder->fsCodeAppend("\t\tedgeAlpha = (sqrt(gFM*gFM+4.0*func*gF2M) - gFM)/(2.0*gF2M);\n");
38 * Version 2:
39 * builder->fsCodeAppend("\t\tedgeAlpha = (gFM - sqrt(gFM*gFM-4.0*func*gF2M))/(2.0*gF2M);\n");
40 *
41 * Also note that 2nd partials of k,l,m are zero
42 *
43 * When comparing the two second order approximations to the first order approximations,
44 * the following results were found. Version 1 tends to underestimate the distances, thus it
45 * basically increases all the error that we were already seeing in the first order
46 * approx. So this version is not the one to use. Version 2 has the opposite effect
47 * and tends to overestimate the distances. This is much closer to what we are
48 * looking for. It is able to render ellipses (even thin ones) without the need to chop.
49 * However, it can not handle thin hyperbolas well and thus would still rely on
50 * chopping to tighten the clipping. Another side effect of the overestimating is
51 * that the curves become much thinner and "ropey". If all that was ever rendered
52 * were "not too thin" curves and ellipses then 2nd order may have an advantage since
53 * only one geometry would need to be rendered. However no benches were run comparing
54 * chopped first order and non chopped 2nd order.
55 */
56class GrGLConicEffect;
57
joshualitt249af152014-09-15 11:41:13 -070058class GrConicEffect : public GrGeometryProcessor {
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +000059public:
bungeman06ca8ec2016-06-09 08:01:03 -070060 static sk_sp<GrGeometryProcessor> Make(GrColor color,
61 const SkMatrix& viewMatrix,
62 const GrPrimitiveEdgeType edgeType,
63 const GrCaps& caps,
64 const SkMatrix& localMatrix,
65 bool usesLocalCoords,
66 uint8_t coverage = 0xff) {
commit-bot@chromium.orgcabf4b22014-03-05 18:27:43 +000067 switch (edgeType) {
joshualittb0a8a372014-09-23 09:50:21 -070068 case kFillAA_GrProcessorEdgeType:
jvanverthe9c0fc62015-04-29 11:18:05 -070069 if (!caps.shaderCaps()->shaderDerivativeSupport()) {
halcanary96fcdcc2015-08-27 07:41:13 -070070 return nullptr;
commit-bot@chromium.orgcabf4b22014-03-05 18:27:43 +000071 }
bungeman06ca8ec2016-06-09 08:01:03 -070072 return sk_sp<GrGeometryProcessor>(
73 new GrConicEffect(color, viewMatrix, coverage, kFillAA_GrProcessorEdgeType,
74 localMatrix, usesLocalCoords));
joshualittb0a8a372014-09-23 09:50:21 -070075 case kHairlineAA_GrProcessorEdgeType:
jvanverthe9c0fc62015-04-29 11:18:05 -070076 if (!caps.shaderCaps()->shaderDerivativeSupport()) {
halcanary96fcdcc2015-08-27 07:41:13 -070077 return nullptr;
commit-bot@chromium.orgcabf4b22014-03-05 18:27:43 +000078 }
bungeman06ca8ec2016-06-09 08:01:03 -070079 return sk_sp<GrGeometryProcessor>(
80 new GrConicEffect(color, viewMatrix, coverage,
81 kHairlineAA_GrProcessorEdgeType, localMatrix,
82 usesLocalCoords));
joshualittb0a8a372014-09-23 09:50:21 -070083 case kFillBW_GrProcessorEdgeType:
bungeman06ca8ec2016-06-09 08:01:03 -070084 return sk_sp<GrGeometryProcessor>(
85 new GrConicEffect(color, viewMatrix, coverage, kFillBW_GrProcessorEdgeType,
86 localMatrix, usesLocalCoords));
commit-bot@chromium.orgcabf4b22014-03-05 18:27:43 +000087 default:
halcanary96fcdcc2015-08-27 07:41:13 -070088 return nullptr;
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +000089 }
90 }
91
Brian Salomond3b65972017-03-22 12:05:03 -040092 ~GrConicEffect() override;
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +000093
mtklein36352bf2015-03-25 18:17:31 -070094 const char* name() const override { return "Conic"; }
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +000095
joshualitt71c92602015-01-14 08:12:47 -080096 inline const Attribute* inPosition() const { return fInPosition; }
97 inline const Attribute* inConicCoeffs() const { return fInConicCoeffs; }
joshualittb0a8a372014-09-23 09:50:21 -070098 inline bool isAntiAliased() const { return GrProcessorEdgeTypeIsAA(fEdgeType); }
99 inline bool isFilled() const { return GrProcessorEdgeTypeIsFill(fEdgeType); }
100 inline GrPrimitiveEdgeType getEdgeType() const { return fEdgeType; }
joshualitt88c23fc2015-05-13 14:18:07 -0700101 GrColor color() const { return fColor; }
joshualitte578a952015-05-14 10:09:13 -0700102 const SkMatrix& viewMatrix() const { return fViewMatrix; }
joshualitte3ababe2015-05-15 07:56:07 -0700103 const SkMatrix& localMatrix() const { return fLocalMatrix; }
joshualittb8c241a2015-05-19 08:23:30 -0700104 bool usesLocalCoords() const { return fUsesLocalCoords; }
105 uint8_t coverageScale() const { return fCoverageScale; }
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +0000106
Brian Salomon94efbf52016-11-29 13:43:05 -0500107 void getGLSLProcessorKey(const GrShaderCaps& caps, GrProcessorKeyBuilder* b) const override;
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +0000108
Brian Salomon94efbf52016-11-29 13:43:05 -0500109 GrGLSLPrimitiveProcessor* createGLSLInstance(const GrShaderCaps&) const override;
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +0000110
111private:
joshualitt8059eb92014-12-29 15:10:07 -0800112 GrConicEffect(GrColor, const SkMatrix& viewMatrix, uint8_t coverage, GrPrimitiveEdgeType,
joshualittb8c241a2015-05-19 08:23:30 -0700113 const SkMatrix& localMatrix, bool usesLocalCoords);
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +0000114
joshualitt88c23fc2015-05-13 14:18:07 -0700115 GrColor fColor;
joshualitte578a952015-05-14 10:09:13 -0700116 SkMatrix fViewMatrix;
joshualitte3ababe2015-05-15 07:56:07 -0700117 SkMatrix fLocalMatrix;
joshualittb8c241a2015-05-19 08:23:30 -0700118 bool fUsesLocalCoords;
joshualitt88c23fc2015-05-13 14:18:07 -0700119 uint8_t fCoverageScale;
120 GrPrimitiveEdgeType fEdgeType;
joshualitt71c92602015-01-14 08:12:47 -0800121 const Attribute* fInPosition;
122 const Attribute* fInConicCoeffs;
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +0000123
joshualittb0a8a372014-09-23 09:50:21 -0700124 GR_DECLARE_GEOMETRY_PROCESSOR_TEST;
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +0000125
joshualitt249af152014-09-15 11:41:13 -0700126 typedef GrGeometryProcessor INHERITED;
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +0000127};
128
129///////////////////////////////////////////////////////////////////////////////
130/**
131 * The output of this effect is a hairline edge for quadratics.
132 * Quadratic specified by 0=u^2-v canonical coords. u and v are the first
133 * two components of the vertex attribute. At the three control points that define
134 * the Quadratic, u, v have the values {0,0}, {1/2, 0}, and {1, 1} respectively.
135 * Coverage for AA is min(0, 1-distance). 3rd & 4th cimponent unused.
136 * Requires shader derivative instruction support.
137 */
138class GrGLQuadEffect;
139
joshualitt249af152014-09-15 11:41:13 -0700140class GrQuadEffect : public GrGeometryProcessor {
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +0000141public:
bungeman06ca8ec2016-06-09 08:01:03 -0700142 static sk_sp<GrGeometryProcessor> Make(GrColor color,
143 const SkMatrix& viewMatrix,
144 const GrPrimitiveEdgeType edgeType,
145 const GrCaps& caps,
146 const SkMatrix& localMatrix,
147 bool usesLocalCoords,
148 uint8_t coverage = 0xff) {
commit-bot@chromium.orgcabf4b22014-03-05 18:27:43 +0000149 switch (edgeType) {
joshualittb0a8a372014-09-23 09:50:21 -0700150 case kFillAA_GrProcessorEdgeType:
jvanverthe9c0fc62015-04-29 11:18:05 -0700151 if (!caps.shaderCaps()->shaderDerivativeSupport()) {
halcanary96fcdcc2015-08-27 07:41:13 -0700152 return nullptr;
commit-bot@chromium.orgcabf4b22014-03-05 18:27:43 +0000153 }
bungeman06ca8ec2016-06-09 08:01:03 -0700154 return sk_sp<GrGeometryProcessor>(
155 new GrQuadEffect(color, viewMatrix, coverage, kFillAA_GrProcessorEdgeType,
156 localMatrix, usesLocalCoords));
joshualittb0a8a372014-09-23 09:50:21 -0700157 case kHairlineAA_GrProcessorEdgeType:
jvanverthe9c0fc62015-04-29 11:18:05 -0700158 if (!caps.shaderCaps()->shaderDerivativeSupport()) {
halcanary96fcdcc2015-08-27 07:41:13 -0700159 return nullptr;
commit-bot@chromium.orgcabf4b22014-03-05 18:27:43 +0000160 }
bungeman06ca8ec2016-06-09 08:01:03 -0700161 return sk_sp<GrGeometryProcessor>(
162 new GrQuadEffect(color, viewMatrix, coverage,
163 kHairlineAA_GrProcessorEdgeType, localMatrix,
164 usesLocalCoords));
joshualittb0a8a372014-09-23 09:50:21 -0700165 case kFillBW_GrProcessorEdgeType:
bungeman06ca8ec2016-06-09 08:01:03 -0700166 return sk_sp<GrGeometryProcessor>(
167 new GrQuadEffect(color, viewMatrix, coverage, kFillBW_GrProcessorEdgeType,
168 localMatrix, usesLocalCoords));
commit-bot@chromium.orgcabf4b22014-03-05 18:27:43 +0000169 default:
halcanary96fcdcc2015-08-27 07:41:13 -0700170 return nullptr;
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +0000171 }
172 }
173
Brian Salomond3b65972017-03-22 12:05:03 -0400174 ~GrQuadEffect() override;
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +0000175
mtklein36352bf2015-03-25 18:17:31 -0700176 const char* name() const override { return "Quad"; }
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +0000177
joshualitt71c92602015-01-14 08:12:47 -0800178 inline const Attribute* inPosition() const { return fInPosition; }
179 inline const Attribute* inHairQuadEdge() const { return fInHairQuadEdge; }
joshualittb0a8a372014-09-23 09:50:21 -0700180 inline bool isAntiAliased() const { return GrProcessorEdgeTypeIsAA(fEdgeType); }
181 inline bool isFilled() const { return GrProcessorEdgeTypeIsFill(fEdgeType); }
182 inline GrPrimitiveEdgeType getEdgeType() const { return fEdgeType; }
joshualitt88c23fc2015-05-13 14:18:07 -0700183 GrColor color() const { return fColor; }
joshualitte578a952015-05-14 10:09:13 -0700184 const SkMatrix& viewMatrix() const { return fViewMatrix; }
joshualitte3ababe2015-05-15 07:56:07 -0700185 const SkMatrix& localMatrix() const { return fLocalMatrix; }
joshualittb8c241a2015-05-19 08:23:30 -0700186 bool usesLocalCoords() const { return fUsesLocalCoords; }
187 uint8_t coverageScale() const { return fCoverageScale; }
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +0000188
Brian Salomon94efbf52016-11-29 13:43:05 -0500189 void getGLSLProcessorKey(const GrShaderCaps& caps, GrProcessorKeyBuilder* b) const override;
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +0000190
Brian Salomon94efbf52016-11-29 13:43:05 -0500191 GrGLSLPrimitiveProcessor* createGLSLInstance(const GrShaderCaps&) const override;
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +0000192
193private:
joshualitt8059eb92014-12-29 15:10:07 -0800194 GrQuadEffect(GrColor, const SkMatrix& viewMatrix, uint8_t coverage, GrPrimitiveEdgeType,
joshualittb8c241a2015-05-19 08:23:30 -0700195 const SkMatrix& localMatrix, bool usesLocalCoords);
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +0000196
joshualitt88c23fc2015-05-13 14:18:07 -0700197 GrColor fColor;
joshualitte578a952015-05-14 10:09:13 -0700198 SkMatrix fViewMatrix;
joshualitte3ababe2015-05-15 07:56:07 -0700199 SkMatrix fLocalMatrix;
joshualittb8c241a2015-05-19 08:23:30 -0700200 bool fUsesLocalCoords;
joshualitt88c23fc2015-05-13 14:18:07 -0700201 uint8_t fCoverageScale;
202 GrPrimitiveEdgeType fEdgeType;
joshualitt71c92602015-01-14 08:12:47 -0800203 const Attribute* fInPosition;
204 const Attribute* fInHairQuadEdge;
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +0000205
joshualittb0a8a372014-09-23 09:50:21 -0700206 GR_DECLARE_GEOMETRY_PROCESSOR_TEST;
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +0000207
joshualitt249af152014-09-15 11:41:13 -0700208 typedef GrGeometryProcessor INHERITED;
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +0000209};
210
211//////////////////////////////////////////////////////////////////////////////
212/**
213 * Shader is based off of "Resolution Independent Curve Rendering using
214 * Programmable Graphics Hardware" by Loop and Blinn.
215 * The output of this effect is a hairline edge for non rational cubics.
216 * Cubics are specified by implicit equation K^3 - LM.
217 * K, L, and M, are the first three values of the vertex attribute,
218 * the fourth value is not used. Distance is calculated using a
219 * first order approximation from the taylor series.
220 * Coverage for AA is max(0, 1-distance).
221 */
222class GrGLCubicEffect;
223
joshualitt249af152014-09-15 11:41:13 -0700224class GrCubicEffect : public GrGeometryProcessor {
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +0000225public:
bungeman06ca8ec2016-06-09 08:01:03 -0700226 static sk_sp<GrGeometryProcessor> Make(GrColor color,
227 const SkMatrix& viewMatrix,
Chris Daltonfebbffa2017-06-08 13:12:02 -0600228 const SkMatrix& klm,
229 bool flipKL,
bungeman06ca8ec2016-06-09 08:01:03 -0700230 const GrPrimitiveEdgeType edgeType,
231 const GrCaps& caps) {
Chris Daltonfebbffa2017-06-08 13:12:02 -0600232 // Map KLM to something that operates in device space.
233 SkMatrix devKLM;
234 if (!viewMatrix.invert(&devKLM)) {
235 return nullptr;
236 }
237 devKLM.postConcat(klm);
238 if (flipKL) {
239 devKLM.postScale(-1, -1);
240 }
241
commit-bot@chromium.orgcabf4b22014-03-05 18:27:43 +0000242 switch (edgeType) {
joshualittb0a8a372014-09-23 09:50:21 -0700243 case kFillAA_GrProcessorEdgeType:
bungeman06ca8ec2016-06-09 08:01:03 -0700244 return sk_sp<GrGeometryProcessor>(
Chris Daltonfebbffa2017-06-08 13:12:02 -0600245 new GrCubicEffect(color, viewMatrix, devKLM, kFillAA_GrProcessorEdgeType));
joshualittb0a8a372014-09-23 09:50:21 -0700246 case kHairlineAA_GrProcessorEdgeType:
bungeman06ca8ec2016-06-09 08:01:03 -0700247 return sk_sp<GrGeometryProcessor>(
Chris Daltonfebbffa2017-06-08 13:12:02 -0600248 new GrCubicEffect(color, viewMatrix, devKLM, kHairlineAA_GrProcessorEdgeType));
joshualittb0a8a372014-09-23 09:50:21 -0700249 case kFillBW_GrProcessorEdgeType:
bungeman06ca8ec2016-06-09 08:01:03 -0700250 return sk_sp<GrGeometryProcessor>(
Chris Daltonfebbffa2017-06-08 13:12:02 -0600251 new GrCubicEffect(color, viewMatrix, devKLM, kFillBW_GrProcessorEdgeType));
commit-bot@chromium.orgcabf4b22014-03-05 18:27:43 +0000252 default:
halcanary96fcdcc2015-08-27 07:41:13 -0700253 return nullptr;
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +0000254 }
255 }
256
Brian Salomond3b65972017-03-22 12:05:03 -0400257 ~GrCubicEffect() override;
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +0000258
mtklein36352bf2015-03-25 18:17:31 -0700259 const char* name() const override { return "Cubic"; }
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +0000260
joshualitt71c92602015-01-14 08:12:47 -0800261 inline const Attribute* inPosition() const { return fInPosition; }
joshualittb0a8a372014-09-23 09:50:21 -0700262 inline bool isAntiAliased() const { return GrProcessorEdgeTypeIsAA(fEdgeType); }
263 inline bool isFilled() const { return GrProcessorEdgeTypeIsFill(fEdgeType); }
264 inline GrPrimitiveEdgeType getEdgeType() const { return fEdgeType; }
joshualitt88c23fc2015-05-13 14:18:07 -0700265 GrColor color() const { return fColor; }
joshualittb8c241a2015-05-19 08:23:30 -0700266 bool colorIgnored() const { return GrColor_ILLEGAL == fColor; }
joshualitte578a952015-05-14 10:09:13 -0700267 const SkMatrix& viewMatrix() const { return fViewMatrix; }
Chris Daltonfebbffa2017-06-08 13:12:02 -0600268 const SkMatrix& devKLMMatrix() const { return fDevKLMMatrix; }
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +0000269
Brian Salomon94efbf52016-11-29 13:43:05 -0500270 void getGLSLProcessorKey(const GrShaderCaps& caps, GrProcessorKeyBuilder* b) const override;
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +0000271
Brian Salomon94efbf52016-11-29 13:43:05 -0500272 GrGLSLPrimitiveProcessor* createGLSLInstance(const GrShaderCaps&) const override;
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +0000273
274private:
Chris Daltonfebbffa2017-06-08 13:12:02 -0600275 GrCubicEffect(GrColor, const SkMatrix& viewMatrix, const SkMatrix& devKLMMatrix,
276 GrPrimitiveEdgeType);
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +0000277
joshualitt88c23fc2015-05-13 14:18:07 -0700278 GrColor fColor;
joshualitte578a952015-05-14 10:09:13 -0700279 SkMatrix fViewMatrix;
Chris Daltonfebbffa2017-06-08 13:12:02 -0600280 SkMatrix fDevKLMMatrix;
joshualitt88c23fc2015-05-13 14:18:07 -0700281 GrPrimitiveEdgeType fEdgeType;
joshualitt71c92602015-01-14 08:12:47 -0800282 const Attribute* fInPosition;
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +0000283
joshualittb0a8a372014-09-23 09:50:21 -0700284 GR_DECLARE_GEOMETRY_PROCESSOR_TEST;
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +0000285
joshualitt249af152014-09-15 11:41:13 -0700286 typedef GrGeometryProcessor INHERITED;
commit-bot@chromium.org07e1c3f2013-08-22 20:41:15 +0000287};
288
289#endif