diff --git a/compiler/noirc_evaluator/src/acir/generated_acir.rs b/compiler/noirc_evaluator/src/acir/generated_acir.rs index 14ceac62461..8b29888a7ad 100644 --- a/compiler/noirc_evaluator/src/acir/generated_acir.rs +++ b/compiler/noirc_evaluator/src/acir/generated_acir.rs @@ -362,12 +362,17 @@ impl GeneratedAcir { bit_size: u32, ) -> Result, RuntimeError> { let radix_big = BigUint::from(radix); + let radix_range = BigUint::from(2u128)..=BigUint::from(256u128); + assert!( + radix_range.contains(&radix_big), + "ICE: Radix must be in the range 2..=256, but found: {:?}", + radix + ); assert_eq!( BigUint::from(2u128).pow(bit_size), radix_big, "ICE: Radix must be a power of 2" ); - let limb_witnesses = self.brillig_to_radix(input_expr, radix, limb_count); let mut composed_limbs = Expression::default(); diff --git a/compiler/noirc_evaluator/src/ssa/ir/instruction/call.rs b/compiler/noirc_evaluator/src/ssa/ir/instruction/call.rs index 992c633ffcd..6ee7aa0192c 100644 --- a/compiler/noirc_evaluator/src/ssa/ir/instruction/call.rs +++ b/compiler/noirc_evaluator/src/ssa/ir/instruction/call.rs @@ -650,7 +650,12 @@ fn constant_to_radix( ) -> SimplifyResult { let bit_size = u32::BITS - (radix - 1).leading_zeros(); let radix_big = BigUint::from(radix); - assert_eq!(BigUint::from(2u128).pow(bit_size), radix_big, "ICE: Radix must be a power of 2"); + let radix_range = BigUint::from(2u128)..=BigUint::from(256u128); + if !radix_range.contains(&radix_big) || BigUint::from(2u128).pow(bit_size) != radix_big { + // NOTE: expect an error to be thrown later in + // acir::generated_acir::radix_le_decompose + return SimplifyResult::None; + } let big_integer = BigUint::from_bytes_be(&field.to_be_bytes()); // Decompose the integer into its radix digits in little endian form. diff --git a/compiler/noirc_frontend/src/hir/comptime/interpreter/builtin.rs b/compiler/noirc_frontend/src/hir/comptime/interpreter/builtin.rs index 3f6b10a0176..7909a423bef 100644 --- a/compiler/noirc_frontend/src/hir/comptime/interpreter/builtin.rs +++ b/compiler/noirc_frontend/src/hir/comptime/interpreter/builtin.rs @@ -66,7 +66,7 @@ impl<'local, 'context> Interpreter<'local, 'context> { "array_as_str_unchecked" => array_as_str_unchecked(interner, arguments, location), "array_len" => array_len(interner, arguments, location), "array_refcount" => Ok(Value::U32(0)), - "assert_constant" => Ok(Value::Bool(true)), + "assert_constant" => Ok(Value::Unit), "as_slice" => as_slice(interner, arguments, location), "ctstring_eq" => ctstring_eq(arguments, location), "ctstring_hash" => ctstring_hash(arguments, location), @@ -175,6 +175,7 @@ impl<'local, 'context> Interpreter<'local, 'context> { "slice_push_front" => slice_push_front(interner, arguments, location), "slice_refcount" => Ok(Value::U32(0)), "slice_remove" => slice_remove(interner, arguments, location, call_stack), + "static_assert" => static_assert(interner, arguments, location, call_stack), "str_as_bytes" => str_as_bytes(interner, arguments, location), "str_as_ctstring" => str_as_ctstring(interner, arguments, location), "struct_def_add_attribute" => struct_def_add_attribute(interner, arguments, location), @@ -327,6 +328,28 @@ fn slice_push_back( Ok(Value::Slice(values, typ)) } +// static_assert(predicate: bool, message: str) +fn static_assert( + interner: &NodeInterner, + arguments: Vec<(Value, Location)>, + location: Location, + call_stack: &im::Vector, +) -> IResult { + let (predicate, message) = check_two_arguments(arguments, location)?; + let predicate = get_bool(predicate)?; + let message = get_str(interner, message)?; + + if predicate { + Ok(Value::Unit) + } else { + failing_constraint( + format!("static_assert failed: {}", message).clone(), + location, + call_stack, + ) + } +} + fn str_as_bytes( interner: &NodeInterner, arguments: Vec<(Value, Location)>, diff --git a/compiler/noirc_frontend/src/tests/metaprogramming.rs b/compiler/noirc_frontend/src/tests/metaprogramming.rs index 8256744e18f..b42342fa47d 100644 --- a/compiler/noirc_frontend/src/tests/metaprogramming.rs +++ b/compiler/noirc_frontend/src/tests/metaprogramming.rs @@ -3,6 +3,7 @@ use noirc_errors::Spanned; use crate::{ ast::Ident, hir::{ + comptime::InterpreterError, def_collector::{ dc_crate::CompilationError, errors::{DefCollectorErrorKind, DuplicateType}, @@ -26,6 +27,26 @@ fn comptime_let() { assert_eq!(errors.len(), 0); } +#[test] +fn comptime_code_rejects_dynamic_variable() { + let src = r#"fn main(x: Field) { + comptime let my_var = (x - x) + 2; + assert_eq(my_var, 2); + }"#; + let errors = get_program_errors(src); + + assert_eq!(errors.len(), 1); + match &errors[0].0 { + CompilationError::InterpreterError(InterpreterError::NonComptimeVarReferenced { + name, + .. + }) => { + assert_eq!(name, "x"); + } + _ => panic!("expected an InterpreterError"), + } +} + #[test] fn comptime_type_in_runtime_code() { let source = "pub fn foo(_f: FunctionDefinition) {}"; diff --git a/noir_stdlib/src/collections/bounded_vec.nr b/noir_stdlib/src/collections/bounded_vec.nr index a1befdd58ec..c030544e791 100644 --- a/noir_stdlib/src/collections/bounded_vec.nr +++ b/noir_stdlib/src/collections/bounded_vec.nr @@ -1,4 +1,4 @@ -use crate::{cmp::Eq, convert::From, runtime::is_unconstrained}; +use crate::{cmp::Eq, convert::From, runtime::is_unconstrained, static_assert}; /// A `BoundedVec` is a growable storage similar to a `Vec` except that it /// is bounded with a maximum possible length. Unlike `Vec`, `BoundedVec` is not implemented @@ -345,7 +345,7 @@ impl BoundedVec { /// let bounded_vec: BoundedVec = BoundedVec::from_array([1, 2, 3]) /// ``` pub fn from_array(array: [T; Len]) -> Self { - assert(Len <= MaxLen, "from array out of bounds"); + static_assert(Len <= MaxLen, "from array out of bounds"); let mut vec: BoundedVec = BoundedVec::new(); vec.extend_from_array(array); vec diff --git a/noir_stdlib/src/field/mod.nr b/noir_stdlib/src/field/mod.nr index d0760447ff1..255236477ff 100644 --- a/noir_stdlib/src/field/mod.nr +++ b/noir_stdlib/src/field/mod.nr @@ -1,5 +1,5 @@ pub mod bn254; -use crate::runtime::is_unconstrained; +use crate::{runtime::is_unconstrained, static_assert}; use bn254::lt as bn254_lt; impl Field { @@ -10,7 +10,10 @@ impl Field { // docs:start:assert_max_bit_size pub fn assert_max_bit_size(self) { // docs:end:assert_max_bit_size - assert(BIT_SIZE < modulus_num_bits() as u32); + static_assert( + BIT_SIZE < modulus_num_bits() as u32, + "BIT_SIZE must be less than modulus_num_bits", + ); self.__assert_max_bit_size(BIT_SIZE); } @@ -61,6 +64,10 @@ impl Field { // docs:start:to_le_bytes pub fn to_le_bytes(self: Self) -> [u8; N] { // docs:end:to_le_bytes + static_assert( + N <= modulus_le_bytes().len(), + "N must be less than or equal to modulus_le_bytes().len()", + ); // Compute the byte decomposition let bytes = self.to_le_radix(256); @@ -94,6 +101,10 @@ impl Field { // docs:start:to_be_bytes pub fn to_be_bytes(self: Self) -> [u8; N] { // docs:end:to_be_bytes + static_assert( + N <= modulus_le_bytes().len(), + "N must be less than or equal to modulus_le_bytes().len()", + ); // Compute the byte decomposition let bytes = self.to_be_radix(256); @@ -119,7 +130,9 @@ impl Field { pub fn to_le_radix(self: Self, radix: u32) -> [u8; N] { // Brillig does not need an immediate radix if !crate::runtime::is_unconstrained() { - crate::assert_constant(radix); + static_assert(1 < radix, "radix must be greater than 1"); + static_assert(radix <= 256, "radix must be less than or equal to 256"); + static_assert(radix & (radix - 1) == 0, "radix must be a power of 2"); } self.__to_le_radix(radix) } @@ -139,6 +152,7 @@ impl Field { #[builtin(to_le_radix)] fn __to_le_radix(self, radix: u32) -> [u8; N] {} + // `_radix` must be less than 256 #[builtin(to_be_radix)] fn __to_be_radix(self, radix: u32) -> [u8; N] {} @@ -172,6 +186,10 @@ impl Field { /// Convert a little endian byte array to a field element. /// If the provided byte array overflows the field modulus then the Field will silently wrap around. pub fn from_le_bytes(bytes: [u8; N]) -> Field { + static_assert( + N <= modulus_le_bytes().len(), + "N must be less than or equal to modulus_le_bytes().len()", + ); let mut v = 1; let mut result = 0; @@ -262,6 +280,7 @@ fn lt_fallback(x: Field, y: Field) -> bool { } mod tests { + use crate::{panic::panic, runtime}; use super::field_less_than; #[test] @@ -322,6 +341,75 @@ mod tests { } // docs:end:to_le_radix_example + #[test(should_fail_with = "radix must be greater than 1")] + fn test_to_le_radix_1() { + // this test should only fail in constrained mode + if !runtime::is_unconstrained() { + let field = 2; + let _: [u8; 8] = field.to_le_radix(1); + } else { + panic(f"radix must be greater than 1"); + } + } + + #[test] + fn test_to_le_radix_brillig_1() { + // this test should only fail in constrained mode + if runtime::is_unconstrained() { + let field = 1; + let out: [u8; 8] = field.to_le_radix(1); + crate::println(out); + let expected = [0; 8]; + assert(out == expected, "unexpected result"); + } + } + + #[test(should_fail_with = "radix must be a power of 2")] + fn test_to_le_radix_3() { + // this test should only fail in constrained mode + if !runtime::is_unconstrained() { + let field = 2; + let _: [u8; 8] = field.to_le_radix(3); + } else { + panic(f"radix must be a power of 2"); + } + } + + #[test] + fn test_to_le_radix_brillig_3() { + // this test should only fail in constrained mode + if runtime::is_unconstrained() { + let field = 1; + let out: [u8; 8] = field.to_le_radix(3); + let mut expected = [0; 8]; + expected[0] = 1; + assert(out == expected, "unexpected result"); + } + } + + #[test(should_fail_with = "radix must be less than or equal to 256")] + fn test_to_le_radix_512() { + // this test should only fail in constrained mode + if !runtime::is_unconstrained() { + let field = 2; + let _: [u8; 8] = field.to_le_radix(512); + } else { + panic(f"radix must be less than or equal to 256") + } + } + + #[test] + fn test_to_le_radix_brillig_512() { + // this test should only fail in constrained mode + if runtime::is_unconstrained() { + let field = 1; + let out: [u8; 8] = field.to_le_radix(512); + let mut expected = [0; 8]; + expected[0] = 1; + assert(out == expected, "unexpected result"); + } + } + #[test] unconstrained fn test_field_less_than() { assert(field_less_than(0, 1)); diff --git a/noir_stdlib/src/meta/ctstring.nr b/noir_stdlib/src/meta/ctstring.nr index e23567ece7d..00b4f1fdb6f 100644 --- a/noir_stdlib/src/meta/ctstring.nr +++ b/noir_stdlib/src/meta/ctstring.nr @@ -7,7 +7,8 @@ impl CtString { "".as_ctstring() } - // Bug: using &mut self as the object results in this method not being found + // TODO(https://github.com/noir-lang/noir/issues/6980): Bug: using &mut self + // as the object results in this method not being found // docs:start:append_str pub comptime fn append_str(self, s: str) -> Self { // docs:end:append_str diff --git a/noir_stdlib/src/uint128.nr b/noir_stdlib/src/uint128.nr index 9aa01e1ea52..446ed4fb092 100644 --- a/noir_stdlib/src/uint128.nr +++ b/noir_stdlib/src/uint128.nr @@ -1,5 +1,6 @@ use crate::cmp::{Eq, Ord, Ordering}; use crate::ops::{Add, BitAnd, BitOr, BitXor, Div, Mul, Not, Rem, Shl, Shr, Sub}; +use crate::static_assert; use super::{convert::AsPrimitive, default::Default}; global pow64: Field = 18446744073709551616; //2^64; @@ -67,11 +68,10 @@ impl U128 { } pub fn from_hex(hex: str) -> U128 { - let N = N as u32; let bytes = hex.as_bytes(); // string must starts with "0x" assert((bytes[0] == 48) & (bytes[1] == 120), "Invalid hexadecimal string"); - assert(N < 35, "Input does not fit into a U128"); + static_assert(N < 35, "Input does not fit into a U128"); let mut lo = 0; let mut hi = 0; diff --git a/test_programs/compile_failure/comptime_static_assert_failure/Nargo.toml b/test_programs/compile_failure/comptime_static_assert_failure/Nargo.toml new file mode 100644 index 00000000000..006fd9f7ffe --- /dev/null +++ b/test_programs/compile_failure/comptime_static_assert_failure/Nargo.toml @@ -0,0 +1,6 @@ +[package] +name = "comptime_static_assert_failure" +type = "bin" +authors = [""] + +[dependencies] \ No newline at end of file diff --git a/test_programs/compile_failure/comptime_static_assert_failure/src/main.nr b/test_programs/compile_failure/comptime_static_assert_failure/src/main.nr new file mode 100644 index 00000000000..fcd757f4c94 --- /dev/null +++ b/test_programs/compile_failure/comptime_static_assert_failure/src/main.nr @@ -0,0 +1,13 @@ +use std::static_assert; + +comptime fn foo(x: Field) -> bool { + static_assert(x == 4, "x != 4"); + x == 4 +} + +fn main() { + comptime { + static_assert(foo(3), "expected message"); + } +} + diff --git a/test_programs/compile_success_empty/comptime_static_assert/Nargo.toml b/test_programs/compile_success_empty/comptime_static_assert/Nargo.toml new file mode 100644 index 00000000000..4c969fe7a79 --- /dev/null +++ b/test_programs/compile_success_empty/comptime_static_assert/Nargo.toml @@ -0,0 +1,6 @@ +[package] +name = "comptime_static_assert" +type = "bin" +authors = [""] + +[dependencies] \ No newline at end of file diff --git a/test_programs/compile_success_empty/comptime_static_assert/src/main.nr b/test_programs/compile_success_empty/comptime_static_assert/src/main.nr new file mode 100644 index 00000000000..2ddbba7b0de --- /dev/null +++ b/test_programs/compile_success_empty/comptime_static_assert/src/main.nr @@ -0,0 +1,19 @@ +use std::static_assert; + +comptime fn foo(x: Field) -> bool { + static_assert(x == 4, "x != 4"); + x == 4 +} + +global C: bool = { + let out = foo(2 + 2); + static_assert(out, "foo did not pass in C"); + out +}; + +fn main() { + comptime { + static_assert(foo(4), "foo did not pass in main"); + static_assert(C, "C did not pass") + } +} diff --git a/tooling/nargo_cli/src/cli/compile_cmd.rs b/tooling/nargo_cli/src/cli/compile_cmd.rs index 0af05703c9a..8a4b991a234 100644 --- a/tooling/nargo_cli/src/cli/compile_cmd.rs +++ b/tooling/nargo_cli/src/cli/compile_cmd.rs @@ -335,6 +335,7 @@ mod tests { use noirc_driver::{CompileOptions, CrateName}; use crate::cli::compile_cmd::{get_target_width, parse_workspace, read_workspace}; + use crate::cli::test_cmd::formatters::diagnostic_to_string; /// Try to find the directory that Cargo sets when it is running; /// otherwise fallback to assuming the CWD is the root of the repository @@ -414,7 +415,12 @@ mod tests { &CompileOptions::default(), None, ) - .expect("failed to compile"); + .unwrap_or_else(|err| { + for diagnostic in err { + println!("{}", diagnostic_to_string(&diagnostic, &file_manager)); + } + panic!("Failed to compile") + }); let width = get_target_width(package.expression_width, None); diff --git a/tooling/nargo_cli/src/cli/test_cmd.rs b/tooling/nargo_cli/src/cli/test_cmd.rs index 9bf3ae9fedf..4ba734386fe 100644 --- a/tooling/nargo_cli/src/cli/test_cmd.rs +++ b/tooling/nargo_cli/src/cli/test_cmd.rs @@ -26,7 +26,7 @@ use crate::{cli::check_cmd::check_crate_and_report_errors, errors::CliError}; use super::{NargoConfig, PackageOptions}; -mod formatters; +pub(crate) mod formatters; /// Run the tests for this program #[derive(Debug, Clone, Args)] diff --git a/tooling/nargo_cli/src/cli/test_cmd/formatters.rs b/tooling/nargo_cli/src/cli/test_cmd/formatters.rs index 75cf14ba120..bc4621c92ea 100644 --- a/tooling/nargo_cli/src/cli/test_cmd/formatters.rs +++ b/tooling/nargo_cli/src/cli/test_cmd/formatters.rs @@ -514,7 +514,10 @@ fn package_start(package_name: &str, test_count: usize) -> std::io::Result<()> { Ok(()) } -fn diagnostic_to_string(file_diagnostic: &FileDiagnostic, file_manager: &FileManager) -> String { +pub(crate) fn diagnostic_to_string( + file_diagnostic: &FileDiagnostic, + file_manager: &FileManager, +) -> String { let file_map = file_manager.as_file_map(); let custom_diagnostic = &file_diagnostic.diagnostic;