2006 lines
		
	
	
		
			56 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			2006 lines
		
	
	
		
			56 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: BSD-3-Clause OR GPL-2.0
 | |
| /* Copyright (c) 2017-2018 Mellanox Technologies. All rights reserved */
 | |
| 
 | |
| #include <linux/kernel.h>
 | |
| #include <linux/errno.h>
 | |
| #include <linux/netdevice.h>
 | |
| #include <net/pkt_cls.h>
 | |
| #include <net/red.h>
 | |
| 
 | |
| #include "spectrum.h"
 | |
| #include "spectrum_span.h"
 | |
| #include "reg.h"
 | |
| 
 | |
| #define MLXSW_SP_PRIO_BAND_TO_TCLASS(band) (IEEE_8021QAZ_MAX_TCS - band - 1)
 | |
| #define MLXSW_SP_PRIO_CHILD_TO_TCLASS(child) \
 | |
| 	MLXSW_SP_PRIO_BAND_TO_TCLASS((child - 1))
 | |
| 
 | |
| enum mlxsw_sp_qdisc_type {
 | |
| 	MLXSW_SP_QDISC_NO_QDISC,
 | |
| 	MLXSW_SP_QDISC_RED,
 | |
| 	MLXSW_SP_QDISC_PRIO,
 | |
| 	MLXSW_SP_QDISC_ETS,
 | |
| 	MLXSW_SP_QDISC_TBF,
 | |
| 	MLXSW_SP_QDISC_FIFO,
 | |
| };
 | |
| 
 | |
| struct mlxsw_sp_qdisc;
 | |
| 
 | |
| struct mlxsw_sp_qdisc_ops {
 | |
| 	enum mlxsw_sp_qdisc_type type;
 | |
| 	int (*check_params)(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 			    void *params);
 | |
| 	int (*replace)(struct mlxsw_sp_port *mlxsw_sp_port, u32 handle,
 | |
| 		       struct mlxsw_sp_qdisc *mlxsw_sp_qdisc, void *params);
 | |
| 	int (*destroy)(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 		       struct mlxsw_sp_qdisc *mlxsw_sp_qdisc);
 | |
| 	int (*get_stats)(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 			 struct mlxsw_sp_qdisc *mlxsw_sp_qdisc,
 | |
| 			 struct tc_qopt_offload_stats *stats_ptr);
 | |
| 	int (*get_xstats)(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 			  struct mlxsw_sp_qdisc *mlxsw_sp_qdisc,
 | |
| 			  void *xstats_ptr);
 | |
| 	void (*clean_stats)(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 			    struct mlxsw_sp_qdisc *mlxsw_sp_qdisc);
 | |
| 	/* unoffload - to be used for a qdisc that stops being offloaded without
 | |
| 	 * being destroyed.
 | |
| 	 */
 | |
| 	void (*unoffload)(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 			  struct mlxsw_sp_qdisc *mlxsw_sp_qdisc, void *params);
 | |
| 	struct mlxsw_sp_qdisc *(*find_class)(struct mlxsw_sp_qdisc *mlxsw_sp_qdisc,
 | |
| 					     u32 parent);
 | |
| 	unsigned int num_classes;
 | |
| };
 | |
| 
 | |
| struct mlxsw_sp_qdisc {
 | |
| 	u32 handle;
 | |
| 	int tclass_num;
 | |
| 	u8 prio_bitmap;
 | |
| 	union {
 | |
| 		struct red_stats red;
 | |
| 	} xstats_base;
 | |
| 	struct mlxsw_sp_qdisc_stats {
 | |
| 		u64 tx_bytes;
 | |
| 		u64 tx_packets;
 | |
| 		u64 drops;
 | |
| 		u64 overlimits;
 | |
| 		u64 backlog;
 | |
| 	} stats_base;
 | |
| 
 | |
| 	struct mlxsw_sp_qdisc_ops *ops;
 | |
| 	struct mlxsw_sp_qdisc *parent;
 | |
| 	struct mlxsw_sp_qdisc *qdiscs;
 | |
| 	unsigned int num_classes;
 | |
| };
 | |
| 
 | |
| struct mlxsw_sp_qdisc_state {
 | |
| 	struct mlxsw_sp_qdisc root_qdisc;
 | |
| 
 | |
| 	/* When a PRIO or ETS are added, the invisible FIFOs in their bands are
 | |
| 	 * created first. When notifications for these FIFOs arrive, it is not
 | |
| 	 * known what qdisc their parent handle refers to. It could be a
 | |
| 	 * newly-created PRIO that will replace the currently-offloaded one, or
 | |
| 	 * it could be e.g. a RED that will be attached below it.
 | |
| 	 *
 | |
| 	 * As the notifications start to arrive, use them to note what the
 | |
| 	 * future parent handle is, and keep track of which child FIFOs were
 | |
| 	 * seen. Then when the parent is known, retroactively offload those
 | |
| 	 * FIFOs.
 | |
| 	 */
 | |
| 	u32 future_handle;
 | |
| 	bool future_fifos[IEEE_8021QAZ_MAX_TCS];
 | |
| 	struct mutex lock; /* Protects qdisc state. */
 | |
| };
 | |
| 
 | |
| static bool
 | |
| mlxsw_sp_qdisc_compare(struct mlxsw_sp_qdisc *mlxsw_sp_qdisc, u32 handle)
 | |
| {
 | |
| 	return mlxsw_sp_qdisc->ops && mlxsw_sp_qdisc->handle == handle;
 | |
| }
 | |
| 
 | |
| static struct mlxsw_sp_qdisc *
 | |
| mlxsw_sp_qdisc_walk(struct mlxsw_sp_qdisc *qdisc,
 | |
| 		    struct mlxsw_sp_qdisc *(*pre)(struct mlxsw_sp_qdisc *,
 | |
| 						  void *),
 | |
| 		    void *data)
 | |
