/* SPDX-License-Identifier: (GPL-2.0 OR BSD-3-Clause) */ /* * Copyright(c) 2020 Intel Corporation. * */ /* * This file contains HFI1 support for IPOIB functionality */ #ifndef HFI1_IPOIB_H #define HFI1_IPOIB_H #include <linux/types.h> #include <linux/stddef.h> #include <linux/atomic.h> #include <linux/netdevice.h> #include <linux/slab.h> #include <linux/skbuff.h> #include <linux/list.h> #include <linux/if_infiniband.h> #include "hfi.h" #include "iowait.h" #include "netdev.h" #include <rdma/ib_verbs.h> #define HFI1_IPOIB_ENTROPY_SHIFT 24 #define HFI1_IPOIB_TXREQ_NAME_LEN 32 #define HFI1_IPOIB_PSEUDO_LEN 20 #define HFI1_IPOIB_ENCAP_LEN 4 struct hfi1_ipoib_dev_priv; union hfi1_ipoib_flow { u16 as_int; struct { u8 tx_queue; u8 sc5; } __attribute__((__packed__)); }; /** * struct ipoib_txreq - IPOIB transmit descriptor * @txreq: sdma transmit request * @sdma_hdr: 9b ib headers * @sdma_status: status returned by sdma engine * @complete: non-zero implies complete * @priv: ipoib netdev private data * @txq: txq on which skb was output * @skb: skb to send */ struct ipoib_txreq { struct sdma_txreq txreq; struct hfi1_sdma_header *sdma_hdr; int sdma_status; int complete; struct hfi1_ipoib_dev_priv *priv; struct hfi1_ipoib_txq *txq; struct sk_buff *skb; }; /** * struct hfi1_ipoib_circ_buf - List of items to be processed * @items: ring of items each a power of two size * @max_items: max items + 1 that the ring can contain * @shift: log2 of size for getting txreq * @sent_txreqs: count of txreqs posted to sdma * @tail: ring tail * @stops: count of stops of queue * @ring_full: ring has been filled * @no_desc: descriptor shortage seen * @complete_txreqs: count of txreqs completed by sdma * @head: ring head */ struct hfi1_ipoib_circ_buf { void *items; u32 max_items; u32 shift; /* consumer cache line */ u64 ____cacheline_aligned_in_smp sent_txreqs; u32 avail; u32 tail; atomic_t stops; atomic_t ring_full; atomic_t no_desc; /* producer cache line */ u64 ____cacheline_aligned_in_smp complete_txreqs; u32 head; }; /** * struct hfi1_ipoib_txq - IPOIB per Tx queue information * @priv: private pointer * @sde: sdma engine * @tx_list: tx request list * @sent_txreqs: count of txreqs posted to sdma * @flow: tracks when list needs to be flushed for a flow change * @q_idx: ipoib Tx queue index * @pkts_sent: indicator packets have been sent from this queue * @wait: iowait structure * @napi: pointer to tx napi interface * @tx_ring: ring of ipoib txreqs to be reaped by napi callback */ struct hfi1_ipoib_txq { struct napi_struct napi; struct hfi1_ipoib_dev_priv *priv; struct sdma_engine *sde; struct list_head tx_list; union hfi1_ipoib_flow flow; u8 q_idx; bool pkts_sent; struct iowait wait; struct hfi1_ipoib_circ_buf ____cacheline_aligned_in_smp tx_ring; }; struct hfi1_ipoib_dev_priv { struct hfi1_devdata *dd; struct net_device *netdev; struct ib_device *device; struct hfi1_ipoib_txq *txqs; const struct net_device_ops *netdev_ops; struct rvt_qp *qp; u32 qkey; u16 pkey; u16 pkey_index; u8 port_num; }; /* hfi1 ipoib rdma netdev's private data structure */ struct hfi1_ipoib_rdma_netdev { struct rdma_netdev rn; /* keep this first */ /* followed by device private data */ struct hfi1_ipoib_dev_priv dev_priv; }; static inline struct hfi1_ipoib_dev_priv * hfi1_ipoib_priv(const struct net_device *dev) { return &((struct hfi1_ipoib_rdma_netdev *)netdev_priv(dev))->dev_priv; } int hfi1_ipoib_send(struct net_device *dev, struct sk_buff *skb, struct ib_ah *address, u32 dqpn); int hfi1_ipoib_txreq_init(struct hfi1_ipoib_dev_priv *priv); void hfi1_ipoib_txreq_deinit(struct hfi1_ipoib_dev_priv *priv); int hfi1_ipoib_rxq_init(struct net_device *dev); void hfi1_ipoib_rxq_deinit(struct net_device *dev); void hfi1_ipoib_napi_tx_enable(struct net_device *dev); void hfi1_ipoib_napi_tx_disable(struct net_device *dev); struct sk_buff *hfi1_ipoib_prepare_skb(struct hfi1_netdev_rxq *rxq, int size, void *data); int hfi1_ipoib_rn_get_params(struct ib_device *device, u32 port_num, enum rdma_netdev_t type, struct rdma_netdev_alloc_params *params); void hfi1_ipoib_tx_timeout(struct net_device *dev, unsigned int q); #endif /* _IPOIB_H */