2020-06-15 14:29:29 -04:00
|
|
|
// Copyright (c) 2009-2010 Satoshi Nakamoto
|
|
|
|
// Copyright (c) 2009-2020 The Bitcoin Core developers
|
|
|
|
// Distributed under the MIT software license, see the accompanying
|
|
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
|
|
|
|
#ifndef BITCOIN_WALLET_BDB_H
|
|
|
|
#define BITCOIN_WALLET_BDB_H
|
|
|
|
|
|
|
|
#include <clientversion.h>
|
|
|
|
#include <fs.h>
|
|
|
|
#include <serialize.h>
|
|
|
|
#include <streams.h>
|
|
|
|
#include <util/system.h>
|
|
|
|
#include <wallet/db.h>
|
|
|
|
|
|
|
|
#include <atomic>
|
|
|
|
#include <map>
|
|
|
|
#include <memory>
|
|
|
|
#include <string>
|
|
|
|
#include <unordered_map>
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
#if defined(__GNUC__) && !defined(__clang__)
|
|
|
|
#pragma GCC diagnostic push
|
|
|
|
#pragma GCC diagnostic ignored "-Wsuggest-override"
|
|
|
|
#endif
|
|
|
|
#include <db_cxx.h>
|
|
|
|
#if defined(__GNUC__) && !defined(__clang__)
|
|
|
|
#pragma GCC diagnostic pop
|
|
|
|
#endif
|
|
|
|
|
|
|
|
struct bilingual_str;
|
|
|
|
|
|
|
|
static const unsigned int DEFAULT_WALLET_DBLOGSIZE = 100;
|
|
|
|
static const bool DEFAULT_WALLET_PRIVDB = true;
|
|
|
|
|
|
|
|
struct WalletDatabaseFileId {
|
|
|
|
u_int8_t value[DB_FILE_ID_LEN];
|
|
|
|
bool operator==(const WalletDatabaseFileId& rhs) const;
|
|
|
|
};
|
|
|
|
|
|
|
|
class BerkeleyDatabase;
|
|
|
|
|
|
|
|
class BerkeleyEnvironment
|
|
|
|
{
|
|
|
|
private:
|
|
|
|
bool fDbEnvInit;
|
|
|
|
bool fMockDb;
|
|
|
|
// Don't change into fs::path, as that can result in
|
|
|
|
// shutdown problems/crashes caused by a static initialized internal pointer.
|
|
|
|
std::string strPath;
|
|
|
|
|
|
|
|
public:
|
|
|
|
std::unique_ptr<DbEnv> dbenv;
|
|
|
|
std::map<std::string, std::reference_wrapper<BerkeleyDatabase>> m_databases;
|
|
|
|
std::unordered_map<std::string, WalletDatabaseFileId> m_fileids;
|
|
|
|
std::condition_variable_any m_db_in_use;
|
|
|
|
|
|
|
|
BerkeleyEnvironment(const fs::path& env_directory);
|
|
|
|
BerkeleyEnvironment();
|
|
|
|
~BerkeleyEnvironment();
|
|
|
|
void Reset();
|
|
|
|
|
|
|
|
bool IsMock() const { return fMockDb; }
|
|
|
|
bool IsInitialized() const { return fDbEnvInit; }
|
|
|
|
bool IsDatabaseLoaded(const std::string& db_filename) const { return m_databases.find(db_filename) != m_databases.end(); }
|
|
|
|
fs::path Directory() const { return strPath; }
|
|
|
|
|
2020-05-07 14:30:04 -04:00
|
|
|
bool Open(bilingual_str& error);
|
2020-06-15 14:29:29 -04:00
|
|
|
void Close();
|
|
|
|
void Flush(bool fShutdown);
|
|
|
|
void CheckpointLSN(const std::string& strFile);
|
|
|
|
|
|
|
|
void CloseDb(const std::string& strFile);
|
|
|
|
void ReloadDbEnv();
|
|
|
|
|
|
|
|
DbTxn* TxnBegin(int flags = DB_TXN_WRITE_NOSYNC)
|
|
|
|
{
|
|
|
|
DbTxn* ptxn = nullptr;
|
|
|
|
int ret = dbenv->txn_begin(nullptr, &ptxn, flags);
|
|
|
|
if (!ptxn || ret != 0)
|
|
|
|
return nullptr;
|
|
|
|
return ptxn;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/** Get BerkeleyEnvironment and database filename given a wallet path. */
|
|
|
|
std::shared_ptr<BerkeleyEnvironment> GetWalletEnv(const fs::path& wallet_path, std::string& database_filename);
|
|
|
|
|
2020-07-11 07:56:13 -04:00
|
|
|
/** Return whether a BDB wallet database is currently loaded. */
|
2020-06-15 14:29:29 -04:00
|
|
|
bool IsBDBWalletLoaded(const fs::path& wallet_path);
|
|
|
|
|
2020-08-04 16:40:31 -04:00
|
|
|
/** Check format of database file */
|
|
|
|
bool IsBerkeleyBtree(const fs::path& path);
|
|
|
|
|
2020-06-15 16:54:58 -04:00
|
|
|
class BerkeleyBatch;
|
|
|
|
|
2020-06-15 14:29:29 -04:00
|
|
|
/** An instance of this class represents one database.
|
|
|
|
* For BerkeleyDB this is just a (env, strFile) tuple.
|
|
|
|
**/
|
2020-06-15 16:24:00 -04:00
|
|
|
class BerkeleyDatabase : public WalletDatabase
|
2020-06-15 14:29:29 -04:00
|
|
|
{
|
|
|
|
public:
|
2020-05-28 17:30:50 -04:00
|
|
|
BerkeleyDatabase() = delete;
|
2020-06-15 14:29:29 -04:00
|
|
|
|
|
|
|
/** Create DB handle to real database */
|
|
|
|
BerkeleyDatabase(std::shared_ptr<BerkeleyEnvironment> env, std::string filename) :
|
2020-06-15 16:24:00 -04:00
|
|
|
WalletDatabase(), env(std::move(env)), strFile(std::move(filename))
|
2020-06-15 14:29:29 -04:00
|
|
|
{
|
|
|
|
auto inserted = this->env->m_databases.emplace(strFile, std::ref(*this));
|
|
|
|
assert(inserted.second);
|
|
|
|
}
|
|
|
|
|
2020-06-15 16:24:00 -04:00
|
|
|
~BerkeleyDatabase() override;
|
2020-06-15 14:29:29 -04:00
|
|
|
|
2020-06-19 20:55:07 -04:00
|
|
|
/** Open the database if it is not already opened.
|
|
|
|
* Dummy function, doesn't do anything right now, but is needed for class abstraction */
|
2020-06-15 16:24:00 -04:00
|
|
|
void Open(const char* mode) override;
|
2020-06-15 14:29:29 -04:00
|
|
|
|
|
|
|
/** Rewrite the entire database on disk, with the exception of key pszSkip if non-zero
|
|
|
|
*/
|
2020-06-15 16:24:00 -04:00
|
|
|
bool Rewrite(const char* pszSkip=nullptr) override;
|
2020-06-15 14:29:29 -04:00
|
|
|
|
2020-06-19 20:51:07 -04:00
|
|
|
/** Indicate the a new database user has began using the database. */
|
2020-06-15 16:24:00 -04:00
|
|
|
void AddRef() override;
|
2020-06-19 20:51:07 -04:00
|
|
|
/** Indicate that database user has stopped using the database and that it could be flushed or closed. */
|
2020-06-15 16:24:00 -04:00
|
|
|
void RemoveRef() override;
|
2020-06-15 14:29:29 -04:00
|
|
|
|
|
|
|
/** Back up the entire database to a file.
|
|
|
|
*/
|
2020-06-15 16:24:00 -04:00
|
|
|
bool Backup(const std::string& strDest) const override;
|
2020-06-15 14:29:29 -04:00
|
|
|
|
2020-06-15 17:59:24 -04:00
|
|
|
/** Make sure all changes are flushed to database file.
|
|
|
|
*/
|
2020-06-15 16:24:00 -04:00
|
|
|
void Flush() override;
|
2020-06-15 17:59:24 -04:00
|
|
|
/** Flush to the database file and close the database.
|
|
|
|
* Also close the environment if no other databases are open in it.
|
2020-06-15 14:29:29 -04:00
|
|
|
*/
|
2020-06-15 16:24:00 -04:00
|
|
|
void Close() override;
|
2020-06-15 14:39:26 -04:00
|
|
|
/* flush the wallet passively (TRY_LOCK)
|
|
|
|
ideal to be called periodically */
|
2020-06-15 16:24:00 -04:00
|
|
|
bool PeriodicFlush() override;
|
2020-06-15 14:29:29 -04:00
|
|
|
|
2020-06-15 16:24:00 -04:00
|
|
|
void IncrementUpdateCounter() override;
|
2020-06-15 14:29:29 -04:00
|
|
|
|
2020-06-15 16:24:00 -04:00
|
|
|
void ReloadDbEnv() override;
|
2020-06-15 14:29:29 -04:00
|
|
|
|
2020-06-15 14:37:29 -04:00
|
|
|
/** Verifies the environment and database file */
|
2020-06-15 16:24:00 -04:00
|
|
|
bool Verify(bilingual_str& error) override;
|
2020-06-15 14:37:29 -04:00
|
|
|
|
2020-06-15 14:29:29 -04:00
|
|
|
/**
|
|
|
|
* Pointer to shared database environment.
|
|
|
|
*
|
|
|
|
* Normally there is only one BerkeleyDatabase object per
|
|
|
|
* BerkeleyEnvivonment, but in the special, backwards compatible case where
|
|
|
|
* multiple wallet BDB data files are loaded from the same directory, this
|
|
|
|
* will point to a shared instance that gets freed when the last data file
|
|
|
|
* is closed.
|
|
|
|
*/
|
|
|
|
std::shared_ptr<BerkeleyEnvironment> env;
|
|
|
|
|
|
|
|
/** Database pointer. This is initialized lazily and reset during flushes, so it can be null. */
|
|
|
|
std::unique_ptr<Db> m_db;
|
|
|
|
|
2020-06-15 16:14:57 -04:00
|
|
|
std::string strFile;
|
|
|
|
|
2020-06-15 16:54:58 -04:00
|
|
|
/** Make a BerkeleyBatch connected to this database */
|
2020-06-15 16:24:00 -04:00
|
|
|
std::unique_ptr<DatabaseBatch> MakeBatch(const char* mode = "r+", bool flush_on_close = true) override;
|
2020-06-15 14:29:29 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
/** RAII class that provides access to a Berkeley database */
|
2020-06-18 16:15:33 -04:00
|
|
|
class BerkeleyBatch : public DatabaseBatch
|
2020-06-15 14:29:29 -04:00
|
|
|
{
|
|
|
|
/** RAII class that automatically cleanses its data on destruction */
|
|
|
|
class SafeDbt final
|
|
|
|
{
|
|
|
|
Dbt m_dbt;
|
|
|
|
|
|
|
|
public:
|
|
|
|
// construct Dbt with internally-managed data
|
|
|
|
SafeDbt();
|
|
|
|
// construct Dbt with provided data
|
|
|
|
SafeDbt(void* data, size_t size);
|
|
|
|
~SafeDbt();
|
|
|
|
|
|
|
|
// delegate to Dbt
|
|
|
|
const void* get_data() const;
|
|
|
|
u_int32_t get_size() const;
|
|
|
|
|
|
|
|
// conversion operator to access the underlying Dbt
|
|
|
|
operator Dbt*();
|
|
|
|
};
|
|
|
|
|
2020-06-15 15:42:53 -04:00
|
|
|
private:
|
2020-06-18 16:15:33 -04:00
|
|
|
bool ReadKey(CDataStream&& key, CDataStream& value) override;
|
|
|
|
bool WriteKey(CDataStream&& key, CDataStream&& value, bool overwrite = true) override;
|
|
|
|
bool EraseKey(CDataStream&& key) override;
|
|
|
|
bool HasKey(CDataStream&& key) override;
|
2020-06-15 15:42:53 -04:00
|
|
|
|
2020-06-15 14:29:29 -04:00
|
|
|
protected:
|
|
|
|
Db* pdb;
|
|
|
|
std::string strFile;
|
|
|
|
DbTxn* activeTxn;
|
2020-05-14 21:17:01 -04:00
|
|
|
Dbc* m_cursor;
|
2020-06-15 14:29:29 -04:00
|
|
|
bool fReadOnly;
|
|
|
|
bool fFlushOnClose;
|
|
|
|
BerkeleyEnvironment *env;
|
2020-06-19 20:51:07 -04:00
|
|
|
BerkeleyDatabase& m_database;
|
2020-06-15 14:29:29 -04:00
|
|
|
|
|
|
|
public:
|
|
|
|
explicit BerkeleyBatch(BerkeleyDatabase& database, const char* pszMode = "r+", bool fFlushOnCloseIn=true);
|
2020-05-10 22:41:34 -04:00
|
|
|
~BerkeleyBatch() override;
|
2020-06-15 14:29:29 -04:00
|
|
|
|
|
|
|
BerkeleyBatch(const BerkeleyBatch&) = delete;
|
|
|
|
BerkeleyBatch& operator=(const BerkeleyBatch&) = delete;
|
|
|
|
|
2020-06-18 16:15:33 -04:00
|
|
|
void Flush() override;
|
|
|
|
void Close() override;
|
2020-06-15 14:29:29 -04:00
|
|
|
|
2020-06-18 16:15:33 -04:00
|
|
|
bool StartCursor() override;
|
|
|
|
bool ReadAtCursor(CDataStream& ssKey, CDataStream& ssValue, bool& complete) override;
|
|
|
|
void CloseCursor() override;
|
|
|
|
bool TxnBegin() override;
|
|
|
|
bool TxnCommit() override;
|
|
|
|
bool TxnAbort() override;
|
2020-06-15 14:29:29 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
std::string BerkeleyDatabaseVersion();
|
|
|
|
|
2020-08-04 16:40:31 -04:00
|
|
|
//! Check if Berkeley database exists at specified path.
|
|
|
|
bool ExistsBerkeleyDatabase(const fs::path& path);
|
|
|
|
|
|
|
|
//! Return object giving access to Berkeley database at specified path.
|
|
|
|
std::unique_ptr<BerkeleyDatabase> MakeBerkeleyDatabase(const fs::path& path, const DatabaseOptions& options, DatabaseStatus& status, bilingual_str& error);
|
|
|
|
|
2020-06-15 14:29:29 -04:00
|
|
|
#endif // BITCOIN_WALLET_BDB_H
|