blob: c5517ee5ecb6952e9d33913084ed5ab6af99f6d5 [file] [log] [blame]
/*
* Copyright 2016 Google Inc.
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#ifndef SKSL_FUNCTIONDEFINITION
#define SKSL_FUNCTIONDEFINITION
#include "include/private/SkSLProgramElement.h"
#include "include/private/SkSLStatement.h"
#include "include/sksl/SkSLPosition.h"
#include "src/sksl/ir/SkSLFunctionDeclaration.h"
#include <memory>
#include <string>
#include <unordered_set>
#include <utility>
namespace SkSL {
class Context;
/**
* A function definition (a declaration plus an associated block of code).
*/
class FunctionDefinition final : public ProgramElement {
public:
inline static constexpr Kind kProgramElementKind = Kind::kFunction;
using FunctionSet = std::unordered_set<const FunctionDeclaration*>;
FunctionDefinition(Position pos, const FunctionDeclaration* declaration, bool builtin,
std::unique_ptr<Statement> body, FunctionSet referencedBuiltinFunctions)
: INHERITED(pos, kProgramElementKind)
, fDeclaration(declaration)
, fBuiltin(builtin)
, fBody(std::move(body))
, fReferencedBuiltinFunctions(std::move(referencedBuiltinFunctions)) {}
/**
* Coerces `return` statements to the return type of the function, and reports errors in the
* function that can't be detected at the individual statement level:
* - `break` and `continue` statements must be in reasonable places.
* - non-void functions are required to return a value on all paths.
* - vertex main() functions don't allow early returns.
*
* This will return a FunctionDefinition even if an error is detected; this leads to better
* diagnostics overall. (Returning null here leads to spurious "function 'f()' was not defined"
* errors when trying to call a function with an error in it.)
*/
static std::unique_ptr<FunctionDefinition> Convert(const Context& context,
Position pos,
const FunctionDeclaration& function,
std::unique_ptr<Statement> body,
bool builtin);
const FunctionDeclaration& declaration() const {
return *fDeclaration;
}
bool isBuiltin() const {
return fBuiltin;
}
std::unique_ptr<Statement>& body() {
return fBody;
}
const std::unique_ptr<Statement>& body() const {
return fBody;
}
const FunctionSet& referencedBuiltinFunctions() const {
return fReferencedBuiltinFunctions;
}
std::unique_ptr<ProgramElement> clone() const override {
return std::make_unique<FunctionDefinition>(fPosition, &this->declaration(),
/*builtin=*/false, this->body()->clone(),
this->referencedBuiltinFunctions());
}
std::string description() const override {
return this->declaration().description() + " " + this->body()->description();
}
private:
const FunctionDeclaration* fDeclaration;
bool fBuiltin;
std::unique_ptr<Statement> fBody;
// We track the builtin functions we reference so that we can ensure that all of them end up
// copied into the final output.
FunctionSet fReferencedBuiltinFunctions;
using INHERITED = ProgramElement;
};
} // namespace SkSL
#endif