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