| {
 | |
| 	struct mlxsw_sp_qdisc *tmp;
 | |
| 	unsigned int i;
 | |
| 
 | |
| 	if (pre) {
 | |
| 		tmp = pre(qdisc, data);
 | |
| 		if (tmp)
 | |
| 			return tmp;
 | |
| 	}
 | |
| 
 | |
| 	if (qdisc->ops) {
 | |
| 		for (i = 0; i < qdisc->num_classes; i++) {
 | |
| 			tmp = &qdisc->qdiscs[i];
 | |
| 			if (qdisc->ops) {
 | |
| 				tmp = mlxsw_sp_qdisc_walk(tmp, pre, data);
 | |
| 				if (tmp)
 | |
| 					return tmp;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return NULL;
 | |
| }
 | |
| 
 | |
| static struct mlxsw_sp_qdisc *
 | |
| mlxsw_sp_qdisc_walk_cb_find(struct mlxsw_sp_qdisc *qdisc, void *data)
 | |
| {
 | |
| 	u32 parent = *(u32 *)data;
 | |
| 
 | |
| 	if (qdisc->ops && TC_H_MAJ(qdisc->handle) == TC_H_MAJ(parent)) {
 | |
| 		if (qdisc->ops->find_class)
 | |
| 			return qdisc->ops->find_class(qdisc, parent);
 | |
| 	}
 | |
| 
 | |
| 	return NULL;
 | |
| }
 | |
| 
 | |
| static struct mlxsw_sp_qdisc *
 | |
| mlxsw_sp_qdisc_find(struct mlxsw_sp_port *mlxsw_sp_port, u32 parent,
 | |
| 		    bool root_only)
 | |
| {
 | |
| 	struct mlxsw_sp_qdisc_state *qdisc_state = mlxsw_sp_port->qdisc;
 | |
| 
 | |
| 	if (!qdisc_state)
 | |
| 		return NULL;
 | |
| 	if (parent == TC_H_ROOT)
 | |
| 		return &qdisc_state->root_qdisc;
 | |
| 	if (root_only)
 | |
| 		return NULL;
 | |
| 	return mlxsw_sp_qdisc_walk(&qdisc_state->root_qdisc,
 | |
| 				   mlxsw_sp_qdisc_walk_cb_find, &parent);
 | |
| }
 | |
| 
 | |
| static struct mlxsw_sp_qdisc *
 | |
| mlxsw_sp_qdisc_walk_cb_find_by_handle(struct mlxsw_sp_qdisc *qdisc, void *data)
 | |
| {
 | |
| 	u32 handle = *(u32 *)data;
 | |
| 
 | |
| 	if (qdisc->ops && qdisc->handle == handle)
 | |
| 		return qdisc;
 | |
| 	return NULL;
 | |
| }
 | |
| 
 | |
| static struct mlxsw_sp_qdisc *
 | |
| mlxsw_sp_qdisc_find_by_handle(struct mlxsw_sp_port *mlxsw_sp_port, u32 handle)
 | |
| {
 | |
| 	struct mlxsw_sp_qdisc_state *qdisc_state = mlxsw_sp_port->qdisc;
 | |
| 
 | |
| 	if (!qdisc_state)
 | |
| 		return NULL;
 | |
| 	return mlxsw_sp_qdisc_walk(&qdisc_state->root_qdisc,
 | |
| 				   mlxsw_sp_qdisc_walk_cb_find_by_handle,
 | |
| 				   &handle);
 | |
| }
 | |
| 
 | |
| static void
 | |
| mlxsw_sp_qdisc_reduce_parent_backlog(struct mlxsw_sp_qdisc *mlxsw_sp_qdisc)
 | |
| {
 | |
| 	struct mlxsw_sp_qdisc *tmp;
 | |
| 
 | |
| 	for (tmp = mlxsw_sp_qdisc->parent; tmp; tmp = tmp->parent)
 | |
| 		tmp->stats_base.backlog -= mlxsw_sp_qdisc->stats_base.backlog;
 | |
| }
 | |
| 
 | |
| static int
 | |
| mlxsw_sp_qdisc_destroy(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 		       struct mlxsw_sp_qdisc *mlxsw_sp_qdisc)
 | |
| {
 | |
| 	struct mlxsw_sp_qdisc *root_qdisc = &mlxsw_sp_port->qdisc->root_qdisc;
 | |
| 	int err_hdroom = 0;
 | |
| 	int err = 0;
 | |
| 
 | |
| 	if (!mlxsw_sp_qdisc)
 | |
| 		return 0;
 | |
| 
 | |
| 	if (root_qdisc == mlxsw_sp_qdisc) {
 | |
| 		struct mlxsw_sp_hdroom hdroom = *mlxsw_sp_port->hdroom;
 | |
| 
 | |
| 		hdroom.mode = MLXSW_SP_HDROOM_MODE_DCB;
 | |
| 		mlxsw_sp_hdroom_prios_reset_buf_idx(&hdroom);
 | |
| 		mlxsw_sp_hdroom_bufs_reset_lossiness(&hdroom);
 | |
| 		mlxsw_sp_hdroom_bufs_reset_sizes(mlxsw_sp_port, &hdroom);
 | |
| 		err_hdroom = mlxsw_sp_hdroom_configure(mlxsw_sp_port, &hdroom);
 | |
| 	}
 | |
| 
 | |
| 	if (!mlxsw_sp_qdisc->ops)
 | |
| 		return 0;
 | |
| 
 | |
| 	mlxsw_sp_qdisc_reduce_parent_backlog(mlxsw_sp_qdisc);
 | |
| 	if (mlxsw_sp_qdisc->ops->destroy)
 | |
| 		err = mlxsw_sp_qdisc->ops->destroy(mlxsw_sp_port,
 | |
| 						   mlxsw_sp_qdisc);
 | |
| 	if (mlxsw_sp_qdisc->ops->clean_stats)
 | |
| 		mlxsw_sp_qdisc->ops->clean_stats(mlxsw_sp_port, mlxsw_sp_qdisc);
 | |
| 
 | |
| 	mlxsw_sp_qdisc->handle = TC_H_UNSPEC;
 | |
| 	mlxsw_sp_qdisc->ops = NULL;
 | |
| 	mlxsw_sp_qdisc->num_classes = 0;
 | |
| 	kfree(mlxsw_sp_qdisc->qdiscs);
 | |
| 	mlxsw_sp_qdisc->qdiscs = NULL;
 | |
| 	return err_hdroom ?: err;
 | |
| }
 | |
| 
 | |
| static int mlxsw_sp_qdisc_create(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 				 u32 handle,
 | |
| 				 struct mlxsw_sp_qdisc *mlxsw_sp_qdisc,
 | |
| 				 struct mlxsw_sp_qdisc_ops *ops, void *params)
 | |
| {
 | |
| 	struct mlxsw_sp_qdisc *root_qdisc = &mlxsw_sp_port->qdisc->root_qdisc;
 | |
| 	struct mlxsw_sp_hdroom orig_hdroom;
 | |
| 	unsigned int i;
 | |
| 	int err;
 | |
| 
 | |
| 	err = ops->check_params(mlxsw_sp_port, params);
 | |
| 	if (err)
 | |
| 		return err;
 | |
| 
 | |
| 	if (ops->num_classes) {
 | |
| 		mlxsw_sp_qdisc->qdiscs = kcalloc(ops->num_classes,
 | |
| 						 sizeof(*mlxsw_sp_qdisc->qdiscs),
 | |
| 						 GFP_KERNEL);
 | |
| 		if (!mlxsw_sp_qdisc->qdiscs)
 | |
| 			return -ENOMEM;
 | |
| 
 | |
| 		for (i = 0; i < ops->num_classes; i++)
 | |
| 			mlxsw_sp_qdisc->qdiscs[i].parent = mlxsw_sp_qdisc;
 | |
| 	}
 | |
| 
 | |
| 	orig_hdroom = *mlxsw_sp_port->hdroom;
 | |
| 	if (root_qdisc == mlxsw_sp_qdisc) {
 | |
| 		struct mlxsw_sp_hdroom hdroom = orig_hdroom;
 | |
| 
 | |
| 		hdroom.mode = MLXSW_SP_HDROOM_MODE_TC;
 | |
| 		mlxsw_sp_hdroom_prios_reset_buf_idx(&hdroom);
 | |
| 		mlxsw_sp_hdroom_bufs_reset_lossiness(&hdroom);
 | |
| 		mlxsw_sp_hdroom_bufs_reset_sizes(mlxsw_sp_port, &hdroom);
 | |
| 
 | |
| 		err = mlxsw_sp_hdroom_configure(mlxsw_sp_port, &hdroom);
 | |
| 		if (err)
 | |
| 			goto err_hdroom_configure;
 | |
| 	}
 | |
| 
 | |
| 	mlxsw_sp_qdisc->num_classes = ops->num_classes;
 | |
| 	mlxsw_sp_qdisc->ops = ops;
 | |
| 	mlxsw_sp_qdisc->handle = handle;
 | |
| 	err = ops->replace(mlxsw_sp_port, handle, mlxsw_sp_qdisc, params);
 | |
| 	if (err)
 | |
| 		goto err_replace;
 | |
| 
 | |
| 	return 0;
 | |
| 
 | |
| err_replace:
 | |
| 	mlxsw_sp_qdisc->handle = TC_H_UNSPEC;
 | |
| 	mlxsw_sp_qdisc->ops = NULL;
 | |
| 	mlxsw_sp_qdisc->num_classes = 0;
 | |
| 	mlxsw_sp_hdroom_configure(mlxsw_sp_port, &orig_hdroom);
 | |
| err_hdroom_configure:
 | |
| 	kfree(mlxsw_sp_qdisc->qdiscs);
 | |
| 	mlxsw_sp_qdisc->qdiscs = NULL;
 | |
| 	return err;
 | |
| }
 | |
| 
 | |
| static int
 | |
| mlxsw_sp_qdisc_change(struct mlxsw_sp_port *mlxsw_sp_port, u32 handle,
 | |
| 		      struct mlxsw_sp_qdisc *mlxsw_sp_qdisc, void *params)
 | |
| {
 | |
| 	struct mlxsw_sp_qdisc_ops *ops = mlxsw_sp_qdisc->ops;
 | |
| 	int err;
 | |
| 
 | |
| 	err = ops->check_params(mlxsw_sp_port, params);
 | |
| 	if (err)
 | |
| 		goto unoffload;
 | |
| 
 | |
| 	err = ops->replace(mlxsw_sp_port, handle, mlxsw_sp_qdisc, params);
 | |
| 	if (err)
 | |
| 		goto unoffload;
 | |
| 
 | |
| 	/* Check if the Qdisc changed. That includes a situation where an
 | |
| 	 * invisible Qdisc replaces another one, or is being added for the
 | |
| 	 * first time.
 | |
| 	 */
 | |
| 	if (mlxsw_sp_qdisc->handle != handle) {
 | |
| 		if (ops->clean_stats)
 | |
| 			ops->clean_stats(mlxsw_sp_port, mlxsw_sp_qdisc);
 | |
| 	}
 | |
| 
 | |
| 	mlxsw_sp_qdisc->handle = handle;
 | |
| 	return 0;
 | |
| 
 | |
| unoffload:
 | |
| 	if (ops->unoffload)
 | |
| 		ops->unoffload(mlxsw_sp_port, mlxsw_sp_qdisc, params);
 | |
| 
 | |
| 	mlxsw_sp_qdisc_destroy(mlxsw_sp_port, mlxsw_sp_qdisc);
 | |
| 	return err;
 | |
| }
 | |
| 
 | |
| static int
 | |
| mlxsw_sp_qdisc_replace(struct mlxsw_sp_port *mlxsw_sp_port, u32 handle,
 | |
| 		       struct mlxsw_sp_qdisc *mlxsw_sp_qdisc,
 | |
| 		       struct mlxsw_sp_qdisc_ops *ops, void *params)
 | |
| {
 | |
| 	if (mlxsw_sp_qdisc->ops && mlxsw_sp_qdisc->ops->type != ops->type)
 | |
| 		/* In case this location contained a different qdisc of the
 | |
| 		 * same type we can override the old qdisc configuration.
 | |
| 		 * Otherwise, we need to remove the old qdisc before setting the
 | |
| 		 * new one.
 | |
| 		 */
 | |
| 		mlxsw_sp_qdisc_destroy(mlxsw_sp_port, mlxsw_sp_qdisc);
 | |
| 
 | |
| 	if (!mlxsw_sp_qdisc->ops)
 | |
| 		return mlxsw_sp_qdisc_create(mlxsw_sp_port, handle,
 | |
| 					     mlxsw_sp_qdisc, ops, params);
 | |
| 	else
 | |
| 		return mlxsw_sp_qdisc_change(mlxsw_sp_port, handle,
 | |
| 					     mlxsw_sp_qdisc, params);
 | |
| }
 | |
| 
 | |
| static int
 | |
| mlxsw_sp_qdisc_get_stats(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 			 struct mlxsw_sp_qdisc *mlxsw_sp_qdisc,
 | |
| 			 struct tc_qopt_offload_stats *stats_ptr)
 | |
| {
 | |
| 	if (mlxsw_sp_qdisc && mlxsw_sp_qdisc->ops &&
 | |
| 	    mlxsw_sp_qdisc->ops->get_stats)
 | |
| 		return mlxsw_sp_qdisc->ops->get_stats(mlxsw_sp_port,
 | |
| 						      mlxsw_sp_qdisc,
 | |
| 						      stats_ptr);
 | |
| 
 | |
| 	return -EOPNOTSUPP;
 | |
| }
 | |
| 
 | |
| static int
 | |
| mlxsw_sp_qdisc_get_xstats(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 			  struct mlxsw_sp_qdisc *mlxsw_sp_qdisc,
 | |
| 			  void *xstats_ptr)
 | |
| {
 | |
| 	if (mlxsw_sp_qdisc && mlxsw_sp_qdisc->ops &&
 | |
| 	    mlxsw_sp_qdisc->ops->get_xstats)
 | |
| 		return mlxsw_sp_qdisc->ops->get_xstats(mlxsw_sp_port,
 | |
| 						      mlxsw_sp_qdisc,
 | |
| 						      xstats_ptr);
 | |
| 
 | |
| 	return -EOPNOTSUPP;
 | |
| }
 | |
| 
 | |
| static u64
 | |
| mlxsw_sp_xstats_backlog(struct mlxsw_sp_port_xstats *xstats, int tclass_num)
 | |
| {
 | |
| 	return xstats->backlog[tclass_num] +
 | |
| 	       xstats->backlog[tclass_num + 8];
 | |
| }
 | |
| 
 | |
| static u64
 | |
| mlxsw_sp_xstats_tail_drop(struct mlxsw_sp_port_xstats *xstats, int tclass_num)
 | |
| {
 | |
| 	return xstats->tail_drop[tclass_num] +
 | |
| 	       xstats->tail_drop[tclass_num + 8];
 | |
| }
 | |
| 
 | |
| static void
 | |
| mlxsw_sp_qdisc_bstats_per_priority_get(struct mlxsw_sp_port_xstats *xstats,
 | |
| 				       u8 prio_bitmap, u64 *tx_packets,
 | |
| 				       u64 *tx_bytes)
 | |
| {
 | |
| 	int i;
 | |
| 
 | |
| 	*tx_packets = 0;
 | |
| 	*tx_bytes = 0;
 | |
| 	for (i = 0; i < IEEE_8021QAZ_MAX_TCS; i++) {
 | |
| 		if (prio_bitmap & BIT(i)) {
 | |
| 			*tx_packets += xstats->tx_packets[i];
 | |
| 			*tx_bytes += xstats->tx_bytes[i];
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static void
 | |
| mlxsw_sp_qdisc_collect_tc_stats(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 				struct mlxsw_sp_qdisc *mlxsw_sp_qdisc,
 | |
| 				u64 *p_tx_bytes, u64 *p_tx_packets,
 | |
| 				u64 *p_drops, u64 *p_backlog)
 | |
| {
 | |
| 	int tclass_num = mlxsw_sp_qdisc->tclass_num;
 | |
| 	struct mlxsw_sp_port_xstats *xstats;
 | |
| 	u64 tx_bytes, tx_packets;
 | |
| 
 | |
| 	xstats = &mlxsw_sp_port->periodic_hw_stats.xstats;
 | |
| 	mlxsw_sp_qdisc_bstats_per_priority_get(xstats,
 | |
| 					       mlxsw_sp_qdisc->prio_bitmap,
 | |
| 					       &tx_packets, &tx_bytes);
 | |
| 
 | |
| 	*p_tx_packets += tx_packets;
 | |
| 	*p_tx_bytes += tx_bytes;
 | |
| 	*p_drops += xstats->wred_drop[tclass_num] +
 | |
| 		    mlxsw_sp_xstats_tail_drop(xstats, tclass_num);
 | |
| 	*p_backlog += mlxsw_sp_xstats_backlog(xstats, tclass_num);
 | |
| }
 | |
| 
 | |
| static void
 | |
| mlxsw_sp_qdisc_update_stats(struct mlxsw_sp *mlxsw_sp,
 | |
| 			    struct mlxsw_sp_qdisc *mlxsw_sp_qdisc,
 | |
| 			    u64 tx_bytes, u64 tx_packets,
 | |
| 			    u64 drops, u64 backlog,
 | |
| 			    struct tc_qopt_offload_stats *stats_ptr)
 | |
| {
 | |
| 	struct mlxsw_sp_qdisc_stats *stats_base = &mlxsw_sp_qdisc->stats_base;
 | |
| 
 | |
| 	tx_bytes -= stats_base->tx_bytes;
 | |
| 	tx_packets -= stats_base->tx_packets;
 | |
| 	drops -= stats_base->drops;
 | |
| 	backlog -= stats_base->backlog;
 | |
| 
 | |
| 	_bstats_update(stats_ptr->bstats, tx_bytes, tx_packets);
 | |
| 	stats_ptr->qstats->drops += drops;
 | |
| 	stats_ptr->qstats->backlog += mlxsw_sp_cells_bytes(mlxsw_sp, backlog);
 | |
| 
 | |
| 	stats_base->backlog += backlog;
 | |
| 	stats_base->drops += drops;
 | |
| 	stats_base->tx_bytes += tx_bytes;
 | |
| 	stats_base->tx_packets += tx_packets;
 | |
| }
 | |
| 
 | |
| static void
 | |
| mlxsw_sp_qdisc_get_tc_stats(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 			    struct mlxsw_sp_qdisc *mlxsw_sp_qdisc,
 | |
| 			    struct tc_qopt_offload_stats *stats_ptr)
 | |
| {
 | |
| 	u64 tx_packets = 0;
 | |
| 	u64 tx_bytes = 0;
 | |
| 	u64 backlog = 0;
 | |
| 	u64 drops = 0;
 | |
| 
 | |
| 	mlxsw_sp_qdisc_collect_tc_stats(mlxsw_sp_port, mlxsw_sp_qdisc,
 | |
| 					&tx_bytes, &tx_packets,
 | |
| 					&drops, &backlog);
 | |
| 	mlxsw_sp_qdisc_update_stats(mlxsw_sp_port->mlxsw_sp, mlxsw_sp_qdisc,
 | |
| 				    tx_bytes, tx_packets, drops, backlog,
 | |
| 				    stats_ptr);
 | |
| }
 | |
| 
 | |
| static int
 | |
| mlxsw_sp_tclass_congestion_enable(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 				  int tclass_num, u32 min, u32 max,
 | |
| 				  u32 probability, bool is_wred, bool is_ecn)
 | |
| {
 | |
| 	char cwtpm_cmd[MLXSW_REG_CWTPM_LEN];
 | |
| 	char cwtp_cmd[MLXSW_REG_CWTP_LEN];
 | |
| 	struct mlxsw_sp *mlxsw_sp = mlxsw_sp_port->mlxsw_sp;
 | |
| 	int err;
 | |
| 
 | |
| 	mlxsw_reg_cwtp_pack(cwtp_cmd, mlxsw_sp_port->local_port, tclass_num);
 | |
| 	mlxsw_reg_cwtp_profile_pack(cwtp_cmd, MLXSW_REG_CWTP_DEFAULT_PROFILE,
 | |
| 				    roundup(min, MLXSW_REG_CWTP_MIN_VALUE),
 | |
| 				    roundup(max, MLXSW_REG_CWTP_MIN_VALUE),
 | |
| 				    probability);
 | |
| 
 | |
| 	err = mlxsw_reg_write(mlxsw_sp->core, MLXSW_REG(cwtp), cwtp_cmd);
 | |
| 	if (err)
 | |
| 		return err;
 | |
| 
 | |
| 	mlxsw_reg_cwtpm_pack(cwtpm_cmd, mlxsw_sp_port->local_port, tclass_num,
 | |
| 			     MLXSW_REG_CWTP_DEFAULT_PROFILE, is_wred, is_ecn);
 | |
| 
 | |
| 	return mlxsw_reg_write(mlxsw_sp->core, MLXSW_REG(cwtpm), cwtpm_cmd);
 | |
| }
 | |
| 
 | |
| static int
 | |
| mlxsw_sp_tclass_congestion_disable(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 				   int tclass_num)
 | |
| {
 | |
| 	struct mlxsw_sp *mlxsw_sp = mlxsw_sp_port->mlxsw_sp;
 | |
| 	char cwtpm_cmd[MLXSW_REG_CWTPM_LEN];
 | |
| 
 | |
| 	mlxsw_reg_cwtpm_pack(cwtpm_cmd, mlxsw_sp_port->local_port, tclass_num,
 | |
| 			     MLXSW_REG_CWTPM_RESET_PROFILE, false, false);
 | |
| 	return mlxsw_reg_write(mlxsw_sp->core, MLXSW_REG(cwtpm), cwtpm_cmd);
 | |
| }
 | |
| 
 | |
| static void
 | |
| mlxsw_sp_setup_tc_qdisc_red_clean_stats(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 					struct mlxsw_sp_qdisc *mlxsw_sp_qdisc)
 | |
| {
 | |
| 	int tclass_num = mlxsw_sp_qdisc->tclass_num;
 | |
| 	struct mlxsw_sp_qdisc_stats *stats_base;
 | |
| 	struct mlxsw_sp_port_xstats *xstats;
 | |
| 	struct red_stats *red_base;
 | |
| 
 | |
| 	xstats = &mlxsw_sp_port->periodic_hw_stats.xstats;
 | |
| 	stats_base = &mlxsw_sp_qdisc->stats_base;
 | |
| 	red_base = &mlxsw_sp_qdisc->xstats_base.red;
 | |
| 
 | |
| 	mlxsw_sp_qdisc_bstats_per_priority_get(xstats,
 | |
| 					       mlxsw_sp_qdisc->prio_bitmap,
 | |
| 					       &stats_base->tx_packets,
 | |
| 					       &stats_base->tx_bytes);
 | |
| 	red_base->prob_drop = xstats->wred_drop[tclass_num];
 | |
| 	red_base->pdrop = mlxsw_sp_xstats_tail_drop(xstats, tclass_num);
 | |
| 
 | |
| 	stats_base->overlimits = red_base->prob_drop + red_base->prob_mark;
 | |
| 	stats_base->drops = red_base->prob_drop + red_base->pdrop;
 | |
| 
 | |
| 	stats_base->backlog = 0;
 | |
| }
 | |
| 
 | |
| static int
 | |
| mlxsw_sp_qdisc_red_destroy(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 			   struct mlxsw_sp_qdisc *mlxsw_sp_qdisc)
 | |
| {
 | |
| 	return mlxsw_sp_tclass_congestion_disable(mlxsw_sp_port,
 | |
| 						  mlxsw_sp_qdisc->tclass_num);
 | |
| }
 | |
| 
 | |
| static int
 | |
| mlxsw_sp_qdisc_red_check_params(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 				void *params)
 | |
| {
 | |
| 	struct mlxsw_sp *mlxsw_sp = mlxsw_sp_port->mlxsw_sp;
 | |
| 	struct tc_red_qopt_offload_params *p = params;
 | |
| 
 | |
| 	if (p->min > p->max) {
 | |
| 		dev_err(mlxsw_sp->bus_info->dev,
 | |
| 			"spectrum: RED: min %u is bigger then max %u\n", p->min,
 | |
| 			p->max);
 | |
| 		return -EINVAL;
 | |
| 	}
 | |
| 	if (p->max > MLXSW_CORE_RES_GET(mlxsw_sp->core,
 | |
| 					GUARANTEED_SHARED_BUFFER)) {
 | |
| 		dev_err(mlxsw_sp->bus_info->dev,
 | |
| 			"spectrum: RED: max value %u is too big\n", p->max);
 | |
| 		return -EINVAL;
 | |
| 	}
 | |
| 	if (p->min == 0 || p->max == 0) {
 | |
| 		dev_err(mlxsw_sp->bus_info->dev,
 | |
| 			"spectrum: RED: 0 value is illegal for min and max\n");
 | |
| 		return -EINVAL;
 | |
| 	}
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int
 | |
| mlxsw_sp_qdisc_red_replace(struct mlxsw_sp_port *mlxsw_sp_port, u32 handle,
 | |
| 			   struct mlxsw_sp_qdisc *mlxsw_sp_qdisc,
 | |
| 			   void *params)
 | |
| {
 | |
| 	struct mlxsw_sp *mlxsw_sp = mlxsw_sp_port->mlxsw_sp;
 | |
| 	struct tc_red_qopt_offload_params *p = params;
 | |
| 	int tclass_num = mlxsw_sp_qdisc->tclass_num;
 | |
| 	u32 min, max;
 | |
| 	u64 prob;
 | |
| 
 | |
| 	/* calculate probability in percentage */
 | |
| 	prob = p->probability;
 | |
| 	prob *= 100;
 | |
| 	prob = DIV_ROUND_UP(prob, 1 << 16);
 | |
| 	prob = DIV_ROUND_UP(prob, 1 << 16);
 | |
| 	min = mlxsw_sp_bytes_cells(mlxsw_sp, p->min);
 | |
| 	max = mlxsw_sp_bytes_cells(mlxsw_sp, p->max);
 | |
| 	return mlxsw_sp_tclass_congestion_enable(mlxsw_sp_port, tclass_num,
 | |
| 						 min, max, prob,
 | |
| 						 !p->is_nodrop, p->is_ecn);
 | |
| }
 | |
| 
 | |
| static void
 | |
| mlxsw_sp_qdisc_leaf_unoffload(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 			      struct mlxsw_sp_qdisc *mlxsw_sp_qdisc,
 | |
| 			      struct gnet_stats_queue *qstats)
 | |
| {
 | |
| 	u64 backlog;
 | |
| 
 | |
| 	backlog = mlxsw_sp_cells_bytes(mlxsw_sp_port->mlxsw_sp,
 | |
| 				       mlxsw_sp_qdisc->stats_base.backlog);
 | |
| 	qstats->backlog -= backlog;
 | |
| 	mlxsw_sp_qdisc->stats_base.backlog = 0;
 | |
| }
 | |
| 
 | |
| static void
 | |
| mlxsw_sp_qdisc_red_unoffload(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 			     struct mlxsw_sp_qdisc *mlxsw_sp_qdisc,
 | |
| 			     void *params)
 | |
| {
 | |
| 	struct tc_red_qopt_offload_params *p = params;
 | |
| 
 | |
| 	mlxsw_sp_qdisc_leaf_unoffload(mlxsw_sp_port, mlxsw_sp_qdisc, p->qstats);
 | |
| }
 | |
| 
 | |
| static int
 | |
| mlxsw_sp_qdisc_get_red_xstats(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 			      struct mlxsw_sp_qdisc *mlxsw_sp_qdisc,
 | |
| 			      void *xstats_ptr)
 | |
| {
 | |
| 	struct red_stats *xstats_base = &mlxsw_sp_qdisc->xstats_base.red;
 | |
| 	int tclass_num = mlxsw_sp_qdisc->tclass_num;
 | |
| 	struct mlxsw_sp_port_xstats *xstats;
 | |
| 	struct red_stats *res = xstats_ptr;
 | |
| 	int early_drops, pdrops;
 | |
| 
 | |
| 	xstats = &mlxsw_sp_port->periodic_hw_stats.xstats;
 | |
| 
 | |
| 	early_drops = xstats->wred_drop[tclass_num] - xstats_base->prob_drop;
 | |
| 	pdrops = mlxsw_sp_xstats_tail_drop(xstats, tclass_num) -
 | |
| 		 xstats_base->pdrop;
 | |
| 
 | |
| 	res->pdrop += pdrops;
 | |
| 	res->prob_drop += early_drops;
 | |
| 
 | |
| 	xstats_base->pdrop += pdrops;
 | |
| 	xstats_base->prob_drop += early_drops;
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int
 | |
| mlxsw_sp_qdisc_get_red_stats(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 			     struct mlxsw_sp_qdisc *mlxsw_sp_qdisc,
 | |
| 			     struct tc_qopt_offload_stats *stats_ptr)
 | |
| {
 | |
| 	int tclass_num = mlxsw_sp_qdisc->tclass_num;
 | |
| 	struct mlxsw_sp_qdisc_stats *stats_base;
 | |
| 	struct mlxsw_sp_port_xstats *xstats;
 | |
| 	u64 overlimits;
 | |
| 
 | |
| 	xstats = &mlxsw_sp_port->periodic_hw_stats.xstats;
 | |
| 	stats_base = &mlxsw_sp_qdisc->stats_base;
 | |
| 
 | |
| 	mlxsw_sp_qdisc_get_tc_stats(mlxsw_sp_port, mlxsw_sp_qdisc, stats_ptr);
 | |
| 	overlimits = xstats->wred_drop[tclass_num] - stats_base->overlimits;
 | |
| 
 | |
| 	stats_ptr->qstats->overlimits += overlimits;
 | |
| 	stats_base->overlimits += overlimits;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static struct mlxsw_sp_qdisc *
 | |
| mlxsw_sp_qdisc_leaf_find_class(struct mlxsw_sp_qdisc *mlxsw_sp_qdisc,
 | |
| 			       u32 parent)
 | |
| {
 | |
| 	return NULL;
 | |
| }
 | |
| 
 | |
| #define MLXSW_SP_PORT_DEFAULT_TCLASS 0
 | |
| 
 | |
| static struct mlxsw_sp_qdisc_ops mlxsw_sp_qdisc_ops_red = {
 | |
| 	.type = MLXSW_SP_QDISC_RED,
 | |
| 	.check_params = mlxsw_sp_qdisc_red_check_params,
 | |
| 	.replace = mlxsw_sp_qdisc_red_replace,
 | |
| 	.unoffload = mlxsw_sp_qdisc_red_unoffload,
 | |
| 	.destroy = mlxsw_sp_qdisc_red_destroy,
 | |
| 	.get_stats = mlxsw_sp_qdisc_get_red_stats,
 | |
| 	.get_xstats = mlxsw_sp_qdisc_get_red_xstats,
 | |
| 	.clean_stats = mlxsw_sp_setup_tc_qdisc_red_clean_stats,
 | |
| 	.find_class = mlxsw_sp_qdisc_leaf_find_class,
 | |
| };
 | |
| 
 | |
| static int __mlxsw_sp_setup_tc_red(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 				   struct tc_red_qopt_offload *p)
 | |
| {
 | |
| 	struct mlxsw_sp_qdisc *mlxsw_sp_qdisc;
 | |
| 
 | |
| 	mlxsw_sp_qdisc = mlxsw_sp_qdisc_find(mlxsw_sp_port, p->parent, false);
 | |
| 	if (!mlxsw_sp_qdisc)
 | |
| 		return -EOPNOTSUPP;
 | |
| 
 | |
| 	if (p->command == TC_RED_REPLACE)
 | |
| 		return mlxsw_sp_qdisc_replace(mlxsw_sp_port, p->handle,
 | |
| 					      mlxsw_sp_qdisc,
 | |
| 					      &mlxsw_sp_qdisc_ops_red,
 | |
| 					      &p->set);
 | |
| 
 | |
| 	if (!mlxsw_sp_qdisc_compare(mlxsw_sp_qdisc, p->handle))
 | |
| 		return -EOPNOTSUPP;
 | |
| 
 | |
| 	switch (p->command) {
 | |
| 	case TC_RED_DESTROY:
 | |
| 		return mlxsw_sp_qdisc_destroy(mlxsw_sp_port, mlxsw_sp_qdisc);
 | |
| 	case TC_RED_XSTATS:
 | |
| 		return mlxsw_sp_qdisc_get_xstats(mlxsw_sp_port, mlxsw_sp_qdisc,
 | |
| 						 p->xstats);
 | |
| 	case TC_RED_STATS:
 | |
| 		return mlxsw_sp_qdisc_get_stats(mlxsw_sp_port, mlxsw_sp_qdisc,
 | |
| 						&p->stats);
 | |
| 	default:
 | |
| 		return -EOPNOTSUPP;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| int mlxsw_sp_setup_tc_red(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 			  struct tc_red_qopt_offload *p)
 | |
| {
 | |
| 	int err;
 | |
| 
 | |
| 	mutex_lock(&mlxsw_sp_port->qdisc->lock);
 | |
| 	err = __mlxsw_sp_setup_tc_red(mlxsw_sp_port, p);
 | |
| 	mutex_unlock(&mlxsw_sp_port->qdisc->lock);
 | |
| 
 | |
| 	return err;
 | |
| }
 | |
| 
 | |
| static void
 | |
| mlxsw_sp_setup_tc_qdisc_leaf_clean_stats(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 					 struct mlxsw_sp_qdisc *mlxsw_sp_qdisc)
 | |
| {
 | |
| 	u64 backlog_cells = 0;
 | |
| 	u64 tx_packets = 0;
 | |
| 	u64 tx_bytes = 0;
 | |
| 	u64 drops = 0;
 | |
| 
 | |
| 	mlxsw_sp_qdisc_collect_tc_stats(mlxsw_sp_port, mlxsw_sp_qdisc,
 | |
| 					&tx_bytes, &tx_packets,
 | |
| 					&drops, &backlog_cells);
 | |
| 
 | |
| 	mlxsw_sp_qdisc->stats_base.tx_packets = tx_packets;
 | |
| 	mlxsw_sp_qdisc->stats_base.tx_bytes = tx_bytes;
 | |
| 	mlxsw_sp_qdisc->stats_base.drops = drops;
 | |
| 	mlxsw_sp_qdisc->stats_base.backlog = 0;
 | |
| }
 | |
| 
 | |
| static int
 | |
| mlxsw_sp_qdisc_tbf_destroy(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 			   struct mlxsw_sp_qdisc *mlxsw_sp_qdisc)
 | |
| {
 | |
| 	return mlxsw_sp_port_ets_maxrate_set(mlxsw_sp_port,
 | |
| 					     MLXSW_REG_QEEC_HR_SUBGROUP,
 | |
| 					     mlxsw_sp_qdisc->tclass_num, 0,
 | |
| 					     MLXSW_REG_QEEC_MAS_DIS, 0);
 | |
| }
 | |
| 
 | |
| static int
 | |
| mlxsw_sp_qdisc_tbf_bs(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 		      u32 max_size, u8 *p_burst_size)
 | |
| {
 | |
| 	/* TBF burst size is configured in bytes. The ASIC burst size value is
 | |
| 	 * ((2 ^ bs) * 512 bits. Convert the TBF bytes to 512-bit units.
 | |
| 	 */
 | |
| 	u32 bs512 = max_size / 64;
 | |
| 	u8 bs = fls(bs512);
 | |
| 
 | |
| 	if (!bs)
 | |
| 		return -EINVAL;
 | |
| 	--bs;
 | |
| 
 | |
| 	/* Demand a power of two. */
 | |
| 	if ((1 << bs) != bs512)
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	if (bs < mlxsw_sp_port->mlxsw_sp->lowest_shaper_bs ||
 | |
| 	    bs > MLXSW_REG_QEEC_HIGHEST_SHAPER_BS)
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	*p_burst_size = bs;
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static u32
 | |
| mlxsw_sp_qdisc_tbf_max_size(u8 bs)
 | |
| {
 | |
| 	return (1U << bs) * 64;
 | |
| }
 | |
| 
 | |
| static u64
 | |
| mlxsw_sp_qdisc_tbf_rate_kbps(struct tc_tbf_qopt_offload_replace_params *p)
 | |
| {
 | |
| 	/* TBF interface is in bytes/s, whereas Spectrum ASIC is configured in
 | |
| 	 * Kbits/s.
 | |
| 	 */
 | |
| 	return div_u64(p->rate.rate_bytes_ps, 1000) * 8;
 | |
| }
 | |
| 
 | |
| static int
 | |
| mlxsw_sp_qdisc_tbf_check_params(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 				void *params)
 | |
| {
 | |
| 	struct tc_tbf_qopt_offload_replace_params *p = params;
 | |
| 	struct mlxsw_sp *mlxsw_sp = mlxsw_sp_port->mlxsw_sp;
 | |
| 	u64 rate_kbps = mlxsw_sp_qdisc_tbf_rate_kbps(p);
 | |
| 	u8 burst_size;
 | |
| 	int err;
 | |
| 
 | |
| 	if (rate_kbps >= MLXSW_REG_QEEC_MAS_DIS) {
 | |
| 		dev_err(mlxsw_sp_port->mlxsw_sp->bus_info->dev,
 | |
| 			"spectrum: TBF: rate of %lluKbps must be below %u\n",
 | |
| 			rate_kbps, MLXSW_REG_QEEC_MAS_DIS);
 | |
| 		return -EINVAL;
 | |
| 	}
 | |
| 
 | |
| 	err = mlxsw_sp_qdisc_tbf_bs(mlxsw_sp_port, p->max_size, &burst_size);
 | |
| 	if (err) {
 | |
| 		u8 highest_shaper_bs = MLXSW_REG_QEEC_HIGHEST_SHAPER_BS;
 | |
| 
 | |
| 		dev_err(mlxsw_sp->bus_info->dev,
 | |
| 			"spectrum: TBF: invalid burst size of %u, must be a power of two between %u and %u",
 | |
| 			p->max_size,
 | |
| 			mlxsw_sp_qdisc_tbf_max_size(mlxsw_sp->lowest_shaper_bs),
 | |
| 			mlxsw_sp_qdisc_tbf_max_size(highest_shaper_bs));
 | |
| 		return -EINVAL;
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int
 | |
| mlxsw_sp_qdisc_tbf_replace(struct mlxsw_sp_port *mlxsw_sp_port, u32 handle,
 | |
| 			   struct mlxsw_sp_qdisc *mlxsw_sp_qdisc,
 | |
| 			   void *params)
 | |
| {
 | |
| 	struct tc_tbf_qopt_offload_replace_params *p = params;
 | |
| 	u64 rate_kbps = mlxsw_sp_qdisc_tbf_rate_kbps(p);
 | |
| 	u8 burst_size;
 | |
| 	int err;
 | |
| 
 | |
| 	err = mlxsw_sp_qdisc_tbf_bs(mlxsw_sp_port, p->max_size, &burst_size);
 | |
| 	if (WARN_ON_ONCE(err))
 | |
| 		/* check_params above was supposed to reject this value. */
 | |
| 		return -EINVAL;
 | |
| 
 | |
| 	/* Configure subgroup shaper, so that both UC and MC traffic is subject
 | |
| 	 * to shaping. That is unlike RED, however UC queue lengths are going to
 | |
| 	 * be different than MC ones due to different pool and quota
 | |
| 	 * configurations, so the configuration is not applicable. For shaper on
 | |
| 	 * the other hand, subjecting the overall stream to the configured
 | |
| 	 * shaper makes sense. Also note that that is what we do for
 | |
| 	 * ieee_setmaxrate().
 | |
| 	 */
 | |
| 	return mlxsw_sp_port_ets_maxrate_set(mlxsw_sp_port,
 | |
| 					     MLXSW_REG_QEEC_HR_SUBGROUP,
 | |
| 					     mlxsw_sp_qdisc->tclass_num, 0,
 | |
| 					     rate_kbps, burst_size);
 | |
| }
 | |
| 
 | |
| static void
 | |
| mlxsw_sp_qdisc_tbf_unoffload(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 			     struct mlxsw_sp_qdisc *mlxsw_sp_qdisc,
 | |
| 			     void *params)
 | |
| {
 | |
| 	struct tc_tbf_qopt_offload_replace_params *p = params;
 | |
| 
 | |
| 	mlxsw_sp_qdisc_leaf_unoffload(mlxsw_sp_port, mlxsw_sp_qdisc, p->qstats);
 | |
| }
 | |
| 
 | |
| static int
 | |
| mlxsw_sp_qdisc_get_tbf_stats(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 			     struct mlxsw_sp_qdisc *mlxsw_sp_qdisc,
 | |
| 			     struct tc_qopt_offload_stats *stats_ptr)
 | |
| {
 | |
| 	mlxsw_sp_qdisc_get_tc_stats(mlxsw_sp_port, mlxsw_sp_qdisc,
 | |
| 				    stats_ptr);
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static struct mlxsw_sp_qdisc_ops mlxsw_sp_qdisc_ops_tbf = {
 | |
| 	.type = MLXSW_SP_QDISC_TBF,
 | |
| 	.check_params = mlxsw_sp_qdisc_tbf_check_params,
 | |
| 	.replace = mlxsw_sp_qdisc_tbf_replace,
 | |
| 	.unoffload = mlxsw_sp_qdisc_tbf_unoffload,
 | |
| 	.destroy = mlxsw_sp_qdisc_tbf_destroy,
 | |
| 	.get_stats = mlxsw_sp_qdisc_get_tbf_stats,
 | |
| 	.clean_stats = mlxsw_sp_setup_tc_qdisc_leaf_clean_stats,
 | |
| 	.find_class = mlxsw_sp_qdisc_leaf_find_class,
 | |
| };
 | |
| 
 | |
| static int __mlxsw_sp_setup_tc_tbf(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 				   struct tc_tbf_qopt_offload *p)
 | |
| {
 | |
| 	struct mlxsw_sp_qdisc *mlxsw_sp_qdisc;
 | |
| 
 | |
| 	mlxsw_sp_qdisc = mlxsw_sp_qdisc_find(mlxsw_sp_port, p->parent, false);
 | |
| 	if (!mlxsw_sp_qdisc)
 | |
| 		return -EOPNOTSUPP;
 | |
| 
 | |
| 	if (p->command == TC_TBF_REPLACE)
 | |
| 		return mlxsw_sp_qdisc_replace(mlxsw_sp_port, p->handle,
 | |
| 					      mlxsw_sp_qdisc,
 | |
| 					      &mlxsw_sp_qdisc_ops_tbf,
 | |
| 					      &p->replace_params);
 | |
| 
 | |
| 	if (!mlxsw_sp_qdisc_compare(mlxsw_sp_qdisc, p->handle))
 | |
| 		return -EOPNOTSUPP;
 | |
| 
 | |
| 	switch (p->command) {
 | |
| 	case TC_TBF_DESTROY:
 | |
| 		return mlxsw_sp_qdisc_destroy(mlxsw_sp_port, mlxsw_sp_qdisc);
 | |
| 	case TC_TBF_STATS:
 | |
| 		return mlxsw_sp_qdisc_get_stats(mlxsw_sp_port, mlxsw_sp_qdisc,
 | |
| 						&p->stats);
 | |
| 	default:
 | |
| 		return -EOPNOTSUPP;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| int mlxsw_sp_setup_tc_tbf(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 			  struct tc_tbf_qopt_offload *p)
 | |
| {
 | |
| 	int err;
 | |
| 
 | |
| 	mutex_lock(&mlxsw_sp_port->qdisc->lock);
 | |
| 	err = __mlxsw_sp_setup_tc_tbf(mlxsw_sp_port, p);
 | |
| 	mutex_unlock(&mlxsw_sp_port->qdisc->lock);
 | |
| 
 | |
| 	return err;
 | |
| }
 | |
| 
 | |
| static int
 | |
| mlxsw_sp_qdisc_fifo_check_params(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 				 void *params)
 | |
| {
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int
 | |
| mlxsw_sp_qdisc_fifo_replace(struct mlxsw_sp_port *mlxsw_sp_port, u32 handle,
 | |
| 			    struct mlxsw_sp_qdisc *mlxsw_sp_qdisc,
 | |
| 			    void *params)
 | |
| {
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int
 | |
| mlxsw_sp_qdisc_get_fifo_stats(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 			      struct mlxsw_sp_qdisc *mlxsw_sp_qdisc,
 | |
| 			      struct tc_qopt_offload_stats *stats_ptr)
 | |
| {
 | |
| 	mlxsw_sp_qdisc_get_tc_stats(mlxsw_sp_port, mlxsw_sp_qdisc,
 | |
| 				    stats_ptr);
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static struct mlxsw_sp_qdisc_ops mlxsw_sp_qdisc_ops_fifo = {
 | |
| 	.type = MLXSW_SP_QDISC_FIFO,
 | |
| 	.check_params = mlxsw_sp_qdisc_fifo_check_params,
 | |
| 	.replace = mlxsw_sp_qdisc_fifo_replace,
 | |
| 	.get_stats = mlxsw_sp_qdisc_get_fifo_stats,
 | |
| 	.clean_stats = mlxsw_sp_setup_tc_qdisc_leaf_clean_stats,
 | |
| };
 | |
| 
 | |
| static int __mlxsw_sp_setup_tc_fifo(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 				    struct tc_fifo_qopt_offload *p)
 | |
| {
 | |
| 	struct mlxsw_sp_qdisc_state *qdisc_state = mlxsw_sp_port->qdisc;
 | |
| 	struct mlxsw_sp_qdisc *mlxsw_sp_qdisc;
 | |
| 	unsigned int band;
 | |
| 	u32 parent_handle;
 | |
| 
 | |
| 	mlxsw_sp_qdisc = mlxsw_sp_qdisc_find(mlxsw_sp_port, p->parent, false);
 | |
| 	if (!mlxsw_sp_qdisc && p->handle == TC_H_UNSPEC) {
 | |
| 		parent_handle = TC_H_MAJ(p->parent);
 | |
| 		if (parent_handle != qdisc_state->future_handle) {
 | |
| 			/* This notifications is for a different Qdisc than
 | |
| 			 * previously. Wipe the future cache.
 | |
| 			 */
 | |
| 			memset(qdisc_state->future_fifos, 0,
 | |
| 			       sizeof(qdisc_state->future_fifos));
 | |
| 			qdisc_state->future_handle = parent_handle;
 | |
| 		}
 | |
| 
 | |
| 		band = TC_H_MIN(p->parent) - 1;
 | |
| 		if (band < IEEE_8021QAZ_MAX_TCS) {
 | |
| 			if (p->command == TC_FIFO_REPLACE)
 | |
| 				qdisc_state->future_fifos[band] = true;
 | |
| 			else if (p->command == TC_FIFO_DESTROY)
 | |
| 				qdisc_state->future_fifos[band] = false;
 | |
| 		}
 | |
| 	}
 | |
| 	if (!mlxsw_sp_qdisc)
 | |
| 		return -EOPNOTSUPP;
 | |
| 
 | |
| 	if (p->command == TC_FIFO_REPLACE) {
 | |
| 		return mlxsw_sp_qdisc_replace(mlxsw_sp_port, p->handle,
 | |
| 					      mlxsw_sp_qdisc,
 | |
| 					      &mlxsw_sp_qdisc_ops_fifo, NULL);
 | |
| 	}
 | |
| 
 | |
| 	if (!mlxsw_sp_qdisc_compare(mlxsw_sp_qdisc, p->handle))
 | |
| 		return -EOPNOTSUPP;
 | |
| 
 | |
| 	switch (p->command) {
 | |
| 	case TC_FIFO_DESTROY:
 | |
| 		return mlxsw_sp_qdisc_destroy(mlxsw_sp_port, mlxsw_sp_qdisc);
 | |
| 	case TC_FIFO_STATS:
 | |
| 		return mlxsw_sp_qdisc_get_stats(mlxsw_sp_port, mlxsw_sp_qdisc,
 | |
| 						&p->stats);
 | |
| 	case TC_FIFO_REPLACE: /* Handled above. */
 | |
| 		break;
 | |
| 	}
 | |
| 
 | |
| 	return -EOPNOTSUPP;
 | |
| }
 | |
| 
 | |
| int mlxsw_sp_setup_tc_fifo(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 			   struct tc_fifo_qopt_offload *p)
 | |
| {
 | |
| 	int err;
 | |
| 
 | |
| 	mutex_lock(&mlxsw_sp_port->qdisc->lock);
 | |
| 	err = __mlxsw_sp_setup_tc_fifo(mlxsw_sp_port, p);
 | |
| 	mutex_unlock(&mlxsw_sp_port->qdisc->lock);
 | |
| 
 | |
| 	return err;
 | |
| }
 | |
| 
 | |
| static int __mlxsw_sp_qdisc_ets_destroy(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 					struct mlxsw_sp_qdisc *mlxsw_sp_qdisc)
 | |
| {
 | |
| 	int i;
 | |
| 
 | |
| 	for (i = 0; i < mlxsw_sp_qdisc->num_classes; i++) {
 | |
| 		mlxsw_sp_port_prio_tc_set(mlxsw_sp_port, i,
 | |
| 					  MLXSW_SP_PORT_DEFAULT_TCLASS);
 | |
| 		mlxsw_sp_port_ets_set(mlxsw_sp_port,
 | |
| 				      MLXSW_REG_QEEC_HR_SUBGROUP,
 | |
| 				      i, 0, false, 0);
 | |
| 		mlxsw_sp_qdisc_destroy(mlxsw_sp_port,
 | |
| 				       &mlxsw_sp_qdisc->qdiscs[i]);
 | |
| 		mlxsw_sp_qdisc->qdiscs[i].prio_bitmap = 0;
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int
 | |
| mlxsw_sp_qdisc_prio_destroy(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 			    struct mlxsw_sp_qdisc *mlxsw_sp_qdisc)
 | |
| {
 | |
| 	return __mlxsw_sp_qdisc_ets_destroy(mlxsw_sp_port, mlxsw_sp_qdisc);
 | |
| }
 | |
| 
 | |
| static int
 | |
| __mlxsw_sp_qdisc_ets_check_params(unsigned int nbands)
 | |
| {
 | |
| 	if (nbands > IEEE_8021QAZ_MAX_TCS)
 | |
| 		return -EOPNOTSUPP;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int
 | |
| mlxsw_sp_qdisc_prio_check_params(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 				 void *params)
 | |
| {
 | |
| 	struct tc_prio_qopt_offload_params *p = params;
 | |
| 
 | |
| 	return __mlxsw_sp_qdisc_ets_check_params(p->bands);
 | |
| }
 | |
| 
 | |
| static int
 | |
| __mlxsw_sp_qdisc_ets_replace(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 			     struct mlxsw_sp_qdisc *mlxsw_sp_qdisc,
 | |
| 			     u32 handle, unsigned int nbands,
 | |
| 			     const unsigned int *quanta,
 | |
| 			     const unsigned int *weights,
 | |
| 			     const u8 *priomap)
 | |
| {
 | |
| 	struct mlxsw_sp_qdisc_state *qdisc_state = mlxsw_sp_port->qdisc;
 | |
| 	struct mlxsw_sp_qdisc *child_qdisc;
 | |
| 	int tclass, i, band, backlog;
 | |
| 	u8 old_priomap;
 | |
| 	int err;
 | |
| 
 | |
| 	for (band = 0; band < nbands; band++) {
 | |
| 		tclass = MLXSW_SP_PRIO_BAND_TO_TCLASS(band);
 | |
| 		child_qdisc = &mlxsw_sp_qdisc->qdiscs[band];
 | |
| 		old_priomap = child_qdisc->prio_bitmap;
 | |
| 		child_qdisc->prio_bitmap = 0;
 | |
| 
 | |
| 		err = mlxsw_sp_port_ets_set(mlxsw_sp_port,
 | |
| 					    MLXSW_REG_QEEC_HR_SUBGROUP,
 | |
| 					    tclass, 0, !!quanta[band],
 | |
| 					    weights[band]);
 | |
| 		if (err)
 | |
| 			return err;
 | |
| 
 | |
| 		for (i = 0; i < IEEE_8021QAZ_MAX_TCS; i++) {
 | |
| 			if (priomap[i] == band) {
 | |
| 				child_qdisc->prio_bitmap |= BIT(i);
 | |
| 				if (BIT(i) & old_priomap)
 | |
| 					continue;
 | |
| 				err = mlxsw_sp_port_prio_tc_set(mlxsw_sp_port,
 | |
| 								i, tclass);
 | |
| 				if (err)
 | |
| 					return err;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		child_qdisc->tclass_num = tclass;
 | |
| 
 | |
| 		if (old_priomap != child_qdisc->prio_bitmap &&
 | |
| 		    child_qdisc->ops && child_qdisc->ops->clean_stats) {
 | |
| 			backlog = child_qdisc->stats_base.backlog;
 | |
| 			child_qdisc->ops->clean_stats(mlxsw_sp_port,
 | |
| 						      child_qdisc);
 | |
| 			child_qdisc->stats_base.backlog = backlog;
 | |
| 		}
 | |
| 
 | |
| 		if (handle == qdisc_state->future_handle &&
 | |
| 		    qdisc_state->future_fifos[band]) {
 | |
| 			err = mlxsw_sp_qdisc_replace(mlxsw_sp_port, TC_H_UNSPEC,
 | |
| 						     child_qdisc,
 | |
| 						     &mlxsw_sp_qdisc_ops_fifo,
 | |
| 						     NULL);
 | |
| 			if (err)
 | |
| 				return err;
 | |
| 		}
 | |
| 	}
 | |
| 	for (; band < IEEE_8021QAZ_MAX_TCS; band++) {
 | |
| 		tclass = MLXSW_SP_PRIO_BAND_TO_TCLASS(band);
 | |
| 		child_qdisc = &mlxsw_sp_qdisc->qdiscs[band];
 | |
| 		child_qdisc->prio_bitmap = 0;
 | |
| 		mlxsw_sp_qdisc_destroy(mlxsw_sp_port, child_qdisc);
 | |
| 		mlxsw_sp_port_ets_set(mlxsw_sp_port,
 | |
| 				      MLXSW_REG_QEEC_HR_SUBGROUP,
 | |
| 				      tclass, 0, false, 0);
 | |
| 	}
 | |
| 
 | |
| 	qdisc_state->future_handle = TC_H_UNSPEC;
 | |
| 	memset(qdisc_state->future_fifos, 0, sizeof(qdisc_state->future_fifos));
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int
 | |
| mlxsw_sp_qdisc_prio_replace(struct mlxsw_sp_port *mlxsw_sp_port, u32 handle,
 | |
| 			    struct mlxsw_sp_qdisc *mlxsw_sp_qdisc,
 | |
| 			    void *params)
 | |
| {
 | |
| 	struct tc_prio_qopt_offload_params *p = params;
 | |
| 	unsigned int zeroes[TCQ_ETS_MAX_BANDS] = {0};
 | |
| 
 | |
| 	return __mlxsw_sp_qdisc_ets_replace(mlxsw_sp_port, mlxsw_sp_qdisc,
 | |
| 					    handle, p->bands, zeroes,
 | |
| 					    zeroes, p->priomap);
 | |
| }
 | |
| 
 | |
| static void
 | |
| __mlxsw_sp_qdisc_ets_unoffload(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 			       struct mlxsw_sp_qdisc *mlxsw_sp_qdisc,
 | |
| 			       struct gnet_stats_queue *qstats)
 | |
| {
 | |
| 	u64 backlog;
 | |
| 
 | |
| 	backlog = mlxsw_sp_cells_bytes(mlxsw_sp_port->mlxsw_sp,
 | |
| 				       mlxsw_sp_qdisc->stats_base.backlog);
 | |
| 	qstats->backlog -= backlog;
 | |
| }
 | |
| 
 | |
| static void
 | |
| mlxsw_sp_qdisc_prio_unoffload(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 			      struct mlxsw_sp_qdisc *mlxsw_sp_qdisc,
 | |
| 			      void *params)
 | |
| {
 | |
| 	struct tc_prio_qopt_offload_params *p = params;
 | |
| 
 | |
| 	__mlxsw_sp_qdisc_ets_unoffload(mlxsw_sp_port, mlxsw_sp_qdisc,
 | |
| 				       p->qstats);
 | |
| }
 | |
| 
 | |
| static int
 | |
| mlxsw_sp_qdisc_get_prio_stats(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 			      struct mlxsw_sp_qdisc *mlxsw_sp_qdisc,
 | |
| 			      struct tc_qopt_offload_stats *stats_ptr)
 | |
| {
 | |
| 	struct mlxsw_sp_qdisc *tc_qdisc;
 | |
| 	u64 tx_packets = 0;
 | |
| 	u64 tx_bytes = 0;
 | |
| 	u64 backlog = 0;
 | |
| 	u64 drops = 0;
 | |
| 	int i;
 | |
| 
 | |
| 	for (i = 0; i < mlxsw_sp_qdisc->num_classes; i++) {
 | |
| 		tc_qdisc = &mlxsw_sp_qdisc->qdiscs[i];
 | |
| 		mlxsw_sp_qdisc_collect_tc_stats(mlxsw_sp_port, tc_qdisc,
 | |
| 						&tx_bytes, &tx_packets,
 | |
| 						&drops, &backlog);
 | |
| 	}
 | |
| 
 | |
| 	mlxsw_sp_qdisc_update_stats(mlxsw_sp_port->mlxsw_sp, mlxsw_sp_qdisc,
 | |
| 				    tx_bytes, tx_packets, drops, backlog,
 | |
| 				    stats_ptr);
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static void
 | |
| mlxsw_sp_setup_tc_qdisc_prio_clean_stats(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 					 struct mlxsw_sp_qdisc *mlxsw_sp_qdisc)
 | |
| {
 | |
| 	struct mlxsw_sp_qdisc_stats *stats_base;
 | |
| 	struct mlxsw_sp_port_xstats *xstats;
 | |
| 	struct rtnl_link_stats64 *stats;
 | |
| 	int i;
 | |
| 
 | |
| 	xstats = &mlxsw_sp_port->periodic_hw_stats.xstats;
 | |
| 	stats = &mlxsw_sp_port->periodic_hw_stats.stats;
 | |
| 	stats_base = &mlxsw_sp_qdisc->stats_base;
 | |
| 
 | |
| 	stats_base->tx_packets = stats->tx_packets;
 | |
| 	stats_base->tx_bytes = stats->tx_bytes;
 | |
| 
 | |
| 	stats_base->drops = 0;
 | |
| 	for (i = 0; i < IEEE_8021QAZ_MAX_TCS; i++) {
 | |
| 		stats_base->drops += mlxsw_sp_xstats_tail_drop(xstats, i);
 | |
| 		stats_base->drops += xstats->wred_drop[i];
 | |
| 	}
 | |
| 
 | |
| 	mlxsw_sp_qdisc->stats_base.backlog = 0;
 | |
| }
 | |
| 
 | |
| static struct mlxsw_sp_qdisc *
 | |
| mlxsw_sp_qdisc_prio_find_class(struct mlxsw_sp_qdisc *mlxsw_sp_qdisc,
 | |
| 			       u32 parent)
 | |
| {
 | |
| 	int child_index = TC_H_MIN(parent);
 | |
| 	int band = child_index - 1;
 | |
| 
 | |
| 	if (band < 0 || band >= mlxsw_sp_qdisc->num_classes)
 | |
| 		return NULL;
 | |
| 	return &mlxsw_sp_qdisc->qdiscs[band];
 | |
| }
 | |
| 
 | |
| static struct mlxsw_sp_qdisc_ops mlxsw_sp_qdisc_ops_prio = {
 | |
| 	.type = MLXSW_SP_QDISC_PRIO,
 | |
| 	.check_params = mlxsw_sp_qdisc_prio_check_params,
 | |
| 	.replace = mlxsw_sp_qdisc_prio_replace,
 | |
| 	.unoffload = mlxsw_sp_qdisc_prio_unoffload,
 | |
| 	.destroy = mlxsw_sp_qdisc_prio_destroy,
 | |
| 	.get_stats = mlxsw_sp_qdisc_get_prio_stats,
 | |
| 	.clean_stats = mlxsw_sp_setup_tc_qdisc_prio_clean_stats,
 | |
| 	.find_class = mlxsw_sp_qdisc_prio_find_class,
 | |
| 	.num_classes = IEEE_8021QAZ_MAX_TCS,
 | |
| };
 | |
| 
 | |
| static int
 | |
| mlxsw_sp_qdisc_ets_check_params(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 				void *params)
 | |
| {
 | |
| 	struct tc_ets_qopt_offload_replace_params *p = params;
 | |
| 
 | |
| 	return __mlxsw_sp_qdisc_ets_check_params(p->bands);
 | |
| }
 | |
| 
 | |
| static int
 | |
| mlxsw_sp_qdisc_ets_replace(struct mlxsw_sp_port *mlxsw_sp_port, u32 handle,
 | |
| 			   struct mlxsw_sp_qdisc *mlxsw_sp_qdisc,
 | |
| 			   void *params)
 | |
| {
 | |
| 	struct tc_ets_qopt_offload_replace_params *p = params;
 | |
| 
 | |
| 	return __mlxsw_sp_qdisc_ets_replace(mlxsw_sp_port, mlxsw_sp_qdisc,
 | |
| 					    handle, p->bands, p->quanta,
 | |
| 					    p->weights, p->priomap);
 | |
| }
 | |
| 
 | |
| static void
 | |
| mlxsw_sp_qdisc_ets_unoffload(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 			     struct mlxsw_sp_qdisc *mlxsw_sp_qdisc,
 | |
| 			     void *params)
 | |
| {
 | |
| 	struct tc_ets_qopt_offload_replace_params *p = params;
 | |
| 
 | |
| 	__mlxsw_sp_qdisc_ets_unoffload(mlxsw_sp_port, mlxsw_sp_qdisc,
 | |
| 				       p->qstats);
 | |
| }
 | |
| 
 | |
| static int
 | |
| mlxsw_sp_qdisc_ets_destroy(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 			   struct mlxsw_sp_qdisc *mlxsw_sp_qdisc)
 | |
| {
 | |
| 	return __mlxsw_sp_qdisc_ets_destroy(mlxsw_sp_port, mlxsw_sp_qdisc);
 | |
| }
 | |
| 
 | |
| static struct mlxsw_sp_qdisc_ops mlxsw_sp_qdisc_ops_ets = {
 | |
| 	.type = MLXSW_SP_QDISC_ETS,
 | |
| 	.check_params = mlxsw_sp_qdisc_ets_check_params,
 | |
| 	.replace = mlxsw_sp_qdisc_ets_replace,
 | |
| 	.unoffload = mlxsw_sp_qdisc_ets_unoffload,
 | |
| 	.destroy = mlxsw_sp_qdisc_ets_destroy,
 | |
| 	.get_stats = mlxsw_sp_qdisc_get_prio_stats,
 | |
| 	.clean_stats = mlxsw_sp_setup_tc_qdisc_prio_clean_stats,
 | |
| 	.find_class = mlxsw_sp_qdisc_prio_find_class,
 | |
| 	.num_classes = IEEE_8021QAZ_MAX_TCS,
 | |
| };
 | |
| 
 | |
| /* Linux allows linking of Qdiscs to arbitrary classes (so long as the resulting
 | |
|  * graph is free of cycles). These operations do not change the parent handle
 | |
|  * though, which means it can be incomplete (if there is more than one class
 | |
|  * where the Qdisc in question is grafted) or outright wrong (if the Qdisc was
 | |
|  * linked to a different class and then removed from the original class).
 | |
|  *
 | |
|  * E.g. consider this sequence of operations:
 | |
|  *
 | |
|  *  # tc qdisc add dev swp1 root handle 1: prio
 | |
|  *  # tc qdisc add dev swp1 parent 1:3 handle 13: red limit 1000000 avpkt 10000
 | |
|  *  RED: set bandwidth to 10Mbit
 | |
|  *  # tc qdisc link dev swp1 handle 13: parent 1:2
 | |
|  *
 | |
|  * At this point, both 1:2 and 1:3 have the same RED Qdisc instance as their
 | |
|  * child. But RED will still only claim that 1:3 is its parent. If it's removed
 | |
|  * from that band, its only parent will be 1:2, but it will continue to claim
 | |
|  * that it is in fact 1:3.
 | |
|  *
 | |
|  * The notification for child Qdisc replace (e.g. TC_RED_REPLACE) comes before
 | |
|  * the notification for parent graft (e.g. TC_PRIO_GRAFT). We take the replace
 | |
|  * notification to offload the child Qdisc, based on its parent handle, and use
 | |
|  * the graft operation to validate that the class where the child is actually
 | |
|  * grafted corresponds to the parent handle. If the two don't match, we
 | |
|  * unoffload the child.
 | |
|  */
 | |
| static int
 | |
| __mlxsw_sp_qdisc_ets_graft(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 			   struct mlxsw_sp_qdisc *mlxsw_sp_qdisc,
 | |
| 			   u8 band, u32 child_handle)
 | |
| {
 | |
| 	struct mlxsw_sp_qdisc *old_qdisc;
 | |
| 
 | |
| 	if (band < mlxsw_sp_qdisc->num_classes &&
 | |
| 	    mlxsw_sp_qdisc->qdiscs[band].handle == child_handle)
 | |
| 		return 0;
 | |
| 
 | |
| 	if (!child_handle) {
 | |
| 		/* This is an invisible FIFO replacing the original Qdisc.
 | |
| 		 * Ignore it--the original Qdisc's destroy will follow.
 | |
| 		 */
 | |
| 		return 0;
 | |
| 	}
 | |
| 
 | |
| 	/* See if the grafted qdisc is already offloaded on any tclass. If so,
 | |
| 	 * unoffload it.
 | |
| 	 */
 | |
| 	old_qdisc = mlxsw_sp_qdisc_find_by_handle(mlxsw_sp_port,
 | |
| 						  child_handle);
 | |
| 	if (old_qdisc)
 | |
| 		mlxsw_sp_qdisc_destroy(mlxsw_sp_port, old_qdisc);
 | |
| 
 | |
| 	mlxsw_sp_qdisc = mlxsw_sp_qdisc->ops->find_class(mlxsw_sp_qdisc, band);
 | |
| 	if (!WARN_ON(!mlxsw_sp_qdisc))
 | |
| 		mlxsw_sp_qdisc_destroy(mlxsw_sp_port, mlxsw_sp_qdisc);
 | |
| 
 | |
| 	return -EOPNOTSUPP;
 | |
| }
 | |
| 
 | |
| static int
 | |
| mlxsw_sp_qdisc_prio_graft(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 			  struct mlxsw_sp_qdisc *mlxsw_sp_qdisc,
 | |
| 			  struct tc_prio_qopt_offload_graft_params *p)
 | |
| {
 | |
| 	return __mlxsw_sp_qdisc_ets_graft(mlxsw_sp_port, mlxsw_sp_qdisc,
 | |
| 					  p->band, p->child_handle);
 | |
| }
 | |
| 
 | |
| static int __mlxsw_sp_setup_tc_prio(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 				    struct tc_prio_qopt_offload *p)
 | |
| {
 | |
| 	struct mlxsw_sp_qdisc *mlxsw_sp_qdisc;
 | |
| 
 | |
| 	mlxsw_sp_qdisc = mlxsw_sp_qdisc_find(mlxsw_sp_port, p->parent, true);
 | |
| 	if (!mlxsw_sp_qdisc)
 | |
| 		return -EOPNOTSUPP;
 | |
| 
 | |
| 	if (p->command == TC_PRIO_REPLACE)
 | |
| 		return mlxsw_sp_qdisc_replace(mlxsw_sp_port, p->handle,
 | |
| 					      mlxsw_sp_qdisc,
 | |
| 					      &mlxsw_sp_qdisc_ops_prio,
 | |
| 					      &p->replace_params);
 | |
| 
 | |
| 	if (!mlxsw_sp_qdisc_compare(mlxsw_sp_qdisc, p->handle))
 | |
| 		return -EOPNOTSUPP;
 | |
| 
 | |
| 	switch (p->command) {
 | |
| 	case TC_PRIO_DESTROY:
 | |
| 		return mlxsw_sp_qdisc_destroy(mlxsw_sp_port, mlxsw_sp_qdisc);
 | |
| 	case TC_PRIO_STATS:
 | |
| 		return mlxsw_sp_qdisc_get_stats(mlxsw_sp_port, mlxsw_sp_qdisc,
 | |
| 						&p->stats);
 | |
| 	case TC_PRIO_GRAFT:
 | |
| 		return mlxsw_sp_qdisc_prio_graft(mlxsw_sp_port, mlxsw_sp_qdisc,
 | |
| 						 &p->graft_params);
 | |
| 	default:
 | |
| 		return -EOPNOTSUPP;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| int mlxsw_sp_setup_tc_prio(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 			   struct tc_prio_qopt_offload *p)
 | |
| {
 | |
| 	int err;
 | |
| 
 | |
| 	mutex_lock(&mlxsw_sp_port->qdisc->lock);
 | |
| 	err = __mlxsw_sp_setup_tc_prio(mlxsw_sp_port, p);
 | |
| 	mutex_unlock(&mlxsw_sp_port->qdisc->lock);
 | |
| 
 | |
| 	return err;
 | |
| }
 | |
| 
 | |
| static int __mlxsw_sp_setup_tc_ets(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 				   struct tc_ets_qopt_offload *p)
 | |
| {
 | |
| 	struct mlxsw_sp_qdisc *mlxsw_sp_qdisc;
 | |
| 
 | |
| 	mlxsw_sp_qdisc = mlxsw_sp_qdisc_find(mlxsw_sp_port, p->parent, true);
 | |
| 	if (!mlxsw_sp_qdisc)
 | |
| 		return -EOPNOTSUPP;
 | |
| 
 | |
| 	if (p->command == TC_ETS_REPLACE)
 | |
| 		return mlxsw_sp_qdisc_replace(mlxsw_sp_port, p->handle,
 | |
| 					      mlxsw_sp_qdisc,
 | |
| 					      &mlxsw_sp_qdisc_ops_ets,
 | |
| 					      &p->replace_params);
 | |
| 
 | |
| 	if (!mlxsw_sp_qdisc_compare(mlxsw_sp_qdisc, p->handle))
 | |
| 		return -EOPNOTSUPP;
 | |
| 
 | |
| 	switch (p->command) {
 | |
| 	case TC_ETS_DESTROY:
 | |
| 		return mlxsw_sp_qdisc_destroy(mlxsw_sp_port, mlxsw_sp_qdisc);
 | |
| 	case TC_ETS_STATS:
 | |
| 		return mlxsw_sp_qdisc_get_stats(mlxsw_sp_port, mlxsw_sp_qdisc,
 | |
| 						&p->stats);
 | |
| 	case TC_ETS_GRAFT:
 | |
| 		return __mlxsw_sp_qdisc_ets_graft(mlxsw_sp_port, mlxsw_sp_qdisc,
 | |
| 						  p->graft_params.band,
 | |
| 						  p->graft_params.child_handle);
 | |
| 	default:
 | |
| 		return -EOPNOTSUPP;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| int mlxsw_sp_setup_tc_ets(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 			  struct tc_ets_qopt_offload *p)
 | |
| {
 | |
| 	int err;
 | |
| 
 | |
| 	mutex_lock(&mlxsw_sp_port->qdisc->lock);
 | |
| 	err = __mlxsw_sp_setup_tc_ets(mlxsw_sp_port, p);
 | |
| 	mutex_unlock(&mlxsw_sp_port->qdisc->lock);
 | |
| 
 | |
| 	return err;
 | |
| }
 | |
| 
 | |
| struct mlxsw_sp_qevent_block {
 | |
| 	struct list_head binding_list;
 | |
| 	struct list_head mall_entry_list;
 | |
| 	struct mlxsw_sp *mlxsw_sp;
 | |
| };
 | |
| 
 | |
| struct mlxsw_sp_qevent_binding {
 | |
| 	struct list_head list;
 | |
| 	struct mlxsw_sp_port *mlxsw_sp_port;
 | |
| 	u32 handle;
 | |
| 	int tclass_num;
 | |
| 	enum mlxsw_sp_span_trigger span_trigger;
 | |
| };
 | |
| 
 | |
| static LIST_HEAD(mlxsw_sp_qevent_block_cb_list);
 | |
| 
 | |
| static int mlxsw_sp_qevent_span_configure(struct mlxsw_sp *mlxsw_sp,
 | |
| 					  struct mlxsw_sp_mall_entry *mall_entry,
 | |
| 					  struct mlxsw_sp_qevent_binding *qevent_binding,
 | |
| 					  const struct mlxsw_sp_span_agent_parms *agent_parms,
 | |
| 					  int *p_span_id)
 | |
| {
 | |
| 	struct mlxsw_sp_port *mlxsw_sp_port = qevent_binding->mlxsw_sp_port;
 | |
| 	struct mlxsw_sp_span_trigger_parms trigger_parms = {};
 | |
| 	int span_id;
 | |
| 	int err;
 | |
| 
 | |
| 	err = mlxsw_sp_span_agent_get(mlxsw_sp, &span_id, agent_parms);
 | |
| 	if (err)
 | |
| 		return err;
 | |
| 
 | |
| 	err = mlxsw_sp_span_analyzed_port_get(mlxsw_sp_port, true);
 | |
| 	if (err)
 | |
| 		goto err_analyzed_port_get;
 | |
| 
 | |
| 	trigger_parms.span_id = span_id;
 | |
| 	trigger_parms.probability_rate = 1;
 | |
| 	err = mlxsw_sp_span_agent_bind(mlxsw_sp, qevent_binding->span_trigger, mlxsw_sp_port,
 | |
| 				       &trigger_parms);
 | |
| 	if (err)
 | |
| 		goto err_agent_bind;
 | |
| 
 | |
| 	err = mlxsw_sp_span_trigger_enable(mlxsw_sp_port, qevent_binding->span_trigger,
 | |
| 					   qevent_binding->tclass_num);
 | |
| 	if (err)
 | |
| 		goto err_trigger_enable;
 | |
| 
 | |
| 	*p_span_id = span_id;
 | |
| 	return 0;
 | |
| 
 | |
| err_trigger_enable:
 | |
| 	mlxsw_sp_span_agent_unbind(mlxsw_sp, qevent_binding->span_trigger, mlxsw_sp_port,
 | |
| 				   &trigger_parms);
 | |
| err_agent_bind:
 | |
| 	mlxsw_sp_span_analyzed_port_put(mlxsw_sp_port, true);
 | |
| err_analyzed_port_get:
 | |
| 	mlxsw_sp_span_agent_put(mlxsw_sp, span_id);
 | |
| 	return err;
 | |
| }
 | |
| 
 | |
| static void mlxsw_sp_qevent_span_deconfigure(struct mlxsw_sp *mlxsw_sp,
 | |
| 					     struct mlxsw_sp_qevent_binding *qevent_binding,
 | |
| 					     int span_id)
 | |
| {
 | |
| 	struct mlxsw_sp_port *mlxsw_sp_port = qevent_binding->mlxsw_sp_port;
 | |
| 	struct mlxsw_sp_span_trigger_parms trigger_parms = {
 | |
| 		.span_id = span_id,
 | |
| 	};
 | |
| 
 | |
| 	mlxsw_sp_span_trigger_disable(mlxsw_sp_port, qevent_binding->span_trigger,
 | |
| 				      qevent_binding->tclass_num);
 | |
| 	mlxsw_sp_span_agent_unbind(mlxsw_sp, qevent_binding->span_trigger, mlxsw_sp_port,
 | |
| 				   &trigger_parms);
 | |
| 	mlxsw_sp_span_analyzed_port_put(mlxsw_sp_port, true);
 | |
| 	mlxsw_sp_span_agent_put(mlxsw_sp, span_id);
 | |
| }
 | |
| 
 | |
| static int mlxsw_sp_qevent_mirror_configure(struct mlxsw_sp *mlxsw_sp,
 | |
| 					    struct mlxsw_sp_mall_entry *mall_entry,
 | |
| 					    struct mlxsw_sp_qevent_binding *qevent_binding)
 | |
| {
 | |
| 	struct mlxsw_sp_span_agent_parms agent_parms = {
 | |
| 		.to_dev = mall_entry->mirror.to_dev,
 | |
| 	};
 | |
| 
 | |
| 	return mlxsw_sp_qevent_span_configure(mlxsw_sp, mall_entry, qevent_binding,
 | |
| 					      &agent_parms, &mall_entry->mirror.span_id);
 | |
| }
 | |
| 
 | |
| static void mlxsw_sp_qevent_mirror_deconfigure(struct mlxsw_sp *mlxsw_sp,
 | |
| 					       struct mlxsw_sp_mall_entry *mall_entry,
 | |
| 					       struct mlxsw_sp_qevent_binding *qevent_binding)
 | |
| {
 | |
| 	mlxsw_sp_qevent_span_deconfigure(mlxsw_sp, qevent_binding, mall_entry->mirror.span_id);
 | |
| }
 | |
| 
 | |
| static int mlxsw_sp_qevent_trap_configure(struct mlxsw_sp *mlxsw_sp,
 | |
| 					  struct mlxsw_sp_mall_entry *mall_entry,
 | |
| 					  struct mlxsw_sp_qevent_binding *qevent_binding)
 | |
| {
 | |
| 	struct mlxsw_sp_span_agent_parms agent_parms = {
 | |
| 		.session_id = MLXSW_SP_SPAN_SESSION_ID_BUFFER,
 | |
| 	};
 | |
| 	int err;
 | |
| 
 | |
| 	err = mlxsw_sp_trap_group_policer_hw_id_get(mlxsw_sp,
 | |
| 						    DEVLINK_TRAP_GROUP_GENERIC_ID_BUFFER_DROPS,
 | |
| 						    &agent_parms.policer_enable,
 | |
| 						    &agent_parms.policer_id);
 | |
| 	if (err)
 | |
| 		return err;
 | |
| 
 | |
| 	return mlxsw_sp_qevent_span_configure(mlxsw_sp, mall_entry, qevent_binding,
 | |
| 					      &agent_parms, &mall_entry->trap.span_id);
 | |
| }
 | |
| 
 | |
| static void mlxsw_sp_qevent_trap_deconfigure(struct mlxsw_sp *mlxsw_sp,
 | |
| 					     struct mlxsw_sp_mall_entry *mall_entry,
 | |
| 					     struct mlxsw_sp_qevent_binding *qevent_binding)
 | |
| {
 | |
| 	mlxsw_sp_qevent_span_deconfigure(mlxsw_sp, qevent_binding, mall_entry->trap.span_id);
 | |
| }
 | |
| 
 | |
| static int mlxsw_sp_qevent_entry_configure(struct mlxsw_sp *mlxsw_sp,
 | |
| 					   struct mlxsw_sp_mall_entry *mall_entry,
 | |
| 					   struct mlxsw_sp_qevent_binding *qevent_binding)
 | |
| {
 | |
| 	switch (mall_entry->type) {
 | |
| 	case MLXSW_SP_MALL_ACTION_TYPE_MIRROR:
 | |
| 		return mlxsw_sp_qevent_mirror_configure(mlxsw_sp, mall_entry, qevent_binding);
 | |
| 	case MLXSW_SP_MALL_ACTION_TYPE_TRAP:
 | |
| 		return mlxsw_sp_qevent_trap_configure(mlxsw_sp, mall_entry, qevent_binding);
 | |
| 	default:
 | |
| 		/* This should have been validated away. */
 | |
| 		WARN_ON(1);
 | |
| 		return -EOPNOTSUPP;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static void mlxsw_sp_qevent_entry_deconfigure(struct mlxsw_sp *mlxsw_sp,
 | |
| 					      struct mlxsw_sp_mall_entry *mall_entry,
 | |
| 					      struct mlxsw_sp_qevent_binding *qevent_binding)
 | |
| {
 | |
| 	switch (mall_entry->type) {
 | |
| 	case MLXSW_SP_MALL_ACTION_TYPE_MIRROR:
 | |
| 		return mlxsw_sp_qevent_mirror_deconfigure(mlxsw_sp, mall_entry, qevent_binding);
 | |
| 	case MLXSW_SP_MALL_ACTION_TYPE_TRAP:
 | |
| 		return mlxsw_sp_qevent_trap_deconfigure(mlxsw_sp, mall_entry, qevent_binding);
 | |
| 	default:
 | |
| 		WARN_ON(1);
 | |
| 		return;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static int mlxsw_sp_qevent_binding_configure(struct mlxsw_sp_qevent_block *qevent_block,
 | |
| 					     struct mlxsw_sp_qevent_binding *qevent_binding)
 | |
| {
 | |
| 	struct mlxsw_sp_mall_entry *mall_entry;
 | |
| 	int err;
 | |
| 
 | |
| 	list_for_each_entry(mall_entry, &qevent_block->mall_entry_list, list) {
 | |
| 		err = mlxsw_sp_qevent_entry_configure(qevent_block->mlxsw_sp, mall_entry,
 | |
| 						      qevent_binding);
 | |
| 		if (err)
 | |
| 			goto err_entry_configure;
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| 
 | |
| err_entry_configure:
 | |
| 	list_for_each_entry_continue_reverse(mall_entry, &qevent_block->mall_entry_list, list)
 | |
| 		mlxsw_sp_qevent_entry_deconfigure(qevent_block->mlxsw_sp, mall_entry,
 | |
| 						  qevent_binding);
 | |
| 	return err;
 | |
| }
 | |
| 
 | |
| static void mlxsw_sp_qevent_binding_deconfigure(struct mlxsw_sp_qevent_block *qevent_block,
 | |
| 						struct mlxsw_sp_qevent_binding *qevent_binding)
 | |
| {
 | |
| 	struct mlxsw_sp_mall_entry *mall_entry;
 | |
| 
 | |
| 	list_for_each_entry(mall_entry, &qevent_block->mall_entry_list, list)
 | |
| 		mlxsw_sp_qevent_entry_deconfigure(qevent_block->mlxsw_sp, mall_entry,
 | |
| 						  qevent_binding);
 | |
| }
 | |
| 
 | |
| static int mlxsw_sp_qevent_block_configure(struct mlxsw_sp_qevent_block *qevent_block)
 | |
| {
 | |
| 	struct mlxsw_sp_qevent_binding *qevent_binding;
 | |
| 	int err;
 | |
| 
 | |
| 	list_for_each_entry(qevent_binding, &qevent_block->binding_list, list) {
 | |
| 		err = mlxsw_sp_qevent_binding_configure(qevent_block, qevent_binding);
 | |
| 		if (err)
 | |
| 			goto err_binding_configure;
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| 
 | |
| err_binding_configure:
 | |
| 	list_for_each_entry_continue_reverse(qevent_binding, &qevent_block->binding_list, list)
 | |
| 		mlxsw_sp_qevent_binding_deconfigure(qevent_block, qevent_binding);
 | |
| 	return err;
 | |
| }
 | |
| 
 | |
| static void mlxsw_sp_qevent_block_deconfigure(struct mlxsw_sp_qevent_block *qevent_block)
 | |
| {
 | |
| 	struct mlxsw_sp_qevent_binding *qevent_binding;
 | |
| 
 | |
| 	list_for_each_entry(qevent_binding, &qevent_block->binding_list, list)
 | |
| 		mlxsw_sp_qevent_binding_deconfigure(qevent_block, qevent_binding);
 | |
| }
 | |
| 
 | |
| static struct mlxsw_sp_mall_entry *
 | |
| mlxsw_sp_qevent_mall_entry_find(struct mlxsw_sp_qevent_block *block, unsigned long cookie)
 | |
| {
 | |
| 	struct mlxsw_sp_mall_entry *mall_entry;
 | |
| 
 | |
| 	list_for_each_entry(mall_entry, &block->mall_entry_list, list)
 | |
| 		if (mall_entry->cookie == cookie)
 | |
| 			return mall_entry;
 | |
| 
 | |
| 	return NULL;
 | |
| }
 | |
| 
 | |
| static int mlxsw_sp_qevent_mall_replace(struct mlxsw_sp *mlxsw_sp,
 | |
| 					struct mlxsw_sp_qevent_block *qevent_block,
 | |
| 					struct tc_cls_matchall_offload *f)
 | |
| {
 | |
| 	struct mlxsw_sp_mall_entry *mall_entry;
 | |
| 	struct flow_action_entry *act;
 | |
| 	int err;
 | |
| 
 | |
| 	/* It should not currently be possible to replace a matchall rule. So
 | |
| 	 * this must be a new rule.
 | |
| 	 */
 | |
| 	if (!list_empty(&qevent_block->mall_entry_list)) {
 | |
| 		NL_SET_ERR_MSG(f->common.extack, "At most one filter supported");
 | |
| 		return -EOPNOTSUPP;
 | |
| 	}
 | |
| 	if (f->rule->action.num_entries != 1) {
 | |
| 		NL_SET_ERR_MSG(f->common.extack, "Only singular actions supported");
 | |
| 		return -EOPNOTSUPP;
 | |
| 	}
 | |
| 	if (f->common.chain_index) {
 | |
| 		NL_SET_ERR_MSG(f->common.extack, "Only chain 0 is supported");
 | |
| 		return -EOPNOTSUPP;
 | |
| 	}
 | |
| 	if (f->common.protocol != htons(ETH_P_ALL)) {
 | |
| 		NL_SET_ERR_MSG(f->common.extack, "Protocol matching not supported");
 | |
| 		return -EOPNOTSUPP;
 | |
| 	}
 | |
| 
 | |
| 	act = &f->rule->action.entries[0];
 | |
| 	if (!(act->hw_stats & FLOW_ACTION_HW_STATS_DISABLED)) {
 | |
| 		NL_SET_ERR_MSG(f->common.extack, "HW counters not supported on qevents");
 | |
| 		return -EOPNOTSUPP;
 | |
| 	}
 | |
| 
 | |
| 	mall_entry = kzalloc(sizeof(*mall_entry), GFP_KERNEL);
 | |
| 	if (!mall_entry)
 | |
| 		return -ENOMEM;
 | |
| 	mall_entry->cookie = f->cookie;
 | |
| 
 | |
| 	if (act->id == FLOW_ACTION_MIRRED) {
 | |
| 		mall_entry->type = MLXSW_SP_MALL_ACTION_TYPE_MIRROR;
 | |
| 		mall_entry->mirror.to_dev = act->dev;
 | |
| 	} else if (act->id == FLOW_ACTION_TRAP) {
 | |
| 		mall_entry->type = MLXSW_SP_MALL_ACTION_TYPE_TRAP;
 | |
| 	} else {
 | |
| 		NL_SET_ERR_MSG(f->common.extack, "Unsupported action");
 | |
| 		err = -EOPNOTSUPP;
 | |
| 		goto err_unsupported_action;
 | |
| 	}
 | |
| 
 | |
| 	list_add_tail(&mall_entry->list, &qevent_block->mall_entry_list);
 | |
| 
 | |
| 	err = mlxsw_sp_qevent_block_configure(qevent_block);
 | |
| 	if (err)
 | |
| 		goto err_block_configure;
 | |
| 
 | |
| 	return 0;
 | |
| 
 | |
| err_block_configure:
 | |
| 	list_del(&mall_entry->list);
 | |
| err_unsupported_action:
 | |
| 	kfree(mall_entry);
 | |
| 	return err;
 | |
| }
 | |
| 
 | |
| static void mlxsw_sp_qevent_mall_destroy(struct mlxsw_sp_qevent_block *qevent_block,
 | |
| 					 struct tc_cls_matchall_offload *f)
 | |
| {
 | |
| 	struct mlxsw_sp_mall_entry *mall_entry;
 | |
| 
 | |
| 	mall_entry = mlxsw_sp_qevent_mall_entry_find(qevent_block, f->cookie);
 | |
| 	if (!mall_entry)
 | |
| 		return;
 | |
| 
 | |
| 	mlxsw_sp_qevent_block_deconfigure(qevent_block);
 | |
| 
 | |
| 	list_del(&mall_entry->list);
 | |
| 	kfree(mall_entry);
 | |
| }
 | |
| 
 | |
| static int mlxsw_sp_qevent_block_mall_cb(struct mlxsw_sp_qevent_block *qevent_block,
 | |
| 					 struct tc_cls_matchall_offload *f)
 | |
| {
 | |
| 	struct mlxsw_sp *mlxsw_sp = qevent_block->mlxsw_sp;
 | |
| 
 | |
| 	switch (f->command) {
 | |
| 	case TC_CLSMATCHALL_REPLACE:
 | |
| 		return mlxsw_sp_qevent_mall_replace(mlxsw_sp, qevent_block, f);
 | |
| 	case TC_CLSMATCHALL_DESTROY:
 | |
| 		mlxsw_sp_qevent_mall_destroy(qevent_block, f);
 | |
| 		return 0;
 | |
| 	default:
 | |
| 		return -EOPNOTSUPP;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static int mlxsw_sp_qevent_block_cb(enum tc_setup_type type, void *type_data, void *cb_priv)
 | |
| {
 | |
| 	struct mlxsw_sp_qevent_block *qevent_block = cb_priv;
 | |
| 
 | |
| 	switch (type) {
 | |
| 	case TC_SETUP_CLSMATCHALL:
 | |
| 		return mlxsw_sp_qevent_block_mall_cb(qevent_block, type_data);
 | |
| 	default:
 | |
| 		return -EOPNOTSUPP;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static struct mlxsw_sp_qevent_block *mlxsw_sp_qevent_block_create(struct mlxsw_sp *mlxsw_sp,
 | |
| 								  struct net *net)
 | |
| {
 | |
| 	struct mlxsw_sp_qevent_block *qevent_block;
 | |
| 
 | |
| 	qevent_block = kzalloc(sizeof(*qevent_block), GFP_KERNEL);
 | |
| 	if (!qevent_block)
 | |
| 		return NULL;
 | |
| 
 | |
| 	INIT_LIST_HEAD(&qevent_block->binding_list);
 | |
| 	INIT_LIST_HEAD(&qevent_block->mall_entry_list);
 | |
| 	qevent_block->mlxsw_sp = mlxsw_sp;
 | |
| 	return qevent_block;
 | |
| }
 | |
| 
 | |
| static void
 | |
| mlxsw_sp_qevent_block_destroy(struct mlxsw_sp_qevent_block *qevent_block)
 | |
| {
 | |
| 	WARN_ON(!list_empty(&qevent_block->binding_list));
 | |
| 	WARN_ON(!list_empty(&qevent_block->mall_entry_list));
 | |
| 	kfree(qevent_block);
 | |
| }
 | |
| 
 | |
| static void mlxsw_sp_qevent_block_release(void *cb_priv)
 | |
| {
 | |
| 	struct mlxsw_sp_qevent_block *qevent_block = cb_priv;
 | |
| 
 | |
| 	mlxsw_sp_qevent_block_destroy(qevent_block);
 | |
| }
 | |
| 
 | |
| static struct mlxsw_sp_qevent_binding *
 | |
| mlxsw_sp_qevent_binding_create(struct mlxsw_sp_port *mlxsw_sp_port, u32 handle, int tclass_num,
 | |
| 			       enum mlxsw_sp_span_trigger span_trigger)
 | |
| {
 | |
| 	struct mlxsw_sp_qevent_binding *binding;
 | |
| 
 | |
| 	binding = kzalloc(sizeof(*binding), GFP_KERNEL);
 | |
| 	if (!binding)
 | |
| 		return ERR_PTR(-ENOMEM);
 | |
| 
 | |
| 	binding->mlxsw_sp_port = mlxsw_sp_port;
 | |
| 	binding->handle = handle;
 | |
| 	binding->tclass_num = tclass_num;
 | |
| 	binding->span_trigger = span_trigger;
 | |
| 	return binding;
 | |
| }
 | |
| 
 | |
| static void
 | |
| mlxsw_sp_qevent_binding_destroy(struct mlxsw_sp_qevent_binding *binding)
 | |
| {
 | |
| 	kfree(binding);
 | |
| }
 | |
| 
 | |
| static struct mlxsw_sp_qevent_binding *
 | |
| mlxsw_sp_qevent_binding_lookup(struct mlxsw_sp_qevent_block *block,
 | |
| 			       struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 			       u32 handle,
 | |
| 			       enum mlxsw_sp_span_trigger span_trigger)
 | |
| {
 | |
| 	struct mlxsw_sp_qevent_binding *qevent_binding;
 | |
| 
 | |
| 	list_for_each_entry(qevent_binding, &block->binding_list, list)
 | |
| 		if (qevent_binding->mlxsw_sp_port == mlxsw_sp_port &&
 | |
| 		    qevent_binding->handle == handle &&
 | |
| 		    qevent_binding->span_trigger == span_trigger)
 | |
| 			return qevent_binding;
 | |
| 	return NULL;
 | |
| }
 | |
| 
 | |
| static int mlxsw_sp_setup_tc_block_qevent_bind(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 					       struct flow_block_offload *f,
 | |
| 					       enum mlxsw_sp_span_trigger span_trigger)
 | |
| {
 | |
| 	struct mlxsw_sp *mlxsw_sp = mlxsw_sp_port->mlxsw_sp;
 | |
| 	struct mlxsw_sp_qevent_binding *qevent_binding;
 | |
| 	struct mlxsw_sp_qevent_block *qevent_block;
 | |
| 	struct flow_block_cb *block_cb;
 | |
| 	struct mlxsw_sp_qdisc *qdisc;
 | |
| 	bool register_block = false;
 | |
| 	int err;
 | |
| 
 | |
| 	block_cb = flow_block_cb_lookup(f->block, mlxsw_sp_qevent_block_cb, mlxsw_sp);
 | |
| 	if (!block_cb) {
 | |
| 		qevent_block = mlxsw_sp_qevent_block_create(mlxsw_sp, f->net);
 | |
| 		if (!qevent_block)
 | |
| 			return -ENOMEM;
 | |
| 		block_cb = flow_block_cb_alloc(mlxsw_sp_qevent_block_cb, mlxsw_sp, qevent_block,
 | |
| 					       mlxsw_sp_qevent_block_release);
 | |
| 		if (IS_ERR(block_cb)) {
 | |
| 			mlxsw_sp_qevent_block_destroy(qevent_block);
 | |
| 			return PTR_ERR(block_cb);
 | |
| 		}
 | |
| 		register_block = true;
 | |
| 	} else {
 | |
| 		qevent_block = flow_block_cb_priv(block_cb);
 | |
| 	}
 | |
| 	flow_block_cb_incref(block_cb);
 | |
| 
 | |
| 	qdisc = mlxsw_sp_qdisc_find_by_handle(mlxsw_sp_port, f->sch->handle);
 | |
| 	if (!qdisc) {
 | |
| 		NL_SET_ERR_MSG(f->extack, "Qdisc not offloaded");
 | |
| 		err = -ENOENT;
 | |
| 		goto err_find_qdisc;
 | |
| 	}
 | |
| 
 | |
| 	if (WARN_ON(mlxsw_sp_qevent_binding_lookup(qevent_block, mlxsw_sp_port, f->sch->handle,
 | |
| 						   span_trigger))) {
 | |
| 		err = -EEXIST;
 | |
| 		goto err_binding_exists;
 | |
| 	}
 | |
| 
 | |
| 	qevent_binding = mlxsw_sp_qevent_binding_create(mlxsw_sp_port, f->sch->handle,
 | |
| 							qdisc->tclass_num, span_trigger);
 | |
| 	if (IS_ERR(qevent_binding)) {
 | |
| 		err = PTR_ERR(qevent_binding);
 | |
| 		goto err_binding_create;
 | |
| 	}
 | |
| 
 | |
| 	err = mlxsw_sp_qevent_binding_configure(qevent_block, qevent_binding);
 | |
| 	if (err)
 | |
| 		goto err_binding_configure;
 | |
| 
 | |
| 	list_add(&qevent_binding->list, &qevent_block->binding_list);
 | |
| 
 | |
| 	if (register_block) {
 | |
| 		flow_block_cb_add(block_cb, f);
 | |
| 		list_add_tail(&block_cb->driver_list, &mlxsw_sp_qevent_block_cb_list);
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| 
 | |
| err_binding_configure:
 | |
| 	mlxsw_sp_qevent_binding_destroy(qevent_binding);
 | |
| err_binding_create:
 | |
| err_binding_exists:
 | |
| err_find_qdisc:
 | |
| 	if (!flow_block_cb_decref(block_cb))
 | |
| 		flow_block_cb_free(block_cb);
 | |
| 	return err;
 | |
| }
 | |
| 
 | |
| static void mlxsw_sp_setup_tc_block_qevent_unbind(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 						  struct flow_block_offload *f,
 | |
| 						  enum mlxsw_sp_span_trigger span_trigger)
 | |
| {
 | |
| 	struct mlxsw_sp *mlxsw_sp = mlxsw_sp_port->mlxsw_sp;
 | |
| 	struct mlxsw_sp_qevent_binding *qevent_binding;
 | |
| 	struct mlxsw_sp_qevent_block *qevent_block;
 | |
| 	struct flow_block_cb *block_cb;
 | |
| 
 | |
| 	block_cb = flow_block_cb_lookup(f->block, mlxsw_sp_qevent_block_cb, mlxsw_sp);
 | |
| 	if (!block_cb)
 | |
| 		return;
 | |
| 	qevent_block = flow_block_cb_priv(block_cb);
 | |
| 
 | |
| 	qevent_binding = mlxsw_sp_qevent_binding_lookup(qevent_block, mlxsw_sp_port, f->sch->handle,
 | |
| 							span_trigger);
 | |
| 	if (!qevent_binding)
 | |
| 		return;
 | |
| 
 | |
| 	list_del(&qevent_binding->list);
 | |
| 	mlxsw_sp_qevent_binding_deconfigure(qevent_block, qevent_binding);
 | |
| 	mlxsw_sp_qevent_binding_destroy(qevent_binding);
 | |
| 
 | |
| 	if (!flow_block_cb_decref(block_cb)) {
 | |
| 		flow_block_cb_remove(block_cb, f);
 | |
| 		list_del(&block_cb->driver_list);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static int mlxsw_sp_setup_tc_block_qevent(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 					  struct flow_block_offload *f,
 | |
| 					  enum mlxsw_sp_span_trigger span_trigger)
 | |
| {
 | |
| 	f->driver_block_list = &mlxsw_sp_qevent_block_cb_list;
 | |
| 
 | |
| 	switch (f->command) {
 | |
| 	case FLOW_BLOCK_BIND:
 | |
| 		return mlxsw_sp_setup_tc_block_qevent_bind(mlxsw_sp_port, f, span_trigger);
 | |
| 	case FLOW_BLOCK_UNBIND:
 | |
| 		mlxsw_sp_setup_tc_block_qevent_unbind(mlxsw_sp_port, f, span_trigger);
 | |
| 		return 0;
 | |
| 	default:
 | |
| 		return -EOPNOTSUPP;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| int mlxsw_sp_setup_tc_block_qevent_early_drop(struct mlxsw_sp_port *mlxsw_sp_port,
 | |
| 					      struct flow_block_offload *f)
 | |
| {
 | |
| 	return mlxsw_sp_setup_tc_block_qevent(mlxsw_sp_port, f, MLXSW_SP_SPAN_TRIGGER_EARLY_DROP);
 | |
| }
 | |
| 
 | |
| int mlxsw_sp_tc_qdisc_init(struct mlxsw_sp_port *mlxsw_sp_port)
 | |
| {
 | |
| 	struct mlxsw_sp_qdisc_state *qdisc_state;
 | |
| 
 | |
| 	qdisc_state = kzalloc(sizeof(*qdisc_state), GFP_KERNEL);
 | |
| 	if (!qdisc_state)
 | |
| 		return -ENOMEM;
 | |
| 
 | |
| 	mutex_init(&qdisc_state->lock);
 | |
| 	qdisc_state->root_qdisc.prio_bitmap = 0xff;
 | |
| 	qdisc_state->root_qdisc.tclass_num = MLXSW_SP_PORT_DEFAULT_TCLASS;
 | |
| 	mlxsw_sp_port->qdisc = qdisc_state;
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| void mlxsw_sp_tc_qdisc_fini(struct mlxsw_sp_port *mlxsw_sp_port)
 | |
| {
 | |
| 	mutex_destroy(&mlxsw_sp_port->qdisc->lock);
 | |
| 	kfree(mlxsw_sp_port->qdisc);
 | |
| }
 |