e05dbb5116
base/traceflags.py: added some more traceflags for ethernet to break it up better dev/etherpkt.hh: since we are not network host order, must reverse bytes for these typechecks. also, overload isTcp/UdpPkt to take an argument so you don't have to reget the ip header if you've already got one. dev/ns_gige.cc: 1) add some functions that reverse Endianness so we can generate adn evaluate checksum adn dprintf data accurately/more understandably 2) forget about the implementation of fifo fill/drain thresholds, it's not used by the driver much, nor does it matter with use sending/receiving in whole packets anyway. get rid of teh associated variables. 3) get rid of txFifoCnt the variable, it's redundant and unnecessary, just use txFifoAvail. 4) change io_enable to ioEnable, just to be picky. 5) modify some DPRINTF's to be clearer, also added a lot, and spread them into better traceflag categories 6) fix the device bug! it's the intrTick = 0 at teh beginning of cpuInterrupt(). 7) clear some bools in regsReset() so they don't holdover wrong state 8) fix pseudo header generation for Tcp checksumming to account for network order dev/ns_gige.hh: change io_enable to ioEnable, get rid of fill/drain thresh related variables and txFifoCnt, which is redundant --HG-- extra : convert_revision : c538b75731f3c9e04354f57e6df9a40aeca5096d
397 lines
10 KiB
C++
397 lines
10 KiB
C++
/*
|
|
* Copyright (c) 2004 The Regents of The University of Michigan
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions are
|
|
* met: redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer;
|
|
* redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution;
|
|
* neither the name of the copyright holders nor the names of its
|
|
* contributors may be used to endorse or promote products derived from
|
|
* this software without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
/* @file
|
|
* Device module for modelling the National Semiconductor
|
|
* DP83820 ethernet controller
|
|
*/
|
|
|
|
#ifndef __NS_GIGE_HH__
|
|
#define __NS_GIGE_HH__
|
|
|
|
//#include "base/range.hh"
|
|
#include "dev/etherint.hh"
|
|
#include "dev/etherpkt.hh"
|
|
#include "sim/eventq.hh"
|
|
#include "dev/ns_gige_reg.h"
|
|
#include "base/statistics.hh"
|
|
#include "dev/pcidev.hh"
|
|
#include "dev/tsunami.hh"
|
|
#include "dev/io_device.hh"
|
|
#include "mem/bus/bus.hh"
|
|
|
|
/** defined by the NS83820 data sheet */
|
|
#define MAX_TX_FIFO_SIZE 8192
|
|
#define MAX_RX_FIFO_SIZE 32768
|
|
|
|
/** length of ethernet address in bytes */
|
|
#define EADDR_LEN 6
|
|
|
|
/**
|
|
* Ethernet device registers
|
|
*/
|
|
struct dp_regs {
|
|
uint32_t command;
|
|
uint32_t config;
|
|
uint32_t mear;
|
|
uint32_t ptscr;
|
|
uint32_t isr;
|
|
uint32_t imr;
|
|
uint32_t ier;
|
|
uint32_t ihr;
|
|
uint32_t txdp;
|
|
uint32_t txdp_hi;
|
|
uint32_t txcfg;
|
|
uint32_t gpior;
|
|
uint32_t rxdp;
|
|
uint32_t rxdp_hi;
|
|
uint32_t rxcfg;
|
|
uint32_t pqcr;
|
|
uint32_t wcsr;
|
|
uint32_t pcr;
|
|
uint32_t rfcr;
|
|
uint32_t rfdr;
|
|
uint32_t srr;
|
|
uint32_t mibc;
|
|
uint32_t vrcr;
|
|
uint32_t vtcr;
|
|
uint32_t vdr;
|
|
uint32_t ccsr;
|
|
uint32_t tbicr;
|
|
uint32_t tbisr;
|
|
uint32_t tanar;
|
|
uint32_t tanlpar;
|
|
uint32_t taner;
|
|
uint32_t tesr;
|
|
};
|
|
|
|
struct dp_rom {
|
|
/** for perfect match memory. the linux driver doesn't use any other ROM */
|
|
uint8_t perfectMatch[EADDR_LEN];
|
|
};
|
|
|
|
class IntrControl;
|
|
class NSGigEInt;
|
|
class PhysicalMemory;
|
|
class BaseInterface;
|
|
class HierParams;
|
|
class Bus;
|
|
class PciConfigAll;
|
|
|
|
/**
|
|
* NS DP82830 Ethernet device model
|
|
*/
|
|
class NSGigE : public PciDev
|
|
{
|
|
public:
|
|
/** Transmit State Machine states */
|
|
enum TxState
|
|
{
|
|
txIdle,
|
|
txDescRefr,
|
|
txDescRead,
|
|
txFifoBlock,
|
|
txFragRead,
|
|
txDescWrite,
|
|
txAdvance
|
|
};
|
|
|
|
/** Receive State Machine States */
|
|
enum RxState
|
|
{
|
|
rxIdle,
|
|
rxDescRefr,
|
|
rxDescRead,
|
|
rxFifoBlock,
|
|
rxFragWrite,
|
|
rxDescWrite,
|
|
rxAdvance
|
|
};
|
|
|
|
enum DmaState
|
|
{
|
|
dmaIdle,
|
|
dmaReading,
|
|
dmaWriting,
|
|
dmaReadWaiting,
|
|
dmaWriteWaiting
|
|
};
|
|
|
|
private:
|
|
/** pointer to the chipset */
|
|
Tsunami *tsunami;
|
|
|
|
private:
|
|
Addr addr;
|
|
static const Addr size = sizeof(dp_regs);
|
|
|
|
protected:
|
|
typedef std::deque<PacketPtr> pktbuf_t;
|
|
typedef pktbuf_t::iterator pktiter_t;
|
|
|
|
/** device register file */
|
|
dp_regs regs;
|
|
dp_rom rom;
|
|
|
|
/** pci settings */
|
|
bool ioEnable;
|
|
#if 0
|
|
bool memEnable;
|
|
bool bmEnable;
|
|
#endif
|
|
|
|
/*** BASIC STRUCTURES FOR TX/RX ***/
|
|
/* Data FIFOs */
|
|
pktbuf_t txFifo;
|
|
pktbuf_t rxFifo;
|
|
|
|
/** various helper vars */
|
|
PacketPtr txPacket;
|
|
PacketPtr rxPacket;
|
|
uint8_t *txPacketBufPtr;
|
|
uint8_t *rxPacketBufPtr;
|
|
uint32_t txXferLen;
|
|
uint32_t rxXferLen;
|
|
bool rxDmaFree;
|
|
bool txDmaFree;
|
|
|
|
/** DescCaches */
|
|
ns_desc txDescCache;
|
|
ns_desc rxDescCache;
|
|
|
|
/* tx State Machine */
|
|
TxState txState;
|
|
/** Current Transmit Descriptor Done */
|
|
bool CTDD;
|
|
/** current amt of free space in txDataFifo in bytes */
|
|
uint32_t txFifoAvail;
|
|
/** halt the tx state machine after next packet */
|
|
bool txHalt;
|
|
/** ptr to the next byte in the current fragment */
|
|
Addr txFragPtr;
|
|
/** count of bytes remaining in the current descriptor */
|
|
uint32_t txDescCnt;
|
|
DmaState txDmaState;
|
|
|
|
/** rx State Machine */
|
|
RxState rxState;
|
|
/** Current Receive Descriptor Done */
|
|
bool CRDD;
|
|
/** num of bytes in the current packet being drained from rxDataFifo */
|
|
uint32_t rxPktBytes;
|
|
/** number of bytes in the rxFifo */
|
|
uint32_t rxFifoCnt;
|
|
/** halt the rx state machine after current packet */
|
|
bool rxHalt;
|
|
/** ptr to the next byte in current fragment */
|
|
Addr rxFragPtr;
|
|
/** count of bytes remaining in the current descriptor */
|
|
uint32_t rxDescCnt;
|
|
DmaState rxDmaState;
|
|
|
|
bool extstsEnable;
|
|
|
|
protected:
|
|
Tick dmaReadDelay;
|
|
Tick dmaWriteDelay;
|
|
|
|
Tick dmaReadFactor;
|
|
Tick dmaWriteFactor;
|
|
|
|
void *rxDmaData;
|
|
Addr rxDmaAddr;
|
|
int rxDmaLen;
|
|
bool doRxDmaRead();
|
|
bool doRxDmaWrite();
|
|
void rxDmaReadCopy();
|
|
void rxDmaWriteCopy();
|
|
|
|
void *txDmaData;
|
|
Addr txDmaAddr;
|
|
int txDmaLen;
|
|
bool doTxDmaRead();
|
|
bool doTxDmaWrite();
|
|
void txDmaReadCopy();
|
|
void txDmaWriteCopy();
|
|
|
|
void rxDmaReadDone();
|
|
friend class EventWrapper<NSGigE, &NSGigE::rxDmaReadDone>;
|
|
EventWrapper<NSGigE, &NSGigE::rxDmaReadDone> rxDmaReadEvent;
|
|
|
|
void rxDmaWriteDone();
|
|
friend class EventWrapper<NSGigE, &NSGigE::rxDmaWriteDone>;
|
|
EventWrapper<NSGigE, &NSGigE::rxDmaWriteDone> rxDmaWriteEvent;
|
|
|
|
void txDmaReadDone();
|
|
friend class EventWrapper<NSGigE, &NSGigE::txDmaReadDone>;
|
|
EventWrapper<NSGigE, &NSGigE::txDmaReadDone> txDmaReadEvent;
|
|
|
|
void txDmaWriteDone();
|
|
friend class EventWrapper<NSGigE, &NSGigE::txDmaWriteDone>;
|
|
EventWrapper<NSGigE, &NSGigE::txDmaWriteDone> txDmaWriteEvent;
|
|
|
|
bool dmaDescFree;
|
|
bool dmaDataFree;
|
|
|
|
|
|
protected:
|
|
Tick txDelay;
|
|
Tick rxDelay;
|
|
|
|
void txReset();
|
|
void rxReset();
|
|
void regsReset();
|
|
|
|
void rxKick();
|
|
Tick rxKickTick;
|
|
typedef EventWrapper<NSGigE, &NSGigE::rxKick> RxKickEvent;
|
|
friend class RxKickEvent;
|
|
|
|
void txKick();
|
|
Tick txKickTick;
|
|
typedef EventWrapper<NSGigE, &NSGigE::txKick> TxKickEvent;
|
|
friend class TxKickEvent;
|
|
|
|
/**
|
|
* Retransmit event
|
|
*/
|
|
void transmit();
|
|
typedef EventWrapper<NSGigE, &NSGigE::transmit> TxEvent;
|
|
friend class TxEvent;
|
|
TxEvent txEvent;
|
|
|
|
void txDump() const;
|
|
void rxDump() const;
|
|
|
|
/**
|
|
* receive address filter
|
|
*/
|
|
bool rxFilterEnable;
|
|
bool rxFilter(PacketPtr packet);
|
|
bool acceptBroadcast;
|
|
bool acceptMulticast;
|
|
bool acceptUnicast;
|
|
bool acceptPerfect;
|
|
bool acceptArp;
|
|
|
|
PhysicalMemory *physmem;
|
|
|
|
/**
|
|
* Interrupt management
|
|
*/
|
|
IntrControl *intctrl;
|
|
void devIntrPost(uint32_t interrupts);
|
|
void devIntrClear(uint32_t interrupts);
|
|
void devIntrChangeMask();
|
|
|
|
Tick intrDelay;
|
|
Tick intrTick;
|
|
bool cpuPendingIntr;
|
|
void cpuIntrPost(Tick when);
|
|
void cpuInterrupt();
|
|
void cpuIntrClear();
|
|
|
|
typedef EventWrapper<NSGigE, &NSGigE::cpuInterrupt> IntrEvent;
|
|
friend class IntrEvent;
|
|
IntrEvent *intrEvent;
|
|
|
|
/**
|
|
* Hardware checksum support
|
|
*/
|
|
bool udpChecksum(PacketPtr packet, bool gen);
|
|
bool tcpChecksum(PacketPtr packet, bool gen);
|
|
bool ipChecksum(PacketPtr packet, bool gen);
|
|
uint16_t checksumCalc(uint16_t *pseudo, uint16_t *buf, uint32_t len);
|
|
|
|
NSGigEInt *interface;
|
|
|
|
public:
|
|
NSGigE(const std::string &name, IntrControl *i, Tick intr_delay,
|
|
PhysicalMemory *pmem, Tick tx_delay, Tick rx_delay,
|
|
MemoryController *mmu, HierParams *hier, Bus *header_bus,
|
|
Bus *payload_bus, Tick pio_latency, bool dma_desc_free,
|
|
bool dma_data_free, Tick dma_read_delay, Tick dma_write_delay,
|
|
Tick dma_read_factor, Tick dma_write_factor, PciConfigAll *cf,
|
|
PciConfigData *cd, Tsunami *t, uint32_t bus, uint32_t dev,
|
|
uint32_t func, bool rx_filter, const int eaddr[6]);
|
|
~NSGigE();
|
|
|
|
virtual void WriteConfig(int offset, int size, uint32_t data);
|
|
virtual void ReadConfig(int offset, int size, uint8_t *data);
|
|
|
|
virtual Fault read(MemReqPtr &req, uint8_t *data);
|
|
virtual Fault write(MemReqPtr &req, const uint8_t *data);
|
|
|
|
bool cpuIntrPending() const;
|
|
void cpuIntrAck() { cpuIntrClear(); }
|
|
|
|
bool recvPacket(PacketPtr packet);
|
|
void transferDone();
|
|
|
|
void setInterface(NSGigEInt *i) { assert(!interface); interface = i; }
|
|
|
|
virtual void serialize(std::ostream &os);
|
|
virtual void unserialize(Checkpoint *cp, const std::string §ion);
|
|
|
|
public:
|
|
void regStats();
|
|
|
|
private:
|
|
Stats::Scalar<> txBytes;
|
|
Stats::Scalar<> rxBytes;
|
|
Stats::Scalar<> txPackets;
|
|
Stats::Scalar<> rxPackets;
|
|
Stats::Formula txBandwidth;
|
|
Stats::Formula rxBandwidth;
|
|
Stats::Formula txPacketRate;
|
|
Stats::Formula rxPacketRate;
|
|
|
|
private:
|
|
Tick pioLatency;
|
|
|
|
public:
|
|
Tick cacheAccess(MemReqPtr &req);
|
|
};
|
|
|
|
/*
|
|
* Ethernet Interface for an Ethernet Device
|
|
*/
|
|
class NSGigEInt : public EtherInt
|
|
{
|
|
private:
|
|
NSGigE *dev;
|
|
|
|
public:
|
|
NSGigEInt(const std::string &name, NSGigE *d)
|
|
: EtherInt(name), dev(d) { dev->setInterface(this); }
|
|
|
|
virtual bool recvPacket(PacketPtr &pkt) { return dev->recvPacket(pkt); }
|
|
virtual void sendDone() { dev->transferDone(); }
|
|
};
|
|
|
|
#endif // __NS_GIGE_HH__
|