/*
 * 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 
namespace ams::fssystem {
    template
    class LruListCache {
        NON_COPYABLE(LruListCache);
        NON_MOVEABLE(LruListCache);
        public:
            class Node : public ::ams::fs::impl::Newable {
                NON_COPYABLE(Node);
                NON_MOVEABLE(Node);
                public:
                    Key m_key;
                    Value m_value;
                    util::IntrusiveListNode m_mru_list_node;
                public:
                    explicit Node(const Value &value) : m_value(value) { /* ... */ }
            };
        private:
            using MruList = typename util::IntrusiveListMemberTraits<&Node::m_mru_list_node>::ListType;
        private:
            MruList m_mru_list;
        public:
            constexpr LruListCache() : m_mru_list() { /* ... */ }
            bool FindValueAndUpdateMru(Value *out, const Key &key) {
                for (auto it = m_mru_list.begin(); it != m_mru_list.end(); ++it) {
                    if (it->m_key == key) {
                        *out = it->m_value;
                        m_mru_list.erase(it);
                        m_mru_list.push_front(*it);
                        return true;
                    }
                }
                return false;
            }
            std::unique_ptr PopLruNode() {
                AMS_ABORT_UNLESS(!m_mru_list.empty());
                Node *lru = std::addressof(*m_mru_list.rbegin());
                m_mru_list.pop_back();
                return std::unique_ptr(lru);
            }
            void PushMruNode(std::unique_ptr &&node, const Key &key) {
                node->m_key = key;
                m_mru_list.push_front(*node);
                node.release();
            }
            void DeleteAllNodes() {
                while (!m_mru_list.empty()) {
                    Node *lru = std::addressof(*m_mru_list.rbegin());
                    m_mru_list.erase(m_mru_list.iterator_to(*lru));
                    delete lru;
                }
            }
            size_t GetSize() const {
                return m_mru_list.size();
            }
            bool IsEmpty() const {
                return m_mru_list.empty();
            }
    };
}