blob: 4f3c40d1eb13ec2110b46da1de068913f3ad1ecc [file] [log] [blame]
//===--- Tweak.cpp -----------------------------------------------*- C++-*-===//
//
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
// See https://llvm.org/LICENSE.txt for license information.
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
//
//===----------------------------------------------------------------------===//
#include "Tweak.h"
#include "Logger.h"
#include "Path.h"
#include "SourceCode.h"
#include "llvm/ADT/None.h"
#include "llvm/ADT/Optional.h"
#include "llvm/ADT/STLExtras.h"
#include "llvm/ADT/StringMap.h"
#include "llvm/ADT/StringRef.h"
#include "llvm/Support/Error.h"
#include "llvm/Support/Registry.h"
#include <functional>
#include <memory>
#include <utility>
LLVM_INSTANTIATE_REGISTRY(llvm::Registry<clang::clangd::Tweak>)
namespace clang {
namespace clangd {
/// A handy typedef to save some typing.
typedef llvm::Registry<Tweak> TweakRegistry;
namespace {
/// Asserts invariants on TweakRegistry. No-op with assertion disabled.
void validateRegistry() {
#ifndef NDEBUG
llvm::StringSet<> Seen;
for (const auto &E : TweakRegistry::entries()) {
// REGISTER_TWEAK ensures E.getName() is equal to the tweak class name.
// We check that id() matches it.
assert(E.instantiate()->id() == E.getName() &&
"id should be equal to class name");
assert(Seen.try_emplace(E.getName()).second && "duplicate check id");
}
#endif
}
} // namespace
Tweak::Selection::Selection(ParsedAST &AST, unsigned RangeBegin,
unsigned RangeEnd)
: AST(AST), SelectionBegin(RangeBegin), SelectionEnd(RangeEnd),
ASTSelection(AST.getASTContext(), AST.getTokens(), RangeBegin, RangeEnd) {
auto &SM = AST.getSourceManager();
Code = SM.getBufferData(SM.getMainFileID());
Cursor = SM.getComposedLoc(SM.getMainFileID(), RangeBegin);
}
std::vector<std::unique_ptr<Tweak>>
prepareTweaks(const Tweak::Selection &S,
llvm::function_ref<bool(const Tweak &)> Filter) {
validateRegistry();
std::vector<std::unique_ptr<Tweak>> Available;
for (const auto &E : TweakRegistry::entries()) {
std::unique_ptr<Tweak> T = E.instantiate();
if (!Filter(*T) || !T->prepare(S))
continue;
Available.push_back(std::move(T));
}
// Ensure deterministic order of the results.
llvm::sort(Available,
[](const std::unique_ptr<Tweak> &L,
const std::unique_ptr<Tweak> &R) { return L->id() < R->id(); });
return Available;
}
llvm::Expected<std::unique_ptr<Tweak>> prepareTweak(StringRef ID,
const Tweak::Selection &S) {
auto It = llvm::find_if(
TweakRegistry::entries(),
[ID](const TweakRegistry::entry &E) { return E.getName() == ID; });
if (It == TweakRegistry::end())
return llvm::createStringError(llvm::inconvertibleErrorCode(),
"id of the tweak is invalid");
std::unique_ptr<Tweak> T = It->instantiate();
if (!T->prepare(S))
return llvm::createStringError(llvm::inconvertibleErrorCode(),
"failed to prepare() a check");
return std::move(T);
}
llvm::Expected<std::pair<Path, Edit>>
Tweak::Effect::fileEdit(const SourceManager &SM, FileID FID,
tooling::Replacements Replacements) {
Edit Ed(SM.getBufferData(FID), std::move(Replacements));
if (auto FilePath = getCanonicalPath(SM.getFileEntryForID(FID), SM))
return std::make_pair(*FilePath, std::move(Ed));
return llvm::createStringError(
llvm::inconvertibleErrorCode(),
"Failed to get absolute path for edited file: " +
SM.getFileEntryForID(FID)->getName());
}
llvm::Expected<Tweak::Effect>
Tweak::Effect::mainFileEdit(const SourceManager &SM,
tooling::Replacements Replacements) {
auto PathAndEdit = fileEdit(SM, SM.getMainFileID(), std::move(Replacements));
if (!PathAndEdit)
return PathAndEdit.takeError();
Tweak::Effect E;
E.ApplyEdits.try_emplace(PathAndEdit->first, PathAndEdit->second);
return E;
}
} // namespace clangd
} // namespace clang