Loading...
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}
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}