// Copyright 2014 Citra Emulator Project // Licensed under GPLv2 or any later version // Refer to the license.txt file included. #pragma once #include #include #include #include #include #include #include "common/common_types.h" #include "common/file_util.h" #include "core/file_sys/archive_backend.h" #include "core/file_sys/directory_backend.h" #include "core/file_sys/file_backend.h" #include "core/file_sys/romfs_reader.h" #include "core/hle/result.h" //////////////////////////////////////////////////////////////////////////////////////////////////// // FileSys namespace namespace FileSys { class IVFCDelayGenerator : public DelayGenerator { u64 GetReadDelayNs(std::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(static_cast(length) * slope + offset, minimum); return IPCDelayNanoseconds; } u64 GetOpenDelayNs() override { // This is the delay measured for a romfs open. // For now we will take that as a default static constexpr u64 IPCDelayNanoseconds(9438006); return IPCDelayNanoseconds; } SERIALIZE_DELAY_GENERATOR }; class RomFSDelayGenerator : public DelayGenerator { public: u64 GetReadDelayNs(std::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(static_cast(length) * slope + offset, minimum); return IPCDelayNanoseconds; } u64 GetOpenDelayNs() override { // This is the delay measured on O3DS and O2DS with // https://gist.github.com/FearlessTobi/eb1d70619c65c7e6f02141d71e79a36e // from the results the average of each length was taken. static constexpr u64 IPCDelayNanoseconds(9438006); return IPCDelayNanoseconds; } SERIALIZE_DELAY_GENERATOR }; class ExeFSDelayGenerator : public DelayGenerator { public: u64 GetReadDelayNs(std::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(static_cast(length) * slope + offset, minimum); return IPCDelayNanoseconds; } u64 GetOpenDelayNs() override { // This is the delay measured on O3DS and O2DS with // https://gist.github.com/FearlessTobi/eb1d70619c65c7e6f02141d71e79a36e // from the results the average of each length was taken. static constexpr u64 IPCDelayNanoseconds(9438006); return IPCDelayNanoseconds; } SERIALIZE_DELAY_GENERATOR }; /** * 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 * input data (load the raw IVFC archive) and override any required methods */ class IVFCArchive : public ArchiveBackend { public: IVFCArchive(std::shared_ptr file, std::unique_ptr delay_generator_); std::string GetName() const override; ResultVal> OpenFile(const Path& path, const Mode& mode) const override; ResultCode DeleteFile(const Path& path) const override; ResultCode RenameFile(const Path& src_path, const Path& dest_path) const override; ResultCode DeleteDirectory(const Path& path) const override; ResultCode DeleteDirectoryRecursively(const Path& path) const override; ResultCode CreateFile(const Path& path, u64 size) const override; ResultCode CreateDirectory(const Path& path) const override; ResultCode RenameDirectory(const Path& src_path, const Path& dest_path) const override; ResultVal> OpenDirectory(const Path& path) const override; u64 GetFreeBytes() const override; protected: std::shared_ptr romfs_file; }; class IVFCFile : public FileBackend { public: IVFCFile(std::shared_ptr file, std::unique_ptr delay_generator_); ResultVal Read(u64 offset, std::size_t length, u8* buffer) const override; ResultVal Write(u64 offset, std::size_t length, bool flush, const u8* buffer) override; u64 GetSize() const override; bool SetSize(u64 size) const override; bool Close() const override { return false; } void Flush() const override {} private: std::shared_ptr romfs_file; IVFCFile() = default; template void serialize(Archive& ar, const unsigned int) { ar& boost::serialization::base_object(*this); ar& romfs_file; } friend class boost::serialization::access; }; class IVFCDirectory : public DirectoryBackend { public: u32 Read(const u32 count, Entry* entries) override { return 0; } bool Close() const override { return false; } }; class IVFCFileInMemory : public FileBackend { public: IVFCFileInMemory(std::vector bytes, u64 offset, u64 size, std::unique_ptr delay_generator_); ResultVal Read(u64 offset, std::size_t length, u8* buffer) const override; ResultVal Write(u64 offset, std::size_t length, bool flush, const u8* buffer) override; u64 GetSize() const override; bool SetSize(u64 size) const override; bool Close() const override { return false; } void Flush() const override {} private: std::vector romfs_file; u64 data_offset; u64 data_size; IVFCFileInMemory() = default; template void serialize(Archive& ar, const unsigned int) { ar& boost::serialization::base_object(*this); ar& romfs_file; ar& data_offset; ar& data_size; } friend class boost::serialization::access; }; } // namespace FileSys BOOST_CLASS_EXPORT_KEY(FileSys::IVFCFile) BOOST_CLASS_EXPORT_KEY(FileSys::IVFCFileInMemory) BOOST_CLASS_EXPORT_KEY(FileSys::IVFCDelayGenerator) BOOST_CLASS_EXPORT_KEY(FileSys::RomFSDelayGenerator) BOOST_CLASS_EXPORT_KEY(FileSys::ExeFSDelayGenerator)