@@ -38,7 +38,8 @@ enum {
IBV_QP_CREATE_SCATTER_FCS |
IBV_QP_CREATE_CVLAN_STRIPPING |
IBV_QP_CREATE_SOURCE_QPN |
- IBV_QP_CREATE_PCI_WRITE_END_PADDING
+ IBV_QP_CREATE_PCI_WRITE_END_PADDING |
+ IBV_QP_CREATE_DYNAMIC_CONTEXT_ATTACH
};
@@ -918,6 +918,7 @@ enum ibv_qp_create_flags {
IBV_QP_CREATE_CVLAN_STRIPPING = 1 << 9,
IBV_QP_CREATE_SOURCE_QPN = 1 << 10,
IBV_QP_CREATE_PCI_WRITE_END_PADDING = 1 << 11,
+ IBV_QP_CREATE_DYNAMIC_CONTEXT_ATTACH = 1 << 13,
};
enum ibv_qp_create_send_ops_flags {
@@ -92,6 +92,10 @@ static const struct verbs_context_ops hns_common_ops = {
static int init_dca_context(struct hns_roce_context *ctx, int page_size)
{
struct hns_roce_dca_ctx *dca_ctx = &ctx->dca_ctx;
+ int unit_size = 0;
+ long max_size = 0;
+ long min_size;
+ char *env;
int ret;
if (!(ctx->cap_flags & HNS_ROCE_CAP_FLAG_DCA_MODE))
@@ -102,8 +106,52 @@ static int init_dca_context(struct hns_roce_context *ctx, int page_size)
if (ret)
return ret;
- dca_ctx->unit_size = page_size * HNS_DCA_DEFAULT_UNIT_PAGES;
- dca_ctx->max_size = HNS_DCA_MAX_MEM_SIZE;
+ env = getenv("HNS_DCA_UNIT_SIZE");
+ if (env) {
+ unit_size = atoi(env);
+ /* Disable DCA only for this process */
+ if (unit_size == 0)
+ return 0;
+ }
+
+ if (unit_size < 1)
+ unit_size = page_size * HNS_DCA_DEFAULT_UNIT_PAGES;
+
+ unit_size = align(unit_size, page_size);
+
+ /*
+ * not set OR 0: Unlimited memory pool increase.
+ * others: Maximum memory pool size to be increased.
+ */
+ env = getenv("HNS_DCA_MAX_SIZE");
+ if (env)
+ max_size = atol(env);
+
+ if (max_size == 0)
+ max_size = HNS_DCA_MAX_MEM_SIZE;
+ else
+ max_size = DIV_ROUND_UP(max_size, unit_size);
+
+ /*
+ * not set: The memory pool cannot be reduced.
+ * others: The size of free memory in the pool cannot exceed this value.
+ * 0: Always reduce the free memory in the pool.
+ */
+ env = getenv("HNS_DCA_MIN_SIZE");
+ if (env) {
+ min_size = atol(env);
+ if (min_size > 0)
+ min_size = DIV_ROUND_UP(min_size, unit_size);
+ else
+ min_size = 0;
+ } else {
+ min_size = HNS_DCA_MAX_MEM_SIZE;
+ }
+
+ dca_ctx->unit_size = unit_size;
+ dca_ctx->max_size = max_size;
+ dca_ctx->min_size = min_size;
+
dca_ctx->mem_cnt = 0;
return 0;
@@ -542,7 +542,7 @@ int hns_roce_u_destroy_srq(struct ibv_srq *srq)
}
enum {
- CREATE_QP_SUP_CREATE_FLAGS = 0,
+ CREATE_QP_SUP_CREATE_FLAGS = IBV_QP_CREATE_DYNAMIC_CONTEXT_ATTACH,
};
enum {
@@ -688,10 +688,11 @@ static int calc_qp_buff_size(struct hns_roce_device *hr_dev,
return 0;
}
-static inline bool check_qp_support_dca(bool pool_en, enum ibv_qp_type qp_type)
+static inline bool check_qp_support_dca(bool pool_en, enum ibv_qp_type qp_type,
+ uint32_t create_flags)
{
if (pool_en && (qp_type == IBV_QPT_RC || qp_type == IBV_QPT_XRC_SEND))
- return true;
+ return !!(create_flags & IBV_QP_CREATE_DYNAMIC_CONTEXT_ATTACH);
return false;
}
@@ -730,7 +731,8 @@ static int qp_alloc_wqe(struct ibv_qp_init_attr_ex *attr,
goto err_alloc;
}
- if (check_qp_support_dca(ctx->dca_ctx.max_size != 0, attr->qp_type)) {
+ if (check_qp_support_dca(ctx->dca_ctx.max_size != 0,
+ attr->qp_type, attr->create_flags)) {
/* when DCA is enabled, use a buffer list to store page addr */
qp->buf.buf = NULL;
qp->page_list.max_cnt = hr_hw_page_count(qp->buf_size);
@@ -901,6 +903,14 @@ static int qp_exec_create_cmd(struct ibv_qp_init_attr_ex *attr,
struct hns_roce_create_qp_ex cmd_ex = {};
int ret;
+ /*
+ * When handling the command in kernel space, the user QP enable
+ * the DCA mode by checking whether the cmd.buf_addr is NULL but
+ * not the attr->create_flags has the DCA enable bit, so clear
+ * this bit before command is ready to run.
+ */
+ attr->create_flags &= ~IBV_QP_CREATE_DYNAMIC_CONTEXT_ATTACH;
+
cmd_ex.sdb_addr = (uintptr_t)qp->sdb;
cmd_ex.db_addr = (uintptr_t)qp->rdb;
cmd_ex.buf_addr = (uintptr_t)qp->buf.buf;