xref: /aosp_15_r20/external/skia/src/sksl/SkSLOperator.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/SkSLOperator.h"
9*c8dee2aaSAndroid Build Coastguard Worker 
10*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkTypes.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "src/base/SkStringView.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLBuiltinTypes.h"
13*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLContext.h"
14*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLProgramSettings.h"
15*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLType.h"
16*c8dee2aaSAndroid Build Coastguard Worker 
17*c8dee2aaSAndroid Build Coastguard Worker #include <memory>
18*c8dee2aaSAndroid Build Coastguard Worker 
19*c8dee2aaSAndroid Build Coastguard Worker namespace SkSL {
20*c8dee2aaSAndroid Build Coastguard Worker 
getBinaryPrecedence() const21*c8dee2aaSAndroid Build Coastguard Worker OperatorPrecedence Operator::getBinaryPrecedence() const {
22*c8dee2aaSAndroid Build Coastguard Worker     switch (this->kind()) {
23*c8dee2aaSAndroid Build Coastguard Worker         case Kind::STAR:         // fall through
24*c8dee2aaSAndroid Build Coastguard Worker         case Kind::SLASH:        // fall through
25*c8dee2aaSAndroid Build Coastguard Worker         case Kind::PERCENT:      return OperatorPrecedence::kMultiplicative;
26*c8dee2aaSAndroid Build Coastguard Worker         case Kind::PLUS:         // fall through
27*c8dee2aaSAndroid Build Coastguard Worker         case Kind::MINUS:        return OperatorPrecedence::kAdditive;
28*c8dee2aaSAndroid Build Coastguard Worker         case Kind::SHL:          // fall through
29*c8dee2aaSAndroid Build Coastguard Worker         case Kind::SHR:          return OperatorPrecedence::kShift;
30*c8dee2aaSAndroid Build Coastguard Worker         case Kind::LT:           // fall through
31*c8dee2aaSAndroid Build Coastguard Worker         case Kind::GT:           // fall through
32*c8dee2aaSAndroid Build Coastguard Worker         case Kind::LTEQ:         // fall through
33*c8dee2aaSAndroid Build Coastguard Worker         case Kind::GTEQ:         return OperatorPrecedence::kRelational;
34*c8dee2aaSAndroid Build Coastguard Worker         case Kind::EQEQ:         // fall through
35*c8dee2aaSAndroid Build Coastguard Worker         case Kind::NEQ:          return OperatorPrecedence::kEquality;
36*c8dee2aaSAndroid Build Coastguard Worker         case Kind::BITWISEAND:   return OperatorPrecedence::kBitwiseAnd;
37*c8dee2aaSAndroid Build Coastguard Worker         case Kind::BITWISEXOR:   return OperatorPrecedence::kBitwiseXor;
38*c8dee2aaSAndroid Build Coastguard Worker         case Kind::BITWISEOR:    return OperatorPrecedence::kBitwiseOr;
39*c8dee2aaSAndroid Build Coastguard Worker         case Kind::LOGICALAND:   return OperatorPrecedence::kLogicalAnd;
40*c8dee2aaSAndroid Build Coastguard Worker         case Kind::LOGICALXOR:   return OperatorPrecedence::kLogicalXor;
41*c8dee2aaSAndroid Build Coastguard Worker         case Kind::LOGICALOR:    return OperatorPrecedence::kLogicalOr;
42*c8dee2aaSAndroid Build Coastguard Worker         case Kind::EQ:           // fall through
43*c8dee2aaSAndroid Build Coastguard Worker         case Kind::PLUSEQ:       // fall through
44*c8dee2aaSAndroid Build Coastguard Worker         case Kind::MINUSEQ:      // fall through
45*c8dee2aaSAndroid Build Coastguard Worker         case Kind::STAREQ:       // fall through
46*c8dee2aaSAndroid Build Coastguard Worker         case Kind::SLASHEQ:      // fall through
47*c8dee2aaSAndroid Build Coastguard Worker         case Kind::PERCENTEQ:    // fall through
48*c8dee2aaSAndroid Build Coastguard Worker         case Kind::SHLEQ:        // fall through
49*c8dee2aaSAndroid Build Coastguard Worker         case Kind::SHREQ:        // fall through
50*c8dee2aaSAndroid Build Coastguard Worker         case Kind::BITWISEANDEQ: // fall through
51*c8dee2aaSAndroid Build Coastguard Worker         case Kind::BITWISEXOREQ: // fall through
52*c8dee2aaSAndroid Build Coastguard Worker         case Kind::BITWISEOREQ:  return OperatorPrecedence::kAssignment;
53*c8dee2aaSAndroid Build Coastguard Worker         case Kind::COMMA:        return OperatorPrecedence::kSequence;
54*c8dee2aaSAndroid Build Coastguard Worker         default: SK_ABORT("unsupported binary operator");
55*c8dee2aaSAndroid Build Coastguard Worker     }
56*c8dee2aaSAndroid Build Coastguard Worker }
57*c8dee2aaSAndroid Build Coastguard Worker 
operatorName() const58*c8dee2aaSAndroid Build Coastguard Worker const char* Operator::operatorName() const {
59*c8dee2aaSAndroid Build Coastguard Worker     switch (this->kind()) {
60*c8dee2aaSAndroid Build Coastguard Worker         case Kind::PLUS:         return " + ";
61*c8dee2aaSAndroid Build Coastguard Worker         case Kind::MINUS:        return " - ";
62*c8dee2aaSAndroid Build Coastguard Worker         case Kind::STAR:         return " * ";
63*c8dee2aaSAndroid Build Coastguard Worker         case Kind::SLASH:        return " / ";
64*c8dee2aaSAndroid Build Coastguard Worker         case Kind::PERCENT:      return " % ";
65*c8dee2aaSAndroid Build Coastguard Worker         case Kind::SHL:          return " << ";
66*c8dee2aaSAndroid Build Coastguard Worker         case Kind::SHR:          return " >> ";
67*c8dee2aaSAndroid Build Coastguard Worker         case Kind::LOGICALNOT:   return "!";
68*c8dee2aaSAndroid Build Coastguard Worker         case Kind::LOGICALAND:   return " && ";
69*c8dee2aaSAndroid Build Coastguard Worker         case Kind::LOGICALOR:    return " || ";
70*c8dee2aaSAndroid Build Coastguard Worker         case Kind::LOGICALXOR:   return " ^^ ";
71*c8dee2aaSAndroid Build Coastguard Worker         case Kind::BITWISENOT:   return "~";
72*c8dee2aaSAndroid Build Coastguard Worker         case Kind::BITWISEAND:   return " & ";
73*c8dee2aaSAndroid Build Coastguard Worker         case Kind::BITWISEOR:    return " | ";
74*c8dee2aaSAndroid Build Coastguard Worker         case Kind::BITWISEXOR:   return " ^ ";
75*c8dee2aaSAndroid Build Coastguard Worker         case Kind::EQ:           return " = ";
76*c8dee2aaSAndroid Build Coastguard Worker         case Kind::EQEQ:         return " == ";
77*c8dee2aaSAndroid Build Coastguard Worker         case Kind::NEQ:          return " != ";
78*c8dee2aaSAndroid Build Coastguard Worker         case Kind::LT:           return " < ";
79*c8dee2aaSAndroid Build Coastguard Worker         case Kind::GT:           return " > ";
80*c8dee2aaSAndroid Build Coastguard Worker         case Kind::LTEQ:         return " <= ";
81*c8dee2aaSAndroid Build Coastguard Worker         case Kind::GTEQ:         return " >= ";
82*c8dee2aaSAndroid Build Coastguard Worker         case Kind::PLUSEQ:       return " += ";
83*c8dee2aaSAndroid Build Coastguard Worker         case Kind::MINUSEQ:      return " -= ";
84*c8dee2aaSAndroid Build Coastguard Worker         case Kind::STAREQ:       return " *= ";
85*c8dee2aaSAndroid Build Coastguard Worker         case Kind::SLASHEQ:      return " /= ";
86*c8dee2aaSAndroid Build Coastguard Worker         case Kind::PERCENTEQ:    return " %= ";
87*c8dee2aaSAndroid Build Coastguard Worker         case Kind::SHLEQ:        return " <<= ";
88*c8dee2aaSAndroid Build Coastguard Worker         case Kind::SHREQ:        return " >>= ";
89*c8dee2aaSAndroid Build Coastguard Worker         case Kind::BITWISEANDEQ: return " &= ";
90*c8dee2aaSAndroid Build Coastguard Worker         case Kind::BITWISEOREQ:  return " |= ";
91*c8dee2aaSAndroid Build Coastguard Worker         case Kind::BITWISEXOREQ: return " ^= ";
92*c8dee2aaSAndroid Build Coastguard Worker         case Kind::PLUSPLUS:     return "++";
93*c8dee2aaSAndroid Build Coastguard Worker         case Kind::MINUSMINUS:   return "--";
94*c8dee2aaSAndroid Build Coastguard Worker         case Kind::COMMA:        return ", ";
95*c8dee2aaSAndroid Build Coastguard Worker         default: SkUNREACHABLE;
96*c8dee2aaSAndroid Build Coastguard Worker     }
97*c8dee2aaSAndroid Build Coastguard Worker }
98*c8dee2aaSAndroid Build Coastguard Worker 
tightOperatorName() const99*c8dee2aaSAndroid Build Coastguard Worker std::string_view Operator::tightOperatorName() const {
100*c8dee2aaSAndroid Build Coastguard Worker     std::string_view name = this->operatorName();
101*c8dee2aaSAndroid Build Coastguard Worker     if (skstd::starts_with(name, ' ')) {
102*c8dee2aaSAndroid Build Coastguard Worker         name.remove_prefix(1);
103*c8dee2aaSAndroid Build Coastguard Worker     }
104*c8dee2aaSAndroid Build Coastguard Worker     if (skstd::ends_with(name, ' ')) {
105*c8dee2aaSAndroid Build Coastguard Worker         name.remove_suffix(1);
106*c8dee2aaSAndroid Build Coastguard Worker     }
107*c8dee2aaSAndroid Build Coastguard Worker     return name;
108*c8dee2aaSAndroid Build Coastguard Worker }
109*c8dee2aaSAndroid Build Coastguard Worker 
isAssignment() const110*c8dee2aaSAndroid Build Coastguard Worker bool Operator::isAssignment() const {
111*c8dee2aaSAndroid Build Coastguard Worker     switch (this->kind()) {
112*c8dee2aaSAndroid Build Coastguard Worker         case Kind::EQ:           // fall through
113*c8dee2aaSAndroid Build Coastguard Worker         case Kind::PLUSEQ:       // fall through
114*c8dee2aaSAndroid Build Coastguard Worker         case Kind::MINUSEQ:      // fall through
115*c8dee2aaSAndroid Build Coastguard Worker         case Kind::STAREQ:       // fall through
116*c8dee2aaSAndroid Build Coastguard Worker         case Kind::SLASHEQ:      // fall through
117*c8dee2aaSAndroid Build Coastguard Worker         case Kind::PERCENTEQ:    // fall through
118*c8dee2aaSAndroid Build Coastguard Worker         case Kind::SHLEQ:        // fall through
119*c8dee2aaSAndroid Build Coastguard Worker         case Kind::SHREQ:        // fall through
120*c8dee2aaSAndroid Build Coastguard Worker         case Kind::BITWISEOREQ:  // fall through
121*c8dee2aaSAndroid Build Coastguard Worker         case Kind::BITWISEXOREQ: // fall through
122*c8dee2aaSAndroid Build Coastguard Worker         case Kind::BITWISEANDEQ:
123*c8dee2aaSAndroid Build Coastguard Worker             return true;
124*c8dee2aaSAndroid Build Coastguard Worker         default:
125*c8dee2aaSAndroid Build Coastguard Worker             return false;
126*c8dee2aaSAndroid Build Coastguard Worker     }
127*c8dee2aaSAndroid Build Coastguard Worker }
128*c8dee2aaSAndroid Build Coastguard Worker 
isCompoundAssignment() const129*c8dee2aaSAndroid Build Coastguard Worker bool Operator::isCompoundAssignment() const {
130*c8dee2aaSAndroid Build Coastguard Worker     return this->isAssignment() && this->kind() != Kind::EQ;
131*c8dee2aaSAndroid Build Coastguard Worker }
132*c8dee2aaSAndroid Build Coastguard Worker 
removeAssignment() const133*c8dee2aaSAndroid Build Coastguard Worker Operator Operator::removeAssignment() const {
134*c8dee2aaSAndroid Build Coastguard Worker     switch (this->kind()) {
135*c8dee2aaSAndroid Build Coastguard Worker         case Kind::PLUSEQ:       return Kind::PLUS;
136*c8dee2aaSAndroid Build Coastguard Worker         case Kind::MINUSEQ:      return Kind::MINUS;
137*c8dee2aaSAndroid Build Coastguard Worker         case Kind::STAREQ:       return Kind::STAR;
138*c8dee2aaSAndroid Build Coastguard Worker         case Kind::SLASHEQ:      return Kind::SLASH;
139*c8dee2aaSAndroid Build Coastguard Worker         case Kind::PERCENTEQ:    return Kind::PERCENT;
140*c8dee2aaSAndroid Build Coastguard Worker         case Kind::SHLEQ:        return Kind::SHL;
141*c8dee2aaSAndroid Build Coastguard Worker         case Kind::SHREQ:        return Kind::SHR;
142*c8dee2aaSAndroid Build Coastguard Worker         case Kind::BITWISEOREQ:  return Kind::BITWISEOR;
143*c8dee2aaSAndroid Build Coastguard Worker         case Kind::BITWISEXOREQ: return Kind::BITWISEXOR;
144*c8dee2aaSAndroid Build Coastguard Worker         case Kind::BITWISEANDEQ: return Kind::BITWISEAND;
145*c8dee2aaSAndroid Build Coastguard Worker         default: return *this;
146*c8dee2aaSAndroid Build Coastguard Worker     }
147*c8dee2aaSAndroid Build Coastguard Worker }
148*c8dee2aaSAndroid Build Coastguard Worker 
isRelational() const149*c8dee2aaSAndroid Build Coastguard Worker bool Operator::isRelational() const {
150*c8dee2aaSAndroid Build Coastguard Worker     switch (this->kind()) {
151*c8dee2aaSAndroid Build Coastguard Worker         case Kind::LT:
152*c8dee2aaSAndroid Build Coastguard Worker         case Kind::GT:
153*c8dee2aaSAndroid Build Coastguard Worker         case Kind::LTEQ:
154*c8dee2aaSAndroid Build Coastguard Worker         case Kind::GTEQ:
155*c8dee2aaSAndroid Build Coastguard Worker             return true;
156*c8dee2aaSAndroid Build Coastguard Worker         default:
157*c8dee2aaSAndroid Build Coastguard Worker             return false;
158*c8dee2aaSAndroid Build Coastguard Worker     }
159*c8dee2aaSAndroid Build Coastguard Worker }
160*c8dee2aaSAndroid Build Coastguard Worker 
isOnlyValidForIntegralTypes() const161*c8dee2aaSAndroid Build Coastguard Worker bool Operator::isOnlyValidForIntegralTypes() const {
162*c8dee2aaSAndroid Build Coastguard Worker     switch (this->kind()) {
163*c8dee2aaSAndroid Build Coastguard Worker         case Kind::SHL:
164*c8dee2aaSAndroid Build Coastguard Worker         case Kind::SHR:
165*c8dee2aaSAndroid Build Coastguard Worker         case Kind::BITWISEAND:
166*c8dee2aaSAndroid Build Coastguard Worker         case Kind::BITWISEOR:
167*c8dee2aaSAndroid Build Coastguard Worker         case Kind::BITWISEXOR:
168*c8dee2aaSAndroid Build Coastguard Worker         case Kind::PERCENT:
169*c8dee2aaSAndroid Build Coastguard Worker         case Kind::SHLEQ:
170*c8dee2aaSAndroid Build Coastguard Worker         case Kind::SHREQ:
171*c8dee2aaSAndroid Build Coastguard Worker         case Kind::BITWISEANDEQ:
172*c8dee2aaSAndroid Build Coastguard Worker         case Kind::BITWISEOREQ:
173*c8dee2aaSAndroid Build Coastguard Worker         case Kind::BITWISEXOREQ:
174*c8dee2aaSAndroid Build Coastguard Worker         case Kind::PERCENTEQ:
175*c8dee2aaSAndroid Build Coastguard Worker             return true;
176*c8dee2aaSAndroid Build Coastguard Worker         default:
177*c8dee2aaSAndroid Build Coastguard Worker             return false;
178*c8dee2aaSAndroid Build Coastguard Worker     }
179*c8dee2aaSAndroid Build Coastguard Worker }
180*c8dee2aaSAndroid Build Coastguard Worker 
isValidForMatrixOrVector() const181*c8dee2aaSAndroid Build Coastguard Worker bool Operator::isValidForMatrixOrVector() const {
182*c8dee2aaSAndroid Build Coastguard Worker     switch (this->kind()) {
183*c8dee2aaSAndroid Build Coastguard Worker         case Kind::PLUS:
184*c8dee2aaSAndroid Build Coastguard Worker         case Kind::MINUS:
185*c8dee2aaSAndroid Build Coastguard Worker         case Kind::STAR:
186*c8dee2aaSAndroid Build Coastguard Worker         case Kind::SLASH:
187*c8dee2aaSAndroid Build Coastguard Worker         case Kind::PERCENT:
188*c8dee2aaSAndroid Build Coastguard Worker         case Kind::SHL:
189*c8dee2aaSAndroid Build Coastguard Worker         case Kind::SHR:
190*c8dee2aaSAndroid Build Coastguard Worker         case Kind::BITWISEAND:
191*c8dee2aaSAndroid Build Coastguard Worker         case Kind::BITWISEOR:
192*c8dee2aaSAndroid Build Coastguard Worker         case Kind::BITWISEXOR:
193*c8dee2aaSAndroid Build Coastguard Worker         case Kind::PLUSEQ:
194*c8dee2aaSAndroid Build Coastguard Worker         case Kind::MINUSEQ:
195*c8dee2aaSAndroid Build Coastguard Worker         case Kind::STAREQ:
196*c8dee2aaSAndroid Build Coastguard Worker         case Kind::SLASHEQ:
197*c8dee2aaSAndroid Build Coastguard Worker         case Kind::PERCENTEQ:
198*c8dee2aaSAndroid Build Coastguard Worker         case Kind::SHLEQ:
199*c8dee2aaSAndroid Build Coastguard Worker         case Kind::SHREQ:
200*c8dee2aaSAndroid Build Coastguard Worker         case Kind::BITWISEANDEQ:
201*c8dee2aaSAndroid Build Coastguard Worker         case Kind::BITWISEOREQ:
202*c8dee2aaSAndroid Build Coastguard Worker         case Kind::BITWISEXOREQ:
203*c8dee2aaSAndroid Build Coastguard Worker             return true;
204*c8dee2aaSAndroid Build Coastguard Worker         default:
205*c8dee2aaSAndroid Build Coastguard Worker             return false;
206*c8dee2aaSAndroid Build Coastguard Worker     }
207*c8dee2aaSAndroid Build Coastguard Worker }
208*c8dee2aaSAndroid Build Coastguard Worker 
isMatrixMultiply(const Type & left,const Type & right) const209*c8dee2aaSAndroid Build Coastguard Worker bool Operator::isMatrixMultiply(const Type& left, const Type& right) const {
210*c8dee2aaSAndroid Build Coastguard Worker     if (this->kind() != Kind::STAR && this->kind() != Kind::STAREQ) {
211*c8dee2aaSAndroid Build Coastguard Worker         return false;
212*c8dee2aaSAndroid Build Coastguard Worker     }
213*c8dee2aaSAndroid Build Coastguard Worker     if (left.isMatrix()) {
214*c8dee2aaSAndroid Build Coastguard Worker         return right.isMatrix() || right.isVector();
215*c8dee2aaSAndroid Build Coastguard Worker     }
216*c8dee2aaSAndroid Build Coastguard Worker     return left.isVector() && right.isMatrix();
217*c8dee2aaSAndroid Build Coastguard Worker }
218*c8dee2aaSAndroid Build Coastguard Worker 
219*c8dee2aaSAndroid Build Coastguard Worker /**
220*c8dee2aaSAndroid Build Coastguard Worker  * Determines the operand and result types of a binary expression. Returns true if the expression is
221*c8dee2aaSAndroid Build Coastguard Worker  * legal, false otherwise. If false, the values of the out parameters are undefined.
222*c8dee2aaSAndroid Build Coastguard Worker  */
determineBinaryType(const Context & context,const Type & left,const Type & right,const Type ** outLeftType,const Type ** outRightType,const Type ** outResultType) const223*c8dee2aaSAndroid Build Coastguard Worker bool Operator::determineBinaryType(const Context& context,
224*c8dee2aaSAndroid Build Coastguard Worker                                    const Type& left,
225*c8dee2aaSAndroid Build Coastguard Worker                                    const Type& right,
226*c8dee2aaSAndroid Build Coastguard Worker                                    const Type** outLeftType,
227*c8dee2aaSAndroid Build Coastguard Worker                                    const Type** outRightType,
228*c8dee2aaSAndroid Build Coastguard Worker                                    const Type** outResultType) const {
229*c8dee2aaSAndroid Build Coastguard Worker     const bool allowNarrowing = context.fConfig->fSettings.fAllowNarrowingConversions;
230*c8dee2aaSAndroid Build Coastguard Worker     switch (this->kind()) {
231*c8dee2aaSAndroid Build Coastguard Worker         case Kind::EQ:  // left = right
232*c8dee2aaSAndroid Build Coastguard Worker             if (left.isVoid()) {
233*c8dee2aaSAndroid Build Coastguard Worker                 return false;
234*c8dee2aaSAndroid Build Coastguard Worker             }
235*c8dee2aaSAndroid Build Coastguard Worker             *outLeftType = &left;
236*c8dee2aaSAndroid Build Coastguard Worker             *outRightType = &left;
237*c8dee2aaSAndroid Build Coastguard Worker             *outResultType = &left;
238*c8dee2aaSAndroid Build Coastguard Worker             return right.canCoerceTo(left, allowNarrowing);
239*c8dee2aaSAndroid Build Coastguard Worker 
240*c8dee2aaSAndroid Build Coastguard Worker         case Kind::EQEQ:   // left == right
241*c8dee2aaSAndroid Build Coastguard Worker         case Kind::NEQ: {  // left != right
242*c8dee2aaSAndroid Build Coastguard Worker             if (left.isVoid() || left.isOpaque()) {
243*c8dee2aaSAndroid Build Coastguard Worker                 return false;
244*c8dee2aaSAndroid Build Coastguard Worker             }
245*c8dee2aaSAndroid Build Coastguard Worker             CoercionCost rightToLeft = right.coercionCost(left),
246*c8dee2aaSAndroid Build Coastguard Worker                          leftToRight = left.coercionCost(right);
247*c8dee2aaSAndroid Build Coastguard Worker             if (rightToLeft < leftToRight) {
248*c8dee2aaSAndroid Build Coastguard Worker                 if (rightToLeft.isPossible(allowNarrowing)) {
249*c8dee2aaSAndroid Build Coastguard Worker                     *outLeftType = &left;
250*c8dee2aaSAndroid Build Coastguard Worker                     *outRightType = &left;
251*c8dee2aaSAndroid Build Coastguard Worker                     *outResultType = context.fTypes.fBool.get();
252*c8dee2aaSAndroid Build Coastguard Worker                     return true;
253*c8dee2aaSAndroid Build Coastguard Worker                 }
254*c8dee2aaSAndroid Build Coastguard Worker             } else {
255*c8dee2aaSAndroid Build Coastguard Worker                 if (leftToRight.isPossible(allowNarrowing)) {
256*c8dee2aaSAndroid Build Coastguard Worker                     *outLeftType = &right;
257*c8dee2aaSAndroid Build Coastguard Worker                     *outRightType = &right;
258*c8dee2aaSAndroid Build Coastguard Worker                     *outResultType = context.fTypes.fBool.get();
259*c8dee2aaSAndroid Build Coastguard Worker                     return true;
260*c8dee2aaSAndroid Build Coastguard Worker                 }
261*c8dee2aaSAndroid Build Coastguard Worker             }
262*c8dee2aaSAndroid Build Coastguard Worker             return false;
263*c8dee2aaSAndroid Build Coastguard Worker         }
264*c8dee2aaSAndroid Build Coastguard Worker         case Kind::LOGICALOR:   // left || right
265*c8dee2aaSAndroid Build Coastguard Worker         case Kind::LOGICALAND:  // left && right
266*c8dee2aaSAndroid Build Coastguard Worker         case Kind::LOGICALXOR:  // left ^^ right
267*c8dee2aaSAndroid Build Coastguard Worker             *outLeftType = context.fTypes.fBool.get();
268*c8dee2aaSAndroid Build Coastguard Worker             *outRightType = context.fTypes.fBool.get();
269*c8dee2aaSAndroid Build Coastguard Worker             *outResultType = context.fTypes.fBool.get();
270*c8dee2aaSAndroid Build Coastguard Worker             return left.canCoerceTo(*context.fTypes.fBool, allowNarrowing) &&
271*c8dee2aaSAndroid Build Coastguard Worker                    right.canCoerceTo(*context.fTypes.fBool, allowNarrowing);
272*c8dee2aaSAndroid Build Coastguard Worker 
273*c8dee2aaSAndroid Build Coastguard Worker         case Operator::Kind::COMMA:  // left, right
274*c8dee2aaSAndroid Build Coastguard Worker             if (left.isOpaque() || right.isOpaque()) {
275*c8dee2aaSAndroid Build Coastguard Worker                 return false;
276*c8dee2aaSAndroid Build Coastguard Worker             }
277*c8dee2aaSAndroid Build Coastguard Worker             *outLeftType = &left;
278*c8dee2aaSAndroid Build Coastguard Worker             *outRightType = &right;
279*c8dee2aaSAndroid Build Coastguard Worker             *outResultType = &right;
280*c8dee2aaSAndroid Build Coastguard Worker             return true;
281*c8dee2aaSAndroid Build Coastguard Worker 
282*c8dee2aaSAndroid Build Coastguard Worker         default:
283*c8dee2aaSAndroid Build Coastguard Worker             break;
284*c8dee2aaSAndroid Build Coastguard Worker     }
285*c8dee2aaSAndroid Build Coastguard Worker 
286*c8dee2aaSAndroid Build Coastguard Worker     // Boolean types only support the operators listed above (, = == != || && ^^).
287*c8dee2aaSAndroid Build Coastguard Worker     // If we've gotten this far with a boolean, we have an unsupported operator.
288*c8dee2aaSAndroid Build Coastguard Worker     const Type& leftComponentType = left.componentType();
289*c8dee2aaSAndroid Build Coastguard Worker     const Type& rightComponentType = right.componentType();
290*c8dee2aaSAndroid Build Coastguard Worker     if (leftComponentType.isBoolean() || rightComponentType.isBoolean()) {
291*c8dee2aaSAndroid Build Coastguard Worker         return false;
292*c8dee2aaSAndroid Build Coastguard Worker     }
293*c8dee2aaSAndroid Build Coastguard Worker 
294*c8dee2aaSAndroid Build Coastguard Worker     bool isAssignment = this->isAssignment();
295*c8dee2aaSAndroid Build Coastguard Worker     if (this->isMatrixMultiply(left, right)) {  // left * right
296*c8dee2aaSAndroid Build Coastguard Worker         // Determine final component type.
297*c8dee2aaSAndroid Build Coastguard Worker         if (!this->determineBinaryType(context, left.componentType(), right.componentType(),
298*c8dee2aaSAndroid Build Coastguard Worker                                        outLeftType, outRightType, outResultType)) {
299*c8dee2aaSAndroid Build Coastguard Worker             return false;
300*c8dee2aaSAndroid Build Coastguard Worker         }
301*c8dee2aaSAndroid Build Coastguard Worker         // Convert component type to compound.
302*c8dee2aaSAndroid Build Coastguard Worker         *outLeftType = &(*outResultType)->toCompound(context, left.columns(), left.rows());
303*c8dee2aaSAndroid Build Coastguard Worker         *outRightType = &(*outResultType)->toCompound(context, right.columns(), right.rows());
304*c8dee2aaSAndroid Build Coastguard Worker         int leftColumns = left.columns(), leftRows = left.rows();
305*c8dee2aaSAndroid Build Coastguard Worker         int rightColumns = right.columns(), rightRows = right.rows();
306*c8dee2aaSAndroid Build Coastguard Worker         if (right.isVector()) {
307*c8dee2aaSAndroid Build Coastguard Worker             // `matrix * vector` treats the vector as a column vector; we need to transpose it.
308*c8dee2aaSAndroid Build Coastguard Worker             std::swap(rightColumns, rightRows);
309*c8dee2aaSAndroid Build Coastguard Worker             SkASSERT(rightColumns == 1);
310*c8dee2aaSAndroid Build Coastguard Worker         }
311*c8dee2aaSAndroid Build Coastguard Worker         if (rightColumns > 1) {
312*c8dee2aaSAndroid Build Coastguard Worker             *outResultType = &(*outResultType)->toCompound(context, rightColumns, leftRows);
313*c8dee2aaSAndroid Build Coastguard Worker         } else {
314*c8dee2aaSAndroid Build Coastguard Worker             // The result was a column vector. Transpose it back to a row.
315*c8dee2aaSAndroid Build Coastguard Worker             *outResultType = &(*outResultType)->toCompound(context, leftRows, rightColumns);
316*c8dee2aaSAndroid Build Coastguard Worker         }
317*c8dee2aaSAndroid Build Coastguard Worker         if (isAssignment && ((*outResultType)->columns() != leftColumns ||
318*c8dee2aaSAndroid Build Coastguard Worker                              (*outResultType)->rows() != leftRows)) {
319*c8dee2aaSAndroid Build Coastguard Worker             return false;
320*c8dee2aaSAndroid Build Coastguard Worker         }
321*c8dee2aaSAndroid Build Coastguard Worker         return leftColumns == rightRows;
322*c8dee2aaSAndroid Build Coastguard Worker     }
323*c8dee2aaSAndroid Build Coastguard Worker 
324*c8dee2aaSAndroid Build Coastguard Worker     bool leftIsVectorOrMatrix = left.isVector() || left.isMatrix();
325*c8dee2aaSAndroid Build Coastguard Worker     bool validMatrixOrVectorOp = this->isValidForMatrixOrVector();
326*c8dee2aaSAndroid Build Coastguard Worker 
327*c8dee2aaSAndroid Build Coastguard Worker     if (leftIsVectorOrMatrix && validMatrixOrVectorOp && right.isScalar()) {
328*c8dee2aaSAndroid Build Coastguard Worker         // Determine final component type.
329*c8dee2aaSAndroid Build Coastguard Worker         if (!this->determineBinaryType(context, left.componentType(), right,
330*c8dee2aaSAndroid Build Coastguard Worker                                        outLeftType, outRightType, outResultType)) {
331*c8dee2aaSAndroid Build Coastguard Worker             return false;
332*c8dee2aaSAndroid Build Coastguard Worker         }
333*c8dee2aaSAndroid Build Coastguard Worker         // Convert component type to compound.
334*c8dee2aaSAndroid Build Coastguard Worker         *outLeftType = &(*outLeftType)->toCompound(context, left.columns(), left.rows());
335*c8dee2aaSAndroid Build Coastguard Worker         if (!this->isRelational()) {
336*c8dee2aaSAndroid Build Coastguard Worker             *outResultType = &(*outResultType)->toCompound(context, left.columns(), left.rows());
337*c8dee2aaSAndroid Build Coastguard Worker         }
338*c8dee2aaSAndroid Build Coastguard Worker         return true;
339*c8dee2aaSAndroid Build Coastguard Worker     }
340*c8dee2aaSAndroid Build Coastguard Worker 
341*c8dee2aaSAndroid Build Coastguard Worker     bool rightIsVectorOrMatrix = right.isVector() || right.isMatrix();
342*c8dee2aaSAndroid Build Coastguard Worker 
343*c8dee2aaSAndroid Build Coastguard Worker     if (!isAssignment && rightIsVectorOrMatrix && validMatrixOrVectorOp && left.isScalar()) {
344*c8dee2aaSAndroid Build Coastguard Worker         // Determine final component type.
345*c8dee2aaSAndroid Build Coastguard Worker         if (!this->determineBinaryType(context, left, right.componentType(),
346*c8dee2aaSAndroid Build Coastguard Worker                                        outLeftType, outRightType, outResultType)) {
347*c8dee2aaSAndroid Build Coastguard Worker             return false;
348*c8dee2aaSAndroid Build Coastguard Worker         }
349*c8dee2aaSAndroid Build Coastguard Worker         // Convert component type to compound.
350*c8dee2aaSAndroid Build Coastguard Worker         *outRightType = &(*outRightType)->toCompound(context, right.columns(), right.rows());
351*c8dee2aaSAndroid Build Coastguard Worker         if (!this->isRelational()) {
352*c8dee2aaSAndroid Build Coastguard Worker             *outResultType = &(*outResultType)->toCompound(context, right.columns(), right.rows());
353*c8dee2aaSAndroid Build Coastguard Worker         }
354*c8dee2aaSAndroid Build Coastguard Worker         return true;
355*c8dee2aaSAndroid Build Coastguard Worker     }
356*c8dee2aaSAndroid Build Coastguard Worker 
357*c8dee2aaSAndroid Build Coastguard Worker     CoercionCost rightToLeftCost = right.coercionCost(left);
358*c8dee2aaSAndroid Build Coastguard Worker     CoercionCost leftToRightCost = isAssignment ? CoercionCost::Impossible()
359*c8dee2aaSAndroid Build Coastguard Worker                                                 : left.coercionCost(right);
360*c8dee2aaSAndroid Build Coastguard Worker 
361*c8dee2aaSAndroid Build Coastguard Worker     if ((left.isScalar() && right.isScalar()) || (leftIsVectorOrMatrix && validMatrixOrVectorOp)) {
362*c8dee2aaSAndroid Build Coastguard Worker         if (this->isOnlyValidForIntegralTypes()) {
363*c8dee2aaSAndroid Build Coastguard Worker             if (!leftComponentType.isInteger() || !rightComponentType.isInteger()) {
364*c8dee2aaSAndroid Build Coastguard Worker                 return false;
365*c8dee2aaSAndroid Build Coastguard Worker             }
366*c8dee2aaSAndroid Build Coastguard Worker         }
367*c8dee2aaSAndroid Build Coastguard Worker         if (rightToLeftCost.isPossible(allowNarrowing) && rightToLeftCost < leftToRightCost) {
368*c8dee2aaSAndroid Build Coastguard Worker             // Right-to-Left conversion is possible and cheaper
369*c8dee2aaSAndroid Build Coastguard Worker             *outLeftType = &left;
370*c8dee2aaSAndroid Build Coastguard Worker             *outRightType = &left;
371*c8dee2aaSAndroid Build Coastguard Worker             *outResultType = &left;
372*c8dee2aaSAndroid Build Coastguard Worker         } else if (leftToRightCost.isPossible(allowNarrowing)) {
373*c8dee2aaSAndroid Build Coastguard Worker             // Left-to-Right conversion is possible (and at least as cheap as Right-to-Left)
374*c8dee2aaSAndroid Build Coastguard Worker             *outLeftType = &right;
375*c8dee2aaSAndroid Build Coastguard Worker             *outRightType = &right;
376*c8dee2aaSAndroid Build Coastguard Worker             *outResultType = &right;
377*c8dee2aaSAndroid Build Coastguard Worker         } else {
378*c8dee2aaSAndroid Build Coastguard Worker             return false;
379*c8dee2aaSAndroid Build Coastguard Worker         }
380*c8dee2aaSAndroid Build Coastguard Worker         if (this->isRelational()) {
381*c8dee2aaSAndroid Build Coastguard Worker             *outResultType = context.fTypes.fBool.get();
382*c8dee2aaSAndroid Build Coastguard Worker         }
383*c8dee2aaSAndroid Build Coastguard Worker         return true;
384*c8dee2aaSAndroid Build Coastguard Worker     }
385*c8dee2aaSAndroid Build Coastguard Worker     return false;
386*c8dee2aaSAndroid Build Coastguard Worker }
387*c8dee2aaSAndroid Build Coastguard Worker 
388*c8dee2aaSAndroid Build Coastguard Worker }  // namespace SkSL
389