gem5  v21.0.1.0
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
sinic.cc
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2004-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 #include "dev/net/sinic.hh"
30 
31 #include <deque>
32 #include <limits>
33 #include <string>
34 
35 #include "base/compiler.hh"
36 #include "base/debug.hh"
37 #include "base/inet.hh"
38 #include "base/types.hh"
39 #include "debug/EthernetAll.hh"
40 #include "dev/net/etherlink.hh"
41 #include "mem/packet.hh"
42 #include "mem/packet_access.hh"
43 #include "sim/eventq.hh"
44 #include "sim/stats.hh"
45 
46 using namespace Net;
47 
48 namespace Sinic {
49 
50 const char *RxStateStrings[] =
51 {
52  "rxIdle",
53  "rxFifoBlock",
54  "rxBeginCopy",
55  "rxCopy",
56  "rxCopyDone"
57 };
58 
59 const char *TxStateStrings[] =
60 {
61  "txIdle",
62  "txFifoBlock",
63  "txBeginCopy",
64  "txCopy",
65  "txCopyDone"
66 };
67 
68 
70 //
71 // Sinic PCI Device
72 //
73 Base::Base(const Params &p)
74  : EtherDevBase(p), rxEnable(false), txEnable(false),
75  intrDelay(p.intr_delay), intrTick(0), cpuIntrEnable(false),
76  cpuPendingIntr(false), intrEvent(0), interface(NULL)
77 {
78 }
79 
81  : Base(p), rxUnique(0), txUnique(0),
82  virtualRegs(p.virtual_count < 1 ? 1 : p.virtual_count),
83  rxFifo(p.rx_fifo_size), txFifo(p.tx_fifo_size),
84  rxKickTick(0), txKickTick(0),
85  txEvent([this]{ txEventTransmit(); }, name()),
86  rxDmaEvent([this]{ rxDmaDone(); }, name()),
87  txDmaEvent([this]{ txDmaDone(); }, name()),
88  dmaReadDelay(p.dma_read_delay), dmaReadFactor(p.dma_read_factor),
89  dmaWriteDelay(p.dma_write_delay), dmaWriteFactor(p.dma_write_factor),
90  sinicDeviceStats(this)
91 {
92  interface = new Interface(name() + ".int0", this);
93  reset();
94 }
95 
97 {}
98 
100  : Stats::Group(parent, "SinicDevice"),
101  ADD_STAT(totalVnicDistance, UNIT_COUNT,
102  "Total vnic distance"),
103  ADD_STAT(numVnicDistance, UNIT_COUNT,
104  "Number of vnic distance measurements"),
105  ADD_STAT(maxVnicDistance, UNIT_COUNT, "Maximum vnic distance"),
106  ADD_STAT(avgVnicDistance,
107  UNIT_RATE(Stats::Units::Count, Stats::Units::Count),
108  "Average vnic distance", totalVnicDistance / numVnicDistance),
109  _maxVnicDistance(0)
110 {
111 }
112 
113 void
115 {
117 
119 }
120 
121 Port &
122 Device::getPort(const std::string &if_name, PortID idx)
123 {
124  if (if_name == "interface")
125  return *interface;
126  return EtherDevBase::getPort(if_name, idx);
127 }
128 
129 
130 void
132 {
133  int size = virtualRegs.size();
134  if (index > size)
135  panic("Trying to access a vnic that doesn't exist %d > %d\n",
136  index, size);
137 }
138 
139 //add stats for head of line blocking
140 //add stats for average fifo length
141 //add stats for average number of vnics busy
142 
143 void
145 {
146  using namespace Regs;
147  prepareIO(cpu, index);
148 
149  VirtualReg &vnic = virtualRegs[index];
150 
151  // update rx registers
152  uint64_t rxdone = vnic.RxDone;
153  rxdone = set_RxDone_Packets(rxdone, rxFifo.countPacketsAfter(rxFifoPtr));
154  rxdone = set_RxDone_Empty(rxdone, rxFifo.empty());
155  rxdone = set_RxDone_High(rxdone, rxFifo.size() > regs.RxFifoHigh);
156  rxdone = set_RxDone_NotHigh(rxdone, rxLow);
157  regs.RxData = vnic.RxData;
158  regs.RxDone = rxdone;
159  regs.RxWait = rxdone;
160 
161  // update tx regsiters
162  uint64_t txdone = vnic.TxDone;
163  txdone = set_TxDone_Packets(txdone, txFifo.packets());
164  txdone = set_TxDone_Full(txdone, txFifo.avail() < regs.TxMaxCopy);
165  txdone = set_TxDone_Low(txdone, txFifo.size() < regs.TxFifoLow);
166  regs.TxData = vnic.TxData;
167  regs.TxDone = txdone;
168  regs.TxWait = txdone;
169 
170  int head = 0xffff;
171 
172  if (!rxFifo.empty()) {
173  int vnic = rxFifo.begin()->priv;
174  if (vnic != -1 && virtualRegs[vnic].rxPacketOffset > 0)
175  head = vnic;
176  }
177 
178  regs.RxStatus = set_RxStatus_Head(regs.RxStatus, head);
179  regs.RxStatus = set_RxStatus_Busy(regs.RxStatus, rxBusyCount);
180  regs.RxStatus = set_RxStatus_Mapped(regs.RxStatus, rxMappedCount);
181  regs.RxStatus = set_RxStatus_Dirty(regs.RxStatus, rxDirtyCount);
182 }
183 
184 void
186 {
187  prepareIO(cpu, index);
188 }
189 
193 Tick
195 {
196  assert(config.command & PCI_CMD_MSE);
197 
198  Addr daddr = pkt->getAddr();
199  assert(BARs[0]->range().contains(daddr));
200  daddr -= BARs[0]->addr();
201 
202  ContextID cpu = pkt->req->contextId();
203  Addr index = daddr >> Regs::VirtualShift;
204  Addr raddr = daddr & Regs::VirtualMask;
205 
206  if (!regValid(raddr))
207  panic("invalid register: cpu=%d vnic=%d da=%#x pa=%#x size=%d",
208  cpu, index, daddr, pkt->getAddr(), pkt->getSize());
209 
210  const Regs::Info &info = regInfo(raddr);
211  if (!info.read)
212  panic("read %s (write only): "
213  "cpu=%d vnic=%d da=%#x pa=%#x size=%d",
214  info.name, cpu, index, daddr, pkt->getAddr(), pkt->getSize());
215 
216  panic("read %s (invalid size): "
217  "cpu=%d vnic=%d da=%#x pa=%#x size=%d",
218  info.name, cpu, index, daddr, pkt->getAddr(), pkt->getSize());
219 
220  prepareRead(cpu, index);
221 
222  M5_VAR_USED uint64_t value = 0;
223  if (pkt->getSize() == 4) {
224  uint32_t reg = regData32(raddr);
225  pkt->setLE(reg);
226  value = reg;
227  }
228 
229  if (pkt->getSize() == 8) {
230  uint64_t reg = regData64(raddr);
231  pkt->setLE(reg);
232  value = reg;
233  }
234 
235  DPRINTF(EthernetPIO,
236  "read %s: cpu=%d vnic=%d da=%#x pa=%#x size=%d val=%#x\n",
237  info.name, cpu, index, daddr, pkt->getAddr(), pkt->getSize(), value);
238 
239  // reading the interrupt status register has the side effect of
240  // clearing it
241  if (raddr == Regs::IntrStatus)
242  devIntrClear();
243 
244  return pioDelay;
245 }
246 
280 Tick
282 {
283  assert(config.command & PCI_CMD_MSE);
284 
285  Addr daddr = pkt->getAddr();
286  assert(BARs[0]->range().contains(daddr));
287  daddr -= BARs[0]->addr();
288 
289  ContextID cpu = pkt->req->contextId();
290  Addr index = daddr >> Regs::VirtualShift;
291  Addr raddr = daddr & Regs::VirtualMask;
292 
293  if (!regValid(raddr))
294  panic("invalid register: cpu=%d, da=%#x pa=%#x size=%d",
295  cpu, daddr, pkt->getAddr(), pkt->getSize());
296 
297  const Regs::Info &info = regInfo(raddr);
298  if (!info.write)
299  panic("write %s (read only): "
300  "cpu=%d vnic=%d da=%#x pa=%#x size=%d",
301  info.name, cpu, index, daddr, pkt->getAddr(), pkt->getSize());
302 
303  if (pkt->getSize() != info.size)
304  panic("write %s (invalid size): "
305  "cpu=%d vnic=%d da=%#x pa=%#x size=%d",
306  info.name, cpu, index, daddr, pkt->getAddr(), pkt->getSize());
307 
308  VirtualReg &vnic = virtualRegs[index];
309 
310  DPRINTF(EthernetPIO,
311  "write %s vnic %d: cpu=%d val=%#x da=%#x pa=%#x size=%d\n",
312  info.name, index, cpu, info.size == 4 ?
313  pkt->getLE<uint32_t>() : pkt->getLE<uint64_t>(),
314  daddr, pkt->getAddr(), pkt->getSize());
315 
316  prepareWrite(cpu, index);
317 
318  switch (raddr) {
319  case Regs::Config:
320  changeConfig(pkt->getLE<uint32_t>());
321  break;
322 
323  case Regs::Command:
324  command(pkt->getLE<uint32_t>());
325  break;
326 
327  case Regs::IntrStatus:
328  devIntrClear(regs.IntrStatus &
329  pkt->getLE<uint32_t>());
330  break;
331 
332  case Regs::IntrMask:
333  devIntrChangeMask(pkt->getLE<uint32_t>());
334  break;
335 
336  case Regs::RxData:
337  if (Regs::get_RxDone_Busy(vnic.RxDone))
338  panic("receive machine busy with another request! rxState=%s",
340 
341  vnic.rxUnique = rxUnique++;
342  vnic.RxDone = Regs::RxDone_Busy;
343  vnic.RxData = pkt->getLE<uint64_t>();
344  rxBusyCount++;
345 
346  if (Regs::get_RxData_Vaddr(pkt->getLE<uint64_t>())) {
347  panic("vtophys not implemented in newmem");
348  } else {
349  DPRINTF(EthernetPIO, "write RxData vnic %d (rxunique %d)\n",
350  index, vnic.rxUnique);
351  }
352 
353  if (vnic.rxIndex == rxFifo.end()) {
354  DPRINTF(EthernetPIO, "request new packet...appending to rxList\n");
355  rxList.push_back(index);
356  } else {
357  DPRINTF(EthernetPIO, "packet exists...appending to rxBusy\n");
358  rxBusy.push_back(index);
359  }
360 
361  if (rxEnable && (rxState == rxIdle || rxState == rxFifoBlock)) {
363  rxKick();
364  }
365  break;
366 
367  case Regs::TxData:
368  if (Regs::get_TxDone_Busy(vnic.TxDone))
369  panic("transmit machine busy with another request! txState=%s",
371 
372  vnic.txUnique = txUnique++;
373  vnic.TxDone = Regs::TxDone_Busy;
374 
375  if (Regs::get_TxData_Vaddr(pkt->getLE<uint64_t>())) {
376  panic("vtophys won't work here in newmem.\n");
377  } else {
378  DPRINTF(EthernetPIO, "write TxData vnic %d (txunique %d)\n",
379  index, vnic.txUnique);
380  }
381 
382  if (txList.empty() || txList.front() != index)
383  txList.push_back(index);
384  if (txEnable && txState == txIdle && txList.front() == index) {
386  txKick();
387  }
388  break;
389  }
390 
391  return pioDelay;
392 }
393 
394 void
395 Device::devIntrPost(uint32_t interrupts)
396 {
397  if ((interrupts & Regs::Intr_Res))
398  panic("Cannot set a reserved interrupt");
399 
400  regs.IntrStatus |= interrupts;
401 
402  DPRINTF(EthernetIntr,
403  "interrupt written to intStatus: intr=%#x status=%#x mask=%#x\n",
404  interrupts, regs.IntrStatus, regs.IntrMask);
405 
406  interrupts = regs.IntrStatus & regs.IntrMask;
407 
408  // Intr_RxHigh is special, we only signal it if we've emptied the fifo
409  // and then filled it above the high watermark
410  if (rxEmpty)
411  rxEmpty = false;
412  else
413  interrupts &= ~Regs::Intr_RxHigh;
414 
415  // Intr_TxLow is special, we only signal it if we've filled up the fifo
416  // and then dropped below the low watermark
417  if (txFull)
418  txFull = false;
419  else
420  interrupts &= ~Regs::Intr_TxLow;
421 
422  if (interrupts) {
423  Tick when = curTick();
424  if ((interrupts & Regs::Intr_NoDelay) == 0)
425  when += intrDelay;
426  cpuIntrPost(when);
427  }
428 }
429 
430 void
431 Device::devIntrClear(uint32_t interrupts)
432 {
433  if ((interrupts & Regs::Intr_Res))
434  panic("Cannot clear a reserved interrupt");
435 
436  regs.IntrStatus &= ~interrupts;
437 
438  DPRINTF(EthernetIntr,
439  "interrupt cleared from intStatus: intr=%x status=%x mask=%x\n",
440  interrupts, regs.IntrStatus, regs.IntrMask);
441 
442  if (!(regs.IntrStatus & regs.IntrMask))
443  cpuIntrClear();
444 }
445 
446 void
447 Device::devIntrChangeMask(uint32_t newmask)
448 {
449  if (regs.IntrMask == newmask)
450  return;
451 
452  regs.IntrMask = newmask;
453 
454  DPRINTF(EthernetIntr,
455  "interrupt mask changed: intStatus=%x intMask=%x masked=%x\n",
456  regs.IntrStatus, regs.IntrMask, regs.IntrStatus & regs.IntrMask);
457 
458  if (regs.IntrStatus & regs.IntrMask)
459  cpuIntrPost(curTick());
460  else
461  cpuIntrClear();
462 }
463 
464 void
466 {
467  // If the interrupt you want to post is later than an interrupt
468  // already scheduled, just let it post in the coming one and don't
469  // schedule another.
470  // HOWEVER, must be sure that the scheduled intrTick is in the
471  // future (this was formerly the source of a bug)
476  assert(when >= curTick());
477  assert(intrTick >= curTick() || intrTick == 0);
478  if (!cpuIntrEnable) {
479  DPRINTF(EthernetIntr, "interrupts not enabled.\n",
480  intrTick);
481  return;
482  }
483 
484  if (when > intrTick && intrTick != 0) {
485  DPRINTF(EthernetIntr, "don't need to schedule event...intrTick=%d\n",
486  intrTick);
487  return;
488  }
489 
490  intrTick = when;
491  if (intrTick < curTick()) {
492  intrTick = curTick();
493  }
494 
495  DPRINTF(EthernetIntr, "going to schedule an interrupt for intrTick=%d\n",
496  intrTick);
497 
498  if (intrEvent)
499  intrEvent->squash();
500 
501  intrEvent = new EventFunctionWrapper([this]{ cpuInterrupt(); },
502  name(), true);
504 }
505 
506 void
508 {
509  assert(intrTick == curTick());
510 
511  // Whether or not there's a pending interrupt, we don't care about
512  // it anymore
513  intrEvent = 0;
514  intrTick = 0;
515 
516  // Don't send an interrupt if there's already one
517  if (cpuPendingIntr) {
518  DPRINTF(EthernetIntr,
519  "would send an interrupt now, but there's already pending\n");
520  } else {
521  // Send interrupt
522  cpuPendingIntr = true;
523 
524  DPRINTF(EthernetIntr, "posting interrupt\n");
525  intrPost();
526  }
527 }
528 
529 void
531 {
532  if (!cpuPendingIntr)
533  return;
534 
535  if (intrEvent) {
536  intrEvent->squash();
537  intrEvent = 0;
538  }
539 
540  intrTick = 0;
541 
542  cpuPendingIntr = false;
543 
544  DPRINTF(EthernetIntr, "clearing cchip interrupt\n");
545  intrClear();
546 }
547 
548 bool
550 { return cpuPendingIntr; }
551 
552 void
553 Device::changeConfig(uint32_t newconf)
554 {
555  uint32_t changed = regs.Config ^ newconf;
556  if (!changed)
557  return;
558 
559  regs.Config = newconf;
560 
561  if ((changed & Regs::Config_IntEn)) {
562  cpuIntrEnable = regs.Config & Regs::Config_IntEn;
563  if (cpuIntrEnable) {
564  if (regs.IntrStatus & regs.IntrMask)
565  cpuIntrPost(curTick());
566  } else {
567  cpuIntrClear();
568  }
569  }
570 
571  if ((changed & Regs::Config_TxEn)) {
572  txEnable = regs.Config & Regs::Config_TxEn;
573  if (txEnable)
574  txKick();
575  }
576 
577  if ((changed & Regs::Config_RxEn)) {
578  rxEnable = regs.Config & Regs::Config_RxEn;
579  if (rxEnable)
580  rxKick();
581  }
582 }
583 
584 void
586 {
587  if (command & Regs::Command_Intr)
588  devIntrPost(Regs::Intr_Soft);
589 
590  if (command & Regs::Command_Reset)
591  reset();
592 }
593 
594 void
596 {
597  using namespace Regs;
598 
599  memset(&regs, 0, sizeof(regs));
600 
601  regs.Config = 0;
602  if (params().rx_thread)
603  regs.Config |= Config_RxThread;
604  if (params().tx_thread)
605  regs.Config |= Config_TxThread;
606  if (params().rss)
607  regs.Config |= Config_RSS;
608  if (params().zero_copy)
609  regs.Config |= Config_ZeroCopy;
610  if (params().delay_copy)
611  regs.Config |= Config_DelayCopy;
612  if (params().virtual_addr)
613  regs.Config |= Config_Vaddr;
614 
615  if (params().delay_copy && params().zero_copy)
616  panic("Can't delay copy and zero copy");
617 
618  regs.IntrMask = Intr_Soft | Intr_RxHigh | Intr_RxPacket | Intr_TxLow;
619  regs.RxMaxCopy = params().rx_max_copy;
620  regs.TxMaxCopy = params().tx_max_copy;
621  regs.ZeroCopySize = params().zero_copy_size;
622  regs.ZeroCopyMark = params().zero_copy_threshold;
623  regs.VirtualCount = params().virtual_count;
624  regs.RxMaxIntr = params().rx_max_intr;
625  regs.RxFifoSize = params().rx_fifo_size;
626  regs.TxFifoSize = params().tx_fifo_size;
627  regs.RxFifoLow = params().rx_fifo_low_mark;
628  regs.TxFifoLow = params().tx_fifo_threshold;
629  regs.RxFifoHigh = params().rx_fifo_threshold;
630  regs.TxFifoHigh = params().tx_fifo_high_mark;
631  regs.HwAddr = params().hardware_address;
632 
633  if (regs.RxMaxCopy < regs.ZeroCopyMark)
634  panic("Must be able to copy at least as many bytes as the threshold");
635 
636  if (regs.ZeroCopySize >= regs.ZeroCopyMark)
637  panic("The number of bytes to copy must be less than the threshold");
638 
639  rxList.clear();
640  rxBusy.clear();
641  rxActive = -1;
642  txList.clear();
643  rxBusyCount = 0;
644  rxDirtyCount = 0;
645  rxMappedCount = 0;
646 
647  rxState = rxIdle;
648  txState = txIdle;
649 
650  rxFifo.clear();
651  rxFifoPtr = rxFifo.end();
652  txFifo.clear();
653  rxEmpty = false;
654  rxLow = true;
655  txFull = false;
656 
657  int size = virtualRegs.size();
658  virtualRegs.clear();
659  virtualRegs.resize(size);
660  for (int i = 0; i < size; ++i)
661  virtualRegs[i].rxIndex = rxFifo.end();
662 }
663 
664 void
666 {
667  assert(rxState == rxCopy);
669  DPRINTF(EthernetDMA, "end rx dma write paddr=%#x len=%d\n",
671  DDUMP(EthernetData, rxDmaData, rxDmaLen);
672 
673  // If the transmit state machine has a pending DMA, let it go first
674  if (txState == txBeginCopy)
675  txKick();
676 
677  rxKick();
678 }
679 
680 void
682 {
683  VirtualReg *vnic = NULL;
684 
685  DPRINTF(EthernetSM, "rxKick: rxState=%s (rxFifo.size=%d)\n",
687 
688  if (rxKickTick > curTick()) {
689  DPRINTF(EthernetSM, "rxKick: exiting, can't run till %d\n",
690  rxKickTick);
691  return;
692  }
693 
694  next:
695  rxFifo.check();
696  if (rxState == rxIdle)
697  goto exit;
698 
699  if (rxActive == -1) {
700  if (rxState != rxFifoBlock)
701  panic("no active vnic while in state %s", RxStateStrings[rxState]);
702 
703  DPRINTF(EthernetSM, "processing rxState=%s\n",
705  } else {
706  vnic = &virtualRegs[rxActive];
707  DPRINTF(EthernetSM,
708  "processing rxState=%s for vnic %d (rxunique %d)\n",
710  }
711 
712  switch (rxState) {
713  case rxFifoBlock:
714  if (DTRACE(EthernetSM)) {
716  int size = virtualRegs.size();
717  for (int i = 0; i < size; ++i) {
718  VirtualReg *vn = &virtualRegs[i];
719  bool busy = Regs::get_RxDone_Busy(vn->RxDone);
720  if (vn->rxIndex != end) {
721 #ifndef NDEBUG
722  bool dirty = vn->rxPacketOffset > 0;
723  const char *status;
724 
725  if (busy && dirty)
726  status = "busy,dirty";
727  else if (busy)
728  status = "busy";
729  else if (dirty)
730  status = "dirty";
731  else
732  status = "mapped";
733 
734  DPRINTF(EthernetSM,
735  "vnic %d %s (rxunique %d), packet %d, slack %d\n",
736  i, status, vn->rxUnique,
738  vn->rxIndex->slack);
739 #endif
740  } else if (busy) {
741  DPRINTF(EthernetSM, "vnic %d unmapped (rxunique %d)\n",
742  i, vn->rxUnique);
743  }
744  }
745  }
746 
747  if (!rxBusy.empty()) {
748  rxActive = rxBusy.front();
749  rxBusy.pop_front();
750  vnic = &virtualRegs[rxActive];
751 
752  if (vnic->rxIndex == rxFifo.end())
753  panic("continuing vnic without packet\n");
754 
755  DPRINTF(EthernetSM,
756  "continue processing for vnic %d (rxunique %d)\n",
757  rxActive, vnic->rxUnique);
758 
760 
761  int vnic_distance = rxFifo.countPacketsBefore(vnic->rxIndex);
762  sinicDeviceStats.totalVnicDistance += vnic_distance;
764  if (vnic_distance > sinicDeviceStats._maxVnicDistance) {
765  sinicDeviceStats.maxVnicDistance = vnic_distance;
766  sinicDeviceStats._maxVnicDistance = vnic_distance;
767  }
768 
769  break;
770  }
771 
772  if (rxFifoPtr == rxFifo.end()) {
773  DPRINTF(EthernetSM, "receive waiting for data. Nothing to do.\n");
774  goto exit;
775  }
776 
777  if (rxList.empty())
778  panic("Not idle, but nothing to do!");
779 
780  assert(!rxFifo.empty());
781 
782  rxActive = rxList.front();
783  rxList.pop_front();
784  vnic = &virtualRegs[rxActive];
785 
786  DPRINTF(EthernetSM,
787  "processing new packet for vnic %d (rxunique %d)\n",
788  rxActive, vnic->rxUnique);
789 
790  // Grab a new packet from the fifo.
791  vnic->rxIndex = rxFifoPtr++;
792  vnic->rxIndex->priv = rxActive;
793  vnic->rxPacketOffset = 0;
794  vnic->rxPacketBytes = vnic->rxIndex->packet->length;
795  assert(vnic->rxPacketBytes);
796  rxMappedCount++;
797 
798  vnic->rxDoneData = 0;
799  /* scope for variables */ {
800  IpPtr ip(vnic->rxIndex->packet);
801  if (ip) {
802  DPRINTF(Ethernet, "ID is %d\n", ip->id());
803  vnic->rxDoneData |= Regs::RxDone_IpPacket;
805  if (cksum(ip) != 0) {
806  DPRINTF(EthernetCksum, "Rx IP Checksum Error\n");
807  vnic->rxDoneData |= Regs::RxDone_IpError;
808  }
809  TcpPtr tcp(ip);
810  UdpPtr udp(ip);
811  if (tcp) {
812  DPRINTF(Ethernet,
813  "Src Port=%d, Dest Port=%d, Seq=%d, Ack=%d\n",
814  tcp->sport(), tcp->dport(), tcp->seq(),
815  tcp->ack());
816  vnic->rxDoneData |= Regs::RxDone_TcpPacket;
818  if (cksum(tcp) != 0) {
819  DPRINTF(EthernetCksum, "Rx TCP Checksum Error\n");
820  vnic->rxDoneData |= Regs::RxDone_TcpError;
821  }
822  } else if (udp) {
823  vnic->rxDoneData |= Regs::RxDone_UdpPacket;
825  if (cksum(udp) != 0) {
826  DPRINTF(EthernetCksum, "Rx UDP Checksum Error\n");
827  vnic->rxDoneData |= Regs::RxDone_UdpError;
828  }
829  }
830  }
831  }
833  break;
834 
835  case rxBeginCopy:
837  goto exit;
838 
839  rxDmaAddr = pciToDma(Regs::get_RxData_Addr(vnic->RxData));
840  rxDmaLen = std::min<unsigned>(Regs::get_RxData_Len(vnic->RxData),
841  vnic->rxPacketBytes);
842 
843  /*
844  * if we're doing zero/delay copy and we're below the fifo
845  * threshold, see if we should try to do the zero/defer copy
846  */
847  if ((Regs::get_Config_ZeroCopy(regs.Config) ||
848  Regs::get_Config_DelayCopy(regs.Config)) &&
849  !Regs::get_RxData_NoDelay(vnic->RxData) && rxLow) {
850  if (rxDmaLen > regs.ZeroCopyMark)
851  rxDmaLen = regs.ZeroCopySize;
852  }
853  rxDmaData = vnic->rxIndex->packet->data + vnic->rxPacketOffset;
854  rxState = rxCopy;
855  if (rxDmaAddr == 1LL) {
857  break;
858  }
859 
861  break;
862 
863  case rxCopy:
864  DPRINTF(EthernetSM, "receive machine still copying\n");
865  goto exit;
866 
867  case rxCopyDone:
868  vnic->RxDone = vnic->rxDoneData;
869  vnic->RxDone |= Regs::RxDone_Complete;
870  rxBusyCount--;
871 
872  if (vnic->rxPacketBytes == rxDmaLen) {
873  if (vnic->rxPacketOffset)
874  rxDirtyCount--;
875 
876  // Packet is complete. Indicate how many bytes were copied
877  vnic->RxDone = Regs::set_RxDone_CopyLen(vnic->RxDone, rxDmaLen);
878 
879  DPRINTF(EthernetSM,
880  "rxKick: packet complete on vnic %d (rxunique %d)\n",
881  rxActive, vnic->rxUnique);
882  rxFifo.remove(vnic->rxIndex);
883  vnic->rxIndex = rxFifo.end();
884  rxMappedCount--;
885  } else {
886  if (!vnic->rxPacketOffset)
887  rxDirtyCount++;
888 
889  vnic->rxPacketBytes -= rxDmaLen;
890  vnic->rxPacketOffset += rxDmaLen;
891  vnic->RxDone |= Regs::RxDone_More;
892  vnic->RxDone = Regs::set_RxDone_CopyLen(vnic->RxDone,
893  vnic->rxPacketBytes);
894  DPRINTF(EthernetSM,
895  "rxKick: packet not complete on vnic %d (rxunique %d): "
896  "%d bytes left\n",
897  rxActive, vnic->rxUnique, vnic->rxPacketBytes);
898  }
899 
900  rxActive = -1;
901  rxState = rxBusy.empty() && rxList.empty() ? rxIdle : rxFifoBlock;
902 
903  if (rxFifo.empty()) {
904  devIntrPost(Regs::Intr_RxEmpty);
905  rxEmpty = true;
906  }
907 
908  if (rxFifo.size() < regs.RxFifoLow)
909  rxLow = true;
910 
911  if (rxFifo.size() > regs.RxFifoHigh)
912  rxLow = false;
913 
914  devIntrPost(Regs::Intr_RxDMA);
915  break;
916 
917  default:
918  panic("Invalid rxState!");
919  }
920 
921  DPRINTF(EthernetSM, "entering next rxState=%s\n",
923 
924  goto next;
925 
926  exit:
930  DPRINTF(EthernetSM, "rx state machine exited rxState=%s\n",
932 }
933 
934 void
936 {
937  assert(txState == txCopy);
939  DPRINTF(EthernetDMA, "tx dma read paddr=%#x len=%d\n",
941  DDUMP(EthernetData, txDmaData, txDmaLen);
942 
943  // If the receive state machine has a pending DMA, let it go first
944  if (rxState == rxBeginCopy)
945  rxKick();
946 
947  txKick();
948 }
949 
950 void
952 {
953  if (txFifo.empty()) {
954  DPRINTF(Ethernet, "nothing to transmit\n");
955  return;
956  }
957 
958  uint32_t interrupts;
959  EthPacketPtr packet = txFifo.front();
960  if (!interface->sendPacket(packet)) {
961  DPRINTF(Ethernet, "Packet Transmit: failed txFifo available %d\n",
962  txFifo.avail());
963  return;
964  }
965 
966  txFifo.pop();
967 #if TRACING_ON
968  if (DTRACE(Ethernet)) {
969  IpPtr ip(packet);
970  if (ip) {
971  DPRINTF(Ethernet, "ID is %d\n", ip->id());
972  TcpPtr tcp(ip);
973  if (tcp) {
974  DPRINTF(Ethernet,
975  "Src Port=%d, Dest Port=%d, Seq=%d, Ack=%d\n",
976  tcp->sport(), tcp->dport(), tcp->seq(),
977  tcp->ack());
978  }
979  }
980  }
981 #endif
982 
983  DDUMP(EthernetData, packet->data, packet->length);
984  etherDeviceStats.txBytes += packet->length;
986 
987  DPRINTF(Ethernet, "Packet Transmit: successful txFifo Available %d\n",
988  txFifo.avail());
989 
990  interrupts = Regs::Intr_TxPacket;
991  if (txFifo.size() < regs.TxFifoLow)
992  interrupts |= Regs::Intr_TxLow;
993  devIntrPost(interrupts);
994 }
995 
996 void
998 {
999  VirtualReg *vnic;
1000  DPRINTF(EthernetSM, "txKick: txState=%s (txFifo.size=%d)\n",
1002 
1003  if (txKickTick > curTick()) {
1004  DPRINTF(EthernetSM, "txKick: exiting, can't run till %d\n",
1005  txKickTick);
1006  return;
1007  }
1008 
1009  next:
1010  if (txState == txIdle)
1011  goto exit;
1012 
1013  assert(!txList.empty());
1014  vnic = &virtualRegs[txList.front()];
1015 
1016  switch (txState) {
1017  case txFifoBlock:
1018  assert(Regs::get_TxDone_Busy(vnic->TxDone));
1019  if (!txPacket) {
1020  // Grab a new packet from the fifo.
1021  txPacket = std::make_shared<EthPacketData>(16384);
1022  txPacketOffset = 0;
1023  }
1024 
1025  if (txFifo.avail() - txPacket->length <
1026  Regs::get_TxData_Len(vnic->TxData)) {
1027  DPRINTF(EthernetSM, "transmit fifo full. Nothing to do.\n");
1028  goto exit;
1029  }
1030 
1031  txState = txBeginCopy;
1032  break;
1033 
1034  case txBeginCopy:
1036  goto exit;
1037 
1038  txDmaAddr = pciToDma(Regs::get_TxData_Addr(vnic->TxData));
1039  txDmaLen = Regs::get_TxData_Len(vnic->TxData);
1040  txDmaData = txPacket->data + txPacketOffset;
1041  txState = txCopy;
1042 
1044  break;
1045 
1046  case txCopy:
1047  DPRINTF(EthernetSM, "transmit machine still copying\n");
1048  goto exit;
1049 
1050  case txCopyDone:
1051  vnic->TxDone = txDmaLen | Regs::TxDone_Complete;
1052  txPacket->simLength += txDmaLen;
1053  txPacket->length += txDmaLen;
1054  if ((vnic->TxData & Regs::TxData_More)) {
1056  txState = txIdle;
1057  devIntrPost(Regs::Intr_TxDMA);
1058  break;
1059  }
1060 
1061  assert(txPacket->length <= txFifo.avail());
1062  if ((vnic->TxData & Regs::TxData_Checksum)) {
1063  IpPtr ip(txPacket);
1064  if (ip) {
1065  TcpPtr tcp(ip);
1066  if (tcp) {
1067  tcp->sum(0);
1068  tcp->sum(cksum(tcp));
1070  }
1071 
1072  UdpPtr udp(ip);
1073  if (udp) {
1074  udp->sum(0);
1075  udp->sum(cksum(udp));
1077  }
1078 
1079  ip->sum(0);
1080  ip->sum(cksum(ip));
1082  }
1083  }
1084 
1085  txFifo.push(txPacket);
1086  if (txFifo.avail() < regs.TxMaxCopy) {
1087  devIntrPost(Regs::Intr_TxFull);
1088  txFull = true;
1089  }
1090  txPacket = 0;
1091  transmit();
1092  txList.pop_front();
1093  txState = txList.empty() ? txIdle : txFifoBlock;
1094  devIntrPost(Regs::Intr_TxDMA);
1095  break;
1096 
1097  default:
1098  panic("Invalid txState!");
1099  }
1100 
1101  DPRINTF(EthernetSM, "entering next txState=%s\n",
1103 
1104  goto next;
1105 
1106  exit:
1110  DPRINTF(EthernetSM, "tx state machine exited txState=%s\n",
1112 }
1113 
1114 void
1116 {
1117  if (txFifo.empty()) {
1118  DPRINTF(Ethernet, "transfer complete: txFifo empty...nothing to do\n");
1119  return;
1120  }
1121 
1122  DPRINTF(Ethernet, "transfer complete: data in txFifo...schedule xmit\n");
1123 
1124  reschedule(txEvent, clockEdge(Cycles(1)), true);
1125 }
1126 
1127 bool
1129 {
1130  if (!Regs::get_Config_Filter(regs.Config))
1131  return false;
1132 
1133  panic("receive filter not implemented\n");
1134  bool drop = true;
1135  return drop;
1136 }
1137 
1138 bool
1140 {
1141  etherDeviceStats.rxBytes += packet->length;
1143 
1144  DPRINTF(Ethernet, "Receiving packet from wire, rxFifo Available is %d\n",
1145  rxFifo.avail());
1146 
1147  if (!rxEnable) {
1148  DPRINTF(Ethernet, "receive disabled...packet dropped\n");
1149  return true;
1150  }
1151 
1152  if (rxFilter(packet)) {
1153  DPRINTF(Ethernet, "packet filtered...dropped\n");
1154  return true;
1155  }
1156 
1157  if (rxFifo.size() >= regs.RxFifoHigh)
1158  devIntrPost(Regs::Intr_RxHigh);
1159 
1160  if (!rxFifo.push(packet)) {
1161  DPRINTF(Ethernet,
1162  "packet will not fit in receive buffer...packet dropped\n");
1163  return false;
1164  }
1165 
1166  // If we were at the last element, back up one ot go to the new
1167  // last element of the list.
1168  if (rxFifoPtr == rxFifo.end())
1169  --rxFifoPtr;
1170 
1171  devIntrPost(Regs::Intr_RxPacket);
1172  rxKick();
1173  return true;
1174 }
1175 
1176 void
1178 {
1180 
1181  // During drain we could have left the state machines in a waiting state and
1182  // they wouldn't get out until some other event occured to kick them.
1183  // This way they'll get out immediately
1184  txKick();
1185  rxKick();
1186 }
1187 
1188 //=====================================================================
1189 //
1190 //
1191 void
1193 {
1194  // Serialize the PciDevice base class
1196 
1200 
1201  /*
1202  * Keep track of pending interrupt status.
1203  */
1206  Tick intrEventTick = 0;
1207  if (intrEvent)
1208  intrEventTick = intrEvent->when();
1209  SERIALIZE_SCALAR(intrEventTick);
1210 }
1211 
1212 void
1214 {
1215  // Unserialize the PciDevice base class
1217 
1221 
1222  /*
1223  * Keep track of pending interrupt status.
1224  */
1227  Tick intrEventTick;
1228  UNSERIALIZE_SCALAR(intrEventTick);
1229  if (intrEventTick) {
1230  intrEvent = new EventFunctionWrapper([this]{ cpuInterrupt(); },
1231  name(), true);
1232  schedule(intrEvent, intrEventTick);
1233  }
1234 }
1235 
1236 void
1238 {
1239  int count;
1240 
1241  // Serialize the PciDevice base class
1243 
1244  if (rxState == rxCopy)
1245  panic("can't serialize with an in flight dma request rxState=%s",
1247 
1248  if (txState == txCopy)
1249  panic("can't serialize with an in flight dma request txState=%s",
1251 
1252  /*
1253  * Serialize the device registers that could be modified by the OS.
1254  */
1255  SERIALIZE_SCALAR(regs.Config);
1256  SERIALIZE_SCALAR(regs.IntrStatus);
1257  SERIALIZE_SCALAR(regs.IntrMask);
1258  SERIALIZE_SCALAR(regs.RxData);
1259  SERIALIZE_SCALAR(regs.TxData);
1260 
1261  /*
1262  * Serialize the virtual nic state
1263  */
1264  int virtualRegsSize = virtualRegs.size();
1265  SERIALIZE_SCALAR(virtualRegsSize);
1266  for (int i = 0; i < virtualRegsSize; ++i) {
1267  const VirtualReg *vnic = &virtualRegs[i];
1268 
1269  std::string reg = csprintf("vnic%d", i);
1270  paramOut(cp, reg + ".RxData", vnic->RxData);
1271  paramOut(cp, reg + ".RxDone", vnic->RxDone);
1272  paramOut(cp, reg + ".TxData", vnic->TxData);
1273  paramOut(cp, reg + ".TxDone", vnic->TxDone);
1274 
1275  bool rxPacketExists = vnic->rxIndex != rxFifo.end();
1276  paramOut(cp, reg + ".rxPacketExists", rxPacketExists);
1277  if (rxPacketExists) {
1278  int rxPacket = 0;
1279  auto i = rxFifo.begin();
1280  while (i != vnic->rxIndex) {
1281  assert(i != rxFifo.end());
1282  ++i;
1283  ++rxPacket;
1284  }
1285 
1286  paramOut(cp, reg + ".rxPacket", rxPacket);
1287  paramOut(cp, reg + ".rxPacketOffset", vnic->rxPacketOffset);
1288  paramOut(cp, reg + ".rxPacketBytes", vnic->rxPacketBytes);
1289  }
1290  paramOut(cp, reg + ".rxDoneData", vnic->rxDoneData);
1291  }
1292 
1293  int rxFifoPtr = -1;
1294  if (this->rxFifoPtr != rxFifo.end())
1295  rxFifoPtr = rxFifo.countPacketsBefore(this->rxFifoPtr);
1297 
1302 
1303  VirtualList::const_iterator i, end;
1304  for (count = 0, i = rxList.begin(), end = rxList.end(); i != end; ++i)
1305  paramOut(cp, csprintf("rxList%d", count++), *i);
1306  int rxListSize = count;
1307  SERIALIZE_SCALAR(rxListSize);
1308 
1309  for (count = 0, i = rxBusy.begin(), end = rxBusy.end(); i != end; ++i)
1310  paramOut(cp, csprintf("rxBusy%d", count++), *i);
1311  int rxBusySize = count;
1312  SERIALIZE_SCALAR(rxBusySize);
1313 
1314  for (count = 0, i = txList.begin(), end = txList.end(); i != end; ++i)
1315  paramOut(cp, csprintf("txList%d", count++), *i);
1316  int txListSize = count;
1317  SERIALIZE_SCALAR(txListSize);
1318 
1319  /*
1320  * Serialize rx state machine
1321  */
1322  int rxState = this->rxState;
1323  SERIALIZE_SCALAR(rxState);
1326  rxFifo.serialize("rxFifo", cp);
1327 
1328  /*
1329  * Serialize tx state machine
1330  */
1331  int txState = this->txState;
1332  SERIALIZE_SCALAR(txState);
1334  txFifo.serialize("txFifo", cp);
1335  bool txPacketExists = txPacket != nullptr;
1336  SERIALIZE_SCALAR(txPacketExists);
1337  if (txPacketExists) {
1338  txPacket->serialize("txPacket", cp);
1341  }
1342 
1343  /*
1344  * If there's a pending transmit, store the time so we can
1345  * reschedule it later
1346  */
1347  Tick transmitTick = txEvent.scheduled() ? txEvent.when() - curTick() : 0;
1348  SERIALIZE_SCALAR(transmitTick);
1349 }
1350 
1351 void
1353 {
1354  // Unserialize the PciDevice base class
1356 
1357  /*
1358  * Unserialize the device registers that may have been written by the OS.
1359  */
1360  UNSERIALIZE_SCALAR(regs.Config);
1361  UNSERIALIZE_SCALAR(regs.IntrStatus);
1362  UNSERIALIZE_SCALAR(regs.IntrMask);
1363  UNSERIALIZE_SCALAR(regs.RxData);
1364  UNSERIALIZE_SCALAR(regs.TxData);
1365 
1370 
1371  int rxListSize;
1372  UNSERIALIZE_SCALAR(rxListSize);
1373  rxList.clear();
1374  for (int i = 0; i < rxListSize; ++i) {
1375  int value;
1376  paramIn(cp, csprintf("rxList%d", i), value);
1377  rxList.push_back(value);
1378  }
1379 
1380  int rxBusySize;
1381  UNSERIALIZE_SCALAR(rxBusySize);
1382  rxBusy.clear();
1383  for (int i = 0; i < rxBusySize; ++i) {
1384  int value;
1385  paramIn(cp, csprintf("rxBusy%d", i), value);
1386  rxBusy.push_back(value);
1387  }
1388 
1389  int txListSize;
1390  UNSERIALIZE_SCALAR(txListSize);
1391  txList.clear();
1392  for (int i = 0; i < txListSize; ++i) {
1393  int value;
1394  paramIn(cp, csprintf("txList%d", i), value);
1395  txList.push_back(value);
1396  }
1397 
1398  /*
1399  * Unserialize rx state machine
1400  */
1401  int rxState;
1405  this->rxState = (RxState) rxState;
1406  rxFifo.unserialize("rxFifo", cp);
1407 
1408  int rxFifoPtr;
1410  if (rxFifoPtr >= 0) {
1411  this->rxFifoPtr = rxFifo.begin();
1412  for (int i = 0; i < rxFifoPtr; ++i)
1413  ++this->rxFifoPtr;
1414  } else {
1415  this->rxFifoPtr = rxFifo.end();
1416  }
1417 
1418  /*
1419  * Unserialize tx state machine
1420  */
1421  int txState;
1424  this->txState = (TxState) txState;
1425  txFifo.unserialize("txFifo", cp);
1426  bool txPacketExists;
1427  UNSERIALIZE_SCALAR(txPacketExists);
1428  txPacket = 0;
1429  if (txPacketExists) {
1430  txPacket = std::make_shared<EthPacketData>(16384);
1431  txPacket->unserialize("txPacket", cp);
1434  }
1435 
1436  /*
1437  * unserialize the virtual nic registers/state
1438  *
1439  * this must be done after the unserialization of the rxFifo
1440  * because the packet iterators depend on the fifo being populated
1441  */
1442  int virtualRegsSize;
1443  UNSERIALIZE_SCALAR(virtualRegsSize);
1444  virtualRegs.clear();
1445  virtualRegs.resize(virtualRegsSize);
1446  for (int i = 0; i < virtualRegsSize; ++i) {
1447  VirtualReg *vnic = &virtualRegs[i];
1448  std::string reg = csprintf("vnic%d", i);
1449 
1450  paramIn(cp, reg + ".RxData", vnic->RxData);
1451  paramIn(cp, reg + ".RxDone", vnic->RxDone);
1452  paramIn(cp, reg + ".TxData", vnic->TxData);
1453  paramIn(cp, reg + ".TxDone", vnic->TxDone);
1454 
1455  vnic->rxUnique = rxUnique++;
1456  vnic->txUnique = txUnique++;
1457 
1458  bool rxPacketExists;
1459  paramIn(cp, reg + ".rxPacketExists", rxPacketExists);
1460  if (rxPacketExists) {
1461  int rxPacket;
1462  paramIn(cp, reg + ".rxPacket", rxPacket);
1463  vnic->rxIndex = rxFifo.begin();
1464  while (rxPacket--)
1465  ++vnic->rxIndex;
1466 
1467  paramIn(cp, reg + ".rxPacketOffset",
1468  vnic->rxPacketOffset);
1469  paramIn(cp, reg + ".rxPacketBytes", vnic->rxPacketBytes);
1470  } else {
1471  vnic->rxIndex = rxFifo.end();
1472  }
1473  paramIn(cp, reg + ".rxDoneData", vnic->rxDoneData);
1474  }
1475 
1476  /*
1477  * If there's a pending transmit, reschedule it now
1478  */
1479  Tick transmitTick;
1480  UNSERIALIZE_SCALAR(transmitTick);
1481  if (transmitTick)
1482  schedule(txEvent, curTick() + transmitTick);
1483 
1485 
1486 }
1487 
1488 } // namespace Sinic
PacketFifo::countPacketsBefore
int countPacketsBefore(const_iterator i) const
Definition: pktfifo.hh:177
Sinic::Device::rxIdle
@ rxIdle
Definition: sinic.hh:88
Sinic::Regs::VirtualMask
static const int VirtualMask
Definition: sinicreg.hh:59
Sinic::Device::txKick
void txKick()
Definition: sinic.cc:997
ArmISA::status
Bitfield< 5, 0 > status
Definition: miscregs_types.hh:417
Sinic::Device::VirtualReg::rxIndex
PacketFifo::iterator rxIndex
Definition: sinic.hh:138
Event::scheduled
bool scheduled() const
Determine if the current event is scheduled.
Definition: eventq.hh:462
PacketFifo::push
bool push(EthPacketPtr ptr)
Definition: pktfifo.hh:120
sinic.hh
Sinic::Regs::Info
Definition: sinicreg.hh:171
LL
#define LL(N)
int64_t constant
Definition: types.hh:48
Sinic::Device::VirtualReg::txUnique
Counter txUnique
Definition: sinic.hh:144
Sinic::Device::rxDmaData
uint8_t * rxDmaData
Definition: sinic.hh:176
Sinic::Device::rxFifo
PacketFifo rxFifo
Definition: sinic.hh:171
DrainState::Running
@ Running
Running normally.
Sinic::Device::rxEmpty
bool rxEmpty
Definition: sinic.hh:173
EtherDevice::EtherDeviceStats::rxUdpChecksums
Stats::Scalar rxUdpChecksums
Definition: etherdevice.hh:77
PacketFifo::serialize
void serialize(const std::string &base, CheckpointOut &cp) const
Serialization stuff.
Definition: pktfifo.cc:84
MipsISA::index
Bitfield< 30, 0 > index
Definition: pra_constants.hh:44
UNSERIALIZE_SCALAR
#define UNSERIALIZE_SCALAR(scalar)
Definition: serialize.hh:591
Sinic::Device::DeviceStats::maxVnicDistance
Stats::Scalar maxVnicDistance
Definition: sinic.hh:279
EtherDevice::EtherDeviceStats::rxBytes
Stats::Scalar rxBytes
Definition: etherdevice.hh:62
Sinic::Device::rxDmaLen
unsigned rxDmaLen
Definition: sinic.hh:177
Packet::getAddr
Addr getAddr() const
Definition: packet.hh:755
Sinic::Device::rxActive
int rxActive
Definition: sinic.hh:158
Sinic::Device::rxDmaEvent
EventFunctionWrapper rxDmaEvent
Definition: sinic.hh:238
PciDevice::BARs
std::array< PciBar *, 6 > BARs
Definition: device.hh:305
PacketFifo::pop
void pop()
Definition: pktfifo.hh:137
PacketFifo::unserialize
void unserialize(const std::string &base, CheckpointIn &cp)
Definition: pktfifo.cc:97
ArmISA::i
Bitfield< 7 > i
Definition: miscregs_types.hh:63
Sinic::Device::txPacketBytes
int txPacketBytes
Definition: sinic.hh:184
X86ISA::exit
Bitfield< 3 > exit
Definition: misc.hh:848
EventManager::reschedule
void reschedule(Event &event, Tick when, bool always=false)
Definition: eventq.hh:1034
Sinic::Base::cpuIntrPost
void cpuIntrPost(Tick when)
Definition: sinic.cc:465
Sinic::Device::DeviceStats::DeviceStats
DeviceStats(Stats::Group *parent)
Definition: sinic.cc:99
Sinic::Device::txFifo
PacketFifo txFifo
Definition: sinic.hh:180
Sinic::Device::drainResume
virtual void drainResume() override
Resume execution after a successful drain.
Definition: sinic.cc:1177
Net::TcpPtr
Definition: inet.hh:637
ContextID
int ContextID
Globally unique thread context ID.
Definition: types.hh:237
Sinic::Device::unserialize
void unserialize(CheckpointIn &cp) override
Unserialize an object.
Definition: sinic.cc:1352
iGbReg::TxdOp::tcp
bool tcp(TxDesc *d)
Definition: i8254xGBe_defs.hh:262
Sinic::Device::txFifoBlock
@ txFifoBlock
Definition: sinic.hh:98
Sinic::Device::read
Tick read(PacketPtr pkt) override
Memory Interface.
Definition: sinic.cc:194
Sinic::Device::write
Tick write(PacketPtr pkt) override
IPR read of device register.
Definition: sinic.cc:281
PacketFifo::remove
void remove(iterator i)
Definition: pktfifo.hh:158
Sinic::Device::rxState
RxState rxState
Definition: sinic.hh:170
Tick
uint64_t Tick
Tick count type.
Definition: types.hh:59
EtherDevice::EtherDeviceStats::rxIpChecksums
Stats::Scalar rxIpChecksums
Definition: etherdevice.hh:71
Net::UdpPtr
Definition: inet.hh:751
PortID
int16_t PortID
Port index/ID type, and a symbolic name for an invalid port id.
Definition: types.hh:243
PciDevice::config
PCIConfig config
The current config space.
Definition: device.hh:272
Sinic::Device::txPacket
EthPacketPtr txPacket
Definition: sinic.hh:182
Packet::req
RequestPtr req
A pointer to the original request.
Definition: packet.hh:341
Sinic::Device::txDmaDone
void txDmaDone()
Definition: sinic.cc:935
DTRACE
#define DTRACE(x)
Definition: debug.hh:156
Sinic::Device::rxMappedCount
int rxMappedCount
Definition: sinic.hh:162
Sinic::Device::txEvent
EventFunctionWrapper txEvent
Definition: sinic.hh:208
Sinic::Device::changeConfig
void changeConfig(uint32_t newconfig)
device configuration
Definition: sinic.cc:553
PciDevice::intrClear
void intrClear()
Definition: device.hh:362
Sinic::RxStateStrings
const char * RxStateStrings[]
Definition: sinic.cc:50
Packet::getSize
unsigned getSize() const
Definition: packet.hh:765
X86ISA::count
count
Definition: misc.hh:703
Sinic::Base::rxEnable
bool rxEnable
Definition: sinic.hh:50
PacketFifo::end
iterator end()
Definition: pktfifo.hh:113
Sinic::Device::rxDirtyCount
int rxDirtyCount
Definition: sinic.hh:163
Stats::reset
void reset()
Definition: statistics.cc:299
Event::when
Tick when() const
Get the time that the event is scheduled.
Definition: eventq.hh:505
Sinic::Device::txKickTick
Tick txKickTick
Definition: sinic.hh:196
Sinic::Device::rxDmaAddr
Addr rxDmaAddr
Definition: sinic.hh:175
X86ISA::reg
Bitfield< 5, 3 > reg
Definition: types.hh:88
Sinic::Device::devIntrChangeMask
void devIntrChangeMask(uint32_t newmask)
Definition: sinic.cc:447
Sinic::Device::rxCopyDone
@ rxCopyDone
Definition: sinic.hh:92
Sinic::Base::cpuPendingIntr
bool cpuPendingIntr
Definition: sinic.hh:57
Sinic::Device::txCopy
@ txCopy
Definition: sinic.hh:100
packet.hh
EventFunctionWrapper
Definition: eventq.hh:1112
Net::UdpHdr::sum
uint16_t sum() const
Definition: inet.hh:739
Sinic::Device::DeviceStats::numVnicDistance
Stats::Scalar numVnicDistance
Definition: sinic.hh:278
Net
Definition: inet.cc:53
Sinic::Device::transferDone
void transferDone()
Definition: sinic.cc:1115
stats.hh
Stats::Group::resetStats
virtual void resetStats()
Callback to reset stats.
Definition: group.cc:81
Sinic::Base::interface
Interface * interface
Definition: sinic.hh:63
Sinic::Regs::VirtualShift
static const int VirtualShift
Definition: sinicreg.hh:58
Sinic::Regs::Info::read
bool read
Definition: sinicreg.hh:174
Drainable::drainResume
virtual void drainResume()
Resume execution after a successful drain.
Definition: drain.hh:289
DmaDevice::dmaRead
void dmaRead(Addr addr, int size, Event *event, uint8_t *data, uint32_t sid, uint32_t ssid, Tick delay=0)
Definition: dma_device.hh:225
Sinic::Device::VirtualReg::rxPacketBytes
unsigned rxPacketBytes
Definition: sinic.hh:140
Sinic::Device::txDmaAddr
Addr txDmaAddr
Definition: sinic.hh:185
cp
Definition: cprintf.cc:37
Sinic::Device::txFull
bool txFull
Definition: sinic.hh:181
Sinic::Device::prepareIO
void prepareIO(ContextID cpu, int index)
Definition: sinic.cc:131
EventManager::schedule
void schedule(Event &event, Tick when)
Definition: eventq.hh:1016
Sinic::Base::txEnable
bool txEnable
Definition: sinic.hh:51
DmaDevice::dmaWrite
void dmaWrite(Addr addr, int size, Event *event, uint8_t *data, uint32_t sid, uint32_t ssid, Tick delay=0)
Definition: dma_device.hh:211
Sinic::Base::intrDelay
Tick intrDelay
Definition: sinic.hh:54
Sinic::Device::VirtualReg::rxDoneData
uint64_t rxDoneData
Definition: sinic.hh:141
Sinic::Device::VirtualReg::rxPacketOffset
unsigned rxPacketOffset
Definition: sinic.hh:139
EtherDevice::EtherDeviceStats::txTcpChecksums
Stats::Scalar txTcpChecksums
Definition: etherdevice.hh:73
Sinic::Device::devIntrClear
void devIntrClear(uint32_t interrupts=Regs::Intr_All)
Definition: sinic.cc:431
PacketFifo::size
unsigned size() const
Definition: pktfifo.hh:98
DPRINTF
#define DPRINTF(x,...)
Definition: trace.hh:237
Sinic::Device::txDmaLen
int txDmaLen
Definition: sinic.hh:187
ADD_STAT
#define ADD_STAT(n,...)
Convenience macro to add a stat to a statistics group.
Definition: group.hh:71
Sinic::Device::resetStats
void resetStats() override
Callback to reset stats.
Definition: sinic.cc:114
Net::IpPtr
Definition: inet.hh:351
PciDevice::serialize
void serialize(CheckpointOut &cp) const override
Serialize this object to the given output stream.
Definition: device.cc:399
Sinic::Device::txUnique
Counter txUnique
Definition: sinic.hh:154
Sinic::Device::rxBusyCount
int rxBusyCount
Definition: sinic.hh:161
Sinic::Base::serialize
void serialize(CheckpointOut &cp) const override
Serialization stuff.
Definition: sinic.cc:1192
Port
Ports are used to interface objects to each other.
Definition: port.hh:56
Sinic
Definition: sinic.cc:48
Sinic::TxStateStrings
const char * TxStateStrings[]
Definition: sinic.cc:59
Sinic::Base
Definition: sinic.hh:47
Sinic::Device::RxState
RxState
Receive State Machine States.
Definition: sinic.hh:87
Sinic::Regs::Info::size
uint8_t size
Definition: sinicreg.hh:173
EtherDevice::EtherDeviceStats::txPackets
Stats::Scalar txPackets
Definition: etherdevice.hh:64
Sinic::regInfo
const Regs::Info & regInfo(Addr daddr)
Definition: sinicreg.hh:182
Clocked::clockEdge
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...
Definition: clocked_object.hh:174
DmaDevice::getPort
Port & getPort(const std::string &if_name, PortID idx=InvalidPortID) override
Get a port with a given name and index.
Definition: dma_device.cc:360
Sinic::Base::cpuIntrPending
bool cpuIntrPending() const
Definition: sinic.cc:549
EtherDevice::EtherDeviceStats::txIpChecksums
Stats::Scalar txIpChecksums
Definition: etherdevice.hh:70
Sinic::Device::rxKick
void rxKick()
Definition: sinic.cc:681
debug.hh
Sinic::Device::txPacketOffset
int txPacketOffset
Definition: sinic.hh:183
Sinic::Device::txBeginCopy
@ txBeginCopy
Definition: sinic.hh:99
PciDevice::unserialize
void unserialize(CheckpointIn &cp) override
Reconstruct the state of this object from a checkpoint.
Definition: device.cc:462
Sinic::Device::getPort
Port & getPort(const std::string &if_name, PortID idx=InvalidPortID) override
Get a port with a given name and index.
Definition: sinic.cc:122
Sinic::Base::intrTick
Tick intrTick
Definition: sinic.hh:55
compiler.hh
PacketFifo::check
void check() const
Definition: pktfifo.hh:192
Sinic::Device::txEventTransmit
void txEventTransmit()
Definition: sinic.hh:202
UNIT_COUNT
#define UNIT_COUNT
Definition: units.hh:49
Sinic::Device::DeviceStats::totalVnicDistance
Stats::Scalar totalVnicDistance
Definition: sinic.hh:277
Sinic::Device::devIntrPost
void devIntrPost(uint32_t interrupts)
Interrupt management.
Definition: sinic.cc:395
Sinic::Regs::Info::name
const char * name
Definition: sinicreg.hh:176
Net::cksum
uint16_t cksum(const IpPtr &ptr)
Definition: inet.cc:203
Addr
uint64_t Addr
Address type This will probably be moved somewhere else in the near future.
Definition: types.hh:148
Sinic::Device::TxState
TxState
Transmit State Machine states.
Definition: sinic.hh:96
Sinic::Device::prepareWrite
void prepareWrite(ContextID cpu, int index)
Definition: sinic.cc:185
PacketFifo::iterator
fifo_list::iterator iterator
Definition: pktfifo.hh:81
Sinic::Device::rxDmaDone
void rxDmaDone()
DMA parameters.
Definition: sinic.cc:665
Sinic::Device::VirtualReg::rxUnique
Counter rxUnique
Definition: sinic.hh:143
Sinic::Device::prepareRead
void prepareRead(ContextID cpu, int index)
Definition: sinic.cc:144
name
const std::string & name()
Definition: trace.cc:48
SERIALIZE_SCALAR
#define SERIALIZE_SCALAR(scalar)
Definition: serialize.hh:584
Sinic::Device::transmit
void transmit()
Retransmit event.
Definition: sinic.cc:951
paramOut
void paramOut(CheckpointOut &cp, const std::string &name, ExtMachInst const &machInst)
Definition: types.cc:37
DDUMP
#define DDUMP(x, data, count)
DPRINTF is a debugging trace facility that allows one to selectively enable tracing statements.
Definition: trace.hh:236
Sinic::Base::cpuIntrEnable
bool cpuIntrEnable
Definition: sinic.hh:56
packet_access.hh
Sinic::Regs::Info::write
bool write
Definition: sinicreg.hh:175
Sinic::regValid
bool regValid(Addr daddr)
Definition: sinicreg.hh:224
Sinic::Device::VirtualReg::RxDone
uint64_t RxDone
Definition: sinic.hh:134
Sinic::Device::sinicDeviceStats
Sinic::Device::DeviceStats sinicDeviceStats
Sinic::Device::rxCopy
@ rxCopy
Definition: sinic.hh:91
Drainable::drainState
DrainState drainState() const
Return the current drain state of an object.
Definition: drain.hh:320
Sinic::Device::txList
VirtualList txList
Definition: sinic.hh:159
Sinic::Base::intrEvent
EventFunctionWrapper * intrEvent
Definition: sinic.hh:62
SimObject::name
virtual const std::string name() const
Definition: sim_object.hh:182
EtherDevice::EtherDeviceStats::rxPackets
Stats::Scalar rxPackets
Definition: etherdevice.hh:65
Sinic::Device::reset
void reset()
Definition: sinic.cc:595
UNIT_RATE
#define UNIT_RATE(T1, T2)
Definition: units.hh:47
Sinic::Device::rxFilter
bool rxFilter(const EthPacketPtr &packet)
receive address filter
Definition: sinic.cc:1128
Sinic::Device::regData64
uint64_t & regData64(Addr daddr)
Definition: sinic.hh:167
EthPacketPtr
std::shared_ptr< EthPacketData > EthPacketPtr
Definition: etherpkt.hh:87
Packet::getLE
T getLE() const
Get the data in the packet byte swapped from little endian to host endian.
Definition: packet_access.hh:75
PacketFifo::packets
unsigned packets() const
Definition: pktfifo.hh:96
EtherDevice::EtherDeviceStats::rxTcpChecksums
Stats::Scalar rxTcpChecksums
Definition: etherdevice.hh:74
PCI_CMD_MSE
#define PCI_CMD_MSE
Definition: pcireg.h:116
Sinic::Device::VirtualReg::TxDone
uint64_t TxDone
Definition: sinic.hh:136
Sinic::Device::recvPacket
bool recvPacket(EthPacketPtr packet)
device ethernet interface
Definition: sinic.cc:1139
Sinic::Device::rxUnique
Counter rxUnique
Definition: sinic.hh:153
EtherDevice::EtherDeviceStats::txBytes
Stats::Scalar txBytes
Definition: etherdevice.hh:61
PacketFifo::empty
bool empty() const
Definition: pktfifo.hh:101
Sinic::Device::txDmaEvent
EventFunctionWrapper txDmaEvent
Definition: sinic.hh:241
types.hh
Sinic::Device::regData32
uint32_t & regData32(Addr daddr)
Definition: sinic.hh:166
Sinic::Device::txCopyDone
@ txCopyDone
Definition: sinic.hh:101
Packet
A Packet is used to encapsulate a transfer between two objects in the memory system (e....
Definition: packet.hh:258
PacketFifo::avail
unsigned avail() const
Definition: pktfifo.hh:100
Stats::Group
Statistics container.
Definition: group.hh:87
Sinic::Base::unserialize
void unserialize(CheckpointIn &cp) override
Unserialize an object.
Definition: sinic.cc:1213
Sinic::Device::command
void command(uint32_t command)
Definition: sinic.cc:585
iGbReg::TxdOp::ip
bool ip(TxDesc *d)
Definition: i8254xGBe_defs.hh:261
Sinic::Device::VirtualReg::TxData
uint64_t TxData
Definition: sinic.hh:135
Sinic::Device::rxBusy
VirtualList rxBusy
Definition: sinic.hh:157
PacketFifo::clear
void clear()
Definition: pktfifo.hh:149
Sinic::Base::cpuIntrClear
void cpuIntrClear()
Definition: sinic.cc:530
Event::squash
void squash()
Squash the current event.
Definition: eventq.hh:469
Cycles
Cycles is a wrapper class for representing cycle counts, i.e.
Definition: types.hh:79
Packet::setLE
void setLE(T v)
Set the value in the data pointer to v as little endian.
Definition: packet_access.hh:105
Sinic::Device::txDmaData
uint8_t * txDmaData
Definition: sinic.hh:186
PciDevice::pioDelay
Tick pioDelay
Definition: device.hh:351
CheckpointOut
std::ostream CheckpointOut
Definition: serialize.hh:64
Sinic::Device::rxBeginCopy
@ rxBeginCopy
Definition: sinic.hh:90
Sinic::Device::rxLow
bool rxLow
Definition: sinic.hh:174
EtherInt::sendPacket
bool sendPacket(EthPacketPtr packet)
Definition: etherint.hh:70
Sinic::Device::rxFifoBlock
@ rxFifoBlock
Definition: sinic.hh:89
PciDevice::intrPost
void intrPost()
Definition: device.hh:361
Stats
Definition: statistics.cc:53
Sinic::Device::VirtualReg
Definition: sinic.hh:132
Sinic::Device::VirtualReg::RxData
uint64_t RxData
Definition: sinic.hh:133
Sinic::Device::rxKickTick
Tick rxKickTick
Definition: sinic.hh:193
curTick
Tick curTick()
The universal simulation clock.
Definition: cur_tick.hh:43
EtherDevBase
Dummy class to keep the Python class hierarchy in sync with the C++ object hierarchy.
Definition: etherdevice.hh:140
Sinic::Device::regs
struct Sinic::Device::@90 regs
device register file
SimObject::params
const Params & params() const
Definition: sim_object.hh:168
Sinic::Device::rxList
VirtualList rxList
Definition: sinic.hh:156
Sinic::Device::virtualRegs
VirtualRegs virtualRegs
Definition: sinic.hh:155
PacketFifo::front
EthPacketPtr front()
Definition: pktfifo.hh:118
MipsISA::p
Bitfield< 0 > p
Definition: pra_constants.hh:323
Sinic::Device::serialize
void serialize(CheckpointOut &cp) const override
Serialization stuff.
Definition: sinic.cc:1237
Sinic::Device::txIdle
@ txIdle
Definition: sinic.hh:97
inet.hh
EtherDevice::etherDeviceStats
EtherDevice::EtherDeviceStats etherDeviceStats
paramIn
void paramIn(CheckpointIn &cp, const std::string &name, ExtMachInst &machInst)
Definition: types.cc:69
Sinic::Device::~Device
~Device()
Definition: sinic.cc:96
Sinic::Base::cpuInterrupt
void cpuInterrupt()
Definition: sinic.cc:507
CheckpointIn
Definition: serialize.hh:68
ResponsePort::sendRangeChange
void sendRangeChange() const
Called by the owner to send a range change.
Definition: port.hh:293
DmaDevice::dmaPending
bool dmaPending() const
Definition: dma_device.hh:238
csprintf
std::string csprintf(const char *format, const Args &...args)
Definition: cprintf.hh:158
Sinic::Device::DeviceStats::_maxVnicDistance
int _maxVnicDistance
Definition: sinic.hh:282
Sinic::Device::Device
Device(const Params &p)
Definition: sinic.cc:80
DmaDevice::Params
DmaDeviceParams Params
Definition: dma_device.hh:206
PacketFifo::countPacketsAfter
int countPacketsAfter(const_iterator i) const
Definition: pktfifo.hh:184
Sinic::Device::rxFifoPtr
PacketFifo::iterator rxFifoPtr
Definition: sinic.hh:172
PciDevice::pciToDma
Addr pciToDma(Addr pci_addr) const
Definition: device.hh:356
panic
#define panic(...)
This implements a cprintf based panic() function.
Definition: logging.hh:171
PioDevice::pioPort
PioPort< PioDevice > pioPort
The pioPort that handles the requests for us and provides us requests that it sees.
Definition: io_device.hh:106
Sinic::Device::txState
TxState txState
Definition: sinic.hh:179
EtherDevice::EtherDeviceStats::txUdpChecksums
Stats::Scalar txUdpChecksums
Definition: etherdevice.hh:76
PacketFifo::begin
iterator begin()
Definition: pktfifo.hh:112
eventq.hh

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