mirror of https://github.com/OpenMW/openmw.git
Merge branch 'platform-file' into 'master'
Move platform specific file code into platform and cleanup LowLevelFile See merge request OpenMW/openmw!2143LTO-timing^2
commit
7c442926f8
@ -0,0 +1,35 @@
|
|||||||
|
#ifndef OPENMW_COMPONENTS_PLATFORM_FILE_HPP
|
||||||
|
#define OPENMW_COMPONENTS_PLATFORM_FILE_HPP
|
||||||
|
|
||||||
|
#include <cstdlib>
|
||||||
|
#include <string_view>
|
||||||
|
|
||||||
|
namespace Platform::File {
|
||||||
|
|
||||||
|
enum class Handle : intptr_t
|
||||||
|
{
|
||||||
|
Invalid = -1
|
||||||
|
};
|
||||||
|
|
||||||
|
enum class SeekType
|
||||||
|
{
|
||||||
|
Begin,
|
||||||
|
Current,
|
||||||
|
End
|
||||||
|
};
|
||||||
|
|
||||||
|
Handle open(const char* filename);
|
||||||
|
|
||||||
|
void close(Handle handle);
|
||||||
|
|
||||||
|
size_t size(Handle handle);
|
||||||
|
|
||||||
|
void seek(Handle handle, size_t Position, SeekType type = SeekType::Begin);
|
||||||
|
size_t tell(Handle handle);
|
||||||
|
|
||||||
|
size_t read(Handle handle, void* data, size_t size);
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif // OPENMW_COMPONENTS_PLATFORM_FILE_HPP
|
@ -0,0 +1,99 @@
|
|||||||
|
#include "file.hpp"
|
||||||
|
|
||||||
|
#include <sys/types.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <fcntl.h>
|
||||||
|
#include <errno.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <stdexcept>
|
||||||
|
#include <string>
|
||||||
|
|
||||||
|
namespace Platform::File {
|
||||||
|
|
||||||
|
static auto getNativeHandle(Handle handle)
|
||||||
|
{
|
||||||
|
return static_cast<int>(handle);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int getNativeSeekType(SeekType seek)
|
||||||
|
{
|
||||||
|
if (seek == SeekType::Begin)
|
||||||
|
return SEEK_SET;
|
||||||
|
if (seek == SeekType::Current)
|
||||||
|
return SEEK_CUR;
|
||||||
|
if (seek == SeekType::End)
|
||||||
|
return SEEK_END;
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
Handle open(const char* filename)
|
||||||
|
{
|
||||||
|
#ifdef O_BINARY
|
||||||
|
static const int openFlags = O_RDONLY | O_BINARY;
|
||||||
|
#else
|
||||||
|
static const int openFlags = O_RDONLY;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
auto handle = ::open(filename, openFlags, 0);
|
||||||
|
if (handle == -1)
|
||||||
|
{
|
||||||
|
throw std::runtime_error(std::string("Failed to open '") + filename + "' for reading: " + strerror(errno));
|
||||||
|
}
|
||||||
|
return static_cast<Handle>(handle);
|
||||||
|
}
|
||||||
|
|
||||||
|
void close(Handle handle)
|
||||||
|
{
|
||||||
|
auto nativeHandle = getNativeHandle(handle);
|
||||||
|
|
||||||
|
::close(nativeHandle);
|
||||||
|
}
|
||||||
|
|
||||||
|
void seek(Handle handle, size_t position, SeekType type /*= SeekType::Begin*/)
|
||||||
|
{
|
||||||
|
const auto nativeHandle = getNativeHandle(handle);
|
||||||
|
const auto nativeSeekType = getNativeSeekType(type);
|
||||||
|
|
||||||
|
if (::lseek(nativeHandle, position, nativeSeekType) == -1)
|
||||||
|
{
|
||||||
|
throw std::runtime_error("An lseek() call failed: " + std::string(strerror(errno)));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t size(Handle handle)
|
||||||
|
{
|
||||||
|
const auto oldPos = tell(handle);
|
||||||
|
|
||||||
|
seek(handle, 0, SeekType::End);
|
||||||
|
const auto fileSize = tell(handle);
|
||||||
|
seek(handle, oldPos, SeekType::Begin);
|
||||||
|
|
||||||
|
|
||||||
|
return static_cast<size_t>(fileSize);
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t tell(Handle handle)
|
||||||
|
{
|
||||||
|
auto nativeHandle = getNativeHandle(handle);
|
||||||
|
|
||||||
|
size_t position = ::lseek(nativeHandle, 0, SEEK_CUR);
|
||||||
|
if (position == size_t(-1))
|
||||||
|
{
|
||||||
|
throw std::runtime_error("An lseek() call failed: " + std::string(strerror(errno)));
|
||||||
|
}
|
||||||
|
return position;
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t read(Handle handle, void* data, size_t size)
|
||||||
|
{
|
||||||
|
auto nativeHandle = getNativeHandle(handle);
|
||||||
|
|
||||||
|
int amount = ::read(nativeHandle, data, size);
|
||||||
|
if (amount == -1)
|
||||||
|
{
|
||||||
|
throw std::runtime_error("An attempt to read " + std::to_string(size) + " bytes failed: " + strerror(errno));
|
||||||
|
}
|
||||||
|
return amount;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -0,0 +1,88 @@
|
|||||||
|
#include "file.hpp"
|
||||||
|
|
||||||
|
#include <errno.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <string>
|
||||||
|
#include <stdexcept>
|
||||||
|
|
||||||
|
namespace Platform::File {
|
||||||
|
|
||||||
|
static auto getNativeHandle(Handle handle)
|
||||||
|
{
|
||||||
|
return reinterpret_cast<FILE*>(static_cast<intptr_t>(handle));
|
||||||
|
}
|
||||||
|
|
||||||
|
static int getNativeSeekType(SeekType seek)
|
||||||
|
{
|
||||||
|
if (seek == SeekType::Begin)
|
||||||
|
return SEEK_SET;
|
||||||
|
if (seek == SeekType::Current)
|
||||||
|
return SEEK_CUR;
|
||||||
|
if (seek == SeekType::End)
|
||||||
|
return SEEK_END;
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
Handle open(const char* filename)
|
||||||
|
{
|
||||||
|
FILE* handle = fopen(filename, "rb");
|
||||||
|
if (handle == nullptr)
|
||||||
|
{
|
||||||
|
throw std::runtime_error(std::string("Failed to open '") + filename + "' for reading: " + strerror(errno));
|
||||||
|
}
|
||||||
|
return static_cast<Handle>(reinterpret_cast<intptr_t>(handle));
|
||||||
|
}
|
||||||
|
|
||||||
|
void close(Handle handle)
|
||||||
|
{
|
||||||
|
auto nativeHandle = getNativeHandle(handle);
|
||||||
|
fclose(nativeHandle);
|
||||||
|
}
|
||||||
|
|
||||||
|
void seek(Handle handle, size_t position, SeekType type /*= SeekType::Begin*/)
|
||||||
|
{
|
||||||
|
const auto nativeHandle = getNativeHandle(handle);
|
||||||
|
const auto nativeSeekType = getNativeSeekType(type);
|
||||||
|
if (fseek(nativeHandle, position, nativeSeekType) != 0)
|
||||||
|
{
|
||||||
|
throw std::runtime_error(std::string("An fseek() call failed: ") + strerror(errno));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t size(Handle handle)
|
||||||
|
{
|
||||||
|
auto nativeHandle = getNativeHandle(handle);
|
||||||
|
|
||||||
|
const auto oldPos = tell(handle);
|
||||||
|
seek(handle, 0, SeekType::End);
|
||||||
|
const auto fileSize = tell(handle);
|
||||||
|
seek(handle, oldPos, SeekType::Begin);
|
||||||
|
|
||||||
|
return static_cast<size_t>(fileSize);
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t tell(Handle handle)
|
||||||
|
{
|
||||||
|
auto nativeHandle = getNativeHandle(handle);
|
||||||
|
|
||||||
|
long position = ftell(nativeHandle);
|
||||||
|
if (position == -1)
|
||||||
|
{
|
||||||
|
throw std::runtime_error(std::string("An ftell() call failed: ") + strerror(errno));
|
||||||
|
}
|
||||||
|
return static_cast<size_t>(position);
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t read(Handle handle, void* data, size_t size)
|
||||||
|
{
|
||||||
|
auto nativeHandle = getNativeHandle(handle);
|
||||||
|
|
||||||
|
int amount = fread(data, 1, size, nativeHandle);
|
||||||
|
if (amount == 0 && ferror(nativeHandle))
|
||||||
|
{
|
||||||
|
throw std::runtime_error(std::string("An attempt to read ") + std::to_string(size) + " bytes failed: " + strerror(errno));
|
||||||
|
}
|
||||||
|
return static_cast<size_t>(amount);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -0,0 +1,96 @@
|
|||||||
|
#include "file.hpp"
|
||||||
|
|
||||||
|
#include <components/windows.hpp>
|
||||||
|
#include <string>
|
||||||
|
#include <stdexcept>
|
||||||
|
#include <boost/locale.hpp>
|
||||||
|
|
||||||
|
namespace Platform::File {
|
||||||
|
|
||||||
|
static auto getNativeHandle(Handle handle)
|
||||||
|
{
|
||||||
|
return reinterpret_cast<HANDLE>(static_cast<intptr_t>(handle));
|
||||||
|
}
|
||||||
|
|
||||||
|
static int getNativeSeekType(SeekType seek)
|
||||||
|
{
|
||||||
|
if (seek == SeekType::Begin)
|
||||||
|
return FILE_BEGIN;
|
||||||
|
if (seek == SeekType::Current)
|
||||||
|
return FILE_CURRENT;
|
||||||
|
if (seek == SeekType::End)
|
||||||
|
return FILE_END;
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
Handle open(const char* filename)
|
||||||
|
{
|
||||||
|
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);
|
||||||
|
if (handle == INVALID_HANDLE_VALUE)
|
||||||
|
{
|
||||||
|
throw std::runtime_error(std::string("Failed to open '") + filename + "' for reading: " + std::to_string(GetLastError()));
|
||||||
|
}
|
||||||
|
return static_cast<Handle>(reinterpret_cast<intptr_t>(handle));
|
||||||
|
}
|
||||||
|
|
||||||
|
void close(Handle handle)
|
||||||
|
{
|
||||||
|
auto nativeHandle = getNativeHandle(handle);
|
||||||
|
CloseHandle(nativeHandle);
|
||||||
|
}
|
||||||
|
|
||||||
|
void seek(Handle handle, size_t position, SeekType type /*= SeekType::Begin*/)
|
||||||
|
{
|
||||||
|
const auto nativeHandle = getNativeHandle(handle);
|
||||||
|
const auto nativeSeekType = getNativeSeekType(type);
|
||||||
|
|
||||||
|
if (SetFilePointer(nativeHandle, static_cast<LONG>(position), nullptr, nativeSeekType) == INVALID_SET_FILE_POINTER)
|
||||||
|
{
|
||||||
|
if (auto errCode = GetLastError(); errCode != ERROR_SUCCESS)
|
||||||
|
{
|
||||||
|
throw std::runtime_error(std::string("An fseek() call failed: ") + std::to_string(errCode));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t size(Handle handle)
|
||||||
|
{
|
||||||
|
auto nativeHandle = getNativeHandle(handle);
|
||||||
|
|
||||||
|
assert(isValidHandle(mHandle));
|
||||||
|
|
||||||
|
BY_HANDLE_FILE_INFORMATION info;
|
||||||
|
|
||||||
|
if (!GetFileInformationByHandle(nativeHandle, &info))
|
||||||
|
throw std::runtime_error("A query operation on a file failed.");
|
||||||
|
|
||||||
|
if (info.nFileSizeHigh != 0)
|
||||||
|
throw std::runtime_error("Files greater that 4GB are not supported.");
|
||||||
|
|
||||||
|
return info.nFileSizeLow;
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t tell(Handle handle)
|
||||||
|
{
|
||||||
|
auto nativeHandle = getNativeHandle(handle);
|
||||||
|
|
||||||
|
DWORD value = SetFilePointer(nativeHandle, 0, nullptr, SEEK_CUR);
|
||||||
|
if (value == INVALID_SET_FILE_POINTER && GetLastError() != NO_ERROR)
|
||||||
|
throw std::runtime_error("A query operation on a file failed.");
|
||||||
|
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t read(Handle handle, void* data, size_t size)
|
||||||
|
{
|
||||||
|
auto nativeHandle = getNativeHandle(handle);
|
||||||
|
|
||||||
|
DWORD bytesRead{};
|
||||||
|
|
||||||
|
if (!ReadFile(nativeHandle, data, static_cast<DWORD>(size), &bytesRead, nullptr))
|
||||||
|
throw std::runtime_error(std::string("A read operation on a file failed: ") + std::to_string(GetLastError()));
|
||||||
|
|
||||||
|
return bytesRead;
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue