1 // SPDX-License-Identifier: GPL-2.0 OR Linux-OpenIB
2 // Copyright (c) 2020 Mellanox Technologies
8 struct mlx5e_skb_cb_hwtstamp {
10 ktime_t port_hwtstamp;
13 void mlx5e_skb_cb_hwtstamp_init(struct sk_buff *skb)
15 memset(skb->cb, 0, sizeof(struct mlx5e_skb_cb_hwtstamp));
18 static struct mlx5e_skb_cb_hwtstamp *mlx5e_skb_cb_get_hwts(struct sk_buff *skb)
20 BUILD_BUG_ON(sizeof(struct mlx5e_skb_cb_hwtstamp) > sizeof(skb->cb));
21 return (struct mlx5e_skb_cb_hwtstamp *)skb->cb;
24 static void mlx5e_skb_cb_hwtstamp_tx(struct sk_buff *skb,
25 struct mlx5e_ptp_cq_stats *cq_stats)
27 struct skb_shared_hwtstamps hwts = {};
30 diff = abs(mlx5e_skb_cb_get_hwts(skb)->port_hwtstamp -
31 mlx5e_skb_cb_get_hwts(skb)->cqe_hwtstamp);
33 /* Maximal allowed diff is 1 / 128 second */
34 if (diff > (NSEC_PER_SEC >> 7)) {
36 cq_stats->abort_abs_diff_ns += diff;
40 hwts.hwtstamp = mlx5e_skb_cb_get_hwts(skb)->port_hwtstamp;
41 skb_tstamp_tx(skb, &hwts);
44 void mlx5e_skb_cb_hwtstamp_handler(struct sk_buff *skb, int hwtstamp_type,
46 struct mlx5e_ptp_cq_stats *cq_stats)
48 switch (hwtstamp_type) {
49 case (MLX5E_SKB_CB_CQE_HWTSTAMP):
50 mlx5e_skb_cb_get_hwts(skb)->cqe_hwtstamp = hwtstamp;
52 case (MLX5E_SKB_CB_PORT_HWTSTAMP):
53 mlx5e_skb_cb_get_hwts(skb)->port_hwtstamp = hwtstamp;
57 /* If both CQEs arrive, check and report the port tstamp, and clear skb cb as
58 * skb soon to be released.
60 if (!mlx5e_skb_cb_get_hwts(skb)->cqe_hwtstamp ||
61 !mlx5e_skb_cb_get_hwts(skb)->port_hwtstamp)
64 mlx5e_skb_cb_hwtstamp_tx(skb, cq_stats);
65 memset(skb->cb, 0, sizeof(struct mlx5e_skb_cb_hwtstamp));
68 static void mlx5e_ptp_handle_ts_cqe(struct mlx5e_ptpsq *ptpsq,
69 struct mlx5_cqe64 *cqe,
72 struct sk_buff *skb = mlx5e_skb_fifo_pop(&ptpsq->skb_fifo);
75 if (unlikely(MLX5E_RX_ERR_CQE(cqe))) {
76 ptpsq->cq_stats->err_cqe++;
80 hwtstamp = mlx5_timecounter_cyc2time(ptpsq->txqsq.clock, get_cqe_ts(cqe));
81 mlx5e_skb_cb_hwtstamp_handler(skb, MLX5E_SKB_CB_PORT_HWTSTAMP,
82 hwtstamp, ptpsq->cq_stats);
83 ptpsq->cq_stats->cqe++;
86 napi_consume_skb(skb, budget);
89 static bool mlx5e_ptp_poll_ts_cq(struct mlx5e_cq *cq, int budget)
91 struct mlx5e_ptpsq *ptpsq = container_of(cq, struct mlx5e_ptpsq, ts_cq);
92 struct mlx5_cqwq *cqwq = &cq->wq;
93 struct mlx5_cqe64 *cqe;
96 if (unlikely(!test_bit(MLX5E_SQ_STATE_ENABLED, &ptpsq->txqsq.state)))
99 cqe = mlx5_cqwq_get_cqe(cqwq);
106 mlx5e_ptp_handle_ts_cqe(ptpsq, cqe, budget);
107 } while ((++work_done < budget) && (cqe = mlx5_cqwq_get_cqe(cqwq)));
109 mlx5_cqwq_update_db_record(cqwq);
111 /* ensure cq space is freed before enabling more cqes */
114 return work_done == budget;
117 static int mlx5e_ptp_napi_poll(struct napi_struct *napi, int budget)
119 struct mlx5e_port_ptp *c = container_of(napi, struct mlx5e_port_ptp,
121 struct mlx5e_ch_stats *ch_stats = c->stats;
130 for (i = 0; i < c->num_tc; i++) {
131 busy |= mlx5e_poll_tx_cq(&c->ptpsq[i].txqsq.cq, budget);
132 busy |= mlx5e_ptp_poll_ts_cq(&c->ptpsq[i].ts_cq, budget);
140 if (unlikely(!napi_complete_done(napi, work_done)))
145 for (i = 0; i < c->num_tc; i++) {
146 mlx5e_cq_arm(&c->ptpsq[i].txqsq.cq);
147 mlx5e_cq_arm(&c->ptpsq[i].ts_cq);
156 static int mlx5e_ptp_alloc_txqsq(struct mlx5e_port_ptp *c, int txq_ix,
157 struct mlx5e_params *params,
158 struct mlx5e_sq_param *param,
159 struct mlx5e_txqsq *sq, int tc,
160 struct mlx5e_ptpsq *ptpsq)
162 void *sqc_wq = MLX5_ADDR_OF(sqc, param->sqc, wq);
163 struct mlx5_core_dev *mdev = c->mdev;
164 struct mlx5_wq_cyc *wq = &sq->wq;
169 sq->tstamp = c->tstamp;
170 sq->clock = &mdev->clock;
171 sq->mkey_be = c->mkey_be;
172 sq->netdev = c->netdev;
177 sq->uar_map = mdev->mlx5e_res.bfreg.map;
178 sq->min_inline_mode = params->tx_min_inline_mode;
179 sq->hw_mtu = MLX5E_SW2HW_MTU(params, params->sw_mtu);
180 sq->stats = &c->priv->port_ptp_stats.sq[tc];
182 INIT_WORK(&sq->recover_work, mlx5e_tx_err_cqe_work);
183 if (!MLX5_CAP_ETH(mdev, wqe_vlan_insert))
184 set_bit(MLX5E_SQ_STATE_VLAN_NEED_L2_INLINE, &sq->state);
185 sq->stop_room = param->stop_room;
187 node = dev_to_node(mlx5_core_dma_dev(mdev));
189 param->wq.db_numa_node = node;
190 err = mlx5_wq_cyc_create(mdev, ¶m->wq, sqc_wq, wq, &sq->wq_ctrl);
193 wq->db = &wq->db[MLX5_SND_DBR];
195 err = mlx5e_alloc_txqsq_db(sq, node);
197 goto err_sq_wq_destroy;
202 mlx5_wq_destroy(&sq->wq_ctrl);
207 static void mlx5e_ptp_destroy_sq(struct mlx5_core_dev *mdev, u32 sqn)
209 mlx5_core_destroy_sq(mdev, sqn);
212 static int mlx5e_ptp_alloc_traffic_db(struct mlx5e_ptpsq *ptpsq, int numa)
214 int wq_sz = mlx5_wq_cyc_get_size(&ptpsq->txqsq.wq);
216 ptpsq->skb_fifo.fifo = kvzalloc_node(array_size(wq_sz, sizeof(*ptpsq->skb_fifo.fifo)),
218 if (!ptpsq->skb_fifo.fifo)
221 ptpsq->skb_fifo.pc = &ptpsq->skb_fifo_pc;
222 ptpsq->skb_fifo.cc = &ptpsq->skb_fifo_cc;
223 ptpsq->skb_fifo.mask = wq_sz - 1;
228 static void mlx5e_ptp_drain_skb_fifo(struct mlx5e_skb_fifo *skb_fifo)
230 while (*skb_fifo->pc != *skb_fifo->cc) {
231 struct sk_buff *skb = mlx5e_skb_fifo_pop(skb_fifo);
233 dev_kfree_skb_any(skb);
237 static void mlx5e_ptp_free_traffic_db(struct mlx5e_skb_fifo *skb_fifo)
239 mlx5e_ptp_drain_skb_fifo(skb_fifo);
240 kvfree(skb_fifo->fifo);
243 static int mlx5e_ptp_open_txqsq(struct mlx5e_port_ptp *c, u32 tisn,
244 int txq_ix, struct mlx5e_ptp_params *cparams,
245 int tc, struct mlx5e_ptpsq *ptpsq)
247 struct mlx5e_sq_param *sqp = &cparams->txq_sq_param;
248 struct mlx5e_txqsq *txqsq = &ptpsq->txqsq;
249 struct mlx5e_create_sq_param csp = {};
252 err = mlx5e_ptp_alloc_txqsq(c, txq_ix, &cparams->params, sqp,
259 csp.cqn = txqsq->cq.mcq.cqn;
260 csp.wq_ctrl = &txqsq->wq_ctrl;
261 csp.min_inline_mode = txqsq->min_inline_mode;
262 csp.ts_cqe_to_dest_cqn = ptpsq->ts_cq.mcq.cqn;
264 err = mlx5e_create_sq_rdy(c->mdev, sqp, &csp, &txqsq->sqn);
268 err = mlx5e_ptp_alloc_traffic_db(ptpsq,
269 dev_to_node(mlx5_core_dma_dev(c->mdev)));
276 mlx5e_free_txqsq(txqsq);
281 static void mlx5e_ptp_close_txqsq(struct mlx5e_ptpsq *ptpsq)
283 struct mlx5e_txqsq *sq = &ptpsq->txqsq;
284 struct mlx5_core_dev *mdev = sq->mdev;
286 mlx5e_ptp_free_traffic_db(&ptpsq->skb_fifo);
287 cancel_work_sync(&sq->recover_work);
288 mlx5e_ptp_destroy_sq(mdev, sq->sqn);
289 mlx5e_free_txqsq_descs(sq);
290 mlx5e_free_txqsq(sq);
293 static int mlx5e_ptp_open_txqsqs(struct mlx5e_port_ptp *c,
294 struct mlx5e_ptp_params *cparams)
296 struct mlx5e_params *params = &cparams->params;
301 ix_base = params->num_tc * params->num_channels;
303 for (tc = 0; tc < params->num_tc; tc++) {
304 int txq_ix = ix_base + tc;
306 err = mlx5e_ptp_open_txqsq(c, c->priv->tisn[c->lag_port][tc], txq_ix,
307 cparams, tc, &c->ptpsq[tc]);
315 for (--tc; tc >= 0; tc--)
316 mlx5e_ptp_close_txqsq(&c->ptpsq[tc]);
321 static void mlx5e_ptp_close_txqsqs(struct mlx5e_port_ptp *c)
325 for (tc = 0; tc < c->num_tc; tc++)
326 mlx5e_ptp_close_txqsq(&c->ptpsq[tc]);
329 static int mlx5e_ptp_open_cqs(struct mlx5e_port_ptp *c,
330 struct mlx5e_ptp_params *cparams)
332 struct mlx5e_params *params = &cparams->params;
333 struct mlx5e_create_cq_param ccp = {};
334 struct dim_cq_moder ptp_moder = {};
335 struct mlx5e_cq_param *cq_param;
339 ccp.node = dev_to_node(mlx5_core_dma_dev(c->mdev));
340 ccp.ch_stats = c->stats;
344 cq_param = &cparams->txq_sq_param.cqp;
346 for (tc = 0; tc < params->num_tc; tc++) {
347 struct mlx5e_cq *cq = &c->ptpsq[tc].txqsq.cq;
349 err = mlx5e_open_cq(c->priv, ptp_moder, cq_param, &ccp, cq);
351 goto out_err_txqsq_cq;
354 for (tc = 0; tc < params->num_tc; tc++) {
355 struct mlx5e_cq *cq = &c->ptpsq[tc].ts_cq;
356 struct mlx5e_ptpsq *ptpsq = &c->ptpsq[tc];
358 err = mlx5e_open_cq(c->priv, ptp_moder, cq_param, &ccp, cq);
362 ptpsq->cq_stats = &c->priv->port_ptp_stats.cq[tc];
368 for (--tc; tc >= 0; tc--)
369 mlx5e_close_cq(&c->ptpsq[tc].ts_cq);
372 for (--tc; tc >= 0; tc--)
373 mlx5e_close_cq(&c->ptpsq[tc].txqsq.cq);
378 static void mlx5e_ptp_close_cqs(struct mlx5e_port_ptp *c)
382 for (tc = 0; tc < c->num_tc; tc++)
383 mlx5e_close_cq(&c->ptpsq[tc].ts_cq);
385 for (tc = 0; tc < c->num_tc; tc++)
386 mlx5e_close_cq(&c->ptpsq[tc].txqsq.cq);
389 static void mlx5e_ptp_build_sq_param(struct mlx5e_priv *priv,
390 struct mlx5e_params *params,
391 struct mlx5e_sq_param *param)
393 void *sqc = param->sqc;
396 mlx5e_build_sq_param_common(priv, param);
398 wq = MLX5_ADDR_OF(sqc, sqc, wq);
399 MLX5_SET(wq, wq, log_wq_sz, params->log_sq_size);
400 param->stop_room = mlx5e_stop_room_for_wqe(MLX5_SEND_WQE_MAX_WQEBBS);
401 mlx5e_build_tx_cq_param(priv, params, ¶m->cqp);
404 static void mlx5e_ptp_build_params(struct mlx5e_port_ptp *c,
405 struct mlx5e_ptp_params *cparams,
406 struct mlx5e_params *orig)
408 struct mlx5e_params *params = &cparams->params;
410 params->tx_min_inline_mode = orig->tx_min_inline_mode;
411 params->num_channels = orig->num_channels;
412 params->hard_mtu = orig->hard_mtu;
413 params->sw_mtu = orig->sw_mtu;
414 params->num_tc = orig->num_tc;
417 params->log_sq_size = orig->log_sq_size;
419 mlx5e_ptp_build_sq_param(c->priv, params, &cparams->txq_sq_param);
422 static int mlx5e_ptp_open_queues(struct mlx5e_port_ptp *c,
423 struct mlx5e_ptp_params *cparams)
427 err = mlx5e_ptp_open_cqs(c, cparams);
431 napi_enable(&c->napi);
433 err = mlx5e_ptp_open_txqsqs(c, cparams);
440 napi_disable(&c->napi);
441 mlx5e_ptp_close_cqs(c);
446 static void mlx5e_ptp_close_queues(struct mlx5e_port_ptp *c)
448 mlx5e_ptp_close_txqsqs(c);
449 napi_disable(&c->napi);
450 mlx5e_ptp_close_cqs(c);
453 int mlx5e_port_ptp_open(struct mlx5e_priv *priv, struct mlx5e_params *params,
454 u8 lag_port, struct mlx5e_port_ptp **cp)
456 struct net_device *netdev = priv->netdev;
457 struct mlx5_core_dev *mdev = priv->mdev;
458 struct mlx5e_ptp_params *cparams;
459 struct mlx5e_port_ptp *c;
464 err = mlx5_vector2eqn(priv->mdev, 0, &eqn, &irq);
468 c = kvzalloc_node(sizeof(*c), GFP_KERNEL, dev_to_node(mlx5_core_dma_dev(mdev)));
469 cparams = kvzalloc(sizeof(*cparams), GFP_KERNEL);
474 c->mdev = priv->mdev;
475 c->tstamp = &priv->tstamp;
477 c->pdev = mlx5_core_dma_dev(priv->mdev);
478 c->netdev = priv->netdev;
479 c->mkey_be = cpu_to_be32(priv->mdev->mlx5e_res.mkey.key);
480 c->num_tc = params->num_tc;
481 c->stats = &priv->port_ptp_stats.ch;
482 c->lag_port = lag_port;
484 netif_napi_add(netdev, &c->napi, mlx5e_ptp_napi_poll, 64);
486 mlx5e_ptp_build_params(c, cparams, params);
488 err = mlx5e_ptp_open_queues(c, cparams);
499 netif_napi_del(&c->napi);
506 void mlx5e_port_ptp_close(struct mlx5e_port_ptp *c)
508 mlx5e_ptp_close_queues(c);
509 netif_napi_del(&c->napi);
514 void mlx5e_ptp_activate_channel(struct mlx5e_port_ptp *c)
518 for (tc = 0; tc < c->num_tc; tc++)
519 mlx5e_activate_txqsq(&c->ptpsq[tc].txqsq);
522 void mlx5e_ptp_deactivate_channel(struct mlx5e_port_ptp *c)
526 for (tc = 0; tc < c->num_tc; tc++)
527 mlx5e_deactivate_txqsq(&c->ptpsq[tc].txqsq);