/* */ #include "DefaultBtRequestFactory.h" #include "LogFactory.h" #include "Logger.h" #include "Piece.h" #include "Peer.h" #include "BtContext.h" #include "PieceStorage.h" #include "BtMessageDispatcher.h" #include "BtMessageFactory.h" #include "BtMessage.h" #include "BtRegistry.h" #include "a2functional.h" #include namespace aria2 { DefaultBtRequestFactory::DefaultBtRequestFactory(): cuid(0), _logger(LogFactory::getInstance()) { LogFactory::getInstance()->debug("DefaultBtRequestFactory::instantiated"); } DefaultBtRequestFactory::~DefaultBtRequestFactory() { LogFactory::getInstance()->debug("DefaultBtRequestFactory::deleted"); } void DefaultBtRequestFactory::addTargetPiece(const PieceHandle& piece) { pieces.push_back(piece); } class AbortCompletedPieceRequest { private: WeakHandle _dispatcher; public: AbortCompletedPieceRequest(const WeakHandle& dispatcher): _dispatcher(dispatcher) {} void operator()(const SharedHandle& piece) { if(piece->pieceComplete()) { _dispatcher->doAbortOutstandingRequestAction(piece); } } }; void DefaultBtRequestFactory::removeCompletedPiece() { std::for_each(pieces.begin(), pieces.end(), AbortCompletedPieceRequest(dispatcher)); pieces.erase(std::remove_if(pieces.begin(), pieces.end(), mem_fun_sh(&Piece::pieceComplete)), pieces.end()); } void DefaultBtRequestFactory::removeTargetPiece(const PieceHandle& piece) { pieces.erase(std::remove(pieces.begin(), pieces.end(), piece), pieces.end()); dispatcher->doAbortOutstandingRequestAction(piece); pieceStorage->cancelPiece(piece); } class ProcessChokedPiece { private: SharedHandle _peer; WeakHandle _pieceStorage; public: ProcessChokedPiece(const SharedHandle& peer, const WeakHandle& pieceStorage): _peer(peer), _pieceStorage(pieceStorage) {} void operator()(const SharedHandle& piece) { if(!_peer->isInPeerAllowedIndexSet(piece->getIndex())) { _pieceStorage->cancelPiece(piece); } } }; class FindChokedPiece { private: SharedHandle _peer; public: FindChokedPiece(const SharedHandle& peer):_peer(peer) {} bool operator()(const SharedHandle& piece) { return !_peer->isInPeerAllowedIndexSet(piece->getIndex()); } }; void DefaultBtRequestFactory::doChokedAction() { std::for_each(pieces.begin(), pieces.end(), ProcessChokedPiece(peer, pieceStorage)); pieces.erase(std::remove_if(pieces.begin(), pieces.end(), FindChokedPiece(peer)), pieces.end()); } void DefaultBtRequestFactory::removeAllTargetPiece() { for(Pieces::iterator itr = pieces.begin(); itr != pieces.end(); ++itr) { dispatcher->doAbortOutstandingRequestAction(*itr); pieceStorage->cancelPiece(*itr); } pieces.clear(); } void DefaultBtRequestFactory::createRequestMessages (std::deque >& requests, size_t max) { for(Pieces::iterator itr = pieces.begin(); itr != pieces.end() && requests.size() < max; ++itr) { PieceHandle& piece = *itr; size_t blockIndex; while(requests.size() < max && piece->getMissingUnusedBlockIndex(blockIndex)) { _logger->debug("Creating RequestMessage index=%u, begin=%u, blockIndex=%u", piece->getIndex(), blockIndex*piece->getBlockLength(), blockIndex); requests.push_back(messageFactory->createRequestMessage(piece, blockIndex)); } } } void DefaultBtRequestFactory::createRequestMessagesOnEndGame (std::deque >& requests, size_t max) { for(Pieces::iterator itr = pieces.begin(); itr != pieces.end() && requests.size() < max; ++itr) { PieceHandle& piece = *itr; std::deque missingBlockIndexes; piece->getAllMissingBlockIndexes(missingBlockIndexes); std::random_shuffle(missingBlockIndexes.begin(), missingBlockIndexes.end()); for(std::deque::const_iterator bitr = missingBlockIndexes.begin(); bitr != missingBlockIndexes.end() && requests.size() < max; bitr++) { size_t blockIndex = *bitr; if(!dispatcher->isOutstandingRequest(piece->getIndex(), blockIndex)) { _logger->debug("Creating RequestMessage index=%u, begin=%u, blockIndex=%u", piece->getIndex(), blockIndex*piece->getBlockLength(), blockIndex); requests.push_back(messageFactory->createRequestMessage(piece, blockIndex)); } } } } class CountMissingBlock { private: size_t _numMissingBlock; public: CountMissingBlock():_numMissingBlock(0) {} size_t getNumMissingBlock() { return _numMissingBlock; } void operator()(const SharedHandle& piece) { _numMissingBlock += piece->countMissingBlock(); } }; size_t DefaultBtRequestFactory::countMissingBlock() { return std::for_each(pieces.begin(), pieces.end(), CountMissingBlock()).getNumMissingBlock(); } std::deque >& DefaultBtRequestFactory::getTargetPieces() { return pieces; } void DefaultBtRequestFactory::setBtContext(const SharedHandle& btContext) { this->btContext = btContext; this->pieceStorage = PIECE_STORAGE(btContext); } void DefaultBtRequestFactory::setPeer(const SharedHandle& peer) { this->peer = peer; } void DefaultBtRequestFactory::setBtMessageDispatcher(const WeakHandle& dispatcher) { this->dispatcher = dispatcher; } void DefaultBtRequestFactory::setBtMessageFactory(const WeakHandle& factory) { this->messageFactory = factory; } } // namespace aria2