NetBSD/sys/dev/marvell/if_mvxpevar.h
hsuenaga ea5bc4c33e separate buffer management codes 'mvxpbm.c' from if_mvxpe.c.
the buffer management(ex. fill the rx descriptors/buffers) is done by H/W in
ARMADA XP/380, and is done by S/W in ARMADA 370. the H/W BM support is not yet
implemented, so all devices use the S/W management mode at this time.
2015-06-03 03:55:47 +00:00

508 lines
13 KiB
C

/* $NetBSD: if_mvxpevar.h,v 1.2 2015/06/03 03:55:47 hsuenaga Exp $ */
/*
* Copyright (c) 2015 Internet Initiative Japan Inc.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. 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.
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``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 AUTHOR 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.
*/
#ifndef _IF_MVXPEVAR_H_
#define _IF_MVXPEVAR_H_
#include <net/if.h>
#include <dev/marvell/mvxpbmvar.h>
/*
* Limit of packet sizes.
*/
#define MVXPE_HWHEADER_SIZE 2 /* Marvell Header */
#define MVXPE_MRU 2000 /* Max Receive Unit */
#define MVXPE_MTU MVXPE_MRU /* Max Transmit Unit */
/*
* Default limit of queue length
*
* queue 0 is lowest priority and queue 7 is highest priority.
* IP packet is received on queue 7 by default.
*
* XXX: packet classifier is not implement yet
*/
#define MVXPE_RX_QUEUE_LIMIT_0 8
#define MVXPE_RX_QUEUE_LIMIT_1 8
#define MVXPE_RX_QUEUE_LIMIT_2 8
#define MVXPE_RX_QUEUE_LIMIT_3 8
#define MVXPE_RX_QUEUE_LIMIT_4 8
#define MVXPE_RX_QUEUE_LIMIT_5 8
#define MVXPE_RX_QUEUE_LIMIT_6 8
#define MVXPE_RX_QUEUE_LIMIT_7 IFQ_MAXLEN
#define MVXPE_TX_QUEUE_LIMIT_0 IFQ_MAXLEN
#define MVXPE_TX_QUEUE_LIMIT_1 8
#define MVXPE_TX_QUEUE_LIMIT_2 8
#define MVXPE_TX_QUEUE_LIMIT_3 8
#define MVXPE_TX_QUEUE_LIMIT_4 8
#define MVXPE_TX_QUEUE_LIMIT_5 8
#define MVXPE_TX_QUEUE_LIMIT_6 8
#define MVXPE_TX_QUEUE_LIMIT_7 8
/* interrupt is triggered when corossing (queuelen / RATIO) */
#define MVXPE_RXTH_RATIO 8
#define MVXPE_RXTH_REFILL_RATIO 2
#define MVXPE_TXTH_RATIO 8
/*
* Device Register access
*/
#define MVXPE_READ(sc, reg) \
bus_space_read_4((sc)->sc_iot, (sc)->sc_ioh, (reg))
#define MVXPE_WRITE(sc, reg, val) \
bus_space_write_4((sc)->sc_iot, (sc)->sc_ioh, (reg), (val))
#define MVXPE_READ_REGION(sc, reg, val, c) \
bus_space_read_region_4((sc)->sc_iot, (sc)->sc_ioh, (reg), (val), (c))
#define MVXPE_WRITE_REGION(sc, reg, val, c) \
bus_space_write_region_4((sc)->sc_iot, (sc)->sc_ioh, (reg), (val), (c))
#define MVXPE_READ_MIB(sc, reg) \
bus_space_read_4((sc)->sc_iot, (sc)->sc_mibh, (reg))
#define MVXPE_IS_LINKUP(sc) \
(MVXPE_READ((sc), MVXPE_PSR) & MVXPE_PSR_LINKUP)
#define MVXPE_IS_QUEUE_BUSY(queues, q) \
((((queues) >> (q)) & 0x1))
/*
* EEE: Lower Power Idle config
* Default timer is duration of MTU sized frame transmission.
* The timer can be negotiated by LLDP protocol, but we have no
* support.
*/
#define MVXPE_LPI_TS (MVXPE_MRU * 8 / 1000) /* [us] */
#define MVXPE_LPI_TW (MVXPE_MRU * 8 / 1000) /* [us] */
#define MVXPE_LPI_LI (MVXPE_MRU * 8 / 1000) /* [us] */
/*
* DMA Descriptor
*
* the ethernet device has 8 rx/tx DMA queues. each of queue has its own
* decriptor list. descriptors are simply index by counter inside the device.
*/
#define MVXPE_TX_RING_CNT IFQ_MAXLEN
#define MVXPE_TX_RING_MSK (MVXPE_TX_RING_CNT - 1)
#define MVXPE_TX_RING_NEXT(x) (((x) + 1) & MVXPE_TX_RING_MSK)
#define MVXPE_RX_RING_CNT IFQ_MAXLEN
#define MVXPE_RX_RING_MSK (MVXPE_RX_RING_CNT - 1)
#define MVXPE_RX_RING_NEXT(x) (((x) + 1) & MVXPE_RX_RING_MSK)
#define MVXPE_TX_SEGLIMIT 32
struct mvxpe_rx_ring {
/* Real descriptors array. shared by RxDMA */
struct mvxpe_rx_desc *rx_descriptors;
bus_dmamap_t rx_descriptors_map;
/* Managment entries for each of descritors */
struct mvxpe_rx_handle {
struct mvxpe_rx_desc *rxdesc_va;
off_t rxdesc_off; /* from rx_descriptors[0] */
struct mvxpbm_chunk *chunk;
} rx_handle[MVXPE_RX_RING_CNT];
/* locks */
kmutex_t rx_ring_mtx;
/* Index */
int rx_dma;
int rx_cpu;
/* Limit */
int rx_queue_len;
int rx_queue_th_received;
int rx_queue_th_free;
int rx_queue_th_time; /* [Tclk] */
};
struct mvxpe_tx_ring {
/* Real descriptors array. shared by TxDMA */
struct mvxpe_tx_desc *tx_descriptors;
bus_dmamap_t tx_descriptors_map;
/* Managment entries for each of descritors */
struct mvxpe_tx_handle {
struct mvxpe_tx_desc *txdesc_va;
off_t txdesc_off; /* from tx_descriptors[0] */
struct mbuf *txdesc_mbuf;
bus_dmamap_t txdesc_mbuf_map;
} tx_handle[MVXPE_TX_RING_CNT];
/* locks */
kmutex_t tx_ring_mtx;
/* Index */
int tx_used;
int tx_dma;
int tx_cpu;
/* Limit */
int tx_queue_len;
int tx_queue_th_free;
};
static inline int
tx_counter_adv(int ctr, int n)
{
/* XXX: lock or atomic */
ctr += n;
while (ctr >= MVXPE_TX_RING_CNT)
ctr -= MVXPE_TX_RING_CNT;
return ctr;
}
static inline int
rx_counter_adv(int ctr, int n)
{
/* XXX: lock or atomic */
ctr += n;
while (ctr >= MVXPE_TX_RING_CNT)
ctr -= MVXPE_TX_RING_CNT;
return ctr;
}
/*
* Timeout control
*/
#define MVXPE_PHY_TIMEOUT 10000 /* msec */
#define RX_DISABLE_TIMEOUT 0x1000000 /* times */
#define TX_DISABLE_TIMEOUT 0x1000000 /* times */
#define TX_FIFO_EMPTY_TIMEOUT 0x1000000 /* times */
/*
* Event counter
*/
#ifdef MVXPE_EVENT_COUNTERS
#define MVXPE_EVCNT_INCR(ev) (ev)->ev_count++
#define MVXPE_EVCNT_ADD(ev, val) (ev)->ev_count += (val)
#else
#define MVXPE_EVCNT_INCR(ev) /* nothing */
#define MVXPE_EVCNT_ADD(ev, val) /* nothing */
#endif
struct mvxpe_evcnt {
/*
* Master Interrupt Handler
*/
struct evcnt ev_i_rxtxth;
struct evcnt ev_i_rxtx;
struct evcnt ev_i_misc;
/*
* RXTXTH Interrupt
*/
struct evcnt ev_rxtxth_txerr;
/*
* MISC Interrupt
*/
struct evcnt ev_misc_phystatuschng;
struct evcnt ev_misc_linkchange;
struct evcnt ev_misc_iae;
struct evcnt ev_misc_rxoverrun;
struct evcnt ev_misc_rxcrc;
struct evcnt ev_misc_rxlargepacket;
struct evcnt ev_misc_txunderrun;
struct evcnt ev_misc_prbserr;
struct evcnt ev_misc_srse;
struct evcnt ev_misc_txreq;
/*
* RxTx Interrupt
*/
struct evcnt ev_rxtx_rreq;
struct evcnt ev_rxtx_rpq;
struct evcnt ev_rxtx_tbrq;
struct evcnt ev_rxtx_rxtxth;
struct evcnt ev_rxtx_txerr;
struct evcnt ev_rxtx_misc;
/*
* Link
*/
struct evcnt ev_link_up;
struct evcnt ev_link_down;
/*
* Rx Descriptor
*/
struct evcnt ev_rxd_ce;
struct evcnt ev_rxd_or;
struct evcnt ev_rxd_mf;
struct evcnt ev_rxd_re;
struct evcnt ev_rxd_scat;
/*
* Tx Descriptor
*/
struct evcnt ev_txd_lc;
struct evcnt ev_txd_ur;
struct evcnt ev_txd_rl;
struct evcnt ev_txd_oth;
/*
* Status Registers
*/
struct evcnt ev_reg_pdfc; /* Rx Port Discard Frame Counter */
struct evcnt ev_reg_pofc; /* Rx Port Overrun Frame Counter */
struct evcnt ev_reg_txbadfcs; /* Tx BAD FCS Counter */
struct evcnt ev_reg_txdropped; /* Tx Dropped Counter */
struct evcnt ev_reg_lpic;
/* Device Driver Errors */
struct evcnt ev_drv_wdogsoft;
struct evcnt ev_drv_txerr;
struct evcnt ev_drv_rxq[MVXPE_QUEUE_SIZE];
struct evcnt ev_drv_rxqe[MVXPE_QUEUE_SIZE];
struct evcnt ev_drv_txq[MVXPE_QUEUE_SIZE];
struct evcnt ev_drv_txqe[MVXPE_QUEUE_SIZE];
};
/*
* Debug
*/
#ifdef MVXPE_DEBUG
#define DPRINTF(fmt, ...) \
do { \
if (mvxpe_debug >= 1) { \
printf("%s: ", __func__); \
printf((fmt), ##__VA_ARGS__); \
} \
} while (/*CONSTCOND*/0)
#define DPRINTFN(level , fmt, ...) \
do { \
if (mvxpe_debug >= (level)) { \
printf("%s: ", __func__); \
printf((fmt), ##__VA_ARGS__); \
} \
} while (/*CONSTCOND*/0)
#define DPRINTDEV(dev, level, fmt, ...) \
do { \
if (mvxpe_debug >= (level)) { \
device_printf((dev), \
"%s: "fmt , __func__, ##__VA_ARGS__); \
} \
} while (/*CONSTCOND*/0)
#define DPRINTSC(sc, level, fmt, ...) \
do { \
device_t dev = (sc)->sc_dev; \
if (mvxpe_debug >= (level)) { \
device_printf(dev, \
"%s: " fmt, __func__, ##__VA_ARGS__); \
} \
} while (/*CONSTCOND*/0)
#define DPRINTIFNET(ifp, level, fmt, ...) \
do { \
const char *xname = (ifp)->if_xname; \
if (mvxpe_debug >= (level)) { \
printf("%s: %s: " fmt, xname, __func__, ##__VA_ARGS__);\
} \
} while (/*CONSTCOND*/0)
#define DPRINTIFNET(ifp, level, fmt, ...) \
do { \
const char *xname = (ifp)->if_xname; \
if (mvxpe_debug >= (level)) { \
printf("%s: %s: " fmt, xname, __func__, ##__VA_ARGS__);\
} \
} while (/*CONSTCOND*/0)
#define DPRINTPRXS(level, q) \
do { \
uint32_t _reg = MVXPE_READ(sc, MVXPE_PRXS(q)); \
if (mvxpe_debug >= (level)) { \
printf("PRXS(queue %d) %#x: Occupied %d, NoOccupied %d.\n", \
q, _reg, MVXPE_PRXS_GET_ODC(_reg), \
MVXPE_PRXS_GET_NODC(_reg)); \
} \
} while (/*CONSTCOND*/0)
#else
#define DPRINTF(fmt, ...)
#define DPRINTFN(level, fmt, ...)
#define DPRINTDEV(dev, level, fmt, ...)
#define DPRINTSC(sc, level, fmt, ...)
#define DPRINTIFNET(ifp, level, fmt, ...)
#define DPRINTPRXS(level, reg)
#endif
#define KASSERT_SC_MTX(sc) \
KASSERT(mutex_owned(&(sc)->sc_mtx))
#define KASSERT_BM_MTX(sc) \
KASSERT(mutex_owned(&(sc)->sc_bm.bm_mtx))
#define KASSERT_RX_MTX(sc, q) \
KASSERT(mutex_owned(&(sc)->sc_rx_ring[(q)].rx_ring_mtx))
#define KASSERT_TX_MTX(sc, q) \
KASSERT(mutex_owned(&(sc)->sc_tx_ring[(q)].tx_ring_mtx))
/*
* Configuration parameters
*/
struct mvxpe_conf {
int cf_lpi; /* EEE Low Power IDLE enable */
int cf_fc; /* Flow Control enable */
};
/*
* sysctl(9) parameters
*/
struct mvxpe_softc;
struct mvxpe_sysctl_queue {
struct mvxpe_softc *sc;
int rxtx;
int queue;
};
#define MVXPE_SYSCTL_RX 0
#define MVXPE_SYSCTL_TX 1
struct mvxpe_sysctl_mib {
struct mvxpe_softc *sc;
int index;
uint64_t counter;
};
/*
* Ethernet Device main context
*/
struct mvxpe_softc {
device_t sc_dev;
int sc_port;
uint32_t sc_version;
/*
* sc_mtx must be held by interface functions to/from
* other frameworks. interrupt hander, sysctl hander,
* ioctl hander, and so on.
*/
kmutex_t sc_mtx;
/*
* Ethernet facilities
*/
struct ethercom sc_ethercom;
struct mii_data sc_mii;
u_int8_t sc_enaddr[ETHER_ADDR_LEN]; /* station addr */
int sc_if_flags;
int sc_wdogsoft;
/*
* Configuration Parameters
*/
struct mvxpe_conf sc_cf;
/*
* I/O Spaces
*/
bus_space_tag_t sc_iot;
bus_space_handle_t sc_ioh; /* all registers handle */
bus_space_handle_t sc_mibh; /* mib counter handle */
/*
* DMA Spaces
*/
bus_dma_tag_t sc_dmat;
struct mvxpe_rx_ring sc_rx_ring[MVXPE_QUEUE_SIZE];
struct mvxpe_tx_ring sc_tx_ring[MVXPE_QUEUE_SIZE];
int sc_tx_pending; /* total number of tx pkt */
/*
* Software Buffer Manager
*/
struct mvxpbm_softc *sc_bm;
/*
* Maintance clock
*/
callout_t sc_tick_ch; /* tick callout */
/*
* Link State control
*/
uint32_t sc_linkstate;
/*
* Act as Rndom source
*/
krndsource_t sc_rnd_source;
/*
* Sysctl interfaces
*/
struct sysctllog *sc_mvxpe_clog;
struct mvxpe_sysctl_queue sc_sysctl_rx_queue[MVXPE_QUEUE_SIZE];
struct mvxpe_sysctl_queue sc_sysctl_tx_queue[MVXPE_QUEUE_SIZE];
/*
* MIB counter
*/
size_t sc_sysctl_mib_size;
struct mvxpe_sysctl_mib *sc_sysctl_mib;
#ifdef MVXPE_EVENT_COUNTERS
/*
* Event counter
*/
struct mvxpe_evcnt sc_ev;
#endif
};
#define MVXPE_RX_RING_MEM_VA(sc, q) \
((sc)->sc_rx_ring[(q)].rx_descriptors)
#define MVXPE_RX_RING_MEM_PA(sc, q) \
((sc)->sc_rx_ring[(q)].rx_descriptors_map->dm_segs[0].ds_addr)
#define MVXPE_RX_RING_MEM_MAP(sc, q) \
((sc)->sc_rx_ring[(q)].rx_descriptors_map)
#define MVXPE_RX_RING(sc, q) \
(&(sc)->sc_rx_ring[(q)])
#define MVXPE_RX_HANDLE(sc, q, i) \
(&(sc)->sc_rx_ring[(q)].rx_handle[(i)])
#define MVXPE_RX_DESC(sc, q, i) \
((sc)->sc_rx_ring[(q)].rx_handle[(i)].rxdesc_va)
#define MVXPE_RX_DESC_OFF(sc, q, i) \
((sc)->sc_rx_ring[(q)].rx_handle[(i)].rxdesc_off)
#define MVXPE_RX_PKTBUF(sc, q, i) \
((sc)->sc_rx_ring[(q)].rx_handle[(i)].chunk)
#define MVXPE_TX_RING_MEM_VA(sc, q) \
((sc)->sc_tx_ring[(q)].tx_descriptors)
#define MVXPE_TX_RING_MEM_PA(sc, q) \
((sc)->sc_tx_ring[(q)].tx_descriptors_map->dm_segs[0].ds_addr)
#define MVXPE_TX_RING_MEM_MAP(sc, q) \
((sc)->sc_tx_ring[(q)].tx_descriptors_map)
#define MVXPE_TX_RING(sc, q) \
(&(sc)->sc_tx_ring[(q)])
#define MVXPE_TX_HANDLE(sc, q, i) \
(&(sc)->sc_tx_ring[(q)].tx_handle[(i)])
#define MVXPE_TX_DESC(sc, q, i) \
((sc)->sc_tx_ring[(q)].tx_handle[(i)].txdesc_va)
#define MVXPE_TX_DESC_OFF(sc, q, i) \
((sc)->sc_tx_ring[(q)].tx_handle[(i)].txdesc_off)
#define MVXPE_TX_MBUF(sc, q, i) \
((sc)->sc_tx_ring[(q)].tx_handle[(i)].txdesc_mbuf)
#define MVXPE_TX_MAP(sc, q, i) \
((sc)->sc_tx_ring[(q)].tx_handle[(i)].txdesc_mbuf_map)
#endif /* _IF_MVXPEVAR_H_ */