1*c8dee2aaSAndroid Build Coastguard Worker /* 2*c8dee2aaSAndroid Build Coastguard Worker * Copyright 2016 Google Inc. 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 #ifndef SKSL_BINARYEXPRESSION 9*c8dee2aaSAndroid Build Coastguard Worker #define SKSL_BINARYEXPRESSION 10*c8dee2aaSAndroid Build Coastguard Worker 11*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkTypes.h" 12*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLOperator.h" 13*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLPosition.h" 14*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLExpression.h" 15*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLIRNode.h" 16*c8dee2aaSAndroid Build Coastguard Worker 17*c8dee2aaSAndroid Build Coastguard Worker #include <memory> 18*c8dee2aaSAndroid Build Coastguard Worker #include <string> 19*c8dee2aaSAndroid Build Coastguard Worker #include <utility> 20*c8dee2aaSAndroid Build Coastguard Worker 21*c8dee2aaSAndroid Build Coastguard Worker namespace SkSL { 22*c8dee2aaSAndroid Build Coastguard Worker 23*c8dee2aaSAndroid Build Coastguard Worker class Context; 24*c8dee2aaSAndroid Build Coastguard Worker class Type; 25*c8dee2aaSAndroid Build Coastguard Worker class VariableReference; 26*c8dee2aaSAndroid Build Coastguard Worker 27*c8dee2aaSAndroid Build Coastguard Worker /** 28*c8dee2aaSAndroid Build Coastguard Worker * A binary operation. 29*c8dee2aaSAndroid Build Coastguard Worker */ 30*c8dee2aaSAndroid Build Coastguard Worker class BinaryExpression final : public Expression { 31*c8dee2aaSAndroid Build Coastguard Worker public: 32*c8dee2aaSAndroid Build Coastguard Worker inline static constexpr Kind kIRNodeKind = Kind::kBinary; 33*c8dee2aaSAndroid Build Coastguard Worker BinaryExpression(Position pos,std::unique_ptr<Expression> left,Operator op,std::unique_ptr<Expression> right,const Type * type)34*c8dee2aaSAndroid Build Coastguard Worker BinaryExpression(Position pos, std::unique_ptr<Expression> left, Operator op, 35*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> right, const Type* type) 36*c8dee2aaSAndroid Build Coastguard Worker : INHERITED(pos, kIRNodeKind, type) 37*c8dee2aaSAndroid Build Coastguard Worker , fLeft(std::move(left)) 38*c8dee2aaSAndroid Build Coastguard Worker , fOperator(op) 39*c8dee2aaSAndroid Build Coastguard Worker , fRight(std::move(right)) { 40*c8dee2aaSAndroid Build Coastguard Worker // If we are assigning to a VariableReference, ensure that it is set to Write or ReadWrite. 41*c8dee2aaSAndroid Build Coastguard Worker SkASSERT(!op.isAssignment() || CheckRef(*this->left())); 42*c8dee2aaSAndroid Build Coastguard Worker } 43*c8dee2aaSAndroid Build Coastguard Worker 44*c8dee2aaSAndroid Build Coastguard Worker // Creates a potentially-simplified form of the expression. Determines the result type 45*c8dee2aaSAndroid Build Coastguard Worker // programmatically. Typechecks and coerces input expressions; reports errors via ErrorReporter. 46*c8dee2aaSAndroid Build Coastguard Worker static std::unique_ptr<Expression> Convert(const Context& context, 47*c8dee2aaSAndroid Build Coastguard Worker Position pos, 48*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> left, 49*c8dee2aaSAndroid Build Coastguard Worker Operator op, 50*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> right); 51*c8dee2aaSAndroid Build Coastguard Worker 52*c8dee2aaSAndroid Build Coastguard Worker // Creates a potentially-simplified form of the expression. Determines the result type 53*c8dee2aaSAndroid Build Coastguard Worker // programmatically. Asserts if the expressions do not typecheck or are otherwise invalid. 54*c8dee2aaSAndroid Build Coastguard Worker static std::unique_ptr<Expression> Make(const Context& context, 55*c8dee2aaSAndroid Build Coastguard Worker Position pos, 56*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> left, 57*c8dee2aaSAndroid Build Coastguard Worker Operator op, 58*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> right); 59*c8dee2aaSAndroid Build Coastguard Worker 60*c8dee2aaSAndroid Build Coastguard Worker // Creates a potentially-simplified form of the expression. Result type is passed in. 61*c8dee2aaSAndroid Build Coastguard Worker // Asserts if the expressions do not typecheck or are otherwise invalid. 62*c8dee2aaSAndroid Build Coastguard Worker static std::unique_ptr<Expression> Make(const Context& context, 63*c8dee2aaSAndroid Build Coastguard Worker Position pos, 64*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> left, 65*c8dee2aaSAndroid Build Coastguard Worker Operator op, 66*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> right, 67*c8dee2aaSAndroid Build Coastguard Worker const Type* resultType); 68*c8dee2aaSAndroid Build Coastguard Worker left()69*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression>& left() { 70*c8dee2aaSAndroid Build Coastguard Worker return fLeft; 71*c8dee2aaSAndroid Build Coastguard Worker } 72*c8dee2aaSAndroid Build Coastguard Worker left()73*c8dee2aaSAndroid Build Coastguard Worker const std::unique_ptr<Expression>& left() const { 74*c8dee2aaSAndroid Build Coastguard Worker return fLeft; 75*c8dee2aaSAndroid Build Coastguard Worker } 76*c8dee2aaSAndroid Build Coastguard Worker right()77*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression>& right() { 78*c8dee2aaSAndroid Build Coastguard Worker return fRight; 79*c8dee2aaSAndroid Build Coastguard Worker } 80*c8dee2aaSAndroid Build Coastguard Worker right()81*c8dee2aaSAndroid Build Coastguard Worker const std::unique_ptr<Expression>& right() const { 82*c8dee2aaSAndroid Build Coastguard Worker return fRight; 83*c8dee2aaSAndroid Build Coastguard Worker } 84*c8dee2aaSAndroid Build Coastguard Worker getOperator()85*c8dee2aaSAndroid Build Coastguard Worker Operator getOperator() const { 86*c8dee2aaSAndroid Build Coastguard Worker return fOperator; 87*c8dee2aaSAndroid Build Coastguard Worker } 88*c8dee2aaSAndroid Build Coastguard Worker 89*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> clone(Position pos) const override; 90*c8dee2aaSAndroid Build Coastguard Worker 91*c8dee2aaSAndroid Build Coastguard Worker std::string description(OperatorPrecedence parentPrecedence) const override; 92*c8dee2aaSAndroid Build Coastguard Worker 93*c8dee2aaSAndroid Build Coastguard Worker /** 94*c8dee2aaSAndroid Build Coastguard Worker * If the expression is an assignment like `a = 1` or `a += sin(b)`, returns the 95*c8dee2aaSAndroid Build Coastguard Worker * VariableReference that will be written to. For other types of expressions, returns null. 96*c8dee2aaSAndroid Build Coastguard Worker * Complex expressions that contain inner assignments, like `(a = b) * 2`, will return null. 97*c8dee2aaSAndroid Build Coastguard Worker */ 98*c8dee2aaSAndroid Build Coastguard Worker VariableReference* isAssignmentIntoVariable(); 99*c8dee2aaSAndroid Build Coastguard Worker 100*c8dee2aaSAndroid Build Coastguard Worker private: 101*c8dee2aaSAndroid Build Coastguard Worker static bool CheckRef(const Expression& expr); 102*c8dee2aaSAndroid Build Coastguard Worker 103*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> fLeft; 104*c8dee2aaSAndroid Build Coastguard Worker Operator fOperator; 105*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> fRight; 106*c8dee2aaSAndroid Build Coastguard Worker 107*c8dee2aaSAndroid Build Coastguard Worker using INHERITED = Expression; 108*c8dee2aaSAndroid Build Coastguard Worker }; 109*c8dee2aaSAndroid Build Coastguard Worker 110*c8dee2aaSAndroid Build Coastguard Worker } // namespace SkSL 111*c8dee2aaSAndroid Build Coastguard Worker 112*c8dee2aaSAndroid Build Coastguard Worker #endif 113