Skip to content

Commit 95b9337

Browse files
committed
Merge #650: Replace macros with traits; other cleanups
164bde9 miniscript: add parsing benchmarks (Andrew Poelstra) bbafc39 expression: add parsing benchmark (Andrew Poelstra) 30d2d11 miniscript: factor out a couple parts of Terminal::from_tree (Andrew Poelstra) f22dc3a miniscript: use new TRUE/FALSE constants throughout the codebase (Andrew Poelstra) be41d8b miniscript: add constants for 1 and 0 (Andrew Poelstra) 270e65d Replace macros with blanket impls (Andrew Poelstra) 5ec0f44 add `blanket_traits` module to replace very noisy trait definitions (Andrew Poelstra) Pull request description: There are several macros we use to implement functions when our `Pk` types satisfy a large bundle of trait conditions. There is a trick rust-lang/rust#20671 (comment) that we can use instead. While I'm at it, do several other small cleanups. As a weekend project I rewrote the expression module and was able to get a 10-15% speedup on parsing miniscripts, while eliminating recursion and simplifying the algorithms. I am still working on cleaning up the code and improving the error handling. This is the first set of commits from that branch, which should be simple and uncontroversial. ACKs for top commit: apoelstra: > ACK [164bde9](164bde9). Are you taking a performance refactor stab at rust-miniscript? sanket1729: ACK 164bde9. Are you taking a performance refactor stab at rust-miniscript? Tree-SHA512: 42e11f5f45fa705e14334e79126a66c97577fc6e807f804beaa1532bf3693a6c41a8b714bc4d1436209a1e0d808dc6a3ccc7198f20ff467f40f12126d1ee02f3
2 parents 4c4597e + 164bde9 commit 95b9337

20 files changed

+397
-377
lines changed

src/blanket_traits.rs

+72
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,72 @@
1+
// SPDX-License-Identifier: CC0-1.0
2+
3+
//! Blanket Traits
4+
//!
5+
//! Because of this library's heavy use of generics, we often require complicated
6+
//! trait bounds (especially when it comes to [`FromStr`] and its
7+
//! associated error types). These blanket traits act as aliases, allowing easier
8+
//! descriptions of them.
9+
//!
10+
//! While these traits are not sealed, they have blanket-impls which prevent you
11+
//! from directly implementing them on your own types. The traits will be
12+
//! automatically implemented if you satisfy all the bounds.
13+
//!
14+
15+
use core::fmt;
16+
use core::str::FromStr;
17+
18+
use crate::MiniscriptKey;
19+
20+
/// Blanket trait describing a key where all associated types implement `FromStr`,
21+
/// and all `FromStr` errors can be displayed.
22+
pub trait FromStrKey:
23+
MiniscriptKey<
24+
Sha256 = Self::_Sha256,
25+
Hash256 = Self::_Hash256,
26+
Ripemd160 = Self::_Ripemd160,
27+
Hash160 = Self::_Hash160,
28+
> + FromStr<Err = Self::_FromStrErr>
29+
{
30+
/// Dummy type. Do not use.
31+
type _Sha256: FromStr<Err = Self::_Sha256FromStrErr>;
32+
/// Dummy type. Do not use.
33+
type _Sha256FromStrErr: fmt::Debug + fmt::Display;
34+
/// Dummy type. Do not use.
35+
type _Hash256: FromStr<Err = Self::_Hash256FromStrErr>;
36+
/// Dummy type. Do not use.
37+
type _Hash256FromStrErr: fmt::Debug + fmt::Display;
38+
/// Dummy type. Do not use.
39+
type _Ripemd160: FromStr<Err = Self::_Ripemd160FromStrErr>;
40+
/// Dummy type. Do not use.
41+
type _Ripemd160FromStrErr: fmt::Debug + fmt::Display;
42+
/// Dummy type. Do not use.
43+
type _Hash160: FromStr<Err = Self::_Hash160FromStrErr>;
44+
/// Dummy type. Do not use.
45+
type _Hash160FromStrErr: fmt::Debug + fmt::Display;
46+
/// Dummy type. Do not use.
47+
type _FromStrErr: fmt::Debug + fmt::Display;
48+
}
49+
50+
impl<T> FromStrKey for T
51+
where
52+
Self: MiniscriptKey + FromStr,
53+
<Self as MiniscriptKey>::Sha256: FromStr,
54+
Self::Hash256: FromStr,
55+
Self::Ripemd160: FromStr,
56+
Self::Hash160: FromStr,
57+
<Self as FromStr>::Err: fmt::Debug + fmt::Display,
58+
<<Self as MiniscriptKey>::Sha256 as FromStr>::Err: fmt::Debug + fmt::Display,
59+
<Self::Hash256 as FromStr>::Err: fmt::Debug + fmt::Display,
60+
<Self::Ripemd160 as FromStr>::Err: fmt::Debug + fmt::Display,
61+
<Self::Hash160 as FromStr>::Err: fmt::Debug + fmt::Display,
62+
{
63+
type _Sha256 = <T as MiniscriptKey>::Sha256;
64+
type _Sha256FromStrErr = <<T as MiniscriptKey>::Sha256 as FromStr>::Err;
65+
type _Hash256 = <T as MiniscriptKey>::Hash256;
66+
type _Hash256FromStrErr = <<T as MiniscriptKey>::Hash256 as FromStr>::Err;
67+
type _Ripemd160 = <T as MiniscriptKey>::Ripemd160;
68+
type _Ripemd160FromStrErr = <<T as MiniscriptKey>::Ripemd160 as FromStr>::Err;
69+
type _Hash160 = <T as MiniscriptKey>::Hash160;
70+
type _Hash160FromStrErr = <<T as MiniscriptKey>::Hash160 as FromStr>::Err;
71+
type _FromStrErr = <T as FromStr>::Err;
72+
}

