forked from OSchip/llvm-project
183 lines
7.0 KiB
C++
183 lines
7.0 KiB
C++
//===--- ClangdUnit.h --------------------------------------------*- 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
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#ifndef LLVM_CLANG_TOOLS_EXTRA_CLANGD_CLANGDUNIT_H
|
|
#define LLVM_CLANG_TOOLS_EXTRA_CLANGD_CLANGDUNIT_H
|
|
|
|
#include "Compiler.h"
|
|
#include "Diagnostics.h"
|
|
#include "FS.h"
|
|
#include "Function.h"
|
|
#include "Headers.h"
|
|
#include "Path.h"
|
|
#include "Protocol.h"
|
|
#include "index/CanonicalIncludes.h"
|
|
#include "index/Index.h"
|
|
#include "clang/Frontend/FrontendAction.h"
|
|
#include "clang/Frontend/PrecompiledPreamble.h"
|
|
#include "clang/Lex/Preprocessor.h"
|
|
#include "clang/Serialization/ASTBitCodes.h"
|
|
#include "clang/Tooling/CompilationDatabase.h"
|
|
#include "clang/Tooling/Core/Replacement.h"
|
|
#include <memory>
|
|
#include <string>
|
|
#include <vector>
|
|
|
|
namespace llvm {
|
|
class raw_ostream;
|
|
|
|
namespace vfs {
|
|
class FileSystem;
|
|
} // namespace vfs
|
|
} // namespace llvm
|
|
|
|
namespace clang {
|
|
|
|
namespace tooling {
|
|
struct CompileCommand;
|
|
} // namespace tooling
|
|
|
|
namespace clangd {
|
|
|
|
// Stores Preamble and associated data.
|
|
struct PreambleData {
|
|
PreambleData(PrecompiledPreamble Preamble, std::vector<Diag> Diags,
|
|
IncludeStructure Includes,
|
|
std::vector<std::string> MainFileMacros,
|
|
std::unique_ptr<PreambleFileStatusCache> StatCache,
|
|
CanonicalIncludes CanonIncludes);
|
|
|
|
tooling::CompileCommand CompileCommand;
|
|
PrecompiledPreamble Preamble;
|
|
std::vector<Diag> Diags;
|
|
// Processes like code completions and go-to-definitions will need #include
|
|
// information, and their compile action skips preamble range.
|
|
IncludeStructure Includes;
|
|
// Macros defined in the preamble section of the main file.
|
|
// Users care about headers vs main-file, not preamble vs non-preamble.
|
|
// These should be treated as main-file entities e.g. for code completion.
|
|
std::vector<std::string> MainFileMacros;
|
|
// Cache of FS operations performed when building the preamble.
|
|
// When reusing a preamble, this cache can be consumed to save IO.
|
|
std::unique_ptr<PreambleFileStatusCache> StatCache;
|
|
CanonicalIncludes CanonIncludes;
|
|
};
|
|
|
|
/// Stores and provides access to parsed AST.
|
|
class ParsedAST {
|
|
public:
|
|
/// Attempts to run Clang and store parsed AST. If \p Preamble is non-null
|
|
/// it is reused during parsing.
|
|
static llvm::Optional<ParsedAST>
|
|
build(std::unique_ptr<clang::CompilerInvocation> CI,
|
|
std::shared_ptr<const PreambleData> Preamble,
|
|
std::unique_ptr<llvm::MemoryBuffer> Buffer,
|
|
IntrusiveRefCntPtr<llvm::vfs::FileSystem> VFS, const SymbolIndex *Index,
|
|
const ParseOptions &Opts);
|
|
|
|
ParsedAST(ParsedAST &&Other);
|
|
ParsedAST &operator=(ParsedAST &&Other);
|
|
|
|
~ParsedAST();
|
|
|
|
/// Note that the returned ast will not contain decls from the preamble that
|
|
/// were not deserialized during parsing. Clients should expect only decls
|
|
/// from the main file to be in the AST.
|
|
ASTContext &getASTContext();
|
|
const ASTContext &getASTContext() const;
|
|
|
|
Preprocessor &getPreprocessor();
|
|
std::shared_ptr<Preprocessor> getPreprocessorPtr();
|
|
const Preprocessor &getPreprocessor() const;
|
|
|
|
SourceManager &getSourceManager() {
|
|
return getASTContext().getSourceManager();
|
|
}
|
|
const SourceManager &getSourceManager() const {
|
|
return getASTContext().getSourceManager();
|
|
}
|
|
|
|
/// This function returns top-level decls present in the main file of the AST.
|
|
/// The result does not include the decls that come from the preamble.
|
|
/// (These should be const, but RecursiveASTVisitor requires Decl*).
|
|
ArrayRef<Decl *> getLocalTopLevelDecls();
|
|
|
|
const std::vector<Diag> &getDiagnostics() const;
|
|
|
|
/// Returns the esitmated size of the AST and the accessory structures, in
|
|
/// bytes. Does not include the size of the preamble.
|
|
std::size_t getUsedBytes() const;
|
|
const IncludeStructure &getIncludeStructure() const;
|
|
const CanonicalIncludes &getCanonicalIncludes() const;
|
|
|
|
private:
|
|
ParsedAST(std::shared_ptr<const PreambleData> Preamble,
|
|
std::unique_ptr<CompilerInstance> Clang,
|
|
std::unique_ptr<FrontendAction> Action,
|
|
std::vector<Decl *> LocalTopLevelDecls, std::vector<Diag> Diags,
|
|
IncludeStructure Includes, CanonicalIncludes CanonIncludes);
|
|
|
|
// In-memory preambles must outlive the AST, it is important that this member
|
|
// goes before Clang and Action.
|
|
std::shared_ptr<const PreambleData> Preamble;
|
|
// We store an "incomplete" FrontendAction (i.e. no EndSourceFile was called
|
|
// on it) and CompilerInstance used to run it. That way we don't have to do
|
|
// complex memory management of all Clang structures on our own. (They are
|
|
// stored in CompilerInstance and cleaned up by
|
|
// FrontendAction.EndSourceFile).
|
|
std::unique_ptr<CompilerInstance> Clang;
|
|
std::unique_ptr<FrontendAction> Action;
|
|
|
|
// Data, stored after parsing.
|
|
std::vector<Diag> Diags;
|
|
// Top-level decls inside the current file. Not that this does not include
|
|
// top-level decls from the preamble.
|
|
std::vector<Decl *> LocalTopLevelDecls;
|
|
IncludeStructure Includes;
|
|
CanonicalIncludes CanonIncludes;
|
|
};
|
|
|
|
using PreambleParsedCallback =
|
|
std::function<void(ASTContext &, std::shared_ptr<clang::Preprocessor>,
|
|
const CanonicalIncludes &)>;
|
|
|
|
/// Rebuild the preamble for the new inputs unless the old one can be reused.
|
|
/// If \p OldPreamble can be reused, it is returned unchanged.
|
|
/// If \p OldPreamble is null, always builds the preamble.
|
|
/// If \p PreambleCallback is set, it will be run on top of the AST while
|
|
/// building the preamble. Note that if the old preamble was reused, no AST is
|
|
/// built and, therefore, the callback will not be executed.
|
|
std::shared_ptr<const PreambleData>
|
|
buildPreamble(PathRef FileName, CompilerInvocation &CI,
|
|
std::shared_ptr<const PreambleData> OldPreamble,
|
|
const tooling::CompileCommand &OldCompileCommand,
|
|
const ParseInputs &Inputs, bool StoreInMemory,
|
|
PreambleParsedCallback PreambleCallback);
|
|
|
|
/// Build an AST from provided user inputs. This function does not check if
|
|
/// preamble can be reused, as this function expects that \p Preamble is the
|
|
/// result of calling buildPreamble.
|
|
llvm::Optional<ParsedAST>
|
|
buildAST(PathRef FileName, std::unique_ptr<CompilerInvocation> Invocation,
|
|
const ParseInputs &Inputs,
|
|
std::shared_ptr<const PreambleData> Preamble);
|
|
|
|
/// Get the beginning SourceLocation at a specified \p Pos.
|
|
/// May be invalid if Pos is, or if there's no identifier.
|
|
SourceLocation getBeginningOfIdentifier(const ParsedAST &Unit,
|
|
const Position &Pos, const FileID FID);
|
|
|
|
/// For testing/debugging purposes. Note that this method deserializes all
|
|
/// unserialized Decls, so use with care.
|
|
void dumpAST(ParsedAST &AST, llvm::raw_ostream &OS);
|
|
|
|
} // namespace clangd
|
|
} // namespace clang
|
|
|
|
#endif // LLVM_CLANG_TOOLS_EXTRA_CLANGD_CLANGDUNIT_H
|