2008-09-07 Tatsuhiro Tsujikawa <tujikawa at rednoah dot com>

Fixed the bug that DiskWriterEntry is not created when its
	FileEntry.isRequested() is false and it doesn't share a piece 
with
	other FileEntries that are requested. This bug causes 
segmentation fault
	in the end.
pull/1/head
Tatsuhiro Tsujikawa 2008-09-07 11:37:15 +00:00
parent 390538d0b5
commit 4e28efd925
6 changed files with 178 additions and 72 deletions

View File

@ -1,3 +1,15 @@
2008-09-07 Tatsuhiro Tsujikawa <tujikawa at rednoah dot com>
Fixed the bug that DiskWriterEntry is not created when its
FileEntry.isRequested() is false and it doesn't share a piece with
other FileEntries that are requested. This bug causes segmentation fault
in the end.
* src/MultiDiskAdaptor.cc
* src/MultiDiskAdaptor.h
* src/MultiFileAllocationIterator.cc
* test/MultiDiskAdaptorTest.cc
* test/MultiFileAllocationIteratorTest.cc
2008-09-07 Tatsuhiro Tsujikawa <tujikawa at rednoah dot com> 2008-09-07 Tatsuhiro Tsujikawa <tujikawa at rednoah dot com>
Fixed the bug that exception is thrown when MultiDiskAdaptor::size() is Fixed the bug that exception is thrown when MultiDiskAdaptor::size() is

View File