src/descriptor/bare.rs

+10-14
Original file line numberDiff line numberDiff line change
@@ -166,24 +166,22 @@ impl<Pk: MiniscriptKey> Liftable<Pk> for Bare<Pk> {
166166
fn lift(&self) -> Result<semantic::Policy<Pk>, Error> { self.ms.lift() }
167167
}
168168

169-
impl_from_tree!(
170-
Bare<Pk>,
169+
impl<Pk: crate::FromStrKey> FromTree for Bare<Pk> {
171170
fn from_tree(top: &expression::Tree) -> Result<Self, Error> {
172171
let sub = Miniscript::<Pk, BareCtx>::from_tree(top)?;
173172
BareCtx::top_level_checks(&sub)?;
174173
Bare::new(sub)
175174
}
176-
);
175+
}
177176

178-
impl_from_str!(
179-
Bare<Pk>,
180-
type Err = Error;,
177+
impl<Pk: crate::FromStrKey> core::str::FromStr for Bare<Pk> {
178+
type Err = Error;
181179
fn from_str(s: &str) -> Result<Self, Self::Err> {
182180
let desc_str = verify_checksum(s)?;
183181
let top = expression::Tree::from_str(desc_str)?;
184182
Self::from_tree(&top)
185183
}
186-
);
184+
}
187185

188186
impl<Pk: MiniscriptKey> ForEachKey<Pk> for Bare<Pk> {
189187
fn for_each_key<'a, F: FnMut(&'a Pk) -> bool>(&'a self, pred: F) -> bool {
@@ -366,8 +364,7 @@ impl<Pk: MiniscriptKey> Liftable<Pk> for Pkh<Pk> {
366364
}
367365
}
368366

369-
impl_from_tree!(
370-
Pkh<Pk>,
367+
impl<Pk: crate::FromStrKey> FromTree for Pkh<Pk> {
371368
fn from_tree(top: &expression::Tree) -> Result<Self, Error> {
372369
if top.name == "pkh" && top.args.len() == 1 {
373370
Ok(Pkh::new(expression::terminal(&top.args[0], |pk| Pk::from_str(pk))?)?)
@@ -379,17 +376,16 @@ impl_from_tree!(
379376
)))
380377
}
381378
}
382-
);
379+
}
383380

