blob: 017f0b57bae7b9191d7aecc23182096d3b1993e3 [file] [log] [blame]
// Part of the Crubit project, under the Apache License v2.0 with LLVM
// Exceptions. See /LICENSE for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
#include "lifetime_annotations/lifetime_annotations.h"
#include <functional>
#include <optional>
#include <utility>
#include "third_party/absl/strings/str_cat.h"
#include "third_party/llvm/llvm-project/clang/include/clang/AST/ASTContext.h"
#include "third_party/llvm/llvm-project/clang/include/clang/AST/DeclCXX.h"
#include "third_party/llvm/llvm-project/clang/include/clang/Lex/Pragma.h"
#include "third_party/llvm/llvm-project/clang/include/clang/Lex/Preprocessor.h"
namespace devtools_rust {
static std::optional<FunctionLifetimes> ElidedLifetimes(
const clang::FunctionDecl* func,
std::function<Lifetime()> lifetime_factory) {
FunctionLifetimes result;
// Every input lifetime is assigned a distinct lifetime.
result.param_lifetimes.resize(func->getNumParams());
llvm::SmallVector<Lifetime> all_input_lifetimes;
for (unsigned i = 0; i < func->getNumParams(); ++i) {
const clang::ParmVarDecl* param = func->getParamDecl(i);
result.param_lifetimes[i] =
CreateLifetimesForType(param->getType(), lifetime_factory);
all_input_lifetimes.append(result.param_lifetimes[i]);
}
if (clang::isa<clang::CXXMethodDecl>(func)) {
Lifetime this_lifetime = lifetime_factory();
result.this_lifetimes.push_back(this_lifetime);
// If we have an implicit `this` parameter, its lifetime is assigned to all
// output lifetimes.
result.return_lifetimes = CreateLifetimesForType(
func->getReturnType(), [this_lifetime]() { return this_lifetime; });
return result;
}
// If we have no output lifetimes, there's nothing left to do.
if (CreateLifetimesForType(func->getReturnType(), Lifetime::Static).empty()) {
return result;
}
// If we have a single input lifetime, its lifetime is assigned to all output
// lifetimes.
if (all_input_lifetimes.size() == 1) {
result.return_lifetimes = CreateLifetimesForType(
func->getReturnType(),
[&all_input_lifetimes]() { return all_input_lifetimes[0]; });
return result;
}
return std::nullopt;
}
llvm::Expected<FunctionLifetimes> GetLifetimeAnnotations(
const clang::FunctionDecl* func, const LifetimeAnnotationContext& context,
LifetimeSymbolTable* symbol_table) {
// TODO(mboehme):
// - Add support for retrieving actual lifetime annotations (not just
// lifetimes implied by elision).
// - If we have multiple declarations of a function, make sure they are all
// annotated with the same lifetimes.
// For the time being, we only return elided lifetimes.
// See whether there are any lifetimes to be elided at all.
std::optional<FunctionLifetimes> elided_lifetimes =
ElidedLifetimes(func, Lifetime::Static);
if (!elided_lifetimes.has_value()) {
return llvm::createStringError(
llvm::inconvertibleErrorCode(),
absl::StrCat("Cannot determine output lifetimes for '",
func->getNameAsString(),
"' because it does not have exactly one input lifetime"));
}
// If the function has any elided lifetimes, we need to check if lifetime
// elision is enabled.
if (elided_lifetimes->ContainsLifetimes()) {
clang::SourceManager& source_manager =
func->getASTContext().getSourceManager();
clang::FileID file_id =
source_manager.getFileID(func->getSourceRange().getBegin());
if (!context.lifetime_elision_files.contains(file_id)) {
return llvm::createStringError(
llvm::inconvertibleErrorCode(),
absl::StrCat("Lifetime elision not enabled for '",
func->getNameAsString(), "'"));
}
}
// We know we're allowed to elide lifetimes, so produce the elided lifetimes
// again, but this time create new variables for the elided lifetimes.
if (symbol_table) {
elided_lifetimes = ElidedLifetimes(func, [symbol_table]() {
Lifetime lifetime = Lifetime::CreateVariable();
symbol_table->LookupLifetimeAndMaybeDeclare(lifetime);
return lifetime;
});
} else {
elided_lifetimes = ElidedLifetimes(func, Lifetime::CreateVariable);
}
assert(elided_lifetimes.has_value());
return std::move(elided_lifetimes).value();
}
namespace {
class LifetimeElisionPragmaHandler : public clang::PragmaHandler {
public:
explicit LifetimeElisionPragmaHandler(
std::shared_ptr<LifetimeAnnotationContext> context)
: clang::PragmaHandler("lifetime_elision"), context_(context) {}
void HandlePragma(clang::Preprocessor& preprocessor,
clang::PragmaIntroducer introducer,
clang::Token&) override {
clang::SourceManager& source_manager = preprocessor.getSourceManager();
clang::FileID file_id = source_manager.getFileID(introducer.Loc);
context_->lifetime_elision_files.insert(file_id);
}
private:
std::shared_ptr<LifetimeAnnotationContext> context_;
};
} // namespace
void AddLifetimeAnnotationHandlers(
clang::CompilerInstance& compiler,
std::shared_ptr<LifetimeAnnotationContext> context) {
// Preprocessor takes ownership of the handler.
compiler.getPreprocessor().AddPragmaHandler(
"clang", new LifetimeElisionPragmaHandler(context));
}
llvm::SmallVector<clang::TypeLoc> GetTemplateArgs(clang::TypeLoc type_loc) {
llvm::SmallVector<clang::TypeLoc> args;
if (auto template_specialization_type_loc =
type_loc.getAs<clang::TemplateSpecializationTypeLoc>()) {
for (unsigned i = 0; i < template_specialization_type_loc.getNumArgs();
++i) {
args.push_back(template_specialization_type_loc.getArgLoc(i)
.getTypeSourceInfo()
->getTypeLoc());
}
} else if (auto dependent_template_specialization_type_loc =
type_loc
.getAs<clang::DependentTemplateSpecializationTypeLoc>()) {
// TODO(mboehme): Where does this occur exactly? Do we need to be handling
// it?
// AFAICT, this happens if we're looking at a dependent template name
// (https://en.cppreference.com/w/cpp/language/dependent_name), which
// probably means that this can only happen in template definitions (as
// opposed to template instantiations), and we aren't analyzing those for
// now. At the least, I haven't been able to trigger this case from a test.
// Putting an assertion in here so that we notice this case if it does come
// up.
assert(false);
for (unsigned i = 0;
i < dependent_template_specialization_type_loc.getNumArgs(); ++i) {
args.push_back(dependent_template_specialization_type_loc.getArgLoc(i)
.getTypeSourceInfo()
->getTypeLoc());
}
}
return args;
}
} // namespace devtools_rust