/* * Copyright (c) Atmosphère-NX * * This program is free software; you can redistribute it and/or modify it * under the terms and conditions of the GNU General Public License, * version 2, as published by the Free Software Foundation. * * This program is distributed in the hope it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for * more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #pragma once #include #include #include #include #include namespace ams::fs { namespace { class ReadOnlyFile : public fsa::IFile, public impl::Newable { NON_COPYABLE(ReadOnlyFile); NON_MOVEABLE(ReadOnlyFile); private: std::unique_ptr base_file; public: explicit ReadOnlyFile(std::unique_ptr &&f) : base_file(std::move(f)) { /* ... */ } virtual ~ReadOnlyFile() { /* ... */ } private: virtual Result DoRead(size_t *out, s64 offset, void *buffer, size_t size, const fs::ReadOption &option) override final { return this->base_file->Read(out, offset, buffer, size, option); } virtual Result DoGetSize(s64 *out) override final { return this->base_file->GetSize(out); } virtual Result DoFlush() override final { return ResultSuccess(); } virtual Result DoWrite(s64 offset, const void *buffer, size_t size, const fs::WriteOption &option) override final { AMS_UNUSED(offset, buffer, size, option); return fs::ResultUnsupportedOperationInReadOnlyFileA(); } virtual Result DoSetSize(s64 size) override final { AMS_UNUSED(size); return fs::ResultUnsupportedOperationInReadOnlyFileA(); } virtual Result DoOperateRange(void *dst, size_t dst_size, fs::OperationId op_id, s64 offset, s64 size, const void *src, size_t src_size) override final { switch (op_id) { case OperationId::Invalidate: case OperationId::QueryRange: return this->base_file->OperateRange(dst, dst_size, op_id, offset, size, src, src_size); default: return fs::ResultUnsupportedOperationInReadOnlyFileB(); } } public: virtual sf::cmif::DomainObjectId GetDomainObjectId() const override { return this->base_file->GetDomainObjectId(); } }; } template class ReadOnlyFileSystemTemplate : public fsa::IFileSystem, public impl::Newable { NON_COPYABLE(ReadOnlyFileSystemTemplate); NON_MOVEABLE(ReadOnlyFileSystemTemplate); private: T base_fs; public: explicit ReadOnlyFileSystemTemplate(T &&fs) : base_fs(std::move(fs)) { /* ... */ } virtual ~ReadOnlyFileSystemTemplate() { /* ... */ } private: virtual Result DoOpenFile(std::unique_ptr *out_file, const char *path, OpenMode mode) override final { /* Only allow opening files with mode = read. */ R_UNLESS((mode & fs::OpenMode_All) == fs::OpenMode_Read, fs::ResultInvalidOpenMode()); std::unique_ptr base_file; R_TRY(this->base_fs->OpenFile(std::addressof(base_file), path, mode)); auto read_only_file = std::make_unique(std::move(base_file)); R_UNLESS(read_only_file != nullptr, fs::ResultAllocationFailureInReadOnlyFileSystemA()); *out_file = std::move(read_only_file); return ResultSuccess(); } virtual Result DoOpenDirectory(std::unique_ptr *out_dir, const char *path, OpenDirectoryMode mode) override final { return this->base_fs->OpenDirectory(out_dir, path, mode); } virtual Result DoGetEntryType(DirectoryEntryType *out, const char *path) override final { return this->base_fs->GetEntryType(out, path); } virtual Result DoCommit() override final { return ResultSuccess(); } virtual Result DoCreateFile(const char *path, s64 size, int flags) override final { AMS_UNUSED(path, size, flags); return fs::ResultUnsupportedOperationInReadOnlyFileSystemTemplateA(); } virtual Result DoDeleteFile(const char *path) override final { AMS_UNUSED(path); return fs::ResultUnsupportedOperationInReadOnlyFileSystemTemplateA(); } virtual Result DoCreateDirectory(const char *path) override final { AMS_UNUSED(path); return fs::ResultUnsupportedOperationInReadOnlyFileSystemTemplateA(); } virtual Result DoDeleteDirectory(const char *path) override final { AMS_UNUSED(path); return fs::ResultUnsupportedOperationInReadOnlyFileSystemTemplateA(); } virtual Result DoDeleteDirectoryRecursively(const char *path) override final { AMS_UNUSED(path); return fs::ResultUnsupportedOperationInReadOnlyFileSystemTemplateA(); } virtual Result DoRenameFile(const char *old_path, const char *new_path) override final { AMS_UNUSED(old_path, new_path); return fs::ResultUnsupportedOperationInReadOnlyFileSystemTemplateA(); } virtual Result DoRenameDirectory(const char *old_path, const char *new_path) override final { AMS_UNUSED(old_path, new_path); return fs::ResultUnsupportedOperationInReadOnlyFileSystemTemplateA(); } virtual Result DoCleanDirectoryRecursively(const char *path) override final { AMS_UNUSED(path); return fs::ResultUnsupportedOperationInReadOnlyFileSystemTemplateA(); } virtual Result DoGetFreeSpaceSize(s64 *out, const char *path) override final { AMS_UNUSED(out, path); return fs::ResultUnsupportedOperationInReadOnlyFileSystemTemplateB(); } virtual Result DoGetTotalSpaceSize(s64 *out, const char *path) override final { AMS_UNUSED(out, path); return fs::ResultUnsupportedOperationInReadOnlyFileSystemTemplateB(); } virtual Result DoCommitProvisionally(s64 counter) override final { AMS_UNUSED(counter); return fs::ResultUnsupportedOperationInReadOnlyFileSystemTemplateC(); } }; using ReadOnlyFileSystem = ReadOnlyFileSystemTemplate>; using ReadOnlyFileSystemShared = ReadOnlyFileSystemTemplate>; }