2011-08-09 13:27:58 +02:00
// Copyright (c) 2009-2010 Satoshi Nakamoto
2020-01-15 02:17:38 +07:00
// Copyright (c) 2009-2020 The Bitcoin Core developers
2014-10-26 15:03:12 +08:00
// Distributed under the MIT software license, see the accompanying
2012-05-18 22:02:28 +08:00
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
2011-06-26 19:23:24 +02:00
2017-11-10 13:57:53 +13:00
# include <wallet/wallet.h>
2013-04-13 00:13:08 -05:00
2017-11-10 13:57:53 +13:00
# include <chain.h>
# include <consensus/consensus.h>
# include <consensus/validation.h>
# include <fs.h>
2017-07-26 10:23:01 -04:00
# include <interfaces/chain.h>
2018-04-08 14:37:50 -04:00
# include <interfaces/wallet.h>
2017-11-10 13:57:53 +13:00
# include <key.h>
2017-09-19 18:12:25 -07:00
# include <key_io.h>
2020-02-01 23:07:19 +02:00
# include <optional.h>
2017-11-10 13:57:53 +13:00
# include <policy/fees.h>
# include <policy/policy.h>
# include <primitives/block.h>
# include <primitives/transaction.h>
2018-11-10 12:29:07 -08:00
# include <script/descriptor.h>
2017-11-10 13:57:53 +13:00
# include <script/script.h>
2019-06-06 22:52:24 +02:00
# include <script/signingprovider.h>
2018-11-06 09:23:43 -05:00
# include <util/bip32.h>
2020-02-24 14:34:17 -05:00
# include <util/check.h>
2019-04-02 17:03:37 -04:00
# include <util/error.h>
# include <util/fees.h>
2018-10-22 15:51:11 -07:00
# include <util/moneystr.h>
2019-04-02 17:03:37 -04:00
# include <util/rbf.h>
2019-06-17 10:56:52 +03:00
# include <util/translation.h>
2019-04-17 13:27:02 -04:00
# include <wallet/coincontrol.h>
2017-11-10 13:57:53 +13:00
# include <wallet/fees.h>
2013-04-13 00:13:08 -05:00
2018-04-17 18:22:23 +01:00
# include <algorithm>
2014-10-01 08:50:24 +02:00
# include <assert.h>
2012-11-03 09:58:41 -05:00
# include <boost/algorithm/string/replace.hpp>
2011-06-26 19:23:24 +02:00
2020-02-24 14:34:17 -05:00
using interfaces : : FoundBlock ;
2018-07-25 16:24:55 +09:00
const std : : map < uint64_t , std : : string > WALLET_FLAG_CAVEATS {
{ WALLET_FLAG_AVOID_REUSE ,
" You need to rescan the blockchain in order to correctly mark used "
" destinations in the past. Until this is done, some destinations may "
" be considered unused, even if the opposite is the case. "
} ,
} ;
2018-07-31 01:50:43 +09:00
static const size_t OUTPUT_GROUP_MAX_ENTRIES = 10 ;
2020-01-07 23:14:15 +07:00
static RecursiveMutex cs_wallets ;
2018-05-22 16:18:07 +01:00
static std : : vector < std : : shared_ptr < CWallet > > vpwallets GUARDED_BY ( cs_wallets ) ;
2019-09-27 07:31:44 -04:00
static std : : list < LoadWalletFn > g_load_wallet_fns GUARDED_BY ( cs_wallets ) ;
2018-04-17 18:22:23 +01:00
2018-05-22 16:18:07 +01:00
bool AddWallet ( const std : : shared_ptr < CWallet > & wallet )
2018-04-17 18:22:23 +01:00
{
2018-04-19 14:44:34 +01:00
LOCK ( cs_wallets ) ;
2018-04-17 18:22:23 +01:00
assert ( wallet ) ;
2018-05-22 16:18:07 +01:00
std : : vector < std : : shared_ptr < CWallet > > : : const_iterator i = std : : find ( vpwallets . begin ( ) , vpwallets . end ( ) , wallet ) ;
2018-04-17 18:22:23 +01:00
if ( i ! = vpwallets . end ( ) ) return false ;
vpwallets . push_back ( wallet ) ;
2019-10-07 14:11:34 -04:00
wallet - > ConnectScriptPubKeyManNotifiers ( ) ;
2018-04-17 18:22:23 +01:00
return true ;
}
2018-05-22 16:18:07 +01:00
bool RemoveWallet ( const std : : shared_ptr < CWallet > & wallet )
2018-04-17 18:22:23 +01:00
{
assert ( wallet ) ;
2020-03-10 15:46:20 -04:00
// Unregister with the validation interface which also drops shared ponters.
wallet - > m_chain_notifications_handler . reset ( ) ;
LOCK ( cs_wallets ) ;
2018-05-22 16:18:07 +01:00
std : : vector < std : : shared_ptr < CWallet > > : : iterator i = std : : find ( vpwallets . begin ( ) , vpwallets . end ( ) , wallet ) ;
2018-04-17 18:22:23 +01:00
if ( i = = vpwallets . end ( ) ) return false ;
vpwallets . erase ( i ) ;
return true ;
}
2018-04-18 13:46:11 +01:00
bool HasWallets ( )
{
2018-04-19 14:44:34 +01:00
LOCK ( cs_wallets ) ;
2018-04-18 13:46:11 +01:00
return ! vpwallets . empty ( ) ;
}
2018-05-22 16:18:07 +01:00
std : : vector < std : : shared_ptr < CWallet > > GetWallets ( )
2018-04-17 18:22:23 +01:00
{
2018-04-19 14:44:34 +01:00
LOCK ( cs_wallets ) ;
2018-04-17 18:22:23 +01:00
return vpwallets ;
}
2018-05-22 16:18:07 +01:00
std : : shared_ptr < CWallet > GetWallet ( const std : : string & name )
2018-04-17 18:22:23 +01:00
{
2018-04-19 14:44:34 +01:00
LOCK ( cs_wallets ) ;
2018-05-22 16:18:07 +01:00
for ( const std : : shared_ptr < CWallet > & wallet : vpwallets ) {
2018-04-17 18:22:23 +01:00
if ( wallet - > GetName ( ) = = name ) return wallet ;
}
return nullptr ;
}
2019-09-27 07:31:44 -04:00
std : : unique_ptr < interfaces : : Handler > HandleLoadWallet ( LoadWalletFn load_wallet )
{
LOCK ( cs_wallets ) ;
auto it = g_load_wallet_fns . emplace ( g_load_wallet_fns . end ( ) , std : : move ( load_wallet ) ) ;
return interfaces : : MakeHandler ( [ it ] { LOCK ( cs_wallets ) ; g_load_wallet_fns . erase ( it ) ; } ) ;
}
2018-12-12 23:21:19 +00:00
static Mutex g_wallet_release_mutex ;
static std : : condition_variable g_wallet_release_cv ;
2019-08-25 02:07:04 +01:00
static std : : set < std : : string > g_unloading_wallet_set ;
2018-12-12 23:21:19 +00:00
2018-04-28 22:36:43 +01:00
// Custom deleter for shared_ptr<CWallet>.
static void ReleaseWallet ( CWallet * wallet )
{
2019-08-25 02:07:04 +01:00
const std : : string name = wallet - > GetName ( ) ;
2018-06-15 19:02:52 -04:00
wallet - > WalletLogPrintf ( " Releasing wallet \n " ) ;
2018-04-28 22:36:43 +01:00
wallet - > Flush ( ) ;
delete wallet ;
2018-12-12 23:21:19 +00:00
// Wallet is now released, notify UnloadWallet, if any.
{
LOCK ( g_wallet_release_mutex ) ;
2019-08-25 02:07:04 +01:00
if ( g_unloading_wallet_set . erase ( name ) = = 0 ) {
2018-12-12 23:21:19 +00:00
// UnloadWallet was not called for this wallet, all done.
return ;
}
}
g_wallet_release_cv . notify_all ( ) ;
}
void UnloadWallet ( std : : shared_ptr < CWallet > & & wallet )
{
// Mark wallet for unloading.
2019-08-25 02:07:04 +01:00
const std : : string name = wallet - > GetName ( ) ;
2018-12-12 23:21:19 +00:00
{
LOCK ( g_wallet_release_mutex ) ;
2019-08-25 02:07:04 +01:00
auto it = g_unloading_wallet_set . insert ( name ) ;
2018-12-12 23:21:19 +00:00
assert ( it . second ) ;
}
// The wallet can be in use so it's not possible to explicitly unload here.
// Notify the unload intent so that all remaining shared pointers are
// released.
2019-08-25 02:07:04 +01:00
wallet - > NotifyUnload ( ) ;
2020-03-10 15:46:20 -04:00
2018-12-12 23:21:19 +00:00
// Time to ditch our shared_ptr and wait for ReleaseWallet call.
wallet . reset ( ) ;
{
WAIT_LOCK ( g_wallet_release_mutex , lock ) ;
2019-08-25 02:07:04 +01:00
while ( g_unloading_wallet_set . count ( name ) = = 1 ) {
2018-12-12 23:21:19 +00:00
g_wallet_release_cv . wait ( lock ) ;
}
}
2018-04-28 22:36:43 +01:00
}
2019-10-06 17:52:05 -04:00
std : : shared_ptr < CWallet > LoadWallet ( interfaces : : Chain & chain , const WalletLocation & location , std : : string & error , std : : vector < std : : string > & warnings )
2019-01-12 11:47:04 +00:00
{
2020-03-28 02:14:08 +00:00
try {
if ( ! CWallet : : Verify ( chain , location , false , error , warnings ) ) {
error = " Wallet file verification failed: " + error ;
return nullptr ;
}
2019-01-12 11:47:04 +00:00
2020-03-28 02:14:08 +00:00
std : : shared_ptr < CWallet > wallet = CWallet : : CreateWalletFromFile ( chain , location , error , warnings ) ;
if ( ! wallet ) {
error = " Wallet loading failed: " + error ;
return nullptr ;
}
AddWallet ( wallet ) ;
wallet - > postInitProcess ( ) ;
return wallet ;
} catch ( const std : : runtime_error & e ) {
error = e . what ( ) ;
2019-01-12 11:47:04 +00:00
return nullptr ;
}
}
2019-10-06 17:52:05 -04:00
std : : shared_ptr < CWallet > LoadWallet ( interfaces : : Chain & chain , const std : : string & name , std : : string & error , std : : vector < std : : string > & warnings )
2019-01-12 11:47:04 +00:00
{
2019-10-06 17:52:05 -04:00
return LoadWallet ( chain , WalletLocation ( name ) , error , warnings ) ;
2019-01-12 11:47:04 +00:00
}
2019-10-06 17:52:05 -04:00
WalletCreationStatus CreateWallet ( interfaces : : Chain & chain , const SecureString & passphrase , uint64_t wallet_creation_flags , const std : : string & name , std : : string & error , std : : vector < std : : string > & warnings , std : : shared_ptr < CWallet > & result )
2019-05-24 17:13:13 -04:00
{
// Indicate that the wallet is actually supposed to be blank and not just blank to make it encrypted
bool create_blank = ( wallet_creation_flags & WALLET_FLAG_BLANK_WALLET ) ;
// Born encrypted wallets need to be created blank first.
if ( ! passphrase . empty ( ) ) {
wallet_creation_flags | = WALLET_FLAG_BLANK_WALLET ;
}
// Check the wallet file location
WalletLocation location ( name ) ;
if ( location . Exists ( ) ) {
error = " Wallet " + location . GetName ( ) + " already exists. " ;
2019-07-10 17:51:39 -04:00
return WalletCreationStatus : : CREATION_FAILED ;
2019-05-24 17:13:13 -04:00
}
// Wallet::Verify will check if we're trying to create a wallet with a duplicate name.
2019-10-06 17:52:05 -04:00
if ( ! CWallet : : Verify ( chain , location , false , error , warnings ) ) {
error = " Wallet file verification failed: " + error ;
2019-07-10 17:51:39 -04:00
return WalletCreationStatus : : CREATION_FAILED ;
}
// Do not allow a passphrase when private keys are disabled
if ( ! passphrase . empty ( ) & & ( wallet_creation_flags & WALLET_FLAG_DISABLE_PRIVATE_KEYS ) ) {
error = " Passphrase provided but private keys are disabled. A passphrase is only used to encrypt private keys, so cannot be used for wallets with private keys disabled. " ;
return WalletCreationStatus : : CREATION_FAILED ;
2019-05-24 17:13:13 -04:00
}
// Make the wallet
2019-10-06 17:52:05 -04:00
std : : shared_ptr < CWallet > wallet = CWallet : : CreateWalletFromFile ( chain , location , error , warnings , wallet_creation_flags ) ;
2019-05-24 17:13:13 -04:00
if ( ! wallet ) {
2019-10-06 17:52:05 -04:00
error = " Wallet creation failed: " + error ;
2019-07-10 17:51:39 -04:00
return WalletCreationStatus : : CREATION_FAILED ;
2019-05-24 17:13:13 -04:00
}
// Encrypt the wallet
if ( ! passphrase . empty ( ) & & ! ( wallet_creation_flags & WALLET_FLAG_DISABLE_PRIVATE_KEYS ) ) {
if ( ! wallet - > EncryptWallet ( passphrase ) ) {
error = " Error: Wallet created but failed to encrypt. " ;
2019-07-10 17:51:39 -04:00
return WalletCreationStatus : : ENCRYPTION_FAILED ;
2019-05-24 17:13:13 -04:00
}
if ( ! create_blank ) {
// Unlock the wallet
if ( ! wallet - > Unlock ( passphrase ) ) {
error = " Error: Wallet was encrypted but could not be unlocked " ;
2019-07-10 17:51:39 -04:00
return WalletCreationStatus : : ENCRYPTION_FAILED ;
2019-05-24 17:13:13 -04:00
}
// Set a seed for the wallet
2019-10-07 14:11:34 -04:00
{
2019-10-07 14:11:34 -04:00
LOCK ( wallet - > cs_wallet ) ;
2019-10-07 14:11:34 -04:00
for ( auto spk_man : wallet - > GetActiveScriptPubKeyMans ( ) ) {
2019-10-07 14:11:34 -04:00
if ( ! spk_man - > SetupGeneration ( ) ) {
error = " Unable to generate initial keys " ;
return WalletCreationStatus : : CREATION_FAILED ;
}
}
}
2019-05-24 17:13:13 -04:00
// Relock the wallet
wallet - > Lock ( ) ;
}
}
AddWallet ( wallet ) ;
wallet - > postInitProcess ( ) ;
2019-07-10 17:51:39 -04:00
result = wallet ;
return WalletCreationStatus : : SUCCESS ;
2019-05-24 17:13:13 -04:00
}
2019-12-30 14:05:27 -05:00
const uint256 CWalletTx : : ABANDON_HASH ( UINT256_ONE ( ) ) ;
2016-01-07 16:31:27 -05:00
2014-10-26 15:03:12 +08:00
/** @defgroup mapWallet
*
* @ {
*/
2011-06-26 19:23:24 +02:00
Split up util.cpp/h
Split up util.cpp/h into:
- string utilities (hex, base32, base64): no internal dependencies, no dependency on boost (apart from foreach)
- money utilities (parsesmoney, formatmoney)
- time utilities (gettime*, sleep, format date):
- and the rest (logging, argument parsing, config file parsing)
The latter is basically the environment and OS handling,
and is stripped of all utility functions, so we may want to
rename it to something else than util.cpp/h for clarity (Matt suggested
osinterface).
Breaks dependency of sha256.cpp on all the things pulled in by util.
2014-08-21 16:11:09 +02:00
std : : string COutput : : ToString ( ) const
{
2016-11-11 16:54:51 -08:00
return strprintf ( " COutput(%s, %d, %d) [ % s ] " , tx->GetHash().ToString(), i, nDepth, FormatMoney(tx->tx->vout[i].nValue)) ;
Split up util.cpp/h
Split up util.cpp/h into:
- string utilities (hex, base32, base64): no internal dependencies, no dependency on boost (apart from foreach)
- money utilities (parsesmoney, formatmoney)
- time utilities (gettime*, sleep, format date):
- and the rest (logging, argument parsing, config file parsing)
The latter is basically the environment and OS handling,
and is stripped of all utility functions, so we may want to
rename it to something else than util.cpp/h for clarity (Matt suggested
osinterface).
Breaks dependency of sha256.cpp on all the things pulled in by util.
2014-08-21 16:11:09 +02:00
}
2014-02-15 16:38:28 -05:00
const CWalletTx * CWallet : : GetWalletTx ( const uint256 & hash ) const
{
LOCK ( cs_wallet ) ;
std : : map < uint256 , CWalletTx > : : const_iterator it = mapWallet . find ( hash ) ;
if ( it = = mapWallet . end ( ) )
2017-08-07 07:36:37 +02:00
return nullptr ;
2014-02-15 16:38:28 -05:00
return & ( it - > second ) ;
}
2019-10-07 14:11:34 -04:00
void CWallet : : UpgradeKeyMetadata ( )
{
2019-10-07 14:11:34 -04:00
if ( IsLocked ( ) | | IsWalletFlagSet ( WALLET_FLAG_KEY_ORIGIN_METADATA ) ) {
return ;
}
2019-11-04 11:00:26 -05:00
auto spk_man = GetLegacyScriptPubKeyMan ( ) ;
if ( ! spk_man ) {
return ;
2019-10-07 14:11:34 -04:00
}
2019-11-04 11:00:26 -05:00
spk_man - > UpgradeKeyMetadata ( ) ;
2019-10-07 14:11:34 -04:00
SetWalletFlag ( WALLET_FLAG_KEY_ORIGIN_METADATA ) ;
2019-10-07 14:11:34 -04:00
}
2016-09-16 16:45:36 +02:00
bool CWallet : : Unlock ( const SecureString & strWalletPassphrase , bool accept_no_keys )
Add wallet privkey encryption.
This commit adds support for ckeys, or enCrypted private keys, to the wallet.
All keys are stored in memory in their encrypted form and thus the passphrase
is required from the user to spend coins, or to create new addresses.
Keys are encrypted with AES-256-CBC using OpenSSL's EVP library. The key is
calculated via EVP_BytesToKey using SHA512 with (by default) 25000 rounds and
a random salt.
By default, the user's wallet remains unencrypted until they call the RPC
command encryptwallet <passphrase> or, from the GUI menu, Options->
Encrypt Wallet.
When the user is attempting to call RPC functions which require the password
to unlock the wallet, an error will be returned unless they call
walletpassphrase <passphrase> <time to keep key in memory> first.
A keypoolrefill command has been added which tops up the users keypool
(requiring the passphrase via walletpassphrase first).
keypoolsize has been added to the output of getinfo to show the user the
number of keys left before they need to specify their passphrase (and call
keypoolrefill).
Note that walletpassphrase will automatically fill keypool in a separate
thread which it spawns when the passphrase is set. This could cause some
delays in other threads waiting for locks on the wallet passphrase, including
one which could cause the passphrase to be stored longer than expected,
however it will not allow the passphrase to be used longer than expected as
ThreadCleanWalletPassphrase will attempt to get a lock on the key as soon
as the specified lock time has arrived.
When the keypool runs out (and wallet is locked) GetOrReuseKeyFromPool
returns vchDefaultKey, meaning miners may start to generate many blocks to
vchDefaultKey instead of a new key each time.
A walletpassphrasechange <oldpassphrase> <newpassphrase> has been added to
allow the user to change their password via RPC.
Whenever keying material (unencrypted private keys, the user's passphrase,
the wallet's AES key) is stored unencrypted in memory, any reasonable attempt
is made to mlock/VirtualLock that memory before storing the keying material.
This is not true in several (commented) cases where mlock/VirtualLocking the
memory is not possible.
Although encryption of private keys in memory can be very useful on desktop
systems (as some small amount of protection against stupid viruses), on an
RPC server, the password is entered fairly insecurely. Thus, the only main
advantage encryption has for RPC servers is for RPC servers that do not spend
coins, except in rare cases, eg. a webserver of a merchant which only receives
payment except for cases of manual intervention.
Thanks to jgarzik for the original patch and sipa, gmaxwell and many others
for all their input.
Conflicts:
src/wallet.cpp
2011-07-08 15:47:35 +02:00
{
2011-08-26 14:37:23 -04:00
CCrypter crypter ;
2016-11-10 08:00:05 +01:00
CKeyingMaterial _vMasterKey ;
Add wallet privkey encryption.
This commit adds support for ckeys, or enCrypted private keys, to the wallet.
All keys are stored in memory in their encrypted form and thus the passphrase
is required from the user to spend coins, or to create new addresses.
Keys are encrypted with AES-256-CBC using OpenSSL's EVP library. The key is
calculated via EVP_BytesToKey using SHA512 with (by default) 25000 rounds and
a random salt.
By default, the user's wallet remains unencrypted until they call the RPC
command encryptwallet <passphrase> or, from the GUI menu, Options->
Encrypt Wallet.
When the user is attempting to call RPC functions which require the password
to unlock the wallet, an error will be returned unless they call
walletpassphrase <passphrase> <time to keep key in memory> first.
A keypoolrefill command has been added which tops up the users keypool
(requiring the passphrase via walletpassphrase first).
keypoolsize has been added to the output of getinfo to show the user the
number of keys left before they need to specify their passphrase (and call
keypoolrefill).
Note that walletpassphrase will automatically fill keypool in a separate
thread which it spawns when the passphrase is set. This could cause some
delays in other threads waiting for locks on the wallet passphrase, including
one which could cause the passphrase to be stored longer than expected,
however it will not allow the passphrase to be used longer than expected as
ThreadCleanWalletPassphrase will attempt to get a lock on the key as soon
as the specified lock time has arrived.
When the keypool runs out (and wallet is locked) GetOrReuseKeyFromPool
returns vchDefaultKey, meaning miners may start to generate many blocks to
vchDefaultKey instead of a new key each time.
A walletpassphrasechange <oldpassphrase> <newpassphrase> has been added to
allow the user to change their password via RPC.
Whenever keying material (unencrypted private keys, the user's passphrase,
the wallet's AES key) is stored unencrypted in memory, any reasonable attempt
is made to mlock/VirtualLock that memory before storing the keying material.
This is not true in several (commented) cases where mlock/VirtualLocking the
memory is not possible.
Although encryption of private keys in memory can be very useful on desktop
systems (as some small amount of protection against stupid viruses), on an
RPC server, the password is entered fairly insecurely. Thus, the only main
advantage encryption has for RPC servers is for RPC servers that do not spend
coins, except in rare cases, eg. a webserver of a merchant which only receives
payment except for cases of manual intervention.
Thanks to jgarzik for the original patch and sipa, gmaxwell and many others
for all their input.
Conflicts:
src/wallet.cpp
2011-07-08 15:47:35 +02:00
2012-04-06 18:39:12 +02:00
{
LOCK ( cs_wallet ) ;
2017-06-02 03:18:57 +02:00
for ( const MasterKeyMap : : value_type & pMasterKey : mapMasterKeys )
Add wallet privkey encryption.
This commit adds support for ckeys, or enCrypted private keys, to the wallet.
All keys are stored in memory in their encrypted form and thus the passphrase
is required from the user to spend coins, or to create new addresses.
Keys are encrypted with AES-256-CBC using OpenSSL's EVP library. The key is
calculated via EVP_BytesToKey using SHA512 with (by default) 25000 rounds and
a random salt.
By default, the user's wallet remains unencrypted until they call the RPC
command encryptwallet <passphrase> or, from the GUI menu, Options->
Encrypt Wallet.
When the user is attempting to call RPC functions which require the password
to unlock the wallet, an error will be returned unless they call
walletpassphrase <passphrase> <time to keep key in memory> first.
A keypoolrefill command has been added which tops up the users keypool
(requiring the passphrase via walletpassphrase first).
keypoolsize has been added to the output of getinfo to show the user the
number of keys left before they need to specify their passphrase (and call
keypoolrefill).
Note that walletpassphrase will automatically fill keypool in a separate
thread which it spawns when the passphrase is set. This could cause some
delays in other threads waiting for locks on the wallet passphrase, including
one which could cause the passphrase to be stored longer than expected,
however it will not allow the passphrase to be used longer than expected as
ThreadCleanWalletPassphrase will attempt to get a lock on the key as soon
as the specified lock time has arrived.
When the keypool runs out (and wallet is locked) GetOrReuseKeyFromPool
returns vchDefaultKey, meaning miners may start to generate many blocks to
vchDefaultKey instead of a new key each time.
A walletpassphrasechange <oldpassphrase> <newpassphrase> has been added to
allow the user to change their password via RPC.
Whenever keying material (unencrypted private keys, the user's passphrase,
the wallet's AES key) is stored unencrypted in memory, any reasonable attempt
is made to mlock/VirtualLock that memory before storing the keying material.
This is not true in several (commented) cases where mlock/VirtualLocking the
memory is not possible.
Although encryption of private keys in memory can be very useful on desktop
systems (as some small amount of protection against stupid viruses), on an
RPC server, the password is entered fairly insecurely. Thus, the only main
advantage encryption has for RPC servers is for RPC servers that do not spend
coins, except in rare cases, eg. a webserver of a merchant which only receives
payment except for cases of manual intervention.
Thanks to jgarzik for the original patch and sipa, gmaxwell and many others
for all their input.
Conflicts:
src/wallet.cpp
2011-07-08 15:47:35 +02:00
{
if ( ! crypter . SetKeyFromPassphrase ( strWalletPassphrase , pMasterKey . second . vchSalt , pMasterKey . second . nDeriveIterations , pMasterKey . second . nDerivationMethod ) )
return false ;
2016-11-10 08:00:05 +01:00
if ( ! crypter . Decrypt ( pMasterKey . second . vchCryptedKey , _vMasterKey ) )
2013-05-07 10:47:00 -04:00
continue ; // try another master key
2019-06-06 23:58:21 +02:00
if ( Unlock ( _vMasterKey , accept_no_keys ) ) {
2018-11-06 09:23:37 -05:00
// Now that we've unlocked, upgrade the key metadata
UpgradeKeyMetadata ( ) ;
Add wallet privkey encryption.
This commit adds support for ckeys, or enCrypted private keys, to the wallet.
All keys are stored in memory in their encrypted form and thus the passphrase
is required from the user to spend coins, or to create new addresses.
Keys are encrypted with AES-256-CBC using OpenSSL's EVP library. The key is
calculated via EVP_BytesToKey using SHA512 with (by default) 25000 rounds and
a random salt.
By default, the user's wallet remains unencrypted until they call the RPC
command encryptwallet <passphrase> or, from the GUI menu, Options->
Encrypt Wallet.
When the user is attempting to call RPC functions which require the password
to unlock the wallet, an error will be returned unless they call
walletpassphrase <passphrase> <time to keep key in memory> first.
A keypoolrefill command has been added which tops up the users keypool
(requiring the passphrase via walletpassphrase first).
keypoolsize has been added to the output of getinfo to show the user the
number of keys left before they need to specify their passphrase (and call
keypoolrefill).
Note that walletpassphrase will automatically fill keypool in a separate
thread which it spawns when the passphrase is set. This could cause some
delays in other threads waiting for locks on the wallet passphrase, including
one which could cause the passphrase to be stored longer than expected,
however it will not allow the passphrase to be used longer than expected as
ThreadCleanWalletPassphrase will attempt to get a lock on the key as soon
as the specified lock time has arrived.
When the keypool runs out (and wallet is locked) GetOrReuseKeyFromPool
returns vchDefaultKey, meaning miners may start to generate many blocks to
vchDefaultKey instead of a new key each time.
A walletpassphrasechange <oldpassphrase> <newpassphrase> has been added to
allow the user to change their password via RPC.
Whenever keying material (unencrypted private keys, the user's passphrase,
the wallet's AES key) is stored unencrypted in memory, any reasonable attempt
is made to mlock/VirtualLock that memory before storing the keying material.
This is not true in several (commented) cases where mlock/VirtualLocking the
memory is not possible.
Although encryption of private keys in memory can be very useful on desktop
systems (as some small amount of protection against stupid viruses), on an
RPC server, the password is entered fairly insecurely. Thus, the only main
advantage encryption has for RPC servers is for RPC servers that do not spend
coins, except in rare cases, eg. a webserver of a merchant which only receives
payment except for cases of manual intervention.
Thanks to jgarzik for the original patch and sipa, gmaxwell and many others
for all their input.
Conflicts:
src/wallet.cpp
2011-07-08 15:47:35 +02:00
return true ;
2018-11-06 09:23:37 -05:00
}
Add wallet privkey encryption.
This commit adds support for ckeys, or enCrypted private keys, to the wallet.
All keys are stored in memory in their encrypted form and thus the passphrase
is required from the user to spend coins, or to create new addresses.
Keys are encrypted with AES-256-CBC using OpenSSL's EVP library. The key is
calculated via EVP_BytesToKey using SHA512 with (by default) 25000 rounds and
a random salt.
By default, the user's wallet remains unencrypted until they call the RPC
command encryptwallet <passphrase> or, from the GUI menu, Options->
Encrypt Wallet.
When the user is attempting to call RPC functions which require the password
to unlock the wallet, an error will be returned unless they call
walletpassphrase <passphrase> <time to keep key in memory> first.
A keypoolrefill command has been added which tops up the users keypool
(requiring the passphrase via walletpassphrase first).
keypoolsize has been added to the output of getinfo to show the user the
number of keys left before they need to specify their passphrase (and call
keypoolrefill).
Note that walletpassphrase will automatically fill keypool in a separate
thread which it spawns when the passphrase is set. This could cause some
delays in other threads waiting for locks on the wallet passphrase, including
one which could cause the passphrase to be stored longer than expected,
however it will not allow the passphrase to be used longer than expected as
ThreadCleanWalletPassphrase will attempt to get a lock on the key as soon
as the specified lock time has arrived.
When the keypool runs out (and wallet is locked) GetOrReuseKeyFromPool
returns vchDefaultKey, meaning miners may start to generate many blocks to
vchDefaultKey instead of a new key each time.
A walletpassphrasechange <oldpassphrase> <newpassphrase> has been added to
allow the user to change their password via RPC.
Whenever keying material (unencrypted private keys, the user's passphrase,
the wallet's AES key) is stored unencrypted in memory, any reasonable attempt
is made to mlock/VirtualLock that memory before storing the keying material.
This is not true in several (commented) cases where mlock/VirtualLocking the
memory is not possible.
Although encryption of private keys in memory can be very useful on desktop
systems (as some small amount of protection against stupid viruses), on an
RPC server, the password is entered fairly insecurely. Thus, the only main
advantage encryption has for RPC servers is for RPC servers that do not spend
coins, except in rare cases, eg. a webserver of a merchant which only receives
payment except for cases of manual intervention.
Thanks to jgarzik for the original patch and sipa, gmaxwell and many others
for all their input.
Conflicts:
src/wallet.cpp
2011-07-08 15:47:35 +02:00
}
2012-04-06 18:39:12 +02:00
}
Add wallet privkey encryption.
This commit adds support for ckeys, or enCrypted private keys, to the wallet.
All keys are stored in memory in their encrypted form and thus the passphrase
is required from the user to spend coins, or to create new addresses.
Keys are encrypted with AES-256-CBC using OpenSSL's EVP library. The key is
calculated via EVP_BytesToKey using SHA512 with (by default) 25000 rounds and
a random salt.
By default, the user's wallet remains unencrypted until they call the RPC
command encryptwallet <passphrase> or, from the GUI menu, Options->
Encrypt Wallet.
When the user is attempting to call RPC functions which require the password
to unlock the wallet, an error will be returned unless they call
walletpassphrase <passphrase> <time to keep key in memory> first.
A keypoolrefill command has been added which tops up the users keypool
(requiring the passphrase via walletpassphrase first).
keypoolsize has been added to the output of getinfo to show the user the
number of keys left before they need to specify their passphrase (and call
keypoolrefill).
Note that walletpassphrase will automatically fill keypool in a separate
thread which it spawns when the passphrase is set. This could cause some
delays in other threads waiting for locks on the wallet passphrase, including
one which could cause the passphrase to be stored longer than expected,
however it will not allow the passphrase to be used longer than expected as
ThreadCleanWalletPassphrase will attempt to get a lock on the key as soon
as the specified lock time has arrived.
When the keypool runs out (and wallet is locked) GetOrReuseKeyFromPool
returns vchDefaultKey, meaning miners may start to generate many blocks to
vchDefaultKey instead of a new key each time.
A walletpassphrasechange <oldpassphrase> <newpassphrase> has been added to
allow the user to change their password via RPC.
Whenever keying material (unencrypted private keys, the user's passphrase,
the wallet's AES key) is stored unencrypted in memory, any reasonable attempt
is made to mlock/VirtualLock that memory before storing the keying material.
This is not true in several (commented) cases where mlock/VirtualLocking the
memory is not possible.
Although encryption of private keys in memory can be very useful on desktop
systems (as some small amount of protection against stupid viruses), on an
RPC server, the password is entered fairly insecurely. Thus, the only main
advantage encryption has for RPC servers is for RPC servers that do not spend
coins, except in rare cases, eg. a webserver of a merchant which only receives
payment except for cases of manual intervention.
Thanks to jgarzik for the original patch and sipa, gmaxwell and many others
for all their input.
Conflicts:
src/wallet.cpp
2011-07-08 15:47:35 +02:00
return false ;
}
2011-11-26 06:02:04 +00:00
bool CWallet : : ChangeWalletPassphrase ( const SecureString & strOldWalletPassphrase , const SecureString & strNewWalletPassphrase )
Add wallet privkey encryption.
This commit adds support for ckeys, or enCrypted private keys, to the wallet.
All keys are stored in memory in their encrypted form and thus the passphrase
is required from the user to spend coins, or to create new addresses.
Keys are encrypted with AES-256-CBC using OpenSSL's EVP library. The key is
calculated via EVP_BytesToKey using SHA512 with (by default) 25000 rounds and
a random salt.
By default, the user's wallet remains unencrypted until they call the RPC
command encryptwallet <passphrase> or, from the GUI menu, Options->
Encrypt Wallet.
When the user is attempting to call RPC functions which require the password
to unlock the wallet, an error will be returned unless they call
walletpassphrase <passphrase> <time to keep key in memory> first.
A keypoolrefill command has been added which tops up the users keypool
(requiring the passphrase via walletpassphrase first).
keypoolsize has been added to the output of getinfo to show the user the
number of keys left before they need to specify their passphrase (and call
keypoolrefill).
Note that walletpassphrase will automatically fill keypool in a separate
thread which it spawns when the passphrase is set. This could cause some
delays in other threads waiting for locks on the wallet passphrase, including
one which could cause the passphrase to be stored longer than expected,
however it will not allow the passphrase to be used longer than expected as
ThreadCleanWalletPassphrase will attempt to get a lock on the key as soon
as the specified lock time has arrived.
When the keypool runs out (and wallet is locked) GetOrReuseKeyFromPool
returns vchDefaultKey, meaning miners may start to generate many blocks to
vchDefaultKey instead of a new key each time.
A walletpassphrasechange <oldpassphrase> <newpassphrase> has been added to
allow the user to change their password via RPC.
Whenever keying material (unencrypted private keys, the user's passphrase,
the wallet's AES key) is stored unencrypted in memory, any reasonable attempt
is made to mlock/VirtualLock that memory before storing the keying material.
This is not true in several (commented) cases where mlock/VirtualLocking the
memory is not possible.
Although encryption of private keys in memory can be very useful on desktop
systems (as some small amount of protection against stupid viruses), on an
RPC server, the password is entered fairly insecurely. Thus, the only main
advantage encryption has for RPC servers is for RPC servers that do not spend
coins, except in rare cases, eg. a webserver of a merchant which only receives
payment except for cases of manual intervention.
Thanks to jgarzik for the original patch and sipa, gmaxwell and many others
for all their input.
Conflicts:
src/wallet.cpp
2011-07-08 15:47:35 +02:00
{
2011-08-26 14:37:23 -04:00
bool fWasLocked = IsLocked ( ) ;
Add wallet privkey encryption.
This commit adds support for ckeys, or enCrypted private keys, to the wallet.
All keys are stored in memory in their encrypted form and thus the passphrase
is required from the user to spend coins, or to create new addresses.
Keys are encrypted with AES-256-CBC using OpenSSL's EVP library. The key is
calculated via EVP_BytesToKey using SHA512 with (by default) 25000 rounds and
a random salt.
By default, the user's wallet remains unencrypted until they call the RPC
command encryptwallet <passphrase> or, from the GUI menu, Options->
Encrypt Wallet.
When the user is attempting to call RPC functions which require the password
to unlock the wallet, an error will be returned unless they call
walletpassphrase <passphrase> <time to keep key in memory> first.
A keypoolrefill command has been added which tops up the users keypool
(requiring the passphrase via walletpassphrase first).
keypoolsize has been added to the output of getinfo to show the user the
number of keys left before they need to specify their passphrase (and call
keypoolrefill).
Note that walletpassphrase will automatically fill keypool in a separate
thread which it spawns when the passphrase is set. This could cause some
delays in other threads waiting for locks on the wallet passphrase, including
one which could cause the passphrase to be stored longer than expected,
however it will not allow the passphrase to be used longer than expected as
ThreadCleanWalletPassphrase will attempt to get a lock on the key as soon
as the specified lock time has arrived.
When the keypool runs out (and wallet is locked) GetOrReuseKeyFromPool
returns vchDefaultKey, meaning miners may start to generate many blocks to
vchDefaultKey instead of a new key each time.
A walletpassphrasechange <oldpassphrase> <newpassphrase> has been added to
allow the user to change their password via RPC.
Whenever keying material (unencrypted private keys, the user's passphrase,
the wallet's AES key) is stored unencrypted in memory, any reasonable attempt
is made to mlock/VirtualLock that memory before storing the keying material.
This is not true in several (commented) cases where mlock/VirtualLocking the
memory is not possible.
Although encryption of private keys in memory can be very useful on desktop
systems (as some small amount of protection against stupid viruses), on an
RPC server, the password is entered fairly insecurely. Thus, the only main
advantage encryption has for RPC servers is for RPC servers that do not spend
coins, except in rare cases, eg. a webserver of a merchant which only receives
payment except for cases of manual intervention.
Thanks to jgarzik for the original patch and sipa, gmaxwell and many others
for all their input.
Conflicts:
src/wallet.cpp
2011-07-08 15:47:35 +02:00
2011-08-26 14:37:23 -04:00
{
2012-04-06 18:39:12 +02:00
LOCK ( cs_wallet ) ;
Add wallet privkey encryption.
This commit adds support for ckeys, or enCrypted private keys, to the wallet.
All keys are stored in memory in their encrypted form and thus the passphrase
is required from the user to spend coins, or to create new addresses.
Keys are encrypted with AES-256-CBC using OpenSSL's EVP library. The key is
calculated via EVP_BytesToKey using SHA512 with (by default) 25000 rounds and
a random salt.
By default, the user's wallet remains unencrypted until they call the RPC
command encryptwallet <passphrase> or, from the GUI menu, Options->
Encrypt Wallet.
When the user is attempting to call RPC functions which require the password
to unlock the wallet, an error will be returned unless they call
walletpassphrase <passphrase> <time to keep key in memory> first.
A keypoolrefill command has been added which tops up the users keypool
(requiring the passphrase via walletpassphrase first).
keypoolsize has been added to the output of getinfo to show the user the
number of keys left before they need to specify their passphrase (and call
keypoolrefill).
Note that walletpassphrase will automatically fill keypool in a separate
thread which it spawns when the passphrase is set. This could cause some
delays in other threads waiting for locks on the wallet passphrase, including
one which could cause the passphrase to be stored longer than expected,
however it will not allow the passphrase to be used longer than expected as
ThreadCleanWalletPassphrase will attempt to get a lock on the key as soon
as the specified lock time has arrived.
When the keypool runs out (and wallet is locked) GetOrReuseKeyFromPool
returns vchDefaultKey, meaning miners may start to generate many blocks to
vchDefaultKey instead of a new key each time.
A walletpassphrasechange <oldpassphrase> <newpassphrase> has been added to
allow the user to change their password via RPC.
Whenever keying material (unencrypted private keys, the user's passphrase,
the wallet's AES key) is stored unencrypted in memory, any reasonable attempt
is made to mlock/VirtualLock that memory before storing the keying material.
This is not true in several (commented) cases where mlock/VirtualLocking the
memory is not possible.
Although encryption of private keys in memory can be very useful on desktop
systems (as some small amount of protection against stupid viruses), on an
RPC server, the password is entered fairly insecurely. Thus, the only main
advantage encryption has for RPC servers is for RPC servers that do not spend
coins, except in rare cases, eg. a webserver of a merchant which only receives
payment except for cases of manual intervention.
Thanks to jgarzik for the original patch and sipa, gmaxwell and many others
for all their input.
Conflicts:
src/wallet.cpp
2011-07-08 15:47:35 +02:00
Lock ( ) ;
CCrypter crypter ;
2016-11-10 08:00:05 +01:00
CKeyingMaterial _vMasterKey ;
2017-06-02 03:18:57 +02:00
for ( MasterKeyMap : : value_type & pMasterKey : mapMasterKeys )
Add wallet privkey encryption.
This commit adds support for ckeys, or enCrypted private keys, to the wallet.
All keys are stored in memory in their encrypted form and thus the passphrase
is required from the user to spend coins, or to create new addresses.
Keys are encrypted with AES-256-CBC using OpenSSL's EVP library. The key is
calculated via EVP_BytesToKey using SHA512 with (by default) 25000 rounds and
a random salt.
By default, the user's wallet remains unencrypted until they call the RPC
command encryptwallet <passphrase> or, from the GUI menu, Options->
Encrypt Wallet.
When the user is attempting to call RPC functions which require the password
to unlock the wallet, an error will be returned unless they call
walletpassphrase <passphrase> <time to keep key in memory> first.
A keypoolrefill command has been added which tops up the users keypool
(requiring the passphrase via walletpassphrase first).
keypoolsize has been added to the output of getinfo to show the user the
number of keys left before they need to specify their passphrase (and call
keypoolrefill).
Note that walletpassphrase will automatically fill keypool in a separate
thread which it spawns when the passphrase is set. This could cause some
delays in other threads waiting for locks on the wallet passphrase, including
one which could cause the passphrase to be stored longer than expected,
however it will not allow the passphrase to be used longer than expected as
ThreadCleanWalletPassphrase will attempt to get a lock on the key as soon
as the specified lock time has arrived.
When the keypool runs out (and wallet is locked) GetOrReuseKeyFromPool
returns vchDefaultKey, meaning miners may start to generate many blocks to
vchDefaultKey instead of a new key each time.
A walletpassphrasechange <oldpassphrase> <newpassphrase> has been added to
allow the user to change their password via RPC.
Whenever keying material (unencrypted private keys, the user's passphrase,
the wallet's AES key) is stored unencrypted in memory, any reasonable attempt
is made to mlock/VirtualLock that memory before storing the keying material.
This is not true in several (commented) cases where mlock/VirtualLocking the
memory is not possible.
Although encryption of private keys in memory can be very useful on desktop
systems (as some small amount of protection against stupid viruses), on an
RPC server, the password is entered fairly insecurely. Thus, the only main
advantage encryption has for RPC servers is for RPC servers that do not spend
coins, except in rare cases, eg. a webserver of a merchant which only receives
payment except for cases of manual intervention.
Thanks to jgarzik for the original patch and sipa, gmaxwell and many others
for all their input.
Conflicts:
src/wallet.cpp
2011-07-08 15:47:35 +02:00
{
if ( ! crypter . SetKeyFromPassphrase ( strOldWalletPassphrase , pMasterKey . second . vchSalt , pMasterKey . second . nDeriveIterations , pMasterKey . second . nDerivationMethod ) )
return false ;
2016-11-10 08:00:05 +01:00
if ( ! crypter . Decrypt ( pMasterKey . second . vchCryptedKey , _vMasterKey ) )
Add wallet privkey encryption.
This commit adds support for ckeys, or enCrypted private keys, to the wallet.
All keys are stored in memory in their encrypted form and thus the passphrase
is required from the user to spend coins, or to create new addresses.
Keys are encrypted with AES-256-CBC using OpenSSL's EVP library. The key is
calculated via EVP_BytesToKey using SHA512 with (by default) 25000 rounds and
a random salt.
By default, the user's wallet remains unencrypted until they call the RPC
command encryptwallet <passphrase> or, from the GUI menu, Options->
Encrypt Wallet.
When the user is attempting to call RPC functions which require the password
to unlock the wallet, an error will be returned unless they call
walletpassphrase <passphrase> <time to keep key in memory> first.
A keypoolrefill command has been added which tops up the users keypool
(requiring the passphrase via walletpassphrase first).
keypoolsize has been added to the output of getinfo to show the user the
number of keys left before they need to specify their passphrase (and call
keypoolrefill).
Note that walletpassphrase will automatically fill keypool in a separate
thread which it spawns when the passphrase is set. This could cause some
delays in other threads waiting for locks on the wallet passphrase, including
one which could cause the passphrase to be stored longer than expected,
however it will not allow the passphrase to be used longer than expected as
ThreadCleanWalletPassphrase will attempt to get a lock on the key as soon
as the specified lock time has arrived.
When the keypool runs out (and wallet is locked) GetOrReuseKeyFromPool
returns vchDefaultKey, meaning miners may start to generate many blocks to
vchDefaultKey instead of a new key each time.
A walletpassphrasechange <oldpassphrase> <newpassphrase> has been added to
allow the user to change their password via RPC.
Whenever keying material (unencrypted private keys, the user's passphrase,
the wallet's AES key) is stored unencrypted in memory, any reasonable attempt
is made to mlock/VirtualLock that memory before storing the keying material.
This is not true in several (commented) cases where mlock/VirtualLocking the
memory is not possible.
Although encryption of private keys in memory can be very useful on desktop
systems (as some small amount of protection against stupid viruses), on an
RPC server, the password is entered fairly insecurely. Thus, the only main
advantage encryption has for RPC servers is for RPC servers that do not spend
coins, except in rare cases, eg. a webserver of a merchant which only receives
payment except for cases of manual intervention.
Thanks to jgarzik for the original patch and sipa, gmaxwell and many others
for all their input.
Conflicts:
src/wallet.cpp
2011-07-08 15:47:35 +02:00
return false ;
2019-06-06 23:58:21 +02:00
if ( Unlock ( _vMasterKey ) )
Add wallet privkey encryption.
This commit adds support for ckeys, or enCrypted private keys, to the wallet.
All keys are stored in memory in their encrypted form and thus the passphrase
is required from the user to spend coins, or to create new addresses.
Keys are encrypted with AES-256-CBC using OpenSSL's EVP library. The key is
calculated via EVP_BytesToKey using SHA512 with (by default) 25000 rounds and
a random salt.
By default, the user's wallet remains unencrypted until they call the RPC
command encryptwallet <passphrase> or, from the GUI menu, Options->
Encrypt Wallet.
When the user is attempting to call RPC functions which require the password
to unlock the wallet, an error will be returned unless they call
walletpassphrase <passphrase> <time to keep key in memory> first.
A keypoolrefill command has been added which tops up the users keypool
(requiring the passphrase via walletpassphrase first).
keypoolsize has been added to the output of getinfo to show the user the
number of keys left before they need to specify their passphrase (and call
keypoolrefill).
Note that walletpassphrase will automatically fill keypool in a separate
thread which it spawns when the passphrase is set. This could cause some
delays in other threads waiting for locks on the wallet passphrase, including
one which could cause the passphrase to be stored longer than expected,
however it will not allow the passphrase to be used longer than expected as
ThreadCleanWalletPassphrase will attempt to get a lock on the key as soon
as the specified lock time has arrived.
When the keypool runs out (and wallet is locked) GetOrReuseKeyFromPool
returns vchDefaultKey, meaning miners may start to generate many blocks to
vchDefaultKey instead of a new key each time.
A walletpassphrasechange <oldpassphrase> <newpassphrase> has been added to
allow the user to change their password via RPC.
Whenever keying material (unencrypted private keys, the user's passphrase,
the wallet's AES key) is stored unencrypted in memory, any reasonable attempt
is made to mlock/VirtualLock that memory before storing the keying material.
This is not true in several (commented) cases where mlock/VirtualLocking the
memory is not possible.
Although encryption of private keys in memory can be very useful on desktop
systems (as some small amount of protection against stupid viruses), on an
RPC server, the password is entered fairly insecurely. Thus, the only main
advantage encryption has for RPC servers is for RPC servers that do not spend
coins, except in rare cases, eg. a webserver of a merchant which only receives
payment except for cases of manual intervention.
Thanks to jgarzik for the original patch and sipa, gmaxwell and many others
for all their input.
Conflicts:
src/wallet.cpp
2011-07-08 15:47:35 +02:00
{
2013-04-13 00:13:08 -05:00
int64_t nStartTime = GetTimeMillis ( ) ;
2011-06-28 15:31:09 +02:00
crypter . SetKeyFromPassphrase ( strNewWalletPassphrase , pMasterKey . second . vchSalt , pMasterKey . second . nDeriveIterations , pMasterKey . second . nDerivationMethod ) ;
2017-09-11 15:43:49 -04:00
pMasterKey . second . nDeriveIterations = static_cast < unsigned int > ( pMasterKey . second . nDeriveIterations * ( 100 / ( ( double ) ( GetTimeMillis ( ) - nStartTime ) ) ) ) ;
2011-06-28 15:31:09 +02:00
nStartTime = GetTimeMillis ( ) ;
crypter . SetKeyFromPassphrase ( strNewWalletPassphrase , pMasterKey . second . vchSalt , pMasterKey . second . nDeriveIterations , pMasterKey . second . nDerivationMethod ) ;
2017-09-11 15:43:49 -04:00
pMasterKey . second . nDeriveIterations = ( pMasterKey . second . nDeriveIterations + static_cast < unsigned int > ( pMasterKey . second . nDeriveIterations * 100 / ( ( double ) ( GetTimeMillis ( ) - nStartTime ) ) ) ) / 2 ;
2011-06-28 15:31:09 +02:00
if ( pMasterKey . second . nDeriveIterations < 25000 )
pMasterKey . second . nDeriveIterations = 25000 ;
2018-06-15 19:02:52 -04:00
WalletLogPrintf ( " Wallet passphrase changed to an nDeriveIterations of %i \n " , pMasterKey . second . nDeriveIterations ) ;
2011-06-28 15:31:09 +02:00
Add wallet privkey encryption.
This commit adds support for ckeys, or enCrypted private keys, to the wallet.
All keys are stored in memory in their encrypted form and thus the passphrase
is required from the user to spend coins, or to create new addresses.
Keys are encrypted with AES-256-CBC using OpenSSL's EVP library. The key is
calculated via EVP_BytesToKey using SHA512 with (by default) 25000 rounds and
a random salt.
By default, the user's wallet remains unencrypted until they call the RPC
command encryptwallet <passphrase> or, from the GUI menu, Options->
Encrypt Wallet.
When the user is attempting to call RPC functions which require the password
to unlock the wallet, an error will be returned unless they call
walletpassphrase <passphrase> <time to keep key in memory> first.
A keypoolrefill command has been added which tops up the users keypool
(requiring the passphrase via walletpassphrase first).
keypoolsize has been added to the output of getinfo to show the user the
number of keys left before they need to specify their passphrase (and call
keypoolrefill).
Note that walletpassphrase will automatically fill keypool in a separate
thread which it spawns when the passphrase is set. This could cause some
delays in other threads waiting for locks on the wallet passphrase, including
one which could cause the passphrase to be stored longer than expected,
however it will not allow the passphrase to be used longer than expected as
ThreadCleanWalletPassphrase will attempt to get a lock on the key as soon
as the specified lock time has arrived.
When the keypool runs out (and wallet is locked) GetOrReuseKeyFromPool
returns vchDefaultKey, meaning miners may start to generate many blocks to
vchDefaultKey instead of a new key each time.
A walletpassphrasechange <oldpassphrase> <newpassphrase> has been added to
allow the user to change their password via RPC.
Whenever keying material (unencrypted private keys, the user's passphrase,
the wallet's AES key) is stored unencrypted in memory, any reasonable attempt
is made to mlock/VirtualLock that memory before storing the keying material.
This is not true in several (commented) cases where mlock/VirtualLocking the
memory is not possible.
Although encryption of private keys in memory can be very useful on desktop
systems (as some small amount of protection against stupid viruses), on an
RPC server, the password is entered fairly insecurely. Thus, the only main
advantage encryption has for RPC servers is for RPC servers that do not spend
coins, except in rare cases, eg. a webserver of a merchant which only receives
payment except for cases of manual intervention.
Thanks to jgarzik for the original patch and sipa, gmaxwell and many others
for all their input.
Conflicts:
src/wallet.cpp
2011-07-08 15:47:35 +02:00
if ( ! crypter . SetKeyFromPassphrase ( strNewWalletPassphrase , pMasterKey . second . vchSalt , pMasterKey . second . nDeriveIterations , pMasterKey . second . nDerivationMethod ) )
return false ;
2016-11-10 08:00:05 +01:00
if ( ! crypter . Encrypt ( _vMasterKey , pMasterKey . second . vchCryptedKey ) )
Add wallet privkey encryption.
This commit adds support for ckeys, or enCrypted private keys, to the wallet.
All keys are stored in memory in their encrypted form and thus the passphrase
is required from the user to spend coins, or to create new addresses.
Keys are encrypted with AES-256-CBC using OpenSSL's EVP library. The key is
calculated via EVP_BytesToKey using SHA512 with (by default) 25000 rounds and
a random salt.
By default, the user's wallet remains unencrypted until they call the RPC
command encryptwallet <passphrase> or, from the GUI menu, Options->
Encrypt Wallet.
When the user is attempting to call RPC functions which require the password
to unlock the wallet, an error will be returned unless they call
walletpassphrase <passphrase> <time to keep key in memory> first.
A keypoolrefill command has been added which tops up the users keypool
(requiring the passphrase via walletpassphrase first).
keypoolsize has been added to the output of getinfo to show the user the
number of keys left before they need to specify their passphrase (and call
keypoolrefill).
Note that walletpassphrase will automatically fill keypool in a separate
thread which it spawns when the passphrase is set. This could cause some
delays in other threads waiting for locks on the wallet passphrase, including
one which could cause the passphrase to be stored longer than expected,
however it will not allow the passphrase to be used longer than expected as
ThreadCleanWalletPassphrase will attempt to get a lock on the key as soon
as the specified lock time has arrived.
When the keypool runs out (and wallet is locked) GetOrReuseKeyFromPool
returns vchDefaultKey, meaning miners may start to generate many blocks to
vchDefaultKey instead of a new key each time.
A walletpassphrasechange <oldpassphrase> <newpassphrase> has been added to
allow the user to change their password via RPC.
Whenever keying material (unencrypted private keys, the user's passphrase,
the wallet's AES key) is stored unencrypted in memory, any reasonable attempt
is made to mlock/VirtualLock that memory before storing the keying material.
This is not true in several (commented) cases where mlock/VirtualLocking the
memory is not possible.
Although encryption of private keys in memory can be very useful on desktop
systems (as some small amount of protection against stupid viruses), on an
RPC server, the password is entered fairly insecurely. Thus, the only main
advantage encryption has for RPC servers is for RPC servers that do not spend
coins, except in rare cases, eg. a webserver of a merchant which only receives
payment except for cases of manual intervention.
Thanks to jgarzik for the original patch and sipa, gmaxwell and many others
for all their input.
Conflicts:
src/wallet.cpp
2011-07-08 15:47:35 +02:00
return false ;
2017-12-08 06:39:22 -05:00
WalletBatch ( * database ) . WriteMasterKey ( pMasterKey . first , pMasterKey . second ) ;
Add wallet privkey encryption.
This commit adds support for ckeys, or enCrypted private keys, to the wallet.
All keys are stored in memory in their encrypted form and thus the passphrase
is required from the user to spend coins, or to create new addresses.
Keys are encrypted with AES-256-CBC using OpenSSL's EVP library. The key is
calculated via EVP_BytesToKey using SHA512 with (by default) 25000 rounds and
a random salt.
By default, the user's wallet remains unencrypted until they call the RPC
command encryptwallet <passphrase> or, from the GUI menu, Options->
Encrypt Wallet.
When the user is attempting to call RPC functions which require the password
to unlock the wallet, an error will be returned unless they call
walletpassphrase <passphrase> <time to keep key in memory> first.
A keypoolrefill command has been added which tops up the users keypool
(requiring the passphrase via walletpassphrase first).
keypoolsize has been added to the output of getinfo to show the user the
number of keys left before they need to specify their passphrase (and call
keypoolrefill).
Note that walletpassphrase will automatically fill keypool in a separate
thread which it spawns when the passphrase is set. This could cause some
delays in other threads waiting for locks on the wallet passphrase, including
one which could cause the passphrase to be stored longer than expected,
however it will not allow the passphrase to be used longer than expected as
ThreadCleanWalletPassphrase will attempt to get a lock on the key as soon
as the specified lock time has arrived.
When the keypool runs out (and wallet is locked) GetOrReuseKeyFromPool
returns vchDefaultKey, meaning miners may start to generate many blocks to
vchDefaultKey instead of a new key each time.
A walletpassphrasechange <oldpassphrase> <newpassphrase> has been added to
allow the user to change their password via RPC.
Whenever keying material (unencrypted private keys, the user's passphrase,
the wallet's AES key) is stored unencrypted in memory, any reasonable attempt
is made to mlock/VirtualLock that memory before storing the keying material.
This is not true in several (commented) cases where mlock/VirtualLocking the
memory is not possible.
Although encryption of private keys in memory can be very useful on desktop
systems (as some small amount of protection against stupid viruses), on an
RPC server, the password is entered fairly insecurely. Thus, the only main
advantage encryption has for RPC servers is for RPC servers that do not spend
coins, except in rare cases, eg. a webserver of a merchant which only receives
payment except for cases of manual intervention.
Thanks to jgarzik for the original patch and sipa, gmaxwell and many others
for all their input.
Conflicts:
src/wallet.cpp
2011-07-08 15:47:35 +02:00
if ( fWasLocked )
Lock ( ) ;
return true ;
}
}
}
2011-08-26 14:37:23 -04:00
Add wallet privkey encryption.
This commit adds support for ckeys, or enCrypted private keys, to the wallet.
All keys are stored in memory in their encrypted form and thus the passphrase
is required from the user to spend coins, or to create new addresses.
Keys are encrypted with AES-256-CBC using OpenSSL's EVP library. The key is
calculated via EVP_BytesToKey using SHA512 with (by default) 25000 rounds and
a random salt.
By default, the user's wallet remains unencrypted until they call the RPC
command encryptwallet <passphrase> or, from the GUI menu, Options->
Encrypt Wallet.
When the user is attempting to call RPC functions which require the password
to unlock the wallet, an error will be returned unless they call
walletpassphrase <passphrase> <time to keep key in memory> first.
A keypoolrefill command has been added which tops up the users keypool
(requiring the passphrase via walletpassphrase first).
keypoolsize has been added to the output of getinfo to show the user the
number of keys left before they need to specify their passphrase (and call
keypoolrefill).
Note that walletpassphrase will automatically fill keypool in a separate
thread which it spawns when the passphrase is set. This could cause some
delays in other threads waiting for locks on the wallet passphrase, including
one which could cause the passphrase to be stored longer than expected,
however it will not allow the passphrase to be used longer than expected as
ThreadCleanWalletPassphrase will attempt to get a lock on the key as soon
as the specified lock time has arrived.
When the keypool runs out (and wallet is locked) GetOrReuseKeyFromPool
returns vchDefaultKey, meaning miners may start to generate many blocks to
vchDefaultKey instead of a new key each time.
A walletpassphrasechange <oldpassphrase> <newpassphrase> has been added to
allow the user to change their password via RPC.
Whenever keying material (unencrypted private keys, the user's passphrase,
the wallet's AES key) is stored unencrypted in memory, any reasonable attempt
is made to mlock/VirtualLock that memory before storing the keying material.
This is not true in several (commented) cases where mlock/VirtualLocking the
memory is not possible.
Although encryption of private keys in memory can be very useful on desktop
systems (as some small amount of protection against stupid viruses), on an
RPC server, the password is entered fairly insecurely. Thus, the only main
advantage encryption has for RPC servers is for RPC servers that do not spend
coins, except in rare cases, eg. a webserver of a merchant which only receives
payment except for cases of manual intervention.
Thanks to jgarzik for the original patch and sipa, gmaxwell and many others
for all their input.
Conflicts:
src/wallet.cpp
2011-07-08 15:47:35 +02:00
return false ;
}
2020-02-26 16:05:49 -05:00
void CWallet : : chainStateFlushed ( const CBlockLocator & loc )
2012-04-15 22:10:54 +02:00
{
2017-12-08 06:39:22 -05:00
WalletBatch batch ( * database ) ;
batch . WriteBestBlock ( loc ) ;
2012-04-15 22:10:54 +02:00
}
2011-07-10 16:07:22 +02:00
2018-07-27 08:22:42 +02:00
void CWallet : : SetMinVersion ( enum WalletFeature nVersion , WalletBatch * batch_in , bool fExplicit )
2012-02-18 14:55:02 +01:00
{
2019-02-18 16:09:45 -05:00
LOCK ( cs_wallet ) ;
2012-02-18 14:55:02 +01:00
if ( nWalletVersion > = nVersion )
2018-07-27 08:22:42 +02:00
return ;
2012-02-18 14:55:02 +01:00
2012-03-22 03:56:31 +01:00
// when doing an explicit upgrade, if we pass the max version permitted, upgrade all the way
if ( fExplicit & & nVersion > nWalletMaxVersion )
nVersion = FEATURE_LATEST ;
2012-02-18 14:55:02 +01:00
nWalletVersion = nVersion ;
2012-03-22 03:56:31 +01:00
if ( nVersion > nWalletMaxVersion )
nWalletMaxVersion = nVersion ;
2012-02-18 14:55:02 +01:00
{
2017-12-08 06:39:22 -05:00
WalletBatch * batch = batch_in ? batch_in : new WalletBatch ( * database ) ;
2012-02-18 14:55:02 +01:00
if ( nWalletVersion > 40000 )
2017-12-08 06:39:22 -05:00
batch - > WriteMinVersion ( nWalletVersion ) ;
if ( ! batch_in )
delete batch ;
2012-02-18 14:55:02 +01:00
}
}
2012-03-22 03:56:31 +01:00
bool CWallet : : SetMaxVersion ( int nVersion )
{
2019-02-18 16:09:45 -05:00
LOCK ( cs_wallet ) ;
2012-03-22 03:56:31 +01:00
// cannot downgrade below current version
if ( nWalletVersion > nVersion )
return false ;
nWalletMaxVersion = nVersion ;
return true ;
}
2017-01-27 10:33:45 +09:00
std : : set < uint256 > CWallet : : GetConflicts ( const uint256 & txid ) const
2014-02-13 20:12:51 -05:00
{
2017-01-27 10:33:45 +09:00
std : : set < uint256 > result ;
2014-02-13 20:12:51 -05:00
AssertLockHeld ( cs_wallet ) ;
std : : map < uint256 , CWalletTx > : : const_iterator it = mapWallet . find ( txid ) ;
if ( it = = mapWallet . end ( ) )
return result ;
const CWalletTx & wtx = it - > second ;
2014-02-15 16:38:28 -05:00
std : : pair < TxSpends : : const_iterator , TxSpends : : const_iterator > range ;
2014-02-13 20:12:51 -05:00
2017-06-02 03:18:57 +02:00
for ( const CTxIn & txin : wtx . tx - > vin )
2014-02-13 20:12:51 -05:00
{
2014-02-15 16:38:28 -05:00
if ( mapTxSpends . count ( txin . prevout ) < = 1 )
continue ; // No conflict if zero or one spends
range = mapTxSpends . equal_range ( txin . prevout ) ;
2016-09-02 18:19:01 +02:00
for ( TxSpends : : const_iterator _it = range . first ; _it ! = range . second ; + + _it )
result . insert ( _it - > second ) ;
2014-02-13 20:12:51 -05:00
}
return result ;
}
2016-12-09 13:45:27 -05:00
bool CWallet : : HasWalletSpend ( const uint256 & txid ) const
{
AssertLockHeld ( cs_wallet ) ;
auto iter = mapTxSpends . lower_bound ( COutPoint ( txid , 0 ) ) ;
return ( iter ! = mapTxSpends . end ( ) & & iter - > first . hash = = txid ) ;
}
2015-02-04 21:19:27 +01:00
void CWallet : : Flush ( bool shutdown )
{
2017-12-08 06:39:22 -05:00
database - > Flush ( shutdown ) ;
2015-02-04 21:19:27 +01:00
}
2017-01-27 10:33:45 +09:00
void CWallet : : SyncMetaData ( std : : pair < TxSpends : : iterator , TxSpends : : iterator > range )
2014-02-13 20:12:51 -05:00
{
// We want all the wallet transactions in range to have the same metadata as
// the oldest (smallest nOrderPos).
// So: find smallest nOrderPos:
int nMinOrderPos = std : : numeric_limits < int > : : max ( ) ;
2017-08-07 07:36:37 +02:00
const CWalletTx * copyFrom = nullptr ;
2018-01-26 16:28:31 +00:00
for ( TxSpends : : iterator it = range . first ; it ! = range . second ; + + it ) {
2017-01-19 16:08:03 -05:00
const CWalletTx * wtx = & mapWallet . at ( it - > second ) ;
2018-01-26 16:28:31 +00:00
if ( wtx - > nOrderPos < nMinOrderPos ) {
2018-06-29 11:30:25 +02:00
nMinOrderPos = wtx - > nOrderPos ;
2018-01-26 16:28:31 +00:00
copyFrom = wtx ;
2014-02-13 20:12:51 -05:00
}
}
2017-08-17 15:26:32 +02:00
2018-05-17 22:07:32 +02:00
if ( ! copyFrom ) {
return ;
}
2017-08-17 15:26:32 +02:00
2014-02-13 20:12:51 -05:00
// Now copy data from copyFrom to rest:
2014-02-15 16:38:28 -05:00
for ( TxSpends : : iterator it = range . first ; it ! = range . second ; + + it )
2014-02-13 20:12:51 -05:00
{
const uint256 & hash = it - > second ;
2017-01-19 16:08:03 -05:00
CWalletTx * copyTo = & mapWallet . at ( hash ) ;
2014-02-13 20:12:51 -05:00
if ( copyFrom = = copyTo ) continue ;
2017-08-28 09:24:17 +02:00
assert ( copyFrom & & " Oldest wallet transaction in range assumed to have been found. " ) ;
2015-03-11 16:48:53 -07:00
if ( ! copyFrom - > IsEquivalentTo ( * copyTo ) ) continue ;
2014-02-13 20:12:51 -05:00
copyTo - > mapValue = copyFrom - > mapValue ;
copyTo - > vOrderForm = copyFrom - > vOrderForm ;
// fTimeReceivedIsTxTime not copied on purpose
// nTimeReceived not copied on purpose
copyTo - > nTimeSmart = copyFrom - > nTimeSmart ;
copyTo - > fFromMe = copyFrom - > fFromMe ;
// nOrderPos not copied on purpose
// cached members not copied on purpose
}
}
2014-10-26 15:03:12 +08:00
/**
* Outpoint is spent if any non - conflicted transaction
* spends it :
*/
2019-04-29 09:52:01 -04:00
bool CWallet : : IsSpent ( const uint256 & hash , unsigned int n ) const
2014-02-13 20:12:51 -05:00
{
2014-02-15 16:38:28 -05:00
const COutPoint outpoint ( hash , n ) ;
2017-01-27 10:33:45 +09:00
std : : pair < TxSpends : : const_iterator , TxSpends : : const_iterator > range ;
2014-02-15 16:38:28 -05:00
range = mapTxSpends . equal_range ( outpoint ) ;
2014-02-13 20:12:51 -05:00
2014-02-15 16:38:28 -05:00
for ( TxSpends : : const_iterator it = range . first ; it ! = range . second ; + + it )
2014-02-13 20:12:51 -05:00
{
2014-02-15 16:38:28 -05:00
const uint256 & wtxid = it - > second ;
std : : map < uint256 , CWalletTx > : : const_iterator mit = mapWallet . find ( wtxid ) ;
2016-01-07 16:31:27 -05:00
if ( mit ! = mapWallet . end ( ) ) {
2019-04-29 09:52:01 -04:00
int depth = mit - > second . GetDepthInMainChain ( ) ;
2016-01-07 16:31:27 -05:00
if ( depth > 0 | | ( depth = = 0 & & ! mit - > second . isAbandoned ( ) ) )
return true ; // Spent
}
2014-02-13 20:12:51 -05:00
}
2014-02-15 16:38:28 -05:00
return false ;
}
void CWallet : : AddToSpends ( const COutPoint & outpoint , const uint256 & wtxid )
{
2017-01-27 10:33:45 +09:00
mapTxSpends . insert ( std : : make_pair ( outpoint , wtxid ) ) ;
2014-02-15 16:38:28 -05:00
2018-05-25 14:27:58 +01:00
setLockedCoins . erase ( outpoint ) ;
2017-01-27 10:33:45 +09:00
std : : pair < TxSpends : : iterator , TxSpends : : iterator > range ;
2014-02-15 16:38:28 -05:00
range = mapTxSpends . equal_range ( outpoint ) ;
SyncMetaData ( range ) ;
}
void CWallet : : AddToSpends ( const uint256 & wtxid )
{
2017-08-13 15:04:57 +01:00
auto it = mapWallet . find ( wtxid ) ;
assert ( it ! = mapWallet . end ( ) ) ;
CWalletTx & thisTx = it - > second ;
2014-02-15 16:38:28 -05:00
if ( thisTx . IsCoinBase ( ) ) // Coinbases don't spend anything!
return ;
2017-06-02 03:18:57 +02:00
for ( const CTxIn & txin : thisTx . tx - > vin )
2014-02-15 16:38:28 -05:00
AddToSpends ( txin . prevout , wtxid ) ;
2014-02-13 20:12:51 -05:00
}
2011-11-26 06:02:04 +00:00
bool CWallet : : EncryptWallet ( const SecureString & strWalletPassphrase )
Add wallet privkey encryption.
This commit adds support for ckeys, or enCrypted private keys, to the wallet.
All keys are stored in memory in their encrypted form and thus the passphrase
is required from the user to spend coins, or to create new addresses.
Keys are encrypted with AES-256-CBC using OpenSSL's EVP library. The key is
calculated via EVP_BytesToKey using SHA512 with (by default) 25000 rounds and
a random salt.
By default, the user's wallet remains unencrypted until they call the RPC
command encryptwallet <passphrase> or, from the GUI menu, Options->
Encrypt Wallet.
When the user is attempting to call RPC functions which require the password
to unlock the wallet, an error will be returned unless they call
walletpassphrase <passphrase> <time to keep key in memory> first.
A keypoolrefill command has been added which tops up the users keypool
(requiring the passphrase via walletpassphrase first).
keypoolsize has been added to the output of getinfo to show the user the
number of keys left before they need to specify their passphrase (and call
keypoolrefill).
Note that walletpassphrase will automatically fill keypool in a separate
thread which it spawns when the passphrase is set. This could cause some
delays in other threads waiting for locks on the wallet passphrase, including
one which could cause the passphrase to be stored longer than expected,
however it will not allow the passphrase to be used longer than expected as
ThreadCleanWalletPassphrase will attempt to get a lock on the key as soon
as the specified lock time has arrived.
When the keypool runs out (and wallet is locked) GetOrReuseKeyFromPool
returns vchDefaultKey, meaning miners may start to generate many blocks to
vchDefaultKey instead of a new key each time.
A walletpassphrasechange <oldpassphrase> <newpassphrase> has been added to
allow the user to change their password via RPC.
Whenever keying material (unencrypted private keys, the user's passphrase,
the wallet's AES key) is stored unencrypted in memory, any reasonable attempt
is made to mlock/VirtualLock that memory before storing the keying material.
This is not true in several (commented) cases where mlock/VirtualLocking the
memory is not possible.
Although encryption of private keys in memory can be very useful on desktop
systems (as some small amount of protection against stupid viruses), on an
RPC server, the password is entered fairly insecurely. Thus, the only main
advantage encryption has for RPC servers is for RPC servers that do not spend
coins, except in rare cases, eg. a webserver of a merchant which only receives
payment except for cases of manual intervention.
Thanks to jgarzik for the original patch and sipa, gmaxwell and many others
for all their input.
Conflicts:
src/wallet.cpp
2011-07-08 15:47:35 +02:00
{
2011-08-26 14:37:23 -04:00
if ( IsCrypted ( ) )
return false ;
Add wallet privkey encryption.
This commit adds support for ckeys, or enCrypted private keys, to the wallet.
All keys are stored in memory in their encrypted form and thus the passphrase
is required from the user to spend coins, or to create new addresses.
Keys are encrypted with AES-256-CBC using OpenSSL's EVP library. The key is
calculated via EVP_BytesToKey using SHA512 with (by default) 25000 rounds and
a random salt.
By default, the user's wallet remains unencrypted until they call the RPC
command encryptwallet <passphrase> or, from the GUI menu, Options->
Encrypt Wallet.
When the user is attempting to call RPC functions which require the password
to unlock the wallet, an error will be returned unless they call
walletpassphrase <passphrase> <time to keep key in memory> first.
A keypoolrefill command has been added which tops up the users keypool
(requiring the passphrase via walletpassphrase first).
keypoolsize has been added to the output of getinfo to show the user the
number of keys left before they need to specify their passphrase (and call
keypoolrefill).
Note that walletpassphrase will automatically fill keypool in a separate
thread which it spawns when the passphrase is set. This could cause some
delays in other threads waiting for locks on the wallet passphrase, including
one which could cause the passphrase to be stored longer than expected,
however it will not allow the passphrase to be used longer than expected as
ThreadCleanWalletPassphrase will attempt to get a lock on the key as soon
as the specified lock time has arrived.
When the keypool runs out (and wallet is locked) GetOrReuseKeyFromPool
returns vchDefaultKey, meaning miners may start to generate many blocks to
vchDefaultKey instead of a new key each time.
A walletpassphrasechange <oldpassphrase> <newpassphrase> has been added to
allow the user to change their password via RPC.
Whenever keying material (unencrypted private keys, the user's passphrase,
the wallet's AES key) is stored unencrypted in memory, any reasonable attempt
is made to mlock/VirtualLock that memory before storing the keying material.
This is not true in several (commented) cases where mlock/VirtualLocking the
memory is not possible.
Although encryption of private keys in memory can be very useful on desktop
systems (as some small amount of protection against stupid viruses), on an
RPC server, the password is entered fairly insecurely. Thus, the only main
advantage encryption has for RPC servers is for RPC servers that do not spend
coins, except in rare cases, eg. a webserver of a merchant which only receives
payment except for cases of manual intervention.
Thanks to jgarzik for the original patch and sipa, gmaxwell and many others
for all their input.
Conflicts:
src/wallet.cpp
2011-07-08 15:47:35 +02:00
2016-11-10 08:00:05 +01:00
CKeyingMaterial _vMasterKey ;
Add wallet privkey encryption.
This commit adds support for ckeys, or enCrypted private keys, to the wallet.
All keys are stored in memory in their encrypted form and thus the passphrase
is required from the user to spend coins, or to create new addresses.
Keys are encrypted with AES-256-CBC using OpenSSL's EVP library. The key is
calculated via EVP_BytesToKey using SHA512 with (by default) 25000 rounds and
a random salt.
By default, the user's wallet remains unencrypted until they call the RPC
command encryptwallet <passphrase> or, from the GUI menu, Options->
Encrypt Wallet.
When the user is attempting to call RPC functions which require the password
to unlock the wallet, an error will be returned unless they call
walletpassphrase <passphrase> <time to keep key in memory> first.
A keypoolrefill command has been added which tops up the users keypool
(requiring the passphrase via walletpassphrase first).
keypoolsize has been added to the output of getinfo to show the user the
number of keys left before they need to specify their passphrase (and call
keypoolrefill).
Note that walletpassphrase will automatically fill keypool in a separate
thread which it spawns when the passphrase is set. This could cause some
delays in other threads waiting for locks on the wallet passphrase, including
one which could cause the passphrase to be stored longer than expected,
however it will not allow the passphrase to be used longer than expected as
ThreadCleanWalletPassphrase will attempt to get a lock on the key as soon
as the specified lock time has arrived.
When the keypool runs out (and wallet is locked) GetOrReuseKeyFromPool
returns vchDefaultKey, meaning miners may start to generate many blocks to
vchDefaultKey instead of a new key each time.
A walletpassphrasechange <oldpassphrase> <newpassphrase> has been added to
allow the user to change their password via RPC.
Whenever keying material (unencrypted private keys, the user's passphrase,
the wallet's AES key) is stored unencrypted in memory, any reasonable attempt
is made to mlock/VirtualLock that memory before storing the keying material.
This is not true in several (commented) cases where mlock/VirtualLocking the
memory is not possible.
Although encryption of private keys in memory can be very useful on desktop
systems (as some small amount of protection against stupid viruses), on an
RPC server, the password is entered fairly insecurely. Thus, the only main
advantage encryption has for RPC servers is for RPC servers that do not spend
coins, except in rare cases, eg. a webserver of a merchant which only receives
payment except for cases of manual intervention.
Thanks to jgarzik for the original patch and sipa, gmaxwell and many others
for all their input.
Conflicts:
src/wallet.cpp
2011-07-08 15:47:35 +02:00
2016-11-10 08:00:05 +01:00
_vMasterKey . resize ( WALLET_CRYPTO_KEY_SIZE ) ;
GetStrongRandBytes ( & _vMasterKey [ 0 ] , WALLET_CRYPTO_KEY_SIZE ) ;
Add wallet privkey encryption.
This commit adds support for ckeys, or enCrypted private keys, to the wallet.
All keys are stored in memory in their encrypted form and thus the passphrase
is required from the user to spend coins, or to create new addresses.
Keys are encrypted with AES-256-CBC using OpenSSL's EVP library. The key is
calculated via EVP_BytesToKey using SHA512 with (by default) 25000 rounds and
a random salt.
By default, the user's wallet remains unencrypted until they call the RPC
command encryptwallet <passphrase> or, from the GUI menu, Options->
Encrypt Wallet.
When the user is attempting to call RPC functions which require the password
to unlock the wallet, an error will be returned unless they call
walletpassphrase <passphrase> <time to keep key in memory> first.
A keypoolrefill command has been added which tops up the users keypool
(requiring the passphrase via walletpassphrase first).
keypoolsize has been added to the output of getinfo to show the user the
number of keys left before they need to specify their passphrase (and call
keypoolrefill).
Note that walletpassphrase will automatically fill keypool in a separate
thread which it spawns when the passphrase is set. This could cause some
delays in other threads waiting for locks on the wallet passphrase, including
one which could cause the passphrase to be stored longer than expected,
however it will not allow the passphrase to be used longer than expected as
ThreadCleanWalletPassphrase will attempt to get a lock on the key as soon
as the specified lock time has arrived.
When the keypool runs out (and wallet is locked) GetOrReuseKeyFromPool
returns vchDefaultKey, meaning miners may start to generate many blocks to
vchDefaultKey instead of a new key each time.
A walletpassphrasechange <oldpassphrase> <newpassphrase> has been added to
allow the user to change their password via RPC.
Whenever keying material (unencrypted private keys, the user's passphrase,
the wallet's AES key) is stored unencrypted in memory, any reasonable attempt
is made to mlock/VirtualLock that memory before storing the keying material.
This is not true in several (commented) cases where mlock/VirtualLocking the
memory is not possible.
Although encryption of private keys in memory can be very useful on desktop
systems (as some small amount of protection against stupid viruses), on an
RPC server, the password is entered fairly insecurely. Thus, the only main
advantage encryption has for RPC servers is for RPC servers that do not spend
coins, except in rare cases, eg. a webserver of a merchant which only receives
payment except for cases of manual intervention.
Thanks to jgarzik for the original patch and sipa, gmaxwell and many others
for all their input.
Conflicts:
src/wallet.cpp
2011-07-08 15:47:35 +02:00
2011-08-26 14:37:23 -04:00
CMasterKey kMasterKey ;
2014-06-24 14:27:32 +02:00
2011-08-26 14:37:23 -04:00
kMasterKey . vchSalt . resize ( WALLET_CRYPTO_SALT_SIZE ) ;
2016-04-16 12:25:12 +02:00
GetStrongRandBytes ( & kMasterKey . vchSalt [ 0 ] , WALLET_CRYPTO_SALT_SIZE ) ;
Add wallet privkey encryption.
This commit adds support for ckeys, or enCrypted private keys, to the wallet.
All keys are stored in memory in their encrypted form and thus the passphrase
is required from the user to spend coins, or to create new addresses.
Keys are encrypted with AES-256-CBC using OpenSSL's EVP library. The key is
calculated via EVP_BytesToKey using SHA512 with (by default) 25000 rounds and
a random salt.
By default, the user's wallet remains unencrypted until they call the RPC
command encryptwallet <passphrase> or, from the GUI menu, Options->
Encrypt Wallet.
When the user is attempting to call RPC functions which require the password
to unlock the wallet, an error will be returned unless they call
walletpassphrase <passphrase> <time to keep key in memory> first.
A keypoolrefill command has been added which tops up the users keypool
(requiring the passphrase via walletpassphrase first).
keypoolsize has been added to the output of getinfo to show the user the
number of keys left before they need to specify their passphrase (and call
keypoolrefill).
Note that walletpassphrase will automatically fill keypool in a separate
thread which it spawns when the passphrase is set. This could cause some
delays in other threads waiting for locks on the wallet passphrase, including
one which could cause the passphrase to be stored longer than expected,
however it will not allow the passphrase to be used longer than expected as
ThreadCleanWalletPassphrase will attempt to get a lock on the key as soon
as the specified lock time has arrived.
When the keypool runs out (and wallet is locked) GetOrReuseKeyFromPool
returns vchDefaultKey, meaning miners may start to generate many blocks to
vchDefaultKey instead of a new key each time.
A walletpassphrasechange <oldpassphrase> <newpassphrase> has been added to
allow the user to change their password via RPC.
Whenever keying material (unencrypted private keys, the user's passphrase,
the wallet's AES key) is stored unencrypted in memory, any reasonable attempt
is made to mlock/VirtualLock that memory before storing the keying material.
This is not true in several (commented) cases where mlock/VirtualLocking the
memory is not possible.
Although encryption of private keys in memory can be very useful on desktop
systems (as some small amount of protection against stupid viruses), on an
RPC server, the password is entered fairly insecurely. Thus, the only main
advantage encryption has for RPC servers is for RPC servers that do not spend
coins, except in rare cases, eg. a webserver of a merchant which only receives
payment except for cases of manual intervention.
Thanks to jgarzik for the original patch and sipa, gmaxwell and many others
for all their input.
Conflicts:
src/wallet.cpp
2011-07-08 15:47:35 +02:00
2011-08-26 14:37:23 -04:00
CCrypter crypter ;
2013-04-13 00:13:08 -05:00
int64_t nStartTime = GetTimeMillis ( ) ;
2011-08-26 14:37:23 -04:00
crypter . SetKeyFromPassphrase ( strWalletPassphrase , kMasterKey . vchSalt , 25000 , kMasterKey . nDerivationMethod ) ;
2017-09-11 15:43:49 -04:00
kMasterKey . nDeriveIterations = static_cast < unsigned int > ( 2500000 / ( ( double ) ( GetTimeMillis ( ) - nStartTime ) ) ) ;
2011-06-28 15:31:09 +02:00
2011-08-26 14:37:23 -04:00
nStartTime = GetTimeMillis ( ) ;
crypter . SetKeyFromPassphrase ( strWalletPassphrase , kMasterKey . vchSalt , kMasterKey . nDeriveIterations , kMasterKey . nDerivationMethod ) ;
2017-09-11 15:43:49 -04:00
kMasterKey . nDeriveIterations = ( kMasterKey . nDeriveIterations + static_cast < unsigned int > ( kMasterKey . nDeriveIterations * 100 / ( ( double ) ( GetTimeMillis ( ) - nStartTime ) ) ) ) / 2 ;
2011-06-28 15:31:09 +02:00
2011-08-26 14:37:23 -04:00
if ( kMasterKey . nDeriveIterations < 25000 )
kMasterKey . nDeriveIterations = 25000 ;
2011-06-28 15:31:09 +02:00
2018-06-15 19:02:52 -04:00
WalletLogPrintf ( " Encrypting Wallet with an nDeriveIterations of %i \n " , kMasterKey . nDeriveIterations ) ;
2011-06-28 15:31:09 +02:00
2011-08-26 14:37:23 -04:00
if ( ! crypter . SetKeyFromPassphrase ( strWalletPassphrase , kMasterKey . vchSalt , kMasterKey . nDeriveIterations , kMasterKey . nDerivationMethod ) )
return false ;
2016-11-10 08:00:05 +01:00
if ( ! crypter . Encrypt ( _vMasterKey , kMasterKey . vchCryptedKey ) )
2011-08-26 14:37:23 -04:00
return false ;
Add wallet privkey encryption.
This commit adds support for ckeys, or enCrypted private keys, to the wallet.
All keys are stored in memory in their encrypted form and thus the passphrase
is required from the user to spend coins, or to create new addresses.
Keys are encrypted with AES-256-CBC using OpenSSL's EVP library. The key is
calculated via EVP_BytesToKey using SHA512 with (by default) 25000 rounds and
a random salt.
By default, the user's wallet remains unencrypted until they call the RPC
command encryptwallet <passphrase> or, from the GUI menu, Options->
Encrypt Wallet.
When the user is attempting to call RPC functions which require the password
to unlock the wallet, an error will be returned unless they call
walletpassphrase <passphrase> <time to keep key in memory> first.
A keypoolrefill command has been added which tops up the users keypool
(requiring the passphrase via walletpassphrase first).
keypoolsize has been added to the output of getinfo to show the user the
number of keys left before they need to specify their passphrase (and call
keypoolrefill).
Note that walletpassphrase will automatically fill keypool in a separate
thread which it spawns when the passphrase is set. This could cause some
delays in other threads waiting for locks on the wallet passphrase, including
one which could cause the passphrase to be stored longer than expected,
however it will not allow the passphrase to be used longer than expected as
ThreadCleanWalletPassphrase will attempt to get a lock on the key as soon
as the specified lock time has arrived.
When the keypool runs out (and wallet is locked) GetOrReuseKeyFromPool
returns vchDefaultKey, meaning miners may start to generate many blocks to
vchDefaultKey instead of a new key each time.
A walletpassphrasechange <oldpassphrase> <newpassphrase> has been added to
allow the user to change their password via RPC.
Whenever keying material (unencrypted private keys, the user's passphrase,
the wallet's AES key) is stored unencrypted in memory, any reasonable attempt
is made to mlock/VirtualLock that memory before storing the keying material.
This is not true in several (commented) cases where mlock/VirtualLocking the
memory is not possible.
Although encryption of private keys in memory can be very useful on desktop
systems (as some small amount of protection against stupid viruses), on an
RPC server, the password is entered fairly insecurely. Thus, the only main
advantage encryption has for RPC servers is for RPC servers that do not spend
coins, except in rare cases, eg. a webserver of a merchant which only receives
payment except for cases of manual intervention.
Thanks to jgarzik for the original patch and sipa, gmaxwell and many others
for all their input.
Conflicts:
src/wallet.cpp
2011-07-08 15:47:35 +02:00
2011-08-26 14:37:23 -04:00
{
2012-04-06 18:39:12 +02:00
LOCK ( cs_wallet ) ;
Add wallet privkey encryption.
This commit adds support for ckeys, or enCrypted private keys, to the wallet.
All keys are stored in memory in their encrypted form and thus the passphrase
is required from the user to spend coins, or to create new addresses.
Keys are encrypted with AES-256-CBC using OpenSSL's EVP library. The key is
calculated via EVP_BytesToKey using SHA512 with (by default) 25000 rounds and
a random salt.
By default, the user's wallet remains unencrypted until they call the RPC
command encryptwallet <passphrase> or, from the GUI menu, Options->
Encrypt Wallet.
When the user is attempting to call RPC functions which require the password
to unlock the wallet, an error will be returned unless they call
walletpassphrase <passphrase> <time to keep key in memory> first.
A keypoolrefill command has been added which tops up the users keypool
(requiring the passphrase via walletpassphrase first).
keypoolsize has been added to the output of getinfo to show the user the
number of keys left before they need to specify their passphrase (and call
keypoolrefill).
Note that walletpassphrase will automatically fill keypool in a separate
thread which it spawns when the passphrase is set. This could cause some
delays in other threads waiting for locks on the wallet passphrase, including
one which could cause the passphrase to be stored longer than expected,
however it will not allow the passphrase to be used longer than expected as
ThreadCleanWalletPassphrase will attempt to get a lock on the key as soon
as the specified lock time has arrived.
When the keypool runs out (and wallet is locked) GetOrReuseKeyFromPool
returns vchDefaultKey, meaning miners may start to generate many blocks to
vchDefaultKey instead of a new key each time.
A walletpassphrasechange <oldpassphrase> <newpassphrase> has been added to
allow the user to change their password via RPC.
Whenever keying material (unencrypted private keys, the user's passphrase,
the wallet's AES key) is stored unencrypted in memory, any reasonable attempt
is made to mlock/VirtualLock that memory before storing the keying material.
This is not true in several (commented) cases where mlock/VirtualLocking the
memory is not possible.
Although encryption of private keys in memory can be very useful on desktop
systems (as some small amount of protection against stupid viruses), on an
RPC server, the password is entered fairly insecurely. Thus, the only main
advantage encryption has for RPC servers is for RPC servers that do not spend
coins, except in rare cases, eg. a webserver of a merchant which only receives
payment except for cases of manual intervention.
Thanks to jgarzik for the original patch and sipa, gmaxwell and many others
for all their input.
Conflicts:
src/wallet.cpp
2011-07-08 15:47:35 +02:00
mapMasterKeys [ + + nMasterKeyMaxID ] = kMasterKey ;
2019-12-05 18:23:57 -05:00
WalletBatch * encrypted_batch = new WalletBatch ( * database ) ;
2017-12-08 06:39:22 -05:00
if ( ! encrypted_batch - > TxnBegin ( ) ) {
delete encrypted_batch ;
encrypted_batch = nullptr ;
2017-03-08 12:08:26 +00:00
return false ;
Add wallet privkey encryption.
This commit adds support for ckeys, or enCrypted private keys, to the wallet.
All keys are stored in memory in their encrypted form and thus the passphrase
is required from the user to spend coins, or to create new addresses.
Keys are encrypted with AES-256-CBC using OpenSSL's EVP library. The key is
calculated via EVP_BytesToKey using SHA512 with (by default) 25000 rounds and
a random salt.
By default, the user's wallet remains unencrypted until they call the RPC
command encryptwallet <passphrase> or, from the GUI menu, Options->
Encrypt Wallet.
When the user is attempting to call RPC functions which require the password
to unlock the wallet, an error will be returned unless they call
walletpassphrase <passphrase> <time to keep key in memory> first.
A keypoolrefill command has been added which tops up the users keypool
(requiring the passphrase via walletpassphrase first).
keypoolsize has been added to the output of getinfo to show the user the
number of keys left before they need to specify their passphrase (and call
keypoolrefill).
Note that walletpassphrase will automatically fill keypool in a separate
thread which it spawns when the passphrase is set. This could cause some
delays in other threads waiting for locks on the wallet passphrase, including
one which could cause the passphrase to be stored longer than expected,
however it will not allow the passphrase to be used longer than expected as
ThreadCleanWalletPassphrase will attempt to get a lock on the key as soon
as the specified lock time has arrived.
When the keypool runs out (and wallet is locked) GetOrReuseKeyFromPool
returns vchDefaultKey, meaning miners may start to generate many blocks to
vchDefaultKey instead of a new key each time.
A walletpassphrasechange <oldpassphrase> <newpassphrase> has been added to
allow the user to change their password via RPC.
Whenever keying material (unencrypted private keys, the user's passphrase,
the wallet's AES key) is stored unencrypted in memory, any reasonable attempt
is made to mlock/VirtualLock that memory before storing the keying material.
This is not true in several (commented) cases where mlock/VirtualLocking the
memory is not possible.
Although encryption of private keys in memory can be very useful on desktop
systems (as some small amount of protection against stupid viruses), on an
RPC server, the password is entered fairly insecurely. Thus, the only main
advantage encryption has for RPC servers is for RPC servers that do not spend
coins, except in rare cases, eg. a webserver of a merchant which only receives
payment except for cases of manual intervention.
Thanks to jgarzik for the original patch and sipa, gmaxwell and many others
for all their input.
Conflicts:
src/wallet.cpp
2011-07-08 15:47:35 +02:00
}
2017-12-08 06:39:22 -05:00
encrypted_batch - > WriteMasterKey ( nMasterKeyMaxID , kMasterKey ) ;
Add wallet privkey encryption.
This commit adds support for ckeys, or enCrypted private keys, to the wallet.
All keys are stored in memory in their encrypted form and thus the passphrase
is required from the user to spend coins, or to create new addresses.
Keys are encrypted with AES-256-CBC using OpenSSL's EVP library. The key is
calculated via EVP_BytesToKey using SHA512 with (by default) 25000 rounds and
a random salt.
By default, the user's wallet remains unencrypted until they call the RPC
command encryptwallet <passphrase> or, from the GUI menu, Options->
Encrypt Wallet.
When the user is attempting to call RPC functions which require the password
to unlock the wallet, an error will be returned unless they call
walletpassphrase <passphrase> <time to keep key in memory> first.
A keypoolrefill command has been added which tops up the users keypool
(requiring the passphrase via walletpassphrase first).
keypoolsize has been added to the output of getinfo to show the user the
number of keys left before they need to specify their passphrase (and call
keypoolrefill).
Note that walletpassphrase will automatically fill keypool in a separate
thread which it spawns when the passphrase is set. This could cause some
delays in other threads waiting for locks on the wallet passphrase, including
one which could cause the passphrase to be stored longer than expected,
however it will not allow the passphrase to be used longer than expected as
ThreadCleanWalletPassphrase will attempt to get a lock on the key as soon
as the specified lock time has arrived.
When the keypool runs out (and wallet is locked) GetOrReuseKeyFromPool
returns vchDefaultKey, meaning miners may start to generate many blocks to
vchDefaultKey instead of a new key each time.
A walletpassphrasechange <oldpassphrase> <newpassphrase> has been added to
allow the user to change their password via RPC.
Whenever keying material (unencrypted private keys, the user's passphrase,
the wallet's AES key) is stored unencrypted in memory, any reasonable attempt
is made to mlock/VirtualLock that memory before storing the keying material.
This is not true in several (commented) cases where mlock/VirtualLocking the
memory is not possible.
Although encryption of private keys in memory can be very useful on desktop
systems (as some small amount of protection against stupid viruses), on an
RPC server, the password is entered fairly insecurely. Thus, the only main
advantage encryption has for RPC servers is for RPC servers that do not spend
coins, except in rare cases, eg. a webserver of a merchant which only receives
payment except for cases of manual intervention.
Thanks to jgarzik for the original patch and sipa, gmaxwell and many others
for all their input.
Conflicts:
src/wallet.cpp
2011-07-08 15:47:35 +02:00
2019-10-07 14:11:34 -04:00
for ( const auto & spk_man_pair : m_spk_managers ) {
auto spk_man = spk_man_pair . second . get ( ) ;
2019-12-05 18:23:57 -05:00
if ( ! spk_man - > Encrypt ( _vMasterKey , encrypted_batch ) ) {
2019-10-07 14:11:34 -04:00
encrypted_batch - > TxnAbort ( ) ;
delete encrypted_batch ;
encrypted_batch = nullptr ;
// We now probably have half of our keys encrypted in memory, and half not...
// die and let the user reload the unencrypted wallet.
assert ( false ) ;
}
2011-07-08 15:08:27 +02:00
}
2012-02-18 14:55:02 +01:00
// Encryption was introduced in version 0.4.0
2017-12-08 06:39:22 -05:00
SetMinVersion ( FEATURE_WALLETCRYPT , encrypted_batch , true ) ;
2012-02-18 14:55:02 +01:00
2017-12-08 06:39:22 -05:00
if ( ! encrypted_batch - > TxnCommit ( ) ) {
delete encrypted_batch ;
2018-09-03 15:41:02 +02:00
encrypted_batch = nullptr ;
2017-03-08 12:08:26 +00:00
// We now have keys encrypted in memory, but not on disk...
// die to avoid confusion and let the user reload the unencrypted wallet.
assert ( false ) ;
2011-07-08 15:08:27 +02:00
}
Add wallet privkey encryption.
This commit adds support for ckeys, or enCrypted private keys, to the wallet.
All keys are stored in memory in their encrypted form and thus the passphrase
is required from the user to spend coins, or to create new addresses.
Keys are encrypted with AES-256-CBC using OpenSSL's EVP library. The key is
calculated via EVP_BytesToKey using SHA512 with (by default) 25000 rounds and
a random salt.
By default, the user's wallet remains unencrypted until they call the RPC
command encryptwallet <passphrase> or, from the GUI menu, Options->
Encrypt Wallet.
When the user is attempting to call RPC functions which require the password
to unlock the wallet, an error will be returned unless they call
walletpassphrase <passphrase> <time to keep key in memory> first.
A keypoolrefill command has been added which tops up the users keypool
(requiring the passphrase via walletpassphrase first).
keypoolsize has been added to the output of getinfo to show the user the
number of keys left before they need to specify their passphrase (and call
keypoolrefill).
Note that walletpassphrase will automatically fill keypool in a separate
thread which it spawns when the passphrase is set. This could cause some
delays in other threads waiting for locks on the wallet passphrase, including
one which could cause the passphrase to be stored longer than expected,
however it will not allow the passphrase to be used longer than expected as
ThreadCleanWalletPassphrase will attempt to get a lock on the key as soon
as the specified lock time has arrived.
When the keypool runs out (and wallet is locked) GetOrReuseKeyFromPool
returns vchDefaultKey, meaning miners may start to generate many blocks to
vchDefaultKey instead of a new key each time.
A walletpassphrasechange <oldpassphrase> <newpassphrase> has been added to
allow the user to change their password via RPC.
Whenever keying material (unencrypted private keys, the user's passphrase,
the wallet's AES key) is stored unencrypted in memory, any reasonable attempt
is made to mlock/VirtualLock that memory before storing the keying material.
This is not true in several (commented) cases where mlock/VirtualLocking the
memory is not possible.
Although encryption of private keys in memory can be very useful on desktop
systems (as some small amount of protection against stupid viruses), on an
RPC server, the password is entered fairly insecurely. Thus, the only main
advantage encryption has for RPC servers is for RPC servers that do not spend
coins, except in rare cases, eg. a webserver of a merchant which only receives
payment except for cases of manual intervention.
Thanks to jgarzik for the original patch and sipa, gmaxwell and many others
for all their input.
Conflicts:
src/wallet.cpp
2011-07-08 15:47:35 +02:00
2017-12-08 06:39:22 -05:00
delete encrypted_batch ;
encrypted_batch = nullptr ;
2017-03-08 12:08:26 +00:00
2011-11-17 14:01:25 -05:00
Lock ( ) ;
Unlock ( strWalletPassphrase ) ;
2016-07-21 21:19:02 +02:00
2018-04-04 12:47:55 -04:00
// if we are using HD, replace the HD seed with a new one
2019-10-07 14:11:34 -04:00
if ( auto spk_man = GetLegacyScriptPubKeyMan ( ) ) {
2019-10-07 14:11:34 -04:00
if ( spk_man - > IsHDEnabled ( ) ) {
2019-11-05 10:53:07 -05:00
if ( ! spk_man - > SetupGeneration ( true ) ) {
return false ;
}
2019-10-07 14:11:34 -04:00
}
2016-07-21 21:19:02 +02:00
}
Add wallet privkey encryption.
This commit adds support for ckeys, or enCrypted private keys, to the wallet.
All keys are stored in memory in their encrypted form and thus the passphrase
is required from the user to spend coins, or to create new addresses.
Keys are encrypted with AES-256-CBC using OpenSSL's EVP library. The key is
calculated via EVP_BytesToKey using SHA512 with (by default) 25000 rounds and
a random salt.
By default, the user's wallet remains unencrypted until they call the RPC
command encryptwallet <passphrase> or, from the GUI menu, Options->
Encrypt Wallet.
When the user is attempting to call RPC functions which require the password
to unlock the wallet, an error will be returned unless they call
walletpassphrase <passphrase> <time to keep key in memory> first.
A keypoolrefill command has been added which tops up the users keypool
(requiring the passphrase via walletpassphrase first).
keypoolsize has been added to the output of getinfo to show the user the
number of keys left before they need to specify their passphrase (and call
keypoolrefill).
Note that walletpassphrase will automatically fill keypool in a separate
thread which it spawns when the passphrase is set. This could cause some
delays in other threads waiting for locks on the wallet passphrase, including
one which could cause the passphrase to be stored longer than expected,
however it will not allow the passphrase to be used longer than expected as
ThreadCleanWalletPassphrase will attempt to get a lock on the key as soon
as the specified lock time has arrived.
When the keypool runs out (and wallet is locked) GetOrReuseKeyFromPool
returns vchDefaultKey, meaning miners may start to generate many blocks to
vchDefaultKey instead of a new key each time.
A walletpassphrasechange <oldpassphrase> <newpassphrase> has been added to
allow the user to change their password via RPC.
Whenever keying material (unencrypted private keys, the user's passphrase,
the wallet's AES key) is stored unencrypted in memory, any reasonable attempt
is made to mlock/VirtualLock that memory before storing the keying material.
This is not true in several (commented) cases where mlock/VirtualLocking the
memory is not possible.
Although encryption of private keys in memory can be very useful on desktop
systems (as some small amount of protection against stupid viruses), on an
RPC server, the password is entered fairly insecurely. Thus, the only main
advantage encryption has for RPC servers is for RPC servers that do not spend
coins, except in rare cases, eg. a webserver of a merchant which only receives
payment except for cases of manual intervention.
Thanks to jgarzik for the original patch and sipa, gmaxwell and many others
for all their input.
Conflicts:
src/wallet.cpp
2011-07-08 15:47:35 +02:00
Lock ( ) ;
2011-08-26 14:37:23 -04:00
2011-11-10 21:12:46 -05:00
// Need to completely rewrite the wallet file; if we don't, bdb might keep
// bits of the unencrypted private key in slack space in the database file.
2017-12-08 06:39:22 -05:00
database - > Rewrite ( ) ;
2012-05-05 16:07:14 +02:00
2018-02-20 16:08:36 -05:00
// BDB seems to have a bad habit of writing old data into
// slack space in .dat files; that is bad if the old data is
// unencrypted private keys. So:
database - > ReloadDbEnv ( ) ;
2011-11-10 21:12:46 -05:00
}
2012-05-06 19:40:58 +02:00
NotifyStatusChanged ( this ) ;
2011-11-10 21:29:23 +01:00
Add wallet privkey encryption.
This commit adds support for ckeys, or enCrypted private keys, to the wallet.
All keys are stored in memory in their encrypted form and thus the passphrase
is required from the user to spend coins, or to create new addresses.
Keys are encrypted with AES-256-CBC using OpenSSL's EVP library. The key is
calculated via EVP_BytesToKey using SHA512 with (by default) 25000 rounds and
a random salt.
By default, the user's wallet remains unencrypted until they call the RPC
command encryptwallet <passphrase> or, from the GUI menu, Options->
Encrypt Wallet.
When the user is attempting to call RPC functions which require the password
to unlock the wallet, an error will be returned unless they call
walletpassphrase <passphrase> <time to keep key in memory> first.
A keypoolrefill command has been added which tops up the users keypool
(requiring the passphrase via walletpassphrase first).
keypoolsize has been added to the output of getinfo to show the user the
number of keys left before they need to specify their passphrase (and call
keypoolrefill).
Note that walletpassphrase will automatically fill keypool in a separate
thread which it spawns when the passphrase is set. This could cause some
delays in other threads waiting for locks on the wallet passphrase, including
one which could cause the passphrase to be stored longer than expected,
however it will not allow the passphrase to be used longer than expected as
ThreadCleanWalletPassphrase will attempt to get a lock on the key as soon
as the specified lock time has arrived.
When the keypool runs out (and wallet is locked) GetOrReuseKeyFromPool
returns vchDefaultKey, meaning miners may start to generate many blocks to
vchDefaultKey instead of a new key each time.
A walletpassphrasechange <oldpassphrase> <newpassphrase> has been added to
allow the user to change their password via RPC.
Whenever keying material (unencrypted private keys, the user's passphrase,
the wallet's AES key) is stored unencrypted in memory, any reasonable attempt
is made to mlock/VirtualLock that memory before storing the keying material.
This is not true in several (commented) cases where mlock/VirtualLocking the
memory is not possible.
Although encryption of private keys in memory can be very useful on desktop
systems (as some small amount of protection against stupid viruses), on an
RPC server, the password is entered fairly insecurely. Thus, the only main
advantage encryption has for RPC servers is for RPC servers that do not spend
coins, except in rare cases, eg. a webserver of a merchant which only receives
payment except for cases of manual intervention.
Thanks to jgarzik for the original patch and sipa, gmaxwell and many others
for all their input.
Conflicts:
src/wallet.cpp
2011-07-08 15:47:35 +02:00
return true ;
2011-06-26 19:23:24 +02:00
}
2016-09-09 19:21:44 -07:00
DBErrors CWallet : : ReorderTransactions ( )
{
2016-09-28 08:57:25 -07:00
LOCK ( cs_wallet ) ;
2017-12-08 06:39:22 -05:00
WalletBatch batch ( * database ) ;
2016-09-28 08:57:25 -07:00
// Old wallets didn't have any defined order for transactions
// Probably a bad idea to change the output of this
2018-07-31 12:23:26 -04:00
// First: get all CWalletTx into a sorted-by-time multimap.
typedef std : : multimap < int64_t , CWalletTx * > TxItems ;
2016-09-28 08:57:25 -07:00
TxItems txByTime ;
2017-06-04 22:02:43 +02:00
for ( auto & entry : mapWallet )
2016-09-28 08:57:25 -07:00
{
2017-06-04 22:02:43 +02:00
CWalletTx * wtx = & entry . second ;
2018-07-31 12:23:26 -04:00
txByTime . insert ( std : : make_pair ( wtx - > nTimeReceived , wtx ) ) ;
2016-09-28 08:57:25 -07:00
}
nOrderPosNext = 0 ;
std : : vector < int64_t > nOrderPosOffsets ;
for ( TxItems : : iterator it = txByTime . begin ( ) ; it ! = txByTime . end ( ) ; + + it )
{
2018-07-31 12:23:26 -04:00
CWalletTx * const pwtx = ( * it ) . second ;
int64_t & nOrderPos = pwtx - > nOrderPos ;
2016-09-28 08:57:25 -07:00
if ( nOrderPos = = - 1 )
{
nOrderPos = nOrderPosNext + + ;
nOrderPosOffsets . push_back ( nOrderPos ) ;
2018-07-31 12:23:26 -04:00
if ( ! batch . WriteTx ( * pwtx ) )
return DBErrors : : LOAD_FAIL ;
2016-09-28 08:57:25 -07:00
}
else
{
int64_t nOrderPosOff = 0 ;
2017-06-02 03:18:57 +02:00
for ( const int64_t & nOffsetStart : nOrderPosOffsets )
2016-09-28 08:57:25 -07:00
{
if ( nOrderPos > = nOffsetStart )
+ + nOrderPosOff ;
}
nOrderPos + = nOrderPosOff ;
nOrderPosNext = std : : max ( nOrderPosNext , nOrderPos + 1 ) ;
if ( ! nOrderPosOff )
continue ;
// Since we're changing the order, write it back
2018-07-31 12:23:26 -04:00
if ( ! batch . WriteTx ( * pwtx ) )
return DBErrors : : LOAD_FAIL ;
2016-09-28 08:57:25 -07:00
}
}
2017-12-08 06:39:22 -05:00
batch . WriteOrderPosNext ( nOrderPosNext ) ;
2016-09-28 08:57:25 -07:00
2018-03-09 15:03:40 +01:00
return DBErrors : : LOAD_OK ;
2016-09-09 19:21:44 -07:00
}
2019-02-18 16:09:45 -05:00
int64_t CWallet : : IncOrderPosNext ( WalletBatch * batch )
2012-09-08 04:55:36 +00:00
{
2019-02-18 16:09:45 -05:00
AssertLockHeld ( cs_wallet ) ;
2013-04-13 00:13:08 -05:00
int64_t nRet = nOrderPosNext + + ;
2017-12-08 06:39:22 -05:00
if ( batch ) {
batch - > WriteOrderPosNext ( nOrderPosNext ) ;
2012-11-13 23:52:37 +01:00
} else {
2017-12-08 06:39:22 -05:00
WalletBatch ( * database ) . WriteOrderPosNext ( nOrderPosNext ) ;
2012-11-13 23:52:37 +01:00
}
2012-09-08 04:55:36 +00:00
return nRet ;
}
2011-07-13 11:56:38 +02:00
void CWallet : : MarkDirty ( )
{
{
2012-04-06 18:39:12 +02:00
LOCK ( cs_wallet ) ;
2017-06-02 03:28:42 +02:00
for ( std : : pair < const uint256 , CWalletTx > & item : mapWallet )
2011-07-13 11:56:38 +02:00
item . second . MarkDirty ( ) ;
}
}
2016-12-09 13:45:27 -05:00
bool CWallet : : MarkReplaced ( const uint256 & originalHash , const uint256 & newHash )
{
LOCK ( cs_wallet ) ;
auto mi = mapWallet . find ( originalHash ) ;
// There is a bug if MarkReplaced is not called on an existing wallet transaction.
assert ( mi ! = mapWallet . end ( ) ) ;
CWalletTx & wtx = ( * mi ) . second ;
// Ensure for now that we're not overwriting data
assert ( wtx . mapValue . count ( " replaced_by_txid " ) = = 0 ) ;
wtx . mapValue [ " replaced_by_txid " ] = newHash . ToString ( ) ;
2017-12-08 06:39:22 -05:00
WalletBatch batch ( * database , " r+ " ) ;
2016-12-09 13:45:27 -05:00
bool success = true ;
2017-12-08 06:39:22 -05:00
if ( ! batch . WriteTx ( wtx ) ) {
2018-06-15 19:02:52 -04:00
WalletLogPrintf ( " %s: Updating batch tx %s failed \n " , __func__ , wtx . GetHash ( ) . ToString ( ) ) ;
2016-12-09 13:45:27 -05:00
success = false ;
}
NotifyTransactionChanged ( this , originalHash , CT_UPDATED ) ;
return success ;
}
2020-02-21 21:15:13 +00:00
void CWallet : : SetSpentKeyState ( WalletBatch & batch , const uint256 & hash , unsigned int n , bool used , std : : set < CTxDestination > & tx_destinations )
2018-09-13 13:53:19 +09:00
{
2019-11-02 16:14:36 +00:00
AssertLockHeld ( cs_wallet ) ;
2018-09-13 13:53:19 +09:00
const CWalletTx * srctx = GetWalletTx ( hash ) ;
if ( ! srctx ) return ;
CTxDestination dst ;
if ( ExtractDestination ( srctx - > tx - > vout [ n ] . scriptPubKey , dst ) ) {
2019-10-07 14:11:34 -04:00
if ( IsMine ( dst ) ) {
2018-09-13 13:53:19 +09:00
if ( used & & ! GetDestData ( dst , " used " , nullptr ) ) {
2019-12-31 19:55:18 +01:00
if ( AddDestData ( batch , dst , " used " , " p " ) ) { // p for "present", opposite of absent (null)
tx_destinations . insert ( dst ) ;
}
2018-09-13 13:53:19 +09:00
} else if ( ! used & & GetDestData ( dst , " used " , nullptr ) ) {
2019-11-02 16:20:45 +00:00
EraseDestData ( batch , dst , " used " ) ;
2018-09-13 13:53:19 +09:00
}
}
}
}
2020-02-21 21:15:13 +00:00
bool CWallet : : IsSpentKey ( const uint256 & hash , unsigned int n ) const
2018-09-13 13:53:19 +09:00
{
2019-11-27 10:56:04 -05:00
AssertLockHeld ( cs_wallet ) ;
2018-09-13 13:53:19 +09:00
CTxDestination dst ;
const CWalletTx * srctx = GetWalletTx ( hash ) ;
2019-11-27 10:56:04 -05:00
if ( srctx ) {
assert ( srctx - > tx - > vout . size ( ) > n ) ;
LegacyScriptPubKeyMan * spk_man = GetLegacyScriptPubKeyMan ( ) ;
// When descriptor wallets arrive, these additional checks are
// likely superfluous and can be optimized out
assert ( spk_man ! = nullptr ) ;
for ( const auto & keyid : GetAffectedKeys ( srctx - > tx - > vout [ n ] . scriptPubKey , * spk_man ) ) {
WitnessV0KeyHash wpkh_dest ( keyid ) ;
if ( GetDestData ( wpkh_dest , " used " , nullptr ) ) {
return true ;
}
2020-01-14 13:23:24 -05:00
ScriptHash sh_wpkh_dest ( GetScriptForDestination ( wpkh_dest ) ) ;
2019-11-27 10:56:04 -05:00
if ( GetDestData ( sh_wpkh_dest , " used " , nullptr ) ) {
return true ;
}
PKHash pkh_dest ( keyid ) ;
if ( GetDestData ( pkh_dest , " used " , nullptr ) ) {
return true ;
}
}
}
return false ;
2018-09-13 13:53:19 +09:00
}
2016-06-07 21:41:03 -07:00
bool CWallet : : AddToWallet ( const CWalletTx & wtxIn , bool fFlushOnClose )
2011-06-26 19:23:24 +02:00
{
2016-06-07 21:41:03 -07:00
LOCK ( cs_wallet ) ;
2017-12-08 06:39:22 -05:00
WalletBatch batch ( * database , " r+ " , fFlushOnClose ) ;
2016-06-07 21:41:03 -07:00
2011-06-26 19:23:24 +02:00
uint256 hash = wtxIn . GetHash ( ) ;
2014-02-13 20:12:51 -05:00
2018-09-13 13:53:19 +09:00
if ( IsWalletFlagSet ( WALLET_FLAG_AVOID_REUSE ) ) {
// Mark used destinations
2019-12-31 19:55:18 +01:00
std : : set < CTxDestination > tx_destinations ;
2018-09-13 13:53:19 +09:00
for ( const CTxIn & txin : wtxIn . tx - > vin ) {
const COutPoint & op = txin . prevout ;
2020-02-21 21:15:13 +00:00
SetSpentKeyState ( batch , op . hash , op . n , true , tx_destinations ) ;
2018-09-13 13:53:19 +09:00
}
2019-12-31 19:55:18 +01:00
MarkDestinationsDirty ( tx_destinations ) ;
2018-09-13 13:53:19 +09:00
}
2016-06-07 21:25:31 -07:00
// Inserts only if not already there, returns tx inserted or tx found
2017-01-27 10:33:45 +09:00
std : : pair < std : : map < uint256 , CWalletTx > : : iterator , bool > ret = mapWallet . insert ( std : : make_pair ( hash , wtxIn ) ) ;
2016-06-07 21:25:31 -07:00
CWalletTx & wtx = ( * ret . first ) . second ;
wtx . BindWallet ( this ) ;
bool fInsertedNew = ret . second ;
2018-06-11 14:09:16 -04:00
if ( fInsertedNew ) {
2017-07-28 22:54:31 -04:00
wtx . nTimeReceived = chain ( ) . getAdjustedTime ( ) ;
2017-12-08 06:39:22 -05:00
wtx . nOrderPos = IncOrderPosNext ( & batch ) ;
2018-07-31 12:23:26 -04:00
wtx . m_it_wtxOrdered = wtxOrdered . insert ( std : : make_pair ( wtx . nOrderPos , & wtx ) ) ;
2016-12-16 10:00:26 -05:00
wtx . nTimeSmart = ComputeTimeSmart ( wtx ) ;
2016-06-07 21:25:31 -07:00
AddToSpends ( hash ) ;
}
2011-06-26 19:23:24 +02:00
2016-06-07 21:25:31 -07:00
bool fUpdated = false ;
if ( ! fInsertedNew )
{
2019-08-12 18:12:12 -04:00
if ( wtxIn . m_confirm . status ! = wtx . m_confirm . status ) {
wtx . m_confirm . status = wtxIn . m_confirm . status ;
wtx . m_confirm . nIndex = wtxIn . m_confirm . nIndex ;
wtx . m_confirm . hashBlock = wtxIn . m_confirm . hashBlock ;
2019-04-20 11:22:59 -04:00
wtx . m_confirm . block_height = wtxIn . m_confirm . block_height ;
2016-06-07 21:25:31 -07:00
fUpdated = true ;
2019-08-12 18:12:12 -04:00
} else {
assert ( wtx . m_confirm . nIndex = = wtxIn . m_confirm . nIndex ) ;
assert ( wtx . m_confirm . hashBlock = = wtxIn . m_confirm . hashBlock ) ;
2019-04-20 11:22:59 -04:00
assert ( wtx . m_confirm . block_height = = wtxIn . m_confirm . block_height ) ;
2011-06-26 19:23:24 +02:00
}
2016-06-07 21:25:31 -07:00
if ( wtxIn . fFromMe & & wtxIn . fFromMe ! = wtx . fFromMe )
{
wtx . fFromMe = wtxIn . fFromMe ;
fUpdated = true ;
}
2017-09-03 08:43:35 -04:00
// If we have a witness-stripped version of this transaction, and we
// see a new version with a witness, then we must be upgrading a pre-segwit
// wallet. Store the new version of the transaction with the witness,
// as the stripped-version must be invalid.
// TODO: Store all versions of the transaction, instead of just one.
if ( wtxIn . tx - > HasWitness ( ) & & ! wtx . tx - > HasWitness ( ) ) {
wtx . SetTx ( wtxIn . tx ) ;
fUpdated = true ;
}
2016-06-07 21:25:31 -07:00
}
2011-06-26 19:23:24 +02:00
2016-06-07 21:25:31 -07:00
//// debug print
2018-06-15 19:02:52 -04:00
WalletLogPrintf ( " AddToWallet %s %s%s \n " , wtxIn . GetHash ( ) . ToString ( ) , ( fInsertedNew ? " new " : " " ) , ( fUpdated ? " update " : " " ) ) ;
2011-06-26 19:23:24 +02:00
2016-06-07 21:25:31 -07:00
// Write to disk
if ( fInsertedNew | | fUpdated )
2017-12-08 06:39:22 -05:00
if ( ! batch . WriteTx ( wtx ) )
2016-06-07 21:25:31 -07:00
return false ;
2013-05-26 14:17:18 -04:00
2016-06-07 21:25:31 -07:00
// Break debit/credit balance caches:
wtx . MarkDirty ( ) ;
2011-06-26 19:23:24 +02:00
2016-06-07 21:25:31 -07:00
// Notify UI of new or updated transaction
NotifyTransactionChanged ( this , hash , fInsertedNew ? CT_NEW : CT_UPDATED ) ;
2012-11-03 09:58:41 -05:00
2019-07-05 18:30:15 +02:00
# if HAVE_SYSTEM
2016-06-07 21:25:31 -07:00
// notify an external script when a wallet transaction comes in or is updated
2017-08-01 21:17:40 +02:00
std : : string strCmd = gArgs . GetArg ( " -walletnotify " , " " ) ;
2012-11-03 09:58:41 -05:00
2017-07-27 01:09:05 +02:00
if ( ! strCmd . empty ( ) )
2016-06-07 21:25:31 -07:00
{
boost : : replace_all ( strCmd , " %s " , wtxIn . GetHash ( ) . GetHex ( ) ) ;
2018-05-29 14:37:53 +01:00
# ifndef WIN32
// Substituting the wallet name isn't currently supported on windows
// because windows shell escaping has not been implemented yet:
// https://github.com/bitcoin/bitcoin/pull/13339#issuecomment-537384875
// A few ways it could be implemented in the future are described in:
// https://github.com/bitcoin/bitcoin/pull/13339#issuecomment-461288094
boost : : replace_all ( strCmd , " %w " , ShellEscape ( GetName ( ) ) ) ;
# endif
2018-02-07 19:19:34 -05:00
std : : thread t ( runCommand , strCmd ) ;
t . detach ( ) ; // thread runs free
2016-06-07 21:25:31 -07:00
}
2019-03-14 11:30:37 +01:00
# endif
2016-06-07 21:25:31 -07:00
return true ;
}
2019-08-22 13:16:40 -04:00
void CWallet : : LoadToWallet ( CWalletTx & wtxIn )
2016-06-07 21:25:31 -07:00
{
2019-12-01 21:30:29 -05:00
// If wallet doesn't have a chain (e.g bitcoin-wallet), lock can't be taken.
2019-08-22 13:16:40 -04:00
auto locked_chain = LockChain ( ) ;
2019-04-20 11:22:59 -04:00
if ( locked_chain ) {
Optional < int > block_height = locked_chain - > getBlockHeight ( wtxIn . m_confirm . hashBlock ) ;
if ( block_height ) {
// Update cached block height variable since it not stored in the
// serialized transaction.
wtxIn . m_confirm . block_height = * block_height ;
} else if ( wtxIn . isConflicted ( ) | | wtxIn . isConfirmed ( ) ) {
// If tx block (or conflicting block) was reorged out of chain
// while the wallet was shutdown, change tx status to UNCONFIRMED
// and reset block height, hash, and index. ABANDONED tx don't have
// associated blocks and don't need to be updated. The case where a
// transaction was reorged out while online and then reconfirmed
// while offline is covered by the rescan logic.
2019-08-22 13:16:40 -04:00
wtxIn . setUnconfirmed ( ) ;
wtxIn . m_confirm . hashBlock = uint256 ( ) ;
2019-04-20 11:22:59 -04:00
wtxIn . m_confirm . block_height = 0 ;
2019-08-22 13:16:40 -04:00
wtxIn . m_confirm . nIndex = 0 ;
}
}
2016-06-07 21:25:31 -07:00
uint256 hash = wtxIn . GetHash ( ) ;
2018-06-11 14:09:16 -04:00
const auto & ins = mapWallet . emplace ( hash , wtxIn ) ;
CWalletTx & wtx = ins . first - > second ;
2016-06-07 21:25:31 -07:00
wtx . BindWallet ( this ) ;
2018-06-11 14:09:16 -04:00
if ( /* insertion took place */ ins . second ) {
2018-07-31 12:23:26 -04:00
wtx . m_it_wtxOrdered = wtxOrdered . insert ( std : : make_pair ( wtx . nOrderPos , & wtx ) ) ;
2018-06-11 14:09:16 -04:00
}
2016-06-07 21:25:31 -07:00
AddToSpends ( hash ) ;
2017-06-02 03:18:57 +02:00
for ( const CTxIn & txin : wtx . tx - > vin ) {
2017-08-13 15:04:57 +01:00
auto it = mapWallet . find ( txin . prevout . hash ) ;
if ( it ! = mapWallet . end ( ) ) {
CWalletTx & prevtx = it - > second ;
2019-08-12 18:12:12 -04:00
if ( prevtx . isConflicted ( ) ) {
2019-04-20 11:22:59 -04:00
MarkConflicted ( prevtx . m_confirm . hashBlock , prevtx . m_confirm . block_height , wtx . GetHash ( ) ) ;
2016-06-07 21:25:31 -07:00
}
}
2012-05-05 16:07:14 +02:00
}
2011-06-26 19:23:24 +02:00
}
2019-10-24 12:53:57 -04:00
bool CWallet : : AddToWalletIfInvolvingMe ( const CTransactionRef & ptx , CWalletTx : : Confirmation confirm , bool fUpdate )
2011-06-26 19:23:24 +02:00
{
2017-03-06 18:21:27 -05:00
const CTransaction & tx = * ptx ;
2011-06-26 19:23:24 +02:00
{
2014-03-09 12:41:22 +01:00
AssertLockHeld ( cs_wallet ) ;
2015-11-26 18:42:07 +01:00
2019-10-24 12:53:57 -04:00
if ( ! confirm . hashBlock . IsNull ( ) ) {
2017-06-02 03:18:57 +02:00
for ( const CTxIn & txin : tx . vin ) {
2015-11-26 18:42:07 +01:00
std : : pair < TxSpends : : const_iterator , TxSpends : : const_iterator > range = mapTxSpends . equal_range ( txin . prevout ) ;
while ( range . first ! = range . second ) {
if ( range . first - > second ! = tx . GetHash ( ) ) {
2019-10-24 12:53:57 -04:00
WalletLogPrintf ( " Transaction %s (in block %s) conflicts with wallet transaction %s (both spend %s:%i) \n " , tx . GetHash ( ) . ToString ( ) , confirm . hashBlock . ToString ( ) , range . first - > second . ToString ( ) , range . first - > first . hash . ToString ( ) , range . first - > first . n ) ;
2019-04-20 11:22:59 -04:00
MarkConflicted ( confirm . hashBlock , confirm . block_height , range . first - > second ) ;
2015-11-26 18:42:07 +01:00
}
range . first + + ;
}
}
}
2014-09-06 15:59:59 -04:00
bool fExisted = mapWallet . count ( tx . GetHash ( ) ) ! = 0 ;
2011-08-26 14:37:23 -04:00
if ( fExisted & & ! fUpdate ) return false ;
2014-07-17 14:09:46 +02:00
if ( fExisted | | IsMine ( tx ) | | IsFromMe ( tx ) )
2011-08-26 14:37:23 -04:00
{
2017-07-18 15:49:56 -04:00
/* Check if any keys in the wallet keypool that were supposed to be unused
* have appeared in a new transaction . If so , remove those keys from the keypool .
* This can happen when restoring an old wallet backup that does not contain
* the mostly recently created transactions from newer versions of the wallet .
*/
// loop though all outputs
for ( const CTxOut & txout : tx . vout ) {
2019-10-07 14:11:34 -04:00
for ( const auto & spk_man_pair : m_spk_managers ) {
spk_man_pair . second - > MarkUnusedAddresses ( txout . scriptPubKey ) ;
2017-07-18 15:49:56 -04:00
}
}
2017-03-06 18:21:27 -05:00
CWalletTx wtx ( this , ptx ) ;
2014-08-31 05:55:27 +02:00
2019-08-12 18:12:12 -04:00
// Block disconnection override an abandoned tx as unconfirmed
// which means user may have to call abandontransaction again
2019-10-24 12:53:57 -04:00
wtx . m_confirm = confirm ;
2014-08-31 05:55:27 +02:00
2016-06-07 21:41:03 -07:00
return AddToWallet ( wtx , false ) ;
2011-08-26 14:37:23 -04:00
}
2011-06-26 19:23:24 +02:00
}
return false ;
}
2017-04-28 14:10:21 -04:00
bool CWallet : : TransactionCanBeAbandoned ( const uint256 & hashTx ) const
{
2017-07-26 10:23:01 -04:00
auto locked_chain = chain ( ) . lock ( ) ;
LOCK ( cs_wallet ) ;
2017-04-28 14:10:21 -04:00
const CWalletTx * wtx = GetWalletTx ( hashTx ) ;
2019-04-29 09:52:01 -04:00
return wtx & & ! wtx - > isAbandoned ( ) & & wtx - > GetDepthInMainChain ( ) = = 0 & & ! wtx - > InMempool ( ) ;
2017-04-28 14:10:21 -04:00
}
2018-07-12 16:02:31 -04:00
void CWallet : : MarkInputsDirty ( const CTransactionRef & tx )
{
for ( const CTxIn & txin : tx - > vin ) {
auto it = mapWallet . find ( txin . prevout . hash ) ;
if ( it ! = mapWallet . end ( ) ) {
it - > second . MarkDirty ( ) ;
}
}
}
2019-04-29 09:52:01 -04:00
bool CWallet : : AbandonTransaction ( const uint256 & hashTx )
2016-01-07 16:31:27 -05:00
{
2019-04-29 09:52:01 -04:00
auto locked_chain = chain ( ) . lock ( ) ; // Temporary. Removed in upcoming lock cleanup
2017-07-26 10:23:01 -04:00
LOCK ( cs_wallet ) ;
2016-01-07 16:31:27 -05:00
2017-12-08 06:39:22 -05:00
WalletBatch batch ( * database , " r+ " ) ;
2016-01-07 16:31:27 -05:00
std : : set < uint256 > todo ;
std : : set < uint256 > done ;
// Can't mark abandoned if confirmed or in mempool
2017-08-13 15:04:57 +01:00
auto it = mapWallet . find ( hashTx ) ;
assert ( it ! = mapWallet . end ( ) ) ;
CWalletTx & origtx = it - > second ;
2019-04-29 09:52:01 -04:00
if ( origtx . GetDepthInMainChain ( ) ! = 0 | | origtx . InMempool ( ) ) {
2016-01-07 16:31:27 -05:00
return false ;
}
todo . insert ( hashTx ) ;
while ( ! todo . empty ( ) ) {
uint256 now = * todo . begin ( ) ;
todo . erase ( now ) ;
done . insert ( now ) ;
2017-08-13 15:04:57 +01:00
auto it = mapWallet . find ( now ) ;
assert ( it ! = mapWallet . end ( ) ) ;
CWalletTx & wtx = it - > second ;
2019-04-29 09:52:01 -04:00
int currentconfirm = wtx . GetDepthInMainChain ( ) ;
2016-01-07 16:31:27 -05:00
// If the orig tx was not in block, none of its spends can be
assert ( currentconfirm < = 0 ) ;
// if (currentconfirm < 0) {Tx and spends are already conflicted, no need to abandon}
if ( currentconfirm = = 0 & & ! wtx . isAbandoned ( ) ) {
// If the orig tx was not in block/mempool, none of its spends can be in mempool
assert ( ! wtx . InMempool ( ) ) ;
wtx . setAbandoned ( ) ;
wtx . MarkDirty ( ) ;
2017-12-08 06:39:22 -05:00
batch . WriteTx ( wtx ) ;
2016-01-11 11:15:41 +01:00
NotifyTransactionChanged ( this , wtx . GetHash ( ) , CT_UPDATED ) ;
2016-01-07 16:31:27 -05:00
// Iterate over all its outputs, and mark transactions in the wallet that spend them abandoned too
2018-07-12 17:19:00 -04:00
TxSpends : : const_iterator iter = mapTxSpends . lower_bound ( COutPoint ( now , 0 ) ) ;
2016-01-07 16:31:27 -05:00
while ( iter ! = mapTxSpends . end ( ) & & iter - > first . hash = = now ) {
if ( ! done . count ( iter - > second ) ) {
todo . insert ( iter - > second ) ;
}
iter + + ;
}
// If a transaction changes 'conflicted' state, that changes the balance
// available of the outputs it spends. So force those to be recomputed
2018-07-12 16:02:31 -04:00
MarkInputsDirty ( wtx . tx ) ;
2016-01-07 16:31:27 -05:00
}
}
return true ;
}
2019-04-20 11:22:59 -04:00
void CWallet : : MarkConflicted ( const uint256 & hashBlock , int conflicting_height , const uint256 & hashTx )
2015-11-26 18:42:07 +01:00
{
2017-07-26 10:23:01 -04:00
auto locked_chain = chain ( ) . lock ( ) ;
LOCK ( cs_wallet ) ;
2015-11-26 18:42:07 +01:00
2019-04-29 10:18:50 -04:00
int conflictconfirms = ( m_last_block_processed_height - conflicting_height + 1 ) * - 1 ;
2016-02-09 20:23:09 +01:00
// If number of conflict confirms cannot be determined, this means
// that the block is still unknown or not yet part of the main chain,
// for example when loading the wallet during a reindex. Do nothing in that
// case.
if ( conflictconfirms > = 0 )
return ;
2015-11-26 18:42:07 +01:00
// Do not flush the wallet here for performance reasons
2017-12-08 06:39:22 -05:00
WalletBatch batch ( * database , " r+ " , false ) ;
2015-11-26 18:42:07 +01:00
2016-01-07 16:31:12 -05:00
std : : set < uint256 > todo ;
2015-11-26 18:42:07 +01:00
std : : set < uint256 > done ;
2016-01-07 16:31:12 -05:00
todo . insert ( hashTx ) ;
2015-11-26 18:42:07 +01:00
while ( ! todo . empty ( ) ) {
2016-01-07 16:31:12 -05:00
uint256 now = * todo . begin ( ) ;
todo . erase ( now ) ;
2015-11-26 18:42:07 +01:00
done . insert ( now ) ;
2017-08-13 15:04:57 +01:00
auto it = mapWallet . find ( now ) ;
assert ( it ! = mapWallet . end ( ) ) ;
CWalletTx & wtx = it - > second ;
2019-04-29 09:52:01 -04:00
int currentconfirm = wtx . GetDepthInMainChain ( ) ;
2015-11-26 18:42:07 +01:00
if ( conflictconfirms < currentconfirm ) {
// Block is 'more conflicted' than current confirm; update.
// Mark transaction as conflicted with this block.
2019-08-12 18:12:12 -04:00
wtx . m_confirm . nIndex = 0 ;
wtx . m_confirm . hashBlock = hashBlock ;
2019-04-20 11:22:59 -04:00
wtx . m_confirm . block_height = conflicting_height ;
2019-08-12 18:12:12 -04:00
wtx . setConflicted ( ) ;
2015-11-26 18:42:07 +01:00
wtx . MarkDirty ( ) ;
2017-12-08 06:39:22 -05:00
batch . WriteTx ( wtx ) ;
2015-11-26 18:42:07 +01:00
// Iterate over all its outputs, and mark transactions in the wallet that spend them conflicted too
TxSpends : : const_iterator iter = mapTxSpends . lower_bound ( COutPoint ( now , 0 ) ) ;
while ( iter ! = mapTxSpends . end ( ) & & iter - > first . hash = = now ) {
if ( ! done . count ( iter - > second ) ) {
2016-01-07 16:31:12 -05:00
todo . insert ( iter - > second ) ;
2015-11-26 18:42:07 +01:00
}
iter + + ;
}
2016-01-06 17:24:30 -05:00
// If a transaction changes 'conflicted' state, that changes the balance
// available of the outputs it spends. So force those to be recomputed
2018-07-12 16:02:31 -04:00
MarkInputsDirty ( wtx . tx ) ;
2015-11-26 18:42:07 +01:00
}
}
}
2019-10-24 12:53:57 -04:00
void CWallet : : SyncTransaction ( const CTransactionRef & ptx , CWalletTx : : Confirmation confirm , bool update_tx )
2019-08-12 18:12:12 -04:00
{
2019-10-24 12:53:57 -04:00
if ( ! AddToWalletIfInvolvingMe ( ptx , confirm , update_tx ) )
2014-02-15 16:38:28 -05:00
return ; // Not one of ours
// If a transaction changes 'conflicted' state, that changes the balance
// available of the outputs it spends. So force those to be
// recomputed, also:
2018-07-12 16:02:31 -04:00
MarkInputsDirty ( ptx ) ;
2013-10-19 18:34:06 +02:00
}
2020-02-26 16:05:49 -05:00
void CWallet : : transactionAddedToMempool ( const CTransactionRef & ptx ) {
2017-07-26 10:23:01 -04:00
auto locked_chain = chain ( ) . lock ( ) ;
LOCK ( cs_wallet ) ;
2019-04-20 11:22:59 -04:00
CWalletTx : : Confirmation confirm ( CWalletTx : : Status : : UNCONFIRMED , /* block_height */ 0 , { } , /* nIndex */ 0 ) ;
2019-10-24 12:53:57 -04:00
SyncTransaction ( ptx , confirm ) ;
Use callbacks to cache whether wallet transactions are in mempool
This avoid calling out to mempool state during coin selection,
balance calculation, etc. In the next commit we ensure all wallet
callbacks from CValidationInterface happen in the same queue,
serialized with each other. This helps to avoid re-introducing one
of the issues described in #9584 [1] by further disconnecting
wallet from current chain/mempool state.
Thanks to @morcos for the suggestion to do this.
Note that there are several race conditions introduced here:
* If a user calls sendrawtransaction from RPC, adding a
transaction which is "trusted" (ie from them) and pays them
change, it may not be immediately used by coin selection until
the notification callbacks finish running. No such race is
introduced in normal transaction-sending RPCs as this case is
explicitly handled.
* Until Block{Connected,Disconnected} and
TransactionAddedToMempool calls also run in the CSceduler
background thread, there is a race where
TransactionAddedToMempool might be called after a
Block{Connected,Disconnected} call happens.
* Wallet will write a new best chain from the SetBestChain
callback prior to having processed the transaction from that
block.
[1] "you could go to select coins, need to use 0-conf change, but
such 0-conf change may have been included in a block who's
callbacks have not yet been processed - resulting in thinking they
are not in mempool and, thus, not selectable."
2017-01-20 16:38:07 -05:00
auto it = mapWallet . find ( ptx - > GetHash ( ) ) ;
if ( it ! = mapWallet . end ( ) ) {
it - > second . fInMempool = true ;
}
}
2020-02-26 16:05:49 -05:00
void CWallet : : transactionRemovedFromMempool ( const CTransactionRef & ptx ) {
Use callbacks to cache whether wallet transactions are in mempool
This avoid calling out to mempool state during coin selection,
balance calculation, etc. In the next commit we ensure all wallet
callbacks from CValidationInterface happen in the same queue,
serialized with each other. This helps to avoid re-introducing one
of the issues described in #9584 [1] by further disconnecting
wallet from current chain/mempool state.
Thanks to @morcos for the suggestion to do this.
Note that there are several race conditions introduced here:
* If a user calls sendrawtransaction from RPC, adding a
transaction which is "trusted" (ie from them) and pays them
change, it may not be immediately used by coin selection until
the notification callbacks finish running. No such race is
introduced in normal transaction-sending RPCs as this case is
explicitly handled.
* Until Block{Connected,Disconnected} and
TransactionAddedToMempool calls also run in the CSceduler
background thread, there is a race where
TransactionAddedToMempool might be called after a
Block{Connected,Disconnected} call happens.
* Wallet will write a new best chain from the SetBestChain
callback prior to having processed the transaction from that
block.
[1] "you could go to select coins, need to use 0-conf change, but
such 0-conf change may have been included in a block who's
callbacks have not yet been processed - resulting in thinking they
are not in mempool and, thus, not selectable."
2017-01-20 16:38:07 -05:00
LOCK ( cs_wallet ) ;
auto it = mapWallet . find ( ptx - > GetHash ( ) ) ;
if ( it ! = mapWallet . end ( ) ) {
it - > second . fInMempool = false ;
}
2017-03-29 21:12:42 -04:00
}
2020-02-26 16:05:49 -05:00
void CWallet : : blockConnected ( const CBlock & block , int height )
2019-07-24 15:41:41 -04:00
{
2017-07-30 16:00:56 -04:00
const uint256 & block_hash = block . GetHash ( ) ;
2017-07-26 10:23:01 -04:00
auto locked_chain = chain ( ) . lock ( ) ;
LOCK ( cs_wallet ) ;
2017-03-29 21:12:42 -04:00
2019-04-20 12:02:52 -04:00
m_last_block_processed_height = height ;
m_last_block_processed = block_hash ;
2019-04-20 11:22:59 -04:00
for ( size_t index = 0 ; index < block . vtx . size ( ) ; index + + ) {
CWalletTx : : Confirmation confirm ( CWalletTx : : Status : : CONFIRMED , height , block_hash , index ) ;
SyncTransaction ( block . vtx [ index ] , confirm ) ;
2020-02-26 16:05:49 -05:00
transactionRemovedFromMempool ( block . vtx [ index ] ) ;
2017-03-29 21:12:42 -04:00
}
}
2020-02-26 16:05:49 -05:00
void CWallet : : blockDisconnected ( const CBlock & block , int height )
2019-07-24 15:41:41 -04:00
{
2017-07-26 10:23:01 -04:00
auto locked_chain = chain ( ) . lock ( ) ;
LOCK ( cs_wallet ) ;
2017-03-08 12:55:33 -05:00
2019-08-12 18:12:12 -04:00
// At block disconnection, this will change an abandoned transaction to
// be unconfirmed, whether or not the transaction is added back to the mempool.
// User may have to call abandontransaction again. It may be addressed in the
// future with a stickier abandoned state or even removing abandontransaction call.
2019-04-20 12:02:52 -04:00
m_last_block_processed_height = height - 1 ;
m_last_block_processed = block . hashPrevBlock ;
2017-07-30 16:00:56 -04:00
for ( const CTransactionRef & ptx : block . vtx ) {
2019-04-20 11:22:59 -04:00
CWalletTx : : Confirmation confirm ( CWalletTx : : Status : : UNCONFIRMED , /* block_height */ 0 , { } , /* nIndex */ 0 ) ;
2019-10-24 12:53:57 -04:00
SyncTransaction ( ptx , confirm ) ;
2017-03-29 21:12:42 -04:00
}
}
2020-02-26 16:05:49 -05:00
void CWallet : : updatedBlockTip ( )
2019-03-28 13:15:47 -04:00
{
m_best_block_time = GetTime ( ) ;
}
2017-03-29 21:12:42 -04:00
2011-06-26 19:23:24 +02:00
2020-03-02 15:57:25 +09:00
void CWallet : : BlockUntilSyncedToCurrentChain ( ) const {
2017-01-17 18:06:16 -05:00
AssertLockNotHeld ( cs_wallet ) ;
2019-04-18 08:21:35 -04:00
// Skip the queue-draining stuff if we know we're caught up with
2019-03-27 11:14:25 -04:00
// ::ChainActive().Tip(), otherwise put a callback in the validation interface queue and wait
2017-01-17 18:06:16 -05:00
// for the queue to drain enough to execute it (indicating we are caught up
// at least with the time we entered this function).
2019-04-18 08:21:35 -04:00
uint256 last_block_hash = WITH_LOCK ( cs_wallet , return m_last_block_processed ) ;
2019-06-24 19:07:09 -04:00
chain ( ) . waitForNotificationsIfTipChanged ( last_block_hash ) ;
2017-01-17 18:06:16 -05:00
}
2013-07-26 01:06:01 +02:00
isminetype CWallet : : IsMine ( const CTxIn & txin ) const
2011-06-26 19:23:24 +02:00
{
{
2012-04-06 18:39:12 +02:00
LOCK ( cs_wallet ) ;
2017-01-27 10:33:45 +09:00
std : : map < uint256 , CWalletTx > : : const_iterator mi = mapWallet . find ( txin . prevout . hash ) ;
2011-06-26 19:23:24 +02:00
if ( mi ! = mapWallet . end ( ) )
{
const CWalletTx & prev = ( * mi ) . second ;
2016-11-11 16:54:51 -08:00
if ( txin . prevout . n < prev . tx - > vout . size ( ) )
return IsMine ( prev . tx - > vout [ txin . prevout . n ] ) ;
2011-06-26 19:23:24 +02:00
}
}
2014-07-01 11:00:22 +02:00
return ISMINE_NO ;
2011-06-26 19:23:24 +02:00
}
2016-12-09 15:31:06 -05:00
// Note that this function doesn't distinguish between a 0-valued input,
// and a not-"is mine" (according to the filter) input.
2014-04-22 15:46:19 -07:00
CAmount CWallet : : GetDebit ( const CTxIn & txin , const isminefilter & filter ) const
2011-06-26 19:23:24 +02:00
{
{
2012-04-06 18:39:12 +02:00
LOCK ( cs_wallet ) ;
2017-01-27 10:33:45 +09:00
std : : map < uint256 , CWalletTx > : : const_iterator mi = mapWallet . find ( txin . prevout . hash ) ;
2011-06-26 19:23:24 +02:00
if ( mi ! = mapWallet . end ( ) )
{
const CWalletTx & prev = ( * mi ) . second ;
2016-11-11 16:54:51 -08:00
if ( txin . prevout . n < prev . tx - > vout . size ( ) )
if ( IsMine ( prev . tx - > vout [ txin . prevout . n ] ) & filter )
return prev . tx - > vout [ txin . prevout . n ] . nValue ;
2011-06-26 19:23:24 +02:00
}
}
return 0 ;
}
2015-02-04 23:19:29 +01:00
isminetype CWallet : : IsMine ( const CTxOut & txout ) const
{
2019-10-07 14:11:34 -04:00
return IsMine ( txout . scriptPubKey ) ;
}
isminetype CWallet : : IsMine ( const CTxDestination & dest ) const
{
return IsMine ( GetScriptForDestination ( dest ) ) ;
2015-02-04 23:19:29 +01:00
}
2019-10-07 14:11:34 -04:00
isminetype CWallet : : IsMine ( const CScript & script ) const
2019-10-07 14:11:34 -04:00
{
2019-10-07 14:11:34 -04:00
isminetype result = ISMINE_NO ;
2019-10-07 14:11:34 -04:00
for ( const auto & spk_man_pair : m_spk_managers ) {
result = std : : max ( result , spk_man_pair . second - > IsMine ( script ) ) ;
2019-10-07 14:11:34 -04:00
}
return result ;
2019-10-07 14:11:34 -04:00
}
2015-02-04 23:19:29 +01:00
CAmount CWallet : : GetCredit ( const CTxOut & txout , const isminefilter & filter ) const
{
if ( ! MoneyRange ( txout . nValue ) )
2016-08-19 18:31:35 +02:00
throw std : : runtime_error ( std : : string ( __func__ ) + " : value out of range " ) ;
2015-02-04 23:19:29 +01:00
return ( ( IsMine ( txout ) & filter ) ? txout . nValue : 0 ) ;
}
2011-10-03 13:05:43 -04:00
bool CWallet : : IsChange ( const CTxOut & txout ) const
2018-10-06 00:24:06 +03:00
{
return IsChange ( txout . scriptPubKey ) ;
}
bool CWallet : : IsChange ( const CScript & script ) const
2011-10-03 13:05:43 -04:00
{
2011-11-08 13:20:29 -05:00
// TODO: fix handling of 'change' outputs. The assumption is that any
2014-06-09 21:11:59 +02:00
// payment to a script that is ours, but is not in the address book
2011-11-08 13:20:29 -05:00
// is change. That assumption is likely to break when we implement multisignature
// wallets that return change back into a multi-signature-protected address;
// a better way of identifying which outputs are 'the send' and which are
// 'the change' will need to be implemented (maybe extend CWalletTx to remember
// which output, if any, was change).
2019-10-07 14:11:34 -04:00
if ( IsMine ( script ) )
2012-04-06 18:39:12 +02:00
{
2014-06-09 21:11:59 +02:00
CTxDestination address ;
2018-10-06 00:24:06 +03:00
if ( ! ExtractDestination ( script , address ) )
2014-06-09 21:11:59 +02:00
return true ;
2012-04-06 18:39:12 +02:00
LOCK ( cs_wallet ) ;
2020-02-22 04:16:36 +00:00
if ( ! FindAddressBookEntry ( address ) ) {
2012-04-06 18:39:12 +02:00
return true ;
2020-02-22 04:16:36 +00:00
}
2012-04-06 18:39:12 +02:00
}
2011-10-03 13:05:43 -04:00
return false ;
}
2015-02-04 23:19:29 +01:00
CAmount CWallet : : GetChange ( const CTxOut & txout ) const
{
if ( ! MoneyRange ( txout . nValue ) )
2016-08-19 18:31:35 +02:00
throw std : : runtime_error ( std : : string ( __func__ ) + " : value out of range " ) ;
2015-02-04 23:19:29 +01:00
return ( IsChange ( txout ) ? txout . nValue : 0 ) ;
}
bool CWallet : : IsMine ( const CTransaction & tx ) const
{
2017-06-02 03:18:57 +02:00
for ( const CTxOut & txout : tx . vout )
2015-02-04 23:19:29 +01:00
if ( IsMine ( txout ) )
return true ;
return false ;
}
bool CWallet : : IsFromMe ( const CTransaction & tx ) const
{
return ( GetDebit ( tx , ISMINE_ALL ) > 0 ) ;
}
CAmount CWallet : : GetDebit ( const CTransaction & tx , const isminefilter & filter ) const
{
CAmount nDebit = 0 ;
2017-06-02 03:18:57 +02:00
for ( const CTxIn & txin : tx . vin )
2015-02-04 23:19:29 +01:00
{
nDebit + = GetDebit ( txin , filter ) ;
if ( ! MoneyRange ( nDebit ) )
2016-08-19 18:31:35 +02:00
throw std : : runtime_error ( std : : string ( __func__ ) + " : value out of range " ) ;
2015-02-04 23:19:29 +01:00
}
return nDebit ;
}
2016-12-09 15:31:06 -05:00
bool CWallet : : IsAllFromMe ( const CTransaction & tx , const isminefilter & filter ) const
{
LOCK ( cs_wallet ) ;
2017-06-02 03:18:57 +02:00
for ( const CTxIn & txin : tx . vin )
2016-12-09 15:31:06 -05:00
{
auto mi = mapWallet . find ( txin . prevout . hash ) ;
if ( mi = = mapWallet . end ( ) )
return false ; // any unknown inputs can't be from us
const CWalletTx & prev = ( * mi ) . second ;
if ( txin . prevout . n > = prev . tx - > vout . size ( ) )
return false ; // invalid input!
if ( ! ( IsMine ( prev . tx - > vout [ txin . prevout . n ] ) & filter ) )
return false ;
}
return true ;
}
2015-02-04 23:19:29 +01:00
CAmount CWallet : : GetCredit ( const CTransaction & tx , const isminefilter & filter ) const
{
CAmount nCredit = 0 ;
2017-06-02 03:18:57 +02:00
for ( const CTxOut & txout : tx . vout )
2015-02-04 23:19:29 +01:00
{
nCredit + = GetCredit ( txout , filter ) ;
if ( ! MoneyRange ( nCredit ) )
2016-08-19 18:31:35 +02:00
throw std : : runtime_error ( std : : string ( __func__ ) + " : value out of range " ) ;
2015-02-04 23:19:29 +01:00
}
return nCredit ;
}
CAmount CWallet : : GetChange ( const CTransaction & tx ) const
{
CAmount nChange = 0 ;
2017-06-02 03:18:57 +02:00
for ( const CTxOut & txout : tx . vout )
2015-02-04 23:19:29 +01:00
{
nChange + = GetChange ( txout ) ;
if ( ! MoneyRange ( nChange ) )
2016-08-19 18:31:35 +02:00
throw std : : runtime_error ( std : : string ( __func__ ) + " : value out of range " ) ;
2015-02-04 23:19:29 +01:00
}
return nChange ;
}
2019-10-07 14:11:34 -04:00
bool CWallet : : IsHDEnabled ( ) const
{
2019-07-08 17:05:05 -04:00
// All Active ScriptPubKeyMans must be HD for this to be true
2019-10-07 14:11:34 -04:00
bool result = true ;
2019-07-08 17:05:05 -04:00
for ( const auto & spk_man : GetActiveScriptPubKeyMans ( ) ) {
result & = spk_man - > IsHDEnabled ( ) ;
2019-10-07 14:11:34 -04:00
}
return result ;
}
2020-03-02 16:35:58 +09:00
bool CWallet : : CanGetAddresses ( bool internal ) const
2019-10-07 14:11:34 -04:00
{
2019-10-07 14:11:34 -04:00
LOCK ( cs_wallet ) ;
2019-10-07 14:11:34 -04:00
if ( m_spk_managers . empty ( ) ) return false ;
for ( OutputType t : OUTPUT_TYPES ) {
auto spk_man = GetScriptPubKeyMan ( t , internal ) ;
2019-10-07 14:11:34 -04:00
if ( spk_man & & spk_man - > CanGetAddresses ( internal ) ) {
return true ;
}
}
return false ;
}
2017-02-16 14:22:18 +01:00
void CWallet : : SetWalletFlag ( uint64_t flags )
{
LOCK ( cs_wallet ) ;
m_wallet_flags | = flags ;
if ( ! WalletBatch ( * database ) . WriteWalletFlags ( m_wallet_flags ) )
throw std : : runtime_error ( std : : string ( __func__ ) + " : writing wallet flags failed " ) ;
}
2019-02-06 21:26:55 -05:00
void CWallet : : UnsetWalletFlag ( uint64_t flag )
2019-05-18 11:01:56 +01:00
{
WalletBatch batch ( * database ) ;
UnsetWalletFlagWithDB ( batch , flag ) ;
}
void CWallet : : UnsetWalletFlagWithDB ( WalletBatch & batch , uint64_t flag )
2019-02-06 21:26:55 -05:00
{
LOCK ( cs_wallet ) ;
m_wallet_flags & = ~ flag ;
2019-05-18 11:01:56 +01:00
if ( ! batch . WriteWalletFlags ( m_wallet_flags ) )
2019-02-06 21:26:55 -05:00
throw std : : runtime_error ( std : : string ( __func__ ) + " : writing wallet flags failed " ) ;
}
2019-10-29 17:53:35 -04:00
void CWallet : : UnsetBlankWalletFlag ( WalletBatch & batch )
{
UnsetWalletFlagWithDB ( batch , WALLET_FLAG_BLANK_WALLET ) ;
}
2018-08-02 16:43:55 +09:00
bool CWallet : : IsWalletFlagSet ( uint64_t flag ) const
2017-02-16 14:22:18 +01:00
{
return ( m_wallet_flags & flag ) ;
}
2017-05-05 08:53:39 +02:00
bool CWallet : : SetWalletFlags ( uint64_t overwriteFlags , bool memonly )
2017-02-16 14:22:18 +01:00
{
LOCK ( cs_wallet ) ;
m_wallet_flags = overwriteFlags ;
2019-05-24 14:55:27 +09:00
if ( ( ( overwriteFlags & KNOWN_WALLET_FLAGS ) > > 32 ) ^ ( overwriteFlags > > 32 ) ) {
2017-05-05 08:53:39 +02:00
// contains unknown non-tolerable wallet flags
return false ;
}
2017-02-16 14:22:18 +01:00
if ( ! memonly & & ! WalletBatch ( * database ) . WriteWalletFlags ( m_wallet_flags ) ) {
throw std : : runtime_error ( std : : string ( __func__ ) + " : writing wallet flags failed " ) ;
}
2017-05-05 08:53:39 +02:00
return true ;
2017-02-16 14:22:18 +01:00
}
2013-04-13 00:13:08 -05:00
int64_t CWalletTx : : GetTxTime ( ) const
2011-06-26 19:23:24 +02:00
{
2013-04-13 00:13:08 -05:00
int64_t n = nTimeSmart ;
2012-05-28 18:45:12 +00:00
return n ? n : nTimeReceived ;
2011-06-26 19:23:24 +02:00
}
2018-07-14 19:19:44 -07:00
// Helper for producing a max-sized low-S low-R signature (eg 71 bytes)
2018-08-07 16:59:53 -07:00
// or a max-sized low-S signature (e.g. 72 bytes) if use_max_sig is true
bool CWallet : : DummySignInput ( CTxIn & tx_in , const CTxOut & txout , bool use_max_sig ) const
2018-03-05 16:37:24 -05:00
{
// Fill in dummy signatures for fee calculation.
const CScript & scriptPubKey = txout . scriptPubKey ;
SignatureData sigdata ;
2020-02-10 21:27:59 -05:00
std : : unique_ptr < SigningProvider > provider = GetSolvingProvider ( scriptPubKey ) ;
2019-10-07 14:11:34 -04:00
if ( ! provider ) {
// We don't know about this scriptpbuKey;
return false ;
}
2019-10-07 14:11:34 -04:00
if ( ! ProduceSignature ( * provider , use_max_sig ? DUMMY_MAXIMUM_SIGNATURE_CREATOR : DUMMY_SIGNATURE_CREATOR , scriptPubKey , sigdata ) ) {
2018-03-05 16:37:24 -05:00
return false ;
}
2018-08-07 16:59:53 -07:00
UpdateInput ( tx_in , sigdata ) ;
2018-03-05 16:37:24 -05:00
return true ;
}
2018-07-14 19:19:44 -07:00
// Helper for producing a bunch of max-sized low-S low-R signatures (eg 71 bytes)
2018-08-07 16:59:53 -07:00
bool CWallet : : DummySignTx ( CMutableTransaction & txNew , const std : : vector < CTxOut > & txouts , bool use_max_sig ) const
2018-03-05 16:37:24 -05:00
{
// Fill in dummy signatures for fee calculation.
int nIn = 0 ;
for ( const auto & txout : txouts )
{
2018-08-07 16:59:53 -07:00
if ( ! DummySignInput ( txNew . vin [ nIn ] , txout , use_max_sig ) ) {
2018-03-05 16:37:24 -05:00
return false ;
}
nIn + + ;
}
return true ;
}
2019-10-07 14:11:34 -04:00
bool CWallet : : ImportScripts ( const std : : set < CScript > scripts , int64_t timestamp )
{
auto spk_man = GetLegacyScriptPubKeyMan ( ) ;
if ( ! spk_man ) {
return false ;
}
2019-10-07 14:11:34 -04:00
LOCK ( spk_man - > cs_KeyStore ) ;
2019-10-07 14:11:34 -04:00
return spk_man - > ImportScripts ( scripts , timestamp ) ;
}
bool CWallet : : ImportPrivKeys ( const std : : map < CKeyID , CKey > & privkey_map , const int64_t timestamp )
{
auto spk_man = GetLegacyScriptPubKeyMan ( ) ;
if ( ! spk_man ) {
return false ;
}
2019-10-07 14:11:34 -04:00
LOCK ( spk_man - > cs_KeyStore ) ;
2019-10-07 14:11:34 -04:00
return spk_man - > ImportPrivKeys ( privkey_map , timestamp ) ;
}
bool CWallet : : ImportPubKeys ( const std : : vector < CKeyID > & ordered_pubkeys , const std : : map < CKeyID , CPubKey > & pubkey_map , const std : : map < CKeyID , std : : pair < CPubKey , KeyOriginInfo > > & key_origins , const bool add_keypool , const bool internal , const int64_t timestamp )
{
auto spk_man = GetLegacyScriptPubKeyMan ( ) ;
if ( ! spk_man ) {
return false ;
}
2019-10-07 14:11:34 -04:00
LOCK ( spk_man - > cs_KeyStore ) ;
2019-10-07 14:11:34 -04:00
return spk_man - > ImportPubKeys ( ordered_pubkeys , pubkey_map , key_origins , add_keypool , internal , timestamp ) ;
}
bool CWallet : : ImportScriptPubKeys ( const std : : string & label , const std : : set < CScript > & script_pub_keys , const bool have_solving_data , const bool apply_label , const int64_t timestamp )
{
auto spk_man = GetLegacyScriptPubKeyMan ( ) ;
if ( ! spk_man ) {
return false ;
}
2019-10-07 14:11:34 -04:00
LOCK ( spk_man - > cs_KeyStore ) ;
2019-10-07 14:11:34 -04:00
if ( ! spk_man - > ImportScriptPubKeys ( script_pub_keys , have_solving_data , timestamp ) ) {
2019-10-07 14:11:34 -04:00
return false ;
}
2019-10-07 14:11:34 -04:00
if ( apply_label ) {
WalletBatch batch ( * database ) ;
for ( const CScript & script : script_pub_keys ) {
CTxDestination dest ;
ExtractDestination ( script , dest ) ;
if ( IsValidDestination ( dest ) ) {
SetAddressBookWithDB ( batch , dest , label , " receive " ) ;
}
}
}
2019-10-07 14:11:34 -04:00
return true ;
}
2018-08-07 16:59:53 -07:00
int64_t CalculateMaximumSignedTxSize ( const CTransaction & tx , const CWallet * wallet , bool use_max_sig )
2018-03-05 16:37:24 -05:00
{
std : : vector < CTxOut > txouts ;
2018-06-18 07:58:28 +02:00
for ( const CTxIn & input : tx . vin ) {
2018-03-05 16:37:24 -05:00
const auto mi = wallet - > mapWallet . find ( input . prevout . hash ) ;
2019-11-21 14:37:26 -05:00
// Can not estimate size without knowing the input details
2018-03-05 16:37:24 -05:00
if ( mi = = wallet - > mapWallet . end ( ) ) {
return - 1 ;
}
assert ( input . prevout . n < mi - > second . tx - > vout . size ( ) ) ;
txouts . emplace_back ( mi - > second . tx - > vout [ input . prevout . n ] ) ;
}
2018-08-07 16:59:53 -07:00
return CalculateMaximumSignedTxSize ( tx , wallet , txouts , use_max_sig ) ;
2018-03-05 16:37:24 -05:00
}
// txouts needs to be in the order of tx.vin
2018-08-07 16:59:53 -07:00
int64_t CalculateMaximumSignedTxSize ( const CTransaction & tx , const CWallet * wallet , const std : : vector < CTxOut > & txouts , bool use_max_sig )
2018-03-05 16:37:24 -05:00
{
CMutableTransaction txNew ( tx ) ;
2018-08-07 16:59:53 -07:00
if ( ! wallet - > DummySignTx ( txNew , txouts , use_max_sig ) ) {
2018-03-05 16:37:24 -05:00
return - 1 ;
}
2018-12-09 22:03:07 -08:00
return GetVirtualTransactionSize ( CTransaction ( txNew ) ) ;
2018-03-05 16:37:24 -05:00
}
2018-08-07 16:59:53 -07:00
int CalculateMaximumSignedInputSize ( const CTxOut & txout , const CWallet * wallet , bool use_max_sig )
2018-03-05 16:37:24 -05:00
{
CMutableTransaction txn ;
txn . vin . push_back ( CTxIn ( COutPoint ( ) ) ) ;
2018-08-07 16:59:53 -07:00
if ( ! wallet - > DummySignInput ( txn . vin [ 0 ] , txout , use_max_sig ) ) {
2018-03-05 16:37:24 -05:00
return - 1 ;
}
return GetVirtualTransactionInputSize ( txn . vin [ 0 ] ) ;
}
2017-01-27 10:33:45 +09:00
void CWalletTx : : GetAmounts ( std : : list < COutputEntry > & listReceived ,
2018-07-31 12:23:26 -04:00
std : : list < COutputEntry > & listSent , CAmount & nFee , const isminefilter & filter ) const
2011-06-26 19:23:24 +02:00
{
2012-06-02 02:33:28 +00:00
nFee = 0 ;
2011-06-26 19:23:24 +02:00
listReceived . clear ( ) ;
listSent . clear ( ) ;
// Compute fee:
2014-04-22 15:46:19 -07:00
CAmount nDebit = GetDebit ( filter ) ;
2011-06-26 19:23:24 +02:00
if ( nDebit > 0 ) // debit>0 means we signed/sent this transaction
{
2016-11-11 16:54:51 -08:00
CAmount nValueOut = tx - > GetValueOut ( ) ;
2011-06-26 19:23:24 +02:00
nFee = nDebit - nValueOut ;
}
2011-10-03 13:05:43 -04:00
// Sent/received.
2016-11-11 16:54:51 -08:00
for ( unsigned int i = 0 ; i < tx - > vout . size ( ) ; + + i )
2011-06-26 19:23:24 +02:00
{
2016-11-11 16:54:51 -08:00
const CTxOut & txout = tx - > vout [ i ] ;
2014-04-29 19:39:01 +02:00
isminetype fIsMine = pwallet - > IsMine ( txout ) ;
2012-09-22 03:20:14 +00:00
// Only need to handle txouts if AT LEAST one of these is true:
// 1) they debit from us (sent)
// 2) the output is to us (received)
if ( nDebit > 0 )
{
// Don't report 'change' txouts
if ( pwallet - > IsChange ( txout ) )
continue ;
}
2014-04-29 19:39:01 +02:00
else if ( ! ( fIsMine & filter ) )
2012-09-22 03:20:14 +00:00
continue ;
// In either case, we need to get the destination address
2012-05-14 23:44:52 +02:00
CTxDestination address ;
2015-12-11 18:07:11 -08:00
2015-12-12 10:45:53 -08:00
if ( ! ExtractDestination ( txout . scriptPubKey , address ) & & ! txout . scriptPubKey . IsUnspendable ( ) )
2011-06-26 19:23:24 +02:00
{
2018-06-15 19:02:52 -04:00
pwallet - > WalletLogPrintf ( " CWalletTx::GetAmounts: Unknown transaction type found, txid %s \n " ,
this - > GetHash ( ) . ToString ( ) ) ;
2012-09-22 03:20:14 +00:00
address = CNoDestination ( ) ;
2011-06-26 19:23:24 +02:00
}
2014-07-18 13:24:38 +02:00
COutputEntry output = { address , txout . nValue , ( int ) i } ;
2014-05-30 00:54:00 +02:00
2012-09-22 03:20:14 +00:00
// If we are debited by the transaction, add the output as a "sent" entry
2011-06-26 19:23:24 +02:00
if ( nDebit > 0 )
2014-05-30 00:54:00 +02:00
listSent . push_back ( output ) ;
2011-06-26 19:23:24 +02:00
2012-09-22 03:20:14 +00:00
// If we are receiving the output, add it as a "received" entry
2014-07-12 17:15:17 +02:00
if ( fIsMine & filter )
2014-05-30 00:54:00 +02:00
listReceived . push_back ( output ) ;
2011-06-26 19:23:24 +02:00
}
}
2017-03-02 15:24:50 -05:00
/**
* Scan active chain for relevant transactions after importing keys . This should
* be called whenever new keys are added to the wallet , with the oldest key
2017-06-22 17:16:24 -04:00
* creation time .
2017-03-02 15:24:50 -05:00
*
* @ return Earliest timestamp that could be successfully scanned from . Timestamp
2017-06-22 17:14:40 -04:00
* returned will be higher than startTime if relevant blocks could not be read .
2017-03-02 15:24:50 -05:00
*/
2017-12-12 13:13:58 -10:00
int64_t CWallet : : RescanFromTime ( int64_t startTime , const WalletRescanReserver & reserver , bool update )
2017-03-02 15:24:50 -05:00
{
// Find starting block. May be null if nCreateTime is greater than the
// highest blockchain timestamp, in which case there is nothing that needs
// to be scanned.
2020-01-21 15:55:19 -05:00
int start_height = 0 ;
2019-01-07 23:38:53 -08:00
uint256 start_block ;
2020-01-21 15:55:19 -05:00
bool start = chain ( ) . findFirstBlockWithTimeAndHeight ( startTime - TIMESTAMP_WINDOW , 0 , FoundBlock ( ) . hash ( start_block ) . height ( start_height ) ) ;
WalletLogPrintf ( " %s: Rescanning last %i blocks \n " , __func__ , start ? WITH_LOCK ( cs_wallet , return GetLastBlockHeight ( ) ) - start_height + 1 : 0 ) ;
2017-03-02 15:24:50 -05:00
2020-01-21 15:55:19 -05:00
if ( start ) {
2018-04-29 15:45:44 +00:00
// TODO: this should take into account failure by ScanResult::USER_ABORT
2020-01-22 16:53:42 -05:00
ScanResult result = ScanForWalletTransactions ( start_block , start_height , { } /* max_height */ , reserver , update ) ;
2019-01-07 23:38:53 -08:00
if ( result . status = = ScanResult : : FAILURE ) {
int64_t time_max ;
2020-02-24 14:34:17 -05:00
CHECK_NONFATAL ( chain ( ) . findBlock ( result . last_failed_block , FoundBlock ( ) . maxTime ( time_max ) ) ) ;
2019-01-07 23:38:53 -08:00
return time_max + TIMESTAMP_WINDOW + 1 ;
2017-03-02 15:24:50 -05:00
}
}
return startTime ;
}
2014-10-26 15:03:12 +08:00
/**
2019-01-07 23:38:53 -08:00
* Scan the block chain ( starting in start_block ) for transactions
2014-10-26 15:03:12 +08:00
* from or to us . If fUpdate is true , found transactions that already
* exist in the wallet will be updated .
2017-02-16 10:49:03 -05:00
*
2019-02-01 16:15:13 -05:00
* @ param [ in ] start_block Scan starting block . If block is not on the active
* chain , the scan will return SUCCESS immediately .
2020-01-22 16:53:42 -05:00
* @ param [ in ] start_height Height of start_block
2020-01-21 17:08:12 -05:00
* @ param [ in ] max_height Optional max scanning height . If unset there is
* no maximum and scanning can continue to the tip
2015-11-19 16:05:37 +01:00
*
2019-01-31 17:48:51 -05:00
* @ return ScanResult returning scan information and indicating success or
* failure . Return status will be set to SUCCESS if scan was
* successful . FAILURE if a complete rescan was not possible ( due to
* pruning or corruption ) . USER_ABORT if the rescan was aborted before
* it could complete .
2017-09-07 16:29:59 -07:00
*
2019-01-07 23:38:53 -08:00
* @ pre Caller needs to make sure start_block ( and the optional stop_block ) are on
2017-09-07 16:29:59 -07:00
* the main chain after to the addition of any new keys you want to detect
* transactions for .
2014-10-26 15:03:12 +08:00
*/
2020-01-22 16:53:42 -05:00
CWallet : : ScanResult CWallet : : ScanForWalletTransactions ( const uint256 & start_block , int start_height , Optional < int > max_height , const WalletRescanReserver & reserver , bool fUpdate )
2011-06-26 19:23:24 +02:00
{
2014-02-17 16:35:37 -08:00
int64_t nNow = GetTime ( ) ;
2019-04-27 10:15:56 -04:00
int64_t start_time = GetTimeMillis ( ) ;
2011-06-26 19:23:24 +02:00
2017-12-12 13:13:58 -10:00
assert ( reserver . isReserved ( ) ) ;
2015-11-19 16:05:37 +01:00
2019-01-07 23:38:53 -08:00
uint256 block_hash = start_block ;
ScanResult result ;
2018-04-09 18:48:19 +02:00
2019-01-07 23:38:53 -08:00
WalletLogPrintf ( " Rescan started from block %s... \n " , start_block . ToString ( ) ) ;
2018-04-09 18:48:19 +02:00
2019-04-27 10:16:33 -04:00
fAbortRescan = false ;
2019-06-28 21:09:58 +03:00
ShowProgress ( strprintf ( " %s " + _ ( " Rescanning... " ) . translated , GetDisplayName ( ) ) , 0 ) ; // show rescan progress in GUI as dialog or on splashscreen, if -rescan on startup
2020-01-22 16:53:42 -05:00
uint256 tip_hash = WITH_LOCK ( cs_wallet , return GetLastBlockHash ( ) ) ;
uint256 end_hash = tip_hash ;
if ( max_height ) chain ( ) . findAncestorByHeight ( tip_hash , * max_height , FoundBlock ( ) . hash ( end_hash ) ) ;
double progress_begin = chain ( ) . guessVerificationProgress ( block_hash ) ;
double progress_end = chain ( ) . guessVerificationProgress ( end_hash ) ;
2019-04-27 10:16:33 -04:00
double progress_current = progress_begin ;
2020-01-22 16:53:42 -05:00
int block_height = start_height ;
while ( ! fAbortRescan & & ! chain ( ) . shutdownRequested ( ) ) {
2019-04-03 17:05:52 +01:00
m_scanning_progress = ( progress_current - progress_begin ) / ( progress_end - progress_begin ) ;
2020-01-22 16:53:42 -05:00
if ( block_height % 100 = = 0 & & progress_end - progress_begin > 0.0 ) {
2019-06-28 21:09:58 +03:00
ShowProgress ( strprintf ( " %s " + _ ( " Rescanning... " ) . translated , GetDisplayName ( ) ) , std : : max ( 1 , std : : min ( 99 , ( int ) ( m_scanning_progress * 100 ) ) ) ) ;
2019-04-27 10:16:33 -04:00
}
if ( GetTime ( ) > = nNow + 60 ) {
nNow = GetTime ( ) ;
2020-01-22 16:53:42 -05:00
WalletLogPrintf ( " Still rescanning. At block %d. Progress=%f \n " , block_height , progress_current ) ;
2017-09-07 16:29:59 -07:00
}
2013-06-10 09:38:13 -04:00
2019-04-27 10:16:33 -04:00
CBlock block ;
2020-01-22 16:53:42 -05:00
bool next_block ;
uint256 next_block_hash ;
bool reorg = false ;
2020-02-24 14:34:17 -05:00
if ( chain ( ) . findBlock ( block_hash , FoundBlock ( ) . data ( block ) ) & & ! block . IsNull ( ) ) {
2019-04-27 10:16:33 -04:00
auto locked_chain = chain ( ) . lock ( ) ;
LOCK ( cs_wallet ) ;
2020-01-22 16:53:42 -05:00
next_block = chain ( ) . findNextBlock ( block_hash , block_height , FoundBlock ( ) . hash ( next_block_hash ) , & reorg ) ;
if ( reorg ) {
2019-04-27 10:16:33 -04:00
// Abort scan if current block is no longer active, to prevent
// marking transactions as coming from the wrong block.
// TODO: This should return success instead of failure, see
// https://github.com/bitcoin/bitcoin/pull/14711#issuecomment-458342518
2019-01-31 17:42:56 -05:00
result . last_failed_block = block_hash ;
2019-01-07 23:38:53 -08:00
result . status = ScanResult : : FAILURE ;
2019-04-27 10:16:33 -04:00
break ;
}
for ( size_t posInBlock = 0 ; posInBlock < block . vtx . size ( ) ; + + posInBlock ) {
2020-01-22 16:53:42 -05:00
CWalletTx : : Confirmation confirm ( CWalletTx : : Status : : CONFIRMED , block_height , block_hash , posInBlock ) ;
2019-10-24 12:53:57 -04:00
SyncTransaction ( block . vtx [ posInBlock ] , confirm , fUpdate ) ;
2011-06-26 19:23:24 +02:00
}
2019-04-27 10:16:33 -04:00
// scan succeeded, record block as most recent successfully scanned
result . last_scanned_block = block_hash ;
2020-01-22 16:53:42 -05:00
result . last_scanned_height = block_height ;
2019-04-27 10:16:33 -04:00
} else {
// could not scan block, keep scanning but record this block as the most recent failure
result . last_failed_block = block_hash ;
result . status = ScanResult : : FAILURE ;
2020-01-22 16:53:42 -05:00
next_block = chain ( ) . findNextBlock ( block_hash , block_height , FoundBlock ( ) . hash ( next_block_hash ) , & reorg ) ;
2019-04-27 10:16:33 -04:00
}
2020-01-22 16:53:42 -05:00
if ( max_height & & block_height > = * max_height ) {
2019-04-27 10:16:33 -04:00
break ;
}
{
auto locked_chain = chain ( ) . lock ( ) ;
2020-01-22 16:53:42 -05:00
if ( ! next_block | | reorg ) {
2019-04-27 10:16:33 -04:00
// break successfully when rescan has reached the tip, or
// previous block is no longer on the chain due to a reorg
2015-11-19 16:05:37 +01:00
break ;
}
2019-01-07 23:38:53 -08:00
2019-04-27 10:16:33 -04:00
// increment block and verification progress
2020-01-22 16:53:42 -05:00
block_hash = next_block_hash ;
+ + block_height ;
2019-04-27 10:16:33 -04:00
progress_current = chain ( ) . guessVerificationProgress ( block_hash ) ;
2019-01-07 23:38:53 -08:00
2019-04-27 10:16:33 -04:00
// handle updated tip hash
const uint256 prev_tip_hash = tip_hash ;
2020-01-22 16:53:42 -05:00
tip_hash = WITH_LOCK ( cs_wallet , return GetLastBlockHash ( ) ) ;
2020-01-21 17:08:12 -05:00
if ( ! max_height & & prev_tip_hash ! = tip_hash ) {
2019-04-27 10:16:33 -04:00
// in case the tip has changed, update progress max
progress_end = chain ( ) . guessVerificationProgress ( tip_hash ) ;
2017-09-07 16:29:59 -07:00
}
2011-06-26 19:23:24 +02:00
}
2019-04-27 10:16:33 -04:00
}
2019-06-28 21:09:58 +03:00
ShowProgress ( strprintf ( " %s " + _ ( " Rescanning... " ) . translated , GetDisplayName ( ) ) , 100 ) ; // hide progress dialog in GUI
2019-04-27 10:16:33 -04:00
if ( block_height & & fAbortRescan ) {
2020-01-22 16:53:42 -05:00
WalletLogPrintf ( " Rescan aborted at block %d. Progress=%f \n " , block_height , progress_current ) ;
2019-04-27 10:16:33 -04:00
result . status = ScanResult : : USER_ABORT ;
} else if ( block_height & & chain ( ) . shutdownRequested ( ) ) {
2020-01-22 16:53:42 -05:00
WalletLogPrintf ( " Rescan interrupted by shutdown request at block %d. Progress=%f \n " , block_height , progress_current ) ;
2019-04-27 10:16:33 -04:00
result . status = ScanResult : : USER_ABORT ;
} else {
WalletLogPrintf ( " Rescan completed in %15dms \n " , GetTimeMillis ( ) - start_time ) ;
2011-06-26 19:23:24 +02:00
}
2019-01-07 23:38:53 -08:00
return result ;
2011-06-26 19:23:24 +02:00
}
2019-04-29 09:52:01 -04:00
void CWallet : : ReacceptWalletTransactions ( )
2011-06-26 19:23:24 +02:00
{
2015-04-28 14:48:28 +00:00
// If transactions aren't being broadcasted, don't let them into local mempool either
2015-03-27 10:34:48 +01:00
if ( ! fBroadcastTransactions )
return ;
2014-12-19 06:59:16 +01:00
std : : map < int64_t , CWalletTx * > mapSorted ;
// Sort pending wallet transactions based on their initial wallet insertion order
2019-04-11 15:58:53 +00:00
for ( std : : pair < const uint256 , CWalletTx > & item : mapWallet ) {
2014-02-15 16:38:28 -05:00
const uint256 & wtxid = item . first ;
CWalletTx & wtx = item . second ;
assert ( wtx . GetHash ( ) = = wtxid ) ;
2011-06-26 19:23:24 +02:00
2019-04-29 09:52:01 -04:00
int nDepth = wtx . GetDepthInMainChain ( ) ;
2014-02-15 16:38:28 -05:00
2016-01-07 16:31:27 -05:00
if ( ! wtx . IsCoinBase ( ) & & ( nDepth = = 0 & & ! wtx . isAbandoned ( ) ) ) {
2014-12-19 06:59:16 +01:00
mapSorted . insert ( std : : make_pair ( wtx . nOrderPos , & wtx ) ) ;
2011-06-26 19:23:24 +02:00
}
}
2014-12-19 06:59:16 +01:00
// Try to add wallet transactions to memory pool
2018-06-18 07:58:28 +02:00
for ( const std : : pair < const int64_t , CWalletTx * > & item : mapSorted ) {
2014-12-19 06:59:16 +01:00
CWalletTx & wtx = * ( item . second ) ;
2019-04-11 15:58:53 +00:00
std : : string unused_err_string ;
2019-04-29 09:52:01 -04:00
wtx . SubmitMemoryPoolAndRelay ( unused_err_string , false ) ;
2014-12-19 06:59:16 +01:00
}
2011-06-26 19:23:24 +02:00
}
2019-04-29 09:52:01 -04:00
bool CWalletTx : : SubmitMemoryPoolAndRelay ( std : : string & err_string , bool relay )
2011-06-26 19:23:24 +02:00
{
2019-03-26 14:56:40 -04:00
// Can't relay if wallet is not broadcasting
if ( ! pwallet - > GetBroadcastTransactions ( ) ) return false ;
// Don't relay abandoned transactions
if ( isAbandoned ( ) ) return false ;
2019-08-06 14:38:34 -04:00
// Don't try to submit coinbase transactions. These would fail anyway but would
// cause log spam.
if ( IsCoinBase ( ) ) return false ;
2019-08-09 11:07:30 -04:00
// Don't try to submit conflicted or confirmed transactions.
2019-04-29 09:52:01 -04:00
if ( GetDepthInMainChain ( ) ! = 0 ) return false ;
2019-03-26 14:56:40 -04:00
2019-04-11 15:58:53 +00:00
// Submit transaction to mempool for relay
pwallet - > WalletLogPrintf ( " Submitting wtx %s to mempool for relay \n " , GetHash ( ) . ToString ( ) ) ;
// We must set fInMempool here - while it will be re-set to true by the
// entered-mempool callback, if we did not there would be a race where a
// user could call sendmoney in a loop and hit spurious out of funds errors
// because we think that this newly generated transaction's change is
// unavailable as we're not yet aware that it is in the mempool.
//
// Irrespective of the failure reason, un-marking fInMempool
// out-of-order is incorrect - it should be unmarked when
// TransactionRemovedFromMempool fires.
2017-12-05 15:57:12 -05:00
bool ret = pwallet - > chain ( ) . broadcastTransaction ( tx , pwallet - > m_default_max_tx_fee , relay , err_string ) ;
2019-04-11 15:58:53 +00:00
fInMempool | = ret ;
return ret ;
2011-06-26 19:23:24 +02:00
}
2017-01-27 10:33:45 +09:00
std : : set < uint256 > CWalletTx : : GetConflicts ( ) const
2014-02-13 20:12:51 -05:00
{
2017-01-27 10:33:45 +09:00
std : : set < uint256 > result ;
2017-08-07 07:36:37 +02:00
if ( pwallet ! = nullptr )
2014-02-13 20:12:51 -05:00
{
uint256 myHash = GetHash ( ) ;
2014-07-17 14:09:46 +02:00
result = pwallet - > GetConflicts ( myHash ) ;
2014-02-13 20:12:51 -05:00
result . erase ( myHash ) ;
}
return result ;
}
2019-04-10 11:40:35 +09:00
CAmount CWalletTx : : GetCachableAmount ( AmountType type , const isminefilter & filter , bool recalculate ) const
{
auto & amount = m_amounts [ type ] ;
if ( recalculate | | ! amount . m_cached [ filter ] ) {
amount . Set ( filter , type = = DEBIT ? pwallet - > GetDebit ( * tx , filter ) : pwallet - > GetCredit ( * tx , filter ) ) ;
2020-01-07 13:47:20 +00:00
m_is_cache_empty = false ;
2019-04-10 11:40:35 +09:00
}
return amount . m_value [ filter ] ;
}
2014-12-18 20:00:01 -05:00
CAmount CWalletTx : : GetDebit ( const isminefilter & filter ) const
{
2016-11-11 16:54:51 -08:00
if ( tx - > vin . empty ( ) )
2014-12-18 20:00:01 -05:00
return 0 ;
CAmount debit = 0 ;
2019-04-10 11:40:35 +09:00
if ( filter & ISMINE_SPENDABLE ) {
debit + = GetCachableAmount ( DEBIT , ISMINE_SPENDABLE ) ;
2014-12-18 20:00:01 -05:00
}
2019-04-10 11:40:35 +09:00
if ( filter & ISMINE_WATCH_ONLY ) {
debit + = GetCachableAmount ( DEBIT , ISMINE_WATCH_ONLY ) ;
2014-12-18 20:00:01 -05:00
}
return debit ;
}
2019-04-29 09:52:01 -04:00
CAmount CWalletTx : : GetCredit ( const isminefilter & filter ) const
2014-12-18 20:00:01 -05:00
{
// Must wait until coinbase is safely deep enough in the chain before valuing it
2019-04-29 09:52:01 -04:00
if ( IsImmatureCoinBase ( ) )
2014-12-18 20:00:01 -05:00
return 0 ;
2016-11-15 09:19:23 -05:00
CAmount credit = 0 ;
2019-04-10 11:40:35 +09:00
if ( filter & ISMINE_SPENDABLE ) {
2014-12-18 20:00:01 -05:00
// GetBalance can assume transactions in mapWallet won't change
2019-04-10 11:40:35 +09:00
credit + = GetCachableAmount ( CREDIT , ISMINE_SPENDABLE ) ;
2014-12-18 20:00:01 -05:00
}
2019-04-10 11:40:35 +09:00
if ( filter & ISMINE_WATCH_ONLY ) {
credit + = GetCachableAmount ( CREDIT , ISMINE_WATCH_ONLY ) ;
2014-12-18 20:00:01 -05:00
}
return credit ;
}
2019-04-29 09:52:01 -04:00
CAmount CWalletTx : : GetImmatureCredit ( bool fUseCache ) const
2014-12-18 20:00:01 -05:00
{
2019-04-29 09:52:01 -04:00
if ( IsImmatureCoinBase ( ) & & IsInMainChain ( ) ) {
2019-04-10 11:40:35 +09:00
return GetCachableAmount ( IMMATURE_CREDIT , ISMINE_SPENDABLE , ! fUseCache ) ;
2014-12-18 20:00:01 -05:00
}
return 0 ;
}
2019-04-29 09:52:01 -04:00
CAmount CWalletTx : : GetAvailableCredit ( bool fUseCache , const isminefilter & filter ) const
2014-12-18 20:00:01 -05:00
{
2017-06-21 21:10:00 +02:00
if ( pwallet = = nullptr )
2014-12-18 20:00:01 -05:00
return 0 ;
2019-04-10 11:40:35 +09:00
// Avoid caching ismine for NO or ALL cases (could remove this check and simplify in the future).
2018-09-13 13:53:19 +09:00
bool allow_cache = ( filter & ISMINE_ALL ) & & ( filter & ISMINE_ALL ) ! = ISMINE_ALL ;
2019-04-10 11:40:35 +09:00
2014-12-18 20:00:01 -05:00
// Must wait until coinbase is safely deep enough in the chain before valuing it
2019-04-29 09:52:01 -04:00
if ( IsImmatureCoinBase ( ) )
2014-12-18 20:00:01 -05:00
return 0 ;
2019-04-10 11:40:35 +09:00
if ( fUseCache & & allow_cache & & m_amounts [ AVAILABLE_CREDIT ] . m_cached [ filter ] ) {
return m_amounts [ AVAILABLE_CREDIT ] . m_value [ filter ] ;
2018-06-25 14:39:36 -04:00
}
2014-12-18 20:00:01 -05:00
2018-09-13 13:53:19 +09:00
bool allow_used_addresses = ( filter & ISMINE_USED ) | | ! pwallet - > IsWalletFlagSet ( WALLET_FLAG_AVOID_REUSE ) ;
2014-12-18 20:00:01 -05:00
CAmount nCredit = 0 ;
uint256 hashTx = GetHash ( ) ;
2016-11-11 16:54:51 -08:00
for ( unsigned int i = 0 ; i < tx - > vout . size ( ) ; i + + )
2014-12-18 20:00:01 -05:00
{
2020-02-21 21:15:13 +00:00
if ( ! pwallet - > IsSpent ( hashTx , i ) & & ( allow_used_addresses | | ! pwallet - > IsSpentKey ( hashTx , i ) ) ) {
2016-11-11 16:54:51 -08:00
const CTxOut & txout = tx - > vout [ i ] ;
2018-06-25 14:39:36 -04:00
nCredit + = pwallet - > GetCredit ( txout , filter ) ;
2014-12-18 20:00:01 -05:00
if ( ! MoneyRange ( nCredit ) )
2017-07-04 22:22:53 +03:00
throw std : : runtime_error ( std : : string ( __func__ ) + " : value out of range " ) ;
2014-12-18 20:00:01 -05:00
}
}
2019-04-10 11:40:35 +09:00
if ( allow_cache ) {
m_amounts [ AVAILABLE_CREDIT ] . Set ( filter , nCredit ) ;
2020-01-07 13:47:20 +00:00
m_is_cache_empty = false ;
2018-06-25 14:39:36 -04:00
}
2019-04-10 11:40:35 +09:00
2014-12-18 20:00:01 -05:00
return nCredit ;
}
2019-04-29 09:52:01 -04:00
CAmount CWalletTx : : GetImmatureWatchOnlyCredit ( const bool fUseCache ) const
2014-12-18 20:00:01 -05:00
{
2019-04-29 09:52:01 -04:00
if ( IsImmatureCoinBase ( ) & & IsInMainChain ( ) ) {
2019-04-10 11:40:35 +09:00
return GetCachableAmount ( IMMATURE_CREDIT , ISMINE_WATCH_ONLY , ! fUseCache ) ;
2014-12-18 20:00:01 -05:00
}
return 0 ;
}
CAmount CWalletTx : : GetChange ( ) const
{
if ( fChangeCached )
return nChangeCached ;
2017-05-09 15:46:26 +09:00
nChangeCached = pwallet - > GetChange ( * tx ) ;
2014-12-18 20:00:01 -05:00
fChangeCached = true ;
return nChangeCached ;
}
2015-11-30 16:15:15 +01:00
bool CWalletTx : : InMempool ( ) const
{
Use callbacks to cache whether wallet transactions are in mempool
This avoid calling out to mempool state during coin selection,
balance calculation, etc. In the next commit we ensure all wallet
callbacks from CValidationInterface happen in the same queue,
serialized with each other. This helps to avoid re-introducing one
of the issues described in #9584 [1] by further disconnecting
wallet from current chain/mempool state.
Thanks to @morcos for the suggestion to do this.
Note that there are several race conditions introduced here:
* If a user calls sendrawtransaction from RPC, adding a
transaction which is "trusted" (ie from them) and pays them
change, it may not be immediately used by coin selection until
the notification callbacks finish running. No such race is
introduced in normal transaction-sending RPCs as this case is
explicitly handled.
* Until Block{Connected,Disconnected} and
TransactionAddedToMempool calls also run in the CSceduler
background thread, there is a race where
TransactionAddedToMempool might be called after a
Block{Connected,Disconnected} call happens.
* Wallet will write a new best chain from the SetBestChain
callback prior to having processed the transaction from that
block.
[1] "you could go to select coins, need to use 0-conf change, but
such 0-conf change may have been included in a block who's
callbacks have not yet been processed - resulting in thinking they
are not in mempool and, thus, not selectable."
2017-01-20 16:38:07 -05:00
return fInMempool ;
2015-11-30 16:15:15 +01:00
}
2017-07-31 15:30:21 -04:00
bool CWalletTx : : IsTrusted ( interfaces : : Chain : : Lock & locked_chain ) const
2019-08-30 13:25:41 -07:00
{
std : : set < uint256 > s ;
return IsTrusted ( locked_chain , s ) ;
}
2019-09-06 09:15:49 -07:00
bool CWalletTx : : IsTrusted ( interfaces : : Chain : : Lock & locked_chain , std : : set < uint256 > & trusted_parents ) const
2014-12-18 20:00:01 -05:00
{
// Quick answer in most cases
2019-09-06 09:24:35 -07:00
if ( ! locked_chain . checkFinalTx ( * tx ) ) return false ;
2019-04-29 09:52:01 -04:00
int nDepth = GetDepthInMainChain ( ) ;
2019-09-06 09:24:35 -07:00
if ( nDepth > = 1 ) return true ;
if ( nDepth < 0 ) return false ;
// using wtx's cached debit
if ( ! pwallet - > m_spend_zero_conf_change | | ! IsFromMe ( ISMINE_ALL ) ) return false ;
2014-12-18 20:00:01 -05:00
2015-11-26 18:42:07 +01:00
// Don't trust unconfirmed transactions from us unless they are in the mempool.
2019-09-06 09:24:35 -07:00
if ( ! InMempool ( ) ) return false ;
2015-11-26 18:42:07 +01:00
2014-12-18 20:00:01 -05:00
// Trusted if all inputs are from us and are in the mempool:
2017-06-02 03:18:57 +02:00
for ( const CTxIn & txin : tx - > vin )
2014-12-18 20:00:01 -05:00
{
// Transactions not sent by us: not trusted
const CWalletTx * parent = pwallet - > GetWalletTx ( txin . prevout . hash ) ;
2019-09-06 09:24:35 -07:00
if ( parent = = nullptr ) return false ;
2016-11-11 16:54:51 -08:00
const CTxOut & parentOut = parent - > tx - > vout [ txin . prevout . n ] ;
2019-08-30 12:39:41 -07:00
// Check that this specific input being spent is trusted
2019-09-06 09:24:35 -07:00
if ( pwallet - > IsMine ( parentOut ) ! = ISMINE_SPENDABLE ) return false ;
2019-08-30 13:25:41 -07:00
// If we've already trusted this parent, continue
2019-09-06 09:24:35 -07:00
if ( trusted_parents . count ( parent - > GetHash ( ) ) ) continue ;
2019-08-30 12:39:41 -07:00
// Recurse to check that the parent is also trusted
2019-09-06 09:24:35 -07:00
if ( ! parent - > IsTrusted ( locked_chain , trusted_parents ) ) return false ;
2019-09-06 09:15:49 -07:00
trusted_parents . insert ( parent - > GetHash ( ) ) ;
2014-12-18 20:00:01 -05:00
}
return true ;
}
2016-11-11 16:54:51 -08:00
bool CWalletTx : : IsEquivalentTo ( const CWalletTx & _tx ) const
2015-07-02 20:57:39 +02:00
{
2018-05-04 17:38:49 -04:00
CMutableTransaction tx1 { * this - > tx } ;
CMutableTransaction tx2 { * _tx . tx } ;
2017-05-18 09:42:14 +02:00
for ( auto & txin : tx1 . vin ) txin . scriptSig = CScript ( ) ;
for ( auto & txin : tx2 . vin ) txin . scriptSig = CScript ( ) ;
2015-07-02 20:57:39 +02:00
return CTransaction ( tx1 ) = = CTransaction ( tx2 ) ;
}
2019-03-20 17:46:38 -04:00
// Rebroadcast transactions from the wallet. We do this on a random timer
// to slightly obfuscate which transactions come from our wallet.
//
// Ideally, we'd only resend transactions that we think should have been
// mined in the most recent block. Any transaction that wasn't in the top
// blockweight of transactions in the mempool shouldn't have been mined,
// and so is probably just sitting in the mempool waiting to be confirmed.
// Rebroadcasting does nothing to speed up confirmation and only damages
// privacy.
2019-03-20 18:07:52 -04:00
void CWallet : : ResendWalletTransactions ( )
2011-06-26 19:23:24 +02:00
{
2019-03-20 17:46:38 -04:00
// During reindex, importing and IBD, old wallet transactions become
// unconfirmed. Don't resend them as that would spam other nodes.
if ( ! chain ( ) . isReadyToBroadcast ( ) ) return ;
2011-06-26 19:23:24 +02:00
// Do this infrequently and randomly to avoid giving away
// that these are our transactions.
2019-03-22 14:37:30 -04:00
if ( GetTime ( ) < nNextResend | | ! fBroadcastTransactions ) return ;
2013-04-19 21:28:25 +00:00
bool fFirst = ( nNextResend = = 0 ) ;
nNextResend = GetTime ( ) + GetRand ( 30 * 60 ) ;
2019-03-22 14:37:30 -04:00
if ( fFirst ) return ;
2011-06-26 19:23:24 +02:00
// Only do it if there's been a new block since last time
2019-03-28 13:15:47 -04:00
if ( m_best_block_time < nLastResend ) return ;
2013-04-19 21:28:25 +00:00
nLastResend = GetTime ( ) ;
2011-06-26 19:23:24 +02:00
2019-04-11 15:58:53 +00:00
int submitted_tx_count = 0 ;
2019-03-22 14:37:30 -04:00
2019-03-20 18:07:52 -04:00
{ // locked_chain and cs_wallet scope
auto locked_chain = chain ( ) . lock ( ) ;
2019-03-22 14:37:30 -04:00
LOCK ( cs_wallet ) ;
// Relay transactions
for ( std : : pair < const uint256 , CWalletTx > & item : mapWallet ) {
CWalletTx & wtx = item . second ;
2019-08-09 11:07:30 -04:00
// Attempt to rebroadcast all txes more than 5 minutes older than
// the last block. SubmitMemoryPoolAndRelay() will not rebroadcast
// any confirmed or conflicting txs.
2019-03-28 13:15:47 -04:00
if ( wtx . nTimeReceived > m_best_block_time - 5 * 60 ) continue ;
2019-04-11 15:58:53 +00:00
std : : string unused_err_string ;
2019-04-29 09:52:01 -04:00
if ( wtx . SubmitMemoryPoolAndRelay ( unused_err_string , true ) ) + + submitted_tx_count ;
2019-03-22 14:37:30 -04:00
}
2019-03-20 18:07:52 -04:00
} // locked_chain and cs_wallet
2019-03-22 14:37:30 -04:00
2019-04-11 15:58:53 +00:00
if ( submitted_tx_count > 0 ) {
WalletLogPrintf ( " %s: resubmit %u unconfirmed transactions \n " , __func__ , submitted_tx_count ) ;
2019-03-22 14:37:30 -04:00
}
2011-06-26 19:23:24 +02:00
}
2014-10-26 15:03:12 +08:00
/** @} */ // end of mapWallet
2011-06-26 19:23:24 +02:00
2019-03-20 17:46:38 -04:00
void MaybeResendWalletTxs ( )
{
for ( const std : : shared_ptr < CWallet > & pwallet : GetWallets ( ) ) {
2019-03-20 18:07:52 -04:00
pwallet - > ResendWalletTransactions ( ) ;
2019-03-20 17:46:38 -04:00
}
}
2011-06-26 19:23:24 +02:00
2014-10-26 15:03:12 +08:00
/** @defgroup Actions
*
* @ {
*/
2011-06-26 19:23:24 +02:00
2018-09-13 13:53:19 +09:00
CWallet : : Balance CWallet : : GetBalance ( const int min_depth , bool avoid_reuse ) const
2011-06-26 19:23:24 +02:00
{
2019-03-11 16:12:58 -04:00
Balance ret ;
2019-06-25 20:18:12 +03:00
isminefilter reuse_filter = avoid_reuse ? ISMINE_NO : ISMINE_USED ;
2011-06-26 19:23:24 +02:00
{
2017-07-26 10:23:01 -04:00
auto locked_chain = chain ( ) . lock ( ) ;
LOCK ( cs_wallet ) ;
2019-09-06 09:15:49 -07:00
std : : set < uint256 > trusted_parents ;
2017-06-04 22:02:43 +02:00
for ( const auto & entry : mapWallet )
2011-06-26 19:23:24 +02:00
{
2019-03-10 17:29:54 -04:00
const CWalletTx & wtx = entry . second ;
2019-09-06 09:15:49 -07:00
const bool is_trusted { wtx . IsTrusted ( * locked_chain , trusted_parents ) } ;
2019-04-29 09:52:01 -04:00
const int tx_depth { wtx . GetDepthInMainChain ( ) } ;
const CAmount tx_credit_mine { wtx . GetAvailableCredit ( /* fUseCache */ true , ISMINE_SPENDABLE | reuse_filter ) } ;
const CAmount tx_credit_watchonly { wtx . GetAvailableCredit ( /* fUseCache */ true , ISMINE_WATCH_ONLY | reuse_filter ) } ;
2019-03-11 16:12:58 -04:00
if ( is_trusted & & tx_depth > = min_depth ) {
ret . m_mine_trusted + = tx_credit_mine ;
ret . m_watchonly_trusted + = tx_credit_watchonly ;
2018-06-25 14:39:36 -04:00
}
2019-03-11 16:12:58 -04:00
if ( ! is_trusted & & tx_depth = = 0 & & wtx . InMempool ( ) ) {
ret . m_mine_untrusted_pending + = tx_credit_mine ;
ret . m_watchonly_untrusted_pending + = tx_credit_watchonly ;
}
2019-04-29 09:52:01 -04:00
ret . m_mine_immature + = wtx . GetImmatureCredit ( ) ;
ret . m_watchonly_immature + = wtx . GetImmatureWatchOnlyCredit ( ) ;
2011-06-26 19:23:24 +02:00
}
}
2019-03-11 16:12:58 -04:00
return ret ;
2014-03-29 05:15:28 +01:00
}
2017-04-28 14:10:21 -04:00
CAmount CWallet : : GetAvailableBalance ( const CCoinControl * coinControl ) const
{
2017-07-26 10:23:01 -04:00
auto locked_chain = chain ( ) . lock ( ) ;
LOCK ( cs_wallet ) ;
2017-05-04 17:02:12 -04:00
2017-04-28 14:10:21 -04:00
CAmount balance = 0 ;
std : : vector < COutput > vCoins ;
2017-07-31 15:30:21 -04:00
AvailableCoins ( * locked_chain , vCoins , true , coinControl ) ;
2017-04-28 14:10:21 -04:00
for ( const COutput & out : vCoins ) {
if ( out . fSpendable ) {
balance + = out . tx - > tx - > vout [ out . i ] . nValue ;
}
}
return balance ;
}
2019-05-09 18:34:38 -07:00
void CWallet : : AvailableCoins ( interfaces : : Chain : : Lock & locked_chain , std : : vector < COutput > & vCoins , bool fOnlySafe , const CCoinControl * coinControl , const CAmount & nMinimumAmount , const CAmount & nMaximumAmount , const CAmount & nMinimumSumAmount , const uint64_t nMaximumCount ) const
2012-02-27 04:19:32 -08:00
{
2018-02-02 11:37:50 +00:00
AssertLockHeld ( cs_wallet ) ;
2012-02-27 04:19:32 -08:00
vCoins . clear ( ) ;
2018-02-08 18:18:51 +00:00
CAmount nTotal = 0 ;
2018-09-13 13:53:19 +09:00
// Either the WALLET_FLAG_AVOID_REUSE flag is not set (in which case we always allow), or we default to avoiding, and only in the case where
// a coin control object is provided, and has the avoid address reuse flag set to false, do we allow already used addresses
bool allow_used_addresses = ! IsWalletFlagSet ( WALLET_FLAG_AVOID_REUSE ) | | ( coinControl & & ! coinControl - > m_avoid_address_reuse ) ;
2019-05-09 18:34:38 -07:00
const int min_depth = { coinControl ? coinControl - > m_min_depth : DEFAULT_MIN_DEPTH } ;
const int max_depth = { coinControl ? coinControl - > m_max_depth : DEFAULT_MAX_DEPTH } ;
2012-02-27 04:19:32 -08:00
2019-09-06 09:15:49 -07:00
std : : set < uint256 > trusted_parents ;
2018-02-08 18:18:51 +00:00
for ( const auto & entry : mapWallet )
2012-02-27 04:19:32 -08:00
{
2018-02-08 18:18:51 +00:00
const uint256 & wtxid = entry . first ;
2019-03-10 17:29:54 -04:00
const CWalletTx & wtx = entry . second ;
2017-03-07 18:17:32 +00:00
2019-03-10 17:29:54 -04:00
if ( ! locked_chain . checkFinalTx ( * wtx . tx ) ) {
2018-02-08 18:18:51 +00:00
continue ;
2019-02-01 17:35:51 -05:00
}
2017-03-07 18:17:32 +00:00
2019-04-29 09:52:01 -04:00
if ( wtx . IsImmatureCoinBase ( ) )
2018-02-08 18:18:51 +00:00
continue ;
2012-02-27 04:19:32 -08:00
2019-04-29 09:52:01 -04:00
int nDepth = wtx . GetDepthInMainChain ( ) ;
2018-02-08 18:18:51 +00:00
if ( nDepth < 0 )
continue ;
2012-05-31 16:01:16 -04:00
2018-02-08 18:18:51 +00:00
// We should not consider coins which aren't at least in our mempool
// It's possible for these to be conflicted via ancestors which we may never be able to detect
2019-03-10 17:29:54 -04:00
if ( nDepth = = 0 & & ! wtx . InMempool ( ) )
2018-02-08 18:18:51 +00:00
continue ;
2012-02-27 04:19:32 -08:00
2019-09-06 09:15:49 -07:00
bool safeTx = wtx . IsTrusted ( locked_chain , trusted_parents ) ;
2018-02-08 18:18:51 +00:00
// We should not consider coins from transactions that are replacing
// other transactions.
//
// Example: There is a transaction A which is replaced by bumpfee
// transaction B. In this case, we want to prevent creation of
// a transaction B' which spends an output of B.
//
// Reason: If transaction A were initially confirmed, transactions B
// and B' would no longer be valid, so the user would have to create
// a new transaction C to replace B'. However, in the case of a
// one-block reorg, transactions B' and C might BOTH be accepted,
// when the user only wanted one of them. Specifically, there could
// be a 1-block reorg away from the chain where transactions A and C
// were accepted to another chain where B, B', and C were all
// accepted.
2019-03-10 17:29:54 -04:00
if ( nDepth = = 0 & & wtx . mapValue . count ( " replaces_txid " ) ) {
2018-02-08 18:18:51 +00:00
safeTx = false ;
}
2014-02-12 13:43:07 -05:00
2018-02-08 18:18:51 +00:00
// Similarly, we should not consider coins from transactions that
// have been replaced. In the example above, we would want to prevent
// creation of a transaction A' spending an output of A, because if
// transaction B were initially confirmed, conflicting with A and
// A', we wouldn't want to the user to create a transaction D
// intending to replace A', but potentially resulting in a scenario
// where A, A', and D could all be accepted (instead of just B and
// D, or just A and A' like the user would want).
2019-03-10 17:29:54 -04:00
if ( nDepth = = 0 & & wtx . mapValue . count ( " replaced_by_txid " ) ) {
2018-02-08 18:18:51 +00:00
safeTx = false ;
}
2016-03-17 12:48:05 -04:00
2018-02-08 18:18:51 +00:00
if ( fOnlySafe & & ! safeTx ) {
continue ;
}
2016-12-09 13:45:27 -05:00
2019-05-09 18:34:38 -07:00
if ( nDepth < min_depth | | nDepth > max_depth ) {
2018-02-08 18:18:51 +00:00
continue ;
2019-05-09 18:34:38 -07:00
}
2017-02-23 11:20:16 -05:00
2019-03-10 17:29:54 -04:00
for ( unsigned int i = 0 ; i < wtx . tx - > vout . size ( ) ; i + + ) {
if ( wtx . tx - > vout [ i ] . nValue < nMinimumAmount | | wtx . tx - > vout [ i ] . nValue > nMaximumAmount )
2016-12-09 13:45:27 -05:00
continue ;
2018-02-08 18:18:51 +00:00
if ( coinControl & & coinControl - > HasSelected ( ) & & ! coinControl - > fAllowOtherInputs & & ! coinControl - > IsSelected ( COutPoint ( entry . first , i ) ) )
2017-03-07 18:17:32 +00:00
continue ;
2018-02-08 18:18:51 +00:00
if ( IsLockedCoin ( entry . first , i ) )
continue ;
2017-03-07 18:17:32 +00:00
2019-04-29 09:52:01 -04:00
if ( IsSpent ( wtxid , i ) )
2018-02-08 18:18:51 +00:00
continue ;
2017-03-07 18:17:32 +00:00
2019-03-10 17:29:54 -04:00
isminetype mine = IsMine ( wtx . tx - > vout [ i ] ) ;
2017-03-07 18:17:32 +00:00
2018-02-08 18:18:51 +00:00
if ( mine = = ISMINE_NO ) {
continue ;
}
2017-03-07 18:17:32 +00:00
2020-02-21 21:15:13 +00:00
if ( ! allow_used_addresses & & IsSpentKey ( wtxid , i ) ) {
2018-09-13 13:53:19 +09:00
continue ;
}
2020-02-10 21:27:59 -05:00
std : : unique_ptr < SigningProvider > provider = GetSolvingProvider ( wtx . tx - > vout [ i ] . scriptPubKey ) ;
2019-10-07 14:11:34 -04:00
bool solvable = provider ? IsSolvable ( * provider , wtx . tx - > vout [ i ] . scriptPubKey ) : false ;
2018-05-01 12:05:55 -07:00
bool spendable = ( ( mine & ISMINE_SPENDABLE ) ! = ISMINE_NO ) | | ( ( ( mine & ISMINE_WATCH_ONLY ) ! = ISMINE_NO ) & & ( coinControl & & coinControl - > fAllowWatchOnly & & solvable ) ) ;
2017-03-07 18:17:32 +00:00
2019-03-10 17:29:54 -04:00
vCoins . push_back ( COutput ( & wtx , i , nDepth , spendable , solvable , safeTx , ( coinControl & & coinControl - > fAllowWatchOnly ) ) ) ;
2017-03-07 18:17:32 +00:00
2018-02-08 18:18:51 +00:00
// Checks the sum amount of all UTXO's.
if ( nMinimumSumAmount ! = MAX_MONEY ) {
2019-03-10 17:29:54 -04:00
nTotal + = wtx . tx - > vout [ i ] . nValue ;
2017-03-07 18:17:32 +00:00
2018-02-08 18:18:51 +00:00
if ( nTotal > = nMinimumSumAmount ) {
2017-03-07 18:17:32 +00:00
return ;
}
2012-09-27 13:52:09 -04:00
}
2018-02-08 18:18:51 +00:00
// Checks the maximum number of UTXO's.
if ( nMaximumCount > 0 & & vCoins . size ( ) > = nMaximumCount ) {
return ;
}
2012-02-27 04:19:32 -08:00
}
}
}
2017-07-31 15:30:21 -04:00
std : : map < CTxDestination , std : : vector < COutput > > CWallet : : ListCoins ( interfaces : : Chain : : Lock & locked_chain ) const
2017-04-28 14:10:21 -04:00
{
2017-06-15 14:56:35 -04:00
AssertLockHeld ( cs_wallet ) ;
2017-04-28 14:10:21 -04:00
std : : map < CTxDestination , std : : vector < COutput > > result ;
std : : vector < COutput > availableCoins ;
2017-07-31 15:30:21 -04:00
AvailableCoins ( locked_chain , availableCoins ) ;
2018-02-02 11:37:50 +00:00
2018-06-18 07:58:28 +02:00
for ( const COutput & coin : availableCoins ) {
2017-04-28 14:10:21 -04:00
CTxDestination address ;
2019-09-23 19:13:55 +02:00
if ( ( coin . fSpendable | | ( IsWalletFlagSet ( WALLET_FLAG_DISABLE_PRIVATE_KEYS ) & & coin . fSolvable ) ) & &
2017-04-28 14:10:21 -04:00
ExtractDestination ( FindNonChangeParentOutput ( * coin . tx - > tx , coin . i ) . scriptPubKey , address ) ) {
result [ address ] . emplace_back ( std : : move ( coin ) ) ;
}
}
std : : vector < COutPoint > lockedCoins ;
ListLockedCoins ( lockedCoins ) ;
2019-12-05 11:53:23 -05:00
// Include watch-only for LegacyScriptPubKeyMan wallets without private keys
const bool include_watch_only = GetLegacyScriptPubKeyMan ( ) & & IsWalletFlagSet ( WALLET_FLAG_DISABLE_PRIVATE_KEYS ) ;
2019-09-23 19:13:55 +02:00
const isminetype is_mine_filter = include_watch_only ? ISMINE_WATCH_ONLY : ISMINE_SPENDABLE ;
2018-06-18 07:58:28 +02:00
for ( const COutPoint & output : lockedCoins ) {
2017-04-28 14:10:21 -04:00
auto it = mapWallet . find ( output . hash ) ;
if ( it ! = mapWallet . end ( ) ) {
2019-04-29 09:52:01 -04:00
int depth = it - > second . GetDepthInMainChain ( ) ;
2017-04-28 14:10:21 -04:00
if ( depth > = 0 & & output . n < it - > second . tx - > vout . size ( ) & &
2019-09-23 19:13:55 +02:00
IsMine ( it - > second . tx - > vout [ output . n ] ) = = is_mine_filter
) {
2017-04-28 14:10:21 -04:00
CTxDestination address ;
if ( ExtractDestination ( FindNonChangeParentOutput ( * it - > second . tx , output . n ) . scriptPubKey , address ) ) {
result [ address ] . emplace_back (
& it - > second , output . n , depth , true /* spendable */ , true /* solvable */ , false /* safe */ ) ;
}
}
}
}
return result ;
}
const CTxOut & CWallet : : FindNonChangeParentOutput ( const CTransaction & tx , int output ) const
{
const CTransaction * ptx = & tx ;
int n = output ;
while ( IsChange ( ptx - > vout [ n ] ) & & ptx - > vin . size ( ) > 0 ) {
const COutPoint & prevout = ptx - > vin [ 0 ] . prevout ;
auto it = mapWallet . find ( prevout . hash ) ;
if ( it = = mapWallet . end ( ) | | it - > second . tx - > vout . size ( ) < = prevout . n | |
! IsMine ( it - > second . tx - > vout [ prevout . n ] ) ) {
break ;
}
ptx = it - > second . tx . get ( ) ;
n = prevout . n ;
}
return ptx - > vout [ n ] ;
}
2018-07-19 11:45:26 +09:00
bool CWallet : : SelectCoinsMinConf ( const CAmount & nTargetValue , const CoinEligibilityFilter & eligibility_filter , std : : vector < OutputGroup > groups ,
2018-03-05 16:42:49 -05:00
std : : set < CInputCoin > & setCoinsRet , CAmount & nValueRet , const CoinSelectionParams & coin_selection_params , bool & bnb_used ) const
2011-06-26 19:23:24 +02:00
{
setCoinsRet . clear ( ) ;
nValueRet = 0 ;
2018-07-19 11:45:26 +09:00
std : : vector < OutputGroup > utxo_pool ;
2018-03-05 16:42:49 -05:00
if ( coin_selection_params . use_bnb ) {
// Get long term estimate
FeeCalculation feeCalc ;
CCoinControl temp ;
temp . m_confirm_target = 1008 ;
2017-07-28 21:40:29 -04:00
CFeeRate long_term_feerate = GetMinimumFeeRate ( * this , temp , & feeCalc ) ;
2016-12-02 15:29:20 -05:00
2018-03-05 16:42:49 -05:00
// Calculate cost of change
2017-07-28 21:40:29 -04:00
CAmount cost_of_change = GetDiscardRate ( * this ) . GetFee ( coin_selection_params . change_spend_size ) + coin_selection_params . effective_fee . GetFee ( coin_selection_params . change_output_size ) ;
2011-06-26 19:23:24 +02:00
2018-03-05 16:42:49 -05:00
// Filter by the min conf specs and add to utxo_pool and calculate effective value
2018-07-19 11:45:26 +09:00
for ( OutputGroup & group : groups ) {
if ( ! group . EligibleForSpending ( eligibility_filter ) ) continue ;
group . fee = 0 ;
group . long_term_fee = 0 ;
group . effective_value = 0 ;
for ( auto it = group . m_outputs . begin ( ) ; it ! = group . m_outputs . end ( ) ; ) {
const CInputCoin & coin = * it ;
CAmount effective_value = coin . txout . nValue - ( coin . m_input_bytes < 0 ? 0 : coin_selection_params . effective_fee . GetFee ( coin . m_input_bytes ) ) ;
// Only include outputs that are positive effective value (i.e. not dust)
if ( effective_value > 0 ) {
group . fee + = coin . m_input_bytes < 0 ? 0 : coin_selection_params . effective_fee . GetFee ( coin . m_input_bytes ) ;
group . long_term_fee + = coin . m_input_bytes < 0 ? 0 : long_term_feerate . GetFee ( coin . m_input_bytes ) ;
2019-10-25 18:45:58 -04:00
if ( coin_selection_params . m_subtract_fee_outputs ) {
group . effective_value + = coin . txout . nValue ;
} else {
group . effective_value + = effective_value ;
}
2018-07-19 11:45:26 +09:00
+ + it ;
} else {
it = group . Discard ( coin ) ;
}
2011-06-26 19:23:24 +02:00
}
2018-07-19 11:45:26 +09:00
if ( group . effective_value > 0 ) utxo_pool . push_back ( group ) ;
2018-03-05 16:42:49 -05:00
}
// Calculate the fees for things that aren't inputs
CAmount not_input_fees = coin_selection_params . effective_fee . GetFee ( coin_selection_params . tx_noinputs_size ) ;
bnb_used = true ;
return SelectCoinsBnB ( utxo_pool , nTargetValue , cost_of_change , setCoinsRet , nValueRet , not_input_fees ) ;
} else {
// Filter by the min conf specs and add to utxo_pool
2018-07-19 11:45:26 +09:00
for ( const OutputGroup & group : groups ) {
if ( ! group . EligibleForSpending ( eligibility_filter ) ) continue ;
utxo_pool . push_back ( group ) ;
2016-12-25 20:19:40 +00:00
}
2018-03-05 16:42:49 -05:00
bnb_used = false ;
return KnapsackSolver ( nTargetValue , utxo_pool , setCoinsRet , nValueRet ) ;
2011-06-26 19:23:24 +02:00
}
}
2018-03-15 02:42:18 -04:00
bool CWallet : : SelectCoins ( const std : : vector < COutput > & vAvailableCoins , const CAmount & nTargetValue , std : : set < CInputCoin > & setCoinsRet , CAmount & nValueRet , const CCoinControl & coin_control , CoinSelectionParams & coin_selection_params , bool & bnb_used ) const
2011-06-26 19:23:24 +02:00
{
2017-01-27 10:33:45 +09:00
std : : vector < COutput > vCoins ( vAvailableCoins ) ;
2019-10-25 18:10:13 -04:00
CAmount value_to_select = nTargetValue ;
2013-08-12 17:03:03 +02:00
2019-11-25 02:19:57 -05:00
// Default to bnb was not used. If we use it, we set it later
bnb_used = false ;
2013-08-12 17:03:03 +02:00
// coin control -> return all selected outputs (we want all selected to go into the transaction for sure)
2018-03-05 16:42:49 -05:00
if ( coin_control . HasSelected ( ) & & ! coin_control . fAllowOtherInputs )
2013-08-12 17:03:03 +02:00
{
2017-06-02 03:18:57 +02:00
for ( const COutput & out : vCoins )
2013-08-12 17:03:03 +02:00
{
2015-04-24 18:29:00 -07:00
if ( ! out . fSpendable )
continue ;
2016-11-11 16:54:51 -08:00
nValueRet + = out . tx - > tx - > vout [ out . i ] . nValue ;
2018-07-19 11:45:26 +09:00
setCoinsRet . insert ( out . GetInputCoin ( ) ) ;
2013-08-12 17:03:03 +02:00
}
return ( nValueRet > = nTargetValue ) ;
}
2012-02-27 04:19:32 -08:00
2015-04-24 18:29:00 -07:00
// calculate value from preset inputs and store them
2017-04-07 09:38:33 +00:00
std : : set < CInputCoin > setPresetCoins ;
2015-04-24 18:29:00 -07:00
CAmount nValueFromPresetInputs = 0 ;
std : : vector < COutPoint > vPresetInputs ;
2018-03-05 16:42:49 -05:00
coin_control . ListSelected ( vPresetInputs ) ;
2017-06-02 03:18:57 +02:00
for ( const COutPoint & outpoint : vPresetInputs )
2015-04-24 18:29:00 -07:00
{
2017-01-27 10:33:45 +09:00
std : : map < uint256 , CWalletTx > : : const_iterator it = mapWallet . find ( outpoint . hash ) ;
2015-04-24 18:29:00 -07:00
if ( it ! = mapWallet . end ( ) )
{
2019-03-10 17:29:54 -04:00
const CWalletTx & wtx = it - > second ;
2015-04-24 18:29:00 -07:00
// Clearly invalid input, fail
2019-10-25 18:10:13 -04:00
if ( wtx . tx - > vout . size ( ) < = outpoint . n ) {
2015-04-24 18:29:00 -07:00
return false ;
2019-10-25 18:10:13 -04:00
}
2018-03-05 16:42:49 -05:00
// Just to calculate the marginal byte size
2019-10-25 18:10:13 -04:00
CInputCoin coin ( wtx . tx , outpoint . n , wtx . GetSpendSize ( outpoint . n , false ) ) ;
nValueFromPresetInputs + = coin . txout . nValue ;
if ( coin . m_input_bytes < = 0 ) {
return false ; // Not solvable, can't estimate size for fee
}
coin . effective_value = coin . txout . nValue - coin_selection_params . effective_fee . GetFee ( coin . m_input_bytes ) ;
if ( coin_selection_params . use_bnb ) {
value_to_select - = coin . effective_value ;
} else {
value_to_select - = coin . txout . nValue ;
}
setPresetCoins . insert ( coin ) ;
} else {
2015-04-24 18:29:00 -07:00
return false ; // TODO: Allow non-wallet inputs
2019-10-25 18:10:13 -04:00
}
2015-04-24 18:29:00 -07:00
}
// remove preset inputs from vCoins
2018-03-05 16:42:49 -05:00
for ( std : : vector < COutput > : : iterator it = vCoins . begin ( ) ; it ! = vCoins . end ( ) & & coin_control . HasSelected ( ) ; )
2015-04-24 18:29:00 -07:00
{
2018-07-19 11:45:26 +09:00
if ( setPresetCoins . count ( it - > GetInputCoin ( ) ) )
2015-04-24 18:29:00 -07:00
it = vCoins . erase ( it ) ;
else
+ + it ;
}
2019-12-29 01:58:46 +01:00
unsigned int limit_ancestor_count = 0 ;
unsigned int limit_descendant_count = 0 ;
chain ( ) . getPackageLimits ( limit_ancestor_count , limit_descendant_count ) ;
size_t max_ancestors = ( size_t ) std : : max < int64_t > ( 1 , limit_ancestor_count ) ;
size_t max_descendants = ( size_t ) std : : max < int64_t > ( 1 , limit_descendant_count ) ;
bool fRejectLongChains = gArgs . GetBoolArg ( " -walletrejectlongchains " , DEFAULT_WALLET_REJECT_LONG_CHAINS ) ;
2018-07-19 11:45:26 +09:00
// form groups from remaining coins; note that preset coins will not
// automatically have their associated (same address) coins included
2018-07-31 01:50:43 +09:00
if ( coin_control . m_avoid_partial_spends & & vCoins . size ( ) > OUTPUT_GROUP_MAX_ENTRIES ) {
// Cases where we have 11+ outputs all pointing to the same destination may result in
// privacy leaks as they will potentially be deterministically sorted. We solve that by
// explicitly shuffling the outputs before processing
2018-10-31 15:42:03 -07:00
Shuffle ( vCoins . begin ( ) , vCoins . end ( ) , FastRandomContext ( ) ) ;
2018-07-31 01:50:43 +09:00
}
2019-12-29 01:58:46 +01:00
std : : vector < OutputGroup > groups = GroupOutputs ( vCoins , ! coin_control . m_avoid_partial_spends , max_ancestors ) ;
2016-12-02 15:29:20 -05:00
2019-10-25 18:10:13 -04:00
bool res = value_to_select < = 0 | |
SelectCoinsMinConf ( value_to_select , CoinEligibilityFilter ( 1 , 6 , 0 ) , groups , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) | |
SelectCoinsMinConf ( value_to_select , CoinEligibilityFilter ( 1 , 1 , 0 ) , groups , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) | |
( m_spend_zero_conf_change & & SelectCoinsMinConf ( value_to_select , CoinEligibilityFilter ( 0 , 1 , 2 ) , groups , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) | |
( m_spend_zero_conf_change & & SelectCoinsMinConf ( value_to_select , CoinEligibilityFilter ( 0 , 1 , std : : min ( ( size_t ) 4 , max_ancestors / 3 ) , std : : min ( ( size_t ) 4 , max_descendants / 3 ) ) , groups , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) | |
( m_spend_zero_conf_change & & SelectCoinsMinConf ( value_to_select , CoinEligibilityFilter ( 0 , 1 , max_ancestors / 2 , max_descendants / 2 ) , groups , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) | |
( m_spend_zero_conf_change & & SelectCoinsMinConf ( value_to_select , CoinEligibilityFilter ( 0 , 1 , max_ancestors - 1 , max_descendants - 1 ) , groups , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) | |
( m_spend_zero_conf_change & & ! fRejectLongChains & & SelectCoinsMinConf ( value_to_select , CoinEligibilityFilter ( 0 , 1 , std : : numeric_limits < uint64_t > : : max ( ) ) , groups , setCoinsRet , nValueRet , coin_selection_params , bnb_used ) ) ;
2015-04-24 18:29:00 -07:00
// because SelectCoinsMinConf clears the setCoinsRet, we now add the possible inputs to the coinset
2018-07-19 11:45:26 +09:00
util : : insert ( setCoinsRet , setPresetCoins ) ;
2015-04-24 18:29:00 -07:00
// add preset inputs to the total value selected
nValueRet + = nValueFromPresetInputs ;
return res ;
2011-06-26 19:23:24 +02:00
}
2020-02-10 19:49:56 -05:00
bool CWallet : : SignTransaction ( CMutableTransaction & tx ) const
2017-03-03 16:15:47 +01:00
{
2019-02-18 16:09:45 -05:00
AssertLockHeld ( cs_wallet ) ;
2017-04-10 14:05:49 +02:00
2020-02-10 19:49:56 -05:00
// Build coins map
std : : map < COutPoint , Coin > coins ;
2018-05-17 17:54:18 -07:00
for ( auto & input : tx . vin ) {
2017-03-03 16:15:47 +01:00
std : : map < uint256 , CWalletTx > : : const_iterator mi = mapWallet . find ( input . prevout . hash ) ;
2017-03-27 14:00:14 +02:00
if ( mi = = mapWallet . end ( ) | | input . prevout . n > = mi - > second . tx - > vout . size ( ) ) {
return false ;
}
2020-02-10 19:49:56 -05:00
const CWalletTx & wtx = mi - > second ;
coins [ input . prevout ] = Coin ( wtx . tx - > vout [ input . prevout . n ] , wtx . m_confirm . block_height , wtx . IsCoinBase ( ) ) ;
}
std : : map < int , std : : string > input_errors ;
return SignTransaction ( tx , coins , SIGHASH_ALL , input_errors ) ;
}
bool CWallet : : SignTransaction ( CMutableTransaction & tx , const std : : map < COutPoint , Coin > & coins , int sighash , std : : map < int , std : : string > & input_errors ) const
{
2020-02-11 17:10:25 -05:00
// Try to sign with all ScriptPubKeyMans
for ( ScriptPubKeyMan * spk_man : GetAllScriptPubKeyMans ( ) ) {
// spk_man->SignTransaction will return true if the transaction is complete,
// so we can exit early and return true if that happens
if ( spk_man - > SignTransaction ( tx , coins , sighash , input_errors ) ) {
return true ;
}
}
// At this point, one input was not fully signed otherwise we would have exited already
// Find that input and figure out what went wrong.
2020-02-10 19:49:56 -05:00
for ( unsigned int i = 0 ; i < tx . vin . size ( ) ; i + + ) {
// Get the prevout
CTxIn & txin = tx . vin [ i ] ;
auto coin = coins . find ( txin . prevout ) ;
if ( coin = = coins . end ( ) | | coin - > second . IsSpent ( ) ) {
input_errors [ i ] = " Input not found or already spent " ;
continue ;
}
2019-10-07 14:11:34 -04:00
2020-02-10 19:49:56 -05:00
// Check if this input is complete
SignatureData sigdata = DataFromTransaction ( tx , i , coin - > second . out ) ;
2020-02-11 17:10:25 -05:00
if ( ! sigdata . complete ) {
2020-02-10 19:49:56 -05:00
input_errors [ i ] = " Unable to sign input, missing keys " ;
continue ;
}
2017-03-03 16:15:47 +01:00
}
2020-02-10 19:49:56 -05:00
return false ;
2017-03-03 16:15:47 +01:00
}
2020-02-10 19:50:56 -05:00
TransactionError CWallet : : FillPSBT ( PartiallySignedTransaction & psbtx , bool & complete , int sighash_type , bool sign , bool bip32derivs ) const
{
LOCK ( cs_wallet ) ;
// Get all of the previous transactions
for ( unsigned int i = 0 ; i < psbtx . tx - > vin . size ( ) ; + + i ) {
const CTxIn & txin = psbtx . tx - > vin [ i ] ;
PSBTInput & input = psbtx . inputs . at ( i ) ;
if ( PSBTInputSigned ( input ) ) {
continue ;
}
// Verify input looks sane. This will check that we have at most one uxto, witness or non-witness.
if ( ! input . IsSane ( ) ) {
return TransactionError : : INVALID_PSBT ;
}
// If we have no utxo, grab it from the wallet.
if ( ! input . non_witness_utxo & & input . witness_utxo . IsNull ( ) ) {
const uint256 & txhash = txin . prevout . hash ;
const auto it = mapWallet . find ( txhash ) ;
if ( it ! = mapWallet . end ( ) ) {
const CWalletTx & wtx = it - > second ;
// We only need the non_witness_utxo, which is a superset of the witness_utxo.
// The signing code will switch to the smaller witness_utxo if this is ok.
input . non_witness_utxo = wtx . tx ;
}
}
2020-02-10 19:50:59 -05:00
}
// Fill in information from ScriptPubKeyMans
// Because each ScriptPubKeyMan may be able to fill more than one input, we need to keep track of each ScriptPubKeyMan that has filled this psbt.
// Each iteration, we may fill more inputs than the input that is specified in that iteration.
// We assume that each input is filled by only one ScriptPubKeyMan
std : : set < uint256 > visited_spk_mans ;
for ( unsigned int i = 0 ; i < psbtx . tx - > vin . size ( ) ; + + i ) {
const CTxIn & txin = psbtx . tx - > vin [ i ] ;
PSBTInput & input = psbtx . inputs . at ( i ) ;
2020-02-10 19:50:56 -05:00
2020-02-10 19:50:59 -05:00
if ( PSBTInputSigned ( input ) ) {
continue ;
2020-02-10 19:50:56 -05:00
}
2020-02-10 19:50:59 -05:00
// Get the scriptPubKey to know which ScriptPubKeyMan to use
2020-02-10 19:50:56 -05:00
CScript script ;
if ( ! input . witness_utxo . IsNull ( ) ) {
script = input . witness_utxo . scriptPubKey ;
} else if ( input . non_witness_utxo ) {
if ( txin . prevout . n > = input . non_witness_utxo - > vout . size ( ) ) {
return TransactionError : : MISSING_INPUTS ;
}
script = input . non_witness_utxo - > vout [ txin . prevout . n ] . scriptPubKey ;
} else {
// There's no UTXO so we can just skip this now
continue ;
}
SignatureData sigdata ;
input . FillSignatureData ( sigdata ) ;
2020-02-10 19:50:59 -05:00
std : : set < ScriptPubKeyMan * > spk_mans = GetScriptPubKeyMans ( script , sigdata ) ;
if ( spk_mans . size ( ) = = 0 ) {
2020-02-10 19:50:56 -05:00
continue ;
}
2020-02-10 19:50:59 -05:00
for ( auto & spk_man : spk_mans ) {
// If we've already been signed by this spk_man, skip it
if ( visited_spk_mans . count ( spk_man - > GetID ( ) ) > 0 ) {
continue ;
}
// Fill in the information from the spk_man
TransactionError res = spk_man - > FillPSBT ( psbtx , sighash_type , sign , bip32derivs ) ;
if ( res ! = TransactionError : : OK ) {
return res ;
}
2020-02-10 19:50:56 -05:00
2020-02-10 19:50:59 -05:00
// Add this spk_man to visited_spk_mans so we can skip it later
visited_spk_mans . insert ( spk_man - > GetID ( ) ) ;
2020-02-10 19:50:56 -05:00
}
}
2020-02-10 19:50:59 -05:00
// Complete if every input is now signed
complete = true ;
for ( const auto & input : psbtx . inputs ) {
complete & = PSBTInputSigned ( input ) ;
}
2020-02-10 19:50:56 -05:00
return TransactionError : : OK ;
}
2020-02-13 17:09:12 -05:00
SigningResult CWallet : : SignMessage ( const std : : string & message , const PKHash & pkhash , std : : string & str_sig ) const
{
SignatureData sigdata ;
CScript script_pub_key = GetScriptForDestination ( pkhash ) ;
for ( const auto & spk_man_pair : m_spk_managers ) {
if ( spk_man_pair . second - > CanProvide ( script_pub_key , sigdata ) ) {
return spk_man_pair . second - > SignMessage ( message , pkhash , str_sig ) ;
}
}
return SigningResult : : PRIVATE_KEY_NOT_AVAILABLE ;
}
2017-07-10 14:29:06 -04:00
bool CWallet : : FundTransaction ( CMutableTransaction & tx , CAmount & nFeeRet , int & nChangePosInOut , std : : string & strFailReason , bool lockUnspents , const std : : set < int > & setSubtractFeeFromOutputs , CCoinControl coinControl )
2015-04-24 18:29:00 -07:00
{
2017-01-27 10:33:45 +09:00
std : : vector < CRecipient > vecSend ;
2015-04-24 18:29:00 -07:00
2017-12-14 03:17:58 +00:00
// Turn the txout set into a CRecipient vector.
for ( size_t idx = 0 ; idx < tx . vout . size ( ) ; idx + + ) {
2016-12-13 13:36:23 -08:00
const CTxOut & txOut = tx . vout [ idx ] ;
CRecipient recipient = { txOut . scriptPubKey , txOut . nValue , setSubtractFeeFromOutputs . count ( idx ) = = 1 } ;
2015-04-24 18:29:00 -07:00
vecSend . push_back ( recipient ) ;
}
coinControl . fAllowOtherInputs = true ;
2016-04-28 22:04:07 +02:00
2017-12-14 03:17:58 +00:00
for ( const CTxIn & txin : tx . vin ) {
2015-04-24 18:29:00 -07:00
coinControl . Select ( txin . prevout ) ;
2017-12-14 03:17:58 +00:00
}
2015-04-24 18:29:00 -07:00
2017-12-14 03:18:29 +00:00
// Acquire the locks to prevent races to the new locked unspents between the
// CreateTransaction call and LockCoin calls (when lockUnspents is true).
2017-07-26 10:23:01 -04:00
auto locked_chain = chain ( ) . lock ( ) ;
LOCK ( cs_wallet ) ;
2015-04-24 18:29:00 -07:00
2017-02-02 15:30:03 -05:00
CTransactionRef tx_new ;
2019-07-10 11:37:45 -04:00
if ( ! CreateTransaction ( * locked_chain , vecSend , tx_new , nFeeRet , nChangePosInOut , strFailReason , coinControl , false ) ) {
2015-04-24 18:29:00 -07:00
return false ;
2017-06-28 16:41:55 -04:00
}
2017-07-10 14:29:06 -04:00
if ( nChangePosInOut ! = - 1 ) {
2017-02-02 15:30:03 -05:00
tx . vout . insert ( tx . vout . begin ( ) + nChangePosInOut , tx_new - > vout [ nChangePosInOut ] ) ;
2017-07-10 14:29:06 -04:00
}
2015-04-24 18:29:00 -07:00
2017-12-14 03:17:58 +00:00
// Copy output sizes from new transaction; they may have had the fee
// subtracted from them.
for ( unsigned int idx = 0 ; idx < tx . vout . size ( ) ; idx + + ) {
2017-02-02 15:30:03 -05:00
tx . vout [ idx ] . nValue = tx_new - > vout [ idx ] . nValue ;
2017-12-14 03:17:58 +00:00
}
2016-12-13 13:36:23 -08:00
2017-12-14 03:17:58 +00:00
// Add new txins while keeping original txin scriptSig/order.
2017-02-02 15:30:03 -05:00
for ( const CTxIn & txin : tx_new - > vin ) {
2017-12-14 03:17:58 +00:00
if ( ! coinControl . IsSelected ( txin . prevout ) ) {
2015-04-24 18:29:00 -07:00
tx . vin . push_back ( txin ) ;
2016-04-06 15:56:14 +01:00
2017-12-14 03:17:58 +00:00
if ( lockUnspents ) {
LockCoin ( txin . prevout ) ;
2016-04-06 15:56:14 +01:00
}
}
2015-04-24 18:29:00 -07:00
}
return true ;
}
2020-01-22 17:11:02 -05:00
static bool IsCurrentForAntiFeeSniping ( interfaces : : Chain & chain , const uint256 & block_hash )
2018-12-23 19:08:57 +01:00
{
2019-03-06 16:47:57 -05:00
if ( chain . isInitialBlockDownload ( ) ) {
2018-12-23 19:08:57 +01:00
return false ;
}
constexpr int64_t MAX_ANTI_FEE_SNIPING_TIP_AGE = 8 * 60 * 60 ; // in seconds
2020-01-22 17:11:02 -05:00
int64_t block_time ;
CHECK_NONFATAL ( chain . findBlock ( block_hash , FoundBlock ( ) . time ( block_time ) ) ) ;
if ( block_time < ( GetTime ( ) - MAX_ANTI_FEE_SNIPING_TIP_AGE ) ) {
2018-12-23 19:08:57 +01:00
return false ;
}
return true ;
}
/**
* Return a height - based locktime for new transactions ( uses the height of the
* current chain tip unless we are not synced with the current chain
*/
2020-01-22 17:11:02 -05:00
static uint32_t GetLocktimeForNewTransaction ( interfaces : : Chain & chain , const uint256 & block_hash , int block_height )
2018-12-23 19:08:57 +01:00
{
uint32_t locktime ;
// Discourage fee sniping.
//
// For a large miner the value of the transactions in the best block and
// the mempool can exceed the cost of deliberately attempting to mine two
// blocks to orphan the current best block. By setting nLockTime such that
// only the next block can include the transaction, we discourage this
// practice as the height restricted and limited blocksize gives miners
// considering fee sniping fewer options for pulling off this attack.
//
// A simple way to think about this is from the wallet's point of view we
// always want the blockchain to move forward. By setting nLockTime this
// way we're basically making the statement that we only want this
// transaction to appear in the next block; we don't want to potentially
// encourage reorgs by allowing transactions to appear at lower heights
// than the next block in forks of the best chain.
//
// Of course, the subsidy is high enough, and transaction volume low
// enough, that fee sniping isn't a problem yet, but by implementing a fix
// now we ensure code won't be written that makes assumptions about
// nLockTime that preclude a fix later.
2020-01-22 17:11:02 -05:00
if ( IsCurrentForAntiFeeSniping ( chain , block_hash ) ) {
locktime = block_height ;
2018-12-23 19:08:57 +01:00
// Secondly occasionally randomly pick a nLockTime even further back, so
// that transactions that are delayed after signing for whatever reason,
// e.g. high-latency mix networks and some CoinJoin implementations, have
// better privacy.
if ( GetRandInt ( 10 ) = = 0 )
locktime = std : : max ( 0 , ( int ) locktime - GetRandInt ( 100 ) ) ;
} else {
// If our chain is lagging behind, we can't discourage fee sniping nor help
// the privacy of high-latency transactions. To avoid leaking a potentially
// unique "nLockTime fingerprint", set nLockTime to a constant.
locktime = 0 ;
}
assert ( locktime < LOCKTIME_THRESHOLD ) ;
return locktime ;
}
2018-01-15 16:10:13 +00:00
OutputType CWallet : : TransactionChangeType ( OutputType change_type , const std : : vector < CRecipient > & vecSend )
2018-01-23 17:56:15 +01:00
{
// If -changetype is specified, always use that change type.
2018-03-19 15:57:11 -04:00
if ( change_type ! = OutputType : : CHANGE_AUTO ) {
2018-01-15 16:10:13 +00:00
return change_type ;
2018-01-23 17:56:15 +01:00
}
2018-02-10 21:06:35 -05:00
// if m_default_address_type is legacy, use legacy address as change (even
2018-01-23 17:56:15 +01:00
// if some of the outputs are P2WPKH or P2WSH).
2018-02-10 21:06:35 -05:00
if ( m_default_address_type = = OutputType : : LEGACY ) {
return OutputType : : LEGACY ;
2018-01-23 17:56:15 +01:00
}
// if any destination is P2WPKH or P2WSH, use P2WPKH for the change
// output.
for ( const auto & recipient : vecSend ) {
// Check if any destination contains a witness program:
int witnessversion = 0 ;
std : : vector < unsigned char > witnessprogram ;
if ( recipient . scriptPubKey . IsWitnessProgram ( witnessversion , witnessprogram ) ) {
2018-02-10 21:06:35 -05:00
return OutputType : : BECH32 ;
2018-01-23 17:56:15 +01:00
}
}
2018-02-10 21:06:35 -05:00
// else use m_default_address_type for change
return m_default_address_type ;
2018-01-23 17:56:15 +01:00
}
2019-07-10 11:37:45 -04:00
bool CWallet : : CreateTransaction ( interfaces : : Chain : : Lock & locked_chain , const std : : vector < CRecipient > & vecSend , CTransactionRef & tx , CAmount & nFeeRet ,
2018-07-19 11:45:26 +09:00
int & nChangePosInOut , std : : string & strFailReason , const CCoinControl & coin_control , bool sign )
2011-06-26 19:23:24 +02:00
{
2014-04-22 15:46:19 -07:00
CAmount nValue = 0 ;
2019-10-24 09:44:20 +01:00
const OutputType change_type = TransactionChangeType ( coin_control . m_change_type ? * coin_control . m_change_type : m_default_change_type , vecSend ) ;
ReserveDestination reservedest ( this , change_type ) ;
2016-03-30 02:04:22 +01:00
int nChangePosRequest = nChangePosInOut ;
2014-07-23 14:34:36 +02:00
unsigned int nSubtractFeeFromAmount = 0 ;
2017-01-04 08:51:14 +00:00
for ( const auto & recipient : vecSend )
2011-06-26 19:23:24 +02:00
{
2014-07-23 14:34:36 +02:00
if ( nValue < 0 | | recipient . nAmount < 0 )
2013-04-25 17:31:22 -04:00
{
2019-06-28 21:09:58 +03:00
strFailReason = _ ( " Transaction amounts must not be negative " ) . translated ;
2011-06-26 19:23:24 +02:00
return false ;
2013-04-25 17:31:22 -04:00
}
2014-07-23 14:34:36 +02:00
nValue + = recipient . nAmount ;
if ( recipient . fSubtractFeeFromAmount )
nSubtractFeeFromAmount + + ;
2011-06-26 19:23:24 +02:00
}
2016-11-22 12:48:01 -05:00
if ( vecSend . empty ( ) )
2013-04-25 17:31:22 -04:00
{
2019-06-28 21:09:58 +03:00
strFailReason = _ ( " Transaction must have at least one recipient " ) . translated ;
2011-06-26 19:23:24 +02:00
return false ;
2013-04-25 17:31:22 -04:00
}
2011-06-26 19:23:24 +02:00
2014-06-07 13:53:27 +02:00
CMutableTransaction txNew ;
2017-04-25 15:39:32 -04:00
FeeCalculation feeCalc ;
2017-08-25 14:49:44 -05:00
CAmount nFeeNeeded ;
2018-03-05 16:37:24 -05:00
int nBytes ;
2011-06-26 19:23:24 +02:00
{
2017-04-07 09:38:33 +00:00
std : : set < CInputCoin > setCoins ;
2017-07-26 10:23:01 -04:00
auto locked_chain = chain ( ) . lock ( ) ;
LOCK ( cs_wallet ) ;
2020-01-22 17:11:02 -05:00
txNew . nLockTime = GetLocktimeForNewTransaction ( chain ( ) , GetLastBlockHash ( ) , GetLastBlockHeight ( ) ) ;
2011-06-26 19:23:24 +02:00
{
2016-03-08 00:15:17 +00:00
std : : vector < COutput > vAvailableCoins ;
2019-05-09 18:34:38 -07:00
AvailableCoins ( * locked_chain , vAvailableCoins , true , & coin_control , 1 , MAX_MONEY , MAX_MONEY , 0 ) ;
2018-03-05 16:42:49 -05:00
CoinSelectionParams coin_selection_params ; // Parameters for coin selection, init with dummy
2016-03-08 00:15:17 +00:00
2017-06-30 12:53:29 -04:00
// Create change script that will be used if we need change
2019-06-18 15:48:20 -04:00
// TODO: pass in scriptChange instead of reservedest so
2017-06-30 12:53:29 -04:00
// change transaction isn't always pay-to-bitcoin-address
CScript scriptChange ;
// coin control: send change to custom address
2017-06-28 16:41:55 -04:00
if ( ! boost : : get < CNoDestination > ( & coin_control . destChange ) ) {
scriptChange = GetScriptForDestination ( coin_control . destChange ) ;
} else { // no coin control: send change to newly generated address
2017-06-30 12:53:29 -04:00
// Note: We use a new key here to keep it from being obvious which side is the change.
// The drawback is that by not reusing a previous key, the change may be lost if a
// backup is restored, if the backup doesn't have the new private key for the change.
// If we reused the old key, it would be possible to add code to look for and
// rediscover unknown transactions that were written with keys of ours to recover
// post-backup change.
2019-10-23 15:21:50 +02:00
// Reserve a new key pair from key pool. If it fails, provide a dummy
// destination in case we don't need change.
2019-06-18 15:48:20 -04:00
CTxDestination dest ;
2019-10-23 15:11:25 +02:00
if ( ! reservedest . GetReservedDestination ( dest , true ) ) {
2019-10-23 15:21:50 +02:00
strFailReason = _ ( " Transaction needs a change address, but we can't generate it. Please call keypoolrefill first. " ) . translated ;
2017-06-30 12:53:29 -04:00
}
2019-06-18 15:48:20 -04:00
scriptChange = GetScriptForDestination ( dest ) ;
2019-10-23 15:21:50 +02:00
assert ( ! dest . empty ( ) | | scriptChange . empty ( ) ) ;
2017-06-30 12:53:29 -04:00
}
2017-06-30 13:16:53 -04:00
CTxOut change_prototype_txout ( 0 , scriptChange ) ;
2018-06-22 18:27:18 +00:00
coin_selection_params . change_output_size = GetSerializeSize ( change_prototype_txout ) ;
2017-06-30 12:53:29 -04:00
2017-07-28 21:40:29 -04:00
CFeeRate discard_rate = GetDiscardRate ( * this ) ;
2018-03-05 16:42:49 -05:00
// Get the fee rate to use effective values in coin selection
2017-07-28 21:40:29 -04:00
CFeeRate nFeeRateNeeded = GetMinimumFeeRate ( * this , coin_control , & feeCalc ) ;
2018-03-05 16:42:49 -05:00
2014-11-02 00:14:47 +01:00
nFeeRet = 0 ;
2017-06-30 13:16:53 -04:00
bool pick_new_inputs = true ;
CAmount nValueIn = 0 ;
2018-03-05 16:42:49 -05:00
// BnB selector is the only selector used when this is true.
// That should only happen on the first pass through the loop.
2019-10-25 18:45:58 -04:00
coin_selection_params . use_bnb = true ;
coin_selection_params . m_subtract_fee_outputs = nSubtractFeeFromAmount ! = 0 ; // If we are doing subtract fee from recipient, don't use effective values
2015-09-14 14:49:59 +02:00
// Start with no fee and loop until there is enough fee
2013-07-31 14:06:44 +10:00
while ( true )
2011-06-26 19:23:24 +02:00
{
2016-03-30 02:04:22 +01:00
nChangePosInOut = nChangePosRequest ;
2014-06-07 13:53:27 +02:00
txNew . vin . clear ( ) ;
txNew . vout . clear ( ) ;
2014-07-23 14:34:36 +02:00
bool fFirst = true ;
2011-06-26 19:23:24 +02:00
2015-03-18 19:22:49 +01:00
CAmount nValueToSelect = nValue ;
2014-07-23 14:34:36 +02:00
if ( nSubtractFeeFromAmount = = 0 )
2015-03-18 19:22:49 +01:00
nValueToSelect + = nFeeRet ;
2018-03-05 16:42:49 -05:00
2011-06-26 19:23:24 +02:00
// vouts to the payees
2019-10-25 18:45:58 -04:00
if ( ! coin_selection_params . m_subtract_fee_outputs ) {
coin_selection_params . tx_noinputs_size = 11 ; // Static vsize overhead + outputs vsize. 4 nVersion, 4 nLocktime, 1 input count, 1 output count, 1 witness overhead (dummy, flag, stack size)
}
2017-01-04 08:51:14 +00:00
for ( const auto & recipient : vecSend )
2013-04-24 18:27:00 -04:00
{
2014-07-23 14:34:36 +02:00
CTxOut txout ( recipient . nAmount , recipient . scriptPubKey ) ;
if ( recipient . fSubtractFeeFromAmount )
{
2017-08-28 09:20:50 +02:00
assert ( nSubtractFeeFromAmount ! = 0 ) ;
2014-07-23 14:34:36 +02:00
txout . nValue - = nFeeRet / nSubtractFeeFromAmount ; // Subtract fee equally from each selected recipient
if ( fFirst ) // first receiver pays the remainder not divisible by output count
{
fFirst = false ;
txout . nValue - = nFeeRet % nSubtractFeeFromAmount ;
}
}
2018-03-05 16:42:49 -05:00
// Include the fee cost for outputs. Note this is only used for BnB right now
2019-10-25 18:45:58 -04:00
if ( ! coin_selection_params . m_subtract_fee_outputs ) {
coin_selection_params . tx_noinputs_size + = : : GetSerializeSize ( txout , PROTOCOL_VERSION ) ;
}
2014-07-23 14:34:36 +02:00
2019-03-06 16:47:57 -05:00
if ( IsDust ( txout , chain ( ) . relayDustFee ( ) ) )
2013-04-25 17:31:22 -04:00
{
2014-07-23 14:34:36 +02:00
if ( recipient . fSubtractFeeFromAmount & & nFeeRet > 0 )
{
if ( txout . nValue < 0 )
2019-06-28 21:09:58 +03:00
strFailReason = _ ( " The transaction amount is too small to pay the fee " ) . translated ;
2014-07-23 14:34:36 +02:00
else
2019-06-28 21:09:58 +03:00
strFailReason = _ ( " The transaction amount is too small to send after the fee has been deducted " ) . translated ;
2014-07-23 14:34:36 +02:00
}
else
2019-06-28 21:09:58 +03:00
strFailReason = _ ( " Transaction amount too small " ) . translated ;
2013-04-24 18:27:00 -04:00
return false ;
2013-04-25 17:31:22 -04:00
}
2014-06-07 13:53:27 +02:00
txNew . vout . push_back ( txout ) ;
2013-04-24 18:27:00 -04:00
}
2011-06-26 19:23:24 +02:00
// Choose coins to use
2019-11-25 02:19:57 -05:00
bool bnb_used = false ;
2017-06-30 13:16:53 -04:00
if ( pick_new_inputs ) {
nValueIn = 0 ;
setCoins . clear ( ) ;
2018-10-03 14:41:03 +09:00
int change_spend_size = CalculateMaximumSignedInputSize ( change_prototype_txout , this ) ;
// If the wallet doesn't know how to sign change output, assume p2sh-p2wpkh
// as lower-bound to allow BnB to do it's thing
if ( change_spend_size = = - 1 ) {
coin_selection_params . change_spend_size = DUMMY_NESTED_P2WPKH_INPUT_SIZE ;
} else {
coin_selection_params . change_spend_size = ( size_t ) change_spend_size ;
}
2018-03-05 16:42:49 -05:00
coin_selection_params . effective_fee = nFeeRateNeeded ;
if ( ! SelectCoins ( vAvailableCoins , nValueToSelect , setCoins , nValueIn , coin_control , coin_selection_params , bnb_used ) )
2017-06-30 13:16:53 -04:00
{
2018-03-05 16:42:49 -05:00
// If BnB was used, it was the first pass. No longer the first pass and continue loop with knapsack.
if ( bnb_used ) {
coin_selection_params . use_bnb = false ;
continue ;
}
else {
2019-06-28 21:09:58 +03:00
strFailReason = _ ( " Insufficient funds " ) . translated ;
2018-03-05 16:42:49 -05:00
return false ;
}
2017-06-30 13:16:53 -04:00
}
2018-06-27 18:06:44 +02:00
} else {
bnb_used = false ;
2013-04-25 17:31:22 -04:00
}
2011-06-26 19:23:24 +02:00
2015-03-18 19:22:49 +01:00
const CAmount nChange = nValueIn - nValueToSelect ;
2011-07-24 16:37:09 +02:00
if ( nChange > 0 )
2011-06-26 19:23:24 +02:00
{
2011-09-28 12:30:06 -04:00
// Fill a vout to ourself
2013-04-24 18:27:00 -04:00
CTxOut newTxOut ( nChange , scriptChange ) ;
// Never create dust outputs; if we would, just
// add the dust to the fee.
2018-03-05 16:42:49 -05:00
// The nChange when BnB is used is always going to go to fees.
if ( IsDust ( newTxOut , discard_rate ) | | bnb_used )
2013-04-24 18:27:00 -04:00
{
2016-03-30 02:04:22 +01:00
nChangePosInOut = - 1 ;
2013-04-24 18:27:00 -04:00
nFeeRet + = nChange ;
}
else
{
2016-03-30 02:04:22 +01:00
if ( nChangePosInOut = = - 1 )
{
// Insert change txn at random position:
nChangePosInOut = GetRandInt ( txNew . vout . size ( ) + 1 ) ;
}
2016-06-08 15:34:18 +02:00
else if ( ( unsigned int ) nChangePosInOut > txNew . vout . size ( ) )
2016-03-30 02:04:22 +01:00
{
2019-06-28 21:09:58 +03:00
strFailReason = _ ( " Change index out of range " ) . translated ;
2016-03-30 02:04:22 +01:00
return false ;
}
2017-01-27 10:33:45 +09:00
std : : vector < CTxOut > : : iterator position = txNew . vout . begin ( ) + nChangePosInOut ;
2014-06-07 13:53:27 +02:00
txNew . vout . insert ( position , newTxOut ) ;
2013-04-24 18:27:00 -04:00
}
2017-04-28 17:22:37 -04:00
} else {
nChangePosInOut = - 1 ;
}
2011-06-26 19:23:24 +02:00
2018-03-15 16:18:48 -04:00
// Dummy fill vin for maximum size estimation
2013-08-25 14:13:25 -04:00
//
2018-03-15 16:18:48 -04:00
for ( const auto & coin : setCoins ) {
txNew . vin . push_back ( CTxIn ( coin . outpoint , CScript ( ) ) ) ;
}
2011-06-26 19:23:24 +02:00
2018-12-09 22:03:07 -08:00
nBytes = CalculateMaximumSignedTxSize ( CTransaction ( txNew ) , this , coin_control . fAllowWatchOnly ) ;
2018-03-05 16:37:24 -05:00
if ( nBytes < 0 ) {
2019-06-28 21:09:58 +03:00
strFailReason = _ ( " Signing transaction failed " ) . translated ;
2017-01-26 12:57:04 -05:00
return false ;
2015-04-24 18:29:00 -07:00
}
2017-07-28 21:40:29 -04:00
nFeeNeeded = GetMinimumFee ( * this , nBytes , coin_control , & feeCalc ) ;
2018-04-07 12:12:46 -04:00
if ( feeCalc . reason = = FeeReason : : FALLBACK & & ! m_allow_fallback_fee ) {
2017-12-12 10:32:50 -10:00
// eventually allow a fallback fee
2019-06-28 21:09:58 +03:00
strFailReason = _ ( " Fee estimation failed. Fallbackfee is disabled. Wait a few blocks or enable -fallbackfee. " ) . translated ;
2017-12-12 10:32:50 -10:00
return false ;
}
2014-05-27 15:44:57 -04:00
2017-01-05 09:12:48 -05:00
if ( nFeeRet > = nFeeNeeded ) {
2017-06-30 13:16:53 -04:00
// Reduce fee to only the needed amount if possible. This
// prevents potential overpayment in fees if the coins
// selected to meet nFeeNeeded result in a transaction that
// requires less fee than the prior iteration.
// If we have no change and a big enough excess fee, then
// try to construct transaction again only without picking
// new inputs. We now know we only need the smaller fee
// (because of reduced tx size) and so we should add a
// change output. Only try this once.
2017-08-25 14:46:07 -05:00
if ( nChangePosInOut = = - 1 & & nSubtractFeeFromAmount = = 0 & & pick_new_inputs ) {
2018-03-05 16:42:49 -05:00
unsigned int tx_size_with_change = nBytes + coin_selection_params . change_output_size + 2 ; // Add 2 as a buffer in case increasing # of outputs changes compact size
2017-07-28 21:40:29 -04:00
CAmount fee_needed_with_change = GetMinimumFee ( * this , tx_size_with_change , coin_control , nullptr ) ;
2017-08-25 14:46:07 -05:00
CAmount minimum_value_for_change = GetDustThreshold ( change_prototype_txout , discard_rate ) ;
if ( nFeeRet > = fee_needed_with_change + minimum_value_for_change ) {
pick_new_inputs = false ;
nFeeRet = fee_needed_with_change ;
continue ;
}
2017-06-30 13:16:53 -04:00
}
// If we have change output already, just increase it
2017-01-05 09:12:48 -05:00
if ( nFeeRet > nFeeNeeded & & nChangePosInOut ! = - 1 & & nSubtractFeeFromAmount = = 0 ) {
CAmount extraFeePaid = nFeeRet - nFeeNeeded ;
2017-01-27 10:33:45 +09:00
std : : vector < CTxOut > : : iterator change_position = txNew . vout . begin ( ) + nChangePosInOut ;
2017-01-05 09:12:48 -05:00
change_position - > nValue + = extraFeePaid ;
nFeeRet - = extraFeePaid ;
}
2014-12-16 01:43:40 -08:00
break ; // Done, enough fee included.
2017-01-05 09:12:48 -05:00
}
2017-06-30 13:16:53 -04:00
else if ( ! pick_new_inputs ) {
// This shouldn't happen, we should have had enough excess
// fee to pay for the new output and still meet nFeeNeeded
2017-07-27 17:01:39 -04:00
// Or we should have just subtracted fee from recipients and
// nFeeNeeded should not have changed
2019-06-28 21:09:58 +03:00
strFailReason = _ ( " Transaction fee and change calculation failed " ) . translated ;
2017-06-30 13:16:53 -04:00
return false ;
}
2014-05-27 15:44:57 -04:00
2017-01-05 09:10:08 -05:00
// Try to reduce change to include necessary fee
if ( nChangePosInOut ! = - 1 & & nSubtractFeeFromAmount = = 0 ) {
CAmount additionalFeeNeeded = nFeeNeeded - nFeeRet ;
2017-01-27 10:33:45 +09:00
std : : vector < CTxOut > : : iterator change_position = txNew . vout . begin ( ) + nChangePosInOut ;
2017-01-05 09:10:08 -05:00
// Only reduce change if remaining amount is still a large enough output.
if ( change_position - > nValue > = MIN_FINAL_CHANGE + additionalFeeNeeded ) {
change_position - > nValue - = additionalFeeNeeded ;
nFeeRet + = additionalFeeNeeded ;
break ; // Done, able to increase fee from change
}
}
2017-07-27 17:01:39 -04:00
// If subtracting fee from recipients, we now know what fee we
// need to subtract, we have no reason to reselect inputs
if ( nSubtractFeeFromAmount > 0 ) {
pick_new_inputs = false ;
}
2014-05-27 15:44:57 -04:00
// Include more fee and try again.
nFeeRet = nFeeNeeded ;
2018-03-05 16:42:49 -05:00
coin_selection_params . use_bnb = false ;
2014-05-27 15:44:57 -04:00
continue ;
2011-06-26 19:23:24 +02:00
}
2019-10-23 15:21:50 +02:00
// Give up if change keypool ran out and we failed to find a solution without change:
if ( scriptChange . empty ( ) & & nChangePosInOut ! = - 1 ) {
return false ;
}
2011-06-26 19:23:24 +02:00
}
2017-01-04 08:51:14 +00:00
2018-03-15 16:18:48 -04:00
// Shuffle selected coins and fill in final vin
txNew . vin . clear ( ) ;
std : : vector < CInputCoin > selected_coins ( setCoins . begin ( ) , setCoins . end ( ) ) ;
2018-10-31 15:42:03 -07:00
Shuffle ( selected_coins . begin ( ) , selected_coins . end ( ) , FastRandomContext ( ) ) ;
2018-03-15 16:18:48 -04:00
// Note how the sequence number is set to non-maxint so that
// the nLockTime set above actually works.
//
// BIP125 defines opt-in RBF as any nSequence < maxint-1, so
// we use the highest possible value in that range (maxint-2)
// to avoid conflicting with other possible uses of nSequence,
// and in the spirit of "smallest possible change from prior
// behavior."
2018-04-07 12:12:46 -04:00
const uint32_t nSequence = coin_control . m_signal_bip125_rbf . get_value_or ( m_signal_rbf ) ? MAX_BIP125_RBF_SEQUENCE : ( CTxIn : : SEQUENCE_FINAL - 1 ) ;
2018-03-15 16:18:48 -04:00
for ( const auto & coin : selected_coins ) {
txNew . vin . push_back ( CTxIn ( coin . outpoint , CScript ( ) , nSequence ) ) ;
}
2020-02-10 19:50:52 -05:00
if ( sign & & ! SignTransaction ( txNew ) ) {
strFailReason = _ ( " Signing transaction failed " ) . translated ;
return false ;
2017-01-04 08:51:14 +00:00
}
2017-02-02 15:30:03 -05:00
// Return the constructed transaction data.
tx = MakeTransactionRef ( std : : move ( txNew ) ) ;
2017-01-04 08:51:14 +00:00
// Limit size
2018-04-27 01:24:48 +08:00
if ( GetTransactionWeight ( * tx ) > MAX_STANDARD_TX_WEIGHT )
2017-01-04 08:51:14 +00:00
{
2019-06-28 21:09:58 +03:00
strFailReason = _ ( " Transaction too large " ) . translated ;
2017-01-04 08:51:14 +00:00
return false ;
}
2011-06-26 19:23:24 +02:00
}
2019-07-02 15:16:36 +01:00
if ( nFeeRet > m_default_max_tx_fee ) {
strFailReason = TransactionErrorString ( TransactionError : : MAX_FEE_EXCEEDED ) ;
return false ;
}
2017-08-01 21:17:40 +02:00
if ( gArgs . GetBoolArg ( " -walletrejectlongchains " , DEFAULT_WALLET_REJECT_LONG_CHAINS ) ) {
2016-12-02 15:45:43 -05:00
// Lastly, ensure this tx will pass the mempool's chain limits
2017-07-28 19:45:45 -04:00
if ( ! chain ( ) . checkChainLimits ( tx ) ) {
2019-06-28 21:09:58 +03:00
strFailReason = _ ( " Transaction has too long of a mempool chain " ) . translated ;
2016-12-02 15:45:43 -05:00
return false ;
}
}
2017-04-25 15:39:32 -04:00
2019-06-13 14:12:01 -04:00
// Before we return success, we assume any change key will be used to prevent
// accidental re-use.
reservedest . KeepDestination ( ) ;
2018-06-15 19:02:52 -04:00
WalletLogPrintf ( " Fee Calculation: Fee:%d Bytes:%u Needed:%d Tgt:%d (requested %d) Reason: \" %s \" Decay %.5f: Estimation: (%g - %g) %.2f%% %.1f/(%.1f %d mem %.1f out) Fail: (%g - %g) %.2f%% %.1f/(%.1f %d mem %.1f out) \n " ,
2017-08-25 14:49:44 -05:00
nFeeRet , nBytes , nFeeNeeded , feeCalc . returnedTarget , feeCalc . desiredTarget , StringForFeeReason ( feeCalc . reason ) , feeCalc . est . decay ,
2017-04-25 15:39:32 -04:00
feeCalc . est . pass . start , feeCalc . est . pass . end ,
100 * feeCalc . est . pass . withinTarget / ( feeCalc . est . pass . totalConfirmed + feeCalc . est . pass . inMempool + feeCalc . est . pass . leftMempool ) ,
feeCalc . est . pass . withinTarget , feeCalc . est . pass . totalConfirmed , feeCalc . est . pass . inMempool , feeCalc . est . pass . leftMempool ,
feeCalc . est . fail . start , feeCalc . est . fail . end ,
100 * feeCalc . est . fail . withinTarget / ( feeCalc . est . fail . totalConfirmed + feeCalc . est . fail . inMempool + feeCalc . est . fail . leftMempool ) ,
feeCalc . est . fail . withinTarget , feeCalc . est . fail . totalConfirmed , feeCalc . est . fail . inMempool , feeCalc . est . fail . leftMempool ) ;
2014-07-23 14:34:36 +02:00
return true ;
2011-06-26 19:23:24 +02:00
}
2019-10-18 09:37:40 -04:00
void CWallet : : CommitTransaction ( CTransactionRef tx , mapValue_t mapValue , std : : vector < std : : pair < std : : string , std : : string > > orderForm )
2011-06-26 19:23:24 +02:00
{
2019-04-03 17:29:21 -04:00
auto locked_chain = chain ( ) . lock ( ) ;
LOCK ( cs_wallet ) ;
2017-02-02 15:30:03 -05:00
2019-04-03 17:29:21 -04:00
CWalletTx wtxNew ( this , std : : move ( tx ) ) ;
wtxNew . mapValue = std : : move ( mapValue ) ;
wtxNew . vOrderForm = std : : move ( orderForm ) ;
wtxNew . fTimeReceivedIsTxTime = true ;
wtxNew . fFromMe = true ;
2017-02-02 15:30:03 -05:00
2019-04-03 17:29:21 -04:00
WalletLogPrintf ( " CommitTransaction: \n %s " , wtxNew . tx - > ToString ( ) ) ; /* Continued */
2011-06-26 19:23:24 +02:00
2019-04-03 17:29:21 -04:00
// Add tx to wallet, because if it has change it's also ours,
// otherwise just for transaction history.
AddToWallet ( wtxNew ) ;
2011-06-26 19:23:24 +02:00
2019-04-03 17:29:21 -04:00
// Notify that old coins are spent
for ( const CTxIn & txin : wtxNew . tx - > vin ) {
CWalletTx & coin = mapWallet . at ( txin . prevout . hash ) ;
coin . BindWallet ( this ) ;
NotifyTransactionChanged ( this , coin . GetHash ( ) , CT_UPDATED ) ;
}
2011-06-26 19:23:24 +02:00
2019-04-03 17:29:21 -04:00
// Get the inserted-CWalletTx from mapWallet so that the
// fInMempool flag is cached properly
CWalletTx & wtx = mapWallet . at ( wtxNew . GetHash ( ) ) ;
Use callbacks to cache whether wallet transactions are in mempool
This avoid calling out to mempool state during coin selection,
balance calculation, etc. In the next commit we ensure all wallet
callbacks from CValidationInterface happen in the same queue,
serialized with each other. This helps to avoid re-introducing one
of the issues described in #9584 [1] by further disconnecting
wallet from current chain/mempool state.
Thanks to @morcos for the suggestion to do this.
Note that there are several race conditions introduced here:
* If a user calls sendrawtransaction from RPC, adding a
transaction which is "trusted" (ie from them) and pays them
change, it may not be immediately used by coin selection until
the notification callbacks finish running. No such race is
introduced in normal transaction-sending RPCs as this case is
explicitly handled.
* Until Block{Connected,Disconnected} and
TransactionAddedToMempool calls also run in the CSceduler
background thread, there is a race where
TransactionAddedToMempool might be called after a
Block{Connected,Disconnected} call happens.
* Wallet will write a new best chain from the SetBestChain
callback prior to having processed the transaction from that
block.
[1] "you could go to select coins, need to use 0-conf change, but
such 0-conf change may have been included in a block who's
callbacks have not yet been processed - resulting in thinking they
are not in mempool and, thus, not selectable."
2017-01-20 16:38:07 -05:00
2019-04-03 17:55:24 -04:00
if ( ! fBroadcastTransactions ) {
// Don't submit tx to the mempool
return ;
2011-06-26 19:23:24 +02:00
}
2019-04-03 17:29:21 -04:00
2019-04-03 17:55:24 -04:00
std : : string err_string ;
2019-04-29 09:52:01 -04:00
if ( ! wtx . SubmitMemoryPoolAndRelay ( err_string , true ) ) {
2019-04-03 17:55:24 -04:00
WalletLogPrintf ( " CommitTransaction(): Transaction cannot be broadcast immediately, %s \n " , err_string ) ;
// TODO: if we expect the failure to be long term or permanent, instead delete wtx from the wallet and return failure.
2011-06-26 19:23:24 +02:00
}
}
2012-09-18 14:30:47 -04:00
DBErrors CWallet : : LoadWallet ( bool & fFirstRunRet )
2011-06-26 19:23:24 +02:00
{
2019-08-22 13:16:40 -04:00
// Even if we don't use this lock in this function, we want to preserve
// lock order in LoadToWallet if query of chain state is needed to know
2019-12-01 21:30:29 -05:00
// tx status. If lock can't be taken (e.g bitcoin-wallet), tx confirmation
2019-08-22 13:16:40 -04:00
// status may be not reliable.
auto locked_chain = LockChain ( ) ;
2017-07-26 10:23:01 -04:00
LOCK ( cs_wallet ) ;
2017-08-24 14:12:21 -04:00
2011-06-26 19:23:24 +02:00
fFirstRunRet = false ;
2017-12-08 06:39:22 -05:00
DBErrors nLoadWalletRet = WalletBatch ( * database , " cr+ " ) . LoadWallet ( this ) ;
2018-03-09 15:03:40 +01:00
if ( nLoadWalletRet = = DBErrors : : NEED_REWRITE )
2011-11-10 21:29:23 +01:00
{
2017-12-08 06:39:22 -05:00
if ( database - > Rewrite ( " \x04 pool " ) )
2011-11-10 21:12:46 -05:00
{
2019-10-07 14:11:34 -04:00
for ( const auto & spk_man_pair : m_spk_managers ) {
spk_man_pair . second - > RewriteDB ( ) ;
2019-10-07 14:11:34 -04:00
}
2011-11-10 21:12:46 -05:00
}
2011-11-10 21:29:23 +01:00
}
2019-10-07 14:11:34 -04:00
// This wallet is in its first run if there are no ScriptPubKeyMans and it isn't blank or no privkeys
2019-10-07 14:11:34 -04:00
fFirstRunRet = m_spk_managers . empty ( ) & & ! IsWalletFlagSet ( WALLET_FLAG_DISABLE_PRIVATE_KEYS ) & & ! IsWalletFlagSet ( WALLET_FLAG_BLANK_WALLET ) ;
if ( fFirstRunRet ) {
assert ( m_external_spk_managers . empty ( ) ) ;
assert ( m_internal_spk_managers . empty ( ) ) ;
2018-04-29 20:15:05 +02:00
}
2017-07-28 17:00:49 -07:00
2018-03-09 15:03:40 +01:00
if ( nLoadWalletRet ! = DBErrors : : LOAD_OK )
2011-07-05 03:06:19 +02:00
return nLoadWalletRet ;
2011-06-26 19:23:24 +02:00
2018-03-09 15:03:40 +01:00
return DBErrors : : LOAD_OK ;
2011-06-26 19:23:24 +02:00
}
2017-01-27 10:33:45 +09:00
DBErrors CWallet : : ZapSelectTx ( std : : vector < uint256 > & vHashIn , std : : vector < uint256 > & vHashOut )
2016-03-07 08:51:06 -05:00
{
2019-02-18 16:09:45 -05:00
AssertLockHeld ( cs_wallet ) ;
DBErrors nZapSelectTxRet = WalletBatch ( * database , " cr+ " ) . ZapSelectTx ( vHashIn , vHashOut ) ;
2018-06-11 14:09:16 -04:00
for ( uint256 hash : vHashOut ) {
const auto & it = mapWallet . find ( hash ) ;
wtxOrdered . erase ( it - > second . m_it_wtxOrdered ) ;
mapWallet . erase ( it ) ;
2019-09-24 11:15:15 +02:00
NotifyTransactionChanged ( this , hash , CT_DELETED ) ;
2018-06-11 14:09:16 -04:00
}
2016-11-12 10:53:18 +01:00
2018-03-09 15:03:40 +01:00
if ( nZapSelectTxRet = = DBErrors : : NEED_REWRITE )
2016-03-07 08:51:06 -05:00
{
2017-12-08 06:39:22 -05:00
if ( database - > Rewrite ( " \x04 pool " ) )
2016-03-07 08:51:06 -05:00
{
2019-10-07 14:11:34 -04:00
for ( const auto & spk_man_pair : m_spk_managers ) {
spk_man_pair . second - > RewriteDB ( ) ;
2019-10-07 14:11:34 -04:00
}
2016-03-07 08:51:06 -05:00
}
}
2018-03-09 15:03:40 +01:00
if ( nZapSelectTxRet ! = DBErrors : : LOAD_OK )
2016-03-07 08:51:06 -05:00
return nZapSelectTxRet ;
MarkDirty ( ) ;
2018-03-09 15:03:40 +01:00
return DBErrors : : LOAD_OK ;
2016-03-07 08:51:06 -05:00
}
2011-07-07 15:22:54 +02:00
2014-02-14 18:27:15 +01:00
DBErrors CWallet : : ZapWalletTx ( std : : vector < CWalletTx > & vWtx )
2014-02-14 11:33:07 -05:00
{
2017-12-08 06:39:22 -05:00
DBErrors nZapWalletTxRet = WalletBatch ( * database , " cr+ " ) . ZapWalletTx ( vWtx ) ;
2018-03-09 15:03:40 +01:00
if ( nZapWalletTxRet = = DBErrors : : NEED_REWRITE )
2014-02-14 11:33:07 -05:00
{
2017-12-08 06:39:22 -05:00
if ( database - > Rewrite ( " \x04 pool " ) )
2014-02-14 11:33:07 -05:00
{
2019-10-07 14:11:34 -04:00
for ( const auto & spk_man_pair : m_spk_managers ) {
spk_man_pair . second - > RewriteDB ( ) ;
2019-10-07 14:11:34 -04:00
}
2014-02-14 11:33:07 -05:00
}
}
2018-03-09 15:03:40 +01:00
if ( nZapWalletTxRet ! = DBErrors : : LOAD_OK )
2014-02-14 11:33:07 -05:00
return nZapWalletTxRet ;
2018-03-09 15:03:40 +01:00
return DBErrors : : LOAD_OK ;
2014-02-14 11:33:07 -05:00
}
2019-04-04 02:33:02 -07:00
bool CWallet : : SetAddressBookWithDB ( WalletBatch & batch , const CTxDestination & address , const std : : string & strName , const std : : string & strPurpose )
2011-07-07 15:22:54 +02:00
{
2014-02-18 12:11:46 -05:00
bool fUpdated = false ;
{
2019-02-07 14:46:08 -05:00
LOCK ( cs_wallet ) ;
2020-02-21 21:52:52 +00:00
std : : map < CTxDestination , CAddressBookData > : : iterator mi = m_address_book . find ( address ) ;
2020-02-22 04:16:36 +00:00
fUpdated = ( mi ! = m_address_book . end ( ) & & ! mi - > second . IsChange ( ) ) ;
2020-02-22 01:52:47 +00:00
m_address_book [ address ] . SetLabel ( strName ) ;
2014-02-18 12:11:46 -05:00
if ( ! strPurpose . empty ( ) ) /* update purpose only if requested */
2020-02-21 21:52:52 +00:00
m_address_book [ address ] . purpose = strPurpose ;
2014-02-18 12:11:46 -05:00
}
2019-10-07 14:11:34 -04:00
NotifyAddressBookChanged ( this , address , strName , IsMine ( address ) ! = ISMINE_NO ,
2014-02-18 12:11:46 -05:00
strPurpose , ( fUpdated ? CT_UPDATED : CT_NEW ) ) ;
2019-04-04 02:33:02 -07:00
if ( ! strPurpose . empty ( ) & & ! batch . WritePurpose ( EncodeDestination ( address ) , strPurpose ) )
2013-07-22 16:50:39 +10:00
return false ;
2019-04-04 02:33:02 -07:00
return batch . WriteName ( EncodeDestination ( address ) , strName ) ;
}
bool CWallet : : SetAddressBook ( const CTxDestination & address , const std : : string & strName , const std : : string & strPurpose )
{
WalletBatch batch ( * database ) ;
return SetAddressBookWithDB ( batch , address , strName , strPurpose ) ;
2011-07-07 15:22:54 +02:00
}
2013-07-22 16:50:39 +10:00
bool CWallet : : DelAddressBook ( const CTxDestination & address )
2011-07-07 15:22:54 +02:00
{
2020-02-22 05:02:18 +00:00
// If we want to delete receiving addresses, we need to take care that DestData "used" (and possibly newer DestData) gets preserved (and the "deleted" address transformed into a change entry instead of actually being deleted)
// NOTE: This isn't a problem for sending addresses because they never have any DestData yet!
// When adding new DestData, it should be considered here whether to retain or delete it (or move it?).
2020-04-06 19:56:06 +00:00
if ( IsMine ( address ) ) {
WalletLogPrintf ( " %s called with IsMine address, NOT SUPPORTED. Please report this bug! %s \n " , __func__ , PACKAGE_BUGREPORT ) ;
return false ;
}
2020-02-22 05:02:18 +00:00
2013-11-18 16:55:54 +01:00
{
2019-02-07 14:46:08 -05:00
LOCK ( cs_wallet ) ;
2014-02-18 12:11:46 -05:00
2017-03-08 12:08:26 +00:00
// Delete destdata tuples associated with address
2017-08-22 18:02:33 -07:00
std : : string strAddress = EncodeDestination ( address ) ;
2020-02-21 21:52:52 +00:00
for ( const std : : pair < const std : : string , std : : string > & item : m_address_book [ address ] . destdata )
2013-11-18 16:55:54 +01:00
{
2017-12-08 06:39:22 -05:00
WalletBatch ( * database ) . EraseDestData ( strAddress , item . first ) ;
2013-11-18 16:55:54 +01:00
}
2020-02-21 21:52:52 +00:00
m_address_book . erase ( address ) ;
2013-11-18 16:55:54 +01:00
}
2019-10-07 14:11:34 -04:00
NotifyAddressBookChanged ( this , address , " " , IsMine ( address ) ! = ISMINE_NO , " " , CT_DELETED ) ;
2014-02-18 12:11:46 -05:00
2017-12-08 06:39:22 -05:00
WalletBatch ( * database ) . ErasePurpose ( EncodeDestination ( address ) ) ;
return WalletBatch ( * database ) . EraseName ( EncodeDestination ( address ) ) ;
2011-07-07 15:22:54 +02:00
}
2020-03-02 16:33:45 +09:00
size_t CWallet : : KeypoolCountExternalKeys ( ) const
2019-10-07 14:11:34 -04:00
{
AssertLockHeld ( cs_wallet ) ;
unsigned int count = 0 ;
2019-10-07 14:11:34 -04:00
for ( auto spk_man : GetActiveScriptPubKeyMans ( ) ) {
2019-10-07 14:11:34 -04:00
count + = spk_man - > KeypoolCountExternalKeys ( ) ;
}
return count ;
}
2019-10-07 14:11:34 -04:00
unsigned int CWallet : : GetKeyPoolSize ( ) const
{
AssertLockHeld ( cs_wallet ) ;
unsigned int count = 0 ;
2019-10-07 14:11:34 -04:00
for ( auto spk_man : GetActiveScriptPubKeyMans ( ) ) {
2019-10-07 14:11:34 -04:00
count + = spk_man - > GetKeyPoolSize ( ) ;
}
return count ;
}
2019-10-07 14:11:34 -04:00
bool CWallet : : TopUpKeyPool ( unsigned int kpSize )
{
2019-10-07 14:11:34 -04:00
LOCK ( cs_wallet ) ;
2019-10-07 14:11:34 -04:00
bool res = true ;
2019-10-07 14:11:34 -04:00
for ( auto spk_man : GetActiveScriptPubKeyMans ( ) ) {
2019-10-07 14:11:34 -04:00
res & = spk_man - > TopUp ( kpSize ) ;
2019-10-07 14:11:34 -04:00
}
return res ;
}
bool CWallet : : GetNewDestination ( const OutputType type , const std : : string label , CTxDestination & dest , std : : string & error )
{
2019-10-07 14:11:34 -04:00
LOCK ( cs_wallet ) ;
2019-10-07 14:11:34 -04:00
error . clear ( ) ;
bool result = false ;
2019-10-07 14:11:34 -04:00
auto spk_man = GetScriptPubKeyMan ( type , false /* internal */ ) ;
2019-10-07 14:11:34 -04:00
if ( spk_man ) {
2019-11-20 12:42:10 -05:00
spk_man - > TopUp ( ) ;
2019-10-07 14:11:34 -04:00
result = spk_man - > GetNewDestination ( type , dest , error ) ;
}
if ( result ) {
SetAddressBook ( dest , label , " receive " ) ;
2019-10-07 14:11:34 -04:00
}
2019-10-07 14:11:34 -04:00
2019-10-07 14:11:34 -04:00
return result ;
}
2019-10-07 14:11:34 -04:00
bool CWallet : : GetNewChangeDestination ( const OutputType type , CTxDestination & dest , std : : string & error )
2011-11-17 14:01:25 -05:00
{
2019-10-07 14:11:34 -04:00
LOCK ( cs_wallet ) ;
2019-10-07 14:11:34 -04:00
error . clear ( ) ;
2019-07-09 10:09:56 -04:00
2019-10-24 09:44:20 +01:00
ReserveDestination reservedest ( this , type ) ;
if ( ! reservedest . GetReservedDestination ( dest , true ) ) {
2019-10-12 12:27:56 +02:00
error = _ ( " Error: Keypool ran out, please call keypoolrefill first " ) . translated ;
2019-06-18 15:49:02 -04:00
return false ;
}
reservedest . KeepDestination ( ) ;
return true ;
}
2020-03-02 16:54:59 +09:00
int64_t CWallet : : GetOldestKeyPoolTime ( ) const
2019-10-07 14:11:34 -04:00
{
2019-10-07 14:11:34 -04:00
LOCK ( cs_wallet ) ;
2019-10-07 14:11:34 -04:00
int64_t oldestKey = std : : numeric_limits < int64_t > : : max ( ) ;
2019-10-07 14:11:34 -04:00
for ( const auto & spk_man_pair : m_spk_managers ) {
oldestKey = std : : min ( oldestKey , spk_man_pair . second - > GetOldestKeyPoolTime ( ) ) ;
2019-10-07 14:11:34 -04:00
}
return oldestKey ;
}
2019-12-31 19:55:18 +01:00
void CWallet : : MarkDestinationsDirty ( const std : : set < CTxDestination > & destinations ) {
for ( auto & entry : mapWallet ) {
CWalletTx & wtx = entry . second ;
2020-01-07 13:47:20 +00:00
if ( wtx . m_is_cache_empty ) continue ;
2019-12-31 19:55:18 +01:00
for ( unsigned int i = 0 ; i < wtx . tx - > vout . size ( ) ; i + + ) {
CTxDestination dst ;
if ( ExtractDestination ( wtx . tx - > vout [ i ] . scriptPubKey , dst ) & & destinations . count ( dst ) ) {
wtx . MarkDirty ( ) ;
break ;
}
}
}
}
2020-03-02 17:17:33 +09:00
std : : map < CTxDestination , CAmount > CWallet : : GetAddressBalances ( interfaces : : Chain : : Lock & locked_chain ) const
2012-08-01 12:48:42 -04:00
{
2017-01-27 10:33:45 +09:00
std : : map < CTxDestination , CAmount > balances ;
2012-08-01 12:48:42 -04:00
{
LOCK ( cs_wallet ) ;
2019-09-06 09:15:49 -07:00
std : : set < uint256 > trusted_parents ;
2017-06-01 11:48:29 -04:00
for ( const auto & walletEntry : mapWallet )
2012-08-01 12:48:42 -04:00
{
2019-03-10 17:29:54 -04:00
const CWalletTx & wtx = walletEntry . second ;
2012-08-01 12:48:42 -04:00
2019-09-06 09:15:49 -07:00
if ( ! wtx . IsTrusted ( locked_chain , trusted_parents ) )
2012-08-01 12:48:42 -04:00
continue ;
2019-04-29 09:52:01 -04:00
if ( wtx . IsImmatureCoinBase ( ) )
2012-08-01 12:48:42 -04:00
continue ;
2019-04-29 09:52:01 -04:00
int nDepth = wtx . GetDepthInMainChain ( ) ;
2019-03-10 17:29:54 -04:00
if ( nDepth < ( wtx . IsFromMe ( ISMINE_ALL ) ? 0 : 1 ) )
2012-08-01 12:48:42 -04:00
continue ;
2019-03-10 17:29:54 -04:00
for ( unsigned int i = 0 ; i < wtx . tx - > vout . size ( ) ; i + + )
2012-08-01 12:48:42 -04:00
{
2012-08-20 13:43:33 -04:00
CTxDestination addr ;
2019-03-10 17:29:54 -04:00
if ( ! IsMine ( wtx . tx - > vout [ i ] ) )
2012-08-01 12:48:42 -04:00
continue ;
2019-03-10 17:29:54 -04:00
if ( ! ExtractDestination ( wtx . tx - > vout [ i ] . scriptPubKey , addr ) )
2012-08-20 13:43:33 -04:00
continue ;
2012-08-01 12:48:42 -04:00
2019-04-29 09:52:01 -04:00
CAmount n = IsSpent ( walletEntry . first , i ) ? 0 : wtx . tx - > vout [ i ] . nValue ;
2012-08-01 12:48:42 -04:00
if ( ! balances . count ( addr ) )
balances [ addr ] = 0 ;
balances [ addr ] + = n ;
}
}
}
return balances ;
}
2020-03-02 17:17:33 +09:00
std : : set < std : : set < CTxDestination > > CWallet : : GetAddressGroupings ( ) const
2012-08-01 12:48:42 -04:00
{
2019-02-18 16:09:45 -05:00
AssertLockHeld ( cs_wallet ) ;
2017-01-27 10:33:45 +09:00
std : : set < std : : set < CTxDestination > > groupings ;
std : : set < CTxDestination > grouping ;
2012-08-01 12:48:42 -04:00
2017-06-01 11:48:29 -04:00
for ( const auto & walletEntry : mapWallet )
2012-08-01 12:48:42 -04:00
{
2019-03-10 17:29:54 -04:00
const CWalletTx & wtx = walletEntry . second ;
2012-08-01 12:48:42 -04:00
2019-03-10 17:29:54 -04:00
if ( wtx . tx - > vin . size ( ) > 0 )
2012-08-01 12:48:42 -04:00
{
2012-09-27 13:29:35 -04:00
bool any_mine = false ;
2012-08-01 12:48:42 -04:00
// group all input addresses with each other
2019-03-10 17:29:54 -04:00
for ( const CTxIn & txin : wtx . tx - > vin )
2012-08-20 13:43:33 -04:00
{
CTxDestination address ;
2012-09-27 13:29:35 -04:00
if ( ! IsMine ( txin ) ) /* If this input isn't mine, ignore it */
continue ;
2017-01-19 16:08:03 -05:00
if ( ! ExtractDestination ( mapWallet . at ( txin . prevout . hash ) . tx - > vout [ txin . prevout . n ] . scriptPubKey , address ) )
2012-08-20 13:43:33 -04:00
continue ;
grouping . insert ( address ) ;
2012-09-27 13:29:35 -04:00
any_mine = true ;
2012-08-20 13:43:33 -04:00
}
2012-08-01 12:48:42 -04:00
// group change with input addresses
2012-09-27 13:29:35 -04:00
if ( any_mine )
{
2019-03-10 17:29:54 -04:00
for ( const CTxOut & txout : wtx . tx - > vout )
2012-09-27 13:29:35 -04:00
if ( IsChange ( txout ) )
{
CTxDestination txoutAddr ;
if ( ! ExtractDestination ( txout . scriptPubKey , txoutAddr ) )
continue ;
grouping . insert ( txoutAddr ) ;
}
}
if ( grouping . size ( ) > 0 )
{
groupings . insert ( grouping ) ;
grouping . clear ( ) ;
}
2012-08-01 12:48:42 -04:00
}
// group lone addrs by themselves
2019-03-10 17:29:54 -04:00
for ( const auto & txout : wtx . tx - > vout )
2017-05-18 09:42:14 +02:00
if ( IsMine ( txout ) )
2012-08-01 12:48:42 -04:00
{
2012-08-20 13:43:33 -04:00
CTxDestination address ;
2017-05-18 09:42:14 +02:00
if ( ! ExtractDestination ( txout . scriptPubKey , address ) )
2012-08-20 13:43:33 -04:00
continue ;
grouping . insert ( address ) ;
2012-08-01 12:48:42 -04:00
groupings . insert ( grouping ) ;
grouping . clear ( ) ;
}
}
2017-01-27 10:33:45 +09:00
std : : set < std : : set < CTxDestination > * > uniqueGroupings ; // a set of pointers to groups of addresses
std : : map < CTxDestination , std : : set < CTxDestination > * > setmap ; // map addresses to the unique group containing it
2017-06-02 03:18:57 +02:00
for ( std : : set < CTxDestination > _grouping : groupings )
2012-08-01 12:48:42 -04:00
{
// make a set of all the groups hit by this new group
2017-01-27 10:33:45 +09:00
std : : set < std : : set < CTxDestination > * > hits ;
std : : map < CTxDestination , std : : set < CTxDestination > * > : : iterator it ;
2018-06-18 07:58:28 +02:00
for ( const CTxDestination & address : _grouping )
2012-08-01 12:48:42 -04:00
if ( ( it = setmap . find ( address ) ) ! = setmap . end ( ) )
hits . insert ( ( * it ) . second ) ;
// merge all hit groups into a new single group and delete old groups
2017-01-27 10:33:45 +09:00
std : : set < CTxDestination > * merged = new std : : set < CTxDestination > ( _grouping ) ;
2017-06-02 03:18:57 +02:00
for ( std : : set < CTxDestination > * hit : hits )
2012-08-01 12:48:42 -04:00
{
merged - > insert ( hit - > begin ( ) , hit - > end ( ) ) ;
uniqueGroupings . erase ( hit ) ;
delete hit ;
}
uniqueGroupings . insert ( merged ) ;
// update setmap
2018-06-18 07:58:28 +02:00
for ( const CTxDestination & element : * merged )
2012-08-01 12:48:42 -04:00
setmap [ element ] = merged ;
}
2017-01-27 10:33:45 +09:00
std : : set < std : : set < CTxDestination > > ret ;
2018-06-18 07:58:28 +02:00
for ( const std : : set < CTxDestination > * uniqueGrouping : uniqueGroupings )
2012-08-01 12:48:42 -04:00
{
ret . insert ( * uniqueGrouping ) ;
delete uniqueGrouping ;
}
return ret ;
}
2017-10-20 13:27:55 -04:00
std : : set < CTxDestination > CWallet : : GetLabelAddresses ( const std : : string & label ) const
2013-07-16 09:01:09 +10:00
{
2014-11-28 19:11:49 +01:00
LOCK ( cs_wallet ) ;
2017-01-27 10:33:45 +09:00
std : : set < CTxDestination > result ;
2020-02-21 21:52:52 +00:00
for ( const std : : pair < const CTxDestination , CAddressBookData > & item : m_address_book )
2013-07-16 09:01:09 +10:00
{
2020-02-22 04:16:36 +00:00
if ( item . second . IsChange ( ) ) continue ;
2013-07-16 09:01:09 +10:00
const CTxDestination & address = item . first ;
2020-04-03 03:02:16 +00:00
const std : : string & strName = item . second . GetLabel ( ) ;
2017-10-20 13:27:55 -04:00
if ( strName = = label )
2013-07-16 09:01:09 +10:00
result . insert ( address ) ;
}
return result ;
}
2019-10-24 09:44:20 +01:00
bool ReserveDestination : : GetReservedDestination ( CTxDestination & dest , bool internal )
2011-06-26 19:23:24 +02:00
{
2019-10-07 14:11:34 -04:00
m_spk_man = pwallet - > GetScriptPubKeyMan ( type , internal ) ;
2019-10-07 14:11:34 -04:00
if ( ! m_spk_man ) {
return false ;
}
2019-02-06 21:26:55 -05:00
2011-06-26 19:23:24 +02:00
if ( nIndex = = - 1 )
{
2019-11-20 12:42:10 -05:00
m_spk_man - > TopUp ( ) ;
2011-06-26 19:23:24 +02:00
CKeyPool keypool ;
2019-10-07 14:11:34 -04:00
if ( ! m_spk_man - > GetReservedDestination ( type , internal , address , nIndex , keypool ) ) {
2014-06-16 14:45:32 +02:00
return false ;
2011-07-14 03:28:31 +02:00
}
2017-04-19 12:55:32 -04:00
fInternal = keypool . fInternal ;
2011-06-26 19:23:24 +02:00
}
2019-06-18 15:48:20 -04:00
dest = address ;
2013-04-25 19:30:28 +02:00
return true ;
2011-06-26 19:23:24 +02:00
}
2019-06-18 15:48:20 -04:00
void ReserveDestination : : KeepDestination ( )
2011-06-26 19:23:24 +02:00
{
2019-10-24 09:45:20 +01:00
if ( nIndex ! = - 1 ) {
2019-10-07 14:11:34 -04:00
m_spk_man - > KeepDestination ( nIndex , type ) ;
2019-10-24 09:45:20 +01:00
}
2011-06-26 19:23:24 +02:00
nIndex = - 1 ;
2019-06-18 15:48:20 -04:00
address = CNoDestination ( ) ;
2011-06-26 19:23:24 +02:00
}
2019-06-18 15:48:20 -04:00
void ReserveDestination : : ReturnDestination ( )
2011-06-26 19:23:24 +02:00
{
2017-04-19 13:11:16 -04:00
if ( nIndex ! = - 1 ) {
2019-10-07 14:11:34 -04:00
m_spk_man - > ReturnDestination ( nIndex , fInternal , address ) ;
2017-04-19 13:11:16 -04:00
}
2011-06-26 19:23:24 +02:00
nIndex = - 1 ;
2019-06-18 15:48:20 -04:00
address = CNoDestination ( ) ;
2011-06-26 19:23:24 +02:00
}
2011-07-07 15:22:54 +02:00
2016-03-30 02:04:22 +01:00
void CWallet : : LockCoin ( const COutPoint & output )
2012-09-27 13:52:09 -04:00
{
2019-02-18 16:09:45 -05:00
AssertLockHeld ( cs_wallet ) ;
2012-09-27 13:52:09 -04:00
setLockedCoins . insert ( output ) ;
}
2016-03-30 02:04:22 +01:00
void CWallet : : UnlockCoin ( const COutPoint & output )
2012-09-27 13:52:09 -04:00
{
2019-02-18 16:09:45 -05:00
AssertLockHeld ( cs_wallet ) ;
2012-09-27 13:52:09 -04:00
setLockedCoins . erase ( output ) ;
}
void CWallet : : UnlockAllCoins ( )
{
2019-02-18 16:09:45 -05:00
AssertLockHeld ( cs_wallet ) ;
2012-09-27 13:52:09 -04:00
setLockedCoins . clear ( ) ;
}
bool CWallet : : IsLockedCoin ( uint256 hash , unsigned int n ) const
{
2019-02-18 16:09:45 -05:00
AssertLockHeld ( cs_wallet ) ;
2012-09-27 13:52:09 -04:00
COutPoint outpt ( hash , n ) ;
return ( setLockedCoins . count ( outpt ) > 0 ) ;
}
2017-04-28 14:10:21 -04:00
void CWallet : : ListLockedCoins ( std : : vector < COutPoint > & vOutpts ) const
2012-09-27 13:52:09 -04:00
{
2019-02-18 16:09:45 -05:00
AssertLockHeld ( cs_wallet ) ;
2012-09-27 13:52:09 -04:00
for ( std : : set < COutPoint > : : iterator it = setLockedCoins . begin ( ) ;
it ! = setLockedCoins . end ( ) ; it + + ) {
COutPoint outpt = ( * it ) ;
vOutpts . push_back ( outpt ) ;
}
}
2014-10-26 15:03:12 +08:00
/** @} */ // end of Actions
2014-08-27 17:46:30 +02:00
2019-04-29 10:14:47 -04:00
void CWallet : : GetKeyBirthTimes ( interfaces : : Chain : : Lock & locked_chain , std : : map < CKeyID , int64_t > & mapKeyBirth ) const {
2019-02-18 16:09:45 -05:00
AssertLockHeld ( cs_wallet ) ;
2013-04-29 19:50:40 +02:00
mapKeyBirth . clear ( ) ;
2019-10-07 14:11:34 -04:00
LegacyScriptPubKeyMan * spk_man = GetLegacyScriptPubKeyMan ( ) ;
assert ( spk_man ! = nullptr ) ;
2019-10-07 14:11:34 -04:00
LOCK ( spk_man - > cs_KeyStore ) ;
2019-10-07 14:11:34 -04:00
2013-04-29 19:50:40 +02:00
// get birth times for keys with metadata
2019-10-07 14:11:34 -04:00
for ( const auto & entry : spk_man - > mapKeyMetadata ) {
2016-11-08 16:55:02 -05:00
if ( entry . second . nCreateTime ) {
mapKeyBirth [ entry . first ] = entry . second . nCreateTime ;
}
}
2013-04-29 19:50:40 +02:00
// map in which we'll infer heights of other keys
2020-01-22 17:15:17 -05:00
std : : map < CKeyID , const CWalletTx : : Confirmation * > mapKeyFirstBlock ;
CWalletTx : : Confirmation max_confirm ;
max_confirm . block_height = GetLastBlockHeight ( ) > 144 ? GetLastBlockHeight ( ) - 144 : 0 ; // the tip can be reorganized; use a 144-block safety margin
CHECK_NONFATAL ( chain ( ) . findAncestorByHeight ( GetLastBlockHash ( ) , max_confirm . block_height , FoundBlock ( ) . hash ( max_confirm . hashBlock ) ) ) ;
2019-10-07 14:11:34 -04:00
for ( const CKeyID & keyid : spk_man - > GetKeys ( ) ) {
2013-04-29 19:50:40 +02:00
if ( mapKeyBirth . count ( keyid ) = = 0 )
2020-01-22 17:15:17 -05:00
mapKeyFirstBlock [ keyid ] = & max_confirm ;
2013-04-29 19:50:40 +02:00
}
// if there are no such keys, we're done
if ( mapKeyFirstBlock . empty ( ) )
return ;
// find first block that affects those keys, if there are any left
2017-06-04 22:02:43 +02:00
for ( const auto & entry : mapWallet ) {
2013-04-29 19:50:40 +02:00
// iterate over all wallet transactions...
2017-06-04 22:02:43 +02:00
const CWalletTx & wtx = entry . second ;
2020-01-22 17:15:17 -05:00
if ( wtx . m_confirm . status = = CWalletTx : : CONFIRMED ) {
2013-04-29 19:50:40 +02:00
// ... which are already in a block
2017-06-02 03:18:57 +02:00
for ( const CTxOut & txout : wtx . tx - > vout ) {
2013-04-29 19:50:40 +02:00
// iterate over all their outputs
2019-10-07 14:11:34 -04:00
for ( const auto & keyid : GetAffectedKeys ( txout . scriptPubKey , * spk_man ) ) {
2013-04-29 19:50:40 +02:00
// ... and all their affected keys
2020-01-22 17:15:17 -05:00
auto rit = mapKeyFirstBlock . find ( keyid ) ;
if ( rit ! = mapKeyFirstBlock . end ( ) & & wtx . m_confirm . block_height < rit - > second - > block_height ) {
rit - > second = & wtx . m_confirm ;
}
2013-04-29 19:50:40 +02:00
}
}
}
}
// Extract block timestamps for those keys
2020-01-22 17:15:17 -05:00
for ( const auto & entry : mapKeyFirstBlock ) {
int64_t block_time ;
CHECK_NONFATAL ( chain ( ) . findBlock ( entry . second - > hashBlock , FoundBlock ( ) . time ( block_time ) ) ) ;
mapKeyBirth [ entry . first ] = block_time - TIMESTAMP_WINDOW ; // block times can be 2h off
}
2013-04-29 19:50:40 +02:00
}
2013-11-18 16:55:54 +01:00
2016-12-19 10:51:45 -05:00
/**
* Compute smart timestamp for a transaction being added to the wallet .
*
* Logic :
* - If sending a transaction , assign its timestamp to the current time .
* - If receiving a transaction outside a block , assign its timestamp to the
* current time .
* - If receiving a block with a future timestamp , assign all its ( not already
* known ) transactions ' timestamps to the current time .
* - If receiving a block with a past timestamp , before the most recent known
* transaction ( that we care about ) , assign all its ( not already known )
* transactions ' timestamps to the same timestamp as that most - recent - known
* transaction .
* - If receiving a block with a past timestamp , but after the most recent known
* transaction , assign all its ( not already known ) transactions ' timestamps to
* the block time .
*
* For more information see CWalletTx : : nTimeSmart ,
* https : //bitcointalk.org/?topic=54527, or
* https : //github.com/bitcoin/bitcoin/pull/1393.
*/
2016-12-16 10:00:26 -05:00
unsigned int CWallet : : ComputeTimeSmart ( const CWalletTx & wtx ) const
{
unsigned int nTimeSmart = wtx . nTimeReceived ;
2019-08-12 18:12:12 -04:00
if ( ! wtx . isUnconfirmed ( ) & & ! wtx . isAbandoned ( ) ) {
2019-01-07 22:56:46 -08:00
int64_t blocktime ;
2020-02-24 14:34:17 -05:00
if ( chain ( ) . findBlock ( wtx . m_confirm . hashBlock , FoundBlock ( ) . time ( blocktime ) ) ) {
2016-12-16 10:00:26 -05:00
int64_t latestNow = wtx . nTimeReceived ;
int64_t latestEntry = 0 ;
2017-02-10 15:00:30 -05:00
// Tolerate times up to the last timestamp in the wallet not more than 5 minutes into the future
int64_t latestTolerated = latestNow + 300 ;
const TxItems & txOrdered = wtxOrdered ;
for ( auto it = txOrdered . rbegin ( ) ; it ! = txOrdered . rend ( ) ; + + it ) {
2018-07-31 12:23:26 -04:00
CWalletTx * const pwtx = it - > second ;
2017-02-10 15:00:30 -05:00
if ( pwtx = = & wtx ) {
continue ;
}
int64_t nSmartTime ;
2018-07-31 12:23:26 -04:00
nSmartTime = pwtx - > nTimeSmart ;
if ( ! nSmartTime ) {
nSmartTime = pwtx - > nTimeReceived ;
2017-02-10 15:00:30 -05:00
}
if ( nSmartTime < = latestTolerated ) {
latestEntry = nSmartTime ;
if ( nSmartTime > latestNow ) {
latestNow = nSmartTime ;
2016-12-16 10:00:26 -05:00
}
2017-02-10 15:00:30 -05:00
break ;
2016-12-16 10:00:26 -05:00
}
}
nTimeSmart = std : : max ( latestEntry , std : : min ( blocktime , latestNow ) ) ;
2017-02-10 15:00:30 -05:00
} else {
2019-08-12 18:12:12 -04:00
WalletLogPrintf ( " %s: found %s in block %s not in index \n " , __func__ , wtx . GetHash ( ) . ToString ( ) , wtx . m_confirm . hashBlock . ToString ( ) ) ;
2016-12-16 10:00:26 -05:00
}
}
return nTimeSmart ;
}
2019-11-02 16:20:45 +00:00
bool CWallet : : AddDestData ( WalletBatch & batch , const CTxDestination & dest , const std : : string & key , const std : : string & value )
2013-11-18 16:55:54 +01:00
{
2014-01-14 05:05:43 +01:00
if ( boost : : get < CNoDestination > ( & dest ) )
return false ;
2020-02-21 21:52:52 +00:00
m_address_book [ dest ] . destdata . insert ( std : : make_pair ( key , value ) ) ;
2019-11-02 16:20:45 +00:00
return batch . WriteDestData ( EncodeDestination ( dest ) , key , value ) ;
2013-11-18 16:55:54 +01:00
}
2019-11-02 16:20:45 +00:00
bool CWallet : : EraseDestData ( WalletBatch & batch , const CTxDestination & dest , const std : : string & key )
2013-11-18 16:55:54 +01:00
{
2020-02-21 21:52:52 +00:00
if ( ! m_address_book [ dest ] . destdata . erase ( key ) )
2013-11-18 16:55:54 +01:00
return false ;
2019-11-02 16:20:45 +00:00
return batch . EraseDestData ( EncodeDestination ( dest ) , key ) ;
2013-11-18 16:55:54 +01:00
}
2018-07-27 08:22:42 +02:00
void CWallet : : LoadDestData ( const CTxDestination & dest , const std : : string & key , const std : : string & value )
2013-11-18 16:55:54 +01:00
{
2020-02-21 21:52:52 +00:00
m_address_book [ dest ] . destdata . insert ( std : : make_pair ( key , value ) ) ;
2013-11-18 16:55:54 +01:00
}
bool CWallet : : GetDestData ( const CTxDestination & dest , const std : : string & key , std : : string * value ) const
{
2020-02-21 21:52:52 +00:00
std : : map < CTxDestination , CAddressBookData > : : const_iterator i = m_address_book . find ( dest ) ;
if ( i ! = m_address_book . end ( ) )
2013-11-18 16:55:54 +01:00
{
CAddressBookData : : StringMap : : const_iterator j = i - > second . destdata . find ( key ) ;
if ( j ! = i - > second . destdata . end ( ) )
{
if ( value )
* value = j - > second ;
return true ;
}
}
return false ;
}
2014-08-21 05:04:43 +02:00
2017-04-28 14:10:21 -04:00
std : : vector < std : : string > CWallet : : GetDestValues ( const std : : string & prefix ) const
{
std : : vector < std : : string > values ;
2020-02-21 21:52:52 +00:00
for ( const auto & address : m_address_book ) {
2017-04-28 14:10:21 -04:00
for ( const auto & data : address . second . destdata ) {
if ( ! data . first . compare ( 0 , prefix . size ( ) , prefix ) ) {
values . emplace_back ( data . second ) ;
}
}
}
return values ;
}
2019-10-06 17:52:05 -04:00
bool CWallet : : Verify ( interfaces : : Chain & chain , const WalletLocation & location , bool salvage_wallet , std : : string & error_string , std : : vector < std : : string > & warnings )
2018-04-18 13:11:28 -04:00
{
// Do some checking on wallet path. It should be either a:
//
// 1. Path where a directory can be created.
// 2. Path to an existing directory.
// 3. Path to a symlink to a directory.
// 4. For backwards compatibility, the name of a data file in -walletdir.
LOCK ( cs_wallets ) ;
2018-09-28 16:50:18 +01:00
const fs : : path & wallet_path = location . GetPath ( ) ;
2018-04-18 13:11:28 -04:00
fs : : file_type path_type = fs : : symlink_status ( wallet_path ) . type ( ) ;
if ( ! ( path_type = = fs : : file_not_found | | path_type = = fs : : directory_file | |
( path_type = = fs : : symlink_file & & fs : : is_directory ( wallet_path ) ) | |
2018-09-28 16:50:18 +01:00
( path_type = = fs : : regular_file & & fs : : path ( location . GetName ( ) ) . filename ( ) = = location . GetName ( ) ) ) ) {
2018-04-18 14:17:09 -04:00
error_string = strprintf (
" Invalid -wallet path '%s'. -wallet path should point to a directory where wallet.dat and "
2018-04-18 13:11:28 -04:00
" database/log.?????????? files can be stored, a location where such a directory could be created, "
2018-04-18 14:17:09 -04:00
" or (for backwards compatibility) the name of an existing data file in -walletdir (%s) " ,
2018-09-28 16:50:18 +01:00
location . GetName ( ) , GetWalletDir ( ) ) ;
2018-04-18 14:17:09 -04:00
return false ;
2018-04-18 13:11:28 -04:00
}
// Make sure that the wallet path doesn't clash with an existing wallet path
2018-10-23 13:26:27 +08:00
if ( IsWalletLoaded ( wallet_path ) ) {
error_string = strprintf ( " Error loading wallet %s. Duplicate -wallet filename specified. " , location . GetName ( ) ) ;
return false ;
2018-04-18 13:11:28 -04:00
}
2018-05-18 16:28:50 -04:00
// Keep same database environment instance across Verify/Recover calls below.
std : : unique_ptr < WalletDatabase > database = WalletDatabase : : Create ( wallet_path ) ;
2018-08-04 12:14:28 -04:00
try {
if ( ! WalletBatch : : VerifyEnvironment ( wallet_path , error_string ) ) {
return false ;
}
} catch ( const fs : : filesystem_error & e ) {
2018-09-28 16:50:18 +01:00
error_string = strprintf ( " Error loading wallet %s. %s " , location . GetName ( ) , fsbridge : : get_filesystem_error_message ( e ) ) ;
2018-04-18 14:17:09 -04:00
return false ;
2018-04-18 13:11:28 -04:00
}
if ( salvage_wallet ) {
// Recover readable keypairs:
2019-03-22 00:24:55 -04:00
CWallet dummyWallet ( & chain , WalletLocation ( ) , WalletDatabase : : CreateDummy ( ) ) ;
2018-04-18 13:11:28 -04:00
std : : string backup_filename ;
2019-08-22 13:16:40 -04:00
// Even if we don't use this lock in this function, we want to preserve
// lock order in LoadToWallet if query of chain state is needed to know
// tx status. If lock can't be taken, tx confirmation status may be not
// reliable.
auto locked_chain = dummyWallet . LockChain ( ) ;
2018-04-18 13:11:28 -04:00
if ( ! WalletBatch : : Recover ( wallet_path , ( void * ) & dummyWallet , WalletBatch : : RecoverKeysOnlyFilter , backup_filename ) ) {
return false ;
}
}
2019-10-06 17:52:05 -04:00
return WalletBatch : : VerifyDatabaseFile ( wallet_path , warnings , error_string ) ;
2018-04-18 13:11:28 -04:00
}
2019-10-06 17:52:05 -04:00
std : : shared_ptr < CWallet > CWallet : : CreateWalletFromFile ( interfaces : : Chain & chain , const WalletLocation & location , std : : string & error , std : : vector < std : : string > & warnings , uint64_t wallet_creation_flags )
2016-02-22 12:07:55 +01:00
{
2019-08-19 17:13:11 -04:00
const std : : string walletFile = WalletDataFilePath ( location . GetPath ( ) ) . string ( ) ;
2017-11-13 21:25:46 -05:00
2016-02-22 12:07:55 +01:00
// needed to restore wallet transaction meta data after -zapwallettxes
std : : vector < CWalletTx > vWtx ;
2017-08-01 21:17:40 +02:00
if ( gArgs . GetBoolArg ( " -zapwallettxes " , false ) ) {
2019-06-28 21:09:58 +03:00
chain . initMessage ( _ ( " Zapping all transactions from wallet... " ) . translated ) ;
2016-02-22 12:07:55 +01:00
2019-03-22 00:24:55 -04:00
std : : unique_ptr < CWallet > tempWallet = MakeUnique < CWallet > ( & chain , location , WalletDatabase : : Create ( location . GetPath ( ) ) ) ;
2016-02-22 12:07:55 +01:00
DBErrors nZapWalletRet = tempWallet - > ZapWalletTx ( vWtx ) ;
2018-03-09 15:03:40 +01:00
if ( nZapWalletRet ! = DBErrors : : LOAD_OK ) {
2019-10-06 17:52:05 -04:00
error = strprintf ( _ ( " Error loading %s: Wallet corrupted " ) . translated , walletFile ) ;
2017-08-07 07:36:37 +02:00
return nullptr ;
2016-02-22 12:07:55 +01:00
}
}
2019-06-28 21:09:58 +03:00
chain . initMessage ( _ ( " Loading wallet... " ) . translated ) ;
2016-02-22 12:07:55 +01:00
int64_t nStart = GetTimeMillis ( ) ;
bool fFirstRun = true ;
2018-04-28 22:36:43 +01:00
// TODO: Can't use std::make_shared because we need a custom deleter but
// should be possible to use std::allocate_shared.
2019-03-22 00:24:55 -04:00
std : : shared_ptr < CWallet > walletInstance ( new CWallet ( & chain , location , WalletDatabase : : Create ( location . GetPath ( ) ) ) , ReleaseWallet ) ;
2016-02-22 12:07:55 +01:00
DBErrors nLoadWalletRet = walletInstance - > LoadWallet ( fFirstRun ) ;
2019-10-06 17:52:05 -04:00
if ( nLoadWalletRet ! = DBErrors : : LOAD_OK ) {
2018-03-09 15:03:40 +01:00
if ( nLoadWalletRet = = DBErrors : : CORRUPT ) {
2019-10-06 17:52:05 -04:00
error = strprintf ( _ ( " Error loading %s: Wallet corrupted " ) . translated , walletFile ) ;
2017-08-07 07:36:37 +02:00
return nullptr ;
2016-09-09 08:44:47 +00:00
}
2018-03-09 15:03:40 +01:00
else if ( nLoadWalletRet = = DBErrors : : NONCRITICAL_ERROR )
2016-02-22 12:07:55 +01:00
{
2019-10-06 17:52:05 -04:00
warnings . push_back ( strprintf ( _ ( " Error reading %s! All keys read correctly, but transaction data "
2019-06-28 21:09:58 +03:00
" or address book entries might be missing or incorrect. " ) . translated ,
2016-03-15 10:30:37 +01:00
walletFile ) ) ;
2016-02-22 12:07:55 +01:00
}
2018-03-09 15:03:40 +01:00
else if ( nLoadWalletRet = = DBErrors : : TOO_NEW ) {
2019-10-06 17:52:05 -04:00
error = strprintf ( _ ( " Error loading %s: Wallet requires newer version of %s " ) . translated , walletFile , PACKAGE_NAME ) ;
2017-08-07 07:36:37 +02:00
return nullptr ;
2016-09-09 08:44:47 +00:00
}
2018-03-09 15:03:40 +01:00
else if ( nLoadWalletRet = = DBErrors : : NEED_REWRITE )
2016-02-22 12:07:55 +01:00
{
2019-10-06 17:52:05 -04:00
error = strprintf ( _ ( " Wallet needed to be rewritten: restart %s to complete " ) . translated , PACKAGE_NAME ) ;
2017-08-07 07:36:37 +02:00
return nullptr ;
2016-09-09 08:44:47 +00:00
}
else {
2019-10-06 17:52:05 -04:00
error = strprintf ( _ ( " Error loading %s " ) . translated , walletFile ) ;
2017-08-07 07:36:37 +02:00
return nullptr ;
2016-02-22 12:07:55 +01:00
}
}
if ( fFirstRun )
{
2017-09-05 15:54:11 -07:00
// ensure this wallet.dat can only be opened by clients supporting HD with chain split and expects no default key
2018-03-31 13:37:27 -04:00
walletInstance - > SetMinVersion ( FEATURE_LATEST ) ;
2017-03-24 10:53:35 +01:00
2018-09-13 13:53:19 +09:00
walletInstance - > SetWalletFlags ( wallet_creation_flags , false ) ;
2019-10-07 14:11:34 -04:00
2019-07-17 17:54:15 -04:00
// Only create LegacyScriptPubKeyMan when not descriptor wallet
if ( ! walletInstance - > IsWalletFlagSet ( WALLET_FLAG_DESCRIPTORS ) ) {
walletInstance - > SetupLegacyScriptPubKeyMan ( ) ;
}
2019-10-07 14:11:34 -04:00
2018-09-13 13:53:19 +09:00
if ( ! ( wallet_creation_flags & ( WALLET_FLAG_DISABLE_PRIVATE_KEYS | WALLET_FLAG_BLANK_WALLET ) ) ) {
2019-10-07 14:11:34 -04:00
LOCK ( walletInstance - > cs_wallet ) ;
2019-10-07 14:11:34 -04:00
for ( auto spk_man : walletInstance - > GetActiveScriptPubKeyMans ( ) ) {
2019-10-07 14:11:34 -04:00
if ( ! spk_man - > SetupGeneration ( ) ) {
error = _ ( " Unable to generate initial keys " ) . translated ;
return nullptr ;
}
}
2016-02-22 12:07:55 +01:00
}
2019-05-16 21:42:34 +02:00
auto locked_chain = chain . lock ( ) ;
2020-02-26 16:05:49 -05:00
walletInstance - > chainStateFlushed ( locked_chain - > getTipLocator ( ) ) ;
2017-05-05 08:53:39 +02:00
} else if ( wallet_creation_flags & WALLET_FLAG_DISABLE_PRIVATE_KEYS ) {
// Make it impossible to disable private keys after creation
2019-10-06 17:52:05 -04:00
error = strprintf ( _ ( " Error loading %s: Private keys can only be disabled during creation " ) . translated , walletFile ) ;
2017-05-05 08:53:39 +02:00
return NULL ;
} else if ( walletInstance - > IsWalletFlagSet ( WALLET_FLAG_DISABLE_PRIVATE_KEYS ) ) {
2019-10-07 14:11:34 -04:00
for ( auto spk_man : walletInstance - > GetActiveScriptPubKeyMans ( ) ) {
if ( spk_man - > HavePrivateKeys ( ) ) {
2019-10-07 14:11:34 -04:00
warnings . push_back ( strprintf ( _ ( " Warning: Private keys detected in wallet {%s} with disabled private keys " ) . translated , walletFile ) ) ;
2019-10-07 14:11:34 -04:00
break ;
2019-10-07 14:11:34 -04:00
}
2017-05-05 08:53:39 +02:00
}
2016-06-13 16:27:41 +02:00
}
2016-02-22 12:07:55 +01:00
2018-03-19 15:57:11 -04:00
if ( ! gArgs . GetArg ( " -addresstype " , " " ) . empty ( ) & & ! ParseOutputType ( gArgs . GetArg ( " -addresstype " , " " ) , walletInstance - > m_default_address_type ) ) {
2019-10-06 17:52:05 -04:00
error = strprintf ( _ ( " Unknown address type '%s' " ) . translated , gArgs . GetArg ( " -addresstype " , " " ) ) ;
2018-02-10 21:06:35 -05:00
return nullptr ;
}
2018-03-19 15:57:11 -04:00
if ( ! gArgs . GetArg ( " -changetype " , " " ) . empty ( ) & & ! ParseOutputType ( gArgs . GetArg ( " -changetype " , " " ) , walletInstance - > m_default_change_type ) ) {
2019-10-06 17:52:05 -04:00
error = strprintf ( _ ( " Unknown change type '%s' " ) . translated , gArgs . GetArg ( " -changetype " , " " ) ) ;
2018-02-10 21:06:35 -05:00
return nullptr ;
}
2018-04-07 12:12:46 -04:00
if ( gArgs . IsArgSet ( " -mintxfee " ) ) {
CAmount n = 0 ;
if ( ! ParseMoney ( gArgs . GetArg ( " -mintxfee " , " " ) , n ) | | 0 = = n ) {
2019-10-06 17:52:05 -04:00
error = AmountErrMsg ( " mintxfee " , gArgs . GetArg ( " -mintxfee " , " " ) ) . translated ;
2018-04-07 12:12:46 -04:00
return nullptr ;
}
if ( n > HIGH_TX_FEE_PER_KB ) {
2019-10-06 17:52:05 -04:00
warnings . push_back ( AmountHighWarn ( " -mintxfee " ) . translated + " " +
2019-06-28 21:09:58 +03:00
_ ( " This is the minimum transaction fee you pay on every transaction. " ) . translated ) ;
2018-04-07 12:12:46 -04:00
}
walletInstance - > m_min_fee = CFeeRate ( n ) ;
}
if ( gArgs . IsArgSet ( " -fallbackfee " ) ) {
CAmount nFeePerK = 0 ;
if ( ! ParseMoney ( gArgs . GetArg ( " -fallbackfee " , " " ) , nFeePerK ) ) {
2019-10-06 17:52:05 -04:00
error = strprintf ( _ ( " Invalid amount for -fallbackfee=<amount>: '%s' " ) . translated , gArgs . GetArg ( " -fallbackfee " , " " ) ) ;
2018-04-07 12:12:46 -04:00
return nullptr ;
}
if ( nFeePerK > HIGH_TX_FEE_PER_KB ) {
2019-10-06 17:52:05 -04:00
warnings . push_back ( AmountHighWarn ( " -fallbackfee " ) . translated + " " +
2019-06-28 21:09:58 +03:00
_ ( " This is the transaction fee you may pay when fee estimates are not available. " ) . translated ) ;
2018-04-07 12:12:46 -04:00
}
walletInstance - > m_fallback_fee = CFeeRate ( nFeePerK ) ;
}
2019-08-01 22:39:46 +02:00
// Disable fallback fee in case value was set to 0, enable if non-null value
walletInstance - > m_allow_fallback_fee = walletInstance - > m_fallback_fee . GetFeePerK ( ) ! = 0 ;
2018-04-07 12:12:46 -04:00
if ( gArgs . IsArgSet ( " -discardfee " ) ) {
CAmount nFeePerK = 0 ;
if ( ! ParseMoney ( gArgs . GetArg ( " -discardfee " , " " ) , nFeePerK ) ) {
2019-10-06 17:52:05 -04:00
error = strprintf ( _ ( " Invalid amount for -discardfee=<amount>: '%s' " ) . translated , gArgs . GetArg ( " -discardfee " , " " ) ) ;
2018-04-07 12:12:46 -04:00
return nullptr ;
}
if ( nFeePerK > HIGH_TX_FEE_PER_KB ) {
2019-10-06 17:52:05 -04:00
warnings . push_back ( AmountHighWarn ( " -discardfee " ) . translated + " " +
2019-06-28 21:09:58 +03:00
_ ( " This is the transaction fee you may discard if change is smaller than dust at this level " ) . translated ) ;
2018-04-07 12:12:46 -04:00
}
walletInstance - > m_discard_rate = CFeeRate ( nFeePerK ) ;
}
if ( gArgs . IsArgSet ( " -paytxfee " ) ) {
CAmount nFeePerK = 0 ;
if ( ! ParseMoney ( gArgs . GetArg ( " -paytxfee " , " " ) , nFeePerK ) ) {
2019-10-06 17:52:05 -04:00
error = AmountErrMsg ( " paytxfee " , gArgs . GetArg ( " -paytxfee " , " " ) ) . translated ;
2018-04-07 12:12:46 -04:00
return nullptr ;
}
if ( nFeePerK > HIGH_TX_FEE_PER_KB ) {
2019-10-06 17:52:05 -04:00
warnings . push_back ( AmountHighWarn ( " -paytxfee " ) . translated + " " +
2019-06-28 21:09:58 +03:00
_ ( " This is the transaction fee you will pay if you send a transaction. " ) . translated ) ;
2018-04-07 12:12:46 -04:00
}
walletInstance - > m_pay_tx_fee = CFeeRate ( nFeePerK , 1000 ) ;
2019-03-06 16:47:57 -05:00
if ( walletInstance - > m_pay_tx_fee < chain . relayMinFee ( ) ) {
2019-10-06 17:52:05 -04:00
error = strprintf ( _ ( " Invalid amount for -paytxfee=<amount>: '%s' (must be at least %s) " ) . translated ,
gArgs . GetArg ( " -paytxfee " , " " ) , chain . relayMinFee ( ) . ToString ( ) ) ;
2018-04-07 12:12:46 -04:00
return nullptr ;
}
}
2019-02-20 13:45:16 -05:00
2019-10-06 17:52:05 -04:00
if ( gArgs . IsArgSet ( " -maxtxfee " ) ) {
2019-02-20 13:45:16 -05:00
CAmount nMaxFee = 0 ;
if ( ! ParseMoney ( gArgs . GetArg ( " -maxtxfee " , " " ) , nMaxFee ) ) {
2019-10-06 17:52:05 -04:00
error = AmountErrMsg ( " maxtxfee " , gArgs . GetArg ( " -maxtxfee " , " " ) ) . translated ;
2019-02-20 13:45:16 -05:00
return nullptr ;
}
if ( nMaxFee > HIGH_MAX_TX_FEE ) {
2019-10-06 17:52:05 -04:00
warnings . push_back ( _ ( " -maxtxfee is set very high! Fees this large could be paid on a single transaction. " ) . translated ) ;
2019-02-20 13:45:16 -05:00
}
if ( CFeeRate ( nMaxFee , 1000 ) < chain . relayMinFee ( ) ) {
2019-10-06 17:52:05 -04:00
error = strprintf ( _ ( " Invalid amount for -maxtxfee=<amount>: '%s' (must be at least the minrelay fee of %s to prevent stuck transactions) " ) . translated ,
gArgs . GetArg ( " -maxtxfee " , " " ) , chain . relayMinFee ( ) . ToString ( ) ) ;
2019-02-20 13:45:16 -05:00
return nullptr ;
}
walletInstance - > m_default_max_tx_fee = nMaxFee ;
}
2019-08-20 13:17:05 -04:00
if ( chain . relayMinFee ( ) . GetFeePerK ( ) > HIGH_TX_FEE_PER_KB ) {
2019-10-06 17:52:05 -04:00
warnings . push_back ( AmountHighWarn ( " -minrelaytxfee " ) . translated + " " +
2019-06-28 21:09:58 +03:00
_ ( " The wallet will avoid paying less than the minimum relay fee. " ) . translated ) ;
2019-08-20 13:17:05 -04:00
}
2019-02-20 13:45:16 -05:00
2018-04-07 12:12:46 -04:00
walletInstance - > m_confirm_target = gArgs . GetArg ( " -txconfirmtarget " , DEFAULT_TX_CONFIRM_TARGET ) ;
walletInstance - > m_spend_zero_conf_change = gArgs . GetBoolArg ( " -spendzeroconfchange " , DEFAULT_SPEND_ZEROCONF_CHANGE ) ;
walletInstance - > m_signal_rbf = gArgs . GetBoolArg ( " -walletrbf " , DEFAULT_WALLET_RBF ) ;
2018-08-08 10:42:13 -04:00
walletInstance - > WalletLogPrintf ( " Wallet completed loading in %15dms \n " , GetTimeMillis ( ) - nStart ) ;
2016-02-22 12:07:55 +01:00
2017-07-18 15:49:56 -04:00
// Try to top up keypool. No-op if the wallet is locked.
walletInstance - > TopUpKeyPool ( ) ;
2017-07-26 10:23:01 -04:00
auto locked_chain = chain . lock ( ) ;
LOCK ( walletInstance - > cs_wallet ) ;
2018-01-12 00:56:27 +00:00
2019-01-08 00:06:24 -08:00
int rescan_height = 0 ;
2017-08-01 21:17:40 +02:00
if ( ! gArgs . GetBoolArg ( " -rescan " , false ) )
2016-02-22 12:07:55 +01:00
{
2017-12-08 06:39:22 -05:00
WalletBatch batch ( * walletInstance - > database ) ;
2016-02-22 12:07:55 +01:00
CBlockLocator locator ;
2019-01-08 00:06:24 -08:00
if ( batch . ReadBestBlock ( locator ) ) {
if ( const Optional < int > fork_height = locked_chain - > findLocatorFork ( locator ) ) {
rescan_height = * fork_height ;
}
}
2016-02-22 12:07:55 +01:00
}
2017-01-17 18:06:16 -05:00
2019-01-08 00:06:24 -08:00
const Optional < int > tip_height = locked_chain - > getHeight ( ) ;
if ( tip_height ) {
walletInstance - > m_last_block_processed = locked_chain - > getBlockHash ( * tip_height ) ;
2019-04-20 12:02:52 -04:00
walletInstance - > m_last_block_processed_height = * tip_height ;
2019-01-08 00:06:24 -08:00
} else {
walletInstance - > m_last_block_processed . SetNull ( ) ;
2019-04-20 12:02:52 -04:00
walletInstance - > m_last_block_processed_height = - 1 ;
2019-01-08 00:06:24 -08:00
}
2017-01-17 18:06:16 -05:00
2019-01-08 00:06:24 -08:00
if ( tip_height & & * tip_height ! = rescan_height )
2016-02-22 12:07:55 +01:00
{
2019-04-22 14:30:02 -04:00
// We can't rescan beyond non-pruned blocks, stop and throw an error.
// This might happen if a user uses an old wallet within a pruned node
// or if they ran -disablewallet for a longer time, then decided to re-enable
if ( chain . havePruned ( ) ) {
// Exit early and print an error.
// If a block is pruned after this check, we will load the wallet,
// but fail the rescan with a generic error.
2019-01-08 00:06:24 -08:00
int block_height = * tip_height ;
while ( block_height > 0 & & locked_chain - > haveBlockOnDisk ( block_height - 1 ) & & rescan_height ! = block_height ) {
- - block_height ;
}
2016-02-22 12:07:55 +01:00
2019-01-08 00:06:24 -08:00
if ( rescan_height ! = block_height ) {
2019-10-06 17:52:05 -04:00
error = _ ( " Prune: last wallet synchronisation goes beyond pruned data. You need to -reindex (download the whole blockchain again in case of pruned node) " ) . translated ;
2017-08-07 07:36:37 +02:00
return nullptr ;
2016-09-09 08:44:47 +00:00
}
2016-02-22 12:07:55 +01:00
}
2019-06-28 21:09:58 +03:00
chain . initMessage ( _ ( " Rescanning... " ) . translated ) ;
2019-01-08 00:06:24 -08:00
walletInstance - > WalletLogPrintf ( " Rescanning last %i blocks (from block %i)... \n " , * tip_height - rescan_height , rescan_height ) ;
2017-03-02 16:14:39 -05:00
// No need to read and scan block if block was created before
// our wallet birthday (as adjusted for block time variability)
2020-02-01 23:07:19 +02:00
// The way the 'time_first_key' is initialized is just a workaround for the gcc bug #47679 since version 4.6.0.
Optional < int64_t > time_first_key = MakeOptional ( false , int64_t ( ) ) ; ;
2019-10-07 14:11:34 -04:00
for ( auto spk_man : walletInstance - > GetAllScriptPubKeyMans ( ) ) {
2019-10-07 14:11:34 -04:00
int64_t time = spk_man - > GetTimeFirstKey ( ) ;
if ( ! time_first_key | | time < * time_first_key ) time_first_key = time ;
}
if ( time_first_key ) {
if ( Optional < int > first_block = locked_chain - > findFirstBlockWithTimeAndHeight ( * time_first_key - TIMESTAMP_WINDOW , rescan_height , nullptr ) ) {
2019-01-08 00:06:24 -08:00
rescan_height = * first_block ;
}
2017-03-02 16:14:39 -05:00
}
2017-12-12 13:13:58 -10:00
{
2020-04-12 00:42:15 +01:00
WalletRescanReserver reserver ( * walletInstance ) ;
2020-01-22 16:53:42 -05:00
if ( ! reserver . reserve ( ) | | ( ScanResult : : SUCCESS ! = walletInstance - > ScanForWalletTransactions ( locked_chain - > getBlockHash ( rescan_height ) , rescan_height , { } /* max height */ , reserver , true /* update */ ) . status ) ) {
2019-10-06 17:52:05 -04:00
error = _ ( " Failed to rescan the wallet during initialization " ) . translated ;
2017-12-12 13:13:58 -10:00
return nullptr ;
}
}
2020-02-26 16:05:49 -05:00
walletInstance - > chainStateFlushed ( locked_chain - > getTipLocator ( ) ) ;
2017-12-08 06:39:22 -05:00
walletInstance - > database - > IncrementUpdateCounter ( ) ;
2016-02-22 12:07:55 +01:00
// Restore wallet transaction metadata after -zapwallettxes=1
2017-08-01 21:17:40 +02:00
if ( gArgs . GetBoolArg ( " -zapwallettxes " , false ) & & gArgs . GetArg ( " -zapwallettxes " , " 1 " ) ! = " 2 " )
2016-02-22 12:07:55 +01:00
{
2017-12-08 06:39:22 -05:00
WalletBatch batch ( * walletInstance - > database ) ;
2016-02-22 12:07:55 +01:00
2017-06-02 03:18:57 +02:00
for ( const CWalletTx & wtxOld : vWtx )
2016-02-22 12:07:55 +01:00
{
uint256 hash = wtxOld . GetHash ( ) ;
std : : map < uint256 , CWalletTx > : : iterator mi = walletInstance - > mapWallet . find ( hash ) ;
if ( mi ! = walletInstance - > mapWallet . end ( ) )
{
const CWalletTx * copyFrom = & wtxOld ;
CWalletTx * copyTo = & mi - > second ;
copyTo - > mapValue = copyFrom - > mapValue ;
copyTo - > vOrderForm = copyFrom - > vOrderForm ;
copyTo - > nTimeReceived = copyFrom - > nTimeReceived ;
copyTo - > nTimeSmart = copyFrom - > nTimeSmart ;
copyTo - > fFromMe = copyFrom - > fFromMe ;
copyTo - > nOrderPos = copyFrom - > nOrderPos ;
2017-12-08 06:39:22 -05:00
batch . WriteTx ( * copyTo ) ;
2016-02-22 12:07:55 +01:00
}
}
}
}
2018-05-07 17:08:03 -04:00
2019-09-27 07:31:44 -04:00
{
LOCK ( cs_wallets ) ;
for ( auto & load_wallet : g_load_wallet_fns ) {
load_wallet ( interfaces : : MakeWallet ( walletInstance ) ) ;
}
}
2018-06-19 21:33:13 +02:00
2017-07-30 16:00:56 -04:00
// Register with the validation interface. It's ok to do this after rescan since we're still holding locked_chain.
2020-03-10 15:46:20 -04:00
walletInstance - > m_chain_notifications_handler = walletInstance - > chain ( ) . handleNotifications ( walletInstance ) ;
2018-05-07 17:08:03 -04:00
2017-08-01 21:17:40 +02:00
walletInstance - > SetBroadcastTransactions ( gArgs . GetBoolArg ( " -walletbroadcast " , DEFAULT_WALLETBROADCAST ) ) ;
2016-02-22 12:07:55 +01:00
2016-08-19 15:50:48 -07:00
{
2018-06-15 19:02:52 -04:00
walletInstance - > WalletLogPrintf ( " setKeyPool.size() = %u \n " , walletInstance - > GetKeyPoolSize ( ) ) ;
walletInstance - > WalletLogPrintf ( " mapWallet.size() = %u \n " , walletInstance - > mapWallet . size ( ) ) ;
2020-02-21 21:52:52 +00:00
walletInstance - > WalletLogPrintf ( " m_address_book.size() = %u \n " , walletInstance - > m_address_book . size ( ) ) ;
2016-08-19 15:50:48 -07:00
}
2016-09-09 08:44:47 +00:00
return walletInstance ;
}
2020-02-22 02:50:46 +00:00
const CAddressBookData * CWallet : : FindAddressBookEntry ( const CTxDestination & dest , bool allow_change ) const
{
const auto & address_book_it = m_address_book . find ( dest ) ;
if ( address_book_it = = m_address_book . end ( ) ) return nullptr ;
if ( ( ! allow_change ) & & address_book_it - > second . IsChange ( ) ) {
return nullptr ;
}
return & address_book_it - > second ;
}
2019-04-22 00:23:33 -04:00
bool CWallet : : UpgradeWallet ( int version , std : : string & error , std : : vector < std : : string > & warnings )
2019-04-06 12:55:34 -04:00
{
2019-04-22 00:23:33 -04:00
int prev_version = GetVersion ( ) ;
2019-04-22 00:18:35 -04:00
int nMaxVersion = version ;
2019-04-22 00:09:39 -04:00
if ( nMaxVersion = = 0 ) // the -upgradewallet without argument case
2019-04-06 12:55:34 -04:00
{
2019-04-22 00:23:33 -04:00
WalletLogPrintf ( " Performing wallet upgrade to %i \n " , FEATURE_LATEST ) ;
2019-04-22 00:09:39 -04:00
nMaxVersion = FEATURE_LATEST ;
2019-04-22 00:23:33 -04:00
SetMinVersion ( FEATURE_LATEST ) ; // permanently upgrade the wallet immediately
2019-04-06 12:55:34 -04:00
}
2019-04-22 00:09:39 -04:00
else
2019-04-22 00:23:33 -04:00
WalletLogPrintf ( " Allowing wallet upgrade up to %i \n " , nMaxVersion ) ;
if ( nMaxVersion < GetVersion ( ) )
2019-04-22 00:09:39 -04:00
{
error = _ ( " Cannot downgrade wallet " ) . translated ;
return false ;
}
2019-04-22 00:23:33 -04:00
SetMaxVersion ( nMaxVersion ) ;
2019-04-06 12:55:34 -04:00
2019-04-22 00:23:33 -04:00
LOCK ( cs_wallet ) ;
2019-04-06 12:55:34 -04:00
2019-04-22 00:09:39 -04:00
// Do not upgrade versions to any version between HD_SPLIT and FEATURE_PRE_SPLIT_KEYPOOL unless already supporting HD_SPLIT
2019-04-22 00:23:33 -04:00
int max_version = GetVersion ( ) ;
if ( ! CanSupportFeature ( FEATURE_HD_SPLIT ) & & max_version > = FEATURE_HD_SPLIT & & max_version < FEATURE_PRE_SPLIT_KEYPOOL ) {
2019-04-06 12:56:15 -04:00
error = _ ( " Cannot upgrade a non HD split wallet without upgrading to support pre split keypool. Please use version 169900 or no version specified. " ) . translated ;
2019-04-22 00:09:39 -04:00
return false ;
}
2019-04-06 12:55:34 -04:00
2019-04-22 00:23:33 -04:00
for ( auto spk_man : GetActiveScriptPubKeyMans ( ) ) {
if ( ! spk_man - > Upgrade ( prev_version , error ) ) {
return false ;
2019-04-06 12:55:34 -04:00
}
}
return true ;
}
2018-04-19 17:42:40 -04:00
void CWallet : : postInitProcess ( )
2016-10-20 09:22:13 +02:00
{
2019-03-27 16:59:47 +00:00
auto locked_chain = chain ( ) . lock ( ) ;
LOCK ( cs_wallet ) ;
2016-10-20 09:22:13 +02:00
// Add wallet transactions that aren't already in a block to mempool
// Do this here as mempool requires genesis block to be loaded
2019-04-29 09:52:01 -04:00
ReacceptWalletTransactions ( ) ;
2019-03-27 17:00:08 +00:00
// Update wallet transactions with current mempool transactions.
chain ( ) . requestMempoolTransactions ( * this ) ;
2016-10-20 09:22:13 +02:00
}
2020-03-02 17:23:34 +09:00
bool CWallet : : BackupWallet ( const std : : string & strDest ) const
2016-05-16 17:31:16 -07:00
{
2017-12-08 06:39:22 -05:00
return database - > Backup ( strDest ) ;
2016-05-16 17:31:16 -07:00
}
2014-08-21 05:04:43 +02:00
CKeyPool : : CKeyPool ( )
{
nTime = GetTime ( ) ;
2017-01-16 11:10:12 +01:00
fInternal = false ;
2018-04-21 03:10:12 -04:00
m_pre_split = false ;
2014-08-21 05:04:43 +02:00
}
2017-01-10 16:45:30 +01:00
CKeyPool : : CKeyPool ( const CPubKey & vchPubKeyIn , bool internalIn )
2014-08-21 05:04:43 +02:00
{
nTime = GetTime ( ) ;
vchPubKey = vchPubKeyIn ;
2017-01-10 16:45:30 +01:00
fInternal = internalIn ;
2018-04-21 03:10:12 -04:00
m_pre_split = false ;
2014-08-21 05:04:43 +02:00
}
2019-04-29 09:52:01 -04:00
int CWalletTx : : GetDepthInMainChain ( ) const
2014-08-28 17:15:21 +02:00
{
2019-04-22 14:22:25 -04:00
assert ( pwallet ! = nullptr ) ;
AssertLockHeld ( pwallet - > cs_wallet ) ;
2019-08-12 18:12:12 -04:00
if ( isUnconfirmed ( ) | | isAbandoned ( ) ) return 0 ;
2016-01-07 16:31:27 -05:00
2019-04-22 14:22:25 -04:00
return ( pwallet - > GetLastBlockHeight ( ) - m_confirm . block_height + 1 ) * ( isConflicted ( ) ? - 1 : 1 ) ;
2014-08-28 17:15:21 +02:00
}
2019-04-29 09:52:01 -04:00
int CWalletTx : : GetBlocksToMaturity ( ) const
2014-08-28 17:15:21 +02:00
{
if ( ! IsCoinBase ( ) )
return 0 ;
2019-04-29 09:52:01 -04:00
int chain_depth = GetDepthInMainChain ( ) ;
2018-07-13 12:41:42 -04:00
assert ( chain_depth > = 0 ) ; // coinbase tx should not be conflicted
return std : : max ( 0 , ( COINBASE_MATURITY + 1 ) - chain_depth ) ;
2014-08-28 17:15:21 +02:00
}
2019-04-29 09:52:01 -04:00
bool CWalletTx : : IsImmatureCoinBase ( ) const
2018-07-11 01:17:59 -04:00
{
// note GetBlocksToMaturity is 0 for non-coinbase tx
2019-04-29 09:52:01 -04:00
return GetBlocksToMaturity ( ) > 0 ;
2018-07-11 01:17:59 -04:00
}
2014-08-28 17:15:21 +02:00
2019-12-29 01:58:46 +01:00
std : : vector < OutputGroup > CWallet : : GroupOutputs ( const std : : vector < COutput > & outputs , bool single_coin , const size_t max_ancestors ) const {
2018-07-19 11:43:03 +09:00
std : : vector < OutputGroup > groups ;
std : : map < CTxDestination , OutputGroup > gmap ;
2019-12-29 01:58:46 +01:00
std : : set < CTxDestination > full_groups ;
2018-07-19 11:43:03 +09:00
for ( const auto & output : outputs ) {
if ( output . fSpendable ) {
2019-12-29 01:58:46 +01:00
CTxDestination dst ;
2018-07-19 11:43:03 +09:00
CInputCoin input_coin = output . GetInputCoin ( ) ;
size_t ancestors , descendants ;
2017-07-28 19:42:27 -04:00
chain ( ) . getTransactionAncestry ( output . tx - > GetHash ( ) , ancestors , descendants ) ;
2018-07-19 11:43:03 +09:00
if ( ! single_coin & & ExtractDestination ( output . tx - > tx - > vout [ output . i ] . scriptPubKey , dst ) ) {
2019-12-29 01:58:46 +01:00
auto it = gmap . find ( dst ) ;
if ( it ! = gmap . end ( ) ) {
// Limit output groups to no more than OUTPUT_GROUP_MAX_ENTRIES
// number of entries, to protect against inadvertently creating
// a too-large transaction when using -avoidpartialspends to
// prevent breaking consensus or surprising users with a very
// high amount of fees.
if ( it - > second . m_outputs . size ( ) > = OUTPUT_GROUP_MAX_ENTRIES ) {
groups . push_back ( it - > second ) ;
it - > second = OutputGroup { } ;
full_groups . insert ( dst ) ;
}
it - > second . Insert ( input_coin , output . nDepth , output . tx - > IsFromMe ( ISMINE_ALL ) , ancestors , descendants ) ;
} else {
gmap [ dst ] . Insert ( input_coin , output . nDepth , output . tx - > IsFromMe ( ISMINE_ALL ) , ancestors , descendants ) ;
2018-07-19 11:43:03 +09:00
}
} else {
groups . emplace_back ( input_coin , output . nDepth , output . tx - > IsFromMe ( ISMINE_ALL ) , ancestors , descendants ) ;
}
}
}
if ( ! single_coin ) {
2019-12-29 01:58:46 +01:00
for ( auto & it : gmap ) {
auto & group = it . second ;
if ( full_groups . count ( it . first ) > 0 ) {
// Make this unattractive as we want coin selection to avoid it if possible
group . m_ancestors = max_ancestors - 1 ;
}
groups . push_back ( group ) ;
}
2018-07-19 11:43:03 +09:00
}
return groups ;
}
2018-07-19 17:56:52 -07:00
2019-12-05 18:14:53 -05:00
bool CWallet : : IsCrypted ( ) const
2019-06-06 23:58:21 +02:00
{
2019-12-05 18:14:53 -05:00
return HasEncryptionKeys ( ) ;
2019-06-06 23:58:21 +02:00
}
bool CWallet : : IsLocked ( ) const
{
if ( ! IsCrypted ( ) ) {
return false ;
}
2019-10-07 14:11:34 -04:00
LOCK ( cs_wallet ) ;
2019-06-06 23:58:21 +02:00
return vMasterKey . empty ( ) ;
}
bool CWallet : : Lock ( )
{
2019-12-05 18:14:53 -05:00
if ( ! IsCrypted ( ) )
2019-06-06 23:58:21 +02:00
return false ;
{
2019-10-07 14:11:34 -04:00
LOCK ( cs_wallet ) ;
2019-06-06 23:58:21 +02:00
vMasterKey . clear ( ) ;
}
NotifyStatusChanged ( this ) ;
return true ;
}
2019-10-07 14:11:34 -04:00
2019-12-05 18:02:11 -05:00
bool CWallet : : Unlock ( const CKeyingMaterial & vMasterKeyIn , bool accept_no_keys )
{
{
2019-10-07 14:11:34 -04:00
LOCK ( cs_wallet ) ;
2019-10-07 14:11:34 -04:00
for ( const auto & spk_man_pair : m_spk_managers ) {
if ( ! spk_man_pair . second - > CheckDecryptionKey ( vMasterKeyIn , accept_no_keys ) ) {
2019-12-05 18:02:11 -05:00
return false ;
}
}
vMasterKey = vMasterKeyIn ;
}
NotifyStatusChanged ( this ) ;
return true ;
}
2019-10-07 14:11:34 -04:00
std : : set < ScriptPubKeyMan * > CWallet : : GetActiveScriptPubKeyMans ( ) const
{
std : : set < ScriptPubKeyMan * > spk_mans ;
for ( bool internal : { false , true } ) {
for ( OutputType t : OUTPUT_TYPES ) {
auto spk_man = GetScriptPubKeyMan ( t , internal ) ;
if ( spk_man ) {
spk_mans . insert ( spk_man ) ;
}
}
}
return spk_mans ;
}
std : : set < ScriptPubKeyMan * > CWallet : : GetAllScriptPubKeyMans ( ) const
{
std : : set < ScriptPubKeyMan * > spk_mans ;
for ( const auto & spk_man_pair : m_spk_managers ) {
spk_mans . insert ( spk_man_pair . second . get ( ) ) ;
}
return spk_mans ;
}
ScriptPubKeyMan * CWallet : : GetScriptPubKeyMan ( const OutputType & type , bool internal ) const
{
const std : : map < OutputType , ScriptPubKeyMan * > & spk_managers = internal ? m_internal_spk_managers : m_external_spk_managers ;
std : : map < OutputType , ScriptPubKeyMan * > : : const_iterator it = spk_managers . find ( type ) ;
if ( it = = spk_managers . end ( ) ) {
WalletLogPrintf ( " %s scriptPubKey Manager for output type %d does not exist \n " , internal ? " Internal " : " External " , static_cast < int > ( type ) ) ;
return nullptr ;
}
return it - > second ;
}
2020-02-10 19:49:56 -05:00
std : : set < ScriptPubKeyMan * > CWallet : : GetScriptPubKeyMans ( const CScript & script , SignatureData & sigdata ) const
{
std : : set < ScriptPubKeyMan * > spk_mans ;
for ( const auto & spk_man_pair : m_spk_managers ) {
if ( spk_man_pair . second - > CanProvide ( script , sigdata ) ) {
spk_mans . insert ( spk_man_pair . second . get ( ) ) ;
}
}
return spk_mans ;
}
2019-10-07 14:11:34 -04:00
ScriptPubKeyMan * CWallet : : GetScriptPubKeyMan ( const CScript & script ) const
{
2019-10-07 14:11:34 -04:00
SignatureData sigdata ;
for ( const auto & spk_man_pair : m_spk_managers ) {
if ( spk_man_pair . second - > CanProvide ( script , sigdata ) ) {
return spk_man_pair . second . get ( ) ;
}
}
return nullptr ;
}
ScriptPubKeyMan * CWallet : : GetScriptPubKeyMan ( const uint256 & id ) const
{
if ( m_spk_managers . count ( id ) > 0 ) {
return m_spk_managers . at ( id ) . get ( ) ;
}
return nullptr ;
2019-10-07 14:11:34 -04:00
}
2020-02-10 21:27:59 -05:00
std : : unique_ptr < SigningProvider > CWallet : : GetSolvingProvider ( const CScript & script ) const
2019-10-07 14:11:34 -04:00
{
2019-10-07 14:11:34 -04:00
SignatureData sigdata ;
2020-02-10 21:27:59 -05:00
return GetSolvingProvider ( script , sigdata ) ;
2019-10-07 14:11:34 -04:00
}
2020-02-10 21:27:59 -05:00
std : : unique_ptr < SigningProvider > CWallet : : GetSolvingProvider ( const CScript & script , SignatureData & sigdata ) const
2019-10-07 14:11:34 -04:00
{
2019-10-07 14:11:34 -04:00
for ( const auto & spk_man_pair : m_spk_managers ) {
if ( spk_man_pair . second - > CanProvide ( script , sigdata ) ) {
2020-02-10 21:27:59 -05:00
return spk_man_pair . second - > GetSolvingProvider ( script ) ;
2019-10-07 14:11:34 -04:00
}
}
return nullptr ;
2019-10-07 14:11:34 -04:00
}
LegacyScriptPubKeyMan * CWallet : : GetLegacyScriptPubKeyMan ( ) const
{
2019-07-08 11:21:31 -04:00
if ( IsWalletFlagSet ( WALLET_FLAG_DESCRIPTORS ) ) {
return nullptr ;
}
2019-10-07 14:11:34 -04:00
// Legacy wallets only have one ScriptPubKeyMan which is a LegacyScriptPubKeyMan.
// Everything in m_internal_spk_managers and m_external_spk_managers point to the same legacyScriptPubKeyMan.
auto it = m_internal_spk_managers . find ( OutputType : : LEGACY ) ;
if ( it = = m_internal_spk_managers . end ( ) ) return nullptr ;
return dynamic_cast < LegacyScriptPubKeyMan * > ( it - > second ) ;
2019-10-07 14:11:34 -04:00
}
2019-12-05 18:01:30 -05:00
2019-10-07 14:11:34 -04:00
LegacyScriptPubKeyMan * CWallet : : GetOrCreateLegacyScriptPubKeyMan ( )
{
SetupLegacyScriptPubKeyMan ( ) ;
return GetLegacyScriptPubKeyMan ( ) ;
}
void CWallet : : SetupLegacyScriptPubKeyMan ( )
{
2019-07-08 11:21:31 -04:00
if ( ! m_internal_spk_managers . empty ( ) | | ! m_external_spk_managers . empty ( ) | | ! m_spk_managers . empty ( ) | | IsWalletFlagSet ( WALLET_FLAG_DESCRIPTORS ) ) {
2019-10-07 14:11:34 -04:00
return ;
}
auto spk_manager = std : : unique_ptr < ScriptPubKeyMan > ( new LegacyScriptPubKeyMan ( * this ) ) ;
for ( const auto & type : OUTPUT_TYPES ) {
m_internal_spk_managers [ type ] = spk_manager . get ( ) ;
m_external_spk_managers [ type ] = spk_manager . get ( ) ;
}
m_spk_managers [ spk_manager - > GetID ( ) ] = std : : move ( spk_manager ) ;
2019-10-07 14:11:34 -04:00
}
2019-12-05 18:01:30 -05:00
const CKeyingMaterial & CWallet : : GetEncryptionKey ( ) const
{
return vMasterKey ;
}
bool CWallet : : HasEncryptionKeys ( ) const
{
return ! mapMasterKeys . empty ( ) ;
}
2019-10-07 14:11:34 -04:00
void CWallet : : ConnectScriptPubKeyManNotifiers ( )
{
for ( const auto & spk_man : GetActiveScriptPubKeyMans ( ) ) {
spk_man - > NotifyWatchonlyChanged . connect ( NotifyWatchonlyChanged ) ;
spk_man - > NotifyCanGetAddressesChanged . connect ( NotifyCanGetAddressesChanged ) ;
}
}
2019-07-08 15:41:31 -04:00
void CWallet : : LoadDescriptorScriptPubKeyMan ( uint256 id , WalletDescriptor & desc )
{
auto spk_manager = std : : unique_ptr < ScriptPubKeyMan > ( new DescriptorScriptPubKeyMan ( * this , desc ) ) ;
m_spk_managers [ id ] = std : : move ( spk_manager ) ;
}
void CWallet : : SetActiveScriptPubKeyMan ( uint256 id , OutputType type , bool internal , bool memonly )
{
auto & spk_mans = internal ? m_internal_spk_managers : m_external_spk_managers ;
auto spk_man = m_spk_managers . at ( id ) . get ( ) ;
spk_man - > SetType ( type , internal ) ;
spk_mans [ type ] = spk_man ;
if ( ! memonly ) {
WalletBatch batch ( * database ) ;
if ( ! batch . WriteActiveScriptPubKeyMan ( static_cast < uint8_t > ( type ) , id , internal ) ) {
throw std : : runtime_error ( std : : string ( __func__ ) + " : writing active ScriptPubKeyMan id failed " ) ;
}
}
NotifyCanGetAddressesChanged ( ) ;
}