2010-08-29 12:58:15 -04:00
// Copyright (c) 2009-2010 Satoshi Nakamoto
2012-02-07 13:28:30 -03:00
// Copyright (c) 2009-2012 The Bitcoin developers
2010-08-29 12:58:15 -04:00
// Distributed under the MIT/X11 software license, see the accompanying
2012-05-18 10:02:28 -04:00
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
2012-08-28 17:04:54 -04:00
# include "alert.h"
2011-09-08 17:50:58 -03:00
# include "checkpoints.h"
2011-05-14 16:57:34 -04:00
# include "db.h"
2012-09-03 16:14:03 -03:00
# include "txdb.h"
2011-05-14 17:20:30 -04:00
# include "net.h"
2011-05-15 17:52:31 -04:00
# include "init.h"
2012-04-15 17:10:54 -03:00
# include "ui_interface.h"
2012-01-03 17:24:28 -03:00
# include <boost/algorithm/string/replace.hpp>
2011-06-19 18:12:31 -04:00
# include <boost/filesystem.hpp>
2011-05-15 23:45:35 -04:00
# include <boost/filesystem/fstream.hpp>
2010-08-29 12:58:15 -04:00
2011-05-15 03:11:04 -04:00
using namespace std ;
using namespace boost ;
2010-08-29 12:58:15 -04:00
//
// Global state
//
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
CCriticalSection cs_setpwalletRegistered ;
set < CWallet * > setpwalletRegistered ;
2010-08-29 12:58:15 -04:00
CCriticalSection cs_main ;
2012-04-13 17:28:07 -03:00
CTxMemPool mempool ;
2010-08-29 12:58:15 -04:00
unsigned int nTransactionsUpdated = 0 ;
map < uint256 , CBlockIndex * > mapBlockIndex ;
2010-10-19 14:16:51 -03:00
uint256 hashGenesisBlock ( " 0x000000000019d6689c085ae165831e934ff763ae46a2a6c172b3f1b60a8ce26f " ) ;
2011-08-11 07:41:01 -04:00
static CBigNum bnProofOfWorkLimit ( ~ uint256 ( 0 ) > > 32 ) ;
2010-08-29 12:58:15 -04:00
CBlockIndex * pindexGenesisBlock = NULL ;
int nBestHeight = - 1 ;
CBigNum bnBestChainWork = 0 ;
CBigNum bnBestInvalidWork = 0 ;
uint256 hashBestChain = 0 ;
CBlockIndex * pindexBest = NULL ;
2012-08-18 18:33:01 -04:00
set < CBlockIndex * , CBlockIndexWorkComparator > setBlockIndexValid ; // may contain all CBlockIndex*'s that have validness >=BLOCK_VALID_TRANSACTIONS, and must contain those who aren't failed
2011-12-21 18:33:19 -03:00
int64 nTimeBestReceived = 0 ;
2012-09-13 09:33:52 -03:00
bool fImporting = false ;
2012-10-21 16:23:13 -03:00
bool fReindex = false ;
2012-12-01 16:10:23 -03:00
bool fBenchmark = false ;
2012-11-04 13:11:48 -03:00
unsigned int nCoinCacheSize = 5000 ;
2010-08-29 12:58:15 -04:00
2013-01-03 18:13:24 -03:00
CMedianFilter < int > cPeerBlockCounts ( 8 , 0 ) ; // Amount of blocks that other nodes claim to have
2011-09-28 16:35:58 -03:00
2010-08-29 12:58:15 -04:00
map < uint256 , CBlock * > mapOrphanBlocks ;
multimap < uint256 , CBlock * > mapOrphanBlocksByPrev ;
map < uint256 , CDataStream * > mapOrphanTransactions ;
2012-05-15 15:53:30 -04:00
map < uint256 , map < uint256 , CDataStream * > > mapOrphanTransactionsByPrev ;
2010-08-29 12:58:15 -04:00
2012-02-06 17:48:00 -03:00
// Constant stuff for coinbase transactions we create:
CScript COINBASE_FLAGS ;
2010-08-29 12:58:15 -04:00
2011-12-23 12:14:57 -03:00
const string strMessageMagic = " Bitcoin Signed Message: \n " ;
2010-08-29 12:58:15 -04:00
double dHashesPerSec ;
2011-12-21 18:33:19 -03:00
int64 nHPSTimerStart ;
2010-08-29 12:58:15 -04:00
// Settings
2011-12-21 18:33:19 -03:00
int64 nTransactionFee = 0 ;
2012-02-16 17:00:16 -03:00
2011-03-26 09:01:27 -03:00
2010-08-29 12:58:15 -04:00
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
//////////////////////////////////////////////////////////////////////////////
//
// dispatching functions
//
2011-11-06 20:05:42 -03:00
// These functions dispatch to one or all registered wallets
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
void RegisterWallet ( CWallet * pwalletIn )
{
{
2012-04-06 13:39:12 -03:00
LOCK ( cs_setpwalletRegistered ) ;
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
setpwalletRegistered . insert ( pwalletIn ) ;
}
}
void UnregisterWallet ( CWallet * pwalletIn )
{
{
2012-04-06 13:39:12 -03:00
LOCK ( cs_setpwalletRegistered ) ;
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
setpwalletRegistered . erase ( pwalletIn ) ;
}
}
2011-11-06 20:05:42 -03:00
// check whether the passed transaction is from us
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
bool static IsFromMe ( CTransaction & tx )
{
BOOST_FOREACH ( CWallet * pwallet , setpwalletRegistered )
if ( pwallet - > IsFromMe ( tx ) )
return true ;
return false ;
}
2011-11-06 20:05:42 -03:00
// get the wallet transaction with the given hash (if it exists)
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
bool static GetTransaction ( const uint256 & hashTx , CWalletTx & wtx )
{
BOOST_FOREACH ( CWallet * pwallet , setpwalletRegistered )
if ( pwallet - > GetTransaction ( hashTx , wtx ) )
return true ;
return false ;
}
2011-11-06 20:05:42 -03:00
// erases transaction with the given hash from all wallets
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
void static EraseFromWallets ( uint256 hash )
{
BOOST_FOREACH ( CWallet * pwallet , setpwalletRegistered )
pwallet - > EraseFromWallet ( hash ) ;
}
2011-11-06 20:05:42 -03:00
// make sure all wallets know about the given transaction, in the given block
2012-07-07 18:06:34 -04:00
void SyncWithWallets ( const uint256 & hash , const CTransaction & tx , const CBlock * pblock , bool fUpdate )
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
{
BOOST_FOREACH ( CWallet * pwallet , setpwalletRegistered )
2012-07-07 18:06:34 -04:00
pwallet - > AddToWalletIfInvolvingMe ( hash , tx , pblock , fUpdate ) ;
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
}
2011-11-06 20:05:42 -03:00
// notify wallets about a new best chain
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
void static SetBestChain ( const CBlockLocator & loc )
{
BOOST_FOREACH ( CWallet * pwallet , setpwalletRegistered )
pwallet - > SetBestChain ( loc ) ;
}
2011-11-06 20:05:42 -03:00
// notify wallets about an updated transaction
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
void static UpdatedTransaction ( const uint256 & hashTx )
{
BOOST_FOREACH ( CWallet * pwallet , setpwalletRegistered )
pwallet - > UpdatedTransaction ( hashTx ) ;
}
2011-11-06 20:05:42 -03:00
// dump all wallets
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
void static PrintWallets ( const CBlock & block )
{
BOOST_FOREACH ( CWallet * pwallet , setpwalletRegistered )
pwallet - > PrintWallet ( block ) ;
}
2011-11-06 20:05:42 -03:00
// notify wallets about an incoming inventory (for request counts)
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
void static Inventory ( const uint256 & hash )
{
BOOST_FOREACH ( CWallet * pwallet , setpwalletRegistered )
pwallet - > Inventory ( hash ) ;
}
2011-11-06 20:05:42 -03:00
// ask wallets to resend their transactions
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
void static ResendWalletTransactions ( )
{
BOOST_FOREACH ( CWallet * pwallet , setpwalletRegistered )
pwallet - > ResendWalletTransactions ( ) ;
}
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
//////////////////////////////////////////////////////////////////////////////
//
// CCoinsView implementations
//
bool CCoinsView : : GetCoins ( uint256 txid , CCoins & coins ) { return false ; }
bool CCoinsView : : SetCoins ( uint256 txid , const CCoins & coins ) { return false ; }
bool CCoinsView : : HaveCoins ( uint256 txid ) { return false ; }
CBlockIndex * CCoinsView : : GetBestBlock ( ) { return NULL ; }
bool CCoinsView : : SetBestBlock ( CBlockIndex * pindex ) { return false ; }
2012-07-06 10:33:34 -04:00
bool CCoinsView : : BatchWrite ( const std : : map < uint256 , CCoins > & mapCoins , CBlockIndex * pindex ) { return false ; }
2012-09-25 18:04:54 -03:00
bool CCoinsView : : GetStats ( CCoinsStats & stats ) { return false ; }
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
2012-07-08 13:04:05 -04:00
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
CCoinsViewBacked : : CCoinsViewBacked ( CCoinsView & viewIn ) : base ( & viewIn ) { }
bool CCoinsViewBacked : : GetCoins ( uint256 txid , CCoins & coins ) { return base - > GetCoins ( txid , coins ) ; }
bool CCoinsViewBacked : : SetCoins ( uint256 txid , const CCoins & coins ) { return base - > SetCoins ( txid , coins ) ; }
bool CCoinsViewBacked : : HaveCoins ( uint256 txid ) { return base - > HaveCoins ( txid ) ; }
CBlockIndex * CCoinsViewBacked : : GetBestBlock ( ) { return base - > GetBestBlock ( ) ; }
bool CCoinsViewBacked : : SetBestBlock ( CBlockIndex * pindex ) { return base - > SetBestBlock ( pindex ) ; }
void CCoinsViewBacked : : SetBackend ( CCoinsView & viewIn ) { base = & viewIn ; }
2012-07-06 10:33:34 -04:00
bool CCoinsViewBacked : : BatchWrite ( const std : : map < uint256 , CCoins > & mapCoins , CBlockIndex * pindex ) { return base - > BatchWrite ( mapCoins , pindex ) ; }
2012-09-25 18:04:54 -03:00
bool CCoinsViewBacked : : GetStats ( CCoinsStats & stats ) { return base - > GetStats ( stats ) ; }
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
CCoinsViewCache : : CCoinsViewCache ( CCoinsView & baseIn , bool fDummy ) : CCoinsViewBacked ( baseIn ) , pindexTip ( NULL ) { }
bool CCoinsViewCache : : GetCoins ( uint256 txid , CCoins & coins ) {
if ( cacheCoins . count ( txid ) ) {
coins = cacheCoins [ txid ] ;
return true ;
}
if ( base - > GetCoins ( txid , coins ) ) {
cacheCoins [ txid ] = coins ;
return true ;
}
return false ;
}
2012-07-08 13:04:05 -04:00
std : : map < uint256 , CCoins > : : iterator CCoinsViewCache : : FetchCoins ( uint256 txid ) {
std : : map < uint256 , CCoins > : : iterator it = cacheCoins . find ( txid ) ;
if ( it ! = cacheCoins . end ( ) )
return it ;
CCoins tmp ;
if ( ! base - > GetCoins ( txid , tmp ) )
return it ;
std : : pair < std : : map < uint256 , CCoins > : : iterator , bool > ret = cacheCoins . insert ( std : : make_pair ( txid , tmp ) ) ;
return ret . first ;
}
CCoins & CCoinsViewCache : : GetCoins ( uint256 txid ) {
std : : map < uint256 , CCoins > : : iterator it = FetchCoins ( txid ) ;
assert ( it ! = cacheCoins . end ( ) ) ;
return it - > second ;
}
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
bool CCoinsViewCache : : SetCoins ( uint256 txid , const CCoins & coins ) {
cacheCoins [ txid ] = coins ;
return true ;
}
bool CCoinsViewCache : : HaveCoins ( uint256 txid ) {
2012-07-08 13:04:05 -04:00
return FetchCoins ( txid ) ! = cacheCoins . end ( ) ;
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
}
CBlockIndex * CCoinsViewCache : : GetBestBlock ( ) {
if ( pindexTip = = NULL )
pindexTip = base - > GetBestBlock ( ) ;
return pindexTip ;
}
bool CCoinsViewCache : : SetBestBlock ( CBlockIndex * pindex ) {
pindexTip = pindex ;
return true ;
}
2012-07-06 10:33:34 -04:00
bool CCoinsViewCache : : BatchWrite ( const std : : map < uint256 , CCoins > & mapCoins , CBlockIndex * pindex ) {
for ( std : : map < uint256 , CCoins > : : const_iterator it = mapCoins . begin ( ) ; it ! = mapCoins . end ( ) ; it + + )
cacheCoins [ it - > first ] = it - > second ;
pindexTip = pindex ;
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
return true ;
}
2012-07-06 10:33:34 -04:00
bool CCoinsViewCache : : Flush ( ) {
bool fOk = base - > BatchWrite ( cacheCoins , pindexTip ) ;
if ( fOk )
cacheCoins . clear ( ) ;
return fOk ;
}
unsigned int CCoinsViewCache : : GetCacheSize ( ) {
return cacheCoins . size ( ) ;
}
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
/** CCoinsView that brings transactions from a memorypool into view.
It does not check for spendings by memory pool transactions . */
CCoinsViewMemPool : : CCoinsViewMemPool ( CCoinsView & baseIn , CTxMemPool & mempoolIn ) : CCoinsViewBacked ( baseIn ) , mempool ( mempoolIn ) { }
bool CCoinsViewMemPool : : GetCoins ( uint256 txid , CCoins & coins ) {
if ( base - > GetCoins ( txid , coins ) )
return true ;
if ( mempool . exists ( txid ) ) {
const CTransaction & tx = mempool . lookup ( txid ) ;
coins = CCoins ( tx , MEMPOOL_HEIGHT ) ;
return true ;
}
return false ;
}
bool CCoinsViewMemPool : : HaveCoins ( uint256 txid ) {
return mempool . exists ( txid ) | | base - > HaveCoins ( txid ) ;
}
2012-07-06 10:33:34 -04:00
CCoinsViewCache * pcoinsTip = NULL ;
2012-09-03 10:26:57 -03:00
CBlockTreeDB * pblocktree = NULL ;
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
2010-08-29 12:58:15 -04:00
//////////////////////////////////////////////////////////////////////////////
//
// mapOrphanTransactions
//
2012-05-15 15:53:30 -04:00
bool AddOrphanTx ( const CDataStream & vMsg )
2010-08-29 12:58:15 -04:00
{
CTransaction tx ;
CDataStream ( vMsg ) > > tx ;
uint256 hash = tx . GetHash ( ) ;
if ( mapOrphanTransactions . count ( hash ) )
2012-05-15 15:53:30 -04:00
return false ;
CDataStream * pvMsg = new CDataStream ( vMsg ) ;
2012-02-29 12:14:18 -03:00
2012-05-15 15:53:30 -04:00
// Ignore big transactions, to avoid a
// send-big-orphans memory exhaustion attack. If a peer has a legitimate
// large transaction with a missing parent then we assume
// it will rebroadcast it later, after the parent transaction(s)
// have been mined or received.
// 10,000 orphans, each of which is at most 5,000 bytes big is
// at most 500 megabytes of orphans:
if ( pvMsg - > size ( ) > 5000 )
{
2012-09-29 06:57:44 -03:00
printf ( " ignoring large orphan tx (size: % " PRIszu " , hash: %s) \n " , pvMsg - > size ( ) , hash . ToString ( ) . substr ( 0 , 10 ) . c_str ( ) ) ;
2012-06-19 15:50:12 -04:00
delete pvMsg ;
2012-05-15 15:53:30 -04:00
return false ;
}
2012-02-29 12:14:18 -03:00
2012-05-15 15:53:30 -04:00
mapOrphanTransactions [ hash ] = pvMsg ;
2011-05-15 03:11:04 -04:00
BOOST_FOREACH ( const CTxIn & txin , tx . vin )
2012-05-15 15:53:30 -04:00
mapOrphanTransactionsByPrev [ txin . prevout . hash ] . insert ( make_pair ( hash , pvMsg ) ) ;
2012-09-29 06:57:44 -03:00
printf ( " stored orphan tx %s (mapsz % " PRIszu " ) \n " , hash . ToString ( ) . substr ( 0 , 10 ) . c_str ( ) ,
2012-05-15 15:53:30 -04:00
mapOrphanTransactions . size ( ) ) ;
return true ;
2010-08-29 12:58:15 -04:00
}
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
void static EraseOrphanTx ( uint256 hash )
2010-08-29 12:58:15 -04:00
{
if ( ! mapOrphanTransactions . count ( hash ) )
return ;
const CDataStream * pvMsg = mapOrphanTransactions [ hash ] ;
CTransaction tx ;
CDataStream ( * pvMsg ) > > tx ;
2011-05-15 03:11:04 -04:00
BOOST_FOREACH ( const CTxIn & txin , tx . vin )
2010-08-29 12:58:15 -04:00
{
2012-05-15 15:53:30 -04:00
mapOrphanTransactionsByPrev [ txin . prevout . hash ] . erase ( hash ) ;
if ( mapOrphanTransactionsByPrev [ txin . prevout . hash ] . empty ( ) )
mapOrphanTransactionsByPrev . erase ( txin . prevout . hash ) ;
2010-08-29 12:58:15 -04:00
}
delete pvMsg ;
mapOrphanTransactions . erase ( hash ) ;
}
2012-04-23 15:14:03 -03:00
unsigned int LimitOrphanTxSize ( unsigned int nMaxOrphans )
2012-02-29 12:14:18 -03:00
{
2012-04-23 15:14:03 -03:00
unsigned int nEvicted = 0 ;
2012-02-29 12:14:18 -03:00
while ( mapOrphanTransactions . size ( ) > nMaxOrphans )
{
// Evict a random orphan:
2012-05-17 12:13:14 -04:00
uint256 randomhash = GetRandHash ( ) ;
2012-02-29 12:14:18 -03:00
map < uint256 , CDataStream * > : : iterator it = mapOrphanTransactions . lower_bound ( randomhash ) ;
if ( it = = mapOrphanTransactions . end ( ) )
it = mapOrphanTransactions . begin ( ) ;
EraseOrphanTx ( it - > first ) ;
+ + nEvicted ;
}
return nEvicted ;
}
2010-08-29 12:58:15 -04:00
//////////////////////////////////////////////////////////////////////////////
//
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
// CTransaction
2010-08-29 12:58:15 -04:00
//
2011-10-03 14:05:43 -03:00
bool CTransaction : : IsStandard ( ) const
{
2012-06-27 12:43:19 -04:00
if ( nVersion > CTransaction : : CURRENT_VERSION )
return false ;
2011-10-03 14:05:43 -03:00
BOOST_FOREACH ( const CTxIn & txin , vin )
{
2011-11-08 15:20:29 -03:00
// Biggest 'standard' txin is a 3-signature 3-of-3 CHECKMULTISIG
2012-01-04 23:40:52 -03:00
// pay-to-script-hash, which is 3 ~80-byte signatures, 3
2011-10-03 14:05:43 -03:00
// ~65-byte public keys, plus a few script ops.
2011-11-08 15:20:29 -03:00
if ( txin . scriptSig . size ( ) > 500 )
2012-01-04 23:40:52 -03:00
return false ;
2011-10-03 14:05:43 -03:00
if ( ! txin . scriptSig . IsPushOnly ( ) )
2012-01-04 23:40:52 -03:00
return false ;
2011-10-03 14:05:43 -03:00
}
2012-08-24 06:14:48 -04:00
BOOST_FOREACH ( const CTxOut & txout , vout ) {
2011-10-03 14:05:43 -03:00
if ( ! : : IsStandard ( txout . scriptPubKey ) )
2012-01-04 23:40:52 -03:00
return false ;
2012-08-24 06:14:48 -04:00
if ( txout . nValue = = 0 )
return false ;
}
2011-10-03 14:05:43 -03:00
return true ;
}
//
// Check transaction inputs, and make sure any
2012-01-04 23:40:52 -03:00
// pay-to-script-hash transactions are evaluating IsStandard scripts
2011-10-03 14:05:43 -03:00
//
// Why bother? To avoid denial-of-service attacks; an attacker
2012-01-04 23:40:52 -03:00
// can submit a standard HASH... OP_EQUAL transaction,
// which will get accepted into blocks. The redemption
// script can be anything; an attacker could use a very
2011-10-03 14:05:43 -03:00
// expensive-to-check-upon-redemption script like:
// DUP CHECKSIG DROP ... repeated 100 times... OP_1
//
2012-07-08 13:04:05 -04:00
bool CTransaction : : AreInputsStandard ( CCoinsViewCache & mapInputs ) const
2011-10-03 14:05:43 -03:00
{
2012-01-10 22:18:00 -03:00
if ( IsCoinBase ( ) )
2012-01-20 22:59:04 -03:00
return true ; // Coinbases don't use vin normally
2012-01-10 22:18:00 -03:00
2012-04-15 17:52:09 -03:00
for ( unsigned int i = 0 ; i < vin . size ( ) ; i + + )
2011-10-03 14:05:43 -03:00
{
2012-01-10 22:18:00 -03:00
const CTxOut & prev = GetOutputFor ( vin [ i ] , mapInputs ) ;
2011-10-03 14:05:43 -03:00
vector < vector < unsigned char > > vSolutions ;
2011-11-08 15:20:29 -03:00
txnouttype whichType ;
// get the scriptPubKey corresponding to this input:
2012-01-10 22:18:00 -03:00
const CScript & prevScript = prev . scriptPubKey ;
2011-11-08 15:20:29 -03:00
if ( ! Solver ( prevScript , whichType , vSolutions ) )
2012-01-04 23:40:52 -03:00
return false ;
2012-01-19 15:30:54 -03:00
int nArgsExpected = ScriptSigArgsExpected ( whichType , vSolutions ) ;
2012-04-22 14:44:12 -03:00
if ( nArgsExpected < 0 )
return false ;
2012-01-19 15:30:54 -03:00
// Transactions with extra stuff in their scriptSigs are
// non-standard. Note that this EvalScript() call will
// be quick, because if there are any operations
// beside "push data" in the scriptSig the
// IsStandard() call returns false
vector < vector < unsigned char > > stack ;
2012-08-22 16:33:21 -04:00
if ( ! EvalScript ( stack , vin [ i ] . scriptSig , * this , i , false , 0 ) )
2012-01-19 15:30:54 -03:00
return false ;
2011-10-03 14:05:43 -03:00
if ( whichType = = TX_SCRIPTHASH )
{
2012-01-04 23:40:52 -03:00
if ( stack . empty ( ) )
2011-10-03 14:05:43 -03:00
return false ;
2011-11-08 15:20:29 -03:00
CScript subscript ( stack . back ( ) . begin ( ) , stack . back ( ) . end ( ) ) ;
2012-01-19 15:30:54 -03:00
vector < vector < unsigned char > > vSolutions2 ;
txnouttype whichType2 ;
if ( ! Solver ( subscript , whichType2 , vSolutions2 ) )
2012-01-04 23:40:52 -03:00
return false ;
2012-01-19 15:30:54 -03:00
if ( whichType2 = = TX_SCRIPTHASH )
return false ;
2012-04-22 14:44:12 -03:00
int tmpExpected ;
tmpExpected = ScriptSigArgsExpected ( whichType2 , vSolutions2 ) ;
if ( tmpExpected < 0 )
return false ;
nArgsExpected + = tmpExpected ;
2011-10-03 14:05:43 -03:00
}
2012-01-19 15:30:54 -03:00
2012-04-22 14:44:12 -03:00
if ( stack . size ( ) ! = ( unsigned int ) nArgsExpected )
2012-01-19 15:30:54 -03:00
return false ;
2011-10-03 14:05:43 -03:00
}
return true ;
}
2012-04-23 15:14:03 -03:00
unsigned int
2012-01-04 23:40:52 -03:00
CTransaction : : GetLegacySigOpCount ( ) const
{
2012-04-23 15:14:03 -03:00
unsigned int nSigOps = 0 ;
2012-01-04 23:40:52 -03:00
BOOST_FOREACH ( const CTxIn & txin , vin )
{
nSigOps + = txin . scriptSig . GetSigOpCount ( false ) ;
}
BOOST_FOREACH ( const CTxOut & txout , vout )
{
nSigOps + = txout . scriptPubKey . GetSigOpCount ( false ) ;
}
return nSigOps ;
}
2010-08-29 12:58:15 -04:00
int CMerkleTx : : SetMerkleBranch ( const CBlock * pblock )
{
if ( fClient )
{
if ( hashBlock = = 0 )
return 0 ;
}
else
{
CBlock blockTmp ;
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
if ( pblock = = NULL ) {
CCoins coins ;
2012-07-06 10:33:34 -04:00
if ( pcoinsTip - > GetCoins ( GetHash ( ) , coins ) ) {
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
CBlockIndex * pindex = FindBlockByHeight ( coins . nHeight ) ;
if ( pindex ) {
if ( ! blockTmp . ReadFromDisk ( pindex ) )
return 0 ;
pblock = & blockTmp ;
}
}
2010-08-29 12:58:15 -04:00
}
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
if ( pblock ) {
2010-08-29 12:58:15 -04:00
// Update the tx's hashBlock
hashBlock = pblock - > GetHash ( ) ;
// Locate the transaction
2012-04-22 14:51:16 -03:00
for ( nIndex = 0 ; nIndex < ( int ) pblock - > vtx . size ( ) ; nIndex + + )
2010-08-29 12:58:15 -04:00
if ( pblock - > vtx [ nIndex ] = = * ( CTransaction * ) this )
break ;
2012-04-22 14:51:16 -03:00
if ( nIndex = = ( int ) pblock - > vtx . size ( ) )
2010-08-29 12:58:15 -04:00
{
vMerkleBranch . clear ( ) ;
nIndex = - 1 ;
printf ( " ERROR: SetMerkleBranch() : couldn't find tx in block \n " ) ;
return 0 ;
}
// Fill in merkle branch
vMerkleBranch = pblock - > GetMerkleBranch ( nIndex ) ;
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
}
2010-08-29 12:58:15 -04:00
}
// Is the tx in a block that's in the main chain
map < uint256 , CBlockIndex * > : : iterator mi = mapBlockIndex . find ( hashBlock ) ;
if ( mi = = mapBlockIndex . end ( ) )
return 0 ;
CBlockIndex * pindex = ( * mi ) . second ;
if ( ! pindex | | ! pindex - > IsInMainChain ( ) )
return 0 ;
return pindexBest - > nHeight - pindex - > nHeight + 1 ;
}
2010-09-30 12:23:07 -04:00
bool CTransaction : : CheckTransaction ( ) const
{
// Basic checks that don't depend on any context
2011-08-15 20:33:00 -04:00
if ( vin . empty ( ) )
2011-09-06 17:59:38 -03:00
return DoS ( 10 , error ( " CTransaction::CheckTransaction() : vin empty " )) ;
2011-08-15 20:33:00 -04:00
if ( vout . empty ( ) )
2011-09-06 17:59:38 -03:00
return DoS ( 10 , error ( " CTransaction::CheckTransaction() : vout empty " )) ;
2010-09-30 12:23:07 -04:00
// Size limits
2012-04-16 09:56:45 -03:00
if ( : : GetSerializeSize ( * this , SER_NETWORK , PROTOCOL_VERSION ) > MAX_BLOCK_SIZE )
2011-09-06 17:59:38 -03:00
return DoS ( 100 , error ( " CTransaction::CheckTransaction() : size limits failed " )) ;
2010-09-30 12:23:07 -04:00
// Check for negative or overflow output values
2011-12-21 18:33:19 -03:00
int64 nValueOut = 0 ;
2011-05-15 03:11:04 -04:00
BOOST_FOREACH ( const CTxOut & txout , vout )
2010-09-30 12:23:07 -04:00
{
if ( txout . nValue < 0 )
2011-09-06 17:59:38 -03:00
return DoS ( 100 , error ( " CTransaction::CheckTransaction() : txout . nValue negative " )) ;
2010-09-30 12:23:07 -04:00
if ( txout . nValue > MAX_MONEY )
2011-09-06 17:59:38 -03:00
return DoS ( 100 , error ( " CTransaction::CheckTransaction() : txout . nValue too high " )) ;
2010-09-30 12:23:07 -04:00
nValueOut + = txout . nValue ;
if ( ! MoneyRange ( nValueOut ) )
2011-09-06 17:59:38 -03:00
return DoS ( 100 , error ( " CTransaction::CheckTransaction() : txout total out of range " )) ;
2010-09-30 12:23:07 -04:00
}
2011-07-30 17:01:45 -04:00
// Check for duplicate inputs
set < COutPoint > vInOutPoints ;
BOOST_FOREACH ( const CTxIn & txin , vin )
{
if ( vInOutPoints . count ( txin . prevout ) )
return false ;
vInOutPoints . insert ( txin . prevout ) ;
}
2010-09-30 12:23:07 -04:00
if ( IsCoinBase ( ) )
{
if ( vin [ 0 ] . scriptSig . size ( ) < 2 | | vin [ 0 ] . scriptSig . size ( ) > 100 )
2011-09-06 17:59:38 -03:00
return DoS ( 100 , error ( " CTransaction::CheckTransaction() : coinbase script size " )) ;
2010-09-30 12:23:07 -04:00
}
else
{
2011-05-15 03:11:04 -04:00
BOOST_FOREACH ( const CTxIn & txin , vin )
2010-09-30 12:23:07 -04:00
if ( txin . prevout . IsNull ( ) )
2011-09-06 17:59:38 -03:00
return DoS ( 10 , error ( " CTransaction::CheckTransaction() : prevout is null " )) ;
2010-09-30 12:23:07 -04:00
}
return true ;
}
2012-09-09 17:39:45 -03:00
int64 CTransaction : : GetMinFee ( unsigned int nBlockSize , bool fAllowFree ,
enum GetMinFee_mode mode ) const
{
// Base fee is either MIN_TX_FEE or MIN_RELAY_TX_FEE
int64 nBaseFee = ( mode = = GMF_RELAY ) ? MIN_RELAY_TX_FEE : MIN_TX_FEE ;
unsigned int nBytes = : : GetSerializeSize ( * this , SER_NETWORK , PROTOCOL_VERSION ) ;
unsigned int nNewBlockSize = nBlockSize + nBytes ;
int64 nMinFee = ( 1 + ( int64 ) nBytes / 1000 ) * nBaseFee ;
if ( fAllowFree )
{
if ( nBlockSize = = 1 )
{
// Transactions under 10K are free
// (about 4500 BTC if made of 50 BTC inputs)
if ( nBytes < 10000 )
nMinFee = 0 ;
}
else
{
// Free transaction area
if ( nNewBlockSize < 27000 )
nMinFee = 0 ;
}
}
// To limit dust spam, require MIN_TX_FEE/MIN_RELAY_TX_FEE if any output is less than 0.01
if ( nMinFee < nBaseFee )
{
BOOST_FOREACH ( const CTxOut & txout , vout )
if ( txout . nValue < CENT )
nMinFee = nBaseFee ;
}
// Raise the price as the block approaches full
if ( nBlockSize ! = 1 & & nNewBlockSize > = MAX_BLOCK_SIZE_GEN / 2 )
{
if ( nNewBlockSize > = MAX_BLOCK_SIZE_GEN )
return MAX_MONEY ;
nMinFee * = MAX_BLOCK_SIZE_GEN / ( MAX_BLOCK_SIZE_GEN - nNewBlockSize ) ;
}
if ( ! MoneyRange ( nMinFee ) )
nMinFee = MAX_MONEY ;
return nMinFee ;
}
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
void CTxMemPool : : pruneSpent ( const uint256 & hashTx , CCoins & coins )
{
LOCK ( cs ) ;
std : : map < COutPoint , CInPoint > : : iterator it = mapNextTx . lower_bound ( COutPoint ( hashTx , 0 ) ) ;
// iterate over all COutPoints in mapNextTx whose hash equals the provided hashTx
while ( it ! = mapNextTx . end ( ) & & it - > first . hash = = hashTx ) {
coins . Spend ( it - > first . n ) ; // and remove those outputs from coins
it + + ;
}
}
2012-09-09 17:39:45 -03:00
2012-07-06 10:33:34 -04:00
bool CTxMemPool : : accept ( CTransaction & tx , bool fCheckInputs ,
2012-04-13 18:34:22 -03:00
bool * pfMissingInputs )
2010-08-29 12:58:15 -04:00
{
if ( pfMissingInputs )
* pfMissingInputs = false ;
2012-04-13 18:34:22 -03:00
if ( ! tx . CheckTransaction ( ) )
return error ( " CTxMemPool::accept() : CheckTransaction failed " ) ;
2010-09-30 12:23:07 -04:00
2010-08-29 12:58:15 -04:00
// Coinbase is only valid in a block, not as a loose transaction
2012-04-13 18:34:22 -03:00
if ( tx . IsCoinBase ( ) )
return tx . DoS ( 100 , error ( " CTxMemPool::accept() : coinbase as individual tx " ) ) ;
2010-08-29 12:58:15 -04:00
// To help v0.1.5 clients who would see it as a negative number
2012-04-13 18:34:22 -03:00
if ( ( int64 ) tx . nLockTime > std : : numeric_limits < int > : : max ( ) )
return error ( " CTxMemPool::accept() : not accepting nLockTime beyond 2038 yet " ) ;
2010-09-06 21:12:53 -04:00
2011-04-20 12:20:33 -03:00
// Rather not work on nonstandard transactions (unless -testnet)
2012-04-13 18:34:22 -03:00
if ( ! fTestNet & & ! tx . IsStandard ( ) )
return error ( " CTxMemPool::accept() : nonstandard transaction type " ) ;
2010-12-12 15:20:36 -03:00
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
// is it already in the memory pool?
2012-04-13 18:34:22 -03:00
uint256 hash = tx . GetHash ( ) ;
2012-04-06 13:39:12 -03:00
{
2012-04-13 18:34:22 -03:00
LOCK ( cs ) ;
if ( mapTx . count ( hash ) )
2010-08-29 12:58:15 -04:00
return false ;
2012-04-06 13:39:12 -03:00
}
2010-08-29 12:58:15 -04:00
// Check for conflicts with in-memory transactions
CTransaction * ptxOld = NULL ;
2012-04-17 15:12:48 -03:00
for ( unsigned int i = 0 ; i < tx . vin . size ( ) ; i + + )
2010-08-29 12:58:15 -04:00
{
2012-04-13 18:34:22 -03:00
COutPoint outpoint = tx . vin [ i ] . prevout ;
2010-08-29 12:58:15 -04:00
if ( mapNextTx . count ( outpoint ) )
{
// Disable replacement feature for now
return false ;
// Allow replacing with a newer version of the same transaction
if ( i ! = 0 )
return false ;
ptxOld = mapNextTx [ outpoint ] . ptx ;
2010-11-19 17:22:46 -03:00
if ( ptxOld - > IsFinal ( ) )
return false ;
2012-04-13 18:34:22 -03:00
if ( ! tx . IsNewerThan ( * ptxOld ) )
2010-08-29 12:58:15 -04:00
return false ;
2012-04-17 15:12:48 -03:00
for ( unsigned int i = 0 ; i < tx . vin . size ( ) ; i + + )
2010-08-29 12:58:15 -04:00
{
2012-04-13 18:34:22 -03:00
COutPoint outpoint = tx . vin [ i ] . prevout ;
2010-08-29 12:58:15 -04:00
if ( ! mapNextTx . count ( outpoint ) | | mapNextTx [ outpoint ] . ptx ! = ptxOld )
return false ;
}
break ;
}
}
2010-12-12 15:20:36 -03:00
if ( fCheckInputs )
2010-08-29 12:58:15 -04:00
{
2012-10-22 20:16:26 -03:00
CCoinsView dummy ;
CCoinsViewCache view ( dummy ) ;
{
LOCK ( cs ) ;
CCoinsViewMemPool viewMemPool ( * pcoinsTip , * this ) ;
view . SetBackend ( viewMemPool ) ;
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
// do we already have it?
if ( view . HaveCoins ( hash ) )
2012-05-10 20:31:46 -04:00
return false ;
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
// do all inputs exist?
2012-10-22 19:21:16 -03:00
// Note that this does not check for the presence of actual outputs (see the next check for that),
// only helps filling in pfMissingInputs (to determine missing vs spent).
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
BOOST_FOREACH ( const CTxIn txin , tx . vin ) {
if ( ! view . HaveCoins ( txin . prevout . hash ) ) {
if ( pfMissingInputs )
* pfMissingInputs = true ;
return false ;
}
2011-10-03 14:05:43 -03:00
}
2012-10-22 19:21:16 -03:00
// are the actual inputs available?
2012-07-08 13:04:05 -04:00
if ( ! tx . HaveInputs ( view ) )
return error ( " CTxMemPool::accept() : inputs already spent " ) ;
2012-11-11 09:11:42 -03:00
2012-10-22 20:16:26 -03:00
// Bring the best block into scope
view . GetBestBlock ( ) ;
// we have all inputs cached now, so switch back to dummy, so we don't need to keep lock on mempool
view . SetBackend ( dummy ) ;
}
2012-07-08 13:04:05 -04:00
2012-01-04 23:40:52 -03:00
// Check for non-standard pay-to-script-hash in inputs
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
if ( ! tx . AreInputsStandard ( view ) & & ! fTestNet )
2012-04-13 18:34:22 -03:00
return error ( " CTxMemPool::accept() : nonstandard transaction input " ) ;
2011-10-03 14:05:43 -03:00
2012-01-20 19:07:40 -03:00
// Note: if you modify this code to accept non-standard transactions, then
// you should add code here to check that the transaction does a
// reasonable number of ECDSA signature verifications.
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
int64 nFees = tx . GetValueIn ( view ) - tx . GetValueOut ( ) ;
2012-04-17 15:12:48 -03:00
unsigned int nSize = : : GetSerializeSize ( tx , SER_NETWORK , PROTOCOL_VERSION ) ;
2012-01-10 22:18:00 -03:00
// Don't accept it if it can't get into a block
2012-09-09 18:05:26 -03:00
int64 txMinFee = tx . GetMinFee ( 1000 , true , GMF_RELAY ) ;
if ( nFees < txMinFee )
return error ( " CTxMemPool::accept() : not enough fees %s, % " PRI64d " < % " PRI64d ,
2012-09-18 16:24:31 -03:00
hash . ToString ( ) . c_str ( ) ,
2012-09-09 18:05:26 -03:00
nFees , txMinFee ) ;
2012-01-04 23:40:52 -03:00
2011-03-11 13:50:16 -03:00
// Continuously rate-limit free transactions
2011-03-13 15:38:07 -03:00
// This mitigates 'penny-flooding' -- sending thousands of free transactions just to
2012-07-25 23:25:26 -04:00
// be annoying or make others' transactions take longer to confirm.
2011-05-25 18:38:20 -04:00
if ( nFees < MIN_RELAY_TX_FEE )
2010-12-12 15:20:36 -03:00
{
2011-03-13 15:38:07 -03:00
static CCriticalSection cs ;
2011-03-11 13:50:16 -03:00
static double dFreeCount ;
2011-12-21 18:33:19 -03:00
static int64 nLastTime ;
int64 nNow = GetTime ( ) ;
2011-03-13 15:38:07 -03:00
2010-12-12 15:20:36 -03:00
{
2011-03-13 15:38:07 -03:00
// Use an exponentially decaying ~10-minute window:
dFreeCount * = pow ( 1.0 - 1.0 / 600.0 , ( double ) ( nNow - nLastTime ) ) ;
nLastTime = nNow ;
// -limitfreerelay unit is thousand-bytes-per-minute
// At default rate it would take over a month to fill 1GB
2012-04-13 18:34:22 -03:00
if ( dFreeCount > GetArg ( " -limitfreerelay " , 15 ) * 10 * 1000 & & ! IsFromMe ( tx ) )
return error ( " CTxMemPool::accept() : free transaction rejected by rate limiter " ) ;
2011-03-13 15:38:07 -03:00
if ( fDebug )
printf ( " Rate limit dFreeCount: %g => %g \n " , dFreeCount , dFreeCount + nSize ) ;
dFreeCount + = nSize ;
2010-12-12 15:20:36 -03:00
}
}
2012-01-10 22:18:00 -03:00
// Check against previous transactions
// This is done last to help prevent CPU exhaustion denial-of-service attacks.
2012-11-13 19:03:25 -03:00
if ( ! tx . CheckInputs ( view , CS_ALWAYS , SCRIPT_VERIFY_P2SH ) )
2012-01-10 22:18:00 -03:00
{
2012-04-13 18:34:22 -03:00
return error ( " CTxMemPool::accept() : ConnectInputs failed % s " , hash.ToString().substr(0,10).c_str()) ;
2012-01-10 22:18:00 -03:00
}
2010-08-29 12:58:15 -04:00
}
// Store transaction in memory
{
2012-04-13 18:34:22 -03:00
LOCK ( cs ) ;
2010-08-29 12:58:15 -04:00
if ( ptxOld )
{
2012-04-13 18:34:22 -03:00
printf ( " CTxMemPool::accept() : replacing tx %s with new version \n " , ptxOld - > GetHash ( ) . ToString ( ) . c_str ( ) ) ;
remove ( * ptxOld ) ;
2010-08-29 12:58:15 -04:00
}
2012-07-04 01:12:44 -04:00
addUnchecked ( hash , tx ) ;
2010-08-29 12:58:15 -04:00
}
///// are we sure this is ok when loading transactions or restoring block txes
// If updated, erase old tx from wallet
if ( ptxOld )
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
EraseFromWallets ( ptxOld - > GetHash ( ) ) ;
2010-08-29 12:58:15 -04:00
2012-09-29 06:57:44 -03:00
printf ( " CTxMemPool::accept() : accepted %s (poolsz % " PRIszu " ) \n " ,
2012-05-10 20:20:31 -04:00
hash . ToString ( ) . substr ( 0 , 10 ) . c_str ( ) ,
mapTx . size ( ) ) ;
2010-08-29 12:58:15 -04:00
return true ;
}
2012-07-06 10:33:34 -04:00
bool CTransaction : : AcceptToMemoryPool ( bool fCheckInputs , bool * pfMissingInputs )
2012-04-13 18:34:22 -03:00
{
2012-07-06 10:33:34 -04:00
return mempool . accept ( * this , fCheckInputs , pfMissingInputs ) ;
2012-04-13 18:34:22 -03:00
}
2011-05-27 00:02:51 -04:00
2012-07-04 01:12:44 -04:00
bool CTxMemPool : : addUnchecked ( const uint256 & hash , CTransaction & tx )
2010-08-29 12:58:15 -04:00
{
// Add to memory pool without checking anything. Don't call this directly,
2012-04-13 18:34:22 -03:00
// call CTxMemPool::accept to properly check the transaction first.
2010-08-29 12:58:15 -04:00
{
2012-04-13 17:28:07 -03:00
mapTx [ hash ] = tx ;
2012-04-17 15:12:48 -03:00
for ( unsigned int i = 0 ; i < tx . vin . size ( ) ; i + + )
2012-04-13 17:28:07 -03:00
mapNextTx [ tx . vin [ i ] . prevout ] = CInPoint ( & mapTx [ hash ] , i ) ;
2010-08-29 12:58:15 -04:00
nTransactionsUpdated + + ;
}
return true ;
}
2012-11-24 10:26:51 -03:00
bool CTxMemPool : : remove ( const CTransaction & tx , bool fRecursive )
2010-08-29 12:58:15 -04:00
{
// Remove transaction from memory pool
{
2012-04-13 17:28:07 -03:00
LOCK ( cs ) ;
uint256 hash = tx . GetHash ( ) ;
if ( mapTx . count ( hash ) )
2012-02-09 10:21:41 -03:00
{
2012-11-24 10:26:51 -03:00
if ( fRecursive ) {
for ( unsigned int i = 0 ; i < tx . vout . size ( ) ; i + + ) {
std : : map < COutPoint , CInPoint > : : iterator it = mapNextTx . find ( COutPoint ( hash , i ) ) ;
if ( it ! = mapNextTx . end ( ) )
remove ( * it - > second . ptx , true ) ;
}
}
2012-04-13 17:28:07 -03:00
BOOST_FOREACH ( const CTxIn & txin , tx . vin )
2012-02-09 10:21:41 -03:00
mapNextTx . erase ( txin . prevout ) ;
2012-04-13 17:28:07 -03:00
mapTx . erase ( hash ) ;
2012-02-09 10:21:41 -03:00
nTransactionsUpdated + + ;
}
2010-08-29 12:58:15 -04:00
}
return true ;
}
2012-11-24 10:26:51 -03:00
bool CTxMemPool : : removeConflicts ( const CTransaction & tx )
{
// Remove transactions which depend on inputs of tx, recursively
LOCK ( cs ) ;
BOOST_FOREACH ( const CTxIn & txin , tx . vin ) {
std : : map < COutPoint , CInPoint > : : iterator it = mapNextTx . find ( txin . prevout ) ;
if ( it ! = mapNextTx . end ( ) ) {
const CTransaction & txConflict = * it - > second . ptx ;
if ( txConflict ! = tx )
remove ( txConflict , true ) ;
}
}
return true ;
}
2012-08-24 02:44:51 -04:00
void CTxMemPool : : clear ( )
2012-05-22 17:55:15 -04:00
{
LOCK ( cs ) ;
mapTx . clear ( ) ;
mapNextTx . clear ( ) ;
+ + nTransactionsUpdated ;
}
2012-06-22 11:43:34 -04:00
void CTxMemPool : : queryHashes ( std : : vector < uint256 > & vtxid )
{
vtxid . clear ( ) ;
2010-08-29 12:58:15 -04:00
2012-06-22 11:43:34 -04:00
LOCK ( cs ) ;
vtxid . reserve ( mapTx . size ( ) ) ;
for ( map < uint256 , CTransaction > : : iterator mi = mapTx . begin ( ) ; mi ! = mapTx . end ( ) ; + + mi )
vtxid . push_back ( ( * mi ) . first ) ;
}
2010-08-29 12:58:15 -04:00
2011-07-11 15:49:45 -04:00
int CMerkleTx : : GetDepthInMainChain ( CBlockIndex * & pindexRet ) const
2010-08-29 12:58:15 -04:00
{
if ( hashBlock = = 0 | | nIndex = = - 1 )
return 0 ;
// Find the block it claims to be in
map < uint256 , CBlockIndex * > : : iterator mi = mapBlockIndex . find ( hashBlock ) ;
if ( mi = = mapBlockIndex . end ( ) )
return 0 ;
CBlockIndex * pindex = ( * mi ) . second ;
if ( ! pindex | | ! pindex - > IsInMainChain ( ) )
return 0 ;
// Make sure the merkle branch connects to this block
if ( ! fMerkleVerified )
{
if ( CBlock : : CheckMerkleBranch ( GetHash ( ) , vMerkleBranch , nIndex ) ! = pindex - > hashMerkleRoot )
return 0 ;
fMerkleVerified = true ;
}
2011-07-11 15:49:45 -04:00
pindexRet = pindex ;
2010-08-29 12:58:15 -04:00
return pindexBest - > nHeight - pindex - > nHeight + 1 ;
}
int CMerkleTx : : GetBlocksToMaturity ( ) const
{
if ( ! IsCoinBase ( ) )
return 0 ;
return max ( 0 , ( COINBASE_MATURITY + 20 ) - GetDepthInMainChain ( ) ) ;
}
2012-07-06 10:33:34 -04:00
bool CMerkleTx : : AcceptToMemoryPool ( bool fCheckInputs )
2010-08-29 12:58:15 -04:00
{
if ( fClient )
{
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
if ( ! IsInMainChain ( ) & & ! ClientCheckInputs ( ) )
2010-08-29 12:58:15 -04:00
return false ;
2012-07-06 10:33:34 -04:00
return CTransaction : : AcceptToMemoryPool ( false ) ;
2010-08-29 12:58:15 -04:00
}
else
{
2012-07-06 10:33:34 -04:00
return CTransaction : : AcceptToMemoryPool ( fCheckInputs ) ;
2010-08-29 12:58:15 -04:00
}
}
2012-07-06 10:33:34 -04:00
bool CWalletTx : : AcceptWalletTransaction ( bool fCheckInputs )
2010-08-29 12:58:15 -04:00
{
{
2012-04-13 17:03:09 -03:00
LOCK ( mempool . cs ) ;
2010-09-06 21:12:53 -04:00
// Add previous supporting transactions first
2011-05-15 03:11:04 -04:00
BOOST_FOREACH ( CMerkleTx & tx , vtxPrev )
2010-08-29 12:58:15 -04:00
{
if ( ! tx . IsCoinBase ( ) )
{
uint256 hash = tx . GetHash ( ) ;
2012-07-06 10:33:34 -04:00
if ( ! mempool . exists ( hash ) & & pcoinsTip - > HaveCoins ( hash ) )
tx . AcceptToMemoryPool ( fCheckInputs ) ;
2010-08-29 12:58:15 -04:00
}
}
2012-07-06 10:33:34 -04:00
return AcceptToMemoryPool ( fCheckInputs ) ;
2010-08-29 12:58:15 -04:00
}
2010-09-06 21:12:53 -04:00
return false ;
2010-08-29 12:58:15 -04:00
}
2011-03-02 18:27:24 -03:00
2012-02-15 13:49:04 -03:00
// Return transaction in tx, and if it was found inside a block, its hash is placed in hashBlock
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
bool GetTransaction ( const uint256 & hash , CTransaction & txOut , uint256 & hashBlock , bool fAllowSlow )
2012-02-15 13:49:04 -03:00
{
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
CBlockIndex * pindexSlow = NULL ;
2012-02-15 13:49:04 -03:00
{
LOCK ( cs_main ) ;
{
LOCK ( mempool . cs ) ;
if ( mempool . exists ( hash ) )
{
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
txOut = mempool . lookup ( hash ) ;
2012-02-15 13:49:04 -03:00
return true ;
}
}
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
if ( fAllowSlow ) { // use coin database to locate block that contains transaction, and scan it
int nHeight = - 1 ;
{
2012-07-06 10:33:34 -04:00
CCoinsViewCache & view = * pcoinsTip ;
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
CCoins coins ;
if ( view . GetCoins ( hash , coins ) )
nHeight = coins . nHeight ;
}
if ( nHeight > 0 )
pindexSlow = FindBlockByHeight ( nHeight ) ;
2012-02-15 13:49:04 -03:00
}
}
2010-08-29 12:58:15 -04:00
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
if ( pindexSlow ) {
CBlock block ;
if ( block . ReadFromDisk ( pindexSlow ) ) {
BOOST_FOREACH ( const CTransaction & tx , block . vtx ) {
if ( tx . GetHash ( ) = = hash ) {
txOut = tx ;
hashBlock = pindexSlow - > GetBlockHash ( ) ;
return true ;
}
}
}
}
2010-08-29 12:58:15 -04:00
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
return false ;
}
2010-08-29 12:58:15 -04:00
//////////////////////////////////////////////////////////////////////////////
//
// CBlock and CBlockIndex
//
2012-07-19 16:06:20 -04:00
static CBlockIndex * pblockindexFBBHLast ;
CBlockIndex * FindBlockByHeight ( int nHeight )
{
CBlockIndex * pblockindex ;
if ( nHeight < nBestHeight / 2 )
pblockindex = pindexGenesisBlock ;
else
pblockindex = pindexBest ;
if ( pblockindexFBBHLast & & abs ( nHeight - pblockindex - > nHeight ) > abs ( nHeight - pblockindexFBBHLast - > nHeight ) )
pblockindex = pblockindexFBBHLast ;
while ( pblockindex - > nHeight > nHeight )
pblockindex = pblockindex - > pprev ;
while ( pblockindex - > nHeight < nHeight )
pblockindex = pblockindex - > pnext ;
pblockindexFBBHLast = pblockindex ;
return pblockindex ;
}
2012-11-14 18:18:10 -03:00
bool CBlock : : ReadFromDisk ( const CBlockIndex * pindex )
2010-08-29 12:58:15 -04:00
{
2012-11-14 18:18:10 -03:00
if ( ! ReadFromDisk ( pindex - > GetBlockPos ( ) ) )
2010-08-29 12:58:15 -04:00
return false ;
if ( GetHash ( ) ! = pindex - > GetBlockHash ( ) )
return error ( " CBlock::ReadFromDisk() : GetHash ( ) doesn ' t match index " ) ;
return true ;
}
2012-11-14 18:18:10 -03:00
uint256 static GetOrphanRoot ( const CBlockHeader * pblock )
2010-08-29 12:58:15 -04:00
{
// Work back to the first block in the orphan chain
while ( mapOrphanBlocks . count ( pblock - > hashPrevBlock ) )
pblock = mapOrphanBlocks [ pblock - > hashPrevBlock ] ;
return pblock - > GetHash ( ) ;
}
2011-12-21 18:33:19 -03:00
int64 static GetBlockValue ( int nHeight , int64 nFees )
2010-08-29 12:58:15 -04:00
{
2011-12-21 18:33:19 -03:00
int64 nSubsidy = 50 * COIN ;
2010-08-29 12:58:15 -04:00
2012-07-18 03:37:05 -04:00
// Subsidy is cut in half every 210000 blocks, which will occur approximately every 4 years
2010-08-29 12:58:15 -04:00
nSubsidy > > = ( nHeight / 210000 ) ;
return nSubsidy + nFees ;
}
2011-12-21 18:33:19 -03:00
static const int64 nTargetTimespan = 14 * 24 * 60 * 60 ; // two weeks
static const int64 nTargetSpacing = 10 * 60 ;
static const int64 nInterval = nTargetTimespan / nTargetSpacing ;
2011-09-08 13:51:43 -03:00
//
// minimum amount of work that could possibly be required nTime after
// minimum work required was nBase
//
2011-12-21 18:33:19 -03:00
unsigned int ComputeMinWork ( unsigned int nBase , int64 nTime )
2011-09-08 13:51:43 -03:00
{
2011-12-05 17:50:22 -03:00
// Testnet has min-difficulty blocks
// after nTargetSpacing*2 time between blocks:
if ( fTestNet & & nTime > nTargetSpacing * 2 )
return bnProofOfWorkLimit . GetCompact ( ) ;
2011-09-08 13:51:43 -03:00
CBigNum bnResult ;
bnResult . SetCompact ( nBase ) ;
while ( nTime > 0 & & bnResult < bnProofOfWorkLimit )
{
// Maximum 400% adjustment...
bnResult * = 4 ;
// ... in best-case exactly 4-times-normal target time
nTime - = nTargetTimespan * 4 ;
}
if ( bnResult > bnProofOfWorkLimit )
bnResult = bnProofOfWorkLimit ;
return bnResult . GetCompact ( ) ;
}
2012-11-14 18:18:10 -03:00
unsigned int static GetNextWorkRequired ( const CBlockIndex * pindexLast , const CBlockHeader * pblock )
2010-08-29 12:58:15 -04:00
{
2011-12-05 17:50:22 -03:00
unsigned int nProofOfWorkLimit = bnProofOfWorkLimit . GetCompact ( ) ;
2010-08-29 12:58:15 -04:00
// Genesis block
if ( pindexLast = = NULL )
2011-12-05 17:50:22 -03:00
return nProofOfWorkLimit ;
2010-08-29 12:58:15 -04:00
// Only change once per interval
if ( ( pindexLast - > nHeight + 1 ) % nInterval ! = 0 )
2011-12-05 17:50:22 -03:00
{
2012-04-13 15:38:05 -03:00
// Special difficulty rule for testnet:
if ( fTestNet )
2011-12-05 17:50:22 -03:00
{
// If the new block's timestamp is more than 2* 10 minutes
// then allow mining of a min-difficulty block.
2012-04-13 15:38:05 -03:00
if ( pblock - > nTime > pindexLast - > nTime + nTargetSpacing * 2 )
2011-12-05 17:50:22 -03:00
return nProofOfWorkLimit ;
else
{
// Return the last non-special-min-difficulty-rules-block
const CBlockIndex * pindex = pindexLast ;
while ( pindex - > pprev & & pindex - > nHeight % nInterval ! = 0 & & pindex - > nBits = = nProofOfWorkLimit )
pindex = pindex - > pprev ;
return pindex - > nBits ;
}
}
2010-08-29 12:58:15 -04:00
return pindexLast - > nBits ;
2011-12-05 17:50:22 -03:00
}
2010-08-29 12:58:15 -04:00
// Go back by what we want to be 14 days worth of blocks
const CBlockIndex * pindexFirst = pindexLast ;
for ( int i = 0 ; pindexFirst & & i < nInterval - 1 ; i + + )
pindexFirst = pindexFirst - > pprev ;
assert ( pindexFirst ) ;
// Limit adjustment step
2011-12-21 18:33:19 -03:00
int64 nActualTimespan = pindexLast - > GetBlockTime ( ) - pindexFirst - > GetBlockTime ( ) ;
2010-08-29 12:58:15 -04:00
printf ( " nActualTimespan = % " PRI64d " before bounds \n " , nActualTimespan ) ;
if ( nActualTimespan < nTargetTimespan / 4 )
nActualTimespan = nTargetTimespan / 4 ;
if ( nActualTimespan > nTargetTimespan * 4 )
nActualTimespan = nTargetTimespan * 4 ;
// Retarget
CBigNum bnNew ;
bnNew . SetCompact ( pindexLast - > nBits ) ;
bnNew * = nActualTimespan ;
bnNew / = nTargetTimespan ;
if ( bnNew > bnProofOfWorkLimit )
bnNew = bnProofOfWorkLimit ;
/// debug print
printf ( " GetNextWorkRequired RETARGET \n " ) ;
printf ( " nTargetTimespan = % " PRI64d " nActualTimespan = % " PRI64d " \n " , nTargetTimespan , nActualTimespan ) ;
printf ( " Before: %08x %s \n " , pindexLast - > nBits , CBigNum ( ) . SetCompact ( pindexLast - > nBits ) . getuint256 ( ) . ToString ( ) . c_str ( ) ) ;
printf ( " After: %08x %s \n " , bnNew . GetCompact ( ) , bnNew . getuint256 ( ) . ToString ( ) . c_str ( ) ) ;
return bnNew . GetCompact ( ) ;
}
bool CheckProofOfWork ( uint256 hash , unsigned int nBits )
{
CBigNum bnTarget ;
bnTarget . SetCompact ( nBits ) ;
// Check range
if ( bnTarget < = 0 | | bnTarget > bnProofOfWorkLimit )
return error ( " CheckProofOfWork() : nBits below minimum work " ) ;
// Check proof of work matches claimed amount
if ( hash > bnTarget . getuint256 ( ) )
return error ( " CheckProofOfWork() : hash doesn ' t match nBits " ) ;
return true ;
}
2011-09-02 13:02:22 -03:00
// Return maximum amount of blocks that other nodes claim to have
2011-09-11 05:49:30 -03:00
int GetNumBlocksOfPeers ( )
2011-09-02 13:02:22 -03:00
{
2011-09-08 17:50:58 -03:00
return std : : max ( cPeerBlockCounts . median ( ) , Checkpoints : : GetTotalBlocksEstimate ( ) ) ;
2011-09-02 13:02:22 -03:00
}
2010-08-29 12:58:15 -04:00
bool IsInitialBlockDownload ( )
{
2012-10-21 16:23:13 -03:00
if ( pindexBest = = NULL | | nBestHeight < Checkpoints : : GetTotalBlocksEstimate ( ) | | fReindex | | fImporting )
2010-08-29 12:58:15 -04:00
return true ;
2011-12-21 18:33:19 -03:00
static int64 nLastUpdate ;
2010-08-29 12:58:15 -04:00
static CBlockIndex * pindexLastBest ;
if ( pindexBest ! = pindexLastBest )
{
pindexLastBest = pindexBest ;
nLastUpdate = GetTime ( ) ;
}
return ( GetTime ( ) - nLastUpdate < 10 & &
pindexBest - > GetBlockTime ( ) < GetTime ( ) - 24 * 60 * 60 ) ;
}
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
void static InvalidChainFound ( CBlockIndex * pindexNew )
2010-08-29 12:58:15 -04:00
{
if ( pindexNew - > bnChainWork > bnBestInvalidWork )
{
bnBestInvalidWork = pindexNew - > bnChainWork ;
2012-09-03 10:26:57 -03:00
pblocktree - > WriteBestInvalidWork ( bnBestInvalidWork ) ;
2012-05-06 13:40:58 -04:00
uiInterface . NotifyBlocksChanged ( ) ;
2010-08-29 12:58:15 -04:00
}
2012-05-14 14:39:50 -04:00
printf ( " InvalidChainFound: invalid block=%s height=%d work=%s date=%s \n " ,
2012-08-13 03:02:44 -04:00
BlockHashStr ( pindexNew - > GetBlockHash ( ) ) . c_str ( ) , pindexNew - > nHeight ,
2013-01-01 17:28:28 -03:00
pindexNew - > bnChainWork . ToString ( ) . c_str ( ) , DateTimeStrFormat ( " %Y-%m-%d %H:%M:%S " ,
2012-05-14 14:39:50 -04:00
pindexNew - > GetBlockTime ( ) ) . c_str ( ) ) ;
printf ( " InvalidChainFound: current best=%s height=%d work=%s date=%s \n " ,
2012-08-13 03:02:44 -04:00
BlockHashStr ( hashBestChain ) . c_str ( ) , nBestHeight , bnBestChainWork . ToString ( ) . c_str ( ) ,
2013-01-01 17:28:28 -03:00
DateTimeStrFormat ( " %Y-%m-%d %H:%M:%S " , pindexBest - > GetBlockTime ( ) ) . c_str ( ) ) ;
2010-08-29 12:58:15 -04:00
if ( pindexBest & & bnBestInvalidWork > bnBestChainWork + pindexBest - > GetBlockWork ( ) * 6 )
2012-07-27 02:36:43 -04:00
printf ( " InvalidChainFound: Warning: Displayed transactions may not be correct! You may need to upgrade, or other nodes may need to upgrade. \n " ) ;
2010-08-29 12:58:15 -04:00
}
2012-08-18 18:33:01 -04:00
void static InvalidBlockFound ( CBlockIndex * pindex ) {
pindex - > nStatus | = BLOCK_FAILED_VALID ;
2012-09-03 10:26:57 -03:00
pblocktree - > WriteBlockIndex ( CDiskBlockIndex ( pindex ) ) ;
2012-08-18 18:33:01 -04:00
setBlockIndexValid . erase ( pindex ) ;
InvalidChainFound ( pindex ) ;
if ( pindex - > pnext )
ConnectBestBlock ( ) ; // reorganise away from the failed block
}
bool ConnectBestBlock ( ) {
do {
CBlockIndex * pindexNewBest ;
{
std : : set < CBlockIndex * , CBlockIndexWorkComparator > : : reverse_iterator it = setBlockIndexValid . rbegin ( ) ;
if ( it = = setBlockIndexValid . rend ( ) )
return true ;
pindexNewBest = * it ;
}
2012-11-04 09:38:51 -03:00
if ( pindexNewBest = = pindexBest | | ( pindexBest & & pindexNewBest - > bnChainWork = = pindexBest - > bnChainWork ) )
2012-08-18 18:33:01 -04:00
return true ; // nothing to do
// check ancestry
CBlockIndex * pindexTest = pindexNewBest ;
std : : vector < CBlockIndex * > vAttach ;
do {
if ( pindexTest - > nStatus & BLOCK_FAILED_MASK ) {
// mark descendants failed
CBlockIndex * pindexFailed = pindexNewBest ;
while ( pindexTest ! = pindexFailed ) {
pindexFailed - > nStatus | = BLOCK_FAILED_CHILD ;
setBlockIndexValid . erase ( pindexFailed ) ;
2012-09-03 10:26:57 -03:00
pblocktree - > WriteBlockIndex ( CDiskBlockIndex ( pindexFailed ) ) ;
2012-08-18 18:33:01 -04:00
pindexFailed = pindexFailed - > pprev ;
}
InvalidChainFound ( pindexNewBest ) ;
break ;
}
if ( pindexBest = = NULL | | pindexTest - > bnChainWork > pindexBest - > bnChainWork )
vAttach . push_back ( pindexTest ) ;
if ( pindexTest - > pprev = = NULL | | pindexTest - > pnext ! = NULL ) {
reverse ( vAttach . begin ( ) , vAttach . end ( ) ) ;
2012-12-04 19:53:26 -03:00
BOOST_FOREACH ( CBlockIndex * pindexSwitch , vAttach ) {
if ( fRequestShutdown )
break ;
2012-08-18 18:33:01 -04:00
if ( ! SetBestChain ( pindexSwitch ) )
return false ;
2012-12-04 19:53:26 -03:00
}
2012-08-18 18:33:01 -04:00
return true ;
}
pindexTest = pindexTest - > pprev ;
} while ( true ) ;
} while ( true ) ;
}
2012-11-14 18:18:10 -03:00
void CBlockHeader : : UpdateTime ( const CBlockIndex * pindexPrev )
2012-02-16 12:22:31 -03:00
{
nTime = max ( pindexPrev - > GetMedianTimePast ( ) + 1 , GetAdjustedTime ( ) ) ;
// Updating time can change work required on testnet:
if ( fTestNet )
nBits = GetNextWorkRequired ( pindexPrev , this ) ;
}
2010-08-29 12:58:15 -04:00
2012-07-08 13:04:05 -04:00
const CTxOut & CTransaction : : GetOutputFor ( const CTxIn & input , CCoinsViewCache & view )
2011-10-03 14:05:43 -03:00
{
2012-07-08 13:04:05 -04:00
const CCoins & coins = view . GetCoins ( input . prevout . hash ) ;
assert ( coins . IsAvailable ( input . prevout . n ) ) ;
return coins . vout [ input . prevout . n ] ;
2012-01-10 22:18:00 -03:00
}
2012-07-08 13:04:05 -04:00
int64 CTransaction : : GetValueIn ( CCoinsViewCache & inputs ) const
2012-01-10 22:18:00 -03:00
{
if ( IsCoinBase ( ) )
return 0 ;
int64 nResult = 0 ;
2012-04-15 17:52:09 -03:00
for ( unsigned int i = 0 ; i < vin . size ( ) ; i + + )
2012-01-10 22:18:00 -03:00
nResult + = GetOutputFor ( vin [ i ] , inputs ) . nValue ;
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
return nResult ;
2012-01-10 22:18:00 -03:00
}
2012-07-08 13:04:05 -04:00
unsigned int CTransaction : : GetP2SHSigOpCount ( CCoinsViewCache & inputs ) const
2012-01-10 22:18:00 -03:00
{
if ( IsCoinBase ( ) )
return 0 ;
2012-04-23 15:14:03 -03:00
unsigned int nSigOps = 0 ;
2012-04-15 17:52:09 -03:00
for ( unsigned int i = 0 ; i < vin . size ( ) ; i + + )
2012-01-10 22:18:00 -03:00
{
2012-07-08 13:04:05 -04:00
const CTxOut & prevout = GetOutputFor ( vin [ i ] , inputs ) ;
2012-01-20 19:07:40 -03:00
if ( prevout . scriptPubKey . IsPayToScriptHash ( ) )
nSigOps + = prevout . scriptPubKey . GetSigOpCount ( vin [ i ] . scriptSig ) ;
2012-01-10 22:18:00 -03:00
}
return nSigOps ;
}
2012-07-08 13:04:05 -04:00
bool CTransaction : : UpdateCoins ( CCoinsViewCache & inputs , CTxUndo & txundo , int nHeight , const uint256 & txhash ) const
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
{
// mark inputs spent
if ( ! IsCoinBase ( ) ) {
BOOST_FOREACH ( const CTxIn & txin , vin ) {
2012-07-08 13:04:05 -04:00
CCoins & coins = inputs . GetCoins ( txin . prevout . hash ) ;
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
CTxInUndo undo ;
if ( ! coins . Spend ( txin . prevout , undo ) )
return error ( " UpdateCoins() : cannot spend input " ) ;
txundo . vprevout . push_back ( undo ) ;
}
}
// add outputs
2012-07-07 18:06:34 -04:00
if ( ! inputs . SetCoins ( txhash , CCoins ( * this , nHeight ) ) )
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
return error ( " UpdateCoins() : cannot update output " ) ;
return true ;
}
2012-07-08 13:04:05 -04:00
bool CTransaction : : HaveInputs ( CCoinsViewCache & inputs ) const
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
{
2012-10-05 14:22:21 -03:00
if ( ! IsCoinBase ( ) ) {
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
// first check whether information about the prevout hash is available
for ( unsigned int i = 0 ; i < vin . size ( ) ; i + + ) {
const COutPoint & prevout = vin [ i ] . prevout ;
if ( ! inputs . HaveCoins ( prevout . hash ) )
return false ;
}
// then check whether the actual outputs are available
for ( unsigned int i = 0 ; i < vin . size ( ) ; i + + ) {
const COutPoint & prevout = vin [ i ] . prevout ;
2012-07-08 13:04:05 -04:00
const CCoins & coins = inputs . GetCoins ( prevout . hash ) ;
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
if ( ! coins . IsAvailable ( prevout . n ) )
return false ;
}
}
return true ;
}
2012-11-13 19:03:25 -03:00
bool CTransaction : : CheckInputs ( CCoinsViewCache & inputs , enum CheckSig_mode csmode , unsigned int flags ) const
2010-08-29 12:58:15 -04:00
{
if ( ! IsCoinBase ( ) )
{
2012-07-08 13:04:05 -04:00
// This doesn't trigger the DoS code on purpose; if it did, it would make it easier
// for an attacker to attempt to split the network.
if ( ! HaveInputs ( inputs ) )
return error ( " CheckInputs() : % s inputs unavailable " , GetHash().ToString().substr(0,10).c_str()) ;
2012-10-22 04:22:15 -03:00
// While checking, GetBestBlock() refers to the parent block.
// This is also true for mempool checks.
2012-11-11 09:11:42 -03:00
int nSpendHeight = inputs . GetBestBlock ( ) - > nHeight + 1 ;
2011-12-21 18:33:19 -03:00
int64 nValueIn = 0 ;
2012-01-10 22:18:00 -03:00
int64 nFees = 0 ;
2012-04-15 17:52:09 -03:00
for ( unsigned int i = 0 ; i < vin . size ( ) ; i + + )
2010-08-29 12:58:15 -04:00
{
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
const COutPoint & prevout = vin [ i ] . prevout ;
2012-07-08 13:04:05 -04:00
const CCoins & coins = inputs . GetCoins ( prevout . hash ) ;
2010-08-29 12:58:15 -04:00
// If prev is coinbase, check that it's matured
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
if ( coins . IsCoinBase ( ) ) {
2012-10-22 04:22:15 -03:00
if ( nSpendHeight - coins . nHeight < COINBASE_MATURITY )
return error ( " CheckInputs() : tried to spend coinbase at depth % d " , nSpendHeight - coins.nHeight) ;
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
}
2010-08-29 12:58:15 -04:00
2012-05-16 11:26:56 -04:00
// Check for negative or overflow input values
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
nValueIn + = coins . vout [ prevout . n ] . nValue ;
if ( ! MoneyRange ( coins . vout [ prevout . n ] . nValue ) | | ! MoneyRange ( nValueIn ) )
return DoS ( 100 , error ( " CheckInputs() : txin values out of range " )) ;
2012-05-16 11:26:56 -04:00
}
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
if ( nValueIn < GetValueOut ( ) )
return DoS ( 100 , error ( " ChecktInputs() : % s value in < value out " , GetHash().ToString().substr(0,10).c_str())) ;
// Tally transaction fees
int64 nTxFee = nValueIn - GetValueOut ( ) ;
if ( nTxFee < 0 )
return DoS ( 100 , error ( " CheckInputs() : % s nTxFee < 0 " , GetHash().ToString().substr(0,10).c_str())) ;
nFees + = nTxFee ;
if ( ! MoneyRange ( nFees ) )
return DoS ( 100 , error ( " CheckInputs() : nFees out of range " )) ;
2012-05-16 11:26:56 -04:00
// The first loop above does all the inexpensive checks.
// Only if ALL inputs pass do we perform expensive ECDSA signature checks.
// Helps prevent CPU exhaustion attacks.
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
// Skip ECDSA signature verification when connecting blocks
2012-10-05 14:22:21 -03:00
// before the last block chain checkpoint. This is safe because block merkle hashes are
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
// still computed and checked, and any change will be caught at the next checkpoint.
2012-10-05 14:22:21 -03:00
if ( csmode = = CS_ALWAYS | |
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
( csmode = = CS_AFTER_CHECKPOINT & & inputs . GetBestBlock ( ) - > nHeight > = Checkpoints : : GetTotalBlocksEstimate ( ) ) ) {
for ( unsigned int i = 0 ; i < vin . size ( ) ; i + + ) {
const COutPoint & prevout = vin [ i ] . prevout ;
2012-07-08 13:04:05 -04:00
const CCoins & coins = inputs . GetCoins ( prevout . hash ) ;
2012-01-10 22:18:00 -03:00
2011-09-02 17:59:47 -03:00
// Verify signature
2012-11-13 19:03:25 -03:00
if ( ! VerifySignature ( coins , * this , i , flags , 0 ) )
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
return DoS ( 100 , error ( " CheckInputs() : % s VerifySignature failed " , GetHash().ToString().substr(0,10).c_str())) ;
2011-11-08 15:20:29 -03:00
}
2010-08-29 12:58:15 -04:00
}
}
return true ;
}
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
bool CTransaction : : ClientCheckInputs ( ) const
2010-08-29 12:58:15 -04:00
{
if ( IsCoinBase ( ) )
return false ;
// Take over previous transactions' spent pointers
{
2012-04-13 17:03:09 -03:00
LOCK ( mempool . cs ) ;
2011-12-21 18:33:19 -03:00
int64 nValueIn = 0 ;
2012-04-15 17:52:09 -03:00
for ( unsigned int i = 0 ; i < vin . size ( ) ; i + + )
2010-08-29 12:58:15 -04:00
{
// Get prev tx from single transactions in memory
COutPoint prevout = vin [ i ] . prevout ;
2012-04-13 19:20:44 -03:00
if ( ! mempool . exists ( prevout . hash ) )
2010-08-29 12:58:15 -04:00
return false ;
2012-04-13 19:20:44 -03:00
CTransaction & txPrev = mempool . lookup ( prevout . hash ) ;
2010-08-29 12:58:15 -04:00
if ( prevout . n > = txPrev . vout . size ( ) )
return false ;
// Verify signature
2012-11-13 19:03:25 -03:00
if ( ! VerifySignature ( CCoins ( txPrev , - 1 ) , * this , i , SCRIPT_VERIFY_P2SH , 0 ) )
2010-08-29 12:58:15 -04:00
return error ( " ConnectInputs() : VerifySignature failed " ) ;
2012-04-13 17:03:09 -03:00
///// this is redundant with the mempool.mapNextTx stuff,
///// not sure which I want to get rid of
2010-08-29 12:58:15 -04:00
///// this has to go away now that posNext is gone
// // Check for conflicts
// if (!txPrev.vout[prevout.n].posNext.IsNull())
// return error("ConnectInputs() : prev tx already used");
//
// // Flag outpoints as used
// txPrev.vout[prevout.n].posNext = posThisTx;
nValueIn + = txPrev . vout [ prevout . n ] . nValue ;
2010-09-06 21:12:53 -04:00
if ( ! MoneyRange ( txPrev . vout [ prevout . n ] . nValue ) | | ! MoneyRange ( nValueIn ) )
return error ( " ClientConnectInputs() : txin values out of range " ) ;
2010-08-29 12:58:15 -04:00
}
if ( GetValueOut ( ) > nValueIn )
return false ;
}
return true ;
}
2012-07-08 13:04:05 -04:00
bool CBlock : : DisconnectBlock ( CBlockIndex * pindex , CCoinsViewCache & view )
2010-08-29 12:58:15 -04:00
{
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
assert ( pindex = = view . GetBestBlock ( ) ) ;
2010-08-29 12:58:15 -04:00
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
CBlockUndo blockUndo ;
2010-08-29 12:58:15 -04:00
{
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
CDiskBlockPos pos = pindex - > GetUndoPos ( ) ;
if ( pos . IsNull ( ) )
return error ( " DisconnectBlock() : no undo data available " ) ;
FILE * file = OpenUndoFile ( pos , true ) ;
if ( file = = NULL )
return error ( " DisconnectBlock() : undo file not available " ) ;
CAutoFile fileUndo ( file , SER_DISK , CLIENT_VERSION ) ;
fileUndo > > blockUndo ;
2010-08-29 12:58:15 -04:00
}
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
assert ( blockUndo . vtxundo . size ( ) + 1 = = vtx . size ( ) ) ;
// undo transactions in reverse order
for ( int i = vtx . size ( ) - 1 ; i > = 0 ; i - - ) {
const CTransaction & tx = vtx [ i ] ;
uint256 hash = tx . GetHash ( ) ;
// check that all outputs are available
2012-07-08 13:04:05 -04:00
if ( ! view . HaveCoins ( hash ) )
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
return error ( " DisconnectBlock() : outputs still spent ? database corrupted " ) ;
2012-07-08 13:04:05 -04:00
CCoins & outs = view . GetCoins ( hash ) ;
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
CCoins outsBlock = CCoins ( tx , pindex - > nHeight ) ;
if ( outs ! = outsBlock )
return error ( " DisconnectBlock() : added transaction mismatch ? database corrupted " ) ;
// remove outputs
2012-07-08 13:04:05 -04:00
outs = CCoins ( ) ;
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
// restore inputs
if ( i > 0 ) { // not coinbases
const CTxUndo & txundo = blockUndo . vtxundo [ i - 1 ] ;
assert ( txundo . vprevout . size ( ) = = tx . vin . size ( ) ) ;
for ( unsigned int j = tx . vin . size ( ) ; j - - > 0 ; ) {
const COutPoint & out = tx . vin [ j ] . prevout ;
const CTxInUndo & undo = txundo . vprevout [ j ] ;
CCoins coins ;
view . GetCoins ( out . hash , coins ) ; // this can fail if the prevout was already entirely spent
if ( coins . IsPruned ( ) ) {
if ( undo . nHeight = = 0 )
return error ( " DisconnectBlock() : undo data doesn ' t contain tx metadata ? database corrupted " ) ;
coins . fCoinBase = undo . fCoinBase ;
coins . nHeight = undo . nHeight ;
coins . nVersion = undo . nVersion ;
} else {
if ( undo . nHeight ! = 0 )
return error ( " DisconnectBlock() : undo data contains unneeded tx metadata ? database corrupted " ) ;
}
if ( coins . IsAvailable ( out . n ) )
return error ( " DisconnectBlock() : prevout output not spent ? database corrupted " ) ;
if ( coins . vout . size ( ) < out . n + 1 )
coins . vout . resize ( out . n + 1 ) ;
coins . vout [ out . n ] = undo . txout ;
if ( ! view . SetCoins ( out . hash , coins ) )
return error ( " DisconnectBlock() : cannot restore coin inputs " ) ;
}
}
}
// move best block pointer to prevout block
view . SetBestBlock ( pindex - > pprev ) ;
2010-08-29 12:58:15 -04:00
return true ;
}
2012-09-05 22:21:18 -03:00
void static FlushBlockFile ( )
{
LOCK ( cs_LastBlockFile ) ;
2012-12-03 06:14:54 -03:00
CDiskBlockPos posOld ( nLastBlockFile , 0 ) ;
2012-09-05 22:21:18 -03:00
FILE * fileOld = OpenBlockFile ( posOld ) ;
2012-12-01 07:36:53 -03:00
if ( fileOld ) {
FileCommit ( fileOld ) ;
fclose ( fileOld ) ;
}
2012-09-05 22:21:18 -03:00
fileOld = OpenUndoFile ( posOld ) ;
2012-12-01 07:36:53 -03:00
if ( fileOld ) {
FileCommit ( fileOld ) ;
fclose ( fileOld ) ;
}
2012-09-05 22:21:18 -03:00
}
2012-09-03 10:26:57 -03:00
bool FindUndoPos ( int nFile , CDiskBlockPos & pos , unsigned int nAddSize ) ;
2012-08-13 13:11:05 -04:00
2012-07-08 13:04:05 -04:00
bool CBlock : : ConnectBlock ( CBlockIndex * pindex , CCoinsViewCache & view , bool fJustCheck )
2010-08-29 12:58:15 -04:00
{
// Check it again in case a previous version let a bad block in
2012-05-09 13:24:44 -04:00
if ( ! CheckBlock ( ! fJustCheck , ! fJustCheck ) )
2010-08-29 12:58:15 -04:00
return false ;
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
// verify that the view's current state corresponds to the previous block
assert ( pindex - > pprev = = view . GetBestBlock ( ) ) ;
2012-02-17 13:58:02 -03:00
// Do not allow blocks that contain transactions which 'overwrite' older transactions,
// unless those are already completely spent.
// If such overwrites are allowed, coinbases and transactions depending upon those
// can be duplicated to remove the ability to spend the first instance -- even after
// being sent to another address.
// See BIP30 and http://r6.ca/blog/20120206T005236Z.html for more information.
// This logic is not necessary for memory pool transactions, as AcceptToMemoryPool
2012-07-25 23:25:26 -04:00
// already refuses previously-known transaction ids entirely.
2012-09-09 21:11:04 -03:00
// This rule was originally applied all blocks whose timestamp was after March 15, 2012, 0:00 UTC.
// Now that the whole chain is irreversibly beyond that time it is applied to all blocks except the
// two in the chain that violate it. This prevents exploiting the issue against nodes in their
// initial block download.
2012-10-24 02:41:52 -03:00
bool fEnforceBIP30 = ( ! pindex - > phashBlock ) | | // Enforce on CreateNewBlock invocations which don't have a hash.
! ( ( pindex - > nHeight = = 91842 & & pindex - > GetBlockHash ( ) = = uint256 ( " 0x00000000000a4d0a398161ffc163c503763b1f4360639393e0e4c8e300e0caec " ) ) | |
2012-09-09 21:11:04 -03:00
( pindex - > nHeight = = 91880 & & pindex - > GetBlockHash ( ) = = uint256 ( " 0x00000000000743f190a18c5577a3c2d2a1f610ae9601ac046a38084ccb7cd721 " ) ) ) ;
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
if ( fEnforceBIP30 ) {
2012-07-07 18:06:34 -04:00
for ( unsigned int i = 0 ; i < vtx . size ( ) ; i + + ) {
uint256 hash = GetTxHash ( i ) ;
2012-07-08 13:04:05 -04:00
if ( view . HaveCoins ( hash ) & & ! view . GetCoins ( hash ) . IsPruned ( ) )
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
return error ( " ConnectBlock() : tried to overwrite transaction " ) ;
}
}
2012-02-17 13:58:02 -03:00
2012-06-22 11:50:52 -04:00
// BIP16 didn't become active until Apr 1 2012
int64 nBIP16SwitchTime = 1333238400 ;
2012-03-21 16:45:58 -03:00
bool fStrictPayToScriptHash = ( pindex - > nTime > = nBIP16SwitchTime ) ;
2012-01-20 19:07:40 -03:00
2012-06-23 08:17:13 -04:00
CBlockUndo blockundo ;
2012-12-01 16:10:23 -03:00
int64 nStart = GetTimeMicros ( ) ;
2011-12-21 18:33:19 -03:00
int64 nFees = 0 ;
2012-12-01 16:10:23 -03:00
int nInputs = 0 ;
2012-04-23 15:14:03 -03:00
unsigned int nSigOps = 0 ;
2012-07-07 18:06:34 -04:00
for ( unsigned int i = 0 ; i < vtx . size ( ) ; i + + )
2010-08-29 12:58:15 -04:00
{
2012-12-01 16:10:23 -03:00
2012-07-07 18:06:34 -04:00
const CTransaction & tx = vtx [ i ] ;
2012-12-01 16:10:23 -03:00
nInputs + = tx . vin . size ( ) ;
2012-01-20 19:07:40 -03:00
nSigOps + = tx . GetLegacySigOpCount ( ) ;
if ( nSigOps > MAX_BLOCK_SIGOPS )
return DoS ( 100 , error ( " ConnectBlock() : too many sigops " )) ;
2012-01-10 22:18:00 -03:00
if ( ! tx . IsCoinBase ( ) )
{
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
if ( ! tx . HaveInputs ( view ) )
return DoS ( 100 , error ( " ConnectBlock() : inputs missing / spent " )) ;
2012-01-04 23:40:52 -03:00
2012-01-20 19:07:40 -03:00
if ( fStrictPayToScriptHash )
{
// Add in sigops done by pay-to-script-hash inputs;
// this is to prevent a "rogue miner" from creating
// an incredibly-expensive-to-validate block.
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
nSigOps + = tx . GetP2SHSigOpCount ( view ) ;
2012-01-20 19:07:40 -03:00
if ( nSigOps > MAX_BLOCK_SIGOPS )
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
return DoS ( 100 , error ( " ConnectBlock() : too many sigops " )) ;
2012-01-20 19:07:40 -03:00
}
2012-01-04 23:40:52 -03:00
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
nFees + = tx . GetValueIn ( view ) - tx . GetValueOut ( ) ;
2012-06-23 08:17:13 -04:00
2012-11-13 19:03:25 -03:00
if ( ! tx . CheckInputs ( view , CS_AFTER_CHECKPOINT , fStrictPayToScriptHash ? SCRIPT_VERIFY_P2SH : SCRIPT_VERIFY_NONE ) )
2012-01-10 22:45:55 -03:00
return false ;
2012-01-10 22:18:00 -03:00
}
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
CTxUndo txundo ;
2012-07-07 18:06:34 -04:00
if ( ! tx . UpdateCoins ( view , txundo , pindex - > nHeight , GetTxHash ( i ) ) )
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
return error ( " ConnectBlock() : UpdateInputs failed " ) ;
if ( ! tx . IsCoinBase ( ) )
blockundo . vtxundo . push_back ( txundo ) ;
2012-12-01 16:10:23 -03:00
2010-08-29 12:58:15 -04:00
}
2012-12-01 16:10:23 -03:00
int64 nTime = GetTimeMicros ( ) - nStart ;
if ( fBenchmark )
printf ( " - Connect %u transactions: %.2fms (%.3fms/tx, %.3fms/txin) \n " , ( unsigned ) vtx . size ( ) , 0.001 * nTime , 0.001 * nTime / vtx . size ( ) , nInputs < = 1 ? 0 : 0.001 * nTime / ( nInputs - 1 ) ) ;
2011-10-03 14:05:43 -03:00
2012-10-22 16:46:00 -03:00
if ( vtx [ 0 ] . GetValueOut ( ) > GetBlockValue ( pindex - > nHeight , nFees ) )
2012-11-16 17:46:16 -03:00
return error ( " ConnectBlock() : coinbase pays too much ( actual = % " PRI64d " vs limit = % " PRI64d " ) " , vtx[0].GetValueOut(), GetBlockValue(pindex->nHeight, nFees)) ;
2012-10-22 16:46:00 -03:00
2012-05-09 13:24:44 -04:00
if ( fJustCheck )
return true ;
2012-08-13 13:11:05 -04:00
// Write undo information to disk
2012-08-18 18:33:01 -04:00
if ( pindex - > GetUndoPos ( ) . IsNull ( ) | | ( pindex - > nStatus & BLOCK_VALID_MASK ) < BLOCK_VALID_SCRIPTS )
2012-08-13 13:11:05 -04:00
{
2012-08-18 18:33:01 -04:00
if ( pindex - > GetUndoPos ( ) . IsNull ( ) ) {
CDiskBlockPos pos ;
2012-09-03 10:26:57 -03:00
if ( ! FindUndoPos ( pindex - > nFile , pos , : : GetSerializeSize ( blockundo , SER_DISK , CLIENT_VERSION ) + 8 ) )
2012-08-18 18:33:01 -04:00
return error ( " ConnectBlock() : FindUndoPos failed " ) ;
if ( ! blockundo . WriteToDisk ( pos ) )
return error ( " ConnectBlock() : CBlockUndo : : WriteToDisk failed " ) ;
// update nUndoPos in block index
pindex - > nUndoPos = pos . nPos ;
pindex - > nStatus | = BLOCK_HAVE_UNDO ;
}
pindex - > nStatus = ( pindex - > nStatus & ~ BLOCK_VALID_MASK ) | BLOCK_VALID_SCRIPTS ;
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
CDiskBlockIndex blockindex ( pindex ) ;
2012-09-03 10:26:57 -03:00
if ( ! pblocktree - > WriteBlockIndex ( blockindex ) )
2010-10-05 22:19:47 -04:00
return error ( " ConnectBlock() : WriteBlockIndex failed " ) ;
2010-08-29 12:58:15 -04:00
}
2012-10-05 14:22:21 -03:00
// add this block to the view's block chain
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
if ( ! view . SetBestBlock ( pindex ) )
return false ;
2010-08-29 12:58:15 -04:00
// Watch for transactions paying to me
2012-07-07 18:06:34 -04:00
for ( unsigned int i = 0 ; i < vtx . size ( ) ; i + + )
SyncWithWallets ( GetTxHash ( i ) , vtx [ i ] , this , true ) ;
2010-08-29 12:58:15 -04:00
return true ;
}
2012-08-18 18:33:01 -04:00
bool SetBestChain ( CBlockIndex * pindexNew )
2010-08-29 12:58:15 -04:00
{
2012-12-01 12:46:23 -03:00
// All modifications to the coin state will be done in this cache.
// Only when all have succeeded, we push it to pcoinsTip.
CCoinsViewCache view ( * pcoinsTip , true ) ;
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
// special case for attaching the genesis block
// note that no ConnectBlock is called, so its coinbase output is non-spendable
if ( pindexGenesisBlock = = NULL & & pindexNew - > GetBlockHash ( ) = = hashGenesisBlock )
{
2012-07-06 10:33:34 -04:00
view . SetBestBlock ( pindexNew ) ;
if ( ! view . Flush ( ) )
return false ;
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
pindexGenesisBlock = pindexNew ;
pindexBest = pindexNew ;
hashBestChain = pindexNew - > GetBlockHash ( ) ;
nBestHeight = pindexBest - > nHeight ;
bnBestChainWork = pindexNew - > bnChainWork ;
return true ;
}
// Find the fork (typically, there is none)
CBlockIndex * pfork = view . GetBestBlock ( ) ;
2010-08-29 12:58:15 -04:00
CBlockIndex * plonger = pindexNew ;
2012-12-02 17:59:22 -03:00
while ( pfork & & pfork ! = plonger )
2010-08-29 12:58:15 -04:00
{
while ( plonger - > nHeight > pfork - > nHeight )
if ( ! ( plonger = plonger - > pprev ) )
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
return error ( " SetBestChain() : plonger - > pprev is null " ) ;
2010-08-29 12:58:15 -04:00
if ( pfork = = plonger )
break ;
if ( ! ( pfork = pfork - > pprev ) )
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
return error ( " SetBestChain() : pfork - > pprev is null " ) ;
2010-08-29 12:58:15 -04:00
}
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
// List of what to disconnect (typically nothing)
2010-08-29 12:58:15 -04:00
vector < CBlockIndex * > vDisconnect ;
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
for ( CBlockIndex * pindex = view . GetBestBlock ( ) ; pindex ! = pfork ; pindex = pindex - > pprev )
2010-08-29 12:58:15 -04:00
vDisconnect . push_back ( pindex ) ;
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
// List of what to connect (typically only pindexNew)
2010-08-29 12:58:15 -04:00
vector < CBlockIndex * > vConnect ;
for ( CBlockIndex * pindex = pindexNew ; pindex ! = pfork ; pindex = pindex - > pprev )
vConnect . push_back ( pindex ) ;
reverse ( vConnect . begin ( ) , vConnect . end ( ) ) ;
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
if ( vDisconnect . size ( ) > 0 ) {
2012-08-13 03:02:44 -04:00
printf ( " REORGANIZE: Disconnect % " PRIszu " blocks; %s..%s \n " , vDisconnect . size ( ) , BlockHashStr ( pfork - > GetBlockHash ( ) ) . c_str ( ) , BlockHashStr ( pindexBest - > GetBlockHash ( ) ) . c_str ( ) ) ;
printf ( " REORGANIZE: Connect % " PRIszu " blocks; %s..%s \n " , vConnect . size ( ) , BlockHashStr ( pfork - > GetBlockHash ( ) ) . c_str ( ) , BlockHashStr ( pindexNew - > GetBlockHash ( ) ) . c_str ( ) ) ;
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
}
2012-03-21 09:15:27 -03:00
2010-08-29 12:58:15 -04:00
// Disconnect shorter branch
vector < CTransaction > vResurrect ;
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
BOOST_FOREACH ( CBlockIndex * pindex , vDisconnect ) {
2010-08-29 12:58:15 -04:00
CBlock block ;
if ( ! block . ReadFromDisk ( pindex ) )
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
return error ( " SetBestBlock() : ReadFromDisk for disconnect failed " ) ;
2012-12-01 16:10:23 -03:00
int64 nStart = GetTimeMicros ( ) ;
2012-12-01 12:46:23 -03:00
if ( ! block . DisconnectBlock ( pindex , view ) )
2012-08-13 03:02:44 -04:00
return error ( " SetBestBlock() : DisconnectBlock % s failed " , BlockHashStr(pindex->GetBlockHash()).c_str()) ;
2012-12-01 16:10:23 -03:00
if ( fBenchmark )
printf ( " - Disconnect: %.2fms \n " , ( GetTimeMicros ( ) - nStart ) * 0.001 ) ;
2010-08-29 12:58:15 -04:00
2012-11-30 20:41:27 -03:00
// Queue memory transactions to resurrect.
// We only do this for blocks after the last checkpoint (reorganisation before that
// point should only happen with -reindex/-loadblock, or a misbehaving peer.
2011-05-15 03:11:04 -04:00
BOOST_FOREACH ( const CTransaction & tx , block . vtx )
2012-11-30 20:41:27 -03:00
if ( ! tx . IsCoinBase ( ) & & pindex - > nHeight > Checkpoints : : GetTotalBlocksEstimate ( ) )
2010-08-29 12:58:15 -04:00
vResurrect . push_back ( tx ) ;
}
// Connect longer branch
vector < CTransaction > vDelete ;
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
BOOST_FOREACH ( CBlockIndex * pindex , vConnect ) {
2010-08-29 12:58:15 -04:00
CBlock block ;
2012-08-18 18:33:01 -04:00
if ( ! block . ReadFromDisk ( pindex ) )
return error ( " SetBestBlock() : ReadFromDisk for connect failed " ) ;
2012-12-01 16:10:23 -03:00
int64 nStart = GetTimeMicros ( ) ;
2012-12-01 12:46:23 -03:00
if ( ! block . ConnectBlock ( pindex , view ) ) {
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
InvalidChainFound ( pindexNew ) ;
2012-08-18 18:33:01 -04:00
InvalidBlockFound ( pindex ) ;
2012-08-13 03:02:44 -04:00
return error ( " SetBestBlock() : ConnectBlock % s failed " , BlockHashStr(pindex->GetBlockHash()).c_str()) ;
2010-08-29 12:58:15 -04:00
}
2012-12-01 16:10:23 -03:00
if ( fBenchmark )
printf ( " - Connect: %.2fms \n " , ( GetTimeMicros ( ) - nStart ) * 0.001 ) ;
2010-08-29 12:58:15 -04:00
// Queue memory transactions to delete
2012-08-18 18:33:01 -04:00
BOOST_FOREACH ( const CTransaction & tx , block . vtx )
2010-08-29 12:58:15 -04:00
vDelete . push_back ( tx ) ;
}
2012-12-01 12:46:23 -03:00
// Flush changes to global coin state
2012-12-01 16:10:23 -03:00
int64 nStart = GetTimeMicros ( ) ;
int nModified = view . GetCacheSize ( ) ;
2012-12-01 12:46:23 -03:00
if ( ! view . Flush ( ) )
return error ( " SetBestBlock() : unable to modify coin state " ) ;
2012-12-01 16:10:23 -03:00
int64 nTime = GetTimeMicros ( ) - nStart ;
if ( fBenchmark )
printf ( " - Flush %i transactions: %.2fms (%.4fms/tx) \n " , nModified , 0.001 * nTime , 0.001 * nTime / nModified ) ;
2012-12-01 12:46:23 -03:00
2010-10-05 22:19:47 -04:00
// Make sure it's successfully written to disk before changing memory structure
2012-07-06 10:33:34 -04:00
bool fIsInitialDownload = IsInitialBlockDownload ( ) ;
2012-12-01 12:46:23 -03:00
if ( ! fIsInitialDownload | | pcoinsTip - > GetCacheSize ( ) > nCoinCacheSize ) {
2012-09-05 22:21:18 -03:00
FlushBlockFile ( ) ;
2012-09-03 16:14:03 -03:00
pblocktree - > Sync ( ) ;
2012-12-01 12:46:23 -03:00
if ( ! pcoinsTip - > Flush ( ) )
2012-07-06 10:33:34 -04:00
return false ;
2012-09-05 22:21:18 -03:00
}
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
// At this point, all changes have been done to the database.
// Proceed by updating the memory structures.
2010-08-29 12:58:15 -04:00
// Disconnect shorter branch
2011-05-15 03:11:04 -04:00
BOOST_FOREACH ( CBlockIndex * pindex , vDisconnect )
2010-08-29 12:58:15 -04:00
if ( pindex - > pprev )
pindex - > pprev - > pnext = NULL ;
// Connect longer branch
2011-05-15 03:11:04 -04:00
BOOST_FOREACH ( CBlockIndex * pindex , vConnect )
2010-08-29 12:58:15 -04:00
if ( pindex - > pprev )
pindex - > pprev - > pnext = pindex ;
// Resurrect memory transactions that were in the disconnected branch
2011-05-15 03:11:04 -04:00
BOOST_FOREACH ( CTransaction & tx , vResurrect )
2012-11-30 20:26:56 -03:00
tx . AcceptToMemoryPool ( ) ;
2010-08-29 12:58:15 -04:00
// Delete redundant memory transactions that are in the connected branch
2012-11-24 10:26:51 -03:00
BOOST_FOREACH ( CTransaction & tx , vDelete ) {
2012-04-13 17:28:07 -03:00
mempool . remove ( tx ) ;
2012-11-24 10:26:51 -03:00
mempool . removeConflicts ( tx ) ;
}
2010-08-29 12:58:15 -04:00
2011-04-13 11:16:30 -03:00
// Update best block in wallet (so we can detect restored wallets)
2012-01-03 17:24:28 -03:00
if ( ! fIsInitialDownload )
2011-04-13 11:16:30 -03:00
{
const CBlockLocator locator ( pindexNew ) ;
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
: : SetBestChain ( locator ) ;
2011-04-13 11:16:30 -03:00
}
2010-08-29 12:58:15 -04:00
// New best block
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
hashBestChain = pindexNew - > GetBlockHash ( ) ;
2010-08-29 12:58:15 -04:00
pindexBest = pindexNew ;
2012-07-19 16:06:20 -04:00
pblockindexFBBHLast = NULL ;
2010-08-29 12:58:15 -04:00
nBestHeight = pindexBest - > nHeight ;
bnBestChainWork = pindexNew - > bnChainWork ;
nTimeBestReceived = GetTime ( ) ;
nTransactionsUpdated + + ;
2012-08-18 18:33:01 -04:00
printf ( " SetBestChain: new best=%s height=%d work=%s tx=%lu date=%s \n " ,
2012-08-13 03:02:44 -04:00
BlockHashStr ( hashBestChain ) . c_str ( ) , nBestHeight , bnBestChainWork . ToString ( ) . c_str ( ) , ( unsigned long ) pindexNew - > nChainTx ,
2013-01-01 17:28:28 -03:00
DateTimeStrFormat ( " %Y-%m-%d %H:%M:%S " , pindexBest - > GetBlockTime ( ) ) . c_str ( ) ) ;
2010-08-29 12:58:15 -04:00
2012-06-27 13:51:51 -04:00
// Check the version of the last 100 blocks to see if we need to upgrade:
if ( ! fIsInitialDownload )
{
int nUpgraded = 0 ;
const CBlockIndex * pindex = pindexBest ;
for ( int i = 0 ; i < 100 & & pindex ! = NULL ; i + + )
{
if ( pindex - > nVersion > CBlock : : CURRENT_VERSION )
+ + nUpgraded ;
pindex = pindex - > pprev ;
}
if ( nUpgraded > 0 )
printf ( " SetBestChain: %d of last 100 blocks above version %d \n " , nUpgraded , CBlock : : CURRENT_VERSION ) ;
if ( nUpgraded > 100 / 2 )
// strMiscWarning is read by GetWarnings(), called by Qt and the JSON-RPC code to warn the user:
2012-07-27 02:36:43 -04:00
strMiscWarning = _ ( " Warning: This version is obsolete, upgrade required! " ) ;
2012-06-27 13:51:51 -04:00
}
2012-01-03 17:24:28 -03:00
std : : string strCmd = GetArg ( " -blocknotify " , " " ) ;
if ( ! fIsInitialDownload & & ! strCmd . empty ( ) )
{
boost : : replace_all ( strCmd , " %s " , hashBestChain . GetHex ( ) ) ;
boost : : thread t ( runCommand , strCmd ) ; // thread runs free
}
2010-08-29 12:58:15 -04:00
return true ;
}
2012-06-18 19:36:43 -04:00
bool CBlock : : AddToBlockIndex ( const CDiskBlockPos & pos )
2010-08-29 12:58:15 -04:00
{
// Check for duplicate
uint256 hash = GetHash ( ) ;
if ( mapBlockIndex . count ( hash ) )
2012-08-13 03:02:44 -04:00
return error ( " AddToBlockIndex() : % s already exists " , BlockHashStr(hash).c_str()) ;
2010-08-29 12:58:15 -04:00
// Construct new block index object
2012-06-18 19:36:43 -04:00
CBlockIndex * pindexNew = new CBlockIndex ( * this ) ;
2010-08-29 12:58:15 -04:00
if ( ! pindexNew )
return error ( " AddToBlockIndex() : new CBlockIndex failed " ) ;
map < uint256 , CBlockIndex * > : : iterator mi = mapBlockIndex . insert ( make_pair ( hash , pindexNew ) ) . first ;
pindexNew - > phashBlock = & ( ( * mi ) . first ) ;
map < uint256 , CBlockIndex * > : : iterator miPrev = mapBlockIndex . find ( hashPrevBlock ) ;
if ( miPrev ! = mapBlockIndex . end ( ) )
{
pindexNew - > pprev = ( * miPrev ) . second ;
pindexNew - > nHeight = pindexNew - > pprev - > nHeight + 1 ;
}
2012-08-18 18:33:01 -04:00
pindexNew - > nTx = vtx . size ( ) ;
2010-08-29 12:58:15 -04:00
pindexNew - > bnChainWork = ( pindexNew - > pprev ? pindexNew - > pprev - > bnChainWork : 0 ) + pindexNew - > GetBlockWork ( ) ;
2012-08-18 18:33:01 -04:00
pindexNew - > nChainTx = ( pindexNew - > pprev ? pindexNew - > pprev - > nChainTx : 0 ) + pindexNew - > nTx ;
pindexNew - > nFile = pos . nFile ;
pindexNew - > nDataPos = pos . nPos ;
2012-08-13 13:11:05 -04:00
pindexNew - > nUndoPos = 0 ;
2012-08-18 18:33:01 -04:00
pindexNew - > nStatus = BLOCK_VALID_TRANSACTIONS | BLOCK_HAVE_DATA ;
setBlockIndexValid . insert ( pindexNew ) ;
2010-08-29 12:58:15 -04:00
2012-09-03 10:26:57 -03:00
pblocktree - > WriteBlockIndex ( CDiskBlockIndex ( pindexNew ) ) ;
2010-08-29 12:58:15 -04:00
2012-08-18 18:33:01 -04:00
// New best?
if ( ! ConnectBestBlock ( ) )
2012-07-06 10:33:34 -04:00
return false ;
2010-08-29 12:58:15 -04:00
if ( pindexNew = = pindexBest )
{
// Notify UI to display prev block's coinbase if it was ours
static uint256 hashPrevBestCoinBase ;
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
UpdatedTransaction ( hashPrevBestCoinBase ) ;
2012-07-07 18:06:34 -04:00
hashPrevBestCoinBase = GetTxHash ( 0 ) ;
2010-08-29 12:58:15 -04:00
}
2012-09-03 10:26:57 -03:00
pblocktree - > Flush ( ) ;
2012-05-06 13:40:58 -04:00
uiInterface . NotifyBlocksChanged ( ) ;
2010-08-29 12:58:15 -04:00
return true ;
}
2012-10-21 16:23:13 -03:00
bool FindBlockPos ( CDiskBlockPos & pos , unsigned int nAddSize , unsigned int nHeight , uint64 nTime , bool fKnown = false )
2012-08-13 13:11:05 -04:00
{
bool fUpdatedLast = false ;
LOCK ( cs_LastBlockFile ) ;
2012-10-21 16:23:13 -03:00
if ( fKnown ) {
if ( nLastBlockFile ! = pos . nFile ) {
nLastBlockFile = pos . nFile ;
infoLastBlockFile . SetNull ( ) ;
pblocktree - > ReadBlockFileInfo ( nLastBlockFile , infoLastBlockFile ) ;
2012-12-04 19:46:30 -03:00
fUpdatedLast = true ;
2012-10-21 16:23:13 -03:00
}
} else {
while ( infoLastBlockFile . nSize + nAddSize > = MAX_BLOCKFILE_SIZE ) {
printf ( " Leaving block file %i: %s \n " , nLastBlockFile , infoLastBlockFile . ToString ( ) . c_str ( ) ) ;
FlushBlockFile ( ) ;
nLastBlockFile + + ;
infoLastBlockFile . SetNull ( ) ;
pblocktree - > ReadBlockFileInfo ( nLastBlockFile , infoLastBlockFile ) ; // check whether data for the new file somehow already exist; can fail just fine
fUpdatedLast = true ;
}
pos . nFile = nLastBlockFile ;
pos . nPos = infoLastBlockFile . nSize ;
2012-08-13 13:11:05 -04:00
}
infoLastBlockFile . nSize + = nAddSize ;
infoLastBlockFile . AddBlock ( nHeight , nTime ) ;
2012-10-21 16:23:13 -03:00
if ( ! fKnown ) {
unsigned int nOldChunks = ( pos . nPos + BLOCKFILE_CHUNK_SIZE - 1 ) / BLOCKFILE_CHUNK_SIZE ;
unsigned int nNewChunks = ( infoLastBlockFile . nSize + BLOCKFILE_CHUNK_SIZE - 1 ) / BLOCKFILE_CHUNK_SIZE ;
if ( nNewChunks > nOldChunks ) {
2012-12-04 03:48:57 -03:00
if ( CheckDiskSpace ( nNewChunks * BLOCKFILE_CHUNK_SIZE - pos . nPos ) ) {
FILE * file = OpenBlockFile ( pos ) ;
if ( file ) {
printf ( " Pre-allocating up to position 0x%x in blk%05u.dat \n " , nNewChunks * BLOCKFILE_CHUNK_SIZE , pos . nFile ) ;
AllocateFileRange ( file , pos . nPos , nNewChunks * BLOCKFILE_CHUNK_SIZE - pos . nPos ) ;
fclose ( file ) ;
}
2012-10-21 16:23:13 -03:00
}
2012-12-04 03:48:57 -03:00
else
return error ( " FindBlockPos() : out of disk space " ) ;
2012-08-15 20:21:28 -04:00
}
}
2012-09-03 10:26:57 -03:00
if ( ! pblocktree - > WriteBlockFileInfo ( nLastBlockFile , infoLastBlockFile ) )
2012-08-13 13:11:05 -04:00
return error ( " FindBlockPos() : cannot write updated block info " ) ;
if ( fUpdatedLast )
2012-09-03 10:26:57 -03:00
pblocktree - > WriteLastBlockFile ( nLastBlockFile ) ;
2012-08-13 13:11:05 -04:00
return true ;
}
2012-09-03 10:26:57 -03:00
bool FindUndoPos ( int nFile , CDiskBlockPos & pos , unsigned int nAddSize )
2012-08-13 13:11:05 -04:00
{
pos . nFile = nFile ;
LOCK ( cs_LastBlockFile ) ;
2012-08-15 20:21:28 -04:00
unsigned int nNewSize ;
2012-08-13 13:11:05 -04:00
if ( nFile = = nLastBlockFile ) {
pos . nPos = infoLastBlockFile . nUndoSize ;
2012-08-15 20:21:28 -04:00
nNewSize = ( infoLastBlockFile . nUndoSize + = nAddSize ) ;
2012-09-03 10:26:57 -03:00
if ( ! pblocktree - > WriteBlockFileInfo ( nLastBlockFile , infoLastBlockFile ) )
2012-08-13 13:11:05 -04:00
return error ( " FindUndoPos() : cannot write updated block info " ) ;
2012-08-15 20:21:28 -04:00
} else {
CBlockFileInfo info ;
2012-09-03 10:26:57 -03:00
if ( ! pblocktree - > ReadBlockFileInfo ( nFile , info ) )
2012-08-15 20:21:28 -04:00
return error ( " FindUndoPos() : cannot read block info " ) ;
pos . nPos = info . nUndoSize ;
nNewSize = ( info . nUndoSize + = nAddSize ) ;
2012-09-03 10:26:57 -03:00
if ( ! pblocktree - > WriteBlockFileInfo ( nFile , info ) )
2012-08-15 20:21:28 -04:00
return error ( " FindUndoPos() : cannot write updated block info " ) ;
}
unsigned int nOldChunks = ( pos . nPos + UNDOFILE_CHUNK_SIZE - 1 ) / UNDOFILE_CHUNK_SIZE ;
unsigned int nNewChunks = ( nNewSize + UNDOFILE_CHUNK_SIZE - 1 ) / UNDOFILE_CHUNK_SIZE ;
if ( nNewChunks > nOldChunks ) {
2012-12-04 03:48:57 -03:00
if ( CheckDiskSpace ( nNewChunks * UNDOFILE_CHUNK_SIZE - pos . nPos ) ) {
FILE * file = OpenUndoFile ( pos ) ;
if ( file ) {
printf ( " Pre-allocating up to position 0x%x in rev%05u.dat \n " , nNewChunks * UNDOFILE_CHUNK_SIZE , pos . nFile ) ;
AllocateFileRange ( file , pos . nPos , nNewChunks * UNDOFILE_CHUNK_SIZE - pos . nPos ) ;
fclose ( file ) ;
}
2012-08-15 20:21:28 -04:00
}
2012-12-04 03:48:57 -03:00
else
return error ( " FindUndoPos() : out of disk space " ) ;
2012-08-13 13:11:05 -04:00
}
return true ;
}
2010-08-29 12:58:15 -04:00
2012-05-09 13:24:44 -04:00
bool CBlock : : CheckBlock ( bool fCheckPOW , bool fCheckMerkleRoot ) const
2010-08-29 12:58:15 -04:00
{
// These are checks that are independent of context
// that can be verified before saving an orphan block.
// Size limits
2012-04-16 09:56:45 -03:00
if ( vtx . empty ( ) | | vtx . size ( ) > MAX_BLOCK_SIZE | | : : GetSerializeSize ( * this , SER_NETWORK , PROTOCOL_VERSION ) > MAX_BLOCK_SIZE )
2011-09-06 17:59:38 -03:00
return DoS ( 100 , error ( " CheckBlock() : size limits failed " )) ;
2010-08-29 12:58:15 -04:00
2010-09-19 17:20:34 -04:00
// Check proof of work matches claimed amount
2012-05-09 13:24:44 -04:00
if ( fCheckPOW & & ! CheckProofOfWork ( GetHash ( ) , nBits ) )
2011-09-06 17:59:38 -03:00
return DoS ( 50 , error ( " CheckBlock() : proof of work failed " )) ;
2010-09-19 17:20:34 -04:00
2010-08-29 12:58:15 -04:00
// Check timestamp
if ( GetBlockTime ( ) > GetAdjustedTime ( ) + 2 * 60 * 60 )
return error ( " CheckBlock() : block timestamp too far in the future " ) ;
// First transaction must be coinbase, the rest must not be
if ( vtx . empty ( ) | | ! vtx [ 0 ] . IsCoinBase ( ) )
2011-09-06 17:59:38 -03:00
return DoS ( 100 , error ( " CheckBlock() : first tx is not coinbase " )) ;
2012-04-15 17:52:09 -03:00
for ( unsigned int i = 1 ; i < vtx . size ( ) ; i + + )
2010-08-29 12:58:15 -04:00
if ( vtx [ i ] . IsCoinBase ( ) )
2011-09-06 17:59:38 -03:00
return DoS ( 100 , error ( " CheckBlock() : more than one coinbase " )) ;
2010-08-29 12:58:15 -04:00
// Check transactions
2011-05-15 03:11:04 -04:00
BOOST_FOREACH ( const CTransaction & tx , vtx )
2010-08-29 12:58:15 -04:00
if ( ! tx . CheckTransaction ( ) )
2011-09-06 17:59:38 -03:00
return DoS ( tx . nDoS , error ( " CheckBlock() : CheckTransaction failed " )) ;
2010-08-29 12:58:15 -04:00
2012-10-22 19:21:16 -03:00
// Build the merkle tree already. We need it anyway later, and it makes the
// block cache the transaction hashes, which means they don't need to be
// recalculated many times during this block's validation.
BuildMerkleTree ( ) ;
2012-04-29 20:56:55 -04:00
// Check for duplicate txids. This is caught by ConnectInputs(),
// but catching it earlier avoids a potential DoS attack:
set < uint256 > uniqueTx ;
2012-07-07 18:06:34 -04:00
for ( unsigned int i = 0 ; i < vtx . size ( ) ; i + + ) {
uniqueTx . insert ( GetTxHash ( i ) ) ;
2012-04-29 20:56:55 -04:00
}
if ( uniqueTx . size ( ) ! = vtx . size ( ) )
return DoS ( 100 , error ( " CheckBlock() : duplicate transaction " )) ;
2012-04-23 15:14:03 -03:00
unsigned int nSigOps = 0 ;
2011-10-03 14:05:43 -03:00
BOOST_FOREACH ( const CTransaction & tx , vtx )
{
2012-01-04 23:40:52 -03:00
nSigOps + = tx . GetLegacySigOpCount ( ) ;
2011-10-03 14:05:43 -03:00
}
if ( nSigOps > MAX_BLOCK_SIGOPS )
2011-09-06 17:59:38 -03:00
return DoS ( 100 , error ( " CheckBlock() : out - of - bounds SigOpCount " )) ;
2010-08-29 12:58:15 -04:00
2012-07-25 20:48:39 -04:00
// Check merkle root
2012-05-09 13:24:44 -04:00
if ( fCheckMerkleRoot & & hashMerkleRoot ! = BuildMerkleTree ( ) )
2011-09-06 17:59:38 -03:00
return DoS ( 100 , error ( " CheckBlock() : hashMerkleRoot mismatch " )) ;
2010-08-29 12:58:15 -04:00
return true ;
}
2012-10-21 16:23:13 -03:00
bool CBlock : : AcceptBlock ( CDiskBlockPos * dbp )
2010-08-29 12:58:15 -04:00
{
// Check for duplicate
uint256 hash = GetHash ( ) ;
if ( mapBlockIndex . count ( hash ) )
return error ( " AcceptBlock() : block already in mapBlockIndex " ) ;
// Get prev block index
2012-10-21 16:23:13 -03:00
CBlockIndex * pindexPrev = NULL ;
int nHeight = 0 ;
if ( hash ! = hashGenesisBlock ) {
2012-11-10 10:26:34 -03:00
map < uint256 , CBlockIndex * > : : iterator mi = mapBlockIndex . find ( hashPrevBlock ) ;
if ( mi = = mapBlockIndex . end ( ) )
return DoS ( 10 , error ( " AcceptBlock() : prev block not found " )) ;
pindexPrev = ( * mi ) . second ;
nHeight = pindexPrev - > nHeight + 1 ;
// Check proof of work
if ( nBits ! = GetNextWorkRequired ( pindexPrev , this ) )
return DoS ( 100 , error ( " AcceptBlock() : incorrect proof of work " )) ;
// Check timestamp against prev
if ( GetBlockTime ( ) < = pindexPrev - > GetMedianTimePast ( ) )
return error ( " AcceptBlock() : block ' s timestamp is too early " ) ;
// Check that all transactions are finalized
BOOST_FOREACH ( const CTransaction & tx , vtx )
if ( ! tx . IsFinal ( nHeight , GetBlockTime ( ) ) )
return DoS ( 10 , error ( " AcceptBlock() : contains a non - final transaction " )) ;
// Check that the block chain matches the known block chain up to a checkpoint
if ( ! Checkpoints : : CheckBlock ( nHeight , hash ) )
return DoS ( 100 , error ( " AcceptBlock() : rejected by checkpoint lock - in at % d " , nHeight)) ;
// Reject block.nVersion=1 blocks when 95% (75% on testnet) of the network has upgraded:
if ( nVersion < 2 )
2012-07-05 21:22:16 -04:00
{
2012-11-10 10:26:34 -03:00
if ( ( ! fTestNet & & CBlockIndex : : IsSuperMajority ( 2 , pindexPrev , 950 , 1000 ) ) | |
( fTestNet & & CBlockIndex : : IsSuperMajority ( 2 , pindexPrev , 75 , 100 ) ) )
{
return error ( " AcceptBlock() : rejected nVersion = 1 block " ) ;
}
2012-07-05 21:22:16 -04:00
}
2012-11-10 10:26:34 -03:00
// Enforce block.nVersion=2 rule that the coinbase starts with serialized block height
if ( nVersion > = 2 )
2012-06-27 19:30:39 -04:00
{
2012-11-10 10:26:34 -03:00
// if 750 of the last 1,000 blocks are version 2 or greater (51/100 if testnet):
if ( ( ! fTestNet & & CBlockIndex : : IsSuperMajority ( 2 , pindexPrev , 750 , 1000 ) ) | |
( fTestNet & & CBlockIndex : : IsSuperMajority ( 2 , pindexPrev , 51 , 100 ) ) )
{
CScript expect = CScript ( ) < < nHeight ;
if ( ! std : : equal ( expect . begin ( ) , expect . end ( ) , vtx [ 0 ] . vin [ 0 ] . scriptSig . begin ( ) ) )
return DoS ( 100 , error ( " AcceptBlock() : block height mismatch in coinbase " )) ;
}
2012-06-27 19:30:39 -04:00
}
}
2010-08-29 12:58:15 -04:00
// Write block to history file
2012-08-13 13:11:05 -04:00
unsigned int nBlockSize = : : GetSerializeSize ( * this , SER_DISK , CLIENT_VERSION ) ;
CDiskBlockPos blockPos ;
2012-12-04 03:48:57 -03:00
if ( dbp ! = NULL )
2012-10-21 16:23:13 -03:00
blockPos = * dbp ;
if ( ! FindBlockPos ( blockPos , nBlockSize + 8 , nHeight , nTime , dbp ! = NULL ) )
2012-09-03 10:26:57 -03:00
return error ( " AcceptBlock() : FindBlockPos failed " ) ;
2012-10-21 16:23:13 -03:00
if ( dbp = = NULL )
if ( ! WriteToDisk ( blockPos ) )
return error ( " AcceptBlock() : WriteToDisk failed " ) ;
2012-06-18 19:36:43 -04:00
if ( ! AddToBlockIndex ( blockPos ) )
2010-08-29 12:58:15 -04:00
return error ( " AcceptBlock() : AddToBlockIndex failed " ) ;
// Relay inventory, but don't relay old inventory during initial block download
2012-03-20 14:45:45 -03:00
int nBlockEstimate = Checkpoints : : GetTotalBlocksEstimate ( ) ;
2010-08-29 12:58:15 -04:00
if ( hashBestChain = = hash )
2012-04-06 13:39:12 -03:00
{
LOCK ( cs_vNodes ) ;
BOOST_FOREACH ( CNode * pnode , vNodes )
if ( nBestHeight > ( pnode - > nStartingHeight ! = - 1 ? pnode - > nStartingHeight - 2000 : nBlockEstimate ) )
pnode - > PushInventory ( CInv ( MSG_BLOCK , hash ) ) ;
}
2010-08-29 12:58:15 -04:00
return true ;
}
2012-06-27 19:30:39 -04:00
bool CBlockIndex : : IsSuperMajority ( int minVersion , const CBlockIndex * pstart , unsigned int nRequired , unsigned int nToCheck )
{
unsigned int nFound = 0 ;
for ( unsigned int i = 0 ; i < nToCheck & & nFound < nRequired & & pstart ! = NULL ; i + + )
{
if ( pstart - > nVersion > = minVersion )
+ + nFound ;
pstart = pstart - > pprev ;
}
return ( nFound > = nRequired ) ;
}
2012-10-21 16:23:13 -03:00
bool ProcessBlock ( CNode * pfrom , CBlock * pblock , CDiskBlockPos * dbp )
2010-08-29 12:58:15 -04:00
{
// Check for duplicate
uint256 hash = pblock - > GetHash ( ) ;
if ( mapBlockIndex . count ( hash ) )
2012-08-13 03:02:44 -04:00
return error ( " ProcessBlock() : already have block % d % s " , mapBlockIndex[hash]->nHeight, BlockHashStr(hash).c_str()) ;
2010-08-29 12:58:15 -04:00
if ( mapOrphanBlocks . count ( hash ) )
2012-08-13 03:02:44 -04:00
return error ( " ProcessBlock() : already have block ( orphan ) % s " , BlockHashStr(hash).c_str()) ;
2010-08-29 12:58:15 -04:00
// Preliminary checks
if ( ! pblock - > CheckBlock ( ) )
return error ( " ProcessBlock() : CheckBlock FAILED " ) ;
2011-09-08 13:51:43 -03:00
CBlockIndex * pcheckpoint = Checkpoints : : GetLastCheckpoint ( mapBlockIndex ) ;
if ( pcheckpoint & & pblock - > hashPrevBlock ! = hashBestChain )
{
// Extra checks to prevent "fill up memory by spamming with bogus blocks"
2011-12-21 18:33:19 -03:00
int64 deltaTime = pblock - > GetBlockTime ( ) - pcheckpoint - > nTime ;
2011-09-08 13:51:43 -03:00
if ( deltaTime < 0 )
{
2012-02-20 18:35:08 -03:00
if ( pfrom )
pfrom - > Misbehaving ( 100 ) ;
2011-09-08 13:51:43 -03:00
return error ( " ProcessBlock() : block with timestamp before last checkpoint " ) ;
}
CBigNum bnNewBlock ;
bnNewBlock . SetCompact ( pblock - > nBits ) ;
CBigNum bnRequired ;
bnRequired . SetCompact ( ComputeMinWork ( pcheckpoint - > nBits , deltaTime ) ) ;
if ( bnNewBlock > bnRequired )
{
2012-02-20 18:35:08 -03:00
if ( pfrom )
pfrom - > Misbehaving ( 100 ) ;
2011-09-08 13:51:43 -03:00
return error ( " ProcessBlock() : block with too little proof - of - work " ) ;
}
}
2012-08-21 13:18:53 -04:00
// If we don't already have its previous block, shunt it off to holding area until we get it
2012-10-21 16:23:13 -03:00
if ( pblock - > hashPrevBlock ! = 0 & & ! mapBlockIndex . count ( pblock - > hashPrevBlock ) )
2010-08-29 12:58:15 -04:00
{
2012-08-13 03:02:44 -04:00
printf ( " ProcessBlock: ORPHAN BLOCK, prev=%s \n " , BlockHashStr ( pblock - > hashPrevBlock ) . c_str ( ) ) ;
2010-08-29 12:58:15 -04:00
2012-08-21 13:18:53 -04:00
// Accept orphans as long as there is a node to request its parents from
if ( pfrom ) {
CBlock * pblock2 = new CBlock ( * pblock ) ;
mapOrphanBlocks . insert ( make_pair ( hash , pblock2 ) ) ;
mapOrphanBlocksByPrev . insert ( make_pair ( pblock2 - > hashPrevBlock , pblock2 ) ) ;
// Ask this guy to fill in what we're missing
2010-11-23 16:16:36 -03:00
pfrom - > PushGetBlocks ( pindexBest , GetOrphanRoot ( pblock2 ) ) ;
2012-08-21 13:18:53 -04:00
}
2010-08-29 12:58:15 -04:00
return true ;
}
// Store to disk
2012-10-21 16:23:13 -03:00
if ( ! pblock - > AcceptBlock ( dbp ) )
2010-08-29 12:58:15 -04:00
return error ( " ProcessBlock() : AcceptBlock FAILED " ) ;
// Recursively process any orphan blocks that depended on this one
vector < uint256 > vWorkQueue ;
vWorkQueue . push_back ( hash ) ;
2012-04-15 17:52:09 -03:00
for ( unsigned int i = 0 ; i < vWorkQueue . size ( ) ; i + + )
2010-08-29 12:58:15 -04:00
{
uint256 hashPrev = vWorkQueue [ i ] ;
for ( multimap < uint256 , CBlock * > : : iterator mi = mapOrphanBlocksByPrev . lower_bound ( hashPrev ) ;
mi ! = mapOrphanBlocksByPrev . upper_bound ( hashPrev ) ;
+ + mi )
{
CBlock * pblockOrphan = ( * mi ) . second ;
if ( pblockOrphan - > AcceptBlock ( ) )
vWorkQueue . push_back ( pblockOrphan - > GetHash ( ) ) ;
mapOrphanBlocks . erase ( pblockOrphan - > GetHash ( ) ) ;
delete pblockOrphan ;
}
mapOrphanBlocksByPrev . erase ( hashPrev ) ;
}
printf ( " ProcessBlock: ACCEPTED \n " ) ;
return true ;
}
2011-12-21 18:33:19 -03:00
bool CheckDiskSpace ( uint64 nAdditionalBytes )
2010-08-29 12:58:15 -04:00
{
2011-12-21 18:33:19 -03:00
uint64 nFreeBytesAvailable = filesystem : : space ( GetDataDir ( ) ) . available ;
2010-08-29 12:58:15 -04:00
2012-05-14 01:49:17 -04:00
// Check for nMinDiskSpace bytes (currently 50MB)
if ( nFreeBytesAvailable < nMinDiskSpace + nAdditionalBytes )
2010-08-29 12:58:15 -04:00
{
fShutdown = true ;
2012-11-05 04:04:21 -03:00
string strMessage = _ ( " Error: Disk space is low! " ) ;
2010-08-29 12:58:15 -04:00
strMiscWarning = strMessage ;
printf ( " *** %s \n " , strMessage . c_str ( ) ) ;
2012-11-05 04:04:21 -03:00
uiInterface . ThreadSafeMessageBox ( strMessage , " " , CClientUIInterface : : MSG_ERROR ) ;
2012-06-11 01:40:14 -04:00
StartShutdown ( ) ;
2010-08-29 12:58:15 -04:00
return false ;
}
return true ;
}
2012-08-13 13:11:05 -04:00
CCriticalSection cs_LastBlockFile ;
CBlockFileInfo infoLastBlockFile ;
int nLastBlockFile = 0 ;
FILE * OpenDiskFile ( const CDiskBlockPos & pos , const char * prefix , bool fReadOnly )
2012-09-04 22:40:26 -03:00
{
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
if ( pos . IsNull ( ) )
2010-08-29 12:58:15 -04:00
return NULL ;
2012-08-13 13:11:05 -04:00
boost : : filesystem : : path path = GetDataDir ( ) / " blocks " / strprintf ( " %s%05u.dat " , prefix , pos . nFile ) ;
boost : : filesystem : : create_directories ( path . parent_path ( ) ) ;
FILE * file = fopen ( path . string ( ) . c_str ( ) , " rb+ " ) ;
if ( ! file & & ! fReadOnly )
file = fopen ( path . string ( ) . c_str ( ) , " wb+ " ) ;
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
if ( ! file ) {
printf ( " Unable to open file %s \n " , path . string ( ) . c_str ( ) ) ;
2010-08-29 12:58:15 -04:00
return NULL ;
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
}
2012-08-13 13:11:05 -04:00
if ( pos . nPos ) {
if ( fseek ( file , pos . nPos , SEEK_SET ) ) {
printf ( " Unable to seek to position %u of %s \n " , pos . nPos , path . string ( ) . c_str ( ) ) ;
fclose ( file ) ;
return NULL ;
}
}
2010-08-29 12:58:15 -04:00
return file ;
}
2012-08-13 13:11:05 -04:00
FILE * OpenBlockFile ( const CDiskBlockPos & pos , bool fReadOnly ) {
return OpenDiskFile ( pos , " blk " , fReadOnly ) ;
}
FILE * OpenUndoFile ( const CDiskBlockPos & pos , bool fReadOnly ) {
return OpenDiskFile ( pos , " rev " , fReadOnly ) ;
}
2012-09-03 16:14:03 -03:00
CBlockIndex * InsertBlockIndex ( uint256 hash )
{
if ( hash = = 0 )
return NULL ;
// Return existing
map < uint256 , CBlockIndex * > : : iterator mi = mapBlockIndex . find ( hash ) ;
if ( mi ! = mapBlockIndex . end ( ) )
return ( * mi ) . second ;
// Create new
CBlockIndex * pindexNew = new CBlockIndex ( ) ;
if ( ! pindexNew )
throw runtime_error ( " LoadBlockIndex() : new CBlockIndex failed " ) ;
mi = mapBlockIndex . insert ( make_pair ( hash , pindexNew ) ) . first ;
pindexNew - > phashBlock = & ( ( * mi ) . first ) ;
return pindexNew ;
}
bool static LoadBlockIndexDB ( )
{
if ( ! pblocktree - > LoadBlockIndexGuts ( ) )
return false ;
if ( fRequestShutdown )
return true ;
// Calculate bnChainWork
vector < pair < int , CBlockIndex * > > vSortedByHeight ;
vSortedByHeight . reserve ( mapBlockIndex . size ( ) ) ;
BOOST_FOREACH ( const PAIRTYPE ( uint256 , CBlockIndex * ) & item , mapBlockIndex )
{
CBlockIndex * pindex = item . second ;
vSortedByHeight . push_back ( make_pair ( pindex - > nHeight , pindex ) ) ;
}
sort ( vSortedByHeight . begin ( ) , vSortedByHeight . end ( ) ) ;
BOOST_FOREACH ( const PAIRTYPE ( int , CBlockIndex * ) & item , vSortedByHeight )
{
CBlockIndex * pindex = item . second ;
pindex - > bnChainWork = ( pindex - > pprev ? pindex - > pprev - > bnChainWork : 0 ) + pindex - > GetBlockWork ( ) ;
pindex - > nChainTx = ( pindex - > pprev ? pindex - > pprev - > nChainTx : 0 ) + pindex - > nTx ;
if ( ( pindex - > nStatus & BLOCK_VALID_MASK ) > = BLOCK_VALID_TRANSACTIONS & & ! ( pindex - > nStatus & BLOCK_FAILED_MASK ) )
setBlockIndexValid . insert ( pindex ) ;
}
// Load block file info
pblocktree - > ReadLastBlockFile ( nLastBlockFile ) ;
printf ( " LoadBlockIndex(): last block file = %i \n " , nLastBlockFile ) ;
if ( pblocktree - > ReadBlockFileInfo ( nLastBlockFile , infoLastBlockFile ) )
printf ( " LoadBlockIndex(): last block file: %s \n " , infoLastBlockFile . ToString ( ) . c_str ( ) ) ;
2012-10-05 14:22:21 -03:00
2012-12-02 17:59:22 -03:00
// Load bnBestInvalidWork, OK if it doesn't exist
pblocktree - > ReadBestInvalidWork ( bnBestInvalidWork ) ;
// Check whether we need to continue reindexing
bool fReindexing = false ;
pblocktree - > ReadReindexing ( fReindexing ) ;
fReindex | = fReindexing ;
2012-09-03 16:14:03 -03:00
// Load hashBestChain pointer to end of best chain
pindexBest = pcoinsTip - > GetBestBlock ( ) ;
if ( pindexBest = = NULL )
2012-12-02 17:59:22 -03:00
return true ;
2012-09-03 16:14:03 -03:00
hashBestChain = pindexBest - > GetBlockHash ( ) ;
nBestHeight = pindexBest - > nHeight ;
bnBestChainWork = pindexBest - > bnChainWork ;
// set 'next' pointers in best chain
CBlockIndex * pindex = pindexBest ;
while ( pindex ! = NULL & & pindex - > pprev ! = NULL ) {
CBlockIndex * pindexPrev = pindex - > pprev ;
pindexPrev - > pnext = pindex ;
pindex = pindexPrev ;
}
printf ( " LoadBlockIndex(): hashBestChain=%s height=%d date=%s \n " ,
2012-08-13 03:02:44 -04:00
BlockHashStr ( hashBestChain ) . c_str ( ) , nBestHeight ,
2013-01-01 17:28:28 -03:00
DateTimeStrFormat ( " %Y-%m-%d %H:%M:%S " , pindexBest - > GetBlockTime ( ) ) . c_str ( ) ) ;
2012-09-03 16:14:03 -03:00
// Verify blocks in the best chain
int nCheckLevel = GetArg ( " -checklevel " , 1 ) ;
int nCheckDepth = GetArg ( " -checkblocks " , 2500 ) ;
if ( nCheckDepth = = 0 )
nCheckDepth = 1000000000 ; // suffices until the year 19000
if ( nCheckDepth > nBestHeight )
nCheckDepth = nBestHeight ;
printf ( " Verifying last %i blocks at level %i \n " , nCheckDepth , nCheckLevel ) ;
CBlockIndex * pindexFork = NULL ;
for ( CBlockIndex * pindex = pindexBest ; pindex & & pindex - > pprev ; pindex = pindex - > pprev )
{
if ( fRequestShutdown | | pindex - > nHeight < nBestHeight - nCheckDepth )
break ;
CBlock block ;
if ( ! block . ReadFromDisk ( pindex ) )
return error ( " LoadBlockIndex() : block . ReadFromDisk failed " ) ;
// check level 1: verify block validity
if ( nCheckLevel > 0 & & ! block . CheckBlock ( ) )
{
printf ( " LoadBlockIndex() : *** found bad block at %d, hash=%s \n " , pindex - > nHeight , pindex - > GetBlockHash ( ) . ToString ( ) . c_str ( ) ) ;
pindexFork = pindex - > pprev ;
}
// TODO: stronger verifications
}
if ( pindexFork & & ! fRequestShutdown )
{
// TODO: reorg back
return error ( " LoadBlockIndex() : chain database corrupted " ) ;
}
return true ;
}
2012-10-21 16:23:13 -03:00
bool LoadBlockIndex ( )
2010-08-29 12:58:15 -04:00
{
2010-10-19 14:16:51 -03:00
if ( fTestNet )
{
2012-07-26 17:20:20 -04:00
pchMessageStart [ 0 ] = 0x0b ;
pchMessageStart [ 1 ] = 0x11 ;
pchMessageStart [ 2 ] = 0x09 ;
pchMessageStart [ 3 ] = 0x07 ;
2012-04-12 22:26:35 -03:00
hashGenesisBlock = uint256 ( " 000000000933ea01ad0ee984209779baaec3ced90fa3f408719526f8d77f4943 " ) ;
2010-10-19 14:16:51 -03:00
}
2012-10-21 16:23:13 -03:00
if ( fReindex )
return true ;
2010-08-29 12:58:15 -04:00
//
2012-09-03 10:26:57 -03:00
// Load block index from databases
2010-08-29 12:58:15 -04:00
//
2012-09-03 10:26:57 -03:00
if ( ! LoadBlockIndexDB ( ) )
2010-08-29 12:58:15 -04:00
return false ;
//
// Init with genesis block
//
if ( mapBlockIndex . empty ( ) )
{
// Genesis Block:
// CBlock(hash=000000000019d6, ver=1, hashPrevBlock=00000000000000, hashMerkleRoot=4a5e1e, nTime=1231006505, nBits=1d00ffff, nNonce=2083236893, vtx=1)
// CTransaction(hash=4a5e1e, ver=1, vin.size=1, vout.size=1, nLockTime=0)
// CTxIn(COutPoint(000000, -1), coinbase 04ffff001d0104455468652054696d65732030332f4a616e2f32303039204368616e63656c6c6f72206f6e206272696e6b206f66207365636f6e64206261696c6f757420666f722062616e6b73)
// CTxOut(nValue=50.00000000, scriptPubKey=0x5F1DF16B2B704C8A578D0B)
// vMerkleTree: 4a5e1e
// Genesis block
const char * pszTimestamp = " The Times 03/Jan/2009 Chancellor on brink of second bailout for banks " ;
CTransaction txNew ;
txNew . vin . resize ( 1 ) ;
txNew . vout . resize ( 1 ) ;
txNew . vin [ 0 ] . scriptSig = CScript ( ) < < 486604799 < < CBigNum ( 4 ) < < vector < unsigned char > ( ( const unsigned char * ) pszTimestamp , ( const unsigned char * ) pszTimestamp + strlen ( pszTimestamp ) ) ;
txNew . vout [ 0 ] . nValue = 50 * COIN ;
txNew . vout [ 0 ] . scriptPubKey = CScript ( ) < < ParseHex ( " 04678afdb0fe5548271967f1a67130b7105cd6a828e03909a67962e0ea1f61deb649f6bc3f4cef38c4f35504e51ec112de5c384df7ba0b8d578a4c702b6bf11d5f " ) < < OP_CHECKSIG ;
CBlock block ;
block . vtx . push_back ( txNew ) ;
block . hashPrevBlock = 0 ;
block . hashMerkleRoot = block . BuildMerkleTree ( ) ;
block . nVersion = 1 ;
block . nTime = 1231006505 ;
block . nBits = 0x1d00ffff ;
block . nNonce = 2083236893 ;
2010-10-19 14:16:51 -03:00
if ( fTestNet )
{
2011-02-03 10:21:21 -03:00
block . nTime = 1296688602 ;
2012-04-12 22:26:35 -03:00
block . nNonce = 414098458 ;
2010-10-19 14:16:51 -03:00
}
2010-08-29 12:58:15 -04:00
2010-10-19 14:16:51 -03:00
//// debug print
2012-06-18 19:36:43 -04:00
uint256 hash = block . GetHash ( ) ;
printf ( " %s \n " , hash . ToString ( ) . c_str ( ) ) ;
2010-10-19 14:16:51 -03:00
printf ( " %s \n " , hashGenesisBlock . ToString ( ) . c_str ( ) ) ;
printf ( " %s \n " , block . hashMerkleRoot . ToString ( ) . c_str ( ) ) ;
assert ( block . hashMerkleRoot = = uint256 ( " 0x4a5e1e4baab89f3a32518a88c31bc87f618f76673e2cc77ab2127b7afdeda33b " ) ) ;
block . print ( ) ;
2012-06-18 19:36:43 -04:00
assert ( hash = = hashGenesisBlock ) ;
2010-08-29 12:58:15 -04:00
// Start new block file
2012-08-13 13:11:05 -04:00
unsigned int nBlockSize = : : GetSerializeSize ( block , SER_DISK , CLIENT_VERSION ) ;
CDiskBlockPos blockPos ;
2012-09-03 10:26:57 -03:00
if ( ! FindBlockPos ( blockPos , nBlockSize + 8 , 0 , block . nTime ) )
return error ( " AcceptBlock() : FindBlockPos failed " ) ;
2012-06-18 19:36:43 -04:00
if ( ! block . WriteToDisk ( blockPos ) )
2010-08-29 12:58:15 -04:00
return error ( " LoadBlockIndex() : writing genesis block to disk failed " ) ;
2012-06-18 19:36:43 -04:00
if ( ! block . AddToBlockIndex ( blockPos ) )
2010-08-29 12:58:15 -04:00
return error ( " LoadBlockIndex() : genesis block not accepted " ) ;
}
return true ;
}
void PrintBlockTree ( )
{
2012-07-25 20:48:39 -04:00
// pre-compute tree structure
2010-08-29 12:58:15 -04:00
map < CBlockIndex * , vector < CBlockIndex * > > mapNext ;
for ( map < uint256 , CBlockIndex * > : : iterator mi = mapBlockIndex . begin ( ) ; mi ! = mapBlockIndex . end ( ) ; + + mi )
{
CBlockIndex * pindex = ( * mi ) . second ;
mapNext [ pindex - > pprev ] . push_back ( pindex ) ;
// test
//while (rand() % 3 == 0)
// mapNext[pindex->pprev].push_back(pindex);
}
vector < pair < int , CBlockIndex * > > vStack ;
vStack . push_back ( make_pair ( 0 , pindexGenesisBlock ) ) ;
int nPrevCol = 0 ;
while ( ! vStack . empty ( ) )
{
int nCol = vStack . back ( ) . first ;
CBlockIndex * pindex = vStack . back ( ) . second ;
vStack . pop_back ( ) ;
// print split or gap
if ( nCol > nPrevCol )
{
for ( int i = 0 ; i < nCol - 1 ; i + + )
printf ( " | " ) ;
printf ( " | \\ \n " ) ;
}
else if ( nCol < nPrevCol )
{
for ( int i = 0 ; i < nCol ; i + + )
printf ( " | " ) ;
printf ( " | \n " ) ;
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
}
2010-08-29 12:58:15 -04:00
nPrevCol = nCol ;
// print columns
for ( int i = 0 ; i < nCol ; i + + )
printf ( " | " ) ;
// print item
CBlock block ;
block . ReadFromDisk ( pindex ) ;
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
printf ( " %d (blk%05u.dat:0x%x) %s tx % " PRIszu " " ,
2010-08-29 12:58:15 -04:00
pindex - > nHeight ,
2012-08-13 13:11:05 -04:00
pindex - > GetBlockPos ( ) . nFile , pindex - > GetBlockPos ( ) . nPos ,
2013-01-01 17:28:28 -03:00
DateTimeStrFormat ( " %Y-%m-%d %H:%M:%S " , block . GetBlockTime ( ) ) . c_str ( ) ,
2010-08-29 12:58:15 -04:00
block . vtx . size ( ) ) ;
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
PrintWallets ( block ) ;
2010-08-29 12:58:15 -04:00
2012-07-25 20:48:39 -04:00
// put the main time-chain first
2010-08-29 12:58:15 -04:00
vector < CBlockIndex * > & vNext = mapNext [ pindex ] ;
2012-04-15 17:52:09 -03:00
for ( unsigned int i = 0 ; i < vNext . size ( ) ; i + + )
2010-08-29 12:58:15 -04:00
{
if ( vNext [ i ] - > pnext )
{
swap ( vNext [ 0 ] , vNext [ i ] ) ;
break ;
}
}
// iterate children
2012-04-15 17:52:09 -03:00
for ( unsigned int i = 0 ; i < vNext . size ( ) ; i + + )
2010-08-29 12:58:15 -04:00
vStack . push_back ( make_pair ( nCol + i , vNext [ i ] ) ) ;
}
}
2012-10-21 16:23:13 -03:00
bool LoadExternalBlockFile ( FILE * fileIn , CDiskBlockPos * dbp )
2012-02-20 16:50:26 -03:00
{
2012-08-16 18:14:40 -04:00
int64 nStart = GetTimeMillis ( ) ;
2012-02-20 16:50:26 -03:00
int nLoaded = 0 ;
{
2012-10-27 17:01:38 -03:00
CBufferedFile blkdat ( fileIn , 2 * MAX_BLOCK_SIZE , MAX_BLOCK_SIZE + 8 , SER_DISK , CLIENT_VERSION ) ;
2012-10-21 16:23:13 -03:00
uint64 nStartByte = 0 ;
if ( dbp ) {
// (try to) skip already indexed part
CBlockFileInfo info ;
if ( pblocktree - > ReadBlockFileInfo ( dbp - > nFile , info ) ) {
nStartByte = info . nSize ;
blkdat . Seek ( info . nSize ) ;
}
}
2012-10-27 17:01:38 -03:00
uint64 nRewind = blkdat . GetPos ( ) ;
2012-12-04 19:53:26 -03:00
while ( blkdat . good ( ) & & ! blkdat . eof ( ) & & ! fRequestShutdown ) {
2012-10-27 17:01:38 -03:00
blkdat . SetPos ( nRewind ) ;
nRewind + + ; // start one byte further next time, in case of failure
blkdat . SetLimit ( ) ; // remove former limit
2012-10-21 16:23:13 -03:00
unsigned int nSize = 0 ;
2012-10-27 17:01:38 -03:00
try {
// locate a header
unsigned char buf [ 4 ] ;
blkdat . FindByte ( pchMessageStart [ 0 ] ) ;
nRewind = blkdat . GetPos ( ) + 1 ;
blkdat > > FLATDATA ( buf ) ;
if ( memcmp ( buf , pchMessageStart , 4 ) )
continue ;
// read size
2012-02-20 16:50:26 -03:00
blkdat > > nSize ;
2012-10-27 17:01:38 -03:00
if ( nSize < 80 | | nSize > MAX_BLOCK_SIZE )
continue ;
2012-10-21 16:23:13 -03:00
} catch ( std : : exception & e ) {
// no valid block header found; don't complain
break ;
}
try {
2012-10-27 17:01:38 -03:00
// read block
2012-10-21 16:23:13 -03:00
uint64 nBlockPos = blkdat . GetPos ( ) ;
blkdat . SetLimit ( nBlockPos + nSize ) ;
2012-10-27 17:01:38 -03:00
CBlock block ;
blkdat > > block ;
nRewind = blkdat . GetPos ( ) ;
2012-10-21 16:23:13 -03:00
// process block
if ( nBlockPos > = nStartByte ) {
2012-09-13 09:33:52 -03:00
LOCK ( cs_main ) ;
2012-10-21 16:23:13 -03:00
if ( dbp )
dbp - > nPos = nBlockPos ;
if ( ProcessBlock ( NULL , & block , dbp ) )
2012-02-20 16:50:26 -03:00
nLoaded + + ;
}
2012-10-27 17:01:38 -03:00
} catch ( std : : exception & e ) {
printf ( " %s() : Deserialize or I/O error caught during load \n " , __PRETTY_FUNCTION__ ) ;
2012-02-20 16:50:26 -03:00
}
}
2012-10-27 17:01:38 -03:00
fclose ( fileIn ) ;
2012-02-20 16:50:26 -03:00
}
2012-10-21 16:23:13 -03:00
if ( nLoaded > 0 )
printf ( " Loaded %i blocks from external file in % " PRI64d " ms \n " , nLoaded , GetTimeMillis ( ) - nStart ) ;
2012-02-20 16:50:26 -03:00
return nLoaded > 0 ;
}
2010-08-29 12:58:15 -04:00
2012-09-13 09:33:52 -03:00
2010-08-29 12:58:15 -04:00
//////////////////////////////////////////////////////////////////////////////
//
// CAlert
//
2012-08-28 17:04:54 -04:00
extern map < uint256 , CAlert > mapAlerts ;
extern CCriticalSection cs_mapAlerts ;
2010-08-29 12:58:15 -04:00
string GetWarnings ( string strFor )
{
int nPriority = 0 ;
string strStatusBar ;
string strRPC ;
2012-10-24 16:47:07 -03:00
2010-12-03 16:38:09 -03:00
if ( GetBoolArg ( " -testsafemode " ) )
2010-08-29 12:58:15 -04:00
strRPC = " test " ;
2012-10-24 16:47:07 -03:00
if ( ! CLIENT_VERSION_IS_RELEASE )
strStatusBar = _ ( " This is a pre-release test build - use at your own risk - do not use for mining or merchant applications " ) ;
2010-08-29 12:58:15 -04:00
// Misc warnings like out of disk space and clock is wrong
if ( strMiscWarning ! = " " )
{
nPriority = 1000 ;
strStatusBar = strMiscWarning ;
}
// Longer invalid proof-of-work chain
if ( pindexBest & & bnBestInvalidWork > bnBestChainWork + pindexBest - > GetBlockWork ( ) * 6 )
{
nPriority = 2000 ;
2012-07-27 02:36:43 -04:00
strStatusBar = strRPC = _ ( " Warning: Displayed transactions may not be correct! You may need to upgrade, or other nodes may need to upgrade. " ) ;
2010-08-29 12:58:15 -04:00
}
// Alerts
{
2012-04-06 13:39:12 -03:00
LOCK ( cs_mapAlerts ) ;
2011-05-15 03:11:04 -04:00
BOOST_FOREACH ( PAIRTYPE ( const uint256 , CAlert ) & item , mapAlerts )
2010-08-29 12:58:15 -04:00
{
const CAlert & alert = item . second ;
if ( alert . AppliesToMe ( ) & & alert . nPriority > nPriority )
{
nPriority = alert . nPriority ;
strStatusBar = alert . strStatusBar ;
}
}
}
if ( strFor = = " statusbar " )
return strStatusBar ;
else if ( strFor = = " rpc " )
return strRPC ;
2011-06-24 13:56:23 -04:00
assert ( ! " GetWarnings() : invalid parameter " ) ;
2010-08-29 12:58:15 -04:00
return " error " ;
}
//////////////////////////////////////////////////////////////////////////////
//
// Messages
//
2012-07-06 10:33:34 -04:00
bool static AlreadyHave ( const CInv & inv )
2010-08-29 12:58:15 -04:00
{
switch ( inv . type )
{
2012-04-13 19:24:55 -03:00
case MSG_TX :
{
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
bool txInMap = false ;
2012-04-17 13:31:51 -03:00
{
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
LOCK ( mempool . cs ) ;
txInMap = mempool . exists ( inv . hash ) ;
2012-04-17 13:31:51 -03:00
}
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
return txInMap | | mapOrphanTransactions . count ( inv . hash ) | |
2012-07-06 10:33:34 -04:00
pcoinsTip - > HaveCoins ( inv . hash ) ;
2012-04-13 19:24:55 -03:00
}
case MSG_BLOCK :
return mapBlockIndex . count ( inv . hash ) | |
mapOrphanBlocks . count ( inv . hash ) ;
2010-08-29 12:58:15 -04:00
}
// Don't know what it is, just say we already got one
return true ;
}
2010-10-19 14:16:51 -03:00
// The message start string is designed to be unlikely to occur in normal data.
2012-07-25 20:48:39 -04:00
// The characters are rarely used upper ASCII, not valid as UTF-8, and produce
2010-10-19 14:16:51 -03:00
// a large 4-byte int at any alignment.
2011-08-07 12:19:14 -04:00
unsigned char pchMessageStart [ 4 ] = { 0xf9 , 0xbe , 0xb4 , 0xd9 } ;
2010-08-29 12:58:15 -04:00
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
bool static ProcessMessage ( CNode * pfrom , string strCommand , CDataStream & vRecv )
2010-08-29 12:58:15 -04:00
{
RandAddSeedPerfmon ( ) ;
2012-05-11 18:09:33 -04:00
if ( fDebug )
2012-09-29 06:57:44 -03:00
printf ( " received: %s (% " PRIszu " bytes) \n " , strCommand . c_str ( ) , vRecv . size ( ) ) ;
2010-08-29 12:58:15 -04:00
if ( mapArgs . count ( " -dropmessagestest " ) & & GetRand ( atoi ( mapArgs [ " -dropmessagestest " ] ) ) = = 0 )
{
printf ( " dropmessagestest DROPPING RECV MESSAGE \n " ) ;
return true ;
}
if ( strCommand = = " version " )
{
// Each connection can only send one version message
if ( pfrom - > nVersion ! = 0 )
2011-09-06 18:41:51 -03:00
{
pfrom - > Misbehaving ( 1 ) ;
2010-08-29 12:58:15 -04:00
return false ;
2011-09-06 18:41:51 -03:00
}
2010-08-29 12:58:15 -04:00
2011-12-21 18:33:19 -03:00
int64 nTime ;
2010-08-29 12:58:15 -04:00
CAddress addrMe ;
CAddress addrFrom ;
2011-12-21 18:33:19 -03:00
uint64 nNonce = 1 ;
2010-08-29 12:58:15 -04:00
vRecv > > pfrom - > nVersion > > pfrom - > nServices > > nTime > > addrMe ;
2012-04-12 21:07:49 -03:00
if ( pfrom - > nVersion < MIN_PROTO_VERSION )
2012-02-19 21:33:31 -03:00
{
2012-02-28 09:31:56 -03:00
// Since February 20, 2012, the protocol is initiated at version 209,
2012-02-19 21:33:31 -03:00
// and earlier versions are no longer supported
printf ( " partner %s using obsolete version %i; disconnecting \n " , pfrom - > addr . ToString ( ) . c_str ( ) , pfrom - > nVersion ) ;
pfrom - > fDisconnect = true ;
return false ;
}
2010-08-29 12:58:15 -04:00
if ( pfrom - > nVersion = = 10300 )
pfrom - > nVersion = 300 ;
2012-02-19 21:33:31 -03:00
if ( ! vRecv . empty ( ) )
2010-08-29 12:58:15 -04:00
vRecv > > addrFrom > > nNonce ;
2012-02-19 21:33:31 -03:00
if ( ! vRecv . empty ( ) )
2010-08-29 12:58:15 -04:00
vRecv > > pfrom - > strSubVer ;
2012-02-19 21:33:31 -03:00
if ( ! vRecv . empty ( ) )
2010-08-29 12:58:15 -04:00
vRecv > > pfrom - > nStartingHeight ;
2012-02-12 09:45:24 -03:00
if ( pfrom - > fInbound & & addrMe . IsRoutable ( ) )
{
pfrom - > addrLocal = addrMe ;
SeenLocal ( addrMe ) ;
}
2010-08-29 12:58:15 -04:00
// Disconnect if we connected to ourself
if ( nNonce = = nLocalHostNonce & & nNonce > 1 )
{
2010-10-05 22:19:47 -04:00
printf ( " connected to self at %s, disconnecting \n " , pfrom - > addr . ToString ( ) . c_str ( ) ) ;
2010-08-29 12:58:15 -04:00
pfrom - > fDisconnect = true ;
return true ;
}
2011-01-24 12:42:17 -03:00
// Be shy and don't send version until we hear
if ( pfrom - > fInbound )
pfrom - > PushVersion ( ) ;
2010-08-29 12:58:15 -04:00
pfrom - > fClient = ! ( pfrom - > nServices & NODE_NETWORK ) ;
2012-01-03 19:33:31 -03:00
AddTimeData ( pfrom - > addr , nTime ) ;
2010-08-29 12:58:15 -04:00
// Change version
2012-02-19 21:33:31 -03:00
pfrom - > PushMessage ( " verack " ) ;
2011-12-16 18:26:14 -03:00
pfrom - > vSend . SetVersion ( min ( pfrom - > nVersion , PROTOCOL_VERSION ) ) ;
2010-08-29 12:58:15 -04:00
2010-10-23 14:43:53 -03:00
if ( ! pfrom - > fInbound )
{
// Advertise our address
2012-05-24 13:02:21 -04:00
if ( ! fNoListen & & ! IsInitialBlockDownload ( ) )
2010-10-23 14:43:53 -03:00
{
2012-02-12 09:45:24 -03:00
CAddress addr = GetLocalAddress ( & pfrom - > addr ) ;
if ( addr . IsRoutable ( ) )
pfrom - > PushAddress ( addr ) ;
2010-10-23 14:43:53 -03:00
}
// Get recent addresses
2012-04-23 21:15:00 -03:00
if ( pfrom - > fOneShot | | pfrom - > nVersion > = CADDR_TIME_VERSION | | addrman . size ( ) < 1000 )
2010-10-23 14:43:53 -03:00
{
pfrom - > PushMessage ( " getaddr " ) ;
pfrom - > fGetAddr = true ;
}
2012-01-04 19:39:45 -03:00
addrman . Good ( pfrom - > addr ) ;
} else {
if ( ( ( CNetAddr ) pfrom - > addr ) = = ( CNetAddr ) addrFrom )
{
addrman . Add ( addrFrom , addrFrom ) ;
addrman . Good ( addrFrom ) ;
}
2010-10-23 14:43:53 -03:00
}
2010-08-29 12:58:15 -04:00
// Ask the first connected node for block updates
2011-12-19 21:04:47 -03:00
static int nAskedForBlocks = 0 ;
2012-10-21 16:23:13 -03:00
if ( ! pfrom - > fClient & & ! pfrom - > fOneShot & & ! fImporting & & ! fReindex & &
2012-09-24 14:26:09 -03:00
( pfrom - > nStartingHeight > ( nBestHeight - 144 ) ) & &
2012-04-12 21:07:49 -03:00
( pfrom - > nVersion < NOBLKS_VERSION_START | |
2012-04-17 13:31:51 -03:00
pfrom - > nVersion > = NOBLKS_VERSION_END ) & &
2011-09-02 13:56:10 -03:00
( nAskedForBlocks < 1 | | vNodes . size ( ) < = 1 ) )
2010-08-29 12:58:15 -04:00
{
nAskedForBlocks + + ;
pfrom - > PushGetBlocks ( pindexBest , uint256 ( 0 ) ) ;
}
// Relay alerts
2012-04-06 13:39:12 -03:00
{
LOCK ( cs_mapAlerts ) ;
2011-05-15 03:11:04 -04:00
BOOST_FOREACH ( PAIRTYPE ( const uint256 , CAlert ) & item , mapAlerts )
2010-08-29 12:58:15 -04:00
item . second . RelayTo ( pfrom ) ;
2012-04-06 13:39:12 -03:00
}
2010-08-29 12:58:15 -04:00
pfrom - > fSuccessfullyConnected = true ;
2012-05-01 16:03:51 -04:00
printf ( " receive version message: version %d, blocks=%d, us=%s, them=%s, peer=%s \n " , pfrom - > nVersion , pfrom - > nStartingHeight , addrMe . ToString ( ) . c_str ( ) , addrFrom . ToString ( ) . c_str ( ) , pfrom - > addr . ToString ( ) . c_str ( ) ) ;
2011-09-28 16:35:58 -03:00
cPeerBlockCounts . input ( pfrom - > nStartingHeight ) ;
2010-08-29 12:58:15 -04:00
}
else if ( pfrom - > nVersion = = 0 )
{
// Must have a version message before anything else
2011-09-06 18:41:51 -03:00
pfrom - > Misbehaving ( 1 ) ;
2010-08-29 12:58:15 -04:00
return false ;
}
else if ( strCommand = = " verack " )
{
2011-12-16 18:26:14 -03:00
pfrom - > vRecv . SetVersion ( min ( pfrom - > nVersion , PROTOCOL_VERSION ) ) ;
2010-08-29 12:58:15 -04:00
}
else if ( strCommand = = " addr " )
{
vector < CAddress > vAddr ;
vRecv > > vAddr ;
2010-10-23 14:43:53 -03:00
// Don't want addr from older versions unless seeding
2012-04-12 21:07:49 -03:00
if ( pfrom - > nVersion < CADDR_TIME_VERSION & & addrman . size ( ) > 1000 )
2010-08-29 12:58:15 -04:00
return true ;
if ( vAddr . size ( ) > 1000 )
2011-09-06 18:41:51 -03:00
{
pfrom - > Misbehaving ( 20 ) ;
2012-09-29 06:57:44 -03:00
return error ( " message addr size() = % " PRIszu " " , vAddr.size()) ;
2011-09-06 18:41:51 -03:00
}
2010-08-29 12:58:15 -04:00
// Store the new addresses
2012-04-10 15:22:04 -03:00
vector < CAddress > vAddrOk ;
2011-12-21 18:33:19 -03:00
int64 nNow = GetAdjustedTime ( ) ;
int64 nSince = nNow - 10 * 60 ;
2011-05-15 03:11:04 -04:00
BOOST_FOREACH ( CAddress & addr , vAddr )
2010-08-29 12:58:15 -04:00
{
if ( fShutdown )
return true ;
2010-10-23 14:43:53 -03:00
if ( addr . nTime < = 100000000 | | addr . nTime > nNow + 10 * 60 )
addr . nTime = nNow - 5 * 24 * 60 * 60 ;
2010-08-29 12:58:15 -04:00
pfrom - > AddAddressKnown ( addr ) ;
2012-04-10 15:22:04 -03:00
bool fReachable = IsReachable ( addr ) ;
2010-10-23 14:43:53 -03:00
if ( addr . nTime > nSince & & ! pfrom - > fGetAddr & & vAddr . size ( ) < = 10 & & addr . IsRoutable ( ) )
2010-08-29 12:58:15 -04:00
{
// Relay to a limited number of other nodes
{
2012-04-06 13:39:12 -03:00
LOCK ( cs_vNodes ) ;
2010-10-19 14:16:51 -03:00
// Use deterministic randomness to send to the same nodes for 24 hours
// at a time so the setAddrKnowns of the chosen nodes prevent repeats
2010-08-29 12:58:15 -04:00
static uint256 hashSalt ;
if ( hashSalt = = 0 )
2012-05-17 12:13:14 -04:00
hashSalt = GetRandHash ( ) ;
2012-05-13 15:31:59 -04:00
uint64 hashAddr = addr . GetHash ( ) ;
2012-01-03 19:33:31 -03:00
uint256 hashRand = hashSalt ^ ( hashAddr < < 32 ) ^ ( ( GetTime ( ) + hashAddr ) / ( 24 * 60 * 60 ) ) ;
2010-10-19 14:16:51 -03:00
hashRand = Hash ( BEGIN ( hashRand ) , END ( hashRand ) ) ;
2010-08-29 12:58:15 -04:00
multimap < uint256 , CNode * > mapMix ;
2011-05-15 03:11:04 -04:00
BOOST_FOREACH ( CNode * pnode , vNodes )
2010-10-19 14:16:51 -03:00
{
2012-04-12 21:07:49 -03:00
if ( pnode - > nVersion < CADDR_TIME_VERSION )
2010-10-23 14:43:53 -03:00
continue ;
2010-10-19 14:16:51 -03:00
unsigned int nPointer ;
memcpy ( & nPointer , & pnode , sizeof ( nPointer ) ) ;
uint256 hashKey = hashRand ^ nPointer ;
hashKey = Hash ( BEGIN ( hashKey ) , END ( hashKey ) ) ;
mapMix . insert ( make_pair ( hashKey , pnode ) ) ;
}
2012-04-10 15:22:04 -03:00
int nRelayNodes = fReachable ? 2 : 1 ; // limited relaying of addresses outside our network(s)
2010-08-29 12:58:15 -04:00
for ( multimap < uint256 , CNode * > : : iterator mi = mapMix . begin ( ) ; mi ! = mapMix . end ( ) & & nRelayNodes - - > 0 ; + + mi )
( ( * mi ) . second ) - > PushAddress ( addr ) ;
}
}
2012-04-10 15:22:04 -03:00
// Do not store addresses outside our network
if ( fReachable )
vAddrOk . push_back ( addr ) ;
2010-08-29 12:58:15 -04:00
}
2012-04-10 15:22:04 -03:00
addrman . Add ( vAddrOk , pfrom - > addr , 2 * 60 * 60 ) ;
2010-08-29 12:58:15 -04:00
if ( vAddr . size ( ) < 1000 )
pfrom - > fGetAddr = false ;
2012-04-23 21:15:00 -03:00
if ( pfrom - > fOneShot )
pfrom - > fDisconnect = true ;
2010-08-29 12:58:15 -04:00
}
else if ( strCommand = = " inv " )
{
vector < CInv > vInv ;
vRecv > > vInv ;
2012-07-31 17:42:35 -04:00
if ( vInv . size ( ) > MAX_INV_SZ )
2011-09-06 18:41:51 -03:00
{
pfrom - > Misbehaving ( 20 ) ;
2012-09-29 06:57:44 -03:00
return error ( " message inv size() = % " PRIszu " " , vInv.size()) ;
2011-09-06 18:41:51 -03:00
}
2010-08-29 12:58:15 -04:00
2012-05-04 22:04:38 -04:00
// find last block in inv vector
unsigned int nLastBlock = ( unsigned int ) ( - 1 ) ;
for ( unsigned int nInv = 0 ; nInv < vInv . size ( ) ; nInv + + ) {
2012-05-07 15:36:30 -04:00
if ( vInv [ vInv . size ( ) - 1 - nInv ] . type = = MSG_BLOCK ) {
2012-05-04 22:04:38 -04:00
nLastBlock = vInv . size ( ) - 1 - nInv ;
2012-05-07 15:36:30 -04:00
break ;
}
2012-05-04 22:04:38 -04:00
}
2012-04-15 17:52:09 -03:00
for ( unsigned int nInv = 0 ; nInv < vInv . size ( ) ; nInv + + )
2010-08-29 12:58:15 -04:00
{
2012-03-18 19:47:26 -03:00
const CInv & inv = vInv [ nInv ] ;
2010-08-29 12:58:15 -04:00
if ( fShutdown )
return true ;
pfrom - > AddInventoryKnown ( inv ) ;
2012-07-06 10:33:34 -04:00
bool fAlreadyHave = AlreadyHave ( inv ) ;
2011-09-17 13:29:41 -03:00
if ( fDebug )
printf ( " got inventory: %s %s \n " , inv . ToString ( ) . c_str ( ) , fAlreadyHave ? " have " : " new " ) ;
2010-08-29 12:58:15 -04:00
2012-09-13 09:33:52 -03:00
if ( ! fAlreadyHave ) {
2012-10-21 16:23:13 -03:00
if ( ! fImporting & & ! fReindex )
2012-09-13 09:33:52 -03:00
pfrom - > AskFor ( inv ) ;
} else if ( inv . type = = MSG_BLOCK & & mapOrphanBlocks . count ( inv . hash ) ) {
2010-08-29 12:58:15 -04:00
pfrom - > PushGetBlocks ( pindexBest , GetOrphanRoot ( mapOrphanBlocks [ inv . hash ] ) ) ;
2012-05-07 15:36:30 -04:00
} else if ( nInv = = nLastBlock ) {
// In case we are on a very long side-chain, it is possible that we already have
// the last block in an inv bundle sent in response to getblocks. Try to detect
// this situation and push another getblocks to continue.
pfrom - > PushGetBlocks ( mapBlockIndex [ inv . hash ] , uint256 ( 0 ) ) ;
if ( fDebug )
printf ( " force request: %s \n " , inv . ToString ( ) . c_str ( ) ) ;
}
2010-08-29 12:58:15 -04:00
// Track requests for our stuff
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
Inventory ( inv . hash ) ;
2010-08-29 12:58:15 -04:00
}
}
else if ( strCommand = = " getdata " )
{
vector < CInv > vInv ;
vRecv > > vInv ;
2012-07-31 17:42:35 -04:00
if ( vInv . size ( ) > MAX_INV_SZ )
2011-09-06 18:41:51 -03:00
{
pfrom - > Misbehaving ( 20 ) ;
2012-09-29 06:57:44 -03:00
return error ( " message getdata size() = % " PRIszu " " , vInv.size()) ;
2011-09-06 18:41:51 -03:00
}
2010-08-29 12:58:15 -04:00
2012-06-23 21:38:33 -04:00
if ( fDebugNet | | ( vInv . size ( ) ! = 1 ) )
2012-09-29 06:57:44 -03:00
printf ( " received getdata (% " PRIszu " invsz) \n " , vInv . size ( ) ) ;
2012-06-23 21:38:33 -04:00
2011-05-15 03:11:04 -04:00
BOOST_FOREACH ( const CInv & inv , vInv )
2010-08-29 12:58:15 -04:00
{
if ( fShutdown )
return true ;
2012-06-23 21:38:33 -04:00
if ( fDebugNet | | ( vInv . size ( ) = = 1 ) )
printf ( " received getdata for: %s \n " , inv . ToString ( ) . c_str ( ) ) ;
2010-08-29 12:58:15 -04:00
if ( inv . type = = MSG_BLOCK )
{
// Send block from disk
map < uint256 , CBlockIndex * > : : iterator mi = mapBlockIndex . find ( inv . hash ) ;
if ( mi ! = mapBlockIndex . end ( ) )
{
CBlock block ;
2010-12-05 06:29:30 -03:00
block . ReadFromDisk ( ( * mi ) . second ) ;
2010-08-29 12:58:15 -04:00
pfrom - > PushMessage ( " block " , block ) ;
// Trigger them to send a getblocks request for the next batch of inventory
if ( inv . hash = = pfrom - > hashContinue )
{
// Bypass PushInventory, this must send even if redundant,
// and we want it right after the last block so they don't
// wait for other stuff first.
vector < CInv > vInv ;
vInv . push_back ( CInv ( MSG_BLOCK , hashBestChain ) ) ;
pfrom - > PushMessage ( " inv " , vInv ) ;
pfrom - > hashContinue = 0 ;
}
}
}
else if ( inv . IsKnownType ( ) )
{
// Send stream from relay memory
2012-07-31 17:42:35 -04:00
bool pushed = false ;
2010-08-29 12:58:15 -04:00
{
2012-04-06 13:39:12 -03:00
LOCK ( cs_mapRelay ) ;
2010-08-29 12:58:15 -04:00
map < CInv , CDataStream > : : iterator mi = mapRelay . find ( inv ) ;
2012-07-31 17:42:35 -04:00
if ( mi ! = mapRelay . end ( ) ) {
2010-08-29 12:58:15 -04:00
pfrom - > PushMessage ( inv . GetCommand ( ) , ( * mi ) . second ) ;
2012-07-31 17:42:35 -04:00
pushed = true ;
}
}
if ( ! pushed & & inv . type = = MSG_TX ) {
LOCK ( mempool . cs ) ;
if ( mempool . exists ( inv . hash ) ) {
CTransaction tx = mempool . lookup ( inv . hash ) ;
CDataStream ss ( SER_NETWORK , PROTOCOL_VERSION ) ;
ss . reserve ( 1000 ) ;
ss < < tx ;
pfrom - > PushMessage ( " tx " , ss ) ;
}
2010-08-29 12:58:15 -04:00
}
}
// Track requests for our stuff
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
Inventory ( inv . hash ) ;
2010-08-29 12:58:15 -04:00
}
}
else if ( strCommand = = " getblocks " )
{
CBlockLocator locator ;
uint256 hashStop ;
vRecv > > locator > > hashStop ;
2010-12-05 06:29:30 -03:00
// Find the last block the caller has in the main chain
2010-08-29 12:58:15 -04:00
CBlockIndex * pindex = locator . GetBlockIndex ( ) ;
// Send the rest of the chain
if ( pindex )
pindex = pindex - > pnext ;
2012-03-21 23:10:50 -03:00
int nLimit = 500 ;
2012-08-13 03:02:44 -04:00
printf ( " getblocks %d to %s limit %d \n " , ( pindex ? pindex - > nHeight : - 1 ) , BlockHashStr ( hashStop ) . c_str ( ) , nLimit ) ;
2010-08-29 12:58:15 -04:00
for ( ; pindex ; pindex = pindex - > pnext )
{
if ( pindex - > GetBlockHash ( ) = = hashStop )
{
2012-08-13 03:02:44 -04:00
printf ( " getblocks stopping at %d %s \n " , pindex - > nHeight , BlockHashStr ( pindex - > GetBlockHash ( ) ) . c_str ( ) ) ;
2010-08-29 12:58:15 -04:00
break ;
}
pfrom - > PushInventory ( CInv ( MSG_BLOCK , pindex - > GetBlockHash ( ) ) ) ;
2012-03-21 23:10:50 -03:00
if ( - - nLimit < = 0 )
2010-08-29 12:58:15 -04:00
{
// When this block is requested, we'll send an inv that'll make them
// getblocks the next batch of inventory.
2012-08-13 03:02:44 -04:00
printf ( " getblocks stopping at limit %d %s \n " , pindex - > nHeight , BlockHashStr ( pindex - > GetBlockHash ( ) ) . c_str ( ) ) ;
2010-08-29 12:58:15 -04:00
pfrom - > hashContinue = pindex - > GetBlockHash ( ) ;
break ;
}
}
}
2010-12-05 06:29:30 -03:00
else if ( strCommand = = " getheaders " )
{
CBlockLocator locator ;
uint256 hashStop ;
vRecv > > locator > > hashStop ;
CBlockIndex * pindex = NULL ;
if ( locator . IsNull ( ) )
{
// If locator is null, return the hashStop block
map < uint256 , CBlockIndex * > : : iterator mi = mapBlockIndex . find ( hashStop ) ;
if ( mi = = mapBlockIndex . end ( ) )
return true ;
pindex = ( * mi ) . second ;
}
else
{
// Find the last block the caller has in the main chain
pindex = locator . GetBlockIndex ( ) ;
if ( pindex )
pindex = pindex - > pnext ;
}
2012-11-14 18:18:10 -03:00
// we must use CBlocks, as CBlockHeaders won't include the 0x00 nTx count at the end
2010-12-05 06:29:30 -03:00
vector < CBlock > vHeaders ;
2012-03-19 01:13:15 -03:00
int nLimit = 2000 ;
2012-08-13 03:02:44 -04:00
printf ( " getheaders %d to %s \n " , ( pindex ? pindex - > nHeight : - 1 ) , BlockHashStr ( hashStop ) . c_str ( ) ) ;
2010-12-05 06:29:30 -03:00
for ( ; pindex ; pindex = pindex - > pnext )
{
vHeaders . push_back ( pindex - > GetBlockHeader ( ) ) ;
if ( - - nLimit < = 0 | | pindex - > GetBlockHash ( ) = = hashStop )
break ;
}
pfrom - > PushMessage ( " headers " , vHeaders ) ;
}
2010-08-29 12:58:15 -04:00
else if ( strCommand = = " tx " )
{
vector < uint256 > vWorkQueue ;
2012-05-17 10:12:04 -04:00
vector < uint256 > vEraseQueue ;
2010-08-29 12:58:15 -04:00
CDataStream vMsg ( vRecv ) ;
CTransaction tx ;
vRecv > > tx ;
CInv inv ( MSG_TX , tx . GetHash ( ) ) ;
pfrom - > AddInventoryKnown ( inv ) ;
bool fMissingInputs = false ;
2012-07-06 10:33:34 -04:00
if ( tx . AcceptToMemoryPool ( true , & fMissingInputs ) )
2010-08-29 12:58:15 -04:00
{
2012-07-07 18:06:34 -04:00
SyncWithWallets ( inv . hash , tx , NULL , true ) ;
2010-08-29 12:58:15 -04:00
RelayMessage ( inv , vMsg ) ;
mapAlreadyAskedFor . erase ( inv ) ;
vWorkQueue . push_back ( inv . hash ) ;
2012-05-17 10:12:04 -04:00
vEraseQueue . push_back ( inv . hash ) ;
2010-08-29 12:58:15 -04:00
// Recursively process any orphan transactions that depended on this one
2012-04-15 17:52:09 -03:00
for ( unsigned int i = 0 ; i < vWorkQueue . size ( ) ; i + + )
2010-08-29 12:58:15 -04:00
{
uint256 hashPrev = vWorkQueue [ i ] ;
2012-05-15 15:53:30 -04:00
for ( map < uint256 , CDataStream * > : : iterator mi = mapOrphanTransactionsByPrev [ hashPrev ] . begin ( ) ;
mi ! = mapOrphanTransactionsByPrev [ hashPrev ] . end ( ) ;
2010-08-29 12:58:15 -04:00
+ + mi )
{
const CDataStream & vMsg = * ( ( * mi ) . second ) ;
CTransaction tx ;
CDataStream ( vMsg ) > > tx ;
CInv inv ( MSG_TX , tx . GetHash ( ) ) ;
2012-05-17 10:12:04 -04:00
bool fMissingInputs2 = false ;
2010-08-29 12:58:15 -04:00
2012-07-06 10:33:34 -04:00
if ( tx . AcceptToMemoryPool ( true , & fMissingInputs2 ) )
2010-08-29 12:58:15 -04:00
{
2010-10-05 22:19:47 -04:00
printf ( " accepted orphan tx %s \n " , inv . hash . ToString ( ) . substr ( 0 , 10 ) . c_str ( ) ) ;
2012-07-07 18:06:34 -04:00
SyncWithWallets ( inv . hash , tx , NULL , true ) ;
2010-08-29 12:58:15 -04:00
RelayMessage ( inv , vMsg ) ;
mapAlreadyAskedFor . erase ( inv ) ;
vWorkQueue . push_back ( inv . hash ) ;
2012-05-17 10:12:04 -04:00
vEraseQueue . push_back ( inv . hash ) ;
}
else if ( ! fMissingInputs2 )
{
// invalid orphan
vEraseQueue . push_back ( inv . hash ) ;
printf ( " removed invalid orphan tx %s \n " , inv . hash . ToString ( ) . substr ( 0 , 10 ) . c_str ( ) ) ;
2010-08-29 12:58:15 -04:00
}
}
}
2012-05-17 10:12:04 -04:00
BOOST_FOREACH ( uint256 hash , vEraseQueue )
2010-08-29 12:58:15 -04:00
EraseOrphanTx ( hash ) ;
}
else if ( fMissingInputs )
{
AddOrphanTx ( vMsg ) ;
2012-02-29 12:14:18 -03:00
// DoS prevention: do not allow mapOrphanTransactions to grow unbounded
2012-04-23 15:14:03 -03:00
unsigned int nEvicted = LimitOrphanTxSize ( MAX_ORPHAN_TRANSACTIONS ) ;
2012-02-29 12:14:18 -03:00
if ( nEvicted > 0 )
2012-04-23 15:14:03 -03:00
printf ( " mapOrphan overflow, removed %u tx \n " , nEvicted ) ;
2010-08-29 12:58:15 -04:00
}
2011-09-06 17:59:38 -03:00
if ( tx . nDoS ) pfrom - > Misbehaving ( tx . nDoS ) ;
2010-08-29 12:58:15 -04:00
}
2012-10-21 16:23:13 -03:00
else if ( strCommand = = " block " & & ! fImporting & & ! fReindex ) // Ignore blocks received while importing
2010-08-29 12:58:15 -04:00
{
2010-12-05 06:29:30 -03:00
CBlock block ;
vRecv > > block ;
2010-08-29 12:58:15 -04:00
2012-08-13 03:02:44 -04:00
printf ( " received block %s \n " , BlockHashStr ( block . GetHash ( ) ) . c_str ( ) ) ;
2010-12-05 06:29:30 -03:00
// block.print();
2010-08-29 12:58:15 -04:00
2010-12-05 06:29:30 -03:00
CInv inv ( MSG_BLOCK , block . GetHash ( ) ) ;
2010-08-29 12:58:15 -04:00
pfrom - > AddInventoryKnown ( inv ) ;
2010-12-05 06:29:30 -03:00
if ( ProcessBlock ( pfrom , & block ) )
2010-08-29 12:58:15 -04:00
mapAlreadyAskedFor . erase ( inv ) ;
2011-09-06 17:59:38 -03:00
if ( block . nDoS ) pfrom - > Misbehaving ( block . nDoS ) ;
2010-08-29 12:58:15 -04:00
}
else if ( strCommand = = " getaddr " )
{
pfrom - > vAddrToSend . clear ( ) ;
2012-01-04 19:39:45 -03:00
vector < CAddress > vAddr = addrman . GetAddr ( ) ;
BOOST_FOREACH ( const CAddress & addr , vAddr )
pfrom - > PushAddress ( addr ) ;
2010-08-29 12:58:15 -04:00
}
2012-07-31 17:42:35 -04:00
else if ( strCommand = = " mempool " )
{
std : : vector < uint256 > vtxid ;
mempool . queryHashes ( vtxid ) ;
vector < CInv > vInv ;
for ( unsigned int i = 0 ; i < vtxid . size ( ) ; i + + ) {
CInv inv ( MSG_TX , vtxid [ i ] ) ;
vInv . push_back ( inv ) ;
if ( i = = ( MAX_INV_SZ - 1 ) )
break ;
}
if ( vInv . size ( ) > 0 )
pfrom - > PushMessage ( " inv " , vInv ) ;
}
2010-08-29 12:58:15 -04:00
else if ( strCommand = = " ping " )
{
2012-04-11 13:38:03 -03:00
if ( pfrom - > nVersion > BIP0031_VERSION )
{
uint64 nonce = 0 ;
vRecv > > nonce ;
// Echo the message back with the nonce. This allows for two useful features:
//
// 1) A remote node can quickly check if the connection is operational
// 2) Remote nodes can measure the latency of the network thread. If this node
// is overloaded it won't respond to pings quickly and the remote node can
// avoid sending us more work, like chain download requests.
//
// The nonce stops the remote getting confused between different pings: without
// it, if the remote node sends a ping once per second and this node takes 5
// seconds to respond to each, the 5th ping the remote sends would appear to
// return very quickly.
pfrom - > PushMessage ( " pong " , nonce ) ;
}
2010-08-29 12:58:15 -04:00
}
else if ( strCommand = = " alert " )
{
CAlert alert ;
vRecv > > alert ;
2012-08-26 17:08:18 -04:00
uint256 alertHash = alert . GetHash ( ) ;
if ( pfrom - > setKnown . count ( alertHash ) = = 0 )
2010-08-29 12:58:15 -04:00
{
2012-08-26 17:08:18 -04:00
if ( alert . ProcessAlert ( ) )
2012-04-06 13:39:12 -03:00
{
2012-08-26 17:08:18 -04:00
// Relay
pfrom - > setKnown . insert ( alertHash ) ;
{
LOCK ( cs_vNodes ) ;
BOOST_FOREACH ( CNode * pnode , vNodes )
alert . RelayTo ( pnode ) ;
}
}
else {
// Small DoS penalty so peers that send us lots of
// duplicate/expired/invalid-signature/whatever alerts
// eventually get banned.
// This isn't a Misbehaving(100) (immediate ban) because the
// peer might be an older or different implementation with
// a different signature key, etc.
pfrom - > Misbehaving ( 10 ) ;
2012-04-06 13:39:12 -03:00
}
2010-08-29 12:58:15 -04:00
}
}
else
{
// Ignore unknown commands for extensibility
}
// Update the last seen time for this node's address
if ( pfrom - > fNetworkNode )
if ( strCommand = = " version " | | strCommand = = " addr " | | strCommand = = " inv " | | strCommand = = " getdata " | | strCommand = = " ping " )
AddressCurrentlyConnected ( pfrom - > addr ) ;
return true ;
}
2011-06-01 12:27:05 -04:00
bool ProcessMessages ( CNode * pfrom )
{
CDataStream & vRecv = pfrom - > vRecv ;
if ( vRecv . empty ( ) )
return true ;
//if (fDebug)
// printf("ProcessMessages(%u bytes)\n", vRecv.size());
2010-08-29 12:58:15 -04:00
2011-06-01 12:27:05 -04:00
//
// Message format
// (4) message start
// (12) command
// (4) size
// (4) checksum
// (x) data
//
2010-08-29 12:58:15 -04:00
2011-06-01 12:27:05 -04:00
loop
{
2012-03-21 23:10:50 -03:00
// Don't bother if send buffer is too full to respond anyway
if ( pfrom - > vSend . size ( ) > = SendBufferSize ( ) )
break ;
2011-06-01 12:27:05 -04:00
// Scan for message start
CDataStream : : iterator pstart = search ( vRecv . begin ( ) , vRecv . end ( ) , BEGIN ( pchMessageStart ) , END ( pchMessageStart ) ) ;
int nHeaderSize = vRecv . GetSerializeSize ( CMessageHeader ( ) ) ;
if ( vRecv . end ( ) - pstart < nHeaderSize )
{
2012-04-22 14:51:16 -03:00
if ( ( int ) vRecv . size ( ) > nHeaderSize )
2011-06-01 12:27:05 -04:00
{
printf ( " \n \n PROCESSMESSAGE MESSAGESTART NOT FOUND \n \n " ) ;
vRecv . erase ( vRecv . begin ( ) , vRecv . end ( ) - nHeaderSize ) ;
}
break ;
}
if ( pstart - vRecv . begin ( ) > 0 )
2012-09-29 06:57:44 -03:00
printf ( " \n \n PROCESSMESSAGE SKIPPED % " PRIpdd " BYTES \n \n " , pstart - vRecv . begin ( ) ) ;
2011-06-01 12:27:05 -04:00
vRecv . erase ( vRecv . begin ( ) , pstart ) ;
2010-08-29 12:58:15 -04:00
2011-06-01 12:27:05 -04:00
// Read header
vector < char > vHeaderSave ( vRecv . begin ( ) , vRecv . begin ( ) + nHeaderSize ) ;
CMessageHeader hdr ;
vRecv > > hdr ;
if ( ! hdr . IsValid ( ) )
{
printf ( " \n \n PROCESSMESSAGE: ERRORS IN HEADER %s \n \n \n " , hdr . GetCommand ( ) . c_str ( ) ) ;
continue ;
}
string strCommand = hdr . GetCommand ( ) ;
// Message size
unsigned int nMessageSize = hdr . nMessageSize ;
if ( nMessageSize > MAX_SIZE )
{
2012-05-22 07:06:08 -04:00
printf ( " ProcessMessages(%s, %u bytes) : nMessageSize > MAX_SIZE \n " , strCommand . c_str ( ) , nMessageSize ) ;
2011-06-01 12:27:05 -04:00
continue ;
}
if ( nMessageSize > vRecv . size ( ) )
{
// Rewind and wait for rest of message
vRecv . insert ( vRecv . begin ( ) , vHeaderSave . begin ( ) , vHeaderSave . end ( ) ) ;
break ;
}
// Checksum
2012-02-19 21:33:31 -03:00
uint256 hash = Hash ( vRecv . begin ( ) , vRecv . begin ( ) + nMessageSize ) ;
unsigned int nChecksum = 0 ;
memcpy ( & nChecksum , & hash , sizeof ( nChecksum ) ) ;
if ( nChecksum ! = hdr . nChecksum )
2011-06-01 12:27:05 -04:00
{
2012-05-22 07:06:08 -04:00
printf ( " ProcessMessages(%s, %u bytes) : CHECKSUM ERROR nChecksum=%08x hdr.nChecksum=%08x \n " ,
2012-02-19 21:33:31 -03:00
strCommand . c_str ( ) , nMessageSize , nChecksum , hdr . nChecksum ) ;
continue ;
2011-06-01 12:27:05 -04:00
}
// Copy message to its own buffer
CDataStream vMsg ( vRecv . begin ( ) , vRecv . begin ( ) + nMessageSize , vRecv . nType , vRecv . nVersion ) ;
vRecv . ignore ( nMessageSize ) ;
// Process message
bool fRet = false ;
try
{
2012-04-06 13:39:12 -03:00
{
LOCK ( cs_main ) ;
2011-06-01 12:27:05 -04:00
fRet = ProcessMessage ( pfrom , strCommand , vMsg ) ;
2012-04-06 13:39:12 -03:00
}
2011-06-01 12:27:05 -04:00
if ( fShutdown )
return true ;
}
catch ( std : : ios_base : : failure & e )
{
if ( strstr ( e . what ( ) , " end of data " ) )
{
2012-07-25 20:48:39 -04:00
// Allow exceptions from under-length message on vRecv
2012-05-22 07:06:08 -04:00
printf ( " ProcessMessages(%s, %u bytes) : Exception '%s' caught, normally caused by a message being shorter than its stated length \n " , strCommand . c_str ( ) , nMessageSize , e . what ( ) ) ;
2011-06-01 12:27:05 -04:00
}
else if ( strstr ( e . what ( ) , " size too large " ) )
{
2012-07-25 20:48:39 -04:00
// Allow exceptions from over-long size
2012-05-22 07:06:08 -04:00
printf ( " ProcessMessages(%s, %u bytes) : Exception '%s' caught \n " , strCommand . c_str ( ) , nMessageSize , e . what ( ) ) ;
2011-06-01 12:27:05 -04:00
}
else
{
2012-05-22 07:06:08 -04:00
PrintExceptionContinue ( & e , " ProcessMessages() " ) ;
2011-06-01 12:27:05 -04:00
}
}
catch ( std : : exception & e ) {
2012-05-22 07:06:08 -04:00
PrintExceptionContinue ( & e , " ProcessMessages() " ) ;
2011-06-01 12:27:05 -04:00
} catch ( . . . ) {
2012-05-22 07:06:08 -04:00
PrintExceptionContinue ( NULL , " ProcessMessages() " ) ;
2011-06-01 12:27:05 -04:00
}
if ( ! fRet )
printf ( " ProcessMessage(%s, %u bytes) FAILED \n " , strCommand . c_str ( ) , nMessageSize ) ;
}
vRecv . Compact ( ) ;
return true ;
}
2010-08-29 12:58:15 -04:00
bool SendMessages ( CNode * pto , bool fSendTrickle )
{
2012-04-17 13:50:45 -03:00
TRY_LOCK ( cs_main , lockMain ) ;
if ( lockMain ) {
2010-08-29 12:58:15 -04:00
// Don't send anything until we get their version message
if ( pto - > nVersion = = 0 )
return true ;
2012-05-17 20:36:55 -04:00
// Keep-alive ping. We send a nonce of zero because we don't use it anywhere
2012-04-11 13:38:03 -03:00
// right now.
if ( pto - > nLastSend & & GetTime ( ) - pto - > nLastSend > 30 * 60 & & pto - > vSend . empty ( ) ) {
2012-06-14 12:31:08 -04:00
uint64 nonce = 0 ;
2012-04-11 13:38:03 -03:00
if ( pto - > nVersion > BIP0031_VERSION )
2012-06-14 12:31:08 -04:00
pto - > PushMessage ( " ping " , nonce ) ;
2012-04-11 13:38:03 -03:00
else
pto - > PushMessage ( " ping " ) ;
}
2010-08-29 12:58:15 -04:00
2010-10-23 14:43:53 -03:00
// Resend wallet transactions that haven't gotten in a block yet
ResendWalletTransactions ( ) ;
2010-08-29 12:58:15 -04:00
// Address refresh broadcast
2011-12-21 18:33:19 -03:00
static int64 nLastRebroadcast ;
2012-02-01 20:08:03 -03:00
if ( ! IsInitialBlockDownload ( ) & & ( GetTime ( ) - nLastRebroadcast > 24 * 60 * 60 ) )
2010-08-29 12:58:15 -04:00
{
{
2012-04-06 13:39:12 -03:00
LOCK ( cs_vNodes ) ;
2011-05-15 03:11:04 -04:00
BOOST_FOREACH ( CNode * pnode , vNodes )
2010-08-29 12:58:15 -04:00
{
// Periodically clear setAddrKnown to allow refresh broadcasts
2012-02-01 20:08:03 -03:00
if ( nLastRebroadcast )
pnode - > setAddrKnown . clear ( ) ;
2010-08-29 12:58:15 -04:00
// Rebroadcast our address
2012-05-24 13:02:21 -04:00
if ( ! fNoListen )
2010-10-23 14:43:53 -03:00
{
2012-02-12 09:45:24 -03:00
CAddress addr = GetLocalAddress ( & pnode - > addr ) ;
if ( addr . IsRoutable ( ) )
pnode - > PushAddress ( addr ) ;
2010-10-23 14:43:53 -03:00
}
2010-08-29 12:58:15 -04:00
}
}
2012-02-01 20:08:03 -03:00
nLastRebroadcast = GetTime ( ) ;
2010-08-29 12:58:15 -04:00
}
//
// Message: addr
//
if ( fSendTrickle )
{
vector < CAddress > vAddr ;
vAddr . reserve ( pto - > vAddrToSend . size ( ) ) ;
2011-05-15 03:11:04 -04:00
BOOST_FOREACH ( const CAddress & addr , pto - > vAddrToSend )
2010-08-29 12:58:15 -04:00
{
// returns true if wasn't already contained in the set
if ( pto - > setAddrKnown . insert ( addr ) . second )
{
vAddr . push_back ( addr ) ;
// receiver rejects addr messages larger than 1000
if ( vAddr . size ( ) > = 1000 )
{
pto - > PushMessage ( " addr " , vAddr ) ;
vAddr . clear ( ) ;
}
}
}
pto - > vAddrToSend . clear ( ) ;
if ( ! vAddr . empty ( ) )
pto - > PushMessage ( " addr " , vAddr ) ;
}
//
// Message: inventory
//
vector < CInv > vInv ;
vector < CInv > vInvWait ;
{
2012-04-06 13:39:12 -03:00
LOCK ( pto - > cs_inventory ) ;
2010-08-29 12:58:15 -04:00
vInv . reserve ( pto - > vInventoryToSend . size ( ) ) ;
vInvWait . reserve ( pto - > vInventoryToSend . size ( ) ) ;
2011-05-15 03:11:04 -04:00
BOOST_FOREACH ( const CInv & inv , pto - > vInventoryToSend )
2010-08-29 12:58:15 -04:00
{
if ( pto - > setInventoryKnown . count ( inv ) )
continue ;
// trickle out tx inv to protect privacy
if ( inv . type = = MSG_TX & & ! fSendTrickle )
{
// 1/4 of tx invs blast to all immediately
static uint256 hashSalt ;
if ( hashSalt = = 0 )
2012-05-17 12:13:14 -04:00
hashSalt = GetRandHash ( ) ;
2010-08-29 12:58:15 -04:00
uint256 hashRand = inv . hash ^ hashSalt ;
hashRand = Hash ( BEGIN ( hashRand ) , END ( hashRand ) ) ;
bool fTrickleWait = ( ( hashRand & 3 ) ! = 0 ) ;
// always trickle our own transactions
if ( ! fTrickleWait )
{
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
CWalletTx wtx ;
if ( GetTransaction ( inv . hash , wtx ) )
if ( wtx . fFromMe )
fTrickleWait = true ;
2010-08-29 12:58:15 -04:00
}
if ( fTrickleWait )
{
vInvWait . push_back ( inv ) ;
continue ;
}
}
// returns true if wasn't already contained in the set
if ( pto - > setInventoryKnown . insert ( inv ) . second )
{
vInv . push_back ( inv ) ;
if ( vInv . size ( ) > = 1000 )
{
pto - > PushMessage ( " inv " , vInv ) ;
vInv . clear ( ) ;
}
}
}
pto - > vInventoryToSend = vInvWait ;
}
if ( ! vInv . empty ( ) )
pto - > PushMessage ( " inv " , vInv ) ;
//
// Message: getdata
//
vector < CInv > vGetData ;
2011-12-21 18:33:19 -03:00
int64 nNow = GetTime ( ) * 1000000 ;
2010-08-29 12:58:15 -04:00
while ( ! pto - > mapAskFor . empty ( ) & & ( * pto - > mapAskFor . begin ( ) ) . first < = nNow )
{
const CInv & inv = ( * pto - > mapAskFor . begin ( ) ) . second ;
2012-07-06 10:33:34 -04:00
if ( ! AlreadyHave ( inv ) )
2010-08-29 12:58:15 -04:00
{
2012-06-22 13:11:57 -04:00
if ( fDebugNet )
printf ( " sending getdata: %s \n " , inv . ToString ( ) . c_str ( ) ) ;
2010-08-29 12:58:15 -04:00
vGetData . push_back ( inv ) ;
if ( vGetData . size ( ) > = 1000 )
{
pto - > PushMessage ( " getdata " , vGetData ) ;
vGetData . clear ( ) ;
}
2012-05-17 18:01:00 -04:00
mapAlreadyAskedFor [ inv ] = nNow ;
2010-08-29 12:58:15 -04:00
}
pto - > mapAskFor . erase ( pto - > mapAskFor . begin ( ) ) ;
}
if ( ! vGetData . empty ( ) )
pto - > PushMessage ( " getdata " , vGetData ) ;
}
return true ;
}
//////////////////////////////////////////////////////////////////////////////
//
// BitcoinMiner
//
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
int static FormatHashBlocks ( void * pbuffer , unsigned int len )
2010-09-13 18:14:24 -04:00
{
unsigned char * pdata = ( unsigned char * ) pbuffer ;
unsigned int blocks = 1 + ( ( len + 8 ) / 64 ) ;
unsigned char * pend = pdata + 64 * blocks ;
memset ( pdata + len , 0 , 64 * blocks - len ) ;
pdata [ len ] = 0x80 ;
unsigned int bits = len * 8 ;
pend [ - 1 ] = ( bits > > 0 ) & 0xff ;
pend [ - 2 ] = ( bits > > 8 ) & 0xff ;
pend [ - 3 ] = ( bits > > 16 ) & 0xff ;
pend [ - 4 ] = ( bits > > 24 ) & 0xff ;
return blocks ;
}
static const unsigned int pSHA256InitState [ 8 ] =
{ 0x6a09e667 , 0xbb67ae85 , 0x3c6ef372 , 0xa54ff53a , 0x510e527f , 0x9b05688c , 0x1f83d9ab , 0x5be0cd19 } ;
2011-09-27 15:16:07 -03:00
void SHA256Transform ( void * pstate , void * pinput , const void * pinit )
2010-09-13 18:14:24 -04:00
{
2011-09-27 15:16:07 -03:00
SHA256_CTX ctx ;
unsigned char data [ 64 ] ;
SHA256_Init ( & ctx ) ;
for ( int i = 0 ; i < 16 ; i + + )
( ( uint32_t * ) data ) [ i ] = ByteReverse ( ( ( uint32_t * ) pinput ) [ i ] ) ;
for ( int i = 0 ; i < 8 ; i + + )
ctx . h [ i ] = ( ( uint32_t * ) pinit ) [ i ] ;
SHA256_Update ( & ctx , data , sizeof ( data ) ) ;
2012-05-17 20:36:55 -04:00
for ( int i = 0 ; i < 8 ; i + + )
2011-09-27 15:16:07 -03:00
( ( uint32_t * ) pstate ) [ i ] = ctx . h [ i ] ;
2010-09-13 18:14:24 -04:00
}
//
// ScanHash scans nonces looking for a hash with at least some zero bits.
// It operates on big endian data. Caller does the byte reversing.
// All input buffers are 16-byte aligned. nNonce is usually preserved
2010-11-23 16:16:36 -03:00
// between calls, but periodically or if nNonce is 0xffff0000 or above,
2010-09-13 18:14:24 -04:00
// the block is rebuilt and nNonce starts over at zero.
//
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
unsigned int static ScanHash_CryptoPP ( char * pmidstate , char * pdata , char * phash1 , char * phash , unsigned int & nHashesDone )
2010-09-13 18:14:24 -04:00
{
2010-11-23 16:16:36 -03:00
unsigned int & nNonce = * ( unsigned int * ) ( pdata + 12 ) ;
2010-09-13 18:14:24 -04:00
for ( ; ; )
{
2012-07-25 20:48:39 -04:00
// Crypto++ SHA256
2010-11-23 16:16:36 -03:00
// Hash pdata using pmidstate as the starting state into
2012-07-25 20:48:39 -04:00
// pre-formatted buffer phash1, then hash phash1 into phash
2010-09-13 18:14:24 -04:00
nNonce + + ;
2010-11-23 16:16:36 -03:00
SHA256Transform ( phash1 , pdata , pmidstate ) ;
2010-09-13 18:14:24 -04:00
SHA256Transform ( phash , phash1 , pSHA256InitState ) ;
// Return the nonce if the hash has at least some zero bits,
// caller will check if it has enough to reach the target
if ( ( ( unsigned short * ) phash ) [ 14 ] = = 0 )
return nNonce ;
// If nothing found after trying for a while, return -1
if ( ( nNonce & 0xffff ) = = 0 )
{
nHashesDone = 0xffff + 1 ;
2012-04-22 14:51:16 -03:00
return ( unsigned int ) - 1 ;
2010-09-13 18:14:24 -04:00
}
}
}
2011-08-15 20:33:00 -04:00
// Some explaining would be appreciated
2010-11-19 17:22:46 -03:00
class COrphan
{
public :
CTransaction * ptx ;
set < uint256 > setDependsOn ;
double dPriority ;
2012-07-12 14:22:32 -04:00
double dFeePerKb ;
2010-11-19 17:22:46 -03:00
COrphan ( CTransaction * ptxIn )
{
ptx = ptxIn ;
2012-07-12 14:22:32 -04:00
dPriority = dFeePerKb = 0 ;
2010-11-19 17:22:46 -03:00
}
void print ( ) const
{
2012-07-12 14:22:32 -04:00
printf ( " COrphan(hash=%s, dPriority=%.1f, dFeePerKb=%.1f) \n " ,
ptx - > GetHash ( ) . ToString ( ) . substr ( 0 , 10 ) . c_str ( ) , dPriority , dFeePerKb ) ;
2011-05-15 03:11:04 -04:00
BOOST_FOREACH ( uint256 hash , setDependsOn )
2010-11-19 17:22:46 -03:00
printf ( " setDependsOn %s \n " , hash . ToString ( ) . substr ( 0 , 10 ) . c_str ( ) ) ;
}
} ;
2011-05-27 00:02:51 -04:00
uint64 nLastBlockTx = 0 ;
uint64 nLastBlockSize = 0 ;
2012-07-12 14:22:32 -04:00
// We want to sort transactions by priority and fee, so:
typedef boost : : tuple < double , double , CTransaction * > TxPriority ;
class TxPriorityCompare
{
bool byFee ;
public :
TxPriorityCompare ( bool _byFee ) : byFee ( _byFee ) { }
bool operator ( ) ( const TxPriority & a , const TxPriority & b )
{
if ( byFee )
{
if ( a . get < 1 > ( ) = = b . get < 1 > ( ) )
return a . get < 0 > ( ) < b . get < 0 > ( ) ;
return a . get < 1 > ( ) < b . get < 1 > ( ) ;
}
else
{
if ( a . get < 0 > ( ) = = b . get < 0 > ( ) )
return a . get < 1 > ( ) < b . get < 1 > ( ) ;
return a . get < 0 > ( ) < b . get < 0 > ( ) ;
}
}
} ;
2010-11-23 16:16:36 -03:00
CBlock * CreateNewBlock ( CReserveKey & reservekey )
{
// Create new block
auto_ptr < CBlock > pblock ( new CBlock ( ) ) ;
if ( ! pblock . get ( ) )
return NULL ;
// Create coinbase tx
CTransaction txNew ;
txNew . vin . resize ( 1 ) ;
txNew . vin [ 0 ] . prevout . SetNull ( ) ;
txNew . vout . resize ( 1 ) ;
txNew . vout [ 0 ] . scriptPubKey < < reservekey . GetReservedKey ( ) < < OP_CHECKSIG ;
// Add our coinbase tx as first transaction
pblock - > vtx . push_back ( txNew ) ;
2012-07-12 14:22:32 -04:00
// Largest block you're willing to create:
unsigned int nBlockMaxSize = GetArg ( " -blockmaxsize " , MAX_BLOCK_SIZE_GEN / 2 ) ;
// Limit to betweeen 1K and MAX_BLOCK_SIZE-1K for sanity:
nBlockMaxSize = std : : max ( ( unsigned int ) 1000 , std : : min ( ( unsigned int ) ( MAX_BLOCK_SIZE - 1000 ) , nBlockMaxSize ) ) ;
// How much of the block should be dedicated to high-priority transactions,
// included regardless of the fees they pay
unsigned int nBlockPrioritySize = GetArg ( " -blockprioritysize " , 27000 ) ;
nBlockPrioritySize = std : : min ( nBlockMaxSize , nBlockPrioritySize ) ;
// Minimum block size you want to create; block will be filled with free transactions
// until there are no more or the block reaches this size:
unsigned int nBlockMinSize = GetArg ( " -blockminsize " , 0 ) ;
nBlockMinSize = std : : min ( nBlockMaxSize , nBlockMinSize ) ;
// Fee-per-kilobyte amount considered the same as "free"
// Be careful setting this: if you set it to zero then
// a transaction spammer can cheaply fill blocks using
// 1-satoshi-fee transactions. It should be set above the real
// cost to you of processing a transaction.
int64 nMinTxFee = MIN_TX_FEE ;
if ( mapArgs . count ( " -mintxfee " ) )
ParseMoney ( mapArgs [ " -mintxfee " ] , nMinTxFee ) ;
2010-11-23 16:16:36 -03:00
// Collect memory pool transactions into the block
2011-12-21 18:33:19 -03:00
int64 nFees = 0 ;
2010-11-23 16:16:36 -03:00
{
2012-04-13 17:03:09 -03:00
LOCK2 ( cs_main , mempool . cs ) ;
2012-10-24 02:41:52 -03:00
CBlockIndex * pindexPrev = pindexBest ;
2012-07-06 10:33:34 -04:00
CCoinsViewCache view ( * pcoinsTip , true ) ;
2010-11-23 16:16:36 -03:00
// Priority order to process transactions
list < COrphan > vOrphan ; // list memory doesn't move
map < uint256 , vector < COrphan * > > mapDependers ;
2012-11-26 13:30:54 -03:00
bool fPrintPriority = GetBoolArg ( " -printpriority " ) ;
2012-07-12 14:22:32 -04:00
// This vector will be sorted into a priority queue:
vector < TxPriority > vecPriority ;
vecPriority . reserve ( mempool . mapTx . size ( ) ) ;
2012-04-13 17:03:09 -03:00
for ( map < uint256 , CTransaction > : : iterator mi = mempool . mapTx . begin ( ) ; mi ! = mempool . mapTx . end ( ) ; + + mi )
2010-11-23 16:16:36 -03:00
{
CTransaction & tx = ( * mi ) . second ;
if ( tx . IsCoinBase ( ) | | ! tx . IsFinal ( ) )
continue ;
COrphan * porphan = NULL ;
double dPriority = 0 ;
2012-07-12 14:22:32 -04:00
int64 nTotalIn = 0 ;
2012-07-26 15:29:59 -04:00
bool fMissingInputs = false ;
2011-05-15 03:11:04 -04:00
BOOST_FOREACH ( const CTxIn & txin , tx . vin )
2010-11-23 16:16:36 -03:00
{
// Read prev transaction
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
CCoins coins ;
if ( ! view . GetCoins ( txin . prevout . hash , coins ) )
2010-11-23 16:16:36 -03:00
{
2012-07-26 15:29:59 -04:00
// This should never happen; all transactions in the memory
// pool should connect to either transactions in the chain
// or other transactions in the memory pool.
if ( ! mempool . mapTx . count ( txin . prevout . hash ) )
{
printf ( " ERROR: mempool transaction missing input \n " ) ;
if ( fDebug ) assert ( " mempool transaction missing input " = = 0 ) ;
fMissingInputs = true ;
if ( porphan )
vOrphan . pop_back ( ) ;
break ;
}
2010-11-23 16:16:36 -03:00
// Has to wait for dependencies
if ( ! porphan )
{
// Use list for automatic deletion
vOrphan . push_back ( COrphan ( & tx ) ) ;
porphan = & vOrphan . back ( ) ;
}
mapDependers [ txin . prevout . hash ] . push_back ( porphan ) ;
porphan - > setDependsOn . insert ( txin . prevout . hash ) ;
2012-07-12 14:22:32 -04:00
nTotalIn + = mempool . mapTx [ txin . prevout . hash ] . vout [ txin . prevout . n ] . nValue ;
2010-11-23 16:16:36 -03:00
continue ;
}
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
int64 nValueIn = coins . vout [ txin . prevout . n ] . nValue ;
2012-07-12 14:22:32 -04:00
nTotalIn + = nValueIn ;
2010-11-23 16:16:36 -03:00
2012-10-22 19:10:12 -03:00
int nConf = pindexPrev - > nHeight - coins . nHeight + 1 ;
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
2010-11-23 16:16:36 -03:00
dPriority + = ( double ) nValueIn * nConf ;
}
2012-07-26 15:29:59 -04:00
if ( fMissingInputs ) continue ;
2010-11-23 16:16:36 -03:00
// Priority is sum(valuein * age) / txsize
2012-07-12 14:22:32 -04:00
unsigned int nTxSize = : : GetSerializeSize ( tx , SER_NETWORK , PROTOCOL_VERSION ) ;
dPriority / = nTxSize ;
2010-11-23 16:16:36 -03:00
2012-07-12 14:22:32 -04:00
// This is a more accurate fee-per-kilobyte than is used by the client code, because the
// client code rounds up the size to the nearest 1K. That's good, because it gives an
// incentive to create smaller transactions.
double dFeePerKb = double ( nTotalIn - tx . GetValueOut ( ) ) / ( double ( nTxSize ) / 1000.0 ) ;
2010-11-23 16:16:36 -03:00
2012-07-12 14:22:32 -04:00
if ( porphan )
2010-11-23 16:16:36 -03:00
{
2012-07-12 14:22:32 -04:00
porphan - > dPriority = dPriority ;
porphan - > dFeePerKb = dFeePerKb ;
2010-11-23 16:16:36 -03:00
}
2012-07-12 14:22:32 -04:00
else
vecPriority . push_back ( TxPriority ( dPriority , dFeePerKb , & ( * mi ) . second ) ) ;
2010-11-23 16:16:36 -03:00
}
// Collect transactions into block
2011-12-21 18:33:19 -03:00
uint64 nBlockSize = 1000 ;
2011-05-27 00:02:51 -04:00
uint64 nBlockTx = 0 ;
2012-01-20 19:07:40 -03:00
int nBlockSigOps = 100 ;
2012-07-12 14:22:32 -04:00
bool fSortedByFee = ( nBlockPrioritySize < = 0 ) ;
TxPriorityCompare comparer ( fSortedByFee ) ;
std : : make_heap ( vecPriority . begin ( ) , vecPriority . end ( ) , comparer ) ;
while ( ! vecPriority . empty ( ) )
2010-11-23 16:16:36 -03:00
{
2012-07-12 14:22:32 -04:00
// Take highest priority transaction off the priority queue:
double dPriority = vecPriority . front ( ) . get < 0 > ( ) ;
double dFeePerKb = vecPriority . front ( ) . get < 1 > ( ) ;
CTransaction & tx = * ( vecPriority . front ( ) . get < 2 > ( ) ) ;
std : : pop_heap ( vecPriority . begin ( ) , vecPriority . end ( ) , comparer ) ;
vecPriority . pop_back ( ) ;
2010-11-23 16:16:36 -03:00
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
// second layer cached modifications just for this transaction
CCoinsViewCache viewTemp ( view , true ) ;
2010-11-23 16:16:36 -03:00
// Size limits
2012-04-16 09:56:45 -03:00
unsigned int nTxSize = : : GetSerializeSize ( tx , SER_NETWORK , PROTOCOL_VERSION ) ;
2012-07-12 14:22:32 -04:00
if ( nBlockSize + nTxSize > = nBlockMaxSize )
2010-11-23 16:16:36 -03:00
continue ;
2012-01-04 23:40:52 -03:00
// Legacy limits on sigOps:
2012-04-23 15:14:03 -03:00
unsigned int nTxSigOps = tx . GetLegacySigOpCount ( ) ;
2012-01-20 19:07:40 -03:00
if ( nBlockSigOps + nTxSigOps > = MAX_BLOCK_SIGOPS )
2012-01-04 23:40:52 -03:00
continue ;
2012-07-12 14:22:32 -04:00
// Skip free transactions if we're past the minimum block size:
if ( fSortedByFee & & ( dFeePerKb < nMinTxFee ) & & ( nBlockSize + nTxSize > = nBlockMinSize ) )
continue ;
// Prioritize by fee once past the priority size or we run out of high-priority
// transactions:
if ( ! fSortedByFee & &
( ( nBlockSize + nTxSize > = nBlockPrioritySize ) | | ( dPriority < COIN * 144 / 250 ) ) )
{
fSortedByFee = true ;
comparer = TxPriorityCompare ( fSortedByFee ) ;
std : : make_heap ( vecPriority . begin ( ) , vecPriority . end ( ) , comparer ) ;
}
2010-11-23 16:16:36 -03:00
2012-07-08 13:04:05 -04:00
if ( ! tx . HaveInputs ( viewTemp ) )
2011-10-03 14:05:43 -03:00
continue ;
2012-01-04 23:40:52 -03:00
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
int64 nTxFees = tx . GetValueIn ( viewTemp ) - tx . GetValueOut ( ) ;
2012-01-10 22:18:00 -03:00
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
nTxSigOps + = tx . GetP2SHSigOpCount ( viewTemp ) ;
2012-01-20 19:07:40 -03:00
if ( nBlockSigOps + nTxSigOps > = MAX_BLOCK_SIGOPS )
2010-11-23 16:16:36 -03:00
continue ;
2012-01-10 22:18:00 -03:00
2012-11-13 19:03:25 -03:00
if ( ! tx . CheckInputs ( viewTemp , CS_ALWAYS , SCRIPT_VERIFY_P2SH ) )
2012-07-08 13:04:05 -04:00
continue ;
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
CTxUndo txundo ;
2012-07-07 18:06:34 -04:00
uint256 hash = tx . GetHash ( ) ;
if ( ! tx . UpdateCoins ( viewTemp , txundo , pindexPrev - > nHeight + 1 , hash ) )
2012-01-10 22:18:00 -03:00
continue ;
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
// push changes from the second layer cache to the first one
viewTemp . Flush ( ) ;
2010-11-23 16:16:36 -03:00
// Added
pblock - > vtx . push_back ( tx ) ;
nBlockSize + = nTxSize ;
2011-05-27 00:02:51 -04:00
+ + nBlockTx ;
2012-01-20 19:07:40 -03:00
nBlockSigOps + = nTxSigOps ;
2012-01-28 12:16:21 -03:00
nFees + = nTxFees ;
2010-11-23 16:16:36 -03:00
2012-11-26 13:30:54 -03:00
if ( fPrintPriority )
2012-07-12 14:22:32 -04:00
{
printf ( " priority %.1f feeperkb %.1f txid %s \n " ,
dPriority , dFeePerKb , tx . GetHash ( ) . ToString ( ) . c_str ( ) ) ;
}
2010-11-23 16:16:36 -03:00
// Add transactions that depend on this one to the priority queue
if ( mapDependers . count ( hash ) )
{
2011-05-15 03:11:04 -04:00
BOOST_FOREACH ( COrphan * porphan , mapDependers [ hash ] )
2010-11-23 16:16:36 -03:00
{
if ( ! porphan - > setDependsOn . empty ( ) )
{
porphan - > setDependsOn . erase ( hash ) ;
if ( porphan - > setDependsOn . empty ( ) )
2012-07-12 14:22:32 -04:00
{
vecPriority . push_back ( TxPriority ( porphan - > dPriority , porphan - > dFeePerKb , porphan - > ptx ) ) ;
std : : push_heap ( vecPriority . begin ( ) , vecPriority . end ( ) , comparer ) ;
}
2010-11-23 16:16:36 -03:00
}
}
}
}
2011-05-27 00:02:51 -04:00
nLastBlockTx = nBlockTx ;
nLastBlockSize = nBlockSize ;
2012-09-29 06:57:44 -03:00
printf ( " CreateNewBlock(): total size % " PRI64u " \n " , nBlockSize ) ;
2011-05-27 00:02:51 -04:00
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
pblock - > vtx [ 0 ] . vout [ 0 ] . nValue = GetBlockValue ( pindexPrev - > nHeight + 1 , nFees ) ;
2010-11-23 16:16:36 -03:00
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
// Fill in header
pblock - > hashPrevBlock = pindexPrev - > GetBlockHash ( ) ;
pblock - > UpdateTime ( pindexPrev ) ;
pblock - > nBits = GetNextWorkRequired ( pindexPrev , pblock . get ( ) ) ;
pblock - > nNonce = 0 ;
2012-10-12 18:49:44 -03:00
pblock - > vtx [ 0 ] . vin [ 0 ] . scriptSig = CScript ( ) < < OP_0 < < OP_0 ;
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
2012-06-18 19:36:43 -04:00
CBlockIndex indexDummy ( * pblock ) ;
2012-05-09 13:24:44 -04:00
indexDummy . pprev = pindexPrev ;
indexDummy . nHeight = pindexPrev - > nHeight + 1 ;
2012-07-06 10:33:34 -04:00
CCoinsViewCache viewNew ( * pcoinsTip , true ) ;
Ultraprune
This switches bitcoin's transaction/block verification logic to use a
"coin database", which contains all unredeemed transaction output scripts,
amounts and heights.
The name ultraprune comes from the fact that instead of a full transaction
index, we only (need to) keep an index with unspent outputs. For now, the
blocks themselves are kept as usual, although they are only necessary for
serving, rescanning and reorganizing.
The basic datastructures are CCoins (representing the coins of a single
transaction), and CCoinsView (representing a state of the coins database).
There are several implementations for CCoinsView. A dummy, one backed by
the coins database (coins.dat), one backed by the memory pool, and one
that adds a cache on top of it. FetchInputs, ConnectInputs, ConnectBlock,
DisconnectBlock, ... now operate on a generic CCoinsView.
The block switching logic now builds a single cached CCoinsView with
changes to be committed to the database before any changes are made.
This means no uncommitted changes are ever read from the database, and
should ease the transition to another database layer which does not
support transactions (but does support atomic writes), like LevelDB.
For the getrawtransaction() RPC call, access to a txid-to-disk index
would be preferable. As this index is not necessary or even useful
for any other part of the implementation, it is not provided. Instead,
getrawtransaction() uses the coin database to find the block height,
and then scans that block to find the requested transaction. This is
slow, but should suffice for debug purposes.
2012-07-01 12:54:00 -04:00
if ( ! pblock - > ConnectBlock ( & indexDummy , viewNew , true ) )
2012-05-09 13:24:44 -04:00
throw std : : runtime_error ( " CreateNewBlock() : ConnectBlock failed " ) ;
}
2010-11-23 16:16:36 -03:00
return pblock . release ( ) ;
}
2011-09-06 17:39:05 -03:00
void IncrementExtraNonce ( CBlock * pblock , CBlockIndex * pindexPrev , unsigned int & nExtraNonce )
2010-11-23 16:16:36 -03:00
{
// Update nExtraNonce
2011-06-12 20:16:54 -04:00
static uint256 hashPrevBlock ;
if ( hashPrevBlock ! = pblock - > hashPrevBlock )
2010-11-23 16:16:36 -03:00
{
2011-06-12 20:16:54 -04:00
nExtraNonce = 0 ;
hashPrevBlock = pblock - > hashPrevBlock ;
2010-11-23 16:16:36 -03:00
}
2011-06-12 20:16:54 -04:00
+ + nExtraNonce ;
2012-06-27 19:30:39 -04:00
unsigned int nHeight = pindexPrev - > nHeight + 1 ; // Height first in coinbase required for block.version=2
pblock - > vtx [ 0 ] . vin [ 0 ] . scriptSig = ( CScript ( ) < < nHeight < < CBigNum ( nExtraNonce ) ) + COINBASE_FLAGS ;
2011-10-13 17:03:58 -03:00
assert ( pblock - > vtx [ 0 ] . vin [ 0 ] . scriptSig . size ( ) < = 100 ) ;
2010-11-23 16:16:36 -03:00
pblock - > hashMerkleRoot = pblock - > BuildMerkleTree ( ) ;
}
void FormatHashBuffers ( CBlock * pblock , char * pmidstate , char * pdata , char * phash1 )
{
//
2012-07-25 20:48:39 -04:00
// Pre-build hash buffers
2010-11-23 16:16:36 -03:00
//
struct
{
struct unnamed2
{
int nVersion ;
uint256 hashPrevBlock ;
uint256 hashMerkleRoot ;
unsigned int nTime ;
unsigned int nBits ;
unsigned int nNonce ;
}
block ;
unsigned char pchPadding0 [ 64 ] ;
uint256 hash1 ;
unsigned char pchPadding1 [ 64 ] ;
}
tmp ;
memset ( & tmp , 0 , sizeof ( tmp ) ) ;
tmp . block . nVersion = pblock - > nVersion ;
tmp . block . hashPrevBlock = pblock - > hashPrevBlock ;
tmp . block . hashMerkleRoot = pblock - > hashMerkleRoot ;
tmp . block . nTime = pblock - > nTime ;
tmp . block . nBits = pblock - > nBits ;
tmp . block . nNonce = pblock - > nNonce ;
FormatHashBlocks ( & tmp . block , sizeof ( tmp . block ) ) ;
FormatHashBlocks ( & tmp . hash1 , sizeof ( tmp . hash1 ) ) ;
// Byte swap all the input buffer
2012-04-15 17:52:09 -03:00
for ( unsigned int i = 0 ; i < sizeof ( tmp ) / 4 ; i + + )
2010-11-23 16:16:36 -03:00
( ( unsigned int * ) & tmp ) [ i ] = ByteReverse ( ( ( unsigned int * ) & tmp ) [ i ] ) ;
// Precalc the first half of the first hash, which stays constant
SHA256Transform ( pmidstate , & tmp . block , pSHA256InitState ) ;
memcpy ( pdata , & tmp . block , 128 ) ;
memcpy ( phash1 , & tmp . hash1 , 64 ) ;
}
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
bool CheckWork ( CBlock * pblock , CWallet & wallet , CReserveKey & reservekey )
2010-11-23 16:16:36 -03:00
{
uint256 hash = pblock - > GetHash ( ) ;
uint256 hashTarget = CBigNum ( ) . SetCompact ( pblock - > nBits ) . getuint256 ( ) ;
if ( hash > hashTarget )
return false ;
//// debug print
printf ( " BitcoinMiner: \n " ) ;
printf ( " proof-of-work found \n hash: %s \n target: %s \n " , hash . GetHex ( ) . c_str ( ) , hashTarget . GetHex ( ) . c_str ( ) ) ;
pblock - > print ( ) ;
printf ( " generated %s \n " , FormatMoney ( pblock - > vtx [ 0 ] . vout [ 0 ] . nValue ) . c_str ( ) ) ;
// Found a solution
{
2012-04-06 13:39:12 -03:00
LOCK ( cs_main ) ;
2010-11-23 16:16:36 -03:00
if ( pblock - > hashPrevBlock ! = hashBestChain )
return error ( " BitcoinMiner : generated block is stale " ) ;
// Remove key from key pool
reservekey . KeepKey ( ) ;
// Track how many getdata requests this block gets
2012-04-06 13:39:12 -03:00
{
LOCK ( wallet . cs_wallet ) ;
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
wallet . mapRequestCount [ pblock - > GetHash ( ) ] = 0 ;
2012-04-06 13:39:12 -03:00
}
2010-11-23 16:16:36 -03:00
// Process this block the same as if we had received it from another node
if ( ! ProcessBlock ( NULL , pblock ) )
return error ( " BitcoinMiner : ProcessBlock, block not accepted " ) ;
}
return true ;
}
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
void static ThreadBitcoinMiner ( void * parg ) ;
2012-02-16 17:00:16 -03:00
static bool fGenerateBitcoins = false ;
static bool fLimitProcessors = false ;
static int nLimitProcessors = - 1 ;
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
void static BitcoinMiner ( CWallet * pwallet )
2010-08-29 12:58:15 -04:00
{
printf ( " BitcoinMiner started \n " ) ;
SetThreadPriority ( THREAD_PRIORITY_LOWEST ) ;
2012-06-24 11:03:57 -04:00
// Make this thread recognisable as the mining thread
2012-06-30 11:14:28 -04:00
RenameThread ( " bitcoin-miner " ) ;
2012-06-24 11:03:57 -04:00
2010-11-23 16:16:36 -03:00
// Each thread has its own key and counter
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
CReserveKey reservekey ( pwallet ) ;
2010-10-19 14:16:51 -03:00
unsigned int nExtraNonce = 0 ;
2010-11-23 16:16:36 -03:00
2010-08-29 12:58:15 -04:00
while ( fGenerateBitcoins )
{
if ( fShutdown )
return ;
while ( vNodes . empty ( ) | | IsInitialBlockDownload ( ) )
{
Sleep ( 1000 ) ;
if ( fShutdown )
return ;
if ( ! fGenerateBitcoins )
return ;
}
//
// Create new block
//
2010-11-23 16:16:36 -03:00
unsigned int nTransactionsUpdatedLast = nTransactionsUpdated ;
CBlockIndex * pindexPrev = pindexBest ;
auto_ptr < CBlock > pblock ( CreateNewBlock ( reservekey ) ) ;
2010-08-29 12:58:15 -04:00
if ( ! pblock . get ( ) )
return ;
2011-09-06 17:39:05 -03:00
IncrementExtraNonce ( pblock . get ( ) , pindexPrev , nExtraNonce ) ;
2010-08-29 12:58:15 -04:00
2012-09-29 06:57:44 -03:00
printf ( " Running BitcoinMiner with % " PRIszu " transactions in block (%u bytes) \n " , pblock - > vtx . size ( ) ,
2012-07-12 14:22:32 -04:00
: : GetSerializeSize ( * pblock , SER_NETWORK , PROTOCOL_VERSION ) ) ;
2010-08-29 12:58:15 -04:00
//
2012-07-25 20:48:39 -04:00
// Pre-build hash buffers
2010-08-29 12:58:15 -04:00
//
2010-11-23 16:16:36 -03:00
char pmidstatebuf [ 32 + 16 ] ; char * pmidstate = alignup < 16 > ( pmidstatebuf ) ;
char pdatabuf [ 128 + 16 ] ; char * pdata = alignup < 16 > ( pdatabuf ) ;
char phash1buf [ 64 + 16 ] ; char * phash1 = alignup < 16 > ( phash1buf ) ;
FormatHashBuffers ( pblock . get ( ) , pmidstate , pdata , phash1 ) ;
unsigned int & nBlockTime = * ( unsigned int * ) ( pdata + 64 + 4 ) ;
2012-02-16 12:22:31 -03:00
unsigned int & nBlockBits = * ( unsigned int * ) ( pdata + 64 + 8 ) ;
2010-11-23 16:16:36 -03:00
unsigned int & nBlockNonce = * ( unsigned int * ) ( pdata + 64 + 12 ) ;
2010-08-29 12:58:15 -04:00
//
// Search
//
2011-12-21 18:33:19 -03:00
int64 nStart = GetTime ( ) ;
2010-08-29 12:58:15 -04:00
uint256 hashTarget = CBigNum ( ) . SetCompact ( pblock - > nBits ) . getuint256 ( ) ;
uint256 hashbuf [ 2 ] ;
uint256 & hash = * alignup < 16 > ( hashbuf ) ;
loop
{
2010-09-13 18:14:24 -04:00
unsigned int nHashesDone = 0 ;
unsigned int nNonceFound ;
2012-07-25 20:48:39 -04:00
// Crypto++ SHA256
2011-04-03 15:21:52 -03:00
nNonceFound = ScanHash_CryptoPP ( pmidstate , pdata + 64 , phash1 ,
( char * ) & hash , nHashesDone ) ;
2010-08-29 12:58:15 -04:00
2010-09-13 18:14:24 -04:00
// Check if something found
2012-04-22 14:51:16 -03:00
if ( nNonceFound ! = ( unsigned int ) - 1 )
2010-08-29 12:58:15 -04:00
{
2012-04-15 17:52:09 -03:00
for ( unsigned int i = 0 ; i < sizeof ( hash ) / 4 ; i + + )
2010-08-29 12:58:15 -04:00
( ( unsigned int * ) & hash ) [ i ] = ByteReverse ( ( ( unsigned int * ) & hash ) [ i ] ) ;
if ( hash < = hashTarget )
{
2010-09-13 18:14:24 -04:00
// Found a solution
pblock - > nNonce = ByteReverse ( nNonceFound ) ;
2010-08-29 12:58:15 -04:00
assert ( hash = = pblock - > GetHash ( ) ) ;
SetThreadPriority ( THREAD_PRIORITY_NORMAL ) ;
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
CheckWork ( pblock . get ( ) , * pwalletMain , reservekey ) ;
2010-08-29 12:58:15 -04:00
SetThreadPriority ( THREAD_PRIORITY_LOWEST ) ;
break ;
}
}
2010-09-13 18:14:24 -04:00
// Meter hashes/sec
2011-12-21 18:33:19 -03:00
static int64 nHashCounter ;
2010-09-13 18:14:24 -04:00
if ( nHPSTimerStart = = 0 )
2010-08-29 12:58:15 -04:00
{
2010-09-13 18:14:24 -04:00
nHPSTimerStart = GetTimeMillis ( ) ;
nHashCounter = 0 ;
}
else
nHashCounter + = nHashesDone ;
if ( GetTimeMillis ( ) - nHPSTimerStart > 4000 )
{
static CCriticalSection cs ;
2010-08-29 12:58:15 -04:00
{
2012-04-06 13:39:12 -03:00
LOCK ( cs ) ;
2010-09-13 18:14:24 -04:00
if ( GetTimeMillis ( ) - nHPSTimerStart > 4000 )
2010-08-29 12:58:15 -04:00
{
2010-09-13 18:14:24 -04:00
dHashesPerSec = 1000.0 * nHashCounter / ( GetTimeMillis ( ) - nHPSTimerStart ) ;
nHPSTimerStart = GetTimeMillis ( ) ;
nHashCounter = 0 ;
2011-12-21 18:33:19 -03:00
static int64 nLogTime ;
2010-09-13 18:14:24 -04:00
if ( GetTime ( ) - nLogTime > 30 * 60 )
2010-08-29 12:58:15 -04:00
{
2010-09-13 18:14:24 -04:00
nLogTime = GetTime ( ) ;
2012-02-07 15:42:29 -03:00
printf ( " hashmeter %3d CPUs %6.0f khash/s \n " , vnThreadsRunning [ THREAD_MINER ] , dHashesPerSec / 1000.0 ) ;
2010-08-29 12:58:15 -04:00
}
}
}
2010-09-13 18:14:24 -04:00
}
2010-08-29 12:58:15 -04:00
2010-09-13 18:14:24 -04:00
// Check for stop or if block needs to be rebuilt
if ( fShutdown )
return ;
if ( ! fGenerateBitcoins )
return ;
2012-02-07 15:42:29 -03:00
if ( fLimitProcessors & & vnThreadsRunning [ THREAD_MINER ] > nLimitProcessors )
2010-09-13 18:14:24 -04:00
return ;
if ( vNodes . empty ( ) )
break ;
2010-11-23 16:16:36 -03:00
if ( nBlockNonce > = 0xffff0000 )
2010-09-13 18:14:24 -04:00
break ;
if ( nTransactionsUpdated ! = nTransactionsUpdatedLast & & GetTime ( ) - nStart > 60 )
break ;
if ( pindexPrev ! = pindexBest )
break ;
2010-08-29 12:58:15 -04:00
2010-09-13 18:14:24 -04:00
// Update nTime every few seconds
2012-02-16 12:22:31 -03:00
pblock - > UpdateTime ( pindexPrev ) ;
2010-11-23 16:16:36 -03:00
nBlockTime = ByteReverse ( pblock - > nTime ) ;
2012-02-16 12:22:31 -03:00
if ( fTestNet )
{
// Changing pblock->nTime can change work required on testnet:
nBlockBits = ByteReverse ( pblock - > nBits ) ;
hashTarget = CBigNum ( ) . SetCompact ( pblock - > nBits ) . getuint256 ( ) ;
}
2010-08-29 12:58:15 -04:00
}
}
}
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
void static ThreadBitcoinMiner ( void * parg )
2010-08-29 12:58:15 -04:00
{
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
CWallet * pwallet = ( CWallet * ) parg ;
2011-06-01 12:27:05 -04:00
try
2010-08-29 12:58:15 -04:00
{
2012-02-07 15:42:29 -03:00
vnThreadsRunning [ THREAD_MINER ] + + ;
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
BitcoinMiner ( pwallet ) ;
2012-02-07 15:42:29 -03:00
vnThreadsRunning [ THREAD_MINER ] - - ;
2011-03-17 18:54:20 -03:00
}
2011-06-01 12:27:05 -04:00
catch ( std : : exception & e ) {
2012-02-07 15:42:29 -03:00
vnThreadsRunning [ THREAD_MINER ] - - ;
2011-06-01 12:27:05 -04:00
PrintException ( & e , " ThreadBitcoinMiner() " ) ;
} catch ( . . . ) {
2012-02-07 15:42:29 -03:00
vnThreadsRunning [ THREAD_MINER ] - - ;
2011-06-01 12:27:05 -04:00
PrintException ( NULL , " ThreadBitcoinMiner() " ) ;
2010-08-29 12:58:15 -04:00
}
2011-06-01 12:27:05 -04:00
nHPSTimerStart = 0 ;
2012-02-07 15:42:29 -03:00
if ( vnThreadsRunning [ THREAD_MINER ] = = 0 )
2011-06-01 12:27:05 -04:00
dHashesPerSec = 0 ;
2012-02-07 15:42:29 -03:00
printf ( " ThreadBitcoinMiner exiting, %d threads remaining \n " , vnThreadsRunning [ THREAD_MINER ] ) ;
2010-11-09 16:47:07 -03:00
}
2010-08-29 12:58:15 -04:00
CWallet class
* A new class CKeyStore manages private keys, and script.cpp depends on access to CKeyStore.
* A new class CWallet extends CKeyStore, and contains all former wallet-specific globals; CWallet depends on script.cpp, not the other way around.
* Wallet-specific functions in CTransaction/CTxIn/CTxOut (GetDebit, GetCredit, GetChange, IsMine, IsFromMe), are moved to CWallet, taking their former 'this' argument as an explicit parameter
* CWalletTx objects know which CWallet they belong to, for convenience, so they have their own direct (and caching) GetDebit/... functions.
* Some code was moved from CWalletDB to CWallet, such as handling of reserve keys.
* Main.cpp keeps a set of all 'registered' wallets, which should be informed about updates to the block chain, and does not have any notion about any 'main' wallet. Function in main.cpp that require a wallet (such as GenerateCoins), take an explicit CWallet* argument.
* The actual CWallet instance used by the application is defined in init.cpp as "CWallet* pwalletMain". rpc.cpp and ui.cpp use this variable.
* Functions in main.cpp and db.cpp that are not used by other modules are marked static.
* The code for handling the 'submitorder' message is removed, as it not really compatible with the idea that a node is independent from the wallet(s) connected to it, and obsolete anyway.
2011-06-01 12:28:20 -04:00
void GenerateBitcoins ( bool fGenerate , CWallet * pwallet )
2010-08-29 12:58:15 -04:00
{
2012-02-16 17:00:16 -03:00
fGenerateBitcoins = fGenerate ;
nLimitProcessors = GetArg ( " -genproclimit " , - 1 ) ;
if ( nLimitProcessors = = 0 )
fGenerateBitcoins = false ;
fLimitProcessors = ( nLimitProcessors ! = - 1 ) ;
if ( fGenerate )
2010-08-29 12:58:15 -04:00
{
2011-06-01 12:27:05 -04:00
int nProcessors = boost : : thread : : hardware_concurrency ( ) ;
printf ( " %d processors \n " , nProcessors ) ;
if ( nProcessors < 1 )
nProcessors = 1 ;
if ( fLimitProcessors & & nProcessors > nLimitProcessors )
nProcessors = nLimitProcessors ;
2012-02-07 15:42:29 -03:00
int nAddThreads = nProcessors - vnThreadsRunning [ THREAD_MINER ] ;
2011-06-01 12:27:05 -04:00
printf ( " Starting %d BitcoinMiner threads \n " , nAddThreads ) ;
for ( int i = 0 ; i < nAddThreads ; i + + )
2010-08-29 12:58:15 -04:00
{
2012-08-29 14:25:37 -04:00
if ( ! NewThread ( ThreadBitcoinMiner , pwallet ) )
printf ( " Error: NewThread(ThreadBitcoinMiner) failed \n " ) ;
2011-06-01 12:27:05 -04:00
Sleep ( 10 ) ;
2010-08-29 12:58:15 -04:00
}
2011-04-04 23:24:35 -03:00
}
2010-08-29 12:58:15 -04:00
}
2012-06-16 07:36:00 -04:00
// Amount compression:
// * If the amount is 0, output 0
// * first, divide the amount (in base units) by the largest power of 10 possible; call the exponent e (e is max 9)
// * if e<9, the last digit of the resulting number cannot be 0; store it as d, and drop it (divide by 10)
// * call the result n
// * output 1 + 10*(9*n + d - 1) + e
// * if e==9, we only know the resulting number is not zero, so output 1 + 10*(n - 1) + 9
// (this is decodable, as d is in [1-9] and e is in [0-9])
uint64 CTxOutCompressor : : CompressAmount ( uint64 n )
{
if ( n = = 0 )
return 0 ;
int e = 0 ;
while ( ( ( n % 10 ) = = 0 ) & & e < 9 ) {
n / = 10 ;
e + + ;
}
if ( e < 9 ) {
int d = ( n % 10 ) ;
assert ( d > = 1 & & d < = 9 ) ;
n / = 10 ;
return 1 + ( n * 9 + d - 1 ) * 10 + e ;
} else {
return 1 + ( n - 1 ) * 10 + 9 ;
}
}
uint64 CTxOutCompressor : : DecompressAmount ( uint64 x )
{
// x = 0 OR x = 1+10*(9*n + d - 1) + e OR x = 1+10*(n - 1) + 9
if ( x = = 0 )
return 0 ;
x - - ;
// x = 10*(9*n + d - 1) + e
int e = x % 10 ;
x / = 10 ;
uint64 n = 0 ;
if ( e < 9 ) {
// x = 9*n + d - 1
int d = ( x % 9 ) + 1 ;
x / = 9 ;
// x = n
n = x * 10 + d ;
} else {
n = x + 1 ;
}
while ( e ) {
n * = 10 ;
e - - ;
}
return n ;
}