1
0
Fork 1
mirror of https://github.com/TES3MP/openmw-tes3mp.git synced 2025-01-19 22:23:51 +00:00
openmw-tes3mp/components/files/lowlevelfile.cpp

359 lines
7.5 KiB
C++
Raw Normal View History

#include "lowlevelfile.hpp"
#include <stdexcept>
#include <sstream>
#include <cassert>
2020-11-13 06:36:26 +00:00
#if FILE_API == FILE_API_STDIO
#include <errno.h>
#include <string.h>
#endif
#if FILE_API == FILE_API_POSIX
#include <sys/types.h>
#include <unistd.h>
#include <fcntl.h>
#include <errno.h>
#include <string.h>
#endif
#if FILE_API == FILE_API_STDIO
/*
*
2015-01-31 18:35:09 +00:00
* Implementation of LowLevelFile methods using c stdio
*
*/
LowLevelFile::LowLevelFile ()
{
2018-10-09 06:21:12 +00:00
mHandle = nullptr;
}
LowLevelFile::~LowLevelFile ()
{
2018-10-09 06:21:12 +00:00
if (mHandle != nullptr)
2015-01-31 18:35:09 +00:00
fclose (mHandle);
}
void LowLevelFile::open (char const * filename)
{
2018-10-09 06:21:12 +00:00
assert (mHandle == nullptr);
2015-01-31 18:35:09 +00:00
mHandle = fopen (filename, "rb");
2018-10-09 06:21:12 +00:00
if (mHandle == nullptr)
2015-01-31 18:35:09 +00:00
{
std::ostringstream os;
2020-11-13 06:36:26 +00:00
os << "Failed to open '" << filename << "' for reading: " << strerror(errno);
2015-01-31 18:35:09 +00:00
throw std::runtime_error (os.str ());
}
}
void LowLevelFile::close ()
{
2018-10-09 06:21:12 +00:00
assert (mHandle != nullptr);
2015-01-31 18:35:09 +00:00
fclose (mHandle);
2018-10-09 06:21:12 +00:00
mHandle = nullptr;
}
size_t LowLevelFile::size ()
{
2018-10-09 06:21:12 +00:00
assert (mHandle != nullptr);
2015-01-31 18:35:09 +00:00
long oldPosition = ftell (mHandle);
if (oldPosition == -1)
2020-11-13 06:36:26 +00:00
{
std::ostringstream os;
os << "An ftell() call failed: " << strerror(errno);
throw std::runtime_error (os.str ());
}
2015-01-31 18:35:09 +00:00
if (fseek (mHandle, 0, SEEK_END) != 0)
2020-11-13 06:36:26 +00:00
{
std::ostringstream os;
os << "An fseek() call failed: " << strerror(errno);
throw std::runtime_error (os.str ());
}
2020-11-13 06:36:26 +00:00
long size = ftell (mHandle);
if (size == -1)
{
std::ostringstream os;
os << "An ftell() call failed: " << strerror(errno);
throw std::runtime_error (os.str ());
}
2015-01-31 18:35:09 +00:00
if (fseek (mHandle, oldPosition, SEEK_SET) != 0)
2020-11-13 06:36:26 +00:00
{
std::ostringstream os;
os << "An fseek() call failed: " << strerror(errno);
throw std::runtime_error (os.str ());
}
2020-11-13 06:36:26 +00:00
return size_t (size);
}
2020-11-13 06:36:26 +00:00
void LowLevelFile::seek (size_t position)
{
2018-10-09 06:21:12 +00:00
assert (mHandle != nullptr);
2020-11-13 06:36:26 +00:00
if (fseek (mHandle, position, SEEK_SET) != 0)
{
std::ostringstream os;
os << "An fseek() call failed: " << strerror(errno);
throw std::runtime_error (os.str ());
}
}
size_t LowLevelFile::tell ()
{
2018-10-09 06:21:12 +00:00
assert (mHandle != nullptr);
2020-11-13 06:36:26 +00:00
long position = ftell (mHandle);
if (position == -1)
{
std::ostringstream os;
os << "An ftell() call failed: " << strerror(errno);
throw std::runtime_error (os.str ());
}
2020-11-13 06:36:26 +00:00
return size_t (position);
}
size_t LowLevelFile::read (void * data, size_t size)
{
2018-10-09 06:21:12 +00:00
assert (mHandle != nullptr);
2015-01-31 18:35:09 +00:00
int amount = fread (data, 1, size, mHandle);
2015-01-31 18:35:09 +00:00
if (amount == 0 && ferror (mHandle))
2020-11-13 06:36:26 +00:00
{
std::ostringstream os;
os << "An attempt to read " << size << " bytes failed: " << strerror(errno);
throw std::runtime_error (os.str ());
}
2015-01-31 18:35:09 +00:00
return amount;
}
#elif FILE_API == FILE_API_POSIX
/*
*
2015-01-31 18:35:09 +00:00
* Implementation of LowLevelFile methods using posix IO calls
*
*/
LowLevelFile::LowLevelFile ()
{
2015-01-31 18:35:09 +00:00
mHandle = -1;
}
LowLevelFile::~LowLevelFile ()
{
2015-01-31 18:35:09 +00:00
if (mHandle != -1)
::close (mHandle);
}
void LowLevelFile::open (char const * filename)
{
2015-01-31 18:35:09 +00:00
assert (mHandle == -1);
#ifdef O_BINARY
2015-01-31 18:35:09 +00:00
static const int openFlags = O_RDONLY | O_BINARY;
#else
2015-01-31 18:35:09 +00:00
static const int openFlags = O_RDONLY;
#endif
2015-01-31 18:35:09 +00:00
mHandle = ::open (filename, openFlags, 0);
2015-01-31 18:35:09 +00:00
if (mHandle == -1)
{
std::ostringstream os;
os << "Failed to open '" << filename << "' for reading: " << strerror(errno);
2015-01-31 18:35:09 +00:00
throw std::runtime_error (os.str ());
}
}
void LowLevelFile::close ()
{
2015-01-31 18:35:09 +00:00
assert (mHandle != -1);
2015-01-31 18:35:09 +00:00
::close (mHandle);
2015-01-31 18:35:09 +00:00
mHandle = -1;
}
size_t LowLevelFile::size ()
{
2015-01-31 18:35:09 +00:00
assert (mHandle != -1);
2015-01-31 18:35:09 +00:00
size_t oldPosition = ::lseek (mHandle, 0, SEEK_CUR);
2015-01-31 18:35:09 +00:00
if (oldPosition == size_t (-1))
{
std::ostringstream os;
2020-11-13 06:36:26 +00:00
os << "An lseek() call failed: " << strerror(errno);
throw std::runtime_error (os.str ());
}
2020-11-13 06:36:26 +00:00
size_t size = ::lseek (mHandle, 0, SEEK_END);
2020-11-13 06:36:26 +00:00
if (size == size_t (-1))
{
std::ostringstream os;
2020-11-13 06:36:26 +00:00
os << "An lseek() call failed: " << strerror(errno);
throw std::runtime_error (os.str ());
}
2015-01-31 18:35:09 +00:00
if (lseek (mHandle, oldPosition, SEEK_SET) == -1)
{
std::ostringstream os;
2020-11-13 06:36:26 +00:00
os << "An lseek() call failed: " << strerror(errno);
throw std::runtime_error (os.str ());
}
2020-11-13 06:36:26 +00:00
return size;
}
2020-11-13 06:36:26 +00:00
void LowLevelFile::seek (size_t position)
{
2015-01-31 18:35:09 +00:00
assert (mHandle != -1);
2020-11-13 06:36:26 +00:00
if (::lseek (mHandle, position, SEEK_SET) == -1)
{
std::ostringstream os;
2020-11-13 06:36:26 +00:00
os << "An lseek() call failed: " << strerror(errno);
throw std::runtime_error (os.str ());
}
}
size_t LowLevelFile::tell ()
{
2015-01-31 18:35:09 +00:00
assert (mHandle != -1);
2020-11-13 06:36:26 +00:00
size_t position = ::lseek (mHandle, 0, SEEK_CUR);
2020-11-13 06:36:26 +00:00
if (position == size_t (-1))
{
std::ostringstream os;
2020-11-13 06:36:26 +00:00
os << "An lseek() call failed: " << strerror(errno);
throw std::runtime_error (os.str ());
}
2020-11-13 06:36:26 +00:00
return position;
}
size_t LowLevelFile::read (void * data, size_t size)
{
2015-01-31 18:35:09 +00:00
assert (mHandle != -1);
2015-01-31 18:35:09 +00:00
int amount = ::read (mHandle, data, size);
2015-01-31 18:35:09 +00:00
if (amount == -1)
{
std::ostringstream os;
2020-11-13 06:36:26 +00:00
os << "An attempt to read " << size << " bytes failed: " << strerror(errno);
throw std::runtime_error (os.str ());
}
2015-01-31 18:35:09 +00:00
return amount;
}
#elif FILE_API == FILE_API_WIN32
#include <boost/locale.hpp>
/*
*
2015-01-31 18:35:09 +00:00
* Implementation of LowLevelFile methods using Win32 API calls
*
*/
LowLevelFile::LowLevelFile ()
{
2015-01-31 18:35:09 +00:00
mHandle = INVALID_HANDLE_VALUE;
}
LowLevelFile::~LowLevelFile ()
{
2015-01-31 18:35:09 +00:00
if (mHandle != INVALID_HANDLE_VALUE)
CloseHandle (mHandle);
}
void LowLevelFile::open (char const * filename)
{
2015-01-31 18:35:09 +00:00
assert (mHandle == INVALID_HANDLE_VALUE);
2015-01-31 18:35:09 +00:00
std::wstring wname = boost::locale::conv::utf_to_utf<wchar_t>(filename);
HANDLE handle = CreateFileW (wname.c_str(), GENERIC_READ, FILE_SHARE_READ, 0, OPEN_EXISTING, 0, 0);
2015-01-31 18:35:09 +00:00
if (handle == INVALID_HANDLE_VALUE)
{
std::ostringstream os;
os << "Failed to open '" << filename << "' for reading: " << GetLastError();
2015-01-31 18:35:09 +00:00
throw std::runtime_error (os.str ());
}
2015-01-31 18:35:09 +00:00
mHandle = handle;
}
void LowLevelFile::close ()
{
2015-01-31 18:35:09 +00:00
assert (mHandle != INVALID_HANDLE_VALUE);
2015-01-31 18:35:09 +00:00
CloseHandle (mHandle);
2015-01-31 18:35:09 +00:00
mHandle = INVALID_HANDLE_VALUE;
}
size_t LowLevelFile::size ()
{
2015-01-31 18:35:09 +00:00
assert (mHandle != INVALID_HANDLE_VALUE);
2015-01-31 18:35:09 +00:00
BY_HANDLE_FILE_INFORMATION info;
2015-01-31 18:35:09 +00:00
if (!GetFileInformationByHandle (mHandle, &info))
throw std::runtime_error ("A query operation on a file failed.");
2015-01-31 18:35:09 +00:00
if (info.nFileSizeHigh != 0)
throw std::runtime_error ("Files greater that 4GB are not supported.");
2015-01-31 18:35:09 +00:00
return info.nFileSizeLow;
}
2020-11-13 06:36:26 +00:00
void LowLevelFile::seek (size_t position)
{
2015-01-31 18:35:09 +00:00
assert (mHandle != INVALID_HANDLE_VALUE);
2021-05-02 06:43:44 +00:00
if (SetFilePointer (mHandle, static_cast<LONG>(position), nullptr, SEEK_SET) == INVALID_SET_FILE_POINTER)
2015-01-31 18:35:09 +00:00
if (GetLastError () != NO_ERROR)
throw std::runtime_error ("A seek operation on a file failed.");
}
size_t LowLevelFile::tell ()
{
2015-01-31 18:35:09 +00:00
assert (mHandle != INVALID_HANDLE_VALUE);
2018-10-09 06:21:12 +00:00
DWORD value = SetFilePointer (mHandle, 0, nullptr, SEEK_CUR);
2015-01-31 18:35:09 +00:00
if (value == INVALID_SET_FILE_POINTER && GetLastError () != NO_ERROR)
throw std::runtime_error ("A query operation on a file failed.");
2015-01-31 18:35:09 +00:00
return value;
}
size_t LowLevelFile::read (void * data, size_t size)
{
2015-01-31 18:35:09 +00:00
assert (mHandle != INVALID_HANDLE_VALUE);
2015-01-31 18:35:09 +00:00
DWORD read;
2021-05-02 06:43:44 +00:00
if (!ReadFile (mHandle, data, static_cast<DWORD>(size), &read, nullptr))
2015-01-31 18:35:09 +00:00
throw std::runtime_error ("A read operation on a file failed.");
2015-01-31 18:35:09 +00:00
return read;
}
#endif