mirror of
https://github.com/capstone-engine/llvm-capstone.git
synced 2025-01-11 02:16:50 +00:00
[clangd] Split locateSymbolAt into several component functions, to allow later reuse. NFC
This commit is contained in:
parent
91cdbd521a
commit
e7de00cf97
@ -172,85 +172,53 @@ llvm::Optional<Location> makeLocation(ASTContext &AST, SourceLocation TokLoc,
|
|||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
std::vector<DocumentLink> getDocumentLinks(ParsedAST &AST) {
|
// Treat #included files as symbols, to enable go-to-definition on them.
|
||||||
const auto &SM = AST.getSourceManager();
|
static llvm::Optional<LocatedSymbol>
|
||||||
auto MainFilePath =
|
locateFileReferent(const Position &Pos, ParsedAST &AST,
|
||||||
getCanonicalPath(SM.getFileEntryForID(SM.getMainFileID()), SM);
|
llvm::StringRef MainFilePath) {
|
||||||
if (!MainFilePath) {
|
|
||||||
elog("Failed to get a path for the main file, so no links");
|
|
||||||
return {};
|
|
||||||
}
|
|
||||||
|
|
||||||
std::vector<DocumentLink> Result;
|
|
||||||
for (auto &Inc : AST.getIncludeStructure().MainFileIncludes) {
|
|
||||||
if (!Inc.Resolved.empty()) {
|
|
||||||
Result.push_back(DocumentLink(
|
|
||||||
{Inc.R, URIForFile::canonicalize(Inc.Resolved, *MainFilePath)}));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return Result;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::vector<LocatedSymbol> locateSymbolAt(ParsedAST &AST, Position Pos,
|
|
||||||
const SymbolIndex *Index) {
|
|
||||||
const auto &SM = AST.getSourceManager();
|
|
||||||
auto MainFilePath =
|
|
||||||
getCanonicalPath(SM.getFileEntryForID(SM.getMainFileID()), SM);
|
|
||||||
if (!MainFilePath) {
|
|
||||||
elog("Failed to get a path for the main file, so no references");
|
|
||||||
return {};
|
|
||||||
}
|
|
||||||
|
|
||||||
// Treat #included files as symbols, to enable go-to-definition on them.
|
|
||||||
for (auto &Inc : AST.getIncludeStructure().MainFileIncludes) {
|
for (auto &Inc : AST.getIncludeStructure().MainFileIncludes) {
|
||||||
if (!Inc.Resolved.empty() && Inc.R.start.line == Pos.line) {
|
if (!Inc.Resolved.empty() && Inc.R.start.line == Pos.line) {
|
||||||
LocatedSymbol File;
|
LocatedSymbol File;
|
||||||
File.Name = std::string(llvm::sys::path::filename(Inc.Resolved));
|
File.Name = std::string(llvm::sys::path::filename(Inc.Resolved));
|
||||||
File.PreferredDeclaration = {
|
File.PreferredDeclaration = {
|
||||||
URIForFile::canonicalize(Inc.Resolved, *MainFilePath), Range{}};
|
URIForFile::canonicalize(Inc.Resolved, MainFilePath), Range{}};
|
||||||
File.Definition = File.PreferredDeclaration;
|
File.Definition = File.PreferredDeclaration;
|
||||||
// We're not going to find any further symbols on #include lines.
|
// We're not going to find any further symbols on #include lines.
|
||||||
return {std::move(File)};
|
return File;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
return llvm::None;
|
||||||
|
}
|
||||||
|
|
||||||
auto CurLoc = sourceLocationInMainFile(SM, Pos);
|
// Macros are simple: there's no declaration/definition distinction.
|
||||||
if (!CurLoc) {
|
// As a consequence, there's no need to look them up in the index either.
|
||||||
elog("locateSymbolAt failed to convert position to source location: {0}",
|
static llvm::Optional<LocatedSymbol>
|
||||||
CurLoc.takeError());
|
locateMacroReferent(const syntax::Token &TouchedIdentifier, ParsedAST &AST,
|
||||||
return {};
|
llvm::StringRef MainFilePath) {
|
||||||
}
|
if (auto M = locateMacroAt(TouchedIdentifier, AST.getPreprocessor())) {
|
||||||
|
if (auto Loc = makeLocation(AST.getASTContext(),
|
||||||
// Macros are simple: there's no declaration/definition distinction.
|
M->Info->getDefinitionLoc(), MainFilePath)) {
|
||||||
// As a consequence, there's no need to look them up in the index either.
|
LocatedSymbol Macro;
|
||||||
std::vector<LocatedSymbol> Result;
|
Macro.Name = std::string(M->Name);
|
||||||
const auto *TouchedIdentifier =
|
Macro.PreferredDeclaration = *Loc;
|
||||||
syntax::spelledIdentifierTouching(*CurLoc, AST.getTokens());
|
Macro.Definition = Loc;
|
||||||
if (TouchedIdentifier) {
|
return Macro;
|
||||||
if (auto M = locateMacroAt(*TouchedIdentifier, AST.getPreprocessor())) {
|
|
||||||
if (auto Loc = makeLocation(AST.getASTContext(),
|
|
||||||
M->Info->getDefinitionLoc(), *MainFilePath)) {
|
|
||||||
LocatedSymbol Macro;
|
|
||||||
Macro.Name = std::string(M->Name);
|
|
||||||
Macro.PreferredDeclaration = *Loc;
|
|
||||||
Macro.Definition = Loc;
|
|
||||||
Result.push_back(std::move(Macro));
|
|
||||||
|
|
||||||
// Don't look at the AST or index if we have a macro result.
|
|
||||||
// (We'd just return declarations referenced from the macro's
|
|
||||||
// expansion.)
|
|
||||||
return Result;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
return llvm::None;
|
||||||
|
}
|
||||||
|
|
||||||
// Decls are more complicated.
|
// Decls are more complicated.
|
||||||
// The AST contains at least a declaration, maybe a definition.
|
// The AST contains at least a declaration, maybe a definition.
|
||||||
// These are up-to-date, and so generally preferred over index results.
|
// These are up-to-date, and so generally preferred over index results.
|
||||||
// We perform a single batch index lookup to find additional definitions.
|
// We perform a single batch index lookup to find additional definitions.
|
||||||
|
static std::vector<LocatedSymbol>
|
||||||
|
locateASTReferent(SourceLocation CurLoc, const syntax::Token *TouchedIdentifier,
|
||||||
|
ParsedAST &AST, llvm::StringRef MainFilePath,
|
||||||
|
const SymbolIndex *Index) {
|
||||||
|
const SourceManager &SM = AST.getSourceManager();
|
||||||
// Results follow the order of Symbols.Decls.
|
// Results follow the order of Symbols.Decls.
|
||||||
|
std::vector<LocatedSymbol> Result;
|
||||||
// Keep track of SymbolID -> index mapping, to fill in index data later.
|
// Keep track of SymbolID -> index mapping, to fill in index data later.
|
||||||
llvm::DenseMap<SymbolID, size_t> ResultIndex;
|
llvm::DenseMap<SymbolID, size_t> ResultIndex;
|
||||||
|
|
||||||
@ -259,7 +227,7 @@ std::vector<LocatedSymbol> locateSymbolAt(ParsedAST &AST, Position Pos,
|
|||||||
const NamedDecl *Preferred = Def ? Def : D;
|
const NamedDecl *Preferred = Def ? Def : D;
|
||||||
|
|
||||||
auto Loc = makeLocation(AST.getASTContext(), nameLocation(*Preferred, SM),
|
auto Loc = makeLocation(AST.getASTContext(), nameLocation(*Preferred, SM),
|
||||||
*MainFilePath);
|
MainFilePath);
|
||||||
if (!Loc)
|
if (!Loc)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
@ -278,7 +246,7 @@ std::vector<LocatedSymbol> locateSymbolAt(ParsedAST &AST, Position Pos,
|
|||||||
// Emit all symbol locations (declaration or definition) from AST.
|
// Emit all symbol locations (declaration or definition) from AST.
|
||||||
DeclRelationSet Relations =
|
DeclRelationSet Relations =
|
||||||
DeclRelation::TemplatePattern | DeclRelation::Alias;
|
DeclRelation::TemplatePattern | DeclRelation::Alias;
|
||||||
for (const NamedDecl *D : getDeclAtPosition(AST, *CurLoc, Relations)) {
|
for (const NamedDecl *D : getDeclAtPosition(AST, CurLoc, Relations)) {
|
||||||
// Special case: void foo() ^override: jump to the overridden method.
|
// Special case: void foo() ^override: jump to the overridden method.
|
||||||
if (const auto *CMD = llvm::dyn_cast<CXXMethodDecl>(D)) {
|
if (const auto *CMD = llvm::dyn_cast<CXXMethodDecl>(D)) {
|
||||||
const InheritableAttr *Attr = D->getAttr<OverrideAttr>();
|
const InheritableAttr *Attr = D->getAttr<OverrideAttr>();
|
||||||
@ -320,23 +288,23 @@ std::vector<LocatedSymbol> locateSymbolAt(ParsedAST &AST, Position Pos,
|
|||||||
if (R.Definition) { // from AST
|
if (R.Definition) { // from AST
|
||||||
// Special case: if the AST yielded a definition, then it may not be
|
// Special case: if the AST yielded a definition, then it may not be
|
||||||
// the right *declaration*. Prefer the one from the index.
|
// the right *declaration*. Prefer the one from the index.
|
||||||
if (auto Loc = toLSPLocation(Sym.CanonicalDeclaration, *MainFilePath))
|
if (auto Loc = toLSPLocation(Sym.CanonicalDeclaration, MainFilePath))
|
||||||
R.PreferredDeclaration = *Loc;
|
R.PreferredDeclaration = *Loc;
|
||||||
|
|
||||||
// We might still prefer the definition from the index, e.g. for
|
// We might still prefer the definition from the index, e.g. for
|
||||||
// generated symbols.
|
// generated symbols.
|
||||||
if (auto Loc = toLSPLocation(
|
if (auto Loc = toLSPLocation(
|
||||||
getPreferredLocation(*R.Definition, Sym.Definition, Scratch),
|
getPreferredLocation(*R.Definition, Sym.Definition, Scratch),
|
||||||
*MainFilePath))
|
MainFilePath))
|
||||||
R.Definition = *Loc;
|
R.Definition = *Loc;
|
||||||
} else {
|
} else {
|
||||||
R.Definition = toLSPLocation(Sym.Definition, *MainFilePath);
|
R.Definition = toLSPLocation(Sym.Definition, MainFilePath);
|
||||||
|
|
||||||
// Use merge logic to choose AST or index declaration.
|
// Use merge logic to choose AST or index declaration.
|
||||||
if (auto Loc = toLSPLocation(
|
if (auto Loc = toLSPLocation(
|
||||||
getPreferredLocation(R.PreferredDeclaration,
|
getPreferredLocation(R.PreferredDeclaration,
|
||||||
Sym.CanonicalDeclaration, Scratch),
|
Sym.CanonicalDeclaration, Scratch),
|
||||||
*MainFilePath))
|
MainFilePath))
|
||||||
R.PreferredDeclaration = *Loc;
|
R.PreferredDeclaration = *Loc;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@ -345,6 +313,60 @@ std::vector<LocatedSymbol> locateSymbolAt(ParsedAST &AST, Position Pos,
|
|||||||
return Result;
|
return Result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::vector<LocatedSymbol> locateSymbolAt(ParsedAST &AST, Position Pos,
|
||||||
|
const SymbolIndex *Index) {
|
||||||
|
const auto &SM = AST.getSourceManager();
|
||||||
|
auto MainFilePath =
|
||||||
|
getCanonicalPath(SM.getFileEntryForID(SM.getMainFileID()), SM);
|
||||||
|
if (!MainFilePath) {
|
||||||
|
elog("Failed to get a path for the main file, so no references");
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (auto File = locateFileReferent(Pos, AST, *MainFilePath))
|
||||||
|
return {std::move(*File)};
|
||||||
|
|
||||||
|
auto CurLoc = sourceLocationInMainFile(SM, Pos);
|
||||||
|
if (!CurLoc) {
|
||||||
|
elog("locateSymbolAt failed to convert position to source location: {0}",
|
||||||
|
CurLoc.takeError());
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
const syntax::Token *TouchedIdentifier =
|
||||||
|
syntax::spelledIdentifierTouching(*CurLoc, AST.getTokens());
|
||||||
|
if (TouchedIdentifier)
|
||||||
|
if (auto Macro =
|
||||||
|
locateMacroReferent(*TouchedIdentifier, AST, *MainFilePath))
|
||||||
|
// Don't look at the AST or index if we have a macro result.
|
||||||
|
// (We'd just return declarations referenced from the macro's
|
||||||
|
// expansion.)
|
||||||
|
return {*std::move(Macro)};
|
||||||
|
|
||||||
|
return locateASTReferent(*CurLoc, TouchedIdentifier, AST, *MainFilePath,
|
||||||
|
Index);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<DocumentLink> getDocumentLinks(ParsedAST &AST) {
|
||||||
|
const auto &SM = AST.getSourceManager();
|
||||||
|
auto MainFilePath =
|
||||||
|
getCanonicalPath(SM.getFileEntryForID(SM.getMainFileID()), SM);
|
||||||
|
if (!MainFilePath) {
|
||||||
|
elog("Failed to get a path for the main file, so no links");
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<DocumentLink> Result;
|
||||||
|
for (auto &Inc : AST.getIncludeStructure().MainFileIncludes) {
|
||||||
|
if (!Inc.Resolved.empty()) {
|
||||||
|
Result.push_back(DocumentLink(
|
||||||
|
{Inc.R, URIForFile::canonicalize(Inc.Resolved, *MainFilePath)}));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return Result;
|
||||||
|
}
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
|
|
||||||
/// Collects references to symbols within the main file.
|
/// Collects references to symbols within the main file.
|
||||||
|
Loading…
x
Reference in New Issue
Block a user