//===--- TestTU.cpp - Scratch source files for testing --------------------===// // // The LLVM Compiler Infrastructure // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// #include "TestTU.h" #include "TestFS.h" #include "index/FileIndex.h" #include "index/MemIndex.h" #include "clang/AST/RecursiveASTVisitor.h" #include "clang/Frontend/CompilerInvocation.h" #include "clang/Frontend/PCHContainerOperations.h" #include "clang/Frontend/Utils.h" namespace clang { namespace clangd { ParsedAST TestTU::build() const { std::string FullFilename = testPath(Filename), FullHeaderName = testPath(HeaderFilename); std::vector Cmd = {"clang", FullFilename.c_str()}; // FIXME: this shouldn't need to be conditional, but it breaks a // GoToDefinition test for some reason (getMacroArgExpandedLocation fails). if (!HeaderCode.empty()) { Cmd.push_back("-include"); Cmd.push_back(FullHeaderName.c_str()); } Cmd.insert(Cmd.end(), ExtraArgs.begin(), ExtraArgs.end()); ParseInputs Inputs; Inputs.CompileCommand.Filename = FullFilename; Inputs.CompileCommand.CommandLine = {Cmd.begin(), Cmd.end()}; Inputs.CompileCommand.Directory = testRoot(); Inputs.Contents = Code; Inputs.FS = buildTestFS({{FullFilename, Code}, {FullHeaderName, HeaderCode}}); auto PCHs = std::make_shared(); auto CI = buildCompilerInvocation(Inputs); assert(CI && "Failed to build compilation invocation."); auto Preamble = buildPreamble(FullFilename, *CI, /*OldPreamble=*/nullptr, /*OldCompileCommand=*/Inputs.CompileCommand, Inputs, PCHs, /*StoreInMemory=*/true, /*PreambleCallback=*/nullptr); auto AST = buildAST(FullFilename, createInvocationFromCommandLine(Cmd), Inputs, Preamble, PCHs); if (!AST.hasValue()) { ADD_FAILURE() << "Failed to build code:\n" << Code; llvm_unreachable("Failed to build TestTU!"); } return std::move(*AST); } SymbolSlab TestTU::headerSymbols() const { auto AST = build(); return indexHeaderSymbols(AST.getASTContext(), AST.getPreprocessorPtr()); } std::unique_ptr TestTU::index() const { auto AST = build(); auto Idx = llvm::make_unique(/*UseDex=*/true); Idx->updatePreamble(Filename, AST.getASTContext(), AST.getPreprocessorPtr()); Idx->updateMain(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) { ADD_FAILURE() << "Multiple symbols named " << QName << ":\n" << *Result << "\n---\n" << S; assert(false && "QName is not unique"); } Result = &S; } if (!Result) { ADD_FAILURE() << "No symbol named " << QName << " in " << ::testing::PrintToString(Slab); assert(false && "No symbol with QName"); } return *Result; } const NamedDecl &findDecl(ParsedAST &AST, llvm::StringRef QName) { llvm::SmallVector Components; QName.split(Components, "::"); 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.size() == 1 && "Lookup returned multiple results"); return *LookupRes.front(); }; const DeclContext *Scope = Ctx.getTranslationUnitDecl(); for (auto NameIt = Components.begin(), End = Components.end() - 1; NameIt != End; ++NameIt) { Scope = &cast(LookupDecl(*Scope, *NameIt)); } return LookupDecl(*Scope, Components.back()); } const NamedDecl &findDecl(ParsedAST &AST, std::function Filter) { struct Visitor : RecursiveASTVisitor { decltype(Filter) F; llvm::SmallVector 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) { ADD_FAILURE() << 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