368 lines
7.8 KiB
C++
368 lines
7.8 KiB
C++
/*
|
|
|
|
Copyright (c) 2003, Magnus Jonsson & 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.
|
|
|
|
*/
|
|
|
|
#include "libtorrent/file.hpp"
|
|
#include "libtorrent/utf8.hpp"
|
|
#include "libtorrent/assert.hpp"
|
|
|
|
#ifdef UNICODE
|
|
#include "libtorrent/storage.hpp"
|
|
#endif
|
|
|
|
#include <sstream>
|
|
#include <windows.h>
|
|
#include <winioctl.h>
|
|
|
|
namespace
|
|
{
|
|
// must be used to not leak memory in case something would throw
|
|
class auto_localfree
|
|
{
|
|
public:
|
|
auto_localfree(HLOCAL memory)
|
|
: m_memory(memory)
|
|
{
|
|
}
|
|
~auto_localfree()
|
|
{
|
|
if (m_memory)
|
|
LocalFree(m_memory);
|
|
}
|
|
private:
|
|
HLOCAL m_memory;
|
|
};
|
|
|
|
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);
|
|
if (size == wchar_t(-1)) return s;
|
|
ret.resize(size);
|
|
return ret;
|
|
}
|
|
catch(std::exception)
|
|
{
|
|
return s;
|
|
}
|
|
}
|
|
|
|
void throw_exception(const char* thrower)
|
|
{
|
|
DWORD err = GetLastError();
|
|
|
|
#ifdef UNICODE
|
|
wchar_t *wbuffer = 0;
|
|
FormatMessage(
|
|
FORMAT_MESSAGE_FROM_SYSTEM
|
|
|FORMAT_MESSAGE_ALLOCATE_BUFFER
|
|
, 0, err, 0, (LPWSTR)&wbuffer, 0, 0);
|
|
auto_localfree auto_free(wbuffer);
|
|
std::string tmp_utf8;
|
|
libtorrent::wchar_utf8(wbuffer, tmp_utf8);
|
|
char const* buffer = tmp_utf8.c_str();
|
|
#else
|
|
char* buffer = 0;
|
|
FormatMessage(
|
|
FORMAT_MESSAGE_FROM_SYSTEM
|
|
|FORMAT_MESSAGE_ALLOCATE_BUFFER
|
|
, 0, err, 0, (LPSTR)&buffer, 0, 0);
|
|
auto_localfree auto_free(buffer);
|
|
#endif
|
|
|
|
std::stringstream s;
|
|
s << (thrower ? thrower : "NULL") << ": " << (buffer ? buffer : "NULL");
|
|
|
|
throw libtorrent::file_error(s.str());
|
|
}
|
|
}
|
|
|
|
namespace libtorrent
|
|
{
|
|
|
|
struct file::impl : boost::noncopyable
|
|
{
|
|
enum open_flags
|
|
{
|
|
read_flag = 1,
|
|
write_flag = 2
|
|
};
|
|
|
|
enum seek_mode
|
|
{
|
|
seek_begin = FILE_BEGIN,
|
|
seek_from_here = FILE_CURRENT,
|
|
seek_end = FILE_END
|
|
};
|
|
|
|
impl()
|
|
{
|
|
m_file_handle = INVALID_HANDLE_VALUE;
|
|
}
|
|
|
|
void open(const char *file_name, open_flags flags)
|
|
{
|
|
TORRENT_ASSERT(file_name);
|
|
TORRENT_ASSERT(flags & (read_flag | write_flag));
|
|
|
|
DWORD access_mask = 0;
|
|
if (flags & read_flag)
|
|
access_mask |= GENERIC_READ;
|
|
if (flags & write_flag)
|
|
access_mask |= GENERIC_WRITE;
|
|
|
|
TORRENT_ASSERT(access_mask & (GENERIC_READ | GENERIC_WRITE));
|
|
|
|
#ifdef UNICODE
|
|
std::wstring wfile_name(safe_convert(file_name));
|
|
HANDLE new_handle = CreateFile(
|
|
wfile_name.c_str()
|
|
, access_mask
|
|
, FILE_SHARE_READ
|
|
, 0
|
|
, (flags & write_flag)?OPEN_ALWAYS:OPEN_EXISTING
|
|
, FILE_ATTRIBUTE_NORMAL
|
|
, 0);
|
|
#else
|
|
HANDLE new_handle = CreateFile(
|
|
utf8_native(file_name).c_str()
|
|
, access_mask
|
|
, FILE_SHARE_READ
|
|
, 0
|
|
, (flags & write_flag)?OPEN_ALWAYS:OPEN_EXISTING
|
|
, FILE_ATTRIBUTE_NORMAL
|
|
, 0);
|
|
#endif
|
|
|
|
if (new_handle == INVALID_HANDLE_VALUE)
|
|
throw_exception(file_name);
|
|
// try to make the file sparse if supported
|
|
if (access_mask & GENERIC_WRITE)
|
|
{
|
|
DWORD temp;
|
|
::DeviceIoControl(new_handle, FSCTL_SET_SPARSE, 0, 0
|
|
, 0, 0, &temp, 0);
|
|
}
|
|
// will only close old file if the open succeeded
|
|
close();
|
|
m_file_handle = new_handle;
|
|
}
|
|
|
|
void close()
|
|
{
|
|
if (m_file_handle != INVALID_HANDLE_VALUE)
|
|
{
|
|
CloseHandle(m_file_handle);
|
|
m_file_handle = INVALID_HANDLE_VALUE;
|
|
}
|
|
}
|
|
|
|
~impl()
|
|
{
|
|
close();
|
|
}
|
|
|
|
size_type write(const char* buffer, size_type num_bytes)
|
|
{
|
|
TORRENT_ASSERT(buffer);
|
|
TORRENT_ASSERT((DWORD)num_bytes == num_bytes);
|
|
DWORD bytes_written = 0;
|
|
if (num_bytes != 0)
|
|
{
|
|
if (FALSE == WriteFile(
|
|
m_file_handle
|
|
, buffer
|
|
, (DWORD)num_bytes
|
|
, &bytes_written
|
|
, 0))
|
|
{
|
|
throw_exception("file::write");
|
|
}
|
|
}
|
|
return bytes_written;
|
|
}
|
|
|
|
size_type read(char* buffer, size_type num_bytes)
|
|
{
|
|
TORRENT_ASSERT(buffer);
|
|
TORRENT_ASSERT(num_bytes >= 0);
|
|
TORRENT_ASSERT((DWORD)num_bytes == num_bytes);
|
|
|
|
DWORD bytes_read = 0;
|
|
if (num_bytes != 0)
|
|
{
|
|
if (FALSE == ReadFile(
|
|
m_file_handle
|
|
, buffer
|
|
, (DWORD)num_bytes
|
|
, &bytes_read
|
|
, 0))
|
|
{
|
|
throw_exception("file::read");
|
|
}
|
|
}
|
|
return bytes_read;
|
|
}
|
|
|
|
void set_size(size_type s)
|
|
{
|
|
size_type pos = tell();
|
|
seek(s, seek_begin);
|
|
if (FALSE == ::SetEndOfFile(m_file_handle))
|
|
throw_exception("file::set_size");
|
|
|
|
seek(pos, seek_begin);
|
|
}
|
|
|
|
size_type seek(size_type pos, seek_mode from_where)
|
|
{
|
|
TORRENT_ASSERT(pos >= 0 || from_where != seek_begin);
|
|
TORRENT_ASSERT(pos <= 0 || from_where != seek_end);
|
|
LARGE_INTEGER offs;
|
|
offs.QuadPart = pos;
|
|
if (FALSE == SetFilePointerEx(
|
|
m_file_handle
|
|
, offs
|
|
, &offs
|
|
, from_where))
|
|
{
|
|
throw_exception("file::seek");
|
|
}
|
|
return offs.QuadPart;
|
|
}
|
|
|
|
size_type tell()
|
|
{
|
|
LARGE_INTEGER offs;
|
|
offs.QuadPart = 0;
|
|
|
|
// is there any other way to get offset?
|
|
if (FALSE == SetFilePointerEx(
|
|
m_file_handle
|
|
, offs
|
|
, &offs
|
|
, FILE_CURRENT))
|
|
{
|
|
throw_exception("file::tell");
|
|
}
|
|
|
|
size_type pos = offs.QuadPart;
|
|
TORRENT_ASSERT(pos >= 0);
|
|
return pos;
|
|
}
|
|
/*
|
|
size_type size()
|
|
{
|
|
LARGE_INTEGER s;
|
|
if (FALSE == GetFileSizeEx(m_file_handle, &s))
|
|
{
|
|
throw_exception("file::size");
|
|
}
|
|
|
|
size_type size = s.QuadPart;
|
|
TORRENT_ASSERT(size >= 0);
|
|
return size;
|
|
}
|
|
*/
|
|
private:
|
|
|
|
HANDLE m_file_handle;
|
|
|
|
};
|
|
}
|
|
|
|
namespace libtorrent
|
|
{
|
|
|
|
const file::seek_mode file::begin(file::impl::seek_begin);
|
|
const file::seek_mode file::end(file::impl::seek_end);
|
|
|
|
const file::open_mode file::in(file::impl::read_flag);
|
|
const file::open_mode file::out(file::impl::write_flag);
|
|
|
|
file::file()
|
|
: m_impl(new libtorrent::file::impl())
|
|
{
|
|
}
|
|
file::file(boost::filesystem::path const& p, open_mode m)
|
|
: m_impl(new libtorrent::file::impl())
|
|
{
|
|
open(p,m);
|
|
}
|
|
|
|
file::~file()
|
|
{
|
|
}
|
|
|
|
void file::open(boost::filesystem::path const& p, open_mode m)
|
|
{
|
|
TORRENT_ASSERT(p.is_complete());
|
|
m_impl->open(p.native_file_string().c_str(), impl::open_flags(m.m_mask));
|
|
}
|
|
|
|
void file::close()
|
|
{
|
|
m_impl->close();
|
|
}
|
|
|
|
size_type file::write(const char* buffer, size_type num_bytes)
|
|
{
|
|
return m_impl->write(buffer, num_bytes);
|
|
}
|
|
|
|
size_type file::read(char* buffer, size_type num_bytes)
|
|
{
|
|
return m_impl->read(buffer, num_bytes);
|
|
}
|
|
|
|
void file::set_size(size_type s)
|
|
{
|
|
m_impl->set_size(s);
|
|
}
|
|
|
|
size_type file::seek(size_type pos, seek_mode m)
|
|
{
|
|
return m_impl->seek(pos,impl::seek_mode(m.m_val));
|
|
}
|
|
|
|
size_type file::tell()
|
|
{
|
|
return m_impl->tell();
|
|
}
|
|
}
|