Skip to content

Commit 8a9495a

Browse files
committed
[ClangImporter] Honor Swift 4 API notes in Swift 3 mode
*** Depends on Clang change "[APINotes] Record what version caused *** *** an annotation to get replaced." Update your Clang checkout! *** More generally, change the meaning of the SwiftVersions section in API notes to be "this version or earlier" rather than "exactly this version". We mostly get this behavior for free from the Clang-side changes, but for SwiftName and the enum annotations we look at inactive attributes as well. The latter is simple, but the former means being careful about finding the annotation we /would/ have picked, i.e. the one closest to the version we requested.
1 parent 9a04bee commit 8a9495a

File tree

6 files changed

+351
-19
lines changed

6 files changed

+351
-19
lines changed

lib/ClangImporter/ImportEnumInfo.cpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -68,7 +68,7 @@ void EnumInfo::classifyEnum(ASTContext &ctx, const clang::EnumDecl *decl,
6868
// If API notes have /removed/ a FlagEnum or EnumExtensibility attribute,
6969
// then we don't need to check the macros.
7070
for (auto *attr : decl->specific_attrs<clang::SwiftVersionedAttr>()) {
71-
if (!attr->getVersion().empty())
71+
if (!attr->getIsReplacedByActive())
7272
continue;
7373
if (isa<clang::FlagEnumAttr>(attr->getAttrToAdd()) ||
7474
isa<clang::EnumExtensibilityAttr>(attr->getAttrToAdd())) {

lib/ClangImporter/ImportName.cpp

+98-18
Original file line numberDiff line numberDiff line change
@@ -545,12 +545,58 @@ determineCtorInitializerKind(const clang::ObjCMethodDecl *method) {
545545
return None;
546546
}
547547

548-
template <typename A>
549-
static bool matchesVersion(A *versionedAttr, ImportNameVersion version) {
550-
clang::VersionTuple attrVersion = versionedAttr->getVersion();
551-
if (attrVersion.empty())
552-
return version == ImportNameVersion::maxVersion();
553-
return attrVersion.getMajor() == version.majorVersionNumber();
548+
namespace {
549+
/// Aggregate struct for the common members of clang::SwiftVersionedAttr and
550+
/// clang::SwiftVersionedRemovalAttr.
551+
///
552+
/// For a SwiftVersionedRemovalAttr, the Attr member will be null.
553+
struct VersionedSwiftNameInfo {
554+
const clang::SwiftNameAttr *Attr;
555+
clang::VersionTuple Version;
556+
bool IsReplacedByActive;
557+
};
558+
559+
/// The action to take upon seeing a particular versioned swift_name annotation.
560+
enum class VersionedSwiftNameAction {
561+
/// This annotation is not interesting.
562+
Ignore,
563+
/// This annotation is better than whatever we have so far.
564+
Use,
565+
/// This annotation is better than nothing, but that's all; don't bother
566+
/// recording its version.
567+
UseAsFallback,
568+
/// This annotation itself isn't interesting, but its version shows that the
569+
/// correct answer is whatever's currently active.
570+
ResetToActive
571+
};
572+
} // end anonymous namespace
573+
574+
static VersionedSwiftNameAction
575+
checkVersionedSwiftName(VersionedSwiftNameInfo info,
576+
clang::VersionTuple bestSoFar,
577+
ImportNameVersion requestedVersion) {
578+
if (!bestSoFar.empty() && bestSoFar <= info.Version)
579+
return VersionedSwiftNameAction::Ignore;
580+
581+
if (info.IsReplacedByActive) {
582+
// We know that there are no versioned names between the active version and
583+
// a replacement version, because otherwise /that/ name would be active.
584+
// So if replacement < requested, we want to use the old value that was
585+
// replaced (but with very low priority), and otherwise we want to use the
586+
// new value that is now active. (Special case: replacement = 0 means that
587+
// a header annotation was replaced by an unversioned API notes annotation.)
588+
if (info.Version.empty() ||
589+
info.Version.getMajor() >= requestedVersion.majorVersionNumber()) {
590+
return VersionedSwiftNameAction::ResetToActive;
591+
}
592+
if (bestSoFar.empty())
593+
return VersionedSwiftNameAction::UseAsFallback;
594+
return VersionedSwiftNameAction::Ignore;
595+
}
596+
597+
if (info.Version.getMajor() < requestedVersion.majorVersionNumber())
598+
return VersionedSwiftNameAction::Ignore;
599+
return VersionedSwiftNameAction::Use;
554600
}
555601

556602

@@ -567,26 +613,60 @@ findSwiftNameAttr(const clang::Decl *decl, ImportNameVersion version) {
567613
return nullptr;
568614

569615
// Handle versioned API notes for Swift 3 and later. This is the common case.
570-
if (version != ImportNameVersion::swift2()) {
616+
if (version > ImportNameVersion::swift2()) {
617+
const auto *activeAttr = decl->getAttr<clang::SwiftNameAttr>();
618+
const clang::SwiftNameAttr *result = activeAttr;
619+
clang::VersionTuple bestSoFar;
571620
for (auto *attr : decl->attrs()) {
621+
VersionedSwiftNameInfo info;
622+
572623
if (auto *versionedAttr = dyn_cast<clang::SwiftVersionedAttr>(attr)) {
573-
if (!matchesVersion(versionedAttr, version))
624+
auto *added =
625+
dyn_cast<clang::SwiftNameAttr>(versionedAttr->getAttrToAdd());
626+
if (!added)
574627
continue;
575-
if (auto *added =
576-
dyn_cast<clang::SwiftNameAttr>(versionedAttr->getAttrToAdd())) {
577-
return added;
578-
}
579-
}
580628

581-
if (auto *removeAttr = dyn_cast<clang::SwiftVersionedRemovalAttr>(attr)) {
582-
if (!matchesVersion(removeAttr, version))
629+
info = {added, versionedAttr->getVersion(),
630+
versionedAttr->getIsReplacedByActive()};
631+
632+
} else if (auto *removeAttr =
633+
dyn_cast<clang::SwiftVersionedRemovalAttr>(attr)) {
634+
if (removeAttr->getAttrKindToRemove() != clang::attr::SwiftName)
583635
continue;
584-
if (removeAttr->getAttrKindToRemove() == clang::attr::SwiftName)
585-
return nullptr;
636+
info = {nullptr, removeAttr->getVersion(),
637+
removeAttr->getIsReplacedByActive()};
638+
639+
} else {
640+
continue;
641+
}
642+
643+
switch (checkVersionedSwiftName(info, bestSoFar, version)) {
644+
case VersionedSwiftNameAction::Ignore:
645+
continue;
646+
case VersionedSwiftNameAction::Use:
647+
result = info.Attr;
648+
bestSoFar = info.Version;
649+
break;
650+
case VersionedSwiftNameAction::UseAsFallback:
651+
// HACK: If there's a swift_name attribute in the headers /and/ in the
652+
// unversioned API notes /and/ in the active versioned API notes, there
653+
// will be two "replacement" attributes, one for each of the first two
654+
// cases. Prefer the first one we see, because that turns out to be the
655+
// one from the API notes, which matches the semantics when there are no
656+
// versioned API notes. (This isn't very principled but there's at least
657+
// a test to tell us if it changes.)
658+
if (result == activeAttr)
659+
result = info.Attr;
660+
assert(bestSoFar.empty());
661+
break;
662+
case VersionedSwiftNameAction::ResetToActive:
663+
result = activeAttr;
664+
bestSoFar = info.Version;
665+
break;
586666
}
587667
}
588668

589-
return decl->getAttr<clang::SwiftNameAttr>();
669+
return result;
590670
}
591671

592672
// The remainder of this function emulates the limited form of swift_name

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

+86
Original file line numberDiff line numberDiff line change
@@ -65,6 +65,23 @@ Functions:
6565
Tags:
6666
- Name: InnerInSwift4
6767
SwiftName: Outer.Inner
68+
Globals:
69+
- Name: multiVersionedGlobal34Notes
70+
SwiftName: multiVersionedGlobal34Notes_NEW
71+
- Name: multiVersionedGlobal34Both
72+
SwiftName: multiVersionedGlobal34Both_NEW
73+
- Name: multiVersionedGlobal345Notes
74+
SwiftName: multiVersionedGlobal345Notes_NEW
75+
- Name: multiVersionedGlobal345Both
76+
SwiftName: multiVersionedGlobal345Both_NEW
77+
- Name: multiVersionedGlobal4Notes
78+
SwiftName: multiVersionedGlobal4Notes_NEW
79+
- Name: multiVersionedGlobal4Both
80+
SwiftName: multiVersionedGlobal4Both_NEW
81+
- Name: multiVersionedGlobal45Notes
82+
SwiftName: multiVersionedGlobal45Notes_NEW
83+
- Name: multiVersionedGlobal45Both
84+
SwiftName: multiVersionedGlobal45Both_NEW
6885
SwiftVersions:
6986
- Version: 3.0
7087
Classes:
@@ -207,3 +224,72 @@ SwiftVersions:
207224
SwiftName: aliasRenamedSwift3
208225
- Name: OptionyEnumRenamed
209226
SwiftName: renamedSwift3
227+
Globals:
228+
- Name: multiVersionedGlobal34
229+
SwiftName: multiVersionedGlobal34_3
230+
- Name: multiVersionedGlobal34Header
231+
SwiftName: multiVersionedGlobal34Header_3
232+
- Name: multiVersionedGlobal34Notes
233+
SwiftName: multiVersionedGlobal34Notes_3
234+
- Name: multiVersionedGlobal34Both
235+
SwiftName: multiVersionedGlobal34Both_3
236+
- Name: multiVersionedGlobal345
237+
SwiftName: multiVersionedGlobal345_3
238+
- Name: multiVersionedGlobal345Header
239+
SwiftName: multiVersionedGlobal345Header_3
240+
- Name: multiVersionedGlobal345Notes
241+
SwiftName: multiVersionedGlobal345Notes_3
242+
- Name: multiVersionedGlobal345Both
243+
SwiftName: multiVersionedGlobal345Both_3
244+
- Version: 5
245+
Globals:
246+
- Name: multiVersionedGlobal345
247+
SwiftName: multiVersionedGlobal345_5
248+
- Name: multiVersionedGlobal345Header
249+
SwiftName: multiVersionedGlobal345Header_5
250+
- Name: multiVersionedGlobal345Notes
251+
SwiftName: multiVersionedGlobal345Notes_5
252+
- Name: multiVersionedGlobal345Both
253+
SwiftName: multiVersionedGlobal345Both_5
254+
- Name: multiVersionedGlobal45
255+
SwiftName: multiVersionedGlobal45_5
256+
- Name: multiVersionedGlobal45Header
257+
SwiftName: multiVersionedGlobal45Header_5
258+
- Name: multiVersionedGlobal45Notes
259+
SwiftName: multiVersionedGlobal45Notes_5
260+
- Name: multiVersionedGlobal45Both
261+
SwiftName: multiVersionedGlobal45Both_5
262+
- Version: 4 # Versions are deliberately ordered as "3, 5, 4" to catch bugs.
263+
Globals:
264+
- Name: multiVersionedGlobal34
265+
SwiftName: multiVersionedGlobal34_4
266+
- Name: multiVersionedGlobal34Header
267+
SwiftName: multiVersionedGlobal34Header_4
268+
- Name: multiVersionedGlobal34Notes
269+
SwiftName: multiVersionedGlobal34Notes_4
270+
- Name: multiVersionedGlobal34Both
271+
SwiftName: multiVersionedGlobal34Both_4
272+
- Name: multiVersionedGlobal345
273+
SwiftName: multiVersionedGlobal345_4
274+
- Name: multiVersionedGlobal345Header
275+
SwiftName: multiVersionedGlobal345Header_4
276+
- Name: multiVersionedGlobal345Notes
277+
SwiftName: multiVersionedGlobal345Notes_4
278+
- Name: multiVersionedGlobal345Both
279+
SwiftName: multiVersionedGlobal345Both_4
280+
- Name: multiVersionedGlobal4
281+
SwiftName: multiVersionedGlobal4_4
282+
- Name: multiVersionedGlobal4Header
283+
SwiftName: multiVersionedGlobal4Header_4
284+
- Name: multiVersionedGlobal4Notes
285+
SwiftName: multiVersionedGlobal4Notes_4
286+
- Name: multiVersionedGlobal4Both
287+
SwiftName: multiVersionedGlobal4Both_4
288+
- Name: multiVersionedGlobal45
289+
SwiftName: multiVersionedGlobal45_4
290+
- Name: multiVersionedGlobal45Header
291+
SwiftName: multiVersionedGlobal45Header_4
292+
- Name: multiVersionedGlobal45Notes
293+
SwiftName: multiVersionedGlobal45Notes_4
294+
- Name: multiVersionedGlobal45Both
295+
SwiftName: multiVersionedGlobal45Both_4

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

+1
Original file line numberDiff line numberDiff line change
@@ -32,6 +32,7 @@ __attribute__((objc_root_class))
3232

3333
#import <APINotesFrameworkTest/Classes.h>
3434
#import <APINotesFrameworkTest/Enums.h>
35+
#import <APINotesFrameworkTest/Globals.h>
3536
#import <APINotesFrameworkTest/ImportAsMember.h>
3637
#import <APINotesFrameworkTest/Properties.h>
3738
#import <APINotesFrameworkTest/Protocols.h>
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
#pragma clang assume_nonnull begin
2+
3+
int multiVersionedGlobal4;
4+
int multiVersionedGlobal4Notes;
5+
int multiVersionedGlobal4Header __attribute__((swift_name("multiVersionedGlobal4Header_NEW")));
6+
int multiVersionedGlobal4Both __attribute__((swift_name("multiVersionedGlobal4Both_OLD")));
7+
8+
int multiVersionedGlobal34;
9+
int multiVersionedGlobal34Notes;
10+
int multiVersionedGlobal34Header __attribute__((swift_name("multiVersionedGlobal34Header_NEW")));
11+
int multiVersionedGlobal34Both __attribute__((swift_name("multiVersionedGlobal34Both_OLD")));
12+
13+
int multiVersionedGlobal45;
14+
int multiVersionedGlobal45Notes;
15+
int multiVersionedGlobal45Header __attribute__((swift_name("multiVersionedGlobal45Header_NEW")));
16+
int multiVersionedGlobal45Both __attribute__((swift_name("multiVersionedGlobal45Both_OLD")));
17+
18+
int multiVersionedGlobal345;
19+
int multiVersionedGlobal345Notes;
20+
int multiVersionedGlobal345Header __attribute__((swift_name("multiVersionedGlobal345Header_NEW")));
21+
int multiVersionedGlobal345Both __attribute__((swift_name("multiVersionedGlobal345Both_OLD")));
22+
23+
#pragma clang assume_nonnull end

0 commit comments

Comments
 (0)