| //===--- ParsedAST.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 "ParsedAST.h" |
| #include "../clang-tidy/ClangTidyCheck.h" |
| #include "../clang-tidy/ClangTidyDiagnosticConsumer.h" |
| #include "../clang-tidy/ClangTidyModule.h" |
| #include "../clang-tidy/ClangTidyModuleRegistry.h" |
| #include "../clang-tidy/ClangTidyOptions.h" |
| #include "AST.h" |
| #include "CollectMacros.h" |
| #include "Compiler.h" |
| #include "Config.h" |
| #include "Diagnostics.h" |
| #include "Feature.h" |
| #include "FeatureModule.h" |
| #include "Headers.h" |
| #include "HeuristicResolver.h" |
| #include "IncludeCleaner.h" |
| #include "IncludeFixer.h" |
| #include "Preamble.h" |
| #include "SourceCode.h" |
| #include "TidyProvider.h" |
| #include "clang-include-cleaner/Record.h" |
| #include "index/Symbol.h" |
| #include "support/Logger.h" |
| #include "support/Path.h" |
| #include "support/Trace.h" |
| #include "clang/AST/ASTContext.h" |
| #include "clang/AST/Decl.h" |
| #include "clang/AST/DeclGroup.h" |
| #include "clang/AST/ExternalASTSource.h" |
| #include "clang/ASTMatchers/ASTMatchFinder.h" |
| #include "clang/Basic/Diagnostic.h" |
| #include "clang/Basic/DiagnosticIDs.h" |
| #include "clang/Basic/DiagnosticSema.h" |
| #include "clang/Basic/FileEntry.h" |
| #include "clang/Basic/LLVM.h" |
| #include "clang/Basic/LangOptions.h" |
| #include "clang/Basic/SourceLocation.h" |
| #include "clang/Basic/SourceManager.h" |
| #include "clang/Basic/TokenKinds.h" |
| #include "clang/Frontend/CompilerInstance.h" |
| #include "clang/Frontend/CompilerInvocation.h" |
| #include "clang/Frontend/FrontendActions.h" |
| #include "clang/Frontend/FrontendOptions.h" |
| #include "clang/Frontend/PrecompiledPreamble.h" |
| #include "clang/Lex/Lexer.h" |
| #include "clang/Lex/PPCallbacks.h" |
| #include "clang/Lex/Preprocessor.h" |
| #include "clang/Serialization/ASTWriter.h" |
| #include "clang/Tooling/CompilationDatabase.h" |
| #include "clang/Tooling/Core/Diagnostic.h" |
| #include "clang/Tooling/Syntax/Tokens.h" |
| #include "llvm/ADT/ArrayRef.h" |
| #include "llvm/ADT/DenseMap.h" |
| #include "llvm/ADT/DenseSet.h" |
| #include "llvm/ADT/STLExtras.h" |
| #include "llvm/ADT/STLFunctionalExtras.h" |
| #include "llvm/ADT/SmallVector.h" |
| #include "llvm/ADT/StringRef.h" |
| #include "llvm/Support/Error.h" |
| #include "llvm/Support/MemoryBuffer.h" |
| #include <cassert> |
| #include <cstddef> |
| #include <iterator> |
| #include <memory> |
| #include <optional> |
| #include <string> |
| #include <tuple> |
| #include <utility> |
| #include <vector> |
| |
| // Force the linker to link in Clang-tidy modules. |
| // clangd doesn't support the static analyzer. |
| #if CLANGD_TIDY_CHECKS |
| #define CLANG_TIDY_DISABLE_STATIC_ANALYZER_CHECKS |
| #include "../clang-tidy/ClangTidyForceLinker.h" |
| #endif |
| |
| namespace clang { |
| namespace clangd { |
| namespace { |
| |
| template <class T> std::size_t getUsedBytes(const std::vector<T> &Vec) { |
| return Vec.capacity() * sizeof(T); |
| } |
| |
| class DeclTrackingASTConsumer : public ASTConsumer { |
| public: |
| DeclTrackingASTConsumer(std::vector<Decl *> &TopLevelDecls) |
| : TopLevelDecls(TopLevelDecls) {} |
| |
| bool HandleTopLevelDecl(DeclGroupRef DG) override { |
| for (Decl *D : DG) { |
| auto &SM = D->getASTContext().getSourceManager(); |
| if (!isInsideMainFile(D->getLocation(), SM)) |
| continue; |
| if (const NamedDecl *ND = dyn_cast<NamedDecl>(D)) |
| if (isImplicitTemplateInstantiation(ND)) |
| continue; |
| |
| // ObjCMethodDecl are not actually top-level decls. |
| if (isa<ObjCMethodDecl>(D)) |
| continue; |
| |
| TopLevelDecls.push_back(D); |
| } |
| return true; |
| } |
| |
| private: |
| std::vector<Decl *> &TopLevelDecls; |
| }; |
| |
| class ClangdFrontendAction : public SyntaxOnlyAction { |
| public: |
| std::vector<Decl *> takeTopLevelDecls() { return std::move(TopLevelDecls); } |
| |
| protected: |
| std::unique_ptr<ASTConsumer> |
| CreateASTConsumer(CompilerInstance &CI, llvm::StringRef InFile) override { |
| return std::make_unique<DeclTrackingASTConsumer>(/*ref*/ TopLevelDecls); |
| } |
| |
| private: |
| std::vector<Decl *> TopLevelDecls; |
| }; |
| |
| // When using a preamble, only preprocessor events outside its bounds are seen. |
| // This is almost what we want: replaying transitive preprocessing wastes time. |
| // However this confuses clang-tidy checks: they don't see any #includes! |
| // So we replay the *non-transitive* #includes that appear in the main-file. |
| // It would be nice to replay other events (macro definitions, ifdefs etc) but |
| // this addresses the most common cases fairly cheaply. |
| class ReplayPreamble : private PPCallbacks { |
| public: |
| // Attach preprocessor hooks such that preamble events will be injected at |
| // the appropriate time. |
| // Events will be delivered to the *currently registered* PP callbacks. |
| static void attach(std::vector<Inclusion> Includes, CompilerInstance &Clang, |
| const PreambleBounds &PB) { |
| auto &PP = Clang.getPreprocessor(); |
| auto *ExistingCallbacks = PP.getPPCallbacks(); |
| // No need to replay events if nobody is listening. |
| if (!ExistingCallbacks) |
| return; |
| PP.addPPCallbacks(std::unique_ptr<PPCallbacks>(new ReplayPreamble( |
| std::move(Includes), ExistingCallbacks, Clang.getSourceManager(), PP, |
| Clang.getLangOpts(), PB))); |
| // We're relying on the fact that addPPCallbacks keeps the old PPCallbacks |
| // around, creating a chaining wrapper. Guard against other implementations. |
| assert(PP.getPPCallbacks() != ExistingCallbacks && |
| "Expected chaining implementation"); |
| } |
| |
| private: |
| ReplayPreamble(std::vector<Inclusion> Includes, PPCallbacks *Delegate, |
| const SourceManager &SM, Preprocessor &PP, |
| const LangOptions &LangOpts, const PreambleBounds &PB) |
| : Includes(std::move(Includes)), Delegate(Delegate), SM(SM), PP(PP) { |
| // Only tokenize the preamble section of the main file, as we are not |
| // interested in the rest of the tokens. |
| MainFileTokens = syntax::tokenize( |
| syntax::FileRange(SM.getMainFileID(), 0, PB.Size), SM, LangOpts); |
| } |
| |
| // In a normal compile, the preamble traverses the following structure: |
| // |
| // mainfile.cpp |
| // <built-in> |
| // ... macro definitions like __cplusplus ... |
| // <command-line> |
| // ... macro definitions for args like -Dfoo=bar ... |
| // "header1.h" |
| // ... header file contents ... |
| // "header2.h" |
| // ... header file contents ... |
| // ... main file contents ... |
| // |
| // When using a preamble, the "header1" and "header2" subtrees get skipped. |
| // We insert them right after the built-in header, which still appears. |
| void FileChanged(SourceLocation Loc, FileChangeReason Reason, |
| SrcMgr::CharacteristicKind Kind, FileID PrevFID) override { |
| // It'd be nice if there was a better way to identify built-in headers... |
| if (Reason == FileChangeReason::ExitFile && |
| SM.getBufferOrFake(PrevFID).getBufferIdentifier() == "<built-in>") |
| replay(); |
| } |
| |
| void replay() { |
| for (const auto &Inc : Includes) { |
| OptionalFileEntryRef File; |
| if (Inc.Resolved != "") |
| File = expectedToOptional(SM.getFileManager().getFileRef(Inc.Resolved)); |
| |
| // Re-lex the #include directive to find its interesting parts. |
| auto HashLoc = SM.getComposedLoc(SM.getMainFileID(), Inc.HashOffset); |
| auto HashTok = llvm::partition_point(MainFileTokens, |
| [&HashLoc](const syntax::Token &T) { |
| return T.location() < HashLoc; |
| }); |
| assert(HashTok != MainFileTokens.end() && HashTok->kind() == tok::hash); |
| |
| auto IncludeTok = std::next(HashTok); |
| assert(IncludeTok != MainFileTokens.end()); |
| |
| auto FileTok = std::next(IncludeTok); |
| assert(FileTok != MainFileTokens.end()); |
| |
| // Create a fake import/include token, none of the callers seem to care |
| // about clang::Token::Flags. |
| Token SynthesizedIncludeTok; |
| SynthesizedIncludeTok.startToken(); |
| SynthesizedIncludeTok.setLocation(IncludeTok->location()); |
| SynthesizedIncludeTok.setLength(IncludeTok->length()); |
| SynthesizedIncludeTok.setKind(tok::raw_identifier); |
| SynthesizedIncludeTok.setRawIdentifierData(IncludeTok->text(SM).data()); |
| PP.LookUpIdentifierInfo(SynthesizedIncludeTok); |
| |
| // Same here, create a fake one for Filename, including angles or quotes. |
| Token SynthesizedFilenameTok; |
| SynthesizedFilenameTok.startToken(); |
| SynthesizedFilenameTok.setLocation(FileTok->location()); |
| // Note that we can't make use of FileTok->length/text in here as in the |
| // case of angled includes this will contain tok::less instead of |
| // filename. Whereas Inc.Written contains the full header name including |
| // quotes/angles. |
| SynthesizedFilenameTok.setLength(Inc.Written.length()); |
| SynthesizedFilenameTok.setKind(tok::header_name); |
| SynthesizedFilenameTok.setLiteralData(Inc.Written.data()); |
| |
| llvm::StringRef WrittenFilename = |
| llvm::StringRef(Inc.Written).drop_front().drop_back(); |
| Delegate->InclusionDirective( |
| HashTok->location(), SynthesizedIncludeTok, WrittenFilename, |
| Inc.Written.front() == '<', |
| syntax::FileRange(SM, SynthesizedFilenameTok.getLocation(), |
| SynthesizedFilenameTok.getEndLoc()) |
| .toCharRange(SM), |
| File, "SearchPath", "RelPath", |
| /*SuggestedModule=*/nullptr, /*ModuleImported=*/false, Inc.FileKind); |
| if (File) |
| Delegate->FileSkipped(*File, SynthesizedFilenameTok, Inc.FileKind); |
| } |
| } |
| |
| const std::vector<Inclusion> Includes; |
| PPCallbacks *Delegate; |
| const SourceManager &SM; |
| Preprocessor &PP; |
| std::vector<syntax::Token> MainFileTokens; |
| }; |
| |
| // Filter for clang diagnostics groups enabled by CTOptions.Checks. |
| // |
| // These are check names like clang-diagnostics-unused. |
| // Note that unlike -Wunused, clang-diagnostics-unused does not imply |
| // subcategories like clang-diagnostics-unused-function. |
| // |
| // This is used to determine which diagnostics can be enabled by ExtraArgs in |
| // the clang-tidy configuration. |
| class TidyDiagnosticGroups { |
| // Whether all diagnostic groups are enabled by default. |
| // True if we've seen clang-diagnostic-*. |
| bool Default = false; |
| // Set of diag::Group whose enablement != Default. |
| // If Default is false, this is foo where we've seen clang-diagnostic-foo. |
| llvm::DenseSet<unsigned> Exceptions; |
| |
| public: |
| TidyDiagnosticGroups(llvm::StringRef Checks) { |
| constexpr llvm::StringLiteral CDPrefix = "clang-diagnostic-"; |
| |
| llvm::StringRef Check; |
| while (!Checks.empty()) { |
| std::tie(Check, Checks) = Checks.split(','); |
| if (Check.empty()) |
| continue; |
| |
| bool Enable = !Check.consume_front("-"); |
| bool Glob = Check.consume_back("*"); |
| if (Glob) { |
| // Is this clang-diagnostic-*, or *, or so? |
| // (We ignore all other types of globs). |
| if (CDPrefix.starts_with(Check)) { |
| Default = Enable; |
| Exceptions.clear(); |
| } |
| continue; |
| } |
| |
| // In "*,clang-diagnostic-foo", the latter is a no-op. |
| if (Default == Enable) |
| continue; |
| // The only non-glob entries we care about are clang-diagnostic-foo. |
| if (!Check.consume_front(CDPrefix)) |
| continue; |
| |
| if (auto Group = DiagnosticIDs::getGroupForWarningOption(Check)) |
| Exceptions.insert(static_cast<unsigned>(*Group)); |
| } |
| } |
| |
| bool operator()(diag::Group GroupID) const { |
| return Exceptions.contains(static_cast<unsigned>(GroupID)) ? !Default |
| : Default; |
| } |
| }; |
| |
| // Find -W<group> and -Wno-<group> options in ExtraArgs and apply them to Diags. |
| // |
| // This is used to handle ExtraArgs in clang-tidy configuration. |
| // We don't use clang's standard handling of this as we want slightly different |
| // behavior (e.g. we want to exclude these from -Wno-error). |
| void applyWarningOptions(llvm::ArrayRef<std::string> ExtraArgs, |
| llvm::function_ref<bool(diag::Group)> EnabledGroups, |
| DiagnosticsEngine &Diags) { |
| for (llvm::StringRef Group : ExtraArgs) { |
| // Only handle args that are of the form -W[no-]<group>. |
| // Other flags are possible but rare and deliberately out of scope. |
| llvm::SmallVector<diag::kind> Members; |
| if (!Group.consume_front("-W") || Group.empty()) |
| continue; |
| bool Enable = !Group.consume_front("no-"); |
| if (Diags.getDiagnosticIDs()->getDiagnosticsInGroup( |
| diag::Flavor::WarningOrError, Group, Members)) |
| continue; |
| |
| // Upgrade (or downgrade) the severity of each diagnostic in the group. |
| // If -Werror is on, newly added warnings will be treated as errors. |
| // We don't want this, so keep track of them to fix afterwards. |
| bool NeedsWerrorExclusion = false; |
| for (diag::kind ID : Members) { |
| if (Enable) { |
| if (Diags.getDiagnosticLevel(ID, SourceLocation()) < |
| DiagnosticsEngine::Warning) { |
| auto Group = DiagnosticIDs::getGroupForDiag(ID); |
| if (!Group || !EnabledGroups(*Group)) |
| continue; |
| Diags.setSeverity(ID, diag::Severity::Warning, SourceLocation()); |
| if (Diags.getWarningsAsErrors()) |
| NeedsWerrorExclusion = true; |
| } |
| } else { |
| Diags.setSeverity(ID, diag::Severity::Ignored, SourceLocation()); |
| } |
| } |
| if (NeedsWerrorExclusion) { |
| // FIXME: there's no API to suppress -Werror for single diagnostics. |
| // In some cases with sub-groups, we may end up erroneously |
| // downgrading diagnostics that were -Werror in the compile command. |
| Diags.setDiagnosticGroupWarningAsError(Group, false); |
| } |
| } |
| } |
| |
| std::vector<Diag> getIncludeCleanerDiags(ParsedAST &AST, llvm::StringRef Code, |
| const ThreadsafeFS &TFS) { |
| auto &Cfg = Config::current(); |
| if (Cfg.Diagnostics.SuppressAll) |
| return {}; |
| bool SuppressMissing = |
| Cfg.Diagnostics.Suppress.contains("missing-includes") || |
| Cfg.Diagnostics.MissingIncludes == Config::IncludesPolicy::None; |
| bool SuppressUnused = |
| Cfg.Diagnostics.Suppress.contains("unused-includes") || |
| Cfg.Diagnostics.UnusedIncludes == Config::IncludesPolicy::None; |
| if (SuppressMissing && SuppressUnused) |
| return {}; |
| auto Findings = computeIncludeCleanerFindings( |
| AST, Cfg.Diagnostics.Includes.AnalyzeAngledIncludes); |
| if (SuppressMissing) |
| Findings.MissingIncludes.clear(); |
| if (SuppressUnused) |
| Findings.UnusedIncludes.clear(); |
| return issueIncludeCleanerDiagnostics(AST, Code, Findings, TFS, |
| Cfg.Diagnostics.Includes.IgnoreHeader); |
| } |
| |
| tidy::ClangTidyCheckFactories |
| filterFastTidyChecks(const tidy::ClangTidyCheckFactories &All, |
| Config::FastCheckPolicy Policy) { |
| if (Policy == Config::FastCheckPolicy::None) |
| return All; |
| bool AllowUnknown = Policy == Config::FastCheckPolicy::Loose; |
| tidy::ClangTidyCheckFactories Fast; |
| for (const auto &Factory : All) { |
| if (isFastTidyCheck(Factory.getKey()).value_or(AllowUnknown)) |
| Fast.registerCheckFactory(Factory.first(), Factory.second); |
| } |
| return Fast; |
| } |
| |
| } // namespace |
| |
| std::optional<ParsedAST> |
| ParsedAST::build(llvm::StringRef Filename, const ParseInputs &Inputs, |
| std::unique_ptr<clang::CompilerInvocation> CI, |
| llvm::ArrayRef<Diag> CompilerInvocationDiags, |
| std::shared_ptr<const PreambleData> Preamble) { |
| trace::Span Tracer("BuildAST"); |
| SPAN_ATTACH(Tracer, "File", Filename); |
| const Config &Cfg = Config::current(); |
| |
| auto VFS = Inputs.TFS->view(Inputs.CompileCommand.Directory); |
| if (Preamble && Preamble->StatCache) |
| VFS = Preamble->StatCache->getConsumingFS(std::move(VFS)); |
| |
| assert(CI); |
| |
| if (CI->getFrontendOpts().Inputs.size() > 0) { |
| auto Lang = CI->getFrontendOpts().Inputs[0].getKind().getLanguage(); |
| if (Lang == Language::Asm || Lang == Language::LLVM_IR) { |
| elog("Clangd does not support assembly or IR source files"); |
| return std::nullopt; |
| } |
| } |
| |
| // Command-line parsing sets DisableFree to true by default, but we don't want |
| // to leak memory in clangd. |
| CI->getFrontendOpts().DisableFree = false; |
| const PrecompiledPreamble *PreamblePCH = |
| Preamble ? &Preamble->Preamble : nullptr; |
| |
| // This is on-by-default in windows to allow parsing SDK headers, but it |
| // breaks many features. Disable it for the main-file (not preamble). |
| CI->getLangOpts().DelayedTemplateParsing = false; |
| |
| std::vector<std::unique_ptr<FeatureModule::ASTListener>> ASTListeners; |
| if (Inputs.FeatureModules) { |
| for (auto &M : *Inputs.FeatureModules) { |
| if (auto Listener = M.astListeners()) |
| ASTListeners.emplace_back(std::move(Listener)); |
| } |
| } |
| StoreDiags ASTDiags; |
| ASTDiags.setDiagCallback( |
| [&ASTListeners](const clang::Diagnostic &D, clangd::Diag &Diag) { |
| for (const auto &L : ASTListeners) |
| L->sawDiagnostic(D, Diag); |
| }); |
| |
| // Adjust header search options to load the built module files recorded |
| // in RequiredModules. |
| if (Preamble && Preamble->RequiredModules) |
| Preamble->RequiredModules->adjustHeaderSearchOptions( |
| CI->getHeaderSearchOpts()); |
| |
| std::optional<PreamblePatch> Patch; |
| // We might use an ignoring diagnostic consumer if they are going to be |
| // dropped later on to not pay for extra latency by processing them. |
| DiagnosticConsumer *DiagConsumer = &ASTDiags; |
| IgnoreDiagnostics DropDiags; |
| if (Preamble) { |
| Patch = PreamblePatch::createFullPatch(Filename, Inputs, *Preamble); |
| Patch->apply(*CI); |
| } |
| auto Clang = prepareCompilerInstance( |
| std::move(CI), PreamblePCH, |
| llvm::MemoryBuffer::getMemBufferCopy(Inputs.Contents, Filename), VFS, |
| *DiagConsumer); |
| |
| if (!Clang) { |
| // The last diagnostic contains information about the reason of this |
| // failure. |
| std::vector<Diag> Diags(ASTDiags.take()); |
| elog("Failed to prepare a compiler instance: {0}", |
| !Diags.empty() ? static_cast<DiagBase &>(Diags.back()).Message |
| : "unknown error"); |
| return std::nullopt; |
| } |
| tidy::ClangTidyOptions ClangTidyOpts; |
| { |
| trace::Span Tracer("ClangTidyOpts"); |
| ClangTidyOpts = getTidyOptionsForFile(Inputs.ClangTidyProvider, Filename); |
| dlog("ClangTidy configuration for file {0}: {1}", Filename, |
| tidy::configurationAsText(ClangTidyOpts)); |
| |
| // If clang-tidy is configured to emit clang warnings, we should too. |
| // |
| // Such clang-tidy configuration consists of two parts: |
| // - ExtraArgs: ["-Wfoo"] causes clang to produce the warnings |
| // - Checks: "clang-diagnostic-foo" prevents clang-tidy filtering them out |
| // |
| // In clang-tidy, diagnostics are emitted if they pass both checks. |
| // When groups contain subgroups, -Wparent includes the child, but |
| // clang-diagnostic-parent does not. |
| // |
| // We *don't* want to change the compile command directly. This can have |
| // too many unexpected effects: breaking the command, interactions with |
| // -- and -Werror, etc. Besides, we've already parsed the command. |
| // Instead we parse the -W<group> flags and handle them directly. |
| // |
| // Similarly, we don't want to use Checks to filter clang diagnostics after |
| // they are generated, as this spreads clang-tidy emulation everywhere. |
| // Instead, we just use these to filter which extra diagnostics we enable. |
| auto &Diags = Clang->getDiagnostics(); |
| TidyDiagnosticGroups TidyGroups(ClangTidyOpts.Checks ? *ClangTidyOpts.Checks |
| : llvm::StringRef()); |
| if (ClangTidyOpts.ExtraArgsBefore) |
| applyWarningOptions(*ClangTidyOpts.ExtraArgsBefore, TidyGroups, Diags); |
| if (ClangTidyOpts.ExtraArgs) |
| applyWarningOptions(*ClangTidyOpts.ExtraArgs, TidyGroups, Diags); |
| } |
| |
| auto Action = std::make_unique<ClangdFrontendAction>(); |
| const FrontendInputFile &MainInput = Clang->getFrontendOpts().Inputs[0]; |
| if (!Action->BeginSourceFile(*Clang, MainInput)) { |
| log("BeginSourceFile() failed when building AST for {0}", |
| MainInput.getFile()); |
| return std::nullopt; |
| } |
| // If we saw an include guard in the preamble section of the main file, |
| // mark the main-file as include-guarded. |
| // This information is part of the HeaderFileInfo but is not loaded from the |
| // preamble as the file's size is part of its identity and may have changed. |
| // (The rest of HeaderFileInfo is not relevant for our purposes). |
| if (Preamble && Preamble->MainIsIncludeGuarded) { |
| const SourceManager &SM = Clang->getSourceManager(); |
| OptionalFileEntryRef MainFE = SM.getFileEntryRefForID(SM.getMainFileID()); |
| Clang->getPreprocessor().getHeaderSearchInfo().MarkFileIncludeOnce(*MainFE); |
| } |
| |
| // Set up ClangTidy. Must happen after BeginSourceFile() so ASTContext exists. |
| // Clang-tidy has some limitations to ensure reasonable performance: |
| // - checks don't see all preprocessor events in the preamble |
| // - matchers run only over the main-file top-level decls (and can't see |
| // ancestors outside this scope). |
| // In practice almost all checks work well without modifications. |
| std::vector<std::unique_ptr<tidy::ClangTidyCheck>> CTChecks; |
| ast_matchers::MatchFinder CTFinder; |
| std::optional<tidy::ClangTidyContext> CTContext; |
| // Must outlive FixIncludes. |
| auto BuildDir = VFS->getCurrentWorkingDirectory(); |
| std::optional<IncludeFixer> FixIncludes; |
| llvm::DenseMap<diag::kind, DiagnosticsEngine::Level> OverriddenSeverity; |
| // No need to run clang-tidy or IncludeFixerif we are not going to surface |
| // diagnostics. |
| { |
| trace::Span Tracer("ClangTidyInit"); |
| static const auto *AllCTFactories = [] { |
| auto *CTFactories = new tidy::ClangTidyCheckFactories; |
| for (const auto &E : tidy::ClangTidyModuleRegistry::entries()) |
| E.instantiate()->addCheckFactories(*CTFactories); |
| return CTFactories; |
| }(); |
| tidy::ClangTidyCheckFactories FastFactories = filterFastTidyChecks( |
| *AllCTFactories, Cfg.Diagnostics.ClangTidy.FastCheckFilter); |
| CTContext.emplace(std::make_unique<tidy::DefaultOptionsProvider>( |
| tidy::ClangTidyGlobalOptions(), ClangTidyOpts)); |
| CTContext->setDiagnosticsEngine(&Clang->getDiagnostics()); |
| CTContext->setASTContext(&Clang->getASTContext()); |
| CTContext->setCurrentFile(Filename); |
| CTContext->setSelfContainedDiags(true); |
| CTChecks = FastFactories.createChecksForLanguage(&*CTContext); |
| Preprocessor *PP = &Clang->getPreprocessor(); |
| for (const auto &Check : CTChecks) { |
| Check->registerPPCallbacks(Clang->getSourceManager(), PP, PP); |
| Check->registerMatchers(&CTFinder); |
| } |
| |
| // Clang only corrects typos for use of undeclared functions in C if that |
| // use is an error. Include fixer relies on typo correction, so pretend |
| // this is an error. (The actual typo correction is nice too). |
| // We restore the original severity in the level adjuster. |
| // FIXME: It would be better to have a real API for this, but what? |
| for (auto ID : {diag::ext_implicit_function_decl_c99, |
| diag::ext_implicit_lib_function_decl, |
| diag::ext_implicit_lib_function_decl_c99, |
| diag::warn_implicit_function_decl}) { |
| OverriddenSeverity.try_emplace( |
| ID, Clang->getDiagnostics().getDiagnosticLevel(ID, SourceLocation())); |
| Clang->getDiagnostics().setSeverity(ID, diag::Severity::Error, |
| SourceLocation()); |
| } |
| |
| ASTDiags.setLevelAdjuster([&](DiagnosticsEngine::Level DiagLevel, |
| const clang::Diagnostic &Info) { |
| if (Cfg.Diagnostics.SuppressAll || |
| isBuiltinDiagnosticSuppressed(Info.getID(), Cfg.Diagnostics.Suppress, |
| Clang->getLangOpts())) |
| return DiagnosticsEngine::Ignored; |
| |
| auto It = OverriddenSeverity.find(Info.getID()); |
| if (It != OverriddenSeverity.end()) |
| DiagLevel = It->second; |
| |
| if (!CTChecks.empty()) { |
| std::string CheckName = CTContext->getCheckName(Info.getID()); |
| bool IsClangTidyDiag = !CheckName.empty(); |
| if (IsClangTidyDiag) { |
| if (Cfg.Diagnostics.Suppress.contains(CheckName)) |
| return DiagnosticsEngine::Ignored; |
| // Check for suppression comment. Skip the check for diagnostics not |
| // in the main file, because we don't want that function to query the |
| // source buffer for preamble files. For the same reason, we ask |
| // shouldSuppressDiagnostic to avoid I/O. |
| // We let suppression comments take precedence over warning-as-error |
| // to match clang-tidy's behaviour. |
| bool IsInsideMainFile = |
| Info.hasSourceManager() && |
| isInsideMainFile(Info.getLocation(), Info.getSourceManager()); |
| SmallVector<tooling::Diagnostic, 1> TidySuppressedErrors; |
| if (IsInsideMainFile && CTContext->shouldSuppressDiagnostic( |
| DiagLevel, Info, TidySuppressedErrors, |
| /*AllowIO=*/false, |
| /*EnableNolintBlocks=*/true)) { |
| // FIXME: should we expose the suppression error (invalid use of |
| // NOLINT comments)? |
| return DiagnosticsEngine::Ignored; |
| } |
| if (!CTContext->getOptions().SystemHeaders.value_or(false) && |
| Info.hasSourceManager() && |
| Info.getSourceManager().isInSystemMacro(Info.getLocation())) |
| return DiagnosticsEngine::Ignored; |
| |
| // Check for warning-as-error. |
| if (DiagLevel == DiagnosticsEngine::Warning && |
| CTContext->treatAsError(CheckName)) { |
| return DiagnosticsEngine::Error; |
| } |
| } |
| } |
| return DiagLevel; |
| }); |
| |
| // Add IncludeFixer which can recover diagnostics caused by missing includes |
| // (e.g. incomplete type) and attach include insertion fixes to diagnostics. |
| if (Inputs.Index && !BuildDir.getError()) { |
| auto Style = |
| getFormatStyleForFile(Filename, Inputs.Contents, *Inputs.TFS, false); |
| auto Inserter = std::make_shared<IncludeInserter>( |
| Filename, Inputs.Contents, Style, BuildDir.get(), |
| &Clang->getPreprocessor().getHeaderSearchInfo()); |
| ArrayRef<Inclusion> MainFileIncludes; |
| if (Preamble) { |
| MainFileIncludes = Preamble->Includes.MainFileIncludes; |
| for (const auto &Inc : Preamble->Includes.MainFileIncludes) |
| Inserter->addExisting(Inc); |
| } |
| // FIXME: Consider piping through ASTSignals to fetch this to handle the |
| // case where a header file contains ObjC decls but no #imports. |
| Symbol::IncludeDirective Directive = |
| Inputs.Opts.ImportInsertions |
| ? preferredIncludeDirective(Filename, Clang->getLangOpts(), |
| MainFileIncludes, {}) |
| : Symbol::Include; |
| FixIncludes.emplace(Filename, Inserter, *Inputs.Index, |
| /*IndexRequestLimit=*/5, Directive); |
| ASTDiags.contributeFixes([&FixIncludes](DiagnosticsEngine::Level DiagLevl, |
| const clang::Diagnostic &Info) { |
| return FixIncludes->fix(DiagLevl, Info); |
| }); |
| Clang->setExternalSemaSource(FixIncludes->unresolvedNameRecorder()); |
| } |
| } |
| |
| IncludeStructure Includes; |
| include_cleaner::PragmaIncludes PI; |
| // If we are using a preamble, copy existing includes. |
| if (Preamble) { |
| Includes = Preamble->Includes; |
| Includes.MainFileIncludes = Patch->preambleIncludes(); |
| // Replay the preamble includes so that clang-tidy checks can see them. |
| ReplayPreamble::attach(Patch->preambleIncludes(), *Clang, |
| Patch->modifiedBounds()); |
| PI = *Preamble->Pragmas; |
| } |
| // Important: collectIncludeStructure is registered *after* ReplayPreamble! |
| // Otherwise we would collect the replayed includes again... |
| // (We can't *just* use the replayed includes, they don't have Resolved path). |
| Includes.collect(*Clang); |
| // Same for pragma-includes, we're already inheriting preamble includes, so we |
| // should only receive callbacks for non-preamble mainfile includes. |
| PI.record(*Clang); |
| // Copy over the macros in the preamble region of the main file, and combine |
| // with non-preamble macros below. |
| MainFileMacros Macros; |
| std::vector<PragmaMark> Marks; |
| if (Preamble) { |
| Macros = Patch->mainFileMacros(); |
| Marks = Patch->marks(); |
| } |
| auto &PP = Clang->getPreprocessor(); |
| PP.addPPCallbacks(std::make_unique<CollectMainFileMacros>(PP, Macros)); |
| |
| PP.addPPCallbacks( |
| collectPragmaMarksCallback(Clang->getSourceManager(), Marks)); |
| |
| // FIXME: Attach a comment handler to take care of |
| // keep/export/no_include etc. IWYU pragmas. |
| |
| // Collect tokens of the main file. |
| syntax::TokenCollector CollectTokens(PP); |
| |
| // To remain consistent with preamble builds, these callbacks must be called |
| // exactly here, after preprocessor is initialized and BeginSourceFile() was |
| // called already. |
| for (const auto &L : ASTListeners) |
| L->beforeExecute(*Clang); |
| |
| if (llvm::Error Err = Action->Execute()) |
| log("Execute() failed when building AST for {0}: {1}", MainInput.getFile(), |
| toString(std::move(Err))); |
| |
| // We have to consume the tokens before running clang-tidy to avoid collecting |
| // tokens from running the preprocessor inside the checks (only |
| // modernize-use-trailing-return-type does that today). |
| syntax::TokenBuffer Tokens = std::move(CollectTokens).consume(); |
| // Makes SelectionTree build much faster. |
| Tokens.indexExpandedTokens(); |
| std::vector<Decl *> ParsedDecls = Action->takeTopLevelDecls(); |
| // AST traversals should exclude the preamble, to avoid performance cliffs. |
| Clang->getASTContext().setTraversalScope(ParsedDecls); |
| if (!CTChecks.empty()) { |
| // Run the AST-dependent part of the clang-tidy checks. |
| // (The preprocessor part ran already, via PPCallbacks). |
| trace::Span Tracer("ClangTidyMatch"); |
| CTFinder.matchAST(Clang->getASTContext()); |
| } |
| |
| // XXX: This is messy: clang-tidy checks flush some diagnostics at EOF. |
| // However Action->EndSourceFile() would destroy the ASTContext! |
| // So just inform the preprocessor of EOF, while keeping everything alive. |
| PP.EndSourceFile(); |
| // UnitDiagsConsumer is local, we can not store it in CompilerInstance that |
| // has a longer lifetime. |
| Clang->getDiagnostics().setClient(new IgnoreDiagnostics); |
| // CompilerInstance won't run this callback, do it directly. |
| ASTDiags.EndSourceFile(); |
| |
| std::vector<Diag> Diags = CompilerInvocationDiags; |
| // FIXME: Also skip generation of diagnostics altogether to speed up ast |
| // builds when we are patching a stale preamble. |
| // Add diagnostics from the preamble, if any. |
| if (Preamble) |
| llvm::append_range(Diags, Patch->patchedDiags()); |
| // Finally, add diagnostics coming from the AST. |
| { |
| std::vector<Diag> D = ASTDiags.take(&*CTContext); |
| Diags.insert(Diags.end(), D.begin(), D.end()); |
| } |
| ParsedAST Result(Filename, Inputs.Version, std::move(Preamble), |
| std::move(Clang), std::move(Action), std::move(Tokens), |
| std::move(Macros), std::move(Marks), std::move(ParsedDecls), |
| std::move(Diags), std::move(Includes), std::move(PI)); |
| llvm::move(getIncludeCleanerDiags(Result, Inputs.Contents, *Inputs.TFS), |
| std::back_inserter(Result.Diags)); |
| return std::move(Result); |
| } |
| |
| ParsedAST::ParsedAST(ParsedAST &&Other) = default; |
| |
| ParsedAST &ParsedAST::operator=(ParsedAST &&Other) = default; |
| |
| ParsedAST::~ParsedAST() { |
| if (Action) { |
| // We already notified the PP of end-of-file earlier, so detach it first. |
| // We must keep it alive until after EndSourceFile(), Sema relies on this. |
| auto PP = Clang->getPreprocessorPtr(); // Keep PP alive for now. |
| Clang->setPreprocessor(nullptr); // Detach so we don't send EOF again. |
| Action->EndSourceFile(); // Destroy ASTContext and Sema. |
| // Now Sema is gone, it's safe for PP to go out of scope. |
| } |
| } |
| |
| ASTContext &ParsedAST::getASTContext() { return Clang->getASTContext(); } |
| |
| const ASTContext &ParsedAST::getASTContext() const { |
| return Clang->getASTContext(); |
| } |
| |
| Sema &ParsedAST::getSema() { return Clang->getSema(); } |
| |
| Preprocessor &ParsedAST::getPreprocessor() { return Clang->getPreprocessor(); } |
| |
| std::shared_ptr<Preprocessor> ParsedAST::getPreprocessorPtr() { |
| return Clang->getPreprocessorPtr(); |
| } |
| |
| const Preprocessor &ParsedAST::getPreprocessor() const { |
| return Clang->getPreprocessor(); |
| } |
| |
| llvm::ArrayRef<Decl *> ParsedAST::getLocalTopLevelDecls() { |
| return LocalTopLevelDecls; |
| } |
| |
| llvm::ArrayRef<const Decl *> ParsedAST::getLocalTopLevelDecls() const { |
| return LocalTopLevelDecls; |
| } |
| |
| const MainFileMacros &ParsedAST::getMacros() const { return Macros; } |
| const std::vector<PragmaMark> &ParsedAST::getMarks() const { return Marks; } |
| |
| std::size_t ParsedAST::getUsedBytes() const { |
| auto &AST = getASTContext(); |
| // FIXME(ibiryukov): we do not account for the dynamically allocated part of |
| // Message and Fixes inside each diagnostic. |
| std::size_t Total = |
| clangd::getUsedBytes(LocalTopLevelDecls) + clangd::getUsedBytes(Diags); |
| |
| // FIXME: the rest of the function is almost a direct copy-paste from |
| // libclang's clang_getCXTUResourceUsage. We could share the implementation. |
| |
| // Sum up various allocators inside the ast context and the preprocessor. |
| Total += AST.getASTAllocatedMemory(); |
| Total += AST.getSideTableAllocatedMemory(); |
| Total += AST.Idents.getAllocator().getTotalMemory(); |
| Total += AST.Selectors.getTotalMemory(); |
| |
| Total += AST.getSourceManager().getContentCacheSize(); |
| Total += AST.getSourceManager().getDataStructureSizes(); |
| Total += AST.getSourceManager().getMemoryBufferSizes().malloc_bytes; |
| |
| if (ExternalASTSource *Ext = AST.getExternalSource()) |
| Total += Ext->getMemoryBufferSizes().malloc_bytes; |
| |
| const Preprocessor &PP = getPreprocessor(); |
| Total += PP.getTotalMemory(); |
| if (PreprocessingRecord *PRec = PP.getPreprocessingRecord()) |
| Total += PRec->getTotalMemory(); |
| Total += PP.getHeaderSearchInfo().getTotalMemory(); |
| |
| return Total; |
| } |
| |
| const IncludeStructure &ParsedAST::getIncludeStructure() const { |
| return Includes; |
| } |
| |
| ParsedAST::ParsedAST(PathRef TUPath, llvm::StringRef Version, |
| std::shared_ptr<const PreambleData> Preamble, |
| std::unique_ptr<CompilerInstance> Clang, |
| std::unique_ptr<FrontendAction> Action, |
| syntax::TokenBuffer Tokens, MainFileMacros Macros, |
| std::vector<PragmaMark> Marks, |
| std::vector<Decl *> LocalTopLevelDecls, |
| std::vector<Diag> Diags, IncludeStructure Includes, |
| include_cleaner::PragmaIncludes PI) |
| : TUPath(TUPath), Version(Version), Preamble(std::move(Preamble)), |
| Clang(std::move(Clang)), Action(std::move(Action)), |
| Tokens(std::move(Tokens)), Macros(std::move(Macros)), |
| Marks(std::move(Marks)), Diags(std::move(Diags)), |
| LocalTopLevelDecls(std::move(LocalTopLevelDecls)), |
| Includes(std::move(Includes)), PI(std::move(PI)), |
| Resolver(std::make_unique<HeuristicResolver>(getASTContext())) { |
| assert(this->Clang); |
| assert(this->Action); |
| } |
| |
| const include_cleaner::PragmaIncludes &ParsedAST::getPragmaIncludes() const { |
| return PI; |
| } |
| |
| std::optional<llvm::StringRef> ParsedAST::preambleVersion() const { |
| if (!Preamble) |
| return std::nullopt; |
| return llvm::StringRef(Preamble->Version); |
| } |
| |
| llvm::ArrayRef<Diag> ParsedAST::getDiagnostics() const { return Diags; } |
| } // namespace clangd |
| } // namespace clang |