/* * 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 #include #include namespace ams::util { template class BoundedMap { private: std::array, N> m_keys; std::array, N> m_values; private: ALWAYS_INLINE void FreeEntry(size_t i) { m_keys[i].reset(); DestroyAt(m_values[i]); } public: constexpr BoundedMap() : m_keys(), m_values() { /* ... */ } Value *Find(const Key &key) { for (size_t i = 0; i < N; i++) { if (m_keys[i] && m_keys[i].value() == key) { return GetPointer(m_values[i]); } } return nullptr; } void Remove(const Key &key) { for (size_t i = 0; i < N; i++) { if (m_keys[i] && m_keys[i].value() == key) { this->FreeEntry(i); break; } } } void RemoveAll() { for (size_t i = 0; i < N; i++) { this->FreeEntry(i); } } bool IsFull() { for (size_t i = 0; i < N; i++) { if (!m_keys[i]) { return false; } } return true; } bool Insert(const Key &key, Value &&value) { /* We can't insert if the key is used. */ if (this->Find(key) != nullptr) { return false; } /* Find a free value. */ for (size_t i = 0; i < N; i++) { if (!m_keys[i]) { m_keys[i] = key; ConstructAt(m_values[i], std::forward(value)); return true; } } return false; } bool InsertOrAssign(const Key &key, Value &&value) { /* Try to find and assign an existing value. */ for (size_t i = 0; i < N; i++) { if (m_keys[i] && m_keys[i].value() == key) { GetReference(m_values[i]) = std::forward(value); return true; } } /* Find a free value. */ for (size_t i = 0; i < N; i++) { if (!m_keys[i]) { m_keys[i] = key; ConstructAt(m_values[i], std::move(value)); return true; } } return false; } template bool Emplace(const Key &key, Args&&... args) { /* We can't emplace if the key is used. */ if (this->Find(key) != nullptr) { return false; } /* Find a free value. */ for (size_t i = 0; i < N; i++) { if (!m_keys[i]) { m_keys[i] = key; ConstructAt(m_values[i], std::forward(args)...); return true; } } return false; } }; }