2014-08-26 16:28:32 -04:00
// Copyright (c) 2009-2010 Satoshi Nakamoto
2020-12-31 09:48:25 +01:00
// Copyright (c) 2009-2020 The Bitcoin Core developers
2014-08-26 16:28:32 -04:00
// Distributed under the MIT software license, see the accompanying
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
2017-11-10 13:57:53 +13:00
# include <policy/fees.h>
2014-08-26 16:28:32 -04:00
2017-11-10 13:57:53 +13:00
# include <clientversion.h>
2020-12-07 14:01:19 +01:00
# include <fs.h>
# include <logging.h>
2017-11-10 13:57:53 +13:00
# include <streams.h>
# include <txmempool.h>
2018-10-22 15:51:11 -07:00
# include <util/system.h>
2014-08-26 16:28:32 -04:00
2020-12-26 13:00:48 +01:00
static const char * FEE_ESTIMATES_FILENAME = " fee_estimates.dat " ;
2020-09-10 14:27:36 +02:00
2017-03-09 14:02:00 -05:00
static constexpr double INF_FEERATE = 1e99 ;
2020-12-26 13:00:48 +01:00
std : : string StringForFeeEstimateHorizon ( FeeEstimateHorizon horizon )
{
switch ( horizon ) {
case FeeEstimateHorizon : : SHORT_HALFLIFE : return " short " ;
case FeeEstimateHorizon : : MED_HALFLIFE : return " medium " ;
case FeeEstimateHorizon : : LONG_HALFLIFE : return " long " ;
} // no default case, so the compiler can warn about missing cases
assert ( false ) ;
2017-06-06 13:08:27 -04:00
}
2017-02-16 17:27:20 -05:00
/**
* We will instantiate an instance of this class to track transactions that were
* included in a block . We will lump transactions into a bucket according to their
* approximate feerate and then track how long it took for those txs to be included in a block
*
* The tracking of unconfirmed ( mempool ) transactions is completely independent of the
* historical tracking of transactions that have been confirmed in a block .
*/
class TxConfirmStats
{
private :
//Define the buckets we will group transactions into
2017-02-23 15:13:41 -05:00
const std : : vector < double > & buckets ; // The upper-bound of the range for the bucket (inclusive)
const std : : map < double , unsigned int > & bucketMap ; // Map of bucket upper-bound to index into all vectors by bucket
2017-02-16 17:27:20 -05:00
// For each bucket X:
// Count the total # of txs in each bucket
// Track the historical moving average of this total over blocks
std : : vector < double > txCtAvg ;
// Count the total # of txs confirmed within Y blocks in each bucket
2019-06-09 18:52:10 +02:00
// Track the historical moving average of these totals over blocks
2017-02-23 15:13:41 -05:00
std : : vector < std : : vector < double > > confAvg ; // confAvg[Y][X]
2017-03-09 15:26:05 -05:00
// Track moving avg of txs which have been evicted from the mempool
// after failing to be confirmed within Y blocks
std : : vector < std : : vector < double > > failAvg ; // failAvg[Y][X]
2017-02-16 17:27:20 -05:00
// Sum the total feerate of all tx's in each bucket
// Track the historical moving average of this total over blocks
2020-07-30 21:19:38 +02:00
std : : vector < double > m_feerate_avg ;
2017-02-16 17:27:20 -05:00
// Combine the conf counts with tx counts to calculate the confirmation % for each Y,X
// Combine the total value with the tx counts to calculate the avg feerate per bucket
double decay ;
2017-04-03 11:31:27 -04:00
// Resolution (# of blocks) with which confirmations are tracked
2017-02-23 15:13:41 -05:00
unsigned int scale ;
2017-02-16 17:27:20 -05:00
// Mempool counts of outstanding transactions
// For each bucket X, track the number of transactions in the mempool
// that are unconfirmed for each possible confirmation value Y
std : : vector < std : : vector < int > > unconfTxs ; //unconfTxs[Y][X]
2017-04-03 11:31:27 -04:00
// transactions still unconfirmed after GetMaxConfirms for each bucket
2017-02-16 17:27:20 -05:00
std : : vector < int > oldUnconfTxs ;
2017-02-23 15:13:41 -05:00
void resizeInMemoryCounters ( size_t newbuckets ) ;
2017-02-16 17:27:20 -05:00
public :
/**
* Create new TxConfirmStats . This is called by BlockPolicyEstimator ' s
* constructor with default values .
* @ param defaultBuckets contains the upper limits for the bucket boundaries
2017-07-16 12:20:49 +02:00
* @ param maxPeriods max number of periods to track
2017-02-16 17:27:20 -05:00
* @ param decay how much to decay the historical moving average per block
*/
2017-02-23 15:13:41 -05:00
TxConfirmStats ( const std : : vector < double > & defaultBuckets , const std : : map < double , unsigned int > & defaultBucketMap ,
2017-04-03 11:31:27 -04:00
unsigned int maxPeriods , double decay , unsigned int scale ) ;
2017-02-16 17:27:20 -05:00
2017-02-28 17:29:42 -05:00
/** Roll the circular buffer for unconfirmed txs*/
2017-02-16 17:27:20 -05:00
void ClearCurrent ( unsigned int nBlockHeight ) ;
/**
* Record a new transaction data point in the current block stats
* @ param blocksToConfirm the number of blocks it took this transaction to confirm
* @ param val the feerate of the transaction
* @ warning blocksToConfirm is 1 - based and has to be > = 1
*/
void Record ( int blocksToConfirm , double val ) ;
/** Record a new transaction entering the mempool*/
unsigned int NewTx ( unsigned int nBlockHeight , double val ) ;
/** Remove a transaction from mempool tracking stats*/
void removeTx ( unsigned int entryHeight , unsigned int nBestSeenHeight ,
2017-03-09 15:26:05 -05:00
unsigned int bucketIndex , bool inBlock ) ;
2017-02-16 17:27:20 -05:00
/** Update our estimates by decaying our historical moving average and updating
with the data gathered from the current block */
void UpdateMovingAverages ( ) ;
/**
* Calculate a feerate estimate . Find the lowest value bucket ( or range of buckets
* to make sure we have enough data points ) whose transactions still have sufficient likelihood
* of being confirmed within the target number of confirmations
* @ param confTarget target number of confirmations
* @ param sufficientTxVal required average number of transactions per block in a bucket range
* @ param minSuccess the success probability we require
* @ param nBlockHeight the current block height
*/
double EstimateMedianVal ( int confTarget , double sufficientTxVal ,
2020-07-30 18:48:57 +02:00
double minSuccess , unsigned int nBlockHeight ,
2017-01-24 16:30:03 -05:00
EstimationResult * result = nullptr ) const ;
2017-02-16 17:27:20 -05:00
/** Return the max number of confirms we're tracking */
2017-04-03 11:31:27 -04:00
unsigned int GetMaxConfirms ( ) const { return scale * confAvg . size ( ) ; }
2017-02-16 17:27:20 -05:00
/** Write state of estimation data to a file*/
void Write ( CAutoFile & fileout ) const ;
/**
* Read saved state of estimation data from a file and replace all internal data structures and
* variables with this state .
*/
2017-02-23 15:13:41 -05:00
void Read ( CAutoFile & filein , int nFileVersion , size_t numBuckets ) ;
2017-02-16 17:27:20 -05:00
} ;
2017-02-16 16:23:15 -05:00
TxConfirmStats : : TxConfirmStats ( const std : : vector < double > & defaultBuckets ,
2017-02-23 15:13:41 -05:00
const std : : map < double , unsigned int > & defaultBucketMap ,
2017-04-03 11:31:27 -04:00
unsigned int maxPeriods , double _decay , unsigned int _scale )
2020-07-30 21:19:38 +02:00
: buckets ( defaultBuckets ) , bucketMap ( defaultBucketMap ) , decay ( _decay ) , scale ( _scale )
2014-08-26 16:28:32 -04:00
{
2017-06-26 16:00:25 +02:00
assert ( _scale ! = 0 & & " _scale must be non-zero " ) ;
2017-04-03 11:31:27 -04:00
confAvg . resize ( maxPeriods ) ;
failAvg . resize ( maxPeriods ) ;
for ( unsigned int i = 0 ; i < maxPeriods ; i + + ) {
2020-07-31 20:28:40 +02:00
confAvg [ i ] . resize ( buckets . size ( ) ) ;
2017-03-09 15:26:05 -05:00
failAvg [ i ] . resize ( buckets . size ( ) ) ;
2014-08-26 16:28:32 -04:00
}
txCtAvg . resize ( buckets . size ( ) ) ;
2020-07-30 21:19:38 +02:00
m_feerate_avg . resize ( buckets . size ( ) ) ;
2017-02-23 15:13:41 -05:00
resizeInMemoryCounters ( buckets . size ( ) ) ;
2014-08-26 16:28:32 -04:00
}
2017-02-23 15:13:41 -05:00
void TxConfirmStats : : resizeInMemoryCounters ( size_t newbuckets ) {
// newbuckets must be passed in because the buckets referred to during Read have not been updated yet.
unconfTxs . resize ( GetMaxConfirms ( ) ) ;
for ( unsigned int i = 0 ; i < unconfTxs . size ( ) ; i + + ) {
unconfTxs [ i ] . resize ( newbuckets ) ;
}
oldUnconfTxs . resize ( newbuckets ) ;
2014-08-26 16:28:32 -04:00
}
2017-02-28 17:29:42 -05:00
// Roll the unconfirmed txs circular buffer
2014-08-26 16:28:32 -04:00
void TxConfirmStats : : ClearCurrent ( unsigned int nBlockHeight )
{
for ( unsigned int j = 0 ; j < buckets . size ( ) ; j + + ) {
2020-07-30 21:19:38 +02:00
oldUnconfTxs [ j ] + = unconfTxs [ nBlockHeight % unconfTxs . size ( ) ] [ j ] ;
2014-08-26 16:28:32 -04:00
unconfTxs [ nBlockHeight % unconfTxs . size ( ) ] [ j ] = 0 ;
}
}
2020-07-30 21:19:38 +02:00
void TxConfirmStats : : Record ( int blocksToConfirm , double feerate )
2014-08-26 16:28:32 -04:00
{
// blocksToConfirm is 1-based
if ( blocksToConfirm < 1 )
return ;
2020-07-30 21:19:38 +02:00
int periodsToConfirm = ( blocksToConfirm + scale - 1 ) / scale ;
unsigned int bucketindex = bucketMap . lower_bound ( feerate ) - > second ;
2017-04-03 11:31:27 -04:00
for ( size_t i = periodsToConfirm ; i < = confAvg . size ( ) ; i + + ) {
2017-02-28 17:29:42 -05:00
confAvg [ i - 1 ] [ bucketindex ] + + ;
2014-08-26 16:28:32 -04:00
}
2017-02-28 17:29:42 -05:00
txCtAvg [ bucketindex ] + + ;
2020-07-30 21:19:38 +02:00
m_feerate_avg [ bucketindex ] + = feerate ;
2014-08-26 16:28:32 -04:00
}
void TxConfirmStats : : UpdateMovingAverages ( )
{
2020-07-31 20:28:40 +02:00
assert ( confAvg . size ( ) = = failAvg . size ( ) ) ;
2014-08-26 16:28:32 -04:00
for ( unsigned int j = 0 ; j < buckets . size ( ) ; j + + ) {
2020-07-31 20:28:40 +02:00
for ( unsigned int i = 0 ; i < confAvg . size ( ) ; i + + ) {
confAvg [ i ] [ j ] * = decay ;
failAvg [ i ] [ j ] * = decay ;
}
2020-07-30 21:19:38 +02:00
m_feerate_avg [ j ] * = decay ;
txCtAvg [ j ] * = decay ;
2014-08-26 16:28:32 -04:00
}
}
// returns -1 on error conditions
double TxConfirmStats : : EstimateMedianVal ( int confTarget , double sufficientTxVal ,
2020-07-30 18:48:57 +02:00
double successBreakPoint , unsigned int nBlockHeight ,
EstimationResult * result ) const
2014-08-26 16:28:32 -04:00
{
// Counters for a bucket (or range of buckets)
double nConf = 0 ; // Number of tx's confirmed within the confTarget
double totalNum = 0 ; // Total number of tx's that were ever confirmed
int extraNum = 0 ; // Number of tx's still in mempool for confTarget or longer
2017-03-09 15:26:05 -05:00
double failNum = 0 ; // Number of tx's that were never confirmed but removed from the mempool after confTarget
2020-07-30 21:19:38 +02:00
const int periodTarget = ( confTarget + scale - 1 ) / scale ;
const int maxbucketindex = buckets . size ( ) - 1 ;
2014-08-26 16:28:32 -04:00
// We'll combine buckets until we have enough samples.
// The near and far variables will define the range we've combined
// The best variables are the last range we saw which still had a high
// enough confirmation rate to count as success.
// The cur variables are the current range we're counting.
2020-07-30 18:48:57 +02:00
unsigned int curNearBucket = maxbucketindex ;
unsigned int bestNearBucket = maxbucketindex ;
unsigned int curFarBucket = maxbucketindex ;
unsigned int bestFarBucket = maxbucketindex ;
2014-08-26 16:28:32 -04:00
bool foundAnswer = false ;
unsigned int bins = unconfTxs . size ( ) ;
2017-02-21 22:18:13 -05:00
bool newBucketRange = true ;
2017-01-24 16:30:03 -05:00
bool passing = true ;
EstimatorBucket passBucket ;
EstimatorBucket failBucket ;
2014-08-26 16:28:32 -04:00
2020-07-30 18:48:57 +02:00
// Start counting from highest feerate transactions
for ( int bucket = maxbucketindex ; bucket > = 0 ; - - bucket ) {
2017-02-21 22:18:13 -05:00
if ( newBucketRange ) {
curNearBucket = bucket ;
newBucketRange = false ;
}
2014-08-26 16:28:32 -04:00
curFarBucket = bucket ;
2017-04-03 11:31:27 -04:00
nConf + = confAvg [ periodTarget - 1 ] [ bucket ] ;
2014-08-26 16:28:32 -04:00
totalNum + = txCtAvg [ bucket ] ;
2017-04-03 11:31:27 -04:00
failNum + = failAvg [ periodTarget - 1 ] [ bucket ] ;
2014-08-26 16:28:32 -04:00
for ( unsigned int confct = confTarget ; confct < GetMaxConfirms ( ) ; confct + + )
2020-07-30 21:19:38 +02:00
extraNum + = unconfTxs [ ( nBlockHeight - confct ) % bins ] [ bucket ] ;
2014-08-26 16:28:32 -04:00
extraNum + = oldUnconfTxs [ bucket ] ;
// If we have enough transaction data points in this range of buckets,
// we can test for success
// (Only count the confirmed data points, so that each confirmation count
// will be looking at the same amount of data and same bucket breaks)
if ( totalNum > = sufficientTxVal / ( 1 - decay ) ) {
2017-03-09 15:26:05 -05:00
double curPct = nConf / ( totalNum + failNum + extraNum ) ;
2014-08-26 16:28:32 -04:00
// Check to see if we are no longer getting confirmed at the success rate
2020-07-30 18:48:57 +02:00
if ( curPct < successBreakPoint ) {
2017-01-24 16:30:03 -05:00
if ( passing = = true ) {
// First time we hit a failure record the failed bucket
unsigned int failMinBucket = std : : min ( curNearBucket , curFarBucket ) ;
unsigned int failMaxBucket = std : : max ( curNearBucket , curFarBucket ) ;
failBucket . start = failMinBucket ? buckets [ failMinBucket - 1 ] : 0 ;
failBucket . end = buckets [ failMaxBucket ] ;
failBucket . withinTarget = nConf ;
failBucket . totalConfirmed = totalNum ;
failBucket . inMempool = extraNum ;
2017-03-09 15:26:05 -05:00
failBucket . leftMempool = failNum ;
2017-01-24 16:30:03 -05:00
passing = false ;
}
2017-03-02 10:08:25 -05:00
continue ;
}
2014-08-26 16:28:32 -04:00
// Otherwise update the cumulative stats, and the bucket variables
// and reset the counters
else {
2017-01-24 16:30:03 -05:00
failBucket = EstimatorBucket ( ) ; // Reset any failed bucket, currently passing
2014-08-26 16:28:32 -04:00
foundAnswer = true ;
2017-01-24 16:30:03 -05:00
passing = true ;
passBucket . withinTarget = nConf ;
2014-08-26 16:28:32 -04:00
nConf = 0 ;
2017-01-24 16:30:03 -05:00
passBucket . totalConfirmed = totalNum ;
2014-08-26 16:28:32 -04:00
totalNum = 0 ;
2017-01-24 16:30:03 -05:00
passBucket . inMempool = extraNum ;
2017-03-09 15:26:05 -05:00
passBucket . leftMempool = failNum ;
failNum = 0 ;
2014-08-26 16:28:32 -04:00
extraNum = 0 ;
bestNearBucket = curNearBucket ;
bestFarBucket = curFarBucket ;
2017-02-21 22:18:13 -05:00
newBucketRange = true ;
2014-08-26 16:28:32 -04:00
}
}
}
double median = - 1 ;
double txSum = 0 ;
2016-03-21 13:04:40 -04:00
// Calculate the "average" feerate of the best bucket range that met success conditions
// Find the bucket with the median transaction and then report the average feerate from that bucket
2014-08-26 16:28:32 -04:00
// This is a compromise between finding the median which we can't since we don't save all tx's
// and reporting the average which is less accurate
2017-01-24 16:30:03 -05:00
unsigned int minBucket = std : : min ( bestNearBucket , bestFarBucket ) ;
unsigned int maxBucket = std : : max ( bestNearBucket , bestFarBucket ) ;
2014-08-26 16:28:32 -04:00
for ( unsigned int j = minBucket ; j < = maxBucket ; j + + ) {
txSum + = txCtAvg [ j ] ;
}
if ( foundAnswer & & txSum ! = 0 ) {
txSum = txSum / 2 ;
for ( unsigned int j = minBucket ; j < = maxBucket ; j + + ) {
if ( txCtAvg [ j ] < txSum )
txSum - = txCtAvg [ j ] ;
else { // we're in the right bucket
2020-07-30 21:19:38 +02:00
median = m_feerate_avg [ j ] / txCtAvg [ j ] ;
2014-08-26 16:28:32 -04:00
break ;
}
}
2017-01-24 16:30:03 -05:00
passBucket . start = minBucket ? buckets [ minBucket - 1 ] : 0 ;
passBucket . end = buckets [ maxBucket ] ;
2014-08-26 16:28:32 -04:00
}
2017-01-24 16:30:03 -05:00
// If we were passing until we reached last few buckets with insufficient data, then report those as failed
if ( passing & & ! newBucketRange ) {
unsigned int failMinBucket = std : : min ( curNearBucket , curFarBucket ) ;
unsigned int failMaxBucket = std : : max ( curNearBucket , curFarBucket ) ;
failBucket . start = failMinBucket ? buckets [ failMinBucket - 1 ] : 0 ;
failBucket . end = buckets [ failMaxBucket ] ;
failBucket . withinTarget = nConf ;
failBucket . totalConfirmed = totalNum ;
failBucket . inMempool = extraNum ;
2017-03-09 15:26:05 -05:00
failBucket . leftMempool = failNum ;
2014-08-26 16:28:32 -04:00
}
2020-08-15 01:57:59 +02:00
float passed_within_target_perc = 0.0 ;
float failed_within_target_perc = 0.0 ;
if ( ( passBucket . totalConfirmed + passBucket . inMempool + passBucket . leftMempool ) ) {
passed_within_target_perc = 100 * passBucket . withinTarget / ( passBucket . totalConfirmed + passBucket . inMempool + passBucket . leftMempool ) ;
}
if ( ( failBucket . totalConfirmed + failBucket . inMempool + failBucket . leftMempool ) ) {
failed_within_target_perc = 100 * failBucket . withinTarget / ( failBucket . totalConfirmed + failBucket . inMempool + failBucket . leftMempool ) ;
}
2020-07-30 18:48:57 +02:00
LogPrint ( BCLog : : ESTIMATEFEE , " FeeEst: %d > %.0f%% decay %.5f: feerate: %g from (%g - %g) %.2f%% %.1f/(%.1f %d mem %.1f out) Fail: (%g - %g) %.2f%% %.1f/(%.1f %d mem %.1f out) \n " ,
confTarget , 100.0 * successBreakPoint , decay ,
2017-01-24 16:30:03 -05:00
median , passBucket . start , passBucket . end ,
2020-08-15 01:57:59 +02:00
passed_within_target_perc ,
2017-03-09 15:26:05 -05:00
passBucket . withinTarget , passBucket . totalConfirmed , passBucket . inMempool , passBucket . leftMempool ,
2017-01-24 16:30:03 -05:00
failBucket . start , failBucket . end ,
2020-08-15 01:57:59 +02:00
failed_within_target_perc ,
2017-03-09 15:26:05 -05:00
failBucket . withinTarget , failBucket . totalConfirmed , failBucket . inMempool , failBucket . leftMempool ) ;
2014-08-26 16:28:32 -04:00
2017-01-24 16:30:03 -05:00
if ( result ) {
result - > pass = passBucket ;
result - > fail = failBucket ;
result - > decay = decay ;
2017-04-03 11:31:27 -04:00
result - > scale = scale ;
2017-01-24 16:30:03 -05:00
}
2014-08-26 16:28:32 -04:00
return median ;
}
2017-02-15 15:48:48 -05:00
void TxConfirmStats : : Write ( CAutoFile & fileout ) const
2014-08-26 16:28:32 -04:00
{
fileout < < decay ;
2017-02-23 15:13:41 -05:00
fileout < < scale ;
2020-07-30 21:19:38 +02:00
fileout < < m_feerate_avg ;
2014-08-26 16:28:32 -04:00
fileout < < txCtAvg ;
fileout < < confAvg ;
2017-02-23 15:13:41 -05:00
fileout < < failAvg ;
2014-08-26 16:28:32 -04:00
}
2017-02-23 15:13:41 -05:00
void TxConfirmStats : : Read ( CAutoFile & filein , int nFileVersion , size_t numBuckets )
2014-08-26 16:28:32 -04:00
{
2017-02-23 15:13:41 -05:00
// Read data file and do some very basic sanity checking
// buckets and bucketMap are not updated yet, so don't access them
// If there is a read failure, we'll just discard this entire object anyway
2017-04-03 11:31:27 -04:00
size_t maxConfirms , maxPeriods ;
2017-02-23 15:13:41 -05:00
// The current version will store the decay with each individual TxConfirmStats and also keep a scale factor
2017-12-19 11:19:28 -05:00
filein > > decay ;
if ( decay < = 0 | | decay > = 1 ) {
throw std : : runtime_error ( " Corrupt estimates file. Decay must be between 0 and 1 (non-inclusive) " ) ;
}
filein > > scale ;
if ( scale = = 0 ) {
throw std : : runtime_error ( " Corrupt estimates file. Scale must be non-zero " ) ;
2017-02-23 15:13:41 -05:00
}
2020-07-30 21:19:38 +02:00
filein > > m_feerate_avg ;
if ( m_feerate_avg . size ( ) ! = numBuckets ) {
2016-03-21 13:04:40 -04:00
throw std : : runtime_error ( " Corrupt estimates file. Mismatch in feerate average bucket count " ) ;
2017-02-23 15:13:41 -05:00
}
filein > > txCtAvg ;
if ( txCtAvg . size ( ) ! = numBuckets ) {
2014-08-26 16:28:32 -04:00
throw std : : runtime_error ( " Corrupt estimates file. Mismatch in tx count bucket count " ) ;
}
2017-02-23 15:13:41 -05:00
filein > > confAvg ;
2017-04-03 11:31:27 -04:00
maxPeriods = confAvg . size ( ) ;
maxConfirms = scale * maxPeriods ;
2014-08-26 16:28:32 -04:00
2017-02-23 15:13:41 -05:00
if ( maxConfirms < = 0 | | maxConfirms > 6 * 24 * 7 ) { // one week
2014-08-26 16:28:32 -04:00
throw std : : runtime_error ( " Corrupt estimates file. Must maintain estimates for between 1 and 1008 (one week) confirms " ) ;
}
2017-04-03 11:31:27 -04:00
for ( unsigned int i = 0 ; i < maxPeriods ; i + + ) {
2017-02-23 15:13:41 -05:00
if ( confAvg [ i ] . size ( ) ! = numBuckets ) {
throw std : : runtime_error ( " Corrupt estimates file. Mismatch in feerate conf average bucket count " ) ;
}
2014-08-26 16:28:32 -04:00
}
2017-12-19 11:19:28 -05:00
filein > > failAvg ;
if ( maxPeriods ! = failAvg . size ( ) ) {
throw std : : runtime_error ( " Corrupt estimates file. Mismatch in confirms tracked for failures " ) ;
}
for ( unsigned int i = 0 ; i < maxPeriods ; i + + ) {
if ( failAvg [ i ] . size ( ) ! = numBuckets ) {
throw std : : runtime_error ( " Corrupt estimates file. Mismatch in one of failure average bucket counts " ) ;
2017-03-09 15:26:05 -05:00
}
2014-08-26 16:28:32 -04:00
}
2017-02-23 15:13:41 -05:00
// Resize the current block variables which aren't stored in the data file
// to match the number of confirms and buckets
resizeInMemoryCounters ( numBuckets ) ;
2014-08-26 16:28:32 -04:00
2016-12-25 20:19:40 +00:00
LogPrint ( BCLog : : ESTIMATEFEE , " Reading estimates: %u buckets counting confirms up to %u blocks \n " ,
2016-03-21 13:04:40 -04:00
numBuckets , maxConfirms ) ;
2014-08-26 16:28:32 -04:00
}
unsigned int TxConfirmStats : : NewTx ( unsigned int nBlockHeight , double val )
{
unsigned int bucketindex = bucketMap . lower_bound ( val ) - > second ;
unsigned int blockIndex = nBlockHeight % unconfTxs . size ( ) ;
unconfTxs [ blockIndex ] [ bucketindex ] + + ;
return bucketindex ;
}
2017-03-09 15:26:05 -05:00
void TxConfirmStats : : removeTx ( unsigned int entryHeight , unsigned int nBestSeenHeight , unsigned int bucketindex , bool inBlock )
2014-08-26 16:28:32 -04:00
{
//nBestSeenHeight is not updated yet for the new block
int blocksAgo = nBestSeenHeight - entryHeight ;
if ( nBestSeenHeight = = 0 ) // the BlockPolicyEstimator hasn't seen any blocks yet
blocksAgo = 0 ;
if ( blocksAgo < 0 ) {
2016-12-25 20:19:40 +00:00
LogPrint ( BCLog : : ESTIMATEFEE , " Blockpolicy error, blocks ago is negative for mempool tx \n " ) ;
2015-08-09 00:17:27 +01:00
return ; //This can't happen because we call this with our best seen height, no entries can have higher
2014-08-26 16:28:32 -04:00
}
if ( blocksAgo > = ( int ) unconfTxs . size ( ) ) {
2016-12-25 20:19:40 +00:00
if ( oldUnconfTxs [ bucketindex ] > 0 ) {
2014-08-26 16:28:32 -04:00
oldUnconfTxs [ bucketindex ] - - ;
2016-12-25 20:19:40 +00:00
} else {
LogPrint ( BCLog : : ESTIMATEFEE , " Blockpolicy error, mempool tx removed from >25 blocks,bucketIndex=%u already \n " ,
2014-08-26 16:28:32 -04:00
bucketindex ) ;
2016-12-25 20:19:40 +00:00
}
2014-08-26 16:28:32 -04:00
}
else {
unsigned int blockIndex = entryHeight % unconfTxs . size ( ) ;
2016-12-25 20:19:40 +00:00
if ( unconfTxs [ blockIndex ] [ bucketindex ] > 0 ) {
2014-08-26 16:28:32 -04:00
unconfTxs [ blockIndex ] [ bucketindex ] - - ;
2016-12-25 20:19:40 +00:00
} else {
LogPrint ( BCLog : : ESTIMATEFEE , " Blockpolicy error, mempool tx removed from blockIndex=%u,bucketIndex=%u already \n " ,
2014-08-26 16:28:32 -04:00
blockIndex , bucketindex ) ;
2016-12-25 20:19:40 +00:00
}
2014-08-26 16:28:32 -04:00
}
2017-04-03 11:31:27 -04:00
if ( ! inBlock & & ( unsigned int ) blocksAgo > = scale ) { // Only counts as a failure if not confirmed for entire period
2017-08-28 09:20:50 +02:00
assert ( scale ! = 0 ) ;
2017-04-03 11:31:27 -04:00
unsigned int periodsAgo = blocksAgo / scale ;
for ( size_t i = 0 ; i < periodsAgo & & i < failAvg . size ( ) ; i + + ) {
2017-03-09 15:26:05 -05:00
failAvg [ i ] [ bucketindex ] + + ;
}
}
2014-08-26 16:28:32 -04:00
}
2016-11-29 13:55:26 -05:00
// This function is called from CTxMemPool::removeUnchecked to ensure
// txs removed from the mempool for any reason are no longer
// tracked. Txs that were part of a block have already been removed in
// processBlockTx to ensure they are never double tracked, but it is
// of no harm to try to remove them again.
2017-03-09 15:26:05 -05:00
bool CBlockPolicyEstimator : : removeTx ( uint256 hash , bool inBlock )
2014-08-26 16:28:32 -04:00
{
2018-10-08 15:18:43 +02:00
LOCK ( m_cs_fee_estimator ) ;
2014-08-26 16:28:32 -04:00
std : : map < uint256 , TxStatsInfo > : : iterator pos = mapMemPoolTxs . find ( hash ) ;
2016-11-10 14:16:42 -05:00
if ( pos ! = mapMemPoolTxs . end ( ) ) {
2017-03-09 15:26:05 -05:00
feeStats - > removeTx ( pos - > second . blockHeight , nBestSeenHeight , pos - > second . bucketIndex , inBlock ) ;
shortStats - > removeTx ( pos - > second . blockHeight , nBestSeenHeight , pos - > second . bucketIndex , inBlock ) ;
longStats - > removeTx ( pos - > second . blockHeight , nBestSeenHeight , pos - > second . bucketIndex , inBlock ) ;
2016-11-10 14:16:42 -05:00
mapMemPoolTxs . erase ( hash ) ;
return true ;
} else {
return false ;
2014-08-26 16:28:32 -04:00
}
}
2017-01-13 16:25:15 -05:00
CBlockPolicyEstimator : : CBlockPolicyEstimator ( )
2017-03-10 16:57:40 -05:00
: nBestSeenHeight ( 0 ) , firstRecordedHeight ( 0 ) , historicalFirst ( 0 ) , historicalBest ( 0 ) , trackedTxs ( 0 ) , untrackedTxs ( 0 )
2014-08-26 16:28:32 -04:00
{
2017-01-13 16:19:25 -05:00
static_assert ( MIN_BUCKET_FEERATE > 0 , " Min feerate must be nonzero " ) ;
2017-02-23 15:13:41 -05:00
size_t bucketIndex = 0 ;
2020-09-10 14:27:36 +02:00
2017-04-12 12:05:10 -04:00
for ( double bucketBoundary = MIN_BUCKET_FEERATE ; bucketBoundary < = MAX_BUCKET_FEERATE ; bucketBoundary * = FEE_SPACING , bucketIndex + + ) {
2017-02-23 15:13:41 -05:00
buckets . push_back ( bucketBoundary ) ;
bucketMap [ bucketBoundary ] = bucketIndex ;
2014-08-26 16:28:32 -04:00
}
2017-02-23 15:13:41 -05:00
buckets . push_back ( INF_FEERATE ) ;
bucketMap [ INF_FEERATE ] = bucketIndex ;
assert ( bucketMap . size ( ) = = buckets . size ( ) ) ;
2017-08-09 15:35:43 +02:00
feeStats = std : : unique_ptr < TxConfirmStats > ( new TxConfirmStats ( buckets , bucketMap , MED_BLOCK_PERIODS , MED_DECAY , MED_SCALE ) ) ;
shortStats = std : : unique_ptr < TxConfirmStats > ( new TxConfirmStats ( buckets , bucketMap , SHORT_BLOCK_PERIODS , SHORT_DECAY , SHORT_SCALE ) ) ;
longStats = std : : unique_ptr < TxConfirmStats > ( new TxConfirmStats ( buckets , bucketMap , LONG_BLOCK_PERIODS , LONG_DECAY , LONG_SCALE ) ) ;
2020-09-10 14:27:36 +02:00
// If the fee estimation file is present, read recorded estimations
fs : : path est_filepath = GetDataDir ( ) / FEE_ESTIMATES_FILENAME ;
CAutoFile est_file ( fsbridge : : fopen ( est_filepath , " rb " ) , SER_DISK , CLIENT_VERSION ) ;
if ( est_file . IsNull ( ) | | ! Read ( est_file ) ) {
LogPrintf ( " Failed to read fee estimates from %s. Continue anyway. \n " , est_filepath . string ( ) ) ;
}
2017-02-16 16:23:15 -05:00
}
CBlockPolicyEstimator : : ~ CBlockPolicyEstimator ( )
{
2014-08-26 16:28:32 -04:00
}
2016-11-11 12:48:01 -05:00
void CBlockPolicyEstimator : : processTransaction ( const CTxMemPoolEntry & entry , bool validFeeEstimate )
2014-08-26 16:28:32 -04:00
{
2018-10-08 15:18:43 +02:00
LOCK ( m_cs_fee_estimator ) ;
2014-08-26 16:28:32 -04:00
unsigned int txHeight = entry . GetHeight ( ) ;
uint256 hash = entry . GetTx ( ) . GetHash ( ) ;
2016-03-21 13:04:40 -04:00
if ( mapMemPoolTxs . count ( hash ) ) {
2016-12-25 20:19:40 +00:00
LogPrint ( BCLog : : ESTIMATEFEE , " Blockpolicy error mempool tx %s already being tracked \n " ,
2019-10-28 13:30:20 +01:00
hash . ToString ( ) ) ;
2017-12-19 11:19:28 -05:00
return ;
2014-08-26 16:28:32 -04:00
}
2016-11-29 15:40:03 -05:00
if ( txHeight ! = nBestSeenHeight ) {
2014-08-26 16:28:32 -04:00
// Ignore side chains and re-orgs; assuming they are random they don't
// affect the estimate. We'll potentially double count transactions in 1-block reorgs.
2019-03-27 11:14:25 -04:00
// Ignore txs if BlockPolicyEstimator is not in sync with ::ChainActive().Tip().
2016-11-29 15:40:03 -05:00
// It will be synced next time a block is processed.
2014-08-26 16:28:32 -04:00
return ;
}
// Only want to be updating estimates when our blockchain is synced,
// otherwise we'll miscalculate how many blocks its taking to get included.
2016-11-11 13:40:27 -05:00
if ( ! validFeeEstimate ) {
untrackedTxs + + ;
2014-08-26 16:28:32 -04:00
return ;
2016-11-11 13:40:27 -05:00
}
trackedTxs + + ;
2014-08-26 16:28:32 -04:00
2016-03-21 13:04:40 -04:00
// Feerates are stored and reported as BTC-per-kb:
2014-08-26 16:28:32 -04:00
CFeeRate feeRate ( entry . GetFee ( ) , entry . GetTxSize ( ) ) ;
mapMemPoolTxs [ hash ] . blockHeight = txHeight ;
2017-03-09 14:02:00 -05:00
unsigned int bucketIndex = feeStats - > NewTx ( txHeight , ( double ) feeRate . GetFeePerK ( ) ) ;
mapMemPoolTxs [ hash ] . bucketIndex = bucketIndex ;
unsigned int bucketIndex2 = shortStats - > NewTx ( txHeight , ( double ) feeRate . GetFeePerK ( ) ) ;
assert ( bucketIndex = = bucketIndex2 ) ;
unsigned int bucketIndex3 = longStats - > NewTx ( txHeight , ( double ) feeRate . GetFeePerK ( ) ) ;
assert ( bucketIndex = = bucketIndex3 ) ;
2014-08-26 16:28:32 -04:00
}
2016-11-11 13:40:27 -05:00
bool CBlockPolicyEstimator : : processBlockTx ( unsigned int nBlockHeight , const CTxMemPoolEntry * entry )
2014-08-26 16:28:32 -04:00
{
2017-03-09 15:26:05 -05:00
if ( ! removeTx ( entry - > GetTx ( ) . GetHash ( ) , true ) ) {
2016-11-11 11:57:51 -05:00
// This transaction wasn't being tracked for fee estimation
2016-11-11 13:40:27 -05:00
return false ;
2014-08-26 16:28:32 -04:00
}
// How many blocks did it take for miners to include this transaction?
// blocksToConfirm is 1-based, so a transaction included in the earliest
// possible block has confirmation count of 1
2016-11-11 14:16:42 -05:00
int blocksToConfirm = nBlockHeight - entry - > GetHeight ( ) ;
2014-08-26 16:28:32 -04:00
if ( blocksToConfirm < = 0 ) {
// This can't happen because we don't process transactions from a block with a height
// lower than our greatest seen height
2016-12-25 20:19:40 +00:00
LogPrint ( BCLog : : ESTIMATEFEE , " Blockpolicy error Transaction had negative blocksToConfirm \n " ) ;
2016-11-11 13:40:27 -05:00
return false ;
2014-08-26 16:28:32 -04:00
}
2016-03-21 13:04:40 -04:00
// Feerates are stored and reported as BTC-per-kb:
2016-11-11 14:16:42 -05:00
CFeeRate feeRate ( entry - > GetFee ( ) , entry - > GetTxSize ( ) ) ;
2014-08-26 16:28:32 -04:00
2017-02-16 16:23:15 -05:00
feeStats - > Record ( blocksToConfirm , ( double ) feeRate . GetFeePerK ( ) ) ;
2017-03-09 14:02:00 -05:00
shortStats - > Record ( blocksToConfirm , ( double ) feeRate . GetFeePerK ( ) ) ;
longStats - > Record ( blocksToConfirm , ( double ) feeRate . GetFeePerK ( ) ) ;
2016-11-11 13:40:27 -05:00
return true ;
2014-08-26 16:28:32 -04:00
}
void CBlockPolicyEstimator : : processBlock ( unsigned int nBlockHeight ,
2016-11-11 14:16:42 -05:00
std : : vector < const CTxMemPoolEntry * > & entries )
2014-08-26 16:28:32 -04:00
{
2018-10-08 15:18:43 +02:00
LOCK ( m_cs_fee_estimator ) ;
2014-08-26 16:28:32 -04:00
if ( nBlockHeight < = nBestSeenHeight ) {
// Ignore side chains and re-orgs; assuming they are random
// they don't affect the estimate.
// And if an attacker can re-org the chain at will, then
// you've got much bigger problems than "attacker can influence
// transaction fees."
return ;
}
2016-11-11 11:57:51 -05:00
// Must update nBestSeenHeight in sync with ClearCurrent so that
// calls to removeTx (via processBlockTx) correctly calculate age
// of unconfirmed txs to remove from tracking.
nBestSeenHeight = nBlockHeight ;
2017-02-28 17:29:42 -05:00
// Update unconfirmed circular buffer
2017-02-16 16:23:15 -05:00
feeStats - > ClearCurrent ( nBlockHeight ) ;
2017-03-09 14:02:00 -05:00
shortStats - > ClearCurrent ( nBlockHeight ) ;
longStats - > ClearCurrent ( nBlockHeight ) ;
2014-08-26 16:28:32 -04:00
2017-02-28 17:29:42 -05:00
// Decay all exponential averages
feeStats - > UpdateMovingAverages ( ) ;
shortStats - > UpdateMovingAverages ( ) ;
longStats - > UpdateMovingAverages ( ) ;
2014-08-26 16:28:32 -04:00
2016-11-11 13:40:27 -05:00
unsigned int countedTxs = 0 ;
2017-02-28 17:29:42 -05:00
// Update averages with data points from current block
2017-05-18 09:42:14 +02:00
for ( const auto & entry : entries ) {
if ( processBlockTx ( nBlockHeight , entry ) )
2016-11-11 13:40:27 -05:00
countedTxs + + ;
}
2014-08-26 16:28:32 -04:00
2017-03-07 15:01:50 -05:00
if ( firstRecordedHeight = = 0 & & countedTxs > 0 ) {
firstRecordedHeight = nBestSeenHeight ;
LogPrint ( BCLog : : ESTIMATEFEE , " Blockpolicy first recorded height %u \n " , firstRecordedHeight ) ;
}
2014-08-26 16:28:32 -04:00
2017-03-10 16:57:40 -05:00
LogPrint ( BCLog : : ESTIMATEFEE , " Blockpolicy estimates updated by %u of %u block txs, since last block %u of %u tracked, mempool map size %u, max target %u from %s \n " ,
countedTxs , entries . size ( ) , trackedTxs , trackedTxs + untrackedTxs , mapMemPoolTxs . size ( ) ,
MaxUsableEstimate ( ) , HistoricalBlockSpan ( ) > BlockSpan ( ) ? " historical " : " current " ) ;
2016-11-11 13:40:27 -05:00
trackedTxs = 0 ;
untrackedTxs = 0 ;
2014-08-26 16:28:32 -04:00
}
2017-02-15 15:23:34 -05:00
CFeeRate CBlockPolicyEstimator : : estimateFee ( int confTarget ) const
2014-08-26 16:28:32 -04:00
{
2017-01-24 16:30:03 -05:00
// It's not possible to get reasonable estimates for confTarget of 1
if ( confTarget < = 1 )
return CFeeRate ( 0 ) ;
return estimateRawFee ( confTarget , DOUBLE_SUCCESS_PCT , FeeEstimateHorizon : : MED_HALFLIFE ) ;
}
CFeeRate CBlockPolicyEstimator : : estimateRawFee ( int confTarget , double successThreshold , FeeEstimateHorizon horizon , EstimationResult * result ) const
{
2020-12-26 13:00:48 +01:00
TxConfirmStats * stats = nullptr ;
2017-01-24 16:30:03 -05:00
double sufficientTxs = SUFFICIENT_FEETXS ;
switch ( horizon ) {
case FeeEstimateHorizon : : SHORT_HALFLIFE : {
2017-08-09 15:35:43 +02:00
stats = shortStats . get ( ) ;
2017-01-24 16:30:03 -05:00
sufficientTxs = SUFFICIENT_TXS_SHORT ;
break ;
}
case FeeEstimateHorizon : : MED_HALFLIFE : {
2017-08-09 15:35:43 +02:00
stats = feeStats . get ( ) ;
2017-01-24 16:30:03 -05:00
break ;
}
case FeeEstimateHorizon : : LONG_HALFLIFE : {
2017-08-09 15:35:43 +02:00
stats = longStats . get ( ) ;
2017-01-24 16:30:03 -05:00
break ;
}
2020-12-26 13:00:48 +01:00
} // no default case, so the compiler can warn about missing cases
assert ( stats ) ;
2017-01-24 16:30:03 -05:00
2018-10-08 15:18:43 +02:00
LOCK ( m_cs_fee_estimator ) ;
2014-08-26 16:28:32 -04:00
// Return failure if trying to analyze a target we're not tracking
2017-01-24 16:30:03 -05:00
if ( confTarget < = 0 | | ( unsigned int ) confTarget > stats - > GetMaxConfirms ( ) )
return CFeeRate ( 0 ) ;
if ( successThreshold > 1 )
2014-08-26 16:28:32 -04:00
return CFeeRate ( 0 ) ;
2020-07-30 18:48:57 +02:00
double median = stats - > EstimateMedianVal ( confTarget , sufficientTxs , successThreshold , nBestSeenHeight , result ) ;
2014-08-26 16:28:32 -04:00
if ( median < 0 )
return CFeeRate ( 0 ) ;
2017-09-11 15:47:09 -04:00
return CFeeRate ( llround ( median ) ) ;
2014-08-26 16:28:32 -04:00
}
2017-06-28 10:03:00 -04:00
unsigned int CBlockPolicyEstimator : : HighestTargetTracked ( FeeEstimateHorizon horizon ) const
{
2018-12-02 21:28:53 +01:00
LOCK ( m_cs_fee_estimator ) ;
2017-06-28 10:03:00 -04:00
switch ( horizon ) {
case FeeEstimateHorizon : : SHORT_HALFLIFE : {
return shortStats - > GetMaxConfirms ( ) ;
}
case FeeEstimateHorizon : : MED_HALFLIFE : {
return feeStats - > GetMaxConfirms ( ) ;
}
case FeeEstimateHorizon : : LONG_HALFLIFE : {
return longStats - > GetMaxConfirms ( ) ;
}
2020-12-26 13:00:48 +01:00
} // no default case, so the compiler can warn about missing cases
assert ( false ) ;
2017-06-28 10:03:00 -04:00
}
2017-03-10 16:57:40 -05:00
unsigned int CBlockPolicyEstimator : : BlockSpan ( ) const
{
if ( firstRecordedHeight = = 0 ) return 0 ;
assert ( nBestSeenHeight > = firstRecordedHeight ) ;
return nBestSeenHeight - firstRecordedHeight ;
}
unsigned int CBlockPolicyEstimator : : HistoricalBlockSpan ( ) const
{
if ( historicalFirst = = 0 ) return 0 ;
assert ( historicalBest > = historicalFirst ) ;
if ( nBestSeenHeight - historicalBest > OLDEST_ESTIMATE_HISTORY ) return 0 ;
return historicalBest - historicalFirst ;
}
unsigned int CBlockPolicyEstimator : : MaxUsableEstimate ( ) const
{
// Block spans are divided by 2 to make sure there are enough potential failing data points for the estimate
return std : : min ( longStats - > GetMaxConfirms ( ) , std : : max ( BlockSpan ( ) , HistoricalBlockSpan ( ) ) / 2 ) ;
}
2017-03-07 11:33:44 -05:00
/** Return a fee estimate at the required successThreshold from the shortest
* time horizon which tracks confirmations up to the desired target . If
* checkShorterHorizon is requested , also allow short time horizon estimates
* for a lower target to reduce the given answer */
2017-04-25 15:39:32 -04:00
double CBlockPolicyEstimator : : estimateCombinedFee ( unsigned int confTarget , double successThreshold , bool checkShorterHorizon , EstimationResult * result ) const
2017-03-07 11:33:44 -05:00
{
double estimate = - 1 ;
if ( confTarget > = 1 & & confTarget < = longStats - > GetMaxConfirms ( ) ) {
// Find estimate from shortest time horizon possible
if ( confTarget < = shortStats - > GetMaxConfirms ( ) ) { // short horizon
2020-07-30 18:48:57 +02:00
estimate = shortStats - > EstimateMedianVal ( confTarget , SUFFICIENT_TXS_SHORT , successThreshold , nBestSeenHeight , result ) ;
2017-03-07 11:33:44 -05:00
}
else if ( confTarget < = feeStats - > GetMaxConfirms ( ) ) { // medium horizon
2020-07-30 18:48:57 +02:00
estimate = feeStats - > EstimateMedianVal ( confTarget , SUFFICIENT_FEETXS , successThreshold , nBestSeenHeight , result ) ;
2017-03-07 11:33:44 -05:00
}
else { // long horizon
2020-07-30 18:48:57 +02:00
estimate = longStats - > EstimateMedianVal ( confTarget , SUFFICIENT_FEETXS , successThreshold , nBestSeenHeight , result ) ;
2017-03-07 11:33:44 -05:00
}
if ( checkShorterHorizon ) {
2017-04-25 15:39:32 -04:00
EstimationResult tempResult ;
2017-03-07 11:33:44 -05:00
// If a lower confTarget from a more recent horizon returns a lower answer use it.
if ( confTarget > feeStats - > GetMaxConfirms ( ) ) {
2020-07-30 18:48:57 +02:00
double medMax = feeStats - > EstimateMedianVal ( feeStats - > GetMaxConfirms ( ) , SUFFICIENT_FEETXS , successThreshold , nBestSeenHeight , & tempResult ) ;
2017-04-25 15:39:32 -04:00
if ( medMax > 0 & & ( estimate = = - 1 | | medMax < estimate ) ) {
2017-03-07 11:33:44 -05:00
estimate = medMax ;
2017-04-25 15:39:32 -04:00
if ( result ) * result = tempResult ;
}
2017-03-07 11:33:44 -05:00
}
if ( confTarget > shortStats - > GetMaxConfirms ( ) ) {
2020-07-30 18:48:57 +02:00
double shortMax = shortStats - > EstimateMedianVal ( shortStats - > GetMaxConfirms ( ) , SUFFICIENT_TXS_SHORT , successThreshold , nBestSeenHeight , & tempResult ) ;
2017-04-25 15:39:32 -04:00
if ( shortMax > 0 & & ( estimate = = - 1 | | shortMax < estimate ) ) {
2017-03-07 11:33:44 -05:00
estimate = shortMax ;
2017-04-25 15:39:32 -04:00
if ( result ) * result = tempResult ;
}
2017-03-07 11:33:44 -05:00
}
}
}
return estimate ;
}
2017-04-12 12:29:03 -04:00
/** Ensure that for a conservative estimate, the DOUBLE_SUCCESS_PCT is also met
* at 2 * target for any longer time horizons .
*/
2017-04-25 15:39:32 -04:00
double CBlockPolicyEstimator : : estimateConservativeFee ( unsigned int doubleTarget , EstimationResult * result ) const
2017-03-07 11:33:44 -05:00
{
double estimate = - 1 ;
2017-04-25 15:39:32 -04:00
EstimationResult tempResult ;
2017-03-07 11:33:44 -05:00
if ( doubleTarget < = shortStats - > GetMaxConfirms ( ) ) {
2020-07-30 18:48:57 +02:00
estimate = feeStats - > EstimateMedianVal ( doubleTarget , SUFFICIENT_FEETXS , DOUBLE_SUCCESS_PCT , nBestSeenHeight , result ) ;
2017-03-07 11:33:44 -05:00
}
if ( doubleTarget < = feeStats - > GetMaxConfirms ( ) ) {
2020-07-30 18:48:57 +02:00
double longEstimate = longStats - > EstimateMedianVal ( doubleTarget , SUFFICIENT_FEETXS , DOUBLE_SUCCESS_PCT , nBestSeenHeight , & tempResult ) ;
2017-03-07 11:33:44 -05:00
if ( longEstimate > estimate ) {
estimate = longEstimate ;
2017-04-25 15:39:32 -04:00
if ( result ) * result = tempResult ;
2017-03-07 11:33:44 -05:00
}
}
return estimate ;
}
2017-04-12 12:29:03 -04:00
/** estimateSmartFee returns the max of the feerates calculated with a 60%
* threshold required at target / 2 , an 85 % threshold required at target and a
* 95 % threshold required at 2 * target . Each calculation is performed at the
* shortest time horizon which tracks the required target . Conservative
* estimates , however , required the 95 % threshold at 2 * target be met for any
* longer time horizons also .
*/
2017-06-29 13:13:23 -04:00
CFeeRate CBlockPolicyEstimator : : estimateSmartFee ( int confTarget , FeeCalculation * feeCalc , bool conservative ) const
2015-11-16 15:10:22 -05:00
{
2018-10-08 15:18:43 +02:00
LOCK ( m_cs_fee_estimator ) ;
2017-06-29 13:13:23 -04:00
2017-04-25 15:39:32 -04:00
if ( feeCalc ) {
feeCalc - > desiredTarget = confTarget ;
feeCalc - > returnedTarget = confTarget ;
}
2016-11-29 12:18:44 -05:00
2015-11-16 15:10:22 -05:00
double median = - 1 ;
2017-04-25 15:39:32 -04:00
EstimationResult tempResult ;
2017-02-15 09:24:11 -05:00
2017-06-29 13:13:23 -04:00
// Return failure if trying to analyze a target we're not tracking
2017-06-29 13:57:33 -04:00
if ( confTarget < = 0 | | ( unsigned int ) confTarget > longStats - > GetMaxConfirms ( ) ) {
2017-07-18 14:02:42 +02:00
return CFeeRate ( 0 ) ; // error condition
2017-06-29 13:57:33 -04:00
}
2017-02-15 09:24:11 -05:00
2017-06-29 13:13:23 -04:00
// It's not possible to get reasonable estimates for confTarget of 1
2017-06-29 13:57:33 -04:00
if ( confTarget = = 1 ) confTarget = 2 ;
2017-02-15 09:24:11 -05:00
2017-06-29 13:13:23 -04:00
unsigned int maxUsableEstimate = MaxUsableEstimate ( ) ;
if ( ( unsigned int ) confTarget > maxUsableEstimate ) {
confTarget = maxUsableEstimate ;
}
2017-06-29 13:57:33 -04:00
if ( feeCalc ) feeCalc - > returnedTarget = confTarget ;
2017-07-18 14:02:42 +02:00
if ( confTarget < = 1 ) return CFeeRate ( 0 ) ; // error condition
2017-03-10 16:57:40 -05:00
2017-06-29 13:13:23 -04:00
assert ( confTarget > 0 ) ; //estimateCombinedFee and estimateConservativeFee take unsigned ints
/** true is passed to estimateCombined fee for target/2 and target so
* that we check the max confirms for shorter time horizons as well .
* This is necessary to preserve monotonically increasing estimates .
* For non - conservative estimates we do the same thing for 2 * target , but
* for conservative estimates we want to skip these shorter horizons
* checks for 2 * target because we are taking the max over all time
* horizons so we already have monotonically increasing estimates and
* the purpose of conservative estimates is not to let short term
* fluctuations lower our estimates by too much .
*/
double halfEst = estimateCombinedFee ( confTarget / 2 , HALF_SUCCESS_PCT , true , & tempResult ) ;
if ( feeCalc ) {
feeCalc - > est = tempResult ;
feeCalc - > reason = FeeReason : : HALF_ESTIMATE ;
}
median = halfEst ;
double actualEst = estimateCombinedFee ( confTarget , SUCCESS_PCT , true , & tempResult ) ;
if ( actualEst > median ) {
median = actualEst ;
2017-04-25 15:39:32 -04:00
if ( feeCalc ) {
feeCalc - > est = tempResult ;
2017-06-29 13:13:23 -04:00
feeCalc - > reason = FeeReason : : FULL_ESTIMATE ;
2017-04-25 15:39:32 -04:00
}
2017-06-29 13:13:23 -04:00
}
double doubleEst = estimateCombinedFee ( 2 * confTarget , DOUBLE_SUCCESS_PCT , ! conservative , & tempResult ) ;
if ( doubleEst > median ) {
median = doubleEst ;
if ( feeCalc ) {
feeCalc - > est = tempResult ;
feeCalc - > reason = FeeReason : : DOUBLE_ESTIMATE ;
2017-03-07 11:33:44 -05:00
}
2017-06-29 13:13:23 -04:00
}
if ( conservative | | median = = - 1 ) {
double consEst = estimateConservativeFee ( 2 * confTarget , & tempResult ) ;
if ( consEst > median ) {
median = consEst ;
2017-04-25 15:39:32 -04:00
if ( feeCalc ) {
feeCalc - > est = tempResult ;
2017-06-29 13:13:23 -04:00
feeCalc - > reason = FeeReason : : CONSERVATIVE ;
2017-04-25 15:39:32 -04:00
}
2017-03-07 11:33:44 -05:00
}
2017-06-29 13:13:23 -04:00
}
2015-11-16 15:10:22 -05:00
2017-07-18 14:02:42 +02:00
if ( median < 0 ) return CFeeRate ( 0 ) ; // error condition
2015-11-16 15:10:22 -05:00
2017-09-11 15:47:09 -04:00
return CFeeRate ( llround ( median ) ) ;
2015-11-16 15:10:22 -05:00
}
2020-09-10 14:27:36 +02:00
void CBlockPolicyEstimator : : Flush ( ) {
FlushUnconfirmed ( ) ;
fs : : path est_filepath = GetDataDir ( ) / FEE_ESTIMATES_FILENAME ;
CAutoFile est_file ( fsbridge : : fopen ( est_filepath , " wb " ) , SER_DISK , CLIENT_VERSION ) ;
if ( est_file . IsNull ( ) | | ! Write ( est_file ) ) {
2020-12-07 14:01:19 +01:00
LogPrintf ( " Failed to write fee estimates to %s. Continue anyway. \n " , est_filepath . string ( ) ) ;
2020-09-10 14:27:36 +02:00
}
}
2017-03-07 11:33:44 -05:00
2017-02-15 15:48:48 -05:00
bool CBlockPolicyEstimator : : Write ( CAutoFile & fileout ) const
2014-08-26 16:28:32 -04:00
{
2017-02-15 15:48:48 -05:00
try {
2018-10-08 15:18:43 +02:00
LOCK ( m_cs_fee_estimator ) ;
2017-02-23 15:13:41 -05:00
fileout < < 149900 ; // version required to read: 0.14.99 or later
2017-02-15 15:48:48 -05:00
fileout < < CLIENT_VERSION ; // version that wrote the file
fileout < < nBestSeenHeight ;
2017-03-10 16:57:40 -05:00
if ( BlockSpan ( ) > HistoricalBlockSpan ( ) / 2 ) {
fileout < < firstRecordedHeight < < nBestSeenHeight ;
}
else {
fileout < < historicalFirst < < historicalBest ;
}
2017-02-23 15:13:41 -05:00
fileout < < buckets ;
2017-02-16 16:23:15 -05:00
feeStats - > Write ( fileout ) ;
2017-02-23 15:13:41 -05:00
shortStats - > Write ( fileout ) ;
longStats - > Write ( fileout ) ;
2017-02-15 15:48:48 -05:00
}
catch ( const std : : exception & ) {
2017-04-23 10:36:26 -07:00
LogPrintf ( " CBlockPolicyEstimator::Write(): unable to write policy estimator data (non-fatal) \n " ) ;
2017-02-15 15:48:48 -05:00
return false ;
}
return true ;
2014-08-26 16:28:32 -04:00
}
2017-02-15 15:48:48 -05:00
bool CBlockPolicyEstimator : : Read ( CAutoFile & filein )
2014-08-26 16:28:32 -04:00
{
2017-02-15 15:48:48 -05:00
try {
2018-10-08 15:18:43 +02:00
LOCK ( m_cs_fee_estimator ) ;
2017-02-23 15:13:41 -05:00
int nVersionRequired , nVersionThatWrote ;
2017-02-15 15:48:48 -05:00
filein > > nVersionRequired > > nVersionThatWrote ;
2020-12-07 14:06:28 +01:00
if ( nVersionRequired > CLIENT_VERSION ) {
throw std : : runtime_error ( strprintf ( " up-version (%d) fee estimate file " , nVersionRequired ) ) ;
}
2017-02-23 15:13:41 -05:00
// Read fee estimates file into temporary variables so existing data
// structures aren't corrupted if there is an exception.
2017-06-04 22:45:22 +02:00
unsigned int nFileBestSeenHeight ;
2017-02-15 15:48:48 -05:00
filein > > nFileBestSeenHeight ;
2017-02-23 15:13:41 -05:00
2017-09-06 16:37:09 -07:00
if ( nVersionRequired < 149900 ) {
LogPrintf ( " %s: incompatible old fee estimation data (non-fatal). Version: %d \n " , __func__ , nVersionRequired ) ;
} else { // New format introduced in 149900
2017-06-04 22:45:22 +02:00
unsigned int nFileHistoricalFirst , nFileHistoricalBest ;
2017-03-10 16:57:40 -05:00
filein > > nFileHistoricalFirst > > nFileHistoricalBest ;
if ( nFileHistoricalFirst > nFileHistoricalBest | | nFileHistoricalBest > nFileBestSeenHeight ) {
throw std : : runtime_error ( " Corrupt estimates file. Historical block range for estimates is invalid " ) ;
}
2017-02-23 15:13:41 -05:00
std : : vector < double > fileBuckets ;
filein > > fileBuckets ;
size_t numBuckets = fileBuckets . size ( ) ;
2020-12-07 14:06:28 +01:00
if ( numBuckets < = 1 | | numBuckets > 1000 ) {
2017-02-23 15:13:41 -05:00
throw std : : runtime_error ( " Corrupt estimates file. Must have between 2 and 1000 feerate buckets " ) ;
2020-12-07 14:06:28 +01:00
}
2017-02-23 15:13:41 -05:00
2017-04-03 11:31:27 -04:00
std : : unique_ptr < TxConfirmStats > fileFeeStats ( new TxConfirmStats ( buckets , bucketMap , MED_BLOCK_PERIODS , MED_DECAY , MED_SCALE ) ) ;
std : : unique_ptr < TxConfirmStats > fileShortStats ( new TxConfirmStats ( buckets , bucketMap , SHORT_BLOCK_PERIODS , SHORT_DECAY , SHORT_SCALE ) ) ;
std : : unique_ptr < TxConfirmStats > fileLongStats ( new TxConfirmStats ( buckets , bucketMap , LONG_BLOCK_PERIODS , LONG_DECAY , LONG_SCALE ) ) ;
2017-02-23 15:13:41 -05:00
fileFeeStats - > Read ( filein , nVersionThatWrote , numBuckets ) ;
fileShortStats - > Read ( filein , nVersionThatWrote , numBuckets ) ;
fileLongStats - > Read ( filein , nVersionThatWrote , numBuckets ) ;
// Fee estimates file parsed correctly
// Copy buckets from file and refresh our bucketmap
buckets = fileBuckets ;
bucketMap . clear ( ) ;
for ( unsigned int i = 0 ; i < buckets . size ( ) ; i + + ) {
bucketMap [ buckets [ i ] ] = i ;
}
// Destroy old TxConfirmStats and point to new ones that already reference buckets and bucketMap
2017-08-09 15:35:43 +02:00
feeStats = std : : move ( fileFeeStats ) ;
shortStats = std : : move ( fileShortStats ) ;
longStats = std : : move ( fileLongStats ) ;
2017-02-23 15:13:41 -05:00
nBestSeenHeight = nFileBestSeenHeight ;
2017-03-10 16:57:40 -05:00
historicalFirst = nFileHistoricalFirst ;
historicalBest = nFileHistoricalBest ;
2017-02-23 15:13:41 -05:00
}
2017-02-15 15:48:48 -05:00
}
2017-02-23 15:13:41 -05:00
catch ( const std : : exception & e ) {
LogPrintf ( " CBlockPolicyEstimator::Read(): unable to read policy estimator data (non-fatal): %s \n " , e . what ( ) ) ;
2017-02-15 15:48:48 -05:00
return false ;
}
return true ;
2014-08-26 16:28:32 -04:00
}
2016-02-12 15:57:15 -05:00
2018-01-19 10:38:58 -05:00
void CBlockPolicyEstimator : : FlushUnconfirmed ( ) {
2017-03-09 15:26:05 -05:00
int64_t startclear = GetTimeMicros ( ) ;
2018-10-08 15:18:43 +02:00
LOCK ( m_cs_fee_estimator ) ;
2018-01-19 10:38:58 -05:00
size_t num_entries = mapMemPoolTxs . size ( ) ;
// Remove every entry in mapMemPoolTxs
while ( ! mapMemPoolTxs . empty ( ) ) {
auto mi = mapMemPoolTxs . begin ( ) ;
removeTx ( mi - > first , false ) ; // this calls erase() on mapMemPoolTxs
2017-03-09 15:26:05 -05:00
}
int64_t endclear = GetTimeMicros ( ) ;
2018-01-19 10:38:58 -05:00
LogPrint ( BCLog : : ESTIMATEFEE , " Recorded %u unconfirmed txs from mempool in %gs \n " , num_entries , ( endclear - startclear ) * 0.000001 ) ;
2017-03-09 15:26:05 -05:00
}
2016-02-12 15:57:15 -05:00
FeeFilterRounder : : FeeFilterRounder ( const CFeeRate & minIncrementalFee )
{
2016-12-05 21:46:08 -05:00
CAmount minFeeLimit = std : : max ( CAmount ( 1 ) , minIncrementalFee . GetFeePerK ( ) / 2 ) ;
2016-02-12 15:57:15 -05:00
feeset . insert ( 0 ) ;
2017-03-09 14:02:00 -05:00
for ( double bucketBoundary = minFeeLimit ; bucketBoundary < = MAX_FILTER_FEERATE ; bucketBoundary * = FEE_FILTER_SPACING ) {
2016-02-12 15:57:15 -05:00
feeset . insert ( bucketBoundary ) ;
}
}
CAmount FeeFilterRounder : : round ( CAmount currentMinFee )
{
std : : set < double > : : iterator it = feeset . lower_bound ( currentMinFee ) ;
2016-10-13 16:19:20 +02:00
if ( ( it ! = feeset . begin ( ) & & insecure_rand . rand32 ( ) % 3 ! = 0 ) | | it = = feeset . end ( ) ) {
2016-02-12 15:57:15 -05:00
it - - ;
}
2017-09-11 15:43:49 -04:00
return static_cast < CAmount > ( * it ) ;
2016-02-12 15:57:15 -05:00
}