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_TERNARYEXPRESSION 9*c8dee2aaSAndroid Build Coastguard Worker #define SKSL_TERNARYEXPRESSION 10*c8dee2aaSAndroid Build Coastguard Worker 11*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkTypes.h" 12*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLPosition.h" 13*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLExpression.h" 14*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLIRNode.h" 15*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLType.h" 16*c8dee2aaSAndroid Build Coastguard Worker 17*c8dee2aaSAndroid Build Coastguard Worker #include <cstdint> 18*c8dee2aaSAndroid Build Coastguard Worker #include <memory> 19*c8dee2aaSAndroid Build Coastguard Worker #include <string> 20*c8dee2aaSAndroid Build Coastguard Worker #include <utility> 21*c8dee2aaSAndroid Build Coastguard Worker 22*c8dee2aaSAndroid Build Coastguard Worker namespace SkSL { 23*c8dee2aaSAndroid Build Coastguard Worker 24*c8dee2aaSAndroid Build Coastguard Worker class Context; 25*c8dee2aaSAndroid Build Coastguard Worker enum class OperatorPrecedence : uint8_t; 26*c8dee2aaSAndroid Build Coastguard Worker 27*c8dee2aaSAndroid Build Coastguard Worker /** 28*c8dee2aaSAndroid Build Coastguard Worker * A ternary expression (test ? ifTrue : ifFalse). 29*c8dee2aaSAndroid Build Coastguard Worker */ 30*c8dee2aaSAndroid Build Coastguard Worker class TernaryExpression final : public Expression { 31*c8dee2aaSAndroid Build Coastguard Worker public: 32*c8dee2aaSAndroid Build Coastguard Worker inline static constexpr Kind kIRNodeKind = Kind::kTernary; 33*c8dee2aaSAndroid Build Coastguard Worker TernaryExpression(Position pos,std::unique_ptr<Expression> test,std::unique_ptr<Expression> ifTrue,std::unique_ptr<Expression> ifFalse)34*c8dee2aaSAndroid Build Coastguard Worker TernaryExpression(Position pos, std::unique_ptr<Expression> test, 35*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> ifTrue, std::unique_ptr<Expression> ifFalse) 36*c8dee2aaSAndroid Build Coastguard Worker : INHERITED(pos, kIRNodeKind, &ifTrue->type()) 37*c8dee2aaSAndroid Build Coastguard Worker , fTest(std::move(test)) 38*c8dee2aaSAndroid Build Coastguard Worker , fIfTrue(std::move(ifTrue)) 39*c8dee2aaSAndroid Build Coastguard Worker , fIfFalse(std::move(ifFalse)) { 40*c8dee2aaSAndroid Build Coastguard Worker SkASSERT(this->ifTrue()->type().matches(this->ifFalse()->type())); 41*c8dee2aaSAndroid Build Coastguard Worker } 42*c8dee2aaSAndroid Build Coastguard Worker 43*c8dee2aaSAndroid Build Coastguard Worker // Creates a potentially-simplified form of the ternary. Typechecks and coerces input 44*c8dee2aaSAndroid Build Coastguard Worker // expressions; reports errors via ErrorReporter. 45*c8dee2aaSAndroid Build Coastguard Worker static std::unique_ptr<Expression> Convert(const Context& context, 46*c8dee2aaSAndroid Build Coastguard Worker Position pos, 47*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> test, 48*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> ifTrue, 49*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> ifFalse); 50*c8dee2aaSAndroid Build Coastguard Worker 51*c8dee2aaSAndroid Build Coastguard Worker // Creates a potentially-simplified form of the ternary; reports errors via ASSERT. 52*c8dee2aaSAndroid Build Coastguard Worker static std::unique_ptr<Expression> Make(const Context& context, 53*c8dee2aaSAndroid Build Coastguard Worker Position pos, 54*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> test, 55*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> ifTrue, 56*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> ifFalse); 57*c8dee2aaSAndroid Build Coastguard Worker test()58*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression>& test() { 59*c8dee2aaSAndroid Build Coastguard Worker return fTest; 60*c8dee2aaSAndroid Build Coastguard Worker } 61*c8dee2aaSAndroid Build Coastguard Worker test()62*c8dee2aaSAndroid Build Coastguard Worker const std::unique_ptr<Expression>& test() const { 63*c8dee2aaSAndroid Build Coastguard Worker return fTest; 64*c8dee2aaSAndroid Build Coastguard Worker } 65*c8dee2aaSAndroid Build Coastguard Worker ifTrue()66*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression>& ifTrue() { 67*c8dee2aaSAndroid Build Coastguard Worker return fIfTrue; 68*c8dee2aaSAndroid Build Coastguard Worker } 69*c8dee2aaSAndroid Build Coastguard Worker ifTrue()70*c8dee2aaSAndroid Build Coastguard Worker const std::unique_ptr<Expression>& ifTrue() const { 71*c8dee2aaSAndroid Build Coastguard Worker return fIfTrue; 72*c8dee2aaSAndroid Build Coastguard Worker } 73*c8dee2aaSAndroid Build Coastguard Worker ifFalse()74*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression>& ifFalse() { 75*c8dee2aaSAndroid Build Coastguard Worker return fIfFalse; 76*c8dee2aaSAndroid Build Coastguard Worker } 77*c8dee2aaSAndroid Build Coastguard Worker ifFalse()78*c8dee2aaSAndroid Build Coastguard Worker const std::unique_ptr<Expression>& ifFalse() const { 79*c8dee2aaSAndroid Build Coastguard Worker return fIfFalse; 80*c8dee2aaSAndroid Build Coastguard Worker } 81*c8dee2aaSAndroid Build Coastguard Worker clone(Position pos)82*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> clone(Position pos) const override { 83*c8dee2aaSAndroid Build Coastguard Worker return std::make_unique<TernaryExpression>(pos, this->test()->clone(), 84*c8dee2aaSAndroid Build Coastguard Worker this->ifTrue()->clone(), 85*c8dee2aaSAndroid Build Coastguard Worker this->ifFalse()->clone()); 86*c8dee2aaSAndroid Build Coastguard Worker } 87*c8dee2aaSAndroid Build Coastguard Worker 88*c8dee2aaSAndroid Build Coastguard Worker std::string description(OperatorPrecedence parentPrecedence) const override; 89*c8dee2aaSAndroid Build Coastguard Worker 90*c8dee2aaSAndroid Build Coastguard Worker private: 91*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> fTest; 92*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> fIfTrue; 93*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> fIfFalse; 94*c8dee2aaSAndroid Build Coastguard Worker 95*c8dee2aaSAndroid Build Coastguard Worker using INHERITED = Expression; 96*c8dee2aaSAndroid Build Coastguard Worker }; 97*c8dee2aaSAndroid Build Coastguard Worker 98*c8dee2aaSAndroid Build Coastguard Worker } // namespace SkSL 99*c8dee2aaSAndroid Build Coastguard Worker 100*c8dee2aaSAndroid Build Coastguard Worker #endif 101