xref: /aosp_15_r20/external/skia/src/sksl/ir/SkSLConstructorCompound.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 
8*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLConstructorCompound.h"
9*c8dee2aaSAndroid Build Coastguard Worker 
10*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkTypes.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/base/SkTArray.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLAnalysis.h"
13*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLConstantFolder.h"
14*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLContext.h"
15*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLProgramSettings.h"
16*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLConstructorSplat.h"
17*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLExpression.h"
18*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLLiteral.h"
19*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLType.h"
20*c8dee2aaSAndroid Build Coastguard Worker 
21*c8dee2aaSAndroid Build Coastguard Worker #include <algorithm>
22*c8dee2aaSAndroid Build Coastguard Worker #include <cstddef>
23*c8dee2aaSAndroid Build Coastguard Worker #include <numeric>
24*c8dee2aaSAndroid Build Coastguard Worker #include <string>
25*c8dee2aaSAndroid Build Coastguard Worker 
26*c8dee2aaSAndroid Build Coastguard Worker namespace SkSL {
27*c8dee2aaSAndroid Build Coastguard Worker 
is_safe_to_eliminate(const Type & type,const Expression & arg)28*c8dee2aaSAndroid Build Coastguard Worker static bool is_safe_to_eliminate(const Type& type, const Expression& arg) {
29*c8dee2aaSAndroid Build Coastguard Worker     if (type.isScalar()) {
30*c8dee2aaSAndroid Build Coastguard Worker         // A scalar "compound type" with a single scalar argument is a no-op and can be eliminated.
31*c8dee2aaSAndroid Build Coastguard Worker         // (Pedantically, this isn't a compound at all, but it's harmless to allow and simplifies
32*c8dee2aaSAndroid Build Coastguard Worker         // call sites which need to narrow a vector and may sometimes end up with a scalar.)
33*c8dee2aaSAndroid Build Coastguard Worker         SkASSERTF(arg.type().matches(type), "Creating type '%s' from '%s'",
34*c8dee2aaSAndroid Build Coastguard Worker                   type.description().c_str(), arg.type().description().c_str());
35*c8dee2aaSAndroid Build Coastguard Worker         return true;
36*c8dee2aaSAndroid Build Coastguard Worker     }
37*c8dee2aaSAndroid Build Coastguard Worker     if (type.isVector() && arg.type().matches(type)) {
38*c8dee2aaSAndroid Build Coastguard Worker         // A vector compound constructor containing a single argument of matching type can trivially
39*c8dee2aaSAndroid Build Coastguard Worker         // be eliminated.
40*c8dee2aaSAndroid Build Coastguard Worker         return true;
41*c8dee2aaSAndroid Build Coastguard Worker     }
42*c8dee2aaSAndroid Build Coastguard Worker     // This is a meaningful single-argument compound constructor (e.g. vector-from-matrix,
43*c8dee2aaSAndroid Build Coastguard Worker     // matrix-from-vector).
44*c8dee2aaSAndroid Build Coastguard Worker     return false;
45*c8dee2aaSAndroid Build Coastguard Worker }
46*c8dee2aaSAndroid Build Coastguard Worker 
make_splat_from_arguments(const Type & type,const ExpressionArray & args)47*c8dee2aaSAndroid Build Coastguard Worker static const Expression* make_splat_from_arguments(const Type& type, const ExpressionArray& args) {
48*c8dee2aaSAndroid Build Coastguard Worker     // Splats cannot represent a matrix.
49*c8dee2aaSAndroid Build Coastguard Worker     if (type.isMatrix()) {
50*c8dee2aaSAndroid Build Coastguard Worker         return nullptr;
51*c8dee2aaSAndroid Build Coastguard Worker     }
52*c8dee2aaSAndroid Build Coastguard Worker     const Expression* splatExpression = nullptr;
53*c8dee2aaSAndroid Build Coastguard Worker     for (int index = 0; index < args.size(); ++index) {
54*c8dee2aaSAndroid Build Coastguard Worker         // Arguments must only be scalars or a splat constructors (which can only contain scalars).
55*c8dee2aaSAndroid Build Coastguard Worker         const Expression* expr;
56*c8dee2aaSAndroid Build Coastguard Worker         if (args[index]->type().isScalar()) {
57*c8dee2aaSAndroid Build Coastguard Worker             expr = args[index].get();
58*c8dee2aaSAndroid Build Coastguard Worker         } else if (args[index]->is<ConstructorSplat>()) {
59*c8dee2aaSAndroid Build Coastguard Worker             expr = args[index]->as<ConstructorSplat>().argument().get();
60*c8dee2aaSAndroid Build Coastguard Worker         } else {
61*c8dee2aaSAndroid Build Coastguard Worker             return nullptr;
62*c8dee2aaSAndroid Build Coastguard Worker         }
63*c8dee2aaSAndroid Build Coastguard Worker         // On the first iteration, just remember the expression we encountered.
64*c8dee2aaSAndroid Build Coastguard Worker         if (index == 0) {
65*c8dee2aaSAndroid Build Coastguard Worker             splatExpression = expr;
66*c8dee2aaSAndroid Build Coastguard Worker             continue;
67*c8dee2aaSAndroid Build Coastguard Worker         }
68*c8dee2aaSAndroid Build Coastguard Worker         // On subsequent iterations, ensure that the expression we found matches the first one.
69*c8dee2aaSAndroid Build Coastguard Worker         // (Note that IsSameExpressionTree will always reject an Expression with side effects.)
70*c8dee2aaSAndroid Build Coastguard Worker         if (!Analysis::IsSameExpressionTree(*expr, *splatExpression)) {
71*c8dee2aaSAndroid Build Coastguard Worker             return nullptr;
72*c8dee2aaSAndroid Build Coastguard Worker         }
73*c8dee2aaSAndroid Build Coastguard Worker     }
74*c8dee2aaSAndroid Build Coastguard Worker 
75*c8dee2aaSAndroid Build Coastguard Worker     return splatExpression;
76*c8dee2aaSAndroid Build Coastguard Worker }
77*c8dee2aaSAndroid Build Coastguard Worker 
Make(const Context & context,Position pos,const Type & type,ExpressionArray args)78*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> ConstructorCompound::Make(const Context& context,
79*c8dee2aaSAndroid Build Coastguard Worker                                                       Position pos,
80*c8dee2aaSAndroid Build Coastguard Worker                                                       const Type& type,
81*c8dee2aaSAndroid Build Coastguard Worker                                                       ExpressionArray args) {
82*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(type.isAllowedInES2(context));
83*c8dee2aaSAndroid Build Coastguard Worker 
84*c8dee2aaSAndroid Build Coastguard Worker     // All the arguments must have matching component type.
85*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(std::all_of(args.begin(), args.end(), [&](const std::unique_ptr<Expression>& arg) {
86*c8dee2aaSAndroid Build Coastguard Worker         const Type& argType = arg->type();
87*c8dee2aaSAndroid Build Coastguard Worker         return (argType.isScalar() || argType.isVector() || argType.isMatrix()) &&
88*c8dee2aaSAndroid Build Coastguard Worker                (argType.componentType().matches(type.componentType()));
89*c8dee2aaSAndroid Build Coastguard Worker     }));
90*c8dee2aaSAndroid Build Coastguard Worker 
91*c8dee2aaSAndroid Build Coastguard Worker     // The slot count of the combined argument list must match the composite type's slot count.
92*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(type.slotCount() ==
93*c8dee2aaSAndroid Build Coastguard Worker              std::accumulate(args.begin(), args.end(), /*initial value*/ (size_t)0,
94*c8dee2aaSAndroid Build Coastguard Worker                              [](size_t n, const std::unique_ptr<Expression>& arg) {
95*c8dee2aaSAndroid Build Coastguard Worker                                  return n + arg->type().slotCount();
96*c8dee2aaSAndroid Build Coastguard Worker                              }));
97*c8dee2aaSAndroid Build Coastguard Worker     // No-op compound constructors (containing a single argument of the same type) are eliminated.
98*c8dee2aaSAndroid Build Coastguard Worker     // (Even though this is a "compound constructor," we let scalars pass through here; it's
99*c8dee2aaSAndroid Build Coastguard Worker     // harmless to allow and simplifies call sites which need to narrow a vector and may sometimes
100*c8dee2aaSAndroid Build Coastguard Worker     // end up with a scalar.)
101*c8dee2aaSAndroid Build Coastguard Worker     if (args.size() == 1 && is_safe_to_eliminate(type, *args.front())) {
102*c8dee2aaSAndroid Build Coastguard Worker         args.front()->fPosition = pos;
103*c8dee2aaSAndroid Build Coastguard Worker         return std::move(args.front());
104*c8dee2aaSAndroid Build Coastguard Worker     }
105*c8dee2aaSAndroid Build Coastguard Worker     // Beyond this point, the type must be a vector or matrix.
106*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(type.isVector() || type.isMatrix());
107*c8dee2aaSAndroid Build Coastguard Worker 
108*c8dee2aaSAndroid Build Coastguard Worker     if (context.fConfig->fSettings.fOptimize) {
109*c8dee2aaSAndroid Build Coastguard Worker         // Find ConstructorCompounds embedded inside other ConstructorCompounds and flatten them.
110*c8dee2aaSAndroid Build Coastguard Worker         //   -  float4(float2(1, 2), 3, 4)                -->  float4(1, 2, 3, 4)
111*c8dee2aaSAndroid Build Coastguard Worker         //   -  float4(w, float3(sin(x), cos(y), tan(z))) -->  float4(w, sin(x), cos(y), tan(z))
112*c8dee2aaSAndroid Build Coastguard Worker         //   -  mat2(float2(a, b), float2(c, d))          -->  mat2(a, b, c, d)
113*c8dee2aaSAndroid Build Coastguard Worker 
114*c8dee2aaSAndroid Build Coastguard Worker         // See how many fields we would have if composite constructors were flattened out.
115*c8dee2aaSAndroid Build Coastguard Worker         int fields = 0;
116*c8dee2aaSAndroid Build Coastguard Worker         for (const std::unique_ptr<Expression>& arg : args) {
117*c8dee2aaSAndroid Build Coastguard Worker             fields += arg->is<ConstructorCompound>()
118*c8dee2aaSAndroid Build Coastguard Worker                               ? arg->as<ConstructorCompound>().arguments().size()
119*c8dee2aaSAndroid Build Coastguard Worker                               : 1;
120*c8dee2aaSAndroid Build Coastguard Worker         }
121*c8dee2aaSAndroid Build Coastguard Worker 
122*c8dee2aaSAndroid Build Coastguard Worker         // If we added up more fields than we're starting with, we found at least one input that can
123*c8dee2aaSAndroid Build Coastguard Worker         // be flattened out.
124*c8dee2aaSAndroid Build Coastguard Worker         if (fields > args.size()) {
125*c8dee2aaSAndroid Build Coastguard Worker             ExpressionArray flattened;
126*c8dee2aaSAndroid Build Coastguard Worker             flattened.reserve_exact(fields);
127*c8dee2aaSAndroid Build Coastguard Worker             for (std::unique_ptr<Expression>& arg : args) {
128*c8dee2aaSAndroid Build Coastguard Worker                 // For non-ConstructorCompound fields, move them over as-is.
129*c8dee2aaSAndroid Build Coastguard Worker                 if (!arg->is<ConstructorCompound>()) {
130*c8dee2aaSAndroid Build Coastguard Worker                     flattened.push_back(std::move(arg));
131*c8dee2aaSAndroid Build Coastguard Worker                     continue;
132*c8dee2aaSAndroid Build Coastguard Worker                 }
133*c8dee2aaSAndroid Build Coastguard Worker                 // For ConstructorCompound fields, move over their inner arguments individually.
134*c8dee2aaSAndroid Build Coastguard Worker                 ConstructorCompound& compositeCtor = arg->as<ConstructorCompound>();
135*c8dee2aaSAndroid Build Coastguard Worker                 for (std::unique_ptr<Expression>& innerArg : compositeCtor.arguments()) {
136*c8dee2aaSAndroid Build Coastguard Worker                     flattened.push_back(std::move(innerArg));
137*c8dee2aaSAndroid Build Coastguard Worker                 }
138*c8dee2aaSAndroid Build Coastguard Worker             }
139*c8dee2aaSAndroid Build Coastguard Worker             args = std::move(flattened);
140*c8dee2aaSAndroid Build Coastguard Worker         }
141*c8dee2aaSAndroid Build Coastguard Worker     }
142*c8dee2aaSAndroid Build Coastguard Worker 
143*c8dee2aaSAndroid Build Coastguard Worker     // Replace constant variables with their corresponding values, so `float2(one, two)` can
144*c8dee2aaSAndroid Build Coastguard Worker     // compile down to `float2(1.0, 2.0)` (the latter is a compile-time constant).
145*c8dee2aaSAndroid Build Coastguard Worker     for (std::unique_ptr<Expression>& arg : args) {
146*c8dee2aaSAndroid Build Coastguard Worker         arg = ConstantFolder::MakeConstantValueForVariable(pos, std::move(arg));
147*c8dee2aaSAndroid Build Coastguard Worker     }
148*c8dee2aaSAndroid Build Coastguard Worker 
149*c8dee2aaSAndroid Build Coastguard Worker     if (context.fConfig->fSettings.fOptimize) {
150*c8dee2aaSAndroid Build Coastguard Worker         // Reduce compound constructors to splats where possible.
151*c8dee2aaSAndroid Build Coastguard Worker         if (const Expression* splat = make_splat_from_arguments(type, args)) {
152*c8dee2aaSAndroid Build Coastguard Worker             return ConstructorSplat::Make(context, pos, type, splat->clone());
153*c8dee2aaSAndroid Build Coastguard Worker         }
154*c8dee2aaSAndroid Build Coastguard Worker     }
155*c8dee2aaSAndroid Build Coastguard Worker 
156*c8dee2aaSAndroid Build Coastguard Worker     return std::make_unique<ConstructorCompound>(pos, type, std::move(args));
157*c8dee2aaSAndroid Build Coastguard Worker }
158*c8dee2aaSAndroid Build Coastguard Worker 
MakeFromConstants(const Context & context,Position pos,const Type & returnType,const double value[])159*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> ConstructorCompound::MakeFromConstants(const Context& context,
160*c8dee2aaSAndroid Build Coastguard Worker                                                                    Position pos,
161*c8dee2aaSAndroid Build Coastguard Worker                                                                    const Type& returnType,
162*c8dee2aaSAndroid Build Coastguard Worker                                                                    const double value[]) {
163*c8dee2aaSAndroid Build Coastguard Worker     int numSlots = returnType.slotCount();
164*c8dee2aaSAndroid Build Coastguard Worker     ExpressionArray array;
165*c8dee2aaSAndroid Build Coastguard Worker     array.reserve_exact(numSlots);
166*c8dee2aaSAndroid Build Coastguard Worker     for (int index = 0; index < numSlots; ++index) {
167*c8dee2aaSAndroid Build Coastguard Worker         array.push_back(Literal::Make(pos, value[index], &returnType.componentType()));
168*c8dee2aaSAndroid Build Coastguard Worker     }
169*c8dee2aaSAndroid Build Coastguard Worker     return ConstructorCompound::Make(context, pos, returnType, std::move(array));
170*c8dee2aaSAndroid Build Coastguard Worker }
171*c8dee2aaSAndroid Build Coastguard Worker 
172*c8dee2aaSAndroid Build Coastguard Worker }  // namespace SkSL
173