A Discrete-Event Network Simulator
API
ripng-simple-network.cc
Go to the documentation of this file.
1/*
2 * Copyright (c) 2014 Universita' di Firenze, Italy
3 *
4 * This program is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License version 2 as
6 * published by the Free Software Foundation;
7 *
8 * This program is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 * GNU General Public License for more details.
12 *
13 * You should have received a copy of the GNU General Public License
14 * along with this program; if not, write to the Free Software
15 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
16 *
17 * Author: Tommaso Pecorella <tommaso.pecorella@unifi.it>
18 */
19
20// Network topology
21//
22// SRC
23// |<=== source network
24// A-----B
25// \ / \ all networks have cost 1, except
26// \ / | for the direct link from C to D, which
27// C / has cost 10
28// | /
29// |/
30// D
31// |<=== target network
32// DST
33//
34//
35// A, B, C and D are RIPng routers.
36// A and D are configured with static addresses.
37// SRC and DST will exchange packets.
38//
39// After about 3 seconds, the topology is built, and Echo Reply will be received.
40// After 40 seconds, the link between B and D will break, causing a route failure.
41// After 44 seconds from the failure, the routers will recovery from the failure.
42// Split Horizoning should affect the recovery time, but it is not. See the manual
43// for an explanation of this effect.
44//
45// If "showPings" is enabled, the user will see:
46// 1) if the ping has been acknowledged
47// 2) if a Destination Unreachable has been received by the sender
48// 3) nothing, when the Echo Request has been received by the destination but
49// the Echo Reply is unable to reach the sender.
50// Examining the .pcap files with Wireshark can confirm this effect.
51
52#include "ns3/core-module.h"
53#include "ns3/csma-module.h"
54#include "ns3/internet-apps-module.h"
55#include "ns3/internet-module.h"
56#include "ns3/ipv6-routing-table-entry.h"
57#include "ns3/ipv6-static-routing-helper.h"
58
59#include <fstream>
60
61using namespace ns3;
62
63NS_LOG_COMPONENT_DEFINE("RipNgSimpleRouting");
64
65void
66TearDownLink(Ptr<Node> nodeA, Ptr<Node> nodeB, uint32_t interfaceA, uint32_t interfaceB)
67{
68 nodeA->GetObject<Ipv6>()->SetDown(interfaceA);
69 nodeB->GetObject<Ipv6>()->SetDown(interfaceB);
70}
71
72int
73main(int argc, char** argv)
74{
75 bool verbose = false;
76 bool printRoutingTables = false;
77 bool showPings = false;
78 std::string SplitHorizon("PoisonReverse");
79
80 CommandLine cmd(__FILE__);
81 cmd.AddValue("verbose", "turn on log components", verbose);
82 cmd.AddValue("printRoutingTables",
83 "Print routing tables at 30, 60 and 90 seconds",
84 printRoutingTables);
85 cmd.AddValue("showPings", "Show Ping6 reception", showPings);
86 cmd.AddValue("splitHorizonStrategy",
87 "Split Horizon strategy to use (NoSplitHorizon, SplitHorizon, PoisonReverse)",
88 SplitHorizon);
89 cmd.Parse(argc, argv);
90
91 if (verbose)
92 {
93 LogComponentEnable("RipNgSimpleRouting", LOG_LEVEL_INFO);
95 LogComponentEnable("Icmpv6L4Protocol", LOG_LEVEL_INFO);
96 LogComponentEnable("Ipv6Interface", LOG_LEVEL_ALL);
97 LogComponentEnable("Icmpv6L4Protocol", LOG_LEVEL_ALL);
98 LogComponentEnable("NdiscCache", LOG_LEVEL_ALL);
99 LogComponentEnable("Ping6Application", LOG_LEVEL_ALL);
100 }
101
102 if (showPings)
103 {
104 LogComponentEnable("Ping6Application", LOG_LEVEL_INFO);
105 }
106
107 if (SplitHorizon == "NoSplitHorizon")
108 {
109 Config::SetDefault("ns3::RipNg::SplitHorizon", EnumValue(RipNg::NO_SPLIT_HORIZON));
110 }
111 else if (SplitHorizon == "SplitHorizon")
112 {
113 Config::SetDefault("ns3::RipNg::SplitHorizon", EnumValue(RipNg::SPLIT_HORIZON));
114 }
115 else
116 {
117 Config::SetDefault("ns3::RipNg::SplitHorizon", EnumValue(RipNg::POISON_REVERSE));
118 }
119
120 NS_LOG_INFO("Create nodes.");
121 Ptr<Node> src = CreateObject<Node>();
122 Names::Add("SrcNode", src);
123 Ptr<Node> dst = CreateObject<Node>();
124 Names::Add("DstNode", dst);
125 Ptr<Node> a = CreateObject<Node>();
126 Names::Add("RouterA", a);
127 Ptr<Node> b = CreateObject<Node>();
128 Names::Add("RouterB", b);
129 Ptr<Node> c = CreateObject<Node>();
130 Names::Add("RouterC", c);
131 Ptr<Node> d = CreateObject<Node>();
132 Names::Add("RouterD", d);
133 NodeContainer net1(src, a);
134 NodeContainer net2(a, b);
135 NodeContainer net3(a, c);
136 NodeContainer net4(b, c);
137 NodeContainer net5(c, d);
138 NodeContainer net6(b, d);
139 NodeContainer net7(d, dst);
140 NodeContainer routers(a, b, c, d);
141 NodeContainer nodes(src, dst);
142
143 NS_LOG_INFO("Create channels.");
145 csma.SetChannelAttribute("DataRate", DataRateValue(5000000));
146 csma.SetChannelAttribute("Delay", TimeValue(MilliSeconds(2)));
147 NetDeviceContainer ndc1 = csma.Install(net1);
148 NetDeviceContainer ndc2 = csma.Install(net2);
149 NetDeviceContainer ndc3 = csma.Install(net3);
150 NetDeviceContainer ndc4 = csma.Install(net4);
151 NetDeviceContainer ndc5 = csma.Install(net5);
152 NetDeviceContainer ndc6 = csma.Install(net6);
153 NetDeviceContainer ndc7 = csma.Install(net7);
154
155 NS_LOG_INFO("Create IPv6 and routing");
156 RipNgHelper ripNgRouting;
157
158 // Rule of thumb:
159 // Interfaces are added sequentially, starting from 0
160 // However, interface 0 is always the loopback...
161 ripNgRouting.ExcludeInterface(a, 1);
162 ripNgRouting.ExcludeInterface(d, 3);
163
164 ripNgRouting.SetInterfaceMetric(c, 3, 10);
165 ripNgRouting.SetInterfaceMetric(d, 1, 10);
166
168 listRH.Add(ripNgRouting, 0);
170 listRH.Add(staticRh, 5);
171
172 InternetStackHelper internetv6;
173 internetv6.SetIpv4StackInstall(false);
174 internetv6.SetRoutingHelper(listRH);
175 internetv6.Install(routers);
176
177 InternetStackHelper internetv6Nodes;
178 internetv6Nodes.SetIpv4StackInstall(false);
179 internetv6Nodes.Install(nodes);
180
181 // Assign addresses.
182 // The source and destination networks have global addresses
183 // The "core" network just needs link-local addresses for routing.
184 // We assign global addresses to the routers as well to receive
185 // ICMPv6 errors.
186 NS_LOG_INFO("Assign IPv6 Addresses.");
188
189 ipv6.SetBase(Ipv6Address("2001:1::"), Ipv6Prefix(64));
190 Ipv6InterfaceContainer iic1 = ipv6.Assign(ndc1);
191 iic1.SetForwarding(1, true);
193
194 ipv6.SetBase(Ipv6Address("2001:0:1::"), Ipv6Prefix(64));
195 Ipv6InterfaceContainer iic2 = ipv6.Assign(ndc2);
196 iic2.SetForwarding(0, true);
197 iic2.SetForwarding(1, true);
198
199 ipv6.SetBase(Ipv6Address("2001:0:2::"), Ipv6Prefix(64));
200 Ipv6InterfaceContainer iic3 = ipv6.Assign(ndc3);
201 iic3.SetForwarding(0, true);
202 iic3.SetForwarding(1, true);
203
204 ipv6.SetBase(Ipv6Address("2001:0:3::"), Ipv6Prefix(64));
205 Ipv6InterfaceContainer iic4 = ipv6.Assign(ndc4);
206 iic4.SetForwarding(0, true);
207 iic4.SetForwarding(1, true);
208
209 ipv6.SetBase(Ipv6Address("2001:0:4::"), Ipv6Prefix(64));
210 Ipv6InterfaceContainer iic5 = ipv6.Assign(ndc5);
211 iic5.SetForwarding(0, true);
212 iic5.SetForwarding(1, true);
213
214 ipv6.SetBase(Ipv6Address("2001:0:5::"), Ipv6Prefix(64));
215 Ipv6InterfaceContainer iic6 = ipv6.Assign(ndc6);
216 iic6.SetForwarding(0, true);
217 iic6.SetForwarding(1, true);
218
219 ipv6.SetBase(Ipv6Address("2001:2::"), Ipv6Prefix(64));
220 Ipv6InterfaceContainer iic7 = ipv6.Assign(ndc7);
221 iic7.SetForwarding(0, true);
223
224 if (printRoutingTables)
225 {
226 RipNgHelper routingHelper;
227
228 Ptr<OutputStreamWrapper> routingStream = Create<OutputStreamWrapper>(&std::cout);
229
230 routingHelper.PrintRoutingTableAt(Seconds(30.0), a, routingStream);
231 routingHelper.PrintRoutingTableAt(Seconds(30.0), b, routingStream);
232 routingHelper.PrintRoutingTableAt(Seconds(30.0), c, routingStream);
233 routingHelper.PrintRoutingTableAt(Seconds(30.0), d, routingStream);
234
235 routingHelper.PrintRoutingTableAt(Seconds(60.0), a, routingStream);
236 routingHelper.PrintRoutingTableAt(Seconds(60.0), b, routingStream);
237 routingHelper.PrintRoutingTableAt(Seconds(60.0), c, routingStream);
238 routingHelper.PrintRoutingTableAt(Seconds(60.0), d, routingStream);
239
240 routingHelper.PrintRoutingTableAt(Seconds(90.0), a, routingStream);
241 routingHelper.PrintRoutingTableAt(Seconds(90.0), b, routingStream);
242 routingHelper.PrintRoutingTableAt(Seconds(90.0), c, routingStream);
243 routingHelper.PrintRoutingTableAt(Seconds(90.0), d, routingStream);
244 }
245
246 NS_LOG_INFO("Create Applications.");
247 uint32_t packetSize = 1024;
248 uint32_t maxPacketCount = 100;
249 Time interPacketInterval = Seconds(1.0);
250 Ping6Helper ping6;
251
252 ping6.SetLocal(iic1.GetAddress(0, 1));
253 ping6.SetRemote(iic7.GetAddress(1, 1));
254 ping6.SetAttribute("MaxPackets", UintegerValue(maxPacketCount));
255 ping6.SetAttribute("Interval", TimeValue(interPacketInterval));
256 ping6.SetAttribute("PacketSize", UintegerValue(packetSize));
257 ApplicationContainer apps = ping6.Install(src);
258 apps.Start(Seconds(1.0));
259 apps.Stop(Seconds(110.0));
260
261 AsciiTraceHelper ascii;
262 csma.EnableAsciiAll(ascii.CreateFileStream("ripng-simple-routing.tr"));
263 csma.EnablePcapAll("ripng-simple-routing", true);
264
265 Simulator::Schedule(Seconds(40), &TearDownLink, b, d, 3, 2);
266
267 /* Now, do the actual simulation. */
268 NS_LOG_INFO("Run Simulation.");
269 Simulator::Stop(Seconds(120));
270 Simulator::Run();
271 Simulator::Destroy();
272 NS_LOG_INFO("Done.");
273
274 return 0;
275}
holds a vector of ns3::Application pointers.
void Start(Time start)
Arrange for all of the Applications in this container to Start() at the Time given as a parameter.
void Stop(Time stop)
Arrange for all of the Applications in this container to Stop() at the Time given as a parameter.
Manage ASCII trace files for device models.
Definition: trace-helper.h:173
Ptr< OutputStreamWrapper > CreateFileStream(std::string filename, std::ios::openmode filemode=std::ios::out)
Create and initialize an output stream object we'll use to write the traced bits.
Parse command-line arguments.
Definition: command-line.h:232
build a set of CsmaNetDevice objects
Definition: csma-helper.h:48
AttributeValue implementation for DataRate.
Hold variables of type enum.
Definition: enum.h:56
aggregate IP/TCP/UDP functionality to existing Nodes.
void SetIpv4StackInstall(bool enable)
Enable/disable IPv4 stack install.
void Install(std::string nodeName) const
Aggregate implementations of the ns3::Ipv4, ns3::Ipv6, ns3::Udp, and ns3::Tcp classes onto the provid...
void SetRoutingHelper(const Ipv4RoutingHelper &routing)
Helper class to auto-assign global IPv6 unicast addresses.
void SetBase(Ipv6Address network, Ipv6Prefix prefix, Ipv6Address base=Ipv6Address("::1"))
Set the base network number, network prefix, and base interface ID.
Ipv6InterfaceContainer Assign(const NetDeviceContainer &c)
Allocate an Ipv6InterfaceContainer with auto-assigned addresses.
Describes an IPv6 address.
Definition: ipv6-address.h:50
Access to the IPv6 forwarding table, interfaces, and configuration.
Definition: ipv6.h:82
Keep track of a set of IPv6 interfaces.
void SetForwarding(uint32_t i, bool state)
Set the state of the stack (act as a router or as an host) for the specified index.
void SetDefaultRouteInAllNodes(uint32_t router)
Set the default route for all the devices (except the router itself).
Ipv6Address GetAddress(uint32_t i, uint32_t j) const
Get the address for the specified index.
Helper class that adds ns3::Ipv6ListRouting objects.
void Add(const Ipv6RoutingHelper &routing, int16_t priority)
Describes an IPv6 prefix.
Definition: ipv6-address.h:456
static void PrintRoutingTableAt(Time printTime, Ptr< Node > node, Ptr< OutputStreamWrapper > stream, Time::Unit unit=Time::S)
prints the routing tables of a node at a particular time.
Helper class that adds ns3::Ipv6StaticRouting objects.
holds a vector of ns3::NetDevice pointers
keep track of a set of node pointers.
Ptr< T > GetObject() const
Get a pointer to the requested aggregated Object.
Definition: object.h:471
Ping6 application helper.
Definition: ping6-helper.h:38
void SetRemote(Ipv6Address ip)
Set the remote IPv6 address.
Definition: ping6-helper.cc:41
ApplicationContainer Install(NodeContainer c)
Install the application in Nodes.
Definition: ping6-helper.cc:53
void SetAttribute(std::string name, const AttributeValue &value)
Set some attributes.
Definition: ping6-helper.cc:47
void SetLocal(Ipv6Address ip)
Set the local IPv6 address.
Definition: ping6-helper.cc:35
Helper class that adds RIPng routing to nodes.
Definition: ripng-helper.h:41
void ExcludeInterface(Ptr< Node > node, uint32_t interface)
Exclude an interface from RIPng protocol.
void SetInterfaceMetric(Ptr< Node > node, uint32_t interface, uint8_t metric)
Set a metric for an interface.
Simulation virtual time values and global simulation resolution.
Definition: nstime.h:105
AttributeValue implementation for Time.
Definition: nstime.h:1425
Hold an unsigned integer type.
Definition: uinteger.h:45
void SetDefault(std::string name, const AttributeValue &value)
Definition: config.cc:891
#define NS_LOG_COMPONENT_DEFINE(name)
Define a Log component with a specific name.
Definition: log.h:202
#define NS_LOG_INFO(msg)
Use NS_LOG to output a message of level LOG_INFO.
Definition: log.h:275
Time Seconds(double value)
Construct a Time in the indicated unit.
Definition: nstime.h:1338
Time MilliSeconds(uint64_t value)
Construct a Time in the indicated unit.
Definition: nstime.h:1350
NodeContainer nodes
Every class exported by the ns3 library is enclosed in the ns3 namespace.
@ LOG_LEVEL_ALL
Print everything.
Definition: log.h:116
@ LOG_LEVEL_INFO
LOG_INFO and above.
Definition: log.h:107
void LogComponentEnable(const char *name, enum LogLevel level)
Enable the logging output associated with that log component.
Definition: log.cc:358
csma
Definition: second.py:56
cmd
Definition: second.py:33
bool verbose
void TearDownLink(Ptr< Node > nodeA, Ptr< Node > nodeB, uint32_t interfaceA, uint32_t interfaceB)
static const uint32_t packetSize
Packet size generated at the AP.