From 498e2d8457f713bd141735b0edecb0314de0844b Mon Sep 17 00:00:00 2001 From: Arvid Norberg Date: Mon, 11 Nov 2024 21:35:43 +0100 Subject: [PATCH] add build option debug-disk-pool to log disk buffers allocated by category --- Jamfile | 3 ++ examples/custom_storage.cpp | 3 ++ include/libtorrent/aux_/disk_buffer_pool.hpp | 19 ++++++-- include/libtorrent/disk_buffer_holder.hpp | 6 +++ src/disabled_disk_io.cpp | 3 ++ src/disk_buffer_holder.cpp | 8 ++++ src/disk_buffer_pool.cpp | 50 +++++++++++++++++--- src/mmap_disk_io.cpp | 12 +++-- src/peer_connection.cpp | 5 +- 9 files changed, 94 insertions(+), 15 deletions(-) diff --git a/Jamfile b/Jamfile index f2e680bd61b..635a9c862ce 100644 --- a/Jamfile +++ b/Jamfile @@ -725,6 +725,9 @@ feature.compose on : TORRENT_EXPORT_EXTRA ; feature debug-disk-thread : off on : composite propagated ; feature.compose on : DEBUG_DISK_THREAD=1 ; +feature debug-disk-pool : off on : composite propagated ; +feature.compose on : TORRENT_DEBUG_BUFFER_POOL=1 ; + feature msvc-version-macro : off on : composite propagated link-incompatible ; # ask the compiler to correctly set __cplusplus version feature.compose on : /Zc\:__cplusplus ; diff --git a/examples/custom_storage.cpp b/examples/custom_storage.cpp index 01b0463b992..686e1e0c766 100644 --- a/examples/custom_storage.cpp +++ b/examples/custom_storage.cpp @@ -249,6 +249,9 @@ struct temp_disk_io final : lt::disk_interface // never free any buffer. We only return buffers owned by the storage // object } +#if TORRENT_DEBUG_BUFFER_POOL + void rename_buffer(char*, char const*) override {} +#endif void update_stats_counters(lt::counters&) const override {} diff --git a/include/libtorrent/aux_/disk_buffer_pool.hpp b/include/libtorrent/aux_/disk_buffer_pool.hpp index 99d39c44506..532fc0425f4 100644 --- a/include/libtorrent/aux_/disk_buffer_pool.hpp +++ b/include/libtorrent/aux_/disk_buffer_pool.hpp @@ -14,8 +14,12 @@ see LICENSE file. #include "libtorrent/config.hpp" -#if TORRENT_USE_INVARIANT_CHECKS -#include +#ifndef TORRENT_DEBUG_BUFFER_POOL +#define TORRENT_DEBUG_BUFFER_POOL 0 +#endif + +#if TORRENT_DEBUG_BUFFER_POOL +#include #endif #include #include @@ -56,6 +60,9 @@ namespace aux { void set_settings(settings_interface const& sett); +#if TORRENT_DEBUG_BUFFER_POOL + void rename_buffer(char* buf, char const* category) override; +#endif private: void free_buffer_impl(char* buf, std::unique_lock& l); @@ -94,8 +101,12 @@ namespace aux { // this is specifically exempt from release_asserts // since it's a quite costly check. Only for debug // builds. -#if TORRENT_USE_INVARIANT_CHECKS - std::set m_buffers_in_use; +#if TORRENT_DEBUG_BUFFER_POOL + std::map m_buffers_in_use; + std::map m_histogram; + time_t m_last_log = std::time(nullptr); + + void maybe_log(); #endif #if TORRENT_USE_ASSERTS int m_magic = 0x1337; diff --git a/include/libtorrent/disk_buffer_holder.hpp b/include/libtorrent/disk_buffer_holder.hpp index 4f50a5f1c9e..2140b6dd9e7 100644 --- a/include/libtorrent/disk_buffer_holder.hpp +++ b/include/libtorrent/disk_buffer_holder.hpp @@ -23,6 +23,9 @@ namespace libtorrent { struct TORRENT_EXPORT buffer_allocator_interface { virtual void free_disk_buffer(char* b) = 0; +#if TORRENT_DEBUG_BUFFER_POOL + virtual void rename_buffer(char* buf, char const* category) = 0; +#endif protected: ~buffer_allocator_interface() = default; }; @@ -77,6 +80,9 @@ namespace libtorrent { std::ptrdiff_t size() const { return m_size; } +#if TORRENT_DEBUG_BUFFER_POOL + void rename(char const* category); +#endif private: buffer_allocator_interface* m_allocator = nullptr; diff --git a/src/disabled_disk_io.cpp b/src/disabled_disk_io.cpp index 9d0d3e787b8..0a99387eab4 100644 --- a/src/disabled_disk_io.cpp +++ b/src/disabled_disk_io.cpp @@ -156,6 +156,9 @@ struct TORRENT_EXTRA_EXPORT disabled_disk_io final // since we just have a single zeroed buffer, we don't need to free anything // here. The buffer is owned by the disabled_disk_io object itself void free_disk_buffer(char*) override {} +#if TORRENT_DEBUG_BUFFER_POOL + void rename_buffer(char*, char const*) override {} +#endif std::vector get_status(storage_index_t) const override { return {}; } diff --git a/src/disk_buffer_holder.cpp b/src/disk_buffer_holder.cpp index 0c2c1a2076d..b1d5ca8edfd 100644 --- a/src/disk_buffer_holder.cpp +++ b/src/disk_buffer_holder.cpp @@ -38,5 +38,13 @@ namespace libtorrent { m_size = 0; } +#if TORRENT_DEBUG_BUFFER_POOL + void disk_buffer_holder::rename(char const* category) + { + if (m_buf != nullptr) + m_allocator->rename_buffer(m_buf, category); + } +#endif + disk_buffer_holder::~disk_buffer_holder() { reset(); } } diff --git a/src/disk_buffer_pool.cpp b/src/disk_buffer_pool.cpp index ea03a576626..8d39ba47a49 100644 --- a/src/disk_buffer_pool.cpp +++ b/src/disk_buffer_pool.cpp @@ -110,12 +110,13 @@ namespace { } char* disk_buffer_pool::allocate_buffer_impl(std::unique_lock& l - , char const*) + , char const* category) { TORRENT_ASSERT(m_settings_set); TORRENT_ASSERT(m_magic == 0x1337); TORRENT_ASSERT(l.owns_lock()); TORRENT_UNUSED(l); + TORRENT_UNUSED(category); char* ret = static_cast(std::malloc(default_block_size)); @@ -127,17 +128,20 @@ namespace { ++m_in_use; -#if TORRENT_USE_INVARIANT_CHECKS +#if TORRENT_DEBUG_BUFFER_POOL try { - TORRENT_ASSERT(m_buffers_in_use.count(ret) == 0); - m_buffers_in_use.insert(ret); + auto const [it, added] = m_buffers_in_use.insert({ret, category}); + TORRENT_UNUSED(it); + TORRENT_ASSERT(added); } catch (...) { free_buffer_impl(ret, l); return nullptr; } + m_histogram[category] += 1; + maybe_log(); #endif if (m_in_use >= m_low_watermark + (m_max_use - m_low_watermark) @@ -192,13 +196,47 @@ namespace { void disk_buffer_pool::remove_buffer_in_use(char* buf) { TORRENT_UNUSED(buf); -#if TORRENT_USE_INVARIANT_CHECKS - std::set::iterator i = m_buffers_in_use.find(buf); +#if TORRENT_DEBUG_BUFFER_POOL + auto i = m_buffers_in_use.find(buf); TORRENT_ASSERT(i != m_buffers_in_use.end()); + TORRENT_ASSERT(m_histogram[i->second] > 0); + m_histogram[i->second] -= 1; m_buffers_in_use.erase(i); + maybe_log(); #endif } +#if TORRENT_DEBUG_BUFFER_POOL + void disk_buffer_pool::maybe_log() + { + time_t const now = std::time(nullptr); + if (now - m_last_log > 1) + { + m_last_log = now; + FILE* f = fopen("buffer_pool.log", "a"); + fprintf(f, "%ld ", now); + for (auto it = m_histogram.begin(); it != m_histogram.end(); ++it) + { + fprintf(f, "%s:%d ", it->first.c_str(), it->second); + } + fputs("\n", f); + fclose(f); + } + } + + void disk_buffer_pool::rename_buffer(char* buf, char const* category) + { + std::unique_lock l(m_pool_mutex); + auto i = m_buffers_in_use.find(buf); + TORRENT_ASSERT(i != m_buffers_in_use.end()); + TORRENT_ASSERT(m_histogram[i->second] > 0); + m_histogram[i->second] -= 1; + i->second = category; + m_histogram[category] += 1; + maybe_log(); + } +#endif + void disk_buffer_pool::free_buffer_impl(char* buf, std::unique_lock& l) { TORRENT_ASSERT(buf); diff --git a/src/mmap_disk_io.cpp b/src/mmap_disk_io.cpp index 3fa5b0245f1..2ad84e9af9d 100644 --- a/src/mmap_disk_io.cpp +++ b/src/mmap_disk_io.cpp @@ -436,7 +436,7 @@ TORRENT_EXPORT std::unique_ptr mmap_disk_io_constructor( status_t mmap_disk_io::do_job(aux::job::read& a, aux::mmap_disk_job* j) { - a.buf = disk_buffer_holder(m_buffer_pool, m_buffer_pool.allocate_buffer("send buffer"), default_block_size); + a.buf = disk_buffer_holder(m_buffer_pool, m_buffer_pool.allocate_buffer("send buffer (cache miss)"), default_block_size); if (!a.buf) { j->error.ec = error::no_memory; @@ -477,6 +477,10 @@ TORRENT_EXPORT std::unique_ptr mmap_disk_io_constructor( m_stats_counters.inc_stats_counter(counters::num_writing_threads, 1); +#if TORRENT_DEBUG_BUFFER_POOL + buffer.rename("flushing"); +#endif + // the actual write operation int const ret = j->storage->write(m_settings, b , a.piece, a.offset, file_mode, j->flags, j->error); @@ -554,7 +558,7 @@ TORRENT_EXPORT std::unique_ptr mmap_disk_io_constructor( int const ret = m_store_buffer.get2(loc1, loc2, [&](char const* buf1, char const* buf2) { buffer = disk_buffer_holder(m_buffer_pool - , m_buffer_pool.allocate_buffer("send buffer") + , m_buffer_pool.allocate_buffer("send buffer (cache hit)") , r.length); if (!buffer) { @@ -605,7 +609,7 @@ TORRENT_EXPORT std::unique_ptr mmap_disk_io_constructor( { if (m_store_buffer.get({ storage, r.piece, block_offset }, [&](char const* buf) { - buffer = disk_buffer_holder(m_buffer_pool, m_buffer_pool.allocate_buffer("send buffer"), r.length); + buffer = disk_buffer_holder(m_buffer_pool, m_buffer_pool.allocate_buffer("send buffer (cache hit)"), r.length); if (!buffer) { ec.ec = error::no_memory; @@ -641,7 +645,7 @@ TORRENT_EXPORT std::unique_ptr mmap_disk_io_constructor( TORRENT_ASSERT(valid_flags(flags)); bool exceeded = false; disk_buffer_holder buffer(m_buffer_pool, m_buffer_pool.allocate_buffer( - exceeded, o, "receive buffer"), default_block_size); + exceeded, o, "store buffer"), default_block_size); if (!buffer) aux::throw_ex(); std::memcpy(buffer.data(), buf, aux::numeric_cast(r.length)); diff --git a/src/peer_connection.cpp b/src/peer_connection.cpp index 9f87ebcccd3..e7c6ee83f8e 100644 --- a/src/peer_connection.cpp +++ b/src/peer_connection.cpp @@ -3173,9 +3173,11 @@ namespace { { // if any other peer has a busy request to this block, we need // to cancel it too - t->cancel_block(block_finished); if (t->has_picker()) + { + t->cancel_block(block_finished); t->picker().write_failed(block_finished); + } if (t->has_storage()) { @@ -3724,6 +3726,7 @@ namespace { TORRENT_ASSERT(block.piece_index != piece_block::invalid.piece_index); TORRENT_ASSERT(block.piece_index < t->torrent_file().end_piece()); TORRENT_ASSERT(block.block_index < t->torrent_file().piece_size(block.piece_index)); + TORRENT_ASSERT(t->has_picker()); // if all the peers that requested this block has been // cancelled, then just ignore the cancel.