-
Notifications
You must be signed in to change notification settings - Fork 13.2k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
[clangd] Support outgoing calls in call hierarchy #77556
Changes from all commits
f3d2263
3556ead
9a39394
4b4bbbc
bad4f72
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change | ||||
---|---|---|---|---|---|---|
|
@@ -1701,6 +1701,7 @@ declToHierarchyItem(const NamedDecl &ND, llvm::StringRef TUPath) { | |||||
|
||||||
HierarchyItem HI; | ||||||
HI.name = printName(Ctx, ND); | ||||||
// FIXME: Populate HI.detail the way we do in symbolToHierarchyItem? | ||||||
HI.kind = SK; | ||||||
HI.range = Range{sourceLocToPosition(SM, DeclRange->getBegin()), | ||||||
sourceLocToPosition(SM, DeclRange->getEnd())}; | ||||||
|
@@ -1752,6 +1753,7 @@ static std::optional<HierarchyItem> symbolToHierarchyItem(const Symbol &S, | |||||
} | ||||||
HierarchyItem HI; | ||||||
HI.name = std::string(S.Name); | ||||||
HI.detail = (S.Scope + S.Name).str(); | ||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. adding signature looks a bit more helpful, maybe one day add return type info too
Suggested change
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I agree, showing the signature here would be quite useful. I actually started implementing this for incoming calls in clangd/clangd#915, but ran into a complication (our current indexing process doesn't actually store the signature for all functions). This is solvable with some indexer changes, but I'll defer to a fix for that issue, which will now (once this patch is merged) benefit both incoming and outgoing calls. |
||||||
HI.kind = indexSymbolKindToSymbolKind(S.SymInfo.Kind); | ||||||
HI.selectionRange = Loc->range; | ||||||
// FIXME: Populate 'range' correctly | ||||||
|
@@ -2304,6 +2306,63 @@ incomingCalls(const CallHierarchyItem &Item, const SymbolIndex *Index) { | |||||
return Results; | ||||||
} | ||||||
|
||||||
std::vector<CallHierarchyOutgoingCall> | ||||||
outgoingCalls(const CallHierarchyItem &Item, const SymbolIndex *Index) { | ||||||
std::vector<CallHierarchyOutgoingCall> Results; | ||||||
if (!Index || Item.data.empty()) | ||||||
return Results; | ||||||
auto ID = SymbolID::fromStr(Item.data); | ||||||
if (!ID) { | ||||||
elog("outgoingCalls failed to find symbol: {0}", ID.takeError()); | ||||||
return Results; | ||||||
} | ||||||
// In this function, we find outgoing calls based on the index only. | ||||||
ContainedRefsRequest Request; | ||||||
Request.ID = *ID; | ||||||
// Initially store the ranges in a map keyed by SymbolID of the callee. | ||||||
// This allows us to group different calls to the same function | ||||||
// into the same CallHierarchyOutgoingCall. | ||||||
llvm::DenseMap<SymbolID, std::vector<Range>> CallsOut; | ||||||
// We can populate the ranges based on a refs request only. As we do so, we | ||||||
// also accumulate the callee IDs into a lookup request. | ||||||
LookupRequest CallsOutLookup; | ||||||
Index->containedRefs(Request, [&](const auto &R) { | ||||||
auto Loc = indexToLSPLocation(R.Location, Item.uri.file()); | ||||||
if (!Loc) { | ||||||
elog("outgoingCalls failed to convert location: {0}", Loc.takeError()); | ||||||
return; | ||||||
} | ||||||
auto It = CallsOut.try_emplace(R.Symbol, std::vector<Range>{}).first; | ||||||
It->second.push_back(Loc->range); | ||||||
|
||||||
CallsOutLookup.IDs.insert(R.Symbol); | ||||||
}); | ||||||
// Perform the lookup request and combine its results with CallsOut to | ||||||
// get complete CallHierarchyOutgoingCall objects. | ||||||
Index->lookup(CallsOutLookup, [&](const Symbol &Callee) { | ||||||
// The containedRefs request should only return symbols which are | ||||||
// function-like, i.e. symbols for which references to them can be "calls". | ||||||
using SK = index::SymbolKind; | ||||||
auto Kind = Callee.SymInfo.Kind; | ||||||
assert(Kind == SK::Function || Kind == SK::InstanceMethod || | ||||||
Kind == SK::ClassMethod || Kind == SK::StaticMethod || | ||||||
Kind == SK::Constructor || Kind == SK::Destructor || | ||||||
Kind == SK::ConversionFunction); | ||||||
|
||||||
auto It = CallsOut.find(Callee.ID); | ||||||
assert(It != CallsOut.end()); | ||||||
if (auto CHI = symbolToCallHierarchyItem(Callee, Item.uri.file())) | ||||||
Results.push_back( | ||||||
CallHierarchyOutgoingCall{std::move(*CHI), std::move(It->second)}); | ||||||
}); | ||||||
// Sort results by name of the callee. | ||||||
llvm::sort(Results, [](const CallHierarchyOutgoingCall &A, | ||||||
const CallHierarchyOutgoingCall &B) { | ||||||
return A.to.name < B.to.name; | ||||||
}); | ||||||
return Results; | ||||||
} | ||||||
|
||||||
llvm::DenseSet<const Decl *> getNonLocalDeclRefs(ParsedAST &AST, | ||||||
const FunctionDecl *FD) { | ||||||
if (!FD->hasBody()) | ||||||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
is this the place which covers c++ constructors?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The difference between
declToHierarchyItem
andsymbolToHierarchyItem
is the former is used for symbols from the AST, and the latter for symbols from the index. They should both ideally produce the same result, but the implementation (and sometimes the amount of detail available in the respective inputs) is different.There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I see. Ideally there should be some recommended use model for
c++
which call hierarchy clients/visualisers (e.g. lsp4e) could implement to show expected function signatures. Like "obtain details field from call hierarchy item and use it as function signature" and then lsp4e could append details to name. We probably can later tune xrefs.cpp implementation to provide required details where needed.Note that function return value is still not available, and as far as I can see there is no place in protocol to put it in. I would envision language-specific details map for that.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
See also clangd/clangd#1940 for some additional discussion of this topic.