Skip to content

Commit a4e9109

Browse files
authored
Merge pull request #17396 from lorentey/anyhashable-is-not-hashable
[stdlib] Fix AnyHashable's Equatable/Hashable conformance
2 parents 6ebdefb + 2f4ad79 commit a4e9109

14 files changed

+965
-272
lines changed

stdlib/private/StdlibUnittest/StdlibUnittest.swift

+27
Original file line numberDiff line numberDiff line change
@@ -2410,6 +2410,33 @@ internal func hash<H: Hashable>(_ value: H, seed: Int? = nil) -> Int {
24102410
return hasher.finalize()
24112411
}
24122412

2413+
/// Test that the elements of `groups` consist of instances that satisfy the
2414+
/// semantic requirements of `Hashable`, with each group defining a distinct
2415+
/// equivalence class under `==`.
2416+
public func checkHashableGroups<Groups: Collection>(
2417+
_ groups: Groups,
2418+
_ message: @autoclosure () -> String = "",
2419+
stackTrace: SourceLocStack = SourceLocStack(),
2420+
showFrame: Bool = true,
2421+
file: String = #file, line: UInt = #line
2422+
) where Groups.Element: Collection, Groups.Element.Element: Hashable {
2423+
let instances = groups.flatMap { $0 }
2424+
// groupIndices[i] is the index of the element in groups that contains
2425+
// instances[i].
2426+
let groupIndices =
2427+
zip(0..., groups).flatMap { i, group in group.map { _ in i } }
2428+
func equalityOracle(_ lhs: Int, _ rhs: Int) -> Bool {
2429+
return groupIndices[lhs] == groupIndices[rhs]
2430+
}
2431+
checkHashable(
2432+
instances,
2433+
equalityOracle: equalityOracle,
2434+
hashEqualityOracle: equalityOracle,
2435+
allowBrokenTransitivity: false,
2436+
stackTrace: stackTrace.pushIf(showFrame, file: file, line: line),
2437+
showFrame: false)
2438+
}
2439+
24132440
/// Test that the elements of `instances` satisfy the semantic requirements of
24142441
/// `Hashable`, using `equalityOracle` to generate equality and hashing
24152442
/// expectations from pairs of positions in `instances`.

stdlib/public/core/AnyHashable.swift

