/*
 * Copyright (c) 2018-2020 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 
namespace ams::fssystem {
    class SparseStorage : public IndirectStorage {
        NON_COPYABLE(SparseStorage);
        NON_MOVEABLE(SparseStorage);
        private:
            class ZeroStorage : public ::ams::fs::IStorage, public ::ams::fs::impl::Newable {
                public:
                    ZeroStorage() { /* ... */ }
                    virtual ~ZeroStorage() { /* ... */ }
                virtual Result Read(s64 offset, void *buffer, size_t size) override {
                    AMS_ASSERT(offset >= 0);
                    AMS_ASSERT(buffer != nullptr || size == 0);
                    if (size > 0) {
                        std::memset(buffer, 0, size);
                    }
                    return ResultSuccess();
                }
                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 {
                    return ResultSuccess();
                }
                virtual Result GetSize(s64 *out) override {
                    AMS_ASSERT(out != nullptr);
                    *out = std::numeric_limits::max();
                    return ResultSuccess();
                }
                virtual Result Flush() override {
                    return ResultSuccess();
                }
                virtual Result Write(s64 offset, const void *buffer, size_t size) override {
                    return fs::ResultUnsupportedOperationInZeroStorageA();
                }
                virtual Result SetSize(s64 size) override {
                    return fs::ResultUnsupportedOperationInZeroStorageB();
                }
            };
        private:
            ZeroStorage zero_storage;
        public:
            SparseStorage() : IndirectStorage(), zero_storage() { /* ... */ }
            virtual ~SparseStorage() { /* ... */ }
            using IndirectStorage::Initialize;
            void Initialize(s64 end_offset) {
                this->GetEntryTable().Initialize(NodeSize, end_offset);
                this->SetZeroStorage();
            }
            void SetDataStorage(fs::SubStorage storage) {
                AMS_ASSERT(this->IsInitialized());
                this->SetStorage(0, storage);
                this->SetZeroStorage();
            }
            template
            void SetDataStorage(T storage, s64 offset, s64 size) {
                AMS_ASSERT(this->IsInitialized());
                this->SetStorage(0, storage, offset, size);
                this->SetZeroStorage();
            }
            virtual Result Read(s64 offset, void *buffer, size_t size) override;
        private:
            void SetZeroStorage() {
                return this->SetStorage(1, std::addressof(this->zero_storage), 0, std::numeric_limits::max());
            }
    };
}