Skip to content

Commit 1602580

Browse files
committedApr 20, 2018
SIL: Sort vtable entries for synthesized methods
Completes the fix for <rdar://problem/35647420> and <https://bugs.swift.org/browse/SR-6468>.
1 parent 0ff2cd9 commit 1602580

File tree

3 files changed

+178
-6
lines changed

3 files changed

+178
-6
lines changed
 

‎include/swift/SIL/SILVTableVisitor.h

+78-6
Original file line numberDiff line numberDiff line change
@@ -18,11 +18,55 @@
1818
#ifndef SWIFT_SIL_SILVTABLEVISITOR_H
1919
#define SWIFT_SIL_SILVTABLEVISITOR_H
2020

21+
#include <string>
22+
2123
#include "swift/AST/Decl.h"
2224
#include "swift/AST/Types.h"
25+
#include "swift/AST/ASTMangler.h"
2326

2427
namespace swift {
2528

29+
// Utility class for deterministically ordering vtable entries for
30+
// synthesized methods.
31+
struct SortedFuncList {
32+
using Entry = std::pair<std::string, AbstractFunctionDecl *>;
33+
SmallVector<Entry, 2> elts;
34+
bool sorted = false;
35+
36+
void add(AbstractFunctionDecl *afd) {
37+
Mangle::ASTMangler mangler;
38+
std::string mangledName;
39+
if (auto *cd = dyn_cast<ConstructorDecl>(afd))
40+
mangledName = mangler.mangleConstructorEntity(cd, 0, 0);
41+
else
42+
mangledName = mangler.mangleEntity(afd, 0);
43+
44+
elts.push_back(std::make_pair(mangledName, afd));
45+
}
46+
47+
bool empty() { return elts.empty(); }
48+
49+
void sort() {
50+
assert(!sorted);
51+
sorted = true;
52+
std::sort(elts.begin(),
53+
elts.end(),
54+
[](const Entry &lhs, const Entry &rhs) -> bool {
55+
return lhs.first < rhs.first;
56+
});
57+
}
58+
59+
decltype(elts)::const_iterator begin() const {
60+
assert(sorted);
61+
return elts.begin();
62+
}
63+
64+
decltype(elts)::const_iterator end() const {
65+
assert(sorted);
66+
return elts.end();
67+
}
68+
};
69+
2670
/// A CRTP class for visiting virtually-dispatched methods of a class.
2771
///
2872
/// You must override these two methods in your subclass:
@@ -76,19 +120,47 @@ template <class T> class SILVTableVisitor {
76120
}
77121
}
78122

123+
void maybeAddMember(Decl *member) {
124+
if (auto *fd = dyn_cast<FuncDecl>(member))
125+
maybeAddMethod(fd);
126+
else if (auto *cd = dyn_cast<ConstructorDecl>(member))
127+
maybeAddConstructor(cd);
128+
else if (auto *placeholder = dyn_cast<MissingMemberDecl>(member))
129+
asDerived().addPlaceholder(placeholder);
130+
}
131+
79132
protected:
80133
void addVTableEntries(ClassDecl *theClass) {
81134
// Imported classes do not have a vtable.
82135
if (!theClass->hasKnownSwiftImplementation())
83136
return;
84137

138+
// Note that while vtable order is not ABI, we still want it to be
139+
// consistent between translation units.
140+
//
141+
// So, sort synthesized members by their mangled name, since they
142+
// are added lazily during type checking, with the remaining ones
143+
// forced at the end.
144+
SortedFuncList synthesizedMembers;
145+
85146
for (auto member : theClass->getMembers()) {
86-
if (auto *fd = dyn_cast<FuncDecl>(member))
87-
maybeAddMethod(fd);
88-
else if (auto *cd = dyn_cast<ConstructorDecl>(member))
89-
maybeAddConstructor(cd);
90-
else if (auto *placeholder = dyn_cast<MissingMemberDecl>(member))
91-
asDerived().addPlaceholder(placeholder);
147+
if (auto *afd = dyn_cast<AbstractFunctionDecl>(member)) {
148+
if (afd->isSynthesized()) {
149+
synthesizedMembers.add(afd);
150+
continue;
151+
}
152+
}
153+
154+
maybeAddMember(member);
155+
}
156+
157+
if (synthesizedMembers.empty())
158+
return;
159+
160+
synthesizedMembers.sort();
161+
162+
for (const auto &pair : synthesizedMembers) {
163+
maybeAddMember(pair.second);
92164
}
93165
}
94166
};

