xref: /aosp_15_r20/external/skia/src/sksl/ir/SkSLFunctionDefinition.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_FUNCTIONDEFINITION
9*c8dee2aaSAndroid Build Coastguard Worker #define SKSL_FUNCTIONDEFINITION
10*c8dee2aaSAndroid Build Coastguard Worker 
11*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLPosition.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLFunctionDeclaration.h"
13*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLIRNode.h"
14*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLProgramElement.h"
15*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLStatement.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 
25*c8dee2aaSAndroid Build Coastguard Worker /**
26*c8dee2aaSAndroid Build Coastguard Worker  * A function definition (a declaration plus an associated block of code).
27*c8dee2aaSAndroid Build Coastguard Worker  */
28*c8dee2aaSAndroid Build Coastguard Worker class FunctionDefinition final : public ProgramElement {
29*c8dee2aaSAndroid Build Coastguard Worker public:
30*c8dee2aaSAndroid Build Coastguard Worker     inline static constexpr Kind kIRNodeKind = Kind::kFunction;
31*c8dee2aaSAndroid Build Coastguard Worker 
FunctionDefinition(Position pos,const FunctionDeclaration * declaration,std::unique_ptr<Statement> body)32*c8dee2aaSAndroid Build Coastguard Worker     FunctionDefinition(Position pos,
33*c8dee2aaSAndroid Build Coastguard Worker                        const FunctionDeclaration* declaration,
34*c8dee2aaSAndroid Build Coastguard Worker                        std::unique_ptr<Statement> body)
35*c8dee2aaSAndroid Build Coastguard Worker             : INHERITED(pos, kIRNodeKind)
36*c8dee2aaSAndroid Build Coastguard Worker             , fDeclaration(declaration)
37*c8dee2aaSAndroid Build Coastguard Worker             , fBody(std::move(body)) {}
38*c8dee2aaSAndroid Build Coastguard Worker 
39*c8dee2aaSAndroid Build Coastguard Worker     /**
40*c8dee2aaSAndroid Build Coastguard Worker      * Coerces `return` statements to the return type of the function, and reports errors in the
41*c8dee2aaSAndroid Build Coastguard Worker      * function that can't be detected at the individual statement level:
42*c8dee2aaSAndroid Build Coastguard Worker      *
43*c8dee2aaSAndroid Build Coastguard Worker      *     - `break` and `continue` statements must be in reasonable places.
44*c8dee2aaSAndroid Build Coastguard Worker      *     - Non-void functions are required to return a value on all paths.
45*c8dee2aaSAndroid Build Coastguard Worker      *     - Vertex main() functions don't allow early returns.
46*c8dee2aaSAndroid Build Coastguard Worker      *     - Limits on overall stack size are enforced.
47*c8dee2aaSAndroid Build Coastguard Worker      *
48*c8dee2aaSAndroid Build Coastguard Worker      * This will return a FunctionDefinition even if an error is detected; this leads to better
49*c8dee2aaSAndroid Build Coastguard Worker      * diagnostics overall. (Returning null here leads to spurious "function 'f()' was not defined"
50*c8dee2aaSAndroid Build Coastguard Worker      * errors when trying to call a function with an error in it.)
51*c8dee2aaSAndroid Build Coastguard Worker      */
52*c8dee2aaSAndroid Build Coastguard Worker     static std::unique_ptr<FunctionDefinition> Convert(const Context& context,
53*c8dee2aaSAndroid Build Coastguard Worker                                                        Position pos,
54*c8dee2aaSAndroid Build Coastguard Worker                                                        const FunctionDeclaration& function,
55*c8dee2aaSAndroid Build Coastguard Worker                                                        std::unique_ptr<Statement> body);
56*c8dee2aaSAndroid Build Coastguard Worker 
57*c8dee2aaSAndroid Build Coastguard Worker     static std::unique_ptr<FunctionDefinition> Make(const Context& context,
58*c8dee2aaSAndroid Build Coastguard Worker                                                     Position pos,
59*c8dee2aaSAndroid Build Coastguard Worker                                                     const FunctionDeclaration& function,
60*c8dee2aaSAndroid Build Coastguard Worker                                                     std::unique_ptr<Statement> body);
61*c8dee2aaSAndroid Build Coastguard Worker 
declaration()62*c8dee2aaSAndroid Build Coastguard Worker     const FunctionDeclaration& declaration() const {
63*c8dee2aaSAndroid Build Coastguard Worker         return *fDeclaration;
64*c8dee2aaSAndroid Build Coastguard Worker     }
65*c8dee2aaSAndroid Build Coastguard Worker 
body()66*c8dee2aaSAndroid Build Coastguard Worker     std::unique_ptr<Statement>& body() {
67*c8dee2aaSAndroid Build Coastguard Worker         return fBody;
68*c8dee2aaSAndroid Build Coastguard Worker     }
69*c8dee2aaSAndroid Build Coastguard Worker 
body()70*c8dee2aaSAndroid Build Coastguard Worker     const std::unique_ptr<Statement>& body() const {
71*c8dee2aaSAndroid Build Coastguard Worker         return fBody;
72*c8dee2aaSAndroid Build Coastguard Worker     }
73*c8dee2aaSAndroid Build Coastguard Worker 
description()74*c8dee2aaSAndroid Build Coastguard Worker     std::string description() const override {
75*c8dee2aaSAndroid Build Coastguard Worker         return this->declaration().description() + " " + this->body()->description();
76*c8dee2aaSAndroid Build Coastguard Worker     }
77*c8dee2aaSAndroid Build Coastguard Worker 
78*c8dee2aaSAndroid Build Coastguard Worker private:
79*c8dee2aaSAndroid Build Coastguard Worker     const FunctionDeclaration* fDeclaration;
80*c8dee2aaSAndroid Build Coastguard Worker     std::unique_ptr<Statement> fBody;
81*c8dee2aaSAndroid Build Coastguard Worker 
82*c8dee2aaSAndroid Build Coastguard Worker     using INHERITED = ProgramElement;
83*c8dee2aaSAndroid Build Coastguard Worker };
84*c8dee2aaSAndroid Build Coastguard Worker 
85*c8dee2aaSAndroid Build Coastguard Worker }  // namespace SkSL
86*c8dee2aaSAndroid Build Coastguard Worker 
87*c8dee2aaSAndroid Build Coastguard Worker #endif
88