Skip to content
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.

Commit 4ce8015

Browse files
committedFeb 28, 2024·
wip: remove backtrack subtree
1 parent 2d5b15c commit 4ce8015

File tree

2 files changed

+49
-89
lines changed

2 files changed

+49
-89
lines changed
 

‎README.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,7 @@ A basic performance comparison between this current [Rust BnB](https://github.co
2828

2929
|implementation|pool size|ns/iter|
3030
|-------------:|---------|-------|
31-
| Rust BnB| 1,000|695,860|
31+
| Rust BnB| 1,000|897,810|
3232
| C++ Core BnB| 1,000|816,374|
3333

3434
Note: The measurements where recorded using rustc 1.75. Expect worse performance with MSRV.

‎src/branch_and_bound.rs

+48-88
Original file line numberDiff line numberDiff line change
@@ -143,7 +143,6 @@ pub fn select_coins_bnb(
143143
let mut iteration = 0;
144144
let mut index = 0;
145145
let mut backtrack;
146-
let mut backtrack_subtree;
147146

148147
let mut value = Amount::ZERO;
149148

@@ -180,59 +179,14 @@ pub fn select_coins_bnb(
180179
}
181180

182181
while iteration < ITERATION_LIMIT {
183-
// There are two conditions for backtracking:
184-
//
185-
// 1: Not enough value to make it to target.
186-
// This condition happens before reaching a leaf node.
187-
// Looking for a leaf node condition should not make a difference.
188-
// This backtrack removes more than one node and instead starts
189-
// the exploration of a new subtree.
190-
//
191-
// From:
192-
// o
193-
// / \
194-
// 4
195-
// \
196-
// 3
197-
// \
198-
// 2
199-
// /
200-
// 1
201-
// To:
202-
// o
203-
// \
204-
// 4
205-
// /
206-
// 3
207-
//
208-
// 2: value meets or exceeded target.
209-
// In this condition, we only backtrack one node
210-
//
211-
// From:
212-
// o
213-
// /
214-
// 4
215-
// /
216-
// 3
217-
//
218-
// To:
219-
// o
220-
// /
221-
// 4
222-
// \
223-
// 3
224-
// Set initial loop state
225182
backtrack = false;
226-
backtrack_subtree = false;
227-
228183
// * not enough value to make it to the target.
229-
// Therefore, explore a new new subtree.
230184
//
231185
// unchecked_add is used here for performance. Before entering the search loop, all
232186
// utxos are summed and checked for overflow. Since there was no overflow then, any
233187
// subset of addition will not overflow.
234188
if available_value.unchecked_add(value) < target {
235-
backtrack_subtree = true;
189+
backtrack = true;
236190
}
237191
// This optimization provides an upper bound on the amount of waste that is acceptable.
238192
// Since value is lost when we create a change output due to increasing the size of the
@@ -273,47 +227,28 @@ pub fn select_coins_bnb(
273227

274228
current_waste = current_waste.checked_sub(waste)?;
275229
}
276-
277230
// * Backtrack one node
278231
if backtrack {
279-
let last_index = index_selection.pop().unwrap();
280-
let (eff_value, utxo_waste, _) = w_utxos[last_index];
281-
282-
current_waste = current_waste.checked_sub(utxo_waste)?;
283-
value = value.checked_sub(eff_value)?;
284-
index -= 1;
285-
assert_eq!(index, last_index);
286-
}
287-
// * Backtrack to new tree
288-
else if backtrack_subtree {
289-
// No new subtree left to explore.
290232
if index_selection.is_empty() {
291233
return index_to_utxo_list(best_selection, w_utxos);
292234
}
293235

294-
// Anchor the new subtree at the next available index.
295-
//
296-
// if our index selection is: [0,1,2,3]
297-
// then copy the head 0 into index.
298-
// At the end of this loop, index is incremented to 1.
299-
// Therefore, we will be starting our next search tree at index 1.
300-
index = index_selection[0];
301-
302-
// Reset waste counter since we are starting a new search branch.
303-
current_waste = SignedAmount::ZERO;
236+
loop {
237+
index -= 1;
304238

305-
// The available value of the next iteration. This should never overflow
306-
// since the value is always less than the last available_value calculation.
307-
available_value = w_utxos[index + 1..].iter().map(|&(v, _, _)| v).sum();
239+
if index <= *index_selection.last().unwrap() {
240+
break;
241+
}
308242

309-
// If the new subtree does not have enough value, we are done searching.
310-
if available_value < target {
311-
return index_to_utxo_list(best_selection, w_utxos);
312-
}
243+
let (eff_value, _, _) = w_utxos[index];
244+
available_value += eff_value;
245+
};
313246

314-
// Start a new selection and add the root of the new subtree to the index selection.
315-
index_selection.clear();
316-
value = Amount::ZERO;
247+
assert_eq!(index, *index_selection.last().unwrap());
248+
let (eff_value, utxo_waste, _) = w_utxos[index];
249+
current_waste = current_waste.checked_sub(utxo_waste)?;
250+
value = value.checked_sub(eff_value)?;
251+
index_selection.pop().unwrap();
317252
}
318253
// * Add next node to the inclusion branch.
319254
else {
@@ -392,7 +327,7 @@ mod tests {
392327
.collect()
393328
}
394329

395-
fn create_weighted_utxos_from_values(fee: Amount, values: Vec<Amount>) -> Vec<WeightedUtxo> {
330+
fn create_weighted_utxos_from_values(values: Vec<Amount>) -> Vec<WeightedUtxo> {
396331
values
397332
.iter()
398333
.map(|amt| WeightedUtxo {
@@ -737,7 +672,31 @@ mod tests {
737672
}
738673

739674
#[test]
740-
fn select_coins_bnb_extended_set() {
675+
fn select_coins_bnb_set_size_five() {
676+
let target = Amount::from_str("6 cBTC").unwrap();
677+
let cost_of_change = Amount::ZERO;
678+
let vals = vec![
679+
Amount::from_str("3 cBTC").unwrap(),
680+
Amount::from_str("2.9 cBTC").unwrap(),
681+
Amount::from_str("2 cBTC").unwrap(),
682+
Amount::from_str("1.9 cBTC").unwrap(),
683+
Amount::from_str("1 cBTC").unwrap(),
684+
];
685+
686+
let weighted_utxos = create_weighted_utxos_from_values(vals);
687+
let list: Vec<_> =
688+
select_coins_bnb(target, cost_of_change, FeeRate::ZERO, FeeRate::ZERO, &weighted_utxos)
689+
.unwrap()
690+
.collect();
691+
692+
assert_eq!(list.len(), 3);
693+
assert_eq!(list[0].utxo.value, Amount::from_str("3 cBTC").unwrap());
694+
assert_eq!(list[1].utxo.value, Amount::from_str("2 cBTC").unwrap());
695+
assert_eq!(list[2].utxo.value, Amount::from_str("1 cBTC").unwrap());
696+
}
697+
698+
#[test]
699+
fn select_coins_bnb_set_size_seven() {
741700
let target = Amount::from_str("18 cBTC").unwrap();
742701
let cost_of_change = Amount::from_str("50 sats").unwrap();
743702
let vals = vec![
@@ -748,17 +707,18 @@ mod tests {
748707
Amount::from_str("3 cBTC").unwrap(),
749708
Amount::from_str("2 cBTC").unwrap(),
750709
Amount::from_str("1 cBTC").unwrap() + Amount::from_str("5 sats").unwrap(),
751-
];
710+
];
752711

753-
let weighted_utxos = create_weighted_utxos_from_values(Amount::ZERO, vals);
712+
let weighted_utxos = create_weighted_utxos_from_values(vals);
754713
let list: Vec<_> =
755714
select_coins_bnb(target, cost_of_change, FeeRate::ZERO, FeeRate::ZERO, &weighted_utxos)
756715
.unwrap()
757716
.collect();
758717

759-
assert_eq!(list.len(), 3);
760-
assert_eq!(list[0].utxo.value, Amount::from_str("10 cBTC").unwrap());
761-
assert_eq!(list[1].utxo.value, Amount::from_str("6 cBTC").unwrap());
762-
assert_eq!(list[2].utxo.value, Amount::from_str("2 cBTC").unwrap());
763-
}
718+
assert_eq!(list.len(), 3);
719+
assert_eq!(list[0].utxo.value, Amount::from_str("10 cBTC").unwrap());
720+
assert_eq!(list[1].utxo.value, Amount::from_str("6 cBTC").unwrap());
721+
assert_eq!(list[2].utxo.value, Amount::from_str("2 cBTC").unwrap());
722+
}
723+
764724
}

0 commit comments

Comments
 (0)
Please sign in to comment.