Skip to content

Commit 1f0486b

Browse files
committed
Remove recursion in normalized
Remove the recursive call in `semantic::Policy::normalized`, using the `TreeLike` trait to iterate nodes instead.
1 parent 6449f38 commit 1f0486b

File tree

1 file changed

+52
-43
lines changed

1 file changed

+52
-43
lines changed

src/policy/semantic.rs

+52-43
Original file line numberDiff line numberDiff line change
@@ -422,61 +422,70 @@ impl<Pk: MiniscriptKey> Policy<Pk> {
422422
(m, n)
423423
}
424424

425-
match self {
426-
Policy::Threshold(k, subs) => {
427-
let mut ret_subs = Vec::with_capacity(subs.len());
425+
use Policy::*;
428426

429-
let subs: Vec<_> = subs
430-
.into_iter()
431-
.map(|sub| Arc::new(sub.as_ref().clone().normalized()))
432-
.collect();
427+
let mut normalized: Vec<Arc<Policy<Pk>>> = vec![];
428+
for data in Arc::new(self).post_order_iter() {
429+
let child_n = |n| Arc::clone(&normalized[data.child_indices[n]]);
433430

434-
let (m, n) = normalized_threshold_values(k, &subs);
431+
let new_policy = match data.node.as_ref() {
432+
Threshold(k, ref subs) => {
433+
let mut ret_subs = Vec::with_capacity(subs.len());
434+
let (m, n) = normalized_threshold_values(*k, &subs);
435435

436436
let parent_is_and = m == n; // (n, n)-thresh is an AND
437437
let parent_is_or = m == 1; // (1, n)-thresh is an OR
438438

439-
for sub in subs {
440-
match sub.as_ref() {
441-
Policy::Trivial | Policy::Unsatisfiable => {}
442-
Policy::Threshold(ref k, ref subs) => {
443-
let child_is_and = *k == subs.len();
444-
let child_is_or = *k == 1;
445-
446-
if parent_is_and && parent_is_or {
447-
// m = n = 1, child must be the non-trivial, non-unsatisfiable node.
448-
ret_subs.push(Arc::clone(&sub));
449-
} else if parent_is_and && child_is_and {
450-
// If both parent and child are ANDs we can flatten them.
451-
subs.iter().for_each(|sub| ret_subs.push(Arc::clone(sub)));
452-
} else if parent_is_or && child_is_or {
453-
// If both parent and child are ORs we can flatten them.
454-
subs.iter().for_each(|sub| ret_subs.push(Arc::clone(sub)));
455-
} else {
456-
ret_subs.push(Arc::clone(&sub));
439+
for sub in (0..subs.len()).map(child_n) {
440+
match sub.as_ref() {
441+
Trivial | Unsatisfiable => {}
442+
Threshold(ref k, ref subs) => {
443+
let child_is_and = *k == subs.len();
444+
let child_is_or = *k == 1;
445+
446+
if parent_is_and && parent_is_or {
447+
// m = n = 1, child must be the non-trivial, non-unsatisfiable node.
448+
ret_subs.push(Arc::clone(&sub));
449+
} else if parent_is_and && child_is_and {
450+
// If both parent and child are ANDs we can flatten them.
451+
subs.iter().for_each(|sub| ret_subs.push(Arc::clone(sub)));
452+
} else if parent_is_or && child_is_or {
453+
// If both parent and child are ORs we can flatten them.
454+
subs.iter().for_each(|sub| ret_subs.push(Arc::clone(sub)));
455+
} else {
456+
ret_subs.push(Arc::clone(&sub));
457+
}
457458
}
459+
_ => ret_subs.push(Arc::clone(&sub)),
458460
}
459-
_ => ret_subs.push(Arc::clone(&sub)),
461+
}
462+
// Now reason about m of n threshold
463+
if m == 0 {
464+
Some(Trivial)
465+
} else if m > ret_subs.len() {
466+
Some(Unsatisfiable)
467+
} else if ret_subs.len() == 1 {
468+
let policy = ret_subs.pop().unwrap();
469+
Some((*policy).clone()) // I'm lost now, can we try_unwrap still?
470+
} else if parent_is_and {
471+
Some(Threshold(ret_subs.len(), ret_subs))
472+
} else if parent_is_or {
473+
Some(Threshold(1, ret_subs))
474+
} else {
475+
Some(Threshold(m, ret_subs))
460476
}
461477
}
462-
// Now reason about m of n threshold
463-
if m == 0 {
464-
Policy::Trivial
465-
} else if m > ret_subs.len() {
466-
Policy::Unsatisfiable
467-
} else if ret_subs.len() == 1 {
468-
let policy = ret_subs.pop().unwrap();
469-
(*policy).clone() // More than one strong reference, can't use `try_unwrap()`.
470-
} else if parent_is_and {
471-
Policy::Threshold(ret_subs.len(), ret_subs)
472-
} else if parent_is_or {
473-
Policy::Threshold(1, ret_subs)
474-
} else {
475-
Policy::Threshold(m, ret_subs)
476-
}
478+
_ => None,
479+
};
480+
match new_policy {
481+
Some(new_policy) => normalized.push(Arc::new(new_policy)),
482+
None => normalized.push(Arc::clone(&data.node)),
477483
}
478-
x => x,
479484
}
485+
// Unwrap is ok because we know we processed at least one node.
486+
let root_node = normalized.pop().unwrap();
487+
// Unwrap is ok because we know `root_node` is the only strong reference.
488+
Arc::try_unwrap(root_node).unwrap()
480489
}
481490

482491
/// Detects a true/trivial policy.

0 commit comments

Comments
 (0)