blob: 48ad9e6513bbb3de52a2da589cc3946c4c0b4ace [file] [log] [blame]
Sam McCall489cc582019-09-03 11:35:50 +00001//===--- FindTarget.h - What does an AST node refer to? ---------*- 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// Many clangd features are concerned with references in the AST:
10// - xrefs, go-to-definition, explicitly talk about references
11// - hover and code actions relate to things you "target" in the editor
12// - refactoring actions need to know about entities that are referenced
13// to determine whether/how the edit can be applied.
14//
15// Historically, we have used libIndex (IndexDataConsumer) to tie source
16// locations to referenced declarations. This file defines a more decoupled
17// approach based around AST nodes (DynTypedNode), and can be combined with
18// SelectionTree or other traversals.
19//
20//===----------------------------------------------------------------------===//
21
Haojian Wuc00627f2019-09-25 12:54:53 +000022#ifndef LLVM_CLANG_TOOLS_EXTRA_UNITTESTS_CLANGD_FINDTARGET_H
23#define LLVM_CLANG_TOOLS_EXTRA_UNITTESTS_CLANGD_FINDTARGET_H
24
Ilya Biryukov87e0cb42019-11-05 19:06:12 +010025#include "clang/AST/ASTContext.h"
Sam McCall489cc582019-09-03 11:35:50 +000026#include "clang/AST/ASTTypeTraits.h"
Ilya Biryukovf96d2e12019-09-25 12:40:22 +000027#include "clang/AST/NestedNameSpecifier.h"
28#include "clang/AST/Stmt.h"
Sam McCall489cc582019-09-03 11:35:50 +000029#include "clang/Basic/SourceLocation.h"
Ilya Biryukovf96d2e12019-09-25 12:40:22 +000030#include "llvm/ADT/Optional.h"
31#include "llvm/ADT/STLExtras.h"
Sam McCall489cc582019-09-03 11:35:50 +000032#include "llvm/ADT/SmallPtrSet.h"
Ilya Biryukovf96d2e12019-09-25 12:40:22 +000033#include "llvm/ADT/SmallVector.h"
34#include "llvm/Support/raw_ostream.h"
Sam McCall489cc582019-09-03 11:35:50 +000035
36#include <bitset>
37
38namespace clang {
39namespace clangd {
40/// Describes the link between an AST node and a Decl it refers to.
41enum class DeclRelation : unsigned;
42/// A bitfield of DeclRelations.
43class DeclRelationSet;
44
45/// targetDecl() finds the declaration referred to by an AST node.
46/// For example a RecordTypeLoc refers to the RecordDecl for the type.
47///
48/// In some cases there are multiple results, e.g. a dependent unresolved
49/// OverloadExpr may have several candidates. All will be returned:
50///
51/// void foo(int); <-- candidate
52/// void foo(double); <-- candidate
53/// template <typename T> callFoo() { foo(T()); }
54/// ^ OverloadExpr
55///
56/// In other cases, there may be choices about what "referred to" means.
57/// e.g. does naming a typedef refer to the underlying type?
58/// The results are marked with a set of DeclRelations, and can be filtered.
59///
60/// struct S{}; <-- candidate (underlying)
61/// using T = S{}; <-- candidate (alias)
62/// T x;
63/// ^ TypedefTypeLoc
64///
65/// Formally, we walk a graph starting at the provided node, and return the
66/// decls that were found. Certain edges in the graph have labels, and for each
67/// decl we return the set of labels seen on a path to the decl.
68/// For the previous example:
69///
70/// TypedefTypeLoc T
71/// |
72/// TypedefType T
73/// / \
74/// [underlying] [alias]
75/// / \
76/// RecordDecl S TypeAliasDecl T
77///
Sam McCallf06f4392020-01-03 17:26:33 +010078/// Note that this function only returns NamedDecls. Generally other decls
79/// don't have references in this sense, just the node itself.
80/// If callers want to support such decls, they should cast the node directly.
81///
Sam McCall489cc582019-09-03 11:35:50 +000082/// FIXME: some AST nodes cannot be DynTypedNodes, these cannot be specified.
Sam McCallf06f4392020-01-03 17:26:33 +010083llvm::SmallVector<const NamedDecl *, 1>
Sam McCall489cc582019-09-03 11:35:50 +000084targetDecl(const ast_type_traits::DynTypedNode &, DeclRelationSet Mask);
85
Sam McCallacc4ffb2020-01-02 17:59:10 +010086/// Similar to targetDecl(), however instead of applying a filter, all possible
87/// decls are returned along with their DeclRelationSets.
88/// This is suitable for indexing, where everything is recorded and filtering
89/// is applied later.
Sam McCallf06f4392020-01-03 17:26:33 +010090llvm::SmallVector<std::pair<const NamedDecl *, DeclRelationSet>, 1>
Sam McCallacc4ffb2020-01-02 17:59:10 +010091allTargetDecls(const ast_type_traits::DynTypedNode &);
92
93enum class DeclRelation : unsigned {
94 // Template options apply when the declaration is an instantiated template.
95 // e.g. [[vector<int>]] vec;
96
97 /// This is the template instantiation that was referred to.
98 /// e.g. template<> class vector<int> (the implicit specialization)
99 TemplateInstantiation,
100 /// This is the pattern the template specialization was instantiated from.
101 /// e.g. class vector<T> (the pattern within the primary template)
102 TemplatePattern,
103
104 // Alias options apply when the declaration is an alias.
105 // e.g. namespace clang { [[StringRef]] S; }
106
107 /// This declaration is an alias that was referred to.
108 /// e.g. using llvm::StringRef (the UsingDecl directly referenced).
109 Alias,
110 /// This is the underlying declaration for an alias, decltype etc.
111 /// e.g. class llvm::StringRef (the underlying declaration referenced).
112 Underlying,
113};
114llvm::raw_ostream &operator<<(llvm::raw_ostream &, DeclRelation);
115
Ilya Biryukovf96d2e12019-09-25 12:40:22 +0000116/// Information about a reference written in the source code, independent of the
117/// actual AST node that this reference lives in.
118/// Useful for tools that are source-aware, e.g. refactorings.
119struct ReferenceLoc {
120 /// Contains qualifier written in the code, if any, e.g. 'ns::' for 'ns::foo'.
121 NestedNameSpecifierLoc Qualifier;
122 /// Start location of the last name part, i.e. 'foo' in 'ns::foo<int>'.
123 SourceLocation NameLoc;
Haojian Wu65f61c02019-10-18 12:07:19 +0000124 /// True if the reference is a declaration or definition;
125 bool IsDecl = false;
Ilya Biryukovf96d2e12019-09-25 12:40:22 +0000126 // FIXME: add info about template arguments.
127 /// A list of targets referenced by this name. Normally this has a single
128 /// element, but multiple is also possible, e.g. in case of using declarations
129 /// or unresolved overloaded functions.
130 /// For dependent and unresolved references, Targets can also be empty.
131 llvm::SmallVector<const NamedDecl *, 1> Targets;
132};
133llvm::raw_ostream &operator<<(llvm::raw_ostream &OS, ReferenceLoc R);
134
135/// Recursively traverse \p S and report all references explicitly written in
136/// the code. The main use-case is refactorings that need to process all
137/// references in some subrange of the file and apply simple edits, e.g. add
138/// qualifiers.
139/// FIXME: currently this does not report references to overloaded operators.
140/// FIXME: extend to report location information about declaration names too.
Kadir Cetinkaya007e4fe2019-09-25 15:44:26 +0000141void findExplicitReferences(const Stmt *S,
Ilya Biryukovf96d2e12019-09-25 12:40:22 +0000142 llvm::function_ref<void(ReferenceLoc)> Out);
Kadir Cetinkaya007e4fe2019-09-25 15:44:26 +0000143void findExplicitReferences(const Decl *D,
Ilya Biryukovf96d2e12019-09-25 12:40:22 +0000144 llvm::function_ref<void(ReferenceLoc)> Out);
Ilya Biryukov87e0cb42019-11-05 19:06:12 +0100145void findExplicitReferences(const ASTContext &AST,
146 llvm::function_ref<void(ReferenceLoc)> Out);
Ilya Biryukovf96d2e12019-09-25 12:40:22 +0000147
Sam McCallacc4ffb2020-01-02 17:59:10 +0100148/// Find declarations explicitly referenced in the source code defined by \p N.
149/// For templates, will prefer to return a template instantiation whenever
150/// possible. However, can also return a template pattern if the specialization
151/// cannot be picked, e.g. in dependent code or when there is no corresponding
Kazuaki Ishizakidd5571d2020-04-05 15:28:11 +0900152/// Decl for a template instantiation, e.g. for templated using decls:
Sam McCallacc4ffb2020-01-02 17:59:10 +0100153/// template <class T> using Ptr = T*;
154/// Ptr<int> x;
155/// ^~~ there is no Decl for 'Ptr<int>', so we return the template pattern.
156/// \p Mask should not contain TemplatePattern or TemplateInstantiation.
157llvm::SmallVector<const NamedDecl *, 1>
158explicitReferenceTargets(ast_type_traits::DynTypedNode N,
159 DeclRelationSet Mask);
Sam McCall489cc582019-09-03 11:35:50 +0000160
Sam McCallacc4ffb2020-01-02 17:59:10 +0100161// Boring implementation details of bitfield.
Sam McCall489cc582019-09-03 11:35:50 +0000162
163class DeclRelationSet {
164 using Set = std::bitset<static_cast<unsigned>(DeclRelation::Underlying) + 1>;
165 Set S;
166 DeclRelationSet(Set S) : S(S) {}
167
168public:
169 DeclRelationSet() = default;
170 DeclRelationSet(DeclRelation R) { S.set(static_cast<unsigned>(R)); }
171
172 explicit operator bool() const { return S.any(); }
173 friend DeclRelationSet operator&(DeclRelationSet L, DeclRelationSet R) {
174 return L.S & R.S;
175 }
176 friend DeclRelationSet operator|(DeclRelationSet L, DeclRelationSet R) {
177 return L.S | R.S;
178 }
179 friend bool operator==(DeclRelationSet L, DeclRelationSet R) {
180 return L.S == R.S;
181 }
182 friend DeclRelationSet operator~(DeclRelationSet R) { return ~R.S; }
183 DeclRelationSet &operator|=(DeclRelationSet Other) {
184 S |= Other.S;
185 return *this;
186 }
187 DeclRelationSet &operator&=(DeclRelationSet Other) {
188 S &= Other.S;
189 return *this;
190 }
191 friend llvm::raw_ostream &operator<<(llvm::raw_ostream &, DeclRelationSet);
192};
193// The above operators can't be looked up if both sides are enums.
194// over.match.oper.html#3.2
195inline DeclRelationSet operator|(DeclRelation L, DeclRelation R) {
196 return DeclRelationSet(L) | DeclRelationSet(R);
197}
198inline DeclRelationSet operator&(DeclRelation L, DeclRelation R) {
199 return DeclRelationSet(L) & DeclRelationSet(R);
200}
201inline DeclRelationSet operator~(DeclRelation R) { return ~DeclRelationSet(R); }
202llvm::raw_ostream &operator<<(llvm::raw_ostream &, DeclRelationSet);
203
204} // namespace clangd
205} // namespace clang
Haojian Wuc00627f2019-09-25 12:54:53 +0000206
207#endif // LLVM_CLANG_TOOLS_EXTRA_UNITTESTS_CLANGD_FINDTARGET_H