217 lines
7.9 KiB
C++
217 lines
7.9 KiB
C++
/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
|
|
/*
|
|
* 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
|
|
*/
|
|
|
|
// Allow ns-3 to ping a real host somewhere, using emulation mode
|
|
//
|
|
// ------------
|
|
// | node n0 |
|
|
// | |
|
|
// | --- |
|
|
// | | | |
|
|
// | |emu| |
|
|
// | | | |
|
|
// | --- |
|
|
// | | |
|
|
// ----|-------
|
|
// |
|
|
// (device on host system, set to promiscuous mode)
|
|
// |
|
|
// --------- (Internet) -------
|
|
//
|
|
// To use this example:
|
|
// 1) You need to decide on a physical device on your real system, and either
|
|
// overwrite the hard-configured device name below (eth0) or pass this
|
|
// device name in as a command-line argument
|
|
// 2) The host device must be set to promiscuous mode
|
|
// (e.g. "sudo ifconfig eth0 promisc")
|
|
// 3) Be aware that ns-3 will generate a fake mac address, and that in
|
|
// some enterprise networks, this may be considered bad form to be
|
|
// sending packets out of your device with "unauthorized" mac addresses
|
|
// 4) You will need to assign an IP address to the ns-3 simulation node that
|
|
// is consistent with the subnet that is active on the host device's link.
|
|
// That is, you will have to assign an IP address to the ns-3 node as if
|
|
// it were on your real subnet. Search for "Ipv4Address localIp" and
|
|
// replace the string "1.2.3.4" with a valid IP address.
|
|
// 5) You will need to configure a default route in the ns-3 node to tell it
|
|
// how to get off of your subnet. One thing you could do is a
|
|
// 'netstat -rn' command and find the IP address of the default gateway
|
|
// on your host. Search for "Ipv4Address gateway" and replace the string
|
|
// "1.2.3.4" string with the gateway IP address.
|
|
|
|
#include "ns3/abort.h"
|
|
#include "ns3/core-module.h"
|
|
#include "ns3/simulator-module.h"
|
|
#include "ns3/node-module.h"
|
|
#include "ns3/emu-module.h"
|
|
#include "ns3/v4ping-module.h"
|
|
#include "ns3/helper-module.h"
|
|
|
|
using namespace ns3;
|
|
|
|
NS_LOG_COMPONENT_DEFINE ("PingEmulationExample");
|
|
|
|
static void
|
|
PingRtt (std::string context, Time rtt)
|
|
{
|
|
NS_LOG_UNCOND ("Received Response with RTT = " << rtt);
|
|
}
|
|
|
|
int
|
|
main (int argc, char *argv[])
|
|
{
|
|
NS_LOG_INFO ("Ping Emulation Example");
|
|
|
|
std::string deviceName ("eth0");
|
|
std::string remote ("208.77.188.166"); // example.com
|
|
|
|
//
|
|
// Allow the user to override any of the defaults at run-time, via
|
|
// command-line arguments
|
|
//
|
|
CommandLine cmd;
|
|
cmd.AddValue("deviceName", "Device name", deviceName);
|
|
cmd.AddValue("remote", "Remote IP address (dotted decimal only please)", remote);
|
|
cmd.Parse (argc, argv);
|
|
|
|
Ipv4Address remoteIp (remote.c_str ());
|
|
Ipv4Address localIp ("1.2.3.4");
|
|
NS_ABORT_MSG_IF (localIp == "1.2.3.4", "You must change the local IP address before running this example");
|
|
|
|
Ipv4Mask localMask ("255.255.255.0");
|
|
|
|
//
|
|
// Since we are using a real piece of hardware we need to use the realtime
|
|
// simulator.
|
|
//
|
|
GlobalValue::Bind ("SimulatorImplementationType", StringValue ("ns3::RealtimeSimulatorImpl"));
|
|
|
|
//
|
|
// Since we are going to be talking to real-world machines, we need to enable
|
|
// calculation of checksums in our protocols.
|
|
//
|
|
GlobalValue::Bind ("ChecksumEnabled", BooleanValue (true));
|
|
|
|
//
|
|
// In such a simple topology, the use of the helper API can be a hindrance
|
|
// so we drop down into the low level API and do it manually.
|
|
//
|
|
// First we need a single node.
|
|
//
|
|
NS_LOG_INFO ("Create Node");
|
|
Ptr<Node> node = CreateObject<Node> ();
|
|
|
|
//
|
|
// Create an emu device, allocate a MAC address and point the device to the
|
|
// Linux device name. The device needs a transmit queueing discipline so
|
|
// create a droptail queue and give it to the device. Finally, "install"
|
|
// the device into the node.
|
|
//
|
|
// Do understand that the ns-3 allocated MAC address will be sent out over
|
|
// your network since the emu net device will spoof it. By default, this
|
|
// address will have an Organizationally Unique Identifier (OUI) of zero.
|
|
// The Internet Assigned Number Authority IANA
|
|
//
|
|
// http://www.iana.org/assignments/ethernet-numbers
|
|
//
|
|
// reports that this OUI is unassigned, and so should not conflict with
|
|
// real hardware on your net. It may raise all kinds of red flags in a
|
|
// real environment to have packets from a device with an obviously bogus
|
|
// OUI flying around. Be aware.
|
|
//
|
|
NS_LOG_INFO ("Create Device");
|
|
Ptr<EmuNetDevice> device = CreateObject<EmuNetDevice> ();
|
|
device->SetAttribute ("Address", Mac48AddressValue (Mac48Address::Allocate ()));
|
|
device->SetAttribute ("DeviceName", StringValue (deviceName));
|
|
|
|
Ptr<Queue> queue = CreateObject<DropTailQueue> ();
|
|
device->SetQueue (queue);
|
|
node->AddDevice (device);
|
|
|
|
//
|
|
// Add a default internet stack to the node. This gets us the ns-3 versions
|
|
// of ARP, IPv4, ICMP, UDP and TCP.
|
|
//
|
|
NS_LOG_INFO ("Add Internet Stack");
|
|
InternetStackHelper internetStackHelper;
|
|
internetStackHelper.Install (node);
|
|
|
|
NS_LOG_INFO ("Create IPv4 Interface");
|
|
Ptr<Ipv4> ipv4 = node->GetObject<Ipv4> ();
|
|
uint32_t interface = ipv4->AddInterface (device);
|
|
Ipv4InterfaceAddress address = Ipv4InterfaceAddress (localIp, localMask);
|
|
ipv4->AddAddress (interface, address);
|
|
ipv4->SetMetric (interface, 1);
|
|
ipv4->SetUp (interface);
|
|
|
|
//
|
|
// When the ping appliation sends its ICMP packet, it will happily send it
|
|
// down the ns-3 protocol stack. We set the IP address of the destination
|
|
// to the address corresponding to example.com above. This address is off
|
|
// our local network so we have got to provide some kind of default route
|
|
// to ns-3 to be able to get that ICMP packet forwarded off of our network.
|
|
//
|
|
// You have got to provide an IP address of a real host that you can send
|
|
// real packets to and have them forwarded off of your local network. One
|
|
// thing you could do is a 'netstat -rn' command and find the IP address of
|
|
// the default gateway on your host and add it below, replacing the
|
|
// "1.2.3.4" string.
|
|
//
|
|
Ipv4Address gateway ("1.2.3.4");
|
|
NS_ABORT_MSG_IF (gateway == "1.2.3.4", "You must change the gateway IP address before running this example");
|
|
|
|
Ipv4StaticRoutingHelper ipv4RoutingHelper;
|
|
Ptr<Ipv4StaticRouting> staticRouting = ipv4RoutingHelper.GetStaticRouting (ipv4);
|
|
staticRouting->SetDefaultRoute (gateway, interface);
|
|
|
|
//
|
|
// Create the ping application. This application knows how to send
|
|
// ICMP echo requests. Setting up the packet sink manually is a bit
|
|
// of a hassle and since there is no law that says we cannot mix the
|
|
// helper API with the low level API, let's just use the helper.
|
|
//
|
|
NS_LOG_INFO ("Create V4Ping Appliation");
|
|
Ptr<V4Ping> app = CreateObject<V4Ping> ();
|
|
app->SetAttribute ("Remote", Ipv4AddressValue (remoteIp));
|
|
node->AddApplication (app);
|
|
app->Start (Seconds (1.0));
|
|
app->Stop (Seconds (5.0));
|
|
|
|
//
|
|
// Give the application a name. This makes life much easier when constructing
|
|
// config paths.
|
|
//
|
|
Names::Add ("app", app);
|
|
|
|
//
|
|
// Hook a trace to print something when the response comes back.
|
|
//
|
|
Config::Connect ("/Names/app/Rtt", MakeCallback (&PingRtt));
|
|
|
|
//
|
|
// Enable a promiscuous pcap trace to see what is coming and going on our device.
|
|
//
|
|
EmuHelper::EnablePcap ("emu-ping", device, true);
|
|
|
|
//
|
|
// Now, do the actual emulation.
|
|
//
|
|
NS_LOG_INFO ("Run Emulation.");
|
|
Simulator::Stop (Seconds (5.0));
|
|
Simulator::Run ();
|
|
Simulator::Destroy ();
|
|
NS_LOG_INFO ("Done.");
|
|
}
|