/root/bitcoin/src/dbwrapper.cpp
Line | Count | Source |
1 | | // Copyright (c) 2012-present 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 | | #include <dbwrapper.h> |
6 | | |
7 | | #include <logging.h> |
8 | | #include <random.h> |
9 | | #include <serialize.h> |
10 | | #include <span.h> |
11 | | #include <streams.h> |
12 | | #include <util/fs.h> |
13 | | #include <util/fs_helpers.h> |
14 | | #include <util/obfuscation.h> |
15 | | #include <util/strencodings.h> |
16 | | |
17 | | #include <algorithm> |
18 | | #include <cassert> |
19 | | #include <cstdarg> |
20 | | #include <cstdint> |
21 | | #include <cstdio> |
22 | | #include <leveldb/cache.h> |
23 | | #include <leveldb/db.h> |
24 | | #include <leveldb/env.h> |
25 | | #include <leveldb/filter_policy.h> |
26 | | #include <leveldb/helpers/memenv/memenv.h> |
27 | | #include <leveldb/iterator.h> |
28 | | #include <leveldb/options.h> |
29 | | #include <leveldb/slice.h> |
30 | | #include <leveldb/status.h> |
31 | | #include <leveldb/write_batch.h> |
32 | | #include <memory> |
33 | | #include <optional> |
34 | | #include <utility> |
35 | | |
36 | 0 | static auto CharCast(const std::byte* data) { return reinterpret_cast<const char*>(data); } |
37 | | |
38 | | bool DestroyDB(const std::string& path_str) |
39 | 0 | { |
40 | 0 | return leveldb::DestroyDB(path_str, {}).ok(); |
41 | 0 | } |
42 | | |
43 | | /** Handle database error by throwing dbwrapper_error exception. |
44 | | */ |
45 | | static void HandleError(const leveldb::Status& status) |
46 | 0 | { |
47 | 0 | if (status.ok()) |
48 | 0 | return; |
49 | 0 | const std::string errmsg = "Fatal LevelDB error: " + status.ToString(); |
50 | 0 | LogPrintf("%s\n", errmsg); |
51 | 0 | LogPrintf("You can use -debug=leveldb to get more complete diagnostic messages\n"); |
52 | 0 | throw dbwrapper_error(errmsg); |
53 | 0 | } |
54 | | |
55 | | class CBitcoinLevelDBLogger : public leveldb::Logger { |
56 | | public: |
57 | | // This code is adapted from posix_logger.h, which is why it is using vsprintf. |
58 | | // Please do not do this in normal code |
59 | 0 | void Logv(const char * format, va_list ap) override { |
60 | 0 | if (!LogAcceptCategory(BCLog::LEVELDB, BCLog::Level::Debug)) { |
61 | 0 | return; |
62 | 0 | } |
63 | 0 | char buffer[500]; |
64 | 0 | for (int iter = 0; iter < 2; iter++) { |
65 | 0 | char* base; |
66 | 0 | int bufsize; |
67 | 0 | if (iter == 0) { |
68 | 0 | bufsize = sizeof(buffer); |
69 | 0 | base = buffer; |
70 | 0 | } |
71 | 0 | else { |
72 | 0 | bufsize = 30000; |
73 | 0 | base = new char[bufsize]; |
74 | 0 | } |
75 | 0 | char* p = base; |
76 | 0 | char* limit = base + bufsize; |
77 | | |
78 | | // Print the message |
79 | 0 | if (p < limit) { |
80 | 0 | va_list backup_ap; |
81 | 0 | va_copy(backup_ap, ap); |
82 | | // Do not use vsnprintf elsewhere in bitcoin source code, see above. |
83 | 0 | p += vsnprintf(p, limit - p, format, backup_ap); |
84 | 0 | va_end(backup_ap); |
85 | 0 | } |
86 | | |
87 | | // Truncate to available space if necessary |
88 | 0 | if (p >= limit) { |
89 | 0 | if (iter == 0) { |
90 | 0 | continue; // Try again with larger buffer |
91 | 0 | } |
92 | 0 | else { |
93 | 0 | p = limit - 1; |
94 | 0 | } |
95 | 0 | } |
96 | | |
97 | | // Add newline if necessary |
98 | 0 | if (p == base || p[-1] != '\n') { |
99 | 0 | *p++ = '\n'; |
100 | 0 | } |
101 | |
|
102 | 0 | assert(p <= limit); |
103 | 0 | base[std::min(bufsize - 1, (int)(p - base))] = '\0'; |
104 | 0 | LogDebug(BCLog::LEVELDB, "%s\n", util::RemoveSuffixView(base, "\n")); |
105 | 0 | if (base != buffer) { |
106 | 0 | delete[] base; |
107 | 0 | } |
108 | 0 | break; |
109 | 0 | } |
110 | 0 | } |
111 | | }; |
112 | | |
113 | 0 | static void SetMaxOpenFiles(leveldb::Options *options) { |
114 | | // On most platforms the default setting of max_open_files (which is 1000) |
115 | | // is optimal. On Windows using a large file count is OK because the handles |
116 | | // do not interfere with select() loops. On 64-bit Unix hosts this value is |
117 | | // also OK, because up to that amount LevelDB will use an mmap |
118 | | // implementation that does not use extra file descriptors (the fds are |
119 | | // closed after being mmap'ed). |
120 | | // |
121 | | // Increasing the value beyond the default is dangerous because LevelDB will |
122 | | // fall back to a non-mmap implementation when the file count is too large. |
123 | | // On 32-bit Unix host we should decrease the value because the handles use |
124 | | // up real fds, and we want to avoid fd exhaustion issues. |
125 | | // |
126 | | // See PR #12495 for further discussion. |
127 | |
|
128 | 0 | int default_open_files = options->max_open_files; |
129 | 0 | #ifndef WIN32 |
130 | 0 | if (sizeof(void*) < 8) { |
131 | 0 | options->max_open_files = 64; |
132 | 0 | } |
133 | 0 | #endif |
134 | 0 | LogDebug(BCLog::LEVELDB, "LevelDB using max_open_files=%d (default=%d)\n", |
135 | 0 | options->max_open_files, default_open_files); |
136 | 0 | } |
137 | | |
138 | | static leveldb::Options GetOptions(size_t nCacheSize) |
139 | 0 | { |
140 | 0 | leveldb::Options options; |
141 | 0 | options.block_cache = leveldb::NewLRUCache(nCacheSize / 2); |
142 | 0 | options.write_buffer_size = nCacheSize / 4; // up to two write buffers may be held in memory simultaneously |
143 | 0 | options.filter_policy = leveldb::NewBloomFilterPolicy(10); |
144 | 0 | options.compression = leveldb::kNoCompression; |
145 | 0 | options.info_log = new CBitcoinLevelDBLogger(); |
146 | 0 | if (leveldb::kMajorVersion > 1 || (leveldb::kMajorVersion == 1 && leveldb::kMinorVersion >= 16)) { |
147 | | // LevelDB versions before 1.16 consider short writes to be corruption. Only trigger error |
148 | | // on corruption in later versions. |
149 | 0 | options.paranoid_checks = true; |
150 | 0 | } |
151 | 0 | options.max_file_size = std::max(options.max_file_size, DBWRAPPER_MAX_FILE_SIZE); |
152 | 0 | SetMaxOpenFiles(&options); |
153 | 0 | return options; |
154 | 0 | } |
155 | | |
156 | | struct CDBBatch::WriteBatchImpl { |
157 | | leveldb::WriteBatch batch; |
158 | | }; |
159 | | |
160 | | CDBBatch::CDBBatch(const CDBWrapper& _parent) |
161 | 0 | : parent{_parent}, |
162 | 0 | m_impl_batch{std::make_unique<CDBBatch::WriteBatchImpl>()} |
163 | 0 | { |
164 | 0 | Clear(); |
165 | 0 | }; |
166 | | |
167 | 0 | CDBBatch::~CDBBatch() = default; |
168 | | |
169 | | void CDBBatch::Clear() |
170 | 0 | { |
171 | 0 | m_impl_batch->batch.Clear(); |
172 | 0 | } |
173 | | |
174 | | void CDBBatch::WriteImpl(std::span<const std::byte> key, DataStream& ssValue) |
175 | 0 | { |
176 | 0 | leveldb::Slice slKey(CharCast(key.data()), key.size()); |
177 | 0 | dbwrapper_private::GetObfuscation(parent)(ssValue); |
178 | 0 | leveldb::Slice slValue(CharCast(ssValue.data()), ssValue.size()); |
179 | 0 | m_impl_batch->batch.Put(slKey, slValue); |
180 | 0 | } |
181 | | |
182 | | void CDBBatch::EraseImpl(std::span<const std::byte> key) |
183 | 0 | { |
184 | 0 | leveldb::Slice slKey(CharCast(key.data()), key.size()); |
185 | 0 | m_impl_batch->batch.Delete(slKey); |
186 | 0 | } |
187 | | |
188 | | size_t CDBBatch::ApproximateSize() const |
189 | 0 | { |
190 | 0 | return m_impl_batch->batch.ApproximateSize(); |
191 | 0 | } |
192 | | |
193 | | struct LevelDBContext { |
194 | | //! custom environment this database is using (may be nullptr in case of default environment) |
195 | | leveldb::Env* penv; |
196 | | |
197 | | //! database options used |
198 | | leveldb::Options options; |
199 | | |
200 | | //! options used when reading from the database |
201 | | leveldb::ReadOptions readoptions; |
202 | | |
203 | | //! options used when iterating over values of the database |
204 | | leveldb::ReadOptions iteroptions; |
205 | | |
206 | | //! options used when writing to the database |
207 | | leveldb::WriteOptions writeoptions; |
208 | | |
209 | | //! options used when sync writing to the database |
210 | | leveldb::WriteOptions syncoptions; |
211 | | |
212 | | //! the database itself |
213 | | leveldb::DB* pdb; |
214 | | }; |
215 | | |
216 | | CDBWrapper::CDBWrapper(const DBParams& params) |
217 | 0 | : m_db_context{std::make_unique<LevelDBContext>()}, m_name{fs::PathToString(params.path.stem())}, m_path{params.path}, m_is_memory{params.memory_only} |
218 | 0 | { |
219 | 0 | DBContext().penv = nullptr; |
220 | 0 | DBContext().readoptions.verify_checksums = true; |
221 | 0 | DBContext().iteroptions.verify_checksums = true; |
222 | 0 | DBContext().iteroptions.fill_cache = false; |
223 | 0 | DBContext().syncoptions.sync = true; |
224 | 0 | DBContext().options = GetOptions(params.cache_bytes); |
225 | 0 | DBContext().options.create_if_missing = true; |
226 | 0 | if (params.memory_only) { |
227 | 0 | DBContext().penv = leveldb::NewMemEnv(leveldb::Env::Default()); |
228 | 0 | DBContext().options.env = DBContext().penv; |
229 | 0 | } else { |
230 | 0 | if (params.wipe_data) { |
231 | 0 | LogInfo("Wiping LevelDB in %s", fs::PathToString(params.path)); |
232 | 0 | leveldb::Status result = leveldb::DestroyDB(fs::PathToString(params.path), DBContext().options); |
233 | 0 | HandleError(result); |
234 | 0 | } |
235 | 0 | TryCreateDirectories(params.path); |
236 | 0 | LogInfo("Opening LevelDB in %s", fs::PathToString(params.path)); |
237 | 0 | } |
238 | | // PathToString() return value is safe to pass to leveldb open function, |
239 | | // because on POSIX leveldb passes the byte string directly to ::open(), and |
240 | | // on Windows it converts from UTF-8 to UTF-16 before calling ::CreateFileW |
241 | | // (see env_posix.cc and env_windows.cc). |
242 | 0 | leveldb::Status status = leveldb::DB::Open(DBContext().options, fs::PathToString(params.path), &DBContext().pdb); |
243 | 0 | HandleError(status); |
244 | 0 | LogInfo("Opened LevelDB successfully"); |
245 | |
|
246 | 0 | if (params.options.force_compact) { |
247 | 0 | LogInfo("Starting database compaction of %s", fs::PathToString(params.path)); |
248 | 0 | DBContext().pdb->CompactRange(nullptr, nullptr); |
249 | 0 | LogInfo("Finished database compaction of %s", fs::PathToString(params.path)); |
250 | 0 | } |
251 | |
|
252 | 0 | if (!Read(OBFUSCATION_KEY, m_obfuscation) && params.obfuscate && IsEmpty()) { |
253 | | // Generate and write the new obfuscation key. |
254 | 0 | const Obfuscation obfuscation{FastRandomContext{}.randbytes<Obfuscation::KEY_SIZE>()}; |
255 | 0 | assert(!m_obfuscation); // Make sure the key is written without obfuscation. |
256 | 0 | Write(OBFUSCATION_KEY, obfuscation); |
257 | 0 | m_obfuscation = obfuscation; |
258 | 0 | LogInfo("Wrote new obfuscation key for %s: %s", fs::PathToString(params.path), m_obfuscation.HexKey()); |
259 | 0 | } |
260 | 0 | LogInfo("Using obfuscation key for %s: %s", fs::PathToString(params.path), m_obfuscation.HexKey()); |
261 | 0 | } |
262 | | |
263 | | CDBWrapper::~CDBWrapper() |
264 | 2 | { |
265 | 2 | delete DBContext().pdb; |
266 | 2 | DBContext().pdb = nullptr; |
267 | 2 | delete DBContext().options.filter_policy; |
268 | 2 | DBContext().options.filter_policy = nullptr; |
269 | 2 | delete DBContext().options.info_log; |
270 | 2 | DBContext().options.info_log = nullptr; |
271 | 2 | delete DBContext().options.block_cache; |
272 | 2 | DBContext().options.block_cache = nullptr; |
273 | 2 | delete DBContext().penv; |
274 | 2 | DBContext().options.env = nullptr; |
275 | 2 | } |
276 | | |
277 | | bool CDBWrapper::WriteBatch(CDBBatch& batch, bool fSync) |
278 | 0 | { |
279 | 0 | const bool log_memory = LogAcceptCategory(BCLog::LEVELDB, BCLog::Level::Debug); |
280 | 0 | double mem_before = 0; |
281 | 0 | if (log_memory) { |
282 | 0 | mem_before = DynamicMemoryUsage() / 1024.0 / 1024; |
283 | 0 | } |
284 | 0 | leveldb::Status status = DBContext().pdb->Write(fSync ? DBContext().syncoptions : DBContext().writeoptions, &batch.m_impl_batch->batch); |
285 | 0 | HandleError(status); |
286 | 0 | if (log_memory) { |
287 | 0 | double mem_after = DynamicMemoryUsage() / 1024.0 / 1024; |
288 | 0 | LogDebug(BCLog::LEVELDB, "WriteBatch memory usage: db=%s, before=%.1fMiB, after=%.1fMiB\n", |
289 | 0 | m_name, mem_before, mem_after); |
290 | 0 | } |
291 | 0 | return true; |
292 | 0 | } |
293 | | |
294 | | size_t CDBWrapper::DynamicMemoryUsage() const |
295 | 0 | { |
296 | 0 | std::string memory; |
297 | 0 | std::optional<size_t> parsed; |
298 | 0 | if (!DBContext().pdb->GetProperty("leveldb.approximate-memory-usage", &memory) || !(parsed = ToIntegral<size_t>(memory))) { |
299 | 0 | LogDebug(BCLog::LEVELDB, "Failed to get approximate-memory-usage property\n"); |
300 | 0 | return 0; |
301 | 0 | } |
302 | 0 | return parsed.value(); |
303 | 0 | } |
304 | | |
305 | | std::optional<std::string> CDBWrapper::ReadImpl(std::span<const std::byte> key) const |
306 | 0 | { |
307 | 0 | leveldb::Slice slKey(CharCast(key.data()), key.size()); |
308 | 0 | std::string strValue; |
309 | 0 | leveldb::Status status = DBContext().pdb->Get(DBContext().readoptions, slKey, &strValue); |
310 | 0 | if (!status.ok()) { |
311 | 0 | if (status.IsNotFound()) |
312 | 0 | return std::nullopt; |
313 | 0 | LogPrintf("LevelDB read failure: %s\n", status.ToString()); |
314 | 0 | HandleError(status); |
315 | 0 | } |
316 | 0 | return strValue; |
317 | 0 | } |
318 | | |
319 | | bool CDBWrapper::ExistsImpl(std::span<const std::byte> key) const |
320 | 0 | { |
321 | 0 | leveldb::Slice slKey(CharCast(key.data()), key.size()); |
322 | |
|
323 | 0 | std::string strValue; |
324 | 0 | leveldb::Status status = DBContext().pdb->Get(DBContext().readoptions, slKey, &strValue); |
325 | 0 | if (!status.ok()) { |
326 | 0 | if (status.IsNotFound()) |
327 | 0 | return false; |
328 | 0 | LogPrintf("LevelDB read failure: %s\n", status.ToString()); |
329 | 0 | HandleError(status); |
330 | 0 | } |
331 | 0 | return true; |
332 | 0 | } |
333 | | |
334 | | size_t CDBWrapper::EstimateSizeImpl(std::span<const std::byte> key1, std::span<const std::byte> key2) const |
335 | 0 | { |
336 | 0 | leveldb::Slice slKey1(CharCast(key1.data()), key1.size()); |
337 | 0 | leveldb::Slice slKey2(CharCast(key2.data()), key2.size()); |
338 | 0 | uint64_t size = 0; |
339 | 0 | leveldb::Range range(slKey1, slKey2); |
340 | 0 | DBContext().pdb->GetApproximateSizes(&range, 1, &size); |
341 | 0 | return size; |
342 | 0 | } |
343 | | |
344 | | bool CDBWrapper::IsEmpty() |
345 | 0 | { |
346 | 0 | std::unique_ptr<CDBIterator> it(NewIterator()); |
347 | 0 | it->SeekToFirst(); |
348 | 0 | return !(it->Valid()); |
349 | 0 | } |
350 | | |
351 | | struct CDBIterator::IteratorImpl { |
352 | | const std::unique_ptr<leveldb::Iterator> iter; |
353 | | |
354 | 0 | explicit IteratorImpl(leveldb::Iterator* _iter) : iter{_iter} {} |
355 | | }; |
356 | | |
357 | 0 | CDBIterator::CDBIterator(const CDBWrapper& _parent, std::unique_ptr<IteratorImpl> _piter) : parent(_parent), |
358 | 0 | m_impl_iter(std::move(_piter)) {} |
359 | | |
360 | | CDBIterator* CDBWrapper::NewIterator() |
361 | 0 | { |
362 | 0 | return new CDBIterator{*this, std::make_unique<CDBIterator::IteratorImpl>(DBContext().pdb->NewIterator(DBContext().iteroptions))}; |
363 | 0 | } |
364 | | |
365 | | void CDBIterator::SeekImpl(std::span<const std::byte> key) |
366 | 0 | { |
367 | 0 | leveldb::Slice slKey(CharCast(key.data()), key.size()); |
368 | 0 | m_impl_iter->iter->Seek(slKey); |
369 | 0 | } |
370 | | |
371 | | std::span<const std::byte> CDBIterator::GetKeyImpl() const |
372 | 0 | { |
373 | 0 | return MakeByteSpan(m_impl_iter->iter->key()); |
374 | 0 | } |
375 | | |
376 | | std::span<const std::byte> CDBIterator::GetValueImpl() const |
377 | 0 | { |
378 | 0 | return MakeByteSpan(m_impl_iter->iter->value()); |
379 | 0 | } |
380 | | |
381 | 0 | CDBIterator::~CDBIterator() = default; |
382 | 0 | bool CDBIterator::Valid() const { return m_impl_iter->iter->Valid(); } |
383 | 0 | void CDBIterator::SeekToFirst() { m_impl_iter->iter->SeekToFirst(); } |
384 | 0 | void CDBIterator::Next() { m_impl_iter->iter->Next(); } |
385 | | |
386 | | namespace dbwrapper_private { |
387 | | |
388 | | const Obfuscation& GetObfuscation(const CDBWrapper& w) |
389 | 0 | { |
390 | 0 | return w.m_obfuscation; |
391 | 0 | } |
392 | | |
393 | | } // namespace dbwrapper_private |