@@ -39,20 +39,29 @@ public protocol _HasCustomAnyHashableRepresentation {
39
39
40
40
@usableFromInline // FIXME(sil-serialize-all)
41
41
internal protocol _AnyHashableBox {
42
- func _unbox < T : Hashable > ( ) -> T ?
42
+ var _canonicalBox : _AnyHashableBox { get }
43
43
44
44
/// Determine whether values in the boxes are equivalent.
45
45
///
46
+ /// - Precondition: `self` and `box` are in canonical form.
46
47
/// - Returns: `nil` to indicate that the boxes store different types, so
47
48
/// no comparison is possible. Otherwise, contains the result of `==`.
48
- func _isEqual( to: _AnyHashableBox ) -> Bool ?
49
+ func _isEqual( to box : _AnyHashableBox ) -> Bool ?
49
50
var _hashValue : Int { get }
50
51
func _hash( into hasher: inout Hasher )
52
+ func _rawHashValue( _seed: ( UInt64 , UInt64 ) ) -> Int
51
53
52
54
var _base : Any { get }
55
+ func _unbox< T: Hashable > ( ) -> T ?
53
56
func _downCastConditional< T> ( into result: UnsafeMutablePointer < T > ) -> Bool
54
57
}
55
58
59
+ extension _AnyHashableBox {
60
+ var _canonicalBox : _AnyHashableBox {
61
+ return self
62
+ }
63
+ }
64
+
56
65
@_fixed_layout // FIXME(sil-serialize-all)
57
66
@usableFromInline // FIXME(sil-serialize-all)
58
67
internal struct _ConcreteHashableBox < Base : Hashable > : _AnyHashableBox {
@@ -87,6 +96,11 @@ internal struct _ConcreteHashableBox<Base : Hashable> : _AnyHashableBox {
87
96
_baseHashable. hash ( into: & hasher)
88
97
}
89
98
99
+ @inlinable // FIXME(sil-serialize-all)
100
+ func _rawHashValue( _seed: ( UInt64 , UInt64 ) ) -> Int {
101
+ return _baseHashable. _rawHashValue ( seed: _seed)
102
+ }
103
+
90
104
@inlinable // FIXME(sil-serialize-all)
91
105
internal var _base : Any {
92
106
return _baseHashable
@@ -101,19 +115,6 @@ internal struct _ConcreteHashableBox<Base : Hashable> : _AnyHashableBox {
101
115
}
102
116
}
103
117
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
-
117
118
/// A type-erased hashable value.
118
119
///
119
120
/// The `AnyHashable` type forwards equality comparisons and hashing operations
@@ -137,8 +138,11 @@ internal func _getBridgedCustomAnyHashable<T>(_ value: T) -> AnyHashable? {
137
138
public struct AnyHashable {
138
139
@usableFromInline // FIXME(sil-serialize-all)
139
140
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
+ }
142
146
143
147
/// Creates a type-erased hashable value that wraps the given instance.
144
148
///
@@ -160,15 +164,13 @@ public struct AnyHashable {
160
164
/// - Parameter base: A hashable value to wrap.
161
165
@inlinable // FIXME(sil-serialize-all)
162
166
public init < H : Hashable > ( _ base: H ) {
163
- if let customRepresentation =
167
+ if let custom =
164
168
( base as? _HasCustomAnyHashableRepresentation ) ? . _toCustomAnyHashable ( ) {
165
- self = customRepresentation
166
- self . _usedCustomRepresentation = true
169
+ self = custom
167
170
return
168
171
}
169
172
170
- self . _box = _ConcreteHashableBox ( 0 as Int )
171
- self . _usedCustomRepresentation = false
173
+ self . init ( _box: _ConcreteHashableBox ( false ) ) // Dummy value
172
174
_makeAnyHashableUpcastingToHashableBaseType (
173
175
base,
174
176
storingResultInto: & self )
@@ -177,7 +179,6 @@ public struct AnyHashable {
177
179
@inlinable // FIXME(sil-serialize-all)
178
180
internal init < H : Hashable > ( _usingDefaultRepresentationOf base: H ) {
179
181
self . _box = _ConcreteHashableBox ( base)
180
- self . _usedCustomRepresentation = false
181
182
}
182
183
183
184
/// The value wrapped by this instance.
@@ -206,13 +207,11 @@ public struct AnyHashable {
206
207
if _box. _downCastConditional ( into: result) { return true }
207
208
208
209
#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
216
215
}
217
216
#endif
218
217
@@ -248,42 +247,15 @@ extension AnyHashable : Equatable {
248
247
/// - rhs: Another type-erased hashable value.
249
248
@inlinable // FIXME(sil-serialize-all)
250
249
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
279
251
}
280
252
}
281
253
282
254
extension AnyHashable : Hashable {
283
255
/// The hash value.
284
256
@inlinable
285
257
public var hashValue : Int {
286
- return _box. _hashValue
258
+ return _box. _canonicalBox . _hashValue
287
259
}
288
260
289
261
/// Hashes the essential components of this value by feeding them into the
@@ -293,7 +265,12 @@ extension AnyHashable : Hashable {
293
265
/// of this instance.
294
266
@inlinable
295
267
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)
297
274
}
298
275
}
299
276
0 commit comments