2006-11-14 14:36:10 +01:00
|
|
|
/*
|
|
|
|
|
2016-01-18 00:57:46 +01:00
|
|
|
Copyright (c) 2006-2016, Arvid Norberg
|
2006-11-14 14:36:10 +01:00
|
|
|
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.
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
2015-08-20 01:33:20 +02:00
|
|
|
#include "libtorrent/config.hpp"
|
|
|
|
|
2009-11-25 07:55:34 +01:00
|
|
|
#include "libtorrent/assert.hpp"
|
2006-11-14 14:36:10 +01:00
|
|
|
#include "libtorrent/file_pool.hpp"
|
2008-07-18 01:41:46 +02:00
|
|
|
#include "libtorrent/error_code.hpp"
|
2016-10-25 23:27:48 +02:00
|
|
|
#include "libtorrent/file_storage.hpp"
|
2016-12-22 16:42:33 +01:00
|
|
|
#include "libtorrent/units.hpp"
|
2016-12-28 19:12:20 +01:00
|
|
|
#ifdef TORRENT_WINDOWS
|
|
|
|
#include "libtorrent/aux_/win_util.hpp"
|
|
|
|
#endif
|
2006-11-14 14:36:10 +01:00
|
|
|
|
2016-08-13 03:31:55 +02:00
|
|
|
#include <limits>
|
|
|
|
|
2006-11-14 14:36:10 +01:00
|
|
|
namespace libtorrent
|
|
|
|
{
|
2011-08-07 06:41:10 +02:00
|
|
|
file_pool::file_pool(int size)
|
|
|
|
: m_size(size)
|
|
|
|
, m_low_prio_io(true)
|
2014-01-06 04:50:25 +01:00
|
|
|
{
|
|
|
|
}
|
2011-08-07 06:41:10 +02:00
|
|
|
|
2016-07-10 13:34:45 +02:00
|
|
|
file_pool::~file_pool() = default;
|
2011-08-07 06:41:10 +02:00
|
|
|
|
2013-01-07 02:56:40 +01:00
|
|
|
#ifdef TORRENT_WINDOWS
|
2014-07-06 21:18:00 +02:00
|
|
|
void set_low_priority(file_handle const& f)
|
2013-01-07 02:56:40 +01:00
|
|
|
{
|
|
|
|
// file prio is only supported on vista and up
|
|
|
|
// so load the functions dynamically
|
2016-04-17 22:56:07 +02:00
|
|
|
typedef enum {
|
2013-01-07 02:56:40 +01:00
|
|
|
FileBasicInfo,
|
|
|
|
FileStandardInfo,
|
|
|
|
FileNameInfo,
|
|
|
|
FileRenameInfo,
|
|
|
|
FileDispositionInfo,
|
|
|
|
FileAllocationInfo,
|
|
|
|
FileEndOfFileInfo,
|
|
|
|
FileStreamInfo,
|
|
|
|
FileCompressionInfo,
|
|
|
|
FileAttributeTagInfo,
|
|
|
|
FileIdBothDirectoryInfo,
|
|
|
|
FileIdBothDirectoryRestartInfo,
|
|
|
|
FileIoPriorityHintInfo,
|
2015-08-18 16:42:03 +02:00
|
|
|
FileRemoteProtocolInfo,
|
2013-01-07 02:56:40 +01:00
|
|
|
MaximumFileInfoByHandleClass
|
2016-04-17 22:56:07 +02:00
|
|
|
} FILE_INFO_BY_HANDLE_CLASS_LOCAL;
|
2013-01-07 02:56:40 +01:00
|
|
|
|
2016-04-17 22:56:07 +02:00
|
|
|
typedef enum {
|
2013-01-07 02:56:40 +01:00
|
|
|
IoPriorityHintVeryLow = 0,
|
|
|
|
IoPriorityHintLow,
|
|
|
|
IoPriorityHintNormal,
|
|
|
|
MaximumIoPriorityHintType
|
2016-04-17 22:56:07 +02:00
|
|
|
} PRIORITY_HINT_LOCAL;
|
2013-01-07 02:56:40 +01:00
|
|
|
|
2016-04-17 22:56:07 +02:00
|
|
|
typedef struct {
|
|
|
|
PRIORITY_HINT_LOCAL PriorityHint;
|
|
|
|
} FILE_IO_PRIORITY_HINT_INFO_LOCAL;
|
2013-01-07 02:56:40 +01:00
|
|
|
|
2016-04-17 22:56:07 +02:00
|
|
|
typedef BOOL (WINAPI *SetFileInformationByHandle_t)(HANDLE hFile, FILE_INFO_BY_HANDLE_CLASS_LOCAL FileInformationClass, LPVOID lpFileInformation, DWORD dwBufferSize);
|
2016-12-28 19:12:20 +01:00
|
|
|
auto SetFileInformationByHandle =
|
|
|
|
aux::get_library_procedure<aux::kernel32, SetFileInformationByHandle_t>("SetFileInformationByHandle");
|
|
|
|
|
|
|
|
if (SetFileInformationByHandle == nullptr) return;
|
2013-01-07 02:56:40 +01:00
|
|
|
|
2016-04-17 22:56:07 +02:00
|
|
|
FILE_IO_PRIORITY_HINT_INFO_LOCAL io_hint;
|
2013-01-07 02:56:40 +01:00
|
|
|
io_hint.PriorityHint = IoPriorityHintLow;
|
|
|
|
SetFileInformationByHandle(f->native_handle(),
|
|
|
|
FileIoPriorityHintInfo, &io_hint, sizeof(io_hint));
|
|
|
|
}
|
|
|
|
#endif // TORRENT_WINDOWS
|
|
|
|
|
2014-07-06 21:18:00 +02:00
|
|
|
file_handle file_pool::open_file(void* st, std::string const& p
|
2016-12-22 16:42:33 +01:00
|
|
|
, file_index_t const file_index, file_storage const& fs, int m, error_code& ec)
|
2006-11-14 14:36:10 +01:00
|
|
|
{
|
2015-02-08 15:44:33 +01:00
|
|
|
// potentially used to hold a reference to a file object that's
|
|
|
|
// about to be destructed. If we have such object we assign it to
|
2016-05-01 00:54:23 +02:00
|
|
|
// this member to be destructed after we release the std::mutex. On some
|
2015-02-08 15:44:33 +01:00
|
|
|
// operating systems (such as OSX) closing a file may take a long
|
2016-05-01 00:54:23 +02:00
|
|
|
// time. We don't want to hold the std::mutex for that.
|
2015-02-08 15:44:33 +01:00
|
|
|
file_handle defer_destruction;
|
|
|
|
|
2016-05-01 00:54:23 +02:00
|
|
|
std::unique_lock<std::mutex> l(m_mutex);
|
2014-07-06 21:18:00 +02:00
|
|
|
|
|
|
|
#if TORRENT_USE_ASSERTS
|
|
|
|
// we're not allowed to open a file
|
|
|
|
// from a deleted storage!
|
2015-08-18 16:42:03 +02:00
|
|
|
TORRENT_ASSERT(std::find(m_deleted_storages.begin(), m_deleted_storages.end()
|
|
|
|
, std::make_pair(fs.name(), static_cast<void const*>(&fs)))
|
2014-07-06 21:18:00 +02:00
|
|
|
== m_deleted_storages.end());
|
|
|
|
#endif
|
|
|
|
|
2016-07-09 22:26:26 +02:00
|
|
|
TORRENT_ASSERT(st != nullptr);
|
2009-10-26 02:29:39 +01:00
|
|
|
TORRENT_ASSERT(is_complete(p));
|
2009-01-11 03:02:34 +01:00
|
|
|
TORRENT_ASSERT((m & file::rw_mask) == file::read_only
|
|
|
|
|| (m & file::rw_mask) == file::read_write);
|
2016-10-25 23:27:48 +02:00
|
|
|
auto const i = m_files.find(std::make_pair(st, file_index));
|
2009-05-09 23:11:25 +02:00
|
|
|
if (i != m_files.end())
|
2006-11-14 14:36:10 +01:00
|
|
|
{
|
2009-05-09 23:11:25 +02:00
|
|
|
lru_file_entry& e = i->second;
|
2015-03-12 05:34:54 +01:00
|
|
|
e.last_use = aux::time_now();
|
2006-11-14 14:36:10 +01:00
|
|
|
|
2009-01-25 02:58:49 +01:00
|
|
|
if (e.key != st && ((e.mode & file::rw_mask) != file::read_only
|
|
|
|
|| (m & file::rw_mask) != file::read_only))
|
2007-04-15 09:15:49 +02:00
|
|
|
{
|
|
|
|
// this means that another instance of the storage
|
|
|
|
// is using the exact same file.
|
2009-11-29 08:06:38 +01:00
|
|
|
ec = errors::file_collision;
|
2014-07-06 21:18:00 +02:00
|
|
|
return file_handle();
|
2007-04-15 09:15:49 +02:00
|
|
|
}
|
2006-11-14 14:36:10 +01:00
|
|
|
|
|
|
|
e.key = st;
|
2008-10-19 07:03:17 +02:00
|
|
|
// if we asked for a file in write mode,
|
|
|
|
// and the cached file is is not opened in
|
|
|
|
// write mode, re-open it
|
2010-01-31 21:44:05 +01:00
|
|
|
if ((((e.mode & file::rw_mask) != file::read_write)
|
2009-01-11 03:02:34 +01:00
|
|
|
&& ((m & file::rw_mask) == file::read_write))
|
2011-04-25 04:15:18 +02:00
|
|
|
|| (e.mode & file::random_access) != (m & file::random_access))
|
2006-11-14 14:36:10 +01:00
|
|
|
{
|
|
|
|
// close the file before we open it with
|
2016-04-14 20:03:11 +02:00
|
|
|
// the new read/write privileges, since windows may
|
2014-07-06 21:18:00 +02:00
|
|
|
// file opening a file twice. However, since there may
|
|
|
|
// be outstanding operations on it, we can't close the
|
|
|
|
// file, we can only delete our reference to it.
|
|
|
|
// if this is the only reference to the file, it will be closed
|
2015-02-08 15:44:33 +01:00
|
|
|
defer_destruction = e.file_ptr;
|
2016-09-01 03:42:18 +02:00
|
|
|
e.file_ptr = std::make_shared<file>();
|
2014-07-06 21:18:00 +02:00
|
|
|
|
2013-08-12 09:30:57 +02:00
|
|
|
std::string full_path = fs.file_path(file_index, p);
|
2010-08-23 10:48:02 +02:00
|
|
|
if (!e.file_ptr->open(full_path, m, ec))
|
2008-02-25 05:41:21 +01:00
|
|
|
{
|
|
|
|
m_files.erase(i);
|
2014-07-06 21:18:00 +02:00
|
|
|
return file_handle();
|
2008-02-25 05:41:21 +01:00
|
|
|
}
|
2010-01-23 04:02:32 +01:00
|
|
|
#ifdef TORRENT_WINDOWS
|
|
|
|
if (m_low_prio_io)
|
2013-01-07 02:56:40 +01:00
|
|
|
set_low_priority(e.file_ptr);
|
2010-01-23 04:02:32 +01:00
|
|
|
#endif
|
2014-07-06 21:18:00 +02:00
|
|
|
|
2008-07-20 18:34:01 +02:00
|
|
|
TORRENT_ASSERT(e.file_ptr->is_open());
|
2006-11-14 14:36:10 +01:00
|
|
|
e.mode = m;
|
|
|
|
}
|
|
|
|
return e.file_ptr;
|
|
|
|
}
|
2014-07-06 21:18:00 +02:00
|
|
|
|
2008-02-25 05:41:21 +01:00
|
|
|
lru_file_entry e;
|
2016-09-01 03:42:18 +02:00
|
|
|
e.file_ptr = std::make_shared<file>();
|
2008-02-25 05:41:21 +01:00
|
|
|
if (!e.file_ptr)
|
|
|
|
{
|
2015-11-22 19:12:11 +01:00
|
|
|
ec = error_code(boost::system::errc::not_enough_memory, generic_category());
|
2008-02-25 05:41:21 +01:00
|
|
|
return e.file_ptr;
|
|
|
|
}
|
2013-08-12 09:30:57 +02:00
|
|
|
std::string full_path = fs.file_path(file_index, p);
|
2010-08-23 10:48:02 +02:00
|
|
|
if (!e.file_ptr->open(full_path, m, ec))
|
2014-07-06 21:18:00 +02:00
|
|
|
return file_handle();
|
2013-01-07 02:56:40 +01:00
|
|
|
#ifdef TORRENT_WINDOWS
|
|
|
|
if (m_low_prio_io)
|
|
|
|
set_low_priority(e.file_ptr);
|
|
|
|
#endif
|
2006-11-14 14:36:10 +01:00
|
|
|
e.mode = m;
|
|
|
|
e.key = st;
|
2013-08-12 09:30:57 +02:00
|
|
|
m_files.insert(std::make_pair(std::make_pair(st, file_index), e));
|
2008-07-20 18:34:01 +02:00
|
|
|
TORRENT_ASSERT(e.file_ptr->is_open());
|
2014-07-06 21:18:00 +02:00
|
|
|
|
|
|
|
file_handle file_ptr = e.file_ptr;
|
|
|
|
|
|
|
|
// the file is not in our cache
|
2015-08-18 16:42:03 +02:00
|
|
|
if (int(m_files.size()) >= m_size)
|
2014-07-06 21:18:00 +02:00
|
|
|
{
|
|
|
|
// the file cache is at its maximum size, close
|
|
|
|
// the least recently used (lru) file from it
|
|
|
|
remove_oldest(l);
|
|
|
|
}
|
|
|
|
return file_ptr;
|
2006-11-14 14:36:10 +01:00
|
|
|
}
|
|
|
|
|
2016-08-14 19:11:59 +02:00
|
|
|
std::vector<pool_file_status> file_pool::get_status(void* st) const
|
2014-07-06 21:18:00 +02:00
|
|
|
{
|
2016-08-14 19:11:59 +02:00
|
|
|
std::vector<pool_file_status> ret;
|
|
|
|
{
|
|
|
|
std::unique_lock<std::mutex> l(m_mutex);
|
2014-07-06 21:18:00 +02:00
|
|
|
|
2016-12-22 16:42:33 +01:00
|
|
|
auto const start = m_files.lower_bound(std::make_pair(st, file_index_t(0)));
|
2016-10-25 23:27:48 +02:00
|
|
|
auto const end = m_files.upper_bound(std::make_pair(st
|
2016-12-22 16:42:33 +01:00
|
|
|
, std::numeric_limits<file_index_t>::max()));
|
2015-08-18 16:42:03 +02:00
|
|
|
|
2016-10-25 23:27:48 +02:00
|
|
|
for (auto i = start; i != end; ++i)
|
2016-08-14 19:11:59 +02:00
|
|
|
ret.push_back({i->first.second, i->second.mode, i->second.last_use});
|
2014-07-06 21:18:00 +02:00
|
|
|
}
|
2016-08-14 19:11:59 +02:00
|
|
|
return ret;
|
2014-07-06 21:18:00 +02:00
|
|
|
}
|
|
|
|
|
2016-05-01 00:54:23 +02:00
|
|
|
void file_pool::remove_oldest(std::unique_lock<std::mutex>& l)
|
2009-05-09 23:11:25 +02:00
|
|
|
{
|
2016-12-22 16:42:33 +01:00
|
|
|
using value_type = decltype(m_files)::value_type;
|
2016-10-25 23:27:48 +02:00
|
|
|
auto const i = std::min_element(m_files.begin(), m_files.end()
|
|
|
|
, [] (value_type const& lhs, value_type const& rhs)
|
2016-05-25 06:31:52 +02:00
|
|
|
{ return lhs.second.last_use < rhs.second.last_use; });
|
2009-05-09 23:11:25 +02:00
|
|
|
if (i == m_files.end()) return;
|
2011-08-07 06:41:10 +02:00
|
|
|
|
2014-07-06 21:18:00 +02:00
|
|
|
file_handle file_ptr = i->second.file_ptr;
|
2009-05-09 23:11:25 +02:00
|
|
|
m_files.erase(i);
|
2014-07-06 21:18:00 +02:00
|
|
|
|
|
|
|
// closing a file may be long running operation (mac os x)
|
|
|
|
l.unlock();
|
|
|
|
file_ptr.reset();
|
|
|
|
l.lock();
|
2009-05-09 23:11:25 +02:00
|
|
|
}
|
|
|
|
|
2016-12-22 16:42:33 +01:00
|
|
|
void file_pool::release(void* st, file_index_t file_index)
|
2008-05-28 10:44:40 +02:00
|
|
|
{
|
2016-05-01 00:54:23 +02:00
|
|
|
std::unique_lock<std::mutex> l(m_mutex);
|
2014-07-06 21:18:00 +02:00
|
|
|
|
2016-10-25 23:27:48 +02:00
|
|
|
auto const i = m_files.find(std::make_pair(st, file_index));
|
2011-08-07 06:41:10 +02:00
|
|
|
if (i == m_files.end()) return;
|
2016-05-25 06:31:52 +02:00
|
|
|
|
2014-07-06 21:18:00 +02:00
|
|
|
file_handle file_ptr = i->second.file_ptr;
|
2011-08-07 06:41:10 +02:00
|
|
|
m_files.erase(i);
|
2014-07-06 21:18:00 +02:00
|
|
|
|
|
|
|
// closing a file may be long running operation (mac os x)
|
|
|
|
l.unlock();
|
|
|
|
file_ptr.reset();
|
2008-05-28 10:44:40 +02:00
|
|
|
}
|
|
|
|
|
2010-01-23 04:02:32 +01:00
|
|
|
// closes files belonging to the specified
|
|
|
|
// storage. If 0 is passed, all files are closed
|
2006-11-14 14:36:10 +01:00
|
|
|
void file_pool::release(void* st)
|
|
|
|
{
|
2016-05-01 00:54:23 +02:00
|
|
|
std::unique_lock<std::mutex> l(m_mutex);
|
2014-07-06 21:18:00 +02:00
|
|
|
|
2016-07-09 22:26:26 +02:00
|
|
|
if (st == nullptr)
|
2010-01-23 04:02:32 +01:00
|
|
|
{
|
2016-12-22 16:42:33 +01:00
|
|
|
std::map<std::pair<void*, file_index_t>, lru_file_entry> tmp;
|
2014-07-06 21:18:00 +02:00
|
|
|
tmp.swap(m_files);
|
|
|
|
l.unlock();
|
2010-01-23 04:02:32 +01:00
|
|
|
return;
|
|
|
|
}
|
2006-11-14 14:36:10 +01:00
|
|
|
|
2014-07-06 21:18:00 +02:00
|
|
|
std::vector<file_handle> to_close;
|
2016-10-25 23:27:48 +02:00
|
|
|
for (auto i = m_files.begin(); i != m_files.end();)
|
2009-05-09 23:11:25 +02:00
|
|
|
{
|
|
|
|
if (i->second.key == st)
|
2014-07-06 21:18:00 +02:00
|
|
|
{
|
|
|
|
to_close.push_back(i->second.file_ptr);
|
2009-05-09 23:11:25 +02:00
|
|
|
m_files.erase(i++);
|
2014-07-06 21:18:00 +02:00
|
|
|
}
|
2009-05-09 23:11:25 +02:00
|
|
|
else
|
|
|
|
++i;
|
|
|
|
}
|
2014-07-06 21:18:00 +02:00
|
|
|
l.unlock();
|
|
|
|
// the files are closed here
|
|
|
|
}
|
|
|
|
|
|
|
|
#if TORRENT_USE_ASSERTS
|
|
|
|
void file_pool::mark_deleted(file_storage const& fs)
|
|
|
|
{
|
2016-05-01 00:54:23 +02:00
|
|
|
std::unique_lock<std::mutex> l(m_mutex);
|
2015-08-19 01:39:01 +02:00
|
|
|
m_deleted_storages.push_back(std::make_pair(fs.name()
|
|
|
|
, static_cast<void const*>(&fs)));
|
2014-07-06 21:18:00 +02:00
|
|
|
if(m_deleted_storages.size() > 100)
|
|
|
|
m_deleted_storages.erase(m_deleted_storages.begin());
|
2006-11-14 14:36:10 +01:00
|
|
|
}
|
|
|
|
|
2014-07-06 21:18:00 +02:00
|
|
|
bool file_pool::assert_idle_files(void* st) const
|
|
|
|
{
|
2016-05-01 00:54:23 +02:00
|
|
|
std::unique_lock<std::mutex> l(m_mutex);
|
2014-07-06 21:18:00 +02:00
|
|
|
|
2016-10-25 23:27:48 +02:00
|
|
|
for (auto const& i : m_files)
|
2014-07-06 21:18:00 +02:00
|
|
|
{
|
2016-10-25 23:27:48 +02:00
|
|
|
if (i.second.key == st && !i.second.file_ptr.unique())
|
2014-07-06 21:18:00 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2006-11-14 16:53:38 +01:00
|
|
|
void file_pool::resize(int size)
|
|
|
|
{
|
2016-05-01 00:54:23 +02:00
|
|
|
std::unique_lock<std::mutex> l(m_mutex);
|
2014-07-06 21:18:00 +02:00
|
|
|
|
2007-10-05 02:30:00 +02:00
|
|
|
TORRENT_ASSERT(size > 0);
|
2014-01-06 04:50:25 +01:00
|
|
|
|
|
|
|
if (size == m_size) return;
|
2006-11-14 16:53:38 +01:00
|
|
|
m_size = size;
|
|
|
|
if (int(m_files.size()) <= m_size) return;
|
|
|
|
|
|
|
|
// close the least recently used files
|
|
|
|
while (int(m_files.size()) > m_size)
|
2014-07-06 21:18:00 +02:00
|
|
|
remove_oldest(l);
|
2006-11-14 16:53:38 +01:00
|
|
|
}
|
|
|
|
|
2006-11-14 14:36:10 +01:00
|
|
|
}
|