A Discrete-Event Network Simulator
API
lena-x2-handover.cc
Go to the documentation of this file.
1 /* -*- Mode: C++; c-file-style: "gnu"; indent-tabs-mode:nil; -*- */
2 /*
3  * Copyright (c) 2012-2018 Centre Tecnologic de Telecomunicacions de Catalunya (CTTC)
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: Manuel Requena <manuel.requena@cttc.es>
19  */
20 
21 #include "ns3/core-module.h"
22 #include "ns3/network-module.h"
23 #include "ns3/internet-module.h"
24 #include "ns3/mobility-module.h"
25 #include "ns3/lte-module.h"
26 #include "ns3/applications-module.h"
27 #include "ns3/point-to-point-module.h"
28 #include "ns3/config-store-module.h"
29 
30 using namespace ns3;
31 
32 NS_LOG_COMPONENT_DEFINE ("LenaX2HandoverExample");
33 
34 void
35 NotifyConnectionEstablishedUe (std::string context,
36  uint64_t imsi,
37  uint16_t cellid,
38  uint16_t rnti)
39 {
40  std::cout << Simulator::Now ().GetSeconds () << " " << context
41  << " UE IMSI " << imsi
42  << ": connected to CellId " << cellid
43  << " with RNTI " << rnti
44  << std::endl;
45 }
46 
47 void
48 NotifyHandoverStartUe (std::string context,
49  uint64_t imsi,
50  uint16_t cellid,
51  uint16_t rnti,
52  uint16_t targetCellId)
53 {
54  std::cout << Simulator::Now ().GetSeconds () << " " << context
55  << " UE IMSI " << imsi
56  << ": previously connected to CellId " << cellid
57  << " with RNTI " << rnti
58  << ", doing handover to CellId " << targetCellId
59  << std::endl;
60 }
61 
62 void
63 NotifyHandoverEndOkUe (std::string context,
64  uint64_t imsi,
65  uint16_t cellid,
66  uint16_t rnti)
67 {
68  std::cout << Simulator::Now ().GetSeconds () << " " << context
69  << " UE IMSI " << imsi
70  << ": successful handover to CellId " << cellid
71  << " with RNTI " << rnti
72  << std::endl;
73 }
74 
75 void
76 NotifyConnectionEstablishedEnb (std::string context,
77  uint64_t imsi,
78  uint16_t cellid,
79  uint16_t rnti)
80 {
81  std::cout << Simulator::Now ().GetSeconds () << " " << context
82  << " eNB CellId " << cellid
83  << ": successful connection of UE with IMSI " << imsi
84  << " RNTI " << rnti
85  << std::endl;
86 }
87 
88 void
89 NotifyHandoverStartEnb (std::string context,
90  uint64_t imsi,
91  uint16_t cellid,
92  uint16_t rnti,
93  uint16_t targetCellId)
94 {
95  std::cout << Simulator::Now ().GetSeconds () << " " << context
96  << " eNB CellId " << cellid
97  << ": start handover of UE with IMSI " << imsi
98  << " RNTI " << rnti
99  << " to CellId " << targetCellId
100  << std::endl;
101 }
102 
103 void
104 NotifyHandoverEndOkEnb (std::string context,
105  uint64_t imsi,
106  uint16_t cellid,
107  uint16_t rnti)
108 {
109  std::cout << Simulator::Now ().GetSeconds () << " " << context
110  << " eNB CellId " << cellid
111  << ": completed handover of UE with IMSI " << imsi
112  << " RNTI " << rnti
113  << std::endl;
114 }
115 
116 
122 int
123 main (int argc, char *argv[])
124 {
125  // LogLevel logLevel = (LogLevel)(LOG_PREFIX_FUNC | LOG_PREFIX_TIME | LOG_LEVEL_ALL);
126 
127  // LogComponentEnable ("LteHelper", logLevel);
128  // LogComponentEnable ("EpcHelper", logLevel);
129  // LogComponentEnable ("EpcEnbApplication", logLevel);
130  // LogComponentEnable ("EpcX2", logLevel);
131  // LogComponentEnable ("EpcSgwPgwApplication", logLevel);
132 
133  // LogComponentEnable ("LteEnbRrc", logLevel);
134  // LogComponentEnable ("LteEnbNetDevice", logLevel);
135  // LogComponentEnable ("LteUeRrc", logLevel);
136  // LogComponentEnable ("LteUeNetDevice", logLevel);
137 
138  uint16_t numberOfUes = 1;
139  uint16_t numberOfEnbs = 2;
140  uint16_t numBearersPerUe = 2;
141  Time simTime = MilliSeconds (490);
142  double distance = 100.0;
143  bool disableDl = false;
144  bool disableUl = false;
145 
146  // change some default attributes so that they are reasonable for
147  // this scenario, but do this before processing command line
148  // arguments, so that the user is allowed to override these settings
149  Config::SetDefault ("ns3::UdpClient::Interval", TimeValue (MilliSeconds (10)));
150  Config::SetDefault ("ns3::UdpClient::MaxPackets", UintegerValue (1000000));
151  Config::SetDefault ("ns3::LteHelper::UseIdealRrc", BooleanValue (false));
152 
153  // Command line arguments
155  cmd.AddValue ("numberOfUes", "Number of UEs", numberOfUes);
156  cmd.AddValue ("numberOfEnbs", "Number of eNodeBs", numberOfEnbs);
157  cmd.AddValue ("simTime", "Total duration of the simulation", simTime);
158  cmd.AddValue ("disableDl", "Disable downlink data flows", disableDl);
159  cmd.AddValue ("disableUl", "Disable uplink data flows", disableUl);
160  cmd.Parse (argc, argv);
161 
162 
163  Ptr<LteHelper> lteHelper = CreateObject<LteHelper> ();
164  Ptr<PointToPointEpcHelper> epcHelper = CreateObject<PointToPointEpcHelper> ();
165  lteHelper->SetEpcHelper (epcHelper);
166  lteHelper->SetSchedulerType ("ns3::RrFfMacScheduler");
167  lteHelper->SetHandoverAlgorithmType ("ns3::NoOpHandoverAlgorithm"); // disable automatic handover
168 
169  Ptr<Node> pgw = epcHelper->GetPgwNode ();
170 
171  // Create a single RemoteHost
172  NodeContainer remoteHostContainer;
173  remoteHostContainer.Create (1);
174  Ptr<Node> remoteHost = remoteHostContainer.Get (0);
175  InternetStackHelper internet;
176  internet.Install (remoteHostContainer);
177 
178  // Create the Internet
179  PointToPointHelper p2ph;
180  p2ph.SetDeviceAttribute ("DataRate", DataRateValue (DataRate ("100Gb/s")));
181  p2ph.SetDeviceAttribute ("Mtu", UintegerValue (1500));
182  p2ph.SetChannelAttribute ("Delay", TimeValue (Seconds (0.010)));
183  NetDeviceContainer internetDevices = p2ph.Install (pgw, remoteHost);
184  Ipv4AddressHelper ipv4h;
185  ipv4h.SetBase ("1.0.0.0", "255.0.0.0");
186  Ipv4InterfaceContainer internetIpIfaces = ipv4h.Assign (internetDevices);
187  Ipv4Address remoteHostAddr = internetIpIfaces.GetAddress (1);
188 
189 
190  // Routing of the Internet Host (towards the LTE network)
191  Ipv4StaticRoutingHelper ipv4RoutingHelper;
192  Ptr<Ipv4StaticRouting> remoteHostStaticRouting = ipv4RoutingHelper.GetStaticRouting (remoteHost->GetObject<Ipv4> ());
193  // interface 0 is localhost, 1 is the p2p device
194  remoteHostStaticRouting->AddNetworkRouteTo (Ipv4Address ("7.0.0.0"), Ipv4Mask ("255.0.0.0"), 1);
195 
196  NodeContainer ueNodes;
197  NodeContainer enbNodes;
198  enbNodes.Create (numberOfEnbs);
199  ueNodes.Create (numberOfUes);
200 
201  // Install Mobility Model
202  Ptr<ListPositionAllocator> positionAlloc = CreateObject<ListPositionAllocator> ();
203  for (uint16_t i = 0; i < numberOfEnbs; i++)
204  {
205  positionAlloc->Add (Vector (distance * 2 * i - distance, 0, 0));
206  }
207  for (uint16_t i = 0; i < numberOfUes; i++)
208  {
209  positionAlloc->Add (Vector (0, 0, 0));
210  }
212  mobility.SetMobilityModel ("ns3::ConstantPositionMobilityModel");
213  mobility.SetPositionAllocator (positionAlloc);
214  mobility.Install (enbNodes);
215  mobility.Install (ueNodes);
216 
217  // Install LTE Devices in eNB and UEs
218  NetDeviceContainer enbLteDevs = lteHelper->InstallEnbDevice (enbNodes);
219  NetDeviceContainer ueLteDevs = lteHelper->InstallUeDevice (ueNodes);
220 
221  // Install the IP stack on the UEs
222  internet.Install (ueNodes);
223  Ipv4InterfaceContainer ueIpIfaces;
224  ueIpIfaces = epcHelper->AssignUeIpv4Address (NetDeviceContainer (ueLteDevs));
225 
226 
227  // Attach all UEs to the first eNodeB
228  for (uint16_t i = 0; i < numberOfUes; i++)
229  {
230  lteHelper->Attach (ueLteDevs.Get (i), enbLteDevs.Get (0));
231  }
232 
233 
234  NS_LOG_LOGIC ("setting up applications");
235 
236  // Install and start applications on UEs and remote host
237  uint16_t dlPort = 10000;
238  uint16_t ulPort = 20000;
239 
240  // randomize a bit start times to avoid simulation artifacts
241  // (e.g., buffer overflows due to packet transmissions happening
242  // exactly at the same time)
243  Ptr<UniformRandomVariable> startTimeSeconds = CreateObject<UniformRandomVariable> ();
244  startTimeSeconds->SetAttribute ("Min", DoubleValue (0.05));
245  startTimeSeconds->SetAttribute ("Max", DoubleValue (0.06));
246 
247  for (uint32_t u = 0; u < numberOfUes; ++u)
248  {
249  Ptr<Node> ue = ueNodes.Get (u);
250  // Set the default gateway for the UE
251  Ptr<Ipv4StaticRouting> ueStaticRouting = ipv4RoutingHelper.GetStaticRouting (ue->GetObject<Ipv4> ());
252  ueStaticRouting->SetDefaultRoute (epcHelper->GetUeDefaultGatewayAddress (), 1);
253 
254  for (uint32_t b = 0; b < numBearersPerUe; ++b)
255  {
258  Ptr<EpcTft> tft = Create<EpcTft> ();
259 
260  if (!disableDl)
261  {
262  ++dlPort;
263 
264  NS_LOG_LOGIC ("installing UDP DL app for UE " << u);
265  UdpClientHelper dlClientHelper (ueIpIfaces.GetAddress (u), dlPort);
266  clientApps.Add (dlClientHelper.Install (remoteHost));
267  PacketSinkHelper dlPacketSinkHelper ("ns3::UdpSocketFactory",
269  serverApps.Add (dlPacketSinkHelper.Install (ue));
270 
272  dlpf.localPortStart = dlPort;
273  dlpf.localPortEnd = dlPort;
274  tft->Add (dlpf);
275  }
276 
277  if (!disableUl)
278  {
279  ++ulPort;
280 
281  NS_LOG_LOGIC ("installing UDP UL app for UE " << u);
282  UdpClientHelper ulClientHelper (remoteHostAddr, ulPort);
283  clientApps.Add (ulClientHelper.Install (ue));
284  PacketSinkHelper ulPacketSinkHelper ("ns3::UdpSocketFactory",
286  serverApps.Add (ulPacketSinkHelper.Install (remoteHost));
287 
289  ulpf.remotePortStart = ulPort;
290  ulpf.remotePortEnd = ulPort;
291  tft->Add (ulpf);
292  }
293 
295  lteHelper->ActivateDedicatedEpsBearer (ueLteDevs.Get (u), bearer, tft);
296 
297  Time startTime = Seconds (startTimeSeconds->GetValue ());
298  serverApps.Start (startTime);
299  clientApps.Start (startTime);
300  clientApps.Stop (simTime);
301 
302  } // end for b
303  }
304 
305 
306  // Add X2 interface
307  lteHelper->AddX2Interface (enbNodes);
308 
309  // X2-based Handover
310  lteHelper->HandoverRequest (MilliSeconds (300), ueLteDevs.Get (0), enbLteDevs.Get (0), enbLteDevs.Get (1));
311 
312  // Uncomment to enable PCAP tracing
313  //p2ph.EnablePcapAll("lena-x2-handover");
314 
315  lteHelper->EnablePhyTraces ();
316  lteHelper->EnableMacTraces ();
317  lteHelper->EnableRlcTraces ();
318  lteHelper->EnablePdcpTraces ();
319  Ptr<RadioBearerStatsCalculator> rlcStats = lteHelper->GetRlcStats ();
320  rlcStats->SetAttribute ("EpochDuration", TimeValue (Seconds (0.05)));
321  Ptr<RadioBearerStatsCalculator> pdcpStats = lteHelper->GetPdcpStats ();
322  pdcpStats->SetAttribute ("EpochDuration", TimeValue (Seconds (0.05)));
323 
324 
325  // connect custom trace sinks for RRC connection establishment and handover notification
326  Config::Connect ("/NodeList/*/DeviceList/*/LteEnbRrc/ConnectionEstablished",
328  Config::Connect ("/NodeList/*/DeviceList/*/LteUeRrc/ConnectionEstablished",
330  Config::Connect ("/NodeList/*/DeviceList/*/LteEnbRrc/HandoverStart",
332  Config::Connect ("/NodeList/*/DeviceList/*/LteUeRrc/HandoverStart",
334  Config::Connect ("/NodeList/*/DeviceList/*/LteEnbRrc/HandoverEndOk",
336  Config::Connect ("/NodeList/*/DeviceList/*/LteUeRrc/HandoverEndOk",
338 
339 
340  Simulator::Stop (simTime + MilliSeconds (20));
341  Simulator::Run ();
342 
343  // GtkConfigStore config;
344  // config.ConfigureAttributes ();
345 
347  return 0;
348 }
holds a vector of ns3::Application pointers.
uint8_t Add(PacketFilter f)
add a PacketFilter to the Traffic Flow Template
Definition: epc-tft.cc:240
Ptr< NetDevice > Get(uint32_t i) const
Get the Ptr<NetDevice> stored in this container at a given index.
Simulation virtual time values and global simulation resolution.
Definition: nstime.h:102
an Inet address class
static Ipv4Address GetAny(void)
AttributeValue implementation for Boolean.
Definition: boolean.h:36
NetDeviceContainer InstallEnbDevice(NodeContainer c)
Create a set of eNodeB devices.
Definition: lte-helper.cc:474
holds a vector of std::pair of Ptr<Ipv4> and interface index.
void SetDefaultRoute(Ipv4Address nextHop, uint32_t interface, uint32_t metric=0)
Add a default route to the static routing table.
void HandoverRequest(Time hoTime, Ptr< NetDevice > ueDev, Ptr< NetDevice > sourceEnbDev, Ptr< NetDevice > targetEnbDev)
Manually trigger an X2-based handover.
Definition: lte-helper.cc:1246
NetDeviceContainer Install(NodeContainer c)
a class to represent an Ipv4 address mask
Definition: ipv4-address.h:258
double GetSeconds(void) const
Get an approximation of the time stored in this instance in the indicated unit.
Definition: nstime.h:355
serverApps
Definition: first.py:45
static void Run(void)
Run the simulation.
Definition: simulator.cc:170
void Attach(NetDeviceContainer ueDevices)
Enables automatic attachment of a set of UE devices to a suitable cell using Idle mode initial cell s...
Definition: lte-helper.cc:961
#define NS_LOG_COMPONENT_DEFINE(name)
Define a Log component with a specific name.
Definition: log.h:204
Time MilliSeconds(uint64_t value)
Construct a Time in the indicated unit.
Definition: nstime.h:1070
void EnableRlcTraces(void)
Enable trace sinks for RLC layer.
Definition: lte-helper.cc:1435
aggregate IP/TCP/UDP functionality to existing Nodes.
uint16_t localPortEnd
end of the port number range of the UE
Definition: epc-tft.h:140
A helper to make it easier to instantiate an ns3::PacketSinkApplication on a set of nodes...
cmd
Definition: second.py:35
Build a set of PointToPointNetDevice objects.
void NotifyConnectionEstablishedEnb(std::string context, uint64_t imsi, uint16_t cellid, uint16_t rnti)
void SetDeviceAttribute(std::string name, const AttributeValue &value)
Set an attribute value to be propagated to each NetDevice created by the helper.
uint8_t ActivateDedicatedEpsBearer(NetDeviceContainer ueDevices, EpsBearer bearer, Ptr< EpcTft > tft)
Activate a dedicated EPS bearer on a given set of UE devices.
Definition: lte-helper.cc:1069
void SetHandoverAlgorithmType(std::string type)
Set the type of handover algorithm to be used by eNodeB devices.
Definition: lte-helper.cc:327
void NotifyHandoverEndOkUe(std::string context, uint64_t imsi, uint16_t cellid, uint16_t rnti)
void SetSchedulerType(std::string type)
Set the type of scheduler to be used by eNodeB devices.
Definition: lte-helper.cc:279
mobility
Definition: third.py:101
Class for representing data rates.
Definition: data-rate.h:88
void EnablePdcpTraces(void)
Enable trace sinks for PDCP layer.
Definition: lte-helper.cc:1578
Create a client application which sends UDP packets carrying a 32bit sequence number and a 64 bit tim...
AttributeValue implementation for Time.
Definition: nstime.h:1124
Ipv4Address GetAddress(uint32_t i, uint32_t j=0) const
Hold an unsigned integer type.
Definition: uinteger.h:44
double startTime
void NotifyConnectionEstablishedUe(std::string context, uint64_t imsi, uint16_t cellid, uint16_t rnti)
holds a vector of ns3::NetDevice pointers
void EnablePhyTraces(void)
Enable trace sinks for PHY layer.
Definition: lte-helper.cc:1489
Callback< R > MakeCallback(R(T::*memPtr)(void), OBJ objPtr)
Definition: callback.h:1489
Parse command-line arguments.
Definition: command-line.h:213
void Connect(std::string path, const CallbackBase &cb)
Definition: config.cc:871
static void Destroy(void)
Execute the events scheduled with ScheduleDestroy().
Definition: simulator.cc:134
void NotifyHandoverEndOkEnb(std::string context, uint64_t imsi, uint16_t cellid, uint16_t rnti)
Access to the IPv4 forwarding table, interfaces, and configuration.
Definition: ipv4.h:76
Ptr< T > GetObject(void) const
Get a pointer to the requested aggregated Object.
Definition: object.h:459
uint16_t remotePortEnd
end of the port number range of the remote host
Definition: epc-tft.h:138
Every class exported by the ns3 library is enclosed in the ns3 namespace.
keep track of a set of node pointers.
double GetValue(double min, double max)
Get the next random value, as a double in the specified range .
void AddX2Interface(NodeContainer enbNodes)
Create an X2 interface between all the eNBs in a given set.
Definition: lte-helper.cc:1221
This class contains the specification of EPS Bearers.
Definition: eps-bearer.h:91
static Time Now(void)
Return the current simulation virtual time.
Definition: simulator.cc:193
NS_LOG_LOGIC("Net device "<< nd<< " is not bridged")
void SetChannelAttribute(std::string name, const AttributeValue &value)
Set an attribute value to be propagated to each Channel created by the helper.
void Install(std::string nodeName) const
Aggregate implementations of the ns3::Ipv4, ns3::Ipv6, ns3::Udp, and ns3::Tcp classes onto the provid...
Ptr< RadioBearerStatsCalculator > GetRlcStats(void)
Definition: lte-helper.cc:1572
NetDeviceContainer InstallUeDevice(NodeContainer c)
Create a set of UE devices.
Definition: lte-helper.cc:489
Ptr< RadioBearerStatsCalculator > GetPdcpStats(void)
Definition: lte-helper.cc:1586
Helper class used to assign positions and mobility models to nodes.
void AddNetworkRouteTo(Ipv4Address network, Ipv4Mask networkMask, Ipv4Address nextHop, uint32_t interface, uint32_t metric=0)
Add a network route to the static routing table.
Ipv4 addresses are stored in host order in this class.
Definition: ipv4-address.h:40
Ipv4InterfaceContainer Assign(const NetDeviceContainer &c)
Assign IP addresses to the net devices specified in the container based on the current network prefix...
void SetEpcHelper(Ptr< EpcHelper > h)
Set the EpcHelper to be used to setup the EPC network in conjunction with the setup of the LTE radio ...
Definition: lte-helper.cc:272
Helper class that adds ns3::Ipv4StaticRouting objects.
AttributeValue implementation for DataRate.
Definition: data-rate.h:242
static void Stop(void)
Tell the Simulator the calling event should be the last one executed.
Definition: simulator.cc:178
Time Seconds(double value)
Construct a Time in the indicated unit.
Definition: nstime.h:1062
void SetDefault(std::string name, const AttributeValue &value)
Definition: config.cc:810
void Add(Vector v)
Add a position to the list of positions.
Ptr< Node > Get(uint32_t i) const
Get the Ptr<Node> stored in this container at a given index.
clientApps
Definition: first.py:54
A helper class to make life easier while doing simple IPv4 address assignment in scripts.
void Create(uint32_t n)
Create n nodes and append pointers to them to the end of this NodeContainer.
void NotifyHandoverStartUe(std::string context, uint64_t imsi, uint16_t cellid, uint16_t rnti, uint16_t targetCellId)
void EnableMacTraces(void)
Enable trace sinks for MAC layer.
Definition: lte-helper.cc:1529
void NotifyHandoverStartEnb(std::string context, uint64_t imsi, uint16_t cellid, uint16_t rnti, uint16_t targetCellId)
This class can be used to hold variables of floating point type such as &#39;double&#39; or &#39;float&#39;...
Definition: double.h:41
Non-GBR TCP-based Video (Buffered Streaming, e.g., www, e-mail...)
Definition: eps-bearer.h:120
void SetAttribute(std::string name, const AttributeValue &value)
Set a single attribute, raising fatal errors if unsuccessful.
Definition: object-base.cc:185
uint16_t remotePortStart
start of the port number range of the remote host
Definition: epc-tft.h:137
Ptr< Ipv4StaticRouting > GetStaticRouting(Ptr< Ipv4 > ipv4) const
Try and find the static routing protocol as either the main routing protocol or in the list of routin...
void SetBase(Ipv4Address network, Ipv4Mask mask, Ipv4Address base="0.0.0.1")
Set the base network number, network mask and base address.
Implement the data structure representing a TrafficFlowTemplate Packet Filter.
Definition: epc-tft.h:74
uint16_t localPortStart
start of the port number range of the UE
Definition: epc-tft.h:139