From d630b44cb132e92c6d42d1b84895022f74d499c8 Mon Sep 17 00:00:00 2001 From: sirasistant Date: Wed, 14 Feb 2024 14:45:04 +0000 Subject: [PATCH 1/4] git subrepo pull --branch=master --force noir subrepo: subdir: "noir" merged: "c44ef1484" upstream: origin: "https://github.com/noir-lang/noir" branch: "master" commit: "c44ef1484" git-subrepo: version: "0.4.6" origin: "???" commit: "???" --- noir/.github/actions/setup/action.yml | 2 +- noir/.github/workflows/release.yml | 2 +- noir/.gitrepo | 4 +- noir/aztec_macros/src/lib.rs | 5 +- .../src/brillig/brillig_gen/brillig_block.rs | 20 +- .../src/ssa/ssa_gen/context.rs | 4 +- noir/compiler/noirc_frontend/src/ast/mod.rs | 63 +++++- .../src/hir/resolution/resolver.rs | 44 ++-- .../src/hir/resolution/type_aliases.rs | 6 +- .../src/hir/type_check/errors.rs | 3 +- .../noirc_frontend/src/hir/type_check/expr.rs | 29 ++- .../noirc_frontend/src/hir/type_check/stmt.rs | 7 +- .../noirc_frontend/src/hir_def/types.rs | 197 ++++++++++++++---- .../noirc_frontend/src/lexer/errors.rs | 10 - .../noirc_frontend/src/lexer/lexer.rs | 2 +- .../noirc_frontend/src/lexer/token.rs | 8 +- .../src/monomorphization/ast.rs | 7 +- .../src/monomorphization/mod.rs | 31 +-- .../noirc_frontend/src/node_interner.rs | 31 ++- .../noirc_frontend/src/parser/errors.rs | 13 +- .../noirc_frontend/src/parser/parser.rs | 22 +- .../noirc_frontend/src/resolve_locations.rs | 4 +- noir/compiler/noirc_frontend/src/tests.rs | 22 ++ noir/cspell.json | 1 + .../docs/noir/concepts/data_types/index.md | 14 ++ .../docs/noir/concepts/data_types/integers.md | 9 +- .../integer_literal_overflow/src/main.nr | 2 +- .../restricted_bit_sizes}/Nargo.toml | 3 +- .../restricted_bit_sizes/src/main.nr | 5 + .../regression_2854/Prover.toml | 1 - .../regression_2854/src/main.nr | 3 - noir/tooling/noirc_abi/src/lib.rs | 16 +- 32 files changed, 420 insertions(+), 170 deletions(-) rename noir/test_programs/{execution_success/regression_2854 => compile_failure/restricted_bit_sizes}/Nargo.toml (63%) create mode 100644 noir/test_programs/compile_failure/restricted_bit_sizes/src/main.nr delete mode 100644 noir/test_programs/execution_success/regression_2854/Prover.toml delete mode 100644 noir/test_programs/execution_success/regression_2854/src/main.nr diff --git a/noir/.github/actions/setup/action.yml b/noir/.github/actions/setup/action.yml index b265a63d29a..d0e83dedf67 100644 --- a/noir/.github/actions/setup/action.yml +++ b/noir/.github/actions/setup/action.yml @@ -7,7 +7,7 @@ runs: - uses: actions/setup-node@v4 id: node with: - node-version: 18.17.1 + node-version: 18.19.0 cache: 'yarn' cache-dependency-path: 'yarn.lock' diff --git a/noir/.github/workflows/release.yml b/noir/.github/workflows/release.yml index 71a0ab6d894..83e8e479181 100644 --- a/noir/.github/workflows/release.yml +++ b/noir/.github/workflows/release.yml @@ -45,7 +45,7 @@ jobs: - uses: actions/setup-node@v3 with: - node-version: 18.17.1 + node-version: 18.19.0 cache: 'yarn' cache-dependency-path: 'yarn.lock' diff --git a/noir/.gitrepo b/noir/.gitrepo index 240e767e6f2..8b34412d5de 100644 --- a/noir/.gitrepo +++ b/noir/.gitrepo @@ -6,7 +6,7 @@ [subrepo] remote = https://github.com/noir-lang/noir branch = master - commit = 78ef0134b82e76a73dadb6c7975def22290e3a1a - parent = e23d048e916fa12966fe01d1a8c0d3bfb50c2943 + commit = c44ef14847a436733206b6dd9590a7ab214ecd97 + parent = 382626cddaa175041695e2eb70ad3c350351ffe3 method = merge cmdver = 0.4.6 diff --git a/noir/aztec_macros/src/lib.rs b/noir/aztec_macros/src/lib.rs index 0f054c262bf..51a8b5361a6 100644 --- a/noir/aztec_macros/src/lib.rs +++ b/noir/aztec_macros/src/lib.rs @@ -1129,7 +1129,10 @@ fn create_context(ty: &str, params: &[Param]) -> Result, AztecMac add_array_to_hasher( &id, &UnresolvedType { - typ: UnresolvedTypeData::Integer(Signedness::Unsigned, 32), + typ: UnresolvedTypeData::Integer( + Signedness::Unsigned, + noirc_frontend::IntegerBitSize::ThirtyTwo, + ), span: None, }, ) diff --git a/noir/compiler/noirc_evaluator/src/brillig/brillig_gen/brillig_block.rs b/noir/compiler/noirc_evaluator/src/brillig/brillig_gen/brillig_block.rs index 84b719f29aa..7697d7e65fa 100644 --- a/noir/compiler/noirc_evaluator/src/brillig/brillig_gen/brillig_block.rs +++ b/noir/compiler/noirc_evaluator/src/brillig/brillig_gen/brillig_block.rs @@ -619,23 +619,29 @@ impl<'block> BrilligBlock<'block> { ); } Instruction::RangeCheck { value, max_bit_size, assert_message } => { - let left = self.convert_ssa_register_value(*value, dfg); - let max = BigUint::from(2_u128).pow(*max_bit_size); - let right = self.brillig_context.allocate_register(); - self.brillig_context.const_instruction( - right, + let value = self.convert_ssa_register_value(*value, dfg); + // Cast original value to field + let left = self.brillig_context.allocate_register(); + self.convert_cast(left, value, &Type::field()); + + // Create a field constant with the max + let max = BigUint::from(2_u128).pow(*max_bit_size) - BigUint::from(1_u128); + let right = self.brillig_context.make_constant( FieldElement::from_be_bytes_reduce(&max.to_bytes_be()).into(), FieldElement::max_num_bits(), ); + // Check if lte max let brillig_binary_op = BrilligBinaryOp::Integer { - op: BinaryIntOp::LessThan, - bit_size: max_bit_size + 1, + op: BinaryIntOp::LessThanEquals, + bit_size: FieldElement::max_num_bits(), }; let condition = self.brillig_context.allocate_register(); self.brillig_context.binary_instruction(left, right, condition, brillig_binary_op); + self.brillig_context.constrain_instruction(condition, assert_message.clone()); self.brillig_context.deallocate_register(condition); + self.brillig_context.deallocate_register(left); self.brillig_context.deallocate_register(right); } Instruction::IncrementRc { value } => { diff --git a/noir/compiler/noirc_evaluator/src/ssa/ssa_gen/context.rs b/noir/compiler/noirc_evaluator/src/ssa/ssa_gen/context.rs index 6ee7f312660..845ffd15413 100644 --- a/noir/compiler/noirc_evaluator/src/ssa/ssa_gen/context.rs +++ b/noir/compiler/noirc_evaluator/src/ssa/ssa_gen/context.rs @@ -219,8 +219,8 @@ impl<'a> FunctionContext<'a> { let element_types = Self::convert_type(element).flatten(); Type::Array(Rc::new(element_types), *len as usize) } - ast::Type::Integer(Signedness::Signed, bits) => Type::signed(*bits), - ast::Type::Integer(Signedness::Unsigned, bits) => Type::unsigned(*bits), + ast::Type::Integer(Signedness::Signed, bits) => Type::signed((*bits).into()), + ast::Type::Integer(Signedness::Unsigned, bits) => Type::unsigned((*bits).into()), ast::Type::Bool => Type::unsigned(1), ast::Type::String(len) => Type::Array(Rc::new(vec![Type::char()]), *len as usize), ast::Type::FmtString(_, _) => { diff --git a/noir/compiler/noirc_frontend/src/ast/mod.rs b/noir/compiler/noirc_frontend/src/ast/mod.rs index 1223f822af3..29edbaca594 100644 --- a/noir/compiler/noirc_frontend/src/ast/mod.rs +++ b/noir/compiler/noirc_frontend/src/ast/mod.rs @@ -28,6 +28,55 @@ use crate::{ }; use iter_extended::vecmap; +#[derive(Debug, PartialEq, Eq, Clone, Copy, Hash, Ord, PartialOrd)] +pub enum IntegerBitSize { + One, + Eight, + ThirtyTwo, + SixtyFour, +} + +impl IntegerBitSize { + pub fn allowed_sizes() -> Vec { + vec![Self::One, Self::Eight, Self::ThirtyTwo, Self::SixtyFour] + } +} + +impl From for u32 { + fn from(size: IntegerBitSize) -> u32 { + use IntegerBitSize::*; + match size { + One => 1, + Eight => 8, + ThirtyTwo => 32, + SixtyFour => 64, + } + } +} + +pub struct InvalidIntegerBitSizeError(pub u32); + +impl TryFrom for IntegerBitSize { + type Error = InvalidIntegerBitSizeError; + + fn try_from(value: u32) -> Result { + use IntegerBitSize::*; + match value { + 1 => Ok(One), + 8 => Ok(Eight), + 32 => Ok(ThirtyTwo), + 64 => Ok(SixtyFour), + _ => Err(InvalidIntegerBitSizeError(value)), + } + } +} + +impl core::fmt::Display for IntegerBitSize { + fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { + write!(f, "{}", u32::from(*self)) + } +} + /// The parser parses types as 'UnresolvedType's which /// require name resolution to resolve any type names used /// for structs within, but are otherwise identical to Types. @@ -35,7 +84,7 @@ use iter_extended::vecmap; pub enum UnresolvedTypeData { FieldElement, Array(Option, Box), // [4]Witness = Array(4, Witness) - Integer(Signedness, u32), // u32 = Integer(unsigned, 32) + Integer(Signedness, IntegerBitSize), // u32 = Integer(unsigned, ThirtyTwo) Bool, Expression(UnresolvedTypeExpression), String(Option), @@ -197,11 +246,17 @@ impl UnresolvedType { } impl UnresolvedTypeData { - pub fn from_int_token(token: IntType) -> UnresolvedTypeData { + pub fn from_int_token( + token: IntType, + ) -> Result { use {IntType::*, UnresolvedTypeData::Integer}; match token { - Signed(num_bits) => Integer(Signedness::Signed, num_bits), - Unsigned(num_bits) => Integer(Signedness::Unsigned, num_bits), + Signed(num_bits) => { + Ok(Integer(Signedness::Signed, IntegerBitSize::try_from(num_bits)?)) + } + Unsigned(num_bits) => { + Ok(Integer(Signedness::Unsigned, IntegerBitSize::try_from(num_bits)?)) + } } } diff --git a/noir/compiler/noirc_frontend/src/hir/resolution/resolver.rs b/noir/compiler/noirc_frontend/src/hir/resolution/resolver.rs index bef5e113428..d4aae133b35 100644 --- a/noir/compiler/noirc_frontend/src/hir/resolution/resolver.rs +++ b/noir/compiler/noirc_frontend/src/hir/resolution/resolver.rs @@ -29,7 +29,7 @@ use crate::hir::def_map::{LocalModuleId, ModuleDefId, TryFromModuleDefId, MAIN_F use crate::hir_def::stmt::{HirAssignStatement, HirForStatement, HirLValue, HirPattern}; use crate::node_interner::{ DefinitionId, DefinitionKind, DependencyId, ExprId, FuncId, GlobalId, NodeInterner, StmtId, - StructId, TraitId, TraitImplId, TraitMethodId, + StructId, TraitId, TraitImplId, TraitMethodId, TypeAliasId, }; use crate::{ hir::{def_map::CrateDefMap, resolution::path_resolver::PathResolver}, @@ -39,9 +39,9 @@ use crate::{ use crate::{ ArrayLiteral, ContractFunctionType, Distinctness, ForRange, FunctionDefinition, FunctionReturnType, FunctionVisibility, Generics, LValue, NoirStruct, NoirTypeAlias, Param, - Path, PathKind, Pattern, Shared, StructType, Type, TypeAliasType, TypeVariable, - TypeVariableKind, UnaryOp, UnresolvedGenerics, UnresolvedTraitConstraint, UnresolvedType, - UnresolvedTypeData, UnresolvedTypeExpression, Visibility, ERROR_IDENT, + Path, PathKind, Pattern, Shared, StructType, Type, TypeAlias, TypeVariable, TypeVariableKind, + UnaryOp, UnresolvedGenerics, UnresolvedTraitConstraint, UnresolvedType, UnresolvedTypeData, + UnresolvedTypeExpression, Visibility, ERROR_IDENT, }; use fm::FileId; use iter_extended::vecmap; @@ -573,16 +573,19 @@ impl<'a> Resolver<'a> { let span = path.span(); let mut args = vecmap(args, |arg| self.resolve_type_inner(arg, new_variables)); - if let Some(type_alias_type) = self.lookup_type_alias(path.clone()) { - let expected_generic_count = type_alias_type.generics.len(); - let type_alias_string = type_alias_type.to_string(); - let id = type_alias_type.id; + if let Some(type_alias) = self.lookup_type_alias(path.clone()) { + let type_alias = type_alias.borrow(); + let expected_generic_count = type_alias.generics.len(); + let type_alias_string = type_alias.to_string(); + let id = type_alias.id; self.verify_generics_count(expected_generic_count, &mut args, span, || { type_alias_string }); - let result = self.interner.get_type_alias(id).get_type(&args); + if let Some(item) = self.current_item { + self.interner.add_type_alias_dependency(item, id); + } // Collecting Type Alias references [Location]s to be used by LSP in order // to resolve the definition of the type alias @@ -593,9 +596,8 @@ impl<'a> Resolver<'a> { // equal to another type alias. Fixing this fully requires an analysis to create a DFG // of definition ordering, but for now we have an explicit check here so that we at // least issue an error that the type was not found instead of silently passing. - if result != Type::Error { - return result; - } + let alias = self.interner.get_type_alias(id); + return Type::Alias(alias, args); } match self.lookup_struct_or_error(path) { @@ -752,12 +754,15 @@ impl<'a> Resolver<'a> { resolved_type } - pub fn resolve_type_aliases( + pub fn resolve_type_alias( mut self, unresolved: NoirTypeAlias, + alias_id: TypeAliasId, ) -> (Type, Generics, Vec) { let generics = self.add_generics(&unresolved.generics); self.resolve_local_globals(); + + self.current_item = Some(DependencyId::Alias(alias_id)); let typ = self.resolve_type(unresolved.typ); (typ, generics, self.errors) @@ -1120,6 +1125,17 @@ impl<'a> Resolver<'a> { } } } + Type::Alias(alias, generics) => { + for (i, generic) in generics.iter().enumerate() { + if let Type::NamedGeneric(type_variable, name) = generic { + if alias.borrow().generic_is_numeric(i) { + found.insert(name.to_string(), type_variable.clone()); + } + } else { + Self::find_numeric_generics_in_type(generic, found); + } + } + } Type::MutableReference(element) => Self::find_numeric_generics_in_type(element, found), Type::String(length) => { if let Type::NamedGeneric(type_variable, name) = length.as_ref() { @@ -1791,7 +1807,7 @@ impl<'a> Resolver<'a> { } } - fn lookup_type_alias(&mut self, path: Path) -> Option<&TypeAliasType> { + fn lookup_type_alias(&mut self, path: Path) -> Option> { self.lookup(path).ok().map(|id| self.interner.get_type_alias(id)) } diff --git a/noir/compiler/noirc_frontend/src/hir/resolution/type_aliases.rs b/noir/compiler/noirc_frontend/src/hir/resolution/type_aliases.rs index f66f6c8dfa7..2e5ce611a7f 100644 --- a/noir/compiler/noirc_frontend/src/hir/resolution/type_aliases.rs +++ b/noir/compiler/noirc_frontend/src/hir/resolution/type_aliases.rs @@ -17,7 +17,7 @@ pub(crate) fn resolve_type_aliases( crate_id: CrateId, ) -> Vec<(CompilationError, FileId)> { let mut errors: Vec<(CompilationError, FileId)> = vec![]; - for (type_id, unresolved_typ) in type_aliases { + for (alias_id, unresolved_typ) in type_aliases { let path_resolver = StandardPathResolver::new(ModuleId { local_id: unresolved_typ.module_id, krate: crate_id, @@ -25,9 +25,9 @@ pub(crate) fn resolve_type_aliases( let file = unresolved_typ.file_id; let (typ, generics, resolver_errors) = Resolver::new(&mut context.def_interner, &path_resolver, &context.def_maps, file) - .resolve_type_aliases(unresolved_typ.type_alias_def); + .resolve_type_alias(unresolved_typ.type_alias_def, alias_id); errors.extend(resolver_errors.iter().cloned().map(|e| (e.into(), file))); - context.def_interner.set_type_alias(type_id, typ, generics); + context.def_interner.set_type_alias(alias_id, typ, generics); } errors } diff --git a/noir/compiler/noirc_frontend/src/hir/type_check/errors.rs b/noir/compiler/noirc_frontend/src/hir/type_check/errors.rs index 3967d7642f7..96d30100d8b 100644 --- a/noir/compiler/noirc_frontend/src/hir/type_check/errors.rs +++ b/noir/compiler/noirc_frontend/src/hir/type_check/errors.rs @@ -8,6 +8,7 @@ use crate::hir_def::expr::HirBinaryOp; use crate::hir_def::types::Type; use crate::BinaryOpKind; use crate::FunctionReturnType; +use crate::IntegerBitSize; use crate::Signedness; #[derive(Error, Debug, Clone, PartialEq, Eq)] @@ -67,7 +68,7 @@ pub enum TypeCheckError { #[error("Integers must have the same signedness LHS is {sign_x:?}, RHS is {sign_y:?}")] IntegerSignedness { sign_x: Signedness, sign_y: Signedness, span: Span }, #[error("Integers must have the same bit width LHS is {bit_width_x}, RHS is {bit_width_y}")] - IntegerBitWidth { bit_width_x: u32, bit_width_y: u32, span: Span }, + IntegerBitWidth { bit_width_x: IntegerBitSize, bit_width_y: IntegerBitSize, span: Span }, #[error("{kind} cannot be used in an infix operation")] InvalidInfixOp { kind: &'static str, span: Span }, #[error("{kind} cannot be used in a unary operation")] diff --git a/noir/compiler/noirc_frontend/src/hir/type_check/expr.rs b/noir/compiler/noirc_frontend/src/hir/type_check/expr.rs index b3180e0dd20..96a79152f69 100644 --- a/noir/compiler/noirc_frontend/src/hir/type_check/expr.rs +++ b/noir/compiler/noirc_frontend/src/hir/type_check/expr.rs @@ -565,6 +565,7 @@ impl<'interner> TypeChecker<'interner> { Type::Integer(..) | Type::FieldElement | Type::TypeVariable(_, TypeVariableKind::IntegerOrField) + | Type::TypeVariable(_, TypeVariableKind::Integer) | Type::Bool => (), Type::TypeVariable(_, _) => { @@ -805,7 +806,7 @@ impl<'interner> TypeChecker<'interner> { // Matches on TypeVariable must be first to follow any type // bindings. - (TypeVariable(int, _), other) | (other, TypeVariable(int, _)) => { + (TypeVariable(int, int_kind), other) | (other, TypeVariable(int, int_kind)) => { if let TypeBinding::Bound(binding) = &*int.borrow() { return self.comparator_operand_type_rules(other, binding, op, span); } @@ -823,7 +824,13 @@ impl<'interner> TypeChecker<'interner> { } let mut bindings = TypeBindings::new(); - if other.try_bind_to_polymorphic_int(int, &mut bindings).is_ok() + if other + .try_bind_to_polymorphic_int( + int, + &mut bindings, + *int_kind == TypeVariableKind::Integer, + ) + .is_ok() || other == &Type::Error { Type::apply_type_bindings(bindings); @@ -837,6 +844,10 @@ impl<'interner> TypeChecker<'interner> { }) } } + (Alias(alias, args), other) | (other, Alias(alias, args)) => { + let alias = alias.borrow().get_type(args); + self.comparator_operand_type_rules(&alias, other, op, span) + } (Integer(sign_x, bit_width_x), Integer(sign_y, bit_width_y)) => { if sign_x != sign_y { return Err(TypeCheckError::IntegerSignedness { @@ -1081,7 +1092,7 @@ impl<'interner> TypeChecker<'interner> { // Matches on TypeVariable must be first so that we follow any type // bindings. - (TypeVariable(int, _), other) | (other, TypeVariable(int, _)) => { + (TypeVariable(int, int_kind), other) | (other, TypeVariable(int, int_kind)) => { if let TypeBinding::Bound(binding) = &*int.borrow() { return self.infix_operand_type_rules(binding, op, other, span); } @@ -1114,7 +1125,13 @@ impl<'interner> TypeChecker<'interner> { } let mut bindings = TypeBindings::new(); - if other.try_bind_to_polymorphic_int(int, &mut bindings).is_ok() + if other + .try_bind_to_polymorphic_int( + int, + &mut bindings, + *int_kind == TypeVariableKind::Integer, + ) + .is_ok() || other == &Type::Error { Type::apply_type_bindings(bindings); @@ -1128,6 +1145,10 @@ impl<'interner> TypeChecker<'interner> { }) } } + (Alias(alias, args), other) | (other, Alias(alias, args)) => { + let alias = alias.borrow().get_type(args); + self.infix_operand_type_rules(&alias, op, other, span) + } (Integer(sign_x, bit_width_x), Integer(sign_y, bit_width_y)) => { if sign_x != sign_y { return Err(TypeCheckError::IntegerSignedness { diff --git a/noir/compiler/noirc_frontend/src/hir/type_check/stmt.rs b/noir/compiler/noirc_frontend/src/hir/type_check/stmt.rs index 1bd6c16277b..03d61b93e0c 100644 --- a/noir/compiler/noirc_frontend/src/hir/type_check/stmt.rs +++ b/noir/compiler/noirc_frontend/src/hir/type_check/stmt.rs @@ -93,7 +93,7 @@ impl<'interner> TypeChecker<'interner> { match pattern { HirPattern::Identifier(ident) => self.interner.push_definition_type(ident.id, typ), HirPattern::Mutable(pattern, _) => self.bind_pattern(pattern, typ), - HirPattern::Tuple(fields, location) => match typ { + HirPattern::Tuple(fields, location) => match typ.follow_bindings() { Type::Tuple(field_types) if field_types.len() == fields.len() => { for (field, field_type) in fields.iter().zip(field_types) { self.bind_pattern(field, field_type); @@ -120,12 +120,12 @@ impl<'interner> TypeChecker<'interner> { source: Source::Assignment, }); - if let Type::Struct(struct_type, generics) = struct_type { + if let Type::Struct(struct_type, generics) = struct_type.follow_bindings() { let struct_type = struct_type.borrow(); for (field_name, field_pattern) in fields { if let Some((type_field, _)) = - struct_type.get_field(&field_name.0.contents, generics) + struct_type.get_field(&field_name.0.contents, &generics) { self.bind_pattern(field_pattern, type_field); } @@ -351,6 +351,7 @@ impl<'interner> TypeChecker<'interner> { HirExpression::Literal(HirLiteral::Integer(value, false)) => { let v = value.to_u128(); if let Type::Integer(_, bit_count) = annotated_type { + let bit_count: u32 = (*bit_count).into(); let max = 1 << bit_count; if v >= max { self.errors.push(TypeCheckError::OverflowingAssignment { diff --git a/noir/compiler/noirc_frontend/src/hir_def/types.rs b/noir/compiler/noirc_frontend/src/hir_def/types.rs index 30ca7054a77..98b47f17cd4 100644 --- a/noir/compiler/noirc_frontend/src/hir_def/types.rs +++ b/noir/compiler/noirc_frontend/src/hir_def/types.rs @@ -8,6 +8,7 @@ use std::{ use crate::{ hir::type_check::TypeCheckError, node_interner::{ExprId, NodeInterner, TraitId, TypeAliasId}, + IntegerBitSize, }; use iter_extended::vecmap; use noirc_errors::{Location, Span}; @@ -27,8 +28,8 @@ pub enum Type { Array(Box, Box), /// A primitive integer type with the given sign and bit count. - /// E.g. `u32` would be `Integer(Unsigned, 32)` - Integer(Signedness, u32), + /// E.g. `u32` would be `Integer(Unsigned, ThirtyTwo)` + Integer(Signedness, IntegerBitSize), /// The primitive `bool` type. Bool, @@ -44,13 +45,18 @@ pub enum Type { /// The unit type `()`. Unit, + /// A tuple type with the given list of fields in the order they appear in source code. + Tuple(Vec), + /// A user-defined struct type. The `Shared` field here refers to /// the shared definition for each instance of this struct type. The `Vec` /// represents the generic arguments (if any) to this struct type. Struct(Shared, Vec), - /// A tuple type with the given list of fields in the order they appear in source code. - Tuple(Vec), + /// A user-defined alias to another type. Similar to a Struct, this carries a shared + /// reference to the definition of the alias along with any generics that may have + /// been applied to the alias. + Alias(Shared, Vec), /// TypeVariables are stand-in variables for some type which is not yet known. /// They are not to be confused with NamedGenerics. While the later mostly works @@ -116,11 +122,16 @@ impl Type { let typ = typ.as_ref(); (length as u32) * typ.field_count() } - Type::Struct(ref def, args) => { + Type::Struct(def, args) => { let struct_type = def.borrow(); let fields = struct_type.get_fields(args); fields.iter().fold(0, |acc, (_, field_type)| acc + field_type.field_count()) } + Type::Alias(def, _) => { + // It is safe to access `typ` without instantiating generics here since generics + // cannot change the number of fields in `typ`. + def.borrow().typ.field_count() + } Type::Tuple(fields) => { fields.iter().fold(0, |acc, field_typ| acc + field_typ.field_count()) } @@ -309,7 +320,7 @@ impl std::fmt::Display for StructType { /// Wrap around an unsolved type #[derive(Debug, Clone, Eq)] -pub struct TypeAliasType { +pub struct TypeAlias { pub name: Ident, pub id: TypeAliasId, pub typ: Type, @@ -317,40 +328,33 @@ pub struct TypeAliasType { pub location: Location, } -impl std::hash::Hash for TypeAliasType { +impl std::hash::Hash for TypeAlias { fn hash(&self, state: &mut H) { self.id.hash(state); } } -impl PartialEq for TypeAliasType { +impl PartialEq for TypeAlias { fn eq(&self, other: &Self) -> bool { self.id == other.id } } -impl std::fmt::Display for TypeAliasType { +impl std::fmt::Display for TypeAlias { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { - write!(f, "{}", self.name)?; - - if !self.generics.is_empty() { - let generics = vecmap(&self.generics, |binding| binding.borrow().to_string()); - write!(f, "{}", generics.join(", "))?; - } - - Ok(()) + write!(f, "{}", self.name) } } -impl TypeAliasType { +impl TypeAlias { pub fn new( id: TypeAliasId, name: Ident, location: Location, typ: Type, generics: Generics, - ) -> TypeAliasType { - TypeAliasType { id, typ, name, location, generics } + ) -> TypeAlias { + TypeAlias { id, typ, name, location, generics } } pub fn set_type_and_generics(&mut self, new_typ: Type, new_generics: Generics) { @@ -371,6 +375,14 @@ impl TypeAliasType { self.typ.substitute(&substitutions) } + + /// True if the given index is the same index as a generic type of this alias + /// which is expected to be a numeric generic. + /// This is needed because we infer type kinds in Noir and don't have extensive kind checking. + pub fn generic_is_numeric(&self, index_of_generic: usize) -> bool { + let target_id = self.generics[index_of_generic].0; + self.typ.contains_numeric_typevar(target_id) + } } /// A shared, mutable reference to some T. @@ -441,6 +453,10 @@ pub enum TypeVariableKind { /// type annotations on each integer literal. IntegerOrField, + /// A generic integer type. This is a more specific kind of TypeVariable + /// that can only be bound to Type::Integer, or other polymorphic integers. + Integer, + /// A potentially constant array size. This will only bind to itself, Type::NotConstant, or /// Type::Constant(n) with a matching size. This defaults to Type::Constant(n) if still unbound /// during monomorphization. @@ -538,7 +554,7 @@ impl Type { } pub fn default_range_loop_type() -> Type { - Type::Integer(Signedness::Unsigned, 64) + Type::Integer(Signedness::Unsigned, IntegerBitSize::SixtyFour) } pub fn type_variable(id: TypeVariableId) -> Type { @@ -637,6 +653,13 @@ impl Type { } }) } + Type::Alias(alias, generics) => generics.iter().enumerate().any(|(i, generic)| { + if named_generic_id_matches_target(generic) { + alias.borrow().generic_is_numeric(i) + } else { + generic.contains_numeric_typevar(target_id) + } + }), Type::MutableReference(element) => element.contains_numeric_typevar(target_id), Type::String(length) => named_generic_id_matches_target(length), Type::FmtString(length, elements) => { @@ -673,6 +696,11 @@ impl Type { | Type::TraitAsType(..) | Type::NotConstant => false, + // This function is called during name resolution before we've verified aliases + // are not cyclic. As a result, it wouldn't be safe to check this alias' definition + // to see if the aliased type is valid. + Type::Alias(..) => false, + Type::Array(length, element) => { length.is_valid_for_program_input() && element.is_valid_for_program_input() } @@ -746,6 +774,13 @@ impl std::fmt::Display for Type { Signedness::Unsigned => write!(f, "u{num_bits}"), }, Type::TypeVariable(var, TypeVariableKind::Normal) => write!(f, "{}", var.borrow()), + Type::TypeVariable(binding, TypeVariableKind::Integer) => { + if let TypeBinding::Unbound(_) = &*binding.borrow() { + write!(f, "{}", TypeVariableKind::Integer.default_type()) + } else { + write!(f, "{}", binding.borrow()) + } + } Type::TypeVariable(binding, TypeVariableKind::IntegerOrField) => { if let TypeBinding::Unbound(_) = &*binding.borrow() { // Show a Field by default if this TypeVariableKind::IntegerOrField is unbound, since that is @@ -772,6 +807,14 @@ impl std::fmt::Display for Type { write!(f, "{}<{}>", s.borrow(), args.join(", ")) } } + Type::Alias(alias, args) => { + let args = vecmap(args, |arg| arg.to_string()); + if args.is_empty() { + write!(f, "{}", alias.borrow()) + } else { + write!(f, "{}<{}>", alias.borrow(), args.join(", ")) + } + } Type::TraitAsType(_id, name, generics) => { write!(f, "impl {}", name)?; if !generics.is_empty() { @@ -863,7 +906,7 @@ impl Type { TypeBinding::Unbound(id) => *id, }; - let this = self.substitute(bindings); + let this = self.substitute(bindings).follow_bindings(); match &this { Type::Constant(length) if *length == target_length => { @@ -910,6 +953,7 @@ impl Type { Ok(()) } TypeVariableKind::IntegerOrField => Err(UnificationError), + TypeVariableKind::Integer => Err(UnificationError), }, } } @@ -924,13 +968,14 @@ impl Type { &self, var: &TypeVariable, bindings: &mut TypeBindings, + only_integer: bool, ) -> Result<(), UnificationError> { let target_id = match &*var.borrow() { TypeBinding::Bound(_) => unreachable!(), TypeBinding::Unbound(id) => *id, }; - let this = self.substitute(bindings); + let this = self.substitute(bindings).follow_bindings(); match &this { Type::FieldElement | Type::Integer(..) => { bindings.insert(target_id, (var.clone(), this)); @@ -939,7 +984,30 @@ impl Type { Type::TypeVariable(self_var, TypeVariableKind::IntegerOrField) => { let borrow = self_var.borrow(); match &*borrow { - TypeBinding::Bound(typ) => typ.try_bind_to_polymorphic_int(var, bindings), + TypeBinding::Bound(typ) => { + typ.try_bind_to_polymorphic_int(var, bindings, only_integer) + } + // Avoid infinitely recursive bindings + TypeBinding::Unbound(id) if *id == target_id => Ok(()), + TypeBinding::Unbound(new_target_id) => { + if only_integer { + // Integer is more specific than IntegerOrField so we bind the type + // variable to Integer instead. + let clone = Type::TypeVariable(var.clone(), TypeVariableKind::Integer); + bindings.insert(*new_target_id, (self_var.clone(), clone)); + } else { + bindings.insert(target_id, (var.clone(), this.clone())); + } + Ok(()) + } + } + } + Type::TypeVariable(self_var, TypeVariableKind::Integer) => { + let borrow = self_var.borrow(); + match &*borrow { + TypeBinding::Bound(typ) => { + typ.try_bind_to_polymorphic_int(var, bindings, only_integer) + } // Avoid infinitely recursive bindings TypeBinding::Unbound(id) if *id == target_id => Ok(()), TypeBinding::Unbound(_) => { @@ -948,18 +1016,23 @@ impl Type { } } } - Type::TypeVariable(binding, TypeVariableKind::Normal) => { - let borrow = binding.borrow(); + Type::TypeVariable(self_var, TypeVariableKind::Normal) => { + let borrow = self_var.borrow(); match &*borrow { - TypeBinding::Bound(typ) => typ.try_bind_to_polymorphic_int(var, bindings), + TypeBinding::Bound(typ) => { + typ.try_bind_to_polymorphic_int(var, bindings, only_integer) + } // Avoid infinitely recursive bindings TypeBinding::Unbound(id) if *id == target_id => Ok(()), TypeBinding::Unbound(new_target_id) => { - // IntegerOrField is more specific than TypeVariable so we bind the type - // variable to IntegerOrField instead. - let clone = - Type::TypeVariable(var.clone(), TypeVariableKind::IntegerOrField); - bindings.insert(*new_target_id, (binding.clone(), clone)); + // Bind to the most specific type variable kind + let clone_kind = if only_integer { + TypeVariableKind::Integer + } else { + TypeVariableKind::IntegerOrField + }; + let clone = Type::TypeVariable(var.clone(), clone_kind); + bindings.insert(*new_target_id, (self_var.clone(), clone)); Ok(()) } } @@ -1046,10 +1119,24 @@ impl Type { match (self, other) { (Error, _) | (_, Error) => Ok(()), + (Alias(alias, args), other) | (other, Alias(alias, args)) => { + let alias = alias.borrow().get_type(args); + alias.try_unify(other, bindings) + } + (TypeVariable(var, Kind::IntegerOrField), other) | (other, TypeVariable(var, Kind::IntegerOrField)) => { other.try_unify_to_type_variable(var, bindings, |bindings| { - other.try_bind_to_polymorphic_int(var, bindings) + let only_integer = false; + other.try_bind_to_polymorphic_int(var, bindings, only_integer) + }) + } + + (TypeVariable(var, Kind::Integer), other) + | (other, TypeVariable(var, Kind::Integer)) => { + other.try_unify_to_type_variable(var, bindings, |bindings| { + let only_integer = true; + other.try_bind_to_polymorphic_int(var, bindings, only_integer) }) } @@ -1407,7 +1494,7 @@ impl Type { Type::NamedGeneric(binding, _) | Type::TypeVariable(binding, _) => { substitute_binding(binding) } - // Do not substitute_helper fields, it ca, substitute_bound_typevarsn lead to infinite recursion + // Do not substitute_helper fields, it can lead to infinite recursion // and we should not match fields when type checking anyway. Type::Struct(fields, args) => { let args = vecmap(args, |arg| { @@ -1415,6 +1502,12 @@ impl Type { }); Type::Struct(fields.clone(), args) } + Type::Alias(alias, args) => { + let args = vecmap(args, |arg| { + arg.substitute_helper(type_bindings, substitute_bound_typevars) + }); + Type::Alias(alias.clone(), args) + } Type::Tuple(fields) => { let fields = vecmap(fields, |field| { field.substitute_helper(type_bindings, substitute_bound_typevars) @@ -1463,7 +1556,9 @@ impl Type { let field_occurs = fields.occurs(target_id); len_occurs || field_occurs } - Type::Struct(_, generic_args) => generic_args.iter().any(|arg| arg.occurs(target_id)), + Type::Struct(_, generic_args) | Type::Alias(_, generic_args) => { + generic_args.iter().any(|arg| arg.occurs(target_id)) + } Type::Tuple(fields) => fields.iter().any(|field| field.occurs(target_id)), Type::NamedGeneric(binding, _) | Type::TypeVariable(binding, _) => { match &*binding.borrow() { @@ -1514,6 +1609,11 @@ impl Type { let args = vecmap(args, |arg| arg.follow_bindings()); Struct(def.clone(), args) } + Alias(def, args) => { + // We don't need to vecmap(args, follow_bindings) since we're recursively + // calling follow_bindings here already. + def.borrow().get_type(args).follow_bindings() + } Tuple(args) => Tuple(vecmap(args, |arg| arg.follow_bindings())), TypeVariable(var, _) | NamedGeneric(var, _) => { if let TypeBinding::Bound(typ) = &*var.borrow() { @@ -1598,6 +1698,7 @@ impl TypeVariableKind { pub(crate) fn default_type(&self) -> Type { match self { TypeVariableKind::IntegerOrField | TypeVariableKind::Normal => Type::default_int_type(), + TypeVariableKind::Integer => Type::default_range_loop_type(), TypeVariableKind::Constant(length) => Type::Constant(*length), } } @@ -1621,8 +1722,14 @@ impl From<&Type> for PrintableType { PrintableType::Array { length, typ: Box::new(typ.into()) } } Type::Integer(sign, bit_width) => match sign { - Signedness::Unsigned => PrintableType::UnsignedInteger { width: *bit_width }, - Signedness::Signed => PrintableType::SignedInteger { width: *bit_width }, + Signedness::Unsigned => { + PrintableType::UnsignedInteger { width: (*bit_width).into() } + } + Signedness::Signed => PrintableType::SignedInteger { width: (*bit_width).into() }, + }, + Type::TypeVariable(binding, TypeVariableKind::Integer) => match &*binding.borrow() { + TypeBinding::Bound(typ) => typ.into(), + TypeBinding::Unbound(_) => Type::default_range_loop_type().into(), }, Type::TypeVariable(binding, TypeVariableKind::IntegerOrField) => { match &*binding.borrow() { @@ -1645,6 +1752,7 @@ impl From<&Type> for PrintableType { let fields = vecmap(fields, |(name, typ)| (name, typ.into())); PrintableType::Struct { fields, name: struct_type.name.to_string() } } + Type::Alias(alias, args) => alias.borrow().get_type(args).into(), Type::TraitAsType(_, _, _) => unreachable!(), Type::Tuple(types) => PrintableType::Tuple { types: vecmap(types, |typ| typ.into()) }, Type::TypeVariable(_, _) => unreachable!(), @@ -1682,15 +1790,26 @@ impl std::fmt::Debug for Type { Type::TypeVariable(binding, TypeVariableKind::IntegerOrField) => { write!(f, "IntOrField{:?}", binding) } + Type::TypeVariable(binding, TypeVariableKind::Integer) => { + write!(f, "Int{:?}", binding) + } Type::TypeVariable(binding, TypeVariableKind::Constant(n)) => { write!(f, "{}{:?}", n, binding) } Type::Struct(s, args) => { let args = vecmap(args, |arg| format!("{:?}", arg)); if args.is_empty() { - write!(f, "{:?}", s.borrow()) + write!(f, "{}", s.borrow()) + } else { + write!(f, "{}<{}>", s.borrow(), args.join(", ")) + } + } + Type::Alias(alias, args) => { + let args = vecmap(args, |arg| format!("{:?}", arg)); + if args.is_empty() { + write!(f, "{}", alias.borrow()) } else { - write!(f, "{:?}<{}>", s.borrow(), args.join(", ")) + write!(f, "{}<{}>", alias.borrow(), args.join(", ")) } } Type::TraitAsType(_id, name, generics) => { diff --git a/noir/compiler/noirc_frontend/src/lexer/errors.rs b/noir/compiler/noirc_frontend/src/lexer/errors.rs index a2a4056f1d0..35a07c11e0a 100644 --- a/noir/compiler/noirc_frontend/src/lexer/errors.rs +++ b/noir/compiler/noirc_frontend/src/lexer/errors.rs @@ -17,8 +17,6 @@ pub enum LexerErrorKind { InvalidIntegerLiteral { span: Span, found: String }, #[error("{:?} is not a valid attribute", found)] MalformedFuncAttribute { span: Span, found: String }, - #[error("Integer type is larger than the maximum supported size of u127")] - TooManyBits { span: Span, max: u32, got: u32 }, #[error("Logical and used instead of bitwise and")] LogicalAnd { span: Span }, #[error("Unterminated block comment")] @@ -45,7 +43,6 @@ impl LexerErrorKind { LexerErrorKind::NotADoubleChar { span, .. } => *span, LexerErrorKind::InvalidIntegerLiteral { span, .. } => *span, LexerErrorKind::MalformedFuncAttribute { span, .. } => *span, - LexerErrorKind::TooManyBits { span, .. } => *span, LexerErrorKind::LogicalAnd { span } => *span, LexerErrorKind::UnterminatedBlockComment { span } => *span, LexerErrorKind::UnterminatedStringLiteral { span } => *span, @@ -85,13 +82,6 @@ impl LexerErrorKind { format!(" {found} is not a valid attribute"), *span, ), - LexerErrorKind::TooManyBits { span, max, got } => ( - "Integer literal too large".to_string(), - format!( - "The maximum number of bits needed to represent a field is {max}, This integer type needs {got} bits" - ), - *span, - ), LexerErrorKind::LogicalAnd { span } => ( "Noir has no logical-and (&&) operator since short-circuiting is much less efficient when compiling to circuits".to_string(), "Try `&` instead, or use `if` only if you require short-circuiting".to_string(), diff --git a/noir/compiler/noirc_frontend/src/lexer/lexer.rs b/noir/compiler/noirc_frontend/src/lexer/lexer.rs index fd8168e36c6..cf66ece0c30 100644 --- a/noir/compiler/noirc_frontend/src/lexer/lexer.rs +++ b/noir/compiler/noirc_frontend/src/lexer/lexer.rs @@ -307,7 +307,7 @@ impl<'a> Lexer<'a> { // Check if word an int type // if no error occurred, then it is either a valid integer type or it is not an int type - let parsed_token = IntType::lookup_int_type(&word, Span::inclusive(start, end))?; + let parsed_token = IntType::lookup_int_type(&word)?; // Check if it is an int type if let Some(int_type_token) = parsed_token { diff --git a/noir/compiler/noirc_frontend/src/lexer/token.rs b/noir/compiler/noirc_frontend/src/lexer/token.rs index 5d08ab03ad3..f7c07c5f5db 100644 --- a/noir/compiler/noirc_frontend/src/lexer/token.rs +++ b/noir/compiler/noirc_frontend/src/lexer/token.rs @@ -306,7 +306,7 @@ impl IntType { // XXX: Result // Is not the best API. We could split this into two functions. One that checks if the the // word is a integer, which only returns an Option - pub(crate) fn lookup_int_type(word: &str, span: Span) -> Result, LexerErrorKind> { + pub(crate) fn lookup_int_type(word: &str) -> Result, LexerErrorKind> { // Check if the first string is a 'u' or 'i' let is_signed = if word.starts_with('i') { @@ -324,12 +324,6 @@ impl IntType { Err(_) => return Ok(None), }; - let max_bits = FieldElement::max_num_bits() / 2; - - if str_as_u32 > max_bits { - return Err(LexerErrorKind::TooManyBits { span, max: max_bits, got: str_as_u32 }); - } - if is_signed { Ok(Some(Token::IntType(IntType::Signed(str_as_u32)))) } else { diff --git a/noir/compiler/noirc_frontend/src/monomorphization/ast.rs b/noir/compiler/noirc_frontend/src/monomorphization/ast.rs index 73e7ef372ab..e4e619d5d92 100644 --- a/noir/compiler/noirc_frontend/src/monomorphization/ast.rs +++ b/noir/compiler/noirc_frontend/src/monomorphization/ast.rs @@ -6,7 +6,8 @@ use noirc_errors::{ }; use crate::{ - hir_def::function::FunctionSignature, BinaryOpKind, Distinctness, Signedness, Visibility, + hir_def::function::FunctionSignature, BinaryOpKind, Distinctness, IntegerBitSize, Signedness, + Visibility, }; /// The monomorphized AST is expression-based, all statements are also @@ -217,8 +218,8 @@ pub struct Function { #[derive(Debug, PartialEq, Eq, Clone, Hash)] pub enum Type { Field, - Array(/*len:*/ u64, Box), // Array(4, Field) = [Field; 4] - Integer(Signedness, /*bits:*/ u32), // u32 = Integer(unsigned, 32) + Array(/*len:*/ u64, Box), // Array(4, Field) = [Field; 4] + Integer(Signedness, /*bits:*/ IntegerBitSize), // u32 = Integer(unsigned, ThirtyTwo) Bool, String(/*len:*/ u64), // String(4) = str[4] FmtString(/*len:*/ u64, Box), diff --git a/noir/compiler/noirc_frontend/src/monomorphization/mod.rs b/noir/compiler/noirc_frontend/src/monomorphization/mod.rs index 21c095eb877..f691a0c9065 100644 --- a/noir/compiler/noirc_frontend/src/monomorphization/mod.rs +++ b/noir/compiler/noirc_frontend/src/monomorphization/mod.rs @@ -27,8 +27,8 @@ use crate::{ }, node_interner::{self, DefinitionKind, NodeInterner, StmtId, TraitImplKind, TraitMethodId}, token::FunctionAttribute, - ContractFunctionType, FunctionKind, Type, TypeBinding, TypeBindings, TypeVariable, - TypeVariableKind, UnaryOp, Visibility, + ContractFunctionType, FunctionKind, IntegerBitSize, Type, TypeBinding, TypeBindings, + TypeVariable, TypeVariableKind, UnaryOp, Visibility, }; use self::ast::{Definition, FuncId, Function, LocalId, Program}; @@ -354,6 +354,7 @@ impl<'interner> Monomorphizer<'interner> { match typ { ast::Type::Field => Literal(Integer(-value, typ, location)), ast::Type::Integer(_, bit_size) => { + let bit_size: u32 = bit_size.into(); let base = 1_u128 << bit_size; Literal(Integer(FieldElement::from(base) - value, typ, location)) } @@ -802,12 +803,14 @@ impl<'interner> Monomorphizer<'interner> { // Default any remaining unbound type variables. // This should only happen if the variable in question is unused // and within a larger generic type. - let default = - if self.is_range_loop && matches!(kind, TypeVariableKind::IntegerOrField) { - Type::default_range_loop_type() - } else { - kind.default_type() - }; + let default = if self.is_range_loop + && (matches!(kind, TypeVariableKind::IntegerOrField) + || matches!(kind, TypeVariableKind::Integer)) + { + Type::default_range_loop_type() + } else { + kind.default_type() + }; let monomorphized_default = self.convert_type(&default); binding.bind(default); @@ -820,6 +823,8 @@ impl<'interner> Monomorphizer<'interner> { ast::Type::Tuple(fields) } + HirType::Alias(def, args) => self.convert_type(&def.borrow().get_type(args)), + HirType::Tuple(fields) => { let fields = vecmap(fields, |x| self.convert_type(x)); ast::Type::Tuple(fields) @@ -1109,19 +1114,19 @@ impl<'interner> Monomorphizer<'interner> { } "modulus_le_bits" => { let bits = FieldElement::modulus().to_radix_le(2); - Some(self.modulus_array_literal(bits, 1, location)) + Some(self.modulus_array_literal(bits, IntegerBitSize::One, location)) } "modulus_be_bits" => { let bits = FieldElement::modulus().to_radix_be(2); - Some(self.modulus_array_literal(bits, 1, location)) + Some(self.modulus_array_literal(bits, IntegerBitSize::One, location)) } "modulus_be_bytes" => { let bytes = FieldElement::modulus().to_bytes_be(); - Some(self.modulus_array_literal(bytes, 8, location)) + Some(self.modulus_array_literal(bytes, IntegerBitSize::Eight, location)) } "modulus_le_bytes" => { let bytes = FieldElement::modulus().to_bytes_le(); - Some(self.modulus_array_literal(bytes, 8, location)) + Some(self.modulus_array_literal(bytes, IntegerBitSize::Eight, location)) } _ => None, }; @@ -1133,7 +1138,7 @@ impl<'interner> Monomorphizer<'interner> { fn modulus_array_literal( &self, bytes: Vec, - arr_elem_bits: u32, + arr_elem_bits: IntegerBitSize, location: Location, ) -> ast::Expression { use ast::*; diff --git a/noir/compiler/noirc_frontend/src/node_interner.rs b/noir/compiler/noirc_frontend/src/node_interner.rs index 0051c1b4f5f..9a45268d111 100644 --- a/noir/compiler/noirc_frontend/src/node_interner.rs +++ b/noir/compiler/noirc_frontend/src/node_interner.rs @@ -29,7 +29,7 @@ use crate::hir_def::{ use crate::token::{Attributes, SecondaryAttribute}; use crate::{ BinaryOpKind, ContractFunctionType, FunctionDefinition, FunctionVisibility, Generics, Shared, - TypeAliasType, TypeBindings, TypeVariable, TypeVariableId, TypeVariableKind, + TypeAlias, TypeBindings, TypeVariable, TypeVariableId, TypeVariableKind, }; /// An arbitrary number to limit the recursion depth when searching for trait impls. @@ -90,11 +90,12 @@ pub struct NodeInterner { structs: HashMap>, struct_attributes: HashMap, - // Type Aliases map. + + // Maps TypeAliasId -> Shared // // Map type aliases to the actual type. // When resolving types, check against this map to see if a type alias is defined. - pub(crate) type_aliases: Vec, + pub(crate) type_aliases: Vec>, // Trait map. // @@ -604,13 +605,13 @@ impl NodeInterner { pub fn push_type_alias(&mut self, typ: &UnresolvedTypeAlias) -> TypeAliasId { let type_id = TypeAliasId(self.type_aliases.len()); - self.type_aliases.push(TypeAliasType::new( + self.type_aliases.push(Shared::new(TypeAlias::new( type_id, typ.type_alias_def.name.clone(), Location::new(typ.type_alias_def.span, typ.file_id), Type::Error, vecmap(&typ.type_alias_def.generics, |_| TypeVariable::unbound(TypeVariableId(0))), - )); + ))); type_id } @@ -632,7 +633,7 @@ impl NodeInterner { pub fn set_type_alias(&mut self, type_id: TypeAliasId, typ: Type, generics: Generics) { let type_alias_type = &mut self.type_aliases[type_id.0]; - type_alias_type.set_type_and_generics(typ, generics); + type_alias_type.borrow_mut().set_type_and_generics(typ, generics); } /// Returns the interned statement corresponding to `stmt_id` @@ -957,8 +958,8 @@ impl NodeInterner { self.traits.get(&id) } - pub fn get_type_alias(&self, id: TypeAliasId) -> &TypeAliasType { - &self.type_aliases[id.0] + pub fn get_type_alias(&self, id: TypeAliasId) -> Shared { + self.type_aliases[id.0].clone() } pub fn get_global(&self, global_id: GlobalId) -> &GlobalInfo { @@ -1539,6 +1540,10 @@ impl NodeInterner { self.add_dependency(dependent, DependencyId::Function(dependency)); } + pub fn add_type_alias_dependency(&mut self, dependent: DependencyId, dependency: TypeAliasId) { + self.add_dependency(dependent, DependencyId::Alias(dependency)); + } + fn add_dependency(&mut self, dependent: DependencyId, dependency: DependencyId) { let dependent_index = self.get_or_insert_dependency(dependent); let dependency_index = self.get_or_insert_dependency(dependency); @@ -1585,6 +1590,12 @@ impl NodeInterner { } DependencyId::Alias(alias_id) => { let alias = self.get_type_alias(alias_id); + // If type aliases form a cycle, we have to manually break the cycle + // here to prevent infinite recursion in the type checker. + alias.borrow_mut().typ = Type::Error; + + // push_error will borrow the alias so we have to drop the mutable borrow + let alias = alias.borrow(); push_error(alias.name.to_string(), &scc, i, alias.location); break; } @@ -1606,7 +1617,7 @@ impl NodeInterner { DependencyId::Struct(id) => Cow::Owned(self.get_struct(id).borrow().name.to_string()), DependencyId::Function(id) => Cow::Borrowed(self.function_name(&id)), DependencyId::Alias(id) => { - Cow::Borrowed(self.get_type_alias(id).name.0.contents.as_ref()) + Cow::Owned(self.get_type_alias(id).borrow().name.to_string()) } DependencyId::Global(id) => { Cow::Borrowed(self.get_global(id).ident.0.contents.as_ref()) @@ -1700,6 +1711,7 @@ fn get_type_method_key(typ: &Type) -> Option { Type::Array(_, _) => Some(Array), Type::Integer(_, _) => Some(FieldOrInt), Type::TypeVariable(_, TypeVariableKind::IntegerOrField) => Some(FieldOrInt), + Type::TypeVariable(_, TypeVariableKind::Integer) => Some(FieldOrInt), Type::Bool => Some(Bool), Type::String(_) => Some(String), Type::FmtString(_, _) => Some(FmtString), @@ -1708,6 +1720,7 @@ fn get_type_method_key(typ: &Type) -> Option { Type::Function(_, _, _) => Some(Function), Type::NamedGeneric(_, _) => Some(Generic), Type::MutableReference(element) => get_type_method_key(element), + Type::Alias(alias, _) => get_type_method_key(&alias.borrow().typ), // We do not support adding methods to these types Type::TypeVariable(_, _) diff --git a/noir/compiler/noirc_frontend/src/parser/errors.rs b/noir/compiler/noirc_frontend/src/parser/errors.rs index 9158c68db72..43a1f96f13f 100644 --- a/noir/compiler/noirc_frontend/src/parser/errors.rs +++ b/noir/compiler/noirc_frontend/src/parser/errors.rs @@ -1,6 +1,7 @@ use crate::lexer::errors::LexerErrorKind; use crate::lexer::token::Token; use crate::Expression; +use crate::IntegerBitSize; use small_ord_set::SmallOrdSet; use thiserror::Error; @@ -40,8 +41,8 @@ pub enum ParserErrorReason { NoFunctionAttributesAllowedOnStruct, #[error("Assert statements can only accept string literals")] AssertMessageNotString, - #[error("Integer bit size {0} won't be supported")] - DeprecatedBitSize(u32), + #[error("Integer bit size {0} isn't supported")] + InvalidBitSize(u32), #[error("{0}")] Lexer(LexerErrorKind), } @@ -132,8 +133,6 @@ impl std::fmt::Display for ParserError { } } -pub(crate) static ALLOWED_INTEGER_BIT_SIZES: &[u32] = &[1, 8, 32, 64]; - impl From for Diagnostic { fn from(error: ParserError) -> Diagnostic { match error.reason { @@ -149,9 +148,9 @@ impl From for Diagnostic { "The 'comptime' keyword has been deprecated. It can be removed without affecting your program".into(), error.span, ), - ParserErrorReason::DeprecatedBitSize(bit_size) => Diagnostic::simple_warning( - format!("Use of deprecated bit size {}", bit_size), - format!("Bit sizes for integers will be restricted to {}", ALLOWED_INTEGER_BIT_SIZES.iter().map(|n| n.to_string()).collect::>().join(", ")), + ParserErrorReason::InvalidBitSize(bit_size) => Diagnostic::simple_error( + format!("Use of invalid bit size {}", bit_size), + format!("Allowed bit sizes for integers are {}", IntegerBitSize::allowed_sizes().iter().map(|n| n.to_string()).collect::>().join(", ")), error.span, ), ParserErrorReason::ExperimentalFeature(_) => Diagnostic::simple_warning( diff --git a/noir/compiler/noirc_frontend/src/parser/parser.rs b/noir/compiler/noirc_frontend/src/parser/parser.rs index b1ec18f5ec5..8bcd7670716 100644 --- a/noir/compiler/noirc_frontend/src/parser/parser.rs +++ b/noir/compiler/noirc_frontend/src/parser/parser.rs @@ -23,7 +23,6 @@ //! prevent other parsers from being tried afterward since there is no longer an error. Thus, they should //! be limited to cases like the above `fn` example where it is clear we shouldn't back out of the //! current parser to try alternative parsers in a `choice` expression. -use super::errors::ALLOWED_INTEGER_BIT_SIZES; use super::{ foldl_with_span, labels::ParsingRuleLabel, parameter_name_recovery, parameter_recovery, parenthesized, then_commit, then_commit_ignore, top_level_statement_recovery, ExprParser, @@ -36,7 +35,7 @@ use crate::ast::{ }; use crate::lexer::Lexer; use crate::parser::{force, ignore_then_commit, statement_recovery}; -use crate::token::{Attribute, Attributes, IntType, Keyword, SecondaryAttribute, Token, TokenKind}; +use crate::token::{Attribute, Attributes, Keyword, SecondaryAttribute, Token, TokenKind}; use crate::{ BinaryOp, BinaryOpKind, BlockExpression, ConstrainKind, ConstrainStatement, Distinctness, ForLoopStatement, ForRange, FunctionDefinition, FunctionReturnType, FunctionVisibility, Ident, @@ -1093,19 +1092,14 @@ fn int_type() -> impl NoirParser { Err(ParserError::expected_label(ParsingRuleLabel::IntegerType, unexpected, span)) } })) - .validate(|int_type, span, emit| { - let bit_size = match int_type.1 { - IntType::Signed(bit_size) | IntType::Unsigned(bit_size) => bit_size, - }; - if !ALLOWED_INTEGER_BIT_SIZES.contains(&bit_size) { - emit(ParserError::with_reason( - ParserErrorReason::DeprecatedBitSize(bit_size), - span, - )); - } - int_type + .validate(|(_, token), span, emit| { + UnresolvedTypeData::from_int_token(token) + .map(|data| data.with_span(span)) + .unwrap_or_else(|err| { + emit(ParserError::with_reason(ParserErrorReason::InvalidBitSize(err.0), span)); + UnresolvedType::error(span) + }) }) - .map_with_span(|(_, token), span| UnresolvedTypeData::from_int_token(token).with_span(span)) } fn named_type(type_parser: impl NoirParser) -> impl NoirParser { diff --git a/noir/compiler/noirc_frontend/src/resolve_locations.rs b/noir/compiler/noirc_frontend/src/resolve_locations.rs index cfb88966b9d..b5f1b1d0c64 100644 --- a/noir/compiler/noirc_frontend/src/resolve_locations.rs +++ b/noir/compiler/noirc_frontend/src/resolve_locations.rs @@ -212,6 +212,8 @@ impl NodeInterner { self.type_alias_ref .iter() .find(|(_, named_type_location)| named_type_location.span.contains(&location.span)) - .map(|(type_alias_id, _found_location)| self.get_type_alias(*type_alias_id).location) + .map(|(type_alias_id, _found_location)| { + self.get_type_alias(*type_alias_id).borrow().location + }) } } diff --git a/noir/compiler/noirc_frontend/src/tests.rs b/noir/compiler/noirc_frontend/src/tests.rs index 1deff446d7e..8a56b337398 100644 --- a/noir/compiler/noirc_frontend/src/tests.rs +++ b/noir/compiler/noirc_frontend/src/tests.rs @@ -1184,4 +1184,26 @@ fn lambda$f1(mut env$l1: (Field)) -> Field { "#; assert_eq!(get_program_errors(src).len(), 1); } + + #[test] + fn deny_cyclic_type_aliases() { + let src = r#" + type A = B; + type B = A; + fn main() {} + "#; + assert_eq!(get_program_errors(src).len(), 1); + } + + #[test] + fn ensure_nested_type_aliases_type_check() { + let src = r#" + type A = B; + type B = u8; + fn main() { + let _a: A = 0 as u16; + } + "#; + assert_eq!(get_program_errors(src).len(), 1); + } } diff --git a/noir/cspell.json b/noir/cspell.json index 19e9a175ce0..2acca0633d3 100644 --- a/noir/cspell.json +++ b/noir/cspell.json @@ -106,6 +106,7 @@ "lvalue", "Maddiaa", "mathbb", + "memfs", "merkle", "metas", "minreq", diff --git a/noir/docs/docs/noir/concepts/data_types/index.md b/noir/docs/docs/noir/concepts/data_types/index.md index 3c9cd4c2437..97b3b2cb094 100644 --- a/noir/docs/docs/noir/concepts/data_types/index.md +++ b/noir/docs/docs/noir/concepts/data_types/index.md @@ -91,6 +91,20 @@ fn main() { } ``` +Type aliases can even refer to other aliases. An error will be issued if they form a cycle: + +```rust +// Ok! +type A = B; +type B = Field; + +type Bad1 = Bad2; + +// error: Dependency cycle found +type Bad2 = Bad1; +// ^^^^^^^^^^^ 'Bad2' recursively depends on itself: Bad2 -> Bad1 -> Bad2 +``` + ### BigInt You can achieve BigInt functionality using the [Noir BigInt](https://github.com/shuklaayush/noir-bigint) library. diff --git a/noir/docs/docs/noir/concepts/data_types/integers.md b/noir/docs/docs/noir/concepts/data_types/integers.md index 30135d76e4a..4d58d96fed5 100644 --- a/noir/docs/docs/noir/concepts/data_types/integers.md +++ b/noir/docs/docs/noir/concepts/data_types/integers.md @@ -5,7 +5,7 @@ keywords: [noir, integer types, methods, examples, arithmetic] sidebar_position: 1 --- -An integer type is a range constrained field type. The Noir frontend supports arbitrarily-sized, both unsigned and signed integer types. +An integer type is a range constrained field type. The Noir frontend supports both unsigned and signed integer types. The allowed sizes are 1, 8, 32 and 64 bits. :::info @@ -45,13 +45,6 @@ fn main() { The bit size determines the maximum and minimum range of value the integer type can store. For example, an `i8` variable can store a value in the range of -128 to 127 (i.e. $\\-2^{7}\\$ to $\\2^{7}-1\\$). -:::tip - -If you are using the default proving backend with Noir, both even (e.g. _u2_, _i2_) and odd (e.g. _u3_, _i3_) arbitrarily-sized integer types up to 127 bits (i.e. _u127_ and _i127_) are supported. - -::: - - ## 128 bits Unsigned Integers The built-in structure `U128` allows you to use 128-bit unsigned integers almost like a native integer type. However, there are some differences to keep in mind: diff --git a/noir/test_programs/compile_failure/integer_literal_overflow/src/main.nr b/noir/test_programs/compile_failure/integer_literal_overflow/src/main.nr index d89505c0085..e4d21b5c3b9 100644 --- a/noir/test_programs/compile_failure/integer_literal_overflow/src/main.nr +++ b/noir/test_programs/compile_failure/integer_literal_overflow/src/main.nr @@ -2,4 +2,4 @@ fn main() { foo(1234) } -fn foo(_x: u4) {} +fn foo(_x: u8) {} diff --git a/noir/test_programs/execution_success/regression_2854/Nargo.toml b/noir/test_programs/compile_failure/restricted_bit_sizes/Nargo.toml similarity index 63% rename from noir/test_programs/execution_success/regression_2854/Nargo.toml rename to noir/test_programs/compile_failure/restricted_bit_sizes/Nargo.toml index fb2b3c42fdd..36f8253e8e7 100644 --- a/noir/test_programs/execution_success/regression_2854/Nargo.toml +++ b/noir/test_programs/compile_failure/restricted_bit_sizes/Nargo.toml @@ -1,6 +1,5 @@ [package] -name = "regression_2854" +name = "restricted_bit_sizes" type = "bin" authors = [""] - [dependencies] diff --git a/noir/test_programs/compile_failure/restricted_bit_sizes/src/main.nr b/noir/test_programs/compile_failure/restricted_bit_sizes/src/main.nr new file mode 100644 index 00000000000..01e72bfcfd7 --- /dev/null +++ b/noir/test_programs/compile_failure/restricted_bit_sizes/src/main.nr @@ -0,0 +1,5 @@ +use dep::std::assert_constant; + +fn main() -> pub u63 { + 5 +} diff --git a/noir/test_programs/execution_success/regression_2854/Prover.toml b/noir/test_programs/execution_success/regression_2854/Prover.toml deleted file mode 100644 index 07890234a19..00000000000 --- a/noir/test_programs/execution_success/regression_2854/Prover.toml +++ /dev/null @@ -1 +0,0 @@ -x = "3" diff --git a/noir/test_programs/execution_success/regression_2854/src/main.nr b/noir/test_programs/execution_success/regression_2854/src/main.nr deleted file mode 100644 index eccff8225b6..00000000000 --- a/noir/test_programs/execution_success/regression_2854/src/main.nr +++ /dev/null @@ -1,3 +0,0 @@ -fn main(x: Field) -> pub i127 { - x as i127 -} diff --git a/noir/tooling/noirc_abi/src/lib.rs b/noir/tooling/noirc_abi/src/lib.rs index 1fc257c1676..26feab65d83 100644 --- a/noir/tooling/noirc_abi/src/lib.rs +++ b/noir/tooling/noirc_abi/src/lib.rs @@ -142,14 +142,13 @@ impl AbiType { Signedness::Signed => Sign::Signed, }; - Self::Integer { sign, width: *bit_width } - } - Type::TypeVariable(binding, TypeVariableKind::IntegerOrField) => { - match &*binding.borrow() { - TypeBinding::Bound(typ) => Self::from_type(context, typ), - TypeBinding::Unbound(_) => Self::from_type(context, &Type::default_int_type()), - } + Self::Integer { sign, width: (*bit_width).into() } } + Type::TypeVariable(binding, TypeVariableKind::IntegerOrField) + | Type::TypeVariable(binding, TypeVariableKind::Integer) => match &*binding.borrow() { + TypeBinding::Bound(typ) => Self::from_type(context, typ), + TypeBinding::Unbound(_) => Self::from_type(context, &Type::default_int_type()), + }, Type::Bool => Self::Boolean, Type::String(size) => { let size = size @@ -158,7 +157,7 @@ impl AbiType { Self::String { length: size } } - Type::Struct(def, ref args) => { + Type::Struct(def, args) => { let struct_type = def.borrow(); let fields = struct_type.get_fields(args); let fields = vecmap(fields, |(name, typ)| (name, Self::from_type(context, &typ))); @@ -167,6 +166,7 @@ impl AbiType { context.fully_qualified_struct_path(context.root_crate_id(), struct_type.id); Self::Struct { fields, path } } + Type::Alias(def, args) => Self::from_type(context, &def.borrow().get_type(args)), Type::Tuple(fields) => { let fields = vecmap(fields, |typ| Self::from_type(context, typ)); Self::Tuple { fields } From 518ec4151111c2879ae2466ebdda34c0cebfe14a Mon Sep 17 00:00:00 2001 From: sirasistant Date: Thu, 15 Feb 2024 08:45:44 +0000 Subject: [PATCH 2/4] wip: restricting bit sizes --- .../contracts/references/storage/main.md | 1 - .../contracts/resources/dependencies.md | 8 - docs/docs/developers/contracts/setup.md | 3 +- .../writing_dapp/contract_deployment.md | 1 - .../tutorials/writing_token_contract.md | 1 - noir-projects/aztec-nr/README.md | 1 - .../aztec-nr/aztec/src/note/note_getter.nr | 18 +- .../aztec/src/note/note_getter_options.nr | 50 ++- .../aztec/src/note/note_viewer_options.nr | 14 +- .../aztec-nr/aztec/src/oracle/notes.nr | 28 +- .../src/easy_private_state.nr | 35 +- noir-projects/aztec-nr/safe-math/Nargo.toml | 8 - noir-projects/aztec-nr/safe-math/src/lib.nr | 3 - .../aztec-nr/safe-math/src/safe_u120.nr | 340 ------------------ .../aztec-nr/value-note/src/filter.nr | 6 +- .../aztec-nr/value-note/src/utils.nr | 7 +- .../contracts/card_game_contract/src/cards.nr | 101 ++---- .../contracts/card_game_contract/src/game.nr | 11 +- .../contracts/counter_contract/src/main.nr | 18 +- .../docs_example_contract/src/main.nr | 37 +- .../easy_private_token_contract/src/main.nr | 20 +- .../contracts/lending_contract/Nargo.toml | 1 - .../contracts/lending_contract/src/asset.nr | 21 +- .../contracts/lending_contract/src/helpers.nr | 50 ++- .../lending_contract/src/interest_math.nr | 2 - .../lending_contract/src/interfaces.nr | 175 +++++---- .../contracts/lending_contract/src/main.nr | 26 +- .../token_blacklist_contract/Nargo.toml | 1 - .../token_blacklist_contract/src/main.nr | 1 - .../src/types/balances_map.nr | 1 - .../src/types/token_note.nr | 1 - .../contracts/token_contract/Nargo.toml | 1 - .../contracts/token_contract/src/main.nr | 1 - .../token_contract/src/types/balances_map.nr | 1 - .../token_contract/src/types/token_note.nr | 1 - .../src/crates/types/src/traits.nr | 8 +- .../crates/types/src/type_serialization.nr | 13 + 37 files changed, 285 insertions(+), 730 deletions(-) delete mode 100644 noir-projects/aztec-nr/safe-math/Nargo.toml delete mode 100644 noir-projects/aztec-nr/safe-math/src/lib.nr delete mode 100644 noir-projects/aztec-nr/safe-math/src/safe_u120.nr diff --git a/docs/docs/developers/contracts/references/storage/main.md b/docs/docs/developers/contracts/references/storage/main.md index 6bc1f653915..b776eeb6f9d 100644 --- a/docs/docs/developers/contracts/references/storage/main.md +++ b/docs/docs/developers/contracts/references/storage/main.md @@ -59,7 +59,6 @@ type = "contract" [dependencies] aztec = { git="https://github.com/AztecProtocol/aztec-packages/", tag="#include_aztec_version", directory="noir-projects/aztec-nr/aztec" } -safe_math = { git="https://github.com/AztecProtocol/aztec-packages/", tag="#include_aztec_version", directory="noir-projects/aztec-nr/safe-math"} authwit={ git="https://github.com/AztecProtocol/aztec-packages/", tag="#include_aztec_version", directory="noir-projects/aztec-nr/authwit"} compressed_string = {git="https://github.com/AztecProtocol/aztec-packages/", tag="#include_aztec_version", directory="noir-projects/aztec-nr/compressed-string"} ``` diff --git a/docs/docs/developers/contracts/resources/dependencies.md b/docs/docs/developers/contracts/resources/dependencies.md index fbee858c1da..02debb3a898 100644 --- a/docs/docs/developers/contracts/resources/dependencies.md +++ b/docs/docs/developers/contracts/resources/dependencies.md @@ -44,14 +44,6 @@ protocol_types = { git="https://github.com/AztecProtocol/aztec-packages/", tag=" This library contains types that are used in the Aztec protocol. Find it on [GitHub](https://github.com/AztecProtocol/aztec-packages/tree/master/noir-projects/noir-protocol-circuits/src/crates/types/src). -## Safe math - -```toml -safe_math = { git="https://github.com/AztecProtocol/aztec-packages/", tag="#include_aztec_version", directory="noir-projects/aztec-nr/safe-math" } -``` - -This is a library for safe arithmetic, similar to OpenZeppelin's safe math library. Find it on [GitHub](https://github.com/AztecProtocol/aztec-packages/tree/master/noir-projects/aztec-nr/safe-math). - ## Value note ```toml diff --git a/docs/docs/developers/contracts/setup.md b/docs/docs/developers/contracts/setup.md index 7ea682a2bc5..efe8bd57a3d 100644 --- a/docs/docs/developers/contracts/setup.md +++ b/docs/docs/developers/contracts/setup.md @@ -57,7 +57,7 @@ Your folder should look like: Before writing the contracts, we must add the aztec.nr library. This adds smart contract utility functions for interacting with the Aztec network. -3. Finally, add relevant aztec-nr dependencies that you might use such as `aztec.nr`, `value_note` and `safe_math` libraries. +3. Finally, add relevant aztec-nr dependencies that you might use such as `aztec.nr` and `value_note` libraries. Open Nargo.toml that is in the `contracts/example_contract` folder, and add the dependency section as follows @@ -74,7 +74,6 @@ aztec = { git="https://github.com/AztecProtocol/aztec-packages/", tag="#include_ # Utility dependencies value_note = { git="https://github.com/AztecProtocol/aztec-packages/", tag="#include_aztec_version", directory="noir-projects/aztec-nr/value-note"} -safe_math = { git="https://github.com/AztecProtocol/aztec-packages/", tag="#include_aztec_version", directory="noir-projects/aztec-nr/safe-math"} ``` :::info diff --git a/docs/docs/developers/tutorials/writing_dapp/contract_deployment.md b/docs/docs/developers/tutorials/writing_dapp/contract_deployment.md index 79cb2293882..8cbaecbc608 100644 --- a/docs/docs/developers/tutorials/writing_dapp/contract_deployment.md +++ b/docs/docs/developers/tutorials/writing_dapp/contract_deployment.md @@ -21,7 +21,6 @@ Then, open the `contracts/token/Nargo.toml` configuration file, and add the `azt [dependencies] aztec = { git="https://github.com/AztecProtocol/aztec-packages/", tag="#include_aztec_version", directory="noir-projects/aztec-nr/aztec" } authwit = { git="https://github.com/AztecProtocol/aztec-packages/", tag="#include_aztec_version", directory="noir-projects/aztec-nr/authwit"} -safe_math = { git="https://github.com/AztecProtocol/aztec-packages/", tag="#include_aztec_version", directory="noir-projects/aztec-nr/safe-math"} compressed_string = {git="https://github.com/AztecProtocol/aztec-packages/", tag="#include_aztec_version", directory="noir-projects/aztec-nr/compressed-string"} ``` diff --git a/docs/docs/developers/tutorials/writing_token_contract.md b/docs/docs/developers/tutorials/writing_token_contract.md index 7a25ad513f5..3ea5bc2b2b3 100644 --- a/docs/docs/developers/tutorials/writing_token_contract.md +++ b/docs/docs/developers/tutorials/writing_token_contract.md @@ -59,7 +59,6 @@ type = "contract" [dependencies] aztec = { git="https://github.com/AztecProtocol/aztec-packages/", tag="#include_aztec_version", directory="noir-projects/aztec-nr/aztec" } -safe_math = { git="https://github.com/AztecProtocol/aztec-packages/", tag="#include_aztec_version", directory="noir-projects/aztec-nr/safe-math"} authwit={ git="https://github.com/AztecProtocol/aztec-packages/", tag="#include_aztec_version", directory="noir-projects/aztec-nr/authwit"} compressed_string = {git="https://github.com/AztecProtocol/aztec-packages/", tag="#include_aztec_version", directory="noir-projects/aztec-nr/compressed-string"} ``` diff --git a/noir-projects/aztec-nr/README.md b/noir-projects/aztec-nr/README.md index 31b4527801c..3a944619c61 100644 --- a/noir-projects/aztec-nr/README.md +++ b/noir-projects/aztec-nr/README.md @@ -44,7 +44,6 @@ aztec = { git = "https://github.com/AztecProtocol/aztec-nr", tag = "master" , di # Optional libraries easy_private_state = { git = "https://github.com/AztecProtocol/aztec-nr", tag = "master" , directory = "easy-private-state" } -safe_math = { git = "https://github.com/AztecProtocol/aztec-nr", tag = "master" , directory = "safe-math" } value_note = { git = "https://github.com/AztecProtocol/aztec-nr", tag = "master" , directory = "value-note" } ``` diff --git a/noir-projects/aztec-nr/aztec/src/note/note_getter.nr b/noir-projects/aztec-nr/aztec/src/note/note_getter.nr index 4591211c4de..23237bb805f 100644 --- a/noir-projects/aztec-nr/aztec/src/note/note_getter.nr +++ b/noir-projects/aztec-nr/aztec/src/note/note_getter.nr @@ -1,19 +1,15 @@ use dep::std::option::Option; use dep::protocol_types::{ constants::{ - MAX_READ_REQUESTS_PER_CALL, - GET_NOTE_ORACLE_RETURN_LENGTH, - GET_NOTES_ORACLE_RETURN_LENGTH, - MAX_NOTES_PER_PAGE, - VIEW_NOTE_ORACLE_RETURN_LENGTH, - }, + MAX_READ_REQUESTS_PER_CALL, GET_NOTE_ORACLE_RETURN_LENGTH, GET_NOTES_ORACLE_RETURN_LENGTH, + MAX_NOTES_PER_PAGE, VIEW_NOTE_ORACLE_RETURN_LENGTH +} }; use crate::context::PrivateContext; use crate::note::{ note_getter_options::{NoteGetterOptions, Select, Sort, SortOrder, Comparator, NoteStatus}, - note_interface::NoteInterface, - note_viewer_options::NoteViewerOptions, - utils::compute_note_hash_for_consumption, + note_interface::NoteInterface, note_viewer_options::NoteViewerOptions, + utils::compute_note_hash_for_consumption }; use crate::oracle; @@ -60,7 +56,7 @@ fn check_notes_order( for i in 0..sorts.len { let sort = sorts.get_unchecked(i).unwrap_unchecked(); let eq = fields_0[sort.field_index] == fields_1[sort.field_index]; - let lt = fields_0[sort.field_index] as u120 < fields_1[sort.field_index] as u120; + let lt = fields_0[sort.field_index].lt(fields_1[sort.field_index]); if sort.order == SortOrder.ASC { assert(eq | lt, "Return notes not sorted in ascending order."); } else if !eq { @@ -195,7 +191,7 @@ unconstrained pub fn view_notes( unconstrained fn flatten_options( selects: BoundedVec, N>, sorts: BoundedVec, N> -) -> (u8, [u8; N], [Field; N], [u3; N], [u8; N], [u2; N]) { +) -> (u8, [u8; N], [Field; N], [u8; N], [u8; N], [u8; N]) { let mut num_selects = 0; let mut select_by = [0; N]; let mut select_values = [0; N]; diff --git a/noir-projects/aztec-nr/aztec/src/note/note_getter_options.nr b/noir-projects/aztec-nr/aztec/src/note/note_getter_options.nr index 8b7dc2fe706..bb013ef946e 100644 --- a/noir-projects/aztec-nr/aztec/src/note/note_getter_options.nr +++ b/noir-projects/aztec-nr/aztec/src/note/note_getter_options.nr @@ -1,16 +1,14 @@ use dep::std::option::Option; -use dep::protocol_types::{ - constants::MAX_READ_REQUESTS_PER_CALL, -}; +use dep::protocol_types::{constants::MAX_READ_REQUESTS_PER_CALL}; use crate::note::note_interface::NoteInterface; struct ComparatorEnum { - EQ: u3, - NEQ: u3, - LT: u3, - LTE: u3, - GT: u3, - GTE: u3, + EQ: u8, + NEQ: u8, + LT: u8, + LTE: u8, + GT: u8, + GTE: u8, } global Comparator = ComparatorEnum { @@ -25,18 +23,18 @@ global Comparator = ComparatorEnum { struct Select { field_index: u8, value: Field, - comparator: u3, + comparator: u8, } impl Select { - pub fn new(field_index: u8, value: Field, comparator: u3) -> Self { + pub fn new(field_index: u8, value: Field, comparator: u8) -> Self { Select { field_index, value, comparator } } } struct SortOrderEnum { - DESC: u2, - ASC: u2, + DESC: u8, + ASC: u8, } global SortOrder = SortOrderEnum { @@ -46,18 +44,18 @@ global SortOrder = SortOrderEnum { struct Sort { field_index: u8, - order: u2, + order: u8, } impl Sort { - pub fn new(field_index: u8, order: u2) -> Self { + pub fn new(field_index: u8, order: u8) -> Self { Sort { field_index, order } } } struct NoteStatusEnum { - ACTIVE: u2, - ACTIVE_OR_NULLIFIED: u2, + ACTIVE: u8, + ACTIVE_OR_NULLIFIED: u8, } global NoteStatus = NoteStatusEnum { @@ -81,7 +79,7 @@ struct NoteGetterOptions { offset: u32, filter: fn ([Option; MAX_READ_REQUESTS_PER_CALL], FILTER_ARGS) -> [Option; MAX_READ_REQUESTS_PER_CALL], filter_args: FILTER_ARGS, - status: u2, + status: u8, } // docs:end:NoteGetterOptions @@ -99,24 +97,24 @@ impl NoteGetterOptions { offset: 0, filter: return_all_notes, filter_args: 0, - status: NoteStatus.ACTIVE, + status: NoteStatus.ACTIVE } } // This function initializes a NoteGetterOptions with a filter, which takes the notes returned from the database and filter_args as its parameters. // `filter_args` allows you to provide additional data or context to the custom filter. pub fn with_filter( - filter: fn ([Option; MAX_READ_REQUESTS_PER_CALL], FILTER_ARGS) -> [Option; MAX_READ_REQUESTS_PER_CALL], - filter_args: FILTER_ARGS, + filter: fn([Option; MAX_READ_REQUESTS_PER_CALL], FILTER_ARGS) -> [Option; MAX_READ_REQUESTS_PER_CALL], + filter_args: FILTER_ARGS ) -> Self where Note: NoteInterface { - NoteGetterOptions { + NoteGetterOptions { selects: BoundedVec::new(Option::none()), sorts: BoundedVec::new(Option::none()), limit: MAX_READ_REQUESTS_PER_CALL as u32, offset: 0, filter, filter_args, - status: NoteStatus.ACTIVE, + status: NoteStatus.ACTIVE } } @@ -124,14 +122,14 @@ impl NoteGetterOptions { // It takes a field_index indicating which field to select, // a value representing the specific value to match in that field, and // a comparator (For possible values of comparators, please see the Comparator enum above) - pub fn select(&mut self, field_index: u8, value: Field, comparator: Option) -> Self { + pub fn select(&mut self, field_index: u8, value: Field, comparator: Option) -> Self { self.selects.push(Option::some(Select::new(field_index, value, comparator.unwrap_or(Comparator.EQ)))); *self } // This method adds a `Sort` criterion to the options. // It takes a field_index indicating which field to sort by and an order (SortOrder) to determine the sorting direction. - pub fn sort(&mut self, field_index: u8, order: u2) -> Self { + pub fn sort(&mut self, field_index: u8, order: u8) -> Self { self.sorts.push(Option::some(Sort::new(field_index, order))); *self } @@ -150,7 +148,7 @@ impl NoteGetterOptions { } // This method sets the status value, which determines whether to retrieve active or nullified notes. - pub fn set_status(&mut self, status: u2) -> Self { + pub fn set_status(&mut self, status: u8) -> Self { self.status = status; *self } diff --git a/noir-projects/aztec-nr/aztec/src/note/note_viewer_options.nr b/noir-projects/aztec-nr/aztec/src/note/note_viewer_options.nr index d778927e744..a291b8f8c5d 100644 --- a/noir-projects/aztec-nr/aztec/src/note/note_viewer_options.nr +++ b/noir-projects/aztec-nr/aztec/src/note/note_viewer_options.nr @@ -1,8 +1,6 @@ use dep::std::option::Option; use crate::note::note_getter_options::{Select, Sort, Comparator, NoteStatus}; -use dep::protocol_types::{ - constants::MAX_NOTES_PER_PAGE, -}; +use dep::protocol_types::{constants::MAX_NOTES_PER_PAGE}; use crate::note::note_interface::NoteInterface; // docs:start:NoteViewerOptions @@ -11,7 +9,7 @@ struct NoteViewerOptions { sorts: BoundedVec, N>, limit: u32, offset: u32, - status: u2, + status: u8, } // docs:end:NoteViewerOptions @@ -22,7 +20,7 @@ impl NoteViewerOptions { sorts: BoundedVec::new(Option::none()), limit: MAX_NOTES_PER_PAGE as u32, offset: 0, - status: NoteStatus.ACTIVE, + status: NoteStatus.ACTIVE } } @@ -30,12 +28,12 @@ impl NoteViewerOptions { // It takes a field_index indicating which field to select, // a value representing the specific value to match in that field, and // a comparator (For possible values of comparators, please see the Comparator enum from note_getter_options) - pub fn select(&mut self, field_index: u8, value: Field, comparator: Option) -> Self { + pub fn select(&mut self, field_index: u8, value: Field, comparator: Option) -> Self { self.selects.push(Option::some(Select::new(field_index, value, comparator.unwrap_or(Comparator.EQ)))); *self } - pub fn sort(&mut self, field_index: u8, order: u2) -> Self { + pub fn sort(&mut self, field_index: u8, order: u8) -> Self { self.sorts.push(Option::some(Sort::new(field_index, order))); *self } @@ -52,7 +50,7 @@ impl NoteViewerOptions { } // This method sets the status value, which determines whether to retrieve active or nullified notes. - pub fn set_status(&mut self, status: u2) -> Self { + pub fn set_status(&mut self, status: u8) -> Self { self.status = status; *self } diff --git a/noir-projects/aztec-nr/aztec/src/oracle/notes.nr b/noir-projects/aztec-nr/aztec/src/oracle/notes.nr index 42b2f356ec8..6b91b35392e 100644 --- a/noir-projects/aztec-nr/aztec/src/oracle/notes.nr +++ b/noir-projects/aztec-nr/aztec/src/oracle/notes.nr @@ -1,13 +1,7 @@ use dep::std::option::Option; -use crate::note::{ - note_header::NoteHeader, - note_interface::NoteInterface, -}; +use crate::note::{note_header::NoteHeader, note_interface::NoteInterface}; -use dep::protocol_types::{ - address::AztecAddress, - utils::arr_copy_slice, -}; +use dep::protocol_types::{address::AztecAddress, utils::arr_copy_slice}; #[oracle(notifyCreatedNote)] fn notify_created_note_oracle( @@ -39,12 +33,12 @@ fn get_notes_oracle( _num_selects: u8, _select_by: [u8; N], _select_values: [Field; N], - _select_comparators: [u3; N], + _select_comparators: [u8; N], _sort_by: [u8; N], - _sort_order: [u2; N], + _sort_order: [u8; N], _limit: u32, _offset: u32, - _status: u2, + _status: u8, _return_size: u32, _placeholder_fields: [Field; S] ) -> [Field; S] {} @@ -54,12 +48,12 @@ unconstrained fn get_notes_oracle_wrapper( num_selects: u8, select_by: [u8; N], select_values: [Field; N], - select_comparators: [u3; N], + select_comparators: [u8; N], sort_by: [u8; N], - sort_order: [u2; N], + sort_order: [u8; N], limit: u32, offset: u32, - status: u2, + status: u8, mut placeholder_fields: [Field; S] ) -> [Field; S] { let return_size = placeholder_fields.len() as u32; @@ -84,12 +78,12 @@ unconstrained pub fn get_notes( num_selects: u8, select_by: [u8; M], select_values: [Field; M], - select_comparators: [u3; M], + select_comparators: [u8; M], sort_by: [u8; M], - sort_order: [u2; M], + sort_order: [u8; M], limit: u32, offset: u32, - status: u2, + status: u8, mut placeholder_opt_notes: [Option; S], // TODO: Remove it and use `limit` to initialize the note array. placeholder_fields: [Field; NS], // TODO: Remove it and use `limit` to initialize the note array. _placeholder_note_length: [Field; N] // Turbofish hack? Compiler breaks calculating read_offset unless we add this parameter diff --git a/noir-projects/aztec-nr/easy-private-state/src/easy_private_state.nr b/noir-projects/aztec-nr/easy-private-state/src/easy_private_state.nr index d6b6fbf650f..93228eaa888 100644 --- a/noir-projects/aztec-nr/easy-private-state/src/easy_private_state.nr +++ b/noir-projects/aztec-nr/easy-private-state/src/easy_private_state.nr @@ -1,13 +1,8 @@ use dep::aztec::{ - protocol_types::address::AztecAddress, - context::Context, - note::note_getter_options::NoteGetterOptions, - state_vars::set::Set, -}; -use dep::value_note::{ - filter::filter_notes_min_sum, - value_note::ValueNote, + protocol_types::address::AztecAddress, context::Context, + note::note_getter_options::NoteGetterOptions, state_vars::set::Set }; +use dep::value_note::{filter::filter_notes_min_sum, value_note::ValueNote}; struct EasyPrivateUint { context: Context, @@ -16,24 +11,14 @@ struct EasyPrivateUint { } impl EasyPrivateUint { - pub fn new( - context: Context, - storage_slot: Field, - ) -> Self { + pub fn new(context: Context, storage_slot: Field) -> Self { assert(storage_slot != 0, "Storage slot 0 not allowed. Storage slots must start from 1."); - let set = Set { - context, - storage_slot - }; - EasyPrivateUint { - context, - set, - storage_slot, - } + let set = Set { context, storage_slot }; + EasyPrivateUint { context, set, storage_slot } } // Very similar to `value_note::utils::increment`. - pub fn add(self, addend: u120, owner: AztecAddress) { + pub fn add(self, addend: u64, owner: AztecAddress) { // Creates new note for the owner. let mut addend_note = ValueNote::new(addend as Field, owner); @@ -44,13 +29,13 @@ impl EasyPrivateUint { } // Very similar to `value_note::utils::decrement`. - pub fn sub(self, subtrahend: u120, owner: AztecAddress) { + pub fn sub(self, subtrahend: u64, owner: AztecAddress) { // docs:start:get_notes let options = NoteGetterOptions::with_filter(filter_notes_min_sum, subtrahend as Field); let maybe_notes = self.set.get_notes(options); // docs:end:get_notes - let mut minuend: u120 = 0; + let mut minuend: u64 = 0; for i in 0..maybe_notes.len() { if maybe_notes[i].is_some() { let note = maybe_notes[i].unwrap_unchecked(); @@ -64,7 +49,7 @@ impl EasyPrivateUint { self.set.remove(note); // docs:end:remove - minuend += note.value as u120; + minuend += note.value as u64; } } diff --git a/noir-projects/aztec-nr/safe-math/Nargo.toml b/noir-projects/aztec-nr/safe-math/Nargo.toml deleted file mode 100644 index aaa75e68977..00000000000 --- a/noir-projects/aztec-nr/safe-math/Nargo.toml +++ /dev/null @@ -1,8 +0,0 @@ -[package] -name = "safe_math" -authors = [""] -compiler_version = ">=0.18.0" -type = "lib" - -[dependencies] -aztec = { path = "../aztec" } \ No newline at end of file diff --git a/noir-projects/aztec-nr/safe-math/src/lib.nr b/noir-projects/aztec-nr/safe-math/src/lib.nr deleted file mode 100644 index 79dd63eb4b8..00000000000 --- a/noir-projects/aztec-nr/safe-math/src/lib.nr +++ /dev/null @@ -1,3 +0,0 @@ -mod safe_u120; - -use crate::safe_u120::{SafeU120, SAFE_U120_SERIALIZED_LEN}; diff --git a/noir-projects/aztec-nr/safe-math/src/safe_u120.nr b/noir-projects/aztec-nr/safe-math/src/safe_u120.nr deleted file mode 100644 index 74acd0db57d..00000000000 --- a/noir-projects/aztec-nr/safe-math/src/safe_u120.nr +++ /dev/null @@ -1,340 +0,0 @@ -use dep::std::cmp::Eq; -use dep::aztec::protocol_types::traits::{Deserialize, Serialize}; - -struct SafeU120 { - value: u120, -} - -impl Eq for SafeU120 { - fn eq( - self: Self, - other: Self - ) -> bool { - self.value == other.value - } -} - -global SAFE_U120_SERIALIZED_LEN: Field = 1; - -impl Serialize for SafeU120 { - fn serialize(value: SafeU120) -> [Field; SAFE_U120_SERIALIZED_LEN] { - [value.value as Field] - } -} - -impl Deserialize for SafeU120 { - // This is safe when reading from storage IF only correct safeu120 was written to storage - fn deserialize(fields: [Field; SAFE_U120_SERIALIZED_LEN]) -> SafeU120 { - let value = fields[0] as u120; - SafeU120 { value } - } -} - -impl SafeU120 { - pub fn min() -> Self { - Self { - value: 0 - } - } - - pub fn max() -> Self { - Self { - value: 0xffffffffffffffffffffffffffffff - } - } - - pub fn new( - value: Field, - ) -> Self { - // Check that it actually will fit. Spending a lot of constraints here :grimacing: - let bytes = value.to_be_bytes(32); - for i in 0..17 { - assert(bytes[i] == 0, "Value too large for SafeU120"); - } - - let value = value as u120; - Self { value } - } - - pub fn is_zero( - self: Self, - ) -> bool { - self.value == 0 - } - - pub fn lt(self: Self, other: Self) -> bool { - self.value < other.value - } - - pub fn le(self: Self, other: Self) -> bool { - self.value <= other.value - } - - pub fn gt(self: Self, other: Self) -> bool { - self.value > other.value - } - - pub fn ge(self: Self, other: Self) -> bool { - self.value >= other.value - } - - pub fn sub( - self: Self, - b: Self, - ) -> Self { - assert(self.value >= b.value, "Underflow"); - Self { - value: self.value - b.value - } - } - - pub fn add( - self: Self, - b: Self, - ) -> Self { - let c: u120 = self.value + b.value; - assert(c >= self.value, "Overflow"); - Self { - value: c - } - } - - pub fn mul( - self: Self, - b: Self, - ) -> Self { - let c: u120 = self.value * b.value; - if !b.is_zero() { - assert(c / b.value == self.value, "Overflow"); - } - Self { - value: c - } - } - - pub fn div( - self: Self, - b: Self, - ) -> Self { - assert(!b.is_zero(), "Divide by zero"); - Self { - value: self.value / b.value - } - } - - pub fn mul_div( - self: Self, - b: Self, - divisor: Self - ) -> Self { - self.mul(b).div(divisor) - } - - pub fn mul_div_up( - self: Self, - b: Self, - divisor: Self - ) -> Self { - let c = self.mul(b); - assert(!divisor.is_zero(), "Divide by zero"); - let adder = ((self.value * b.value % divisor.value) as u120 > 0) as u120; - c.div(divisor).add(Self {value: adder}) - } - - // todo: implement mul_div with 240 bit intermediate values. -} - -#[test] -fn test_init() { - let a = SafeU120::new(1); - assert(a.value == 1); -} - -#[test] -fn test_init_max() { - let a = SafeU120::max(); - assert(a.value == 0xffffffffffffffffffffffffffffff); -} - -#[test] -fn test_init_min() { - let a = SafeU120::min(); - assert(a.value == 0); -} - -#[test] -fn test_is_zero() { - let a = SafeU120::min(); - assert(a.value == 0); - assert(a.is_zero() == true); -} - -#[test] -fn test_eq() { - let a = SafeU120::new(1); - let b = SafeU120::new(1); - assert(a.eq(b)); -} - -#[test] -fn test_lt() { - let a = SafeU120::new(1); - let b = SafeU120::new(2); - assert(a.lt(b)); - assert(b.lt(a) == false); -} - -#[test] -fn test_le() { - let a = SafeU120::new(2); - let b = SafeU120::new(2); - let c = SafeU120::new(5); - assert(a.le(b)); - assert(a.le(c)); - assert(c.le(a) == false); -} - -#[test] -fn test_gt() { - let a = SafeU120::new(1); - let b = SafeU120::new(2); - assert(b.gt(a)); - assert(a.gt(b) == false); -} - -#[test] -fn test_ge() { - let a = SafeU120::new(2); - let b = SafeU120::new(2); - let c = SafeU120::new(5); - assert(a.ge(b)); - assert(a.ge(c) == false); - assert(c.ge(a)); -} - -#[test(should_fail)] -fn test_init_too_large() { - let b = SafeU120::max().value as Field + 1; // max + 1 - let _a = SafeU120::new(b); -} - -#[test] -fn test_add() { - let a = SafeU120::new(1); - let b = SafeU120::new(2); - let c = SafeU120::add(a, b); - assert(c.value == 3); -} - -#[test(should_fail)] -fn test_add_overflow() { - let a = SafeU120::max(); - let b = SafeU120::new(1); - let _c = SafeU120::add(a, b); -} - -#[test] -fn test_sub() { - let a = SafeU120::new(2); - let b = SafeU120::new(1); - let c = SafeU120::sub(a, b); - assert(c.value == 1); -} - -#[test(should_fail)] -fn test_sub_underflow() { - let a = SafeU120::new(1); - let b = SafeU120::new(2); - let _c = SafeU120::sub(a, b); -} - -#[test] -fn test_mul() { - let a = SafeU120::new(2); - let b = SafeU120::new(3); - let c = SafeU120::mul(a, b); - assert(c.value == 6); -} - -#[test(should_fail)] -fn test_mul_overflow() { - let a = SafeU120::max(); - let b = SafeU120::new(2); - let _c = SafeU120::mul(a, b); -} - -#[test] -fn test_div() { - let a = SafeU120::new(6); - let b = SafeU120::new(3); - let c = SafeU120::div(a, b); - assert(c.value == 2); -} - -#[test(should_fail)] -fn test_div_by_zero() { - let a = SafeU120::new(6); - let b = SafeU120::new(0); - let _c = SafeU120::div(a, b); -} - -#[test] -fn test_mul_div() { - let a = SafeU120::new(6); - let b = SafeU120::new(3); - let c = SafeU120::new(2); - let d = SafeU120::mul_div(a, b, c); - assert(d.value == 9); -} - -#[test(should_fail)] -fn test_mul_div_zero_divisor() { - let a = SafeU120::new(6); - let b = SafeU120::new(3); - let c = SafeU120::new(0); - let _d = SafeU120::mul_div(a, b, c); -} - -#[test(should_fail)] -fn test_mul_div_ghost_overflow() { - let a = SafeU120::max(); - let b = SafeU120::new(2); - let c = SafeU120::new(4); - let _d = SafeU120::mul_div(a, b, c); -} - -#[test] -fn test_mul_div_up_rounding() { - let a = SafeU120::new(6); - let b = SafeU120::new(3); - let c = SafeU120::new(5); - let d = SafeU120::mul_div_up(a, b, c); - assert(d.value == 4); -} - -#[test] -fn test_mul_div_up_non_rounding() { - let a = SafeU120::new(6); - let b = SafeU120::new(3); - let c = SafeU120::new(2); - let d = SafeU120::mul_div_up(a, b, c); - assert(d.value == 9); -} - -#[test(should_fail)] -fn test_mul_div_up_ghost_overflow() { - let a = SafeU120::max(); - let b = SafeU120::new(2); - let c = SafeU120::new(9); - let _d = SafeU120::mul_div_up(a, b, c); -} - -// It should not be possible for us to overflow `mul_div_up` through the adder, since that require the divisor to be 1 -// since we otherwise would not be at the max value. If divisor is 1, adder is 0. -#[test(should_fail)] -fn test_mul_div_up_zero_divisor() { - let a = SafeU120::new(6); - let b = SafeU120::new(3); - let c = SafeU120::new(0); - let _d = SafeU120::mul_div_up(a, b, c); -} diff --git a/noir-projects/aztec-nr/value-note/src/filter.nr b/noir-projects/aztec-nr/value-note/src/filter.nr index 56dd89a5ca9..9d8a51a33bf 100644 --- a/noir-projects/aztec-nr/value-note/src/filter.nr +++ b/noir-projects/aztec-nr/value-note/src/filter.nr @@ -7,12 +7,12 @@ pub fn filter_notes_min_sum( min_sum: Field ) -> [Option; MAX_READ_REQUESTS_PER_CALL] { let mut selected = [Option::none(); MAX_READ_REQUESTS_PER_CALL]; - let mut sum = 0; + let mut sum = U128::from_integer(0); for i in 0..notes.len() { - if notes[i].is_some() & (sum < min_sum as u120) { + if notes[i].is_some() & (sum < U128::from_integer(min_sum)) { let note = notes[i].unwrap_unchecked(); selected[i] = Option::some(note); - sum += note.value as u120; + sum += U128::from_integer(note.value); } } selected diff --git a/noir-projects/aztec-nr/value-note/src/utils.nr b/noir-projects/aztec-nr/value-note/src/utils.nr index d7ef7c948dc..93894a5d4b4 100644 --- a/noir-projects/aztec-nr/value-note/src/utils.nr +++ b/noir-projects/aztec-nr/value-note/src/utils.nr @@ -3,10 +3,7 @@ use dep::aztec::context::PrivateContext; use dep::aztec::note::note_getter_options::{NoteGetterOptions, SortOrder}; use dep::aztec::oracle::get_public_key::get_public_key; use dep::aztec::state_vars::set::Set; -use crate::{ - filter::filter_notes_min_sum, - value_note::{ValueNote, VALUE_NOTE_LEN}, -}; +use crate::{filter::filter_notes_min_sum, value_note::{ValueNote, VALUE_NOTE_LEN}}; use dep::aztec::protocol_types::address::AztecAddress; // Sort the note values (0th field) in descending order. @@ -53,7 +50,7 @@ pub fn decrement_by_at_most(balance: Set, max_amount: Field, owner: A // Add the change value back to the owner's balance. let mut change_value = 0; - if decremented as u120 > max_amount as u120 { + if max_amount.lt(decremented) { change_value = decremented - max_amount; decremented -= change_value; } diff --git a/noir-projects/noir-contracts/contracts/card_game_contract/src/cards.nr b/noir-projects/noir-contracts/contracts/card_game_contract/src/cards.nr index 9569a41b774..f52115df6de 100644 --- a/noir-projects/noir-contracts/contracts/card_game_contract/src/cards.nr +++ b/noir-projects/noir-contracts/contracts/card_game_contract/src/cards.nr @@ -1,45 +1,32 @@ use dep::aztec::{ - protocol_types::{ - address::AztecAddress, - constants::{ - MAX_NOTES_PER_PAGE, - MAX_READ_REQUESTS_PER_CALL, - }, - }, + protocol_types::{address::AztecAddress, constants::{MAX_NOTES_PER_PAGE, MAX_READ_REQUESTS_PER_CALL}}, context::{PrivateContext, PublicContext, Context}, note::{ - note_getter_options::NoteGetterOptions, - note_viewer_options::NoteViewerOptions, - note_getter::view_notes, - }, - state_vars::set::Set, + note_getter_options::NoteGetterOptions, note_viewer_options::NoteViewerOptions, + note_getter::view_notes +}, + state_vars::set::Set }; use dep::std; -use dep::std::{ - option::Option, -}; -use dep::value_note::{ - value_note::{ValueNote, VALUE_NOTE_LEN}, -}; +use dep::std::{option::Option}; +use dep::value_note::{value_note::{ValueNote, VALUE_NOTE_LEN}}; struct Card { - strength: u16, - points: u16, + // We use u32s since u16s are unsupported + strength: u32, + points: u32, } impl Card { pub fn from_field(field: Field) -> Card { let value_bytes = field.to_le_bytes(32); - let strength = (value_bytes[0] as u16) + (value_bytes[1] as u16) * 256; - let points = (value_bytes[2] as u16) + (value_bytes[3] as u16) * 256; - Card { - strength, - points, - } + let strength = (value_bytes[0] as u32) + (value_bytes[1] as u32) * 256; + let points = (value_bytes[2] as u32) + (value_bytes[3] as u32) * 256; + Card { strength, points } } pub fn to_field(self) -> Field { - self.strength as Field + (self.points as Field)*65536 + self.strength as Field + (self.points as Field) * 65536 } pub fn serialize(self) -> [Field; 2] { @@ -60,30 +47,17 @@ struct CardNote { } impl CardNote { - fn new( - strength: u16, - points: u16, - owner: AztecAddress, - ) -> Self { - let card = Card { - strength, - points, - }; + fn new(strength: u32, points: u32, owner: AztecAddress) -> Self { + let card = Card { strength, points }; CardNote::from_card(card, owner) } pub fn from_card(card: Card, owner: AztecAddress) -> CardNote { - CardNote { - card, - note: ValueNote::new(card.to_field(), owner), - } + CardNote { card, note: ValueNote::new(card.to_field(), owner) } } pub fn from_note(note: ValueNote) -> CardNote { - CardNote { - card: Card::from_field(note.value), - note, - } + CardNote { card: Card::from_field(note.value), note } } } @@ -118,21 +92,13 @@ pub fn filter_cards( } impl Deck { - pub fn new( - context: Context, - storage_slot: Field, - ) -> Self { - let set = Set { - context, - storage_slot, - }; - Deck { - set - } + pub fn new(context: Context, storage_slot: Field) -> Self { + let set = Set { context, storage_slot }; + Deck { set } } - pub fn add_cards(&mut self, cards: [Card; N], owner: AztecAddress) -> [CardNote]{ - let context = self.set.context.private.unwrap(); + pub fn add_cards(&mut self, cards: [Card; N], owner: AztecAddress) -> [CardNote] { + let _context = self.set.context.private.unwrap(); let mut inserted_cards = []; for card in cards { @@ -150,25 +116,27 @@ impl Deck { let mut found_cards = [Option::none(); N]; for i in 0..maybe_notes.len() { if maybe_notes[i].is_some() { - let card_note = CardNote::from_note( - maybe_notes[i].unwrap_unchecked() - ); + let card_note = CardNote::from_note(maybe_notes[i].unwrap_unchecked()); // Ensure the notes are actually owned by the owner (to prevent user from generating a valid proof while // spending someone else's notes). assert(card_note.note.owner.eq(owner)); for j in 0..cards.len() { - if found_cards[j].is_none() & (cards[j].strength == card_note.card.strength) & (cards[j].points == card_note.card.points) { + if found_cards[j].is_none() + & (cards[j].strength == card_note.card.strength) + & (cards[j].points == card_note.card.points) { found_cards[j] = Option::some(card_note); } } } } - found_cards.map(|card_note: Option| { + found_cards.map( + |card_note: Option| { assert(card_note.is_some(), "Card not found"); card_note.unwrap_unchecked() - }) + } + ) } pub fn remove_cards(&mut self, cards: [Card; N], owner: AztecAddress) { @@ -182,14 +150,13 @@ impl Deck { let options = NoteViewerOptions::new().set_offset(offset); let opt_notes = self.set.view_notes(options); let mut opt_cards = [Option::none(); MAX_NOTES_PER_PAGE]; - + for i in 0..opt_notes.len() { opt_cards[i] = opt_notes[i].map(|note: ValueNote| Card::from_field(note.value)); } opt_cards } - } global PACK_CARDS = 3; // Limited by number of write requests (max 4) @@ -204,8 +171,8 @@ pub fn get_pack_cards(seed: Field, owner: AztecAddress, context: &mut PrivateCon // we generate PACK_CARDS cards assert((PACK_CARDS as u64) < 8, "Cannot generate more than 8 cards"); for i in 0..PACK_CARDS { - let strength = (random_bytes[i] as u16) + (random_bytes[i + 1] as u16) * 256; - let points = (random_bytes[i + 2] as u16) + (random_bytes[i + 3] as u16) * 256; + let strength = (random_bytes[i] as u32) + (random_bytes[i + 1] as u32) * 256; + let points = (random_bytes[i + 2] as u32) + (random_bytes[i + 3] as u32) * 256; cards[i] = Card { strength, points }; diff --git a/noir-projects/noir-contracts/contracts/card_game_contract/src/game.nr b/noir-projects/noir-contracts/contracts/card_game_contract/src/game.nr index f7b388c42b8..1a1415284db 100644 --- a/noir-projects/noir-contracts/contracts/card_game_contract/src/game.nr +++ b/noir-projects/noir-contracts/contracts/card_game_contract/src/game.nr @@ -1,7 +1,4 @@ -use dep::aztec::protocol_types::{ - address::AztecAddress, - traits::{Serialize, Deserialize}, -}; +use dep::aztec::protocol_types::{address::AztecAddress, traits::{Serialize, Deserialize}}; use crate::cards::Card; global NUMBER_OF_PLAYERS = 2; @@ -10,7 +7,7 @@ global NUMBER_OF_CARDS_DECK = 2; struct PlayerEntry { address: AztecAddress, deck_strength: u32, - points: u120, + points: u64, } impl PlayerEntry { @@ -25,7 +22,7 @@ impl Deserialize for PlayerEntry { fn deserialize(fields: [Field; PLAYER_SERIALIZED_LEN]) -> PlayerEntry { let address = AztecAddress::from_field(fields[0]); let deck_strength = fields[1] as u32; - let points = fields[2] as u120; + let points = fields[2] as u64; PlayerEntry { address, @@ -163,7 +160,7 @@ impl Game { for i in 0..NUMBER_OF_PLAYERS { let card = self.rounds_cards[round_offset + i]; - round_points += (card.points as u120); + round_points += (card.points as u64); if card.strength > winner_strength { winner_strength = card.strength; winner_index = i; diff --git a/noir-projects/noir-contracts/contracts/counter_contract/src/main.nr b/noir-projects/noir-contracts/contracts/counter_contract/src/main.nr index 02059ef0d3f..f2d99f1b1e8 100644 --- a/noir-projects/noir-contracts/contracts/counter_contract/src/main.nr +++ b/noir-projects/noir-contracts/contracts/counter_contract/src/main.nr @@ -2,20 +2,10 @@ contract Counter { // docs:start:imports use dep::aztec::protocol_types::address::AztecAddress; use dep::aztec::{ - context::{PrivateContext, Context}, - note::{ - note_header::NoteHeader, - utils as note_utils, - }, - state_vars::map::Map, - }; - use dep::value_note::{ - balance_utils, - value_note::{ - ValueNote, - VALUE_NOTE_LEN, - }, + context::{PrivateContext, Context}, note::{note_header::NoteHeader, utils as note_utils}, + state_vars::map::Map }; + use dep::value_note::{balance_utils, value_note::{ValueNote, VALUE_NOTE_LEN}}; use dep::easy_private_state::easy_private_state::EasyPrivateUint; // docs:end:imports @@ -27,7 +17,7 @@ contract Counter { // docs:start:constructor #[aztec(private)] - fn constructor(headstart: u120, owner: AztecAddress) { + fn constructor(headstart: u64, owner: AztecAddress) { let counters = storage.counters; counters.at(owner).add(headstart, owner); } diff --git a/noir-projects/noir-contracts/contracts/docs_example_contract/src/main.nr b/noir-projects/noir-contracts/contracts/docs_example_contract/src/main.nr index 86c91046462..8ebe2b316ab 100644 --- a/noir-projects/noir-contracts/contracts/docs_example_contract/src/main.nr +++ b/noir-projects/noir-contracts/contracts/docs_example_contract/src/main.nr @@ -13,26 +13,21 @@ mod types; contract DocsExample { // how to import dependencies defined in your workspace - use dep::aztec::protocol_types::{ - abis::function_selector::FunctionSelector, - address::AztecAddress, - }; + use dep::aztec::protocol_types::{abis::function_selector::FunctionSelector, address::AztecAddress}; use dep::aztec::{ note::{ - note_header::NoteHeader, - note_getter_options::{NoteGetterOptions, Comparator}, - note_viewer_options::{NoteViewerOptions}, - utils as note_utils, - }, + note_header::NoteHeader, note_getter_options::{NoteGetterOptions, Comparator}, + note_viewer_options::{NoteViewerOptions}, utils as note_utils + }, context::{PrivateContext, PublicContext, Context}, - state_vars::{map::Map, public_state::PublicState,singleton::Singleton, immutable_singleton::ImmutableSingleton, set::Set, stable_public_state::StablePublicState}, + state_vars::{ + map::Map, public_state::PublicState, singleton::Singleton, + immutable_singleton::ImmutableSingleton, set::Set, stable_public_state::StablePublicState + } }; // how to import methods from other files/folders within your workspace use crate::options::create_account_card_getter_options; - use crate::types::{ - card_note::{CardNote, CARD_NOTE_LEN}, - leader::Leader, - }; + use crate::types::{card_note::{CardNote, CARD_NOTE_LEN}, leader::Leader}; struct Storage { // Shows how to create a custom struct in PublicState @@ -64,10 +59,7 @@ contract DocsExample { fn init(context: Context) -> Self { Storage { // docs:start:storage-leader-init - leader: PublicState::new( - context, - 1 - ), + leader: PublicState::new(context, 1), // docs:end:storage-leader-init // docs:start:start_vars_singleton legendary_card: Singleton::new(context, 3), @@ -79,7 +71,7 @@ contract DocsExample { 4, |context, slot| { Singleton::new(context, slot) - }, + } ), // docs:end:state_vars-MapSingleton // docs:start:storage-set-init @@ -95,9 +87,8 @@ contract DocsExample { 8, |context, slot| { PublicState::new(context, slot) - }, - ), - // docs:end:storage-minters-init + } + )// docs:end:storage-minters-init } } } @@ -155,7 +146,7 @@ contract DocsExample { } // docs:start:state_vars-NoteGetterOptionsComparatorExampleNoir - unconstrained fn read_note(amount: Field, comparator: u3) -> pub [Option; 10] { + unconstrained fn read_note(amount: Field, comparator: u8) -> pub [Option; 10] { let options = NoteViewerOptions::new().select(0, amount, Option::some(comparator)); let notes = storage.set.view_notes(options); diff --git a/noir-projects/noir-contracts/contracts/easy_private_token_contract/src/main.nr b/noir-projects/noir-contracts/contracts/easy_private_token_contract/src/main.nr index 29137e82834..f37a26da733 100644 --- a/noir-projects/noir-contracts/contracts/easy_private_token_contract/src/main.nr +++ b/noir-projects/noir-contracts/contracts/easy_private_token_contract/src/main.nr @@ -4,19 +4,9 @@ contract EasyPrivateToken { use dep::std::option::Option; use dep::aztec::{ context::{PrivateContext, PublicContext, Context}, - note::{ - note_header::NoteHeader, - utils as note_utils, - }, - state_vars::map::Map, - }; - use dep::value_note::{ - balance_utils, - value_note::{ - ValueNote, - VALUE_NOTE_LEN, - }, + note::{note_header::NoteHeader, utils as note_utils}, state_vars::map::Map }; + use dep::value_note::{balance_utils, value_note::{ValueNote, VALUE_NOTE_LEN}}; use dep::easy_private_state::easy_private_state::EasyPrivateUint; struct Storage { @@ -27,7 +17,7 @@ contract EasyPrivateToken { * initialize the contract's initial state variables. */ #[aztec(private)] - fn constructor(initial_supply: u120, owner: AztecAddress) { + fn constructor(initial_supply: u64, owner: AztecAddress) { let balances = storage.balances; balances.at(owner).add(initial_supply, owner); @@ -35,7 +25,7 @@ contract EasyPrivateToken { // Mints `amount` of tokens to `owner`. #[aztec(private)] - fn mint(amount: u120, owner: AztecAddress) { + fn mint(amount: u64, owner: AztecAddress) { let balances = storage.balances; balances.at(owner).add(amount, owner); @@ -43,7 +33,7 @@ contract EasyPrivateToken { // Transfers `amount` of tokens from `sender` to a `recipient`. #[aztec(private)] - fn transfer(amount: u120, sender: AztecAddress, recipient: AztecAddress) { + fn transfer(amount: u64, sender: AztecAddress, recipient: AztecAddress) { let balances = storage.balances; balances.at(sender).sub(amount, sender); diff --git a/noir-projects/noir-contracts/contracts/lending_contract/Nargo.toml b/noir-projects/noir-contracts/contracts/lending_contract/Nargo.toml index dba6d562715..a23ee238cdb 100644 --- a/noir-projects/noir-contracts/contracts/lending_contract/Nargo.toml +++ b/noir-projects/noir-contracts/contracts/lending_contract/Nargo.toml @@ -6,4 +6,3 @@ type = "contract" [dependencies] aztec = { path = "../../../aztec-nr/aztec" } -safe_math = { path = "../../../aztec-nr/safe-math" } diff --git a/noir-projects/noir-contracts/contracts/lending_contract/src/asset.nr b/noir-projects/noir-contracts/contracts/lending_contract/src/asset.nr index a3ef3db1695..88f95f6e848 100644 --- a/noir-projects/noir-contracts/contracts/lending_contract/src/asset.nr +++ b/noir-projects/noir-contracts/contracts/lending_contract/src/asset.nr @@ -1,7 +1,4 @@ -use dep::aztec::protocol_types::{ - address::AztecAddress, - traits::{Deserialize, Serialize} -}; +use dep::aztec::protocol_types::{address::AztecAddress, traits::{Deserialize, Serialize}}; // Struct to be used to represent "totals". Generally, there should be one per asset. // It stores the global values that are shared among all users, such as an accumulator @@ -9,9 +6,9 @@ use dep::aztec::protocol_types::{ // In practice, it should also point to an oracle and have more fields related to // loan to value ratios and other things, but we did not have enough reads/writes for this. struct Asset { - interest_accumulator: u120, - last_updated_ts: u120, - loan_to_value: u120, + interest_accumulator: U128, + last_updated_ts: u64, + loan_to_value: U128, oracle: AztecAddress, } @@ -20,9 +17,9 @@ global ASSET_SERIALIZED_LEN: Field = 4; impl Serialize for Asset { fn serialize(asset: Asset) -> [Field; ASSET_SERIALIZED_LEN] { [ - asset.interest_accumulator as Field, + asset.interest_accumulator.to_integer(), asset.last_updated_ts as Field, - asset.loan_to_value as Field, + asset.loan_to_value.to_integer(), asset.oracle.to_field() ] } @@ -32,9 +29,9 @@ impl Deserialize for Asset { // Right now we are wasting so many writes. If changing last_updated_ts // we will end up rewriting all of them, wasting writes. fn deserialize(fields: [Field; ASSET_SERIALIZED_LEN]) -> Asset { - let interest_accumulator = fields[0] as u120; - let last_updated_ts = fields[1] as u120; - let loan_to_value = fields[2] as u120; + let interest_accumulator = U128::from_integer(fields[0]); + let last_updated_ts = fields[1] as u64; + let loan_to_value = U128::from_integer(fields[2]); let oracle = AztecAddress::from_field(fields[3]); Asset { diff --git a/noir-projects/noir-contracts/contracts/lending_contract/src/helpers.nr b/noir-projects/noir-contracts/contracts/lending_contract/src/helpers.nr index 63e72fc7aa2..7a43a13de78 100644 --- a/noir-projects/noir-contracts/contracts/lending_contract/src/helpers.nr +++ b/noir-projects/noir-contracts/contracts/lending_contract/src/helpers.nr @@ -1,5 +1,4 @@ use crate::interest_math::compute_multiplier; -use dep::safe_math::SafeU120; use dep::aztec::hash::pedersen_hash; // Utility used to easily get a "id" for a private user that sits in the same @@ -16,45 +15,38 @@ pub fn compute_identifier(secret: Field, on_behalf_of: Field, self: Field) -> Fi } pub fn covered_by_collateral( - price: u120, - loan_to_value: u120, - collateral: u120, - increase: u120, - decrease: u120 -) -> u120 { - let price_precision = SafeU120 { value: 1000000000 }; - let ltv_precision = SafeU120 { value: 10000 }; + price: U128, + loan_to_value: U128, + collateral: U128, + increase: U128, + decrease: U128 +) -> U128 { + let price_precision = U128::from_integer(1000000000); + let ltv_precision = U128::from_integer(10000); - let price = SafeU120 { value: price }; - let collateral = SafeU120 { value: collateral }.add(SafeU120 { value: increase }).sub(SafeU120 { value: decrease }); - let loan_to_value = SafeU120 { value: loan_to_value }; + let collateral = (collateral + increase) - decrease; - let collateral_value = collateral.mul_div(price, price_precision); - let debt_covered = collateral_value.mul_div(loan_to_value, ltv_precision); + let collateral_value = (collateral* price) / price_precision; + let debt_covered = (collateral_value * loan_to_value) / ltv_precision; - debt_covered.value + debt_covered } struct DebtReturn { - debt_value: u120, - static_debt: u120, + debt_value: U128, + static_debt: U128, } pub fn debt_updates( - interest_accumulator: u120, - static_debt: u120, - increase: u120, - decrease: u120 + interest_accumulator: U128, + static_debt: U128, + increase: U128, + decrease: U128 ) -> DebtReturn { - assert(interest_accumulator > 0); - let accumulator_precision = SafeU120 { value: 1000000000 }; - - let static_debt = SafeU120 { value: static_debt }; - let interest_accumulator = SafeU120 { value: interest_accumulator }; - let increase = SafeU120 { value: increase }; - let decrease = SafeU120 { value: decrease }; + assert(interest_accumulator > U128::from_integer(0)); + let accumulator_precision = U128::from_integer(1000000000); - let current_debt_value = static_debt.mul_div(interest_accumulator, accumulator_precision); + let current_debt_value = (static_debt * interest_accumulator) / accumulator_precision; let new_debt_value = current_debt_value.add(increase).sub(decrease); // static_debt_increase = amount / accumulator diff --git a/noir-projects/noir-contracts/contracts/lending_contract/src/interest_math.nr b/noir-projects/noir-contracts/contracts/lending_contract/src/interest_math.nr index 9f9d3c518fb..24f661a143e 100644 --- a/noir-projects/noir-contracts/contracts/lending_contract/src/interest_math.nr +++ b/noir-projects/noir-contracts/contracts/lending_contract/src/interest_math.nr @@ -1,5 +1,3 @@ -use dep::safe_math::SafeU120; - // Binomial approximation of exponential // using lower than desired precisions for everything due to u120 limit // (1+x)^n = 1+n*x+[n/2*(n-1)]*x^2+[n/6*(n-1)*(n-2)*x^3]... diff --git a/noir-projects/noir-contracts/contracts/lending_contract/src/interfaces.nr b/noir-projects/noir-contracts/contracts/lending_contract/src/interfaces.nr index c65d8033a9d..631f5eed5ca 100644 --- a/noir-projects/noir-contracts/contracts/lending_contract/src/interfaces.nr +++ b/noir-projects/noir-contracts/contracts/lending_contract/src/interfaces.nr @@ -1,33 +1,26 @@ -use dep::aztec::context::{ - PrivateContext, - PublicContext -}; +use dep::aztec::context::{PrivateContext, PublicContext}; use crate::asset::Asset; -use dep::aztec::protocol_types::{ - abis::function_selector::FunctionSelector, - address::AztecAddress, - constants::RETURN_VALUES_LENGTH, -}; +use dep::aztec::protocol_types::{abis::function_selector::FunctionSelector, address::AztecAddress, constants::RETURN_VALUES_LENGTH}; struct PriceFeed { address: AztecAddress, } impl PriceFeed { - pub fn at(address: AztecAddress) -> Self { - Self { address } - } - - pub fn get_price(self: Self, context: PublicContext) -> u120 { - let return_values = context.call_public_function( - self.address, - FunctionSelector::from_signature("get_price(Field)"), - [0] - ); - - return_values[0] as u120 - } + pub fn at(address: AztecAddress) -> Self { + Self { address } + } + + pub fn get_price(self: Self, context: PublicContext) -> U128 { + let return_values = context.call_public_function( + self.address, + FunctionSelector::from_signature("get_price(Field)"), + [0] + ); + + U128::from_integer(return_values[0]) + } } struct Token { @@ -35,50 +28,76 @@ struct Token { } impl Token { - pub fn at(address: AztecAddress) -> Self { - Self { address } - } - - pub fn transfer_public(self: Self, context: PublicContext, from: AztecAddress, to: AztecAddress, amount: Field, nonce: Field) { - context.call_public_function( - self.address, - FunctionSelector::from_signature("transfer_public((Field),(Field),Field,Field)"), - [from.to_field(), to.to_field(), amount, nonce] - ); - } - - pub fn mint_public(self: Self, context: PublicContext, to: AztecAddress, amount: Field) { - context.call_public_function( - self.address, - FunctionSelector::from_signature("mint_public((Field),Field)"), - [to.to_field(), amount] - ); - } - - pub fn burn_public(self: Self, context: PublicContext, from: AztecAddress, amount: Field, nonce: Field) { - context.call_public_function( - self.address, - FunctionSelector::from_signature("burn_public((Field),Field,Field)"), - [from.to_field(), amount, nonce] - ); - } - - // Private - pub fn unshield(self: Self, context: &mut PrivateContext, from: AztecAddress, to: AztecAddress, amount: Field, nonce: Field) -> [Field; RETURN_VALUES_LENGTH] { - context.call_private_function( - self.address, - FunctionSelector::from_signature("unshield((Field),(Field),Field,Field)"), - [from.to_field(), to.to_field(), amount, nonce] - ) - } - - pub fn burn(self: Self, context: &mut PrivateContext, from: AztecAddress, amount: Field, nonce: Field) -> [Field; RETURN_VALUES_LENGTH] { - context.call_private_function( - self.address, - FunctionSelector::from_signature("burn((Field),Field,Field)"), - [from.to_field(), amount, nonce] - ) - } + pub fn at(address: AztecAddress) -> Self { + Self { address } + } + + pub fn transfer_public( + self: Self, + context: PublicContext, + from: AztecAddress, + to: AztecAddress, + amount: Field, + nonce: Field + ) { + context.call_public_function( + self.address, + FunctionSelector::from_signature("transfer_public((Field),(Field),Field,Field)"), + [from.to_field(), to.to_field(), amount, nonce] + ); + } + + pub fn mint_public(self: Self, context: PublicContext, to: AztecAddress, amount: Field) { + context.call_public_function( + self.address, + FunctionSelector::from_signature("mint_public((Field),Field)"), + [to.to_field(), amount] + ); + } + + pub fn burn_public( + self: Self, + context: PublicContext, + from: AztecAddress, + amount: Field, + nonce: Field + ) { + context.call_public_function( + self.address, + FunctionSelector::from_signature("burn_public((Field),Field,Field)"), + [from.to_field(), amount, nonce] + ); + } + + // Private + pub fn unshield( + self: Self, + context: &mut PrivateContext, + from: AztecAddress, + to: AztecAddress, + amount: Field, + nonce: Field + ) -> [Field; RETURN_VALUES_LENGTH] { + context.call_private_function( + self.address, + FunctionSelector::from_signature("unshield((Field),(Field),Field,Field)"), + [from.to_field(), to.to_field(), amount, nonce] + ) + } + + pub fn burn( + self: Self, + context: &mut PrivateContext, + from: AztecAddress, + amount: Field, + nonce: Field + ) -> [Field; RETURN_VALUES_LENGTH] { + context.call_private_function( + self.address, + FunctionSelector::from_signature("burn((Field),Field,Field)"), + [from.to_field(), amount, nonce] + ) + } } struct Lending { @@ -86,16 +105,16 @@ struct Lending { } impl Lending { - pub fn at(address: AztecAddress) -> Self { - Self { address } - } - - pub fn update_accumulator(self: Self, context: PublicContext) -> Asset { - let return_values = context.call_public_function_no_args( - self.address, - FunctionSelector::from_signature("update_accumulator()"), - ); - - Asset::deserialize(return_values) - } + pub fn at(address: AztecAddress) -> Self { + Self { address } + } + + pub fn update_accumulator(self: Self, context: PublicContext) -> Asset { + let return_values = context.call_public_function_no_args( + self.address, + FunctionSelector::from_signature("update_accumulator()") + ); + + Asset::deserialize(return_values) + } } diff --git a/noir-projects/noir-contracts/contracts/lending_contract/src/main.nr b/noir-projects/noir-contracts/contracts/lending_contract/src/main.nr index 64bab5a6e72..4c22c5ec95a 100644 --- a/noir-projects/noir-contracts/contracts/lending_contract/src/main.nr +++ b/noir-projects/noir-contracts/contracts/lending_contract/src/main.nr @@ -15,7 +15,6 @@ contract Lending { abis::function_selector::FunctionSelector, address::AztecAddress, }; - use dep::safe_math::SafeU120; use dep::std::option::Option; use dep::aztec::{ context::{PrivateContext, PublicContext, Context}, @@ -59,16 +58,17 @@ contract Lending { let asset_loc = storage.assets.at(0); let asset = asset_loc.read(); - assert(loan_to_value as u120 <= 10000); + let loan_to_value = U128::from_integer(loan_to_value); + + assert(loan_to_value <= U128::from_integer(10000)); assert(asset.last_updated_ts == 0); - assert(asset.interest_accumulator == 0); + assert(asset.interest_accumulator == U128::from_integer(0)); - let last_updated_ts = context.timestamp() as u120; - let loan_to_value = loan_to_value as u120; + let last_updated_ts = context.timestamp() as u64; asset_loc.write( Asset { - interest_accumulator: 1000000000, + interest_accumulator: U128::from_integer(1000000000), last_updated_ts, loan_to_value, oracle @@ -85,19 +85,19 @@ contract Lending { let asset_loc = storage.assets.at(0); let mut asset = asset_loc.read(); - let timestamp = context.timestamp() as u120; - let dt: SafeU120 = SafeU120 { value: timestamp }.sub(SafeU120 { value: asset.last_updated_ts }); + let timestamp = context.timestamp() as u64; + let dt = timestamp - asset.last_updated_ts; // Only update if time has passed. - if (!dt.is_zero()) { - let precision: SafeU120 = SafeU120 { value: 1000000000 }; - let rate_per_second: u120 = 1268391679; // 4% yearly rate / (60 * 60 * 24 * 365) + if !(dt == 0) { + let precision = U128::from_integer(1000000000); + let rate_per_second = U128::from_integer(1268391679); // 4% yearly rate / (60 * 60 * 24 * 365) // if rate_per_second < sqrt(WAD) our approx is eq precision + rate * dt let multiplier = compute_multiplier(rate_per_second, dt); // accumulator *= multiplier, and multiplier >= 1 - asset.interest_accumulator = SafeU120{value: asset.interest_accumulator}.mul_div(multiplier, precision).value; - asset.last_updated_ts = context.timestamp() as u120; + asset.interest_accumulator = (asset.interest_accumulator * multiplier) / precision; + asset.last_updated_ts = context.timestamp() as u64; asset_loc.write(asset); } diff --git a/noir-projects/noir-contracts/contracts/token_blacklist_contract/Nargo.toml b/noir-projects/noir-contracts/contracts/token_blacklist_contract/Nargo.toml index a742335f1ef..d178cd2ba2d 100644 --- a/noir-projects/noir-contracts/contracts/token_blacklist_contract/Nargo.toml +++ b/noir-projects/noir-contracts/contracts/token_blacklist_contract/Nargo.toml @@ -6,6 +6,5 @@ type = "contract" [dependencies] aztec = { path = "../../../aztec-nr/aztec" } -safe_math = { path = "../../../aztec-nr/safe-math" } field_note = { path = "../../../aztec-nr/field-note" } authwit = { path = "../../../aztec-nr/authwit" } diff --git a/noir-projects/noir-contracts/contracts/token_blacklist_contract/src/main.nr b/noir-projects/noir-contracts/contracts/token_blacklist_contract/src/main.nr index 0a0394289a1..aee7ff2c834 100644 --- a/noir-projects/noir-contracts/contracts/token_blacklist_contract/src/main.nr +++ b/noir-projects/noir-contracts/contracts/token_blacklist_contract/src/main.nr @@ -16,7 +16,6 @@ contract TokenBlacklist { // Libs use dep::std::option::Option; - use dep::safe_math::SafeU120; use dep::aztec::protocol_types::{ abis::function_selector::FunctionSelector, address::AztecAddress, diff --git a/noir-projects/noir-contracts/contracts/token_blacklist_contract/src/types/balances_map.nr b/noir-projects/noir-contracts/contracts/token_blacklist_contract/src/types/balances_map.nr index cbc423cc4ff..81ab66cd07a 100644 --- a/noir-projects/noir-contracts/contracts/token_blacklist_contract/src/types/balances_map.nr +++ b/noir-projects/noir-contracts/contracts/token_blacklist_contract/src/types/balances_map.nr @@ -1,5 +1,4 @@ use dep::std::option::Option; -use dep::safe_math::SafeU120; use dep::aztec::{ context::Context, protocol_types::{address::AztecAddress, constants::MAX_READ_REQUESTS_PER_CALL}, state_vars::{set::Set, map::Map}, diff --git a/noir-projects/noir-contracts/contracts/token_blacklist_contract/src/types/token_note.nr b/noir-projects/noir-contracts/contracts/token_blacklist_contract/src/types/token_note.nr index a7e2a45f1ec..13773ebee0a 100644 --- a/noir-projects/noir-contracts/contracts/token_blacklist_contract/src/types/token_note.nr +++ b/noir-projects/noir-contracts/contracts/token_blacklist_contract/src/types/token_note.nr @@ -4,7 +4,6 @@ use dep::aztec::{ context::PrivateContext, state_vars::set::Set, log::emit_encrypted_log, hash::pedersen_hash }; use dep::aztec::oracle::{rand::rand, nullifier_key::get_nullifier_secret_key, get_public_key::get_public_key}; -use dep::safe_math::SafeU120; use dep::std::option::Option; trait OwnedNote { diff --git a/noir-projects/noir-contracts/contracts/token_contract/Nargo.toml b/noir-projects/noir-contracts/contracts/token_contract/Nargo.toml index f8409f858b9..2c06fc4c8b5 100644 --- a/noir-projects/noir-contracts/contracts/token_contract/Nargo.toml +++ b/noir-projects/noir-contracts/contracts/token_contract/Nargo.toml @@ -6,6 +6,5 @@ type = "contract" [dependencies] aztec = { path = "../../../aztec-nr/aztec" } -safe_math = { path = "../../../aztec-nr/safe-math" } compressed_string = { path = "../../../aztec-nr/compressed-string" } authwit = { path = "../../../aztec-nr/authwit" } diff --git a/noir-projects/noir-contracts/contracts/token_contract/src/main.nr b/noir-projects/noir-contracts/contracts/token_contract/src/main.nr index f48b637ef6d..e72e39fa87f 100644 --- a/noir-projects/noir-contracts/contracts/token_contract/src/main.nr +++ b/noir-projects/noir-contracts/contracts/token_contract/src/main.nr @@ -13,7 +13,6 @@ contract Token { // Libs use dep::std::option::Option; - use dep::safe_math::SafeU120; use dep::compressed_string::FieldCompressedString; use dep::aztec::{ diff --git a/noir-projects/noir-contracts/contracts/token_contract/src/types/balances_map.nr b/noir-projects/noir-contracts/contracts/token_contract/src/types/balances_map.nr index a47fdce4f3f..5666abccb59 100644 --- a/noir-projects/noir-contracts/contracts/token_contract/src/types/balances_map.nr +++ b/noir-projects/noir-contracts/contracts/token_contract/src/types/balances_map.nr @@ -1,5 +1,4 @@ use dep::std::option::Option; -use dep::safe_math::SafeU120; use dep::aztec::{ context::{PrivateContext, PublicContext, Context}, hash::pedersen_hash, protocol_types::{address::AztecAddress, constants::MAX_READ_REQUESTS_PER_CALL}, diff --git a/noir-projects/noir-contracts/contracts/token_contract/src/types/token_note.nr b/noir-projects/noir-contracts/contracts/token_contract/src/types/token_note.nr index 2be2ecb3e00..3746d53089f 100644 --- a/noir-projects/noir-contracts/contracts/token_contract/src/types/token_note.nr +++ b/noir-projects/noir-contracts/contracts/token_contract/src/types/token_note.nr @@ -4,7 +4,6 @@ use dep::aztec::{ context::PrivateContext, state_vars::set::Set, log::emit_encrypted_log, hash::pedersen_hash }; use dep::aztec::oracle::{rand::rand, nullifier_key::get_nullifier_secret_key, get_public_key::get_public_key}; -use dep::safe_math::SafeU120; use dep::std::option::Option; trait OwnedNote { diff --git a/noir-projects/noir-protocol-circuits/src/crates/types/src/traits.nr b/noir-projects/noir-protocol-circuits/src/crates/types/src/traits.nr index e4cd56320cc..b7e0438e660 100644 --- a/noir-projects/noir-protocol-circuits/src/crates/types/src/traits.nr +++ b/noir-projects/noir-protocol-circuits/src/crates/types/src/traits.nr @@ -19,9 +19,9 @@ impl Empty for Field { fn empty() -> Self {0} } impl Empty for u1 { fn empty() -> Self {0} } impl Empty for u8 { fn empty() -> Self {0} } -impl Empty for u16 { fn empty() -> Self {0} } impl Empty for u32 { fn empty() -> Self {0} } impl Empty for u64 { fn empty() -> Self {0} } +impl Empty for U128 { fn empty() -> Self {U128::from_integer(0)} } pub fn is_empty(item: T) -> bool where T: Empty + Eq { item.eq(T::empty()) @@ -45,6 +45,12 @@ impl ToField for Field { } } +impl ToField for U128 { + fn to_field(self) -> Field { + self.to_integer() + } +} + // docs:start:serialize trait Serialize { fn serialize(self) -> [Field; N]; diff --git a/noir-projects/noir-protocol-circuits/src/crates/types/src/type_serialization.nr b/noir-projects/noir-protocol-circuits/src/crates/types/src/type_serialization.nr index 7bc12f02d27..1a9377bfb72 100644 --- a/noir-projects/noir-protocol-circuits/src/crates/types/src/type_serialization.nr +++ b/noir-projects/noir-protocol-circuits/src/crates/types/src/type_serialization.nr @@ -54,3 +54,16 @@ impl Deserialize for bool { fields[0] as bool } } + +impl Serialize<2> for U128 { + fn serialize(self) -> [Field; 2] { + [self.lo, self.hi] + } + +} + +impl Deserialize<2> for U128 { + fn deserialize(fields: [Field; 2]) -> Self { + U128 {lo: fields[0], hi: fields[1]} + } +} From 636c316123f4df094b81b631700b12c56e608fe5 Mon Sep 17 00:00:00 2001 From: sirasistant Date: Thu, 15 Feb 2024 14:06:44 +0000 Subject: [PATCH 3/4] fix: more fixes for restricted bit sizes --- .../contracts/lending_contract/src/helpers.nr | 25 ++++--- .../lending_contract/src/interest_math.nr | 23 +++--- .../contracts/lending_contract/src/main.nr | 70 +++++++++++-------- .../price_feed_contract/src/asset.nr | 6 +- .../contracts/price_feed_contract/src/main.nr | 10 +-- 5 files changed, 73 insertions(+), 61 deletions(-) diff --git a/noir-projects/noir-contracts/contracts/lending_contract/src/helpers.nr b/noir-projects/noir-contracts/contracts/lending_contract/src/helpers.nr index 7a43a13de78..7cfba1fc667 100644 --- a/noir-projects/noir-contracts/contracts/lending_contract/src/helpers.nr +++ b/noir-projects/noir-contracts/contracts/lending_contract/src/helpers.nr @@ -26,7 +26,7 @@ pub fn covered_by_collateral( let collateral = (collateral + increase) - decrease; - let collateral_value = (collateral* price) / price_precision; + let collateral_value = (collateral * price) / price_precision; let debt_covered = (collateral_value * loan_to_value) / ltv_precision; debt_covered @@ -37,6 +37,15 @@ struct DebtReturn { static_debt: U128, } +fn div_up(a: U128, b: U128) -> U128 { + let div = a / b; + if div * b < a { + div + U128::from_integer(1) + } else { + div + } +} + pub fn debt_updates( interest_accumulator: U128, static_debt: U128, @@ -51,20 +60,18 @@ pub fn debt_updates( // static_debt_increase = amount / accumulator // rounding up new debt. - let static_debt_increase = increase.mul_div_up(accumulator_precision, interest_accumulator); + let static_debt_increase = div_up(increase * accumulator_precision, interest_accumulator); // rounding down repayment. - let static_debt_decrease = decrease.mul_div(accumulator_precision, interest_accumulator); + let static_debt_decrease = (decrease * accumulator_precision) / interest_accumulator; // We need to allow repaying of the entire debt as well etc. This is very prone to failing // if you try to repay exact due to time diff between sim and execution. let new_static_debt = static_debt.add(static_debt_increase).sub(static_debt_decrease); - DebtReturn { debt_value: new_debt_value.value, static_debt: new_static_debt.value } + DebtReturn { debt_value: new_debt_value, static_debt: new_static_debt } } -pub fn debt_value(static_debt: u120, interest_accumulator: u120) -> u120 { - let static_debt = SafeU120 { value: static_debt }; - let accumulator_precision = SafeU120 { value: 1000000000 }; - let interest_accumulator = SafeU120 { value: interest_accumulator }; - static_debt.mul_div_up(interest_accumulator, accumulator_precision).value +pub fn debt_value(static_debt: U128, interest_accumulator: U128) -> U128 { + let accumulator_precision = U128::from_integer(1000000000); + div_up(static_debt * interest_accumulator, accumulator_precision) } diff --git a/noir-projects/noir-contracts/contracts/lending_contract/src/interest_math.nr b/noir-projects/noir-contracts/contracts/lending_contract/src/interest_math.nr index 24f661a143e..a678da6baf0 100644 --- a/noir-projects/noir-contracts/contracts/lending_contract/src/interest_math.nr +++ b/noir-projects/noir-contracts/contracts/lending_contract/src/interest_math.nr @@ -5,23 +5,24 @@ // dividing with 31536000 (seconds per year). // rate must be measured with higher precision than 10^9. // we use e18, and rates >= 4% yearly. Otherwise need more precision -pub fn compute_multiplier(rate_per_second: u120, dt: SafeU120) -> SafeU120 { - let base = SafeU120 { value: 1000000000 }; // 1e9 - let WAD = SafeU120 { value: 1000000000000000000 }; // 1e18 +pub fn compute_multiplier(rate_per_second: U128, dt: u64) -> U128 { + let base = U128::from_integer(1000000000); // 1e9 + let WAD = U128::from_integer(1000000000000000000); // 1e18 let diff = WAD.div(base); let mut res = base; - if (!dt.is_zero()) { - let exp_minus_one = SafeU120 { value: dt.value - 1 }; - let exp_minus_two = SafeU120 { value: if (dt.value > 2) { dt.value - 2 } else { 0 } }; + if dt != 0 { + let exp_minus_one = U128::from_integer(dt - 1); + let exp_minus_two = U128::from_integer(if (dt > 2) { dt - 2 } else { 0 }); + let dt = U128::from_integer(dt); // if rate_per_second < sqrt(WAD), then base_power_two and base_power_three = 0 - let rate = SafeU120 { value: rate_per_second }; - let base_power_two = rate.mul_div(rate, WAD); - let base_power_three = base_power_two.mul_div(rate, WAD); + let rate = rate_per_second; + let base_power_two = (rate * rate) / WAD; + let base_power_three = (base_power_two * rate) / WAD; let temp = dt.mul(exp_minus_one); - let second_term = temp.mul(base_power_two).div(SafeU120 { value: 2 }); - let third_term = temp.mul(exp_minus_two).mul(base_power_three).div(SafeU120 { value: 6 }); + let second_term = temp.mul(base_power_two).div(U128::from_integer(2)); + let third_term = temp.mul(exp_minus_two).mul(base_power_three).div(U128::from_integer(6)); // throwing away precision to keep us under u120 :sob: let offset = dt.mul(rate).add(second_term).add(third_term).div(diff); diff --git a/noir-projects/noir-contracts/contracts/lending_contract/src/main.nr b/noir-projects/noir-contracts/contracts/lending_contract/src/main.nr index 4c22c5ec95a..b6ec053fad9 100644 --- a/noir-projects/noir-contracts/contracts/lending_contract/src/main.nr +++ b/noir-projects/noir-contracts/contracts/lending_contract/src/main.nr @@ -11,18 +11,9 @@ mod interfaces; // - A way to repay all debt at once // - Liquidations contract Lending { - use dep::aztec::protocol_types::{ - abis::function_selector::FunctionSelector, - address::AztecAddress, - }; + use dep::aztec::protocol_types::{abis::function_selector::FunctionSelector, address::AztecAddress}; use dep::std::option::Option; - use dep::aztec::{ - context::{PrivateContext, PublicContext, Context}, - state_vars::{ - map::Map, - public_state::PublicState, - } - }; + use dep::aztec::{context::{PrivateContext, PublicContext, Context}, state_vars::{map::Map, public_state::PublicState}}; use crate::asset::Asset; use crate::interest_math::compute_multiplier; use crate::helpers::{covered_by_collateral, DebtReturn, debt_updates, debt_value, compute_identifier}; @@ -67,12 +58,7 @@ contract Lending { let last_updated_ts = context.timestamp() as u64; asset_loc.write( - Asset { - interest_accumulator: U128::from_integer(1000000000), - last_updated_ts, - loan_to_value, - oracle - } + Asset { interest_accumulator: U128::from_integer(1000000000), last_updated_ts, loan_to_value, oracle } ); storage.collateral_asset.write(collateral_asset); @@ -191,11 +177,16 @@ contract Lending { let debt_covered = covered_by_collateral( price, asset.loan_to_value, - collateral as u120, - 0, - amount as u120 + U128::from_integer(collateral), + U128::from_integer(0), + U128::from_integer(amount) + ); + let debt_returns = debt_updates( + asset.interest_accumulator, + U128::from_integer(static_debt), + U128::from_integer(0), + U128::from_integer(0) ); - let debt_returns = debt_updates(asset.interest_accumulator, static_debt as u120, 0, 0); assert(debt_returns.debt_value < debt_covered); @@ -233,15 +224,26 @@ contract Lending { let price = PriceFeed::at(asset.oracle).get_price(context); // Fetch collateral and static_debt, compute health of current position - let collateral = storage.collateral.at(owner).read() as u120; - let static_debt = storage.static_debt.at(owner).read() as u120; + let collateral = U128::from_integer(storage.collateral.at(owner).read()); + let static_debt = U128::from_integer(storage.static_debt.at(owner).read()); - let debt_covered = covered_by_collateral(price, asset.loan_to_value, collateral, 0, 0); - let debt_returns = debt_updates(asset.interest_accumulator, static_debt, amount as u120, 0); + let debt_covered = covered_by_collateral( + price, + asset.loan_to_value, + collateral, + U128::from_integer(0), + U128::from_integer(0) + ); + let debt_returns = debt_updates( + asset.interest_accumulator, + static_debt, + U128::from_integer(amount), + U128::from_integer(0) + ); assert(debt_returns.debt_value < debt_covered); - storage.static_debt.at(owner).write(debt_returns.static_debt as Field); + storage.static_debt.at(owner).write(debt_returns.static_debt.to_integer()); // @todo @LHerskind Need to support both private and public minting. let stable_coin = storage.stable_coin.read(); @@ -285,10 +287,15 @@ contract Lending { // To ensure that private is using the correct token. assert(stable_coin.eq(storage.stable_coin.read())); - let static_debt = storage.static_debt.at(owner).read() as u120; - let debt_returns = debt_updates(asset.interest_accumulator, static_debt, 0, amount as u120); + let static_debt = U128::from_integer(storage.static_debt.at(owner).read()); + let debt_returns = debt_updates( + asset.interest_accumulator, + static_debt, + U128::from_integer(0), + U128::from_integer(amount) + ); - storage.static_debt.at(owner).write(debt_returns.static_debt as Field); + storage.static_debt.at(owner).write(debt_returns.static_debt.to_integer()); } unconstrained fn get_asset(assetId: Field) -> pub Asset { @@ -299,7 +306,10 @@ contract Lending { let collateral = storage.collateral.at(owner).read(); let static_debt = storage.static_debt.at(owner).read(); let asset = storage.assets.at(0).read(); - let debt = debt_value(static_debt as u120, asset.interest_accumulator as u120) as Field; + let debt = debt_value( + U128::from_integer(static_debt), + U128::from_integer(asset.interest_accumulator) + ).to_integer(); Position { collateral, static_debt, debt } } diff --git a/noir-projects/noir-contracts/contracts/price_feed_contract/src/asset.nr b/noir-projects/noir-contracts/contracts/price_feed_contract/src/asset.nr index b795fdf68a3..b1358e1abb7 100644 --- a/noir-projects/noir-contracts/contracts/price_feed_contract/src/asset.nr +++ b/noir-projects/noir-contracts/contracts/price_feed_contract/src/asset.nr @@ -1,20 +1,20 @@ use dep::aztec::protocol_types::traits::{Serialize, Deserialize}; struct Asset { - price: u120, + price: U128, } global ASSET_SERIALIZED_LEN: Field = 1; impl Serialize for Asset { fn serialize(asset: Asset) -> [Field; ASSET_SERIALIZED_LEN] { - [asset.price as Field] + [asset.price.to_integer()] } } impl Deserialize for Asset { fn deserialize(fields: [Field; ASSET_SERIALIZED_LEN]) -> Asset { - let price = fields[0] as u120; + let price = fields[0] as w; Asset { price } } } diff --git a/noir-projects/noir-contracts/contracts/price_feed_contract/src/main.nr b/noir-projects/noir-contracts/contracts/price_feed_contract/src/main.nr index 5d2ef98e196..fe5c919c025 100644 --- a/noir-projects/noir-contracts/contracts/price_feed_contract/src/main.nr +++ b/noir-projects/noir-contracts/contracts/price_feed_contract/src/main.nr @@ -2,13 +2,7 @@ mod asset; contract PriceFeed { use dep::std::option::Option; - use dep::aztec::{ - context::{PrivateContext, PublicContext, Context}, - state_vars::{ - map::Map, - public_state::PublicState, - }, - }; + use dep::aztec::{context::{PrivateContext, PublicContext, Context}, state_vars::{map::Map, public_state::PublicState}}; use dep::aztec::protocol_types::address::AztecAddress; use crate::asset::Asset; @@ -21,7 +15,7 @@ contract PriceFeed { fn constructor() {} #[aztec(public)] - fn set_price(asset_id: Field, price: u120) { + fn set_price(asset_id: Field, price: Field) { let asset = storage.assets.at(asset_id); asset.write(Asset { price }); } From f92dcdd8bf790f1b074abd380d2c3f29ee36219b Mon Sep 17 00:00:00 2001 From: sirasistant Date: Fri, 16 Feb 2024 10:58:00 +0000 Subject: [PATCH 4/4] more u128 --- .../slow-updates-tree/src/slow_map.nr | 332 +++++++++--------- .../price_feed_contract/src/asset.nr | 2 +- .../contracts/price_feed_contract/src/main.nr | 2 +- .../contracts/token_contract/src/main.nr | 66 ++-- .../crates/types/src/type_serialization.nr | 12 +- 5 files changed, 192 insertions(+), 222 deletions(-) diff --git a/noir-projects/aztec-nr/slow-updates-tree/src/slow_map.nr b/noir-projects/aztec-nr/slow-updates-tree/src/slow_map.nr index 065ab009197..20482969143 100644 --- a/noir-projects/aztec-nr/slow-updates-tree/src/slow_map.nr +++ b/noir-projects/aztec-nr/slow-updates-tree/src/slow_map.nr @@ -6,10 +6,10 @@ use dep::std::merkle::compute_merkle_root; use dep::std::option::Option; // The epoch length is just a random number for now. -global EPOCH_LENGTH: u120 = 100; +global EPOCH_LENGTH: u64 = 100; fn compute_next_change(time: Field) -> Field { - ((time as u120 / EPOCH_LENGTH + 1) * EPOCH_LENGTH) as Field + ((time as u64 / EPOCH_LENGTH + 1) * EPOCH_LENGTH) as Field } // A leaf in the tree. @@ -54,42 +54,36 @@ pub fn deserialize_slow_update_proof(serialized: [Field; M]) -> SlowUpdate } impl SlowUpdateProof { - pub fn serialize(self: Self) -> [Field; M] { - let mut serialized = [0; M]; - serialized[0] = self.index; - serialized[1] = self.new_value; - serialized[2] = self.before.value; - serialized[3 + N] = self.after.value; - - for i in 0..N { - serialized[3 + i] = self.before.sibling_path[i]; - serialized[4 + N + i] = self.after.sibling_path[i]; - } - serialized - } - - pub fn deserialize(serialized: [Field; M]) -> Self { - let mut before_sibling_path = [0; N]; - let mut after_sibling_path = [0; N]; - - for i in 0..N { - before_sibling_path[i] = serialized[3 + i]; - after_sibling_path[i] = serialized[4 + N + i]; + pub fn serialize(self: Self) -> [Field; M] { + let mut serialized = [0; M]; + serialized[0] = self.index; + serialized[1] = self.new_value; + serialized[2] = self.before.value; + serialized[3 + N] = self.after.value; + + for i in 0..N { + serialized[3 + i] = self.before.sibling_path[i]; + serialized[4 + N + i] = self.after.sibling_path[i]; + } + serialized } - Self { - index: serialized[0], - new_value: serialized[1], - before: SlowUpdateInner { - value: serialized[2], - sibling_path: before_sibling_path, - }, - after: SlowUpdateInner { - value: serialized[3 + N], - sibling_path: after_sibling_path, - }, + pub fn deserialize(serialized: [Field; M]) -> Self { + let mut before_sibling_path = [0; N]; + let mut after_sibling_path = [0; N]; + + for i in 0..N { + before_sibling_path[i] = serialized[3 + i]; + after_sibling_path[i] = serialized[4 + N + i]; + } + + Self { + index: serialized[0], + new_value: serialized[1], + before: SlowUpdateInner { value: serialized[2], sibling_path: before_sibling_path }, + after: SlowUpdateInner { value: serialized[3 + N], sibling_path: after_sibling_path } + } } - } } // The simple slow map which stores a sparse tree @@ -99,161 +93,153 @@ struct SlowMap { } impl SlowMap { - pub fn new( - context: Context, - storage_slot: Field - ) -> Self { - assert(storage_slot != 0, "Storage slot 0 not allowed. Storage slots must start from 1."); - Self { - context, - storage_slot, + pub fn new(context: Context, storage_slot: Field) -> Self { + assert(storage_slot != 0, "Storage slot 0 not allowed. Storage slots must start from 1."); + Self { context, storage_slot } } - } - pub fn read_root(self: Self) -> Leaf { - let fields = storage_read(self.storage_slot); - Leaf::deserialize(fields) - } + pub fn read_root(self: Self) -> Leaf { + let fields = storage_read(self.storage_slot); + Leaf::deserialize(fields) + } - // Beware that the initial root could include much state that is not shown by the public storage! - pub fn initialize(self: Self, initial_root: Field) { - let mut root_object = self.read_root(); - assert(root_object.next_change == 0, "cannot initialize twice"); - root_object = Leaf { + // Beware that the initial root could include much state that is not shown by the public storage! + pub fn initialize(self: Self, initial_root: Field) { + let mut root_object = self.read_root(); + assert(root_object.next_change == 0, "cannot initialize twice"); + root_object = Leaf { next_change: 0xffffffffffffffffffffffffffffff, before: initial_root, after: initial_root, }; - let fields = root_object.serialize(); - storage_write(self.storage_slot, fields); - } + let fields = root_object.serialize(); + storage_write(self.storage_slot, fields); + } - // Reads the "CURRENT" value of the root - pub fn current_root(self: Self) -> Field { - let time = self.context.public.unwrap().timestamp() as u120; - let root_object = self.read_root(); - if time <= root_object.next_change as u120 { - root_object.before - } else { - root_object.after + // Reads the "CURRENT" value of the root + pub fn current_root(self: Self) -> Field { + let time = self.context.public.unwrap().timestamp() as u64; + let root_object = self.read_root(); + if time <= root_object.next_change as u64 { + root_object.before + } else { + root_object.after + } } - } - // docs:start:read_leaf_at - pub fn read_leaf_at(self: Self, key: Field) -> Leaf { - let derived_storage_slot = pedersen_hash([self.storage_slot, key]); - let fields = storage_read(derived_storage_slot); - Leaf::deserialize(fields) - } - // docs:end:read_leaf_at - - // docs:start:read_at - // Reads the "CURRENT" value of the leaf - pub fn read_at(self: Self, key: Field) -> Field { - let time = self.context.public.unwrap().timestamp() as u120; - let leaf = self.read_leaf_at(key); - if time <= leaf.next_change as u120 { - leaf.before - } else { - leaf.after + // docs:start:read_leaf_at + pub fn read_leaf_at(self: Self, key: Field) -> Leaf { + let derived_storage_slot = pedersen_hash([self.storage_slot, key]); + let fields = storage_read(derived_storage_slot); + Leaf::deserialize(fields) } - } - // docs:end:read_at - - // Will update values in the "AFTER" tree - // - updates the leaf and root to follow current values, moving from after to before if - // needed. - // - checks that the provided merkle paths match state values - // - update the leaf and compute the net root - // Should only be used when updates from public are desired, since the hashing will be - // costly since done by sequencer. - pub fn update_at(self: Self, p: SlowUpdateProof) { - // The calling function should ensure that the index is within the tree. - // This must be done separately to ensure we are not constraining too tight here. - - let time = self.context.public.unwrap().timestamp() as u120; - let next_change = compute_next_change(time as Field); - - let mut root = self.read_root(); - let mut leaf = self.read_leaf_at(p.index); - - // Move leaf if needed - if time > leaf.next_change as u120 { - leaf.before = leaf.after; + // docs:end:read_leaf_at + + // docs:start:read_at + // Reads the "CURRENT" value of the leaf + pub fn read_at(self: Self, key: Field) -> Field { + let time = self.context.public.unwrap().timestamp() as u64; + let leaf = self.read_leaf_at(key); + if time <= leaf.next_change as u64 { + leaf.before + } else { + leaf.after + } } - - // Move root if needed - if time > root.next_change as u120 { - root.before = root.after; + // docs:end:read_at + + // Will update values in the "AFTER" tree + // - updates the leaf and root to follow current values, moving from after to before if + // needed. + // - checks that the provided merkle paths match state values + // - update the leaf and compute the net root + // Should only be used when updates from public are desired, since the hashing will be + // costly since done by sequencer. + pub fn update_at(self: Self, p: SlowUpdateProof) { + // The calling function should ensure that the index is within the tree. + // This must be done separately to ensure we are not constraining too tight here. + + let time = self.context.public.unwrap().timestamp() as u64; + let next_change = compute_next_change(time as Field); + + let mut root = self.read_root(); + let mut leaf = self.read_leaf_at(p.index); + + // Move leaf if needed + if time > leaf.next_change as u64 { + leaf.before = leaf.after; + } + + // Move root if needed + if time > root.next_change as u64 { + root.before = root.after; + } + + // Ensures that when before is active, it is not altered by this update + assert( + root.before == compute_merkle_root(leaf.before, p.index, p.before.sibling_path), "Before root don't match" + ); + + // Ensures that the provided sibling path is valid for the CURRENT "after" tree. + // Without this check, someone could provide a sibling path for a different tree + // and update the entire "after" tree at once, causing it to be out of sync with leaf storage. + assert( + root.after == compute_merkle_root(leaf.after, p.index, p.after.sibling_path), "After root don't match" + ); + + // Update the leaf + leaf.after = p.new_value; + leaf.next_change = next_change; + + // Update the after root + root.after = compute_merkle_root(leaf.after, p.index, p.after.sibling_path); + root.next_change = next_change; + + self.update_unsafe(p.index, leaf, root); } - // Ensures that when before is active, it is not altered by this update - assert( - root.before == compute_merkle_root(leaf.before, p.index, p.before.sibling_path), - "Before root don't match" - ); - - // Ensures that the provided sibling path is valid for the CURRENT "after" tree. - // Without this check, someone could provide a sibling path for a different tree - // and update the entire "after" tree at once, causing it to be out of sync with leaf storage. - assert( - root.after == compute_merkle_root(leaf.after, p.index, p.after.sibling_path), - "After root don't match" - ); - - // Update the leaf - leaf.after = p.new_value; - leaf.next_change = next_change; - - // Update the after root - root.after = compute_merkle_root(leaf.after, p.index, p.after.sibling_path); - root.next_change = next_change; - - self.update_unsafe(p.index, leaf, root); - } - - // A variation of `update_at` that skips the merkle-membership checks. - // To be used by a contract which has already checked the merkle-membership. - // This allows us to check the merkle-memberships in private and then update - // in public, limiting the cost of the update. - pub fn update_unsafe_at(self: Self, index: Field, leaf_value: Field, new_root: Field) { - // User must ensure that the checks from update_at is performed for safety - let time = self.context.public.unwrap().timestamp() as u120; - let next_change = compute_next_change(time as Field); - - let mut root = self.read_root(); - let mut leaf = self.read_leaf_at(index); - - // Move leaf if needed - if time > leaf.next_change as u120 { - leaf.before = leaf.after; + // A variation of `update_at` that skips the merkle-membership checks. + // To be used by a contract which has already checked the merkle-membership. + // This allows us to check the merkle-memberships in private and then update + // in public, limiting the cost of the update. + pub fn update_unsafe_at(self: Self, index: Field, leaf_value: Field, new_root: Field) { + // User must ensure that the checks from update_at is performed for safety + let time = self.context.public.unwrap().timestamp() as u64; + let next_change = compute_next_change(time as Field); + + let mut root = self.read_root(); + let mut leaf = self.read_leaf_at(index); + + // Move leaf if needed + if time > leaf.next_change as u64 { + leaf.before = leaf.after; + } + + // Move root if needed + if time > root.next_change as u64 { + root.before = root.after; + } + + // Update the leaf + leaf.after = leaf_value; + leaf.next_change = next_change; + + // Update the root + root.after = new_root; + root.next_change = next_change; + + self.update_unsafe(index, leaf, root); } - - // Move root if needed - if time > root.next_change as u120 { - root.before = root.after; - } - - // Update the leaf - leaf.after = leaf_value; - leaf.next_change = next_change; - - // Update the root - root.after = new_root; - root.next_change = next_change; - self.update_unsafe(index, leaf, root); - } + // Updates the value in the in storage with no checks. + fn update_unsafe(self: Self, index: Field, leaf: Leaf, root: Leaf) { + let derived_storage_slot = pedersen_hash([self.storage_slot, index]); + let fields = leaf.serialize(); + storage_write(derived_storage_slot, fields); - // Updates the value in the in storage with no checks. - fn update_unsafe(self: Self, index: Field, leaf: Leaf, root: Leaf) { - let derived_storage_slot = pedersen_hash([self.storage_slot, index]); - let fields = leaf.serialize(); - storage_write(derived_storage_slot, fields); - - let fields = root.serialize(); - storage_write(self.storage_slot, fields); - } + let fields = root.serialize(); + storage_write(self.storage_slot, fields); + } } /*pub fn compute_merkle_root(leaf: Field, index: Field, hash_path: [Field; N]) -> Field { diff --git a/noir-projects/noir-contracts/contracts/price_feed_contract/src/asset.nr b/noir-projects/noir-contracts/contracts/price_feed_contract/src/asset.nr index b1358e1abb7..0f34a0429b1 100644 --- a/noir-projects/noir-contracts/contracts/price_feed_contract/src/asset.nr +++ b/noir-projects/noir-contracts/contracts/price_feed_contract/src/asset.nr @@ -14,7 +14,7 @@ impl Serialize for Asset { impl Deserialize for Asset { fn deserialize(fields: [Field; ASSET_SERIALIZED_LEN]) -> Asset { - let price = fields[0] as w; + let price = U128::from_integer(fields[0]); Asset { price } } } diff --git a/noir-projects/noir-contracts/contracts/price_feed_contract/src/main.nr b/noir-projects/noir-contracts/contracts/price_feed_contract/src/main.nr index fe5c919c025..b092547220c 100644 --- a/noir-projects/noir-contracts/contracts/price_feed_contract/src/main.nr +++ b/noir-projects/noir-contracts/contracts/price_feed_contract/src/main.nr @@ -17,7 +17,7 @@ contract PriceFeed { #[aztec(public)] fn set_price(asset_id: Field, price: Field) { let asset = storage.assets.at(asset_id); - asset.write(Asset { price }); + asset.write(Asset { price: U128::from_integer(price) }); } #[aztec(public)] diff --git a/noir-projects/noir-contracts/contracts/token_contract/src/main.nr b/noir-projects/noir-contracts/contracts/token_contract/src/main.nr index e72e39fa87f..4a3667b383a 100644 --- a/noir-projects/noir-contracts/contracts/token_contract/src/main.nr +++ b/noir-projects/noir-contracts/contracts/token_contract/src/main.nr @@ -16,33 +16,17 @@ contract Token { use dep::compressed_string::FieldCompressedString; use dep::aztec::{ - note::{ - note_getter_options::NoteGetterOptions, - note_header::NoteHeader, - utils as note_utils, - }, + note::{note_getter_options::NoteGetterOptions, note_header::NoteHeader, utils as note_utils}, hash::{compute_secret_hash}, - state_vars::{map::Map, public_state::PublicState, stable_public_state::StablePublicState, set::Set}, - protocol_types::{ - abis::function_selector::FunctionSelector, - address::AztecAddress - } + state_vars::{map::Map, public_state::PublicState, stable_public_state::StablePublicState, set::Set}, + protocol_types::{abis::function_selector::FunctionSelector, address::AztecAddress} }; // docs:start:import_authwit - use dep::authwit::{ - auth::{ - assert_current_call_valid_authwit, - assert_current_call_valid_authwit_public, - }, - }; + use dep::authwit::{auth::{assert_current_call_valid_authwit, assert_current_call_valid_authwit_public}}; // docs:end:import_authwit - use crate::types::{ - transparent_note::TransparentNote, - token_note::{TokenNote, TOKEN_NOTE_LEN}, - balances_map::BalancesMap - }; + use crate::types::{transparent_note::TransparentNote, token_note::{TokenNote, TOKEN_NOTE_LEN}, balances_map::BalancesMap}; // docs:end::imports // docs:start:storage_struct @@ -56,11 +40,11 @@ contract Token { // docs:start:storage_balances balances: BalancesMap, // docs:end:storage_balances - total_supply: PublicState, + total_supply: PublicState, // docs:start:storage_pending_shields pending_shields: Set, // docs:end:storage_pending_shields - public_balances: Map>, + public_balances: Map>, symbol: StablePublicState, name: StablePublicState, // docs:start:storage_decimals @@ -157,7 +141,7 @@ contract Token { // docs:start:read_minter assert(storage.minters.at(context.msg_sender()).read(), "caller is not minter"); // docs:end:read_minter - let amount = SafeU120::new(amount); + let amount = U128::from_integer(amount); let new_balance = storage.public_balances.at(to).read().add(amount); let supply = storage.total_supply.read().add(amount); @@ -172,7 +156,7 @@ contract Token { assert(storage.minters.at(context.msg_sender()).read(), "caller is not minter"); let pending_shields = storage.pending_shields; let mut note = TransparentNote::new(amount, secret_hash); - let supply = storage.total_supply.read().add(SafeU120::new(amount)); + let supply = storage.total_supply.read().add(U128::from_integer(amount)); storage.total_supply.write(supply); // docs:start:insert_from_public @@ -191,11 +175,11 @@ contract Token { assert(nonce == 0, "invalid nonce"); } - let amount = SafeU120::new(amount); + let amount = U128::from_integer(amount); let from_balance = storage.public_balances.at(from).read().sub(amount); let pending_shields = storage.pending_shields; - let mut note = TransparentNote::new(amount.value as Field, secret_hash); + let mut note = TransparentNote::new(amount.to_integer(), secret_hash); storage.public_balances.at(from).write(from_balance); pending_shields.insert_from_public(&mut note); @@ -211,7 +195,7 @@ contract Token { assert(nonce == 0, "invalid nonce"); } - let amount = SafeU120::new(amount); + let amount = U128::from_integer(amount); let from_balance = storage.public_balances.at(from).read().sub(amount); storage.public_balances.at(from).write(from_balance); @@ -231,7 +215,7 @@ contract Token { } // docs:end:assert_current_call_valid_authwit_public - let amount = SafeU120::new(amount); + let amount = U128::from_integer(amount); let from_balance = storage.public_balances.at(from).read().sub(amount); storage.public_balances.at(from).write(from_balance); @@ -254,7 +238,7 @@ contract Token { pending_shields.remove(note); // Add the token note to user's balances set - storage.balances.add(to, SafeU120::new(amount)); + storage.balances.add(to, U128::from_integer(amount)); } // docs:end:redeem_shield @@ -267,7 +251,7 @@ contract Token { assert(nonce == 0, "invalid nonce"); } - storage.balances.sub(from, SafeU120::new(amount)); + storage.balances.sub(from, U128::from_integer(amount)); let selector = FunctionSelector::from_signature("_increase_public_balance((Field),Field)"); let _void = context.call_public_function(context.this_address(), selector, [to.to_field(), amount]); @@ -285,7 +269,7 @@ contract Token { } // docs:end:assert_current_call_valid_authwit - let amount = SafeU120::new(amount); + let amount = U128::from_integer(amount); storage.balances.sub(from, amount); // docs:start:increase_private_balance storage.balances.add(to, amount); @@ -302,7 +286,7 @@ contract Token { assert(nonce == 0, "invalid nonce"); } - storage.balances.sub(from, SafeU120::new(amount)); + storage.balances.sub(from, U128::from_integer(amount)); let selector = FunctionSelector::from_signature("_reduce_total_supply(Field)"); let _void = context.call_public_function(context.this_address(), selector, [amount]); @@ -333,7 +317,7 @@ contract Token { // docs:start:increase_public_balance #[aztec(public)] internal fn _increase_public_balance(to: AztecAddress, amount: Field) { - let new_balance = storage.public_balances.at(to).read().add(SafeU120::new(amount)); + let new_balance = storage.public_balances.at(to).read().add(U128::from_integer(amount)); storage.public_balances.at(to).write(new_balance); } // docs:end:increase_public_balance @@ -342,7 +326,7 @@ contract Token { #[aztec(public)] internal fn _reduce_total_supply(amount: Field) { // Only to be called from burn. - let new_supply = storage.total_supply.read().sub(SafeU120::new(amount)); + let new_supply = storage.total_supply.read().sub(U128::from_integer(amount)); storage.total_supply.write(new_supply); } // docs:end:reduce_total_supply @@ -362,20 +346,20 @@ contract Token { // docs:end:is_minter // docs:start:total_supply - unconstrained fn total_supply() -> pub u120 { - storage.total_supply.read().value + unconstrained fn total_supply() -> pub Field { + storage.total_supply.read().to_integer() } // docs:end:total_supply // docs:start:balance_of_private - unconstrained fn balance_of_private(owner: AztecAddress) -> pub u120 { - storage.balances.balance_of(owner).value + unconstrained fn balance_of_private(owner: AztecAddress) -> pub Field { + storage.balances.balance_of(owner).to_integer() } // docs:end:balance_of_private // docs:start:balance_of_public - unconstrained fn balance_of_public(owner: AztecAddress) -> pub u120 { - storage.public_balances.at(owner).read().value + unconstrained fn balance_of_public(owner: AztecAddress) -> pub Field { + storage.public_balances.at(owner).read().to_integer() } // docs:end:balance_of_public diff --git a/noir-projects/noir-protocol-circuits/src/crates/types/src/type_serialization.nr b/noir-projects/noir-protocol-circuits/src/crates/types/src/type_serialization.nr index 1a9377bfb72..6ce66c3f63a 100644 --- a/noir-projects/noir-protocol-circuits/src/crates/types/src/type_serialization.nr +++ b/noir-projects/noir-protocol-circuits/src/crates/types/src/type_serialization.nr @@ -55,15 +55,15 @@ impl Deserialize for bool { } } -impl Serialize<2> for U128 { - fn serialize(self) -> [Field; 2] { - [self.lo, self.hi] +impl Serialize<1> for U128 { + fn serialize(self) -> [Field; 1] { + [self.to_integer()] } } -impl Deserialize<2> for U128 { - fn deserialize(fields: [Field; 2]) -> Self { - U128 {lo: fields[0], hi: fields[1]} +impl Deserialize<1> for U128 { + fn deserialize(fields: [Field; 1]) -> Self { + U128::from_integer(fields[0]) } }