Skip to content

Commit 4714b20

Browse files
committed
Increase the range of MillerRabin::test_random_base to include 2.
1 parent 7c08405 commit 4714b20

File tree

3 files changed

+35
-24
lines changed

3 files changed

+35
-24
lines changed

src/hazmat.rs

+9
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,8 @@ pub use lucas::{lucas_test, AStarBase, BruteForceBase, LucasBase, LucasCheck, Se
1717
pub use miller_rabin::{minimum_mr_iterations, MillerRabin};
1818
pub use sieve::{random_odd_integer, SetBits, SieveFactory, SmallPrimesSieve, SmallPrimesSieveFactory};
1919

20+
use crypto_bigint::{Integer, Word};
21+
2022
/// Possible results of various primality tests.
2123
#[derive(Copy, Clone, Debug, PartialEq, Eq)]
2224
pub enum Primality {
@@ -40,6 +42,13 @@ impl Primality {
4042
}
4143
}
4244

45+
pub(crate) fn equals_primitive<T>(num: &T, primitive: Word) -> bool
46+
where
47+
T: Integer,
48+
{
49+
num.bits_vartime() <= Word::BITS && num.as_ref()[0].0 == primitive
50+
}
51+
4352
#[cfg(test)]
4453
mod tests {
4554
use alloc::format;

src/hazmat/miller_rabin.rs

+23-15
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@ use crypto_bigint::{Integer, Limb, Monty, NonZero as CTNonZero, Odd, PowBoundedE
44
use rand_core::CryptoRng;
55

66
use super::{
7+
equals_primitive,
78
float::{floor_sqrt, two_powf_upper_bound, two_powi},
89
Primality,
910
};
@@ -107,25 +108,30 @@ impl<T: Integer + RandomMod> MillerRabin<T> {
107108
self.test(&T::from_limb_like(Limb::from(2u32), &self.candidate))
108109
}
109110

110-
/// Perform a Miller-Rabin check with a random base (in the range `[3, candidate-2]`)
111-
/// drawn using the provided RNG.
111+
/// Perform a Miller-Rabin check with a random base (in the range `[2, candidate-2]`,
112+
/// because the test holds trivially for bases 1 or `candidate-1`) drawn using the provided RNG.
112113
///
113-
/// Note: panics if `candidate == 3` (so the range above is empty).
114+
/// *Note:* if `candidate == 1` or `candidate == 3` (which would make the above range contain no numbers)
115+
/// no check is actually performed, since we already know the result
116+
/// ([`Primality::Composite`] for 1, [`Primality::Prime`] for 3).
114117
pub fn test_random_base<R: CryptoRng + ?Sized>(&self, rng: &mut R) -> Primality {
115-
// We sample a random base from the range `[3, candidate-2]`:
116-
// - we have a separate method for base 2;
117-
// - the test holds trivially for bases 1 or `candidate-1`.
118-
if self.candidate.bits() < 3 {
119-
panic!("No suitable random base possible when `candidate == 3`; use the base 2 test.")
118+
if equals_primitive(&self.candidate, 1) {
119+
// As per standard convention
120+
return Primality::Composite;
120121
}
121122

122-
let range = self.candidate.wrapping_sub(&T::from(4u32));
123-
// Can unwrap here since `candidate` is odd, and `candidate >= 4` (as checked above)
123+
if equals_primitive(&self.candidate, 3) {
124+
// As per standard convention
125+
return Primality::Prime;
126+
}
127+
128+
// The candidate is odd, so by now it is guaranteed to be >= 5.
129+
let range = self.candidate.wrapping_sub(&T::from(3u32));
124130
let range_nonzero = CTNonZero::new(range).expect("the range should be non-zero by construction");
125131
// This should not overflow as long as `random_mod()` behaves according to the contract
126132
// (that is, returns a number within the given range).
127133
let random = T::random_mod(rng, &range_nonzero)
128-
.checked_add(&T::from(3u32))
134+
.checked_add(&T::from(2u32))
129135
.expect("addition should not overflow by construction");
130136
self.test(&random)
131137
}
@@ -236,7 +242,7 @@ mod tests {
236242
use num_prime::nt_funcs::is_prime64;
237243

238244
use super::{minimum_mr_iterations, MillerRabin};
239-
use crate::hazmat::{primes, pseudoprimes, random_odd_integer, SetBits, SmallPrimesSieve};
245+
use crate::hazmat::{primes, pseudoprimes, random_odd_integer, Primality, SetBits, SmallPrimesSieve};
240246

241247
#[test]
242248
fn miller_rabin_derived_traits() {
@@ -246,10 +252,12 @@ mod tests {
246252
}
247253

248254
#[test]
249-
#[should_panic(expected = "No suitable random base possible when `candidate == 3`; use the base 2 test.")]
250-
fn random_base_range_check() {
255+
fn random_base_corner_cases() {
256+
let mr = MillerRabin::new(Odd::new(U64::from(1u32)).unwrap());
257+
assert!(mr.test_random_base(&mut OsRng.unwrap_err()) == Primality::Composite);
258+
251259
let mr = MillerRabin::new(Odd::new(U64::from(3u32)).unwrap());
252-
mr.test_random_base(&mut OsRng.unwrap_err());
260+
assert!(mr.test_random_base(&mut OsRng.unwrap_err()) == Primality::Prime);
253261
}
254262

255263
fn is_spsp(num: u32) -> bool {

src/presets.rs

+3-9
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,11 @@
1-
use crypto_bigint::{Integer, Odd, RandomBits, RandomMod, Word};
1+
use crypto_bigint::{Integer, Odd, RandomBits, RandomMod};
22
use rand_core::CryptoRng;
33

44
use crate::{
55
generic::sieve_and_find,
66
hazmat::{
7-
lucas_test, AStarBase, LucasCheck, MillerRabin, Primality, SelfridgeBase, SetBits, SmallPrimesSieveFactory,
7+
equals_primitive, lucas_test, AStarBase, LucasCheck, MillerRabin, Primality, SelfridgeBase, SetBits,
8+
SmallPrimesSieveFactory,
89
},
910
};
1011

@@ -35,13 +36,6 @@ where
3536
.expect("will produce a result eventually")
3637
}
3738

38-
fn equals_primitive<T>(num: &T, primitive: Word) -> bool
39-
where
40-
T: Integer,
41-
{
42-
num.bits_vartime() <= u16::BITS && num.as_ref()[0].0 == primitive
43-
}
44-
4539
/// Checks if the given number is prime.
4640
///
4741
/// Performed tests:

0 commit comments

Comments
 (0)