achow101
commented at 11:09 pm on May 21, 2021:
member
Instead of returning a set of selected coins and their total value as separate items, encapsulate both of these, and other variables, into a new SelectionResult struct. This allows us to have all of the things relevant to a coin selection solution be in a single object. SelectionResult enables us to implement the waste calculation in a cleaner way.
All of the coin selection functions (SelectCoinsBnB, KnapsackSolver, AttemptSelection, and SelectCoins) are changed to use a SelectionResult as the output parameter.
DrahtBot
commented at 3:49 am on May 22, 2021:
member
The following sections might be updated with supplementary metadata relevant to reviewers and maintainers.
Conflicts
Reviewers, this pull request conflicts with the following ones:
#23534 (wallet: Allow negtive effective value inputs when subtracting fee from outputs by achow101)
#23497 (Add src/node/ and src/wallet/ code to node:: and wallet:: namespaces by ryanofsky)
#23475 (wallet: add config to prioritize a solution that doesn’t create change in coin selection by brunoerg)
#23367 (Optimize coin selection by dropping BnB upper limit by S3RK)
#23202 (wallet: allow psbtbumpfee to work with txs with external inputs by achow101)
#23201 (wallet: Allow users to specify input weights when funding a transaction by achow101)
#13226 (Optimize SelectCoinsBnB by tracking the selection by index rather than by position by Empact)
If you consider this pull request important, please also help to review the conflicting pull requests. Ideally, start with the one that should be merged first.
achow101 force-pushed
on May 25, 2021
achow101 force-pushed
on Jun 1, 2021
achow101 force-pushed
on Jun 2, 2021
achow101 force-pushed
on Jun 9, 2021
achow101 force-pushed
on Jul 3, 2021
glozow
commented at 3:18 pm on August 24, 2021:
member
Concept ACK
achow101 force-pushed
on Aug 26, 2021
achow101 force-pushed
on Sep 1, 2021
achow101 marked this as ready for review
on Sep 1, 2021
DrahtBot added the label
Needs rebase
on Sep 3, 2021
in
src/wallet/test/coinselector_tests.cpp:200
in
bf455047a0outdated
Commit 80402a13e52671aa7359e3b87d85098805926a63: This may be more a comment on the commit message, but shouldn’t the result of BnB be deterministic? Are the OutputGroups unsorted, or how is this happening?
achow101
commented at 10:12 pm on September 3, 2021:
I don’t remember the specifics, but I think it had to with different txids being in utxo_pool and actual_selection
in
src/wallet/coinselection.h:198
in
187a9d9285outdated
192@@ -193,6 +193,17 @@ struct SelectionResult
193 * hit a specific target.
194 */
195 CAmount input_fees{0};
196+ /** Whether the input values for calculations should be the the effective value (true) or normal value (false) */
0 /** Whether the input values for calculations should be the effective value (true) or normal value (false) */
achow101
commented at 0:34 am on September 4, 2021:
Fixed
in
src/wallet/coinselection.h:202
in
187a9d9285outdated
192@@ -193,6 +193,17 @@ struct SelectionResult
193 * hit a specific target.
194 */
195 CAmount input_fees{0};
196+ /** Whether the input values for calculations should be the the effective value (true) or normal value (false) */
197+ bool m_use_effective{true};
198+ /** The algorithm used supports making change outputs. */
199+ bool m_make_change{false};
200+ /** The cost of making a change output and spending it in the future. Since this is largely a static parameter independent of the selection, it is not cleared by Clear() */
What do you mean with “independent of the selection”? Do you mean that it’s the same across different algorithms, independent of the inputs that get picked, or smth else? It’s definitely dependent on the selection parameters, especially the feerate.
achow101
commented at 10:16 pm on September 3, 2021:
Independent of the algorithm.
achow101
commented at 0:34 am on September 4, 2021:
Clarified the comment.
in
src/wallet/test/coinselector_tests.cpp:157
in
405e780110outdated
achow101
commented at 0:34 am on September 4, 2021:
Fixed
in
src/wallet/spend.cpp:411
in
10492b60adoutdated
414- continue;
415- nValueRet += out.tx->tx->vout[out.i].nValue;
416- setCoinsRet.insert(out.GetInputCoin());
417+ for (const COutput& out : vCoins) {
418+ if (!out.fSpendable) continue;
419+ /* Set depth, from_me, ancestors, and descendants to 0 or false as don't matter for preset inputs as no actual selection is being done.
achow101
commented at 0:34 am on September 4, 2021:
fixed
Xekyo
commented at 9:43 pm on September 3, 2021:
member
Light review, concept ACK
DrahtBot removed the label
Needs rebase
on Sep 3, 2021
achow101 force-pushed
on Sep 4, 2021
theStack
commented at 2:38 pm on September 8, 2021:
member
Concept ACK
DrahtBot added the label
Needs rebase
on Sep 9, 2021
achow101 force-pushed
on Sep 9, 2021
achow101 force-pushed
on Sep 9, 2021
DrahtBot removed the label
Needs rebase
on Sep 9, 2021
achow101 force-pushed
on Sep 28, 2021
achow101
commented at 11:18 pm on September 28, 2021:
member
Rebased. I’ve also made some additional changes. Notably, instead of using SelectionResult as an out parameter, all of the coin selection functions will return an std::optional<SelectionResult> instead.
achow101 force-pushed
on Sep 29, 2021
laanwj added this to the "Blockers" column in a project
DrahtBot added the label
Needs rebase
on Oct 4, 2021
achow101 force-pushed
on Oct 4, 2021
DrahtBot removed the label
Needs rebase
on Oct 4, 2021
S3RK
commented at 7:53 am on October 7, 2021:
member
Concept ACK, going to review the code
achow101 force-pushed
on Oct 7, 2021
DrahtBot added the label
Needs rebase
on Oct 8, 2021
achow101 force-pushed
on Oct 8, 2021
DrahtBot removed the label
Needs rebase
on Oct 8, 2021
Empact
commented at 12:12 pm on October 13, 2021:
member
In fe4d2eff23f22ec09b71bd4f218a28c66a386f10:
Should this be private? Especially since you have functions to AddInput() and GetInputVector(), nobody should have direct access to this set.
achow101
commented at 10:03 pm on November 23, 2021:
Inserted a commit which makes them private.
in
src/wallet/coinselection.h:216
in
fe4d2eff23outdated
205+ const CAmount m_target;
206+ /** Whether the input values for calculations should be the effective value (true) or normal value (false) */
207+ bool m_use_effective{false};
208+
209+ explicit SelectionResult(const CAmount target)
210+ : m_target(target) {}
This seems strange, since m_use_effective should never really change. IIRC, m_subtract_fee_outputs is known right from the start (CreateTransactionInternal()) right? It seems that SelectionResult::m_use_effective should be a const bool and set in the constructor.
My expectation would be to Assume(group.m_subtract_fee_outputs != m_use_effective). Maybe I’m misunderstanding?
achow101
commented at 6:54 pm on November 23, 2021:
I did it this way because SelectionResults are constructed by the SelectCoins* functions which do not know about m_subtract_fee_outputs and didn’t feel like adding more parameters to those functions just to have m_use_effective be set in the constructor.
in
src/wallet/spend.cpp:525
in
c97801c398outdated
524 // If possible, fund the transaction with confirmed UTXOs only. Prefer at least six
525 // confirmations on outputs received from other wallets and only spend confirmed change.
526- if (AttemptSelection(wallet, value_to_select, CoinEligibilityFilter(1, 6, 0), vCoins, setCoinsRet, nValueRet, coin_selection_params)) return true;
527- if (AttemptSelection(wallet, value_to_select, CoinEligibilityFilter(1, 1, 0), vCoins, setCoinsRet, nValueRet, coin_selection_params)) return true;
528+ std::optional<SelectionResult> r1 = AttemptSelection(wallet, value_to_select, CoinEligibilityFilter(1, 6, 0), vCoins, coin_selection_params);
529+ if (r1) return r1;
I think this is cleaner, and keeps the scope of the result to this if block. There’s no reason why r1’s scope should persist through the end of the lambda.
is there any benefit to using a std::make_optional here?
0 return result;
achow101
commented at 10:03 pm on November 23, 2021:
No, changed to just return result.
in
src/wallet/spend.cpp:514
in
c97801c398outdated
514@@ -515,60 +515,60 @@ bool SelectCoins(const CWallet& wallet, const std::vector<COutput>& vAvailableCo
515 // Coin Selection attempts to select inputs from a pool of eligible UTXOs to fund the
516 // transaction at a target feerate. If an attempt fails, more attempts may be made using a more
517 // permissive CoinEligibilityFilter.
518- const bool res = [&] {
519+ std::optional<SelectionResult> res = [&] {
520 // Pre-selected inputs already cover the target amount.
521- if (value_to_select <= 0) return true;
522+ if (value_to_select <= 0) return std::make_optional(SelectionResult(nTargetValue));
Can this be written into a comparison operator for SelectionResult? The logic of waste and what makes one selection result better than another is inherent to the SelectionResult type; imo it fits better there than within AttemptSelection.
If you have a < operator defined, you can sort and/or just return the min element.
achow101
commented at 10:05 pm on November 23, 2021:
I didn’t want to always be computing waste every time, especially because we may not have access to information such as the cost of change. So the approach I’ve done for this is to have a function which computes and stores the waste inside of SelectionResult, and then operator< will use that when doing the comparison.
glozow
commented at 3:43 pm on November 14, 2021:
member
I really like the interface of returning a std::optional<SelectionResult>. My main suggestion is instead of res = Select(...); if (res) return res; declare it within the if condition like if (auto res{Select(...)}) return *res;.
in
src/wallet/coinselection.cpp:422
in
c97801c398outdated
I think randomizing in a Get function is very unexpected. You wouldn’t expect it to return a different value every time. I’d prefer to either rename this function to e.g. RandomizedInputVector() or, do it at the call site, or do the shuffle somewhere else.
achow101
commented at 10:05 pm on November 23, 2021:
I’ve added GetInputSet to just return m_selected_inputs and changed this to be GetShuffledInputVector.
Maybe move this under the global ::GetSelectionWaste? Currently no problem but if you’d ever want to make it static.
achow101
commented at 10:05 pm on November 23, 2021:
Done
in
src/wallet/test/coinselector_tests.cpp:89
in
c97801c398outdated
83@@ -81,10 +84,30 @@ static void add_coin(std::vector<COutput>& coins, CWallet& wallet, const CAmount
84 coins.push_back(output);
85 }
8687-static bool equal_sets(CoinSet a, CoinSet b)
88+/** Check if SelectionResult a is equivalent to SelectionResult b.
89+ * Equivalent means same input values, but maybe different inputs (i.e. same value, different prevout) */
90+bool EquivalentResult(const SelectionResult& a, const SelectionResult& b)
I don’t think this function is ever used. I think I’d prefer to remove it, in general it seems unnecessary to have a Clear function instead of replacing the structure with a new, empty one.
achow101
commented at 10:06 pm on November 23, 2021:
I had intended to use it in the tests, and with the changes to make m_selected_inputs private, it is now being used by those tests.
Would it be better if, when waste is nullopt, calculate it? The *m_waste would throw an exception if m_waste == std::nullopt, so the Assume doesn’t do much.
achow101
commented at 6:34 pm on December 5, 2021:
The cost of change is not known to SelectionResult so it cannot be calculated.
achow101
commented at 6:50 pm on December 5, 2021:
Done
in
src/wallet/spend.cpp:381
in
0211023c25outdated
377@@ -378,16 +378,14 @@ bool AttemptSelection(const CWallet& wallet, const CAmount& nTargetValue, const
378 {
379 setCoinsRet.clear();
380 nValueRet = 0;
381- // Vector of results for use with waste calculation
382- // In order: calculated waste, selected inputs, selected input value (sum of input values)
383- // TODO: Use a struct representing the selection result
384- std::vector<std::tuple<CAmount, std::set<CInputCoin>, CAmount>> results;
385+ // Vector of result and waste pairs. We will choose the best one based on waste.
Really nice that it’s just SelectionResults instead of tuples now :D but the comment needs to be updated
achow101
commented at 6:50 pm on December 5, 2021:
Done
in
src/wallet/spend.cpp:386
in
0211023c25outdated
388 // Note that unlike KnapsackSolver, we do not include the fee for creating a change output as BnB will not create a change output.
389 std::vector<OutputGroup> positive_groups = GroupOutputs(wallet, coins, coin_selection_params, eligibility_filter, true /* positive_only */);
390 if (auto bnb_result{SelectCoinsBnB(positive_groups, nTargetValue, coin_selection_params.m_cost_of_change)}) {
391 bnb_result->ComputeAndSetWaste(CAmount(0));
392- results.emplace_back(std::make_tuple(bnb_result->GetWaste(), bnb_result->GetInputSet(), bnb_result->GetSelectedValue()));
393+ results.push_back(*bnb_result);
achow101
commented at 6:42 pm on December 5, 2021:
The result is already constructed.
in
src/wallet/spend.cpp:415
in
0211023c25outdated
411@@ -414,11 +412,9 @@ bool AttemptSelection(const CWallet& wallet, const CAmount& nTargetValue, const
412413 // Choose the result with the least waste
414 // If the waste is the same, choose the one which spends more inputs.
415- const auto& best_result = std::min_element(results.begin(), results.end(), [](const auto& a, const auto& b) {
416- return std::get<0>(a) < std::get<0>(b) || (std::get<0>(a) == std::get<0>(b) && std::get<1>(a).size() > std::get<1>(b).size());
417- });
418- setCoinsRet = std::get<1>(*best_result);
419- nValueRet = std::get<2>(*best_result);
420+ auto best_result = *std::min_element(results.begin(), results.end());
In 1b1ad42
Should this function also clear m_waste?
achow101
commented at 6:50 pm on December 5, 2021:
Done
glozow
commented at 4:47 pm on December 2, 2021:
member
utACKe142d9af81, a few suggestions/questions
Introduce SelectionResult struct
Introduces a SelectionResult struct which contains the set of selected
inputs and the total transaction fee for the transaction. This will be
used by the various SelectCoins* functions. Additionally helpers are
provided to compute the total input value and result comparisons.
9d1d86da04
scripted-diff: Use SelectionResult in coin selector tests
Replace the CoinSet actual_selection with a SelectionResult
expected_result. We don't use the SelectionResult functions yet, but
will soon.
-BEGIN VERIFY SCRIPT-
sed -i 's/CoinSet actual_selection/SelectionResult expected_result(CAmount(0))/' src/wallet/test/coinselector_tests.cpp
sed -i 's/actual_selection/expected_result.m_selected_inputs/' src/wallet/test/coinselector_tests.cpp
sed -i 's/expected_result.m_selected_inputs.clear/expected_result.Clear/' src/wallet/test/coinselector_tests.cpp
-END VERIFY SCRIPT-
cbf0b9f4ff
Make member variables of SelectionResult privatea339add471
Return SelectionResult from SelectCoinsBnB
Removes coins_out and value_ret has SelectCoinsBnB return a
std::optional<SelectionResult>
60d2ca72e3
Return SelectionResult from KnapsackSolver
Returns a std::optional<SelectionResult> from KnapsackSolver instead of
using out parameters for the inputs set and selected value.
0ef6184575
Return SelectionResult from SelectCoinsSRD
Changes SelectCoinsSRD to return a SelectionResult.
51a9c00b4d
Make an OutputGroup for preset inputs
In SelectCoins, for our preset inputs, we combine all of the preset
inputs into a single OutputGroup. This allows us to combine the preset
inputs with additional selection algo results.
e8f7ae5eb3
Use SelectionResult for waste calculationbb50850a44
Use SelectionResult in AttemptSelection
Replace setCoinsRet and nValueRet with a SelectionResult in
AttemptSelection
9d9b101d20
Use SelectionResult in SelectCoins
Replace setCoinsRet and nValueRet with SelectionResult
05300c1439
achow101 force-pushed
on Dec 5, 2021
laanwj
commented at 4:20 pm on December 9, 2021:
member
426+bool SelectionResult::operator<(SelectionResult other) const
427+{
428+ Assume(m_waste != std::nullopt);
429+ Assume(other.m_waste != std::nullopt);
430+ // As this operator is only used in std::min_element, we want the result that has more inputs when waste are equal.
431+ return *m_waste < *other.m_waste || (*m_waste == *other.m_waste && m_selected_inputs.size() > other.m_selected_inputs.size());
MarcoFalke
commented at 5:07 pm on December 9, 2021:
same
laanwj removed this from the "Blockers" column in a project
MarcoFalke
commented at 5:16 pm on December 9, 2021:
member
left some q
RandyMcMillan referenced this in commit
caf667a3e1
on Dec 23, 2021
This is a metadata mirror of the GitHub repository
bitcoin/bitcoin.
This site is not affiliated with GitHub.
Content is generated from a GitHub metadata backup.
generated: 2025-04-06 15:12 UTC
This site is hosted by @0xB10C More mirrored repositories can be found on mirror.b10c.me