A Discrete-Event Network Simulator
API
mesh-point-device.cc
Go to the documentation of this file.
1 /* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
2 /*
3  * Copyright (c) 2008,2009 IITP RAS
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: Kirill Andreev <andreev@iitp.ru>
19  * Pavel Boyko <boyko@iitp.ru>
20  */
21 
22 #include "ns3/node.h"
23 #include "ns3/packet.h"
24 #include "ns3/log.h"
25 #include "ns3/pointer.h"
26 #include "ns3/mesh-point-device.h"
27 #include "ns3/wifi-net-device.h"
28 #include "ns3/mesh-wifi-interface-mac.h"
29 
30 namespace ns3 {
31 
32 NS_LOG_COMPONENT_DEFINE ("MeshPointDevice");
33 
34 NS_OBJECT_ENSURE_REGISTERED (MeshPointDevice);
35 
36 TypeId
38 {
39  static TypeId tid = TypeId ("ns3::MeshPointDevice")
40  .SetParent<NetDevice> ()
41  .AddConstructor<MeshPointDevice> ()
42  .AddAttribute ("Mtu", "The MAC-level Maximum Transmission Unit",
43  UintegerValue (0xffff),
46  MakeUintegerChecker<uint16_t> ())
47  .AddAttribute ( "RoutingProtocol",
48  "The mesh routing protocol used by this mesh point.",
49  PointerValue (),
54  return tid;
55 }
56 
58  m_ifIndex (0)
59 {
60  NS_LOG_FUNCTION (this);
61  m_channel = CreateObject<BridgeChannel> ();
62 }
63 
65 {
66  NS_LOG_FUNCTION (this);
67  m_node = 0;
68  m_channel = 0;
70 }
71 
72 void
74 {
75  NS_LOG_FUNCTION (this);
76  for (std::vector<Ptr<NetDevice> >::iterator iter = m_ifaces.begin (); iter != m_ifaces.end (); iter++)
77  {
78  *iter = 0;
79  }
80  m_ifaces.clear ();
81  m_node = 0;
82  m_channel = 0;
85 
86 }
87 
88 //-----------------------------------------------------------------------------
89 // NetDevice interface implementation
90 //-----------------------------------------------------------------------------
91 
92 void
93 MeshPointDevice::ReceiveFromDevice (Ptr<NetDevice> incomingPort, Ptr<const Packet> packet, uint16_t protocol,
94  Address const &src, Address const &dst, PacketType packetType)
95 {
96  NS_LOG_FUNCTION (this << incomingPort << packet);
97  NS_LOG_DEBUG ("UID is " << packet->GetUid ());
98  const Mac48Address src48 = Mac48Address::ConvertFrom (src);
99  const Mac48Address dst48 = Mac48Address::ConvertFrom (dst);
100  uint16_t& realProtocol = protocol;
101  NS_LOG_DEBUG ("SRC=" << src48 << ", DST = " << dst48 << ", I am: " << m_address);
102  if (!m_promiscRxCallback.IsNull ())
103  {
104  m_promiscRxCallback (this, packet, protocol, src, dst, packetType);
105  }
106  if (dst48.IsGroup ())
107  {
108  Ptr<Packet> packet_copy = packet->Copy ();
109  if (m_routingProtocol->RemoveRoutingStuff (incomingPort->GetIfIndex (), src48, dst48, packet_copy, realProtocol))
110  {
111  m_rxCallback (this, packet_copy, realProtocol, src);
112  Forward (incomingPort, packet, protocol, src48, dst48);
113 
115  m_rxStats.broadcastDataBytes += packet->GetSize ();
116  }
117  return;
118  }
119  if (dst48 == m_address)
120  {
121  Ptr<Packet> packet_copy = packet->Copy ();
122  if (m_routingProtocol->RemoveRoutingStuff (incomingPort->GetIfIndex (), src48, dst48, packet_copy, realProtocol))
123  {
124  m_rxCallback (this, packet_copy, realProtocol, src);
126  m_rxStats.unicastDataBytes += packet->GetSize ();
127  }
128  return;
129  }
130  else
131  Forward (incomingPort, packet->Copy (), protocol, src48, dst48);
132 }
133 
134 void
135 MeshPointDevice::Forward (Ptr<NetDevice> inport, Ptr<const Packet> packet, uint16_t protocol,
136  const Mac48Address src, const Mac48Address dst)
137 {
138  // pass through routing protocol
139  m_routingProtocol->RequestRoute (inport->GetIfIndex (), src, dst, packet, protocol, MakeCallback (
140  &MeshPointDevice::DoSend, this));
141 }
142 
143 void
144 MeshPointDevice::SetIfIndex (const uint32_t index)
145 {
146  NS_LOG_FUNCTION (this);
147  m_ifIndex = index;
148 }
149 
150 uint32_t
152 {
153  NS_LOG_FUNCTION (this);
154  return m_ifIndex;
155 }
156 
159 {
160  NS_LOG_FUNCTION (this);
161  return m_channel;
162 }
163 
164 Address
166 {
167  NS_LOG_FUNCTION (this);
168  return m_address;
169 }
170 
171 void
173 {
174  NS_LOG_FUNCTION (this);
175  NS_LOG_WARN ("Manual changing mesh point address can cause routing errors.");
177 }
178 
179 bool
180 MeshPointDevice::SetMtu (const uint16_t mtu)
181 {
182  NS_LOG_FUNCTION (this);
183  m_mtu = mtu;
184  return true;
185 }
186 
187 uint16_t
189 {
190  NS_LOG_FUNCTION (this);
191  return m_mtu;
192 }
193 
194 bool
196 {
197  NS_LOG_FUNCTION (this);
198  return true;
199 }
200 
201 void
203 {
204  NS_LOG_FUNCTION (this);
205  // do nothing
206  NS_LOG_WARN ("AddLinkChangeCallback does nothing");
207 }
208 
209 bool
211 {
212  NS_LOG_FUNCTION (this);
213  return true;
214 }
215 
216 Address
218 {
219  NS_LOG_FUNCTION (this);
220  return Mac48Address ("ff:ff:ff:ff:ff:ff");
221 }
222 
223 bool
225 {
226  NS_LOG_FUNCTION (this);
227  return true;
228 }
229 
230 Address
232 {
233  NS_LOG_FUNCTION (this << multicastGroup);
234  Mac48Address multicast = Mac48Address::GetMulticast (multicastGroup);
235  return multicast;
236 }
237 
238 bool
240 {
241  NS_LOG_FUNCTION (this);
242  return false;
243 }
244 
245 bool
247 {
248  NS_LOG_FUNCTION (this);
249  return false;
250 }
251 
252 bool
253 MeshPointDevice::Send (Ptr<Packet> packet, const Address& dest, uint16_t protocolNumber)
254 {
255  NS_LOG_FUNCTION (this);
256  const Mac48Address dst48 = Mac48Address::ConvertFrom (dest);
257  return m_routingProtocol->RequestRoute (m_ifIndex, m_address, dst48, packet, protocolNumber, MakeCallback (
258  &MeshPointDevice::DoSend, this));
259 }
260 
261 bool
262 MeshPointDevice::SendFrom (Ptr<Packet> packet, const Address& src, const Address& dest,
263  uint16_t protocolNumber)
264 {
265  NS_LOG_FUNCTION (this);
266  const Mac48Address src48 = Mac48Address::ConvertFrom (src);
267  const Mac48Address dst48 = Mac48Address::ConvertFrom (dest);
268  return m_routingProtocol->RequestRoute (m_ifIndex, src48, dst48, packet, protocolNumber, MakeCallback (
269  &MeshPointDevice::DoSend, this));
270 }
271 
272 Ptr<Node>
274 {
275  NS_LOG_FUNCTION (this);
276  return m_node;
277 }
278 
279 void
281 {
282  NS_LOG_FUNCTION (this);
283  m_node = node;
284 }
285 
286 bool
288 {
289  NS_LOG_FUNCTION (this);
290  return true;
291 }
292 
293 void
295 {
296  NS_LOG_FUNCTION (this);
297  m_rxCallback = cb;
298 }
299 
300 void
302 {
303  NS_LOG_FUNCTION (this);
304  m_promiscRxCallback = cb;
305 }
306 
307 bool
309 {
310  NS_LOG_FUNCTION (this);
311  return false; // don't allow to bridge mesh network with something else.
312 }
313 
314 Address
316 {
317  NS_LOG_FUNCTION (this << addr);
318  return Mac48Address::GetMulticast (addr);
319 }
320 
321 //-----------------------------------------------------------------------------
322 // Interfaces
323 //-----------------------------------------------------------------------------
324 uint32_t
326 {
327  NS_LOG_FUNCTION (this);
328  return m_ifaces.size ();
329 }
330 
333 {
334  NS_LOG_FUNCTION (this << n);
335  for (std::vector<Ptr<NetDevice> >::const_iterator i = m_ifaces.begin (); i != m_ifaces.end (); i++)
336  {
337  if ((*i)->GetIfIndex () == n)
338  {
339  return (*i);
340  }
341  }
342  NS_FATAL_ERROR ("Mesh point interface is not found by index");
343  return 0;
344 }
345 std::vector<Ptr<NetDevice> >
347 {
348  return m_ifaces;
349 }
350 void
352 {
353  NS_LOG_FUNCTION (this << iface);
354 
355  NS_ASSERT (iface != this);
356  if (!Mac48Address::IsMatchingType (iface->GetAddress ()))
357  {
358  NS_FATAL_ERROR ("Device does not support eui 48 addresses: cannot be used as a mesh point interface.");
359  }
360  if (!iface->SupportsSendFrom ())
361  {
362  NS_FATAL_ERROR ("Device does not support SendFrom: cannot be used as a mesh point interface.");
363  }
364 
365  // Mesh point has MAC address of it's first interface
366  if (m_ifaces.empty ())
367  {
368  m_address = Mac48Address::ConvertFrom (iface->GetAddress ());
369  }
370  Ptr<WifiNetDevice> wifiNetDev = iface->GetObject<WifiNetDevice> ();
371  if (wifiNetDev == 0)
372  {
373  NS_FATAL_ERROR ("Device is not a WiFi NIC: cannot be used as a mesh point interface.");
374  }
375  Ptr<MeshWifiInterfaceMac> ifaceMac = wifiNetDev->GetMac ()->GetObject<MeshWifiInterfaceMac> ();
376  if (ifaceMac == 0)
377  {
379  "WiFi device doesn't have correct MAC installed: cannot be used as a mesh point interface.");
380  }
381  ifaceMac->SetMeshPointAddress (m_address);
382 
383  // Receive frames from this interface
384  m_node->RegisterProtocolHandler (MakeCallback (&MeshPointDevice::ReceiveFromDevice, this), 0, iface, /*promiscuous = */
385  true);
386  m_ifaces.push_back (iface);
387  m_channel->AddChannel (iface->GetChannel ());
388 }
389 
390 //-----------------------------------------------------------------------------
391 // Protocols
392 //-----------------------------------------------------------------------------
393 
394 void
396 {
397  NS_LOG_FUNCTION (this << protocol);
398  NS_ASSERT_MSG (PeekPointer (protocol->GetMeshPoint ()) == this,
399  "Routing protocol must be installed on mesh point to be useful.");
400  m_routingProtocol = protocol;
401 }
402 
405 {
406  NS_LOG_FUNCTION (this);
407  return m_routingProtocol;
408 }
409 
410 void
412  uint16_t protocol, uint32_t outIface)
413 {
414  NS_LOG_FUNCTION (this << success << packet << src << dst << protocol << outIface);
415  if (!success)
416  {
417  NS_LOG_DEBUG ("Resolve failed");
418  return;
419  }
420 
421  // Count statistics
422  Statistics * stats = ((src == m_address) ? &m_txStats : &m_fwdStats);
423 
424  if (dst.IsBroadcast ())
425  {
426  stats->broadcastData++;
427  stats->broadcastDataBytes += packet->GetSize ();
428  }
429  else
430  {
431  stats->unicastData++;
432  stats->unicastDataBytes += packet->GetSize ();
433  }
434 
435  // Send
436  if (outIface != 0xffffffff)
437  {
438  GetInterface (outIface)->SendFrom (packet, src, dst, protocol);
439  }
440  else
441  {
442  for (std::vector<Ptr<NetDevice> >::iterator i = m_ifaces.begin (); i != m_ifaces.end (); i++)
443  {
444  (*i)->SendFrom (packet->Copy (), src, dst, protocol);
445  }
446  }
447 }
449  unicastData (0), unicastDataBytes (0), broadcastData (0), broadcastDataBytes (0)
450 {
451  NS_LOG_FUNCTION (this);
452 }
453 
454 void
455 MeshPointDevice::Report (std::ostream & os) const
456 {
457  NS_LOG_FUNCTION (this);
458  os << "<Statistics" << std::endl <<
459  "txUnicastData=\"" << m_txStats.unicastData << "\"" << std::endl <<
460  "txUnicastDataBytes=\"" << m_txStats.unicastDataBytes << "\"" << std::endl <<
461  "txBroadcastData=\"" << m_txStats.broadcastData << "\"" << std::endl <<
462  "txBroadcastDataBytes=\"" << m_txStats.broadcastDataBytes << "\"" << std::endl <<
463  "rxUnicastData=\"" << m_rxStats.unicastData << "\"" << std::endl <<
464  "rxUnicastDataBytes=\"" << m_rxStats.unicastDataBytes << "\"" << std::endl <<
465  "rxBroadcastData=\"" << m_rxStats.broadcastData << "\"" << std::endl <<
466  "rxBroadcastDataBytes=\"" << m_rxStats.broadcastDataBytes << "\"" << std::endl <<
467  "fwdUnicastData=\"" << m_fwdStats.unicastData << "\"" << std::endl <<
468  "fwdUnicastDataBytes=\"" << m_fwdStats.unicastDataBytes << "\"" << std::endl <<
469  "fwdBroadcastData=\"" << m_fwdStats.broadcastData << "\"" << std::endl <<
470  "fwdBroadcastDataBytes=\"" << m_fwdStats.broadcastDataBytes << "\"" << std::endl <<
471  "/>" << std::endl;
472 }
473 
474 void
476 {
477  NS_LOG_FUNCTION (this);
478  m_rxStats = Statistics ();
479  m_txStats = Statistics ();
480  m_fwdStats = Statistics ();
481 }
482 
483 } // namespace ns3
uint32_t GetNInterfaces() const
static bool IsMatchingType(const Address &address)
Mac48Address m_address
Mesh point MAC address, supposed to be the address of the first added interface.
Smart pointer class similar to boost::intrusive_ptr.
Definition: ptr.h:73
#define NS_LOG_FUNCTION(parameters)
If log level LOG_FUNCTION is enabled, this macro will output all input parameters separated by "...
virtual bool IsBridge() const
Return true if the net device is acting as a bridge.
#define NS_OBJECT_ENSURE_REGISTERED(type)
Register an Object subclass with the TypeId system.
Definition: object-base.h:44
NetDevice::ReceiveCallback m_rxCallback
Receive action.
Ptr< AttributeChecker > MakePointerChecker(void)
Definition: pointer.h:214
void ResetStats()
Reset statistics counters.
virtual bool IsBroadcast() const
uint64_t GetUid(void) const
Returns the packet's Uid.
Definition: packet.cc:380
virtual bool SetMtu(const uint16_t mtu)
bool IsNull(void) const
Check for null implementation.
Definition: callback.h:1072
virtual ~MeshPointDevice()
D-tor.
#define NS_ASSERT(condition)
At runtime, in debugging builds, if this condition is not true, the program prints the source file...
Definition: assert.h:61
PacketType
Packet types are used as they are in Linux.
Definition: net-device.h:272
U * PeekPointer(const Ptr< U > &p)
Definition: ptr.h:562
#define NS_LOG_COMPONENT_DEFINE(name)
Define a Log component with a specific name.
Definition: log.h:201
uint32_t GetSize(void) const
Returns the the size in bytes of the packet (including the zero-filled initial payload).
Definition: packet.h:766
bool IsBroadcast(void) const
virtual void AddLinkChangeCallback(Callback< void > callback)
Statistics m_txStats
Counters.
virtual void DoDispose(void)
Destructor implementation.
Definition: object.cc:338
#define NS_FATAL_ERROR(msg)
Fatal error handling.
Definition: fatal-error.h:100
virtual bool IsLinkUp() const
virtual bool IsMulticast() const
Ptr< MeshL2RoutingProtocol > m_routingProtocol
Current routing protocol, used mainly by GetRoutingProtocol.
virtual void DoDispose()
Destructor implementation.
Ptr< BridgeChannel > m_channel
Virtual channel for upper layers.
virtual void SetReceiveCallback(NetDevice::ReceiveCallback cb)
a polymophic address class
Definition: address.h:90
virtual Address GetAddress() const
std::vector< Ptr< NetDevice > > GetInterfaces() const
virtual void SetAddress(Address a)
Set the address of this interface.
Ptr< NetDevice > GetInterface(uint32_t id) const
void SetRoutingProtocol(Ptr< MeshL2RoutingProtocol > protocol)
Register routing protocol to be used. Protocol must be already installed on this mesh point...
Statistics m_rxStats
Counters.
Ptr< const AttributeAccessor > MakePointerAccessor(T1 a1)
Create an AttributeAccessor for a class data member, or a lone class get functor or set method...
Definition: pointer.h:210
static Mac48Address GetMulticast(Ipv4Address address)
void Forward(Ptr< NetDevice > incomingPort, Ptr< const Packet > packet, uint16_t protocol, const Mac48Address src, const Mac48Address dst)
Forward packet down to interfaces.
Hold an unsigned integer type.
Definition: uinteger.h:44
Hold together all Wifi-related objects.
Callback< R > MakeCallback(R(T::*memPtr)(void), OBJ objPtr)
Definition: callback.h:1290
virtual uint32_t GetIfIndex() const
virtual Address GetMulticast(Ipv4Address multicastGroup) const
Make and return a MAC multicast address using the provided multicast group.
virtual Ptr< Node > GetNode() const
virtual uint16_t GetMtu() const
void AddInterface(Ptr< NetDevice > port)
Attach new interface to the station.
virtual void SetNode(Ptr< Node > node)
static Mac48Address ConvertFrom(const Address &address)
void Report(std::ostream &os) const
Print statistics counters.
Ptr< Packet > Copy(void) const
performs a COW copy of the packet.
Definition: packet.cc:122
uint16_t m_mtu
MTU in bytes.
Every class exported by the ns3 library is enclosed in the ns3 namespace.
Hold objects of type Ptr.
Definition: pointer.h:36
virtual bool SendFrom(Ptr< Packet > packet, const Address &source, const Address &dest, uint16_t protocolNumber)
virtual void SetPromiscReceiveCallback(NetDevice::PromiscReceiveCallback cb)
bool IsGroup(void) const
an EUI-48 address
Definition: mac48-address.h:43
Statistics m_fwdStats
Counters.
Ptr< Node > m_node
Parent node.
void DoSend(bool success, Ptr< Packet > packet, Mac48Address src, Mac48Address dst, uint16_t protocol, uint32_t iface)
Response callback for L2 routing protocol.
virtual Address GetBroadcast() const
static TypeId GetTypeId()
Object type ID for NS3 object system.
virtual Ptr< Channel > GetChannel() const
uint32_t m_ifIndex
If index.
Interface for L2 mesh routing protocol and mesh point communication.
void RegisterProtocolHandler(ProtocolHandler handler, uint16_t protocolType, Ptr< NetDevice > device, bool promiscuous=false)
Definition: node.cc:219
#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:84
MeshPointDevice()
C-tor create empty (without interfaces and protocols) mesh point.
Describes an IPv6 address.
Definition: ipv6-address.h:47
Ipv4 addresses are stored in host order in this class.
Definition: ipv4-address.h:40
Network layer to device interface.
Definition: net-device.h:75
#define NS_LOG_WARN(msg)
Use NS_LOG to output a message of level LOG_WARN.
Definition: log.h:228
virtual bool NeedsArp() const
virtual bool IsPointToPoint() const
Return true if the net device is on a point-to-point link.
#define NS_LOG_DEBUG(msg)
Use NS_LOG to output a message of level LOG_DEBUG.
Definition: log.h:236
std::vector< Ptr< NetDevice > > m_ifaces
List of interfaces.
virtual void SetIfIndex(const uint32_t index)
virtual bool Send(Ptr< Packet > packet, const Address &dest, uint16_t protocolNumber)
void ReceiveFromDevice(Ptr< NetDevice > device, Ptr< const Packet > packet, uint16_t protocol, Address const &source, Address const &destination, PacketType packetType)
Receive packet from interface.
NetDevice::PromiscReceiveCallback m_promiscRxCallback
Promisc receive action.
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:51
virtual bool SupportsSendFrom() const
TypeId SetParent(TypeId tid)
Definition: type-id.cc:631
Basic MAC of mesh point Wi-Fi interface.
Ptr< MeshL2RoutingProtocol > GetRoutingProtocol() const
Access current routing protocol.