From 7dfb1738b7bf40b1ab816395ef671b0cf895ce6d Mon Sep 17 00:00:00 2001 From: sklppy88 Date: Thu, 22 Aug 2024 12:26:41 +0000 Subject: [PATCH] init --- noir-projects/aztec-nr/Nargo.toml | 1 + noir-projects/aztec-nr/uint-note/Nargo.toml | 8 ++ noir-projects/aztec-nr/uint-note/src/lib.nr | 1 + .../aztec-nr/uint-note/src/uint_note.nr | 130 ++++++++++++++++++ 4 files changed, 140 insertions(+) create mode 100644 noir-projects/aztec-nr/uint-note/Nargo.toml create mode 100644 noir-projects/aztec-nr/uint-note/src/lib.nr create mode 100644 noir-projects/aztec-nr/uint-note/src/uint_note.nr diff --git a/noir-projects/aztec-nr/Nargo.toml b/noir-projects/aztec-nr/Nargo.toml index 8f2ac888ccf1..f4185b088a68 100644 --- a/noir-projects/aztec-nr/Nargo.toml +++ b/noir-projects/aztec-nr/Nargo.toml @@ -6,4 +6,5 @@ members = [ "compressed-string", "easy-private-state", "value-note", + "uint-note", ] diff --git a/noir-projects/aztec-nr/uint-note/Nargo.toml b/noir-projects/aztec-nr/uint-note/Nargo.toml new file mode 100644 index 000000000000..b04f25f78d4e --- /dev/null +++ b/noir-projects/aztec-nr/uint-note/Nargo.toml @@ -0,0 +1,8 @@ +[package] +name = "uint_note" +authors = ["aztec-labs"] +compiler_version = ">=0.18.0" +type = "lib" + +[dependencies] +aztec = { path = "../aztec" } diff --git a/noir-projects/aztec-nr/uint-note/src/lib.nr b/noir-projects/aztec-nr/uint-note/src/lib.nr new file mode 100644 index 000000000000..2225eccc2a25 --- /dev/null +++ b/noir-projects/aztec-nr/uint-note/src/lib.nr @@ -0,0 +1 @@ +mod uint_note; diff --git a/noir-projects/aztec-nr/uint-note/src/uint_note.nr b/noir-projects/aztec-nr/uint-note/src/uint_note.nr new file mode 100644 index 000000000000..c431cb36c705 --- /dev/null +++ b/noir-projects/aztec-nr/uint-note/src/uint_note.nr @@ -0,0 +1,130 @@ +use dep::aztec::{ + generators::{Ga1 as G_amt, Ga2 as G_npk, Ga3 as G_rnd, G_slot}, + prelude::{NoteHeader, NoteInterface, PrivateContext}, + protocol_types::{ + constants::GENERATOR_INDEX__NOTE_NULLIFIER, point::{Point, POINT_LENGTH}, scalar::Scalar, + hash::poseidon2_hash_with_separator, traits::Serialize +}, + note::utils::compute_note_hash_for_nullify, oracle::unsafe_rand::unsafe_rand, + keys::getters::get_nsk_app +}; +use dep::std::{embedded_curve_ops::multi_scalar_mul, hash::from_field_unsafe}; + +trait OwnedNote { + fn new(amount: U128, owner_npk_m_hash: Field) -> Self; + fn get_amount(self) -> U128; +} + +global UINT_NOTE_LEN: Field = 3; // 3 plus a header. +global UINT_NOTE_BYTES_LEN: Field = 3 * 32 + 64; + +#[aztec(note)] +struct UintNote { + // The integer stored by the note + value: U128, + // The nullifying public key hash is used with the nsk_app to ensure that the note can be privately spent. + npk_m_hash: Field, + // Randomness of the note to hide its contents + randomness: Field, +} + +impl NoteInterface for UintNote { + fn compute_nullifier(self, context: &mut PrivateContext, note_hash_for_nullify: Field) -> Field { + let secret = context.request_nsk_app(self.npk_m_hash); + poseidon2_hash_with_separator([ + note_hash_for_nullify, + secret + ], + GENERATOR_INDEX__NOTE_NULLIFIER as Field, + ) + } + + fn compute_nullifier_without_context(self) -> Field { + let note_hash_for_nullify = compute_note_hash_for_nullify(self); + let secret = get_nsk_app(self.npk_m_hash); + poseidon2_hash_with_separator([note_hash_for_nullify, secret],GENERATOR_INDEX__NOTE_NULLIFIER) + } + + fn compute_note_hiding_point(self) -> Point { + // We use the unsafe version because the multi_scalar_mul will constrain the scalars. + let amount_scalar = from_field_unsafe(self.value.to_integer()); + let npk_m_hash_scalar = from_field_unsafe(self.npk_m_hash); + let randomness_scalar = from_field_unsafe(self.randomness); + let slot_scalar = from_field_unsafe(self.header.storage_slot); + // We compute the note hiding point as: + // `G_amt * amount + G_npk * npk_m_hash + G_rnd * randomness + G_slot * slot` + // instead of using pedersen or poseidon2 because it allows us to privately add and subtract from amount + // in public by leveraging homomorphism. + multi_scalar_mul( + [G_amt, G_npk, G_rnd, G_slot], + [amount_scalar, npk_m_hash_scalar, randomness_scalar, slot_scalar] + ) + } +} + +impl UintNote { + // TODO: Merge this func with `compute_note_hiding_point`. I (benesjan) didn't do it in the initial PR to not have + // to modify macros and all the related funcs in it. + fn to_note_hiding_point(self) -> IntNoteHidingPoint { + IntNoteHidingPoint::new(self.compute_note_hiding_point()) + } +} + +struct IntNoteHidingPoint { + inner: Point +} + +impl IntNoteHidingPoint { + fn new(point: Point) -> Self { + Self { inner: point } + } + + fn add_amount(&mut self, amount: U128) { + self.inner = multi_scalar_mul([G_amt], [from_field_unsafe(amount.to_integer())]) + self.inner; + } + + fn add_npk_m_hash(&mut self, npk_m_hash: Field) { + self.inner = multi_scalar_mul([G_npk], [from_field_unsafe(npk_m_hash)]) + self.inner; + } + + fn add_randomness(&mut self, randomness: Field) { + self.inner = multi_scalar_mul([G_rnd], [from_field_unsafe(randomness)]) + self.inner; + } + + fn add_slot(&mut self, slot: Field) { + self.inner = multi_scalar_mul([G_slot], [from_field_unsafe(slot)]) + self.inner; + } + + fn finalize(self) -> Field { + self.inner.x + } +} + +impl Serialize for IntNoteHidingPoint { + fn serialize(self) -> [Field; POINT_LENGTH] { + self.inner.serialize() + } +} + +impl Eq for UintNote { + fn eq(self, other: Self) -> bool { + (self.value == other.value) & + (self.npk_m_hash == other.npk_m_hash) & + (self.randomness == other.randomness) + } +} + +impl OwnedNote for UintNote { + fn new(amount: U128, owner_npk_m_hash: Field) -> Self { + Self { + value: amount, + npk_m_hash: owner_npk_m_hash, + randomness: unsafe_rand(), + header: NoteHeader::empty(), + } + } + + fn get_amount(self) -> U128 { + self.value + } +}