From d2cce9649ec7ee4168fad5ea6809c86af69440ce Mon Sep 17 00:00:00 2001 From: TomAFrench Date: Mon, 23 Sep 2024 20:07:47 +0000 Subject: [PATCH 1/2] feat: remove unnecessary branching in keccak impl --- noir_stdlib/src/hash/keccak.nr | 42 ++++++++++++---------------------- 1 file changed, 15 insertions(+), 27 deletions(-) diff --git a/noir_stdlib/src/hash/keccak.nr b/noir_stdlib/src/hash/keccak.nr index 96e53429ac5..1675e111026 100644 --- a/noir_stdlib/src/hash/keccak.nr +++ b/noir_stdlib/src/hash/keccak.nr @@ -1,10 +1,10 @@ use crate::collections::vec::Vec; use crate::runtime::is_unconstrained; -global LIMBS_PER_BLOCK = 17; //BLOCK_SIZE / 8; +global BLOCK_SIZE_IN_BYTES: u32 = 136; //(1600 - BITS * 2) / WORD_SIZE; +global WORD_SIZE: u32 = 8; // Limbs are made up of u64s so 8 bytes each. +global LIMBS_PER_BLOCK: u32 = BLOCK_SIZE_IN_BYTES / WORD_SIZE; global NUM_KECCAK_LANES: u32 = 25; -global BLOCK_SIZE = 136; //(1600 - BITS * 2) / WORD_SIZE; -global WORD_SIZE = 8; #[foreign(keccakf1600)] fn keccakf1600(input: [u64; 25]) -> [u64; 25] {} @@ -12,7 +12,7 @@ fn keccakf1600(input: [u64; 25]) -> [u64; 25] {} #[no_predicates] pub(crate) fn keccak256(input: [u8; N], message_size: u32) -> [u8; 32] { assert(N >= message_size); - let mut block_bytes = [0; BLOCK_SIZE]; + let mut block_bytes = [0; BLOCK_SIZE_IN_BYTES]; if is_unconstrained() { for i in 0..message_size { block_bytes[i] = input[i]; @@ -26,38 +26,26 @@ pub(crate) fn keccak256(input: [u8; N], message_size: u32) -> [u8; 3 } //1. format_input_lanes - let max_blocks = (N + BLOCK_SIZE) / BLOCK_SIZE; + let max_blocks = (N + BLOCK_SIZE_IN_BYTES) / BLOCK_SIZE_IN_BYTES; //maximum number of bytes to hash - let max_blocks_length = (BLOCK_SIZE * (max_blocks)); - let real_max_blocks = (message_size + BLOCK_SIZE) / BLOCK_SIZE; - let real_blocks_bytes = real_max_blocks * BLOCK_SIZE; + let max_blocks_length = (BLOCK_SIZE_IN_BYTES * max_blocks); + let real_max_blocks = (message_size + BLOCK_SIZE_IN_BYTES) / BLOCK_SIZE_IN_BYTES; + let real_blocks_bytes = real_max_blocks * BLOCK_SIZE_IN_BYTES; block_bytes[message_size] = 1; block_bytes[real_blocks_bytes - 1] = 0x80; - let num_limbs = max_blocks * LIMBS_PER_BLOCK; //max_blocks_length / WORD_SIZE; - let mut sliced_buffer = Vec::new(); // populate a vector of 64-bit limbs from our byte array + let num_limbs = max_blocks_length / WORD_SIZE; + let mut sliced_buffer = Vec::new(); for i in 0..num_limbs { let limb_start = WORD_SIZE * i; - + let mut sliced = 0; - if (limb_start + WORD_SIZE > max_blocks_length) { - let slice_size = max_blocks_length - limb_start; - let byte_shift = (WORD_SIZE - slice_size) * 8; - let mut v = 1; - for k in 0..slice_size { - sliced += v * (block_bytes[limb_start+k] as Field); - v *= 256; - } - let w = 1 << (byte_shift as u8); - sliced *= w as Field; - } else { - let mut v = 1; - for k in 0..WORD_SIZE { - sliced += v * (block_bytes[limb_start+k] as Field); - v *= 256; - } + let mut v = 1; + for k in 0..WORD_SIZE { + sliced += v * (block_bytes[limb_start+k] as Field); + v *= 256; } sliced_buffer.push(sliced as u64); From d70449f9997711cad2f552010f529925e86f9b1c Mon Sep 17 00:00:00 2001 From: TomAFrench Date: Mon, 23 Sep 2024 20:14:35 +0000 Subject: [PATCH 2/2] . --- noir_stdlib/src/hash/keccak.nr | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/noir_stdlib/src/hash/keccak.nr b/noir_stdlib/src/hash/keccak.nr index 1675e111026..1acb2a59045 100644 --- a/noir_stdlib/src/hash/keccak.nr +++ b/noir_stdlib/src/hash/keccak.nr @@ -3,7 +3,7 @@ use crate::runtime::is_unconstrained; global BLOCK_SIZE_IN_BYTES: u32 = 136; //(1600 - BITS * 2) / WORD_SIZE; global WORD_SIZE: u32 = 8; // Limbs are made up of u64s so 8 bytes each. -global LIMBS_PER_BLOCK: u32 = BLOCK_SIZE_IN_BYTES / WORD_SIZE; +global LIMBS_PER_BLOCK: u32 = BLOCK_SIZE_IN_BYTES / WORD_SIZE; global NUM_KECCAK_LANES: u32 = 25; #[foreign(keccakf1600)] @@ -40,7 +40,7 @@ pub(crate) fn keccak256(input: [u8; N], message_size: u32) -> [u8; 3 let mut sliced_buffer = Vec::new(); for i in 0..num_limbs { let limb_start = WORD_SIZE * i; - + let mut sliced = 0; let mut v = 1; for k in 0..WORD_SIZE {