/* Copyright (c) 2012-2018, Arvid Norberg All rights reserved. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: * Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. * Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. * Neither the name of the author nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #ifndef TORRENT_DISK_INTERFACE_HPP #define TORRENT_DISK_INTERFACE_HPP #include "libtorrent/bdecode.hpp" #include #include #include "libtorrent/fwd.hpp" #include "libtorrent/units.hpp" #include "libtorrent/disk_buffer_holder.hpp" #include "libtorrent/aux_/vector.hpp" #include "libtorrent/aux_/export.hpp" #include "libtorrent/storage_defs.hpp" #include "libtorrent/time.hpp" #include "libtorrent/sha1_hash.hpp" #include "libtorrent/flags.hpp" #include "libtorrent/session_types.hpp" namespace libtorrent { struct disk_observer; struct counters; struct storage_holder; using file_open_mode_t = flags::bitfield_flag; // this is a bittorrent constant constexpr int default_block_size = 0x4000; namespace file_open_mode { // open the file for reading only constexpr file_open_mode_t read_only{}; // open the file for writing only constexpr file_open_mode_t write_only = 0_bit; // open the file for reading and writing constexpr file_open_mode_t read_write = 1_bit; // the mask for the bits determining read or write mode constexpr file_open_mode_t rw_mask = read_only | write_only | read_write; // open the file in sparse mode (if supported by the // filesystem). constexpr file_open_mode_t sparse = 2_bit; // don't update the access timestamps on the file (if // supported by the operating system and filesystem). // this generally improves disk performance. constexpr file_open_mode_t no_atime = 3_bit; // open the file for random access. This disables read-ahead // logic constexpr file_open_mode_t random_access = 5_bit; #if TORRENT_ABI_VERSION == 1 // prevent the file from being opened by another process // while it's still being held open by this handle constexpr file_open_mode_t TORRENT_DEPRECATED locked = 6_bit; #endif } // this contains information about a file that's currently open by the // libtorrent disk I/O subsystem. It's associated with a single torrent. struct TORRENT_EXPORT open_file_state { // the index of the file this entry refers to into the ``file_storage`` // file list of this torrent. This starts indexing at 0. file_index_t file_index; // ``open_mode`` is a bitmask of the file flags this file is currently // opened with. These are the flags used in the ``file::open()`` function. // The flags used in this bitfield are defined by the file_open_mode enum. // // Note that the read/write mode is not a bitmask. The two least significant bits are used // to represent the read/write mode. Those bits can be masked out using the ``rw_mask`` constant. file_open_mode_t open_mode; // a (high precision) timestamp of when the file was last used. time_point last_use; }; #if TORRENT_ABI_VERSION == 1 using pool_file_status = open_file_state; #endif using disk_job_flags_t = flags::bitfield_flag; struct TORRENT_EXTRA_EXPORT disk_interface { // force making a copy of the cached block, rather // than getting a reference to the block already in // the cache. static constexpr disk_job_flags_t force_copy = 0_bit; // hint that there may be more disk operations with sequential access to // the file static constexpr disk_job_flags_t sequential_access = 3_bit; // don't keep the read block in cache static constexpr disk_job_flags_t volatile_read = 4_bit; // this flag is set on a job when a read operation did // not hit the disk, but found the data in the read cache. static constexpr disk_job_flags_t cache_hit = 5_bit; virtual storage_holder new_torrent(storage_constructor_type sc , storage_params p, std::shared_ptr const&) = 0; virtual void remove_torrent(storage_index_t) = 0; virtual storage_interface* get_torrent(storage_index_t) = 0; virtual void async_read(storage_index_t storage, peer_request const& r , std::function handler , disk_job_flags_t flags = {}) = 0; virtual bool async_write(storage_index_t storage, peer_request const& r , char const* buf, std::shared_ptr o , std::function handler , disk_job_flags_t flags = {}) = 0; virtual void async_hash(storage_index_t storage, piece_index_t piece, disk_job_flags_t flags , std::function handler) = 0; virtual void async_move_storage(storage_index_t storage, std::string p, move_flags_t flags , std::function handler) = 0; virtual void async_release_files(storage_index_t storage , std::function handler = std::function()) = 0; virtual void async_check_files(storage_index_t storage , add_torrent_params const* resume_data , aux::vector& links , std::function handler) = 0; virtual void async_flush_piece(storage_index_t storage, piece_index_t piece , std::function handler = std::function()) = 0; virtual void async_stop_torrent(storage_index_t storage , std::function handler = std::function()) = 0; virtual void async_rename_file(storage_index_t storage , file_index_t index, std::string name , std::function handler) = 0; virtual void async_delete_files(storage_index_t storage, remove_flags_t options , std::function handler) = 0; virtual void async_set_file_priority(storage_index_t storage , aux::vector prio , std::function handler) = 0; virtual void async_clear_piece(storage_index_t storage, piece_index_t index , std::function handler) = 0; virtual void clear_piece(storage_index_t storage, piece_index_t index) = 0; virtual void update_stats_counters(counters& c) const = 0; virtual void get_cache_info(cache_status* ret, storage_index_t storage , bool no_pieces = true, bool session = true) const = 0; virtual std::vector get_status(storage_index_t) const = 0; virtual void submit_jobs() = 0; #if TORRENT_USE_ASSERTS virtual bool is_disk_buffer(char* buffer) const = 0; #endif protected: ~disk_interface() {} }; struct storage_holder { storage_holder() = default; storage_holder(storage_index_t idx, disk_interface& disk_io) : m_disk_io(&disk_io) , m_idx(idx) {} ~storage_holder() { if (m_disk_io) m_disk_io->remove_torrent(m_idx); } explicit operator bool() const { return m_disk_io != nullptr; } operator storage_index_t() const { TORRENT_ASSERT(m_disk_io); return m_idx; } void reset() { if (m_disk_io) m_disk_io->remove_torrent(m_idx); m_disk_io = nullptr; } storage_holder(storage_holder const&) = delete; storage_holder& operator=(storage_holder const&) = delete; storage_holder(storage_holder&& rhs) noexcept : m_disk_io(rhs.m_disk_io) , m_idx(rhs.m_idx) { rhs.m_disk_io = nullptr; } storage_holder& operator=(storage_holder&& rhs) noexcept { if (m_disk_io) m_disk_io->remove_torrent(m_idx); m_disk_io = rhs.m_disk_io; m_idx = rhs.m_idx; rhs.m_disk_io = nullptr; return *this; } private: disk_interface* m_disk_io = nullptr; storage_index_t m_idx{0}; }; } #endif