/* * 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 . */ #include namespace ams::fs { Result FileStorage::UpdateSize() { R_SUCCEED_IF(m_size != InvalidSize); return m_base_file->GetSize(std::addressof(m_size)); } Result FileStorage::Read(s64 offset, void *buffer, size_t size) { /* Immediately succeed if there's nothing to read. */ R_SUCCEED_IF(size == 0); /* Validate buffer. */ R_UNLESS(buffer != nullptr, fs::ResultNullptrArgument()); /* Ensure our size is valid. */ R_TRY(this->UpdateSize()); /* Ensure our access is valid. */ R_UNLESS(IStorage::CheckAccessRange(offset, size, m_size), fs::ResultOutOfRange()); size_t read_size; return m_base_file->Read(std::addressof(read_size), offset, buffer, size); } Result FileStorage::Write(s64 offset, const void *buffer, size_t size) { /* Immediately succeed if there's nothing to write. */ R_SUCCEED_IF(size == 0); /* Validate buffer. */ R_UNLESS(buffer != nullptr, fs::ResultNullptrArgument()); /* Ensure our size is valid. */ R_TRY(this->UpdateSize()); /* Ensure our access is valid. */ R_UNLESS(IStorage::CheckAccessRange(offset, size, m_size), fs::ResultOutOfRange()); return m_base_file->Write(offset, buffer, size, fs::WriteOption()); } Result FileStorage::Flush() { return m_base_file->Flush(); } Result FileStorage::GetSize(s64 *out_size) { R_TRY(this->UpdateSize()); *out_size = m_size; return ResultSuccess(); } Result FileStorage::SetSize(s64 size) { m_size = InvalidSize; return m_base_file->SetSize(size); } Result FileStorage::OperateRange(void *dst, size_t dst_size, OperationId op_id, s64 offset, s64 size, const void *src, size_t src_size) { switch (op_id) { case OperationId::Invalidate: case OperationId::QueryRange: if (size == 0) { if (op_id == OperationId::QueryRange) { R_UNLESS(dst != nullptr, fs::ResultNullptrArgument()); R_UNLESS(dst_size == sizeof(QueryRangeInfo), fs::ResultInvalidSize()); reinterpret_cast(dst)->Clear(); } return ResultSuccess(); } R_TRY(this->UpdateSize()); R_UNLESS(IStorage::CheckOffsetAndSize(offset, size), fs::ResultOutOfRange()); return m_base_file->OperateRange(dst, dst_size, op_id, offset, size, src, src_size); default: return fs::ResultUnsupportedOperateRangeForFileStorage(); } } Result FileStorageBasedFileSystem::Initialize(std::shared_ptr base_file_system, const fs::Path &path, fs::OpenMode mode) { /* Open the file. */ std::unique_ptr base_file; R_TRY(base_file_system->OpenFile(std::addressof(base_file), path, mode)); /* Set the file. */ this->SetFile(std::move(base_file)); m_base_file_system = std::move(base_file_system); return ResultSuccess(); } Result FileHandleStorage::UpdateSize() { R_SUCCEED_IF(m_size != InvalidSize); return GetFileSize(std::addressof(m_size), m_handle); } Result FileHandleStorage::Read(s64 offset, void *buffer, size_t size) { /* Lock the mutex. */ std::scoped_lock lk(m_mutex); /* Immediately succeed if there's nothing to read. */ R_SUCCEED_IF(size == 0); /* Validate buffer. */ R_UNLESS(buffer != nullptr, fs::ResultNullptrArgument()); /* Ensure our size is valid. */ R_TRY(this->UpdateSize()); /* Ensure our access is valid. */ R_UNLESS(IStorage::CheckAccessRange(offset, size, m_size), fs::ResultOutOfRange()); return ReadFile(m_handle, offset, buffer, size, fs::ReadOption()); } Result FileHandleStorage::Write(s64 offset, const void *buffer, size_t size) { /* Lock the mutex. */ std::scoped_lock lk(m_mutex); /* Immediately succeed if there's nothing to write. */ R_SUCCEED_IF(size == 0); /* Validate buffer. */ R_UNLESS(buffer != nullptr, fs::ResultNullptrArgument()); /* Ensure our size is valid. */ R_TRY(this->UpdateSize()); /* Ensure our access is valid. */ R_UNLESS(IStorage::CheckAccessRange(offset, size, m_size), fs::ResultOutOfRange()); return WriteFile(m_handle, offset, buffer, size, fs::WriteOption()); } Result FileHandleStorage::Flush() { return FlushFile(m_handle); } Result FileHandleStorage::GetSize(s64 *out_size) { R_TRY(this->UpdateSize()); *out_size = m_size; return ResultSuccess(); } Result FileHandleStorage::SetSize(s64 size) { m_size = InvalidSize; return SetFileSize(m_handle, size); } Result FileHandleStorage::OperateRange(void *dst, size_t dst_size, OperationId op_id, s64 offset, s64 size, const void *src, size_t src_size) { AMS_UNUSED(src, src_size); switch (op_id) { case OperationId::QueryRange: /* Validate buffer and size. */ R_UNLESS(dst != nullptr, fs::ResultNullptrArgument()); R_UNLESS(dst_size == sizeof(QueryRangeInfo), fs::ResultInvalidSize()); return QueryRange(static_cast(dst), m_handle, offset, size); default: return fs::ResultUnsupportedOperateRangeForFileHandleStorage(); } } }