1#![doc = include_str!("../README.md")]
2
3#![no_std]
33#![cfg_attr(docsrs, feature(doc_cfg))]
34#![forbid(unsafe_code, clippy::unwrap_used)]
35#![warn(missing_docs, rust_2018_idioms)]
36#![doc(
37 html_logo_url = "https://raw.githubusercontent.com/RustCrypto/media/8f1a9894/logo.svg",
38 html_favicon_url = "https://raw.githubusercontent.com/RustCrypto/media/8f1a9894/logo.svg"
39)]
40
41use crypto_bigint::{ArrayEncoding, ByteArray, Integer};
42use hmac::{
43 digest::{
44 core_api::BlockSizeUser, generic_array::GenericArray, Digest, FixedOutput,
45 FixedOutputReset, Mac,
46 },
47 SimpleHmac,
48};
49use zeroize::{Zeroize, Zeroizing};
50
51#[inline]
60pub fn generate_k<D, I>(x: &I, n: &I, h: &ByteArray<I>, data: &[u8]) -> Zeroizing<I>
61where
62 D: Digest + BlockSizeUser + FixedOutput<OutputSize = I::ByteSize> + FixedOutputReset,
63 I: ArrayEncoding + Integer + Zeroize,
64{
65 let mut x = x.to_be_byte_array();
66 let mut hmac_drbg = HmacDrbg::<D>::new(&x, h, data);
67 x.zeroize();
68
69 loop {
70 let mut bytes = ByteArray::<I>::default();
71 hmac_drbg.fill_bytes(&mut bytes);
72 let k = I::from_be_byte_array(bytes);
73
74 if (!k.is_zero() & k.ct_lt(n)).into() {
75 return Zeroizing::new(k);
76 }
77 }
78}
79
80pub struct HmacDrbg<D>
87where
88 D: Digest + BlockSizeUser + FixedOutputReset,
89{
90 k: SimpleHmac<D>,
92
93 v: GenericArray<u8, D::OutputSize>,
95}
96
97impl<D> HmacDrbg<D>
98where
99 D: Digest + BlockSizeUser + FixedOutputReset,
100{
101 pub fn new(entropy_input: &[u8], nonce: &[u8], additional_data: &[u8]) -> Self {
103 let mut k = SimpleHmac::new(&Default::default());
104 let mut v = GenericArray::default();
105
106 for b in &mut v {
107 *b = 0x01;
108 }
109
110 for i in 0..=1 {
111 k.update(&v);
112 k.update(&[i]);
113 k.update(entropy_input);
114 k.update(nonce);
115 k.update(additional_data);
116 k = SimpleHmac::new_from_slice(&k.finalize().into_bytes()).expect("HMAC error");
117
118 k.update(&v);
120 v = k.finalize_reset().into_bytes();
121 }
122
123 Self { k, v }
124 }
125
126 pub fn fill_bytes(&mut self, out: &mut [u8]) {
128 for out_chunk in out.chunks_mut(self.v.len()) {
129 self.k.update(&self.v);
130 self.v = self.k.finalize_reset().into_bytes();
131 out_chunk.copy_from_slice(&self.v[..out_chunk.len()]);
132 }
133
134 self.k.update(&self.v);
135 self.k.update(&[0x00]);
136 self.k =
137 SimpleHmac::new_from_slice(&self.k.finalize_reset().into_bytes()).expect("HMAC error");
138 self.k.update(&self.v);
139 self.v = self.k.finalize_reset().into_bytes();
140 }
141}