+37-60
Original file line numberDiff line numberDiff line change
@@ -39,20 +39,29 @@ public protocol _HasCustomAnyHashableRepresentation {
3939

4040
@usableFromInline // FIXME(sil-serialize-all)
4141
internal protocol _AnyHashableBox {
42-
func _unbox<T : Hashable>() -> T?
42+
var _canonicalBox: _AnyHashableBox { get }
4343

4444
/// Determine whether values in the boxes are equivalent.
4545
///
46+
/// - Precondition: `self` and `box` are in canonical form.
4647
/// - Returns: `nil` to indicate that the boxes store different types, so
4748
/// no comparison is possible. Otherwise, contains the result of `==`.
48-
func _isEqual(to: _AnyHashableBox) -> Bool?
49+
func _isEqual(to box: _AnyHashableBox) -> Bool?
4950
var _hashValue: Int { get }
5051
func _hash(into hasher: inout Hasher)
52+
func _rawHashValue(_seed: (UInt64, UInt64)) -> Int
5153

5254
var _base: Any { get }
55+
func _unbox<T: Hashable>() -> T?
5356
func _downCastConditional<T>(into result: UnsafeMutablePointer<T>) -> Bool
5457
}
5558

59+
extension _AnyHashableBox {
60+
var _canonicalBox: _AnyHashableBox {
61+
return self
62+
}
63+
}
64+
5665
@_fixed_layout // FIXME(sil-serialize-all)
5766
@usableFromInline // FIXME(sil-serialize-all)
5867
internal struct _ConcreteHashableBox<Base : Hashable> : _AnyHashableBox {
@@ -87,6 +96,11 @@ internal struct _ConcreteHashableBox<Base : Hashable> : _AnyHashableBox {
8796
_baseHashable.hash(into: &hasher)
8897
}
8998

99+
@inlinable // FIXME(sil-serialize-all)
100+
func _rawHashValue(_seed: (UInt64, UInt64)) -> Int {
101+
return _baseHashable._rawHashValue(seed: _seed)
102+
}
103+
90104
@inlinable // FIXME(sil-serialize-all)
91105
internal var _base: Any {
92106
return _baseHashable
@@ -101,19 +115,6 @@ internal struct _ConcreteHashableBox<Base : Hashable> : _AnyHashableBox {
101115
}
102116
}
103117

104-
#if _runtime(_ObjC)
105-
// Retrieve the custom AnyHashable representation of the value after it
106-
// has been bridged to Objective-C. This mapping to Objective-C and back
107-
// turns a non-custom representation into a custom one, which is used as
108-
// the lowest-common-denominator for comparisons.
109-
@inlinable // FIXME(sil-serialize-all)
110-
internal func _getBridgedCustomAnyHashable<T>(_ value: T) -> AnyHashable? {
111-
let bridgedValue = _bridgeAnythingToObjectiveC(value)
112-
return (bridgedValue as?
113-
_HasCustomAnyHashableRepresentation)?._toCustomAnyHashable()
114-
}
115-
#endif
116-
117118
/// A type-erased hashable value.
118119
///
119120
/// The `AnyHashable` type forwards equality comparisons and hashing operations
@@ -137,8 +138,11 @@ internal func _getBridgedCustomAnyHashable<T>(_ value: T) -> AnyHashable? {
137138
public struct AnyHashable {
138139
@usableFromInline // FIXME(sil-serialize-all)
139140
internal var _box: _AnyHashableBox
140-
@usableFromInline // FIXME(sil-serialize-all)
141-
internal var _usedCustomRepresentation: Bool
141+
142+
@inlinable // FIXME(sil-serialize-all)
143+
internal init(_box box: _AnyHashableBox) {
144+
self._box = box
145+
}
142146

143147
/// Creates a type-erased hashable value that wraps the given instance.
144148
///
@@ -160,15 +164,13 @@ public struct AnyHashable {
160164
/// - Parameter base: A hashable value to wrap.
161165
@inlinable // FIXME(sil-serialize-all)
162166
public init<H : Hashable>(_ base: H) {
163-
if let customRepresentation =
167+
if let custom =
164168
(base as? _HasCustomAnyHashableRepresentation)?._toCustomAnyHashable() {
165-
self = customRepresentation
166-
self._usedCustomRepresentation = true
169+
self = custom
167170
return
168171
}
169172

170-
self._box = _ConcreteHashableBox(0 as Int)
171-
self._usedCustomRepresentation = false
173+
self.init(_box: _ConcreteHashableBox(false)) // Dummy value
172174
_makeAnyHashableUpcastingToHashableBaseType(
173175
base,
174176
storingResultInto: &self)
@@ -177,7 +179,6 @@ public struct AnyHashable {
177179
@inlinable // FIXME(sil-serialize-all)
178180
internal init<H : Hashable>(_usingDefaultRepresentationOf base: H) {
179181
self._box = _ConcreteHashableBox(base)
180-
self._usedCustomRepresentation = false
181182
}
182183

183184
/// The value wrapped by this instance.
@@ -206,13 +207,11 @@ public struct AnyHashable {
206207
if _box._downCastConditional(into: result) { return true }
207208

208209
#if _runtime(_ObjC)
209-
// If we used a custom representation, bridge to Objective-C and then
210-
// attempt the cast from there.
211-
if _usedCustomRepresentation {
212-
if let value = _bridgeAnythingToObjectiveC(_box._base) as? T {
213-
result.initialize(to: value)
214-
return true
215-
}
210+
// Bridge to Objective-C and then attempt the cast from there.
211+
// FIXME: This should also work without the Objective-C runtime.
212+
if let value = _bridgeAnythingToObjectiveC(_box._base) as? T {
213+
result.initialize(to: value)
214+
return true
216215
}
217216
#endif
218217

@@ -248,42 +247,15 @@ extension AnyHashable : Equatable {
248247
/// - rhs: Another type-erased hashable value.
249248
@inlinable // FIXME(sil-serialize-all)
250249
public static func == (lhs: AnyHashable, rhs: AnyHashable) -> Bool {
251-
// If they're equal, we're done.
252-
if let result = lhs._box._isEqual(to: rhs._box) { return result }
253-
254-
#if _runtime(_ObjC)
255-
// If one used a custom representation but the other did not, bridge
256-
// the one that did *not* use the custom representation to Objective-C:
257-
// if the bridged result has a custom representation, compare those custom
258-
// custom representations.
259-
if lhs._usedCustomRepresentation != rhs._usedCustomRepresentation {
260-
// If the lhs used a custom representation, try comparing against the
261-
// custom representation of the bridged rhs (if there is one).
262-
if lhs._usedCustomRepresentation {
263-
if let customRHS = _getBridgedCustomAnyHashable(rhs._box._base) {
264-
return lhs._box._isEqual(to: customRHS._box) ?? false
265-
}
266-
return false
267-
}
268-
269-
// Otherwise, try comparing the rhs against the custom representation of
270-
// the bridged lhs (if there is one).
271-
if let customLHS = _getBridgedCustomAnyHashable(lhs._box._base) {
272-
return customLHS._box._isEqual(to: rhs._box) ?? false
273-
}
274-
return false
275-
}
276-
#endif
277-
278-
return false
250+
return lhs._box._canonicalBox._isEqual(to: rhs._box._canonicalBox) ?? false
279251
}
280252
}
281253

282254
extension AnyHashable : Hashable {
283255
/// The hash value.
284256
@inlinable
285257
public var hashValue: Int {
286-
return _box._hashValue
258+
return _box._canonicalBox._hashValue
287259
}
288260

289261
/// Hashes the essential components of this value by feeding them into the
@@ -293,7 +265,12 @@ extension AnyHashable : Hashable {
293265
/// of this instance.
294266
@inlinable
295267
public func hash(into hasher: inout Hasher) {
296-
_box._hash(into: &hasher)
268+
_box._canonicalBox._hash(into: &hasher)
269+
}
270+
271+
@inlinable // FIXME(sil-serialize-all)
272+
public func _rawHashValue(seed: (UInt64, UInt64)) -> Int {
273+
return _box._canonicalBox._rawHashValue(_seed: seed)
297274
}
298275
}
299276

stdlib/public/core/Array.swift

+73
Original file line numberDiff line numberDiff line change
@@ -1758,3 +1758,76 @@ extension Array {
17581758
}
17591759
}
17601760
#endif
1761+
1762+
extension Array: _HasCustomAnyHashableRepresentation
1763+
where Element: Hashable {
1764+
public func _toCustomAnyHashable() -> AnyHashable? {
1765+
return AnyHashable(_box: _ArrayAnyHashableBox(self))
1766+
}
1767+
}
1768+
1769+
internal protocol _ArrayAnyHashableProtocol: _AnyHashableBox {
1770+
var count: Int { get }
1771+
subscript(index: Int) -> AnyHashable { get }
1772+
}
1773+
1774+
internal struct _ArrayAnyHashableBox<Element: Hashable>
1775+
: _ArrayAnyHashableProtocol {
1776+
internal let _value: [Element]
1777+
1778+
internal init(_ value: [Element]) {
1779+
self._value = value
1780+
}
1781+
1782+
internal var _base: Any {
1783+
return _value
1784+
}
1785+
1786+
internal var count: Int {
1787+
return _value.count
1788+
}
1789+
1790+
internal subscript(index: Int) -> AnyHashable {
1791+
return _value[index] as AnyHashable
1792+
}
1793+
1794+
func _isEqual(to other: _AnyHashableBox) -> Bool? {
1795+
guard let other = other as? _ArrayAnyHashableProtocol else { return nil }
1796+
guard _value.count == other.count else { return false }
1797+
for i in 0 ..< _value.count {
1798+
if self[i] != other[i] { return false }
1799+
}
1800+
return true
1801+
}
1802+
1803+
var _hashValue: Int {
1804+
var hasher = Hasher()
1805+
_hash(into: &hasher)
1806+
return hasher.finalize()
1807+
}
1808+
1809+
func _hash(into hasher: inout Hasher) {
1810+
hasher.combine(_value.count) // discriminator
1811+
for i in 0 ..< _value.count {
1812+
hasher.combine(self[i])
1813+
}
1814+
}
1815+
1816+
func _rawHashValue(_seed: (UInt64, UInt64)) -> Int {
1817+
var hasher = Hasher(_seed: _seed)
1818+
self._hash(into: &hasher)
1819+
return hasher._finalize()
1820+
}
1821+
1822+
internal func _unbox<T : Hashable>() -> T? {
1823+
return _value as? T
1824+
}
1825+
1826+
internal func _downCastConditional<T>(
1827+
into result: UnsafeMutablePointer<T>
1828+
) -> Bool {
1829+
guard let value = _value as? T else { return false }
1830+
result.initialize(to: value)
1831+
return true
1832+
}
1833+
}

stdlib/public/core/Dictionary.swift

+59
Original file line numberDiff line numberDiff line change
@@ -1471,6 +1471,65 @@ extension Dictionary: Hashable where Value: Hashable {
14711471
}
14721472
}
14731473

1474+
extension Dictionary: _HasCustomAnyHashableRepresentation
1475+
where Value: Hashable {
1476+
public func _toCustomAnyHashable() -> AnyHashable? {
1477+
return AnyHashable(_box: _DictionaryAnyHashableBox(self))
1478+
}
1479+
}
1480+
1481+
internal struct _DictionaryAnyHashableBox<Key: Hashable, Value: Hashable>
1482+
: _AnyHashableBox {
1483+
internal let _value: Dictionary<Key, Value>
1484+
internal let _canonical: Dictionary<AnyHashable, AnyHashable>
1485+
1486+
internal init(_ value: Dictionary<Key, Value>) {
1487+
self._value = value
1488+
self._canonical = value as Dictionary<AnyHashable, AnyHashable>
1489+
}
1490+
1491+
internal var _base: Any {
1492+
return _value
1493+
}
1494+
1495+
internal var _canonicalBox: _AnyHashableBox {
1496+
return _DictionaryAnyHashableBox<AnyHashable, AnyHashable>(_canonical)
1497+
}
1498+
1499+
internal func _isEqual(to other: _AnyHashableBox) -> Bool? {
1500+
guard
1501+
let other = other as? _DictionaryAnyHashableBox<AnyHashable, AnyHashable>
1502+
else {
1503+
return nil
1504+
}
1505+
return _canonical == other._value
1506+
}
1507+
1508+
internal var _hashValue: Int {
1509+
return _canonical.hashValue
1510+
}
1511+
1512+
internal func _hash(into hasher: inout Hasher) {
1513+
_canonical.hash(into: &hasher)
1514+
}
1515+
1516+
internal func _rawHashValue(_seed: (UInt64, UInt64)) -> Int {
1517+
return _canonical._rawHashValue(seed: _seed)
1518+
}
1519+
1520+
internal func _unbox<T: Hashable>() -> T? {
1521+
return _value as? T
1522+
}
1523+
1524+
internal func _downCastConditional<T>(
1525+
into result: UnsafeMutablePointer<T>
1526+
) -> Bool {
1527+
guard let value = _value as? T else { return false }
1528+
result.initialize(to: value)
1529+
return true
1530+
}
1531+
}
1532+
14741533
extension Dictionary: CustomStringConvertible, CustomDebugStringConvertible {
14751534
@inlinable // FIXME(sil-serialize-all)
14761535
internal func _makeDescription() -> String {

0 commit comments

Comments
 (0)