linux/drivers/vdpa/mlx5/core/resources.c
<<
>>
Prefs
   1// SPDX-License-Identifier: GPL-2.0 OR Linux-OpenIB
   2/* Copyright (c) 2020 Mellanox Technologies Ltd. */
   3
   4#include <linux/mlx5/driver.h>
   5#include "mlx5_vdpa.h"
   6
   7static int alloc_pd(struct mlx5_vdpa_dev *dev, u32 *pdn, u16 uid)
   8{
   9        struct mlx5_core_dev *mdev = dev->mdev;
  10
  11        u32 out[MLX5_ST_SZ_DW(alloc_pd_out)] = {};
  12        u32 in[MLX5_ST_SZ_DW(alloc_pd_in)] = {};
  13        int err;
  14
  15        MLX5_SET(alloc_pd_in, in, opcode, MLX5_CMD_OP_ALLOC_PD);
  16        MLX5_SET(alloc_pd_in, in, uid, uid);
  17
  18        err = mlx5_cmd_exec_inout(mdev, alloc_pd, in, out);
  19        if (!err)
  20                *pdn = MLX5_GET(alloc_pd_out, out, pd);
  21
  22        return err;
  23}
  24
  25static int dealloc_pd(struct mlx5_vdpa_dev *dev, u32 pdn, u16 uid)
  26{
  27        u32 in[MLX5_ST_SZ_DW(dealloc_pd_in)] = {};
  28        struct mlx5_core_dev *mdev = dev->mdev;
  29
  30        MLX5_SET(dealloc_pd_in, in, opcode, MLX5_CMD_OP_DEALLOC_PD);
  31        MLX5_SET(dealloc_pd_in, in, pd, pdn);
  32        MLX5_SET(dealloc_pd_in, in, uid, uid);
  33        return mlx5_cmd_exec_in(mdev, dealloc_pd, in);
  34}
  35
  36static int get_null_mkey(struct mlx5_vdpa_dev *dev, u32 *null_mkey)
  37{
  38        u32 out[MLX5_ST_SZ_DW(query_special_contexts_out)] = {};
  39        u32 in[MLX5_ST_SZ_DW(query_special_contexts_in)] = {};
  40        struct mlx5_core_dev *mdev = dev->mdev;
  41        int err;
  42
  43        MLX5_SET(query_special_contexts_in, in, opcode, MLX5_CMD_OP_QUERY_SPECIAL_CONTEXTS);
  44        err = mlx5_cmd_exec_inout(mdev, query_special_contexts, in, out);
  45        if (!err)
  46                *null_mkey = MLX5_GET(query_special_contexts_out, out, null_mkey);
  47        return err;
  48}
  49
  50static int create_uctx(struct mlx5_vdpa_dev *mvdev, u16 *uid)
  51{
  52        u32 out[MLX5_ST_SZ_DW(create_uctx_out)] = {};
  53        int inlen;
  54        void *in;
  55        int err;
  56
  57        if (MLX5_CAP_GEN(mvdev->mdev, umem_uid_0))
  58                return 0;
  59
  60        /* 0 means not supported */
  61        if (!MLX5_CAP_GEN(mvdev->mdev, log_max_uctx))
  62                return -EOPNOTSUPP;
  63
  64        inlen = MLX5_ST_SZ_BYTES(create_uctx_in);
  65        in = kzalloc(inlen, GFP_KERNEL);
  66        if (!in)
  67                return -ENOMEM;
  68
  69        MLX5_SET(create_uctx_in, in, opcode, MLX5_CMD_OP_CREATE_UCTX);
  70        MLX5_SET(create_uctx_in, in, uctx.cap, MLX5_UCTX_CAP_RAW_TX);
  71
  72        err = mlx5_cmd_exec(mvdev->mdev, in, inlen, out, sizeof(out));
  73        kfree(in);
  74        if (!err)
  75                *uid = MLX5_GET(create_uctx_out, out, uid);
  76
  77        return err;
  78}
  79
  80static void destroy_uctx(struct mlx5_vdpa_dev *mvdev, u32 uid)
  81{
  82        u32 out[MLX5_ST_SZ_DW(destroy_uctx_out)] = {};
  83        u32 in[MLX5_ST_SZ_DW(destroy_uctx_in)] = {};
  84
  85        if (!uid)
  86                return;
  87
  88        MLX5_SET(destroy_uctx_in, in, opcode, MLX5_CMD_OP_DESTROY_UCTX);
  89        MLX5_SET(destroy_uctx_in, in, uid, uid);
  90
  91        mlx5_cmd_exec(mvdev->mdev, in, sizeof(in), out, sizeof(out));
  92}
  93
  94int mlx5_vdpa_create_tis(struct mlx5_vdpa_dev *mvdev, void *in, u32 *tisn)
  95{
  96        u32 out[MLX5_ST_SZ_DW(create_tis_out)] = {};
  97        int err;
  98
  99        MLX5_SET(create_tis_in, in, opcode, MLX5_CMD_OP_CREATE_TIS);
 100        MLX5_SET(create_tis_in, in, uid, mvdev->res.uid);
 101        err = mlx5_cmd_exec_inout(mvdev->mdev, create_tis, in, out);
 102        if (!err)
 103                *tisn = MLX5_GET(create_tis_out, out, tisn);
 104
 105        return err;
 106}
 107
 108void mlx5_vdpa_destroy_tis(struct mlx5_vdpa_dev *mvdev, u32 tisn)
 109{
 110        u32 in[MLX5_ST_SZ_DW(destroy_tis_in)] = {};
 111
 112        MLX5_SET(destroy_tis_in, in, opcode, MLX5_CMD_OP_DESTROY_TIS);
 113        MLX5_SET(destroy_tis_in, in, uid, mvdev->res.uid);
 114        MLX5_SET(destroy_tis_in, in, tisn, tisn);
 115        mlx5_cmd_exec_in(mvdev->mdev, destroy_tis, in);
 116}
 117
 118int mlx5_vdpa_create_rqt(struct mlx5_vdpa_dev *mvdev, void *in, int inlen, u32 *rqtn)
 119{
 120        u32 out[MLX5_ST_SZ_DW(create_rqt_out)] = {};
 121        int err;
 122
 123        MLX5_SET(create_rqt_in, in, opcode, MLX5_CMD_OP_CREATE_RQT);
 124        err = mlx5_cmd_exec(mvdev->mdev, in, inlen, out, sizeof(out));
 125        if (!err)
 126                *rqtn = MLX5_GET(create_rqt_out, out, rqtn);
 127
 128        return err;
 129}
 130
 131void mlx5_vdpa_destroy_rqt(struct mlx5_vdpa_dev *mvdev, u32 rqtn)
 132{
 133        u32 in[MLX5_ST_SZ_DW(destroy_rqt_in)] = {};
 134
 135        MLX5_SET(destroy_rqt_in, in, opcode, MLX5_CMD_OP_DESTROY_RQT);
 136        MLX5_SET(destroy_rqt_in, in, uid, mvdev->res.uid);
 137        MLX5_SET(destroy_rqt_in, in, rqtn, rqtn);
 138        mlx5_cmd_exec_in(mvdev->mdev, destroy_rqt, in);
 139}
 140
 141int mlx5_vdpa_create_tir(struct mlx5_vdpa_dev *mvdev, void *in, u32 *tirn)
 142{
 143        u32 out[MLX5_ST_SZ_DW(create_tir_out)] = {};
 144        int err;
 145
 146        MLX5_SET(create_tir_in, in, opcode, MLX5_CMD_OP_CREATE_TIR);
 147        err = mlx5_cmd_exec_inout(mvdev->mdev, create_tir, in, out);
 148        if (!err)
 149                *tirn = MLX5_GET(create_tir_out, out, tirn);
 150
 151        return err;
 152}
 153
 154void mlx5_vdpa_destroy_tir(struct mlx5_vdpa_dev *mvdev, u32 tirn)
 155{
 156        u32 in[MLX5_ST_SZ_DW(destroy_tir_in)] = {};
 157
 158        MLX5_SET(destroy_tir_in, in, opcode, MLX5_CMD_OP_DESTROY_TIR);
 159        MLX5_SET(destroy_tir_in, in, uid, mvdev->res.uid);
 160        MLX5_SET(destroy_tir_in, in, tirn, tirn);
 161        mlx5_cmd_exec_in(mvdev->mdev, destroy_tir, in);
 162}
 163
 164int mlx5_vdpa_alloc_transport_domain(struct mlx5_vdpa_dev *mvdev, u32 *tdn)
 165{
 166        u32 out[MLX5_ST_SZ_DW(alloc_transport_domain_out)] = {};
 167        u32 in[MLX5_ST_SZ_DW(alloc_transport_domain_in)] = {};
 168        int err;
 169
 170        MLX5_SET(alloc_transport_domain_in, in, opcode, MLX5_CMD_OP_ALLOC_TRANSPORT_DOMAIN);
 171        MLX5_SET(alloc_transport_domain_in, in, uid, mvdev->res.uid);
 172
 173        err = mlx5_cmd_exec_inout(mvdev->mdev, alloc_transport_domain, in, out);
 174        if (!err)
 175                *tdn = MLX5_GET(alloc_transport_domain_out, out, transport_domain);
 176
 177        return err;
 178}
 179
 180void mlx5_vdpa_dealloc_transport_domain(struct mlx5_vdpa_dev *mvdev, u32 tdn)
 181{
 182        u32 in[MLX5_ST_SZ_DW(dealloc_transport_domain_in)] = {};
 183
 184        MLX5_SET(dealloc_transport_domain_in, in, opcode, MLX5_CMD_OP_DEALLOC_TRANSPORT_DOMAIN);
 185        MLX5_SET(dealloc_transport_domain_in, in, uid, mvdev->res.uid);
 186        MLX5_SET(dealloc_transport_domain_in, in, transport_domain, tdn);
 187        mlx5_cmd_exec_in(mvdev->mdev, dealloc_transport_domain, in);
 188}
 189
 190int mlx5_vdpa_create_mkey(struct mlx5_vdpa_dev *mvdev, struct mlx5_core_mkey *mkey, u32 *in,
 191                          int inlen)
 192{
 193        u32 lout[MLX5_ST_SZ_DW(create_mkey_out)] = {};
 194        u32 mkey_index;
 195        void *mkc;
 196        int err;
 197
 198        MLX5_SET(create_mkey_in, in, opcode, MLX5_CMD_OP_CREATE_MKEY);
 199        MLX5_SET(create_mkey_in, in, uid, mvdev->res.uid);
 200
 201        err = mlx5_cmd_exec(mvdev->mdev, in, inlen, lout, sizeof(lout));
 202        if (err)
 203                return err;
 204
 205        mkc = MLX5_ADDR_OF(create_mkey_in, in, memory_key_mkey_entry);
 206        mkey_index = MLX5_GET(create_mkey_out, lout, mkey_index);
 207        mkey->iova = MLX5_GET64(mkc, mkc, start_addr);
 208        mkey->size = MLX5_GET64(mkc, mkc, len);
 209        mkey->key |= mlx5_idx_to_mkey(mkey_index);
 210        mkey->pd = MLX5_GET(mkc, mkc, pd);
 211        return 0;
 212}
 213
 214int mlx5_vdpa_destroy_mkey(struct mlx5_vdpa_dev *mvdev, struct mlx5_core_mkey *mkey)
 215{
 216        u32 in[MLX5_ST_SZ_DW(destroy_mkey_in)] = {};
 217
 218        MLX5_SET(destroy_mkey_in, in, uid, mvdev->res.uid);
 219        MLX5_SET(destroy_mkey_in, in, opcode, MLX5_CMD_OP_DESTROY_MKEY);
 220        MLX5_SET(destroy_mkey_in, in, mkey_index, mlx5_mkey_to_idx(mkey->key));
 221        return mlx5_cmd_exec_in(mvdev->mdev, destroy_mkey, in);
 222}
 223
 224int mlx5_vdpa_alloc_resources(struct mlx5_vdpa_dev *mvdev)
 225{
 226        u64 offset = MLX5_CAP64_DEV_VDPA_EMULATION(mvdev->mdev, doorbell_bar_offset);
 227        struct mlx5_vdpa_resources *res = &mvdev->res;
 228        struct mlx5_core_dev *mdev = mvdev->mdev;
 229        u64 kick_addr;
 230        int err;
 231
 232        if (res->valid) {
 233                mlx5_vdpa_warn(mvdev, "resources already allocated\n");
 234                return -EINVAL;
 235        }
 236        mutex_init(&mvdev->mr.mkey_mtx);
 237        res->uar = mlx5_get_uars_page(mdev);
 238        if (IS_ERR(res->uar)) {
 239                err = PTR_ERR(res->uar);
 240                goto err_uars;
 241        }
 242
 243        err = create_uctx(mvdev, &res->uid);
 244        if (err)
 245                goto err_uctx;
 246
 247        err = alloc_pd(mvdev, &res->pdn, res->uid);
 248        if (err)
 249                goto err_pd;
 250
 251        err = get_null_mkey(mvdev, &res->null_mkey);
 252        if (err)
 253                goto err_key;
 254
 255        kick_addr = mdev->bar_addr + offset;
 256        res->phys_kick_addr = kick_addr;
 257
 258        res->kick_addr = ioremap(kick_addr, PAGE_SIZE);
 259        if (!res->kick_addr) {
 260                err = -ENOMEM;
 261                goto err_key;
 262        }
 263        res->valid = true;
 264
 265        return 0;
 266
 267err_key:
 268        dealloc_pd(mvdev, res->pdn, res->uid);
 269err_pd:
 270        destroy_uctx(mvdev, res->uid);
 271err_uctx:
 272        mlx5_put_uars_page(mdev, res->uar);
 273err_uars:
 274        mutex_destroy(&mvdev->mr.mkey_mtx);
 275        return err;
 276}
 277
 278void mlx5_vdpa_free_resources(struct mlx5_vdpa_dev *mvdev)
 279{
 280        struct mlx5_vdpa_resources *res = &mvdev->res;
 281
 282        if (!res->valid)
 283                return;
 284
 285        iounmap(res->kick_addr);
 286        res->kick_addr = NULL;
 287        dealloc_pd(mvdev, res->pdn, res->uid);
 288        destroy_uctx(mvdev, res->uid);
 289        mlx5_put_uars_page(mvdev->mdev, res->uar);
 290        mutex_destroy(&mvdev->mr.mkey_mtx);
 291        res->valid = false;
 292}
 293