@ -45,11 +45,13 @@
#include "Logger.h" #include "Logger.h"
#include "SimpleRandomizer.h" #include "SimpleRandomizer.h"
#include <algorithm> #include <algorithm>
#include <cassert>
namespace aria2 { namespace aria2 {
DiskWriterEntry::DiskWriterEntry(const SharedHandle<FileEntry>& fileEntry): DiskWriterEntry::DiskWriterEntry(const SharedHandle<FileEntry>& fileEntry):
fileEntry(fileEntry), _open(false), _directIO(false) {} fileEntry(fileEntry), _open(false), _directIO(false),
_needsFileAllocation(false) {}
DiskWriterEntry::~DiskWriterEntry() {} DiskWriterEntry::~DiskWriterEntry() {}
@ -60,29 +62,35 @@ std::string DiskWriterEntry::getFilePath(const std::string& topDir) const
void DiskWriterEntry::initAndOpenFile(const std::string& topDir) void DiskWriterEntry::initAndOpenFile(const std::string& topDir)
{ {
diskWriter->initAndOpenFile(getFilePath(topDir), fileEntry->getLength()); if(!diskWriter.isNull()) {
if(_directIO) { diskWriter->initAndOpenFile(getFilePath(topDir), fileEntry->getLength());
diskWriter->enableDirectIO(); if(_directIO) {
diskWriter->enableDirectIO();
}
_open = true;
} }
_open = true;
} }
void DiskWriterEntry::openFile(const std::string& topDir) void DiskWriterEntry::openFile(const std::string& topDir)
{ {
diskWriter->openFile(getFilePath(topDir), fileEntry->getLength()); if(!diskWriter.isNull()) {
if(_directIO) { diskWriter->openFile(getFilePath(topDir), fileEntry->getLength());
diskWriter->enableDirectIO(); if(_directIO) {
diskWriter->enableDirectIO();
}
_open = true;
} }
_open = true;
} }
void DiskWriterEntry::openExistingFile(const std::string& topDir) void DiskWriterEntry::openExistingFile(const std::string& topDir)
{ {
diskWriter->openExistingFile(getFilePath(topDir), fileEntry->getLength()); if(!diskWriter.isNull()) {
if(_directIO) { diskWriter->openExistingFile(getFilePath(topDir), fileEntry->getLength());
diskWriter->enableDirectIO(); if(_directIO) {
diskWriter->enableDirectIO();
}
_open = true;
} }
_open = true;
} }
bool DiskWriterEntry::isOpen() const bool DiskWriterEntry::isOpen() const
@ -105,6 +113,7 @@ bool DiskWriterEntry::fileExists(const std::string& topDir)
uint64_t DiskWriterEntry::size() const uint64_t DiskWriterEntry::size() const
{ {
assert(!diskWriter.isNull());
return diskWriter->size(); return diskWriter->size();
} }
@ -144,6 +153,16 @@ void DiskWriterEntry::disableDirectIO()
_directIO = false; _directIO = false;
} }
bool DiskWriterEntry::needsFileAllocation() const
{
return _needsFileAllocation;
}
void DiskWriterEntry::needsFileAllocation(bool f)
{
_needsFileAllocation = f;
}
MultiDiskAdaptor::MultiDiskAdaptor(): MultiDiskAdaptor::MultiDiskAdaptor():
pieceLength(0), pieceLength(0),
_maxOpenFiles(DEFAULT_MAX_OPEN_FILES), _maxOpenFiles(DEFAULT_MAX_OPEN_FILES),
@ -153,13 +172,10 @@ MultiDiskAdaptor::~MultiDiskAdaptor() {}
static SharedHandle<DiskWriterEntry> createDiskWriterEntry static SharedHandle<DiskWriterEntry> createDiskWriterEntry
(const SharedHandle<FileEntry>& fileEntry, (const SharedHandle<FileEntry>& fileEntry,
DiskWriterFactory& dwFactory, bool needsFileAllocation)
bool directIOAllowed)
{ {
SharedHandle<DiskWriterEntry> entry(new DiskWriterEntry(fileEntry)); SharedHandle<DiskWriterEntry> entry(new DiskWriterEntry(fileEntry));
entry->setDiskWriter(dwFactory.newDiskWriter()); entry->needsFileAllocation(needsFileAllocation);
entry->getDiskWriter()->setDirectIOAllowed(directIOAllowed);
return entry; return entry;
} }
@ -172,49 +188,42 @@ void MultiDiskAdaptor::resetDiskWriterEntries()
return; return;
} }
DefaultDiskWriterFactory dwFactory; for(std::deque<SharedHandle<FileEntry> >::const_iterator i =
if(pieceLength == 0) { fileEntries.begin(); i != fileEntries.end(); ++i) {
for(std::deque<SharedHandle<FileEntry> >::const_iterator itr = diskWriterEntries.push_back
fileEntries.begin(); itr != fileEntries.end(); ++itr) { (createDiskWriterEntry(*i, (*i)->isRequested()));
if((*itr)->isRequested()) { }
diskWriterEntries.push_back
(createDiskWriterEntry(*itr, dwFactory, _directIOAllowed)); // TODO Currently, pieceLength == 0 is used for unit testing only.
} if(pieceLength > 0) {
} std::deque<SharedHandle<DiskWriterEntry> >::iterator done =
} else { diskWriterEntries.begin();
std::deque<SharedHandle<FileEntry> >::const_iterator done = fileEntries.begin(); for(std::deque<SharedHandle<DiskWriterEntry> >::iterator itr =
for(std::deque<SharedHandle<FileEntry> >::const_iterator itr = diskWriterEntries.begin(); itr != diskWriterEntries.end();) {
fileEntries.begin(); itr != fileEntries.end();) { if(!(*itr)->getFileEntry()->isRequested()) {
if(!(*itr)->isRequested()) {
++itr; ++itr;
continue; continue;
} }
off_t pieceStartOffset = ((*itr)->getOffset()/pieceLength)*pieceLength; off_t pieceStartOffset =
std::deque<SharedHandle<DiskWriterEntry> >::iterator insertionPoint = ((*itr)->getFileEntry()->getOffset()/pieceLength)*pieceLength;
diskWriterEntries.end(); if(itr != diskWriterEntries.begin()) {
for(std::deque<SharedHandle<DiskWriterEntry> >::iterator i =
if(itr != fileEntries.begin()) { itr-1; i != done; --i) {
for(std::deque<SharedHandle<FileEntry> >::const_iterator i = itr-1; const SharedHandle<FileEntry>& fileEntry = (*i)->getFileEntry();
i != done; --i) { if((uint64_t)pieceStartOffset <
if((uint64_t)pieceStartOffset < (*i)->getOffset()+(*i)->getLength()) { fileEntry->getOffset()+fileEntry->getLength()) {
insertionPoint = diskWriterEntries.insert (*i)->needsFileAllocation(true);
(insertionPoint,
createDiskWriterEntry(*i, dwFactory, _directIOAllowed));
} else { } else {
break; break;
} }
} }
} }
diskWriterEntries.push_back
(createDiskWriterEntry(*itr, dwFactory, _directIOAllowed));
++itr; ++itr;
for(; itr != fileEntries.end(); ++itr) { for(; itr != diskWriterEntries.end(); ++itr) {
if((*itr)->getOffset() < pieceStartOffset+pieceLength) { if((*itr)->getFileEntry()->getOffset() < pieceStartOffset+pieceLength) {
diskWriterEntries.push_back (*itr)->needsFileAllocation(true);
(createDiskWriterEntry(*itr, dwFactory, _directIOAllowed));
} else { } else {
break; break;
} }
@ -223,6 +232,16 @@ void MultiDiskAdaptor::resetDiskWriterEntries()
done = itr-1; done = itr-1;
} }
} }
DefaultDiskWriterFactory dwFactory;
for(std::deque<SharedHandle<DiskWriterEntry> >::iterator i =
diskWriterEntries.begin(); i != diskWriterEntries.end(); ++i) {
if((*i)->needsFileAllocation() || (*i)->fileExists(getTopDirPath())) {
logger->debug("Creating DiskWriter for filename=%s",
(*i)->getFilePath(getTopDirPath()).c_str());
(*i)->setDiskWriter(dwFactory.newDiskWriter());
(*i)->getDiskWriter()->setDirectIOAllowed(_directIOAllowed);
}
}
} }
std::string MultiDiskAdaptor::getTopDirPath() const std::string MultiDiskAdaptor::getTopDirPath() const

