Go to the documentation of this file.
37 #include "debug/EthernetAll.hh"
43 for (
int i = 0;
i <
p.port_interface_connection_count; ++
i) {
44 std::string interfaceName =
csprintf(
"%s.interface%d",
name(),
i);
46 p.output_buffer_size,
p.delay,
47 p.delay_var,
p.fabric_speed,
i);
63 if (if_name ==
"interface") {
81 DPRINTF(Ethernet,
"Fifo is full. Drop packet: len=%d\n",
82 std::prev(
fifo.end())->packet->length);
84 _size -= std::prev(
fifo.end())->packet->length;
89 warn(
"EtherSwitch: Packet length (%d) exceeds the maximum storage "
90 "capacity of port fifo (%d)", ptr->length,
_maxsize);
97 if (!
empty() &&
fifo.begin()->packet == ptr) {
109 assert(_size >= fifo.begin()->packet->length);
111 _size -= fifo.begin()->packet->length;
112 fifo.erase(fifo.begin());
124 uint64_t outputBufferSize,
Tick delay,
125 Tick delay_var,
double rate,
unsigned id)
139 learnSenderAddr(srcMacAddr,
this);
140 Interface *receiver = lookupDestPort(destMacAddr);
143 for (
auto it : parent->interfaces)
145 it->enqueue(packet, interfaceId);
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());
151 receiver->
enqueue(packet, interfaceId);
171 if (outputFifo.push(packet, senderId)) {
172 parent->reschedule(txEvent,
curTick() + switchingDelay(),
true);
180 assert(!outputFifo.empty());
182 if (!sendPacket(outputFifo.front())) {
183 DPRINTF(Ethernet,
"output port busy...retry later\n");
184 if (!txEvent.scheduled())
187 DPRINTF(Ethernet,
"packet sent: len=%d\n", outputFifo.front()->length);
191 if (!outputFifo.empty()) {
192 parent->schedule(txEvent,
curTick() + switchingDelay());
200 Tick delay = (
Tick)ceil(((
double)outputFifo.front()->simLength
201 * ticksPerByte) + 1.0);
204 delay += switchDelay;
211 auto it = parent->forwardingTable.find(uint64_t(destMacAddr));
213 if (it == parent->forwardingTable.end()) {
214 DPRINTF(Ethernet,
"no entry in forwaring table for MAC: "
215 "%x\n", uint64_t(destMacAddr));
220 if ((
curTick() - it->second.lastUseTime) > parent->ttl) {
223 parent->forwardingTable.erase(it);
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;
237 auto it = parent->forwardingTable.find(uint64_t(srcMacAddr));
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),
248 parent->forwardingTable.insert(std::make_pair(uint64_t(srcMacAddr),
249 forwardingTableEntry));
251 it->second.lastUseTime =
curTick();
259 it->serializeSection(
cp, it->name());
267 it->unserializeSection(
cp, it->name());
274 bool event_scheduled = txEvent.scheduled();
277 if (event_scheduled) {
278 Tick event_time = txEvent.when();
281 outputFifo.serializeSection(
cp,
"outputFifo");
287 bool event_scheduled;
290 if (event_scheduled) {
293 parent->schedule(txEvent, event_time);
295 outputFifo.unserializeSection(
cp,
"outputFifo");
301 packet->serialize(
"packet",
cp);
309 packet = std::make_shared<EthPacketData>(16384);
310 packet->unserialize(
"packet",
cp);
319 int fifosize = fifo.size();
324 for (
const auto &entry : fifo)
325 entry.serializeSection(
cp,
csprintf(
"entry%d",
i++));
337 for (
int i = 0;
i < fifosize; ++
i) {
void unserializeSection(CheckpointIn &cp, const char *name)
Unserialize an a child object.
void serialize(CheckpointOut &cp) const
Serialize an object.
std::set< PortFifoEntry, EntryOrder > fifo
Port & getPort(const std::string &if_name, PortID idx=InvalidPortID) override
Get a port with a given name and index.
#define UNSERIALIZE_SCALAR(scalar)
std::vector< Interface * > interfaces
void serialize(CheckpointOut &cp) const
Serialize an object.
const double ticksPerByte
void serialize(CheckpointOut &cp) const
Serialization stuff.
uint64_t Tick
Tick count type.
Model for an Ethernet switch port.
int16_t PortID
Port index/ID type, and a symbolic name for an invalid port id.
bool push(EthPacketPtr ptr, unsigned senderId)
Push a packet into the fifo and sort the packets with same recv tick by port id.
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...
void unserialize(CheckpointIn &cp)
Unserialize an object.
void unserialize(CheckpointIn &cp) override
Unserialize an object.
EtherSwitch(const Params &p)
void learnSenderAddr(Net::EthAddr srcMacAddr, Interface *sender)
virtual Port & getPort(const std::string &if_name, PortID idx=InvalidPortID)
Get a port with a given name and index.
Ports are used to interface objects to each other.
void enqueue(EthPacketPtr packet, unsigned senderId)
enqueue packet to the outputFifo
const PortID id
A numeric identifier to distinguish ports in a vector, and set to InvalidPortID in case this port is ...
#define SERIALIZE_SCALAR(scalar)
PortFifo outputFifo
output fifo at each interface
virtual const std::string name() const
std::shared_ptr< EthPacketData > EthPacketPtr
Interface(const std::string &name, EtherSwitch *_etherSwitch, uint64_t outputBufferSize, Tick delay, Tick delay_var, double rate, unsigned id)
#define panic_if(cond,...)
Conditional panic macro that checks the supplied condition and only panics if the condition is true a...
const std::string & name() const
Return port name (for DPRINTF).
Interface * lookupDestPort(Net::EthAddr destAddr)
bool recvPacket(EthPacketPtr packet)
When a packet is received from a device, route it through an (several) output queue(s)
std::ostream CheckpointOut
EventFunctionWrapper txEvent
void serialize(CheckpointOut &cp) const override
Serialize an object.
Tick curTick()
The universal simulation clock.
void unserialize(CheckpointIn &cp)
Unserialize an object.
const unsigned interfaceId
void unserialize(CheckpointIn &cp)
Unserialize an object.
std::string csprintf(const char *format, const Args &...args)
Abstract superclass for simulation objects.
Generated on Tue Jun 22 2021 15:28:28 for gem5 by doxygen 1.8.17