- For qualified completion (foo::a^)
- unresolved qualifier - use global namespace ("::")
- resolved qualifier - use all accessible namespaces inside the resolved qualifier.
- For unqualified completion (vec^), use scopes that are accessible from the scope from which code completion occurs.
Details
- Reviewers
sammccall ilya-biryukov - Commits
- rG061c73eb288d: [clangd] Use accessible scopes to query indexes for global code completion.
rCTE323189: [clangd] Use accessible scopes to query indexes for global code completion.
rL323189: [clangd] Use accessible scopes to query indexes for global code completion.
Diff Detail
- Repository
- rL LLVM
Event Timeline
Discussed this offline.
If we move most to collect visited DeclContexts to CodeCompletionDeclConsumer (from CodeComplete.cpp) and provide a list of DeclContexts to CodeCompletionConsumer, we will be able to get rid of running lookup manually in clangd.
Sounds like implementation is in flux, but a few comments that might be relevant.
clangd/CodeComplete.cpp | ||
---|---|---|
264 ↗ | (On Diff #129862) | this could also be just a using, rather than a struct - do you see more members here? |
270 ↗ | (On Diff #129862) | Just to check, if the user types: is this right? |
300 ↗ | (On Diff #129862) | Probably not for this patch, but I just realized... Currently if the completion is not qualified, we don't track what namespaces are in scope. So I'd suggest either here or in queryscopes: // TODO: for unqualified completions, capture scopes and use for scoring. |
658 ↗ | (On Diff #129862) | you need to set loadexternal to false to avoid deserializing the whole preamble. |
clangd/CodeComplete.cpp | ||
---|---|---|
270 ↗ | (On Diff #129862) | You're probably right, just want to be sure we're talking about the same thing. There's two layers here: the context detected from sema, and what we're going to send the index. The layering is more relevant now that more of this moves out of clangd. for "vec", we should be sending {""} to the index for now, and later move towards doing global completion. That said, per our discussion this morning, the detected state shouldn't really be Optional<vector<string>>, but rather struct { vector<string> AccessibleScope, optional<string> UnresolvedQualifier } or something like that... |
clangd/CodeComplete.cpp | ||
---|---|---|
270 ↗ | (On Diff #129862) |
|
- Rebase the patch to latest
- Update the patch based on the offline discussion
Updating D42073: [clangd] Query all visible scopes based on all visible using-namespace declarations
and containing namespace for global qualified code completion.
clangd/CodeComplete.cpp | ||
---|---|---|
270 ↗ | (On Diff #129862) | @ilya-biryukov your are right. Based on our discussion last morning, the accessible scopes: // qualified completion // "::vec" => {"::"} // "using namespace std; ::vec^" => {"::", "std"} // "namespace ns {using namespace std;} ns::^" => {"ns", "std"} // "std::vec^" => {"::"} // unresolved "std::" // // unqualified completion // "vec^" => {"::"} // "using namespace std; vec^" => {"::", "std"} // "using namespace std; namespace ns { vec^ }" => {"ns", "std", "::"} With rL322945, now it is sensible to include all (qualified/unqualified completion) in this patch. |
658 ↗ | (On Diff #129862) | not needed now as we don't do lookup manually in clangd. |
I think this mostly does the right thing, but I find some of the code structure/description confusing.
I might have made misguided comments, happy to sit down together and work this out :-)
clangd/CodeComplete.cpp | ||
---|---|---|
316 ↗ | (On Diff #130618) | This comment is a bit confusing. |
316 ↗ | (On Diff #130618) | nit: avoid \brief where possible - the first sentence is used by default. |
319 ↗ | (On Diff #130618) | it's not totally clear to me whether you're talking about *our* sema-based results, or the sema infrastructure we're using, or it's a mistake and you mean our index code completion. Maybe just: // FIXME: When we resolve part of a scope chain (e.g. "known::unknown::ident"), we should // expand the known part rather than treating the whole thing as unknown. I think this should go in the implementation, rather than the type comment. |
327–328 ↗ | (On Diff #130618) | Hmm, you're renamed this from SpecifiedScope to QualifiedScope. I don't understand this name, can you explain what it means? |
329 ↗ | (On Diff #130618) | This is a bit hard to parse, and doesn't seem to describe the "unresolved by sema" case in a meaningful way. What about: The scopes we should look in, determined by Sema. If the qualifier was fully resolved, we should look for completions in these scopes. If there is an unresolved part of the qualifier, it should be resolved within these scopes. That way we describe what we aim to do (which is pretty simple), and we can document deviations with FIXMEs. |
334 ↗ | (On Diff #130618) | for this case: why not "the containing namespace and all accessible namespaces"? You've implemented that below, and it seems like the right thing to do here. |
340 ↗ | (On Diff #130618) | you seem to have a comment-in-a-comment here. I'd like to see an example `"namespace ns { vec^ }" => {"ns", ""} |
342 ↗ | (On Diff #130618) | This is ambiguous, does it mean The suffix of the user-writter qualifiers that Sema didn't resolve or The full scope qualifier as typed by the user? |
346 ↗ | (On Diff #130618) | this needs a new name. Previously it described one scope, and was being formatted to match the index representation. |
348 ↗ | (On Diff #130618) | what is "VS"? |
350 ↗ | (On Diff #130618) | It seems like you have stored the scopes in an unknown format, and call "normalizeScope" defensively? Seems cleaner if you ensure both the scopes and unknown are in the form:
Then this code can produce similarly-formatted output with just: Results.push_back(VS); if (UnresolvedQualifier) Results.back() += *UnresolvedQualifier; We need to trim leading :: before sending to the index, but I think that's because we got the index API wrong. I'll send a patch to fix it. |
845 ↗ | (On Diff #130618) | This seems like too much fiddly logic inlined here. Can't getQualifiedScopes(CodeCompletionContext, Sema) -> QualifiedScopes handle both cases here? |
849 ↗ | (On Diff #130618) | Please use a QualifiedScopes object to handle this case too. |
863 ↗ | (On Diff #130618) | the global scope is spelled "" in both conventions we use (the current one, and the one we should be using instead :-)). |
868 ↗ | (On Diff #130618) | log doesn't work that way - you need to combine into a single message :-) I think since we don't have verbosity levels, it's best to remove this - it's much more fine-grained than anything else we log, so it will be spammy. |
clangd/CodeComplete.cpp | ||
---|---|---|
664 ↗ | (On Diff #130618) | NIT: s/dureing/during |
Thanks for the comments!
clangd/CodeComplete.cpp | ||
---|---|---|
319 ↗ | (On Diff #130618) | I was talking about the sema infrastructure . |
327–328 ↗ | (On Diff #130618) | My original thought was that this structure is used only for qualified completion (not unqualified completion). |
334 ↗ | (On Diff #130618) | Currently Sema doesn't support it, (that being said the visited contexts for this case is empty). |
340 ↗ | (On Diff #130618) | This was focusing on qualified completions only, the unqualified sample is provided separately (below). Have made it together. |
342 ↗ | (On Diff #130618) | unfortunately, it is the second one. |
348 ↗ | (On Diff #130618) | means visible scope, renamed it to AS. |
350 ↗ | (On Diff #130618) | Thanks. Done. |
845 ↗ | (On Diff #130618) | Yes, we can, that would simplify the code. |
868 ↗ | (On Diff #130618) | I think the log here is pretty useful for debugging. |
clangd/CodeComplete.cpp | ||
---|---|---|
652 ↗ | (On Diff #130883) | could you move this to be right after the closely-related SpecifiedScope struct? I can't remember why they got separated, probably my fault :-) |
653 ↗ | (On Diff #130883) | as Ilya pointed out to me, passing Sema around is a big scary thing that we should try to avoid. Here it's only used to get the text if the CXXScopeSpec is invalid. Can we pass just the SourceManager, or better yet, the text as a StringPiece? |
695 ↗ | (On Diff #130883) | do you need to remove any leading :: here? |
868 ↗ | (On Diff #130618) | SG. It might be worth more clearly tying the log message to the other code complete one:, e.g. "Code complete: fuzzyFind(\"{0}\", Scopes=[{1}])" |
unittests/clangd/CodeCompleteTests.cpp | ||
679 ↗ | (On Diff #130883) | Can you factor out the common code into a function? e.g. vector<FuzzyFindRequest> captureIndexRequests(string example) { IndexRequestCollector Index; // actually the class can be local here ... } Like completions(), this avoids boilerplate in the tests. |
remove the leading "::".
clangd/CodeComplete.cpp | ||
---|---|---|
695 ↗ | (On Diff #130883) | Aha, sorry -- I misread the comment. Yeah, we need to remove the leading "::", added a test to catch this bug. |