Line data Source code
1 : // Copyright (c) 2020-2021 The Bitcoin Core developers
2 : // Distributed under the MIT software license, see the accompanying
3 : // file COPYING or http://www.opensource.org/licenses/mit-license.php.
4 :
5 : #ifndef BITCOIN_WALLET_SQLITE_H
6 : #define BITCOIN_WALLET_SQLITE_H
7 :
8 : #include <sync.h>
9 : #include <wallet/db.h>
10 :
11 : struct bilingual_str;
12 :
13 : struct sqlite3_stmt;
14 : struct sqlite3;
15 :
16 : namespace wallet {
17 : class SQLiteDatabase;
18 :
19 : /** RAII class that provides a database cursor */
20 : class SQLiteCursor : public DatabaseCursor
21 : {
22 : public:
23 : sqlite3_stmt* m_cursor_stmt{nullptr};
24 : // Copies of the prefix things for the prefix cursor.
25 : // Prevents SQLite from accessing temp variables for the prefix things.
26 : std::vector<std::byte> m_prefix_range_start;
27 : std::vector<std::byte> m_prefix_range_end;
28 :
29 0 : explicit SQLiteCursor() = default;
30 0 : explicit SQLiteCursor(std::vector<std::byte> start_range, std::vector<std::byte> end_range)
31 0 : : m_prefix_range_start(std::move(start_range)),
32 0 : m_prefix_range_end(std::move(end_range))
33 0 : {}
34 : ~SQLiteCursor() override;
35 :
36 : Status Next(DataStream& key, DataStream& value) override;
37 : };
38 :
39 : /** Class responsible for executing SQL statements in SQLite databases.
40 : * Methods are virtual so they can be overridden by unit tests testing unusual database conditions. */
41 0 : class SQliteExecHandler
42 : {
43 : public:
44 0 : virtual ~SQliteExecHandler() = default;
45 : virtual int Exec(SQLiteDatabase& database, const std::string& statement);
46 : };
47 :
48 : /** RAII class that provides access to a WalletDatabase */
49 : class SQLiteBatch : public DatabaseBatch
50 : {
51 : private:
52 : SQLiteDatabase& m_database;
53 : std::unique_ptr<SQliteExecHandler> m_exec_handler{std::make_unique<SQliteExecHandler>()};
54 :
55 : sqlite3_stmt* m_read_stmt{nullptr};
56 : sqlite3_stmt* m_insert_stmt{nullptr};
57 : sqlite3_stmt* m_overwrite_stmt{nullptr};
58 : sqlite3_stmt* m_delete_stmt{nullptr};
59 : sqlite3_stmt* m_delete_prefix_stmt{nullptr};
60 :
61 : /** Whether this batch has started a database transaction and whether it owns SQLiteDatabase::m_write_semaphore.
62 : * If the batch starts a db tx, it acquires the semaphore and sets this to true, keeping the semaphore
63 : * until the transaction ends to prevent other batch objects from writing to the database.
64 : *
65 : * If this batch did not start a transaction, the semaphore is acquired transiently when writing and m_txn
66 : * is not set.
67 : *
68 : * m_txn is different from HasActiveTxn() as it is only true when this batch has started the transaction,
69 : * not just when any batch has started a transaction.
70 : */
71 : bool m_txn{false};
72 :
73 : void SetupSQLStatements();
74 : bool ExecStatement(sqlite3_stmt* stmt, Span<const std::byte> blob);
75 :
76 : bool ReadKey(DataStream&& key, DataStream& value) override;
77 : bool WriteKey(DataStream&& key, DataStream&& value, bool overwrite = true) override;
78 : bool EraseKey(DataStream&& key) override;
79 : bool HasKey(DataStream&& key) override;
80 : bool ErasePrefix(Span<const std::byte> prefix) override;
81 :
82 : public:
83 : explicit SQLiteBatch(SQLiteDatabase& database);
84 0 : ~SQLiteBatch() override { Close(); }
85 :
86 : void SetExecHandler(std::unique_ptr<SQliteExecHandler>&& handler) { m_exec_handler = std::move(handler); }
87 :
88 : /* No-op. See comment on SQLiteDatabase::Flush */
89 0 : void Flush() override {}
90 :
91 : void Close() override;
92 :
93 : std::unique_ptr<DatabaseCursor> GetNewCursor() override;
94 : std::unique_ptr<DatabaseCursor> GetNewPrefixCursor(Span<const std::byte> prefix) override;
95 : bool TxnBegin() override;
96 : bool TxnCommit() override;
97 : bool TxnAbort() override;
98 0 : bool HasActiveTxn() override { return m_txn; }
99 : };
100 :
101 : /** An instance of this class represents one SQLite3 database.
102 : **/
103 : class SQLiteDatabase : public WalletDatabase
104 : {
105 : private:
106 : const bool m_mock{false};
107 :
108 : const std::string m_dir_path;
109 :
110 : const std::string m_file_path;
111 :
112 : /**
113 : * This mutex protects SQLite initialization and shutdown.
114 : * sqlite3_config() and sqlite3_shutdown() are not thread-safe (sqlite3_initialize() is).
115 : * Concurrent threads that execute SQLiteDatabase::SQLiteDatabase() should have just one
116 : * of them do the init and the rest wait for it to complete before all can proceed.
117 : */
118 : static Mutex g_sqlite_mutex;
119 : static int g_sqlite_count GUARDED_BY(g_sqlite_mutex);
120 :
121 : void Cleanup() noexcept EXCLUSIVE_LOCKS_REQUIRED(!g_sqlite_mutex);
122 :
123 : public:
124 : SQLiteDatabase() = delete;
125 :
126 : /** Create DB handle to real database */
127 : SQLiteDatabase(const fs::path& dir_path, const fs::path& file_path, const DatabaseOptions& options, bool mock = false);
128 :
129 : ~SQLiteDatabase();
130 :
131 : // Batches must acquire this semaphore on writing, and release when done writing.
132 : // This ensures that only one batch is modifying the database at a time.
133 : CSemaphore m_write_semaphore;
134 :
135 : bool Verify(bilingual_str& error);
136 :
137 : /** Open the database if it is not already opened */
138 : void Open() override;
139 :
140 : /** Close the database */
141 : void Close() override;
142 :
143 : /* These functions are unused */
144 0 : void AddRef() override { assert(false); }
145 0 : void RemoveRef() override { assert(false); }
146 :
147 : /** Rewrite the entire database on disk */
148 : bool Rewrite(const char* skip = nullptr) override;
149 :
150 : /** Back up the entire database to a file.
151 : */
152 : bool Backup(const std::string& dest) const override;
153 :
154 : /** No-ops
155 : *
156 : * SQLite always flushes everything to the database file after each transaction
157 : * (each Read/Write/Erase that we do is its own transaction unless we called
158 : * TxnBegin) so there is no need to have Flush or Periodic Flush.
159 : *
160 : * There is no DB env to reload, so ReloadDbEnv has nothing to do
161 : */
162 0 : void Flush() override {}
163 0 : bool PeriodicFlush() override { return false; }
164 0 : void ReloadDbEnv() override {}
165 :
166 0 : void IncrementUpdateCounter() override { ++nUpdateCounter; }
167 :
168 0 : std::string Filename() override { return m_file_path; }
169 0 : std::string Format() override { return "sqlite"; }
170 :
171 : /** Make a SQLiteBatch connected to this database */
172 : std::unique_ptr<DatabaseBatch> MakeBatch(bool flush_on_close = true) override;
173 :
174 : /** Return true if there is an on-going txn in this connection */
175 : bool HasActiveTxn();
176 :
177 : sqlite3* m_db{nullptr};
178 : bool m_use_unsafe_sync;
179 : };
180 :
181 : std::unique_ptr<SQLiteDatabase> MakeSQLiteDatabase(const fs::path& path, const DatabaseOptions& options, DatabaseStatus& status, bilingual_str& error);
182 :
183 : std::string SQLiteDatabaseVersion();
184 : } // namespace wallet
185 :
186 : #endif // BITCOIN_WALLET_SQLITE_H
|