384-
impl_from_str!(
385-
Pkh<Pk>,
386-
type Err = Error;,
381+
impl<Pk: crate::FromStrKey> core::str::FromStr for Pkh<Pk> {
382+
type Err = Error;
387383
fn from_str(s: &str) -> Result<Self, Self::Err> {
388384
let desc_str = verify_checksum(s)?;
389385
let top = expression::Tree::from_str(desc_str)?;
390386
Self::from_tree(&top)
391387
}
392-
);
388+
}
393389

394390
impl<Pk: MiniscriptKey> ForEachKey<Pk> for Pkh<Pk> {
395391
fn for_each_key<'a, F: FnMut(&'a Pk) -> bool>(&'a self, mut pred: F) -> bool { pred(&self.pk) }

src/descriptor/mod.rs

+5-7
Original file line numberDiff line numberDiff line change
@@ -918,8 +918,7 @@ impl Descriptor<DefiniteDescriptorKey> {
918918
}
919919
}
920920

921-
impl_from_tree!(
922-
Descriptor<Pk>,
921+
impl<Pk: crate::FromStrKey> crate::expression::FromTree for Descriptor<Pk> {
923922
/// Parse an expression tree into a descriptor.
924923
fn from_tree(top: &expression::Tree) -> Result<Descriptor<Pk>, Error> {
925924
Ok(match (top.name, top.args.len() as u32) {
@@ -931,11 +930,10 @@ impl_from_tree!(
931930
_ => Descriptor::Bare(Bare::from_tree(top)?),
932931
})
933932
}
934-
);
933+
}
935934

936-
impl_from_str!(
937-
Descriptor<Pk>,
938-
type Err = Error;,
935+
impl<Pk: crate::FromStrKey> FromStr for Descriptor<Pk> {
936+
type Err = Error;
939937
fn from_str(s: &str) -> Result<Descriptor<Pk>, Error> {
940938
// tr tree parsing has special code
941939
// Tr::from_str will check the checksum
@@ -950,7 +948,7 @@ impl_from_str!(
950948

951949
Ok(desc)
952950
}
953-
);
951+
}
954952

955953
impl<Pk: MiniscriptKey> fmt::Debug for Descriptor<Pk> {
956954
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {

src/descriptor/segwitv0.rs

+10-14
Original file line numberDiff line numberDiff line change
@@ -231,8 +231,7 @@ impl<Pk: MiniscriptKey> Liftable<Pk> for Wsh<Pk> {
231231
}
232232
}
233233

234-
impl_from_tree!(
235-
Wsh<Pk>,
234+
impl<Pk: crate::FromStrKey> crate::expression::FromTree for Wsh<Pk> {
236235
fn from_tree(top: &expression::Tree) -> Result<Self, Error> {
237236
if top.name == "wsh" && top.args.len() == 1 {
238237
let top = &top.args[0];
@@ -250,7 +249,7 @@ impl_from_tree!(
250249
)))
251250
}
252251
}
253-
);
252+
}
254253

255254
impl<Pk: MiniscriptKey> fmt::Debug for Wsh<Pk> {
256255
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
@@ -270,15 +269,14 @@ impl<Pk: MiniscriptKey> fmt::Display for Wsh<Pk> {
270269
}
271270
}
272271

273-
impl_from_str!(
274-
Wsh<Pk>,
275-
type Err = Error;,
272+
impl<Pk: crate::FromStrKey> core::str::FromStr for Wsh<Pk> {
273+
type Err = Error;
276274
fn from_str(s: &str) -> Result<Self, Self::Err> {
277275
let desc_str = verify_checksum(s)?;
278276
let top = expression::Tree::from_str(desc_str)?;
279277
Wsh::<Pk>::from_tree(&top)
280278
}
281-
);
279+
}
282280

283281
impl<Pk: MiniscriptKey> ForEachKey<Pk> for Wsh<Pk> {
284282
fn for_each_key<'a, F: FnMut(&'a Pk) -> bool>(&'a self, pred: F) -> bool {
@@ -475,8 +473,7 @@ impl<Pk: MiniscriptKey> Liftable<Pk> for Wpkh<Pk> {
475473
}
476474
}
477475

478-
impl_from_tree!(
479-
Wpkh<Pk>,
476+
impl<Pk: crate::FromStrKey> crate::expression::FromTree for Wpkh<Pk> {
480477
fn from_tree(top: &expression::Tree) -> Result<Self, Error> {
481478
if top.name == "wpkh" && top.args.len() == 1 {
482479
Ok(Wpkh::new(expression::terminal(&top.args[0], |pk| Pk::from_str(pk))?)?)
@@ -488,17 +485,16 @@ impl_from_tree!(
488485
)))
489486
}
490487
}
491-
);
488+
}
492489

