Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add feature = "zeroize" #92

Open
wants to merge 1 commit into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 5 additions & 0 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -53,6 +53,11 @@ version = "1.0"
default-features = false
features = ["std"]

[dependencies.zeroize]
optional = true
version = "0.6"
default-features = false

[dev-dependencies.serde_test]
version = "1.0"

Expand Down
4 changes: 4 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,10 @@ Implementations for `i128` and `u128` are only available with Rust 1.26 and
later. The build script automatically detects this, but you can make it
mandatory by enabling the `i128` crate feature.

The [`zeroize`](https://crates.io/crates/zeroize) feature causes
`BigInt`/`BigUint` to be wiped from memory on `Drop`, which may improve security
in some applications. `zeroize` requires Rust 1.31.

## Releases

Release notes are available in [RELEASES.md](RELEASES.md).
Expand Down
3 changes: 3 additions & 0 deletions ci/test_full.sh
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,9 @@ fi
if [[ "$TRAVIS_RUST_VERSION" =~ ^(nightly|beta|stable|1.26.0)$ ]]; then
FEATURES="$FEATURES i128"
fi
if [[ "$TRAVIS_RUST_VERSION" =~ ^(nightly|beta|stable|1.31.0)$ ]]; then
FEATURES="$FEATURES zeroize"
fi

# num-bigint should build and test everywhere.
cargo build --verbose
Expand Down
4 changes: 4 additions & 0 deletions src/algorithms.rs
Original file line number Diff line number Diff line change
Expand Up @@ -610,6 +610,7 @@ pub fn ilog2<T: traits::PrimInt>(v: T) -> usize {
pub fn biguint_shl(n: Cow<BigUint>, bits: usize) -> BigUint {
let n_unit = bits / big_digit::BITS;
let mut data = match n_unit {
#[cfg(not(feature = "zeroize"))]
0 => n.into_owned().data,
_ => {
let len = n_unit + n.data.len() + 1;
Expand Down Expand Up @@ -644,6 +645,9 @@ pub fn biguint_shr(n: Cow<BigUint>, bits: usize) -> BigUint {
}
let mut data = match n {
Cow::Borrowed(n) => n.data[n_unit..].to_vec(),
#[cfg(feature = "zeroize")]
Cow::Owned(mut n) => n.data[n_unit..].to_vec(),
#[cfg(not(feature = "zeroize"))]
Cow::Owned(mut n) => {
n.data.drain(..n_unit);
n.data
Expand Down
8 changes: 8 additions & 0 deletions src/biguint.rs
Original file line number Diff line number Diff line change
Expand Up @@ -79,6 +79,14 @@ impl Default for BigUint {
}
}

#[cfg(feature = "zeroize")]
impl Drop for BigUint {
fn drop(&mut self) {
use zeroize::Zeroize;
self.data.zeroize();
}
}

impl fmt::Display for BigUint {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
f.pad_integral(true, "", &self.to_str_radix(10))
Expand Down
2 changes: 2 additions & 0 deletions src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -84,6 +84,8 @@
extern crate rand;
#[cfg(feature = "serde")]
extern crate serde;
#[cfg(feature = "zeroize")]
extern crate zeroize;

extern crate num_integer as integer;
extern crate num_traits as traits;
Expand Down
11 changes: 5 additions & 6 deletions src/monty.rs
Original file line number Diff line number Diff line change
Expand Up @@ -57,13 +57,12 @@ impl<'a> MontyReducer<'a> {
//
// Reference:
// Brent & Zimmermann, Modern Computer Arithmetic, v0.5.9, Algorithm 2.6
fn monty_redc(a: BigUint, mr: &MontyReducer) -> BigUint {
let mut c = a.data;
fn monty_redc(mut a: BigUint, mr: &MontyReducer) -> BigUint {
let n = &mr.n.data;
let n_size = n.len();

// Allocate sufficient work space
c.resize(2 * n_size + 2, 0);
a.data.resize(2 * n_size + 2, 0);

// β is the size of a word, in this case 32 bits. So "a mod β" is
// equivalent to masking a to 32 bits.
Expand All @@ -73,15 +72,15 @@ fn monty_redc(a: BigUint, mr: &MontyReducer) -> BigUint {
// 1: for i = 0 to (n-1)
for i in 0..n_size {
// 2: q_i <- mu*c_i mod β
let q_i = c[i].wrapping_mul(mu);
let q_i = a.data[i].wrapping_mul(mu);

// 3: C <- C + q_i * N * β^i
super::algorithms::mac_digit(&mut c[i..], n, q_i);
super::algorithms::mac_digit(&mut a.data[i..], n, q_i);
}

// 4: R <- C * β^(-n)
// This is an n-word bitshift, equivalent to skipping n words.
let ret = BigUint::new(c[n_size..].to_vec());
let ret = BigUint::new(a.data[n_size..].to_vec());

// 5: if R >= β^n then return R-N else return R.
if &ret < mr.n {
Expand Down