xref: /linux/drivers/net/ethernet/microchip/sparx5/sparx5_qos.c (revision 9a87ffc99ec8eb8d35eed7c4f816d75f5cc9662e)
1ab0e493eSDaniel Machon // SPDX-License-Identifier: GPL-2.0+
2ab0e493eSDaniel Machon /* Microchip Sparx5 Switch driver
3ab0e493eSDaniel Machon  *
4ab0e493eSDaniel Machon  * Copyright (c) 2022 Microchip Technology Inc. and its subsidiaries.
5ab0e493eSDaniel Machon  */
6ab0e493eSDaniel Machon 
7e02a5ac6SDaniel Machon #include <net/pkt_cls.h>
8e02a5ac6SDaniel Machon 
9ab0e493eSDaniel Machon #include "sparx5_main.h"
10ab0e493eSDaniel Machon #include "sparx5_qos.h"
11ab0e493eSDaniel Machon 
129e02131eSDaniel Machon /* Calculate new base_time based on cycle_time.
139e02131eSDaniel Machon  *
149e02131eSDaniel Machon  * The hardware requires a base_time that is always in the future.
159e02131eSDaniel Machon  * We define threshold_time as current_time + (2 * cycle_time).
169e02131eSDaniel Machon  * If base_time is below threshold_time this function recalculates it to be in
179e02131eSDaniel Machon  * the interval:
189e02131eSDaniel Machon  * threshold_time <= base_time < (threshold_time + cycle_time)
199e02131eSDaniel Machon  *
209e02131eSDaniel Machon  * A very simple algorithm could be like this:
219e02131eSDaniel Machon  * new_base_time = org_base_time + N * cycle_time
229e02131eSDaniel Machon  * using the lowest N so (new_base_time >= threshold_time
239e02131eSDaniel Machon  */
sparx5_new_base_time(struct sparx5 * sparx5,const u32 cycle_time,const ktime_t org_base_time,ktime_t * new_base_time)249e02131eSDaniel Machon void sparx5_new_base_time(struct sparx5 *sparx5, const u32 cycle_time,
259e02131eSDaniel Machon 			  const ktime_t org_base_time, ktime_t *new_base_time)
269e02131eSDaniel Machon {
279e02131eSDaniel Machon 	ktime_t current_time, threshold_time, new_time;
289e02131eSDaniel Machon 	struct timespec64 ts;
299e02131eSDaniel Machon 	u64 nr_of_cycles_p2;
309e02131eSDaniel Machon 	u64 nr_of_cycles;
319e02131eSDaniel Machon 	u64 diff_time;
329e02131eSDaniel Machon 
339e02131eSDaniel Machon 	new_time = org_base_time;
349e02131eSDaniel Machon 
359e02131eSDaniel Machon 	sparx5_ptp_gettime64(&sparx5->phc[SPARX5_PHC_PORT].info, &ts);
369e02131eSDaniel Machon 	current_time = timespec64_to_ktime(ts);
379e02131eSDaniel Machon 	threshold_time = current_time + (2 * cycle_time);
389e02131eSDaniel Machon 	diff_time = threshold_time - new_time;
399e02131eSDaniel Machon 	nr_of_cycles = div_u64(diff_time, cycle_time);
409e02131eSDaniel Machon 	nr_of_cycles_p2 = 1; /* Use 2^0 as start value */
419e02131eSDaniel Machon 
429e02131eSDaniel Machon 	if (new_time >= threshold_time) {
439e02131eSDaniel Machon 		*new_base_time = new_time;
449e02131eSDaniel Machon 		return;
459e02131eSDaniel Machon 	}
469e02131eSDaniel Machon 
479e02131eSDaniel Machon 	/* Calculate the smallest power of 2 (nr_of_cycles_p2)
489e02131eSDaniel Machon 	 * that is larger than nr_of_cycles.
499e02131eSDaniel Machon 	 */
509e02131eSDaniel Machon 	while (nr_of_cycles_p2 < nr_of_cycles)
519e02131eSDaniel Machon 		nr_of_cycles_p2 <<= 1; /* Next (higher) power of 2 */
529e02131eSDaniel Machon 
539e02131eSDaniel Machon 	/* Add as big chunks (power of 2 * cycle_time)
549e02131eSDaniel Machon 	 * as possible for each power of 2
559e02131eSDaniel Machon 	 */
569e02131eSDaniel Machon 	while (nr_of_cycles_p2) {
579e02131eSDaniel Machon 		if (new_time < threshold_time) {
589e02131eSDaniel Machon 			new_time += cycle_time * nr_of_cycles_p2;
599e02131eSDaniel Machon 			while (new_time < threshold_time)
609e02131eSDaniel Machon 				new_time += cycle_time * nr_of_cycles_p2;
619e02131eSDaniel Machon 			new_time -= cycle_time * nr_of_cycles_p2;
629e02131eSDaniel Machon 		}
639e02131eSDaniel Machon 		nr_of_cycles_p2 >>= 1; /* Next (lower) power of 2 */
649e02131eSDaniel Machon 	}
659e02131eSDaniel Machon 	new_time += cycle_time;
669e02131eSDaniel Machon 	*new_base_time = new_time;
679e02131eSDaniel Machon }
689e02131eSDaniel Machon 
69e02a5ac6SDaniel Machon /* Max rates for leak groups */
70e02a5ac6SDaniel Machon static const u32 spx5_hsch_max_group_rate[SPX5_HSCH_LEAK_GRP_CNT] = {
71e02a5ac6SDaniel Machon 	1048568, /*  1.049 Gbps */
72e02a5ac6SDaniel Machon 	2621420, /*  2.621 Gbps */
73e02a5ac6SDaniel Machon 	10485680, /* 10.486 Gbps */
74e02a5ac6SDaniel Machon 	26214200 /* 26.214 Gbps */
75e02a5ac6SDaniel Machon };
76e02a5ac6SDaniel Machon 
77e02a5ac6SDaniel Machon static struct sparx5_layer layers[SPX5_HSCH_LAYER_CNT];
78e02a5ac6SDaniel Machon 
sparx5_lg_get_leak_time(struct sparx5 * sparx5,u32 layer,u32 group)79e02a5ac6SDaniel Machon static u32 sparx5_lg_get_leak_time(struct sparx5 *sparx5, u32 layer, u32 group)
80e02a5ac6SDaniel Machon {
81e02a5ac6SDaniel Machon 	u32 value;
82e02a5ac6SDaniel Machon 
83e02a5ac6SDaniel Machon 	value = spx5_rd(sparx5, HSCH_HSCH_TIMER_CFG(layer, group));
84e02a5ac6SDaniel Machon 	return HSCH_HSCH_TIMER_CFG_LEAK_TIME_GET(value);
85e02a5ac6SDaniel Machon }
86e02a5ac6SDaniel Machon 
sparx5_lg_set_leak_time(struct sparx5 * sparx5,u32 layer,u32 group,u32 leak_time)87e02a5ac6SDaniel Machon static void sparx5_lg_set_leak_time(struct sparx5 *sparx5, u32 layer, u32 group,
88e02a5ac6SDaniel Machon 				    u32 leak_time)
89e02a5ac6SDaniel Machon {
90e02a5ac6SDaniel Machon 	spx5_wr(HSCH_HSCH_TIMER_CFG_LEAK_TIME_SET(leak_time), sparx5,
91e02a5ac6SDaniel Machon 		HSCH_HSCH_TIMER_CFG(layer, group));
92e02a5ac6SDaniel Machon }
93e02a5ac6SDaniel Machon 
sparx5_lg_get_first(struct sparx5 * sparx5,u32 layer,u32 group)94e02a5ac6SDaniel Machon static u32 sparx5_lg_get_first(struct sparx5 *sparx5, u32 layer, u32 group)
95e02a5ac6SDaniel Machon {
96e02a5ac6SDaniel Machon 	u32 value;
97e02a5ac6SDaniel Machon 
98e02a5ac6SDaniel Machon 	value = spx5_rd(sparx5, HSCH_HSCH_LEAK_CFG(layer, group));
99e02a5ac6SDaniel Machon 	return HSCH_HSCH_LEAK_CFG_LEAK_FIRST_GET(value);
100e02a5ac6SDaniel Machon }
101e02a5ac6SDaniel Machon 
sparx5_lg_get_next(struct sparx5 * sparx5,u32 layer,u32 group,u32 idx)102e02a5ac6SDaniel Machon static u32 sparx5_lg_get_next(struct sparx5 *sparx5, u32 layer, u32 group,
103e02a5ac6SDaniel Machon 			      u32 idx)
104e02a5ac6SDaniel Machon 
105e02a5ac6SDaniel Machon {
106e02a5ac6SDaniel Machon 	u32 value;
107e02a5ac6SDaniel Machon 
108e02a5ac6SDaniel Machon 	value = spx5_rd(sparx5, HSCH_SE_CONNECT(idx));
109e02a5ac6SDaniel Machon 	return HSCH_SE_CONNECT_SE_LEAK_LINK_GET(value);
110e02a5ac6SDaniel Machon }
111e02a5ac6SDaniel Machon 
sparx5_lg_get_last(struct sparx5 * sparx5,u32 layer,u32 group)112e02a5ac6SDaniel Machon static u32 sparx5_lg_get_last(struct sparx5 *sparx5, u32 layer, u32 group)
113e02a5ac6SDaniel Machon {
114e02a5ac6SDaniel Machon 	u32 itr, next;
115e02a5ac6SDaniel Machon 
116e02a5ac6SDaniel Machon 	itr = sparx5_lg_get_first(sparx5, layer, group);
117e02a5ac6SDaniel Machon 
118e02a5ac6SDaniel Machon 	for (;;) {
119e02a5ac6SDaniel Machon 		next = sparx5_lg_get_next(sparx5, layer, group, itr);
120e02a5ac6SDaniel Machon 		if (itr == next)
121e02a5ac6SDaniel Machon 			return itr;
122e02a5ac6SDaniel Machon 
123e02a5ac6SDaniel Machon 		itr = next;
124e02a5ac6SDaniel Machon 	}
125e02a5ac6SDaniel Machon }
126e02a5ac6SDaniel Machon 
sparx5_lg_is_last(struct sparx5 * sparx5,u32 layer,u32 group,u32 idx)127e02a5ac6SDaniel Machon static bool sparx5_lg_is_last(struct sparx5 *sparx5, u32 layer, u32 group,
128e02a5ac6SDaniel Machon 			      u32 idx)
129e02a5ac6SDaniel Machon {
130e02a5ac6SDaniel Machon 	return idx == sparx5_lg_get_next(sparx5, layer, group, idx);
131e02a5ac6SDaniel Machon }
132e02a5ac6SDaniel Machon 
sparx5_lg_is_first(struct sparx5 * sparx5,u32 layer,u32 group,u32 idx)133e02a5ac6SDaniel Machon static bool sparx5_lg_is_first(struct sparx5 *sparx5, u32 layer, u32 group,
134e02a5ac6SDaniel Machon 			       u32 idx)
135e02a5ac6SDaniel Machon {
136e02a5ac6SDaniel Machon 	return idx == sparx5_lg_get_first(sparx5, layer, group);
137e02a5ac6SDaniel Machon }
138e02a5ac6SDaniel Machon 
sparx5_lg_is_empty(struct sparx5 * sparx5,u32 layer,u32 group)139e02a5ac6SDaniel Machon static bool sparx5_lg_is_empty(struct sparx5 *sparx5, u32 layer, u32 group)
140e02a5ac6SDaniel Machon {
141e02a5ac6SDaniel Machon 	return sparx5_lg_get_leak_time(sparx5, layer, group) == 0;
142e02a5ac6SDaniel Machon }
143e02a5ac6SDaniel Machon 
sparx5_lg_is_singular(struct sparx5 * sparx5,u32 layer,u32 group)144e02a5ac6SDaniel Machon static bool sparx5_lg_is_singular(struct sparx5 *sparx5, u32 layer, u32 group)
145e02a5ac6SDaniel Machon {
146e02a5ac6SDaniel Machon 	if (sparx5_lg_is_empty(sparx5, layer, group))
147e02a5ac6SDaniel Machon 		return false;
148e02a5ac6SDaniel Machon 
149e02a5ac6SDaniel Machon 	return sparx5_lg_get_first(sparx5, layer, group) ==
150e02a5ac6SDaniel Machon 	       sparx5_lg_get_last(sparx5, layer, group);
151e02a5ac6SDaniel Machon }
152e02a5ac6SDaniel Machon 
sparx5_lg_enable(struct sparx5 * sparx5,u32 layer,u32 group,u32 leak_time)153e02a5ac6SDaniel Machon static void sparx5_lg_enable(struct sparx5 *sparx5, u32 layer, u32 group,
154e02a5ac6SDaniel Machon 			     u32 leak_time)
155e02a5ac6SDaniel Machon {
156e02a5ac6SDaniel Machon 	sparx5_lg_set_leak_time(sparx5, layer, group, leak_time);
157e02a5ac6SDaniel Machon }
158e02a5ac6SDaniel Machon 
sparx5_lg_disable(struct sparx5 * sparx5,u32 layer,u32 group)159e02a5ac6SDaniel Machon static void sparx5_lg_disable(struct sparx5 *sparx5, u32 layer, u32 group)
160e02a5ac6SDaniel Machon {
161e02a5ac6SDaniel Machon 	sparx5_lg_set_leak_time(sparx5, layer, group, 0);
162e02a5ac6SDaniel Machon }
163e02a5ac6SDaniel Machon 
sparx5_lg_get_group_by_index(struct sparx5 * sparx5,u32 layer,u32 idx,u32 * group)164e02a5ac6SDaniel Machon static int sparx5_lg_get_group_by_index(struct sparx5 *sparx5, u32 layer,
165e02a5ac6SDaniel Machon 					u32 idx, u32 *group)
166e02a5ac6SDaniel Machon {
167e02a5ac6SDaniel Machon 	u32 itr, next;
168e02a5ac6SDaniel Machon 	int i;
169e02a5ac6SDaniel Machon 
170e02a5ac6SDaniel Machon 	for (i = 0; i < SPX5_HSCH_LEAK_GRP_CNT; i++) {
171e02a5ac6SDaniel Machon 		if (sparx5_lg_is_empty(sparx5, layer, i))
172e02a5ac6SDaniel Machon 			continue;
173e02a5ac6SDaniel Machon 
174e02a5ac6SDaniel Machon 		itr = sparx5_lg_get_first(sparx5, layer, i);
175e02a5ac6SDaniel Machon 
176e02a5ac6SDaniel Machon 		for (;;) {
177e02a5ac6SDaniel Machon 			next = sparx5_lg_get_next(sparx5, layer, i, itr);
178e02a5ac6SDaniel Machon 
179e02a5ac6SDaniel Machon 			if (itr == idx) {
180e02a5ac6SDaniel Machon 				*group = i;
181e02a5ac6SDaniel Machon 				return 0; /* Found it */
182e02a5ac6SDaniel Machon 			}
183e02a5ac6SDaniel Machon 			if (itr == next)
184e02a5ac6SDaniel Machon 				break; /* Was not found */
185e02a5ac6SDaniel Machon 
186e02a5ac6SDaniel Machon 			itr = next;
187e02a5ac6SDaniel Machon 		}
188e02a5ac6SDaniel Machon 	}
189e02a5ac6SDaniel Machon 
190e02a5ac6SDaniel Machon 	return -1;
191e02a5ac6SDaniel Machon }
192e02a5ac6SDaniel Machon 
sparx5_lg_get_group_by_rate(u32 layer,u32 rate,u32 * group)193e02a5ac6SDaniel Machon static int sparx5_lg_get_group_by_rate(u32 layer, u32 rate, u32 *group)
194e02a5ac6SDaniel Machon {
195e02a5ac6SDaniel Machon 	struct sparx5_layer *l = &layers[layer];
196e02a5ac6SDaniel Machon 	struct sparx5_lg *lg;
197e02a5ac6SDaniel Machon 	u32 i;
198e02a5ac6SDaniel Machon 
199e02a5ac6SDaniel Machon 	for (i = 0; i < SPX5_HSCH_LEAK_GRP_CNT; i++) {
200e02a5ac6SDaniel Machon 		lg = &l->leak_groups[i];
201e02a5ac6SDaniel Machon 		if (rate <= lg->max_rate) {
202e02a5ac6SDaniel Machon 			*group = i;
203e02a5ac6SDaniel Machon 			return 0;
204e02a5ac6SDaniel Machon 		}
205e02a5ac6SDaniel Machon 	}
206e02a5ac6SDaniel Machon 
207e02a5ac6SDaniel Machon 	return -1;
208e02a5ac6SDaniel Machon }
209e02a5ac6SDaniel Machon 
sparx5_lg_get_adjacent(struct sparx5 * sparx5,u32 layer,u32 group,u32 idx,u32 * prev,u32 * next,u32 * first)210e02a5ac6SDaniel Machon static int sparx5_lg_get_adjacent(struct sparx5 *sparx5, u32 layer, u32 group,
211e02a5ac6SDaniel Machon 				  u32 idx, u32 *prev, u32 *next, u32 *first)
212e02a5ac6SDaniel Machon {
213e02a5ac6SDaniel Machon 	u32 itr;
214e02a5ac6SDaniel Machon 
215e02a5ac6SDaniel Machon 	*first = sparx5_lg_get_first(sparx5, layer, group);
216e02a5ac6SDaniel Machon 	*prev = *first;
217e02a5ac6SDaniel Machon 	*next = *first;
218e02a5ac6SDaniel Machon 	itr = *first;
219e02a5ac6SDaniel Machon 
220e02a5ac6SDaniel Machon 	for (;;) {
221e02a5ac6SDaniel Machon 		*next = sparx5_lg_get_next(sparx5, layer, group, itr);
222e02a5ac6SDaniel Machon 
223e02a5ac6SDaniel Machon 		if (itr == idx)
224e02a5ac6SDaniel Machon 			return 0; /* Found it */
225e02a5ac6SDaniel Machon 
226e02a5ac6SDaniel Machon 		if (itr == *next)
227e02a5ac6SDaniel Machon 			return -1; /* Was not found */
228e02a5ac6SDaniel Machon 
229e02a5ac6SDaniel Machon 		*prev = itr;
230e02a5ac6SDaniel Machon 		itr = *next;
231e02a5ac6SDaniel Machon 	}
232e02a5ac6SDaniel Machon 
233e02a5ac6SDaniel Machon 	return -1;
234e02a5ac6SDaniel Machon }
235e02a5ac6SDaniel Machon 
sparx5_lg_conf_set(struct sparx5 * sparx5,u32 layer,u32 group,u32 se_first,u32 idx,u32 idx_next,bool empty)236e02a5ac6SDaniel Machon static int sparx5_lg_conf_set(struct sparx5 *sparx5, u32 layer, u32 group,
237e02a5ac6SDaniel Machon 			      u32 se_first, u32 idx, u32 idx_next, bool empty)
238e02a5ac6SDaniel Machon {
239e02a5ac6SDaniel Machon 	u32 leak_time = layers[layer].leak_groups[group].leak_time;
240e02a5ac6SDaniel Machon 
241e02a5ac6SDaniel Machon 	/* Stop leaking */
242e02a5ac6SDaniel Machon 	sparx5_lg_disable(sparx5, layer, group);
243e02a5ac6SDaniel Machon 
244e02a5ac6SDaniel Machon 	if (empty)
245e02a5ac6SDaniel Machon 		return 0;
246e02a5ac6SDaniel Machon 
247e02a5ac6SDaniel Machon 	/* Select layer */
248e02a5ac6SDaniel Machon 	spx5_rmw(HSCH_HSCH_CFG_CFG_HSCH_LAYER_SET(layer),
249e02a5ac6SDaniel Machon 		 HSCH_HSCH_CFG_CFG_HSCH_LAYER, sparx5, HSCH_HSCH_CFG_CFG);
250e02a5ac6SDaniel Machon 
251e02a5ac6SDaniel Machon 	/* Link elements */
252e02a5ac6SDaniel Machon 	spx5_wr(HSCH_SE_CONNECT_SE_LEAK_LINK_SET(idx_next), sparx5,
253e02a5ac6SDaniel Machon 		HSCH_SE_CONNECT(idx));
254e02a5ac6SDaniel Machon 
255e02a5ac6SDaniel Machon 	/* Set the first element. */
256e02a5ac6SDaniel Machon 	spx5_rmw(HSCH_HSCH_LEAK_CFG_LEAK_FIRST_SET(se_first),
257e02a5ac6SDaniel Machon 		 HSCH_HSCH_LEAK_CFG_LEAK_FIRST, sparx5,
258e02a5ac6SDaniel Machon 		 HSCH_HSCH_LEAK_CFG(layer, group));
259e02a5ac6SDaniel Machon 
260e02a5ac6SDaniel Machon 	/* Start leaking */
261e02a5ac6SDaniel Machon 	sparx5_lg_enable(sparx5, layer, group, leak_time);
262e02a5ac6SDaniel Machon 
263e02a5ac6SDaniel Machon 	return 0;
264e02a5ac6SDaniel Machon }
265e02a5ac6SDaniel Machon 
sparx5_lg_del(struct sparx5 * sparx5,u32 layer,u32 group,u32 idx)266e02a5ac6SDaniel Machon static int sparx5_lg_del(struct sparx5 *sparx5, u32 layer, u32 group, u32 idx)
267e02a5ac6SDaniel Machon {
268e02a5ac6SDaniel Machon 	u32 first, next, prev;
269e02a5ac6SDaniel Machon 	bool empty = false;
270e02a5ac6SDaniel Machon 
271e02a5ac6SDaniel Machon 	/* idx *must* be present in the leak group */
272e02a5ac6SDaniel Machon 	WARN_ON(sparx5_lg_get_adjacent(sparx5, layer, group, idx, &prev, &next,
273e02a5ac6SDaniel Machon 				       &first) < 0);
274e02a5ac6SDaniel Machon 
275e02a5ac6SDaniel Machon 	if (sparx5_lg_is_singular(sparx5, layer, group)) {
276e02a5ac6SDaniel Machon 		empty = true;
277e02a5ac6SDaniel Machon 	} else if (sparx5_lg_is_last(sparx5, layer, group, idx)) {
278e02a5ac6SDaniel Machon 		/* idx is removed, prev is now last */
279e02a5ac6SDaniel Machon 		idx = prev;
280e02a5ac6SDaniel Machon 		next = prev;
281e02a5ac6SDaniel Machon 	} else if (sparx5_lg_is_first(sparx5, layer, group, idx)) {
282e02a5ac6SDaniel Machon 		/* idx is removed and points to itself, first is next */
283e02a5ac6SDaniel Machon 		first = next;
284e02a5ac6SDaniel Machon 		next = idx;
285e02a5ac6SDaniel Machon 	} else {
286e02a5ac6SDaniel Machon 		/* Next is not touched */
287e02a5ac6SDaniel Machon 		idx = prev;
288e02a5ac6SDaniel Machon 	}
289e02a5ac6SDaniel Machon 
290e02a5ac6SDaniel Machon 	return sparx5_lg_conf_set(sparx5, layer, group, first, idx, next,
291e02a5ac6SDaniel Machon 				  empty);
292e02a5ac6SDaniel Machon }
293e02a5ac6SDaniel Machon 
sparx5_lg_add(struct sparx5 * sparx5,u32 layer,u32 new_group,u32 idx)294e02a5ac6SDaniel Machon static int sparx5_lg_add(struct sparx5 *sparx5, u32 layer, u32 new_group,
295e02a5ac6SDaniel Machon 			 u32 idx)
296e02a5ac6SDaniel Machon {
297e02a5ac6SDaniel Machon 	u32 first, next, old_group;
298e02a5ac6SDaniel Machon 
299e02a5ac6SDaniel Machon 	pr_debug("ADD: layer: %d, new_group: %d, idx: %d", layer, new_group,
300e02a5ac6SDaniel Machon 		 idx);
301e02a5ac6SDaniel Machon 
302e02a5ac6SDaniel Machon 	/* Is this SE already shaping ? */
303e02a5ac6SDaniel Machon 	if (sparx5_lg_get_group_by_index(sparx5, layer, idx, &old_group) >= 0) {
304e02a5ac6SDaniel Machon 		if (old_group != new_group) {
305e02a5ac6SDaniel Machon 			/* Delete from old group */
306e02a5ac6SDaniel Machon 			sparx5_lg_del(sparx5, layer, old_group, idx);
307e02a5ac6SDaniel Machon 		} else {
308e02a5ac6SDaniel Machon 			/* Nothing to do here */
309e02a5ac6SDaniel Machon 			return 0;
310e02a5ac6SDaniel Machon 		}
311e02a5ac6SDaniel Machon 	}
312e02a5ac6SDaniel Machon 
313e02a5ac6SDaniel Machon 	/* We always add to head of the list */
314e02a5ac6SDaniel Machon 	first = idx;
315e02a5ac6SDaniel Machon 
316e02a5ac6SDaniel Machon 	if (sparx5_lg_is_empty(sparx5, layer, new_group))
317e02a5ac6SDaniel Machon 		next = idx;
318e02a5ac6SDaniel Machon 	else
319e02a5ac6SDaniel Machon 		next = sparx5_lg_get_first(sparx5, layer, new_group);
320e02a5ac6SDaniel Machon 
321e02a5ac6SDaniel Machon 	return sparx5_lg_conf_set(sparx5, layer, new_group, first, idx, next,
322e02a5ac6SDaniel Machon 				  false);
323e02a5ac6SDaniel Machon }
324e02a5ac6SDaniel Machon 
sparx5_shaper_conf_set(struct sparx5_port * port,const struct sparx5_shaper * sh,u32 layer,u32 idx,u32 group)325e02a5ac6SDaniel Machon static int sparx5_shaper_conf_set(struct sparx5_port *port,
326e02a5ac6SDaniel Machon 				  const struct sparx5_shaper *sh, u32 layer,
327e02a5ac6SDaniel Machon 				  u32 idx, u32 group)
328e02a5ac6SDaniel Machon {
329e02a5ac6SDaniel Machon 	int (*sparx5_lg_action)(struct sparx5 *, u32, u32, u32);
330e02a5ac6SDaniel Machon 	struct sparx5 *sparx5 = port->sparx5;
331e02a5ac6SDaniel Machon 
332e02a5ac6SDaniel Machon 	if (!sh->rate && !sh->burst)
333e02a5ac6SDaniel Machon 		sparx5_lg_action = &sparx5_lg_del;
334e02a5ac6SDaniel Machon 	else
335e02a5ac6SDaniel Machon 		sparx5_lg_action = &sparx5_lg_add;
336e02a5ac6SDaniel Machon 
337e02a5ac6SDaniel Machon 	/* Select layer */
338e02a5ac6SDaniel Machon 	spx5_rmw(HSCH_HSCH_CFG_CFG_HSCH_LAYER_SET(layer),
339e02a5ac6SDaniel Machon 		 HSCH_HSCH_CFG_CFG_HSCH_LAYER, sparx5, HSCH_HSCH_CFG_CFG);
340e02a5ac6SDaniel Machon 
341e02a5ac6SDaniel Machon 	/* Set frame mode */
342e02a5ac6SDaniel Machon 	spx5_rmw(HSCH_SE_CFG_SE_FRM_MODE_SET(sh->mode), HSCH_SE_CFG_SE_FRM_MODE,
343e02a5ac6SDaniel Machon 		 sparx5, HSCH_SE_CFG(idx));
344e02a5ac6SDaniel Machon 
345e02a5ac6SDaniel Machon 	/* Set committed rate and burst */
346e02a5ac6SDaniel Machon 	spx5_wr(HSCH_CIR_CFG_CIR_RATE_SET(sh->rate) |
347e02a5ac6SDaniel Machon 			HSCH_CIR_CFG_CIR_BURST_SET(sh->burst),
348e02a5ac6SDaniel Machon 		sparx5, HSCH_CIR_CFG(idx));
349e02a5ac6SDaniel Machon 
350e02a5ac6SDaniel Machon 	/* This has to be done after the shaper configuration has been set */
351e02a5ac6SDaniel Machon 	sparx5_lg_action(sparx5, layer, group, idx);
352e02a5ac6SDaniel Machon 
353e02a5ac6SDaniel Machon 	return 0;
354e02a5ac6SDaniel Machon }
355e02a5ac6SDaniel Machon 
sparx5_weight_to_hw_cost(u32 weight_min,u32 weight)35621122542SDaniel Machon static u32 sparx5_weight_to_hw_cost(u32 weight_min, u32 weight)
35721122542SDaniel Machon {
35821122542SDaniel Machon 	return ((((SPX5_DWRR_COST_MAX << 4) * weight_min / weight) + 8) >> 4) -
35921122542SDaniel Machon 	       1;
36021122542SDaniel Machon }
36121122542SDaniel Machon 
sparx5_dwrr_conf_set(struct sparx5_port * port,struct sparx5_dwrr * dwrr)36221122542SDaniel Machon static int sparx5_dwrr_conf_set(struct sparx5_port *port,
36321122542SDaniel Machon 				struct sparx5_dwrr *dwrr)
36421122542SDaniel Machon {
36521122542SDaniel Machon 	int i;
36621122542SDaniel Machon 
36721122542SDaniel Machon 	spx5_rmw(HSCH_HSCH_CFG_CFG_HSCH_LAYER_SET(2) |
36821122542SDaniel Machon 		 HSCH_HSCH_CFG_CFG_CFG_SE_IDX_SET(port->portno),
36921122542SDaniel Machon 		 HSCH_HSCH_CFG_CFG_HSCH_LAYER | HSCH_HSCH_CFG_CFG_CFG_SE_IDX,
37021122542SDaniel Machon 		 port->sparx5, HSCH_HSCH_CFG_CFG);
37121122542SDaniel Machon 
37221122542SDaniel Machon 	/* Number of *lower* indexes that are arbitrated dwrr */
37321122542SDaniel Machon 	spx5_rmw(HSCH_SE_CFG_SE_DWRR_CNT_SET(dwrr->count),
37421122542SDaniel Machon 		 HSCH_SE_CFG_SE_DWRR_CNT, port->sparx5,
37521122542SDaniel Machon 		 HSCH_SE_CFG(port->portno));
37621122542SDaniel Machon 
37721122542SDaniel Machon 	for (i = 0; i < dwrr->count; i++) {
37821122542SDaniel Machon 		spx5_rmw(HSCH_DWRR_ENTRY_DWRR_COST_SET(dwrr->cost[i]),
37921122542SDaniel Machon 			 HSCH_DWRR_ENTRY_DWRR_COST, port->sparx5,
38021122542SDaniel Machon 			 HSCH_DWRR_ENTRY(i));
38121122542SDaniel Machon 	}
38221122542SDaniel Machon 
38321122542SDaniel Machon 	return 0;
38421122542SDaniel Machon }
38521122542SDaniel Machon 
sparx5_leak_groups_init(struct sparx5 * sparx5)386e02a5ac6SDaniel Machon static int sparx5_leak_groups_init(struct sparx5 *sparx5)
387e02a5ac6SDaniel Machon {
388e02a5ac6SDaniel Machon 	struct sparx5_layer *layer;
389e02a5ac6SDaniel Machon 	u32 sys_clk_per_100ps;
390e02a5ac6SDaniel Machon 	struct sparx5_lg *lg;
391e02a5ac6SDaniel Machon 	u32 leak_time_us;
392e02a5ac6SDaniel Machon 	int i, ii;
393e02a5ac6SDaniel Machon 
394e02a5ac6SDaniel Machon 	sys_clk_per_100ps = spx5_rd(sparx5, HSCH_SYS_CLK_PER);
395e02a5ac6SDaniel Machon 
396e02a5ac6SDaniel Machon 	for (i = 0; i < SPX5_HSCH_LAYER_CNT; i++) {
397e02a5ac6SDaniel Machon 		layer = &layers[i];
398e02a5ac6SDaniel Machon 		for (ii = 0; ii < SPX5_HSCH_LEAK_GRP_CNT; ii++) {
399e02a5ac6SDaniel Machon 			lg = &layer->leak_groups[ii];
400e02a5ac6SDaniel Machon 			lg->max_rate = spx5_hsch_max_group_rate[ii];
401e02a5ac6SDaniel Machon 
402e02a5ac6SDaniel Machon 			/* Calculate the leak time in us, to serve a maximum
403e02a5ac6SDaniel Machon 			 * rate of 'max_rate' for this group
404e02a5ac6SDaniel Machon 			 */
405e02a5ac6SDaniel Machon 			leak_time_us = (SPX5_SE_RATE_MAX * 1000) / lg->max_rate;
406e02a5ac6SDaniel Machon 
407e02a5ac6SDaniel Machon 			/* Hardware wants leak time in ns */
408e02a5ac6SDaniel Machon 			lg->leak_time = 1000 * leak_time_us;
409e02a5ac6SDaniel Machon 
410e02a5ac6SDaniel Machon 			/* Calculate resolution */
411e02a5ac6SDaniel Machon 			lg->resolution = 1000 / leak_time_us;
412e02a5ac6SDaniel Machon 
413e02a5ac6SDaniel Machon 			/* Maximum number of shapers that can be served by
414e02a5ac6SDaniel Machon 			 * this leak group
415e02a5ac6SDaniel Machon 			 */
416e02a5ac6SDaniel Machon 			lg->max_ses = (1000 * leak_time_us) / sys_clk_per_100ps;
417e02a5ac6SDaniel Machon 
418e02a5ac6SDaniel Machon 			/* Example:
419e02a5ac6SDaniel Machon 			 * Wanted bandwidth is 100Mbit:
420e02a5ac6SDaniel Machon 			 *
421e02a5ac6SDaniel Machon 			 * 100 mbps can be served by leak group zero.
422e02a5ac6SDaniel Machon 			 *
423e02a5ac6SDaniel Machon 			 * leak_time is 125000 ns.
424e02a5ac6SDaniel Machon 			 * resolution is: 8
425e02a5ac6SDaniel Machon 			 *
426e02a5ac6SDaniel Machon 			 * cir          = 100000 / 8 = 12500
427e02a5ac6SDaniel Machon 			 * leaks_pr_sec = 125000 / 10^9 = 8000
428e02a5ac6SDaniel Machon 			 * bw           = 12500 * 8000 = 10^8 (100 Mbit)
429e02a5ac6SDaniel Machon 			 */
430e02a5ac6SDaniel Machon 
431e02a5ac6SDaniel Machon 			/* Disable by default - this also indicates an empty
432e02a5ac6SDaniel Machon 			 * leak group
433e02a5ac6SDaniel Machon 			 */
434e02a5ac6SDaniel Machon 			sparx5_lg_disable(sparx5, i, ii);
435e02a5ac6SDaniel Machon 		}
436e02a5ac6SDaniel Machon 	}
437e02a5ac6SDaniel Machon 
438e02a5ac6SDaniel Machon 	return 0;
439e02a5ac6SDaniel Machon }
440e02a5ac6SDaniel Machon 
sparx5_qos_init(struct sparx5 * sparx5)441e02a5ac6SDaniel Machon int sparx5_qos_init(struct sparx5 *sparx5)
442e02a5ac6SDaniel Machon {
443e02a5ac6SDaniel Machon 	int ret;
444e02a5ac6SDaniel Machon 
445e02a5ac6SDaniel Machon 	ret = sparx5_leak_groups_init(sparx5);
446e02a5ac6SDaniel Machon 	if (ret < 0)
447e02a5ac6SDaniel Machon 		return ret;
448e02a5ac6SDaniel Machon 
44992ef3d01SDaniel Machon 	ret = sparx5_dcb_init(sparx5);
45092ef3d01SDaniel Machon 	if (ret < 0)
45192ef3d01SDaniel Machon 		return ret;
45292ef3d01SDaniel Machon 
453*e116b19dSDaniel Machon 	sparx5_psfp_init(sparx5);
454*e116b19dSDaniel Machon 
455e02a5ac6SDaniel Machon 	return 0;
456e02a5ac6SDaniel Machon }
457e02a5ac6SDaniel Machon 
sparx5_tc_mqprio_add(struct net_device * ndev,u8 num_tc)458ab0e493eSDaniel Machon int sparx5_tc_mqprio_add(struct net_device *ndev, u8 num_tc)
459ab0e493eSDaniel Machon {
460ab0e493eSDaniel Machon 	int i;
461ab0e493eSDaniel Machon 
462ab0e493eSDaniel Machon 	if (num_tc != SPX5_PRIOS) {
463ab0e493eSDaniel Machon 		netdev_err(ndev, "Only %d traffic classes supported\n",
464ab0e493eSDaniel Machon 			   SPX5_PRIOS);
465ab0e493eSDaniel Machon 		return -EINVAL;
466ab0e493eSDaniel Machon 	}
467ab0e493eSDaniel Machon 
468ab0e493eSDaniel Machon 	netdev_set_num_tc(ndev, num_tc);
469ab0e493eSDaniel Machon 
470ab0e493eSDaniel Machon 	for (i = 0; i < num_tc; i++)
471ab0e493eSDaniel Machon 		netdev_set_tc_queue(ndev, i, 1, i);
472ab0e493eSDaniel Machon 
473ab0e493eSDaniel Machon 	netdev_dbg(ndev, "dev->num_tc %u dev->real_num_tx_queues %u\n",
474ab0e493eSDaniel Machon 		   ndev->num_tc, ndev->real_num_tx_queues);
475ab0e493eSDaniel Machon 
476ab0e493eSDaniel Machon 	return 0;
477ab0e493eSDaniel Machon }
478ab0e493eSDaniel Machon 
sparx5_tc_mqprio_del(struct net_device * ndev)479ab0e493eSDaniel Machon int sparx5_tc_mqprio_del(struct net_device *ndev)
480ab0e493eSDaniel Machon {
481ab0e493eSDaniel Machon 	netdev_reset_tc(ndev);
482ab0e493eSDaniel Machon 
483ab0e493eSDaniel Machon 	netdev_dbg(ndev, "dev->num_tc %u dev->real_num_tx_queues %u\n",
484ab0e493eSDaniel Machon 		   ndev->num_tc, ndev->real_num_tx_queues);
485ab0e493eSDaniel Machon 
486ab0e493eSDaniel Machon 	return 0;
487ab0e493eSDaniel Machon }
488e02a5ac6SDaniel Machon 
sparx5_tc_tbf_add(struct sparx5_port * port,struct tc_tbf_qopt_offload_replace_params * params,u32 layer,u32 idx)489e02a5ac6SDaniel Machon int sparx5_tc_tbf_add(struct sparx5_port *port,
490e02a5ac6SDaniel Machon 		      struct tc_tbf_qopt_offload_replace_params *params,
491e02a5ac6SDaniel Machon 		      u32 layer, u32 idx)
492e02a5ac6SDaniel Machon {
493e02a5ac6SDaniel Machon 	struct sparx5_shaper sh = {
494e02a5ac6SDaniel Machon 		.mode = SPX5_SE_MODE_DATARATE,
495e02a5ac6SDaniel Machon 		.rate = div_u64(params->rate.rate_bytes_ps, 1000) * 8,
496e02a5ac6SDaniel Machon 		.burst = params->max_size,
497e02a5ac6SDaniel Machon 	};
498e02a5ac6SDaniel Machon 	struct sparx5_lg *lg;
499e02a5ac6SDaniel Machon 	u32 group;
500e02a5ac6SDaniel Machon 
501e02a5ac6SDaniel Machon 	/* Find suitable group for this se */
502e02a5ac6SDaniel Machon 	if (sparx5_lg_get_group_by_rate(layer, sh.rate, &group) < 0) {
503e02a5ac6SDaniel Machon 		pr_debug("Could not find leak group for se with rate: %d",
504e02a5ac6SDaniel Machon 			 sh.rate);
505e02a5ac6SDaniel Machon 		return -EINVAL;
506e02a5ac6SDaniel Machon 	}
507e02a5ac6SDaniel Machon 
508e02a5ac6SDaniel Machon 	lg = &layers[layer].leak_groups[group];
509e02a5ac6SDaniel Machon 
510e02a5ac6SDaniel Machon 	pr_debug("Found matching group (speed: %d)\n", lg->max_rate);
511e02a5ac6SDaniel Machon 
512e02a5ac6SDaniel Machon 	if (sh.rate < SPX5_SE_RATE_MIN || sh.burst < SPX5_SE_BURST_MIN)
513e02a5ac6SDaniel Machon 		return -EINVAL;
514e02a5ac6SDaniel Machon 
515e02a5ac6SDaniel Machon 	/* Calculate committed rate and burst */
516e02a5ac6SDaniel Machon 	sh.rate = DIV_ROUND_UP(sh.rate, lg->resolution);
517e02a5ac6SDaniel Machon 	sh.burst = DIV_ROUND_UP(sh.burst, SPX5_SE_BURST_UNIT);
518e02a5ac6SDaniel Machon 
519e02a5ac6SDaniel Machon 	if (sh.rate > SPX5_SE_RATE_MAX || sh.burst > SPX5_SE_BURST_MAX)
520e02a5ac6SDaniel Machon 		return -EINVAL;
521e02a5ac6SDaniel Machon 
522e02a5ac6SDaniel Machon 	return sparx5_shaper_conf_set(port, &sh, layer, idx, group);
523e02a5ac6SDaniel Machon }
524e02a5ac6SDaniel Machon 
sparx5_tc_tbf_del(struct sparx5_port * port,u32 layer,u32 idx)525e02a5ac6SDaniel Machon int sparx5_tc_tbf_del(struct sparx5_port *port, u32 layer, u32 idx)
526e02a5ac6SDaniel Machon {
527e02a5ac6SDaniel Machon 	struct sparx5_shaper sh = {0};
528e02a5ac6SDaniel Machon 	u32 group;
529e02a5ac6SDaniel Machon 
530e02a5ac6SDaniel Machon 	sparx5_lg_get_group_by_index(port->sparx5, layer, idx, &group);
531e02a5ac6SDaniel Machon 
532e02a5ac6SDaniel Machon 	return sparx5_shaper_conf_set(port, &sh, layer, idx, group);
533e02a5ac6SDaniel Machon }
53421122542SDaniel Machon 
sparx5_tc_ets_add(struct sparx5_port * port,struct tc_ets_qopt_offload_replace_params * params)53521122542SDaniel Machon int sparx5_tc_ets_add(struct sparx5_port *port,
53621122542SDaniel Machon 		      struct tc_ets_qopt_offload_replace_params *params)
53721122542SDaniel Machon {
53821122542SDaniel Machon 	struct sparx5_dwrr dwrr = {0};
53921122542SDaniel Machon 	/* Minimum weight for each iteration */
54021122542SDaniel Machon 	unsigned int w_min = 100;
54121122542SDaniel Machon 	int i;
54221122542SDaniel Machon 
54321122542SDaniel Machon 	/* Find minimum weight for all dwrr bands */
54421122542SDaniel Machon 	for (i = 0; i < SPX5_PRIOS; i++) {
54521122542SDaniel Machon 		if (params->quanta[i] == 0)
54621122542SDaniel Machon 			continue;
54721122542SDaniel Machon 		w_min = min(w_min, params->weights[i]);
54821122542SDaniel Machon 	}
54921122542SDaniel Machon 
55021122542SDaniel Machon 	for (i = 0; i < SPX5_PRIOS; i++) {
55121122542SDaniel Machon 		/* Strict band; skip */
55221122542SDaniel Machon 		if (params->quanta[i] == 0)
55321122542SDaniel Machon 			continue;
55421122542SDaniel Machon 
55521122542SDaniel Machon 		dwrr.count++;
55621122542SDaniel Machon 
55721122542SDaniel Machon 		/* On the sparx5, bands with higher indexes are preferred and
55821122542SDaniel Machon 		 * arbitrated strict. Strict bands are put in the lower indexes,
55921122542SDaniel Machon 		 * by tc, so we reverse the bands here.
56021122542SDaniel Machon 		 *
56121122542SDaniel Machon 		 * Also convert the weight to something the hardware
56221122542SDaniel Machon 		 * understands.
56321122542SDaniel Machon 		 */
56421122542SDaniel Machon 		dwrr.cost[SPX5_PRIOS - i - 1] =
56521122542SDaniel Machon 			sparx5_weight_to_hw_cost(w_min, params->weights[i]);
56621122542SDaniel Machon 	}
56721122542SDaniel Machon 
56821122542SDaniel Machon 	return sparx5_dwrr_conf_set(port, &dwrr);
56921122542SDaniel Machon }
57021122542SDaniel Machon 
sparx5_tc_ets_del(struct sparx5_port * port)57121122542SDaniel Machon int sparx5_tc_ets_del(struct sparx5_port *port)
57221122542SDaniel Machon {
57321122542SDaniel Machon 	struct sparx5_dwrr dwrr = {0};
57421122542SDaniel Machon 
57521122542SDaniel Machon 	return sparx5_dwrr_conf_set(port, &dwrr);
57621122542SDaniel Machon }
577