A Discrete-Event Network Simulator
API
dsdv-routing-protocol.cc
Go to the documentation of this file.
1 /* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
2 /*
3  * Copyright (c) 2010 Hemanth Narra, Yufei Cheng
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: Hemanth Narra <hemanth@ittc.ku.com>
19  * Author: Yufei Cheng <yfcheng@ittc.ku.edu>
20  *
21  * James P.G. Sterbenz <jpgs@ittc.ku.edu>, director
22  * ResiliNets Research Group http://wiki.ittc.ku.edu/resilinets
23  * Information and Telecommunication Technology Center (ITTC)
24  * and Department of Electrical Engineering and Computer Science
25  * The University of Kansas Lawrence, KS USA.
26  *
27  * Work supported in part by NSF FIND (Future Internet Design) Program
28  * under grant CNS-0626918 (Postmodern Internet Architecture),
29  * NSF grant CNS-1050226 (Multilayer Network Resilience Analysis and Experimentation on GENI),
30  * US Department of Defense (DoD), and ITTC at The University of Kansas.
31  */
32 
33 #include "dsdv-routing-protocol.h"
34 #include "ns3/log.h"
35 #include "ns3/inet-socket-address.h"
36 #include "ns3/trace-source-accessor.h"
37 #include "ns3/udp-socket-factory.h"
38 #include "ns3/boolean.h"
39 #include "ns3/double.h"
40 #include "ns3/uinteger.h"
41 
42 namespace ns3 {
43 
44 NS_LOG_COMPONENT_DEFINE ("DsdvRoutingProtocol");
45 
46 namespace dsdv {
47 
48 NS_OBJECT_ENSURE_REGISTERED (RoutingProtocol);
49 
51 const uint32_t RoutingProtocol::DSDV_PORT = 269;
52 
54 struct DeferredRouteOutputTag : public Tag
55 {
57  int32_t oif;
58 
64  DeferredRouteOutputTag (int32_t o = -1)
65  : Tag (),
66  oif (o)
67  {
68  }
69 
74  static TypeId
76  {
77  static TypeId tid = TypeId ("ns3::dsdv::DeferredRouteOutputTag")
78  .SetParent<Tag> ()
79  .SetGroupName ("Dsdv")
80  .AddConstructor<DeferredRouteOutputTag> ()
81  ;
82  return tid;
83  }
84 
85  TypeId
87  {
88  return GetTypeId ();
89  }
90 
91  uint32_t
93  {
94  return sizeof(int32_t);
95  }
96 
97  void
98  Serialize (TagBuffer i) const
99  {
100  i.WriteU32 (oif);
101  }
102 
103  void
105  {
106  oif = i.ReadU32 ();
107  }
108 
109  void
110  Print (std::ostream &os) const
111  {
112  os << "DeferredRouteOutputTag: output interface = " << oif;
113  }
114 };
115 
116 TypeId
118 {
119  static TypeId tid = TypeId ("ns3::dsdv::RoutingProtocol")
121  .SetGroupName ("Dsdv")
122  .AddConstructor<RoutingProtocol> ()
123  .AddAttribute ("PeriodicUpdateInterval","Periodic interval between exchange of full routing tables among nodes. ",
124  TimeValue (Seconds (15)),
126  MakeTimeChecker ())
127  .AddAttribute ("SettlingTime", "Minimum time an update is to be stored in adv table before sending out"
128  "in case of change in metric (in seconds)",
129  TimeValue (Seconds (5)),
131  MakeTimeChecker ())
132  .AddAttribute ("MaxQueueLen", "Maximum number of packets that we allow a routing protocol to buffer.",
133  UintegerValue (500 /*assuming maximum nodes in simulation is 100*/),
135  MakeUintegerChecker<uint32_t> ())
136  .AddAttribute ("MaxQueuedPacketsPerDst", "Maximum number of packets that we allow per destination to buffer.",
137  UintegerValue (5),
139  MakeUintegerChecker<uint32_t> ())
140  .AddAttribute ("MaxQueueTime","Maximum time packets can be queued (in seconds)",
141  TimeValue (Seconds (30)),
143  MakeTimeChecker ())
144  .AddAttribute ("EnableBuffering","Enables buffering of data packets if no route to destination is available",
145  BooleanValue (true),
149  .AddAttribute ("EnableWST","Enables Weighted Settling Time for the updates before advertising",
150  BooleanValue (true),
154  .AddAttribute ("Holdtimes","Times the forwarding Interval to purge the route.",
155  UintegerValue (3),
157  MakeUintegerChecker<uint32_t> ())
158  .AddAttribute ("WeightedFactor","WeightedFactor for the settling time if Weighted Settling Time is enabled",
159  DoubleValue (0.875),
161  MakeDoubleChecker<double> ())
162  .AddAttribute ("EnableRouteAggregation","Enables Weighted Settling Time for the updates before advertising",
163  BooleanValue (false),
167  .AddAttribute ("RouteAggregationTime","Time to aggregate updates before sending them out (in seconds)",
168  TimeValue (Seconds (1)),
170  MakeTimeChecker ());
171  return tid;
172 }
173 
174 void
176 {
177  EnableBuffering = f;
178 }
179 bool
181 {
182  return EnableBuffering;
183 }
184 void
186 {
187  EnableWST = f;
188 }
189 bool
191 {
192  return EnableWST;
193 }
194 void
196 {
198 }
199 bool
201 {
202  return EnableRouteAggregation;
203 }
204 
205 int64_t
207 {
208  NS_LOG_FUNCTION (this << stream);
210  return 1;
211 }
212 
214  : m_routingTable (),
215  m_advRoutingTable (),
216  m_queue (),
217  m_periodicUpdateTimer (Timer::CANCEL_ON_DESTROY)
218 {
219  m_uniformRandomVariable = CreateObject<UniformRandomVariable> ();
220 }
221 
223 {
224 }
225 
226 void
228 {
229  m_ipv4 = 0;
230  for (std::map<Ptr<Socket>, Ipv4InterfaceAddress>::iterator iter = m_socketAddresses.begin (); iter
231  != m_socketAddresses.end (); iter++)
232  {
233  iter->first->Close ();
234  }
235  m_socketAddresses.clear ();
237 }
238 
239 void
241 {
242  *stream->GetStream () << "Node: " << m_ipv4->GetObject<Node> ()->GetId ()
243  << ", Time: " << Now ().As (unit)
244  << ", Local time: " << GetObject<Node> ()->GetLocalTime ().As (unit)
245  << ", DSDV Routing table" << std::endl;
246 
247  m_routingTable.Print (stream, unit);
248  *stream->GetStream () << std::endl;
249 }
250 
251 void
253 {
263 }
264 
267  const Ipv4Header &header,
268  Ptr<NetDevice> oif,
269  Socket::SocketErrno &sockerr)
270 {
271  NS_LOG_FUNCTION (this << header << (oif ? oif->GetIfIndex () : 0));
272 
273  if (!p)
274  {
275  return LoopbackRoute (header,oif);
276  }
277  if (m_socketAddresses.empty ())
278  {
279  sockerr = Socket::ERROR_NOROUTETOHOST;
280  NS_LOG_LOGIC ("No dsdv interfaces");
281  Ptr<Ipv4Route> route;
282  return route;
283  }
284  std::map<Ipv4Address, RoutingTableEntry> removedAddresses;
285  sockerr = Socket::ERROR_NOTERROR;
286  Ptr<Ipv4Route> route;
287  Ipv4Address dst = header.GetDestination ();
288  NS_LOG_DEBUG ("Packet Size: " << p->GetSize ()
289  << ", Packet id: " << p->GetUid () << ", Destination address in Packet: " << dst);
291  m_routingTable.Purge (removedAddresses);
292  for (std::map<Ipv4Address, RoutingTableEntry>::iterator rmItr = removedAddresses.begin ();
293  rmItr != removedAddresses.end (); ++rmItr)
294  {
295  rmItr->second.SetEntriesChanged (true);
296  rmItr->second.SetSeqNo (rmItr->second.GetSeqNo () + 1);
297  m_advRoutingTable.AddRoute (rmItr->second);
298  }
299  if (!removedAddresses.empty ())
300  {
302  }
303  if (m_routingTable.LookupRoute (dst,rt))
304  {
305  if (EnableBuffering)
306  {
308  }
309  if (rt.GetHop () == 1)
310  {
311  route = rt.GetRoute ();
312  NS_ASSERT (route != 0);
313  NS_LOG_DEBUG ("A route exists from " << route->GetSource ()
314  << " to neighboring destination "
315  << route->GetDestination ());
316  if (oif != 0 && route->GetOutputDevice () != oif)
317  {
318  NS_LOG_DEBUG ("Output device doesn't match. Dropped.");
319  sockerr = Socket::ERROR_NOROUTETOHOST;
320  return Ptr<Ipv4Route> ();
321  }
322  return route;
323  }
324  else
325  {
326  RoutingTableEntry newrt;
327  if (m_routingTable.LookupRoute (rt.GetNextHop (),newrt))
328  {
329  route = newrt.GetRoute ();
330  NS_ASSERT (route != 0);
331  NS_LOG_DEBUG ("A route exists from " << route->GetSource ()
332  << " to destination " << dst << " via "
333  << rt.GetNextHop ());
334  if (oif != 0 && route->GetOutputDevice () != oif)
335  {
336  NS_LOG_DEBUG ("Output device doesn't match. Dropped.");
337  sockerr = Socket::ERROR_NOROUTETOHOST;
338  return Ptr<Ipv4Route> ();
339  }
340  return route;
341  }
342  }
343  }
344 
345  if (EnableBuffering)
346  {
347  uint32_t iif = (oif ? m_ipv4->GetInterfaceForDevice (oif) : -1);
348  DeferredRouteOutputTag tag (iif);
349  if (!p->PeekPacketTag (tag))
350  {
351  p->AddPacketTag (tag);
352  }
353  }
354  return LoopbackRoute (header,oif);
355 }
356 
357 void
359  const Ipv4Header & header,
360  UnicastForwardCallback ucb,
361  ErrorCallback ecb)
362 {
363  NS_LOG_FUNCTION (this << p << header);
364  NS_ASSERT (p != 0 && p != Ptr<Packet> ());
365  QueueEntry newEntry (p,header,ucb,ecb);
366  bool result = m_queue.Enqueue (newEntry);
367  if (result)
368  {
369  NS_LOG_DEBUG ("Added packet " << p->GetUid () << " to queue.");
370  }
371 }
372 
373 bool
375  const Ipv4Header &header,
377  UnicastForwardCallback ucb,
380  ErrorCallback ecb)
381 {
382  NS_LOG_FUNCTION (m_mainAddress << " received packet " << p->GetUid ()
383  << " from " << header.GetSource ()
384  << " on interface " << idev->GetAddress ()
385  << " to destination " << header.GetDestination ());
386  if (m_socketAddresses.empty ())
387  {
388  NS_LOG_DEBUG ("No dsdv interfaces");
389  return false;
390  }
391  NS_ASSERT (m_ipv4 != 0);
392  // Check if input device supports IP
393  NS_ASSERT (m_ipv4->GetInterfaceForDevice (idev) >= 0);
394  int32_t iif = m_ipv4->GetInterfaceForDevice (idev);
395 
396  Ipv4Address dst = header.GetDestination ();
397  Ipv4Address origin = header.GetSource ();
398 
399  // DSDV is not a multicast routing protocol
400  if (dst.IsMulticast ())
401  {
402  return false;
403  }
404 
405  // Deferred route request
406  if (EnableBuffering == true && idev == m_lo)
407  {
409  if (p->PeekPacketTag (tag))
410  {
411  DeferredRouteOutput (p,header,ucb,ecb);
412  return true;
413  }
414  }
415  for (std::map<Ptr<Socket>, Ipv4InterfaceAddress>::const_iterator j =
416  m_socketAddresses.begin (); j != m_socketAddresses.end (); ++j)
417  {
418  Ipv4InterfaceAddress iface = j->second;
419  if (origin == iface.GetLocal ())
420  {
421  return true;
422  }
423  }
424  // LOCAL DELIVARY TO DSDV INTERFACES
425  for (std::map<Ptr<Socket>, Ipv4InterfaceAddress>::const_iterator j = m_socketAddresses.begin (); j
426  != m_socketAddresses.end (); ++j)
427  {
428  Ipv4InterfaceAddress iface = j->second;
429  if (m_ipv4->GetInterfaceForAddress (iface.GetLocal ()) == iif)
430  {
431  if (dst == iface.GetBroadcast () || dst.IsBroadcast ())
432  {
433  Ptr<Packet> packet = p->Copy ();
434  if (lcb.IsNull () == false)
435  {
436  NS_LOG_LOGIC ("Broadcast local delivery to " << iface.GetLocal ());
437  lcb (p, header, iif);
438  // Fall through to additional processing
439  }
440  else
441  {
442  NS_LOG_ERROR ("Unable to deliver packet locally due to null callback " << p->GetUid () << " from " << origin);
443  ecb (p, header, Socket::ERROR_NOROUTETOHOST);
444  }
445  if (header.GetTtl () > 1)
446  {
447  NS_LOG_LOGIC ("Forward broadcast. TTL " << (uint16_t) header.GetTtl ());
448  RoutingTableEntry toBroadcast;
449  if (m_routingTable.LookupRoute (dst,toBroadcast,true))
450  {
451  Ptr<Ipv4Route> route = toBroadcast.GetRoute ();
452  ucb (route,packet,header);
453  }
454  else
455  {
456  NS_LOG_DEBUG ("No route to forward. Drop packet " << p->GetUid ());
457  }
458  }
459  return true;
460  }
461  }
462  }
463 
464  if (m_ipv4->IsDestinationAddress (dst, iif))
465  {
466  if (lcb.IsNull () == false)
467  {
468  NS_LOG_LOGIC ("Unicast local delivery to " << dst);
469  lcb (p, header, iif);
470  }
471  else
472  {
473  NS_LOG_ERROR ("Unable to deliver packet locally due to null callback " << p->GetUid () << " from " << origin);
474  ecb (p, header, Socket::ERROR_NOROUTETOHOST);
475  }
476  return true;
477  }
478 
479  // Check if input device supports IP forwarding
480  if (m_ipv4->IsForwarding (iif) == false)
481  {
482  NS_LOG_LOGIC ("Forwarding disabled for this interface");
483  ecb (p, header, Socket::ERROR_NOROUTETOHOST);
484  return true;
485  }
486 
487  RoutingTableEntry toDst;
488  if (m_routingTable.LookupRoute (dst,toDst))
489  {
491  if (m_routingTable.LookupRoute (toDst.GetNextHop (),ne))
492  {
493  Ptr<Ipv4Route> route = ne.GetRoute ();
494  NS_LOG_LOGIC (m_mainAddress << " is forwarding packet " << p->GetUid ()
495  << " to " << dst
496  << " from " << header.GetSource ()
497  << " via nexthop neighbor " << toDst.GetNextHop ());
498  ucb (route,p,header);
499  return true;
500  }
501  }
502  NS_LOG_LOGIC ("Drop packet " << p->GetUid ()
503  << " as there is no route to forward it.");
504  return false;
505 }
506 
509 {
510  NS_ASSERT (m_lo != 0);
511  Ptr<Ipv4Route> rt = Create<Ipv4Route> ();
512  rt->SetDestination (hdr.GetDestination ());
513  // rt->SetSource (hdr.GetSource ());
514  //
515  // Source address selection here is tricky. The loopback route is
516  // returned when DSDV does not have a route; this causes the packet
517  // to be looped back and handled (cached) in RouteInput() method
518  // while a route is found. However, connection-oriented protocols
519  // like TCP need to create an endpoint four-tuple (src, src port,
520  // dst, dst port) and create a pseudo-header for checksumming. So,
521  // DSDV needs to guess correctly what the eventual source address
522  // will be.
523  //
524  // For single interface, single address nodes, this is not a problem.
525  // When there are possibly multiple outgoing interfaces, the policy
526  // implemented here is to pick the first available DSDV interface.
527  // If RouteOutput() caller specified an outgoing interface, that
528  // further constrains the selection of source address
529  //
530  std::map<Ptr<Socket>, Ipv4InterfaceAddress>::const_iterator j = m_socketAddresses.begin ();
531  if (oif)
532  {
533  // Iterate to find an address on the oif device
534  for (j = m_socketAddresses.begin (); j != m_socketAddresses.end (); ++j)
535  {
536  Ipv4Address addr = j->second.GetLocal ();
537  int32_t interface = m_ipv4->GetInterfaceForAddress (addr);
538  if (oif == m_ipv4->GetNetDevice (static_cast<uint32_t> (interface)))
539  {
540  rt->SetSource (addr);
541  break;
542  }
543  }
544  }
545  else
546  {
547  rt->SetSource (j->second.GetLocal ());
548  }
549  NS_ASSERT_MSG (rt->GetSource () != Ipv4Address (), "Valid DSDV source address not found");
550  rt->SetGateway (Ipv4Address ("127.0.0.1"));
551  rt->SetOutputDevice (m_lo);
552  return rt;
553 }
554 
555 void
557 {
558  Address sourceAddress;
559  Ptr<Packet> advpacket = Create<Packet> ();
560  Ptr<Packet> packet = socket->RecvFrom (sourceAddress);
561  InetSocketAddress inetSourceAddr = InetSocketAddress::ConvertFrom (sourceAddress);
562  Ipv4Address sender = inetSourceAddr.GetIpv4 ();
563  Ipv4Address receiver = m_socketAddresses[socket].GetLocal ();
564  Ptr<NetDevice> dev = m_ipv4->GetNetDevice (m_ipv4->GetInterfaceForAddress (receiver));
565  uint32_t packetSize = packet->GetSize ();
566  NS_LOG_FUNCTION (m_mainAddress << " received dsdv packet of size: " << packetSize
567  << " and packet id: " << packet->GetUid ());
568  uint32_t count = 0;
569  for (; packetSize > 0; packetSize = packetSize - 12)
570  {
571  count = 0;
572  DsdvHeader dsdvHeader, tempDsdvHeader;
573  packet->RemoveHeader (dsdvHeader);
574  NS_LOG_DEBUG ("Processing new update for " << dsdvHeader.GetDst ());
575  /*Verifying if the packets sent by me were returned back to me. If yes, discarding them!*/
576  for (std::map<Ptr<Socket>, Ipv4InterfaceAddress>::const_iterator j = m_socketAddresses.begin (); j
577  != m_socketAddresses.end (); ++j)
578  {
579  Ipv4InterfaceAddress interface = j->second;
580  if (dsdvHeader.GetDst () == interface.GetLocal ())
581  {
582  if (dsdvHeader.GetDstSeqno () % 2 == 1)
583  {
584  NS_LOG_DEBUG ("Sent Dsdv update back to the same Destination, "
585  "with infinite metric. Time left to send fwd update: "
587  count++;
588  }
589  else
590  {
591  NS_LOG_DEBUG ("Received update for my address. Discarding this.");
592  count++;
593  }
594  }
595  }
596  if (count > 0)
597  {
598  continue;
599  }
600  NS_LOG_DEBUG ("Received a DSDV packet from "
601  << sender << " to " << receiver << ". Details are: Destination: " << dsdvHeader.GetDst () << ", Seq No: "
602  << dsdvHeader.GetDstSeqno () << ", HopCount: " << dsdvHeader.GetHopCount ());
603  RoutingTableEntry fwdTableEntry, advTableEntry;
604  EventId event;
605  bool permanentTableVerifier = m_routingTable.LookupRoute (dsdvHeader.GetDst (),fwdTableEntry);
606  if (permanentTableVerifier == false)
607  {
608  if (dsdvHeader.GetDstSeqno () % 2 != 1)
609  {
610  NS_LOG_DEBUG ("Received New Route!");
611  RoutingTableEntry newEntry (
612  /*device=*/ dev, /*dst=*/
613  dsdvHeader.GetDst (), /*seqno=*/
614  dsdvHeader.GetDstSeqno (),
615  /*iface=*/ m_ipv4->GetAddress (m_ipv4->GetInterfaceForAddress (receiver), 0),
616  /*hops=*/ dsdvHeader.GetHopCount (), /*next hop=*/
617  sender, /*lifetime=*/
618  Simulator::Now (), /*settlingTime*/
619  m_settlingTime, /*entries changed*/
620  true);
621  newEntry.SetFlag (VALID);
622  m_routingTable.AddRoute (newEntry);
623  NS_LOG_DEBUG ("New Route added to both tables");
624  m_advRoutingTable.AddRoute (newEntry);
625  }
626  else
627  {
628  // received update not present in main routing table and also with infinite metric
629  NS_LOG_DEBUG ("Discarding this update as this route is not present in "
630  "main routing table and received with infinite metric");
631  }
632  }
633  else
634  {
635  if (!m_advRoutingTable.LookupRoute (dsdvHeader.GetDst (),advTableEntry))
636  {
638  std::map<Ipv4Address, RoutingTableEntry> allRoutes;
640  for (std::map<Ipv4Address, RoutingTableEntry>::const_iterator i = allRoutes.begin (); i != allRoutes.end (); ++i)
641  {
642  NS_LOG_DEBUG ("ADV table routes are:" << i->second.GetDestination ());
643  }
644  // present in fwd table and not in advtable
645  m_advRoutingTable.AddRoute (fwdTableEntry);
646  m_advRoutingTable.LookupRoute (dsdvHeader.GetDst (),advTableEntry);
647  }
648  if (dsdvHeader.GetDstSeqno () % 2 != 1)
649  {
650  if (dsdvHeader.GetDstSeqno () > advTableEntry.GetSeqNo ())
651  {
652  // Received update with better seq number. Clear any old events that are running
653  if (m_advRoutingTable.ForceDeleteIpv4Event (dsdvHeader.GetDst ()))
654  {
655  NS_LOG_DEBUG ("Canceling the timer to update route with better seq number");
656  }
657  // if its a changed metric *nomatter* where the update came from, wait for WST
658  if (dsdvHeader.GetHopCount () != advTableEntry.GetHop ())
659  {
660  advTableEntry.SetSeqNo (dsdvHeader.GetDstSeqno ());
661  advTableEntry.SetLifeTime (Simulator::Now ());
662  advTableEntry.SetFlag (VALID);
663  advTableEntry.SetEntriesChanged (true);
664  advTableEntry.SetNextHop (sender);
665  advTableEntry.SetHop (dsdvHeader.GetHopCount ());
666  NS_LOG_DEBUG ("Received update with better sequence number and changed metric.Waiting for WST");
667  Time tempSettlingtime = GetSettlingTime (dsdvHeader.GetDst ());
668  advTableEntry.SetSettlingTime (tempSettlingtime);
669  NS_LOG_DEBUG ("Added Settling Time:" << tempSettlingtime.As (Time::S)
670  << " as there is no event running for this route");
671  event = Simulator::Schedule (tempSettlingtime,&RoutingProtocol::SendTriggeredUpdate,this);
672  m_advRoutingTable.AddIpv4Event (dsdvHeader.GetDst (),event);
673  NS_LOG_DEBUG ("EventCreated EventUID: " << event.GetUid ());
674  // if received changed metric, use it but adv it only after wst
675  m_routingTable.Update (advTableEntry);
676  m_advRoutingTable.Update (advTableEntry);
677  }
678  else
679  {
680  // Received update with better seq number and same metric.
681  advTableEntry.SetSeqNo (dsdvHeader.GetDstSeqno ());
682  advTableEntry.SetLifeTime (Simulator::Now ());
683  advTableEntry.SetFlag (VALID);
684  advTableEntry.SetEntriesChanged (true);
685  advTableEntry.SetNextHop (sender);
686  advTableEntry.SetHop (dsdvHeader.GetHopCount ());
687  m_advRoutingTable.Update (advTableEntry);
688  NS_LOG_DEBUG ("Route with better sequence number and same metric received. Advertised without WST");
689  }
690  }
691  else if (dsdvHeader.GetDstSeqno () == advTableEntry.GetSeqNo ())
692  {
693  if (dsdvHeader.GetHopCount () < advTableEntry.GetHop ())
694  {
695  /*Received update with same seq number and better hop count.
696  * As the metric is changed, we will have to wait for WST before sending out this update.
697  */
698  NS_LOG_DEBUG ("Canceling any existing timer to update route with same sequence number "
699  "and better hop count");
701  advTableEntry.SetSeqNo (dsdvHeader.GetDstSeqno ());
702  advTableEntry.SetLifeTime (Simulator::Now ());
703  advTableEntry.SetFlag (VALID);
704  advTableEntry.SetEntriesChanged (true);
705  advTableEntry.SetNextHop (sender);
706  advTableEntry.SetHop (dsdvHeader.GetHopCount ());
707  Time tempSettlingtime = GetSettlingTime (dsdvHeader.GetDst ());
708  advTableEntry.SetSettlingTime (tempSettlingtime);
709  NS_LOG_DEBUG ("Added Settling Time," << tempSettlingtime.As (Time::S)
710  << " as there is no current event running for this route");
711  event = Simulator::Schedule (tempSettlingtime,&RoutingProtocol::SendTriggeredUpdate,this);
712  m_advRoutingTable.AddIpv4Event (dsdvHeader.GetDst (),event);
713  NS_LOG_DEBUG ("EventCreated EventUID: " << event.GetUid ());
714  // if received changed metric, use it but adv it only after wst
715  m_routingTable.Update (advTableEntry);
716  m_advRoutingTable.Update (advTableEntry);
717  }
718  else
719  {
720  /*Received update with same seq number but with same or greater hop count.
721  * Discard that update.
722  */
723  if (!m_advRoutingTable.AnyRunningEvent (dsdvHeader.GetDst ()))
724  {
725  /*update the timer only if nexthop address matches thus discarding
726  * updates to that destination from other nodes.
727  */
728  if (advTableEntry.GetNextHop () == sender)
729  {
730  advTableEntry.SetLifeTime (Simulator::Now ());
731  m_routingTable.Update (advTableEntry);
732  }
734  dsdvHeader.GetDst ());
735  }
736  NS_LOG_DEBUG ("Received update with same seq number and "
737  "same/worst metric for, " << dsdvHeader.GetDst () << ". Discarding the update.");
738  }
739  }
740  else
741  {
742  // Received update with an old sequence number. Discard the update
743  if (!m_advRoutingTable.AnyRunningEvent (dsdvHeader.GetDst ()))
744  {
745  m_advRoutingTable.DeleteRoute (dsdvHeader.GetDst ());
746  }
747  NS_LOG_DEBUG (dsdvHeader.GetDst () << " : Received update with old seq number. Discarding the update.");
748  }
749  }
750  else
751  {
752  NS_LOG_DEBUG ("Route with infinite metric received for "
753  << dsdvHeader.GetDst () << " from " << sender);
754  // Delete route only if update was received from my nexthop neighbor
755  if (sender == advTableEntry.GetNextHop ())
756  {
757  NS_LOG_DEBUG ("Triggering an update for this unreachable route:");
758  std::map<Ipv4Address, RoutingTableEntry> dstsWithNextHopSrc;
759  m_routingTable.GetListOfDestinationWithNextHop (dsdvHeader.GetDst (),dstsWithNextHopSrc);
760  m_routingTable.DeleteRoute (dsdvHeader.GetDst ());
761  advTableEntry.SetSeqNo (dsdvHeader.GetDstSeqno ());
762  advTableEntry.SetEntriesChanged (true);
763  m_advRoutingTable.Update (advTableEntry);
764  for (std::map<Ipv4Address, RoutingTableEntry>::iterator i = dstsWithNextHopSrc.begin (); i
765  != dstsWithNextHopSrc.end (); ++i)
766  {
767  i->second.SetSeqNo (i->second.GetSeqNo () + 1);
768  i->second.SetEntriesChanged (true);
769  m_advRoutingTable.AddRoute (i->second);
770  m_routingTable.DeleteRoute (i->second.GetDestination ());
771  }
772  }
773  else
774  {
775  if (!m_advRoutingTable.AnyRunningEvent (dsdvHeader.GetDst ()))
776  {
777  m_advRoutingTable.DeleteRoute (dsdvHeader.GetDst ());
778  }
779  NS_LOG_DEBUG (dsdvHeader.GetDst () <<
780  " : Discard this link break update as it was received from a different neighbor "
781  "and I can reach the destination");
782  }
783  }
784  }
785  }
786  std::map<Ipv4Address, RoutingTableEntry> allRoutes;
788  if (EnableRouteAggregation && allRoutes.size () > 0)
789  {
791  }
792  else
793  {
795  }
796 }
797 
798 
799 void
801 {
802  NS_LOG_FUNCTION (m_mainAddress << " is sending a triggered update");
803  std::map<Ipv4Address, RoutingTableEntry> allRoutes;
805  for (std::map<Ptr<Socket>, Ipv4InterfaceAddress>::const_iterator j = m_socketAddresses.begin (); j
806  != m_socketAddresses.end (); ++j)
807  {
808  DsdvHeader dsdvHeader;
809  Ptr<Socket> socket = j->first;
810  Ipv4InterfaceAddress iface = j->second;
811  Ptr<Packet> packet = Create<Packet> ();
812  for (std::map<Ipv4Address, RoutingTableEntry>::const_iterator i = allRoutes.begin (); i != allRoutes.end (); ++i)
813  {
814  NS_LOG_LOGIC ("Destination: " << i->second.GetDestination ()
815  << " SeqNo:" << i->second.GetSeqNo () << " HopCount:"
816  << i->second.GetHop () + 1);
817  RoutingTableEntry temp = i->second;
818  if ((i->second.GetEntriesChanged () == true) && (!m_advRoutingTable.AnyRunningEvent (temp.GetDestination ())))
819  {
820  dsdvHeader.SetDst (i->second.GetDestination ());
821  dsdvHeader.SetDstSeqno (i->second.GetSeqNo ());
822  dsdvHeader.SetHopCount (i->second.GetHop () + 1);
823  temp.SetFlag (VALID);
824  temp.SetEntriesChanged (false);
825  m_advRoutingTable.DeleteIpv4Event (temp.GetDestination ());
826  if (!(temp.GetSeqNo () % 2))
827  {
828  m_routingTable.Update (temp);
829  }
830  packet->AddHeader (dsdvHeader);
831  m_advRoutingTable.DeleteRoute (temp.GetDestination ());
832  NS_LOG_DEBUG ("Deleted this route from the advertised table");
833  }
834  else
835  {
836  EventId event = m_advRoutingTable.GetEventId (temp.GetDestination ());
837  NS_ASSERT (event.GetUid () != 0);
838  NS_LOG_DEBUG ("EventID " << event.GetUid () << " associated with "
839  << temp.GetDestination () << " has not expired, waiting in adv table");
840  }
841  }
842  if (packet->GetSize () >= 12)
843  {
844  RoutingTableEntry temp2;
845  m_routingTable.LookupRoute (m_ipv4->GetAddress (1, 0).GetBroadcast (), temp2);
846  dsdvHeader.SetDst (m_ipv4->GetAddress (1, 0).GetLocal ());
847  dsdvHeader.SetDstSeqno (temp2.GetSeqNo ());
848  dsdvHeader.SetHopCount (temp2.GetHop () + 1);
849  NS_LOG_DEBUG ("Adding my update as well to the packet");
850  packet->AddHeader (dsdvHeader);
851  // Send to all-hosts broadcast if on /32 addr, subnet-directed otherwise
852  Ipv4Address destination;
853  if (iface.GetMask () == Ipv4Mask::GetOnes ())
854  {
855  destination = Ipv4Address ("255.255.255.255");
856  }
857  else
858  {
859  destination = iface.GetBroadcast ();
860  }
861  socket->SendTo (packet, 0, InetSocketAddress (destination, DSDV_PORT));
862  NS_LOG_FUNCTION ("Sent Triggered Update from "
863  << dsdvHeader.GetDst ()
864  << " with packet id : " << packet->GetUid () << " and packet Size: " << packet->GetSize ());
865  }
866  else
867  {
868  NS_LOG_FUNCTION ("Update not sent as there are no updates to be triggered");
869  }
870  }
871 }
872 
873 void
875 {
876  std::map<Ipv4Address, RoutingTableEntry> removedAddresses, allRoutes;
877  m_routingTable.Purge (removedAddresses);
880  if (allRoutes.empty ())
881  {
882  return;
883  }
884  NS_LOG_FUNCTION (m_mainAddress << " is sending out its periodic update");
885  for (std::map<Ptr<Socket>, Ipv4InterfaceAddress>::const_iterator j = m_socketAddresses.begin (); j
886  != m_socketAddresses.end (); ++j)
887  {
888  Ptr<Socket> socket = j->first;
889  Ipv4InterfaceAddress iface = j->second;
890  Ptr<Packet> packet = Create<Packet> ();
891  for (std::map<Ipv4Address, RoutingTableEntry>::const_iterator i = allRoutes.begin (); i != allRoutes.end (); ++i)
892  {
893  DsdvHeader dsdvHeader;
894  if (i->second.GetHop () == 0)
895  {
896  RoutingTableEntry ownEntry;
897  dsdvHeader.SetDst (m_ipv4->GetAddress (1,0).GetLocal ());
898  dsdvHeader.SetDstSeqno (i->second.GetSeqNo () + 2);
899  dsdvHeader.SetHopCount (i->second.GetHop () + 1);
900  m_routingTable.LookupRoute (m_ipv4->GetAddress (1,0).GetBroadcast (),ownEntry);
901  ownEntry.SetSeqNo (dsdvHeader.GetDstSeqno ());
902  m_routingTable.Update (ownEntry);
903  packet->AddHeader (dsdvHeader);
904  }
905  else
906  {
907  dsdvHeader.SetDst (i->second.GetDestination ());
908  dsdvHeader.SetDstSeqno ((i->second.GetSeqNo ()));
909  dsdvHeader.SetHopCount (i->second.GetHop () + 1);
910  packet->AddHeader (dsdvHeader);
911  }
912  NS_LOG_DEBUG ("Forwarding the update for " << i->first);
913  NS_LOG_DEBUG ("Forwarding details are, Destination: " << dsdvHeader.GetDst ()
914  << ", SeqNo:" << dsdvHeader.GetDstSeqno ()
915  << ", HopCount:" << dsdvHeader.GetHopCount ()
916  << ", LifeTime: " << i->second.GetLifeTime ().As (Time::S));
917  }
918  for (std::map<Ipv4Address, RoutingTableEntry>::const_iterator rmItr = removedAddresses.begin (); rmItr
919  != removedAddresses.end (); ++rmItr)
920  {
921  DsdvHeader removedHeader;
922  removedHeader.SetDst (rmItr->second.GetDestination ());
923  removedHeader.SetDstSeqno (rmItr->second.GetSeqNo () + 1);
924  removedHeader.SetHopCount (rmItr->second.GetHop () + 1);
925  packet->AddHeader (removedHeader);
926  NS_LOG_DEBUG ("Update for removed record is: Destination: " << removedHeader.GetDst ()
927  << " SeqNo:" << removedHeader.GetDstSeqno ()
928  << " HopCount:" << removedHeader.GetHopCount ());
929  }
930  socket->Send (packet);
931  // Send to all-hosts broadcast if on /32 addr, subnet-directed otherwise
932  Ipv4Address destination;
933  if (iface.GetMask () == Ipv4Mask::GetOnes ())
934  {
935  destination = Ipv4Address ("255.255.255.255");
936  }
937  else
938  {
939  destination = iface.GetBroadcast ();
940  }
941  socket->SendTo (packet, 0, InetSocketAddress (destination, DSDV_PORT));
942  NS_LOG_FUNCTION ("PeriodicUpdate Packet UID is : " << packet->GetUid ());
943  }
945 }
946 
947 void
949 {
950  NS_ASSERT (ipv4 != 0);
951  NS_ASSERT (m_ipv4 == 0);
952  m_ipv4 = ipv4;
953  // Create lo route. It is asserted that the only one interface up for now is loopback
954  NS_ASSERT (m_ipv4->GetNInterfaces () == 1 && m_ipv4->GetAddress (0, 0).GetLocal () == Ipv4Address ("127.0.0.1"));
955  m_lo = m_ipv4->GetNetDevice (0);
956  NS_ASSERT (m_lo != 0);
957  // Remember lo route
958  RoutingTableEntry rt (
959  /*device=*/ m_lo, /*dst=*/
960  Ipv4Address::GetLoopback (), /*seqno=*/
961  0,
962  /*iface=*/ Ipv4InterfaceAddress (Ipv4Address::GetLoopback (),Ipv4Mask ("255.0.0.0")),
963  /*hops=*/ 0, /*next hop=*/
965  /*lifetime=*/ Simulator::GetMaximumSimulationTime ());
966  rt.SetFlag (INVALID);
967  rt.SetEntriesChanged (false);
970 }
971 
972 void
974 {
975  NS_LOG_FUNCTION (this << m_ipv4->GetAddress (i, 0).GetLocal ()
976  << " interface is up");
977  Ptr<Ipv4L3Protocol> l3 = m_ipv4->GetObject<Ipv4L3Protocol> ();
978  Ipv4InterfaceAddress iface = l3->GetAddress (i,0);
979  if (iface.GetLocal () == Ipv4Address ("127.0.0.1"))
980  {
981  return;
982  }
983  // Create a socket to listen only on this interface
984  Ptr<Socket> socket = Socket::CreateSocket (GetObject<Node> (),UdpSocketFactory::GetTypeId ());
985  NS_ASSERT (socket != 0);
987  socket->BindToNetDevice (l3->GetNetDevice (i));
989  socket->SetAllowBroadcast (true);
990  socket->SetAttribute ("IpTtl",UintegerValue (1));
991  m_socketAddresses.insert (std::make_pair (socket,iface));
992  // Add local broadcast record to the routing table
993  Ptr<NetDevice> dev = m_ipv4->GetNetDevice (m_ipv4->GetInterfaceForAddress (iface.GetLocal ()));
994  RoutingTableEntry rt (/*device=*/ dev, /*dst=*/ iface.GetBroadcast (), /*seqno=*/ 0,/*iface=*/ iface,/*hops=*/ 0,
995  /*next hop=*/ iface.GetBroadcast (), /*lifetime=*/ Simulator::GetMaximumSimulationTime ());
997  if (m_mainAddress == Ipv4Address ())
998  {
999  m_mainAddress = iface.GetLocal ();
1000  }
1002 }
1003 
1004 void
1006 {
1007  Ptr<Ipv4L3Protocol> l3 = m_ipv4->GetObject<Ipv4L3Protocol> ();
1008  Ptr<NetDevice> dev = l3->GetNetDevice (i);
1009  Ptr<Socket> socket = FindSocketWithInterfaceAddress (m_ipv4->GetAddress (i,0));
1010  NS_ASSERT (socket);
1011  socket->Close ();
1012  m_socketAddresses.erase (socket);
1013  if (m_socketAddresses.empty ())
1014  {
1015  NS_LOG_LOGIC ("No dsdv interfaces");
1016  m_routingTable.Clear ();
1017  return;
1018  }
1021 }
1022 
1023 void
1026 {
1027  NS_LOG_FUNCTION (this << " interface " << i << " address " << address);
1028  Ptr<Ipv4L3Protocol> l3 = m_ipv4->GetObject<Ipv4L3Protocol> ();
1029  if (!l3->IsUp (i))
1030  {
1031  return;
1032  }
1033  Ipv4InterfaceAddress iface = l3->GetAddress (i,0);
1035  if (!socket)
1036  {
1037  if (iface.GetLocal () == Ipv4Address ("127.0.0.1"))
1038  {
1039  return;
1040  }
1041  Ptr<Socket> socket = Socket::CreateSocket (GetObject<Node> (),UdpSocketFactory::GetTypeId ());
1042  NS_ASSERT (socket != 0);
1044  // Bind to any IP address so that broadcasts can be received
1045  socket->BindToNetDevice (l3->GetNetDevice (i));
1047  socket->SetAllowBroadcast (true);
1048  m_socketAddresses.insert (std::make_pair (socket,iface));
1049  Ptr<NetDevice> dev = m_ipv4->GetNetDevice (m_ipv4->GetInterfaceForAddress (iface.GetLocal ()));
1050  RoutingTableEntry rt (/*device=*/ dev, /*dst=*/ iface.GetBroadcast (),/*seqno=*/ 0, /*iface=*/ iface,/*hops=*/ 0,
1051  /*next hop=*/ iface.GetBroadcast (), /*lifetime=*/ Simulator::GetMaximumSimulationTime ());
1052  m_routingTable.AddRoute (rt);
1053  }
1054 }
1055 
1056 void
1059 {
1061  if (socket)
1062  {
1063  m_socketAddresses.erase (socket);
1064  Ptr<Ipv4L3Protocol> l3 = m_ipv4->GetObject<Ipv4L3Protocol> ();
1065  if (l3->GetNAddresses (i))
1066  {
1067  Ipv4InterfaceAddress iface = l3->GetAddress (i,0);
1068  // Create a socket to listen only on this interface
1069  Ptr<Socket> socket = Socket::CreateSocket (GetObject<Node> (),UdpSocketFactory::GetTypeId ());
1070  NS_ASSERT (socket != 0);
1072  // Bind to any IP address so that broadcasts can be received
1074  socket->SetAllowBroadcast (true);
1075  m_socketAddresses.insert (std::make_pair (socket,iface));
1076  }
1077  }
1078 }
1079 
1082 {
1083  for (std::map<Ptr<Socket>, Ipv4InterfaceAddress>::const_iterator j = m_socketAddresses.begin (); j
1084  != m_socketAddresses.end (); ++j)
1085  {
1086  Ptr<Socket> socket = j->first;
1087  Ipv4InterfaceAddress iface = j->second;
1088  if (iface == addr)
1089  {
1090  return socket;
1091  }
1092  }
1093  Ptr<Socket> socket;
1094  return socket;
1095 }
1096 
1097 void
1099  Ptr<const Packet> packet,
1100  const Ipv4Header & header)
1101 {
1102  Ptr<Ipv4L3Protocol> l3 = m_ipv4->GetObject<Ipv4L3Protocol> ();
1103  NS_ASSERT (l3 != 0);
1104  Ptr<Packet> p = packet->Copy ();
1105  l3->Send (p,route->GetSource (),header.GetDestination (),header.GetProtocol (),route);
1106 }
1107 
1108 void
1110  const Ipv4Header & header,
1111  Socket::SocketErrno err)
1112 {
1113  NS_LOG_DEBUG (m_mainAddress << " drop packet " << packet->GetUid () << " to "
1114  << header.GetDestination () << " from queue. Error " << err);
1115 }
1116 
1117 void
1119 {
1120  NS_LOG_FUNCTION (this);
1121  Ptr<Ipv4Route> route;
1122  std::map<Ipv4Address, RoutingTableEntry> allRoutes;
1123  m_routingTable.GetListOfAllRoutes (allRoutes);
1124  for (std::map<Ipv4Address, RoutingTableEntry>::const_iterator i = allRoutes.begin (); i != allRoutes.end (); ++i)
1125  {
1126  RoutingTableEntry rt;
1127  rt = i->second;
1128  if (m_queue.Find (rt.GetDestination ()))
1129  {
1130  if (rt.GetHop () == 1)
1131  {
1132  route = rt.GetRoute ();
1133  NS_LOG_LOGIC ("A route exists from " << route->GetSource ()
1134  << " to neighboring destination "
1135  << route->GetDestination ());
1136  NS_ASSERT (route != 0);
1137  }
1138  else
1139  {
1140  RoutingTableEntry newrt;
1141  m_routingTable.LookupRoute (rt.GetNextHop (),newrt);
1142  route = newrt.GetRoute ();
1143  NS_LOG_LOGIC ("A route exists from " << route->GetSource ()
1144  << " to destination " << route->GetDestination () << " via "
1145  << rt.GetNextHop ());
1146  NS_ASSERT (route != 0);
1147  }
1148  SendPacketFromQueue (rt.GetDestination (),route);
1149  }
1150  }
1151 }
1152 
1153 void
1155  Ptr<Ipv4Route> route)
1156 {
1157  NS_LOG_DEBUG (m_mainAddress << " is sending a queued packet to destination " << dst);
1158  QueueEntry queueEntry;
1159  if (m_queue.Dequeue (dst,queueEntry))
1160  {
1162  Ptr<Packet> p = ConstCast<Packet> (queueEntry.GetPacket ());
1163  if (p->RemovePacketTag (tag))
1164  {
1165  if (tag.oif != -1 && tag.oif != m_ipv4->GetInterfaceForDevice (route->GetOutputDevice ()))
1166  {
1167  NS_LOG_DEBUG ("Output device doesn't match. Dropped.");
1168  return;
1169  }
1170  }
1172  Ipv4Header header = queueEntry.GetIpv4Header ();
1173  header.SetSource (route->GetSource ());
1174  header.SetTtl (header.GetTtl () + 1); // compensate extra TTL decrement by fake loopback routing
1175  ucb (route,p,header);
1176  if (m_queue.GetSize () != 0 && m_queue.Find (dst))
1177  {
1179  &RoutingProtocol::SendPacketFromQueue,this,dst,route);
1180  }
1181  }
1182 }
1183 
1184 Time
1186 {
1187  NS_LOG_FUNCTION ("Calculating the settling time for " << address);
1188  RoutingTableEntry mainrt;
1189  Time weightedTime;
1191  if (EnableWST)
1192  {
1193  if (mainrt.GetSettlingTime () == Seconds (0))
1194  {
1195  return Seconds (0);
1196  }
1197  else
1198  {
1199  NS_LOG_DEBUG ("Route SettlingTime: " << mainrt.GetSettlingTime ().As (Time::S)
1200  << " and LifeTime:" << mainrt.GetLifeTime ().As (Time::S));
1201  weightedTime = m_weightedFactor * mainrt.GetSettlingTime () +
1202  (1.0 - m_weightedFactor) * mainrt.GetLifeTime ();
1203  NS_LOG_DEBUG ("Calculated weightedTime:" << weightedTime.As (Time::S));
1204  return weightedTime;
1205  }
1206  }
1207  return mainrt.GetSettlingTime ();
1208 }
1209 
1210 void
1212 {
1213  NS_LOG_FUNCTION ("Merging advertised table changes with main table before sending out periodic update");
1214  std::map<Ipv4Address, RoutingTableEntry> allRoutes;
1216  if (allRoutes.size () > 0)
1217  {
1218  for (std::map<Ipv4Address, RoutingTableEntry>::const_iterator i = allRoutes.begin (); i != allRoutes.end (); ++i)
1219  {
1220  RoutingTableEntry advEntry = i->second;
1221  if ((advEntry.GetEntriesChanged () == true) && (!m_advRoutingTable.AnyRunningEvent (advEntry.GetDestination ())))
1222  {
1223  if (!(advEntry.GetSeqNo () % 2))
1224  {
1225  advEntry.SetFlag (VALID);
1226  advEntry.SetEntriesChanged (false);
1227  m_routingTable.Update (advEntry);
1228  NS_LOG_DEBUG ("Merged update for " << advEntry.GetDestination () << " with main routing Table");
1229  }
1231  }
1232  else
1233  {
1234  NS_LOG_DEBUG ("Event currently running. Cannot Merge Routing Tables");
1235  }
1236  }
1237  }
1238 }
1239 }
1240 }
void SetSource(Ipv4Address source)
Definition: ipv4-header.cc:285
void Start()
Start protocol operation.
static TypeId GetTypeId(void)
Get the type ID.
Ptr< Ipv4Route > LoopbackRoute(const Ipv4Header &header, Ptr< NetDevice > oif) const
Create loopback route for given header.
static EventId Schedule(Time const &delay, FUNC f, Ts &&... args)
Schedule an event to expire after delay.
Definition: simulator.h:557
static const uint32_t DSDV_PORT
UDP Port for DSDV control traffic.
static Ipv4Mask GetOnes(void)
uint64_t GetUid(void) const
Returns the packet&#39;s Uid.
Definition: packet.cc:390
Simulation virtual time values and global simulation resolution.
Definition: nstime.h:103
int64_t AssignStreams(int64_t stream)
Assign a fixed random variable stream number to the random variables used by this model...
Routing table entry.
Definition: dsdv-rtable.h:56
an Inet address class
static Ipv4Address GetAny(void)
void SetEntriesChanged(bool entriesChanged)
Set entries changed indicator.
Definition: dsdv-rtable.h:253
#define NS_LOG_FUNCTION(parameters)
If log level LOG_FUNCTION is enabled, this macro will output all input parameters separated by "...
std::map< Ptr< Socket >, Ipv4InterfaceAddress > m_socketAddresses
Raw socket per each IP interface, map socket -> iface address (IP + mask)
void SetStream(int64_t stream)
Specifies the stream number for the RngStream.
uint32_t GetHop() const
Get hop.
Definition: dsdv-rtable.h:190
AttributeValue implementation for Boolean.
Definition: boolean.h:36
Callback template class.
Definition: callback.h:1278
Definition: second.py:1
bool EnableRouteAggregation
This is a flag to enable route aggregation.
bool IsBroadcast(void) const
uint32_t GetInteger(uint32_t min, uint32_t max)
Get the next random value, as an unsigned integer in the specified range .
void MergeTriggerPeriodicUpdates()
Merge periodic updates.
A simple virtual Timer class.
Definition: timer.h:73
Time m_periodicUpdateInterval
PeriodicUpdateInterval specifies the periodic time interval between which the a node broadcasts its e...
#define NS_OBJECT_ENSURE_REGISTERED(type)
Register an Object subclass with the TypeId system.
Definition: object-base.h:45
bool EnableBuffering
Flag that is used to enable or disable buffering.
bool AnyRunningEvent(Ipv4Address address)
Force delete an update waiting for settling time to complete as a better update to same destination w...
Definition: dsdv-rtable.cc:272
void SetDst(Ipv4Address destination)
Set destination address.
Definition: dsdv-packet.h:86
uint32_t GetSize(void) const
Returns the the size in bytes of the packet (including the zero-filled initial payload).
Definition: packet.h:852
virtual bool SetAllowBroadcast(bool allowBroadcast)=0
Configure whether broadcast datagram transmissions are allowed.
ErrorCallback m_ecb
Error callback for own packets.
a class to represent an Ipv4 address mask
Definition: ipv4-address.h:269
static const uint32_t packetSize
Ptr< const AttributeAccessor > MakeBooleanAccessor(T1 a1)
Create an AttributeAccessor for a class data member, or a lone class get functor or set method...
Definition: boolean.h:85
virtual void SetIpv4(Ptr< Ipv4 > ipv4)
static TypeId GetTypeId()
Get the type ID.
Ptr< const Packet > GetPacket() const
Get packet.
EventId GetEventId(Ipv4Address address)
Get the EcentId associated with that address.
Definition: dsdv-rtable.cc:338
Ipv4Address GetSource(void) const
Definition: ipv4-header.cc:291
bool EnableWST
Flag that is used to enable or disable Weighted Settling Time.
#define NS_ASSERT(condition)
At runtime, in debugging builds, if this condition is not true, the program prints the source file...
Definition: assert.h:67
uint32_t GetSeqNo() const
Get sequence number.
Definition: dsdv-rtable.h:172
Time GetDelayLeft(void) const
Definition: timer.cc:87
void SetFlag(RouteFlags flag)
Set route flags.
Definition: dsdv-rtable.h:235
#define NS_LOG_COMPONENT_DEFINE(name)
Define a Log component with a specific name.
Definition: log.h:205
void SetDstSeqno(uint32_t sequenceNumber)
Set destination sequence number.
Definition: dsdv-packet.h:122
Time MilliSeconds(uint64_t value)
Construct a Time in the indicated unit.
Definition: nstime.h:1286
void SetMaxPacketsPerDst(uint32_t len)
Set maximum packets per destination.
virtual void DoDispose()
Destructor implementation.
virtual void NotifyInterfaceDown(uint32_t interface)
void GetListOfAllRoutes(std::map< Ipv4Address, RoutingTableEntry > &allRoutes)
Lookup list of all addresses in the routing table.
Definition: dsdv-rtable.cc:170
double m_weightedFactor
This is the wighted factor to determine the weighted settling time.
uint32_t GetDstSeqno() const
Get destination sequence number.
Definition: dsdv-packet.h:131
virtual void DoDispose(void)
Destructor implementation.
Definition: object.cc:346
Ipv4Address GetDst() const
Get destination address.
Definition: dsdv-packet.h:95
TAG_BUFFER_INLINE uint32_t ReadU32(void)
Definition: tag-buffer.h:215
TimeWithUnit As(const enum Unit unit) const
Attach a unit to a Time, to facilitate output in a specific unit.
Definition: time.cc:389
void Print(std::ostream &os) const
SocketErrno
Enumeration of the possible errors returned by a socket.
Definition: socket.h:82
void SendPacketFromQueue(Ipv4Address dst, Ptr< Ipv4Route > route)
Send packet from queue.
int32_t oif
Positive if output device is fixed in RouteOutput.
bool Update(RoutingTableEntry &rt)
Updating the routing Table with routing table entry rt.
Definition: dsdv-rtable.cc:136
DSDV routing protocol.
virtual void PrintRoutingTable(Ptr< OutputStreamWrapper > stream, Time::Unit unit=Time::S) const
Print the Routing Table entries.
bool GetEntriesChanged() const
Get entries changed.
Definition: dsdv-rtable.h:262
a polymophic address class
Definition: address.h:90
void SendPeriodicUpdate()
Broadcasts the entire routing table for every PeriodicUpdateInterval.
uint8_t GetProtocol(void) const
Definition: ipv4-header.cc:272
void SetSource(Ipv4Address src)
Definition: ipv4-route.cc:49
Ptr< Socket > FindSocketWithInterfaceAddress(Ipv4InterfaceAddress iface) const
Find socket with local interface address iface.
Packet header for IPv4.
Definition: ipv4-header.h:33
bool ForceDeleteIpv4Event(Ipv4Address address)
Force delete an update waiting for settling time to complete as a better update to same destination w...
Definition: dsdv-rtable.cc:296
bool IsMulticast(void) const
static EventId ScheduleNow(FUNC f, Ts &&... args)
Schedule an event to expire Now.
Definition: simulator.h:588
AttributeValue implementation for Time.
Definition: nstime.h:1342
void Schedule(void)
Schedule a new event using the currently-configured delay, function, and arguments.
Definition: timer.cc:158
DSDV Queue Entry.
void SetGateway(Ipv4Address gw)
Definition: ipv4-route.cc:63
void SetFunction(FN fn)
Definition: timer.h:278
Hold an unsigned integer type.
Definition: uinteger.h:44
TAG_BUFFER_INLINE void WriteU32(uint32_t v)
Definition: tag-buffer.h:186
void SetSeqNo(uint32_t sequenceNumber)
Set sequence number.
Definition: dsdv-rtable.h:163
bool GetEnableRAFlag() const
Get enable route aggregation (RA) flag.
Unit
The unit to use to interpret a number representing time.
Definition: nstime.h:109
void Purge(std::map< Ipv4Address, RoutingTableEntry > &removedAddresses)
Delete all outdated entries if Lifetime is expired.
Definition: dsdv-rtable.cc:208
uint32_t m_maxQueueLen
The maximum number of packets that we allow a routing protocol to buffer.
bool RouteInput(Ptr< const Packet > p, const Ipv4Header &header, Ptr< const NetDevice > idev, UnicastForwardCallback ucb, MulticastForwardCallback mcb, LocalDeliverCallback lcb, ErrorCallback ecb)
Route input packet.
Ipv4Address GetDestination(void) const
Definition: ipv4-route.cc:42
void SetRecvCallback(Callback< void, Ptr< Socket > >)
Notify application when new data is available to be read.
Definition: socket.cc:128
static Ptr< Socket > CreateSocket(Ptr< Node > node, TypeId tid)
This method wraps the creation of sockets that is performed on a given node by a SocketFactory specif...
Definition: socket.cc:71
Ipv4Address GetBroadcast(void) const
Get the broadcast address.
Ipv4Mask GetMask(void) const
Get the network mask.
virtual void NotifyRemoveAddress(uint32_t interface, Ipv4InterfaceAddress address)
static TypeId GetTypeId(void)
Get the type ID.
#define NS_LOG_LOGIC(msg)
Use NS_LOG to output a message of level LOG_LOGIC.
Definition: log.h:289
void SetMaxQueueLen(uint32_t len)
Set maximum queue length.
UnicastForwardCallback m_scb
Unicast callback for own packets.
TypeId GetInstanceTypeId() const
Get the most derived TypeId for this Object.
double f(double x, void *params)
Definition: 80211b.c:70
tag a set of bytes in a packet
Definition: tag.h:36
Ptr< Ipv4Route > RouteOutput(Ptr< Packet > p, const Ipv4Header &header, Ptr< NetDevice > oif, Socket::SocketErrno &sockerr)
Query routing cache for an existing route, for an outbound packet.
Implement the IPv4 layer.
Time GetSettlingTime(Ipv4Address dst)
Get settlingTime for a destination.
Ptr< NetDevice > GetOutputDevice(void) const
Definition: ipv4-route.cc:84
virtual int Bind(const Address &address)=0
Allocate a local endpoint for this socket.
void DeferredRouteOutput(Ptr< const Packet > p, const Ipv4Header &header, UnicastForwardCallback ucb, ErrorCallback ecb)
Queue packet until we find a route.
void LookForQueuedPackets(void)
Look for any queued packets to send them out.
Every class exported by the ns3 library is enclosed in the ns3 namespace.
Time m_routeAggregationTime
Parameter that holds the route aggregation time interval.
static InetSocketAddress ConvertFrom(const Address &address)
Returns an InetSocketAddress which corresponds to the input Address.
address
Definition: first.py:44
Ptr< Ipv4Route > GetRoute() const
Get route.
Definition: dsdv-rtable.h:91
bool Dequeue(Ipv4Address dst, QueueEntry &entry)
Return first found (the earliest) entry for given destination.
Time GetSettlingTime() const
Get settling time.
Definition: dsdv-rtable.h:226
PacketQueue m_queue
A "drop front on full" queue used by the routing layer to buffer packets to which it does not have a ...
void DeleteAllRoutesFromInterface(Ipv4InterfaceAddress iface)
Delete all route from interface with address iface.
Definition: dsdv-rtable.cc:148
Time m_maxQueueTime
The maximum period of time that a routing protocol is allowed to buffer a packet for.
Ptr< Packet > Copy(void) const
performs a COW copy of the packet.
Definition: packet.cc:121
UnicastForwardCallback GetUnicastForwardCallback() const
Get unicast forward callback function.
bool AddRoute(RoutingTableEntry &r)
Add routing table entry if it doesn&#39;t yet exist in routing table.
Definition: dsdv-rtable.cc:128
void RecvDsdv(Ptr< Socket > socket)
Receive and process dsdv control packet.
void SetQueueTimeout(Time t)
Set queue timeout.
Time GetLifeTime() const
Get lifetime.
Definition: dsdv-rtable.h:208
void Print(Ptr< OutputStreamWrapper > stream, Time::Unit unit=Time::S) const
Print routing table.
Definition: dsdv-rtable.cc:252
Ipv4Address m_mainAddress
Nodes IP address.
Ptr< const AttributeAccessor > MakeTimeAccessor(T1 a1)
Create an AttributeAccessor for a class data member, or a lone class get functor or set method...
Definition: nstime.h:1343
static Time Now(void)
Return the current simulation virtual time.
Definition: simulator.cc:195
void SetOutputDevice(Ptr< NetDevice > outputDevice)
Equivalent in Linux to dst_entry.dev.
Definition: ipv4-route.cc:77
static Ipv4Address GetLoopback(void)
bool GetWSTFlag() const
Get weighted settling time (WST) flag.
void SendTriggeredUpdate()
Sends trigger update from a node.
uint32_t m_maxQueuedPacketsPerDst
The maximum number of packets that we allow per destination to buffer.
virtual void BindToNetDevice(Ptr< NetDevice > netdevice)
Bind a socket to specific device.
Definition: socket.cc:330
virtual void NotifyAddAddress(uint32_t interface, Ipv4InterfaceAddress address)
uint8_t GetTtl(void) const
Definition: ipv4-header.cc:265
Ipv4Address GetDestination(void) const
Definition: ipv4-header.cc:304
#define NS_ASSERT_MSG(condition, message)
At runtime, in debugging builds, if this condition is not true, the program prints the message to out...
Definition: assert.h:88
Ptr< const AttributeAccessor > MakeDoubleAccessor(T1 a1)
Create an AttributeAccessor for a class data member, or a lone class get functor or set method...
Definition: double.h:42
void GetListOfDestinationWithNextHop(Ipv4Address nxtHp, std::map< Ipv4Address, RoutingTableEntry > &dstList)
Lookup list of addresses for which nxtHp is the next Hop address.
Definition: dsdv-rtable.cc:183
Ipv4Header GetIpv4Header() const
Get IP header.
bool DeleteIpv4Event(Ipv4Address address)
Clear up the entry from the map after the event is completed.
Definition: dsdv-rtable.cc:311
uint32_t GetUid(void) const
Definition: event-id.cc:95
void Send(Ptr< Ipv4Route >, Ptr< const Packet >, const Ipv4Header &)
Send packet.
Ipv4 addresses are stored in host order in this class.
Definition: ipv4-address.h:41
uint32_t GetHopCount() const
Get hop count.
Definition: dsdv-packet.h:113
read and write tag data
Definition: tag-buffer.h:51
bool DeleteRoute(Ipv4Address dst)
Delete routing table entry with destination address dst, if it exists.
Definition: dsdv-rtable.cc:111
a class to store IPv4 address information on an interface
void SetEnableRAFlag(bool f)
Set enable route aggregation (RA) flag.
An identifier for simulation events.
Definition: event-id.h:53
void AddPacketTag(const Tag &tag) const
Add a packet tag.
Definition: packet.cc:956
Ptr< NetDevice > m_lo
Loopback device used to defer route requests until a route is found.
Ptr< UniformRandomVariable > m_uniformRandomVariable
Provides uniform random variables.
bool RemovePacketTag(Tag &tag)
Remove a packet tag.
Definition: packet.cc:963
bool GetEnableBufferFlag() const
Get enable buffer flag.
A network Node.
Definition: node.h:56
#define NS_LOG_DEBUG(msg)
Use NS_LOG to output a message of level LOG_DEBUG.
Definition: log.h:273
Time Seconds(double value)
Construct a Time in the indicated unit.
Definition: nstime.h:1278
Ptr< const AttributeChecker > MakeBooleanChecker(void)
Definition: boolean.cc:121
void SetTtl(uint8_t ttl)
Definition: ipv4-header.cc:259
void SetHopCount(uint32_t hopCount)
Set hop count.
Definition: dsdv-packet.h:104
Tag used by DSDV implementation.
DSDV Update Packet Format
Definition: dsdv-packet.h:58
Ipv4Address GetDestination() const
Get destination IP address.
Definition: dsdv-rtable.h:82
Ipv4Address GetLocal(void) const
Get the local address.
void SetEnableBufferFlag(bool f)
Set enable buffer flag.
Abstract base class for IPv4 routing protocols.
void Clear()
Delete all entries from routing table.
Definition: dsdv-rtable.h:383
DeferredRouteOutputTag(int32_t o=-1)
Constructor.
Ptr< const AttributeChecker > MakeTimeChecker(const Time min, const Time max)
Helper to make a Time checker with bounded range.
Definition: time.cc:472
virtual int SendTo(Ptr< Packet > p, uint32_t flags, const Address &toAddress)=0
Send data to a specified peer.
Ptr< Ipv4 > m_ipv4
IP protocol.
void Setholddowntime(Time t)
Set hold down time (time until an invalid route may be deleted)
Definition: dsdv-rtable.h:457
RoutingTable m_advRoutingTable
Advertised Routing table for the node.
Time MicroSeconds(uint64_t value)
Construct a Time in the indicated unit.
Definition: nstime.h:1294
Time Now(void)
create an ns3::Time instance which contains the current simulation time.
Definition: simulator.cc:287
#define NS_LOG_ERROR(msg)
Use NS_LOG to output a message of level LOG_ERROR.
Definition: log.h:257
virtual void NotifyInterfaceUp(uint32_t interface)
second
Definition: nstime.h:115
bool AddIpv4Event(Ipv4Address address, EventId id)
Add an event for a destination address so that the update to for that destination is sent after the e...
Definition: dsdv-rtable.cc:264
bool LookupRoute(Ipv4Address dst, RoutingTableEntry &rt)
Lookup routing table entry with destination address dst.
Definition: dsdv-rtable.cc:72
virtual Ptr< Packet > RecvFrom(uint32_t maxSize, uint32_t flags, Address &fromAddress)=0
Read a single packet from the socket and retrieve the sender address.
Timer m_periodicUpdateTimer
Timer to trigger periodic updates from a node.
uint32_t GetSize()
Get the number of entries.
bool PeekPacketTag(Tag &tag) const
Search a matching tag and call Tag::Deserialize if it is found.
Definition: packet.cc:978
void SetWSTFlag(bool f)
Set weighted settling time (WST) flag.
virtual int Send(Ptr< Packet > p, uint32_t flags)=0
Send data (or dummy data) to the remote host.
Ipv4Address GetSource(void) const
Definition: ipv4-route.cc:56
virtual int Close(void)=0
Close a socket.
RoutingTable m_routingTable
Main Routing table for the node.
bool IsNull(void) const
Check for null implementation.
Definition: callback.h:1386
void Drop(Ptr< const Packet >, const Ipv4Header &, Socket::SocketErrno)
Notify that packet is dropped for some reason.
This class can be used to hold variables of floating point type such as &#39;double&#39; or &#39;float&#39;...
Definition: double.h:41
void SetAttribute(std::string name, const AttributeValue &value)
Set a single attribute, raising fatal errors if unsuccessful.
Definition: object-base.cc:185
Ptr< const AttributeAccessor > MakeUintegerAccessor(T1 a1)
Create an AttributeAccessor for a class data member, or a lone class get functor or set method...
Definition: uinteger.h:45
a unique identifier for an interface.
Definition: type-id.h:58
Ipv4Address GetNextHop() const
Get next hop.
Definition: dsdv-rtable.h:118
Time m_settlingTime
SettlingTime specifies the time for which a node waits before propagating an update.
TypeId SetParent(TypeId tid)
Set the parent TypeId.
Definition: type-id.cc:923
Callback< R, Ts... > MakeCallback(R(T::*memPtr)(Ts...), OBJ objPtr)
Build Callbacks for class method members which take varying numbers of arguments and potentially retu...
Definition: callback.h:1642
static Time GetMaximumSimulationTime(void)
Get the maximum representable simulation time.
Definition: simulator.cc:293
std::ostream * GetStream(void)
Return a pointer to an ostream previously set in the wrapper.
virtual Address GetAddress(void) const =0
void AddHeader(const Header &header)
Add header to this packet.
Definition: packet.cc:256
bool Find(Ipv4Address dst)
Finds whether a packet with destination dst exists in the queue.
void SetDestination(Ipv4Address dest)
Definition: ipv4-route.cc:35
Ipv4Address GetIpv4(void) const
Callback< void, Ptr< Ipv4Route >, Ptr< const Packet >, const Ipv4Header & > UnicastForwardCallback
Callback for unicast packets to be forwarded.
uint32_t Holdtimes
Holdtimes is the multiplicative factor of PeriodicUpdateInterval for which the node waits since the l...
bool Enqueue(QueueEntry &entry)
Push entry in queue, if there is no entry with the same packet and destination address in queue...