‎test/stdlib/CodableMultifile.swift

+79
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,79 @@
1+
// RUN: %empty-directory(%t)
2+
// RUN: cp %s %t/main.swift
3+
// RUN: %target-build-swift %t/main.swift %S/Inputs/CodableMultifileOther.swift -module-name main -o %t/main
4+
// RUN: %target-run %t/main
5+
// REQUIRES: executable_test
6+
7+
// FIXME: This test could run on Linux too, if we could either use
8+
// corelibs-foundation, or implement a mock Encoder for testing.
9+
10+
// REQUIRES: objc_interop
11+
12+
import StdlibUnittest
13+
import Foundation
14+
15+
var CodableMultifileTestSuite = TestSuite("CodableMultifile")
16+
17+
// The first test doesn't synthesize encode(to:) at all.
18+
CodableMultifileTestSuite.test("1") {
19+
let derived = DerivedFirst()
20+
21+
// Make sure the vtable offset matches between this translation unit and
22+
// the other file, which requires us to force the encode(to:) member when
23+
// type checking this translation unit.
24+
expectEqual(false, derived.derivedMember)
25+
}
26+
27+
// The second test synthesizes init(from:) before encode(to:).
28+
29+
// We define a wrapper so that the virtual method BaseSecond.encode(to:) method
30+
// is called from outside its module. If we use the conformance of BaseSecond
31+
// to Codable, we don't expose the bug because the virtual method call is
32+
// encapsulated in the conformance, which is emitted in the same translation unit
33+
// as BaseSecond.
34+
struct WrapperSecond : Encodable {
35+
let base: BaseSecond
36+
37+
func encode(to encoder: Encoder) throws {
38+
try base.encode(to: encoder)
39+
}
40+
}
41+
42+
CodableMultifileTestSuite.test("2") {
43+
// Make sure we synthesize the init(from:) member before encode(to:) in
44+
// this translation unit.
45+
_ = BaseSecond.init(from:)
46+
47+
let base = WrapperSecond(base: BaseSecond())
48+
let encoder = JSONEncoder()
49+
50+
expectEqual(
51+
"{\"baseMember\":2}",
52+
String(data: try! encoder.encode(base), encoding: .utf8)!)
53+
}
54+
55+
// The third test synthesizes encode(to:) before init(from:).
56+
57+
// See above.
58+
struct WrapperThird : Encodable {
59+
let base: BaseThird
60+
61+
func encode(to encoder: Encoder) throws {
62+
try base.encode(to: encoder)
63+
}
64+
}
65+
66+
CodableMultifileTestSuite.test("3") {
67+
// Make sure we synthesize the encode(to:) member before init(from:) in
68+
// this translation unit.
69+
_ = BaseThird.encode(to:)
70+
71+
let base = WrapperThird(base: BaseThird())
72+
let encoder = JSONEncoder()
73+
74+
expectEqual(
75+
"{\"baseMember\":3}",
76+
String(data: try! encoder.encode(base), encoding: .utf8)!)
77+
}
78+
79+
runAllTests()
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
class BaseFirst: Codable {
2+
var baseMember = 1
3+
4+
init() {}
5+
}
6+
7+
class DerivedFirst: BaseFirst {
8+
var derivedMember = false
9+
}
10+
11+
class BaseSecond: Codable {
12+
var baseMember = 2
13+
14+
init() {}
15+
}
16+
17+
class BaseThird: Codable {
18+
var baseMember = 3
19+
20+
init() {}
21+
}

0 commit comments

Comments
 (0)
Please sign in to comment.