Skip to content

Commit 280dace

Browse files
committed
[ClangImporter] Import ObjC members under Swift 3 and 4 names.
...so that developers get proper diagnostics when they use the wrong name. rdar://problem/29170671
1 parent 478fdd7 commit 280dace

File tree

9 files changed

+224
-82
lines changed

9 files changed

+224
-82
lines changed

lib/ClangImporter/ImportDecl.cpp

+30-52
Original file line numberDiff line numberDiff line change
@@ -3661,12 +3661,6 @@ namespace {
36613661
{decl->param_begin(), decl->param_size()},
36623662
decl->isVariadic(), redundant);
36633663

3664-
if (auto rawDecl = Impl.importDecl(decl, ImportNameVersion::Raw)) {
3665-
// We expect the raw decl to always be a method.
3666-
assert(isa<FuncDecl>(rawDecl));
3667-
Impl.addAlternateDecl(result, cast<ValueDecl>(rawDecl));
3668-
}
3669-
36703664
return result;
36713665
}
36723666

@@ -3935,13 +3929,6 @@ namespace {
39353929
importObjCGenericParams(const clang::ObjCInterfaceDecl *decl,
39363930
DeclContext *dc);
39373931

3938-
/// Import members of the given Objective-C container and add them to the
3939-
/// list of corresponding Swift members.
3940-
void importObjCMembers(const clang::ObjCContainerDecl *decl,
3941-
DeclContext *swiftContext,
3942-
llvm::SmallPtrSet<Decl *, 4> &knownMembers,
3943-
SmallVectorImpl<Decl *> &members);
3944-
39453932
/// \brief Import the members of all of the protocols to which the given
39463933
/// Objective-C class, category, or extension explicitly conforms into
39473934
/// the given list of members, so long as the method was not already
@@ -6421,37 +6408,6 @@ Optional<GenericParamList *> SwiftDeclConverter::importObjCGenericParams(
64216408
genericParams, Impl.importSourceLoc(typeParamList->getRAngleLoc()));
64226409
}
64236410