493-
impl_from_str!(
494-
Wpkh<Pk>,
495-
type Err = Error;,
490+
impl<Pk: crate::FromStrKey> core::str::FromStr for Wpkh<Pk> {
491+
type Err = Error;
496492
fn from_str(s: &str) -> Result<Self, Self::Err> {
497493
let desc_str = verify_checksum(s)?;
498494
let top = expression::Tree::from_str(desc_str)?;
499495
Self::from_tree(&top)
500496
}
501-
);
497+
}
502498

503499
impl<Pk: MiniscriptKey> ForEachKey<Pk> for Wpkh<Pk> {
504500
fn for_each_key<'a, F: FnMut(&'a Pk) -> bool>(&'a self, mut pred: F) -> bool { pred(&self.pk) }

src/descriptor/sh.rs

+5-7
Original file line numberDiff line numberDiff line change
@@ -81,8 +81,7 @@ impl<Pk: MiniscriptKey> fmt::Display for Sh<Pk> {
8181
}
8282
}
8383

84-
impl_from_tree!(
85-
Sh<Pk>,
84+
impl<Pk: crate::FromStrKey> crate::expression::FromTree for Sh<Pk> {
8685
fn from_tree(top: &expression::Tree) -> Result<Self, Error> {
8786
if top.name == "sh" && top.args.len() == 1 {
8887
let top = &top.args[0];
@@ -105,17 +104,16 @@ impl_from_tree!(
105104
)))
106105
}
107106
}
108-
);
107+
}
109108

110-
impl_from_str!(
111-
Sh<Pk>,
112-
type Err = Error;,
109+
impl<Pk: crate::FromStrKey> core::str::FromStr for Sh<Pk> {
110+
type Err = Error;
113111
fn from_str(s: &str) -> Result<Self, Self::Err> {
114112
let desc_str = verify_checksum(s)?;
115113
let top = expression::Tree::from_str(desc_str)?;
116114
Self::from_tree(&top)
117115
}
118-
);
116+
}
119117

