xref: /aosp_15_r20/external/skia/src/sksl/ir/SkSLPrefixExpression.cpp (revision c8dee2aa9b3f27cf6c858bd81872bdeb2c07ed17)
1*c8dee2aaSAndroid Build Coastguard Worker /*
2*c8dee2aaSAndroid Build Coastguard Worker  * Copyright 2020 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/SkSLPrefixExpression.h"
9*c8dee2aaSAndroid Build Coastguard Worker 
10*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkTypes.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLAnalysis.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLConstantFolder.h"
13*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLContext.h"
14*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLDefines.h"
15*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLErrorReporter.h"
16*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLOperator.h"
17*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLProgramSettings.h"
18*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLBinaryExpression.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/SkSLConstructorDiagonalMatrix.h"
22*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLLiteral.h"
23*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLType.h"
24*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLVariableReference.h"
25*c8dee2aaSAndroid Build Coastguard Worker 
26*c8dee2aaSAndroid Build Coastguard Worker #include <cstddef>
27*c8dee2aaSAndroid Build Coastguard Worker #include <optional>
28*c8dee2aaSAndroid Build Coastguard Worker 
29*c8dee2aaSAndroid Build Coastguard Worker namespace SkSL {
30*c8dee2aaSAndroid Build Coastguard Worker 
31*c8dee2aaSAndroid Build Coastguard Worker static ExpressionArray negate_operands(const Context& context,
32*c8dee2aaSAndroid Build Coastguard Worker                                        Position pos,
33*c8dee2aaSAndroid Build Coastguard Worker                                        const ExpressionArray& operands);
34*c8dee2aaSAndroid Build Coastguard Worker 
negate_value(double value)35*c8dee2aaSAndroid Build Coastguard Worker static double negate_value(double value) {
36*c8dee2aaSAndroid Build Coastguard Worker     return -value;
37*c8dee2aaSAndroid Build Coastguard Worker }
38*c8dee2aaSAndroid Build Coastguard Worker 
bitwise_not_value(double value)39*c8dee2aaSAndroid Build Coastguard Worker static double bitwise_not_value(double value) {
40*c8dee2aaSAndroid Build Coastguard Worker     return ~static_cast<SKSL_INT>(value);
41*c8dee2aaSAndroid Build Coastguard Worker }
42*c8dee2aaSAndroid Build Coastguard Worker 
apply_to_elements(const Context & context,Position pos,const Expression & expr,double (* fn)(double))43*c8dee2aaSAndroid Build Coastguard Worker static std::unique_ptr<Expression> apply_to_elements(const Context& context,
44*c8dee2aaSAndroid Build Coastguard Worker                                                      Position pos,
45*c8dee2aaSAndroid Build Coastguard Worker                                                      const Expression& expr,
46*c8dee2aaSAndroid Build Coastguard Worker                                                      double (*fn)(double)) {
47*c8dee2aaSAndroid Build Coastguard Worker     const Type& elementType = expr.type().componentType();
48*c8dee2aaSAndroid Build Coastguard Worker 
49*c8dee2aaSAndroid Build Coastguard Worker     double values[16];
50*c8dee2aaSAndroid Build Coastguard Worker     size_t numSlots = expr.type().slotCount();
51*c8dee2aaSAndroid Build Coastguard Worker     if (numSlots > std::size(values)) {
52*c8dee2aaSAndroid Build Coastguard Worker         // The expression has more slots than we expected.
53*c8dee2aaSAndroid Build Coastguard Worker         return nullptr;
54*c8dee2aaSAndroid Build Coastguard Worker     }
55*c8dee2aaSAndroid Build Coastguard Worker 
56*c8dee2aaSAndroid Build Coastguard Worker     for (size_t index = 0; index < numSlots; ++index) {
57*c8dee2aaSAndroid Build Coastguard Worker         if (std::optional<double> slotValue = expr.getConstantValue(index)) {
58*c8dee2aaSAndroid Build Coastguard Worker             values[index] = fn(*slotValue);
59*c8dee2aaSAndroid Build Coastguard Worker             if (elementType.checkForOutOfRangeLiteral(context, values[index], pos)) {
60*c8dee2aaSAndroid Build Coastguard Worker                 // We can't simplify the expression if the new value is out-of-range for the type.
61*c8dee2aaSAndroid Build Coastguard Worker                 return nullptr;
62*c8dee2aaSAndroid Build Coastguard Worker             }
63*c8dee2aaSAndroid Build Coastguard Worker         } else {
64*c8dee2aaSAndroid Build Coastguard Worker             // There's a non-constant element; we can't simplify this expression.
65*c8dee2aaSAndroid Build Coastguard Worker             return nullptr;
66*c8dee2aaSAndroid Build Coastguard Worker         }
67*c8dee2aaSAndroid Build Coastguard Worker     }
68*c8dee2aaSAndroid Build Coastguard Worker     return ConstructorCompound::MakeFromConstants(context, pos, expr.type(), values);
69*c8dee2aaSAndroid Build Coastguard Worker }
70*c8dee2aaSAndroid Build Coastguard Worker 
simplify_negation(const Context & context,Position pos,const Expression & originalExpr)71*c8dee2aaSAndroid Build Coastguard Worker static std::unique_ptr<Expression> simplify_negation(const Context& context,
72*c8dee2aaSAndroid Build Coastguard Worker                                                      Position pos,
73*c8dee2aaSAndroid Build Coastguard Worker                                                      const Expression& originalExpr) {
74*c8dee2aaSAndroid Build Coastguard Worker     const Expression* value = ConstantFolder::GetConstantValueForVariable(originalExpr);
75*c8dee2aaSAndroid Build Coastguard Worker     switch (value->kind()) {
76*c8dee2aaSAndroid Build Coastguard Worker         case Expression::Kind::kLiteral:
77*c8dee2aaSAndroid Build Coastguard Worker         case Expression::Kind::kConstructorSplat:
78*c8dee2aaSAndroid Build Coastguard Worker         case Expression::Kind::kConstructorCompound: {
79*c8dee2aaSAndroid Build Coastguard Worker             // Convert `-vecN(literal, ...)` into `vecN(-literal, ...)`.
80*c8dee2aaSAndroid Build Coastguard Worker             if (std::unique_ptr<Expression> expr = apply_to_elements(context, pos, *value,
81*c8dee2aaSAndroid Build Coastguard Worker                                                                      negate_value)) {
82*c8dee2aaSAndroid Build Coastguard Worker                 return expr;
83*c8dee2aaSAndroid Build Coastguard Worker             }
84*c8dee2aaSAndroid Build Coastguard Worker             break;
85*c8dee2aaSAndroid Build Coastguard Worker         }
86*c8dee2aaSAndroid Build Coastguard Worker         case Expression::Kind::kPrefix: {
87*c8dee2aaSAndroid Build Coastguard Worker             // Convert `-(-expression)` into `expression`.
88*c8dee2aaSAndroid Build Coastguard Worker             const PrefixExpression& prefix = value->as<PrefixExpression>();
89*c8dee2aaSAndroid Build Coastguard Worker             if (prefix.getOperator().kind() == Operator::Kind::MINUS) {
90*c8dee2aaSAndroid Build Coastguard Worker                 return prefix.operand()->clone(pos);
91*c8dee2aaSAndroid Build Coastguard Worker             }
92*c8dee2aaSAndroid Build Coastguard Worker             break;
93*c8dee2aaSAndroid Build Coastguard Worker         }
94*c8dee2aaSAndroid Build Coastguard Worker         case Expression::Kind::kConstructorArray:
95*c8dee2aaSAndroid Build Coastguard Worker             // Convert `-array[N](literal, ...)` into `array[N](-literal, ...)`.
96*c8dee2aaSAndroid Build Coastguard Worker             if (Analysis::IsCompileTimeConstant(*value)) {
97*c8dee2aaSAndroid Build Coastguard Worker                 const ConstructorArray& ctor = value->as<ConstructorArray>();
98*c8dee2aaSAndroid Build Coastguard Worker                 return ConstructorArray::Make(context, pos, ctor.type(),
99*c8dee2aaSAndroid Build Coastguard Worker                                               negate_operands(context, pos, ctor.arguments()));
100*c8dee2aaSAndroid Build Coastguard Worker             }
101*c8dee2aaSAndroid Build Coastguard Worker             break;
102*c8dee2aaSAndroid Build Coastguard Worker 
103*c8dee2aaSAndroid Build Coastguard Worker         case Expression::Kind::kConstructorDiagonalMatrix:
104*c8dee2aaSAndroid Build Coastguard Worker             // Convert `-matrix(literal)` into `matrix(-literal)`.
105*c8dee2aaSAndroid Build Coastguard Worker             if (Analysis::IsCompileTimeConstant(*value)) {
106*c8dee2aaSAndroid Build Coastguard Worker                 const ConstructorDiagonalMatrix& ctor = value->as<ConstructorDiagonalMatrix>();
107*c8dee2aaSAndroid Build Coastguard Worker                 if (std::unique_ptr<Expression> simplified = simplify_negation(context,
108*c8dee2aaSAndroid Build Coastguard Worker                                                                                pos,
109*c8dee2aaSAndroid Build Coastguard Worker                                                                                *ctor.argument())) {
110*c8dee2aaSAndroid Build Coastguard Worker                     return ConstructorDiagonalMatrix::Make(context, pos, ctor.type(),
111*c8dee2aaSAndroid Build Coastguard Worker                                                            std::move(simplified));
112*c8dee2aaSAndroid Build Coastguard Worker                 }
113*c8dee2aaSAndroid Build Coastguard Worker             }
114*c8dee2aaSAndroid Build Coastguard Worker             break;
115*c8dee2aaSAndroid Build Coastguard Worker 
116*c8dee2aaSAndroid Build Coastguard Worker         default:
117*c8dee2aaSAndroid Build Coastguard Worker             break;
118*c8dee2aaSAndroid Build Coastguard Worker     }
119*c8dee2aaSAndroid Build Coastguard Worker     return nullptr;
120*c8dee2aaSAndroid Build Coastguard Worker }
121*c8dee2aaSAndroid Build Coastguard Worker 
negate_operands(const Context & context,Position pos,const ExpressionArray & array)122*c8dee2aaSAndroid Build Coastguard Worker static ExpressionArray negate_operands(const Context& context,
123*c8dee2aaSAndroid Build Coastguard Worker                                        Position pos,
124*c8dee2aaSAndroid Build Coastguard Worker                                        const ExpressionArray& array) {
125*c8dee2aaSAndroid Build Coastguard Worker     ExpressionArray replacement;
126*c8dee2aaSAndroid Build Coastguard Worker     replacement.reserve_exact(array.size());
127*c8dee2aaSAndroid Build Coastguard Worker     for (const std::unique_ptr<Expression>& expr : array) {
128*c8dee2aaSAndroid Build Coastguard Worker         // The logic below is very similar to `negate_operand`, but with different ownership rules.
129*c8dee2aaSAndroid Build Coastguard Worker         if (std::unique_ptr<Expression> simplified = simplify_negation(context, pos, *expr)) {
130*c8dee2aaSAndroid Build Coastguard Worker             replacement.push_back(std::move(simplified));
131*c8dee2aaSAndroid Build Coastguard Worker         } else {
132*c8dee2aaSAndroid Build Coastguard Worker             replacement.push_back(std::make_unique<PrefixExpression>(pos, Operator::Kind::MINUS,
133*c8dee2aaSAndroid Build Coastguard Worker                                                                      expr->clone()));
134*c8dee2aaSAndroid Build Coastguard Worker         }
135*c8dee2aaSAndroid Build Coastguard Worker     }
136*c8dee2aaSAndroid Build Coastguard Worker     return replacement;
137*c8dee2aaSAndroid Build Coastguard Worker }
138*c8dee2aaSAndroid Build Coastguard Worker 
negate_operand(const Context & context,Position pos,std::unique_ptr<Expression> value)139*c8dee2aaSAndroid Build Coastguard Worker static std::unique_ptr<Expression> negate_operand(const Context& context,
140*c8dee2aaSAndroid Build Coastguard Worker                                                   Position pos,
141*c8dee2aaSAndroid Build Coastguard Worker                                                   std::unique_ptr<Expression> value) {
142*c8dee2aaSAndroid Build Coastguard Worker     // Attempt to simplify this negation (e.g. eliminate double negation, literal values)
143*c8dee2aaSAndroid Build Coastguard Worker     if (std::unique_ptr<Expression> simplified = simplify_negation(context, pos, *value)) {
144*c8dee2aaSAndroid Build Coastguard Worker         return simplified;
145*c8dee2aaSAndroid Build Coastguard Worker     }
146*c8dee2aaSAndroid Build Coastguard Worker 
147*c8dee2aaSAndroid Build Coastguard Worker     // No simplified form; convert expression to Prefix(MINUS, expression).
148*c8dee2aaSAndroid Build Coastguard Worker     return std::make_unique<PrefixExpression>(pos, Operator::Kind::MINUS, std::move(value));
149*c8dee2aaSAndroid Build Coastguard Worker }
150*c8dee2aaSAndroid Build Coastguard Worker 
logical_not_operand(const Context & context,Position pos,std::unique_ptr<Expression> operand)151*c8dee2aaSAndroid Build Coastguard Worker static std::unique_ptr<Expression> logical_not_operand(const Context& context,
152*c8dee2aaSAndroid Build Coastguard Worker                                                        Position pos,
153*c8dee2aaSAndroid Build Coastguard Worker                                                        std::unique_ptr<Expression> operand) {
154*c8dee2aaSAndroid Build Coastguard Worker     const Expression* value = ConstantFolder::GetConstantValueForVariable(*operand);
155*c8dee2aaSAndroid Build Coastguard Worker     switch (value->kind()) {
156*c8dee2aaSAndroid Build Coastguard Worker         case Expression::Kind::kLiteral: {
157*c8dee2aaSAndroid Build Coastguard Worker             // Convert !boolLiteral(true) to boolLiteral(false).
158*c8dee2aaSAndroid Build Coastguard Worker             SkASSERT(value->type().isBoolean());
159*c8dee2aaSAndroid Build Coastguard Worker             const Literal& b = value->as<Literal>();
160*c8dee2aaSAndroid Build Coastguard Worker             return Literal::MakeBool(pos, !b.boolValue(), &operand->type());
161*c8dee2aaSAndroid Build Coastguard Worker         }
162*c8dee2aaSAndroid Build Coastguard Worker         case Expression::Kind::kPrefix: {
163*c8dee2aaSAndroid Build Coastguard Worker             // Convert `!(!expression)` into `expression`.
164*c8dee2aaSAndroid Build Coastguard Worker             PrefixExpression& prefix = operand->as<PrefixExpression>();
165*c8dee2aaSAndroid Build Coastguard Worker             if (prefix.getOperator().kind() == Operator::Kind::LOGICALNOT) {
166*c8dee2aaSAndroid Build Coastguard Worker                 prefix.operand()->fPosition = pos;
167*c8dee2aaSAndroid Build Coastguard Worker                 return std::move(prefix.operand());
168*c8dee2aaSAndroid Build Coastguard Worker             }
169*c8dee2aaSAndroid Build Coastguard Worker             break;
170*c8dee2aaSAndroid Build Coastguard Worker         }
171*c8dee2aaSAndroid Build Coastguard Worker         case Expression::Kind::kBinary: {
172*c8dee2aaSAndroid Build Coastguard Worker             BinaryExpression& binary = operand->as<BinaryExpression>();
173*c8dee2aaSAndroid Build Coastguard Worker             std::optional<Operator> replacement;
174*c8dee2aaSAndroid Build Coastguard Worker             switch (binary.getOperator().kind()) {
175*c8dee2aaSAndroid Build Coastguard Worker                 case OperatorKind::EQEQ: replacement = OperatorKind::NEQ;  break;
176*c8dee2aaSAndroid Build Coastguard Worker                 case OperatorKind::NEQ:  replacement = OperatorKind::EQEQ; break;
177*c8dee2aaSAndroid Build Coastguard Worker                 case OperatorKind::LT:   replacement = OperatorKind::GTEQ; break;
178*c8dee2aaSAndroid Build Coastguard Worker                 case OperatorKind::LTEQ: replacement = OperatorKind::GT;   break;
179*c8dee2aaSAndroid Build Coastguard Worker                 case OperatorKind::GT:   replacement = OperatorKind::LTEQ; break;
180*c8dee2aaSAndroid Build Coastguard Worker                 case OperatorKind::GTEQ: replacement = OperatorKind::LT;   break;
181*c8dee2aaSAndroid Build Coastguard Worker                 default:                                                   break;
182*c8dee2aaSAndroid Build Coastguard Worker             }
183*c8dee2aaSAndroid Build Coastguard Worker             if (replacement.has_value()) {
184*c8dee2aaSAndroid Build Coastguard Worker                 return BinaryExpression::Make(context, pos, std::move(binary.left()),
185*c8dee2aaSAndroid Build Coastguard Worker                                               *replacement, std::move(binary.right()),
186*c8dee2aaSAndroid Build Coastguard Worker                                               &binary.type());
187*c8dee2aaSAndroid Build Coastguard Worker             }
188*c8dee2aaSAndroid Build Coastguard Worker             break;
189*c8dee2aaSAndroid Build Coastguard Worker         }
190*c8dee2aaSAndroid Build Coastguard Worker         default:
191*c8dee2aaSAndroid Build Coastguard Worker             break;
192*c8dee2aaSAndroid Build Coastguard Worker     }
193*c8dee2aaSAndroid Build Coastguard Worker 
194*c8dee2aaSAndroid Build Coastguard Worker     // No simplified form; convert expression to Prefix(LOGICALNOT, expression).
195*c8dee2aaSAndroid Build Coastguard Worker     return std::make_unique<PrefixExpression>(pos, Operator::Kind::LOGICALNOT, std::move(operand));
196*c8dee2aaSAndroid Build Coastguard Worker }
197*c8dee2aaSAndroid Build Coastguard Worker 
bitwise_not_operand(const Context & context,Position pos,std::unique_ptr<Expression> operand)198*c8dee2aaSAndroid Build Coastguard Worker static std::unique_ptr<Expression> bitwise_not_operand(const Context& context,
199*c8dee2aaSAndroid Build Coastguard Worker                                                        Position pos,
200*c8dee2aaSAndroid Build Coastguard Worker                                                        std::unique_ptr<Expression> operand) {
201*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(operand->type().componentType().isInteger());
202*c8dee2aaSAndroid Build Coastguard Worker 
203*c8dee2aaSAndroid Build Coastguard Worker     const Expression* value = ConstantFolder::GetConstantValueForVariable(*operand);
204*c8dee2aaSAndroid Build Coastguard Worker 
205*c8dee2aaSAndroid Build Coastguard Worker     switch (value->kind()) {
206*c8dee2aaSAndroid Build Coastguard Worker         case Expression::Kind::kLiteral:
207*c8dee2aaSAndroid Build Coastguard Worker         case Expression::Kind::kConstructorSplat:
208*c8dee2aaSAndroid Build Coastguard Worker         case Expression::Kind::kConstructorCompound: {
209*c8dee2aaSAndroid Build Coastguard Worker             // Convert ~vecN(1, 2, ...) to vecN(~1, ~2, ...).
210*c8dee2aaSAndroid Build Coastguard Worker             if (std::unique_ptr<Expression> expr = apply_to_elements(context, pos, *value,
211*c8dee2aaSAndroid Build Coastguard Worker                                                                      bitwise_not_value)) {
212*c8dee2aaSAndroid Build Coastguard Worker                 return expr;
213*c8dee2aaSAndroid Build Coastguard Worker             }
214*c8dee2aaSAndroid Build Coastguard Worker             break;
215*c8dee2aaSAndroid Build Coastguard Worker         }
216*c8dee2aaSAndroid Build Coastguard Worker         case Expression::Kind::kPrefix: {
217*c8dee2aaSAndroid Build Coastguard Worker             // Convert `~(~expression)` into `expression`.
218*c8dee2aaSAndroid Build Coastguard Worker             PrefixExpression& prefix = operand->as<PrefixExpression>();
219*c8dee2aaSAndroid Build Coastguard Worker             if (prefix.getOperator().kind() == Operator::Kind::BITWISENOT) {
220*c8dee2aaSAndroid Build Coastguard Worker                 prefix.operand()->fPosition = pos;
221*c8dee2aaSAndroid Build Coastguard Worker                 return std::move(prefix.operand());
222*c8dee2aaSAndroid Build Coastguard Worker             }
223*c8dee2aaSAndroid Build Coastguard Worker             break;
224*c8dee2aaSAndroid Build Coastguard Worker         }
225*c8dee2aaSAndroid Build Coastguard Worker         default:
226*c8dee2aaSAndroid Build Coastguard Worker             break;
227*c8dee2aaSAndroid Build Coastguard Worker     }
228*c8dee2aaSAndroid Build Coastguard Worker 
229*c8dee2aaSAndroid Build Coastguard Worker     // No simplified form; convert expression to Prefix(BITWISENOT, expression).
230*c8dee2aaSAndroid Build Coastguard Worker     return std::make_unique<PrefixExpression>(pos, Operator::Kind::BITWISENOT, std::move(operand));
231*c8dee2aaSAndroid Build Coastguard Worker }
232*c8dee2aaSAndroid Build Coastguard Worker 
Convert(const Context & context,Position pos,Operator op,std::unique_ptr<Expression> base)233*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> PrefixExpression::Convert(const Context& context,
234*c8dee2aaSAndroid Build Coastguard Worker                                                       Position pos,
235*c8dee2aaSAndroid Build Coastguard Worker                                                       Operator op,
236*c8dee2aaSAndroid Build Coastguard Worker                                                       std::unique_ptr<Expression> base) {
237*c8dee2aaSAndroid Build Coastguard Worker     const Type& baseType = base->type();
238*c8dee2aaSAndroid Build Coastguard Worker     switch (op.kind()) {
239*c8dee2aaSAndroid Build Coastguard Worker         case Operator::Kind::PLUS:
240*c8dee2aaSAndroid Build Coastguard Worker             if (baseType.isArray() || !baseType.componentType().isNumber()) {
241*c8dee2aaSAndroid Build Coastguard Worker                 context.fErrors->error(pos,
242*c8dee2aaSAndroid Build Coastguard Worker                                        "'+' cannot operate on '" + baseType.displayName() + "'");
243*c8dee2aaSAndroid Build Coastguard Worker                 return nullptr;
244*c8dee2aaSAndroid Build Coastguard Worker             }
245*c8dee2aaSAndroid Build Coastguard Worker             break;
246*c8dee2aaSAndroid Build Coastguard Worker 
247*c8dee2aaSAndroid Build Coastguard Worker         case Operator::Kind::MINUS:
248*c8dee2aaSAndroid Build Coastguard Worker             if (baseType.isArray() || !baseType.componentType().isNumber()) {
249*c8dee2aaSAndroid Build Coastguard Worker                 context.fErrors->error(pos,
250*c8dee2aaSAndroid Build Coastguard Worker                                        "'-' cannot operate on '" + baseType.displayName() + "'");
251*c8dee2aaSAndroid Build Coastguard Worker                 return nullptr;
252*c8dee2aaSAndroid Build Coastguard Worker             }
253*c8dee2aaSAndroid Build Coastguard Worker             break;
254*c8dee2aaSAndroid Build Coastguard Worker 
255*c8dee2aaSAndroid Build Coastguard Worker         case Operator::Kind::PLUSPLUS:
256*c8dee2aaSAndroid Build Coastguard Worker         case Operator::Kind::MINUSMINUS:
257*c8dee2aaSAndroid Build Coastguard Worker             if (baseType.isArray() || !baseType.componentType().isNumber()) {
258*c8dee2aaSAndroid Build Coastguard Worker                 context.fErrors->error(pos,
259*c8dee2aaSAndroid Build Coastguard Worker                                        "'" + std::string(op.tightOperatorName()) +
260*c8dee2aaSAndroid Build Coastguard Worker                                        "' cannot operate on '" + baseType.displayName() + "'");
261*c8dee2aaSAndroid Build Coastguard Worker                 return nullptr;
262*c8dee2aaSAndroid Build Coastguard Worker             }
263*c8dee2aaSAndroid Build Coastguard Worker             if (!Analysis::UpdateVariableRefKind(base.get(), VariableReference::RefKind::kReadWrite,
264*c8dee2aaSAndroid Build Coastguard Worker                                                  context.fErrors)) {
265*c8dee2aaSAndroid Build Coastguard Worker                 return nullptr;
266*c8dee2aaSAndroid Build Coastguard Worker             }
267*c8dee2aaSAndroid Build Coastguard Worker             break;
268*c8dee2aaSAndroid Build Coastguard Worker 
269*c8dee2aaSAndroid Build Coastguard Worker         case Operator::Kind::LOGICALNOT:
270*c8dee2aaSAndroid Build Coastguard Worker             if (!baseType.isBoolean()) {
271*c8dee2aaSAndroid Build Coastguard Worker                 context.fErrors->error(pos,
272*c8dee2aaSAndroid Build Coastguard Worker                                        "'" + std::string(op.tightOperatorName()) +
273*c8dee2aaSAndroid Build Coastguard Worker                                        "' cannot operate on '" + baseType.displayName() + "'");
274*c8dee2aaSAndroid Build Coastguard Worker                 return nullptr;
275*c8dee2aaSAndroid Build Coastguard Worker             }
276*c8dee2aaSAndroid Build Coastguard Worker             break;
277*c8dee2aaSAndroid Build Coastguard Worker 
278*c8dee2aaSAndroid Build Coastguard Worker         case Operator::Kind::BITWISENOT:
279*c8dee2aaSAndroid Build Coastguard Worker             if (context.fConfig->strictES2Mode()) {
280*c8dee2aaSAndroid Build Coastguard Worker                 // GLSL ES 1.00, Section 5.1
281*c8dee2aaSAndroid Build Coastguard Worker                 context.fErrors->error(
282*c8dee2aaSAndroid Build Coastguard Worker                         pos,
283*c8dee2aaSAndroid Build Coastguard Worker                         "operator '" + std::string(op.tightOperatorName()) + "' is not allowed");
284*c8dee2aaSAndroid Build Coastguard Worker                 return nullptr;
285*c8dee2aaSAndroid Build Coastguard Worker             }
286*c8dee2aaSAndroid Build Coastguard Worker             if (baseType.isArray() || !baseType.componentType().isInteger()) {
287*c8dee2aaSAndroid Build Coastguard Worker                 context.fErrors->error(pos,
288*c8dee2aaSAndroid Build Coastguard Worker                                        "'" + std::string(op.tightOperatorName()) +
289*c8dee2aaSAndroid Build Coastguard Worker                                        "' cannot operate on '" + baseType.displayName() + "'");
290*c8dee2aaSAndroid Build Coastguard Worker                 return nullptr;
291*c8dee2aaSAndroid Build Coastguard Worker             }
292*c8dee2aaSAndroid Build Coastguard Worker             break;
293*c8dee2aaSAndroid Build Coastguard Worker 
294*c8dee2aaSAndroid Build Coastguard Worker         default:
295*c8dee2aaSAndroid Build Coastguard Worker             SK_ABORT("unsupported prefix operator");
296*c8dee2aaSAndroid Build Coastguard Worker     }
297*c8dee2aaSAndroid Build Coastguard Worker 
298*c8dee2aaSAndroid Build Coastguard Worker     std::unique_ptr<Expression> result = PrefixExpression::Make(context, pos, op, std::move(base));
299*c8dee2aaSAndroid Build Coastguard Worker     SkASSERT(result->fPosition == pos);
300*c8dee2aaSAndroid Build Coastguard Worker     return result;
301*c8dee2aaSAndroid Build Coastguard Worker }
302*c8dee2aaSAndroid Build Coastguard Worker 
Make(const Context & context,Position pos,Operator op,std::unique_ptr<Expression> base)303*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> PrefixExpression::Make(const Context& context,
304*c8dee2aaSAndroid Build Coastguard Worker                                                    Position pos,
305*c8dee2aaSAndroid Build Coastguard Worker                                                    Operator op,
306*c8dee2aaSAndroid Build Coastguard Worker                                                    std::unique_ptr<Expression> base) {
307*c8dee2aaSAndroid Build Coastguard Worker     const Type& baseType = base->type();
308*c8dee2aaSAndroid Build Coastguard Worker     switch (op.kind()) {
309*c8dee2aaSAndroid Build Coastguard Worker         case Operator::Kind::PLUS:
310*c8dee2aaSAndroid Build Coastguard Worker             SkASSERT(!baseType.isArray());
311*c8dee2aaSAndroid Build Coastguard Worker             SkASSERT(baseType.componentType().isNumber());
312*c8dee2aaSAndroid Build Coastguard Worker             base->fPosition = pos;
313*c8dee2aaSAndroid Build Coastguard Worker             return base;
314*c8dee2aaSAndroid Build Coastguard Worker 
315*c8dee2aaSAndroid Build Coastguard Worker         case Operator::Kind::MINUS:
316*c8dee2aaSAndroid Build Coastguard Worker             SkASSERT(!baseType.isArray());
317*c8dee2aaSAndroid Build Coastguard Worker             SkASSERT(baseType.componentType().isNumber());
318*c8dee2aaSAndroid Build Coastguard Worker             return negate_operand(context, pos, std::move(base));
319*c8dee2aaSAndroid Build Coastguard Worker 
320*c8dee2aaSAndroid Build Coastguard Worker         case Operator::Kind::LOGICALNOT:
321*c8dee2aaSAndroid Build Coastguard Worker             SkASSERT(baseType.isBoolean());
322*c8dee2aaSAndroid Build Coastguard Worker             return logical_not_operand(context, pos, std::move(base));
323*c8dee2aaSAndroid Build Coastguard Worker 
324*c8dee2aaSAndroid Build Coastguard Worker         case Operator::Kind::PLUSPLUS:
325*c8dee2aaSAndroid Build Coastguard Worker         case Operator::Kind::MINUSMINUS:
326*c8dee2aaSAndroid Build Coastguard Worker             SkASSERT(!baseType.isArray());
327*c8dee2aaSAndroid Build Coastguard Worker             SkASSERT(baseType.componentType().isNumber());
328*c8dee2aaSAndroid Build Coastguard Worker             SkASSERT(Analysis::IsAssignable(*base));
329*c8dee2aaSAndroid Build Coastguard Worker             break;
330*c8dee2aaSAndroid Build Coastguard Worker 
331*c8dee2aaSAndroid Build Coastguard Worker         case Operator::Kind::BITWISENOT:
332*c8dee2aaSAndroid Build Coastguard Worker             SkASSERT(!context.fConfig->strictES2Mode());
333*c8dee2aaSAndroid Build Coastguard Worker             SkASSERT(!baseType.isArray());
334*c8dee2aaSAndroid Build Coastguard Worker             SkASSERT(baseType.componentType().isInteger());
335*c8dee2aaSAndroid Build Coastguard Worker             if (baseType.isLiteral()) {
336*c8dee2aaSAndroid Build Coastguard Worker                 // The expression `~123` is no longer a literal; coerce to the actual type.
337*c8dee2aaSAndroid Build Coastguard Worker                 base = baseType.scalarTypeForLiteral().coerceExpression(std::move(base), context);
338*c8dee2aaSAndroid Build Coastguard Worker                 SkASSERT(base);
339*c8dee2aaSAndroid Build Coastguard Worker             }
340*c8dee2aaSAndroid Build Coastguard Worker             return bitwise_not_operand(context, pos, std::move(base));
341*c8dee2aaSAndroid Build Coastguard Worker 
342*c8dee2aaSAndroid Build Coastguard Worker         default:
343*c8dee2aaSAndroid Build Coastguard Worker             SkDEBUGFAILF("unsupported prefix operator: %s", op.operatorName());
344*c8dee2aaSAndroid Build Coastguard Worker     }
345*c8dee2aaSAndroid Build Coastguard Worker 
346*c8dee2aaSAndroid Build Coastguard Worker     return std::make_unique<PrefixExpression>(pos, op, std::move(base));
347*c8dee2aaSAndroid Build Coastguard Worker }
348*c8dee2aaSAndroid Build Coastguard Worker 
description(OperatorPrecedence parentPrecedence) const349*c8dee2aaSAndroid Build Coastguard Worker std::string PrefixExpression::description(OperatorPrecedence parentPrecedence) const {
350*c8dee2aaSAndroid Build Coastguard Worker     bool needsParens = (OperatorPrecedence::kPrefix >= parentPrecedence);
351*c8dee2aaSAndroid Build Coastguard Worker     return std::string(needsParens ? "(" : "") +
352*c8dee2aaSAndroid Build Coastguard Worker            std::string(this->getOperator().tightOperatorName()) +
353*c8dee2aaSAndroid Build Coastguard Worker            this->operand()->description(OperatorPrecedence::kPrefix) +
354*c8dee2aaSAndroid Build Coastguard Worker            std::string(needsParens ? ")" : "");
355*c8dee2aaSAndroid Build Coastguard Worker }
356*c8dee2aaSAndroid Build Coastguard Worker 
357*c8dee2aaSAndroid Build Coastguard Worker }  // namespace SkSL
358