From 134303714bdff50c611b0e3deddb9b84091f174e Mon Sep 17 00:00:00 2001 From: Xiang Zhu Date: Wed, 6 Jul 2022 15:35:32 -0700 Subject: [PATCH] Fix the order assumption of the pubkeys created by Pubkey::new_unique() (#26451) new_unique() does not gurantee the increment order due to the bytes array storage and its eq-partial trait interpreting the bytes in the big-endian way. --- runtime/src/accounts.rs | 20 +++++++++++++++++--- 1 file changed, 17 insertions(+), 3 deletions(-) diff --git a/runtime/src/accounts.rs b/runtime/src/accounts.rs index 2ed469df9cc49a..881e96f2b78a47 100644 --- a/runtime/src/accounts.rs +++ b/runtime/src/accounts.rs @@ -3620,13 +3620,27 @@ mod tests { AccountShrinkThreshold::default(), ); - let pubkey0 = Pubkey::new_unique(); + /* This test assumes pubkey0 < pubkey1 < pubkey2. + * But the keys created with new_unique() does not gurantee this + * order because of the endianness. new_unique() calls add 1 at each + * key generaration as the little endian integer. A pubkey stores its + * value in a 32-byte array bytes, and its eq-partial trait considers + * the lower-address bytes more significant, which is the big-endian + * order. + * So, sort first to ensure the order assumption holds. + */ + let mut keys = vec![]; + for _idx in 0..3 { + keys.push(Pubkey::new_unique()); + } + keys.sort(); + let pubkey2 = keys.pop().unwrap(); + let pubkey1 = keys.pop().unwrap(); + let pubkey0 = keys.pop().unwrap(); let account0 = AccountSharedData::new(42, 0, &Pubkey::default()); accounts.store_slow_uncached(0, &pubkey0, &account0); - let pubkey1 = Pubkey::new_unique(); let account1 = AccountSharedData::new(42, 0, &Pubkey::default()); accounts.store_slow_uncached(0, &pubkey1, &account1); - let pubkey2 = Pubkey::new_unique(); let account2 = AccountSharedData::new(41, 0, &Pubkey::default()); accounts.store_slow_uncached(0, &pubkey2, &account2);