/*
 * 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 
#include 
#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 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 g_local_fs_creator;
        constinit util::optional 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(g_device_buffer), DeviceBufferSize, reinterpret_cast(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
    }
}