mirror of
https://github.com/aria2/aria2.git
synced 2025-04-06 14:07:37 +03:00
2007-02-06 Tatsuhiro Tsujikawa <tujikawa dot rednoah dot com>
To fix the bug that causes crash on Max OS X: * src/SimpleRandomizer.h (getInstance): Create new instance if the static variable is null. * src/SimpleRandomizer.cc (randomizer): Initialized to 0. * src/BitfieldManFactory.h (getNewFactory): Removed the call to setRandomizer(). To fix the miscalculation of the range of checksum: * src/BitfieldMan.h (isBitSetOffsetRange): New function. * src/BitfieldMan.cc (isBitSetOffsetRange): New function. * src/SegmentMan.cc (tryChunkChecksumValidation): Use BitfieldMan::isBitSetOffsetRange(). Use bitfield->getBlockLength() instead of segment.segmentLength.
This commit is contained in:
parent
a4e7bd7ec4
commit
5b2f8f036e
8 changed files with 90 additions and 8 deletions
21
ChangeLog
21
ChangeLog
|
@ -1,3 +1,24 @@
|
||||||
|
2007-02-06 Tatsuhiro Tsujikawa <tujikawa dot rednoah dot com>
|
||||||
|
|
||||||
|
To fix the bug that causes crash on Max OS X:
|
||||||
|
|
||||||
|
* src/SimpleRandomizer.h
|
||||||
|
(getInstance): Create new instance if the static variable is null.
|
||||||
|
* src/SimpleRandomizer.cc
|
||||||
|
(randomizer): Initialized to 0.
|
||||||
|
* src/BitfieldManFactory.h
|
||||||
|
(getNewFactory): Removed the call to setRandomizer().
|
||||||
|
|
||||||
|
To fix the miscalculation of the range of checksum:
|
||||||
|
|
||||||
|
* src/BitfieldMan.h
|
||||||
|
(isBitSetOffsetRange): New function.
|
||||||
|
* src/BitfieldMan.cc
|
||||||
|
(isBitSetOffsetRange): New function.
|
||||||
|
* src/SegmentMan.cc
|
||||||
|
(tryChunkChecksumValidation): Use BitfieldMan::isBitSetOffsetRange().
|
||||||
|
Use bitfield->getBlockLength() instead of segment.segmentLength.
|
||||||
|
|
||||||
2007-02-03 Tatsuhiro Tsujikawa <tujikawa at rednoah dot com>
|
2007-02-03 Tatsuhiro Tsujikawa <tujikawa at rednoah dot com>
|
||||||
|
|
||||||
To lower CPU usage in BitTorrent download when --max-upload-limit
|
To lower CPU usage in BitTorrent download when --max-upload-limit
|
||||||
|
|
|
@ -644,3 +644,24 @@ void BitfieldMan::unsetBitRange(int32_t startIndex, int32_t endIndex)
|
||||||
}
|
}
|
||||||
updateCache();
|
updateCache();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool BitfieldMan::isBitSetOffsetRange(int64_t offset, int64_t length) const
|
||||||
|
{
|
||||||
|
if(length <= 0) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if(totalLength <= offset) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if(totalLength < offset+length) {
|
||||||
|
length = totalLength-offset;
|
||||||
|
}
|
||||||
|
int32_t startBlock = offset/blockLength;
|
||||||
|
int32_t endBlock = (offset+length-1)/blockLength;
|
||||||
|
for(int32_t i = startBlock; i <= endBlock; i++) {
|
||||||
|
if(!isBitSet(i)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
|
@ -254,6 +254,9 @@ public:
|
||||||
bool isBitRangeSet(int32_t startIndex, int32_t endIndex) const;
|
bool isBitRangeSet(int32_t startIndex, int32_t endIndex) const;
|
||||||
|
|
||||||
void unsetBitRange(int32_t startIndex, int32_t endIndex);
|
void unsetBitRange(int32_t startIndex, int32_t endIndex);
|
||||||
|
|
||||||
|
bool isBitSetOffsetRange(int64_t offset, int64_t length) const;
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // _D_BITFIELD_MAN_H_
|
#endif // _D_BITFIELD_MAN_H_
|
||||||
|
|
|
@ -53,9 +53,7 @@ public:
|
||||||
~BitfieldManFactory() {}
|
~BitfieldManFactory() {}
|
||||||
|
|
||||||
static BitfieldManFactoryHandle getNewFactory() {
|
static BitfieldManFactoryHandle getNewFactory() {
|
||||||
BitfieldManFactoryHandle factory =
|
BitfieldManFactoryHandle factory = new BitfieldManFactory();
|
||||||
BitfieldManFactoryHandle(new BitfieldManFactory());
|
|
||||||
factory->setRandomizer(defaultRandomizer);
|
|
||||||
return factory;
|
return factory;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -497,10 +497,12 @@ void SegmentMan::tryChunkChecksumValidation(const Segment& segment)
|
||||||
segment.getPosition(),
|
segment.getPosition(),
|
||||||
segment.writtenLength,
|
segment.writtenLength,
|
||||||
chunkHashLength);
|
chunkHashLength);
|
||||||
if(hashStartIndex*chunkHashLength < segment.getPosition() && !bitfield->isBitSet(segment.index-1)) {
|
if(!bitfield->isBitSetOffsetRange((int64_t)hashStartIndex*chunkHashLength,
|
||||||
|
chunkHashLength)) {
|
||||||
++hashStartIndex;
|
++hashStartIndex;
|
||||||
}
|
}
|
||||||
if(hashEndIndex*(chunkHashLength+1) > segment.getPosition()+segment.segmentLength && !bitfield->isBitSet(segment.index+1)) {
|
if(!bitfield->isBitSetOffsetRange((int64_t)hashEndIndex*chunkHashLength,
|
||||||
|
chunkHashLength)) {
|
||||||
--hashEndIndex;
|
--hashEndIndex;
|
||||||
}
|
}
|
||||||
logger->debug("hashStartIndex=%d, hashEndIndex=%d",
|
logger->debug("hashStartIndex=%d, hashEndIndex=%d",
|
||||||
|
@ -515,7 +517,7 @@ void SegmentMan::tryChunkChecksumValidation(const Segment& segment)
|
||||||
Util::indexRange(startIndex, endIndex,
|
Util::indexRange(startIndex, endIndex,
|
||||||
hashOffset,
|
hashOffset,
|
||||||
(hashEndIndex-hashStartIndex+1)*chunkHashLength,
|
(hashEndIndex-hashStartIndex+1)*chunkHashLength,
|
||||||
segment.segmentLength);
|
bitfield->getBlockLength());
|
||||||
logger->debug("startIndex=%d, endIndex=%d", startIndex, endIndex);
|
logger->debug("startIndex=%d, endIndex=%d", startIndex, endIndex);
|
||||||
if(bitfield->isBitRangeSet(startIndex, endIndex)) {
|
if(bitfield->isBitRangeSet(startIndex, endIndex)) {
|
||||||
for(int32_t index = hashStartIndex; index <= hashEndIndex; ++index) {
|
for(int32_t index = hashStartIndex; index <= hashEndIndex; ++index) {
|
||||||
|
|
|
@ -34,4 +34,4 @@
|
||||||
/* copyright --> */
|
/* copyright --> */
|
||||||
#include "SimpleRandomizer.h"
|
#include "SimpleRandomizer.h"
|
||||||
|
|
||||||
RandomizerHandle SimpleRandomizer::randomizer = RandomizerHandle(new SimpleRandomizer());
|
RandomizerHandle SimpleRandomizer::randomizer = 0;
|
||||||
|
|
|
@ -47,11 +47,14 @@ private:
|
||||||
public:
|
public:
|
||||||
|
|
||||||
static RandomizerHandle getInstance() {
|
static RandomizerHandle getInstance() {
|
||||||
|
if(randomizer.isNull()) {
|
||||||
|
randomizer = new SimpleRandomizer();
|
||||||
|
}
|
||||||
return randomizer;
|
return randomizer;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void init() {
|
static void init() {
|
||||||
srandom(time(NULL));
|
srandom(time(0));
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual ~SimpleRandomizer() {}
|
virtual ~SimpleRandomizer() {}
|
||||||
|
|
|
@ -15,6 +15,7 @@ class BitfieldManTest:public CppUnit::TestFixture {
|
||||||
CPPUNIT_TEST(testFilter);
|
CPPUNIT_TEST(testFilter);
|
||||||
CPPUNIT_TEST(testGetMissingIndex);
|
CPPUNIT_TEST(testGetMissingIndex);
|
||||||
CPPUNIT_TEST(testGetSparceMissingUnusedIndex);
|
CPPUNIT_TEST(testGetSparceMissingUnusedIndex);
|
||||||
|
CPPUNIT_TEST(testIsBitSetOffsetRange);
|
||||||
CPPUNIT_TEST_SUITE_END();
|
CPPUNIT_TEST_SUITE_END();
|
||||||
private:
|
private:
|
||||||
RandomizerHandle fixedNumberRandomizer;
|
RandomizerHandle fixedNumberRandomizer;
|
||||||
|
@ -35,6 +36,7 @@ public:
|
||||||
void testFilter();
|
void testFilter();
|
||||||
void testGetMissingIndex();
|
void testGetMissingIndex();
|
||||||
void testGetSparceMissingUnusedIndex();
|
void testGetSparceMissingUnusedIndex();
|
||||||
|
void testIsBitSetOffsetRange();
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -233,3 +235,35 @@ void BitfieldManTest::testGetSparceMissingUnusedIndex() {
|
||||||
bitfield.setBit(9);
|
bitfield.setBit(9);
|
||||||
CPPUNIT_ASSERT_EQUAL(-1, bitfield.getSparseMissingUnusedIndex());
|
CPPUNIT_ASSERT_EQUAL(-1, bitfield.getSparseMissingUnusedIndex());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void BitfieldManTest::testIsBitSetOffsetRange()
|
||||||
|
{
|
||||||
|
int64_t totalLength = (int64_t)4*1024*1024*1024;
|
||||||
|
int32_t pieceLength = 4*1024*1024;
|
||||||
|
BitfieldMan bitfield(pieceLength, totalLength);
|
||||||
|
bitfield.setAllBit();
|
||||||
|
|
||||||
|
CPPUNIT_ASSERT(!bitfield.isBitSetOffsetRange(0, 0));
|
||||||
|
CPPUNIT_ASSERT(!bitfield.isBitSetOffsetRange(0, -1));
|
||||||
|
CPPUNIT_ASSERT(!bitfield.isBitSetOffsetRange(totalLength, 100));
|
||||||
|
CPPUNIT_ASSERT(!bitfield.isBitSetOffsetRange(totalLength+1, 100));
|
||||||
|
|
||||||
|
CPPUNIT_ASSERT(bitfield.isBitSetOffsetRange(0, totalLength));
|
||||||
|
CPPUNIT_ASSERT(bitfield.isBitSetOffsetRange(0, totalLength+1));
|
||||||
|
|
||||||
|
bitfield.clearAllBit();
|
||||||
|
|
||||||
|
bitfield.setBit(100);
|
||||||
|
bitfield.setBit(101);
|
||||||
|
|
||||||
|
CPPUNIT_ASSERT(bitfield.isBitSetOffsetRange(pieceLength*100, pieceLength*2));
|
||||||
|
CPPUNIT_ASSERT(!bitfield.isBitSetOffsetRange(pieceLength*100-10, pieceLength*2));
|
||||||
|
CPPUNIT_ASSERT(!bitfield.isBitSetOffsetRange(pieceLength*100, pieceLength*2+1));
|
||||||
|
|
||||||
|
bitfield.clearAllBit();
|
||||||
|
|
||||||
|
bitfield.setBit(100);
|
||||||
|
bitfield.setBit(102);
|
||||||
|
|
||||||
|
CPPUNIT_ASSERT(!bitfield.isBitSetOffsetRange(pieceLength*100, pieceLength*3));
|
||||||
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue