xref: /aosp_15_r20/external/skia/src/sksl/ir/SkSLIndexExpression.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/SkSLIndexExpression.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/SkSLBuiltinTypes.h"
14*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLConstantFolder.h"
15*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLContext.h"
16*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLDefines.h"
17*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLErrorReporter.h"
18*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLOperator.h"
19*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLConstructorArray.h"
20*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLConstructorCompound.h"
21*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLLiteral.h"
22*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLSwizzle.h"
23*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLSymbolTable.h"  // IWYU pragma: keep
24*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLType.h"
25*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLTypeReference.h"
26*c8dee2aaSAndroid Build Coastguard Worker 
27*c8dee2aaSAndroid Build Coastguard Worker #include <cstdint>
28*c8dee2aaSAndroid Build Coastguard Worker #include <optional>
29*c8dee2aaSAndroid Build Coastguard Worker 
30*c8dee2aaSAndroid Build Coastguard Worker namespace SkSL {
31*c8dee2aaSAndroid Build Coastguard Worker 
index_out_of_range(const Context & context,Position pos,SKSL_INT index,const Expression & base)32*c8dee2aaSAndroid Build Coastguard Worker static bool index_out_of_range(const Context& context, Position pos, SKSL_INT index,
33*c8dee2aaSAndroid Build Coastguard Worker         const Expression& base) {
34*c8dee2aaSAndroid Build Coastguard Worker     if (index >= 0) {
35*c8dee2aaSAndroid Build Coastguard Worker         if (base.type().columns() == Type::kUnsizedArray) {
36*c8dee2aaSAndroid Build Coastguard Worker             return false;
37*c8dee2aaSAndroid Build Coastguard Worker         } else if (index < base.type().columns()) {
38*c8dee2aaSAndroid Build Coastguard Worker             return false;
39*c8dee2aaSAndroid Build Coastguard Worker         }
40*c8dee2aaSAndroid Build Coastguard Worker     }
41*c8dee2aaSAndroid Build Coastguard Worker     context.fErrors->error(pos, "index " + std::to_string(index) + " out of range for '" +
42*c8dee2aaSAndroid Build Coastguard Worker                                 base.type().displayName() + "'");
43*c8dee2aaSAndroid Build Coastguard Worker     return true;
44*c8dee2aaSAndroid Build Coastguard Worker }
45*c8dee2aaSAndroid Build Coastguard Worker 
IndexType(const Context & context,const Type & type)46*c8dee2aaSAndroid Build Coastguard Worker const Type& IndexExpression::IndexType(const Context& context, const Type& type) {
47*c8dee2aaSAndroid Build Coastguard Worker     if (type.isMatrix()) {
48*c8dee2aaSAndroid Build Coastguard Worker         if (type.componentType().matches(*context.fTypes.fFloat)) {
49*c8dee2aaSAndroid Build Coastguard Worker             switch (type.rows()) {
50*c8dee2aaSAndroid Build Coastguard Worker                 case 2: return *context.fTypes.fFloat2;
51*c8dee2aaSAndroid Build Coastguard Worker                 case 3: return *context.fTypes.fFloat3;
52*c8dee2aaSAndroid Build Coastguard Worker                 case 4: return *context.fTypes.fFloat4;
53*c8dee2aaSAndroid Build Coastguard Worker                 default: SkASSERT(false);
54*c8dee2aaSAndroid Build Coastguard Worker             }
55*c8dee2aaSAndroid Build Coastguard Worker         } else if (type.componentType().matches(*context.fTypes.fHalf)) {
56*c8dee2aaSAndroid Build Coastguard Worker             switch (type.rows()) {
57*c8dee2aaSAndroid Build Coastguard Worker                 case 2: return *context.fTypes.fHalf2;
58*c8dee2aaSAndroid Build Coastguard Worker                 case 3: return *context.fTypes.fHalf3;
59*c8dee2aaSAndroid Build Coastguard Worker                 case 4: return *context.fTypes.fHalf4;
60*c8dee2aaSAndroid Build Coastguard Worker                 default: SkASSERT(false);
61*c8dee2aaSAndroid Build Coastguard Worker             }
62*c8dee2aaSAndroid Build Coastguard Worker         }
63*c8dee2aaSAndroid Build Coastguard Worker     }
64*c8dee2aaSAndroid Build Coastguard Worker     return type.componentType();
65*c8dee2aaSAndroid Build Coastguard Worker }
66*c8dee2aaSAndroid Build Coastguard Worker 
Convert(const Context & context,Position pos,std::unique_ptr<Expression> base,std::unique_ptr<Expression> index)67*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> IndexExpression::Convert(const Context& context,
68*c8dee2aaSAndroid Build Coastguard Worker                                                      Position pos,
69*c8dee2aaSAndroid Build Coastguard Worker                                                      std::unique_ptr<Expression> base,
70*c8dee2aaSAndroid Build Coastguard Worker                                                      std::unique_ptr<Expression> index) {
71*c8dee2aaSAndroid Build Coastguard Worker     // Convert an array type reference: `int[10]`.
72*c8dee2aaSAndroid Build Coastguard Worker     if (base->is<TypeReference>()) {
73*c8dee2aaSAndroid Build Coastguard Worker         const Type& baseType = base->as<TypeReference>().value();
74*c8dee2aaSAndroid Build Coastguard Worker         SKSL_INT arraySize = baseType.convertArraySize(context, pos, std::move(index));
75*c8dee2aaSAndroid Build Coastguard Worker         if (!arraySize) {
76*c8dee2aaSAndroid Build Coastguard Worker             return nullptr;
77*c8dee2aaSAndroid Build Coastguard Worker         }
78*c8dee2aaSAndroid Build Coastguard Worker         return TypeReference::Convert(
79*c8dee2aaSAndroid Build Coastguard Worker                 context,
80*c8dee2aaSAndroid Build Coastguard Worker                 pos,
81*c8dee2aaSAndroid Build Coastguard Worker                 context.fSymbolTable->addArrayDimension(context, &baseType, arraySize));
82*c8dee2aaSAndroid Build Coastguard Worker     }
83*c8dee2aaSAndroid Build Coastguard Worker     // Convert an index expression with an expression inside of it: `arr[a * 3]`.
84*c8dee2aaSAndroid Build Coastguard Worker     const Type& baseType = base->type();
85*c8dee2aaSAndroid Build Coastguard Worker     if (!baseType.isArray() && !baseType.isMatrix() && !baseType.isVector()) {
86*c8dee2aaSAndroid Build Coastguard Worker         context.fErrors->error(base->fPosition,
87*c8dee2aaSAndroid Build Coastguard Worker                                "expected array, but found '" + baseType.displayName() + "'");
88*c8dee2aaSAndroid Build Coastguard Worker         return nullptr;
89*c8dee2aaSAndroid Build Coastguard Worker     }
90*c8dee2aaSAndroid Build Coastguard Worker     if (!index->type().isInteger()) {
91*c8dee2aaSAndroid Build Coastguard Worker         index = context.fTypes.fInt->coerceExpression(std::move(index), context);
92*c8dee2aaSAndroid Build Coastguard Worker         if (!index) {
93*c8dee2aaSAndroid Build Coastguard Worker             return nullptr;
94*c8dee2aaSAndroid Build Coastguard Worker         }
95*c8dee2aaSAndroid Build Coastguard Worker     }
96*c8dee2aaSAndroid Build Coastguard Worker     // Perform compile-time bounds checking on constant-expression indices.
97*c8dee2aaSAndroid Build Coastguard Worker     const Expression* indexExpr = ConstantFolder::GetConstantValueForVariable(*index);
98*c8dee2aaSAndroid Build Coastguard Worker     if (indexExpr->isIntLiteral()) {
99*c8dee2aaSAndroid Build Coastguard Worker         SKSL_INT indexValue = indexExpr->as<Literal>().intValue();
100*c8dee2aaSAndroid Build Coastguard Worker         if (index_out_of_range(context, index->fPosition, indexValue, *base)) {
101*c8dee2aaSAndroid Build Coastguard Worker             return nullptr;
102*c8dee2aaSAndroid Build Coastguard Worker         }
103*c8dee2aaSAndroid Build Coastguard Worker     }
104*c8dee2aaSAndroid Build Coastguard Worker     return IndexExpression::Make(context, pos, std::move(base), std::move(index));
105*c8dee2aaSAndroid Build Coastguard Worker }
106*c8dee2aaSAndroid Build Coastguard Worker 
Make(const Context & context,Position pos,std::unique_ptr<Expression> base,std::unique_ptr<Expression> index)107*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> IndexExpression::Make(const Context& context,
108*c8dee2aaSAndroid Build Coastguard Worker                                                   Position pos,
109*c8dee2aaSAndroid Build Coastguard Worker                                                   std::unique_ptr<Expression> base,
110*c8dee2aaSAndroid Build Coastguard Worker                                                   std::unique_ptr<Expression> index) {
111*c8dee2aaSAndroid Build Coastguard Worker     const Type& baseType = base->type();
112*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(baseType.isArray() || baseType.isMatrix() || baseType.isVector());
113*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(index->type().isInteger());
114*c8dee2aaSAndroid Build Coastguard Worker 
115*c8dee2aaSAndroid Build Coastguard Worker     const Expression* indexExpr = ConstantFolder::GetConstantValueForVariable(*index);
116*c8dee2aaSAndroid Build Coastguard Worker     if (indexExpr->isIntLiteral()) {
117*c8dee2aaSAndroid Build Coastguard Worker         SKSL_INT indexValue = indexExpr->as<Literal>().intValue();
118*c8dee2aaSAndroid Build Coastguard Worker         if (!index_out_of_range(context, index->fPosition, indexValue, *base)) {
119*c8dee2aaSAndroid Build Coastguard Worker             if (baseType.isVector()) {
120*c8dee2aaSAndroid Build Coastguard Worker                 // Constant array indexes on vectors can be converted to swizzles: `v[2]` --> `v.z`.
121*c8dee2aaSAndroid Build Coastguard Worker                 // Swizzling is harmless and can unlock further simplifications for some base types.
122*c8dee2aaSAndroid Build Coastguard Worker                 return Swizzle::Make(context, pos, std::move(base),
123*c8dee2aaSAndroid Build Coastguard Worker                         ComponentArray{(int8_t)indexValue});
124*c8dee2aaSAndroid Build Coastguard Worker             }
125*c8dee2aaSAndroid Build Coastguard Worker 
126*c8dee2aaSAndroid Build Coastguard Worker             if (baseType.isArray() && !Analysis::HasSideEffects(*base)) {
127*c8dee2aaSAndroid Build Coastguard Worker                 // Indexing an constant array constructor with a constant index can just pluck out
128*c8dee2aaSAndroid Build Coastguard Worker                 // the requested value from the array.
129*c8dee2aaSAndroid Build Coastguard Worker                 const Expression* baseExpr = ConstantFolder::GetConstantValueForVariable(*base);
130*c8dee2aaSAndroid Build Coastguard Worker                 if (baseExpr->is<ConstructorArray>()) {
131*c8dee2aaSAndroid Build Coastguard Worker                     const ConstructorArray& arrayCtor = baseExpr->as<ConstructorArray>();
132*c8dee2aaSAndroid Build Coastguard Worker                     const ExpressionArray& arguments = arrayCtor.arguments();
133*c8dee2aaSAndroid Build Coastguard Worker                     SkASSERT(arguments.size() == baseType.columns());
134*c8dee2aaSAndroid Build Coastguard Worker 
135*c8dee2aaSAndroid Build Coastguard Worker                     return arguments[indexValue]->clone(pos);
136*c8dee2aaSAndroid Build Coastguard Worker                 }
137*c8dee2aaSAndroid Build Coastguard Worker             }
138*c8dee2aaSAndroid Build Coastguard Worker 
139*c8dee2aaSAndroid Build Coastguard Worker             if (baseType.isMatrix() && !Analysis::HasSideEffects(*base)) {
140*c8dee2aaSAndroid Build Coastguard Worker                 // Matrices can be constructed with vectors that don't line up on column boundaries,
141*c8dee2aaSAndroid Build Coastguard Worker                 // so extracting out the values from the constructor can be tricky. Fortunately, we
142*c8dee2aaSAndroid Build Coastguard Worker                 // can reconstruct an equivalent vector using `getConstantValue`. If we
143*c8dee2aaSAndroid Build Coastguard Worker                 // can't extract the data using `getConstantValue`, it wasn't constant and
144*c8dee2aaSAndroid Build Coastguard Worker                 // we're not obligated to simplify anything.
145*c8dee2aaSAndroid Build Coastguard Worker                 const Expression* baseExpr = ConstantFolder::GetConstantValueForVariable(*base);
146*c8dee2aaSAndroid Build Coastguard Worker                 int vecWidth = baseType.rows();
147*c8dee2aaSAndroid Build Coastguard Worker                 const Type& vecType = baseType.columnType(context);
148*c8dee2aaSAndroid Build Coastguard Worker                 indexValue *= vecWidth;
149*c8dee2aaSAndroid Build Coastguard Worker 
150*c8dee2aaSAndroid Build Coastguard Worker                 double ctorArgs[4];
151*c8dee2aaSAndroid Build Coastguard Worker                 bool allConstant = true;
152*c8dee2aaSAndroid Build Coastguard Worker                 for (int slot = 0; slot < vecWidth; ++slot) {
153*c8dee2aaSAndroid Build Coastguard Worker                     std::optional<double> slotVal = baseExpr->getConstantValue(indexValue + slot);
154*c8dee2aaSAndroid Build Coastguard Worker                     if (slotVal.has_value()) {
155*c8dee2aaSAndroid Build Coastguard Worker                         ctorArgs[slot] = *slotVal;
156*c8dee2aaSAndroid Build Coastguard Worker                     } else {
157*c8dee2aaSAndroid Build Coastguard Worker                         allConstant = false;
158*c8dee2aaSAndroid Build Coastguard Worker                         break;
159*c8dee2aaSAndroid Build Coastguard Worker                     }
160*c8dee2aaSAndroid Build Coastguard Worker                 }
161*c8dee2aaSAndroid Build Coastguard Worker 
162*c8dee2aaSAndroid Build Coastguard Worker                 if (allConstant) {
163*c8dee2aaSAndroid Build Coastguard Worker                     return ConstructorCompound::MakeFromConstants(context, pos, vecType, ctorArgs);
164*c8dee2aaSAndroid Build Coastguard Worker                 }
165*c8dee2aaSAndroid Build Coastguard Worker             }
166*c8dee2aaSAndroid Build Coastguard Worker         }
167*c8dee2aaSAndroid Build Coastguard Worker     }
168*c8dee2aaSAndroid Build Coastguard Worker 
169*c8dee2aaSAndroid Build Coastguard Worker     return std::make_unique<IndexExpression>(context, pos, std::move(base), std::move(index));
170*c8dee2aaSAndroid Build Coastguard Worker }
171*c8dee2aaSAndroid Build Coastguard Worker 
description(OperatorPrecedence) const172*c8dee2aaSAndroid Build Coastguard Worker std::string IndexExpression::description(OperatorPrecedence) const {
173*c8dee2aaSAndroid Build Coastguard Worker     return this->base()->description(OperatorPrecedence::kPostfix) + "[" +
174*c8dee2aaSAndroid Build Coastguard Worker            this->index()->description(OperatorPrecedence::kExpression) + "]";
175*c8dee2aaSAndroid Build Coastguard Worker }
176*c8dee2aaSAndroid Build Coastguard Worker 
177*c8dee2aaSAndroid Build Coastguard Worker }  // namespace SkSL
178