Browse Source
fs: Move fsp_srv subclasses to separate files
fs: Move fsp_srv subclasses to separate files
fs: Move additional files to the fsp directorynce_cpp
committed by
Liam
17 changed files with 687 additions and 542 deletions
-
13src/core/CMakeLists.txt
-
6src/core/hle/service/filesystem/filesystem.cpp
-
79src/core/hle/service/filesystem/fsp/fs_i_directory.cpp
-
27src/core/hle/service/filesystem/fsp/fs_i_directory.h
-
127src/core/hle/service/filesystem/fsp/fs_i_file.cpp
-
26src/core/hle/service/filesystem/fsp/fs_i_file.h
-
262src/core/hle/service/filesystem/fsp/fs_i_filesystem.cpp
-
38src/core/hle/service/filesystem/fsp/fs_i_filesystem.h
-
62src/core/hle/service/filesystem/fsp/fs_i_storage.cpp
-
23src/core/hle/service/filesystem/fsp/fs_i_storage.h
-
2src/core/hle/service/filesystem/fsp/fsp_ldr.cpp
-
0src/core/hle/service/filesystem/fsp/fsp_ldr.h
-
2src/core/hle/service/filesystem/fsp/fsp_pr.cpp
-
0src/core/hle/service/filesystem/fsp/fsp_pr.h
-
540src/core/hle/service/filesystem/fsp/fsp_srv.cpp
-
0src/core/hle/service/filesystem/fsp/fsp_srv.h
-
22src/core/hle/service/filesystem/fsp/fsp_util.h
@ -0,0 +1,79 @@ |
|||||
|
// SPDX-FileCopyrightText: Copyright 2023 yuzu Emulator Project
|
||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||
|
|
||||
|
#include "core/file_sys/savedata_factory.h"
|
||||
|
#include "core/hle/service/filesystem/fsp/fs_i_directory.h"
|
||||
|
#include "core/hle/service/ipc_helpers.h"
|
||||
|
|
||||
|
namespace Service::FileSystem { |
||||
|
|
||||
|
template <typename T> |
||||
|
static void BuildEntryIndex(std::vector<FileSys::Entry>& entries, const std::vector<T>& new_data, |
||||
|
FileSys::EntryType type) { |
||||
|
entries.reserve(entries.size() + new_data.size()); |
||||
|
|
||||
|
for (const auto& new_entry : new_data) { |
||||
|
auto name = new_entry->GetName(); |
||||
|
|
||||
|
if (type == FileSys::EntryType::File && name == FileSys::GetSaveDataSizeFileName()) { |
||||
|
continue; |
||||
|
} |
||||
|
|
||||
|
entries.emplace_back(name, type, |
||||
|
type == FileSys::EntryType::Directory ? 0 : new_entry->GetSize()); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
IDirectory::IDirectory(Core::System& system_, FileSys::VirtualDir backend_, OpenDirectoryMode mode) |
||||
|
: ServiceFramework{system_, "IDirectory"}, backend(std::move(backend_)) { |
||||
|
static const FunctionInfo functions[] = { |
||||
|
{0, &IDirectory::Read, "Read"}, |
||||
|
{1, &IDirectory::GetEntryCount, "GetEntryCount"}, |
||||
|
}; |
||||
|
RegisterHandlers(functions); |
||||
|
|
||||
|
// TODO(DarkLordZach): Verify that this is the correct behavior.
|
||||
|
// Build entry index now to save time later.
|
||||
|
if (True(mode & OpenDirectoryMode::Directory)) { |
||||
|
BuildEntryIndex(entries, backend->GetSubdirectories(), FileSys::EntryType::Directory); |
||||
|
} |
||||
|
if (True(mode & OpenDirectoryMode::File)) { |
||||
|
BuildEntryIndex(entries, backend->GetFiles(), FileSys::EntryType::File); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
void IDirectory::Read(HLERequestContext& ctx) { |
||||
|
LOG_DEBUG(Service_FS, "called."); |
||||
|
|
||||
|
// Calculate how many entries we can fit in the output buffer
|
||||
|
const u64 count_entries = ctx.GetWriteBufferNumElements<FileSys::Entry>(); |
||||
|
|
||||
|
// Cap at total number of entries.
|
||||
|
const u64 actual_entries = std::min(count_entries, entries.size() - next_entry_index); |
||||
|
|
||||
|
// Determine data start and end
|
||||
|
const auto* begin = reinterpret_cast<u8*>(entries.data() + next_entry_index); |
||||
|
const auto* end = reinterpret_cast<u8*>(entries.data() + next_entry_index + actual_entries); |
||||
|
const auto range_size = static_cast<std::size_t>(std::distance(begin, end)); |
||||
|
|
||||
|
next_entry_index += actual_entries; |
||||
|
|
||||
|
// Write the data to memory
|
||||
|
ctx.WriteBuffer(begin, range_size); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 4}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
rb.Push(actual_entries); |
||||
|
} |
||||
|
|
||||
|
void IDirectory::GetEntryCount(HLERequestContext& ctx) { |
||||
|
LOG_DEBUG(Service_FS, "called"); |
||||
|
|
||||
|
u64 count = entries.size() - next_entry_index; |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 4}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
rb.Push(count); |
||||
|
} |
||||
|
|
||||
|
} // namespace Service::FileSystem
|
||||
@ -0,0 +1,27 @@ |
|||||
|
// SPDX-FileCopyrightText: Copyright 2023 yuzu Emulator Project |
||||
|
// SPDX-License-Identifier: GPL-2.0-or-later |
||||
|
|
||||
|
#pragma once |
||||
|
|
||||
|
#include "core/file_sys/vfs.h" |
||||
|
#include "core/hle/service/filesystem/filesystem.h" |
||||
|
#include "core/hle/service/filesystem/fsp_util.h" |
||||
|
#include "core/hle/service/service.h" |
||||
|
|
||||
|
namespace Service::FileSystem { |
||||
|
|
||||
|
class IDirectory final : public ServiceFramework<IDirectory> { |
||||
|
public: |
||||
|
explicit IDirectory(Core::System& system_, FileSys::VirtualDir backend_, |
||||
|
OpenDirectoryMode mode); |
||||
|
|
||||
|
private: |
||||
|
FileSys::VirtualDir backend; |
||||
|
std::vector<FileSys::Entry> entries; |
||||
|
u64 next_entry_index = 0; |
||||
|
|
||||
|
void Read(HLERequestContext& ctx); |
||||
|
void GetEntryCount(HLERequestContext& ctx); |
||||
|
}; |
||||
|
|
||||
|
} // namespace Service::FileSystem |
||||
@ -0,0 +1,127 @@ |
|||||
|
// SPDX-FileCopyrightText: Copyright 2023 yuzu Emulator Project
|
||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||
|
|
||||
|
#include "core/file_sys/errors.h"
|
||||
|
#include "core/hle/service/filesystem/fsp/fs_i_file.h"
|
||||
|
#include "core/hle/service/ipc_helpers.h"
|
||||
|
|
||||
|
namespace Service::FileSystem { |
||||
|
|
||||
|
IFile::IFile(Core::System& system_, FileSys::VirtualFile backend_) |
||||
|
: ServiceFramework{system_, "IFile"}, backend(std::move(backend_)) { |
||||
|
static const FunctionInfo functions[] = { |
||||
|
{0, &IFile::Read, "Read"}, |
||||
|
{1, &IFile::Write, "Write"}, |
||||
|
{2, &IFile::Flush, "Flush"}, |
||||
|
{3, &IFile::SetSize, "SetSize"}, |
||||
|
{4, &IFile::GetSize, "GetSize"}, |
||||
|
{5, nullptr, "OperateRange"}, |
||||
|
{6, nullptr, "OperateRangeWithBuffer"}, |
||||
|
}; |
||||
|
RegisterHandlers(functions); |
||||
|
} |
||||
|
|
||||
|
void IFile::Read(HLERequestContext& ctx) { |
||||
|
IPC::RequestParser rp{ctx}; |
||||
|
const u64 option = rp.Pop<u64>(); |
||||
|
const s64 offset = rp.Pop<s64>(); |
||||
|
const s64 length = rp.Pop<s64>(); |
||||
|
|
||||
|
LOG_DEBUG(Service_FS, "called, option={}, offset=0x{:X}, length={}", option, offset, length); |
||||
|
|
||||
|
// Error checking
|
||||
|
if (length < 0) { |
||||
|
LOG_ERROR(Service_FS, "Length is less than 0, length={}", length); |
||||
|
IPC::ResponseBuilder rb{ctx, 2}; |
||||
|
rb.Push(FileSys::ERROR_INVALID_SIZE); |
||||
|
return; |
||||
|
} |
||||
|
if (offset < 0) { |
||||
|
LOG_ERROR(Service_FS, "Offset is less than 0, offset={}", offset); |
||||
|
IPC::ResponseBuilder rb{ctx, 2}; |
||||
|
rb.Push(FileSys::ERROR_INVALID_OFFSET); |
||||
|
return; |
||||
|
} |
||||
|
|
||||
|
// Read the data from the Storage backend
|
||||
|
std::vector<u8> output = backend->ReadBytes(length, offset); |
||||
|
|
||||
|
// Write the data to memory
|
||||
|
ctx.WriteBuffer(output); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 4}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
rb.Push(static_cast<u64>(output.size())); |
||||
|
} |
||||
|
|
||||
|
void IFile::Write(HLERequestContext& ctx) { |
||||
|
IPC::RequestParser rp{ctx}; |
||||
|
const u64 option = rp.Pop<u64>(); |
||||
|
const s64 offset = rp.Pop<s64>(); |
||||
|
const s64 length = rp.Pop<s64>(); |
||||
|
|
||||
|
LOG_DEBUG(Service_FS, "called, option={}, offset=0x{:X}, length={}", option, offset, length); |
||||
|
|
||||
|
// Error checking
|
||||
|
if (length < 0) { |
||||
|
LOG_ERROR(Service_FS, "Length is less than 0, length={}", length); |
||||
|
IPC::ResponseBuilder rb{ctx, 2}; |
||||
|
rb.Push(FileSys::ERROR_INVALID_SIZE); |
||||
|
return; |
||||
|
} |
||||
|
if (offset < 0) { |
||||
|
LOG_ERROR(Service_FS, "Offset is less than 0, offset={}", offset); |
||||
|
IPC::ResponseBuilder rb{ctx, 2}; |
||||
|
rb.Push(FileSys::ERROR_INVALID_OFFSET); |
||||
|
return; |
||||
|
} |
||||
|
|
||||
|
const auto data = ctx.ReadBuffer(); |
||||
|
|
||||
|
ASSERT_MSG(static_cast<s64>(data.size()) <= length, |
||||
|
"Attempting to write more data than requested (requested={:016X}, actual={:016X}).", |
||||
|
length, data.size()); |
||||
|
|
||||
|
// Write the data to the Storage backend
|
||||
|
const auto write_size = |
||||
|
static_cast<std::size_t>(std::distance(data.begin(), data.begin() + length)); |
||||
|
const std::size_t written = backend->Write(data.data(), write_size, offset); |
||||
|
|
||||
|
ASSERT_MSG(static_cast<s64>(written) == length, |
||||
|
"Could not write all bytes to file (requested={:016X}, actual={:016X}).", length, |
||||
|
written); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 2}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
} |
||||
|
|
||||
|
void IFile::Flush(HLERequestContext& ctx) { |
||||
|
LOG_DEBUG(Service_FS, "called"); |
||||
|
|
||||
|
// Exists for SDK compatibiltity -- No need to flush file.
|
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 2}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
} |
||||
|
|
||||
|
void IFile::SetSize(HLERequestContext& ctx) { |
||||
|
IPC::RequestParser rp{ctx}; |
||||
|
const u64 size = rp.Pop<u64>(); |
||||
|
LOG_DEBUG(Service_FS, "called, size={}", size); |
||||
|
|
||||
|
backend->Resize(size); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 2}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
} |
||||
|
|
||||
|
void IFile::GetSize(HLERequestContext& ctx) { |
||||
|
const u64 size = backend->GetSize(); |
||||
|
LOG_DEBUG(Service_FS, "called, size={}", size); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 4}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
rb.Push<u64>(size); |
||||
|
} |
||||
|
|
||||
|
} // namespace Service::FileSystem
|
||||
@ -0,0 +1,26 @@ |
|||||
|
// SPDX-FileCopyrightText: Copyright 2023 yuzu Emulator Project |
||||
|
// SPDX-License-Identifier: GPL-2.0-or-later |
||||
|
|
||||
|
#pragma once |
||||
|
|
||||
|
#include "core/file_sys/vfs.h" |
||||
|
#include "core/hle/service/filesystem/filesystem.h" |
||||
|
#include "core/hle/service/service.h" |
||||
|
|
||||
|
namespace Service::FileSystem { |
||||
|
|
||||
|
class IFile final : public ServiceFramework<IFile> { |
||||
|
public: |
||||
|
explicit IFile(Core::System& system_, FileSys::VirtualFile backend_); |
||||
|
|
||||
|
private: |
||||
|
FileSys::VirtualFile backend; |
||||
|
|
||||
|
void Read(HLERequestContext& ctx); |
||||
|
void Write(HLERequestContext& ctx); |
||||
|
void Flush(HLERequestContext& ctx); |
||||
|
void SetSize(HLERequestContext& ctx); |
||||
|
void GetSize(HLERequestContext& ctx); |
||||
|
}; |
||||
|
|
||||
|
} // namespace Service::FileSystem |
||||
@ -0,0 +1,262 @@ |
|||||
|
// SPDX-FileCopyrightText: Copyright 2023 yuzu Emulator Project
|
||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||
|
|
||||
|
#include "common/string_util.h"
|
||||
|
#include "core/hle/service/filesystem/fsp/fs_i_directory.h"
|
||||
|
#include "core/hle/service/filesystem/fsp/fs_i_file.h"
|
||||
|
#include "core/hle/service/filesystem/fsp/fs_i_filesystem.h"
|
||||
|
#include "core/hle/service/ipc_helpers.h"
|
||||
|
|
||||
|
namespace Service::FileSystem { |
||||
|
|
||||
|
IFileSystem::IFileSystem(Core::System& system_, FileSys::VirtualDir backend_, SizeGetter size_) |
||||
|
: ServiceFramework{system_, "IFileSystem"}, backend{std::move(backend_)}, |
||||
|
size{std::move(size_)} { |
||||
|
static const FunctionInfo functions[] = { |
||||
|
{0, &IFileSystem::CreateFile, "CreateFile"}, |
||||
|
{1, &IFileSystem::DeleteFile, "DeleteFile"}, |
||||
|
{2, &IFileSystem::CreateDirectory, "CreateDirectory"}, |
||||
|
{3, &IFileSystem::DeleteDirectory, "DeleteDirectory"}, |
||||
|
{4, &IFileSystem::DeleteDirectoryRecursively, "DeleteDirectoryRecursively"}, |
||||
|
{5, &IFileSystem::RenameFile, "RenameFile"}, |
||||
|
{6, nullptr, "RenameDirectory"}, |
||||
|
{7, &IFileSystem::GetEntryType, "GetEntryType"}, |
||||
|
{8, &IFileSystem::OpenFile, "OpenFile"}, |
||||
|
{9, &IFileSystem::OpenDirectory, "OpenDirectory"}, |
||||
|
{10, &IFileSystem::Commit, "Commit"}, |
||||
|
{11, &IFileSystem::GetFreeSpaceSize, "GetFreeSpaceSize"}, |
||||
|
{12, &IFileSystem::GetTotalSpaceSize, "GetTotalSpaceSize"}, |
||||
|
{13, &IFileSystem::CleanDirectoryRecursively, "CleanDirectoryRecursively"}, |
||||
|
{14, &IFileSystem::GetFileTimeStampRaw, "GetFileTimeStampRaw"}, |
||||
|
{15, nullptr, "QueryEntry"}, |
||||
|
{16, &IFileSystem::GetFileSystemAttribute, "GetFileSystemAttribute"}, |
||||
|
}; |
||||
|
RegisterHandlers(functions); |
||||
|
} |
||||
|
|
||||
|
void IFileSystem::CreateFile(HLERequestContext& ctx) { |
||||
|
IPC::RequestParser rp{ctx}; |
||||
|
|
||||
|
const auto file_buffer = ctx.ReadBuffer(); |
||||
|
const std::string name = Common::StringFromBuffer(file_buffer); |
||||
|
|
||||
|
const u64 file_mode = rp.Pop<u64>(); |
||||
|
const u32 file_size = rp.Pop<u32>(); |
||||
|
|
||||
|
LOG_DEBUG(Service_FS, "called. file={}, mode=0x{:X}, size=0x{:08X}", name, file_mode, |
||||
|
file_size); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 2}; |
||||
|
rb.Push(backend.CreateFile(name, file_size)); |
||||
|
} |
||||
|
|
||||
|
void IFileSystem::DeleteFile(HLERequestContext& ctx) { |
||||
|
const auto file_buffer = ctx.ReadBuffer(); |
||||
|
const std::string name = Common::StringFromBuffer(file_buffer); |
||||
|
|
||||
|
LOG_DEBUG(Service_FS, "called. file={}", name); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 2}; |
||||
|
rb.Push(backend.DeleteFile(name)); |
||||
|
} |
||||
|
|
||||
|
void IFileSystem::CreateDirectory(HLERequestContext& ctx) { |
||||
|
const auto file_buffer = ctx.ReadBuffer(); |
||||
|
const std::string name = Common::StringFromBuffer(file_buffer); |
||||
|
|
||||
|
LOG_DEBUG(Service_FS, "called. directory={}", name); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 2}; |
||||
|
rb.Push(backend.CreateDirectory(name)); |
||||
|
} |
||||
|
|
||||
|
void IFileSystem::DeleteDirectory(HLERequestContext& ctx) { |
||||
|
const auto file_buffer = ctx.ReadBuffer(); |
||||
|
const std::string name = Common::StringFromBuffer(file_buffer); |
||||
|
|
||||
|
LOG_DEBUG(Service_FS, "called. directory={}", name); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 2}; |
||||
|
rb.Push(backend.DeleteDirectory(name)); |
||||
|
} |
||||
|
|
||||
|
void IFileSystem::DeleteDirectoryRecursively(HLERequestContext& ctx) { |
||||
|
const auto file_buffer = ctx.ReadBuffer(); |
||||
|
const std::string name = Common::StringFromBuffer(file_buffer); |
||||
|
|
||||
|
LOG_DEBUG(Service_FS, "called. directory={}", name); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 2}; |
||||
|
rb.Push(backend.DeleteDirectoryRecursively(name)); |
||||
|
} |
||||
|
|
||||
|
void IFileSystem::CleanDirectoryRecursively(HLERequestContext& ctx) { |
||||
|
const auto file_buffer = ctx.ReadBuffer(); |
||||
|
const std::string name = Common::StringFromBuffer(file_buffer); |
||||
|
|
||||
|
LOG_DEBUG(Service_FS, "called. Directory: {}", name); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 2}; |
||||
|
rb.Push(backend.CleanDirectoryRecursively(name)); |
||||
|
} |
||||
|
|
||||
|
void IFileSystem::RenameFile(HLERequestContext& ctx) { |
||||
|
const std::string src_name = Common::StringFromBuffer(ctx.ReadBuffer(0)); |
||||
|
const std::string dst_name = Common::StringFromBuffer(ctx.ReadBuffer(1)); |
||||
|
|
||||
|
LOG_DEBUG(Service_FS, "called. file '{}' to file '{}'", src_name, dst_name); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 2}; |
||||
|
rb.Push(backend.RenameFile(src_name, dst_name)); |
||||
|
} |
||||
|
|
||||
|
void IFileSystem::OpenFile(HLERequestContext& ctx) { |
||||
|
IPC::RequestParser rp{ctx}; |
||||
|
|
||||
|
const auto file_buffer = ctx.ReadBuffer(); |
||||
|
const std::string name = Common::StringFromBuffer(file_buffer); |
||||
|
|
||||
|
const auto mode = static_cast<FileSys::Mode>(rp.Pop<u32>()); |
||||
|
|
||||
|
LOG_DEBUG(Service_FS, "called. file={}, mode={}", name, mode); |
||||
|
|
||||
|
FileSys::VirtualFile vfs_file{}; |
||||
|
auto result = backend.OpenFile(&vfs_file, name, mode); |
||||
|
if (result != ResultSuccess) { |
||||
|
IPC::ResponseBuilder rb{ctx, 2}; |
||||
|
rb.Push(result); |
||||
|
return; |
||||
|
} |
||||
|
|
||||
|
auto file = std::make_shared<IFile>(system, vfs_file); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 2, 0, 1}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
rb.PushIpcInterface<IFile>(std::move(file)); |
||||
|
} |
||||
|
|
||||
|
void IFileSystem::OpenDirectory(HLERequestContext& ctx) { |
||||
|
IPC::RequestParser rp{ctx}; |
||||
|
|
||||
|
const auto file_buffer = ctx.ReadBuffer(); |
||||
|
const std::string name = Common::StringFromBuffer(file_buffer); |
||||
|
const auto mode = rp.PopRaw<OpenDirectoryMode>(); |
||||
|
|
||||
|
LOG_DEBUG(Service_FS, "called. directory={}, mode={}", name, mode); |
||||
|
|
||||
|
FileSys::VirtualDir vfs_dir{}; |
||||
|
auto result = backend.OpenDirectory(&vfs_dir, name); |
||||
|
if (result != ResultSuccess) { |
||||
|
IPC::ResponseBuilder rb{ctx, 2}; |
||||
|
rb.Push(result); |
||||
|
return; |
||||
|
} |
||||
|
|
||||
|
auto directory = std::make_shared<IDirectory>(system, vfs_dir, mode); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 2, 0, 1}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
rb.PushIpcInterface<IDirectory>(std::move(directory)); |
||||
|
} |
||||
|
|
||||
|
void IFileSystem::GetEntryType(HLERequestContext& ctx) { |
||||
|
const auto file_buffer = ctx.ReadBuffer(); |
||||
|
const std::string name = Common::StringFromBuffer(file_buffer); |
||||
|
|
||||
|
LOG_DEBUG(Service_FS, "called. file={}", name); |
||||
|
|
||||
|
FileSys::EntryType vfs_entry_type{}; |
||||
|
auto result = backend.GetEntryType(&vfs_entry_type, name); |
||||
|
if (result != ResultSuccess) { |
||||
|
IPC::ResponseBuilder rb{ctx, 2}; |
||||
|
rb.Push(result); |
||||
|
return; |
||||
|
} |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 3}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
rb.Push<u32>(static_cast<u32>(vfs_entry_type)); |
||||
|
} |
||||
|
|
||||
|
void IFileSystem::Commit(HLERequestContext& ctx) { |
||||
|
LOG_WARNING(Service_FS, "(STUBBED) called"); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 2}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
} |
||||
|
|
||||
|
void IFileSystem::GetFreeSpaceSize(HLERequestContext& ctx) { |
||||
|
LOG_DEBUG(Service_FS, "called"); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 4}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
rb.Push(size.get_free_size()); |
||||
|
} |
||||
|
|
||||
|
void IFileSystem::GetTotalSpaceSize(HLERequestContext& ctx) { |
||||
|
LOG_DEBUG(Service_FS, "called"); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 4}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
rb.Push(size.get_total_size()); |
||||
|
} |
||||
|
|
||||
|
void IFileSystem::GetFileTimeStampRaw(HLERequestContext& ctx) { |
||||
|
const auto file_buffer = ctx.ReadBuffer(); |
||||
|
const std::string name = Common::StringFromBuffer(file_buffer); |
||||
|
|
||||
|
LOG_WARNING(Service_FS, "(Partial Implementation) called. file={}", name); |
||||
|
|
||||
|
FileSys::FileTimeStampRaw vfs_timestamp{}; |
||||
|
auto result = backend.GetFileTimeStampRaw(&vfs_timestamp, name); |
||||
|
if (result != ResultSuccess) { |
||||
|
IPC::ResponseBuilder rb{ctx, 2}; |
||||
|
rb.Push(result); |
||||
|
return; |
||||
|
} |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 10}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
rb.PushRaw(vfs_timestamp); |
||||
|
} |
||||
|
|
||||
|
void IFileSystem::GetFileSystemAttribute(HLERequestContext& ctx) { |
||||
|
LOG_WARNING(Service_FS, "(STUBBED) called"); |
||||
|
|
||||
|
struct FileSystemAttribute { |
||||
|
u8 dir_entry_name_length_max_defined; |
||||
|
u8 file_entry_name_length_max_defined; |
||||
|
u8 dir_path_name_length_max_defined; |
||||
|
u8 file_path_name_length_max_defined; |
||||
|
INSERT_PADDING_BYTES_NOINIT(0x5); |
||||
|
u8 utf16_dir_entry_name_length_max_defined; |
||||
|
u8 utf16_file_entry_name_length_max_defined; |
||||
|
u8 utf16_dir_path_name_length_max_defined; |
||||
|
u8 utf16_file_path_name_length_max_defined; |
||||
|
INSERT_PADDING_BYTES_NOINIT(0x18); |
||||
|
s32 dir_entry_name_length_max; |
||||
|
s32 file_entry_name_length_max; |
||||
|
s32 dir_path_name_length_max; |
||||
|
s32 file_path_name_length_max; |
||||
|
INSERT_PADDING_WORDS_NOINIT(0x5); |
||||
|
s32 utf16_dir_entry_name_length_max; |
||||
|
s32 utf16_file_entry_name_length_max; |
||||
|
s32 utf16_dir_path_name_length_max; |
||||
|
s32 utf16_file_path_name_length_max; |
||||
|
INSERT_PADDING_WORDS_NOINIT(0x18); |
||||
|
INSERT_PADDING_WORDS_NOINIT(0x1); |
||||
|
}; |
||||
|
static_assert(sizeof(FileSystemAttribute) == 0xc0, "FileSystemAttribute has incorrect size"); |
||||
|
|
||||
|
FileSystemAttribute savedata_attribute{}; |
||||
|
savedata_attribute.dir_entry_name_length_max_defined = true; |
||||
|
savedata_attribute.file_entry_name_length_max_defined = true; |
||||
|
savedata_attribute.dir_entry_name_length_max = 0x40; |
||||
|
savedata_attribute.file_entry_name_length_max = 0x40; |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 50}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
rb.PushRaw(savedata_attribute); |
||||
|
} |
||||
|
|
||||
|
} // namespace Service::FileSystem
|
||||
@ -0,0 +1,38 @@ |
|||||
|
// SPDX-FileCopyrightText: Copyright 2023 yuzu Emulator Project |
||||
|
// SPDX-License-Identifier: GPL-2.0-or-later |
||||
|
|
||||
|
#pragma once |
||||
|
|
||||
|
#include "core/file_sys/vfs.h" |
||||
|
#include "core/hle/service/filesystem/filesystem.h" |
||||
|
#include "core/hle/service/filesystem/fsp/fsp_util.h" |
||||
|
#include "core/hle/service/service.h" |
||||
|
|
||||
|
namespace Service::FileSystem { |
||||
|
|
||||
|
class IFileSystem final : public ServiceFramework<IFileSystem> { |
||||
|
public: |
||||
|
explicit IFileSystem(Core::System& system_, FileSys::VirtualDir backend_, SizeGetter size_); |
||||
|
|
||||
|
void CreateFile(HLERequestContext& ctx); |
||||
|
void DeleteFile(HLERequestContext& ctx); |
||||
|
void CreateDirectory(HLERequestContext& ctx); |
||||
|
void DeleteDirectory(HLERequestContext& ctx); |
||||
|
void DeleteDirectoryRecursively(HLERequestContext& ctx); |
||||
|
void CleanDirectoryRecursively(HLERequestContext& ctx); |
||||
|
void RenameFile(HLERequestContext& ctx); |
||||
|
void OpenFile(HLERequestContext& ctx); |
||||
|
void OpenDirectory(HLERequestContext& ctx); |
||||
|
void GetEntryType(HLERequestContext& ctx); |
||||
|
void Commit(HLERequestContext& ctx); |
||||
|
void GetFreeSpaceSize(HLERequestContext& ctx); |
||||
|
void GetTotalSpaceSize(HLERequestContext& ctx); |
||||
|
void GetFileTimeStampRaw(HLERequestContext& ctx); |
||||
|
void GetFileSystemAttribute(HLERequestContext& ctx); |
||||
|
|
||||
|
private: |
||||
|
VfsDirectoryServiceWrapper backend; |
||||
|
SizeGetter size; |
||||
|
}; |
||||
|
|
||||
|
} // namespace Service::FileSystem |
||||
@ -0,0 +1,62 @@ |
|||||
|
// SPDX-FileCopyrightText: Copyright 2023 yuzu Emulator Project
|
||||
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||
|
|
||||
|
#include "core/file_sys/errors.h"
|
||||
|
#include "core/hle/service/filesystem/fsp/fs_i_storage.h"
|
||||
|
#include "core/hle/service/ipc_helpers.h"
|
||||
|
|
||||
|
namespace Service::FileSystem { |
||||
|
|
||||
|
IStorage::IStorage(Core::System& system_, FileSys::VirtualFile backend_) |
||||
|
: ServiceFramework{system_, "IStorage"}, backend(std::move(backend_)) { |
||||
|
static const FunctionInfo functions[] = { |
||||
|
{0, &IStorage::Read, "Read"}, |
||||
|
{1, nullptr, "Write"}, |
||||
|
{2, nullptr, "Flush"}, |
||||
|
{3, nullptr, "SetSize"}, |
||||
|
{4, &IStorage::GetSize, "GetSize"}, |
||||
|
{5, nullptr, "OperateRange"}, |
||||
|
}; |
||||
|
RegisterHandlers(functions); |
||||
|
} |
||||
|
|
||||
|
void IStorage::Read(HLERequestContext& ctx) { |
||||
|
IPC::RequestParser rp{ctx}; |
||||
|
const s64 offset = rp.Pop<s64>(); |
||||
|
const s64 length = rp.Pop<s64>(); |
||||
|
|
||||
|
LOG_DEBUG(Service_FS, "called, offset=0x{:X}, length={}", offset, length); |
||||
|
|
||||
|
// Error checking
|
||||
|
if (length < 0) { |
||||
|
LOG_ERROR(Service_FS, "Length is less than 0, length={}", length); |
||||
|
IPC::ResponseBuilder rb{ctx, 2}; |
||||
|
rb.Push(FileSys::ERROR_INVALID_SIZE); |
||||
|
return; |
||||
|
} |
||||
|
if (offset < 0) { |
||||
|
LOG_ERROR(Service_FS, "Offset is less than 0, offset={}", offset); |
||||
|
IPC::ResponseBuilder rb{ctx, 2}; |
||||
|
rb.Push(FileSys::ERROR_INVALID_OFFSET); |
||||
|
return; |
||||
|
} |
||||
|
|
||||
|
// Read the data from the Storage backend
|
||||
|
std::vector<u8> output = backend->ReadBytes(length, offset); |
||||
|
// Write the data to memory
|
||||
|
ctx.WriteBuffer(output); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 2}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
} |
||||
|
|
||||
|
void IStorage::GetSize(HLERequestContext& ctx) { |
||||
|
const u64 size = backend->GetSize(); |
||||
|
LOG_DEBUG(Service_FS, "called, size={}", size); |
||||
|
|
||||
|
IPC::ResponseBuilder rb{ctx, 4}; |
||||
|
rb.Push(ResultSuccess); |
||||
|
rb.Push<u64>(size); |
||||
|
} |
||||
|
|
||||
|
} // namespace Service::FileSystem
|
||||
@ -0,0 +1,23 @@ |
|||||
|
// SPDX-FileCopyrightText: Copyright 2023 yuzu Emulator Project |
||||
|
// SPDX-License-Identifier: GPL-2.0-or-later |
||||
|
|
||||
|
#pragma once |
||||
|
|
||||
|
#include "core/file_sys/vfs.h" |
||||
|
#include "core/hle/service/filesystem/filesystem.h" |
||||
|
#include "core/hle/service/service.h" |
||||
|
|
||||
|
namespace Service::FileSystem { |
||||
|
|
||||
|
class IStorage final : public ServiceFramework<IStorage> { |
||||
|
public: |
||||
|
explicit IStorage(Core::System& system_, FileSys::VirtualFile backend_); |
||||
|
|
||||
|
private: |
||||
|
FileSys::VirtualFile backend; |
||||
|
|
||||
|
void Read(HLERequestContext& ctx); |
||||
|
void GetSize(HLERequestContext& ctx); |
||||
|
}; |
||||
|
|
||||
|
} // namespace Service::FileSystem |
||||
@ -1,7 +1,7 @@ |
|||||
// SPDX-FileCopyrightText: Copyright 2018 yuzu Emulator Project
|
// SPDX-FileCopyrightText: Copyright 2018 yuzu Emulator Project
|
||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||
|
|
||||
#include "core/hle/service/filesystem/fsp_ldr.h"
|
|
||||
|
#include "core/hle/service/filesystem/fsp/fsp_ldr.h"
|
||||
|
|
||||
namespace Service::FileSystem { |
namespace Service::FileSystem { |
||||
|
|
||||
@ -1,7 +1,7 @@ |
|||||
// SPDX-FileCopyrightText: Copyright 2018 yuzu Emulator Project
|
// SPDX-FileCopyrightText: Copyright 2018 yuzu Emulator Project
|
||||
// SPDX-License-Identifier: GPL-2.0-or-later
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
||||
|
|
||||
#include "core/hle/service/filesystem/fsp_pr.h"
|
|
||||
|
#include "core/hle/service/filesystem/fsp/fsp_pr.h"
|
||||
|
|
||||
namespace Service::FileSystem { |
namespace Service::FileSystem { |
||||
|
|
||||
@ -0,0 +1,22 @@ |
|||||
|
// SPDX-FileCopyrightText: Copyright 2023 yuzu Emulator Project |
||||
|
// SPDX-License-Identifier: GPL-2.0-or-later |
||||
|
|
||||
|
#pragma once |
||||
|
|
||||
|
#include "core/hle/service/filesystem/filesystem.h" |
||||
|
|
||||
|
namespace Service::FileSystem { |
||||
|
|
||||
|
struct SizeGetter { |
||||
|
std::function<u64()> get_free_size; |
||||
|
std::function<u64()> get_total_size; |
||||
|
|
||||
|
static SizeGetter FromStorageId(const FileSystemController& fsc, FileSys::StorageId id) { |
||||
|
return { |
||||
|
[&fsc, id] { return fsc.GetFreeSpaceSize(id); }, |
||||
|
[&fsc, id] { return fsc.GetTotalSpaceSize(id); }, |
||||
|
}; |
||||
|
} |
||||
|
}; |
||||
|
|
||||
|
} // namespace Service::FileSystem |
||||
Write
Preview
Loading…
Cancel
Save
Reference in new issue