From 211f34c35116e4bb39a0da4279a7c421bcafb634 Mon Sep 17 00:00:00 2001 From: Denys Zadorozhnyi Date: Wed, 23 Oct 2024 16:15:39 +0300 Subject: [PATCH] feature: add `miden-package` crate with `Package` type to represent compiled Miden program/library. --- Cargo.lock | 53 ++++++ Cargo.toml | 1 + assembly/Cargo.toml | 5 + assembly/src/ast/ident.rs | 35 ++++ core/src/utils/mod.rs | 17 ++ package/Cargo.toml | 31 ++++ package/README.md | 7 + package/src/de.rs | 57 ++++++ package/src/dep/mod.rs | 67 +++++++ package/src/dep/resolver.rs | 69 +++++++ package/src/lib.rs | 34 ++++ package/src/package.rs | 258 +++++++++++++++++++++++++++ package/src/rodata.rs | 67 +++++++ package/src/se.rs | 33 ++++ package/src/tests.rs | 150 ++++++++++++++++ package/tests/data/basic_wallet.masp | Bin 0 -> 145327 bytes processor/Cargo.toml | 1 + 17 files changed, 885 insertions(+) create mode 100644 package/Cargo.toml create mode 100644 package/README.md create mode 100644 package/src/de.rs create mode 100644 package/src/dep/mod.rs create mode 100644 package/src/dep/resolver.rs create mode 100644 package/src/lib.rs create mode 100644 package/src/package.rs create mode 100644 package/src/rodata.rs create mode 100644 package/src/se.rs create mode 100644 package/src/tests.rs create mode 100644 package/tests/data/basic_wallet.masp diff --git a/Cargo.lock b/Cargo.lock index 0ecc546d0..64583c237 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -172,6 +172,16 @@ version = "0.6.3" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "349f9b6a179ed607305526ca489b34ad0a41aed5f7980fa90eb03160b69598fb" +[[package]] +name = "bitcode" +version = "0.6.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ee1bce7608560cd4bf0296a4262d0dbf13e6bcec5ff2105724c8ab88cc7fc784" +dependencies = [ + "bytemuck", + "serde", +] + [[package]] name = "bitflags" version = "1.3.2" @@ -229,6 +239,12 @@ version = "3.16.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "79296716171880943b8470b5f8d03aa55eb2e645a4874bdbb28adb49162e012c" +[[package]] +name = "bytemuck" +version = "1.19.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8334215b81e418a0a7bdb8ef0849474f40bb10c8b71f1c4ed315cff49f32494d" + [[package]] name = "byteorder" version = "1.5.0" @@ -1025,6 +1041,7 @@ dependencies = [ "pretty_assertions", "regex", "rustc_version 0.4.1", + "serde", "smallvec", "tracing", "unicode-width 0.2.0", @@ -1131,6 +1148,18 @@ dependencies = [ "syn", ] +[[package]] +name = "miden-package" +version = "0.11.0" +dependencies = [ + "bitcode", + "miden-assembly", + "miden-core", + "serde", + "serde_bytes", + "serde_repr", +] + [[package]] name = "miden-processor" version = "0.11.0" @@ -1139,6 +1168,7 @@ dependencies = [ "miden-air", "miden-assembly", "miden-core", + "miden-package", "miden-test-utils", "tracing", "winter-fri", @@ -1874,6 +1904,15 @@ dependencies = [ "serde_derive", ] +[[package]] +name = "serde_bytes" +version = "0.11.15" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "387cc504cb06bb40a96c8e04e951fe01854cf6bc921053c954e4a606d9675c6a" +dependencies = [ + "serde", +] + [[package]] name = "serde_derive" version = "1.0.214" @@ -1897,6 +1936,17 @@ dependencies = [ "serde", ] +[[package]] +name = "serde_repr" +version = "0.1.19" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6c64451ba24fc7a6a2d60fc75dd9c83c90903b19028d4eff35e88fc1e86564e9" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] + [[package]] name = "serde_spanned" version = "0.6.8" @@ -1953,6 +2003,9 @@ name = "smallvec" version = "1.13.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3c5e1a9a646d36c3599cd173a41282daf47c44583ad367b8e6837255952e5c67" +dependencies = [ + "serde", +] [[package]] name = "smawk" diff --git a/Cargo.toml b/Cargo.toml index 30d5cdbdf..ac1587911 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -4,6 +4,7 @@ members = [ "assembly", "core", "miden", + "package", "processor", "prover", "stdlib", diff --git a/assembly/Cargo.toml b/assembly/Cargo.toml index e72f780fb..712223bed 100644 --- a/assembly/Cargo.toml +++ b/assembly/Cargo.toml @@ -21,6 +21,10 @@ doctest = false default = ["std"] std = ["aho-corasick/std", "miette/fancy", "miette/std", "thiserror/std", "vm-core/std"] testing = ["dep:regex"] +serde = [ + "dep:serde", + "smallvec/serde", +] [dependencies] aho-corasick = { version = "1.1", default-features = false } @@ -37,6 +41,7 @@ unicode-width = { version = "0.2", features = ["no_std"] } vm-core = { package = "miden-core", path = "../core", version = "0.11", default-features = false, features = [ "diagnostics", ] } +serde = { version = "1.0.208", features = ["alloc", "rc", "serde_derive"], optional = true } [dev-dependencies] pretty_assertions = "1.4" diff --git a/assembly/src/ast/ident.rs b/assembly/src/ast/ident.rs index ea2d13a23..10e999ef4 100644 --- a/assembly/src/ast/ident.rs +++ b/assembly/src/ast/ident.rs @@ -183,3 +183,38 @@ impl FromStr for Ident { Ok(Self { span: SourceSpan::default(), name }) } } + +#[cfg(feature = "serde")] +impl serde::Serialize for Ident { + fn serialize(&self, serializer: S) -> Result + where + S: serde::Serializer, + { + self.as_str().serialize(serializer) + } +} + +#[cfg(feature = "serde")] +impl<'de> serde::Deserialize<'de> for Ident { + fn deserialize(deserializer: D) -> Result + where + D: serde::Deserializer<'de>, + { + struct IdentVisitor; + impl serde::de::Visitor<'_> for IdentVisitor { + type Value = Ident; + + fn expecting(&self, formatter: &mut fmt::Formatter) -> fmt::Result { + formatter.write_str("ident") + } + + fn visit_str(self, v: &str) -> Result + where + E: serde::de::Error, + { + Ok(Ident::new_unchecked(Span::unknown(Arc::from(v)))) + } + } + deserializer.deserialize_str(IdentVisitor) + } +} diff --git a/core/src/utils/mod.rs b/core/src/utils/mod.rs index f610ed7f8..3cbf5fb70 100644 --- a/core/src/utils/mod.rs +++ b/core/src/utils/mod.rs @@ -41,6 +41,23 @@ impl ToElements for Vec { } } +// TODO(denysz): add roundtrip proptest +impl ToElements for &[u8] { + fn to_elements(&self) -> Vec { + self.chunks(4) + .map(|chunk| { + if chunk.len() < 4 { + let mut bytes = [0; 4]; + bytes[..chunk.len()].copy_from_slice(chunk); + Felt::new(u32::from_le_bytes(bytes) as u64) + } else { + Felt::new(u32::from_le_bytes(chunk.try_into().unwrap()) as u64) + } + }) + .collect() + } +} + // INTO BYTES // ================================================================================================ diff --git a/package/Cargo.toml b/package/Cargo.toml new file mode 100644 index 000000000..a9f524b1d --- /dev/null +++ b/package/Cargo.toml @@ -0,0 +1,31 @@ +[package] +name = "miden-package" +version = "0.11.0" +description = "Miden VM package" +documentation = "https://docs.rs/miden-package/0.10.5" +readme = "README.md" +categories = ["compilers", "no-std"] +keywords = ["package", "language", "miden"] +license.workspace = true +authors.workspace = true +homepage.workspace = true +repository.workspace = true +rust-version.workspace = true +edition.workspace = true + +[lib] +bench = false +doctest = false + +[dependencies] +vm-core = { package = "miden-core", path = "../core", version = "0.11", default-features = false } +# processor = { package = "miden-processor", path = "../processor", version = "0.11", default-features = false } +assembly = { package = "miden-assembly", path = "../assembly", version = "0.11", default-features = false, features = [ + "serde", +] } +serde = { version = "1.0.208", features = ["alloc", "rc", "serde_derive"] } +serde_bytes = "0.11.15" +serde_repr = "0.1.19" +bitcode = { version = "0.6.3", default-features = false, features = ["serde"] } + +[dev-dependencies] diff --git a/package/README.md b/package/README.md new file mode 100644 index 000000000..747fc473c --- /dev/null +++ b/package/README.md @@ -0,0 +1,7 @@ +## Overview + +TBD + +## Binary Format + +TBD \ No newline at end of file diff --git a/package/src/de.rs b/package/src/de.rs new file mode 100644 index 000000000..faa7e913c --- /dev/null +++ b/package/src/de.rs @@ -0,0 +1,57 @@ +use alloc::{fmt, sync::Arc}; + +use assembly::Library; +use vm_core::{utils::Deserializable, Program}; + +use crate::{package::MastArtifact, Digest}; + +/// Deserialize a [Digest] from a byte array +pub fn deserialize_digest<'de, D>(deserializer: D) -> Result +where + D: serde::Deserializer<'de>, +{ + const DIGEST_BYTES: usize = 32; + + let bytes: [u8; DIGEST_BYTES] = serde_bytes::deserialize(deserializer)?; + + Digest::try_from(bytes).map_err(serde::de::Error::custom) +} + +/// Deserialize a [MastArtifact] from a byte array +pub fn deserialize_mast<'de, D>(deserializer: D) -> Result +where + D: serde::Deserializer<'de>, +{ + struct MastArtifactVisitor; + + impl serde::de::Visitor<'_> for MastArtifactVisitor { + type Value = MastArtifact; + + fn expecting(&self, formatter: &mut fmt::Formatter) -> fmt::Result { + formatter.write_str("mast artifact") + } + + fn visit_bytes(self, v: &[u8]) -> Result + where + E: serde::de::Error, + { + if let Some(bytes) = v.strip_prefix(b"PRG\0") { + Program::read_from_bytes(bytes) + .map(Arc::new) + .map(MastArtifact::Executable) + .map_err(serde::de::Error::custom) + } else if let Some(bytes) = v.strip_prefix(b"LIB\0") { + Library::read_from_bytes(bytes) + .map(Arc::new) + .map(MastArtifact::Library) + .map_err(serde::de::Error::custom) + } else { + Err(serde::de::Error::invalid_value( + serde::de::Unexpected::Bytes(v.get(0..4).unwrap_or(v)), + &"expected valid mast artifact type tag", + )) + } + } + } + deserializer.deserialize_bytes(MastArtifactVisitor) +} diff --git a/package/src/dep/mod.rs b/package/src/dep/mod.rs new file mode 100644 index 000000000..6daa61358 --- /dev/null +++ b/package/src/dep/mod.rs @@ -0,0 +1,67 @@ +use alloc::string::String; + +use serde::{Deserialize, Serialize}; + +use super::{de, se}; +use crate::Digest; + +pub(crate) mod resolver; + +/// A system library identifier +#[derive( + Debug, Copy, Clone, PartialEq, Eq, serde_repr::Serialize_repr, serde_repr::Deserialize_repr, +)] +#[repr(u8)] +pub enum SystemLibraryId { + /// The standard library + Stdlib, + /// The base library + Miden, +} + +impl core::str::FromStr for SystemLibraryId { + type Err = (); + fn from_str(s: &str) -> Result { + match s { + // Compiler uses "std" and "base" to identify the standard and base libraries + // respectively. We also accept "stdlib" and "miden" as aliases for these libraries. + "std" | "stdlib" => Ok(Self::Stdlib), + "base" | "miden" => Ok(Self::Miden), + _ => Err(()), + } + } +} + +/// The name of a dependency +#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)] +pub enum DependencyName { + /// The dependency is a system library + System(SystemLibraryId), + /// The dependency is a user library with the given name + User(String), +} + +impl From for DependencyName { + fn from(s: String) -> Self { + if let Ok(id) = s.parse() { + DependencyName::System(id) + } else { + DependencyName::User(s) + } + } +} + +/// A package dependency +#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)] +pub struct Dependency { + /// The name of the dependency. + /// Serves as a human-readable identifier for the dependency and a search hint for the resolver + pub name: DependencyName, + /// The digest of the dependency. + /// Serves as an ultimate source of truth for identifying the dependency. + #[serde( + serialize_with = "se::serialize_digest", + deserialize_with = "de::deserialize_digest" + )] + pub digest: Digest, +} diff --git a/package/src/dep/resolver.rs b/package/src/dep/resolver.rs new file mode 100644 index 000000000..9f92b7306 --- /dev/null +++ b/package/src/dep/resolver.rs @@ -0,0 +1,69 @@ +use alloc::{collections::BTreeMap, sync::Arc}; + +use assembly::Library; + +use super::Dependency; +use crate::{Digest, Package}; + +// DEPENDENCY RESOLUTION +// ================================================================================================ + +#[derive(Debug, Clone)] +pub enum DependencyResolution { + Local(LocalResolution), + // Registry(...), +} + +impl From> for DependencyResolution { + fn from(library: Arc) -> Self { + Self::Local(LocalResolution::Library(library)) + } +} + +impl From> for DependencyResolution { + fn from(package: Arc) -> Self { + Self::Local(LocalResolution::Package(package)) + } +} + +#[derive(Debug, Clone)] +pub enum LocalResolution { + Library(Arc), + Package(Arc), +} + +impl From> for LocalResolution { + fn from(library: Arc) -> Self { + Self::Library(library) + } +} + +impl From> for LocalResolution { + fn from(package: Arc) -> Self { + Self::Package(package) + } +} + +// RESOLVER +// ================================================================================================ + +pub trait DependencyResolver { + fn resolve(&self, dependency: &Dependency) -> Option; +} + +#[derive(Debug, Default)] +pub struct MemDependencyResolverByDigest { + resolved: BTreeMap, +} + +impl MemDependencyResolverByDigest { + pub fn add(&mut self, digest: Digest, resolution: DependencyResolution) { + self.resolved.insert(digest, resolution); + } +} + +impl DependencyResolver for MemDependencyResolverByDigest { + fn resolve(&self, dependency: &Dependency) -> Option { + self.resolved.get(&dependency.digest).cloned() + } +} diff --git a/package/src/lib.rs b/package/src/lib.rs new file mode 100644 index 000000000..376d6fcb1 --- /dev/null +++ b/package/src/lib.rs @@ -0,0 +1,34 @@ +//! The [Package] containing a [vm_core::Program] or [assembly::Library], rodata segments, +//! and a manifest(exports and dependencies). +//! +//! Serves as the unit of deployment and distribution in the Miden ecosystem. +//! Contains everything needed to execute a program or link a library. + +#![no_std] + +extern crate alloc; + +mod de; +mod dep; +mod package; +mod rodata; +mod se; + +#[cfg(test)] +extern crate std; +#[cfg(test)] +mod tests; + +pub use self::{ + dep::{ + resolver::{ + DependencyResolution, DependencyResolver, LocalResolution, + MemDependencyResolverByDigest, + }, + Dependency, DependencyName, SystemLibraryId, + }, + package::{MastArtifact, Package, PackageExport, PackageManifest}, + rodata::{PtrDesc, Rodata}, +}; + +type Digest = vm_core::chiplets::hasher::Digest; diff --git a/package/src/package.rs b/package/src/package.rs new file mode 100644 index 000000000..efc94a7f2 --- /dev/null +++ b/package/src/package.rs @@ -0,0 +1,258 @@ +use alloc::{ + collections::{BTreeMap, BTreeSet}, + format, + string::String, + sync::Arc, + vec::Vec, +}; +use core::fmt; + +use assembly::{ast::QualifiedProcedureName, Library, Report}; +use serde::{Deserialize, Serialize}; +use vm_core::{mast::MastForest, utils::DisplayHex, Felt, Program}; + +use super::{de, se}; +use crate::{Dependency, Digest, Rodata}; + +// MAST ARTIFACT +// ================================================================================================ + +/// The artifact produced by lowering a [Program] to a Merkelized Abstract Syntax Tree +/// +/// This type is used in compilation pipelines to abstract over the type of output requested. +#[derive(Debug, Clone)] +pub enum MastArtifact { + /// A MAST artifact which can be executed by the VM directly + Executable(Arc), + /// A MAST artifact which can be used as a dependency by a [Program] + Library(Arc), +} + +impl MastArtifact { + /// Get the underlying [Program] for this artifact, or panic if this is a [Library] + pub fn unwrap_program(self) -> Arc { + match self { + Self::Executable(prog) => prog, + Self::Library(_) => panic!("attempted to unwrap 'mast' library as program"), + } + } + + /// Get the underlying [Library] for this artifact, or panic if this is a [Program] + pub fn unwrap_library(self) -> Arc { + match self { + Self::Executable(_) => panic!("attempted to unwrap 'mast' program as library"), + Self::Library(lib) => lib, + } + } + + /// Get the content digest associated with this artifact + pub fn digest(&self) -> Digest { + match self { + Self::Executable(ref prog) => prog.hash(), + Self::Library(ref lib) => *lib.digest(), + } + } + + /// Get the underlying [MastForest] for this artifact + pub fn mast_forest(&self) -> &MastForest { + match self { + Self::Executable(ref prog) => prog.mast_forest(), + Self::Library(ref lib) => lib.mast_forest(), + } + } +} + +// PACKAGE MANIFEST +// ================================================================================================ + +/// The manifest of a package, containing the set of package dependencies(libraries or packages) and +/// exported procedures and their signatures, if known. +#[derive(Debug, Default, Clone, PartialEq, Eq, Serialize, Deserialize)] +#[serde(default)] +pub struct PackageManifest { + /// The set of exports in this package. + pub exports: BTreeSet, + /// The libraries(packages) linked against by this package, which must be provided when + /// executing the program. + pub dependencies: Vec, +} + +/// A procedure exported by a package, along with its digest and +/// signature(will be added after MASM type attributes are implemented). +#[derive(Clone, PartialEq, Eq, Serialize, Deserialize, Ord, PartialOrd)] +pub struct PackageExport { + /// The fully-qualified name of the procedure exported by this package + pub name: String, + /// The digest of the procedure exported by this package + #[serde( + serialize_with = "se::serialize_digest", + deserialize_with = "de::deserialize_digest" + )] + pub digest: Digest, + // Signature will be added in the future when the type signatures are available in the Assembly + // #[serde(default)] + // pub signature: Option, +} + +impl fmt::Debug for PackageExport { + fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { + f.debug_struct("PackageExport") + .field("name", &format_args!("{}", self.name)) + .field("digest", &format_args!("{}", DisplayHex::new(&self.digest.as_bytes()))) + // .field("signature", &self.signature) + .finish() + } +} + +// PACKAGE +// ================================================================================================ + +/// A package containing a [Program]/[Library], rodata segments, +/// and a manifest(exports and dependencies). +#[derive(Debug, Clone, Serialize, Deserialize)] +pub struct Package { + /// Name of the package + pub name: String, + /// Content digest of the package + #[serde( + serialize_with = "se::serialize_digest", + deserialize_with = "de::deserialize_digest" + )] + pub digest: Digest, + /// The MAST artifact ([Program] or [Library]) of the package + #[serde(serialize_with = "se::serialize_mast", deserialize_with = "de::deserialize_mast")] + pub mast: MastArtifact, + /// The rodata segments required to be loaded in the advice provider before executing the code + /// in this package + pub rodata: Vec, + /// The package manifest, containing the set of exported procedures and their signatures, + /// if known. + pub manifest: PackageManifest, +} + +impl Package { + const MAGIC: &'static [u8] = b"MASP\0"; + const FORMAT_VERSION: &'static [u8] = b"1.0\0"; + + /// Parses a package from the provided bytes + pub fn read_from_bytes(bytes: B) -> Result + where + B: AsRef<[u8]>, + { + use alloc::borrow::Cow; + + let bytes = bytes.as_ref(); + + let bytes = bytes + .strip_prefix(Self::MAGIC) + .ok_or_else(|| Report::msg("invalid package: missing header"))?; + let bytes = bytes.strip_prefix(Self::FORMAT_VERSION).ok_or_else(|| { + Report::msg(format!( + "invalid package: incorrect version, expected '1.0', got '{}'", + bytes.get(0..4).map(String::from_utf8_lossy).unwrap_or(Cow::Borrowed("")), + )) + })?; + + bitcode::deserialize(bytes).map_err(Report::msg) + } + + /// Serializes the package into a byte array + pub fn write_to_bytes(&self) -> Result, Report> { + let mut bytes = Vec::new(); + bytes.extend_from_slice(Self::MAGIC); + bytes.extend_from_slice(Self::FORMAT_VERSION); + let mut data = bitcode::serialize(self).map_err(Report::msg)?; + bytes.append(&mut data); + Ok(bytes) + } + + /// Checks if the package's MAST artifact is a [Program] + pub fn is_program(&self) -> bool { + matches!(self.mast, MastArtifact::Executable(_)) + } + + /// Checks if the package's MAST artifact is a [Library] + pub fn is_library(&self) -> bool { + matches!(self.mast, MastArtifact::Library(_)) + } + + /// Unwraps the package's MAST artifact as a [Program] or panics if it is a [Library] + pub fn unwrap_program(&self) -> Arc { + match self.mast { + MastArtifact::Executable(ref prog) => Arc::clone(prog), + _ => panic!("expected package to contain a program, but got a library"), + } + } + + /// Unwraps the package's MAST artifact as a [Library] or panics if it is a [Program] + pub fn unwrap_library(&self) -> Arc { + match self.mast { + MastArtifact::Library(ref lib) => Arc::clone(lib), + _ => panic!("expected package to contain a library, but got an executable"), + } + } + + /// Creates a new package with [Program] from this [Library] package and the given + /// entrypoint (should be a procedure in the library). + pub fn make_executable(&self, entrypoint: &QualifiedProcedureName) -> Result { + let MastArtifact::Library(ref library) = self.mast else { + return Err(Report::msg("expected library but got an executable")); + }; + + let module = library + .module_infos() + .find(|info| info.path() == &entrypoint.module) + .ok_or_else(|| { + Report::msg(format!( + "invalid entrypoint: library does not contain a module named '{}'", + entrypoint.module + )) + })?; + if let Some(digest) = module.get_procedure_digest_by_name(&entrypoint.name) { + let node_id = library.mast_forest().find_procedure_root(digest).ok_or_else(|| { + Report::msg( + "invalid entrypoint: malformed library - procedure exported, but digest has \ + no node in the forest", + ) + })?; + + let exports = BTreeSet::from_iter(self.manifest.exports.iter().find_map(|export| { + if export.digest == digest { + Some(export.clone()) + } else { + None + } + })); + + Ok(Self { + name: self.name.clone(), + digest, + mast: MastArtifact::Executable(Arc::new(Program::new( + library.mast_forest().clone(), + node_id, + ))), + rodata: self.rodata.clone(), + manifest: PackageManifest { + exports, + dependencies: self.manifest.dependencies.clone(), + }, + }) + } else { + Err(Report::msg(format!( + "invalid entrypoint: library does not export '{}'", + entrypoint + ))) + } + } + + /// The advice map that is expected to be loaded into the advice provider so that the + /// program's compiler-generated prologue will load the rodata from the advice provider into + /// memory before the program is executed + pub fn advice_map(&self) -> BTreeMap> { + let mut advice_inputs = BTreeMap::default(); + for rodata in self.rodata.iter() { + advice_inputs.extend([(rodata.digest, rodata.to_elements())]); + } + advice_inputs + } +} diff --git a/package/src/rodata.rs b/package/src/rodata.rs new file mode 100644 index 000000000..fdb5d3d42 --- /dev/null +++ b/package/src/rodata.rs @@ -0,0 +1,67 @@ +use alloc::vec::Vec; + +use serde::{Deserialize, Serialize}; +use vm_core::{utils::ToElements, Felt, FieldElement}; + +use super::{de, se}; +use crate::Digest; + +/// This represents a descriptor for a pointer referencing data in Miden's linear memory. +#[derive(Debug, Copy, Clone, PartialEq, Eq, Serialize, Deserialize)] +pub struct PtrDesc { + /// This is the address of the word containing the first byte of data + pub waddr: u32, + /// This is the element index of the word referenced by `waddr` containing the first byte of + /// data + /// + /// Each element is assumed to be a 32-bit value/chunk + pub index: u8, + /// This is the byte offset into the 32-bit chunk referenced by `index` + /// + /// This offset is where the data referenced by the pointer actually starts. + pub offset: u8, +} + +/// Represents a read-only data segment, combined with its content digest +#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)] +pub struct Rodata { + /// The content digest computed for `data` + #[serde( + serialize_with = "se::serialize_digest", + deserialize_with = "de::deserialize_digest" + )] + pub digest: Digest, + /// The address at which the data for this segment begins + pub start: PtrDesc, + /// The raw binary data for this segment + pub data: Vec, +} + +impl Rodata { + /// Returns the size of the data in bytes + pub fn size_in_bytes(&self) -> usize { + self.data.len() + } + + /// Returns the size of the data in felts + pub fn size_in_felts(&self) -> usize { + self.data.len().next_multiple_of(4) / 4 + } + + /// Returns the size of the data in VM words + pub fn size_in_words(&self) -> usize { + self.size_in_felts().next_multiple_of(4) / 4 + } + + /// Converts this rodata object to its equivalent representation in felts + /// + /// The resulting felts will be in padded out to the nearest number of words, i.e. if the data + /// only takes up 3 felts worth of bytes, then the resulting `Vec` will contain 4 felts, so that + /// the total size is a valid number of words. + pub fn to_elements(&self) -> Vec { + let mut felts = self.data.as_slice().to_elements(); + let padding = (self.size_in_words() * 4).abs_diff(felts.len()); + felts.resize(felts.len() + padding, Felt::ZERO); + felts + } +} diff --git a/package/src/se.rs b/package/src/se.rs new file mode 100644 index 000000000..91dba3de7 --- /dev/null +++ b/package/src/se.rs @@ -0,0 +1,33 @@ +use alloc::vec; + +use vm_core::utils::Serializable; + +use crate::{package::MastArtifact, Digest}; + +/// Serialize a [Digest] into a byte array +pub fn serialize_digest(digest: &Digest, serializer: S) -> Result +where + S: serde::Serializer, +{ + serde_bytes::serialize(&digest.as_bytes(), serializer) +} + +/// Serialize a [MastArtifact] into a byte array +pub fn serialize_mast(mast: &MastArtifact, serializer: S) -> Result +where + S: serde::Serializer, +{ + let mut buffer = vec![]; + match mast { + MastArtifact::Executable(program) => { + buffer.extend(b"PRG\0"); + program.write_into(&mut buffer); + }, + MastArtifact::Library(library) => { + buffer.extend(b"LIB\0"); + library.write_into(&mut buffer); + }, + } + + serde_bytes::serialize(&buffer, serializer) +} diff --git a/package/src/tests.rs b/package/src/tests.rs new file mode 100644 index 000000000..325e63bdb --- /dev/null +++ b/package/src/tests.rs @@ -0,0 +1,150 @@ +use std::{dbg, sync::Arc}; + +use assembly::Report; + +use super::*; + +#[ignore = "not implemented"] +#[test] +fn packaging_serialization() { + let package = example_package().unwrap(); + bitcode::serialize(package.as_ref()).unwrap(); + todo!("do it via roundtrip in proptest") +} + +#[ignore = "not implemented"] +#[test] +fn packaging_deserialization() -> Result<(), Report> { + let _expected = example_package()?; + todo!("do it via roundtrip in proptest") + + // let mut bytes = vec![]; + // expected + // .write_to(&mut bytes, midenc_session::OutputMode::Binary, &context.session) + // .into_diagnostic()?; + + // let package = Package::read_from_bytes(bytes)?; + + // assert_eq!(package.name, expected.name); + // assert_eq!(package.digest, expected.digest); + // assert_eq!(package.rodata, expected.rodata); + // assert_eq!(package.manifest, expected.manifest); + // assert!(package.is_program()); + + // // Verify rodata serialization + // assert!(!package.rodata.is_empty()); + // let expected_rodata_offset = PtrDesc::from_ptr(65536 * 4); + // let foo_data = package + // .rodata + // .iter() + // .find(|rodata| rodata.start == expected_rodata_offset) + // .unwrap(); + // let foo_bytes = foo_data.data.as_slice(); + + // let foo_ty = StructType::new([Type::U8, Type::U32, Type::U64]); + // let offset_u8 = foo_ty.get(0).offset as usize; + // let offset_u32 = foo_ty.get(1).offset as usize; + // let offset_u64 = foo_ty.get(2).offset as usize; + // assert_eq!(foo_bytes[offset_u8], 1); + // assert_eq!( + // u32::from_be_bytes([ + // foo_bytes[offset_u32], + // foo_bytes[offset_u32 + 1], + // foo_bytes[offset_u32 + 2], + // foo_bytes[offset_u32 + 3] + // ]), + // 2 + // ); + // assert_eq!( + // u32::from_be_bytes([ + // foo_bytes[offset_u64], + // foo_bytes[offset_u64 + 1], + // foo_bytes[offset_u64 + 2], + // foo_bytes[offset_u64 + 3] + // ]), + // 0 + // ); + // assert_eq!( + // u32::from_be_bytes([ + // foo_bytes[offset_u64 + 4], + // foo_bytes[offset_u64 + 5], + // foo_bytes[offset_u64 + 6], + // foo_bytes[offset_u64 + 7] + // ]), + // 3 + // ); + + // // Verify the MAST + // let expected = expected.unwrap_program(); + // let program = package.unwrap_program(); + // assert_eq!(program.hash(), expected.hash()); + // assert_eq!(program.mast_forest(), expected.mast_forest()); + + // Ok(()) +} + +fn example_package() -> Result, Report> { + todo!() + // use midenc_hir::ProgramBuilder; + + // // Build a simple program + // let mut builder = ProgramBuilder::new(&context.session.diagnostics); + + // // Build test module with fib function + // let mut mb = builder.module("test"); + // midenc_hir::testing::fib1(mb.as_mut(), context); + + // // Ensure we have an example data segment or two to work with + // let foo_ty = StructType::new([Type::U8, Type::U32, Type::U64]); + // // Initialize the struct with some data + // let offset_u8 = foo_ty.get(0).offset as usize; + // let offset_u32 = foo_ty.get(1).offset as usize; + // let offset_u64 = foo_ty.get(2).offset as usize; + // let foo_ty = Type::Struct(foo_ty); + // let foo_size = foo_ty.size_in_bytes(); + // let mut data = vec![0u8; foo_size]; + // unsafe { + // let data_ptr_range = data.as_mut_ptr_range(); + // core::ptr::write(data_ptr_range.start.byte_add(offset_u8), 1u8); + // core::ptr::write(data_ptr_range.start.byte_add(offset_u32).cast(), + // 2u32.to_be_bytes()); core::ptr::write(data_ptr_range.start.byte_add(offset_u64). + // cast(), 0u32.to_be_bytes()); // hi bits core::ptr::write(data_ptr_range.start. + // byte_add(offset_u64 + 4).cast(), 3u32.to_be_bytes()); // lo bits + // } + // mb.declare_data_segment(65536 * 4, foo_size as u32, data, true)?; + + // mb.build().expect("unexpected error constructing test module"); + + // // Link the program + // let mut program = builder + // .with_entrypoint("test::fib".parse().unwrap()) + // .link() + // .expect("failed to link program"); + + // program.add_library(StdLibrary::default().into()); + + // // Compile the program + // let mut compiler = crate::MasmCompiler::new(&context.session); + // let program = compiler.compile(program).expect("compilation failed").unwrap_executable(); + + // // Assemble the program + // let masm_artifact = MasmArtifact::Executable(program); + // let mast_artifact = masm_artifact.assemble(&context.session)?; + + // // Package the program + // Ok(Arc::new(Package::new(mast_artifact, &masm_artifact, &context.session))) +} + +#[ignore = "update the binary atfer the changes in the Miden package format are settled"] +#[test] +fn basic_wallet_package_deserialization() { + // Test for the https://github.com/0xPolygonMiden/compiler/issues/347 + // The included Miden package file is built at + // https://github.com/0xPolygonMiden/compiler/blob/6cd29e17b34c5abef7f6328c33af06f8bf203344/tests/integration/src/rust_masm_tests/rust_sdk.rs#L48-L63 + + let bytes = include_bytes!("../tests/data/basic_wallet.masp"); + + let package = Package::read_from_bytes(bytes).unwrap(); + dbg!(&package.manifest); + assert_eq!(package.name, "basic_wallet"); +} diff --git a/package/tests/data/basic_wallet.masp b/package/tests/data/basic_wallet.masp new file mode 100644 index 0000000000000000000000000000000000000000..84f2b7555e94abe2a92789be84ac45e67f1512ac GIT binary patch literal 145327 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

