Linux Audio

Check our new training course

Loading...
v6.2
  1// SPDX-License-Identifier: GPL-2.0
  2/* Copyright (c)  2019 Intel Corporation */
  3
  4#include "igc.h"
  5#include "igc_tsn.h"
  6
  7static bool is_any_launchtime(struct igc_adapter *adapter)
  8{
  9	int i;
 10
 11	for (i = 0; i < adapter->num_tx_queues; i++) {
 12		struct igc_ring *ring = adapter->tx_ring[i];
 13
 14		if (ring->launchtime_enable)
 15			return true;
 16	}
 17
 18	return false;
 19}
 20
 21static bool is_cbs_enabled(struct igc_adapter *adapter)
 22{
 23	int i;
 24
 25	for (i = 0; i < adapter->num_tx_queues; i++) {
 26		struct igc_ring *ring = adapter->tx_ring[i];
 27
 28		if (ring->cbs_enable)
 29			return true;
 30	}
 31
 32	return false;
 33}
 34
 35static unsigned int igc_tsn_new_flags(struct igc_adapter *adapter)
 36{
 37	unsigned int new_flags = adapter->flags & ~IGC_FLAG_TSN_ANY_ENABLED;
 38
 39	if (adapter->qbv_enable)
 40		new_flags |= IGC_FLAG_TSN_QBV_ENABLED;
 41
 42	if (is_any_launchtime(adapter))
 43		new_flags |= IGC_FLAG_TSN_QBV_ENABLED;
 44
 45	if (is_cbs_enabled(adapter))
 46		new_flags |= IGC_FLAG_TSN_QAV_ENABLED;
 47
 48	return new_flags;
 49}
 50
 51void igc_tsn_adjust_txtime_offset(struct igc_adapter *adapter)
 52{
 53	struct igc_hw *hw = &adapter->hw;
 54	u16 txoffset;
 55
 56	if (!is_any_launchtime(adapter))
 57		return;
 58
 59	switch (adapter->link_speed) {
 60	case SPEED_10:
 61		txoffset = IGC_TXOFFSET_SPEED_10;
 62		break;
 63	case SPEED_100:
 64		txoffset = IGC_TXOFFSET_SPEED_100;
 65		break;
 66	case SPEED_1000:
 67		txoffset = IGC_TXOFFSET_SPEED_1000;
 68		break;
 69	case SPEED_2500:
 70		txoffset = IGC_TXOFFSET_SPEED_2500;
 71		break;
 72	default:
 73		txoffset = 0;
 74		break;
 75	}
 76
 77	wr32(IGC_GTXOFFSET, txoffset);
 78}
 79
 80/* Returns the TSN specific registers to their default values after
 81 * the adapter is reset.
 82 */
 83static int igc_tsn_disable_offload(struct igc_adapter *adapter)
 84{
 85	struct igc_hw *hw = &adapter->hw;
 86	u32 tqavctrl;
 87	int i;
 88
 89	wr32(IGC_GTXOFFSET, 0);
 
 
 
 
 90	wr32(IGC_TXPBS, I225_TXPBSIZE_DEFAULT);
 91	wr32(IGC_DTXMXPKTSZ, IGC_DTXMXPKTSZ_DEFAULT);
 92
 93	tqavctrl = rd32(IGC_TQAVCTRL);
 94	tqavctrl &= ~(IGC_TQAVCTRL_TRANSMIT_MODE_TSN |
 95		      IGC_TQAVCTRL_ENHANCED_QAV);
 96	wr32(IGC_TQAVCTRL, tqavctrl);
 97
 98	for (i = 0; i < adapter->num_tx_queues; i++) {
 
 
 
 
 
 
 99		wr32(IGC_TXQCTL(i), 0);
100		wr32(IGC_STQT(i), 0);
101		wr32(IGC_ENDQT(i), NSEC_PER_SEC);
102	}
103
104	wr32(IGC_QBVCYCLET_S, 0);
105	wr32(IGC_QBVCYCLET, NSEC_PER_SEC);
106
107	adapter->flags &= ~IGC_FLAG_TSN_QBV_ENABLED;
108
109	return 0;
110}
111
112static int igc_tsn_enable_offload(struct igc_adapter *adapter)
113{
114	struct igc_hw *hw = &adapter->hw;
115	u32 tqavctrl, baset_l, baset_h;
116	u32 sec, nsec, cycle;
117	ktime_t base_time, systim;
118	int i;
119
 
 
 
120	cycle = adapter->cycle_time;
121	base_time = adapter->base_time;
122
123	wr32(IGC_TSAUXC, 0);
124	wr32(IGC_DTXMXPKTSZ, IGC_DTXMXPKTSZ_TSN);
125	wr32(IGC_TXPBS, IGC_TXPBSIZE_TSN);
126
127	tqavctrl = rd32(IGC_TQAVCTRL);
128	tqavctrl |= IGC_TQAVCTRL_TRANSMIT_MODE_TSN | IGC_TQAVCTRL_ENHANCED_QAV;
129	wr32(IGC_TQAVCTRL, tqavctrl);
130
131	wr32(IGC_QBVCYCLET_S, cycle);
132	wr32(IGC_QBVCYCLET, cycle);
133
134	for (i = 0; i < adapter->num_tx_queues; i++) {
135		struct igc_ring *ring = adapter->tx_ring[i];
136		u32 txqctl = 0;
137		u16 cbs_value;
138		u32 tqavcc;
139
140		wr32(IGC_STQT(i), ring->start_time);
141		wr32(IGC_ENDQT(i), ring->end_time);
142
143		txqctl |= IGC_TXQCTL_STRICT_CYCLE |
144			IGC_TXQCTL_STRICT_END;
 
 
 
 
 
 
 
145
146		if (ring->launchtime_enable)
147			txqctl |= IGC_TXQCTL_QUEUE_MODE_LAUNCHT;
148
149		/* Skip configuring CBS for Q2 and Q3 */
150		if (i > 1)
151			goto skip_cbs;
152
153		if (ring->cbs_enable) {
154			if (i == 0)
155				txqctl |= IGC_TXQCTL_QAV_SEL_CBS0;
156			else
157				txqctl |= IGC_TXQCTL_QAV_SEL_CBS1;
158
159			/* According to i225 datasheet section 7.5.2.7, we
160			 * should set the 'idleSlope' field from TQAVCC
161			 * register following the equation:
162			 *
163			 * value = link-speed   0x7736 * BW * 0.2
164			 *         ---------- *  -----------------         (E1)
165			 *          100Mbps            2.5
166			 *
167			 * Note that 'link-speed' is in Mbps.
168			 *
169			 * 'BW' is the percentage bandwidth out of full
170			 * link speed which can be found with the
171			 * following equation. Note that idleSlope here
172			 * is the parameter from this function
173			 * which is in kbps.
174			 *
175			 *     BW =     idleSlope
176			 *          -----------------                      (E2)
177			 *          link-speed * 1000
178			 *
179			 * That said, we can come up with a generic
180			 * equation to calculate the value we should set
181			 * it TQAVCC register by replacing 'BW' in E1 by E2.
182			 * The resulting equation is:
183			 *
184			 * value = link-speed * 0x7736 * idleSlope * 0.2
185			 *         -------------------------------------   (E3)
186			 *             100 * 2.5 * link-speed * 1000
187			 *
188			 * 'link-speed' is present in both sides of the
189			 * fraction so it is canceled out. The final
190			 * equation is the following:
191			 *
192			 *     value = idleSlope * 61036
193			 *             -----------------                   (E4)
194			 *                  2500000
195			 *
196			 * NOTE: For i225, given the above, we can see
197			 *       that idleslope is represented in
198			 *       40.959433 kbps units by the value at
199			 *       the TQAVCC register (2.5Gbps / 61036),
200			 *       which reduces the granularity for
201			 *       idleslope increments.
202			 *
203			 * In i225 controller, the sendSlope and loCredit
204			 * parameters from CBS are not configurable
205			 * by software so we don't do any
206			 * 'controller configuration' in respect to
207			 * these parameters.
208			 */
209			cbs_value = DIV_ROUND_UP_ULL(ring->idleslope
210						     * 61036ULL, 2500000);
211
212			tqavcc = rd32(IGC_TQAVCC(i));
213			tqavcc &= ~IGC_TQAVCC_IDLESLOPE_MASK;
214			tqavcc |= cbs_value | IGC_TQAVCC_KEEP_CREDITS;
215			wr32(IGC_TQAVCC(i), tqavcc);
216
217			wr32(IGC_TQAVHC(i),
218			     0x80000000 + ring->hicredit * 0x7735);
219		} else {
220			/* Disable any CBS for the queue */
221			txqctl &= ~(IGC_TXQCTL_QAV_SEL_MASK);
222
223			/* Set idleSlope to zero. */
224			tqavcc = rd32(IGC_TQAVCC(i));
225			tqavcc &= ~(IGC_TQAVCC_IDLESLOPE_MASK |
226				    IGC_TQAVCC_KEEP_CREDITS);
227			wr32(IGC_TQAVCC(i), tqavcc);
228
229			/* Set hiCredit to zero. */
230			wr32(IGC_TQAVHC(i), 0);
231		}
232skip_cbs:
233		wr32(IGC_TXQCTL(i), txqctl);
234	}
235
236	nsec = rd32(IGC_SYSTIML);
237	sec = rd32(IGC_SYSTIMH);
238
239	systim = ktime_set(sec, nsec);
240
241	if (ktime_compare(systim, base_time) > 0) {
242		s64 n;
243
244		n = div64_s64(ktime_sub_ns(systim, base_time), cycle);
245		base_time = ktime_add_ns(base_time, (n + 1) * cycle);
246	}
247
248	baset_h = div_s64_rem(base_time, NSEC_PER_SEC, &baset_l);
249
250	wr32(IGC_BASET_H, baset_h);
251	wr32(IGC_BASET_L, baset_l);
252
 
 
253	return 0;
254}
255
256int igc_tsn_reset(struct igc_adapter *adapter)
257{
258	unsigned int new_flags;
259	int err = 0;
260
261	new_flags = igc_tsn_new_flags(adapter);
262
263	if (!(new_flags & IGC_FLAG_TSN_ANY_ENABLED))
264		return igc_tsn_disable_offload(adapter);
265
266	err = igc_tsn_enable_offload(adapter);
267	if (err < 0)
268		return err;
269
270	adapter->flags = new_flags;
 
271
272	return err;
273}
274
275int igc_tsn_offload_apply(struct igc_adapter *adapter)
276{
277	int err;
 
 
 
278
279	if (netif_running(adapter->netdev)) {
280		schedule_work(&adapter->reset_task);
281		return 0;
282	}
283
284	err = igc_tsn_enable_offload(adapter);
285	if (err < 0)
286		return err;
287
288	adapter->flags = igc_tsn_new_flags(adapter);
289	return 0;
290}
v5.14.15
  1// SPDX-License-Identifier: GPL-2.0
  2/* Copyright (c)  2019 Intel Corporation */
  3
  4#include "igc.h"
  5#include "igc_tsn.h"
  6
  7static bool is_any_launchtime(struct igc_adapter *adapter)
  8{
  9	int i;
 10
 11	for (i = 0; i < adapter->num_tx_queues; i++) {
 12		struct igc_ring *ring = adapter->tx_ring[i];
 13
 14		if (ring->launchtime_enable)
 15			return true;
 16	}
 17
 18	return false;
 19}
 20
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 21/* Returns the TSN specific registers to their default values after
 22 * TSN offloading is disabled.
 23 */
 24static int igc_tsn_disable_offload(struct igc_adapter *adapter)
 25{
 26	struct igc_hw *hw = &adapter->hw;
 27	u32 tqavctrl;
 28	int i;
 29
 30	if (!(adapter->flags & IGC_FLAG_TSN_QBV_ENABLED))
 31		return 0;
 32
 33	adapter->cycle_time = 0;
 34
 35	wr32(IGC_TXPBS, I225_TXPBSIZE_DEFAULT);
 36	wr32(IGC_DTXMXPKTSZ, IGC_DTXMXPKTSZ_DEFAULT);
 37
 38	tqavctrl = rd32(IGC_TQAVCTRL);
 39	tqavctrl &= ~(IGC_TQAVCTRL_TRANSMIT_MODE_TSN |
 40		      IGC_TQAVCTRL_ENHANCED_QAV);
 41	wr32(IGC_TQAVCTRL, tqavctrl);
 42
 43	for (i = 0; i < adapter->num_tx_queues; i++) {
 44		struct igc_ring *ring = adapter->tx_ring[i];
 45
 46		ring->start_time = 0;
 47		ring->end_time = 0;
 48		ring->launchtime_enable = false;
 49
 50		wr32(IGC_TXQCTL(i), 0);
 51		wr32(IGC_STQT(i), 0);
 52		wr32(IGC_ENDQT(i), NSEC_PER_SEC);
 53	}
 54
 55	wr32(IGC_QBVCYCLET_S, NSEC_PER_SEC);
 56	wr32(IGC_QBVCYCLET, NSEC_PER_SEC);
 57
 58	adapter->flags &= ~IGC_FLAG_TSN_QBV_ENABLED;
 59
 60	return 0;
 61}
 62
 63static int igc_tsn_enable_offload(struct igc_adapter *adapter)
 64{
 65	struct igc_hw *hw = &adapter->hw;
 66	u32 tqavctrl, baset_l, baset_h;
 67	u32 sec, nsec, cycle;
 68	ktime_t base_time, systim;
 69	int i;
 70
 71	if (adapter->flags & IGC_FLAG_TSN_QBV_ENABLED)
 72		return 0;
 73
 74	cycle = adapter->cycle_time;
 75	base_time = adapter->base_time;
 76
 77	wr32(IGC_TSAUXC, 0);
 78	wr32(IGC_DTXMXPKTSZ, IGC_DTXMXPKTSZ_TSN);
 79	wr32(IGC_TXPBS, IGC_TXPBSIZE_TSN);
 80
 81	tqavctrl = rd32(IGC_TQAVCTRL);
 82	tqavctrl |= IGC_TQAVCTRL_TRANSMIT_MODE_TSN | IGC_TQAVCTRL_ENHANCED_QAV;
 83	wr32(IGC_TQAVCTRL, tqavctrl);
 84
 85	wr32(IGC_QBVCYCLET_S, cycle);
 86	wr32(IGC_QBVCYCLET, cycle);
 87
 88	for (i = 0; i < adapter->num_tx_queues; i++) {
 89		struct igc_ring *ring = adapter->tx_ring[i];
 90		u32 txqctl = 0;
 
 
 91
 92		wr32(IGC_STQT(i), ring->start_time);
 93		wr32(IGC_ENDQT(i), ring->end_time);
 94
 95		if (adapter->base_time) {
 96			/* If we have a base_time we are in "taprio"
 97			 * mode and we need to be strict about the
 98			 * cycles: only transmit a packet if it can be
 99			 * completed during that cycle.
100			 */
101			txqctl |= IGC_TXQCTL_STRICT_CYCLE |
102				IGC_TXQCTL_STRICT_END;
103		}
104
105		if (ring->launchtime_enable)
106			txqctl |= IGC_TXQCTL_QUEUE_MODE_LAUNCHT;
107
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
108		wr32(IGC_TXQCTL(i), txqctl);
109	}
110
111	nsec = rd32(IGC_SYSTIML);
112	sec = rd32(IGC_SYSTIMH);
113
114	systim = ktime_set(sec, nsec);
115
116	if (ktime_compare(systim, base_time) > 0) {
117		s64 n;
118
119		n = div64_s64(ktime_sub_ns(systim, base_time), cycle);
120		base_time = ktime_add_ns(base_time, (n + 1) * cycle);
121	}
122
123	baset_h = div_s64_rem(base_time, NSEC_PER_SEC, &baset_l);
124
125	wr32(IGC_BASET_H, baset_h);
126	wr32(IGC_BASET_L, baset_l);
127
128	adapter->flags |= IGC_FLAG_TSN_QBV_ENABLED;
129
130	return 0;
131}
132
133int igc_tsn_offload_apply(struct igc_adapter *adapter)
134{
135	bool is_any_enabled = adapter->base_time || is_any_launchtime(adapter);
 
 
 
 
 
 
136
137	if (!(adapter->flags & IGC_FLAG_TSN_QBV_ENABLED) && !is_any_enabled)
138		return 0;
 
139
140	if (!is_any_enabled) {
141		int err = igc_tsn_disable_offload(adapter);
142
143		if (err < 0)
144			return err;
145
146		/* The BASET registers aren't cleared when writing
147		 * into them, force a reset if the interface is
148		 * running.
149		 */
150		if (netif_running(adapter->netdev))
151			schedule_work(&adapter->reset_task);
152
 
 
153		return 0;
154	}
155
156	return igc_tsn_enable_offload(adapter);
 
 
 
 
 
157}