/*
 * 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 <http://www.gnu.org/licenses/>.
 */
#include <stratosphere.hpp>
#include <stratosphere/fssrv/fssrv_interface_adapters.hpp>
#include "fs_library.hpp"
#include "fs_file_system_service_object_adapter.hpp"
#include "../../fssrv/impl/fssrv_allocator_for_service_framework.hpp"

namespace ams::fs::impl {

    #if !defined(ATMOSPHERE_OS_HORIZON)
    namespace {

        constexpr size_t SystemHeapSize = 8_MB;
        alignas(os::MemoryPageSize) constinit u8 g_system_heap[SystemHeapSize];

        ALWAYS_INLINE auto &GetSystemHeapAllocator() {
            AMS_FUNCTION_LOCAL_STATIC(mem::StandardAllocator, s_system_heap_allocator, g_system_heap, sizeof(g_system_heap));
            return s_system_heap_allocator;
        }

        constinit util::optional<fssrv::MemoryResourceFromStandardAllocator> g_system_heap_memory_resource;

        void *AllocateForSystem(size_t size) { return g_system_heap_memory_resource->Allocate(size); }
        void DeallocateForSystem(void *p, size_t size) { return g_system_heap_memory_resource->Deallocate(p, size); }

        [[maybe_unused]] constexpr size_t BufferPoolSize        = 6_MB;
        [[maybe_unused]] constexpr size_t DeviceBufferSize      = 8_MB;
        [[maybe_unused]] constexpr size_t DeviceWorkBufferSize  = os::MemoryPageSize;
        [[maybe_unused]] constexpr size_t BufferManagerHeapSize = 14_MB;

        constexpr size_t DeviceWorkBufferRequiredSize = 0x140;

        static_assert(util::IsAligned(BufferManagerHeapSize, os::MemoryBlockUnitSize));

        //alignas(os::MemoryPageSize) u8 g_buffer_pool[BufferPoolSize];
        alignas(os::MemoryPageSize) u8 g_device_buffer[DeviceBufferSize];
        alignas(os::MemoryPageSize) u8 g_device_work_buffer[DeviceWorkBufferSize];
        //alignas(os::MemoryPageSize) u8 g_buffer_manager_heap[BufferManagerHeapSize];
        //
        //alignas(os::MemoryPageSize) u8 g_buffer_manager_work_buffer[64_KB];
        /* TODO: Other work buffers. */

        /* TODO: Implement pooled threads. */
        // constexpr int    PooledThreadCount     = 12;
        // constexpr size_t PooledThreadStackSize = 32_KB;
        // fssystem::PooledThread g_pooled_threads[PooledThreadCount];

        /* FileSystem creators. */
        constinit util::optional<fssrv::fscreator::LocalFileSystemCreator> g_local_fs_creator;
        constinit util::optional<fssrv::fscreator::SubDirectoryFileSystemCreator> g_subdir_fs_creator;

        constinit fssrv::fscreator::FileSystemCreatorInterfaces g_fs_creator_interfaces = {};

        Result InitializeFileSystemLibraryImpl() {
            /* Set system allocator. */
            fssystem::InitializeAllocator(::ams::fs::impl::Allocate, ::ams::fs::impl::Deallocate);
            fssystem::InitializeAllocatorForSystem(::ams::fs::impl::AllocateForSystem, ::ams::fs::impl::DeallocateForSystem);

            /* TODO: Many things. */
            g_system_heap_memory_resource.emplace(std::addressof(GetSystemHeapAllocator()));

            fssystem::InitializeBufferPool(reinterpret_cast<char *>(g_device_buffer), DeviceBufferSize, reinterpret_cast<char *>(g_device_work_buffer), DeviceWorkBufferRequiredSize);

            /* Setup fscreators/interfaces. */
            g_local_fs_creator.emplace(true);
            g_subdir_fs_creator.emplace();

            g_fs_creator_interfaces.local_fs_creator = std::addressof(*g_local_fs_creator);
            g_fs_creator_interfaces.subdir_fs_creator = std::addressof(*g_subdir_fs_creator);

            /* Initialize fssrv. */
            const fssrv::FileSystemProxyConfiguration config  = {
                .m_fs_creator_interfaces = std::addressof(g_fs_creator_interfaces),
                .m_base_storage_service_impl = nullptr /* TODO */,
                .m_base_file_system_service_impl = nullptr /* TODO */,
                .m_nca_file_system_service_impl = nullptr /* TODO */,
                .m_save_data_file_system_service_impl = nullptr /* TODO */,
                .m_access_failure_management_service_impl = nullptr /* TODO */,
                .m_time_service_impl = nullptr /* TODO */,
                .m_status_report_service_impl = nullptr /* TODO */,
                .m_program_registry_service_impl = nullptr /* TODO */,
                .m_access_log_service_impl = nullptr /* TODO */,
                .m_debug_configuration_service_impl = nullptr /* TODO */,
            };

            fssrv::InitializeForFileSystemProxy(config);
            R_SUCCEED();
        }

        class FileSystemLibraryInitializer {
            public:
                FileSystemLibraryInitializer() {
                    R_ABORT_UNLESS(InitializeFileSystemLibraryImpl());
                }
        };

    }
    #endif

    void InitializeFileSystemLibrary() {
        #if !defined(ATMOSPHERE_OS_HORIZON)
        AMS_FUNCTION_LOCAL_STATIC(FileSystemLibraryInitializer, s_library_initializer);
        AMS_UNUSED(s_library_initializer);
        #endif
    }

}