120118
impl<Pk: MiniscriptKey> Sh<Pk> {
121119
/// Get the Inner

src/descriptor/sortedmulti.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -60,7 +60,7 @@ impl<Pk: MiniscriptKey, Ctx: ScriptContext> SortedMultiVec<Pk, Ctx> {
6060
pub fn from_tree(tree: &expression::Tree) -> Result<Self, Error>
6161
where
6262
Pk: FromStr,
63-
<Pk as FromStr>::Err: ToString,
63+
<Pk as FromStr>::Err: fmt::Display,
6464
{
6565
if tree.args.is_empty() {
6666
return Err(errstr("no arguments given for sortedmulti"));

src/descriptor/tr.rs

+7-10
Original file line numberDiff line numberDiff line change
@@ -467,8 +467,7 @@ where
467467
}
468468

469469
#[rustfmt::skip]
470-
impl_block_str!(
471-
Tr<Pk>,
470+
impl<Pk: crate::FromStrKey> Tr<Pk> {
472471
// Helper function to parse taproot script path
473472
fn parse_tr_script_spend(tree: &expression::Tree,) -> Result<TapTree<Pk>, Error> {
474473
match tree {
@@ -487,10 +486,9 @@ impl_block_str!(
487486
)),
488487
}
489488
}
490-
);
489+
}
491490

492-
impl_from_tree!(
493-
Tr<Pk>,
491+
impl<Pk: crate::FromStrKey> crate::expression::FromTree for Tr<Pk> {
494492
fn from_tree(top: &expression::Tree) -> Result<Self, Error> {
495493
if top.name == "tr" {
496494
match top.args.len() {
@@ -530,17 +528,16 @@ impl_from_tree!(
530528
)))
531529
}
532530
}
533-
);
531+
}
534532

535-
impl_from_str!(
536-
Tr<Pk>,
537-
type Err = Error;,
533+
impl<Pk: crate::FromStrKey> core::str::FromStr for Tr<Pk> {
534+
type Err = Error;
538535
fn from_str(s: &str) -> Result<Self, Self::Err> {
539536
let desc_str = verify_checksum(s)?;
540537
let top = parse_tr_tree(desc_str)?;
541538
Self::from_tree(&top)
542539
}
543-
);
540+
}
544541

545542
impl<Pk: MiniscriptKey> fmt::Debug for Tr<Pk> {
546543
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {

src/expression.rs

+29-2
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@
22

33
//! # Function-like Expression Language
44
//!
5+
use core::fmt;
56
use core::str::FromStr;
67

78
use crate::prelude::*;
@@ -218,7 +219,7 @@ pub fn parse_num(s: &str) -> Result<u32, Error> {
218219
pub fn terminal<T, F, Err>(term: &Tree, convert: F) -> Result<T, Error>
219220
where
220221
F: FnOnce(&str) -> Result<T, Err>,
221-
Err: ToString,
222+
Err: fmt::Display,
222223
{
223224
if term.args.is_empty() {
224225
convert(term.name).map_err(|e| Error::Unexpected(e.to_string()))
@@ -259,7 +260,6 @@ where
259260

260261
#[cfg(test)]
261262
mod tests {
262-
263263
use super::parse_num;
264264

265265
#[test]
@@ -281,3 +281,30 @@ mod tests {
281281
assert_eq!(valid_chars, super::VALID_CHARS);
282282
}
283283
}
284+
285+
#[cfg(bench)]
286+
mod benches {
287+
use test::{black_box, Bencher};
288+
289+
use super::*;
290+
291+
#[bench]
292+
pub fn parse_tree(bh: &mut Bencher) {
293+
bh.iter(|| {
294+
let tree = Tree::from_str(
295+
"and(thresh(2,and(sha256(H),or(sha256(H),pk(A))),pk(B),pk(C),pk(D),sha256(H)),pk(E))",
296+
).unwrap();
297+
black_box(tree);
298+
});
299+
}
300+
301+
#[bench]
302+
pub fn parse_tree_deep(bh: &mut Bencher) {
303+
bh.iter(|| {
304+
let tree = Tree::from_str(
305+
"and(and(and(and(and(and(and(and(and(and(and(and(and(and(and(and(and(and(and(and(1,2),3),4),5),6),7),8),9),10),11),12),13),14),15),16),17),18),19),20),21)"
306+
).unwrap();
307+
black_box(tree);
308+
});
309+
}
310+
}

src/interpreter/inner.rs

+2-6
Original file line numberDiff line numberDiff line change
@@ -46,12 +46,8 @@ fn script_from_stack_elem<Ctx: ScriptContext>(
4646
Miniscript::parse_with_ext(bitcoin::Script::from_bytes(sl), &ExtParams::allow_all())
4747
.map_err(Error::from)
4848
}
49-
stack::Element::Satisfied => {
50-
Miniscript::from_ast(crate::Terminal::True).map_err(Error::from)
51-
}
52-
stack::Element::Dissatisfied => {
53-
Miniscript::from_ast(crate::Terminal::False).map_err(Error::from)
54-
}
49+
stack::Element::Satisfied => Ok(Miniscript::TRUE),
50+
stack::Element::Dissatisfied => Ok(Miniscript::FALSE),
5551
}
5652
}
5753

0 commit comments

Comments
 (0)