forked from swiftlang/swift
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathITCNameLookup.cpp
174 lines (152 loc) · 6.53 KB
/
ITCNameLookup.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
//===--- ITCNameLookup.cpp - Iterative Type Checker Name Lookup -----------===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2015 Apple Inc. and the Swift project authors
// Licensed under Apache License v2.0 with Runtime Library Exception
//
// See http://swift.org/LICENSE.txt for license information
// See http://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
//
//===----------------------------------------------------------------------===//
//
// This file implements the portions of the IterativeTypeChecker
// class that involve name lookup.
//
//===----------------------------------------------------------------------===//
#include "TypeChecker.h"
#include "swift/Sema/IterativeTypeChecker.h"
#include "swift/AST/ASTContext.h"
#include "swift/AST/Decl.h"
#include <tuple>
using namespace swift;
//----------------------------------------------------------------------------//
// Qualified name lookup handling
//----------------------------------------------------------------------------//
bool IterativeTypeChecker::isQualifiedLookupInDeclContextSatisfied(
TypeCheckRequest::DeclContextLookupPayloadType payload) {
auto dc = payload.DC;
NominalTypeDecl *nominal = nullptr;
switch (dc->getContextKind()) {
case DeclContextKind::AbstractClosureExpr:
case DeclContextKind::AbstractFunctionDecl:
case DeclContextKind::Initializer:
case DeclContextKind::TopLevelCodeDecl:
case DeclContextKind::SerializedLocal:
llvm_unreachable("not a DeclContext that supports name lookup");
case DeclContextKind::Module:
case DeclContextKind::FileUnit:
// Modules and file units can always handle name lookup.
return true;
case DeclContextKind::NominalTypeDecl:
// Get the nominal type.
nominal = cast<NominalTypeDecl>(dc);
break;
case DeclContextKind::ExtensionDecl: {
auto ext = cast<ExtensionDecl>(dc);
// FIXME: bind the extension. We currently assume this is done.
nominal = ext->isNominalTypeOrNominalTypeExtensionContext();
if (!nominal) return true;
break;
}
}
assert(nominal && "Only nominal types are handled here");
// FIXME: Cache a bit indicating when qualified lookup is possible.
// If we needed them for this query, did we already add implicit
// initializers?
auto name = payload.Name;
if ((!name || name.matchesRef(getASTContext().Id_init)) &&
!nominal->addedImplicitInitializers())
return false;
// For classes, check the superclass.
if (auto classDecl = dyn_cast<ClassDecl>(nominal)) {
if (!isSatisfied(requestTypeCheckSuperclass(classDecl)))
return false;
if (auto superclass = classDecl->getSuperclass()) {
if (auto superclassDecl = superclass->getAnyNominal()) {
if (!isSatisfied(requestQualifiedLookupInDeclContext({ superclassDecl,
payload.Name,
payload.Loc })))
return false;
}
}
}
return true;
}
void IterativeTypeChecker::processQualifiedLookupInDeclContext(
TypeCheckRequest::DeclContextLookupPayloadType payload,
UnsatisfiedDependency unsatisfiedDependency) {
auto nominal = payload.DC->isNominalTypeOrNominalTypeExtensionContext();
assert(nominal && "Only nominal types are handled here");
// For classes, we need the superclass (if any) to support qualified lookup.
if (auto classDecl = dyn_cast<ClassDecl>(nominal)) {
if (unsatisfiedDependency(requestTypeCheckSuperclass(classDecl)))
return;
if (auto superclass = classDecl->getSuperclass()) {
if (auto superclassDecl = superclass->getAnyNominal()) {
if (unsatisfiedDependency(
requestQualifiedLookupInDeclContext({ superclassDecl,
payload.Name,
payload.Loc })))
return;
}
}
}
// FIXME: we need to resolve the set of protocol conformances to do
// unqualified lookup.
// If we're looking for all names or for an initializer, resolve
// implicitly-declared initializers.
// FIXME: Recursion into old type checker.
auto name = payload.Name;
if (!name || name.matchesRef(getASTContext().Id_init))
TC.resolveImplicitConstructors(nominal);
}
bool IterativeTypeChecker::breakCycleForQualifiedLookupInDeclContext(
TypeCheckRequest::DeclContextLookupPayloadType payload) {
return false;
}
//----------------------------------------------------------------------------//
// Qualified name lookup handling
//----------------------------------------------------------------------------//
bool IterativeTypeChecker::isUnqualifiedLookupInDeclContextSatisfied(
TypeCheckRequest::DeclContextLookupPayloadType payload) {
auto dc = payload.DC;
switch (dc->getContextKind()) {
case DeclContextKind::AbstractClosureExpr:
case DeclContextKind::AbstractFunctionDecl:
case DeclContextKind::Initializer:
case DeclContextKind::TopLevelCodeDecl:
case DeclContextKind::SerializedLocal:
// FIXME: Actually do the lookup in these contexts, because if we find
// something, we don't need to continue outward.
return isUnqualifiedLookupInDeclContextSatisfied({dc->getParent(),
payload.Name,
payload.Loc});
case DeclContextKind::Module:
case DeclContextKind::FileUnit:
// Modules and file units can always handle name lookup.
return true;
case DeclContextKind::NominalTypeDecl:
case DeclContextKind::ExtensionDecl:
// Check whether we can perform qualified lookup into this
// declaration context.
if (!isSatisfied(
requestQualifiedLookupInDeclContext({dc, payload.Name, payload.Loc})))
return false;
// FIXME: If there is a name, actually perform qualified lookup
// into this DeclContext. If it succeeds, there's nothing more to
// do.
return isUnqualifiedLookupInDeclContextSatisfied({dc->getParent(),
payload.Name,
payload.Loc});
}
}
void IterativeTypeChecker::processUnqualifiedLookupInDeclContext(
TypeCheckRequest::DeclContextLookupPayloadType payload,
UnsatisfiedDependency unsatisfiedDependency) {
// Everything is handled by the dependencies.
}
bool IterativeTypeChecker::breakCycleForUnqualifiedLookupInDeclContext(
TypeCheckRequest::DeclContextLookupPayloadType payload) {
return false;
}