| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287 |
- /* <!-- copyright */
- /*
- * aria2 - The high speed download utility
- *
- * Copyright (C) 2006 Tatsuhiro Tsujikawa
- *
- * 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.
- *
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License
- * along with this program; if not, write to the Free Software
- * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
- *
- * In addition, as a special exception, the copyright holders give
- * permission to link the code of portions of this program with the
- * OpenSSL library under certain conditions as described in each
- * individual source file, and distribute linked combinations
- * including the two.
- * You must obey the GNU General Public License in all respects
- * for all of the code used other than OpenSSL. If you modify
- * file(s) with this exception, you may extend this exception to your
- * version of the file(s), but you are not obligated to do so. If you
- * do not wish to do so, delete this exception statement from your
- * version. If you delete this exception statement from all source
- * files in the program, then also delete it here.
- */
- /* copyright --> */
- #include "DefaultBtRequestFactory.h"
- #include <algorithm>
- #include "LogFactory.h"
- #include "Logger.h"
- #include "Piece.h"
- #include "Peer.h"
- #include "PieceStorage.h"
- #include "BtMessageDispatcher.h"
- #include "BtMessageFactory.h"
- #include "BtMessage.h"
- #include "a2functional.h"
- #include "SimpleRandomizer.h"
- #include "array_fun.h"
- #include "fmt.h"
- namespace aria2 {
- DefaultBtRequestFactory::DefaultBtRequestFactory()
- : pieceStorage_(0),
- dispatcher_(0),
- messageFactory_(0),
- cuid_(0)
- {}
- DefaultBtRequestFactory::~DefaultBtRequestFactory() {}
- void DefaultBtRequestFactory::addTargetPiece(const std::shared_ptr<Piece>& piece)
- {
- pieces_.push_back(piece);
- }
- namespace {
- class AbortCompletedPieceRequest
- {
- private:
- BtMessageDispatcher* dispatcher_;
- public:
- AbortCompletedPieceRequest(BtMessageDispatcher* dispatcher):
- dispatcher_(dispatcher) {}
- void operator()(const std::shared_ptr<Piece>& piece)
- {
- if(piece->pieceComplete()) {
- dispatcher_->doAbortOutstandingRequestAction(piece);
- }
- }
- };
- } // namespace
- void DefaultBtRequestFactory::removeCompletedPiece() {
- std::for_each(pieces_.begin(), pieces_.end(),
- AbortCompletedPieceRequest(dispatcher_));
- pieces_.erase(std::remove_if(pieces_.begin(), pieces_.end(),
- std::mem_fn(&Piece::pieceComplete)),
- pieces_.end());
- }
- void DefaultBtRequestFactory::removeTargetPiece
- (const std::shared_ptr<Piece>& piece)
- {
- pieces_.erase(std::remove_if(pieces_.begin(), pieces_.end(),
- derefEqual(piece)),
- pieces_.end());
- dispatcher_->doAbortOutstandingRequestAction(piece);
- pieceStorage_->cancelPiece(piece, cuid_);
- }
- namespace {
- class ProcessChokedPiece {
- private:
- std::shared_ptr<Peer> peer_;
- PieceStorage* pieceStorage_;
- cuid_t cuid_;
- public:
- ProcessChokedPiece(const std::shared_ptr<Peer>& peer,
- PieceStorage* pieceStorage,
- cuid_t cuid):
- peer_(peer),
- pieceStorage_(pieceStorage),
- cuid_(cuid)
- {}
- void operator()(const std::shared_ptr<Piece>& piece)
- {
- if(!peer_->isInPeerAllowedIndexSet(piece->getIndex())) {
- pieceStorage_->cancelPiece(piece, cuid_);
- }
- }
- };
- } // namespace
- namespace {
- class FindChokedPiece {
- private:
- std::shared_ptr<Peer> peer_;
- public:
- FindChokedPiece(const std::shared_ptr<Peer>& peer):peer_(peer) {}
- bool operator()(const std::shared_ptr<Piece>& piece)
- {
- return !peer_->isInPeerAllowedIndexSet(piece->getIndex());
- }
- };
- } // namespace
- void DefaultBtRequestFactory::doChokedAction()
- {
- std::for_each(pieces_.begin(), pieces_.end(),
- ProcessChokedPiece(peer_, pieceStorage_, cuid_));
- pieces_.erase(std::remove_if(pieces_.begin(), pieces_.end(),
- FindChokedPiece(peer_)),
- pieces_.end());
- }
- void DefaultBtRequestFactory::removeAllTargetPiece() {
- for(std::deque<std::shared_ptr<Piece> >::iterator itr = pieces_.begin(),
- eoi = pieces_.end(); itr != eoi; ++itr) {
- dispatcher_->doAbortOutstandingRequestAction(*itr);
- pieceStorage_->cancelPiece(*itr, cuid_);
- }
- pieces_.clear();
- }
- void DefaultBtRequestFactory::createRequestMessages
- (std::vector<std::shared_ptr<BtMessage> >& requests, size_t max)
- {
- if(requests.size() >= max) {
- return;
- }
- size_t getnum = max-requests.size();
- std::vector<size_t> blockIndexes;
- blockIndexes.reserve(getnum);
- for(std::deque<std::shared_ptr<Piece> >::iterator itr = pieces_.begin(),
- eoi = pieces_.end(); itr != eoi && getnum; ++itr) {
- std::shared_ptr<Piece>& piece = *itr;
- if(piece->getMissingUnusedBlockIndex(blockIndexes, getnum)) {
- getnum -= blockIndexes.size();
- for(std::vector<size_t>::const_iterator i = blockIndexes.begin(),
- eoi2 = blockIndexes.end(); i != eoi2; ++i) {
- A2_LOG_DEBUG
- (fmt("Creating RequestMessage index=%lu, begin=%u,"
- " blockIndex=%lu",
- static_cast<unsigned long>(piece->getIndex()),
- static_cast<unsigned int>((*i)*piece->getBlockLength()),
- static_cast<unsigned long>(*i)));
- requests.push_back
- (messageFactory_->createRequestMessage(piece, *i));
- }
- blockIndexes.clear();
- }
- }
- }
- void DefaultBtRequestFactory::createRequestMessagesOnEndGame
- (std::vector<std::shared_ptr<BtMessage> >& requests, size_t max)
- {
- for(std::deque<std::shared_ptr<Piece> >::iterator itr = pieces_.begin(),
- eoi = pieces_.end(); itr != eoi && requests.size() < max; ++itr) {
- std::shared_ptr<Piece>& piece = *itr;
- const size_t mislen = piece->getBitfieldLength();
- array_ptr<unsigned char> misbitfield(new unsigned char[mislen]);
- piece->getAllMissingBlockIndexes(misbitfield, mislen);
- std::vector<size_t> missingBlockIndexes;
- size_t blockIndex = 0;
- for(size_t i = 0; i < mislen; ++i) {
- unsigned char bits = misbitfield[i];
- unsigned char mask = 128;
- for(size_t bi = 0; bi < 8; ++bi, mask >>= 1, ++blockIndex) {
- if(bits & mask) {
- missingBlockIndexes.push_back(blockIndex);
- }
- }
- }
- std::random_shuffle(missingBlockIndexes.begin(), missingBlockIndexes.end(),
- *SimpleRandomizer::getInstance());
- for(std::vector<size_t>::const_iterator bitr = missingBlockIndexes.begin(),
- eoi2 = missingBlockIndexes.end();
- bitr != eoi2 && requests.size() < max; ++bitr) {
- const size_t& blockIndex = *bitr;
- if(!dispatcher_->isOutstandingRequest(piece->getIndex(),
- blockIndex)) {
- A2_LOG_DEBUG
- (fmt("Creating RequestMessage index=%lu, begin=%u,"
- " blockIndex=%lu",
- static_cast<unsigned long>(piece->getIndex()),
- static_cast<unsigned int>(blockIndex*piece->getBlockLength()),
- static_cast<unsigned long>(blockIndex)));
- requests.push_back(messageFactory_->createRequestMessage
- (piece, blockIndex));
- }
- }
- }
- }
- namespace {
- class CountMissingBlock
- {
- private:
- size_t numMissingBlock_;
- public:
- CountMissingBlock():numMissingBlock_(0) {}
- size_t getNumMissingBlock()
- {
- return numMissingBlock_;
- }
- void operator()(const std::shared_ptr<Piece>& piece)
- {
- numMissingBlock_ += piece->countMissingBlock();
- }
- };
- } // namespace
- size_t DefaultBtRequestFactory::countMissingBlock()
- {
- return std::for_each(pieces_.begin(), pieces_.end(),
- CountMissingBlock()).getNumMissingBlock();
- }
- void DefaultBtRequestFactory::getTargetPieceIndexes
- (std::vector<size_t>& indexes) const
- {
- std::transform(pieces_.begin(), pieces_.end(), std::back_inserter(indexes),
- std::mem_fn(&Piece::getIndex));
- }
- void DefaultBtRequestFactory::setPieceStorage(PieceStorage* pieceStorage)
- {
- pieceStorage_ = pieceStorage;
- }
- void DefaultBtRequestFactory::setPeer(const std::shared_ptr<Peer>& peer)
- {
- peer_ = peer;
- }
- void DefaultBtRequestFactory::setBtMessageDispatcher
- (BtMessageDispatcher* dispatcher)
- {
- dispatcher_ = dispatcher;
- }
- void DefaultBtRequestFactory::setBtMessageFactory(BtMessageFactory* factory)
- {
- messageFactory_ = factory;
- }
- } // namespace aria2
|