| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871 | /* <!-- 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 "DefaultPieceStorage.h"#include <numeric>#include <algorithm>#include "DownloadContext.h"#include "Piece.h"#include "Peer.h"#include "LogFactory.h"#include "Logger.h"#include "prefs.h"#include "DirectDiskAdaptor.h"#include "MultiDiskAdaptor.h"#include "DiskWriter.h"#include "BitfieldMan.h"#include "message.h"#include "DefaultDiskWriterFactory.h"#include "FileEntry.h"#include "DlAbortEx.h"#include "util.h"#include "a2functional.h"#include "Option.h"#include "fmt.h"#include "RarestPieceSelector.h"#include "DefaultStreamPieceSelector.h"#include "InorderStreamPieceSelector.h"#include "GeomStreamPieceSelector.h"#include "array_fun.h"#include "PieceStatMan.h"#include "wallclock.h"#include "bitfield.h"#include "SingletonHolder.h"#include "Notifier.h"#include "WrDiskCache.h"#include "RequestGroup.h"#ifdef ENABLE_BITTORRENT# include "bittorrent_helper.h"#endif // ENABLE_BITTORRENTnamespace aria2 {DefaultPieceStorage::DefaultPieceStorage(const std::shared_ptr<DownloadContext>& downloadContext, const Option* option) : downloadContext_(downloadContext),   bitfieldMan_(new BitfieldMan(downloadContext->getPieceLength(),                                downloadContext->getTotalLength())),   diskWriterFactory_(new DefaultDiskWriterFactory()),   endGame_(false),   endGamePieceNum_(END_GAME_PIECE_NUM),   option_(option),   pieceStatMan_(new PieceStatMan(downloadContext->getNumPieces(), true)),   pieceSelector_(new RarestPieceSelector(pieceStatMan_)),   wrDiskCache_(0){  const std::string& pieceSelectorOpt =    option_->get(PREF_STREAM_PIECE_SELECTOR);  if(pieceSelectorOpt.empty() || pieceSelectorOpt == A2_V_DEFAULT) {    streamPieceSelector_.reset(new DefaultStreamPieceSelector(bitfieldMan_));  } else if(pieceSelectorOpt == V_INORDER) {    streamPieceSelector_.reset(new InorderStreamPieceSelector(bitfieldMan_));  } else if(pieceSelectorOpt == A2_V_GEOM) {    streamPieceSelector_.reset(new GeomStreamPieceSelector(bitfieldMan_, 1.5));  }}DefaultPieceStorage::~DefaultPieceStorage(){  delete bitfieldMan_;}std::shared_ptr<Piece> DefaultPieceStorage::checkOutPiece(size_t index, cuid_t cuid){  bitfieldMan_->setUseBit(index);  std::shared_ptr<Piece> piece = findUsedPiece(index);  if(!piece) {    piece.reset(new Piece(index, bitfieldMan_->getBlockLength(index)));#ifdef ENABLE_MESSAGE_DIGEST    piece->setHashType(downloadContext_->getPieceHashType());#endif // ENABLE_MESSAGE_DIGEST    addUsedPiece(piece);  }  piece->addUser(cuid);  RequestGroup* group = downloadContext_->getOwnerRequestGroup();  if((!group || !group->inMemoryDownload()) &&     wrDiskCache_ && !piece->getWrDiskCacheEntry()) {    // So, we rely on the fact that diskAdaptor_ is not reinitialized    // in the session.    piece->initWrCache(wrDiskCache_, diskAdaptor_);  }  return piece;}/** * Newly instantiated piece is not added to usedPieces. * Because it is waste of memory and there is no chance to use them later. */std::shared_ptr<Piece> DefaultPieceStorage::getPiece(size_t index){  std::shared_ptr<Piece> piece;  if(index <= bitfieldMan_->getMaxIndex()) {    piece = findUsedPiece(index);    if(!piece) {      piece.reset(new Piece(index, bitfieldMan_->getBlockLength(index)));      if(hasPiece(index)) {        piece->setAllBlock();      }    }  }  return piece;}void DefaultPieceStorage::addUsedPiece(const std::shared_ptr<Piece>& piece){  usedPieces_.insert(piece);  A2_LOG_DEBUG(fmt("usedPieces_.size()=%lu",                   static_cast<unsigned long>(usedPieces_.size())));}std::shared_ptr<Piece> DefaultPieceStorage::findUsedPiece(size_t index) const{  std::shared_ptr<Piece> p(new Piece());  p->setIndex(index);  UsedPieceSet::iterator i = usedPieces_.find(p);  if(i == usedPieces_.end()) {    p.reset();    return p;  } else {    return *i;  }}#ifdef ENABLE_BITTORRENTbool DefaultPieceStorage::hasMissingPiece(const std::shared_ptr<Peer>& peer){  return bitfieldMan_->hasMissingPiece(peer->getBitfield(),                                       peer->getBitfieldLength());}void DefaultPieceStorage::getMissingPiece(std::vector<std::shared_ptr<Piece> >& pieces, size_t minMissingBlocks, const unsigned char* bitfield, size_t length, cuid_t cuid){  const size_t mislen = bitfieldMan_->getBitfieldLength();  array_ptr<unsigned char> misbitfield(new unsigned char[mislen]);  size_t blocks = bitfieldMan_->countBlock();  size_t misBlock = 0;  if(isEndGame()) {    bool r = bitfieldMan_->getAllMissingIndexes      (misbitfield, mislen, bitfield, length);    if(!r) {      return;    }    std::vector<size_t> indexes;    for(size_t i = 0; i < blocks; ++i) {      if(bitfield::test(misbitfield, blocks, i)) {        indexes.push_back(i);      }    }    std::random_shuffle(indexes.begin(), indexes.end());    for(std::vector<size_t>::const_iterator i = indexes.begin(),          eoi = indexes.end(); i != eoi && misBlock < minMissingBlocks; ++i) {      std::shared_ptr<Piece> piece = checkOutPiece(*i, cuid);      if(piece->getUsedBySegment()) {        // We don't share piece downloaded via HTTP/FTP        piece->removeUser(cuid);      } else {        pieces.push_back(piece);        misBlock += piece->countMissingBlock();      }    }  } else {    bool r = bitfieldMan_->getAllMissingUnusedIndexes      (misbitfield, mislen, bitfield, length);    if(!r) {      return;    }    while(misBlock < minMissingBlocks) {      size_t index;      if(pieceSelector_->select(index, misbitfield, blocks)) {        pieces.push_back(checkOutPiece(index, cuid));        bitfield::flipBit(misbitfield, blocks, index);        misBlock += pieces.back()->countMissingBlock();      } else {        break;      }    }  }}namespace {void unsetExcludedIndexes(BitfieldMan& bitfield,                          const std::vector<size_t>& excludedIndexes){  using namespace std::placeholders;  std::for_each(excludedIndexes.begin(), excludedIndexes.end(),                std::bind(&BitfieldMan::unsetBit, &bitfield, _1));}} // namespacevoid DefaultPieceStorage::createFastIndexBitfield(BitfieldMan& bitfield, const std::shared_ptr<Peer>& peer){  for(std::set<size_t>::const_iterator itr =        peer->getPeerAllowedIndexSet().begin(),        eoi = peer->getPeerAllowedIndexSet().end(); itr != eoi; ++itr) {    if(!bitfieldMan_->isBitSet(*itr) && peer->hasPiece(*itr)) {      bitfield.setBit(*itr);    }  }}void DefaultPieceStorage::getMissingPiece(std::vector<std::shared_ptr<Piece> >& pieces, size_t minMissingBlocks, const std::shared_ptr<Peer>& peer, cuid_t cuid){  getMissingPiece(pieces, minMissingBlocks,                  peer->getBitfield(), peer->getBitfieldLength(),                  cuid);}void DefaultPieceStorage::getMissingPiece(std::vector<std::shared_ptr<Piece> >& pieces, size_t minMissingBlocks, const std::shared_ptr<Peer>& peer, const std::vector<size_t>& excludedIndexes, cuid_t cuid){  BitfieldMan tempBitfield(bitfieldMan_->getBlockLength(),                           bitfieldMan_->getTotalLength());  tempBitfield.setBitfield(peer->getBitfield(), peer->getBitfieldLength());  unsetExcludedIndexes(tempBitfield, excludedIndexes);  getMissingPiece(pieces, minMissingBlocks,                  tempBitfield.getBitfield(), tempBitfield.getBitfieldLength(),                  cuid);}void DefaultPieceStorage::getMissingFastPiece(std::vector<std::shared_ptr<Piece> >& pieces, size_t minMissingBlocks, const std::shared_ptr<Peer>& peer, cuid_t cuid){  if(peer->isFastExtensionEnabled() && peer->countPeerAllowedIndexSet() > 0) {    BitfieldMan tempBitfield(bitfieldMan_->getBlockLength(),                             bitfieldMan_->getTotalLength());    createFastIndexBitfield(tempBitfield, peer);    getMissingPiece(pieces, minMissingBlocks,                    tempBitfield.getBitfield(),                    tempBitfield.getBitfieldLength(),                    cuid);  }}void DefaultPieceStorage::getMissingFastPiece(std::vector<std::shared_ptr<Piece> >& pieces, size_t minMissingBlocks, const std::shared_ptr<Peer>& peer, const std::vector<size_t>& excludedIndexes, cuid_t cuid){  if(peer->isFastExtensionEnabled() && peer->countPeerAllowedIndexSet() > 0) {    BitfieldMan tempBitfield(bitfieldMan_->getBlockLength(),                             bitfieldMan_->getTotalLength());    createFastIndexBitfield(tempBitfield, peer);    unsetExcludedIndexes(tempBitfield, excludedIndexes);    getMissingPiece(pieces, minMissingBlocks,                    tempBitfield.getBitfield(),                    tempBitfield.getBitfieldLength(),                    cuid);  }}std::shared_ptr<Piece>DefaultPieceStorage::getMissingPiece(const std::shared_ptr<Peer>& peer, cuid_t cuid){  std::vector<std::shared_ptr<Piece> > pieces;  getMissingPiece(pieces, 1, peer, cuid);  if(pieces.empty()) {    return std::shared_ptr<Piece>();  } else {    return pieces.front();  }}std::shared_ptr<Piece> DefaultPieceStorage::getMissingPiece(const std::shared_ptr<Peer>& peer, const std::vector<size_t>& excludedIndexes, cuid_t cuid){  std::vector<std::shared_ptr<Piece> > pieces;  getMissingPiece(pieces, 1, peer, excludedIndexes, cuid);  if(pieces.empty()) {    return std::shared_ptr<Piece>();  } else {    return pieces.front();  }}std::shared_ptr<Piece> DefaultPieceStorage::getMissingFastPiece(const std::shared_ptr<Peer>& peer, cuid_t cuid){  std::vector<std::shared_ptr<Piece> > pieces;  getMissingFastPiece(pieces, 1, peer, cuid);  if(pieces.empty()) {    return std::shared_ptr<Piece>();  } else {    return pieces.front();  }}std::shared_ptr<Piece> DefaultPieceStorage::getMissingFastPiece(const std::shared_ptr<Peer>& peer, const std::vector<size_t>& excludedIndexes, cuid_t cuid){  std::vector<std::shared_ptr<Piece> > pieces;  getMissingFastPiece(pieces, 1, peer, excludedIndexes, cuid);  if(pieces.empty()) {    return std::shared_ptr<Piece>();  } else {    return pieces.front();  }}#endif // ENABLE_BITTORRENTbool DefaultPieceStorage::hasMissingUnusedPiece(){  size_t index;  return bitfieldMan_->getFirstMissingUnusedIndex(index);}std::shared_ptr<Piece> DefaultPieceStorage::getMissingPiece(size_t minSplitSize, const unsigned char* ignoreBitfield, size_t length, cuid_t cuid){  size_t index;  if(streamPieceSelector_->select     (index, minSplitSize, ignoreBitfield, length)) {    return checkOutPiece(index, cuid);  } else {    return std::shared_ptr<Piece>();  }}std::shared_ptr<Piece> DefaultPieceStorage::getMissingPiece(size_t index, cuid_t cuid){  if(hasPiece(index) || isPieceUsed(index)) {    return std::shared_ptr<Piece>();  } else {    return checkOutPiece(index, cuid);  }}void DefaultPieceStorage::deleteUsedPiece(const std::shared_ptr<Piece>& piece){  if(!piece) {    return;  }  usedPieces_.erase(piece);  piece->releaseWrCache(wrDiskCache_);}// void DefaultPieceStorage::reduceUsedPieces(size_t upperBound)// {//   size_t usedPiecesSize = usedPieces.size();//   if(usedPiecesSize <= upperBound) {//     return;//   }//   size_t delNum = usedPiecesSize-upperBound;//   int fillRate = 10;//   while(delNum && fillRate <= 15) {//     delNum -= deleteUsedPiecesByFillRate(fillRate, delNum);//     fillRate += 5;//   }// }// size_t DefaultPieceStorage::deleteUsedPiecesByFillRate(int fillRate,//                                                     size_t delNum)// {//   size_t deleted = 0;//   for(Pieces::iterator itr = usedPieces.begin();//       itr != usedPieces.end() && deleted < delNum;) {//     std::shared_ptr<Piece>& piece = *itr;//     if(!bitfieldMan->isUseBitSet(piece->getIndex()) &&//        piece->countCompleteBlock() <= piece->countBlock()*(fillRate/100.0)) {//       logger->info(MSG_DELETING_USED_PIECE,//                  piece->getIndex(),//                  (piece->countCompleteBlock()*100)/piece->countBlock(),//                  fillRate);//       itr = usedPieces.erase(itr);//       ++deleted;//     } else {//       ++itr;//     }//   }//   return deleted;// }void DefaultPieceStorage::completePiece(const std::shared_ptr<Piece>& piece){  if(!piece) {    return;  }  deleteUsedPiece(piece);  //   if(!isEndGame()) {  //     reduceUsedPieces(100);  //   }  if(allDownloadFinished()) {    return;  }  bitfieldMan_->setBit(piece->getIndex());  bitfieldMan_->unsetUseBit(piece->getIndex());  addPieceStats(piece->getIndex());  if(downloadFinished()) {    downloadContext_->resetDownloadStopTime();    if(isSelectiveDownloadingMode()) {      A2_LOG_NOTICE(MSG_SELECTIVE_DOWNLOAD_COMPLETED);      // following line was commented out in order to stop sending request      // message after user-specified files were downloaded.      //finishSelectiveDownloadingMode();    } else {      A2_LOG_INFO(MSG_DOWNLOAD_COMPLETED);    }#ifdef ENABLE_BITTORRENT    if(downloadContext_->hasAttribute(CTX_ATTR_BT)) {      std::shared_ptr<TorrentAttribute> torrentAttrs =        bittorrent::getTorrentAttrs(downloadContext_);      if(!torrentAttrs->metadata.empty()) {#ifdef __MINGW32__        // On Windows, if aria2 opens files with GENERIC_WRITE access        // right, some programs cannot open them aria2 is seeding. To        // avoid this situation, re-open the files with read-only        // enabled.        A2_LOG_INFO("Closing files and re-open them with read-only mode"                    " enabled.");        diskAdaptor_->closeFile();        diskAdaptor_->enableReadOnly();        diskAdaptor_->openFile();#endif // __MINGW32__        util::executeHookByOptName(downloadContext_->getOwnerRequestGroup(),                                   option_, PREF_ON_BT_DOWNLOAD_COMPLETE);        SingletonHolder<Notifier>::instance()->          notifyDownloadEvent(EVENT_ON_BT_DOWNLOAD_COMPLETE,                              downloadContext_->getOwnerRequestGroup());      }    }#endif // ENABLE_BITTORRENT  }}bool DefaultPieceStorage::isSelectiveDownloadingMode(){  return bitfieldMan_->isFilterEnabled();}// not unittestedvoid DefaultPieceStorage::cancelPiece(const std::shared_ptr<Piece>& piece, cuid_t cuid){  if(!piece) {    return;  }  piece->removeUser(cuid);  if(!piece->getUsed()) {    bitfieldMan_->unsetUseBit(piece->getIndex());  }  if(!isEndGame()) {    if(piece->getCompletedLength() == 0) {      deleteUsedPiece(piece);    }  }}bool DefaultPieceStorage::hasPiece(size_t index){  return bitfieldMan_->isBitSet(index);}bool DefaultPieceStorage::isPieceUsed(size_t index){  return bitfieldMan_->isUseBitSet(index);}int64_t DefaultPieceStorage::getTotalLength(){  return bitfieldMan_->getTotalLength();}int64_t DefaultPieceStorage::getFilteredTotalLength(){  return bitfieldMan_->getFilteredTotalLength();}int64_t DefaultPieceStorage::getCompletedLength(){  int64_t completedLength =    bitfieldMan_->getCompletedLength()+getInFlightPieceCompletedLength();  int64_t totalLength = getTotalLength();  if(completedLength > totalLength) {    completedLength = totalLength;  }  return completedLength;}int64_t DefaultPieceStorage::getFilteredCompletedLength(){  return bitfieldMan_->getFilteredCompletedLength()+    getInFlightPieceFilteredCompletedLength();}int64_t DefaultPieceStorage::getInFlightPieceCompletedLength() const{  int64_t len = 0;  for(UsedPieceSet::const_iterator i = usedPieces_.begin(),        eoi = usedPieces_.end(); i != eoi; ++i) {    len += (*i)->getCompletedLength();  }  return len;}int64_t DefaultPieceStorage::getInFlightPieceFilteredCompletedLength() const{  int64_t len = 0;  for(UsedPieceSet::const_iterator i = usedPieces_.begin(),        eoi = usedPieces_.end(); i != eoi; ++i) {    if(bitfieldMan_->isFilterBitSet((*i)->getIndex())) {      len += (*i)->getCompletedLength();    }  }  return len;}// not unittestedvoid DefaultPieceStorage::setupFileFilter(){  const std::vector<std::shared_ptr<FileEntry> >& fileEntries =    downloadContext_->getFileEntries();  bool allSelected = true;  for(std::vector<std::shared_ptr<FileEntry> >::const_iterator i =        fileEntries.begin(), eoi = fileEntries.end();      i != eoi; ++i) {    if(!(*i)->isRequested()) {      allSelected = false;      break;    }  }  if(allSelected) {    return;  }  for(std::vector<std::shared_ptr<FileEntry> >::const_iterator i =        fileEntries.begin(), eoi = fileEntries.end(); i != eoi; ++i) {    if((*i)->isRequested()) {      bitfieldMan_->addFilter((*i)->getOffset(), (*i)->getLength());    }  }  bitfieldMan_->enableFilter();}// not unittestedvoid DefaultPieceStorage::clearFileFilter(){  bitfieldMan_->clearFilter();}// not unittestedbool DefaultPieceStorage::downloadFinished(){  // TODO iterate all requested FileEntry and Call  // bitfieldMan->isBitSetOffsetRange()  return bitfieldMan_->isFilteredAllBitSet();}// not unittestedbool DefaultPieceStorage::allDownloadFinished(){  return bitfieldMan_->isAllBitSet();}// not unittestedvoid DefaultPieceStorage::initStorage(){  if(downloadContext_->getFileEntries().size() == 1) {    A2_LOG_DEBUG("Instantiating DirectDiskAdaptor");    DirectDiskAdaptor* directDiskAdaptor(new DirectDiskAdaptor());    directDiskAdaptor->setTotalLength(downloadContext_->getTotalLength());    directDiskAdaptor->setFileEntries      (downloadContext_->getFileEntries().begin(),       downloadContext_->getFileEntries().end());    std::shared_ptr<DiskWriter> writer =      diskWriterFactory_->newDiskWriter(directDiskAdaptor->getFilePath());    directDiskAdaptor->setDiskWriter(writer);    diskAdaptor_.reset(directDiskAdaptor);  } else {    A2_LOG_DEBUG("Instantiating MultiDiskAdaptor");    MultiDiskAdaptor* multiDiskAdaptor(new MultiDiskAdaptor());    multiDiskAdaptor->setFileEntries(downloadContext_->getFileEntries().begin(),                                     downloadContext_->getFileEntries().end());    multiDiskAdaptor->setPieceLength(downloadContext_->getPieceLength());    multiDiskAdaptor->setMaxOpenFiles      (option_->getAsInt(PREF_BT_MAX_OPEN_FILES));    diskAdaptor_.reset(multiDiskAdaptor);  }  if(option_->get(PREF_FILE_ALLOCATION) == V_FALLOC) {    diskAdaptor_->setFileAllocationMethod(DiskAdaptor::FILE_ALLOC_FALLOC);  } else if(option_->get(PREF_FILE_ALLOCATION) == V_TRUNC) {    diskAdaptor_->setFileAllocationMethod(DiskAdaptor::FILE_ALLOC_TRUNC);  }}void DefaultPieceStorage::setBitfield(const unsigned char* bitfield,                                      size_t bitfieldLength){  bitfieldMan_->setBitfield(bitfield, bitfieldLength);  addPieceStats(bitfield, bitfieldLength);}size_t DefaultPieceStorage::getBitfieldLength(){  return bitfieldMan_->getBitfieldLength();}const unsigned char* DefaultPieceStorage::getBitfield(){  return bitfieldMan_->getBitfield();}std::shared_ptr<DiskAdaptor> DefaultPieceStorage::getDiskAdaptor() {  return diskAdaptor_;}WrDiskCache* DefaultPieceStorage::getWrDiskCache(){  return wrDiskCache_;}void DefaultPieceStorage::flushWrDiskCacheEntry(){  if(!wrDiskCache_) {    return;  }  // UsedPieceSet is sorted by piece index. It means we can flush  // cache by non-decreasing offset, which is good to reduce disk seek  // unless the file is heavily fragmented.  for(UsedPieceSet::const_iterator i = usedPieces_.begin(),        eoi = usedPieces_.end(); i != eoi; ++i) {    WrDiskCacheEntry* ce = (*i)->getWrDiskCacheEntry();    if(ce) {      (*i)->flushWrCache(wrDiskCache_);      (*i)->releaseWrCache(wrDiskCache_);    }  }}int32_t DefaultPieceStorage::getPieceLength(size_t index){  return bitfieldMan_->getBlockLength(index);}void DefaultPieceStorage::advertisePiece(cuid_t cuid, size_t index){  HaveEntry entry(cuid, index, global::wallclock());  haves_.push_front(entry);}voidDefaultPieceStorage::getAdvertisedPieceIndexes(std::vector<size_t>& indexes,                                               cuid_t myCuid,                                               const Timer& lastCheckTime){  for(std::deque<HaveEntry>::const_iterator itr = haves_.begin(),        eoi = haves_.end(); itr != eoi; ++itr) {    const HaveEntry& have = *itr;    if(lastCheckTime > have.getRegisteredTime()) {      break;    }    indexes.push_back(have.getIndex());  }}namespace {class FindElapsedHave{private:  time_t elapsed;public:  FindElapsedHave(time_t elapsed):elapsed(elapsed) {}  bool operator()(const HaveEntry& have) {    if(have.getRegisteredTime().difference(global::wallclock()) >= elapsed) {      return true;    } else {      return false;    }  }};} // namespacevoid DefaultPieceStorage::removeAdvertisedPiece(time_t elapsed){  std::deque<HaveEntry>::iterator itr =    std::find_if(haves_.begin(), haves_.end(), FindElapsedHave(elapsed));  if(itr != haves_.end()) {    A2_LOG_DEBUG(fmt(MSG_REMOVED_HAVE_ENTRY,                     static_cast<unsigned long>(haves_.end()-itr)));    haves_.erase(itr, haves_.end());  }}void DefaultPieceStorage::markAllPiecesDone(){  bitfieldMan_->setAllBit();}void DefaultPieceStorage::markPiecesDone(int64_t length){  if(length == bitfieldMan_->getTotalLength()) {    bitfieldMan_->setAllBit();  } else if(length == 0) {    // TODO this would go to markAllPiecesUndone()    bitfieldMan_->clearAllBit();    usedPieces_.clear();  } else {    size_t numPiece = length/bitfieldMan_->getBlockLength();    if(numPiece > 0) {      bitfieldMan_->setBitRange(0, numPiece-1);    }    size_t r = (length%bitfieldMan_->getBlockLength())/Piece::BLOCK_LENGTH;    if(r > 0) {      std::shared_ptr<Piece> p        (new Piece(numPiece, bitfieldMan_->getBlockLength(numPiece)));      for(size_t i = 0; i < r; ++i) {        p->completeBlock(i);      }#ifdef ENABLE_MESSAGE_DIGEST      p->setHashType(downloadContext_->getPieceHashType());#endif // ENABLE_MESSAGE_DIGEST      addUsedPiece(p);    }  }}void DefaultPieceStorage::markPieceMissing(size_t index){  bitfieldMan_->unsetBit(index);}void DefaultPieceStorage::addInFlightPiece(const std::vector<std::shared_ptr<Piece> >& pieces){  usedPieces_.insert(pieces.begin(), pieces.end());}size_t DefaultPieceStorage::countInFlightPiece(){  return usedPieces_.size();}void DefaultPieceStorage::getInFlightPieces(std::vector<std::shared_ptr<Piece> >& pieces){  pieces.insert(pieces.end(), usedPieces_.begin(), usedPieces_.end());}void DefaultPieceStorage::setDiskWriterFactory(const std::shared_ptr<DiskWriterFactory>& diskWriterFactory){  diskWriterFactory_ = diskWriterFactory;}void DefaultPieceStorage::addPieceStats(const unsigned char* bitfield,                                        size_t bitfieldLength){  pieceStatMan_->addPieceStats(bitfield, bitfieldLength);}void DefaultPieceStorage::subtractPieceStats(const unsigned char* bitfield,                                             size_t bitfieldLength){  pieceStatMan_->subtractPieceStats(bitfield, bitfieldLength);}void DefaultPieceStorage::updatePieceStats(const unsigned char* newBitfield,                                           size_t newBitfieldLength,                                           const unsigned char* oldBitfield){  pieceStatMan_->updatePieceStats(newBitfield, newBitfieldLength,                                  oldBitfield);}void DefaultPieceStorage::addPieceStats(size_t index){  pieceStatMan_->addPieceStats(index);}size_t DefaultPieceStorage::getNextUsedIndex(size_t index){  for(size_t i = index+1; i < bitfieldMan_->countBlock(); ++i) {    if(bitfieldMan_->isUseBitSet(i) || bitfieldMan_->isBitSet(i)) {      return i;    }  }  return bitfieldMan_->countBlock();}void DefaultPieceStorage::onDownloadIncomplete(){  streamPieceSelector_->onBitfieldInit();}} // namespace aria2
 |