xref: /aosp_15_r20/external/skia/src/gpu/ganesh/tessellate/GrTessellationShader.cpp (revision c8dee2aa9b3f27cf6c858bd81872bdeb2c07ed17)
1*c8dee2aaSAndroid Build Coastguard Worker /*
2*c8dee2aaSAndroid Build Coastguard Worker  * Copyright 2021 Google LLC.
3*c8dee2aaSAndroid Build Coastguard Worker  *
4*c8dee2aaSAndroid Build Coastguard Worker  * Use of this source code is governed by a BSD-style license that can be
5*c8dee2aaSAndroid Build Coastguard Worker  * found in the LICENSE file.
6*c8dee2aaSAndroid Build Coastguard Worker  */
7*c8dee2aaSAndroid Build Coastguard Worker #include "src/gpu/ganesh/tessellate/GrTessellationShader.h"
8*c8dee2aaSAndroid Build Coastguard Worker 
9*c8dee2aaSAndroid Build Coastguard Worker #include "src/gpu/Swizzle.h"
10*c8dee2aaSAndroid Build Coastguard Worker #include "src/gpu/ganesh/GrDstProxyView.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "src/gpu/ganesh/GrPipeline.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "src/gpu/ganesh/GrSurfaceProxyView.h"
13*c8dee2aaSAndroid Build Coastguard Worker #include "src/gpu/tessellate/WangsFormula.h"
14*c8dee2aaSAndroid Build Coastguard Worker 
15*c8dee2aaSAndroid Build Coastguard Worker #include <utility>
16*c8dee2aaSAndroid Build Coastguard Worker 
17*c8dee2aaSAndroid Build Coastguard Worker enum class GrAAType : unsigned int;
18*c8dee2aaSAndroid Build Coastguard Worker 
MakePipeline(const ProgramArgs & args,GrAAType aaType,GrAppliedClip && appliedClip,GrProcessorSet && processors)19*c8dee2aaSAndroid Build Coastguard Worker const GrPipeline* GrTessellationShader::MakePipeline(const ProgramArgs& args,
20*c8dee2aaSAndroid Build Coastguard Worker                                                      GrAAType aaType,
21*c8dee2aaSAndroid Build Coastguard Worker                                                      GrAppliedClip&& appliedClip,
22*c8dee2aaSAndroid Build Coastguard Worker                                                      GrProcessorSet&& processors) {
23*c8dee2aaSAndroid Build Coastguard Worker     GrPipeline::InitArgs pipelineArgs;
24*c8dee2aaSAndroid Build Coastguard Worker 
25*c8dee2aaSAndroid Build Coastguard Worker     pipelineArgs.fCaps = args.fCaps;
26*c8dee2aaSAndroid Build Coastguard Worker     pipelineArgs.fDstProxyView = *args.fDstProxyView;
27*c8dee2aaSAndroid Build Coastguard Worker     pipelineArgs.fWriteSwizzle = args.fWriteView.swizzle();
28*c8dee2aaSAndroid Build Coastguard Worker 
29*c8dee2aaSAndroid Build Coastguard Worker     return args.fArena->make<GrPipeline>(pipelineArgs,
30*c8dee2aaSAndroid Build Coastguard Worker                                          std::move(processors),
31*c8dee2aaSAndroid Build Coastguard Worker                                          std::move(appliedClip));
32*c8dee2aaSAndroid Build Coastguard Worker }
33*c8dee2aaSAndroid Build Coastguard Worker 
WangsFormulaSkSL()34*c8dee2aaSAndroid Build Coastguard Worker const char* GrTessellationShader::WangsFormulaSkSL() {
35*c8dee2aaSAndroid Build Coastguard Worker     static_assert(skgpu::wangs_formula::length_term<3>(1) == 0.75);
36*c8dee2aaSAndroid Build Coastguard Worker     static_assert(skgpu::wangs_formula::length_term_p2<3>(1) == 0.5625);
37*c8dee2aaSAndroid Build Coastguard Worker 
38*c8dee2aaSAndroid Build Coastguard Worker     return
39*c8dee2aaSAndroid Build Coastguard Worker // Returns the length squared of the largest forward difference from Wang's cubic formula.
40*c8dee2aaSAndroid Build Coastguard Worker "float wangs_formula_max_fdiff_p2(float2 p0, float2 p1, float2 p2, float2 p3,"
41*c8dee2aaSAndroid Build Coastguard Worker                                  "float2x2 matrix) {"
42*c8dee2aaSAndroid Build Coastguard Worker     "float2 d0 = matrix * (fma(float2(-2), p1, p2) + p0);"
43*c8dee2aaSAndroid Build Coastguard Worker     "float2 d1 = matrix * (fma(float2(-2), p2, p3) + p1);"
44*c8dee2aaSAndroid Build Coastguard Worker     "return max(dot(d0,d0), dot(d1,d1));"
45*c8dee2aaSAndroid Build Coastguard Worker "}"
46*c8dee2aaSAndroid Build Coastguard Worker "float wangs_formula_cubic(float _precision_, float2 p0, float2 p1, float2 p2, float2 p3,"
47*c8dee2aaSAndroid Build Coastguard Worker                           "float2x2 matrix) {"
48*c8dee2aaSAndroid Build Coastguard Worker     "float m = wangs_formula_max_fdiff_p2(p0, p1, p2, p3, matrix);"
49*c8dee2aaSAndroid Build Coastguard Worker     "return max(ceil(sqrt(0.75 * _precision_ * sqrt(m))), 1.0);"
50*c8dee2aaSAndroid Build Coastguard Worker "}"
51*c8dee2aaSAndroid Build Coastguard Worker "float wangs_formula_cubic_log2(float _precision_, float2 p0, float2 p1, float2 p2, float2 p3,"
52*c8dee2aaSAndroid Build Coastguard Worker                                "float2x2 matrix) {"
53*c8dee2aaSAndroid Build Coastguard Worker     "float m = wangs_formula_max_fdiff_p2(p0, p1, p2, p3, matrix);"
54*c8dee2aaSAndroid Build Coastguard Worker     "return ceil(log2(max(0.5625 * _precision_ * _precision_ * m, 1.0)) * .25);"
55*c8dee2aaSAndroid Build Coastguard Worker "}"
56*c8dee2aaSAndroid Build Coastguard Worker "float wangs_formula_conic_p2(float _precision_, float2 p0, float2 p1, float2 p2, float w) {"
57*c8dee2aaSAndroid Build Coastguard Worker     // Translate the bounding box center to the origin.
58*c8dee2aaSAndroid Build Coastguard Worker     "float2 C = (min(min(p0, p1), p2) + max(max(p0, p1), p2)) * 0.5;"
59*c8dee2aaSAndroid Build Coastguard Worker     "p0 -= C;"
60*c8dee2aaSAndroid Build Coastguard Worker     "p1 -= C;"
61*c8dee2aaSAndroid Build Coastguard Worker     "p2 -= C;"
62*c8dee2aaSAndroid Build Coastguard Worker 
63*c8dee2aaSAndroid Build Coastguard Worker     // Compute max length.
64*c8dee2aaSAndroid Build Coastguard Worker     "float m = sqrt(max(max(dot(p0,p0), dot(p1,p1)), dot(p2,p2)));"
65*c8dee2aaSAndroid Build Coastguard Worker 
66*c8dee2aaSAndroid Build Coastguard Worker     // Compute forward differences.
67*c8dee2aaSAndroid Build Coastguard Worker     "float2 dp = fma(float2(-2.0 * w), p1, p0) + p2;"
68*c8dee2aaSAndroid Build Coastguard Worker     "float dw = abs(fma(-2.0, w, 2.0));"
69*c8dee2aaSAndroid Build Coastguard Worker 
70*c8dee2aaSAndroid Build Coastguard Worker     // Compute numerator and denominator for parametric step size of linearization. Here, the
71*c8dee2aaSAndroid Build Coastguard Worker     // epsilon referenced from the cited paper is 1/precision.
72*c8dee2aaSAndroid Build Coastguard Worker     "float rp_minus_1 = max(0.0, fma(m, _precision_, -1.0));"
73*c8dee2aaSAndroid Build Coastguard Worker     "float numer = length(dp) * _precision_ + rp_minus_1 * dw;"
74*c8dee2aaSAndroid Build Coastguard Worker     "float denom = 4 * min(w, 1.0);"
75*c8dee2aaSAndroid Build Coastguard Worker 
76*c8dee2aaSAndroid Build Coastguard Worker     "return numer/denom;"
77*c8dee2aaSAndroid Build Coastguard Worker "}"
78*c8dee2aaSAndroid Build Coastguard Worker "float wangs_formula_conic(float _precision_, float2 p0, float2 p1, float2 p2, float w) {"
79*c8dee2aaSAndroid Build Coastguard Worker     "float n2 = wangs_formula_conic_p2(_precision_, p0, p1, p2, w);"
80*c8dee2aaSAndroid Build Coastguard Worker     "return max(ceil(sqrt(n2)), 1.0);"
81*c8dee2aaSAndroid Build Coastguard Worker "}"
82*c8dee2aaSAndroid Build Coastguard Worker "float wangs_formula_conic_log2(float _precision_, float2 p0, float2 p1, float2 p2, float w) {"
83*c8dee2aaSAndroid Build Coastguard Worker     "float n2 = wangs_formula_conic_p2(_precision_, p0, p1, p2, w);"
84*c8dee2aaSAndroid Build Coastguard Worker     "return ceil(log2(max(n2, 1.0)) * .5);"
85*c8dee2aaSAndroid Build Coastguard Worker "}"
86*c8dee2aaSAndroid Build Coastguard Worker ;
87*c8dee2aaSAndroid Build Coastguard Worker }
88