blob: 31ff9b2e32b2147caf148a017de41035b5a71226 [file] [log] [blame]
Alexander Afanasyev0ab833e2011-08-18 15:49:13 -07001/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil -*- */
Alexander Afanasyev45b92d42011-08-14 23:11:38 -07002/*
3 * Copyright (c) 2011 UCLA
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 *
Ilya Moiseenko25f7d4d2011-09-29 18:41:06 -070018 * Author: Alexander Afanasyev <alexander.afanasyev@ucla.edu>
Ilya Moiseenkofbd0a8b2011-10-28 13:07:16 -070019 * Ilya Moiseenko <iliamo@cs.ucla.edu>
Alexander Afanasyev45b92d42011-08-14 23:11:38 -070020 */
21
Alexander Afanasyev45b92d42011-08-14 23:11:38 -070022#include "ns3/assert.h"
23#include "ns3/log.h"
24#include "ns3/object.h"
25#include "ns3/names.h"
Alexander Afanasyev45b92d42011-08-14 23:11:38 -070026#include "ns3/packet-socket-factory.h"
27#include "ns3/config.h"
28#include "ns3/simulator.h"
29#include "ns3/string.h"
30#include "ns3/net-device.h"
31#include "ns3/callback.h"
32#include "ns3/node.h"
33#include "ns3/core-config.h"
Alexander Afanasyev07827182011-12-13 01:07:32 -080034#include "ns3/point-to-point-net-device.h"
35#include "ns3/point-to-point-helper.h"
36
37#include "../model/ccnx-forwarding-strategy.h"
Alexander Afanasyevf9f4eb02011-12-16 01:51:14 -080038#include "../model/ccnx-net-device-face.h"
39#include "../model/ccnx-l3-protocol.h"
40#include "../model/ccnx-fib.h"
Alexander Afanasyev07827182011-12-13 01:07:32 -080041
Alexander Afanasyev52e9aa92011-11-15 20:23:20 -080042#include "ns3/node-list.h"
Alexander Afanasyev57bcbc32012-06-01 01:46:24 -070043// #include "ns3/loopback-net-device.h"
Alexander Afanasyevf9f4eb02011-12-16 01:51:14 -080044
Alexander Afanasyev11453142011-11-25 16:13:33 -080045#include "ns3/data-rate.h"
Alexander Afanasyev45b92d42011-08-14 23:11:38 -070046
Alexander Afanasyev0ab833e2011-08-18 15:49:13 -070047#include "ccnx-face-container.h"
Alexander Afanasyev45b92d42011-08-14 23:11:38 -070048#include "ccnx-stack-helper.h"
Alexander Afanasyev45b92d42011-08-14 23:11:38 -070049
50#include <limits>
51#include <map>
Alexander Afanasyev52e9aa92011-11-15 20:23:20 -080052#include <boost/foreach.hpp>
Alexander Afanasyevb7626842012-01-12 13:43:33 -080053#include <boost/lexical_cast.hpp>
Alexander Afanasyeva5bbe0e2011-11-22 17:28:39 -080054
Alexander Afanasyev45b92d42011-08-14 23:11:38 -070055NS_LOG_COMPONENT_DEFINE ("CcnxStackHelper");
56
57namespace ns3 {
Ilya Moiseenko25f7d4d2011-09-29 18:41:06 -070058
Alexander Afanasyev45b92d42011-08-14 23:11:38 -070059CcnxStackHelper::CcnxStackHelper ()
Alexander Afanasyev4a5c2c12011-12-12 18:50:57 -080060 : m_limitsEnabled (false)
61 , m_needSetDefaultRoutes (false)
Alexander Afanasyev45b92d42011-08-14 23:11:38 -070062{
Alexander Afanasyev3a4a0b32012-06-28 14:14:22 -070063 m_ccnxFactory. SetTypeId ("ns3::CcnxL3Protocol");
64 m_strategyFactory. SetTypeId ("ns3::CcnxFloodingStrategy");
Alexander Afanasyevd9fecdd2012-06-08 16:22:24 -070065 m_contentStoreFactory.SetTypeId ("ns3::CcnxContentStoreLru");
Alexander Afanasyev95a4fa32012-07-09 15:23:59 -070066 m_fibFactory. SetTypeId ("ns3::CcnxFibImpl");
Alexander Afanasyev3a4a0b32012-06-28 14:14:22 -070067 m_pitFactory. SetTypeId ("ns3::CcnxPit");
Ilya Moiseenko25f7d4d2011-09-29 18:41:06 -070068}
69
Alexander Afanasyev45b92d42011-08-14 23:11:38 -070070CcnxStackHelper::~CcnxStackHelper ()
71{
Alexander Afanasyev45b92d42011-08-14 23:11:38 -070072}
73
Alexander Afanasyev3a4a0b32012-06-28 14:14:22 -070074void
75CcnxStackHelper::SetCcnxAttributes (const std::string &attr1, const std::string &value1,
76 const std::string &attr2, const std::string &value2,
77 const std::string &attr3, const std::string &value3,
78 const std::string &attr4, const std::string &value4)
79{
80 if (attr1 != "")
81 m_ccnxFactory.Set (attr1, StringValue (value1));
82 if (attr2 != "")
83 m_ccnxFactory.Set (attr2, StringValue (value2));
84 if (attr3 != "")
85 m_ccnxFactory.Set (attr3, StringValue (value3));
86 if (attr4 != "")
87 m_ccnxFactory.Set (attr4, StringValue (value4));
88}
89
Alexander Afanasyev45b92d42011-08-14 23:11:38 -070090void
Alexander Afanasyev3a4a0b32012-06-28 14:14:22 -070091CcnxStackHelper::SetForwardingStrategy (const std::string &strategy,
92 const std::string &attr1, const std::string &value1,
93 const std::string &attr2, const std::string &value2,
94 const std::string &attr3, const std::string &value3,
95 const std::string &attr4, const std::string &value4)
Alexander Afanasyev45b92d42011-08-14 23:11:38 -070096{
Alexander Afanasyev11453142011-11-25 16:13:33 -080097 m_strategyFactory.SetTypeId (strategy);
Alexander Afanasyev3a4a0b32012-06-28 14:14:22 -070098 if (attr1 != "")
99 m_strategyFactory.Set (attr1, StringValue (value1));
100 if (attr2 != "")
101 m_strategyFactory.Set (attr2, StringValue (value2));
102 if (attr3 != "")
103 m_strategyFactory.Set (attr3, StringValue (value3));
104 if (attr4 != "")
105 m_strategyFactory.Set (attr4, StringValue (value4));
Alexander Afanasyev11453142011-11-25 16:13:33 -0800106}
107
Alexander Afanasyevd9fecdd2012-06-08 16:22:24 -0700108void
Alexander Afanasyev3a4a0b32012-06-28 14:14:22 -0700109CcnxStackHelper::SetContentStore (const std::string &contentStore,
110 const std::string &attr1, const std::string &value1,
111 const std::string &attr2, const std::string &value2,
112 const std::string &attr3, const std::string &value3,
113 const std::string &attr4, const std::string &value4)
Alexander Afanasyevd9fecdd2012-06-08 16:22:24 -0700114{
115 m_contentStoreFactory.SetTypeId (contentStore);
Alexander Afanasyev3a4a0b32012-06-28 14:14:22 -0700116 if (attr1 != "")
117 m_contentStoreFactory.Set (attr1, StringValue (value1));
118 if (attr2 != "")
119 m_contentStoreFactory.Set (attr2, StringValue (value2));
120 if (attr3 != "")
121 m_contentStoreFactory.Set (attr3, StringValue (value3));
122 if (attr4 != "")
123 m_contentStoreFactory.Set (attr4, StringValue (value4));
124}
125
126void
127CcnxStackHelper::SetPit (const std::string &pitClass,
128 const std::string &attr1, const std::string &value1,
129 const std::string &attr2, const std::string &value2,
130 const std::string &attr3, const std::string &value3,
131 const std::string &attr4, const std::string &value4)
132{
133 m_pitFactory.SetTypeId (pitClass);
134 if (attr1 != "")
135 m_pitFactory.Set (attr1, StringValue (value1));
136 if (attr2 != "")
137 m_pitFactory.Set (attr2, StringValue (value2));
138 if (attr3 != "")
139 m_pitFactory.Set (attr3, StringValue (value3));
140 if (attr4 != "")
141 m_pitFactory.Set (attr4, StringValue (value4));
142}
143
144void
145CcnxStackHelper::SetFib (const std::string &fibClass,
146 const std::string &attr1, const std::string &value1,
147 const std::string &attr2, const std::string &value2,
148 const std::string &attr3, const std::string &value3,
149 const std::string &attr4, const std::string &value4)
150{
151 m_fibFactory.SetTypeId (fibClass);
152 if (attr1 != "")
153 m_fibFactory.Set (attr1, StringValue (value1));
154 if (attr2 != "")
155 m_fibFactory.Set (attr2, StringValue (value2));
156 if (attr3 != "")
157 m_fibFactory.Set (attr3, StringValue (value3));
158 if (attr4 != "")
159 m_fibFactory.Set (attr4, StringValue (value4));
Alexander Afanasyevd9fecdd2012-06-08 16:22:24 -0700160}
Alexander Afanasyev4a5c2c12011-12-12 18:50:57 -0800161
162void
163CcnxStackHelper::SetDefaultRoutes (bool needSet)
164{
165 NS_LOG_FUNCTION (this << needSet);
166 m_needSetDefaultRoutes = needSet;
167}
168
Alexander Afanasyev11453142011-11-25 16:13:33 -0800169void
Alexander Afanasyev33d62312012-01-09 13:50:20 -0800170CcnxStackHelper::EnableLimits (bool enable/* = true*/,
171 Time avgRtt/*=Seconds(0.1)*/,
172 uint32_t avgContentObject/*=1100*/,
173 uint32_t avgInterest/*=40*/)
Alexander Afanasyev11453142011-11-25 16:13:33 -0800174{
Alexander Afanasyevc39f0b42011-11-28 12:51:12 -0800175 NS_LOG_INFO ("EnableLimits: " << enable);
Alexander Afanasyev11453142011-11-25 16:13:33 -0800176 m_limitsEnabled = enable;
Alexander Afanasyev8f5a9bb2011-12-18 19:49:02 -0800177 m_avgRtt = avgRtt;
Alexander Afanasyevc39f0b42011-11-28 12:51:12 -0800178 m_avgContentObjectSize = avgContentObject;
179 m_avgInterestSize = avgInterest;
Alexander Afanasyev45b92d42011-08-14 23:11:38 -0700180}
181
Alexander Afanasyev0ab833e2011-08-18 15:49:13 -0700182Ptr<CcnxFaceContainer>
Alexander Afanasyev45b92d42011-08-14 23:11:38 -0700183CcnxStackHelper::Install (NodeContainer c) const
184{
Alexander Afanasyev0ab833e2011-08-18 15:49:13 -0700185 Ptr<CcnxFaceContainer> faces = Create<CcnxFaceContainer> ();
Alexander Afanasyev45b92d42011-08-14 23:11:38 -0700186 for (NodeContainer::Iterator i = c.Begin (); i != c.End (); ++i)
187 {
Alexander Afanasyev0ab833e2011-08-18 15:49:13 -0700188 faces->AddAll (Install (*i));
Alexander Afanasyev45b92d42011-08-14 23:11:38 -0700189 }
Alexander Afanasyev0ab833e2011-08-18 15:49:13 -0700190 return faces;
Alexander Afanasyev45b92d42011-08-14 23:11:38 -0700191}
192
Alexander Afanasyev0ab833e2011-08-18 15:49:13 -0700193Ptr<CcnxFaceContainer>
Alexander Afanasyev45b92d42011-08-14 23:11:38 -0700194CcnxStackHelper::InstallAll (void) const
195{
Alexander Afanasyev0ab833e2011-08-18 15:49:13 -0700196 return Install (NodeContainer::GetGlobal ());
Alexander Afanasyev45b92d42011-08-14 23:11:38 -0700197}
198
Alexander Afanasyev0ab833e2011-08-18 15:49:13 -0700199Ptr<CcnxFaceContainer>
Alexander Afanasyev45b92d42011-08-14 23:11:38 -0700200CcnxStackHelper::Install (Ptr<Node> node) const
201{
Alexander Afanasyev0ab833e2011-08-18 15:49:13 -0700202 // NS_ASSERT_MSG (m_forwarding, "SetForwardingHelper() should be set prior calling Install() method");
203 Ptr<CcnxFaceContainer> faces = Create<CcnxFaceContainer> ();
Alexander Afanasyev45b92d42011-08-14 23:11:38 -0700204
Alexander Afanasyevc5a23e22011-09-07 00:37:36 -0700205 if (node->GetObject<Ccnx> () != 0)
Alexander Afanasyev45b92d42011-08-14 23:11:38 -0700206 {
Alexander Afanasyevc5a23e22011-09-07 00:37:36 -0700207 NS_FATAL_ERROR ("CcnxStackHelper::Install (): Installing "
208 "a CcnxStack to a node with an existing Ccnx object");
209 return 0;
Alexander Afanasyev45b92d42011-08-14 23:11:38 -0700210 }
Alexander Afanasyev0ab833e2011-08-18 15:49:13 -0700211
Alexander Afanasyev3a4a0b32012-06-28 14:14:22 -0700212 // Create CcnxL3Protocol
213 Ptr<Ccnx> ccnx = m_ccnxFactory.Create<Ccnx> ();
Alexander Afanasyevc5a23e22011-09-07 00:37:36 -0700214
Alexander Afanasyev3a4a0b32012-06-28 14:14:22 -0700215 // Create and aggregate FIB
Alexander Afanasyev44bb6ea2012-07-09 08:44:41 -0700216 Ptr<CcnxFib> fib = m_fibFactory.Create<CcnxFib> ();
217 ccnx->AggregateObject (fib);
Alexander Afanasyev3a4a0b32012-06-28 14:14:22 -0700218
219 // Create and aggregate PIT
220 ccnx->AggregateObject (m_pitFactory.Create<CcnxPit> ());
221
222 // Create and aggregate forwarding strategy
223 ccnx->AggregateObject (m_strategyFactory.Create<CcnxForwardingStrategy> ());
224
225 // Create and aggregate content store
226 ccnx->AggregateObject (m_contentStoreFactory.Create<CcnxContentStore> ());
227
228 // Aggregate CcnxL3Protocol on node
Alexander Afanasyevc5a23e22011-09-07 00:37:36 -0700229 node->AggregateObject (ccnx);
Alexander Afanasyev11453142011-11-25 16:13:33 -0800230
Alexander Afanasyevc5a23e22011-09-07 00:37:36 -0700231 for (uint32_t index=0; index < node->GetNDevices (); index++)
232 {
Alexander Afanasyev11453142011-11-25 16:13:33 -0800233 Ptr<NetDevice> device = node->GetDevice (index);
Alexander Afanasyev57bcbc32012-06-01 01:46:24 -0700234 // This check does not make sense: LoopbackNetDevice is installed only if IP stack is installed,
235 // Normally, ndnSIM works without IP stack, so no reason to check
236 // if (DynamicCast<LoopbackNetDevice> (device) != 0)
237 // continue; // don't create face for a LoopbackNetDevice
Alexander Afanasyev11453142011-11-25 16:13:33 -0800238
Alexander Afanasyevcbe92ae2011-12-16 13:06:18 -0800239 Ptr<CcnxNetDeviceFace> face = CreateObject<CcnxNetDeviceFace> (node, device);
Alexander Afanasyev19426ef2011-11-23 20:55:28 -0800240
Alexander Afanasyev4a5c2c12011-12-12 18:50:57 -0800241 ccnx->AddFace (face);
242 NS_LOG_LOGIC ("Node " << node->GetId () << ": added CcnxNetDeviceFace as face #" << *face);
Alexander Afanasyev11453142011-11-25 16:13:33 -0800243
Alexander Afanasyev4a5c2c12011-12-12 18:50:57 -0800244 if (m_needSetDefaultRoutes)
245 {
246 // default route with lowest priority possible
Alexander Afanasyev463fe222012-05-31 13:46:51 -0700247 AddRoute (node, "/", face, std::numeric_limits<int32_t>::max ());
Alexander Afanasyev4a5c2c12011-12-12 18:50:57 -0800248 }
249
Alexander Afanasyev11453142011-11-25 16:13:33 -0800250 if (m_limitsEnabled)
251 {
Alexander Afanasyevc39f0b42011-11-28 12:51:12 -0800252 NS_LOG_INFO ("Limits are enabled");
Alexander Afanasyev11453142011-11-25 16:13:33 -0800253 Ptr<PointToPointNetDevice> p2p = DynamicCast<PointToPointNetDevice> (device);
Alexander Afanasyev3f1c8b02012-01-19 19:41:34 -0800254 if (p2p != 0)
Alexander Afanasyevc39f0b42011-11-28 12:51:12 -0800255 {
Alexander Afanasyev3f1c8b02012-01-19 19:41:34 -0800256 // Setup bucket filtering
257 // Assume that we know average data packet size, and this size is equal default size
258 // Set maximum buckets (averaging over 1 second)
Ilya Moiseenkoc9266042011-11-02 17:49:21 -0700259
Alexander Afanasyev3f1c8b02012-01-19 19:41:34 -0800260 DataRateValue dataRate; device->GetAttribute ("DataRate", dataRate);
Alexander Afanasyev11453142011-11-25 16:13:33 -0800261
Alexander Afanasyev3f1c8b02012-01-19 19:41:34 -0800262 NS_LOG_INFO("DataRate for this link is " << dataRate.Get());
Alexander Afanasyevc39f0b42011-11-28 12:51:12 -0800263
Alexander Afanasyev3f1c8b02012-01-19 19:41:34 -0800264 double maxInterestPackets = 1.0 * dataRate.Get ().GetBitRate () / 8.0 / (m_avgContentObjectSize + m_avgInterestSize);
265 NS_LOG_INFO ("Max packets per second: " << maxInterestPackets);
266 NS_LOG_INFO ("Max burst: " << m_avgRtt.ToDouble (Time::S) * maxInterestPackets);
Alexander Afanasyevc39f0b42011-11-28 12:51:12 -0800267
Alexander Afanasyev3f1c8b02012-01-19 19:41:34 -0800268 // Set bucket max to BDP
269 face->SetBucketMax (m_avgRtt.ToDouble (Time::S) * maxInterestPackets); // number of interests allowed
270 face->SetBucketLeak (maxInterestPackets);
271 }
Alexander Afanasyev11453142011-11-25 16:13:33 -0800272 }
Ilya Moiseenkoc9266042011-11-02 17:49:21 -0700273
Ilya Moiseenko25f7d4d2011-09-29 18:41:06 -0700274 face->SetUp ();
Alexander Afanasyevc5a23e22011-09-07 00:37:36 -0700275 faces->Add (face);
276 }
Ilya Moiseenko25f7d4d2011-09-29 18:41:06 -0700277
Alexander Afanasyev0ab833e2011-08-18 15:49:13 -0700278 return faces;
Alexander Afanasyev45b92d42011-08-14 23:11:38 -0700279}
280
Alexander Afanasyev0ab833e2011-08-18 15:49:13 -0700281Ptr<CcnxFaceContainer>
Alexander Afanasyev45b92d42011-08-14 23:11:38 -0700282CcnxStackHelper::Install (std::string nodeName) const
283{
284 Ptr<Node> node = Names::Find<Node> (nodeName);
Alexander Afanasyev0ab833e2011-08-18 15:49:13 -0700285 return Install (node);
Alexander Afanasyev45b92d42011-08-14 23:11:38 -0700286}
287
Alexander Afanasyevc5a23e22011-09-07 00:37:36 -0700288
289void
Alexander Afanasyevb7626842012-01-12 13:43:33 -0800290CcnxStackHelper::AddRoute (Ptr<Node> node, std::string prefix, Ptr<CcnxFace> face, int32_t metric)
Alexander Afanasyev52e9aa92011-11-15 20:23:20 -0800291{
Alexander Afanasyev4a5c2c12011-12-12 18:50:57 -0800292 NS_LOG_LOGIC ("[" << node->GetId () << "]$ route add " << prefix << " via " << *face << " metric " << metric);
293
Alexander Afanasyev52e9aa92011-11-15 20:23:20 -0800294 Ptr<CcnxFib> fib = node->GetObject<CcnxFib> ();
295
296 CcnxNameComponentsValue prefixValue;
297 prefixValue.DeserializeFromString (prefix, MakeCcnxNameComponentsChecker ());
298 fib->Add (prefixValue.Get (), face, metric);
299}
300
301void
Alexander Afanasyevb7626842012-01-12 13:43:33 -0800302CcnxStackHelper::AddRoute (std::string nodeName, std::string prefix, uint32_t faceId, int32_t metric)
Alexander Afanasyevc5a23e22011-09-07 00:37:36 -0700303{
Alexander Afanasyevc5a23e22011-09-07 00:37:36 -0700304 Ptr<Node> node = Names::Find<Node> (nodeName);
305 NS_ASSERT_MSG (node != 0, "Node [" << nodeName << "] does not exist");
Alexander Afanasyeva4e3f852011-11-15 20:39:33 -0800306
Alexander Afanasyev52e9aa92011-11-15 20:23:20 -0800307 Ptr<Ccnx> ccnx = node->GetObject<Ccnx> ();
308 NS_ASSERT_MSG (ccnx != 0, "Ccnx stack should be installed on the node");
309
310 Ptr<CcnxFace> face = ccnx->GetFace (faceId);
311 NS_ASSERT_MSG (face != 0, "Face with ID [" << faceId << "] does not exist on node [" << nodeName << "]");
312
313 AddRoute (node, prefix, face, metric);
Alexander Afanasyevc5a23e22011-09-07 00:37:36 -0700314}
Alexander Afanasyev4a5c2c12011-12-12 18:50:57 -0800315
Alexander Afanasyev45b92d42011-08-14 23:11:38 -0700316
Alexander Afanasyev45b92d42011-08-14 23:11:38 -0700317} // namespace ns3