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