1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
use core::fmt::{Debug, Display, Formatter};
use std::slice;
use arrayref::array_ref;
use generic_array::GenericArray;
use sha2::{compress256, Digest as ShaDigest, Sha256};
use typenum::U64;
use crate::{fp::Fp, fp4::Fp4};
pub const DIGEST_WORDS: usize = 8;
#[derive(Clone, Copy, Default, Eq, Hash, Ord, PartialOrd, PartialEq)]
pub struct Digest(pub [u32; DIGEST_WORDS]);
static INIT_256: [u32; DIGEST_WORDS] = [
0x6a09e667, 0xbb67ae85, 0x3c6ef372, 0xa54ff53a, 0x510e527f, 0x9b05688c, 0x1f83d9ab, 0x5be0cd19,
];
fn set_word(buf: &mut [u8], idx: usize, word: u32) {
buf[(4 * idx)..(4 * idx + 4)].copy_from_slice(&word.to_be_bytes());
}
impl Digest {
pub fn from_u32s(words: &[u32]) -> Self {
Digest(words.try_into().expect("slice with incorrect length"))
}
pub fn hash_bytes(bytes: &[u8]) -> Self {
let mut hasher = Sha256::new();
hasher.update(bytes);
let raw_ret = hasher.finalize();
let mut ret = [0 as u32; DIGEST_WORDS];
for i in 0..DIGEST_WORDS {
ret[i] = u32::from_be_bytes(*array_ref![raw_ret, i * 4, 4]);
}
Digest(ret)
}
pub fn hash_fps(fps: &[Fp]) -> Self {
let mut state = INIT_256;
let mut block: GenericArray<u8, U64> = GenericArray::default();
let mut off = 0;
for i in 0..fps.len() {
set_word(block.as_mut_slice(), off, u32::from(fps[i]));
off += 1;
if off == 16 {
compress256(&mut state, slice::from_ref(&block));
off = 0;
}
}
if off != 0 {
block[off * 4..].fill(0);
compress256(&mut state, slice::from_ref(&block));
}
Digest(state)
}
pub fn hash_fp4s(fp4s: &[Fp4]) -> Self {
let mut flat: Vec<Fp> = vec![];
for i in 0..fp4s.len() {
for j in 0..4 {
flat.push(fp4s[i].elems()[j]);
}
}
return Self::hash_fps(&flat);
}
pub fn hash_pair(a: &Digest, b: &Digest) -> Self {
let mut state = INIT_256;
let mut block: GenericArray<u8, U64> = GenericArray::default();
for i in 0..8 {
set_word(block.as_mut_slice(), i, a.0[i]);
set_word(block.as_mut_slice(), 8 + i, b.0[i]);
}
compress256(&mut state, slice::from_ref(&block));
Digest(state)
}
}
impl Digest {
pub fn new(data: [u32; DIGEST_WORDS]) -> Self {
Self(data)
}
pub fn as_slice(&self) -> &[u32] {
&self.0
}
}
impl Display for Digest {
fn fmt(&self, f: &mut Formatter) -> core::fmt::Result {
for word in self.0 {
core::write!(f, "{:08x?}", word)?;
}
Ok(())
}
}
impl Debug for Digest {
fn fmt(&self, f: &mut Formatter) -> core::fmt::Result {
for word in self.0 {
core::write!(f, "{:08x?}", word)?;
}
Ok(())
}
}