/*
 * 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 "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(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(GetPointer(holder->impl_storage));
        AMS_ASSERT(holder_base->IsNotLinked());
        /* Destroy. */
        static_assert(std::is_trivially_destructible::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(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(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;
    }
}