Go to the documentation of this file.
35 #if defined(__OpenBSD__) || defined(__APPLE__)
36 #include <sys/param.h>
40 #if HAVE_TUNTAP && defined(__linux__)
41 #if 1 // Hide from the style checker since these have to be out of order.
42 #include <sys/socket.h>
47 #include <linux/if_tun.h>
52 #include <netinet/in.h>
53 #include <sys/ioctl.h>
64 #include "debug/Ethernet.hh"
65 #include "debug/EthernetData.hh"
98 txEvent([this]{
retransmit(); },
"EtherTapBase retransmit")
100 buffer =
new uint8_t[buflen];
115 uint8_t *
buffer = (uint8_t *)this->buffer;
118 bool tapevent_present =
false;
120 tapevent_present =
true;
122 event->serialize(cp);
132 uint8_t *
buffer = (uint8_t *)this->buffer;
135 bool tapevent_present;
137 if (tapevent_present) {
139 event->unserialize(cp);
150 event =
new TapEvent(
this,
fd, POLLIN|POLLERR);
166 if (if_name ==
"tap")
177 DPRINTF(Ethernet,
"EtherTap sim->real len=%d\n", packet->length);
178 DDUMP(EthernetData, packet->data, packet->length);
180 bool success =
sendReal(packet->data, packet->length);
191 packet = std::make_shared<EthPacketData>(
len);
192 packet->length =
len;
193 packet->simLength =
len;
194 memcpy(packet->data,
data,
len);
196 DPRINTF(Ethernet,
"EtherTap real->sim len=%d\n", packet->length);
197 DDUMP(EthernetData, packet->data, packet->length);
199 DPRINTF(Ethernet,
"bus busy...buffer for retransmission\n");
218 DPRINTF(Ethernet,
"EtherTap retransmit\n");
263 DPRINTF(Ethernet,
"TapListener(listen): Can't bind port %d\n",
port);
267 ccprintf(std::cerr,
"Listening for tap connection on port %d\n",
port);
281 panic(
"TapListener(accept): cannot accept if we're not listening!");
292 fatal(
"All listeners are disabled! EtherTapStub can't work!");
333 DPRINTF(Ethernet,
"EtherTapStub attached\n");
340 DPRINTF(Ethernet,
"EtherTapStub detached\n");
349 if (revent & POLLERR) {
354 if (!(revent & POLLIN))
372 DPRINTF(Ethernet,
"Received data from peer: len=%d buffer_used=%d "
375 uint8_t *frame_start = &
buffer[
sizeof(uint32_t)];
407 int fd = open(
p.tun_clone_device.c_str(), O_RDWR | O_NONBLOCK);
409 panic(
"Couldn't open %s.\n",
p.tun_clone_device);
412 memset(&ifr, 0,
sizeof(ifr));
413 ifr.ifr_flags = IFF_TAP | IFF_NO_PI;
414 strncpy(ifr.ifr_name,
p.tap_device_name.c_str(), IFNAMSIZ - 1);
416 if (ioctl(
fd, TUNSETIFF, (
void *)&ifr) < 0)
417 panic(
"Failed to access tap device %s.\n", ifr.ifr_name);
423 EtherTap::~EtherTap()
431 EtherTap::recvReal(
int revent)
433 if (revent & POLLERR)
434 panic(
"Error polling for tap data.\n");
436 if (!(revent & POLLIN))
440 while ((ret = read(tap, buffer, buflen))) {
444 panic(
"Failed to read from tap device.\n");
447 sendSimulated(buffer, ret);
452 EtherTap::sendReal(
const void *
data,
size_t len)
457 pfd->events = POLLOUT;
463 panic(
"Failed to write data to tap device.\n");
465 int ret = poll(pfd, 1, -1);
468 if (ret == -1 || (ret == 1 && (pfd->revents & POLLERR))) {
469 panic(
"Failed when polling to write data to tap device.\n");
475 #endif // HAVE_TUNTAP
Tick curTick()
The universal simulation clock.
#define fatal(...)
This implements a cprintf based fatal() function.
int16_t PortID
Port index/ID type, and a symbolic name for an invalid port id.
virtual Port & getPort(const std::string &if_name, PortID idx=InvalidPortID)
Get a port with a given name and index.
Port & getPort(const std::string &if_name, PortID idx=InvalidPortID) override
Get a port with a given name and index.
bool sendReal(const void *data, size_t len) override
EventQueue * eventQueue() const
#define UNSERIALIZE_SCALAR(scalar)
EtherTapBase(const Params &p)
std::queue< EthPacketPtr > packetBuffer
bool sendPacket(EthPacketPtr packet)
TapListener(EtherTapStub *t, int p)
void schedule(Event &event, Tick when)
void sendSimulated(void *data, size_t len)
TapEvent(EtherTapBase *_tap, int fd, int e)
void ccprintf(cp::Print &print)
virtual int accept(bool nodelay=false)
void dump(EthPacketPtr &pkt)
EtherTapStub(const Params &p)
EventFunctionWrapper txEvent
std::shared_ptr< EthPacketData > EthPacketPtr
void unserialize(CheckpointIn &cp) override
Unserialize an object.
void dump()
Dump all statistics data to the registered outputs.
void process(int revent) override
virtual bool listen(int port, bool reuse=true)
void schedule(PollEvent *event)
#define SERIALIZE_ARRAY(member, size)
Abstract superclass for simulation objects.
static bool allDisabled()
const std::string & name()
#define SERIALIZE_SCALAR(scalar)
#define DDUMP(x, data, count)
DPRINTF is a debugging trace facility that allows one to selectively enable tracing statements.
void recvReal(int revent) override
void unserialize(CheckpointIn &cp) override
Unserialize an object.
Ports are used to interface objects to each other.
#define UNSERIALIZE_ARRAY(member, size)
void serialize(CheckpointOut &cp) const override
Serialize an object.
virtual void recvReal(int revent)=0
Event(TapListener *l, int fd, int e)
std::ostream CheckpointOut
void serialize(CheckpointOut &cp) const override
Serialize an object.
void process(int revent) override
Reference material can be found at the JEDEC website: UFS standard http://www.jedec....
bool recvSimulated(EthPacketPtr packet)
virtual bool sendReal(const void *data, size_t len)=0
bool scheduled() const
Determine if the current event is scheduled.
#define panic(...)
This implements a cprintf based panic() function.
EtherTapBaseParams Params
Generated on Wed Jul 13 2022 10:39:20 for gem5 by doxygen 1.8.17