Files
unison/examples/wireless/wifi-multi-tos.cc
2022-10-14 14:13:12 +00:00

200 lines
7.2 KiB
C++

/*
* Copyright (c) 2016
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation;
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*
* Author: Sebastien Deronne <sebastien.deronne@gmail.com>
*/
#include "ns3/boolean.h"
#include "ns3/command-line.h"
#include "ns3/config.h"
#include "ns3/internet-stack-helper.h"
#include "ns3/ipv4-address-helper.h"
#include "ns3/ipv4-global-routing-helper.h"
#include "ns3/log.h"
#include "ns3/mobility-helper.h"
#include "ns3/on-off-helper.h"
#include "ns3/packet-sink-helper.h"
#include "ns3/packet-sink.h"
#include "ns3/ssid.h"
#include "ns3/string.h"
#include "ns3/uinteger.h"
#include "ns3/yans-wifi-channel.h"
#include "ns3/yans-wifi-helper.h"
// This is a simple example in order to show how to configure an IEEE 802.11n Wi-Fi network
// with multiple TOS. It outputs the aggregated UDP throughput, which depends on the number of
// stations, the HT MCS value (0 to 7), the channel width (20 or 40 MHz) and the guard interval
// (long or short). The user can also specify the distance between the access point and the
// stations (in meters), and can specify whether RTS/CTS is used or not.
using namespace ns3;
NS_LOG_COMPONENT_DEFINE("WifiMultiTos");
int
main(int argc, char* argv[])
{
uint32_t nWifi = 4;
double simulationTime = 10; // seconds
double distance = 1.0; // meters
uint16_t mcs = 7;
uint8_t channelWidth = 20; // MHz
bool useShortGuardInterval = false;
bool useRts = false;
CommandLine cmd(__FILE__);
cmd.AddValue("nWifi", "Number of stations", nWifi);
cmd.AddValue("distance",
"Distance in meters between the stations and the access point",
distance);
cmd.AddValue("simulationTime", "Simulation time in seconds", simulationTime);
cmd.AddValue("useRts", "Enable/disable RTS/CTS", useRts);
cmd.AddValue("mcs", "MCS value (0 - 7)", mcs);
cmd.AddValue("channelWidth", "Channel width in MHz", channelWidth);
cmd.AddValue("useShortGuardInterval",
"Enable/disable short guard interval",
useShortGuardInterval);
cmd.Parse(argc, argv);
NodeContainer wifiStaNodes;
wifiStaNodes.Create(nWifi);
NodeContainer wifiApNode;
wifiApNode.Create(1);
YansWifiChannelHelper channel = YansWifiChannelHelper::Default();
YansWifiPhyHelper phy;
phy.SetChannel(channel.Create());
WifiMacHelper mac;
WifiHelper wifi;
wifi.SetStandard(WIFI_STANDARD_80211n);
std::ostringstream oss;
oss << "HtMcs" << mcs;
wifi.SetRemoteStationManager("ns3::ConstantRateWifiManager",
"DataMode",
StringValue(oss.str()),
"ControlMode",
StringValue(oss.str()),
"RtsCtsThreshold",
UintegerValue(useRts ? 0 : 999999));
Ssid ssid = Ssid("ns3-80211n");
mac.SetType("ns3::StaWifiMac", "Ssid", SsidValue(ssid));
NetDeviceContainer staDevices;
staDevices = wifi.Install(phy, mac, wifiStaNodes);
mac.SetType("ns3::ApWifiMac", "Ssid", SsidValue(ssid));
NetDeviceContainer apDevice;
apDevice = wifi.Install(phy, mac, wifiApNode);
// Set channel width
Config::Set("/NodeList/*/DeviceList/*/$ns3::WifiNetDevice/Phy/ChannelSettings",
StringValue("{0, " + std::to_string(channelWidth) + ", BAND_2_4GHZ, 0}"));
// Set guard interval
Config::Set(
"/NodeList/*/DeviceList/*/$ns3::WifiNetDevice/HtConfiguration/ShortGuardIntervalSupported",
BooleanValue(useShortGuardInterval));
// mobility
MobilityHelper mobility;
Ptr<ListPositionAllocator> positionAlloc = CreateObject<ListPositionAllocator>();
positionAlloc->Add(Vector(0.0, 0.0, 0.0));
for (uint32_t i = 0; i < nWifi; i++)
{
positionAlloc->Add(Vector(distance, 0.0, 0.0));
}
mobility.SetPositionAllocator(positionAlloc);
mobility.SetMobilityModel("ns3::ConstantPositionMobilityModel");
mobility.Install(wifiApNode);
mobility.Install(wifiStaNodes);
// Internet stack
InternetStackHelper stack;
stack.Install(wifiApNode);
stack.Install(wifiStaNodes);
Ipv4AddressHelper address;
address.SetBase("192.168.1.0", "255.255.255.0");
Ipv4InterfaceContainer staNodeInterfaces;
Ipv4InterfaceContainer apNodeInterface;
staNodeInterfaces = address.Assign(staDevices);
apNodeInterface = address.Assign(apDevice);
// Setting applications
ApplicationContainer sourceApplications;
ApplicationContainer sinkApplications;
std::vector<uint8_t> tosValues = {0x70, 0x28, 0xb8, 0xc0}; // AC_BE, AC_BK, AC_VI, AC_VO
uint32_t portNumber = 9;
for (uint32_t index = 0; index < nWifi; ++index)
{
for (uint8_t tosValue : tosValues)
{
auto ipv4 = wifiApNode.Get(0)->GetObject<Ipv4>();
const auto address = ipv4->GetAddress(1, 0).GetLocal();
InetSocketAddress sinkSocket(address, portNumber++);
sinkSocket.SetTos(tosValue);
OnOffHelper onOffHelper("ns3::UdpSocketFactory", sinkSocket);
onOffHelper.SetAttribute("OnTime",
StringValue("ns3::ConstantRandomVariable[Constant=1]"));
onOffHelper.SetAttribute("OffTime",
StringValue("ns3::ConstantRandomVariable[Constant=0]"));
onOffHelper.SetAttribute("DataRate", DataRateValue(50000000 / nWifi));
onOffHelper.SetAttribute("PacketSize", UintegerValue(1472)); // bytes
sourceApplications.Add(onOffHelper.Install(wifiStaNodes.Get(index)));
PacketSinkHelper packetSinkHelper("ns3::UdpSocketFactory", sinkSocket);
sinkApplications.Add(packetSinkHelper.Install(wifiApNode.Get(0)));
}
}
sinkApplications.Start(Seconds(0.0));
sinkApplications.Stop(Seconds(simulationTime + 1));
sourceApplications.Start(Seconds(1.0));
sourceApplications.Stop(Seconds(simulationTime + 1));
Ipv4GlobalRoutingHelper::PopulateRoutingTables();
Simulator::Stop(Seconds(simulationTime + 1));
Simulator::Run();
double throughput = 0;
for (uint32_t index = 0; index < sinkApplications.GetN(); ++index)
{
uint64_t totalPacketsThrough =
DynamicCast<PacketSink>(sinkApplications.Get(index))->GetTotalRx();
throughput += ((totalPacketsThrough * 8) / (simulationTime * 1000000.0)); // Mbit/s
}
Simulator::Destroy();
if (throughput > 0)
{
std::cout << "Aggregated throughput: " << throughput << " Mbit/s" << std::endl;
}
else
{
NS_LOG_ERROR("Obtained throughput is 0!");
exit(1);
}
return 0;
}