Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[DNM] Added hints felt unpacking for blake. #1994 #2013

Closed
wants to merge 1 commit into from
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,8 @@

* feat: replace `thiserror-no-std` with `thiserror 2` [#1919](https://github.com/lambdaclass/cairo-vm/pull/1919)

* feat: Support hints for new blake felt serialization library code [#1994](https://github.com/lambdaclass/cairo-vm/pull/1994)

#### [2.0.1] - 2025-03-17

* feat: Limited padding of builtin segments to >=16 [#1981](https://github.com/lambdaclass/cairo-vm/pull/1981)
Expand Down
177 changes: 177 additions & 0 deletions vm/src/hint_processor/builtin_hint_processor/blake2s_utils.rs
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
use crate::hint_processor::hint_processor_utils::felt_to_usize;
use crate::stdlib::{borrow::Cow, collections::HashMap, prelude::*};

use crate::types::errors::math_errors::MathError;
Expand All @@ -17,6 +18,8 @@ use crate::{
vm::{errors::hint_errors::HintError, vm_core::VirtualMachine},
};

use num_bigint::BigUint;
use num_integer::Integer;
use num_traits::ToPrimitive;

use super::hint_utils::get_integer_from_var_name;
Expand Down Expand Up @@ -242,6 +245,83 @@ pub fn blake2s_add_uint256_bigend(
Ok(())
}

/* Implements Hint:
memory[ap] = (ids.end != ids.packed_values) and (memory[ids.packed_values] < 2**63)
*/
pub fn is_over_63_bits_or_end(
vm: &mut VirtualMachine,
ids_data: &HashMap<String, HintReference>,
ap_tracking: &ApTracking,
) -> Result<(), HintError> {
let end = get_ptr_from_var_name("end", vm, ids_data, ap_tracking)?;
let packed_values = get_ptr_from_var_name("packed_values", vm, ids_data, ap_tracking)?;
let ap = vm.get_ap();

if end == packed_values {
vm.insert_value(ap, 0)?
} else {
let val = vm.get_integer(packed_values)?;
vm.insert_value(
ap,
(val.to_biguint() < (BigUint::from(1_u32) << 63)) as usize,
)?
}
Ok(())
}

/* Implements Hint:
offset = 0
for i in range(ids.packed_values_len):
val = (memory[ids.packed_values + i] % PRIME)
val_len = 2 if val < 2**63 else 8
if val_len == 8:
val += 2**255
for i in range(val_len - 1, -1, -1):
val, memory[ids.unpacked_u32s + offset + i] = divmod(val, 2**32)
assert val == 0
offset += val_len
*/
pub fn blake2s_unpack_felts(
vm: &mut VirtualMachine,
ids_data: &HashMap<String, HintReference>,
ap_tracking: &ApTracking,
) -> Result<(), HintError> {
let packed_values_len =
get_integer_from_var_name("packed_values_len", vm, ids_data, ap_tracking)?;
let packed_values = get_ptr_from_var_name("packed_values", vm, ids_data, ap_tracking)?;
let unpacked_u32s = get_ptr_from_var_name("unpacked_u32s", vm, ids_data, ap_tracking)?;

let vals = vm.get_integer_range(packed_values, felt_to_usize(&packed_values_len)?)?;
let pow2 = |exp: u32| BigUint::from(1_u32) << exp;

// Split value into either 2 or 8 32-bit limbs.
let out: Vec<MaybeRelocatable> = vals
.into_iter()
.map(|val| val.to_biguint())
.flat_map(|val| {
if val < pow2(63) {
let (high, low) = val.div_rem(&(pow2(32)));
vec![high, low]
} else {
let mut limbs = vec![BigUint::from(0_u32); 8];
let mut val: BigUint = val + (pow2(255));
for i in (0..8).rev() {
let (q, r) = val.div_rem(&(pow2(32)));
limbs[i] = r;
val = q;
}
limbs
}
})
.map(Felt252::from)
.map(MaybeRelocatable::from)
.collect();

vm.load_data(unpacked_u32s, &out)
.map_err(HintError::Memory)?;
Ok(())
}

/* Implements Hint:
%{
from starkware.cairo.common.cairo_blake2s.blake2s_utils import IV, blake2s_compress
Expand Down Expand Up @@ -604,6 +684,103 @@ mod tests {
.is_none());
}

#[test]
#[cfg_attr(target_arch = "wasm32", wasm_bindgen_test)]
fn is_over_63_bits_or_end_ends() {
let hint_code = hint_code::IS_OVER_63_BITS_OR_END;
//Create vm
let mut vm = vm!();
//Insert ids into memory
vm.segments = segments![((1, 0), (1, 2)), ((1, 1), (1, 2)), ((1, 2), 123)];
vm.set_fp(3);
vm.set_ap(3);
let ids_data = ids_data!["end", "packed_values", "value"];
//Execute the hint
assert_matches!(run_hint!(vm, ids_data, hint_code), Ok(()));
//Check data ptr
check_memory![vm.segments.memory, ((1, 3), 0)];
}

#[test]
#[cfg_attr(target_arch = "wasm32", wasm_bindgen_test)]
fn is_over_63_bits_or_end_small() {
let hint_code = hint_code::IS_OVER_63_BITS_OR_END;
//Create vm
let mut vm = vm!();
//Insert ids into memory
vm.segments = segments![((1, 0), (1, 3)), ((1, 1), (1, 2)), ((1, 2), 123)];
vm.set_fp(3);
vm.set_ap(3);
let ids_data = ids_data!["end", "packed_values", "value"];
//Execute the hint
assert_matches!(run_hint!(vm, ids_data, hint_code), Ok(()));
//Check data ptr
check_memory![vm.segments.memory, ((1, 3), 1)];
}

#[test]
#[cfg_attr(target_arch = "wasm32", wasm_bindgen_test)]
fn is_over_63_bits_or_end_big() {
let hint_code = hint_code::IS_OVER_63_BITS_OR_END;
//Create vm
let mut vm = vm!();
//Insert ids into memory
vm.segments = segments![
((1, 0), (1, 3)),
((1, 1), (1, 2)),
((1, 2), 0x10000000000000000)
];
vm.set_fp(3);
vm.set_ap(3);
let ids_data = ids_data!["end", "packed_values", "value"];
//Execute the hint
assert_matches!(run_hint!(vm, ids_data, hint_code), Ok(()));
//Check data ptr
check_memory![vm.segments.memory, ((1, 3), 0)];
}

#[test]
#[cfg_attr(target_arch = "wasm32", wasm_bindgen_test)]
fn blake2s_unpack_felts() {
let hint_code = hint_code::BLAKE2S_UNPACK_FELTS;
//Create vm
let mut vm = vm!();
//Insert ids into memory
vm.segments = segments![
((1, 0), 2),
((1, 1), (1, 3)),
((1, 2), (2, 0)),
((1, 3), 0x123456781234),
((1, 4), 0x1234abcd5678efab1234abcd)
];
vm.set_fp(5);
vm.set_ap(5);
let ids_data = ids_data![
"packed_values_len",
"packed_values",
"unpacked_u32s",
"small_value",
"big_value"
];
vm.segments.add();
//Execute the hint
assert_matches!(run_hint!(vm, ids_data, hint_code), Ok(()));
//Check data ptr
check_memory![
vm.segments.memory,
((2, 0), 0x1234),
((2, 1), 0x56781234),
((2, 2), 0x80000000),
((2, 3), 0),
((2, 4), 0),
((2, 5), 0),
((2, 6), 0),
((2, 7), 0x1234abcd),
((2, 8), 0x5678efab),
((2, 9), 0x1234abcd)
];
}

#[test]
#[cfg_attr(target_arch = "wasm32", wasm_bindgen_test)]
fn blake2s_add_uint256_bigend_valid_non_zero() {
Expand Down
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
#[cfg(feature = "cairo-0-secp-hints")]
use super::secp::cairo0_hints;
use super::{
blake2s_utils::finalize_blake2s_v3,
blake2s_utils::{blake2s_unpack_felts, finalize_blake2s_v3, is_over_63_bits_or_end},
ec_recover::{
ec_recover_divmod_n_packed, ec_recover_product_div_m, ec_recover_product_mod,
ec_recover_sub_a_b,
Expand Down Expand Up @@ -360,6 +360,12 @@ impl HintProcessorLogic for BuiltinHintProcessor {
hint_code::BLAKE2S_ADD_UINT256_BIGEND => {
blake2s_add_uint256_bigend(vm, &hint_data.ids_data, &hint_data.ap_tracking)
}
hint_code::IS_OVER_63_BITS_OR_END => {
is_over_63_bits_or_end(vm, &hint_data.ids_data, &hint_data.ap_tracking)
}
hint_code::BLAKE2S_UNPACK_FELTS => {
blake2s_unpack_felts(vm, &hint_data.ids_data, &hint_data.ap_tracking)
}
hint_code::UNSAFE_KECCAK => {
unsafe_keccak(vm, exec_scopes, &hint_data.ids_data, &hint_data.ap_tracking)
}
Expand Down
11 changes: 11 additions & 0 deletions vm/src/hint_processor/builtin_hint_processor/hint_code.rs
Original file line number Diff line number Diff line change
Expand Up @@ -426,6 +426,17 @@ segments.write_arg(ids.data + 4, [(ids.high >> (B * i)) & MASK for i in range(4)
MASK = 2 ** 32 - 1
segments.write_arg(ids.data, [(ids.high >> (B * (3 - i))) & MASK for i in range(4)])
segments.write_arg(ids.data + 4, [(ids.low >> (B * (3 - i))) & MASK for i in range(4)])"#}),
(IS_OVER_63_BITS_OR_END, indoc! {r#"memory[ap] = to_felt_or_relocatable((ids.end != ids.packed_values) and (memory[ids.packed_values] < 2**63))"#}),
(BLAKE2S_UNPACK_FELTS, indoc! {r#"offset = 0
for i in range(ids.packed_values_len):
val = (memory[ids.packed_values + i] % PRIME)
val_len = 2 if val < 2**63 else 8
if val_len == 8:
val += 2**255
for i in range(val_len - 1, -1, -1):
val, memory[ids.unpacked_u32s + offset + i] = divmod(val, 2**32)
assert val == 0
offset += val_len"#}),
(EXAMPLE_BLAKE2S_COMPRESS, indoc! {r#"from starkware.cairo.common.cairo_blake2s.blake2s_utils import IV, blake2s_compress

_blake2s_input_chunk_size_felts = int(ids.BLAKE2S_INPUT_CHUNK_SIZE_FELTS)
Expand Down
Loading