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_SWIZZLE 9*c8dee2aaSAndroid Build Coastguard Worker #define SKSL_SWIZZLE 10*c8dee2aaSAndroid Build Coastguard Worker 11*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkTypes.h" 12*c8dee2aaSAndroid Build Coastguard Worker #include "src/base/SkFixedArray.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 #include "src/sksl/ir/SkSLType.h" 17*c8dee2aaSAndroid Build Coastguard Worker 18*c8dee2aaSAndroid Build Coastguard Worker #include <cstdint> 19*c8dee2aaSAndroid Build Coastguard Worker #include <initializer_list> 20*c8dee2aaSAndroid Build Coastguard Worker #include <memory> 21*c8dee2aaSAndroid Build Coastguard Worker #include <string> 22*c8dee2aaSAndroid Build Coastguard Worker #include <string_view> 23*c8dee2aaSAndroid Build Coastguard Worker #include <utility> 24*c8dee2aaSAndroid Build Coastguard Worker 25*c8dee2aaSAndroid Build Coastguard Worker namespace SkSL { 26*c8dee2aaSAndroid Build Coastguard Worker 27*c8dee2aaSAndroid Build Coastguard Worker class Context; 28*c8dee2aaSAndroid Build Coastguard Worker enum class OperatorPrecedence : uint8_t; 29*c8dee2aaSAndroid Build Coastguard Worker 30*c8dee2aaSAndroid Build Coastguard Worker namespace SwizzleComponent { 31*c8dee2aaSAndroid Build Coastguard Worker 32*c8dee2aaSAndroid Build Coastguard Worker enum Type : int8_t { 33*c8dee2aaSAndroid Build Coastguard Worker X = 0, Y = 1, Z = 2, W = 3, 34*c8dee2aaSAndroid Build Coastguard Worker R = 4, G = 5, B = 6, A = 7, 35*c8dee2aaSAndroid Build Coastguard Worker S = 8, T = 9, P = 10, Q = 11, 36*c8dee2aaSAndroid Build Coastguard Worker UL = 12, UT = 13, UR = 14, UB = 15, 37*c8dee2aaSAndroid Build Coastguard Worker ZERO, 38*c8dee2aaSAndroid Build Coastguard Worker ONE 39*c8dee2aaSAndroid Build Coastguard Worker }; 40*c8dee2aaSAndroid Build Coastguard Worker 41*c8dee2aaSAndroid Build Coastguard Worker } // namespace SwizzleComponent 42*c8dee2aaSAndroid Build Coastguard Worker 43*c8dee2aaSAndroid Build Coastguard Worker /** 44*c8dee2aaSAndroid Build Coastguard Worker * Represents the components of a vector swizzle. 45*c8dee2aaSAndroid Build Coastguard Worker */ 46*c8dee2aaSAndroid Build Coastguard Worker using ComponentArray = skia_private::FixedArray<4, int8_t>; 47*c8dee2aaSAndroid Build Coastguard Worker 48*c8dee2aaSAndroid Build Coastguard Worker /** 49*c8dee2aaSAndroid Build Coastguard Worker * Represents a vector swizzle operation such as 'float3(1, 2, 3).zyx'. 50*c8dee2aaSAndroid Build Coastguard Worker */ 51*c8dee2aaSAndroid Build Coastguard Worker class Swizzle final : public Expression { 52*c8dee2aaSAndroid Build Coastguard Worker public: 53*c8dee2aaSAndroid Build Coastguard Worker inline static constexpr Kind kIRNodeKind = Kind::kSwizzle; 54*c8dee2aaSAndroid Build Coastguard Worker 55*c8dee2aaSAndroid Build Coastguard Worker using Component = SwizzleComponent::Type; 56*c8dee2aaSAndroid Build Coastguard Worker Swizzle(const Context & context,Position pos,std::unique_ptr<Expression> base,const ComponentArray & components)57*c8dee2aaSAndroid Build Coastguard Worker Swizzle(const Context& context, Position pos, std::unique_ptr<Expression> base, 58*c8dee2aaSAndroid Build Coastguard Worker const ComponentArray& components) 59*c8dee2aaSAndroid Build Coastguard Worker : INHERITED(pos, kIRNodeKind, 60*c8dee2aaSAndroid Build Coastguard Worker &base->type().componentType().toCompound(context, components.size(), 1)) 61*c8dee2aaSAndroid Build Coastguard Worker , fBase(std::move(base)) 62*c8dee2aaSAndroid Build Coastguard Worker , fComponents(components) { 63*c8dee2aaSAndroid Build Coastguard Worker SkASSERT(this->components().size() >= 1 && this->components().size() <= 4); 64*c8dee2aaSAndroid Build Coastguard Worker } 65*c8dee2aaSAndroid Build Coastguard Worker 66*c8dee2aaSAndroid Build Coastguard Worker // Swizzle::Convert permits component strings containing ZERO or ONE, does typechecking, reports 67*c8dee2aaSAndroid Build Coastguard Worker // errors via ErrorReporter, and returns an expression that combines constructors and native 68*c8dee2aaSAndroid Build Coastguard Worker // swizzles (comprised solely of X/Y/W/Z). 69*c8dee2aaSAndroid Build Coastguard Worker static std::unique_ptr<Expression> Convert(const Context& context, 70*c8dee2aaSAndroid Build Coastguard Worker Position pos, 71*c8dee2aaSAndroid Build Coastguard Worker Position maskPos, 72*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> base, 73*c8dee2aaSAndroid Build Coastguard Worker std::string_view componentString); 74*c8dee2aaSAndroid Build Coastguard Worker 75*c8dee2aaSAndroid Build Coastguard Worker // Swizzle::Make does not permit ZERO or ONE in the component array, just X/Y/Z/W; errors are 76*c8dee2aaSAndroid Build Coastguard Worker // reported via ASSERT. 77*c8dee2aaSAndroid Build Coastguard Worker static std::unique_ptr<Expression> Make(const Context& context, 78*c8dee2aaSAndroid Build Coastguard Worker Position pos, 79*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> expr, 80*c8dee2aaSAndroid Build Coastguard Worker ComponentArray components); 81*c8dee2aaSAndroid Build Coastguard Worker 82*c8dee2aaSAndroid Build Coastguard Worker // Swizzle::MakeExact creates the swizzle as requested, without simplification. 83*c8dee2aaSAndroid Build Coastguard Worker static std::unique_ptr<Expression> MakeExact(const Context& context, 84*c8dee2aaSAndroid Build Coastguard Worker Position pos, 85*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> expr, 86*c8dee2aaSAndroid Build Coastguard Worker ComponentArray components); 87*c8dee2aaSAndroid Build Coastguard Worker base()88*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression>& base() { 89*c8dee2aaSAndroid Build Coastguard Worker return fBase; 90*c8dee2aaSAndroid Build Coastguard Worker } 91*c8dee2aaSAndroid Build Coastguard Worker base()92*c8dee2aaSAndroid Build Coastguard Worker const std::unique_ptr<Expression>& base() const { 93*c8dee2aaSAndroid Build Coastguard Worker return fBase; 94*c8dee2aaSAndroid Build Coastguard Worker } 95*c8dee2aaSAndroid Build Coastguard Worker components()96*c8dee2aaSAndroid Build Coastguard Worker const ComponentArray& components() const { 97*c8dee2aaSAndroid Build Coastguard Worker return fComponents; 98*c8dee2aaSAndroid Build Coastguard Worker } 99*c8dee2aaSAndroid Build Coastguard Worker clone(Position pos)100*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> clone(Position pos) const override { 101*c8dee2aaSAndroid Build Coastguard Worker return std::unique_ptr<Expression>(new Swizzle(pos, &this->type(), this->base()->clone(), 102*c8dee2aaSAndroid Build Coastguard Worker this->components())); 103*c8dee2aaSAndroid Build Coastguard Worker } 104*c8dee2aaSAndroid Build Coastguard Worker 105*c8dee2aaSAndroid Build Coastguard Worker std::string description(OperatorPrecedence) const override; 106*c8dee2aaSAndroid Build Coastguard Worker 107*c8dee2aaSAndroid Build Coastguard Worker // Converts an array of swizzle components into a string. 108*c8dee2aaSAndroid Build Coastguard Worker static std::string MaskString(const ComponentArray& inComponents); 109*c8dee2aaSAndroid Build Coastguard Worker 110*c8dee2aaSAndroid Build Coastguard Worker // Returns true if a swizzle keeps each component in its original position. 111*c8dee2aaSAndroid Build Coastguard Worker static bool IsIdentity(const ComponentArray& components); 112*c8dee2aaSAndroid Build Coastguard Worker 113*c8dee2aaSAndroid Build Coastguard Worker private: Swizzle(Position pos,const Type * type,std::unique_ptr<Expression> base,const ComponentArray & components)114*c8dee2aaSAndroid Build Coastguard Worker Swizzle(Position pos, const Type* type, std::unique_ptr<Expression> base, 115*c8dee2aaSAndroid Build Coastguard Worker const ComponentArray& components) 116*c8dee2aaSAndroid Build Coastguard Worker : INHERITED(pos, kIRNodeKind, type) 117*c8dee2aaSAndroid Build Coastguard Worker , fBase(std::move(base)) 118*c8dee2aaSAndroid Build Coastguard Worker , fComponents(components) { 119*c8dee2aaSAndroid Build Coastguard Worker SkASSERT(this->components().size() >= 1 && this->components().size() <= 4); 120*c8dee2aaSAndroid Build Coastguard Worker } 121*c8dee2aaSAndroid Build Coastguard Worker 122*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> fBase; 123*c8dee2aaSAndroid Build Coastguard Worker ComponentArray fComponents; 124*c8dee2aaSAndroid Build Coastguard Worker 125*c8dee2aaSAndroid Build Coastguard Worker using INHERITED = Expression; 126*c8dee2aaSAndroid Build Coastguard Worker }; 127*c8dee2aaSAndroid Build Coastguard Worker 128*c8dee2aaSAndroid Build Coastguard Worker } // namespace SkSL 129*c8dee2aaSAndroid Build Coastguard Worker 130*c8dee2aaSAndroid Build Coastguard Worker #endif 131