A Discrete-Event Network Simulator
API
 All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Properties Friends Macros Groups Pages
ping6.cc
Go to the documentation of this file.
1 /* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
2 /*
3  * Copyright (c) 2007-2009 Strasbourg University
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: Sebastien Vincent <vincent@clarinet.u-strasbg.fr>
19  */
20 
21 #include "ns3/log.h"
22 #include "ns3/nstime.h"
23 #include "ns3/simulator.h"
24 #include "ns3/socket-factory.h"
25 #include "ns3/packet.h"
26 #include "ns3/socket.h"
27 #include "ns3/uinteger.h"
28 #include "ns3/ipv6.h"
29 #include "ns3/ipv6-address.h"
30 #include "ns3/inet6-socket-address.h"
31 #include "ns3/icmpv6-header.h"
32 #include "ns3/ipv6-raw-socket-factory.h"
33 #include "ns3/ipv6-header.h"
34 #include "ns3/ipv6-extension-header.h"
35 
36 #include "ping6.h"
37 
38 namespace ns3
39 {
40 
41 NS_LOG_COMPONENT_DEFINE ("Ping6Application");
42 
44 
46 {
47  static TypeId tid = TypeId ("ns3::Ping6")
49  .AddConstructor<Ping6>()
50  .AddAttribute ("MaxPackets",
51  "The maximum number of packets the application will send",
52  UintegerValue (100),
53  MakeUintegerAccessor (&Ping6::m_count),
54  MakeUintegerChecker<uint32_t>())
55  .AddAttribute ("Interval",
56  "The time to wait between packets",
57  TimeValue (Seconds (1.0)),
58  MakeTimeAccessor (&Ping6::m_interval),
59  MakeTimeChecker ())
60  .AddAttribute ("RemoteIpv6",
61  "The Ipv6Address of the outbound packets",
63  MakeIpv6AddressAccessor (&Ping6::m_peerAddress),
64  MakeIpv6AddressChecker ())
65  .AddAttribute ("LocalIpv6",
66  "Local Ipv6Address of the sender",
68  MakeIpv6AddressAccessor (&Ping6::m_localAddress),
69  MakeIpv6AddressChecker ())
70  .AddAttribute ("PacketSize",
71  "Size of packets generated",
72  UintegerValue (100),
73  MakeUintegerAccessor (&Ping6::m_size),
74  MakeUintegerChecker<uint32_t>())
75  ;
76  return tid;
77 }
78 
80 {
82  m_sent = 0;
83  m_socket = 0;
84  m_seq = 0;
85  m_sendEvent = EventId ();
86 }
87 
89 {
91  m_socket = 0;
92 }
93 
95 {
98 }
99 
101 {
103 
104  if (!m_socket)
105  {
106  TypeId tid = TypeId::LookupByName ("ns3::Ipv6RawSocketFactory");
108 
110 
114  }
115 
116  ScheduleTransmit (Seconds (0.));
117 }
118 
120 {
121  NS_LOG_FUNCTION (this << ipv6);
122  m_localAddress = ipv6;
123 }
124 
126 {
127  NS_LOG_FUNCTION (this << ipv6);
128  m_peerAddress = ipv6;
129 }
130 
132 {
134 
135  if (m_socket)
136  {
138  }
139 
141 }
142 
143 void Ping6::SetIfIndex (uint32_t ifIndex)
144 {
145  m_ifIndex = ifIndex;
146 }
147 
149 {
150  NS_LOG_FUNCTION (this << dt);
152 }
153 
154 void Ping6::SetRouters (std::vector<Ipv6Address> routers)
155 {
156  m_routers = routers;
157 }
158 
159 void Ping6::Send ()
160 {
163  Ptr<Packet> p = 0;
164  uint8_t* data = new uint8_t[m_size];
165  Ipv6Address src;
166  Ptr<Ipv6> ipv6 = GetNode ()->GetObject<Ipv6> ();
167 
168  if (m_ifIndex > 0)
169  {
170  /* hack to have ifIndex in Ipv6RawSocketImpl
171  * maybe add a SetIfIndex in Ipv6RawSocketImpl directly
172  */
173  for (uint32_t i = 0; i < GetNode ()->GetObject<Ipv6> ()->GetNAddresses (m_ifIndex); i++)
174  {
175  Ipv6InterfaceAddress srcIa;
176  srcIa = GetNode ()->GetObject<Ipv6> ()->GetAddress (m_ifIndex, i);
177 
178  if (srcIa.IsInSameSubnet (m_peerAddress))
179  {
180  src = srcIa.GetAddress ();
181  break;
182  }
183  }
184  }
185  else
186  {
187  src = m_localAddress;
188  }
189 
190  NS_ASSERT_MSG (m_size >= 4, "ICMPv6 echo request payload size must be >= 4");
191  data[0] = 0xDE;
192  data[1] = 0xAD;
193  data[2] = 0xBE;
194  data[3] = 0xEF;
195 
196  p = Create<Packet> (data, 4);
197  p->AddAtEnd (Create<Packet> (m_size - 4));
198  Icmpv6Echo req (1);
199 
200  req.SetId (0xBEEF);
201  req.SetSeq (m_seq);
202  m_seq++;
203 
204  /* we do not calculate pseudo header checksum here, because we are not sure about
205  * source IPv6 address. Checksum is calculated in Ipv6RawSocketImpl.
206  */
207 
208  p->AddHeader (req);
209  m_socket->Bind (Inet6SocketAddress (src, 0));
210 
211  /* use Loose Routing (routing type 0) */
212  if (m_routers.size ())
213  {
214  Ipv6ExtensionLooseRoutingHeader routingHeader;
215  routingHeader.SetNextHeader (Ipv6Header::IPV6_ICMPV6);
216  routingHeader.SetLength (m_routers.size () * 16 + 8);
217  routingHeader.SetTypeRouting (0);
218  routingHeader.SetSegmentsLeft (m_routers.size ());
219  routingHeader.SetRoutersAddress (m_routers);
220  p->AddHeader (routingHeader);
222  }
223 
225  ++m_sent;
226 
227  NS_LOG_INFO ("Sent " << p->GetSize () << " bytes to " << m_peerAddress);
228 
229  if (m_sent < m_count)
230  {
232  }
233 
234  delete[] data;
235 }
236 
238 {
239  NS_LOG_FUNCTION (this << socket);
240 
241  Ptr<Packet> packet=0;
242  Address from;
243  while ((packet = socket->RecvFrom (from)))
244  {
246  {
247  Ipv6Header hdr;
248  Icmpv6Echo reply (0);
249  Icmpv6DestinationUnreachable destUnreach;
250  Icmpv6TimeExceeded timeExceeded;
252 
253  packet->RemoveHeader (hdr);
254 
255  uint8_t type;
256  packet->CopyData (&type, sizeof(type));
257 
258  switch (type)
259  {
261  packet->RemoveHeader (reply);
262 
263  NS_LOG_INFO ("Received Echo Reply size = " << std::dec << packet->GetSize () <<
264  " bytes from " << address.GetIpv6 () <<
265  " id = " << (uint16_t)reply.GetId () <<
266  " seq = " << (uint16_t)reply.GetSeq () <<
267  " Hop Count = " << (uint16_t) (64 - hdr.GetHopLimit ()));
268  break;
270  packet->RemoveHeader (destUnreach);
271 
272  NS_LOG_INFO ("Received Destination Unreachable from " << address.GetIpv6 ());
273  break;
275  packet->RemoveHeader (timeExceeded);
276 
277  NS_LOG_INFO ("Received Time Exceeded from " << address.GetIpv6 ());
278  break;
279  default:
280  break;
281  }
282  }
283  }
284 }
285 
286 } /* namespace ns3 */
287 
uint32_t RemoveHeader(Header &header)
Deserialize and remove the header from the internal buffer.
Definition: packet.cc:268
virtual ~Ping6()
Destructor.
Definition: ping6.cc:88
uint16_t GetId() const
Get the ID of the packet.
Ipv6Address GetIpv6(void) const
Get the IPv6 address.
Simulation virtual time values and global simulation resolution.
Definition: nstime.h:79
Doxygen introspection did not find any typical Config paths.
Definition: ipv6-header.h:33
#define NS_LOG_FUNCTION(parameters)
If log level LOG_FUNCTION is enabled, this macro will output all input parameters separated by "...
virtual void StopApplication()
Stop the application.
Definition: ping6.cc:131
#define NS_OBJECT_ENSURE_REGISTERED(type)
Register the class in the ns-3 factory.
Definition: object-base.h:38
Access to the IPv6 forwarding table, interfaces, and configuration.
Definition: ipv6.h:80
void SetTypeRouting(uint8_t typeRouting)
Set the "Type of Routing" field.
Doxygen introspection did not find any typical Config paths.
#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
#define NS_LOG_COMPONENT_DEFINE(name)
Define a Log component with a specific name.
Definition: log.h:170
IPv6 address associated with an interface.
void SetRouters(std::vector< Ipv6Address > routers)
Set routers for routing type 0 (loose routing).
Definition: ping6.cc:154
uint32_t GetSize(void) const
Returns the the size in bytes of the packet (including the zero-filled initial payload).
Definition: packet.h:744
#define NS_LOG_INFO(msg)
Use NS_LOG to output a message of level LOG_INFO.
Definition: log.h:223
static void Cancel(const EventId &id)
Set the cancel bit on this event: the event's associated function will not be invoked when it expires...
Definition: simulator.cc:268
#define NS_LOG_FUNCTION_NOARGS()
Output the name of the function.
Callback< R > MakeNullCallback(void)
Definition: callback.h:1399
static EventId Schedule(Time const &time, MEM mem_ptr, OBJ obj)
Schedule an event to expire at the relative time "time" is reached.
Definition: simulator.h:825
Time m_interval
Intervall between packets sent.
Definition: ping6.h:147
void Send()
Send a packet.
Definition: ping6.cc:159
a polymophic address class
Definition: address.h:86
void SetIfIndex(uint32_t ifIndex)
Set the out interface index.
Definition: ping6.cc:143
Doxygen introspection did not find any typical Config paths.
Ipv6Address GetAddress() const
Get the IPv6 address.
Doxygen introspection did not find any typical Config paths.
The base class for all ns3 applications.
Definition: application.h:60
hold objects of type ns3::Time
Definition: nstime.h:1008
Hold an unsigned integer type.
Definition: uinteger.h:46
uint8_t data[writeSize]
Ptr< Node > GetNode() const
Definition: application.cc:103
uint32_t m_sent
Number of packets sent.
Definition: ping6.h:137
An Inet6 address class.
Callback< R > MakeCallback(R(T::*memPtr)(void), OBJ objPtr)
Definition: callback.h:1242
virtual void StartApplication()
Start the application.
Definition: ping6.cc:100
void SetRecvCallback(Callback< void, Ptr< Socket > >)
Notify application when new data is available to be read.
Definition: socket.cc:127
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:70
std::vector< Ipv6Address > m_routers
Routers addresses for routing type 0.
Definition: ping6.h:182
void SetLocal(Ipv6Address ipv6)
Set the local address.
Definition: ping6.cc:119
uint16_t GetSeq() const
Get the sequence number.
uint32_t m_size
Size of the packet.
Definition: ping6.h:142
bool IsInSameSubnet(Ipv6Address b) const
Checks if the address is in the same subnet.
virtual void DoDispose(void)
This method is called by Object::Dispose or by the object's destructor, whichever comes first...
Definition: application.cc:82
uint8_t GetHopLimit(void) const
Get the "Hop limit" field (TTL).
Definition: ipv6-header.cc:91
virtual int Bind(const Address &address)=0
Allocate a local endpoint for this socket.
void SetNextHeader(uint8_t nextHeader)
Set the "Next header" field.
Ptr< Socket > m_socket
Local socket.
Definition: ping6.h:162
Ping6()
Constructor.
Definition: ping6.cc:79
Hold objects of type ns3::Ipv6Address.
Doxygen introspection did not find any typical Config paths.
#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
Describes an IPv6 address.
Definition: ipv6-address.h:46
void SetRoutersAddress(std::vector< Ipv6Address > routersAddress)
Set the vector of routers' address.
void SetRemote(Ipv6Address ipv6)
Set the remote peer.
Definition: ping6.cc:125
an identifier for simulation events.
Definition: event-id.h:46
uint16_t m_seq
Sequence number.
Definition: ping6.h:167
uint32_t m_ifIndex
Out interface (i.e.
Definition: ping6.h:177
static Inet6SocketAddress ConvertFrom(const Address &addr)
Convert the address to a InetSocketAddress.
Ipv6Address m_localAddress
Local address.
Definition: ping6.h:152
static bool IsMatchingType(const Address &addr)
If the address match.
void ScheduleTransmit(Time dt)
Schedule sending a packet.
Definition: ping6.cc:148
void SetLength(uint16_t length)
brief Set the length of the extension.
Ptr< const AttributeChecker > MakeTimeChecker(const Time min, const Time max)
Helper to make a Time checker with bounded range.
Definition: time.cc:441
uint32_t CopyData(uint8_t *buffer, uint32_t size) const
Copy the packet contents to a byte buffer.
Definition: packet.cc:381
virtual int SendTo(Ptr< Packet > p, uint32_t flags, const Address &toAddress)=0
Send data to a specified peer.
virtual void DoDispose()
Dispose this object;.
Definition: ping6.cc:94
void SetSeq(uint16_t seq)
Set the sequence number.
tuple address
Definition: first.py:37
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.
EventId m_sendEvent
Event ID.
Definition: ping6.h:172
static TypeId GetTypeId()
Get the type ID.
Definition: ping6.cc:45
void SetAttribute(std::string name, const AttributeValue &value)
Definition: object-base.cc:176
bool IsExpired(void) const
This method is syntactic sugar for the ns3::Simulator::isExpired method.
Definition: event-id.cc:53
Ptr< T > GetObject(void) const
Definition: object.h:362
a unique identifier for an interface.
Definition: type-id.h:49
TypeId SetParent(TypeId tid)
Definition: type-id.cc:610
Ipv6Address m_peerAddress
Peer address.
Definition: ping6.h:157
void SetId(uint16_t id)
Set the ID of the packet.
void HandleRead(Ptr< Socket > socket)
Receive method.
Definition: ping6.cc:237
static TypeId LookupByName(std::string name)
Definition: type-id.cc:535
uint32_t m_count
Number of "Echo request" packets that will be sent.
Definition: ping6.h:132
void SetSegmentsLeft(uint8_t segmentsLeft)
Set the "Segments left" field.