/* */ #include "AbstractSingleDiskAdaptor.h" #include "File.h" #include "AdaptiveFileAllocationIterator.h" #include "DiskWriter.h" #include "FileEntry.h" #include "TruncFileAllocationIterator.h" #include "WrDiskCacheEntry.h" #include "LogFactory.h" #ifdef HAVE_SOME_FALLOCATE # include "FallocFileAllocationIterator.h" #endif // HAVE_SOME_FALLOCATE namespace aria2 { AbstractSingleDiskAdaptor::AbstractSingleDiskAdaptor(): totalLength_(0), readOnly_(false) {} AbstractSingleDiskAdaptor::~AbstractSingleDiskAdaptor() {} void AbstractSingleDiskAdaptor::initAndOpenFile() { diskWriter_->initAndOpenFile(totalLength_); } void AbstractSingleDiskAdaptor::openFile() { diskWriter_->openFile(totalLength_); } void AbstractSingleDiskAdaptor::closeFile() { diskWriter_->closeFile(); } void AbstractSingleDiskAdaptor::openExistingFile() { diskWriter_->openExistingFile(totalLength_); } void AbstractSingleDiskAdaptor::writeData (const unsigned char* data, size_t len, int64_t offset) { diskWriter_->writeData(data, len, offset); } ssize_t AbstractSingleDiskAdaptor::readData (unsigned char* data, size_t len, int64_t offset) { return diskWriter_->readData(data, len, offset); } void AbstractSingleDiskAdaptor::writeCache(const WrDiskCacheEntry* entry) { // Write cached data in 4KiB aligned offset. This reduces disk // activity especially on Windows 7 NTFS. In this code, we assume // that maximum length of DataCell data is 16KiB to simplify the // code. unsigned char buf[16*1024]; int64_t start = 0; size_t buflen = 0; size_t buffoffset = 0; const WrDiskCacheEntry::DataCellSet& dataSet = entry->getDataSet(); for(auto & d : dataSet) { if(start+static_cast(buflen) < d->goff) { A2_LOG_DEBUG(fmt("Cache flush goff=%" PRId64 ", len=%lu", start, static_cast(buflen))); writeData(buf+buffoffset, buflen-buffoffset, start); start = d->goff; buflen = buffoffset = 0; } if(buflen == 0 && (d->goff & 0xfff) == 0 && (d->len & 0xfff) == 0) { // Already aligned. Write it without copy. A2_LOG_DEBUG(fmt("Cache flush goff=%" PRId64 ", len=%lu", start, static_cast(d->len))); writeData(d->data + d->offset, d->len, start); start += d->len; } else { if(buflen == 0) { buflen = buffoffset = d->goff & 0xfff; } size_t wlen = std::min(sizeof(buf) - buflen, d->len); memcpy(buf+buflen, d->data + d->offset, wlen); buflen += wlen; if(buflen == sizeof(buf)) { A2_LOG_DEBUG(fmt("Cache flush goff=%" PRId64 ", len=%lu", start, static_cast(buflen))); writeData(buf+buffoffset, buflen-buffoffset, start); memcpy(buf, d->data + d->offset + wlen, d->len - wlen); start += sizeof(buf) - buffoffset; buflen = d->len - wlen; buffoffset = 0; } } } writeData(buf+buffoffset, buflen-buffoffset, start); } bool AbstractSingleDiskAdaptor::fileExists() { return File(getFilePath()).exists(); } int64_t AbstractSingleDiskAdaptor::size() { return File(getFilePath()).size(); } void AbstractSingleDiskAdaptor::truncate(int64_t length) { diskWriter_->truncate(length); } std::unique_ptr AbstractSingleDiskAdaptor::fileAllocationIterator() { switch(getFileAllocationMethod()) { #ifdef HAVE_SOME_FALLOCATE case(DiskAdaptor::FILE_ALLOC_FALLOC): return make_unique (diskWriter_.get(), size() ,totalLength_); #endif // HAVE_SOME_FALLOCATE case(DiskAdaptor::FILE_ALLOC_TRUNC): return make_unique (diskWriter_.get(), size(), totalLength_); default: return make_unique (diskWriter_.get(), size(), totalLength_); } } void AbstractSingleDiskAdaptor::enableReadOnly() { diskWriter_->enableReadOnly(); readOnly_ = true; } void AbstractSingleDiskAdaptor::disableReadOnly() { diskWriter_->disableReadOnly(); readOnly_ = false; } void AbstractSingleDiskAdaptor::enableMmap() { diskWriter_->enableMmap(); } void AbstractSingleDiskAdaptor::cutTrailingGarbage() { if(File(getFilePath()).size() > totalLength_) { diskWriter_->truncate(totalLength_); } } void AbstractSingleDiskAdaptor::setDiskWriter (std::unique_ptr diskWriter) { diskWriter_ = std::move(diskWriter); } void AbstractSingleDiskAdaptor::setTotalLength(int64_t totalLength) { totalLength_ = totalLength; } } // namespace aria2