mirror of https://github.com/aria2/aria2
Merge branch 'random-webseeding'
commit
88f20e4191
|
@ -327,6 +327,8 @@ HTTP/FTP/SFTP Options
|
|||
:option:`--min-split-size <-k>` option,
|
||||
so it will be necessary to specify a reasonable value to
|
||||
:option:`--min-split-size <-k>` option.
|
||||
If ``random`` is given, aria2 selects piece randomly. Like
|
||||
``inorder``, :option:`--min-split-size <-k>` option is honored.
|
||||
If ``geom`` is given, at the beginning aria2 selects piece which has
|
||||
minimum index like ``inorder``, but it exponentially increasingly
|
||||
keeps space from previously selected piece. This will reduce the
|
||||
|
|
|
@ -375,18 +375,19 @@ bool BitfieldMan::getGeomMissingUnusedIndex(size_t& index, int32_t minSplitSize,
|
|||
|
||||
namespace {
|
||||
template <typename Array>
|
||||
bool getInorderMissingUnusedIndex(size_t& index, int32_t minSplitSize,
|
||||
bool getInorderMissingUnusedIndex(size_t& index, size_t startIndex,
|
||||
size_t lastIndex, int32_t minSplitSize,
|
||||
const Array& bitfield,
|
||||
const unsigned char* useBitfield,
|
||||
int32_t blockLength, size_t blocks)
|
||||
{
|
||||
// We always return first piece if it is available.
|
||||
if (!bitfield::test(bitfield, blocks, 0) &&
|
||||
!bitfield::test(useBitfield, blocks, 0)) {
|
||||
index = 0;
|
||||
if (!bitfield::test(bitfield, blocks, startIndex) &&
|
||||
!bitfield::test(useBitfield, blocks, startIndex)) {
|
||||
index = startIndex;
|
||||
return true;
|
||||
}
|
||||
for (size_t i = 1; i < blocks;) {
|
||||
for (size_t i = startIndex + 1; i < lastIndex;) {
|
||||
if (!bitfield::test(bitfield, blocks, i) &&
|
||||
!bitfield::test(useBitfield, blocks, i)) {
|
||||
// If previous piece has already been retrieved, we can download
|
||||
|
@ -396,7 +397,8 @@ bool getInorderMissingUnusedIndex(size_t& index, int32_t minSplitSize,
|
|||
index = i;
|
||||
return true;
|
||||
}
|
||||
// Check free space of minSplitSize.
|
||||
// Check free space of minSplitSize. When checking this, we use
|
||||
// blocks instead of lastIndex.
|
||||
size_t j;
|
||||
for (j = i; j < blocks; ++j) {
|
||||
if (bitfield::test(bitfield, blocks, j) ||
|
||||
|
@ -424,13 +426,34 @@ bool BitfieldMan::getInorderMissingUnusedIndex(
|
|||
{
|
||||
if (filterEnabled_) {
|
||||
return aria2::getInorderMissingUnusedIndex(
|
||||
index, minSplitSize, array(ignoreBitfield) | ~array(filterBitfield_) |
|
||||
array(bitfield_) | array(useBitfield_),
|
||||
index, 0, blocks_, minSplitSize,
|
||||
array(ignoreBitfield) | ~array(filterBitfield_) | array(bitfield_) |
|
||||
array(useBitfield_),
|
||||
useBitfield_, blockLength_, blocks_);
|
||||
}
|
||||
else {
|
||||
return aria2::getInorderMissingUnusedIndex(
|
||||
index, minSplitSize,
|
||||
index, 0, blocks_, minSplitSize,
|
||||
array(ignoreBitfield) | array(bitfield_) | array(useBitfield_),
|
||||
useBitfield_, blockLength_, blocks_);
|
||||
}
|
||||
}
|
||||
|
||||
bool BitfieldMan::getInorderMissingUnusedIndex(
|
||||
size_t& index, size_t startIndex, size_t endIndex, int32_t minSplitSize,
|
||||
const unsigned char* ignoreBitfield, size_t ignoreBitfieldLength) const
|
||||
{
|
||||
endIndex = std::min(endIndex, blocks_);
|
||||
if (filterEnabled_) {
|
||||
return aria2::getInorderMissingUnusedIndex(
|
||||
index, startIndex, endIndex, minSplitSize,
|
||||
array(ignoreBitfield) | ~array(filterBitfield_) | array(bitfield_) |
|
||||
array(useBitfield_),
|
||||
useBitfield_, blockLength_, blocks_);
|
||||
}
|
||||
else {
|
||||
return aria2::getInorderMissingUnusedIndex(
|
||||
index, startIndex, endIndex, minSplitSize,
|
||||
array(ignoreBitfield) | array(bitfield_) | array(useBitfield_),
|
||||
useBitfield_, blockLength_, blocks_);
|
||||
}
|
||||
|
|
|
@ -161,8 +161,20 @@ public:
|
|||
const unsigned char* ignoreBitfield,
|
||||
size_t ignoreBitfieldLength) const;
|
||||
|
||||
// Just like getInorderMissingUnusedIndex() above, but limit the
|
||||
// search area in [startIndex, endIndex). |endIndex| is normalized
|
||||
// to min(|endIndex|, blocks_)
|
||||
//
|
||||
// affected by filter
|
||||
bool getAllMissingIndexes(unsigned char* misbitfield, size_t mislen) const;
|
||||
bool getInorderMissingUnusedIndex
|
||||
(size_t& index,
|
||||
size_t startIndex, size_t endIndex,
|
||||
int32_t minSplitSize,
|
||||
const unsigned char* ignoreBitfield,
|
||||
size_t ignoreBitfieldLength) const;
|
||||
|
||||
// affected by filter
|
||||
bool getAllMissingIndexes(unsigned char *misbitfield, size_t mislen) const;
|
||||
|
||||
// affected by filter
|
||||
bool getAllMissingIndexes(unsigned char* misbitfield, size_t mislen,
|
||||
|
|
|
@ -58,6 +58,7 @@
|
|||
#include "RarestPieceSelector.h"
|
||||
#include "DefaultStreamPieceSelector.h"
|
||||
#include "InorderStreamPieceSelector.h"
|
||||
#include "RandomStreamPieceSelector.h"
|
||||
#include "GeomStreamPieceSelector.h"
|
||||
#include "array_fun.h"
|
||||
#include "PieceStatMan.h"
|
||||
|
@ -99,6 +100,10 @@ DefaultPieceStorage::DefaultPieceStorage(
|
|||
streamPieceSelector_ =
|
||||
make_unique<InorderStreamPieceSelector>(bitfieldMan_.get());
|
||||
}
|
||||
else if (pieceSelectorOpt == A2_V_RANDOM) {
|
||||
streamPieceSelector_ =
|
||||
make_unique<RandomStreamPieceSelector>(bitfieldMan_.get());
|
||||
}
|
||||
else if (pieceSelectorOpt == A2_V_GEOM) {
|
||||
streamPieceSelector_ =
|
||||
make_unique<GeomStreamPieceSelector>(bitfieldMan_.get(), 1.5);
|
||||
|
|
|
@ -140,6 +140,7 @@ SRCS = \
|
|||
InitiateConnectionCommand.cc InitiateConnectionCommand.h\
|
||||
InitiateConnectionCommandFactory.cc InitiateConnectionCommandFactory.h\
|
||||
InorderStreamPieceSelector.cc InorderStreamPieceSelector.h\
|
||||
RandomStreamPieceSelector.cc RandomStreamPieceSelector.h\
|
||||
InorderURISelector.cc InorderURISelector.h\
|
||||
IOFile.cc IOFile.h\
|
||||
IteratableChecksumValidator.cc IteratableChecksumValidator.h\
|
||||
|
|
|
@ -954,7 +954,7 @@ std::vector<OptionHandler*> OptionHandlerFactory::createOptionHandlers()
|
|||
{
|
||||
OptionHandler* op(new ParameterOptionHandler(
|
||||
PREF_STREAM_PIECE_SELECTOR, TEXT_STREAM_PIECE_SELECTOR, A2_V_DEFAULT,
|
||||
{A2_V_DEFAULT, V_INORDER, A2_V_GEOM}));
|
||||
{A2_V_DEFAULT, V_INORDER, A2_V_RANDOM, A2_V_GEOM}));
|
||||
op->addTag(TAG_FTP);
|
||||
op->addTag(TAG_HTTP);
|
||||
op->setInitialOption(true);
|
||||
|
|
|
@ -0,0 +1,76 @@
|
|||
/* <!-- copyright */
|
||||
/*
|
||||
* aria2 - The high speed download utility
|
||||
*
|
||||
* Copyright (C) 2015 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 "RandomStreamPieceSelector.h"
|
||||
#include "BitfieldMan.h"
|
||||
#include "SimpleRandomizer.h"
|
||||
|
||||
namespace aria2 {
|
||||
|
||||
RandomStreamPieceSelector::RandomStreamPieceSelector
|
||||
(BitfieldMan* bitfieldMan)
|
||||
: bitfieldMan_(bitfieldMan)
|
||||
{}
|
||||
|
||||
RandomStreamPieceSelector::~RandomStreamPieceSelector() {}
|
||||
|
||||
bool RandomStreamPieceSelector::select
|
||||
(size_t& index,
|
||||
size_t minSplitSize,
|
||||
const unsigned char* ignoreBitfield,
|
||||
size_t length)
|
||||
{
|
||||
size_t start = SimpleRandomizer::getInstance()->getRandomNumber
|
||||
(bitfieldMan_->countBlock());
|
||||
|
||||
auto rv = bitfieldMan_->getInorderMissingUnusedIndex
|
||||
(index, start, bitfieldMan_->countBlock(), minSplitSize, ignoreBitfield,
|
||||
length);
|
||||
if (rv) {
|
||||
return true;
|
||||
}
|
||||
rv = bitfieldMan_->getInorderMissingUnusedIndex(index, 0, start, minSplitSize,
|
||||
ignoreBitfield, length);
|
||||
if (rv) {
|
||||
return true;
|
||||
}
|
||||
// Fall back to inorder search because randomized search may fail
|
||||
// because of |minSplitSize| constraint.
|
||||
return bitfieldMan_->getInorderMissingUnusedIndex(index, minSplitSize,
|
||||
ignoreBitfield, length);
|
||||
}
|
||||
|
||||
void RandomStreamPieceSelector::onBitfieldInit() {}
|
||||
|
||||
} // namespace aria2
|
|
@ -0,0 +1,62 @@
|
|||
/* <!-- copyright */
|
||||
/*
|
||||
* aria2 - The high speed download utility
|
||||
*
|
||||
* Copyright (C) 2015 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 --> */
|
||||
#ifndef D_RANDOM_STREAM_PIECE_SELECTOR_H
|
||||
#define D_RANDOM_STREAM_PIECE_SELECTOR_H
|
||||
|
||||
#include "StreamPieceSelector.h"
|
||||
|
||||
namespace aria2 {
|
||||
|
||||
class BitfieldMan;
|
||||
|
||||
class RandomStreamPieceSelector:public StreamPieceSelector {
|
||||
public:
|
||||
RandomStreamPieceSelector(BitfieldMan* bitfieldMan);
|
||||
virtual ~RandomStreamPieceSelector();
|
||||
|
||||
virtual bool select
|
||||
(size_t& index,
|
||||
size_t minSplitSize,
|
||||
const unsigned char* ignoreBitfield,
|
||||
size_t length) CXX11_OVERRIDE;
|
||||
|
||||
virtual void onBitfieldInit() CXX11_OVERRIDE;
|
||||
private:
|
||||
BitfieldMan* bitfieldMan_;
|
||||
};
|
||||
|
||||
} // namespace aria2
|
||||
|
||||
#endif // D_RANDOM_STREAM_PIECE_SELECTOR_H
|
|
@ -557,8 +557,7 @@ void RequestGroup::initPieceStorage()
|
|||
downloadContext_->getFileEntries().begin(),
|
||||
downloadContext_->getFileEntries().end())) {
|
||||
// Use LongestSequencePieceSelector when HTTP/FTP/BitTorrent
|
||||
// integrated downloads. Currently multi-file integrated
|
||||
// download is not supported.
|
||||
// integrated downloads.
|
||||
A2_LOG_DEBUG("Using LongestSequencePieceSelector");
|
||||
ps->setPieceSelector(make_unique<LongestSequencePieceSelector>());
|
||||
}
|
||||
|
|
|
@ -131,6 +131,7 @@ const std::string V_NOTICE("notice");
|
|||
const std::string V_WARN("warn");
|
||||
const std::string V_ERROR("error");
|
||||
const std::string V_INORDER("inorder");
|
||||
const std::string A2_V_RANDOM("random");
|
||||
const std::string V_FEEDBACK("feedback");
|
||||
const std::string V_ADAPTIVE("adaptive");
|
||||
const std::string V_LIBUV("libuv");
|
||||
|
|
|
@ -88,6 +88,7 @@ extern const std::string V_NOTICE;
|
|||
extern const std::string V_WARN;
|
||||
extern const std::string V_ERROR;
|
||||
extern const std::string V_INORDER;
|
||||
extern const std::string A2_V_RANDOM;
|
||||
extern const std::string V_FEEDBACK;
|
||||
extern const std::string V_ADAPTIVE;
|
||||
extern const std::string V_LIBUV;
|
||||
|
|
|
@ -851,6 +851,9 @@
|
|||
" --min-split-size option, so it will be necessary\n" \
|
||||
" to specify a reasonable value to\n" \
|
||||
" --min-split-size option.\n" \
|
||||
" If 'random' is given, aria2 selects piece\n" \
|
||||
" randomly. Like 'inorder', --min-split-size\n" \
|
||||
" option is honored.\n" \
|
||||
" If 'geom' is given, at the beginning aria2\n" \
|
||||
" selects piece which has minimum index like\n" \
|
||||
" 'inorder', but it exponentially increasingly\n" \
|
||||
|
|
Loading…
Reference in New Issue