clang-tools 22.0.0git
AST.h
Go to the documentation of this file.
1//===--- AST.h - Utility AST functions -------------------------*- C++ -*-===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8//
9// Various code that examines C++ source code using AST.
10//
11//===----------------------------------------------------------------------===//
12
13#ifndef LLVM_CLANG_TOOLS_EXTRA_CLANGD_AST_H
14#define LLVM_CLANG_TOOLS_EXTRA_CLANGD_AST_H
15
16#include "Headers.h"
17#include "index/Symbol.h"
18#include "index/SymbolID.h"
19#include "clang/AST/Decl.h"
20#include "clang/AST/DeclObjC.h"
21#include "clang/AST/TypeLoc.h"
22#include "clang/Basic/SourceLocation.h"
23#include "clang/Lex/MacroInfo.h"
24#include "llvm/ADT/StringRef.h"
25#include <optional>
26#include <string>
27#include <vector>
28
29namespace clang {
30class SourceManager;
31class Decl;
32class DynTypedNode;
33class HeuristicResolver;
34
35namespace clangd {
36
37/// Returns true if the declaration is considered implementation detail based on
38/// heuristics. For example, a declaration whose name is not explicitly spelled
39/// in code is considered implementation detail.
40bool isImplementationDetail(const Decl *D);
41
42/// Find the source location of the identifier for \p D.
43/// Transforms macro locations to locations spelled inside files. All code
44/// that needs locations of declaration names (e.g. the index) should go through
45/// this function.
46SourceLocation nameLocation(const clang::Decl &D, const SourceManager &SM);
47
48/// Returns the qualified name of ND. The scope doesn't contain unwritten scopes
49/// like inline namespaces.
50std::string printQualifiedName(const NamedDecl &ND);
51
52/// Returns the first enclosing namespace scope starting from \p DC.
53std::string printNamespaceScope(const DeclContext &DC);
54
55/// Returns the name of the namespace inside the 'using namespace' directive, as
56/// written in the code. E.g., passing 'using namespace ::std' will result in
57/// '::std'.
58std::string printUsingNamespaceName(const ASTContext &Ctx,
59 const UsingDirectiveDecl &D);
60
61/// Prints unqualified name of the decl for the purpose of displaying it to the
62/// user. Anonymous decls return names of the form "(anonymous {kind})", e.g.
63/// "(anonymous struct)" or "(anonymous namespace)".
64std::string printName(const ASTContext &Ctx, const NamedDecl &ND);
65
66/// Prints template arguments of a decl as written in the source code, including
67/// enclosing '<' and '>', e.g for a partial specialization like: template
68/// <typename U> struct Foo<int, U> will return '<int, U>'. Returns an empty
69/// string if decl is not a template specialization.
70std::string printTemplateSpecializationArgs(const NamedDecl &ND);
71
72/// Print the Objective-C method name, including the full container name, e.g.
73/// `-[MyClass(Category) method:]`
74std::string printObjCMethod(const ObjCMethodDecl &Method);
75
76/// Print the Objective-C container name including categories, e.g. `MyClass`,
77// `MyClass()`, `MyClass(Category)`, and `MyProtocol`.
78std::string printObjCContainer(const ObjCContainerDecl &C);
79
80/// Returns true if this is a NamedDecl with a reserved name.
81bool hasReservedName(const Decl &);
82/// Returns true if this scope would be written with a reserved name.
83/// This does not include unwritten scope elements like __1 in std::__1::vector.
84bool hasReservedScope(const DeclContext &);
85
86/// Gets the symbol ID for a declaration. Returned SymbolID might be null.
87SymbolID getSymbolID(const Decl *D);
88
89/// Gets the symbol ID for a macro. Returned SymbolID might be null.
90/// Currently, this is an encoded USR of the macro, which incorporates macro
91/// locations (e.g. file name, offset in file).
92/// FIXME: the USR semantics might not be stable enough as the ID for index
93/// macro (e.g. a change in definition offset can result in a different USR). We
94/// could change these semantics in the future by reimplementing this funcure
95/// (e.g. avoid USR for macros).
96SymbolID getSymbolID(const llvm::StringRef MacroName, const MacroInfo *MI,
97 const SourceManager &SM);
98
99/// Return the corresponding implementation/definition for the given ObjC
100/// container if it has one, otherwise, return nullptr.
101///
102/// Objective-C classes can have three types of declarations:
103///
104/// - forward declaration: "@class MyClass;"
105/// - true declaration (interface definition): "@interface MyClass ... @end"
106/// - true definition (implementation): "@implementation MyClass ... @end"
107///
108/// Objective-C categories are extensions on classes:
109///
110/// - declaration: "@interface MyClass (Ext) ... @end"
111/// - definition: "@implementation MyClass (Ext) ... @end"
112///
113/// With one special case, a class extension, which is normally used to keep
114/// some declarations internal to a file without exposing them in a header.
115///
116/// - class extension declaration: "@interface MyClass () ... @end"
117/// - which really links to class definition: "@implementation MyClass ... @end"
118///
119/// For Objective-C protocols, e.g. "@protocol MyProtocol ... @end" this will
120/// return nullptr as protocols don't have an implementation.
121const ObjCImplDecl *getCorrespondingObjCImpl(const ObjCContainerDecl *D);
122
123/// Infer the include directive to use for the given \p FileName. It aims for
124/// #import for ObjC files and #include for the rest.
125///
126/// - For source files we use LangOpts directly to infer ObjC-ness.
127/// - For header files we also check for symbols declared by the file and
128/// existing include directives, as the language can be set to ObjC++ as a
129/// fallback in the absence of compile flags.
131preferredIncludeDirective(llvm::StringRef FileName, const LangOptions &LangOpts,
132 ArrayRef<Inclusion> MainFileIncludes,
133 ArrayRef<const Decl *> TopLevelDecls);
134
135/// Returns a QualType as string. The result doesn't contain unwritten scopes
136/// like anonymous/inline namespace.
137std::string printType(const QualType QT, const DeclContext &CurContext,
138 llvm::StringRef Placeholder = "",
139 bool FullyQualify = false);
140
141/// Indicates if \p D is a template instantiation implicitly generated by the
142/// compiler, e.g.
143/// template <class T> struct vector {};
144/// vector<int> v; // 'vector<int>' is an implicit instantiation
145bool isImplicitTemplateInstantiation(const NamedDecl *D);
146/// Indicates if \p D is an explicit template specialization, e.g.
147/// template <class T> struct vector {};
148/// template <> struct vector<bool> {}; // <-- explicit specialization
149///
150/// Note that explicit instantiations are NOT explicit specializations, albeit
151/// they look similar.
152/// template struct vector<bool>; // <-- explicit instantiation, NOT an
153/// explicit specialization.
154bool isExplicitTemplateSpecialization(const NamedDecl *D);
155
156/// Returns a nested name specifier loc of \p ND if it was present in the
157/// source, e.g.
158/// void ns::something::foo() -> returns 'ns::something'
159/// void foo() -> returns null
160NestedNameSpecifierLoc getQualifierLoc(const NamedDecl &ND);
161
162// Returns a type corresponding to a declaration of that type.
163// Unlike the method on ASTContext, attempts to preserve the type as-written
164// (i.e. vector<T*> rather than vector<type-parameter-0-0 *>.
165QualType declaredType(const TypeDecl *D);
166
167/// Retrieves the deduced type at a given location (auto, decltype).
168/// It will return the underlying type.
169/// If the type is an undeduced auto, returns the type itself.
170std::optional<QualType> getDeducedType(ASTContext &, const HeuristicResolver *,
171 SourceLocation Loc);
172
173// Find the abbreviated-function-template `auto` within a type, or returns null.
174// Similar to getContainedAutoTypeLoc, but these `auto`s are
175// TemplateTypeParmTypes for implicit TTPs, instead of AutoTypes.
176// Also we don't look very hard, just stripping const, references, pointers.
177// FIXME: handle more type patterns.
178TemplateTypeParmTypeLoc getContainedAutoParamType(TypeLoc TL);
179
180// If TemplatedDecl is the generic body of a template, and the template has
181// exactly one visible instantiation, return the instantiated body.
182NamedDecl *getOnlyInstantiation(NamedDecl *TemplatedDecl);
183
184/// Return attributes attached directly to a node.
185std::vector<const Attr *> getAttributes(const DynTypedNode &);
186
187/// Gets the nested name specifier necessary for spelling \p ND in \p
188/// DestContext, at \p InsertionPoint. It selects the shortest suffix of \p ND
189/// such that it is visible in \p DestContext.
190/// Returns an empty string if no qualification is necessary. For example, if
191/// you want to qualify clang::clangd::bar::foo in clang::clangd::x, this
192/// function will return bar. Note that the result might be sub-optimal for
193/// classes, e.g. when the \p ND is a member of the base class.
194///
195/// This version considers all the using namespace directives before \p
196/// InsertionPoint. i.e, if you have `using namespace
197/// clang::clangd::bar`, this function will return an empty string for the
198/// example above since no qualification is necessary in that case.
199/// FIXME: Also take using directives and namespace aliases inside function body
200/// into account.
201std::string getQualification(ASTContext &Context,
202 const DeclContext *DestContext,
203 SourceLocation InsertionPoint,
204 const NamedDecl *ND);
205
206/// This function uses the \p VisibleNamespaces to figure out if a shorter
207/// qualification is sufficient for \p ND, and ignores any using namespace
208/// directives. It can be useful if there's no AST for the DestContext, but some
209/// pseudo-parsing is done. i.e. if \p ND is ns1::ns2::X and \p DestContext is
210/// ns1::, users can provide `ns2::` as visible to change the result to be
211/// empty.
212/// Elements in VisibleNamespaces should be in the form: `ns::`, with trailing
213/// "::".
214/// Note that this is just textual and might be incorrect. e.g. when there are
215/// two namespaces ns1::a and ns2::a, the function will early exit if "a::" is
216/// present in \p VisibleNamespaces, no matter whether it is from ns1:: or ns2::
217std::string getQualification(ASTContext &Context,
218 const DeclContext *DestContext,
219 const NamedDecl *ND,
220 llvm::ArrayRef<std::string> VisibleNamespaces);
221
222/// Whether we must avoid computing linkage for D during code completion.
223/// Clang aggressively caches linkage computation, which is stable after the AST
224/// is built. Unfortunately the AST is incomplete during code completion, so
225/// linkage may still change.
226///
227/// Example: `auto x = []{^}` at file scope.
228/// During code completion, the initializer for x hasn't been parsed yet.
229/// x has type `undeduced auto`, and external linkage.
230/// If we compute linkage at this point, the external linkage will be cached.
231///
232/// After code completion the initializer is attached, and x has a lambda type.
233/// This means x has "unique external" linkage. If we computed linkage above,
234/// the cached value is incorrect. (clang catches this with an assertion).
235bool hasUnstableLinkage(const Decl *D);
236
237/// Checks whether \p D is more than \p MaxDepth away from translation unit
238/// scope.
239/// This is useful for limiting traversals to keep operation latencies
240/// reasonable.
241bool isDeeplyNested(const Decl *D, unsigned MaxDepth = 10);
242
243/// Recursively resolves the parameters of a FunctionDecl that forwards its
244/// parameters to another function via variadic template parameters. This can
245/// for example be used to retrieve the constructor parameter ParmVarDecl for a
246/// make_unique or emplace_back call.
247llvm::SmallVector<const ParmVarDecl *>
248resolveForwardingParameters(const FunctionDecl *D, unsigned MaxDepth = 10);
249
250/// Checks whether D is instantiated from a function parameter pack
251/// whose type is a bare type parameter pack (e.g. `Args...`), or a
252/// reference to one (e.g. `Args&...` or `Args&&...`).
253bool isExpandedFromParameterPack(const ParmVarDecl *D);
254
255/// Heuristic that checks if FT is likely to be forwarding a parameter pack to
256/// another function (e.g. `make_unique`).
257bool isLikelyForwardingFunction(const FunctionTemplateDecl *FT);
258
259/// Only call if FD is a likely forwarding function. Returns
260/// constructors that might be forwarded to.
261SmallVector<const CXXConstructorDecl *, 1>
262searchConstructorsInForwardingFunction(const FunctionDecl *FD);
263
264} // namespace clangd
265} // namespace clang
266
267#endif // LLVM_CLANG_TOOLS_EXTRA_CLANGD_AST_H
A context is an immutable container for per-request data that must be propagated through layers that ...
Definition Context.h:69
FIXME: Skip testing on windows temporarily due to the different escaping code mode.
Definition AST.cpp:44
SmallVector< const ParmVarDecl * > resolveForwardingParameters(const FunctionDecl *D, unsigned MaxDepth)
Recursively resolves the parameters of a FunctionDecl that forwards its parameters to another functio...
Definition AST.cpp:981
std::string printTemplateSpecializationArgs(const NamedDecl &ND)
Prints template arguments of a decl as written in the source code, including enclosing '<' and '>',...
Definition AST.cpp:286
std::string printObjCMethod(const ObjCMethodDecl &Method)
Print the Objective-C method name, including the full container name, e.g.
Definition AST.cpp:315
bool isLikelyForwardingFunction(const FunctionTemplateDecl *FT)
Heuristic that checks if FT is likely to be forwarding a parameter pack to another function (e....
Definition AST.cpp:1042
SymbolID getSymbolID(const Decl *D)
Gets the symbol ID for a declaration. Returned SymbolID might be null.
Definition AST.cpp:353
std::string printName(const ASTContext &Ctx, const NamedDecl &ND)
Prints unqualified name of the decl for the purpose of displaying it to the user.
Definition AST.cpp:247
std::string printObjCContainer(const ObjCContainerDecl &C)
Print the Objective-C container name including categories, e.g. MyClass,.
Definition AST.cpp:334
std::string printType(const QualType QT, const DeclContext &CurContext, const llvm::StringRef Placeholder, bool FullyQualify)
Returns a QualType as string.
Definition AST.cpp:416
std::string getQualification(ASTContext &Context, const DeclContext *DestContext, SourceLocation InsertionPoint, const NamedDecl *ND)
Gets the nested name specifier necessary for spelling ND in DestContext, at InsertionPoint.
Definition AST.cpp:697
bool isExplicitTemplateSpecialization(const NamedDecl *D)
Indicates if D is an explicit template specialization, e.g.
Definition AST.cpp:187
NamedDecl * getOnlyInstantiation(NamedDecl *TemplatedDecl)
Definition AST.cpp:661
SourceLocation nameLocation(const clang::Decl &D, const SourceManager &SM)
Find the source location of the identifier for D.
Definition AST.cpp:196
NestedNameSpecifierLoc getQualifierLoc(const NamedDecl &ND)
Returns a nested name specifier loc of ND if it was present in the source, e.g.
Definition AST.cpp:228
std::optional< QualType > getDeducedType(ASTContext &ASTCtx, const HeuristicResolver *Resolver, SourceLocation Loc)
Retrieves the deduced type at a given location (auto, decltype).
Definition AST.cpp:622
Symbol::IncludeDirective preferredIncludeDirective(llvm::StringRef FileName, const LangOptions &LangOpts, ArrayRef< Inclusion > MainFileIncludes, ArrayRef< const Decl * > TopLevelDecls)
Infer the include directive to use for the given FileName.
Definition AST.cpp:385
bool isExpandedFromParameterPack(const ParmVarDecl *D)
Checks whether D is instantiated from a function parameter pack whose type is a bare type parameter p...
Definition AST.cpp:1038
bool hasUnstableLinkage(const Decl *D)
Whether we must avoid computing linkage for D during code completion.
Definition AST.cpp:734
std::string printUsingNamespaceName(const ASTContext &Ctx, const UsingDirectiveDecl &D)
Returns the name of the namespace inside the 'using namespace' directive, as written in the code.
Definition AST.cpp:236
bool hasReservedName(const Decl &D)
Returns true if this is a NamedDecl with a reserved name.
Definition AST.cpp:443
std::vector< const Attr * > getAttributes(const DynTypedNode &N)
Return attributes attached directly to a node.
Definition AST.cpp:673
SmallVector< const CXXConstructorDecl *, 1 > searchConstructorsInForwardingFunction(const FunctionDecl *FD)
Only call if FD is a likely forwarding function.
Definition AST.cpp:1109
QualType declaredType(const TypeDecl *D)
Definition AST.cpp:461
bool isImplementationDetail(const Decl *D)
Returns true if the declaration is considered implementation detail based on heuristics.
Definition AST.cpp:191
const ObjCImplDecl * getCorrespondingObjCImpl(const ObjCContainerDecl *D)
Return the corresponding implementation/definition for the given ObjC container if it has one,...
Definition AST.cpp:370
bool isImplicitTemplateInstantiation(const NamedDecl *D)
Indicates if D is a template instantiation implicitly generated by the compiler, e....
Definition AST.cpp:183
bool hasReservedScope(const DeclContext &DC)
Returns true if this scope would be written with a reserved name.
Definition AST.cpp:450
std::string printQualifiedName(const NamedDecl &ND)
Returns the qualified name of ND.
Definition AST.cpp:206
TemplateTypeParmTypeLoc getContainedAutoParamType(TypeLoc TL)
Definition AST.cpp:634
bool isDeeplyNested(const Decl *D, unsigned MaxDepth)
Checks whether D is more than MaxDepth away from translation unit scope.
Definition AST.cpp:741
std::string printNamespaceScope(const DeclContext &DC)
Returns the first enclosing namespace scope starting from DC.
Definition AST.cpp:302
===– Representation.cpp - ClangDoc Representation --------—*- C++ -*-===//