/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */ /** * Copyright (c) 2014-2016, Regents of the University of California, * Arizona Board of Regents, * Colorado State University, * University Pierre & Marie Curie, Sorbonne University, * Washington University in St. Louis, * Beijing Institute of Technology, * The University of Memphis. * * This file is part of NFD (Named Data Networking Forwarding Daemon). * See AUTHORS.md for complete list of NFD authors and contributors. * * NFD is free software: you can redistribute it and/or modify it under the terms * of the GNU General Public License as published by the Free Software Foundation, * either version 3 of the License, or (at your option) any later version. * * NFD is distributed in the hope that 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 * NFD, e.g., in COPYING.md file. If not, see . */ #include "fib.hpp" #include "pit-entry.hpp" #include "measurements-entry.hpp" #include "core/asserts.hpp" namespace nfd { namespace fib { NFD_ASSERT_FORWARD_ITERATOR(Fib::const_iterator); const unique_ptr Fib::s_emptyEntry = make_unique(Name()); static inline bool nteHasFibEntry(const name_tree::Entry& nte) { return nte.getFibEntry() != nullptr; } Fib::Fib(NameTree& nameTree) : m_nameTree(nameTree) , m_nItems(0) { } template const Entry& Fib::findLongestPrefixMatchImpl(const K& key) const { name_tree::Entry* nte = m_nameTree.findLongestPrefixMatch(key, &nteHasFibEntry); if (nte != nullptr) { return *nte->getFibEntry(); } return *s_emptyEntry; } const Entry& Fib::findLongestPrefixMatch(const Name& prefix) const { return this->findLongestPrefixMatchImpl(prefix); } const Entry& Fib::findLongestPrefixMatch(const pit::Entry& pitEntry) const { return this->findLongestPrefixMatchImpl(pitEntry); } const Entry& Fib::findLongestPrefixMatch(const measurements::Entry& measurementsEntry) const { return this->findLongestPrefixMatchImpl(measurementsEntry); } Entry* Fib::findExactMatch(const Name& prefix) { name_tree::Entry* nte = m_nameTree.findExactMatch(prefix); if (nte != nullptr) return nte->getFibEntry(); return nullptr; } std::pair Fib::insert(const Name& prefix) { name_tree::Entry& nte = m_nameTree.lookup(prefix); Entry* entry = nte.getFibEntry(); if (entry != nullptr) { return std::make_pair(entry, false); } nte.setFibEntry(make_unique(prefix)); ++m_nItems; return std::make_pair(nte.getFibEntry(), true); } void Fib::erase(name_tree::Entry* nte, bool canDeleteNte) { BOOST_ASSERT(nte != nullptr); nte->setFibEntry(nullptr); if (canDeleteNte) { m_nameTree.eraseIfEmpty(nte); } --m_nItems; } void Fib::erase(const Name& prefix) { name_tree::Entry* nte = m_nameTree.findExactMatch(prefix); if (nte != nullptr) { this->erase(nte); } } void Fib::erase(const Entry& entry) { name_tree::Entry* nte = m_nameTree.getEntry(entry); if (nte == nullptr) { // don't try to erase s_emptyEntry BOOST_ASSERT(&entry == s_emptyEntry.get()); return; } this->erase(nte); } void Fib::removeNextHop(Entry& entry, const Face& face) { entry.removeNextHop(face); if (!entry.hasNextHops()) { name_tree::Entry* nte = m_nameTree.getEntry(entry); this->erase(nte, false); } } Fib::Range Fib::getRange() const { return m_nameTree.fullEnumerate(&nteHasFibEntry) | boost::adaptors::transformed(name_tree::GetTableEntry(&name_tree::Entry::getFibEntry)); } } // namespace fib } // namespace nfd