/* * 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 namespace ams::fssystem { namespace impl { template struct PimplHelper { static void Construct(void *); static void Destroy(void *); }; } template class Pimpl { private: #if defined(ATMOSPHERE_OS_HORIZON) || defined(ATMOSPHERE_OS_WINDOWS) || defined(ATMOSPHERE_OS_LINUX) static constexpr size_t ExtraSizeToEnsureCompatibility = 0; #elif defined(ATMOSPHERE_OS_MACOS) static constexpr size_t ExtraSizeToEnsureCompatibility = 0x20; #endif static constexpr size_t StorageSize = Size + ExtraSizeToEnsureCompatibility; private: alignas(0x10) u8 m_storage[StorageSize]; public: ALWAYS_INLINE Pimpl() { impl::PimplHelper::Construct(m_storage); } ALWAYS_INLINE ~Pimpl() { impl::PimplHelper::Destroy(m_storage); } ALWAYS_INLINE T *Get() { return reinterpret_cast(m_storage + 0); } ALWAYS_INLINE T *operator->() { return reinterpret_cast(m_storage + 0); } }; #define AMS_FSSYSTEM_ENABLE_PIMPL(_CLASSNAME_) \ namespace ams::fssystem::impl { \ \ template \ struct PimplHelper<_CLASSNAME_, Size> { \ static ALWAYS_INLINE void Construct(void *p) { \ static_assert(sizeof(_CLASSNAME_) <= Size); \ static_assert(alignof(_CLASSNAME_) <= 0x10); \ \ std::construct_at(static_cast<_CLASSNAME_ *>(p)); \ } \ \ static ALWAYS_INLINE void Destroy(void *p) { \ std::destroy_at(static_cast<_CLASSNAME_ *>(p)); \ } \ }; \ \ } }