blob: 28cc5837cefbff06ddf0fa911788ae1c776d12ba [file] [log] [blame]
Kirill Bobyrev8e35f1e2018-08-14 16:03:32 +00001//===--- CodeComplete.cpp ----------------------------------------*- C++-*-===//
Sam McCall98775c52017-12-04 13:49:59 +00002//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
Kirill Bobyrev8e35f1e2018-08-14 16:03:32 +00008//===----------------------------------------------------------------------===//
Sam McCall98775c52017-12-04 13:49:59 +00009//
Sam McCallc18c2802018-06-15 11:06:29 +000010// Code completion has several moving parts:
11// - AST-based completions are provided using the completion hooks in Sema.
12// - external completions are retrieved from the index (using hints from Sema)
13// - the two sources overlap, and must be merged and overloads bundled
14// - results must be scored and ranked (see Quality.h) before rendering
Sam McCall98775c52017-12-04 13:49:59 +000015//
Sam McCallc18c2802018-06-15 11:06:29 +000016// Signature help works in a similar way as code completion, but it is simpler:
17// it's purely AST-based, and there are few candidates.
Sam McCall98775c52017-12-04 13:49:59 +000018//
Kirill Bobyrev8e35f1e2018-08-14 16:03:32 +000019//===----------------------------------------------------------------------===//
Sam McCall98775c52017-12-04 13:49:59 +000020
21#include "CodeComplete.h"
Eric Liu7ad16962018-06-22 10:46:59 +000022#include "AST.h"
Eric Liub1d75422018-10-02 10:43:55 +000023#include "ClangdUnit.h"
Eric Liu63696e12017-12-20 17:24:31 +000024#include "CodeCompletionStrings.h"
Sam McCall98775c52017-12-04 13:49:59 +000025#include "Compiler.h"
Kadir Cetinkaya2f84d912018-08-08 08:59:29 +000026#include "Diagnostics.h"
Ilya Biryukov647da3e2018-11-26 15:38:01 +000027#include "ExpectedTypes.h"
Sam McCall3f0243f2018-07-03 08:09:29 +000028#include "FileDistance.h"
Sam McCall84652cc2018-01-12 16:16:09 +000029#include "FuzzyMatch.h"
Eric Liu63f419a2018-05-15 15:29:32 +000030#include "Headers.h"
Eric Liu6f648df2017-12-19 16:50:37 +000031#include "Logger.h"
Sam McCallc5707b62018-05-15 17:43:27 +000032#include "Quality.h"
Eric Liuc5105f92018-02-16 14:15:55 +000033#include "SourceCode.h"
Eric Liu25d74e92018-08-24 11:23:56 +000034#include "TUScheduler.h"
Sam McCall2b780162018-01-30 17:20:54 +000035#include "Trace.h"
Eric Liu63f419a2018-05-15 15:29:32 +000036#include "URI.h"
Eric Liu6f648df2017-12-19 16:50:37 +000037#include "index/Index.h"
Eric Liu3fac4ef2018-10-17 11:19:02 +000038#include "clang/AST/Decl.h"
39#include "clang/AST/DeclBase.h"
Marc-Andre Laperle945b5a32018-06-05 14:01:40 +000040#include "clang/ASTMatchers/ASTMatchFinder.h"
Ilya Biryukovc22d3442018-05-16 12:32:49 +000041#include "clang/Basic/LangOptions.h"
Kadir Cetinkayaa9c9d002018-08-13 08:23:01 +000042#include "clang/Basic/SourceLocation.h"
Eric Liuc5105f92018-02-16 14:15:55 +000043#include "clang/Format/Format.h"
Sam McCall98775c52017-12-04 13:49:59 +000044#include "clang/Frontend/CompilerInstance.h"
45#include "clang/Frontend/FrontendActions.h"
Sam McCallebef8122018-09-14 12:36:06 +000046#include "clang/Lex/PreprocessorOptions.h"
Sam McCall98775c52017-12-04 13:49:59 +000047#include "clang/Sema/CodeCompleteConsumer.h"
48#include "clang/Sema/Sema.h"
Eric Liu670c1472018-09-27 18:46:00 +000049#include "llvm/ADT/ArrayRef.h"
Eric Liu3fac4ef2018-10-17 11:19:02 +000050#include "llvm/ADT/None.h"
Eric Liu25d74e92018-08-24 11:23:56 +000051#include "llvm/ADT/Optional.h"
Eric Liu83f63e42018-09-03 10:18:21 +000052#include "llvm/ADT/SmallVector.h"
Eric Liu25d74e92018-08-24 11:23:56 +000053#include "llvm/Support/Error.h"
Haojian Wuba28e9a2018-01-10 14:44:34 +000054#include "llvm/Support/Format.h"
Eric Liubc25ef72018-07-05 08:29:33 +000055#include "llvm/Support/FormatVariadic.h"
Sam McCall2161ec72018-07-05 06:20:41 +000056#include "llvm/Support/ScopedPrinter.h"
Eric Liu83f63e42018-09-03 10:18:21 +000057#include <algorithm>
58#include <iterator>
Sam McCall98775c52017-12-04 13:49:59 +000059
Sam McCallc5707b62018-05-15 17:43:27 +000060// We log detailed candidate here if you run with -debug-only=codecomplete.
Sam McCall27c979a2018-06-29 14:47:57 +000061#define DEBUG_TYPE "CodeComplete"
Sam McCallc5707b62018-05-15 17:43:27 +000062
Sam McCallc008af62018-10-20 15:30:37 +000063using namespace llvm;
Sam McCall98775c52017-12-04 13:49:59 +000064namespace clang {
65namespace clangd {
66namespace {
67
Eric Liu6f648df2017-12-19 16:50:37 +000068CompletionItemKind toCompletionItemKind(index::SymbolKind Kind) {
69 using SK = index::SymbolKind;
70 switch (Kind) {
71 case SK::Unknown:
72 return CompletionItemKind::Missing;
73 case SK::Module:
74 case SK::Namespace:
75 case SK::NamespaceAlias:
76 return CompletionItemKind::Module;
77 case SK::Macro:
78 return CompletionItemKind::Text;
79 case SK::Enum:
80 return CompletionItemKind::Enum;
81 // FIXME(ioeric): use LSP struct instead of class when it is suppoted in the
82 // protocol.
83 case SK::Struct:
84 case SK::Class:
85 case SK::Protocol:
86 case SK::Extension:
87 case SK::Union:
88 return CompletionItemKind::Class;
89 // FIXME(ioeric): figure out whether reference is the right type for aliases.
90 case SK::TypeAlias:
91 case SK::Using:
92 return CompletionItemKind::Reference;
93 case SK::Function:
94 // FIXME(ioeric): this should probably be an operator. This should be fixed
95 // when `Operator` is support type in the protocol.
96 case SK::ConversionFunction:
97 return CompletionItemKind::Function;
98 case SK::Variable:
99 case SK::Parameter:
100 return CompletionItemKind::Variable;
101 case SK::Field:
102 return CompletionItemKind::Field;
103 // FIXME(ioeric): use LSP enum constant when it is supported in the protocol.
104 case SK::EnumConstant:
105 return CompletionItemKind::Value;
106 case SK::InstanceMethod:
107 case SK::ClassMethod:
108 case SK::StaticMethod:
109 case SK::Destructor:
110 return CompletionItemKind::Method;
111 case SK::InstanceProperty:
112 case SK::ClassProperty:
113 case SK::StaticProperty:
114 return CompletionItemKind::Property;
115 case SK::Constructor:
116 return CompletionItemKind::Constructor;
117 }
118 llvm_unreachable("Unhandled clang::index::SymbolKind.");
119}
120
Sam McCall83305892018-06-08 21:17:19 +0000121CompletionItemKind
122toCompletionItemKind(CodeCompletionResult::ResultKind ResKind,
Sam McCall4c077f92018-09-18 09:08:28 +0000123 const NamedDecl *Decl,
124 CodeCompletionContext::Kind CtxKind) {
Sam McCall83305892018-06-08 21:17:19 +0000125 if (Decl)
126 return toCompletionItemKind(index::getSymbolInfo(Decl).Kind);
Sam McCall4c077f92018-09-18 09:08:28 +0000127 if (CtxKind == CodeCompletionContext::CCC_IncludedFile)
128 return CompletionItemKind::File;
Sam McCall83305892018-06-08 21:17:19 +0000129 switch (ResKind) {
130 case CodeCompletionResult::RK_Declaration:
131 llvm_unreachable("RK_Declaration without Decl");
132 case CodeCompletionResult::RK_Keyword:
133 return CompletionItemKind::Keyword;
134 case CodeCompletionResult::RK_Macro:
135 return CompletionItemKind::Text; // unfortunately, there's no 'Macro'
136 // completion items in LSP.
137 case CodeCompletionResult::RK_Pattern:
138 return CompletionItemKind::Snippet;
139 }
140 llvm_unreachable("Unhandled CodeCompletionResult::ResultKind.");
141}
142
Sam McCall98775c52017-12-04 13:49:59 +0000143/// Get the optional chunk as a string. This function is possibly recursive.
144///
145/// The parameter info for each parameter is appended to the Parameters.
Kadir Cetinkayae486e372018-08-13 08:40:05 +0000146std::string getOptionalParameters(const CodeCompletionString &CCS,
147 std::vector<ParameterInformation> &Parameters,
148 SignatureQualitySignals &Signal) {
Sam McCall98775c52017-12-04 13:49:59 +0000149 std::string Result;
150 for (const auto &Chunk : CCS) {
151 switch (Chunk.Kind) {
152 case CodeCompletionString::CK_Optional:
153 assert(Chunk.Optional &&
154 "Expected the optional code completion string to be non-null.");
Kadir Cetinkayae486e372018-08-13 08:40:05 +0000155 Result += getOptionalParameters(*Chunk.Optional, Parameters, Signal);
Sam McCall98775c52017-12-04 13:49:59 +0000156 break;
157 case CodeCompletionString::CK_VerticalSpace:
158 break;
159 case CodeCompletionString::CK_Placeholder:
160 // A string that acts as a placeholder for, e.g., a function call
161 // argument.
162 // Intentional fallthrough here.
163 case CodeCompletionString::CK_CurrentParameter: {
164 // A piece of text that describes the parameter that corresponds to
165 // the code-completion location within a function call, message send,
166 // macro invocation, etc.
167 Result += Chunk.Text;
168 ParameterInformation Info;
169 Info.label = Chunk.Text;
170 Parameters.push_back(std::move(Info));
Kadir Cetinkayae486e372018-08-13 08:40:05 +0000171 Signal.ContainsActiveParameter = true;
172 Signal.NumberOfOptionalParameters++;
Sam McCall98775c52017-12-04 13:49:59 +0000173 break;
174 }
175 default:
176 Result += Chunk.Text;
177 break;
178 }
179 }
180 return Result;
181}
182
Eric Liu63f419a2018-05-15 15:29:32 +0000183/// Creates a `HeaderFile` from \p Header which can be either a URI or a literal
184/// include.
Sam McCallc008af62018-10-20 15:30:37 +0000185static Expected<HeaderFile> toHeaderFile(StringRef Header, StringRef HintPath) {
Eric Liu63f419a2018-05-15 15:29:32 +0000186 if (isLiteralInclude(Header))
187 return HeaderFile{Header.str(), /*Verbatim=*/true};
188 auto U = URI::parse(Header);
189 if (!U)
190 return U.takeError();
191
192 auto IncludePath = URI::includeSpelling(*U);
193 if (!IncludePath)
194 return IncludePath.takeError();
195 if (!IncludePath->empty())
196 return HeaderFile{std::move(*IncludePath), /*Verbatim=*/true};
197
198 auto Resolved = URI::resolve(*U, HintPath);
199 if (!Resolved)
200 return Resolved.takeError();
201 return HeaderFile{std::move(*Resolved), /*Verbatim=*/false};
202}
203
Sam McCall545a20d2018-01-19 14:34:02 +0000204/// A code completion result, in clang-native form.
Sam McCall98775c52017-12-04 13:49:59 +0000205/// It may be promoted to a CompletionItem if it's among the top-ranked results.
206struct CompletionCandidate {
Sam McCallc008af62018-10-20 15:30:37 +0000207 StringRef Name; // Used for filtering and sorting.
Sam McCall545a20d2018-01-19 14:34:02 +0000208 // We may have a result from Sema, from the index, or both.
209 const CodeCompletionResult *SemaResult = nullptr;
210 const Symbol *IndexResult = nullptr;
Sam McCallc008af62018-10-20 15:30:37 +0000211 SmallVector<StringRef, 1> RankedIncludeHeaders;
Sam McCall98775c52017-12-04 13:49:59 +0000212
Sam McCallc18c2802018-06-15 11:06:29 +0000213 // Returns a token identifying the overload set this is part of.
214 // 0 indicates it's not part of any overload set.
215 size_t overloadSet() const {
216 SmallString<256> Scratch;
217 if (IndexResult) {
218 switch (IndexResult->SymInfo.Kind) {
219 case index::SymbolKind::ClassMethod:
220 case index::SymbolKind::InstanceMethod:
221 case index::SymbolKind::StaticMethod:
Fangrui Song12e4ee72018-11-02 05:59:29 +0000222#ifndef NDEBUG
223 llvm_unreachable("Don't expect members from index in code completion");
224#else
Fangrui Songa3ed05b2018-11-02 04:23:50 +0000225 LLVM_FALLTHROUGH;
Fangrui Song12e4ee72018-11-02 05:59:29 +0000226#endif
Sam McCallc18c2802018-06-15 11:06:29 +0000227 case index::SymbolKind::Function:
228 // We can't group overloads together that need different #includes.
229 // This could break #include insertion.
230 return hash_combine(
231 (IndexResult->Scope + IndexResult->Name).toStringRef(Scratch),
Eric Liu83f63e42018-09-03 10:18:21 +0000232 headerToInsertIfAllowed().getValueOr(""));
Sam McCallc18c2802018-06-15 11:06:29 +0000233 default:
234 return 0;
235 }
236 }
237 assert(SemaResult);
238 // We need to make sure we're consistent with the IndexResult case!
239 const NamedDecl *D = SemaResult->Declaration;
240 if (!D || !D->isFunctionOrFunctionTemplate())
241 return 0;
242 {
Sam McCallc008af62018-10-20 15:30:37 +0000243 raw_svector_ostream OS(Scratch);
Sam McCallc18c2802018-06-15 11:06:29 +0000244 D->printQualifiedName(OS);
245 }
Eric Liu83f63e42018-09-03 10:18:21 +0000246 return hash_combine(Scratch, headerToInsertIfAllowed().getValueOr(""));
Sam McCallc18c2802018-06-15 11:06:29 +0000247 }
248
Eric Liu83f63e42018-09-03 10:18:21 +0000249 // The best header to include if include insertion is allowed.
Sam McCallc008af62018-10-20 15:30:37 +0000250 Optional<StringRef> headerToInsertIfAllowed() const {
Eric Liu83f63e42018-09-03 10:18:21 +0000251 if (RankedIncludeHeaders.empty())
Sam McCallc008af62018-10-20 15:30:37 +0000252 return None;
Sam McCallc18c2802018-06-15 11:06:29 +0000253 if (SemaResult && SemaResult->Declaration) {
254 // Avoid inserting new #include if the declaration is found in the current
255 // file e.g. the symbol is forward declared.
256 auto &SM = SemaResult->Declaration->getASTContext().getSourceManager();
257 for (const Decl *RD : SemaResult->Declaration->redecls())
Stephen Kelly43465bf2018-08-09 22:42:26 +0000258 if (SM.isInMainFile(SM.getExpansionLoc(RD->getBeginLoc())))
Sam McCallc008af62018-10-20 15:30:37 +0000259 return None;
Sam McCallc18c2802018-06-15 11:06:29 +0000260 }
Eric Liu83f63e42018-09-03 10:18:21 +0000261 return RankedIncludeHeaders[0];
Sam McCallc18c2802018-06-15 11:06:29 +0000262 }
263
Sam McCallc008af62018-10-20 15:30:37 +0000264 using Bundle = SmallVector<CompletionCandidate, 4>;
Sam McCall98775c52017-12-04 13:49:59 +0000265};
Sam McCallc18c2802018-06-15 11:06:29 +0000266using ScoredBundle =
Sam McCall27c979a2018-06-29 14:47:57 +0000267 std::pair<CompletionCandidate::Bundle, CodeCompletion::Scores>;
Sam McCallc18c2802018-06-15 11:06:29 +0000268struct ScoredBundleGreater {
269 bool operator()(const ScoredBundle &L, const ScoredBundle &R) {
Sam McCall27c979a2018-06-29 14:47:57 +0000270 if (L.second.Total != R.second.Total)
271 return L.second.Total > R.second.Total;
Sam McCallc18c2802018-06-15 11:06:29 +0000272 return L.first.front().Name <
273 R.first.front().Name; // Earlier name is better.
274 }
275};
Sam McCall98775c52017-12-04 13:49:59 +0000276
Sam McCall27c979a2018-06-29 14:47:57 +0000277// Assembles a code completion out of a bundle of >=1 completion candidates.
278// Many of the expensive strings are only computed at this point, once we know
279// the candidate bundle is going to be returned.
280//
281// Many fields are the same for all candidates in a bundle (e.g. name), and are
282// computed from the first candidate, in the constructor.
283// Others vary per candidate, so add() must be called for remaining candidates.
284struct CodeCompletionBuilder {
285 CodeCompletionBuilder(ASTContext &ASTCtx, const CompletionCandidate &C,
286 CodeCompletionString *SemaCCS,
Sam McCallc008af62018-10-20 15:30:37 +0000287 ArrayRef<std::string> QueryScopes,
Sam McCall27c979a2018-06-29 14:47:57 +0000288 const IncludeInserter &Includes, StringRef FileName,
Sam McCall4c077f92018-09-18 09:08:28 +0000289 CodeCompletionContext::Kind ContextKind,
Sam McCall27c979a2018-06-29 14:47:57 +0000290 const CodeCompleteOptions &Opts)
Kadir Cetinkaya516fcda2018-08-23 12:19:39 +0000291 : ASTCtx(ASTCtx), ExtractDocumentation(Opts.IncludeComments),
292 EnableFunctionArgSnippets(Opts.EnableFunctionArgSnippets) {
Sam McCall27c979a2018-06-29 14:47:57 +0000293 add(C, SemaCCS);
294 if (C.SemaResult) {
Sam McCall4e5742a2018-07-06 11:50:49 +0000295 Completion.Origin |= SymbolOrigin::AST;
Sam McCallc008af62018-10-20 15:30:37 +0000296 Completion.Name = StringRef(SemaCCS->getTypedText());
Eric Liuf433c2d2018-07-18 15:31:14 +0000297 if (Completion.Scope.empty()) {
298 if ((C.SemaResult->Kind == CodeCompletionResult::RK_Declaration) ||
299 (C.SemaResult->Kind == CodeCompletionResult::RK_Pattern))
Sam McCall27c979a2018-06-29 14:47:57 +0000300 if (const auto *D = C.SemaResult->getDeclaration())
Sam McCallc008af62018-10-20 15:30:37 +0000301 if (const auto *ND = dyn_cast<NamedDecl>(D))
Sam McCall27c979a2018-06-29 14:47:57 +0000302 Completion.Scope =
303 splitQualifiedName(printQualifiedName(*ND)).first;
Eric Liuf433c2d2018-07-18 15:31:14 +0000304 }
Sam McCall4c077f92018-09-18 09:08:28 +0000305 Completion.Kind = toCompletionItemKind(
306 C.SemaResult->Kind, C.SemaResult->Declaration, ContextKind);
Kadir Cetinkaya0ed5d292018-09-27 14:21:07 +0000307 // Sema could provide more info on whether the completion was a file or
308 // folder.
309 if (Completion.Kind == CompletionItemKind::File &&
310 Completion.Name.back() == '/')
311 Completion.Kind = CompletionItemKind::Folder;
Kadir Cetinkaya2f84d912018-08-08 08:59:29 +0000312 for (const auto &FixIt : C.SemaResult->FixIts) {
313 Completion.FixIts.push_back(
314 toTextEdit(FixIt, ASTCtx.getSourceManager(), ASTCtx.getLangOpts()));
315 }
Kirill Bobyrev4a5ff882018-10-07 14:49:41 +0000316 llvm::sort(Completion.FixIts, [](const TextEdit &X, const TextEdit &Y) {
317 return std::tie(X.range.start.line, X.range.start.character) <
318 std::tie(Y.range.start.line, Y.range.start.character);
319 });
Eric Liu6df66002018-09-06 18:52:26 +0000320 Completion.Deprecated |=
321 (C.SemaResult->Availability == CXAvailability_Deprecated);
Sam McCall27c979a2018-06-29 14:47:57 +0000322 }
323 if (C.IndexResult) {
Sam McCall4e5742a2018-07-06 11:50:49 +0000324 Completion.Origin |= C.IndexResult->Origin;
Sam McCall27c979a2018-06-29 14:47:57 +0000325 if (Completion.Scope.empty())
326 Completion.Scope = C.IndexResult->Scope;
327 if (Completion.Kind == CompletionItemKind::Missing)
328 Completion.Kind = toCompletionItemKind(C.IndexResult->SymInfo.Kind);
329 if (Completion.Name.empty())
330 Completion.Name = C.IndexResult->Name;
Eric Liu670c1472018-09-27 18:46:00 +0000331 // If the completion was visible to Sema, no qualifier is needed. This
332 // avoids unneeded qualifiers in cases like with `using ns::X`.
333 if (Completion.RequiredQualifier.empty() && !C.SemaResult) {
334 StringRef ShortestQualifier = C.IndexResult->Scope;
335 for (StringRef Scope : QueryScopes) {
336 StringRef Qualifier = C.IndexResult->Scope;
337 if (Qualifier.consume_front(Scope) &&
338 Qualifier.size() < ShortestQualifier.size())
339 ShortestQualifier = Qualifier;
340 }
341 Completion.RequiredQualifier = ShortestQualifier;
342 }
Eric Liu6df66002018-09-06 18:52:26 +0000343 Completion.Deprecated |= (C.IndexResult->Flags & Symbol::Deprecated);
Sam McCall27c979a2018-06-29 14:47:57 +0000344 }
Eric Liu83f63e42018-09-03 10:18:21 +0000345
346 // Turn absolute path into a literal string that can be #included.
347 auto Inserted =
348 [&](StringRef Header) -> Expected<std::pair<std::string, bool>> {
349 auto ResolvedDeclaring =
350 toHeaderFile(C.IndexResult->CanonicalDeclaration.FileURI, FileName);
351 if (!ResolvedDeclaring)
352 return ResolvedDeclaring.takeError();
353 auto ResolvedInserted = toHeaderFile(Header, FileName);
354 if (!ResolvedInserted)
355 return ResolvedInserted.takeError();
356 return std::make_pair(
357 Includes.calculateIncludePath(*ResolvedDeclaring, *ResolvedInserted),
358 Includes.shouldInsertInclude(*ResolvedDeclaring, *ResolvedInserted));
359 };
360 bool ShouldInsert = C.headerToInsertIfAllowed().hasValue();
361 // Calculate include paths and edits for all possible headers.
362 for (const auto &Inc : C.RankedIncludeHeaders) {
363 if (auto ToInclude = Inserted(Inc)) {
364 CodeCompletion::IncludeCandidate Include;
365 Include.Header = ToInclude->first;
366 if (ToInclude->second && ShouldInsert)
367 Include.Insertion = Includes.insert(ToInclude->first);
368 Completion.Includes.push_back(std::move(Include));
Sam McCall27c979a2018-06-29 14:47:57 +0000369 } else
Sam McCallbed58852018-07-11 10:35:11 +0000370 log("Failed to generate include insertion edits for adding header "
Sam McCall27c979a2018-06-29 14:47:57 +0000371 "(FileURI='{0}', IncludeHeader='{1}') into {2}",
Eric Liu83f63e42018-09-03 10:18:21 +0000372 C.IndexResult->CanonicalDeclaration.FileURI, Inc, FileName);
Sam McCall27c979a2018-06-29 14:47:57 +0000373 }
Eric Liu83f63e42018-09-03 10:18:21 +0000374 // Prefer includes that do not need edits (i.e. already exist).
375 std::stable_partition(Completion.Includes.begin(),
376 Completion.Includes.end(),
377 [](const CodeCompletion::IncludeCandidate &I) {
378 return !I.Insertion.hasValue();
379 });
Sam McCall27c979a2018-06-29 14:47:57 +0000380 }
381
382 void add(const CompletionCandidate &C, CodeCompletionString *SemaCCS) {
383 assert(bool(C.SemaResult) == bool(SemaCCS));
384 Bundled.emplace_back();
385 BundledEntry &S = Bundled.back();
386 if (C.SemaResult) {
387 getSignature(*SemaCCS, &S.Signature, &S.SnippetSuffix,
388 &Completion.RequiredQualifier);
389 S.ReturnType = getReturnType(*SemaCCS);
390 } else if (C.IndexResult) {
391 S.Signature = C.IndexResult->Signature;
392 S.SnippetSuffix = C.IndexResult->CompletionSnippetSuffix;
Sam McCall2e5700f2018-08-31 13:55:01 +0000393 S.ReturnType = C.IndexResult->ReturnType;
Sam McCall27c979a2018-06-29 14:47:57 +0000394 }
395 if (ExtractDocumentation && Completion.Documentation.empty()) {
Sam McCall2e5700f2018-08-31 13:55:01 +0000396 if (C.IndexResult)
397 Completion.Documentation = C.IndexResult->Documentation;
Sam McCall27c979a2018-06-29 14:47:57 +0000398 else if (C.SemaResult)
399 Completion.Documentation = getDocComment(ASTCtx, *C.SemaResult,
400 /*CommentsFromHeader=*/false);
401 }
402 }
403
404 CodeCompletion build() {
405 Completion.ReturnType = summarizeReturnType();
406 Completion.Signature = summarizeSignature();
407 Completion.SnippetSuffix = summarizeSnippet();
408 Completion.BundleSize = Bundled.size();
409 return std::move(Completion);
410 }
411
412private:
413 struct BundledEntry {
414 std::string SnippetSuffix;
415 std::string Signature;
416 std::string ReturnType;
417 };
418
419 // If all BundledEntrys have the same value for a property, return it.
420 template <std::string BundledEntry::*Member>
421 const std::string *onlyValue() const {
422 auto B = Bundled.begin(), E = Bundled.end();
423 for (auto I = B + 1; I != E; ++I)
424 if (I->*Member != B->*Member)
425 return nullptr;
426 return &(B->*Member);
427 }
428
Kadir Cetinkayaf8b85a32018-08-23 13:14:50 +0000429 template <bool BundledEntry::*Member> const bool *onlyValue() const {
430 auto B = Bundled.begin(), E = Bundled.end();
431 for (auto I = B + 1; I != E; ++I)
432 if (I->*Member != B->*Member)
433 return nullptr;
434 return &(B->*Member);
435 }
436
Sam McCall27c979a2018-06-29 14:47:57 +0000437 std::string summarizeReturnType() const {
438 if (auto *RT = onlyValue<&BundledEntry::ReturnType>())
439 return *RT;
440 return "";
441 }
442
443 std::string summarizeSnippet() const {
Kadir Cetinkaya516fcda2018-08-23 12:19:39 +0000444 auto *Snippet = onlyValue<&BundledEntry::SnippetSuffix>();
445 if (!Snippet)
446 // All bundles are function calls.
Ilya Biryukov4f984702018-09-26 05:45:31 +0000447 // FIXME(ibiryukov): sometimes add template arguments to a snippet, e.g.
448 // we need to complete 'forward<$1>($0)'.
Kadir Cetinkaya516fcda2018-08-23 12:19:39 +0000449 return "($0)";
Ilya Biryukov4f984702018-09-26 05:45:31 +0000450 if (EnableFunctionArgSnippets)
451 return *Snippet;
452
453 // Replace argument snippets with a simplified pattern.
454 if (Snippet->empty())
455 return "";
456 if (Completion.Kind == CompletionItemKind::Function ||
457 Completion.Kind == CompletionItemKind::Method) {
458 // Functions snippets can be of 2 types:
459 // - containing only function arguments, e.g.
460 // foo(${1:int p1}, ${2:int p2});
461 // We transform this pattern to '($0)' or '()'.
462 // - template arguments and function arguments, e.g.
463 // foo<${1:class}>(${2:int p1}).
464 // We transform this pattern to '<$1>()$0' or '<$0>()'.
465
Sam McCallc008af62018-10-20 15:30:37 +0000466 bool EmptyArgs = StringRef(*Snippet).endswith("()");
Ilya Biryukov4f984702018-09-26 05:45:31 +0000467 if (Snippet->front() == '<')
468 return EmptyArgs ? "<$1>()$0" : "<$1>($0)";
469 if (Snippet->front() == '(')
470 return EmptyArgs ? "()" : "($0)";
471 return *Snippet; // Not an arg snippet?
472 }
473 if (Completion.Kind == CompletionItemKind::Reference ||
474 Completion.Kind == CompletionItemKind::Class) {
475 if (Snippet->front() != '<')
476 return *Snippet; // Not an arg snippet?
477
478 // Classes and template using aliases can only have template arguments,
479 // e.g. Foo<${1:class}>.
Sam McCallc008af62018-10-20 15:30:37 +0000480 if (StringRef(*Snippet).endswith("<>"))
Ilya Biryukov4f984702018-09-26 05:45:31 +0000481 return "<>"; // can happen with defaulted template arguments.
482 return "<$0>";
483 }
Kadir Cetinkaya516fcda2018-08-23 12:19:39 +0000484 return *Snippet;
Sam McCall27c979a2018-06-29 14:47:57 +0000485 }
486
487 std::string summarizeSignature() const {
488 if (auto *Signature = onlyValue<&BundledEntry::Signature>())
489 return *Signature;
490 // All bundles are function calls.
491 return "(…)";
492 }
493
494 ASTContext &ASTCtx;
495 CodeCompletion Completion;
496 SmallVector<BundledEntry, 1> Bundled;
497 bool ExtractDocumentation;
Kadir Cetinkaya516fcda2018-08-23 12:19:39 +0000498 bool EnableFunctionArgSnippets;
Sam McCall27c979a2018-06-29 14:47:57 +0000499};
500
Sam McCall545a20d2018-01-19 14:34:02 +0000501// Determine the symbol ID for a Sema code completion result, if possible.
Sam McCallc008af62018-10-20 15:30:37 +0000502Optional<SymbolID> getSymbolID(const CodeCompletionResult &R,
503 const SourceManager &SM) {
Sam McCall545a20d2018-01-19 14:34:02 +0000504 switch (R.Kind) {
505 case CodeCompletionResult::RK_Declaration:
506 case CodeCompletionResult::RK_Pattern: {
Haojian Wuc6ddb462018-08-07 08:57:52 +0000507 return clang::clangd::getSymbolID(R.Declaration);
Sam McCall545a20d2018-01-19 14:34:02 +0000508 }
509 case CodeCompletionResult::RK_Macro:
Eric Liud25f1212018-09-06 09:59:37 +0000510 return clang::clangd::getSymbolID(*R.Macro, R.MacroDefInfo, SM);
Sam McCall545a20d2018-01-19 14:34:02 +0000511 case CodeCompletionResult::RK_Keyword:
512 return None;
513 }
514 llvm_unreachable("unknown CodeCompletionResult kind");
515}
516
Haojian Wu061c73e2018-01-23 11:37:26 +0000517// Scopes of the paritial identifier we're trying to complete.
518// It is used when we query the index for more completion results.
Eric Liu6f648df2017-12-19 16:50:37 +0000519struct SpecifiedScope {
Haojian Wu061c73e2018-01-23 11:37:26 +0000520 // The scopes we should look in, determined by Sema.
521 //
522 // If the qualifier was fully resolved, we look for completions in these
523 // scopes; if there is an unresolved part of the qualifier, it should be
524 // resolved within these scopes.
525 //
526 // Examples of qualified completion:
527 //
528 // "::vec" => {""}
529 // "using namespace std; ::vec^" => {"", "std::"}
530 // "namespace ns {using namespace std;} ns::^" => {"ns::", "std::"}
531 // "std::vec^" => {""} // "std" unresolved
532 //
533 // Examples of unqualified completion:
534 //
535 // "vec^" => {""}
536 // "using namespace std; vec^" => {"", "std::"}
537 // "using namespace std; namespace ns { vec^ }" => {"ns::", "std::", ""}
538 //
539 // "" for global namespace, "ns::" for normal namespace.
540 std::vector<std::string> AccessibleScopes;
541 // The full scope qualifier as typed by the user (without the leading "::").
542 // Set if the qualifier is not fully resolved by Sema.
Sam McCallc008af62018-10-20 15:30:37 +0000543 Optional<std::string> UnresolvedQualifier;
Sam McCall545a20d2018-01-19 14:34:02 +0000544
Eric Liuabbd7132018-11-06 11:17:40 +0000545 // Construct scopes being queried in indexes. The results are deduplicated.
Haojian Wu061c73e2018-01-23 11:37:26 +0000546 // This method format the scopes to match the index request representation.
547 std::vector<std::string> scopesForIndexQuery() {
Eric Liuabbd7132018-11-06 11:17:40 +0000548 std::set<std::string> Results;
549 for (StringRef AS : AccessibleScopes)
550 Results.insert(
551 ((UnresolvedQualifier ? *UnresolvedQualifier : "") + AS).str());
552 return {Results.begin(), Results.end()};
Sam McCall545a20d2018-01-19 14:34:02 +0000553 }
Eric Liu6f648df2017-12-19 16:50:37 +0000554};
555
Eric Liu670c1472018-09-27 18:46:00 +0000556// Get all scopes that will be queried in indexes and whether symbols from
Eric Liu3fac4ef2018-10-17 11:19:02 +0000557// any scope is allowed. The first scope in the list is the preferred scope
558// (e.g. enclosing namespace).
Eric Liu670c1472018-09-27 18:46:00 +0000559std::pair<std::vector<std::string>, bool>
Eric Liu3fac4ef2018-10-17 11:19:02 +0000560getQueryScopes(CodeCompletionContext &CCContext, const Sema &CCSema,
Eric Liu670c1472018-09-27 18:46:00 +0000561 const CodeCompleteOptions &Opts) {
Kirill Bobyrev5a267ed2018-05-29 11:50:51 +0000562 auto GetAllAccessibleScopes = [](CodeCompletionContext &CCContext) {
Haojian Wu061c73e2018-01-23 11:37:26 +0000563 SpecifiedScope Info;
Kirill Bobyrev5a267ed2018-05-29 11:50:51 +0000564 for (auto *Context : CCContext.getVisitedContexts()) {
Haojian Wu061c73e2018-01-23 11:37:26 +0000565 if (isa<TranslationUnitDecl>(Context))
566 Info.AccessibleScopes.push_back(""); // global namespace
Mikael Holmenfa41add2018-10-18 06:00:39 +0000567 else if (isa<NamespaceDecl>(Context))
Eric Liu3fac4ef2018-10-17 11:19:02 +0000568 Info.AccessibleScopes.push_back(printNamespaceScope(*Context));
Haojian Wu061c73e2018-01-23 11:37:26 +0000569 }
570 return Info;
571 };
572
573 auto SS = CCContext.getCXXScopeSpecifier();
574
575 // Unqualified completion (e.g. "vec^").
576 if (!SS) {
Eric Liu3fac4ef2018-10-17 11:19:02 +0000577 std::vector<std::string> Scopes;
578 std::string EnclosingScope = printNamespaceScope(*CCSema.CurContext);
579 Scopes.push_back(EnclosingScope);
580 for (auto &S : GetAllAccessibleScopes(CCContext).scopesForIndexQuery()) {
581 if (EnclosingScope != S)
582 Scopes.push_back(std::move(S));
583 }
Eric Liu670c1472018-09-27 18:46:00 +0000584 // Allow AllScopes completion only for there is no explicit scope qualifier.
585 return {Scopes, Opts.AllScopes};
Haojian Wu061c73e2018-01-23 11:37:26 +0000586 }
587
588 // Qualified completion ("std::vec^"), we have two cases depending on whether
589 // the qualifier can be resolved by Sema.
590 if ((*SS)->isValid()) { // Resolved qualifier.
Eric Liu670c1472018-09-27 18:46:00 +0000591 return {GetAllAccessibleScopes(CCContext).scopesForIndexQuery(), false};
Haojian Wu061c73e2018-01-23 11:37:26 +0000592 }
593
594 // Unresolved qualifier.
595 // FIXME: When Sema can resolve part of a scope chain (e.g.
596 // "known::unknown::id"), we should expand the known part ("known::") rather
597 // than treating the whole thing as unknown.
598 SpecifiedScope Info;
599 Info.AccessibleScopes.push_back(""); // global namespace
600
601 Info.UnresolvedQualifier =
Eric Liu3fac4ef2018-10-17 11:19:02 +0000602 Lexer::getSourceText(CharSourceRange::getCharRange((*SS)->getRange()),
603 CCSema.SourceMgr, clang::LangOptions())
Kirill Bobyrev5a267ed2018-05-29 11:50:51 +0000604 .ltrim("::");
Haojian Wu061c73e2018-01-23 11:37:26 +0000605 // Sema excludes the trailing "::".
606 if (!Info.UnresolvedQualifier->empty())
607 *Info.UnresolvedQualifier += "::";
608
Eric Liu670c1472018-09-27 18:46:00 +0000609 return {Info.scopesForIndexQuery(), false};
Haojian Wu061c73e2018-01-23 11:37:26 +0000610}
611
Eric Liu42abe412018-05-24 11:20:19 +0000612// Should we perform index-based completion in a context of the specified kind?
613// FIXME: consider allowing completion, but restricting the result types.
614bool contextAllowsIndex(enum CodeCompletionContext::Kind K) {
615 switch (K) {
616 case CodeCompletionContext::CCC_TopLevel:
617 case CodeCompletionContext::CCC_ObjCInterface:
618 case CodeCompletionContext::CCC_ObjCImplementation:
619 case CodeCompletionContext::CCC_ObjCIvarList:
620 case CodeCompletionContext::CCC_ClassStructUnion:
621 case CodeCompletionContext::CCC_Statement:
622 case CodeCompletionContext::CCC_Expression:
623 case CodeCompletionContext::CCC_ObjCMessageReceiver:
624 case CodeCompletionContext::CCC_EnumTag:
625 case CodeCompletionContext::CCC_UnionTag:
626 case CodeCompletionContext::CCC_ClassOrStructTag:
627 case CodeCompletionContext::CCC_ObjCProtocolName:
628 case CodeCompletionContext::CCC_Namespace:
629 case CodeCompletionContext::CCC_Type:
Eric Liu42abe412018-05-24 11:20:19 +0000630 case CodeCompletionContext::CCC_ParenthesizedExpression:
631 case CodeCompletionContext::CCC_ObjCInterfaceName:
632 case CodeCompletionContext::CCC_ObjCCategoryName:
Kadir Cetinkayab9157902018-10-24 15:24:29 +0000633 case CodeCompletionContext::CCC_Symbol:
634 case CodeCompletionContext::CCC_SymbolOrNewName:
Eric Liu42abe412018-05-24 11:20:19 +0000635 return true;
Eric Liu42abe412018-05-24 11:20:19 +0000636 case CodeCompletionContext::CCC_OtherWithMacros:
637 case CodeCompletionContext::CCC_DotMemberAccess:
638 case CodeCompletionContext::CCC_ArrowMemberAccess:
639 case CodeCompletionContext::CCC_ObjCPropertyAccess:
640 case CodeCompletionContext::CCC_MacroName:
641 case CodeCompletionContext::CCC_MacroNameUse:
642 case CodeCompletionContext::CCC_PreprocessorExpression:
643 case CodeCompletionContext::CCC_PreprocessorDirective:
Eric Liu42abe412018-05-24 11:20:19 +0000644 case CodeCompletionContext::CCC_SelectorName:
645 case CodeCompletionContext::CCC_TypeQualifiers:
646 case CodeCompletionContext::CCC_ObjCInstanceMessage:
647 case CodeCompletionContext::CCC_ObjCClassMessage:
Sam McCall4c077f92018-09-18 09:08:28 +0000648 case CodeCompletionContext::CCC_IncludedFile:
Kadir Cetinkayab9157902018-10-24 15:24:29 +0000649 // FIXME: Provide identifier based completions for the following contexts:
650 case CodeCompletionContext::CCC_Other: // Be conservative.
651 case CodeCompletionContext::CCC_NaturalLanguage:
Eric Liu42abe412018-05-24 11:20:19 +0000652 case CodeCompletionContext::CCC_Recovery:
Kadir Cetinkayab9157902018-10-24 15:24:29 +0000653 case CodeCompletionContext::CCC_NewName:
Eric Liu42abe412018-05-24 11:20:19 +0000654 return false;
655 }
656 llvm_unreachable("unknown code completion context");
657}
658
Sam McCall4caa8512018-06-07 12:49:17 +0000659// Some member calls are blacklisted because they're so rarely useful.
660static bool isBlacklistedMember(const NamedDecl &D) {
661 // Destructor completion is rarely useful, and works inconsistently.
662 // (s.^ completes ~string, but s.~st^ is an error).
663 if (D.getKind() == Decl::CXXDestructor)
664 return true;
665 // Injected name may be useful for A::foo(), but who writes A::A::foo()?
666 if (auto *R = dyn_cast_or_null<RecordDecl>(&D))
667 if (R->isInjectedClassName())
668 return true;
669 // Explicit calls to operators are also rare.
670 auto NameKind = D.getDeclName().getNameKind();
671 if (NameKind == DeclarationName::CXXOperatorName ||
672 NameKind == DeclarationName::CXXLiteralOperatorName ||
673 NameKind == DeclarationName::CXXConversionFunctionName)
674 return true;
675 return false;
676}
677
Sam McCall545a20d2018-01-19 14:34:02 +0000678// The CompletionRecorder captures Sema code-complete output, including context.
679// It filters out ignored results (but doesn't apply fuzzy-filtering yet).
680// It doesn't do scoring or conversion to CompletionItem yet, as we want to
681// merge with index results first.
Ilya Biryukovddf6a332018-03-02 12:28:27 +0000682// Generally the fields and methods of this object should only be used from
683// within the callback.
Sam McCall545a20d2018-01-19 14:34:02 +0000684struct CompletionRecorder : public CodeCompleteConsumer {
Ilya Biryukovddf6a332018-03-02 12:28:27 +0000685 CompletionRecorder(const CodeCompleteOptions &Opts,
Sam McCallc008af62018-10-20 15:30:37 +0000686 unique_function<void()> ResultsCallback)
Sam McCall545a20d2018-01-19 14:34:02 +0000687 : CodeCompleteConsumer(Opts.getClangCompleteOpts(),
Sam McCall98775c52017-12-04 13:49:59 +0000688 /*OutputIsBinary=*/false),
Sam McCall545a20d2018-01-19 14:34:02 +0000689 CCContext(CodeCompletionContext::CCC_Other), Opts(Opts),
690 CCAllocator(std::make_shared<GlobalCodeCompletionAllocator>()),
Ilya Biryukovddf6a332018-03-02 12:28:27 +0000691 CCTUInfo(CCAllocator), ResultsCallback(std::move(ResultsCallback)) {
692 assert(this->ResultsCallback);
693 }
694
Sam McCall545a20d2018-01-19 14:34:02 +0000695 std::vector<CodeCompletionResult> Results;
696 CodeCompletionContext CCContext;
697 Sema *CCSema = nullptr; // Sema that created the results.
698 // FIXME: Sema is scary. Can we store ASTContext and Preprocessor, instead?
Sam McCall98775c52017-12-04 13:49:59 +0000699
Sam McCall545a20d2018-01-19 14:34:02 +0000700 void ProcessCodeCompleteResults(class Sema &S, CodeCompletionContext Context,
701 CodeCompletionResult *InResults,
Sam McCall98775c52017-12-04 13:49:59 +0000702 unsigned NumResults) override final {
Eric Liu485074f2018-07-11 13:15:31 +0000703 // Results from recovery mode are generally useless, and the callback after
704 // recovery (if any) is usually more interesting. To make sure we handle the
705 // future callback from sema, we just ignore all callbacks in recovery mode,
706 // as taking only results from recovery mode results in poor completion
707 // results.
708 // FIXME: in case there is no future sema completion callback after the
709 // recovery mode, we might still want to provide some results (e.g. trivial
710 // identifier-based completion).
711 if (Context.getKind() == CodeCompletionContext::CCC_Recovery) {
712 log("Code complete: Ignoring sema code complete callback with Recovery "
713 "context.");
714 return;
715 }
Eric Liu42abe412018-05-24 11:20:19 +0000716 // If a callback is called without any sema result and the context does not
717 // support index-based completion, we simply skip it to give way to
718 // potential future callbacks with results.
719 if (NumResults == 0 && !contextAllowsIndex(Context.getKind()))
720 return;
Ilya Biryukov94da7bd2018-03-16 15:23:44 +0000721 if (CCSema) {
Sam McCallbed58852018-07-11 10:35:11 +0000722 log("Multiple code complete callbacks (parser backtracked?). "
Ilya Biryukov94da7bd2018-03-16 15:23:44 +0000723 "Dropping results from context {0}, keeping results from {1}.",
Eric Liu42abe412018-05-24 11:20:19 +0000724 getCompletionKindString(Context.getKind()),
Sam McCallbed58852018-07-11 10:35:11 +0000725 getCompletionKindString(this->CCContext.getKind()));
Ilya Biryukov94da7bd2018-03-16 15:23:44 +0000726 return;
727 }
Sam McCall545a20d2018-01-19 14:34:02 +0000728 // Record the completion context.
Sam McCall545a20d2018-01-19 14:34:02 +0000729 CCSema = &S;
730 CCContext = Context;
Eric Liu6f648df2017-12-19 16:50:37 +0000731
Sam McCall545a20d2018-01-19 14:34:02 +0000732 // Retain the results we might want.
Sam McCall98775c52017-12-04 13:49:59 +0000733 for (unsigned I = 0; I < NumResults; ++I) {
Sam McCall545a20d2018-01-19 14:34:02 +0000734 auto &Result = InResults[I];
Sam McCalle8437cb2018-10-24 13:51:44 +0000735 // Class members that are shadowed by subclasses are usually noise.
736 if (Result.Hidden && Result.Declaration &&
737 Result.Declaration->isCXXClassMember())
Ilya Biryukovf60bf342018-01-10 13:51:09 +0000738 continue;
Sam McCall545a20d2018-01-19 14:34:02 +0000739 if (!Opts.IncludeIneligibleResults &&
Sam McCall98775c52017-12-04 13:49:59 +0000740 (Result.Availability == CXAvailability_NotAvailable ||
741 Result.Availability == CXAvailability_NotAccessible))
742 continue;
Sam McCall4caa8512018-06-07 12:49:17 +0000743 if (Result.Declaration &&
744 !Context.getBaseType().isNull() // is this a member-access context?
745 && isBlacklistedMember(*Result.Declaration))
Sam McCalld2a95922018-01-22 21:05:00 +0000746 continue;
Ilya Biryukov53d6d932018-03-06 16:45:21 +0000747 // We choose to never append '::' to completion results in clangd.
748 Result.StartsNestedNameSpecifier = false;
Sam McCall545a20d2018-01-19 14:34:02 +0000749 Results.push_back(Result);
Sam McCall98775c52017-12-04 13:49:59 +0000750 }
Ilya Biryukovddf6a332018-03-02 12:28:27 +0000751 ResultsCallback();
Sam McCall98775c52017-12-04 13:49:59 +0000752 }
753
Sam McCall545a20d2018-01-19 14:34:02 +0000754 CodeCompletionAllocator &getAllocator() override { return *CCAllocator; }
Sam McCall98775c52017-12-04 13:49:59 +0000755 CodeCompletionTUInfo &getCodeCompletionTUInfo() override { return CCTUInfo; }
756
Sam McCall545a20d2018-01-19 14:34:02 +0000757 // Returns the filtering/sorting name for Result, which must be from Results.
758 // Returned string is owned by this recorder (or the AST).
Sam McCallc008af62018-10-20 15:30:37 +0000759 StringRef getName(const CodeCompletionResult &Result) {
Sam McCall98775c52017-12-04 13:49:59 +0000760 switch (Result.Kind) {
761 case CodeCompletionResult::RK_Declaration:
762 if (auto *ID = Result.Declaration->getIdentifier())
Sam McCall545a20d2018-01-19 14:34:02 +0000763 return ID->getName();
Sam McCall98775c52017-12-04 13:49:59 +0000764 break;
765 case CodeCompletionResult::RK_Keyword:
Sam McCall545a20d2018-01-19 14:34:02 +0000766 return Result.Keyword;
Sam McCall98775c52017-12-04 13:49:59 +0000767 case CodeCompletionResult::RK_Macro:
Sam McCall545a20d2018-01-19 14:34:02 +0000768 return Result.Macro->getName();
Sam McCall98775c52017-12-04 13:49:59 +0000769 case CodeCompletionResult::RK_Pattern:
Sam McCall545a20d2018-01-19 14:34:02 +0000770 return Result.Pattern->getTypedText();
Sam McCall98775c52017-12-04 13:49:59 +0000771 }
Ilya Biryukov43714502018-05-16 12:32:44 +0000772 auto *CCS = codeCompletionString(Result);
Sam McCall545a20d2018-01-19 14:34:02 +0000773 return CCS->getTypedText();
Sam McCall98775c52017-12-04 13:49:59 +0000774 }
775
Sam McCall545a20d2018-01-19 14:34:02 +0000776 // Build a CodeCompletion string for R, which must be from Results.
777 // The CCS will be owned by this recorder.
Ilya Biryukov43714502018-05-16 12:32:44 +0000778 CodeCompletionString *codeCompletionString(const CodeCompletionResult &R) {
Sam McCall545a20d2018-01-19 14:34:02 +0000779 // CodeCompletionResult doesn't seem to be const-correct. We own it, anyway.
780 return const_cast<CodeCompletionResult &>(R).CreateCodeCompletionString(
Ilya Biryukov43714502018-05-16 12:32:44 +0000781 *CCSema, CCContext, *CCAllocator, CCTUInfo,
782 /*IncludeBriefComments=*/false);
Sam McCall98775c52017-12-04 13:49:59 +0000783 }
784
Sam McCall545a20d2018-01-19 14:34:02 +0000785private:
786 CodeCompleteOptions Opts;
787 std::shared_ptr<GlobalCodeCompletionAllocator> CCAllocator;
Sam McCall98775c52017-12-04 13:49:59 +0000788 CodeCompletionTUInfo CCTUInfo;
Sam McCallc008af62018-10-20 15:30:37 +0000789 unique_function<void()> ResultsCallback;
Sam McCall545a20d2018-01-19 14:34:02 +0000790};
791
Ilya Biryukov8a0f76b2018-08-17 09:32:30 +0000792struct ScoredSignature {
793 // When set, requires documentation to be requested from the index with this
794 // ID.
Sam McCallc008af62018-10-20 15:30:37 +0000795 Optional<SymbolID> IDForDoc;
Ilya Biryukov8a0f76b2018-08-17 09:32:30 +0000796 SignatureInformation Signature;
797 SignatureQualitySignals Quality;
798};
Kadir Cetinkayae486e372018-08-13 08:40:05 +0000799
Sam McCall98775c52017-12-04 13:49:59 +0000800class SignatureHelpCollector final : public CodeCompleteConsumer {
Sam McCall98775c52017-12-04 13:49:59 +0000801public:
802 SignatureHelpCollector(const clang::CodeCompleteOptions &CodeCompleteOpts,
Sam McCall046557b2018-09-03 16:37:59 +0000803 const SymbolIndex *Index, SignatureHelp &SigHelp)
Ilya Biryukov8a0f76b2018-08-17 09:32:30 +0000804 : CodeCompleteConsumer(CodeCompleteOpts,
805 /*OutputIsBinary=*/false),
Sam McCall98775c52017-12-04 13:49:59 +0000806 SigHelp(SigHelp),
807 Allocator(std::make_shared<clang::GlobalCodeCompletionAllocator>()),
Ilya Biryukov8a0f76b2018-08-17 09:32:30 +0000808 CCTUInfo(Allocator), Index(Index) {}
Sam McCall98775c52017-12-04 13:49:59 +0000809
810 void ProcessOverloadCandidates(Sema &S, unsigned CurrentArg,
811 OverloadCandidate *Candidates,
Ilya Biryukov43c292c2018-08-30 13:14:31 +0000812 unsigned NumCandidates,
813 SourceLocation OpenParLoc) override {
814 assert(!OpenParLoc.isInvalid());
815 SourceManager &SrcMgr = S.getSourceManager();
816 OpenParLoc = SrcMgr.getFileLoc(OpenParLoc);
817 if (SrcMgr.isInMainFile(OpenParLoc))
818 SigHelp.argListStart = sourceLocToPosition(SrcMgr, OpenParLoc);
819 else
820 elog("Location oustide main file in signature help: {0}",
821 OpenParLoc.printToString(SrcMgr));
822
Kadir Cetinkayae486e372018-08-13 08:40:05 +0000823 std::vector<ScoredSignature> ScoredSignatures;
Sam McCall98775c52017-12-04 13:49:59 +0000824 SigHelp.signatures.reserve(NumCandidates);
Kadir Cetinkayae486e372018-08-13 08:40:05 +0000825 ScoredSignatures.reserve(NumCandidates);
Sam McCall98775c52017-12-04 13:49:59 +0000826 // FIXME(rwols): How can we determine the "active overload candidate"?
827 // Right now the overloaded candidates seem to be provided in a "best fit"
828 // order, so I'm not too worried about this.
829 SigHelp.activeSignature = 0;
830 assert(CurrentArg <= (unsigned)std::numeric_limits<int>::max() &&
831 "too many arguments");
832 SigHelp.activeParameter = static_cast<int>(CurrentArg);
833 for (unsigned I = 0; I < NumCandidates; ++I) {
Ilya Biryukov8fd44bb2018-08-14 09:36:32 +0000834 OverloadCandidate Candidate = Candidates[I];
835 // We want to avoid showing instantiated signatures, because they may be
836 // long in some cases (e.g. when 'T' is substituted with 'std::string', we
837 // would get 'std::basic_string<char>').
838 if (auto *Func = Candidate.getFunction()) {
839 if (auto *Pattern = Func->getTemplateInstantiationPattern())
840 Candidate = OverloadCandidate(Pattern);
841 }
842
Sam McCall98775c52017-12-04 13:49:59 +0000843 const auto *CCS = Candidate.CreateSignatureString(
844 CurrentArg, S, *Allocator, CCTUInfo, true);
845 assert(CCS && "Expected the CodeCompletionString to be non-null");
Kadir Cetinkayae486e372018-08-13 08:40:05 +0000846 ScoredSignatures.push_back(processOverloadCandidate(
Ilya Biryukov43714502018-05-16 12:32:44 +0000847 Candidate, *CCS,
Ilya Biryukov5f4a3512018-08-17 09:29:38 +0000848 Candidate.getFunction()
849 ? getDeclComment(S.getASTContext(), *Candidate.getFunction())
850 : ""));
Sam McCall98775c52017-12-04 13:49:59 +0000851 }
Ilya Biryukov8a0f76b2018-08-17 09:32:30 +0000852
853 // Sema does not load the docs from the preamble, so we need to fetch extra
854 // docs from the index instead.
Sam McCallc008af62018-10-20 15:30:37 +0000855 DenseMap<SymbolID, std::string> FetchedDocs;
Ilya Biryukov8a0f76b2018-08-17 09:32:30 +0000856 if (Index) {
857 LookupRequest IndexRequest;
858 for (const auto &S : ScoredSignatures) {
859 if (!S.IDForDoc)
860 continue;
861 IndexRequest.IDs.insert(*S.IDForDoc);
862 }
863 Index->lookup(IndexRequest, [&](const Symbol &S) {
Sam McCall2e5700f2018-08-31 13:55:01 +0000864 if (!S.Documentation.empty())
865 FetchedDocs[S.ID] = S.Documentation;
Ilya Biryukov8a0f76b2018-08-17 09:32:30 +0000866 });
867 log("SigHelp: requested docs for {0} symbols from the index, got {1} "
868 "symbols with non-empty docs in the response",
869 IndexRequest.IDs.size(), FetchedDocs.size());
870 }
871
Kirill Bobyrev4a5ff882018-10-07 14:49:41 +0000872 llvm::sort(
873 ScoredSignatures,
Ilya Biryukov8a0f76b2018-08-17 09:32:30 +0000874 [](const ScoredSignature &L, const ScoredSignature &R) {
875 // Ordering follows:
876 // - Less number of parameters is better.
877 // - Function is better than FunctionType which is better than
878 // Function Template.
879 // - High score is better.
880 // - Shorter signature is better.
881 // - Alphebatically smaller is better.
882 if (L.Quality.NumberOfParameters != R.Quality.NumberOfParameters)
883 return L.Quality.NumberOfParameters < R.Quality.NumberOfParameters;
884 if (L.Quality.NumberOfOptionalParameters !=
885 R.Quality.NumberOfOptionalParameters)
886 return L.Quality.NumberOfOptionalParameters <
887 R.Quality.NumberOfOptionalParameters;
888 if (L.Quality.Kind != R.Quality.Kind) {
889 using OC = CodeCompleteConsumer::OverloadCandidate;
890 switch (L.Quality.Kind) {
891 case OC::CK_Function:
892 return true;
893 case OC::CK_FunctionType:
894 return R.Quality.Kind != OC::CK_Function;
895 case OC::CK_FunctionTemplate:
896 return false;
897 }
898 llvm_unreachable("Unknown overload candidate type.");
899 }
900 if (L.Signature.label.size() != R.Signature.label.size())
901 return L.Signature.label.size() < R.Signature.label.size();
902 return L.Signature.label < R.Signature.label;
903 });
904
905 for (auto &SS : ScoredSignatures) {
906 auto IndexDocIt =
907 SS.IDForDoc ? FetchedDocs.find(*SS.IDForDoc) : FetchedDocs.end();
908 if (IndexDocIt != FetchedDocs.end())
909 SS.Signature.documentation = IndexDocIt->second;
910
911 SigHelp.signatures.push_back(std::move(SS.Signature));
912 }
Sam McCall98775c52017-12-04 13:49:59 +0000913 }
914
915 GlobalCodeCompletionAllocator &getAllocator() override { return *Allocator; }
916
917 CodeCompletionTUInfo &getCodeCompletionTUInfo() override { return CCTUInfo; }
918
919private:
Eric Liu63696e12017-12-20 17:24:31 +0000920 // FIXME(ioeric): consider moving CodeCompletionString logic here to
921 // CompletionString.h.
Kadir Cetinkayae486e372018-08-13 08:40:05 +0000922 ScoredSignature processOverloadCandidate(const OverloadCandidate &Candidate,
923 const CodeCompletionString &CCS,
Sam McCallc008af62018-10-20 15:30:37 +0000924 StringRef DocComment) const {
Ilya Biryukov8a0f76b2018-08-17 09:32:30 +0000925 SignatureInformation Signature;
Kadir Cetinkayae486e372018-08-13 08:40:05 +0000926 SignatureQualitySignals Signal;
Sam McCall98775c52017-12-04 13:49:59 +0000927 const char *ReturnType = nullptr;
928
Ilya Biryukov8a0f76b2018-08-17 09:32:30 +0000929 Signature.documentation = formatDocumentation(CCS, DocComment);
Kadir Cetinkayae486e372018-08-13 08:40:05 +0000930 Signal.Kind = Candidate.getKind();
Sam McCall98775c52017-12-04 13:49:59 +0000931
932 for (const auto &Chunk : CCS) {
933 switch (Chunk.Kind) {
934 case CodeCompletionString::CK_ResultType:
935 // A piece of text that describes the type of an entity or,
936 // for functions and methods, the return type.
937 assert(!ReturnType && "Unexpected CK_ResultType");
938 ReturnType = Chunk.Text;
939 break;
940 case CodeCompletionString::CK_Placeholder:
941 // A string that acts as a placeholder for, e.g., a function call
942 // argument.
943 // Intentional fallthrough here.
944 case CodeCompletionString::CK_CurrentParameter: {
945 // A piece of text that describes the parameter that corresponds to
946 // the code-completion location within a function call, message send,
947 // macro invocation, etc.
Ilya Biryukov8a0f76b2018-08-17 09:32:30 +0000948 Signature.label += Chunk.Text;
Sam McCall98775c52017-12-04 13:49:59 +0000949 ParameterInformation Info;
950 Info.label = Chunk.Text;
Ilya Biryukov8a0f76b2018-08-17 09:32:30 +0000951 Signature.parameters.push_back(std::move(Info));
Kadir Cetinkayae486e372018-08-13 08:40:05 +0000952 Signal.NumberOfParameters++;
953 Signal.ContainsActiveParameter = true;
Sam McCall98775c52017-12-04 13:49:59 +0000954 break;
955 }
956 case CodeCompletionString::CK_Optional: {
957 // The rest of the parameters are defaulted/optional.
958 assert(Chunk.Optional &&
959 "Expected the optional code completion string to be non-null.");
Ilya Biryukov8a0f76b2018-08-17 09:32:30 +0000960 Signature.label += getOptionalParameters(*Chunk.Optional,
961 Signature.parameters, Signal);
Sam McCall98775c52017-12-04 13:49:59 +0000962 break;
963 }
964 case CodeCompletionString::CK_VerticalSpace:
965 break;
966 default:
Ilya Biryukov8a0f76b2018-08-17 09:32:30 +0000967 Signature.label += Chunk.Text;
Sam McCall98775c52017-12-04 13:49:59 +0000968 break;
969 }
970 }
971 if (ReturnType) {
Ilya Biryukov8a0f76b2018-08-17 09:32:30 +0000972 Signature.label += " -> ";
973 Signature.label += ReturnType;
Sam McCall98775c52017-12-04 13:49:59 +0000974 }
Ilya Biryukov8a0f76b2018-08-17 09:32:30 +0000975 dlog("Signal for {0}: {1}", Signature, Signal);
976 ScoredSignature Result;
977 Result.Signature = std::move(Signature);
978 Result.Quality = Signal;
979 Result.IDForDoc =
980 Result.Signature.documentation.empty() && Candidate.getFunction()
981 ? clangd::getSymbolID(Candidate.getFunction())
Sam McCallc008af62018-10-20 15:30:37 +0000982 : None;
Ilya Biryukov8a0f76b2018-08-17 09:32:30 +0000983 return Result;
Sam McCall98775c52017-12-04 13:49:59 +0000984 }
985
986 SignatureHelp &SigHelp;
987 std::shared_ptr<clang::GlobalCodeCompletionAllocator> Allocator;
988 CodeCompletionTUInfo CCTUInfo;
Ilya Biryukov8a0f76b2018-08-17 09:32:30 +0000989 const SymbolIndex *Index;
Sam McCall98775c52017-12-04 13:49:59 +0000990}; // SignatureHelpCollector
991
Sam McCall545a20d2018-01-19 14:34:02 +0000992struct SemaCompleteInput {
993 PathRef FileName;
994 const tooling::CompileCommand &Command;
Eric Liub1d75422018-10-02 10:43:55 +0000995 const PreambleData *Preamble;
Sam McCall545a20d2018-01-19 14:34:02 +0000996 StringRef Contents;
997 Position Pos;
Sam McCallc008af62018-10-20 15:30:37 +0000998 IntrusiveRefCntPtr<vfs::FileSystem> VFS;
Sam McCall545a20d2018-01-19 14:34:02 +0000999 std::shared_ptr<PCHContainerOperations> PCHs;
1000};
1001
1002// Invokes Sema code completion on a file.
Sam McCall3f0243f2018-07-03 08:09:29 +00001003// If \p Includes is set, it will be updated based on the compiler invocation.
Sam McCalld1a7a372018-01-31 13:40:48 +00001004bool semaCodeComplete(std::unique_ptr<CodeCompleteConsumer> Consumer,
Sam McCall545a20d2018-01-19 14:34:02 +00001005 const clang::CodeCompleteOptions &Options,
Eric Liu63f419a2018-05-15 15:29:32 +00001006 const SemaCompleteInput &Input,
Sam McCall3f0243f2018-07-03 08:09:29 +00001007 IncludeStructure *Includes = nullptr) {
Ilya Biryukovddf6a332018-03-02 12:28:27 +00001008 trace::Span Tracer("Sema completion");
Sam McCall98775c52017-12-04 13:49:59 +00001009 std::vector<const char *> ArgStrs;
Sam McCall545a20d2018-01-19 14:34:02 +00001010 for (const auto &S : Input.Command.CommandLine)
Sam McCall98775c52017-12-04 13:49:59 +00001011 ArgStrs.push_back(S.c_str());
1012
Ilya Biryukova9cf3112018-02-13 17:15:06 +00001013 if (Input.VFS->setCurrentWorkingDirectory(Input.Command.Directory)) {
1014 log("Couldn't set working directory");
1015 // We run parsing anyway, our lit-tests rely on results for non-existing
1016 // working dirs.
1017 }
Sam McCall98775c52017-12-04 13:49:59 +00001018
Sam McCallc008af62018-10-20 15:30:37 +00001019 IntrusiveRefCntPtr<vfs::FileSystem> VFS = Input.VFS;
Eric Liub1d75422018-10-02 10:43:55 +00001020 if (Input.Preamble && Input.Preamble->StatCache)
1021 VFS = Input.Preamble->StatCache->getConsumingFS(std::move(VFS));
Sam McCall98775c52017-12-04 13:49:59 +00001022 IgnoreDiagnostics DummyDiagsConsumer;
1023 auto CI = createInvocationFromCommandLine(
1024 ArgStrs,
1025 CompilerInstance::createDiagnostics(new DiagnosticOptions,
1026 &DummyDiagsConsumer, false),
Eric Liub1d75422018-10-02 10:43:55 +00001027 VFS);
Ilya Biryukovb6ad25c2018-02-09 13:51:57 +00001028 if (!CI) {
Sam McCallbed58852018-07-11 10:35:11 +00001029 elog("Couldn't create CompilerInvocation");
Ilya Biryukovb6ad25c2018-02-09 13:51:57 +00001030 return false;
1031 }
Ilya Biryukov981a35d2018-05-28 12:11:37 +00001032 auto &FrontendOpts = CI->getFrontendOpts();
1033 FrontendOpts.DisableFree = false;
Sam McCall98775c52017-12-04 13:49:59 +00001034 FrontendOpts.SkipFunctionBodies = true;
Ilya Biryukov981a35d2018-05-28 12:11:37 +00001035 CI->getLangOpts()->CommentOpts.ParseAllComments = true;
1036 // Disable typo correction in Sema.
1037 CI->getLangOpts()->SpellChecking = false;
1038 // Setup code completion.
Sam McCall98775c52017-12-04 13:49:59 +00001039 FrontendOpts.CodeCompleteOpts = Options;
Sam McCall545a20d2018-01-19 14:34:02 +00001040 FrontendOpts.CodeCompletionAt.FileName = Input.FileName;
Sam McCalla4962cc2018-04-27 11:59:28 +00001041 auto Offset = positionToOffset(Input.Contents, Input.Pos);
1042 if (!Offset) {
Sam McCallbed58852018-07-11 10:35:11 +00001043 elog("Code completion position was invalid {0}", Offset.takeError());
Sam McCalla4962cc2018-04-27 11:59:28 +00001044 return false;
1045 }
1046 std::tie(FrontendOpts.CodeCompletionAt.Line,
1047 FrontendOpts.CodeCompletionAt.Column) =
1048 offsetToClangLineColumn(Input.Contents, *Offset);
Sam McCall98775c52017-12-04 13:49:59 +00001049
Sam McCallc008af62018-10-20 15:30:37 +00001050 std::unique_ptr<MemoryBuffer> ContentsBuffer =
1051 MemoryBuffer::getMemBufferCopy(Input.Contents, Input.FileName);
Ilya Biryukov981a35d2018-05-28 12:11:37 +00001052 // The diagnostic options must be set before creating a CompilerInstance.
1053 CI->getDiagnosticOpts().IgnoreWarnings = true;
1054 // We reuse the preamble whether it's valid or not. This is a
1055 // correctness/performance tradeoff: building without a preamble is slow, and
1056 // completion is latency-sensitive.
Sam McCallebef8122018-09-14 12:36:06 +00001057 // However, if we're completing *inside* the preamble section of the draft,
1058 // overriding the preamble will break sema completion. Fortunately we can just
1059 // skip all includes in this case; these completions are really simple.
1060 bool CompletingInPreamble =
1061 ComputePreambleBounds(*CI->getLangOpts(), ContentsBuffer.get(), 0).Size >
1062 *Offset;
Ilya Biryukov981a35d2018-05-28 12:11:37 +00001063 // NOTE: we must call BeginSourceFile after prepareCompilerInstance. Otherwise
1064 // the remapped buffers do not get freed.
1065 auto Clang = prepareCompilerInstance(
Eric Liub1d75422018-10-02 10:43:55 +00001066 std::move(CI),
1067 (Input.Preamble && !CompletingInPreamble) ? &Input.Preamble->Preamble
1068 : nullptr,
1069 std::move(ContentsBuffer), std::move(Input.PCHs), std::move(VFS),
Sam McCallebef8122018-09-14 12:36:06 +00001070 DummyDiagsConsumer);
1071 Clang->getPreprocessorOpts().SingleFileParseMode = CompletingInPreamble;
Sam McCall98775c52017-12-04 13:49:59 +00001072 Clang->setCodeCompletionConsumer(Consumer.release());
1073
1074 SyntaxOnlyAction Action;
1075 if (!Action.BeginSourceFile(*Clang, Clang->getFrontendOpts().Inputs[0])) {
Sam McCallbed58852018-07-11 10:35:11 +00001076 log("BeginSourceFile() failed when running codeComplete for {0}",
Sam McCalld1a7a372018-01-31 13:40:48 +00001077 Input.FileName);
Sam McCall98775c52017-12-04 13:49:59 +00001078 return false;
1079 }
Sam McCall3f0243f2018-07-03 08:09:29 +00001080 if (Includes)
1081 Clang->getPreprocessor().addPPCallbacks(
1082 collectIncludeStructureCallback(Clang->getSourceManager(), Includes));
Sam McCall98775c52017-12-04 13:49:59 +00001083 if (!Action.Execute()) {
Sam McCallbed58852018-07-11 10:35:11 +00001084 log("Execute() failed when running codeComplete for {0}", Input.FileName);
Sam McCall98775c52017-12-04 13:49:59 +00001085 return false;
1086 }
Sam McCall98775c52017-12-04 13:49:59 +00001087 Action.EndSourceFile();
1088
1089 return true;
1090}
1091
Ilya Biryukova907ba42018-05-14 10:50:04 +00001092// Should we allow index completions in the specified context?
1093bool allowIndex(CodeCompletionContext &CC) {
1094 if (!contextAllowsIndex(CC.getKind()))
1095 return false;
1096 // We also avoid ClassName::bar (but allow namespace::bar).
1097 auto Scope = CC.getCXXScopeSpecifier();
1098 if (!Scope)
1099 return true;
1100 NestedNameSpecifier *NameSpec = (*Scope)->getScopeRep();
1101 if (!NameSpec)
1102 return true;
1103 // We only query the index when qualifier is a namespace.
1104 // If it's a class, we rely solely on sema completions.
1105 switch (NameSpec->getKind()) {
1106 case NestedNameSpecifier::Global:
1107 case NestedNameSpecifier::Namespace:
1108 case NestedNameSpecifier::NamespaceAlias:
1109 return true;
1110 case NestedNameSpecifier::Super:
1111 case NestedNameSpecifier::TypeSpec:
1112 case NestedNameSpecifier::TypeSpecWithTemplate:
1113 // Unresolved inside a template.
1114 case NestedNameSpecifier::Identifier:
1115 return false;
1116 }
Ilya Biryukova6556e22018-05-14 11:47:30 +00001117 llvm_unreachable("invalid NestedNameSpecifier kind");
Ilya Biryukova907ba42018-05-14 10:50:04 +00001118}
1119
Eric Liu25d74e92018-08-24 11:23:56 +00001120std::future<SymbolSlab> startAsyncFuzzyFind(const SymbolIndex &Index,
1121 const FuzzyFindRequest &Req) {
1122 return runAsync<SymbolSlab>([&Index, Req]() {
1123 trace::Span Tracer("Async fuzzyFind");
1124 SymbolSlab::Builder Syms;
1125 Index.fuzzyFind(Req, [&Syms](const Symbol &Sym) { Syms.insert(Sym); });
1126 return std::move(Syms).build();
1127 });
1128}
1129
1130// Creates a `FuzzyFindRequest` based on the cached index request from the
1131// last completion, if any, and the speculated completion filter text in the
1132// source code.
Sam McCallc008af62018-10-20 15:30:37 +00001133Optional<FuzzyFindRequest> speculativeFuzzyFindRequestForCompletion(
Eric Liu25d74e92018-08-24 11:23:56 +00001134 FuzzyFindRequest CachedReq, PathRef File, StringRef Content, Position Pos) {
1135 auto Filter = speculateCompletionFilter(Content, Pos);
1136 if (!Filter) {
1137 elog("Failed to speculate filter text for code completion at Pos "
1138 "{0}:{1}: {2}",
1139 Pos.line, Pos.character, Filter.takeError());
Sam McCallc008af62018-10-20 15:30:37 +00001140 return None;
Eric Liu25d74e92018-08-24 11:23:56 +00001141 }
1142 CachedReq.Query = *Filter;
1143 return CachedReq;
1144}
1145
Sam McCall98775c52017-12-04 13:49:59 +00001146} // namespace
1147
1148clang::CodeCompleteOptions CodeCompleteOptions::getClangCompleteOpts() const {
1149 clang::CodeCompleteOptions Result;
1150 Result.IncludeCodePatterns = EnableSnippets && IncludeCodePatterns;
1151 Result.IncludeMacros = IncludeMacros;
Sam McCalld8169a82018-01-18 15:31:30 +00001152 Result.IncludeGlobals = true;
Ilya Biryukov43714502018-05-16 12:32:44 +00001153 // We choose to include full comments and not do doxygen parsing in
1154 // completion.
1155 // FIXME: ideally, we should support doxygen in some form, e.g. do markdown
1156 // formatting of the comments.
1157 Result.IncludeBriefComments = false;
Sam McCall98775c52017-12-04 13:49:59 +00001158
Sam McCall3d139c52018-01-12 18:30:08 +00001159 // When an is used, Sema is responsible for completing the main file,
1160 // the index can provide results from the preamble.
1161 // Tell Sema not to deserialize the preamble to look for results.
1162 Result.LoadExternal = !Index;
Kadir Cetinkaya2f84d912018-08-08 08:59:29 +00001163 Result.IncludeFixIts = IncludeFixIts;
Eric Liu6f648df2017-12-19 16:50:37 +00001164
Sam McCall98775c52017-12-04 13:49:59 +00001165 return Result;
1166}
1167
Eric Liu83f63e42018-09-03 10:18:21 +00001168// Returns the most popular include header for \p Sym. If two headers are
1169// equally popular, prefer the shorter one. Returns empty string if \p Sym has
1170// no include header.
Sam McCallc008af62018-10-20 15:30:37 +00001171SmallVector<StringRef, 1> getRankedIncludes(const Symbol &Sym) {
Eric Liu83f63e42018-09-03 10:18:21 +00001172 auto Includes = Sym.IncludeHeaders;
1173 // Sort in descending order by reference count and header length.
Kirill Bobyrev4a5ff882018-10-07 14:49:41 +00001174 llvm::sort(Includes, [](const Symbol::IncludeHeaderWithReferences &LHS,
1175 const Symbol::IncludeHeaderWithReferences &RHS) {
1176 if (LHS.References == RHS.References)
1177 return LHS.IncludeHeader.size() < RHS.IncludeHeader.size();
1178 return LHS.References > RHS.References;
1179 });
Sam McCallc008af62018-10-20 15:30:37 +00001180 SmallVector<StringRef, 1> Headers;
Eric Liu83f63e42018-09-03 10:18:21 +00001181 for (const auto &Include : Includes)
1182 Headers.push_back(Include.IncludeHeader);
1183 return Headers;
1184}
1185
Sam McCall545a20d2018-01-19 14:34:02 +00001186// Runs Sema-based (AST) and Index-based completion, returns merged results.
1187//
1188// There are a few tricky considerations:
1189// - the AST provides information needed for the index query (e.g. which
1190// namespaces to search in). So Sema must start first.
1191// - we only want to return the top results (Opts.Limit).
1192// Building CompletionItems for everything else is wasteful, so we want to
1193// preserve the "native" format until we're done with scoring.
1194// - the data underlying Sema completion items is owned by the AST and various
1195// other arenas, which must stay alive for us to build CompletionItems.
1196// - we may get duplicate results from Sema and the Index, we need to merge.
1197//
Ilya Biryukovddf6a332018-03-02 12:28:27 +00001198// So we start Sema completion first, and do all our work in its callback.
Sam McCall545a20d2018-01-19 14:34:02 +00001199// We use the Sema context information to query the index.
1200// Then we merge the two result sets, producing items that are Sema/Index/Both.
1201// These items are scored, and the top N are synthesized into the LSP response.
1202// Finally, we can clean up the data structures created by Sema completion.
1203//
1204// Main collaborators are:
1205// - semaCodeComplete sets up the compiler machinery to run code completion.
1206// - CompletionRecorder captures Sema completion results, including context.
1207// - SymbolIndex (Opts.Index) provides index completion results as Symbols
1208// - CompletionCandidates are the result of merging Sema and Index results.
1209// Each candidate points to an underlying CodeCompletionResult (Sema), a
1210// Symbol (Index), or both. It computes the result quality score.
1211// CompletionCandidate also does conversion to CompletionItem (at the end).
1212// - FuzzyMatcher scores how the candidate matches the partial identifier.
1213// This score is combined with the result quality score for the final score.
1214// - TopN determines the results with the best score.
1215class CodeCompleteFlow {
Eric Liuc5105f92018-02-16 14:15:55 +00001216 PathRef FileName;
Sam McCall3f0243f2018-07-03 08:09:29 +00001217 IncludeStructure Includes; // Complete once the compiler runs.
Eric Liu25d74e92018-08-24 11:23:56 +00001218 SpeculativeFuzzyFind *SpecFuzzyFind; // Can be nullptr.
Sam McCall545a20d2018-01-19 14:34:02 +00001219 const CodeCompleteOptions &Opts;
Eric Liu25d74e92018-08-24 11:23:56 +00001220
Sam McCall545a20d2018-01-19 14:34:02 +00001221 // Sema takes ownership of Recorder. Recorder is valid until Sema cleanup.
Ilya Biryukovddf6a332018-03-02 12:28:27 +00001222 CompletionRecorder *Recorder = nullptr;
Sam McCall545a20d2018-01-19 14:34:02 +00001223 int NSema = 0, NIndex = 0, NBoth = 0; // Counters for logging.
1224 bool Incomplete = false; // Would more be available with a higher limit?
Sam McCallc008af62018-10-20 15:30:37 +00001225 Optional<FuzzyMatcher> Filter; // Initialized once Sema runs.
Eric Liu670c1472018-09-27 18:46:00 +00001226 std::vector<std::string> QueryScopes; // Initialized once Sema runs.
Eric Liu3fac4ef2018-10-17 11:19:02 +00001227 // Initialized once QueryScopes is initialized, if there are scopes.
Sam McCallc008af62018-10-20 15:30:37 +00001228 Optional<ScopeDistance> ScopeProximity;
Ilya Biryukov647da3e2018-11-26 15:38:01 +00001229 llvm::Optional<OpaqueType> PreferredType; // Initialized once Sema runs.
Eric Liu670c1472018-09-27 18:46:00 +00001230 // Whether to query symbols from any scope. Initialized once Sema runs.
1231 bool AllScopes = false;
Sam McCall3f0243f2018-07-03 08:09:29 +00001232 // Include-insertion and proximity scoring rely on the include structure.
1233 // This is available after Sema has run.
Sam McCallc008af62018-10-20 15:30:37 +00001234 Optional<IncludeInserter> Inserter; // Available during runWithSema.
1235 Optional<URIDistance> FileProximity; // Initialized once Sema runs.
Eric Liu25d74e92018-08-24 11:23:56 +00001236 /// Speculative request based on the cached request and the filter text before
1237 /// the cursor.
1238 /// Initialized right before sema run. This is only set if `SpecFuzzyFind` is
1239 /// set and contains a cached request.
Sam McCallc008af62018-10-20 15:30:37 +00001240 Optional<FuzzyFindRequest> SpecReq;
Sam McCall545a20d2018-01-19 14:34:02 +00001241
1242public:
1243 // A CodeCompleteFlow object is only useful for calling run() exactly once.
Sam McCall3f0243f2018-07-03 08:09:29 +00001244 CodeCompleteFlow(PathRef FileName, const IncludeStructure &Includes,
Eric Liu25d74e92018-08-24 11:23:56 +00001245 SpeculativeFuzzyFind *SpecFuzzyFind,
Sam McCall3f0243f2018-07-03 08:09:29 +00001246 const CodeCompleteOptions &Opts)
Eric Liu25d74e92018-08-24 11:23:56 +00001247 : FileName(FileName), Includes(Includes), SpecFuzzyFind(SpecFuzzyFind),
1248 Opts(Opts) {}
Sam McCall545a20d2018-01-19 14:34:02 +00001249
Sam McCall27c979a2018-06-29 14:47:57 +00001250 CodeCompleteResult run(const SemaCompleteInput &SemaCCInput) && {
Sam McCalld1a7a372018-01-31 13:40:48 +00001251 trace::Span Tracer("CodeCompleteFlow");
Eric Liu25d74e92018-08-24 11:23:56 +00001252 if (Opts.Index && SpecFuzzyFind && SpecFuzzyFind->CachedReq.hasValue()) {
1253 assert(!SpecFuzzyFind->Result.valid());
1254 if ((SpecReq = speculativeFuzzyFindRequestForCompletion(
1255 *SpecFuzzyFind->CachedReq, SemaCCInput.FileName,
1256 SemaCCInput.Contents, SemaCCInput.Pos)))
1257 SpecFuzzyFind->Result = startAsyncFuzzyFind(*Opts.Index, *SpecReq);
1258 }
Eric Liu63f419a2018-05-15 15:29:32 +00001259
Sam McCall545a20d2018-01-19 14:34:02 +00001260 // We run Sema code completion first. It builds an AST and calculates:
Ilya Biryukovddf6a332018-03-02 12:28:27 +00001261 // - completion results based on the AST.
Sam McCall545a20d2018-01-19 14:34:02 +00001262 // - partial identifier and context. We need these for the index query.
Sam McCall27c979a2018-06-29 14:47:57 +00001263 CodeCompleteResult Output;
Ilya Biryukovddf6a332018-03-02 12:28:27 +00001264 auto RecorderOwner = llvm::make_unique<CompletionRecorder>(Opts, [&]() {
1265 assert(Recorder && "Recorder is not set");
Sam McCall3f0243f2018-07-03 08:09:29 +00001266 auto Style =
Eric Liu9338a882018-07-03 14:51:23 +00001267 format::getStyle(format::DefaultFormatStyle, SemaCCInput.FileName,
1268 format::DefaultFallbackStyle, SemaCCInput.Contents,
1269 SemaCCInput.VFS.get());
Sam McCall3f0243f2018-07-03 08:09:29 +00001270 if (!Style) {
Sam McCallbed58852018-07-11 10:35:11 +00001271 log("getStyle() failed for file {0}: {1}. Fallback is LLVM style.",
1272 SemaCCInput.FileName, Style.takeError());
Sam McCall3f0243f2018-07-03 08:09:29 +00001273 Style = format::getLLVMStyle();
1274 }
Eric Liu63f419a2018-05-15 15:29:32 +00001275 // If preprocessor was run, inclusions from preprocessor callback should
Sam McCall3f0243f2018-07-03 08:09:29 +00001276 // already be added to Includes.
1277 Inserter.emplace(
1278 SemaCCInput.FileName, SemaCCInput.Contents, *Style,
1279 SemaCCInput.Command.Directory,
1280 Recorder->CCSema->getPreprocessor().getHeaderSearchInfo());
1281 for (const auto &Inc : Includes.MainFileIncludes)
1282 Inserter->addExisting(Inc);
1283
1284 // Most of the cost of file proximity is in initializing the FileDistance
1285 // structures based on the observed includes, once per query. Conceptually
1286 // that happens here (though the per-URI-scheme initialization is lazy).
1287 // The per-result proximity scoring is (amortized) very cheap.
1288 FileDistanceOptions ProxOpts{}; // Use defaults.
1289 const auto &SM = Recorder->CCSema->getSourceManager();
Sam McCallc008af62018-10-20 15:30:37 +00001290 StringMap<SourceParams> ProxSources;
Sam McCall3f0243f2018-07-03 08:09:29 +00001291 for (auto &Entry : Includes.includeDepth(
1292 SM.getFileEntryForID(SM.getMainFileID())->getName())) {
1293 auto &Source = ProxSources[Entry.getKey()];
1294 Source.Cost = Entry.getValue() * ProxOpts.IncludeCost;
1295 // Symbols near our transitive includes are good, but only consider
1296 // things in the same directory or below it. Otherwise there can be
1297 // many false positives.
1298 if (Entry.getValue() > 0)
1299 Source.MaxUpTraversals = 1;
1300 }
1301 FileProximity.emplace(ProxSources, ProxOpts);
1302
Ilya Biryukovddf6a332018-03-02 12:28:27 +00001303 Output = runWithSema();
Sam McCall3f0243f2018-07-03 08:09:29 +00001304 Inserter.reset(); // Make sure this doesn't out-live Clang.
Ilya Biryukovddf6a332018-03-02 12:28:27 +00001305 SPAN_ATTACH(Tracer, "sema_completion_kind",
1306 getCompletionKindString(Recorder->CCContext.getKind()));
Ilya Biryukov647da3e2018-11-26 15:38:01 +00001307 log("Code complete: sema context {0}, query scopes [{1}] (AnyScope={2}), "
1308 "expected type {3}",
Eric Liubc25ef72018-07-05 08:29:33 +00001309 getCompletionKindString(Recorder->CCContext.getKind()),
Ilya Biryukov647da3e2018-11-26 15:38:01 +00001310 join(QueryScopes.begin(), QueryScopes.end(), ","), AllScopes,
1311 PreferredType ? Recorder->CCContext.getPreferredType().getAsString()
1312 : "<none>");
Ilya Biryukovddf6a332018-03-02 12:28:27 +00001313 });
1314
1315 Recorder = RecorderOwner.get();
Eric Liu25d74e92018-08-24 11:23:56 +00001316
Sam McCalld1a7a372018-01-31 13:40:48 +00001317 semaCodeComplete(std::move(RecorderOwner), Opts.getClangCompleteOpts(),
Eric Liu63f419a2018-05-15 15:29:32 +00001318 SemaCCInput, &Includes);
Sam McCall545a20d2018-01-19 14:34:02 +00001319
Sam McCall2b780162018-01-30 17:20:54 +00001320 SPAN_ATTACH(Tracer, "sema_results", NSema);
1321 SPAN_ATTACH(Tracer, "index_results", NIndex);
1322 SPAN_ATTACH(Tracer, "merged_results", NBoth);
Sam McCalld20d7982018-07-09 14:25:59 +00001323 SPAN_ATTACH(Tracer, "returned_results", int64_t(Output.Completions.size()));
Sam McCall27c979a2018-06-29 14:47:57 +00001324 SPAN_ATTACH(Tracer, "incomplete", Output.HasMore);
Sam McCallbed58852018-07-11 10:35:11 +00001325 log("Code complete: {0} results from Sema, {1} from Index, "
1326 "{2} matched, {3} returned{4}.",
1327 NSema, NIndex, NBoth, Output.Completions.size(),
1328 Output.HasMore ? " (incomplete)" : "");
Sam McCall27c979a2018-06-29 14:47:57 +00001329 assert(!Opts.Limit || Output.Completions.size() <= Opts.Limit);
Sam McCall545a20d2018-01-19 14:34:02 +00001330 // We don't assert that isIncomplete means we hit a limit.
1331 // Indexes may choose to impose their own limits even if we don't have one.
1332 return Output;
1333 }
1334
1335private:
1336 // This is called by run() once Sema code completion is done, but before the
1337 // Sema data structures are torn down. It does all the real work.
Sam McCall27c979a2018-06-29 14:47:57 +00001338 CodeCompleteResult runWithSema() {
Kadir Cetinkayaa9c9d002018-08-13 08:23:01 +00001339 const auto &CodeCompletionRange = CharSourceRange::getCharRange(
1340 Recorder->CCSema->getPreprocessor().getCodeCompletionTokenRange());
1341 Range TextEditRange;
1342 // When we are getting completions with an empty identifier, for example
1343 // std::vector<int> asdf;
1344 // asdf.^;
1345 // Then the range will be invalid and we will be doing insertion, use
1346 // current cursor position in such cases as range.
1347 if (CodeCompletionRange.isValid()) {
1348 TextEditRange = halfOpenToRange(Recorder->CCSema->getSourceManager(),
1349 CodeCompletionRange);
1350 } else {
1351 const auto &Pos = sourceLocToPosition(
1352 Recorder->CCSema->getSourceManager(),
1353 Recorder->CCSema->getPreprocessor().getCodeCompletionLoc());
1354 TextEditRange.start = TextEditRange.end = Pos;
1355 }
Sam McCall545a20d2018-01-19 14:34:02 +00001356 Filter = FuzzyMatcher(
Ilya Biryukovddf6a332018-03-02 12:28:27 +00001357 Recorder->CCSema->getPreprocessor().getCodeCompletionFilter());
Eric Liu3fac4ef2018-10-17 11:19:02 +00001358 std::tie(QueryScopes, AllScopes) =
1359 getQueryScopes(Recorder->CCContext, *Recorder->CCSema, Opts);
1360 if (!QueryScopes.empty())
1361 ScopeProximity.emplace(QueryScopes);
Ilya Biryukov647da3e2018-11-26 15:38:01 +00001362 PreferredType =
1363 OpaqueType::fromType(Recorder->CCSema->getASTContext(),
1364 Recorder->CCContext.getPreferredType());
Sam McCall545a20d2018-01-19 14:34:02 +00001365 // Sema provides the needed context to query the index.
1366 // FIXME: in addition to querying for extra/overlapping symbols, we should
1367 // explicitly request symbols corresponding to Sema results.
1368 // We can use their signals even if the index can't suggest them.
1369 // We must copy index results to preserve them, but there are at most Limit.
Eric Liu8f3678d2018-06-15 13:34:18 +00001370 auto IndexResults = (Opts.Index && allowIndex(Recorder->CCContext))
1371 ? queryIndex()
1372 : SymbolSlab();
Eric Liu25d74e92018-08-24 11:23:56 +00001373 trace::Span Tracer("Populate CodeCompleteResult");
Kadir Cetinkayab15b8dc2018-10-02 09:42:17 +00001374 // Merge Sema and Index results, score them, and pick the winners.
1375 auto Top = mergeResults(Recorder->Results, IndexResults);
Sam McCall27c979a2018-06-29 14:47:57 +00001376 CodeCompleteResult Output;
Kadir Cetinkayaf8b85a32018-08-23 13:14:50 +00001377
1378 // Convert the results to final form, assembling the expensive strings.
Sam McCall27c979a2018-06-29 14:47:57 +00001379 for (auto &C : Top) {
1380 Output.Completions.push_back(toCodeCompletion(C.first));
1381 Output.Completions.back().Score = C.second;
Kadir Cetinkayaa9c9d002018-08-13 08:23:01 +00001382 Output.Completions.back().CompletionTokenRange = TextEditRange;
Sam McCall27c979a2018-06-29 14:47:57 +00001383 }
1384 Output.HasMore = Incomplete;
Eric Liu5d2a8072018-07-23 10:56:37 +00001385 Output.Context = Recorder->CCContext.getKind();
Kadir Cetinkayaf8b85a32018-08-23 13:14:50 +00001386
Sam McCall545a20d2018-01-19 14:34:02 +00001387 return Output;
1388 }
1389
1390 SymbolSlab queryIndex() {
Sam McCalld1a7a372018-01-31 13:40:48 +00001391 trace::Span Tracer("Query index");
Sam McCalld20d7982018-07-09 14:25:59 +00001392 SPAN_ATTACH(Tracer, "limit", int64_t(Opts.Limit));
Sam McCall2b780162018-01-30 17:20:54 +00001393
Sam McCall545a20d2018-01-19 14:34:02 +00001394 // Build the query.
1395 FuzzyFindRequest Req;
Haojian Wu48b48652018-01-25 09:20:09 +00001396 if (Opts.Limit)
Kirill Bobyreve6dd0802018-09-13 14:27:03 +00001397 Req.Limit = Opts.Limit;
Sam McCall545a20d2018-01-19 14:34:02 +00001398 Req.Query = Filter->pattern();
Marc-Andre Laperle945b5a32018-06-05 14:01:40 +00001399 Req.RestrictForCodeCompletion = true;
Eric Liubc25ef72018-07-05 08:29:33 +00001400 Req.Scopes = QueryScopes;
Eric Liu670c1472018-09-27 18:46:00 +00001401 Req.AnyScope = AllScopes;
Sam McCall3f0243f2018-07-03 08:09:29 +00001402 // FIXME: we should send multiple weighted paths here.
Eric Liu6de95ec2018-06-12 08:48:20 +00001403 Req.ProximityPaths.push_back(FileName);
Kirill Bobyrev09f00dc2018-09-10 11:51:05 +00001404 vlog("Code complete: fuzzyFind({0:2})", toJSON(Req));
Eric Liu25d74e92018-08-24 11:23:56 +00001405
1406 if (SpecFuzzyFind)
1407 SpecFuzzyFind->NewReq = Req;
1408 if (SpecFuzzyFind && SpecFuzzyFind->Result.valid() && (*SpecReq == Req)) {
1409 vlog("Code complete: speculative fuzzy request matches the actual index "
1410 "request. Waiting for the speculative index results.");
1411 SPAN_ATTACH(Tracer, "Speculative results", true);
1412
1413 trace::Span WaitSpec("Wait speculative results");
1414 return SpecFuzzyFind->Result.get();
1415 }
1416
1417 SPAN_ATTACH(Tracer, "Speculative results", false);
1418
Sam McCall545a20d2018-01-19 14:34:02 +00001419 // Run the query against the index.
Eric Liu25d74e92018-08-24 11:23:56 +00001420 SymbolSlab::Builder ResultsBuilder;
Sam McCallab8e3932018-02-19 13:04:41 +00001421 if (Opts.Index->fuzzyFind(
1422 Req, [&](const Symbol &Sym) { ResultsBuilder.insert(Sym); }))
1423 Incomplete = true;
Sam McCall545a20d2018-01-19 14:34:02 +00001424 return std::move(ResultsBuilder).build();
1425 }
1426
Kadir Cetinkayab15b8dc2018-10-02 09:42:17 +00001427 // Merges Sema and Index results where possible, to form CompletionCandidates.
1428 // Groups overloads if desired, to form CompletionCandidate::Bundles. The
1429 // bundles are scored and top results are returned, best to worst.
Sam McCallc18c2802018-06-15 11:06:29 +00001430 std::vector<ScoredBundle>
Kadir Cetinkayaf8b85a32018-08-23 13:14:50 +00001431 mergeResults(const std::vector<CodeCompletionResult> &SemaResults,
Kadir Cetinkayab15b8dc2018-10-02 09:42:17 +00001432 const SymbolSlab &IndexResults) {
Sam McCalld1a7a372018-01-31 13:40:48 +00001433 trace::Span Tracer("Merge and score results");
Sam McCallc18c2802018-06-15 11:06:29 +00001434 std::vector<CompletionCandidate::Bundle> Bundles;
Sam McCallc008af62018-10-20 15:30:37 +00001435 DenseMap<size_t, size_t> BundleLookup;
Sam McCallc18c2802018-06-15 11:06:29 +00001436 auto AddToBundles = [&](const CodeCompletionResult *SemaResult,
Kadir Cetinkayab15b8dc2018-10-02 09:42:17 +00001437 const Symbol *IndexResult) {
Sam McCallc18c2802018-06-15 11:06:29 +00001438 CompletionCandidate C;
1439 C.SemaResult = SemaResult;
1440 C.IndexResult = IndexResult;
Eric Liu83f63e42018-09-03 10:18:21 +00001441 if (C.IndexResult)
1442 C.RankedIncludeHeaders = getRankedIncludes(*C.IndexResult);
Sam McCallc18c2802018-06-15 11:06:29 +00001443 C.Name = IndexResult ? IndexResult->Name : Recorder->getName(*SemaResult);
1444 if (auto OverloadSet = Opts.BundleOverloads ? C.overloadSet() : 0) {
1445 auto Ret = BundleLookup.try_emplace(OverloadSet, Bundles.size());
1446 if (Ret.second)
1447 Bundles.emplace_back();
1448 Bundles[Ret.first->second].push_back(std::move(C));
1449 } else {
1450 Bundles.emplace_back();
1451 Bundles.back().push_back(std::move(C));
1452 }
1453 };
Sam McCallc008af62018-10-20 15:30:37 +00001454 DenseSet<const Symbol *> UsedIndexResults;
Sam McCall545a20d2018-01-19 14:34:02 +00001455 auto CorrespondingIndexResult =
1456 [&](const CodeCompletionResult &SemaResult) -> const Symbol * {
Eric Liud25f1212018-09-06 09:59:37 +00001457 if (auto SymID =
1458 getSymbolID(SemaResult, Recorder->CCSema->getSourceManager())) {
Sam McCall545a20d2018-01-19 14:34:02 +00001459 auto I = IndexResults.find(*SymID);
1460 if (I != IndexResults.end()) {
1461 UsedIndexResults.insert(&*I);
1462 return &*I;
1463 }
1464 }
1465 return nullptr;
1466 };
1467 // Emit all Sema results, merging them with Index results if possible.
Ilya Biryukovddf6a332018-03-02 12:28:27 +00001468 for (auto &SemaResult : Recorder->Results)
Kadir Cetinkayab15b8dc2018-10-02 09:42:17 +00001469 AddToBundles(&SemaResult, CorrespondingIndexResult(SemaResult));
Sam McCall545a20d2018-01-19 14:34:02 +00001470 // Now emit any Index-only results.
1471 for (const auto &IndexResult : IndexResults) {
1472 if (UsedIndexResults.count(&IndexResult))
1473 continue;
Kadir Cetinkayab15b8dc2018-10-02 09:42:17 +00001474 AddToBundles(/*SemaResult=*/nullptr, &IndexResult);
Sam McCall545a20d2018-01-19 14:34:02 +00001475 }
Sam McCallc18c2802018-06-15 11:06:29 +00001476 // We only keep the best N results at any time, in "native" format.
1477 TopN<ScoredBundle, ScoredBundleGreater> Top(
1478 Opts.Limit == 0 ? std::numeric_limits<size_t>::max() : Opts.Limit);
1479 for (auto &Bundle : Bundles)
1480 addCandidate(Top, std::move(Bundle));
Sam McCall545a20d2018-01-19 14:34:02 +00001481 return std::move(Top).items();
1482 }
1483
Sam McCall80ad7072018-06-08 13:32:25 +00001484 Optional<float> fuzzyScore(const CompletionCandidate &C) {
1485 // Macros can be very spammy, so we only support prefix completion.
1486 // We won't end up with underfull index results, as macros are sema-only.
1487 if (C.SemaResult && C.SemaResult->Kind == CodeCompletionResult::RK_Macro &&
1488 !C.Name.startswith_lower(Filter->pattern()))
1489 return None;
1490 return Filter->match(C.Name);
1491 }
1492
Sam McCall545a20d2018-01-19 14:34:02 +00001493 // Scores a candidate and adds it to the TopN structure.
Sam McCallc18c2802018-06-15 11:06:29 +00001494 void addCandidate(TopN<ScoredBundle, ScoredBundleGreater> &Candidates,
1495 CompletionCandidate::Bundle Bundle) {
Sam McCallc5707b62018-05-15 17:43:27 +00001496 SymbolQualitySignals Quality;
1497 SymbolRelevanceSignals Relevance;
Eric Liu5d2a8072018-07-23 10:56:37 +00001498 Relevance.Context = Recorder->CCContext.getKind();
Sam McCalld9b54f02018-06-05 16:30:25 +00001499 Relevance.Query = SymbolRelevanceSignals::CodeComplete;
Sam McCallf84dd022018-07-05 08:26:53 +00001500 Relevance.FileProximityMatch = FileProximity.getPointer();
Eric Liu3fac4ef2018-10-17 11:19:02 +00001501 if (ScopeProximity)
1502 Relevance.ScopeProximityMatch = ScopeProximity.getPointer();
Ilya Biryukov647da3e2018-11-26 15:38:01 +00001503 if (PreferredType)
1504 Relevance.HadContextType = true;
Eric Liu670c1472018-09-27 18:46:00 +00001505
Sam McCallc18c2802018-06-15 11:06:29 +00001506 auto &First = Bundle.front();
1507 if (auto FuzzyScore = fuzzyScore(First))
Sam McCallc5707b62018-05-15 17:43:27 +00001508 Relevance.NameMatch = *FuzzyScore;
Sam McCall545a20d2018-01-19 14:34:02 +00001509 else
1510 return;
Sam McCall2161ec72018-07-05 06:20:41 +00001511 SymbolOrigin Origin = SymbolOrigin::Unknown;
Sam McCall4e5742a2018-07-06 11:50:49 +00001512 bool FromIndex = false;
Sam McCallc18c2802018-06-15 11:06:29 +00001513 for (const auto &Candidate : Bundle) {
1514 if (Candidate.IndexResult) {
1515 Quality.merge(*Candidate.IndexResult);
1516 Relevance.merge(*Candidate.IndexResult);
Sam McCall4e5742a2018-07-06 11:50:49 +00001517 Origin |= Candidate.IndexResult->Origin;
1518 FromIndex = true;
Ilya Biryukov647da3e2018-11-26 15:38:01 +00001519 if (!Candidate.IndexResult->Type.empty())
1520 Relevance.HadSymbolType |= true;
1521 if (PreferredType &&
1522 PreferredType->raw() == Candidate.IndexResult->Type) {
1523 Relevance.TypeMatchesPreferred = true;
1524 }
Sam McCallc18c2802018-06-15 11:06:29 +00001525 }
1526 if (Candidate.SemaResult) {
1527 Quality.merge(*Candidate.SemaResult);
1528 Relevance.merge(*Candidate.SemaResult);
Ilya Biryukov647da3e2018-11-26 15:38:01 +00001529 if (PreferredType) {
1530 if (auto CompletionType = OpaqueType::fromCompletionResult(
1531 Recorder->CCSema->getASTContext(), *Candidate.SemaResult)) {
1532 Relevance.HadSymbolType |= true;
1533 if (PreferredType == CompletionType)
1534 Relevance.TypeMatchesPreferred = true;
1535 }
1536 }
Sam McCall4e5742a2018-07-06 11:50:49 +00001537 Origin |= SymbolOrigin::AST;
Sam McCallc18c2802018-06-15 11:06:29 +00001538 }
Sam McCallc5707b62018-05-15 17:43:27 +00001539 }
1540
Sam McCall27c979a2018-06-29 14:47:57 +00001541 CodeCompletion::Scores Scores;
1542 Scores.Quality = Quality.evaluate();
1543 Scores.Relevance = Relevance.evaluate();
1544 Scores.Total = evaluateSymbolAndRelevance(Scores.Quality, Scores.Relevance);
1545 // NameMatch is in fact a multiplier on total score, so rescoring is sound.
1546 Scores.ExcludingName = Relevance.NameMatch
1547 ? Scores.Total / Relevance.NameMatch
1548 : Scores.Quality;
Sam McCallc5707b62018-05-15 17:43:27 +00001549
Sam McCallbed58852018-07-11 10:35:11 +00001550 dlog("CodeComplete: {0} ({1}) = {2}\n{3}{4}\n", First.Name,
Sam McCallc008af62018-10-20 15:30:37 +00001551 to_string(Origin), Scores.Total, to_string(Quality),
1552 to_string(Relevance));
Sam McCall545a20d2018-01-19 14:34:02 +00001553
Sam McCall2161ec72018-07-05 06:20:41 +00001554 NSema += bool(Origin & SymbolOrigin::AST);
Sam McCall4e5742a2018-07-06 11:50:49 +00001555 NIndex += FromIndex;
1556 NBoth += bool(Origin & SymbolOrigin::AST) && FromIndex;
Sam McCallc18c2802018-06-15 11:06:29 +00001557 if (Candidates.push({std::move(Bundle), Scores}))
Sam McCallab8e3932018-02-19 13:04:41 +00001558 Incomplete = true;
Sam McCall545a20d2018-01-19 14:34:02 +00001559 }
1560
Sam McCall27c979a2018-06-29 14:47:57 +00001561 CodeCompletion toCodeCompletion(const CompletionCandidate::Bundle &Bundle) {
Sam McCallc008af62018-10-20 15:30:37 +00001562 Optional<CodeCompletionBuilder> Builder;
Sam McCall27c979a2018-06-29 14:47:57 +00001563 for (const auto &Item : Bundle) {
1564 CodeCompletionString *SemaCCS =
1565 Item.SemaResult ? Recorder->codeCompletionString(*Item.SemaResult)
1566 : nullptr;
1567 if (!Builder)
1568 Builder.emplace(Recorder->CCSema->getASTContext(), Item, SemaCCS,
Eric Liu670c1472018-09-27 18:46:00 +00001569 QueryScopes, *Inserter, FileName,
1570 Recorder->CCContext.getKind(), Opts);
Sam McCall27c979a2018-06-29 14:47:57 +00001571 else
1572 Builder->add(Item, SemaCCS);
Ilya Biryukov43714502018-05-16 12:32:44 +00001573 }
Sam McCall27c979a2018-06-29 14:47:57 +00001574 return Builder->build();
Sam McCall545a20d2018-01-19 14:34:02 +00001575 }
1576};
1577
Sam McCallc008af62018-10-20 15:30:37 +00001578Expected<StringRef> speculateCompletionFilter(StringRef Content, Position Pos) {
Eric Liu25d74e92018-08-24 11:23:56 +00001579 auto Offset = positionToOffset(Content, Pos);
1580 if (!Offset)
Sam McCallc008af62018-10-20 15:30:37 +00001581 return make_error<StringError>(
Eric Liu25d74e92018-08-24 11:23:56 +00001582 "Failed to convert position to offset in content.",
Sam McCallc008af62018-10-20 15:30:37 +00001583 inconvertibleErrorCode());
Eric Liu25d74e92018-08-24 11:23:56 +00001584 if (*Offset == 0)
1585 return "";
1586
1587 // Start from the character before the cursor.
1588 int St = *Offset - 1;
1589 // FIXME(ioeric): consider UTF characters?
1590 auto IsValidIdentifierChar = [](char c) {
1591 return ((c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z') ||
1592 (c >= '0' && c <= '9') || (c == '_'));
1593 };
1594 size_t Len = 0;
1595 for (; (St >= 0) && IsValidIdentifierChar(Content[St]); --St, ++Len) {
1596 }
1597 if (Len > 0)
1598 St++; // Shift to the first valid character.
1599 return Content.substr(St, Len);
1600}
1601
1602CodeCompleteResult
1603codeComplete(PathRef FileName, const tooling::CompileCommand &Command,
Eric Liub1d75422018-10-02 10:43:55 +00001604 const PreambleData *Preamble, StringRef Contents, Position Pos,
Sam McCallc008af62018-10-20 15:30:37 +00001605 IntrusiveRefCntPtr<vfs::FileSystem> VFS,
Eric Liu25d74e92018-08-24 11:23:56 +00001606 std::shared_ptr<PCHContainerOperations> PCHs,
1607 CodeCompleteOptions Opts, SpeculativeFuzzyFind *SpecFuzzyFind) {
Eric Liub1d75422018-10-02 10:43:55 +00001608 return CodeCompleteFlow(FileName,
1609 Preamble ? Preamble->Includes : IncludeStructure(),
1610 SpecFuzzyFind, Opts)
Sam McCall3f0243f2018-07-03 08:09:29 +00001611 .run({FileName, Command, Preamble, Contents, Pos, VFS, PCHs});
Sam McCall98775c52017-12-04 13:49:59 +00001612}
1613
Sam McCalld1a7a372018-01-31 13:40:48 +00001614SignatureHelp signatureHelp(PathRef FileName,
Ilya Biryukov940901e2017-12-13 12:51:22 +00001615 const tooling::CompileCommand &Command,
Eric Liub1d75422018-10-02 10:43:55 +00001616 const PreambleData *Preamble, StringRef Contents,
1617 Position Pos,
Sam McCallc008af62018-10-20 15:30:37 +00001618 IntrusiveRefCntPtr<vfs::FileSystem> VFS,
Ilya Biryukov8a0f76b2018-08-17 09:32:30 +00001619 std::shared_ptr<PCHContainerOperations> PCHs,
Sam McCall046557b2018-09-03 16:37:59 +00001620 const SymbolIndex *Index) {
Sam McCall98775c52017-12-04 13:49:59 +00001621 SignatureHelp Result;
1622 clang::CodeCompleteOptions Options;
1623 Options.IncludeGlobals = false;
1624 Options.IncludeMacros = false;
1625 Options.IncludeCodePatterns = false;
Ilya Biryukov43714502018-05-16 12:32:44 +00001626 Options.IncludeBriefComments = false;
Sam McCall3f0243f2018-07-03 08:09:29 +00001627 IncludeStructure PreambleInclusions; // Unused for signatureHelp
Ilya Biryukov8a0f76b2018-08-17 09:32:30 +00001628 semaCodeComplete(
1629 llvm::make_unique<SignatureHelpCollector>(Options, Index, Result),
1630 Options,
1631 {FileName, Command, Preamble, Contents, Pos, std::move(VFS),
1632 std::move(PCHs)});
Sam McCall98775c52017-12-04 13:49:59 +00001633 return Result;
1634}
1635
Marc-Andre Laperle945b5a32018-06-05 14:01:40 +00001636bool isIndexedForCodeCompletion(const NamedDecl &ND, ASTContext &ASTCtx) {
1637 using namespace clang::ast_matchers;
1638 auto InTopLevelScope = hasDeclContext(
1639 anyOf(namespaceDecl(), translationUnitDecl(), linkageSpecDecl()));
1640 return !match(decl(anyOf(InTopLevelScope,
1641 hasDeclContext(
1642 enumDecl(InTopLevelScope, unless(isScoped()))))),
1643 ND, ASTCtx)
1644 .empty();
1645}
1646
Sam McCall27c979a2018-06-29 14:47:57 +00001647CompletionItem CodeCompletion::render(const CodeCompleteOptions &Opts) const {
1648 CompletionItem LSP;
Eric Liu83f63e42018-09-03 10:18:21 +00001649 const auto *InsertInclude = Includes.empty() ? nullptr : &Includes[0];
1650 LSP.label = ((InsertInclude && InsertInclude->Insertion)
1651 ? Opts.IncludeIndicator.Insert
1652 : Opts.IncludeIndicator.NoInsert) +
Sam McCallc008af62018-10-20 15:30:37 +00001653 (Opts.ShowOrigins ? "[" + to_string(Origin) + "]" : "") +
Sam McCall27c979a2018-06-29 14:47:57 +00001654 RequiredQualifier + Name + Signature;
Sam McCall2161ec72018-07-05 06:20:41 +00001655
Sam McCall27c979a2018-06-29 14:47:57 +00001656 LSP.kind = Kind;
Sam McCallc008af62018-10-20 15:30:37 +00001657 LSP.detail =
1658 BundleSize > 1 ? formatv("[{0} overloads]", BundleSize) : ReturnType;
Eric Liu6df66002018-09-06 18:52:26 +00001659 LSP.deprecated = Deprecated;
Eric Liu83f63e42018-09-03 10:18:21 +00001660 if (InsertInclude)
1661 LSP.detail += "\n" + InsertInclude->Header;
Sam McCall27c979a2018-06-29 14:47:57 +00001662 LSP.documentation = Documentation;
1663 LSP.sortText = sortText(Score.Total, Name);
1664 LSP.filterText = Name;
Kadir Cetinkayaa9c9d002018-08-13 08:23:01 +00001665 LSP.textEdit = {CompletionTokenRange, RequiredQualifier + Name};
Fangrui Song445bdd12018-09-05 08:01:37 +00001666 // Merge continuous additionalTextEdits into main edit. The main motivation
Kadir Cetinkayaa9c9d002018-08-13 08:23:01 +00001667 // behind this is to help LSP clients, it seems most of them are confused when
1668 // they are provided with additionalTextEdits that are consecutive to main
1669 // edit.
1670 // Note that we store additional text edits from back to front in a line. That
1671 // is mainly to help LSP clients again, so that changes do not effect each
1672 // other.
1673 for (const auto &FixIt : FixIts) {
1674 if (IsRangeConsecutive(FixIt.range, LSP.textEdit->range)) {
1675 LSP.textEdit->newText = FixIt.newText + LSP.textEdit->newText;
1676 LSP.textEdit->range.start = FixIt.range.start;
1677 } else {
1678 LSP.additionalTextEdits.push_back(FixIt);
1679 }
1680 }
Kadir Cetinkaya516fcda2018-08-23 12:19:39 +00001681 if (Opts.EnableSnippets)
1682 LSP.textEdit->newText += SnippetSuffix;
Kadir Cetinkaya6c9f15c2018-08-17 15:42:54 +00001683
Kadir Cetinkayaa9c9d002018-08-13 08:23:01 +00001684 // FIXME(kadircet): Do not even fill insertText after making sure textEdit is
1685 // compatible with most of the editors.
1686 LSP.insertText = LSP.textEdit->newText;
Sam McCall27c979a2018-06-29 14:47:57 +00001687 LSP.insertTextFormat = Opts.EnableSnippets ? InsertTextFormat::Snippet
1688 : InsertTextFormat::PlainText;
Eric Liu83f63e42018-09-03 10:18:21 +00001689 if (InsertInclude && InsertInclude->Insertion)
1690 LSP.additionalTextEdits.push_back(*InsertInclude->Insertion);
Eric Liu6df66002018-09-06 18:52:26 +00001691
Sam McCall27c979a2018-06-29 14:47:57 +00001692 return LSP;
1693}
1694
Sam McCalle746a2b2018-07-02 11:13:16 +00001695raw_ostream &operator<<(raw_ostream &OS, const CodeCompletion &C) {
1696 // For now just lean on CompletionItem.
1697 return OS << C.render(CodeCompleteOptions());
1698}
1699
1700raw_ostream &operator<<(raw_ostream &OS, const CodeCompleteResult &R) {
1701 OS << "CodeCompleteResult: " << R.Completions.size() << (R.HasMore ? "+" : "")
Eric Liu5d2a8072018-07-23 10:56:37 +00001702 << " (" << getCompletionKindString(R.Context) << ")"
Sam McCalle746a2b2018-07-02 11:13:16 +00001703 << " items:\n";
1704 for (const auto &C : R.Completions)
1705 OS << C << "\n";
1706 return OS;
1707}
1708
Sam McCall98775c52017-12-04 13:49:59 +00001709} // namespace clangd
1710} // namespace clang