@aztec/protocol-contracts 5.0.0-nightly.20260430 → 5.0.0-nightly.20260501
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
|
@@ -1162,7 +1162,7 @@
|
|
|
1162
1162
|
"path": "/home/aztec-dev/aztec-packages/noir-projects/aztec-nr/aztec/src/oracle/call_private_function.nr",
|
|
1163
1163
|
"source": "use crate::protocol::{abis::function_selector::FunctionSelector, address::AztecAddress, utils::reader::Reader};\n\n#[oracle(aztec_prv_callPrivateFunction)]\nunconstrained fn call_private_function_oracle(\n _contract_address: AztecAddress,\n _function_selector: FunctionSelector,\n _args_hash: Field,\n _start_side_effect_counter: u32,\n _is_static_call: bool,\n) -> [Field; 2] {}\n\npub unconstrained fn call_private_function_internal(\n contract_address: AztecAddress,\n function_selector: FunctionSelector,\n args_hash: Field,\n start_side_effect_counter: u32,\n is_static_call: bool,\n) -> (u32, Field) {\n let fields = call_private_function_oracle(\n contract_address,\n function_selector,\n args_hash,\n start_side_effect_counter,\n is_static_call,\n );\n\n let mut reader = Reader::new(fields);\n let end_side_effect_counter = reader.read_u32();\n let returns_hash = reader.read();\n\n (end_side_effect_counter, returns_hash)\n}\n"
|
|
1164
1164
|
},
|
|
1165
|
-
"
|
|
1165
|
+
"178": {
|
|
1166
1166
|
"function_locations": [
|
|
1167
1167
|
{
|
|
1168
1168
|
"name": "store",
|
|
@@ -1256,7 +1256,7 @@
|
|
|
1256
1256
|
"path": "/home/aztec-dev/aztec-packages/noir-projects/aztec-nr/aztec/src/oracle/capsules.nr",
|
|
1257
1257
|
"source": "use crate::protocol::{address::AztecAddress, traits::{Deserialize, Serialize}};\n\n/// Stores arbitrary information in a per-contract non-volatile database, which can later be retrieved with `load`. If\n/// data was already stored at this slot, it is overwritten.\n// TODO(F-498): review naming consistency\npub unconstrained fn store<T>(contract_address: AztecAddress, slot: Field, value: T, scope: AztecAddress)\nwhere\n T: Serialize,\n{\n let serialized = value.serialize();\n set_capsule_oracle(contract_address, slot, serialized, scope);\n}\n\n/// Returns data previously stored via `storeCapsule` in the per-contract non-volatile database. Returns\n/// Option::none() if nothing was stored at the given slot.\n// TODO(F-498): review naming consistency\npub unconstrained fn load<T>(contract_address: AztecAddress, slot: Field, scope: AztecAddress) -> Option<T>\nwhere\n T: Deserialize,\n{\n let serialized_option = get_capsule_oracle(contract_address, slot, <T as Deserialize>::N, scope);\n serialized_option.map(|arr| Deserialize::deserialize(arr))\n}\n\n/// Deletes data in the per-contract non-volatile database. Does nothing if no data was present.\npub unconstrained fn delete(contract_address: AztecAddress, slot: Field, scope: AztecAddress) {\n delete_oracle(contract_address, slot, scope);\n}\n\n/// Copies a number of contiguous entries in the per-contract non-volatile database. This allows for efficient data\n/// structures by avoiding repeated calls to `loadCapsule` and `storeCapsule`. Supports overlapping source and\n/// destination regions (which will result in the overlapped source values being overwritten). All copied slots must\n/// exist in the database (i.e. have been stored and not deleted)\npub unconstrained fn copy(\n contract_address: AztecAddress,\n src_slot: Field,\n dst_slot: Field,\n num_entries: u32,\n scope: AztecAddress,\n) {\n copy_oracle(contract_address, src_slot, dst_slot, num_entries, scope);\n}\n\n#[oracle(aztec_utl_setCapsule)]\nunconstrained fn set_capsule_oracle<let N: u32>(\n contract_address: AztecAddress,\n slot: Field,\n values: [Field; N],\n scope: AztecAddress,\n) {}\n\n/// We need to pass in `array_len` (the value of N) as a parameter to tell the oracle how many fields the response must\n/// have.\n///\n/// Note that the oracle returns an Option<[Field; N]> because we cannot return an Option<T> directly. That would\n/// require for the oracle resolver to know the shape of T (e.g. if T were a struct of 3 u32 values then the expected\n/// response shape would be 3 single items, whereas it were a struct containing `u32, [Field;10], u32` then the\n/// expected shape would be single, array, single.). Instead, we return the serialization and deserialize in Noir.\n#[oracle(aztec_utl_getCapsule)]\nunconstrained fn get_capsule_oracle<let N: u32>(\n contract_address: AztecAddress,\n slot: Field,\n array_len: u32,\n scope: AztecAddress,\n) -> Option<[Field; N]> {}\n\n#[oracle(aztec_utl_deleteCapsule)]\nunconstrained fn delete_oracle(contract_address: AztecAddress, slot: Field, scope: AztecAddress) {}\n\n#[oracle(aztec_utl_copyCapsule)]\nunconstrained fn copy_oracle(\n contract_address: AztecAddress,\n src_slot: Field,\n dst_slot: Field,\n num_entries: u32,\n scope: AztecAddress,\n) {}\n\nmod test {\n // These tests are sort of redundant since we already test the oracle implementation directly in TypeScript, but\n // they are cheap regardless and help ensure both that the TXE implementation works accordingly and that the Noir\n // oracles are hooked up correctly.\n\n use crate::{\n oracle::capsules::{copy, delete, load, store},\n test::{helpers::test_environment::TestEnvironment, mocks::MockStruct},\n };\n use crate::protocol::{address::AztecAddress, traits::{FromField, ToField}};\n\n global SLOT: Field = 1;\n\n unconstrained fn setup() -> (TestEnvironment, AztecAddress) {\n let mut env = TestEnvironment::new();\n let scope = env.create_light_account();\n (env, scope)\n }\n\n #[test]\n unconstrained fn stores_and_loads() {\n let (env, scope) = setup();\n env.private_context(|context| {\n let contract_address = context.this_address();\n\n let value = MockStruct::new(5, 6);\n store(contract_address, SLOT, value, scope);\n\n assert_eq(load(contract_address, SLOT, scope).unwrap(), value);\n });\n }\n\n #[test]\n unconstrained fn store_overwrites() {\n let (env, scope) = setup();\n env.private_context(|context| {\n let contract_address = context.this_address();\n\n let value = MockStruct::new(5, 6);\n store(contract_address, SLOT, value, scope);\n\n let new_value = MockStruct::new(7, 8);\n store(contract_address, SLOT, new_value, scope);\n\n assert_eq(load(contract_address, SLOT, scope).unwrap(), new_value);\n });\n }\n\n #[test]\n unconstrained fn loads_empty_slot() {\n let (env, scope) = setup();\n env.private_context(|context| {\n let contract_address = context.this_address();\n\n let loaded_value: Option<MockStruct> = load(contract_address, SLOT, scope);\n assert_eq(loaded_value, Option::none());\n });\n }\n\n #[test]\n unconstrained fn deletes_stored_value() {\n let (env, scope) = setup();\n env.private_context(|context| {\n let contract_address = context.this_address();\n\n let value = MockStruct::new(5, 6);\n store(contract_address, SLOT, value, scope);\n delete(contract_address, SLOT, scope);\n\n let loaded_value: Option<MockStruct> = load(contract_address, SLOT, scope);\n assert_eq(loaded_value, Option::none());\n });\n }\n\n #[test]\n unconstrained fn deletes_empty_slot() {\n let (env, scope) = setup();\n env.private_context(|context| {\n let contract_address = context.this_address();\n\n delete(contract_address, SLOT, scope);\n let loaded_value: Option<MockStruct> = load(contract_address, SLOT, scope);\n assert_eq(loaded_value, Option::none());\n });\n }\n\n #[test]\n unconstrained fn copies_non_overlapping_values() {\n let (env, scope) = setup();\n env.private_context(|context| {\n let contract_address = context.this_address();\n\n let src = 5;\n\n let values = [MockStruct::new(5, 6), MockStruct::new(7, 8), MockStruct::new(9, 10)];\n store(contract_address, src, values[0], scope);\n store(contract_address, src + 1, values[1], scope);\n store(contract_address, src + 2, values[2], scope);\n\n let dst = 10;\n copy(contract_address, src, dst, 3, scope);\n\n assert_eq(load(contract_address, dst, scope).unwrap(), values[0]);\n assert_eq(load(contract_address, dst + 1, scope).unwrap(), values[1]);\n assert_eq(load(contract_address, dst + 2, scope).unwrap(), values[2]);\n });\n }\n\n #[test]\n unconstrained fn copies_overlapping_values_with_src_ahead() {\n let (env, scope) = setup();\n env.private_context(|context| {\n let contract_address = context.this_address();\n\n let src = 1;\n\n let values = [MockStruct::new(5, 6), MockStruct::new(7, 8), MockStruct::new(9, 10)];\n store(contract_address, src, values[0], scope);\n store(contract_address, src + 1, values[1], scope);\n store(contract_address, src + 2, values[2], scope);\n\n let dst = 2;\n copy(contract_address, src, dst, 3, scope);\n\n assert_eq(load(contract_address, dst, scope).unwrap(), values[0]);\n assert_eq(load(contract_address, dst + 1, scope).unwrap(), values[1]);\n assert_eq(load(contract_address, dst + 2, scope).unwrap(), values[2]);\n\n // src[1] and src[2] should have been overwritten since they are also dst[0] and dst[1]\n assert_eq(load(contract_address, src, scope).unwrap(), values[0]); // src[0] (unchanged)\n assert_eq(load(contract_address, src + 1, scope).unwrap(), values[0]); // dst[0]\n assert_eq(load(contract_address, src + 2, scope).unwrap(), values[1]); // dst[1]\n });\n }\n\n #[test]\n unconstrained fn copies_overlapping_values_with_dst_ahead() {\n let (env, scope) = setup();\n env.private_context(|context| {\n let contract_address = context.this_address();\n\n let src = 2;\n\n let values = [MockStruct::new(5, 6), MockStruct::new(7, 8), MockStruct::new(9, 10)];\n store(contract_address, src, values[0], scope);\n store(contract_address, src + 1, values[1], scope);\n store(contract_address, src + 2, values[2], scope);\n\n let dst = 1;\n copy(contract_address, src, dst, 3, scope);\n\n assert_eq(load(contract_address, dst, scope).unwrap(), values[0]);\n assert_eq(load(contract_address, dst + 1, scope).unwrap(), values[1]);\n assert_eq(load(contract_address, dst + 2, scope).unwrap(), values[2]);\n\n // src[0] and src[1] should have been overwritten since they are also dst[1] and dst[2]\n assert_eq(load(contract_address, src, scope).unwrap(), values[1]); // dst[1]\n assert_eq(load(contract_address, src + 1, scope).unwrap(), values[2]); // dst[2]\n assert_eq(load(contract_address, src + 2, scope).unwrap(), values[2]); // src[2] (unchanged)\n });\n }\n\n #[test(should_fail_with = \"copy empty slot\")]\n unconstrained fn cannot_copy_empty_values() {\n let (env, scope) = setup();\n env.private_context(|context| {\n let contract_address = context.this_address();\n\n copy(contract_address, SLOT, SLOT, 1, scope);\n });\n }\n\n #[test(should_fail_with = \"not allowed to access\")]\n unconstrained fn cannot_store_other_contract() {\n let (env, scope) = setup();\n env.private_context(|context| {\n let contract_address = context.this_address();\n let other_contract_address = AztecAddress::from_field(contract_address.to_field() + 1);\n\n let value = MockStruct::new(5, 6);\n store(other_contract_address, SLOT, value, scope);\n });\n }\n\n #[test(should_fail_with = \"not allowed to access\")]\n unconstrained fn cannot_load_other_contract() {\n let (env, scope) = setup();\n env.private_context(|context| {\n let contract_address = context.this_address();\n let other_contract_address = AztecAddress::from_field(contract_address.to_field() + 1);\n\n let _: Option<MockStruct> = load(other_contract_address, SLOT, scope);\n });\n }\n\n #[test(should_fail_with = \"not allowed to access\")]\n unconstrained fn cannot_delete_other_contract() {\n let (env, scope) = setup();\n env.private_context(|context| {\n let contract_address = context.this_address();\n let other_contract_address = AztecAddress::from_field(contract_address.to_field() + 1);\n\n delete(other_contract_address, SLOT, scope);\n });\n }\n\n #[test(should_fail_with = \"not allowed to access\")]\n unconstrained fn cannot_copy_other_contract() {\n let (env, scope) = setup();\n env.private_context(|context| {\n let contract_address = context.this_address();\n let other_contract_address = AztecAddress::from_field(contract_address.to_field() + 1);\n\n copy(other_contract_address, SLOT, SLOT, 0, scope);\n });\n }\n}\n"
|
|
1258
1258
|
},
|
|
1259
|
-
"
|
|
1259
|
+
"179": {
|
|
1260
1260
|
"function_locations": [
|
|
1261
1261
|
{
|
|
1262
1262
|
"name": "set_contract_sync_cache_invalid_oracle",
|
|
@@ -1424,7 +1424,7 @@
|
|
|
1424
1424
|
"path": "std/hash/mod.nr",
|
|
1425
1425
|
"source": "// Exposed only for usage in `std::meta`\npub(crate) mod poseidon2;\n\nuse crate::default::Default;\nuse crate::embedded_curve_ops::{\n EmbeddedCurvePoint, EmbeddedCurveScalar, multi_scalar_mul, multi_scalar_mul_array_return,\n};\nuse crate::meta::derive_via;\nuse crate::static_assert;\n\n/// The size of the state accepted by the backend in `poseidon2_permutation`.\nglobal POSEIDON2_CONFIG_STATE_SIZE: u32 = poseidon2_config_state_size();\n\n#[foreign(sha256_compression)]\n// docs:start:sha256_compression\npub fn sha256_compression(input: [u32; 16], state: [u32; 8]) -> [u32; 8] {}\n// docs:end:sha256_compression\n\n#[foreign(keccakf1600)]\n// docs:start:keccakf1600\npub fn keccakf1600(input: [u64; 25]) -> [u64; 25] {}\n// docs:end:keccakf1600\n\npub mod keccak {\n #[deprecated(\"This function has been moved to std::hash::keccakf1600\")]\n pub fn keccakf1600(input: [u64; 25]) -> [u64; 25] {\n super::keccakf1600(input)\n }\n}\n\n#[foreign(blake2s)]\n// docs:start:blake2s\npub fn blake2s<let N: u32>(input: [u8; N]) -> [u8; 32]\n// docs:end:blake2s\n{}\n\n// docs:start:blake3\npub fn blake3<let N: u32>(input: [u8; N]) -> [u8; 32]\n// docs:end:blake3\n{\n if crate::runtime::is_unconstrained() {\n // Temporary measure while Barretenberg is main proving system.\n // Please open an issue if you're working on another proving system and running into problems due to this.\n crate::static_assert(\n N <= 1024,\n \"Barretenberg cannot prove blake3 hashes with inputs larger than 1024 bytes\",\n );\n }\n __blake3(input)\n}\n\n#[foreign(blake3)]\nfn __blake3<let N: u32>(input: [u8; N]) -> [u8; 32] {}\n\n// docs:start:pedersen_commitment\npub fn pedersen_commitment<let N: u32>(input: [Field; N]) -> EmbeddedCurvePoint {\n // docs:end:pedersen_commitment\n pedersen_commitment_with_separator(input, 0)\n}\n\n#[inline_always]\npub fn pedersen_commitment_with_separator<let N: u32>(\n input: [Field; N],\n separator: u32,\n) -> EmbeddedCurvePoint {\n let mut points = [EmbeddedCurveScalar { lo: 0, hi: 0 }; N];\n for i in 0..N {\n points[i] = EmbeddedCurveScalar::from_field(input[i]);\n }\n let generators = derive_generators(\"DEFAULT_DOMAIN_SEPARATOR\".as_bytes(), separator);\n multi_scalar_mul(generators, points)\n}\n\n// docs:start:pedersen_hash\npub fn pedersen_hash<let N: u32>(input: [Field; N]) -> Field\n// docs:end:pedersen_hash\n{\n pedersen_hash_with_separator(input, 0)\n}\n\n#[no_predicates]\npub fn pedersen_hash_with_separator<let N: u32>(input: [Field; N], separator: u32) -> Field {\n let mut scalars: [EmbeddedCurveScalar; N + 1] = [EmbeddedCurveScalar { lo: 0, hi: 0 }; N + 1];\n let mut generators: [EmbeddedCurvePoint; N + 1] =\n [EmbeddedCurvePoint::point_at_infinity(); N + 1];\n crate::assert_constant(separator);\n let domain_generators: [EmbeddedCurvePoint; N] =\n derive_generators(\"DEFAULT_DOMAIN_SEPARATOR\".as_bytes(), separator);\n\n for i in 0..N {\n scalars[i] = EmbeddedCurveScalar::from_field(input[i]);\n generators[i] = domain_generators[i];\n }\n scalars[N] = EmbeddedCurveScalar { lo: N as Field, hi: 0 as Field };\n\n let length_generator: [EmbeddedCurvePoint; 1] =\n derive_generators(\"pedersen_hash_length\".as_bytes(), 0);\n generators[N] = length_generator[0];\n multi_scalar_mul_array_return(generators, scalars, true)[0].x\n}\n\n#[field(bn254)]\n#[inline_always]\npub fn derive_generators<let N: u32, let M: u32>(\n domain_separator_bytes: [u8; M],\n starting_index: u32,\n) -> [EmbeddedCurvePoint; N] {\n crate::assert_constant(domain_separator_bytes);\n crate::assert_constant(starting_index);\n __derive_generators(domain_separator_bytes, starting_index)\n}\n\n#[builtin(derive_pedersen_generators)]\n#[field(bn254)]\nfn __derive_generators<let N: u32, let M: u32>(\n domain_separator_bytes: [u8; M],\n starting_index: u32,\n) -> [EmbeddedCurvePoint; N] {}\n\npub fn poseidon2_permutation<let N: u32>(input: [Field; N]) -> [Field; N] {\n static_assert(\n N == POSEIDON2_CONFIG_STATE_SIZE,\n f\"the input length must equal the state size in the Poseidon2 config; expected {POSEIDON2_CONFIG_STATE_SIZE}, got {N}\",\n );\n poseidon2_permutation_internal(input)\n}\n\n#[foreign(poseidon2_permutation)]\nfn poseidon2_permutation_internal<let N: u32>(input: [Field; N]) -> [Field; N] {}\n\n#[foreign(poseidon2_config_state_size)]\ncomptime fn poseidon2_config_state_size() -> u32 {}\n\n// Generic hashing support.\n// Partially ported and impacted by rust.\n\n// Hash trait shall be implemented per type.\n#[derive_via(derive_hash)]\npub trait Hash {\n fn hash<H>(self, state: &mut H)\n where\n H: Hasher;\n}\n\n// docs:start:derive_hash\ncomptime fn derive_hash(s: TypeDefinition) -> Quoted {\n let name = quote { $crate::hash::Hash };\n let signature = quote { fn hash<H>(_self: Self, _state: &mut H) where H: $crate::hash::Hasher };\n let for_each_field = |name| quote { _self.$name.hash(_state); };\n crate::meta::make_trait_impl(\n s,\n name,\n signature,\n for_each_field,\n quote {},\n |fields| fields,\n )\n}\n// docs:end:derive_hash\n\n// Hasher trait shall be implemented by algorithms to provide hash-agnostic means.\n// TODO: consider making the types generic here ([u8], [Field], etc.)\npub trait Hasher {\n fn finish(self) -> Field;\n\n /// Returns the hash value without consuming the hasher.\n /// Override this for more efficient implementations that avoid copying.\n /// TODO: deprecate finish() and replace it\n fn finish_ref(&self) -> Field {\n (*self).finish()\n }\n\n fn write(&mut self, input: Field);\n}\n\n// BuildHasher is a factory trait, responsible for production of specific Hasher.\npub trait BuildHasher {\n type H: Hasher;\n\n fn build_hasher(self) -> H;\n}\n\npub struct BuildHasherDefault<H>;\n\nimpl<H> BuildHasher for BuildHasherDefault<H>\nwhere\n H: Hasher + Default,\n{\n type H = H;\n\n fn build_hasher(_self: Self) -> H {\n H::default()\n }\n}\n\nimpl<H> Default for BuildHasherDefault<H>\nwhere\n H: Hasher + Default,\n{\n fn default() -> Self {\n BuildHasherDefault {}\n }\n}\n\nimpl Hash for Field {\n fn hash<H>(self, state: &mut H)\n where\n H: Hasher,\n {\n H::write(state, self);\n }\n}\n\nimpl Hash for u8 {\n fn hash<H>(self, state: &mut H)\n where\n H: Hasher,\n {\n H::write(state, self as Field);\n }\n}\n\nimpl Hash for u16 {\n fn hash<H>(self, state: &mut H)\n where\n H: Hasher,\n {\n H::write(state, self as Field);\n }\n}\n\nimpl Hash for u32 {\n fn hash<H>(self, state: &mut H)\n where\n H: Hasher,\n {\n H::write(state, self as Field);\n }\n}\n\nimpl Hash for u64 {\n fn hash<H>(self, state: &mut H)\n where\n H: Hasher,\n {\n H::write(state, self as Field);\n }\n}\n\nimpl Hash for u128 {\n fn hash<H>(self, state: &mut H)\n where\n H: Hasher,\n {\n H::write(state, self as Field);\n }\n}\n\nimpl Hash for i8 {\n fn hash<H>(self, state: &mut H)\n where\n H: Hasher,\n {\n H::write(state, self as u8 as Field);\n }\n}\n\nimpl Hash for i16 {\n fn hash<H>(self, state: &mut H)\n where\n H: Hasher,\n {\n H::write(state, self as u16 as Field);\n }\n}\n\nimpl Hash for i32 {\n fn hash<H>(self, state: &mut H)\n where\n H: Hasher,\n {\n H::write(state, self as u32 as Field);\n }\n}\n\nimpl Hash for i64 {\n fn hash<H>(self, state: &mut H)\n where\n H: Hasher,\n {\n H::write(state, self as u64 as Field);\n }\n}\n\nimpl Hash for bool {\n fn hash<H>(self, state: &mut H)\n where\n H: Hasher,\n {\n H::write(state, self as Field);\n }\n}\n\nimpl Hash for () {\n fn hash<H>(_self: Self, _state: &mut H)\n where\n H: Hasher,\n {}\n}\n\nimpl<T, let N: u32> Hash for [T; N]\nwhere\n T: Hash,\n{\n fn hash<H>(self, state: &mut H)\n where\n H: Hasher,\n {\n for elem in self {\n elem.hash(state);\n }\n }\n}\n\nimpl<T> Hash for [T]\nwhere\n T: Hash,\n{\n fn hash<H>(self, state: &mut H)\n where\n H: Hasher,\n {\n self.len().hash(state);\n for elem in self {\n elem.hash(state);\n }\n }\n}\n\nimpl<A, B> Hash for (A, B)\nwhere\n A: Hash,\n B: Hash,\n{\n fn hash<H>(self, state: &mut H)\n where\n H: Hasher,\n {\n self.0.hash(state);\n self.1.hash(state);\n }\n}\n\nimpl<A, B, C> Hash for (A, B, C)\nwhere\n A: Hash,\n B: Hash,\n C: Hash,\n{\n fn hash<H>(self, state: &mut H)\n where\n H: Hasher,\n {\n self.0.hash(state);\n self.1.hash(state);\n self.2.hash(state);\n }\n}\n\nimpl<A, B, C, D> Hash for (A, B, C, D)\nwhere\n A: Hash,\n B: Hash,\n C: Hash,\n D: Hash,\n{\n fn hash<H>(self, state: &mut H)\n where\n H: Hasher,\n {\n self.0.hash(state);\n self.1.hash(state);\n self.2.hash(state);\n self.3.hash(state);\n }\n}\n\nimpl<A, B, C, D, E> Hash for (A, B, C, D, E)\nwhere\n A: Hash,\n B: Hash,\n C: Hash,\n D: Hash,\n E: Hash,\n{\n fn hash<H>(self, state: &mut H)\n where\n H: Hasher,\n {\n self.0.hash(state);\n self.1.hash(state);\n self.2.hash(state);\n self.3.hash(state);\n self.4.hash(state);\n }\n}\n\n// Some test vectors for Pedersen hash and Pedersen Commitment.\n// They have been generated using the same functions so the tests are for now useless\n// but they will be useful when we switch to Noir implementation.\n#[test]\nfn assert_pedersen() {\n assert_eq(\n pedersen_hash_with_separator([1], 1),\n 0x1b3f4b1a83092a13d8d1a59f7acb62aba15e7002f4440f2275edb99ebbc2305f,\n );\n assert_eq(\n pedersen_commitment_with_separator([1], 1),\n EmbeddedCurvePoint {\n x: 0x054aa86a73cb8a34525e5bbed6e43ba1198e860f5f3950268f71df4591bde402,\n y: 0x209dcfbf2cfb57f9f6046f44d71ac6faf87254afc7407c04eb621a6287cac126,\n },\n );\n\n assert_eq(\n pedersen_hash_with_separator([1, 2], 2),\n 0x26691c129448e9ace0c66d11f0a16d9014a9e8498ee78f4d69f0083168188255,\n );\n assert_eq(\n pedersen_commitment_with_separator([1, 2], 2),\n EmbeddedCurvePoint {\n x: 0x2e2b3b191e49541fe468ec6877721d445dcaffe41728df0a0eafeb15e87b0753,\n y: 0x2ff4482400ad3a6228be17a2af33e2bcdf41be04795f9782bd96efe7e24f8778,\n },\n );\n assert_eq(\n pedersen_hash_with_separator([1, 2, 3], 3),\n 0x0bc694b7a1f8d10d2d8987d07433f26bd616a2d351bc79a3c540d85b6206dbe4,\n );\n assert_eq(\n pedersen_commitment_with_separator([1, 2, 3], 3),\n EmbeddedCurvePoint {\n x: 0x1fee4e8cf8d2f527caa2684236b07c4b1bad7342c01b0f75e9a877a71827dc85,\n y: 0x2f9fedb9a090697ab69bf04c8bc15f7385b3e4b68c849c1536e5ae15ff138fd1,\n },\n );\n assert_eq(\n pedersen_hash_with_separator([1, 2, 3, 4], 4),\n 0xdae10fb32a8408521803905981a2b300d6a35e40e798743e9322b223a5eddc,\n );\n assert_eq(\n pedersen_commitment_with_separator([1, 2, 3, 4], 4),\n EmbeddedCurvePoint {\n x: 0x07ae3e202811e1fca39c2d81eabe6f79183978e6f12be0d3b8eda095b79bdbc9,\n y: 0x0afc6f892593db6fbba60f2da558517e279e0ae04f95758587760ba193145014,\n },\n );\n assert_eq(\n pedersen_hash_with_separator([1, 2, 3, 4, 5], 5),\n 0xfc375b062c4f4f0150f7100dfb8d9b72a6d28582dd9512390b0497cdad9c22,\n );\n assert_eq(\n pedersen_commitment_with_separator([1, 2, 3, 4, 5], 5),\n EmbeddedCurvePoint {\n x: 0x1754b12bd475a6984a1094b5109eeca9838f4f81ac89c5f0a41dbce53189bb29,\n y: 0x2da030e3cfcdc7ddad80eaf2599df6692cae0717d4e9f7bfbee8d073d5d278f7,\n },\n );\n assert_eq(\n pedersen_hash_with_separator([1, 2, 3, 4, 5, 6], 6),\n 0x1696ed13dc2730062a98ac9d8f9de0661bb98829c7582f699d0273b18c86a572,\n );\n assert_eq(\n pedersen_commitment_with_separator([1, 2, 3, 4, 5, 6], 6),\n EmbeddedCurvePoint {\n x: 0x190f6c0e97ad83e1e28da22a98aae156da083c5a4100e929b77e750d3106a697,\n y: 0x1f4b60f34ef91221a0b49756fa0705da93311a61af73d37a0c458877706616fb,\n },\n );\n assert_eq(\n pedersen_hash_with_separator([1, 2, 3, 4, 5, 6, 7], 7),\n 0x128c0ff144fc66b6cb60eeac8a38e23da52992fc427b92397a7dffd71c45ede3,\n );\n assert_eq(\n pedersen_commitment_with_separator([1, 2, 3, 4, 5, 6, 7], 7),\n EmbeddedCurvePoint {\n x: 0x015441e9d29491b06563fac16fc76abf7a9534c715421d0de85d20dbe2965939,\n y: 0x1d2575b0276f4e9087e6e07c2cb75aa1baafad127af4be5918ef8a2ef2fea8fc,\n },\n );\n assert_eq(\n pedersen_hash_with_separator([1, 2, 3, 4, 5, 6, 7, 8], 8),\n 0x2f960e117482044dfc99d12fece2ef6862fba9242be4846c7c9a3e854325a55c,\n );\n assert_eq(\n pedersen_commitment_with_separator([1, 2, 3, 4, 5, 6, 7, 8], 8),\n EmbeddedCurvePoint {\n x: 0x1657737676968887fceb6dd516382ea13b3a2c557f509811cd86d5d1199bc443,\n y: 0x1f39f0cb569040105fa1e2f156521e8b8e08261e635a2b210bdc94e8d6d65f77,\n },\n );\n assert_eq(\n pedersen_hash_with_separator([1, 2, 3, 4, 5, 6, 7, 8, 9], 9),\n 0x0c96db0790602dcb166cc4699e2d306c479a76926b81c2cb2aaa92d249ec7be7,\n );\n assert_eq(\n pedersen_commitment_with_separator([1, 2, 3, 4, 5, 6, 7, 8, 9], 9),\n EmbeddedCurvePoint {\n x: 0x0a3ceae42d14914a432aa60ec7fded4af7dad7dd4acdbf2908452675ec67e06d,\n y: 0xfc19761eaaf621ad4aec9a8b2e84a4eceffdba78f60f8b9391b0bd9345a2f2,\n },\n );\n assert_eq(\n pedersen_hash_with_separator([1, 2, 3, 4, 5, 6, 7, 8, 9, 10], 10),\n 0x2cd37505871bc460a62ea1e63c7fe51149df5d0801302cf1cbc48beb8dff7e94,\n );\n assert_eq(\n pedersen_commitment_with_separator([1, 2, 3, 4, 5, 6, 7, 8, 9, 10], 10),\n EmbeddedCurvePoint {\n x: 0x2fb3f8b3d41ddde007c8c3c62550f9a9380ee546fcc639ffbb3fd30c8d8de30c,\n y: 0x300783be23c446b11a4c0fabf6c91af148937cea15fcf5fb054abf7f752ee245,\n },\n );\n}\n"
|
|
1426
1426
|
},
|
|
1427
|
-
"
|
|
1427
|
+
"181": {
|
|
1428
1428
|
"function_locations": [
|
|
1429
1429
|
{
|
|
1430
1430
|
"name": "get_utility_context_oracle",
|
|
@@ -1438,7 +1438,7 @@
|
|
|
1438
1438
|
"path": "/home/aztec-dev/aztec-packages/noir-projects/aztec-nr/aztec/src/oracle/execution.nr",
|
|
1439
1439
|
"source": "use crate::context::UtilityContext;\n\n#[oracle(aztec_utl_getUtilityContext)]\nunconstrained fn get_utility_context_oracle() -> UtilityContext {}\n\n/// Returns a utility context built from the global variables of anchor block and the contract address of the function\n/// being executed.\npub unconstrained fn get_utility_context() -> UtilityContext {\n get_utility_context_oracle()\n}\n"
|
|
1440
1440
|
},
|
|
1441
|
-
"
|
|
1441
|
+
"182": {
|
|
1442
1442
|
"function_locations": [
|
|
1443
1443
|
{
|
|
1444
1444
|
"name": "store",
|
|
@@ -1464,7 +1464,7 @@
|
|
|
1464
1464
|
"path": "/home/aztec-dev/aztec-packages/noir-projects/aztec-nr/aztec/src/oracle/execution_cache.nr",
|
|
1465
1465
|
"source": "/// Stores values represented as slice in execution cache to be later obtained by its hash.\n// TODO(F-498): review naming consistency\npub fn store<let N: u32>(values: [Field; N], hash: Field) {\n // Safety: This oracle call returns nothing: we only call it for its side effects. It is therefore always safe to\n // call. When loading the values, however, the caller must check that the values are indeed the preimage.\n unsafe { set_hash_preimage_oracle_wrapper(values, hash) };\n}\n\nunconstrained fn set_hash_preimage_oracle_wrapper<let N: u32>(values: [Field; N], hash: Field) {\n set_hash_preimage_oracle(values, hash);\n}\n\n// TODO(F-498): review naming consistency\npub unconstrained fn load<let N: u32>(hash: Field) -> [Field; N] {\n get_hash_preimage_oracle(hash)\n}\n\n#[oracle(aztec_prv_setHashPreimage)]\nunconstrained fn set_hash_preimage_oracle<let N: u32>(_values: [Field; N], _hash: Field) {}\n\n#[oracle(aztec_prv_getHashPreimage)]\nunconstrained fn get_hash_preimage_oracle<let N: u32>(_hash: Field) -> [Field; N] {}\n"
|
|
1466
1466
|
},
|
|
1467
|
-
"
|
|
1467
|
+
"191": {
|
|
1468
1468
|
"function_locations": [
|
|
1469
1469
|
{
|
|
1470
1470
|
"name": "get_pending_tagged_logs",
|
|
@@ -1502,7 +1502,7 @@
|
|
|
1502
1502
|
"path": "/home/aztec-dev/aztec-packages/noir-projects/aztec-nr/aztec/src/oracle/message_processing.nr",
|
|
1503
1503
|
"source": "use crate::ephemeral::EphemeralArray;\nuse crate::messages::processing::{\n log_retrieval_request::LogRetrievalRequest, log_retrieval_response::LogRetrievalResponse, MessageContext,\n pending_tagged_log::PendingTaggedLog,\n};\nuse crate::protocol::address::AztecAddress;\n\n/// Finds new private logs that may have been sent to all registered accounts in PXE in the current contract and\n/// returns them in an ephemeral array with an oracle-allocated base slot.\npub(crate) unconstrained fn get_pending_tagged_logs(scope: AztecAddress) -> EphemeralArray<PendingTaggedLog> {\n let result_slot = get_pending_tagged_logs_oracle(scope);\n EphemeralArray::at(result_slot)\n}\n\n#[oracle(aztec_utl_getPendingTaggedLogs_v2)]\nunconstrained fn get_pending_tagged_logs_oracle(scope: AztecAddress) -> Field {}\n\n/// Validates note/event requests stored in ephemeral arrays.\npub(crate) unconstrained fn validate_and_store_enqueued_notes_and_events(\n note_validation_requests_array_slot: Field,\n event_validation_requests_array_slot: Field,\n max_note_packed_len: Field,\n max_event_serialized_len: Field,\n scope: AztecAddress,\n) {\n validate_and_store_enqueued_notes_and_events_oracle(\n note_validation_requests_array_slot,\n event_validation_requests_array_slot,\n max_note_packed_len,\n max_event_serialized_len,\n scope,\n );\n}\n\n#[oracle(aztec_utl_validateAndStoreEnqueuedNotesAndEvents_v2)]\nunconstrained fn validate_and_store_enqueued_notes_and_events_oracle(\n note_validation_requests_array_slot: Field,\n event_validation_requests_array_slot: Field,\n max_note_packed_len: Field,\n max_event_serialized_len: Field,\n scope: AztecAddress,\n) {}\n\n/// Fetches logs by tag from an ephemeral request array and returns a response ephemeral array.\npub(crate) unconstrained fn get_logs_by_tag(\n requests: EphemeralArray<LogRetrievalRequest>,\n) -> EphemeralArray<Option<LogRetrievalResponse>> {\n let response_slot = get_logs_by_tag_v2_oracle(requests.slot);\n EphemeralArray::at(response_slot)\n}\n\n#[oracle(aztec_utl_getLogsByTag_v2)]\nunconstrained fn get_logs_by_tag_v2_oracle(request_array_slot: Field) -> Field {}\n\n/// Resolves message contexts for tx hashes in an ephemeral request array and returns a response ephemeral array.\npub(crate) unconstrained fn get_message_contexts_by_tx_hash(\n requests: EphemeralArray<Field>,\n) -> EphemeralArray<Option<MessageContext>> {\n let response_slot = get_message_contexts_by_tx_hash_v2_oracle(requests.slot);\n EphemeralArray::at(response_slot)\n}\n\n#[oracle(aztec_utl_getMessageContextsByTxHash_v2)]\nunconstrained fn get_message_contexts_by_tx_hash_v2_oracle(request_array_slot: Field) -> Field {}\n"
|
|
1504
1504
|
},
|
|
1505
|
-
"
|
|
1505
|
+
"194": {
|
|
1506
1506
|
"function_locations": [
|
|
1507
1507
|
{
|
|
1508
1508
|
"name": "notify_created_nullifier",
|
|
@@ -1532,7 +1532,7 @@
|
|
|
1532
1532
|
"path": "/home/aztec-dev/aztec-packages/noir-projects/aztec-nr/aztec/src/oracle/nullifiers.nr",
|
|
1533
1533
|
"source": "//! Nullifier creation, existence checks, etc.\n\nuse crate::protocol::address::aztec_address::AztecAddress;\n\n/// Notifies the simulator that a nullifier has been created, so that its correct status (pending or settled) can be\n/// determined when reading nullifiers in subsequent private function calls. The first non-revertible nullifier emitted\n/// is also used to compute note nonces.\npub fn notify_created_nullifier(inner_nullifier: Field) {\n // Safety: This oracle call returns nothing: we only call it for its side effects. It is therefore always safe to\n // call.\n unsafe { notify_created_nullifier_oracle(inner_nullifier) };\n}\n\n#[oracle(aztec_prv_notifyCreatedNullifier)]\nunconstrained fn notify_created_nullifier_oracle(_inner_nullifier: Field) {}\n\n/// Returns `true` if the nullifier has been emitted in the same transaction, i.e. if [`notify_created_nullifier`] has\n/// been\n/// called for this inner nullifier from the contract with the specified address.\n///\n/// Note that despite sharing pending transaction information with the app, this is not a privacy leak: anyone in the\n/// network can always determine in which transaction a inner nullifier was emitted by a given contract by simply\n/// inspecting transaction effects. What _would_ constitute a leak would be to share the list of inner pending\n/// nullifiers, as that would reveal their preimages.\npub unconstrained fn is_nullifier_pending(inner_nullifier: Field, contract_address: AztecAddress) -> bool {\n is_nullifier_pending_oracle(inner_nullifier, contract_address)\n}\n\n#[oracle(aztec_prv_isNullifierPending)]\nunconstrained fn is_nullifier_pending_oracle(_inner_nullifier: Field, _contract_address: AztecAddress) -> bool {}\n\n/// Returns `true` if the nullifier exists. Note that a `true` value can be constrained by proving existence of the\n/// nullifier, but a `false` value should not be relied upon since other transactions may emit this nullifier before\n/// the current transaction is included in a block. While this might seem of little use at first, certain design\n/// patterns benefit from this abstraction (see e.g. `PrivateMutable`).\n// TODO(F-498): review naming consistency\npub unconstrained fn check_nullifier_exists(inner_nullifier: Field) -> bool {\n does_nullifier_exist_oracle(inner_nullifier)\n}\n\n#[oracle(aztec_utl_doesNullifierExist)]\nunconstrained fn does_nullifier_exist_oracle(_inner_nullifier: Field) -> bool {}\n"
|
|
1534
1534
|
},
|
|
1535
|
-
"
|
|
1535
|
+
"195": {
|
|
1536
1536
|
"function_locations": [
|
|
1537
1537
|
{
|
|
1538
1538
|
"name": "emit_offchain_effect",
|
|
@@ -1546,7 +1546,7 @@
|
|
|
1546
1546
|
"path": "/home/aztec-dev/aztec-packages/noir-projects/aztec-nr/aztec/src/oracle/offchain_effect.nr",
|
|
1547
1547
|
"source": "use crate::protocol::traits::Serialize;\n\n/// Emits data that will be delivered to PXE unaltered. This data can be interpreted freely by a downstream consumer\n/// (such as a wallet).\n///\n/// # Arguments\n///\n/// * `data` - The data to emit.\npub unconstrained fn emit_offchain_effect<T>(data: T)\nwhere\n T: Serialize,\n{\n emit_offchain_effect_oracle(data.serialize());\n}\n\n#[oracle(aztec_utl_emitOffchainEffect)]\nunconstrained fn emit_offchain_effect_oracle<let N: u32>(data: [Field; N]) {}\n"
|
|
1548
1548
|
},
|
|
1549
|
-
"
|
|
1549
|
+
"196": {
|
|
1550
1550
|
"function_locations": [
|
|
1551
1551
|
{
|
|
1552
1552
|
"name": "assert_valid_public_call_data",
|
|
@@ -1564,7 +1564,7 @@
|
|
|
1564
1564
|
"path": "/home/aztec-dev/aztec-packages/noir-projects/aztec-nr/aztec/src/oracle/public_call.nr",
|
|
1565
1565
|
"source": "/// Validates public calldata by checking that the preimage exists and the cumulative size is within limits.\n///\n/// The check is unconstrained and the only purpose of it is to fail early in case of calldata overflow or a bug in\n/// calldata hashing.\npub(crate) fn assert_valid_public_call_data(calldata_hash: Field) {\n // Safety: This oracle call returns nothing: we only call it for its side effects (validating the calldata).\n // It is therefore always safe to call.\n unsafe {\n assert_valid_public_call_data_oracle_wrapper(calldata_hash)\n }\n}\n\nunconstrained fn assert_valid_public_call_data_oracle_wrapper(calldata_hash: Field) {\n assert_valid_public_call_data_oracle(calldata_hash)\n}\n\n#[oracle(aztec_prv_assertValidPublicCalldata)]\nunconstrained fn assert_valid_public_call_data_oracle(_calldata_hash: Field) {}\n"
|
|
1566
1566
|
},
|
|
1567
|
-
"
|
|
1567
|
+
"198": {
|
|
1568
1568
|
"function_locations": [
|
|
1569
1569
|
{
|
|
1570
1570
|
"name": "get_shared_secret_oracle",
|
|
@@ -1578,7 +1578,7 @@
|
|
|
1578
1578
|
"path": "/home/aztec-dev/aztec-packages/noir-projects/aztec-nr/aztec/src/oracle/shared_secret.nr",
|
|
1579
1579
|
"source": "use crate::protocol::address::aztec_address::AztecAddress;\nuse crate::protocol::point::Point;\n\n#[oracle(aztec_utl_getSharedSecret)]\nunconstrained fn get_shared_secret_oracle(\n address: AztecAddress,\n ephPk: Point,\n contract_address: AztecAddress,\n) -> Field {}\n\n/// Returns an app-siloed shared secret between `address` and someone who knows the secret key behind an ephemeral\n/// public key `ephPk`.\n///\n/// The returned value is a Field `s_app`, computed as:\n///\n/// ```text\n/// S = address_secret * ephPk (raw ECDH point)\n/// s_app = h(DOM_SEP, S.x, S.y, contract) (app-siloed scalar)\n/// ```\n///\n/// where `contract` is the address of the calling contract. The oracle host validates this matches its execution\n/// context.\n///\n/// Without app-siloing, a malicious contract could call this oracle with public information (address, ephPk) and\n/// obtain the same raw secret as the legitimate contract, enabling cross-contract decryption. By including the\n/// contract address in the hash, each contract receives a different `s_app`, preventing this attack.\n///\n/// Callers derive indexed subkeys from `s_app` via\n/// [`derive_shared_secret_subkey`](crate::keys::ecdh_shared_secret::derive_shared_secret_subkey).\npub unconstrained fn get_shared_secret(address: AztecAddress, ephPk: Point, contract_address: AztecAddress) -> Field {\n get_shared_secret_oracle(address, ephPk, contract_address)\n}\n"
|
|
1580
1580
|
},
|
|
1581
|
-
"
|
|
1581
|
+
"199": {
|
|
1582
1582
|
"function_locations": [
|
|
1583
1583
|
{
|
|
1584
1584
|
"name": "get_from_public_storage_oracle",
|
|
@@ -1596,7 +1596,7 @@
|
|
|
1596
1596
|
"path": "/home/aztec-dev/aztec-packages/noir-projects/aztec-nr/aztec/src/oracle/storage.nr",
|
|
1597
1597
|
"source": "use crate::protocol::{abis::block_header::BlockHeader, address::AztecAddress, traits::{Hash, Packable, ToField}};\n\n#[oracle(aztec_utl_getFromPublicStorage)]\nunconstrained fn get_from_public_storage_oracle<let N: u32>(\n block_hash: Field,\n address: Field,\n storage_slot: Field,\n length: u32,\n) -> [Field; N] {}\n\n// TODO(F-498): review naming consistency\npub unconstrained fn raw_storage_read<let N: u32>(\n block_hash_to_read_from: Field,\n address: AztecAddress,\n storage_slot: Field,\n) -> [Field; N] {\n get_from_public_storage_oracle(block_hash_to_read_from, address.to_field(), storage_slot, N)\n}\n\n// TODO(F-498): review naming consistency\npub unconstrained fn storage_read<T>(header_to_read_from: BlockHeader, address: AztecAddress, storage_slot: Field) -> T\nwhere\n T: Packable,\n{\n let block_hash_to_read_from = header_to_read_from.hash();\n T::unpack(\n raw_storage_read(block_hash_to_read_from, address, storage_slot),\n )\n}\n"
|
|
1598
1598
|
},
|
|
1599
|
-
"
|
|
1599
|
+
"200": {
|
|
1600
1600
|
"function_locations": [
|
|
1601
1601
|
{
|
|
1602
1602
|
"name": "notify_revertible_phase_start",
|
|
@@ -1622,7 +1622,7 @@
|
|
|
1622
1622
|
"path": "/home/aztec-dev/aztec-packages/noir-projects/aztec-nr/aztec/src/oracle/tx_phase.nr",
|
|
1623
1623
|
"source": "/// Notifies PXE of the side effect counter at which the revertible phase begins.\n///\n/// PXE uses it to classify notes and nullifiers as revertible or non-revertible in its note cache. This information is\n/// then fed to kernels as hints.\npub(crate) fn notify_revertible_phase_start(counter: u32) {\n // Safety: This oracle call returns nothing: we only call it for its side effects. It is therefore always safe to\n // call.\n unsafe { notify_revertible_phase_start_oracle_wrapper(counter) };\n}\n\n/// Returns whether a side effect counter falls in the revertible phase of the transaction.\npub(crate) unconstrained fn is_execution_in_revertible_phase(current_counter: u32) -> bool {\n is_execution_in_revertible_phase_oracle(current_counter)\n}\n\nunconstrained fn notify_revertible_phase_start_oracle_wrapper(counter: u32) {\n notify_revertible_phase_start_oracle(counter);\n}\n\n#[oracle(aztec_prv_notifyRevertiblePhaseStart)]\nunconstrained fn notify_revertible_phase_start_oracle(_counter: u32) {}\n\n#[oracle(aztec_prv_isExecutionInRevertiblePhase)]\nunconstrained fn is_execution_in_revertible_phase_oracle(current_counter: u32) -> bool {}\n"
|
|
1624
1624
|
},
|
|
1625
|
-
"
|
|
1625
|
+
"201": {
|
|
1626
1626
|
"function_locations": [
|
|
1627
1627
|
{
|
|
1628
1628
|
"name": "assert_compatible_oracle_version",
|
|
@@ -1646,53 +1646,53 @@
|
|
|
1646
1646
|
}
|
|
1647
1647
|
],
|
|
1648
1648
|
"path": "/home/aztec-dev/aztec-packages/noir-projects/aztec-nr/aztec/src/oracle/version.nr",
|
|
1649
|
-
"source": "/// The oracle version constants are used to check that the oracle interface is in sync between PXE and Aztec.nr.\n/// We version the oracle interface as `major.minor` where:\n/// - `major` = backward-breaking changes (must match exactly between PXE and Aztec.nr)\n/// - `minor` = oracle additions (non-breaking; PXE minor >= contract minor)\n///\n/// The TypeScript counterparts are in `oracle_version.ts`.\n///\n/// @dev Whenever a contract function or Noir test is run, the `aztec_utl_assertCompatibleOracleVersion` oracle is\n/// called. If the major version is incompatible, an error is thrown immediately. The minor version is recorded by\n/// the PXE and used to provide helpful error messages if a contract calls an oracle that doesn't exist. We don't throw\n/// immediately if AZTEC_NR_MINOR > PXE_MINOR because if a contract is updated to use a newer Aztec.nr dependency\n/// without actually using any of the new oracles then there is no reason to throw.\npub global ORACLE_VERSION_MAJOR: Field = 22;\npub global ORACLE_VERSION_MINOR: Field =
|
|
1649
|
+
"source": "/// The oracle version constants are used to check that the oracle interface is in sync between PXE and Aztec.nr.\n/// We version the oracle interface as `major.minor` where:\n/// - `major` = backward-breaking changes (must match exactly between PXE and Aztec.nr)\n/// - `minor` = oracle additions (non-breaking; PXE minor >= contract minor)\n///\n/// The TypeScript counterparts are in `oracle_version.ts`.\n///\n/// @dev Whenever a contract function or Noir test is run, the `aztec_utl_assertCompatibleOracleVersion` oracle is\n/// called. If the major version is incompatible, an error is thrown immediately. The minor version is recorded by\n/// the PXE and used to provide helpful error messages if a contract calls an oracle that doesn't exist. We don't throw\n/// immediately if AZTEC_NR_MINOR > PXE_MINOR because if a contract is updated to use a newer Aztec.nr dependency\n/// without actually using any of the new oracles then there is no reason to throw.\npub global ORACLE_VERSION_MAJOR: Field = 22;\npub global ORACLE_VERSION_MINOR: Field = 2;\n\n/// Asserts that the version of the oracle is compatible with the version expected by the contract.\npub fn assert_compatible_oracle_version() {\n // Safety: This oracle call returns nothing: we only call it to check Aztec.nr and Oracle interface versions are\n // compatible. It is therefore always safe to call.\n unsafe {\n assert_compatible_oracle_version_wrapper();\n }\n}\n\nunconstrained fn assert_compatible_oracle_version_wrapper() {\n assert_compatible_oracle_version_oracle(ORACLE_VERSION_MAJOR, ORACLE_VERSION_MINOR);\n}\n\n#[oracle(aztec_utl_assertCompatibleOracleVersionV2)]\nunconstrained fn assert_compatible_oracle_version_oracle(major: Field, minor: Field) {}\n\nmod test {\n use super::{assert_compatible_oracle_version_oracle, ORACLE_VERSION_MAJOR, ORACLE_VERSION_MINOR};\n\n #[test]\n unconstrained fn compatible_oracle_version() {\n assert_compatible_oracle_version_oracle(ORACLE_VERSION_MAJOR, ORACLE_VERSION_MINOR);\n }\n\n #[test(should_fail_with = \"Incompatible aztec cli version:\")]\n unconstrained fn incompatible_oracle_version_major() {\n let arbitrary_incorrect_major = 318183437;\n assert_compatible_oracle_version_oracle(arbitrary_incorrect_major, ORACLE_VERSION_MINOR);\n }\n}\n"
|
|
1650
1650
|
},
|
|
1651
|
-
"
|
|
1651
|
+
"205": {
|
|
1652
1652
|
"function_locations": [
|
|
1653
1653
|
{
|
|
1654
1654
|
"name": "<impl StateVariable<1, Context> for Map<K, V, Context>>::new",
|
|
1655
|
-
"start":
|
|
1655
|
+
"start": 4080
|
|
1656
1656
|
},
|
|
1657
1657
|
{
|
|
1658
1658
|
"name": "<impl StateVariable<1, Context> for Map<K, V, Context>>::get_storage_slot",
|
|
1659
|
-
"start":
|
|
1659
|
+
"start": 4265
|
|
1660
1660
|
},
|
|
1661
1661
|
{
|
|
1662
1662
|
"name": "Map<K, V, Context>::at",
|
|
1663
|
-
"start":
|
|
1663
|
+
"start": 4999
|
|
1664
1664
|
}
|
|
1665
1665
|
],
|
|
1666
1666
|
"path": "/home/aztec-dev/aztec-packages/noir-projects/aztec-nr/aztec/src/state_vars/map.nr",
|
|
1667
|
-
"source": "use crate::protocol::{storage::map::derive_storage_slot_in_map, traits::ToField};\nuse crate::state_vars::StateVariable;\n\n/// A key-value container for state variables.\n///\n/// A
|
|
1667
|
+
"source": "use crate::protocol::{storage::map::derive_storage_slot_in_map, traits::ToField};\nuse crate::state_vars::StateVariable;\n\n/// A key-value container for state variables.\n///\n/// A `Map` wraps another state variable type (the 'value') and creates independent instances of it for each key,\n/// resulting in effectively as many state variables as there are key values. This behavior is similar to a Solidity\n/// `mapping (K => V)`.\n///\n/// ## Use Cases\n///\n/// Any scenario in which a fixed number of variables is insufficient to represent contract state requires `Map` e.g.\n/// per-election configuration in a voting contract, per-user state, etc. `Map` also composes naturally into more\n/// complex containers: an array could be implemented as a `Map` with indices as keys.\n///\n/// Note that some private state variables, such as [`PrivateMutable`](crate::state_vars::PrivateMutable) and\n/// [`PrivateSet`](crate::state_vars::PrivateSet) cannot be wrapped in a `Map`. These types implement the\n/// [`OwnedStateVariable`](crate::state_vars::OwnedStateVariable) trait, and as such should instead be wrapped using the\n/// [`Owned`](crate::state_vars::Owned) container, which can be thought of as a specialization of `Map` for private\n/// state 'owned' by a single account.\n///\n/// ## Examples\n///\n/// Since `Map` is also a [`StateVariable`], it is declared in the contract's\n/// [`#[storage]`](crate::macros::storage::storage) struct along with all others:\n///\n/// ```noir\n/// #[storage]\n/// struct Storage<C> {\n/// is_admin: Map<AztecAddress, PublicMutable<bool, C>, C>,\n/// vote_tallies: Map<ElectionId, PublicMutable<u128, C>, C>,\n/// }\n/// ```\n///\n/// ### Multiple `Map`s\n///\n/// There is no limit to how many `Map` containers a contract can have, and `Map`s can themselves wrap other `Map`s,\n/// resulting in nested layouts.\n///\n/// ```noir\n/// #[storage]\n/// struct Storage<C> {\n/// // A nested map where the first key is an address and the second a `Year` type, such that\n/// // self.storage.user_yearly_config.at(user).at(year) results in distinct `UserConfig` variables for each\n/// // `(user, year)` tuple.\n/// user_yearly_config: Map<AztecAddress, Map<Year, UserConfig<C>, C>, C>,\n/// }\n/// ```\n///\n/// ## Requirements\n///\n/// The value type `V` must implement the [`StateVariable`] trait. The key type `K` must implement the [`ToField`] trait\n/// in such a way that the `Field` representation is unique for each key.\n///\n/// For key types that cannot be converted into a `Field`, consider wrapping them in a type that implements `ToField` as\n/// the hash of the key.\n///\n/// ```noir\n/// struct MyKey {\n/// a: Field,\n/// b: Field,\n/// }\n///\n/// struct MyKeyWrapper {\n/// inner: MyKey,\n/// }\n///\n/// impl ToField for MyKeyWrapper {\n/// fn to_field(self) -> Field {\n/// poseidon2_hash([self.inner.a, self.inner.b])\n/// }\n/// }\n///\n/// #[storage]\n/// struct Storage<C> {\n/// a: Map<MyKeyWrapper, PublicMutable<bool, C>, C>,\n/// }\n/// ```\n///\n/// ## Implementation Details\n///\n/// Like all other state variables, `Map` gets assigned a unique storage slot. It uses this value to derive unique\n/// storage slots for each key, resulting in independent state variables. Nothing gets stored at `Map`'s storage slot.\n/// This is equivalent to Solidity's implementation of `mapping`.\n///\n/// Because the storage slot derivation is done using a hash function, it is not possible to compute the key (preimage)\n/// used to obtain a given derived state variable slot.\npub struct Map<K, V, Context> {\n pub context: Context,\n storage_slot: Field,\n}\n\n// Map reserves a single storage slot regardless of what it stores because nothing is stored at said slot: it is only\n// used to derive the storage slots of nested state variables, which is expected to never result in collisions or slots\n// being close to one another due to these being hashes. This mirrors the strategy adopted by Solidity mappings.\nimpl<K, V, Context> StateVariable<1, Context> for Map<K, V, Context> {\n fn new(context: Context, storage_slot: Field) -> Self {\n assert(storage_slot != 0, \"Storage slot 0 not allowed. Storage slots must start from 1.\");\n Map { context, storage_slot }\n }\n\n fn get_storage_slot(self) -> Field {\n self.storage_slot\n }\n}\n\nimpl<K, V, Context> Map<K, V, Context> {\n /// Returns the state variable for a key.\n ///\n /// This derives a unique storage slot for `key` and returns a state variable of type `V` at that slot. It is\n /// equivalent to accessing a `mapping` via the `[]` operator in Solidity (e.g. `balances[user]` would be\n /// `balances.at(user)`).\n ///\n /// ## Example\n ///\n /// ```noir\n /// #[external(\"utility\")]\n /// fn get_election_vote_tallies(election: ElectionId) -> u128 {\n /// self.storage.vote_tallies.at(election).read()\n /// }\n /// ```\n pub fn at<let N: u32>(self, key: K) -> V\n where\n K: ToField,\n V: StateVariable<N, Context>,\n {\n V::new(\n self.context,\n derive_storage_slot_in_map(self.storage_slot, key),\n )\n }\n}\n"
|
|
1668
1668
|
},
|
|
1669
|
-
"
|
|
1669
|
+
"218": {
|
|
1670
1670
|
"function_locations": [
|
|
1671
1671
|
{
|
|
1672
1672
|
"name": "<impl StateVariable<M, Context> for PublicMutable<T, Context>>::new",
|
|
1673
|
-
"start":
|
|
1673
|
+
"start": 4374
|
|
1674
1674
|
},
|
|
1675
1675
|
{
|
|
1676
1676
|
"name": "<impl StateVariable<M, Context> for PublicMutable<T, Context>>::get_storage_slot",
|
|
1677
|
-
"start":
|
|
1677
|
+
"start": 4569
|
|
1678
1678
|
},
|
|
1679
1679
|
{
|
|
1680
1680
|
"name": "PublicMutable<T, PublicContext>::read",
|
|
1681
|
-
"start":
|
|
1681
|
+
"start": 6177
|
|
1682
1682
|
},
|
|
1683
1683
|
{
|
|
1684
1684
|
"name": "PublicMutable<T, PublicContext>::write",
|
|
1685
|
-
"start":
|
|
1685
|
+
"start": 7896
|
|
1686
1686
|
},
|
|
1687
1687
|
{
|
|
1688
1688
|
"name": "PublicMutable<T, UtilityContext>::read",
|
|
1689
|
-
"start":
|
|
1689
|
+
"start": 8774
|
|
1690
1690
|
}
|
|
1691
1691
|
],
|
|
1692
1692
|
"path": "/home/aztec-dev/aztec-packages/noir-projects/aztec-nr/aztec/src/state_vars/public_mutable.nr",
|
|
1693
|
-
"source": "use crate::context::{PublicContext, UtilityContext};\nuse crate::protocol::traits::Packable;\nuse crate::state_vars::StateVariable;\n\nmod test;\n\n/// Mutable public values.\n///\n/// This is one of the most basic public state variables. It is equivalent to a non-`immutable` non-`constant` Solidity\n/// state variable.\n///\n/// It represents a public value of type `T` that can be written to repeatedly over the lifetime of the contract,\n/// allowing the last value that was written to be read.\n///\n/// ## Access Patterns\n///\n/// A value stored in a `PublicMutable` can be read and written from public contract functions.\n///\n/// It is not possible to read or write a `PublicMutable` from private contract functions. A common pattern is to have\n/// these functions [enqueue a public self\n/// calls](crate::contract_self::ContractSelfPrivate::enqueue) in which the\n/// required operation is performed.\n///\n/// For an immutable variant which can be read from private functions, see\n/// [`PublicImmutable`](crate::state_vars::PublicImmutable).\n///\n/// For a mutable (with restrictions) variant which can be read from private functions see\n/// [`DelayedPublicMutable`](crate::state_vars::DelayedPublicMutable).\n///\n/// ## Privacy\n///\n/// `PublicMutable` provides zero privacy. All write and read operations are public: the entire network can see these\n/// accesses and the data involved.\n///\n/// ## Use Cases\n///\n/// This is suitable for any kind of global state that needs to be accessible by everyone. For example, a token may\n/// have a public total supply, or a voting contract may have public vote tallies.\n///\n/// Note that contracts having public values does not necessarily mean the actions that update these values must\n/// themselves be wholly public. For example, the token could allow for private minting and burning, and casting a vote\n/// could be kept private: these private functions would enqueue a public function that writes to the `PublicMutable`.\n///\n/// Similarly, private functions can enqueue a public call in which the `PublicMutable` is checked to meet some\n/// condition. For example, a private action might be executable only if the vote count has exceeded some threshold, in\n/// which case the private function would enqueue a public function that reads from the `PublicMutable`.\n///\n/// Such patterns preserve the privacy of the account that executed the action, as well as details related to the\n/// private execution itself, but they _do_ reveal that the transaction interacted with the `PublicMutable` value (and\n/// hence that the contract was called), as all accesses to it are public. The\n/// [`only_self`](crate::macros::functions::only_self) attribute is very useful when implementing this.\n///\n/// ## Examples\n///\n/// Declaring a `PublicMutable` in the contract's [
|
|
1693
|
+
"source": "use crate::context::{PublicContext, UtilityContext};\nuse crate::protocol::traits::Packable;\nuse crate::state_vars::StateVariable;\n\nmod test;\n\n/// Mutable public values.\n///\n/// This is one of the most basic public state variables. It is equivalent to a non-`immutable` non-`constant` Solidity\n/// state variable.\n///\n/// It represents a public value of type `T` that can be written to repeatedly over the lifetime of the contract,\n/// allowing the last value that was written to be read.\n///\n/// ## Access Patterns\n///\n/// A value stored in a `PublicMutable` can be read and written from public contract functions.\n///\n/// It is not possible to read or write a `PublicMutable` from private contract functions. A common pattern is to have\n/// these functions [enqueue a public self\n/// calls](crate::contract_self::ContractSelfPrivate::enqueue) in which the\n/// required operation is performed.\n///\n/// For an immutable variant which can be read from private functions, see\n/// [`PublicImmutable`](crate::state_vars::PublicImmutable).\n///\n/// For a mutable (with restrictions) variant which can be read from private functions see\n/// [`DelayedPublicMutable`](crate::state_vars::DelayedPublicMutable).\n///\n/// ## Privacy\n///\n/// `PublicMutable` provides zero privacy. All write and read operations are public: the entire network can see these\n/// accesses and the data involved.\n///\n/// ## Use Cases\n///\n/// This is suitable for any kind of global state that needs to be accessible by everyone. For example, a token may\n/// have a public total supply, or a voting contract may have public vote tallies.\n///\n/// Note that contracts having public values does not necessarily mean the actions that update these values must\n/// themselves be wholly public. For example, the token could allow for private minting and burning, and casting a vote\n/// could be kept private: these private functions would enqueue a public function that writes to the `PublicMutable`.\n///\n/// Similarly, private functions can enqueue a public call in which the `PublicMutable` is checked to meet some\n/// condition. For example, a private action might be executable only if the vote count has exceeded some threshold, in\n/// which case the private function would enqueue a public function that reads from the `PublicMutable`.\n///\n/// Such patterns preserve the privacy of the account that executed the action, as well as details related to the\n/// private execution itself, but they _do_ reveal that the transaction interacted with the `PublicMutable` value (and\n/// hence that the contract was called), as all accesses to it are public. The\n/// [`only_self`](crate::macros::functions::only_self) attribute is very useful when implementing this.\n///\n/// ## Examples\n///\n/// Declaring a `PublicMutable` in the contract's [`#[storage]`](crate::macros::storage::storage) struct requires\n/// specifying the type `T` that is stored in the variable:\n///\n/// ```noir\n/// #[storage]\n/// struct Storage<C> {\n/// total_supply: PublicMutable<u128, C>,\n/// public_balances: Map<AztecAddress, PublicMutable<u128, C>, C>,\n///\n/// vote_tallies: Map<ElectionId, PublicMutable<u128, C>, C>,\n/// }\n/// ```\n///\n/// ## Requirements\n///\n/// The type `T` stored in the `PublicMutable` must implement the [`Packable`](crate::protocol::traits::Packable)\n/// trait.\n///\n/// ## Implementation Details\n///\n/// Values are packed and stored directly in the public storage tree, with no overhead. A `PublicMutable` therefore\n/// takes up as many storage slots as the packing length of the stored type `T`.\n///\n/// Private reads are not possible because private functions do not have access to the current network state, only the\n/// _past_ state at the anchor block. They _can_ perform historical reads of `PublicMutable` values at past times, but\n/// they have no way to guarantee that the value has not changed since then.\n/// [`PublicImmutable`](crate::state_vars::PublicImmutable) and\n/// [`DelayedPublicMutable`](crate::state_vars::DelayedPublicMutable) are examples of public state variables that _can_\n/// be read privately by restricting mutation.\npub struct PublicMutable<T, Context> {\n context: Context,\n storage_slot: Field,\n}\n\nimpl<T, Context, let M: u32> StateVariable<M, Context> for PublicMutable<T, Context>\nwhere\n T: Packable<N = M>,\n{\n fn new(context: Context, storage_slot: Field) -> Self {\n assert(storage_slot != 0, \"Storage slot 0 not allowed. Storage slots must start from 1.\");\n PublicMutable { context, storage_slot }\n }\n\n fn get_storage_slot(self) -> Field {\n self.storage_slot\n }\n}\n\nimpl<T> PublicMutable<T, PublicContext> {\n /// Returns the current value.\n ///\n /// If [`write`](PublicMutable::write) has never been called, then this returns the default empty public storage\n /// value, which is all zeroes - equivalent to `let t = T::unpack(std::mem::zeroed());`.\n ///\n /// It is not possible to detect if a `PublicMutable` has ever been initialized or not other than by testing for\n /// the zero sentinel value. For a more robust solution, store an `Option<T>` in the `PublicMutable`.\n ///\n /// ## Examples\n ///\n /// A public getter that returns the current value:\n /// ```noir\n /// #[external(\"public\")]\n /// fn get_total_supply() -> u128 {\n /// self.storage.total_supply.read()\n /// }\n /// ```\n ///\n /// An [`only_self`](crate::macros::functions::only_self) helper that asserts a condition a private function\n /// requires:\n /// ```noir\n /// #[external(\"private\")]\n /// fn execute_proposal(election_id: ElectionId) {\n /// self.enqueue_self._assert_vote_passed(election_id);\n ///\n /// // execute the proposal - this remains private\n /// }\n ///\n /// #[external(\"public\")]\n /// #[only_self]\n /// fn _assert_vote_passed(election_id: ElectionId) {\n /// assert(self.storage.vote_tallies.at(election_id).read() >= VOTE_PASSED_THRESHOLD);\n /// }\n /// ```\n ///\n /// ## Cost\n ///\n /// The `SLOAD` AVM opcode is invoked a number of times equal to `T`'s packed length.\n pub fn read(self) -> T\n where\n T: Packable,\n {\n self.context.storage_read(self.storage_slot)\n }\n\n /// Stores a new value.\n ///\n /// The old value is overridden and cannot be recovered. The new value can be immediately retrieved by\n /// [`read`](PublicMutable::read).\n ///\n /// ## Examples\n ///\n /// A public setter that updates the current value:\n /// ```noir\n /// #[external(\"public\")]\n /// fn mint_tokens(recipient: AztecAddress, amount: u128) {\n /// let current_recipient_balance = self.storage.public_balances.at(recipient).read();\n /// self.storage.public_balances.at(recipient).write(current_recipient_balance + amount);\n ///\n /// let current_supply = self.storage.total_supply.read();\n /// self.storage.total_supply.write(current_supply + amount);\n /// }\n /// ```\n ///\n /// An [`only_self`](crate::macros::functions::only_self) helper that updates public state triggered by a private\n /// function:\n /// ```noir\n /// #[external(\"private\")]\n /// fn vote_for_proposal(election_id: ElectionId, votes: u128) {\n /// // validate the sender can cast this many votes - this remains private\n ///\n /// self.enqueue_self._tally_vote(election_id, votes);\n /// }\n ///\n /// #[external(\"public\")]\n /// #[only_self]\n /// fn _tally_vote(election_id: ElectionId, votes: u128) {\n /// let current = self.storage.vote_tallies.at(election_id).read();\n /// self.storage.vote_tallies.at(election_id).write(current + votes);\n /// }\n /// ```\n ///\n /// ## Cost\n ///\n /// The `SSTORE` AVM opcode is invoked a number of times equal to `T`'s packed length.\n pub fn write(self, value: T)\n where\n T: Packable,\n {\n self.context.storage_write(self.storage_slot, value);\n }\n}\n\nimpl<T> PublicMutable<T, UtilityContext> {\n /// Returns the value at the anchor block.\n ///\n /// If [`write`](PublicMutable::write) has never been called, then this returns the default empty public storage\n /// value, which is all zeroes - equivalent to `let t = T::unpack(std::mem::zeroed());`.\n ///\n /// It is not possible to detect if a `PublicMutable` has ever been initialized or not other than by testing for\n /// the zero sentinel value. For a more robust solution, store an `Option<T>` in the `PublicMutable`.\n ///\n /// ## Examples\n ///\n /// ```noir\n /// #[external(\"utility\")]\n /// fn get_total_supply() -> u128 {\n /// self.storage.total_supply.read()\n /// }\n /// ```\n pub unconstrained fn read(self) -> T\n where\n T: Packable,\n {\n self.context.storage_read(self.storage_slot)\n }\n}\n"
|
|
1694
1694
|
},
|
|
1695
|
-
"
|
|
1695
|
+
"247": {
|
|
1696
1696
|
"function_locations": [
|
|
1697
1697
|
{
|
|
1698
1698
|
"name": "append",
|
|
@@ -1714,7 +1714,7 @@
|
|
|
1714
1714
|
"path": "/home/aztec-dev/aztec-packages/noir-projects/aztec-nr/aztec/src/utils/array/append.nr",
|
|
1715
1715
|
"source": "/// Appends the elements of the second `BoundedVec` to the end of the first one. The resulting `BoundedVec` can have\n/// any arbitrary maximum length, but it must be large enough to fit all of the elements of both the first and second\n/// vectors.\npub fn append<T, let ALen: u32, let BLen: u32, let DstLen: u32>(\n a: BoundedVec<T, ALen>,\n b: BoundedVec<T, BLen>,\n) -> BoundedVec<T, DstLen> {\n let mut dst = BoundedVec::new();\n\n dst.extend_from_bounded_vec(a);\n dst.extend_from_bounded_vec(b);\n\n dst\n}\n\nmod test {\n use super::append;\n\n #[test]\n unconstrained fn append_empty_vecs() {\n let a: BoundedVec<_, 3> = BoundedVec::new();\n let b: BoundedVec<_, 14> = BoundedVec::new();\n\n let result: BoundedVec<Field, 5> = append(a, b);\n\n assert_eq(result.len(), 0);\n assert_eq(result.storage(), std::mem::zeroed());\n }\n\n #[test]\n unconstrained fn append_non_empty_vecs() {\n let a: BoundedVec<_, 3> = BoundedVec::from_array([1, 2, 3]);\n let b: BoundedVec<_, 14> = BoundedVec::from_array([4, 5, 6]);\n\n let result: BoundedVec<Field, 8> = append(a, b);\n\n assert_eq(result.len(), 6);\n assert_eq(result.storage(), [1, 2, 3, 4, 5, 6, std::mem::zeroed(), std::mem::zeroed()]);\n }\n\n #[test(should_fail_with = \"out of bounds\")]\n unconstrained fn append_non_empty_vecs_insufficient_max_len() {\n let a: BoundedVec<_, 3> = BoundedVec::from_array([1, 2, 3]);\n let b: BoundedVec<_, 14> = BoundedVec::from_array([4, 5, 6]);\n\n let _: BoundedVec<Field, 5> = append(a, b);\n }\n}\n"
|
|
1716
1716
|
},
|
|
1717
|
-
"
|
|
1717
|
+
"250": {
|
|
1718
1718
|
"function_locations": [
|
|
1719
1719
|
{
|
|
1720
1720
|
"name": "subarray",
|
|
@@ -1744,7 +1744,7 @@
|
|
|
1744
1744
|
"path": "/home/aztec-dev/aztec-packages/noir-projects/aztec-nr/aztec/src/utils/array/subarray.nr",
|
|
1745
1745
|
"source": "/// Returns `DstLen` elements from a source array, starting at `offset`. `DstLen` must not be larger than the number of\n/// elements past `offset`.\n///\n/// Examples:\n/// ```\n/// let foo: [Field; 2] = subarray([1, 2, 3, 4, 5], 2);\n/// assert_eq(foo, [3, 4]);\n///\n/// let bar: [Field; 5] = subarray([1, 2, 3, 4, 5], 2); // fails - we can't return 5 elements since only 3 remain\n/// ```\npub fn subarray<T, let SrcLen: u32, let DstLen: u32>(src: [T; SrcLen], offset: u32) -> [T; DstLen] {\n assert(offset + DstLen <= SrcLen, \"DstLen too large for offset\");\n\n let mut dst: [T; DstLen] = std::mem::zeroed();\n for i in 0..DstLen {\n dst[i] = src[i + offset];\n }\n\n dst\n}\n\nmod test {\n use super::subarray;\n\n #[test]\n unconstrained fn subarray_into_empty() {\n // In all of these cases we're setting DstLen to be 0, so we always get back an empty array.\n assert_eq(subarray::<Field, _, _>([], 0), []);\n assert_eq(subarray([1, 2, 3, 4, 5], 0), []);\n assert_eq(subarray([1, 2, 3, 4, 5], 2), []);\n }\n\n #[test]\n unconstrained fn subarray_complete() {\n assert_eq(subarray::<Field, _, _>([], 0), []);\n assert_eq(subarray([1, 2, 3, 4, 5], 0), [1, 2, 3, 4, 5]);\n }\n\n #[test]\n unconstrained fn subarray_different_end_sizes() {\n // We implicitly select how many values to read in the size of the return array\n assert_eq(subarray([1, 2, 3, 4, 5], 1), [2, 3, 4, 5]);\n assert_eq(subarray([1, 2, 3, 4, 5], 1), [2, 3, 4]);\n assert_eq(subarray([1, 2, 3, 4, 5], 1), [2, 3]);\n assert_eq(subarray([1, 2, 3, 4, 5], 1), [2]);\n }\n\n #[test(should_fail_with = \"DstLen too large for offset\")]\n unconstrained fn subarray_offset_too_large() {\n // With an offset of 1 we can only request up to 4 elements\n let _: [_; 5] = subarray([1, 2, 3, 4, 5], 1);\n }\n\n #[test(should_fail)]\n unconstrained fn subarray_bad_return_value() {\n assert_eq(subarray([1, 2, 3, 4, 5], 1), [3, 3, 4, 5]);\n }\n}\n"
|
|
1746
1746
|
},
|
|
1747
|
-
"
|
|
1747
|
+
"251": {
|
|
1748
1748
|
"function_locations": [
|
|
1749
1749
|
{
|
|
1750
1750
|
"name": "subbvec",
|
|
@@ -1786,7 +1786,7 @@
|
|
|
1786
1786
|
"path": "/home/aztec-dev/aztec-packages/noir-projects/aztec-nr/aztec/src/utils/array/subbvec.nr",
|
|
1787
1787
|
"source": "use crate::utils::array;\n\n/// Returns `DstMaxLen` elements from a source BoundedVec, starting at `offset`. `offset` must not be larger than the\n/// original length, and `DstLen` must not be larger than the total number of elements past `offset` (including the\n/// zeroed elements past `len()`).\n///\n/// Only elements at the beginning of the vector can be removed: it is not possible to also remove elements at the end\n/// of the vector by passing a value for `DstLen` that is smaller than `len() - offset`.\n///\n/// Examples:\n/// ```\n/// let foo = BoundedVec::<_, 10>::from_array([1, 2, 3, 4, 5]);\n/// assert_eq(subbvec(foo, 2), BoundedVec::<_, 8>::from_array([3, 4, 5]));\n///\n/// let bar: BoundedVec<_, 1> = subbvec(foo, 2); // fails - we can't return just 1 element since 3 remain\n/// let baz: BoundedVec<_, 10> = subbvec(foo, 3); // fails - we can't return 10 elements since only 7 remain\n/// ```\npub fn subbvec<T, let SrcMaxLen: u32, let DstMaxLen: u32>(\n bvec: BoundedVec<T, SrcMaxLen>,\n offset: u32,\n) -> BoundedVec<T, DstMaxLen> {\n // from_parts_unchecked does not verify that the elements past len are zeroed, but that is not an issue in our case\n // because we're constructing the new storage array as a subarray of the original one (which should have zeroed\n // storage past len), guaranteeing correctness. This is because `subarray` does not allow extending arrays past\n // their original length.\n BoundedVec::from_parts_unchecked(array::subarray(bvec.storage(), offset), bvec.len() - offset)\n}\n\nmod test {\n use super::subbvec;\n\n #[test]\n unconstrained fn subbvec_empty() {\n let bvec = BoundedVec::<Field, 0>::from_array([]);\n assert_eq(subbvec(bvec, 0), bvec);\n }\n\n #[test]\n unconstrained fn subbvec_complete() {\n let bvec = BoundedVec::<_, 10>::from_array([1, 2, 3, 4, 5]);\n assert_eq(subbvec(bvec, 0), bvec);\n\n let smaller_capacity = BoundedVec::<_, 5>::from_array([1, 2, 3, 4, 5]);\n assert_eq(subbvec(bvec, 0), smaller_capacity);\n }\n\n #[test]\n unconstrained fn subbvec_partial() {\n let bvec = BoundedVec::<_, 10>::from_array([1, 2, 3, 4, 5]);\n\n assert_eq(subbvec(bvec, 2), BoundedVec::<_, 8>::from_array([3, 4, 5]));\n assert_eq(subbvec(bvec, 2), BoundedVec::<_, 3>::from_array([3, 4, 5]));\n }\n\n #[test]\n unconstrained fn subbvec_into_empty() {\n let bvec: BoundedVec<_, 10> = BoundedVec::from_array([1, 2, 3, 4, 5]);\n assert_eq(subbvec(bvec, 5), BoundedVec::<_, 5>::from_array([]));\n }\n\n #[test(should_fail)]\n unconstrained fn subbvec_offset_past_len() {\n let bvec = BoundedVec::<_, 10>::from_array([1, 2, 3, 4, 5]);\n let _: BoundedVec<_, 1> = subbvec(bvec, 6);\n }\n\n #[test(should_fail)]\n unconstrained fn subbvec_insufficient_dst_len() {\n let bvec = BoundedVec::<_, 10>::from_array([1, 2, 3, 4, 5]);\n\n // We're not providing enough space to hold all of the items inside the original BoundedVec. subbvec can cause\n // for the capacity to reduce, but not the length (other than by len - offset).\n let _: BoundedVec<_, 1> = subbvec(bvec, 2);\n }\n\n #[test(should_fail_with = \"DstLen too large for offset\")]\n unconstrained fn subbvec_dst_len_causes_enlarge() {\n let bvec = BoundedVec::<_, 10>::from_array([1, 2, 3, 4, 5]);\n\n // subbvec does not support capacity increases\n let _: BoundedVec<_, 11> = subbvec(bvec, 0);\n }\n\n #[test(should_fail_with = \"DstLen too large for offset\")]\n unconstrained fn subbvec_dst_len_too_large_for_offset() {\n let bvec = BoundedVec::<_, 10>::from_array([1, 2, 3, 4, 5]);\n\n // This effectively requests a capacity increase, since there'd be just one element plus the 5 empty slots,\n // which is less than 7.\n let _: BoundedVec<_, 7> = subbvec(bvec, 4);\n }\n}\n"
|
|
1788
1788
|
},
|
|
1789
|
-
"
|
|
1789
|
+
"254": {
|
|
1790
1790
|
"function_locations": [
|
|
1791
1791
|
{
|
|
1792
1792
|
"name": "encode_bytes_as_fields",
|
|
@@ -1812,7 +1812,7 @@
|
|
|
1812
1812
|
"path": "/home/aztec-dev/aztec-packages/noir-projects/aztec-nr/aztec/src/utils/conversion/bytes_as_fields.nr",
|
|
1813
1813
|
"source": "use std::static_assert;\n\n/// Encodes an array of bytes as fields.\n///\n/// Use\n/// [`decode_bytes_from_fields`](crate::utils::conversion::bytes_as_fields::decode_bytes_from_fields) to recover\n/// the original bytes.\n///\n/// The `bytes` array length must be a multiple of 31. If padding is added, it will need to be manually removed\n/// after decoding.\n///\n/// ## Encoding\n///\n/// Each 31-byte chunk is interpreted as a big-endian integer and stored in a `Field`. For input `[1, 10, 3, ..., 0]`\n/// (31 bytes), the resulting `Field` is `1 * 256^30 + 10 * 256^29 + 3 * 256^28 + ... + 0`.\npub fn encode_bytes_as_fields<let N: u32>(bytes: [u8; N]) -> [Field; N / 31] {\n static_assert(N % 31 == 0, \"N must be a multiple of 31\");\n\n let mut fields = [0; N / 31];\n for i in 0..N / 31 {\n let mut field = 0;\n for j in 0..31 {\n field = field * 256 + bytes[i * 31 + j] as Field;\n }\n fields[i] = field;\n }\n\n fields\n}\n\n/// Decodes fields back into bytes.\n///\n/// Inverse of\n/// [`encode_bytes_as_fields`](crate::utils::conversion::bytes_as_fields::encode_bytes_as_fields).\n/// Each input `Field` must fit in 248 bits; `Field::to_be_bytes::<31>()` fails the proof otherwise.\npub fn decode_bytes_from_fields<let N: u32>(fields: BoundedVec<Field, N>) -> BoundedVec<u8, N * 31> {\n let mut bytes = BoundedVec::new();\n for i in 0..fields.len() {\n let chunk: [u8; 31] = fields.get(i).to_be_bytes();\n for j in 0..31 {\n bytes.push(chunk[j]);\n }\n }\n bytes\n}\n\nmod tests {\n use crate::utils::array::subarray;\n use super::{decode_bytes_from_fields, encode_bytes_as_fields};\n\n #[test]\n unconstrained fn round_trips_bytes(input: [u8; 93]) {\n let fields = encode_bytes_as_fields(input);\n\n // In production the fields fly through the system and arrive as a BoundedVec on the other end.\n let fields_bvec = BoundedVec::<_, 6>::from_array(fields);\n let bytes_back = decode_bytes_from_fields(fields_bvec);\n\n assert_eq(bytes_back.len(), input.len());\n assert_eq(subarray(bytes_back.storage(), 0), input);\n }\n\n #[test(should_fail_with = \"N must be a multiple of 31\")]\n unconstrained fn encode_rejects_length_not_multiple_of_31() {\n let _fields = encode_bytes_as_fields([0; 32]);\n }\n\n #[test(should_fail_with = \"Field failed to decompose into specified 31 limbs\")]\n unconstrained fn decode_rejects_oversized_field() {\n // `Field::to_be_bytes::<31>()` fails the proof when a field has any bit above position 247 set.\n let oversized: Field = (1 as Field) * 2.pow_32(249);\n let input = BoundedVec::<_, 1>::from_array([oversized]);\n let _bytes = decode_bytes_from_fields(input);\n }\n}\n"
|
|
1814
1814
|
},
|
|
1815
|
-
"
|
|
1815
|
+
"255": {
|
|
1816
1816
|
"function_locations": [
|
|
1817
1817
|
{
|
|
1818
1818
|
"name": "encode_fields_as_bytes",
|
|
@@ -1862,7 +1862,7 @@
|
|
|
1862
1862
|
"path": "/home/aztec-dev/aztec-packages/noir-projects/aztec-nr/aztec/src/utils/conversion/fields_as_bytes.nr",
|
|
1863
1863
|
"source": "/// Encodes an array of fields as bytes.\n///\n/// Losslessly preserves any field value; use\n/// [`try_decode_fields_from_bytes`](crate::utils::conversion::fields_as_bytes::try_decode_fields_from_bytes) to\n/// recover the original fields.\n///\n/// ## Encoding\n///\n/// Each field is written as 32 big-endian bytes and the chunks are concatenated. The field array `[5, 42]` becomes:\n///\n/// ```text\n/// [0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,5, // First field (32 bytes)\n/// 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,42] // Second field (32 bytes)\n/// ```\n///\n/// ## Privacy\n///\n/// The BN254 modulus is `< 2^254`, so every 32-byte chunk has its top bit at zero and the next bit biased. The output\n/// is therefore distinguishable from uniform random bytes; take this into account when feeding it into anything that\n/// assumes uniform randomness (e.g. ciphertexts meant to look random).\npub fn encode_fields_as_bytes<let N: u32>(fields: [Field; N]) -> [u8; 32 * N] {\n let mut bytes = [0; 32 * N];\n for i in 0..N {\n let chunk: [u8; 32] = fields[i].to_be_bytes();\n for j in 0..32 {\n bytes[i * 32 + j] = chunk[j];\n }\n }\n bytes\n}\n\n/// Decodes bytes back into fields.\n///\n/// Panics if the input length is not a multiple of 32 or if any chunk exceeds the BN254 field modulus. See\n/// [`try_decode_fields_from_bytes`](crate::utils::conversion::fields_as_bytes::try_decode_fields_from_bytes)\n/// for a non-panicking variant.\npub fn decode_fields_from_bytes<let N: u32>(bytes: BoundedVec<u8, N>) -> BoundedVec<Field, N / 32> {\n assert(bytes.len() % 32 == 0, \"Input length must be a multiple of 32\");\n try_decode_fields_from_bytes(bytes).expect(f\"Value does not fit in field\")\n}\n\n/// Decodes bytes back into fields, returning None on failure.\n///\n/// Inverse of\n/// [`encode_fields_as_bytes`](crate::utils::conversion::fields_as_bytes::encode_fields_as_bytes).\n/// Returns `Option::none()` if the input length is not a multiple of 32, or if any 32-byte chunk is `>=` the BN254\n/// field modulus.\npub fn try_decode_fields_from_bytes<let N: u32>(bytes: BoundedVec<u8, N>) -> Option<BoundedVec<Field, N / 32>> {\n if bytes.len() % 32 == 0 {\n let num_chunks = bytes.len() / 32;\n let mut fields: BoundedVec<Field, N / 32> = BoundedVec::new();\n for i in 0..num_chunks {\n let maybe_field = try_decode_field_from_bytes(bytes, i * 32);\n if maybe_field.is_some() {\n fields.push(maybe_field.unwrap());\n }\n }\n if fields.len() == num_chunks {\n Option::some(fields)\n } else {\n Option::none()\n }\n } else {\n Option::none()\n }\n}\n\nfn try_decode_field_from_bytes<let N: u32>(bytes: BoundedVec<u8, N>, offset: u32) -> Option<Field> {\n // Field arithmetic silently wraps values >= the modulus, so we compare each chunk against the modulus\n // byte-by-byte (big-endian) while building `field`. cmp: 0 = equal so far, 1 = less than modulus, 2 = exceeds.\n let p = std::field::modulus_be_bytes();\n let mut field = 0;\n let mut cmp: u8 = 0;\n for j in 0..32 {\n let byte = bytes.get(offset + j);\n field = field * 256 + byte as Field;\n if cmp == 0 {\n if byte < p[j] {\n cmp = 1;\n } else if byte > p[j] {\n cmp = 2;\n }\n }\n }\n\n if cmp == 1 {\n Option::some(field)\n } else {\n Option::none()\n }\n}\n\nmod tests {\n use crate::utils::array::subarray;\n use super::{decode_fields_from_bytes, encode_fields_as_bytes, try_decode_fields_from_bytes};\n\n #[test]\n unconstrained fn round_trips_fields(input: [Field; 3]) {\n let bytes = encode_fields_as_bytes(input);\n\n // In production the bytes fly through the system and arrive as a BoundedVec on the other end. 113 is an\n // arbitrary max length larger than the input length of 96.\n let bytes_bvec = BoundedVec::<_, 113>::from_array(bytes);\n let fields_back = try_decode_fields_from_bytes(bytes_bvec).unwrap();\n\n assert_eq(fields_back.len(), input.len());\n assert_eq(subarray(fields_back.storage(), 0), input);\n }\n\n #[test]\n unconstrained fn try_decode_returns_none_on_length_not_multiple_of_32() {\n let input = BoundedVec::<_, 64>::from_parts([0 as u8; 64], 33);\n assert(try_decode_fields_from_bytes(input).is_none());\n }\n\n #[test]\n unconstrained fn try_decode_accepts_max_field() {\n // -1 in field arithmetic wraps to `modulus - 1`, the largest valid field value.\n let max_field_as_bytes: [u8; 32] = (-1).to_be_bytes();\n let input = BoundedVec::<_, 32>::from_array(max_field_as_bytes);\n\n let fields = try_decode_fields_from_bytes(input).unwrap();\n\n assert_eq(fields.get(0), -1);\n }\n\n // Verifies the overflow check: take the max allowed value, bump a random byte, feed it in.\n #[test]\n unconstrained fn try_decode_returns_none_on_chunk_above_modulus(random_value: u8) {\n let index_of_byte_to_bump = random_value % 32;\n let max_field_value_as_bytes: [u8; 32] = (-1).to_be_bytes();\n let byte_to_bump = max_field_value_as_bytes[index_of_byte_to_bump as u32];\n\n // Skip if the selected byte is already 255. Acceptable under fuzz testing.\n if byte_to_bump != 255 {\n let mut input = BoundedVec::<_, 32>::from_array(max_field_value_as_bytes);\n input.set(index_of_byte_to_bump as u32, byte_to_bump + 1);\n\n assert(try_decode_fields_from_bytes(input).is_none());\n }\n }\n\n #[test]\n unconstrained fn try_decode_returns_none_on_chunk_equal_to_modulus() {\n // The field modulus itself is not a valid field value (it wraps to 0).\n let p: [u8; 32] = std::field::modulus_be_bytes().as_array();\n let input = BoundedVec::<u8, 32>::from_array(p);\n assert(try_decode_fields_from_bytes(input).is_none());\n }\n\n #[test(should_fail_with = \"Input length must be a multiple of 32\")]\n unconstrained fn decode_asserts_length_multiple_of_32() {\n let input = BoundedVec::<_, 143>::from_array([\n 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,\n 30, 31, 32, 33,\n ]);\n let _fields = decode_fields_from_bytes(input);\n }\n\n #[test(should_fail_with = \"Value does not fit in field\")]\n unconstrained fn decode_panics_on_chunk_above_modulus(random_value: u8) {\n let index_of_byte_to_bump = random_value % 32;\n let max_field_value_as_bytes: [u8; 32] = (-1).to_be_bytes();\n let byte_to_bump = max_field_value_as_bytes[index_of_byte_to_bump as u32];\n\n if byte_to_bump != 255 {\n let mut input = BoundedVec::<_, 32>::from_array(max_field_value_as_bytes);\n input.set(index_of_byte_to_bump as u32, byte_to_bump + 1);\n let _fields = decode_fields_from_bytes(input);\n }\n }\n}\n"
|
|
1864
1864
|
},
|
|
1865
|
-
"
|
|
1865
|
+
"258": {
|
|
1866
1866
|
"function_locations": [
|
|
1867
1867
|
{
|
|
1868
1868
|
"name": "get_sign_of_point",
|
|
@@ -1912,7 +1912,7 @@
|
|
|
1912
1912
|
"path": "/home/aztec-dev/aztec-packages/noir-projects/aztec-nr/aztec/src/utils/point.nr",
|
|
1913
1913
|
"source": "use crate::protocol::{point::Point, utils::field::sqrt};\n\n// I am storing the modulus minus 1 divided by 2 here because full modulus would throw \"String literal too large\" error\n// Full modulus is 21888242871839275222246405745257275088548364400416034343698204186575808495617\nglobal BN254_FR_MODULUS_DIV_2: Field = 10944121435919637611123202872628637544274182200208017171849102093287904247808;\n\n/// Returns: true if p.y <= MOD_DIV_2, else false.\npub fn get_sign_of_point(p: Point) -> bool {\n // We store only a \"sign\" of the y coordinate because the rest can be derived from the x coordinate. To get the\n // sign we check if the y coordinate is less or equal than the field's modulus minus 1 divided by 2. Ideally we'd\n // do `y <= MOD_DIV_2`, but there's no `lte` function, so instead we do `!(y > MOD_DIV_2)`, which is equivalent,\n // and then rewrite that as `!(MOD_DIV_2 < y)`, since we also have no `gt` function.\n !BN254_FR_MODULUS_DIV_2.lt(p.y)\n}\n\n/// Returns a `Point` in the Grumpkin curve given its x coordinate.\n///\n/// Because not all values in the field are valid x coordinates of points in the curve (i.e. there is no corresponding\n/// y value in the field that satisfies the curve equation), it may not be possible to reconstruct a `Point`.\n/// `Option::none()` is returned in such cases.\npub fn point_from_x_coord(x: Field) -> Option<Point> {\n // y ^ 2 = x ^ 3 - 17\n let rhs = x * x * x - 17;\n sqrt(rhs).map(|y| Point { x, y, is_infinite: false })\n}\n\n/// Returns a `Point` in the Grumpkin curve given its x coordinate and sign for the y coordinate.\n///\n/// Because not all values in the field are valid x coordinates of points in the curve (i.e. there is no corresponding\n/// y value in the field that satisfies the curve equation), it may not be possible to reconstruct a `Point`.\n/// `Option::none()` is returned in such cases.\n///\n/// @param x - The x coordinate of the point @param sign - The \"sign\" of the y coordinate - determines whether y <=\n/// (Fr.MODULUS - 1) / 2\npub fn point_from_x_coord_and_sign(x: Field, sign: bool) -> Option<Point> {\n // y ^ 2 = x ^ 3 - 17\n let rhs = x * x * x - 17;\n\n sqrt(rhs).map(|y| {\n // If there is a square root, we need to ensure it has the correct \"sign\"\n let y_is_positive = !BN254_FR_MODULUS_DIV_2.lt(y);\n let final_y = if y_is_positive == sign { y } else { -y };\n Point { x, y: final_y, is_infinite: false }\n })\n}\n\nmod test {\n use crate::protocol::point::Point;\n use crate::utils::point::{\n BN254_FR_MODULUS_DIV_2, get_sign_of_point, point_from_x_coord, point_from_x_coord_and_sign,\n };\n\n #[test]\n unconstrained fn test_point_from_x_coord_and_sign() {\n // Test positive y coordinate\n let x = 0x1af41f5de96446dc3776a1eb2d98bb956b7acd9979a67854bec6fa7c2973bd73;\n let sign = true;\n let p = point_from_x_coord_and_sign(x, sign).unwrap();\n\n assert_eq(p.x, x);\n assert_eq(p.y, 0x07fc22c7f2c7057571f137fe46ea9c95114282bc95d37d71ec4bfb88de457d4a);\n assert_eq(p.is_infinite, false);\n\n // Test negative y coordinate\n let x2 = 0x247371652e55dd74c9af8dbe9fb44931ba29a9229994384bd7077796c14ee2b5;\n let sign2 = false;\n let p2 = point_from_x_coord_and_sign(x2, sign2).unwrap();\n\n assert_eq(p2.x, x2);\n assert_eq(p2.y, 0x26441aec112e1ae4cee374f42556932001507ad46e255ffb27369c7e3766e5c0);\n assert_eq(p2.is_infinite, false);\n }\n\n #[test]\n unconstrained fn test_point_from_x_coord_valid() {\n // x = 8 is a known quadratic residue - should give a valid point\n let result = point_from_x_coord(Field::from(8));\n assert(result.is_some());\n\n let point = result.unwrap();\n assert_eq(point.x, Field::from(8));\n // Check curve equation y^2 = x^3 - 17\n assert_eq(point.y * point.y, point.x * point.x * point.x - 17);\n }\n\n #[test]\n unconstrained fn test_point_from_x_coord_invalid() {\n // x = 3 is a non-residue for this curve - should give None\n let x = Field::from(3);\n let maybe_point = point_from_x_coord(x);\n assert(maybe_point.is_none());\n }\n\n #[test]\n unconstrained fn test_both_roots_satisfy_curve() {\n // Derive a point from x = 8 (known to be valid from test_point_from_x_coord_valid)\n let x: Field = 8;\n let point = point_from_x_coord(x).unwrap();\n\n // Check y satisfies curve equation\n assert_eq(point.y * point.y, x * x * x - 17);\n\n // Check -y also satisfies curve equation\n let neg_y = 0 - point.y;\n assert_eq(neg_y * neg_y, x * x * x - 17);\n\n // Verify they are different (unless y = 0)\n assert(point.y != neg_y);\n }\n\n #[test]\n unconstrained fn test_point_from_x_coord_and_sign_invalid() {\n // x = 3 has no valid point on the curve (from test_point_from_x_coord_invalid)\n let x = Field::from(3);\n let result_positive = point_from_x_coord_and_sign(x, true);\n let result_negative = point_from_x_coord_and_sign(x, false);\n\n assert(result_positive.is_none());\n assert(result_negative.is_none());\n }\n\n #[test]\n unconstrained fn test_get_sign_of_point() {\n // Derive a point from x = 8, then test both possible y values\n let point = point_from_x_coord(8).unwrap();\n let neg_point = Point { x: point.x, y: 0 - point.y, is_infinite: false };\n\n // One should be \"positive\" (y <= MOD_DIV_2) and one \"negative\"\n let sign1 = get_sign_of_point(point);\n let sign2 = get_sign_of_point(neg_point);\n assert(sign1 != sign2);\n\n // y = 0 should return true (0 <= MOD_DIV_2)\n let zero_y_point = Point { x: 0, y: 0, is_infinite: false };\n assert(get_sign_of_point(zero_y_point) == true);\n\n // y = MOD_DIV_2 should return true (exactly at boundary)\n let boundary_point = Point { x: 0, y: BN254_FR_MODULUS_DIV_2, is_infinite: false };\n assert(get_sign_of_point(boundary_point) == true);\n\n // y = MOD_DIV_2 + 1 should return false (just over boundary)\n let over_boundary_point = Point { x: 0, y: BN254_FR_MODULUS_DIV_2 + 1, is_infinite: false };\n assert(get_sign_of_point(over_boundary_point) == false);\n }\n\n #[test]\n unconstrained fn test_point_from_x_coord_zero() {\n // x = 0: y^2 = 0^3 - 17 = -17, which is not a quadratic residue in BN254 scalar field\n let result = point_from_x_coord(0);\n assert(result.is_none());\n }\n\n #[test]\n unconstrained fn test_bn254_fr_modulus_div_2() {\n // Verify that BN254_FR_MODULUS_DIV_2 == (p - 1) / 2 This means: 2 * BN254_FR_MODULUS_DIV_2 + 1 == p == 0 (in\n // the field)\n assert_eq(2 * BN254_FR_MODULUS_DIV_2 + 1, 0);\n }\n\n}\n"
|
|
1914
1914
|
},
|
|
1915
|
-
"
|
|
1915
|
+
"268": {
|
|
1916
1916
|
"function_locations": [
|
|
1917
1917
|
{
|
|
1918
1918
|
"name": "Poseidon2::hash",
|
|
@@ -1954,7 +1954,7 @@
|
|
|
1954
1954
|
"path": "/home/aztec-dev/nargo/github.com/noir-lang/poseidon/v0.3.0/src/poseidon2.nr",
|
|
1955
1955
|
"source": "use std::default::Default;\nuse std::hash::Hasher;\n\nglobal RATE: u32 = 3;\n\npub struct Poseidon2 {\n cache: [Field; 3],\n state: [Field; 4],\n cache_size: u32,\n squeeze_mode: bool, // 0 => absorb, 1 => squeeze\n}\n\nimpl Poseidon2 {\n #[no_predicates]\n pub fn hash<let N: u32>(input: [Field; N], message_size: u32) -> Field {\n Poseidon2::hash_internal(input, message_size)\n }\n\n pub(crate) fn new(iv: Field) -> Poseidon2 {\n let mut result =\n Poseidon2 { cache: [0; 3], state: [0; 4], cache_size: 0, squeeze_mode: false };\n result.state[RATE] = iv;\n result\n }\n\n fn perform_duplex(&mut self) {\n // add the cache into sponge state\n self.state[0] += self.cache[0];\n self.state[1] += self.cache[1];\n self.state[2] += self.cache[2];\n self.state = crate::poseidon2_permutation(self.state);\n }\n\n fn absorb(&mut self, input: Field) {\n assert(!self.squeeze_mode);\n if self.cache_size == RATE {\n // If we're absorbing, and the cache is full, apply the sponge permutation to compress the cache\n self.perform_duplex();\n self.cache[0] = input;\n self.cache_size = 1;\n } else {\n // If we're absorbing, and the cache is not full, add the input into the cache\n self.cache[self.cache_size] = input;\n self.cache_size += 1;\n }\n }\n\n fn squeeze(&mut self) -> Field {\n assert(!self.squeeze_mode);\n // If we're in absorb mode, apply sponge permutation to compress the cache.\n self.perform_duplex();\n self.squeeze_mode = true;\n\n // Pop one item off the top of the permutation and return it.\n self.state[0]\n }\n\n fn hash_internal<let N: u32>(input: [Field; N], in_len: u32) -> Field {\n let two_pow_64 = 18446744073709551616;\n let iv: Field = (in_len as Field) * two_pow_64;\n let mut state = [0; 4];\n state[RATE] = iv;\n\n if std::runtime::is_unconstrained() {\n for i in 0..(in_len / RATE) {\n state[0] += input[i * RATE];\n state[1] += input[i * RATE + 1];\n state[2] += input[i * RATE + 2];\n state = crate::poseidon2_permutation(state);\n }\n\n // handle remaining elements after last full RATE-sized chunk\n let num_extra_fields = in_len % RATE;\n if num_extra_fields != 0 {\n let remainder_start = in_len - num_extra_fields;\n state[0] += input[remainder_start];\n if num_extra_fields > 1 {\n state[1] += input[remainder_start + 1];\n }\n }\n } else {\n let mut states: [[Field; 4]; N / RATE + 1] = [[0; 4]; N / RATE + 1];\n states[0] = state;\n\n // process all full RATE-sized chunks, storing state after each permutation\n for chunk_idx in 0..(N / RATE) {\n for i in 0..RATE {\n state[i] += input[chunk_idx * RATE + i];\n }\n state = crate::poseidon2_permutation(state);\n states[chunk_idx + 1] = state;\n }\n\n // get state at the last full block before in_len\n let first_partially_filled_chunk = in_len / RATE;\n state = states[first_partially_filled_chunk];\n\n // handle remaining elements after last full RATE-sized chunk\n let remainder_start = (in_len / RATE) * RATE;\n for j in 0..RATE {\n let idx = remainder_start + j;\n if idx < in_len {\n state[j] += input[idx];\n }\n }\n }\n\n // always run final permutation unless we just completed a full chunk\n // still need to permute once if in_len is 0\n if (in_len == 0) | (in_len % RATE != 0) {\n state = crate::poseidon2_permutation(state);\n };\n\n state[0]\n }\n}\n\npub struct Poseidon2Hasher {\n _state: [Field],\n}\n\nimpl Hasher for Poseidon2Hasher {\n fn finish(self) -> Field {\n let iv: Field = (self._state.len() as Field) * 18446744073709551616; // iv = (self._state.len() << 64)\n let mut sponge = Poseidon2::new(iv);\n for i in 0..self._state.len() {\n sponge.absorb(self._state[i]);\n }\n sponge.squeeze()\n }\n\n fn write(&mut self, input: Field) {\n self._state = self._state.push_back(input);\n }\n}\n\nimpl Default for Poseidon2Hasher {\n fn default() -> Self {\n Poseidon2Hasher { _state: @[] }\n }\n}\n"
|
|
1956
1956
|
},
|
|
1957
|
-
"
|
|
1957
|
+
"279": {
|
|
1958
1958
|
"function_locations": [
|
|
1959
1959
|
{
|
|
1960
1960
|
"name": "BlockHeader::chain_id",
|
|
@@ -2138,7 +2138,7 @@
|
|
|
2138
2138
|
"path": "std/array/mod.nr",
|
|
2139
2139
|
"source": "use crate::cmp::{Eq, Ord};\nuse crate::convert::From;\nuse crate::runtime::is_unconstrained;\n\nmod check_shuffle;\nmod quicksort;\n\nimpl<T, let N: u32> [T; N] {\n /// Returns the length of this array.\n ///\n /// ```noir\n /// fn len(self) -> Field\n /// ```\n ///\n /// example\n ///\n /// ```noir\n /// fn main() {\n /// let array = [42, 42];\n /// assert(array.len() == 2);\n /// }\n /// ```\n #[builtin(array_len)]\n pub fn len(self) -> u32 {}\n\n /// Returns this array as a vector.\n ///\n /// ```noir\n /// let array = [1, 2];\n /// let vector = array.as_vector();\n /// assert_eq(vector, [1, 2].as_vector());\n /// ```\n #[builtin(as_vector)]\n pub fn as_vector(self) -> [T] {}\n\n /// Returns this array as a vector.\n /// This method is deprecated in favor of `as_vector`.\n ///\n /// ```noir\n /// let array = [1, 2];\n /// let vector = array.as_slice();\n /// assert_eq(vector, [1, 2].as_vector());\n /// ```\n #[builtin(as_vector)]\n #[deprecated(\"This method has been renamed to `as_vector`\")]\n pub fn as_slice(self) -> [T] {}\n\n /// Applies a function to each element of this array, returning a new array containing the mapped elements.\n ///\n /// Example:\n ///\n /// ```rust\n /// let a = [1, 2, 3];\n /// let b = a.map(|a| a * 2);\n /// assert_eq(b, [2, 4, 6]);\n /// ```\n pub fn map<U, Env>(&self, f: fn[Env](T) -> U) -> [U; N] {\n let uninitialized = crate::mem::zeroed();\n let mut ret = [uninitialized; N];\n\n for i in 0..self.len() {\n ret[i] = f(self[i]);\n }\n\n ret\n }\n\n /// Applies a function to each element of this array along with its index,\n /// returning a new array containing the mapped elements.\n ///\n /// Example:\n ///\n /// ```rust\n /// let a = [1, 2, 3];\n /// let b = a.mapi(|i, a| i + a * 2);\n /// assert_eq(b, [2, 5, 8]);\n /// ```\n pub fn mapi<U, Env>(&self, f: fn[Env](u32, T) -> U) -> [U; N] {\n let uninitialized = crate::mem::zeroed();\n let mut ret = [uninitialized; N];\n\n for i in 0..self.len() {\n ret[i] = f(i, self[i]);\n }\n\n ret\n }\n\n /// Applies a function to each element of this array.\n ///\n /// Example:\n ///\n /// ```rust\n /// let a = [1, 2, 3];\n /// let mut b = [0; 3];\n /// let mut i = 0;\n /// a.for_each(|x| {\n /// b[i] = x;\n /// i += 1;\n /// });\n /// assert_eq(a, b);\n /// ```\n pub fn for_each<Env>(&self, f: fn[Env](T) -> ()) {\n for i in 0..self.len() {\n f(self[i]);\n }\n }\n\n /// Applies a function to each element of this array along with its index.\n ///\n /// Example:\n ///\n /// ```rust\n /// let a = [1, 2, 3];\n /// let mut b = [0; 3];\n /// a.for_eachi(|i, x| {\n /// b[i] = x;\n /// });\n /// assert_eq(a, b);\n /// ```\n pub fn for_eachi<Env>(&self, f: fn[Env](u32, T) -> ()) {\n for i in 0..self.len() {\n f(i, self[i]);\n }\n }\n\n /// Applies a function to each element of the array, returning the final accumulated value. The first\n /// parameter is the initial value.\n ///\n /// This is a left fold, so the given function will be applied to the accumulator and first element of\n /// the array, then the second, and so on. For a given call the expected result would be equivalent to:\n ///\n /// ```rust\n /// let a1 = [1];\n /// let a2 = [1, 2];\n /// let a3 = [1, 2, 3];\n ///\n /// let f = |a, b| a - b;\n /// a1.fold(10, f); //=> f(10, 1)\n /// a2.fold(10, f); //=> f(f(10, 1), 2)\n /// a3.fold(10, f); //=> f(f(f(10, 1), 2), 3)\n ///\n /// assert_eq(a3.fold(10, f), 10 - 1 - 2 - 3);\n /// ```\n pub fn fold<U, Env>(&self, mut accumulator: U, f: fn[Env](U, T) -> U) -> U {\n for elem in self {\n accumulator = f(accumulator, elem);\n }\n accumulator\n }\n\n /// Same as fold, but uses the first element as the starting element.\n ///\n /// Requires the input array to be non-empty.\n ///\n /// Example:\n ///\n /// ```noir\n /// fn main() {\n /// let arr = [1, 2, 3, 4];\n /// let reduced = arr.reduce(|a, b| a + b);\n /// assert(reduced == 10);\n /// }\n /// ```\n pub fn reduce<Env>(&self, f: fn[Env](T, T) -> T) -> T {\n let mut accumulator = self[0];\n for i in 1..self.len() {\n accumulator = f(accumulator, self[i]);\n }\n accumulator\n }\n\n /// Returns true if all the elements in this array satisfy the given predicate.\n ///\n /// Example:\n ///\n /// ```noir\n /// fn main() {\n /// let arr = [2, 2, 2, 2, 2];\n /// let all = arr.all(|a| a == 2);\n /// assert(all);\n /// }\n /// ```\n pub fn all<Env>(&self, predicate: fn[Env](T) -> bool) -> bool {\n let mut ret = true;\n for elem in self {\n ret &= predicate(elem);\n }\n ret\n }\n\n /// Returns true if any of the elements in this array satisfy the given predicate.\n ///\n /// Example:\n ///\n /// ```noir\n /// fn main() {\n /// let arr = [2, 2, 2, 2, 5];\n /// let any = arr.any(|a| a == 5);\n /// assert(any);\n /// }\n /// ```\n pub fn any<Env>(&self, predicate: fn[Env](T) -> bool) -> bool {\n let mut ret = false;\n for elem in self {\n ret |= predicate(elem);\n }\n ret\n }\n\n /// Concatenates this array with another array.\n ///\n /// Example:\n ///\n /// ```noir\n /// fn main() {\n /// let arr1 = [1, 2, 3, 4];\n /// let arr2 = [6, 7, 8, 9, 10, 11];\n /// let concatenated_arr = arr1.concat(arr2);\n /// assert(concatenated_arr == [1, 2, 3, 4, 6, 7, 8, 9, 10, 11]);\n /// }\n /// ```\n pub fn concat<let M: u32>(&self, array2: [T; M]) -> [T; N + M] {\n let mut result = [crate::mem::zeroed(); N + M];\n for i in 0..N {\n result[i] = self[i];\n }\n for i in 0..M {\n result[i + N] = array2[i];\n }\n result\n }\n}\n\nimpl<T, let N: u32> [T; N]\nwhere\n T: Ord + Eq,\n{\n /// Returns a new sorted array. The original array remains untouched. Notice that this function will\n /// only work for arrays of fields or integers, not for any arbitrary type. This is because the sorting\n /// logic it uses internally is optimized specifically for these values. If you need a sort function to\n /// sort any type, you should use the [`Self::sort_via`] function.\n ///\n /// Example:\n ///\n /// ```rust\n /// fn main() {\n /// let arr = [42, 32];\n /// let sorted = arr.sort();\n /// assert(sorted == [32, 42]);\n /// }\n /// ```\n pub fn sort(&self) -> Self {\n self.sort_via(|a, b| a <= b)\n }\n}\n\nimpl<T, let N: u32> [T; N]\nwhere\n T: Eq,\n{\n /// Returns a new sorted array by sorting it with a custom comparison function.\n /// The original array remains untouched.\n /// The ordering function must return true if the first argument should be sorted to be before the second argument or is equal to the second argument.\n ///\n /// Using this method with an operator like `<` that does not return `true` for equal values will result in an assertion failure for arrays with equal elements.\n ///\n /// Example:\n ///\n /// ```rust\n /// fn main() {\n /// let arr = [42, 32]\n /// let sorted_ascending = arr.sort_via(|a, b| a <= b);\n /// assert(sorted_ascending == [32, 42]); // verifies\n ///\n /// let sorted_descending = arr.sort_via(|a, b| a >= b);\n /// assert(sorted_descending == [32, 42]); // does not verify\n /// }\n /// ```\n pub fn sort_via<Env>(&self, ordering: fn[Env](T, T) -> bool) -> Self {\n // Safety: `sorted` array is checked to be:\n // a. a permutation of `input`'s elements\n // b. satisfying the predicate `ordering`\n let sorted = unsafe { quicksort::quicksort(self, ordering) };\n\n if !is_unconstrained() {\n for i in 0..N - 1 {\n assert(\n ordering(sorted[i], sorted[i + 1]),\n \"Array has not been sorted correctly according to `ordering`.\",\n );\n }\n check_shuffle::check_shuffle(self, &sorted);\n }\n sorted\n }\n}\n\nimpl<let N: u32> [u8; N] {\n /// Converts a byte array of type `[u8; N]` to a string. Note that this performs no UTF-8 validation -\n /// the given array is interpreted as-is as a string.\n ///\n /// Example:\n ///\n /// ```rust\n /// fn main() {\n /// let hi = [104, 105].as_str_unchecked();\n /// assert_eq(hi, \"hi\");\n /// }\n /// ```\n #[builtin(array_as_str_unchecked)]\n pub fn as_str_unchecked(self) -> str<N> {}\n}\n\nimpl<let N: u32> From<str<N>> for [u8; N] {\n /// Returns an array of the string bytes.\n fn from(s: str<N>) -> Self {\n s.as_bytes()\n }\n}\n\nmod test {\n #[test]\n fn map_empty() {\n assert_eq([].map(|x| x + 1), []);\n }\n\n global arr_with_100_values: [u32; 100] = [\n 42, 123, 87, 93, 48, 80, 50, 5, 104, 84, 70, 47, 119, 66, 71, 121, 3, 29, 42, 118, 2, 54,\n 89, 44, 81, 0, 26, 106, 68, 96, 84, 48, 95, 54, 45, 32, 89, 100, 109, 19, 37, 41, 19, 98,\n 53, 114, 107, 66, 6, 74, 13, 19, 105, 64, 123, 28, 44, 50, 89, 58, 123, 126, 21, 43, 86, 35,\n 21, 62, 82, 0, 108, 120, 72, 72, 62, 80, 12, 71, 70, 86, 116, 73, 38, 15, 127, 81, 30, 8,\n 125, 28, 26, 69, 114, 63, 27, 28, 61, 42, 13, 32,\n ];\n global expected_with_100_values: [u32; 100] = [\n 0, 0, 2, 3, 5, 6, 8, 12, 13, 13, 15, 19, 19, 19, 21, 21, 26, 26, 27, 28, 28, 28, 29, 30, 32,\n 32, 35, 37, 38, 41, 42, 42, 42, 43, 44, 44, 45, 47, 48, 48, 50, 50, 53, 54, 54, 58, 61, 62,\n 62, 63, 64, 66, 66, 68, 69, 70, 70, 71, 71, 72, 72, 73, 74, 80, 80, 81, 81, 82, 84, 84, 86,\n 86, 87, 89, 89, 89, 93, 95, 96, 98, 100, 104, 105, 106, 107, 108, 109, 114, 114, 116, 118,\n 119, 120, 121, 123, 123, 123, 125, 126, 127,\n ];\n fn sort_u32(a: u32, b: u32) -> bool {\n a <= b\n }\n\n #[test]\n fn test_sort() {\n let arr: [u32; 7] = [3, 6, 8, 10, 1, 2, 1];\n\n let sorted = arr.sort();\n\n let expected: [u32; 7] = [1, 1, 2, 3, 6, 8, 10];\n assert(sorted == expected);\n }\n\n #[test]\n fn test_sort_100_values() {\n let arr: [u32; 100] = [\n 42, 123, 87, 93, 48, 80, 50, 5, 104, 84, 70, 47, 119, 66, 71, 121, 3, 29, 42, 118, 2,\n 54, 89, 44, 81, 0, 26, 106, 68, 96, 84, 48, 95, 54, 45, 32, 89, 100, 109, 19, 37, 41,\n 19, 98, 53, 114, 107, 66, 6, 74, 13, 19, 105, 64, 123, 28, 44, 50, 89, 58, 123, 126, 21,\n 43, 86, 35, 21, 62, 82, 0, 108, 120, 72, 72, 62, 80, 12, 71, 70, 86, 116, 73, 38, 15,\n 127, 81, 30, 8, 125, 28, 26, 69, 114, 63, 27, 28, 61, 42, 13, 32,\n ];\n\n let sorted = arr.sort();\n\n let expected: [u32; 100] = [\n 0, 0, 2, 3, 5, 6, 8, 12, 13, 13, 15, 19, 19, 19, 21, 21, 26, 26, 27, 28, 28, 28, 29, 30,\n 32, 32, 35, 37, 38, 41, 42, 42, 42, 43, 44, 44, 45, 47, 48, 48, 50, 50, 53, 54, 54, 58,\n 61, 62, 62, 63, 64, 66, 66, 68, 69, 70, 70, 71, 71, 72, 72, 73, 74, 80, 80, 81, 81, 82,\n 84, 84, 86, 86, 87, 89, 89, 89, 93, 95, 96, 98, 100, 104, 105, 106, 107, 108, 109, 114,\n 114, 116, 118, 119, 120, 121, 123, 123, 123, 125, 126, 127,\n ];\n assert(sorted == expected);\n }\n\n #[test]\n fn test_sort_100_values_comptime() {\n let sorted = arr_with_100_values.sort();\n assert(sorted == expected_with_100_values);\n }\n\n #[test]\n fn test_sort_via() {\n let arr: [u32; 7] = [3, 6, 8, 10, 1, 2, 1];\n\n let sorted = arr.sort_via(sort_u32);\n\n let expected: [u32; 7] = [1, 1, 2, 3, 6, 8, 10];\n assert(sorted == expected);\n }\n\n #[test]\n fn test_sort_via_100_values() {\n let arr: [u32; 100] = [\n 42, 123, 87, 93, 48, 80, 50, 5, 104, 84, 70, 47, 119, 66, 71, 121, 3, 29, 42, 118, 2,\n 54, 89, 44, 81, 0, 26, 106, 68, 96, 84, 48, 95, 54, 45, 32, 89, 100, 109, 19, 37, 41,\n 19, 98, 53, 114, 107, 66, 6, 74, 13, 19, 105, 64, 123, 28, 44, 50, 89, 58, 123, 126, 21,\n 43, 86, 35, 21, 62, 82, 0, 108, 120, 72, 72, 62, 80, 12, 71, 70, 86, 116, 73, 38, 15,\n 127, 81, 30, 8, 125, 28, 26, 69, 114, 63, 27, 28, 61, 42, 13, 32,\n ];\n\n let sorted = arr.sort_via(sort_u32);\n\n let expected: [u32; 100] = [\n 0, 0, 2, 3, 5, 6, 8, 12, 13, 13, 15, 19, 19, 19, 21, 21, 26, 26, 27, 28, 28, 28, 29, 30,\n 32, 32, 35, 37, 38, 41, 42, 42, 42, 43, 44, 44, 45, 47, 48, 48, 50, 50, 53, 54, 54, 58,\n 61, 62, 62, 63, 64, 66, 66, 68, 69, 70, 70, 71, 71, 72, 72, 73, 74, 80, 80, 81, 81, 82,\n 84, 84, 86, 86, 87, 89, 89, 89, 93, 95, 96, 98, 100, 104, 105, 106, 107, 108, 109, 114,\n 114, 116, 118, 119, 120, 121, 123, 123, 123, 125, 126, 127,\n ];\n assert(sorted == expected);\n }\n\n #[test]\n fn mapi_empty() {\n assert_eq([].mapi(|i, x| i * x + 1), []);\n }\n\n #[test]\n fn for_each_empty() {\n let empty_array: [Field; 0] = [];\n empty_array.for_each(|_x| assert(false));\n }\n\n #[test]\n fn for_eachi_empty() {\n let empty_array: [Field; 0] = [];\n empty_array.for_eachi(|_i, _x| assert(false));\n }\n\n #[test]\n fn map_example() {\n let a = [1, 2, 3];\n let b = a.map(|a| a * 2);\n assert_eq(b, [2, 4, 6]);\n }\n\n #[test]\n fn mapi_example() {\n let a = [1, 2, 3];\n let b = a.mapi(|i, a| i + a * 2);\n assert_eq(b, [2, 5, 8]);\n }\n\n #[test]\n fn for_each_example() {\n let a = [1, 2, 3];\n let mut b = [0, 0, 0];\n let b_ref = &mut b;\n let mut i = 0;\n let i_ref = &mut i;\n a.for_each(|x| {\n b_ref[*i_ref] = x * 2;\n *i_ref += 1;\n });\n assert_eq(b, [2, 4, 6]);\n assert_eq(i, 3);\n }\n\n #[test]\n fn for_eachi_example() {\n let a = [1, 2, 3];\n let mut b = [0, 0, 0];\n let b_ref = &mut b;\n a.for_eachi(|i, a| { b_ref[i] = i + a * 2; });\n assert_eq(b, [2, 5, 8]);\n }\n\n #[test]\n fn concat() {\n let arr1 = [1, 2, 3, 4];\n let arr2 = [6, 7, 8, 9, 10, 11];\n let concatenated_arr = arr1.concat(arr2);\n assert_eq(concatenated_arr, [1, 2, 3, 4, 6, 7, 8, 9, 10, 11]);\n }\n\n #[test]\n fn concat_zero_length_with_something() {\n let arr1 = [];\n let arr2 = [1];\n let concatenated_arr = arr1.concat(arr2);\n assert_eq(concatenated_arr, [1]);\n }\n\n #[test]\n fn concat_something_with_zero_length() {\n let arr1 = [1];\n let arr2 = [];\n let concatenated_arr = arr1.concat(arr2);\n assert_eq(concatenated_arr, [1]);\n }\n\n #[test]\n fn concat_zero_lengths() {\n let arr1: [Field; 0] = [];\n let arr2: [Field; 0] = [];\n let concatenated_arr = arr1.concat(arr2);\n assert_eq(concatenated_arr, []);\n }\n}\n"
|
|
2140
2140
|
},
|
|
2141
|
-
"
|
|
2141
|
+
"357": {
|
|
2142
2142
|
"function_locations": [
|
|
2143
2143
|
{
|
|
2144
2144
|
"name": "sha256_to_field",
|
|
@@ -2260,7 +2260,7 @@
|
|
|
2260
2260
|
"path": "/home/aztec-dev/aztec-packages/noir-projects/noir-protocol-circuits/crates/types/src/hash.nr",
|
|
2261
2261
|
"source": "mod poseidon2_chunks;\n\nuse crate::{\n abis::{\n contract_class_function_leaf_preimage::ContractClassFunctionLeafPreimage,\n function_selector::FunctionSelector, nullifier::Nullifier, private_log::PrivateLog,\n transaction::tx_request::TxRequest,\n },\n address::{AztecAddress, EthAddress},\n constants::{\n CONTRACT_CLASS_LOG_SIZE_IN_FIELDS, DOM_SEP__NOTE_HASH_NONCE,\n DOM_SEP__PRIVATE_LOG_FIRST_FIELD, DOM_SEP__SILOED_NOTE_HASH, DOM_SEP__SILOED_NULLIFIER,\n DOM_SEP__UNIQUE_NOTE_HASH, FUNCTION_TREE_HEIGHT, NULL_MSG_SENDER_CONTRACT_ADDRESS,\n TWO_POW_64,\n },\n merkle_tree::root_from_sibling_path,\n messaging::l2_to_l1_message::L2ToL1Message,\n poseidon2::Poseidon2Sponge,\n side_effect::{Counted, Scoped},\n traits::{FromField, Hash, ToField},\n utils::field::{field_from_bytes, field_from_bytes_32_trunc},\n};\n\npub use poseidon2_chunks::poseidon2_absorb_in_chunks_existing_sponge;\nuse poseidon2_chunks::poseidon2_absorb_in_chunks;\nuse std::embedded_curve_ops::EmbeddedCurveScalar;\n\n// TODO: refactor these into their own files: sha256, poseidon2, some protocol-specific hash computations, some merkle computations.\n\npub fn sha256_to_field<let N: u32>(bytes_to_hash: [u8; N]) -> Field {\n let sha256_hashed = sha256::digest(bytes_to_hash);\n let hash_in_a_field = field_from_bytes_32_trunc(sha256_hashed);\n\n hash_in_a_field\n}\n\npub fn private_functions_root_from_siblings(\n selector: FunctionSelector,\n vk_hash: Field,\n function_leaf_index: Field,\n function_leaf_sibling_path: [Field; FUNCTION_TREE_HEIGHT],\n) -> Field {\n let function_leaf_preimage = ContractClassFunctionLeafPreimage { selector, vk_hash };\n let function_leaf = function_leaf_preimage.hash();\n root_from_sibling_path(\n function_leaf,\n function_leaf_index,\n function_leaf_sibling_path,\n )\n}\n\n/// Siloing in the context of Aztec refers to the process of hashing a note hash with a contract address (this way\n/// the note hash is scoped to a specific contract). This is used to prevent intermingling of notes between contracts.\npub fn compute_siloed_note_hash(contract_address: AztecAddress, note_hash: Field) -> Field {\n poseidon2_hash_with_separator(\n [contract_address.to_field(), note_hash],\n DOM_SEP__SILOED_NOTE_HASH,\n )\n}\n\n/// Computes unique, siloed note hashes from siloed note hashes.\n///\n/// The protocol injects uniqueness into every note_hash, so that every single note_hash in the\n/// tree is unique. This prevents faerie gold attacks, where a malicious sender could create\n/// two identical note_hashes for a recipient (meaning only one would be nullifiable in future).\n///\n/// Most privacy protocols will inject the note's leaf_index (its position in the Note Hashes Tree)\n/// into the note, but this requires the creator of a note to wait until their tx is included in\n/// a block to know the note's final note hash (the unique, siloed note hash), because inserting\n/// leaves into trees is the job of a block producer.\n///\n/// We took a different approach so that the creator of a note will know each note's unique, siloed\n/// note hash before broadcasting their tx to the network.\n/// (There was also a historical requirement relating to \"chained transactions\" -- a feature that\n/// Aztec Connect had to enable notes to be spent from distinct txs earlier in the same block,\n/// and hence before an archive block root had been established for that block -- but that feature\n/// was abandoned for the Aztec Network for having too many bad tradeoffs).\n///\n/// (\n/// Edit: it is no longer true that all final note_hashes will be known by the creator of a tx\n/// before they send it to the network. If a tx makes public function calls, then _revertible_\n/// note_hashes that are created in private will not be made unique in private by the Reset circuit,\n/// but will instead be made unique by the AVM, because the `note_index_in_tx` will not be known\n/// until the AVM has executed the public functions of the tx. (See an explanation in\n/// reset_output_composer.nr for why).\n/// For some such txs, the `note_index_in_tx` might still be predictable through simulation, but\n/// for txs whose public functions create a varying number of non-revertible notes (determined at\n/// runtime), the `note_index_in_tx` will not be deterministically derivable before submitting the\n/// tx to the network.\n/// )\n///\n/// We use the `first_nullifier` of a tx as a seed of uniqueness. We have a guarantee that there will\n/// always be at least one nullifier per tx, because the init circuit will create one if one isn't\n/// created naturally by any functions of the tx. (Search \"protocol_nullifier\").\n/// We combine the `first_nullifier` with the note's index (its position within this tx's new\n/// note_hashes array) (`note_index_in_tx`) to get a truly unique value to inject into a note, which\n/// we call a `note_nonce`.\npub fn compute_unique_note_hash(note_nonce: Field, siloed_note_hash: Field) -> Field {\n let inputs = [note_nonce, siloed_note_hash];\n poseidon2_hash_with_separator(inputs, DOM_SEP__UNIQUE_NOTE_HASH)\n}\n\npub fn compute_note_hash_nonce(first_nullifier_in_tx: Field, note_index_in_tx: u32) -> Field {\n // Hashing the first nullifier with note index in tx is guaranteed to be unique (because all nullifiers are also\n // unique).\n poseidon2_hash_with_separator(\n [first_nullifier_in_tx, note_index_in_tx as Field],\n DOM_SEP__NOTE_HASH_NONCE,\n )\n}\n\npub fn compute_note_nonce_and_unique_note_hash(\n siloed_note_hash: Field,\n first_nullifier: Field,\n note_index_in_tx: u32,\n) -> Field {\n let note_nonce = compute_note_hash_nonce(first_nullifier, note_index_in_tx);\n compute_unique_note_hash(note_nonce, siloed_note_hash)\n}\n\npub fn compute_siloed_nullifier(contract_address: AztecAddress, nullifier: Field) -> Field {\n poseidon2_hash_with_separator(\n [contract_address.to_field(), nullifier],\n DOM_SEP__SILOED_NULLIFIER,\n )\n}\n\npub fn create_protocol_nullifier(tx_request: TxRequest) -> Scoped<Counted<Nullifier>> {\n // The protocol nullifier is ascribed a special side-effect counter of 1. No other side-effect\n // can have counter 1 (see `validate_as_first_call` for that assertion).\n Nullifier { value: tx_request.hash(), note_hash: 0 }.count(1).scope(\n NULL_MSG_SENDER_CONTRACT_ADDRESS,\n )\n}\n\npub fn compute_log_tag(raw_tag: Field, dom_sep: u32) -> Field {\n poseidon2_hash_with_separator([raw_tag], dom_sep)\n}\n\npub fn compute_siloed_private_log_first_field(\n contract_address: AztecAddress,\n field: Field,\n) -> Field {\n poseidon2_hash_with_separator(\n [contract_address.to_field(), field],\n DOM_SEP__PRIVATE_LOG_FIRST_FIELD,\n )\n}\n\npub fn compute_siloed_private_log(contract_address: AztecAddress, log: PrivateLog) -> PrivateLog {\n let mut fields = log.fields;\n fields[0] = compute_siloed_private_log_first_field(contract_address, fields[0]);\n PrivateLog::new(fields, log.length)\n}\n\npub fn compute_contract_class_log_hash(log: [Field; CONTRACT_CLASS_LOG_SIZE_IN_FIELDS]) -> Field {\n poseidon2_hash(log)\n}\n\npub fn compute_app_siloed_secret_key(\n master_secret_key: EmbeddedCurveScalar,\n app_address: AztecAddress,\n key_type_domain_separator: Field,\n) -> Field {\n poseidon2_hash_with_separator(\n [master_secret_key.hi, master_secret_key.lo, app_address.to_field()],\n key_type_domain_separator,\n )\n}\n\npub fn compute_l2_to_l1_message_hash(\n message: Scoped<L2ToL1Message>,\n rollup_version_id: Field,\n chain_id: Field,\n) -> Field {\n let contract_address_bytes: [u8; 32] = message.contract_address.to_field().to_be_bytes();\n let recipient_bytes: [u8; 20] = message.inner.recipient.to_be_bytes();\n let content_bytes: [u8; 32] = message.inner.content.to_be_bytes();\n let rollup_version_id_bytes: [u8; 32] = rollup_version_id.to_be_bytes();\n let chain_id_bytes: [u8; 32] = chain_id.to_be_bytes();\n\n let mut bytes: [u8; 148] = std::mem::zeroed();\n for i in 0..32 {\n bytes[i] = contract_address_bytes[i];\n bytes[i + 32] = rollup_version_id_bytes[i];\n // 64 - 84 are for recipient.\n bytes[i + 84] = chain_id_bytes[i];\n bytes[i + 116] = content_bytes[i];\n }\n\n for i in 0..20 {\n bytes[64 + i] = recipient_bytes[i];\n }\n\n sha256_to_field(bytes)\n}\n\n// TODO: consider a variant that enables domain separation with a u32 (we seem to have standardised u32s for domain separators)\n/// Computes sha256 hash of 2 input fields.\n///\n/// @returns A truncated field (i.e., the first byte is always 0).\npub fn accumulate_sha256(v0: Field, v1: Field) -> Field {\n // Concatenate two fields into 32 x 2 = 64 bytes\n let v0_as_bytes: [u8; 32] = v0.to_be_bytes();\n let v1_as_bytes: [u8; 32] = v1.to_be_bytes();\n let hash_input_flattened = v0_as_bytes.concat(v1_as_bytes);\n\n sha256_to_field(hash_input_flattened)\n}\n\npub fn poseidon2_hash<let N: u32>(inputs: [Field; N]) -> Field {\n poseidon::poseidon2::Poseidon2::hash(inputs, N)\n}\n\n#[no_predicates]\npub fn poseidon2_hash_with_separator<let N: u32, T>(inputs: [Field; N], separator: T) -> Field\nwhere\n T: ToField,\n{\n let inputs_with_separator = [separator.to_field()].concat(inputs);\n poseidon2_hash(inputs_with_separator)\n}\n\n/// Computes a Poseidon2 hash over a dynamic-length subarray of the given input.\n/// Only the first `in_len` fields of `input` are absorbed; any remaining fields are ignored.\n/// The caller is responsible for ensuring that the input is padded with zeros if required.\n#[no_predicates]\npub fn poseidon2_hash_subarray<let N: u32>(input: [Field; N], in_len: u32) -> Field {\n let mut sponge = poseidon2_absorb_in_chunks(input, in_len);\n sponge.squeeze()\n}\n\n// This function is unconstrained because it is intended to be used in unconstrained context only as\n// in constrained contexts it would be too inefficient.\npub unconstrained fn poseidon2_hash_with_separator_bounded_vec<let N: u32, T>(\n inputs: BoundedVec<Field, N>,\n separator: T,\n) -> Field\nwhere\n T: ToField,\n{\n let in_len = inputs.len() + 1;\n let iv: Field = (in_len as Field) * TWO_POW_64;\n let mut sponge = Poseidon2Sponge::new(iv);\n sponge.absorb(separator.to_field());\n\n for i in 0..inputs.len() {\n sponge.absorb(inputs.get(i));\n }\n\n sponge.squeeze()\n}\n\n#[no_predicates]\npub fn poseidon2_hash_bytes<let N: u32>(inputs: [u8; N]) -> Field {\n let mut fields = [0; (N + 30) / 31];\n let mut field_index = 0;\n let mut current_field = [0; 31];\n for i in 0..inputs.len() {\n let index = i % 31;\n current_field[index] = inputs[i];\n if index == 30 {\n fields[field_index] = field_from_bytes(current_field, false);\n current_field = [0; 31];\n field_index += 1;\n }\n }\n if field_index != fields.len() {\n fields[field_index] = field_from_bytes(current_field, false);\n }\n poseidon2_hash(fields)\n}\n\n#[test]\nfn subarray_hash_matches_fixed() {\n let values_to_hash = [3; 17];\n let padded = values_to_hash.concat([0; 11]);\n let subarray_hash = poseidon2_hash_subarray(padded, values_to_hash.len());\n\n // Hash the entire values_to_hash.\n let fixed_len_hash = poseidon::poseidon2::Poseidon2::hash(values_to_hash, values_to_hash.len());\n\n assert_eq(subarray_hash, fixed_len_hash);\n}\n\n#[test]\nfn subarray_hash_matches_variable() {\n let values_to_hash = [3; 17];\n let padded = values_to_hash.concat([0; 11]);\n let subarray_hash = poseidon2_hash_subarray(padded, values_to_hash.len());\n\n // Hash up to values_to_hash.len() fields of the padded array.\n let variable_len_hash = poseidon::poseidon2::Poseidon2::hash(padded, values_to_hash.len());\n\n assert_eq(subarray_hash, variable_len_hash);\n}\n\n#[test]\nfn smoke_sha256_to_field() {\n let full_buffer = [\n 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24,\n 25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47,\n 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, 62, 63, 64, 65, 66, 67, 68, 69, 70,\n 71, 72, 73, 74, 75, 76, 77, 78, 79, 80, 81, 82, 83, 84, 85, 86, 87, 88, 89, 90, 91, 92, 93,\n 94, 95, 96, 97, 98, 99, 100, 101, 102, 103, 104, 105, 106, 107, 108, 109, 110, 111, 112,\n 113, 114, 115, 116, 117, 118, 119, 120, 121, 122, 123, 124, 125, 126, 127, 128, 129, 130,\n 131, 132, 133, 134, 135, 136, 137, 138, 139, 140, 141, 142, 143, 144, 145, 146, 147, 148,\n 149, 150, 151, 152, 153, 154, 155, 156, 157, 158, 159,\n ];\n let result = sha256_to_field(full_buffer);\n\n assert(result == 0x448ebbc9e1a31220a2f3830c18eef61b9bd070e5084b7fa2a359fe729184c7);\n\n // to show correctness of the current ver (truncate one byte) vs old ver (mod full bytes):\n let result_bytes = sha256::digest(full_buffer);\n let truncated_field = crate::utils::field::field_from_bytes_32_trunc(result_bytes);\n assert(truncated_field == result);\n let mod_res = result + (result_bytes[31] as Field);\n assert(mod_res == 0x448ebbc9e1a31220a2f3830c18eef61b9bd070e5084b7fa2a359fe729184e0);\n}\n\n#[test]\nfn unique_siloed_note_hash_matches_typescript() {\n let inner_note_hash = 1;\n let contract_address = AztecAddress::from_field(2);\n let first_nullifier = 3;\n let note_index_in_tx = 4;\n\n let siloed_note_hash = compute_siloed_note_hash(contract_address, inner_note_hash);\n let siloed_note_hash_from_ts =\n 0x1986a4bea3eddb1fff917d629a13e10f63f514f401bdd61838c6b475db949169;\n assert_eq(siloed_note_hash, siloed_note_hash_from_ts);\n\n let nonce: Field = compute_note_hash_nonce(first_nullifier, note_index_in_tx);\n let note_hash_nonce_from_ts =\n 0x28e7799791bf066a57bb51fdd0fbcaf3f0926414314c7db515ea343f44f5d58b;\n assert_eq(nonce, note_hash_nonce_from_ts);\n\n let unique_siloed_note_hash_from_nonce = compute_unique_note_hash(nonce, siloed_note_hash);\n let unique_siloed_note_hash = compute_note_nonce_and_unique_note_hash(\n siloed_note_hash,\n first_nullifier,\n note_index_in_tx,\n );\n assert_eq(unique_siloed_note_hash_from_nonce, unique_siloed_note_hash);\n\n let unique_siloed_note_hash_from_ts =\n 0x29949aef207b715303b24639737c17fbfeb375c1d965ecfa85c7e4f0febb7d16;\n assert_eq(unique_siloed_note_hash, unique_siloed_note_hash_from_ts);\n}\n\n#[test]\nfn siloed_nullifier_matches_typescript() {\n let contract_address = AztecAddress::from_field(123);\n let nullifier = 456;\n\n let res = compute_siloed_nullifier(contract_address, nullifier);\n\n let siloed_nullifier_from_ts =\n 0x169b50336c1f29afdb8a03d955a81e485f5ac7d5f0b8065673d1e407e5877813;\n\n assert_eq(res, siloed_nullifier_from_ts);\n}\n\n#[test]\nfn siloed_private_log_first_field_matches_typescript() {\n let contract_address = AztecAddress::from_field(123);\n let field = 456;\n let res = compute_siloed_private_log_first_field(contract_address, field);\n\n let siloed_private_log_first_field_from_ts =\n 0x29480984f7b9257fded523d50addbcfc8d1d33adcf2db73ef3390a8fd5cdffaa;\n\n assert_eq(res, siloed_private_log_first_field_from_ts);\n}\n\n#[test]\nfn empty_l2_to_l1_message_hash_matches_typescript() {\n // All zeroes\n let res = compute_l2_to_l1_message_hash(\n L2ToL1Message { recipient: EthAddress::zero(), content: 0 }.scope(AztecAddress::from_field(\n 0,\n )),\n 0,\n 0,\n );\n\n let empty_l2_to_l1_msg_hash_from_ts =\n 0x003b18c58c739716e76429634a61375c45b3b5cd470c22ab6d3e14cee23dd992;\n\n assert_eq(res, empty_l2_to_l1_msg_hash_from_ts);\n}\n\n#[test]\nfn l2_to_l1_message_hash_matches_typescript() {\n let message = L2ToL1Message { recipient: EthAddress::from_field(1), content: 2 }.scope(\n AztecAddress::from_field(3),\n );\n let version = 4;\n let chainId = 5;\n\n let hash = compute_l2_to_l1_message_hash(message, version, chainId);\n\n // The following value was generated by `yarn-project/stdlib/src/hash/hash.test.ts`\n let l2_to_l1_message_hash_from_ts =\n 0x0081edf209e087ad31b3fd24263698723d57190bd1d6e9fe056fc0c0a68ee661;\n\n assert_eq(hash, l2_to_l1_message_hash_from_ts);\n}\n\n#[test]\nunconstrained fn poseidon2_hash_with_separator_bounded_vec_matches_non_bounded_vec_version() {\n let inputs = BoundedVec::<Field, 4>::from_array([1, 2, 3]);\n let separator = 42;\n\n // Hash using bounded vec version\n let bounded_result = poseidon2_hash_with_separator_bounded_vec(inputs, separator);\n\n // Hash using regular version\n let regular_result = poseidon2_hash_with_separator([1, 2, 3], separator);\n\n // Results should match\n assert_eq(bounded_result, regular_result);\n}\n"
|
|
2262
2262
|
},
|
|
2263
|
-
"
|
|
2263
|
+
"359": {
|
|
2264
2264
|
"function_locations": [
|
|
2265
2265
|
{
|
|
2266
2266
|
"name": "fatal_log",
|
|
@@ -2334,7 +2334,7 @@
|
|
|
2334
2334
|
"path": "/home/aztec-dev/aztec-packages/noir-projects/noir-protocol-circuits/crates/types/src/logging.nr",
|
|
2335
2335
|
"source": "// Log levels matching the JS logger:\n\n// global SILENT_LOG_LEVEL: u8 = 0;\nglobal FATAL_LOG_LEVEL: u8 = 1;\nglobal ERROR_LOG_LEVEL: u8 = 2;\nglobal WARN_LOG_LEVEL: u8 = 3;\nglobal INFO_LOG_LEVEL: u8 = 4;\nglobal VERBOSE_LOG_LEVEL: u8 = 5;\nglobal DEBUG_LOG_LEVEL: u8 = 6;\nglobal TRACE_LOG_LEVEL: u8 = 7;\n\n// --- Per-level log functions (no format args) ---\n\npub fn fatal_log<let N: u32>(msg: str<N>) {\n fatal_log_format(msg, []);\n}\n\npub fn error_log<let N: u32>(msg: str<N>) {\n error_log_format(msg, []);\n}\n\npub fn warn_log<let N: u32>(msg: str<N>) {\n warn_log_format(msg, []);\n}\n\npub fn info_log<let N: u32>(msg: str<N>) {\n info_log_format(msg, []);\n}\n\npub fn verbose_log<let N: u32>(msg: str<N>) {\n verbose_log_format(msg, []);\n}\n\npub fn debug_log<let N: u32>(msg: str<N>) {\n debug_log_format(msg, []);\n}\n\npub fn trace_log<let N: u32>(msg: str<N>) {\n trace_log_format(msg, []);\n}\n\n// --- Per-level log functions (with format args) ---\n\npub fn fatal_log_format<let M: u32, let N: u32>(msg: str<M>, args: [Field; N]) {\n log_format(FATAL_LOG_LEVEL, msg, args);\n}\n\npub fn error_log_format<let M: u32, let N: u32>(msg: str<M>, args: [Field; N]) {\n log_format(ERROR_LOG_LEVEL, msg, args);\n}\n\npub fn warn_log_format<let M: u32, let N: u32>(msg: str<M>, args: [Field; N]) {\n log_format(WARN_LOG_LEVEL, msg, args);\n}\n\npub fn info_log_format<let M: u32, let N: u32>(msg: str<M>, args: [Field; N]) {\n log_format(INFO_LOG_LEVEL, msg, args);\n}\n\npub fn verbose_log_format<let M: u32, let N: u32>(msg: str<M>, args: [Field; N]) {\n log_format(VERBOSE_LOG_LEVEL, msg, args);\n}\n\npub fn debug_log_format<let M: u32, let N: u32>(msg: str<M>, args: [Field; N]) {\n log_format(DEBUG_LOG_LEVEL, msg, args);\n}\n\npub fn trace_log_format<let M: u32, let N: u32>(msg: str<M>, args: [Field; N]) {\n log_format(TRACE_LOG_LEVEL, msg, args);\n}\n\nfn log_format<let M: u32, let N: u32>(log_level: u8, msg: str<M>, args: [Field; N]) {\n // Safety: This oracle call returns nothing: we only call it for its side effects. It is therefore always safe\n // to call.\n unsafe { log_oracle_wrapper(log_level, msg, args) };\n}\n\nunconstrained fn log_oracle_wrapper<let M: u32, let N: u32>(\n log_level: u8,\n msg: str<M>,\n args: [Field; N],\n) {\n log_oracle(log_level, msg, N, args);\n}\n\n// While the length parameter might seem unnecessary given that we have N, we keep it around because at the AVM\n// bytecode level we want to support non-comptime-known lengths for such opcodes, even if Noir code will not generally\n// take that route. The AVM transpiler maps this oracle to the DEBUGLOG opcode, which reads the fields size from memory.\n#[oracle(aztec_utl_log)]\nunconstrained fn log_oracle<let M: u32, let N: u32>(\n log_level: u8,\n msg: str<M>,\n length: u32,\n args: [Field; N],\n) {}\n"
|
|
2336
2336
|
},
|
|
2337
|
-
"
|
|
2337
|
+
"378": {
|
|
2338
2338
|
"function_locations": [
|
|
2339
2339
|
{
|
|
2340
2340
|
"name": "Poseidon2Sponge::hash",
|
|
@@ -2364,7 +2364,7 @@
|
|
|
2364
2364
|
"path": "/home/aztec-dev/aztec-packages/noir-projects/noir-protocol-circuits/crates/types/src/poseidon2.nr",
|
|
2365
2365
|
"source": "use crate::constants::TWO_POW_64;\nuse crate::traits::{Deserialize, Serialize};\nuse std::meta::derive;\n// NB: This is a clone of noir/noir-repo/noir_stdlib/src/hash/poseidon2.nr\n// It exists as we sometimes need to perform custom absorption, but the stdlib version\n// has a private absorb() method (it's also designed to just be a hasher)\n// Can be removed when standalone noir poseidon lib exists: See noir#6679\n// TODO: Poseidon is stand-alone now\n\nglobal RATE: u32 = 3;\n\n#[derive(Deserialize, Eq, Serialize)]\npub struct Poseidon2Sponge {\n pub cache: [Field; 3],\n pub state: [Field; 4],\n pub cache_size: u32,\n pub squeeze_mode: bool, // 0 => absorb, 1 => squeeze\n}\n\nimpl Poseidon2Sponge {\n #[no_predicates]\n pub fn hash<let N: u32>(input: [Field; N], message_size: u32) -> Field {\n Poseidon2Sponge::hash_internal(input, message_size, message_size != N)\n }\n\n pub(crate) fn new(iv: Field) -> Poseidon2Sponge {\n let mut result =\n Poseidon2Sponge { cache: [0; 3], state: [0; 4], cache_size: 0, squeeze_mode: false };\n result.state[RATE] = iv;\n result\n }\n\n fn perform_duplex(&mut self) {\n // add the cache into sponge state\n for i in 0..RATE {\n // We effectively zero-pad the cache by only adding to the state\n // cache that is less than the specified `cache_size`\n if i < self.cache_size {\n self.state[i] += self.cache[i];\n }\n }\n self.state = std::hash::poseidon2_permutation(self.state);\n }\n\n pub fn absorb(&mut self, input: Field) {\n assert(!self.squeeze_mode);\n if self.cache_size == RATE {\n // If we're absorbing, and the cache is full, apply the sponge permutation to compress the cache\n self.perform_duplex();\n self.cache[0] = input;\n self.cache_size = 1;\n } else {\n // If we're absorbing, and the cache is not full, add the input into the cache\n self.cache[self.cache_size] = input;\n self.cache_size += 1;\n }\n }\n\n pub fn squeeze(&mut self) -> Field {\n assert(!self.squeeze_mode);\n // If we're in absorb mode, apply sponge permutation to compress the cache.\n self.perform_duplex();\n self.squeeze_mode = true;\n\n // Pop one item off the top of the permutation and return it.\n self.state[0]\n }\n\n fn hash_internal<let N: u32>(\n input: [Field; N],\n in_len: u32,\n is_variable_length: bool,\n ) -> Field {\n let iv: Field = (in_len as Field) * TWO_POW_64;\n let mut sponge = Poseidon2Sponge::new(iv);\n for i in 0..input.len() {\n if i < in_len {\n sponge.absorb(input[i]);\n }\n }\n\n sponge.squeeze()\n }\n}\n"
|
|
2366
2366
|
},
|
|
2367
|
-
"
|
|
2367
|
+
"390": {
|
|
2368
2368
|
"function_locations": [
|
|
2369
2369
|
{
|
|
2370
2370
|
"name": "derive_storage_slot_in_map",
|
|
@@ -2378,7 +2378,7 @@
|
|
|
2378
2378
|
"path": "/home/aztec-dev/aztec-packages/noir-projects/noir-protocol-circuits/crates/types/src/storage/map.nr",
|
|
2379
2379
|
"source": "use crate::{\n constants::DOM_SEP__PUBLIC_STORAGE_MAP_SLOT, hash::poseidon2_hash_with_separator,\n traits::ToField,\n};\n\n// TODO: Move this to src/public_data/storage/map.nr\npub fn derive_storage_slot_in_map<K>(storage_slot: Field, key: K) -> Field\nwhere\n K: ToField,\n{\n poseidon2_hash_with_separator(\n [storage_slot, key.to_field()],\n DOM_SEP__PUBLIC_STORAGE_MAP_SLOT,\n )\n}\n\nmod test {\n use crate::{address::AztecAddress, storage::map::derive_storage_slot_in_map, traits::FromField};\n\n #[test]\n fn test_derive_storage_slot_in_map_matches_typescript() {\n let map_slot = 0x132258fb6962c4387ba659d9556521102d227549a386d39f0b22d1890d59c2b5;\n let key = AztecAddress::from_field(\n 0x302dbc2f9b50a73283d5fb2f35bc01eae8935615817a0b4219a057b2ba8a5a3f,\n );\n\n let slot = derive_storage_slot_in_map(map_slot, key);\n\n // The following value was generated by `map_slot.test.ts`\n let slot_from_typescript =\n 0x2d225f361108379adc2da91378b9702675c5546b57e78bafc1e74ec7fec55967;\n\n assert_eq(slot, slot_from_typescript);\n }\n}\n"
|
|
2380
2380
|
},
|
|
2381
|
-
"
|
|
2381
|
+
"397": {
|
|
2382
2382
|
"function_locations": [
|
|
2383
2383
|
{
|
|
2384
2384
|
"name": "<impl ToField for Field>::to_field",
|
|
@@ -2416,7 +2416,7 @@
|
|
|
2416
2416
|
"path": "/home/aztec-dev/aztec-packages/noir-projects/noir-protocol-circuits/crates/types/src/traits/to_field.nr",
|
|
2417
2417
|
"source": "use crate::utils::field::field_from_bytes;\n\npub trait ToField {\n fn to_field(self) -> Field;\n}\n\nimpl ToField for Field {\n #[inline_always]\n fn to_field(self) -> Field {\n self\n }\n}\n\nimpl ToField for bool {\n #[inline_always]\n fn to_field(self) -> Field {\n self as Field\n }\n}\nimpl ToField for u8 {\n #[inline_always]\n fn to_field(self) -> Field {\n self as Field\n }\n}\nimpl ToField for u16 {\n fn to_field(self) -> Field {\n self as Field\n }\n}\nimpl ToField for u32 {\n #[inline_always]\n fn to_field(self) -> Field {\n self as Field\n }\n}\nimpl ToField for u64 {\n #[inline_always]\n fn to_field(self) -> Field {\n self as Field\n }\n}\nimpl ToField for u128 {\n #[inline_always]\n fn to_field(self) -> Field {\n self as Field\n }\n}\nimpl<let N: u32> ToField for str<N> {\n #[inline_always]\n fn to_field(self) -> Field {\n assert(N < 32, \"String doesn't fit in a field, consider using Serialize instead\");\n field_from_bytes(self.as_bytes(), true)\n }\n}\n"
|
|
2418
2418
|
},
|
|
2419
|
-
"
|
|
2419
|
+
"398": {
|
|
2420
2420
|
"function_locations": [
|
|
2421
2421
|
{
|
|
2422
2422
|
"name": "<impl Packable for bool>::pack",
|
|
@@ -2538,7 +2538,7 @@
|
|
|
2538
2538
|
"path": "/home/aztec-dev/aztec-packages/noir-projects/noir-protocol-circuits/crates/types/src/type_packing.nr",
|
|
2539
2539
|
"source": "use crate::traits::Packable;\n\nglobal BOOL_PACKED_LEN: u32 = 1;\nglobal U8_PACKED_LEN: u32 = 1;\nglobal U16_PACKED_LEN: u32 = 1;\nglobal U32_PACKED_LEN: u32 = 1;\nglobal U64_PACKED_LEN: u32 = 1;\nglobal U128_PACKED_LEN: u32 = 1;\nglobal FIELD_PACKED_LEN: u32 = 1;\nglobal I8_PACKED_LEN: u32 = 1;\nglobal I16_PACKED_LEN: u32 = 1;\nglobal I32_PACKED_LEN: u32 = 1;\nglobal I64_PACKED_LEN: u32 = 1;\n\nimpl Packable for bool {\n let N: u32 = BOOL_PACKED_LEN;\n\n #[inline_always]\n fn pack(self) -> [Field; Self::N] {\n [self as Field]\n }\n\n #[inline_always]\n fn unpack(fields: [Field; Self::N]) -> bool {\n (fields[0] as u8) % 2 != 0\n }\n}\n\nimpl Packable for u8 {\n let N: u32 = U8_PACKED_LEN;\n\n #[inline_always]\n fn pack(self) -> [Field; Self::N] {\n [self as Field]\n }\n\n #[inline_always]\n fn unpack(fields: [Field; Self::N]) -> Self {\n fields[0] as u8\n }\n}\n\nimpl Packable for u16 {\n let N: u32 = U16_PACKED_LEN;\n\n #[inline_always]\n fn pack(self) -> [Field; Self::N] {\n [self as Field]\n }\n\n #[inline_always]\n fn unpack(fields: [Field; Self::N]) -> Self {\n fields[0] as u16\n }\n}\n\nimpl Packable for u32 {\n let N: u32 = U32_PACKED_LEN;\n\n #[inline_always]\n fn pack(self) -> [Field; Self::N] {\n [self as Field]\n }\n\n #[inline_always]\n fn unpack(fields: [Field; Self::N]) -> Self {\n fields[0] as u32\n }\n}\n\nimpl Packable for u64 {\n let N: u32 = U64_PACKED_LEN;\n\n #[inline_always]\n fn pack(self) -> [Field; Self::N] {\n [self as Field]\n }\n\n #[inline_always]\n fn unpack(fields: [Field; Self::N]) -> Self {\n fields[0] as u64\n }\n}\n\nimpl Packable for u128 {\n let N: u32 = U128_PACKED_LEN;\n\n #[inline_always]\n fn pack(self) -> [Field; Self::N] {\n [self as Field]\n }\n\n #[inline_always]\n fn unpack(fields: [Field; Self::N]) -> Self {\n fields[0] as u128\n }\n}\n\nimpl Packable for Field {\n let N: u32 = FIELD_PACKED_LEN;\n\n #[inline_always]\n fn pack(self) -> [Field; Self::N] {\n [self]\n }\n\n #[inline_always]\n fn unpack(fields: [Field; Self::N]) -> Self {\n fields[0]\n }\n}\n\nimpl Packable for i8 {\n let N: u32 = I8_PACKED_LEN;\n\n #[inline_always]\n fn pack(self) -> [Field; Self::N] {\n [self as u8 as Field]\n }\n\n #[inline_always]\n fn unpack(fields: [Field; Self::N]) -> Self {\n fields[0] as u8 as i8\n }\n}\n\nimpl Packable for i16 {\n let N: u32 = I16_PACKED_LEN;\n\n #[inline_always]\n fn pack(self) -> [Field; Self::N] {\n [self as u16 as Field]\n }\n\n #[inline_always]\n fn unpack(fields: [Field; Self::N]) -> Self {\n fields[0] as u16 as i16\n }\n}\n\nimpl Packable for i32 {\n let N: u32 = I32_PACKED_LEN;\n\n #[inline_always]\n fn pack(self) -> [Field; Self::N] {\n [self as u32 as Field]\n }\n\n #[inline_always]\n fn unpack(fields: [Field; Self::N]) -> Self {\n fields[0] as u32 as i32\n }\n}\n\nimpl Packable for i64 {\n let N: u32 = I64_PACKED_LEN;\n\n #[inline_always]\n fn pack(self) -> [Field; Self::N] {\n [self as u64 as Field]\n }\n\n #[inline_always]\n fn unpack(fields: [Field; Self::N]) -> Self {\n fields[0] as u64 as i64\n }\n}\n\nimpl<T, let M: u32> Packable for [T; M]\nwhere\n T: Packable,\n{\n let N: u32 = M * <T as Packable>::N;\n\n #[inline_always]\n fn pack(self) -> [Field; Self::N] {\n let mut result: [Field; Self::N] = std::mem::zeroed();\n for i in 0..M {\n let serialized = self[i].pack();\n for j in 0..<T as Packable>::N {\n result[i * <T as Packable>::N + j] = serialized[j];\n }\n }\n result\n }\n\n #[inline_always]\n fn unpack(fields: [Field; Self::N]) -> Self {\n let mut reader = crate::utils::reader::Reader::new(fields);\n let result: [T; M] = std::mem::zeroed();\n reader.read_struct_array::<T, <T as Packable>::N, M>(Packable::unpack, result)\n }\n}\n\n#[test]\nfn test_u16_packing() {\n let a: u16 = 10;\n assert_eq(a, u16::unpack(a.pack()));\n}\n\n#[test]\nfn test_i8_packing() {\n let a: i8 = -10;\n assert_eq(a, i8::unpack(a.pack()));\n}\n\n#[test]\nfn test_i16_packing() {\n let a: i16 = -10;\n assert_eq(a, i16::unpack(a.pack()));\n}\n\n#[test]\nfn test_i32_packing() {\n let a: i32 = -10;\n assert_eq(a, i32::unpack(a.pack()));\n}\n\n#[test]\nfn test_i64_packing() {\n let a: i64 = -10;\n assert_eq(a, i64::unpack(a.pack()));\n}\n"
|
|
2540
2540
|
},
|
|
2541
|
-
"
|
|
2541
|
+
"405": {
|
|
2542
2542
|
"function_locations": [
|
|
2543
2543
|
{
|
|
2544
2544
|
"name": "field_from_bytes",
|
|
@@ -2750,7 +2750,7 @@
|
|
|
2750
2750
|
"path": "std/option.nr",
|
|
2751
2751
|
"source": "use crate::cmp::{Eq, Ord, Ordering};\nuse crate::default::Default;\nuse crate::hash::{Hash, Hasher};\n\n/// Represents a value of type T or its absence.\n/// Use `Option::some(value)` to construct a value or `Option::none()` to record the absence of one.\npub struct Option<T> {\n _is_some: bool,\n _value: T,\n}\n\nimpl<T> Option<T> {\n /// Constructs a None value\n pub fn none() -> Self {\n Self { _is_some: false, _value: crate::mem::zeroed() }\n }\n\n /// Constructs a Some wrapper around the given value\n pub fn some(_value: T) -> Self {\n Self { _is_some: true, _value }\n }\n\n /// True if this Option is None\n pub fn is_none(&self) -> bool {\n !self._is_some\n }\n\n /// True if this Option is Some\n pub fn is_some(&self) -> bool {\n self._is_some\n }\n\n /// Asserts `self.is_some()` and returns the wrapped value.\n pub fn unwrap(self) -> T {\n assert(self._is_some);\n self._value\n }\n\n /// Returns the inner value without asserting `self.is_some()`\n /// Note that if `self` is `None`, there is no guarantee what value will be returned,\n /// only that it will be of type `T`.\n pub fn unwrap_unchecked(self) -> T {\n self._value\n }\n\n /// Returns the wrapped value if `self.is_some()`. Otherwise, returns the given default value.\n pub fn unwrap_or(self, default: T) -> T {\n if self._is_some {\n self._value\n } else {\n default\n }\n }\n\n /// Returns the wrapped value if `self.is_some()`. Otherwise, calls the given function to return\n /// a default value.\n pub fn unwrap_or_else<Env>(self, default: fn[Env]() -> T) -> T {\n if self._is_some {\n self._value\n } else {\n default()\n }\n }\n\n /// Asserts `self.is_some()` with a provided custom message and returns the contained `Some` value\n pub fn expect<let N: u32, MessageTypes>(self, message: fmtstr<N, MessageTypes>) -> T {\n assert(self.is_some(), message);\n self._value\n }\n\n /// If self is `Some(x)`, this returns `Some(f(x))`. Otherwise, this returns `None`.\n pub fn map<U, Env>(self, f: fn[Env](T) -> U) -> Option<U> {\n if self._is_some {\n Option::some(f(self._value))\n } else {\n Option::none()\n }\n }\n\n /// If self is `Some(x)`, this returns `f(x)`. Otherwise, this returns the given default value.\n pub fn map_or<U, Env>(self, default: U, f: fn[Env](T) -> U) -> U {\n if self._is_some {\n f(self._value)\n } else {\n default\n }\n }\n\n /// If self is `Some(x)`, this returns `f(x)`. Otherwise, this returns `default()`.\n pub fn map_or_else<U, Env1, Env2>(self, default: fn[Env1]() -> U, f: fn[Env2](T) -> U) -> U {\n if self._is_some {\n f(self._value)\n } else {\n default()\n }\n }\n\n /// Returns None if self is None. Otherwise, this returns `other`.\n pub fn and(self, other: Self) -> Self {\n if self.is_none() {\n Option::none()\n } else {\n other\n }\n }\n\n /// If self is None, this returns None. Otherwise, this calls the given function\n /// with the Some value contained within self, and returns the result of that call.\n ///\n /// In some languages this function is called `flat_map` or `bind`.\n pub fn and_then<U, Env>(self, f: fn[Env](T) -> Option<U>) -> Option<U> {\n if self._is_some {\n f(self._value)\n } else {\n Option::none()\n }\n }\n\n /// If self is Some, return self. Otherwise, return `other`.\n pub fn or(self, other: Self) -> Self {\n if self._is_some {\n self\n } else {\n other\n }\n }\n\n /// If self is Some, return self. Otherwise, return `default()`.\n pub fn or_else<Env>(self, default: fn[Env]() -> Self) -> Self {\n if self._is_some {\n self\n } else {\n default()\n }\n }\n\n // If only one of the two Options is Some, return that option.\n // Otherwise, if both options are Some or both are None, None is returned.\n pub fn xor(self, other: Self) -> Self {\n if self._is_some {\n if other._is_some {\n Option::none()\n } else {\n self\n }\n } else if other._is_some {\n other\n } else {\n Option::none()\n }\n }\n\n /// Returns `Some(x)` if self is `Some(x)` and `predicate(x)` is true.\n /// Otherwise, this returns `None`\n pub fn filter<Env>(self, predicate: fn[Env](T) -> bool) -> Self {\n if self._is_some {\n if predicate(self._value) {\n self\n } else {\n Option::none()\n }\n } else {\n Option::none()\n }\n }\n\n /// Flattens an Option<Option<T>> into a Option<T>.\n /// This returns None if the outer Option is None. Otherwise, this returns the inner Option.\n pub fn flatten(option: Option<Option<T>>) -> Option<T> {\n if option._is_some {\n option._value\n } else {\n Option::none()\n }\n }\n}\n\nimpl<T> Default for Option<T> {\n fn default() -> Self {\n Option::none()\n }\n}\n\nimpl<T> Eq for Option<T>\nwhere\n T: Eq,\n{\n fn eq(self, other: Self) -> bool {\n if self._is_some == other._is_some {\n if self._is_some {\n self._value == other._value\n } else {\n true\n }\n } else {\n false\n }\n }\n}\n\nimpl<T> Hash for Option<T>\nwhere\n T: Hash,\n{\n fn hash<H>(self, state: &mut H)\n where\n H: Hasher,\n {\n self._is_some.hash(state);\n if self._is_some {\n self._value.hash(state);\n }\n }\n}\n\n// For this impl we're declaring Option::none < Option::some\nimpl<T> Ord for Option<T>\nwhere\n T: Ord,\n{\n fn cmp(self, other: Self) -> Ordering {\n if self._is_some {\n if other._is_some {\n self._value.cmp(other._value)\n } else {\n Ordering::greater()\n }\n } else if other._is_some {\n Ordering::less()\n } else {\n Ordering::equal()\n }\n }\n}\n"
|
|
2752
2752
|
},
|
|
2753
|
-
"
|
|
2753
|
+
"411": {
|
|
2754
2754
|
"function_locations": [
|
|
2755
2755
|
{
|
|
2756
2756
|
"name": "Reader<N>::new",
|
|
@@ -2800,7 +2800,7 @@
|
|
|
2800
2800
|
"path": "/home/aztec-dev/aztec-packages/noir-projects/noir-protocol-circuits/crates/serde/src/reader.nr",
|
|
2801
2801
|
"source": "pub struct Reader<let N: u32> {\n data: [Field; N],\n offset: u32,\n}\n\nimpl<let N: u32> Reader<N> {\n pub fn new(data: [Field; N]) -> Self {\n Self { data, offset: 0 }\n }\n\n pub fn read(&mut self) -> Field {\n let result = self.data[self.offset];\n self.offset += 1;\n result\n }\n\n pub fn read_u32(&mut self) -> u32 {\n self.read() as u32\n }\n\n pub fn read_u64(&mut self) -> u64 {\n self.read() as u64\n }\n\n pub fn read_bool(&mut self) -> bool {\n self.read() != 0\n }\n\n pub fn read_array<let K: u32>(&mut self) -> [Field; K] {\n let mut result = [0; K];\n for i in 0..K {\n result[i] = self.data[self.offset + i];\n }\n self.offset += K;\n result\n }\n\n pub fn read_struct<T, let K: u32>(&mut self, deserialise: fn([Field; K]) -> T) -> T {\n let result = deserialise(self.read_array());\n result\n }\n\n pub fn read_struct_array<T, let K: u32, let C: u32>(\n &mut self,\n deserialise: fn([Field; K]) -> T,\n mut result: [T; C],\n ) -> [T; C] {\n for i in 0..C {\n result[i] = self.read_struct(deserialise);\n }\n result\n }\n\n pub fn peek_offset(&mut self, offset: u32) -> Field {\n self.data[self.offset + offset]\n }\n\n pub fn advance_offset(&mut self, offset: u32) {\n self.offset += offset;\n }\n\n pub fn finish(self) {\n assert_eq(self.offset, self.data.len(), \"Reader did not read all data\");\n }\n}\n"
|
|
2802
2802
|
},
|
|
2803
|
-
"
|
|
2803
|
+
"412": {
|
|
2804
2804
|
"function_locations": [
|
|
2805
2805
|
{
|
|
2806
2806
|
"name": "derive_serialize",
|
|
@@ -2826,7 +2826,7 @@
|
|
|
2826
2826
|
"path": "/home/aztec-dev/aztec-packages/noir-projects/noir-protocol-circuits/crates/serde/src/serialization.nr",
|
|
2827
2827
|
"source": "use crate::{reader::Reader, writer::Writer};\n\n/// Trait for serializing Noir types into arrays of Fields.\n///\n/// An implementation of the Serialize trait has to follow Noir's intrinsic serialization (each member of a struct\n/// converted directly into one or more Fields without any packing or compression). This trait (and Deserialize) are\n/// typically used to communicate between Noir and TypeScript (via oracles and function arguments).\n///\n/// # On Following Noir's Intrinsic Serialization\n/// When calling a Noir function from TypeScript (TS), first the function arguments are serialized into an array\n/// of fields. This array is then included in the initial witness. Noir's intrinsic serialization is then used\n/// to deserialize the arguments from the witness. When the same Noir function is called from Noir this Serialize trait\n/// is used instead of the serialization in TS. For this reason we need to have a match between TS serialization,\n/// Noir's intrinsic serialization and the implementation of this trait. If there is a mismatch, the function calls\n/// fail with an arguments hash mismatch error message.\n///\n/// # Associated Constants\n/// * `N` - The length of the output Field array, known at compile time\n///\n/// # Example\n/// ```\n/// impl<let N: u32> Serialize for str<N> {\n/// let N: u32 = N;\n///\n/// fn serialize(self) -> [Field; Self::N] {\n/// let mut writer: Writer<Self::N> = Writer::new();\n/// self.stream_serialize(&mut writer);\n/// writer.finish()\n/// }\n///\n/// fn stream_serialize<let K: u32>(self, writer: &mut Writer<K>) {\n/// let bytes = self.as_bytes();\n/// for i in 0..bytes.len() {\n/// writer.write(bytes[i] as Field);\n/// }\n/// }\n/// }\n/// ```\n#[derive_via(derive_serialize)]\npub trait Serialize {\n let N: u32;\n\n fn serialize(self) -> [Field; Self::N];\n\n fn stream_serialize<let K: u32>(self, writer: &mut Writer<K>);\n}\n\n/// Generates a `Serialize` trait implementation for a struct type.\n///\n/// # Parameters\n/// - `s`: The struct type definition to generate the implementation for\n///\n/// # Returns\n/// A quoted code block containing the trait implementation\n///\n/// # Example\n/// For a struct defined as:\n/// ```\n/// struct Log<N> {\n/// fields: [Field; N],\n/// length: u32\n/// }\n/// ```\n///\n/// This function generates code equivalent to:\n/// ```\n/// impl<let N: u32> Serialize for Log<N> {\n/// let N: u32 = <[Field; N] as Serialize>::N + <u32 as Serialize>::N;\n///\n/// fn serialize(self) -> [Field; Self::N] {\n/// let mut writer: Writer<Self::N> = Writer::new();\n/// self.stream_serialize(&mut writer);\n/// writer.finish()\n/// }\n///\n/// #[inline_always]\n/// fn stream_serialize<let K: u32>(self, writer: &mut Writer<K>) {\n/// Serialize::stream_serialize(self.fields, writer);\n/// Serialize::stream_serialize(self.length, writer);\n/// }\n/// }\n/// ```\npub comptime fn derive_serialize(s: TypeDefinition) -> Quoted {\n let typ = s.as_type();\n let nested_struct = typ.as_data_type().unwrap();\n\n // We care only about the name and type so we drop the last item of the tuple\n let params = nested_struct.0.fields(nested_struct.1).map(|(name, typ, _)| (name, typ));\n\n // Generates the generic parameter declarations (to be placed after the `impl` keyword) and the `where` clause\n // for the `Serialize` trait.\n let generics_declarations = get_generics_declarations(s);\n let where_serialize_clause = get_where_trait_clause(s, quote {Serialize});\n\n let params_len_quote = get_params_len_quote(params);\n\n let function_body = params\n .map(|(name, _typ): (Quoted, Type)| {\n quote {\n $crate::serialization::Serialize::stream_serialize(self.$name, writer);\n }\n })\n .join(quote {});\n\n quote {\n impl$generics_declarations $crate::serialization::Serialize for $typ\n $where_serialize_clause\n {\n let N: u32 = $params_len_quote;\n\n\n fn serialize(self) -> [Field; Self::N] {\n let mut writer: $crate::writer::Writer<Self::N> = $crate::writer::Writer::new();\n $crate::serialization::Serialize::stream_serialize(self, &mut writer);\n writer.finish()\n }\n\n\n #[inline_always]\n fn stream_serialize<let K: u32>(self, writer: &mut $crate::writer::Writer<K>) {\n $function_body\n }\n }\n }\n}\n\n/// Trait for deserializing Noir types from arrays of Fields.\n///\n/// An implementation of the Deserialize trait has to follow Noir's intrinsic serialization (each member of a struct\n/// converted directly into one or more Fields without any packing or compression). This trait is typically used when\n/// deserializing return values from function calls in Noir. Since the same function could be called from TypeScript\n/// (TS), in which case the TS deserialization would get used, we need to have a match between the 2.\n///\n/// # Associated Constants\n/// * `N` - The length of the input Field array, known at compile time\n///\n/// # Example\n/// ```\n/// impl<let M: u32> Deserialize for str<M> {\n/// let N: u32 = M;\n///\n/// fn deserialize(fields: [Field; Self::N]) -> Self {\n/// let mut reader = Reader::new(fields);\n/// let result = Self::stream_deserialize(&mut reader);\n/// reader.finish();\n/// result\n/// }\n///\n/// fn stream_deserialize<let K: u32>(reader: &mut Reader<K>) -> Self {\n/// let mut bytes = [0 as u8; M];\n/// for i in 0..M {\n/// bytes[i] = reader.read() as u8;\n/// }\n/// str::<M>::from(bytes)\n/// }\n/// }\n/// ```\n#[derive_via(derive_deserialize)]\npub trait Deserialize {\n let N: u32;\n\n fn deserialize(fields: [Field; Self::N]) -> Self;\n\n fn stream_deserialize<let K: u32>(reader: &mut Reader<K>) -> Self;\n}\n\n/// Generates a `Deserialize` trait implementation for a given struct `s`.\n///\n/// # Arguments\n/// * `s` - The struct type definition to generate the implementation for\n///\n/// # Returns\n/// A `Quoted` block containing the generated trait implementation\n///\n/// # Requirements\n/// Each struct member type must implement the `Deserialize` trait (it gets used in the generated code).\n///\n/// # Example\n/// For a struct like:\n/// ```\n/// struct MyStruct {\n/// x: AztecAddress,\n/// y: Field,\n/// }\n/// ```\n///\n/// This generates:\n/// ```\n/// impl Deserialize for MyStruct {\n/// let N: u32 = <AztecAddress as Deserialize>::N + <Field as Deserialize>::N;\n///\n/// fn deserialize(fields: [Field; Self::N]) -> Self {\n/// let mut reader = Reader::new(fields);\n/// let result = Self::stream_deserialize(&mut reader);\n/// reader.finish();\n/// result\n/// }\n///\n/// #[inline_always]\n/// fn stream_deserialize<let K: u32>(reader: &mut Reader<K>) -> Self {\n/// let x = <AztecAddress as Deserialize>::stream_deserialize(reader);\n/// let y = <Field as Deserialize>::stream_deserialize(reader);\n/// Self { x, y }\n/// }\n/// }\n/// ```\npub comptime fn derive_deserialize(s: TypeDefinition) -> Quoted {\n let typ = s.as_type();\n let nested_struct = typ.as_data_type().unwrap();\n let params = nested_struct.0.fields(nested_struct.1);\n\n // Generates the generic parameter declarations (to be placed after the `impl` keyword) and the `where` clause\n // for the `Deserialize` trait.\n let generics_declarations = get_generics_declarations(s);\n let where_deserialize_clause = get_where_trait_clause(s, quote {Deserialize});\n\n // The following will give us:\n // <type_of_struct_member_1 as Deserialize>::N + <type_of_struct_member_2 as Deserialize>::N + ...\n // (or 0 if the struct has no members)\n let right_hand_side_of_definition_of_n = if params.len() > 0 {\n params\n .map(|(_, param_type, _): (Quoted, Type, Quoted)| {\n quote {\n <$param_type as $crate::serialization::Deserialize>::N\n }\n })\n .join(quote {+})\n } else {\n quote {0}\n };\n\n // For structs containing a single member, we can enhance performance by directly deserializing the input array,\n // bypassing the need for loop-based array construction. While this optimization yields significant benefits in\n // Brillig where the loops are expected to not be optimized, it is not relevant in ACIR where the loops are\n // expected to be optimized away.\n let function_body = if params.len() > 1 {\n // This generates deserialization code for each struct member and concatenates them together.\n let deserialization_of_struct_members = params\n .map(|(param_name, param_type, _): (Quoted, Type, Quoted)| {\n quote {\n let $param_name = <$param_type as Deserialize>::stream_deserialize(reader);\n }\n })\n .join(quote {});\n\n // We join the struct member names with a comma to be used in the `Self { ... }` syntax\n // This will give us e.g. `a, b, c` for a struct with three fields named `a`, `b`, and `c`.\n let struct_members = params\n .map(|(param_name, _, _): (Quoted, Type, Quoted)| quote { $param_name })\n .join(quote {,});\n\n quote {\n $deserialization_of_struct_members\n\n Self { $struct_members }\n }\n } else if params.len() == 1 {\n let param_name = params[0].0;\n quote {\n Self { $param_name: $crate::serialization::Deserialize::stream_deserialize(reader) }\n }\n } else {\n quote {\n Self {}\n }\n };\n\n quote {\n impl$generics_declarations $crate::serialization::Deserialize for $typ\n $where_deserialize_clause\n {\n let N: u32 = $right_hand_side_of_definition_of_n;\n\n fn deserialize(fields: [Field; Self::N]) -> Self {\n let mut reader = $crate::reader::Reader::new(fields);\n let result = Self::stream_deserialize(&mut reader);\n reader.finish();\n result\n }\n\n #[inline_always]\n fn stream_deserialize<let K: u32>(reader: &mut $crate::reader::Reader<K>) -> Self {\n $function_body\n }\n }\n }\n}\n\n/// Generates a quoted expression that computes the total serialized length of function parameters.\n///\n/// # Parameters\n/// * `params` - An array of tuples where each tuple contains a quoted parameter name and its Type. The type needs\n/// to implement the Serialize trait.\n///\n/// # Returns\n/// A quoted expression that evaluates to:\n/// * `0` if there are no parameters\n/// * `(<type1 as Serialize>::N + <type2 as Serialize>::N + ...)` for one or more parameters\ncomptime fn get_params_len_quote(params: [(Quoted, Type)]) -> Quoted {\n if params.len() == 0 {\n quote { 0 }\n } else {\n let params_quote_without_parentheses = params\n .map(|(_, param_type): (Quoted, Type)| {\n quote {\n <$param_type as $crate::serialization::Serialize>::N\n }\n })\n .join(quote {+});\n quote { ($params_quote_without_parentheses) }\n }\n}\n\ncomptime fn get_generics_declarations(s: TypeDefinition) -> Quoted {\n let generics = s.generics();\n\n if generics.len() > 0 {\n let generics_declarations_items = generics\n .map(|(name, maybe_integer_typ)| {\n // The second item in the generics tuple is an Option of an integer type that is Some only if\n // the generic is numeric.\n if maybe_integer_typ.is_some() {\n // The generic is numeric, so we return a quote defined as e.g. \"let N: u32\"\n let integer_type = maybe_integer_typ.unwrap();\n quote {let $name: $integer_type}\n } else {\n // The generic is not numeric, so we return a quote containing the name of the generic (e.g. \"T\")\n quote {$name}\n }\n })\n .join(quote {,});\n quote {<$generics_declarations_items>}\n } else {\n // The struct doesn't have any generics defined, so we just return an empty quote.\n quote {}\n }\n}\n\ncomptime fn get_where_trait_clause(s: TypeDefinition, trait_name: Quoted) -> Quoted {\n let generics = s.generics();\n\n // The second item in the generics tuple is an Option of an integer type that is Some only if the generic is\n // numeric.\n let non_numeric_generics =\n generics.filter(|(_, maybe_integer_typ)| maybe_integer_typ.is_none());\n\n if non_numeric_generics.len() > 0 {\n let non_numeric_generics_declarations =\n non_numeric_generics.map(|(name, _)| quote {$name: $trait_name}).join(quote {,});\n quote {where $non_numeric_generics_declarations}\n } else {\n // There are no non-numeric generics, so we return an empty quote.\n quote {}\n }\n}\n"
|
|
2828
2828
|
},
|
|
2829
|
-
"
|
|
2829
|
+
"414": {
|
|
2830
2830
|
"function_locations": [
|
|
2831
2831
|
{
|
|
2832
2832
|
"name": "<impl Serialize for bool>::serialize",
|
|
@@ -3140,7 +3140,7 @@
|
|
|
3140
3140
|
"path": "/home/aztec-dev/aztec-packages/noir-projects/noir-protocol-circuits/crates/serde/src/type_impls.nr",
|
|
3141
3141
|
"source": "use crate::{reader::Reader, serialization::{Deserialize, Serialize}, writer::Writer};\nuse std::embedded_curve_ops::EmbeddedCurvePoint;\nuse std::embedded_curve_ops::EmbeddedCurveScalar;\n\nglobal BOOL_SERIALIZED_LEN: u32 = 1;\nglobal U8_SERIALIZED_LEN: u32 = 1;\nglobal U16_SERIALIZED_LEN: u32 = 1;\nglobal U32_SERIALIZED_LEN: u32 = 1;\nglobal U64_SERIALIZED_LEN: u32 = 1;\nglobal U128_SERIALIZED_LEN: u32 = 1;\nglobal FIELD_SERIALIZED_LEN: u32 = 1;\nglobal I8_SERIALIZED_LEN: u32 = 1;\nglobal I16_SERIALIZED_LEN: u32 = 1;\nglobal I32_SERIALIZED_LEN: u32 = 1;\nglobal I64_SERIALIZED_LEN: u32 = 1;\n\nimpl Serialize for bool {\n let N: u32 = BOOL_SERIALIZED_LEN;\n\n fn serialize(self) -> [Field; Self::N] {\n let mut writer: Writer<Self::N> = Writer::new();\n self.stream_serialize(&mut writer);\n writer.finish()\n }\n\n #[inline_always]\n fn stream_serialize<let K: u32>(self, writer: &mut Writer<K>) {\n writer.write(self as Field);\n }\n}\n\nimpl Deserialize for bool {\n let N: u32 = BOOL_SERIALIZED_LEN;\n\n fn deserialize(fields: [Field; Self::N]) -> Self {\n let mut reader = Reader::new(fields);\n let result = Self::stream_deserialize(&mut reader);\n reader.finish();\n result\n }\n\n #[inline_always]\n fn stream_deserialize<let K: u32>(reader: &mut Reader<K>) -> bool {\n reader.read() != 0\n }\n}\n\nimpl Serialize for u8 {\n let N: u32 = U8_SERIALIZED_LEN;\n\n fn serialize(self) -> [Field; Self::N] {\n let mut writer: Writer<Self::N> = Writer::new();\n self.stream_serialize(&mut writer);\n writer.finish()\n }\n\n #[inline_always]\n fn stream_serialize<let K: u32>(self, writer: &mut Writer<K>) {\n writer.write(self as Field);\n }\n}\n\nimpl Deserialize for u8 {\n let N: u32 = U8_SERIALIZED_LEN;\n\n fn deserialize(fields: [Field; Self::N]) -> Self {\n let mut reader = Reader::new(fields);\n let result = Self::stream_deserialize(&mut reader);\n reader.finish();\n result\n }\n\n #[inline_always]\n fn stream_deserialize<let K: u32>(reader: &mut Reader<K>) -> Self {\n reader.read() as u8\n }\n}\n\nimpl Serialize for u16 {\n let N: u32 = U16_SERIALIZED_LEN;\n\n fn serialize(self) -> [Field; Self::N] {\n let mut writer: Writer<Self::N> = Writer::new();\n self.stream_serialize(&mut writer);\n writer.finish()\n }\n\n #[inline_always]\n fn stream_serialize<let K: u32>(self, writer: &mut Writer<K>) {\n writer.write(self as Field);\n }\n}\n\nimpl Deserialize for u16 {\n let N: u32 = U16_SERIALIZED_LEN;\n\n fn deserialize(fields: [Field; Self::N]) -> Self {\n let mut reader = Reader::new(fields);\n let result = Self::stream_deserialize(&mut reader);\n reader.finish();\n result\n }\n\n #[inline_always]\n fn stream_deserialize<let K: u32>(reader: &mut Reader<K>) -> Self {\n reader.read() as u16\n }\n}\n\nimpl Serialize for u32 {\n let N: u32 = U32_SERIALIZED_LEN;\n\n fn serialize(self) -> [Field; Self::N] {\n let mut writer: Writer<Self::N> = Writer::new();\n self.stream_serialize(&mut writer);\n writer.finish()\n }\n\n #[inline_always]\n fn stream_serialize<let K: u32>(self, writer: &mut Writer<K>) {\n writer.write(self as Field);\n }\n}\n\nimpl Deserialize for u32 {\n let N: u32 = U32_SERIALIZED_LEN;\n\n fn deserialize(fields: [Field; Self::N]) -> Self {\n let mut reader = Reader::new(fields);\n let result = Self::stream_deserialize(&mut reader);\n reader.finish();\n result\n }\n\n #[inline_always]\n fn stream_deserialize<let K: u32>(reader: &mut Reader<K>) -> Self {\n reader.read() as u32\n }\n}\n\nimpl Serialize for u64 {\n let N: u32 = U64_SERIALIZED_LEN;\n\n fn serialize(self) -> [Field; Self::N] {\n let mut writer: Writer<Self::N> = Writer::new();\n self.stream_serialize(&mut writer);\n writer.finish()\n }\n\n #[inline_always]\n fn stream_serialize<let K: u32>(self, writer: &mut Writer<K>) {\n writer.write(self as Field);\n }\n}\n\nimpl Deserialize for u64 {\n let N: u32 = U64_SERIALIZED_LEN;\n\n fn deserialize(fields: [Field; Self::N]) -> Self {\n let mut reader = Reader::new(fields);\n let result = Self::stream_deserialize(&mut reader);\n reader.finish();\n result\n }\n\n #[inline_always]\n fn stream_deserialize<let K: u32>(reader: &mut Reader<K>) -> Self {\n reader.read() as u64\n }\n}\n\nimpl Serialize for u128 {\n let N: u32 = U128_SERIALIZED_LEN;\n\n fn serialize(self) -> [Field; Self::N] {\n let mut writer: Writer<Self::N> = Writer::new();\n self.stream_serialize(&mut writer);\n writer.finish()\n }\n\n #[inline_always]\n fn stream_serialize<let K: u32>(self, writer: &mut Writer<K>) {\n writer.write(self as Field);\n }\n}\n\nimpl Deserialize for u128 {\n let N: u32 = U128_SERIALIZED_LEN;\n\n fn deserialize(fields: [Field; Self::N]) -> Self {\n let mut reader = Reader::new(fields);\n let result = Self::stream_deserialize(&mut reader);\n reader.finish();\n result\n }\n\n #[inline_always]\n fn stream_deserialize<let K: u32>(reader: &mut Reader<K>) -> Self {\n reader.read() as u128\n }\n}\n\nimpl Serialize for Field {\n let N: u32 = FIELD_SERIALIZED_LEN;\n\n fn serialize(self) -> [Field; Self::N] {\n let mut writer: Writer<Self::N> = Writer::new();\n self.stream_serialize(&mut writer);\n writer.finish()\n }\n\n #[inline_always]\n fn stream_serialize<let K: u32>(self, writer: &mut Writer<K>) {\n writer.write(self);\n }\n}\n\nimpl Deserialize for Field {\n let N: u32 = FIELD_SERIALIZED_LEN;\n\n fn deserialize(fields: [Field; Self::N]) -> Self {\n let mut reader = Reader::new(fields);\n let result = Self::stream_deserialize(&mut reader);\n reader.finish();\n result\n }\n\n #[inline_always]\n fn stream_deserialize<let K: u32>(reader: &mut Reader<K>) -> Self {\n reader.read()\n }\n}\n\nimpl Serialize for i8 {\n let N: u32 = I8_SERIALIZED_LEN;\n\n fn serialize(self) -> [Field; Self::N] {\n let mut writer: Writer<Self::N> = Writer::new();\n self.stream_serialize(&mut writer);\n writer.finish()\n }\n\n #[inline_always]\n fn stream_serialize<let K: u32>(self, writer: &mut Writer<K>) {\n writer.write(self as u8 as Field);\n }\n}\n\nimpl Deserialize for i8 {\n let N: u32 = I8_SERIALIZED_LEN;\n\n fn deserialize(fields: [Field; Self::N]) -> Self {\n let mut reader = Reader::new(fields);\n let result = Self::stream_deserialize(&mut reader);\n reader.finish();\n result\n }\n\n #[inline_always]\n fn stream_deserialize<let K: u32>(reader: &mut Reader<K>) -> Self {\n reader.read() as u8 as i8\n }\n}\n\nimpl Serialize for i16 {\n let N: u32 = I16_SERIALIZED_LEN;\n\n fn serialize(self) -> [Field; Self::N] {\n let mut writer: Writer<Self::N> = Writer::new();\n self.stream_serialize(&mut writer);\n writer.finish()\n }\n\n #[inline_always]\n fn stream_serialize<let K: u32>(self, writer: &mut Writer<K>) {\n writer.write(self as u16 as Field);\n }\n}\n\nimpl Deserialize for i16 {\n let N: u32 = I16_SERIALIZED_LEN;\n\n fn deserialize(fields: [Field; Self::N]) -> Self {\n let mut reader = Reader::new(fields);\n let result = Self::stream_deserialize(&mut reader);\n reader.finish();\n result\n }\n\n #[inline_always]\n fn stream_deserialize<let K: u32>(reader: &mut Reader<K>) -> Self {\n reader.read() as u16 as i16\n }\n}\n\nimpl Serialize for i32 {\n let N: u32 = I32_SERIALIZED_LEN;\n\n fn serialize(self) -> [Field; Self::N] {\n let mut writer: Writer<Self::N> = Writer::new();\n self.stream_serialize(&mut writer);\n writer.finish()\n }\n\n #[inline_always]\n fn stream_serialize<let K: u32>(self, writer: &mut Writer<K>) {\n writer.write(self as u32 as Field);\n }\n}\n\nimpl Deserialize for i32 {\n let N: u32 = I32_SERIALIZED_LEN;\n\n fn deserialize(fields: [Field; Self::N]) -> Self {\n let mut reader = Reader::new(fields);\n let result = Self::stream_deserialize(&mut reader);\n reader.finish();\n result\n }\n\n #[inline_always]\n fn stream_deserialize<let K: u32>(reader: &mut Reader<K>) -> Self {\n reader.read() as u32 as i32\n }\n}\n\nimpl Serialize for i64 {\n let N: u32 = I64_SERIALIZED_LEN;\n\n fn serialize(self) -> [Field; Self::N] {\n let mut writer: Writer<Self::N> = Writer::new();\n self.stream_serialize(&mut writer);\n writer.finish()\n }\n\n #[inline_always]\n fn stream_serialize<let K: u32>(self, writer: &mut Writer<K>) {\n writer.write(self as u64 as Field);\n }\n}\n\nimpl Deserialize for i64 {\n let N: u32 = I64_SERIALIZED_LEN;\n\n fn deserialize(fields: [Field; Self::N]) -> Self {\n let mut reader = Reader::new(fields);\n let result = Self::stream_deserialize(&mut reader);\n reader.finish();\n result\n }\n\n #[inline_always]\n fn stream_deserialize<let K: u32>(reader: &mut Reader<K>) -> Self {\n reader.read() as u64 as i64\n }\n}\n\nimpl<T, let M: u32> Serialize for [T; M]\nwhere\n T: Serialize,\n{\n let N: u32 = <T as Serialize>::N * M;\n\n fn serialize(self) -> [Field; Self::N] {\n let mut writer: Writer<Self::N> = Writer::new();\n self.stream_serialize(&mut writer);\n writer.finish()\n }\n\n #[inline_always]\n fn stream_serialize<let K: u32>(self, writer: &mut Writer<K>) {\n for i in 0..M {\n self[i].stream_serialize(writer);\n }\n }\n}\n\nimpl<T, let M: u32> Deserialize for [T; M]\nwhere\n T: Deserialize,\n{\n let N: u32 = <T as Deserialize>::N * M;\n\n fn deserialize(fields: [Field; Self::N]) -> Self {\n let mut reader = Reader::new(fields);\n let result = Self::stream_deserialize(&mut reader);\n reader.finish();\n result\n }\n\n #[inline_always]\n fn stream_deserialize<let K: u32>(reader: &mut Reader<K>) -> Self {\n let mut result: [T; M] = std::mem::zeroed();\n for i in 0..M {\n result[i] = T::stream_deserialize(reader);\n }\n result\n }\n}\n\nimpl<T> Serialize for Option<T>\nwhere\n T: Serialize,\n{\n let N: u32 = <T as Serialize>::N + 1;\n\n fn serialize(self) -> [Field; Self::N] {\n let mut writer: Writer<Self::N> = Writer::new();\n self.stream_serialize(&mut writer);\n writer.finish()\n }\n\n #[inline_always]\n fn stream_serialize<let K: u32>(self, writer: &mut Writer<K>) {\n writer.write_bool(self.is_some());\n if self.is_some() {\n self.unwrap_unchecked().stream_serialize(writer);\n } else {\n writer.advance_offset(<T as Serialize>::N);\n }\n }\n}\n\nimpl<T> Deserialize for Option<T>\nwhere\n T: Deserialize,\n{\n let N: u32 = <T as Deserialize>::N + 1;\n\n fn deserialize(fields: [Field; Self::N]) -> Self {\n let mut reader = Reader::new(fields);\n let result = Self::stream_deserialize(&mut reader);\n reader.finish();\n result\n }\n\n #[inline_always]\n fn stream_deserialize<let K: u32>(reader: &mut Reader<K>) -> Self {\n if reader.read_bool() {\n Option::some(<T as Deserialize>::stream_deserialize(reader))\n } else {\n reader.advance_offset(<T as Deserialize>::N);\n Option::none()\n }\n }\n}\n\nglobal SCALAR_SIZE: u32 = 2;\n\nimpl Serialize for EmbeddedCurveScalar {\n\n let N: u32 = SCALAR_SIZE;\n\n fn serialize(self) -> [Field; SCALAR_SIZE] {\n [self.lo, self.hi]\n }\n\n #[inline_always]\n fn stream_serialize<let K: u32>(self, writer: &mut Writer<K>) {\n writer.write(self.lo);\n writer.write(self.hi);\n }\n}\n\nimpl Deserialize for EmbeddedCurveScalar {\n let N: u32 = SCALAR_SIZE;\n\n fn deserialize(fields: [Field; Self::N]) -> Self {\n Self { lo: fields[0], hi: fields[1] }\n }\n\n #[inline_always]\n fn stream_deserialize<let K: u32>(reader: &mut Reader<K>) -> Self {\n Self { lo: reader.read(), hi: reader.read() }\n }\n}\n\nglobal POINT_SIZE: u32 = 2;\n\nimpl Serialize for EmbeddedCurvePoint {\n let N: u32 = POINT_SIZE;\n\n fn serialize(self) -> [Field; Self::N] {\n [self.x, self.y]\n }\n\n fn stream_serialize<let K: u32>(self, writer: &mut Writer<K>) {\n writer.write(self.x);\n writer.write(self.y);\n }\n}\n\nimpl Deserialize for EmbeddedCurvePoint {\n let N: u32 = POINT_SIZE;\n\n fn deserialize(fields: [Field; Self::N]) -> Self {\n Self { x: fields[0], y: fields[1] }\n }\n\n #[inline_always]\n fn stream_deserialize<let K: u32>(reader: &mut Reader<K>) -> Self {\n Self { x: reader.read(), y: reader.read() }\n }\n}\n\nimpl<let M: u32> Deserialize for str<M> {\n let N: u32 = M;\n\n fn deserialize(fields: [Field; Self::N]) -> Self {\n let mut reader = Reader::new(fields);\n let result = Self::stream_deserialize(&mut reader);\n reader.finish();\n result\n }\n\n #[inline_always]\n fn stream_deserialize<let K: u32>(reader: &mut Reader<K>) -> Self {\n let u8_arr = <[u8; Self::N] as Deserialize>::stream_deserialize(reader);\n str::<Self::N>::from(u8_arr)\n }\n}\n\nimpl<let M: u32> Serialize for str<M> {\n let N: u32 = M;\n\n fn serialize(self) -> [Field; Self::N] {\n let mut writer: Writer<Self::N> = Writer::new();\n self.stream_serialize(&mut writer);\n writer.finish()\n }\n\n #[inline_always]\n fn stream_serialize<let K: u32>(self, writer: &mut Writer<K>) {\n self.as_bytes().stream_serialize(writer);\n }\n}\n\n// Note: Not deriving this because it's not supported to call derive_serialize on a \"remote\" struct (and it will never\n// be supported).\nimpl<T, let M: u32> Deserialize for BoundedVec<T, M>\nwhere\n T: Deserialize,\n{\n let N: u32 = <T as Deserialize>::N * M + 1;\n\n fn deserialize(fields: [Field; Self::N]) -> Self {\n let mut reader = Reader::new(fields);\n let result = Self::stream_deserialize(&mut reader);\n reader.finish();\n result\n }\n\n #[inline_always]\n fn stream_deserialize<let K: u32>(reader: &mut Reader<K>) -> Self {\n let mut new_bounded_vec: BoundedVec<T, M> = BoundedVec::new();\n let payload_len = Self::N - 1;\n\n // Length is stored in the last field as we need to match intrinsic Noir serialization and the `len` struct\n // field is after `storage` struct field (see `bounded_vec.nr` in noir-stdlib)\n let len = reader.peek_offset(payload_len) as u32;\n\n for i in 0..M {\n if i < len {\n new_bounded_vec.push(<T as Deserialize>::stream_deserialize(reader));\n }\n }\n\n // +1 for the length of the BoundedVec\n reader.advance_offset((M - len) * <T as Deserialize>::N + 1);\n\n new_bounded_vec\n }\n}\n\n// This may cause issues if used as program input, because noir disallows empty arrays for program input.\n// I think this is okay because I don't foresee a unit type being used as input. But leaving this comment as a hint\n// if someone does run into this in the future.\nimpl Deserialize for () {\n let N: u32 = 0;\n\n fn deserialize(fields: [Field; Self::N]) -> Self {\n let mut reader = Reader::new(fields);\n let result = Self::stream_deserialize(&mut reader);\n reader.finish();\n result\n }\n\n #[inline_always]\n fn stream_deserialize<let K: u32>(_reader: &mut Reader<K>) -> Self {\n ()\n }\n}\n\n// Note: Not deriving this because it's not supported to call derive_serialize on a \"remote\" struct (and it will never\n// be supported).\nimpl<T, let M: u32> Serialize for BoundedVec<T, M>\nwhere\n T: Serialize,\n{\n let N: u32 = <T as Serialize>::N * M + 1; // +1 for the length of the BoundedVec\n\n fn serialize(self) -> [Field; Self::N] {\n let mut writer: Writer<Self::N> = Writer::new();\n self.stream_serialize(&mut writer);\n writer.finish()\n }\n\n #[inline_always]\n fn stream_serialize<let K: u32>(self, writer: &mut Writer<K>) {\n self.storage().stream_serialize(writer);\n // Length is stored in the last field as we need to match intrinsic Noir serialization and the `len` struct\n // field is after `storage` struct field (see `bounded_vec.nr` in noir-stdlib)\n writer.write_u32(self.len() as u32);\n }\n}\n\n// Create a slice of the given length with each element made from `f(i)` where `i` is the current index\ncomptime fn make_slice<Env, T>(length: u32, f: fn[Env](u32) -> T) -> [T] {\n let mut slice = @[];\n for i in 0..length {\n slice = slice.push_back(f(i));\n }\n slice\n}\n\n// Implements Serialize and Deserialize for an arbitrary tuple type\ncomptime fn impl_serialize_for_tuple(_m: Module, length: u32) -> Quoted {\n // `T0`, `T1`, `T2`\n let type_names = make_slice(length, |i| f\"T{i}\".quoted_contents());\n\n // `result0`, `result1`, `result2`\n let result_names = make_slice(length, |i| f\"result{i}\".quoted_contents());\n\n // `T0, T1, T2`\n let field_generics = type_names.join(quote [,]);\n\n // `<T0 as Serialize>::N + <T1 as Serialize>::N + <T2 as Serialize>::N`\n let full_size_serialize = type_names\n .map(|type_name| quote {\n <$type_name as Serialize>::N\n })\n .join(quote [+]);\n\n // `<T0 as Deserialize>::N + <T1 as Deserialize>::N + <T2 as Deserialize>::N`\n let full_size_deserialize = type_names\n .map(|type_name| quote {\n <$type_name as Deserialize>::N\n })\n .join(quote [+]);\n\n // `T0: Serialize, T1: Serialize, T2: Serialize,`\n let serialize_constraints = type_names\n .map(|field_name| quote {\n $field_name: Serialize,\n })\n .join(quote []);\n\n // `T0: Deserialize, T1: Deserialize, T2: Deserialize,`\n let deserialize_constraints = type_names\n .map(|field_name| quote {\n $field_name: Deserialize,\n })\n .join(quote []);\n\n // Statements to serialize each field\n let serialized_fields = type_names\n .mapi(|i, _type_name| quote {\n $crate::serialization::Serialize::stream_serialize(self.$i, writer);\n })\n .join(quote []);\n\n // Statements to deserialize each field\n let deserialized_fields = type_names\n .mapi(|i, type_name| {\n let result_name = result_names[i];\n quote {\n let $result_name = <$type_name as $crate::serialization::Deserialize>::stream_deserialize(reader);\n }\n })\n .join(quote []);\n let deserialize_results = result_names.join(quote [,]);\n\n quote {\n impl<$field_generics> Serialize for ($field_generics) where $serialize_constraints {\n let N: u32 = $full_size_serialize;\n\n fn serialize(self) -> [Field; Self::N] {\n let mut writer: $crate::writer::Writer<Self::N> = $crate::writer::Writer::new();\n self.stream_serialize(&mut writer);\n writer.finish()\n }\n\n #[inline_always]\n fn stream_serialize<let K: u32>(self, writer: &mut $crate::writer::Writer<K>) {\n\n $serialized_fields\n }\n }\n\n impl<$field_generics> Deserialize for ($field_generics) where $deserialize_constraints {\n let N: u32 = $full_size_deserialize;\n\n fn deserialize(fields: [Field; Self::N]) -> Self {\n let mut reader = $crate::reader::Reader::new(fields);\n let result = Self::stream_deserialize(&mut reader);\n reader.finish();\n result\n }\n \n #[inline_always]\n fn stream_deserialize<let K: u32>(reader: &mut $crate::reader::Reader<K>) -> Self {\n $deserialized_fields\n ($deserialize_results)\n }\n }\n }\n}\n\n// Keeping these manual impls. They are more efficient since they do not\n// require copying sub-arrays from any serialized arrays.\nimpl<T1> Serialize for (T1,)\nwhere\n T1: Serialize,\n{\n let N: u32 = <T1 as Serialize>::N;\n\n fn serialize(self) -> [Field; Self::N] {\n let mut writer: crate::writer::Writer<Self::N> = crate::writer::Writer::new();\n self.stream_serialize(&mut writer);\n writer.finish()\n }\n\n #[inline_always]\n fn stream_serialize<let K: u32>(self, writer: &mut Writer<K>) {\n self.0.stream_serialize(writer);\n }\n}\n\nimpl<T1> Deserialize for (T1,)\nwhere\n T1: Deserialize,\n{\n let N: u32 = <T1 as Deserialize>::N;\n\n fn deserialize(fields: [Field; Self::N]) -> Self {\n let mut reader = crate::reader::Reader::new(fields);\n let result = Self::stream_deserialize(&mut reader);\n reader.finish();\n result\n }\n\n #[inline_always]\n fn stream_deserialize<let K: u32>(reader: &mut Reader<K>) -> Self {\n (<T1 as Deserialize>::stream_deserialize(reader),)\n }\n}\n\n#[impl_serialize_for_tuple(2)]\n#[impl_serialize_for_tuple(3)]\n#[impl_serialize_for_tuple(4)]\n#[impl_serialize_for_tuple(5)]\n#[impl_serialize_for_tuple(6)]\nmod impls {\n use crate::serialization::{Deserialize, Serialize};\n}\n\n#[test]\nunconstrained fn bounded_vec_serialization() {\n // Test empty BoundedVec\n let empty_vec: BoundedVec<Field, 3> = BoundedVec::from_array([]);\n let serialized = empty_vec.serialize();\n let deserialized = BoundedVec::<Field, 3>::deserialize(serialized);\n assert_eq(empty_vec, deserialized);\n assert_eq(deserialized.len(), 0);\n\n // Test partially filled BoundedVec\n let partial_vec: BoundedVec<[u32; 2], 3> = BoundedVec::from_array([[1, 2]]);\n let serialized = partial_vec.serialize();\n let deserialized = BoundedVec::<[u32; 2], 3>::deserialize(serialized);\n assert_eq(partial_vec, deserialized);\n assert_eq(deserialized.len(), 1);\n assert_eq(deserialized.get(0), [1, 2]);\n\n // Test full BoundedVec\n let full_vec: BoundedVec<[u32; 2], 3> = BoundedVec::from_array([[1, 2], [3, 4], [5, 6]]);\n let serialized = full_vec.serialize();\n let deserialized = BoundedVec::<[u32; 2], 3>::deserialize(serialized);\n assert_eq(full_vec, deserialized);\n assert_eq(deserialized.len(), 3);\n assert_eq(deserialized.get(0), [1, 2]);\n assert_eq(deserialized.get(1), [3, 4]);\n assert_eq(deserialized.get(2), [5, 6]);\n}\n"
|
|
3142
3142
|
},
|
|
3143
|
-
"
|
|
3143
|
+
"415": {
|
|
3144
3144
|
"function_locations": [
|
|
3145
3145
|
{
|
|
3146
3146
|
"name": "Writer<N>::new",
|
|
@@ -4811,7 +4811,7 @@
|
|
|
4811
4811
|
"abi_public",
|
|
4812
4812
|
"abi_only_self"
|
|
4813
4813
|
],
|
|
4814
|
-
"debug_symbols": "
|
|
4814
|
+
"debug_symbols": "tZfbbtswDIbfxde5EEXqwLzKMBRp6hYBDCdwkwFDkXcf1Yg+FJDg1dtN8pm2flP0T1n+aF7a59vb06l/Pb83+x8fzfNw6rrT21N3Ph6up3Mv0Y/GpB8A3+xx14CFZh/SvxwDCKAEIJ1Bn4FMPkUug0sRl8Bl8BrxMUNAhSDgBaLNwEZBhlu5hTVGgTOgRpAU9BRphDTiNOJiBm8VQoYACj5D1FtETSMmQcnZMirEB6BBBY2ARkAjNqXhE3AG1Aima6IAoYJGnEY8KKRnkwSDUXAZokaiRlgjTArxAZRyfkDIAFYh34LS839AEgwJOENK/gEaIY2QRpxGkgEwJnAKcg0ZgUAKGokhA0s+RAlSJNzvu0a9+nQd2jZZdWZesfTlMLT9tdn3t67bNb8O3e3zovfLof/8vx4GOSvabf8i/yL4euraRPfdNNqUh4rzXB5t0bpRQDpmIQEVCUPOqoYh70cRHxcatqIRoiowTUkEWj0PH70qhBiL86CyBBJpKVCe6ygR7ELB/YNK+P9YCQSjOSBAKFYiliWkcbSYJIteqRJcmYUMG+cBLkCpElCbCbFOxJdLUc9icgV4jMUsKtYka6Za8KjgvlXM2Ty+FBMqzmRQY7Kd5kDwLUtguckrEs56zcFZnlzllisNhIqEcaMEmCkLa5a1hEox5S2mE+GpEvJWWCpUPMHTgsdIdtJwYblYVYrBhlA1DJuyBtTMrRI4W67s1yWztmZCmMwd6XsaFjUNm8YVNahm8DB2iDWzBWd9GsgBxucauJhG3V6GR3uBL9rLVjTIcBh7FaiosbJRpO9KjWJ5e6Og2dooCNsbBe32Rkkm3NYoSNsbpaqxslHQb26UWhprG6Vur3WNghUN2SRrHkRmVo4vc1nbKFhsFKpYNLDRBxsYbTELqi09BGYsqTR9cadCFY86mF6O8277mzSMpzENE4tbR6rsP+UDDLUgwmyLIhWXoo/jZiFYXO67fsrh4XgaFt/Y9yQ2nA7PXZsPX2/9cXb2+vuiZ/Qb/TKcj+3LbWiT0vxDXT572O04/Lynu/0B",
|
|
4815
4815
|
"is_unconstrained": true,
|
|
4816
4816
|
"name": "_set_authorized"
|
|
4817
4817
|
},
|
|
@@ -4867,7 +4867,7 @@
|
|
|
4867
4867
|
"custom_attributes": [
|
|
4868
4868
|
"abi_public"
|
|
4869
4869
|
],
|
|
4870
|
-
"debug_symbols": "
|
|
4870
|
+
"debug_symbols": "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",
|
|
4871
4871
|
"is_unconstrained": true,
|
|
4872
4872
|
"name": "consume"
|
|
4873
4873
|
},
|
|
@@ -4920,7 +4920,7 @@
|
|
|
4920
4920
|
"abi_public",
|
|
4921
4921
|
"abi_view"
|
|
4922
4922
|
],
|
|
4923
|
-
"debug_symbols": "
|
|
4923
|
+
"debug_symbols": "tZhNbtswEIXvorUXHM4PyVylKAIncQoDhhO4doEi8N3LiTiSbYCMWrqb6PNIehpK74mMPoaXzdPpx+N2//r2c3j49jE8Hba73fbH4+7teX3cvu1z9WNw+kfi8ICrIeDwEPIm/wK3GmL+CZC3cdwmGbfgIENUkALABdAq5AxsF1uFrSJWETKIBbSVEUKB6A3sEsnaSFnQO4U0gndkYBWwCljFWwVzGx4UuABZhfQYVEgF2CpsFbFK8Aa5Z58yaM+oOikLou7SVhXQOQOtUAZwBlrhDN4Z5Ar5DNrqCGSQCpBVyCpsFY4FxBuEAgEMpEC0S0S7aFRBbSyhQRyBHBpYBawCVtH7TKyQCqAeExRiAbIKg0Huh/V0NckIXCBYJVglWiWSQSygPY8QRmDnDcolGMBABUUhFdDmR7AKWgWtQlZRh3NQYAM9JjuB1SQjWEVN8glqEmGFXBE9S00ieZeoSUawipokOAUuoCYJoMAG+RLBn8+rwZL+eDxsNhr0i+jnF8L7+rDZH4eH/Wm3Ww2/1rvT50E/39f7z+1xfch780U2+5e8zYKv291G6byaz3b1U3O0uJzts4knAQC5koCGhCP2puFIZBKReKXhGxohmkKiuYlAi8chUUwhxFgdB9Ulcj7tVmC28iTx+aBmBb7DnZD/eCcQQ5iG4ah6J2JDAhxZF5kl1caR6hrgpgcC4FL/SKQ6koYEx8neHBEmCb4OCDSsyY4nCXCzLXzmKw1sPNRoEc0v5kkB000XDXOmOacJyc8aHK41GvZMjtA0XHJ1DWmFxCTwImX+xhcQGvcCwpRUiPRvGh6tDa/nVTUaBiUI9lDIuzC7a3kbmPRtXp5rSPU2mvZyabIXSNVevqFBmKwPyvP8rHEzlqVBYVcLim9YVCBaXMWDr3bhGxZFL8HeoJkTVueSlkfzKpMmEQdQFQn3mNVi78v8i7FQmsdyMUPftIGtFynz5FNxvt5I+8nI/GTChctuG/Eti0wOwYt3WPobBZkUQq8C1XtoRkXCFJWUalFB6Z9TMPTOKRj75xRM/XMKud45haB/TmlqLJxTCLvnlFYbS+eUtr2WzSnU0CCXzOUEQFWNhUFJF23cBIVif1Ao9QaFXX9QGPqDwr43KIz9QWlqLAwKc3dQWm0sDUrbXsuCwvE/L74ugoJcC4q4/sWXwB0WX+LvsPgSvMc/0tS7+PpiLMsWXyL9i68vnsyyxZfE3sVXW2HJ4muhwr8uvhzNUQlXUfmef62ft4erz+dn1Tps10+7Tfn5eto/X+w9/n63Pfb5/f3w9rx5OR02qjR/g9ePkN8gr/cB3fezXu8P",
|
|
4924
4924
|
"is_unconstrained": true,
|
|
4925
4925
|
"name": "is_consumable"
|
|
4926
4926
|
},
|
|
@@ -4966,7 +4966,7 @@
|
|
|
4966
4966
|
"abi_public",
|
|
4967
4967
|
"abi_view"
|
|
4968
4968
|
],
|
|
4969
|
-
"debug_symbols": "
|
|
4969
|
+
"debug_symbols": "tZbbbuIwEIbfJde58Bw8tnmVqqpSmlaRooBSWGlV8e5r00wOSPZ2l5YL/GdMPmY8v518VC/t8/ntqRteD+/V7uGjeh67vu/envrDvjl1hyFGPyqTvoSrHdWVhGrn6srFKzBxjJcAdeX5cww4jS6Ovq7AyCQAJkGoQqdYp1gjViNWI2JUsIowCUcq/CS8/oXXNEIEoknCfgo0RoVGQCOgEdQIxjQwloUEKjTC6TeUhJ2E1YjViGhE/CRSzhiSiBFKHB+BlKZSqp9iipBJEU5CJpEWk2wSMgm8Ri6XutLWPZ3Gtk2dW/UydvjYjO1wqnbDue/r6lfTn68/ej82w3U8NWOcjUW3w0scI/C169ukLvVyt8nfimTsdDcS2hkAIBsEFBCGLSrDsMgMEb9hYIHhvBICL0k4/nId4kUJzvtsHZxHELMuBbEsS+FwQ7DfsBLygytB5NxchuHsSvgCAgxrFlFLyNUR8gwwc0MATLi/EslWUkBYm/btFWHjZ0bY7QaBgjWtsWoLC2ZBxKNoy6BCU71uUQqLNyncZFEwZ1j2aSDGhWHdllGwZzBMyjDB5BlS2iSKoNUuwxtfgCusBbh5p4Ln/2MgaRqY7ssyCgZlcNoURuMWd309DQoO5r66kE+jaC8zOzQZO2cvLDCYgubBbFbLcVPLVzeK49xGwYJFJT7AJ4QgYDYLLFiUUJyeoFEHyj5LSh6FEHiGGIAsxH3HU83fe5j/pRYOSy2rJ/RNGlQ6SK2dfSoG84mUOyNLZ9zKZbeJYMkis0NodYaFfyHITHD3Evgmh8d42ey7cfNCfUmosWue+3a6fD0P+9Xs6fdRZ/SF/Dge9u3LeWwTaXkrTy+CD8K1yOMl/dsf",
|
|
4970
4970
|
"is_unconstrained": true,
|
|
4971
4971
|
"name": "is_reject_all"
|
|
4972
4972
|
},
|
|
@@ -5000,7 +5000,7 @@
|
|
|
5000
5000
|
"custom_attributes": [
|
|
5001
5001
|
"abi_public"
|
|
5002
5002
|
],
|
|
5003
|
-
"debug_symbols": "
|
|
5003
|
+
"debug_symbols": "tVdbbtswELyLvv3BffDlqxRF4DhKYECwDcUuUAS+e5c2Vw8DJNSo/bFGI2k0XM7S4lfz1r5eP14Ox/fTZ7P98dW89oeuO3y8dKf97nI4HYX9akz6AcBmSxs5+mbr5YhyDpCAEJCuECpw+RKDgsRYARYUKONYQczAU7NFTCBkEPwDoMkMJjcPoJdQGVSGlEl+7oCNAqsgZmBZgb7CkYIkyAI8KHAZBGWCMlGZmBkyyYZNwGcAykC6R4pJCAqUIWUoZsDikJIghwwsKVDGKeOU8ajAZZA8P4DNIBoF+ooYH4BNEnQJ+AzuObgDZVAZVIaUoWRDzDOTgnRPFGBRgTLOZuDFD1MCwrC73TaNBvLl0rdtyuMkoZLb865vj5dme7x23ab5teuu95s+z7vj/XjZ9XLVbJr2+CZHEXw/dG1Ct834tCk/imRsflpyZQcBADeTgIqEkTGrhmHnBhEXZhpY0fBBFSKPJjwvHocLThV8CMVxcFlCgqilIJmzQcLjTMH+g0q4/1kJcn6Y0YCDhJ1nIpQVpBW0lgy2WIhY8SCPDcMA66FUCDC1+Yg6Ha5cibqLMRTgKBRdVJLJaMZaxLGa3yrmZBxPxYRKMCNoLiOOY2BYnoi05OVETEL5lAiopNIaqx4smDESaOaFAF+xEbSSFMdhyCI9V6jUMo6LVSQeo03WzzUqoYgmrcAPDRNNUQOryVQJmiw1+Lzc1dZM8GMyA39PQ+ZVNdJzRQ2qpdMP8Ubjxy5bboOih2FefSzaqMfLxCFe4IrxwooGm+iHRgMuaixtlMnMPjUKhvWNgnFto5BZ3ygE6xuFcG2jEK1vlKrGwkYhu7pRajaWNko9XssahWr/SBTVB7OZlONpLAsbZSrx1ChckfDR6MT6SFh0wbWlh8EMJZWmL35mcCWjdvh3tTjttr+xYRwPNkwofvZx7etTekULIjhiUaSSUtke6fenbJBo/tH0U053+0M/2wTfklh/2L12bT59vx73k6uX32e9opvoc3/at2/Xvk1K0520bFlC2ETz85be9gc=",
|
|
5004
5004
|
"is_unconstrained": true,
|
|
5005
5005
|
"name": "set_authorized"
|
|
5006
5006
|
},
|
|
@@ -6819,11 +6819,11 @@
|
|
|
6819
6819
|
"visibility": "databus"
|
|
6820
6820
|
}
|
|
6821
6821
|
},
|
|
6822
|
-
"bytecode": "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",
|
|
6822
|
+
"bytecode": "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",
|
|
6823
6823
|
"custom_attributes": [
|
|
6824
6824
|
"abi_private"
|
|
6825
6825
|
],
|
|
6826
|
-
"debug_symbols": "
|
|
6826
|
+
"debug_symbols": "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",
|
|
6827
6827
|
"is_unconstrained": false,
|
|
6828
6828
|
"name": "set_authorized_private",
|
|
6829
6829
|
"verification_key": "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"
|
|
@@ -6851,7 +6851,7 @@
|
|
|
6851
6851
|
"custom_attributes": [
|
|
6852
6852
|
"abi_public"
|
|
6853
6853
|
],
|
|
6854
|
-
"debug_symbols": "
|
|
6854
|
+
"debug_symbols": "tVbtbuMgEHwX/84P9gMW8iqnqkpTt7JkOZGbnHSq8u4HLWvsSKC2uvsTj5cwHmYH7PfuuX+6vj4O08vprdv/eu+e5mEch9fH8XQ8XIbTFKvvnUk/YHy3p10HQN1e0jXeA0SAsQBpBH0GhHmIJANOFZuAZGC14owCqyB0e8QIxGfg6ROgYQV5CEGHQCuoFdRK0vMJXAYMCmwG1ijQR9gsA10i5AQkA0EFWvFa8VoJWglJRlwXGVKgFUj/kQQkA9QKaoVsBhwVEiQQK8S3267TNj1e5r5PXVr1LXbzfJj76dLtp+s47rrfh/H68ae382H6uF4Ocxw1u66fnuM1Er4MY5/QbVdmm/pUJGPz7GiyXQgA3IYCGhSGLSqHYecWEuc3HNjgEK8MgYsI4S+vw3mnDOJ9dR1cpyBmtYLYFSsENwz2Hzjh/qcTAktHhUok7DYTvs7AgOolg60aERoa4rRlGWAFakaAafUjaDtc3Ym2ihIKcOSrKhrJZDTFi1Dc/JGZq3XcmQmNYIaloQHLGhi+sTeWZghwLRHQSKU1VjVYMCUSaLZGgDRkeHWSQlkGhTsVDS9DOawCMRYOK1uORiiCYVIOE0yVA5vJVApaHTV4f9y1zkyQkkzPP+NAUhmY5lU5qJVOWeKNRsou+7oMCgJLXyVUZbTjZcISL3DVeGGDgymoDmazsuNuLV/dKBRqGwUbEZVgtLESCKsqMLTiBWaxlAGqpyc11mKXQ8Pierd9R4ZxvMgwvvo2o8bxSXGvqCERB6ySNFJKzutrlQRp+y54iLeH4zBvvnhviWweDk9jn29frtNxNXr5c9YR/WI+z6dj/3yd+8S0+myOH5q/HOwcPdzS0/4C",
|
|
6855
6855
|
"is_unconstrained": true,
|
|
6856
6856
|
"name": "set_reject_all"
|
|
6857
6857
|
},
|
|
@@ -6903,11 +6903,11 @@
|
|
|
6903
6903
|
"visibility": "public"
|
|
6904
6904
|
}
|
|
6905
6905
|
},
|
|
6906
|
-
"bytecode": "H4sIAAAAAAAA/+
|
|
6906
|
+
"bytecode": "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",
|
|
6907
6907
|
"custom_attributes": [
|
|
6908
6908
|
"abi_utility"
|
|
6909
6909
|
],
|
|
6910
|
-
"debug_symbols": "
|
|
6910
|
+
"debug_symbols": "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",
|
|
6911
6911
|
"is_unconstrained": true,
|
|
6912
6912
|
"name": "utility_is_consumable"
|
|
6913
6913
|
},
|
|
@@ -7045,7 +7045,7 @@
|
|
|
7045
7045
|
"custom_attributes": [
|
|
7046
7046
|
"abi_utility"
|
|
7047
7047
|
],
|
|
7048
|
-
"debug_symbols": "
|
|
7048
|
+
"debug_symbols": "tZndbhs5DEbfxde+GFI/FPMqQVG4qbswYDiBmyywKPLuS2b0acZZSPDa7U10kniOKQ4paexfm+/7b29/fT2cfjz/3Dw8/tp8Ox+Ox8NfX4/PT7vXw/PJ/vprM/mPzGXzELY26uYh2ximOlIduY6hjrGOqY65jlLHUsfqi9UXqy9WX7LXiY3Z/q4+ch3NT+SQAPYO5CFmARSAVpAJQAAGBEAEJADMArPALDAXmAvMBeYCc4G5wFxgLjAXmAvMCrPCrDArzAqzVrNMflVyIAAD/DXZIQMEYO/Ok4NWoAlAAAYEgL07++WUABkgADeLg1bgCeBmdTBz8OA5ACIgATJAAAWgFbw2ZyAAzAHmALMXaPC0eIXOIAA3e4RepB/gVTqDX+4xR3txJAd7cXRh1AppAlgYMTowIAAiIAEyQABu9niSVsgTgABu9sByAESAm4tDBgigALSCN8gMBHCzz9QbZIYISAAzJ0+CN8gMBWDmxAbeIDMQgAEBEAEJ4GbPoTfIDAWgFbxBkifKGyR5sXmDzBAAbvZseIPMkAECKACdoXgTzeDm4sCAAIgAX+kmhwwQgC925KAVvK0yOxCAAQHgS2h2cLO/hbfVDAIoAK3gbTUDAdysDgFgZvEwvK1myAAzi7+7t9UMWsHbagYCMCAA3BwcEiADBODm6KAVvK1mIIBflRwEUAB+lc/L+2sGAng8PkHvrxkiIAEyQAAFYObi98v7awYCMMDN+f19u8Em+fX1vN/7HrnaNW0vfdmd96fXzcPp7Xjcbv7eHd8+XvTzZXf6GF93Z/uvKfen7zaa8MfhuHd63y5XT/1LabJFrF5uLNQUNF1KqC+Jvjh/KGKUJpDL67l/fUiYQZCyBFDo+llYl7VZlJK6s4h9ScocqiPZSWJRpHihSH0FZ+aqYDt8NEW+OgbbKhGD7VLdGAYKOzRJVdgsYldR+grb06vBlvJFIHr9NFoMyTacbgw0uKeBSVEVHFInl2ODtroKU+kZBoUZiWGIFPSG+2mLIxJRYugnYuBQgkJ5mUSkq2PQhA7PE8ebamoKiyL2FSSDHtWMqiKVxZEuq4oGdUlCaC+yU0JXoaNkIggtq6XGlp2rDbqsl5S6Ch6sl8wRDubUvafMoxWvrTQ0JeqHEYbFya04V10a6fKu8rAyJl2Ka+o7RqtmTG3RlJWBbisM7ReGjO5JoHZPVp36ScGDKHJKLRdJ401RUMbCafF0owiDVS+mthlLvy7CaDunzG3V0/U95UvHqD5TuyEprAzp+ii4VVZkkX4Ug+okLtjSKRD14xg4goS2D5Wp9OPIf9ZBmWLLaSYJ3V4LgxrVpMiIrmv0P45hJGFZA+2zln7Xx9HunJdDX+zemaEhtqRerF6fDKP9fQpYeuyoJP15DBwcMA9OfFMUXNr6V3K/RmV0dJ3aSm5l0p9H/g25yHfnIt+di3x/LtJ0fy5GjutyMTTcnQsLvrR5ZLqp26/ORfqzjivzme7O53BP0jaPMFjHk/6GPWlUn4KVL60fVD+fYkcHjVBwTOD1U+L/OatEaqfYtDqCflLkcO9ZJY9250nR7ZbM/p6Y0+g5MS9PmoVvdLSpmK70HXJvhQ/zeVVdjI7z3NJpX870n1dlUJz2BQdSYd9shM5D89BQuH2EYZ/U3WSQ0gxyw4O7fQUVWiJC/4l3eNbK7QSrmfJt57W1I/TrW37Djihy/woud9f30NBfwb/Yr7unw/nia8d3V50Pu2/Hff31x9vpafXf139e8B98bflyfn7af3877920fHdpPx6LfapclL9sN/b57aOdwbeZxX7zj98fi+pWJ/ry7rH8Cw==",
|
|
7049
7049
|
"is_unconstrained": true,
|
|
7050
7050
|
"name": "offchain_receive"
|
|
7051
7051
|
},
|
|
@@ -7101,7 +7101,7 @@
|
|
|
7101
7101
|
"custom_attributes": [
|
|
7102
7102
|
"abi_public"
|
|
7103
7103
|
],
|
|
7104
|
-
"debug_symbols": "
|
|
7104
|
+
"debug_symbols": "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",
|
|
7105
7105
|
"is_unconstrained": true,
|
|
7106
7106
|
"name": "public_dispatch"
|
|
7107
7107
|
},
|
|
@@ -7234,7 +7234,7 @@
|
|
|
7234
7234
|
"custom_attributes": [
|
|
7235
7235
|
"abi_utility"
|
|
7236
7236
|
],
|
|
7237
|
-
"debug_symbols": "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Ka96yque8qqnvOopr3rKq57yqp95Nfez/cyrg0viw9WCLbEnPlw9uCXuiQ/X3If2M6MOLoklsSa2xJ64Jm6Je2K848iomNczjiyar4et48ii+XGeOo4sOrkmbol74gEfWXRySSyJD68FW2JPXBO3xD3xgI8sOrkklsTJK8krySvJK8krySvJq8mryavJq8mryavJq8mryavJq8lryWvJa8lryWvJa8lryWvJa8lryevJ68nryevJ68nryevJ68nryevJW5O3Jm9N3pq8NXlr8tbkrclbk7cmb0velrwteVvytuRtyduStyVvS96WvD15e/L25O3J25O3J29P3p68PXl78o7kHck7knck70jekbwjeUfyjuQdy9sej0fiklgSa2JL7Ilr4pa4J07ekrwleUvyluQtyVuStyRvSd6SvCV5JXkleSV5JXkleSV5JXkleSV5JXk1eTV5NXk1eTV5NXk1eTV5NXk1eS15LXkteS15LXkteS15LXkteS15PXk9eT15PXk9eT15PXk9eT15PXnPvOrBJbEkPlwj2BPXxC1xTzzgM6MOLoklcfzGOR+yPY6MOtkT18QtcU884COjTi6JJXHy9uTtyduTtydvT96evCN5R/KO5B3JO5J3JO9I3pG8I3kH3vJ4JC6JJbEmtsSeuCZuiXvi5C3JW5K3JG9J3pK8JXlL8pbkLclbkleSV5JXkleSV5JXkleSV5JXkleSV5NXk1eTV5NXk1eTV5NXk1eTV5PXkteS15LXkteS15LXkteS15LXkteT15PXk9eT15PXk9eT15PXk9eTtyZvTd6avDV5a/LW5K3JW5O3Jm9N3pa8LXlb8qa8KimvSsqrkvKqpLwqKa9KyquS8qqkvCopr0rKq5LyqqS8KimvSsqrkvKqpLwqKa9KyquS8qqkvCopr0rKq5LyqqS8KimvSsorSXklKa8k5ZWkvJKUV5LySlJeScorSXklKa8k5ZWkvJKUV5LySlJeScorSXklKa8k5ZWkvJKUV5LySlJeScorSXklKa8k5ZWkvJKUV5LySlJeScorSXklKa8k5ZWkvJKUV5LySlJeScorSXklKa8k5ZWkvJKUV5LySlJeScorSXklKa8k5ZWkvJKUV5LySlJeScorSXklKa8k5ZWkvJKUV5LySlJeScorSXklKa8k5ZWkvJKUV5LySlJeScorSXklKa8k5ZWkvJKUV5LySlJeScorSXklKa8k5ZWkvJKUV5LySlJeScorSXklKa8k5ZWkvJKUV5LySlJeScorSXklKa/kzCsPHov1zKuDwzVnXjc9MmrOt256ZNTJnrgmbol74gEfGXVySSyJk7ckb0nekrwleUvyHlk07yc2PbLl5Ja4Jz6Wcx6365EtJ5fEklgTW2JPXBO3xD1x8lryWvJa8lryHtliEuyJa+LwzrnwTY9sOXnAR7Z4/P2RLSdLYk1siT1xTdwS98QDrslbk7cmb03emrw1eWvy1uStyVuTtyVvS96WvC15W/K25G3J25K3JW9L3p68PXl78vbk7cnbk7cnb0/enrxHtsxHEZoe2XLykS2xzcdnEduBDlbwKB6BcATIyWOxHQFyckl8/KgarIktsSeuiVvinnjAR5icXBInb0nekrwleUvyluQtyVuSV5JXkleSV5JXkleSV5JXkleSV5JXk1eTV5NXk1eTV5NXk1eTV5NXk9eS15LXkteS15LXkteS15LXkteS15PXk9eT15PXk9eT15PXk9eT15O3Jm9N3pq8NXlr8tbkrclbk7cmb03elrwteVvytuRtyduStyVvS94zZFrw8ffzgpgdBxjztefNjgOMky2xJ47688WNzY58OLknjt9VZ+D4kQ8nl8SSWBNbYk98eC24Je6JB3xkwpw13fzo8TlBuvnR4yf3xAM+enzOl25+9PjJkvhY5hFsiT1xTdwS98ThnXPMmh893mKZjx5vEhzeFuNw9G+L33j078k98YCP/m3xu44+nTOzmx89Ejl/fOE8NofjC+cnHtaDNbEl9sQ1cUvcEw/42JpbjMKxNfcYhWNrPrkmbol74gEfu9geI3jsYk+WxJrYEnvimrjBx+6zx1o5OudkTXzUjLV1dM7JNXFL3BOPxfXonJNL4qOmBdfELfFR04MHfOw1Ty6JJbEmtsSe+Kg5t6R6dMvJJfFRswVrYkvsiWvilrgnPrxz+6lHt8x3CbV6dMvJklgTW2JPHN45pbzVo6NO7okHfHTUySWxJNbEnTGxNG6exu3Ywx2/0dO4eRo3T+Pmadw8jZuncTv2cMdYHXu44/fWNG41jVtN41bTuNU0bke/H7+rpnGradxqGreaxq2lcWtp3Foat6PH50T+dk54nRP12znh9eCjx08uiY860SNHj59siT1xTdwSH97o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",
|
|
7237
|
+
"debug_symbols": "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Ka96yque8qqnvOopr3rKq57yqp95Nfez/cyrg0viw9WCLbEnPlw9uCXuiQ/X3If2M6MOLoklsSa2xJ64Jm6Je2K848iomNczjiyar4et48ii+XGeOo4sOrkmbol74gEfWXRySSyJD68FW2JPXBO3xD3xgI8sOrkklsTJK8krySvJK8krySvJq8mryavJq8mryavJq8mryavJq8lryWvJa8lryWvJa8lryWvJa8lryevJ68nryevJ68nryevJ68nryevJW5O3Jm9N3pq8NXlr8tbkrclbk7cmb0velrwteVvytuRtyduStyVvS96WvD15e/L25O3J25O3J29P3p68PXl78o7kHck7knck70jekbwjeUfyjuQdy9sej0fiklgSa2JL7Ilr4pa4J07ekrwleUvyluQtyVuStyRvSd6SvCV5JXkleSV5JXkleSV5JXkleSV5JXk1eTV5NXk1eTV5NXk1eTV5NXk1eS15LXkteS15LXkteS15LXkteS15PXk9eT15PXk9eT15PXk9eT15PXnPvOrBJbEkPlwj2BPXxC1xTzzgM6MOLoklcazjnA/ZHkdGneyJa+KWuCce8JFRJ5fEkjh5e/L25O3J25O3J29P3pG8I3lH8o7kHck7knck70jekbwDb3k8EpfEklgTW2JPXBO3xD1x8pbkLclbkrckb0nekrwleUvyluQtySvJK8krySvJK8krySvJK8krySvJq8mryavJq8mryavJq8mryavJq8lryWvJa8lryWvJa8lryWvJa8lryevJ68nryevJ68nryevJ68nryevJW5O3Jm9N3pq8NXlr8tbkrclbk7cmb0velrwteVNelZRXJeVVSXlVUl6VlFcl5VVJeVVSXpWUVyXlVUl5VVJelZRXJeVVSXlVUl6VlFcl5VVJeVVSXpWUVyXlVUl5VVJelZRXJeWVpLySlFeS8kpSXknKK0l5JSmvJOWVpLySlFeS8kpSXknKK0l5JSmvJOWVpLySlFeS8kpSXknKK0l5JSmvJOWVpLySlFeS8kpSXknKK0l5JSmvJOWVpLySlFeS8kpSXknKK0l5JSmvJOWVpLySlFeS8kpSXknKK0l5JSmvJOWVpLySlFeS8kpSXknKK0l5JSmvJOWVpLySlFeS8kpSXknKK0l5JSmvJOWVpLySlFeS8kpSXknKK0l5JSmvJOWVpLyS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",
|
|
7238
7238
|
"is_unconstrained": true,
|
|
7239
7239
|
"name": "sync_state"
|
|
7240
7240
|
}
|