Commit f5f502a3 authored by Jakub Kicinski's avatar Jakub Kicinski

Merge tag 'mlx5-updates-2023-08-07' of git://git.kernel.org/pub/scm/linux/kernel/git/saeed/linux

Saeed Mahameed says:

====================
mlx5-updates-2023-08-07

1) Few cleanups

2) Dynamic completion EQs

The driver creates completion EQs for all vectors directly on driver
load, even if those EQs will not be utilized later on.

To allow more flexibility in managing completion EQs and to reduce the
memory overhead of driver load, this series will adjust completion EQs
creation to be dynamic. Meaning, completion EQs will be created only
when needed.

Patch #1 introduces a counter for tracking the current number of
completion EQs.
Patches #2-6 refactor the existing infrastructure of managing completion
EQs and completion IRQs to be compatible with per-vector
allocation/release requests.
Patches #7-8 modify the CPU-to-IRQ affinity calculation to be resilient
in case the affinity is requested but completion IRQ is not allocated yet.
Patch #9 function rename.
Patch #10 handles the corner case of SF performing an IRQ request when no
SF IRQ pool is found, and no PF IRQ exists for the same vector.
Patch #11 modify driver to use dynamically allocate completion EQs.

* tag 'mlx5-updates-2023-08-07' of git://git.kernel.org/pub/scm/linux/kernel/git/saeed/linux:
  net/mlx5: Bridge, Only handle registered netdev bridge events
  net/mlx5: E-Switch, Remove redundant arg ignore_flow_lvl
  net/mlx5: Fix typo reminder -> remainder
  net/mlx5: remove many unnecessary NULL values
  net/mlx5: Allocate completion EQs dynamically
  net/mlx5: Handle SF IRQ request in the absence of SF IRQ pool
  net/mlx5: Rename mlx5_comp_vectors_count() to mlx5_comp_vectors_max()
  net/mlx5: Add IRQ vector to CPU lookup function
  net/mlx5: Introduce mlx5_cpumask_default_spread
  net/mlx5: Implement single completion EQ create/destroy methods
  net/mlx5: Use xarray to store and manage completion EQs
  net/mlx5: Refactor completion IRQ request/release handlers in EQ layer
  net/mlx5: Use xarray to store and manage completion IRQs
  net/mlx5: Refactor completion IRQ request/release API
  net/mlx5: Track the current number of completion EQs
====================

