forked from OSchip/llvm-project
263 lines
9.3 KiB
C++
263 lines
9.3 KiB
C++
//===--- TestTU.cpp - Scratch source files for testing --------------------===//
|
|
//
|
|
// 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 "TestTU.h"
|
|
#include "Compiler.h"
|
|
#include "Diagnostics.h"
|
|
#include "TestFS.h"
|
|
#include "index/FileIndex.h"
|
|
#include "index/MemIndex.h"
|
|
#include "clang/AST/RecursiveASTVisitor.h"
|
|
#include "clang/Basic/Diagnostic.h"
|
|
#include "clang/Frontend/CompilerInvocation.h"
|
|
#include "clang/Frontend/Utils.h"
|
|
#include "llvm/ADT/ScopeExit.h"
|
|
#include "llvm/Support/ScopedPrinter.h"
|
|
#include "llvm/Support/raw_ostream.h"
|
|
#include <cstdlib>
|
|
|
|
namespace clang {
|
|
namespace clangd {
|
|
|
|
ParseInputs TestTU::inputs(MockFS &FS) const {
|
|
std::string FullFilename = testPath(Filename),
|
|
FullHeaderName = testPath(HeaderFilename),
|
|
ImportThunk = testPath("import_thunk.h");
|
|
// We want to implicitly include HeaderFilename without messing up offsets.
|
|
// -include achieves this, but sometimes we want #import (to simulate a header
|
|
// guard without messing up offsets). In this case, use an intermediate file.
|
|
std::string ThunkContents = "#import \"" + FullHeaderName + "\"\n";
|
|
|
|
FS.Files = AdditionalFiles;
|
|
FS.Files[FullFilename] = Code;
|
|
FS.Files[FullHeaderName] = HeaderCode;
|
|
FS.Files[ImportThunk] = ThunkContents;
|
|
|
|
ParseInputs Inputs;
|
|
Inputs.FeatureModules = FeatureModules;
|
|
auto &Argv = Inputs.CompileCommand.CommandLine;
|
|
Argv = {"clang"};
|
|
// FIXME: this shouldn't need to be conditional, but it breaks a
|
|
// GoToDefinition test for some reason (getMacroArgExpandedLocation fails).
|
|
if (!HeaderCode.empty()) {
|
|
Argv.push_back("-include");
|
|
Argv.push_back(ImplicitHeaderGuard ? ImportThunk : FullHeaderName);
|
|
// ms-compatibility changes the meaning of #import.
|
|
// The default is OS-dependent (on on windows), ensure it's off.
|
|
if (ImplicitHeaderGuard)
|
|
Inputs.CompileCommand.CommandLine.push_back("-fno-ms-compatibility");
|
|
}
|
|
Argv.insert(Argv.end(), ExtraArgs.begin(), ExtraArgs.end());
|
|
// Put the file name at the end -- this allows the extra arg (-xc++) to
|
|
// override the language setting.
|
|
Argv.push_back(FullFilename);
|
|
Inputs.CompileCommand.Filename = FullFilename;
|
|
Inputs.CompileCommand.Directory = testRoot();
|
|
Inputs.Contents = Code;
|
|
if (OverlayRealFileSystemForModules)
|
|
FS.OverlayRealFileSystemForModules = true;
|
|
Inputs.TFS = &FS;
|
|
Inputs.Opts = ParseOptions();
|
|
if (ClangTidyProvider)
|
|
Inputs.ClangTidyProvider = ClangTidyProvider;
|
|
Inputs.Index = ExternalIndex;
|
|
return Inputs;
|
|
}
|
|
|
|
void initializeModuleCache(CompilerInvocation &CI) {
|
|
llvm::SmallString<128> ModuleCachePath;
|
|
if (llvm::sys::fs::createUniqueDirectory("module-cache", ModuleCachePath)) {
|
|
llvm::errs() << "Failed to create temp directory for module-cache";
|
|
std::abort();
|
|
}
|
|
CI.getHeaderSearchOpts().ModuleCachePath = ModuleCachePath.c_str();
|
|
}
|
|
|
|
void deleteModuleCache(const std::string ModuleCachePath) {
|
|
if (!ModuleCachePath.empty()) {
|
|
if (llvm::sys::fs::remove_directories(ModuleCachePath)) {
|
|
llvm::errs() << "Failed to delete temp directory for module-cache";
|
|
std::abort();
|
|
}
|
|
}
|
|
}
|
|
|
|
std::shared_ptr<const PreambleData>
|
|
TestTU::preamble(PreambleParsedCallback PreambleCallback) const {
|
|
MockFS FS;
|
|
auto Inputs = inputs(FS);
|
|
IgnoreDiagnostics Diags;
|
|
auto CI = buildCompilerInvocation(Inputs, Diags);
|
|
assert(CI && "Failed to build compilation invocation.");
|
|
if (OverlayRealFileSystemForModules)
|
|
initializeModuleCache(*CI);
|
|
auto ModuleCacheDeleter = llvm::make_scope_exit(
|
|
std::bind(deleteModuleCache, CI->getHeaderSearchOpts().ModuleCachePath));
|
|
return clang::clangd::buildPreamble(testPath(Filename), *CI, Inputs,
|
|
/*StoreInMemory=*/true, PreambleCallback);
|
|
}
|
|
|
|
ParsedAST TestTU::build() const {
|
|
MockFS FS;
|
|
auto Inputs = inputs(FS);
|
|
StoreDiags Diags;
|
|
auto CI = buildCompilerInvocation(Inputs, Diags);
|
|
assert(CI && "Failed to build compilation invocation.");
|
|
if (OverlayRealFileSystemForModules)
|
|
initializeModuleCache(*CI);
|
|
auto ModuleCacheDeleter = llvm::make_scope_exit(
|
|
std::bind(deleteModuleCache, CI->getHeaderSearchOpts().ModuleCachePath));
|
|
|
|
auto Preamble = clang::clangd::buildPreamble(testPath(Filename), *CI, Inputs,
|
|
/*StoreInMemory=*/true,
|
|
/*PreambleCallback=*/nullptr);
|
|
auto AST = ParsedAST::build(testPath(Filename), Inputs, std::move(CI),
|
|
Diags.take(), Preamble);
|
|
if (!AST.hasValue()) {
|
|
llvm::errs() << "Failed to build code:\n" << Code;
|
|
std::abort();
|
|
}
|
|
assert(AST->getDiagnostics() &&
|
|
"TestTU should always build an AST with a fresh Preamble");
|
|
// Check for error diagnostics and report gtest failures (unless expected).
|
|
// This guards against accidental syntax errors silently subverting tests.
|
|
// error-ok is awfully primitive - using clang -verify would be nicer.
|
|
// Ownership and layering makes it pretty hard.
|
|
bool ErrorOk = [&, this] {
|
|
llvm::StringLiteral Marker = "error-ok";
|
|
if (llvm::StringRef(Code).contains(Marker) ||
|
|
llvm::StringRef(HeaderCode).contains(Marker))
|
|
return true;
|
|
for (const auto &KV : this->AdditionalFiles)
|
|
if (llvm::StringRef(KV.second).contains(Marker))
|
|
return true;
|
|
return false;
|
|
}();
|
|
if (!ErrorOk) {
|
|
// We always build AST with a fresh preamble in TestTU.
|
|
for (const auto &D : *AST->getDiagnostics())
|
|
if (D.Severity >= DiagnosticsEngine::Error) {
|
|
llvm::errs()
|
|
<< "TestTU failed to build (suppress with /*error-ok*/): \n"
|
|
<< D << "\n\nFor code:\n"
|
|
<< Code;
|
|
std::abort(); // Stop after first error for simplicity.
|
|
}
|
|
}
|
|
return std::move(*AST);
|
|
}
|
|
|
|
SymbolSlab TestTU::headerSymbols() const {
|
|
auto AST = build();
|
|
return std::get<0>(indexHeaderSymbols(/*Version=*/"null", AST.getASTContext(),
|
|
AST.getPreprocessorPtr(),
|
|
AST.getCanonicalIncludes()));
|
|
}
|
|
|
|
RefSlab TestTU::headerRefs() const {
|
|
auto AST = build();
|
|
return std::get<1>(indexMainDecls(AST));
|
|
}
|
|
|
|
std::unique_ptr<SymbolIndex> TestTU::index() const {
|
|
auto AST = build();
|
|
auto Idx = std::make_unique<FileIndex>();
|
|
Idx->updatePreamble(testPath(Filename), /*Version=*/"null",
|
|
AST.getASTContext(), AST.getPreprocessorPtr(),
|
|
AST.getCanonicalIncludes());
|
|
Idx->updateMain(testPath(Filename), AST);
|
|
return std::move(Idx);
|
|
}
|
|
|
|
const Symbol &findSymbol(const SymbolSlab &Slab, llvm::StringRef QName) {
|
|
const Symbol *Result = nullptr;
|
|
for (const Symbol &S : Slab) {
|
|
if (QName != (S.Scope + S.Name).str())
|
|
continue;
|
|
if (Result) {
|
|
llvm::errs() << "Multiple symbols named " << QName << ":\n"
|
|
<< *Result << "\n---\n"
|
|
<< S;
|
|
assert(false && "QName is not unique");
|
|
}
|
|
Result = &S;
|
|
}
|
|
if (!Result) {
|
|
llvm::errs() << "No symbol named " << QName << " in "
|
|
<< llvm::to_string(Slab);
|
|
assert(false && "No symbol with QName");
|
|
}
|
|
return *Result;
|
|
}
|
|
|
|
// RAII scoped class to disable TraversalScope for a ParsedAST.
|
|
class TraverseHeadersToo {
|
|
ASTContext &Ctx;
|
|
std::vector<Decl *> ScopeToRestore;
|
|
|
|
public:
|
|
TraverseHeadersToo(ParsedAST &AST)
|
|
: Ctx(AST.getASTContext()), ScopeToRestore(Ctx.getTraversalScope()) {
|
|
Ctx.setTraversalScope({Ctx.getTranslationUnitDecl()});
|
|
}
|
|
~TraverseHeadersToo() { Ctx.setTraversalScope(std::move(ScopeToRestore)); }
|
|
};
|
|
|
|
const NamedDecl &findDecl(ParsedAST &AST, llvm::StringRef QName) {
|
|
auto &Ctx = AST.getASTContext();
|
|
auto LookupDecl = [&Ctx](const DeclContext &Scope,
|
|
llvm::StringRef Name) -> const NamedDecl & {
|
|
auto LookupRes = Scope.lookup(DeclarationName(&Ctx.Idents.get(Name)));
|
|
assert(!LookupRes.empty() && "Lookup failed");
|
|
assert(LookupRes.isSingleResult() && "Lookup returned multiple results");
|
|
return *LookupRes.front();
|
|
};
|
|
|
|
const DeclContext *Scope = Ctx.getTranslationUnitDecl();
|
|
|
|
StringRef Cur, Rest;
|
|
for (std::tie(Cur, Rest) = QName.split("::"); !Rest.empty();
|
|
std::tie(Cur, Rest) = Rest.split("::")) {
|
|
Scope = &cast<DeclContext>(LookupDecl(*Scope, Cur));
|
|
}
|
|
return LookupDecl(*Scope, Cur);
|
|
}
|
|
|
|
const NamedDecl &findDecl(ParsedAST &AST,
|
|
std::function<bool(const NamedDecl &)> Filter) {
|
|
TraverseHeadersToo Too(AST);
|
|
struct Visitor : RecursiveASTVisitor<Visitor> {
|
|
decltype(Filter) F;
|
|
llvm::SmallVector<const NamedDecl *, 1> Decls;
|
|
bool VisitNamedDecl(const NamedDecl *ND) {
|
|
if (F(*ND))
|
|
Decls.push_back(ND);
|
|
return true;
|
|
}
|
|
} Visitor;
|
|
Visitor.F = Filter;
|
|
Visitor.TraverseDecl(AST.getASTContext().getTranslationUnitDecl());
|
|
if (Visitor.Decls.size() != 1) {
|
|
llvm::errs() << Visitor.Decls.size() << " symbols matched.";
|
|
assert(Visitor.Decls.size() == 1);
|
|
}
|
|
return *Visitor.Decls.front();
|
|
}
|
|
|
|
const NamedDecl &findUnqualifiedDecl(ParsedAST &AST, llvm::StringRef Name) {
|
|
return findDecl(AST, [Name](const NamedDecl &ND) {
|
|
if (auto *ID = ND.getIdentifier())
|
|
if (ID->getName() == Name)
|
|
return true;
|
|
return false;
|
|
});
|
|
}
|
|
|
|
} // namespace clangd
|
|
} // namespace clang
|