2004-01-16 03:57:45 +01:00
|
|
|
/*
|
|
|
|
|
|
|
|
Copyright (c) 2003, 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.
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
2007-03-17 18:15:16 +01:00
|
|
|
#include "libtorrent/pch.hpp"
|
2008-07-20 13:14:54 +02:00
|
|
|
#include "libtorrent/config.hpp"
|
2009-01-11 03:02:34 +01:00
|
|
|
#include "libtorrent/alloca.hpp"
|
2007-03-17 18:15:16 +01:00
|
|
|
|
2008-02-14 04:48:20 +01:00
|
|
|
#include <boost/scoped_ptr.hpp>
|
2008-07-20 13:14:54 +02:00
|
|
|
#ifdef TORRENT_WINDOWS
|
2004-09-24 12:50:03 +02:00
|
|
|
// windows part
|
2005-08-18 13:20:17 +02:00
|
|
|
#include "libtorrent/utf8.hpp"
|
|
|
|
|
2008-07-20 13:14:54 +02:00
|
|
|
#include <windows.h>
|
|
|
|
#include <winioctl.h>
|
2004-03-31 01:55:52 +02:00
|
|
|
|
|
|
|
#else
|
2008-07-20 13:14:54 +02:00
|
|
|
// posix part
|
2004-03-31 01:55:52 +02:00
|
|
|
#define _FILE_OFFSET_BITS 64
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <fcntl.h>
|
2004-04-02 12:43:37 +02:00
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <sys/types.h>
|
2009-01-17 10:37:40 +01:00
|
|
|
#include <sys/statvfs.h>
|
2004-10-31 09:20:05 +01:00
|
|
|
#include <errno.h>
|
2004-03-31 01:55:52 +02:00
|
|
|
|
2005-08-23 11:59:56 +02:00
|
|
|
#include <boost/static_assert.hpp>
|
|
|
|
// make sure the _FILE_OFFSET_BITS define worked
|
|
|
|
// on this platform
|
|
|
|
BOOST_STATIC_ASSERT(sizeof(lseek(0, 0, 0)) >= 8);
|
|
|
|
|
2004-03-31 01:55:52 +02:00
|
|
|
#endif
|
|
|
|
|
2005-08-23 11:59:56 +02:00
|
|
|
#include <boost/filesystem/operations.hpp>
|
|
|
|
#include "libtorrent/file.hpp"
|
|
|
|
#include <sstream>
|
2007-11-18 08:27:40 +01:00
|
|
|
#include <cstring>
|
2008-07-01 01:14:31 +02:00
|
|
|
#include <vector>
|
2005-08-23 11:59:56 +02:00
|
|
|
|
2008-12-09 04:45:58 +01:00
|
|
|
#ifdef TORRENT_USE_WPATH
|
|
|
|
// for safe_convert
|
2006-01-09 01:07:00 +01:00
|
|
|
#include "libtorrent/storage.hpp"
|
|
|
|
#endif
|
|
|
|
|
2007-10-06 19:27:53 +02:00
|
|
|
#include "libtorrent/assert.hpp"
|
2004-04-05 00:15:31 +02:00
|
|
|
|
2009-01-11 23:27:43 +01:00
|
|
|
#ifdef TORRENT_DEBUG
|
2009-01-11 03:02:34 +01:00
|
|
|
BOOST_STATIC_ASSERT((libtorrent::file::rw_mask & libtorrent::file::no_buffer) == 0);
|
2009-01-11 23:27:43 +01:00
|
|
|
BOOST_STATIC_ASSERT((libtorrent::file::rw_mask & libtorrent::file::attribute_mask) == 0);
|
|
|
|
BOOST_STATIC_ASSERT((libtorrent::file::no_buffer & libtorrent::file::attribute_mask) == 0);
|
|
|
|
#endif
|
2009-01-11 03:02:34 +01:00
|
|
|
|
2004-01-16 03:57:45 +01:00
|
|
|
namespace
|
|
|
|
{
|
2008-07-20 13:14:54 +02:00
|
|
|
#ifdef TORRENT_WINDOWS
|
2005-08-04 00:51:21 +02:00
|
|
|
std::string utf8_native(std::string const& s)
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
std::wstring ws;
|
|
|
|
libtorrent::utf8_wchar(s, ws);
|
|
|
|
std::size_t size = wcstombs(0, ws.c_str(), 0);
|
|
|
|
if (size == std::size_t(-1)) return s;
|
|
|
|
std::string ret;
|
|
|
|
ret.resize(size);
|
|
|
|
size = wcstombs(&ret[0], ws.c_str(), size + 1);
|
2008-09-25 02:16:41 +02:00
|
|
|
if (size == std::size_t(-1)) return s;
|
2005-08-04 00:51:21 +02:00
|
|
|
ret.resize(size);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
catch(std::exception)
|
|
|
|
{
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2004-01-16 03:57:45 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
namespace libtorrent
|
|
|
|
{
|
2007-06-10 22:46:09 +02:00
|
|
|
namespace fs = boost::filesystem;
|
|
|
|
|
2008-07-20 13:14:54 +02:00
|
|
|
file::file()
|
2008-07-18 01:41:46 +02:00
|
|
|
#ifdef TORRENT_WINDOWS
|
2008-07-20 13:14:54 +02:00
|
|
|
: m_file_handle(INVALID_HANDLE_VALUE)
|
2005-08-17 03:57:30 +02:00
|
|
|
#else
|
2008-07-20 13:14:54 +02:00
|
|
|
: m_fd(-1)
|
2008-07-18 01:41:46 +02:00
|
|
|
#endif
|
2008-07-20 13:14:54 +02:00
|
|
|
, m_open_mode(0)
|
2009-01-17 09:35:48 +01:00
|
|
|
#if defined TORRENT_WINDOWS || defined TORRENT_LINUX
|
|
|
|
, m_sector_size(0)
|
|
|
|
#endif
|
2008-07-20 13:14:54 +02:00
|
|
|
{}
|
2008-07-18 01:41:46 +02:00
|
|
|
|
2008-10-19 07:03:17 +02:00
|
|
|
file::file(fs::path const& path, int mode, error_code& ec)
|
2008-07-18 01:41:46 +02:00
|
|
|
#ifdef TORRENT_WINDOWS
|
2008-07-20 13:14:54 +02:00
|
|
|
: m_file_handle(INVALID_HANDLE_VALUE)
|
|
|
|
#else
|
|
|
|
: m_fd(-1)
|
|
|
|
#endif
|
|
|
|
, m_open_mode(0)
|
|
|
|
{
|
|
|
|
open(path, mode, ec);
|
|
|
|
}
|
2008-07-18 01:41:46 +02:00
|
|
|
|
2008-07-20 13:14:54 +02:00
|
|
|
file::~file()
|
|
|
|
{
|
|
|
|
close();
|
|
|
|
}
|
2004-01-16 03:57:45 +01:00
|
|
|
|
2008-10-19 07:03:17 +02:00
|
|
|
bool file::open(fs::path const& path, int mode, error_code& ec)
|
2008-07-20 13:14:54 +02:00
|
|
|
{
|
|
|
|
close();
|
|
|
|
#ifdef TORRENT_WINDOWS
|
2004-04-03 00:21:20 +02:00
|
|
|
|
2008-12-09 04:45:58 +01:00
|
|
|
#ifdef TORRENT_USE_WPATH
|
2009-01-11 03:02:34 +01:00
|
|
|
m_path = safe_convert(path.external_file_string());
|
2006-11-14 01:08:16 +01:00
|
|
|
#else
|
2009-01-11 03:02:34 +01:00
|
|
|
m_path = utf8_native(path.external_file_string());
|
2006-11-14 01:08:16 +01:00
|
|
|
#endif
|
2004-01-16 03:57:45 +01:00
|
|
|
|
2008-07-20 18:34:01 +02:00
|
|
|
m_file_handle = CreateFile(
|
2009-01-11 03:02:34 +01:00
|
|
|
m_path.c_str()
|
|
|
|
, mode & rw_mask
|
2009-01-11 19:01:28 +01:00
|
|
|
, FILE_SHARE_READ | ((mode & no_buffer) ? FILE_SHARE_WRITE : 0)
|
2008-07-20 13:14:54 +02:00
|
|
|
, 0
|
2009-01-11 03:02:34 +01:00
|
|
|
, ((mode & rw_mask) == read_write || (mode & rw_mask) == write_only)?OPEN_ALWAYS:OPEN_EXISTING
|
2009-01-11 23:27:43 +01:00
|
|
|
, FILE_FLAG_RANDOM_ACCESS
|
2009-01-11 19:01:28 +01:00
|
|
|
| ((mode & no_buffer)?FILE_FLAG_OVERLAPPED | FILE_FLAG_NO_BUFFERING:0)
|
2009-01-11 23:27:43 +01:00
|
|
|
| ((mode & attribute_mask)?(mode & attribute_mask):FILE_ATTRIBUTE_NORMAL)
|
2008-07-20 13:14:54 +02:00
|
|
|
, 0);
|
|
|
|
|
|
|
|
if (m_file_handle == INVALID_HANDLE_VALUE)
|
2004-01-16 03:57:45 +01:00
|
|
|
{
|
2008-07-20 13:14:54 +02:00
|
|
|
ec = error_code(GetLastError(), get_system_category());
|
|
|
|
return false;
|
|
|
|
}
|
2004-03-31 01:55:52 +02:00
|
|
|
|
2008-07-20 13:14:54 +02:00
|
|
|
// try to make the file sparse if supported
|
2009-01-11 03:02:34 +01:00
|
|
|
if ((mode & rw_mask) == write_only || (mode & rw_mask) == read_write)
|
2008-07-20 13:14:54 +02:00
|
|
|
{
|
|
|
|
DWORD temp;
|
|
|
|
::DeviceIoControl(m_file_handle, FSCTL_SET_SPARSE, 0, 0
|
|
|
|
, 0, 0, &temp, 0);
|
2004-01-16 03:57:45 +01:00
|
|
|
}
|
2008-07-20 13:14:54 +02:00
|
|
|
#else
|
|
|
|
// rely on default umask to filter x and w permissions
|
|
|
|
// for group and others
|
2008-07-22 15:02:06 +02:00
|
|
|
int permissions = S_IRUSR | S_IWUSR
|
|
|
|
| S_IRGRP | S_IWGRP
|
|
|
|
| S_IROTH | S_IWOTH;
|
|
|
|
|
2009-01-11 23:27:43 +01:00
|
|
|
if (mode & attribute_executable)
|
|
|
|
permissions |= S_IXGRP | S_IXOTH | S_IXUSR;
|
|
|
|
|
2008-11-22 00:38:07 +01:00
|
|
|
m_fd = ::open(path.external_file_string().c_str()
|
2009-01-11 23:27:43 +01:00
|
|
|
, mode & (rw_mask | no_buffer), permissions);
|
2004-01-16 03:57:45 +01:00
|
|
|
|
2009-01-17 10:37:40 +01:00
|
|
|
#ifdef TORRENT_LINUX
|
|
|
|
// workaround for linux bug
|
|
|
|
// https://bugs.launchpad.net/ubuntu/+source/linux/+bug/269946
|
|
|
|
if (m_fd == -1 && (mode & no_buffer) && errno == EINVAL)
|
|
|
|
{
|
|
|
|
mode &= ~no_buffer;
|
|
|
|
m_fd = ::open(path.external_file_string().c_str()
|
|
|
|
, mode & (rw_mask | no_buffer), permissions);
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
2008-07-20 13:14:54 +02:00
|
|
|
if (m_fd == -1)
|
2004-01-16 03:57:45 +01:00
|
|
|
{
|
2008-07-20 13:14:54 +02:00
|
|
|
ec = error_code(errno, get_posix_category());
|
|
|
|
return false;
|
|
|
|
}
|
2009-01-11 03:02:34 +01:00
|
|
|
|
|
|
|
#ifdef F_NOCACHE
|
|
|
|
if (mode & no_buffer)
|
|
|
|
{
|
|
|
|
int yes = 1;
|
|
|
|
fcntl(m_fd, F_NOCACHE, &yes);
|
|
|
|
}
|
2008-07-20 13:14:54 +02:00
|
|
|
#endif
|
2009-01-11 03:02:34 +01:00
|
|
|
|
|
|
|
#ifdef POSIX_FADV_RANDOM
|
|
|
|
// disable read-ahead
|
|
|
|
posix_fadvise(m_fd, 0, 0, POSIX_FADV_RANDOM);
|
2008-07-20 13:14:54 +02:00
|
|
|
#endif
|
2009-01-11 03:02:34 +01:00
|
|
|
|
|
|
|
#endif
|
|
|
|
m_open_mode = mode;
|
|
|
|
|
2008-07-20 18:34:01 +02:00
|
|
|
TORRENT_ASSERT(is_open());
|
2008-07-20 13:14:54 +02:00
|
|
|
return true;
|
|
|
|
}
|
2004-03-31 01:55:52 +02:00
|
|
|
|
2008-07-20 13:14:54 +02:00
|
|
|
bool file::is_open() const
|
|
|
|
{
|
|
|
|
#ifdef TORRENT_WINDOWS
|
|
|
|
return m_file_handle != INVALID_HANDLE_VALUE;
|
|
|
|
#else
|
|
|
|
return m_fd != -1;
|
|
|
|
#endif
|
|
|
|
}
|
2006-07-08 21:41:39 +02:00
|
|
|
|
2009-01-17 09:35:48 +01:00
|
|
|
int file::pos_alignment() const
|
|
|
|
{
|
|
|
|
// on linux and windows, file offsets needs
|
|
|
|
// to be aligned to the disk sector size
|
|
|
|
#if defined TORRENT_LINUX
|
|
|
|
if (m_sector_size == 0)
|
|
|
|
{
|
|
|
|
struct statvfs fs;
|
|
|
|
if (fstatvfs(m_fd, &fs) == 0)
|
|
|
|
m_sector_size = fs.f_bsize;
|
|
|
|
else
|
|
|
|
m_sector_size = 4096;
|
|
|
|
}
|
|
|
|
return m_sector_size;
|
|
|
|
#elif defined TORRENT_WINDOWS
|
|
|
|
if (m_sector_size == 0)
|
|
|
|
{
|
|
|
|
DWORD sectors_per_cluster;
|
|
|
|
DWORD bytes_per_sector;
|
|
|
|
DWORD free_clusters;
|
|
|
|
DWORD total_clusters;
|
|
|
|
#ifdef TORRENT_USE_WPATH
|
|
|
|
wchar_t backslash = L'\\';
|
|
|
|
#else
|
|
|
|
char backslash = '\\';
|
|
|
|
#endif
|
|
|
|
if (GetDiskFreeSpace(m_path.substr(0, m_path.find_first_of(backslash)+1).c_str()
|
|
|
|
, §ors_per_cluster, &bytes_per_sector
|
|
|
|
, &free_clusters, &total_clusters))
|
|
|
|
m_sector_size = bytes_per_sector;
|
|
|
|
else
|
|
|
|
m_sector_size = 4096;
|
|
|
|
}
|
|
|
|
return m_sector_size;
|
|
|
|
#else
|
|
|
|
return 1;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2008-07-20 13:14:54 +02:00
|
|
|
void file::close()
|
|
|
|
{
|
2009-01-17 09:35:48 +01:00
|
|
|
#if defined TORRENT_WINDOWS || defined TORRENT_LINUX
|
|
|
|
m_sector_size = 0;
|
|
|
|
#endif
|
|
|
|
|
2008-07-20 13:14:54 +02:00
|
|
|
#ifdef TORRENT_WINDOWS
|
|
|
|
if (m_file_handle == INVALID_HANDLE_VALUE) return;
|
|
|
|
CloseHandle(m_file_handle);
|
|
|
|
m_file_handle = INVALID_HANDLE_VALUE;
|
2009-01-11 03:02:34 +01:00
|
|
|
m_path.clear();
|
2006-11-14 01:08:16 +01:00
|
|
|
#else
|
2008-07-20 13:14:54 +02:00
|
|
|
if (m_fd == -1) return;
|
|
|
|
::close(m_fd);
|
|
|
|
m_fd = -1;
|
2006-11-14 01:08:16 +01:00
|
|
|
#endif
|
2008-07-20 13:14:54 +02:00
|
|
|
m_open_mode = 0;
|
2009-01-11 03:02:34 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// defined in storage.cpp
|
|
|
|
int bufs_size(file::iovec_t const* bufs, int num_bufs);
|
|
|
|
|
2009-01-17 01:09:44 +01:00
|
|
|
#if defined TORRENT_WINDOWS || defined TORRENT_LINUX || defined TORRENT_DEBUG
|
2009-01-11 03:02:34 +01:00
|
|
|
|
|
|
|
int file::m_page_size = 0;
|
|
|
|
|
|
|
|
void file::init_file()
|
|
|
|
{
|
|
|
|
if (m_page_size != 0) return;
|
|
|
|
|
|
|
|
#ifdef TORRENT_WINDOWS
|
|
|
|
SYSTEM_INFO si;
|
|
|
|
GetSystemInfo(&si);
|
|
|
|
m_page_size = si.dwPageSize;
|
|
|
|
#else
|
|
|
|
m_page_size = sysconf(_SC_PAGESIZE);
|
2008-07-20 13:14:54 +02:00
|
|
|
#endif
|
|
|
|
}
|
2004-01-16 03:57:45 +01:00
|
|
|
|
2009-01-11 03:02:34 +01:00
|
|
|
#endif
|
|
|
|
|
|
|
|
size_type file::readv(size_type file_offset, iovec_t const* bufs, int num_bufs, error_code& ec)
|
2008-07-20 13:14:54 +02:00
|
|
|
{
|
2009-01-11 03:02:34 +01:00
|
|
|
TORRENT_ASSERT((m_open_mode & rw_mask) == read_only || (m_open_mode & rw_mask) == read_write);
|
|
|
|
TORRENT_ASSERT(bufs);
|
|
|
|
TORRENT_ASSERT(num_bufs > 0);
|
2008-07-20 13:14:54 +02:00
|
|
|
TORRENT_ASSERT(is_open());
|
|
|
|
|
2009-01-17 01:09:44 +01:00
|
|
|
#if defined TORRENT_WINDOWS || defined TORRENT_LINUX || defined TORRENT_DEBUG
|
2009-01-11 03:02:34 +01:00
|
|
|
// make sure m_page_size is initialized
|
|
|
|
init_file();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef TORRENT_DEBUG
|
|
|
|
if (m_open_mode & no_buffer)
|
|
|
|
{
|
|
|
|
bool eof = false;
|
|
|
|
int size = 0;
|
2009-01-17 09:35:48 +01:00
|
|
|
// when opened in no_buffer mode, the file_offset must
|
|
|
|
// be aligned to pos_alignment()
|
|
|
|
TORRENT_ASSERT((file_offset & (pos_alignment()-1)) == 0);
|
2009-01-11 03:02:34 +01:00
|
|
|
for (file::iovec_t const* i = bufs, *end(bufs + num_bufs); i < end; ++i)
|
|
|
|
{
|
|
|
|
TORRENT_ASSERT((int(i->iov_base) & (m_page_size-1)) == 0);
|
|
|
|
// every buffer must be a multiple of the page size
|
|
|
|
// except for the last one
|
|
|
|
TORRENT_ASSERT((i->iov_len & (m_page_size-1)) == 0 || i == end-1);
|
|
|
|
if ((i->iov_len & (m_page_size-1)) != 0) eof = true;
|
|
|
|
size += i->iov_len;
|
|
|
|
}
|
|
|
|
error_code code;
|
|
|
|
if (eof) TORRENT_ASSERT(file_offset + size >= get_size(code));
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2008-07-20 13:14:54 +02:00
|
|
|
#ifdef TORRENT_WINDOWS
|
2009-01-11 03:02:34 +01:00
|
|
|
|
2008-07-20 13:14:54 +02:00
|
|
|
DWORD ret = 0;
|
2009-01-11 03:02:34 +01:00
|
|
|
|
|
|
|
// since the ReadFileScatter requires the file to be opened
|
|
|
|
// with no buffering, and no buffering requires page aligned
|
|
|
|
// buffers, open the file in non-buffered mode in case the
|
|
|
|
// buffer is not aligned. Most of the times the buffer should
|
|
|
|
// be aligned though
|
|
|
|
|
|
|
|
if ((m_open_mode & no_buffer) == 0)
|
2007-04-19 05:06:15 +02:00
|
|
|
{
|
2009-01-11 03:02:34 +01:00
|
|
|
// this means the buffer base or the buffer size is not aligned
|
|
|
|
// to the page size. Use a regular file for this operation.
|
|
|
|
|
|
|
|
LARGE_INTEGER offs;
|
|
|
|
offs.QuadPart = file_offset;
|
|
|
|
if (SetFilePointerEx(m_file_handle, offs, &offs, FILE_BEGIN) == FALSE)
|
2007-08-11 17:22:06 +02:00
|
|
|
{
|
2008-07-20 13:14:54 +02:00
|
|
|
ec = error_code(GetLastError(), get_system_category());
|
|
|
|
return -1;
|
2007-08-11 17:22:06 +02:00
|
|
|
}
|
2009-01-11 03:02:34 +01:00
|
|
|
|
|
|
|
for (file::iovec_t const* i = bufs, *end(bufs + num_bufs); i < end; ++i)
|
|
|
|
{
|
|
|
|
DWORD intermediate = 0;
|
2009-01-13 20:40:14 +01:00
|
|
|
if (ReadFile(m_file_handle, (char*)i->iov_base
|
|
|
|
, (DWORD)i->iov_len, &intermediate, 0) == FALSE)
|
2009-01-11 03:02:34 +01:00
|
|
|
{
|
|
|
|
ec = error_code(GetLastError(), get_system_category());
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
ret += intermediate;
|
|
|
|
}
|
|
|
|
return ret;
|
2007-04-19 05:06:15 +02:00
|
|
|
}
|
2004-04-05 00:15:31 +02:00
|
|
|
|
2009-01-11 03:02:34 +01:00
|
|
|
int size = bufs_size(bufs, num_bufs);
|
|
|
|
// number of pages for the read. round up
|
|
|
|
int num_pages = (size + m_page_size - 1) / m_page_size;
|
|
|
|
// allocate array of FILE_SEGMENT_ELEMENT for ReadFileScatter
|
|
|
|
FILE_SEGMENT_ELEMENT* segment_array = TORRENT_ALLOCA(FILE_SEGMENT_ELEMENT, num_pages + 1);
|
|
|
|
FILE_SEGMENT_ELEMENT* cur_seg = segment_array;
|
2009-01-01 02:47:57 +01:00
|
|
|
|
2009-01-11 03:02:34 +01:00
|
|
|
for (file::iovec_t const* i = bufs, *end(bufs + num_bufs); i < end; ++i)
|
2009-01-01 02:47:57 +01:00
|
|
|
{
|
2009-01-11 03:02:34 +01:00
|
|
|
for (int k = 0; k < i->iov_len; k += m_page_size)
|
|
|
|
{
|
|
|
|
cur_seg->Buffer = ((char*)i->iov_base) + k;
|
|
|
|
++cur_seg;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// terminate the array
|
|
|
|
cur_seg->Buffer = 0;
|
|
|
|
|
|
|
|
OVERLAPPED ol;
|
|
|
|
ol.Internal = 0;
|
|
|
|
ol.InternalHigh = 0;
|
|
|
|
ol.OffsetHigh = file_offset >> 32;
|
|
|
|
ol.Offset = file_offset & 0xffffffff;
|
|
|
|
ol.hEvent = CreateEvent(0, true, false, 0);
|
|
|
|
|
|
|
|
ret += size;
|
|
|
|
size = num_pages * m_page_size;
|
|
|
|
if (ReadFileScatter(m_file_handle, segment_array, size, 0, &ol) == 0)
|
|
|
|
{
|
|
|
|
DWORD last_error = GetLastError();
|
2009-01-16 19:51:54 +01:00
|
|
|
if (last_error != ERROR_IO_PENDING)
|
2009-01-01 02:47:57 +01:00
|
|
|
{
|
|
|
|
ec = error_code(GetLastError(), get_system_category());
|
2009-01-11 03:02:34 +01:00
|
|
|
CloseHandle(ol.hEvent);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
if (GetOverlappedResult(m_file_handle, &ol, &ret, true) == 0)
|
|
|
|
{
|
|
|
|
ec = error_code(GetLastError(), get_system_category());
|
|
|
|
CloseHandle(ol.hEvent);
|
2009-01-01 02:47:57 +01:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
2009-01-17 19:25:27 +01:00
|
|
|
CloseHandle(ol.hEvent);
|
2009-01-11 03:02:34 +01:00
|
|
|
return ret;
|
2009-01-01 02:47:57 +01:00
|
|
|
#else
|
2009-01-11 03:02:34 +01:00
|
|
|
size_type ret = lseek(m_fd, file_offset, SEEK_SET);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
ec = error_code(errno, get_posix_category());
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
#ifdef TORRENT_LINUX
|
|
|
|
bool aligned = false;
|
|
|
|
int size = 0;
|
|
|
|
// if we're not opened in no-buffer mode, we don't need alignment
|
|
|
|
if ((m_open_mode & no_buffer) == 0) aligned = true;
|
|
|
|
if (!aligned)
|
|
|
|
{
|
|
|
|
size = bufs_size(bufs, num_bufs);
|
|
|
|
if (size & (m_page_size-1) == 0) aligned = true;
|
|
|
|
}
|
|
|
|
if (aligned)
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
ret = ::readv(m_fd, bufs, num_bufs);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
ec = error_code(errno, get_posix_category());
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
#ifdef TORRENT_LINUX
|
|
|
|
file::iovec_t* temp_bufs = TORRENT_ALLOCA(file::iovec_t, num_bufs);
|
|
|
|
memcpy(temp_bufs, bufs, sizeof(file::iovec_t) * num_bufs);
|
|
|
|
iovec_t& last = temp_bufs[num_bufs-1];
|
|
|
|
last.iov_len = (last.iov_len & ~(m_page_size-1)) + m_page_size;
|
|
|
|
ret = ::readv(m_fd, temp_bufs, num_bufs);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
ec = error_code(errno, get_posix_category());
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return (std::min)(ret, size_type(size));
|
|
|
|
#endif
|
2009-01-01 02:47:57 +01:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2009-01-11 03:02:34 +01:00
|
|
|
size_type file::writev(size_type file_offset, iovec_t const* bufs, int num_bufs, error_code& ec)
|
2009-01-01 02:47:57 +01:00
|
|
|
{
|
2009-01-11 03:02:34 +01:00
|
|
|
TORRENT_ASSERT((m_open_mode & rw_mask) == write_only || (m_open_mode & rw_mask) == read_write);
|
2009-01-01 02:47:57 +01:00
|
|
|
TORRENT_ASSERT(bufs);
|
2009-01-11 03:02:34 +01:00
|
|
|
TORRENT_ASSERT(num_bufs > 0);
|
2009-01-01 02:47:57 +01:00
|
|
|
TORRENT_ASSERT(is_open());
|
|
|
|
|
2009-01-17 01:09:44 +01:00
|
|
|
#if defined TORRENT_WINDOWS || defined TORRENT_LINUX || defined TORRENT_DEBUG
|
2009-01-11 03:02:34 +01:00
|
|
|
// make sure m_page_size is initialized
|
|
|
|
init_file();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef TORRENT_DEBUG
|
|
|
|
if (m_open_mode & no_buffer)
|
|
|
|
{
|
|
|
|
bool eof = false;
|
|
|
|
int size = 0;
|
2009-01-17 09:35:48 +01:00
|
|
|
// when opened in no_buffer mode, the file_offset must
|
|
|
|
// be aligned to pos_alignment()
|
|
|
|
TORRENT_ASSERT((file_offset & (pos_alignment()-1)) == 0);
|
2009-01-11 03:02:34 +01:00
|
|
|
for (file::iovec_t const* i = bufs, *end(bufs + num_bufs); i < end; ++i)
|
|
|
|
{
|
|
|
|
TORRENT_ASSERT((int(i->iov_base) & (m_page_size-1)) == 0);
|
|
|
|
// every buffer must be a multiple of the page size
|
|
|
|
// except for the last one
|
|
|
|
TORRENT_ASSERT((i->iov_len & (m_page_size-1)) == 0 || i == end-1);
|
|
|
|
if ((i->iov_len & (m_page_size-1)) != 0) eof = true;
|
|
|
|
size += i->iov_len;
|
|
|
|
}
|
|
|
|
error_code code;
|
|
|
|
if (eof) TORRENT_ASSERT(file_offset + size >= get_size(code));
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2009-01-01 02:47:57 +01:00
|
|
|
#ifdef TORRENT_WINDOWS
|
2009-01-11 03:02:34 +01:00
|
|
|
|
|
|
|
DWORD ret = 0;
|
|
|
|
|
|
|
|
// since the ReadFileScatter requires the file to be opened
|
|
|
|
// with no buffering, and no buffering requires page aligned
|
|
|
|
// buffers, open the file in non-buffered mode in case the
|
|
|
|
// buffer is not aligned. Most of the times the buffer should
|
|
|
|
// be aligned though
|
|
|
|
|
|
|
|
if ((m_open_mode & no_buffer) == 0)
|
2009-01-01 02:47:57 +01:00
|
|
|
{
|
2009-01-11 03:02:34 +01:00
|
|
|
// this means the buffer base or the buffer size is not aligned
|
|
|
|
// to the page size. Use a regular file for this operation.
|
|
|
|
|
|
|
|
LARGE_INTEGER offs;
|
|
|
|
offs.QuadPart = file_offset;
|
|
|
|
if (SetFilePointerEx(m_file_handle, offs, &offs, SEEK_SET) == FALSE)
|
2009-01-01 02:47:57 +01:00
|
|
|
{
|
|
|
|
ec = error_code(GetLastError(), get_system_category());
|
|
|
|
return -1;
|
|
|
|
}
|
2009-01-11 03:02:34 +01:00
|
|
|
|
|
|
|
for (file::iovec_t const* i = bufs, *end(bufs + num_bufs); i < end; ++i)
|
|
|
|
{
|
|
|
|
DWORD intermediate = 0;
|
2009-01-13 20:40:14 +01:00
|
|
|
if (WriteFile(m_file_handle, (char const*)i->iov_base
|
|
|
|
, (DWORD)i->iov_len, &intermediate, 0) == FALSE)
|
2009-01-11 03:02:34 +01:00
|
|
|
{
|
|
|
|
ec = error_code(GetLastError(), get_system_category());
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
ret += intermediate;
|
|
|
|
}
|
|
|
|
return ret;
|
2009-01-01 02:47:57 +01:00
|
|
|
}
|
|
|
|
|
2009-01-11 03:02:34 +01:00
|
|
|
int size = bufs_size(bufs, num_bufs);
|
|
|
|
// number of pages for the write. round up
|
|
|
|
int num_pages = (size + m_page_size - 1) / m_page_size;
|
|
|
|
// allocate array of FILE_SEGMENT_ELEMENT for WriteFileGather
|
|
|
|
FILE_SEGMENT_ELEMENT* segment_array = TORRENT_ALLOCA(FILE_SEGMENT_ELEMENT, num_pages + 1);
|
|
|
|
FILE_SEGMENT_ELEMENT* cur_seg = segment_array;
|
2004-01-16 03:57:45 +01:00
|
|
|
|
2009-01-11 03:02:34 +01:00
|
|
|
for (file::iovec_t const* i = bufs, *end(bufs + num_bufs); i < end; ++i)
|
|
|
|
{
|
|
|
|
for (int k = 0; k < i->iov_len; k += m_page_size)
|
|
|
|
{
|
|
|
|
cur_seg->Buffer = ((char*)i->iov_base) + k;
|
|
|
|
++cur_seg;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// terminate the array
|
|
|
|
cur_seg->Buffer = 0;
|
|
|
|
|
|
|
|
OVERLAPPED ol;
|
|
|
|
ol.Internal = 0;
|
|
|
|
ol.InternalHigh = 0;
|
|
|
|
ol.OffsetHigh = file_offset >> 32;
|
|
|
|
ol.Offset = file_offset & 0xffffffff;
|
|
|
|
ol.hEvent = CreateEvent(0, true, false, 0);
|
|
|
|
|
|
|
|
ret += size;
|
|
|
|
// if file_size is > 0, the file will be opened in unbuffered
|
|
|
|
// mode after the write completes, and truncate the file to
|
|
|
|
// file_size.
|
|
|
|
size_type file_size = 0;
|
|
|
|
|
2009-01-11 10:59:05 +01:00
|
|
|
if ((size & (m_page_size-1)) != 0)
|
2009-01-11 03:02:34 +01:00
|
|
|
{
|
|
|
|
// if size is not an even multiple, this must be the tail
|
|
|
|
// of the file. Write the whole page and then open a new
|
|
|
|
// file without FILE_FLAG_NO_BUFFERING and set the
|
|
|
|
// file size to file_offset + size
|
|
|
|
|
|
|
|
file_size = file_offset + size;
|
|
|
|
size = num_pages * m_page_size;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (WriteFileGather(m_file_handle, segment_array, size, 0, &ol) == 0)
|
2008-07-20 13:14:54 +02:00
|
|
|
{
|
2009-01-11 03:02:34 +01:00
|
|
|
if (GetLastError() != ERROR_IO_PENDING)
|
2008-07-20 13:14:54 +02:00
|
|
|
{
|
|
|
|
ec = error_code(GetLastError(), get_system_category());
|
2009-01-11 03:02:34 +01:00
|
|
|
CloseHandle(ol.hEvent);
|
2008-07-20 13:14:54 +02:00
|
|
|
return -1;
|
|
|
|
}
|
2009-01-11 03:02:34 +01:00
|
|
|
if (GetOverlappedResult(m_file_handle, &ol, &ret, true) == 0)
|
|
|
|
{
|
|
|
|
ec = error_code(GetLastError(), get_system_category());
|
|
|
|
CloseHandle(ol.hEvent);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
CloseHandle(ol.hEvent);
|
|
|
|
|
|
|
|
if (file_size > 0)
|
|
|
|
{
|
|
|
|
HANDLE f = CreateFile(m_path.c_str(), GENERIC_WRITE
|
2009-01-11 20:54:20 +01:00
|
|
|
, FILE_SHARE_READ | FILE_SHARE_WRITE, 0, OPEN_EXISTING
|
2009-01-11 03:02:34 +01:00
|
|
|
, FILE_ATTRIBUTE_NORMAL | FILE_FLAG_RANDOM_ACCESS, 0);
|
|
|
|
|
|
|
|
if (f == INVALID_HANDLE_VALUE)
|
|
|
|
{
|
|
|
|
ec = error_code(GetLastError(), get_system_category());
|
2009-01-11 11:21:18 +01:00
|
|
|
return -1;
|
2009-01-11 03:02:34 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
LARGE_INTEGER offs;
|
|
|
|
offs.QuadPart = file_size;
|
2009-01-11 19:10:52 +01:00
|
|
|
if (SetFilePointerEx(f, offs, &offs, FILE_BEGIN) == FALSE)
|
2009-01-11 03:02:34 +01:00
|
|
|
{
|
|
|
|
CloseHandle(f);
|
|
|
|
ec = error_code(GetLastError(), get_system_category());
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
if (::SetEndOfFile(f) == FALSE)
|
|
|
|
{
|
|
|
|
ec = error_code(GetLastError(), get_system_category());
|
|
|
|
CloseHandle(f);
|
2009-01-11 11:21:18 +01:00
|
|
|
return -1;
|
2009-01-11 03:02:34 +01:00
|
|
|
}
|
|
|
|
CloseHandle(f);
|
2008-07-20 13:14:54 +02:00
|
|
|
}
|
2009-01-11 03:02:34 +01:00
|
|
|
|
|
|
|
return ret;
|
2004-02-19 17:47:12 +01:00
|
|
|
#else
|
2009-01-11 03:02:34 +01:00
|
|
|
size_type ret = lseek(m_fd, file_offset, SEEK_SET);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
ec = error_code(errno, get_posix_category());
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
#ifdef TORRENT_LINUX
|
|
|
|
bool aligned = false;
|
|
|
|
int size = 0;
|
|
|
|
// if we're not opened in no-buffer mode, we don't need alignment
|
|
|
|
if ((m_open_mode & no_buffer) == 0) aligned = true;
|
|
|
|
if (!aligned)
|
|
|
|
{
|
|
|
|
size = bufs_size(bufs, num_bufs);
|
|
|
|
if (size & (m_page_size-1) == 0) aligned = true;
|
|
|
|
}
|
|
|
|
if (aligned)
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
ret = ::writev(m_fd, bufs, num_bufs);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
ec = error_code(errno, get_posix_category());
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
#ifdef TORRENT_LINUX
|
|
|
|
file::iovec_t* temp_bufs = TORRENT_ALLOCA(file::iovec_t, num_bufs);
|
|
|
|
memcpy(temp_bufs, bufs, sizeof(file::iovec_t) * num_bufs);
|
|
|
|
iovec_t& last = temp_bufs[num_bufs-1];
|
|
|
|
last.iov_len = (last.iov_len & ~(m_page_size-1)) + m_page_size;
|
|
|
|
ret = ::writev(m_fd, temp_bufs, num_bufs);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
ec = error_code(errno, get_posix_category());
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
if (ftruncate(m_fd, file_offset + size) < 0)
|
|
|
|
{
|
|
|
|
ec = error_code(errno, get_posix_category());
|
2009-01-11 11:21:18 +01:00
|
|
|
return -1;
|
2009-01-11 03:02:34 +01:00
|
|
|
}
|
|
|
|
return (std::min)(ret, size_type(size));
|
|
|
|
#endif
|
2004-02-19 17:47:12 +01:00
|
|
|
#endif
|
2008-07-20 13:14:54 +02:00
|
|
|
}
|
2004-01-16 03:57:45 +01:00
|
|
|
|
2008-07-20 13:14:54 +02:00
|
|
|
bool file::set_size(size_type s, error_code& ec)
|
|
|
|
{
|
|
|
|
TORRENT_ASSERT(is_open());
|
|
|
|
TORRENT_ASSERT(s >= 0);
|
2004-01-16 03:57:45 +01:00
|
|
|
|
2008-07-20 13:14:54 +02:00
|
|
|
#ifdef TORRENT_WINDOWS
|
2009-01-11 03:02:34 +01:00
|
|
|
LARGE_INTEGER offs;
|
|
|
|
offs.QuadPart = s;
|
|
|
|
if (SetFilePointerEx(m_file_handle, offs, &offs, FILE_BEGIN) == FALSE)
|
|
|
|
{
|
|
|
|
ec = error_code(GetLastError(), get_system_category());
|
|
|
|
return false;
|
|
|
|
}
|
2008-07-20 13:14:54 +02:00
|
|
|
if (::SetEndOfFile(m_file_handle) == FALSE)
|
|
|
|
{
|
|
|
|
ec = error_code(GetLastError(), get_system_category());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
if (ftruncate(m_fd, s) < 0)
|
|
|
|
{
|
|
|
|
ec = error_code(errno, get_posix_category());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
return true;
|
2004-01-16 03:57:45 +01:00
|
|
|
}
|
|
|
|
|
2009-01-11 03:02:34 +01:00
|
|
|
size_type file::get_size(error_code& ec)
|
2004-01-16 03:57:45 +01:00
|
|
|
{
|
2008-07-20 13:14:54 +02:00
|
|
|
#ifdef TORRENT_WINDOWS
|
2009-01-11 03:02:34 +01:00
|
|
|
LARGE_INTEGER file_size;
|
|
|
|
if (!GetFileSizeEx(m_file_handle, &file_size))
|
2008-07-20 13:14:54 +02:00
|
|
|
{
|
|
|
|
ec = error_code(GetLastError(), get_system_category());
|
|
|
|
return -1;
|
|
|
|
}
|
2009-01-11 03:02:34 +01:00
|
|
|
return file_size.QuadPart;
|
2008-07-20 13:14:54 +02:00
|
|
|
#else
|
2009-01-11 03:02:34 +01:00
|
|
|
struct stat fs;
|
|
|
|
if (fstat(m_fd, &fs) != 0)
|
2008-07-20 13:14:54 +02:00
|
|
|
{
|
2009-01-11 03:02:34 +01:00
|
|
|
ec = error_code(errno, get_posix_category());
|
2008-07-20 13:14:54 +02:00
|
|
|
return -1;
|
|
|
|
}
|
2009-01-11 03:02:34 +01:00
|
|
|
return fs.st_size;
|
2008-07-20 13:14:54 +02:00
|
|
|
#endif
|
2004-01-16 03:57:45 +01:00
|
|
|
}
|
|
|
|
}
|
2008-07-20 13:14:54 +02:00
|
|
|
|