/*
 * 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 "htclow_mux_channel_impl_map.hpp"
namespace ams::htclow::mux {
    ChannelImplMap::ChannelImplMap(PacketFactory *pf, ctrl::HtcctrlStateMachine *sm, TaskManager *tm, os::Event *ev)
        : m_packet_factory(pf), m_state_machine(sm), m_task_manager(tm), m_event(ev), m_map()
    {
        /* Initialize the map. */
        m_map.Initialize(MaxChannelCount, m_map_buffer, sizeof(m_map_buffer));
        /* Set all storages as invalid. */
        for (auto i = 0; i < MaxChannelCount; ++i) {
            m_storage_valid[i] = false;
        }
    }
    ChannelImpl &ChannelImplMap::GetChannelImpl(int index) {
        return GetReference(m_channel_storage[index]);
    }
    ChannelImpl &ChannelImplMap::GetChannelImpl(impl::ChannelInternalType channel) {
        /* Find the channel. */
        auto it = m_map.find(channel);
        AMS_ASSERT(it != m_map.end());
        /* Return the implementation object. */
        return this->GetChannelImpl(it->second);
    }
    Result ChannelImplMap::AddChannel(impl::ChannelInternalType channel) {
        /* Find a free storage. */
        int idx;
        for (idx = 0; idx < MaxChannelCount; ++idx) {
            if (!m_storage_valid[idx]) {
                break;
            }
        }
        /* Validate that the storage is free. */
        R_UNLESS(idx < MaxChannelCount, htclow::ResultOutOfChannel());
        /* Create the channel impl. */
        util::ConstructAt(m_channel_storage[idx], channel, m_packet_factory, m_state_machine, m_task_manager, m_event);
        /* Mark the storage valid. */
        m_storage_valid[idx] = true;
        /* Insert into our map. */
        m_map.insert(std::pair{channel, idx});
        return ResultSuccess();
    }
    Result ChannelImplMap::RemoveChannel(impl::ChannelInternalType channel) {
        /* Find the storage. */
        auto it = m_map.find(channel);
        AMS_ASSERT(it != m_map.end());
        /* Get the channel index. */
        const auto index = it->second;
        AMS_ASSERT(0 <= index && index < MaxChannelCount);
        /* Get the channel impl. */
        auto *channel_impl = GetPointer(m_channel_storage[index]);
        /* Mark the storage as invalid. */
        m_storage_valid[index] = false;
        /* Erase the channel from the map. */
        m_map.erase(channel);
        /* Destroy the channel. */
        std::destroy_at(channel_impl);
        return ResultSuccess();
    }
}