blob: a55e9fddd33e3cf80ec2fe7373ddef0c0557f428 [file] [log] [blame]
Alexander Afanasyev78057c32012-07-06 15:18:46 -07001/* -*- Mode: C++; c-file-style: "gnu"; indent-tabs-mode:nil -*- */
2/*
3 * Copyright (c) 2011 University of California, Los Angeles
4 *
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License version 2 as
7 * published by the Free Software Foundation;
8 *
9 * This program is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
13 *
14 * You should have received a copy of the GNU General Public License
15 * along with this program; if not, write to the Free Software
16 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
17 *
18 * Author: Alexander Afanasyev <alexander.afanasyev@ucla.edu>
19 */
20
Alexander Afanasyev4aac5572012-08-09 10:49:55 -070021#include "ndn-fib-impl.h"
Alexander Afanasyev78057c32012-07-06 15:18:46 -070022
Alexander Afanasyev4aac5572012-08-09 10:49:55 -070023#include "ns3/ndn.h"
24#include "ns3/ndn-face.h"
25#include "ns3/ndn-interest-header.h"
Alexander Afanasyev78057c32012-07-06 15:18:46 -070026
27#include "ns3/node.h"
28#include "ns3/assert.h"
29#include "ns3/names.h"
30#include "ns3/log.h"
31
32#include <boost/ref.hpp>
33#include <boost/lambda/lambda.hpp>
34#include <boost/lambda/bind.hpp>
35namespace ll = boost::lambda;
36
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -070037NS_LOG_COMPONENT_DEFINE ("ndn.fib.FibImpl");
Alexander Afanasyev78057c32012-07-06 15:18:46 -070038
39namespace ns3 {
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -070040namespace ndn {
41namespace fib {
Alexander Afanasyev78057c32012-07-06 15:18:46 -070042
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -070043NS_OBJECT_ENSURE_REGISTERED (FibImpl);
Alexander Afanasyev78057c32012-07-06 15:18:46 -070044
45TypeId
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -070046FibImpl::GetTypeId (void)
Alexander Afanasyev78057c32012-07-06 15:18:46 -070047{
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -070048 static TypeId tid = TypeId ("ns3::ndn::fib::Default") // cheating ns3 object system
49 .SetParent<Fib> ()
Alexander Afanasyev4aac5572012-08-09 10:49:55 -070050 .SetGroupName ("Ndn")
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -070051 .AddConstructor<FibImpl> ()
Alexander Afanasyev78057c32012-07-06 15:18:46 -070052 ;
53 return tid;
54}
55
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -070056FibImpl::FibImpl ()
Alexander Afanasyev78057c32012-07-06 15:18:46 -070057{
58}
59
60void
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -070061FibImpl::NotifyNewAggregate ()
Alexander Afanasyev78057c32012-07-06 15:18:46 -070062{
63 Object::NotifyNewAggregate ();
64}
65
66void
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -070067FibImpl::DoDispose (void)
Alexander Afanasyev78057c32012-07-06 15:18:46 -070068{
69 clear ();
70 Object::DoDispose ();
71}
72
73
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -070074Ptr<Entry>
75FibImpl::LongestPrefixMatch (const InterestHeader &interest)
Alexander Afanasyev78057c32012-07-06 15:18:46 -070076{
Alexander Afanasyev30f60e32012-07-10 14:21:16 -070077 super::iterator item = super::longest_prefix_match (interest.GetName ());
Alexander Afanasyev78057c32012-07-06 15:18:46 -070078 // @todo use predicate to search with exclude filters
79
80 if (item == super::end ())
81 return 0;
82 else
83 return item->payload ();
84}
85
86
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -070087Ptr<Entry>
88FibImpl::Add (const NameComponents &prefix, Ptr<Face> face, int32_t metric)
Alexander Afanasyev78057c32012-07-06 15:18:46 -070089{
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -070090 return Add (Create<NameComponents> (prefix), face, metric);
Alexander Afanasyev78057c32012-07-06 15:18:46 -070091}
92
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -070093Ptr<Entry>
94FibImpl::Add (const Ptr<const NameComponents> &prefix, Ptr<Face> face, int32_t metric)
Alexander Afanasyev78057c32012-07-06 15:18:46 -070095{
Alexander Afanasyev1ba09b82012-07-09 09:16:14 -070096 NS_LOG_FUNCTION (this->GetObject<Node> ()->GetId () << boost::cref(*prefix) << boost::cref(*face) << metric);
Alexander Afanasyev78057c32012-07-06 15:18:46 -070097
98 // will add entry if doesn't exists, or just return an iterator to the existing entry
Alexander Afanasyev30f60e32012-07-10 14:21:16 -070099 std::pair< super::iterator, bool > result = super::insert (*prefix, 0);
100 if (result.first != super::end ())
101 {
102 if (result.second)
103 {
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -0700104 Ptr<EntryImpl> newEntry = Create<EntryImpl> (prefix);
Alexander Afanasyev30f60e32012-07-10 14:21:16 -0700105 newEntry->SetTrie (result.first);
106 result.first->set_payload (newEntry);
107 }
Alexander Afanasyev78057c32012-07-06 15:18:46 -0700108
Alexander Afanasyev30f60e32012-07-10 14:21:16 -0700109 super::modify (result.first,
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -0700110 ll::bind (&Entry::AddOrUpdateRoutingMetric, ll::_1, face, metric));
Alexander Afanasyev78057c32012-07-06 15:18:46 -0700111
Alexander Afanasyev30f60e32012-07-10 14:21:16 -0700112 return result.first->payload ();
113 }
114 else
115 return 0;
Alexander Afanasyev78057c32012-07-06 15:18:46 -0700116}
117
118void
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -0700119FibImpl::Remove (const Ptr<const NameComponents> &prefix)
Alexander Afanasyev78057c32012-07-06 15:18:46 -0700120{
121 NS_LOG_FUNCTION (this->GetObject<Node> ()->GetId () << boost::cref(*prefix));
122
123 super::erase (*prefix);
124}
125
Alexander Afanasyev44bb6ea2012-07-09 08:44:41 -0700126// void
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -0700127// FibImpl::Invalidate (const Ptr<const NameComponents> &prefix)
Alexander Afanasyev44bb6ea2012-07-09 08:44:41 -0700128// {
129// NS_LOG_FUNCTION (this->GetObject<Node> ()->GetId () << boost::cref(*prefix));
Alexander Afanasyev78057c32012-07-06 15:18:46 -0700130
Alexander Afanasyev44bb6ea2012-07-09 08:44:41 -0700131// super::iterator foundItem, lastItem;
132// bool reachLast;
133// boost::tie (foundItem, reachLast, lastItem) = super::getTrie ().find (*prefix);
Alexander Afanasyev78057c32012-07-06 15:18:46 -0700134
Alexander Afanasyev44bb6ea2012-07-09 08:44:41 -0700135// if (!reachLast || lastItem->payload () == 0)
136// return; // nothing to invalidate
Alexander Afanasyev78057c32012-07-06 15:18:46 -0700137
Alexander Afanasyev44bb6ea2012-07-09 08:44:41 -0700138// super::modify (lastItem,
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -0700139// ll::bind (&Entry::Invalidate, ll::_1));
Alexander Afanasyev44bb6ea2012-07-09 08:44:41 -0700140// }
Alexander Afanasyev78057c32012-07-06 15:18:46 -0700141
142void
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -0700143FibImpl::InvalidateAll ()
Alexander Afanasyev78057c32012-07-06 15:18:46 -0700144{
Alexander Afanasyev44bb6ea2012-07-09 08:44:41 -0700145 NS_LOG_FUNCTION (this->GetObject<Node> ()->GetId ());
Alexander Afanasyev78057c32012-07-06 15:18:46 -0700146
Alexander Afanasyev44bb6ea2012-07-09 08:44:41 -0700147 super::parent_trie::recursive_iterator item (super::getTrie ());
148 super::parent_trie::recursive_iterator end (0);
149 for (; item != end; item++)
150 {
151 if (item->payload () == 0) continue;
152
153 super::modify (&(*item),
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -0700154 ll::bind (&Entry::Invalidate, ll::_1));
Alexander Afanasyev44bb6ea2012-07-09 08:44:41 -0700155 }
Alexander Afanasyev78057c32012-07-06 15:18:46 -0700156}
157
158void
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -0700159FibImpl::RemoveFace (super::parent_trie &item, Ptr<Face> face)
Alexander Afanasyev78057c32012-07-06 15:18:46 -0700160{
Alexander Afanasyev44bb6ea2012-07-09 08:44:41 -0700161 if (item.payload () == 0) return;
162 NS_LOG_FUNCTION (this);
Alexander Afanasyev78057c32012-07-06 15:18:46 -0700163
Alexander Afanasyev44bb6ea2012-07-09 08:44:41 -0700164 super::modify (&item,
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -0700165 ll::bind (&Entry::RemoveFace, ll::_1, face));
Alexander Afanasyev78057c32012-07-06 15:18:46 -0700166}
167
168void
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -0700169FibImpl::RemoveFromAll (Ptr<Face> face)
Alexander Afanasyev78057c32012-07-06 15:18:46 -0700170{
Alexander Afanasyev44bb6ea2012-07-09 08:44:41 -0700171 NS_LOG_FUNCTION (this);
Alexander Afanasyev78057c32012-07-06 15:18:46 -0700172
Alexander Afanasyev44bb6ea2012-07-09 08:44:41 -0700173 std::for_each (super::parent_trie::recursive_iterator (super::getTrie ()),
174 super::parent_trie::recursive_iterator (0),
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -0700175 ll::bind (&FibImpl::RemoveFace,
Alexander Afanasyev44bb6ea2012-07-09 08:44:41 -0700176 this, ll::_1, face));
177
178 super::parent_trie::recursive_iterator trieNode (super::getTrie ());
179 super::parent_trie::recursive_iterator end (0);
180 for (; trieNode != end; trieNode++)
181 {
182 if (trieNode->payload () == 0) continue;
183
184 if (trieNode->payload ()->m_faces.size () == 0)
185 {
186 trieNode = super::parent_trie::recursive_iterator (trieNode->erase ());
187 }
188 }
Alexander Afanasyev78057c32012-07-06 15:18:46 -0700189}
190
191void
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -0700192FibImpl::Print (std::ostream &os) const
Alexander Afanasyev78057c32012-07-06 15:18:46 -0700193{
Alexander Afanasyev44bb6ea2012-07-09 08:44:41 -0700194 // !!! unordered_set imposes "random" order of item in the same level !!!
195 super::parent_trie::const_recursive_iterator item (super::getTrie ());
196 super::parent_trie::const_recursive_iterator end (0);
197 for (; item != end; item++)
198 {
199 if (item->payload () == 0) continue;
200
201 os << item->payload ()->GetPrefix () << "\t" << *item->payload () << "\n";
202 }
Alexander Afanasyev78057c32012-07-06 15:18:46 -0700203}
204
Alexander Afanasyevf1e013f2012-07-11 17:59:40 -0700205uint32_t
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -0700206FibImpl::GetSize () const
Alexander Afanasyevf1e013f2012-07-11 17:59:40 -0700207{
208 return super::getPolicy ().size ();
209}
210
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -0700211Ptr<const Entry>
212FibImpl::Begin ()
Alexander Afanasyev95a4fa32012-07-09 15:23:59 -0700213{
214 super::parent_trie::const_recursive_iterator item (super::getTrie ());
215 super::parent_trie::const_recursive_iterator end (0);
216 for (; item != end; item++)
217 {
218 if (item->payload () == 0) continue;
219 break;
220 }
221
222 if (item == end)
223 return End ();
224 else
225 return item->payload ();
226}
227
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -0700228Ptr<const Entry>
229FibImpl::End ()
Alexander Afanasyev95a4fa32012-07-09 15:23:59 -0700230{
231 return 0;
232}
233
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -0700234Ptr<const Entry>
235FibImpl::Next (Ptr<const Entry> from)
Alexander Afanasyev95a4fa32012-07-09 15:23:59 -0700236{
237 if (from == 0) return 0;
238
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -0700239 super::parent_trie::const_recursive_iterator item (*StaticCast<const EntryImpl> (from)->to_iterator ());
Alexander Afanasyev95a4fa32012-07-09 15:23:59 -0700240 super::parent_trie::const_recursive_iterator end (0);
241 for (item++; item != end; item++)
242 {
243 if (item->payload () == 0) continue;
244 break;
245 }
246
247 if (item == end)
248 return End ();
249 else
250 return item->payload ();
251}
252
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -0700253} // namespace fib
254} // namespace ndn
Alexander Afanasyev78057c32012-07-06 15:18:46 -0700255} // namespace ns3