Making stateless file IO platform-agnostic.
This commit is contained in:
parent
376307cfa2
commit
42400d06a9
|
@ -11,14 +11,12 @@
|
||||||
#define XENIA_BASE_FILESYSTEM_H_
|
#define XENIA_BASE_FILESYSTEM_H_
|
||||||
|
|
||||||
#include <iterator>
|
#include <iterator>
|
||||||
|
#include <memory>
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
#include "xenia/base/string.h"
|
#include "xenia/base/string.h"
|
||||||
|
|
||||||
// TOODO(benvanik): remove windows headers.
|
|
||||||
#undef DeleteFile
|
|
||||||
|
|
||||||
namespace xe {
|
namespace xe {
|
||||||
namespace filesystem {
|
namespace filesystem {
|
||||||
|
|
||||||
|
@ -33,6 +31,50 @@ bool IsFolder(const std::wstring& path);
|
||||||
FILE* OpenFile(const std::wstring& path, const char* mode);
|
FILE* OpenFile(const std::wstring& path, const char* mode);
|
||||||
bool DeleteFile(const std::wstring& path);
|
bool DeleteFile(const std::wstring& path);
|
||||||
|
|
||||||
|
struct FileAccess {
|
||||||
|
// Implies kFileReadData.
|
||||||
|
static const uint32_t kGenericRead = 0x80000000;
|
||||||
|
// Implies kFileWriteData.
|
||||||
|
static const uint32_t kGenericWrite = 0x40000000;
|
||||||
|
static const uint32_t kGenericExecute = 0x20000000;
|
||||||
|
static const uint32_t kGenericAll = 0x10000000;
|
||||||
|
static const uint32_t kFileReadData = 0x00000001;
|
||||||
|
static const uint32_t kFileWriteData = 0x00000002;
|
||||||
|
static const uint32_t kFileAppendData = 0x00000004;
|
||||||
|
};
|
||||||
|
|
||||||
|
class FileHandle {
|
||||||
|
public:
|
||||||
|
// Opens the file, failing if it doesn't exist.
|
||||||
|
// The desired_access bitmask denotes the permissions on the file.
|
||||||
|
static std::unique_ptr<FileHandle> OpenExisting(std::wstring path,
|
||||||
|
uint32_t desired_access);
|
||||||
|
|
||||||
|
virtual ~FileHandle() = default;
|
||||||
|
|
||||||
|
std::wstring path() const { return path_; }
|
||||||
|
|
||||||
|
// Reads the requested number of bytes from the file starting at the given
|
||||||
|
// offset. The total number of bytes read is returned only if the complete
|
||||||
|
// read succeeds.
|
||||||
|
virtual bool Read(size_t file_offset, void* buffer, size_t buffer_length,
|
||||||
|
size_t* out_bytes_read) = 0;
|
||||||
|
|
||||||
|
// Writes the given buffer to the file starting at the given offset.
|
||||||
|
// The total number of bytes written is returned only if the complete
|
||||||
|
// write succeeds.
|
||||||
|
virtual bool Write(size_t file_offset, const void* buffer,
|
||||||
|
size_t buffer_length, size_t* out_bytes_written) = 0;
|
||||||
|
|
||||||
|
// Flushes any pending write buffers to the underlying filesystem.
|
||||||
|
virtual void Flush() = 0;
|
||||||
|
|
||||||
|
protected:
|
||||||
|
FileHandle(std::wstring path) : path_(std::move(path)) {}
|
||||||
|
|
||||||
|
std::wstring path_;
|
||||||
|
};
|
||||||
|
|
||||||
struct FileInfo {
|
struct FileInfo {
|
||||||
enum class Type {
|
enum class Type {
|
||||||
kFile,
|
kFile,
|
||||||
|
|
|
@ -56,6 +56,89 @@ bool DeleteFile(const std::wstring& path) {
|
||||||
return DeleteFileW(path.c_str()) ? true : false;
|
return DeleteFileW(path.c_str()) ? true : false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class Win32FileHandle : public FileHandle {
|
||||||
|
public:
|
||||||
|
Win32FileHandle(std::wstring path, HANDLE handle)
|
||||||
|
: FileHandle(std::move(path)), handle_(handle) {}
|
||||||
|
~Win32FileHandle() override {
|
||||||
|
CloseHandle(handle_);
|
||||||
|
handle_ = nullptr;
|
||||||
|
}
|
||||||
|
bool Read(size_t file_offset, void* buffer, size_t buffer_length,
|
||||||
|
size_t* out_bytes_read) override {
|
||||||
|
*out_bytes_read = 0;
|
||||||
|
OVERLAPPED overlapped;
|
||||||
|
overlapped.Pointer = (PVOID)file_offset;
|
||||||
|
overlapped.hEvent = NULL;
|
||||||
|
DWORD bytes_read = 0;
|
||||||
|
BOOL read = ReadFile(handle_, buffer, (DWORD)buffer_length, &bytes_read,
|
||||||
|
&overlapped);
|
||||||
|
if (read) {
|
||||||
|
*out_bytes_read = bytes_read;
|
||||||
|
return true;
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
bool Write(size_t file_offset, const void* buffer, size_t buffer_length,
|
||||||
|
size_t* out_bytes_written) override {
|
||||||
|
*out_bytes_written = 0;
|
||||||
|
OVERLAPPED overlapped;
|
||||||
|
overlapped.Pointer = (PVOID)file_offset;
|
||||||
|
overlapped.hEvent = NULL;
|
||||||
|
DWORD bytes_written = 0;
|
||||||
|
BOOL wrote = WriteFile(handle_, buffer, (DWORD)buffer_length,
|
||||||
|
&bytes_written, &overlapped);
|
||||||
|
if (wrote) {
|
||||||
|
*out_bytes_written = bytes_written;
|
||||||
|
return true;
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
void Flush() override { FlushFileBuffers(handle_); }
|
||||||
|
|
||||||
|
private:
|
||||||
|
HANDLE handle_ = nullptr;
|
||||||
|
};
|
||||||
|
|
||||||
|
std::unique_ptr<FileHandle> FileHandle::OpenExisting(std::wstring path,
|
||||||
|
uint32_t desired_access) {
|
||||||
|
DWORD open_access = 0;
|
||||||
|
if (desired_access & FileAccess::kGenericRead) {
|
||||||
|
open_access |= GENERIC_READ;
|
||||||
|
}
|
||||||
|
if (desired_access & FileAccess::kGenericWrite) {
|
||||||
|
open_access |= GENERIC_WRITE;
|
||||||
|
}
|
||||||
|
if (desired_access & FileAccess::kGenericExecute) {
|
||||||
|
open_access |= GENERIC_EXECUTE;
|
||||||
|
}
|
||||||
|
if (desired_access & FileAccess::kGenericAll) {
|
||||||
|
open_access |= GENERIC_ALL;
|
||||||
|
}
|
||||||
|
if (desired_access & FileAccess::kFileReadData) {
|
||||||
|
open_access |= FILE_READ_DATA;
|
||||||
|
}
|
||||||
|
if (desired_access & FileAccess::kFileWriteData) {
|
||||||
|
open_access |= FILE_WRITE_DATA;
|
||||||
|
}
|
||||||
|
if (desired_access & FileAccess::kFileAppendData) {
|
||||||
|
open_access |= FILE_APPEND_DATA;
|
||||||
|
}
|
||||||
|
DWORD share_mode = FILE_SHARE_READ;
|
||||||
|
// We assume we've already created the file in the caller.
|
||||||
|
DWORD creation_disposition = OPEN_EXISTING;
|
||||||
|
HANDLE handle =
|
||||||
|
CreateFileW(path.c_str(), open_access, share_mode, NULL,
|
||||||
|
creation_disposition, FILE_FLAG_BACKUP_SEMANTICS, NULL);
|
||||||
|
if (handle == INVALID_HANDLE_VALUE) {
|
||||||
|
// TODO(benvanik): pick correct response.
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
return std::make_unique<Win32FileHandle>(path, handle);
|
||||||
|
}
|
||||||
|
|
||||||
#define COMBINE_TIME(t) (((uint64_t)t.dwHighDateTime << 32) | t.dwLowDateTime)
|
#define COMBINE_TIME(t) (((uint64_t)t.dwHighDateTime << 32) | t.dwLowDateTime)
|
||||||
|
|
||||||
bool GetInfo(const std::wstring& path, FileInfo* out_info) {
|
bool GetInfo(const std::wstring& path, FileInfo* out_info) {
|
||||||
|
|
|
@ -9,10 +9,10 @@
|
||||||
|
|
||||||
#include "xenia/vfs/devices/host_path_entry.h"
|
#include "xenia/vfs/devices/host_path_entry.h"
|
||||||
|
|
||||||
|
#include "xenia/base/filesystem.h"
|
||||||
#include "xenia/base/logging.h"
|
#include "xenia/base/logging.h"
|
||||||
#include "xenia/base/mapped_memory.h"
|
#include "xenia/base/mapped_memory.h"
|
||||||
#include "xenia/base/math.h"
|
#include "xenia/base/math.h"
|
||||||
#include "xenia/base/platform_win.h"
|
|
||||||
#include "xenia/base/string.h"
|
#include "xenia/base/string.h"
|
||||||
#include "xenia/vfs/devices/host_path_file.h"
|
#include "xenia/vfs/devices/host_path_file.h"
|
||||||
|
|
||||||
|
@ -54,41 +54,14 @@ X_STATUS HostPathEntry::Open(KernelState* kernel_state, uint32_t desired_access,
|
||||||
XELOGE("Attempting to open file for write access on read-only device");
|
XELOGE("Attempting to open file for write access on read-only device");
|
||||||
return X_STATUS_ACCESS_DENIED;
|
return X_STATUS_ACCESS_DENIED;
|
||||||
}
|
}
|
||||||
DWORD open_access = 0;
|
auto file_handle =
|
||||||
if (desired_access & FileAccess::kGenericRead) {
|
xe::filesystem::FileHandle::OpenExisting(local_path_, desired_access);
|
||||||
open_access |= GENERIC_READ;
|
if (!file_handle) {
|
||||||
}
|
|
||||||
if (desired_access & FileAccess::kGenericWrite) {
|
|
||||||
open_access |= GENERIC_WRITE;
|
|
||||||
}
|
|
||||||
if (desired_access & FileAccess::kGenericExecute) {
|
|
||||||
open_access |= GENERIC_EXECUTE;
|
|
||||||
}
|
|
||||||
if (desired_access & FileAccess::kGenericAll) {
|
|
||||||
open_access |= GENERIC_ALL;
|
|
||||||
}
|
|
||||||
if (desired_access & FileAccess::kFileReadData) {
|
|
||||||
open_access |= FILE_READ_DATA;
|
|
||||||
}
|
|
||||||
if (desired_access & FileAccess::kFileWriteData) {
|
|
||||||
open_access |= FILE_WRITE_DATA;
|
|
||||||
}
|
|
||||||
if (desired_access & FileAccess::kFileAppendData) {
|
|
||||||
open_access |= FILE_APPEND_DATA;
|
|
||||||
}
|
|
||||||
DWORD share_mode = FILE_SHARE_READ;
|
|
||||||
// We assume we've already created the file in the caller.
|
|
||||||
DWORD creation_disposition = OPEN_EXISTING;
|
|
||||||
HANDLE file =
|
|
||||||
CreateFileW(local_path_.c_str(), desired_access, share_mode, NULL,
|
|
||||||
creation_disposition, FILE_FLAG_BACKUP_SEMANTICS, NULL);
|
|
||||||
if (file == INVALID_HANDLE_VALUE) {
|
|
||||||
// TODO(benvanik): pick correct response.
|
// TODO(benvanik): pick correct response.
|
||||||
return X_STATUS_NO_SUCH_FILE;
|
return X_STATUS_NO_SUCH_FILE;
|
||||||
}
|
}
|
||||||
|
*out_file = object_ref<XFile>(new HostPathFile(kernel_state, desired_access,
|
||||||
*out_file = object_ref<XFile>(
|
this, std::move(file_handle)));
|
||||||
new HostPathFile(kernel_state, desired_access, this, file));
|
|
||||||
return X_STATUS_SUCCESS;
|
return X_STATUS_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -9,19 +9,19 @@
|
||||||
|
|
||||||
#include "xenia/vfs/devices/host_path_file.h"
|
#include "xenia/vfs/devices/host_path_file.h"
|
||||||
|
|
||||||
#include "xenia/base/platform_win.h"
|
|
||||||
#include "xenia/vfs/devices/host_path_entry.h"
|
#include "xenia/vfs/devices/host_path_entry.h"
|
||||||
|
|
||||||
namespace xe {
|
namespace xe {
|
||||||
namespace vfs {
|
namespace vfs {
|
||||||
|
|
||||||
HostPathFile::HostPathFile(KernelState* kernel_state, uint32_t file_access,
|
HostPathFile::HostPathFile(
|
||||||
HostPathEntry* entry, HANDLE file_handle)
|
KernelState* kernel_state, uint32_t file_access, HostPathEntry* entry,
|
||||||
|
std::unique_ptr<xe::filesystem::FileHandle> file_handle)
|
||||||
: XFile(kernel_state, file_access, entry),
|
: XFile(kernel_state, file_access, entry),
|
||||||
entry_(entry),
|
entry_(entry),
|
||||||
file_handle_(file_handle) {}
|
file_handle_(std::move(file_handle)) {}
|
||||||
|
|
||||||
HostPathFile::~HostPathFile() { CloseHandle(file_handle_); }
|
HostPathFile::~HostPathFile() = default;
|
||||||
|
|
||||||
X_STATUS HostPathFile::ReadSync(void* buffer, size_t buffer_length,
|
X_STATUS HostPathFile::ReadSync(void* buffer, size_t buffer_length,
|
||||||
size_t byte_offset, size_t* out_bytes_read) {
|
size_t byte_offset, size_t* out_bytes_read) {
|
||||||
|
@ -29,14 +29,7 @@ X_STATUS HostPathFile::ReadSync(void* buffer, size_t buffer_length,
|
||||||
return X_STATUS_ACCESS_DENIED;
|
return X_STATUS_ACCESS_DENIED;
|
||||||
}
|
}
|
||||||
|
|
||||||
OVERLAPPED overlapped;
|
if (file_handle_->Read(byte_offset, buffer, buffer_length, out_bytes_read)) {
|
||||||
overlapped.Pointer = (PVOID)byte_offset;
|
|
||||||
overlapped.hEvent = NULL;
|
|
||||||
DWORD bytes_read = 0;
|
|
||||||
BOOL read = ReadFile(file_handle_, buffer, (DWORD)buffer_length, &bytes_read,
|
|
||||||
&overlapped);
|
|
||||||
if (read) {
|
|
||||||
*out_bytes_read = bytes_read;
|
|
||||||
return X_STATUS_SUCCESS;
|
return X_STATUS_SUCCESS;
|
||||||
} else {
|
} else {
|
||||||
return X_STATUS_END_OF_FILE;
|
return X_STATUS_END_OF_FILE;
|
||||||
|
@ -51,14 +44,8 @@ X_STATUS HostPathFile::WriteSync(const void* buffer, size_t buffer_length,
|
||||||
return X_STATUS_ACCESS_DENIED;
|
return X_STATUS_ACCESS_DENIED;
|
||||||
}
|
}
|
||||||
|
|
||||||
OVERLAPPED overlapped;
|
if (file_handle_->Write(byte_offset, buffer, buffer_length,
|
||||||
overlapped.Pointer = (PVOID)byte_offset;
|
out_bytes_written)) {
|
||||||
overlapped.hEvent = NULL;
|
|
||||||
DWORD bytes_written = 0;
|
|
||||||
BOOL wrote = WriteFile(file_handle_, buffer, (DWORD)buffer_length,
|
|
||||||
&bytes_written, &overlapped);
|
|
||||||
if (wrote) {
|
|
||||||
*out_bytes_written = bytes_written;
|
|
||||||
return X_STATUS_SUCCESS;
|
return X_STATUS_SUCCESS;
|
||||||
} else {
|
} else {
|
||||||
return X_STATUS_END_OF_FILE;
|
return X_STATUS_END_OF_FILE;
|
||||||
|
|
|
@ -12,6 +12,7 @@
|
||||||
|
|
||||||
#include <string>
|
#include <string>
|
||||||
|
|
||||||
|
#include "xenia/base/filesystem.h"
|
||||||
#include "xenia/kernel/objects/xfile.h"
|
#include "xenia/kernel/objects/xfile.h"
|
||||||
|
|
||||||
typedef void* HANDLE;
|
typedef void* HANDLE;
|
||||||
|
@ -24,7 +25,8 @@ class HostPathEntry;
|
||||||
class HostPathFile : public XFile {
|
class HostPathFile : public XFile {
|
||||||
public:
|
public:
|
||||||
HostPathFile(KernelState* kernel_state, uint32_t file_access,
|
HostPathFile(KernelState* kernel_state, uint32_t file_access,
|
||||||
HostPathEntry* entry, HANDLE file_handle);
|
HostPathEntry* entry,
|
||||||
|
std::unique_ptr<xe::filesystem::FileHandle> file_handle);
|
||||||
~HostPathFile() override;
|
~HostPathFile() override;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
@ -35,7 +37,7 @@ class HostPathFile : public XFile {
|
||||||
|
|
||||||
private:
|
private:
|
||||||
HostPathEntry* entry_;
|
HostPathEntry* entry_;
|
||||||
HANDLE file_handle_;
|
std::unique_ptr<xe::filesystem::FileHandle> file_handle_;
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace vfs
|
} // namespace vfs
|
||||||
|
|
|
@ -14,6 +14,7 @@
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
|
#include "xenia/base/filesystem.h"
|
||||||
#include "xenia/base/mapped_memory.h"
|
#include "xenia/base/mapped_memory.h"
|
||||||
#include "xenia/base/string_buffer.h"
|
#include "xenia/base/string_buffer.h"
|
||||||
#include "xenia/kernel/xobject.h"
|
#include "xenia/kernel/xobject.h"
|
||||||
|
@ -61,17 +62,8 @@ enum class FileDisposition {
|
||||||
kOverwriteIf = 5,
|
kOverwriteIf = 5,
|
||||||
};
|
};
|
||||||
|
|
||||||
struct FileAccess {
|
// Reuse xe::filesystem definition.
|
||||||
// Implies kFileReadData.
|
using FileAccess = xe::filesystem::FileAccess;
|
||||||
static const uint32_t kGenericRead = 0x80000000;
|
|
||||||
// Implies kFileWriteData.
|
|
||||||
static const uint32_t kGenericWrite = 0x40000000;
|
|
||||||
static const uint32_t kGenericExecute = 0x20000000;
|
|
||||||
static const uint32_t kGenericAll = 0x10000000;
|
|
||||||
static const uint32_t kFileReadData = 0x00000001;
|
|
||||||
static const uint32_t kFileWriteData = 0x00000002;
|
|
||||||
static const uint32_t kFileAppendData = 0x00000004;
|
|
||||||
};
|
|
||||||
|
|
||||||
enum FileAttributeFlags : uint32_t {
|
enum FileAttributeFlags : uint32_t {
|
||||||
kFileAttributeNone = 0x0000,
|
kFileAttributeNone = 0x0000,
|
||||||
|
|
Loading…
Reference in New Issue