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