2016-10-18 09:27:51 +00:00
|
|
|
// Copyright 2016 Citra Emulator Project
|
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
|
|
|
#include <memory>
|
2020-01-02 23:34:26 +00:00
|
|
|
#include "common/archives.h"
|
2016-10-18 09:27:51 +00:00
|
|
|
#include "common/file_util.h"
|
|
|
|
#include "core/file_sys/archive_sdmcwriteonly.h"
|
|
|
|
#include "core/file_sys/directory_backend.h"
|
|
|
|
#include "core/file_sys/errors.h"
|
|
|
|
#include "core/file_sys/file_backend.h"
|
|
|
|
#include "core/settings.h"
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
// FileSys namespace
|
|
|
|
|
2020-01-02 23:34:26 +00:00
|
|
|
SERIALIZE_EXPORT_IMPL(FileSys::SDMCWriteOnlyArchive)
|
|
|
|
SERIALIZE_EXPORT_IMPL(FileSys::ArchiveFactory_SDMCWriteOnly)
|
|
|
|
|
2016-10-18 09:27:51 +00:00
|
|
|
namespace FileSys {
|
|
|
|
|
2018-11-02 17:11:22 +00:00
|
|
|
class SDMCWriteOnlyDelayGenerator : public DelayGenerator {
|
|
|
|
public:
|
|
|
|
u64 GetReadDelayNs(std::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;
|
|
|
|
}
|
|
|
|
|
|
|
|
u64 GetOpenDelayNs() override {
|
|
|
|
// This is the delay measured on O3DS and O2DS with
|
|
|
|
// https://gist.github.com/FearlessTobi/c37e143c314789251f98f2c45cd706d2
|
|
|
|
// from the results the average of each length was taken.
|
|
|
|
static constexpr u64 IPCDelayNanoseconds(269082);
|
|
|
|
return IPCDelayNanoseconds;
|
|
|
|
}
|
2020-01-08 22:13:56 +00:00
|
|
|
|
|
|
|
SERIALIZE_DELAY_GENERATOR
|
2018-11-02 17:11:22 +00:00
|
|
|
};
|
|
|
|
|
2016-10-18 09:27:51 +00:00
|
|
|
ResultVal<std::unique_ptr<FileBackend>> SDMCWriteOnlyArchive::OpenFile(const Path& path,
|
|
|
|
const Mode& mode) const {
|
|
|
|
if (mode.read_flag) {
|
2018-06-29 11:18:07 +00:00
|
|
|
LOG_ERROR(Service_FS, "Read flag is not supported");
|
2016-10-18 09:27:51 +00:00
|
|
|
return ERROR_INVALID_READ_FLAG;
|
|
|
|
}
|
2016-10-20 03:45:01 +00:00
|
|
|
return SDMCArchive::OpenFileBase(path, mode);
|
2016-10-18 09:27:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ResultVal<std::unique_ptr<DirectoryBackend>> SDMCWriteOnlyArchive::OpenDirectory(
|
|
|
|
const Path& path) const {
|
2018-06-29 11:18:07 +00:00
|
|
|
LOG_ERROR(Service_FS, "Not supported");
|
2016-10-18 09:27:51 +00:00
|
|
|
return ERROR_UNSUPPORTED_OPEN_FLAGS;
|
|
|
|
}
|
|
|
|
|
|
|
|
ArchiveFactory_SDMCWriteOnly::ArchiveFactory_SDMCWriteOnly(const std::string& mount_point)
|
|
|
|
: sdmc_directory(mount_point) {
|
2018-06-29 11:18:07 +00:00
|
|
|
LOG_DEBUG(Service_FS, "Directory {} set as SDMCWriteOnly.", sdmc_directory);
|
2016-10-18 09:27:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool ArchiveFactory_SDMCWriteOnly::Initialize() {
|
|
|
|
if (!Settings::values.use_virtual_sd) {
|
2018-06-29 11:18:07 +00:00
|
|
|
LOG_WARNING(Service_FS, "SDMC disabled by config.");
|
2016-10-18 09:27:51 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!FileUtil::CreateFullPath(sdmc_directory)) {
|
2018-06-29 11:18:07 +00:00
|
|
|
LOG_ERROR(Service_FS, "Unable to create SDMC path.");
|
2016-10-18 09:27:51 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-02-03 01:14:01 +00:00
|
|
|
ResultVal<std::unique_ptr<ArchiveBackend>> ArchiveFactory_SDMCWriteOnly::Open(const Path& path,
|
|
|
|
u64 program_id) {
|
2018-11-02 17:11:22 +00:00
|
|
|
std::unique_ptr<DelayGenerator> delay_generator =
|
|
|
|
std::make_unique<SDMCWriteOnlyDelayGenerator>();
|
|
|
|
auto archive =
|
|
|
|
std::make_unique<SDMCWriteOnlyArchive>(sdmc_directory, std::move(delay_generator));
|
2016-10-18 09:27:51 +00:00
|
|
|
return MakeResult<std::unique_ptr<ArchiveBackend>>(std::move(archive));
|
|
|
|
}
|
|
|
|
|
|
|
|
ResultCode ArchiveFactory_SDMCWriteOnly::Format(const Path& path,
|
2019-02-03 01:14:01 +00:00
|
|
|
const FileSys::ArchiveFormatInfo& format_info,
|
|
|
|
u64 program_id) {
|
2016-10-18 09:27:51 +00:00
|
|
|
// TODO(wwylele): hwtest this
|
2018-06-29 11:18:07 +00:00
|
|
|
LOG_ERROR(Service_FS, "Attempted to format a SDMC write-only archive.");
|
2016-10-18 09:27:51 +00:00
|
|
|
return ResultCode(-1);
|
|
|
|
}
|
|
|
|
|
2019-02-03 01:14:01 +00:00
|
|
|
ResultVal<ArchiveFormatInfo> ArchiveFactory_SDMCWriteOnly::GetFormatInfo(const Path& path,
|
|
|
|
u64 program_id) const {
|
2016-10-18 09:27:51 +00:00
|
|
|
// TODO(Subv): Implement
|
2018-06-29 11:18:07 +00:00
|
|
|
LOG_ERROR(Service_FS, "Unimplemented GetFormatInfo archive {}", GetName());
|
2016-10-18 09:27:51 +00:00
|
|
|
return ResultCode(-1);
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace FileSys
|
2020-01-08 22:13:56 +00:00
|
|
|
|
|
|
|
SERIALIZE_EXPORT_IMPL(FileSys::SDMCWriteOnlyDelayGenerator)
|