kde-extraapps/kget/transfer-plugins/bittorrent/btchunkselector.cpp
2015-02-27 11:02:43 +00:00

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"