blob: bbc84554a9e5e830eaaa3d808e802027b2214a39 [file] [log] [blame]
/*
* Generic Broadcom Home Networking Division (HND) DMA engine SW interface
* This supports the following chips: BCM42xx, 44xx, 47xx .
*
* Copyright (C) 2020, Broadcom.
*
* Unless you and Broadcom execute a separate written software license
* agreement governing use of this software, this software is licensed to you
* under the terms of the GNU General Public License version 2 (the "GPL"),
* available at http://www.broadcom.com/licenses/GPLv2.php, with the
* following added to such license:
*
* As a special exception, the copyright holders of this software give you
* permission to link this software with independent modules, and to copy and
* distribute the resulting executable under terms of your choice, provided that
* you also meet, for each linked independent module, the terms and conditions of
* the license of that module. An independent module is a module which is not
* derived from this software. The special exception does not apply to any
* modifications of the software.
*
*
* <<Broadcom-WL-IPTag/Dual:>>
*/
#ifndef _hnddma_h_
#define _hnddma_h_
#include <typedefs.h>
#include <osl_decl.h>
#include <siutils.h>
#include <sbhnddma.h>
#include <hnd_pktq.h>
#include <hnd_pktpool.h>
#ifndef _hnddma_pub_
#define _hnddma_pub_
typedef const struct hnddma_pub hnddma_t;
#endif /* _hnddma_pub_ */
/* range param for dma_getnexttxp() and dma_txreclaim */
typedef enum txd_range {
HNDDMA_RANGE_ALL = 1,
HNDDMA_RANGE_TRANSMITTED,
HNDDMA_RANGE_TRANSFERED
} txd_range_t;
/* dma parameters id */
enum dma_param_id {
HNDDMA_PID_TX_MULTI_OUTSTD_RD = 0,
HNDDMA_PID_TX_PREFETCH_CTL,
HNDDMA_PID_TX_PREFETCH_THRESH,
HNDDMA_PID_TX_BURSTLEN,
HNDDMA_PID_TX_CHAN_SWITCH,
HNDDMA_PID_RX_PREFETCH_CTL = 0x100,
HNDDMA_PID_RX_PREFETCH_THRESH,
HNDDMA_PID_RX_BURSTLEN,
HNDDMA_PID_BURSTLEN_CAP,
HNDDMA_PID_BURSTLEN_WAR,
HNDDMA_SEP_RX_HDR, /**< SPLITRX related */
HNDDMA_SPLIT_FIFO,
HNDDMA_PID_D11RX_WAR,
HNDDMA_PID_RX_WAIT_CMPL,
HNDDMA_NRXPOST,
HNDDMA_NRXBUFSZ,
HNDDMA_PID_RXCTL_MOW,
HNDDMA_M2M_RXBUF_RAW /* rx buffers are raw buffers, not lbufs/lfrags */
};
#define SPLIT_FIFO_0 1
#define SPLIT_FIFO_1 2
typedef void (*setup_context_t)(void *ctx, void *p, uint8 **desc0, uint16 *len0,
uint8 **desc1, uint16 *len1);
/**
* Exported data structure (read-only)
*/
/* export structure */
struct hnddma_pub {
uint dmastflags; /* dma status flags */
uint dmactrlflags; /**< dma control flags */
/* rx error counters */
uint rxgiants; /**< rx giant frames */
uint rxnobuf; /**< rx out of dma descriptors */
/* tx error counters */
uint txnobuf; /**< tx out of dma descriptors */
uint txnodesc; /**< tx out of dma descriptors running count */
};
/* DMA status flags */
#define BCM_DMA_STF_RX (1u << 0u) /* the channel is RX DMA */
typedef struct dma_common dma_common_t;
typedef struct dma_dd_pool dma_dd_pool_t;
/* Flags for dma_attach_ext function */
#define BCM_DMA_IND_INTF_FLAG 0x00000001 /* set for using INDIRECT DMA INTERFACE */
#define BCM_DMA_DESC_ONLY_FLAG 0x00000002 /* For DMA that posts descriptors only and
* no packets
*/
#define BCM_DMA_CHAN_SWITCH_EN 0x00000008 /* for d11 corerev 64+ to help arbitrate
* btw dma channels.
*/
#define BCM_DMA_ROEXT_SUPPORT 0x00000010 /* for d11 corerev 128+ to support receive
* frame offset >=128B and <= 255B
*/
#define BCM_DMA_RX_ALIGN_8BYTE 0x00000020 /* RXDMA address 8-byte aligned */
#define BCM_DMA_DESC_SHARED_POOL 0x00000100 /* For TX DMA that uses shared desc pool */
#define BCM_DMA_RXP_LIST 0x00000200 /* linked list for RXP instead of array */
typedef int (*rxpkt_error_check_t)(const void* ctx, void* pkt);
extern dma_common_t * dma_common_attach(osl_t *osh, volatile uint32 *indqsel,
volatile uint32 *suspreq, volatile uint32 *flushreq, rxpkt_error_check_t cb, void *ctx);
extern void dma_common_detach(dma_common_t *dmacommon);
extern void dma_common_set_ddpool_ctx(dma_common_t *dmacommon, void *desc_pool);
extern void * dma_common_get_ddpool_ctx(dma_common_t *dmacommon, void **va);
extern bool dma_check_last_desc(hnddma_t *dmah);
extern void dma_txfrwd(hnddma_t *dmah);
#ifdef BCM_DMA_INDIRECT
/* Use indirect registers for non-ctmode */
#define DMA_INDQSEL_IA (1 << 31)
extern void dma_set_indqsel(hnddma_t *di, bool force);
extern bool dma_is_indirect(hnddma_t *dmah);
#else
#define dma_set_indqsel(a, b)
#define dma_is_indirect(a) FALSE
#endif /* #ifdef BCM_DMA_INDIRECT */
extern hnddma_t * dma_attach_ext(dma_common_t *dmac, osl_t *osh, const char *name, si_t *sih,
volatile void *dmaregstx, volatile void *dmaregsrx, uint32 flags, uint8 qnum,
uint ntxd, uint nrxd, uint rxbufsize, int rxextheadroom, uint nrxpost, uint rxoffset,
uint *msg_level, uint coreunit);
extern hnddma_t * dma_attach(osl_t *osh, const char *name, si_t *sih,
volatile void *dmaregstx, volatile void *dmaregsrx,
uint ntxd, uint nrxd, uint rxbufsize, int rxextheadroom, uint nrxpost,
uint rxoffset, uint *msg_level);
void dma_rx_desc_init(hnddma_t *dmah, uint rxfifo);
void dma_detach(hnddma_t *dmah);
bool dma_txreset(hnddma_t *dmah);
bool dma_rxreset(hnddma_t *dmah);
bool dma_rxidle(hnddma_t *dmah);
void dma_txinit(hnddma_t *dmah);
bool dma_txenabled(hnddma_t *dmah);
void dma_rxinit(hnddma_t *dmah);
void dma_txsuspend(hnddma_t *dmah);
void dma_txresume(hnddma_t *dmah);
bool dma_txsuspended(hnddma_t *dmah);
bool dma_txsuspendedidle(hnddma_t *dmah);
void dma_txflush(hnddma_t *dmah);
void dma_txflush_clear(hnddma_t *dmah);
int dma_txfast_ext(hnddma_t *dmah, void *p0, bool commit, uint16 *pre_txout, uint16 *numd);
int dma_txfast_alfrag(hnddma_t *dmah, hnddma_t *aqm_dmah, void *p, bool commit, dma64dd_t *aqmdesc,
uint d11_txh_len, bool ptxd_hw_enab);
#define dma_txfast(dmah, p0, commit) \
dma_txfast_ext((dmah), (p0), (commit), NULL, NULL)
void dma_txcommit(hnddma_t *dmah);
int dma_txunframed(hnddma_t *dmah, void *buf, uint len, bool commit);
void *dma_getpos(hnddma_t *dmah, bool direction);
void dma_fifoloopbackenable(hnddma_t *dmah);
void dma_fifoloopbackdisable(hnddma_t *dmah);
bool dma_txstopped(hnddma_t *dmah);
bool dma_rxstopped(hnddma_t *dmah);
void dma_rxenable(hnddma_t *dmah);
bool dma_rxenabled(hnddma_t *dmah);
void *dma_rx(hnddma_t *dmah);
#ifdef APP_RX
void dma_getnextrxp_app(hnddma_t *dmah, bool forceall, uint *pktcnt,
void **head, void **tail);
void dma_rxfill_haddr_getparams(hnddma_t *dmah, uint *nrxd, uint16 *rxout,
dma64dd_t **ddring, uint *rxextrahdrroom, uint32 **rxpktid);
void dma_rxfill_haddr_setparams(hnddma_t *dmah, uint16 rxout);
#endif /* APP_RX */
uint dma_rx_get_rxoffset(hnddma_t *dmah);
bool dma_rxfill(hnddma_t *dmah);
bool dma_rxfill_required(hnddma_t *dmah);
void dma_txreclaim(hnddma_t *dmah, txd_range_t range);
void dma_rxreclaim(hnddma_t *dmah);
#define _DMA_GETUINTVARPTR_
uint *dma_getuintvarptr(hnddma_t *dmah, const char *name);
uint8 dma_getuint8var(hnddma_t *dmah, const char *name);
uint16 dma_getuint16var(hnddma_t *dmah, const char *name);
uint32 dma_getuint32var(hnddma_t *dmah, const char *name);
void * dma_getnexttxp(hnddma_t *dmah, txd_range_t range);
void * dma_getnextp(hnddma_t *dmah);
void * dma_getnextrxp(hnddma_t *dmah, bool forceall);
void * dma_peeknexttxp(hnddma_t *dmah, txd_range_t range);
int dma_peekntxp(hnddma_t *dmah, int *len, void *txps[], txd_range_t range);
void * dma_peeknextrxp(hnddma_t *dmah);
void dma_rxparam_get(hnddma_t *dmah, uint16 *rxoffset, uint16 *rxbufsize);
bool dma_is_rxfill_suspend(hnddma_t *dmah);
void dma_txblock(hnddma_t *dmah);
void dma_txunblock(hnddma_t *dmah);
uint dma_txactive(hnddma_t *dmah);
uint dma_rxactive(hnddma_t *dmah);
void dma_txrotate(hnddma_t *dmah);
void dma_counterreset(hnddma_t *dmah);
uint dma_ctrlflags(hnddma_t *dmah, uint mask, uint flags);
uint dma_txpending(hnddma_t *dmah);
uint dma_txcommitted(hnddma_t *dmah);
int dma_pktpool_set(hnddma_t *dmah, pktpool_t *pool);
int dma_rxdatapool_set(hnddma_t *dmah, pktpool_t *pktpool);
pktpool_t *dma_rxdatapool_get(hnddma_t *dmah);
void dma_dump_txdmaregs(hnddma_t *dmah, uint32 **buf);
void dma_dump_rxdmaregs(hnddma_t *dmah, uint32 **buf);
#if defined(BCMDBG) || defined(BCMDBG_DUMP) || defined(BCMDBG_DMA)
void dma_dump(hnddma_t *dmah, struct bcmstrbuf *b, bool dumpring);
void dma_dumptx(hnddma_t *dmah, struct bcmstrbuf *b, bool dumpring);
void dma_dumprx(hnddma_t *dmah, struct bcmstrbuf *b, bool dumpring);
#endif
bool dma_rxtxerror(hnddma_t *dmah, bool istx);
void dma_burstlen_set(hnddma_t *dmah, uint8 rxburstlen, uint8 txburstlen);
uint dma_avoidance_cnt(hnddma_t *dmah);
void dma_param_set(hnddma_t *dmah, uint16 paramid, uint16 paramval);
void dma_param_get(hnddma_t *dmah, uint16 paramid, uint *paramval);
void dma_context(hnddma_t *dmah, setup_context_t fn, void *ctx);
bool dma_glom_enable(hnddma_t *dmah, uint32 val);
uint dma_activerxbuf(hnddma_t *dmah);
bool dma_rxidlestatus(hnddma_t *dmah);
uint dma_get_rxpost(hnddma_t *dmah);
/* return addresswidth allowed
* This needs to be done after SB attach but before dma attach.
* SB attach provides ability to probe backplane and dma core capabilities
* This info is needed by DMA_ALLOC_CONSISTENT in dma attach
*/
extern uint dma_addrwidth(si_t *sih, void *dmaregs);
/* count the number of tx packets that are queued to the dma ring */
extern uint dma_txp(hnddma_t *di);
extern void dma_txrewind(hnddma_t *di);
/* pio helpers */
extern int dma_msgbuf_txfast(hnddma_t *di, dma64addr_t p0, bool com, uint32 ln, bool fst, bool lst);
extern int dma_ptrbuf_txfast(hnddma_t *dmah, dma64addr_t p0, void *p, bool commit,
uint32 len, bool first, bool last);
extern int dma_rxfast(hnddma_t *di, dma64addr_t p, uint32 len);
extern int dma_rxfill_suspend(hnddma_t *dmah, bool suspended);
extern void dma_link_handle(hnddma_t *dmah1, hnddma_t *dmah2);
extern void dma_unlink_handle(hnddma_t *dmah1, hnddma_t *dmah2);
extern int dma_rxfill_unframed(hnddma_t *di, void *buf, uint len, bool commit);
extern uint16 dma_get_next_txd_idx(hnddma_t *di, bool txout);
extern uint16 dma_get_txd_count(hnddma_t *dmah, uint16 start, bool txout);
extern uintptr dma_get_txd_addr(hnddma_t *di, uint16 idx);
/* returns the memory address (hi and low) of the buffer associated with the dma descriptor
* having index idx.
*/
extern void dma_get_txd_memaddr(hnddma_t *dmah, uint32 *addrlo, uint32 *addrhi, uint idx);
extern int dma_txdesc(hnddma_t *dmah, dma64dd_t *dd, bool commit);
extern int dma_nexttxdd(hnddma_t *dmah, txd_range_t range, uint32 *flags1, uint32 *flags2,
bool advance);
extern void dma_update_rxfill(hnddma_t *dmah);
extern void dma_rxchan_reset(hnddma_t *di);
extern void dma_txchan_reset(hnddma_t *di);
extern void dma_chan_reset(hnddma_t *dmah);
extern pktpool_t* dma_pktpool_get(hnddma_t *dmah);
extern void dma_clearrxp(hnddma_t *dmah);
extern void dma_cleartxp(hnddma_t *dmah);
#define dma_getnexttxdd(dmah, range, flags1, flags2) \
dma_nexttxdd((dmah), (range), (flags1), (flags2), TRUE)
#define dma_peeknexttxdd(dmah, range, flags1, flags2) \
dma_nexttxdd((dmah), (range), (flags1), (flags2), FALSE)
#define NUM_VEC_PCIE 4
#define XFER_FROM_LBUF 0x1
#define XFER_TO_LBUF 0x2
#define XFER_INJ_ERR 0x4
typedef struct m2m_vec_s {
dma64addr_t addr;
uint32 len;
} m2m_vec_t;
typedef struct m2m_desc_s {
uint8 num_rx_vec;
uint8 num_tx_vec;
uint8 flags;
bool commit;
m2m_vec_t vec[];
} m2m_desc_t;
#define INIT_M2M_DESC(desc) \
{\
desc->num_rx_vec = 0; \
desc->num_tx_vec = 0; \
desc->flags = 0; \
desc->commit = TRUE; \
}
#define SETUP_RX_DESC(desc, rxaddr, rxlen) \
{\
ASSERT(desc->num_tx_vec == 0); \
desc->vec[desc->num_rx_vec].addr = rxaddr; \
desc->vec[desc->num_rx_vec].len = rxlen; \
desc->num_rx_vec++; \
}
#define SETUP_TX_DESC(desc, txaddr, txlen) \
{\
desc->vec[desc->num_tx_vec + desc->num_rx_vec].addr = txaddr; \
desc->vec[desc->num_tx_vec + desc->num_rx_vec].len = txlen; \
desc->num_tx_vec++; \
}
#define SETUP_XFER_FLAGS(desc, flag) \
{\
desc->flags |= flag; \
}
#define DD_IS_SHARED_POOL(di) ((di)->dmactrlflags & DMA_CTRL_SHARED_POOL)
extern int dma_m2m_submit(hnddma_t *dmah, m2m_desc_t *desc, bool implicit);
extern void dma_chan_enable(hnddma_t *dmah, bool enable);
extern bool dma_rxfill_p(hnddma_t *dmah, void *p);
extern void dma_aqm_di_link(hnddma_t *dmah_aqm, hnddma_t *dmah_hw);
extern void dma_dump_aqminfo(hnddma_t * dmah, struct bcmstrbuf *b, uint16 fifonum);
/* To dump ntxd and nrxd from the DMA ring */
void dma_dump_info(hnddma_t *dmah, uint16 fifonum, struct bcmstrbuf *b);
#endif /* _hnddma_h_ */