/*
 * 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 {
    /* ACCURATE_TO_VERSION: Unknown */
    class FileStorage : public IStorage, public impl::Newable {
        NON_COPYABLE(FileStorage);
        NON_MOVEABLE(FileStorage);
        private:
            static constexpr s64 InvalidSize = -1;
        private:
            std::unique_ptr m_unique_file;
            std::shared_ptr m_shared_file;
            fsa::IFile *m_base_file;
            s64 m_size;
        public:
            FileStorage(fsa::IFile *f) : m_unique_file(f), m_size(InvalidSize) {
                m_base_file = m_unique_file.get();
            }
            FileStorage(std::unique_ptr f) : m_unique_file(std::move(f)), m_size(InvalidSize) {
                m_base_file = m_unique_file.get();
            }
            FileStorage(std::shared_ptr f) : m_shared_file(f), m_size(InvalidSize) {
                m_base_file = m_shared_file.get();
            }
            virtual ~FileStorage() { /* ... */ }
        private:
            Result UpdateSize();
        protected:
            constexpr FileStorage() : m_unique_file(), m_shared_file(), m_base_file(nullptr), m_size(InvalidSize) { /* ... */ }
            void SetFile(fs::fsa::IFile *file) {
                AMS_ASSERT(file != nullptr);
                AMS_ASSERT(m_base_file == nullptr);
                m_base_file = file;
            }
            void SetFile(std::unique_ptr &&file) {
                AMS_ASSERT(file != nullptr);
                AMS_ASSERT(m_base_file == nullptr);
                AMS_ASSERT(m_unique_file == nullptr);
                m_unique_file = std::move(file);
                m_base_file   = m_unique_file.get();
            }
        public:
            virtual Result Read(s64 offset, void *buffer, size_t size) override;
            virtual Result Write(s64 offset, const void *buffer, size_t size) override;
            virtual Result Flush() override;
            virtual Result GetSize(s64 *out_size) override;
            virtual Result SetSize(s64 size) override;
            virtual Result OperateRange(void *dst, size_t dst_size, OperationId op_id, s64 offset, s64 size, const void *src, size_t src_size) override;
    };
    /* ACCURATE_TO_VERSION: Unknown */
    class FileStorageBasedFileSystem : public FileStorage {
        NON_COPYABLE(FileStorageBasedFileSystem);
        NON_MOVEABLE(FileStorageBasedFileSystem);
        private:
            std::shared_ptr m_base_file_system;
        public:
            constexpr FileStorageBasedFileSystem() : FileStorage(), m_base_file_system(nullptr) { /* ... */ }
            Result Initialize(std::shared_ptr base_file_system, const fs::Path &path, fs::OpenMode mode);
    };
    class FileHandleStorage : public IStorage, public impl::Newable {
        private:
            static constexpr s64 InvalidSize = -1;
        private:
            FileHandle m_handle;
            bool m_close_file;
            s64 m_size;
            os::SdkMutex m_mutex;
        public:
            constexpr explicit FileHandleStorage(FileHandle handle, bool close_file) : m_handle(handle), m_close_file(close_file), m_size(InvalidSize), m_mutex() { /* ... */ }
            constexpr explicit FileHandleStorage(FileHandle handle) : FileHandleStorage(handle, false) { /* ... */ }
            virtual ~FileHandleStorage() override {
                if (m_close_file) {
                    CloseFile(m_handle);
                }
             }
        protected:
            Result UpdateSize();
        public:
            virtual Result Read(s64 offset, void *buffer, size_t size) override;
            virtual Result Write(s64 offset, const void *buffer, size_t size) override;
            virtual Result Flush() override;
            virtual Result GetSize(s64 *out_size) override;
            virtual Result SetSize(s64 size) override;
            virtual Result OperateRange(void *dst, size_t dst_size, OperationId op_id, s64 offset, s64 size, const void *src, size_t src_size) override;
    };
}