/* */ #include "MultiDiskAdaptor.h" #include "DefaultDiskWriter.h" #include "message.h" #include "Util.h" #include "FileEntry.h" #include "MultiFileAllocationIterator.h" #include "DefaultDiskWriterFactory.h" #include "DlAbortEx.h" #include "File.h" namespace aria2 { DiskWriterEntry::DiskWriterEntry(const SharedHandle& fileEntry): fileEntry(fileEntry), diskWriter(0) {} DiskWriterEntry::~DiskWriterEntry() {} std::string DiskWriterEntry::getFilePath(const std::string& topDir) const { return topDir+"/"+fileEntry->getPath(); } void DiskWriterEntry::initAndOpenFile(const std::string& topDir) { diskWriter->initAndOpenFile(getFilePath(topDir), fileEntry->getLength()); } void DiskWriterEntry::openFile(const std::string& topDir) { diskWriter->openFile(getFilePath(topDir), fileEntry->getLength()); } void DiskWriterEntry::openExistingFile(const std::string& topDir) { diskWriter->openExistingFile(getFilePath(topDir), fileEntry->getLength()); } void DiskWriterEntry::closeFile() { diskWriter->closeFile(); } bool DiskWriterEntry::fileExists(const std::string& topDir) { return File(getFilePath(topDir)).exists(); } uint64_t DiskWriterEntry::size() const { return diskWriter->size(); } SharedHandle DiskWriterEntry::getFileEntry() const { return fileEntry; } void DiskWriterEntry::setDiskWriter(const SharedHandle& diskWriter) { this->diskWriter = diskWriter; } SharedHandle DiskWriterEntry::getDiskWriter() const { return diskWriter; } bool DiskWriterEntry::operator<(const DiskWriterEntry& entry) const { return fileEntry < entry.fileEntry; } MultiDiskAdaptor::MultiDiskAdaptor(): pieceLength(0) {} MultiDiskAdaptor::~MultiDiskAdaptor() {} void MultiDiskAdaptor::resetDiskWriterEntries() { diskWriterEntries.clear(); for(FileEntries::const_iterator itr = fileEntries.begin(); itr != fileEntries.end(); itr++) { DiskWriterEntryHandle entry = new DiskWriterEntry(*itr); if((*itr)->isRequested()) { entry->setDiskWriter(DefaultDiskWriterFactory().newDiskWriter()); } else { entry->setDiskWriter(new DefaultDiskWriter()); } entry->getDiskWriter()->setDirectIOAllowed(_directIOAllowed); diskWriterEntries.push_back(entry); } } std::string MultiDiskAdaptor::getTopDirPath() const { return storeDir+"/"+topDir; } void MultiDiskAdaptor::mkdir() const { for(FileEntries::const_iterator itr = fileEntries.begin(); itr != fileEntries.end(); itr++) { (*itr)->setupDir(getTopDirPath()); } } void MultiDiskAdaptor::openFile() { mkdir(); resetDiskWriterEntries(); for(DiskWriterEntries::iterator itr = diskWriterEntries.begin(); itr != diskWriterEntries.end(); itr++) { (*itr)->openFile(getTopDirPath()); } } void MultiDiskAdaptor::initAndOpenFile() { mkdir(); resetDiskWriterEntries(); for(DiskWriterEntries::iterator itr = diskWriterEntries.begin(); itr != diskWriterEntries.end(); itr++) { (*itr)->initAndOpenFile(getTopDirPath()); } } void MultiDiskAdaptor::openExistingFile() { resetDiskWriterEntries(); for(DiskWriterEntries::iterator itr = diskWriterEntries.begin(); itr != diskWriterEntries.end(); itr++) { (*itr)->openExistingFile(getTopDirPath()); } } void MultiDiskAdaptor::closeFile() { for(DiskWriterEntries::iterator itr = diskWriterEntries.begin(); itr != diskWriterEntries.end(); itr++) { (*itr)->closeFile(); } } void MultiDiskAdaptor::onDownloadComplete() { closeFile(); openFile(); } void MultiDiskAdaptor::writeData(const unsigned char* data, size_t len, off_t offset) { off_t fileOffset = offset; bool writing = false; size_t rem = len; for(DiskWriterEntries::iterator itr = diskWriterEntries.begin(); itr != diskWriterEntries.end() && rem != 0; itr++) { if(isInRange(*itr, offset) || writing) { size_t writeLength = calculateLength(*itr, fileOffset, rem); (*itr)->getDiskWriter()->writeData(data+(len-rem), writeLength, fileOffset); rem -= writeLength; writing = true; fileOffset = 0; } else { fileOffset -= (*itr)->getFileEntry()->getLength(); } } if(!writing) { throw new DlAbortEx(EX_FILE_OFFSET_OUT_OF_RANGE, Util::itos(offset, true).c_str()); } } bool MultiDiskAdaptor::isInRange(const DiskWriterEntryHandle entry, off_t offset) const { return entry->getFileEntry()->getOffset() <= offset && (uint64_t)offset < entry->getFileEntry()->getOffset()+entry->getFileEntry()->getLength(); } size_t MultiDiskAdaptor::calculateLength(const DiskWriterEntryHandle entry, off_t fileOffset, size_t rem) const { size_t length; if(entry->getFileEntry()->getLength() < (uint64_t)fileOffset+rem) { length = entry->getFileEntry()->getLength()-fileOffset; } else { length = rem; } return length; } ssize_t MultiDiskAdaptor::readData(unsigned char* data, size_t len, off_t offset) { off_t fileOffset = offset; bool reading = false; size_t rem = len; size_t totalReadLength = 0; for(DiskWriterEntries::iterator itr = diskWriterEntries.begin(); itr != diskWriterEntries.end() && rem != 0; itr++) { if(isInRange(*itr, offset) || reading) { size_t readLength = calculateLength((*itr), fileOffset, rem); totalReadLength += (*itr)->getDiskWriter()->readData(data+(len-rem), readLength, fileOffset); rem -= readLength; reading = true; fileOffset = 0; } else { fileOffset -= (*itr)->getFileEntry()->getLength(); } } if(!reading) { throw new DlAbortEx(EX_FILE_OFFSET_OUT_OF_RANGE, Util::itos(offset, true).c_str()); } return totalReadLength; } bool MultiDiskAdaptor::fileExists() { if(diskWriterEntries.empty()) { resetDiskWriterEntries(); } for(DiskWriterEntries::iterator itr = diskWriterEntries.begin(); itr != diskWriterEntries.end(); itr++) { if((*itr)->fileExists(getTopDirPath())) { return true; } } return false; } // TODO call DiskWriter::openFile() before calling this function. uint64_t MultiDiskAdaptor::size() const { uint64_t size = 0; for(DiskWriterEntries::const_iterator itr = diskWriterEntries.begin(); itr != diskWriterEntries.end(); itr++) { size += (*itr)->size(); } return size; } FileAllocationIteratorHandle MultiDiskAdaptor::fileAllocationIterator() { return new MultiFileAllocationIterator(this); } void MultiDiskAdaptor::enableDirectIO() { for(DiskWriterEntries::const_iterator itr = diskWriterEntries.begin(); itr != diskWriterEntries.end(); ++itr) { (*itr)->getDiskWriter()->enableDirectIO(); } } void MultiDiskAdaptor::disableDirectIO() { for(DiskWriterEntries::const_iterator itr = diskWriterEntries.begin(); itr != diskWriterEntries.end(); ++itr) { (*itr)->getDiskWriter()->disableDirectIO(); } } } // namespace aria2