/*
 * Copyright (c) 2018-2020 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/>.
 */
#pragma once
#include <vapours.hpp>
#include <stratosphere/os.hpp>

namespace ams::ddsf {

    class EventHandlerManager;

    class IEventHandler {
        NON_COPYABLE(IEventHandler);
        NON_MOVEABLE(IEventHandler);
        friend class EventHandlerManager;
        private:
            os::WaitableHolderType holder;
            uintptr_t user_data;
            bool is_initialized;
            bool is_registered;
        private:
            void Link(os::WaitableManagerType *manager) {
                AMS_ASSERT(this->IsInitialized());
                AMS_ASSERT(!this->IsRegistered());
                AMS_ASSERT(manager != nullptr);
                os::LinkWaitableHolder(manager, std::addressof(this->holder));
            }

            void Unlink() {
                AMS_ASSERT(this->IsInitialized());
                AMS_ASSERT(this->IsRegistered());
                os::UnlinkWaitableHolder(std::addressof(this->holder));
            }

            static IEventHandler &ToEventHandler(os::WaitableHolderType *holder) {
                AMS_ASSERT(holder != nullptr);
                auto &event_handler = *reinterpret_cast<IEventHandler *>(os::GetWaitableHolderUserData(holder));
                AMS_ASSERT(event_handler.IsInitialized());
                return event_handler;
            }
        public:
            IEventHandler() : holder(), user_data(0), is_initialized(false), is_registered(false) { /* ... */ }

            ~IEventHandler() {
                if (this->IsRegistered()) {
                    this->Unlink();
                }
                if (this->IsInitialized()) {
                    this->Finalize();
                }
            }

            bool IsInitialized() const { return this->is_initialized; }
            bool IsRegistered() const { return this->is_registered; }

            uintptr_t GetUserData() const { return this->user_data; }
            void SetUserData(uintptr_t d) { this->user_data = d; }

            template<typename T>
            void Initialize(T *object) {
                AMS_ASSERT(object != nullptr);
                AMS_ASSERT(!this->IsInitialized());
                os::InitializeWaitableHolder(std::addressof(this->holder), object);
                os::SetWaitableHolderUserData(std::addressof(this->holder), reinterpret_cast<uintptr_t>(this));
                this->is_initialized = true;
                this->is_registered  = false;
            }

            void Finalize() {
                AMS_ASSERT(this->IsInitialized());
                AMS_ASSERT(!this->IsRegistered());
                os::FinalizeWaitableHolder(std::addressof(this->holder));
                this->is_initialized = false;
                this->is_registered  = false;
            }
        protected:
            virtual void HandleEvent() = 0;
    };

}