src/fd-net-device/helper/planetlab-fd-net-device-helper.cc
author Peter D. Barnes, Jr. <barnes26@llnl.gov>
Fri, 26 Sep 2014 15:51:00 -0700
changeset 10968 2d29fee2b7b8
parent 9274 ef5b324097d0
permissions -rw-r--r--
[Bug 1551] Redux: NS_LOG_COMPONENT_DEFINE inside or outside of ns3 namespace?
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
9271
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
     1
/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
     2
/*
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
     3
 * Copyright (c) 2012 INRIA
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
     4
 *
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
     5
 * This program is free software; you can redistribute it and/or modify
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
     6
 * it under the terms of the GNU General Public License version 2 as
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
     7
 * published by the Free Software Foundation;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
     8
 *
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
     9
 * This program is distributed in the hope that it will be useful,
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    10
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    11
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    12
 * GNU General Public License for more details.
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    13
 *
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    14
 * You should have received a copy of the GNU General Public License
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    15
 * along with this program; if not, write to the Free Software
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    16
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    17
 *
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    18
 * Author: Alina Quereilhac <alina.quereilhac@inria.fr>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    19
 *         Claudio Freire <klaussfreire@sourceforge.net>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    20
 *
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    21
 */
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    22
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    23
#include "planetlab-fd-net-device-helper.h"
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    24
#include "encode-decode.h"
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    25
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    26
#include "ns3/abort.h"
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    27
#include "ns3/config.h"
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    28
#include "ns3/fd-net-device.h"
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    29
#include "ns3/log.h"
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    30
#include "ns3/names.h"
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    31
#include "ns3/object-factory.h"
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    32
#include "ns3/packet.h"
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    33
#include "ns3/simulator.h"
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    34
#include "ns3/trace-helper.h"
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    35
#include "ns3/internet-module.h"
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    36
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    37
#include <arpa/inet.h>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    38
#include <errno.h>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    39
#include <iostream>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    40
#include <iomanip>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    41
#include <limits>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    42
#include <linux/if_tun.h>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    43
#include <memory>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    44
#include <net/ethernet.h>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    45
#include <net/if.h>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    46
#include <netinet/in.h>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    47
#include <netpacket/packet.h>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    48
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    49
#include <stdlib.h>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    50
#include <string.h>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    51
#include <sys/wait.h>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    52
#include <sys/stat.h>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    53
#include <sys/socket.h>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    54
#include <sys/un.h>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    55
#include <sys/ioctl.h>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    56
#include <time.h>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    57
#include <unistd.h>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    58
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    59
#include <string>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    60
10968
2d29fee2b7b8 [Bug 1551] Redux: NS_LOG_COMPONENT_DEFINE inside or outside of ns3 namespace?
Peter D. Barnes, Jr. <barnes26@llnl.gov>
parents: 9274
diff changeset
    61
namespace ns3 {
9271
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    62
10968
2d29fee2b7b8 [Bug 1551] Redux: NS_LOG_COMPONENT_DEFINE inside or outside of ns3 namespace?
Peter D. Barnes, Jr. <barnes26@llnl.gov>
parents: 9274
diff changeset
    63
NS_LOG_COMPONENT_DEFINE ("PlanetLabFdNetDeviceHelper");
9271
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    64
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    65
#define PLANETLAB_MAGIC 75867
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    66
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    67
PlanetLabFdNetDeviceHelper::PlanetLabFdNetDeviceHelper ()
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    68
{
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    69
  m_tapIp = Ipv4Address ("255.255.255.255");
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    70
  m_tapMask = Ipv4Mask ("255.255.255.255");
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    71
}
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    72
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    73
void
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    74
PlanetLabFdNetDeviceHelper::SetTapIpAddress (Ipv4Address address)
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    75
{
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    76
  m_tapIp = address;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    77
}
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    78
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    79
void
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    80
PlanetLabFdNetDeviceHelper::SetTapMask (Ipv4Mask mask)
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    81
{
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    82
  m_tapMask = mask;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    83
}
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    84
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    85
Ptr<NetDevice>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    86
PlanetLabFdNetDeviceHelper::InstallPriv (Ptr<Node> node) const
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    87
{
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    88
  Ptr<NetDevice> d = FdNetDeviceHelper::InstallPriv (node);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    89
  Ptr<FdNetDevice> device = d->GetObject<FdNetDevice> ();
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    90
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    91
  //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    92
  // The PlanetLab mechanism to create a TAP device doesn't allow
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    93
  // for the moment to set the IFF_NOPI flag. In consequence, a PI
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    94
  // header will be present in the traffic.
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    95
  // We need to explicitly set the encapsulation mode to DIXPI,
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    96
  // so the FdNetDevice is able to treat correctly the traffic
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    97
  // traversing TAP device.
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    98
  //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    99
  Ptr<FdNetDevice> fdnd = device->GetObject<FdNetDevice> ();
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   100
  fdnd->SetEncapsulationMode (FdNetDevice::DIXPI);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   101
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   102
  SetFileDescriptor (device);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   103
  return device;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   104
}
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   105
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   106
void
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   107
PlanetLabFdNetDeviceHelper::SetFileDescriptor (Ptr<FdNetDevice> device) const
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   108
{
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   109
  NS_LOG_LOGIC ("Creating TAP device");
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   110
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   111
  //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   112
  // Call out to a separate process running as suid root in order to create a
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   113
  // TAP device.  We do this to avoid having the entire simulation running as root.
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   114
  //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   115
  int fd = CreateFileDescriptor ();
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   116
  device->SetFileDescriptor (fd);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   117
}
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   118
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   119
int
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   120
PlanetLabFdNetDeviceHelper::CreateFileDescriptor (void) const
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   121
{
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   122
  NS_LOG_FUNCTION (this);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   123
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   124
  //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   125
  // We're going to fork and exec that program soon, but first we need to have
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   126
  // a socket to talk to it with.  So we create a local interprocess (Unix)
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   127
  // socket for that purpose.
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   128
  //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   129
  int sock = socket (PF_UNIX, SOCK_DGRAM, 0);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   130
  NS_ABORT_MSG_IF (sock == -1, "PlanetLabFdNetDeviceHelper::CreateFileDescriptor(): Unix socket creation error, errno = " << strerror (errno));
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   131
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   132
  //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   133
  // Bind to that socket and let the kernel allocate an endpoint
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   134
  //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   135
  struct sockaddr_un un;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   136
  memset (&un, 0, sizeof (un));
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   137
  un.sun_family = AF_UNIX;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   138
  int status = bind (sock, (struct sockaddr*)&un, sizeof (sa_family_t));
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   139
  NS_ABORT_MSG_IF (status == -1, "PlanetLabFdNetDeviceHelper::CreateFileDescriptor(): Could not bind(): errno = " << strerror (errno));
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   140
  NS_LOG_INFO ("Created Unix socket");
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   141
  NS_LOG_INFO ("sun_family = " << un.sun_family);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   142
  NS_LOG_INFO ("sun_path = " << un.sun_path);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   143
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   144
  //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   145
  // We have a socket here, but we want to get it there -- to the program we're
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   146
  // going to exec.  What we'll do is to do a getsockname and then encode the
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   147
  // resulting address information as a string, and then send the string to the
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   148
  // program as an argument.  So we need to get the sock name.
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   149
  //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   150
  socklen_t len = sizeof (un);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   151
  status = getsockname (sock, (struct sockaddr*)&un, &len);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   152
  NS_ABORT_MSG_IF (status == -1, "PlanetLabFdNetDeviceHelper::CreateFileDescriptor(): Could not getsockname(): errno = " << strerror (errno));
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   153
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   154
  //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   155
  // Now encode that socket name (family and path) as a string of hex digits
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   156
  //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   157
  std::string path = BufferToString ((uint8_t *)&un, len);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   158
  NS_LOG_INFO ("Encoded Unix socket as \"" << path << "\"");
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   159
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   160
  //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   161
  // Fork and exec the process to create our socket.  If we're us (the parent)
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   162
  // we wait for the child (the creator) to complete and read the socket it
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   163
  // created and passed back using the ancillary data mechanism.
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   164
  //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   165
  pid_t pid = ::fork ();
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   166
  if (pid == 0)
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   167
    {
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   168
      NS_LOG_DEBUG ("Child process");
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   169
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   170
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   171
      // build a command line argument from the encoded endpoint string that
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   172
      // the socket creation process will use to figure out how to respond to
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   173
      // the (now) parent process.  We're going to have to give this program
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   174
      // quite a bit of information.
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   175
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   176
      // -i<IP-address> The IP address to assign to the new tap device;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   177
      // -n<network-prefix> The network prefix to assign to the new tap device;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   178
      // -t Set teh IFF_TAP flag
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   179
      // -p<path> the path to the unix socket described above.
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   180
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   181
      // Example tap-creator -i1.2.3.1 -n24 -t -pblah
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   182
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   183
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   184
      std::ostringstream ossIp;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   185
      ossIp << "-i" << m_tapIp;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   186
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   187
      std::ostringstream ossPrefix;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   188
      ossPrefix << "-n" << m_tapMask.GetPrefixLength ();
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   189
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   190
      std::ostringstream ossMode;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   191
      ossMode << "-t";
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   192
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   193
      std::ostringstream ossPath;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   194
      ossPath << "-p" << path;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   195
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   196
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   197
      // Execute the socket creation process image.
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   198
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   199
      status = ::execlp (PLANETLAB_TAP_CREATOR,
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   200
                         PLANETLAB_TAP_CREATOR,               // argv[0] (filename)
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   201
                         ossIp.str ().c_str (),               // argv[1] (-i<IP address>)
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   202
                         ossPrefix.str ().c_str (),           // argv[2] (-n<prefix>)
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   203
                         ossMode.str ().c_str (),             // argv[3] (-t <tap>)
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   204
                         ossPath.str ().c_str (),             // argv[4] (-p<path>)
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   205
                         (char *)NULL);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   206
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   207
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   208
      // If the execlp successfully completes, it never returns.  If it returns it failed or the OS is
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   209
      // broken.  In either case, we bail.
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   210
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   211
      NS_FATAL_ERROR ("PlanetLabFdNetDeviceHelper::CreateFileDescriptor(): Back from execlp(), errno = " << ::strerror (errno));
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   212
    }
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   213
  else
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   214
    {
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   215
      NS_LOG_DEBUG ("Parent process");
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   216
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   217
      // We're the process running the emu net device.  We need to wait for the
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   218
      // socket creator process to finish its job.
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   219
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   220
      int st;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   221
      pid_t waited = waitpid (pid, &st, 0);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   222
      NS_ABORT_MSG_IF (waited == -1, "PlanetLabFdNetDeviceHelper::CreateFileDescriptor(): waitpid() fails, errno = " << strerror (errno));
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   223
      NS_ASSERT_MSG (pid == waited, "PlanetLabFdNetDeviceHelper::CreateFileDescriptor(): pid mismatch");
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   224
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   225
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   226
      // Check to see if the socket creator exited normally and then take a
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   227
      // look at the exit code.  If it bailed, so should we.  If it didn't
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   228
      // even exit normally, we bail too.
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   229
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   230
      if (WIFEXITED (st))
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   231
        {
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   232
          int exitStatus = WEXITSTATUS (st);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   233
          NS_ABORT_MSG_IF (exitStatus != 0,
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   234
                           "PlanetLabFdNetDeviceHelper::CreateFileDescriptor(): socket creator exited normally with status " << exitStatus);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   235
        }
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   236
      else
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   237
        {
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   238
          NS_FATAL_ERROR ("PlanetLabFdNetDeviceHelper::CreateFileDescriptor(): socket creator exited abnormally");
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   239
        }
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   240
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   241
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   242
      // At this point, the socket creator has run successfully and should
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   243
      // have created our tap device, initialized it with the information we
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   244
      // passed and sent it back to the socket address we provided.  A socket
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   245
      // (fd) we can use to talk to this tap device should be waiting on the
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   246
      // Unix socket we set up to receive information back from the creator
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   247
      // program.  We've got to do a bunch of grunt work to get at it, though.
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   248
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   249
      // The struct iovec below is part of a scatter-gather list.  It describes a
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   250
      // buffer.  In this case, it describes a buffer (an integer) that will
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   251
      // get the data that comes back from the socket creator process.  It will
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   252
      // be a magic number that we use as a consistency/sanity check.
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   253
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   254
      struct iovec iov;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   255
      uint32_t magic;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   256
      iov.iov_base = &magic;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   257
      iov.iov_len = sizeof(magic);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   258
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   259
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   260
      // The CMSG macros you'll see below are used to create and access control
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   261
      // messages (which is another name for ancillary data).  The ancillary
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   262
      // data is made up of pairs of struct cmsghdr structures and associated
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   263
      // data arrays.
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   264
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   265
      // First, we're going to allocate a buffer on the stack to receive our
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   266
      // data array (that contains the socket).  Sometimes you'll see this called
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   267
      // an "ancillary element" but the msghdr uses the control message termimology
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   268
      // so we call it "control."
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   269
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   270
      size_t msg_size = sizeof(int);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   271
      char control[CMSG_SPACE (msg_size)];
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   272
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   273
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   274
      // There is a msghdr that is used to minimize the number of parameters
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   275
      // passed to recvmsg (which we will use to receive our ancillary data).
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   276
      // This structure uses terminology corresponding to control messages, so
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   277
      // you'll see msg_control, which is the pointer to the ancillary data and
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   278
      // controllen which is the size of the ancillary data array.
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   279
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   280
      // So, initialize the message header that describes the ancillary/control
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   281
      // data we expect to receive and point it to buffer.
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   282
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   283
      struct msghdr msg;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   284
      msg.msg_name = 0;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   285
      msg.msg_namelen = 0;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   286
      msg.msg_iov = &iov;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   287
      msg.msg_iovlen = 1;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   288
      msg.msg_control = control;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   289
      msg.msg_controllen = sizeof (control);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   290
      msg.msg_flags = 0;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   291
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   292
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   293
      // Now we can actually receive the interesting bits from the tap
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   294
      // creator process.  Lots of pain to get four bytes.
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   295
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   296
      ssize_t bytesRead = recvmsg (sock, &msg, 0);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   297
      NS_ABORT_MSG_IF (bytesRead != sizeof(int), "PlanetLabFdNetDeviceHelper::CreateFileDescriptor(): Wrong byte count from socket creator");
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   298
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   299
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   300
      // There may be a number of message headers/ancillary data arrays coming in.
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   301
      // Let's look for the one with a type SCM_RIGHTS which indicates it's the
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   302
      // one we're interested in.
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   303
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   304
      struct cmsghdr *cmsg;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   305
      for (cmsg = CMSG_FIRSTHDR (&msg); cmsg != NULL; cmsg = CMSG_NXTHDR (&msg, cmsg))
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   306
        {
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   307
          if (cmsg->cmsg_level == SOL_SOCKET
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   308
              && cmsg->cmsg_type == SCM_RIGHTS)
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   309
            {
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   310
              //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   311
              // This is the type of message we want.  Check to see if the magic
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   312
              // number is correct and then pull out the socket we care about if
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   313
              // it matches
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   314
              //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   315
              if (magic == PLANETLAB_MAGIC)
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   316
                {
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   317
                  NS_LOG_INFO ("Got SCM_RIGHTS with correct magic " << magic);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   318
                  int *rawSocket = (int*)CMSG_DATA (cmsg);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   319
                  NS_LOG_INFO ("Got the socket from the socket creator = " << *rawSocket);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   320
                  return *rawSocket;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   321
                }
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   322
              else
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   323
                {
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   324
                  NS_LOG_INFO ("Got SCM_RIGHTS, but with bad magic " << magic);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   325
                }
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   326
            }
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   327
        }
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   328
      NS_FATAL_ERROR ("Did not get the raw socket from the socket creator");
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   329
    }
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   330
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   331
}
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   332
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   333
} // namespace ns3
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   334
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   335