mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-01-18 02:46:06 +00:00
Merge branches 'cxgb4', 'misc', 'mlx4', 'nes' and 'uapi' into for-next
This commit is contained in:
commit
d0951d2133
@ -311,6 +311,7 @@ void c2_ae_event(struct c2_dev *c2dev, u32 mq_index)
|
|||||||
if (cq->ibcq.event_handler)
|
if (cq->ibcq.event_handler)
|
||||||
cq->ibcq.event_handler(&ib_event,
|
cq->ibcq.event_handler(&ib_event,
|
||||||
cq->ibcq.cq_context);
|
cq->ibcq.cq_context);
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
default:
|
default:
|
||||||
|
@ -128,9 +128,8 @@ static void stop_ep_timer(struct iwch_ep *ep)
|
|||||||
{
|
{
|
||||||
PDBG("%s ep %p\n", __func__, ep);
|
PDBG("%s ep %p\n", __func__, ep);
|
||||||
if (!timer_pending(&ep->timer)) {
|
if (!timer_pending(&ep->timer)) {
|
||||||
printk(KERN_ERR "%s timer stopped when its not running! ep %p state %u\n",
|
WARN(1, "%s timer stopped when its not running! ep %p state %u\n",
|
||||||
__func__, ep, ep->com.state);
|
__func__, ep, ep->com.state);
|
||||||
WARN_ON(1);
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
del_timer_sync(&ep->timer);
|
del_timer_sync(&ep->timer);
|
||||||
@ -1756,9 +1755,8 @@ static void ep_timeout(unsigned long arg)
|
|||||||
__state_set(&ep->com, ABORTING);
|
__state_set(&ep->com, ABORTING);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
printk(KERN_ERR "%s unexpected state ep %p state %u\n",
|
WARN(1, "%s unexpected state ep %p state %u\n",
|
||||||
__func__, ep, ep->com.state);
|
__func__, ep, ep->com.state);
|
||||||
WARN_ON(1);
|
|
||||||
abort = 0;
|
abort = 0;
|
||||||
}
|
}
|
||||||
spin_unlock_irqrestore(&ep->com.lock, flags);
|
spin_unlock_irqrestore(&ep->com.lock, flags);
|
||||||
|
@ -718,16 +718,6 @@ int ipath_init_chip(struct ipath_devdata *dd, int reinit)
|
|||||||
if (ret)
|
if (ret)
|
||||||
goto done;
|
goto done;
|
||||||
|
|
||||||
/*
|
|
||||||
* we ignore most issues after reporting them, but have to specially
|
|
||||||
* handle hardware-disabled chips.
|
|
||||||
*/
|
|
||||||
if (ret == 2) {
|
|
||||||
/* unique error, known to ipath_init_one */
|
|
||||||
ret = -EPERM;
|
|
||||||
goto done;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* We could bump this to allow for full rcvegrcnt + rcvtidcnt,
|
* We could bump this to allow for full rcvegrcnt + rcvtidcnt,
|
||||||
* but then it no longer nicely fits power of two, and since
|
* but then it no longer nicely fits power of two, and since
|
||||||
|
@ -66,7 +66,7 @@ static void mlx4_ib_cq_event(struct mlx4_cq *cq, enum mlx4_event type)
|
|||||||
|
|
||||||
static void *get_cqe_from_buf(struct mlx4_ib_cq_buf *buf, int n)
|
static void *get_cqe_from_buf(struct mlx4_ib_cq_buf *buf, int n)
|
||||||
{
|
{
|
||||||
return mlx4_buf_offset(&buf->buf, n * sizeof (struct mlx4_cqe));
|
return mlx4_buf_offset(&buf->buf, n * buf->entry_size);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void *get_cqe(struct mlx4_ib_cq *cq, int n)
|
static void *get_cqe(struct mlx4_ib_cq *cq, int n)
|
||||||
@ -77,8 +77,9 @@ static void *get_cqe(struct mlx4_ib_cq *cq, int n)
|
|||||||
static void *get_sw_cqe(struct mlx4_ib_cq *cq, int n)
|
static void *get_sw_cqe(struct mlx4_ib_cq *cq, int n)
|
||||||
{
|
{
|
||||||
struct mlx4_cqe *cqe = get_cqe(cq, n & cq->ibcq.cqe);
|
struct mlx4_cqe *cqe = get_cqe(cq, n & cq->ibcq.cqe);
|
||||||
|
struct mlx4_cqe *tcqe = ((cq->buf.entry_size == 64) ? (cqe + 1) : cqe);
|
||||||
|
|
||||||
return (!!(cqe->owner_sr_opcode & MLX4_CQE_OWNER_MASK) ^
|
return (!!(tcqe->owner_sr_opcode & MLX4_CQE_OWNER_MASK) ^
|
||||||
!!(n & (cq->ibcq.cqe + 1))) ? NULL : cqe;
|
!!(n & (cq->ibcq.cqe + 1))) ? NULL : cqe;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -99,12 +100,13 @@ static int mlx4_ib_alloc_cq_buf(struct mlx4_ib_dev *dev, struct mlx4_ib_cq_buf *
|
|||||||
{
|
{
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
err = mlx4_buf_alloc(dev->dev, nent * sizeof(struct mlx4_cqe),
|
err = mlx4_buf_alloc(dev->dev, nent * dev->dev->caps.cqe_size,
|
||||||
PAGE_SIZE * 2, &buf->buf);
|
PAGE_SIZE * 2, &buf->buf);
|
||||||
|
|
||||||
if (err)
|
if (err)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
|
buf->entry_size = dev->dev->caps.cqe_size;
|
||||||
err = mlx4_mtt_init(dev->dev, buf->buf.npages, buf->buf.page_shift,
|
err = mlx4_mtt_init(dev->dev, buf->buf.npages, buf->buf.page_shift,
|
||||||
&buf->mtt);
|
&buf->mtt);
|
||||||
if (err)
|
if (err)
|
||||||
@ -120,8 +122,7 @@ err_mtt:
|
|||||||
mlx4_mtt_cleanup(dev->dev, &buf->mtt);
|
mlx4_mtt_cleanup(dev->dev, &buf->mtt);
|
||||||
|
|
||||||
err_buf:
|
err_buf:
|
||||||
mlx4_buf_free(dev->dev, nent * sizeof(struct mlx4_cqe),
|
mlx4_buf_free(dev->dev, nent * buf->entry_size, &buf->buf);
|
||||||
&buf->buf);
|
|
||||||
|
|
||||||
out:
|
out:
|
||||||
return err;
|
return err;
|
||||||
@ -129,7 +130,7 @@ out:
|
|||||||
|
|
||||||
static void mlx4_ib_free_cq_buf(struct mlx4_ib_dev *dev, struct mlx4_ib_cq_buf *buf, int cqe)
|
static void mlx4_ib_free_cq_buf(struct mlx4_ib_dev *dev, struct mlx4_ib_cq_buf *buf, int cqe)
|
||||||
{
|
{
|
||||||
mlx4_buf_free(dev->dev, (cqe + 1) * sizeof(struct mlx4_cqe), &buf->buf);
|
mlx4_buf_free(dev->dev, (cqe + 1) * buf->entry_size, &buf->buf);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int mlx4_ib_get_cq_umem(struct mlx4_ib_dev *dev, struct ib_ucontext *context,
|
static int mlx4_ib_get_cq_umem(struct mlx4_ib_dev *dev, struct ib_ucontext *context,
|
||||||
@ -137,8 +138,9 @@ static int mlx4_ib_get_cq_umem(struct mlx4_ib_dev *dev, struct ib_ucontext *cont
|
|||||||
u64 buf_addr, int cqe)
|
u64 buf_addr, int cqe)
|
||||||
{
|
{
|
||||||
int err;
|
int err;
|
||||||
|
int cqe_size = dev->dev->caps.cqe_size;
|
||||||
|
|
||||||
*umem = ib_umem_get(context, buf_addr, cqe * sizeof (struct mlx4_cqe),
|
*umem = ib_umem_get(context, buf_addr, cqe * cqe_size,
|
||||||
IB_ACCESS_LOCAL_WRITE, 1);
|
IB_ACCESS_LOCAL_WRITE, 1);
|
||||||
if (IS_ERR(*umem))
|
if (IS_ERR(*umem))
|
||||||
return PTR_ERR(*umem);
|
return PTR_ERR(*umem);
|
||||||
@ -331,16 +333,23 @@ static void mlx4_ib_cq_resize_copy_cqes(struct mlx4_ib_cq *cq)
|
|||||||
{
|
{
|
||||||
struct mlx4_cqe *cqe, *new_cqe;
|
struct mlx4_cqe *cqe, *new_cqe;
|
||||||
int i;
|
int i;
|
||||||
|
int cqe_size = cq->buf.entry_size;
|
||||||
|
int cqe_inc = cqe_size == 64 ? 1 : 0;
|
||||||
|
|
||||||
i = cq->mcq.cons_index;
|
i = cq->mcq.cons_index;
|
||||||
cqe = get_cqe(cq, i & cq->ibcq.cqe);
|
cqe = get_cqe(cq, i & cq->ibcq.cqe);
|
||||||
|
cqe += cqe_inc;
|
||||||
|
|
||||||
while ((cqe->owner_sr_opcode & MLX4_CQE_OPCODE_MASK) != MLX4_CQE_OPCODE_RESIZE) {
|
while ((cqe->owner_sr_opcode & MLX4_CQE_OPCODE_MASK) != MLX4_CQE_OPCODE_RESIZE) {
|
||||||
new_cqe = get_cqe_from_buf(&cq->resize_buf->buf,
|
new_cqe = get_cqe_from_buf(&cq->resize_buf->buf,
|
||||||
(i + 1) & cq->resize_buf->cqe);
|
(i + 1) & cq->resize_buf->cqe);
|
||||||
memcpy(new_cqe, get_cqe(cq, i & cq->ibcq.cqe), sizeof(struct mlx4_cqe));
|
memcpy(new_cqe, get_cqe(cq, i & cq->ibcq.cqe), cqe_size);
|
||||||
|
new_cqe += cqe_inc;
|
||||||
|
|
||||||
new_cqe->owner_sr_opcode = (cqe->owner_sr_opcode & ~MLX4_CQE_OWNER_MASK) |
|
new_cqe->owner_sr_opcode = (cqe->owner_sr_opcode & ~MLX4_CQE_OWNER_MASK) |
|
||||||
(((i + 1) & (cq->resize_buf->cqe + 1)) ? MLX4_CQE_OWNER_MASK : 0);
|
(((i + 1) & (cq->resize_buf->cqe + 1)) ? MLX4_CQE_OWNER_MASK : 0);
|
||||||
cqe = get_cqe(cq, ++i & cq->ibcq.cqe);
|
cqe = get_cqe(cq, ++i & cq->ibcq.cqe);
|
||||||
|
cqe += cqe_inc;
|
||||||
}
|
}
|
||||||
++cq->mcq.cons_index;
|
++cq->mcq.cons_index;
|
||||||
}
|
}
|
||||||
@ -438,6 +447,7 @@ err_buf:
|
|||||||
|
|
||||||
out:
|
out:
|
||||||
mutex_unlock(&cq->resize_mutex);
|
mutex_unlock(&cq->resize_mutex);
|
||||||
|
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -586,6 +596,9 @@ repoll:
|
|||||||
if (!cqe)
|
if (!cqe)
|
||||||
return -EAGAIN;
|
return -EAGAIN;
|
||||||
|
|
||||||
|
if (cq->buf.entry_size == 64)
|
||||||
|
cqe++;
|
||||||
|
|
||||||
++cq->mcq.cons_index;
|
++cq->mcq.cons_index;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@ -807,6 +820,7 @@ void __mlx4_ib_cq_clean(struct mlx4_ib_cq *cq, u32 qpn, struct mlx4_ib_srq *srq)
|
|||||||
int nfreed = 0;
|
int nfreed = 0;
|
||||||
struct mlx4_cqe *cqe, *dest;
|
struct mlx4_cqe *cqe, *dest;
|
||||||
u8 owner_bit;
|
u8 owner_bit;
|
||||||
|
int cqe_inc = cq->buf.entry_size == 64 ? 1 : 0;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* First we need to find the current producer index, so we
|
* First we need to find the current producer index, so we
|
||||||
@ -825,12 +839,16 @@ void __mlx4_ib_cq_clean(struct mlx4_ib_cq *cq, u32 qpn, struct mlx4_ib_srq *srq)
|
|||||||
*/
|
*/
|
||||||
while ((int) --prod_index - (int) cq->mcq.cons_index >= 0) {
|
while ((int) --prod_index - (int) cq->mcq.cons_index >= 0) {
|
||||||
cqe = get_cqe(cq, prod_index & cq->ibcq.cqe);
|
cqe = get_cqe(cq, prod_index & cq->ibcq.cqe);
|
||||||
|
cqe += cqe_inc;
|
||||||
|
|
||||||
if ((be32_to_cpu(cqe->vlan_my_qpn) & MLX4_CQE_QPN_MASK) == qpn) {
|
if ((be32_to_cpu(cqe->vlan_my_qpn) & MLX4_CQE_QPN_MASK) == qpn) {
|
||||||
if (srq && !(cqe->owner_sr_opcode & MLX4_CQE_IS_SEND_MASK))
|
if (srq && !(cqe->owner_sr_opcode & MLX4_CQE_IS_SEND_MASK))
|
||||||
mlx4_ib_free_srq_wqe(srq, be16_to_cpu(cqe->wqe_index));
|
mlx4_ib_free_srq_wqe(srq, be16_to_cpu(cqe->wqe_index));
|
||||||
++nfreed;
|
++nfreed;
|
||||||
} else if (nfreed) {
|
} else if (nfreed) {
|
||||||
dest = get_cqe(cq, (prod_index + nfreed) & cq->ibcq.cqe);
|
dest = get_cqe(cq, (prod_index + nfreed) & cq->ibcq.cqe);
|
||||||
|
dest += cqe_inc;
|
||||||
|
|
||||||
owner_bit = dest->owner_sr_opcode & MLX4_CQE_OWNER_MASK;
|
owner_bit = dest->owner_sr_opcode & MLX4_CQE_OWNER_MASK;
|
||||||
memcpy(dest, cqe, sizeof *cqe);
|
memcpy(dest, cqe, sizeof *cqe);
|
||||||
dest->owner_sr_opcode = owner_bit |
|
dest->owner_sr_opcode = owner_bit |
|
||||||
|
@ -563,15 +563,24 @@ static struct ib_ucontext *mlx4_ib_alloc_ucontext(struct ib_device *ibdev,
|
|||||||
{
|
{
|
||||||
struct mlx4_ib_dev *dev = to_mdev(ibdev);
|
struct mlx4_ib_dev *dev = to_mdev(ibdev);
|
||||||
struct mlx4_ib_ucontext *context;
|
struct mlx4_ib_ucontext *context;
|
||||||
|
struct mlx4_ib_alloc_ucontext_resp_v3 resp_v3;
|
||||||
struct mlx4_ib_alloc_ucontext_resp resp;
|
struct mlx4_ib_alloc_ucontext_resp resp;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
if (!dev->ib_active)
|
if (!dev->ib_active)
|
||||||
return ERR_PTR(-EAGAIN);
|
return ERR_PTR(-EAGAIN);
|
||||||
|
|
||||||
resp.qp_tab_size = dev->dev->caps.num_qps;
|
if (ibdev->uverbs_abi_ver == MLX4_IB_UVERBS_NO_DEV_CAPS_ABI_VERSION) {
|
||||||
resp.bf_reg_size = dev->dev->caps.bf_reg_size;
|
resp_v3.qp_tab_size = dev->dev->caps.num_qps;
|
||||||
resp.bf_regs_per_page = dev->dev->caps.bf_regs_per_page;
|
resp_v3.bf_reg_size = dev->dev->caps.bf_reg_size;
|
||||||
|
resp_v3.bf_regs_per_page = dev->dev->caps.bf_regs_per_page;
|
||||||
|
} else {
|
||||||
|
resp.dev_caps = dev->dev->caps.userspace_caps;
|
||||||
|
resp.qp_tab_size = dev->dev->caps.num_qps;
|
||||||
|
resp.bf_reg_size = dev->dev->caps.bf_reg_size;
|
||||||
|
resp.bf_regs_per_page = dev->dev->caps.bf_regs_per_page;
|
||||||
|
resp.cqe_size = dev->dev->caps.cqe_size;
|
||||||
|
}
|
||||||
|
|
||||||
context = kmalloc(sizeof *context, GFP_KERNEL);
|
context = kmalloc(sizeof *context, GFP_KERNEL);
|
||||||
if (!context)
|
if (!context)
|
||||||
@ -586,7 +595,11 @@ static struct ib_ucontext *mlx4_ib_alloc_ucontext(struct ib_device *ibdev,
|
|||||||
INIT_LIST_HEAD(&context->db_page_list);
|
INIT_LIST_HEAD(&context->db_page_list);
|
||||||
mutex_init(&context->db_page_mutex);
|
mutex_init(&context->db_page_mutex);
|
||||||
|
|
||||||
err = ib_copy_to_udata(udata, &resp, sizeof resp);
|
if (ibdev->uverbs_abi_ver == MLX4_IB_UVERBS_NO_DEV_CAPS_ABI_VERSION)
|
||||||
|
err = ib_copy_to_udata(udata, &resp_v3, sizeof(resp_v3));
|
||||||
|
else
|
||||||
|
err = ib_copy_to_udata(udata, &resp, sizeof(resp));
|
||||||
|
|
||||||
if (err) {
|
if (err) {
|
||||||
mlx4_uar_free(to_mdev(ibdev)->dev, &context->uar);
|
mlx4_uar_free(to_mdev(ibdev)->dev, &context->uar);
|
||||||
kfree(context);
|
kfree(context);
|
||||||
@ -1342,7 +1355,11 @@ static void *mlx4_ib_add(struct mlx4_dev *dev)
|
|||||||
ibdev->ib_dev.num_comp_vectors = dev->caps.num_comp_vectors;
|
ibdev->ib_dev.num_comp_vectors = dev->caps.num_comp_vectors;
|
||||||
ibdev->ib_dev.dma_device = &dev->pdev->dev;
|
ibdev->ib_dev.dma_device = &dev->pdev->dev;
|
||||||
|
|
||||||
ibdev->ib_dev.uverbs_abi_ver = MLX4_IB_UVERBS_ABI_VERSION;
|
if (dev->caps.userspace_caps)
|
||||||
|
ibdev->ib_dev.uverbs_abi_ver = MLX4_IB_UVERBS_ABI_VERSION;
|
||||||
|
else
|
||||||
|
ibdev->ib_dev.uverbs_abi_ver = MLX4_IB_UVERBS_NO_DEV_CAPS_ABI_VERSION;
|
||||||
|
|
||||||
ibdev->ib_dev.uverbs_cmd_mask =
|
ibdev->ib_dev.uverbs_cmd_mask =
|
||||||
(1ull << IB_USER_VERBS_CMD_GET_CONTEXT) |
|
(1ull << IB_USER_VERBS_CMD_GET_CONTEXT) |
|
||||||
(1ull << IB_USER_VERBS_CMD_QUERY_DEVICE) |
|
(1ull << IB_USER_VERBS_CMD_QUERY_DEVICE) |
|
||||||
|
@ -90,6 +90,7 @@ struct mlx4_ib_xrcd {
|
|||||||
struct mlx4_ib_cq_buf {
|
struct mlx4_ib_cq_buf {
|
||||||
struct mlx4_buf buf;
|
struct mlx4_buf buf;
|
||||||
struct mlx4_mtt mtt;
|
struct mlx4_mtt mtt;
|
||||||
|
int entry_size;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct mlx4_ib_cq_resize {
|
struct mlx4_ib_cq_resize {
|
||||||
|
@ -40,7 +40,9 @@
|
|||||||
* Increment this value if any changes that break userspace ABI
|
* Increment this value if any changes that break userspace ABI
|
||||||
* compatibility are made.
|
* compatibility are made.
|
||||||
*/
|
*/
|
||||||
#define MLX4_IB_UVERBS_ABI_VERSION 3
|
|
||||||
|
#define MLX4_IB_UVERBS_NO_DEV_CAPS_ABI_VERSION 3
|
||||||
|
#define MLX4_IB_UVERBS_ABI_VERSION 4
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Make sure that all structs defined in this file remain laid out so
|
* Make sure that all structs defined in this file remain laid out so
|
||||||
@ -50,12 +52,20 @@
|
|||||||
* instead.
|
* instead.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
struct mlx4_ib_alloc_ucontext_resp {
|
struct mlx4_ib_alloc_ucontext_resp_v3 {
|
||||||
__u32 qp_tab_size;
|
__u32 qp_tab_size;
|
||||||
__u16 bf_reg_size;
|
__u16 bf_reg_size;
|
||||||
__u16 bf_regs_per_page;
|
__u16 bf_regs_per_page;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct mlx4_ib_alloc_ucontext_resp {
|
||||||
|
__u32 dev_caps;
|
||||||
|
__u32 qp_tab_size;
|
||||||
|
__u16 bf_reg_size;
|
||||||
|
__u16 bf_regs_per_page;
|
||||||
|
__u32 cqe_size;
|
||||||
|
};
|
||||||
|
|
||||||
struct mlx4_ib_alloc_pd_resp {
|
struct mlx4_ib_alloc_pd_resp {
|
||||||
__u32 pdn;
|
__u32 pdn;
|
||||||
__u32 reserved;
|
__u32 reserved;
|
||||||
|
@ -629,11 +629,9 @@ static void build_rdma0_msg(struct nes_cm_node *cm_node, struct nes_qp **nesqp_a
|
|||||||
|
|
||||||
case SEND_RDMA_READ_ZERO:
|
case SEND_RDMA_READ_ZERO:
|
||||||
default:
|
default:
|
||||||
if (cm_node->send_rdma0_op != SEND_RDMA_READ_ZERO) {
|
if (cm_node->send_rdma0_op != SEND_RDMA_READ_ZERO)
|
||||||
printk(KERN_ERR "%s[%u]: Unsupported RDMA0 len operation=%u\n",
|
WARN(1, "Unsupported RDMA0 len operation=%u\n",
|
||||||
__func__, __LINE__, cm_node->send_rdma0_op);
|
cm_node->send_rdma0_op);
|
||||||
WARN_ON(1);
|
|
||||||
}
|
|
||||||
nes_debug(NES_DBG_CM, "Sending first rdma operation.\n");
|
nes_debug(NES_DBG_CM, "Sending first rdma operation.\n");
|
||||||
wqe->wqe_words[NES_IWARP_SQ_WQE_MISC_IDX] =
|
wqe->wqe_words[NES_IWARP_SQ_WQE_MISC_IDX] =
|
||||||
cpu_to_le32(NES_IWARP_SQ_OP_RDMAR);
|
cpu_to_le32(NES_IWARP_SQ_OP_RDMAR);
|
||||||
|
@ -210,6 +210,9 @@ static struct sk_buff *nes_get_next_skb(struct nes_device *nesdev, struct nes_qp
|
|||||||
}
|
}
|
||||||
|
|
||||||
while (1) {
|
while (1) {
|
||||||
|
if (skb_queue_empty(&nesqp->pau_list))
|
||||||
|
goto out;
|
||||||
|
|
||||||
seq = nes_get_seq(skb, ack, wnd, fin_rcvd, rst_rcvd);
|
seq = nes_get_seq(skb, ack, wnd, fin_rcvd, rst_rcvd);
|
||||||
if (seq == nextseq) {
|
if (seq == nextseq) {
|
||||||
if (skb->len || processacks)
|
if (skb->len || processacks)
|
||||||
@ -218,14 +221,13 @@ static struct sk_buff *nes_get_next_skb(struct nes_device *nesdev, struct nes_qp
|
|||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (skb->next == (struct sk_buff *)&nesqp->pau_list)
|
|
||||||
goto out;
|
|
||||||
|
|
||||||
old_skb = skb;
|
old_skb = skb;
|
||||||
skb = skb->next;
|
skb = skb->next;
|
||||||
skb_unlink(old_skb, &nesqp->pau_list);
|
skb_unlink(old_skb, &nesqp->pau_list);
|
||||||
nes_mgt_free_skb(nesdev, old_skb, PCI_DMA_TODEVICE);
|
nes_mgt_free_skb(nesdev, old_skb, PCI_DMA_TODEVICE);
|
||||||
nes_rem_ref_cm_node(nesqp->cm_node);
|
nes_rem_ref_cm_node(nesqp->cm_node);
|
||||||
|
if (skb == (struct sk_buff *)&nesqp->pau_list)
|
||||||
|
goto out;
|
||||||
}
|
}
|
||||||
return skb;
|
return skb;
|
||||||
|
|
||||||
@ -245,7 +247,6 @@ static int get_fpdu_info(struct nes_device *nesdev, struct nes_qp *nesqp,
|
|||||||
struct nes_rskb_cb *cb;
|
struct nes_rskb_cb *cb;
|
||||||
struct pau_fpdu_info *fpdu_info = NULL;
|
struct pau_fpdu_info *fpdu_info = NULL;
|
||||||
struct pau_fpdu_frag frags[MAX_FPDU_FRAGS];
|
struct pau_fpdu_frag frags[MAX_FPDU_FRAGS];
|
||||||
unsigned long flags;
|
|
||||||
u32 fpdu_len = 0;
|
u32 fpdu_len = 0;
|
||||||
u32 tmp_len;
|
u32 tmp_len;
|
||||||
int frag_cnt = 0;
|
int frag_cnt = 0;
|
||||||
@ -260,12 +261,10 @@ static int get_fpdu_info(struct nes_device *nesdev, struct nes_qp *nesqp,
|
|||||||
|
|
||||||
*pau_fpdu_info = NULL;
|
*pau_fpdu_info = NULL;
|
||||||
|
|
||||||
spin_lock_irqsave(&nesqp->pau_lock, flags);
|
|
||||||
skb = nes_get_next_skb(nesdev, nesqp, NULL, nesqp->pau_rcv_nxt, &ack, &wnd, &fin_rcvd, &rst_rcvd);
|
skb = nes_get_next_skb(nesdev, nesqp, NULL, nesqp->pau_rcv_nxt, &ack, &wnd, &fin_rcvd, &rst_rcvd);
|
||||||
if (!skb) {
|
if (!skb)
|
||||||
spin_unlock_irqrestore(&nesqp->pau_lock, flags);
|
|
||||||
goto out;
|
goto out;
|
||||||
}
|
|
||||||
cb = (struct nes_rskb_cb *)&skb->cb[0];
|
cb = (struct nes_rskb_cb *)&skb->cb[0];
|
||||||
if (skb->len) {
|
if (skb->len) {
|
||||||
fpdu_len = be16_to_cpu(*(__be16 *) skb->data) + MPA_FRAMING;
|
fpdu_len = be16_to_cpu(*(__be16 *) skb->data) + MPA_FRAMING;
|
||||||
@ -290,10 +289,9 @@ static int get_fpdu_info(struct nes_device *nesdev, struct nes_qp *nesqp,
|
|||||||
|
|
||||||
skb = nes_get_next_skb(nesdev, nesqp, skb,
|
skb = nes_get_next_skb(nesdev, nesqp, skb,
|
||||||
nesqp->pau_rcv_nxt + frag_tot, &ack, &wnd, &fin_rcvd, &rst_rcvd);
|
nesqp->pau_rcv_nxt + frag_tot, &ack, &wnd, &fin_rcvd, &rst_rcvd);
|
||||||
if (!skb) {
|
if (!skb)
|
||||||
spin_unlock_irqrestore(&nesqp->pau_lock, flags);
|
|
||||||
goto out;
|
goto out;
|
||||||
} else if (rst_rcvd) {
|
if (rst_rcvd) {
|
||||||
/* rst received in the middle of fpdu */
|
/* rst received in the middle of fpdu */
|
||||||
for (; i >= 0; i--) {
|
for (; i >= 0; i--) {
|
||||||
skb_unlink(frags[i].skb, &nesqp->pau_list);
|
skb_unlink(frags[i].skb, &nesqp->pau_list);
|
||||||
@ -320,8 +318,6 @@ static int get_fpdu_info(struct nes_device *nesdev, struct nes_qp *nesqp,
|
|||||||
frag_cnt = 1;
|
frag_cnt = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
spin_unlock_irqrestore(&nesqp->pau_lock, flags);
|
|
||||||
|
|
||||||
/* Found one */
|
/* Found one */
|
||||||
fpdu_info = kzalloc(sizeof(*fpdu_info), GFP_ATOMIC);
|
fpdu_info = kzalloc(sizeof(*fpdu_info), GFP_ATOMIC);
|
||||||
if (fpdu_info == NULL) {
|
if (fpdu_info == NULL) {
|
||||||
@ -383,9 +379,8 @@ static int get_fpdu_info(struct nes_device *nesdev, struct nes_qp *nesqp,
|
|||||||
|
|
||||||
if (frags[i].skb->len == 0) {
|
if (frags[i].skb->len == 0) {
|
||||||
/* Pull skb off the list - it will be freed in the callback */
|
/* Pull skb off the list - it will be freed in the callback */
|
||||||
spin_lock_irqsave(&nesqp->pau_lock, flags);
|
if (!skb_queue_empty(&nesqp->pau_list))
|
||||||
skb_unlink(frags[i].skb, &nesqp->pau_list);
|
skb_unlink(frags[i].skb, &nesqp->pau_list);
|
||||||
spin_unlock_irqrestore(&nesqp->pau_lock, flags);
|
|
||||||
} else {
|
} else {
|
||||||
/* Last skb still has data so update the seq */
|
/* Last skb still has data so update the seq */
|
||||||
iph = (struct iphdr *)(cb->data_start + ETH_HLEN);
|
iph = (struct iphdr *)(cb->data_start + ETH_HLEN);
|
||||||
@ -414,14 +409,18 @@ static int forward_fpdus(struct nes_vnic *nesvnic, struct nes_qp *nesqp)
|
|||||||
struct pau_fpdu_info *fpdu_info;
|
struct pau_fpdu_info *fpdu_info;
|
||||||
struct nes_hw_cqp_wqe *cqp_wqe;
|
struct nes_hw_cqp_wqe *cqp_wqe;
|
||||||
struct nes_cqp_request *cqp_request;
|
struct nes_cqp_request *cqp_request;
|
||||||
|
unsigned long flags;
|
||||||
u64 u64tmp;
|
u64 u64tmp;
|
||||||
u32 u32tmp;
|
u32 u32tmp;
|
||||||
int rc;
|
int rc;
|
||||||
|
|
||||||
while (1) {
|
while (1) {
|
||||||
|
spin_lock_irqsave(&nesqp->pau_lock, flags);
|
||||||
rc = get_fpdu_info(nesdev, nesqp, &fpdu_info);
|
rc = get_fpdu_info(nesdev, nesqp, &fpdu_info);
|
||||||
if (fpdu_info == NULL)
|
if (rc || (fpdu_info == NULL)) {
|
||||||
|
spin_unlock_irqrestore(&nesqp->pau_lock, flags);
|
||||||
return rc;
|
return rc;
|
||||||
|
}
|
||||||
|
|
||||||
cqp_request = fpdu_info->cqp_request;
|
cqp_request = fpdu_info->cqp_request;
|
||||||
cqp_wqe = &cqp_request->cqp_wqe;
|
cqp_wqe = &cqp_request->cqp_wqe;
|
||||||
@ -447,7 +446,7 @@ static int forward_fpdus(struct nes_vnic *nesvnic, struct nes_qp *nesqp)
|
|||||||
set_wqe_32bit_value(cqp_wqe->wqe_words, NES_NIC_SQ_WQE_FRAG0_LOW_IDX,
|
set_wqe_32bit_value(cqp_wqe->wqe_words, NES_NIC_SQ_WQE_FRAG0_LOW_IDX,
|
||||||
lower_32_bits(u64tmp));
|
lower_32_bits(u64tmp));
|
||||||
set_wqe_32bit_value(cqp_wqe->wqe_words, NES_NIC_SQ_WQE_FRAG0_HIGH_IDX,
|
set_wqe_32bit_value(cqp_wqe->wqe_words, NES_NIC_SQ_WQE_FRAG0_HIGH_IDX,
|
||||||
upper_32_bits(u64tmp >> 32));
|
upper_32_bits(u64tmp));
|
||||||
|
|
||||||
set_wqe_32bit_value(cqp_wqe->wqe_words, NES_NIC_SQ_WQE_FRAG1_LOW_IDX,
|
set_wqe_32bit_value(cqp_wqe->wqe_words, NES_NIC_SQ_WQE_FRAG1_LOW_IDX,
|
||||||
lower_32_bits(fpdu_info->frags[0].physaddr));
|
lower_32_bits(fpdu_info->frags[0].physaddr));
|
||||||
@ -475,6 +474,7 @@ static int forward_fpdus(struct nes_vnic *nesvnic, struct nes_qp *nesqp)
|
|||||||
|
|
||||||
atomic_set(&cqp_request->refcount, 1);
|
atomic_set(&cqp_request->refcount, 1);
|
||||||
nes_post_cqp_request(nesdev, cqp_request);
|
nes_post_cqp_request(nesdev, cqp_request);
|
||||||
|
spin_unlock_irqrestore(&nesqp->pau_lock, flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
@ -649,11 +649,9 @@ static void nes_chg_qh_handler(struct nes_device *nesdev, struct nes_cqp_request
|
|||||||
nesqp = qh_chg->nesqp;
|
nesqp = qh_chg->nesqp;
|
||||||
|
|
||||||
/* Should we handle the bad completion */
|
/* Should we handle the bad completion */
|
||||||
if (cqp_request->major_code) {
|
if (cqp_request->major_code)
|
||||||
printk(KERN_ERR PFX "Invalid cqp_request major_code=0x%x\n",
|
WARN(1, PFX "Invalid cqp_request major_code=0x%x\n",
|
||||||
cqp_request->major_code);
|
cqp_request->major_code);
|
||||||
WARN_ON(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
switch (nesqp->pau_state) {
|
switch (nesqp->pau_state) {
|
||||||
case PAU_DEL_QH:
|
case PAU_DEL_QH:
|
||||||
|
@ -944,12 +944,13 @@ static void nes_netdev_set_multicast_list(struct net_device *netdev)
|
|||||||
addr,
|
addr,
|
||||||
perfect_filter_register_address+(mc_index * 8),
|
perfect_filter_register_address+(mc_index * 8),
|
||||||
mc_nic_index);
|
mc_nic_index);
|
||||||
macaddr_high = ((u16) addr[0]) << 8;
|
macaddr_high = ((u8) addr[0]) << 8;
|
||||||
macaddr_high += (u16) addr[1];
|
macaddr_high += (u8) addr[1];
|
||||||
macaddr_low = ((u32) addr[2]) << 24;
|
macaddr_low = ((u8) addr[2]) << 24;
|
||||||
macaddr_low += ((u32) addr[3]) << 16;
|
macaddr_low += ((u8) addr[3]) << 16;
|
||||||
macaddr_low += ((u32) addr[4]) << 8;
|
macaddr_low += ((u8) addr[4]) << 8;
|
||||||
macaddr_low += (u32) addr[5];
|
macaddr_low += (u8) addr[5];
|
||||||
|
|
||||||
nes_write_indexed(nesdev,
|
nes_write_indexed(nesdev,
|
||||||
perfect_filter_register_address+(mc_index * 8),
|
perfect_filter_register_address+(mc_index * 8),
|
||||||
macaddr_low);
|
macaddr_low);
|
||||||
|
@ -1755,7 +1755,7 @@ int mlx4_multi_func_init(struct mlx4_dev *dev)
|
|||||||
spin_lock_init(&s_state->lock);
|
spin_lock_init(&s_state->lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
memset(&priv->mfunc.master.cmd_eqe, 0, sizeof(struct mlx4_eqe));
|
memset(&priv->mfunc.master.cmd_eqe, 0, dev->caps.eqe_size);
|
||||||
priv->mfunc.master.cmd_eqe.type = MLX4_EVENT_TYPE_CMD;
|
priv->mfunc.master.cmd_eqe.type = MLX4_EVENT_TYPE_CMD;
|
||||||
INIT_WORK(&priv->mfunc.master.comm_work,
|
INIT_WORK(&priv->mfunc.master.comm_work,
|
||||||
mlx4_master_comm_channel);
|
mlx4_master_comm_channel);
|
||||||
|
@ -51,7 +51,7 @@ int mlx4_en_create_cq(struct mlx4_en_priv *priv,
|
|||||||
int err;
|
int err;
|
||||||
|
|
||||||
cq->size = entries;
|
cq->size = entries;
|
||||||
cq->buf_size = cq->size * sizeof(struct mlx4_cqe);
|
cq->buf_size = cq->size * mdev->dev->caps.cqe_size;
|
||||||
|
|
||||||
cq->ring = ring;
|
cq->ring = ring;
|
||||||
cq->is_tx = mode;
|
cq->is_tx = mode;
|
||||||
|
@ -1600,6 +1600,7 @@ int mlx4_en_init_netdev(struct mlx4_en_dev *mdev, int port,
|
|||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
priv->rx_ring_num = prof->rx_ring_num;
|
priv->rx_ring_num = prof->rx_ring_num;
|
||||||
|
priv->cqe_factor = (mdev->dev->caps.cqe_size == 64) ? 1 : 0;
|
||||||
priv->mac_index = -1;
|
priv->mac_index = -1;
|
||||||
priv->msg_enable = MLX4_EN_MSG_LEVEL;
|
priv->msg_enable = MLX4_EN_MSG_LEVEL;
|
||||||
spin_lock_init(&priv->stats_lock);
|
spin_lock_init(&priv->stats_lock);
|
||||||
|
@ -566,6 +566,7 @@ int mlx4_en_process_rx_cq(struct net_device *dev, struct mlx4_en_cq *cq, int bud
|
|||||||
struct ethhdr *ethh;
|
struct ethhdr *ethh;
|
||||||
dma_addr_t dma;
|
dma_addr_t dma;
|
||||||
u64 s_mac;
|
u64 s_mac;
|
||||||
|
int factor = priv->cqe_factor;
|
||||||
|
|
||||||
if (!priv->port_up)
|
if (!priv->port_up)
|
||||||
return 0;
|
return 0;
|
||||||
@ -574,7 +575,7 @@ int mlx4_en_process_rx_cq(struct net_device *dev, struct mlx4_en_cq *cq, int bud
|
|||||||
* descriptor offset can be deduced from the CQE index instead of
|
* descriptor offset can be deduced from the CQE index instead of
|
||||||
* reading 'cqe->index' */
|
* reading 'cqe->index' */
|
||||||
index = cq->mcq.cons_index & ring->size_mask;
|
index = cq->mcq.cons_index & ring->size_mask;
|
||||||
cqe = &cq->buf[index];
|
cqe = &cq->buf[(index << factor) + factor];
|
||||||
|
|
||||||
/* Process all completed CQEs */
|
/* Process all completed CQEs */
|
||||||
while (XNOR(cqe->owner_sr_opcode & MLX4_CQE_OWNER_MASK,
|
while (XNOR(cqe->owner_sr_opcode & MLX4_CQE_OWNER_MASK,
|
||||||
@ -709,7 +710,7 @@ next:
|
|||||||
|
|
||||||
++cq->mcq.cons_index;
|
++cq->mcq.cons_index;
|
||||||
index = (cq->mcq.cons_index) & ring->size_mask;
|
index = (cq->mcq.cons_index) & ring->size_mask;
|
||||||
cqe = &cq->buf[index];
|
cqe = &cq->buf[(index << factor) + factor];
|
||||||
if (++polled == budget) {
|
if (++polled == budget) {
|
||||||
/* We are here because we reached the NAPI budget -
|
/* We are here because we reached the NAPI budget -
|
||||||
* flush only pending LRO sessions */
|
* flush only pending LRO sessions */
|
||||||
|
@ -315,12 +315,13 @@ static void mlx4_en_process_tx_cq(struct net_device *dev, struct mlx4_en_cq *cq)
|
|||||||
struct mlx4_cqe *buf = cq->buf;
|
struct mlx4_cqe *buf = cq->buf;
|
||||||
u32 packets = 0;
|
u32 packets = 0;
|
||||||
u32 bytes = 0;
|
u32 bytes = 0;
|
||||||
|
int factor = priv->cqe_factor;
|
||||||
|
|
||||||
if (!priv->port_up)
|
if (!priv->port_up)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
index = cons_index & size_mask;
|
index = cons_index & size_mask;
|
||||||
cqe = &buf[index];
|
cqe = &buf[(index << factor) + factor];
|
||||||
ring_index = ring->cons & size_mask;
|
ring_index = ring->cons & size_mask;
|
||||||
|
|
||||||
/* Process all completed CQEs */
|
/* Process all completed CQEs */
|
||||||
@ -349,7 +350,7 @@ static void mlx4_en_process_tx_cq(struct net_device *dev, struct mlx4_en_cq *cq)
|
|||||||
|
|
||||||
++cons_index;
|
++cons_index;
|
||||||
index = cons_index & size_mask;
|
index = cons_index & size_mask;
|
||||||
cqe = &buf[index];
|
cqe = &buf[(index << factor) + factor];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -101,15 +101,21 @@ static void eq_set_ci(struct mlx4_eq *eq, int req_not)
|
|||||||
mb();
|
mb();
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct mlx4_eqe *get_eqe(struct mlx4_eq *eq, u32 entry)
|
static struct mlx4_eqe *get_eqe(struct mlx4_eq *eq, u32 entry, u8 eqe_factor)
|
||||||
{
|
{
|
||||||
unsigned long off = (entry & (eq->nent - 1)) * MLX4_EQ_ENTRY_SIZE;
|
/* (entry & (eq->nent - 1)) gives us a cyclic array */
|
||||||
return eq->page_list[off / PAGE_SIZE].buf + off % PAGE_SIZE;
|
unsigned long offset = (entry & (eq->nent - 1)) * (MLX4_EQ_ENTRY_SIZE << eqe_factor);
|
||||||
|
/* CX3 is capable of extending the EQE from 32 to 64 bytes.
|
||||||
|
* When this feature is enabled, the first (in the lower addresses)
|
||||||
|
* 32 bytes in the 64 byte EQE are reserved and the next 32 bytes
|
||||||
|
* contain the legacy EQE information.
|
||||||
|
*/
|
||||||
|
return eq->page_list[offset / PAGE_SIZE].buf + (offset + (eqe_factor ? MLX4_EQ_ENTRY_SIZE : 0)) % PAGE_SIZE;
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct mlx4_eqe *next_eqe_sw(struct mlx4_eq *eq)
|
static struct mlx4_eqe *next_eqe_sw(struct mlx4_eq *eq, u8 eqe_factor)
|
||||||
{
|
{
|
||||||
struct mlx4_eqe *eqe = get_eqe(eq, eq->cons_index);
|
struct mlx4_eqe *eqe = get_eqe(eq, eq->cons_index, eqe_factor);
|
||||||
return !!(eqe->owner & 0x80) ^ !!(eq->cons_index & eq->nent) ? NULL : eqe;
|
return !!(eqe->owner & 0x80) ^ !!(eq->cons_index & eq->nent) ? NULL : eqe;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -177,7 +183,7 @@ static void slave_event(struct mlx4_dev *dev, u8 slave, struct mlx4_eqe *eqe)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
memcpy(s_eqe, eqe, sizeof(struct mlx4_eqe) - 1);
|
memcpy(s_eqe, eqe, dev->caps.eqe_size - 1);
|
||||||
s_eqe->slave_id = slave;
|
s_eqe->slave_id = slave;
|
||||||
/* ensure all information is written before setting the ownersip bit */
|
/* ensure all information is written before setting the ownersip bit */
|
||||||
wmb();
|
wmb();
|
||||||
@ -441,7 +447,7 @@ static int mlx4_eq_int(struct mlx4_dev *dev, struct mlx4_eq *eq)
|
|||||||
int i;
|
int i;
|
||||||
enum slave_port_gen_event gen_event;
|
enum slave_port_gen_event gen_event;
|
||||||
|
|
||||||
while ((eqe = next_eqe_sw(eq))) {
|
while ((eqe = next_eqe_sw(eq, dev->caps.eqe_factor))) {
|
||||||
/*
|
/*
|
||||||
* Make sure we read EQ entry contents after we've
|
* Make sure we read EQ entry contents after we've
|
||||||
* checked the ownership bit.
|
* checked the ownership bit.
|
||||||
@ -864,7 +870,8 @@ static int mlx4_create_eq(struct mlx4_dev *dev, int nent,
|
|||||||
|
|
||||||
eq->dev = dev;
|
eq->dev = dev;
|
||||||
eq->nent = roundup_pow_of_two(max(nent, 2));
|
eq->nent = roundup_pow_of_two(max(nent, 2));
|
||||||
npages = PAGE_ALIGN(eq->nent * MLX4_EQ_ENTRY_SIZE) / PAGE_SIZE;
|
/* CX3 is capable of extending the CQE/EQE from 32 to 64 bytes */
|
||||||
|
npages = PAGE_ALIGN(eq->nent * (MLX4_EQ_ENTRY_SIZE << dev->caps.eqe_factor)) / PAGE_SIZE;
|
||||||
|
|
||||||
eq->page_list = kmalloc(npages * sizeof *eq->page_list,
|
eq->page_list = kmalloc(npages * sizeof *eq->page_list,
|
||||||
GFP_KERNEL);
|
GFP_KERNEL);
|
||||||
@ -966,8 +973,9 @@ static void mlx4_free_eq(struct mlx4_dev *dev,
|
|||||||
struct mlx4_priv *priv = mlx4_priv(dev);
|
struct mlx4_priv *priv = mlx4_priv(dev);
|
||||||
struct mlx4_cmd_mailbox *mailbox;
|
struct mlx4_cmd_mailbox *mailbox;
|
||||||
int err;
|
int err;
|
||||||
int npages = PAGE_ALIGN(MLX4_EQ_ENTRY_SIZE * eq->nent) / PAGE_SIZE;
|
|
||||||
int i;
|
int i;
|
||||||
|
/* CX3 is capable of extending the CQE/EQE from 32 to 64 bytes */
|
||||||
|
int npages = PAGE_ALIGN((MLX4_EQ_ENTRY_SIZE << dev->caps.eqe_factor) * eq->nent) / PAGE_SIZE;
|
||||||
|
|
||||||
mailbox = mlx4_alloc_cmd_mailbox(dev);
|
mailbox = mlx4_alloc_cmd_mailbox(dev);
|
||||||
if (IS_ERR(mailbox))
|
if (IS_ERR(mailbox))
|
||||||
|
@ -110,6 +110,8 @@ static void dump_dev_cap_flags(struct mlx4_dev *dev, u64 flags)
|
|||||||
[42] = "Multicast VEP steering support",
|
[42] = "Multicast VEP steering support",
|
||||||
[48] = "Counters support",
|
[48] = "Counters support",
|
||||||
[59] = "Port management change event support",
|
[59] = "Port management change event support",
|
||||||
|
[61] = "64 byte EQE support",
|
||||||
|
[62] = "64 byte CQE support",
|
||||||
};
|
};
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
@ -235,7 +237,7 @@ int mlx4_QUERY_FUNC_CAP_wrapper(struct mlx4_dev *dev, int slave,
|
|||||||
field = dev->caps.num_ports;
|
field = dev->caps.num_ports;
|
||||||
MLX4_PUT(outbox->buf, field, QUERY_FUNC_CAP_NUM_PORTS_OFFSET);
|
MLX4_PUT(outbox->buf, field, QUERY_FUNC_CAP_NUM_PORTS_OFFSET);
|
||||||
|
|
||||||
size = 0; /* no PF behaviour is set for now */
|
size = dev->caps.function_caps; /* set PF behaviours */
|
||||||
MLX4_PUT(outbox->buf, size, QUERY_FUNC_CAP_PF_BHVR_OFFSET);
|
MLX4_PUT(outbox->buf, size, QUERY_FUNC_CAP_PF_BHVR_OFFSET);
|
||||||
|
|
||||||
field = 0; /* protected FMR support not available as yet */
|
field = 0; /* protected FMR support not available as yet */
|
||||||
@ -1237,6 +1239,24 @@ int mlx4_INIT_HCA(struct mlx4_dev *dev, struct mlx4_init_hca_param *param)
|
|||||||
if (dev->caps.flags & MLX4_DEV_CAP_FLAG_COUNTERS)
|
if (dev->caps.flags & MLX4_DEV_CAP_FLAG_COUNTERS)
|
||||||
*(inbox + INIT_HCA_FLAGS_OFFSET / 4) |= cpu_to_be32(1 << 4);
|
*(inbox + INIT_HCA_FLAGS_OFFSET / 4) |= cpu_to_be32(1 << 4);
|
||||||
|
|
||||||
|
/* CX3 is capable of extending CQEs/EQEs from 32 to 64 bytes */
|
||||||
|
if (dev->caps.flags & MLX4_DEV_CAP_FLAG_64B_EQE) {
|
||||||
|
*(inbox + INIT_HCA_EQE_CQE_OFFSETS / 4) |= cpu_to_be32(1 << 29);
|
||||||
|
dev->caps.eqe_size = 64;
|
||||||
|
dev->caps.eqe_factor = 1;
|
||||||
|
} else {
|
||||||
|
dev->caps.eqe_size = 32;
|
||||||
|
dev->caps.eqe_factor = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (dev->caps.flags & MLX4_DEV_CAP_FLAG_64B_CQE) {
|
||||||
|
*(inbox + INIT_HCA_EQE_CQE_OFFSETS / 4) |= cpu_to_be32(1 << 30);
|
||||||
|
dev->caps.cqe_size = 64;
|
||||||
|
dev->caps.userspace_caps |= MLX4_USER_DEV_CAP_64B_CQE;
|
||||||
|
} else {
|
||||||
|
dev->caps.cqe_size = 32;
|
||||||
|
}
|
||||||
|
|
||||||
/* QPC/EEC/CQC/EQC/RDMARC attributes */
|
/* QPC/EEC/CQC/EQC/RDMARC attributes */
|
||||||
|
|
||||||
MLX4_PUT(inbox, param->qpc_base, INIT_HCA_QPC_BASE_OFFSET);
|
MLX4_PUT(inbox, param->qpc_base, INIT_HCA_QPC_BASE_OFFSET);
|
||||||
@ -1319,6 +1339,7 @@ int mlx4_QUERY_HCA(struct mlx4_dev *dev,
|
|||||||
struct mlx4_cmd_mailbox *mailbox;
|
struct mlx4_cmd_mailbox *mailbox;
|
||||||
__be32 *outbox;
|
__be32 *outbox;
|
||||||
int err;
|
int err;
|
||||||
|
u8 byte_field;
|
||||||
|
|
||||||
#define QUERY_HCA_GLOBAL_CAPS_OFFSET 0x04
|
#define QUERY_HCA_GLOBAL_CAPS_OFFSET 0x04
|
||||||
|
|
||||||
@ -1370,6 +1391,13 @@ int mlx4_QUERY_HCA(struct mlx4_dev *dev,
|
|||||||
INIT_HCA_LOG_MC_TABLE_SZ_OFFSET);
|
INIT_HCA_LOG_MC_TABLE_SZ_OFFSET);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* CX3 is capable of extending CQEs/EQEs from 32 to 64 bytes */
|
||||||
|
MLX4_GET(byte_field, outbox, INIT_HCA_EQE_CQE_OFFSETS);
|
||||||
|
if (byte_field & 0x20) /* 64-bytes eqe enabled */
|
||||||
|
param->dev_cap_enabled |= MLX4_DEV_CAP_64B_EQE_ENABLED;
|
||||||
|
if (byte_field & 0x40) /* 64-bytes cqe enabled */
|
||||||
|
param->dev_cap_enabled |= MLX4_DEV_CAP_64B_CQE_ENABLED;
|
||||||
|
|
||||||
/* TPT attributes */
|
/* TPT attributes */
|
||||||
|
|
||||||
MLX4_GET(param->dmpt_base, outbox, INIT_HCA_DMPT_BASE_OFFSET);
|
MLX4_GET(param->dmpt_base, outbox, INIT_HCA_DMPT_BASE_OFFSET);
|
||||||
|
@ -172,6 +172,7 @@ struct mlx4_init_hca_param {
|
|||||||
u8 log_uar_sz;
|
u8 log_uar_sz;
|
||||||
u8 uar_page_sz; /* log pg sz in 4k chunks */
|
u8 uar_page_sz; /* log pg sz in 4k chunks */
|
||||||
u8 fs_hash_enable_bits;
|
u8 fs_hash_enable_bits;
|
||||||
|
u64 dev_cap_enabled;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct mlx4_init_ib_param {
|
struct mlx4_init_ib_param {
|
||||||
|
@ -95,8 +95,14 @@ MODULE_PARM_DESC(log_num_mgm_entry_size, "log mgm size, that defines the num"
|
|||||||
" Not in use with device managed"
|
" Not in use with device managed"
|
||||||
" flow steering");
|
" flow steering");
|
||||||
|
|
||||||
|
static bool enable_64b_cqe_eqe;
|
||||||
|
module_param(enable_64b_cqe_eqe, bool, 0444);
|
||||||
|
MODULE_PARM_DESC(enable_64b_cqe_eqe,
|
||||||
|
"Enable 64 byte CQEs/EQEs when the the FW supports this");
|
||||||
|
|
||||||
#define HCA_GLOBAL_CAP_MASK 0
|
#define HCA_GLOBAL_CAP_MASK 0
|
||||||
#define PF_CONTEXT_BEHAVIOUR_MASK 0
|
|
||||||
|
#define PF_CONTEXT_BEHAVIOUR_MASK MLX4_FUNC_CAP_64B_EQE_CQE
|
||||||
|
|
||||||
static char mlx4_version[] __devinitdata =
|
static char mlx4_version[] __devinitdata =
|
||||||
DRV_NAME ": Mellanox ConnectX core driver v"
|
DRV_NAME ": Mellanox ConnectX core driver v"
|
||||||
@ -386,6 +392,21 @@ static int mlx4_dev_cap(struct mlx4_dev *dev, struct mlx4_dev_cap *dev_cap)
|
|||||||
dev->caps.reserved_qps_cnt[MLX4_QP_REGION_FC_EXCH];
|
dev->caps.reserved_qps_cnt[MLX4_QP_REGION_FC_EXCH];
|
||||||
|
|
||||||
dev->caps.sqp_demux = (mlx4_is_master(dev)) ? MLX4_MAX_NUM_SLAVES : 0;
|
dev->caps.sqp_demux = (mlx4_is_master(dev)) ? MLX4_MAX_NUM_SLAVES : 0;
|
||||||
|
|
||||||
|
if (!enable_64b_cqe_eqe) {
|
||||||
|
if (dev_cap->flags &
|
||||||
|
(MLX4_DEV_CAP_FLAG_64B_CQE | MLX4_DEV_CAP_FLAG_64B_EQE)) {
|
||||||
|
mlx4_warn(dev, "64B EQEs/CQEs supported by the device but not enabled\n");
|
||||||
|
dev->caps.flags &= ~MLX4_DEV_CAP_FLAG_64B_CQE;
|
||||||
|
dev->caps.flags &= ~MLX4_DEV_CAP_FLAG_64B_EQE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((dev_cap->flags &
|
||||||
|
(MLX4_DEV_CAP_FLAG_64B_CQE | MLX4_DEV_CAP_FLAG_64B_EQE)) &&
|
||||||
|
mlx4_is_master(dev))
|
||||||
|
dev->caps.function_caps |= MLX4_FUNC_CAP_64B_EQE_CQE;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
/*The function checks if there are live vf, return the num of them*/
|
/*The function checks if there are live vf, return the num of them*/
|
||||||
@ -599,6 +620,21 @@ static int mlx4_slave_cap(struct mlx4_dev *dev)
|
|||||||
goto err_mem;
|
goto err_mem;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (hca_param.dev_cap_enabled & MLX4_DEV_CAP_64B_EQE_ENABLED) {
|
||||||
|
dev->caps.eqe_size = 64;
|
||||||
|
dev->caps.eqe_factor = 1;
|
||||||
|
} else {
|
||||||
|
dev->caps.eqe_size = 32;
|
||||||
|
dev->caps.eqe_factor = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (hca_param.dev_cap_enabled & MLX4_DEV_CAP_64B_CQE_ENABLED) {
|
||||||
|
dev->caps.cqe_size = 64;
|
||||||
|
dev->caps.userspace_caps |= MLX4_USER_DEV_CAP_64B_CQE;
|
||||||
|
} else {
|
||||||
|
dev->caps.cqe_size = 32;
|
||||||
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
err_mem:
|
err_mem:
|
||||||
|
@ -487,6 +487,7 @@ struct mlx4_en_priv {
|
|||||||
int mac_index;
|
int mac_index;
|
||||||
unsigned max_mtu;
|
unsigned max_mtu;
|
||||||
int base_qpn;
|
int base_qpn;
|
||||||
|
int cqe_factor;
|
||||||
|
|
||||||
struct mlx4_en_rss_map rss_map;
|
struct mlx4_en_rss_map rss_map;
|
||||||
__be32 ctrl_flags;
|
__be32 ctrl_flags;
|
||||||
|
@ -142,6 +142,8 @@ enum {
|
|||||||
MLX4_DEV_CAP_FLAG_COUNTERS = 1LL << 48,
|
MLX4_DEV_CAP_FLAG_COUNTERS = 1LL << 48,
|
||||||
MLX4_DEV_CAP_FLAG_SENSE_SUPPORT = 1LL << 55,
|
MLX4_DEV_CAP_FLAG_SENSE_SUPPORT = 1LL << 55,
|
||||||
MLX4_DEV_CAP_FLAG_PORT_MNG_CHG_EV = 1LL << 59,
|
MLX4_DEV_CAP_FLAG_PORT_MNG_CHG_EV = 1LL << 59,
|
||||||
|
MLX4_DEV_CAP_FLAG_64B_EQE = 1LL << 61,
|
||||||
|
MLX4_DEV_CAP_FLAG_64B_CQE = 1LL << 62
|
||||||
};
|
};
|
||||||
|
|
||||||
enum {
|
enum {
|
||||||
@ -151,6 +153,20 @@ enum {
|
|||||||
MLX4_DEV_CAP_FLAG2_FS_EN = 1LL << 3
|
MLX4_DEV_CAP_FLAG2_FS_EN = 1LL << 3
|
||||||
};
|
};
|
||||||
|
|
||||||
|
enum {
|
||||||
|
MLX4_DEV_CAP_64B_EQE_ENABLED = 1LL << 0,
|
||||||
|
MLX4_DEV_CAP_64B_CQE_ENABLED = 1LL << 1
|
||||||
|
};
|
||||||
|
|
||||||
|
enum {
|
||||||
|
MLX4_USER_DEV_CAP_64B_CQE = 1L << 0
|
||||||
|
};
|
||||||
|
|
||||||
|
enum {
|
||||||
|
MLX4_FUNC_CAP_64B_EQE_CQE = 1L << 0
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
#define MLX4_ATTR_EXTENDED_PORT_INFO cpu_to_be16(0xff90)
|
#define MLX4_ATTR_EXTENDED_PORT_INFO cpu_to_be16(0xff90)
|
||||||
|
|
||||||
enum {
|
enum {
|
||||||
@ -419,6 +435,11 @@ struct mlx4_caps {
|
|||||||
u32 max_counters;
|
u32 max_counters;
|
||||||
u8 port_ib_mtu[MLX4_MAX_PORTS + 1];
|
u8 port_ib_mtu[MLX4_MAX_PORTS + 1];
|
||||||
u16 sqp_demux;
|
u16 sqp_demux;
|
||||||
|
u32 eqe_size;
|
||||||
|
u32 cqe_size;
|
||||||
|
u8 eqe_factor;
|
||||||
|
u32 userspace_caps; /* userspace must be aware of these */
|
||||||
|
u32 function_caps; /* VFs must be aware of these */
|
||||||
};
|
};
|
||||||
|
|
||||||
struct mlx4_buf_list {
|
struct mlx4_buf_list {
|
||||||
|
@ -1,6 +0,0 @@
|
|||||||
header-y += ib_user_cm.h
|
|
||||||
header-y += ib_user_mad.h
|
|
||||||
header-y += ib_user_sa.h
|
|
||||||
header-y += ib_user_verbs.h
|
|
||||||
header-y += rdma_netlink.h
|
|
||||||
header-y += rdma_user_cm.h
|
|
@ -1,41 +1,9 @@
|
|||||||
#ifndef _RDMA_NETLINK_H
|
#ifndef _RDMA_NETLINK_H
|
||||||
#define _RDMA_NETLINK_H
|
#define _RDMA_NETLINK_H
|
||||||
|
|
||||||
#include <linux/types.h>
|
|
||||||
|
|
||||||
enum {
|
|
||||||
RDMA_NL_RDMA_CM = 1
|
|
||||||
};
|
|
||||||
|
|
||||||
#define RDMA_NL_GET_CLIENT(type) ((type & (((1 << 6) - 1) << 10)) >> 10)
|
|
||||||
#define RDMA_NL_GET_OP(type) (type & ((1 << 10) - 1))
|
|
||||||
#define RDMA_NL_GET_TYPE(client, op) ((client << 10) + op)
|
|
||||||
|
|
||||||
enum {
|
|
||||||
RDMA_NL_RDMA_CM_ID_STATS = 0,
|
|
||||||
RDMA_NL_RDMA_CM_NUM_OPS
|
|
||||||
};
|
|
||||||
|
|
||||||
enum {
|
|
||||||
RDMA_NL_RDMA_CM_ATTR_SRC_ADDR = 1,
|
|
||||||
RDMA_NL_RDMA_CM_ATTR_DST_ADDR,
|
|
||||||
RDMA_NL_RDMA_CM_NUM_ATTR,
|
|
||||||
};
|
|
||||||
|
|
||||||
struct rdma_cm_id_stats {
|
|
||||||
__u32 qp_num;
|
|
||||||
__u32 bound_dev_if;
|
|
||||||
__u32 port_space;
|
|
||||||
__s32 pid;
|
|
||||||
__u8 cm_state;
|
|
||||||
__u8 node_type;
|
|
||||||
__u8 port_num;
|
|
||||||
__u8 qp_type;
|
|
||||||
};
|
|
||||||
|
|
||||||
#ifdef __KERNEL__
|
|
||||||
|
|
||||||
#include <linux/netlink.h>
|
#include <linux/netlink.h>
|
||||||
|
#include <uapi/rdma/rdma_netlink.h>
|
||||||
|
|
||||||
struct ibnl_client_cbs {
|
struct ibnl_client_cbs {
|
||||||
int (*dump)(struct sk_buff *skb, struct netlink_callback *nlcb);
|
int (*dump)(struct sk_buff *skb, struct netlink_callback *nlcb);
|
||||||
@ -88,6 +56,4 @@ void *ibnl_put_msg(struct sk_buff *skb, struct nlmsghdr **nlh, int seq,
|
|||||||
int ibnl_put_attr(struct sk_buff *skb, struct nlmsghdr *nlh,
|
int ibnl_put_attr(struct sk_buff *skb, struct nlmsghdr *nlh,
|
||||||
int len, void *data, int type);
|
int len, void *data, int type);
|
||||||
|
|
||||||
#endif /* __KERNEL__ */
|
|
||||||
|
|
||||||
#endif /* _RDMA_NETLINK_H */
|
#endif /* _RDMA_NETLINK_H */
|
||||||
|
@ -1 +1,7 @@
|
|||||||
# UAPI Header export list
|
# UAPI Header export list
|
||||||
|
header-y += ib_user_cm.h
|
||||||
|
header-y += ib_user_mad.h
|
||||||
|
header-y += ib_user_sa.h
|
||||||
|
header-y += ib_user_verbs.h
|
||||||
|
header-y += rdma_netlink.h
|
||||||
|
header-y += rdma_user_cm.h
|
||||||
|
37
include/uapi/rdma/rdma_netlink.h
Normal file
37
include/uapi/rdma/rdma_netlink.h
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
#ifndef _UAPI_RDMA_NETLINK_H
|
||||||
|
#define _UAPI_RDMA_NETLINK_H
|
||||||
|
|
||||||
|
#include <linux/types.h>
|
||||||
|
|
||||||
|
enum {
|
||||||
|
RDMA_NL_RDMA_CM = 1
|
||||||
|
};
|
||||||
|
|
||||||
|
#define RDMA_NL_GET_CLIENT(type) ((type & (((1 << 6) - 1) << 10)) >> 10)
|
||||||
|
#define RDMA_NL_GET_OP(type) (type & ((1 << 10) - 1))
|
||||||
|
#define RDMA_NL_GET_TYPE(client, op) ((client << 10) + op)
|
||||||
|
|
||||||
|
enum {
|
||||||
|
RDMA_NL_RDMA_CM_ID_STATS = 0,
|
||||||
|
RDMA_NL_RDMA_CM_NUM_OPS
|
||||||
|
};
|
||||||
|
|
||||||
|
enum {
|
||||||
|
RDMA_NL_RDMA_CM_ATTR_SRC_ADDR = 1,
|
||||||
|
RDMA_NL_RDMA_CM_ATTR_DST_ADDR,
|
||||||
|
RDMA_NL_RDMA_CM_NUM_ATTR,
|
||||||
|
};
|
||||||
|
|
||||||
|
struct rdma_cm_id_stats {
|
||||||
|
__u32 qp_num;
|
||||||
|
__u32 bound_dev_if;
|
||||||
|
__u32 port_space;
|
||||||
|
__s32 pid;
|
||||||
|
__u8 cm_state;
|
||||||
|
__u8 node_type;
|
||||||
|
__u8 port_num;
|
||||||
|
__u8 qp_type;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
#endif /* _UAPI_RDMA_NETLINK_H */
|
Loading…
x
Reference in New Issue
Block a user