mirror of
https://github.com/apple/swift.git
synced 2025-12-14 20:36:38 +01:00
The logic here was unnecessary, the SourceRange gets converted to a CharSourceRange in the function being called, which handles raw identifiers correctly. 6.2-only since this is already fixed on main. rdar://152273926
1065 lines
36 KiB
C++
1065 lines
36 KiB
C++
//===--- SourceEntityWalker.cpp - Routines for semantic source info -------===//
|
|
//
|
|
// 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 "swift/AST/ASTContext.h"
|
|
#include "swift/AST/ASTWalker.h"
|
|
#include "swift/AST/Decl.h"
|
|
#include "swift/AST/Expr.h"
|
|
#include "swift/AST/Module.h"
|
|
#include "swift/AST/Pattern.h"
|
|
#include "swift/AST/ParameterList.h"
|
|
#include "swift/AST/SourceFile.h"
|
|
#include "swift/AST/Stmt.h"
|
|
#include "swift/AST/TypeCheckRequests.h"
|
|
#include "swift/AST/TypeRepr.h"
|
|
#include "swift/AST/Types.h"
|
|
#include "swift/Basic/Assertions.h"
|
|
#include "swift/Basic/Defer.h"
|
|
#include "swift/Basic/SourceManager.h"
|
|
#include "swift/Parse/Lexer.h"
|
|
#include "clang/Basic/Module.h"
|
|
#include "swift/IDE/SourceEntityWalker.h"
|
|
#include "swift/IDE/Utils.h"
|
|
|
|
using namespace swift;
|
|
|
|
namespace {
|
|
|
|
class SemaAnnotator : public ASTWalker {
|
|
SourceEntityWalker &SEWalker;
|
|
SmallVector<ConstructorRefCallExpr *, 2> CtorRefs;
|
|
SmallVector<ExtensionDecl *, 2> ExtDecls;
|
|
llvm::SmallDenseMap<OpaqueValueExpr *, Expr *, 4> OpaqueValueMap;
|
|
llvm::SmallPtrSet<Expr *, 16> ExprsToSkip;
|
|
std::optional<AccessKind> OpAccess;
|
|
|
|
public:
|
|
explicit SemaAnnotator(SourceEntityWalker &SEWalker)
|
|
: SEWalker(SEWalker) { }
|
|
|
|
private:
|
|
|
|
// FIXME: Remove this
|
|
bool shouldWalkAccessorsTheOldWay() override { return true; }
|
|
|
|
bool shouldWalkIntoGenericParams() override {
|
|
return SEWalker.shouldWalkIntoGenericParams();
|
|
}
|
|
|
|
bool shouldWalkSerializedTopLevelInternalDecls() override {
|
|
return false;
|
|
}
|
|
|
|
MacroWalking getMacroWalkingBehavior() const override {
|
|
return SEWalker.getMacroWalkingBehavior();
|
|
}
|
|
|
|
QualifiedIdentTypeReprWalkingScheme
|
|
getQualifiedIdentTypeReprWalkingScheme() const override {
|
|
return QualifiedIdentTypeReprWalkingScheme::SourceOrderRecursive;
|
|
}
|
|
|
|
PreWalkAction walkToDeclPre(Decl *D) override;
|
|
PreWalkAction walkToDeclPreProper(Decl *D);
|
|
PreWalkResult<Expr *> walkToExprPre(Expr *E) override;
|
|
PreWalkAction walkToTypeReprPre(TypeRepr *T) override;
|
|
|
|
PostWalkAction walkToDeclPost(Decl *D) override;
|
|
PostWalkAction walkToDeclPostProper(Decl *D);
|
|
PostWalkResult<Expr *> walkToExprPost(Expr *E) override;
|
|
PostWalkAction walkToTypeReprPost(TypeRepr *T) override;
|
|
|
|
PreWalkResult<Stmt *> walkToStmtPre(Stmt *S) override;
|
|
PostWalkResult<Stmt *> walkToStmtPost(Stmt *S) override;
|
|
|
|
PreWalkResult<ArgumentList *>
|
|
walkToArgumentListPre(ArgumentList *ArgList) override;
|
|
|
|
PreWalkResult<Pattern *> walkToPatternPre(Pattern *P) override;
|
|
PostWalkResult<Pattern *> walkToPatternPost(Pattern *P) override;
|
|
|
|
bool handleImports(ImportDecl *Import);
|
|
bool handleCustomAttributes(Decl *D);
|
|
bool handleCustomTypeAttribute(const CustomAttr *customAttr);
|
|
bool handleClosureAttributes(ClosureExpr *E);
|
|
bool handleTypeAttributes(AttributedTypeRepr *T);
|
|
bool passModulePathElements(ImportPath::Module Path,
|
|
const clang::Module *ClangMod);
|
|
|
|
bool passReference(ValueDecl *D, Type Ty, SourceLoc Loc, SourceRange Range,
|
|
ReferenceMetaData Data);
|
|
bool passReference(ValueDecl *D, Type Ty, DeclNameLoc Loc, ReferenceMetaData Data);
|
|
bool passReference(ModuleEntity Mod, ImportPath::Element IdLoc);
|
|
|
|
bool passSubscriptReference(ValueDecl *D, SourceLoc Loc,
|
|
ReferenceMetaData Data, bool IsOpenBracket);
|
|
bool passCallAsFunctionReference(ValueDecl *D, SourceLoc Loc,
|
|
ReferenceMetaData Data);
|
|
|
|
bool passCallArgNames(Expr *Fn, ArgumentList *ArgList);
|
|
|
|
bool shouldIgnore(Decl *D);
|
|
};
|
|
|
|
} // end anonymous namespace
|
|
|
|
ASTWalker::PreWalkAction SemaAnnotator::walkToDeclPre(Decl *D) {
|
|
if (shouldIgnore(D)) {
|
|
// If we return true here, the children will still be visited, but we won't
|
|
// call walkToDeclPre on SEWalker. The corresponding walkToDeclPost call
|
|
// on SEWalker will be prevented by the check for shouldIgnore in
|
|
// walkToDeclPost in SemaAnnotator.
|
|
return Action::VisitNodeIf(isa<PatternBindingDecl>(D));
|
|
}
|
|
|
|
SEWalker.beginBalancedASTOrderDeclVisit(D);
|
|
auto Result = walkToDeclPreProper(D);
|
|
|
|
if (Result.Action != PreWalkAction::Continue) {
|
|
// To satisfy the contract of balanced calls to
|
|
// begin/endBalancedASTOrderDeclVisit, we must call
|
|
// endBalancedASTOrderDeclVisit here if walkToDeclPost isn't going to be
|
|
// called.
|
|
SEWalker.endBalancedASTOrderDeclVisit(D);
|
|
}
|
|
|
|
return Result;
|
|
}
|
|
|
|
ASTWalker::PreWalkAction SemaAnnotator::walkToDeclPreProper(Decl *D) {
|
|
if (!handleCustomAttributes(D))
|
|
return Action::Stop();
|
|
|
|
SourceLoc Loc = D->getLoc();
|
|
unsigned NameLen = 0;
|
|
bool IsExtension = false;
|
|
|
|
if (auto *VD = dyn_cast<ValueDecl>(D)) {
|
|
if (!VD->isImplicit()) {
|
|
SourceManager &SM = VD->getASTContext().SourceMgr;
|
|
if (VD->hasName()) {
|
|
NameLen = VD->getBaseName().userFacingName().size();
|
|
if (Loc.isValid() && SM.extractText({Loc, 1}) == "`")
|
|
NameLen += 2;
|
|
} else if (Loc.isValid() && SM.extractText({Loc, 1}) == "_") {
|
|
NameLen = 1;
|
|
}
|
|
}
|
|
|
|
auto ReportParamList = [&](ParameterList *PL) {
|
|
for (auto *PD : *PL) {
|
|
auto Loc = PD->getArgumentNameLoc();
|
|
if (Loc.isInvalid())
|
|
continue;
|
|
if (!SEWalker.visitDeclarationArgumentName(PD->getArgumentName(), Loc,
|
|
VD)) {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
};
|
|
|
|
if (isa<AbstractFunctionDecl>(VD) || isa<SubscriptDecl>(VD)) {
|
|
auto ParamList = VD->getParameterList();
|
|
if (!ReportParamList(ParamList))
|
|
return Action::Stop();
|
|
}
|
|
|
|
if (auto proto = dyn_cast<ProtocolDecl>(VD)) {
|
|
// Report a primary associated type as a references to the associated type
|
|
// declaration.
|
|
for (auto parsedName : proto->getPrimaryAssociatedTypeNames()) {
|
|
Identifier name = parsedName.first;
|
|
SourceLoc loc = parsedName.second;
|
|
if (auto assocTypeDecl = proto->getAssociatedType(name)) {
|
|
auto Continue = passReference(
|
|
assocTypeDecl, assocTypeDecl->getInterfaceType(),
|
|
DeclNameLoc(loc),
|
|
ReferenceMetaData(SemaReferenceKind::TypeRef, std::nullopt));
|
|
if (!Continue)
|
|
return Action::Stop();
|
|
}
|
|
}
|
|
}
|
|
} else if (auto *ED = dyn_cast<ExtensionDecl>(D)) {
|
|
SourceRange SR = SourceRange();
|
|
if (auto *repr = ED->getExtendedTypeRepr())
|
|
SR = repr->getSourceRange();
|
|
Loc = SR.Start;
|
|
if (Loc.isValid())
|
|
NameLen = ED->getASTContext().SourceMgr.getByteDistance(SR.Start, SR.End);
|
|
IsExtension = true;
|
|
} else if (auto Import = dyn_cast<ImportDecl>(D)) {
|
|
if (!handleImports(Import))
|
|
return Action::Stop();
|
|
|
|
} else if (auto OpD = dyn_cast<OperatorDecl>(D)) {
|
|
Loc = OpD->getLoc();
|
|
if (Loc.isValid())
|
|
NameLen = OpD->getName().getLength();
|
|
|
|
} else if (auto PrecD = dyn_cast<PrecedenceGroupDecl>(D)) {
|
|
Loc = PrecD->getLoc();
|
|
if (Loc.isValid())
|
|
NameLen = PrecD->getName().getLength();
|
|
} else if (auto *MD = dyn_cast<MacroExpansionDecl>(D)) {
|
|
if (auto *macro =
|
|
dyn_cast_or_null<MacroDecl>(MD->getMacroRef().getDecl())) {
|
|
auto macroRefType = macro->getDeclaredInterfaceType();
|
|
if (!passReference(
|
|
macro, macroRefType, MD->getMacroNameLoc(),
|
|
ReferenceMetaData(SemaReferenceKind::DeclRef, std::nullopt)))
|
|
return Action::Stop();
|
|
}
|
|
}
|
|
|
|
CharSourceRange Range = (Loc.isValid()) ? CharSourceRange(Loc, NameLen)
|
|
: CharSourceRange();
|
|
bool ShouldVisitChildren = SEWalker.walkToDeclPre(D, Range);
|
|
// walkToDeclPost is only called when visiting children, so make sure to only
|
|
// push the extension decl in that case (otherwise it won't be popped)
|
|
if (IsExtension && ShouldVisitChildren) {
|
|
ExtDecls.push_back(static_cast<ExtensionDecl*>(D));
|
|
}
|
|
return Action::VisitNodeIf(ShouldVisitChildren);
|
|
}
|
|
|
|
ASTWalker::PostWalkAction SemaAnnotator::walkToDeclPost(Decl *D) {
|
|
auto Action = walkToDeclPostProper(D);
|
|
SEWalker.endBalancedASTOrderDeclVisit(D);
|
|
|
|
if (Action.Action == PostWalkAction::Stop)
|
|
return Action;
|
|
|
|
// Walk into peer and conformance expansions if walking expansions
|
|
if (shouldWalkMacroArgumentsAndExpansion().second) {
|
|
D->visitAuxiliaryDecls([&](Decl *auxDecl) {
|
|
if (Action.Action == PostWalkAction::Stop)
|
|
return;
|
|
|
|
if (auxDecl->walk(*this)) {
|
|
Action = Action::Stop();
|
|
}
|
|
}, /*visitFreestandingExpanded=*/false);
|
|
}
|
|
|
|
return Action;
|
|
}
|
|
|
|
ASTWalker::PostWalkAction SemaAnnotator::walkToDeclPostProper(Decl *D) {
|
|
if (shouldIgnore(D))
|
|
return Action::Continue();
|
|
|
|
if (isa<ExtensionDecl>(D)) {
|
|
assert(ExtDecls.back() == D);
|
|
ExtDecls.pop_back();
|
|
}
|
|
|
|
bool Continue = SEWalker.walkToDeclPost(D);
|
|
return Action::StopIf(!Continue);
|
|
}
|
|
|
|
ASTWalker::PreWalkResult<Stmt *> SemaAnnotator::walkToStmtPre(Stmt *S) {
|
|
bool TraverseChildren = SEWalker.walkToStmtPre(S);
|
|
if (TraverseChildren) {
|
|
if (auto *DeferS = dyn_cast<DeferStmt>(S)) {
|
|
// Since 'DeferStmt::getTempDecl()' is marked as implicit, we manually
|
|
// walk into the body.
|
|
if (auto *FD = DeferS->getTempDecl()) {
|
|
auto *Body = FD->getBody();
|
|
if (!Body)
|
|
return Action::Stop();
|
|
|
|
auto *RetS = Body->walk(*this);
|
|
if (!RetS)
|
|
return Action::Stop();
|
|
assert(RetS == Body);
|
|
}
|
|
bool Continue = SEWalker.walkToStmtPost(DeferS);
|
|
if (!Continue)
|
|
return Action::Stop();
|
|
|
|
// Already walked children.
|
|
return Action::SkipNode(DeferS);
|
|
}
|
|
}
|
|
return Action::VisitNodeIf(TraverseChildren, S);
|
|
}
|
|
|
|
ASTWalker::PostWalkResult<Stmt *> SemaAnnotator::walkToStmtPost(Stmt *S) {
|
|
bool Continue = SEWalker.walkToStmtPost(S);
|
|
return Action::StopIf(!Continue, S);
|
|
}
|
|
|
|
static SemaReferenceKind getReferenceKind(Expr *Parent, Expr *E) {
|
|
if (auto SA = dyn_cast_or_null<SelfApplyExpr>(Parent)) {
|
|
if (SA->getFn() == E)
|
|
return SemaReferenceKind::DeclMemberRef;
|
|
}
|
|
return SemaReferenceKind::DeclRef;
|
|
}
|
|
|
|
ASTWalker::PreWalkResult<ArgumentList *>
|
|
SemaAnnotator::walkToArgumentListPre(ArgumentList *ArgList) {
|
|
// Don't consider the argument labels for an implicit ArgumentList.
|
|
if (ArgList->isImplicit())
|
|
return Action::Continue(ArgList);
|
|
|
|
// FIXME(https://github.com/apple/swift/issues/57390): What about SubscriptExpr and KeyPathExpr arg labels?
|
|
if (auto CallE = dyn_cast_or_null<CallExpr>(Parent.getAsExpr())) {
|
|
if (!passCallArgNames(CallE->getFn(), ArgList))
|
|
return Action::Stop();
|
|
}
|
|
return Action::Continue(ArgList);
|
|
}
|
|
|
|
ASTWalker::PreWalkResult<Expr *> SemaAnnotator::walkToExprPre(Expr *E) {
|
|
assert(E);
|
|
|
|
if (ExprsToSkip.count(E) != 0) {
|
|
// We are skipping the expression. Call neither walkToExprPr nor
|
|
// walkToExprPost on it
|
|
return Action::SkipNode(E);
|
|
}
|
|
|
|
// Skip the synthesized curry thunks and just walk over the unwrapped
|
|
// expression
|
|
if (auto *ACE = dyn_cast<AutoClosureExpr>(E)) {
|
|
if (auto *SubExpr = ACE->getUnwrappedCurryThunkExpr()) {
|
|
if (!SubExpr->walk(*this))
|
|
return Action::Stop();
|
|
|
|
return Action::SkipNode(E);
|
|
}
|
|
}
|
|
|
|
if (!SEWalker.walkToExprPre(E)) {
|
|
return Action::SkipNode(E);
|
|
}
|
|
|
|
if (auto *CtorRefE = dyn_cast<ConstructorRefCallExpr>(E))
|
|
CtorRefs.push_back(CtorRefE);
|
|
|
|
if (auto *DRE = dyn_cast<DeclRefExpr>(E)) {
|
|
auto *FD = dyn_cast<FuncDecl>(DRE->getDecl());
|
|
// Handle implicit callAsFunction reference. An explicit reference will be
|
|
// handled by the usual DeclRefExpr case below.
|
|
if (DRE->isImplicit() && FD && FD->isCallAsFunctionMethod()) {
|
|
ReferenceMetaData data(SemaReferenceKind::DeclMemberRef, OpAccess);
|
|
if (!passCallAsFunctionReference(FD, DRE->getLoc(), data))
|
|
return Action::Stop();
|
|
|
|
return Action::Continue(E);
|
|
}
|
|
}
|
|
|
|
if (!isa<InOutExpr>(E) && !isa<LoadExpr>(E) && !isa<OpenExistentialExpr>(E) &&
|
|
!isa<MakeTemporarilyEscapableExpr>(E) &&
|
|
!isa<CollectionUpcastConversionExpr>(E) && !isa<OpaqueValueExpr>(E) &&
|
|
!isa<SubscriptExpr>(E) && !isa<KeyPathExpr>(E) && !isa<LiteralExpr>(E) &&
|
|
!isa<CollectionExpr>(E) && E->isImplicit())
|
|
return Action::Continue(E);
|
|
|
|
if (auto LE = dyn_cast<LiteralExpr>(E)) {
|
|
if (LE->getInitializer() &&
|
|
!passReference(LE->getInitializer().getDecl(), LE->getType(), {},
|
|
LE->getSourceRange(),
|
|
ReferenceMetaData(SemaReferenceKind::DeclRef, OpAccess,
|
|
/*isImplicit=*/true))) {
|
|
return Action::Stop();
|
|
}
|
|
return Action::Continue(E);
|
|
} else if (auto CE = dyn_cast<CollectionExpr>(E)) {
|
|
if (CE->getInitializer() &&
|
|
!passReference(CE->getInitializer().getDecl(), CE->getType(), {},
|
|
CE->getSourceRange(),
|
|
ReferenceMetaData(SemaReferenceKind::DeclRef, OpAccess,
|
|
/*isImplicit=*/true))) {
|
|
return Action::Stop();
|
|
}
|
|
return Action::Continue(E);
|
|
} else if (auto *DRE = dyn_cast<DeclRefExpr>(E)) {
|
|
if (auto *module = dyn_cast<ModuleDecl>(DRE->getDecl())) {
|
|
if (!passReference(ModuleEntity(module),
|
|
{module->getName(), E->getLoc()}))
|
|
return Action::Stop();
|
|
} else if (!passReference(DRE->getDecl(), DRE->getType(),
|
|
DRE->getNameLoc(),
|
|
ReferenceMetaData(getReferenceKind(Parent.getAsExpr(), DRE),
|
|
OpAccess))) {
|
|
return Action::Stop();
|
|
}
|
|
} else if (auto *MRE = dyn_cast<MemberRefExpr>(E)) {
|
|
{
|
|
// This could be made more accurate if the member is nonmutating,
|
|
// or whatever.
|
|
std::optional<AccessKind> NewOpAccess;
|
|
if (OpAccess) {
|
|
if (*OpAccess == AccessKind::Write)
|
|
NewOpAccess = AccessKind::ReadWrite;
|
|
else
|
|
NewOpAccess = OpAccess;
|
|
}
|
|
|
|
llvm::SaveAndRestore<std::optional<AccessKind>> C(this->OpAccess,
|
|
NewOpAccess);
|
|
|
|
// Visit in source order.
|
|
if (!MRE->getBase()->walk(*this))
|
|
return Action::Stop();
|
|
}
|
|
|
|
if (!passReference(MRE->getMember().getDecl(), MRE->getType(),
|
|
MRE->getNameLoc(),
|
|
ReferenceMetaData(SemaReferenceKind::DeclMemberRef,
|
|
OpAccess))) {
|
|
return Action::Stop();
|
|
}
|
|
// We already visited the children.
|
|
return Action::SkipChildren(E);
|
|
|
|
} else if (auto OtherCtorE = dyn_cast<OtherConstructorDeclRefExpr>(E)) {
|
|
if (!passReference(OtherCtorE->getDecl(), OtherCtorE->getType(),
|
|
OtherCtorE->getConstructorLoc(),
|
|
ReferenceMetaData(SemaReferenceKind::DeclConstructorRef,
|
|
OpAccess)))
|
|
return Action::Stop();
|
|
|
|
} else if (auto *SE = dyn_cast<SubscriptExpr>(E)) {
|
|
// Visit in source order.
|
|
if (!SE->getBase()->walk(*this))
|
|
return Action::Stop();
|
|
|
|
ValueDecl *SubscrD = nullptr;
|
|
if (SE->hasDecl())
|
|
SubscrD = SE->getDecl().getDecl();
|
|
|
|
ReferenceMetaData data(SemaReferenceKind::SubscriptRef, OpAccess,
|
|
SE->isImplicit());
|
|
|
|
if (SubscrD) {
|
|
if (!passSubscriptReference(SubscrD, E->getLoc(), data, true))
|
|
return Action::Stop();
|
|
}
|
|
|
|
if (!SE->getArgs()->walk(*this))
|
|
return Action::Stop();
|
|
|
|
if (SubscrD) {
|
|
if (!passSubscriptReference(SubscrD, E->getEndLoc(), data, false))
|
|
return Action::Stop();
|
|
}
|
|
|
|
// We already visited the children.
|
|
return Action::SkipChildren(E);
|
|
|
|
} else if (auto *KPE = dyn_cast<KeyPathExpr>(E)) {
|
|
for (auto &component : KPE->getComponents()) {
|
|
switch (component.getKind()) {
|
|
case KeyPathExpr::Component::Kind::Member:
|
|
case KeyPathExpr::Component::Kind::Subscript: {
|
|
auto *decl = component.getDeclRef().getDecl();
|
|
auto loc = component.getLoc();
|
|
SourceRange range(loc, loc);
|
|
auto Continue = passReference(
|
|
decl, component.getComponentType(), loc, range,
|
|
ReferenceMetaData((isa<SubscriptDecl>(decl)
|
|
? SemaReferenceKind::SubscriptRef
|
|
: SemaReferenceKind::DeclMemberRef),
|
|
OpAccess));
|
|
if (!Continue)
|
|
return Action::Stop();
|
|
break;
|
|
}
|
|
|
|
case KeyPathExpr::Component::Kind::TupleElement:
|
|
case KeyPathExpr::Component::Kind::Invalid:
|
|
case KeyPathExpr::Component::Kind::UnresolvedMember:
|
|
case KeyPathExpr::Component::Kind::UnresolvedSubscript:
|
|
case KeyPathExpr::Component::Kind::UnresolvedApply:
|
|
case KeyPathExpr::Component::Kind::Apply:
|
|
case KeyPathExpr::Component::Kind::OptionalChain:
|
|
case KeyPathExpr::Component::Kind::OptionalWrap:
|
|
case KeyPathExpr::Component::Kind::OptionalForce:
|
|
case KeyPathExpr::Component::Kind::Identity:
|
|
case KeyPathExpr::Component::Kind::DictionaryKey:
|
|
case KeyPathExpr::Component::Kind::CodeCompletion:
|
|
break;
|
|
}
|
|
}
|
|
} else if (auto *BinE = dyn_cast<BinaryExpr>(E)) {
|
|
// Visit in source order.
|
|
if (!BinE->getLHS()->walk(*this))
|
|
return Action::Stop();
|
|
if (!BinE->getFn()->walk(*this))
|
|
return Action::Stop();
|
|
if (!BinE->getRHS()->walk(*this))
|
|
return Action::Stop();
|
|
|
|
// We already visited the children.
|
|
return Action::SkipChildren(E);
|
|
} else if (auto IOE = dyn_cast<InOutExpr>(E)) {
|
|
llvm::SaveAndRestore<std::optional<AccessKind>> C(this->OpAccess,
|
|
AccessKind::ReadWrite);
|
|
|
|
if (!IOE->getSubExpr()->walk(*this))
|
|
return Action::Stop();
|
|
|
|
// We already visited the children.
|
|
return Action::SkipChildren(E);
|
|
} else if (auto LE = dyn_cast<LoadExpr>(E)) {
|
|
llvm::SaveAndRestore<std::optional<AccessKind>> C(this->OpAccess,
|
|
AccessKind::Read);
|
|
|
|
if (!LE->getSubExpr()->walk(*this))
|
|
return Action::Stop();
|
|
|
|
// We already visited the children.
|
|
return Action::SkipChildren(E);
|
|
} else if (auto AE = dyn_cast<AssignExpr>(E)) {
|
|
{
|
|
llvm::SaveAndRestore<std::optional<AccessKind>> C(this->OpAccess,
|
|
AccessKind::Write);
|
|
|
|
if (AE->getDest() && !AE->getDest()->walk(*this))
|
|
return Action::Stop();
|
|
}
|
|
|
|
if (AE->getSrc() && !AE->getSrc()->walk(*this))
|
|
return Action::Stop();
|
|
|
|
// We already visited the children.
|
|
return Action::SkipChildren(E);
|
|
} else if (auto OEE = dyn_cast<OpenExistentialExpr>(E)) {
|
|
// Record opaque value.
|
|
OpaqueValueMap[OEE->getOpaqueValue()] = OEE->getExistentialValue();
|
|
SWIFT_DEFER {
|
|
OpaqueValueMap.erase(OEE->getOpaqueValue());
|
|
};
|
|
|
|
if (!OEE->getSubExpr()->walk(*this))
|
|
return Action::Stop();
|
|
|
|
return Action::SkipChildren(E);
|
|
} else if (auto MTEE = dyn_cast<MakeTemporarilyEscapableExpr>(E)) {
|
|
// Manually walk to original arguments in order. We don't handle
|
|
// OpaqueValueExpr here.
|
|
|
|
// Original non-escaping closure.
|
|
if (!MTEE->getNonescapingClosureValue()->walk(*this))
|
|
return Action::Stop();
|
|
|
|
// Body, which is called by synthesized CallExpr.
|
|
auto *callExpr = cast<CallExpr>(MTEE->getSubExpr());
|
|
if (!callExpr->getFn()->walk(*this))
|
|
return Action::Stop();
|
|
|
|
return Action::SkipChildren(E);
|
|
} else if (auto CUCE = dyn_cast<CollectionUpcastConversionExpr>(E)) {
|
|
// Ignore conversion expressions. We don't handle OpaqueValueExpr here
|
|
// because it's only in conversion expressions. Instead, just walk into
|
|
// sub expression.
|
|
if (!CUCE->getSubExpr()->walk(*this))
|
|
return Action::Stop();
|
|
|
|
return Action::SkipChildren(E);
|
|
} else if (auto OVE = dyn_cast<OpaqueValueExpr>(E)) {
|
|
// Walk into mapped value.
|
|
auto value = OpaqueValueMap.find(OVE);
|
|
if (value != OpaqueValueMap.end()) {
|
|
if (!value->second->walk(*this))
|
|
return Action::Stop();
|
|
|
|
return Action::SkipChildren(E);
|
|
}
|
|
} else if (auto DMRE = dyn_cast<DynamicMemberRefExpr>(E)) {
|
|
// Visit in source order.
|
|
if (!DMRE->getBase()->walk(*this))
|
|
return Action::Stop();
|
|
if (!passReference(DMRE->getMember().getDecl(), DMRE->getType(),
|
|
DMRE->getNameLoc(),
|
|
ReferenceMetaData(SemaReferenceKind::DynamicMemberRef,
|
|
OpAccess))) {
|
|
return Action::Stop();
|
|
}
|
|
// We already visited the children.
|
|
return Action::SkipChildren(E);
|
|
} else if (auto ME = dyn_cast<MacroExpansionExpr>(E)) {
|
|
// Add a reference to the macro if this is a true macro expansion *expression*.
|
|
// If this is a `MacroExpansionExpr` that expands a declaration macro, the
|
|
// substitute decl will be visited by ASTWalker and we would be passing its
|
|
// reference if we didn't have this check.
|
|
if (!ME->getSubstituteDecl()) {
|
|
auto macroRef = ME->getMacroRef();
|
|
if (auto *macroDecl = dyn_cast_or_null<MacroDecl>(macroRef.getDecl())) {
|
|
auto macroRefType = macroDecl->getDeclaredInterfaceType();
|
|
if (!passReference(
|
|
macroDecl, macroRefType, ME->getMacroNameLoc(),
|
|
ReferenceMetaData(SemaReferenceKind::DeclRef, std::nullopt)))
|
|
return Action::Stop();
|
|
}
|
|
}
|
|
} else if (auto CE = dyn_cast<ClosureExpr>(E)) {
|
|
if (!handleClosureAttributes(CE))
|
|
return Action::Stop();
|
|
return Action::Continue(E);
|
|
}
|
|
|
|
return Action::Continue(E);
|
|
}
|
|
|
|
ASTWalker::PostWalkResult<Expr *> SemaAnnotator::walkToExprPost(Expr *E) {
|
|
if (isa<ConstructorRefCallExpr>(E)) {
|
|
assert(CtorRefs.back() == E);
|
|
CtorRefs.pop_back();
|
|
}
|
|
|
|
bool Continue = SEWalker.walkToExprPost(E);
|
|
return Action::StopIf(!Continue, E);
|
|
}
|
|
|
|
ASTWalker::PreWalkAction SemaAnnotator::walkToTypeReprPre(TypeRepr *T) {
|
|
bool Continue = SEWalker.walkToTypeReprPre(T);
|
|
if (!Continue)
|
|
return Action::Stop();
|
|
|
|
if (auto *DeclRefT = dyn_cast<DeclRefTypeRepr>(T)) {
|
|
if (ValueDecl *VD = DeclRefT->getBoundDecl()) {
|
|
if (auto *ModD = dyn_cast<ModuleDecl>(VD)) {
|
|
auto ident = DeclRefT->getNameRef().getBaseIdentifier();
|
|
auto Continue = passReference(ModD, {ident, DeclRefT->getLoc()});
|
|
return Action::StopIf(!Continue);
|
|
}
|
|
auto Continue = passReference(
|
|
VD, Type(), DeclRefT->getNameLoc(),
|
|
ReferenceMetaData(SemaReferenceKind::TypeRef, std::nullopt));
|
|
return Action::StopIf(!Continue);
|
|
}
|
|
} else if (auto FT = dyn_cast<FixedTypeRepr>(T)) {
|
|
if (ValueDecl *VD = FT->getType()->getAnyGeneric()) {
|
|
auto Data = ReferenceMetaData(SemaReferenceKind::TypeRef, std::nullopt);
|
|
Data.isImplicitCtorType = true;
|
|
auto Continue = passReference(VD, FT->getType(), FT->getLoc(),
|
|
FT->getSourceRange(), Data);
|
|
return Action::StopIf(!Continue);
|
|
}
|
|
} else if (auto ST = dyn_cast<SelfTypeRepr>(T)) {
|
|
ValueDecl *VD = ST->getType()->getAnyGeneric();
|
|
if (auto DT = ST->getType()->getAs<DynamicSelfType>())
|
|
VD = DT->getSelfType()->getAnyGeneric();
|
|
|
|
if (VD) {
|
|
auto Data = ReferenceMetaData(SemaReferenceKind::TypeRef, std::nullopt);
|
|
Data.isImplicitCtorType = true;
|
|
auto Continue = passReference(VD, ST->getType(), ST->getLoc(),
|
|
ST->getSourceRange(), Data);
|
|
return Action::StopIf(!Continue);
|
|
}
|
|
} else if (auto AT = dyn_cast<AttributedTypeRepr>(T)) {
|
|
auto Continue = handleTypeAttributes(AT);
|
|
return Action::StopIf(!Continue);
|
|
}
|
|
|
|
return Action::Continue();
|
|
}
|
|
|
|
ASTWalker::PostWalkAction SemaAnnotator::walkToTypeReprPost(TypeRepr *T) {
|
|
bool Continue = SEWalker.walkToTypeReprPost(T);
|
|
return Action::StopIf(!Continue);
|
|
}
|
|
|
|
ASTWalker::PreWalkResult<Pattern *>
|
|
SemaAnnotator::walkToPatternPre(Pattern *P) {
|
|
if (!SEWalker.walkToPatternPre(P))
|
|
return Action::SkipNode(P);
|
|
|
|
if (P->isImplicit())
|
|
return Action::Continue(P);
|
|
|
|
if (auto *EP = dyn_cast<EnumElementPattern>(P)) {
|
|
auto *Element = EP->getElementDecl();
|
|
if (!Element)
|
|
return Action::Continue(P);
|
|
Type T = EP->hasType() ? EP->getType() : Type();
|
|
auto Continue = passReference(
|
|
Element, T, DeclNameLoc(EP->getLoc()),
|
|
ReferenceMetaData(SemaReferenceKind::EnumElementRef, std::nullopt));
|
|
return Action::StopIf(!Continue, P);
|
|
}
|
|
|
|
auto *TP = dyn_cast<TypedPattern>(P);
|
|
if (!TP || !TP->isPropagatedType())
|
|
return Action::Continue(P);
|
|
|
|
// If the typed pattern was propagated from somewhere, just walk the
|
|
// subpattern. The type will be walked as a part of another TypedPattern.
|
|
TP->getSubPattern()->walk(*this);
|
|
return Action::SkipNode(P);
|
|
}
|
|
|
|
ASTWalker::PostWalkResult<Pattern *>
|
|
SemaAnnotator::walkToPatternPost(Pattern *P) {
|
|
bool Continue = SEWalker.walkToPatternPost(P);
|
|
return Action::StopIf(!Continue, P);
|
|
}
|
|
|
|
bool SemaAnnotator::handleCustomAttributes(Decl *D) {
|
|
// CustomAttrs of non-param VarDecls are handled when this method is called
|
|
// on their containing PatternBindingDecls (see below).
|
|
if (isa<VarDecl>(D) && !isa<ParamDecl>(D))
|
|
return true;
|
|
|
|
if (auto *PBD = dyn_cast<PatternBindingDecl>(D)) {
|
|
if (auto *SingleVar = PBD->getSingleVar()) {
|
|
D = SingleVar;
|
|
} else {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
ModuleDecl *MD = D->getModuleContext();
|
|
for (auto *customAttr :
|
|
D->getSemanticAttrs().getAttributes<CustomAttr, true>()) {
|
|
SourceFile *SF =
|
|
MD->getSourceFileContainingLocation(customAttr->getLocation());
|
|
ASTNode expansion = SF ? SF->getMacroExpansion() : nullptr;
|
|
if (!shouldWalkMacroArgumentsAndExpansion().second && expansion)
|
|
continue;
|
|
|
|
if (auto *Repr = customAttr->getTypeRepr()) {
|
|
// It's a little weird that attached macros have a `TypeRepr` to begin
|
|
// with, but given they aren't types they then don't get bound. So check
|
|
// for a macro here and and pass a reference to it.
|
|
auto *mutableAttr = const_cast<CustomAttr *>(customAttr);
|
|
if (auto macroDecl = D->getResolvedMacro(mutableAttr)) {
|
|
Type macroRefType = macroDecl->getDeclaredInterfaceType();
|
|
auto customAttrRef =
|
|
std::make_pair(customAttr, expansion ? expansion.get<Decl *>() : D);
|
|
auto refMetadata =
|
|
ReferenceMetaData(SemaReferenceKind::DeclRef, std::nullopt,
|
|
/*isImplicit=*/false, customAttrRef);
|
|
if (!passReference(macroDecl, macroRefType,
|
|
DeclNameLoc(Repr->getStartLoc()), refMetadata))
|
|
return false;
|
|
}
|
|
|
|
if (!Repr->walk(*this))
|
|
return false;
|
|
}
|
|
|
|
if (auto *SemaInit = customAttr->getSemanticInit()) {
|
|
if (!SemaInit->isImplicit()) {
|
|
assert(customAttr->hasArgs());
|
|
if (!SemaInit->walk(*this))
|
|
return false;
|
|
// Don't walk this again via the associated PatternBindingDecl's
|
|
// initializer
|
|
ExprsToSkip.insert(SemaInit);
|
|
}
|
|
} else if (auto *Args = customAttr->getArgs()) {
|
|
if (!Args->walk(*this))
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool SemaAnnotator::handleCustomTypeAttribute(const CustomAttr *customAttr) {
|
|
if (auto *Repr = customAttr->getTypeRepr())
|
|
if (!Repr->walk(*this))
|
|
return false;
|
|
|
|
if (auto *Args = customAttr->getArgs())
|
|
if (!Args->walk(*this))
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
bool SemaAnnotator::handleClosureAttributes(ClosureExpr *E) {
|
|
for (auto *customAttr : E->getAttrs().getAttributes<CustomAttr, true>())
|
|
if (!handleCustomTypeAttribute(customAttr))
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
bool SemaAnnotator::handleTypeAttributes(AttributedTypeRepr *T) {
|
|
for (auto attr : T->getAttrs()) {
|
|
if (!attr.is<CustomAttr *>())
|
|
continue;
|
|
|
|
CustomAttr *customAttr = attr.get<CustomAttr *>();
|
|
if (!handleCustomTypeAttribute(customAttr))
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool SemaAnnotator::handleImports(ImportDecl *Import) {
|
|
auto Mod = Import->getModule();
|
|
if (!Mod)
|
|
return true;
|
|
|
|
auto ClangMod = Mod->findUnderlyingClangModule();
|
|
if (ClangMod && ClangMod->isSubModule()) {
|
|
if (!passModulePathElements(Import->getModulePath(), ClangMod))
|
|
return false;
|
|
} else {
|
|
if (!passReference(Mod, Import->getModulePath().front()))
|
|
return false;
|
|
}
|
|
|
|
auto Decls = Import->getDecls();
|
|
if (Decls.size() == 1) {
|
|
// FIXME: ImportDecl should store a DeclNameLoc.
|
|
// FIXME: Handle overloaded funcs too by passing a reference for each?
|
|
if (!passReference(
|
|
Decls.front(), Type(), DeclNameLoc(Import->getEndLoc()),
|
|
ReferenceMetaData(SemaReferenceKind::DeclRef, std::nullopt)))
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool SemaAnnotator::passModulePathElements(
|
|
ImportPath::Module Path,
|
|
const clang::Module *ClangMod) {
|
|
|
|
assert(ClangMod && "can't passModulePathElements of null ClangMod");
|
|
|
|
// Visit parent, if any, first.
|
|
if (ClangMod->Parent && Path.hasSubmodule())
|
|
if (!passModulePathElements(Path.getParentPath(), ClangMod->Parent))
|
|
return false;
|
|
|
|
return passReference(ClangMod, Path.back());
|
|
}
|
|
|
|
bool SemaAnnotator::passSubscriptReference(ValueDecl *D, SourceLoc Loc,
|
|
ReferenceMetaData Data,
|
|
bool IsOpenBracket) {
|
|
CharSourceRange Range = Loc.isValid()
|
|
? CharSourceRange(Loc, 1)
|
|
: CharSourceRange();
|
|
|
|
return SEWalker.visitSubscriptReference(D, Range, Data, IsOpenBracket);
|
|
}
|
|
|
|
bool SemaAnnotator::passCallAsFunctionReference(ValueDecl *D, SourceLoc Loc,
|
|
ReferenceMetaData Data) {
|
|
CharSourceRange Range =
|
|
Loc.isValid() ? CharSourceRange(Loc, 1) : CharSourceRange();
|
|
|
|
return SEWalker.visitCallAsFunctionReference(D, Range, Data);
|
|
}
|
|
|
|
bool SemaAnnotator::
|
|
passReference(ValueDecl *D, Type Ty, DeclNameLoc Loc, ReferenceMetaData Data) {
|
|
SourceLoc BaseLoc = Loc.getBaseNameLoc();
|
|
return passReference(D, Ty, BaseLoc, BaseLoc, Data);
|
|
}
|
|
|
|
bool SemaAnnotator::
|
|
passReference(ValueDecl *D, Type Ty, SourceLoc BaseNameLoc, SourceRange Range,
|
|
ReferenceMetaData Data) {
|
|
TypeDecl *CtorTyRef = nullptr;
|
|
ExtensionDecl *ExtDecl = nullptr;
|
|
|
|
if (auto *TD = dyn_cast<TypeDecl>(D)) {
|
|
if (!CtorRefs.empty() && BaseNameLoc.isValid()) {
|
|
ConstructorRefCallExpr *Ctor = CtorRefs.back();
|
|
SourceLoc CtorLoc = Ctor->getFn()->getLoc();
|
|
// Get the location of the type, ignoring parens, rather than the start of
|
|
// the Expr, to match the lookup.
|
|
if (auto *TE = dyn_cast<TypeExpr>(Ctor->getBase()))
|
|
CtorLoc = TE->getTypeRepr()->getWithoutParens()->getLoc();
|
|
|
|
bool isImplicit = false;
|
|
Expr *Fn = Ctor->getFn();
|
|
while (auto *ICE = dyn_cast<ImplicitConversionExpr>(Fn))
|
|
Fn = ICE->getSubExpr();
|
|
if (auto *DRE = dyn_cast<DeclRefExpr>(Fn))
|
|
isImplicit = DRE->isImplicit();
|
|
|
|
if (isImplicit && CtorLoc == BaseNameLoc) {
|
|
D = ide::getReferencedDecl(Ctor->getFn()).second.getDecl();
|
|
if (D == nullptr) {
|
|
assert(false && "Unhandled constructor reference");
|
|
return true;
|
|
}
|
|
CtorTyRef = TD;
|
|
}
|
|
}
|
|
|
|
if (!ExtDecls.empty() && BaseNameLoc.isValid()) {
|
|
SourceLoc ExtTyLoc = SourceLoc();
|
|
if (auto *repr = ExtDecls.back()->getExtendedTypeRepr())
|
|
ExtTyLoc = repr->getLoc();
|
|
if (ExtTyLoc.isValid() && ExtTyLoc == BaseNameLoc) {
|
|
ExtDecl = ExtDecls.back();
|
|
}
|
|
}
|
|
}
|
|
|
|
CharSourceRange CharRange =
|
|
Lexer::getCharSourceRangeFromSourceRange(D->getASTContext().SourceMgr,
|
|
Range);
|
|
|
|
return SEWalker.visitDeclReference(D, CharRange, CtorTyRef, ExtDecl, Ty,
|
|
Data);
|
|
}
|
|
|
|
bool SemaAnnotator::passReference(ModuleEntity Mod,
|
|
ImportPath::Element IdLoc) {
|
|
if (IdLoc.Loc.isInvalid())
|
|
return true;
|
|
unsigned NameLen = IdLoc.Item.getLength();
|
|
CharSourceRange Range{ IdLoc.Loc, NameLen };
|
|
return SEWalker.visitModuleReference(Mod, Range);
|
|
}
|
|
|
|
bool SemaAnnotator::passCallArgNames(Expr *Fn, ArgumentList *ArgList) {
|
|
ValueDecl *D = ide::getReferencedDecl(Fn).second.getDecl();
|
|
if (!D)
|
|
return true; // continue.
|
|
|
|
for (auto Arg : *ArgList) {
|
|
Identifier Name = Arg.getLabel();
|
|
if (Name.empty())
|
|
continue;
|
|
|
|
SourceLoc Loc = Arg.getLabelLoc();
|
|
if (Loc.isInvalid())
|
|
continue;
|
|
|
|
CharSourceRange Range{ Loc, Name.getLength() };
|
|
bool Continue = SEWalker.visitCallArgName(Name, Range, D);
|
|
if (!Continue)
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool SemaAnnotator::shouldIgnore(Decl *D) {
|
|
if (!D->isImplicit())
|
|
return false;
|
|
|
|
// TODO: There should really be a separate field controlling whether
|
|
// constructors are visited or not
|
|
if (isa<ConstructorDecl>(D))
|
|
return false;
|
|
|
|
// Walk into missing decls to visit their attributes if they were generated
|
|
// by a member attribute expansion. Note that we would have already skipped
|
|
// this decl if we were ignoring expansions, so no need to check that.
|
|
if (isa<MissingDecl>(D)) {
|
|
if (D->isInMacroExpansionInContext())
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool SourceEntityWalker::walk(SourceFile &SrcFile) {
|
|
SemaAnnotator Annotator(*this);
|
|
return performWalk(Annotator, [&]() { return SrcFile.walk(Annotator); });
|
|
}
|
|
|
|
bool SourceEntityWalker::walk(ModuleDecl &Mod) {
|
|
SemaAnnotator Annotator(*this);
|
|
return performWalk(Annotator, [&]() { return Mod.walk(Annotator); });
|
|
}
|
|
|
|
bool SourceEntityWalker::walk(Stmt *S) {
|
|
SemaAnnotator Annotator(*this);
|
|
return performWalk(Annotator, [&]() { return S->walk(Annotator); });
|
|
}
|
|
|
|
bool SourceEntityWalker::walk(Expr *E) {
|
|
SemaAnnotator Annotator(*this);
|
|
return performWalk(Annotator, [&]() { return E->walk(Annotator); });
|
|
}
|
|
|
|
bool SourceEntityWalker::walk(Pattern *P) {
|
|
SemaAnnotator Annotator(*this);
|
|
return performWalk(Annotator, [&]() { return P->walk(Annotator); });
|
|
}
|
|
|
|
bool SourceEntityWalker::walk(Decl *D) {
|
|
SemaAnnotator Annotator(*this);
|
|
return performWalk(Annotator, [&]() { return D->walk(Annotator); });
|
|
}
|
|
|
|
bool SourceEntityWalker::walk(DeclContext *DC) {
|
|
SemaAnnotator Annotator(*this);
|
|
return performWalk(Annotator, [&]() { return DC->walkContext(Annotator); });
|
|
}
|
|
|
|
bool SourceEntityWalker::walk(ASTNode N) {
|
|
if (auto *E = N.dyn_cast<Expr*>())
|
|
return walk(E);
|
|
if (auto *S = N.dyn_cast<Stmt*>())
|
|
return walk(S);
|
|
if (auto *D = N.dyn_cast<Decl*>())
|
|
return walk(D);
|
|
|
|
llvm_unreachable("unsupported AST node");
|
|
}
|
|
|
|
bool SourceEntityWalker::visitDeclReference(ValueDecl *D, CharSourceRange Range,
|
|
TypeDecl *CtorTyRef,
|
|
ExtensionDecl *ExtTyRef, Type T,
|
|
ReferenceMetaData Data) {
|
|
return true;
|
|
}
|
|
|
|
bool SourceEntityWalker::visitSubscriptReference(ValueDecl *D,
|
|
CharSourceRange Range,
|
|
ReferenceMetaData Data,
|
|
bool IsOpenBracket) {
|
|
// Most of the clients treat subscript reference the same way as a
|
|
// regular reference when called on the open bracket and
|
|
// ignore the closing one.
|
|
return IsOpenBracket
|
|
? visitDeclReference(D, Range, nullptr, nullptr, Type(), Data)
|
|
: true;
|
|
}
|
|
|
|
bool SourceEntityWalker::visitCallAsFunctionReference(ValueDecl *D,
|
|
CharSourceRange Range,
|
|
ReferenceMetaData Data) {
|
|
return true;
|
|
}
|
|
|
|
bool SourceEntityWalker::visitCallArgName(Identifier Name,
|
|
CharSourceRange Range,
|
|
ValueDecl *D) {
|
|
return true;
|
|
}
|
|
|
|
bool SourceEntityWalker::
|
|
visitDeclarationArgumentName(Identifier Name, SourceLoc Start, ValueDecl *D) {
|
|
return true;
|
|
}
|
|
|
|
bool SourceEntityWalker::visitModuleReference(ModuleEntity Mod,
|
|
CharSourceRange Range) {
|
|
return true;
|
|
}
|
|
|
|
void SourceEntityWalker::anchor() {}
|