forked from premiere/premiere-libtorrent
224 lines
5.9 KiB
C++
224 lines
5.9 KiB
C++
/*
|
|
|
|
Copyright (c) 2009, Arvid Norberg
|
|
All rights reserved.
|
|
|
|
Redistribution and use in source and binary forms, with or without
|
|
modification, are permitted provided that the following conditions
|
|
are met:
|
|
|
|
* Redistributions of source code must retain the above copyright
|
|
notice, this list of conditions and the following disclaimer.
|
|
* Redistributions in binary form must reproduce the above copyright
|
|
notice, this list of conditions and the following disclaimer in
|
|
the documentation and/or other materials provided with the distribution.
|
|
* Neither the name of the author nor the names of its
|
|
contributors may be used to endorse or promote products derived
|
|
from this software without specific prior written permission.
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
|
|
LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*/
|
|
|
|
#include "libtorrent/bandwidth_manager.hpp"
|
|
#include "libtorrent/time.hpp"
|
|
|
|
namespace libtorrent
|
|
{
|
|
|
|
bandwidth_manager::bandwidth_manager(int channel
|
|
#ifdef TORRENT_VERBOSE_BANDWIDTH_LIMIT
|
|
, bool log = false
|
|
#endif
|
|
)
|
|
: m_queued_bytes(0)
|
|
, m_channel(channel)
|
|
, m_abort(false)
|
|
{
|
|
#ifdef TORRENT_VERBOSE_BANDWIDTH_LIMIT
|
|
if (log)
|
|
m_log.open("bandwidth_limiter.log", std::ios::trunc);
|
|
m_start = time_now();
|
|
#endif
|
|
}
|
|
|
|
void bandwidth_manager::close()
|
|
{
|
|
m_abort = true;
|
|
m_queue.clear();
|
|
m_queued_bytes = 0;
|
|
}
|
|
|
|
#if defined TORRENT_DEBUG || TORRENT_RELEASE_ASSERTS
|
|
bool bandwidth_manager::is_queued(bandwidth_socket const* peer) const
|
|
{
|
|
for (queue_t::const_iterator i = m_queue.begin()
|
|
, end(m_queue.end()); i != end; ++i)
|
|
{
|
|
if (i->peer.get() == peer) return true;
|
|
}
|
|
return false;
|
|
}
|
|
#endif
|
|
|
|
int bandwidth_manager::queue_size() const
|
|
{
|
|
return m_queue.size();
|
|
}
|
|
|
|
int bandwidth_manager::queued_bytes() const
|
|
{
|
|
return m_queued_bytes;
|
|
}
|
|
|
|
// non prioritized means that, if there's a line for bandwidth,
|
|
// others will cut in front of the non-prioritized peers.
|
|
// this is used by web seeds
|
|
int bandwidth_manager::request_bandwidth(boost::intrusive_ptr<bandwidth_socket> const& peer
|
|
, int blk, int priority
|
|
, bandwidth_channel* chan1
|
|
, bandwidth_channel* chan2
|
|
, bandwidth_channel* chan3
|
|
, bandwidth_channel* chan4
|
|
, bandwidth_channel* chan5
|
|
)
|
|
{
|
|
INVARIANT_CHECK;
|
|
if (m_abort) return 0;
|
|
|
|
TORRENT_ASSERT(blk > 0);
|
|
TORRENT_ASSERT(priority > 0);
|
|
TORRENT_ASSERT(!is_queued(peer.get()));
|
|
|
|
bw_request bwr(peer, blk, priority);
|
|
int i = 0;
|
|
if (chan1 && chan1->throttle() > 0) bwr.channel[i++] = chan1;
|
|
if (chan2 && chan2->throttle() > 0) bwr.channel[i++] = chan2;
|
|
if (chan3 && chan3->throttle() > 0) bwr.channel[i++] = chan3;
|
|
if (chan4 && chan4->throttle() > 0) bwr.channel[i++] = chan4;
|
|
if (chan5 && chan5->throttle() > 0) bwr.channel[i++] = chan5;
|
|
if (i == 0)
|
|
{
|
|
// the connection is not rate limited by any of its
|
|
// bandwidth channels, or it doesn't belong to any
|
|
// channels. There's no point in adding it to
|
|
// the queue, just satisfy the request immediately
|
|
return blk;
|
|
}
|
|
m_queued_bytes += blk;
|
|
m_queue.push_back(bwr);
|
|
return 0;
|
|
}
|
|
|
|
#ifdef TORRENT_DEBUG
|
|
void bandwidth_manager::check_invariant() const
|
|
{
|
|
int queued = 0;
|
|
for (queue_t::const_iterator i = m_queue.begin()
|
|
, end(m_queue.end()); i != end; ++i)
|
|
{
|
|
queued += i->request_size - i->assigned;
|
|
}
|
|
TORRENT_ASSERT(queued == m_queued_bytes);
|
|
}
|
|
#endif
|
|
|
|
void bandwidth_manager::update_quotas(time_duration const& dt)
|
|
{
|
|
if (m_abort) return;
|
|
if (m_queue.empty()) return;
|
|
|
|
INVARIANT_CHECK;
|
|
|
|
int dt_milliseconds = total_milliseconds(dt);
|
|
if (dt_milliseconds > 3000) dt_milliseconds = 3000;
|
|
|
|
// for each bandwidth channel, call update_quota(dt)
|
|
|
|
std::vector<bandwidth_channel*> channels;
|
|
|
|
for (queue_t::iterator i = m_queue.begin();
|
|
i != m_queue.end();)
|
|
{
|
|
if (i->peer->is_disconnecting())
|
|
{
|
|
m_queued_bytes -= i->request_size - i->assigned;
|
|
|
|
// return all assigned quota to all the
|
|
// bandwidth channels this peer belongs to
|
|
for (int j = 0; j < 5 && i->channel[j]; ++j)
|
|
{
|
|
bandwidth_channel* bwc = i->channel[j];
|
|
bwc->return_quota(i->assigned);
|
|
}
|
|
|
|
i = m_queue.erase(i);
|
|
continue;
|
|
}
|
|
for (int j = 0; j < 5 && i->channel[j]; ++j)
|
|
{
|
|
bandwidth_channel* bwc = i->channel[j];
|
|
bwc->tmp = 0;
|
|
}
|
|
++i;
|
|
}
|
|
|
|
for (queue_t::iterator i = m_queue.begin()
|
|
, end(m_queue.end()); i != end; ++i)
|
|
{
|
|
for (int j = 0; j < 5 && i->channel[j]; ++j)
|
|
{
|
|
bandwidth_channel* bwc = i->channel[j];
|
|
if (bwc->tmp == 0) channels.push_back(bwc);
|
|
TORRENT_ASSERT(INT_MAX - bwc->tmp > i->priority);
|
|
bwc->tmp += i->priority;
|
|
}
|
|
}
|
|
|
|
for (std::vector<bandwidth_channel*>::iterator i = channels.begin()
|
|
, end(channels.end()); i != end; ++i)
|
|
{
|
|
(*i)->update_quota(dt_milliseconds);
|
|
}
|
|
|
|
queue_t tm;
|
|
|
|
for (queue_t::iterator i = m_queue.begin();
|
|
i != m_queue.end();)
|
|
{
|
|
int a = i->assign_bandwidth();
|
|
if (i->assigned == i->request_size
|
|
|| (i->ttl <= 0 && i->assigned > 0))
|
|
{
|
|
a += i->request_size - i->assigned;
|
|
TORRENT_ASSERT(i->assigned <= i->request_size);
|
|
tm.push_back(*i);
|
|
i = m_queue.erase(i);
|
|
}
|
|
else
|
|
{
|
|
++i;
|
|
}
|
|
m_queued_bytes -= a;
|
|
}
|
|
|
|
while (!tm.empty())
|
|
{
|
|
bw_request& bwr = tm.back();
|
|
bwr.peer->assign_bandwidth(m_channel, bwr.assigned);
|
|
tm.pop_back();
|
|
}
|
|
}
|
|
}
|
|
|