39 #include "debug/EthernetAll.hh" 76 intrDelay(p->intr_delay), intrTick(0), cpuIntrEnable(false),
77 cpuPendingIntr(false), intrEvent(0), interface(NULL)
82 :
Base(p), rxUnique(0), txUnique(0),
83 virtualRegs(p->virtual_count < 1 ? 1 : p->virtual_count),
84 rxFifo(p->rx_fifo_size), txFifo(p->tx_fifo_size),
85 rxKickTick(0), txKickTick(0),
92 interface = new Interface(name() + ".int0", this);
109 .
desc(
"maximum vnic distance")
113 .
name(
name() +
".totalVnicDistance")
114 .
desc(
"total vnic distance")
118 .
desc(
"number of vnic distance measurements")
123 .
desc(
"average vnic distance")
140 if (if_name ==
"interface")
151 panic(
"Trying to access a vnic that doesn't exist %d > %d\n",
162 using namespace Regs;
168 uint64_t rxdone = vnic.
RxDone;
171 rxdone = set_RxDone_High(rxdone,
rxFifo.
size() >
regs.RxFifoHigh);
172 rxdone = set_RxDone_NotHigh(rxdone,
rxLow);
174 regs.RxDone = rxdone;
175 regs.RxWait = rxdone;
178 uint64_t txdone = vnic.
TxDone;
183 regs.TxDone = txdone;
184 regs.TxWait = txdone;
190 if (vnic != -1 &&
virtualRegs[vnic].rxPacketOffset > 0)
194 regs.RxStatus = set_RxStatus_Head(
regs.RxStatus, head);
221 panic(
"invalid register: cpu=%d vnic=%d da=%#x pa=%#x size=%d",
226 panic(
"read %s (write only): " 227 "cpu=%d vnic=%d da=%#x pa=%#x size=%d",
230 panic(
"read %s (invalid size): " 231 "cpu=%d vnic=%d da=%#x pa=%#x size=%d",
236 uint64_t value M5_VAR_USED = 0;
250 "read %s: cpu=%d vnic=%d da=%#x pa=%#x size=%d val=%#x\n",
251 info.name, cpu, index, daddr, pkt->
getAddr(), pkt->
getSize(), value);
255 if (raddr == Regs::IntrStatus)
306 panic(
"invalid register: cpu=%d, da=%#x pa=%#x size=%d",
311 panic(
"write %s (read only): " 312 "cpu=%d vnic=%d da=%#x pa=%#x size=%d",
315 if (pkt->
getSize() != info.size)
316 panic(
"write %s (invalid size): " 317 "cpu=%d vnic=%d da=%#x pa=%#x size=%d",
323 "write %s vnic %d: cpu=%d val=%#x da=%#x pa=%#x size=%d\n",
324 info.name, index, cpu, info.size == 4 ?
325 pkt->
getLE<uint32_t>() : pkt->
getLE<uint64_t>(),
339 case Regs::IntrStatus:
341 pkt->
getLE<uint32_t>());
349 if (Regs::get_RxDone_Busy(vnic.
RxDone))
350 panic(
"receive machine busy with another request! rxState=%s",
354 vnic.
RxDone = Regs::RxDone_Busy;
358 if (Regs::get_RxData_Vaddr(pkt->
getLE<uint64_t>())) {
359 panic(
"vtophys not implemented in newmem");
361 DPRINTF(EthernetPIO,
"write RxData vnic %d (rxunique %d)\n",
366 DPRINTF(EthernetPIO,
"request new packet...appending to rxList\n");
369 DPRINTF(EthernetPIO,
"packet exists...appending to rxBusy\n");
380 if (Regs::get_TxDone_Busy(vnic.
TxDone))
381 panic(
"transmit machine busy with another request! txState=%s",
385 vnic.
TxDone = Regs::TxDone_Busy;
387 if (Regs::get_TxData_Vaddr(pkt->
getLE<uint64_t>())) {
388 panic(
"vtophys won't work here in newmem.\n");
390 DPRINTF(EthernetPIO,
"write TxData vnic %d (txunique %d)\n",
409 if ((interrupts & Regs::Intr_Res))
410 panic(
"Cannot set a reserved interrupt");
412 regs.IntrStatus |= interrupts;
415 "interrupt written to intStatus: intr=%#x status=%#x mask=%#x\n",
416 interrupts,
regs.IntrStatus,
regs.IntrMask);
418 interrupts =
regs.IntrStatus &
regs.IntrMask;
425 interrupts &= ~Regs::Intr_RxHigh;
432 interrupts &= ~Regs::Intr_TxLow;
436 if ((interrupts & Regs::Intr_NoDelay) == 0)
445 if ((interrupts & Regs::Intr_Res))
446 panic(
"Cannot clear a reserved interrupt");
448 regs.IntrStatus &= ~interrupts;
451 "interrupt cleared from intStatus: intr=%x status=%x mask=%x\n",
452 interrupts,
regs.IntrStatus,
regs.IntrMask);
454 if (!(
regs.IntrStatus &
regs.IntrMask))
461 if (
regs.IntrMask == newmask)
464 regs.IntrMask = newmask;
467 "interrupt mask changed: intStatus=%x intMask=%x masked=%x\n",
470 if (
regs.IntrStatus &
regs.IntrMask)
491 DPRINTF(EthernetIntr,
"interrupts not enabled.\n",
497 DPRINTF(EthernetIntr,
"don't need to schedule event...intrTick=%d\n",
507 DPRINTF(EthernetIntr,
"going to schedule an interrupt for intrTick=%d\n",
531 "would send an interrupt now, but there's already pending\n");
536 DPRINTF(EthernetIntr,
"posting interrupt\n");
556 DPRINTF(EthernetIntr,
"clearing cchip interrupt\n");
567 uint32_t changed =
regs.Config ^ newconf;
571 regs.Config = newconf;
573 if ((changed & Regs::Config_IntEn)) {
576 if (
regs.IntrStatus &
regs.IntrMask)
583 if ((changed & Regs::Config_TxEn)) {
589 if ((changed & Regs::Config_RxEn)) {
599 if (command & Regs::Command_Intr)
602 if (command & Regs::Command_Reset)
609 using namespace Regs;
615 regs.Config |= Config_RxThread;
617 regs.Config |= Config_TxThread;
619 regs.Config |= Config_RSS;
621 regs.Config |= Config_ZeroCopy;
623 regs.Config |= Config_DelayCopy;
624 if (
params()->virtual_addr)
625 regs.Config |= Config_Vaddr;
628 panic(
"Can't delay copy and zero copy");
630 regs.IntrMask = Intr_Soft | Intr_RxHigh | Intr_RxPacket | Intr_TxLow;
634 regs.ZeroCopyMark =
params()->zero_copy_threshold;
641 regs.RxFifoHigh =
params()->rx_fifo_threshold;
642 regs.TxFifoHigh =
params()->tx_fifo_high_mark;
645 if (
regs.RxMaxCopy <
regs.ZeroCopyMark)
646 panic(
"Must be able to copy at least as many bytes as the threshold");
648 if (
regs.ZeroCopySize >=
regs.ZeroCopyMark)
649 panic(
"The number of bytes to copy must be less than the threshold");
672 for (
int i = 0;
i < size; ++
i)
681 DPRINTF(EthernetDMA,
"end rx dma write paddr=%#x len=%d\n",
697 DPRINTF(EthernetSM,
"rxKick: rxState=%s (rxFifo.size=%d)\n",
701 DPRINTF(EthernetSM,
"rxKick: exiting, can't run till %d\n",
713 panic(
"no active vnic while in state %s", RxStateStrings[rxState]);
715 DPRINTF(EthernetSM,
"processing rxState=%s\n",
716 RxStateStrings[rxState]);
720 "processing rxState=%s for vnic %d (rxunique %d)\n",
729 for (
int i = 0;
i < size; ++
i) {
731 bool busy = Regs::get_RxDone_Busy(vn->
RxDone);
738 status =
"busy,dirty";
747 "vnic %d %s (rxunique %d), packet %d, slack %d\n",
753 DPRINTF(EthernetSM,
"vnic %d unmapped (rxunique %d)\n",
765 panic(
"continuing vnic without packet\n");
768 "continue processing for vnic %d (rxunique %d)\n",
785 DPRINTF(EthernetSM,
"receive waiting for data. Nothing to do.\n");
790 panic(
"Not idle, but nothing to do!");
799 "processing new packet for vnic %d (rxunique %d)\n",
814 DPRINTF(Ethernet,
"ID is %d\n", ip->
id());
817 if (
cksum(ip) != 0) {
818 DPRINTF(EthernetCksum,
"Rx IP Checksum Error\n");
825 "Src Port=%d, Dest Port=%d, Seq=%d, Ack=%d\n",
830 if (
cksum(tcp) != 0) {
831 DPRINTF(EthernetCksum,
"Rx TCP Checksum Error\n");
837 if (
cksum(udp) != 0) {
838 DPRINTF(EthernetCksum,
"Rx UDP Checksum Error\n");
859 if ((Regs::get_Config_ZeroCopy(
regs.Config) ||
860 Regs::get_Config_DelayCopy(
regs.Config)) &&
861 !Regs::get_RxData_NoDelay(vnic->
RxData) &&
rxLow) {
876 DPRINTF(EthernetSM,
"receive machine still copying\n");
881 vnic->
RxDone |= Regs::RxDone_Complete;
892 "rxKick: packet complete on vnic %d (rxunique %d)\n",
903 vnic->
RxDone |= Regs::RxDone_More;
907 "rxKick: packet not complete on vnic %d (rxunique %d): " 930 panic(
"Invalid rxState!");
933 DPRINTF(EthernetSM,
"entering next rxState=%s\n",
934 RxStateStrings[rxState]);
942 DPRINTF(EthernetSM,
"rx state machine exited rxState=%s\n",
943 RxStateStrings[rxState]);
951 DPRINTF(EthernetDMA,
"tx dma read paddr=%#x len=%d\n",
966 DPRINTF(Ethernet,
"nothing to transmit\n");
973 DPRINTF(Ethernet,
"Packet Transmit: failed txFifo available %d\n",
983 DPRINTF(Ethernet,
"ID is %d\n", ip->
id());
987 "Src Port=%d, Dest Port=%d, Seq=%d, Ack=%d\n",
988 tcp->sport(), tcp->dport(), tcp->seq(),
995 DDUMP(EthernetData, packet->data, packet->length);
999 DPRINTF(Ethernet,
"Packet Transmit: successful txFifo Available %d\n",
1002 interrupts = Regs::Intr_TxPacket;
1004 interrupts |= Regs::Intr_TxLow;
1012 DPRINTF(EthernetSM,
"txKick: txState=%s (txFifo.size=%d)\n",
1016 DPRINTF(EthernetSM,
"txKick: exiting, can't run till %d\n",
1030 assert(Regs::get_TxDone_Busy(vnic->TxDone));
1033 txPacket = make_shared<EthPacketData>(16384);
1038 Regs::get_TxData_Len(vnic->TxData)) {
1039 DPRINTF(EthernetSM,
"transmit fifo full. Nothing to do.\n");
1051 txDmaLen = Regs::get_TxData_Len(vnic->TxData);
1059 DPRINTF(EthernetSM,
"transmit machine still copying\n");
1063 vnic->TxDone =
txDmaLen | Regs::TxDone_Complete;
1066 if ((vnic->TxData & Regs::TxData_More)) {
1074 if ((vnic->TxData & Regs::TxData_Checksum)) {
1110 panic(
"Invalid txState!");
1113 DPRINTF(EthernetSM,
"entering next txState=%s\n",
1114 TxStateStrings[txState]);
1122 DPRINTF(EthernetSM,
"tx state machine exited txState=%s\n",
1123 TxStateStrings[txState]);
1130 DPRINTF(Ethernet,
"transfer complete: txFifo empty...nothing to do\n");
1134 DPRINTF(Ethernet,
"transfer complete: data in txFifo...schedule xmit\n");
1142 if (!Regs::get_Config_Filter(
regs.Config))
1145 panic(
"receive filter not implemented\n");
1156 DPRINTF(Ethernet,
"Receiving packet from wire, rxFifo Available is %d\n",
1160 DPRINTF(Ethernet,
"receive disabled...packet dropped\n");
1165 DPRINTF(Ethernet,
"packet filtered...dropped\n");
1174 "packet will not fit in receive buffer...packet dropped\n");
1218 Tick intrEventTick = 0;
1241 if (intrEventTick) {
1257 panic(
"can't serialize with an in flight dma request rxState=%s",
1261 panic(
"can't serialize with an in flight dma request txState=%s",
1278 for (
int i = 0;
i < virtualRegsSize; ++
i) {
1288 paramOut(cp, reg +
".rxPacketExists", rxPacketExists);
1289 if (rxPacketExists) {
1298 paramOut(cp, reg +
".rxPacket", rxPacket);
1315 VirtualList::const_iterator
i, end;
1316 for (count = 0, i =
rxList.begin(), end =
rxList.end(); i != end; ++
i)
1318 int rxListSize =
count;
1321 for (count = 0, i =
rxBusy.begin(), end =
rxBusy.end(); i != end; ++
i)
1323 int rxBusySize =
count;
1326 for (count = 0, i =
txList.begin(), end =
txList.end(); i != end; ++
i)
1328 int txListSize =
count;
1347 bool txPacketExists =
txPacket !=
nullptr;
1349 if (txPacketExists) {
1350 txPacket->serialize(
"txPacket", cp);
1386 for (
int i = 0;
i < rxListSize; ++
i) {
1395 for (
int i = 0;
i < rxBusySize; ++
i) {
1404 for (
int i = 0;
i < txListSize; ++
i) {
1417 this->rxState = (
RxState) rxState;
1422 if (rxFifoPtr >= 0) {
1436 this->txState = (
TxState) txState;
1438 bool txPacketExists;
1441 if (txPacketExists) {
1442 txPacket = make_shared<EthPacketData>(16384);
1443 txPacket->unserialize(
"txPacket", cp);
1454 int virtualRegsSize;
1458 for (
int i = 0;
i < virtualRegsSize; ++
i) {
1470 bool rxPacketExists;
1471 paramIn(cp, reg +
".rxPacketExists", rxPacketExists);
1472 if (rxPacketExists) {
1474 paramIn(cp, reg +
".rxPacket", rxPacket);
1479 paramIn(cp, reg +
".rxPacketOffset",
1503 SinicParams::create()
#define panic(...)
This implements a cprintf based panic() function.
Ports are used to interface objects to each other.
const char * TxStateStrings[]
virtual void resetStats()
Callback to reset stats.
void sendRangeChange() const
Called by the owner to send a range change.
Cycles is a wrapper class for representing cycle counts, i.e.
const Regs::Info & regInfo(Addr daddr)
void unserialize(CheckpointIn &cp) override
Reconstruct the state of this object from a checkpoint.
int countPacketsBefore(const_iterator i) const
virtual void drainResume()
Resume execution after a successful drain.
PacketFifo::iterator rxFifoPtr
Addr pciToDma(Addr pci_addr) const
void unserialize(const std::string &base, CheckpointIn &cp)
void unserialize(CheckpointIn &cp) override
Unserialize an object.
Dummy class to keep the Python class hierarchy in sync with the C++ object hierarchy.
void devIntrPost(uint32_t interrupts)
Interrupt management.
#define LL(N)
int64_t constant
void dmaWrite(Addr addr, int size, Event *event, uint8_t *data, uint32_t sid, uint32_t ssid, Tick delay=0)
EventFunctionWrapper txEvent
TxState
Transmit State Machine states.
void prepareRead(ContextID cpu, int index)
uint16_t cksum(const IpPtr &ptr)
void transmit()
Retransmit event.
void squash()
Squash the current event.
Stats::Scalar rxIpChecksums
void cpuIntrPost(Tick when)
Stats::Scalar rxUdpChecksums
#define DDUMP(x, data, count)
const Params * params() const
void regStats() override
Callback to set stat parameters.
EventFunctionWrapper rxDmaEvent
Overload hash function for BasicBlockRange type.
Stats::Scalar rxTcpChecksums
void command(uint32_t command)
Stats::Scalar maxVnicDistance
void setLE(T v)
Set the value in the data pointer to v as little endian.
PioPort< PioDevice > pioPort
The pioPort that handles the requests for us and provides us requests that it sees.
static const int VirtualShift
EventFunctionWrapper * intrEvent
RequestPtr req
A pointer to the original request.
DrainState drainState() const
Return the current drain state of an object.
void serialize(CheckpointOut &cp) const override
Serialization stuff.
bool rxFilter(const EthPacketPtr &packet)
receive address filter
uint64_t & regData64(Addr daddr)
#define UNSERIALIZE_SCALAR(scalar)
Tick curTick()
The current simulated tick.
std::string csprintf(const char *format, const Args &...args)
void regStats()
Callback to set stat parameters.
uint64_t Tick
Tick count type.
void dmaRead(Addr addr, int size, Event *event, uint8_t *data, uint32_t sid, uint32_t ssid, Tick delay=0)
bool regValid(Addr daddr)
void prepareIO(ContextID cpu, int index)
Stats::Scalar numVnicDistance
void paramOut(CheckpointOut &cp, const string &name, ExtMachInst const &machInst)
Tick read(PacketPtr pkt) override
Memory Interface.
Stats::Formula avgVnicDistance
void unserialize(CheckpointIn &cp) override
Unserialize an object.
void resetStats() override
Callback to reset stats.
Stats::Scalar txIpChecksums
std::shared_ptr< EthPacketData > EthPacketPtr
void schedule(Event &event, Tick when)
void rxDmaDone()
DMA parameters.
void devIntrClear(uint32_t interrupts=Regs::Intr_All)
void reschedule(Event &event, Tick when, bool always=false)
Port & getPort(const std::string &if_name, PortID idx=InvalidPortID) override
Get a port with a given name and index.
Defines global host-dependent types: Counter, Tick, and (indirectly) {int,uint}{8,16,32,64}_t.
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
A Packet is used to encapsulate a transfer between two objects in the memory system (e...
bool recvPacket(EthPacketPtr packet)
device ethernet interface
Tick clockEdge(Cycles cycles=Cycles(0)) const
Determine the tick when a cycle begins, by default the current one, but the argument also enables the...
Addr BARAddrs[6]
The current address mapping of the BARs.
bool cpuIntrPending() const
#define SERIALIZE_SCALAR(scalar)
Stats::Scalar totalVnicDistance
Statistics.
bool scheduled() const
Determine if the current event is scheduled.
bool sendPacket(EthPacketPtr packet)
Derived & name(const std::string &name)
Set the name and marks this stat to print at the end of simulation.
uint32_t & regData32(Addr daddr)
Stats::Scalar txUdpChecksums
Tick write(PacketPtr pkt) override
IPR read of device register.
virtual const std::string name() const
Declaration of the Packet class.
std::ostream CheckpointOut
fifo_list::iterator iterator
void prepareWrite(ContextID cpu, int index)
void changeConfig(uint32_t newconfig)
device configuration
int countPacketsAfter(const_iterator i) const
PacketFifo::iterator rxIndex
void devIntrChangeMask(uint32_t newmask)
EventFunctionWrapper txDmaEvent
void paramIn(CheckpointIn &cp, const string &name, ExtMachInst &machInst)
RxState
Receive State Machine States.
T getLE() const
Get the data in the packet byte swapped from little endian to host endian.
Derived & desc(const std::string &_desc)
Set the description and marks this stat to print at the end of simulation.
int16_t PortID
Port index/ID type, and a symbolic name for an invalid port id.
virtual void drainResume() override
Resume execution after a successful drain.
Port & getPort(const std::string &if_name, PortID idx=InvalidPortID) override
Get a port with a given name and index.
bool push(EthPacketPtr ptr)
void serialize(const std::string &base, CheckpointOut &cp) const
Serialization stuff.
PCIConfig config
The current config space.
Tick when() const
Get the time that the event is scheduled.
static const int VirtualMask
int ContextID
Globally unique thread context ID.
struct Sinic::Device::@93 regs
device register file
void serialize(CheckpointOut &cp) const override
Serialization stuff.
void serialize(CheckpointOut &cp) const override
Serialize this object to the given output stream.
uint32_t BARSize[6]
The size of the BARs.
Stats::Scalar txTcpChecksums
const char * RxStateStrings[]