/* * 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 "htcs_manager.hpp" namespace ams::htcs::impl::HtcsManagerHolder { namespace { constinit os::SdkMutex g_holder_mutex; constinit int g_holder_reference_count = 0; mem::StandardAllocator g_allocator; constinit HtcsManager *g_manager = nullptr; alignas(os::MemoryPageSize) u8 g_heap_buffer[416_KB]; } void AddReference() { std::scoped_lock lk(g_holder_mutex); if ((g_holder_reference_count++) == 0) { /* Add reference to the htclow manager. */ htclow::HtclowManagerHolder::AddReference(); /* Initialize the allocator for the manager. */ g_allocator.Initialize(g_heap_buffer, sizeof(g_heap_buffer)); /* Allocate the manager. */ g_manager = static_cast<HtcsManager *>(g_allocator.Allocate(sizeof(HtcsManager), alignof(HtcsManager))); /* Construct the manager. */ std::construct_at(g_manager, std::addressof(g_allocator), htclow::HtclowManagerHolder::GetHtclowManager()); } AMS_ASSERT(g_holder_reference_count > 0); } void Release() { std::scoped_lock lk(g_holder_mutex); AMS_ASSERT(g_holder_reference_count > 0); if ((--g_holder_reference_count) == 0) { /* Destroy the manager. */ std::destroy_at(g_manager); g_allocator.Free(g_manager); g_manager = nullptr; /* Finalize the allocator. */ g_allocator.Finalize(); /* Release reference to the htclow manager. */ htclow::HtclowManagerHolder::Release(); } } HtcsManager *GetHtcsManager() { std::scoped_lock lk(g_holder_mutex); return g_manager; } }