gem5  v21.0.1.0
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
etherswitch.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2014 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  * Device model for an ethernet switch
31  */
32 
33 #include "dev/net/etherswitch.hh"
34 
35 #include "base/random.hh"
36 #include "base/trace.hh"
37 #include "debug/EthernetAll.hh"
38 #include "sim/core.hh"
39 
41  : SimObject(p), ttl(p.time_to_live)
42 {
43  for (int i = 0; i < p.port_interface_connection_count; ++i) {
44  std::string interfaceName = csprintf("%s.interface%d", name(), i);
45  Interface *interface = new Interface(interfaceName, this,
46  p.output_buffer_size, p.delay,
47  p.delay_var, p.fabric_speed, i);
48  interfaces.push_back(interface);
49  }
50 }
51 
53 {
54  for (auto it : interfaces)
55  delete it;
56 
57  interfaces.clear();
58 }
59 
60 Port &
61 EtherSwitch::getPort(const std::string &if_name, PortID idx)
62 {
63  if (if_name == "interface") {
64  panic_if(idx < 0 || idx >= interfaces.size(), "index out of bounds");
65  return *interfaces.at(idx);
66  }
67 
68  return SimObject::getPort(if_name, idx);
69 }
70 
71 bool
73 {
74  assert(ptr->length);
75 
76  _size += ptr->length;
77  fifo.emplace_hint(fifo.end(), ptr, curTick(), senderId);
78 
79  // Drop the extra pushed packets from end of the fifo
80  while (avail() < 0) {
81  DPRINTF(Ethernet, "Fifo is full. Drop packet: len=%d\n",
82  std::prev(fifo.end())->packet->length);
83 
84  _size -= std::prev(fifo.end())->packet->length;
85  fifo.erase(std::prev(fifo.end()));
86  }
87 
88  if (empty()) {
89  warn("EtherSwitch: Packet length (%d) exceeds the maximum storage "
90  "capacity of port fifo (%d)", ptr->length, _maxsize);
91  }
92 
93  // Return true if the newly pushed packet gets inserted
94  // at the head of the queue, otherwise return false
95  // We need this information to deschedule the event that has been
96  // scheduled for the old head of queue packet and schedule a new one
97  if (!empty() && fifo.begin()->packet == ptr) {
98  return true;
99  }
100  return false;
101 }
102 
103 void
105 {
106  if (empty())
107  return;
108 
109  assert(_size >= fifo.begin()->packet->length);
110  // Erase the packet at the head of the queue
111  _size -= fifo.begin()->packet->length;
112  fifo.erase(fifo.begin());
113 }
114 
115 void
117 {
118  fifo.clear();
119  _size = 0;
120 }
121 
123  EtherSwitch *etherSwitch,
124  uint64_t outputBufferSize, Tick delay,
125  Tick delay_var, double rate, unsigned id)
126  : EtherInt(name), ticksPerByte(rate), switchDelay(delay),
127  delayVar(delay_var), interfaceId(id), parent(etherSwitch),
128  outputFifo(name + ".outputFifo", outputBufferSize),
129  txEvent([this]{ transmit(); }, name)
130 {
131 }
132 
133 bool
135 {
136  Net::EthAddr destMacAddr(packet->data);
137  Net::EthAddr srcMacAddr(&packet->data[6]);
138 
139  learnSenderAddr(srcMacAddr, this);
140  Interface *receiver = lookupDestPort(destMacAddr);
141 
142  if (!receiver || destMacAddr.multicast() || destMacAddr.broadcast()) {
143  for (auto it : parent->interfaces)
144  if (it != this)
145  it->enqueue(packet, interfaceId);
146  } else {
147  DPRINTF(Ethernet, "sending packet from MAC %x on port "
148  "%s to MAC %x on port %s\n", uint64_t(srcMacAddr),
149  this->name(), uint64_t(destMacAddr), receiver->name());
150 
151  receiver->enqueue(packet, interfaceId);
152  }
153  // At the output port, we either have buffer space (no drop) or
154  // don't (drop packet); in both cases packet is received on
155  // the interface successfully and there is no notion of busy
156  // interface here (as we don't have inputFifo)
157  return true;
158 }
159 
160 void
162 {
163  // assuming per-interface transmission events,
164  // if the newly push packet gets inserted at the head of the queue
165  // (either there was nothing in the queue or the priority of the new
166  // packet was higher than the packets already in the fifo)
167  // then we need to schedule an event at
168  // "curTick" + "switchingDelay of the packet at the head of the fifo"
169  // to send this packet out the external link
170  // otherwise, there is already a txEvent scheduled
171  if (outputFifo.push(packet, senderId)) {
172  parent->reschedule(txEvent, curTick() + switchingDelay(), true);
173  }
174 }
175 
176 void
178 {
179  // there should be something in the output queue
180  assert(!outputFifo.empty());
181 
182  if (!sendPacket(outputFifo.front())) {
183  DPRINTF(Ethernet, "output port busy...retry later\n");
184  if (!txEvent.scheduled())
185  parent->schedule(txEvent, curTick() + SimClock::Int::ns);
186  } else {
187  DPRINTF(Ethernet, "packet sent: len=%d\n", outputFifo.front()->length);
188  outputFifo.pop();
189  // schedule an event to send the pkt at
190  // the head of queue, if there is any
191  if (!outputFifo.empty()) {
192  parent->schedule(txEvent, curTick() + switchingDelay());
193  }
194  }
195 }
196 
197 Tick
199 {
200  Tick delay = (Tick)ceil(((double)outputFifo.front()->simLength
201  * ticksPerByte) + 1.0);
202  if (delayVar != 0)
203  delay += random_mt.random<Tick>(0, delayVar);
204  delay += switchDelay;
205  return delay;
206 }
207 
210 {
211  auto it = parent->forwardingTable.find(uint64_t(destMacAddr));
212 
213  if (it == parent->forwardingTable.end()) {
214  DPRINTF(Ethernet, "no entry in forwaring table for MAC: "
215  "%x\n", uint64_t(destMacAddr));
216  return nullptr;
217  }
218 
219  // check if this entry is valid based on TTL and lastUseTime
220  if ((curTick() - it->second.lastUseTime) > parent->ttl) {
221  // TTL for this mapping has been expired, so this item is not
222  // valide anymore, let's remove it from the map
223  parent->forwardingTable.erase(it);
224  return nullptr;
225  }
226 
227  DPRINTF(Ethernet, "found entry for MAC address %x on port %s\n",
228  uint64_t(destMacAddr), it->second.interface->name());
229  return it->second.interface;
230 }
231 
232 void
234  Interface *sender)
235 {
236  // learn the port for the sending MAC address
237  auto it = parent->forwardingTable.find(uint64_t(srcMacAddr));
238 
239  // if the port for sender's MAC address is not cached,
240  // cache it now, otherwise just update lastUseTime time
241  if (it == parent->forwardingTable.end()) {
242  DPRINTF(Ethernet, "adding forwarding table entry for MAC "
243  " address %x on port %s\n", uint64_t(srcMacAddr),
244  sender->name());
245  EtherSwitch::SwitchTableEntry forwardingTableEntry;
246  forwardingTableEntry.interface = sender;
247  forwardingTableEntry.lastUseTime = curTick();
248  parent->forwardingTable.insert(std::make_pair(uint64_t(srcMacAddr),
249  forwardingTableEntry));
250  } else {
251  it->second.lastUseTime = curTick();
252  }
253 }
254 
255 void
257 {
258  for (auto it : interfaces)
259  it->serializeSection(cp, it->name());
260 
261 }
262 
263 void
265 {
266  for (auto it : interfaces)
267  it->unserializeSection(cp, it->name());
268 
269 }
270 
271 void
273 {
274  bool event_scheduled = txEvent.scheduled();
275  SERIALIZE_SCALAR(event_scheduled);
276 
277  if (event_scheduled) {
278  Tick event_time = txEvent.when();
279  SERIALIZE_SCALAR(event_time);
280  }
281  outputFifo.serializeSection(cp, "outputFifo");
282 }
283 
284 void
286 {
287  bool event_scheduled;
288  UNSERIALIZE_SCALAR(event_scheduled);
289 
290  if (event_scheduled) {
291  Tick event_time;
292  UNSERIALIZE_SCALAR(event_time);
293  parent->schedule(txEvent, event_time);
294  }
295  outputFifo.unserializeSection(cp, "outputFifo");
296 }
297 
298 void
300 {
301  packet->serialize("packet", cp);
302  SERIALIZE_SCALAR(recvTick);
303  SERIALIZE_SCALAR(srcId);
304 }
305 
306 void
308 {
309  packet = std::make_shared<EthPacketData>(16384);
310  packet->unserialize("packet", cp);
311  UNSERIALIZE_SCALAR(recvTick);
312  UNSERIALIZE_SCALAR(srcId);
313 }
314 
315 void
317 {
318  SERIALIZE_SCALAR(_size);
319  int fifosize = fifo.size();
320 
321  SERIALIZE_SCALAR(fifosize);
322 
323  int i = 0;
324  for (const auto &entry : fifo)
325  entry.serializeSection(cp, csprintf("entry%d", i++));
326 }
327 
328 void
330 {
331  UNSERIALIZE_SCALAR(_size);
332  int fifosize;
333 
334  UNSERIALIZE_SCALAR(fifosize);
335  fifo.clear();
336 
337  for (int i = 0; i < fifosize; ++i) {
338  PortFifoEntry entry(nullptr, 0, 0);
339 
340  entry.unserializeSection(cp, csprintf("entry%d", i));
341 
342  fifo.insert(entry);
343 
344  }
345 }
EtherSwitch::Interface::PortFifo::_size
unsigned _size
Definition: etherswitch.hh:134
Serializable::unserializeSection
void unserializeSection(CheckpointIn &cp, const char *name)
Unserialize an a child object.
Definition: serialize.cc:177
SimClock::Int::ns
Tick ns
nanosecond
Definition: core.cc:62
warn
#define warn(...)
Definition: logging.hh:239
EtherSwitch::Interface::PortFifoEntry::serialize
void serialize(CheckpointOut &cp) const
Serialize an object.
Definition: etherswitch.cc:299
EtherSwitch::Interface::PortFifo::fifo
std::set< PortFifoEntry, EntryOrder > fifo
Definition: etherswitch.hh:130
EtherSwitch::getPort
Port & getPort(const std::string &if_name, PortID idx=InvalidPortID) override
Get a port with a given name and index.
Definition: etherswitch.cc:61
UNSERIALIZE_SCALAR
#define UNSERIALIZE_SCALAR(scalar)
Definition: serialize.hh:591
EtherSwitch::interfaces
std::vector< Interface * > interfaces
Definition: etherswitch.hh:183
EtherSwitch::Interface::serialize
void serialize(CheckpointOut &cp) const
Serialize an object.
Definition: etherswitch.cc:272
etherswitch.hh
EtherSwitch::Interface::parent
EtherSwitch * parent
Definition: etherswitch.hh:96
EtherSwitch::SwitchTableEntry
Definition: etherswitch.hh:174
ArmISA::i
Bitfield< 7 > i
Definition: miscregs_types.hh:63
EtherSwitch::Interface::ticksPerByte
const double ticksPerByte
Definition: etherswitch.hh:91
EtherInt
Definition: etherint.hh:47
EtherSwitch::Params
EtherSwitchParams Params
Definition: etherswitch.hh:54
EtherSwitch
Definition: etherswitch.hh:51
EtherSwitch::Interface::PortFifo::serialize
void serialize(CheckpointOut &cp) const
Serialization stuff.
Definition: etherswitch.cc:316
Net::EthAddr
Definition: inet.hh:72
EtherSwitch::Interface::PortFifo::_maxsize
const unsigned _maxsize
Definition: etherswitch.hh:133
random.hh
Tick
uint64_t Tick
Tick count type.
Definition: types.hh:59
EtherSwitch::Interface
Model for an Ethernet switch port.
Definition: etherswitch.hh:66
PortID
int16_t PortID
Port index/ID type, and a symbolic name for an invalid port id.
Definition: types.hh:243
EtherSwitch::Interface::PortFifo::push
bool push(EthPacketPtr ptr, unsigned senderId)
Push a packet into the fifo and sort the packets with same recv tick by port id.
Definition: etherswitch.cc:72
EtherSwitch::Interface::transmit
void transmit()
Definition: etherswitch.cc:177
Random::random
std::enable_if_t< std::is_integral< T >::value, T > random()
Use the SFINAE idiom to choose an implementation based on whether the type is integral or floating po...
Definition: random.hh:86
EtherSwitch::SwitchTableEntry::lastUseTime
Tick lastUseTime
Definition: etherswitch.hh:176
EtherSwitch::Interface::PortFifoEntry::unserialize
void unserialize(CheckpointIn &cp)
Unserialize an object.
Definition: etherswitch.cc:307
random_mt
Random random_mt
Definition: random.cc:96
EtherSwitch::unserialize
void unserialize(CheckpointIn &cp) override
Unserialize an object.
Definition: etherswitch.cc:264
cp
Definition: cprintf.cc:37
Net::EthAddr::broadcast
bool broadcast() const
Definition: inet.hh:111
EtherSwitch::SwitchTableEntry::interface
Interface * interface
Definition: etherswitch.hh:175
EtherSwitch::EtherSwitch
EtherSwitch(const Params &p)
Definition: etherswitch.cc:40
EtherSwitch::Interface::learnSenderAddr
void learnSenderAddr(Net::EthAddr srcMacAddr, Interface *sender)
Definition: etherswitch.cc:233
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
EtherSwitch::Interface::PortFifo::pop
void pop()
Definition: etherswitch.cc:104
DPRINTF
#define DPRINTF(x,...)
Definition: trace.hh:237
EtherSwitch::Interface::PortFifo::empty
bool empty() const
Definition: etherswitch.hh:150
Port
Ports are used to interface objects to each other.
Definition: port.hh:56
EtherSwitch::Interface::enqueue
void enqueue(EthPacketPtr packet, unsigned senderId)
enqueue packet to the outputFifo
Definition: etherswitch.cc:161
EtherSwitch::Interface::PortFifo::clear
void clear()
Definition: etherswitch.cc:116
Port::id
const PortID id
A numeric identifier to distinguish ports in a vector, and set to InvalidPortID in case this port is ...
Definition: port.hh:74
core.hh
SERIALIZE_SCALAR
#define SERIALIZE_SCALAR(scalar)
Definition: serialize.hh:584
EtherSwitch::Interface::outputFifo
PortFifo outputFifo
output fifo at each interface
Definition: etherswitch.hh:169
SimObject::name
virtual const std::string name() const
Definition: sim_object.hh:182
EthPacketPtr
std::shared_ptr< EthPacketData > EthPacketPtr
Definition: etherpkt.hh:87
EtherSwitch::Interface::Interface
Interface(const std::string &name, EtherSwitch *_etherSwitch, uint64_t outputBufferSize, Tick delay, Tick delay_var, double rate, unsigned id)
Definition: etherswitch.cc:122
panic_if
#define panic_if(cond,...)
Conditional panic macro that checks the supplied condition and only panics if the condition is true a...
Definition: logging.hh:197
Net::EthAddr::multicast
bool multicast() const
Definition: inet.hh:110
EtherInt::name
const std::string & name() const
Return port name (for DPRINTF).
Definition: etherint.hh:59
EtherSwitch::~EtherSwitch
~EtherSwitch()
Definition: etherswitch.cc:52
EtherSwitch::Interface::PortFifo::avail
int avail() const
Definition: etherswitch.hh:147
EtherSwitch::Interface::PortFifoEntry
Definition: etherswitch.hh:98
EtherSwitch::Interface::switchingDelay
Tick switchingDelay()
Definition: etherswitch.cc:198
EtherSwitch::Interface::lookupDestPort
Interface * lookupDestPort(Net::EthAddr destAddr)
Definition: etherswitch.cc:209
EtherSwitch::Interface::recvPacket
bool recvPacket(EthPacketPtr packet)
When a packet is received from a device, route it through an (several) output queue(s)
Definition: etherswitch.cc:134
EtherSwitch::Interface::delayVar
const Tick delayVar
Definition: etherswitch.hh:93
CheckpointOut
std::ostream CheckpointOut
Definition: serialize.hh:64
EtherSwitch::Interface::txEvent
EventFunctionWrapper txEvent
Definition: etherswitch.hh:171
EtherSwitch::serialize
void serialize(CheckpointOut &cp) const override
Serialize an object.
Definition: etherswitch.cc:256
curTick
Tick curTick()
The universal simulation clock.
Definition: cur_tick.hh:43
trace.hh
EtherSwitch::Interface::PortFifo::unserialize
void unserialize(CheckpointIn &cp)
Unserialize an object.
Definition: etherswitch.cc:329
EtherSwitch::Interface::interfaceId
const unsigned interfaceId
Definition: etherswitch.hh:94
MipsISA::p
Bitfield< 0 > p
Definition: pra_constants.hh:323
CheckpointIn
Definition: serialize.hh:68
EtherSwitch::Interface::unserialize
void unserialize(CheckpointIn &cp)
Unserialize an object.
Definition: etherswitch.cc:285
ArmISA::ttl
Bitfield< 51, 48 > ttl
Definition: miscregs_types.hh:148
csprintf
std::string csprintf(const char *format, const Args &...args)
Definition: cprintf.hh:158
EtherSwitch::Interface::switchDelay
const Tick switchDelay
Definition: etherswitch.hh:92
SimObject
Abstract superclass for simulation objects.
Definition: sim_object.hh:141

Generated on Tue Jun 22 2021 15:28:28 for gem5 by doxygen 1.8.17