Do not allow users to get keys from keypool without reserving them
[bitcoinplatinum.git] / src / util.cpp
blobb76c173f90602e56d5457d1c1e9467c662df0663
1 // Copyright (c) 2009-2010 Satoshi Nakamoto
2 // Copyright (c) 2009-2016 The Bitcoin Core developers
3 // Distributed under the MIT software license, see the accompanying
4 // file COPYING or http://www.opensource.org/licenses/mit-license.php.
6 #if defined(HAVE_CONFIG_H)
7 #include "config/bitcoin-config.h"
8 #endif
10 #include "util.h"
12 #include "chainparamsbase.h"
13 #include "fs.h"
14 #include "random.h"
15 #include "serialize.h"
16 #include "utilstrencodings.h"
17 #include "utiltime.h"
19 #include <stdarg.h>
21 #if (defined(__FreeBSD__) || defined(__OpenBSD__) || defined(__DragonFly__))
22 #include <pthread.h>
23 #include <pthread_np.h>
24 #endif
26 #ifndef WIN32
27 // for posix_fallocate
28 #ifdef __linux__
30 #ifdef _POSIX_C_SOURCE
31 #undef _POSIX_C_SOURCE
32 #endif
34 #define _POSIX_C_SOURCE 200112L
36 #endif // __linux__
38 #include <algorithm>
39 #include <fcntl.h>
40 #include <sys/resource.h>
41 #include <sys/stat.h>
43 #else
45 #ifdef _MSC_VER
46 #pragma warning(disable:4786)
47 #pragma warning(disable:4804)
48 #pragma warning(disable:4805)
49 #pragma warning(disable:4717)
50 #endif
52 #ifdef _WIN32_WINNT
53 #undef _WIN32_WINNT
54 #endif
55 #define _WIN32_WINNT 0x0501
57 #ifdef _WIN32_IE
58 #undef _WIN32_IE
59 #endif
60 #define _WIN32_IE 0x0501
62 #define WIN32_LEAN_AND_MEAN 1
63 #ifndef NOMINMAX
64 #define NOMINMAX
65 #endif
67 #include <io.h> /* for _commit */
68 #include <shlobj.h>
69 #endif
71 #ifdef HAVE_SYS_PRCTL_H
72 #include <sys/prctl.h>
73 #endif
75 #ifdef HAVE_MALLOPT_ARENA_MAX
76 #include <malloc.h>
77 #endif
79 #include <boost/algorithm/string/case_conv.hpp> // for to_lower()
80 #include <boost/algorithm/string/predicate.hpp> // for startswith() and endswith()
81 #include <boost/program_options/detail/config_file.hpp>
82 #include <boost/thread.hpp>
83 #include <openssl/crypto.h>
84 #include <openssl/rand.h>
85 #include <openssl/conf.h>
87 // Application startup time (used for uptime calculation)
88 const int64_t nStartupTime = GetTime();
90 const char * const BITCOIN_CONF_FILENAME = "bitcoin.conf";
91 const char * const BITCOIN_PID_FILENAME = "bitcoind.pid";
93 ArgsManager gArgs;
94 bool fPrintToConsole = false;
95 bool fPrintToDebugLog = true;
97 bool fLogTimestamps = DEFAULT_LOGTIMESTAMPS;
98 bool fLogTimeMicros = DEFAULT_LOGTIMEMICROS;
99 bool fLogIPs = DEFAULT_LOGIPS;
100 std::atomic<bool> fReopenDebugLog(false);
101 CTranslationInterface translationInterface;
103 /** Log categories bitfield. */
104 std::atomic<uint32_t> logCategories(0);
106 /** Init OpenSSL library multithreading support */
107 static std::unique_ptr<CCriticalSection[]> ppmutexOpenSSL;
108 void locking_callback(int mode, int i, const char* file, int line) NO_THREAD_SAFETY_ANALYSIS
110 if (mode & CRYPTO_LOCK) {
111 ENTER_CRITICAL_SECTION(ppmutexOpenSSL[i]);
112 } else {
113 LEAVE_CRITICAL_SECTION(ppmutexOpenSSL[i]);
117 // Singleton for wrapping OpenSSL setup/teardown.
118 class CInit
120 public:
121 CInit()
123 // Init OpenSSL library multithreading support
124 ppmutexOpenSSL.reset(new CCriticalSection[CRYPTO_num_locks()]);
125 CRYPTO_set_locking_callback(locking_callback);
127 // OpenSSL can optionally load a config file which lists optional loadable modules and engines.
128 // We don't use them so we don't require the config. However some of our libs may call functions
129 // which attempt to load the config file, possibly resulting in an exit() or crash if it is missing
130 // or corrupt. Explicitly tell OpenSSL not to try to load the file. The result for our libs will be
131 // that the config appears to have been loaded and there are no modules/engines available.
132 OPENSSL_no_config();
134 #ifdef WIN32
135 // Seed OpenSSL PRNG with current contents of the screen
136 RAND_screen();
137 #endif
139 // Seed OpenSSL PRNG with performance counter
140 RandAddSeed();
142 ~CInit()
144 // Securely erase the memory used by the PRNG
145 RAND_cleanup();
146 // Shutdown OpenSSL library multithreading support
147 CRYPTO_set_locking_callback(NULL);
148 // Clear the set of locks now to maintain symmetry with the constructor.
149 ppmutexOpenSSL.reset();
152 instance_of_cinit;
155 * LogPrintf() has been broken a couple of times now
156 * by well-meaning people adding mutexes in the most straightforward way.
157 * It breaks because it may be called by global destructors during shutdown.
158 * Since the order of destruction of static/global objects is undefined,
159 * defining a mutex as a global object doesn't work (the mutex gets
160 * destroyed, and then some later destructor calls OutputDebugStringF,
161 * maybe indirectly, and you get a core dump at shutdown trying to lock
162 * the mutex).
165 static boost::once_flag debugPrintInitFlag = BOOST_ONCE_INIT;
168 * We use boost::call_once() to make sure mutexDebugLog and
169 * vMsgsBeforeOpenLog are initialized in a thread-safe manner.
171 * NOTE: fileout, mutexDebugLog and sometimes vMsgsBeforeOpenLog
172 * are leaked on exit. This is ugly, but will be cleaned up by
173 * the OS/libc. When the shutdown sequence is fully audited and
174 * tested, explicit destruction of these objects can be implemented.
176 static FILE* fileout = NULL;
177 static boost::mutex* mutexDebugLog = NULL;
178 static std::list<std::string>* vMsgsBeforeOpenLog;
180 static int FileWriteStr(const std::string &str, FILE *fp)
182 return fwrite(str.data(), 1, str.size(), fp);
185 static void DebugPrintInit()
187 assert(mutexDebugLog == NULL);
188 mutexDebugLog = new boost::mutex();
189 vMsgsBeforeOpenLog = new std::list<std::string>;
192 void OpenDebugLog()
194 boost::call_once(&DebugPrintInit, debugPrintInitFlag);
195 boost::mutex::scoped_lock scoped_lock(*mutexDebugLog);
197 assert(fileout == NULL);
198 assert(vMsgsBeforeOpenLog);
199 fs::path pathDebug = GetDataDir() / "debug.log";
200 fileout = fsbridge::fopen(pathDebug, "a");
201 if (fileout) {
202 setbuf(fileout, NULL); // unbuffered
203 // dump buffered messages from before we opened the log
204 while (!vMsgsBeforeOpenLog->empty()) {
205 FileWriteStr(vMsgsBeforeOpenLog->front(), fileout);
206 vMsgsBeforeOpenLog->pop_front();
210 delete vMsgsBeforeOpenLog;
211 vMsgsBeforeOpenLog = NULL;
214 struct CLogCategoryDesc
216 uint32_t flag;
217 std::string category;
220 const CLogCategoryDesc LogCategories[] =
222 {BCLog::NONE, "0"},
223 {BCLog::NET, "net"},
224 {BCLog::TOR, "tor"},
225 {BCLog::MEMPOOL, "mempool"},
226 {BCLog::HTTP, "http"},
227 {BCLog::BENCH, "bench"},
228 {BCLog::ZMQ, "zmq"},
229 {BCLog::DB, "db"},
230 {BCLog::RPC, "rpc"},
231 {BCLog::ESTIMATEFEE, "estimatefee"},
232 {BCLog::ADDRMAN, "addrman"},
233 {BCLog::SELECTCOINS, "selectcoins"},
234 {BCLog::REINDEX, "reindex"},
235 {BCLog::CMPCTBLOCK, "cmpctblock"},
236 {BCLog::RAND, "rand"},
237 {BCLog::PRUNE, "prune"},
238 {BCLog::PROXY, "proxy"},
239 {BCLog::MEMPOOLREJ, "mempoolrej"},
240 {BCLog::LIBEVENT, "libevent"},
241 {BCLog::COINDB, "coindb"},
242 {BCLog::QT, "qt"},
243 {BCLog::LEVELDB, "leveldb"},
244 {BCLog::ALL, "1"},
245 {BCLog::ALL, "all"},
248 bool GetLogCategory(uint32_t *f, const std::string *str)
250 if (f && str) {
251 if (*str == "") {
252 *f = BCLog::ALL;
253 return true;
255 for (unsigned int i = 0; i < ARRAYLEN(LogCategories); i++) {
256 if (LogCategories[i].category == *str) {
257 *f = LogCategories[i].flag;
258 return true;
262 return false;
265 std::string ListLogCategories()
267 std::string ret;
268 int outcount = 0;
269 for (unsigned int i = 0; i < ARRAYLEN(LogCategories); i++) {
270 // Omit the special cases.
271 if (LogCategories[i].flag != BCLog::NONE && LogCategories[i].flag != BCLog::ALL) {
272 if (outcount != 0) ret += ", ";
273 ret += LogCategories[i].category;
274 outcount++;
277 return ret;
280 std::vector<CLogCategoryActive> ListActiveLogCategories()
282 std::vector<CLogCategoryActive> ret;
283 for (unsigned int i = 0; i < ARRAYLEN(LogCategories); i++) {
284 // Omit the special cases.
285 if (LogCategories[i].flag != BCLog::NONE && LogCategories[i].flag != BCLog::ALL) {
286 CLogCategoryActive catActive;
287 catActive.category = LogCategories[i].category;
288 catActive.active = LogAcceptCategory(LogCategories[i].flag);
289 ret.push_back(catActive);
292 return ret;
296 * fStartedNewLine is a state variable held by the calling context that will
297 * suppress printing of the timestamp when multiple calls are made that don't
298 * end in a newline. Initialize it to true, and hold it, in the calling context.
300 static std::string LogTimestampStr(const std::string &str, std::atomic_bool *fStartedNewLine)
302 std::string strStamped;
304 if (!fLogTimestamps)
305 return str;
307 if (*fStartedNewLine) {
308 int64_t nTimeMicros = GetTimeMicros();
309 strStamped = DateTimeStrFormat("%Y-%m-%d %H:%M:%S", nTimeMicros/1000000);
310 if (fLogTimeMicros)
311 strStamped += strprintf(".%06d", nTimeMicros%1000000);
312 int64_t mocktime = GetMockTime();
313 if (mocktime) {
314 strStamped += " (mocktime: " + DateTimeStrFormat("%Y-%m-%d %H:%M:%S", mocktime) + ")";
316 strStamped += ' ' + str;
317 } else
318 strStamped = str;
320 if (!str.empty() && str[str.size()-1] == '\n')
321 *fStartedNewLine = true;
322 else
323 *fStartedNewLine = false;
325 return strStamped;
328 int LogPrintStr(const std::string &str)
330 int ret = 0; // Returns total number of characters written
331 static std::atomic_bool fStartedNewLine(true);
333 std::string strTimestamped = LogTimestampStr(str, &fStartedNewLine);
335 if (fPrintToConsole)
337 // print to console
338 ret = fwrite(strTimestamped.data(), 1, strTimestamped.size(), stdout);
339 fflush(stdout);
341 else if (fPrintToDebugLog)
343 boost::call_once(&DebugPrintInit, debugPrintInitFlag);
344 boost::mutex::scoped_lock scoped_lock(*mutexDebugLog);
346 // buffer if we haven't opened the log yet
347 if (fileout == NULL) {
348 assert(vMsgsBeforeOpenLog);
349 ret = strTimestamped.length();
350 vMsgsBeforeOpenLog->push_back(strTimestamped);
352 else
354 // reopen the log file, if requested
355 if (fReopenDebugLog) {
356 fReopenDebugLog = false;
357 fs::path pathDebug = GetDataDir() / "debug.log";
358 if (fsbridge::freopen(pathDebug,"a",fileout) != NULL)
359 setbuf(fileout, NULL); // unbuffered
362 ret = FileWriteStr(strTimestamped, fileout);
365 return ret;
368 /** Interpret string as boolean, for argument parsing */
369 static bool InterpretBool(const std::string& strValue)
371 if (strValue.empty())
372 return true;
373 return (atoi(strValue) != 0);
376 /** Turn -noX into -X=0 */
377 static void InterpretNegativeSetting(std::string& strKey, std::string& strValue)
379 if (strKey.length()>3 && strKey[0]=='-' && strKey[1]=='n' && strKey[2]=='o')
381 strKey = "-" + strKey.substr(3);
382 strValue = InterpretBool(strValue) ? "0" : "1";
386 void ArgsManager::ParseParameters(int argc, const char* const argv[])
388 LOCK(cs_args);
389 mapArgs.clear();
390 mapMultiArgs.clear();
392 for (int i = 1; i < argc; i++)
394 std::string str(argv[i]);
395 std::string strValue;
396 size_t is_index = str.find('=');
397 if (is_index != std::string::npos)
399 strValue = str.substr(is_index+1);
400 str = str.substr(0, is_index);
402 #ifdef WIN32
403 boost::to_lower(str);
404 if (boost::algorithm::starts_with(str, "/"))
405 str = "-" + str.substr(1);
406 #endif
408 if (str[0] != '-')
409 break;
411 // Interpret --foo as -foo.
412 // If both --foo and -foo are set, the last takes effect.
413 if (str.length() > 1 && str[1] == '-')
414 str = str.substr(1);
415 InterpretNegativeSetting(str, strValue);
417 mapArgs[str] = strValue;
418 mapMultiArgs[str].push_back(strValue);
422 std::vector<std::string> ArgsManager::GetArgs(const std::string& strArg)
424 LOCK(cs_args);
425 if (IsArgSet(strArg))
426 return mapMultiArgs.at(strArg);
427 return {};
430 bool ArgsManager::IsArgSet(const std::string& strArg)
432 LOCK(cs_args);
433 return mapArgs.count(strArg);
436 std::string ArgsManager::GetArg(const std::string& strArg, const std::string& strDefault)
438 LOCK(cs_args);
439 if (mapArgs.count(strArg))
440 return mapArgs[strArg];
441 return strDefault;
444 int64_t ArgsManager::GetArg(const std::string& strArg, int64_t nDefault)
446 LOCK(cs_args);
447 if (mapArgs.count(strArg))
448 return atoi64(mapArgs[strArg]);
449 return nDefault;
452 bool ArgsManager::GetBoolArg(const std::string& strArg, bool fDefault)
454 LOCK(cs_args);
455 if (mapArgs.count(strArg))
456 return InterpretBool(mapArgs[strArg]);
457 return fDefault;
460 bool ArgsManager::SoftSetArg(const std::string& strArg, const std::string& strValue)
462 LOCK(cs_args);
463 if (mapArgs.count(strArg))
464 return false;
465 ForceSetArg(strArg, strValue);
466 return true;
469 bool ArgsManager::SoftSetBoolArg(const std::string& strArg, bool fValue)
471 if (fValue)
472 return SoftSetArg(strArg, std::string("1"));
473 else
474 return SoftSetArg(strArg, std::string("0"));
477 void ArgsManager::ForceSetArg(const std::string& strArg, const std::string& strValue)
479 LOCK(cs_args);
480 mapArgs[strArg] = strValue;
481 mapMultiArgs[strArg].clear();
482 mapMultiArgs[strArg].push_back(strValue);
487 static const int screenWidth = 79;
488 static const int optIndent = 2;
489 static const int msgIndent = 7;
491 std::string HelpMessageGroup(const std::string &message) {
492 return std::string(message) + std::string("\n\n");
495 std::string HelpMessageOpt(const std::string &option, const std::string &message) {
496 return std::string(optIndent,' ') + std::string(option) +
497 std::string("\n") + std::string(msgIndent,' ') +
498 FormatParagraph(message, screenWidth - msgIndent, msgIndent) +
499 std::string("\n\n");
502 static std::string FormatException(const std::exception* pex, const char* pszThread)
504 #ifdef WIN32
505 char pszModule[MAX_PATH] = "";
506 GetModuleFileNameA(NULL, pszModule, sizeof(pszModule));
507 #else
508 const char* pszModule = "bitcoin";
509 #endif
510 if (pex)
511 return strprintf(
512 "EXCEPTION: %s \n%s \n%s in %s \n", typeid(*pex).name(), pex->what(), pszModule, pszThread);
513 else
514 return strprintf(
515 "UNKNOWN EXCEPTION \n%s in %s \n", pszModule, pszThread);
518 void PrintExceptionContinue(const std::exception* pex, const char* pszThread)
520 std::string message = FormatException(pex, pszThread);
521 LogPrintf("\n\n************************\n%s\n", message);
522 fprintf(stderr, "\n\n************************\n%s\n", message.c_str());
525 fs::path GetDefaultDataDir()
527 // Windows < Vista: C:\Documents and Settings\Username\Application Data\Bitcoin
528 // Windows >= Vista: C:\Users\Username\AppData\Roaming\Bitcoin
529 // Mac: ~/Library/Application Support/Bitcoin
530 // Unix: ~/.bitcoin
531 #ifdef WIN32
532 // Windows
533 return GetSpecialFolderPath(CSIDL_APPDATA) / "Bitcoin";
534 #else
535 fs::path pathRet;
536 char* pszHome = getenv("HOME");
537 if (pszHome == NULL || strlen(pszHome) == 0)
538 pathRet = fs::path("/");
539 else
540 pathRet = fs::path(pszHome);
541 #ifdef MAC_OSX
542 // Mac
543 return pathRet / "Library/Application Support/Bitcoin";
544 #else
545 // Unix
546 return pathRet / ".bitcoin";
547 #endif
548 #endif
551 static fs::path pathCached;
552 static fs::path pathCachedNetSpecific;
553 static CCriticalSection csPathCached;
555 const fs::path &GetDataDir(bool fNetSpecific)
558 LOCK(csPathCached);
560 fs::path &path = fNetSpecific ? pathCachedNetSpecific : pathCached;
562 // This can be called during exceptions by LogPrintf(), so we cache the
563 // value so we don't have to do memory allocations after that.
564 if (!path.empty())
565 return path;
567 if (IsArgSet("-datadir")) {
568 path = fs::system_complete(GetArg("-datadir", ""));
569 if (!fs::is_directory(path)) {
570 path = "";
571 return path;
573 } else {
574 path = GetDefaultDataDir();
576 if (fNetSpecific)
577 path /= BaseParams().DataDir();
579 fs::create_directories(path);
581 return path;
584 void ClearDatadirCache()
586 LOCK(csPathCached);
588 pathCached = fs::path();
589 pathCachedNetSpecific = fs::path();
592 fs::path GetConfigFile(const std::string& confPath)
594 fs::path pathConfigFile(confPath);
595 if (!pathConfigFile.is_complete())
596 pathConfigFile = GetDataDir(false) / pathConfigFile;
598 return pathConfigFile;
601 void ArgsManager::ReadConfigFile(const std::string& confPath)
603 fs::ifstream streamConfig(GetConfigFile(confPath));
604 if (!streamConfig.good())
605 return; // No bitcoin.conf file is OK
608 LOCK(cs_args);
609 std::set<std::string> setOptions;
610 setOptions.insert("*");
612 for (boost::program_options::detail::config_file_iterator it(streamConfig, setOptions), end; it != end; ++it)
614 // Don't overwrite existing settings so command line settings override bitcoin.conf
615 std::string strKey = std::string("-") + it->string_key;
616 std::string strValue = it->value[0];
617 InterpretNegativeSetting(strKey, strValue);
618 if (mapArgs.count(strKey) == 0)
619 mapArgs[strKey] = strValue;
620 mapMultiArgs[strKey].push_back(strValue);
623 // If datadir is changed in .conf file:
624 ClearDatadirCache();
627 #ifndef WIN32
628 fs::path GetPidFile()
630 fs::path pathPidFile(GetArg("-pid", BITCOIN_PID_FILENAME));
631 if (!pathPidFile.is_complete()) pathPidFile = GetDataDir() / pathPidFile;
632 return pathPidFile;
635 void CreatePidFile(const fs::path &path, pid_t pid)
637 FILE* file = fsbridge::fopen(path, "w");
638 if (file)
640 fprintf(file, "%d\n", pid);
641 fclose(file);
644 #endif
646 bool RenameOver(fs::path src, fs::path dest)
648 #ifdef WIN32
649 return MoveFileExA(src.string().c_str(), dest.string().c_str(),
650 MOVEFILE_REPLACE_EXISTING) != 0;
651 #else
652 int rc = std::rename(src.string().c_str(), dest.string().c_str());
653 return (rc == 0);
654 #endif /* WIN32 */
658 * Ignores exceptions thrown by Boost's create_directories if the requested directory exists.
659 * Specifically handles case where path p exists, but it wasn't possible for the user to
660 * write to the parent directory.
662 bool TryCreateDirectories(const fs::path& p)
666 return fs::create_directories(p);
667 } catch (const fs::filesystem_error&) {
668 if (!fs::exists(p) || !fs::is_directory(p))
669 throw;
672 // create_directories didn't create the directory, it had to have existed already
673 return false;
676 void FileCommit(FILE *file)
678 fflush(file); // harmless if redundantly called
679 #ifdef WIN32
680 HANDLE hFile = (HANDLE)_get_osfhandle(_fileno(file));
681 FlushFileBuffers(hFile);
682 #else
683 #if defined(__linux__) || defined(__NetBSD__)
684 fdatasync(fileno(file));
685 #elif defined(__APPLE__) && defined(F_FULLFSYNC)
686 fcntl(fileno(file), F_FULLFSYNC, 0);
687 #else
688 fsync(fileno(file));
689 #endif
690 #endif
693 bool TruncateFile(FILE *file, unsigned int length) {
694 #if defined(WIN32)
695 return _chsize(_fileno(file), length) == 0;
696 #else
697 return ftruncate(fileno(file), length) == 0;
698 #endif
702 * this function tries to raise the file descriptor limit to the requested number.
703 * It returns the actual file descriptor limit (which may be more or less than nMinFD)
705 int RaiseFileDescriptorLimit(int nMinFD) {
706 #if defined(WIN32)
707 return 2048;
708 #else
709 struct rlimit limitFD;
710 if (getrlimit(RLIMIT_NOFILE, &limitFD) != -1) {
711 if (limitFD.rlim_cur < (rlim_t)nMinFD) {
712 limitFD.rlim_cur = nMinFD;
713 if (limitFD.rlim_cur > limitFD.rlim_max)
714 limitFD.rlim_cur = limitFD.rlim_max;
715 setrlimit(RLIMIT_NOFILE, &limitFD);
716 getrlimit(RLIMIT_NOFILE, &limitFD);
718 return limitFD.rlim_cur;
720 return nMinFD; // getrlimit failed, assume it's fine
721 #endif
725 * this function tries to make a particular range of a file allocated (corresponding to disk space)
726 * it is advisory, and the range specified in the arguments will never contain live data
728 void AllocateFileRange(FILE *file, unsigned int offset, unsigned int length) {
729 #if defined(WIN32)
730 // Windows-specific version
731 HANDLE hFile = (HANDLE)_get_osfhandle(_fileno(file));
732 LARGE_INTEGER nFileSize;
733 int64_t nEndPos = (int64_t)offset + length;
734 nFileSize.u.LowPart = nEndPos & 0xFFFFFFFF;
735 nFileSize.u.HighPart = nEndPos >> 32;
736 SetFilePointerEx(hFile, nFileSize, 0, FILE_BEGIN);
737 SetEndOfFile(hFile);
738 #elif defined(MAC_OSX)
739 // OSX specific version
740 fstore_t fst;
741 fst.fst_flags = F_ALLOCATECONTIG;
742 fst.fst_posmode = F_PEOFPOSMODE;
743 fst.fst_offset = 0;
744 fst.fst_length = (off_t)offset + length;
745 fst.fst_bytesalloc = 0;
746 if (fcntl(fileno(file), F_PREALLOCATE, &fst) == -1) {
747 fst.fst_flags = F_ALLOCATEALL;
748 fcntl(fileno(file), F_PREALLOCATE, &fst);
750 ftruncate(fileno(file), fst.fst_length);
751 #elif defined(__linux__)
752 // Version using posix_fallocate
753 off_t nEndPos = (off_t)offset + length;
754 posix_fallocate(fileno(file), 0, nEndPos);
755 #else
756 // Fallback version
757 // TODO: just write one byte per block
758 static const char buf[65536] = {};
759 fseek(file, offset, SEEK_SET);
760 while (length > 0) {
761 unsigned int now = 65536;
762 if (length < now)
763 now = length;
764 fwrite(buf, 1, now, file); // allowed to fail; this function is advisory anyway
765 length -= now;
767 #endif
770 void ShrinkDebugFile()
772 // Amount of debug.log to save at end when shrinking (must fit in memory)
773 constexpr size_t RECENT_DEBUG_HISTORY_SIZE = 10 * 1000000;
774 // Scroll debug.log if it's getting too big
775 fs::path pathLog = GetDataDir() / "debug.log";
776 FILE* file = fsbridge::fopen(pathLog, "r");
777 // If debug.log file is more than 10% bigger the RECENT_DEBUG_HISTORY_SIZE
778 // trim it down by saving only the last RECENT_DEBUG_HISTORY_SIZE bytes
779 if (file && fs::file_size(pathLog) > 11 * (RECENT_DEBUG_HISTORY_SIZE / 10))
781 // Restart the file with some of the end
782 std::vector<char> vch(RECENT_DEBUG_HISTORY_SIZE, 0);
783 fseek(file, -((long)vch.size()), SEEK_END);
784 int nBytes = fread(vch.data(), 1, vch.size(), file);
785 fclose(file);
787 file = fsbridge::fopen(pathLog, "w");
788 if (file)
790 fwrite(vch.data(), 1, nBytes, file);
791 fclose(file);
794 else if (file != NULL)
795 fclose(file);
798 #ifdef WIN32
799 fs::path GetSpecialFolderPath(int nFolder, bool fCreate)
801 char pszPath[MAX_PATH] = "";
803 if(SHGetSpecialFolderPathA(NULL, pszPath, nFolder, fCreate))
805 return fs::path(pszPath);
808 LogPrintf("SHGetSpecialFolderPathA() failed, could not obtain requested path.\n");
809 return fs::path("");
811 #endif
813 void runCommand(const std::string& strCommand)
815 int nErr = ::system(strCommand.c_str());
816 if (nErr)
817 LogPrintf("runCommand error: system(%s) returned %d\n", strCommand, nErr);
820 void RenameThread(const char* name)
822 #if defined(PR_SET_NAME)
823 // Only the first 15 characters are used (16 - NUL terminator)
824 ::prctl(PR_SET_NAME, name, 0, 0, 0);
825 #elif (defined(__FreeBSD__) || defined(__OpenBSD__) || defined(__DragonFly__))
826 pthread_set_name_np(pthread_self(), name);
828 #elif defined(MAC_OSX)
829 pthread_setname_np(name);
830 #else
831 // Prevent warnings for unused parameters...
832 (void)name;
833 #endif
836 void SetupEnvironment()
838 #ifdef HAVE_MALLOPT_ARENA_MAX
839 // glibc-specific: On 32-bit systems set the number of arenas to 1.
840 // By default, since glibc 2.10, the C library will create up to two heap
841 // arenas per core. This is known to cause excessive virtual address space
842 // usage in our usage. Work around it by setting the maximum number of
843 // arenas to 1.
844 if (sizeof(void*) == 4) {
845 mallopt(M_ARENA_MAX, 1);
847 #endif
848 // On most POSIX systems (e.g. Linux, but not BSD) the environment's locale
849 // may be invalid, in which case the "C" locale is used as fallback.
850 #if !defined(WIN32) && !defined(MAC_OSX) && !defined(__FreeBSD__) && !defined(__OpenBSD__)
851 try {
852 std::locale(""); // Raises a runtime error if current locale is invalid
853 } catch (const std::runtime_error&) {
854 setenv("LC_ALL", "C", 1);
856 #endif
857 // The path locale is lazy initialized and to avoid deinitialization errors
858 // in multithreading environments, it is set explicitly by the main thread.
859 // A dummy locale is used to extract the internal default locale, used by
860 // fs::path, which is then used to explicitly imbue the path.
861 std::locale loc = fs::path::imbue(std::locale::classic());
862 fs::path::imbue(loc);
865 bool SetupNetworking()
867 #ifdef WIN32
868 // Initialize Windows Sockets
869 WSADATA wsadata;
870 int ret = WSAStartup(MAKEWORD(2,2), &wsadata);
871 if (ret != NO_ERROR || LOBYTE(wsadata.wVersion ) != 2 || HIBYTE(wsadata.wVersion) != 2)
872 return false;
873 #endif
874 return true;
877 int GetNumCores()
879 #if BOOST_VERSION >= 105600
880 return boost::thread::physical_concurrency();
881 #else // Must fall back to hardware_concurrency, which unfortunately counts virtual cores
882 return boost::thread::hardware_concurrency();
883 #endif
886 std::string CopyrightHolders(const std::string& strPrefix)
888 std::string strCopyrightHolders = strPrefix + strprintf(_(COPYRIGHT_HOLDERS), _(COPYRIGHT_HOLDERS_SUBSTITUTION));
890 // Check for untranslated substitution to make sure Bitcoin Core copyright is not removed by accident
891 if (strprintf(COPYRIGHT_HOLDERS, COPYRIGHT_HOLDERS_SUBSTITUTION).find("Bitcoin Core") == std::string::npos) {
892 strCopyrightHolders += "\n" + strPrefix + "The Bitcoin Core developers";
894 return strCopyrightHolders;
897 // Obtain the application startup time (used for uptime calculation)
898 int64_t GetStartupTime()
900 return nStartupTime;