2011-05-14 20:10:21 +02:00
// Copyright (c) 2009-2010 Satoshi Nakamoto
2012-02-07 17:28:30 +01:00
// Copyright (c) 2009-2012 The Bitcoin developers
2011-05-14 20:10:21 +02:00
// Distributed under the MIT/X11 software license, see the accompanying
// file license.txt or http://www.opensource.org/licenses/mit-license.php.
2011-06-18 18:46:01 +02:00
# include "db.h"
2012-04-15 23:39:49 +02:00
# include "walletdb.h"
2011-07-03 20:53:56 +02:00
# include "bitcoinrpc.h"
2011-06-18 18:46:01 +02:00
# include "net.h"
# include "init.h"
2012-04-15 22:10:54 +02:00
# include "util.h"
# include "ui_interface.h"
2011-06-26 19:23:24 +02:00
# include <boost/filesystem.hpp>
2011-06-18 18:46:01 +02:00
# include <boost/filesystem/fstream.hpp>
2011-12-20 00:49:07 +01:00
# include <boost/filesystem/convenience.hpp>
2011-06-18 18:46:01 +02:00
# include <boost/interprocess/sync/file_lock.hpp>
2011-05-14 20:10:21 +02:00
2012-04-15 22:10:54 +02:00
# ifndef WIN32
# include <signal.h>
2012-03-18 23:14:03 +01:00
# endif
2011-10-07 16:46:56 +02:00
2011-05-14 20:10:21 +02:00
using namespace std ;
using namespace boost ;
2011-06-26 19:23:24 +02:00
CWallet * pwalletMain ;
2011-05-14 20:10:21 +02:00
//////////////////////////////////////////////////////////////////////////////
//
// Shutdown
//
void ExitTimeout ( void * parg )
{
2011-10-07 17:02:21 +02:00
# ifdef WIN32
2011-05-14 20:10:21 +02:00
Sleep ( 5000 ) ;
ExitProcess ( 0 ) ;
# endif
}
void Shutdown ( void * parg )
{
static CCriticalSection cs_Shutdown ;
static bool fTaken ;
2011-11-11 03:12:46 +01:00
bool fFirstThread = false ;
2011-05-14 20:10:21 +02:00
{
2012-04-06 18:39:12 +02:00
TRY_LOCK ( cs_Shutdown , lockShutdown ) ;
if ( lockShutdown )
{
fFirstThread = ! fTaken ;
fTaken = true ;
}
2011-05-14 20:10:21 +02:00
}
static bool fExit ;
if ( fFirstThread )
{
fShutdown = true ;
nTransactionsUpdated + + ;
2011-11-17 20:21:32 +01:00
DBFlush ( false ) ;
2011-05-14 20:10:21 +02:00
StopNode ( ) ;
2011-11-17 20:21:32 +01:00
DBFlush ( true ) ;
2011-05-14 20:10:21 +02:00
boost : : filesystem : : remove ( GetPidFile ( ) ) ;
2011-06-26 19:23:24 +02:00
UnregisterWallet ( pwalletMain ) ;
delete pwalletMain ;
2011-05-14 20:10:21 +02:00
CreateThread ( ExitTimeout , NULL ) ;
Sleep ( 50 ) ;
printf ( " Bitcoin exiting \n \n " ) ;
fExit = true ;
exit ( 0 ) ;
}
else
{
while ( ! fExit )
Sleep ( 500 ) ;
Sleep ( 100 ) ;
ExitThread ( 0 ) ;
}
}
void HandleSIGTERM ( int )
{
fRequestShutdown = true ;
}
//////////////////////////////////////////////////////////////////////////////
//
// Start
//
2011-09-26 16:04:04 +02:00
# if !defined(QT_GUI)
2011-05-14 20:10:21 +02:00
int main ( int argc , char * argv [ ] )
{
bool fRet = false ;
fRet = AppInit ( argc , argv ) ;
if ( fRet & & fDaemon )
return 0 ;
return 1 ;
}
# endif
bool AppInit ( int argc , char * argv [ ] )
{
bool fRet = false ;
try
{
fRet = AppInit2 ( argc , argv ) ;
}
catch ( std : : exception & e ) {
PrintException ( & e , " AppInit() " ) ;
} catch ( . . . ) {
PrintException ( NULL , " AppInit() " ) ;
}
if ( ! fRet )
Shutdown ( NULL ) ;
return fRet ;
}
bool AppInit2 ( int argc , char * argv [ ] )
{
# ifdef _MSC_VER
// Turn off microsoft heap dump noise
_CrtSetReportMode ( _CRT_WARN , _CRTDBG_MODE_FILE ) ;
_CrtSetReportFile ( _CRT_WARN , CreateFileA ( " NUL " , GENERIC_WRITE , 0 , NULL , OPEN_EXISTING , 0 , 0 ) ) ;
# endif
# if _MSC_VER >= 1400
// Disable confusing "helpful" text message on abort, ctrl-c
_set_abort_behavior ( 0 , _WRITE_ABORT_MSG | _CALL_REPORTFAULT ) ;
# endif
2011-10-07 17:02:21 +02:00
# ifndef WIN32
2011-05-14 20:10:21 +02:00
umask ( 077 ) ;
# endif
2011-10-07 17:02:21 +02:00
# ifndef WIN32
2011-05-14 20:10:21 +02:00
// Clean shutdown on SIGTERM
struct sigaction sa ;
sa . sa_handler = HandleSIGTERM ;
sigemptyset ( & sa . sa_mask ) ;
sa . sa_flags = 0 ;
sigaction ( SIGTERM , & sa , NULL ) ;
sigaction ( SIGINT , & sa , NULL ) ;
sigaction ( SIGHUP , & sa , NULL ) ;
# endif
//
// Parameters
//
2012-02-17 03:09:41 +01:00
// If Qt is used, parameters/bitcoin.conf are parsed in qt/bitcoin.cpp's main()
2011-12-09 19:19:27 +01:00
# if !defined(QT_GUI)
2011-05-14 20:10:21 +02:00
ParseParameters ( argc , argv ) ;
2012-04-22 14:35:22 +02:00
if ( ! boost : : filesystem : : is_directory ( GetDataDir ( false ) ) )
2011-05-14 20:10:21 +02:00
{
2012-02-17 03:09:41 +01:00
fprintf ( stderr , " Error: Specified directory does not exist \n " ) ;
Shutdown ( NULL ) ;
2011-05-14 20:10:21 +02:00
}
2012-04-22 14:35:22 +02:00
ReadConfigFile ( mapArgs , mapMultiArgs ) ;
2012-02-17 03:09:41 +01:00
# endif
2011-05-14 20:10:21 +02:00
if ( mapArgs . count ( " -? " ) | | mapArgs . count ( " --help " ) )
{
string strUsage = string ( ) +
_ ( " Bitcoin version " ) + " " + FormatFullVersion ( ) + " \n \n " +
_ ( " Usage: " ) + " \t \t \t \t \t \t \t \t \t \t \n " +
2011-11-27 20:53:30 +01:00
" bitcoind [options] \t " + " \n " +
2011-12-13 06:46:37 +01:00
" bitcoind [options] <command> [params] \t " + _ ( " Send command to -server or bitcoind " ) + " \n " +
" bitcoind [options] help \t \t " + _ ( " List commands " ) + " \n " +
" bitcoind [options] help <command> \t \t " + _ ( " Get help for a command " ) + " \n " +
_ ( " Options: " ) + " \n " +
" -conf=<file> \t \t " + _ ( " Specify configuration file (default: bitcoin.conf) " ) + " \n " +
" -pid=<file> \t \t " + _ ( " Specify pid file (default: bitcoind.pid) " ) + " \n " +
" -gen \t \t " + _ ( " Generate coins " ) + " \n " +
" -gen=0 \t \t " + _ ( " Don't generate coins " ) + " \n " +
" -min \t \t " + _ ( " Start minimized " ) + " \n " +
2012-02-25 03:54:18 +01:00
" -splash \t \t " + _ ( " Show splash screen on startup (default: 1) " ) + " \n " +
2011-12-13 06:46:37 +01:00
" -datadir=<dir> \t \t " + _ ( " Specify data directory " ) + " \n " +
2012-03-21 00:03:02 +01:00
" -dbcache=<n> \t \t " + _ ( " Set database cache size in megabytes (default: 25) " ) + " \n " +
2012-04-05 17:54:30 +02:00
" -dblogsize=<n> \t \t " + _ ( " Set database disk log size in megabytes (default: 100) " ) + " \n " +
2011-12-13 06:46:37 +01:00
" -timeout=<n> \t " + _ ( " Specify connection timeout (in milliseconds) " ) + " \n " +
2012-04-01 20:25:48 +02:00
" -proxy=<ip:port> \t " + _ ( " Connect through socks proxy " ) + " \n " +
" -socks=<n> \t " + _ ( " Select the version of socks proxy to use (4 or 5, 5 is default) " ) + " \n " +
2012-05-04 16:55:19 +02:00
" -noproxy=<net> \t " + _ ( " Do not use proxy for connections to network net (ipv4 or ipv6) " ) + " \n " +
2012-04-24 02:15:00 +02:00
" -dns \t " + _ ( " Allow DNS lookups for -addnode, -seednode and -connect " ) + " \n " +
2012-04-19 17:38:03 +02:00
" -proxydns \t " + _ ( " Pass DNS requests to (SOCKS5) proxy " ) + " \n " +
2011-12-13 06:46:37 +01:00
" -port=<port> \t \t " + _ ( " Listen for connections on <port> (default: 8333 or testnet: 18333) " ) + " \n " +
" -maxconnections=<n> \t " + _ ( " Maintain at most <n> connections to peers (default: 125) " ) + " \n " +
2011-12-17 01:48:03 +01:00
" -addnode=<ip> \t " + _ ( " Add a node to connect to and attempt to keep the connection open " ) + " \n " +
2011-12-13 06:46:37 +01:00
" -connect=<ip> \t \t " + _ ( " Connect only to the specified node " ) + " \n " +
2012-04-24 02:15:00 +02:00
" -seednode=<ip> \t \t " + _ ( " Connect to a node to retrieve peer addresses, and disconnect " ) + " \n " +
2012-02-19 20:44:35 +01:00
" -externalip=<ip> \t " + _ ( " Specify your own public address " ) + " \n " +
2012-05-04 16:46:22 +02:00
" -blocknet=<net> \t " + _ ( " Do not connect to addresses in network net (ipv4, ipv6) " ) + " \n " +
2012-02-19 20:44:35 +01:00
" -discover \t " + _ ( " Try to discover public IP address (default: 1) " ) + " \n " +
2012-02-06 20:35:57 +01:00
" -irc \t " + _ ( " Find peers using internet relay chat (default: 0) " ) + " \n " +
" -listen \t " + _ ( " Accept connections from outside (default: 1) " ) + " \n " +
2012-02-17 18:44:39 +01:00
# ifdef QT_GUI
" -lang=<lang> \t \t " + _ ( " Set language, for example \" de_DE \" (default: system locale) " ) + " \n " +
# endif
2012-02-06 20:35:57 +01:00
" -dnsseed \t " + _ ( " Find peers using DNS lookup (default: 1) " ) + " \n " +
2011-12-13 06:46:37 +01:00
" -banscore=<n> \t " + _ ( " Threshold for disconnecting misbehaving peers (default: 100) " ) + " \n " +
" -bantime=<n> \t " + _ ( " Number of seconds to keep misbehaving peers from reconnecting (default: 86400) " ) + " \n " +
" -maxreceivebuffer=<n> \t " + _ ( " Maximum per-connection receive buffer, <n>*1000 bytes (default: 10000) " ) + " \n " +
" -maxsendbuffer=<n> \t " + _ ( " Maximum per-connection send buffer, <n>*1000 bytes (default: 10000) " ) + " \n " +
2011-05-14 20:10:21 +02:00
# ifdef USE_UPNP
# if USE_UPNP
2012-02-06 20:35:57 +01:00
" -upnp \t " + _ ( " Use Universal Plug and Play to map the listening port (default: 1) " ) + " \n " +
2011-05-14 20:10:21 +02:00
# else
2012-02-06 20:35:57 +01:00
" -upnp \t " + _ ( " Use Universal Plug and Play to map the listening port (default: 0) " ) + " \n " +
2011-05-14 20:10:21 +02:00
# endif
2012-04-17 23:03:24 +02:00
" -detachdb \t " + _ ( " Detach block and address databases. Increases shutdown time (default: 0) " ) + " \n " +
2011-05-14 20:10:21 +02:00
# endif
2011-12-13 06:46:37 +01:00
" -paytxfee=<amt> \t " + _ ( " Fee per KB to add to transactions you send " ) + " \n " +
2012-02-17 18:00:41 +01:00
# ifdef QT_GUI
2011-12-13 06:46:37 +01:00
" -server \t \t " + _ ( " Accept command line and JSON-RPC commands " ) + " \n " +
2011-05-14 20:10:21 +02:00
# endif
2012-02-17 18:00:41 +01:00
# if !defined(WIN32) && !defined(QT_GUI)
2011-12-13 06:46:37 +01:00
" -daemon \t \t " + _ ( " Run in the background as a daemon and accept commands " ) + " \n " +
2011-05-14 20:10:21 +02:00
# endif
2011-12-13 06:46:37 +01:00
" -testnet \t \t " + _ ( " Use the test network " ) + " \n " +
" -debug \t \t " + _ ( " Output extra debugging information " ) + " \n " +
" -logtimestamps \t " + _ ( " Prepend debug output with timestamp " ) + " \n " +
" -printtoconsole \t " + _ ( " Send trace/debug info to console instead of debug.log file " ) + " \n " +
2011-11-27 20:53:30 +01:00
# ifdef WIN32
2011-12-13 06:46:37 +01:00
" -printtodebugger \t " + _ ( " Send trace/debug info to debugger " ) + " \n " +
2011-11-27 20:53:30 +01:00
# endif
2011-12-13 06:46:37 +01:00
" -rpcuser=<user> \t " + _ ( " Username for JSON-RPC connections " ) + " \n " +
" -rpcpassword=<pw> \t " + _ ( " Password for JSON-RPC connections " ) + " \n " +
" -rpcport=<port> \t \t " + _ ( " Listen for JSON-RPC connections on <port> (default: 8332) " ) + " \n " +
" -rpcallowip=<ip> \t \t " + _ ( " Allow JSON-RPC connections from specified IP address " ) + " \n " +
" -rpcconnect=<ip> \t " + _ ( " Send commands to node running on <ip> (default: 127.0.0.1) " ) + " \n " +
2012-01-03 21:24:28 +01:00
" -blocknotify=<cmd> " + _ ( " Execute command when the best block changes (%s in cmd is replaced by block hash) " ) + " \n " +
2012-03-22 03:56:31 +01:00
" -upgradewallet \t " + _ ( " Upgrade wallet to latest format " ) + " \n " +
2011-12-13 06:46:37 +01:00
" -keypool=<n> \t " + _ ( " Set key pool size to <n> (default: 100) " ) + " \n " +
2012-03-21 23:08:13 +01:00
" -rescan \t " + _ ( " Rescan the block chain for missing wallet transactions " ) + " \n " +
" -checkblocks=<n> \t \t " + _ ( " How many blocks to check at startup (default: 2500, 0 = all) " ) + " \n " +
2012-02-20 20:50:26 +01:00
" -checklevel=<n> \t \t " + _ ( " How thorough the block verification is (0-6, default: 1) " ) + " \n " +
" -loadblock=<file> \t " + _ ( " Imports blocks from external blk000?.dat file " ) + " \n " ;
2011-05-14 20:10:21 +02:00
strUsage + = string ( ) +
2011-12-13 06:46:37 +01:00
_ ( " \n SSL options: (see the Bitcoin Wiki for SSL setup instructions) " ) + " \n " +
" -rpcssl \t " + _ ( " Use OpenSSL (https) for JSON-RPC connections " ) + " \n " +
" -rpcsslcertificatechainfile=<file.cert> \t " + _ ( " Server certificate file (default: server.cert) " ) + " \n " +
" -rpcsslprivatekeyfile=<file.pem> \t " + _ ( " Server private key (default: server.pem) " ) + " \n " +
" -rpcsslciphers=<ciphers> \t " + _ ( " Acceptable ciphers (default: TLSv1+HIGH:!SSLv2:!aNULL:!eNULL:!AH:!3DES:@STRENGTH) " ) + " \n " ;
2011-05-14 20:10:21 +02:00
strUsage + = string ( ) +
2011-12-13 06:46:37 +01:00
" -? \t \t " + _ ( " This help message " ) + " \n " ;
2011-05-14 20:10:21 +02:00
// Remove tabs
strUsage . erase ( std : : remove ( strUsage . begin ( ) , strUsage . end ( ) , ' \t ' ) , strUsage . end ( ) ) ;
2012-02-17 17:53:41 +01:00
# if defined(QT_GUI) && defined(WIN32)
// On windows, show a message box, as there is no stderr
2012-03-25 20:47:33 +02:00
ThreadSafeMessageBox ( strUsage , _ ( " Usage " ) , wxOK | wxMODAL ) ;
2012-02-17 17:53:41 +01:00
# else
2011-05-14 20:10:21 +02:00
fprintf ( stderr , " %s " , strUsage . c_str ( ) ) ;
2012-02-17 17:53:41 +01:00
# endif
2011-05-14 20:10:21 +02:00
return false ;
}
2012-01-03 17:17:04 +01:00
fTestNet = GetBoolArg ( " -testnet " ) ;
2012-02-22 05:17:49 +01:00
if ( fTestNet )
{
SoftSetBoolArg ( " -irc " , true ) ;
}
2011-05-14 20:10:21 +02:00
fDebug = GetBoolArg ( " -debug " ) ;
2012-04-17 23:03:24 +02:00
fDetachDB = GetBoolArg ( " -detachdb " , false ) ;
2011-05-14 20:10:21 +02:00
2012-02-17 18:00:41 +01:00
# if !defined(WIN32) && !defined(QT_GUI)
2011-05-14 20:10:21 +02:00
fDaemon = GetBoolArg ( " -daemon " ) ;
# else
fDaemon = false ;
# endif
if ( fDaemon )
fServer = true ;
else
fServer = GetBoolArg ( " -server " ) ;
/* force fServer when running without GUI */
2011-09-26 16:04:04 +02:00
# if !defined(QT_GUI)
2011-05-14 20:10:21 +02:00
fServer = true ;
2011-05-22 19:32:18 +02:00
# endif
2011-05-14 20:10:21 +02:00
fPrintToConsole = GetBoolArg ( " -printtoconsole " ) ;
fPrintToDebugger = GetBoolArg ( " -printtodebugger " ) ;
fLogTimestamps = GetBoolArg ( " -logtimestamps " ) ;
2011-09-02 18:05:08 +02:00
# ifndef QT_GUI
2011-05-14 20:10:21 +02:00
for ( int i = 1 ; i < argc ; i + + )
2011-12-24 05:27:12 +01:00
if ( ! IsSwitchChar ( argv [ i ] [ 0 ] ) & & ! ( strlen ( argv [ i ] ) > 7 & & strncasecmp ( argv [ i ] , " bitcoin: " , 8 ) = = 0 ) )
2011-05-14 20:10:21 +02:00
fCommandLine = true ;
if ( fCommandLine )
{
int ret = CommandLineRPC ( argc , argv ) ;
exit ( ret ) ;
}
2011-09-02 18:05:08 +02:00
# endif
2011-05-14 20:10:21 +02:00
2012-02-17 18:00:41 +01:00
# if !defined(WIN32) && !defined(QT_GUI)
2011-05-14 20:10:21 +02:00
if ( fDaemon )
{
// Daemonize
pid_t pid = fork ( ) ;
if ( pid < 0 )
{
fprintf ( stderr , " Error: fork() returned %d errno %d \n " , pid , errno ) ;
return false ;
}
if ( pid > 0 )
{
CreatePidFile ( GetPidFile ( ) , pid ) ;
return true ;
}
pid_t sid = setsid ( ) ;
if ( sid < 0 )
fprintf ( stderr , " Error: setsid() returned %d errno %d \n " , sid , errno ) ;
}
# endif
2012-04-09 23:50:56 +02:00
if ( ! fDebug )
2011-05-14 20:10:21 +02:00
ShrinkDebugFile ( ) ;
printf ( " \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n " ) ;
2012-04-07 02:06:53 +02:00
printf ( " Bitcoin version %s (%s) \n " , FormatFullVersion ( ) . c_str ( ) , CLIENT_DATE . c_str ( ) ) ;
2012-04-09 23:50:56 +02:00
printf ( " Default data directory %s \n " , GetDefaultDataDir ( ) . string ( ) . c_str ( ) ) ;
2011-05-14 20:10:21 +02:00
if ( GetBoolArg ( " -loadblockindextest " ) )
{
CTxDB txdb ( " r " ) ;
txdb . LoadBlockIndex ( ) ;
PrintBlockTree ( ) ;
return false ;
}
// Make sure only a single bitcoin process is using the data directory.
2012-04-09 23:50:56 +02:00
boost : : filesystem : : path pathLockFile = GetDataDir ( ) / " .lock " ;
FILE * file = fopen ( pathLockFile . string ( ) . c_str ( ) , " a " ) ; // empty lock file; created if it doesn't exist.
2011-05-14 20:10:21 +02:00
if ( file ) fclose ( file ) ;
2012-04-09 23:50:56 +02:00
static boost : : interprocess : : file_lock lock ( pathLockFile . string ( ) . c_str ( ) ) ;
2011-05-14 20:10:21 +02:00
if ( ! lock . try_lock ( ) )
{
2012-04-09 23:50:56 +02:00
ThreadSafeMessageBox ( strprintf ( _ ( " Cannot obtain a lock on data directory %s. Bitcoin is probably already running. " ) , GetDataDir ( ) . string ( ) . c_str ( ) ) , _ ( " Bitcoin " ) , wxOK | wxMODAL ) ;
2011-05-14 20:10:21 +02:00
return false ;
}
2011-12-16 23:13:45 +01:00
std : : ostringstream strErrors ;
2011-05-14 20:10:21 +02:00
//
// Load data files
//
if ( fDaemon )
fprintf ( stdout , " bitcoin server starting \n " ) ;
2011-12-21 22:33:19 +01:00
int64 nStart ;
2011-05-14 20:10:21 +02:00
2011-08-28 14:12:26 +02:00
InitMessage ( _ ( " Loading addresses... " ) ) ;
2011-05-14 20:10:21 +02:00
printf ( " Loading addresses... \n " ) ;
nStart = GetTimeMillis ( ) ;
if ( ! LoadAddresses ( ) )
2011-12-16 23:13:45 +01:00
strErrors < < _ ( " Error loading addr.dat " ) < < " \n " ;
2011-05-14 20:10:21 +02:00
printf ( " addresses %15 " PRI64d " ms \n " , GetTimeMillis ( ) - nStart ) ;
2011-08-28 14:12:26 +02:00
InitMessage ( _ ( " Loading block index... " ) ) ;
2011-05-14 20:10:21 +02:00
printf ( " Loading block index... \n " ) ;
nStart = GetTimeMillis ( ) ;
if ( ! LoadBlockIndex ( ) )
2011-12-16 23:13:45 +01:00
strErrors < < _ ( " Error loading blkindex.dat " ) < < " \n " ;
2012-04-18 13:30:24 +02:00
// as LoadBlockIndex can take several minutes, it's possible the user
// requested to kill bitcoin-qt during the last operation. If so, exit.
// As the program has not fully started yet, Shutdown() is possibly overkill.
if ( fRequestShutdown )
{
printf ( " Shutdown requested. Exiting. \n " ) ;
return false ;
}
2011-05-14 20:10:21 +02:00
printf ( " block index %15 " PRI64d " ms \n " , GetTimeMillis ( ) - nStart ) ;
2012-02-20 20:50:26 +01:00
if ( mapArgs . count ( " -loadblock " ) )
{
BOOST_FOREACH ( string strFile , mapMultiArgs [ " -loadblock " ] )
{
FILE * file = fopen ( strFile . c_str ( ) , " rb " ) ;
if ( file )
LoadExternalBlockFile ( file ) ;
}
}
2011-08-28 14:12:26 +02:00
InitMessage ( _ ( " Loading wallet... " ) ) ;
2011-05-14 20:10:21 +02:00
printf ( " Loading wallet... \n " ) ;
nStart = GetTimeMillis ( ) ;
bool fFirstRun ;
2011-06-26 19:23:24 +02:00
pwalletMain = new CWallet ( " wallet.dat " ) ;
2011-07-05 03:06:19 +02:00
int nLoadWalletRet = pwalletMain - > LoadWallet ( fFirstRun ) ;
if ( nLoadWalletRet ! = DB_LOAD_OK )
{
if ( nLoadWalletRet = = DB_CORRUPT )
2011-12-16 23:13:45 +01:00
strErrors < < _ ( " Error loading wallet.dat: Wallet corrupted " ) < < " \n " ;
2011-07-05 03:06:19 +02:00
else if ( nLoadWalletRet = = DB_TOO_NEW )
2011-12-16 23:13:45 +01:00
strErrors < < _ ( " Error loading wallet.dat: Wallet requires newer version of Bitcoin " ) < < " \n " ;
2011-11-11 03:12:46 +01:00
else if ( nLoadWalletRet = = DB_NEED_REWRITE )
{
2011-12-16 23:13:45 +01:00
strErrors < < _ ( " Wallet needed to be rewritten: restart Bitcoin to complete " ) < < " \n " ;
2012-03-11 22:57:44 +01:00
printf ( " %s " , strErrors . str ( ) . c_str ( ) ) ;
2012-03-25 20:47:33 +02:00
ThreadSafeMessageBox ( strErrors . str ( ) , _ ( " Bitcoin " ) , wxOK | wxICON_ERROR | wxMODAL ) ;
2011-11-11 03:12:46 +01:00
return false ;
}
2011-07-05 03:06:19 +02:00
else
2011-12-16 23:13:45 +01:00
strErrors < < _ ( " Error loading wallet.dat " ) < < " \n " ;
2011-07-05 03:06:19 +02:00
}
2012-03-22 03:56:31 +01:00
if ( GetBoolArg ( " -upgradewallet " , fFirstRun ) )
{
int nMaxVersion = GetArg ( " -upgradewallet " , 0 ) ;
if ( nMaxVersion = = 0 ) // the -walletupgrade without argument case
{
printf ( " Performing wallet upgrade to %i \n " , FEATURE_LATEST ) ;
nMaxVersion = CLIENT_VERSION ;
pwalletMain - > SetMinVersion ( FEATURE_LATEST ) ; // permanently upgrade the wallet immediately
}
else
printf ( " Allowing wallet upgrade up to %i \n " , nMaxVersion ) ;
if ( nMaxVersion < pwalletMain - > GetVersion ( ) )
strErrors < < _ ( " Cannot downgrade wallet " ) < < " \n " ;
pwalletMain - > SetMaxVersion ( nMaxVersion ) ;
}
if ( fFirstRun )
{
// Create new keyUser and set as default key
RandAddSeedPerfmon ( ) ;
std : : vector < unsigned char > newDefaultKey ;
if ( ! pwalletMain - > GetKeyFromPool ( newDefaultKey , false ) )
strErrors < < _ ( " Cannot initialize keypool " ) < < " \n " ;
pwalletMain - > SetDefaultKey ( newDefaultKey ) ;
if ( ! pwalletMain - > SetAddressBookName ( CBitcoinAddress ( pwalletMain - > vchDefaultKey ) , " " ) )
strErrors < < _ ( " Cannot write default address " ) < < " \n " ;
}
2012-03-11 22:57:44 +01:00
printf ( " %s " , strErrors . str ( ) . c_str ( ) ) ;
2011-05-14 20:10:21 +02:00
printf ( " wallet %15 " PRI64d " ms \n " , GetTimeMillis ( ) - nStart ) ;
2011-06-26 19:23:24 +02:00
RegisterWallet ( pwalletMain ) ;
2011-05-14 20:10:21 +02:00
CBlockIndex * pindexRescan = pindexBest ;
if ( GetBoolArg ( " -rescan " ) )
pindexRescan = pindexGenesisBlock ;
else
{
2011-06-26 19:23:24 +02:00
CWalletDB walletdb ( " wallet.dat " ) ;
2011-05-14 20:10:21 +02:00
CBlockLocator locator ;
if ( walletdb . ReadBestBlock ( locator ) )
pindexRescan = locator . GetBlockIndex ( ) ;
}
if ( pindexBest ! = pindexRescan )
{
2011-08-28 14:12:26 +02:00
InitMessage ( _ ( " Rescanning... " ) ) ;
2011-05-14 20:10:21 +02:00
printf ( " Rescanning last %i blocks (from block %i)... \n " , pindexBest - > nHeight - pindexRescan - > nHeight , pindexRescan - > nHeight ) ;
nStart = GetTimeMillis ( ) ;
2011-06-26 19:23:24 +02:00
pwalletMain - > ScanForWalletTransactions ( pindexRescan , true ) ;
2011-05-14 20:10:21 +02:00
printf ( " rescan %15 " PRI64d " ms \n " , GetTimeMillis ( ) - nStart ) ;
}
2011-08-28 14:12:26 +02:00
InitMessage ( _ ( " Done loading " ) ) ;
2011-05-14 20:10:21 +02:00
printf ( " Done loading \n " ) ;
2011-12-16 23:13:45 +01:00
//// debug print
printf ( " mapBlockIndex.size() = %d \n " , mapBlockIndex . size ( ) ) ;
printf ( " nBestHeight = %d \n " , nBestHeight ) ;
printf ( " setKeyPool.size() = %d \n " , pwalletMain - > setKeyPool . size ( ) ) ;
printf ( " mapWallet.size() = %d \n " , pwalletMain - > mapWallet . size ( ) ) ;
printf ( " mapAddressBook.size() = %d \n " , pwalletMain - > mapAddressBook . size ( ) ) ;
2011-05-14 20:10:21 +02:00
2011-12-16 23:13:45 +01:00
if ( ! strErrors . str ( ) . empty ( ) )
2011-05-14 20:10:21 +02:00
{
2012-03-25 20:47:33 +02:00
ThreadSafeMessageBox ( strErrors . str ( ) , _ ( " Bitcoin " ) , wxOK | wxICON_ERROR | wxMODAL ) ;
2011-05-14 20:10:21 +02:00
return false ;
}
// Add wallet transactions that aren't already in a block to mapTransactions
2011-06-26 19:23:24 +02:00
pwalletMain - > ReacceptWalletTransactions ( ) ;
2011-05-14 20:10:21 +02:00
2012-01-03 16:14:22 +01:00
// Note: Bitcoin-QT stores several settings in the wallet, so we want
// to load the wallet BEFORE parsing command-line arguments, so
// the command-line/bitcoin.conf settings override GUI setting.
2011-05-14 20:10:21 +02:00
//
// Parameters
//
if ( GetBoolArg ( " -printblockindex " ) | | GetBoolArg ( " -printblocktree " ) )
{
PrintBlockTree ( ) ;
return false ;
}
2011-06-12 12:27:01 +02:00
if ( mapArgs . count ( " -timeout " ) )
{
int nNewTimeout = GetArg ( " -timeout " , 5000 ) ;
if ( nNewTimeout > 0 & & nNewTimeout < 600000 )
nConnectTimeout = nNewTimeout ;
}
2011-05-14 20:10:21 +02:00
if ( mapArgs . count ( " -printblock " ) )
{
string strMatch = mapArgs [ " -printblock " ] ;
int nFound = 0 ;
for ( map < uint256 , CBlockIndex * > : : iterator mi = mapBlockIndex . begin ( ) ; mi ! = mapBlockIndex . end ( ) ; + + mi )
{
uint256 hash = ( * mi ) . first ;
if ( strncmp ( hash . ToString ( ) . c_str ( ) , strMatch . c_str ( ) , strMatch . size ( ) ) = = 0 )
{
CBlockIndex * pindex = ( * mi ) . second ;
CBlock block ;
block . ReadFromDisk ( pindex ) ;
block . BuildMerkleTree ( ) ;
block . print ( ) ;
printf ( " \n " ) ;
nFound + + ;
}
}
if ( nFound = = 0 )
printf ( " No blocks matching %s were found \n " , strMatch . c_str ( ) ) ;
return false ;
}
if ( mapArgs . count ( " -proxy " ) )
{
fUseProxy = true ;
2012-01-17 02:12:35 +01:00
addrProxy = CService ( mapArgs [ " -proxy " ] , 9050 ) ;
2011-05-14 20:10:21 +02:00
if ( ! addrProxy . IsValid ( ) )
{
2012-03-25 20:47:33 +02:00
ThreadSafeMessageBox ( _ ( " Invalid -proxy address " ) , _ ( " Bitcoin " ) , wxOK | wxMODAL ) ;
2011-05-14 20:10:21 +02:00
return false ;
}
}
2012-05-04 16:55:19 +02:00
if ( mapArgs . count ( " -noproxy " ) )
{
BOOST_FOREACH ( std : : string snet , mapMultiArgs [ " -noproxy " ] ) {
enum Network net = ParseNetwork ( snet ) ;
if ( net = = NET_UNROUTABLE ) {
ThreadSafeMessageBox ( _ ( " Unknown network specified in -noproxy " ) , _ ( " Bitcoin " ) , wxOK | wxMODAL ) ;
return false ;
}
SetNoProxy ( net ) ;
}
}
2012-02-19 20:44:35 +01:00
if ( mapArgs . count ( " -connect " ) )
SoftSetBoolArg ( " -dnsseed " , false ) ;
2012-01-03 23:33:31 +01:00
bool fTor = ( fUseProxy & & addrProxy . GetPort ( ) = = 9050 ) ;
2012-01-03 16:14:22 +01:00
if ( fTor )
{
2012-02-06 21:48:00 +01:00
// Use SoftSetBoolArg here so user can override any of these if they wish.
2012-01-03 16:14:22 +01:00
// Note: the GetBoolArg() calls for all of these must happen later.
2012-02-06 20:35:57 +01:00
SoftSetBoolArg ( " -listen " , false ) ;
SoftSetBoolArg ( " -irc " , false ) ;
2012-04-19 17:38:03 +02:00
SoftSetBoolArg ( " -proxydns " , true ) ;
2012-02-06 21:48:00 +01:00
SoftSetBoolArg ( " -upnp " , false ) ;
2012-02-19 20:44:35 +01:00
SoftSetBoolArg ( " -discover " , false ) ;
2012-01-03 16:14:22 +01:00
}
2012-05-04 16:46:22 +02:00
if ( mapArgs . count ( " -blocknet " ) ) {
BOOST_FOREACH ( std : : string snet , mapMultiArgs [ " -blocknet " ] ) {
enum Network net = ParseNetwork ( snet ) ;
if ( net = = NET_UNROUTABLE ) {
ThreadSafeMessageBox ( _ ( " Unknown network specified in -blocknet " ) , _ ( " Bitcoin " ) , wxOK | wxMODAL ) ;
return false ;
}
SetLimited ( net ) ;
}
}
2012-04-19 17:38:03 +02:00
fNameLookup = GetBoolArg ( " -dns " ) ;
fProxyNameLookup = GetBoolArg ( " -proxydns " ) ;
if ( fProxyNameLookup )
fNameLookup = true ;
2012-02-06 20:35:57 +01:00
fNoListen = ! GetBoolArg ( " -listen " , true ) ;
2012-04-19 17:38:03 +02:00
nSocksVersion = GetArg ( " -socks " , 5 ) ;
2012-01-03 16:14:22 +01:00
2012-04-24 02:15:00 +02:00
BOOST_FOREACH ( string strDest , mapMultiArgs [ " -seednode " ] )
AddOneShot ( strDest ) ;
2012-01-03 16:14:22 +01:00
2012-03-21 20:45:58 +01:00
// Continue to put "/P2SH/" in the coinbase to monitor
// BIP16 support.
// This can be removed eventually...
const char * pszP2SH = " /P2SH/ " ;
COINBASE_FLAGS < < std : : vector < unsigned char > ( pszP2SH , pszP2SH + strlen ( pszP2SH ) ) ;
2012-02-06 21:48:00 +01:00
2012-01-03 16:14:22 +01:00
if ( ! fNoListen )
{
std : : string strError ;
if ( ! BindListenPort ( strError ) )
{
2012-03-25 20:47:33 +02:00
ThreadSafeMessageBox ( strError , _ ( " Bitcoin " ) , wxOK | wxMODAL ) ;
2012-01-03 16:14:22 +01:00
return false ;
}
}
2012-02-19 20:44:35 +01:00
if ( mapArgs . count ( " -externalip " ) )
2011-05-14 20:10:21 +02:00
{
2012-02-19 20:44:35 +01:00
BOOST_FOREACH ( string strAddr , mapMultiArgs [ " -externalip " ] )
AddLocal ( CNetAddr ( strAddr , fNameLookup ) , LOCAL_MANUAL ) ;
2011-05-14 20:10:21 +02:00
}
if ( mapArgs . count ( " -paytxfee " ) )
{
if ( ! ParseMoney ( mapArgs [ " -paytxfee " ] , nTransactionFee ) )
{
2012-03-25 20:47:33 +02:00
ThreadSafeMessageBox ( _ ( " Invalid amount for -paytxfee=<amount> " ) , _ ( " Bitcoin " ) , wxOK | wxMODAL ) ;
2011-05-14 20:10:21 +02:00
return false ;
}
if ( nTransactionFee > 0.25 * COIN )
2012-03-25 20:47:33 +02:00
ThreadSafeMessageBox ( _ ( " Warning: -paytxfee is set very high. This is the transaction fee you will pay if you send a transaction. " ) , _ ( " Bitcoin " ) , wxOK | wxICON_EXCLAMATION | wxMODAL ) ;
2011-05-14 20:10:21 +02:00
}
//
2011-09-26 16:04:04 +02:00
// Start the node
2011-05-14 20:10:21 +02:00
//
if ( ! CheckDiskSpace ( ) )
return false ;
RandAddSeedPerfmon ( ) ;
if ( ! CreateThread ( StartNode , NULL ) )
2012-03-25 20:47:33 +02:00
ThreadSafeMessageBox ( _ ( " Error: CreateThread(StartNode) failed " ) , _ ( " Bitcoin " ) , wxOK | wxMODAL ) ;
2011-05-14 20:10:21 +02:00
if ( fServer )
CreateThread ( ThreadRPCServer , NULL ) ;
2011-12-20 00:49:07 +01:00
# ifdef QT_GUI
2012-05-01 01:59:54 +02:00
if ( GetStartOnSystemStartup ( ) )
SetStartOnSystemStartup ( true ) ; // Remove startup links
2011-12-20 00:49:07 +01:00
# endif
2011-09-26 16:04:04 +02:00
# if !defined(QT_GUI)
2011-05-14 20:10:21 +02:00
while ( 1 )
Sleep ( 5000 ) ;
# endif
return true ;
}
2011-12-20 00:49:07 +01:00