1c24eef28SGrygorii Strashko // SPDX-License-Identifier: GPL-2.0
2c24eef28SGrygorii Strashko /*
3c24eef28SGrygorii Strashko * Texas Instruments Ethernet Switch Driver ethtool intf
4c24eef28SGrygorii Strashko *
5c24eef28SGrygorii Strashko * Copyright (C) 2019 Texas Instruments
6c24eef28SGrygorii Strashko */
7c24eef28SGrygorii Strashko
8c24eef28SGrygorii Strashko #include <linux/if_ether.h>
9c24eef28SGrygorii Strashko #include <linux/if_vlan.h>
10c24eef28SGrygorii Strashko #include <linux/kmemleak.h>
11c24eef28SGrygorii Strashko #include <linux/module.h>
12c24eef28SGrygorii Strashko #include <linux/netdevice.h>
13c24eef28SGrygorii Strashko #include <linux/net_tstamp.h>
14c24eef28SGrygorii Strashko #include <linux/phy.h>
15c24eef28SGrygorii Strashko #include <linux/pm_runtime.h>
16c24eef28SGrygorii Strashko #include <linux/skbuff.h>
17c24eef28SGrygorii Strashko
18c24eef28SGrygorii Strashko #include "cpsw.h"
19c24eef28SGrygorii Strashko #include "cpts.h"
20c24eef28SGrygorii Strashko #include "cpsw_ale.h"
21c24eef28SGrygorii Strashko #include "cpsw_priv.h"
22c24eef28SGrygorii Strashko #include "davinci_cpdma.h"
23c24eef28SGrygorii Strashko
24c24eef28SGrygorii Strashko struct cpsw_hw_stats {
25c24eef28SGrygorii Strashko u32 rxgoodframes;
26c24eef28SGrygorii Strashko u32 rxbroadcastframes;
27c24eef28SGrygorii Strashko u32 rxmulticastframes;
28c24eef28SGrygorii Strashko u32 rxpauseframes;
29c24eef28SGrygorii Strashko u32 rxcrcerrors;
30c24eef28SGrygorii Strashko u32 rxaligncodeerrors;
31c24eef28SGrygorii Strashko u32 rxoversizedframes;
32c24eef28SGrygorii Strashko u32 rxjabberframes;
33c24eef28SGrygorii Strashko u32 rxundersizedframes;
34c24eef28SGrygorii Strashko u32 rxfragments;
35c24eef28SGrygorii Strashko u32 __pad_0[2];
36c24eef28SGrygorii Strashko u32 rxoctets;
37c24eef28SGrygorii Strashko u32 txgoodframes;
38c24eef28SGrygorii Strashko u32 txbroadcastframes;
39c24eef28SGrygorii Strashko u32 txmulticastframes;
40c24eef28SGrygorii Strashko u32 txpauseframes;
41c24eef28SGrygorii Strashko u32 txdeferredframes;
42c24eef28SGrygorii Strashko u32 txcollisionframes;
43c24eef28SGrygorii Strashko u32 txsinglecollframes;
44c24eef28SGrygorii Strashko u32 txmultcollframes;
45c24eef28SGrygorii Strashko u32 txexcessivecollisions;
46c24eef28SGrygorii Strashko u32 txlatecollisions;
47c24eef28SGrygorii Strashko u32 txunderrun;
48c24eef28SGrygorii Strashko u32 txcarriersenseerrors;
49c24eef28SGrygorii Strashko u32 txoctets;
50c24eef28SGrygorii Strashko u32 octetframes64;
51c24eef28SGrygorii Strashko u32 octetframes65t127;
52c24eef28SGrygorii Strashko u32 octetframes128t255;
53c24eef28SGrygorii Strashko u32 octetframes256t511;
54c24eef28SGrygorii Strashko u32 octetframes512t1023;
55c24eef28SGrygorii Strashko u32 octetframes1024tup;
56c24eef28SGrygorii Strashko u32 netoctets;
57c24eef28SGrygorii Strashko u32 rxsofoverruns;
58c24eef28SGrygorii Strashko u32 rxmofoverruns;
59c24eef28SGrygorii Strashko u32 rxdmaoverruns;
60c24eef28SGrygorii Strashko };
61c24eef28SGrygorii Strashko
62c24eef28SGrygorii Strashko struct cpsw_stats {
63c24eef28SGrygorii Strashko char stat_string[ETH_GSTRING_LEN];
64c24eef28SGrygorii Strashko int type;
65c24eef28SGrygorii Strashko int sizeof_stat;
66c24eef28SGrygorii Strashko int stat_offset;
67c24eef28SGrygorii Strashko };
68c24eef28SGrygorii Strashko
69c24eef28SGrygorii Strashko enum {
70c24eef28SGrygorii Strashko CPSW_STATS,
71c24eef28SGrygorii Strashko CPDMA_RX_STATS,
72c24eef28SGrygorii Strashko CPDMA_TX_STATS,
73c24eef28SGrygorii Strashko };
74c24eef28SGrygorii Strashko
75c24eef28SGrygorii Strashko #define CPSW_STAT(m) CPSW_STATS, \
76c593642cSPankaj Bharadiya sizeof_field(struct cpsw_hw_stats, m), \
77c24eef28SGrygorii Strashko offsetof(struct cpsw_hw_stats, m)
78c24eef28SGrygorii Strashko #define CPDMA_RX_STAT(m) CPDMA_RX_STATS, \
79c593642cSPankaj Bharadiya sizeof_field(struct cpdma_chan_stats, m), \
80c24eef28SGrygorii Strashko offsetof(struct cpdma_chan_stats, m)
81c24eef28SGrygorii Strashko #define CPDMA_TX_STAT(m) CPDMA_TX_STATS, \
82c593642cSPankaj Bharadiya sizeof_field(struct cpdma_chan_stats, m), \
83c24eef28SGrygorii Strashko offsetof(struct cpdma_chan_stats, m)
84c24eef28SGrygorii Strashko
85c24eef28SGrygorii Strashko static const struct cpsw_stats cpsw_gstrings_stats[] = {
86c24eef28SGrygorii Strashko { "Good Rx Frames", CPSW_STAT(rxgoodframes) },
87c24eef28SGrygorii Strashko { "Broadcast Rx Frames", CPSW_STAT(rxbroadcastframes) },
88c24eef28SGrygorii Strashko { "Multicast Rx Frames", CPSW_STAT(rxmulticastframes) },
89c24eef28SGrygorii Strashko { "Pause Rx Frames", CPSW_STAT(rxpauseframes) },
90c24eef28SGrygorii Strashko { "Rx CRC Errors", CPSW_STAT(rxcrcerrors) },
91c24eef28SGrygorii Strashko { "Rx Align/Code Errors", CPSW_STAT(rxaligncodeerrors) },
92c24eef28SGrygorii Strashko { "Oversize Rx Frames", CPSW_STAT(rxoversizedframes) },
93c24eef28SGrygorii Strashko { "Rx Jabbers", CPSW_STAT(rxjabberframes) },
94c24eef28SGrygorii Strashko { "Undersize (Short) Rx Frames", CPSW_STAT(rxundersizedframes) },
95c24eef28SGrygorii Strashko { "Rx Fragments", CPSW_STAT(rxfragments) },
96c24eef28SGrygorii Strashko { "Rx Octets", CPSW_STAT(rxoctets) },
97c24eef28SGrygorii Strashko { "Good Tx Frames", CPSW_STAT(txgoodframes) },
98c24eef28SGrygorii Strashko { "Broadcast Tx Frames", CPSW_STAT(txbroadcastframes) },
99c24eef28SGrygorii Strashko { "Multicast Tx Frames", CPSW_STAT(txmulticastframes) },
100c24eef28SGrygorii Strashko { "Pause Tx Frames", CPSW_STAT(txpauseframes) },
101c24eef28SGrygorii Strashko { "Deferred Tx Frames", CPSW_STAT(txdeferredframes) },
102c24eef28SGrygorii Strashko { "Collisions", CPSW_STAT(txcollisionframes) },
103c24eef28SGrygorii Strashko { "Single Collision Tx Frames", CPSW_STAT(txsinglecollframes) },
104c24eef28SGrygorii Strashko { "Multiple Collision Tx Frames", CPSW_STAT(txmultcollframes) },
105c24eef28SGrygorii Strashko { "Excessive Collisions", CPSW_STAT(txexcessivecollisions) },
106c24eef28SGrygorii Strashko { "Late Collisions", CPSW_STAT(txlatecollisions) },
107c24eef28SGrygorii Strashko { "Tx Underrun", CPSW_STAT(txunderrun) },
108c24eef28SGrygorii Strashko { "Carrier Sense Errors", CPSW_STAT(txcarriersenseerrors) },
109c24eef28SGrygorii Strashko { "Tx Octets", CPSW_STAT(txoctets) },
110c24eef28SGrygorii Strashko { "Rx + Tx 64 Octet Frames", CPSW_STAT(octetframes64) },
111c24eef28SGrygorii Strashko { "Rx + Tx 65-127 Octet Frames", CPSW_STAT(octetframes65t127) },
112c24eef28SGrygorii Strashko { "Rx + Tx 128-255 Octet Frames", CPSW_STAT(octetframes128t255) },
113c24eef28SGrygorii Strashko { "Rx + Tx 256-511 Octet Frames", CPSW_STAT(octetframes256t511) },
114c24eef28SGrygorii Strashko { "Rx + Tx 512-1023 Octet Frames", CPSW_STAT(octetframes512t1023) },
115c24eef28SGrygorii Strashko { "Rx + Tx 1024-Up Octet Frames", CPSW_STAT(octetframes1024tup) },
116c24eef28SGrygorii Strashko { "Net Octets", CPSW_STAT(netoctets) },
117c24eef28SGrygorii Strashko { "Rx Start of Frame Overruns", CPSW_STAT(rxsofoverruns) },
118c24eef28SGrygorii Strashko { "Rx Middle of Frame Overruns", CPSW_STAT(rxmofoverruns) },
119c24eef28SGrygorii Strashko { "Rx DMA Overruns", CPSW_STAT(rxdmaoverruns) },
120c24eef28SGrygorii Strashko };
121c24eef28SGrygorii Strashko
122c24eef28SGrygorii Strashko static const struct cpsw_stats cpsw_gstrings_ch_stats[] = {
123c24eef28SGrygorii Strashko { "head_enqueue", CPDMA_RX_STAT(head_enqueue) },
124c24eef28SGrygorii Strashko { "tail_enqueue", CPDMA_RX_STAT(tail_enqueue) },
125c24eef28SGrygorii Strashko { "pad_enqueue", CPDMA_RX_STAT(pad_enqueue) },
126c24eef28SGrygorii Strashko { "misqueued", CPDMA_RX_STAT(misqueued) },
127c24eef28SGrygorii Strashko { "desc_alloc_fail", CPDMA_RX_STAT(desc_alloc_fail) },
128c24eef28SGrygorii Strashko { "pad_alloc_fail", CPDMA_RX_STAT(pad_alloc_fail) },
129c24eef28SGrygorii Strashko { "runt_receive_buf", CPDMA_RX_STAT(runt_receive_buff) },
130c24eef28SGrygorii Strashko { "runt_transmit_buf", CPDMA_RX_STAT(runt_transmit_buff) },
131c24eef28SGrygorii Strashko { "empty_dequeue", CPDMA_RX_STAT(empty_dequeue) },
132c24eef28SGrygorii Strashko { "busy_dequeue", CPDMA_RX_STAT(busy_dequeue) },
133c24eef28SGrygorii Strashko { "good_dequeue", CPDMA_RX_STAT(good_dequeue) },
134c24eef28SGrygorii Strashko { "requeue", CPDMA_RX_STAT(requeue) },
135c24eef28SGrygorii Strashko { "teardown_dequeue", CPDMA_RX_STAT(teardown_dequeue) },
136c24eef28SGrygorii Strashko };
137c24eef28SGrygorii Strashko
138c24eef28SGrygorii Strashko #define CPSW_STATS_COMMON_LEN ARRAY_SIZE(cpsw_gstrings_stats)
139c24eef28SGrygorii Strashko #define CPSW_STATS_CH_LEN ARRAY_SIZE(cpsw_gstrings_ch_stats)
140c24eef28SGrygorii Strashko
cpsw_get_msglevel(struct net_device * ndev)141c24eef28SGrygorii Strashko u32 cpsw_get_msglevel(struct net_device *ndev)
142c24eef28SGrygorii Strashko {
143c24eef28SGrygorii Strashko struct cpsw_priv *priv = netdev_priv(ndev);
144c24eef28SGrygorii Strashko
145c24eef28SGrygorii Strashko return priv->msg_enable;
146c24eef28SGrygorii Strashko }
147c24eef28SGrygorii Strashko
cpsw_set_msglevel(struct net_device * ndev,u32 value)148c24eef28SGrygorii Strashko void cpsw_set_msglevel(struct net_device *ndev, u32 value)
149c24eef28SGrygorii Strashko {
150c24eef28SGrygorii Strashko struct cpsw_priv *priv = netdev_priv(ndev);
151c24eef28SGrygorii Strashko
152c24eef28SGrygorii Strashko priv->msg_enable = value;
153c24eef28SGrygorii Strashko }
154c24eef28SGrygorii Strashko
cpsw_get_coalesce(struct net_device * ndev,struct ethtool_coalesce * coal,struct kernel_ethtool_coalesce * kernel_coal,struct netlink_ext_ack * extack)155f3ccfda1SYufeng Mo int cpsw_get_coalesce(struct net_device *ndev, struct ethtool_coalesce *coal,
156f3ccfda1SYufeng Mo struct kernel_ethtool_coalesce *kernel_coal,
157f3ccfda1SYufeng Mo struct netlink_ext_ack *extack)
158c24eef28SGrygorii Strashko {
159c24eef28SGrygorii Strashko struct cpsw_common *cpsw = ndev_to_cpsw(ndev);
160c24eef28SGrygorii Strashko
161c24eef28SGrygorii Strashko coal->rx_coalesce_usecs = cpsw->coal_intvl;
162c24eef28SGrygorii Strashko return 0;
163c24eef28SGrygorii Strashko }
164c24eef28SGrygorii Strashko
cpsw_set_coalesce(struct net_device * ndev,struct ethtool_coalesce * coal,struct kernel_ethtool_coalesce * kernel_coal,struct netlink_ext_ack * extack)165f3ccfda1SYufeng Mo int cpsw_set_coalesce(struct net_device *ndev, struct ethtool_coalesce *coal,
166f3ccfda1SYufeng Mo struct kernel_ethtool_coalesce *kernel_coal,
167f3ccfda1SYufeng Mo struct netlink_ext_ack *extack)
168c24eef28SGrygorii Strashko {
169c24eef28SGrygorii Strashko struct cpsw_priv *priv = netdev_priv(ndev);
170c24eef28SGrygorii Strashko u32 int_ctrl;
171c24eef28SGrygorii Strashko u32 num_interrupts = 0;
172c24eef28SGrygorii Strashko u32 prescale = 0;
173c24eef28SGrygorii Strashko u32 addnl_dvdr = 1;
174c24eef28SGrygorii Strashko u32 coal_intvl = 0;
175c24eef28SGrygorii Strashko struct cpsw_common *cpsw = priv->cpsw;
176c24eef28SGrygorii Strashko
177c24eef28SGrygorii Strashko coal_intvl = coal->rx_coalesce_usecs;
178c24eef28SGrygorii Strashko
179c24eef28SGrygorii Strashko int_ctrl = readl(&cpsw->wr_regs->int_control);
180c24eef28SGrygorii Strashko prescale = cpsw->bus_freq_mhz * 4;
181c24eef28SGrygorii Strashko
182c24eef28SGrygorii Strashko if (!coal->rx_coalesce_usecs) {
183c24eef28SGrygorii Strashko int_ctrl &= ~(CPSW_INTPRESCALE_MASK | CPSW_INTPACEEN);
184c24eef28SGrygorii Strashko goto update_return;
185c24eef28SGrygorii Strashko }
186c24eef28SGrygorii Strashko
187c24eef28SGrygorii Strashko if (coal_intvl < CPSW_CMINTMIN_INTVL)
188c24eef28SGrygorii Strashko coal_intvl = CPSW_CMINTMIN_INTVL;
189c24eef28SGrygorii Strashko
190c24eef28SGrygorii Strashko if (coal_intvl > CPSW_CMINTMAX_INTVL) {
191c24eef28SGrygorii Strashko /* Interrupt pacer works with 4us Pulse, we can
192c24eef28SGrygorii Strashko * throttle further by dilating the 4us pulse.
193c24eef28SGrygorii Strashko */
194c24eef28SGrygorii Strashko addnl_dvdr = CPSW_INTPRESCALE_MASK / prescale;
195c24eef28SGrygorii Strashko
196c24eef28SGrygorii Strashko if (addnl_dvdr > 1) {
197c24eef28SGrygorii Strashko prescale *= addnl_dvdr;
198c24eef28SGrygorii Strashko if (coal_intvl > (CPSW_CMINTMAX_INTVL * addnl_dvdr))
199c24eef28SGrygorii Strashko coal_intvl = (CPSW_CMINTMAX_INTVL
200c24eef28SGrygorii Strashko * addnl_dvdr);
201c24eef28SGrygorii Strashko } else {
202c24eef28SGrygorii Strashko addnl_dvdr = 1;
203c24eef28SGrygorii Strashko coal_intvl = CPSW_CMINTMAX_INTVL;
204c24eef28SGrygorii Strashko }
205c24eef28SGrygorii Strashko }
206c24eef28SGrygorii Strashko
207c24eef28SGrygorii Strashko num_interrupts = (1000 * addnl_dvdr) / coal_intvl;
208c24eef28SGrygorii Strashko writel(num_interrupts, &cpsw->wr_regs->rx_imax);
209c24eef28SGrygorii Strashko writel(num_interrupts, &cpsw->wr_regs->tx_imax);
210c24eef28SGrygorii Strashko
211c24eef28SGrygorii Strashko int_ctrl |= CPSW_INTPACEEN;
212c24eef28SGrygorii Strashko int_ctrl &= (~CPSW_INTPRESCALE_MASK);
213c24eef28SGrygorii Strashko int_ctrl |= (prescale & CPSW_INTPRESCALE_MASK);
214c24eef28SGrygorii Strashko
215c24eef28SGrygorii Strashko update_return:
216c24eef28SGrygorii Strashko writel(int_ctrl, &cpsw->wr_regs->int_control);
217c24eef28SGrygorii Strashko
218c24eef28SGrygorii Strashko cpsw_notice(priv, timer, "Set coalesce to %d usecs.\n", coal_intvl);
219c24eef28SGrygorii Strashko cpsw->coal_intvl = coal_intvl;
220c24eef28SGrygorii Strashko
221c24eef28SGrygorii Strashko return 0;
222c24eef28SGrygorii Strashko }
223c24eef28SGrygorii Strashko
cpsw_get_sset_count(struct net_device * ndev,int sset)224c24eef28SGrygorii Strashko int cpsw_get_sset_count(struct net_device *ndev, int sset)
225c24eef28SGrygorii Strashko {
226c24eef28SGrygorii Strashko struct cpsw_common *cpsw = ndev_to_cpsw(ndev);
227c24eef28SGrygorii Strashko
228c24eef28SGrygorii Strashko switch (sset) {
229c24eef28SGrygorii Strashko case ETH_SS_STATS:
230c24eef28SGrygorii Strashko return (CPSW_STATS_COMMON_LEN +
231c24eef28SGrygorii Strashko (cpsw->rx_ch_num + cpsw->tx_ch_num) *
232c24eef28SGrygorii Strashko CPSW_STATS_CH_LEN);
233c24eef28SGrygorii Strashko default:
234c24eef28SGrygorii Strashko return -EOPNOTSUPP;
235c24eef28SGrygorii Strashko }
236c24eef28SGrygorii Strashko }
237c24eef28SGrygorii Strashko
cpsw_add_ch_strings(u8 ** p,int ch_num,int rx_dir)238c24eef28SGrygorii Strashko static void cpsw_add_ch_strings(u8 **p, int ch_num, int rx_dir)
239c24eef28SGrygorii Strashko {
240c24eef28SGrygorii Strashko int ch_stats_len;
241c24eef28SGrygorii Strashko int line;
242c24eef28SGrygorii Strashko int i;
243c24eef28SGrygorii Strashko
244c24eef28SGrygorii Strashko ch_stats_len = CPSW_STATS_CH_LEN * ch_num;
245c24eef28SGrygorii Strashko for (i = 0; i < ch_stats_len; i++) {
246c24eef28SGrygorii Strashko line = i % CPSW_STATS_CH_LEN;
247c24eef28SGrygorii Strashko snprintf(*p, ETH_GSTRING_LEN,
248c24eef28SGrygorii Strashko "%s DMA chan %ld: %s", rx_dir ? "Rx" : "Tx",
249c24eef28SGrygorii Strashko (long)(i / CPSW_STATS_CH_LEN),
250c24eef28SGrygorii Strashko cpsw_gstrings_ch_stats[line].stat_string);
251c24eef28SGrygorii Strashko *p += ETH_GSTRING_LEN;
252c24eef28SGrygorii Strashko }
253c24eef28SGrygorii Strashko }
254c24eef28SGrygorii Strashko
cpsw_get_strings(struct net_device * ndev,u32 stringset,u8 * data)255c24eef28SGrygorii Strashko void cpsw_get_strings(struct net_device *ndev, u32 stringset, u8 *data)
256c24eef28SGrygorii Strashko {
257c24eef28SGrygorii Strashko struct cpsw_common *cpsw = ndev_to_cpsw(ndev);
258c24eef28SGrygorii Strashko u8 *p = data;
259c24eef28SGrygorii Strashko int i;
260c24eef28SGrygorii Strashko
261c24eef28SGrygorii Strashko switch (stringset) {
262c24eef28SGrygorii Strashko case ETH_SS_STATS:
263c24eef28SGrygorii Strashko for (i = 0; i < CPSW_STATS_COMMON_LEN; i++) {
264c24eef28SGrygorii Strashko memcpy(p, cpsw_gstrings_stats[i].stat_string,
265c24eef28SGrygorii Strashko ETH_GSTRING_LEN);
266c24eef28SGrygorii Strashko p += ETH_GSTRING_LEN;
267c24eef28SGrygorii Strashko }
268c24eef28SGrygorii Strashko
269c24eef28SGrygorii Strashko cpsw_add_ch_strings(&p, cpsw->rx_ch_num, 1);
270c24eef28SGrygorii Strashko cpsw_add_ch_strings(&p, cpsw->tx_ch_num, 0);
271c24eef28SGrygorii Strashko break;
272c24eef28SGrygorii Strashko }
273c24eef28SGrygorii Strashko }
274c24eef28SGrygorii Strashko
cpsw_get_ethtool_stats(struct net_device * ndev,struct ethtool_stats * stats,u64 * data)275c24eef28SGrygorii Strashko void cpsw_get_ethtool_stats(struct net_device *ndev,
276c24eef28SGrygorii Strashko struct ethtool_stats *stats, u64 *data)
277c24eef28SGrygorii Strashko {
278c24eef28SGrygorii Strashko u8 *p;
279c24eef28SGrygorii Strashko struct cpsw_common *cpsw = ndev_to_cpsw(ndev);
280c24eef28SGrygorii Strashko struct cpdma_chan_stats ch_stats;
281c24eef28SGrygorii Strashko int i, l, ch;
282c24eef28SGrygorii Strashko
283c24eef28SGrygorii Strashko /* Collect Davinci CPDMA stats for Rx and Tx Channel */
284c24eef28SGrygorii Strashko for (l = 0; l < CPSW_STATS_COMMON_LEN; l++)
285c24eef28SGrygorii Strashko data[l] = readl(cpsw->hw_stats +
286c24eef28SGrygorii Strashko cpsw_gstrings_stats[l].stat_offset);
287c24eef28SGrygorii Strashko
288c24eef28SGrygorii Strashko for (ch = 0; ch < cpsw->rx_ch_num; ch++) {
289c24eef28SGrygorii Strashko cpdma_chan_get_stats(cpsw->rxv[ch].ch, &ch_stats);
290c24eef28SGrygorii Strashko for (i = 0; i < CPSW_STATS_CH_LEN; i++, l++) {
291c24eef28SGrygorii Strashko p = (u8 *)&ch_stats +
292c24eef28SGrygorii Strashko cpsw_gstrings_ch_stats[i].stat_offset;
293c24eef28SGrygorii Strashko data[l] = *(u32 *)p;
294c24eef28SGrygorii Strashko }
295c24eef28SGrygorii Strashko }
296c24eef28SGrygorii Strashko
297c24eef28SGrygorii Strashko for (ch = 0; ch < cpsw->tx_ch_num; ch++) {
298c24eef28SGrygorii Strashko cpdma_chan_get_stats(cpsw->txv[ch].ch, &ch_stats);
299c24eef28SGrygorii Strashko for (i = 0; i < CPSW_STATS_CH_LEN; i++, l++) {
300c24eef28SGrygorii Strashko p = (u8 *)&ch_stats +
301c24eef28SGrygorii Strashko cpsw_gstrings_ch_stats[i].stat_offset;
302c24eef28SGrygorii Strashko data[l] = *(u32 *)p;
303c24eef28SGrygorii Strashko }
304c24eef28SGrygorii Strashko }
305c24eef28SGrygorii Strashko }
306c24eef28SGrygorii Strashko
cpsw_get_pauseparam(struct net_device * ndev,struct ethtool_pauseparam * pause)307c24eef28SGrygorii Strashko void cpsw_get_pauseparam(struct net_device *ndev,
308c24eef28SGrygorii Strashko struct ethtool_pauseparam *pause)
309c24eef28SGrygorii Strashko {
310c24eef28SGrygorii Strashko struct cpsw_priv *priv = netdev_priv(ndev);
311c24eef28SGrygorii Strashko
312c24eef28SGrygorii Strashko pause->autoneg = AUTONEG_DISABLE;
313c24eef28SGrygorii Strashko pause->rx_pause = priv->rx_pause ? true : false;
314c24eef28SGrygorii Strashko pause->tx_pause = priv->tx_pause ? true : false;
315c24eef28SGrygorii Strashko }
316c24eef28SGrygorii Strashko
cpsw_get_wol(struct net_device * ndev,struct ethtool_wolinfo * wol)317c24eef28SGrygorii Strashko void cpsw_get_wol(struct net_device *ndev, struct ethtool_wolinfo *wol)
318c24eef28SGrygorii Strashko {
319c24eef28SGrygorii Strashko struct cpsw_priv *priv = netdev_priv(ndev);
320c24eef28SGrygorii Strashko struct cpsw_common *cpsw = priv->cpsw;
321c24eef28SGrygorii Strashko int slave_no = cpsw_slave_index(cpsw, priv);
322c24eef28SGrygorii Strashko
323c24eef28SGrygorii Strashko wol->supported = 0;
324c24eef28SGrygorii Strashko wol->wolopts = 0;
325c24eef28SGrygorii Strashko
326c24eef28SGrygorii Strashko if (cpsw->slaves[slave_no].phy)
327c24eef28SGrygorii Strashko phy_ethtool_get_wol(cpsw->slaves[slave_no].phy, wol);
328c24eef28SGrygorii Strashko }
329c24eef28SGrygorii Strashko
cpsw_set_wol(struct net_device * ndev,struct ethtool_wolinfo * wol)330c24eef28SGrygorii Strashko int cpsw_set_wol(struct net_device *ndev, struct ethtool_wolinfo *wol)
331c24eef28SGrygorii Strashko {
332c24eef28SGrygorii Strashko struct cpsw_priv *priv = netdev_priv(ndev);
333c24eef28SGrygorii Strashko struct cpsw_common *cpsw = priv->cpsw;
334c24eef28SGrygorii Strashko int slave_no = cpsw_slave_index(cpsw, priv);
335c24eef28SGrygorii Strashko
336c24eef28SGrygorii Strashko if (cpsw->slaves[slave_no].phy)
337c24eef28SGrygorii Strashko return phy_ethtool_set_wol(cpsw->slaves[slave_no].phy, wol);
338c24eef28SGrygorii Strashko else
339c24eef28SGrygorii Strashko return -EOPNOTSUPP;
340c24eef28SGrygorii Strashko }
341c24eef28SGrygorii Strashko
cpsw_get_regs_len(struct net_device * ndev)342c24eef28SGrygorii Strashko int cpsw_get_regs_len(struct net_device *ndev)
343c24eef28SGrygorii Strashko {
344c24eef28SGrygorii Strashko struct cpsw_common *cpsw = ndev_to_cpsw(ndev);
345c24eef28SGrygorii Strashko
346b574bf0cSGrygorii Strashko return cpsw_ale_get_num_entries(cpsw->ale) *
347b574bf0cSGrygorii Strashko ALE_ENTRY_WORDS * sizeof(u32);
348c24eef28SGrygorii Strashko }
349c24eef28SGrygorii Strashko
cpsw_get_regs(struct net_device * ndev,struct ethtool_regs * regs,void * p)350c24eef28SGrygorii Strashko void cpsw_get_regs(struct net_device *ndev, struct ethtool_regs *regs, void *p)
351c24eef28SGrygorii Strashko {
352c24eef28SGrygorii Strashko u32 *reg = p;
353c24eef28SGrygorii Strashko struct cpsw_common *cpsw = ndev_to_cpsw(ndev);
354c24eef28SGrygorii Strashko
355c24eef28SGrygorii Strashko /* update CPSW IP version */
356c24eef28SGrygorii Strashko regs->version = cpsw->version;
357c24eef28SGrygorii Strashko
358c24eef28SGrygorii Strashko cpsw_ale_dump(cpsw->ale, reg);
359c24eef28SGrygorii Strashko }
360c24eef28SGrygorii Strashko
cpsw_ethtool_op_begin(struct net_device * ndev)361c24eef28SGrygorii Strashko int cpsw_ethtool_op_begin(struct net_device *ndev)
362c24eef28SGrygorii Strashko {
363c24eef28SGrygorii Strashko struct cpsw_priv *priv = netdev_priv(ndev);
364c24eef28SGrygorii Strashko struct cpsw_common *cpsw = priv->cpsw;
365c24eef28SGrygorii Strashko int ret;
366c24eef28SGrygorii Strashko
367*2844e243SSondhauß, Jan ret = pm_runtime_resume_and_get(cpsw->dev);
368*2844e243SSondhauß, Jan if (ret < 0)
369c24eef28SGrygorii Strashko cpsw_err(priv, drv, "ethtool begin failed %d\n", ret);
370c24eef28SGrygorii Strashko
371c24eef28SGrygorii Strashko return ret;
372c24eef28SGrygorii Strashko }
373c24eef28SGrygorii Strashko
cpsw_ethtool_op_complete(struct net_device * ndev)374c24eef28SGrygorii Strashko void cpsw_ethtool_op_complete(struct net_device *ndev)
375c24eef28SGrygorii Strashko {
376c24eef28SGrygorii Strashko struct cpsw_priv *priv = netdev_priv(ndev);
377c24eef28SGrygorii Strashko int ret;
378c24eef28SGrygorii Strashko
379c24eef28SGrygorii Strashko ret = pm_runtime_put(priv->cpsw->dev);
380c24eef28SGrygorii Strashko if (ret < 0)
381c24eef28SGrygorii Strashko cpsw_err(priv, drv, "ethtool complete failed %d\n", ret);
382c24eef28SGrygorii Strashko }
383c24eef28SGrygorii Strashko
cpsw_get_channels(struct net_device * ndev,struct ethtool_channels * ch)384c24eef28SGrygorii Strashko void cpsw_get_channels(struct net_device *ndev, struct ethtool_channels *ch)
385c24eef28SGrygorii Strashko {
386c24eef28SGrygorii Strashko struct cpsw_common *cpsw = ndev_to_cpsw(ndev);
387c24eef28SGrygorii Strashko
388c24eef28SGrygorii Strashko ch->max_rx = cpsw->quirk_irq ? 1 : CPSW_MAX_QUEUES;
389c24eef28SGrygorii Strashko ch->max_tx = cpsw->quirk_irq ? 1 : CPSW_MAX_QUEUES;
390c24eef28SGrygorii Strashko ch->max_combined = 0;
391c24eef28SGrygorii Strashko ch->max_other = 0;
392c24eef28SGrygorii Strashko ch->other_count = 0;
393c24eef28SGrygorii Strashko ch->rx_count = cpsw->rx_ch_num;
394c24eef28SGrygorii Strashko ch->tx_count = cpsw->tx_ch_num;
395c24eef28SGrygorii Strashko ch->combined_count = 0;
396c24eef28SGrygorii Strashko }
397c24eef28SGrygorii Strashko
cpsw_get_link_ksettings(struct net_device * ndev,struct ethtool_link_ksettings * ecmd)398c24eef28SGrygorii Strashko int cpsw_get_link_ksettings(struct net_device *ndev,
399c24eef28SGrygorii Strashko struct ethtool_link_ksettings *ecmd)
400c24eef28SGrygorii Strashko {
401c24eef28SGrygorii Strashko struct cpsw_priv *priv = netdev_priv(ndev);
402c24eef28SGrygorii Strashko struct cpsw_common *cpsw = priv->cpsw;
403c24eef28SGrygorii Strashko int slave_no = cpsw_slave_index(cpsw, priv);
404c24eef28SGrygorii Strashko
405c24eef28SGrygorii Strashko if (!cpsw->slaves[slave_no].phy)
406c24eef28SGrygorii Strashko return -EOPNOTSUPP;
407c24eef28SGrygorii Strashko
408c24eef28SGrygorii Strashko phy_ethtool_ksettings_get(cpsw->slaves[slave_no].phy, ecmd);
409c24eef28SGrygorii Strashko return 0;
410c24eef28SGrygorii Strashko }
411c24eef28SGrygorii Strashko
cpsw_set_link_ksettings(struct net_device * ndev,const struct ethtool_link_ksettings * ecmd)412c24eef28SGrygorii Strashko int cpsw_set_link_ksettings(struct net_device *ndev,
413c24eef28SGrygorii Strashko const struct ethtool_link_ksettings *ecmd)
414c24eef28SGrygorii Strashko {
415c24eef28SGrygorii Strashko struct cpsw_priv *priv = netdev_priv(ndev);
416c24eef28SGrygorii Strashko struct cpsw_common *cpsw = priv->cpsw;
417c24eef28SGrygorii Strashko int slave_no = cpsw_slave_index(cpsw, priv);
418c24eef28SGrygorii Strashko
419c24eef28SGrygorii Strashko if (!cpsw->slaves[slave_no].phy)
420c24eef28SGrygorii Strashko return -EOPNOTSUPP;
421c24eef28SGrygorii Strashko
422c24eef28SGrygorii Strashko return phy_ethtool_ksettings_set(cpsw->slaves[slave_no].phy, ecmd);
423c24eef28SGrygorii Strashko }
424c24eef28SGrygorii Strashko
cpsw_get_eee(struct net_device * ndev,struct ethtool_eee * edata)425c24eef28SGrygorii Strashko int cpsw_get_eee(struct net_device *ndev, struct ethtool_eee *edata)
426c24eef28SGrygorii Strashko {
427c24eef28SGrygorii Strashko struct cpsw_priv *priv = netdev_priv(ndev);
428c24eef28SGrygorii Strashko struct cpsw_common *cpsw = priv->cpsw;
429c24eef28SGrygorii Strashko int slave_no = cpsw_slave_index(cpsw, priv);
430c24eef28SGrygorii Strashko
431c24eef28SGrygorii Strashko if (cpsw->slaves[slave_no].phy)
432c24eef28SGrygorii Strashko return phy_ethtool_get_eee(cpsw->slaves[slave_no].phy, edata);
433c24eef28SGrygorii Strashko else
434c24eef28SGrygorii Strashko return -EOPNOTSUPP;
435c24eef28SGrygorii Strashko }
436c24eef28SGrygorii Strashko
cpsw_set_eee(struct net_device * ndev,struct ethtool_eee * edata)437c24eef28SGrygorii Strashko int cpsw_set_eee(struct net_device *ndev, struct ethtool_eee *edata)
438c24eef28SGrygorii Strashko {
439c24eef28SGrygorii Strashko struct cpsw_priv *priv = netdev_priv(ndev);
440c24eef28SGrygorii Strashko struct cpsw_common *cpsw = priv->cpsw;
441c24eef28SGrygorii Strashko int slave_no = cpsw_slave_index(cpsw, priv);
442c24eef28SGrygorii Strashko
443c24eef28SGrygorii Strashko if (cpsw->slaves[slave_no].phy)
444c24eef28SGrygorii Strashko return phy_ethtool_set_eee(cpsw->slaves[slave_no].phy, edata);
445c24eef28SGrygorii Strashko else
446c24eef28SGrygorii Strashko return -EOPNOTSUPP;
447c24eef28SGrygorii Strashko }
448c24eef28SGrygorii Strashko
cpsw_nway_reset(struct net_device * ndev)449c24eef28SGrygorii Strashko int cpsw_nway_reset(struct net_device *ndev)
450c24eef28SGrygorii Strashko {
451c24eef28SGrygorii Strashko struct cpsw_priv *priv = netdev_priv(ndev);
452c24eef28SGrygorii Strashko struct cpsw_common *cpsw = priv->cpsw;
453c24eef28SGrygorii Strashko int slave_no = cpsw_slave_index(cpsw, priv);
454c24eef28SGrygorii Strashko
455c24eef28SGrygorii Strashko if (cpsw->slaves[slave_no].phy)
456c24eef28SGrygorii Strashko return genphy_restart_aneg(cpsw->slaves[slave_no].phy);
457c24eef28SGrygorii Strashko else
458c24eef28SGrygorii Strashko return -EOPNOTSUPP;
459c24eef28SGrygorii Strashko }
460c24eef28SGrygorii Strashko
cpsw_suspend_data_pass(struct net_device * ndev)461c24eef28SGrygorii Strashko static void cpsw_suspend_data_pass(struct net_device *ndev)
462c24eef28SGrygorii Strashko {
463c24eef28SGrygorii Strashko struct cpsw_common *cpsw = ndev_to_cpsw(ndev);
464c24eef28SGrygorii Strashko int i;
465c24eef28SGrygorii Strashko
466c24eef28SGrygorii Strashko /* Disable NAPI scheduling */
467c24eef28SGrygorii Strashko cpsw_intr_disable(cpsw);
468c24eef28SGrygorii Strashko
469c24eef28SGrygorii Strashko /* Stop all transmit queues for every network device.
470c24eef28SGrygorii Strashko */
4719126e75eSIvan Khoronzhuk for (i = 0; i < cpsw->data.slaves; i++) {
4729126e75eSIvan Khoronzhuk ndev = cpsw->slaves[i].ndev;
4739126e75eSIvan Khoronzhuk if (!(ndev && netif_running(ndev)))
474c24eef28SGrygorii Strashko continue;
475c24eef28SGrygorii Strashko
4769126e75eSIvan Khoronzhuk netif_tx_stop_all_queues(ndev);
477871e8465SIvan Khoronzhuk
478871e8465SIvan Khoronzhuk /* Barrier, so that stop_queue visible to other cpus */
479871e8465SIvan Khoronzhuk smp_mb__after_atomic();
480c24eef28SGrygorii Strashko }
481c24eef28SGrygorii Strashko
482c24eef28SGrygorii Strashko /* Handle rest of tx packets and stop cpdma channels */
483c24eef28SGrygorii Strashko cpdma_ctlr_stop(cpsw->dma);
484c24eef28SGrygorii Strashko }
485c24eef28SGrygorii Strashko
cpsw_resume_data_pass(struct net_device * ndev)486c24eef28SGrygorii Strashko static int cpsw_resume_data_pass(struct net_device *ndev)
487c24eef28SGrygorii Strashko {
488c24eef28SGrygorii Strashko struct cpsw_priv *priv = netdev_priv(ndev);
489c24eef28SGrygorii Strashko struct cpsw_common *cpsw = priv->cpsw;
490c24eef28SGrygorii Strashko int i, ret;
491c24eef28SGrygorii Strashko
492c24eef28SGrygorii Strashko /* After this receive is started */
493c24eef28SGrygorii Strashko if (cpsw->usage_count) {
494c24eef28SGrygorii Strashko ret = cpsw_fill_rx_channels(priv);
495c24eef28SGrygorii Strashko if (ret)
496c24eef28SGrygorii Strashko return ret;
497c24eef28SGrygorii Strashko
498c24eef28SGrygorii Strashko cpdma_ctlr_start(cpsw->dma);
499c24eef28SGrygorii Strashko cpsw_intr_enable(cpsw);
500c24eef28SGrygorii Strashko }
501c24eef28SGrygorii Strashko
502c24eef28SGrygorii Strashko /* Resume transmit for every affected interface */
5039126e75eSIvan Khoronzhuk for (i = 0; i < cpsw->data.slaves; i++) {
5049126e75eSIvan Khoronzhuk ndev = cpsw->slaves[i].ndev;
5059126e75eSIvan Khoronzhuk if (ndev && netif_running(ndev))
5069126e75eSIvan Khoronzhuk netif_tx_start_all_queues(ndev);
5079126e75eSIvan Khoronzhuk }
508c24eef28SGrygorii Strashko
509c24eef28SGrygorii Strashko return 0;
510c24eef28SGrygorii Strashko }
511c24eef28SGrygorii Strashko
cpsw_check_ch_settings(struct cpsw_common * cpsw,struct ethtool_channels * ch)512c24eef28SGrygorii Strashko static int cpsw_check_ch_settings(struct cpsw_common *cpsw,
513c24eef28SGrygorii Strashko struct ethtool_channels *ch)
514c24eef28SGrygorii Strashko {
515c24eef28SGrygorii Strashko if (cpsw->quirk_irq) {
516c24eef28SGrygorii Strashko dev_err(cpsw->dev, "Maximum one tx/rx queue is allowed");
517c24eef28SGrygorii Strashko return -EOPNOTSUPP;
518c24eef28SGrygorii Strashko }
519c24eef28SGrygorii Strashko
520c24eef28SGrygorii Strashko if (ch->combined_count)
521c24eef28SGrygorii Strashko return -EINVAL;
522c24eef28SGrygorii Strashko
523c24eef28SGrygorii Strashko /* verify we have at least one channel in each direction */
524c24eef28SGrygorii Strashko if (!ch->rx_count || !ch->tx_count)
525c24eef28SGrygorii Strashko return -EINVAL;
526c24eef28SGrygorii Strashko
527c24eef28SGrygorii Strashko if (ch->rx_count > cpsw->data.channels ||
528c24eef28SGrygorii Strashko ch->tx_count > cpsw->data.channels)
529c24eef28SGrygorii Strashko return -EINVAL;
530c24eef28SGrygorii Strashko
531c24eef28SGrygorii Strashko return 0;
532c24eef28SGrygorii Strashko }
533c24eef28SGrygorii Strashko
cpsw_update_channels_res(struct cpsw_priv * priv,int ch_num,int rx,cpdma_handler_fn rx_handler)534c24eef28SGrygorii Strashko static int cpsw_update_channels_res(struct cpsw_priv *priv, int ch_num, int rx,
535c24eef28SGrygorii Strashko cpdma_handler_fn rx_handler)
536c24eef28SGrygorii Strashko {
537c24eef28SGrygorii Strashko struct cpsw_common *cpsw = priv->cpsw;
538c24eef28SGrygorii Strashko void (*handler)(void *, int, int);
539c24eef28SGrygorii Strashko struct netdev_queue *queue;
540c24eef28SGrygorii Strashko struct cpsw_vector *vec;
541c24eef28SGrygorii Strashko int ret, *ch, vch;
542c24eef28SGrygorii Strashko
543c24eef28SGrygorii Strashko if (rx) {
544c24eef28SGrygorii Strashko ch = &cpsw->rx_ch_num;
545c24eef28SGrygorii Strashko vec = cpsw->rxv;
546c24eef28SGrygorii Strashko handler = rx_handler;
547c24eef28SGrygorii Strashko } else {
548c24eef28SGrygorii Strashko ch = &cpsw->tx_ch_num;
549c24eef28SGrygorii Strashko vec = cpsw->txv;
550c24eef28SGrygorii Strashko handler = cpsw_tx_handler;
551c24eef28SGrygorii Strashko }
552c24eef28SGrygorii Strashko
553c24eef28SGrygorii Strashko while (*ch < ch_num) {
554c24eef28SGrygorii Strashko vch = rx ? *ch : 7 - *ch;
555c24eef28SGrygorii Strashko vec[*ch].ch = cpdma_chan_create(cpsw->dma, vch, handler, rx);
556c24eef28SGrygorii Strashko queue = netdev_get_tx_queue(priv->ndev, *ch);
557c24eef28SGrygorii Strashko queue->tx_maxrate = 0;
558c24eef28SGrygorii Strashko
559c24eef28SGrygorii Strashko if (IS_ERR(vec[*ch].ch))
560c24eef28SGrygorii Strashko return PTR_ERR(vec[*ch].ch);
561c24eef28SGrygorii Strashko
562c24eef28SGrygorii Strashko if (!vec[*ch].ch)
563c24eef28SGrygorii Strashko return -EINVAL;
564c24eef28SGrygorii Strashko
565c24eef28SGrygorii Strashko cpsw_info(priv, ifup, "created new %d %s channel\n", *ch,
566c24eef28SGrygorii Strashko (rx ? "rx" : "tx"));
567c24eef28SGrygorii Strashko (*ch)++;
568c24eef28SGrygorii Strashko }
569c24eef28SGrygorii Strashko
570c24eef28SGrygorii Strashko while (*ch > ch_num) {
571c24eef28SGrygorii Strashko (*ch)--;
572c24eef28SGrygorii Strashko
573c24eef28SGrygorii Strashko ret = cpdma_chan_destroy(vec[*ch].ch);
574c24eef28SGrygorii Strashko if (ret)
575c24eef28SGrygorii Strashko return ret;
576c24eef28SGrygorii Strashko
577c24eef28SGrygorii Strashko cpsw_info(priv, ifup, "destroyed %d %s channel\n", *ch,
578c24eef28SGrygorii Strashko (rx ? "rx" : "tx"));
579c24eef28SGrygorii Strashko }
580c24eef28SGrygorii Strashko
581c24eef28SGrygorii Strashko return 0;
582c24eef28SGrygorii Strashko }
583c24eef28SGrygorii Strashko
cpsw_fail(struct cpsw_common * cpsw)5849ed4050cSIvan Khoronzhuk static void cpsw_fail(struct cpsw_common *cpsw)
5859ed4050cSIvan Khoronzhuk {
5869ed4050cSIvan Khoronzhuk struct net_device *ndev;
5879ed4050cSIvan Khoronzhuk int i;
5889ed4050cSIvan Khoronzhuk
5899ed4050cSIvan Khoronzhuk for (i = 0; i < cpsw->data.slaves; i++) {
5909ed4050cSIvan Khoronzhuk ndev = cpsw->slaves[i].ndev;
5919ed4050cSIvan Khoronzhuk if (ndev)
5929ed4050cSIvan Khoronzhuk dev_close(ndev);
5939ed4050cSIvan Khoronzhuk }
5949ed4050cSIvan Khoronzhuk }
5959ed4050cSIvan Khoronzhuk
cpsw_set_channels_common(struct net_device * ndev,struct ethtool_channels * chs,cpdma_handler_fn rx_handler)596c24eef28SGrygorii Strashko int cpsw_set_channels_common(struct net_device *ndev,
597c24eef28SGrygorii Strashko struct ethtool_channels *chs,
598c24eef28SGrygorii Strashko cpdma_handler_fn rx_handler)
599c24eef28SGrygorii Strashko {
600c24eef28SGrygorii Strashko struct cpsw_priv *priv = netdev_priv(ndev);
601c24eef28SGrygorii Strashko struct cpsw_common *cpsw = priv->cpsw;
6029126e75eSIvan Khoronzhuk struct net_device *sl_ndev;
6039ed4050cSIvan Khoronzhuk int i, new_pools, ret;
604c24eef28SGrygorii Strashko
605c24eef28SGrygorii Strashko ret = cpsw_check_ch_settings(cpsw, chs);
606c24eef28SGrygorii Strashko if (ret < 0)
607c24eef28SGrygorii Strashko return ret;
608c24eef28SGrygorii Strashko
609c24eef28SGrygorii Strashko cpsw_suspend_data_pass(ndev);
610c24eef28SGrygorii Strashko
6119ed4050cSIvan Khoronzhuk new_pools = (chs->rx_count != cpsw->rx_ch_num) && cpsw->usage_count;
6129ed4050cSIvan Khoronzhuk
613c24eef28SGrygorii Strashko ret = cpsw_update_channels_res(priv, chs->rx_count, 1, rx_handler);
614c24eef28SGrygorii Strashko if (ret)
615c24eef28SGrygorii Strashko goto err;
616c24eef28SGrygorii Strashko
617c24eef28SGrygorii Strashko ret = cpsw_update_channels_res(priv, chs->tx_count, 0, rx_handler);
618c24eef28SGrygorii Strashko if (ret)
619c24eef28SGrygorii Strashko goto err;
620c24eef28SGrygorii Strashko
6219126e75eSIvan Khoronzhuk for (i = 0; i < cpsw->data.slaves; i++) {
6229126e75eSIvan Khoronzhuk sl_ndev = cpsw->slaves[i].ndev;
6239126e75eSIvan Khoronzhuk if (!(sl_ndev && netif_running(sl_ndev)))
624c24eef28SGrygorii Strashko continue;
625c24eef28SGrygorii Strashko
626c24eef28SGrygorii Strashko /* Inform stack about new count of queues */
6279126e75eSIvan Khoronzhuk ret = netif_set_real_num_tx_queues(sl_ndev, cpsw->tx_ch_num);
628c24eef28SGrygorii Strashko if (ret) {
629c24eef28SGrygorii Strashko dev_err(priv->dev, "cannot set real number of tx queues\n");
630c24eef28SGrygorii Strashko goto err;
631c24eef28SGrygorii Strashko }
632c24eef28SGrygorii Strashko
6339126e75eSIvan Khoronzhuk ret = netif_set_real_num_rx_queues(sl_ndev, cpsw->rx_ch_num);
634c24eef28SGrygorii Strashko if (ret) {
635c24eef28SGrygorii Strashko dev_err(priv->dev, "cannot set real number of rx queues\n");
636c24eef28SGrygorii Strashko goto err;
637c24eef28SGrygorii Strashko }
638c24eef28SGrygorii Strashko }
639c24eef28SGrygorii Strashko
640c24eef28SGrygorii Strashko cpsw_split_res(cpsw);
641c24eef28SGrygorii Strashko
6429ed4050cSIvan Khoronzhuk if (new_pools) {
6439ed4050cSIvan Khoronzhuk cpsw_destroy_xdp_rxqs(cpsw);
6449ed4050cSIvan Khoronzhuk ret = cpsw_create_xdp_rxqs(cpsw);
6459ed4050cSIvan Khoronzhuk if (ret)
6469ed4050cSIvan Khoronzhuk goto err;
6479ed4050cSIvan Khoronzhuk }
6489ed4050cSIvan Khoronzhuk
649c24eef28SGrygorii Strashko ret = cpsw_resume_data_pass(ndev);
650c24eef28SGrygorii Strashko if (!ret)
651c24eef28SGrygorii Strashko return 0;
652c24eef28SGrygorii Strashko err:
653c24eef28SGrygorii Strashko dev_err(priv->dev, "cannot update channels number, closing device\n");
6549ed4050cSIvan Khoronzhuk cpsw_fail(cpsw);
655c24eef28SGrygorii Strashko return ret;
656c24eef28SGrygorii Strashko }
657c24eef28SGrygorii Strashko
cpsw_get_ringparam(struct net_device * ndev,struct ethtool_ringparam * ering,struct kernel_ethtool_ringparam * kernel_ering,struct netlink_ext_ack * extack)658c24eef28SGrygorii Strashko void cpsw_get_ringparam(struct net_device *ndev,
65974624944SHao Chen struct ethtool_ringparam *ering,
66074624944SHao Chen struct kernel_ethtool_ringparam *kernel_ering,
66174624944SHao Chen struct netlink_ext_ack *extack)
662c24eef28SGrygorii Strashko {
663c24eef28SGrygorii Strashko struct cpsw_priv *priv = netdev_priv(ndev);
664c24eef28SGrygorii Strashko struct cpsw_common *cpsw = priv->cpsw;
665c24eef28SGrygorii Strashko
666c24eef28SGrygorii Strashko /* not supported */
66709faf5a7SIvan Khoronzhuk ering->tx_max_pending = cpsw->descs_pool_size - CPSW_MAX_QUEUES;
668c24eef28SGrygorii Strashko ering->tx_pending = cpdma_get_num_tx_descs(cpsw->dma);
669c24eef28SGrygorii Strashko ering->rx_max_pending = cpsw->descs_pool_size - CPSW_MAX_QUEUES;
670c24eef28SGrygorii Strashko ering->rx_pending = cpdma_get_num_rx_descs(cpsw->dma);
671c24eef28SGrygorii Strashko }
672c24eef28SGrygorii Strashko
cpsw_set_ringparam(struct net_device * ndev,struct ethtool_ringparam * ering,struct kernel_ethtool_ringparam * kernel_ering,struct netlink_ext_ack * extack)673c24eef28SGrygorii Strashko int cpsw_set_ringparam(struct net_device *ndev,
67474624944SHao Chen struct ethtool_ringparam *ering,
67574624944SHao Chen struct kernel_ethtool_ringparam *kernel_ering,
67674624944SHao Chen struct netlink_ext_ack *extack)
677c24eef28SGrygorii Strashko {
6789ed4050cSIvan Khoronzhuk struct cpsw_common *cpsw = ndev_to_cpsw(ndev);
679962fb618SIvan Khoronzhuk int descs_num, ret;
680c24eef28SGrygorii Strashko
681c24eef28SGrygorii Strashko /* ignore ering->tx_pending - only rx_pending adjustment is supported */
682c24eef28SGrygorii Strashko
683c24eef28SGrygorii Strashko if (ering->rx_mini_pending || ering->rx_jumbo_pending ||
684c24eef28SGrygorii Strashko ering->rx_pending < CPSW_MAX_QUEUES ||
685c24eef28SGrygorii Strashko ering->rx_pending > (cpsw->descs_pool_size - CPSW_MAX_QUEUES))
686c24eef28SGrygorii Strashko return -EINVAL;
687c24eef28SGrygorii Strashko
688962fb618SIvan Khoronzhuk descs_num = cpdma_get_num_rx_descs(cpsw->dma);
689962fb618SIvan Khoronzhuk if (ering->rx_pending == descs_num)
690c24eef28SGrygorii Strashko return 0;
691c24eef28SGrygorii Strashko
692c24eef28SGrygorii Strashko cpsw_suspend_data_pass(ndev);
693c24eef28SGrygorii Strashko
694962fb618SIvan Khoronzhuk ret = cpdma_set_num_rx_descs(cpsw->dma, ering->rx_pending);
695962fb618SIvan Khoronzhuk if (ret) {
696962fb618SIvan Khoronzhuk if (cpsw_resume_data_pass(ndev))
697962fb618SIvan Khoronzhuk goto err;
698c24eef28SGrygorii Strashko
699962fb618SIvan Khoronzhuk return ret;
700962fb618SIvan Khoronzhuk }
701c24eef28SGrygorii Strashko
7029ed4050cSIvan Khoronzhuk if (cpsw->usage_count) {
7039ed4050cSIvan Khoronzhuk cpsw_destroy_xdp_rxqs(cpsw);
7049ed4050cSIvan Khoronzhuk ret = cpsw_create_xdp_rxqs(cpsw);
7059ed4050cSIvan Khoronzhuk if (ret)
7069ed4050cSIvan Khoronzhuk goto err;
7079ed4050cSIvan Khoronzhuk }
7089ed4050cSIvan Khoronzhuk
709c24eef28SGrygorii Strashko ret = cpsw_resume_data_pass(ndev);
710c24eef28SGrygorii Strashko if (!ret)
711c24eef28SGrygorii Strashko return 0;
712962fb618SIvan Khoronzhuk err:
713962fb618SIvan Khoronzhuk cpdma_set_num_rx_descs(cpsw->dma, descs_num);
714c24eef28SGrygorii Strashko dev_err(cpsw->dev, "cannot set ring params, closing device\n");
7159ed4050cSIvan Khoronzhuk cpsw_fail(cpsw);
716c24eef28SGrygorii Strashko return ret;
717c24eef28SGrygorii Strashko }
718c24eef28SGrygorii Strashko
719c24eef28SGrygorii Strashko #if IS_ENABLED(CONFIG_TI_CPTS)
cpsw_get_ts_info(struct net_device * ndev,struct ethtool_ts_info * info)720c24eef28SGrygorii Strashko int cpsw_get_ts_info(struct net_device *ndev, struct ethtool_ts_info *info)
721c24eef28SGrygorii Strashko {
722c24eef28SGrygorii Strashko struct cpsw_common *cpsw = ndev_to_cpsw(ndev);
723c24eef28SGrygorii Strashko
724c24eef28SGrygorii Strashko info->so_timestamping =
725c24eef28SGrygorii Strashko SOF_TIMESTAMPING_TX_HARDWARE |
726c24eef28SGrygorii Strashko SOF_TIMESTAMPING_TX_SOFTWARE |
727c24eef28SGrygorii Strashko SOF_TIMESTAMPING_RX_HARDWARE |
728c24eef28SGrygorii Strashko SOF_TIMESTAMPING_RX_SOFTWARE |
729c24eef28SGrygorii Strashko SOF_TIMESTAMPING_SOFTWARE |
730c24eef28SGrygorii Strashko SOF_TIMESTAMPING_RAW_HARDWARE;
731c24eef28SGrygorii Strashko info->phc_index = cpsw->cpts->phc_index;
732c24eef28SGrygorii Strashko info->tx_types =
733c24eef28SGrygorii Strashko (1 << HWTSTAMP_TX_OFF) |
734c24eef28SGrygorii Strashko (1 << HWTSTAMP_TX_ON);
735c24eef28SGrygorii Strashko info->rx_filters =
736c24eef28SGrygorii Strashko (1 << HWTSTAMP_FILTER_NONE) |
737c24eef28SGrygorii Strashko (1 << HWTSTAMP_FILTER_PTP_V2_EVENT);
738c24eef28SGrygorii Strashko return 0;
739c24eef28SGrygorii Strashko }
740c24eef28SGrygorii Strashko #else
cpsw_get_ts_info(struct net_device * ndev,struct ethtool_ts_info * info)741c24eef28SGrygorii Strashko int cpsw_get_ts_info(struct net_device *ndev, struct ethtool_ts_info *info)
742c24eef28SGrygorii Strashko {
743c24eef28SGrygorii Strashko info->so_timestamping =
744c24eef28SGrygorii Strashko SOF_TIMESTAMPING_TX_SOFTWARE |
745c24eef28SGrygorii Strashko SOF_TIMESTAMPING_RX_SOFTWARE |
746c24eef28SGrygorii Strashko SOF_TIMESTAMPING_SOFTWARE;
747c24eef28SGrygorii Strashko info->phc_index = -1;
748c24eef28SGrygorii Strashko info->tx_types = 0;
749c24eef28SGrygorii Strashko info->rx_filters = 0;
750c24eef28SGrygorii Strashko return 0;
751c24eef28SGrygorii Strashko }
752c24eef28SGrygorii Strashko #endif
753