premiere-libtorrent/include/libtorrent/disk_interface.hpp

249 lines
8.8 KiB
C++

/*
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 <string>
#include <memory>
#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<std::uint8_t, struct file_open_mode_tag>;
// internal
// 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.
// For possible flags, see file_open_mode_t.
//
// 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<std::uint8_t, struct disk_job_flags_tag>;
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<void> 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<void(disk_buffer_holder block, disk_job_flags_t flags, storage_error const& se)> 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<disk_observer> o
, std::function<void(storage_error const&)> 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<void(piece_index_t, sha1_hash const&, storage_error const&)> handler) = 0;
virtual void async_move_storage(storage_index_t storage, std::string p, move_flags_t flags
, std::function<void(status_t, std::string const&, storage_error const&)> handler) = 0;
virtual void async_release_files(storage_index_t storage
, std::function<void()> handler = std::function<void()>()) = 0;
virtual void async_check_files(storage_index_t storage
, add_torrent_params const* resume_data
, aux::vector<std::string, file_index_t>& links
, std::function<void(status_t, storage_error const&)> handler) = 0;
virtual void async_flush_piece(storage_index_t storage, piece_index_t piece
, std::function<void()> handler = std::function<void()>()) = 0;
virtual void async_stop_torrent(storage_index_t storage
, std::function<void()> handler = std::function<void()>()) = 0;
virtual void async_rename_file(storage_index_t storage
, file_index_t index, std::string name
, std::function<void(std::string const&, file_index_t, storage_error const&)> handler) = 0;
virtual void async_delete_files(storage_index_t storage, remove_flags_t options
, std::function<void(storage_error const&)> handler) = 0;
virtual void async_set_file_priority(storage_index_t storage
, aux::vector<download_priority_t, file_index_t> prio
, std::function<void(storage_error const&, aux::vector<download_priority_t, file_index_t>)> handler) = 0;
virtual void async_clear_piece(storage_index_t storage, piece_index_t index
, std::function<void(piece_index_t)> 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<open_file_state> 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