5e15e799d6
Before this change memory for the private data of all physical ports where allocated with single rte_zmalloc() call. Specific port private data was accessed by means of an offset into this memory. This scheme is problematic when attempting to free only one port's private data at a time. To address this, a new entity is created called struct nfp_pf_dev. This struct represents the PF device. It has a number of PF specific members. Notably it has a pointer of type rte_eth_dev that points to the eth_dev associated with the first physical port of the device. It also has an array of nfp_net_hw's containing pointers to all the physical ports under the PF. Memory is first allocated for the PF and PF specific initialization is attempted. Next, all the physical ports under the PF is iterated and memory is allocated separately for the private data of each port. Port 0 is skipped during this phase because memory has already been allocated and an eth_dev already exits for the 0th port. Signed-off-by: Heinrich Kuhn <heinrich.kuhn@netronome.com> Signed-off-by: Simon Horman <simon.horman@netronome.com> Reviewed-by: Louis Peens <louis.peens@netronome.com>
511 lines
13 KiB
C
511 lines
13 KiB
C
/* SPDX-License-Identifier: BSD-3-Clause
|
|
* Copyright (c) 2014-2018 Netronome Systems, Inc.
|
|
* All rights reserved.
|
|
*/
|
|
|
|
/*
|
|
* vim:shiftwidth=8:noexpandtab
|
|
*
|
|
* @file dpdk/pmd/nfp_net_pmd.h
|
|
*
|
|
* Netronome NFP_NET PMD driver
|
|
*/
|
|
|
|
#ifndef _NFP_NET_PMD_H_
|
|
#define _NFP_NET_PMD_H_
|
|
|
|
#define NFP_NET_PMD_VERSION "0.1"
|
|
#define PCI_VENDOR_ID_NETRONOME 0x19ee
|
|
#define PCI_DEVICE_ID_NFP4000_PF_NIC 0x4000
|
|
#define PCI_DEVICE_ID_NFP6000_PF_NIC 0x6000
|
|
#define PCI_DEVICE_ID_NFP6000_VF_NIC 0x6003
|
|
|
|
/* Forward declaration */
|
|
struct nfp_net_adapter;
|
|
|
|
/*
|
|
* The maximum number of descriptors is limited by design as
|
|
* DPDK uses uint16_t variables for these values
|
|
*/
|
|
#define NFP_NET_MAX_TX_DESC (32 * 1024)
|
|
#define NFP_NET_MIN_TX_DESC 64
|
|
|
|
#define NFP_NET_MAX_RX_DESC (32 * 1024)
|
|
#define NFP_NET_MIN_RX_DESC 64
|
|
|
|
/* Descriptor alignment */
|
|
#define NFP_ALIGN_RING_DESC 128
|
|
|
|
#define NFP_TX_MAX_SEG UINT8_MAX
|
|
#define NFP_TX_MAX_MTU_SEG 8
|
|
|
|
/* Bar allocation */
|
|
#define NFP_NET_CRTL_BAR 0
|
|
#define NFP_NET_TX_BAR 2
|
|
#define NFP_NET_RX_BAR 2
|
|
#define NFP_QCP_QUEUE_AREA_SZ 0x80000
|
|
|
|
/* Macros for accessing the Queue Controller Peripheral 'CSRs' */
|
|
#define NFP_QCP_QUEUE_OFF(_x) ((_x) * 0x800)
|
|
#define NFP_QCP_QUEUE_ADD_RPTR 0x0000
|
|
#define NFP_QCP_QUEUE_ADD_WPTR 0x0004
|
|
#define NFP_QCP_QUEUE_STS_LO 0x0008
|
|
#define NFP_QCP_QUEUE_STS_LO_READPTR_mask (0x3ffff)
|
|
#define NFP_QCP_QUEUE_STS_HI 0x000c
|
|
#define NFP_QCP_QUEUE_STS_HI_WRITEPTR_mask (0x3ffff)
|
|
|
|
/* Interrupt definitions */
|
|
#define NFP_NET_IRQ_LSC_IDX 0
|
|
|
|
/* Default values for RX/TX configuration */
|
|
#define DEFAULT_RX_FREE_THRESH 32
|
|
#define DEFAULT_RX_PTHRESH 8
|
|
#define DEFAULT_RX_HTHRESH 8
|
|
#define DEFAULT_RX_WTHRESH 0
|
|
|
|
#define DEFAULT_TX_RS_THRESH 32
|
|
#define DEFAULT_TX_FREE_THRESH 32
|
|
#define DEFAULT_TX_PTHRESH 32
|
|
#define DEFAULT_TX_HTHRESH 0
|
|
#define DEFAULT_TX_WTHRESH 0
|
|
#define DEFAULT_TX_RSBIT_THRESH 32
|
|
|
|
/* Alignment for dma zones */
|
|
#define NFP_MEMZONE_ALIGN 128
|
|
|
|
/*
|
|
* This is used by the reconfig protocol. It sets the maximum time waiting in
|
|
* milliseconds before a reconfig timeout happens.
|
|
*/
|
|
#define NFP_NET_POLL_TIMEOUT 5000
|
|
|
|
#define NFP_QCP_QUEUE_ADDR_SZ (0x800)
|
|
|
|
#define NFP_NET_LINK_DOWN_CHECK_TIMEOUT 4000 /* ms */
|
|
#define NFP_NET_LINK_UP_CHECK_TIMEOUT 1000 /* ms */
|
|
|
|
/* Version number helper defines */
|
|
#define NFD_CFG_CLASS_VER_msk 0xff
|
|
#define NFD_CFG_CLASS_VER_shf 24
|
|
#define NFD_CFG_CLASS_VER(x) (((x) & 0xff) << 24)
|
|
#define NFD_CFG_CLASS_VER_of(x) (((x) >> 24) & 0xff)
|
|
#define NFD_CFG_CLASS_TYPE_msk 0xff
|
|
#define NFD_CFG_CLASS_TYPE_shf 16
|
|
#define NFD_CFG_CLASS_TYPE(x) (((x) & 0xff) << 16)
|
|
#define NFD_CFG_CLASS_TYPE_of(x) (((x) >> 16) & 0xff)
|
|
#define NFD_CFG_MAJOR_VERSION_msk 0xff
|
|
#define NFD_CFG_MAJOR_VERSION_shf 8
|
|
#define NFD_CFG_MAJOR_VERSION(x) (((x) & 0xff) << 8)
|
|
#define NFD_CFG_MAJOR_VERSION_of(x) (((x) >> 8) & 0xff)
|
|
#define NFD_CFG_MINOR_VERSION_msk 0xff
|
|
#define NFD_CFG_MINOR_VERSION_shf 0
|
|
#define NFD_CFG_MINOR_VERSION(x) (((x) & 0xff) << 0)
|
|
#define NFD_CFG_MINOR_VERSION_of(x) (((x) >> 0) & 0xff)
|
|
|
|
/* Number of supported physical ports */
|
|
#define NFP_MAX_PHYPORTS 12
|
|
|
|
#include <linux/types.h>
|
|
#include <rte_io.h>
|
|
|
|
static inline uint8_t nn_readb(volatile const void *addr)
|
|
{
|
|
return rte_read8(addr);
|
|
}
|
|
|
|
static inline void nn_writeb(uint8_t val, volatile void *addr)
|
|
{
|
|
rte_write8(val, addr);
|
|
}
|
|
|
|
static inline uint32_t nn_readl(volatile const void *addr)
|
|
{
|
|
return rte_read32(addr);
|
|
}
|
|
|
|
static inline void nn_writel(uint32_t val, volatile void *addr)
|
|
{
|
|
rte_write32(val, addr);
|
|
}
|
|
|
|
static inline void nn_writew(uint16_t val, volatile void *addr)
|
|
{
|
|
rte_write16(val, addr);
|
|
}
|
|
|
|
static inline uint64_t nn_readq(volatile void *addr)
|
|
{
|
|
const volatile uint32_t *p = addr;
|
|
uint32_t low, high;
|
|
|
|
high = nn_readl((volatile const void *)(p + 1));
|
|
low = nn_readl((volatile const void *)p);
|
|
|
|
return low + ((uint64_t)high << 32);
|
|
}
|
|
|
|
static inline void nn_writeq(uint64_t val, volatile void *addr)
|
|
{
|
|
nn_writel(val >> 32, (volatile char *)addr + 4);
|
|
nn_writel(val, addr);
|
|
}
|
|
|
|
/* TX descriptor format */
|
|
#define PCIE_DESC_TX_EOP (1 << 7)
|
|
#define PCIE_DESC_TX_OFFSET_MASK (0x7f)
|
|
|
|
/* Flags in the host TX descriptor */
|
|
#define PCIE_DESC_TX_CSUM (1 << 7)
|
|
#define PCIE_DESC_TX_IP4_CSUM (1 << 6)
|
|
#define PCIE_DESC_TX_TCP_CSUM (1 << 5)
|
|
#define PCIE_DESC_TX_UDP_CSUM (1 << 4)
|
|
#define PCIE_DESC_TX_VLAN (1 << 3)
|
|
#define PCIE_DESC_TX_LSO (1 << 2)
|
|
#define PCIE_DESC_TX_ENCAP_NONE (0)
|
|
#define PCIE_DESC_TX_ENCAP_VXLAN (1 << 1)
|
|
#define PCIE_DESC_TX_ENCAP_GRE (1 << 0)
|
|
|
|
struct nfp_net_tx_desc {
|
|
union {
|
|
struct {
|
|
uint8_t dma_addr_hi; /* High bits of host buf address */
|
|
__le16 dma_len; /* Length to DMA for this desc */
|
|
uint8_t offset_eop; /* Offset in buf where pkt starts +
|
|
* highest bit is eop flag.
|
|
*/
|
|
__le32 dma_addr_lo; /* Low 32bit of host buf addr */
|
|
|
|
__le16 mss; /* MSS to be used for LSO */
|
|
uint8_t lso_hdrlen; /* LSO, where the data starts */
|
|
uint8_t flags; /* TX Flags, see @PCIE_DESC_TX_* */
|
|
|
|
union {
|
|
struct {
|
|
/*
|
|
* L3 and L4 header offsets required
|
|
* for TSOv2
|
|
*/
|
|
uint8_t l3_offset;
|
|
uint8_t l4_offset;
|
|
};
|
|
__le16 vlan; /* VLAN tag to add if indicated */
|
|
};
|
|
__le16 data_len; /* Length of frame + meta data */
|
|
} __rte_packed;
|
|
__le32 vals[4];
|
|
};
|
|
};
|
|
|
|
struct nfp_net_txq {
|
|
struct nfp_net_hw *hw; /* Backpointer to nfp_net structure */
|
|
|
|
/*
|
|
* Queue information: @qidx is the queue index from Linux's
|
|
* perspective. @tx_qcidx is the index of the Queue
|
|
* Controller Peripheral queue relative to the TX queue BAR.
|
|
* @cnt is the size of the queue in number of
|
|
* descriptors. @qcp_q is a pointer to the base of the queue
|
|
* structure on the NFP
|
|
*/
|
|
uint8_t *qcp_q;
|
|
|
|
/*
|
|
* Read and Write pointers. @wr_p and @rd_p are host side pointer,
|
|
* they are free running and have little relation to the QCP pointers *
|
|
* @qcp_rd_p is a local copy queue controller peripheral read pointer
|
|
*/
|
|
|
|
uint32_t wr_p;
|
|
uint32_t rd_p;
|
|
|
|
uint32_t tx_count;
|
|
|
|
uint32_t tx_free_thresh;
|
|
|
|
/*
|
|
* For each descriptor keep a reference to the mbuf and
|
|
* DMA address used until completion is signalled.
|
|
*/
|
|
struct {
|
|
struct rte_mbuf *mbuf;
|
|
} *txbufs;
|
|
|
|
/*
|
|
* Information about the host side queue location. @txds is
|
|
* the virtual address for the queue, @dma is the DMA address
|
|
* of the queue and @size is the size in bytes for the queue
|
|
* (needed for free)
|
|
*/
|
|
struct nfp_net_tx_desc *txds;
|
|
|
|
/*
|
|
* At this point 48 bytes have been used for all the fields in the
|
|
* TX critical path. We have room for 8 bytes and still all placed
|
|
* in a cache line. We are not using the threshold values below but
|
|
* if we need to, we can add the most used in the remaining bytes.
|
|
*/
|
|
uint32_t tx_rs_thresh; /* not used by now. Future? */
|
|
uint32_t tx_pthresh; /* not used by now. Future? */
|
|
uint32_t tx_hthresh; /* not used by now. Future? */
|
|
uint32_t tx_wthresh; /* not used by now. Future? */
|
|
uint16_t port_id;
|
|
int qidx;
|
|
int tx_qcidx;
|
|
__le64 dma;
|
|
} __rte_aligned(64);
|
|
|
|
/* RX and freelist descriptor format */
|
|
#define PCIE_DESC_RX_DD (1 << 7)
|
|
#define PCIE_DESC_RX_META_LEN_MASK (0x7f)
|
|
|
|
/* Flags in the RX descriptor */
|
|
#define PCIE_DESC_RX_RSS (1 << 15)
|
|
#define PCIE_DESC_RX_I_IP4_CSUM (1 << 14)
|
|
#define PCIE_DESC_RX_I_IP4_CSUM_OK (1 << 13)
|
|
#define PCIE_DESC_RX_I_TCP_CSUM (1 << 12)
|
|
#define PCIE_DESC_RX_I_TCP_CSUM_OK (1 << 11)
|
|
#define PCIE_DESC_RX_I_UDP_CSUM (1 << 10)
|
|
#define PCIE_DESC_RX_I_UDP_CSUM_OK (1 << 9)
|
|
#define PCIE_DESC_RX_SPARE (1 << 8)
|
|
#define PCIE_DESC_RX_EOP (1 << 7)
|
|
#define PCIE_DESC_RX_IP4_CSUM (1 << 6)
|
|
#define PCIE_DESC_RX_IP4_CSUM_OK (1 << 5)
|
|
#define PCIE_DESC_RX_TCP_CSUM (1 << 4)
|
|
#define PCIE_DESC_RX_TCP_CSUM_OK (1 << 3)
|
|
#define PCIE_DESC_RX_UDP_CSUM (1 << 2)
|
|
#define PCIE_DESC_RX_UDP_CSUM_OK (1 << 1)
|
|
#define PCIE_DESC_RX_VLAN (1 << 0)
|
|
|
|
#define PCIE_DESC_RX_L4_CSUM_OK (PCIE_DESC_RX_TCP_CSUM_OK | \
|
|
PCIE_DESC_RX_UDP_CSUM_OK)
|
|
struct nfp_net_rx_desc {
|
|
union {
|
|
/* Freelist descriptor */
|
|
struct {
|
|
uint8_t dma_addr_hi;
|
|
__le16 spare;
|
|
uint8_t dd;
|
|
|
|
__le32 dma_addr_lo;
|
|
} __rte_packed fld;
|
|
|
|
/* RX descriptor */
|
|
struct {
|
|
__le16 data_len;
|
|
uint8_t reserved;
|
|
uint8_t meta_len_dd;
|
|
|
|
__le16 flags;
|
|
__le16 vlan;
|
|
} __rte_packed rxd;
|
|
|
|
__le32 vals[2];
|
|
};
|
|
};
|
|
|
|
struct nfp_net_rx_buff {
|
|
struct rte_mbuf *mbuf;
|
|
};
|
|
|
|
struct nfp_net_rxq {
|
|
struct nfp_net_hw *hw; /* Backpointer to nfp_net structure */
|
|
|
|
/*
|
|
* @qcp_fl and @qcp_rx are pointers to the base addresses of the
|
|
* freelist and RX queue controller peripheral queue structures on the
|
|
* NFP
|
|
*/
|
|
uint8_t *qcp_fl;
|
|
uint8_t *qcp_rx;
|
|
|
|
/*
|
|
* Read and Write pointers. @wr_p and @rd_p are host side
|
|
* pointer, they are free running and have little relation to
|
|
* the QCP pointers. @wr_p is where the driver adds new
|
|
* freelist descriptors and @rd_p is where the driver start
|
|
* reading descriptors for newly arrive packets from.
|
|
*/
|
|
uint32_t rd_p;
|
|
|
|
/*
|
|
* For each buffer placed on the freelist, record the
|
|
* associated SKB
|
|
*/
|
|
struct nfp_net_rx_buff *rxbufs;
|
|
|
|
/*
|
|
* Information about the host side queue location. @rxds is
|
|
* the virtual address for the queue
|
|
*/
|
|
struct nfp_net_rx_desc *rxds;
|
|
|
|
/*
|
|
* The mempool is created by the user specifying a mbuf size.
|
|
* We save here the reference of the mempool needed in the RX
|
|
* path and the mbuf size for checking received packets can be
|
|
* safely copied to the mbuf using the NFP_NET_RX_OFFSET
|
|
*/
|
|
struct rte_mempool *mem_pool;
|
|
uint16_t mbuf_size;
|
|
|
|
/*
|
|
* Next two fields are used for giving more free descriptors
|
|
* to the NFP
|
|
*/
|
|
uint16_t rx_free_thresh;
|
|
uint16_t nb_rx_hold;
|
|
|
|
/* the size of the queue in number of descriptors */
|
|
uint16_t rx_count;
|
|
|
|
/*
|
|
* Fields above this point fit in a single cache line and are all used
|
|
* in the RX critical path. Fields below this point are just used
|
|
* during queue configuration or not used at all (yet)
|
|
*/
|
|
|
|
/* referencing dev->data->port_id */
|
|
uint16_t port_id;
|
|
|
|
uint8_t crc_len; /* Not used by now */
|
|
uint8_t drop_en; /* Not used by now */
|
|
|
|
/* DMA address of the queue */
|
|
__le64 dma;
|
|
|
|
/*
|
|
* Queue information: @qidx is the queue index from Linux's
|
|
* perspective. @fl_qcidx is the index of the Queue
|
|
* Controller peripheral queue relative to the RX queue BAR
|
|
* used for the freelist and @rx_qcidx is the Queue Controller
|
|
* Peripheral index for the RX queue.
|
|
*/
|
|
int qidx;
|
|
int fl_qcidx;
|
|
int rx_qcidx;
|
|
} __rte_aligned(64);
|
|
|
|
struct nfp_pf_dev {
|
|
/* Backpointer to associated pci device */
|
|
struct rte_pci_device *pci_dev;
|
|
|
|
/* First physical port's eth device */
|
|
struct rte_eth_dev *eth_dev;
|
|
|
|
/* Array of physical ports belonging to this PF */
|
|
struct nfp_net_hw *ports[NFP_MAX_PHYPORTS];
|
|
|
|
/* Current values for control */
|
|
uint32_t ctrl;
|
|
|
|
uint8_t *ctrl_bar;
|
|
uint8_t *tx_bar;
|
|
uint8_t *rx_bar;
|
|
|
|
uint8_t *qcp_cfg;
|
|
rte_spinlock_t reconfig_lock;
|
|
|
|
uint16_t flbufsz;
|
|
uint16_t device_id;
|
|
uint16_t vendor_id;
|
|
uint16_t subsystem_device_id;
|
|
uint16_t subsystem_vendor_id;
|
|
#if defined(DSTQ_SELECTION)
|
|
#if DSTQ_SELECTION
|
|
uint16_t device_function;
|
|
#endif
|
|
#endif
|
|
|
|
struct nfp_cpp *cpp;
|
|
struct nfp_cpp_area *ctrl_area;
|
|
struct nfp_cpp_area *hwqueues_area;
|
|
struct nfp_cpp_area *msix_area;
|
|
|
|
uint8_t *hw_queues;
|
|
uint8_t total_phyports;
|
|
bool multiport;
|
|
|
|
union eth_table_entry *eth_table;
|
|
|
|
struct nfp_hwinfo *hwinfo;
|
|
struct nfp_rtsym_table *sym_tbl;
|
|
uint32_t nfp_cpp_service_id;
|
|
};
|
|
|
|
struct nfp_net_hw {
|
|
/* Backpointer to the PF this port belongs to */
|
|
struct nfp_pf_dev *pf_dev;
|
|
|
|
/* Backpointer to the eth_dev of this port*/
|
|
struct rte_eth_dev *eth_dev;
|
|
|
|
/* Info from the firmware */
|
|
uint32_t ver;
|
|
uint32_t cap;
|
|
uint32_t max_mtu;
|
|
uint32_t mtu;
|
|
uint32_t rx_offset;
|
|
|
|
/* Current values for control */
|
|
uint32_t ctrl;
|
|
|
|
uint8_t *ctrl_bar;
|
|
uint8_t *tx_bar;
|
|
uint8_t *rx_bar;
|
|
|
|
int stride_rx;
|
|
int stride_tx;
|
|
|
|
uint8_t *qcp_cfg;
|
|
rte_spinlock_t reconfig_lock;
|
|
|
|
uint32_t max_tx_queues;
|
|
uint32_t max_rx_queues;
|
|
uint16_t flbufsz;
|
|
uint16_t device_id;
|
|
uint16_t vendor_id;
|
|
uint16_t subsystem_device_id;
|
|
uint16_t subsystem_vendor_id;
|
|
#if defined(DSTQ_SELECTION)
|
|
#if DSTQ_SELECTION
|
|
uint16_t device_function;
|
|
#endif
|
|
#endif
|
|
|
|
uint8_t mac_addr[RTE_ETHER_ADDR_LEN];
|
|
|
|
/* Records starting point for counters */
|
|
struct rte_eth_stats eth_stats_base;
|
|
|
|
struct nfp_cpp *cpp;
|
|
struct nfp_cpp_area *ctrl_area;
|
|
struct nfp_cpp_area *hwqueues_area;
|
|
struct nfp_cpp_area *msix_area;
|
|
|
|
uint8_t *hw_queues;
|
|
uint8_t idx;
|
|
bool is_phyport;
|
|
|
|
union eth_table_entry *eth_table;
|
|
|
|
uint32_t nfp_cpp_service_id;
|
|
};
|
|
|
|
struct nfp_net_adapter {
|
|
struct nfp_net_hw hw;
|
|
};
|
|
|
|
#define NFP_NET_DEV_PRIVATE_TO_HW(adapter)\
|
|
(&((struct nfp_net_adapter *)adapter)->hw)
|
|
|
|
#define NFP_NET_DEV_PRIVATE_TO_PF(dev_priv)\
|
|
(((struct nfp_net_hw *)dev_priv)->pf_dev)
|
|
|
|
#endif /* _NFP_NET_PMD_H_ */
|
|
/*
|
|
* Local variables:
|
|
* c-file-style: "Linux"
|
|
* indent-tabs-mode: t
|
|
* End:
|
|
*/
|