2018-01-17 04:20:12 +01:00
|
|
|
// Copyright 2018 yuzu emulator team
|
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
2018-02-19 06:33:48 +01:00
|
|
|
#include <cinttypes>
|
2018-01-17 04:20:12 +01:00
|
|
|
#include "common/logging/log.h"
|
|
|
|
#include "core/core.h"
|
2018-03-20 05:01:47 +01:00
|
|
|
#include "core/file_sys/directory.h"
|
2018-01-17 04:20:12 +01:00
|
|
|
#include "core/file_sys/filesystem.h"
|
|
|
|
#include "core/file_sys/storage.h"
|
|
|
|
#include "core/hle/ipc_helpers.h"
|
|
|
|
#include "core/hle/kernel/client_port.h"
|
|
|
|
#include "core/hle/kernel/client_session.h"
|
|
|
|
#include "core/hle/service/filesystem/filesystem.h"
|
|
|
|
#include "core/hle/service/filesystem/fsp_srv.h"
|
|
|
|
|
|
|
|
namespace Service {
|
|
|
|
namespace FileSystem {
|
|
|
|
|
|
|
|
class IStorage final : public ServiceFramework<IStorage> {
|
|
|
|
public:
|
|
|
|
IStorage(std::unique_ptr<FileSys::StorageBackend>&& backend)
|
|
|
|
: ServiceFramework("IStorage"), backend(std::move(backend)) {
|
|
|
|
static const FunctionInfo functions[] = {
|
2018-04-10 18:30:27 +02:00
|
|
|
{0, &IStorage::Read, "Read"},
|
|
|
|
{1, nullptr, "Write"},
|
|
|
|
{2, nullptr, "Flush"},
|
|
|
|
{3, nullptr, "SetSize"},
|
|
|
|
{4, nullptr, "GetSize"},
|
|
|
|
{5, nullptr, "OperateRange"},
|
2018-01-17 04:20:12 +01:00
|
|
|
};
|
|
|
|
RegisterHandlers(functions);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::unique_ptr<FileSys::StorageBackend> backend;
|
|
|
|
|
|
|
|
void Read(Kernel::HLERequestContext& ctx) {
|
|
|
|
IPC::RequestParser rp{ctx};
|
2018-01-21 03:32:36 +01:00
|
|
|
const s64 offset = rp.Pop<s64>();
|
|
|
|
const s64 length = rp.Pop<s64>();
|
2018-01-17 04:20:12 +01:00
|
|
|
|
2018-03-19 17:43:04 +01:00
|
|
|
LOG_DEBUG(Service_FS, "called, offset=0x%ld, length=0x%ld", offset, length);
|
2018-01-17 04:20:12 +01:00
|
|
|
|
2018-01-21 03:32:36 +01:00
|
|
|
// Error checking
|
|
|
|
if (length < 0) {
|
2018-01-24 01:52:18 +01:00
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
2018-01-21 03:32:36 +01:00
|
|
|
rb.Push(ResultCode(ErrorModule::FS, ErrorDescription::InvalidLength));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (offset < 0) {
|
2018-01-24 01:52:18 +01:00
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
2018-01-21 03:32:36 +01:00
|
|
|
rb.Push(ResultCode(ErrorModule::FS, ErrorDescription::InvalidOffset));
|
|
|
|
return;
|
|
|
|
}
|
2018-01-17 04:20:12 +01:00
|
|
|
|
2018-01-21 03:32:36 +01:00
|
|
|
// Read the data from the Storage backend
|
|
|
|
std::vector<u8> output(length);
|
2018-01-17 04:20:12 +01:00
|
|
|
ResultVal<size_t> res = backend->Read(offset, length, output.data());
|
|
|
|
if (res.Failed()) {
|
2018-01-24 01:52:18 +01:00
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
2018-01-17 04:20:12 +01:00
|
|
|
rb.Push(res.Code());
|
2018-01-21 03:32:36 +01:00
|
|
|
return;
|
2018-01-17 04:20:12 +01:00
|
|
|
}
|
|
|
|
|
2018-01-21 03:32:36 +01:00
|
|
|
// Write the data to memory
|
2018-02-14 05:16:19 +01:00
|
|
|
ctx.WriteBuffer(output);
|
2018-01-17 04:20:12 +01:00
|
|
|
|
2018-01-24 01:52:18 +01:00
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
2018-01-17 04:20:12 +01:00
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2018-02-19 06:33:48 +01:00
|
|
|
class IFile final : public ServiceFramework<IFile> {
|
|
|
|
public:
|
|
|
|
explicit IFile(std::unique_ptr<FileSys::StorageBackend>&& backend)
|
|
|
|
: ServiceFramework("IFile"), backend(std::move(backend)) {
|
|
|
|
static const FunctionInfo functions[] = {
|
2018-04-10 18:30:27 +02:00
|
|
|
{0, &IFile::Read, "Read"},
|
|
|
|
{1, &IFile::Write, "Write"},
|
|
|
|
{2, nullptr, "Flush"},
|
|
|
|
{3, &IFile::SetSize, "SetSize"},
|
|
|
|
{4, &IFile::GetSize, "GetSize"},
|
|
|
|
{5, nullptr, "OperateRange"},
|
2018-02-19 06:33:48 +01:00
|
|
|
};
|
|
|
|
RegisterHandlers(functions);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::unique_ptr<FileSys::StorageBackend> backend;
|
|
|
|
|
|
|
|
void Read(Kernel::HLERequestContext& ctx) {
|
|
|
|
IPC::RequestParser rp{ctx};
|
|
|
|
const u64 unk = rp.Pop<u64>();
|
|
|
|
const s64 offset = rp.Pop<s64>();
|
|
|
|
const s64 length = rp.Pop<s64>();
|
|
|
|
|
2018-03-19 17:43:04 +01:00
|
|
|
LOG_DEBUG(Service_FS, "called, offset=0x%ld, length=0x%ld", offset, length);
|
2018-02-19 06:33:48 +01:00
|
|
|
|
|
|
|
// Error checking
|
|
|
|
if (length < 0) {
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
rb.Push(ResultCode(ErrorModule::FS, ErrorDescription::InvalidLength));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (offset < 0) {
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
rb.Push(ResultCode(ErrorModule::FS, ErrorDescription::InvalidOffset));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Read the data from the Storage backend
|
|
|
|
std::vector<u8> output(length);
|
|
|
|
ResultVal<size_t> res = backend->Read(offset, length, output.data());
|
|
|
|
if (res.Failed()) {
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
rb.Push(res.Code());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Write the data to memory
|
|
|
|
ctx.WriteBuffer(output);
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 4};
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
rb.Push(static_cast<u64>(*res));
|
|
|
|
}
|
|
|
|
|
|
|
|
void Write(Kernel::HLERequestContext& ctx) {
|
|
|
|
IPC::RequestParser rp{ctx};
|
|
|
|
const u64 unk = rp.Pop<u64>();
|
|
|
|
const s64 offset = rp.Pop<s64>();
|
|
|
|
const s64 length = rp.Pop<s64>();
|
|
|
|
|
2018-03-19 17:43:04 +01:00
|
|
|
LOG_DEBUG(Service_FS, "called, offset=0x%ld, length=0x%ld", offset, length);
|
2018-02-19 06:33:48 +01:00
|
|
|
|
|
|
|
// Error checking
|
|
|
|
if (length < 0) {
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
rb.Push(ResultCode(ErrorModule::FS, ErrorDescription::InvalidLength));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (offset < 0) {
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
rb.Push(ResultCode(ErrorModule::FS, ErrorDescription::InvalidOffset));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Write the data to the Storage backend
|
|
|
|
std::vector<u8> data = ctx.ReadBuffer();
|
|
|
|
ResultVal<size_t> res = backend->Write(offset, length, true, data.data());
|
|
|
|
if (res.Failed()) {
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
rb.Push(res.Code());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
}
|
2018-03-31 22:02:21 +02:00
|
|
|
|
|
|
|
void SetSize(Kernel::HLERequestContext& ctx) {
|
|
|
|
IPC::RequestParser rp{ctx};
|
|
|
|
const u64 size = rp.Pop<u64>();
|
|
|
|
backend->SetSize(size);
|
2018-04-01 05:30:00 +02:00
|
|
|
LOG_DEBUG(Service_FS, "called, size=%" PRIu64, size);
|
2018-03-31 22:02:21 +02:00
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
}
|
|
|
|
|
|
|
|
void GetSize(Kernel::HLERequestContext& ctx) {
|
|
|
|
const u64 size = backend->GetSize();
|
2018-04-01 05:30:00 +02:00
|
|
|
LOG_DEBUG(Service_FS, "called, size=%" PRIu64, size);
|
2018-03-31 22:02:21 +02:00
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 4};
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
rb.Push<u64>(size);
|
|
|
|
}
|
2018-02-19 06:33:48 +01:00
|
|
|
};
|
|
|
|
|
2018-03-20 05:01:47 +01:00
|
|
|
class IDirectory final : public ServiceFramework<IDirectory> {
|
|
|
|
public:
|
|
|
|
explicit IDirectory(std::unique_ptr<FileSys::DirectoryBackend>&& backend)
|
|
|
|
: ServiceFramework("IDirectory"), backend(std::move(backend)) {
|
|
|
|
static const FunctionInfo functions[] = {
|
|
|
|
{0, &IDirectory::Read, "Read"},
|
|
|
|
{1, &IDirectory::GetEntryCount, "GetEntryCount"},
|
|
|
|
};
|
|
|
|
RegisterHandlers(functions);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::unique_ptr<FileSys::DirectoryBackend> backend;
|
|
|
|
|
|
|
|
void Read(Kernel::HLERequestContext& ctx) {
|
|
|
|
IPC::RequestParser rp{ctx};
|
|
|
|
const u64 unk = rp.Pop<u64>();
|
|
|
|
|
|
|
|
LOG_DEBUG(Service_FS, "called, unk=0x%llx", unk);
|
|
|
|
|
|
|
|
// Calculate how many entries we can fit in the output buffer
|
|
|
|
u64 count_entries = ctx.GetWriteBufferSize() / sizeof(FileSys::Entry);
|
|
|
|
|
|
|
|
// Read the data from the Directory backend
|
|
|
|
std::vector<FileSys::Entry> entries(count_entries);
|
|
|
|
u64 read_entries = backend->Read(count_entries, entries.data());
|
|
|
|
|
|
|
|
// Convert the data into a byte array
|
|
|
|
std::vector<u8> output(entries.size() * sizeof(FileSys::Entry));
|
|
|
|
std::memcpy(output.data(), entries.data(), output.size());
|
|
|
|
|
|
|
|
// Write the data to memory
|
|
|
|
ctx.WriteBuffer(output);
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 4};
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
rb.Push(read_entries);
|
|
|
|
}
|
|
|
|
|
|
|
|
void GetEntryCount(Kernel::HLERequestContext& ctx) {
|
|
|
|
LOG_DEBUG(Service_FS, "called");
|
|
|
|
|
|
|
|
u64 count = backend->GetEntryCount();
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 4};
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
rb.Push(count);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2018-02-19 06:33:48 +01:00
|
|
|
class IFileSystem final : public ServiceFramework<IFileSystem> {
|
|
|
|
public:
|
|
|
|
explicit IFileSystem(std::unique_ptr<FileSys::FileSystemBackend>&& backend)
|
|
|
|
: ServiceFramework("IFileSystem"), backend(std::move(backend)) {
|
|
|
|
static const FunctionInfo functions[] = {
|
|
|
|
{0, &IFileSystem::CreateFile, "CreateFile"},
|
2018-04-10 18:30:27 +02:00
|
|
|
{1, nullptr, "DeleteFile"},
|
2018-03-21 15:36:26 +01:00
|
|
|
{2, &IFileSystem::CreateDirectory, "CreateDirectory"},
|
2018-04-10 18:30:27 +02:00
|
|
|
{3, nullptr, "DeleteDirectory"},
|
|
|
|
{4, nullptr, "DeleteDirectoryRecursively"},
|
|
|
|
{5, nullptr, "RenameFile"},
|
|
|
|
{6, nullptr, "RenameDirectory"},
|
2018-02-19 06:33:48 +01:00
|
|
|
{7, &IFileSystem::GetEntryType, "GetEntryType"},
|
|
|
|
{8, &IFileSystem::OpenFile, "OpenFile"},
|
2018-03-20 05:02:30 +01:00
|
|
|
{9, &IFileSystem::OpenDirectory, "OpenDirectory"},
|
2018-02-19 06:33:48 +01:00
|
|
|
{10, &IFileSystem::Commit, "Commit"},
|
2018-04-10 18:30:27 +02:00
|
|
|
{11, nullptr, "GetFreeSpaceSize"},
|
|
|
|
{12, nullptr, "GetTotalSpaceSize"},
|
|
|
|
{13, nullptr, "CleanDirectoryRecursively"},
|
|
|
|
{14, nullptr, "GetFileTimeStampRaw"},
|
|
|
|
{15, nullptr, "QueryEntry"},
|
2018-02-19 06:33:48 +01:00
|
|
|
};
|
|
|
|
RegisterHandlers(functions);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CreateFile(Kernel::HLERequestContext& ctx) {
|
|
|
|
IPC::RequestParser rp{ctx};
|
|
|
|
|
|
|
|
auto file_buffer = ctx.ReadBuffer();
|
|
|
|
auto end = std::find(file_buffer.begin(), file_buffer.end(), '\0');
|
|
|
|
|
|
|
|
std::string name(file_buffer.begin(), end);
|
|
|
|
|
|
|
|
u64 mode = rp.Pop<u64>();
|
|
|
|
u32 size = rp.Pop<u32>();
|
|
|
|
|
|
|
|
LOG_DEBUG(Service_FS, "called file %s mode 0x%" PRIX64 " size 0x%08X", name.c_str(), mode,
|
|
|
|
size);
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
rb.Push(backend->CreateFile(name, size));
|
|
|
|
}
|
|
|
|
|
2018-03-21 15:36:26 +01:00
|
|
|
void CreateDirectory(Kernel::HLERequestContext& ctx) {
|
|
|
|
IPC::RequestParser rp{ctx};
|
|
|
|
|
|
|
|
auto file_buffer = ctx.ReadBuffer();
|
|
|
|
auto end = std::find(file_buffer.begin(), file_buffer.end(), '\0');
|
|
|
|
|
|
|
|
std::string name(file_buffer.begin(), end);
|
|
|
|
|
|
|
|
LOG_DEBUG(Service_FS, "called directory %s", name.c_str());
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
rb.Push(backend->CreateDirectory(name));
|
|
|
|
}
|
|
|
|
|
2018-02-19 06:33:48 +01:00
|
|
|
void OpenFile(Kernel::HLERequestContext& ctx) {
|
|
|
|
IPC::RequestParser rp{ctx};
|
|
|
|
|
|
|
|
auto file_buffer = ctx.ReadBuffer();
|
|
|
|
auto end = std::find(file_buffer.begin(), file_buffer.end(), '\0');
|
|
|
|
|
|
|
|
std::string name(file_buffer.begin(), end);
|
|
|
|
|
|
|
|
auto mode = static_cast<FileSys::Mode>(rp.Pop<u32>());
|
|
|
|
|
|
|
|
LOG_DEBUG(Service_FS, "called file %s mode %u", name.c_str(), static_cast<u32>(mode));
|
|
|
|
|
|
|
|
auto result = backend->OpenFile(name, mode);
|
|
|
|
if (result.Failed()) {
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
rb.Push(result.Code());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto file = std::move(result.Unwrap());
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2, 0, 1};
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
rb.PushIpcInterface<IFile>(std::move(file));
|
|
|
|
}
|
|
|
|
|
2018-03-20 05:02:30 +01:00
|
|
|
void OpenDirectory(Kernel::HLERequestContext& ctx) {
|
|
|
|
IPC::RequestParser rp{ctx};
|
|
|
|
|
|
|
|
auto file_buffer = ctx.ReadBuffer();
|
|
|
|
auto end = std::find(file_buffer.begin(), file_buffer.end(), '\0');
|
|
|
|
|
|
|
|
std::string name(file_buffer.begin(), end);
|
|
|
|
|
|
|
|
// TODO(Subv): Implement this filter.
|
|
|
|
u32 filter_flags = rp.Pop<u32>();
|
|
|
|
|
|
|
|
LOG_DEBUG(Service_FS, "called directory %s filter %u", name.c_str(), filter_flags);
|
|
|
|
|
|
|
|
auto result = backend->OpenDirectory(name);
|
|
|
|
if (result.Failed()) {
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
rb.Push(result.Code());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto directory = std::move(result.Unwrap());
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2, 0, 1};
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
rb.PushIpcInterface<IDirectory>(std::move(directory));
|
|
|
|
}
|
|
|
|
|
2018-02-19 06:33:48 +01:00
|
|
|
void GetEntryType(Kernel::HLERequestContext& ctx) {
|
|
|
|
IPC::RequestParser rp{ctx};
|
|
|
|
|
|
|
|
auto file_buffer = ctx.ReadBuffer();
|
|
|
|
auto end = std::find(file_buffer.begin(), file_buffer.end(), '\0');
|
|
|
|
|
|
|
|
std::string name(file_buffer.begin(), end);
|
|
|
|
|
|
|
|
LOG_DEBUG(Service_FS, "called file %s", name.c_str());
|
|
|
|
|
|
|
|
auto result = backend->GetEntryType(name);
|
|
|
|
if (result.Failed()) {
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
rb.Push(result.Code());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 3};
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
rb.Push<u32>(static_cast<u32>(*result));
|
|
|
|
}
|
|
|
|
|
|
|
|
void Commit(Kernel::HLERequestContext& ctx) {
|
|
|
|
LOG_WARNING(Service_FS, "(STUBBED) called");
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::unique_ptr<FileSys::FileSystemBackend> backend;
|
|
|
|
};
|
|
|
|
|
2018-01-17 04:20:12 +01:00
|
|
|
FSP_SRV::FSP_SRV() : ServiceFramework("fsp-srv") {
|
|
|
|
static const FunctionInfo functions[] = {
|
2018-04-10 18:30:27 +02:00
|
|
|
{0, nullptr, "MountContent"},
|
2018-04-07 15:20:39 +02:00
|
|
|
{1, &FSP_SRV::Initialize, "Initialize"},
|
2018-04-10 18:30:27 +02:00
|
|
|
{2, nullptr, "OpenDataFileSystemByCurrentProcess"},
|
|
|
|
{7, nullptr, "OpenFileSystemWithPatch"},
|
|
|
|
{8, nullptr, "OpenFileSystemWithId"},
|
|
|
|
{9, nullptr, "OpenDataFileSystemByApplicationId"},
|
|
|
|
{11, nullptr, "OpenBisFileSystem"},
|
|
|
|
{12, nullptr, "OpenBisStorage"},
|
|
|
|
{13, nullptr, "InvalidateBisCache"},
|
|
|
|
{17, nullptr, "OpenHostFileSystem"},
|
2018-02-10 05:30:28 +01:00
|
|
|
{18, &FSP_SRV::MountSdCard, "MountSdCard"},
|
2018-04-10 18:30:27 +02:00
|
|
|
{19, nullptr, "FormatSdCardFileSystem"},
|
|
|
|
{21, nullptr, "DeleteSaveDataFileSystem"},
|
2018-03-04 20:31:57 +01:00
|
|
|
{22, &FSP_SRV::CreateSaveData, "CreateSaveData"},
|
2018-04-10 18:30:27 +02:00
|
|
|
{23, nullptr, "CreateSaveDataFileSystemBySystemSaveDataId"},
|
|
|
|
{24, nullptr, "RegisterSaveDataFileSystemAtomicDeletion"},
|
|
|
|
{25, nullptr, "DeleteSaveDataFileSystemBySaveDataSpaceId"},
|
|
|
|
{26, nullptr, "FormatSdCardDryRun"},
|
|
|
|
{27, nullptr, "IsExFatSupported"},
|
|
|
|
{28, nullptr, "DeleteSaveDataFileSystemBySaveDataAttribute"},
|
|
|
|
{30, nullptr, "OpenGameCardStorage"},
|
|
|
|
{31, nullptr, "OpenGameCardFileSystem"},
|
|
|
|
{32, nullptr, "ExtendSaveDataFileSystem"},
|
|
|
|
{33, nullptr, "DeleteCacheStorage"},
|
|
|
|
{34, nullptr, "GetCacheStorageSize"},
|
2018-02-19 06:33:48 +01:00
|
|
|
{51, &FSP_SRV::MountSaveData, "MountSaveData"},
|
2018-04-10 18:30:27 +02:00
|
|
|
{52, nullptr, "OpenSaveDataFileSystemBySystemSaveDataId"},
|
|
|
|
{53, nullptr, "OpenReadOnlySaveDataFileSystem"},
|
|
|
|
{57, nullptr, "ReadSaveDataFileSystemExtraDataBySaveDataSpaceId"},
|
|
|
|
{58, nullptr, "ReadSaveDataFileSystemExtraData"},
|
|
|
|
{59, nullptr, "WriteSaveDataFileSystemExtraData"},
|
|
|
|
{60, nullptr, "OpenSaveDataInfoReader"},
|
|
|
|
{61, nullptr, "OpenSaveDataInfoReaderBySaveDataSpaceId"},
|
|
|
|
{62, nullptr, "OpenCacheStorageList"},
|
|
|
|
{64, nullptr, "OpenSaveDataInternalStorageFileSystem"},
|
|
|
|
{65, nullptr, "UpdateSaveDataMacForDebug"},
|
|
|
|
{66, nullptr, "WriteSaveDataFileSystemExtraData2"},
|
|
|
|
{80, nullptr, "OpenSaveDataMetaFile"},
|
|
|
|
{81, nullptr, "OpenSaveDataTransferManager"},
|
|
|
|
{82, nullptr, "OpenSaveDataTransferManagerVersion2"},
|
|
|
|
{100, nullptr, "OpenImageDirectoryFileSystem"},
|
|
|
|
{110, nullptr, "OpenContentStorageFileSystem"},
|
2018-01-17 04:20:12 +01:00
|
|
|
{200, &FSP_SRV::OpenDataStorageByCurrentProcess, "OpenDataStorageByCurrentProcess"},
|
2018-04-10 18:30:27 +02:00
|
|
|
{201, nullptr, "OpenDataStorageByProgramId"},
|
2018-02-07 13:11:17 +01:00
|
|
|
{202, nullptr, "OpenDataStorageByDataId"},
|
2018-01-17 04:20:12 +01:00
|
|
|
{203, &FSP_SRV::OpenRomStorage, "OpenRomStorage"},
|
2018-04-10 18:30:27 +02:00
|
|
|
{400, nullptr, "OpenDeviceOperator"},
|
|
|
|
{500, nullptr, "OpenSdCardDetectionEventNotifier"},
|
|
|
|
{501, nullptr, "OpenGameCardDetectionEventNotifier"},
|
|
|
|
{510, nullptr, "OpenSystemDataUpdateEventNotifier"},
|
|
|
|
{511, nullptr, "NotifySystemDataUpdateEvent"},
|
|
|
|
{600, nullptr, "SetCurrentPosixTime"},
|
|
|
|
{601, nullptr, "QuerySaveDataTotalSize"},
|
|
|
|
{602, nullptr, "VerifySaveDataFileSystem"},
|
|
|
|
{603, nullptr, "CorruptSaveDataFileSystem"},
|
|
|
|
{604, nullptr, "CreatePaddingFile"},
|
|
|
|
{605, nullptr, "DeleteAllPaddingFiles"},
|
|
|
|
{606, nullptr, "GetRightsId"},
|
|
|
|
{607, nullptr, "RegisterExternalKey"},
|
|
|
|
{608, nullptr, "UnregisterAllExternalKey"},
|
|
|
|
{609, nullptr, "GetRightsIdByPath"},
|
|
|
|
{610, nullptr, "GetRightsIdAndKeyGenerationByPath"},
|
|
|
|
{611, nullptr, "SetCurrentPosixTimeWithTimeDifference"},
|
|
|
|
{612, nullptr, "GetFreeSpaceSizeForSaveData"},
|
|
|
|
{613, nullptr, "VerifySaveDataFileSystemBySaveDataSpaceId"},
|
|
|
|
{614, nullptr, "CorruptSaveDataFileSystemBySaveDataSpaceId"},
|
|
|
|
{615, nullptr, "QuerySaveDataInternalStorageTotalSize"},
|
|
|
|
{620, nullptr, "SetSdCardEncryptionSeed"},
|
|
|
|
{630, nullptr, "SetSdCardAccessibility"},
|
|
|
|
{631, nullptr, "IsSdCardAccessible"},
|
|
|
|
{640, nullptr, "IsSignedSystemPartitionOnSdCardValid"},
|
|
|
|
{700, nullptr, "OpenAccessFailureResolver"},
|
|
|
|
{701, nullptr, "GetAccessFailureDetectionEvent"},
|
|
|
|
{702, nullptr, "IsAccessFailureDetected"},
|
|
|
|
{710, nullptr, "ResolveAccessFailure"},
|
|
|
|
{720, nullptr, "AbandonAccessFailure"},
|
|
|
|
{800, nullptr, "GetAndClearFileSystemProxyErrorInfo"},
|
|
|
|
{1000, nullptr, "SetBisRootForHost"},
|
|
|
|
{1001, nullptr, "SetSaveDataSize"},
|
|
|
|
{1002, nullptr, "SetSaveDataRootPath"},
|
|
|
|
{1003, nullptr, "DisableAutoSaveDataCreation"},
|
|
|
|
{1004, nullptr, "SetGlobalAccessLogMode"},
|
2018-01-17 04:20:12 +01:00
|
|
|
{1005, &FSP_SRV::GetGlobalAccessLogMode, "GetGlobalAccessLogMode"},
|
2018-04-10 18:30:27 +02:00
|
|
|
{1006, nullptr, "OutputAccessLogToSdCard"},
|
|
|
|
{1007, nullptr, "RegisterUpdatePartition"},
|
|
|
|
{1008, nullptr, "OpenRegisteredUpdatePartition"},
|
|
|
|
{1009, nullptr, "GetAndClearMemoryReportInfo"},
|
|
|
|
{1100, nullptr, "OverrideSaveDataTransferTokenSignVerificationKey"},
|
2018-01-17 04:20:12 +01:00
|
|
|
};
|
|
|
|
RegisterHandlers(functions);
|
|
|
|
}
|
|
|
|
|
2018-01-21 03:32:36 +01:00
|
|
|
void FSP_SRV::TryLoadRomFS() {
|
|
|
|
if (romfs) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
FileSys::Path unused;
|
|
|
|
auto res = OpenFileSystem(Type::RomFS, unused);
|
|
|
|
if (res.Succeeded()) {
|
|
|
|
romfs = std::move(res.Unwrap());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-07 15:20:39 +02:00
|
|
|
void FSP_SRV::Initialize(Kernel::HLERequestContext& ctx) {
|
2018-01-21 03:32:36 +01:00
|
|
|
LOG_WARNING(Service_FS, "(STUBBED) called");
|
|
|
|
|
2018-01-24 01:52:18 +01:00
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
2018-01-17 04:20:12 +01:00
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
}
|
|
|
|
|
2018-02-10 05:30:28 +01:00
|
|
|
void FSP_SRV::MountSdCard(Kernel::HLERequestContext& ctx) {
|
2018-03-20 03:21:49 +01:00
|
|
|
LOG_DEBUG(Service_FS, "called");
|
2018-02-10 05:30:28 +01:00
|
|
|
|
2018-03-20 03:21:49 +01:00
|
|
|
FileSys::Path unused;
|
|
|
|
auto filesystem = OpenFileSystem(Type::SDMC, unused).Unwrap();
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2, 0, 1};
|
2018-02-10 05:30:28 +01:00
|
|
|
rb.Push(RESULT_SUCCESS);
|
2018-03-20 03:21:49 +01:00
|
|
|
rb.PushIpcInterface<IFileSystem>(std::move(filesystem));
|
2018-02-10 05:30:28 +01:00
|
|
|
}
|
|
|
|
|
2018-03-04 20:31:57 +01:00
|
|
|
void FSP_SRV::CreateSaveData(Kernel::HLERequestContext& ctx) {
|
|
|
|
IPC::RequestParser rp{ctx};
|
|
|
|
|
|
|
|
auto save_struct = rp.PopRaw<std::array<u8, 0x40>>();
|
|
|
|
auto save_create_struct = rp.PopRaw<std::array<u8, 0x40>>();
|
|
|
|
u128 uid = rp.PopRaw<u128>();
|
|
|
|
|
|
|
|
LOG_WARNING(Service_FS, "(STUBBED) called uid = %016" PRIX64 "%016" PRIX64, uid[1], uid[0]);
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
}
|
|
|
|
|
2018-02-19 06:33:48 +01:00
|
|
|
void FSP_SRV::MountSaveData(Kernel::HLERequestContext& ctx) {
|
|
|
|
LOG_WARNING(Service_FS, "(STUBBED) called");
|
|
|
|
|
|
|
|
FileSys::Path unused;
|
|
|
|
auto filesystem = OpenFileSystem(Type::SaveData, unused).Unwrap();
|
|
|
|
|
|
|
|
IPC::ResponseBuilder rb{ctx, 2, 0, 1};
|
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
rb.PushIpcInterface<IFileSystem>(std::move(filesystem));
|
|
|
|
}
|
|
|
|
|
2018-01-17 04:20:12 +01:00
|
|
|
void FSP_SRV::GetGlobalAccessLogMode(Kernel::HLERequestContext& ctx) {
|
2018-01-21 03:32:36 +01:00
|
|
|
LOG_WARNING(Service_FS, "(STUBBED) called");
|
|
|
|
|
2018-01-24 01:52:18 +01:00
|
|
|
IPC::ResponseBuilder rb{ctx, 3};
|
2018-01-17 04:20:12 +01:00
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
rb.Push<u32>(5);
|
|
|
|
}
|
|
|
|
|
|
|
|
void FSP_SRV::OpenDataStorageByCurrentProcess(Kernel::HLERequestContext& ctx) {
|
2018-01-21 03:32:36 +01:00
|
|
|
LOG_DEBUG(Service_FS, "called");
|
|
|
|
|
|
|
|
TryLoadRomFS();
|
|
|
|
if (!romfs) {
|
|
|
|
// TODO (bunnei): Find the right error code to use here
|
|
|
|
LOG_CRITICAL(Service_FS, "no file system interface available!");
|
2018-01-24 01:52:18 +01:00
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
2018-01-21 03:32:36 +01:00
|
|
|
rb.Push(ResultCode(-1));
|
2018-01-17 04:20:12 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-01-21 03:32:36 +01:00
|
|
|
// Attempt to open a StorageBackend interface to the RomFS
|
|
|
|
auto storage = romfs->OpenFile({}, {});
|
2018-01-17 04:20:12 +01:00
|
|
|
if (storage.Failed()) {
|
2018-01-21 03:32:36 +01:00
|
|
|
LOG_CRITICAL(Service_FS, "no storage interface available!");
|
2018-01-24 01:52:18 +01:00
|
|
|
IPC::ResponseBuilder rb{ctx, 2};
|
2018-01-17 04:20:12 +01:00
|
|
|
rb.Push(storage.Code());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-01-24 01:52:18 +01:00
|
|
|
IPC::ResponseBuilder rb{ctx, 2, 0, 1};
|
2018-01-17 04:20:12 +01:00
|
|
|
rb.Push(RESULT_SUCCESS);
|
|
|
|
rb.PushIpcInterface<IStorage>(std::move(storage.Unwrap()));
|
|
|
|
}
|
|
|
|
|
|
|
|
void FSP_SRV::OpenRomStorage(Kernel::HLERequestContext& ctx) {
|
2018-01-21 03:32:36 +01:00
|
|
|
LOG_WARNING(Service_FS, "(STUBBED) called, using OpenDataStorageByCurrentProcess");
|
2018-01-17 04:20:12 +01:00
|
|
|
OpenDataStorageByCurrentProcess(ctx);
|
|
|
|
}
|
|
|
|
|
2018-01-21 03:32:36 +01:00
|
|
|
} // namespace FileSystem
|
2018-01-17 04:20:12 +01:00
|
|
|
} // namespace Service
|