1 // SPDX-License-Identifier: GPL-2.0
2 /* Copyright (c) 2019 Intel Corporation */
7 static bool is_any_launchtime(struct igc_adapter *adapter)
11 for (i = 0; i < adapter->num_tx_queues; i++) {
12 struct igc_ring *ring = adapter->tx_ring[i];
14 if (ring->launchtime_enable)
21 /* Returns the TSN specific registers to their default values after
22 * TSN offloading is disabled.
24 static int igc_tsn_disable_offload(struct igc_adapter *adapter)
26 struct igc_hw *hw = &adapter->hw;
30 if (!(adapter->flags & IGC_FLAG_TSN_QBV_ENABLED))
33 adapter->cycle_time = 0;
35 wr32(IGC_TXPBS, I225_TXPBSIZE_DEFAULT);
36 wr32(IGC_DTXMXPKTSZ, IGC_DTXMXPKTSZ_DEFAULT);
38 tqavctrl = rd32(IGC_TQAVCTRL);
39 tqavctrl &= ~(IGC_TQAVCTRL_TRANSMIT_MODE_TSN |
40 IGC_TQAVCTRL_ENHANCED_QAV);
41 wr32(IGC_TQAVCTRL, tqavctrl);
43 for (i = 0; i < adapter->num_tx_queues; i++) {
44 struct igc_ring *ring = adapter->tx_ring[i];
48 ring->launchtime_enable = false;
50 wr32(IGC_TXQCTL(i), 0);
52 wr32(IGC_ENDQT(i), NSEC_PER_SEC);
55 wr32(IGC_QBVCYCLET_S, NSEC_PER_SEC);
56 wr32(IGC_QBVCYCLET, NSEC_PER_SEC);
58 adapter->flags &= ~IGC_FLAG_TSN_QBV_ENABLED;
63 static int igc_tsn_enable_offload(struct igc_adapter *adapter)
65 struct igc_hw *hw = &adapter->hw;
66 u32 tqavctrl, baset_l, baset_h;
68 ktime_t base_time, systim;
71 if (adapter->flags & IGC_FLAG_TSN_QBV_ENABLED)
74 cycle = adapter->cycle_time;
75 base_time = adapter->base_time;
78 wr32(IGC_DTXMXPKTSZ, IGC_DTXMXPKTSZ_TSN);
79 wr32(IGC_TXPBS, IGC_TXPBSIZE_TSN);
81 tqavctrl = rd32(IGC_TQAVCTRL);
82 tqavctrl |= IGC_TQAVCTRL_TRANSMIT_MODE_TSN | IGC_TQAVCTRL_ENHANCED_QAV;
83 wr32(IGC_TQAVCTRL, tqavctrl);
85 wr32(IGC_QBVCYCLET_S, cycle);
86 wr32(IGC_QBVCYCLET, cycle);
88 for (i = 0; i < adapter->num_tx_queues; i++) {
89 struct igc_ring *ring = adapter->tx_ring[i];
92 wr32(IGC_STQT(i), ring->start_time);
93 wr32(IGC_ENDQT(i), ring->end_time);
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.
101 txqctl |= IGC_TXQCTL_STRICT_CYCLE |
102 IGC_TXQCTL_STRICT_END;
105 if (ring->launchtime_enable)
106 txqctl |= IGC_TXQCTL_QUEUE_MODE_LAUNCHT;
108 wr32(IGC_TXQCTL(i), txqctl);
111 nsec = rd32(IGC_SYSTIML);
112 sec = rd32(IGC_SYSTIMH);
114 systim = ktime_set(sec, nsec);
116 if (ktime_compare(systim, base_time) > 0) {
119 n = div64_s64(ktime_sub_ns(systim, base_time), cycle);
120 base_time = ktime_add_ns(base_time, (n + 1) * cycle);
123 baset_h = div_s64_rem(base_time, NSEC_PER_SEC, &baset_l);
125 wr32(IGC_BASET_H, baset_h);
126 wr32(IGC_BASET_L, baset_l);
128 adapter->flags |= IGC_FLAG_TSN_QBV_ENABLED;
133 int igc_tsn_offload_apply(struct igc_adapter *adapter)
135 bool is_any_enabled = adapter->base_time || is_any_launchtime(adapter);
137 if (!(adapter->flags & IGC_FLAG_TSN_QBV_ENABLED) && !is_any_enabled)
140 if (!is_any_enabled) {
141 int err = igc_tsn_disable_offload(adapter);
146 /* The BASET registers aren't cleared when writing
147 * into them, force a reset if the interface is
150 if (netif_running(adapter->netdev))
151 schedule_work(&adapter->reset_task);
156 return igc_tsn_enable_offload(adapter);