/* * 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::fssystem { /* ACCURATE_TO_VERSION: Unknown */ template class AlignmentMatchingStorage : public ::ams::fs::IStorage, public ::ams::fs::impl::Newable { NON_COPYABLE(AlignmentMatchingStorage); NON_MOVEABLE(AlignmentMatchingStorage); public: static constexpr size_t DataAlign = _DataAlign; static constexpr size_t BufferAlign = _BufferAlign; static constexpr size_t DataAlignMax = 0x200; static_assert(DataAlign <= DataAlignMax); static_assert(util::IsPowerOfTwo(DataAlign)); static_assert(util::IsPowerOfTwo(BufferAlign)); private: std::shared_ptr m_shared_base_storage; fs::IStorage * const m_base_storage; s64 m_base_storage_size; bool m_is_base_storage_size_dirty; public: explicit AlignmentMatchingStorage(fs::IStorage *bs) : m_base_storage(bs), m_is_base_storage_size_dirty(true) { /* ... */ } explicit AlignmentMatchingStorage(std::shared_ptr bs) : m_shared_base_storage(std::move(bs)), m_base_storage(m_shared_base_storage.get()), m_is_base_storage_size_dirty(true) { /* ... */ } virtual Result Read(s64 offset, void *buffer, size_t size) override { /* Allocate a work buffer on stack. */ __attribute__((aligned(DataAlignMax))) char work_buf[DataAlign]; static_assert(util::IsAligned(alignof(work_buf), BufferAlign)); /* Succeed if zero size. */ R_SUCCEED_IF(size == 0); /* Validate arguments. */ R_UNLESS(buffer != nullptr, fs::ResultNullptrArgument()); s64 bs_size = 0; R_TRY(this->GetSize(std::addressof(bs_size))); R_TRY(fs::IStorage::CheckAccessRange(offset, size, bs_size)); R_RETURN(AlignmentMatchingStorageImpl::Read(m_base_storage, work_buf, sizeof(work_buf), DataAlign, BufferAlign, offset, static_cast(buffer), size)); } virtual Result Write(s64 offset, const void *buffer, size_t size) override { /* Allocate a work buffer on stack. */ __attribute__((aligned(DataAlignMax))) char work_buf[DataAlign]; static_assert(util::IsAligned(alignof(work_buf), BufferAlign)); /* Succeed if zero size. */ R_SUCCEED_IF(size == 0); /* Validate arguments. */ R_UNLESS(buffer != nullptr, fs::ResultNullptrArgument()); s64 bs_size = 0; R_TRY(this->GetSize(std::addressof(bs_size))); R_TRY(fs::IStorage::CheckAccessRange(offset, size, bs_size)); R_RETURN(AlignmentMatchingStorageImpl::Write(m_base_storage, work_buf, sizeof(work_buf), DataAlign, BufferAlign, offset, static_cast(buffer), size)); } virtual Result Flush() override { R_RETURN(m_base_storage->Flush()); } virtual Result SetSize(s64 size) override { ON_SCOPE_EXIT { m_is_base_storage_size_dirty = true; }; R_RETURN(m_base_storage->SetSize(util::AlignUp(size, DataAlign))); } virtual Result GetSize(s64 *out) override { AMS_ASSERT(out != nullptr); if (m_is_base_storage_size_dirty) { s64 size; R_TRY(m_base_storage->GetSize(std::addressof(size))); m_base_storage_size = size; m_is_base_storage_size_dirty = false; } *out = m_base_storage_size; R_SUCCEED(); } virtual Result OperateRange(void *dst, size_t dst_size, fs::OperationId op_id, s64 offset, s64 size, const void *src, size_t src_size) override { if (op_id == fs::OperationId::Invalidate) { R_RETURN(m_base_storage->OperateRange(fs::OperationId::Invalidate, offset, size)); } else { /* Succeed if zero size. */ R_SUCCEED_IF(size == 0); /* Get the base storage size. */ s64 bs_size = 0; R_TRY(this->GetSize(std::addressof(bs_size))); R_TRY(fs::IStorage::CheckOffsetAndSize(offset, size)); /* Operate on the base storage. */ const auto valid_size = std::min(size, bs_size - offset); const auto aligned_offset = util::AlignDown(offset, DataAlign); const auto aligned_offset_end = util::AlignUp(offset + valid_size, DataAlign); const auto aligned_size = aligned_offset_end - aligned_offset; R_RETURN(m_base_storage->OperateRange(dst, dst_size, op_id, aligned_offset, aligned_size, src, src_size)); } } }; /* ACCURATE_TO_VERSION: Unknown */ template class AlignmentMatchingStoragePooledBuffer : public ::ams::fs::IStorage, public ::ams::fs::impl::Newable { NON_COPYABLE(AlignmentMatchingStoragePooledBuffer); NON_MOVEABLE(AlignmentMatchingStoragePooledBuffer); public: static constexpr size_t BufferAlign = _BufferAlign; static_assert(util::IsPowerOfTwo(BufferAlign)); private: BasePointer m_base_storage; s64 m_base_storage_size; size_t m_data_align; bool m_is_base_storage_size_dirty; public: explicit AlignmentMatchingStoragePooledBuffer(BasePointer bs, size_t da) : m_base_storage(std::move(bs)), m_data_align(da), m_is_base_storage_size_dirty(true) { AMS_ASSERT(util::IsPowerOfTwo(da)); } virtual Result Read(s64 offset, void *buffer, size_t size) override { /* Succeed if zero size. */ R_SUCCEED_IF(size == 0); /* Validate arguments. */ R_UNLESS(buffer != nullptr, fs::ResultNullptrArgument()); s64 bs_size = 0; R_TRY(this->GetSize(std::addressof(bs_size))); R_TRY(fs::IStorage::CheckAccessRange(offset, size, bs_size)); /* Allocate a pooled buffer. */ PooledBuffer pooled_buffer; pooled_buffer.AllocateParticularlyLarge(m_data_align, m_data_align); R_RETURN(AlignmentMatchingStorageImpl::Read(m_base_storage, pooled_buffer.GetBuffer(), pooled_buffer.GetSize(), m_data_align, BufferAlign, offset, static_cast(buffer), size)); } virtual Result Write(s64 offset, const void *buffer, size_t size) override { /* Succeed if zero size. */ R_SUCCEED_IF(size == 0); /* Validate arguments. */ R_UNLESS(buffer != nullptr, fs::ResultNullptrArgument()); s64 bs_size = 0; R_TRY(this->GetSize(std::addressof(bs_size))); R_TRY(fs::IStorage::CheckAccessRange(offset, size, bs_size)); /* Allocate a pooled buffer. */ PooledBuffer pooled_buffer; pooled_buffer.AllocateParticularlyLarge(m_data_align, m_data_align); R_RETURN(AlignmentMatchingStorageImpl::Write(m_base_storage, pooled_buffer.GetBuffer(), pooled_buffer.GetSize(), m_data_align, BufferAlign, offset, static_cast(buffer), size)); } virtual Result Flush() override { R_RETURN(m_base_storage->Flush()); } virtual Result SetSize(s64 size) override { ON_SCOPE_EXIT { m_is_base_storage_size_dirty = true; }; R_RETURN(m_base_storage->SetSize(util::AlignUp(size, m_data_align))); } virtual Result GetSize(s64 *out) override { AMS_ASSERT(out != nullptr); if (m_is_base_storage_size_dirty) { s64 size; R_TRY(m_base_storage->GetSize(std::addressof(size))); m_base_storage_size = size; m_is_base_storage_size_dirty = false; } *out = m_base_storage_size; R_SUCCEED(); } virtual Result OperateRange(void *dst, size_t dst_size, fs::OperationId op_id, s64 offset, s64 size, const void *src, size_t src_size) override { if (op_id == fs::OperationId::Invalidate) { R_RETURN(m_base_storage->OperateRange(fs::OperationId::Invalidate, offset, size)); } else { /* Succeed if zero size. */ R_SUCCEED_IF(size == 0); /* Get the base storage size. */ s64 bs_size = 0; R_TRY(this->GetSize(std::addressof(bs_size))); R_TRY(fs::IStorage::CheckOffsetAndSize(offset, size)); /* Operate on the base storage. */ const auto valid_size = std::min(size, bs_size - offset); const auto aligned_offset = util::AlignDown(offset, m_data_align); const auto aligned_offset_end = util::AlignUp(offset + valid_size, m_data_align); const auto aligned_size = aligned_offset_end - aligned_offset; R_RETURN(m_base_storage->OperateRange(dst, dst_size, op_id, aligned_offset, aligned_size, src, src_size)); } } }; /* ACCURATE_TO_VERSION: Unknown */ template class AlignmentMatchingStorageInBulkRead : public ::ams::fs::IStorage, public ::ams::fs::impl::Newable { NON_COPYABLE(AlignmentMatchingStorageInBulkRead); NON_MOVEABLE(AlignmentMatchingStorageInBulkRead); public: static constexpr size_t BufferAlign = _BufferAlign; static_assert(util::IsPowerOfTwo(BufferAlign)); private: std::shared_ptr m_shared_base_storage; fs::IStorage * const m_base_storage; s64 m_base_storage_size; size_t m_data_align; public: explicit AlignmentMatchingStorageInBulkRead(fs::IStorage *bs, size_t da) : m_shared_base_storage(), m_base_storage(bs), m_base_storage_size(-1), m_data_align(da) { AMS_ASSERT(util::IsPowerOfTwo(m_data_align)); } explicit AlignmentMatchingStorageInBulkRead(std::shared_ptr bs, size_t da) : m_shared_base_storage(bs), m_base_storage(m_shared_base_storage.get()), m_base_storage_size(-1), m_data_align(da) { AMS_ASSERT(util::IsPowerOfTwo(da)); } virtual Result Read(s64 offset, void *buffer, size_t size) override; virtual Result Write(s64 offset, const void *buffer, size_t size) override { /* Succeed if zero size. */ R_SUCCEED_IF(size == 0); /* Validate arguments. */ R_UNLESS(buffer != nullptr, fs::ResultNullptrArgument()); s64 bs_size = 0; R_TRY(this->GetSize(std::addressof(bs_size))); R_TRY(fs::IStorage::CheckAccessRange(offset, size, bs_size)); /* Allocate a pooled buffer. */ PooledBuffer pooled_buffer(m_data_align, m_data_align); R_RETURN(AlignmentMatchingStorageImpl::Write(m_base_storage, pooled_buffer.GetBuffer(), pooled_buffer.GetSize(), m_data_align, BufferAlign, offset, static_cast(buffer), size)); } virtual Result Flush() override { R_RETURN(m_base_storage->Flush()); } virtual Result SetSize(s64 size) override { ON_SCOPE_EXIT { m_base_storage_size = -1; }; R_RETURN(m_base_storage->SetSize(util::AlignUp(size, m_data_align))); } virtual Result GetSize(s64 *out) override { AMS_ASSERT(out != nullptr); if (m_base_storage_size < 0) { s64 size; R_TRY(m_base_storage->GetSize(std::addressof(size))); m_base_storage_size = size; } *out = m_base_storage_size; R_SUCCEED(); } virtual Result OperateRange(void *dst, size_t dst_size, fs::OperationId op_id, s64 offset, s64 size, const void *src, size_t src_size) override { if (op_id == fs::OperationId::Invalidate) { R_RETURN(m_base_storage->OperateRange(fs::OperationId::Invalidate, offset, size)); } else { /* Succeed if zero size. */ R_SUCCEED_IF(size == 0); /* Get the base storage size. */ s64 bs_size = 0; R_TRY(this->GetSize(std::addressof(bs_size))); R_TRY(fs::IStorage::CheckOffsetAndSize(offset, size)); /* Operate on the base storage. */ const auto valid_size = std::min(size, bs_size - offset); const auto aligned_offset = util::AlignDown(offset, m_data_align); const auto aligned_offset_end = util::AlignUp(offset + valid_size, m_data_align); const auto aligned_size = aligned_offset_end - aligned_offset; R_RETURN(m_base_storage->OperateRange(dst, dst_size, op_id, aligned_offset, aligned_size, src, src_size)); } } }; }