A Discrete-Event Network Simulator
API
 All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Properties Friends Macros Groups Pages
packet-sink.cc
Go to the documentation of this file.
1 /* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
2 /*
3  * Copyright 2007 University of Washington
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: Tom Henderson (tomhend@u.washington.edu)
19  */
20 #include "ns3/address.h"
21 #include "ns3/address-utils.h"
22 #include "ns3/log.h"
23 #include "ns3/inet-socket-address.h"
24 #include "ns3/inet6-socket-address.h"
25 #include "ns3/node.h"
26 #include "ns3/socket.h"
27 #include "ns3/udp-socket.h"
28 #include "ns3/simulator.h"
29 #include "ns3/socket-factory.h"
30 #include "ns3/packet.h"
31 #include "ns3/trace-source-accessor.h"
32 #include "ns3/udp-socket-factory.h"
33 #include "packet-sink.h"
34 
35 namespace ns3 {
36 
37 NS_LOG_COMPONENT_DEFINE ("PacketSink");
38 NS_OBJECT_ENSURE_REGISTERED (PacketSink);
39 
40 TypeId
42 {
43  static TypeId tid = TypeId ("ns3::PacketSink")
45  .AddConstructor<PacketSink> ()
46  .AddAttribute ("Local", "The Address on which to Bind the rx socket.",
47  AddressValue (),
48  MakeAddressAccessor (&PacketSink::m_local),
49  MakeAddressChecker ())
50  .AddAttribute ("Protocol", "The type id of the protocol to use for the rx socket.",
52  MakeTypeIdAccessor (&PacketSink::m_tid),
53  MakeTypeIdChecker ())
54  .AddTraceSource ("Rx", "A packet has been received",
56  ;
57  return tid;
58 }
59 
61 {
62  NS_LOG_FUNCTION (this);
63  m_socket = 0;
64  m_totalRx = 0;
65 }
66 
68 {
69  NS_LOG_FUNCTION (this);
70 }
71 
72 uint32_t PacketSink::GetTotalRx () const
73 {
74  NS_LOG_FUNCTION (this);
75  return m_totalRx;
76 }
77 
80 {
81  NS_LOG_FUNCTION (this);
82  return m_socket;
83 }
84 
85 std::list<Ptr<Socket> >
87 {
88  NS_LOG_FUNCTION (this);
89  return m_socketList;
90 }
91 
93 {
94  NS_LOG_FUNCTION (this);
95  m_socket = 0;
96  m_socketList.clear ();
97 
98  // chain up
100 }
101 
102 
103 // Application Methods
104 void PacketSink::StartApplication () // Called at time specified by Start
105 {
106  NS_LOG_FUNCTION (this);
107  // Create the socket if not already
108  if (!m_socket)
109  {
111  m_socket->Bind (m_local);
112  m_socket->Listen ();
115  {
116  Ptr<UdpSocket> udpSocket = DynamicCast<UdpSocket> (m_socket);
117  if (udpSocket)
118  {
119  // equivalent to setsockopt (MCAST_JOIN_GROUP)
120  udpSocket->MulticastJoinGroup (0, m_local);
121  }
122  else
123  {
124  NS_FATAL_ERROR ("Error: joining multicast on a non-UDP socket");
125  }
126  }
127  }
128 
131  MakeNullCallback<bool, Ptr<Socket>, const Address &> (),
136 }
137 
138 void PacketSink::StopApplication () // Called at time specified by Stop
139 {
140  NS_LOG_FUNCTION (this);
141  while(!m_socketList.empty ()) //these are accepted sockets, close them
142  {
143  Ptr<Socket> acceptedSocket = m_socketList.front ();
144  m_socketList.pop_front ();
145  acceptedSocket->Close ();
146  }
147  if (m_socket)
148  {
149  m_socket->Close ();
151  }
152 }
153 
155 {
156  NS_LOG_FUNCTION (this << socket);
157  Ptr<Packet> packet;
158  Address from;
159  while ((packet = socket->RecvFrom (from)))
160  {
161  if (packet->GetSize () == 0)
162  { //EOF
163  break;
164  }
165  m_totalRx += packet->GetSize ();
167  {
168  NS_LOG_INFO ("At time " << Simulator::Now ().GetSeconds ()
169  << "s packet sink received "
170  << packet->GetSize () << " bytes from "
172  << " port " << InetSocketAddress::ConvertFrom (from).GetPort ()
173  << " total Rx " << m_totalRx << " bytes");
174  }
175  else if (Inet6SocketAddress::IsMatchingType (from))
176  {
177  NS_LOG_INFO ("At time " << Simulator::Now ().GetSeconds ()
178  << "s packet sink received "
179  << packet->GetSize () << " bytes from "
181  << " port " << Inet6SocketAddress::ConvertFrom (from).GetPort ()
182  << " total Rx " << m_totalRx << " bytes");
183  }
184  m_rxTrace (packet, from);
185  }
186 }
187 
188 
190 {
191  NS_LOG_FUNCTION (this << socket);
192 }
193 
195 {
196  NS_LOG_FUNCTION (this << socket);
197 }
198 
199 
201 {
202  NS_LOG_FUNCTION (this << s << from);
204  m_socketList.push_back (s);
205 }
206 
207 } // Namespace ns3