6424-
void SwiftDeclConverter::importObjCMembers(
6425-
const clang::ObjCContainerDecl *decl, DeclContext *swiftContext,
6426-
llvm::SmallPtrSet<Decl *, 4> &knownMembers,
6427-
SmallVectorImpl<Decl *> &members) {
6428-
for (auto m = decl->decls_begin(), mEnd = decl->decls_end(); m != mEnd; ++m) {
6429-
auto nd = dyn_cast<clang::NamedDecl>(*m);
6430-
if (!nd || nd != nd->getCanonicalDecl())
6431-
continue;
6432-
6433-
auto member = Impl.importDecl(nd, getVersion());
6434-
if (!member)
6435-
continue;
6436-
6437-
if (auto objcMethod = dyn_cast<clang::ObjCMethodDecl>(nd)) {
6438-
// If there is are alternate declarations for this member, add it.
6439-
for (auto alternate : Impl.getAlternateDecls(member)) {
6440-
if (alternate->getDeclContext() == member->getDeclContext() &&
6441-
knownMembers.insert(alternate).second)
6442-
members.push_back(alternate);
6443-
}
6444-
6445-
// If this declaration shouldn't be visible, don't add it to
6446-
// the list.
6447-
if (shouldSuppressDeclImport(objcMethod))
6448-
continue;
6449-
}
6450-
6451-
members.push_back(member);
6452-
}
6453-
}
6454-
64556411
void SwiftDeclConverter::importMirroredProtocolMembers(
64566412
const clang::ObjCContainerDecl *decl, DeclContext *dc,
64576413
ArrayRef<ProtocolDecl *> protocols, SmallVectorImpl<Decl *> &members,
@@ -7810,10 +7766,6 @@ ClangImporter::Implementation::loadAllMembers(Decl *D, uint64_t extra) {
78107766
Instance->getSourceManager(),
78117767
"loading members for");
78127768

7813-
// TODO: accommodate deprecated versions as well
7814-
SwiftDeclConverter converter(*this, CurrentVersion);
7815-
SwiftDeclConverter swift2Converter(*this, ImportNameVersion::Swift2);
7816-
78177769
DeclContext *DC;
78187770
IterableDeclContext *IDC;
78197771
SmallVector<ProtocolDecl *, 4> protos;
@@ -7848,9 +7800,36 @@ ClangImporter::Implementation::loadAllMembers(Decl *D, uint64_t extra) {
78487800
ImportingEntityRAII Importing(*this);
78497801

78507802
SmallVector<Decl *, 16> members;
7851-
llvm::SmallPtrSet<Decl *, 4> knownMembers;
7852-
converter.importObjCMembers(objcContainer, DC, knownMembers, members);
7853-
swift2Converter.importObjCMembers(objcContainer, DC, knownMembers, members);
7803+
llvm::SmallPtrSet<Decl *, 4> knownAlternateMembers;
7804+
for (const clang::Decl *m : objcContainer->decls()) {
7805+
auto nd = dyn_cast<clang::NamedDecl>(m);
7806+
if (!nd || nd != nd->getCanonicalDecl())
7807+
continue;
7808+
7809+
forEachDistinctName(*this, nd,
7810+
[&](ImportedName name, ImportNameVersion nameVersion) {
7811+
auto member = importDecl(nd, nameVersion);
7812+
if (!member)
7813+
return;
7814+
7815+
// If there are alternate declarations for this member, add them.
7816+
for (auto alternate : getAlternateDecls(member)) {
7817+
if (alternate->getDeclContext() == member->getDeclContext() &&
7818+
knownAlternateMembers.insert(alternate).second) {
7819+
members.push_back(alternate);
7820+
}
7821+
}
7822+
7823+
// If this declaration shouldn't be visible, don't add it to
7824+
// the list.
7825+
if (shouldSuppressDeclImport(nd))
7826+
return;
7827+
7828+
members.push_back(member);
7829+
});
7830+
}
7831+
7832+
SwiftDeclConverter converter(*this, CurrentVersion);
78547833

78557834
protos = takeImportedProtocols(D);
78567835
if (auto clangClass = dyn_cast<clang::ObjCInterfaceDecl>(objcContainer)) {
@@ -7879,7 +7858,6 @@ ClangImporter::Implementation::loadAllMembers(Decl *D, uint64_t extra) {
78797858
for (auto member : members) {
78807859
IDC->addMember(member);
78817860
}
7882-
78837861
}
78847862

78857863
void ClangImporter::Implementation::loadAllConformances(

test/APINotes/Inputs/custom-frameworks/APINotesFrameworkTest.framework/Headers/APINotesFrameworkTest.apinotes

+34-1
Original file line numberDiff line numberDiff line change
@@ -88,12 +88,39 @@ SwiftVersions:
8888
SwiftImportAsNonGeneric: true
8989
- Name: RenamedGeneric
9090
SwiftName: OldRenamedGeneric
91+
- Name: ClassWithManyRenames
92+
Methods:
93+
- Selector: "classWithManyRenamesForInt:"
94+
MethodKind: Class
95+
SwiftName: "init(swift3Factory:)"
96+
- Selector: "initWithBoolean:"
97+
MethodKind: Instance
98+
SwiftName: "init(swift3Boolean:)"
99+
- Selector: "doImportantThings"
100+
MethodKind: Instance
101+
SwiftName: "swift3DoImportantThings()"
102+
Properties:
103+
- Name: "importantClassProperty"
104+
PropertyKind: Class
105+
SwiftName: "swift3ClassProperty"
91106
Protocols:
92107
- Name: ProtoWithVersionedUnavailableMember
93108
Methods:
94109
- Selector: requirement
95110
MethodKind: Instance
96111
ResultType: 'ForwardClass * _Nullable'
112+
- Name: ProtoWithManyRenames
113+
Methods:
114+
- Selector: "initWithBoolean:"
115+
MethodKind: Instance
116+
SwiftName: "init(swift3Boolean:)"
117+
- Selector: "doImportantThings"
118+
MethodKind: Instance
119+
SwiftName: "swift3DoImportantThings()"
120+
Properties:
121+
- Name: "importantClassProperty"
122+
PropertyKind: Class
123+
SwiftName: "swift3ClassProperty"
97124
Functions:
98125
- Name: acceptDoublePointer
99126
SwiftName: 'acceptPointer(_:)'
@@ -119,4 +146,10 @@ SwiftVersions:
119146
SwiftName: aliasRenamedSwift3
120147
- Name: OptionyEnumRenamed
121148
SwiftName: renamedSwift3
122-
149+
- Version: 4
150+
Classes:
151+
- Name: ClassWithManyRenames
152+
Methods:
153+
- Selector: "classWithManyRenamesForInt:"
154+
MethodKind: Class
155+
SwiftName: "init(swift4Factory:)"

test/APINotes/Inputs/custom-frameworks/APINotesFrameworkTest.framework/Headers/Classes.h

+8
Original file line numberDiff line numberDiff line change
@@ -8,5 +8,13 @@
88
@interface RenamedGeneric<Element: Base *> : Base
99
@end
1010

11+
@interface ClassWithManyRenames : Base
12+
+ (instancetype)classWithManyRenamesForInt:(int)value;
13+
- (instancetype)initWithBoolean:(_Bool)value __attribute__((swift_name("init(finalBoolean:)")));
14+
15+
- (void)doImportantThings __attribute__((swift_name("finalDoImportantThings()")));
16+
@property (class, nullable) id importantClassProperty __attribute__((swift_name("finalClassProperty")));
17+
@end
18+
1119
#pragma clang assume_nonnull end
1220
#endif // __OBJC__

test/APINotes/Inputs/custom-frameworks/APINotesFrameworkTest.framework/Headers/Protocols.h

+6
Original file line numberDiff line numberDiff line change
@@ -7,5 +7,11 @@
77
- (nullable id)requirement;
88
@end
99

10+
@protocol ProtoWithManyRenames
11+
- (instancetype)initWithBoolean:(_Bool)value __attribute__((swift_name("init(finalBoolean:)")));
12+
- (void)doImportantThings __attribute__((swift_name("finalDoImportantThings()")));
13+
@property (class, nullable) id importantClassProperty __attribute__((swift_name("finalClassProperty")));
14+
@end
15+
1016
#pragma clang assume_nonnull end
1117
#endif // __OBJC__

test/APINotes/versioned-objc.swift

+97
Original file line numberDiff line numberDiff line change
@@ -43,4 +43,101 @@ func testRenamedGeneric() {
4343
// CHECK-DIAGS-4:[[@LINE-1]]:{{[0-9]+}}: error: 'RenamedGeneric' requires that 'SwiftClass' inherit from 'Base'
4444
}
4545

46+
func testRenamedClassMembers(obj: ClassWithManyRenames) {
47+
// CHECK-DIAGS-3: [[@LINE+1]]:{{[0-9]+}}: error: 'classWithManyRenamesForInt' has been replaced by 'init(swift3Factory:)'
48+
_ = ClassWithManyRenames.classWithManyRenamesForInt(0)
49+
// CHECK-DIAGS-4: [[@LINE-1]]:{{[0-9]+}}: error: 'classWithManyRenamesForInt' has been replaced by 'init(swift4Factory:)'
50+
51+
// CHECK-DIAGS-3: [[@LINE+1]]:{{[0-9]+}}: error: 'init(forInt:)' has been replaced by 'init(swift3Factory:)'
52+
_ = ClassWithManyRenames(forInt: 0)
53+
// CHECK-DIAGS-4: [[@LINE-1]]:{{[0-9]+}}: error: 'init(forInt:)' has been replaced by 'init(swift4Factory:)'
54+
55+
// CHECK-DIAGS-3-NOT: :[[@LINE+1]]:{{[0-9]+}}:
56+
_ = ClassWithManyRenames(swift3Factory: 0)
57+
// CHECK-DIAGS-4: [[@LINE-1]]:{{[0-9]+}}: error: 'init(swift3Factory:)' has been replaced by 'init(swift4Factory:)'
58+
59+
// CHECK-DIAGS-3: [[@LINE+1]]:{{[0-9]+}}: error: 'init(swift4Factory:)' has been replaced by 'init(swift3Factory:)'
60+
_ = ClassWithManyRenames(swift4Factory: 0)
61+
// CHECK-DIAGS-4-NOT: :[[@LINE-1]]:{{[0-9]+}}:
62+
63+
64+
// CHECK-DIAGS-3: [[@LINE+1]]:{{[0-9]+}}: error: 'init(boolean:)' has been renamed to 'init(swift3Boolean:)'
65+
_ = ClassWithManyRenames(boolean: false)
66+
// CHECK-DIAGS-4: [[@LINE-1]]:{{[0-9]+}}: error: 'init(boolean:)' has been renamed to 'init(finalBoolean:)'
67+
68+
// CHECK-DIAGS-3-NOT: :[[@LINE+1]]:{{[0-9]+}}:
69+
_ = ClassWithManyRenames(swift3Boolean: false)
70+
// CHECK-DIAGS-4: [[@LINE-1]]:{{[0-9]+}}: error: 'init(swift3Boolean:)' has been renamed to 'init(finalBoolean:)'
71+
72+
// CHECK-DIAGS-3: [[@LINE+1]]:{{[0-9]+}}: error: 'init(finalBoolean:)' has been renamed to 'init(swift3Boolean:)'
73+
_ = ClassWithManyRenames(finalBoolean: false)
74+
// CHECK-DIAGS-4-NOT: :[[@LINE-1]]:{{[0-9]+}}:
75+
76+
77+
// CHECK-DIAGS-3: [[@LINE+1]]:{{[0-9]+}}: error: 'doImportantThings()' has been renamed to 'swift3DoImportantThings()'
78+
obj.doImportantThings()
79+
// CHECK-DIAGS-4: [[@LINE-1]]:{{[0-9]+}}: error: 'doImportantThings()' has been renamed to 'finalDoImportantThings()'
80+
81+
// CHECK-DIAGS-3-NOT: :[[@LINE+1]]:{{[0-9]+}}:
82+
obj.swift3DoImportantThings()
83+
// CHECK-DIAGS-4: [[@LINE-1]]:{{[0-9]+}}: error: 'swift3DoImportantThings()' has been renamed to 'finalDoImportantThings()'
84+
85+
// CHECK-DIAGS-3: [[@LINE+1]]:{{[0-9]+}}: error: 'finalDoImportantThings()' has been renamed to 'swift3DoImportantThings()'
86+
obj.finalDoImportantThings()
87+
// CHECK-DIAGS-4-NOT: :[[@LINE-1]]:{{[0-9]+}}:
88+
89+
90+
// CHECK-DIAGS-3: [[@LINE+1]]:{{[0-9]+}}: error: 'importantClassProperty' has been renamed to 'swift3ClassProperty'
91+
_ = ClassWithManyRenames.importantClassProperty
92+
// CHECK-DIAGS-4: [[@LINE-1]]:{{[0-9]+}}: error: 'importantClassProperty' has been renamed to 'finalClassProperty'
93+
94+
// CHECK-DIAGS-3-NOT: :[[@LINE+1]]:{{[0-9]+}}:
95+
_ = ClassWithManyRenames.swift3ClassProperty
96+
// CHECK-DIAGS-4: [[@LINE-1]]:{{[0-9]+}}: error: 'swift3ClassProperty' has been renamed to 'finalClassProperty'
97+
98+
// CHECK-DIAGS-3: [[@LINE+1]]:{{[0-9]+}}: error: 'finalClassProperty' has been renamed to 'swift3ClassProperty'
99+
_ = ClassWithManyRenames.finalClassProperty
100+
// CHECK-DIAGS-4-NOT: :[[@LINE-1]]:{{[0-9]+}}:
101+
}
102+
103+
func testRenamedProtocolMembers(obj: ProtoWithManyRenames) {
104+
// CHECK-DIAGS-3: [[@LINE+1]]:{{[0-9]+}}: error: 'init(boolean:)' has been renamed to 'init(swift3Boolean:)'
105+
_ = type(of: obj).init(boolean: false)
106+
// CHECK-DIAGS-4: [[@LINE-1]]:{{[0-9]+}}: error: 'init(boolean:)' has been renamed to 'init(finalBoolean:)'
107+
108+
// CHECK-DIAGS-3-NOT: :[[@LINE+1]]:{{[0-9]+}}:
109+
_ = type(of: obj).init(swift3Boolean: false)
110+
// CHECK-DIAGS-4: [[@LINE-1]]:{{[0-9]+}}: error: 'init(swift3Boolean:)' has been renamed to 'init(finalBoolean:)'
111+
112+
// CHECK-DIAGS-3: [[@LINE+1]]:{{[0-9]+}}: error: 'init(finalBoolean:)' has been renamed to 'init(swift3Boolean:)'
113+
_ = type(of: obj).init(finalBoolean: false)
114+
// CHECK-DIAGS-4-NOT: :[[@LINE-1]]:{{[0-9]+}}:
115+
116+
117+
// CHECK-DIAGS-3: [[@LINE+1]]:{{[0-9]+}}: error: 'doImportantThings()' has been renamed to 'swift3DoImportantThings()'
118+
obj.doImportantThings()
119+
// CHECK-DIAGS-4: [[@LINE-1]]:{{[0-9]+}}: error: 'doImportantThings()' has been renamed to 'finalDoImportantThings()'
120+
121+
// CHECK-DIAGS-3-NOT: :[[@LINE+1]]:{{[0-9]+}}:
122+
obj.swift3DoImportantThings()
123+
// CHECK-DIAGS-4: [[@LINE-1]]:{{[0-9]+}}: error: 'swift3DoImportantThings()' has been renamed to 'finalDoImportantThings()'
124+
125+
// CHECK-DIAGS-3: [[@LINE+1]]:{{[0-9]+}}: error: 'finalDoImportantThings()' has been renamed to 'swift3DoImportantThings()'
126+
obj.finalDoImportantThings()
127+
// CHECK-DIAGS-4-NOT: :[[@LINE-1]]:{{[0-9]+}}:
128+
129+
130+
// CHECK-DIAGS-3: [[@LINE+1]]:{{[0-9]+}}: error: 'importantClassProperty' has been renamed to 'swift3ClassProperty'
131+
_ = type(of: obj).importantClassProperty
132+
// CHECK-DIAGS-4: [[@LINE-1]]:{{[0-9]+}}: error: 'importantClassProperty' has been renamed to 'finalClassProperty'
133+
134+
// CHECK-DIAGS-3-NOT: :[[@LINE+1]]:{{[0-9]+}}:
135+
_ = type(of: obj).swift3ClassProperty
136+
// CHECK-DIAGS-4: [[@LINE-1]]:{{[0-9]+}}: error: 'swift3ClassProperty' has been renamed to 'finalClassProperty'
137+
138+
// CHECK-DIAGS-3: [[@LINE+1]]:{{[0-9]+}}: error: 'finalClassProperty' has been renamed to 'swift3ClassProperty'
139+
_ = type(of: obj).finalClassProperty
140+
// CHECK-DIAGS-4-NOT: :[[@LINE-1]]:{{[0-9]+}}:
141+
}
142+
46143
let unrelatedDiagnostic: Int = nil

test/ClangImporter/Inputs/SwiftPrivateAttr.txt

+5-5
Original file line numberDiff line numberDiff line change
@@ -9,21 +9,21 @@ class Foo : NSObject, __PrivProto {
99
func __oneArg(_ arg: Int32)
1010
func __twoArgs(_ arg: Int32, other arg2: Int32)
1111
class func __withNoArgs() -> Self!
12+
@available(swift, obsoleted: 3, renamed: "__withNoArgs()")
13+
class func __fooWithNoArgs() -> Self!
14+
convenience init!(__oneArg arg: Int32)
1215
@available(*, unavailable, renamed: "init(__oneArg:)", message: "Not available in Swift")
1316
class func __fooWithOneArg(_ arg: Int32) -> Self!
14-
convenience init!(__oneArg arg: Int32)
17+
convenience init!(__twoArgs arg: Int32, other arg2: Int32)
1518
@available(*, unavailable, renamed: "init(__twoArgs:other:)", message: "Not available in Swift")
1619
class func __fooWithTwoArgs(_ arg: Int32, other arg2: Int32) -> Self!
17-
convenience init!(__twoArgs arg: Int32, other arg2: Int32)
20+
convenience init!(__ arg: Int32)
1821
@available(*, unavailable, renamed: "init(__:)", message: "Not available in Swift")
1922
class func __foo(_ arg: Int32) -> Self!
20-
convenience init!(__ arg: Int32)
2123
func objectForKeyedSubscript(_ index: Any!) -> Any!
2224
func __setObject(_ object: Any!, forKeyedSubscript index: Any!)
2325
func __objectAtIndexedSubscript(_ index: Int32) -> Any!
2426
func setObject(_ object: Any!, atIndexedSubscript index: Int32)
25-
@available(swift, obsoleted: 3, renamed: "__withNoArgs()")
26-
class func __fooWithNoArgs() -> Self!
2727
init()
2828
}
2929
class Bar : NSObject {

test/IDE/print_clang_decls_AppKit.swift

+4
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,11 @@
2020
// APPKIT-LABEL: {{^}}class NSView : NSObject, NSCoding, NSAccessibility {{{$}}
2121
// APPKIT-NEXT: init?(coder aDecoder: NSCoder)
2222
// APPKIT-NEXT: func isDescendant(of aView: NSView) -> Bool
23+
// APPKIT-NEXT: @available(swift, obsoleted: 3, renamed: "isDescendant(of:)")
24+
// APPKIT-NEXT: func isDescendantOf(_ aView: NSView) -> Bool
2325
// APPKIT-NEXT: func ancestorShared(with aView: NSView) -> NSView?
26+
// APPKIT-NEXT: @available(swift, obsoleted: 3, renamed: "ancestorShared(with:)")
27+
// APPKIT-NEXT: func ancestorSharedWithView(_ aView: NSView) -> NSView?
2428
// APPKIT-NEXT: func addSubview(_ aView: NSView)
2529
// APPKIT-NEXT: func addSubview(_ aView: NSView, positioned place: UInt32, relativeTo otherView: NSView?)
2630
// APPKIT-NEXT: unowned(unsafe) var superview: @sil_unmanaged NSView? { get }

0 commit comments

Comments
 (0)