#![allow(non_snake_case)]
mod sha256_chip;
mod sha256_varlen;
mod types;
pub(crate) mod utils;
use midnight_proofs::{circuit::Layouter, plonk::Error};
use sha2::Digest;
pub use sha256_chip::{Sha256Chip, Sha256Config, NB_SHA256_ADVICE_COLS, NB_SHA256_FIXED_COLS};
pub use sha256_varlen::VarLenSha256Gadget;
use crate::{
instructions::{
hash::{HashCPU, VarHashInstructions},
DecompositionInstructions, HashInstructions,
},
types::AssignedByte,
vec::AssignedVector,
CircuitField,
};
impl<F: CircuitField> HashCPU<u8, [u8; 32]> for Sha256Chip<F> {
fn hash(inputs: &[u8]) -> [u8; 32] {
let output = sha2::Sha256::digest(inputs);
output.into_iter().collect::<Vec<_>>().try_into().unwrap()
}
}
impl<F: CircuitField> HashInstructions<F, AssignedByte<F>, [AssignedByte<F>; 32]>
for Sha256Chip<F>
{
fn hash(
&self,
layouter: &mut impl Layouter<F>,
inputs: &[AssignedByte<F>],
) -> Result<[AssignedByte<F>; 32], Error> {
let mut output_bytes = Vec::with_capacity(32);
for word in self.sha256(layouter, inputs)? {
let bytes = self.native_gadget.assigned_to_be_bytes(layouter, &word.0, Some(4))?;
output_bytes.extend(bytes)
}
Ok(output_bytes.try_into().unwrap())
}
}
impl<F: CircuitField> HashCPU<u8, [u8; 32]> for VarLenSha256Gadget<F> {
fn hash(inputs: &[u8]) -> [u8; 32] {
let output = sha2::Sha256::digest(inputs);
output.into_iter().collect::<Vec<_>>().try_into().unwrap()
}
}
impl<F: CircuitField, const MAX_LEN: usize>
VarHashInstructions<F, MAX_LEN, AssignedByte<F>, [AssignedByte<F>; 32], 64>
for VarLenSha256Gadget<F>
{
fn varhash(
&self,
layouter: &mut impl Layouter<F>,
inputs: &AssignedVector<F, AssignedByte<F>, MAX_LEN, 64>,
) -> Result<[AssignedByte<F>; 32], Error> {
let mut output_bytes = Vec::with_capacity(32);
for word in self.sha256_varlen(layouter, inputs)? {
let bytes =
self.sha256chip.native_gadget.assigned_to_be_bytes(layouter, &word.0, Some(4))?;
output_bytes.extend(bytes)
}
Ok(output_bytes.try_into().unwrap())
}
}
#[cfg(test)]
mod tests {
use midnight_curves::Fq as Scalar;
use super::sha256_varlen::VarLenSha256Gadget;
use crate::{
field::NativeGadget,
hash::sha256::Sha256Chip,
instructions::hash::tests::{test_hash, test_varhash},
types::AssignedByte,
};
#[test]
fn test_sha256_hash() {
fn test_wrapper(input_size: usize, cost_model: bool) {
test_hash::<
Scalar,
AssignedByte<Scalar>,
[AssignedByte<Scalar>; 32],
Sha256Chip<Scalar>,
NativeGadget<Scalar, _, _>,
>(cost_model, "SHA256", input_size)
}
const SHA256_BLOCK_SIZE: usize = 64;
const SHA256_EDGE_PADDING: usize = 55;
test_wrapper(4 * SHA256_BLOCK_SIZE, true);
test_wrapper(SHA256_BLOCK_SIZE, false);
test_wrapper(SHA256_BLOCK_SIZE - 1, false);
test_wrapper(SHA256_BLOCK_SIZE - 2, false);
test_wrapper(2 * SHA256_BLOCK_SIZE, false);
test_wrapper(SHA256_EDGE_PADDING, false);
test_wrapper(SHA256_EDGE_PADDING - 1, false);
test_wrapper(0, false);
test_wrapper(1, false);
test_wrapper(2, false);
}
#[test]
fn test_sha256_varhash() {
fn test_wrapper<const M: usize>(input_size: usize, cost_model: bool) {
test_varhash::<
Scalar,
AssignedByte<Scalar>,
[AssignedByte<Scalar>; 32],
VarLenSha256Gadget<Scalar>,
M,
64,
>(cost_model, "VarSHA256", input_size)
}
test_wrapper::<512>(64, false);
test_wrapper::<512>(63, false);
test_wrapper::<256>(128, true);
test_wrapper::<256>(127, false);
test_wrapper::<128>(55, false); test_wrapper::<128>(56, false);
test_wrapper::<128>(0, false);
test_wrapper::<128>(1, false);
}
}