mirror of
https://bitbucket.org/smil3y/kde-extraapps.git
synced 2025-02-24 19:02:53 +00:00
301 lines
8.3 KiB
C++
301 lines
8.3 KiB
C++
/* This file is part of the KDE project
|
|
|
|
Copyright (C) 2008 Lukas Appelhans <l.appelhans@gmx.de>
|
|
Copyright (C) 2005 Joris Guisson <joris.guisson@gmail.com>
|
|
|
|
This program is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU General Public
|
|
License as published by the Free Software Foundation; either
|
|
version 2 of the License, or (at your option) any later version.
|
|
*/
|
|
|
|
#include "btchunkselector.h"
|
|
|
|
#include <stdlib.h>
|
|
#include <vector>
|
|
#include <algorithm>
|
|
#include <util/log.h>
|
|
#include <util/bitset.h>
|
|
#include <peer/chunkcounter.h>
|
|
#include <diskio/chunkmanager.h>
|
|
#include <interfaces/piecedownloader.h>
|
|
#include <peer/peer.h>
|
|
#include <peer/peermanager.h>
|
|
#include <download/downloader.h>
|
|
|
|
using namespace bt;
|
|
|
|
struct RareCmp
|
|
{
|
|
ChunkManager & cman;
|
|
ChunkCounter & cc;
|
|
bool warmup;
|
|
|
|
RareCmp(ChunkManager & cman,ChunkCounter & cc,bool warmup) : cman(cman),cc(cc),warmup(warmup) {}
|
|
|
|
bool operator()(Uint32 a,Uint32 b)
|
|
{
|
|
if (a >= cman.getNumChunks() || b >= cman.getNumChunks())
|
|
return false;
|
|
// the sorting is done on two criteria, priority and rareness
|
|
Priority pa = cman.getChunk(a)->getPriority();
|
|
Priority pb = cman.getChunk(b)->getPriority();
|
|
if (pa == pb)
|
|
return normalCmp(a,b); // if both have same priority compare on rareness
|
|
else if (pa > pb) // pa has priority over pb, so select pa
|
|
return true;
|
|
else // pb has priority over pa, so select pb
|
|
return false;
|
|
}
|
|
|
|
bool normalCmp(Uint32 a,Uint32 b)
|
|
{
|
|
// during warmup mode choose most common chunks
|
|
if (!warmup)
|
|
return cc.get(a) < cc.get(b);
|
|
else
|
|
return cc.get(a) > cc.get(b);
|
|
}
|
|
};
|
|
|
|
BTChunkSelector::BTChunkSelector(ChunkManager & cman,Downloader & downer,PeerManager & pman)
|
|
: ChunkSelectorInterface(cman,downer,pman)
|
|
{
|
|
std::vector<Uint32> tmp;
|
|
for (Uint32 i = 0;i < cman.getNumChunks();++i)
|
|
{
|
|
if (!cman.getBitSet().get(i))
|
|
{
|
|
tmp.push_back(i);
|
|
}
|
|
}
|
|
std::random_shuffle(tmp.begin(),tmp.end());
|
|
// std::list does not support random_shuffle so we use a vector as a temporary storage
|
|
// for the random_shuffle
|
|
chunks.insert(chunks.begin(),tmp.begin(),tmp.end());
|
|
sort_timer.update();
|
|
}
|
|
|
|
|
|
BTChunkSelector::~BTChunkSelector()
|
|
{
|
|
}
|
|
|
|
Uint32 BTChunkSelector::leastPeers(const std::list<Uint32> & lp)
|
|
{
|
|
Uint32 sel = lp.front();
|
|
Uint32 cnt = downer.numDownloadersForChunk(sel);
|
|
std::list<Uint32>::const_iterator itEnd = lp.end();
|
|
for (std::list<Uint32>::const_iterator i = lp.begin();i != itEnd;++i)
|
|
{
|
|
Uint32 cnt_i = downer.numDownloadersForChunk(*i);
|
|
if (cnt_i < cnt)
|
|
{
|
|
sel = *i;
|
|
cnt = cnt_i;
|
|
}
|
|
}
|
|
return sel;
|
|
}
|
|
|
|
bool BTChunkSelector::select(PieceDownloader* pd,Uint32 & chunk)
|
|
{
|
|
const BitSet & bs = cman.getBitSet();
|
|
|
|
std::list<Uint32> preview;
|
|
std::list<Uint32> normal;
|
|
std::list<Uint32> first;
|
|
Uint32 sel = cman.getNumChunks() + 1;
|
|
|
|
// sort the chunks every 2 seconds
|
|
if (sort_timer.getElapsedSinceUpdate() > 2000)
|
|
{
|
|
bool warmup = cman.getNumChunks() - cman.chunksLeft() <= 4;
|
|
chunks.sort(RareCmp(cman,pman.getChunkCounter(),warmup));
|
|
sort_timer.update();
|
|
}
|
|
|
|
std::list<Uint32>::iterator itr;
|
|
std::list<Uint32>::iterator itrEnd = chunks.end();
|
|
for (itr = chunks.begin(); itr != itrEnd; )
|
|
{
|
|
const Uint32 i = *itr;
|
|
Chunk* c = cman.getChunk(i);
|
|
|
|
// if we have the chunk remove it from the list
|
|
if (bs.get(i))
|
|
{
|
|
itr = chunks.erase(itr);
|
|
}
|
|
else
|
|
{
|
|
// pd has to have the selected chunk and it needs to be not excluded
|
|
if (pd->hasChunk(i) && !c->isExcluded() && !c->isExcludedForDownloading())
|
|
{
|
|
if (!downer.areWeDownloading(i))
|
|
{
|
|
// we have a chunk
|
|
sel = i;
|
|
break;
|
|
}
|
|
|
|
switch (cman.getChunk(i)->getPriority())
|
|
{
|
|
case PREVIEW_PRIORITY:
|
|
preview.push_back(i);
|
|
break;
|
|
case FIRST_PRIORITY:
|
|
first.push_back(i);
|
|
break;
|
|
case NORMAL_PRIORITY:
|
|
normal.push_back(i);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
++itr;
|
|
}
|
|
}
|
|
|
|
if (sel >= cman.getNumChunks())
|
|
return false;
|
|
|
|
// we have found one, now try to see if we cannot assign this PieceDownloader to a higher priority chunk
|
|
switch (cman.getChunk(sel)->getPriority())
|
|
{
|
|
case PREVIEW_PRIORITY:
|
|
chunk = sel;
|
|
return true;
|
|
case FIRST_PRIORITY:
|
|
if (preview.size() > 0)
|
|
{
|
|
chunk = leastPeers(preview);
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
chunk = sel;
|
|
return true;
|
|
}
|
|
break;
|
|
case NORMAL_PRIORITY:
|
|
if (preview.size() > 0)
|
|
{
|
|
chunk = leastPeers(preview);
|
|
return true;
|
|
}
|
|
else if (first.size() > 0)
|
|
{
|
|
chunk = leastPeers(first);
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
chunk = sel;
|
|
return true;
|
|
}
|
|
break;
|
|
case LAST_PRIORITY:
|
|
if (preview.size() > 0)
|
|
{
|
|
chunk = leastPeers(preview);
|
|
return true;
|
|
}
|
|
else if (first.size() > 0)
|
|
{
|
|
chunk = leastPeers(first);
|
|
return true;
|
|
}
|
|
else if (normal.size() > 0)
|
|
{
|
|
chunk = leastPeers(normal);
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
chunk = sel;
|
|
return true;
|
|
}
|
|
break;
|
|
default:
|
|
chunk = sel;
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void BTChunkSelector::dataChecked(const BitSet & ok_chunks)
|
|
{
|
|
for (Uint32 i = 0;i < ok_chunks.getNumBits();++i)
|
|
{
|
|
bool in_chunks = std::find(chunks.begin(),chunks.end(),i) != chunks.end();
|
|
if (in_chunks && ok_chunks.get(i))
|
|
{
|
|
// if we have the chunk, remove it from the chunks list
|
|
chunks.remove(i);
|
|
}
|
|
else if (!in_chunks && !ok_chunks.get(i))
|
|
{
|
|
// if we don't have the chunk, add it to the list if it wasn't allrready in there
|
|
chunks.push_back(i);
|
|
}
|
|
}
|
|
}
|
|
|
|
void BTChunkSelector::reincluded(Uint32 from, Uint32 to)
|
|
{
|
|
// lets do a safety check first
|
|
if (from >= cman.getNumChunks() || to >= cman.getNumChunks())
|
|
{
|
|
Out(SYS_DIO|LOG_NOTICE) << "Internal error in chunkselector" << endl;
|
|
return;
|
|
}
|
|
|
|
for (Uint32 i = from;i <= to;++i)
|
|
{
|
|
bool in_chunks = std::find(chunks.begin(),chunks.end(),i) != chunks.end();
|
|
if (!in_chunks && cman.getChunk(i)->getStatus() != Chunk::ON_DISK)
|
|
{
|
|
// Out(SYS_DIO|LOG_DEBUG) << "BTChunkSelector::reIncluded " << i << endl;
|
|
chunks.push_back(i);
|
|
}
|
|
}
|
|
}
|
|
|
|
void BTChunkSelector::reinsert(Uint32 chunk)
|
|
{
|
|
bool in_chunks = std::find(chunks.begin(),chunks.end(),chunk) != chunks.end();
|
|
if (!in_chunks)
|
|
chunks.push_back(chunk);
|
|
}
|
|
|
|
void BTChunkSelector::excludeAll()
|
|
{
|
|
chunks.clear();
|
|
}
|
|
|
|
void BTChunkSelector::exclude(Uint32 chunk)
|
|
{
|
|
bool in_chunks = std::find(chunks.begin(),chunks.end(),chunk) != chunks.end();
|
|
if (in_chunks)
|
|
chunks.remove(chunk);
|
|
}
|
|
|
|
BTChunkSelectorFactory::BTChunkSelectorFactory()
|
|
{
|
|
}
|
|
|
|
BTChunkSelectorFactory::~BTChunkSelectorFactory()
|
|
{
|
|
}
|
|
|
|
bt::ChunkSelectorInterface* BTChunkSelectorFactory::createChunkSelector(bt::ChunkManager & cman, bt::Downloader & downer, bt::PeerManager & pman)
|
|
{
|
|
BTChunkSelector *selector = new BTChunkSelector(cman, downer, pman);
|
|
emit selectorAdded(selector);
|
|
return selector;
|
|
}
|
|
|
|
#include "moc_btchunkselector.cpp"
|