| /* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */ |
| /** |
| * Copyright (c) 2014-2017, The University of Memphis, |
| * Regents of the University of California, |
| * Arizona Board of Regents. |
| * |
| * This file is part of NLSR (Named-data Link State Routing). |
| * See AUTHORS.md for complete list of NLSR authors and contributors. |
| * |
| * NLSR 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. |
| * |
| * NLSR 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 |
| * NLSR, e.g., in COPYING.md file. If not, see <http://www.gnu.org/licenses/>. |
| **/ |
| |
| #include "adjacency-list.hpp" |
| |
| #include "adjacent.hpp" |
| #include "common.hpp" |
| #include "logger.hpp" |
| |
| #include <algorithm> |
| |
| namespace nlsr { |
| |
| INIT_LOGGER("AdjacencyList"); |
| |
| AdjacencyList::AdjacencyList() |
| { |
| } |
| |
| AdjacencyList::~AdjacencyList() |
| { |
| } |
| |
| int32_t |
| AdjacencyList::insert(Adjacent& adjacent) |
| { |
| std::list<Adjacent>::iterator it = find(adjacent.getName()); |
| if (it != m_adjList.end()) { |
| return -1; |
| } |
| m_adjList.push_back(adjacent); |
| return 0; |
| } |
| |
| void |
| AdjacencyList::addAdjacents(AdjacencyList& adl) |
| { |
| for (std::list<Adjacent>::iterator it = adl.getAdjList().begin(); |
| it != adl.getAdjList().end(); ++it) { |
| insert((*it)); |
| } |
| } |
| |
| bool |
| AdjacencyList::updateAdjacentStatus(const ndn::Name& adjName, Adjacent::Status s) |
| { |
| std::list<Adjacent>::iterator it = find(adjName); |
| |
| if (it == m_adjList.end()) { |
| return false; |
| } |
| else { |
| it->setStatus(s); |
| return true; |
| } |
| } |
| |
| Adjacent |
| AdjacencyList::getAdjacent(const ndn::Name& adjName) |
| { |
| Adjacent adj(adjName); |
| std::list<Adjacent>::iterator it = find(adjName); |
| if (it != m_adjList.end()) { |
| return (*it); |
| } |
| return adj; |
| } |
| |
| bool |
| AdjacencyList::operator==(AdjacencyList& adl) const |
| { |
| if (size() != adl.size()) { |
| return false; |
| } |
| |
| auto comparator = |
| [] (const Adjacent* lhs, const Adjacent* rhs) { |
| return *lhs < *rhs; |
| }; |
| |
| std::vector<const Adjacent*> ourList; |
| std::transform(m_adjList.begin(), m_adjList.end(), |
| std::back_inserter(ourList), std::pointer_traits<const Adjacent*>::pointer_to); |
| |
| std::vector<const Adjacent*> theirList; |
| std::transform(adl.getAdjList().begin(), adl.getAdjList().end(), |
| std::back_inserter(theirList), std::pointer_traits<const Adjacent*>::pointer_to); |
| |
| std::sort(ourList.begin(), ourList.end(), std::bind(comparator, _1, _2)); |
| std::sort(theirList.begin(), theirList.end(), std::bind(comparator, _1, _2)); |
| |
| for (size_t i = 0; i < ourList.size(); i++) { |
| if (*(ourList[i]) != *(theirList[i])) { |
| return false; |
| } |
| } |
| return true; |
| } |
| |
| int32_t |
| AdjacencyList::updateAdjacentLinkCost(const ndn::Name& adjName, double lc) |
| { |
| std::list<Adjacent>::iterator it = find(adjName); |
| if (it == m_adjList.end()) { |
| return -1; |
| } |
| (*it).setLinkCost(lc); |
| return 0; |
| } |
| |
| bool |
| AdjacencyList::isNeighbor(const ndn::Name& adjName) |
| { |
| std::list<Adjacent>::iterator it = find(adjName); |
| if (it == m_adjList.end()) |
| { |
| return false; |
| } |
| return true; |
| } |
| |
| void |
| AdjacencyList::incrementTimedOutInterestCount(const ndn::Name& neighbor) |
| { |
| std::list<Adjacent>::iterator it = find(neighbor); |
| if (it == m_adjList.end()) { |
| return ; |
| } |
| (*it).setInterestTimedOutNo((*it).getInterestTimedOutNo() + 1); |
| } |
| |
| void |
| AdjacencyList::setTimedOutInterestCount(const ndn::Name& neighbor, |
| uint32_t count) |
| { |
| std::list<Adjacent>::iterator it = find(neighbor); |
| if (it != m_adjList.end()) { |
| (*it).setInterestTimedOutNo(count); |
| } |
| } |
| |
| int32_t |
| AdjacencyList::getTimedOutInterestCount(const ndn::Name& neighbor) |
| { |
| std::list<Adjacent>::iterator it = find(neighbor); |
| if (it == m_adjList.end()) { |
| return -1; |
| } |
| return (*it).getInterestTimedOutNo(); |
| } |
| |
| Adjacent::Status |
| AdjacencyList::getStatusOfNeighbor(const ndn::Name& neighbor) |
| { |
| std::list<Adjacent>::iterator it = find(neighbor); |
| |
| if (it == m_adjList.end()) { |
| return Adjacent::STATUS_UNKNOWN; |
| } |
| else { |
| return it->getStatus(); |
| } |
| } |
| |
| void |
| AdjacencyList::setStatusOfNeighbor(const ndn::Name& neighbor, Adjacent::Status status) |
| { |
| std::list<Adjacent>::iterator it = find(neighbor); |
| if (it != m_adjList.end()) { |
| it->setStatus(status); |
| } |
| } |
| |
| std::list<Adjacent>& |
| AdjacencyList::getAdjList() |
| { |
| return m_adjList; |
| } |
| |
| const std::list<Adjacent>& |
| AdjacencyList::getAdjList() const |
| { |
| return m_adjList; |
| } |
| |
| bool |
| AdjacencyList::isAdjLsaBuildable(const uint32_t interestRetryNo) const |
| { |
| uint32_t nTimedOutNeighbors = 0; |
| |
| for (const Adjacent& adjacency : m_adjList) { |
| |
| if (adjacency.getStatus() == Adjacent::STATUS_ACTIVE) { |
| return true; |
| } |
| else if (adjacency.getInterestTimedOutNo() >= interestRetryNo) { |
| nTimedOutNeighbors++; |
| } |
| } |
| |
| if (nTimedOutNeighbors == m_adjList.size()) { |
| return true; |
| } |
| else { |
| return false; |
| } |
| } |
| |
| int32_t |
| AdjacencyList::getNumOfActiveNeighbor() |
| { |
| int32_t actNbrCount = 0; |
| for (std::list<Adjacent>::iterator it = m_adjList.begin(); it != m_adjList.end(); it++) { |
| |
| if (it->getStatus() == Adjacent::STATUS_ACTIVE) { |
| actNbrCount++; |
| } |
| } |
| return actNbrCount; |
| } |
| |
| std::list<Adjacent>::iterator |
| AdjacencyList::find(const ndn::Name& adjName) |
| { |
| std::list<Adjacent>::iterator it = std::find_if(m_adjList.begin(), |
| m_adjList.end(), |
| std::bind(&Adjacent::compare, |
| _1, std::cref(adjName))); |
| return it; |
| } |
| |
| AdjacencyList::iterator |
| AdjacencyList::findAdjacent(const ndn::Name& adjName) |
| { |
| return std::find_if(m_adjList.begin(), |
| m_adjList.end(), |
| std::bind(&Adjacent::compare, |
| _1, std::cref(adjName))); |
| } |
| |
| AdjacencyList::iterator |
| AdjacencyList::findAdjacent(uint64_t faceId) |
| { |
| return std::find_if(m_adjList.begin(), |
| m_adjList.end(), |
| std::bind(&Adjacent::compareFaceId, |
| _1, faceId)); |
| } |
| |
| AdjacencyList::iterator |
| AdjacencyList::findAdjacent(const ndn::util::FaceUri& faceUri) |
| { |
| return std::find_if(m_adjList.begin(), |
| m_adjList.end(), |
| std::bind(&Adjacent::compareFaceUri, |
| _1, faceUri)); |
| } |
| |
| uint64_t |
| AdjacencyList::getFaceId(const ndn::util::FaceUri& faceUri) |
| { |
| std::list<Adjacent>::iterator it = std::find_if(m_adjList.begin(), |
| m_adjList.end(), |
| std::bind(&Adjacent::compareFaceUri, |
| _1, faceUri)); |
| if (it != m_adjList.end()) { |
| return it->getFaceId(); |
| } |
| |
| return 0; |
| } |
| |
| void |
| AdjacencyList::writeLog() |
| { |
| NLSR_LOG_DEBUG("-------Adjacency List--------"); |
| for (std::list<Adjacent>::iterator it = m_adjList.begin(); |
| it != m_adjList.end(); it++) { |
| (*it).writeLog(); |
| } |
| } |
| |
| } // namespace nlsr |