mirror of
https://github.com/PabloMK7/citra.git
synced 2025-01-19 10:23:06 +01:00
Add DelayGenerator for all file backends
This commit is contained in:
parent
58b16c5459
commit
06a7676ed1
13 changed files with 172 additions and 71 deletions
|
@ -57,6 +57,7 @@ add_library(core STATIC
|
||||||
file_sys/disk_archive.h
|
file_sys/disk_archive.h
|
||||||
file_sys/errors.h
|
file_sys/errors.h
|
||||||
file_sys/file_backend.h
|
file_sys/file_backend.h
|
||||||
|
file_sys/delay_generator.h
|
||||||
file_sys/ivfc_archive.cpp
|
file_sys/ivfc_archive.cpp
|
||||||
file_sys/ivfc_archive.h
|
file_sys/ivfc_archive.h
|
||||||
file_sys/ncch_container.cpp
|
file_sys/ncch_container.cpp
|
||||||
|
|
|
@ -27,7 +27,9 @@ namespace FileSys {
|
||||||
*/
|
*/
|
||||||
class FixSizeDiskFile : public DiskFile {
|
class FixSizeDiskFile : public DiskFile {
|
||||||
public:
|
public:
|
||||||
FixSizeDiskFile(FileUtil::IOFile&& file, const Mode& mode) : DiskFile(std::move(file), mode) {
|
FixSizeDiskFile(FileUtil::IOFile&& file, const Mode& mode,
|
||||||
|
std::unique_ptr<DelayGenerator> delay_generator_)
|
||||||
|
: DiskFile(std::move(file), mode, std::move(delay_generator_)) {
|
||||||
size = GetSize();
|
size = GetSize();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -53,6 +55,20 @@ private:
|
||||||
u64 size{};
|
u64 size{};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class ExtSaveDataDelayGenerator : public DelayGenerator {
|
||||||
|
public:
|
||||||
|
u64 GetReadDelayNs(size_t length) override {
|
||||||
|
// This is the delay measured for a savedate read,
|
||||||
|
// not for extsaveData
|
||||||
|
// For now we will take that
|
||||||
|
static constexpr u64 slope(183);
|
||||||
|
static constexpr u64 offset(524879);
|
||||||
|
static constexpr u64 minimum(631826);
|
||||||
|
u64 IPCDelayNanoseconds = std::max<u64>(static_cast<u64>(length) * slope + offset, minimum);
|
||||||
|
return IPCDelayNanoseconds;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Archive backend for general extsave data archive type.
|
* Archive backend for general extsave data archive type.
|
||||||
* The behaviour of ExtSaveDataArchive is almost the same as SaveDataArchive, except for
|
* The behaviour of ExtSaveDataArchive is almost the same as SaveDataArchive, except for
|
||||||
|
@ -118,7 +134,10 @@ public:
|
||||||
Mode rwmode;
|
Mode rwmode;
|
||||||
rwmode.write_flag.Assign(1);
|
rwmode.write_flag.Assign(1);
|
||||||
rwmode.read_flag.Assign(1);
|
rwmode.read_flag.Assign(1);
|
||||||
auto disk_file = std::make_unique<FixSizeDiskFile>(std::move(file), rwmode);
|
std::unique_ptr<DelayGenerator> delay_generator =
|
||||||
|
std::make_unique<ExtSaveDataDelayGenerator>();
|
||||||
|
auto disk_file =
|
||||||
|
std::make_unique<FixSizeDiskFile>(std::move(file), rwmode, std::move(delay_generator));
|
||||||
return MakeResult<std::unique_ptr<FileBackend>>(std::move(disk_file));
|
return MakeResult<std::unique_ptr<FileBackend>>(std::move(disk_file));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -77,14 +77,17 @@ ResultVal<std::unique_ptr<FileBackend>> NCCHArchive::OpenFile(const Path& path,
|
||||||
u64 romfs_size = 0;
|
u64 romfs_size = 0;
|
||||||
|
|
||||||
result = ncch_container.ReadRomFS(romfs_file, romfs_offset, romfs_size);
|
result = ncch_container.ReadRomFS(romfs_file, romfs_offset, romfs_size);
|
||||||
file = std::make_unique<IVFCFile>(std::move(romfs_file), romfs_offset, romfs_size);
|
std::unique_ptr<DelayGenerator> delay_generator = std::make_unique<RomFSDelayGenerator>();
|
||||||
|
file = std::make_unique<IVFCFile>(std::move(romfs_file), romfs_offset, romfs_size,
|
||||||
|
std::move(delay_generator));
|
||||||
} else if (filepath_type == NCCHFilePathType::Code ||
|
} else if (filepath_type == NCCHFilePathType::Code ||
|
||||||
filepath_type == NCCHFilePathType::ExeFS) {
|
filepath_type == NCCHFilePathType::ExeFS) {
|
||||||
std::vector<u8> buffer;
|
std::vector<u8> buffer;
|
||||||
|
|
||||||
// Load NCCH .code or icon/banner/logo
|
// Load NCCH .code or icon/banner/logo
|
||||||
result = ncch_container.LoadSectionExeFS(openfile_path.exefs_filepath.data(), buffer);
|
result = ncch_container.LoadSectionExeFS(openfile_path.exefs_filepath.data(), buffer);
|
||||||
file = std::make_unique<NCCHFile>(std::move(buffer));
|
std::unique_ptr<DelayGenerator> delay_generator = std::make_unique<ExeFSDelayGenerator>();
|
||||||
|
file = std::make_unique<NCCHFile>(std::move(buffer), std::move(delay_generator));
|
||||||
} else {
|
} else {
|
||||||
LOG_ERROR(Service_FS, "Unknown NCCH archive type %u!", openfile_path.filepath_type);
|
LOG_ERROR(Service_FS, "Unknown NCCH archive type %u!", openfile_path.filepath_type);
|
||||||
result = Loader::ResultStatus::Error;
|
result = Loader::ResultStatus::Error;
|
||||||
|
@ -194,7 +197,10 @@ u64 NCCHArchive::GetFreeBytes() const {
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
NCCHFile::NCCHFile(std::vector<u8> buffer) : file_buffer(std::move(buffer)) {}
|
NCCHFile::NCCHFile(std::vector<u8> buffer, std::unique_ptr<DelayGenerator> delay_generator_)
|
||||||
|
: file_buffer(std::move(buffer)) {
|
||||||
|
delay_generator = std::move(delay_generator_);
|
||||||
|
}
|
||||||
|
|
||||||
ResultVal<size_t> NCCHFile::Read(const u64 offset, const size_t length, u8* buffer) const {
|
ResultVal<size_t> NCCHFile::Read(const u64 offset, const size_t length, u8* buffer) const {
|
||||||
LOG_TRACE(Service_FS, "called offset=%" PRIu64 ", length=%zu", offset, length);
|
LOG_TRACE(Service_FS, "called offset=%" PRIu64 ", length=%zu", offset, length);
|
||||||
|
|
|
@ -17,7 +17,7 @@ namespace Service {
|
||||||
namespace FS {
|
namespace FS {
|
||||||
enum class MediaType : u32;
|
enum class MediaType : u32;
|
||||||
}
|
}
|
||||||
}
|
} // namespace Service
|
||||||
|
|
||||||
namespace FileSys {
|
namespace FileSys {
|
||||||
|
|
||||||
|
@ -51,7 +51,7 @@ protected:
|
||||||
// File backend for NCCH files
|
// File backend for NCCH files
|
||||||
class NCCHFile : public FileBackend {
|
class NCCHFile : public FileBackend {
|
||||||
public:
|
public:
|
||||||
explicit NCCHFile(std::vector<u8> buffer);
|
explicit NCCHFile(std::vector<u8> buffer, std::unique_ptr<DelayGenerator> delay_generator_);
|
||||||
|
|
||||||
ResultVal<size_t> Read(u64 offset, size_t length, u8* buffer) const override;
|
ResultVal<size_t> Read(u64 offset, size_t length, u8* buffer) const override;
|
||||||
ResultVal<size_t> Write(u64 offset, size_t length, bool flush, const u8* buffer) override;
|
ResultVal<size_t> Write(u64 offset, size_t length, bool flush, const u8* buffer) override;
|
||||||
|
|
|
@ -17,6 +17,20 @@
|
||||||
|
|
||||||
namespace FileSys {
|
namespace FileSys {
|
||||||
|
|
||||||
|
class SDMCDelayGenerator : public DelayGenerator {
|
||||||
|
public:
|
||||||
|
u64 GetReadDelayNs(size_t length) override {
|
||||||
|
// This is the delay measured on O3DS and O2DS with
|
||||||
|
// https://gist.github.com/B3n30/ac40eac20603f519ff106107f4ac9182
|
||||||
|
// from the results the average of each length was taken.
|
||||||
|
static constexpr u64 slope(183);
|
||||||
|
static constexpr u64 offset(524879);
|
||||||
|
static constexpr u64 minimum(631826);
|
||||||
|
u64 IPCDelayNanoseconds = std::max<u64>(static_cast<u64>(length) * slope + offset, minimum);
|
||||||
|
return IPCDelayNanoseconds;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
ResultVal<std::unique_ptr<FileBackend>> SDMCArchive::OpenFile(const Path& path,
|
ResultVal<std::unique_ptr<FileBackend>> SDMCArchive::OpenFile(const Path& path,
|
||||||
const Mode& mode) const {
|
const Mode& mode) const {
|
||||||
Mode modified_mode;
|
Mode modified_mode;
|
||||||
|
@ -82,7 +96,8 @@ ResultVal<std::unique_ptr<FileBackend>> SDMCArchive::OpenFileBase(const Path& pa
|
||||||
return ERROR_NOT_FOUND;
|
return ERROR_NOT_FOUND;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto disk_file = std::make_unique<DiskFile>(std::move(file), mode);
|
std::unique_ptr<DelayGenerator> delay_generator = std::make_unique<SDMCDelayGenerator>();
|
||||||
|
auto disk_file = std::make_unique<DiskFile>(std::move(file), mode, std::move(delay_generator));
|
||||||
return MakeResult<std::unique_ptr<FileBackend>>(std::move(disk_file));
|
return MakeResult<std::unique_ptr<FileBackend>>(std::move(disk_file));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -343,6 +358,7 @@ u64 SDMCArchive::GetFreeBytes() const {
|
||||||
|
|
||||||
ArchiveFactory_SDMC::ArchiveFactory_SDMC(const std::string& sdmc_directory)
|
ArchiveFactory_SDMC::ArchiveFactory_SDMC(const std::string& sdmc_directory)
|
||||||
: sdmc_directory(sdmc_directory) {
|
: sdmc_directory(sdmc_directory) {
|
||||||
|
|
||||||
LOG_DEBUG(Service_FS, "Directory %s set as SDMC.", sdmc_directory.c_str());
|
LOG_DEBUG(Service_FS, "Directory %s set as SDMC.", sdmc_directory.c_str());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -56,17 +56,6 @@ public:
|
||||||
return ERROR_UNSUPPORTED_OPEN_FLAGS;
|
return ERROR_UNSUPPORTED_OPEN_FLAGS;
|
||||||
}
|
}
|
||||||
|
|
||||||
u64 GetReadDelayNs(size_t length) const {
|
|
||||||
// The delay was measured on O3DS and O2DS with
|
|
||||||
// https://gist.github.com/B3n30/ac40eac20603f519ff106107f4ac9182
|
|
||||||
// from the results the average of each length was taken.
|
|
||||||
static constexpr u64 slope(94);
|
|
||||||
static constexpr u64 offset(582778);
|
|
||||||
static constexpr u64 minimum(663124);
|
|
||||||
u64 IPCDelayNanoseconds = std::max<u64>(static_cast<u64>(length) * slope + offset, minimum);
|
|
||||||
return IPCDelayNanoseconds;
|
|
||||||
}
|
|
||||||
|
|
||||||
u64 GetSize() const override {
|
u64 GetSize() const override {
|
||||||
return data->size();
|
return data->size();
|
||||||
}
|
}
|
||||||
|
@ -182,8 +171,11 @@ public:
|
||||||
private:
|
private:
|
||||||
ResultVal<std::unique_ptr<FileBackend>> OpenRomFS() const {
|
ResultVal<std::unique_ptr<FileBackend>> OpenRomFS() const {
|
||||||
if (ncch_data.romfs_file) {
|
if (ncch_data.romfs_file) {
|
||||||
return MakeResult<std::unique_ptr<FileBackend>>(std::make_unique<IVFCFile>(
|
std::unique_ptr<DelayGenerator> delay_generator =
|
||||||
ncch_data.romfs_file, ncch_data.romfs_offset, ncch_data.romfs_size));
|
std::make_unique<RomFSDelayGenerator>();
|
||||||
|
return MakeResult<std::unique_ptr<FileBackend>>(
|
||||||
|
std::make_unique<IVFCFile>(ncch_data.romfs_file, ncch_data.romfs_offset,
|
||||||
|
ncch_data.romfs_size, std::move(delay_generator)));
|
||||||
} else {
|
} else {
|
||||||
LOG_INFO(Service_FS, "Unable to read RomFS");
|
LOG_INFO(Service_FS, "Unable to read RomFS");
|
||||||
return ERROR_ROMFS_NOT_FOUND;
|
return ERROR_ROMFS_NOT_FOUND;
|
||||||
|
@ -192,9 +184,11 @@ private:
|
||||||
|
|
||||||
ResultVal<std::unique_ptr<FileBackend>> OpenUpdateRomFS() const {
|
ResultVal<std::unique_ptr<FileBackend>> OpenUpdateRomFS() const {
|
||||||
if (ncch_data.update_romfs_file) {
|
if (ncch_data.update_romfs_file) {
|
||||||
|
std::unique_ptr<DelayGenerator> delay_generator =
|
||||||
|
std::make_unique<RomFSDelayGenerator>();
|
||||||
return MakeResult<std::unique_ptr<FileBackend>>(std::make_unique<IVFCFile>(
|
return MakeResult<std::unique_ptr<FileBackend>>(std::make_unique<IVFCFile>(
|
||||||
ncch_data.update_romfs_file, ncch_data.update_romfs_offset,
|
ncch_data.update_romfs_file, ncch_data.update_romfs_offset,
|
||||||
ncch_data.update_romfs_size));
|
ncch_data.update_romfs_size, std::move(delay_generator)));
|
||||||
} else {
|
} else {
|
||||||
LOG_INFO(Service_FS, "Unable to read update RomFS");
|
LOG_INFO(Service_FS, "Unable to read update RomFS");
|
||||||
return ERROR_ROMFS_NOT_FOUND;
|
return ERROR_ROMFS_NOT_FOUND;
|
||||||
|
@ -251,7 +245,8 @@ void ArchiveFactory_SelfNCCH::Register(Loader::AppLoader& app_loader) {
|
||||||
program_id);
|
program_id);
|
||||||
|
|
||||||
if (ncch_data.find(program_id) != ncch_data.end()) {
|
if (ncch_data.find(program_id) != ncch_data.end()) {
|
||||||
LOG_WARNING(Service_FS, "Registering program %016" PRIX64
|
LOG_WARNING(Service_FS,
|
||||||
|
"Registering program %016" PRIX64
|
||||||
" with SelfNCCH will override existing mapping",
|
" with SelfNCCH will override existing mapping",
|
||||||
program_id);
|
program_id);
|
||||||
}
|
}
|
||||||
|
@ -266,8 +261,8 @@ void ArchiveFactory_SelfNCCH::Register(Loader::AppLoader& app_loader) {
|
||||||
}
|
}
|
||||||
|
|
||||||
std::shared_ptr<FileUtil::IOFile> update_romfs_file;
|
std::shared_ptr<FileUtil::IOFile> update_romfs_file;
|
||||||
if (Loader::ResultStatus::Success ==
|
if (Loader::ResultStatus::Success == app_loader.ReadUpdateRomFS(update_romfs_file,
|
||||||
app_loader.ReadUpdateRomFS(update_romfs_file, data.update_romfs_offset,
|
data.update_romfs_offset,
|
||||||
data.update_romfs_size)) {
|
data.update_romfs_size)) {
|
||||||
|
|
||||||
data.update_romfs_file = std::move(update_romfs_file);
|
data.update_romfs_file = std::move(update_romfs_file);
|
||||||
|
|
29
src/core/file_sys/delay_generator.h
Normal file
29
src/core/file_sys/delay_generator.h
Normal file
|
@ -0,0 +1,29 @@
|
||||||
|
// Copyright 2018 Citra Emulator Project
|
||||||
|
// Licensed under GPLv2 or any later version
|
||||||
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
namespace FileSys {
|
||||||
|
|
||||||
|
class DelayGenerator {
|
||||||
|
public:
|
||||||
|
virtual u64 GetReadDelayNs(size_t length) = 0;
|
||||||
|
|
||||||
|
// TODO (B3N30): Add getter for all other file/directory io operations
|
||||||
|
};
|
||||||
|
|
||||||
|
class DefaultDelayGenerator : public DelayGenerator {
|
||||||
|
public:
|
||||||
|
u64 GetReadDelayNs(size_t length) override {
|
||||||
|
// This is the delay measured for a romfs read.
|
||||||
|
// For now we will take that as a default
|
||||||
|
static constexpr u64 slope(94);
|
||||||
|
static constexpr u64 offset(582778);
|
||||||
|
static constexpr u64 minimum(663124);
|
||||||
|
u64 IPCDelayNanoseconds = std::max<u64>(static_cast<u64>(length) * slope + offset, minimum);
|
||||||
|
return IPCDelayNanoseconds;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace FileSys
|
|
@ -36,20 +36,6 @@ ResultVal<size_t> DiskFile::Write(const u64 offset, const size_t length, const b
|
||||||
return MakeResult<size_t>(written);
|
return MakeResult<size_t>(written);
|
||||||
}
|
}
|
||||||
|
|
||||||
u64 DiskFile::GetReadDelayNs(size_t length) const {
|
|
||||||
// TODO(B3N30): figure out the time a 3ds needs for those write
|
|
||||||
// for that backend.
|
|
||||||
// For now take the results from the romfs test.
|
|
||||||
// The delay was measured on O3DS and O2DS with
|
|
||||||
// https://gist.github.com/B3n30/ac40eac20603f519ff106107f4ac9182
|
|
||||||
// from the results the average of each length was taken.
|
|
||||||
static constexpr u64 slope(183);
|
|
||||||
static constexpr u64 offset(524879);
|
|
||||||
static constexpr u64 minimum(631826);
|
|
||||||
u64 IPCDelayNanoseconds = std::max<u64>(static_cast<u64>(length) * slope + offset, minimum);
|
|
||||||
return IPCDelayNanoseconds;
|
|
||||||
}
|
|
||||||
|
|
||||||
u64 DiskFile::GetSize() const {
|
u64 DiskFile::GetSize() const {
|
||||||
return file->GetSize();
|
return file->GetSize();
|
||||||
}
|
}
|
||||||
|
|
|
@ -22,14 +22,15 @@ namespace FileSys {
|
||||||
|
|
||||||
class DiskFile : public FileBackend {
|
class DiskFile : public FileBackend {
|
||||||
public:
|
public:
|
||||||
DiskFile(FileUtil::IOFile&& file_, const Mode& mode_)
|
DiskFile(FileUtil::IOFile&& file_, const Mode& mode_,
|
||||||
|
std::unique_ptr<DelayGenerator> delay_generator_)
|
||||||
: file(new FileUtil::IOFile(std::move(file_))) {
|
: file(new FileUtil::IOFile(std::move(file_))) {
|
||||||
|
delay_generator = std::move(delay_generator_);
|
||||||
mode.hex = mode_.hex;
|
mode.hex = mode_.hex;
|
||||||
}
|
}
|
||||||
|
|
||||||
ResultVal<size_t> Read(u64 offset, size_t length, u8* buffer) const override;
|
ResultVal<size_t> Read(u64 offset, size_t length, u8* buffer) const override;
|
||||||
ResultVal<size_t> Write(u64 offset, size_t length, bool flush, const u8* buffer) override;
|
ResultVal<size_t> Write(u64 offset, size_t length, bool flush, const u8* buffer) override;
|
||||||
u64 GetReadDelayNs(size_t length) const override;
|
|
||||||
u64 GetSize() const override;
|
u64 GetSize() const override;
|
||||||
bool SetSize(u64 size) const override;
|
bool SetSize(u64 size) const override;
|
||||||
bool Close() const override;
|
bool Close() const override;
|
||||||
|
|
|
@ -8,6 +8,7 @@
|
||||||
#include <cstddef>
|
#include <cstddef>
|
||||||
#include "common/common_types.h"
|
#include "common/common_types.h"
|
||||||
#include "core/hle/result.h"
|
#include "core/hle/result.h"
|
||||||
|
#include "delay_generator.h"
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
// FileSys namespace
|
// FileSys namespace
|
||||||
|
@ -43,17 +44,13 @@ public:
|
||||||
* @param length Length in bytes of data read from file
|
* @param length Length in bytes of data read from file
|
||||||
* @return Nanoseconds for the delay
|
* @return Nanoseconds for the delay
|
||||||
*/
|
*/
|
||||||
virtual u64 GetReadDelayNs(size_t length) const {
|
u64 GetReadDelayNs(size_t length) {
|
||||||
// Return the default delay for the case that the subclass backend didn't
|
if (delay_generator != nullptr) {
|
||||||
// implement one. We take the one measured for romfs reads
|
return delay_generator->GetReadDelayNs(length);
|
||||||
// This should be removed as soon as every subclass backend
|
}
|
||||||
// has one implemented
|
LOG_ERROR(Service_FS, "Delay generator was not initalized. Using default");
|
||||||
LOG_WARNING(Service_FS, "Using default delay for read");
|
delay_generator = std::make_unique<DefaultDelayGenerator>();
|
||||||
static constexpr u64 slope(94);
|
return delay_generator->GetReadDelayNs(length);
|
||||||
static constexpr u64 offset(582778);
|
|
||||||
static constexpr u64 minimum(663124);
|
|
||||||
u64 IPCDelayNanoseconds = std::max<u64>(static_cast<u64>(length) * slope + offset, minimum);
|
|
||||||
return IPCDelayNanoseconds;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -79,6 +76,9 @@ public:
|
||||||
* Flushes the file
|
* Flushes the file
|
||||||
*/
|
*/
|
||||||
virtual void Flush() const = 0;
|
virtual void Flush() const = 0;
|
||||||
|
|
||||||
|
protected:
|
||||||
|
std::unique_ptr<DelayGenerator> delay_generator;
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace FileSys
|
} // namespace FileSys
|
||||||
|
|
|
@ -23,8 +23,9 @@ std::string IVFCArchive::GetName() const {
|
||||||
|
|
||||||
ResultVal<std::unique_ptr<FileBackend>> IVFCArchive::OpenFile(const Path& path,
|
ResultVal<std::unique_ptr<FileBackend>> IVFCArchive::OpenFile(const Path& path,
|
||||||
const Mode& mode) const {
|
const Mode& mode) const {
|
||||||
|
std::unique_ptr<DelayGenerator> delay_generator = std::make_unique<IVFCDelayGenerator>();
|
||||||
return MakeResult<std::unique_ptr<FileBackend>>(
|
return MakeResult<std::unique_ptr<FileBackend>>(
|
||||||
std::make_unique<IVFCFile>(romfs_file, data_offset, data_size));
|
std::make_unique<IVFCFile>(romfs_file, data_offset, data_size, std::move(delay_generator)));
|
||||||
}
|
}
|
||||||
|
|
||||||
ResultCode IVFCArchive::DeleteFile(const Path& path) const {
|
ResultCode IVFCArchive::DeleteFile(const Path& path) const {
|
||||||
|
@ -89,8 +90,11 @@ u64 IVFCArchive::GetFreeBytes() const {
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
IVFCFile::IVFCFile(std::shared_ptr<FileUtil::IOFile> file, u64 offset, u64 size)
|
IVFCFile::IVFCFile(std::shared_ptr<FileUtil::IOFile> file, u64 offset, u64 size,
|
||||||
: romfs_file(std::move(file)), data_offset(offset), data_size(size) {}
|
std::unique_ptr<DelayGenerator> delay_generator_)
|
||||||
|
: romfs_file(std::move(file)), data_offset(offset), data_size(size) {
|
||||||
|
delay_generator = std::move(delay_generator_);
|
||||||
|
}
|
||||||
|
|
||||||
ResultVal<size_t> IVFCFile::Read(const u64 offset, const size_t length, u8* buffer) const {
|
ResultVal<size_t> IVFCFile::Read(const u64 offset, const size_t length, u8* buffer) const {
|
||||||
LOG_TRACE(Service_FS, "called offset=%llu, length=%zu", offset, length);
|
LOG_TRACE(Service_FS, "called offset=%llu, length=%zu", offset, length);
|
||||||
|
@ -107,17 +111,6 @@ ResultVal<size_t> IVFCFile::Write(const u64 offset, const size_t length, const b
|
||||||
return MakeResult<size_t>(0);
|
return MakeResult<size_t>(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
u64 IVFCFile::GetReadDelayNs(size_t length) const {
|
|
||||||
// The delay was measured on O3DS and O2DS with
|
|
||||||
// https://gist.github.com/B3n30/ac40eac20603f519ff106107f4ac9182
|
|
||||||
// from the results the average of each length was taken.
|
|
||||||
static constexpr u64 slope(94);
|
|
||||||
static constexpr u64 offset(582778);
|
|
||||||
static constexpr u64 minimum(663124);
|
|
||||||
u64 IPCDelayNanoseconds = std::max<u64>(static_cast<u64>(length) * slope + offset, minimum);
|
|
||||||
return IPCDelayNanoseconds;
|
|
||||||
}
|
|
||||||
|
|
||||||
u64 IVFCFile::GetSize() const {
|
u64 IVFCFile::GetSize() const {
|
||||||
return data_size;
|
return data_size;
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,6 +19,46 @@
|
||||||
|
|
||||||
namespace FileSys {
|
namespace FileSys {
|
||||||
|
|
||||||
|
class IVFCDelayGenerator : public DelayGenerator {
|
||||||
|
u64 GetReadDelayNs(size_t length) override {
|
||||||
|
// This is the delay measured for a romfs read.
|
||||||
|
// For now we will take that as a default
|
||||||
|
static constexpr u64 slope(94);
|
||||||
|
static constexpr u64 offset(582778);
|
||||||
|
static constexpr u64 minimum(663124);
|
||||||
|
u64 IPCDelayNanoseconds = std::max<u64>(static_cast<u64>(length) * slope + offset, minimum);
|
||||||
|
return IPCDelayNanoseconds;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
class RomFSDelayGenerator : public DelayGenerator {
|
||||||
|
public:
|
||||||
|
u64 GetReadDelayNs(size_t length) override {
|
||||||
|
// The delay was measured on O3DS and O2DS with
|
||||||
|
// https://gist.github.com/B3n30/ac40eac20603f519ff106107f4ac9182
|
||||||
|
// from the results the average of each length was taken.
|
||||||
|
static constexpr u64 slope(94);
|
||||||
|
static constexpr u64 offset(582778);
|
||||||
|
static constexpr u64 minimum(663124);
|
||||||
|
u64 IPCDelayNanoseconds = std::max<u64>(static_cast<u64>(length) * slope + offset, minimum);
|
||||||
|
return IPCDelayNanoseconds;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
class ExeFSDelayGenerator : public DelayGenerator {
|
||||||
|
public:
|
||||||
|
u64 GetReadDelayNs(size_t length) override {
|
||||||
|
// The delay was measured on O3DS and O2DS with
|
||||||
|
// https://gist.github.com/B3n30/ac40eac20603f519ff106107f4ac9182
|
||||||
|
// from the results the average of each length was taken.
|
||||||
|
static constexpr u64 slope(94);
|
||||||
|
static constexpr u64 offset(582778);
|
||||||
|
static constexpr u64 minimum(663124);
|
||||||
|
u64 IPCDelayNanoseconds = std::max<u64>(static_cast<u64>(length) * slope + offset, minimum);
|
||||||
|
return IPCDelayNanoseconds;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Helper which implements an interface to deal with IVFC images used in some archives
|
* Helper which implements an interface to deal with IVFC images used in some archives
|
||||||
* This should be subclassed by concrete archive types, which will provide the
|
* This should be subclassed by concrete archive types, which will provide the
|
||||||
|
@ -50,11 +90,11 @@ protected:
|
||||||
|
|
||||||
class IVFCFile : public FileBackend {
|
class IVFCFile : public FileBackend {
|
||||||
public:
|
public:
|
||||||
IVFCFile(std::shared_ptr<FileUtil::IOFile> file, u64 offset, u64 size);
|
IVFCFile(std::shared_ptr<FileUtil::IOFile> file, u64 offset, u64 size,
|
||||||
|
std::unique_ptr<DelayGenerator> delay_generator_);
|
||||||
|
|
||||||
ResultVal<size_t> Read(u64 offset, size_t length, u8* buffer) const override;
|
ResultVal<size_t> Read(u64 offset, size_t length, u8* buffer) const override;
|
||||||
ResultVal<size_t> Write(u64 offset, size_t length, bool flush, const u8* buffer) override;
|
ResultVal<size_t> Write(u64 offset, size_t length, bool flush, const u8* buffer) override;
|
||||||
u64 GetReadDelayNs(size_t length) const override;
|
|
||||||
u64 GetSize() const override;
|
u64 GetSize() const override;
|
||||||
bool SetSize(u64 size) const override;
|
bool SetSize(u64 size) const override;
|
||||||
bool Close() const override {
|
bool Close() const override {
|
||||||
|
|
|
@ -13,6 +13,20 @@
|
||||||
|
|
||||||
namespace FileSys {
|
namespace FileSys {
|
||||||
|
|
||||||
|
class SaveDataDelayGenerator : public DelayGenerator {
|
||||||
|
public:
|
||||||
|
u64 GetReadDelayNs(size_t length) override {
|
||||||
|
// The delay was measured on O3DS and O2DS with
|
||||||
|
// https://gist.github.com/B3n30/ac40eac20603f519ff106107f4ac9182
|
||||||
|
// from the results the average of each length was taken.
|
||||||
|
static constexpr u64 slope(183);
|
||||||
|
static constexpr u64 offset(524879);
|
||||||
|
static constexpr u64 minimum(631826);
|
||||||
|
u64 IPCDelayNanoseconds = std::max<u64>(static_cast<u64>(length) * slope + offset, minimum);
|
||||||
|
return IPCDelayNanoseconds;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
ResultVal<std::unique_ptr<FileBackend>> SaveDataArchive::OpenFile(const Path& path,
|
ResultVal<std::unique_ptr<FileBackend>> SaveDataArchive::OpenFile(const Path& path,
|
||||||
const Mode& mode) const {
|
const Mode& mode) const {
|
||||||
LOG_DEBUG(Service_FS, "called path=%s mode=%01X", path.DebugStr().c_str(), mode.hex);
|
LOG_DEBUG(Service_FS, "called path=%s mode=%01X", path.DebugStr().c_str(), mode.hex);
|
||||||
|
@ -67,7 +81,8 @@ ResultVal<std::unique_ptr<FileBackend>> SaveDataArchive::OpenFile(const Path& pa
|
||||||
return ERROR_FILE_NOT_FOUND;
|
return ERROR_FILE_NOT_FOUND;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto disk_file = std::make_unique<DiskFile>(std::move(file), mode);
|
std::unique_ptr<DelayGenerator> delay_generator = std::make_unique<SaveDataDelayGenerator>();
|
||||||
|
auto disk_file = std::make_unique<DiskFile>(std::move(file), mode, std::move(delay_generator));
|
||||||
return MakeResult<std::unique_ptr<FileBackend>>(std::move(disk_file));
|
return MakeResult<std::unique_ptr<FileBackend>>(std::move(disk_file));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue