src/fd-net-device/helper/tap-fd-net-device-helper.cc
author alina
Wed, 27 Mar 2013 02:14:57 +0100
changeset 9271 85e753b6676c
child 9274 ef5b324097d0
permissions -rw-r--r--
Merge fd-net-device into ns-3-dev
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, 2012 University of Washington
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
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    19
#include "tap-fd-net-device-helper.h"
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    20
#include "encode-decode.h"
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
#include "ns3/abort.h"
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    23
#include "ns3/config.h"
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    24
#include "ns3/fd-net-device.h"
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    25
#include "ns3/log.h"
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    26
#include "ns3/names.h"
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    27
#include "ns3/object-factory.h"
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    28
#include "ns3/packet.h"
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    29
#include "ns3/simulator.h"
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    30
#include "ns3/trace-helper.h"
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    31
#include "ns3/internet-module.h"
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    32
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    33
#include <arpa/inet.h>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    34
#include <errno.h>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    35
#include <iostream>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    36
#include <iomanip>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    37
#include <limits>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    38
#include <linux/if_tun.h>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    39
#include <memory>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    40
#include <net/ethernet.h>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    41
#include <net/if.h>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    42
#include <netinet/in.h>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    43
#include <netpacket/packet.h>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    44
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    45
#include <stdlib.h>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    46
#include <string.h>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    47
#include <sys/wait.h>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    48
#include <sys/stat.h>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    49
#include <sys/socket.h>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    50
#include <sys/un.h>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    51
#include <sys/ioctl.h>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    52
#include <time.h>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    53
#include <unistd.h>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    54
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    55
#include <string>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    56
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    57
NS_LOG_COMPONENT_DEFINE ("TapFdNetDeviceHelper");
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
namespace ns3 {
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    60
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    61
#define TAP_MAGIC 95549
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    62
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    63
TapFdNetDeviceHelper::TapFdNetDeviceHelper ()
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
  m_deviceName = "";
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    66
  m_modePi = false;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    67
  m_tapIp4 = "";
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    68
  m_tapMask4 = "";
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    69
  m_tapIp6 = "";
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    70
  m_tapPrefix6 = 64;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    71
  m_tapMac = Mac48Address::Allocate ();
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
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    74
void
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    75
TapFdNetDeviceHelper::SetModePi (bool modePi)
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    76
{
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    77
  m_modePi = modePi;
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
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    80
void
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    81
TapFdNetDeviceHelper::SetTapIpv4Address (Ipv4Address address)
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    82
{
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    83
  m_tapIp4 = address;
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
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    86
void
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    87
TapFdNetDeviceHelper::SetTapIpv4Mask (Ipv4Mask mask)
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    88
{
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    89
  m_tapMask4 = mask;
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
void
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    93
TapFdNetDeviceHelper::SetTapIpv6Address (Ipv6Address address)
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    94
{
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    95
  m_tapIp6 = address;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    96
}
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    97
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    98
void
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
    99
TapFdNetDeviceHelper::SetTapIpv6Prefix (int prefix)
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   100
{
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   101
  m_tapPrefix6 = prefix;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   102
}
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   103
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   104
void
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   105
TapFdNetDeviceHelper::SetTapMacAddress (Mac48Address mac)
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   106
{
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   107
  m_tapMac = mac;
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
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   110
Ptr<NetDevice>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   111
TapFdNetDeviceHelper::InstallPriv (Ptr<Node> node) const
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   112
{
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   113
  Ptr<NetDevice> d = FdNetDeviceHelper::InstallPriv (node);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   114
  Ptr<FdNetDevice> device = d->GetObject<FdNetDevice> ();
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   115
  Ptr<FdNetDevice> fdnd = device->GetObject<FdNetDevice> ();
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   116
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
  // We need to explicitly set the encapsulation mode for the traffic
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   119
  // traversing the TAP device, so the FdNetDevice is able to know
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   120
  // how to treat the traffic in a way that in compatible with the
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   121
  // TAP device.
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   122
  //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   123
  if (m_modePi)
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
      fdnd->SetEncapsulationMode (FdNetDevice::DIXPI);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   126
    }
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   127
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   128
  SetFileDescriptor (device);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   129
  return device;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   130
}
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
void
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   133
TapFdNetDeviceHelper::SetFileDescriptor (Ptr<FdNetDevice> device) const
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
  NS_LOG_LOGIC ("Creating TAP device");
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   136
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   137
  //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   138
  // 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
   139
  // 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
   140
  //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   141
  int fd = CreateFileDescriptor ();
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   142
  device->SetFileDescriptor (fd);
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
int
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   146
TapFdNetDeviceHelper::CreateFileDescriptor (void) const
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   147
{
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   148
  NS_LOG_FUNCTION (this);
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
#ifdef HAVE_TAP_CREATOR
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   151
  //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   152
  // 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
   153
  // 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
   154
  // socket for that purpose.
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   155
  //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   156
  int sock = socket (PF_UNIX, SOCK_DGRAM, 0);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   157
  NS_ABORT_MSG_IF (sock == -1, "TapFdNetDeviceHelper::CreateFileDescriptor(): Unix socket creation error, errno = " << strerror (errno));
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   158
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
  // Bind to that socket and let the kernel allocate an endpoint
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   161
  //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   162
  struct sockaddr_un un;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   163
  memset (&un, 0, sizeof (un));
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   164
  un.sun_family = AF_UNIX;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   165
  int status = bind (sock, (struct sockaddr*)&un, sizeof (sa_family_t));
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   166
  NS_ABORT_MSG_IF (status == -1, "TapFdNetDeviceHelper::CreateFileDescriptor(): Could not bind(): errno = " << strerror (errno));
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   167
  NS_LOG_INFO ("Created Unix socket");
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   168
  NS_LOG_INFO ("sun_family = " << un.sun_family);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   169
  NS_LOG_INFO ("sun_path = " << un.sun_path);
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
  //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   172
  // 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
   173
  // 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
   174
  // 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
   175
  // 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
   176
  //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   177
  socklen_t len = sizeof (un);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   178
  status = getsockname (sock, (struct sockaddr*)&un, &len);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   179
  NS_ABORT_MSG_IF (status == -1, "TapFdNetDeviceHelper::CreateFileDescriptor(): Could not getsockname(): errno = " << strerror (errno));
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
  //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   182
  // 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
   183
  //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   184
  std::string path = BufferToString ((uint8_t *)&un, len);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   185
  NS_LOG_INFO ("Encoded Unix socket as \"" << path << "\"");
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
  //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   188
  // 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
   189
  // 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
   190
  // created and passed back using the ancillary data mechanism.
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   191
  //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   192
  pid_t pid = ::fork ();
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   193
  if (pid == 0)
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   194
    {
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   195
      NS_LOG_DEBUG ("Child process");
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
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   198
      // build a command line argument from the encoded endpoint string that
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   199
      // 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
   200
      // 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
   201
      // quite a bit of information.
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   202
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   203
      // -d<device-name> The name of the tap device we want to create;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   204
      // -m<MAC-address> The MAC-48 address to assign to the new tap device;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   205
      // -i<IPv4-address> The IP v4 address to assign to the new tap device;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   206
      // -I<IPv6-address> The IP v6 address to assign to the new tap device;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   207
      // -n<network-IPv4-mask> The network IPv4 mask to assign to the new tap device;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   208
      // -N<network-IPv6-mask> The network IPv6 mask to assign to the new tap device;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   209
      // -t Set teh IFF_TAP flag
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   210
      // -h Set the IFF_NO_PI flag
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   211
      // -p<path> the path to the unix socket described above.
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
      // Example tap-creator -dnewdev -i1.2.3.1 -m08:00:2e:00:01:23 -n255.255.255.0 -t -h -pblah
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
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
      // The device-name is something we may want the system to make up in
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   218
      // every case.  We also rely on it being configured via an Attribute
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   219
      // through the helper.  By default, it is set to the empty string
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   220
      // which tells the system to make up a device name such as "tap123".
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   221
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   222
      std::ostringstream ossDeviceName;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   223
      if (m_deviceName != "")
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
          ossDeviceName << "-d" << m_deviceName;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   226
        }
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   227
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   228
      std::ostringstream ossMac;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   229
      ossMac << "-m" << m_tapMac;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   230
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   231
      std::ostringstream ossIp4;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   232
      if (m_tapIp4 != "")
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   233
        {
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   234
          ossIp4 << "-i" << m_tapIp4;
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
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   237
      std::ostringstream ossIp6;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   238
      if (m_tapIp6 != "")
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
          ossIp6 << "-I" << m_tapIp6;
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
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   243
      std::ostringstream ossNetmask4;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   244
      if (m_tapMask4 != "" )
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   245
        {
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   246
          ossNetmask4 << "-n" << m_tapMask4;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   247
        }
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
      std::ostringstream ossPrefix6;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   250
      ossPrefix6 << "-P" << m_tapPrefix6;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   251
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   252
      std::ostringstream ossMode;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   253
      ossMode << "-t";
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   254
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   255
      std::ostringstream ossPI;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   256
      if (m_modePi)
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   257
        {
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   258
          ossPI << "-h";
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
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   261
      std::ostringstream ossPath;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   262
      ossPath << "-p" << path;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   263
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
      // Execute the socket creation process image.
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   266
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   267
      status = ::execlp (TAP_CREATOR,
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   268
                         TAP_CREATOR,                       // argv[0] (filename)
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   269
                         ossDeviceName.str ().c_str (),     // argv[1] (-d<device name>)
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   270
                         ossMac.str ().c_str (),            // argv[2] (-m<MAC address>
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   271
                         ossIp4.str ().c_str (),            // argv[3] (-i<IP v4 address>)
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   272
                         ossIp6.str ().c_str (),            // argv[4] (-I<IP v6 address>)
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   273
                         ossNetmask4.str ().c_str (),       // argv[5] (-n<IP v4 net mask>)
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   274
                         ossPrefix6.str ().c_str (),        // argv[6] (-P<IP v6 prefix>)
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   275
                         ossMode.str ().c_str (),           // argv[7] (-t <tap>)
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   276
                         ossPI.str ().c_str (),             // argv[8] (-h <pi>)
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   277
                         ossPath.str ().c_str (),           // argv[9] (-p<path>)
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   278
                         (char *)NULL);
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
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   281
      // 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
   282
      // broken.  In either case, we bail.
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   283
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   284
      NS_FATAL_ERROR ("TapFdNetDeviceHelper::CreateFileDescriptor(): Back from execlp(), errno = " << ::strerror (errno));
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   285
    }
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   286
  else
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   287
    {
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   288
      NS_LOG_DEBUG ("Parent process");
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   289
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   290
      // 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
   291
      // socket creator process to finish its job.
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
      int st;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   294
      pid_t waited = waitpid (pid, &st, 0);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   295
      NS_ABORT_MSG_IF (waited == -1, "TapFdNetDeviceHelper::CreateFileDescriptor(): waitpid() fails, errno = " << strerror (errno));
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   296
      NS_ASSERT_MSG (pid == waited, "TapFdNetDeviceHelper::CreateFileDescriptor(): pid mismatch");
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   297
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
      // 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
   300
      // 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
   301
      // even exit normally, we bail too.
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   302
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   303
      if (WIFEXITED (st))
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   304
        {
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   305
          int exitStatus = WEXITSTATUS (st);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   306
          NS_ABORT_MSG_IF (exitStatus != 0,
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   307
                           "TapFdNetDeviceHelper::CreateFileDescriptor(): socket creator exited normally with status " << exitStatus);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   308
        }
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   309
      else
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
          NS_FATAL_ERROR ("TapFdNetDeviceHelper::CreateFileDescriptor(): socket creator exited abnormally");
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   312
        }
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   313
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
      // At this point, the socket creator has run successfully and should
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   316
      // have created our tap device, initialized it with the information we
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   317
      // 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
   318
      // (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
   319
      // 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
   320
      // 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
   321
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   322
      // 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
   323
      // 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
   324
      // 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
   325
      // 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
   326
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   327
      struct iovec iov;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   328
      uint32_t magic;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   329
      iov.iov_base = &magic;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   330
      iov.iov_len = sizeof(magic);
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
      // 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
   334
      // messages (which is another name for ancillary data).  The ancillary
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   335
      // 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
   336
      // data arrays.
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   337
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   338
      // 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
   339
      // 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
   340
      // an "ancillary element" but the msghdr uses the control message termimology
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   341
      // so we call it "control."
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   342
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   343
      size_t msg_size = sizeof(int);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   344
      char control[CMSG_SPACE (msg_size)];
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   345
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   346
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   347
      // 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
   348
      // 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
   349
      // This structure uses terminology corresponding to control messages, so
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   350
      // 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
   351
      // controllen which is the size of the ancillary data array.
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   352
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   353
      // So, initialize the message header that describes the ancillary/control
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   354
      // data we expect to receive and point it to buffer.
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   355
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   356
      struct msghdr msg;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   357
      msg.msg_name = 0;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   358
      msg.msg_namelen = 0;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   359
      msg.msg_iov = &iov;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   360
      msg.msg_iovlen = 1;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   361
      msg.msg_control = control;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   362
      msg.msg_controllen = sizeof (control);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   363
      msg.msg_flags = 0;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   364
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   365
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   366
      // Now we can actually receive the interesting bits from the tap
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   367
      // creator process.  Lots of pain to get four bytes.
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   368
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   369
      ssize_t bytesRead = recvmsg (sock, &msg, 0);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   370
      NS_ABORT_MSG_IF (bytesRead != sizeof(int), "TapFdNetDeviceHelper::CreateFileDescriptor(): Wrong byte count from socket creator");
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   371
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   372
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   373
      // 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
   374
      // 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
   375
      // one we're interested in.
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   376
      //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   377
      struct cmsghdr *cmsg;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   378
      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
   379
        {
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   380
          if (cmsg->cmsg_level == SOL_SOCKET
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   381
              && cmsg->cmsg_type == SCM_RIGHTS)
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   382
            {
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   383
              //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   384
              // 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
   385
              // 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
   386
              // it matches
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   387
              //
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   388
              if (magic == TAP_MAGIC)
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   389
                {
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   390
                  NS_LOG_INFO ("Got SCM_RIGHTS with correct magic " << magic);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   391
                  int *rawSocket = (int*)CMSG_DATA (cmsg);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   392
                  NS_LOG_INFO ("Got the socket from the socket creator = " << *rawSocket);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   393
                  return *rawSocket;
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   394
                }
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   395
              else
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   396
                {
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   397
                  NS_LOG_INFO ("Got SCM_RIGHTS, but with bad magic " << magic);
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   398
                }
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   399
            }
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   400
        }
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   401
      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
   402
    }
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   403
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   404
#else
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   405
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   406
  NS_FATAL_ERROR ("TAP_CREATOR is not defined in your system.");
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   407
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   408
#endif /* HAVE_TAP_CREATOR */
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   409
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   410
}
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   411
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   412
} // namespace ns3
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   413
85e753b6676c Merge fd-net-device into ns-3-dev
alina
parents:
diff changeset
   414