From a2d034505ebaa8097877cbcd0619ca930b251d6c Mon Sep 17 00:00:00 2001 From: Blaine Bublitz Date: Fri, 18 Aug 2023 14:54:36 -0700 Subject: [PATCH] chore: Decouple `noirc_abi` from frontend by introducing `PrintableType` --- Cargo.lock | 14 +- Cargo.toml | 2 + crates/nargo/Cargo.toml | 1 + crates/nargo/src/ops/foreign_calls.rs | 53 +++-- crates/noirc_abi/Cargo.toml | 1 + crates/noirc_abi/src/input_parser/mod.rs | 33 +-- crates/noirc_abi/src/lib.rs | 75 ++++++- crates/noirc_driver/src/lib.rs | 2 +- crates/noirc_evaluator/src/ssa/abi_gen/mod.rs | 6 +- crates/noirc_frontend/Cargo.toml | 3 +- crates/noirc_frontend/src/ast/expression.rs | 3 +- crates/noirc_frontend/src/ast/mod.rs | 21 -- crates/noirc_frontend/src/hir_def/types.rs | 107 +++++----- .../src/monomorphization/mod.rs | 24 +-- crates/noirc_printable_type/Cargo.toml | 14 ++ crates/noirc_printable_type/src/lib.rs | 199 ++++++++++++++++++ 16 files changed, 407 insertions(+), 151 deletions(-) create mode 100644 crates/noirc_printable_type/Cargo.toml create mode 100644 crates/noirc_printable_type/src/lib.rs diff --git a/Cargo.lock b/Cargo.lock index be1270f4fa2..d032e2074eb 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -2234,6 +2234,7 @@ dependencies = [ "noirc_driver", "noirc_errors", "noirc_frontend", + "noirc_printable_type", "regex", "rustc_version", "serde", @@ -2354,6 +2355,7 @@ version = "0.10.3" dependencies = [ "acvm", "iter-extended", + "noirc_frontend", "serde", "serde_json", "strum", @@ -2413,11 +2415,10 @@ dependencies = [ "chumsky", "fm", "iter-extended", - "noirc_abi", "noirc_errors", + "noirc_printable_type", "regex", "rustc-hash", - "serde", "serde_json", "small-ord-set", "smol_str", @@ -2426,6 +2427,15 @@ dependencies = [ "thiserror", ] +[[package]] +name = "noirc_printable_type" +version = "0.10.3" +dependencies = [ + "acvm", + "iter-extended", + "serde", +] + [[package]] name = "nom" version = "7.1.3" diff --git a/Cargo.toml b/Cargo.toml index 76ec9edfa0d..998eb2bb65b 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -5,6 +5,7 @@ members = [ "crates/noirc_frontend", "crates/noirc_errors", "crates/noirc_driver", + "crates/noirc_printable_type", "crates/nargo", "crates/nargo_cli", "crates/nargo_toml", @@ -38,6 +39,7 @@ noirc_driver = { path = "crates/noirc_driver" } noirc_errors = { path = "crates/noirc_errors" } noirc_evaluator = { path = "crates/noirc_evaluator" } noirc_frontend = { path = "crates/noirc_frontend" } +noirc_printable_type = { path = "crates/noirc_printable_type" } noir_wasm = { path = "crates/wasm" } cfg-if = "1.0.0" clap = { version = "4.3.19", features = ["derive"] } diff --git a/crates/nargo/Cargo.toml b/crates/nargo/Cargo.toml index 32ca04ad34f..cb72ba7b072 100644 --- a/crates/nargo/Cargo.toml +++ b/crates/nargo/Cargo.toml @@ -16,6 +16,7 @@ fm.workspace = true noirc_abi.workspace = true noirc_driver.workspace = true noirc_frontend.workspace = true +noirc_printable_type.workspace = true iter-extended.workspace = true serde.workspace = true serde_json.workspace = true diff --git a/crates/nargo/src/ops/foreign_calls.rs b/crates/nargo/src/ops/foreign_calls.rs index e1768d71c44..b94220110d4 100644 --- a/crates/nargo/src/ops/foreign_calls.rs +++ b/crates/nargo/src/ops/foreign_calls.rs @@ -3,7 +3,9 @@ use acvm::{ pwg::ForeignCallWaitInfo, }; use iter_extended::vecmap; -use noirc_abi::{decode_string_value, input_parser::InputValueDisplay, AbiType}; +use noirc_printable_type::{ + decode_string_value, decode_value, PrintableType, PrintableValueDisplay, +}; use regex::{Captures, Regex}; use crate::errors::ForeignCallError; @@ -93,18 +95,18 @@ impl ForeignCall { } fn convert_string_inputs(foreign_call_inputs: &[Vec]) -> Result { - // Fetch the abi type from the foreign call input + // Fetch the PrintableType from the foreign call input // The remaining input values should hold what is to be printed - let (abi_type_as_values, input_values) = + let (printable_type_as_values, input_values) = foreign_call_inputs.split_last().ok_or(ForeignCallError::MissingForeignCallInputs)?; - let abi_type = fetch_abi_type(abi_type_as_values)?; + let printable_type = fetch_printable_type(printable_type_as_values)?; // We must use a flat map here as each value in a struct will be in a separate input value let mut input_values_as_fields = input_values.iter().flat_map(|values| vecmap(values, |value| value.to_field())); - let input_value_display = - InputValueDisplay::try_from_fields(&mut input_values_as_fields, abi_type)?; + let value = decode_value(&mut input_values_as_fields, &printable_type); + let input_value_display = PrintableValueDisplay::new(&value, &printable_type); Ok(input_value_display.to_string()) } @@ -112,34 +114,37 @@ fn convert_string_inputs(foreign_call_inputs: &[Vec]) -> Result], ) -> Result { - let (message_as_values, input_and_abi_values) = + let (message_as_values, input_and_printable_values) = foreign_call_inputs.split_first().ok_or(ForeignCallError::MissingForeignCallInputs)?; let message_as_fields = vecmap(message_as_values, |value| value.to_field()); let message_as_string = decode_string_value(&message_as_fields); - let (num_values, input_and_abi_values) = - input_and_abi_values.split_first().ok_or(ForeignCallError::MissingForeignCallInputs)?; + let (num_values, input_and_printable_values) = input_and_printable_values + .split_first() + .ok_or(ForeignCallError::MissingForeignCallInputs)?; let mut output_strings = Vec::new(); let num_values = num_values[0].to_field().to_u128() as usize; - let mut abi_types = Vec::new(); - for abi_values in input_and_abi_values.iter().skip(input_and_abi_values.len() - num_values) { - let abi_type = fetch_abi_type(abi_values)?; - abi_types.push(abi_type); + let mut printable_types = Vec::new(); + for printable_value in + input_and_printable_values.iter().skip(input_and_printable_values.len() - num_values) + { + let printable_type = fetch_printable_type(printable_value)?; + printable_types.push(printable_type); } for i in 0..num_values { - let abi_type = &abi_types[i]; - let type_size = abi_type.field_count() as usize; + let printable_type = &printable_types[i]; + let type_size = printable_type.field_count() as usize; - let mut input_values_as_fields = input_and_abi_values[i..(i + type_size)] + let mut input_values_as_fields = input_and_printable_values[i..(i + type_size)] .iter() .flat_map(|values| vecmap(values, |value| value.to_field())); - let input_value_display = - InputValueDisplay::try_from_fields(&mut input_values_as_fields, abi_type.clone())?; + let value = decode_value(&mut input_values_as_fields, printable_type); + let input_value_display = PrintableValueDisplay::new(&value, printable_type); output_strings.push(input_value_display.to_string()); } @@ -157,11 +162,13 @@ fn convert_fmt_string_inputs( Ok(formatted_str.into_owned()) } -fn fetch_abi_type(abi_type_as_values: &[Value]) -> Result { - let abi_type_as_fields = vecmap(abi_type_as_values, |value| value.to_field()); - let abi_type_as_string = decode_string_value(&abi_type_as_fields); - let abi_type: AbiType = serde_json::from_str(&abi_type_as_string) +fn fetch_printable_type( + printable_type_as_values: &[Value], +) -> Result { + let printable_type_as_fields = vecmap(printable_type_as_values, |value| value.to_field()); + let printable_type_as_string = decode_string_value(&printable_type_as_fields); + let printable_type: PrintableType = serde_json::from_str(&printable_type_as_string) .map_err(|err| ForeignCallError::InputParserError(err.into()))?; - Ok(abi_type) + Ok(printable_type) } diff --git a/crates/noirc_abi/Cargo.toml b/crates/noirc_abi/Cargo.toml index 6af0cfe78b3..45466061fde 100644 --- a/crates/noirc_abi/Cargo.toml +++ b/crates/noirc_abi/Cargo.toml @@ -9,6 +9,7 @@ edition.workspace = true [dependencies] acvm.workspace = true iter-extended.workspace = true +noirc_frontend.workspace = true toml.workspace = true serde_json = "1.0" serde.workspace = true diff --git a/crates/noirc_abi/src/input_parser/mod.rs b/crates/noirc_abi/src/input_parser/mod.rs index 3a317697534..11d40f338d5 100644 --- a/crates/noirc_abi/src/input_parser/mod.rs +++ b/crates/noirc_abi/src/input_parser/mod.rs @@ -6,8 +6,8 @@ use std::collections::BTreeMap; use acvm::FieldElement; use serde::Serialize; -use crate::errors::{AbiError, InputParserError}; -use crate::{decode_value, Abi, AbiType}; +use crate::errors::InputParserError; +use crate::{Abi, AbiType}; /// This is what all formats eventually transform into /// For example, a toml file will parse into TomlTypes /// and those TomlTypes will be mapped to Value @@ -67,35 +67,6 @@ impl InputValue { } } -/// In order to display an `InputValue` we need an `AbiType` to accurately -/// convert the value into a human-readable format. -pub struct InputValueDisplay { - input_value: InputValue, - abi_type: AbiType, -} - -impl InputValueDisplay { - pub fn try_from_fields( - field_iterator: &mut impl Iterator, - abi_type: AbiType, - ) -> Result { - let input_value = decode_value(field_iterator, &abi_type)?; - Ok(InputValueDisplay { input_value, abi_type }) - } -} - -impl std::fmt::Display for InputValueDisplay { - fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { - // From the docs: https://doc.rust-lang.org/std/fmt/struct.Error.html - // This type does not support transmission of an error other than that an error - // occurred. Any extra information must be arranged to be transmitted through - // some other means. - let json_value = json::JsonTypes::try_from_input_value(&self.input_value, &self.abi_type) - .map_err(|_| std::fmt::Error)?; - write!(f, "{}", serde_json::to_string(&json_value).map_err(|_| std::fmt::Error)?) - } -} - /// The different formats that are supported when parsing /// the initial witness values #[cfg_attr(test, derive(strum_macros::EnumIter))] diff --git a/crates/noirc_abi/src/lib.rs b/crates/noirc_abi/src/lib.rs index b68a03f270d..cc5144df00a 100644 --- a/crates/noirc_abi/src/lib.rs +++ b/crates/noirc_abi/src/lib.rs @@ -12,6 +12,7 @@ use acvm::{ use errors::AbiError; use input_parser::InputValue; use iter_extended::{try_btree_map, try_vecmap, vecmap}; +use noirc_frontend::{Signedness, Type, TypeBinding, TypeVariableKind, Visibility}; use serde::{Deserialize, Serialize}; // This is the ABI used to bridge the different TOML formats for the initial // witness, the partial witness generator and the interpreter. @@ -74,6 +75,24 @@ pub enum AbiVisibility { Private, } +impl From for AbiVisibility { + fn from(value: Visibility) -> Self { + match value { + Visibility::Public => AbiVisibility::Public, + Visibility::Private => AbiVisibility::Private, + } + } +} + +impl From<&Visibility> for AbiVisibility { + fn from(value: &Visibility) -> Self { + match value { + Visibility::Public => AbiVisibility::Public, + Visibility::Private => AbiVisibility::Private, + } + } +} + #[derive(Clone, Copy, Debug, PartialEq, Eq, Serialize, Deserialize)] #[serde(rename_all = "lowercase")] /// Represents whether the return value should compromise of unique witness indices such that no @@ -97,6 +116,60 @@ pub enum Sign { } impl AbiType { + // TODO: Add `Context` argument for resolving fully qualified struct paths + pub fn from_type(typ: &Type) -> Self { + // Note; use strict_eq instead of partial_eq when comparing field types + // in this method, you most likely want to distinguish between public and private + match typ { + Type::FieldElement => Self::Field, + Type::Array(size, typ) => { + let length = size + .evaluate_to_u64() + .expect("Cannot have variable sized arrays as a parameter to main"); + let typ = typ.as_ref(); + Self::Array { length, typ: Box::new(Self::from_type(typ)) } + } + Type::Integer(sign, bit_width) => { + let sign = match sign { + Signedness::Unsigned => Sign::Unsigned, + Signedness::Signed => Sign::Signed, + }; + + Self::Integer { sign, width: *bit_width } + } + Type::TypeVariable(binding, TypeVariableKind::IntegerOrField) => { + match &*binding.borrow() { + TypeBinding::Bound(typ) => Self::from_type(typ), + TypeBinding::Unbound(_) => Self::from_type(&Type::default_int_type()), + } + } + Type::Bool => Self::Boolean, + Type::String(size) => { + let size = size + .evaluate_to_u64() + .expect("Cannot have variable sized strings as a parameter to main"); + Self::String { length: size } + } + Type::FmtString(_, _) => unreachable!("format strings cannot be used in the abi"), + Type::Error => unreachable!(), + Type::Unit => unreachable!(), + Type::Constant(_) => unreachable!(), + Type::Struct(def, ref args) => { + let struct_type = def.borrow(); + let fields = struct_type.get_fields(args); + let fields = vecmap(fields, |(name, typ)| (name, Self::from_type(&typ))); + Self::Struct { fields, name: struct_type.name.to_string() } + } + Type::Tuple(_) => todo!("as_abi_type not yet implemented for tuple types"), + Type::TypeVariable(_, _) => unreachable!(), + Type::NamedGeneric(..) => unreachable!(), + Type::Forall(..) => unreachable!(), + Type::Function(_, _, _) => unreachable!(), + Type::MutableReference(_) => unreachable!("&mut cannot be used in the abi"), + Type::NotConstant => unreachable!(), + } + } + /// Returns the number of field elements required to represent the type once encoded. pub fn field_count(&self) -> u32 { match self { @@ -390,7 +463,7 @@ fn decode_value( Ok(value) } -pub fn decode_string_value(field_elements: &[FieldElement]) -> String { +fn decode_string_value(field_elements: &[FieldElement]) -> String { let string_as_slice = vecmap(field_elements, |e| { let mut field_as_bytes = e.to_be_bytes(); let char_byte = field_as_bytes.pop().unwrap(); // A character in a string is represented by a u8, thus we just want the last byte of the element diff --git a/crates/noirc_driver/src/lib.rs b/crates/noirc_driver/src/lib.rs index 1192416b98f..9fcc412d5a4 100644 --- a/crates/noirc_driver/src/lib.rs +++ b/crates/noirc_driver/src/lib.rs @@ -135,7 +135,7 @@ pub fn compute_function_abi( let (parameters, return_type) = func_meta.into_function_signature(); let parameters = into_abi_params(parameters, &context.def_interner); - let return_type = return_type.map(|typ| typ.as_abi_type()); + let return_type = return_type.map(|typ| AbiType::from_type(&typ)); Some((parameters, return_type)) } diff --git a/crates/noirc_evaluator/src/ssa/abi_gen/mod.rs b/crates/noirc_evaluator/src/ssa/abi_gen/mod.rs index cfefb3c9f73..f2c61715e37 100644 --- a/crates/noirc_evaluator/src/ssa/abi_gen/mod.rs +++ b/crates/noirc_evaluator/src/ssa/abi_gen/mod.rs @@ -2,7 +2,7 @@ use std::collections::BTreeMap; use acvm::acir::native_types::Witness; use iter_extended::{btree_map, vecmap}; -use noirc_abi::{Abi, AbiParameter}; +use noirc_abi::{Abi, AbiParameter, AbiType}; use noirc_frontend::{ hir_def::{ function::{FunctionSignature, Param}, @@ -27,7 +27,7 @@ pub fn into_abi_params(params: Vec, interner: &NodeInterner) -> Vec Abi { let (parameters, return_type) = func_sig; let parameters = into_abi_params(parameters, interner); - let return_type = return_type.map(|typ| typ.as_abi_type()); + let return_type = return_type.map(|typ| AbiType::from_type(&typ)); let param_witnesses = param_witnesses_from_abi_param(¶meters, input_witnesses); Abi { parameters, return_type, param_witnesses, return_witnesses } } diff --git a/crates/noirc_frontend/Cargo.toml b/crates/noirc_frontend/Cargo.toml index 1f902d2d399..636f2e74b2a 100644 --- a/crates/noirc_frontend/Cargo.toml +++ b/crates/noirc_frontend/Cargo.toml @@ -8,15 +8,14 @@ edition.workspace = true [dependencies] acvm.workspace = true -noirc_abi.workspace = true noirc_errors.workspace = true +noirc_printable_type.workspace = true fm.workspace = true arena.workspace = true iter-extended.workspace = true chumsky.workspace = true thiserror.workspace = true smol_str.workspace = true -serde.workspace = true serde_json.workspace = true rustc-hash = "1.1.0" small-ord-set = "0.1.3" diff --git a/crates/noirc_frontend/src/ast/expression.rs b/crates/noirc_frontend/src/ast/expression.rs index b1ef26a7cd8..8b15f6e3b9d 100644 --- a/crates/noirc_frontend/src/ast/expression.rs +++ b/crates/noirc_frontend/src/ast/expression.rs @@ -382,8 +382,7 @@ pub enum FunctionReturnType { /// Describes the types of smart contract functions that are allowed. /// - All Noir programs in the non-contract context can be seen as `Secret`. -#[derive(serde::Serialize, serde::Deserialize, Debug, Copy, Clone, PartialEq, Eq)] -#[serde(rename_all = "snake_case")] +#[derive(Debug, Copy, Clone, PartialEq, Eq)] pub enum ContractFunctionType { /// This function will be executed in a private /// context. diff --git a/crates/noirc_frontend/src/ast/mod.rs b/crates/noirc_frontend/src/ast/mod.rs index 19e0d5d2e62..0a4e69aa55f 100644 --- a/crates/noirc_frontend/src/ast/mod.rs +++ b/crates/noirc_frontend/src/ast/mod.rs @@ -14,7 +14,6 @@ mod type_alias; pub use expression::*; pub use function::*; -use noirc_abi::AbiVisibility; use noirc_errors::Span; pub use statement::*; pub use structure::*; @@ -248,26 +247,6 @@ impl std::fmt::Display for Visibility { } } -// TODO: Move this into noirc_abi when it depends upon noirc_frontend (instead of other way around) -impl From for AbiVisibility { - fn from(value: Visibility) -> Self { - match value { - Visibility::Public => AbiVisibility::Public, - Visibility::Private => AbiVisibility::Private, - } - } -} - -// TODO: Move this into noirc_abi when it depends upon noirc_frontend (instead of other way around) -impl From<&Visibility> for AbiVisibility { - fn from(value: &Visibility) -> Self { - match value { - Visibility::Public => AbiVisibility::Public, - Visibility::Private => AbiVisibility::Private, - } - } -} - #[derive(Clone, Copy, Debug, PartialEq, Eq)] /// Represents whether the return value should compromise of unique witness indices such that no /// index occurs within the program's abi more than once. diff --git a/crates/noirc_frontend/src/hir_def/types.rs b/crates/noirc_frontend/src/hir_def/types.rs index 1f2b9e8be74..7988c20d244 100644 --- a/crates/noirc_frontend/src/hir_def/types.rs +++ b/crates/noirc_frontend/src/hir_def/types.rs @@ -9,8 +9,8 @@ use crate::{ node_interner::{ExprId, NodeInterner, TypeAliasId}, }; use iter_extended::vecmap; -use noirc_abi::AbiType; use noirc_errors::Span; +use noirc_printable_type::PrintableType; use crate::{node_interner::StructId, node_interner::TraitId, Ident, Signedness}; @@ -1012,58 +1012,6 @@ impl Type { } } - // Note; use strict_eq instead of partial_eq when comparing field types - // in this method, you most likely want to distinguish between public and private - pub fn as_abi_type(&self) -> AbiType { - match self { - Type::FieldElement => AbiType::Field, - Type::Array(size, typ) => { - let length = size - .evaluate_to_u64() - .expect("Cannot have variable sized arrays as a parameter to main"); - AbiType::Array { length, typ: Box::new(typ.as_abi_type()) } - } - Type::Integer(sign, bit_width) => { - let sign = match sign { - Signedness::Unsigned => noirc_abi::Sign::Unsigned, - Signedness::Signed => noirc_abi::Sign::Signed, - }; - - AbiType::Integer { sign, width: *bit_width } - } - Type::TypeVariable(binding, TypeVariableKind::IntegerOrField) => { - match &*binding.borrow() { - TypeBinding::Bound(typ) => typ.as_abi_type(), - TypeBinding::Unbound(_) => Type::default_int_type().as_abi_type(), - } - } - Type::Bool => AbiType::Boolean, - Type::String(size) => { - let size = size - .evaluate_to_u64() - .expect("Cannot have variable sized strings as a parameter to main"); - AbiType::String { length: size } - } - Type::FmtString(_, _) => unreachable!("format strings cannot be used in the abi"), - Type::Error => unreachable!(), - Type::Unit => unreachable!(), - Type::Constant(_) => unreachable!(), - Type::Struct(def, args) => { - let struct_type = def.borrow(); - let fields = struct_type.get_fields(args); - let fields = vecmap(fields, |(name, typ)| (name, typ.as_abi_type())); - AbiType::Struct { fields, name: struct_type.name.to_string() } - } - Type::Tuple(_) => todo!("as_abi_type not yet implemented for tuple types"), - Type::TypeVariable(_, _) => unreachable!(), - Type::NamedGeneric(..) => unreachable!(), - Type::Forall(..) => unreachable!(), - Type::Function(_, _, _) => unreachable!(), - Type::MutableReference(_) => unreachable!("&mut cannot be used in the abi"), - Type::NotConstant => unreachable!(), - } - } - /// Iterate over the fields of this type. /// Panics if the type is not a struct or tuple. pub fn iter_fields(&self) -> impl Iterator { @@ -1334,3 +1282,56 @@ impl TypeVariableKind { } } } + +impl From for PrintableType { + fn from(value: Type) -> Self { + Self::from(&value) + } +} + +impl From<&Type> for PrintableType { + fn from(value: &Type) -> Self { + // Note; use strict_eq instead of partial_eq when comparing field types + // in this method, you most likely want to distinguish between public and private + match value { + Type::FieldElement => PrintableType::Field, + Type::Array(size, typ) => { + let length = size.evaluate_to_u64().expect("Cannot print variable sized arrays"); + let typ = typ.as_ref(); + 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 }, + }, + Type::TypeVariable(binding, TypeVariableKind::IntegerOrField) => { + match &*binding.borrow() { + TypeBinding::Bound(typ) => typ.into(), + TypeBinding::Unbound(_) => Type::default_int_type().into(), + } + } + Type::Bool => PrintableType::Boolean, + Type::String(size) => { + let size = size.evaluate_to_u64().expect("Cannot print variable sized strings"); + PrintableType::String { length: size } + } + Type::FmtString(_, _) => unreachable!("format strings cannot be printed"), + Type::Error => unreachable!(), + Type::Unit => unreachable!(), + Type::Constant(_) => unreachable!(), + Type::Struct(def, ref args) => { + let struct_type = def.borrow(); + let fields = struct_type.get_fields(args); + let fields = vecmap(fields, |(name, typ)| (name, typ.into())); + PrintableType::Struct { fields, name: struct_type.name.to_string() } + } + Type::Tuple(_) => todo!("printing tuple types is not yet implemented"), + Type::TypeVariable(_, _) => unreachable!(), + Type::NamedGeneric(..) => unreachable!(), + Type::Forall(..) => unreachable!(), + Type::Function(_, _, _) => unreachable!(), + Type::MutableReference(_) => unreachable!("cannot print &mut"), + Type::NotConstant => unreachable!(), + } + } +} diff --git a/crates/noirc_frontend/src/monomorphization/mod.rs b/crates/noirc_frontend/src/monomorphization/mod.rs index 0d3297bf8a4..477c923c4e8 100644 --- a/crates/noirc_frontend/src/monomorphization/mod.rs +++ b/crates/noirc_frontend/src/monomorphization/mod.rs @@ -10,6 +10,7 @@ //! function, will monomorphize the entire reachable program. use acvm::FieldElement; use iter_extended::{btree_map, vecmap}; +use noirc_printable_type::PrintableType; use std::collections::{BTreeMap, HashMap, VecDeque}; use crate::{ @@ -775,7 +776,7 @@ impl<'interner> Monomorphizer<'interner> { if name.as_str() == "println" { // Oracle calls are required to be wrapped in an unconstrained function // Thus, the only argument to the `println` oracle is expected to always be an ident - self.append_abi_arg(&hir_arguments[0], &mut arguments); + self.append_printable_type_info(&hir_arguments[0], &mut arguments); } } } @@ -828,17 +829,16 @@ impl<'interner> Monomorphizer<'interner> { } /// Adds a function argument that contains type metadata that is required to tell - /// a caller (such as nargo) how to convert values passed to an foreign call - /// back to a human-readable string. + /// `println` how to convert values passed to an foreign call back to a human-readable string. /// The values passed to an foreign call will be a simple list of field elements, /// thus requiring extra metadata to correctly decode this list of elements. /// - /// The Noir compiler has an `AbiType` that handles encoding/decoding a list + /// The Noir compiler has a `PrintableType` that handles encoding/decoding a list /// of field elements to/from JSON. The type metadata attached in this method - /// is the serialized `AbiType` for the argument passed to the function. - /// The caller that is running a Noir program should then deserialize the `AbiType`, + /// is the serialized `PrintableType` for the argument passed to the function. + /// The caller that is running a Noir program should then deserialize the `PrintableType`, /// and accurately decode the list of field elements passed to the foreign call. - fn append_abi_arg( + fn append_printable_type_info( &mut self, hir_argument: &HirExpression, arguments: &mut Vec, @@ -854,7 +854,7 @@ impl<'interner> Monomorphizer<'interner> { match *elements { Type::Tuple(element_types) => { for typ in element_types { - Self::append_abi_arg_inner(&typ, arguments); + Self::append_printable_type_info_inner(&typ, arguments); } } _ => unreachable!( @@ -864,7 +864,7 @@ impl<'interner> Monomorphizer<'interner> { true } _ => { - Self::append_abi_arg_inner(&typ, arguments); + Self::append_printable_type_info_inner(&typ, arguments); false } }; @@ -875,15 +875,15 @@ impl<'interner> Monomorphizer<'interner> { } } - fn append_abi_arg_inner(typ: &Type, arguments: &mut Vec) { + fn append_printable_type_info_inner(typ: &Type, arguments: &mut Vec) { if let HirType::Array(size, _) = typ { if let HirType::NotConstant = **size { unreachable!("println does not support slices. Convert the slice to an array before passing it to println"); } } - let abi_type = typ.as_abi_type(); + let printable_type: PrintableType = typ.into(); let abi_as_string = - serde_json::to_string(&abi_type).expect("ICE: expected Abi type to serialize"); + serde_json::to_string(&printable_type).expect("ICE: expected Abi type to serialize"); arguments.push(ast::Expression::Literal(ast::Literal::Str(abi_as_string))); } diff --git a/crates/noirc_printable_type/Cargo.toml b/crates/noirc_printable_type/Cargo.toml new file mode 100644 index 00000000000..3ef4944ca12 --- /dev/null +++ b/crates/noirc_printable_type/Cargo.toml @@ -0,0 +1,14 @@ +[package] +name = "noirc_printable_type" +version.workspace = true +authors.workspace = true +edition.workspace = true + +# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html + +[dependencies] +acvm.workspace = true +iter-extended.workspace = true +serde.workspace = true + +[dev-dependencies] diff --git a/crates/noirc_printable_type/src/lib.rs b/crates/noirc_printable_type/src/lib.rs new file mode 100644 index 00000000000..fff09954bab --- /dev/null +++ b/crates/noirc_printable_type/src/lib.rs @@ -0,0 +1,199 @@ +use std::{collections::BTreeMap, str}; + +use acvm::FieldElement; +use iter_extended::vecmap; +use serde::{Deserialize, Serialize}; + +#[derive(Clone, Debug, PartialEq, Eq, Serialize, Deserialize)] +#[serde(tag = "kind", rename_all = "lowercase")] +pub enum PrintableType { + Field, + Array { + length: u64, + #[serde(rename = "type")] + typ: Box, + }, + SignedInteger { + width: u32, + }, + UnsignedInteger { + width: u32, + }, + Boolean, + Struct { + name: String, + fields: Vec<(String, PrintableType)>, + }, + String { + length: u64, + }, +} + +impl PrintableType { + /// Returns the number of field elements required to represent the type once encoded. + pub fn field_count(&self) -> u32 { + match self { + Self::Field + | Self::SignedInteger { .. } + | Self::UnsignedInteger { .. } + | Self::Boolean => 1, + Self::Array { length, typ } => typ.field_count() * (*length as u32), + Self::Struct { fields, .. } => { + fields.iter().fold(0, |acc, (_, field_type)| acc + field_type.field_count()) + } + Self::String { length } => *length as u32, + } + } +} + +/// This is what all formats eventually transform into +/// For example, a toml file will parse into TomlTypes +/// and those TomlTypes will be mapped to Value +#[derive(Debug, Clone, Serialize, PartialEq)] +pub enum PrintableValue { + Field(FieldElement), + String(String), + Vec(Vec), + Struct(BTreeMap), +} + +/// In order to display an `PrintableValue` we need an `PrintableType` to accurately +/// convert the value into a human-readable format. +pub struct PrintableValueDisplay<'a> { + value: &'a PrintableValue, + typ: &'a PrintableType, +} + +impl<'a> PrintableValueDisplay<'a> { + #[must_use] + pub fn new(value: &'a PrintableValue, typ: &'a PrintableType) -> Self { + Self { value, typ } + } +} + +impl<'a> std::fmt::Display for PrintableValueDisplay<'a> { + fn fmt(&self, fmt: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { + match (&self.value, &self.typ) { + ( + PrintableValue::Field(f), + PrintableType::Field + // TODO: We should print the sign for these and probably print normal integers instead of field strings + | PrintableType::SignedInteger { .. } + | PrintableType::UnsignedInteger { .. }, + ) => { + write!(fmt, "{}", format_field_string(*f))?; + } + (PrintableValue::Field(f), PrintableType::Boolean) => { + if f.is_one() { + write!(fmt, "true")?; + } else { + write!(fmt, "false")?; + } + } + (PrintableValue::Vec(vector), PrintableType::Array { typ, .. }) => { + write!(fmt, "[")?; + let mut values = vector.iter().peekable(); + while let Some(value) = values.next() { + write!(fmt, "{}", PrintableValueDisplay::new(value, typ))?; + if values.peek().is_some() { + write!(fmt, ", ")?; + } + } + write!(fmt, "]")?; + } + + (PrintableValue::String(s), PrintableType::String { .. }) => { + write!(fmt, r#""{s}""#)?; + } + + (PrintableValue::Struct(map), PrintableType::Struct { name, fields, .. }) => { + write!(fmt, "{name} {{ ")?; + + let mut fields = fields.iter().peekable(); + while let Some((key, field_type)) = fields.next() { + let value = &map[key]; + write!(fmt, "{key}: {}", PrintableValueDisplay::new(value, field_type))?; + if fields.peek().is_some() { + write!(fmt, ", ")?; + } + } + + write!(fmt, " }}")?; + } + + _ => return Err(std::fmt::Error), + }; + Ok(()) + } +} + +/// This trims any leading zeroes. +/// A singular '0' will be prepended as well if the trimmed string has an odd length. +/// A hex string's length needs to be even to decode into bytes, as two digits correspond to +/// one byte. +fn format_field_string(field: FieldElement) -> String { + if field.is_zero() { + return "0x00".to_owned(); + } + let mut trimmed_field = field.to_hex().trim_start_matches('0').to_owned(); + if trimmed_field.len() % 2 != 0 { + trimmed_field = "0".to_owned() + &trimmed_field; + } + "0x".to_owned() + &trimmed_field +} + +// TODO: Figure out a better API for this to avoid exporting the function +/// Assumes that `field_iterator` contains enough [FieldElement] in order to decode the [PrintableType] +pub fn decode_value( + field_iterator: &mut impl Iterator, + typ: &PrintableType, +) -> PrintableValue { + match typ { + PrintableType::Field + | PrintableType::SignedInteger { .. } + | PrintableType::UnsignedInteger { .. } + | PrintableType::Boolean => { + let field_element = field_iterator.next().unwrap(); + + PrintableValue::Field(field_element) + } + PrintableType::Array { length, typ } => { + let length = *length as usize; + let mut array_elements = Vec::with_capacity(length); + for _ in 0..length { + array_elements.push(decode_value(field_iterator, typ)); + } + + PrintableValue::Vec(array_elements) + } + PrintableType::String { length } => { + let field_elements: Vec = field_iterator.take(*length as usize).collect(); + + PrintableValue::String(decode_string_value(&field_elements)) + } + PrintableType::Struct { fields, .. } => { + let mut struct_map = BTreeMap::new(); + + for (field_key, param_type) in fields { + let field_value = decode_value(field_iterator, param_type); + + struct_map.insert(field_key.to_owned(), field_value); + } + + PrintableValue::Struct(struct_map) + } + } +} + +// TODO: Figure out a better API for this to avoid exporting the function +pub fn decode_string_value(field_elements: &[FieldElement]) -> String { + let string_as_slice = vecmap(field_elements, |e| { + let mut field_as_bytes = e.to_be_bytes(); + let char_byte = field_as_bytes.pop().unwrap(); // A character in a string is represented by a u8, thus we just want the last byte of the element + assert!(field_as_bytes.into_iter().all(|b| b == 0)); // Assert that the rest of the field element's bytes are empty + char_byte + }); + + let final_string = str::from_utf8(&string_as_slice).unwrap(); + final_string.to_owned() +}