T4d(yO?sp$3QN4 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 z�X(W&PdYBR{Vqe*@#sn~>V>Wn$jic~rSqFy1Fv=L?<-1fTl_zXt?17PU_G8_OaF zs#pGoDe~5DF@ql#j8tfN>$jx7_C;^~9-|T?8{YcW)Ymkhdk-TU+LJQa$hd|{K2=}Q zdTs%DkGPWI&rp*ZYP!7FXs@=|bMH0eHO3b2r+v15D*822GCs%io)ZF-OI3jg0&$8 zUtM#v_A42r|C*8eDXB2@+HcP*AEGQls({W08E+YQWaElGq}i}Nxo?m5(|%8V732{q z8IKC)9|UWiU_T~!|0o1dFOcb3&rQRuv1^aZaQE|nk&>B*s$)c|Uv+DkU29Ez_UOE# zq0tRX65fo*@?&(;8Ydeja!N|}(}H(rnU@a>=F>$8=qdzWBZM#_j3o*CP3p;8AK9-k zfk=kgFW91uT5lO^Z6eq?JkUxFjV95aW7qn~Fuku}bO`3Ag4Iv(WCG{*31+Q7V*&*G z6@s@q%HZRTa#_|AEx)lGC`0u@f-zWdSZp2*E3GoNYOiEuh+qyC>|sKqxipi3)^J8Q z7id7fc|y-BIWJ059$z|2yGOLk$KuhUrTe+6Q4LIzEArkZ!H4V|jBpTYAfX0dBZM(R zy-x_|Q_hiAp(UTRjpn1DtvrM_oS zT4|kaU2ffK-Dj<{x@oXEx`d5MQ7Ma4YSOl*<)rs(TXXg1t8E$0+xKd}vi-^q+dFu5jOiGfnV#7%vobTZ z^S-PzS^iyOyJUAM=@Qj7v+Lll0!QpSnK2eKY$N_nX?U zrr*Z?x*Kf+Vg}aU7MFCOkYqI$%c5iuhd zk8D0_;V7??xh1Pha!S{ix<=QG4jPj_re@5UF+pXM%MO;Mm(MQWQ|_vmQL((jf9#C0 z$Hz{sJX|@TYIRlSEpu-o}~WqHI`6`LKfW^z>0hqtesMibjvjD=5k> z8d)@|sHDPNQaajEQ6@uTU5W8i;`){sm6gX8R_7I#mX%fyEg3f~uDq-uZq%^CqLMbl z%Hj%23yX#nmBfw8D<2hiZsKt=?O2A{3&)O*jrSZqwtQ%8f_Y>`k!N1HOtQjzRB5G* zajEexW6>CUL19_xXa*OS#3s3);Bxtew3@s`cQB-IhomW_;mMYO9E0H_0R34w5L~?Y|Eky;f z7e#hO`R5q-5T<*}KywSO%@`em6aE?uIxdVwUDl2Q}Oi6Mk zrX(gMCWg5aOG!vdbEPIFr8(WTa*M`XRIyv5nXtqct(1GYFUlsN(KFq7lvi10F7YaZ z1b6hQc3IAm6;@U&L1l2M=+rZup5;|}qhp=k?iFx1aI(34Y`%My5?yJDX~`+Bv?NcN zX8F*O{3$7|?+|KeUP+;vR?69EQ`}X{Sc=jzmhS9a5-7W(87$Ejk1VYkRx%_vudq<& z?+&cmr@j%@6qD>~q-t5U3Tcf(kou)%<8sS~RTo9qKesB@MZ2UXBqye}QS0BhB^ymh zwL^K8Q`NsjqeNb`*bFK!xa#GTA zBd+Zep|;H2^1>0hGODy7Ha2tYsL>rMMvQgEN92{OKLyh9V`IJDlSNK=I7_R3X`*FhanXgwP&Kegl^-IaidE|7qBZS8sn~W zF1#=qTUC}fdNl1ME9@hWN0k=FCWf7DF4enTN%8I^5-+GPO~=%!?@Z>Ig~KXi6B$Io zncQ9_ou1w(y|XP5qPnCjIW;jMDKROzvzP7E#yfJ zt*JIFHdNW=CS7t})tXMbXw-llXJXZ`iroBRC57@rl3P~cbQR>~535f``kyl~J|)qa zlJ6`?PAg6-&dX})4(NgeTbw^k&1v9;b2@*bx#!O@KG&6+8$Y%_^nZSFE?V33XVgti z_52~1nDx9%Ea>P`=^ScXPFG{wxVQAU^B;6@#>U6Sx2Py8uV_X z1Xr3XEjiwm;!IGxd$o&j?_b}d1kup&Zs zpw9TSx#m?A$ztcFrpjK&nNa9Ta^*YI5`Jf~FX-TBCz5i03&{S3L$|zs;=o zys+wm9q6dBBSTb&Y-rXt=TFq#tmU~wDjcd7Egbc4$R$-p>JFtKtuQ`6IWH+GDcP0m zOilcqP(tcojboG3k`r8KcbQISqSNV8?VRd<(MWOsNq(w+i~b+&QJ!i)=X$s$Wq7FUfr(Iz@C;9U{4Ws2qKdICr!u-J-ym5Fejdz#&nJD=Doo z>!R1pG%sEKiu_;fk?k*9Upew9AC+4mEty*&-M!p-en(QhQ@!X)2UH7xuB#rqXhL$} zl5+lbb~eA**h}Qsc(W@VT^<|DQE6;!XOWN-c__IixtwWZJH*ESgQKj%sww{{BiB%QKQL0> zp_Y}EmPrHU#*`$Jpz%ti~_=zt@!Y;1{ptI91cQHxht zSdgDuoKTRKl#p7Omz0tfp!P!Qa52gKh9vjmEpozzGy12GCEUlYX~`EH{JD}7)eg{= zIIcdKKePDmX~et#q`K$iihsX;k}3c0?Q(C!oc=@m9`~z8vU(4is!a3b%!o7I$9*D1 zZjn{Xl%6g}a_Q*>W#dLyl%}T-%_|>TB&TZfN9K(vN^n2Ubyk!*)dW-Bjgfjb!PLYh z_3QOVy6Yd=&8XR=xyD>(y8m#~RL@UJR(}#xfBy}doMv&qL36KFqPu$}xXzvgNITc( zoi2BH&ie3N_g%Pv|H_tM|CTiMy!T{I*PK7SHF0|Byu&QBY`KjO<<6Z>X?zUcaO&X6 z3ztFOAUYG>Z}(+HQEvIT@>JeM$y@94l)Qq1(y=A-UaCw^XH*vDa`;;zJG!)@w3LL@ z1bK-{C@gfQ7Gz!aO>V<%>-XU9JxKc5DW|8)inu46b++d=Is`bkRuBF0MR!%);&c|s z0YgPmuACQQ?_3a{l9u3<6Q)jCdRJmXmiq`V&HX3!whJfz$M$sPMJ0u8R4QltSE{>{ zCf4^UcP~sxu78~1?w!er3+nsY|3Lcn6OZ?HpEYxx{p+0h4`=d6a$0g>GJmAUq%}0F z1n(+HEh}qJoNHrSh(+sJx(Tm~1dZa^-zxX<0?NtZ;mON^*W$azS2V zft-p>Nz1xmbF4qWDwlU%MTPCo@9%0;xwGjZ8)`; z@;eJ4wHq&h47~gT$jIMc0C_8GEG?6(6%`ksMwKzYvnVpQ@uJAU%P-3Bj+7IR94X45 z^LFuaRIW~tl;@7DIPaRs`Gbqp$x`Pf$IAJ$#%EhbmsYvlJD>z-xXu!mi+_>KDY`HSDEUtc_J z>216o$tnKi9%>)acsxiKRvT3)b!V}c_qqAxM5sBy(1m#EliBV&r^HnQY|%dMMSvATTM zkqzwf&nx?fJzsjqUcgmDJdbDFQ(Qw@whsT7#Cn>Ff)vyj;hxcJG9D&ooN4_0=iqz2(xNqIj39g4+7zMY&bXWk7!y0%M zYGE6^3wz)@I0<}JzQZ3Pzy%#32YSN|FbGD#IG6@^!5o+mkHKo#1l!?L*bDpN5S#$B zLGXhX;Dq+j1Ngov$1tb@zWm!U50=0)@G|iGX%4>^o)x zcc44C3EsiY>JIKDcW}?Q<8$DCX$N;MJFIf^lOq5kAptrPM9=ngl+5Eu(nU=BPCE8rz~6F!8m z;TUM;yW}0-5CW|r4Z1>KD1b3A0cOD>coNpZ>+nA8g(Gkp_>y`D+q0u3Bta**9&%w6 z+zQiS0Xz6C$xtiFc5}872E;y;C^@rmclC72yek|H~`1t zSJ23p+&lQr6o&)CAr9I>ceoLTz*v|9bKqfE0WZOu@F9E+$3QDz+VAj&5NHKy&=vYZ z0gQnOFbfvJlduk6hxcJG9D&oouK_p$p(P|iC%7JRVHDg7(_sNT4r}05cn3a%LvRB4 zg%gJ#v;ZfxhaNBxhCvnF0rOx9JOeMoHuxC6fuBGxxAi%EAPm|-Tj&P;p$N+17Pt*2 z!#%JVo`&`C2JC`Q;0yQ~4#E-m5l+Av(94(hIrza4hbQ<#09*;-5CyFu9+Ds(+CyjP z2073hZh%3M4?|!ijDbqH6(+(RFaz#}d*K0C43EMSumV=YI@kcOLM?2AcVQ>&hR@+E z_!bVsG57^e13#GM;P+}A9^eE15Dd*A5@H|@5+MyTpc7mJJ)jr#hnpY|ieUtdhOtl$ z6X1534zpn%+z$`IQdkC0!?Um!UV=^VI&6h^-~;#=_Q02L0KSJh_!&-tRt`lRCg9gH z98JIvf*=ecAR1yJ0aBqIWI|WC7OsbWFc5BrA{Y*(Pyyp$JWPQ*VHV7V1@Is&ff{%U zR>AYI9$tah;7!;L@54v%8SH~^;5+yMj>E5@ksk+f@TE@<2ZTc$w1e(&BMgDDFa_qo z!>|Hgf;Ztq_!^GEFK`<86*fl`2!+;=4%a|GD1Qkov<4|hp*sUI1I<&7dQ?4pqzu>b8>iq5BNhc zG=oTpfjCHnG{}HXa1Hc;UeF(If;=dO5ilCYLN!c)+hID)h6iCeya2WE0qldLa0Wc& z*Iyh#a1|s&XXphtLkav1X2AVW15d#!cpkREd+<4Y55Iy{ep|*70FjUY9ib--f)Owd zronyiC_D?Vz}xT%d<#DVUqR+@KsdxfJLnEK!Vnk>Q(z7}3@hLzcoRN^ui+SIg9UF0 zfmVcpvt{5jYKYd6VM^gqDy5o#1-Ng;8)TOos*VIIMwJ z;a%7R-@!?+$Pe~7{2>Bd&;fGbCKwL4z*M*wmclC72yek|H~`0iALMiRLNkbkt05b1 zfMTeC$#4%WhNodYyaBu5E2snc;E}xdgDW8hQlSg67muQ z{3MctFHCdrsSXDpsBrKJ3P&q&LMmiHXXp;S;0DNrA{YTCEj3K`HDvf+B@4}+ip zhCwNeh1+2sEQM!b6TAa^;Cna)_7*}AL_;ccg?>;36)*+n!V*{oufTTr48DV3LB14E z8Vat06zBqdp%BVpGR%QT;2GEmZ^Nf>2u^}EQn&(ILNa7QA1Ht_m<0F0!>|%whPU7o zI0z@e5+wvc6eK|w^nn5>gGq1?JPa%0Wq1odfkSW-tSyBrpd}*ae@$J~#k};YT&N}?I9VihEC88dO}|q2zf9BMnMHs!$g<{vtS-P01w0C@Dw}?>)>VB z0&l`QuoFIoeQ*E{!;kPQXk!F3cmn6l96`_wqM$XnAQdv8Gi1Yc&>sdtA&h`APzATa z6qpWo!+o#_mcSFR5}t#X;8plLY=;kEH+%v6;X9~L=qTquH(P!8i@0^9*JVGb;Shu~3I4y#}-Y=F)125g59U^jdL`{6sNgA?!@ z7+VW=@PPmbfff)AagYR8Lnr74J)tk$1o6sSa6&4yhb-t0*F%38 z1O+f0Mnfgs22)@<+zt1^B3J^;UZy7jDae+4W__!xEtofgYYOUhiBmh*a)w|R(KaagwNnB zI0QeyFK`BoPC>rZPxi481kIo&w1Gru3mu^=w*8;W(TEdx8)I(U1xm&>6Df zMi>euPyy929qxx^uoi0JE!Y7c!2$Rg%!z_OL_rdChTf15V_+iO0}sPWcp2V?_u*ss z9FD*lFeC{!Xadb34w9fPbcE~SW+;X6Fbf`nr(pwZgWd2g`~v1=!5<Vu70b`&FZi89yAS{I^U?psXx8Z&G7(Rz1a0a~E3L(%M+Cq02 z0K?!Gm<9`=2A+r4-~;#)>Ogz7-~-Jd4l*DI2Ej-ugIi!c%!Y^HX;=*}z(&{(d*CZL z2uDHFPVk0Mh=3S~hxX7D2E!=$8_a}-@Cekv)9?~(gWd2g`~v0-!5<$cHg7 z5$=Y?umU!~HrNf{!Y^QMFZe?wBtj?X1$i(ACc@pY7*@ar*ao}dTlfXc9Rz>45}LzR zkOEzxFBC#KOoln|2&{rvU^{#U-@&h7>nH@mRgeN*pf40cIZTE*@CdAeS71AQ2H(N2 zz>jV_0^ur%g+xe)Y#0E;p&DkuLU;sf;Az+ZZ^5VV9h?G>PC^K@fM|$=4sadZ45MKp z%z-7a8eV{n@EYudui+;!br$>~3R2)2xB-U3EpR7108he;uoZU0K{yHaEWrl?AOvC| z13E)CTnBkD1}4EASOTkIGwgtUa18Wa1a6RUw15QY1ihgE%3%uJ3y;Bb@H*^-ui+;! zb`|^}5|SVb`oR#m1;)eeFas9BQ}8UTgO_1D?18V~ARGnlHG&U>gA+PJFUW^7m<)5_ zQFsnsgAd><_z{fV1V4y`B*=n(Fa)Y#8Z3Zi@B+L6AHf0m1+3XZ5X3-R=ney61l$TU zVG%q9FTpnW1P;L|@aQgF39aF3xE5}Lk?=RT3m$~0VFSDcpTYNV8a%HRLLe40AO{9R z35o8Vp83qOGFdchZ(g9|dD7vw`3OoqAeD6D}k@IHJA z$H34_XbKUK2%Vu16hZ~u4)?-Bcm!%-E!4s;*bm3S)LZa}C`g7b&<~2C5~jj@sDZUm z3%g)H{0x>p!W9q=Y0wRBgkdlaro#j9Fgy-V!3KB>K81a701m@%(4?>64_88SNPtez z4SGUfD27Uy3U|R=xF4Q?7vW9#7`}xQVC^RaLJY)1GF%PU!XOw0x4|rU2v)$$upK^! zqoC<8G=XLi4;|n-xEV&nM7ReYfmN^x-i0sVC}?gFnm{->p(9)m`A`OvVIDjN&%tZ3 z6ZXT;V7XBUgfNJNR*(+e;0721RWKFqg1K-%JOL}=Iam*G!bk8890%h7!45tU08x+x zouN16!x)$dcf(>>0WZT_@Ch7*lVBYvTmda18M2@c6hIkFhB@#Etb$iyJA4M;39_5u zU;I@bC&IUW{Kn>$-w$|d+Kx|t-4qnqYWTYYg719riJY~42IckFYC6b4XME=`9{u2s zK|gM|^Sjq_u5GdZ(EImxy|v53@2&o*`OZhLe&(yUVsx5KWm4~j5|6fnTRhk0nYBHx z|2#VU>-TSc$WRba-E`Uv{gw9A3Aab;~UVZI2ukz0_^L*wXJb2Wg+0;Vn zrQh`IlPj0slBpdZ^~$fiR^6Z-8ueb~=+_HEW0DX5I`{74&yDgW>ryXiP1@0GZoU0^ zTeIx{QdcM+Gc1^?ttj+neW^G!x}6%lJ#autrq;VdLdOzd!(CGY%^#O*Qp<2h9`#w1K z{1XqZeC@lR%$iL*q~6BA`A3ZyH)mdc&Wk%Yym;?3KcysH{iyFoy(Z@I%88>MzhbFH zGf}?1KK^l^{$p?N`N%VWYg^ELcFmQhmip!!e%;x=TlRgUfBgGPgMR5{)pU^GGl~D^ z_7e-Q?Ktzq-N%c|Uhn=?v|pPmHXj_fdf0J`^@jtl88^%g4`ddiCM(vf#)j zzZ|S>`F7MVhY3JsX#Qu;uxu_j(Cx4KT&TsOX#mqxQmCv|U2Jo>ojJCj1T1uckbe)G4di>Dua<-VJDTfV*R z#rIEj8$QHaese?WHF?~&XMEuBt3Ai3d7LUfoVTiVhX;4O+;sBefp+tbX76lR(nO=_ z!1U&J?bX9Re#wI6zu0H3_ZWKJZOhKwxv%}AlYX81Y+3YP=RzN;!}Mmh?)LHi_nO_Y zuFqA!#Wy)%2~5envH!e#tCkE~I{H{q_(5pd< zD|d8i{rJu`sRskI_S@Iye`wcqI4br2(e^RQLuRVi#GjeD;(;sYww<(K^-o8ho3?K0 z(HA%T?P_g@Te{z~t;5(oKlbwX)NJZ3_5RWJG09WY;Towo>b^T8yas&J;*)Lb2G8C5 z`p32FHu&FOchA`4M~A*_No;5c{h5@3U3a~>)Jrq*ajEx@whvElO^2tX-uU4K5vz~%_KAAF&-c-j zADmXU78v5;T484f-J_`OX_x1U_oMuIGx_Ab;y?o%6gydY+u+b=JV6We?6ta4mK=WYmV^3|5CXqm4~>P_@(|8vJ%Z|QO8 zrhAgJvl8AvTC=w9??(@P9)QV`J#yL8m21Arc=h&H zbMHDmr}*)96QBIzbK}hkTl@KFHno;|CCgWJnHIKwR_h09hvq$fUs>1Z7wdB?yZC$S z_7%0y+c5a=vV$2d^>VtterWkEPwFTA`1%_wymq`YWbNFKvbXjeKJVD`(;xim;F~9W zG!rYN-mr|x%L+apUe~gBQ1>0puK4*4SM_5a-I^6YQ#$|E?cF>k49h9GZz+q~2B9h%^n@6Ol0 zaZ}M%MN6OGG-Tx)#|Ol%SvGXS%|jiUO|MJ6$nA&vCl-w>9R8B?wc2soDKF?>YkeU3 z`=_35@$<~n($4kng&OE+4ZYZCd9QA3XET?wVa4`?Y=S$1hhc zNDP%7mdPmF?{~)z@9p{YmtX7~yHEB1^_x;bBY9>0Rp7Z5>Up`x)51l!6 z{2SN3+pXZajJmqJ1|52@Z?{U*t|oiC1xP>OwdTn$`^OzGXt^e_`S_P=R(DP!jhTSXRn$tqG;XRw;#K$_q7Y}X??elvh2C($8Nc~txQjLS(+_9t~2JF4?lgqX(f8$y2=oxRm zoA0mLG+gSH_iFw6r*Y=GZ}dk3TXp#P{-`^iUO#l^#J6%j{7UyqT{Jh~O_bMG&9-Cj ze)C&W!1ED%D*H?>e(#5aEzC(T2PSvw+j3g*7;Co@S>F!So77?M(qDI!&KmM{tH_{x zUJWn*Xx)GS`uHWJTSB@U=@f)ZcZY(sDAFL^4I093Mxsh#h8Oq6uv( z@BOpy#T7*cHEAhI{X#D{ez#TjVF7>P;J+G%ExOs&J=FTyCm_0Irk-sft}T?(J_AvT zO}CIqYseTN0~nx3Kp@u8i)1}q-^*68fc>}gTDtGReQjn!F6B^}8 zEfqlgz+WqYP@dt^dn?MPQPZI&Fy-a@*KvBiNkW|R2JYPistf`~iXbk4gQ1;a%8S`+ zKT2JX@({{red|F1_pI~A?e&CN#cZxCVx&(yB@xl?Kx zwUBB!M{3Jdq<5rG219_4FWoth0CgQW)b1CIvE3G8j?(7KB^x%eNsqZq+R~&fA3jyowAK1U_TL`?Da_%U<@%Jae#1!Tx%26^+y}_w8SU zFTF!1J-^fAe6B&ehk$!1XN-I^ou|*UvptOQmU^j(TRmDyTKzfPcrl>2`AiaQ%tVl* z!FZ9I4L`lJ=_q{g!vLH9o6h1(7X!Kd5%RrK`;BwqhG}1mK+J|0{ z&`)U2Os19|5rmFgo#00WrkG8eHkAMlybzstyXga8@KbYLZDoU3Y*FT6sY|}cBwv^( z)NDrR$oS?72gt#A54h7-*N1w4*+0i*##Kv+-Y;Wvp~L7qjQWV~T${LX@pua4DI5%e zuv03kY7La?C)4-dw=Q;-rdyK>u?aYod4^E@sn2;Fv z2GD`$Z#bAM!ZvvV1NdB|3jcfcL4x8>&ViAa2U_=ckfR zoJ@42SFxjGLP<1n-*ghM2yq0+_y@L*F1*1LEQ}OfPd&Vv{=^^PMpRJWBbj+0rq+mE zJ>dK&R=UdJidvY`3p|QKKRR1m@rFQF%uN;wBwn2Un15w*10DvZUpLUsmIG1?m+99* z{JcQ?gc%?DqF1@kA)Sw6KgfPYhl^+8=XR~SVv3jjOlgQh3OtI!Ab%BVzic-R+Md60 z)Nvm%Kv+;Ws-Z~i8krML6Xh5EWFR*N{({)R2;-Vw{XyMRTh#Pi+Wg2s^4Dr7w}n3n zQS-PWQci4}))e>)=I#WSTlu5MZrnm>Uq}v!nZ<`62z{s(GCmFYWn%ul@8P-mF31D$ zQJFjMH#N{M(1T%h-M5if4!(U=vw4QDVnlQvMn^7&{9R-Cn3%uv#ozm8pnoB*dxbv5 zh036*_mg4Ux4`|}_KtULoq7hEeiBo!CFD%qErH$xz}fn4QFKdr^k(JwJ_He&98ZifcKwNxz(GFDgouNi|6?14EA!b_1ck z!vVk#6`9j7_lz7p5qse#D8)?ayB6zYOwxSQ`EiPizjNYe!Omht2JU?0TQr_e9*d}sYY_h&gT=ZLwHH6V^c zI85P!#&R{N(wwz|9u;3i<2&p87n}q=}P( zwW(W<8(V3VtTo{ISlZ@y>v!F5+{^3^AP*=3N6cPK6Wjh1WhjFh(;McBqH=(NK#dI` z9?$gW>~H7laHfF4Gk|?p^eW%%@NFuIceb8N;$;(MD6$kBUz(&9?IPpl%szb`1L9Hu zxHnx=*W35R7pPL+@&6g2`zVa$GND&dFR-+*(zfa*<<2{nG8edq zHhY>|j|%w#Bw+`>ZE6i82kxltJ7=jp4|>f9(nT%2K05n4UFOQTD9*Zp_(6Khm)9*G zRONw7yT+efUWa%SuVDnxUaQcKR{)#<9{{@ZKKBClLH_2o z8=Hzd4Cal!$jF*yY}cnfCl*y*oRfdgd9p0H=t}7g;z9$sOPA<>eVx*cA7@(1i9<1J z39SM)KIfrW-Dg`vd!wSvSq1vH>_4>I0{XY|KXAUFO8XQ9Q@>2lhc6cZV#M;-|AdBC!|mEG4x*+44)l&yx23&RYq*I~W&6 z0;dHKHUFXADG-!~|G));HtE2nnTH#r5)|{X{z%yllV`rr;1gXvVRD!V`7a@iJREBN zU|=7_H8CrmlZQMziD5Ql(4CRKm-QMMf5HlN5QheZH4AOY?GK3GXVC66o_w3v*!wOt z{`BxDw1JPBW=U+1lO{htDt?du$U}SmJ|!#&0XV%eqvDcH3POgeUhYZBPwPi@>5S+v z#TN8z75f4YEHNY~L7fJ0cgU#BwCL#U*@4j86#r z^9M@{>aFz#6*X;q!gnZP@GZb)%v~LdXKBv%nU}sQrrptHQ+>akL`EE<)S}Kwl>k&u~}%~{tZ7#&Q|3PVp*i*Rp*B0C7{*- zL$Z%=H`2`$sLr}9?*9ajjUM=Teb_veqZC6tX{c204gOPuAnvaDMhsv;O#+r&$(%P` z{7$9JWit3H`TA}USJpyE2&cgQ-cYd$f$~Z;(1W-QdL@!UOjop<)fgZ0dG-9-P`Aap z2mPg0?mS7H&1P@n00Vw>EM^!}JTfP0fs6M)2R!2aE`t;HX<~57g8Jk2xYXp^`{}@6 za3=ZAc|<%AgZxlO=c|qml;7GZ^dR@uvu6IXn4e~@YRXl(aI>mo_J=wF*avDRFw#6- zQ}0Myat^$m@?L^HJBNfs!3Tlf^v)UlR%S!2A%T3QTJ0SDD| zGW2s3!|EM(hE#f~JfV@Y5E>VMl<#26$Igz2Rf%k7HbbC<5^v#s4>QC4#~ zAkLsRMX+FsQm298goKbq?D^`{o1h0amR)~)CE1Uht&?+w|7D*B25$y>clBE?h#k;_ zrPE>u zdXy(ih4TV%-GDpRnfovnGIXP+JS{6SGAyte;#?!ApReB|Pa5EXMR$8h3q$Myu9v0u zVfDuEPd+@;Wif0P2L>i#J;!HVBLSUkYGwBxn|!%X79#hk=5vF_*J z9tUe()(F18TFF$azE_lGtGqLmzXA>{1t|fps{dU5_Q-kdU&?s68n*c(F2ujrvc`XM z+_?m@`4(S&P6s}Ku@|;TQ16~hWVjc~5wjn$BU?_v_0N~&l$3L4=9zEa8f^d+KZuk2 zy!(#}ttC=YM;E(iqeZ$dSe-_8CvvM#*qR}PptBipK@ew6C4>*xYW1&$IT!SjqL@C{ zU8iD8#{N+{meSky#zM6k#2MmLh-||X=oy&(q%Hqy>k>uX!p|G|I9iOHc;t*TakPS< zICB9GPp6z+;-p$tpl^Om@p*bZ@1~I4nrU#!8p+vx_k!JsI?yh}eHh9*qeptRY5b~{ z;=^ji_0pk_gbuA8bE%*m){xBir(iz$|NiR0gsBNlubB4Sl1C|1x+la1J1@`|toM;B z!n&5ck_Lc%@P~jsM^gEhUW?%ug$6UjkeT6e#mm4?eKd~Vu*+g=cfH; z9k+GUdA>MNFD5ike+F@Z@+z!&;Pl=-oTbiw5l&rxri#w^DcmH&m(ND#-Q>e{9f`hx zb^%AA(v`l(LV?;HqV=y@`-!zIVcZy(*l6rK`7;8dy`OniiC}QH0(o%PxbYRV3ylW> zkzYOc!&qN25sQ#9CyyfgUZ|d|hf+}sm`~=HFsDp31BM4&(;@Hy*|qC0G^A-VFC(Nv zSU7!;R^pYus4X)Hru8ek7ShAu;6)jCjvM`ea{`>zx?2r^qHfr?QkH&ld7vtQRH&h|u|_;HWP& zP=p2CnWGRJlIN}#A&sIhs09SuIoaE6roklV0B|2DWxtAVn=LD>cISGtJ7N&n$T*!o z=;gE7IRA_->Y2mL1lBb$?_S?h29pvTz`ZM_x|t-oD%P~CbtPmPkl!E@T1Gxw7?*AF z{S+>b{5K!yF$3_e{;+uyu;rul^&B{?-g<5&6QKR{myfIQfh@C=iZ zMV5Ck+*p{H=+Q8Fn3do^cz25DQh}j+>6(H@DHU9H}&E;&qVuR z&|F#K-BP)`&L1gOeMW3Qsa1=0*9_60%BJUMxV=KK7 zMZ|!JVBEYz%Y|>Cces^M%=lIx?HI6+53p(o*Wy72)1{R∋ewBQC*(xu4A_3maYQ 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