midnight-circuits 7.0.0

Circuit and gadget implementations for Midnight zero-knowledge proofs
Documentation
// This file is part of MIDNIGHT-ZK.
// Copyright (C) Midnight Foundation
// SPDX-License-Identifier: Apache-2.0
// Licensed under the Apache License, Version 2.0 (the "License");
// You may not use this file except in compliance with the License.
// You may obtain a copy of the License at
// http://www.apache.org/licenses/LICENSE-2.0
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

//! Implementation in-circuit of the SHA512 hash function.

#![allow(non_snake_case)]

mod sha512_chip;
mod types;
mod utils;

use midnight_proofs::{circuit::Layouter, plonk::Error};
use sha2::Digest;
pub use sha512_chip::{Sha512Chip, Sha512Config, NB_SHA512_ADVICE_COLS, NB_SHA512_FIXED_COLS};

use crate::{
    instructions::{hash::HashCPU, DecompositionInstructions, HashInstructions},
    types::AssignedByte,
    CircuitField,
};

impl<F: CircuitField> HashCPU<u8, [u8; 64]> for Sha512Chip<F> {
    fn hash(inputs: &[u8]) -> [u8; 64] {
        let output = sha2::Sha512::digest(inputs);
        output.into_iter().collect::<Vec<_>>().try_into().unwrap()
    }
}

impl<F: CircuitField> HashInstructions<F, AssignedByte<F>, [AssignedByte<F>; 64]>
    for Sha512Chip<F>
{
    fn hash(
        &self,
        layouter: &mut impl Layouter<F>,
        inputs: &[AssignedByte<F>],
    ) -> Result<[AssignedByte<F>; 64], Error> {
        let mut output_bytes = Vec::with_capacity(64);

        // We convert each `AssignedPlain<64>` returned by `self.sha512` into 8 bytes.
        for word in self.sha512(layouter, inputs)? {
            let bytes = self.native_gadget.assigned_to_be_bytes(layouter, &word.0, Some(8))?;
            output_bytes.extend(bytes)
        }

        Ok(output_bytes.try_into().unwrap())
    }
}

#[cfg(test)]
mod tests {
    use midnight_curves::Fq as Scalar;

    use crate::{
        field::NativeGadget, hash::sha512::Sha512Chip, instructions::hash::tests::test_hash,
        types::AssignedByte,
    };

    #[test]
    fn test_sha512_hash() {
        fn test_wrapper(input_size: usize, cost_model: bool) {
            test_hash::<
                Scalar,
                AssignedByte<Scalar>,
                [AssignedByte<Scalar>; 64],
                Sha512Chip<Scalar>,
                NativeGadget<Scalar, _, _>,
            >(cost_model, "SHA512", input_size)
        }

        const SHA512_BLOCK_SIZE: usize = 128;
        const SHA512_EDGE_PADDING: usize = 111;

        // Cost model update with input size = 256
        test_wrapper(2 * SHA512_BLOCK_SIZE, true);

        test_wrapper(SHA512_BLOCK_SIZE, false);
        test_wrapper(SHA512_BLOCK_SIZE - 1, false);
        test_wrapper(SHA512_BLOCK_SIZE - 2, false);
        test_wrapper(4 * SHA512_BLOCK_SIZE, false);

        test_wrapper(SHA512_EDGE_PADDING, false);
        test_wrapper(SHA512_EDGE_PADDING - 1, false);

        test_wrapper(0, false);
        test_wrapper(1, false);
        test_wrapper(2, false);
    }
}