xref: /aosp_15_r20/external/skia/src/sksl/ir/SkSLSwizzle.h (revision c8dee2aa9b3f27cf6c858bd81872bdeb2c07ed17)
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