diff --git a/Cargo.lock b/Cargo.lock index 38b900266..f9644b74e 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -430,6 +430,15 @@ dependencies = [ "libc", ] +[[package]] +name = "core2" +version = "0.3.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "239fa3ae9b63c2dc74bd3fa852d4792b8b305ae64eeede946265b6af62f1fff3" +dependencies = [ + "memchr", +] + [[package]] name = "cpp_demangle" version = "0.4.3" @@ -1303,9 +1312,9 @@ dependencies = [ [[package]] name = "memuse" -version = "0.2.1" +version = "0.2.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2145869435ace5ea6ea3d35f59be559317ec9a0d04e1812d5f185a87b6d36f1a" +checksum = "3d97bbf43eb4f088f8ca469930cde17fa036207c9a5e02ccc5107c4e8b17c964" dependencies = [ "nonempty", ] @@ -1443,6 +1452,7 @@ dependencies = [ "bitvec", "blake2b_simd", "bridgetree", + "core2", "criterion", "ff", "fpe", diff --git a/Cargo.toml b/Cargo.toml index 6bc4b1896..4dc52d42b 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -24,14 +24,14 @@ rustdoc-args = ["--cfg", "docsrs", "--html-in-header", "katex-header.html"] [dependencies] aes = "0.8" -bitvec = "1" -blake2b_simd = "1" +bitvec = { version = "1", default-features = false } +blake2b_simd = { version = "1", default-features = false } ff = "0.13" fpe = "0.6" group = { version = "0.13", features = ["wnaf-memuse"] } -hex = "0.4" +hex = { version = "0.4", default-features = false } lazy_static = "1" -memuse = { version = "0.2.1", features = ["nonempty"] } +memuse = { version = "0.2.2", default-features = false } pasta_curves = "0.5" proptest = { version = "1.0.0", optional = true } rand = "0.8" @@ -57,6 +57,9 @@ getset = "0.1" # Logging tracing = "0.1" +# No-std support +core2 = { version = "0.3", default-features = false, features = ["alloc"] } + # Developer tooling dependencies image = { version = "0.24", optional = true } plotters = { version = "0.3.0", optional = true } @@ -78,8 +81,9 @@ pprof = { version = "0.11", features = ["criterion", "flamegraph"] } bench = false [features] -default = ["circuit", "multicore"] -circuit = ["dep:halo2_gadgets", "dep:halo2_proofs"] +default = ["circuit", "multicore", "std"] +std = ["core2/std", "memuse/nonempty"] +circuit = ["dep:halo2_gadgets", "dep:halo2_proofs", "std"] unstable-frost = [] multicore = ["halo2_proofs?/multicore"] dev-graph = ["halo2_proofs?/dev-graph", "image", "plotters"] diff --git a/src/builder.rs b/src/builder.rs index 622230cb1..988247ec7 100644 --- a/src/builder.rs +++ b/src/builder.rs @@ -1,9 +1,9 @@ //! Logic for building Orchard components of transactions. +use alloc::collections::BTreeMap; +use alloc::vec::Vec; use core::fmt; use core::iter; -use std::collections::BTreeMap; -use std::fmt::Display; use ff::Field; use pasta_curves::pallas; @@ -141,7 +141,7 @@ pub enum BuildError { BundleTypeNotSatisfiable, } -impl Display for BuildError { +impl fmt::Display for BuildError { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { use BuildError::*; match self { @@ -163,6 +163,7 @@ impl Display for BuildError { } } +#[cfg(feature = "std")] impl std::error::Error for BuildError {} #[cfg(feature = "circuit")] @@ -189,7 +190,7 @@ pub enum SpendError { FvkMismatch, } -impl Display for SpendError { +impl fmt::Display for SpendError { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { use SpendError::*; f.write_str(match self { @@ -200,18 +201,20 @@ impl Display for SpendError { } } +#[cfg(feature = "std")] impl std::error::Error for SpendError {} /// The only error that can occur here is if outputs are disabled for this builder. #[derive(Debug, PartialEq, Eq)] pub struct OutputError; -impl Display for OutputError { +impl fmt::Display for OutputError { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { f.write_str("Outputs are not enabled for this builder") } } +#[cfg(feature = "std")] impl std::error::Error for OutputError {} /// Information about a specific note to be spent in an [`Action`]. diff --git a/src/bundle.rs b/src/bundle.rs index 36bc14fa9..16c12affe 100644 --- a/src/bundle.rs +++ b/src/bundle.rs @@ -1,5 +1,7 @@ //! Structs related to bundles of Orchard actions. +use alloc::vec::Vec; + pub mod commitments; #[cfg(feature = "circuit")] @@ -10,10 +12,12 @@ pub use batch::BatchValidator; use core::fmt; use blake2b_simd::Hash as Blake2bHash; -use memuse::DynamicUsage; use nonempty::NonEmpty; use zcash_note_encryption::{try_note_decryption, try_output_recovery_with_ovk}; +#[cfg(feature = "std")] +use memuse::DynamicUsage; + use crate::{ action::Action, address::Address, @@ -472,6 +476,7 @@ impl Bundle { } } +#[cfg(feature = "std")] impl DynamicUsage for Bundle { fn dynamic_usage(&self) -> usize { self.actions.dynamic_usage() diff --git a/src/bundle/batch.rs b/src/bundle/batch.rs index 6626b9162..2786bdd69 100644 --- a/src/bundle/batch.rs +++ b/src/bundle/batch.rs @@ -1,3 +1,5 @@ +use alloc::vec::Vec; + use halo2_proofs::plonk; use pasta_curves::vesta; use rand::{CryptoRng, RngCore}; diff --git a/src/circuit.rs b/src/circuit.rs index 67715d639..7d7abbf60 100644 --- a/src/circuit.rs +++ b/src/circuit.rs @@ -1,5 +1,7 @@ //! The Orchard Action circuit implementation. +use alloc::vec::Vec; + use group::{Curve, GroupEncoding}; use halo2_proofs::{ circuit::{floor_planner, Layouter, Value}, diff --git a/src/constants.rs b/src/constants.rs index 642fa1b24..7f40b5394 100644 --- a/src/constants.rs +++ b/src/constants.rs @@ -3,8 +3,8 @@ pub mod fixed_bases; pub mod sinsemilla; pub mod util; -pub use fixed_bases::{NullifierK, OrchardFixedBases, OrchardFixedBasesFull, ValueCommitV}; pub use self::sinsemilla::{OrchardCommitDomains, OrchardHashDomains}; +pub use fixed_bases::{NullifierK, OrchardFixedBases, OrchardFixedBasesFull, ValueCommitV}; /// $\mathsf{MerkleDepth^{Orchard}}$ pub const MERKLE_DEPTH_ORCHARD: usize = 32; diff --git a/src/constants/fixed_bases.rs b/src/constants/fixed_bases.rs index 23d110c26..0e3a73906 100644 --- a/src/constants/fixed_bases.rs +++ b/src/constants/fixed_bases.rs @@ -1,4 +1,8 @@ //! Orchard fixed bases. + +#[cfg(feature = "circuit")] +use alloc::vec::Vec; + use super::{L_ORCHARD_SCALAR, L_VALUE}; #[cfg(feature = "circuit")] diff --git a/src/keys.rs b/src/keys.rs index f66928eeb..5d6cb3f8d 100644 --- a/src/keys.rs +++ b/src/keys.rs @@ -1,6 +1,7 @@ //! Key structures for Orchard. -use std::io::{self, Read, Write}; +use alloc::vec::Vec; +use core2::io::{self, Read, Write}; use ::zip32::{AccountId, ChildIndex}; use aes::Aes256; @@ -405,7 +406,7 @@ impl FullViewingKey { Self::from_bytes(&data).ok_or_else(|| { io::Error::new( io::ErrorKind::InvalidInput, - "Unable to deserialize a valid Orchard FullViewingKey from bytes".to_owned(), + "Unable to deserialize a valid Orchard FullViewingKey from bytes", ) }) } diff --git a/src/lib.rs b/src/lib.rs index 2ac5dcecb..3a52af5c6 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -7,6 +7,7 @@ //! implicitly mean it is an Orchard payment address (as opposed to e.g. a Sapling payment //! address, which is also shielded). +#![no_std] #![cfg_attr(docsrs, feature(doc_cfg))] // Temporary until we have more of the crate implemented. #![allow(dead_code)] @@ -16,6 +17,14 @@ #![deny(missing_docs)] #![deny(unsafe_code)] +#[macro_use] +extern crate alloc; + +#[cfg(feature = "std")] +extern crate std; + +use alloc::vec::Vec; + mod action; mod address; pub mod builder; diff --git a/src/note_encryption.rs b/src/note_encryption.rs index 56d5599cf..1e13c3aa5 100644 --- a/src/note_encryption.rs +++ b/src/note_encryption.rs @@ -1,5 +1,6 @@ //! In-band secret distribution for Orchard bundles. +use alloc::vec::Vec; use core::fmt; use blake2b_simd::{Hash, Params}; diff --git a/src/pczt.rs b/src/pczt.rs index cebc3eab5..38cc0a757 100644 --- a/src/pczt.rs +++ b/src/pczt.rs @@ -1,7 +1,9 @@ //! PCZT support for Orchard. -use std::collections::BTreeMap; -use std::fmt; +use alloc::collections::BTreeMap; +use alloc::string::String; +use alloc::vec::Vec; +use core::fmt; use getset::Getters; use pasta_curves::pallas; diff --git a/src/pczt/io_finalizer.rs b/src/pczt/io_finalizer.rs index b227df3be..6f9a7ec9c 100644 --- a/src/pczt/io_finalizer.rs +++ b/src/pczt/io_finalizer.rs @@ -1,3 +1,5 @@ +use alloc::vec::Vec; + use rand::{CryptoRng, RngCore}; use crate::{ diff --git a/src/pczt/parse.rs b/src/pczt/parse.rs index eb0c66861..f36e09de6 100644 --- a/src/pczt/parse.rs +++ b/src/pczt/parse.rs @@ -1,4 +1,6 @@ -use std::collections::BTreeMap; +use alloc::collections::BTreeMap; +use alloc::string::String; +use alloc::vec::Vec; use ff::PrimeField; use incrementalmerkletree::Hashable; diff --git a/src/pczt/prover.rs b/src/pczt/prover.rs index 57a055a42..73ce0c863 100644 --- a/src/pczt/prover.rs +++ b/src/pczt/prover.rs @@ -1,3 +1,5 @@ +use alloc::vec::Vec; + use halo2_proofs::plonk; use rand::{CryptoRng, RngCore}; diff --git a/src/pczt/updater.rs b/src/pczt/updater.rs index 5c2e00713..c7691f777 100644 --- a/src/pczt/updater.rs +++ b/src/pczt/updater.rs @@ -1,3 +1,6 @@ +use alloc::string::String; +use alloc::vec::Vec; + use super::{Action, Bundle, Zip32Derivation}; impl Bundle { diff --git a/src/tree.rs b/src/tree.rs index 98824ac83..57fdae308 100644 --- a/src/tree.rs +++ b/src/tree.rs @@ -1,5 +1,6 @@ //! Types related to Orchard note commitment trees and anchors. +use alloc::vec::Vec; use core::iter; use crate::{ diff --git a/src/value.rs b/src/value.rs index 058feb103..dac380ece 100644 --- a/src/value.rs +++ b/src/value.rs @@ -81,6 +81,7 @@ impl fmt::Display for OverflowError { } } +#[cfg(feature = "std")] impl std::error::Error for OverflowError {} /// The non-negative value of an individual Orchard note.