mirror of
https://github.com/transmission/transmission
synced 2025-01-04 22:11:23 +00:00
74f367c150
* fix: sonarcloud useless-sequence-of-pointer-operators warning
Xref: https://sonarcloud.io/project/issues\?id\=transmission_transmission\&issues\=AXzyMRYbK9dvryvWm8XC\&open\=AXzyMRYbK9dvryvWm8XC
* fix: sonarcloud unused-function warning
Xref: https://sonarcloud.io/project/issues\?id\=transmission_transmission\&issues\=AX06Stxd1usi2gyYkPTE\&open\=AX06Stxd1usi2gyYkPTE
* fix: sonarcloud init-statement warning
Xref: https://sonarcloud.io/project/issues\?id\=transmission_transmission\&issues\=AXzyMRABK9dvryvWm8Wv\&open\=AXzyMRABK9dvryvWm8Wv
* fix: sonarcloud assignment-in-expression warning
Xref: https://sonarcloud.io/project/issues\?id\=transmission_transmission\&issues\=AXztC8rqIH_lVCrUGWcX\&open\=AXztC8rqIH_lVCrUGWcX
* fix: sonarcloud use-init-statement warning
Xref: https://sonarcloud.io/project/issues\?id\=transmission_transmission\&issues\=AXy-sssdW63O2GMuTBKY\&open\=AXy-sssdW63O2GMuTBKY
* fix: replace-redundant-type-with-auto warning
Xref: https://sonarcloud.io/project/issues\?id\=transmission_transmission\&issues\=AXy-sssdW63O2GMuTBKZ\&open\=AXy-sssdW63O2GMuTBKZ
* fix: use-init-statement
Xref: https://sonarcloud.io/project/issues\?id\=transmission_transmission\&issues\=AXzOK_MX7S1ht0f4sjIh\&open\=AXzOK_MX7S1ht0f4sjIh
* fix: use-structured-binding-declaration warning
Xref: https://sonarcloud.io/project/issues\?id\=transmission_transmission\&issues\=AX06Stxd1usi2gyYkPTK\&open\=AX06Stxd1usi2gyYkPTK
* fix: use-init-statement
Xref: https://sonarcloud.io/project/issues\?id\=transmission_transmission\&issues\=AX06Stxd1usi2gyYkPTL\&open\=AX06Stxd1usi2gyYkPTL
* fix: use-structured-binding-declaration warning
Xref: https://sonarcloud.io/project/issues\?id\=transmission_transmission\&issues\=AX06Stxd1usi2gyYkPTP\&open\=AX06Stxd1usi2gyYkPTP
* fixup! fix: use-structured-binding-declaration warning
* Revert "fix: sonarcloud useless-sequence-of-pointer-operators warning"
This reverts commit 26c99e75c6
.
cannot convert from 'const _InIt' to 'const rpc_method *'
237 lines
5.1 KiB
C++
237 lines
5.1 KiB
C++
/*
|
|
* This file Copyright (C) 2021 Mnemosyne LLC
|
|
*
|
|
* It may be used under the GNU GPL versions 2 or 3
|
|
* or any future license endorsed by Mnemosyne LLC.
|
|
*
|
|
*/
|
|
|
|
#include <algorithm>
|
|
#include <functional>
|
|
#include <memory>
|
|
#include <numeric>
|
|
#include <utility>
|
|
#include <unordered_map>
|
|
#include <unordered_set>
|
|
#include <vector>
|
|
|
|
#define LIBTRANSMISSION_PEER_MODULE
|
|
|
|
#include "peer-mgr-active-requests.h"
|
|
|
|
namespace
|
|
{
|
|
|
|
struct peer_at
|
|
{
|
|
tr_peer* peer;
|
|
time_t when;
|
|
|
|
peer_at(tr_peer* p, time_t w)
|
|
: peer{ p }
|
|
, when{ w }
|
|
{
|
|
}
|
|
|
|
int compare(peer_at const& that) const // <=>
|
|
{
|
|
if (peer != that.peer)
|
|
{
|
|
return peer < that.peer ? -1 : 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
bool operator==(peer_at const& that) const
|
|
{
|
|
return compare(that) == 0;
|
|
}
|
|
};
|
|
|
|
struct PeerAtHash
|
|
{
|
|
std::size_t operator()(peer_at const& pa) const noexcept
|
|
{
|
|
return std::hash<tr_peer*>{}(pa.peer);
|
|
}
|
|
};
|
|
|
|
} // namespace
|
|
|
|
class ActiveRequests::Impl
|
|
{
|
|
public:
|
|
size_t size() const
|
|
{
|
|
return size_;
|
|
}
|
|
|
|
size_t count(tr_peer const* peer) const
|
|
{
|
|
auto const it = count_.find(peer);
|
|
return it != std::end(count_) ? it->second : size_t{};
|
|
}
|
|
|
|
void incCount(tr_peer const* peer)
|
|
{
|
|
++count_[peer];
|
|
++size_;
|
|
}
|
|
|
|
void decCount(tr_peer const* peer)
|
|
{
|
|
auto it = count_.find(peer);
|
|
TR_ASSERT(it != std::end(count_));
|
|
TR_ASSERT(it->second > 0);
|
|
TR_ASSERT(size_ > 0);
|
|
|
|
if (it != std::end(count_))
|
|
{
|
|
if (--it->second == 0)
|
|
{
|
|
count_.erase(it);
|
|
}
|
|
--size_;
|
|
}
|
|
}
|
|
|
|
std::unordered_map<tr_peer const*, size_t> count_;
|
|
|
|
std::unordered_map<tr_block_index_t, std::unordered_set<peer_at, PeerAtHash>> blocks_;
|
|
|
|
private:
|
|
size_t size_ = 0;
|
|
};
|
|
|
|
ActiveRequests::ActiveRequests()
|
|
: impl_{ std::make_unique<Impl>() }
|
|
{
|
|
}
|
|
|
|
ActiveRequests::~ActiveRequests() = default;
|
|
|
|
bool ActiveRequests::add(tr_block_index_t block, tr_peer* peer, time_t when)
|
|
{
|
|
bool const added = impl_->blocks_[block].emplace(peer, when).second;
|
|
|
|
if (added)
|
|
{
|
|
impl_->incCount(peer);
|
|
}
|
|
|
|
return added;
|
|
}
|
|
|
|
// remove a request to `peer` for `block`
|
|
bool ActiveRequests::remove(tr_block_index_t block, tr_peer const* peer)
|
|
{
|
|
auto const it = impl_->blocks_.find(block);
|
|
auto const key = peer_at{ const_cast<tr_peer*>(peer), 0 };
|
|
auto const removed = it != std::end(impl_->blocks_) && it->second.erase(key) != 0;
|
|
|
|
if (removed)
|
|
{
|
|
impl_->decCount(peer);
|
|
|
|
if (std::empty(it->second))
|
|
{
|
|
impl_->blocks_.erase(it);
|
|
}
|
|
}
|
|
|
|
return removed;
|
|
}
|
|
|
|
// remove requests to `peer` and return the associated blocks
|
|
std::vector<tr_block_index_t> ActiveRequests::remove(tr_peer const* peer)
|
|
{
|
|
auto removed = std::vector<tr_block_index_t>{};
|
|
removed.reserve(impl_->blocks_.size());
|
|
|
|
auto const key = peer_at{ const_cast<tr_peer*>(peer), 0 };
|
|
for (auto const& [block, peers_at] : impl_->blocks_)
|
|
{
|
|
if (peers_at.count(key))
|
|
{
|
|
removed.push_back(block);
|
|
}
|
|
}
|
|
|
|
for (auto block : removed)
|
|
{
|
|
remove(block, peer);
|
|
}
|
|
|
|
return removed;
|
|
}
|
|
|
|
// remove requests for `block` and return the associated peers
|
|
std::vector<tr_peer*> ActiveRequests::remove(tr_block_index_t block)
|
|
{
|
|
auto removed = std::vector<tr_peer*>{};
|
|
|
|
if (auto it = impl_->blocks_.find(block); it != std::end(impl_->blocks_))
|
|
{
|
|
auto const n = std::size(it->second);
|
|
removed.resize(n);
|
|
std::transform(
|
|
std::begin(it->second),
|
|
std::end(it->second),
|
|
std::begin(removed),
|
|
[](auto const& sent) { return sent.peer; });
|
|
impl_->blocks_.erase(block);
|
|
}
|
|
|
|
for (auto* peer : removed)
|
|
{
|
|
impl_->decCount(peer);
|
|
}
|
|
|
|
return removed;
|
|
}
|
|
|
|
// return true if there's an active request to `peer` for `block`
|
|
bool ActiveRequests::has(tr_block_index_t block, tr_peer const* peer) const
|
|
{
|
|
auto const it = impl_->blocks_.find(block);
|
|
return it != std::end(impl_->blocks_) && it->second.count(peer_at{ const_cast<tr_peer*>(peer), 0 });
|
|
}
|
|
|
|
// count how many peers we're asking for `block`
|
|
size_t ActiveRequests::count(tr_block_index_t block) const
|
|
{
|
|
return std::size(impl_->blocks_[block]);
|
|
}
|
|
|
|
// count how many active block requests we have to `peer`
|
|
size_t ActiveRequests::count(tr_peer const* peer) const
|
|
{
|
|
return impl_->count(peer);
|
|
}
|
|
|
|
// return the total number of active requests
|
|
size_t ActiveRequests::size() const
|
|
{
|
|
return impl_->size();
|
|
}
|
|
|
|
// returns the active requests sent before `when`
|
|
std::vector<std::pair<tr_block_index_t, tr_peer*>> ActiveRequests::sentBefore(time_t when) const
|
|
{
|
|
auto sent_before = std::vector<std::pair<tr_block_index_t, tr_peer*>>{};
|
|
sent_before.reserve(std::size(impl_->blocks_));
|
|
|
|
for (auto& [block, peers_at] : impl_->blocks_)
|
|
{
|
|
for (auto& sent : peers_at)
|
|
{
|
|
if (sent.when < when)
|
|
{
|
|
sent_before.emplace_back(block, sent.peer);
|
|
}
|
|
}
|
|
}
|
|
|
|
return sent_before;
|
|
}
|