@@ -238,15 +238,21 @@ pub trait ToStableHashKey<HCX> {
238
238
/// The associated constant `CAN_USE_UNSTABLE_SORT` denotes whether
239
239
/// unstable sorting can be used for this type. Set to true if and
240
240
/// only if `a == b` implies `a` and `b` are fully indistinguishable.
241
- ///
242
- /// **Be careful when implementing this trait, as an incorrect
243
- /// implementation can cause miscompilation!**
244
241
pub trait StableOrd : Ord {
245
242
const CAN_USE_UNSTABLE_SORT : bool ;
243
+
244
+ /// Marker to ensure that implementors have carefully considered
245
+ /// whether their `Ord` implementation obeys this trait's contract.
246
+ const THIS_IMPLEMENTATION_HAS_BEEN_TRIPLE_CHECKED : ( ) ;
246
247
}
247
248
248
249
impl < T : StableOrd > StableOrd for & T {
249
250
const CAN_USE_UNSTABLE_SORT : bool = T :: CAN_USE_UNSTABLE_SORT ;
251
+
252
+ // Ordering of a reference is exactly that of the referent, and since
253
+ // the ordering of the referet is stable so must be the ordering of the
254
+ // reference.
255
+ const THIS_IMPLEMENTATION_HAS_BEEN_TRIPLE_CHECKED : ( ) = ( ) ;
250
256
}
251
257
252
258
/// This is a companion trait to `StableOrd`. Some types like `Symbol` can be
@@ -295,6 +301,10 @@ macro_rules! impl_stable_traits_for_trivial_type {
295
301
296
302
impl $crate:: stable_hasher:: StableOrd for $t {
297
303
const CAN_USE_UNSTABLE_SORT : bool = true ;
304
+
305
+ // Encoding and decoding doesn't change the bytes of trivial types
306
+ // and `Ord::cmp` depends only on those bytes.
307
+ const THIS_IMPLEMENTATION_HAS_BEEN_TRIPLE_CHECKED : ( ) = ( ) ;
298
308
}
299
309
} ;
300
310
}
@@ -332,6 +342,10 @@ impl<CTX> HashStable<CTX> for Hash128 {
332
342
333
343
impl StableOrd for Hash128 {
334
344
const CAN_USE_UNSTABLE_SORT : bool = true ;
345
+
346
+ // Encoding and decoding doesn't change the bytes of `Hash128`
347
+ // and `Ord::cmp` depends only on those bytes.
348
+ const THIS_IMPLEMENTATION_HAS_BEEN_TRIPLE_CHECKED : ( ) = ( ) ;
335
349
}
336
350
337
351
impl < CTX > HashStable < CTX > for ! {
@@ -397,6 +411,10 @@ impl<T1: HashStable<CTX>, T2: HashStable<CTX>, CTX> HashStable<CTX> for (T1, T2)
397
411
398
412
impl < T1 : StableOrd , T2 : StableOrd > StableOrd for ( T1 , T2 ) {
399
413
const CAN_USE_UNSTABLE_SORT : bool = T1 :: CAN_USE_UNSTABLE_SORT && T2 :: CAN_USE_UNSTABLE_SORT ;
414
+
415
+ // Ordering of tuples is a pure function of their elements' ordering, and since
416
+ // the ordering of each element is stable so must be the ordering of the tuple.
417
+ const THIS_IMPLEMENTATION_HAS_BEEN_TRIPLE_CHECKED : ( ) = ( ) ;
400
418
}
401
419
402
420
impl < T1 , T2 , T3 , CTX > HashStable < CTX > for ( T1 , T2 , T3 )
@@ -416,6 +434,10 @@ where
416
434
impl < T1 : StableOrd , T2 : StableOrd , T3 : StableOrd > StableOrd for ( T1 , T2 , T3 ) {
417
435
const CAN_USE_UNSTABLE_SORT : bool =
418
436
T1 :: CAN_USE_UNSTABLE_SORT && T2 :: CAN_USE_UNSTABLE_SORT && T3 :: CAN_USE_UNSTABLE_SORT ;
437
+
438
+ // Ordering of tuples is a pure function of their elements' ordering, and since
439
+ // the ordering of each element is stable so must be the ordering of the tuple.
440
+ const THIS_IMPLEMENTATION_HAS_BEEN_TRIPLE_CHECKED : ( ) = ( ) ;
419
441
}
420
442
421
443
impl < T1 , T2 , T3 , T4 , CTX > HashStable < CTX > for ( T1 , T2 , T3 , T4 )
@@ -439,6 +461,10 @@ impl<T1: StableOrd, T2: StableOrd, T3: StableOrd, T4: StableOrd> StableOrd for (
439
461
&& T2 :: CAN_USE_UNSTABLE_SORT
440
462
&& T3 :: CAN_USE_UNSTABLE_SORT
441
463
&& T4 :: CAN_USE_UNSTABLE_SORT ;
464
+
465
+ // Ordering of tuples is a pure function of their elements' ordering, and since
466
+ // the ordering of each element is stable so must be the ordering of the tuple.
467
+ const THIS_IMPLEMENTATION_HAS_BEEN_TRIPLE_CHECKED : ( ) = ( ) ;
442
468
}
443
469
444
470
impl < T : HashStable < CTX > , CTX > HashStable < CTX > for [ T ] {
@@ -533,6 +559,10 @@ impl<CTX> HashStable<CTX> for str {
533
559
534
560
impl StableOrd for & str {
535
561
const CAN_USE_UNSTABLE_SORT : bool = true ;
562
+
563
+ // Encoding and decoding doesn't change the bytes of string slices
564
+ // and `Ord::cmp` depends only on those bytes.
565
+ const THIS_IMPLEMENTATION_HAS_BEEN_TRIPLE_CHECKED : ( ) = ( ) ;
536
566
}
537
567
538
568
impl < CTX > HashStable < CTX > for String {
@@ -542,10 +572,12 @@ impl<CTX> HashStable<CTX> for String {
542
572
}
543
573
}
544
574
545
- // String comparison only depends on their contents and the
546
- // contents are not changed by (de-)serialization.
547
575
impl StableOrd for String {
548
576
const CAN_USE_UNSTABLE_SORT : bool = true ;
577
+
578
+ // String comparison only depends on their contents and the
579
+ // contents are not changed by (de-)serialization.
580
+ const THIS_IMPLEMENTATION_HAS_BEEN_TRIPLE_CHECKED : ( ) = ( ) ;
549
581
}
550
582
551
583
impl < HCX > ToStableHashKey < HCX > for String {
@@ -571,9 +603,11 @@ impl<CTX> HashStable<CTX> for bool {
571
603
}
572
604
}
573
605
574
- // sort order of bools is not changed by (de-)serialization.
575
606
impl StableOrd for bool {
576
607
const CAN_USE_UNSTABLE_SORT : bool = true ;
608
+
609
+ // sort order of bools is not changed by (de-)serialization.
610
+ const THIS_IMPLEMENTATION_HAS_BEEN_TRIPLE_CHECKED : ( ) = ( ) ;
577
611
}
578
612
579
613
impl < T , CTX > HashStable < CTX > for Option < T >
@@ -591,9 +625,11 @@ where
591
625
}
592
626
}
593
627
594
- // the Option wrapper does not add instability to comparison.
595
628
impl < T : StableOrd > StableOrd for Option < T > {
596
629
const CAN_USE_UNSTABLE_SORT : bool = T :: CAN_USE_UNSTABLE_SORT ;
630
+
631
+ // the Option wrapper does not add instability to comparison.
632
+ const THIS_IMPLEMENTATION_HAS_BEEN_TRIPLE_CHECKED : ( ) = ( ) ;
597
633
}
598
634
599
635
impl < T1 , T2 , CTX > HashStable < CTX > for Result < T1 , T2 >
0 commit comments