View File

@ -49,6 +49,7 @@ private:
SharedHandle<DiskWriter> diskWriter; SharedHandle<DiskWriter> diskWriter;
bool _open; bool _open;
bool _directIO; bool _directIO;
bool _needsFileAllocation;
public: public:
DiskWriterEntry(const SharedHandle<FileEntry>& fileEntry); DiskWriterEntry(const SharedHandle<FileEntry>& fileEntry);
@ -87,6 +88,10 @@ public:
// Additionally, if diskWriter is opened, diskWriter->disableDirectIO() is // Additionally, if diskWriter is opened, diskWriter->disableDirectIO() is
// called. // called.
void disableDirectIO(); void disableDirectIO();
bool needsFileAllocation() const;
void needsFileAllocation(bool f);
}; };
typedef SharedHandle<DiskWriterEntry> DiskWriterEntryHandle; typedef SharedHandle<DiskWriterEntry> DiskWriterEntryHandle;

View File

@ -61,7 +61,7 @@ void MultiFileAllocationIterator::allocateChunk()
_diskAdaptor->openIfNot(entry, &DiskWriterEntry::openFile, _diskAdaptor->openIfNot(entry, &DiskWriterEntry::openFile,
_diskAdaptor->getTopDirPath()); _diskAdaptor->getTopDirPath());
entry->enableDirectIO(); entry->enableDirectIO();
if(entry->size() < fileEntry->getLength()) { if(entry->needsFileAllocation() && entry->size() < fileEntry->getLength()) {
// Calling private function of MultiDiskAdaptor. // Calling private function of MultiDiskAdaptor.
_fileAllocationIterator.reset _fileAllocationIterator.reset
(new SingleFileAllocationIterator(entry->getDiskWriter().get(), (new SingleFileAllocationIterator(entry->getDiskWriter().get(),

View File

@ -158,6 +158,7 @@ void MultiDiskAdaptorTest::testCutTrailingGarbage()
adaptor.setTopDir(topDir); adaptor.setTopDir(topDir);
adaptor.setFileEntries(fileEntries); adaptor.setFileEntries(fileEntries);
adaptor.setMaxOpenFiles(1); adaptor.setMaxOpenFiles(1);
adaptor.setPieceLength(1);
adaptor.openFile(); adaptor.openFile();
@ -191,6 +192,7 @@ void MultiDiskAdaptorTest::testSize()
adaptor.setTopDir(topDir); adaptor.setTopDir(topDir);
adaptor.setFileEntries(fileEntries); adaptor.setFileEntries(fileEntries);
adaptor.setMaxOpenFiles(1); adaptor.setMaxOpenFiles(1);
adaptor.setPieceLength(1);
adaptor.openFile(); adaptor.openFile();

View File

@ -4,6 +4,8 @@
#include "FileEntry.h" #include "FileEntry.h"
#include "Exception.h" #include "Exception.h"
#include "array_fun.h" #include "array_fun.h"
#include "TestUtil.h"
#include "DiskWriter.h"
#include <algorithm> #include <algorithm>
#include <iostream> #include <iostream>
#include <cppunit/extensions/HelperMacros.h> #include <cppunit/extensions/HelperMacros.h>
@ -51,36 +53,92 @@ void MultiFileAllocationIteratorTest::testMakeDiskWriterEntries()
fs[8]->setRequested(false); // file9 fs[8]->setRequested(false); // file9
fs[9]->setRequested(false); // fileA fs[9]->setRequested(false); // fileA
std::string storeDir = "/tmp/aria2_MultiFileAllocationIteratorTest_testMakeDiskWriterEntries"; std::string topDir = "top";
std::string storeDir = "/tmp/aria2_MultiFileAllocationIteratorTest"
"_testMakeDiskWriterEntries";
std::string prefix = storeDir+"/"+topDir;
// create empty file4
createFile(prefix+std::string("/file4"), 0);
SharedHandle<MultiDiskAdaptor> diskAdaptor(new MultiDiskAdaptor()); SharedHandle<MultiDiskAdaptor> diskAdaptor(new MultiDiskAdaptor());
diskAdaptor->setFileEntries(std::deque<SharedHandle<FileEntry> >(&fs[0], &fs[arrayLength(fs)])); diskAdaptor->setFileEntries
(std::deque<SharedHandle<FileEntry> >(&fs[0], &fs[arrayLength(fs)]));
diskAdaptor->setPieceLength(1024); diskAdaptor->setPieceLength(1024);
diskAdaptor->setStoreDir(storeDir); diskAdaptor->setStoreDir(storeDir);
diskAdaptor->setTopDir(topDir);
diskAdaptor->openFile(); diskAdaptor->openFile();
SharedHandle<MultiFileAllocationIterator> itr SharedHandle<MultiFileAllocationIterator> itr
(dynamic_pointer_cast<MultiFileAllocationIterator>(diskAdaptor->fileAllocationIterator())); (dynamic_pointer_cast<MultiFileAllocationIterator>
(diskAdaptor->fileAllocationIterator()));
DiskWriterEntries entries = itr->getDiskWriterEntries(); DiskWriterEntries entries = itr->getDiskWriterEntries();
std::sort(entries.begin(), entries.end()); CPPUNIT_ASSERT_EQUAL((size_t)11, entries.size());
CPPUNIT_ASSERT_EQUAL((size_t)8, entries.size()); // file1
CPPUNIT_ASSERT_EQUAL(prefix+std::string("/file1"),
CPPUNIT_ASSERT_EQUAL(storeDir+std::string("/file1"), entries[0]->getFilePath(storeDir)); entries[0]->getFilePath(prefix));
CPPUNIT_ASSERT_EQUAL(storeDir+std::string("/file2"), entries[1]->getFilePath(storeDir)); CPPUNIT_ASSERT(entries[0]->needsFileAllocation());
CPPUNIT_ASSERT_EQUAL(storeDir+std::string("/file3"), entries[2]->getFilePath(storeDir)); CPPUNIT_ASSERT(!entries[0]->getDiskWriter().isNull());
CPPUNIT_ASSERT_EQUAL(storeDir+std::string("/file6"), entries[3]->getFilePath(storeDir)); // file2
CPPUNIT_ASSERT_EQUAL(storeDir+std::string("/file7"), entries[4]->getFilePath(storeDir)); CPPUNIT_ASSERT_EQUAL(prefix+std::string("/file2"),
CPPUNIT_ASSERT_EQUAL(storeDir+std::string("/file8"), entries[5]->getFilePath(storeDir)); entries[1]->getFilePath(prefix));
CPPUNIT_ASSERT_EQUAL(storeDir+std::string("/file9"), entries[6]->getFilePath(storeDir)); CPPUNIT_ASSERT(entries[1]->needsFileAllocation());
CPPUNIT_ASSERT_EQUAL(storeDir+std::string("/fileB"), entries[7]->getFilePath(storeDir)); CPPUNIT_ASSERT(!entries[1]->getDiskWriter().isNull());
// file3
CPPUNIT_ASSERT_EQUAL(prefix+std::string("/file3"),
entries[2]->getFilePath(prefix));
CPPUNIT_ASSERT(entries[2]->needsFileAllocation());
CPPUNIT_ASSERT(!entries[2]->getDiskWriter().isNull());
// file4, diskWriter is not null, because file exists.
CPPUNIT_ASSERT_EQUAL(prefix+std::string("/file4"),
entries[3]->getFilePath(prefix));
CPPUNIT_ASSERT(!entries[3]->needsFileAllocation());
CPPUNIT_ASSERT(!entries[3]->getDiskWriter().isNull());
// file5
CPPUNIT_ASSERT_EQUAL(prefix+std::string("/file5"),
entries[4]->getFilePath(prefix));
CPPUNIT_ASSERT(!entries[4]->needsFileAllocation());
CPPUNIT_ASSERT(entries[4]->getDiskWriter().isNull());
// file6
CPPUNIT_ASSERT_EQUAL(prefix+std::string("/file6"),
entries[5]->getFilePath(prefix));
CPPUNIT_ASSERT(entries[5]->needsFileAllocation());
CPPUNIT_ASSERT(!entries[5]->getDiskWriter().isNull());
// file7
CPPUNIT_ASSERT_EQUAL(prefix+std::string("/file7"),
entries[6]->getFilePath(prefix));
CPPUNIT_ASSERT(entries[6]->needsFileAllocation());
CPPUNIT_ASSERT(!entries[6]->getDiskWriter().isNull());
// file8
CPPUNIT_ASSERT_EQUAL(prefix+std::string("/file8"),
entries[7]->getFilePath(prefix));
CPPUNIT_ASSERT(entries[7]->needsFileAllocation());
CPPUNIT_ASSERT(!entries[7]->getDiskWriter().isNull());
// file9
CPPUNIT_ASSERT_EQUAL(prefix+std::string("/file9"),
entries[8]->getFilePath(prefix));
CPPUNIT_ASSERT(entries[8]->needsFileAllocation());
CPPUNIT_ASSERT(!entries[8]->getDiskWriter().isNull());
// fileA
CPPUNIT_ASSERT_EQUAL(prefix+std::string("/fileA"),
entries[9]->getFilePath(prefix));
CPPUNIT_ASSERT(!entries[9]->needsFileAllocation());
CPPUNIT_ASSERT(entries[9]->getDiskWriter().isNull());
// fileB
CPPUNIT_ASSERT_EQUAL(prefix+std::string("/fileB"),
entries[10]->getFilePath(prefix));
CPPUNIT_ASSERT(entries[10]->needsFileAllocation());
CPPUNIT_ASSERT(!entries[10]->getDiskWriter().isNull());
} }
void MultiFileAllocationIteratorTest::testAllocate() void MultiFileAllocationIteratorTest::testAllocate()
{ {
std::string dir = "/tmp"; std::string dir = "/tmp";
std::string topDir = "aria2_MultiFileAllocationIteratorTest_testAllocate"; std::string topDir = "aria2_MultiFileAllocationIteratorTest_testAllocate";
std::string prefix = dir+"/"+topDir;
std::string fname1 = "file1"; std::string fname1 = "file1";
std::string fname2 = "file2"; std::string fname2 = "file2";
std::string fname3 = "file3"; std::string fname3 = "file3";
@ -98,6 +156,7 @@ void MultiFileAllocationIteratorTest::testAllocate()
SharedHandle<MultiDiskAdaptor> diskAdaptor(new MultiDiskAdaptor()); SharedHandle<MultiDiskAdaptor> diskAdaptor(new MultiDiskAdaptor());
diskAdaptor->setStoreDir(dir); diskAdaptor->setStoreDir(dir);
diskAdaptor->setTopDir(topDir); diskAdaptor->setTopDir(topDir);
diskAdaptor->setPieceLength(1);
int64_t offset = 0; int64_t offset = 0;
SharedHandle<FileEntry> fileEntry1(new FileEntry(fname1, SharedHandle<FileEntry> fileEntry1(new FileEntry(fname1,
@ -139,20 +198,29 @@ void MultiFileAllocationIteratorTest::testAllocate()
fs.push_back(fileEntry6); fs.push_back(fileEntry6);
diskAdaptor->setFileEntries(fs); diskAdaptor->setFileEntries(fs);
File(prefix+"/"+fname1).remove();
File(prefix+"/"+fname2).remove();
File(prefix+"/"+fname3).remove();
File(prefix+"/"+fname4).remove();
File(prefix+"/"+fname5).remove();
File(prefix+"/"+fname6).remove();
// we have to open file first. // we have to open file first.
diskAdaptor->initAndOpenFile(); diskAdaptor->initAndOpenFile();
SharedHandle<MultiFileAllocationIterator> itr SharedHandle<MultiFileAllocationIterator> itr
(dynamic_pointer_cast<MultiFileAllocationIterator>(diskAdaptor->fileAllocationIterator())); (dynamic_pointer_cast<MultiFileAllocationIterator>
(diskAdaptor->fileAllocationIterator()));
while(!itr->finished()) { while(!itr->finished()) {
itr->allocateChunk(); itr->allocateChunk();
} }
CPPUNIT_ASSERT_EQUAL((uint64_t)length1, File(dir+"/"+topDir+"/"+fname1).size()); CPPUNIT_ASSERT_EQUAL((uint64_t)length1, File(prefix+"/"+fname1).size());
CPPUNIT_ASSERT_EQUAL((uint64_t)length2, File(dir+"/"+topDir+"/"+fname2).size()); CPPUNIT_ASSERT_EQUAL((uint64_t)length2, File(prefix+"/"+fname2).size());
CPPUNIT_ASSERT_EQUAL((uint64_t)length3, File(dir+"/"+topDir+"/"+fname3).size()); CPPUNIT_ASSERT_EQUAL((uint64_t)length3, File(prefix+"/"+fname3).size());
CPPUNIT_ASSERT(!File(dir+"/"+topDir+"/"+fname4).isFile()); CPPUNIT_ASSERT(!File(prefix+"/"+fname4).isFile());
CPPUNIT_ASSERT_EQUAL((uint64_t)length5, File(dir+"/"+topDir+"/"+fname5).size()); CPPUNIT_ASSERT_EQUAL((uint64_t)length5, File(prefix+"/"+fname5).size());
CPPUNIT_ASSERT(!File(dir+"/"+topDir+"/"+fname6).isFile()); CPPUNIT_ASSERT(!File(prefix+"/"+fname6).isFile());
} catch(Exception& e) { } catch(Exception& e) {
CPPUNIT_FAIL(e.stackTrace()); CPPUNIT_FAIL(e.stackTrace());