diff --git a/src/download/download_wrapper.cc b/src/download/download_wrapper.cc index 59e817814..3530b7601 100644 --- a/src/download/download_wrapper.cc +++ b/src/download/download_wrapper.cc @@ -46,6 +46,7 @@ #include "protocol/handshake_manager.h" #include "protocol/peer_connection_base.h" #include "torrent/exceptions.h" +#include "torrent/download.h" #include "torrent/object.h" #include "torrent/tracker_list.h" #include "torrent/data/file.h" @@ -103,7 +104,7 @@ DownloadWrapper::~DownloadWrapper() { } void -DownloadWrapper::initialize(const std::string& hash, const std::string& id) { +DownloadWrapper::initialize(const std::string& hash, const std::string& id, int flags) { char hashObfuscated[20]; sha1_salt("req2", 4, hash.c_str(), hash.length(), hashObfuscated); @@ -124,7 +125,7 @@ DownloadWrapper::initialize(const std::string& hash, const std::string& id) { // Connect various signals and slots. m_hashChecker->slot_check_chunk() = std::bind(&DownloadWrapper::check_chunk_hash, this, std::placeholders::_1); - m_hashChecker->delay_checked().slot() = std::bind(&DownloadWrapper::receive_initial_hash, this); + m_hashChecker->delay_checked().slot() = std::bind(&DownloadWrapper::receive_initial_hash, this, flags); } void @@ -156,7 +157,7 @@ DownloadWrapper::is_stopped() const { } void -DownloadWrapper::receive_initial_hash() { +DownloadWrapper::receive_initial_hash(int flags) { if (info()->is_active()) throw internal_error("DownloadWrapper::receive_initial_hash() but we're in a bad state."); @@ -172,7 +173,7 @@ DownloadWrapper::receive_initial_hash() { // Initialize the ChunkSelector here so that no chunks will be // marked by HashTorrent that are not accounted for. m_main->chunk_selector()->initialize(m_main->chunk_statistics()); - receive_update_priorities(); + receive_update_priorities(flags); } if (data()->slot_initial_hash()) @@ -324,7 +325,7 @@ DownloadWrapper::receive_tick(uint32_t ticks) { } void -DownloadWrapper::receive_update_priorities() { +DownloadWrapper::receive_update_priorities(int flags) { if (m_main->chunk_selector()->empty()) return; @@ -332,9 +333,15 @@ DownloadWrapper::receive_update_priorities() { data()->mutable_normal_priority()->clear(); for (FileList::iterator itr = m_main->file_list()->begin(); itr != m_main->file_list()->end(); ++itr) { + // Unset fallocate flag by default. + (*itr)->unset_flags(File::flag_fallocate); + switch ((*itr)->priority()) { case PRIORITY_NORMAL: { + if (flags & torrent::Download::open_enable_fallocate) + (*itr)->set_flags(File::flag_fallocate); + File::range_type range = (*itr)->range(); if ((*itr)->has_flags(File::flag_prioritize_first) && range.first != range.second) { @@ -351,6 +358,9 @@ DownloadWrapper::receive_update_priorities() { break; } case PRIORITY_HIGH: + if (flags & torrent::Download::open_enable_fallocate) + (*itr)->set_flags(File::flag_fallocate); + data()->mutable_high_priority()->insert((*itr)->range().first, (*itr)->range().second); break; default: diff --git a/src/download/download_wrapper.h b/src/download/download_wrapper.h index 6b4b860f9..d0063b698 100644 --- a/src/download/download_wrapper.h +++ b/src/download/download_wrapper.h @@ -64,7 +64,7 @@ class DownloadWrapper { ChunkList* chunk_list() { return m_main->chunk_list(); } // Initialize hash checker and various download stuff. - void initialize(const std::string& hash, const std::string& id); + void initialize(const std::string& hash, const std::string& id, int flags = 0); void close(); @@ -91,7 +91,7 @@ class DownloadWrapper { // Internal: // - void receive_initial_hash(); + void receive_initial_hash(int flags = 0); void receive_hash_done(ChunkHandle handle, const char* hash); void check_chunk_hash(ChunkHandle handle); @@ -102,7 +102,7 @@ class DownloadWrapper { void receive_tick(uint32_t ticks); - void receive_update_priorities(); + void receive_update_priorities(int flags = 0); private: DownloadWrapper(const DownloadWrapper&); diff --git a/src/torrent/data/file.cc b/src/torrent/data/file.cc index 0dfe6ea8a..5b410cd18 100644 --- a/src/torrent/data/file.cc +++ b/src/torrent/data/file.cc @@ -184,6 +184,7 @@ File::resize_file() { if (m_flags & flag_fallocate) { flags |= SocketFile::flag_fallocate; flags |= SocketFile::flag_fallocate_blocking; + m_flags &= ~flag_fallocate; } return SocketFile(m_fd).set_size(m_size, flags); diff --git a/src/torrent/data/file.h b/src/torrent/data/file.h index 4c58994bd..58ea52928 100644 --- a/src/torrent/data/file.h +++ b/src/torrent/data/file.h @@ -68,8 +68,11 @@ class LIBTORRENT_EXPORT lt_cacheline_aligned File { bool is_create_queued() const { return m_flags & flag_create_queued; } bool is_resize_queued() const { return m_flags & flag_resize_queued; } + bool is_fallocatable() const { return m_flags & flag_fallocate; } bool is_previously_created() const { return m_flags & flag_previously_created; } + bool is_fallocatable_file() { return has_flags(flag_resize_queued) && has_flags(flag_fallocate); } + bool has_flags(int flags) { return m_flags & flags; } void set_flags(int flags); diff --git a/src/torrent/data/file_list.cc b/src/torrent/data/file_list.cc index 4721bdbd6..9e57c7b14 100644 --- a/src/torrent/data/file_list.cc +++ b/src/torrent/data/file_list.cc @@ -198,9 +198,9 @@ FileList::set_max_file_size(uint64_t size) { uint64_t FileList::free_diskspace() const { uint64_t freeDiskspace = std::numeric_limits::max(); + rak::fs_stat stat; for (path_list::const_iterator itr = m_indirectLinks.begin(), last = m_indirectLinks.end(); itr != last; ++itr) { - rak::fs_stat stat; if (!stat.update(*itr)) continue; @@ -208,9 +208,74 @@ FileList::free_diskspace() const { freeDiskspace = std::min(freeDiskspace, stat.bytes_avail()); } + // Check the base directory of download if files haven't been created yet + if (freeDiskspace == std::numeric_limits::max()) { + std::string dirPath = is_multi_file() ? m_rootDir.substr(0, m_rootDir.find_last_of("\\/")) : m_rootDir; + + if (stat.update(dirPath)) + freeDiskspace = std::min(freeDiskspace, stat.bytes_avail()); + } + return freeDiskspace != std::numeric_limits::max() ? freeDiskspace : 0; } +uint64_t +FileList::allocatable_size_bytes() const { + uint64_t allocatableSizeBytes = 0; + + if (data()->is_partially_done()) + return allocatableSizeBytes; + + uint32_t allocatableSizeChunks = 0; + uint32_t prevChunk = -1; + bool areAllFilesAllocatable = true; + bool isLastFileAllocatable = false; + + for (FileList::const_iterator itr = begin(), last = end(); itr != last; itr++) { + + // Checks flag_fallocate and flag_resize_queued as well, it will take care of restarting client. + if ((*itr)->is_fallocatable_file()) { + allocatableSizeChunks += (*itr)->size_chunks() - ((*itr)->range_first() == prevChunk ? 1 : 0); + prevChunk = (*itr)->range_second() - 1; + + if (itr == end() - 1) + isLastFileAllocatable = true; + } else { + areAllFilesAllocatable = false; + } + + } + + if (areAllFilesAllocatable) + return m_torrentSize; + + allocatableSizeBytes = (uint64_t)allocatableSizeChunks * (uint64_t)m_chunkSize; + + // Dealing with size of last chunk as it's usually smaller than the rest. + uint64_t reminder = m_torrentSize % (uint64_t)m_chunkSize; + + if (isLastFileAllocatable && reminder != 0) + allocatableSizeBytes = allocatableSizeBytes - (uint64_t)m_chunkSize + reminder; + + return allocatableSizeBytes; +} + +bool +FileList::is_enough_diskspace() const { + uint64_t allocatable_size = allocatable_size_bytes(); + + if (allocatable_size > 0) { + uint64_t free_disk_space = free_diskspace(); + + if (free_disk_space < allocatable_size) { + LT_LOG_FL(INFO, "File allocation is set and not enough disk space to start torrent: allocatable size:%i free space:%i", allocatable_size, free_disk_space); + return false; + } + } + + return true; +} + FileList::iterator_range FileList::split(iterator position, split_type* first, split_type* last) { if (is_open()) @@ -580,7 +645,12 @@ FileList::open_file(File* node, const Path& lastPath, int flags) { return false; } - return node->prepare(MemoryChunk::prot_read, 0); + // File allocation will be done if fallocate flag is set, + // create zero-length file otherwise. + if (node->has_flags(File::flag_fallocate)) + return node->prepare(MemoryChunk::prot_write, 0); + else + return node->prepare(MemoryChunk::prot_read, 0); } MemoryChunk diff --git a/src/torrent/data/file_list.h b/src/torrent/data/file_list.h index 2cb64f669..d82ecc1e6 100644 --- a/src/torrent/data/file_list.h +++ b/src/torrent/data/file_list.h @@ -107,6 +107,7 @@ class LIBTORRENT_EXPORT FileList : private std::vector { size_t size_files() const { return base_type::size(); } uint64_t size_bytes() const { return m_torrentSize; } + uint64_t allocatable_size_bytes() const; uint32_t size_chunks() const { return bitfield()->size_bits(); } uint32_t completed_chunks() const { return bitfield()->size_set(); } @@ -132,6 +133,10 @@ class LIBTORRENT_EXPORT FileList : private std::vector { // of free diskspace will be returned. uint64_t free_diskspace() const; + // Determines whether there is enough disk space for fallocating + // selected files. + bool is_enough_diskspace() const; + // List of directories in the torrent that might be on different // volumes as they are links, including the root directory. Used by // 'free_diskspace()'. diff --git a/src/torrent/download.cc b/src/torrent/download.cc index edddedfbf..d004ba7a2 100644 --- a/src/torrent/download.cc +++ b/src/torrent/download.cc @@ -137,7 +137,13 @@ Download::start(int flags) { throw internal_error("Tried to start a download with empty bitfield."); if (info->is_active()) - return; + throw internal_error("Tried to start an already started download."); + + // Don't start the download if there's not enough disk space for it + // when the open_enable_fallocate was set and at least one of the + // files has fallocate flag (libtorrent would crash otherwise) + if (flags & open_enable_fallocate && !m_ptr->main()->file_list()->is_enough_diskspace()) + throw internal_error("Tried to start a download with not enough disk space for it."); LT_LOG_THIS(INFO, "Starting torrent: flags:%0x.", flags); @@ -145,9 +151,9 @@ Download::start(int flags) { // file_list()->open(flags); - // If the FileList::open_no_create flag was not set, our new - // behavior is to create all zero-length files with - // flag_queued_create set. + // If the open_enable_fallocate or the FileList::open_no_create + // flag was not set, then create all zero-length files with + // flag_create_queued set. file_list()->open(flags & ~FileList::open_no_create); if (m_ptr->connection_type() == CONNECTION_INITIAL_SEED) { @@ -592,8 +598,8 @@ Download::set_download_choke_heuristic(HeuristicType t) { } void -Download::update_priorities() { - m_ptr->receive_update_priorities(); +Download::update_priorities(int flags) { + m_ptr->receive_update_priorities(flags); } void diff --git a/src/torrent/download.h b/src/torrent/download.h index 4968c2de2..d104ad40e 100644 --- a/src/torrent/download.h +++ b/src/torrent/download.h @@ -195,7 +195,7 @@ class LIBTORRENT_EXPORT Download { // Call this when you want the modifications of the download priorities // in the entries to take effect. It is slightly expensive as it rechecks // all the peer bitfields to see if we are still interested. - void update_priorities(); + void update_priorities(int flags = 0); void add_peer(const sockaddr* addr, int port); diff --git a/src/torrent/torrent.cc b/src/torrent/torrent.cc index 339c2c4f8..865745ac4 100644 --- a/src/torrent/torrent.cc +++ b/src/torrent/torrent.cc @@ -164,7 +164,7 @@ encoding_list() { } Download -download_add(Object* object) { +download_add(Object* object, int flags) { std::auto_ptr download(new DownloadWrapper); DownloadConstructor ctor; @@ -190,7 +190,7 @@ download_add(Object* object) { } download->set_hash_queue(manager->hash_queue()); - download->initialize(infoHash, PEER_NAME + rak::generate_random(20 - std::string(PEER_NAME).size())); + download->initialize(infoHash, PEER_NAME + rak::generate_random(20 - std::string(PEER_NAME).size()), flags); // Add trackers, etc, after setting the info hash so that log // entries look sane. diff --git a/src/torrent/torrent.h b/src/torrent/torrent.h index 7bcf88feb..3f61dd658 100644 --- a/src/torrent/torrent.h +++ b/src/torrent/torrent.h @@ -93,7 +93,7 @@ EncodingList* encoding_list() LIBTORRENT_EXPORT; // is done by 'download_remove'. // // Might consider redesigning that... -Download download_add(Object* s) LIBTORRENT_EXPORT; +Download download_add(Object* s, int flags = 0) LIBTORRENT_EXPORT; void download_remove(Download d) LIBTORRENT_EXPORT; // Add all downloads to dlist. The client is responsible for clearing