]> git.lizzy.rs Git - rust.git/commitdiff
Make `Fingerprint::combine_commutative` associative
authorTomasz Miąsko <tomasz.miasko@gmail.com>
Mon, 3 Jan 2022 00:00:00 +0000 (00:00 +0000)
committerTomasz Miąsko <tomasz.miasko@gmail.com>
Mon, 3 Jan 2022 18:07:29 +0000 (19:07 +0100)
The previous implementation swapped lower and upper 64-bits of a result
of modular addition, so the function was non-associative.

compiler/rustc_data_structures/src/fingerprint.rs
compiler/rustc_data_structures/src/fingerprint/tests.rs [new file with mode: 0644]

index c9af35da4bcef52309b80b5684f21a09a463ab57..18a607f9442895ec12a5230f297c3e1f82a5387b 100644 (file)
@@ -3,6 +3,9 @@
 use std::convert::TryInto;
 use std::hash::{Hash, Hasher};
 
+#[cfg(test)]
+mod tests;
+
 #[derive(Eq, PartialEq, Ord, PartialOrd, Debug, Clone, Copy)]
 #[repr(C)]
 pub struct Fingerprint(u64, u64);
@@ -54,7 +57,7 @@ pub fn combine_commutative(self, other: Fingerprint) -> Fingerprint {
 
         let c = a.wrapping_add(b);
 
-        Fingerprint((c >> 64) as u64, c as u64)
+        Fingerprint(c as u64, (c >> 64) as u64)
     }
 
     pub fn to_hex(&self) -> String {
diff --git a/compiler/rustc_data_structures/src/fingerprint/tests.rs b/compiler/rustc_data_structures/src/fingerprint/tests.rs
new file mode 100644 (file)
index 0000000..9b0783e
--- /dev/null
@@ -0,0 +1,14 @@
+use super::*;
+
+// Check that `combine_commutative` is order independent.
+#[test]
+fn combine_commutative_is_order_independent() {
+    let a = Fingerprint::new(0xf6622fb349898b06, 0x70be9377b2f9c610);
+    let b = Fingerprint::new(0xa9562bf5a2a5303c, 0x67d9b6c82034f13d);
+    let c = Fingerprint::new(0x0d013a27811dbbc3, 0x9a3f7b3d9142ec43);
+    let permutations = [(a, b, c), (a, c, b), (b, a, c), (b, c, a), (c, a, b), (c, b, a)];
+    let f = a.combine_commutative(b).combine_commutative(c);
+    for p in &permutations {
+        assert_eq!(f, p.0.combine_commutative(p.1).combine_commutative(p.2));
+    }
+}