blob: 9434ed826468ced19f0d41d6771e744648bd2d1e [file] [log] [blame]
Alexander Afanasyevc74a6022011-08-15 20:01:35 -07001/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil -*- */
Alexander Afanasyev08d984e2011-08-13 19:20:22 -07002/*
Ilya Moiseenko956d0542012-01-02 15:26:40 -08003 * Copyright (c) 2011 University of California, Los Angeles
Alexander Afanasyev08d984e2011-08-13 19:20:22 -07004 *
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 *
Alexander Afanasyev56f79ea2011-08-17 23:54:27 -070018 * Author: Alexander Afanasyev <alexander.afanasyev@ucla.edu>
Alexander Afanasyev08d984e2011-08-13 19:20:22 -070019 *
20 */
21
Alexander Afanasyev4aac5572012-08-09 10:49:55 -070022#include "ndn-face.h"
Alexander Afanasyev08d984e2011-08-13 19:20:22 -070023
Alexander Afanasyev19426ef2011-11-23 20:55:28 -080024#include "ns3/packet.h"
Alexander Afanasyev08d984e2011-08-13 19:20:22 -070025#include "ns3/log.h"
Alexander Afanasyev08d984e2011-08-13 19:20:22 -070026#include "ns3/node.h"
Alexander Afanasyev7fd74f92011-08-25 19:40:17 -070027#include "ns3/assert.h"
Alexander Afanasyevcbe92ae2011-12-16 13:06:18 -080028#include "ns3/uinteger.h"
29#include "ns3/double.h"
Alexander Afanasyev8e0d2812012-01-19 22:38:14 -080030#include "ns3/boolean.h"
Alexander Afanasyev4975f732011-12-20 17:52:19 -080031#include "ns3/simulator.h"
Alexander Afanasyev98d85e12012-08-09 10:06:05 -070032#include "ns3/random-variable.h"
Alexander Afanasyevea9b3e62012-08-13 19:02:54 -070033#include "ns3/pointer.h"
Alexander Afanasyev08d984e2011-08-13 19:20:22 -070034
Alexander Afanasyevfaa01f92013-07-10 18:34:31 -070035#include "ns3/ndn-header-helper.h"
Alexander Afanasyev1a0fff62013-01-19 14:29:51 -080036#include "ns3/ndnSIM/utils/ndn-fw-hop-count-tag.h"
Alexander Afanasyevb989b122013-07-10 17:15:46 -070037#include "ns3/ndnSIM/model/wire/ndnsim.h"
Alexander Afanasyev5ab2bcd2013-07-15 18:21:19 -070038#include "ns3/ndnSIM/model/wire/ccnb.h"
Alexander Afanasyev8e0d2812012-01-19 22:38:14 -080039
Alexander Afanasyev23d2b542011-12-07 18:54:46 -080040#include <boost/ref.hpp>
41
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -070042NS_LOG_COMPONENT_DEFINE ("ndn.Face");
Alexander Afanasyev08d984e2011-08-13 19:20:22 -070043
44namespace ns3 {
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -070045namespace ndn {
Alexander Afanasyev08d984e2011-08-13 19:20:22 -070046
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -070047NS_OBJECT_ENSURE_REGISTERED (Face);
Alexander Afanasyev30c33e32012-06-05 21:28:44 -070048
Alexander Afanasyevbdc0d982011-12-16 01:15:26 -080049TypeId
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -070050Face::GetTypeId ()
Alexander Afanasyevbdc0d982011-12-16 01:15:26 -080051{
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -070052 static TypeId tid = TypeId ("ns3::ndn::Face")
Alexander Afanasyevbdc0d982011-12-16 01:15:26 -080053 .SetParent<Object> ()
Alexander Afanasyev4aac5572012-08-09 10:49:55 -070054 .SetGroupName ("Ndn")
55 .AddAttribute ("Id", "Face id (unique integer for the Ndn stack on this node)",
Alexander Afanasyevcbe92ae2011-12-16 13:06:18 -080056 TypeId::ATTR_GET, // allow only getting it.
57 UintegerValue (0),
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -070058 MakeUintegerAccessor (&Face::m_id),
Alexander Afanasyevcbe92ae2011-12-16 13:06:18 -080059 MakeUintegerChecker<uint32_t> ())
Alexander Afanasyev5ab2bcd2013-07-15 18:21:19 -070060
61 .AddAttribute ("WireFormat",
62 "Default wire format for the face. The face will be accepting packets "
63 "in any supported packet formats, but if encoding requested, it will "
64 "use default wire format to encode (0 for ndnSIM (default), 1 for CCNb)",
65 UintegerValue (WIRE_FORMAT_NDNSIM),
66 MakeUintegerAccessor (&Face::m_wireFormat),
67 MakeUintegerChecker<uint32_t> ())
Alexander Afanasyevbdc0d982011-12-16 01:15:26 -080068 ;
69 return tid;
70}
71
Alexander Afanasyev91d7c7c2013-01-31 15:01:08 -080072/**
Alexander Afanasyev4aac5572012-08-09 10:49:55 -070073 * By default, Ndn face are created in the "down" state
Alexander Afanasyev91d7c7c2013-01-31 15:01:08 -080074 * with no IP addresses. Before becoming useable, the user must
Alexander Afanasyev4aac5572012-08-09 10:49:55 -070075 * invoke SetUp on them once an Ndn address and mask have been set.
Alexander Afanasyev08d984e2011-08-13 19:20:22 -070076 */
Alexander Afanasyev91d7c7c2013-01-31 15:01:08 -080077Face::Face (Ptr<Node> node)
Alexander Afanasyev19426ef2011-11-23 20:55:28 -080078 : m_node (node)
Alexander Afanasyevfaa01f92013-07-10 18:34:31 -070079 , m_upstreamInterestHandler (MakeNullCallback< void, Ptr<Face>, Ptr<Interest> > ())
80 , m_upstreamDataHandler (MakeNullCallback< void, Ptr<Face>, Ptr<ContentObject> > ())
Alexander Afanasyev56f79ea2011-08-17 23:54:27 -070081 , m_ifup (false)
82 , m_id ((uint32_t)-1)
Alexander Afanasyev8e0d2812012-01-19 22:38:14 -080083 , m_metric (0)
Alexander Afanasyevdd9fa4f2013-05-15 16:35:04 -070084 , m_flags (0)
Alexander Afanasyev08d984e2011-08-13 19:20:22 -070085{
86 NS_LOG_FUNCTION (this);
Alexander Afanasyev19426ef2011-11-23 20:55:28 -080087
88 NS_ASSERT_MSG (node != 0, "node cannot be NULL. Check the code");
Alexander Afanasyev08d984e2011-08-13 19:20:22 -070089}
90
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -070091Face::~Face ()
Alexander Afanasyev08d984e2011-08-13 19:20:22 -070092{
93 NS_LOG_FUNCTION_NOARGS ();
94}
95
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -070096Face::Face (const Face &)
Alexander Afanasyev56f79ea2011-08-17 23:54:27 -070097{
98}
99
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -0700100Face& Face::operator= (const Face &)
Alexander Afanasyev56f79ea2011-08-17 23:54:27 -0700101{
102 return *this;
103}
104
Alexander Afanasyeve9c9d722012-01-19 16:59:30 -0800105Ptr<Node>
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -0700106Face::GetNode () const
Alexander Afanasyeve9c9d722012-01-19 16:59:30 -0800107{
108 return m_node;
109}
110
Alexander Afanasyev09c7deb2011-11-23 14:50:10 -0800111void
Alexander Afanasyev5bee19e2013-07-10 14:33:57 -0700112Face::RegisterProtocolHandlers (const InterestHandler &interestHandler, const DataHandler &dataHandler)
Alexander Afanasyev08d984e2011-08-13 19:20:22 -0700113{
Alexander Afanasyev19426ef2011-11-23 20:55:28 -0800114 NS_LOG_FUNCTION_NOARGS ();
115
Alexander Afanasyev5bee19e2013-07-10 14:33:57 -0700116 m_upstreamInterestHandler = interestHandler;
117 m_upstreamDataHandler = dataHandler;
118}
119
120void
121Face::UnRegisterProtocolHandlers ()
122{
123 NS_LOG_FUNCTION_NOARGS ();
124
Alexander Afanasyevfaa01f92013-07-10 18:34:31 -0700125 m_upstreamInterestHandler = MakeNullCallback< void, Ptr<Face>, Ptr<Interest> > ();
126 m_upstreamDataHandler = MakeNullCallback< void, Ptr<Face>, Ptr<ContentObject> > ();
Alexander Afanasyev5bee19e2013-07-10 14:33:57 -0700127}
128
129
130bool
Alexander Afanasyevb989b122013-07-10 17:15:46 -0700131Face::SendInterest (Ptr<const Interest> interest)
Alexander Afanasyev5bee19e2013-07-10 14:33:57 -0700132{
Alexander Afanasyeve4795ae2013-07-11 20:01:31 -0700133 NS_LOG_FUNCTION (this << boost::cref (*this) << interest->GetName ());
Alexander Afanasyev5bee19e2013-07-10 14:33:57 -0700134
135 if (!IsUp ())
136 {
137 return false;
138 }
139
Alexander Afanasyev5ab2bcd2013-07-15 18:21:19 -0700140 Ptr<Packet> packet;
141 if (m_wireFormat == WIRE_FORMAT_NDNSIM)
142 packet = wire::ndnSIM::Interest::ToWire (interest);
143 else if (m_wireFormat == WIRE_FORMAT_CCNB)
144 packet = wire::ccnb::Interest::ToWire (interest);
145 else
146 {
147 NS_FATAL_ERROR ("Unsupported format requested");
148 }
Alexander Afanasyevb989b122013-07-10 17:15:46 -0700149 return Send (packet);
Alexander Afanasyev5bee19e2013-07-10 14:33:57 -0700150}
151
152bool
Alexander Afanasyevb989b122013-07-10 17:15:46 -0700153Face::SendData (Ptr<const ContentObject> data)
Alexander Afanasyev5bee19e2013-07-10 14:33:57 -0700154{
Alexander Afanasyevfaa01f92013-07-10 18:34:31 -0700155 NS_LOG_FUNCTION (this << data);
Alexander Afanasyev5bee19e2013-07-10 14:33:57 -0700156
157 if (!IsUp ())
158 {
159 return false;
160 }
161
Alexander Afanasyev5ab2bcd2013-07-15 18:21:19 -0700162 Ptr<Packet> packet;
163 if (m_wireFormat == WIRE_FORMAT_NDNSIM)
164 packet = wire::ndnSIM::Data::ToWire (data);
165 else if (m_wireFormat == WIRE_FORMAT_CCNB)
166 packet = wire::ccnb::Data::ToWire (data);
167 else
168 {
169 NS_FATAL_ERROR ("Unsupported format requested");
170 }
Alexander Afanasyevb989b122013-07-10 17:15:46 -0700171 return Send (packet);
Alexander Afanasyev09c7deb2011-11-23 14:50:10 -0800172}
173
174bool
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -0700175Face::Send (Ptr<Packet> packet)
Alexander Afanasyev09c7deb2011-11-23 14:50:10 -0800176{
Alexander Afanasyev1a0fff62013-01-19 14:29:51 -0800177 FwHopCountTag hopCount;
178 bool tagExists = packet->RemovePacketTag (hopCount);
179 if (tagExists)
180 {
181 hopCount.Increment ();
182 packet->AddPacketTag (hopCount);
183 }
Alexander Afanasyev30c33e32012-06-05 21:28:44 -0700184
Alexander Afanasyev5bee19e2013-07-10 14:33:57 -0700185 return true;
Alexander Afanasyev09c7deb2011-11-23 14:50:10 -0800186}
187
188bool
Alexander Afanasyev5bee19e2013-07-10 14:33:57 -0700189Face::Receive (Ptr<const Packet> p)
Alexander Afanasyev09c7deb2011-11-23 14:50:10 -0800190{
Alexander Afanasyevfaa01f92013-07-10 18:34:31 -0700191 NS_LOG_FUNCTION (this << p << p->GetSize ());
Alexander Afanasyev19426ef2011-11-23 20:55:28 -0800192
Alexander Afanasyev09c7deb2011-11-23 14:50:10 -0800193 if (!IsUp ())
Alexander Afanasyev30c33e32012-06-05 21:28:44 -0700194 {
195 // no tracing here. If we were off while receiving, we shouldn't even know that something was there
196 return false;
197 }
Alexander Afanasyev09c7deb2011-11-23 14:50:10 -0800198
Alexander Afanasyev5bee19e2013-07-10 14:33:57 -0700199 Ptr<Packet> packet = p->Copy (); // give upper layers a rw copy of the packet
200 try
201 {
202 HeaderHelper::Type type = HeaderHelper::GetNdnHeaderType (packet);
203 switch (type)
204 {
205 case HeaderHelper::INTEREST_NDNSIM:
206 {
Alexander Afanasyevb989b122013-07-10 17:15:46 -0700207 Ptr<Interest> interest = wire::ndnSIM::Interest::FromWire (packet);
208 return ReceiveInterest (interest);
Alexander Afanasyev5bee19e2013-07-10 14:33:57 -0700209 }
210 case HeaderHelper::CONTENT_OBJECT_NDNSIM:
211 {
Alexander Afanasyevb989b122013-07-10 17:15:46 -0700212 Ptr<ContentObject> data = wire::ndnSIM::Data::FromWire (packet);
213 return ReceiveData (data);
Alexander Afanasyev5bee19e2013-07-10 14:33:57 -0700214 }
215 case HeaderHelper::INTEREST_CCNB:
Alexander Afanasyev5ab2bcd2013-07-15 18:21:19 -0700216 {
217 Ptr<Interest> interest = wire::ccnb::Interest::FromWire (packet);
218 return ReceiveInterest (interest);
219 }
Alexander Afanasyev5bee19e2013-07-10 14:33:57 -0700220 case HeaderHelper::CONTENT_OBJECT_CCNB:
Alexander Afanasyev5ab2bcd2013-07-15 18:21:19 -0700221 {
222 Ptr<ContentObject> data = wire::ccnb::Data::FromWire (packet);
223 return ReceiveData (data);
224 }
Alexander Afanasyev5bee19e2013-07-10 14:33:57 -0700225 }
226
227 // exception will be thrown if packet is not recognized
228 }
229 catch (UnknownHeaderException)
230 {
231 NS_ASSERT_MSG (false, "Unknown NDN header. Should not happen");
232 NS_LOG_ERROR ("Unknown NDN header. Should not happen");
233 return false;
234 }
235}
236
237bool
Alexander Afanasyevb989b122013-07-10 17:15:46 -0700238Face::ReceiveInterest (Ptr<Interest> interest)
Alexander Afanasyev5bee19e2013-07-10 14:33:57 -0700239{
240 if (!IsUp ())
241 {
242 // no tracing here. If we were off while receiving, we shouldn't even know that something was there
243 return false;
244 }
245
Alexander Afanasyevb989b122013-07-10 17:15:46 -0700246 m_upstreamInterestHandler (this, interest);
Alexander Afanasyev5bee19e2013-07-10 14:33:57 -0700247 return true;
248}
249
250bool
Alexander Afanasyevb989b122013-07-10 17:15:46 -0700251Face::ReceiveData (Ptr<ContentObject> data)
Alexander Afanasyev5bee19e2013-07-10 14:33:57 -0700252{
253 if (!IsUp ())
254 {
255 // no tracing here. If we were off while receiving, we shouldn't even know that something was there
256 return false;
257 }
258
Alexander Afanasyevb989b122013-07-10 17:15:46 -0700259 m_upstreamDataHandler (this, data);
Alexander Afanasyev19426ef2011-11-23 20:55:28 -0800260 return true;
Alexander Afanasyev08d984e2011-08-13 19:20:22 -0700261}
262
Alexander Afanasyevcbe92ae2011-12-16 13:06:18 -0800263void
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -0700264Face::SetMetric (uint16_t metric)
Alexander Afanasyev8e0d2812012-01-19 22:38:14 -0800265{
266 NS_LOG_FUNCTION (metric);
267 m_metric = metric;
268}
Alexander Afanasyev08d984e2011-08-13 19:20:22 -0700269
Alexander Afanasyev8e0d2812012-01-19 22:38:14 -0800270uint16_t
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -0700271Face::GetMetric (void) const
Alexander Afanasyev8e0d2812012-01-19 22:38:14 -0800272{
Alexander Afanasyev8e0d2812012-01-19 22:38:14 -0800273 return m_metric;
274}
Alexander Afanasyev08d984e2011-08-13 19:20:22 -0700275
Alexander Afanasyevdd9fa4f2013-05-15 16:35:04 -0700276void
277Face::SetFlags (uint32_t flags)
278{
279 m_flags = flags;
280}
281
Alexander Afanasyev56f79ea2011-08-17 23:54:27 -0700282bool
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -0700283Face::operator== (const Face &face) const
Alexander Afanasyev08d984e2011-08-13 19:20:22 -0700284{
Alexander Afanasyeva46844b2011-11-21 19:13:26 -0800285 NS_ASSERT_MSG (m_node->GetId () == face.m_node->GetId (),
Alexander Afanasyev91d7c7c2013-01-31 15:01:08 -0800286 "Faces of different nodes should not be compared to each other: " << *this << " == " << face);
Alexander Afanasyev7fd74f92011-08-25 19:40:17 -0700287
288 return (m_id == face.m_id);
289}
290
291bool
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -0700292Face::operator< (const Face &face) const
Alexander Afanasyev7fd74f92011-08-25 19:40:17 -0700293{
Alexander Afanasyeva46844b2011-11-21 19:13:26 -0800294 NS_ASSERT_MSG (m_node->GetId () == face.m_node->GetId (),
Alexander Afanasyev91d7c7c2013-01-31 15:01:08 -0800295 "Faces of different nodes should not be compared to each other: " << *this << " == " << face);
Alexander Afanasyev7fd74f92011-08-25 19:40:17 -0700296
297 return (m_id < face.m_id);
Alexander Afanasyev08d984e2011-08-13 19:20:22 -0700298}
299
Alexander Afanasyevc5a23e22011-09-07 00:37:36 -0700300std::ostream&
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -0700301Face::Print (std::ostream &os) const
Alexander Afanasyevc5a23e22011-09-07 00:37:36 -0700302{
303 os << "id=" << GetId ();
304 return os;
305}
306
Alexander Afanasyev4aac5572012-08-09 10:49:55 -0700307std::ostream&
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -0700308operator<< (std::ostream& os, const Face &face)
Alexander Afanasyev98256102011-08-14 01:00:02 -0700309{
Alexander Afanasyevc5a23e22011-09-07 00:37:36 -0700310 face.Print (os);
Alexander Afanasyev98256102011-08-14 01:00:02 -0700311 return os;
312}
313
Alexander Afanasyev2b4c9472012-08-09 15:00:38 -0700314} // namespace ndn
315} // namespace ns3
Alexander Afanasyev08d984e2011-08-13 19:20:22 -0700316