|
@@ -513,7 +513,13 @@ static int mlx5e_create_rq(struct mlx5e_channel *c,
|
|
|
rq->channel = c;
|
|
|
rq->ix = c->ix;
|
|
|
rq->priv = c->priv;
|
|
|
- rq->xdp_prog = priv->xdp_prog;
|
|
|
+
|
|
|
+ rq->xdp_prog = priv->xdp_prog ? bpf_prog_inc(priv->xdp_prog) : NULL;
|
|
|
+ if (IS_ERR(rq->xdp_prog)) {
|
|
|
+ err = PTR_ERR(rq->xdp_prog);
|
|
|
+ rq->xdp_prog = NULL;
|
|
|
+ goto err_rq_wq_destroy;
|
|
|
+ }
|
|
|
|
|
|
rq->buff.map_dir = DMA_FROM_DEVICE;
|
|
|
if (rq->xdp_prog)
|
|
@@ -590,12 +596,11 @@ static int mlx5e_create_rq(struct mlx5e_channel *c,
|
|
|
rq->page_cache.head = 0;
|
|
|
rq->page_cache.tail = 0;
|
|
|
|
|
|
- if (rq->xdp_prog)
|
|
|
- bpf_prog_add(rq->xdp_prog, 1);
|
|
|
-
|
|
|
return 0;
|
|
|
|
|
|
err_rq_wq_destroy:
|
|
|
+ if (rq->xdp_prog)
|
|
|
+ bpf_prog_put(rq->xdp_prog);
|
|
|
mlx5_wq_destroy(&rq->wq_ctrl);
|
|
|
|
|
|
return err;
|
|
@@ -3139,11 +3144,21 @@ static int mlx5e_xdp_set(struct net_device *netdev, struct bpf_prog *prog)
|
|
|
|
|
|
if (was_opened && reset)
|
|
|
mlx5e_close_locked(netdev);
|
|
|
+ if (was_opened && !reset) {
|
|
|
+ /* num_channels is invariant here, so we can take the
|
|
|
+ * batched reference right upfront.
|
|
|
+ */
|
|
|
+ prog = bpf_prog_add(prog, priv->params.num_channels);
|
|
|
+ if (IS_ERR(prog)) {
|
|
|
+ err = PTR_ERR(prog);
|
|
|
+ goto unlock;
|
|
|
+ }
|
|
|
+ }
|
|
|
|
|
|
- /* exchange programs */
|
|
|
+ /* exchange programs, extra prog reference we got from caller
|
|
|
+ * as long as we don't fail from this point onwards.
|
|
|
+ */
|
|
|
old_prog = xchg(&priv->xdp_prog, prog);
|
|
|
- if (prog)
|
|
|
- bpf_prog_add(prog, 1);
|
|
|
if (old_prog)
|
|
|
bpf_prog_put(old_prog);
|
|
|
|
|
@@ -3159,7 +3174,6 @@ static int mlx5e_xdp_set(struct net_device *netdev, struct bpf_prog *prog)
|
|
|
/* exchanging programs w/o reset, we update ref counts on behalf
|
|
|
* of the channels RQs here.
|
|
|
*/
|
|
|
- bpf_prog_add(prog, priv->params.num_channels);
|
|
|
for (i = 0; i < priv->params.num_channels; i++) {
|
|
|
struct mlx5e_channel *c = priv->channel[i];
|
|
|
|
|
@@ -3691,6 +3705,9 @@ static void mlx5e_nic_cleanup(struct mlx5e_priv *priv)
|
|
|
|
|
|
if (MLX5_CAP_GEN(mdev, vport_group_manager))
|
|
|
mlx5_eswitch_unregister_vport_rep(esw, 0);
|
|
|
+
|
|
|
+ if (priv->xdp_prog)
|
|
|
+ bpf_prog_put(priv->xdp_prog);
|
|
|
}
|
|
|
|
|
|
static int mlx5e_init_nic_rx(struct mlx5e_priv *priv)
|