2018-02-09 16:39:19 +03:00
|
|
|
/*
|
|
|
|
* QEMU paravirtual RDMA - QP implementation
|
|
|
|
*
|
|
|
|
* Copyright (C) 2018 Oracle
|
|
|
|
* Copyright (C) 2018 Red Hat Inc
|
|
|
|
*
|
|
|
|
* Authors:
|
|
|
|
* Yuval Shaia <yuval.shaia@oracle.com>
|
|
|
|
* Marcel Apfelbaum <marcel@redhat.com>
|
|
|
|
*
|
|
|
|
* This work is licensed under the terms of the GNU GPL, version 2 or later.
|
|
|
|
* See the COPYING file in the top-level directory.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2018-03-21 18:22:07 +03:00
|
|
|
#include "qemu/osdep.h"
|
2018-02-09 16:39:19 +03:00
|
|
|
|
|
|
|
#include "../rdma_utils.h"
|
|
|
|
#include "../rdma_rm.h"
|
|
|
|
#include "../rdma_backend.h"
|
|
|
|
|
2019-03-11 13:29:05 +03:00
|
|
|
#include "trace.h"
|
|
|
|
|
2018-02-09 16:39:19 +03:00
|
|
|
#include "pvrdma.h"
|
2018-03-21 18:22:07 +03:00
|
|
|
#include "standard-headers/rdma/vmw_pvrdma-abi.h"
|
2018-02-09 16:39:19 +03:00
|
|
|
#include "pvrdma_qp_ops.h"
|
|
|
|
|
|
|
|
typedef struct CompHandlerCtx {
|
|
|
|
PVRDMADev *dev;
|
|
|
|
uint32_t cq_handle;
|
|
|
|
struct pvrdma_cqe cqe;
|
|
|
|
} CompHandlerCtx;
|
|
|
|
|
|
|
|
/* Send Queue WQE */
|
|
|
|
typedef struct PvrdmaSqWqe {
|
|
|
|
struct pvrdma_sq_wqe_hdr hdr;
|
misc: Replace zero-length arrays with flexible array member (automatic)
Description copied from Linux kernel commit from Gustavo A. R. Silva
(see [3]):
--v-- description start --v--
The current codebase makes use of the zero-length array language
extension to the C90 standard, but the preferred mechanism to
declare variable-length types such as these ones is a flexible
array member [1], introduced in C99:
struct foo {
int stuff;
struct boo array[];
};
By making use of the mechanism above, we will get a compiler
warning in case the flexible array does not occur last in the
structure, which will help us prevent some kind of undefined
behavior bugs from being unadvertenly introduced [2] to the
Linux codebase from now on.
--^-- description end --^--
Do the similar housekeeping in the QEMU codebase (which uses
C99 since commit 7be41675f7cb).
All these instances of code were found with the help of the
following Coccinelle script:
@@
identifier s, m, a;
type t, T;
@@
struct s {
...
t m;
- T a[0];
+ T a[];
};
@@
identifier s, m, a;
type t, T;
@@
struct s {
...
t m;
- T a[0];
+ T a[];
} QEMU_PACKED;
[1] https://gcc.gnu.org/onlinedocs/gcc/Zero-Length.html
[2] https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git/commit/?id=76497732932f
[3] https://git.kernel.org/pub/scm/linux/kernel/git/gustavoars/linux.git/commit/?id=17642a2fbd2c1
Inspired-by: Gustavo A. R. Silva <gustavo@embeddedor.com>
Reviewed-by: David Hildenbrand <david@redhat.com>
Signed-off-by: Philippe Mathieu-Daudé <philmd@redhat.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2020-03-04 18:38:15 +03:00
|
|
|
struct pvrdma_sge sge[];
|
2018-02-09 16:39:19 +03:00
|
|
|
} PvrdmaSqWqe;
|
|
|
|
|
|
|
|
/* Recv Queue WQE */
|
|
|
|
typedef struct PvrdmaRqWqe {
|
|
|
|
struct pvrdma_rq_wqe_hdr hdr;
|
misc: Replace zero-length arrays with flexible array member (automatic)
Description copied from Linux kernel commit from Gustavo A. R. Silva
(see [3]):
--v-- description start --v--
The current codebase makes use of the zero-length array language
extension to the C90 standard, but the preferred mechanism to
declare variable-length types such as these ones is a flexible
array member [1], introduced in C99:
struct foo {
int stuff;
struct boo array[];
};
By making use of the mechanism above, we will get a compiler
warning in case the flexible array does not occur last in the
structure, which will help us prevent some kind of undefined
behavior bugs from being unadvertenly introduced [2] to the
Linux codebase from now on.
--^-- description end --^--
Do the similar housekeeping in the QEMU codebase (which uses
C99 since commit 7be41675f7cb).
All these instances of code were found with the help of the
following Coccinelle script:
@@
identifier s, m, a;
type t, T;
@@
struct s {
...
t m;
- T a[0];
+ T a[];
};
@@
identifier s, m, a;
type t, T;
@@
struct s {
...
t m;
- T a[0];
+ T a[];
} QEMU_PACKED;
[1] https://gcc.gnu.org/onlinedocs/gcc/Zero-Length.html
[2] https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git/commit/?id=76497732932f
[3] https://git.kernel.org/pub/scm/linux/kernel/git/gustavoars/linux.git/commit/?id=17642a2fbd2c1
Inspired-by: Gustavo A. R. Silva <gustavo@embeddedor.com>
Reviewed-by: David Hildenbrand <david@redhat.com>
Signed-off-by: Philippe Mathieu-Daudé <philmd@redhat.com>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2020-03-04 18:38:15 +03:00
|
|
|
struct pvrdma_sge sge[];
|
2018-02-09 16:39:19 +03:00
|
|
|
} PvrdmaRqWqe;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* 1. Put CQE on send CQ ring
|
|
|
|
* 2. Put CQ number on dsr completion ring
|
|
|
|
* 3. Interrupt host
|
|
|
|
*/
|
|
|
|
static int pvrdma_post_cqe(PVRDMADev *dev, uint32_t cq_handle,
|
2018-12-21 17:40:30 +03:00
|
|
|
struct pvrdma_cqe *cqe, struct ibv_wc *wc)
|
2018-02-09 16:39:19 +03:00
|
|
|
{
|
|
|
|
struct pvrdma_cqe *cqe1;
|
|
|
|
struct pvrdma_cqne *cqne;
|
|
|
|
PvrdmaRing *ring;
|
|
|
|
RdmaRmCQ *cq = rdma_rm_get_cq(&dev->rdma_dev_res, cq_handle);
|
|
|
|
|
|
|
|
if (unlikely(!cq)) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
ring = (PvrdmaRing *)cq->opaque;
|
|
|
|
|
|
|
|
/* Step #1: Put CQE on CQ ring */
|
|
|
|
cqe1 = pvrdma_ring_next_elem_write(ring);
|
|
|
|
if (unlikely(!cqe1)) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2018-08-05 18:35:09 +03:00
|
|
|
memset(cqe1, 0, sizeof(*cqe1));
|
2018-02-09 16:39:19 +03:00
|
|
|
cqe1->wr_id = cqe->wr_id;
|
2019-04-03 14:33:43 +03:00
|
|
|
cqe1->qp = cqe->qp ? cqe->qp : wc->qp_num;
|
2018-02-09 16:39:19 +03:00
|
|
|
cqe1->opcode = cqe->opcode;
|
2018-12-21 17:40:30 +03:00
|
|
|
cqe1->status = wc->status;
|
|
|
|
cqe1->byte_len = wc->byte_len;
|
|
|
|
cqe1->src_qp = wc->src_qp;
|
|
|
|
cqe1->wc_flags = wc->wc_flags;
|
|
|
|
cqe1->vendor_err = wc->vendor_err;
|
|
|
|
|
2019-03-11 13:29:05 +03:00
|
|
|
trace_pvrdma_post_cqe(cq_handle, cq->notify, cqe1->wr_id, cqe1->qp,
|
|
|
|
cqe1->opcode, cqe1->status, cqe1->byte_len,
|
|
|
|
cqe1->src_qp, cqe1->wc_flags, cqe1->vendor_err);
|
2018-02-09 16:39:19 +03:00
|
|
|
|
|
|
|
pvrdma_ring_write_inc(ring);
|
|
|
|
|
|
|
|
/* Step #2: Put CQ number on dsr completion ring */
|
|
|
|
cqne = pvrdma_ring_next_elem_write(&dev->dsr_info.cq);
|
|
|
|
if (unlikely(!cqne)) {
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
cqne->info = cq_handle;
|
|
|
|
pvrdma_ring_write_inc(&dev->dsr_info.cq);
|
|
|
|
|
2018-12-21 17:40:16 +03:00
|
|
|
if (cq->notify != CNT_CLEAR) {
|
|
|
|
if (cq->notify == CNT_ARM) {
|
|
|
|
cq->notify = CNT_CLEAR;
|
|
|
|
}
|
2018-02-09 16:39:19 +03:00
|
|
|
post_interrupt(dev, INTR_VEC_CMD_COMPLETION_Q);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-12-21 17:40:30 +03:00
|
|
|
static void pvrdma_qp_ops_comp_handler(void *ctx, struct ibv_wc *wc)
|
2018-02-09 16:39:19 +03:00
|
|
|
{
|
|
|
|
CompHandlerCtx *comp_ctx = (CompHandlerCtx *)ctx;
|
|
|
|
|
2018-12-21 17:40:30 +03:00
|
|
|
pvrdma_post_cqe(comp_ctx->dev, comp_ctx->cq_handle, &comp_ctx->cqe, wc);
|
|
|
|
|
2018-02-09 16:39:19 +03:00
|
|
|
g_free(ctx);
|
|
|
|
}
|
|
|
|
|
2019-01-09 22:41:23 +03:00
|
|
|
static void complete_with_error(uint32_t vendor_err, void *ctx)
|
|
|
|
{
|
2019-03-14 18:30:30 +03:00
|
|
|
struct ibv_wc wc = {};
|
2019-01-09 22:41:23 +03:00
|
|
|
|
|
|
|
wc.status = IBV_WC_GENERAL_ERR;
|
|
|
|
wc.vendor_err = vendor_err;
|
|
|
|
|
|
|
|
pvrdma_qp_ops_comp_handler(ctx, &wc);
|
|
|
|
}
|
|
|
|
|
2018-02-09 16:39:19 +03:00
|
|
|
void pvrdma_qp_ops_fini(void)
|
|
|
|
{
|
|
|
|
rdma_backend_unregister_comp_handler();
|
|
|
|
}
|
|
|
|
|
|
|
|
int pvrdma_qp_ops_init(void)
|
|
|
|
{
|
|
|
|
rdma_backend_register_comp_handler(pvrdma_qp_ops_comp_handler);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-01-09 23:21:40 +03:00
|
|
|
void pvrdma_qp_send(PVRDMADev *dev, uint32_t qp_handle)
|
2018-02-09 16:39:19 +03:00
|
|
|
{
|
|
|
|
RdmaRmQP *qp;
|
|
|
|
PvrdmaSqWqe *wqe;
|
|
|
|
PvrdmaRing *ring;
|
2018-12-21 17:40:25 +03:00
|
|
|
int sgid_idx;
|
|
|
|
union ibv_gid *sgid;
|
2018-02-09 16:39:19 +03:00
|
|
|
|
|
|
|
qp = rdma_rm_get_qp(&dev->rdma_dev_res, qp_handle);
|
|
|
|
if (unlikely(!qp)) {
|
2019-01-09 23:21:40 +03:00
|
|
|
return;
|
2018-02-09 16:39:19 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
ring = (PvrdmaRing *)qp->opaque;
|
|
|
|
|
|
|
|
wqe = (struct PvrdmaSqWqe *)pvrdma_ring_next_elem_read(ring);
|
|
|
|
while (wqe) {
|
|
|
|
CompHandlerCtx *comp_ctx;
|
|
|
|
|
|
|
|
/* Prepare CQE */
|
|
|
|
comp_ctx = g_malloc(sizeof(CompHandlerCtx));
|
|
|
|
comp_ctx->dev = dev;
|
|
|
|
comp_ctx->cq_handle = qp->send_cq_handle;
|
|
|
|
comp_ctx->cqe.wr_id = wqe->hdr.wr_id;
|
|
|
|
comp_ctx->cqe.qp = qp_handle;
|
2018-12-21 17:40:23 +03:00
|
|
|
comp_ctx->cqe.opcode = IBV_WC_SEND;
|
2018-02-09 16:39:19 +03:00
|
|
|
|
2018-12-21 17:40:25 +03:00
|
|
|
sgid = rdma_rm_get_gid(&dev->rdma_dev_res, wqe->hdr.wr.ud.av.gid_index);
|
|
|
|
if (!sgid) {
|
2019-03-11 13:29:05 +03:00
|
|
|
rdma_error_report("Failed to get gid for idx %d",
|
|
|
|
wqe->hdr.wr.ud.av.gid_index);
|
2019-01-09 23:15:59 +03:00
|
|
|
complete_with_error(VENDOR_ERR_INV_GID_IDX, comp_ctx);
|
|
|
|
continue;
|
2018-12-21 17:40:25 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
sgid_idx = rdma_rm_get_backend_gid_index(&dev->rdma_dev_res,
|
|
|
|
&dev->backend_dev,
|
|
|
|
wqe->hdr.wr.ud.av.gid_index);
|
|
|
|
if (sgid_idx <= 0) {
|
2019-03-11 13:29:05 +03:00
|
|
|
rdma_error_report("Failed to get bk sgid_idx for sgid_idx %d",
|
|
|
|
wqe->hdr.wr.ud.av.gid_index);
|
2019-01-09 23:15:59 +03:00
|
|
|
complete_with_error(VENDOR_ERR_INV_GID_IDX, comp_ctx);
|
|
|
|
continue;
|
2018-12-21 17:40:25 +03:00
|
|
|
}
|
|
|
|
|
2019-01-09 22:41:23 +03:00
|
|
|
if (wqe->hdr.num_sge > dev->dev_attr.max_sge) {
|
2019-03-11 13:29:05 +03:00
|
|
|
rdma_error_report("Invalid num_sge=%d (max %d)", wqe->hdr.num_sge,
|
|
|
|
dev->dev_attr.max_sge);
|
2019-01-09 22:41:23 +03:00
|
|
|
complete_with_error(VENDOR_ERR_INV_NUM_SGE, comp_ctx);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2018-02-09 16:39:19 +03:00
|
|
|
rdma_backend_post_send(&dev->backend_dev, &qp->backend_qp, qp->qp_type,
|
|
|
|
(struct ibv_sge *)&wqe->sge[0], wqe->hdr.num_sge,
|
2018-12-21 17:40:25 +03:00
|
|
|
sgid_idx, sgid,
|
2018-02-09 16:39:19 +03:00
|
|
|
(union ibv_gid *)wqe->hdr.wr.ud.av.dgid,
|
|
|
|
wqe->hdr.wr.ud.remote_qpn,
|
|
|
|
wqe->hdr.wr.ud.remote_qkey, comp_ctx);
|
|
|
|
|
|
|
|
pvrdma_ring_read_inc(ring);
|
|
|
|
|
|
|
|
wqe = pvrdma_ring_next_elem_read(ring);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-09 23:21:40 +03:00
|
|
|
void pvrdma_qp_recv(PVRDMADev *dev, uint32_t qp_handle)
|
2018-02-09 16:39:19 +03:00
|
|
|
{
|
|
|
|
RdmaRmQP *qp;
|
|
|
|
PvrdmaRqWqe *wqe;
|
|
|
|
PvrdmaRing *ring;
|
|
|
|
|
|
|
|
qp = rdma_rm_get_qp(&dev->rdma_dev_res, qp_handle);
|
|
|
|
if (unlikely(!qp)) {
|
2019-01-09 23:21:40 +03:00
|
|
|
return;
|
2018-02-09 16:39:19 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
ring = &((PvrdmaRing *)qp->opaque)[1];
|
|
|
|
|
|
|
|
wqe = (struct PvrdmaRqWqe *)pvrdma_ring_next_elem_read(ring);
|
|
|
|
while (wqe) {
|
|
|
|
CompHandlerCtx *comp_ctx;
|
|
|
|
|
|
|
|
/* Prepare CQE */
|
|
|
|
comp_ctx = g_malloc(sizeof(CompHandlerCtx));
|
|
|
|
comp_ctx->dev = dev;
|
|
|
|
comp_ctx->cq_handle = qp->recv_cq_handle;
|
|
|
|
comp_ctx->cqe.wr_id = wqe->hdr.wr_id;
|
2018-12-21 17:40:22 +03:00
|
|
|
comp_ctx->cqe.qp = qp_handle;
|
|
|
|
comp_ctx->cqe.opcode = IBV_WC_RECV;
|
2018-02-09 16:39:19 +03:00
|
|
|
|
2019-01-09 22:41:23 +03:00
|
|
|
if (wqe->hdr.num_sge > dev->dev_attr.max_sge) {
|
2019-03-11 13:29:05 +03:00
|
|
|
rdma_error_report("Invalid num_sge=%d (max %d)", wqe->hdr.num_sge,
|
|
|
|
dev->dev_attr.max_sge);
|
2019-01-09 22:41:23 +03:00
|
|
|
complete_with_error(VENDOR_ERR_INV_NUM_SGE, comp_ctx);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2019-03-11 13:29:12 +03:00
|
|
|
rdma_backend_post_recv(&dev->backend_dev, &qp->backend_qp, qp->qp_type,
|
2018-02-09 16:39:19 +03:00
|
|
|
(struct ibv_sge *)&wqe->sge[0], wqe->hdr.num_sge,
|
|
|
|
comp_ctx);
|
|
|
|
|
|
|
|
pvrdma_ring_read_inc(ring);
|
|
|
|
|
|
|
|
wqe = pvrdma_ring_next_elem_read(ring);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-03 14:33:43 +03:00
|
|
|
void pvrdma_srq_recv(PVRDMADev *dev, uint32_t srq_handle)
|
|
|
|
{
|
|
|
|
RdmaRmSRQ *srq;
|
|
|
|
PvrdmaRqWqe *wqe;
|
|
|
|
PvrdmaRing *ring;
|
|
|
|
|
|
|
|
srq = rdma_rm_get_srq(&dev->rdma_dev_res, srq_handle);
|
|
|
|
if (unlikely(!srq)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
ring = (PvrdmaRing *)srq->opaque;
|
|
|
|
|
|
|
|
wqe = (struct PvrdmaRqWqe *)pvrdma_ring_next_elem_read(ring);
|
|
|
|
while (wqe) {
|
|
|
|
CompHandlerCtx *comp_ctx;
|
|
|
|
|
|
|
|
/* Prepare CQE */
|
|
|
|
comp_ctx = g_malloc(sizeof(CompHandlerCtx));
|
|
|
|
comp_ctx->dev = dev;
|
|
|
|
comp_ctx->cq_handle = srq->recv_cq_handle;
|
|
|
|
comp_ctx->cqe.wr_id = wqe->hdr.wr_id;
|
|
|
|
comp_ctx->cqe.qp = 0;
|
|
|
|
comp_ctx->cqe.opcode = IBV_WC_RECV;
|
|
|
|
|
|
|
|
if (wqe->hdr.num_sge > dev->dev_attr.max_sge) {
|
|
|
|
rdma_error_report("Invalid num_sge=%d (max %d)", wqe->hdr.num_sge,
|
|
|
|
dev->dev_attr.max_sge);
|
|
|
|
complete_with_error(VENDOR_ERR_INV_NUM_SGE, comp_ctx);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
rdma_backend_post_srq_recv(&dev->backend_dev, &srq->backend_srq,
|
|
|
|
(struct ibv_sge *)&wqe->sge[0],
|
|
|
|
wqe->hdr.num_sge,
|
|
|
|
comp_ctx);
|
|
|
|
|
|
|
|
pvrdma_ring_read_inc(ring);
|
|
|
|
|
|
|
|
wqe = pvrdma_ring_next_elem_read(ring);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2018-02-09 16:39:19 +03:00
|
|
|
void pvrdma_cq_poll(RdmaDeviceResources *dev_res, uint32_t cq_handle)
|
|
|
|
{
|
|
|
|
RdmaRmCQ *cq;
|
|
|
|
|
|
|
|
cq = rdma_rm_get_cq(dev_res, cq_handle);
|
|
|
|
if (!cq) {
|
2018-04-30 23:02:18 +03:00
|
|
|
return;
|
2018-02-09 16:39:19 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
rdma_backend_poll_cq(dev_res, &cq->backend_cq);
|
|
|
|
}
|