/* * 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 "impl/os_multiple_wait_impl.hpp" #include "impl/os_multiple_wait_holder_base.hpp" #include "impl/os_multiple_wait_holder_impl.hpp" namespace ams::os { namespace { ALWAYS_INLINE impl::MultiWaitImpl &GetMultiWaitImpl(MultiWaitType *multi_wait) { return GetReference(multi_wait->impl_storage); } ALWAYS_INLINE MultiWaitHolderType *CastToMultiWaitHolder(impl::MultiWaitHolderBase *base) { return reinterpret_cast<MultiWaitHolderType *>(base); } } void InitializeMultiWait(MultiWaitType *multi_wait) { /* Initialize storage. */ util::ConstructAt(multi_wait->impl_storage); /* Mark initialized. */ multi_wait->state = MultiWaitType::State_Initialized; } void FinalizeMultiWait(MultiWaitType *multi_wait) { auto &impl = GetMultiWaitImpl(multi_wait); AMS_ASSERT(multi_wait->state == MultiWaitType::State_Initialized); AMS_ASSERT(impl.IsListEmpty()); AMS_UNUSED(impl); /* Mark not initialized. */ multi_wait->state = MultiWaitType::State_NotInitialized; /* Destroy. */ util::DestroyAt(multi_wait->impl_storage); } MultiWaitHolderType *WaitAny(MultiWaitType *multi_wait) { auto &impl = GetMultiWaitImpl(multi_wait); AMS_ASSERT(multi_wait->state == MultiWaitType::State_Initialized); AMS_ASSERT(impl.IsListNotEmpty()); auto *holder = CastToMultiWaitHolder(impl.WaitAny()); AMS_ASSERT(holder != nullptr); return holder; } MultiWaitHolderType *TryWaitAny(MultiWaitType *multi_wait) { auto &impl = GetMultiWaitImpl(multi_wait); AMS_ASSERT(multi_wait->state == MultiWaitType::State_Initialized); AMS_ASSERT(impl.IsListNotEmpty()); auto *holder = CastToMultiWaitHolder(impl.TryWaitAny()); return holder; } MultiWaitHolderType *TimedWaitAny(MultiWaitType *multi_wait, TimeSpan timeout) { auto &impl = GetMultiWaitImpl(multi_wait); AMS_ASSERT(multi_wait->state == MultiWaitType::State_Initialized); AMS_ASSERT(impl.IsListNotEmpty()); AMS_ASSERT(timeout.GetNanoSeconds() >= 0); auto *holder = CastToMultiWaitHolder(impl.TimedWaitAny(timeout)); return holder; } void FinalizeMultiWaitHolder(MultiWaitHolderType *holder) { auto *holder_base = reinterpret_cast<impl::MultiWaitHolderBase *>(GetPointer(holder->impl_storage)); AMS_ASSERT(holder_base->IsNotLinked()); /* Destroy. */ static_assert(std::is_trivially_destructible<impl::MultiWaitHolderBase>::value); /* std::destroy_at(holder_base); */ AMS_UNUSED(holder_base); } void LinkMultiWaitHolder(MultiWaitType *multi_wait, MultiWaitHolderType *holder) { auto &impl = GetMultiWaitImpl(multi_wait); auto *holder_base = reinterpret_cast<impl::MultiWaitHolderBase *>(GetPointer(holder->impl_storage)); AMS_ASSERT(multi_wait->state == MultiWaitType::State_Initialized); AMS_ASSERT(holder_base->IsNotLinked()); impl.PushBackToList(*holder_base); holder_base->SetMultiWait(std::addressof(impl)); } void UnlinkMultiWaitHolder(MultiWaitHolderType *holder) { auto *holder_base = reinterpret_cast<impl::MultiWaitHolderBase *>(GetPointer(holder->impl_storage)); /* Don't allow unlinking of an unlinked holder. */ AMS_ABORT_UNLESS(holder_base->IsLinked()); holder_base->GetMultiWait()->EraseFromList(*holder_base); holder_base->SetMultiWait(nullptr); } void UnlinkAllMultiWaitHolder(MultiWaitType *multi_wait) { auto &impl = GetMultiWaitImpl(multi_wait); AMS_ASSERT(multi_wait->state == MultiWaitType::State_Initialized); return impl.EraseAllFromList(); } void MoveAllMultiWaitHolder(MultiWaitType *_dst, MultiWaitType *_src) { auto &dst = GetMultiWaitImpl(_dst); auto &src = GetMultiWaitImpl(_src); AMS_ASSERT(_dst->state == MultiWaitType::State_Initialized); AMS_ASSERT(_src->state == MultiWaitType::State_Initialized); return dst.MoveAllFromOther(src); } void SetMultiWaitHolderUserData(MultiWaitHolderType *holder, uintptr_t user_data) { holder->user_data = user_data; } uintptr_t GetMultiWaitHolderUserData(const MultiWaitHolderType *holder) { return holder->user_data; } void InitializeMultiWaitHolder(MultiWaitHolderType *holder, NativeHandle handle) { AMS_ASSERT(handle != os::InvalidNativeHandle); util::ConstructAt(GetReference(holder->impl_storage).holder_of_native_handle_storage, handle); holder->user_data = 0; } }