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/ir/SkSLChildCall.h"
9*c8dee2aaSAndroid Build Coastguard Worker
10*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkTypes.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/base/SkTArray.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/SkSLOperator.h"
15*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLString.h"
16*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLType.h"
17*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLVariable.h"
18*c8dee2aaSAndroid Build Coastguard Worker
19*c8dee2aaSAndroid Build Coastguard Worker using namespace skia_private;
20*c8dee2aaSAndroid Build Coastguard Worker
21*c8dee2aaSAndroid Build Coastguard Worker namespace SkSL {
22*c8dee2aaSAndroid Build Coastguard Worker
clone(Position pos) const23*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> ChildCall::clone(Position pos) const {
24*c8dee2aaSAndroid Build Coastguard Worker return std::make_unique<ChildCall>(pos, &this->type(), &this->child(),
25*c8dee2aaSAndroid Build Coastguard Worker this->arguments().clone());
26*c8dee2aaSAndroid Build Coastguard Worker }
27*c8dee2aaSAndroid Build Coastguard Worker
description(OperatorPrecedence) const28*c8dee2aaSAndroid Build Coastguard Worker std::string ChildCall::description(OperatorPrecedence) const {
29*c8dee2aaSAndroid Build Coastguard Worker std::string result = std::string(this->child().name()) + ".eval(";
30*c8dee2aaSAndroid Build Coastguard Worker auto separator = SkSL::String::Separator();
31*c8dee2aaSAndroid Build Coastguard Worker for (const std::unique_ptr<Expression>& arg : this->arguments()) {
32*c8dee2aaSAndroid Build Coastguard Worker result += separator();
33*c8dee2aaSAndroid Build Coastguard Worker result += arg->description(OperatorPrecedence::kSequence);
34*c8dee2aaSAndroid Build Coastguard Worker }
35*c8dee2aaSAndroid Build Coastguard Worker result += ")";
36*c8dee2aaSAndroid Build Coastguard Worker return result;
37*c8dee2aaSAndroid Build Coastguard Worker }
38*c8dee2aaSAndroid Build Coastguard Worker
call_signature_is_valid(const Context & context,const Variable & child,const ExpressionArray & arguments)39*c8dee2aaSAndroid Build Coastguard Worker [[maybe_unused]] static bool call_signature_is_valid(const Context& context,
40*c8dee2aaSAndroid Build Coastguard Worker const Variable& child,
41*c8dee2aaSAndroid Build Coastguard Worker const ExpressionArray& arguments) {
42*c8dee2aaSAndroid Build Coastguard Worker const Type* half4 = context.fTypes.fHalf4.get();
43*c8dee2aaSAndroid Build Coastguard Worker const Type* float2 = context.fTypes.fFloat2.get();
44*c8dee2aaSAndroid Build Coastguard Worker
45*c8dee2aaSAndroid Build Coastguard Worker auto params = [&]() -> STArray<2, const Type*> {
46*c8dee2aaSAndroid Build Coastguard Worker switch (child.type().typeKind()) {
47*c8dee2aaSAndroid Build Coastguard Worker case Type::TypeKind::kBlender: return { half4, half4 };
48*c8dee2aaSAndroid Build Coastguard Worker case Type::TypeKind::kColorFilter: return { half4 };
49*c8dee2aaSAndroid Build Coastguard Worker case Type::TypeKind::kShader: return { float2 };
50*c8dee2aaSAndroid Build Coastguard Worker default:
51*c8dee2aaSAndroid Build Coastguard Worker SkUNREACHABLE;
52*c8dee2aaSAndroid Build Coastguard Worker }
53*c8dee2aaSAndroid Build Coastguard Worker }();
54*c8dee2aaSAndroid Build Coastguard Worker
55*c8dee2aaSAndroid Build Coastguard Worker if (params.size() != arguments.size()) {
56*c8dee2aaSAndroid Build Coastguard Worker return false;
57*c8dee2aaSAndroid Build Coastguard Worker }
58*c8dee2aaSAndroid Build Coastguard Worker for (int i = 0; i < arguments.size(); i++) {
59*c8dee2aaSAndroid Build Coastguard Worker if (!arguments[i]->type().matches(*params[i])) {
60*c8dee2aaSAndroid Build Coastguard Worker return false;
61*c8dee2aaSAndroid Build Coastguard Worker }
62*c8dee2aaSAndroid Build Coastguard Worker }
63*c8dee2aaSAndroid Build Coastguard Worker return true;
64*c8dee2aaSAndroid Build Coastguard Worker }
65*c8dee2aaSAndroid Build Coastguard Worker
Make(const Context & context,Position pos,const Type * returnType,const Variable & child,ExpressionArray arguments)66*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> ChildCall::Make(const Context& context,
67*c8dee2aaSAndroid Build Coastguard Worker Position pos,
68*c8dee2aaSAndroid Build Coastguard Worker const Type* returnType,
69*c8dee2aaSAndroid Build Coastguard Worker const Variable& child,
70*c8dee2aaSAndroid Build Coastguard Worker ExpressionArray arguments) {
71*c8dee2aaSAndroid Build Coastguard Worker SkASSERT(call_signature_is_valid(context, child, arguments));
72*c8dee2aaSAndroid Build Coastguard Worker return std::make_unique<ChildCall>(pos, returnType, &child, std::move(arguments));
73*c8dee2aaSAndroid Build Coastguard Worker }
74*c8dee2aaSAndroid Build Coastguard Worker
75*c8dee2aaSAndroid Build Coastguard Worker } // namespace SkSL
76