mirror of
https://github.com/bitcoin/bitcoin.git
synced 2025-12-18 12:01:02 +01:00
Use to simplify code determining the chainstate leveldb paths. New method is the now the only code that needs to figure out the storage path, so the path doesn't need to be constructed multiple places and backed out of leveldb.
393 lines
13 KiB
C++
393 lines
13 KiB
C++
// Copyright (c) 2012-present The Bitcoin Core developers
|
|
// Distributed under the MIT software license, see the accompanying
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
#include <dbwrapper.h>
|
|
|
|
#include <logging.h>
|
|
#include <random.h>
|
|
#include <serialize.h>
|
|
#include <span.h>
|
|
#include <streams.h>
|
|
#include <util/fs.h>
|
|
#include <util/fs_helpers.h>
|
|
#include <util/obfuscation.h>
|
|
#include <util/strencodings.h>
|
|
|
|
#include <algorithm>
|
|
#include <cassert>
|
|
#include <cstdarg>
|
|
#include <cstdint>
|
|
#include <cstdio>
|
|
#include <leveldb/cache.h>
|
|
#include <leveldb/db.h>
|
|
#include <leveldb/env.h>
|
|
#include <leveldb/filter_policy.h>
|
|
#include <leveldb/helpers/memenv/memenv.h>
|
|
#include <leveldb/iterator.h>
|
|
#include <leveldb/options.h>
|
|
#include <leveldb/slice.h>
|
|
#include <leveldb/status.h>
|
|
#include <leveldb/write_batch.h>
|
|
#include <memory>
|
|
#include <optional>
|
|
#include <utility>
|
|
|
|
static auto CharCast(const std::byte* data) { return reinterpret_cast<const char*>(data); }
|
|
|
|
bool DestroyDB(const std::string& path_str)
|
|
{
|
|
return leveldb::DestroyDB(path_str, {}).ok();
|
|
}
|
|
|
|
/** Handle database error by throwing dbwrapper_error exception.
|
|
*/
|
|
static void HandleError(const leveldb::Status& status)
|
|
{
|
|
if (status.ok())
|
|
return;
|
|
const std::string errmsg = "Fatal LevelDB error: " + status.ToString();
|
|
LogError("%s", errmsg);
|
|
LogInfo("You can use -debug=leveldb to get more complete diagnostic messages");
|
|
throw dbwrapper_error(errmsg);
|
|
}
|
|
|
|
class CBitcoinLevelDBLogger : public leveldb::Logger {
|
|
public:
|
|
// This code is adapted from posix_logger.h, which is why it is using vsprintf.
|
|
// Please do not do this in normal code
|
|
void Logv(const char * format, va_list ap) override {
|
|
if (!LogAcceptCategory(BCLog::LEVELDB, BCLog::Level::Debug)) {
|
|
return;
|
|
}
|
|
char buffer[500];
|
|
for (int iter = 0; iter < 2; iter++) {
|
|
char* base;
|
|
int bufsize;
|
|
if (iter == 0) {
|
|
bufsize = sizeof(buffer);
|
|
base = buffer;
|
|
}
|
|
else {
|
|
bufsize = 30000;
|
|
base = new char[bufsize];
|
|
}
|
|
char* p = base;
|
|
char* limit = base + bufsize;
|
|
|
|
// Print the message
|
|
if (p < limit) {
|
|
va_list backup_ap;
|
|
va_copy(backup_ap, ap);
|
|
// Do not use vsnprintf elsewhere in bitcoin source code, see above.
|
|
p += vsnprintf(p, limit - p, format, backup_ap);
|
|
va_end(backup_ap);
|
|
}
|
|
|
|
// Truncate to available space if necessary
|
|
if (p >= limit) {
|
|
if (iter == 0) {
|
|
continue; // Try again with larger buffer
|
|
}
|
|
else {
|
|
p = limit - 1;
|
|
}
|
|
}
|
|
|
|
// Add newline if necessary
|
|
if (p == base || p[-1] != '\n') {
|
|
*p++ = '\n';
|
|
}
|
|
|
|
assert(p <= limit);
|
|
base[std::min(bufsize - 1, (int)(p - base))] = '\0';
|
|
LogDebug(BCLog::LEVELDB, "%s\n", util::RemoveSuffixView(base, "\n"));
|
|
if (base != buffer) {
|
|
delete[] base;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
};
|
|
|
|
static void SetMaxOpenFiles(leveldb::Options *options) {
|
|
// On most platforms the default setting of max_open_files (which is 1000)
|
|
// is optimal. On Windows using a large file count is OK because the handles
|
|
// do not interfere with select() loops. On 64-bit Unix hosts this value is
|
|
// also OK, because up to that amount LevelDB will use an mmap
|
|
// implementation that does not use extra file descriptors (the fds are
|
|
// closed after being mmap'ed).
|
|
//
|
|
// Increasing the value beyond the default is dangerous because LevelDB will
|
|
// fall back to a non-mmap implementation when the file count is too large.
|
|
// On 32-bit Unix host we should decrease the value because the handles use
|
|
// up real fds, and we want to avoid fd exhaustion issues.
|
|
//
|
|
// See PR #12495 for further discussion.
|
|
|
|
int default_open_files = options->max_open_files;
|
|
#ifndef WIN32
|
|
if (sizeof(void*) < 8) {
|
|
options->max_open_files = 64;
|
|
}
|
|
#endif
|
|
LogDebug(BCLog::LEVELDB, "LevelDB using max_open_files=%d (default=%d)\n",
|
|
options->max_open_files, default_open_files);
|
|
}
|
|
|
|
static leveldb::Options GetOptions(size_t nCacheSize)
|
|
{
|
|
leveldb::Options options;
|
|
options.block_cache = leveldb::NewLRUCache(nCacheSize / 2);
|
|
options.write_buffer_size = nCacheSize / 4; // up to two write buffers may be held in memory simultaneously
|
|
options.filter_policy = leveldb::NewBloomFilterPolicy(10);
|
|
options.compression = leveldb::kNoCompression;
|
|
options.info_log = new CBitcoinLevelDBLogger();
|
|
if (leveldb::kMajorVersion > 1 || (leveldb::kMajorVersion == 1 && leveldb::kMinorVersion >= 16)) {
|
|
// LevelDB versions before 1.16 consider short writes to be corruption. Only trigger error
|
|
// on corruption in later versions.
|
|
options.paranoid_checks = true;
|
|
}
|
|
options.max_file_size = std::max(options.max_file_size, DBWRAPPER_MAX_FILE_SIZE);
|
|
SetMaxOpenFiles(&options);
|
|
return options;
|
|
}
|
|
|
|
struct CDBBatch::WriteBatchImpl {
|
|
leveldb::WriteBatch batch;
|
|
};
|
|
|
|
CDBBatch::CDBBatch(const CDBWrapper& _parent)
|
|
: parent{_parent},
|
|
m_impl_batch{std::make_unique<CDBBatch::WriteBatchImpl>()}
|
|
{
|
|
Clear();
|
|
};
|
|
|
|
CDBBatch::~CDBBatch() = default;
|
|
|
|
void CDBBatch::Clear()
|
|
{
|
|
m_impl_batch->batch.Clear();
|
|
}
|
|
|
|
void CDBBatch::WriteImpl(std::span<const std::byte> key, DataStream& ssValue)
|
|
{
|
|
leveldb::Slice slKey(CharCast(key.data()), key.size());
|
|
dbwrapper_private::GetObfuscation(parent)(ssValue);
|
|
leveldb::Slice slValue(CharCast(ssValue.data()), ssValue.size());
|
|
m_impl_batch->batch.Put(slKey, slValue);
|
|
}
|
|
|
|
void CDBBatch::EraseImpl(std::span<const std::byte> key)
|
|
{
|
|
leveldb::Slice slKey(CharCast(key.data()), key.size());
|
|
m_impl_batch->batch.Delete(slKey);
|
|
}
|
|
|
|
size_t CDBBatch::ApproximateSize() const
|
|
{
|
|
return m_impl_batch->batch.ApproximateSize();
|
|
}
|
|
|
|
struct LevelDBContext {
|
|
//! custom environment this database is using (may be nullptr in case of default environment)
|
|
leveldb::Env* penv;
|
|
|
|
//! database options used
|
|
leveldb::Options options;
|
|
|
|
//! options used when reading from the database
|
|
leveldb::ReadOptions readoptions;
|
|
|
|
//! options used when iterating over values of the database
|
|
leveldb::ReadOptions iteroptions;
|
|
|
|
//! options used when writing to the database
|
|
leveldb::WriteOptions writeoptions;
|
|
|
|
//! options used when sync writing to the database
|
|
leveldb::WriteOptions syncoptions;
|
|
|
|
//! the database itself
|
|
leveldb::DB* pdb;
|
|
};
|
|
|
|
CDBWrapper::CDBWrapper(const DBParams& params)
|
|
: m_db_context{std::make_unique<LevelDBContext>()}, m_name{fs::PathToString(params.path.stem())}
|
|
{
|
|
DBContext().penv = nullptr;
|
|
DBContext().readoptions.verify_checksums = true;
|
|
DBContext().iteroptions.verify_checksums = true;
|
|
DBContext().iteroptions.fill_cache = false;
|
|
DBContext().syncoptions.sync = true;
|
|
DBContext().options = GetOptions(params.cache_bytes);
|
|
DBContext().options.create_if_missing = true;
|
|
if (params.memory_only) {
|
|
DBContext().penv = leveldb::NewMemEnv(leveldb::Env::Default());
|
|
DBContext().options.env = DBContext().penv;
|
|
} else {
|
|
if (params.wipe_data) {
|
|
LogInfo("Wiping LevelDB in %s", fs::PathToString(params.path));
|
|
leveldb::Status result = leveldb::DestroyDB(fs::PathToString(params.path), DBContext().options);
|
|
HandleError(result);
|
|
}
|
|
TryCreateDirectories(params.path);
|
|
LogInfo("Opening LevelDB in %s", fs::PathToString(params.path));
|
|
}
|
|
// PathToString() return value is safe to pass to leveldb open function,
|
|
// because on POSIX leveldb passes the byte string directly to ::open(), and
|
|
// on Windows it converts from UTF-8 to UTF-16 before calling ::CreateFileW
|
|
// (see env_posix.cc and env_windows.cc).
|
|
leveldb::Status status = leveldb::DB::Open(DBContext().options, fs::PathToString(params.path), &DBContext().pdb);
|
|
HandleError(status);
|
|
LogInfo("Opened LevelDB successfully");
|
|
|
|
if (params.options.force_compact) {
|
|
LogInfo("Starting database compaction of %s", fs::PathToString(params.path));
|
|
DBContext().pdb->CompactRange(nullptr, nullptr);
|
|
LogInfo("Finished database compaction of %s", fs::PathToString(params.path));
|
|
}
|
|
|
|
if (!Read(OBFUSCATION_KEY, m_obfuscation) && params.obfuscate && IsEmpty()) {
|
|
// Generate and write the new obfuscation key.
|
|
const Obfuscation obfuscation{FastRandomContext{}.randbytes<Obfuscation::KEY_SIZE>()};
|
|
assert(!m_obfuscation); // Make sure the key is written without obfuscation.
|
|
Write(OBFUSCATION_KEY, obfuscation);
|
|
m_obfuscation = obfuscation;
|
|
LogInfo("Wrote new obfuscation key for %s: %s", fs::PathToString(params.path), m_obfuscation.HexKey());
|
|
}
|
|
LogInfo("Using obfuscation key for %s: %s", fs::PathToString(params.path), m_obfuscation.HexKey());
|
|
}
|
|
|
|
CDBWrapper::~CDBWrapper()
|
|
{
|
|
delete DBContext().pdb;
|
|
DBContext().pdb = nullptr;
|
|
delete DBContext().options.filter_policy;
|
|
DBContext().options.filter_policy = nullptr;
|
|
delete DBContext().options.info_log;
|
|
DBContext().options.info_log = nullptr;
|
|
delete DBContext().options.block_cache;
|
|
DBContext().options.block_cache = nullptr;
|
|
delete DBContext().penv;
|
|
DBContext().options.env = nullptr;
|
|
}
|
|
|
|
void CDBWrapper::WriteBatch(CDBBatch& batch, bool fSync)
|
|
{
|
|
const bool log_memory = LogAcceptCategory(BCLog::LEVELDB, BCLog::Level::Debug);
|
|
double mem_before = 0;
|
|
if (log_memory) {
|
|
mem_before = DynamicMemoryUsage() / 1024.0 / 1024;
|
|
}
|
|
leveldb::Status status = DBContext().pdb->Write(fSync ? DBContext().syncoptions : DBContext().writeoptions, &batch.m_impl_batch->batch);
|
|
HandleError(status);
|
|
if (log_memory) {
|
|
double mem_after = DynamicMemoryUsage() / 1024.0 / 1024;
|
|
LogDebug(BCLog::LEVELDB, "WriteBatch memory usage: db=%s, before=%.1fMiB, after=%.1fMiB\n",
|
|
m_name, mem_before, mem_after);
|
|
}
|
|
}
|
|
|
|
size_t CDBWrapper::DynamicMemoryUsage() const
|
|
{
|
|
std::string memory;
|
|
std::optional<size_t> parsed;
|
|
if (!DBContext().pdb->GetProperty("leveldb.approximate-memory-usage", &memory) || !(parsed = ToIntegral<size_t>(memory))) {
|
|
LogDebug(BCLog::LEVELDB, "Failed to get approximate-memory-usage property\n");
|
|
return 0;
|
|
}
|
|
return parsed.value();
|
|
}
|
|
|
|
std::optional<std::string> CDBWrapper::ReadImpl(std::span<const std::byte> key) const
|
|
{
|
|
leveldb::Slice slKey(CharCast(key.data()), key.size());
|
|
std::string strValue;
|
|
leveldb::Status status = DBContext().pdb->Get(DBContext().readoptions, slKey, &strValue);
|
|
if (!status.ok()) {
|
|
if (status.IsNotFound())
|
|
return std::nullopt;
|
|
LogError("LevelDB read failure: %s", status.ToString());
|
|
HandleError(status);
|
|
}
|
|
return strValue;
|
|
}
|
|
|
|
bool CDBWrapper::ExistsImpl(std::span<const std::byte> key) const
|
|
{
|
|
leveldb::Slice slKey(CharCast(key.data()), key.size());
|
|
|
|
std::string strValue;
|
|
leveldb::Status status = DBContext().pdb->Get(DBContext().readoptions, slKey, &strValue);
|
|
if (!status.ok()) {
|
|
if (status.IsNotFound())
|
|
return false;
|
|
LogError("LevelDB read failure: %s", status.ToString());
|
|
HandleError(status);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
size_t CDBWrapper::EstimateSizeImpl(std::span<const std::byte> key1, std::span<const std::byte> key2) const
|
|
{
|
|
leveldb::Slice slKey1(CharCast(key1.data()), key1.size());
|
|
leveldb::Slice slKey2(CharCast(key2.data()), key2.size());
|
|
uint64_t size = 0;
|
|
leveldb::Range range(slKey1, slKey2);
|
|
DBContext().pdb->GetApproximateSizes(&range, 1, &size);
|
|
return size;
|
|
}
|
|
|
|
bool CDBWrapper::IsEmpty()
|
|
{
|
|
std::unique_ptr<CDBIterator> it(NewIterator());
|
|
it->SeekToFirst();
|
|
return !(it->Valid());
|
|
}
|
|
|
|
struct CDBIterator::IteratorImpl {
|
|
const std::unique_ptr<leveldb::Iterator> iter;
|
|
|
|
explicit IteratorImpl(leveldb::Iterator* _iter) : iter{_iter} {}
|
|
};
|
|
|
|
CDBIterator::CDBIterator(const CDBWrapper& _parent, std::unique_ptr<IteratorImpl> _piter) : parent(_parent),
|
|
m_impl_iter(std::move(_piter)) {}
|
|
|
|
CDBIterator* CDBWrapper::NewIterator()
|
|
{
|
|
return new CDBIterator{*this, std::make_unique<CDBIterator::IteratorImpl>(DBContext().pdb->NewIterator(DBContext().iteroptions))};
|
|
}
|
|
|
|
void CDBIterator::SeekImpl(std::span<const std::byte> key)
|
|
{
|
|
leveldb::Slice slKey(CharCast(key.data()), key.size());
|
|
m_impl_iter->iter->Seek(slKey);
|
|
}
|
|
|
|
std::span<const std::byte> CDBIterator::GetKeyImpl() const
|
|
{
|
|
return MakeByteSpan(m_impl_iter->iter->key());
|
|
}
|
|
|
|
std::span<const std::byte> CDBIterator::GetValueImpl() const
|
|
{
|
|
return MakeByteSpan(m_impl_iter->iter->value());
|
|
}
|
|
|
|
CDBIterator::~CDBIterator() = default;
|
|
bool CDBIterator::Valid() const { return m_impl_iter->iter->Valid(); }
|
|
void CDBIterator::SeekToFirst() { m_impl_iter->iter->SeekToFirst(); }
|
|
void CDBIterator::Next() { m_impl_iter->iter->Next(); }
|
|
|
|
namespace dbwrapper_private {
|
|
|
|
const Obfuscation& GetObfuscation(const CDBWrapper& w)
|
|
{
|
|
return w.m_obfuscation;
|
|
}
|
|
|
|
} // namespace dbwrapper_private
|