Link: https://lore.kernel.org/r/20230807175642.20834-1-saeed@kernel.orgSigned-off-by: default avatarJakub Kicinski <kuba@kernel.org>
parents 2c2b8874 b56fb19c
...@@ -993,7 +993,7 @@ int mlx5_ib_create_cq(struct ib_cq *ibcq, const struct ib_cq_init_attr *attr, ...@@ -993,7 +993,7 @@ int mlx5_ib_create_cq(struct ib_cq *ibcq, const struct ib_cq_init_attr *attr,
INIT_WORK(&cq->notify_work, notify_soft_wc_handler); INIT_WORK(&cq->notify_work, notify_soft_wc_handler);
} }
err = mlx5_vector2eqn(dev->mdev, vector, &eqn); err = mlx5_comp_eqn_get(dev->mdev, vector, &eqn);
if (err) if (err)
goto err_cqb; goto err_cqb;
......
...@@ -1002,7 +1002,7 @@ static int UVERBS_HANDLER(MLX5_IB_METHOD_DEVX_QUERY_EQN)( ...@@ -1002,7 +1002,7 @@ static int UVERBS_HANDLER(MLX5_IB_METHOD_DEVX_QUERY_EQN)(
return PTR_ERR(c); return PTR_ERR(c);
dev = to_mdev(c->ibucontext.device); dev = to_mdev(c->ibucontext.device);
err = mlx5_vector2eqn(dev->mdev, user_vector, &dev_eqn); err = mlx5_comp_eqn_get(dev->mdev, user_vector, &dev_eqn);
if (err < 0) if (err < 0)
return err; return err;
......
...@@ -3685,7 +3685,7 @@ static int mlx5_ib_stage_init_init(struct mlx5_ib_dev *dev) ...@@ -3685,7 +3685,7 @@ static int mlx5_ib_stage_init_init(struct mlx5_ib_dev *dev)
if (mlx5_use_mad_ifc(dev)) if (mlx5_use_mad_ifc(dev))
get_ext_port_caps(dev); get_ext_port_caps(dev);
dev->ib_dev.num_comp_vectors = mlx5_comp_vectors_count(mdev); dev->ib_dev.num_comp_vectors = mlx5_comp_vectors_max(mdev);
mutex_init(&dev->cap_mask_mutex); mutex_init(&dev->cap_mask_mutex);
INIT_LIST_HEAD(&dev->qp_list); INIT_LIST_HEAD(&dev->qp_list);
......
...@@ -193,7 +193,7 @@ static inline int mlx5e_get_max_num_channels(struct mlx5_core_dev *mdev) ...@@ -193,7 +193,7 @@ static inline int mlx5e_get_max_num_channels(struct mlx5_core_dev *mdev)
{ {
return is_kdump_kernel() ? return is_kdump_kernel() ?
MLX5E_MIN_NUM_CHANNELS : MLX5E_MIN_NUM_CHANNELS :
min_t(int, mlx5_comp_vectors_count(mdev), MLX5E_MAX_NUM_CHANNELS); min_t(int, mlx5_comp_vectors_max(mdev), MLX5E_MAX_NUM_CHANNELS);
} }
/* The maximum WQE size can be retrieved by max_wqe_sz_sq in /* The maximum WQE size can be retrieved by max_wqe_sz_sq in
......
...@@ -77,6 +77,10 @@ mlx5_esw_bridge_rep_vport_num_vhca_id_get(struct net_device *dev, struct mlx5_es ...@@ -77,6 +77,10 @@ mlx5_esw_bridge_rep_vport_num_vhca_id_get(struct net_device *dev, struct mlx5_es
return NULL; return NULL;
priv = netdev_priv(dev); priv = netdev_priv(dev);
if (!priv->mdev->priv.eswitch->br_offloads)
return NULL;
rpriv = priv->ppriv; rpriv = priv->ppriv;
*vport_num = rpriv->rep->vport; *vport_num = rpriv->rep->vport;
*esw_owner_vhca_id = MLX5_CAP_GEN(priv->mdev, vhca_id); *esw_owner_vhca_id = MLX5_CAP_GEN(priv->mdev, vhca_id);
......
...@@ -127,7 +127,7 @@ static void mlx5e_build_trap_params(struct mlx5_core_dev *mdev, ...@@ -127,7 +127,7 @@ static void mlx5e_build_trap_params(struct mlx5_core_dev *mdev,
static struct mlx5e_trap *mlx5e_open_trap(struct mlx5e_priv *priv) static struct mlx5e_trap *mlx5e_open_trap(struct mlx5e_priv *priv)
{ {
int cpu = cpumask_first(mlx5_comp_irq_get_affinity_mask(priv->mdev, 0)); int cpu = mlx5_comp_vector_get_cpu(priv->mdev, 0);
struct net_device *netdev = priv->netdev; struct net_device *netdev = priv->netdev;
struct mlx5e_trap *t; struct mlx5e_trap *t;
int err; int err;
......
...@@ -1989,7 +1989,7 @@ static int mlx5e_create_cq(struct mlx5e_cq *cq, struct mlx5e_cq_param *param) ...@@ -1989,7 +1989,7 @@ static int mlx5e_create_cq(struct mlx5e_cq *cq, struct mlx5e_cq_param *param)
int eqn; int eqn;
int err; int err;
err = mlx5_vector2eqn(mdev, param->eq_ix, &eqn); err = mlx5_comp_eqn_get(mdev, param->eq_ix, &eqn);
if (err) if (err)
return err; return err;
...@@ -2445,14 +2445,14 @@ static int mlx5e_open_channel(struct mlx5e_priv *priv, int ix, ...@@ -2445,14 +2445,14 @@ static int mlx5e_open_channel(struct mlx5e_priv *priv, int ix,
struct xsk_buff_pool *xsk_pool, struct xsk_buff_pool *xsk_pool,
struct mlx5e_channel **cp) struct mlx5e_channel **cp)
{ {
int cpu = cpumask_first(mlx5_comp_irq_get_affinity_mask(priv->mdev, ix)); int cpu = mlx5_comp_vector_get_cpu(priv->mdev, ix);
struct net_device *netdev = priv->netdev; struct net_device *netdev = priv->netdev;
struct mlx5e_xsk_param xsk; struct mlx5e_xsk_param xsk;
struct mlx5e_channel *c; struct mlx5e_channel *c;
unsigned int irq; unsigned int irq;
int err; int err;
err = mlx5_vector2irqn(priv->mdev, ix, &irq); err = mlx5_comp_irqn_get(priv->mdev, ix, &irq);
if (err) if (err)
return err; return err;
...@@ -2856,13 +2856,13 @@ static void mlx5e_set_default_xps_cpumasks(struct mlx5e_priv *priv, ...@@ -2856,13 +2856,13 @@ static void mlx5e_set_default_xps_cpumasks(struct mlx5e_priv *priv,
struct mlx5_core_dev *mdev = priv->mdev; struct mlx5_core_dev *mdev = priv->mdev;
int num_comp_vectors, ix, irq; int num_comp_vectors, ix, irq;
num_comp_vectors = mlx5_comp_vectors_count(mdev); num_comp_vectors = mlx5_comp_vectors_max(mdev);
for (ix = 0; ix < params->num_channels; ix++) { for (ix = 0; ix < params->num_channels; ix++) {
cpumask_clear(priv->scratchpad.cpumask); cpumask_clear(priv->scratchpad.cpumask);
for (irq = ix; irq < num_comp_vectors; irq += params->num_channels) { for (irq = ix; irq < num_comp_vectors; irq += params->num_channels) {
int cpu = cpumask_first(mlx5_comp_irq_get_affinity_mask(mdev, irq)); int cpu = mlx5_comp_vector_get_cpu(mdev, irq);
cpumask_set_cpu(cpu, priv->scratchpad.cpumask); cpumask_set_cpu(cpu, priv->scratchpad.cpumask);
} }
......
...@@ -740,7 +740,7 @@ int mlx5_esw_qos_modify_vport_rate(struct mlx5_eswitch *esw, u16 vport_num, u32 ...@@ -740,7 +740,7 @@ int mlx5_esw_qos_modify_vport_rate(struct mlx5_eswitch *esw, u16 vport_num, u32
static int esw_qos_devlink_rate_to_mbps(struct mlx5_core_dev *mdev, const char *name, static int esw_qos_devlink_rate_to_mbps(struct mlx5_core_dev *mdev, const char *name,
u64 *rate, struct netlink_ext_ack *extack) u64 *rate, struct netlink_ext_ack *extack)
{ {
u32 link_speed_max, reminder; u32 link_speed_max, remainder;
u64 value; u64 value;
int err; int err;
...@@ -750,8 +750,8 @@ static int esw_qos_devlink_rate_to_mbps(struct mlx5_core_dev *mdev, const char * ...@@ -750,8 +750,8 @@ static int esw_qos_devlink_rate_to_mbps(struct mlx5_core_dev *mdev, const char *
return err; return err;
} }
value = div_u64_rem(*rate, MLX5_LINKSPEED_UNIT, &reminder); value = div_u64_rem(*rate, MLX5_LINKSPEED_UNIT, &remainder);
if (reminder) { if (remainder) {
pr_err("%s rate value %lluBps not in link speed units of 1Mbps.\n", pr_err("%s rate value %lluBps not in link speed units of 1Mbps.\n",
name, *rate); name, *rate);
NL_SET_ERR_MSG_MOD(extack, "TX rate value not in link speed units of 1Mbps"); NL_SET_ERR_MSG_MOD(extack, "TX rate value not in link speed units of 1Mbps");
......
...@@ -375,7 +375,6 @@ esw_setup_indir_table(struct mlx5_flow_destination *dest, ...@@ -375,7 +375,6 @@ esw_setup_indir_table(struct mlx5_flow_destination *dest,
struct mlx5_flow_act *flow_act, struct mlx5_flow_act *flow_act,
struct mlx5_eswitch *esw, struct mlx5_eswitch *esw,
struct mlx5_flow_attr *attr, struct mlx5_flow_attr *attr,
bool ignore_flow_lvl,
int *i) int *i)
{ {
struct mlx5_esw_flow_attr *esw_attr = attr->esw_attr; struct mlx5_esw_flow_attr *esw_attr = attr->esw_attr;
...@@ -385,8 +384,7 @@ esw_setup_indir_table(struct mlx5_flow_destination *dest, ...@@ -385,8 +384,7 @@ esw_setup_indir_table(struct mlx5_flow_destination *dest,
return -EOPNOTSUPP; return -EOPNOTSUPP;
for (j = esw_attr->split_count; j < esw_attr->out_count; j++, (*i)++) { for (j = esw_attr->split_count; j < esw_attr->out_count; j++, (*i)++) {
if (ignore_flow_lvl) flow_act->flags |= FLOW_ACT_IGNORE_FLOW_LEVEL;
flow_act->flags |= FLOW_ACT_IGNORE_FLOW_LEVEL;
dest[*i].type = MLX5_FLOW_DESTINATION_TYPE_FLOW_TABLE; dest[*i].type = MLX5_FLOW_DESTINATION_TYPE_FLOW_TABLE;
dest[*i].ft = mlx5_esw_indir_table_get(esw, attr, dest[*i].ft = mlx5_esw_indir_table_get(esw, attr,
...@@ -569,7 +567,7 @@ esw_setup_dests(struct mlx5_flow_destination *dest, ...@@ -569,7 +567,7 @@ esw_setup_dests(struct mlx5_flow_destination *dest,
err = esw_setup_mtu_dest(dest, &attr->meter_attr, *i); err = esw_setup_mtu_dest(dest, &attr->meter_attr, *i);
(*i)++; (*i)++;
} else if (esw_is_indir_table(esw, attr)) { } else if (esw_is_indir_table(esw, attr)) {
err = esw_setup_indir_table(dest, flow_act, esw, attr, true, i); err = esw_setup_indir_table(dest, flow_act, esw, attr, i);
} else if (esw_is_chain_src_port_rewrite(esw, esw_attr)) { } else if (esw_is_chain_src_port_rewrite(esw, esw_attr)) {
err = esw_setup_chain_src_port_rewrite(dest, flow_act, esw, chains, attr, i); err = esw_setup_chain_src_port_rewrite(dest, flow_act, esw, chains, attr, i);
} else { } else {
......
...@@ -445,7 +445,7 @@ static int mlx5_fpga_conn_create_cq(struct mlx5_fpga_conn *conn, int cq_size) ...@@ -445,7 +445,7 @@ static int mlx5_fpga_conn_create_cq(struct mlx5_fpga_conn *conn, int cq_size)
goto err_cqwq; goto err_cqwq;
} }
err = mlx5_vector2eqn(mdev, smp_processor_id(), &eqn); err = mlx5_comp_eqn_get(mdev, smp_processor_id(), &eqn);
if (err) { if (err) {
kvfree(in); kvfree(in);
goto err_cqwq; goto err_cqwq;
......
...@@ -57,7 +57,7 @@ static const char * const mlx5_fpga_qp_error_strings[] = { ...@@ -57,7 +57,7 @@ static const char * const mlx5_fpga_qp_error_strings[] = {
}; };
static struct mlx5_fpga_device *mlx5_fpga_device_alloc(void) static struct mlx5_fpga_device *mlx5_fpga_device_alloc(void)
{ {
struct mlx5_fpga_device *fdev = NULL; struct mlx5_fpga_device *fdev;
fdev = kzalloc(sizeof(*fdev), GFP_KERNEL); fdev = kzalloc(sizeof(*fdev), GFP_KERNEL);
if (!fdev) if (!fdev)
...@@ -252,7 +252,7 @@ int mlx5_fpga_device_start(struct mlx5_core_dev *mdev) ...@@ -252,7 +252,7 @@ int mlx5_fpga_device_start(struct mlx5_core_dev *mdev)
int mlx5_fpga_init(struct mlx5_core_dev *mdev) int mlx5_fpga_init(struct mlx5_core_dev *mdev)
{ {
struct mlx5_fpga_device *fdev = NULL; struct mlx5_fpga_device *fdev;
if (!MLX5_CAP_GEN(mdev, fpga)) { if (!MLX5_CAP_GEN(mdev, fpga)) {
mlx5_core_dbg(mdev, "FPGA capability not present\n"); mlx5_core_dbg(mdev, "FPGA capability not present\n");
......
...@@ -156,67 +156,57 @@ mlx5_irq_affinity_request(struct mlx5_irq_pool *pool, struct irq_affinity_desc * ...@@ -156,67 +156,57 @@ mlx5_irq_affinity_request(struct mlx5_irq_pool *pool, struct irq_affinity_desc *
return least_loaded_irq; return least_loaded_irq;
} }
void mlx5_irq_affinity_irqs_release(struct mlx5_core_dev *dev, struct mlx5_irq **irqs, void mlx5_irq_affinity_irq_release(struct mlx5_core_dev *dev, struct mlx5_irq *irq)
int num_irqs)
{ {
struct mlx5_irq_pool *pool = mlx5_irq_pool_get(dev); struct mlx5_irq_pool *pool = mlx5_irq_pool_get(dev);
int i; int cpu;
for (i = 0; i < num_irqs; i++) {
int cpu = cpumask_first(mlx5_irq_get_affinity_mask(irqs[i]));
synchronize_irq(pci_irq_vector(pool->dev->pdev, cpu = cpumask_first(mlx5_irq_get_affinity_mask(irq));
mlx5_irq_get_index(irqs[i]))); synchronize_irq(pci_irq_vector(pool->dev->pdev,
if (mlx5_irq_put(irqs[i])) mlx5_irq_get_index(irq)));
if (pool->irqs_per_cpu) if (mlx5_irq_put(irq))
cpu_put(pool, cpu); if (pool->irqs_per_cpu)
} cpu_put(pool, cpu);
} }
/** /**
* mlx5_irq_affinity_irqs_request_auto - request one or more IRQs for mlx5 device. * mlx5_irq_affinity_irq_request_auto - request one IRQ for mlx5 device.
* @dev: mlx5 device that is requesting the IRQs. * @dev: mlx5 device that is requesting the IRQ.
* @nirqs: number of IRQs to request. * @used_cpus: cpumask of bounded cpus by the device
* @irqs: an output array of IRQs pointers. * @vecidx: vector index to request an IRQ for.
* *
* Each IRQ is bounded to at most 1 CPU. * Each IRQ is bounded to at most 1 CPU.
* This function is requesting IRQs according to the default assignment. * This function is requesting an IRQ according to the default assignment.
* The default assignment policy is: * The default assignment policy is:
* - in each iteration, request the least loaded IRQ which is not bound to any * - request the least loaded IRQ which is not bound to any
* CPU of the previous IRQs requested. * CPU of the previous IRQs requested.
* *
* This function returns the number of IRQs requested, (which might be smaller than * On success, this function updates used_cpus mask and returns an irq pointer.
* @nirqs), if successful, or a negative error code in case of an error. * In case of an error, an appropriate error pointer is returned.
*/ */
int mlx5_irq_affinity_irqs_request_auto(struct mlx5_core_dev *dev, int nirqs, struct mlx5_irq *mlx5_irq_affinity_irq_request_auto(struct mlx5_core_dev *dev,
struct mlx5_irq **irqs) struct cpumask *used_cpus, u16 vecidx)
{ {
struct mlx5_irq_pool *pool = mlx5_irq_pool_get(dev); struct mlx5_irq_pool *pool = mlx5_irq_pool_get(dev);
struct irq_affinity_desc af_desc = {}; struct irq_affinity_desc af_desc = {};
struct mlx5_irq *irq; struct mlx5_irq *irq;
int i = 0;
if (!mlx5_irq_pool_is_sf_pool(pool))
return ERR_PTR(-ENOENT);
af_desc.is_managed = 1; af_desc.is_managed = 1;
cpumask_copy(&af_desc.mask, cpu_online_mask); cpumask_copy(&af_desc.mask, cpu_online_mask);
for (i = 0; i < nirqs; i++) { cpumask_andnot(&af_desc.mask, &af_desc.mask, used_cpus);
if (mlx5_irq_pool_is_sf_pool(pool)) irq = mlx5_irq_affinity_request(pool, &af_desc);
irq = mlx5_irq_affinity_request(pool, &af_desc);
else if (IS_ERR(irq))
/* In case SF pool doesn't exists, fallback to the PF IRQs. return irq;
* The PF IRQs are already allocated and binded to CPU
* at this point. Hence, only an index is needed. cpumask_or(used_cpus, used_cpus, mlx5_irq_get_affinity_mask(irq));
*/ mlx5_core_dbg(pool->dev, "IRQ %u mapped to cpu %*pbl, %u EQs on this irq\n",
irq = mlx5_irq_request(dev, i, NULL, NULL); pci_irq_vector(dev->pdev, mlx5_irq_get_index(irq)),
if (IS_ERR(irq)) cpumask_pr_args(mlx5_irq_get_affinity_mask(irq)),
break; mlx5_irq_read_locked(irq) / MLX5_EQ_REFS_PER_IRQ);
irqs[i] = irq;
cpumask_clear_cpu(cpumask_first(mlx5_irq_get_affinity_mask(irq)), &af_desc.mask); return irq;
mlx5_core_dbg(pool->dev, "IRQ %u mapped to cpu %*pbl, %u EQs on this irq\n",
pci_irq_vector(dev->pdev, mlx5_irq_get_index(irq)),
cpumask_pr_args(mlx5_irq_get_affinity_mask(irq)),
mlx5_irq_read_locked(irq) / MLX5_EQ_REFS_PER_IRQ);
}
if (!i)
return PTR_ERR(irq);
return i;
} }
...@@ -81,7 +81,7 @@ static int create_aso_cq(struct mlx5_aso_cq *cq, void *cqc_data) ...@@ -81,7 +81,7 @@ static int create_aso_cq(struct mlx5_aso_cq *cq, void *cqc_data)
int inlen, eqn; int inlen, eqn;
int err; int err;
err = mlx5_vector2eqn(mdev, 0, &eqn); err = mlx5_comp_eqn_get(mdev, 0, &eqn);
if (err) if (err)
return err; return err;
......
...@@ -104,6 +104,6 @@ void mlx5_core_eq_free_irqs(struct mlx5_core_dev *dev); ...@@ -104,6 +104,6 @@ void mlx5_core_eq_free_irqs(struct mlx5_core_dev *dev);
struct cpu_rmap *mlx5_eq_table_get_rmap(struct mlx5_core_dev *dev); struct cpu_rmap *mlx5_eq_table_get_rmap(struct mlx5_core_dev *dev);
#endif #endif
int mlx5_vector2irqn(struct mlx5_core_dev *dev, int vector, unsigned int *irqn); int mlx5_comp_irqn_get(struct mlx5_core_dev *dev, int vector, unsigned int *irqn);
#endif #endif
...@@ -40,7 +40,7 @@ struct mlx5_hv_vhca_agent { ...@@ -40,7 +40,7 @@ struct mlx5_hv_vhca_agent {
struct mlx5_hv_vhca *mlx5_hv_vhca_create(struct mlx5_core_dev *dev) struct mlx5_hv_vhca *mlx5_hv_vhca_create(struct mlx5_core_dev *dev)
{ {
struct mlx5_hv_vhca *hv_vhca = NULL; struct mlx5_hv_vhca *hv_vhca;
hv_vhca = kzalloc(sizeof(*hv_vhca), GFP_KERNEL); hv_vhca = kzalloc(sizeof(*hv_vhca), GFP_KERNEL);
if (!hv_vhca) if (!hv_vhca)
......
...@@ -29,9 +29,9 @@ void mlx5_ctrl_irq_release(struct mlx5_irq *ctrl_irq); ...@@ -29,9 +29,9 @@ void mlx5_ctrl_irq_release(struct mlx5_irq *ctrl_irq);
struct mlx5_irq *mlx5_irq_request(struct mlx5_core_dev *dev, u16 vecidx, struct mlx5_irq *mlx5_irq_request(struct mlx5_core_dev *dev, u16 vecidx,
struct irq_affinity_desc *af_desc, struct irq_affinity_desc *af_desc,
struct cpu_rmap **rmap); struct cpu_rmap **rmap);
int mlx5_irqs_request_vectors(struct mlx5_core_dev *dev, u16 *cpus, int nirqs, struct mlx5_irq *mlx5_irq_request_vector(struct mlx5_core_dev *dev, u16 cpu,
struct mlx5_irq **irqs, struct cpu_rmap **rmap); u16 vecidx, struct cpu_rmap **rmap);
void mlx5_irqs_release_vectors(struct mlx5_irq **irqs, int nirqs); void mlx5_irq_release_vector(struct mlx5_irq *irq);
int mlx5_irq_attach_nb(struct mlx5_irq *irq, struct notifier_block *nb); int mlx5_irq_attach_nb(struct mlx5_irq *irq, struct notifier_block *nb);
int mlx5_irq_detach_nb(struct mlx5_irq *irq, struct notifier_block *nb); int mlx5_irq_detach_nb(struct mlx5_irq *irq, struct notifier_block *nb);
struct cpumask *mlx5_irq_get_affinity_mask(struct mlx5_irq *irq); struct cpumask *mlx5_irq_get_affinity_mask(struct mlx5_irq *irq);
...@@ -39,17 +39,17 @@ int mlx5_irq_get_index(struct mlx5_irq *irq); ...@@ -39,17 +39,17 @@ int mlx5_irq_get_index(struct mlx5_irq *irq);
struct mlx5_irq_pool; struct mlx5_irq_pool;
#ifdef CONFIG_MLX5_SF #ifdef CONFIG_MLX5_SF
int mlx5_irq_affinity_irqs_request_auto(struct mlx5_core_dev *dev, int nirqs, struct mlx5_irq *mlx5_irq_affinity_irq_request_auto(struct mlx5_core_dev *dev,
struct mlx5_irq **irqs); struct cpumask *used_cpus, u16 vecidx);
struct mlx5_irq *mlx5_irq_affinity_request(struct mlx5_irq_pool *pool, struct mlx5_irq *mlx5_irq_affinity_request(struct mlx5_irq_pool *pool,
struct irq_affinity_desc *af_desc); struct irq_affinity_desc *af_desc);
void mlx5_irq_affinity_irqs_release(struct mlx5_core_dev *dev, struct mlx5_irq **irqs, void mlx5_irq_affinity_irq_release(struct mlx5_core_dev *dev, struct mlx5_irq *irq);
int num_irqs);
#else #else
static inline int mlx5_irq_affinity_irqs_request_auto(struct mlx5_core_dev *dev, int nirqs, static inline
struct mlx5_irq **irqs) struct mlx5_irq *mlx5_irq_affinity_irq_request_auto(struct mlx5_core_dev *dev,
struct cpumask *used_cpus, u16 vecidx)
{ {
return -EOPNOTSUPP; return ERR_PTR(-EOPNOTSUPP);
} }
static inline struct mlx5_irq * static inline struct mlx5_irq *
...@@ -58,7 +58,9 @@ mlx5_irq_affinity_request(struct mlx5_irq_pool *pool, struct irq_affinity_desc * ...@@ -58,7 +58,9 @@ mlx5_irq_affinity_request(struct mlx5_irq_pool *pool, struct irq_affinity_desc *
return ERR_PTR(-EOPNOTSUPP); return ERR_PTR(-EOPNOTSUPP);
} }
static inline void mlx5_irq_affinity_irqs_release(struct mlx5_core_dev *dev, static inline
struct mlx5_irq **irqs, int num_irqs) {} void mlx5_irq_affinity_irq_release(struct mlx5_core_dev *dev, struct mlx5_irq *irq)
{
}
#endif #endif
#endif /* __MLX5_IRQ_H__ */ #endif /* __MLX5_IRQ_H__ */
...@@ -432,19 +432,10 @@ static struct mlx5_irq_pool *ctrl_irq_pool_get(struct mlx5_core_dev *dev) ...@@ -432,19 +432,10 @@ static struct mlx5_irq_pool *ctrl_irq_pool_get(struct mlx5_core_dev *dev)
return pool ? pool : irq_table->pcif_pool; return pool ? pool : irq_table->pcif_pool;
} }
/** static void _mlx5_irq_release(struct mlx5_irq *irq)
* mlx5_irqs_release - release one or more IRQs back to the system.
* @irqs: IRQs to be released.
* @nirqs: number of IRQs to be released.
*/
static void mlx5_irqs_release(struct mlx5_irq **irqs, int nirqs)
{ {
int i; synchronize_irq(irq->map.virq);
mlx5_irq_put(irq);
for (i = 0; i < nirqs; i++) {
synchronize_irq(irqs[i]->map.virq);
mlx5_irq_put(irqs[i]);
}
} }
/** /**
...@@ -453,7 +444,7 @@ static void mlx5_irqs_release(struct mlx5_irq **irqs, int nirqs) ...@@ -453,7 +444,7 @@ static void mlx5_irqs_release(struct mlx5_irq **irqs, int nirqs)
*/ */
void mlx5_ctrl_irq_release(struct mlx5_irq *ctrl_irq) void mlx5_ctrl_irq_release(struct mlx5_irq *ctrl_irq)
{ {
mlx5_irqs_release(&ctrl_irq, 1); _mlx5_irq_release(ctrl_irq);
} }
/** /**
...@@ -569,53 +560,42 @@ void mlx5_msix_free(struct mlx5_core_dev *dev, struct msi_map map) ...@@ -569,53 +560,42 @@ void mlx5_msix_free(struct mlx5_core_dev *dev, struct msi_map map)
EXPORT_SYMBOL(mlx5_msix_free); EXPORT_SYMBOL(mlx5_msix_free);
/** /**
* mlx5_irqs_release_vectors - release one or more IRQs back to the system. * mlx5_irq_release_vector - release one IRQ back to the system.
* @irqs: IRQs to be released. * @irq: the irq to release.
* @nirqs: number of IRQs to be released.
*/ */
void mlx5_irqs_release_vectors(struct mlx5_irq **irqs, int nirqs) void mlx5_irq_release_vector(struct mlx5_irq *irq)
{ {
mlx5_irqs_release(irqs, nirqs); _mlx5_irq_release(irq);
} }
/** /**
* mlx5_irqs_request_vectors - request one or more IRQs for mlx5 device. * mlx5_irq_request_vector - request one IRQ for mlx5 device.
* @dev: mlx5 device that is requesting the IRQs. * @dev: mlx5 device that is requesting the IRQ.
* @cpus: CPUs array for binding the IRQs * @cpu: CPU to bind the IRQ to.
* @nirqs: number of IRQs to request. * @vecidx: vector index to request an IRQ for.
* @irqs: an output array of IRQs pointers.
* @rmap: pointer to reverse map pointer for completion interrupts * @rmap: pointer to reverse map pointer for completion interrupts
* *
* Each IRQ is bound to at most 1 CPU. * Each IRQ is bound to at most 1 CPU.
* This function is requests nirqs IRQs, starting from @vecidx. * This function is requests one IRQ, for the given @vecidx.
* *
* This function returns the number of IRQs requested, (which might be smaller than * This function returns a pointer to the irq on success, or an error pointer
* @nirqs), if successful, or a negative error code in case of an error. * in case of an error.
*/ */
int mlx5_irqs_request_vectors(struct mlx5_core_dev *dev, u16 *cpus, int nirqs, struct mlx5_irq *mlx5_irq_request_vector(struct mlx5_core_dev *dev, u16 cpu,
struct mlx5_irq **irqs, struct cpu_rmap **rmap) u16 vecidx, struct cpu_rmap **rmap)
{ {
struct mlx5_irq_table *table = mlx5_irq_table_get(dev); struct mlx5_irq_table *table = mlx5_irq_table_get(dev);
struct mlx5_irq_pool *pool = table->pcif_pool; struct mlx5_irq_pool *pool = table->pcif_pool;
struct irq_affinity_desc af_desc; struct irq_affinity_desc af_desc;
struct mlx5_irq *irq;
int offset = 1; int offset = 1;
int i;
if (!pool->xa_num_irqs.max) if (!pool->xa_num_irqs.max)
offset = 0; offset = 0;
af_desc.is_managed = false; af_desc.is_managed = false;
for (i = 0; i < nirqs; i++) { cpumask_clear(&af_desc.mask);
cpumask_clear(&af_desc.mask); cpumask_set_cpu(cpu, &af_desc.mask);
cpumask_set_cpu(cpus[i], &af_desc.mask); return mlx5_irq_request(dev, vecidx + offset, &af_desc, rmap);
irq = mlx5_irq_request(dev, i + offset, &af_desc, rmap);
if (IS_ERR(irq))
break;
irqs[i] = irq;
}
return i ? i : PTR_ERR(irq);
} }
static struct mlx5_irq_pool * static struct mlx5_irq_pool *
......
...@@ -1096,8 +1096,8 @@ static struct mlx5dr_cq *dr_create_cq(struct mlx5_core_dev *mdev, ...@@ -1096,8 +1096,8 @@ static struct mlx5dr_cq *dr_create_cq(struct mlx5_core_dev *mdev,
if (!in) if (!in)
goto err_cqwq; goto err_cqwq;
vector = raw_smp_processor_id() % mlx5_comp_vectors_count(mdev); vector = raw_smp_processor_id() % mlx5_comp_vectors_max(mdev);
err = mlx5_vector2eqn(mdev, vector, &eqn); err = mlx5_comp_eqn_get(mdev, vector, &eqn);
if (err) { if (err) {
kvfree(in); kvfree(in);
goto err_cqwq; goto err_cqwq;
......
...@@ -580,7 +580,7 @@ static int cq_create(struct mlx5_vdpa_net *ndev, u16 idx, u32 num_ent) ...@@ -580,7 +580,7 @@ static int cq_create(struct mlx5_vdpa_net *ndev, u16 idx, u32 num_ent)
/* Use vector 0 by default. Consider adding code to choose least used /* Use vector 0 by default. Consider adding code to choose least used
* vector. * vector.
*/ */
err = mlx5_vector2eqn(mdev, 0, &eqn); err = mlx5_comp_eqn_get(mdev, 0, &eqn);
if (err) if (err)
goto err_vec; goto err_vec;
......
...@@ -1025,8 +1025,8 @@ static int mlx5vf_create_cq(struct mlx5_core_dev *mdev, ...@@ -1025,8 +1025,8 @@ static int mlx5vf_create_cq(struct mlx5_core_dev *mdev,
goto err_buff; goto err_buff;
} }
vector = raw_smp_processor_id() % mlx5_comp_vectors_count(mdev); vector = raw_smp_processor_id() % mlx5_comp_vectors_max(mdev);
err = mlx5_vector2eqn(mdev, vector, &eqn); err = mlx5_comp_eqn_get(mdev, vector, &eqn);
if (err) if (err)
goto err_vec; goto err_vec;
......
...@@ -1058,7 +1058,7 @@ void mlx5_unregister_debugfs(void); ...@@ -1058,7 +1058,7 @@ void mlx5_unregister_debugfs(void);
void mlx5_fill_page_frag_array_perm(struct mlx5_frag_buf *buf, __be64 *pas, u8 perm); void mlx5_fill_page_frag_array_perm(struct mlx5_frag_buf *buf, __be64 *pas, u8 perm);
void mlx5_fill_page_frag_array(struct mlx5_frag_buf *frag_buf, __be64 *pas); void mlx5_fill_page_frag_array(struct mlx5_frag_buf *frag_buf, __be64 *pas);
int mlx5_vector2eqn(struct mlx5_core_dev *dev, int vector, int *eqn); int mlx5_comp_eqn_get(struct mlx5_core_dev *dev, u16 vecidx, int *eqn);
int mlx5_core_attach_mcg(struct mlx5_core_dev *dev, union ib_gid *mgid, u32 qpn); int mlx5_core_attach_mcg(struct mlx5_core_dev *dev, union ib_gid *mgid, u32 qpn);
int mlx5_core_detach_mcg(struct mlx5_core_dev *dev, union ib_gid *mgid, u32 qpn); int mlx5_core_detach_mcg(struct mlx5_core_dev *dev, union ib_gid *mgid, u32 qpn);
...@@ -1108,9 +1108,8 @@ int mlx5_alloc_bfreg(struct mlx5_core_dev *mdev, struct mlx5_sq_bfreg *bfreg, ...@@ -1108,9 +1108,8 @@ int mlx5_alloc_bfreg(struct mlx5_core_dev *mdev, struct mlx5_sq_bfreg *bfreg,
bool map_wc, bool fast_path); bool map_wc, bool fast_path);
void mlx5_free_bfreg(struct mlx5_core_dev *mdev, struct mlx5_sq_bfreg *bfreg); void mlx5_free_bfreg(struct mlx5_core_dev *mdev, struct mlx5_sq_bfreg *bfreg);
unsigned int mlx5_comp_vectors_count(struct mlx5_core_dev *dev); unsigned int mlx5_comp_vectors_max(struct mlx5_core_dev *dev);
struct cpumask * int mlx5_comp_vector_get_cpu(struct mlx5_core_dev *dev, int vector);
mlx5_comp_irq_get_affinity_mask(struct mlx5_core_dev *dev, int vector);
unsigned int mlx5_core_reserved_gids_count(struct mlx5_core_dev *dev); unsigned int mlx5_core_reserved_gids_count(struct mlx5_core_dev *dev);
int mlx5_core_roce_gid_set(struct mlx5_core_dev *dev, unsigned int index, int mlx5_core_roce_gid_set(struct mlx5_core_dev *dev, unsigned int index,
u8 roce_version, u8 roce_l3_type, const u8 *gid, u8 roce_version, u8 roce_l3_type, const u8 *gid,
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment