gem5  v21.0.0.0
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
ethertap.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2003-2005 The Regents of The University of Michigan
3  * All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions are
7  * met: redistributions of source code must retain the above copyright
8  * notice, this list of conditions and the following disclaimer;
9  * redistributions in binary form must reproduce the above copyright
10  * notice, this list of conditions and the following disclaimer in the
11  * documentation and/or other materials provided with the distribution;
12  * neither the name of the copyright holders nor the names of its
13  * contributors may be used to endorse or promote products derived from
14  * this software without specific prior written permission.
15  *
16  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
17  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
18  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
19  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
20  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
21  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
22  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
26  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27  */
28 
29 /* @file
30  * Interface to connect a simulated ethernet device to the real world
31  */
32 
33 #include "dev/net/ethertap.hh"
34 
35 #if defined(__OpenBSD__) || defined(__APPLE__)
36 #include <sys/param.h>
37 
38 #endif
39 
40 #if USE_TUNTAP && defined(__linux__)
41 #if 1 // Hide from the style checker since these have to be out of order.
42 #include <sys/socket.h> // Has to be included before if.h for some reason.
43 
44 #endif
45 
46 #include <linux/if.h>
47 #include <linux/if_tun.h>
48 
49 #endif
50 
51 #include <fcntl.h>
52 #include <netinet/in.h>
53 #include <sys/ioctl.h>
54 #include <unistd.h>
55 
56 #include <cstring>
57 #include <deque>
58 #include <string>
59 
60 #include "base/logging.hh"
61 #include "base/pollevent.hh"
62 #include "base/socket.hh"
63 #include "base/trace.hh"
64 #include "debug/Ethernet.hh"
65 #include "debug/EthernetData.hh"
66 #include "dev/net/etherdump.hh"
67 #include "dev/net/etherint.hh"
68 #include "dev/net/etherpkt.hh"
69 
70 class TapEvent : public PollEvent
71 {
72  protected:
74 
75  public:
76  TapEvent(EtherTapBase *_tap, int fd, int e)
77  : PollEvent(fd, e), tap(_tap) {}
78 
79  void
80  process(int revent) override
81  {
82  // Ensure that our event queue is active. It may not be since we get
83  // here from the PollQueue whenever a real packet happens to arrive.
85 
86  tap->recvReal(revent);
87  }
88 };
89 
91  : SimObject(p), buflen(p.bufsz), dump(p.dump), event(NULL),
92  interface(NULL),
93  txEvent([this]{ retransmit(); }, "EtherTapBase retransmit")
94 {
95  buffer = new uint8_t[buflen];
96  interface = new EtherTapInt(name() + ".interface", this);
97 }
98 
100 {
101  delete buffer;
102  delete event;
103  delete interface;
104 }
105 
106 void
108 {
110  uint8_t *buffer = (uint8_t *)this->buffer;
112 
113  bool tapevent_present = false;
114  if (event) {
115  tapevent_present = true;
116  SERIALIZE_SCALAR(tapevent_present);
117  event->serialize(cp);
118  } else {
119  SERIALIZE_SCALAR(tapevent_present);
120  }
121 }
122 
123 void
125 {
127  uint8_t *buffer = (uint8_t *)this->buffer;
129 
130  bool tapevent_present;
131  UNSERIALIZE_SCALAR(tapevent_present);
132  if (tapevent_present) {
133  event = new TapEvent(this, 0, 0);
134  event->unserialize(cp);
135  if (event->queued())
137  }
138 }
139 
140 
141 void
143 {
144  assert(!event);
145  event = new TapEvent(this, fd, POLLIN|POLLERR);
147 }
148 
149 void
151 {
152  assert(event);
153  delete event;
154  event = NULL;
155 }
156 
157 
158 Port &
159 EtherTapBase::getPort(const std::string &if_name, PortID idx)
160 {
161  if (if_name == "tap")
162  return *interface;
163  return SimObject::getPort(if_name, idx);
164 }
165 
166 bool
168 {
169  if (dump)
170  dump->dump(packet);
171 
172  DPRINTF(Ethernet, "EtherTap sim->real len=%d\n", packet->length);
173  DDUMP(EthernetData, packet->data, packet->length);
174 
175  bool success = sendReal(packet->data, packet->length);
176 
177  interface->recvDone();
178 
179  return success;
180 }
181 
182 void
184 {
185  EthPacketPtr packet;
186  packet = std::make_shared<EthPacketData>(len);
187  packet->length = len;
188  packet->simLength = len;
189  memcpy(packet->data, data, len);
190 
191  DPRINTF(Ethernet, "EtherTap real->sim len=%d\n", packet->length);
192  DDUMP(EthernetData, packet->data, packet->length);
193  if (!packetBuffer.empty() || !interface->sendPacket(packet)) {
194  DPRINTF(Ethernet, "bus busy...buffer for retransmission\n");
195  packetBuffer.push(packet);
196  if (!txEvent.scheduled())
198  } else if (dump) {
199  dump->dump(packet);
200  }
201 }
202 
203 void
205 {
206  if (packetBuffer.empty())
207  return;
208 
209  EthPacketPtr packet = packetBuffer.front();
210  if (interface->sendPacket(packet)) {
211  if (dump)
212  dump->dump(packet);
213  DPRINTF(Ethernet, "EtherTap retransmit\n");
214  packetBuffer.front() = NULL;
215  packetBuffer.pop();
216  }
217 
218  if (!packetBuffer.empty() && !txEvent.scheduled())
220 }
221 
222 
224 {
225  protected:
226  class Event : public PollEvent
227  {
228  protected:
230 
231  public:
232  Event(TapListener *l, int fd, int e) : PollEvent(fd, e), listener(l) {}
233 
234  void process(int revent) override { listener->accept(); }
235  };
236 
237  friend class Event;
239 
240  void accept();
241 
242  protected:
245  int port;
246 
247  public:
248  TapListener(EtherTapStub *t, int p) : event(NULL), tap(t), port(p) {}
249  ~TapListener() { delete event; }
250 
251  void listen();
252 };
253 
254 void
256 {
257  while (!listener.listen(port, true)) {
258  DPRINTF(Ethernet, "TapListener(listen): Can't bind port %d\n", port);
259  port++;
260  }
261 
262  ccprintf(std::cerr, "Listening for tap connection on port %d\n", port);
263  event = new Event(this, listener.getfd(), POLLIN|POLLERR);
265 }
266 
267 void
269 {
270  // As a consequence of being called from the PollQueue, we might
271  // have been called from a different thread. Migrate to "our"
272  // thread.
274 
275  if (!listener.islistening())
276  panic("TapListener(accept): cannot accept if we're not listening!");
277 
278  int sfd = listener.accept(true);
279  if (sfd != -1)
280  tap->attach(sfd);
281 }
282 
283 
285 {
287  fatal("All listeners are disabled! EtherTapStub can't work!");
288 
289  listener = new TapListener(this, p.port);
290  listener->listen();
291 }
292 
294 {
295  delete listener;
296 }
297 
298 void
300 {
302 
306 }
307 
308 void
310 {
312 
316 }
317 
318 
319 void
321 {
322  if (socket != -1)
323  close(fd);
324 
325  buffer_used = 0;
326  frame_len = 0;
327  socket = fd;
328  DPRINTF(Ethernet, "EtherTapStub attached\n");
329  pollFd(socket);
330 }
331 
332 void
334 {
335  DPRINTF(Ethernet, "EtherTapStub detached\n");
336  stopPolling();
337  close(socket);
338  socket = -1;
339 }
340 
341 void
343 {
344  if (revent & POLLERR) {
345  detach();
346  return;
347  }
348 
349  if (!(revent & POLLIN))
350  return;
351 
352  // Read in as much of the new data as we can.
353  int len = read(socket, buffer + buffer_used, buflen - buffer_used);
354  if (len == 0) {
355  detach();
356  return;
357  }
358  buffer_used += len;
359 
360  // If there's not enough data for the frame length, wait for more.
361  if (buffer_used < sizeof(uint32_t))
362  return;
363 
364  if (frame_len == 0)
365  frame_len = ntohl(*(uint32_t *)buffer);
366 
367  DPRINTF(Ethernet, "Received data from peer: len=%d buffer_used=%d "
368  "frame_len=%d\n", len, buffer_used, frame_len);
369 
370  uint8_t *frame_start = &buffer[sizeof(uint32_t)];
371  while (frame_len != 0 && buffer_used >= frame_len + sizeof(uint32_t)) {
372  sendSimulated(frame_start, frame_len);
373 
374  // Bookkeeping.
375  buffer_used -= frame_len + sizeof(uint32_t);
376  if (buffer_used > 0) {
377  // If there's still any data left, move it into position.
378  memmove(buffer, frame_start + frame_len, buffer_used);
379  }
380  frame_len = 0;
381 
382  if (buffer_used >= sizeof(uint32_t))
383  frame_len = ntohl(*(uint32_t *)buffer);
384  }
385 }
386 
387 bool
388 EtherTapStub::sendReal(const void *data, size_t len)
389 {
390  uint32_t frame_len = htonl(len);
391  ssize_t ret = write(socket, &frame_len, sizeof(frame_len));
392  if (ret != sizeof(frame_len))
393  return false;
394  return write(socket, data, len) == len;
395 }
396 
397 
398 #if USE_TUNTAP
399 
400 EtherTap::EtherTap(const Params &p) : EtherTapBase(p)
401 {
402  int fd = open(p.tun_clone_device.c_str(), O_RDWR | O_NONBLOCK);
403  if (fd < 0)
404  panic("Couldn't open %s.\n", p.tun_clone_device);
405 
406  struct ifreq ifr;
407  memset(&ifr, 0, sizeof(ifr));
408  ifr.ifr_flags = IFF_TAP | IFF_NO_PI;
409  strncpy(ifr.ifr_name, p.tap_device_name.c_str(), IFNAMSIZ - 1);
410 
411  if (ioctl(fd, TUNSETIFF, (void *)&ifr) < 0)
412  panic("Failed to access tap device %s.\n", ifr.ifr_name);
413  // fd now refers to the tap device.
414  tap = fd;
415  pollFd(tap);
416 }
417 
418 EtherTap::~EtherTap()
419 {
420  stopPolling();
421  close(tap);
422  tap = -1;
423 }
424 
425 void
426 EtherTap::recvReal(int revent)
427 {
428  if (revent & POLLERR)
429  panic("Error polling for tap data.\n");
430 
431  if (!(revent & POLLIN))
432  return;
433 
434  ssize_t ret;
435  while ((ret = read(tap, buffer, buflen))) {
436  if (ret < 0) {
437  if (errno == EAGAIN)
438  break;
439  panic("Failed to read from tap device.\n");
440  }
441 
442  sendSimulated(buffer, ret);
443  }
444 }
445 
446 bool
447 EtherTap::sendReal(const void *data, size_t len)
448 {
449  int n;
450  pollfd pfd[1];
451  pfd->fd = tap;
452  pfd->events = POLLOUT;
453 
454  // `tap` is a nonblock fd. Here we try to write until success, and use
455  // poll to make a blocking wait.
456  while ((n = write(tap, data, len)) != len) {
457  if (errno != EAGAIN)
458  panic("Failed to write data to tap device.\n");
459  pfd->revents = 0;
460  int ret = poll(pfd, 1, -1);
461  // timeout is set to inf, we shouldn't get 0 in any case.
462  assert(ret != 0);
463  if (ret == -1 || (ret == 1 && (pfd->revents & POLLERR))) {
464  panic("Failed when polling to write data to tap device.\n");
465  }
466  }
467  return true;
468 }
469 
470 #endif
fatal
#define fatal(...)
This implements a cprintf based fatal() function.
Definition: logging.hh:183
Event::scheduled
bool scheduled() const
Determine if the current event is scheduled.
Definition: eventq.hh:462
socket.hh
EtherTapBase::EtherTapBase
EtherTapBase(const Params &p)
Definition: ethertap.cc:90
SimClock::Int::ns
Tick ns
nanosecond
Definition: core.cc:62
ListenSocket
Definition: socket.hh:32
TapListener::listener
ListenSocket listener
Definition: ethertap.cc:243
EtherTapStub::recvReal
void recvReal(int revent) override
Definition: ethertap.cc:342
ListenSocket::listen
virtual bool listen(int port, bool reuse=true)
Definition: socket.cc:97
TapListener::TapListener
TapListener(EtherTapStub *t, int p)
Definition: ethertap.cc:248
data
const char data[]
Definition: circlebuf.test.cc:47
EtherTapBase::sendReal
virtual bool sendReal(const void *data, size_t len)=0
UNSERIALIZE_SCALAR
#define UNSERIALIZE_SCALAR(scalar)
Definition: serialize.hh:591
TapEvent::TapEvent
TapEvent(EtherTapBase *_tap, int fd, int e)
Definition: ethertap.cc:76
EtherTapBase::stopPolling
void stopPolling()
Definition: ethertap.cc:150
EtherTapBase::retransmit
void retransmit()
Definition: ethertap.cc:204
TapListener::port
int port
Definition: ethertap.cc:245
TapListener::Event::listener
TapListener * listener
Definition: ethertap.cc:229
TapListener::accept
void accept()
Definition: ethertap.cc:268
ArmISA::fd
Bitfield< 14, 12 > fd
Definition: types.hh:159
etherint.hh
etherdump.hh
ListenSocket::accept
virtual int accept(bool nodelay=false)
Definition: socket.cc:146
EtherTapBase::sendSimulated
void sendSimulated(void *data, size_t len)
Definition: ethertap.cc:183
PortID
int16_t PortID
Port index/ID type, and a symbolic name for an invalid port id.
Definition: types.hh:243
EtherTapStub
Definition: ethertap.hh:130
EtherTapBase::recvReal
virtual void recvReal(int revent)=0
EtherTapBase::buflen
int buflen
Definition: ethertap.hh:68
PollQueue::schedule
void schedule(PollEvent *event)
Definition: pollevent.cc:157
TapEvent
Definition: ethertap.cc:70
TapListener::listen
void listen()
Definition: ethertap.cc:255
EtherTapStub::unserialize
void unserialize(CheckpointIn &cp) override
Unserialize an object.
Definition: ethertap.cc:309
EtherTapBase::interface
EtherTapInt * interface
Definition: ethertap.hh:92
EtherTapBase::Params
EtherTapBaseParams Params
Definition: ethertap.hh:59
EtherTapStub::buffer_used
uint32_t buffer_used
Definition: ethertap.hh:150
EtherTapBase::TapEvent
friend class TapEvent
Definition: ethertap.hh:77
ArmISA::n
Bitfield< 31 > n
Definition: miscregs_types.hh:450
pollQueue
PollQueue pollQueue
Definition: pollevent.cc:53
EtherTapStub::socket
int socket
Definition: ethertap.hh:145
cp
Definition: cprintf.cc:37
EventManager::schedule
void schedule(Event &event, Tick when)
Definition: eventq.hh:1016
EtherTapStub::listener
TapListener * listener
Definition: ethertap.hh:143
SimObject::getPort
virtual Port & getPort(const std::string &if_name, PortID idx=InvalidPortID)
Get a port with a given name and index.
Definition: sim_object.cc:120
DPRINTF
#define DPRINTF(x,...)
Definition: trace.hh:237
EtherTapInt
Definition: ethertap.hh:107
ListenSocket::getfd
int getfd() const
Definition: socket.hh:72
MipsISA::event
Bitfield< 10, 5 > event
Definition: pra_constants.hh:297
pollevent.hh
Port
Ports are used to interface objects to each other.
Definition: port.hh:56
ListenSocket::islistening
bool islistening() const
Definition: socket.hh:73
EtherTapBase::unserialize
void unserialize(CheckpointIn &cp) override
Unserialize an object.
Definition: ethertap.cc:124
TapListener::tap
EtherTapStub * tap
Definition: ethertap.cc:244
SERIALIZE_ARRAY
#define SERIALIZE_ARRAY(member, size)
Definition: serialize.hh:626
EtherTapBase::serialize
void serialize(CheckpointOut &cp) const override
Serialize an object.
Definition: ethertap.cc:107
EventManager::eventQueue
EventQueue * eventQueue() const
Definition: eventq.hh:1007
name
const std::string & name()
Definition: trace.cc:48
SERIALIZE_SCALAR
#define SERIALIZE_SCALAR(scalar)
Definition: serialize.hh:584
DDUMP
#define DDUMP(x, data, count)
DPRINTF is a debugging trace facility that allows one to selectively enable tracing statements.
Definition: trace.hh:236
ethertap.hh
EtherTapBase::event
TapEvent * event
Definition: ethertap.hh:78
EtherTapBase::getPort
Port & getPort(const std::string &if_name, PortID idx=InvalidPortID) override
Get a port with a given name and index.
Definition: ethertap.cc:159
PollEvent::queued
bool queued()
Definition: pollevent.hh:70
TapListener
Definition: ethertap.cc:223
EtherTapStub::attach
void attach(int fd)
Definition: ethertap.cc:320
ArmISA::e
Bitfield< 9 > e
Definition: miscregs_types.hh:61
TapListener::Event::process
void process(int revent) override
Definition: ethertap.cc:234
EthPacketPtr
std::shared_ptr< EthPacketData > EthPacketPtr
Definition: etherpkt.hh:87
ListenSocket::allDisabled
static bool allDisabled()
Definition: socket.cc:68
EventQueue::ScopedMigration
Definition: eventq.hh:669
TapEvent::process
void process(int revent) override
Definition: ethertap.cc:80
EtherInt::recvDone
void recvDone()
Definition: etherint.hh:67
UNSERIALIZE_ARRAY
#define UNSERIALIZE_ARRAY(member, size)
Definition: serialize.hh:634
ArmISA::t
Bitfield< 5 > t
Definition: miscregs_types.hh:67
TapListener::Event
friend class Event
Definition: ethertap.cc:237
Stats::dump
void dump()
Dump all statistics data to the registered outputs.
Definition: statistics.cc:290
ArmISA::len
Bitfield< 18, 16 > len
Definition: miscregs_types.hh:439
ccprintf
void ccprintf(cp::Print &print)
Definition: cprintf.hh:127
EtherTapStub::frame_len
uint32_t frame_len
Definition: ethertap.hh:151
logging.hh
etherpkt.hh
TapEvent::tap
EtherTapBase * tap
Definition: ethertap.cc:73
CheckpointOut
std::ostream CheckpointOut
Definition: serialize.hh:64
EtherInt::sendPacket
bool sendPacket(EthPacketPtr packet)
Definition: etherint.hh:70
curTick
Tick curTick()
The universal simulation clock.
Definition: cur_tick.hh:43
EtherTapBase::~EtherTapBase
virtual ~EtherTapBase()
Definition: ethertap.cc:99
EtherTapBase::pollFd
void pollFd(int fd)
Definition: ethertap.cc:142
trace.hh
EtherTapBase::buffer
uint8_t * buffer
Definition: ethertap.hh:67
EtherTapStub::detach
void detach()
Definition: ethertap.cc:333
PollEvent
Definition: pollevent.hh:41
MipsISA::p
Bitfield< 0 > p
Definition: pra_constants.hh:323
TapListener::event
Event * event
Definition: ethertap.cc:238
TapListener::Event::Event
Event(TapListener *l, int fd, int e)
Definition: ethertap.cc:232
CheckpointIn
Definition: serialize.hh:68
MipsISA::l
Bitfield< 5 > l
Definition: pra_constants.hh:320
EtherTapStub::EtherTapStub
EtherTapStub(const Params &p)
Definition: ethertap.cc:284
EtherTapBase::txEvent
EventFunctionWrapper txEvent
Definition: ethertap.hh:104
EtherTapStub::~EtherTapStub
~EtherTapStub()
Definition: ethertap.cc:293
EtherTapStub::TapListener
friend class TapListener
Definition: ethertap.hh:142
TapListener::Event
Definition: ethertap.cc:226
EtherTapBase::packetBuffer
std::queue< EthPacketPtr > packetBuffer
Definition: ethertap.hh:102
EtherTapBase::recvSimulated
bool recvSimulated(EthPacketPtr packet)
Definition: ethertap.cc:167
EtherTapStub::serialize
void serialize(CheckpointOut &cp) const override
Serialize an object.
Definition: ethertap.cc:299
EtherTapBase
Definition: ethertap.hh:56
TapListener::~TapListener
~TapListener()
Definition: ethertap.cc:249
EtherDump::dump
void dump(EthPacketPtr &pkt)
Definition: etherdump.hh:57
EtherTapBase::dump
EtherDump * dump
Definition: ethertap.hh:70
panic
#define panic(...)
This implements a cprintf based panic() function.
Definition: logging.hh:171
EtherTapStub::sendReal
bool sendReal(const void *data, size_t len) override
Definition: ethertap.cc:388
SimObject
Abstract superclass for simulation objects.
Definition: sim_object.hh:141

Generated on Tue Mar 23 2021 19:41:26 for gem5 by doxygen 1.8.17