@@ -422,61 +422,70 @@ impl<Pk: MiniscriptKey> Policy<Pk> {
422
422
( m, n)
423
423
}
424
424
425
- match self {
426
- Policy :: Threshold ( k, subs) => {
427
- let mut ret_subs = Vec :: with_capacity ( subs. len ( ) ) ;
425
+ use Policy :: * ;
428
426
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] ] ) ;
433
430
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) ;
435
435
436
436
let parent_is_and = m == n; // (n, n)-thresh is an AND
437
437
let parent_is_or = m == 1 ; // (1, n)-thresh is an OR
438
438
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
+ }
457
458
}
459
+ _ => ret_subs. push ( Arc :: clone ( & sub) ) ,
458
460
}
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) )
460
476
}
461
477
}
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 ) ) ,
477
483
}
478
- x => x,
479
484
}
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 ( )
480
489
}
481
490
482
491
/// Detects a true/trivial policy.
0 commit comments