mirror of
https://github.com/apple/swift.git
synced 2025-12-14 20:36:38 +01:00
Note that in all cases it was either nullptr or ctx.getLazyResolver().
While passing in nullptr might appear at first glance to mean something
("don't type check anything"), in practice we would check for a nullptr
value and pull out ctx.getLazyResolver() instead. Furthermore, with
the lazy resolver going away (at least for resolveDeclSignature() calls),
it won't make sense to do that anymore anyway.
2243 lines
82 KiB
C++
2243 lines
82 KiB
C++
//===--- SwiftSourceDocInfo.cpp -------------------------------------------===//
|
|
//
|
|
// This source file is part of the Swift.org open source project
|
|
//
|
|
// Copyright (c) 2014 - 2017 Apple Inc. and the Swift project authors
|
|
// Licensed under Apache License v2.0 with Runtime Library Exception
|
|
//
|
|
// See https://swift.org/LICENSE.txt for license information
|
|
// See https://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "SwiftASTManager.h"
|
|
#include "SwiftLangSupport.h"
|
|
#include "SourceKit/Support/FileSystemProvider.h"
|
|
#include "SourceKit/Support/ImmutableTextBuffer.h"
|
|
#include "SourceKit/Support/Logging.h"
|
|
#include "SourceKit/Support/UIdent.h"
|
|
|
|
#include "swift/AST/ASTDemangler.h"
|
|
#include "swift/AST/ASTPrinter.h"
|
|
#include "swift/AST/Decl.h"
|
|
#include "swift/AST/NameLookup.h"
|
|
#include "swift/AST/SwiftNameTranslation.h"
|
|
#include "swift/AST/GenericSignature.h"
|
|
#include "swift/Basic/SourceManager.h"
|
|
#include "swift/Frontend/Frontend.h"
|
|
#include "swift/Frontend/PrintingDiagnosticConsumer.h"
|
|
#include "swift/IDE/CommentConversion.h"
|
|
#include "swift/IDE/ModuleInterfacePrinting.h"
|
|
#include "swift/IDE/SourceEntityWalker.h"
|
|
#include "swift/IDE/Utils.h"
|
|
#include "swift/IDE/Refactoring.h"
|
|
#include "swift/IDE/IDERequests.h"
|
|
#include "swift/Markup/XMLUtils.h"
|
|
#include "swift/Sema/IDETypeChecking.h"
|
|
|
|
#include "clang/AST/ASTContext.h"
|
|
#include "clang/AST/DeclObjC.h"
|
|
#include "clang/Basic/CharInfo.h"
|
|
#include "clang/Basic/Module.h"
|
|
#include "clang/Index/USRGeneration.h"
|
|
#include "clang/Lex/Lexer.h"
|
|
|
|
#include "llvm/Support/MemoryBuffer.h"
|
|
|
|
#include <numeric>
|
|
|
|
using namespace SourceKit;
|
|
using namespace swift;
|
|
using namespace swift::ide;
|
|
|
|
namespace {
|
|
class AnnotatedDeclarationPrinter : public XMLEscapingPrinter {
|
|
public:
|
|
AnnotatedDeclarationPrinter(raw_ostream &OS)
|
|
:XMLEscapingPrinter(OS) { }
|
|
|
|
private:
|
|
void printTypeRef(Type T, const TypeDecl *TD, Identifier Name) override {
|
|
printXML("<Type usr=\"");
|
|
SwiftLangSupport::printUSR(TD, OS);
|
|
printXML("\">");
|
|
StreamPrinter::printTypeRef(T, TD, Name);
|
|
printXML("</Type>");
|
|
}
|
|
};
|
|
} // end anonymous namespace
|
|
|
|
static StringRef getTagForDecl(const Decl *D, bool isRef) {
|
|
auto UID = SwiftLangSupport::getUIDForDecl(D, isRef);
|
|
static const char *prefix = "source.lang.swift.";
|
|
assert(UID.getName().startswith(prefix));
|
|
return UID.getName().drop_front(strlen(prefix));
|
|
}
|
|
|
|
static StringRef ExternalParamNameTag = "decl.var.parameter.argument_label";
|
|
static StringRef LocalParamNameTag = "decl.var.parameter.name";
|
|
static StringRef GenericParamNameTag = "decl.generic_type_param.name";
|
|
static StringRef SyntaxKeywordTag = "syntaxtype.keyword";
|
|
|
|
static StringRef getTagForParameter(PrintStructureKind context) {
|
|
switch (context) {
|
|
case PrintStructureKind::FunctionParameter:
|
|
return "decl.var.parameter";
|
|
case PrintStructureKind::FunctionReturnType:
|
|
return "decl.function.returntype";
|
|
case PrintStructureKind::FunctionType:
|
|
return "";
|
|
case PrintStructureKind::TupleType:
|
|
return "tuple";
|
|
case PrintStructureKind::TupleElement:
|
|
return "tuple.element";
|
|
case PrintStructureKind::GenericParameter:
|
|
return "decl.generic_type_param";
|
|
case PrintStructureKind::GenericRequirement:
|
|
return "decl.generic_type_requirement";
|
|
case PrintStructureKind::BuiltinAttribute:
|
|
return "syntaxtype.attribute.builtin";
|
|
case PrintStructureKind::NumberLiteral:
|
|
return "syntaxtype.number";
|
|
case PrintStructureKind::StringLiteral:
|
|
return "syntaxtype.string";
|
|
}
|
|
llvm_unreachable("unexpected parameter kind");
|
|
}
|
|
|
|
static StringRef getDeclNameTagForDecl(const Decl *D) {
|
|
switch (D->getKind()) {
|
|
case DeclKind::Param:
|
|
// When we're examining the parameter itself, it is the local name that is
|
|
// the name of the variable.
|
|
return LocalParamNameTag;
|
|
case DeclKind::GenericTypeParam:
|
|
return ""; // Handled by printName.
|
|
case DeclKind::Constructor:
|
|
case DeclKind::Destructor:
|
|
case DeclKind::Subscript:
|
|
// The names 'init'/'deinit'/'subscript' are actually keywords.
|
|
return SyntaxKeywordTag;
|
|
default:
|
|
return "decl.name";
|
|
}
|
|
}
|
|
|
|
namespace {
|
|
/// A typesafe union of contexts that the printer can be inside.
|
|
/// Currently: Decl, PrintStructureKind
|
|
class PrintContext {
|
|
// Use the low bit to determine the type; store the enum value shifted left
|
|
// to leave the low bit free.
|
|
const uintptr_t value;
|
|
static constexpr unsigned declTag = 0;
|
|
static constexpr unsigned PrintStructureKindTag = 1;
|
|
static constexpr unsigned typeTag = 2;
|
|
static constexpr unsigned tagMask = 3;
|
|
static constexpr unsigned tagShift = 2;
|
|
bool hasTag(unsigned tag) const { return (value & tagMask) == tag; }
|
|
|
|
public:
|
|
PrintContext(const Decl *D) : value(uintptr_t(D)) {
|
|
static_assert(llvm::PointerLikeTypeTraits<Decl *>::NumLowBitsAvailable >=
|
|
tagShift,
|
|
"missing spare bit in Decl *");
|
|
}
|
|
PrintContext(PrintStructureKind K)
|
|
: value((uintptr_t(K) << tagShift) | PrintStructureKindTag) {}
|
|
PrintContext(TypeLoc unused) : value(typeTag) {}
|
|
|
|
/// Get the context as a Decl, or nullptr.
|
|
const Decl *getDecl() const {
|
|
return hasTag(declTag) ? (const Decl *)value : nullptr;
|
|
}
|
|
/// Get the context as a PrintStructureKind, or None.
|
|
Optional<PrintStructureKind> getPrintStructureKind() const {
|
|
if (!hasTag(PrintStructureKindTag))
|
|
return None;
|
|
return PrintStructureKind(value >> tagShift);
|
|
}
|
|
/// Whether this is a PrintStructureKind context of the given \p kind.
|
|
bool is(PrintStructureKind kind) const {
|
|
auto storedKind = getPrintStructureKind();
|
|
return storedKind && *storedKind == kind;
|
|
}
|
|
bool isType() const { return hasTag(typeTag); }
|
|
};
|
|
|
|
/// An ASTPrinter for annotating declarations with XML tags that describe the
|
|
/// key substructure of the declaration for CursorInfo/DocInfo.
|
|
///
|
|
/// Prints declarations with decl- and type-specific tags derived from the
|
|
/// UIDs used for decl/refs. For example (including newlines purely for ease of
|
|
/// reading):
|
|
///
|
|
/// \verbatim
|
|
/// <decl.function.free>
|
|
/// func <decl.name>foo</decl.name>
|
|
/// (
|
|
/// <decl.var.parameter>
|
|
/// <decl.var.parameter.name>x</decl.var.parameter.name>:
|
|
/// <ref.struct usr="Si">Int</ref.struct>
|
|
/// </decl.var.parameter>
|
|
/// ) -> <decl.function.returntype>
|
|
/// <ref.struct usr="Si">Int</ref.struct></decl.function.returntype>
|
|
/// </decl.function.free>
|
|
/// \endverbatim
|
|
class FullyAnnotatedDeclarationPrinter final : public XMLEscapingPrinter {
|
|
public:
|
|
FullyAnnotatedDeclarationPrinter(raw_ostream &OS) : XMLEscapingPrinter(OS) {}
|
|
|
|
private:
|
|
|
|
// MARK: The ASTPrinter callback interface.
|
|
|
|
void printDeclPre(const Decl *D, Optional<BracketOptions> Bracket) override {
|
|
contextStack.emplace_back(PrintContext(D));
|
|
openTag(getTagForDecl(D, /*isRef=*/false));
|
|
}
|
|
void printDeclPost(const Decl *D, Optional<BracketOptions> Bracket) override {
|
|
assert(contextStack.back().getDecl() == D && "unmatched printDeclPre");
|
|
contextStack.pop_back();
|
|
closeTag(getTagForDecl(D, /*isRef=*/false));
|
|
}
|
|
|
|
void printDeclLoc(const Decl *D) override {
|
|
auto tag = getDeclNameTagForDecl(D);
|
|
if (!tag.empty())
|
|
openTag(tag);
|
|
}
|
|
void printDeclNameEndLoc(const Decl *D) override {
|
|
auto tag = getDeclNameTagForDecl(D);
|
|
if (!tag.empty())
|
|
closeTag(tag);
|
|
}
|
|
|
|
void printTypePre(const TypeLoc &TL) override {
|
|
auto tag = getTypeTagForCurrentContext();
|
|
contextStack.emplace_back(PrintContext(TL));
|
|
if (!tag.empty())
|
|
openTag(tag);
|
|
}
|
|
void printTypePost(const TypeLoc &TL) override {
|
|
assert(contextStack.back().isType());
|
|
contextStack.pop_back();
|
|
auto tag = getTypeTagForCurrentContext();
|
|
if (!tag.empty())
|
|
closeTag(tag);
|
|
}
|
|
|
|
void printStructurePre(PrintStructureKind kind, const Decl *D) override {
|
|
if (kind == PrintStructureKind::TupleElement ||
|
|
kind == PrintStructureKind::TupleType)
|
|
fixupTuple(kind);
|
|
|
|
contextStack.emplace_back(PrintContext(kind));
|
|
auto tag = getTagForParameter(kind);
|
|
if (tag.empty())
|
|
return;
|
|
|
|
if (D && kind == PrintStructureKind::GenericParameter) {
|
|
assert(isa<ValueDecl>(D) && "unexpected non-value decl for param");
|
|
openTagWithUSRForDecl(tag, cast<ValueDecl>(D));
|
|
} else {
|
|
openTag(tag);
|
|
}
|
|
}
|
|
void printStructurePost(PrintStructureKind kind, const Decl *D) override {
|
|
if (kind == PrintStructureKind::TupleElement ||
|
|
kind == PrintStructureKind::TupleType) {
|
|
auto prev = contextStack.pop_back_val();
|
|
(void)prev;
|
|
fixupTuple(kind);
|
|
assert(prev.is(kind) && "unmatched printStructurePre");
|
|
} else {
|
|
assert(contextStack.back().is(kind) && "unmatched printStructurePre");
|
|
contextStack.pop_back();
|
|
}
|
|
|
|
auto tag = getTagForParameter(kind);
|
|
if (!tag.empty())
|
|
closeTag(tag);
|
|
}
|
|
|
|
void printNamePre(PrintNameContext context) override {
|
|
auto tag = getTagForPrintNameContext(context);
|
|
if (!tag.empty())
|
|
openTag(tag);
|
|
}
|
|
void printNamePost(PrintNameContext context) override {
|
|
auto tag = getTagForPrintNameContext(context);
|
|
if (!tag.empty())
|
|
closeTag(tag);
|
|
}
|
|
|
|
void printTypeRef(Type T, const TypeDecl *TD, Identifier name) override {
|
|
auto tag = getTagForDecl(TD, /*isRef=*/true);
|
|
openTagWithUSRForDecl(tag, TD);
|
|
insideRef = true;
|
|
XMLEscapingPrinter::printTypeRef(T, TD, name);
|
|
insideRef = false;
|
|
closeTag(tag);
|
|
}
|
|
|
|
// MARK: Convenience functions for printing.
|
|
|
|
void openTag(StringRef tag) { OS << "<" << tag << ">"; }
|
|
void closeTag(StringRef tag) { OS << "</" << tag << ">"; }
|
|
|
|
void openTagWithUSRForDecl(StringRef tag, const ValueDecl *VD) {
|
|
OS << "<" << tag << " usr=\"";
|
|
SwiftLangSupport::printUSR(VD, OS);
|
|
OS << "\">";
|
|
}
|
|
|
|
// MARK: Misc.
|
|
|
|
StringRef getTypeTagForCurrentContext() const {
|
|
if (contextStack.empty())
|
|
return "";
|
|
|
|
static StringRef parameterTypeTag = "decl.var.parameter.type";
|
|
static StringRef genericParamTypeTag = "decl.generic_type_param.constraint";
|
|
|
|
auto context = contextStack.back();
|
|
if (context.is(PrintStructureKind::FunctionParameter))
|
|
return parameterTypeTag;
|
|
if (context.is(PrintStructureKind::GenericParameter))
|
|
return genericParamTypeTag;
|
|
if (context.is(PrintStructureKind::TupleElement))
|
|
return "tuple.element.type";
|
|
if (context.getPrintStructureKind().hasValue() || context.isType())
|
|
return "";
|
|
|
|
assert(context.getDecl() && "unexpected context kind");
|
|
switch (context.getDecl()->getKind()) {
|
|
case DeclKind::Param:
|
|
return parameterTypeTag;
|
|
case DeclKind::GenericTypeParam:
|
|
return genericParamTypeTag;
|
|
case DeclKind::Var:
|
|
return "decl.var.type";
|
|
case DeclKind::Subscript:
|
|
case DeclKind::Func:
|
|
default:
|
|
return "";
|
|
}
|
|
}
|
|
|
|
StringRef getTagForPrintNameContext(PrintNameContext context) {
|
|
if (insideRef)
|
|
return "";
|
|
|
|
bool insideParam =
|
|
!contextStack.empty() &&
|
|
contextStack.back().is(PrintStructureKind::FunctionParameter);
|
|
|
|
switch (context) {
|
|
case PrintNameContext::FunctionParameterExternal:
|
|
return ExternalParamNameTag;
|
|
case PrintNameContext::FunctionParameterLocal:
|
|
return LocalParamNameTag;
|
|
case PrintNameContext::TupleElement:
|
|
if (insideParam)
|
|
return ExternalParamNameTag;
|
|
return "tuple.element.argument_label";
|
|
case PrintNameContext::Keyword:
|
|
return SyntaxKeywordTag;
|
|
case PrintNameContext::GenericParameter:
|
|
return GenericParamNameTag;
|
|
case PrintNameContext::Attribute:
|
|
return "syntaxtype.attribute.name";
|
|
default:
|
|
return "";
|
|
}
|
|
}
|
|
|
|
/// 'Fix' a tuple or tuple element structure kind to be a function parameter
|
|
/// or function type if we are currently inside a function type. This
|
|
/// simplifies functions that need to differentiate a tuple from the input
|
|
/// part of a function type.
|
|
void fixupTuple(PrintStructureKind &kind) {
|
|
assert(kind == PrintStructureKind::TupleElement ||
|
|
kind == PrintStructureKind::TupleType);
|
|
// Skip over 'type's in the context stack.
|
|
for (auto I = contextStack.rbegin(), E = contextStack.rend(); I != E; ++I) {
|
|
if (I->is(PrintStructureKind::FunctionType)) {
|
|
if (kind == PrintStructureKind::TupleElement)
|
|
kind = PrintStructureKind::FunctionParameter;
|
|
else
|
|
kind = PrintStructureKind::FunctionType;
|
|
break;
|
|
} else if (!I->isType()) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
private:
|
|
/// A stack of contexts being printed, used to determine the context for
|
|
/// subsequent ASTPrinter callbacks.
|
|
llvm::SmallVector<PrintContext, 3> contextStack;
|
|
bool insideRef = false;
|
|
};
|
|
} // end anonymous namespace
|
|
|
|
static Type findBaseTypeForReplacingArchetype(const ValueDecl *VD, const Type Ty) {
|
|
if (Ty.isNull())
|
|
return Type();
|
|
|
|
// Find the nominal type decl related to VD.
|
|
if (!VD->getDeclContext()->isTypeContext())
|
|
return Type();
|
|
|
|
return Ty->getRValueType()->getInOutObjectType()->getMetatypeInstanceType();
|
|
}
|
|
|
|
static void printAnnotatedDeclaration(const ValueDecl *VD,
|
|
const Type BaseTy,
|
|
raw_ostream &OS) {
|
|
AnnotatedDeclarationPrinter Printer(OS);
|
|
PrintOptions PO = PrintOptions::printQuickHelpDeclaration();
|
|
if (BaseTy) {
|
|
PO.setBaseType(BaseTy);
|
|
PO.PrintAsMember = true;
|
|
}
|
|
|
|
// If it's implicit, try to find an overridden ValueDecl that's not implicit.
|
|
// This will ensure we can properly annotate TypeRepr with a usr
|
|
// in AnnotatedDeclarationPrinter.
|
|
while (VD->isImplicit() && VD->getOverriddenDecl())
|
|
VD = VD->getOverriddenDecl();
|
|
|
|
// If this is a property wrapper backing property (_foo) or projected value
|
|
// ($foo) and the wrapped property is not implicit, still print it.
|
|
if (auto *VarD = dyn_cast<VarDecl>(VD)) {
|
|
if (auto *Wrapped = VarD->getOriginalWrappedProperty()) {
|
|
if (!Wrapped->isImplicit())
|
|
PO.TreatAsExplicitDeclList.push_back(VD);
|
|
}
|
|
}
|
|
|
|
// Wrap this up in XML, as that's what we'll use for documentation comments.
|
|
OS<<"<Declaration>";
|
|
VD->print(Printer, PO);
|
|
OS<<"</Declaration>";
|
|
}
|
|
|
|
void SwiftLangSupport::printFullyAnnotatedDeclaration(const ValueDecl *VD,
|
|
Type BaseTy,
|
|
raw_ostream &OS) {
|
|
FullyAnnotatedDeclarationPrinter Printer(OS);
|
|
PrintOptions PO = PrintOptions::printQuickHelpDeclaration();
|
|
if (BaseTy) {
|
|
PO.setBaseType(BaseTy);
|
|
PO.PrintAsMember = true;
|
|
}
|
|
|
|
// If it's implicit, try to find an overridden ValueDecl that's not implicit.
|
|
// This will ensure we can properly annotate TypeRepr with a usr
|
|
// in AnnotatedDeclarationPrinter.
|
|
while (VD->isImplicit() && VD->getOverriddenDecl())
|
|
VD = VD->getOverriddenDecl();
|
|
|
|
// If this is a property wrapper backing property (_foo) or projected value
|
|
// ($foo) and the wrapped property is not implicit, still print it.
|
|
if (auto *VarD = dyn_cast<VarDecl>(VD)) {
|
|
if (auto *Wrapped = VarD->getOriginalWrappedProperty()) {
|
|
if (!Wrapped->isImplicit())
|
|
PO.TreatAsExplicitDeclList.push_back(VD);
|
|
}
|
|
}
|
|
VD->print(Printer, PO);
|
|
}
|
|
|
|
void SwiftLangSupport::printFullyAnnotatedGenericReq(
|
|
const swift::GenericSignature *Sig, llvm::raw_ostream &OS) {
|
|
assert(Sig);
|
|
FullyAnnotatedDeclarationPrinter Printer(OS);
|
|
PrintOptions PO = PrintOptions::printQuickHelpDeclaration();
|
|
Sig->print(Printer, PO);
|
|
}
|
|
|
|
void SwiftLangSupport::printFullyAnnotatedSynthesizedDeclaration(
|
|
const swift::ValueDecl *VD, TypeOrExtensionDecl Target,
|
|
llvm::raw_ostream &OS) {
|
|
// FIXME: Mutable global variable - gross!
|
|
static llvm::SmallDenseMap<swift::ValueDecl*,
|
|
std::unique_ptr<swift::SynthesizedExtensionAnalyzer>> TargetToAnalyzerMap;
|
|
FullyAnnotatedDeclarationPrinter Printer(OS);
|
|
PrintOptions PO = PrintOptions::printQuickHelpDeclaration();
|
|
NominalTypeDecl *TargetNTD = Target.getBaseNominal();
|
|
|
|
if (TargetToAnalyzerMap.count(TargetNTD) == 0) {
|
|
std::unique_ptr<SynthesizedExtensionAnalyzer> Analyzer(
|
|
new SynthesizedExtensionAnalyzer(TargetNTD, PO));
|
|
TargetToAnalyzerMap.insert({TargetNTD, std::move(Analyzer)});
|
|
}
|
|
PO.initForSynthesizedExtension(Target);
|
|
PO.PrintAsMember = true;
|
|
VD->print(Printer, PO);
|
|
}
|
|
|
|
template <typename FnTy>
|
|
void walkRelatedDecls(const ValueDecl *VD, const FnTy &Fn) {
|
|
llvm::SmallDenseMap<DeclName, unsigned, 16> NamesSeen;
|
|
++NamesSeen[VD->getFullName()];
|
|
SmallVector<LookupResultEntry, 8> RelatedDecls;
|
|
|
|
if (isa<ParamDecl>(VD))
|
|
return; // Parameters don't have interesting related declarations.
|
|
|
|
// FIXME: Extract useful related declarations, overloaded functions,
|
|
// if VD is an initializer, we should extract other initializers etc.
|
|
// For now we use UnqualifiedLookup to fetch other declarations with the same
|
|
// base name.
|
|
UnqualifiedLookup Lookup(VD->getBaseName(), VD->getDeclContext());
|
|
for (auto result : Lookup.Results) {
|
|
ValueDecl *RelatedVD = result.getValueDecl();
|
|
if (RelatedVD->getAttrs().isUnavailable(VD->getASTContext()))
|
|
continue;
|
|
|
|
if (RelatedVD != VD) {
|
|
++NamesSeen[RelatedVD->getFullName()];
|
|
RelatedDecls.push_back(result);
|
|
}
|
|
}
|
|
|
|
// Now provide the results along with whether the name is duplicate or not.
|
|
ValueDecl *OriginalBase = VD->getDeclContext()->getSelfNominalTypeDecl();
|
|
for (auto Related : RelatedDecls) {
|
|
ValueDecl *RelatedVD = Related.getValueDecl();
|
|
bool SameBase = Related.getBaseDecl() && Related.getBaseDecl() == OriginalBase;
|
|
Fn(RelatedVD, SameBase, NamesSeen[RelatedVD->getFullName()] > 1);
|
|
}
|
|
}
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
// SwiftLangSupport::getCursorInfo
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
static StringRef getSourceToken(unsigned Offset,
|
|
ImmutableTextSnapshotRef Snap) {
|
|
auto MemBuf = Snap->getBuffer()->getInternalBuffer();
|
|
|
|
// FIXME: Invalid offset shouldn't reach here.
|
|
if (Offset >= MemBuf->getBufferSize())
|
|
return StringRef();
|
|
|
|
SourceManager SM;
|
|
auto MemBufRef = llvm::MemoryBuffer::getMemBuffer(MemBuf->getBuffer(),
|
|
MemBuf->getBufferIdentifier());
|
|
auto BufId = SM.addNewSourceBuffer(std::move(MemBufRef));
|
|
SourceLoc Loc = SM.getLocForOffset(BufId, Offset);
|
|
return Lexer::getTokenAtLocation(SM, Loc).getText();
|
|
}
|
|
|
|
static llvm::Optional<unsigned>
|
|
mapOffsetToOlderSnapshot(unsigned Offset,
|
|
ImmutableTextSnapshotRef NewSnap,
|
|
ImmutableTextSnapshotRef OldSnap) {
|
|
SmallVector<ReplaceImmutableTextUpdateRef, 16> Updates;
|
|
OldSnap->foreachReplaceUntil(NewSnap,
|
|
[&](ReplaceImmutableTextUpdateRef Upd)->bool {
|
|
Updates.push_back(Upd);
|
|
return true;
|
|
});
|
|
|
|
// Walk the updates backwards and "undo" them.
|
|
for (auto I = Updates.rbegin(), E = Updates.rend(); I != E; ++I) {
|
|
auto Upd = *I;
|
|
if (Upd->getByteOffset() <= Offset &&
|
|
Offset < Upd->getByteOffset() + Upd->getText().size())
|
|
return None; // Offset is part of newly inserted text.
|
|
|
|
if (Upd->getByteOffset() <= Offset) {
|
|
Offset += Upd->getLength(); // "bring back" what was removed.
|
|
Offset -= Upd->getText().size(); // "remove" what was added.
|
|
}
|
|
}
|
|
return Offset;
|
|
}
|
|
|
|
static llvm::Optional<unsigned>
|
|
mapOffsetToNewerSnapshot(unsigned Offset,
|
|
ImmutableTextSnapshotRef OldSnap,
|
|
ImmutableTextSnapshotRef NewSnap) {
|
|
bool Completed = OldSnap->foreachReplaceUntil(NewSnap,
|
|
[&](ReplaceImmutableTextUpdateRef Upd)->bool {
|
|
if (Upd->getByteOffset() <= Offset &&
|
|
Offset < Upd->getByteOffset() + Upd->getLength())
|
|
return false; // Offset is part of removed text.
|
|
|
|
if (Upd->getByteOffset() <= Offset) {
|
|
Offset += Upd->getText().size();
|
|
Offset -= Upd->getLength();
|
|
}
|
|
return true;
|
|
});
|
|
|
|
if (Completed)
|
|
return Offset;
|
|
return None;
|
|
}
|
|
|
|
/// Tries to remap the location from a previous snapshot to the latest one.
|
|
static llvm::Optional<std::pair<unsigned, unsigned>>
|
|
tryRemappingLocToLatestSnapshot(SwiftLangSupport &Lang,
|
|
std::pair<unsigned, unsigned> Range,
|
|
StringRef Filename,
|
|
ArrayRef<ImmutableTextSnapshotRef> PreviousASTSnaps) {
|
|
ImmutableTextSnapshotRef LatestSnap;
|
|
if (auto EditorDoc = Lang.getEditorDocuments()->findByPath(Filename))
|
|
LatestSnap = EditorDoc->getLatestSnapshot();
|
|
if (!LatestSnap)
|
|
return Range;
|
|
|
|
for (auto &PrevSnap : PreviousASTSnaps) {
|
|
if (PrevSnap->isFromSameBuffer(LatestSnap)) {
|
|
if (PrevSnap->getStamp() == LatestSnap->getStamp())
|
|
return Range;
|
|
|
|
auto OptBegin = mapOffsetToNewerSnapshot(Range.first,
|
|
PrevSnap, LatestSnap);
|
|
if (!OptBegin.hasValue())
|
|
return None;
|
|
|
|
auto OptEnd = mapOffsetToNewerSnapshot(Range.first+Range.second,
|
|
PrevSnap, LatestSnap);
|
|
if (!OptEnd.hasValue())
|
|
return None;
|
|
|
|
return std::make_pair(*OptBegin, *OptEnd-*OptBegin);
|
|
}
|
|
}
|
|
|
|
return Range;
|
|
}
|
|
|
|
|
|
/// Returns true for error.
|
|
static bool passCursorInfoForModule(ModuleEntity Mod,
|
|
SwiftInterfaceGenMap &IFaceGenContexts,
|
|
const CompilerInvocation &Invok,
|
|
std::function<void(const RequestResult<CursorInfoData> &)> Receiver) {
|
|
std::string Name = Mod.getName();
|
|
std::string FullName = Mod.getFullName();
|
|
CursorInfoData Info;
|
|
Info.Kind = SwiftLangSupport::getUIDForModuleRef();
|
|
Info.Name = Name;
|
|
Info.ModuleName = FullName;
|
|
if (auto IFaceGenRef = IFaceGenContexts.find(Info.ModuleName, Invok))
|
|
Info.ModuleInterfaceName = IFaceGenRef->getDocumentName();
|
|
Info.IsSystem = Mod.isSystemModule();
|
|
std::vector<StringRef> Groups;
|
|
if (auto MD = Mod.getAsSwiftModule()) {
|
|
Info.ModuleGroupArray = ide::collectModuleGroups(const_cast<ModuleDecl*>(MD),
|
|
Groups);
|
|
}
|
|
Receiver(RequestResult<CursorInfoData>::fromResult(Info));
|
|
return false;
|
|
}
|
|
|
|
static void
|
|
collectAvailableRenameInfo(const ValueDecl *VD,
|
|
std::vector<UIdent> &RefactoringIds,
|
|
DelayedStringRetriever &RefactroingNameOS,
|
|
DelayedStringRetriever &RefactoringReasonOS) {
|
|
std::vector<ide::RenameAvailabiliyInfo> Scratch;
|
|
for (auto Info : ide::collectRenameAvailabilityInfo(VD, Scratch)) {
|
|
RefactoringIds.push_back(SwiftLangSupport::
|
|
getUIDForRefactoringKind(Info.Kind));
|
|
RefactroingNameOS.startPiece();
|
|
RefactroingNameOS << ide::getDescriptiveRefactoringKindName(Info.Kind);
|
|
RefactroingNameOS.endPiece();
|
|
RefactoringReasonOS.startPiece();
|
|
RefactoringReasonOS << ide::getDescriptiveRenameUnavailableReason(Info.
|
|
AvailableKind);
|
|
RefactoringReasonOS.endPiece();
|
|
}
|
|
}
|
|
|
|
static void
|
|
serializeRefactoringKinds(ArrayRef<RefactoringKind> AllKinds,
|
|
std::vector<UIdent> &RefactoringIds,
|
|
DelayedStringRetriever &RefactroingNameOS,
|
|
DelayedStringRetriever &RefactoringReasonOS) {
|
|
for (auto Kind : AllKinds) {
|
|
RefactoringIds.push_back(SwiftLangSupport::getUIDForRefactoringKind(Kind));
|
|
RefactroingNameOS.startPiece();
|
|
RefactroingNameOS << ide::getDescriptiveRefactoringKindName(Kind);
|
|
RefactroingNameOS.endPiece();
|
|
RefactoringReasonOS.startPiece();
|
|
RefactoringReasonOS.endPiece();
|
|
}
|
|
}
|
|
|
|
static void
|
|
collectAvailableRefactoringsOtherThanRename(SourceFile *SF,
|
|
ResolvedCursorInfo CursorInfo,
|
|
std::vector<UIdent> &RefactoringIds,
|
|
DelayedStringRetriever &RefactroingNameOS,
|
|
DelayedStringRetriever &RefactoringReasonOS) {
|
|
std::vector<RefactoringKind> Scratch;
|
|
serializeRefactoringKinds(collectAvailableRefactorings(SF, CursorInfo, Scratch,
|
|
/*ExcludeRename*/true), RefactoringIds, RefactroingNameOS,
|
|
RefactoringReasonOS);
|
|
}
|
|
|
|
static Optional<unsigned>
|
|
getParamParentNameOffset(const ValueDecl *VD, SourceLoc Cursor) {
|
|
if (Cursor.isInvalid())
|
|
return None;
|
|
SourceLoc Loc;
|
|
if (auto PD = dyn_cast<ParamDecl>(VD)) {
|
|
|
|
// Avoid returning parent loc for internal-only names.
|
|
if (PD->getArgumentNameLoc().isValid() && PD->getArgumentNameLoc() != Cursor)
|
|
return None;
|
|
auto *DC = PD->getDeclContext();
|
|
switch (DC->getContextKind()) {
|
|
case DeclContextKind::SubscriptDecl:
|
|
Loc = cast<SubscriptDecl>(DC)->getNameLoc();
|
|
break;
|
|
case DeclContextKind::AbstractFunctionDecl:
|
|
Loc = cast<AbstractFunctionDecl>(DC)->getNameLoc();
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
if (Loc.isInvalid())
|
|
return None;
|
|
auto &SM = VD->getASTContext().SourceMgr;
|
|
return SM.getLocOffsetInBuffer(Loc, SM.findBufferContainingLoc(Loc));
|
|
}
|
|
|
|
/// Returns true on success, false on error (and sets `Diagnostic` accordingly).
|
|
static bool passCursorInfoForDecl(SourceFile* SF,
|
|
const ValueDecl *VD,
|
|
const ModuleDecl *MainModule,
|
|
const Type ContainerTy,
|
|
bool IsRef,
|
|
bool RetrieveRefactoring,
|
|
ResolvedCursorInfo TheTok,
|
|
Optional<unsigned> OrigBufferID,
|
|
SourceLoc CursorLoc,
|
|
ArrayRef<RefactoringInfo> KownRefactoringInfoFromRange,
|
|
SwiftLangSupport &Lang,
|
|
const CompilerInvocation &Invok,
|
|
std::string &Diagnostic,
|
|
ArrayRef<ImmutableTextSnapshotRef> PreviousASTSnaps,
|
|
std::function<void(const RequestResult<CursorInfoData> &)> Receiver) {
|
|
if (AvailableAttr::isUnavailable(VD)) {
|
|
Diagnostic = "Unavailable in the current compilation context.";
|
|
return false;
|
|
}
|
|
|
|
SmallString<64> SS;
|
|
auto BaseType = findBaseTypeForReplacingArchetype(VD, ContainerTy);
|
|
bool InSynthesizedExtension = false;
|
|
if (BaseType) {
|
|
if (auto Target = BaseType->getAnyNominal()) {
|
|
SynthesizedExtensionAnalyzer Analyzer(Target,
|
|
PrintOptions::printModuleInterface());
|
|
InSynthesizedExtension = Analyzer.isInSynthesizedExtension(VD);
|
|
}
|
|
}
|
|
|
|
unsigned NameBegin = SS.size();
|
|
{
|
|
llvm::raw_svector_ostream OS(SS);
|
|
SwiftLangSupport::printDisplayName(VD, OS);
|
|
}
|
|
unsigned NameEnd = SS.size();
|
|
|
|
// If VD is the syntehsized property wrapper backing storage (_foo) or
|
|
// projected value ($foo) of a property (foo), use that property's USR instead
|
|
// so that a rename refactoring renames all three (foo, $foo, and _foo).
|
|
const ValueDecl* OriginalProperty = VD;
|
|
if (auto *VarD = dyn_cast<VarDecl>(VD)) {
|
|
if (auto *Wrapped = VarD->getOriginalWrappedProperty())
|
|
OriginalProperty = Wrapped;
|
|
}
|
|
|
|
unsigned USRBegin = SS.size();
|
|
{
|
|
llvm::raw_svector_ostream OS(SS);
|
|
SwiftLangSupport::printUSR(OriginalProperty, OS);
|
|
if (InSynthesizedExtension) {
|
|
OS << LangSupport::SynthesizedUSRSeparator;
|
|
SwiftLangSupport::printUSR(BaseType->getAnyNominal(), OS);
|
|
}
|
|
}
|
|
unsigned USREnd = SS.size();
|
|
|
|
unsigned TypenameBegin = SS.size();
|
|
if (VD->hasInterfaceType()) {
|
|
llvm::raw_svector_ostream OS(SS);
|
|
PrintOptions Options;
|
|
Options.PrintTypeAliasUnderlyingType = true;
|
|
VD->getInterfaceType().print(OS, Options);
|
|
}
|
|
unsigned TypenameEnd = SS.size();
|
|
|
|
unsigned MangledTypeStart = SS.size();
|
|
{
|
|
llvm::raw_svector_ostream OS(SS);
|
|
SwiftLangSupport::printDeclTypeUSR(VD, OS);
|
|
}
|
|
unsigned MangledTypeEnd = SS.size();
|
|
|
|
unsigned MangledContainerTypeStart = SS.size();
|
|
if (ContainerTy && !ContainerTy->hasArchetype()) {
|
|
llvm::raw_svector_ostream OS(SS);
|
|
SwiftLangSupport::printTypeUSR(ContainerTy, OS);
|
|
}
|
|
unsigned MangledContainerTypeEnd = SS.size();
|
|
|
|
// If VD is the syntehsized property wrapper backing storage (_foo) or
|
|
// projected value ($foo) of a property (foo), use that property's
|
|
// documentation instead.
|
|
unsigned DocCommentBegin = SS.size();
|
|
{
|
|
llvm::raw_svector_ostream OS(SS);
|
|
ide::getDocumentationCommentAsXML(OriginalProperty, OS);
|
|
}
|
|
unsigned DocCommentEnd = SS.size();
|
|
|
|
unsigned DeclBegin = SS.size();
|
|
{
|
|
llvm::raw_svector_ostream OS(SS);
|
|
printAnnotatedDeclaration(VD, BaseType, OS);
|
|
}
|
|
unsigned DeclEnd = SS.size();
|
|
|
|
unsigned FullDeclBegin = SS.size();
|
|
{
|
|
llvm::raw_svector_ostream OS(SS);
|
|
SwiftLangSupport::printFullyAnnotatedDeclaration(VD, BaseType, OS);
|
|
}
|
|
unsigned FullDeclEnd = SS.size();
|
|
|
|
unsigned GroupBegin = SS.size();
|
|
{
|
|
llvm::raw_svector_ostream OS(SS);
|
|
auto *GroupVD = InSynthesizedExtension ? BaseType->getAnyNominal() : VD;
|
|
if (auto OP = GroupVD->getGroupName())
|
|
OS << OP.getValue();
|
|
}
|
|
unsigned GroupEnd = SS.size();
|
|
|
|
unsigned LocalizationBegin = SS.size();
|
|
{
|
|
llvm::raw_svector_ostream OS(SS);
|
|
ide::getLocalizationKey(VD, OS);
|
|
}
|
|
unsigned LocalizationEnd = SS.size();
|
|
|
|
std::vector<UIdent> RefactoringIds;
|
|
DelayedStringRetriever RefactoringNameOS(SS);
|
|
DelayedStringRetriever RefactoringReasonOS(SS);
|
|
if (RetrieveRefactoring) {
|
|
collectAvailableRenameInfo(VD, RefactoringIds, RefactoringNameOS,
|
|
RefactoringReasonOS);
|
|
collectAvailableRefactoringsOtherThanRename(SF, TheTok, RefactoringIds,
|
|
RefactoringNameOS, RefactoringReasonOS);
|
|
}
|
|
|
|
DelayedStringRetriever OverUSRsStream(SS);
|
|
|
|
ide::walkOverriddenDecls(VD,
|
|
[&](llvm::PointerUnion<const ValueDecl*, const clang::NamedDecl*> D) {
|
|
OverUSRsStream.startPiece();
|
|
if (auto VD = D.dyn_cast<const ValueDecl*>()) {
|
|
if (SwiftLangSupport::printUSR(VD, OverUSRsStream))
|
|
return;
|
|
} else {
|
|
llvm::SmallString<128> Buf;
|
|
if (clang::index::generateUSRForDecl(
|
|
D.get<const clang::NamedDecl*>(), Buf))
|
|
return;
|
|
OverUSRsStream << Buf.str();
|
|
}
|
|
OverUSRsStream.endPiece();
|
|
});
|
|
|
|
DelayedStringRetriever RelDeclsStream(SS);
|
|
walkRelatedDecls(VD, [&](const ValueDecl *RelatedDecl, bool UseOriginalBase, bool DuplicateName) {
|
|
RelDeclsStream.startPiece();
|
|
{
|
|
RelDeclsStream<<"<RelatedName usr=\"";
|
|
SwiftLangSupport::printUSR(RelatedDecl, RelDeclsStream);
|
|
RelDeclsStream<<"\">";
|
|
if (isa<AbstractFunctionDecl>(RelatedDecl) && DuplicateName) {
|
|
// Related decls are generally overloads, so print parameter types to
|
|
// differentiate them.
|
|
PrintOptions PO;
|
|
PO.SkipAttributes = true;
|
|
PO.SkipIntroducerKeywords = true;
|
|
PO.ArgAndParamPrinting = PrintOptions::ArgAndParamPrintingMode::ArgumentOnly;
|
|
XMLEscapingPrinter Printer(RelDeclsStream);
|
|
if (UseOriginalBase && BaseType) {
|
|
PO.setBaseType(BaseType);
|
|
PO.PrintAsMember = true;
|
|
}
|
|
RelatedDecl->print(Printer, PO);
|
|
} else {
|
|
llvm::SmallString<128> Buf;
|
|
{
|
|
llvm::raw_svector_ostream OSBuf(Buf);
|
|
SwiftLangSupport::printDisplayName(RelatedDecl, OSBuf);
|
|
}
|
|
swift::markup::appendWithXMLEscaping(RelDeclsStream, Buf);
|
|
}
|
|
RelDeclsStream<<"</RelatedName>";
|
|
}
|
|
RelDeclsStream.endPiece();
|
|
});
|
|
|
|
ASTContext &Ctx = VD->getASTContext();
|
|
|
|
ClangImporter *Importer = static_cast<ClangImporter*>(
|
|
Ctx.getClangModuleLoader());
|
|
std::string ModuleName;
|
|
auto ClangNode = VD->getClangNode();
|
|
if (ClangNode) {
|
|
auto ClangMod = Importer->getClangOwningModule(ClangNode);
|
|
if (ClangMod)
|
|
ModuleName = ClangMod->getFullModuleName();
|
|
} else if (VD->getLoc().isInvalid() && VD->getModuleContext() != MainModule) {
|
|
ModuleName = VD->getModuleContext()->getName().str();
|
|
}
|
|
StringRef ModuleInterfaceName;
|
|
if (auto IFaceGenRef = Lang.getIFaceGenContexts().find(ModuleName, Invok))
|
|
ModuleInterfaceName = IFaceGenRef->getDocumentName();
|
|
|
|
UIdent Kind = SwiftLangSupport::getUIDForDecl(VD, IsRef);
|
|
StringRef Name = StringRef(SS.begin()+NameBegin, NameEnd-NameBegin);
|
|
StringRef USR = StringRef(SS.begin()+USRBegin, USREnd-USRBegin);
|
|
StringRef TypeName = StringRef(SS.begin()+TypenameBegin,
|
|
TypenameEnd-TypenameBegin);
|
|
StringRef TypeUsr = StringRef(SS.begin()+MangledTypeStart,
|
|
MangledTypeEnd - MangledTypeStart);
|
|
|
|
StringRef ContainerTypeUsr = StringRef(SS.begin()+MangledContainerTypeStart,
|
|
MangledContainerTypeEnd - MangledContainerTypeStart);
|
|
StringRef DocComment = StringRef(SS.begin()+DocCommentBegin,
|
|
DocCommentEnd-DocCommentBegin);
|
|
StringRef AnnotatedDecl = StringRef(SS.begin()+DeclBegin,
|
|
DeclEnd-DeclBegin);
|
|
StringRef FullyAnnotatedDecl =
|
|
StringRef(SS.begin() + FullDeclBegin, FullDeclEnd - FullDeclBegin);
|
|
StringRef GroupName = StringRef(SS.begin() + GroupBegin, GroupEnd - GroupBegin);
|
|
StringRef LocalizationKey = StringRef(SS.begin() + LocalizationBegin,
|
|
LocalizationEnd - LocalizationBegin);
|
|
|
|
// If VD is the syntehsized property wrapper backing storage (_foo) or
|
|
// projected value ($foo) of a property (foo), base the location on that
|
|
// property instead.
|
|
llvm::Optional<std::pair<unsigned, unsigned>> DeclarationLoc;
|
|
StringRef Filename;
|
|
getLocationInfo(OriginalProperty, DeclarationLoc, Filename);
|
|
if (DeclarationLoc.hasValue()) {
|
|
DeclarationLoc = tryRemappingLocToLatestSnapshot(Lang,
|
|
*DeclarationLoc,
|
|
Filename,
|
|
PreviousASTSnaps);
|
|
if (!DeclarationLoc.hasValue()) {
|
|
Diagnostic = "Failed to remap declaration to latest snapshot.";
|
|
return false;
|
|
}
|
|
}
|
|
|
|
SmallVector<StringRef, 4> OverUSRs;
|
|
OverUSRsStream.retrieve([&](StringRef S) { OverUSRs.push_back(S); });
|
|
|
|
SmallVector<StringRef, 4> AnnotatedRelatedDecls;
|
|
RelDeclsStream.retrieve([&](StringRef S) { AnnotatedRelatedDecls.push_back(S); });
|
|
|
|
SmallVector<RefactoringInfo, 4> RefactoringInfoBuffer;
|
|
for (unsigned I = 0, N = RefactoringIds.size(); I < N; I ++) {
|
|
RefactoringInfoBuffer.push_back({RefactoringIds[I], RefactoringNameOS[I],
|
|
RefactoringReasonOS[I]});
|
|
}
|
|
|
|
// Add available refactoring inheritted from range.
|
|
RefactoringInfoBuffer.insert(RefactoringInfoBuffer.end(),
|
|
KownRefactoringInfoFromRange.begin(),
|
|
KownRefactoringInfoFromRange.end());
|
|
|
|
bool IsSystem = VD->getModuleContext()->isSystemModule();
|
|
std::string TypeInterface;
|
|
|
|
CursorInfoData Info;
|
|
Info.Kind = Kind;
|
|
Info.Name = Name;
|
|
Info.USR = USR;
|
|
Info.TypeName = TypeName;
|
|
Info.TypeUSR = TypeUsr;
|
|
Info.ContainerTypeUSR = ContainerTypeUsr;
|
|
Info.DocComment = DocComment;
|
|
Info.AnnotatedDeclaration = AnnotatedDecl;
|
|
Info.FullyAnnotatedDeclaration = FullyAnnotatedDecl;
|
|
Info.ModuleName = ModuleName;
|
|
Info.ModuleInterfaceName = ModuleInterfaceName;
|
|
Info.DeclarationLoc = DeclarationLoc;
|
|
Info.Filename = Filename;
|
|
Info.OverrideUSRs = OverUSRs;
|
|
Info.AnnotatedRelatedDeclarations = AnnotatedRelatedDecls;
|
|
Info.GroupName = GroupName;
|
|
Info.LocalizationKey = LocalizationKey;
|
|
Info.IsSystem = IsSystem;
|
|
Info.TypeInterface = StringRef();
|
|
Info.AvailableActions = llvm::makeArrayRef(RefactoringInfoBuffer);
|
|
Info.ParentNameOffset = getParamParentNameOffset(VD, CursorLoc);
|
|
Receiver(RequestResult<CursorInfoData>::fromResult(Info));
|
|
return true;
|
|
}
|
|
|
|
static clang::DeclarationName
|
|
getClangDeclarationName(const clang::NamedDecl *ND, NameTranslatingInfo &Info) {
|
|
auto &Ctx = ND->getASTContext();
|
|
auto OrigName = ND->getDeclName();
|
|
assert(SwiftLangSupport::getNameKindForUID(Info.NameKind) == NameKind::ObjC);
|
|
if (Info.BaseName.empty() == Info.ArgNames.empty()) {
|
|
// cannot have both.
|
|
return clang::DeclarationName();
|
|
}
|
|
if (!Info.BaseName.empty()) {
|
|
return clang::DeclarationName(&Ctx.Idents.get(Info.BaseName));
|
|
} else {
|
|
switch (OrigName.getNameKind()) {
|
|
case clang::DeclarationName::ObjCZeroArgSelector:
|
|
case clang::DeclarationName::ObjCOneArgSelector:
|
|
case clang::DeclarationName::ObjCMultiArgSelector:
|
|
break;
|
|
default:
|
|
return clang::DeclarationName();
|
|
}
|
|
|
|
auto OrigSel = OrigName.getObjCSelector();
|
|
unsigned NumPieces = OrigSel.isUnarySelector() ? 1 : OrigSel.getNumArgs();
|
|
if (Info.ArgNames.size() > NumPieces)
|
|
return clang::DeclarationName();
|
|
|
|
ArrayRef<StringRef> Args = llvm::makeArrayRef(Info.ArgNames);
|
|
std::vector<clang::IdentifierInfo *> Pieces;
|
|
for (unsigned i = 0; i < NumPieces; ++i) {
|
|
if (i >= Info.ArgNames.size() || Info.ArgNames[i].empty()) {
|
|
Pieces.push_back(OrigSel.getIdentifierInfoForSlot(i));
|
|
} else {
|
|
StringRef T = Args[i];
|
|
Pieces.push_back(&Ctx.Idents.get(T.endswith(":") ? T.drop_back() : T));
|
|
}
|
|
}
|
|
return clang::DeclarationName(
|
|
Ctx.Selectors.getSelector(OrigSel.getNumArgs(), Pieces.data()));
|
|
}
|
|
}
|
|
|
|
static DeclName getSwiftDeclName(const ValueDecl *VD,
|
|
NameTranslatingInfo &Info) {
|
|
auto &Ctx = VD->getDeclContext()->getASTContext();
|
|
assert(SwiftLangSupport::getNameKindForUID(Info.NameKind) == NameKind::Swift);
|
|
DeclName OrigName = VD->getFullName();
|
|
DeclBaseName BaseName = Info.BaseName.empty()
|
|
? OrigName.getBaseName()
|
|
: DeclBaseName(
|
|
Info.BaseName == "init"
|
|
? DeclBaseName::createConstructor()
|
|
: Ctx.getIdentifier(Info.BaseName));
|
|
auto OrigArgs = OrigName.getArgumentNames();
|
|
SmallVector<Identifier, 8> Args(OrigArgs.begin(), OrigArgs.end());
|
|
if (Info.ArgNames.size() > OrigArgs.size())
|
|
return DeclName();
|
|
for (unsigned i = 0; i < OrigArgs.size(); ++i) {
|
|
if (i < Info.ArgNames.size() && !Info.ArgNames[i].empty()) {
|
|
StringRef Arg = Info.ArgNames[i];
|
|
Args[i] = Ctx.getIdentifier(Arg == "_" ? StringRef() : Arg);
|
|
}
|
|
}
|
|
return DeclName(Ctx, BaseName, llvm::makeArrayRef(Args));
|
|
}
|
|
|
|
/// Returns true on success, false on error (and sets `Diagnostic` accordingly).
|
|
static bool passNameInfoForDecl(ResolvedCursorInfo CursorInfo,
|
|
NameTranslatingInfo &Info,
|
|
std::string &Diagnostic,
|
|
std::function<void(const RequestResult<NameTranslatingInfo> &)> Receiver) {
|
|
auto *VD = CursorInfo.ValueD;
|
|
|
|
// If the given name is not a function name, and the cursor points to
|
|
// a contructor call, we use the type declaration instead of the init
|
|
// declaration to translate the name.
|
|
if (Info.ArgNames.empty() && !Info.IsZeroArgSelector) {
|
|
if (auto *TD = CursorInfo.CtorTyRef) {
|
|
VD = TD;
|
|
}
|
|
}
|
|
switch (SwiftLangSupport::getNameKindForUID(Info.NameKind)) {
|
|
case NameKind::Swift: {
|
|
NameTranslatingInfo Result;
|
|
auto DeclName = getSwiftDeclName(VD, Info);
|
|
if (!DeclName) {
|
|
Diagnostic = "Unable to resolve Swift declaration name.";
|
|
return false;
|
|
}
|
|
auto ResultPair =
|
|
swift::objc_translation::getObjCNameForSwiftDecl(VD, DeclName);
|
|
Identifier Name = ResultPair.first;
|
|
if (!Name.empty()) {
|
|
Result.NameKind = SwiftLangSupport::getUIDForNameKind(NameKind::ObjC);
|
|
Result.BaseName = Name.str();
|
|
Receiver(RequestResult<NameTranslatingInfo>::fromResult(Result));
|
|
} else if (ObjCSelector Selector = ResultPair.second) {
|
|
Result.NameKind = SwiftLangSupport::getUIDForNameKind(NameKind::ObjC);
|
|
SmallString<64> Buffer;
|
|
StringRef Total = Selector.getString(Buffer);
|
|
SmallVector<StringRef, 4> Pieces;
|
|
Total.split(Pieces, ":");
|
|
if (Selector.getNumArgs()) {
|
|
assert(Pieces.back().empty());
|
|
Pieces.pop_back();
|
|
} else {
|
|
Result.IsZeroArgSelector = true;
|
|
}
|
|
Result.ArgNames.insert(Result.ArgNames.begin(), Pieces.begin(), Pieces.end());
|
|
Receiver(RequestResult<NameTranslatingInfo>::fromResult(Result));
|
|
} else {
|
|
Diagnostic = "Unable to resolve name info.";
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
case NameKind::ObjC: {
|
|
ClangImporter *Importer = static_cast<ClangImporter *>(VD->getDeclContext()->
|
|
getASTContext().getClangModuleLoader());
|
|
|
|
const clang::NamedDecl *Named = nullptr;
|
|
auto *BaseDecl = VD;
|
|
|
|
while (!Named && BaseDecl) {
|
|
Named = dyn_cast_or_null<clang::NamedDecl>(BaseDecl->getClangDecl());
|
|
BaseDecl = BaseDecl->getOverriddenDecl();
|
|
}
|
|
if (!Named) {
|
|
Diagnostic = "Unable to resolve a named declaration.";
|
|
return false;
|
|
}
|
|
|
|
auto ObjCName = getClangDeclarationName(Named, Info);
|
|
if (!ObjCName) {
|
|
Diagnostic = "Unable to resolve ObjC declaration name.";
|
|
return false;
|
|
}
|
|
|
|
DeclName Name = Importer->importName(Named, ObjCName);
|
|
NameTranslatingInfo Result;
|
|
Result.NameKind = SwiftLangSupport::getUIDForNameKind(NameKind::Swift);
|
|
Result.BaseName = Name.getBaseName().userFacingName();
|
|
std::transform(Name.getArgumentNames().begin(),
|
|
Name.getArgumentNames().end(),
|
|
std::back_inserter(Result.ArgNames),
|
|
[](Identifier Id) { return Id.str(); });
|
|
Receiver(RequestResult<NameTranslatingInfo>::fromResult(Result));
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
class CursorRangeInfoConsumer : public SwiftASTConsumer {
|
|
protected:
|
|
SwiftLangSupport ⟪
|
|
SwiftInvocationRef ASTInvok;
|
|
StringRef InputFile;
|
|
unsigned Offset;
|
|
unsigned Length;
|
|
|
|
private:
|
|
const bool TryExistingAST;
|
|
SmallVector<ImmutableTextSnapshotRef, 4> PreviousASTSnaps;
|
|
|
|
protected:
|
|
bool CancelOnSubsequentRequest;
|
|
protected:
|
|
ArrayRef<ImmutableTextSnapshotRef> getPreviousASTSnaps() {
|
|
return llvm::makeArrayRef(PreviousASTSnaps);
|
|
}
|
|
|
|
public:
|
|
CursorRangeInfoConsumer(StringRef InputFile, unsigned Offset, unsigned Length,
|
|
SwiftLangSupport &Lang, SwiftInvocationRef ASTInvok,
|
|
bool TryExistingAST, bool CancelOnSubsequentRequest)
|
|
: Lang(Lang), ASTInvok(ASTInvok),InputFile(InputFile), Offset(Offset),
|
|
Length(Length), TryExistingAST(TryExistingAST),
|
|
CancelOnSubsequentRequest(CancelOnSubsequentRequest) {}
|
|
|
|
bool canUseASTWithSnapshots(ArrayRef<ImmutableTextSnapshotRef> Snapshots) override {
|
|
if (!TryExistingAST) {
|
|
LOG_INFO_FUNC(High, "will resolve using up-to-date AST");
|
|
return false;
|
|
}
|
|
|
|
// If there is an existing AST and the offset can be mapped back to the
|
|
// document snapshot that was used to create it, then use that AST.
|
|
// The downside is that we may return stale information, but we get the
|
|
// benefit of increased responsiveness, since the request will not be
|
|
// blocked waiting on the AST to be fully typechecked.
|
|
|
|
ImmutableTextSnapshotRef InputSnap;
|
|
if (auto EditorDoc = Lang.getEditorDocuments()->findByPath(InputFile))
|
|
InputSnap = EditorDoc->getLatestSnapshot();
|
|
if (!InputSnap)
|
|
return false;
|
|
|
|
auto mappedBackOffset = [&]()->llvm::Optional<unsigned> {
|
|
for (auto &Snap : Snapshots) {
|
|
if (Snap->isFromSameBuffer(InputSnap)) {
|
|
if (Snap->getStamp() == InputSnap->getStamp())
|
|
return Offset;
|
|
|
|
auto OptOffset = mapOffsetToOlderSnapshot(Offset, InputSnap, Snap);
|
|
if (!OptOffset.hasValue())
|
|
return None;
|
|
|
|
// Check that the new and old offset still point to the same token.
|
|
StringRef NewTok = getSourceToken(Offset, InputSnap);
|
|
if (NewTok.empty())
|
|
return None;
|
|
if (NewTok == getSourceToken(OptOffset.getValue(), Snap))
|
|
return OptOffset;
|
|
|
|
return None;
|
|
}
|
|
}
|
|
return None;
|
|
};
|
|
|
|
auto OldOffsetOpt = mappedBackOffset();
|
|
if (OldOffsetOpt.hasValue()) {
|
|
Offset = *OldOffsetOpt;
|
|
PreviousASTSnaps.append(Snapshots.begin(), Snapshots.end());
|
|
LOG_INFO_FUNC(High, "will try existing AST");
|
|
return true;
|
|
}
|
|
|
|
LOG_INFO_FUNC(High, "will resolve using up-to-date AST");
|
|
return false;
|
|
}
|
|
};
|
|
|
|
static void resolveCursor(
|
|
SwiftLangSupport &Lang, StringRef InputFile, unsigned Offset,
|
|
unsigned Length, bool Actionables, SwiftInvocationRef Invok,
|
|
bool TryExistingAST, bool CancelOnSubsequentRequest,
|
|
llvm::IntrusiveRefCntPtr<llvm::vfs::FileSystem> fileSystem,
|
|
std::function<void(const RequestResult<CursorInfoData> &)> Receiver) {
|
|
assert(Invok);
|
|
assert(fileSystem);
|
|
|
|
class CursorInfoConsumer : public CursorRangeInfoConsumer {
|
|
bool Actionables;
|
|
std::function<void(const RequestResult<CursorInfoData> &)> Receiver;
|
|
|
|
public:
|
|
CursorInfoConsumer(StringRef InputFile, unsigned Offset,
|
|
unsigned Length, bool Actionables,
|
|
SwiftLangSupport &Lang,
|
|
SwiftInvocationRef ASTInvok,
|
|
bool TryExistingAST,
|
|
bool CancelOnSubsequentRequest,
|
|
std::function<void(const RequestResult<CursorInfoData> &)> Receiver)
|
|
: CursorRangeInfoConsumer(InputFile, Offset, Length, Lang, ASTInvok,
|
|
TryExistingAST, CancelOnSubsequentRequest),
|
|
Actionables(Actionables),
|
|
Receiver(std::move(Receiver)){ }
|
|
|
|
void handlePrimaryAST(ASTUnitRef AstUnit) override {
|
|
auto &CompIns = AstUnit->getCompilerInstance();
|
|
ModuleDecl *MainModule = CompIns.getMainModule();
|
|
SourceManager &SM = CompIns.getSourceMgr();
|
|
unsigned BufferID = AstUnit->getPrimarySourceFile().getBufferID().getValue();
|
|
SourceLoc Loc =
|
|
Lexer::getLocForStartOfToken(SM, BufferID, Offset);
|
|
if (Loc.isInvalid()) {
|
|
Receiver(RequestResult<CursorInfoData>::fromError(
|
|
"Unable to resolve the start of the token."));
|
|
return;
|
|
}
|
|
|
|
// Sanitize length.
|
|
if (Length) {
|
|
SourceLoc TokEnd = Lexer::getLocForEndOfToken(SM, Loc);
|
|
SourceLoc EndLoc = SM.getLocForOffset(BufferID, Offset + Length);
|
|
|
|
// If TokEnd is not before the given EndLoc, the EndLoc contains no
|
|
// more stuff than this token, so set the length to 0.
|
|
if (SM.isBeforeInBuffer(EndLoc, TokEnd) || TokEnd == EndLoc)
|
|
Length = 0;
|
|
}
|
|
|
|
// Retrive relevant actions on the code under selection.
|
|
std::vector<RefactoringInfo> AvailableRefactorings;
|
|
if (Actionables && Length) {
|
|
std::vector<RefactoringKind> Scratch;
|
|
RangeConfig Range;
|
|
Range.BufferId = BufferID;
|
|
auto Pair = SM.getLineAndColumn(Loc);
|
|
Range.Line = Pair.first;
|
|
Range.Column = Pair.second;
|
|
Range.Length = Length;
|
|
bool RangeStartMayNeedRename = false;
|
|
for (RefactoringKind Kind :
|
|
collectAvailableRefactorings(&AstUnit->getPrimarySourceFile(),
|
|
Range, RangeStartMayNeedRename, Scratch, {})) {
|
|
AvailableRefactorings.push_back({
|
|
SwiftLangSupport::getUIDForRefactoringKind(Kind),
|
|
getDescriptiveRefactoringKindName(Kind),
|
|
/*UnavailableReason*/ StringRef()
|
|
});
|
|
}
|
|
if (!RangeStartMayNeedRename) {
|
|
CursorInfoData Info;
|
|
|
|
// FIXME: This Kind does not mean anything.
|
|
Info.Kind = SwiftLangSupport::getUIDForModuleRef();
|
|
Info.AvailableActions = llvm::makeArrayRef(AvailableRefactorings);
|
|
|
|
// If Length is given, then the cursor-info request should only about
|
|
// collecting available refactorings for the range.
|
|
Receiver(RequestResult<CursorInfoData>::fromResult(Info));
|
|
return;
|
|
}
|
|
// If the range start may need rename, we fall back to a regular cursor
|
|
// info request to get the available rename kinds.
|
|
}
|
|
|
|
auto *File = &AstUnit->getPrimarySourceFile();
|
|
ResolvedCursorInfo CursorInfo =
|
|
evaluateOrDefault(File->getASTContext().evaluator,
|
|
CursorInfoRequest{CursorInfoOwner(File, Loc)},
|
|
ResolvedCursorInfo());
|
|
|
|
if (CursorInfo.isInvalid()) {
|
|
CursorInfoData Info;
|
|
Info.InternalDiagnostic = "Unable to resolve cursor info.";
|
|
Receiver(RequestResult<CursorInfoData>::fromResult(Info));
|
|
return;
|
|
}
|
|
CompilerInvocation CompInvok;
|
|
ASTInvok->applyTo(CompInvok);
|
|
|
|
switch (CursorInfo.Kind) {
|
|
case CursorInfoKind::ModuleRef:
|
|
passCursorInfoForModule(CursorInfo.Mod, Lang.getIFaceGenContexts(),
|
|
CompInvok, Receiver);
|
|
return;
|
|
case CursorInfoKind::ValueRef: {
|
|
ValueDecl *VD = CursorInfo.ValueD;
|
|
Type ContainerType = CursorInfo.ContainerType;
|
|
if (CursorInfo.CtorTyRef) {
|
|
// Treat constructor calls, e.g. MyType(), as the type itself,
|
|
// rather than its constructor.
|
|
VD = CursorInfo.CtorTyRef;
|
|
ContainerType = Type();
|
|
}
|
|
std::string Diagnostic;
|
|
bool Success = passCursorInfoForDecl(&AstUnit->getPrimarySourceFile(),
|
|
VD, MainModule,
|
|
ContainerType,
|
|
CursorInfo.IsRef,
|
|
Actionables,
|
|
CursorInfo,
|
|
BufferID, Loc,
|
|
AvailableRefactorings,
|
|
Lang, CompInvok, Diagnostic,
|
|
getPreviousASTSnaps(),
|
|
Receiver);
|
|
if (!Success) {
|
|
if (!getPreviousASTSnaps().empty()) {
|
|
// Attempt again using the up-to-date AST.
|
|
resolveCursor(Lang, InputFile, Offset, Length, Actionables,
|
|
ASTInvok,
|
|
/*TryExistingAST=*/false, CancelOnSubsequentRequest,
|
|
SM.getFileSystem(), Receiver);
|
|
} else {
|
|
CursorInfoData Info;
|
|
Info.InternalDiagnostic = Diagnostic;
|
|
Receiver(RequestResult<CursorInfoData>::fromResult(Info));
|
|
}
|
|
}
|
|
return;
|
|
}
|
|
case CursorInfoKind::ExprStart:
|
|
case CursorInfoKind::StmtStart: {
|
|
if (Actionables) {
|
|
SmallString<64> SS;
|
|
std::vector<UIdent> RefactoringIds;
|
|
DelayedStringRetriever NameRetriever(SS);
|
|
DelayedStringRetriever ReasonRetriever(SS);
|
|
collectAvailableRefactoringsOtherThanRename(
|
|
&AstUnit->getPrimarySourceFile(), CursorInfo, RefactoringIds,
|
|
NameRetriever, ReasonRetriever);
|
|
if (auto Size = RefactoringIds.size()) {
|
|
CursorInfoData Info;
|
|
|
|
// FIXME: This Kind does not mean anything.
|
|
Info.Kind = SwiftLangSupport::getUIDForModuleRef();
|
|
for (unsigned I = 0; I < Size; I ++) {
|
|
AvailableRefactorings.push_back({RefactoringIds[I],
|
|
NameRetriever[I], ReasonRetriever[I]});
|
|
}
|
|
Info.AvailableActions = llvm::makeArrayRef(AvailableRefactorings);
|
|
Receiver(RequestResult<CursorInfoData>::fromResult(Info));
|
|
return;
|
|
}
|
|
}
|
|
|
|
CursorInfoData Info;
|
|
Info.InternalDiagnostic =
|
|
"Resolved to incomplete expression or statement.";
|
|
Receiver(RequestResult<CursorInfoData>::fromResult(Info));
|
|
return;
|
|
}
|
|
case CursorInfoKind::Invalid: {
|
|
llvm_unreachable("bad sema token kind");
|
|
}
|
|
}
|
|
}
|
|
|
|
void cancelled() override {
|
|
Receiver(RequestResult<CursorInfoData>::cancelled());
|
|
}
|
|
|
|
void failed(StringRef Error) override {
|
|
LOG_WARN_FUNC("cursor info failed: " << Error);
|
|
Receiver(RequestResult<CursorInfoData>::fromError(Error));
|
|
}
|
|
};
|
|
|
|
auto Consumer = std::make_shared<CursorInfoConsumer>(
|
|
InputFile, Offset, Length, Actionables, Lang, Invok, TryExistingAST,
|
|
CancelOnSubsequentRequest, Receiver);
|
|
|
|
/// FIXME: When request cancellation is implemented and Xcode adopts it,
|
|
/// don't use 'OncePerASTToken'.
|
|
static const char OncePerASTToken = 0;
|
|
static const char OncePerASTTokenWithActionables = 0;
|
|
const void *Once = nullptr;
|
|
if (CancelOnSubsequentRequest)
|
|
Once = Actionables ? &OncePerASTTokenWithActionables : &OncePerASTToken;
|
|
Lang.getASTManager()->processASTAsync(Invok, std::move(Consumer), Once,
|
|
fileSystem);
|
|
}
|
|
|
|
static void resolveName(SwiftLangSupport &Lang, StringRef InputFile,
|
|
unsigned Offset, SwiftInvocationRef Invok,
|
|
bool TryExistingAST,
|
|
NameTranslatingInfo &Input,
|
|
std::function<void(const RequestResult<NameTranslatingInfo> &)> Receiver) {
|
|
assert(Invok);
|
|
|
|
class NameInfoConsumer : public CursorRangeInfoConsumer {
|
|
NameTranslatingInfo Input;
|
|
std::function<void(const RequestResult<NameTranslatingInfo> &)> Receiver;
|
|
|
|
public:
|
|
NameInfoConsumer(StringRef InputFile, unsigned Offset,
|
|
SwiftLangSupport &Lang, SwiftInvocationRef ASTInvok,
|
|
bool TryExistingAST, NameTranslatingInfo Input,
|
|
std::function<void(const RequestResult<NameTranslatingInfo> &)> Receiver)
|
|
: CursorRangeInfoConsumer(InputFile, Offset, 0, Lang, ASTInvok,
|
|
TryExistingAST,
|
|
/*CancelOnSubsequentRequest=*/false),
|
|
Input(std::move(Input)), Receiver(std::move(Receiver)){ }
|
|
|
|
void handlePrimaryAST(ASTUnitRef AstUnit) override {
|
|
auto &CompIns = AstUnit->getCompilerInstance();
|
|
|
|
unsigned BufferID = AstUnit->getPrimarySourceFile().getBufferID().getValue();
|
|
SourceLoc Loc =
|
|
Lexer::getLocForStartOfToken(CompIns.getSourceMgr(), BufferID, Offset);
|
|
if (Loc.isInvalid()) {
|
|
Receiver(RequestResult<NameTranslatingInfo>::fromError(
|
|
"Unable to resolve the start of the token."));
|
|
return;
|
|
}
|
|
|
|
auto *File = &AstUnit->getPrimarySourceFile();
|
|
ResolvedCursorInfo CursorInfo =
|
|
evaluateOrDefault(File->getASTContext().evaluator,
|
|
CursorInfoRequest{CursorInfoOwner(File, Loc)},
|
|
ResolvedCursorInfo());
|
|
if (CursorInfo.isInvalid()) {
|
|
NameTranslatingInfo Info;
|
|
Info.InternalDiagnostic = "Unable to resolve cursor info.";
|
|
Receiver(RequestResult<NameTranslatingInfo>::fromResult(Info));
|
|
return;
|
|
}
|
|
|
|
CompilerInvocation CompInvok;
|
|
ASTInvok->applyTo(CompInvok);
|
|
|
|
switch(CursorInfo.Kind) {
|
|
case CursorInfoKind::ModuleRef:
|
|
return;
|
|
|
|
case CursorInfoKind::ValueRef: {
|
|
std::string Diagnostic;
|
|
bool Success = passNameInfoForDecl(CursorInfo, Input, Diagnostic,
|
|
Receiver);
|
|
if (!Success) {
|
|
if (!getPreviousASTSnaps().empty()) {
|
|
// Attempt again using the up-to-date AST.
|
|
resolveName(Lang, InputFile, Offset, ASTInvok,
|
|
/*TryExistingAST=*/false, Input, Receiver);
|
|
} else {
|
|
NameTranslatingInfo Info;
|
|
Info.InternalDiagnostic = Diagnostic;
|
|
Receiver(RequestResult<NameTranslatingInfo>::fromResult(Info));
|
|
}
|
|
}
|
|
return;
|
|
}
|
|
case CursorInfoKind::ExprStart:
|
|
case CursorInfoKind::StmtStart: {
|
|
NameTranslatingInfo Info;
|
|
Info.InternalDiagnostic =
|
|
"Resolved to incomplete expression or statement.";
|
|
Receiver(RequestResult<NameTranslatingInfo>::fromResult(Info));
|
|
return;
|
|
}
|
|
case CursorInfoKind::Invalid:
|
|
llvm_unreachable("bad sema token kind.");
|
|
}
|
|
}
|
|
|
|
void cancelled() override {
|
|
Receiver(RequestResult<NameTranslatingInfo>::cancelled());
|
|
}
|
|
|
|
void failed(StringRef Error) override {
|
|
LOG_WARN_FUNC("name info failed: " << Error);
|
|
Receiver(RequestResult<NameTranslatingInfo>::fromError(Error));
|
|
}
|
|
};
|
|
|
|
auto Consumer = std::make_shared<NameInfoConsumer>(
|
|
InputFile, Offset, Lang, Invok, TryExistingAST, Input, Receiver);
|
|
|
|
Lang.getASTManager()->processASTAsync(Invok, std::move(Consumer), nullptr,
|
|
llvm::vfs::getRealFileSystem());
|
|
}
|
|
|
|
static void resolveRange(SwiftLangSupport &Lang,
|
|
StringRef InputFile, unsigned Offset, unsigned Length,
|
|
SwiftInvocationRef Invok,
|
|
bool TryExistingAST, bool CancelOnSubsequentRequest,
|
|
std::function<void(const RequestResult<RangeInfo> &)> Receiver) {
|
|
assert(Invok);
|
|
|
|
class RangeInfoConsumer : public CursorRangeInfoConsumer {
|
|
std::function<void(const RequestResult<RangeInfo> &)> Receiver;
|
|
|
|
public:
|
|
RangeInfoConsumer(StringRef InputFile, unsigned Offset, unsigned Length,
|
|
SwiftLangSupport &Lang, SwiftInvocationRef ASTInvok,
|
|
bool TryExistingAST, bool CancelOnSubsequentRequest,
|
|
std::function<void(const RequestResult<RangeInfo> &)> Receiver)
|
|
: CursorRangeInfoConsumer(InputFile, Offset, Length, Lang, ASTInvok,
|
|
TryExistingAST, CancelOnSubsequentRequest),
|
|
Receiver(std::move(Receiver)){ }
|
|
|
|
void handlePrimaryAST(ASTUnitRef AstUnit) override {
|
|
// FIXME: Implement tracing
|
|
auto *File = &AstUnit->getPrimarySourceFile();
|
|
ResolvedRangeInfo Info = evaluateOrDefault(File->getASTContext().evaluator,
|
|
RangeInfoRequest(RangeInfoOwner({File, Offset, Length})),
|
|
ResolvedRangeInfo());
|
|
|
|
CompilerInvocation CompInvok;
|
|
ASTInvok->applyTo(CompInvok);
|
|
|
|
RangeInfo Result;
|
|
Result.RangeKind = Lang.getUIDForRangeKind(Info.Kind);
|
|
if (Info.Kind == RangeKind::Invalid) {
|
|
Result.RangeContent = "";
|
|
} else {
|
|
assert(Info.ContentRange.isValid());
|
|
Result.RangeContent = Info.ContentRange.str();
|
|
}
|
|
|
|
switch (Info.Kind) {
|
|
case RangeKind::SingleExpression: {
|
|
SmallString<64> SS;
|
|
llvm::raw_svector_ostream OS(SS);
|
|
Info.ExitInfo.ReturnType->print(OS);
|
|
Result.ExprType = OS.str();
|
|
Receiver(RequestResult<RangeInfo>::fromResult(Result));
|
|
return;
|
|
}
|
|
case RangeKind::SingleDecl:
|
|
case RangeKind::MultiTypeMemberDecl:
|
|
case RangeKind::MultiStatement:
|
|
case RangeKind::SingleStatement: {
|
|
Receiver(RequestResult<RangeInfo>::fromResult(Result));
|
|
return;
|
|
}
|
|
case RangeKind::PartOfExpression:
|
|
case RangeKind::Invalid:
|
|
if (!getPreviousASTSnaps().empty()) {
|
|
// Attempt again using the up-to-date AST.
|
|
resolveRange(Lang, InputFile, Offset, Length, ASTInvok,
|
|
/*TryExistingAST=*/false, CancelOnSubsequentRequest,
|
|
Receiver);
|
|
} else {
|
|
Receiver(RequestResult<RangeInfo>::fromResult(Result));
|
|
}
|
|
return;
|
|
}
|
|
}
|
|
|
|
void cancelled() override {
|
|
Receiver(RequestResult<RangeInfo>::cancelled());
|
|
}
|
|
|
|
void failed(StringRef Error) override {
|
|
LOG_WARN_FUNC("range info failed: " << Error);
|
|
Receiver(RequestResult<RangeInfo>::fromError(Error));
|
|
}
|
|
};
|
|
|
|
auto Consumer = std::make_shared<RangeInfoConsumer>(
|
|
InputFile, Offset, Length, Lang, Invok, TryExistingAST,
|
|
CancelOnSubsequentRequest, Receiver);
|
|
/// FIXME: When request cancellation is implemented and Xcode adopts it,
|
|
/// don't use 'OncePerASTToken'.
|
|
static const char OncePerASTToken = 0;
|
|
const void *Once = CancelOnSubsequentRequest ? &OncePerASTToken : nullptr;
|
|
Lang.getASTManager()->processASTAsync(Invok, std::move(Consumer), Once,
|
|
llvm::vfs::getRealFileSystem());
|
|
}
|
|
|
|
void SwiftLangSupport::getCursorInfo(
|
|
StringRef InputFile, unsigned Offset, unsigned Length, bool Actionables,
|
|
bool CancelOnSubsequentRequest, ArrayRef<const char *> Args,
|
|
Optional<VFSOptions> vfsOptions,
|
|
std::function<void(const RequestResult<CursorInfoData> &)> Receiver) {
|
|
|
|
std::string error;
|
|
auto fileSystem = getFileSystem(vfsOptions, InputFile, error);
|
|
if (!fileSystem)
|
|
return Receiver(RequestResult<CursorInfoData>::fromError(error));
|
|
|
|
if (auto IFaceGenRef = IFaceGenContexts.get(InputFile)) {
|
|
IFaceGenRef->accessASTAsync([this, IFaceGenRef, Offset, Actionables, Receiver] {
|
|
SwiftInterfaceGenContext::ResolvedEntity Entity;
|
|
Entity = IFaceGenRef->resolveEntityForOffset(Offset);
|
|
if (Entity.isResolved()) {
|
|
CompilerInvocation Invok;
|
|
IFaceGenRef->applyTo(Invok);
|
|
if (Entity.Mod) {
|
|
passCursorInfoForModule(Entity.Mod, IFaceGenContexts, Invok,
|
|
Receiver);
|
|
} else {
|
|
std::string Diagnostic; // Unused.
|
|
// FIXME: Should pass the main module for the interface but currently
|
|
// it's not necessary.
|
|
passCursorInfoForDecl(
|
|
/*SourceFile*/nullptr, Entity.Dcl, /*MainModule*/ nullptr,
|
|
Type(), Entity.IsRef, Actionables, ResolvedCursorInfo(),
|
|
/*OrigBufferID=*/None, SourceLoc(),
|
|
{}, *this, Invok, Diagnostic, {}, Receiver);
|
|
}
|
|
} else {
|
|
CursorInfoData Info;
|
|
Info.InternalDiagnostic =
|
|
"Unable to resolve entity from generated interface.";
|
|
Receiver(RequestResult<CursorInfoData>::fromResult(Info));
|
|
}
|
|
});
|
|
return;
|
|
}
|
|
|
|
std::string Error;
|
|
SwiftInvocationRef Invok =
|
|
ASTMgr->getInvocation(Args, InputFile, fileSystem, Error);
|
|
if (!Invok) {
|
|
LOG_WARN_FUNC("failed to create an ASTInvocation: " << Error);
|
|
Receiver(RequestResult<CursorInfoData>::fromError(Error));
|
|
return;
|
|
}
|
|
|
|
resolveCursor(*this, InputFile, Offset, Length, Actionables, Invok,
|
|
/*TryExistingAST=*/true, CancelOnSubsequentRequest, fileSystem,
|
|
Receiver);
|
|
}
|
|
|
|
void SwiftLangSupport::
|
|
getRangeInfo(StringRef InputFile, unsigned Offset, unsigned Length,
|
|
bool CancelOnSubsequentRequest, ArrayRef<const char *> Args,
|
|
std::function<void(const RequestResult<RangeInfo> &)> Receiver) {
|
|
if (IFaceGenContexts.get(InputFile)) {
|
|
// FIXME: return range info for generated interfaces.
|
|
Receiver(RequestResult<RangeInfo>::fromError(
|
|
"Range info for generated interfaces is not implemented."));
|
|
return;
|
|
}
|
|
std::string Error;
|
|
SwiftInvocationRef Invok = ASTMgr->getInvocation(Args, InputFile, Error);
|
|
if (!Invok) {
|
|
LOG_WARN_FUNC("failed to create an ASTInvocation: " << Error);
|
|
Receiver(RequestResult<RangeInfo>::fromError(Error));
|
|
return;
|
|
}
|
|
if (Length == 0) {
|
|
Receiver(RequestResult<RangeInfo>::fromError("Invalid range length."));
|
|
return;
|
|
}
|
|
resolveRange(*this, InputFile, Offset, Length, Invok, /*TryExistingAST=*/true,
|
|
CancelOnSubsequentRequest, Receiver);
|
|
}
|
|
|
|
void SwiftLangSupport::
|
|
getNameInfo(StringRef InputFile, unsigned Offset, NameTranslatingInfo &Input,
|
|
ArrayRef<const char *> Args,
|
|
std::function<void(const RequestResult<NameTranslatingInfo> &)> Receiver) {
|
|
|
|
if (auto IFaceGenRef = IFaceGenContexts.get(InputFile)) {
|
|
IFaceGenRef->accessASTAsync([IFaceGenRef, Offset, Input, Receiver] {
|
|
SwiftInterfaceGenContext::ResolvedEntity Entity;
|
|
Entity = IFaceGenRef->resolveEntityForOffset(Offset);
|
|
if (Entity.isResolved()) {
|
|
CompilerInvocation Invok;
|
|
IFaceGenRef->applyTo(Invok);
|
|
if (Entity.Mod) {
|
|
// Module is ignored
|
|
} else {
|
|
// FIXME: Should pass the main module for the interface but currently
|
|
// it's not necessary.
|
|
}
|
|
} else {
|
|
NameTranslatingInfo Info;
|
|
Info.InternalDiagnostic =
|
|
"Unable to resolve entity from generated interface.";
|
|
Receiver(RequestResult<NameTranslatingInfo>::fromResult(Info));
|
|
}
|
|
});
|
|
return;
|
|
}
|
|
|
|
std::string Error;
|
|
SwiftInvocationRef Invok = ASTMgr->getInvocation(Args, InputFile, Error);
|
|
if (!Invok) {
|
|
LOG_WARN_FUNC("failed to create an ASTInvocation: " << Error);
|
|
Receiver(RequestResult<NameTranslatingInfo>::fromError(Error));
|
|
return;
|
|
}
|
|
|
|
resolveName(*this, InputFile, Offset, Invok, /*TryExistingAST=*/true, Input,
|
|
Receiver);
|
|
}
|
|
|
|
static void resolveCursorFromUSR(
|
|
SwiftLangSupport &Lang, StringRef InputFile, StringRef USR,
|
|
SwiftInvocationRef Invok, bool TryExistingAST,
|
|
bool CancelOnSubsequentRequest,
|
|
llvm::IntrusiveRefCntPtr<llvm::vfs::FileSystem> fileSystem,
|
|
std::function<void(const RequestResult<CursorInfoData> &)> Receiver) {
|
|
assert(Invok);
|
|
|
|
class CursorInfoConsumer : public SwiftASTConsumer {
|
|
std::string InputFile;
|
|
StringRef USR;
|
|
SwiftLangSupport ⟪
|
|
SwiftInvocationRef ASTInvok;
|
|
const bool TryExistingAST;
|
|
bool CancelOnSubsequentRequest;
|
|
std::function<void(const RequestResult<CursorInfoData> &)> Receiver;
|
|
SmallVector<ImmutableTextSnapshotRef, 4> PreviousASTSnaps;
|
|
|
|
public:
|
|
CursorInfoConsumer(StringRef InputFile, StringRef USR,
|
|
SwiftLangSupport &Lang, SwiftInvocationRef ASTInvok,
|
|
bool TryExistingAST, bool CancelOnSubsequentRequest,
|
|
std::function<void(const RequestResult<CursorInfoData> &)> Receiver)
|
|
: InputFile(InputFile), USR(USR), Lang(Lang),
|
|
ASTInvok(std::move(ASTInvok)), TryExistingAST(TryExistingAST),
|
|
CancelOnSubsequentRequest(CancelOnSubsequentRequest),
|
|
Receiver(std::move(Receiver)) {}
|
|
|
|
bool canUseASTWithSnapshots(
|
|
ArrayRef<ImmutableTextSnapshotRef> Snapshots) override {
|
|
if (!TryExistingAST) {
|
|
LOG_INFO_FUNC(High, "will resolve using up-to-date AST");
|
|
return false;
|
|
}
|
|
|
|
if (!Snapshots.empty()) {
|
|
PreviousASTSnaps.append(Snapshots.begin(), Snapshots.end());
|
|
LOG_INFO_FUNC(High, "will try existing AST");
|
|
return true;
|
|
}
|
|
|
|
LOG_INFO_FUNC(High, "will resolve using up-to-date AST");
|
|
return false;
|
|
}
|
|
|
|
void handlePrimaryAST(ASTUnitRef AstUnit) override {
|
|
auto &CompIns = AstUnit->getCompilerInstance();
|
|
ModuleDecl *MainModule = CompIns.getMainModule();
|
|
|
|
unsigned BufferID =
|
|
AstUnit->getPrimarySourceFile().getBufferID().getValue();
|
|
|
|
if (USR.startswith("c:")) {
|
|
LOG_WARN_FUNC("lookup for C/C++/ObjC USRs not implemented");
|
|
CursorInfoData Info;
|
|
Info.InternalDiagnostic = "Lookup for C/C++/ObjC USRs not implemented.";
|
|
Receiver(RequestResult<CursorInfoData>::fromResult(Info));
|
|
return;
|
|
}
|
|
|
|
auto &context = CompIns.getASTContext();
|
|
TypeDecl *D = Demangle::getTypeDeclForUSR(context, USR);
|
|
|
|
if (!D) {
|
|
CursorInfoData Info;
|
|
Info.InternalDiagnostic = "Unable to resolve type from USR.";
|
|
Receiver(RequestResult<CursorInfoData>::fromResult(Info));
|
|
return;
|
|
}
|
|
|
|
CompilerInvocation CompInvok;
|
|
ASTInvok->applyTo(CompInvok);
|
|
|
|
if (auto *M = dyn_cast<ModuleDecl>(D)) {
|
|
passCursorInfoForModule(M, Lang.getIFaceGenContexts(), CompInvok,
|
|
Receiver);
|
|
} else {
|
|
auto *DC = D->getDeclContext();
|
|
Type selfTy;
|
|
if (DC->isTypeContext()) {
|
|
selfTy = DC->getSelfInterfaceType();
|
|
selfTy = D->getInnermostDeclContext()->mapTypeIntoContext(selfTy);
|
|
}
|
|
std::string Diagnostic;
|
|
bool Success =
|
|
passCursorInfoForDecl(/*SourceFile*/nullptr, D, MainModule, selfTy,
|
|
/*IsRef=*/false, false, ResolvedCursorInfo(),
|
|
BufferID, SourceLoc(), {}, Lang, CompInvok,
|
|
Diagnostic, PreviousASTSnaps, Receiver);
|
|
if (!Success) {
|
|
if (!PreviousASTSnaps.empty()) {
|
|
// Attempt again using the up-to-date AST.
|
|
resolveCursorFromUSR(
|
|
Lang, InputFile, USR, ASTInvok,
|
|
/*TryExistingAST=*/false, CancelOnSubsequentRequest,
|
|
CompIns.getSourceMgr().getFileSystem(), Receiver);
|
|
} else {
|
|
CursorInfoData Info;
|
|
Info.InternalDiagnostic = Diagnostic;
|
|
Receiver(RequestResult<CursorInfoData>::fromResult(Info));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void cancelled() override {
|
|
Receiver(RequestResult<CursorInfoData>::cancelled());
|
|
}
|
|
|
|
void failed(StringRef Error) override {
|
|
LOG_WARN_FUNC("cursor info failed: " << Error);
|
|
Receiver(RequestResult<CursorInfoData>::fromError(Error));
|
|
}
|
|
};
|
|
|
|
auto Consumer = std::make_shared<CursorInfoConsumer>(
|
|
InputFile, USR, Lang, Invok, TryExistingAST, CancelOnSubsequentRequest,
|
|
Receiver);
|
|
/// FIXME: When request cancellation is implemented and Xcode adopts it,
|
|
/// don't use 'OncePerASTToken'.
|
|
static const char OncePerASTToken = 0;
|
|
const void *Once = CancelOnSubsequentRequest ? &OncePerASTToken : nullptr;
|
|
Lang.getASTManager()->processASTAsync(Invok, std::move(Consumer), Once,
|
|
fileSystem);
|
|
}
|
|
|
|
void SwiftLangSupport::getCursorInfoFromUSR(
|
|
StringRef filename, StringRef USR, bool CancelOnSubsequentRequest,
|
|
ArrayRef<const char *> Args, Optional<VFSOptions> vfsOptions,
|
|
std::function<void(const RequestResult<CursorInfoData> &)> Receiver) {
|
|
std::string error;
|
|
|
|
auto fileSystem = getFileSystem(vfsOptions, filename, error);
|
|
if (!fileSystem)
|
|
return Receiver(RequestResult<CursorInfoData>::fromError(error));
|
|
|
|
if (auto IFaceGenRef = IFaceGenContexts.get(filename)) {
|
|
LOG_WARN_FUNC(
|
|
"Info from usr for generated interface not implemented yet.");
|
|
CursorInfoData Info;
|
|
Info.InternalDiagnostic =
|
|
"Info for generated interfaces not implemented.";
|
|
Receiver(RequestResult<CursorInfoData>::fromResult(Info));
|
|
return;
|
|
}
|
|
|
|
std::string Error;
|
|
SwiftInvocationRef Invok =
|
|
ASTMgr->getInvocation(Args, filename, fileSystem, Error);
|
|
if (!Invok) {
|
|
LOG_WARN_FUNC("failed to create an ASTInvocation: " << Error);
|
|
Receiver(RequestResult<CursorInfoData>::fromError(Error));
|
|
return;
|
|
}
|
|
|
|
resolveCursorFromUSR(*this, filename, USR, Invok, /*TryExistingAST=*/true,
|
|
CancelOnSubsequentRequest, fileSystem, Receiver);
|
|
}
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
// SwiftLangSupport::findUSRRange
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
llvm::Optional<std::pair<unsigned, unsigned>>
|
|
SwiftLangSupport::findUSRRange(StringRef DocumentName, StringRef USR) {
|
|
if (auto IFaceGenRef = IFaceGenContexts.get(DocumentName))
|
|
return IFaceGenRef->findUSRRange(USR);
|
|
|
|
// Only works for a module interface document currently.
|
|
// FIXME: Report it as failed request.
|
|
return None;
|
|
}
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
// SwiftLangSupport::findRelatedIdentifiersInFile
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
namespace {
|
|
class RelatedIdScanner : public SourceEntityWalker {
|
|
ValueDecl *Dcl;
|
|
llvm::SmallVectorImpl<std::pair<unsigned, unsigned>> &Ranges;
|
|
SourceManager &SourceMgr;
|
|
unsigned BufferID = -1;
|
|
bool Cancelled = false;
|
|
|
|
public:
|
|
explicit RelatedIdScanner(SourceFile &SrcFile, unsigned BufferID,
|
|
ValueDecl *D,
|
|
llvm::SmallVectorImpl<std::pair<unsigned, unsigned>> &Ranges)
|
|
: Ranges(Ranges), SourceMgr(SrcFile.getASTContext().SourceMgr),
|
|
BufferID(BufferID) {
|
|
if (auto *V = dyn_cast<VarDecl>(D)) {
|
|
// Always use the canonical var decl for comparison. This is so we
|
|
// pick up all occurrences of x in case statements like the below:
|
|
// case .first(let x), .second(let x)
|
|
// fallthrough
|
|
// case .third(let x)
|
|
// print(x)
|
|
Dcl = V->getCanonicalVarDecl();
|
|
|
|
// If we have a prioperty wrapper backing property or projected value, use
|
|
// the wrapped property instead (i.e. if this is _foo or $foo, pretend
|
|
// it's foo).
|
|
if (auto *Wrapped = V->getOriginalWrappedProperty()) {
|
|
Dcl = Wrapped;
|
|
}
|
|
} else {
|
|
Dcl = D;
|
|
}
|
|
}
|
|
|
|
private:
|
|
bool walkToDeclPre(Decl *D, CharSourceRange Range) override {
|
|
if (Cancelled)
|
|
return false;
|
|
if (auto *V = dyn_cast<VarDecl>(D)) {
|
|
// Handle references to the implicitly generated vars in case statements
|
|
// matching multiple patterns
|
|
D = V->getCanonicalVarDecl();
|
|
}
|
|
if (D == Dcl)
|
|
return passId(Range);
|
|
return true;
|
|
}
|
|
|
|
bool visitDeclReference(ValueDecl *D, CharSourceRange Range,
|
|
TypeDecl *CtorTyRef, ExtensionDecl *ExtTyRef, Type T,
|
|
ReferenceMetaData Data) override {
|
|
if (Cancelled)
|
|
return false;
|
|
|
|
if (auto *V = dyn_cast<VarDecl>(D)) {
|
|
D = V->getCanonicalVarDecl();
|
|
|
|
// If we have a prioperty wrapper backing property or projected value, use
|
|
// the wrapped property for comparison instead (i.e. if this is _foo or
|
|
// $foo, pretend it's foo).
|
|
if (auto *Wrapped = V->getOriginalWrappedProperty()) {
|
|
assert(Range.getByteLength() > 1 &&
|
|
(Range.str().front() == '_' || Range.str().front() == '$'));
|
|
D = Wrapped;
|
|
Range = CharSourceRange(Range.getStart().getAdvancedLoc(1), Range.getByteLength() - 1);
|
|
}
|
|
} else if (CtorTyRef) {
|
|
D = CtorTyRef;
|
|
}
|
|
|
|
if (D == Dcl)
|
|
return passId(Range);
|
|
return true;
|
|
}
|
|
|
|
bool passId(CharSourceRange Range) {
|
|
unsigned Offset = SourceMgr.getLocOffsetInBuffer(Range.getStart(),BufferID);
|
|
Ranges.push_back({ Offset, Range.getByteLength() });
|
|
return !Cancelled;
|
|
}
|
|
};
|
|
|
|
} // end anonymous namespace
|
|
|
|
void SwiftLangSupport::findRelatedIdentifiersInFile(
|
|
StringRef InputFile, unsigned Offset,
|
|
bool CancelOnSubsequentRequest,
|
|
ArrayRef<const char *> Args,
|
|
std::function<void(const RequestResult<RelatedIdentsInfo> &)> Receiver) {
|
|
|
|
std::string Error;
|
|
SwiftInvocationRef Invok = ASTMgr->getInvocation(Args, InputFile, Error);
|
|
if (!Invok) {
|
|
LOG_WARN_FUNC("failed to create an ASTInvocation: " << Error);
|
|
Receiver(RequestResult<RelatedIdentsInfo>::fromError(Error));
|
|
return;
|
|
}
|
|
|
|
class RelatedIdConsumer : public SwiftASTConsumer {
|
|
unsigned Offset;
|
|
std::function<void(const RequestResult<RelatedIdentsInfo> &)> Receiver;
|
|
SwiftInvocationRef Invok;
|
|
|
|
public:
|
|
RelatedIdConsumer(unsigned Offset,
|
|
std::function<void(const RequestResult<RelatedIdentsInfo> &)> Receiver,
|
|
SwiftInvocationRef Invok)
|
|
: Offset(Offset), Receiver(std::move(Receiver)), Invok(Invok) { }
|
|
|
|
// FIXME: Don't silently eat errors here.
|
|
void handlePrimaryAST(ASTUnitRef AstUnit) override {
|
|
auto &CompInst = AstUnit->getCompilerInstance();
|
|
auto &SrcFile = AstUnit->getPrimarySourceFile();
|
|
|
|
|
|
SmallVector<std::pair<unsigned, unsigned>, 8> Ranges;
|
|
|
|
auto Action = [&]() {
|
|
unsigned BufferID = SrcFile.getBufferID().getValue();
|
|
SourceLoc Loc =
|
|
Lexer::getLocForStartOfToken(CompInst.getSourceMgr(), BufferID, Offset);
|
|
if (Loc.isInvalid())
|
|
return;
|
|
|
|
ResolvedCursorInfo CursorInfo =
|
|
evaluateOrDefault(SrcFile.getASTContext().evaluator,
|
|
CursorInfoRequest{CursorInfoOwner(&SrcFile, Loc)},
|
|
ResolvedCursorInfo());
|
|
if (CursorInfo.isInvalid())
|
|
return;
|
|
if (CursorInfo.IsKeywordArgument)
|
|
return;
|
|
|
|
ValueDecl *VD = CursorInfo.CtorTyRef ? CursorInfo.CtorTyRef : CursorInfo.ValueD;
|
|
if (!VD)
|
|
return; // This was a module reference.
|
|
|
|
// Only accept pointing to an identifier.
|
|
if (!CursorInfo.IsRef &&
|
|
(isa<ConstructorDecl>(VD) ||
|
|
isa<DestructorDecl>(VD) ||
|
|
isa<SubscriptDecl>(VD)))
|
|
return;
|
|
if (VD->isOperator())
|
|
return;
|
|
|
|
RelatedIdScanner Scanner(SrcFile, BufferID, VD, Ranges);
|
|
|
|
if (auto *Case = getCaseStmtOfCanonicalVar(VD)) {
|
|
Scanner.walk(Case);
|
|
while ((Case = Case->getFallthroughDest().getPtrOrNull())) {
|
|
Scanner.walk(Case);
|
|
}
|
|
} else if (DeclContext *LocalDC = VD->getDeclContext()->getLocalContext()) {
|
|
Scanner.walk(LocalDC);
|
|
} else {
|
|
Scanner.walk(SrcFile);
|
|
}
|
|
};
|
|
Action();
|
|
|
|
RelatedIdentsInfo Info;
|
|
Info.Ranges = Ranges;
|
|
Receiver(RequestResult<RelatedIdentsInfo>::fromResult(Info));
|
|
}
|
|
|
|
void cancelled() override {
|
|
Receiver(RequestResult<RelatedIdentsInfo>::cancelled());
|
|
}
|
|
|
|
void failed(StringRef Error) override {
|
|
LOG_WARN_FUNC("related idents failed: " << Error);
|
|
Receiver(RequestResult<RelatedIdentsInfo>::fromError(Error));
|
|
}
|
|
|
|
static CaseStmt *getCaseStmtOfCanonicalVar(Decl *D) {
|
|
assert(D);
|
|
if (auto *VD = dyn_cast<VarDecl>(D)) {
|
|
if (auto *Canonical = VD->getCanonicalVarDecl()) {
|
|
return dyn_cast_or_null<CaseStmt>(Canonical->getRecursiveParentPatternStmt());
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
};
|
|
|
|
auto Consumer = std::make_shared<RelatedIdConsumer>(Offset, Receiver, Invok);
|
|
/// FIXME: When request cancellation is implemented and Xcode adopts it,
|
|
/// don't use 'OncePerASTToken'.
|
|
static const char OncePerASTToken = 0;
|
|
const void *Once = CancelOnSubsequentRequest ? &OncePerASTToken : nullptr;
|
|
ASTMgr->processASTAsync(Invok, std::move(Consumer), Once,
|
|
llvm::vfs::getRealFileSystem());
|
|
}
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
// SwiftLangSupport::semanticRefactoring
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
static RefactoringKind getIDERefactoringKind(SemanticRefactoringInfo Info) {
|
|
switch(Info.Kind) {
|
|
case SemanticRefactoringKind::None: return RefactoringKind::None;
|
|
#define SEMANTIC_REFACTORING(KIND, NAME, ID) \
|
|
case SemanticRefactoringKind::KIND: return RefactoringKind::KIND;
|
|
#include "swift/IDE/RefactoringKinds.def"
|
|
}
|
|
}
|
|
|
|
void SwiftLangSupport::
|
|
semanticRefactoring(StringRef Filename, SemanticRefactoringInfo Info,
|
|
ArrayRef<const char*> Args,
|
|
CategorizedEditsReceiver Receiver) {
|
|
std::string Error;
|
|
SwiftInvocationRef Invok = ASTMgr->getInvocation(Args, Filename, Error);
|
|
if (!Invok) {
|
|
LOG_WARN_FUNC("failed to create an ASTInvocation: " << Error);
|
|
Receiver(RequestResult<ArrayRef<CategorizedEdits>>::fromError(Error));
|
|
return;
|
|
}
|
|
assert(Invok);
|
|
|
|
class SemaRefactoringConsumer : public SwiftASTConsumer {
|
|
SemanticRefactoringInfo Info;
|
|
CategorizedEditsReceiver Receiver;
|
|
|
|
public:
|
|
SemaRefactoringConsumer(SemanticRefactoringInfo Info,
|
|
CategorizedEditsReceiver Receiver) : Info(Info),
|
|
Receiver(std::move(Receiver)) {}
|
|
|
|
void handlePrimaryAST(ASTUnitRef AstUnit) override {
|
|
auto &CompIns = AstUnit->getCompilerInstance();
|
|
ModuleDecl *MainModule = CompIns.getMainModule();
|
|
RefactoringOptions Opts(getIDERefactoringKind(Info));
|
|
Opts.Range.BufferId = AstUnit->getPrimarySourceFile().getBufferID().
|
|
getValue();
|
|
Opts.Range.Line = Info.Line;
|
|
Opts.Range.Column = Info.Column;
|
|
Opts.Range.Length = Info.Length;
|
|
Opts.PreferredName = Info.PreferredName;
|
|
|
|
RequestRefactoringEditConsumer EditConsumer(Receiver);
|
|
refactorSwiftModule(MainModule, Opts, EditConsumer, EditConsumer);
|
|
}
|
|
|
|
void cancelled() override {
|
|
Receiver(RequestResult<ArrayRef<CategorizedEdits>>::cancelled());
|
|
}
|
|
|
|
void failed(StringRef Error) override {
|
|
Receiver(RequestResult<ArrayRef<CategorizedEdits>>::fromError(Error));
|
|
}
|
|
};
|
|
|
|
auto Consumer = std::make_shared<SemaRefactoringConsumer>(Info, Receiver);
|
|
/// FIXME: When request cancellation is implemented and Xcode adopts it,
|
|
/// don't use 'OncePerASTToken'.
|
|
static const char OncePerASTToken = 0;
|
|
getASTManager()->processASTAsync(Invok, std::move(Consumer), &OncePerASTToken,
|
|
llvm::vfs::getRealFileSystem());
|
|
}
|
|
|
|
void SwiftLangSupport::collectExpressionTypes(StringRef FileName,
|
|
ArrayRef<const char *> Args,
|
|
ArrayRef<const char *> ExpectedProtocols,
|
|
bool CanonicalType,
|
|
std::function<void(const RequestResult<ExpressionTypesInFile> &)> Receiver) {
|
|
std::string Error;
|
|
SwiftInvocationRef Invok = ASTMgr->getInvocation(Args, FileName, Error);
|
|
if (!Invok) {
|
|
LOG_WARN_FUNC("failed to create an ASTInvocation: " << Error);
|
|
Receiver(RequestResult<ExpressionTypesInFile>::fromError(Error));
|
|
return;
|
|
}
|
|
assert(Invok);
|
|
class ExpressionTypeCollector: public SwiftASTConsumer {
|
|
std::function<void(const RequestResult<ExpressionTypesInFile> &)> Receiver;
|
|
std::vector<const char *> ExpectedProtocols;
|
|
bool CanonicalType;
|
|
public:
|
|
ExpressionTypeCollector(
|
|
std::function<void(const RequestResult<ExpressionTypesInFile> &)> Receiver,
|
|
ArrayRef<const char *> ExpectedProtocols,
|
|
bool CanonicalType):
|
|
Receiver(std::move(Receiver)),
|
|
ExpectedProtocols(ExpectedProtocols.vec()),
|
|
CanonicalType(CanonicalType) {}
|
|
void handlePrimaryAST(ASTUnitRef AstUnit) override {
|
|
auto *SF = AstUnit->getCompilerInstance().getPrimarySourceFile();
|
|
std::vector<ExpressionTypeInfo> Scratch;
|
|
llvm::SmallString<256> TypeBuffer;
|
|
llvm::raw_svector_ostream OS(TypeBuffer);
|
|
ExpressionTypesInFile Result;
|
|
for (auto Item: collectExpressionType(*SF, ExpectedProtocols, Scratch,
|
|
CanonicalType, OS)) {
|
|
Result.Results.push_back({Item.offset, Item.length, Item.typeOffset, {}});
|
|
for (auto P: Item.protocols) {
|
|
Result.Results.back().ProtocolOffsets.push_back(P.first);
|
|
}
|
|
}
|
|
Result.TypeBuffer = OS.str();
|
|
Receiver(RequestResult<ExpressionTypesInFile>::fromResult(Result));
|
|
}
|
|
|
|
void cancelled() override {
|
|
Receiver(RequestResult<ExpressionTypesInFile>::cancelled());
|
|
}
|
|
|
|
void failed(StringRef Error) override {
|
|
Receiver(RequestResult<ExpressionTypesInFile>::fromError(Error));
|
|
}
|
|
};
|
|
auto Collector = std::make_shared<ExpressionTypeCollector>(Receiver,
|
|
ExpectedProtocols,
|
|
CanonicalType);
|
|
/// FIXME: When request cancellation is implemented and Xcode adopts it,
|
|
/// don't use 'OncePerASTToken'.
|
|
static const char OncePerASTToken = 0;
|
|
getASTManager()->processASTAsync(Invok, std::move(Collector),
|
|
&OncePerASTToken,
|
|
llvm::vfs::getRealFileSystem());
|
|
}
|