/* */ #include "MultiFileAllocationIterator.h" #include "MultiDiskAdaptor.h" #include "FileEntry.h" #include "AdaptiveFileAllocationIterator.h" #include "TruncFileAllocationIterator.h" #ifdef HAVE_SOME_FALLOCATE #include "FallocFileAllocationIterator.h" #endif // HAVE_SOME_FALLOCATE #include "DiskWriter.h" #include "DefaultDiskWriterFactory.h" #include "LogFactory.h" namespace aria2 { MultiFileAllocationIterator::MultiFileAllocationIterator( MultiDiskAdaptor* diskAdaptor) : diskAdaptor_{diskAdaptor}, entryItr_{std::begin(diskAdaptor_->getDiskWriterEntries())} { } MultiFileAllocationIterator::~MultiFileAllocationIterator() { if (diskWriter_) { diskWriter_->closeFile(); } } void MultiFileAllocationIterator::allocateChunk() { if (fileAllocationIterator_) { if (!fileAllocationIterator_->finished()) { fileAllocationIterator_->allocateChunk(); return; } if (diskWriter_) { diskWriter_->closeFile(); diskWriter_.reset(); } fileAllocationIterator_.reset(); ++entryItr_; } while (entryItr_ != std::end(diskAdaptor_->getDiskWriterEntries())) { if (!(*entryItr_)->getDiskWriter()) { ++entryItr_; continue; } auto& fileEntry = (*entryItr_)->getFileEntry(); // we use dedicated DiskWriter instead of // (*entryItr_)->getDiskWriter(). This is because // SingleFileAllocationIterator cannot reopen file if file is // closed by OpenedFileCounter. diskWriter_ = DefaultDiskWriterFactory().newDiskWriter((*entryItr_)->getFilePath()); // Open file before calling DiskWriterEntry::size(). Calling // private function of MultiDiskAdaptor. diskWriter_->openFile(fileEntry->getLength()); if ((*entryItr_)->needsFileAllocation() && (*entryItr_)->size() < fileEntry->getLength()) { A2_LOG_INFO(fmt("Allocating file %s: target size=%" PRId64 ", current size=%" PRId64, (*entryItr_)->getFilePath().c_str(), fileEntry->getLength(), (*entryItr_)->size())); switch (diskAdaptor_->getFileAllocationMethod()) { #ifdef HAVE_SOME_FALLOCATE case (DiskAdaptor::FILE_ALLOC_FALLOC): fileAllocationIterator_ = make_unique( diskWriter_.get(), (*entryItr_)->size(), fileEntry->getLength()); break; #endif // HAVE_SOME_FALLOCATE case (DiskAdaptor::FILE_ALLOC_TRUNC): fileAllocationIterator_ = make_unique( diskWriter_.get(), (*entryItr_)->size(), fileEntry->getLength()); break; default: fileAllocationIterator_ = make_unique( diskWriter_.get(), (*entryItr_)->size(), fileEntry->getLength()); break; } fileAllocationIterator_->allocateChunk(); return; } diskWriter_->closeFile(); diskWriter_.reset(); ++entryItr_; } } bool MultiFileAllocationIterator::finished() { return entryItr_ == std::end(diskAdaptor_->getDiskWriterEntries()) && (!fileAllocationIterator_ || fileAllocationIterator_->finished()); } int64_t MultiFileAllocationIterator::getCurrentLength() { if (!fileAllocationIterator_) { return 0; } else { return fileAllocationIterator_->getCurrentLength(); } } int64_t MultiFileAllocationIterator::getTotalLength() { if (!fileAllocationIterator_) { return 0; } else { return fileAllocationIterator_->getTotalLength(); } } const DiskWriterEntries& MultiFileAllocationIterator::getDiskWriterEntries() const { return diskAdaptor_->getDiskWriterEntries(); } } // namespace aria2