blob: d39875e267d9ebaaa8332b48d30841ec6a781c4b [file] [log] [blame]
Mugunthan V Ndf828592012-03-18 20:17:54 +00001/*
2 * Texas Instruments Ethernet Switch Driver
3 *
4 * Copyright (C) 2012 Texas Instruments
5 *
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License as
8 * published by the Free Software Foundation version 2.
9 *
10 * This program is distributed "as is" WITHOUT ANY WARRANTY of any
11 * kind, whether express or implied; without even the implied warranty
12 * of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
14 */
15
16#include <linux/kernel.h>
17#include <linux/io.h>
18#include <linux/clk.h>
19#include <linux/timer.h>
20#include <linux/module.h>
21#include <linux/platform_device.h>
22#include <linux/irqreturn.h>
23#include <linux/interrupt.h>
24#include <linux/if_ether.h>
25#include <linux/etherdevice.h>
26#include <linux/netdevice.h>
Richard Cochran2e5b38a2012-10-29 08:45:20 +000027#include <linux/net_tstamp.h>
Mugunthan V Ndf828592012-03-18 20:17:54 +000028#include <linux/phy.h>
29#include <linux/workqueue.h>
30#include <linux/delay.h>
Mugunthan V Nf150bd72012-07-17 08:09:50 +000031#include <linux/pm_runtime.h>
Mugunthan V N1d147cc2015-09-07 15:16:44 +053032#include <linux/gpio.h>
Mugunthan V N2eb32b02012-07-30 10:17:14 +000033#include <linux/of.h>
Heiko Schocher9e42f712015-10-17 06:04:35 +020034#include <linux/of_mdio.h>
Mugunthan V N2eb32b02012-07-30 10:17:14 +000035#include <linux/of_net.h>
36#include <linux/of_device.h>
Mugunthan V N3b72c2f2013-02-05 08:26:48 +000037#include <linux/if_vlan.h>
Mugunthan V Ndf828592012-03-18 20:17:54 +000038
Mugunthan V N739683b2013-06-06 23:45:14 +053039#include <linux/pinctrl/consumer.h>
Mugunthan V Ndf828592012-03-18 20:17:54 +000040
Mugunthan V Ndbe34722013-08-19 17:47:40 +053041#include "cpsw.h"
Mugunthan V Ndf828592012-03-18 20:17:54 +000042#include "cpsw_ale.h"
Richard Cochran2e5b38a2012-10-29 08:45:20 +000043#include "cpts.h"
Mugunthan V Ndf828592012-03-18 20:17:54 +000044#include "davinci_cpdma.h"
45
46#define CPSW_DEBUG (NETIF_MSG_HW | NETIF_MSG_WOL | \
47 NETIF_MSG_DRV | NETIF_MSG_LINK | \
48 NETIF_MSG_IFUP | NETIF_MSG_INTR | \
49 NETIF_MSG_PROBE | NETIF_MSG_TIMER | \
50 NETIF_MSG_IFDOWN | NETIF_MSG_RX_ERR | \
51 NETIF_MSG_TX_ERR | NETIF_MSG_TX_DONE | \
52 NETIF_MSG_PKTDATA | NETIF_MSG_TX_QUEUED | \
53 NETIF_MSG_RX_STATUS)
54
55#define cpsw_info(priv, type, format, ...) \
56do { \
57 if (netif_msg_##type(priv) && net_ratelimit()) \
58 dev_info(priv->dev, format, ## __VA_ARGS__); \
59} while (0)
60
61#define cpsw_err(priv, type, format, ...) \
62do { \
63 if (netif_msg_##type(priv) && net_ratelimit()) \
64 dev_err(priv->dev, format, ## __VA_ARGS__); \
65} while (0)
66
67#define cpsw_dbg(priv, type, format, ...) \
68do { \
69 if (netif_msg_##type(priv) && net_ratelimit()) \
70 dev_dbg(priv->dev, format, ## __VA_ARGS__); \
71} while (0)
72
73#define cpsw_notice(priv, type, format, ...) \
74do { \
75 if (netif_msg_##type(priv) && net_ratelimit()) \
76 dev_notice(priv->dev, format, ## __VA_ARGS__); \
77} while (0)
78
Mugunthan V N5c50a852012-10-29 08:45:11 +000079#define ALE_ALL_PORTS 0x7
80
Mugunthan V Ndf828592012-03-18 20:17:54 +000081#define CPSW_MAJOR_VERSION(reg) (reg >> 8 & 0x7)
82#define CPSW_MINOR_VERSION(reg) (reg & 0xff)
83#define CPSW_RTL_VERSION(reg) ((reg >> 11) & 0x1f)
84
Richard Cochrane90cfac2012-10-29 08:45:14 +000085#define CPSW_VERSION_1 0x19010a
86#define CPSW_VERSION_2 0x19010c
Mugunthan V Nc193f362013-08-05 17:30:05 +053087#define CPSW_VERSION_3 0x19010f
Mugunthan V N926489b2013-08-12 17:11:15 +053088#define CPSW_VERSION_4 0x190112
Richard Cochran549985e2012-11-14 09:07:56 +000089
90#define HOST_PORT_NUM 0
91#define SLIVER_SIZE 0x40
92
93#define CPSW1_HOST_PORT_OFFSET 0x028
94#define CPSW1_SLAVE_OFFSET 0x050
95#define CPSW1_SLAVE_SIZE 0x040
96#define CPSW1_CPDMA_OFFSET 0x100
97#define CPSW1_STATERAM_OFFSET 0x200
Mugunthan V Nd9718542013-07-23 15:38:17 +053098#define CPSW1_HW_STATS 0x400
Richard Cochran549985e2012-11-14 09:07:56 +000099#define CPSW1_CPTS_OFFSET 0x500
100#define CPSW1_ALE_OFFSET 0x600
101#define CPSW1_SLIVER_OFFSET 0x700
102
103#define CPSW2_HOST_PORT_OFFSET 0x108
104#define CPSW2_SLAVE_OFFSET 0x200
105#define CPSW2_SLAVE_SIZE 0x100
106#define CPSW2_CPDMA_OFFSET 0x800
Mugunthan V Nd9718542013-07-23 15:38:17 +0530107#define CPSW2_HW_STATS 0x900
Richard Cochran549985e2012-11-14 09:07:56 +0000108#define CPSW2_STATERAM_OFFSET 0xa00
109#define CPSW2_CPTS_OFFSET 0xc00
110#define CPSW2_ALE_OFFSET 0xd00
111#define CPSW2_SLIVER_OFFSET 0xd80
112#define CPSW2_BD_OFFSET 0x2000
113
Mugunthan V Ndf828592012-03-18 20:17:54 +0000114#define CPDMA_RXTHRESH 0x0c0
115#define CPDMA_RXFREE 0x0e0
116#define CPDMA_TXHDP 0x00
117#define CPDMA_RXHDP 0x20
118#define CPDMA_TXCP 0x40
119#define CPDMA_RXCP 0x60
120
Mugunthan V Ndf828592012-03-18 20:17:54 +0000121#define CPSW_POLL_WEIGHT 64
122#define CPSW_MIN_PACKET_SIZE 60
123#define CPSW_MAX_PACKET_SIZE (1500 + 14 + 4 + 4)
124
125#define RX_PRIORITY_MAPPING 0x76543210
126#define TX_PRIORITY_MAPPING 0x33221100
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +0300127#define CPDMA_TX_PRIORITY_MAP 0x01234567
Mugunthan V Ndf828592012-03-18 20:17:54 +0000128
Mugunthan V N3b72c2f2013-02-05 08:26:48 +0000129#define CPSW_VLAN_AWARE BIT(1)
130#define CPSW_ALE_VLAN_AWARE 1
131
John Ogness35717d82014-11-14 15:42:52 +0100132#define CPSW_FIFO_NORMAL_MODE (0 << 16)
133#define CPSW_FIFO_DUAL_MAC_MODE (1 << 16)
134#define CPSW_FIFO_RATE_LIMIT_MODE (2 << 16)
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +0000135
Mugunthan V Nff5b8ef2013-03-11 23:16:37 +0000136#define CPSW_INTPACEEN (0x3f << 16)
137#define CPSW_INTPRESCALE_MASK (0x7FF << 0)
138#define CPSW_CMINTMAX_CNT 63
139#define CPSW_CMINTMIN_CNT 2
140#define CPSW_CMINTMAX_INTVL (1000 / CPSW_CMINTMIN_CNT)
141#define CPSW_CMINTMIN_INTVL ((1000 / CPSW_CMINTMAX_CNT) + 1)
142
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +0300143#define cpsw_slave_index(cpsw, priv) \
144 ((cpsw->data.dual_emac) ? priv->emac_port : \
145 cpsw->data.active_slave)
Ivan Khoronzhuke38b5a32016-08-10 02:22:41 +0300146#define IRQ_NUM 2
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +0300147#define CPSW_MAX_QUEUES 8
Grygorii Strashko90225bf2017-01-06 14:07:33 -0600148#define CPSW_CPDMA_DESCS_POOL_SIZE_DEFAULT 256
Mugunthan V Nd3bb9c52013-03-11 23:16:36 +0000149
Mugunthan V Ndf828592012-03-18 20:17:54 +0000150static int debug_level;
151module_param(debug_level, int, 0);
152MODULE_PARM_DESC(debug_level, "cpsw debug level (NETIF_MSG bits)");
153
154static int ale_ageout = 10;
155module_param(ale_ageout, int, 0);
156MODULE_PARM_DESC(ale_ageout, "cpsw ale ageout interval (seconds)");
157
158static int rx_packet_max = CPSW_MAX_PACKET_SIZE;
159module_param(rx_packet_max, int, 0);
160MODULE_PARM_DESC(rx_packet_max, "maximum receive packet size (bytes)");
161
Grygorii Strashko90225bf2017-01-06 14:07:33 -0600162static int descs_pool_size = CPSW_CPDMA_DESCS_POOL_SIZE_DEFAULT;
163module_param(descs_pool_size, int, 0444);
164MODULE_PARM_DESC(descs_pool_size, "Number of CPDMA CPPI descriptors in pool");
165
Richard Cochran996a5c22012-10-29 08:45:12 +0000166struct cpsw_wr_regs {
Mugunthan V Ndf828592012-03-18 20:17:54 +0000167 u32 id_ver;
168 u32 soft_reset;
169 u32 control;
170 u32 int_control;
171 u32 rx_thresh_en;
172 u32 rx_en;
173 u32 tx_en;
174 u32 misc_en;
Mugunthan V Nff5b8ef2013-03-11 23:16:37 +0000175 u32 mem_allign1[8];
176 u32 rx_thresh_stat;
177 u32 rx_stat;
178 u32 tx_stat;
179 u32 misc_stat;
180 u32 mem_allign2[8];
181 u32 rx_imax;
182 u32 tx_imax;
183
Mugunthan V Ndf828592012-03-18 20:17:54 +0000184};
185
Richard Cochran996a5c22012-10-29 08:45:12 +0000186struct cpsw_ss_regs {
Mugunthan V Ndf828592012-03-18 20:17:54 +0000187 u32 id_ver;
188 u32 control;
189 u32 soft_reset;
190 u32 stat_port_en;
191 u32 ptype;
Richard Cochranbd357af2012-10-29 08:45:13 +0000192 u32 soft_idle;
193 u32 thru_rate;
194 u32 gap_thresh;
195 u32 tx_start_wds;
196 u32 flow_control;
197 u32 vlan_ltype;
198 u32 ts_ltype;
199 u32 dlr_ltype;
Mugunthan V Ndf828592012-03-18 20:17:54 +0000200};
201
Richard Cochran9750a3a2012-10-29 08:45:15 +0000202/* CPSW_PORT_V1 */
203#define CPSW1_MAX_BLKS 0x00 /* Maximum FIFO Blocks */
204#define CPSW1_BLK_CNT 0x04 /* FIFO Block Usage Count (Read Only) */
205#define CPSW1_TX_IN_CTL 0x08 /* Transmit FIFO Control */
206#define CPSW1_PORT_VLAN 0x0c /* VLAN Register */
207#define CPSW1_TX_PRI_MAP 0x10 /* Tx Header Priority to Switch Pri Mapping */
208#define CPSW1_TS_CTL 0x14 /* Time Sync Control */
209#define CPSW1_TS_SEQ_LTYPE 0x18 /* Time Sync Sequence ID Offset and Msg Type */
210#define CPSW1_TS_VLAN 0x1c /* Time Sync VLAN1 and VLAN2 */
211
212/* CPSW_PORT_V2 */
213#define CPSW2_CONTROL 0x00 /* Control Register */
214#define CPSW2_MAX_BLKS 0x08 /* Maximum FIFO Blocks */
215#define CPSW2_BLK_CNT 0x0c /* FIFO Block Usage Count (Read Only) */
216#define CPSW2_TX_IN_CTL 0x10 /* Transmit FIFO Control */
217#define CPSW2_PORT_VLAN 0x14 /* VLAN Register */
218#define CPSW2_TX_PRI_MAP 0x18 /* Tx Header Priority to Switch Pri Mapping */
219#define CPSW2_TS_SEQ_MTYPE 0x1c /* Time Sync Sequence ID Offset and Msg Type */
220
221/* CPSW_PORT_V1 and V2 */
222#define SA_LO 0x20 /* CPGMAC_SL Source Address Low */
223#define SA_HI 0x24 /* CPGMAC_SL Source Address High */
224#define SEND_PERCENT 0x28 /* Transmit Queue Send Percentages */
225
226/* CPSW_PORT_V2 only */
227#define RX_DSCP_PRI_MAP0 0x30 /* Rx DSCP Priority to Rx Packet Mapping */
228#define RX_DSCP_PRI_MAP1 0x34 /* Rx DSCP Priority to Rx Packet Mapping */
229#define RX_DSCP_PRI_MAP2 0x38 /* Rx DSCP Priority to Rx Packet Mapping */
230#define RX_DSCP_PRI_MAP3 0x3c /* Rx DSCP Priority to Rx Packet Mapping */
231#define RX_DSCP_PRI_MAP4 0x40 /* Rx DSCP Priority to Rx Packet Mapping */
232#define RX_DSCP_PRI_MAP5 0x44 /* Rx DSCP Priority to Rx Packet Mapping */
233#define RX_DSCP_PRI_MAP6 0x48 /* Rx DSCP Priority to Rx Packet Mapping */
234#define RX_DSCP_PRI_MAP7 0x4c /* Rx DSCP Priority to Rx Packet Mapping */
235
236/* Bit definitions for the CPSW2_CONTROL register */
237#define PASS_PRI_TAGGED (1<<24) /* Pass Priority Tagged */
238#define VLAN_LTYPE2_EN (1<<21) /* VLAN LTYPE 2 enable */
239#define VLAN_LTYPE1_EN (1<<20) /* VLAN LTYPE 1 enable */
240#define DSCP_PRI_EN (1<<16) /* DSCP Priority Enable */
241#define TS_320 (1<<14) /* Time Sync Dest Port 320 enable */
242#define TS_319 (1<<13) /* Time Sync Dest Port 319 enable */
243#define TS_132 (1<<12) /* Time Sync Dest IP Addr 132 enable */
244#define TS_131 (1<<11) /* Time Sync Dest IP Addr 131 enable */
245#define TS_130 (1<<10) /* Time Sync Dest IP Addr 130 enable */
246#define TS_129 (1<<9) /* Time Sync Dest IP Addr 129 enable */
George Cherian09c55372014-05-02 12:02:02 +0530247#define TS_TTL_NONZERO (1<<8) /* Time Sync Time To Live Non-zero enable */
248#define TS_ANNEX_F_EN (1<<6) /* Time Sync Annex F enable */
Richard Cochran9750a3a2012-10-29 08:45:15 +0000249#define TS_ANNEX_D_EN (1<<4) /* Time Sync Annex D enable */
250#define TS_LTYPE2_EN (1<<3) /* Time Sync LTYPE 2 enable */
251#define TS_LTYPE1_EN (1<<2) /* Time Sync LTYPE 1 enable */
252#define TS_TX_EN (1<<1) /* Time Sync Transmit Enable */
253#define TS_RX_EN (1<<0) /* Time Sync Receive Enable */
254
George Cherian09c55372014-05-02 12:02:02 +0530255#define CTRL_V2_TS_BITS \
256 (TS_320 | TS_319 | TS_132 | TS_131 | TS_130 | TS_129 |\
257 TS_TTL_NONZERO | TS_ANNEX_D_EN | TS_LTYPE1_EN)
Richard Cochran9750a3a2012-10-29 08:45:15 +0000258
George Cherian09c55372014-05-02 12:02:02 +0530259#define CTRL_V2_ALL_TS_MASK (CTRL_V2_TS_BITS | TS_TX_EN | TS_RX_EN)
260#define CTRL_V2_TX_TS_BITS (CTRL_V2_TS_BITS | TS_TX_EN)
261#define CTRL_V2_RX_TS_BITS (CTRL_V2_TS_BITS | TS_RX_EN)
262
263
264#define CTRL_V3_TS_BITS \
265 (TS_320 | TS_319 | TS_132 | TS_131 | TS_130 | TS_129 |\
266 TS_TTL_NONZERO | TS_ANNEX_F_EN | TS_ANNEX_D_EN |\
267 TS_LTYPE1_EN)
268
269#define CTRL_V3_ALL_TS_MASK (CTRL_V3_TS_BITS | TS_TX_EN | TS_RX_EN)
270#define CTRL_V3_TX_TS_BITS (CTRL_V3_TS_BITS | TS_TX_EN)
271#define CTRL_V3_RX_TS_BITS (CTRL_V3_TS_BITS | TS_RX_EN)
Richard Cochran9750a3a2012-10-29 08:45:15 +0000272
273/* Bit definitions for the CPSW2_TS_SEQ_MTYPE register */
274#define TS_SEQ_ID_OFFSET_SHIFT (16) /* Time Sync Sequence ID Offset */
275#define TS_SEQ_ID_OFFSET_MASK (0x3f)
276#define TS_MSG_TYPE_EN_SHIFT (0) /* Time Sync Message Type Enable */
277#define TS_MSG_TYPE_EN_MASK (0xffff)
278
279/* The PTP event messages - Sync, Delay_Req, Pdelay_Req, and Pdelay_Resp. */
280#define EVENT_MSG_BITS ((1<<0) | (1<<1) | (1<<2) | (1<<3))
Mugunthan V Ndf828592012-03-18 20:17:54 +0000281
Richard Cochran2e5b38a2012-10-29 08:45:20 +0000282/* Bit definitions for the CPSW1_TS_CTL register */
283#define CPSW_V1_TS_RX_EN BIT(0)
284#define CPSW_V1_TS_TX_EN BIT(4)
285#define CPSW_V1_MSG_TYPE_OFS 16
286
287/* Bit definitions for the CPSW1_TS_SEQ_LTYPE register */
288#define CPSW_V1_SEQ_ID_OFS_SHIFT 16
289
Mugunthan V Ndf828592012-03-18 20:17:54 +0000290struct cpsw_host_regs {
291 u32 max_blks;
292 u32 blk_cnt;
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +0000293 u32 tx_in_ctl;
Mugunthan V Ndf828592012-03-18 20:17:54 +0000294 u32 port_vlan;
295 u32 tx_pri_map;
296 u32 cpdma_tx_pri_map;
297 u32 cpdma_rx_chan_map;
298};
299
300struct cpsw_sliver_regs {
301 u32 id_ver;
302 u32 mac_control;
303 u32 mac_status;
304 u32 soft_reset;
305 u32 rx_maxlen;
306 u32 __reserved_0;
307 u32 rx_pause;
308 u32 tx_pause;
309 u32 __reserved_1;
310 u32 rx_pri_map;
311};
312
Mugunthan V Nd9718542013-07-23 15:38:17 +0530313struct cpsw_hw_stats {
314 u32 rxgoodframes;
315 u32 rxbroadcastframes;
316 u32 rxmulticastframes;
317 u32 rxpauseframes;
318 u32 rxcrcerrors;
319 u32 rxaligncodeerrors;
320 u32 rxoversizedframes;
321 u32 rxjabberframes;
322 u32 rxundersizedframes;
323 u32 rxfragments;
324 u32 __pad_0[2];
325 u32 rxoctets;
326 u32 txgoodframes;
327 u32 txbroadcastframes;
328 u32 txmulticastframes;
329 u32 txpauseframes;
330 u32 txdeferredframes;
331 u32 txcollisionframes;
332 u32 txsinglecollframes;
333 u32 txmultcollframes;
334 u32 txexcessivecollisions;
335 u32 txlatecollisions;
336 u32 txunderrun;
337 u32 txcarriersenseerrors;
338 u32 txoctets;
339 u32 octetframes64;
340 u32 octetframes65t127;
341 u32 octetframes128t255;
342 u32 octetframes256t511;
343 u32 octetframes512t1023;
344 u32 octetframes1024tup;
345 u32 netoctets;
346 u32 rxsofoverruns;
347 u32 rxmofoverruns;
348 u32 rxdmaoverruns;
349};
350
Mugunthan V Ndf828592012-03-18 20:17:54 +0000351struct cpsw_slave {
Richard Cochran9750a3a2012-10-29 08:45:15 +0000352 void __iomem *regs;
Mugunthan V Ndf828592012-03-18 20:17:54 +0000353 struct cpsw_sliver_regs __iomem *sliver;
354 int slave_num;
355 u32 mac_control;
356 struct cpsw_slave_data *data;
357 struct phy_device *phy;
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +0000358 struct net_device *ndev;
359 u32 port_vlan;
360 u32 open_stat;
Mugunthan V Ndf828592012-03-18 20:17:54 +0000361};
362
Richard Cochran9750a3a2012-10-29 08:45:15 +0000363static inline u32 slave_read(struct cpsw_slave *slave, u32 offset)
364{
365 return __raw_readl(slave->regs + offset);
366}
367
368static inline void slave_write(struct cpsw_slave *slave, u32 val, u32 offset)
369{
370 __raw_writel(val, slave->regs + offset);
371}
372
Ivan Khoronzhuk8feb0a12016-11-29 17:00:51 +0200373struct cpsw_vector {
374 struct cpdma_chan *ch;
375 int budget;
376};
377
Ivan Khoronzhuk649a1682016-08-10 02:22:37 +0300378struct cpsw_common {
Ivan Khoronzhuk56e31bd2016-08-10 02:22:38 +0300379 struct device *dev;
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +0300380 struct cpsw_platform_data data;
Ivan Khoronzhukdbc4ec52016-08-10 02:22:43 +0300381 struct napi_struct napi_rx;
382 struct napi_struct napi_tx;
Ivan Khoronzhuk5d8d0d42016-08-10 02:22:39 +0300383 struct cpsw_ss_regs __iomem *regs;
384 struct cpsw_wr_regs __iomem *wr_regs;
385 u8 __iomem *hw_stats;
386 struct cpsw_host_regs __iomem *host_port_regs;
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +0300387 u32 version;
388 u32 coal_intvl;
389 u32 bus_freq_mhz;
390 int rx_packet_max;
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +0300391 struct cpsw_slave *slaves;
Ivan Khoronzhuk2c836bd2016-08-10 02:22:40 +0300392 struct cpdma_ctlr *dma;
Ivan Khoronzhuk8feb0a12016-11-29 17:00:51 +0200393 struct cpsw_vector txv[CPSW_MAX_QUEUES];
394 struct cpsw_vector rxv[CPSW_MAX_QUEUES];
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +0300395 struct cpsw_ale *ale;
Ivan Khoronzhuke38b5a32016-08-10 02:22:41 +0300396 bool quirk_irq;
397 bool rx_irq_disabled;
398 bool tx_irq_disabled;
399 u32 irqs_table[IRQ_NUM];
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +0300400 struct cpts *cpts;
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +0300401 int rx_ch_num, tx_ch_num;
Ivan Khoronzhuk0be01b82016-12-10 14:23:49 +0200402 int speed;
Ivan Khoronzhuk649a1682016-08-10 02:22:37 +0300403};
404
405struct cpsw_priv {
Mugunthan V Ndf828592012-03-18 20:17:54 +0000406 struct net_device *ndev;
Mugunthan V Ndf828592012-03-18 20:17:54 +0000407 struct device *dev;
Mugunthan V Ndf828592012-03-18 20:17:54 +0000408 u32 msg_enable;
Mugunthan V Ndf828592012-03-18 20:17:54 +0000409 u8 mac_addr[ETH_ALEN];
Mugunthan V N1923d6e2014-09-08 22:54:02 +0530410 bool rx_pause;
411 bool tx_pause;
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +0000412 u32 emac_port;
Ivan Khoronzhuk649a1682016-08-10 02:22:37 +0300413 struct cpsw_common *cpsw;
Mugunthan V Ndf828592012-03-18 20:17:54 +0000414};
415
Mugunthan V Nd9718542013-07-23 15:38:17 +0530416struct cpsw_stats {
417 char stat_string[ETH_GSTRING_LEN];
418 int type;
419 int sizeof_stat;
420 int stat_offset;
421};
422
423enum {
424 CPSW_STATS,
425 CPDMA_RX_STATS,
426 CPDMA_TX_STATS,
427};
428
429#define CPSW_STAT(m) CPSW_STATS, \
430 sizeof(((struct cpsw_hw_stats *)0)->m), \
431 offsetof(struct cpsw_hw_stats, m)
432#define CPDMA_RX_STAT(m) CPDMA_RX_STATS, \
433 sizeof(((struct cpdma_chan_stats *)0)->m), \
434 offsetof(struct cpdma_chan_stats, m)
435#define CPDMA_TX_STAT(m) CPDMA_TX_STATS, \
436 sizeof(((struct cpdma_chan_stats *)0)->m), \
437 offsetof(struct cpdma_chan_stats, m)
438
439static const struct cpsw_stats cpsw_gstrings_stats[] = {
440 { "Good Rx Frames", CPSW_STAT(rxgoodframes) },
441 { "Broadcast Rx Frames", CPSW_STAT(rxbroadcastframes) },
442 { "Multicast Rx Frames", CPSW_STAT(rxmulticastframes) },
443 { "Pause Rx Frames", CPSW_STAT(rxpauseframes) },
444 { "Rx CRC Errors", CPSW_STAT(rxcrcerrors) },
445 { "Rx Align/Code Errors", CPSW_STAT(rxaligncodeerrors) },
446 { "Oversize Rx Frames", CPSW_STAT(rxoversizedframes) },
447 { "Rx Jabbers", CPSW_STAT(rxjabberframes) },
448 { "Undersize (Short) Rx Frames", CPSW_STAT(rxundersizedframes) },
449 { "Rx Fragments", CPSW_STAT(rxfragments) },
450 { "Rx Octets", CPSW_STAT(rxoctets) },
451 { "Good Tx Frames", CPSW_STAT(txgoodframes) },
452 { "Broadcast Tx Frames", CPSW_STAT(txbroadcastframes) },
453 { "Multicast Tx Frames", CPSW_STAT(txmulticastframes) },
454 { "Pause Tx Frames", CPSW_STAT(txpauseframes) },
455 { "Deferred Tx Frames", CPSW_STAT(txdeferredframes) },
456 { "Collisions", CPSW_STAT(txcollisionframes) },
457 { "Single Collision Tx Frames", CPSW_STAT(txsinglecollframes) },
458 { "Multiple Collision Tx Frames", CPSW_STAT(txmultcollframes) },
459 { "Excessive Collisions", CPSW_STAT(txexcessivecollisions) },
460 { "Late Collisions", CPSW_STAT(txlatecollisions) },
461 { "Tx Underrun", CPSW_STAT(txunderrun) },
462 { "Carrier Sense Errors", CPSW_STAT(txcarriersenseerrors) },
463 { "Tx Octets", CPSW_STAT(txoctets) },
464 { "Rx + Tx 64 Octet Frames", CPSW_STAT(octetframes64) },
465 { "Rx + Tx 65-127 Octet Frames", CPSW_STAT(octetframes65t127) },
466 { "Rx + Tx 128-255 Octet Frames", CPSW_STAT(octetframes128t255) },
467 { "Rx + Tx 256-511 Octet Frames", CPSW_STAT(octetframes256t511) },
468 { "Rx + Tx 512-1023 Octet Frames", CPSW_STAT(octetframes512t1023) },
469 { "Rx + Tx 1024-Up Octet Frames", CPSW_STAT(octetframes1024tup) },
470 { "Net Octets", CPSW_STAT(netoctets) },
471 { "Rx Start of Frame Overruns", CPSW_STAT(rxsofoverruns) },
472 { "Rx Middle of Frame Overruns", CPSW_STAT(rxmofoverruns) },
473 { "Rx DMA Overruns", CPSW_STAT(rxdmaoverruns) },
Mugunthan V Nd9718542013-07-23 15:38:17 +0530474};
475
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +0300476static const struct cpsw_stats cpsw_gstrings_ch_stats[] = {
477 { "head_enqueue", CPDMA_RX_STAT(head_enqueue) },
478 { "tail_enqueue", CPDMA_RX_STAT(tail_enqueue) },
479 { "pad_enqueue", CPDMA_RX_STAT(pad_enqueue) },
480 { "misqueued", CPDMA_RX_STAT(misqueued) },
481 { "desc_alloc_fail", CPDMA_RX_STAT(desc_alloc_fail) },
482 { "pad_alloc_fail", CPDMA_RX_STAT(pad_alloc_fail) },
483 { "runt_receive_buf", CPDMA_RX_STAT(runt_receive_buff) },
484 { "runt_transmit_buf", CPDMA_RX_STAT(runt_transmit_buff) },
485 { "empty_dequeue", CPDMA_RX_STAT(empty_dequeue) },
486 { "busy_dequeue", CPDMA_RX_STAT(busy_dequeue) },
487 { "good_dequeue", CPDMA_RX_STAT(good_dequeue) },
488 { "requeue", CPDMA_RX_STAT(requeue) },
489 { "teardown_dequeue", CPDMA_RX_STAT(teardown_dequeue) },
490};
491
492#define CPSW_STATS_COMMON_LEN ARRAY_SIZE(cpsw_gstrings_stats)
493#define CPSW_STATS_CH_LEN ARRAY_SIZE(cpsw_gstrings_ch_stats)
Mugunthan V Nd9718542013-07-23 15:38:17 +0530494
Ivan Khoronzhuk649a1682016-08-10 02:22:37 +0300495#define ndev_to_cpsw(ndev) (((struct cpsw_priv *)netdev_priv(ndev))->cpsw)
Ivan Khoronzhukdbc4ec52016-08-10 02:22:43 +0300496#define napi_to_cpsw(napi) container_of(napi, struct cpsw_common, napi)
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +0000497#define for_each_slave(priv, func, arg...) \
498 do { \
Sebastian Siewior6e6ceae2013-04-24 08:48:24 +0000499 struct cpsw_slave *slave; \
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +0300500 struct cpsw_common *cpsw = (priv)->cpsw; \
Sebastian Siewior6e6ceae2013-04-24 08:48:24 +0000501 int n; \
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +0300502 if (cpsw->data.dual_emac) \
503 (func)((cpsw)->slaves + priv->emac_port, ##arg);\
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +0000504 else \
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +0300505 for (n = cpsw->data.slaves, \
506 slave = cpsw->slaves; \
Sebastian Siewior6e6ceae2013-04-24 08:48:24 +0000507 n; n--) \
508 (func)(slave++, ##arg); \
Mugunthan V Ndf828592012-03-18 20:17:54 +0000509 } while (0)
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +0000510
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +0300511#define cpsw_dual_emac_src_port_detect(cpsw, status, ndev, skb) \
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +0000512 do { \
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +0300513 if (!cpsw->data.dual_emac) \
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +0000514 break; \
515 if (CPDMA_RX_SOURCE_PORT(status) == 1) { \
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +0300516 ndev = cpsw->slaves[0].ndev; \
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +0000517 skb->dev = ndev; \
518 } else if (CPDMA_RX_SOURCE_PORT(status) == 2) { \
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +0300519 ndev = cpsw->slaves[1].ndev; \
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +0000520 skb->dev = ndev; \
521 } \
522 } while (0)
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +0300523#define cpsw_add_mcast(cpsw, priv, addr) \
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +0000524 do { \
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +0300525 if (cpsw->data.dual_emac) { \
526 struct cpsw_slave *slave = cpsw->slaves + \
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +0000527 priv->emac_port; \
Ivan Khoronzhuk6f1f5832016-08-10 02:22:34 +0300528 int slave_port = cpsw_get_slave_port( \
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +0000529 slave->slave_num); \
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +0300530 cpsw_ale_add_mcast(cpsw->ale, addr, \
Grygorii Strashko71a2cbb2016-04-07 15:16:44 +0300531 1 << slave_port | ALE_PORT_HOST, \
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +0000532 ALE_VLAN, slave->port_vlan, 0); \
533 } else { \
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +0300534 cpsw_ale_add_mcast(cpsw->ale, addr, \
Grygorii Strashko61f1cef2016-04-07 15:16:43 +0300535 ALE_ALL_PORTS, \
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +0000536 0, 0, 0); \
537 } \
538 } while (0)
539
Ivan Khoronzhuk6f1f5832016-08-10 02:22:34 +0300540static inline int cpsw_get_slave_port(u32 slave_num)
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +0000541{
Grygorii Strashko71a2cbb2016-04-07 15:16:44 +0300542 return slave_num + 1;
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +0000543}
Mugunthan V Ndf828592012-03-18 20:17:54 +0000544
Mugunthan V N0cd8f9c2014-01-23 00:03:12 +0530545static void cpsw_set_promiscious(struct net_device *ndev, bool enable)
546{
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +0300547 struct cpsw_common *cpsw = ndev_to_cpsw(ndev);
548 struct cpsw_ale *ale = cpsw->ale;
Mugunthan V N0cd8f9c2014-01-23 00:03:12 +0530549 int i;
550
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +0300551 if (cpsw->data.dual_emac) {
Mugunthan V N0cd8f9c2014-01-23 00:03:12 +0530552 bool flag = false;
553
554 /* Enabling promiscuous mode for one interface will be
555 * common for both the interface as the interface shares
556 * the same hardware resource.
557 */
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +0300558 for (i = 0; i < cpsw->data.slaves; i++)
559 if (cpsw->slaves[i].ndev->flags & IFF_PROMISC)
Mugunthan V N0cd8f9c2014-01-23 00:03:12 +0530560 flag = true;
561
562 if (!enable && flag) {
563 enable = true;
564 dev_err(&ndev->dev, "promiscuity not disabled as the other interface is still in promiscuity mode\n");
565 }
566
567 if (enable) {
568 /* Enable Bypass */
569 cpsw_ale_control_set(ale, 0, ALE_BYPASS, 1);
570
571 dev_dbg(&ndev->dev, "promiscuity enabled\n");
572 } else {
573 /* Disable Bypass */
574 cpsw_ale_control_set(ale, 0, ALE_BYPASS, 0);
575 dev_dbg(&ndev->dev, "promiscuity disabled\n");
576 }
577 } else {
578 if (enable) {
579 unsigned long timeout = jiffies + HZ;
580
Lennart Sorensen6f979eb2014-10-31 13:28:54 -0400581 /* Disable Learn for all ports (host is port 0 and slaves are port 1 and up */
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +0300582 for (i = 0; i <= cpsw->data.slaves; i++) {
Mugunthan V N0cd8f9c2014-01-23 00:03:12 +0530583 cpsw_ale_control_set(ale, i,
584 ALE_PORT_NOLEARN, 1);
585 cpsw_ale_control_set(ale, i,
586 ALE_PORT_NO_SA_UPDATE, 1);
587 }
588
589 /* Clear All Untouched entries */
590 cpsw_ale_control_set(ale, 0, ALE_AGEOUT, 1);
591 do {
592 cpu_relax();
593 if (cpsw_ale_control_get(ale, 0, ALE_AGEOUT))
594 break;
595 } while (time_after(timeout, jiffies));
596 cpsw_ale_control_set(ale, 0, ALE_AGEOUT, 1);
597
598 /* Clear all mcast from ALE */
Grygorii Strashko61f1cef2016-04-07 15:16:43 +0300599 cpsw_ale_flush_multicast(ale, ALE_ALL_PORTS, -1);
Mugunthan V N0cd8f9c2014-01-23 00:03:12 +0530600
601 /* Flood All Unicast Packets to Host port */
602 cpsw_ale_control_set(ale, 0, ALE_P0_UNI_FLOOD, 1);
603 dev_dbg(&ndev->dev, "promiscuity enabled\n");
604 } else {
Lennart Sorensen6f979eb2014-10-31 13:28:54 -0400605 /* Don't Flood All Unicast Packets to Host port */
Mugunthan V N0cd8f9c2014-01-23 00:03:12 +0530606 cpsw_ale_control_set(ale, 0, ALE_P0_UNI_FLOOD, 0);
607
Lennart Sorensen6f979eb2014-10-31 13:28:54 -0400608 /* Enable Learn for all ports (host is port 0 and slaves are port 1 and up */
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +0300609 for (i = 0; i <= cpsw->data.slaves; i++) {
Mugunthan V N0cd8f9c2014-01-23 00:03:12 +0530610 cpsw_ale_control_set(ale, i,
611 ALE_PORT_NOLEARN, 0);
612 cpsw_ale_control_set(ale, i,
613 ALE_PORT_NO_SA_UPDATE, 0);
614 }
615 dev_dbg(&ndev->dev, "promiscuity disabled\n");
616 }
617 }
618}
619
Mugunthan V N5c50a852012-10-29 08:45:11 +0000620static void cpsw_ndo_set_rx_mode(struct net_device *ndev)
621{
622 struct cpsw_priv *priv = netdev_priv(ndev);
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +0300623 struct cpsw_common *cpsw = priv->cpsw;
Mugunthan V N25906052015-01-13 17:35:49 +0530624 int vid;
625
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +0300626 if (cpsw->data.dual_emac)
627 vid = cpsw->slaves[priv->emac_port].port_vlan;
Mugunthan V N25906052015-01-13 17:35:49 +0530628 else
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +0300629 vid = cpsw->data.default_vlan;
Mugunthan V N5c50a852012-10-29 08:45:11 +0000630
631 if (ndev->flags & IFF_PROMISC) {
632 /* Enable promiscuous mode */
Mugunthan V N0cd8f9c2014-01-23 00:03:12 +0530633 cpsw_set_promiscious(ndev, true);
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +0300634 cpsw_ale_set_allmulti(cpsw->ale, IFF_ALLMULTI);
Mugunthan V N5c50a852012-10-29 08:45:11 +0000635 return;
Mugunthan V N0cd8f9c2014-01-23 00:03:12 +0530636 } else {
637 /* Disable promiscuous mode */
638 cpsw_set_promiscious(ndev, false);
Mugunthan V N5c50a852012-10-29 08:45:11 +0000639 }
640
Lennart Sorensen1e5c4bc2014-10-31 13:38:52 -0400641 /* Restore allmulti on vlans if necessary */
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +0300642 cpsw_ale_set_allmulti(cpsw->ale, priv->ndev->flags & IFF_ALLMULTI);
Lennart Sorensen1e5c4bc2014-10-31 13:38:52 -0400643
Mugunthan V N5c50a852012-10-29 08:45:11 +0000644 /* Clear all mcast from ALE */
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +0300645 cpsw_ale_flush_multicast(cpsw->ale, ALE_ALL_PORTS, vid);
Mugunthan V N5c50a852012-10-29 08:45:11 +0000646
647 if (!netdev_mc_empty(ndev)) {
648 struct netdev_hw_addr *ha;
649
650 /* program multicast address list into ALE register */
651 netdev_for_each_mc_addr(ha, ndev) {
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +0300652 cpsw_add_mcast(cpsw, priv, (u8 *)ha->addr);
Mugunthan V N5c50a852012-10-29 08:45:11 +0000653 }
654 }
655}
656
Ivan Khoronzhuk2c836bd2016-08-10 02:22:40 +0300657static void cpsw_intr_enable(struct cpsw_common *cpsw)
Mugunthan V Ndf828592012-03-18 20:17:54 +0000658{
Ivan Khoronzhuk5d8d0d42016-08-10 02:22:39 +0300659 __raw_writel(0xFF, &cpsw->wr_regs->tx_en);
660 __raw_writel(0xFF, &cpsw->wr_regs->rx_en);
Mugunthan V Ndf828592012-03-18 20:17:54 +0000661
Ivan Khoronzhuk2c836bd2016-08-10 02:22:40 +0300662 cpdma_ctlr_int_ctrl(cpsw->dma, true);
Mugunthan V Ndf828592012-03-18 20:17:54 +0000663 return;
664}
665
Ivan Khoronzhuk2c836bd2016-08-10 02:22:40 +0300666static void cpsw_intr_disable(struct cpsw_common *cpsw)
Mugunthan V Ndf828592012-03-18 20:17:54 +0000667{
Ivan Khoronzhuk5d8d0d42016-08-10 02:22:39 +0300668 __raw_writel(0, &cpsw->wr_regs->tx_en);
669 __raw_writel(0, &cpsw->wr_regs->rx_en);
Mugunthan V Ndf828592012-03-18 20:17:54 +0000670
Ivan Khoronzhuk2c836bd2016-08-10 02:22:40 +0300671 cpdma_ctlr_int_ctrl(cpsw->dma, false);
Mugunthan V Ndf828592012-03-18 20:17:54 +0000672 return;
673}
674
Olof Johansson1a3b5052013-12-11 15:58:07 -0800675static void cpsw_tx_handler(void *token, int len, int status)
Mugunthan V Ndf828592012-03-18 20:17:54 +0000676{
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +0300677 struct netdev_queue *txq;
Mugunthan V Ndf828592012-03-18 20:17:54 +0000678 struct sk_buff *skb = token;
679 struct net_device *ndev = skb->dev;
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +0300680 struct cpsw_common *cpsw = ndev_to_cpsw(ndev);
Mugunthan V Ndf828592012-03-18 20:17:54 +0000681
Mugunthan V Nfae50822013-01-17 06:31:34 +0000682 /* Check whether the queue is stopped due to stalled tx dma, if the
683 * queue is stopped then start the queue as we have free desc for tx
684 */
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +0300685 txq = netdev_get_tx_queue(ndev, skb_get_queue_mapping(skb));
686 if (unlikely(netif_tx_queue_stopped(txq)))
687 netif_tx_wake_queue(txq);
688
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +0300689 cpts_tx_timestamp(cpsw->cpts, skb);
Tobias Klauser8dc43dd2014-03-10 13:12:23 +0100690 ndev->stats.tx_packets++;
691 ndev->stats.tx_bytes += len;
Mugunthan V Ndf828592012-03-18 20:17:54 +0000692 dev_kfree_skb_any(skb);
693}
694
Olof Johansson1a3b5052013-12-11 15:58:07 -0800695static void cpsw_rx_handler(void *token, int len, int status)
Mugunthan V Ndf828592012-03-18 20:17:54 +0000696{
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +0300697 struct cpdma_chan *ch;
Mugunthan V Ndf828592012-03-18 20:17:54 +0000698 struct sk_buff *skb = token;
Sebastian Siewiorb4727e62013-04-23 07:31:39 +0000699 struct sk_buff *new_skb;
Mugunthan V Ndf828592012-03-18 20:17:54 +0000700 struct net_device *ndev = skb->dev;
Mugunthan V Ndf828592012-03-18 20:17:54 +0000701 int ret = 0;
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +0300702 struct cpsw_common *cpsw = ndev_to_cpsw(ndev);
Mugunthan V Ndf828592012-03-18 20:17:54 +0000703
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +0300704 cpsw_dual_emac_src_port_detect(cpsw, status, ndev, skb);
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +0000705
Mugunthan V N16e5c572014-04-10 14:23:23 +0530706 if (unlikely(status < 0) || unlikely(!netif_running(ndev))) {
Mugunthan V Na0e2c822014-09-10 16:38:09 +0530707 bool ndev_status = false;
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +0300708 struct cpsw_slave *slave = cpsw->slaves;
Mugunthan V Na0e2c822014-09-10 16:38:09 +0530709 int n;
710
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +0300711 if (cpsw->data.dual_emac) {
Mugunthan V Na0e2c822014-09-10 16:38:09 +0530712 /* In dual emac mode check for all interfaces */
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +0300713 for (n = cpsw->data.slaves; n; n--, slave++)
Mugunthan V Na0e2c822014-09-10 16:38:09 +0530714 if (netif_running(slave->ndev))
715 ndev_status = true;
716 }
717
718 if (ndev_status && (status >= 0)) {
719 /* The packet received is for the interface which
720 * is already down and the other interface is up
Joe Perchesdbedd442015-03-06 20:49:12 -0800721 * and running, instead of freeing which results
Mugunthan V Na0e2c822014-09-10 16:38:09 +0530722 * in reducing of the number of rx descriptor in
723 * DMA engine, requeue skb back to cpdma.
724 */
725 new_skb = skb;
726 goto requeue;
727 }
728
Sebastian Siewiorb4727e62013-04-23 07:31:39 +0000729 /* the interface is going down, skbs are purged */
Mugunthan V Ndf828592012-03-18 20:17:54 +0000730 dev_kfree_skb_any(skb);
731 return;
732 }
Sebastian Siewiorb4727e62013-04-23 07:31:39 +0000733
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +0300734 new_skb = netdev_alloc_skb_ip_align(ndev, cpsw->rx_packet_max);
Sebastian Siewiorb4727e62013-04-23 07:31:39 +0000735 if (new_skb) {
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +0300736 skb_copy_queue_mapping(new_skb, skb);
Mugunthan V Ndf828592012-03-18 20:17:54 +0000737 skb_put(skb, len);
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +0300738 cpts_rx_timestamp(cpsw->cpts, skb);
Mugunthan V Ndf828592012-03-18 20:17:54 +0000739 skb->protocol = eth_type_trans(skb, ndev);
740 netif_receive_skb(skb);
Tobias Klauser8dc43dd2014-03-10 13:12:23 +0100741 ndev->stats.rx_bytes += len;
742 ndev->stats.rx_packets++;
Grygorii Strashko254a49d2016-08-09 15:09:44 +0300743 kmemleak_not_leak(new_skb);
Sebastian Siewiorb4727e62013-04-23 07:31:39 +0000744 } else {
Tobias Klauser8dc43dd2014-03-10 13:12:23 +0100745 ndev->stats.rx_dropped++;
Sebastian Siewiorb4727e62013-04-23 07:31:39 +0000746 new_skb = skb;
Mugunthan V Ndf828592012-03-18 20:17:54 +0000747 }
748
Mugunthan V Na0e2c822014-09-10 16:38:09 +0530749requeue:
Ivan Khoronzhukce52c742016-08-22 21:18:28 +0300750 if (netif_dormant(ndev)) {
751 dev_kfree_skb_any(new_skb);
752 return;
753 }
754
Ivan Khoronzhuk8feb0a12016-11-29 17:00:51 +0200755 ch = cpsw->rxv[skb_get_queue_mapping(new_skb)].ch;
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +0300756 ret = cpdma_chan_submit(ch, new_skb, new_skb->data,
Ivan Khoronzhuk2c836bd2016-08-10 02:22:40 +0300757 skb_tailroom(new_skb), 0);
Sebastian Siewiorb4727e62013-04-23 07:31:39 +0000758 if (WARN_ON(ret < 0))
759 dev_kfree_skb_any(new_skb);
Mugunthan V Ndf828592012-03-18 20:17:54 +0000760}
761
Ivan Khoronzhuk32b78d82016-12-10 14:23:48 +0200762static void cpsw_split_res(struct net_device *ndev)
Ivan Khoronzhuk48e0a832016-12-06 03:45:00 +0200763{
764 struct cpsw_priv *priv = netdev_priv(ndev);
Ivan Khoronzhuk32b78d82016-12-10 14:23:48 +0200765 u32 consumed_rate = 0, bigest_rate = 0;
Ivan Khoronzhuk48e0a832016-12-06 03:45:00 +0200766 struct cpsw_common *cpsw = priv->cpsw;
767 struct cpsw_vector *txv = cpsw->txv;
Ivan Khoronzhuk32b78d82016-12-10 14:23:48 +0200768 int i, ch_weight, rlim_ch_num = 0;
Ivan Khoronzhuk48e0a832016-12-06 03:45:00 +0200769 int budget, bigest_rate_ch = 0;
Ivan Khoronzhuk48e0a832016-12-06 03:45:00 +0200770 u32 ch_rate, max_rate;
771 int ch_budget = 0;
772
Ivan Khoronzhuk48e0a832016-12-06 03:45:00 +0200773 for (i = 0; i < cpsw->tx_ch_num; i++) {
774 ch_rate = cpdma_chan_get_rate(txv[i].ch);
775 if (!ch_rate)
776 continue;
777
778 rlim_ch_num++;
779 consumed_rate += ch_rate;
780 }
781
782 if (cpsw->tx_ch_num == rlim_ch_num) {
783 max_rate = consumed_rate;
Ivan Khoronzhuk32b78d82016-12-10 14:23:48 +0200784 } else if (!rlim_ch_num) {
785 ch_budget = CPSW_POLL_WEIGHT / cpsw->tx_ch_num;
786 bigest_rate = 0;
787 max_rate = consumed_rate;
Ivan Khoronzhuk48e0a832016-12-06 03:45:00 +0200788 } else {
Ivan Khoronzhuk0be01b82016-12-10 14:23:49 +0200789 max_rate = cpsw->speed * 1000;
790
791 /* if max_rate is less then expected due to reduced link speed,
792 * split proportionally according next potential max speed
793 */
794 if (max_rate < consumed_rate)
795 max_rate *= 10;
796
797 if (max_rate < consumed_rate)
798 max_rate *= 10;
Ivan Khoronzhuk32b78d82016-12-10 14:23:48 +0200799
Ivan Khoronzhuk48e0a832016-12-06 03:45:00 +0200800 ch_budget = (consumed_rate * CPSW_POLL_WEIGHT) / max_rate;
801 ch_budget = (CPSW_POLL_WEIGHT - ch_budget) /
802 (cpsw->tx_ch_num - rlim_ch_num);
803 bigest_rate = (max_rate - consumed_rate) /
804 (cpsw->tx_ch_num - rlim_ch_num);
805 }
806
Ivan Khoronzhuk32b78d82016-12-10 14:23:48 +0200807 /* split tx weight/budget */
Ivan Khoronzhuk48e0a832016-12-06 03:45:00 +0200808 budget = CPSW_POLL_WEIGHT;
809 for (i = 0; i < cpsw->tx_ch_num; i++) {
810 ch_rate = cpdma_chan_get_rate(txv[i].ch);
811 if (ch_rate) {
812 txv[i].budget = (ch_rate * CPSW_POLL_WEIGHT) / max_rate;
813 if (!txv[i].budget)
Ivan Khoronzhuk32b78d82016-12-10 14:23:48 +0200814 txv[i].budget++;
Ivan Khoronzhuk48e0a832016-12-06 03:45:00 +0200815 if (ch_rate > bigest_rate) {
816 bigest_rate_ch = i;
817 bigest_rate = ch_rate;
818 }
Ivan Khoronzhuk32b78d82016-12-10 14:23:48 +0200819
820 ch_weight = (ch_rate * 100) / max_rate;
821 if (!ch_weight)
822 ch_weight++;
823 cpdma_chan_set_weight(cpsw->txv[i].ch, ch_weight);
Ivan Khoronzhuk48e0a832016-12-06 03:45:00 +0200824 } else {
825 txv[i].budget = ch_budget;
826 if (!bigest_rate_ch)
827 bigest_rate_ch = i;
Ivan Khoronzhuk32b78d82016-12-10 14:23:48 +0200828 cpdma_chan_set_weight(cpsw->txv[i].ch, 0);
Ivan Khoronzhuk48e0a832016-12-06 03:45:00 +0200829 }
830
831 budget -= txv[i].budget;
832 }
833
834 if (budget)
835 txv[bigest_rate_ch].budget += budget;
836
837 /* split rx budget */
838 budget = CPSW_POLL_WEIGHT;
839 ch_budget = budget / cpsw->rx_ch_num;
840 for (i = 0; i < cpsw->rx_ch_num; i++) {
841 cpsw->rxv[i].budget = ch_budget;
842 budget -= ch_budget;
843 }
844
845 if (budget)
846 cpsw->rxv[0].budget += budget;
847}
848
Felipe Balbic03abd82015-01-16 10:11:12 -0600849static irqreturn_t cpsw_tx_interrupt(int irq, void *dev_id)
Mugunthan V Ndf828592012-03-18 20:17:54 +0000850{
Ivan Khoronzhukdbc4ec52016-08-10 02:22:43 +0300851 struct cpsw_common *cpsw = dev_id;
Felipe Balbi7ce67a32015-01-02 16:15:59 -0600852
Ivan Khoronzhuk5d8d0d42016-08-10 02:22:39 +0300853 writel(0, &cpsw->wr_regs->tx_en);
Ivan Khoronzhuk2c836bd2016-08-10 02:22:40 +0300854 cpdma_ctlr_eoi(cpsw->dma, CPDMA_EOI_TX);
Felipe Balbic03abd82015-01-16 10:11:12 -0600855
Ivan Khoronzhuke38b5a32016-08-10 02:22:41 +0300856 if (cpsw->quirk_irq) {
857 disable_irq_nosync(cpsw->irqs_table[1]);
858 cpsw->tx_irq_disabled = true;
Mugunthan V N7da11602015-08-12 15:22:53 +0530859 }
860
Ivan Khoronzhukdbc4ec52016-08-10 02:22:43 +0300861 napi_schedule(&cpsw->napi_tx);
Felipe Balbic03abd82015-01-16 10:11:12 -0600862 return IRQ_HANDLED;
863}
864
865static irqreturn_t cpsw_rx_interrupt(int irq, void *dev_id)
866{
Ivan Khoronzhukdbc4ec52016-08-10 02:22:43 +0300867 struct cpsw_common *cpsw = dev_id;
Felipe Balbic03abd82015-01-16 10:11:12 -0600868
Ivan Khoronzhuk2c836bd2016-08-10 02:22:40 +0300869 cpdma_ctlr_eoi(cpsw->dma, CPDMA_EOI_RX);
Ivan Khoronzhuk5d8d0d42016-08-10 02:22:39 +0300870 writel(0, &cpsw->wr_regs->rx_en);
Sebastian Siewiorfd51cf12013-04-23 07:31:37 +0000871
Ivan Khoronzhuke38b5a32016-08-10 02:22:41 +0300872 if (cpsw->quirk_irq) {
873 disable_irq_nosync(cpsw->irqs_table[0]);
874 cpsw->rx_irq_disabled = true;
Mugunthan V N7da11602015-08-12 15:22:53 +0530875 }
876
Ivan Khoronzhukdbc4ec52016-08-10 02:22:43 +0300877 napi_schedule(&cpsw->napi_rx);
Mugunthan V Nd354eb82015-08-04 16:06:19 +0530878 return IRQ_HANDLED;
Mugunthan V Ndf828592012-03-18 20:17:54 +0000879}
880
Mugunthan V N32a74322015-08-04 16:06:20 +0530881static int cpsw_tx_poll(struct napi_struct *napi_tx, int budget)
Mugunthan V Ndf828592012-03-18 20:17:54 +0000882{
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +0300883 u32 ch_map;
Ivan Khoronzhuk8feb0a12016-11-29 17:00:51 +0200884 int num_tx, cur_budget, ch;
Ivan Khoronzhukdbc4ec52016-08-10 02:22:43 +0300885 struct cpsw_common *cpsw = napi_to_cpsw(napi_tx);
Ivan Khoronzhuk8feb0a12016-11-29 17:00:51 +0200886 struct cpsw_vector *txv;
Mugunthan V N32a74322015-08-04 16:06:20 +0530887
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +0300888 /* process every unprocessed channel */
889 ch_map = cpdma_ctrl_txchs_state(cpsw->dma);
Ivan Khoronzhuk342934a2016-11-29 17:00:50 +0200890 for (ch = 0, num_tx = 0; ch_map; ch_map >>= 1, ch++) {
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +0300891 if (!(ch_map & 0x01))
892 continue;
893
Ivan Khoronzhuk8feb0a12016-11-29 17:00:51 +0200894 txv = &cpsw->txv[ch];
895 if (unlikely(txv->budget > budget - num_tx))
896 cur_budget = budget - num_tx;
897 else
898 cur_budget = txv->budget;
899
900 num_tx += cpdma_chan_process(txv->ch, cur_budget);
Ivan Khoronzhuk342934a2016-11-29 17:00:50 +0200901 if (num_tx >= budget)
902 break;
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +0300903 }
904
Mugunthan V N32a74322015-08-04 16:06:20 +0530905 if (num_tx < budget) {
906 napi_complete(napi_tx);
Ivan Khoronzhuk5d8d0d42016-08-10 02:22:39 +0300907 writel(0xff, &cpsw->wr_regs->tx_en);
Ivan Khoronzhuke38b5a32016-08-10 02:22:41 +0300908 if (cpsw->quirk_irq && cpsw->tx_irq_disabled) {
909 cpsw->tx_irq_disabled = false;
910 enable_irq(cpsw->irqs_table[1]);
Mugunthan V N7da11602015-08-12 15:22:53 +0530911 }
Mugunthan V N32a74322015-08-04 16:06:20 +0530912 }
913
Mugunthan V N32a74322015-08-04 16:06:20 +0530914 return num_tx;
915}
916
917static int cpsw_rx_poll(struct napi_struct *napi_rx, int budget)
918{
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +0300919 u32 ch_map;
Ivan Khoronzhuk8feb0a12016-11-29 17:00:51 +0200920 int num_rx, cur_budget, ch;
Ivan Khoronzhukdbc4ec52016-08-10 02:22:43 +0300921 struct cpsw_common *cpsw = napi_to_cpsw(napi_rx);
Ivan Khoronzhuk8feb0a12016-11-29 17:00:51 +0200922 struct cpsw_vector *rxv;
Mugunthan V N510a1e722013-02-17 22:19:20 +0000923
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +0300924 /* process every unprocessed channel */
925 ch_map = cpdma_ctrl_rxchs_state(cpsw->dma);
Ivan Khoronzhuk342934a2016-11-29 17:00:50 +0200926 for (ch = 0, num_rx = 0; ch_map; ch_map >>= 1, ch++) {
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +0300927 if (!(ch_map & 0x01))
928 continue;
929
Ivan Khoronzhuk8feb0a12016-11-29 17:00:51 +0200930 rxv = &cpsw->rxv[ch];
931 if (unlikely(rxv->budget > budget - num_rx))
932 cur_budget = budget - num_rx;
933 else
934 cur_budget = rxv->budget;
935
936 num_rx += cpdma_chan_process(rxv->ch, cur_budget);
Ivan Khoronzhuk342934a2016-11-29 17:00:50 +0200937 if (num_rx >= budget)
938 break;
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +0300939 }
940
Mugunthan V N510a1e722013-02-17 22:19:20 +0000941 if (num_rx < budget) {
Mugunthan V N32a74322015-08-04 16:06:20 +0530942 napi_complete(napi_rx);
Ivan Khoronzhuk5d8d0d42016-08-10 02:22:39 +0300943 writel(0xff, &cpsw->wr_regs->rx_en);
Ivan Khoronzhuke38b5a32016-08-10 02:22:41 +0300944 if (cpsw->quirk_irq && cpsw->rx_irq_disabled) {
945 cpsw->rx_irq_disabled = false;
946 enable_irq(cpsw->irqs_table[0]);
Mugunthan V N7da11602015-08-12 15:22:53 +0530947 }
Mugunthan V N510a1e722013-02-17 22:19:20 +0000948 }
Mugunthan V Ndf828592012-03-18 20:17:54 +0000949
Mugunthan V Ndf828592012-03-18 20:17:54 +0000950 return num_rx;
951}
952
953static inline void soft_reset(const char *module, void __iomem *reg)
954{
955 unsigned long timeout = jiffies + HZ;
956
957 __raw_writel(1, reg);
958 do {
959 cpu_relax();
960 } while ((__raw_readl(reg) & 1) && time_after(timeout, jiffies));
961
962 WARN(__raw_readl(reg) & 1, "failed to soft-reset %s\n", module);
963}
964
965#define mac_hi(mac) (((mac)[0] << 0) | ((mac)[1] << 8) | \
966 ((mac)[2] << 16) | ((mac)[3] << 24))
967#define mac_lo(mac) (((mac)[4] << 0) | ((mac)[5] << 8))
968
969static void cpsw_set_slave_mac(struct cpsw_slave *slave,
970 struct cpsw_priv *priv)
971{
Richard Cochran9750a3a2012-10-29 08:45:15 +0000972 slave_write(slave, mac_hi(priv->mac_addr), SA_HI);
973 slave_write(slave, mac_lo(priv->mac_addr), SA_LO);
Mugunthan V Ndf828592012-03-18 20:17:54 +0000974}
975
976static void _cpsw_adjust_link(struct cpsw_slave *slave,
977 struct cpsw_priv *priv, bool *link)
978{
979 struct phy_device *phy = slave->phy;
980 u32 mac_control = 0;
981 u32 slave_port;
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +0300982 struct cpsw_common *cpsw = priv->cpsw;
Mugunthan V Ndf828592012-03-18 20:17:54 +0000983
984 if (!phy)
985 return;
986
Ivan Khoronzhuk6f1f5832016-08-10 02:22:34 +0300987 slave_port = cpsw_get_slave_port(slave->slave_num);
Mugunthan V Ndf828592012-03-18 20:17:54 +0000988
989 if (phy->link) {
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +0300990 mac_control = cpsw->data.mac_control;
Mugunthan V Ndf828592012-03-18 20:17:54 +0000991
992 /* enable forwarding */
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +0300993 cpsw_ale_control_set(cpsw->ale, slave_port,
Mugunthan V Ndf828592012-03-18 20:17:54 +0000994 ALE_PORT_STATE, ALE_PORT_STATE_FORWARD);
995
996 if (phy->speed == 1000)
997 mac_control |= BIT(7); /* GIGABITEN */
998 if (phy->duplex)
999 mac_control |= BIT(0); /* FULLDUPLEXEN */
Daniel Mack342b7b72012-09-27 09:19:34 +00001000
1001 /* set speed_in input in case RMII mode is used in 100Mbps */
1002 if (phy->speed == 100)
1003 mac_control |= BIT(15);
Mugunthan V Na81d8762013-12-13 18:42:55 +05301004 else if (phy->speed == 10)
1005 mac_control |= BIT(18); /* In Band mode */
Daniel Mack342b7b72012-09-27 09:19:34 +00001006
Mugunthan V N1923d6e2014-09-08 22:54:02 +05301007 if (priv->rx_pause)
1008 mac_control |= BIT(3);
1009
1010 if (priv->tx_pause)
1011 mac_control |= BIT(4);
1012
Mugunthan V Ndf828592012-03-18 20:17:54 +00001013 *link = true;
1014 } else {
1015 mac_control = 0;
1016 /* disable forwarding */
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001017 cpsw_ale_control_set(cpsw->ale, slave_port,
Mugunthan V Ndf828592012-03-18 20:17:54 +00001018 ALE_PORT_STATE, ALE_PORT_STATE_DISABLE);
1019 }
1020
1021 if (mac_control != slave->mac_control) {
1022 phy_print_status(phy);
1023 __raw_writel(mac_control, &slave->sliver->mac_control);
1024 }
1025
1026 slave->mac_control = mac_control;
1027}
1028
Ivan Khoronzhuk0be01b82016-12-10 14:23:49 +02001029static int cpsw_get_common_speed(struct cpsw_common *cpsw)
1030{
1031 int i, speed;
1032
1033 for (i = 0, speed = 0; i < cpsw->data.slaves; i++)
1034 if (cpsw->slaves[i].phy && cpsw->slaves[i].phy->link)
1035 speed += cpsw->slaves[i].phy->speed;
1036
1037 return speed;
1038}
1039
1040static int cpsw_need_resplit(struct cpsw_common *cpsw)
1041{
1042 int i, rlim_ch_num;
1043 int speed, ch_rate;
1044
1045 /* re-split resources only in case speed was changed */
1046 speed = cpsw_get_common_speed(cpsw);
1047 if (speed == cpsw->speed || !speed)
1048 return 0;
1049
1050 cpsw->speed = speed;
1051
1052 for (i = 0, rlim_ch_num = 0; i < cpsw->tx_ch_num; i++) {
1053 ch_rate = cpdma_chan_get_rate(cpsw->txv[i].ch);
1054 if (!ch_rate)
1055 break;
1056
1057 rlim_ch_num++;
1058 }
1059
1060 /* cases not dependent on speed */
1061 if (!rlim_ch_num || rlim_ch_num == cpsw->tx_ch_num)
1062 return 0;
1063
1064 return 1;
1065}
1066
Mugunthan V Ndf828592012-03-18 20:17:54 +00001067static void cpsw_adjust_link(struct net_device *ndev)
1068{
1069 struct cpsw_priv *priv = netdev_priv(ndev);
Ivan Khoronzhuk0be01b82016-12-10 14:23:49 +02001070 struct cpsw_common *cpsw = priv->cpsw;
Mugunthan V Ndf828592012-03-18 20:17:54 +00001071 bool link = false;
1072
1073 for_each_slave(priv, _cpsw_adjust_link, priv, &link);
1074
1075 if (link) {
Ivan Khoronzhuk0be01b82016-12-10 14:23:49 +02001076 if (cpsw_need_resplit(cpsw))
1077 cpsw_split_res(ndev);
1078
Mugunthan V Ndf828592012-03-18 20:17:54 +00001079 netif_carrier_on(ndev);
1080 if (netif_running(ndev))
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03001081 netif_tx_wake_all_queues(ndev);
Mugunthan V Ndf828592012-03-18 20:17:54 +00001082 } else {
1083 netif_carrier_off(ndev);
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03001084 netif_tx_stop_all_queues(ndev);
Mugunthan V Ndf828592012-03-18 20:17:54 +00001085 }
1086}
1087
Mugunthan V Nff5b8ef2013-03-11 23:16:37 +00001088static int cpsw_get_coalesce(struct net_device *ndev,
1089 struct ethtool_coalesce *coal)
1090{
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001091 struct cpsw_common *cpsw = ndev_to_cpsw(ndev);
Mugunthan V Nff5b8ef2013-03-11 23:16:37 +00001092
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001093 coal->rx_coalesce_usecs = cpsw->coal_intvl;
Mugunthan V Nff5b8ef2013-03-11 23:16:37 +00001094 return 0;
1095}
1096
1097static int cpsw_set_coalesce(struct net_device *ndev,
1098 struct ethtool_coalesce *coal)
1099{
1100 struct cpsw_priv *priv = netdev_priv(ndev);
1101 u32 int_ctrl;
1102 u32 num_interrupts = 0;
1103 u32 prescale = 0;
1104 u32 addnl_dvdr = 1;
1105 u32 coal_intvl = 0;
Ivan Khoronzhuk5d8d0d42016-08-10 02:22:39 +03001106 struct cpsw_common *cpsw = priv->cpsw;
Mugunthan V Nff5b8ef2013-03-11 23:16:37 +00001107
Mugunthan V Nff5b8ef2013-03-11 23:16:37 +00001108 coal_intvl = coal->rx_coalesce_usecs;
1109
Ivan Khoronzhuk5d8d0d42016-08-10 02:22:39 +03001110 int_ctrl = readl(&cpsw->wr_regs->int_control);
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001111 prescale = cpsw->bus_freq_mhz * 4;
Mugunthan V Nff5b8ef2013-03-11 23:16:37 +00001112
Mugunthan V Na84bc2a2014-07-15 20:26:53 +05301113 if (!coal->rx_coalesce_usecs) {
1114 int_ctrl &= ~(CPSW_INTPRESCALE_MASK | CPSW_INTPACEEN);
1115 goto update_return;
1116 }
1117
Mugunthan V Nff5b8ef2013-03-11 23:16:37 +00001118 if (coal_intvl < CPSW_CMINTMIN_INTVL)
1119 coal_intvl = CPSW_CMINTMIN_INTVL;
1120
1121 if (coal_intvl > CPSW_CMINTMAX_INTVL) {
1122 /* Interrupt pacer works with 4us Pulse, we can
1123 * throttle further by dilating the 4us pulse.
1124 */
1125 addnl_dvdr = CPSW_INTPRESCALE_MASK / prescale;
1126
1127 if (addnl_dvdr > 1) {
1128 prescale *= addnl_dvdr;
1129 if (coal_intvl > (CPSW_CMINTMAX_INTVL * addnl_dvdr))
1130 coal_intvl = (CPSW_CMINTMAX_INTVL
1131 * addnl_dvdr);
1132 } else {
1133 addnl_dvdr = 1;
1134 coal_intvl = CPSW_CMINTMAX_INTVL;
1135 }
1136 }
1137
1138 num_interrupts = (1000 * addnl_dvdr) / coal_intvl;
Ivan Khoronzhuk5d8d0d42016-08-10 02:22:39 +03001139 writel(num_interrupts, &cpsw->wr_regs->rx_imax);
1140 writel(num_interrupts, &cpsw->wr_regs->tx_imax);
Mugunthan V Nff5b8ef2013-03-11 23:16:37 +00001141
1142 int_ctrl |= CPSW_INTPACEEN;
1143 int_ctrl &= (~CPSW_INTPRESCALE_MASK);
1144 int_ctrl |= (prescale & CPSW_INTPRESCALE_MASK);
Mugunthan V Na84bc2a2014-07-15 20:26:53 +05301145
1146update_return:
Ivan Khoronzhuk5d8d0d42016-08-10 02:22:39 +03001147 writel(int_ctrl, &cpsw->wr_regs->int_control);
Mugunthan V Nff5b8ef2013-03-11 23:16:37 +00001148
1149 cpsw_notice(priv, timer, "Set coalesce to %d usecs.\n", coal_intvl);
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001150 cpsw->coal_intvl = coal_intvl;
Mugunthan V Nff5b8ef2013-03-11 23:16:37 +00001151
1152 return 0;
1153}
1154
Mugunthan V Nd9718542013-07-23 15:38:17 +05301155static int cpsw_get_sset_count(struct net_device *ndev, int sset)
1156{
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03001157 struct cpsw_common *cpsw = ndev_to_cpsw(ndev);
1158
Mugunthan V Nd9718542013-07-23 15:38:17 +05301159 switch (sset) {
1160 case ETH_SS_STATS:
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03001161 return (CPSW_STATS_COMMON_LEN +
1162 (cpsw->rx_ch_num + cpsw->tx_ch_num) *
1163 CPSW_STATS_CH_LEN);
Mugunthan V Nd9718542013-07-23 15:38:17 +05301164 default:
1165 return -EOPNOTSUPP;
1166 }
1167}
1168
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03001169static void cpsw_add_ch_strings(u8 **p, int ch_num, int rx_dir)
1170{
1171 int ch_stats_len;
1172 int line;
1173 int i;
1174
1175 ch_stats_len = CPSW_STATS_CH_LEN * ch_num;
1176 for (i = 0; i < ch_stats_len; i++) {
1177 line = i % CPSW_STATS_CH_LEN;
1178 snprintf(*p, ETH_GSTRING_LEN,
1179 "%s DMA chan %d: %s", rx_dir ? "Rx" : "Tx",
1180 i / CPSW_STATS_CH_LEN,
1181 cpsw_gstrings_ch_stats[line].stat_string);
1182 *p += ETH_GSTRING_LEN;
1183 }
1184}
1185
Mugunthan V Nd9718542013-07-23 15:38:17 +05301186static void cpsw_get_strings(struct net_device *ndev, u32 stringset, u8 *data)
1187{
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03001188 struct cpsw_common *cpsw = ndev_to_cpsw(ndev);
Mugunthan V Nd9718542013-07-23 15:38:17 +05301189 u8 *p = data;
1190 int i;
1191
1192 switch (stringset) {
1193 case ETH_SS_STATS:
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03001194 for (i = 0; i < CPSW_STATS_COMMON_LEN; i++) {
Mugunthan V Nd9718542013-07-23 15:38:17 +05301195 memcpy(p, cpsw_gstrings_stats[i].stat_string,
1196 ETH_GSTRING_LEN);
1197 p += ETH_GSTRING_LEN;
1198 }
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03001199
1200 cpsw_add_ch_strings(&p, cpsw->rx_ch_num, 1);
1201 cpsw_add_ch_strings(&p, cpsw->tx_ch_num, 0);
Mugunthan V Nd9718542013-07-23 15:38:17 +05301202 break;
1203 }
1204}
1205
1206static void cpsw_get_ethtool_stats(struct net_device *ndev,
1207 struct ethtool_stats *stats, u64 *data)
1208{
Mugunthan V Nd9718542013-07-23 15:38:17 +05301209 u8 *p;
Ivan Khoronzhuk2c836bd2016-08-10 02:22:40 +03001210 struct cpsw_common *cpsw = ndev_to_cpsw(ndev);
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03001211 struct cpdma_chan_stats ch_stats;
1212 int i, l, ch;
Mugunthan V Nd9718542013-07-23 15:38:17 +05301213
1214 /* Collect Davinci CPDMA stats for Rx and Tx Channel */
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03001215 for (l = 0; l < CPSW_STATS_COMMON_LEN; l++)
1216 data[l] = readl(cpsw->hw_stats +
1217 cpsw_gstrings_stats[l].stat_offset);
Mugunthan V Nd9718542013-07-23 15:38:17 +05301218
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03001219 for (ch = 0; ch < cpsw->rx_ch_num; ch++) {
Ivan Khoronzhuk8feb0a12016-11-29 17:00:51 +02001220 cpdma_chan_get_stats(cpsw->rxv[ch].ch, &ch_stats);
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03001221 for (i = 0; i < CPSW_STATS_CH_LEN; i++, l++) {
1222 p = (u8 *)&ch_stats +
1223 cpsw_gstrings_ch_stats[i].stat_offset;
1224 data[l] = *(u32 *)p;
1225 }
1226 }
Mugunthan V Nd9718542013-07-23 15:38:17 +05301227
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03001228 for (ch = 0; ch < cpsw->tx_ch_num; ch++) {
Ivan Khoronzhuk8feb0a12016-11-29 17:00:51 +02001229 cpdma_chan_get_stats(cpsw->txv[ch].ch, &ch_stats);
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03001230 for (i = 0; i < CPSW_STATS_CH_LEN; i++, l++) {
1231 p = (u8 *)&ch_stats +
1232 cpsw_gstrings_ch_stats[i].stat_offset;
1233 data[l] = *(u32 *)p;
Mugunthan V Nd9718542013-07-23 15:38:17 +05301234 }
1235 }
1236}
1237
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03001238static int cpsw_common_res_usage_state(struct cpsw_common *cpsw)
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00001239{
1240 u32 i;
1241 u32 usage_count = 0;
1242
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03001243 if (!cpsw->data.dual_emac)
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00001244 return 0;
1245
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03001246 for (i = 0; i < cpsw->data.slaves; i++)
1247 if (cpsw->slaves[i].open_stat)
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00001248 usage_count++;
1249
1250 return usage_count;
1251}
1252
Ivan Khoronzhuk27e9e102016-08-10 02:22:32 +03001253static inline int cpsw_tx_packet_submit(struct cpsw_priv *priv,
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03001254 struct sk_buff *skb,
1255 struct cpdma_chan *txch)
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00001256{
Ivan Khoronzhuk2c836bd2016-08-10 02:22:40 +03001257 struct cpsw_common *cpsw = priv->cpsw;
1258
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03001259 return cpdma_chan_submit(txch, skb, skb->data, skb->len,
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03001260 priv->emac_port + cpsw->data.dual_emac);
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00001261}
1262
1263static inline void cpsw_add_dual_emac_def_ale_entries(
1264 struct cpsw_priv *priv, struct cpsw_slave *slave,
1265 u32 slave_port)
1266{
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001267 struct cpsw_common *cpsw = priv->cpsw;
Grygorii Strashko71a2cbb2016-04-07 15:16:44 +03001268 u32 port_mask = 1 << slave_port | ALE_PORT_HOST;
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00001269
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001270 if (cpsw->version == CPSW_VERSION_1)
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00001271 slave_write(slave, slave->port_vlan, CPSW1_PORT_VLAN);
1272 else
1273 slave_write(slave, slave->port_vlan, CPSW2_PORT_VLAN);
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001274 cpsw_ale_add_vlan(cpsw->ale, slave->port_vlan, port_mask,
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00001275 port_mask, port_mask, 0);
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001276 cpsw_ale_add_mcast(cpsw->ale, priv->ndev->broadcast,
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00001277 port_mask, ALE_VLAN, slave->port_vlan, 0);
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001278 cpsw_ale_add_ucast(cpsw->ale, priv->mac_addr,
1279 HOST_PORT_NUM, ALE_VLAN |
1280 ALE_SECURE, slave->port_vlan);
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00001281}
1282
Daniel Mack1e7a2e22013-11-15 08:29:16 +01001283static void soft_reset_slave(struct cpsw_slave *slave)
Mugunthan V Ndf828592012-03-18 20:17:54 +00001284{
1285 char name[32];
Daniel Mack1e7a2e22013-11-15 08:29:16 +01001286
1287 snprintf(name, sizeof(name), "slave-%d", slave->slave_num);
1288 soft_reset(name, &slave->sliver->soft_reset);
1289}
1290
1291static void cpsw_slave_open(struct cpsw_slave *slave, struct cpsw_priv *priv)
1292{
Mugunthan V Ndf828592012-03-18 20:17:54 +00001293 u32 slave_port;
Ivan Khoronzhuk649a1682016-08-10 02:22:37 +03001294 struct cpsw_common *cpsw = priv->cpsw;
Mugunthan V Ndf828592012-03-18 20:17:54 +00001295
Daniel Mack1e7a2e22013-11-15 08:29:16 +01001296 soft_reset_slave(slave);
Mugunthan V Ndf828592012-03-18 20:17:54 +00001297
1298 /* setup priority mapping */
1299 __raw_writel(RX_PRIORITY_MAPPING, &slave->sliver->rx_pri_map);
Richard Cochran9750a3a2012-10-29 08:45:15 +00001300
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001301 switch (cpsw->version) {
Richard Cochran9750a3a2012-10-29 08:45:15 +00001302 case CPSW_VERSION_1:
1303 slave_write(slave, TX_PRIORITY_MAPPING, CPSW1_TX_PRI_MAP);
1304 break;
1305 case CPSW_VERSION_2:
Mugunthan V Nc193f362013-08-05 17:30:05 +05301306 case CPSW_VERSION_3:
Mugunthan V N926489b2013-08-12 17:11:15 +05301307 case CPSW_VERSION_4:
Richard Cochran9750a3a2012-10-29 08:45:15 +00001308 slave_write(slave, TX_PRIORITY_MAPPING, CPSW2_TX_PRI_MAP);
1309 break;
1310 }
Mugunthan V Ndf828592012-03-18 20:17:54 +00001311
1312 /* setup max packet size, and mac address */
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001313 __raw_writel(cpsw->rx_packet_max, &slave->sliver->rx_maxlen);
Mugunthan V Ndf828592012-03-18 20:17:54 +00001314 cpsw_set_slave_mac(slave, priv);
1315
1316 slave->mac_control = 0; /* no link yet */
1317
Ivan Khoronzhuk6f1f5832016-08-10 02:22:34 +03001318 slave_port = cpsw_get_slave_port(slave->slave_num);
Mugunthan V Ndf828592012-03-18 20:17:54 +00001319
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03001320 if (cpsw->data.dual_emac)
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00001321 cpsw_add_dual_emac_def_ale_entries(priv, slave, slave_port);
1322 else
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001323 cpsw_ale_add_mcast(cpsw->ale, priv->ndev->broadcast,
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00001324 1 << slave_port, 0, 0, ALE_MCAST_FWD_2);
Mugunthan V Ndf828592012-03-18 20:17:54 +00001325
David Rivshind733f7542016-04-27 21:32:31 -04001326 if (slave->data->phy_node) {
David Rivshin552165b2016-04-27 21:25:25 -04001327 slave->phy = of_phy_connect(priv->ndev, slave->data->phy_node,
Heiko Schocher9e42f712015-10-17 06:04:35 +02001328 &cpsw_adjust_link, 0, slave->data->phy_if);
David Rivshind733f7542016-04-27 21:32:31 -04001329 if (!slave->phy) {
1330 dev_err(priv->dev, "phy \"%s\" not found on slave %d\n",
1331 slave->data->phy_node->full_name,
1332 slave->slave_num);
1333 return;
1334 }
1335 } else {
Heiko Schocher9e42f712015-10-17 06:04:35 +02001336 slave->phy = phy_connect(priv->ndev, slave->data->phy_id,
Florian Fainellif9a8f832013-01-14 00:52:52 +00001337 &cpsw_adjust_link, slave->data->phy_if);
David Rivshind733f7542016-04-27 21:32:31 -04001338 if (IS_ERR(slave->phy)) {
1339 dev_err(priv->dev,
1340 "phy \"%s\" not found on slave %d, err %ld\n",
1341 slave->data->phy_id, slave->slave_num,
1342 PTR_ERR(slave->phy));
1343 slave->phy = NULL;
1344 return;
1345 }
Mugunthan V Ndf828592012-03-18 20:17:54 +00001346 }
David Rivshind733f7542016-04-27 21:32:31 -04001347
1348 phy_attached_info(slave->phy);
1349
1350 phy_start(slave->phy);
1351
1352 /* Configure GMII_SEL register */
Ivan Khoronzhuk56e31bd2016-08-10 02:22:38 +03001353 cpsw_phy_sel(cpsw->dev, slave->phy->interface, slave->slave_num);
Mugunthan V Ndf828592012-03-18 20:17:54 +00001354}
1355
Mugunthan V N3b72c2f2013-02-05 08:26:48 +00001356static inline void cpsw_add_default_vlan(struct cpsw_priv *priv)
1357{
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03001358 struct cpsw_common *cpsw = priv->cpsw;
1359 const int vlan = cpsw->data.default_vlan;
Mugunthan V N3b72c2f2013-02-05 08:26:48 +00001360 u32 reg;
1361 int i;
Lennart Sorensen1e5c4bc2014-10-31 13:38:52 -04001362 int unreg_mcast_mask;
Mugunthan V N3b72c2f2013-02-05 08:26:48 +00001363
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001364 reg = (cpsw->version == CPSW_VERSION_1) ? CPSW1_PORT_VLAN :
Mugunthan V N3b72c2f2013-02-05 08:26:48 +00001365 CPSW2_PORT_VLAN;
1366
Ivan Khoronzhuk5d8d0d42016-08-10 02:22:39 +03001367 writel(vlan, &cpsw->host_port_regs->port_vlan);
Mugunthan V N3b72c2f2013-02-05 08:26:48 +00001368
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03001369 for (i = 0; i < cpsw->data.slaves; i++)
1370 slave_write(cpsw->slaves + i, vlan, reg);
Mugunthan V N3b72c2f2013-02-05 08:26:48 +00001371
Lennart Sorensen1e5c4bc2014-10-31 13:38:52 -04001372 if (priv->ndev->flags & IFF_ALLMULTI)
1373 unreg_mcast_mask = ALE_ALL_PORTS;
1374 else
1375 unreg_mcast_mask = ALE_PORT_1 | ALE_PORT_2;
1376
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001377 cpsw_ale_add_vlan(cpsw->ale, vlan, ALE_ALL_PORTS,
Grygorii Strashko61f1cef2016-04-07 15:16:43 +03001378 ALE_ALL_PORTS, ALE_ALL_PORTS,
1379 unreg_mcast_mask);
Mugunthan V N3b72c2f2013-02-05 08:26:48 +00001380}
1381
Mugunthan V Ndf828592012-03-18 20:17:54 +00001382static void cpsw_init_host_port(struct cpsw_priv *priv)
1383{
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00001384 u32 fifo_mode;
Ivan Khoronzhuk5d8d0d42016-08-10 02:22:39 +03001385 u32 control_reg;
1386 struct cpsw_common *cpsw = priv->cpsw;
Mugunthan V N3b72c2f2013-02-05 08:26:48 +00001387
Mugunthan V Ndf828592012-03-18 20:17:54 +00001388 /* soft reset the controller and initialize ale */
Ivan Khoronzhuk5d8d0d42016-08-10 02:22:39 +03001389 soft_reset("cpsw", &cpsw->regs->soft_reset);
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001390 cpsw_ale_start(cpsw->ale);
Mugunthan V Ndf828592012-03-18 20:17:54 +00001391
1392 /* switch to vlan unaware mode */
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001393 cpsw_ale_control_set(cpsw->ale, HOST_PORT_NUM, ALE_VLAN_AWARE,
Mugunthan V N3b72c2f2013-02-05 08:26:48 +00001394 CPSW_ALE_VLAN_AWARE);
Ivan Khoronzhuk5d8d0d42016-08-10 02:22:39 +03001395 control_reg = readl(&cpsw->regs->control);
Mugunthan V N3b72c2f2013-02-05 08:26:48 +00001396 control_reg |= CPSW_VLAN_AWARE;
Ivan Khoronzhuk5d8d0d42016-08-10 02:22:39 +03001397 writel(control_reg, &cpsw->regs->control);
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03001398 fifo_mode = (cpsw->data.dual_emac) ? CPSW_FIFO_DUAL_MAC_MODE :
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00001399 CPSW_FIFO_NORMAL_MODE;
Ivan Khoronzhuk5d8d0d42016-08-10 02:22:39 +03001400 writel(fifo_mode, &cpsw->host_port_regs->tx_in_ctl);
Mugunthan V Ndf828592012-03-18 20:17:54 +00001401
1402 /* setup host port priority mapping */
1403 __raw_writel(CPDMA_TX_PRIORITY_MAP,
Ivan Khoronzhuk5d8d0d42016-08-10 02:22:39 +03001404 &cpsw->host_port_regs->cpdma_tx_pri_map);
1405 __raw_writel(0, &cpsw->host_port_regs->cpdma_rx_chan_map);
Mugunthan V Ndf828592012-03-18 20:17:54 +00001406
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001407 cpsw_ale_control_set(cpsw->ale, HOST_PORT_NUM,
Mugunthan V Ndf828592012-03-18 20:17:54 +00001408 ALE_PORT_STATE, ALE_PORT_STATE_FORWARD);
1409
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03001410 if (!cpsw->data.dual_emac) {
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001411 cpsw_ale_add_ucast(cpsw->ale, priv->mac_addr, HOST_PORT_NUM,
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00001412 0, 0);
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001413 cpsw_ale_add_mcast(cpsw->ale, priv->ndev->broadcast,
Grygorii Strashko71a2cbb2016-04-07 15:16:44 +03001414 ALE_PORT_HOST, 0, 0, ALE_MCAST_FWD_2);
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00001415 }
Mugunthan V Ndf828592012-03-18 20:17:54 +00001416}
1417
Ivan Khoronzhuk3802dce12016-08-22 21:18:24 +03001418static int cpsw_fill_rx_channels(struct cpsw_priv *priv)
1419{
1420 struct cpsw_common *cpsw = priv->cpsw;
1421 struct sk_buff *skb;
1422 int ch_buf_num;
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03001423 int ch, i, ret;
Ivan Khoronzhuk3802dce12016-08-22 21:18:24 +03001424
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03001425 for (ch = 0; ch < cpsw->rx_ch_num; ch++) {
Ivan Khoronzhuk8feb0a12016-11-29 17:00:51 +02001426 ch_buf_num = cpdma_chan_get_rx_buf_num(cpsw->rxv[ch].ch);
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03001427 for (i = 0; i < ch_buf_num; i++) {
1428 skb = __netdev_alloc_skb_ip_align(priv->ndev,
1429 cpsw->rx_packet_max,
1430 GFP_KERNEL);
1431 if (!skb) {
1432 cpsw_err(priv, ifup, "cannot allocate skb\n");
1433 return -ENOMEM;
1434 }
1435
1436 skb_set_queue_mapping(skb, ch);
Ivan Khoronzhuk8feb0a12016-11-29 17:00:51 +02001437 ret = cpdma_chan_submit(cpsw->rxv[ch].ch, skb,
1438 skb->data, skb_tailroom(skb),
1439 0);
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03001440 if (ret < 0) {
1441 cpsw_err(priv, ifup,
1442 "cannot submit skb to channel %d rx, error %d\n",
1443 ch, ret);
1444 kfree_skb(skb);
1445 return ret;
1446 }
1447 kmemleak_not_leak(skb);
Ivan Khoronzhuk3802dce12016-08-22 21:18:24 +03001448 }
1449
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03001450 cpsw_info(priv, ifup, "ch %d rx, submitted %d descriptors\n",
1451 ch, ch_buf_num);
Ivan Khoronzhuk3802dce12016-08-22 21:18:24 +03001452 }
1453
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03001454 return 0;
Ivan Khoronzhuk3802dce12016-08-22 21:18:24 +03001455}
1456
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001457static void cpsw_slave_stop(struct cpsw_slave *slave, struct cpsw_common *cpsw)
Sebastian Siewioraacebbf2013-04-23 07:31:36 +00001458{
Schuyler Patton3995d262014-03-03 16:19:06 +05301459 u32 slave_port;
1460
Ivan Khoronzhuk6f1f5832016-08-10 02:22:34 +03001461 slave_port = cpsw_get_slave_port(slave->slave_num);
Schuyler Patton3995d262014-03-03 16:19:06 +05301462
Sebastian Siewioraacebbf2013-04-23 07:31:36 +00001463 if (!slave->phy)
1464 return;
1465 phy_stop(slave->phy);
1466 phy_disconnect(slave->phy);
1467 slave->phy = NULL;
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001468 cpsw_ale_control_set(cpsw->ale, slave_port,
Schuyler Patton3995d262014-03-03 16:19:06 +05301469 ALE_PORT_STATE, ALE_PORT_STATE_DISABLE);
Grygorii Strashko1f95ba02016-06-24 21:23:41 +03001470 soft_reset_slave(slave);
Sebastian Siewioraacebbf2013-04-23 07:31:36 +00001471}
1472
Mugunthan V Ndf828592012-03-18 20:17:54 +00001473static int cpsw_ndo_open(struct net_device *ndev)
1474{
1475 struct cpsw_priv *priv = netdev_priv(ndev);
Ivan Khoronzhuk649a1682016-08-10 02:22:37 +03001476 struct cpsw_common *cpsw = priv->cpsw;
Ivan Khoronzhuk3802dce12016-08-22 21:18:24 +03001477 int ret;
Mugunthan V Ndf828592012-03-18 20:17:54 +00001478 u32 reg;
1479
Ivan Khoronzhuk56e31bd2016-08-10 02:22:38 +03001480 ret = pm_runtime_get_sync(cpsw->dev);
Grygorii Strashko108a6532016-06-24 21:23:42 +03001481 if (ret < 0) {
Ivan Khoronzhuk56e31bd2016-08-10 02:22:38 +03001482 pm_runtime_put_noidle(cpsw->dev);
Grygorii Strashko108a6532016-06-24 21:23:42 +03001483 return ret;
1484 }
Grygorii Strashko3fa88c52016-04-19 21:09:49 +03001485
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03001486 if (!cpsw_common_res_usage_state(cpsw))
Ivan Khoronzhuk2c836bd2016-08-10 02:22:40 +03001487 cpsw_intr_disable(cpsw);
Mugunthan V Ndf828592012-03-18 20:17:54 +00001488 netif_carrier_off(ndev);
1489
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03001490 /* Notify the stack of the actual queue counts. */
1491 ret = netif_set_real_num_tx_queues(ndev, cpsw->tx_ch_num);
1492 if (ret) {
1493 dev_err(priv->dev, "cannot set real number of tx queues\n");
1494 goto err_cleanup;
1495 }
1496
1497 ret = netif_set_real_num_rx_queues(ndev, cpsw->rx_ch_num);
1498 if (ret) {
1499 dev_err(priv->dev, "cannot set real number of rx queues\n");
1500 goto err_cleanup;
1501 }
1502
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001503 reg = cpsw->version;
Mugunthan V Ndf828592012-03-18 20:17:54 +00001504
1505 dev_info(priv->dev, "initializing cpsw version %d.%d (%d)\n",
1506 CPSW_MAJOR_VERSION(reg), CPSW_MINOR_VERSION(reg),
1507 CPSW_RTL_VERSION(reg));
1508
1509 /* initialize host and slave ports */
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03001510 if (!cpsw_common_res_usage_state(cpsw))
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00001511 cpsw_init_host_port(priv);
Mugunthan V Ndf828592012-03-18 20:17:54 +00001512 for_each_slave(priv, cpsw_slave_open, priv);
1513
Mugunthan V N3b72c2f2013-02-05 08:26:48 +00001514 /* Add default VLAN */
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03001515 if (!cpsw->data.dual_emac)
Mugunthan V Ne6afea02014-06-18 17:21:48 +05301516 cpsw_add_default_vlan(priv);
1517 else
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001518 cpsw_ale_add_vlan(cpsw->ale, cpsw->data.default_vlan,
Grygorii Strashko61f1cef2016-04-07 15:16:43 +03001519 ALE_ALL_PORTS, ALE_ALL_PORTS, 0, 0);
Mugunthan V N3b72c2f2013-02-05 08:26:48 +00001520
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03001521 if (!cpsw_common_res_usage_state(cpsw)) {
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00001522 /* disable priority elevation */
Ivan Khoronzhuk5d8d0d42016-08-10 02:22:39 +03001523 __raw_writel(0, &cpsw->regs->ptype);
Mugunthan V Ndf828592012-03-18 20:17:54 +00001524
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00001525 /* enable statistics collection only on all ports */
Ivan Khoronzhuk5d8d0d42016-08-10 02:22:39 +03001526 __raw_writel(0x7, &cpsw->regs->stat_port_en);
Mugunthan V Ndf828592012-03-18 20:17:54 +00001527
Mugunthan V N1923d6e2014-09-08 22:54:02 +05301528 /* Enable internal fifo flow control */
Ivan Khoronzhuk5d8d0d42016-08-10 02:22:39 +03001529 writel(0x7, &cpsw->regs->flow_control);
Mugunthan V N1923d6e2014-09-08 22:54:02 +05301530
Ivan Khoronzhukdbc4ec52016-08-10 02:22:43 +03001531 napi_enable(&cpsw->napi_rx);
1532 napi_enable(&cpsw->napi_tx);
Mugunthan V Nd354eb82015-08-04 16:06:19 +05301533
Ivan Khoronzhuke38b5a32016-08-10 02:22:41 +03001534 if (cpsw->tx_irq_disabled) {
1535 cpsw->tx_irq_disabled = false;
1536 enable_irq(cpsw->irqs_table[1]);
Mugunthan V N7da11602015-08-12 15:22:53 +05301537 }
1538
Ivan Khoronzhuke38b5a32016-08-10 02:22:41 +03001539 if (cpsw->rx_irq_disabled) {
1540 cpsw->rx_irq_disabled = false;
1541 enable_irq(cpsw->irqs_table[0]);
Mugunthan V N7da11602015-08-12 15:22:53 +05301542 }
1543
Ivan Khoronzhuk3802dce12016-08-22 21:18:24 +03001544 ret = cpsw_fill_rx_channels(priv);
1545 if (ret < 0)
1546 goto err_cleanup;
Mugunthan V Nf280e892013-12-11 22:09:05 -06001547
Grygorii Strashko8a2c9a52016-12-06 18:00:41 -06001548 if (cpts_register(cpsw->cpts))
Mugunthan V Nf280e892013-12-11 22:09:05 -06001549 dev_err(priv->dev, "error registering cpts device\n");
1550
Mugunthan V Ndf828592012-03-18 20:17:54 +00001551 }
Mugunthan V Ndf828592012-03-18 20:17:54 +00001552
Mugunthan V Nff5b8ef2013-03-11 23:16:37 +00001553 /* Enable Interrupt pacing if configured */
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001554 if (cpsw->coal_intvl != 0) {
Mugunthan V Nff5b8ef2013-03-11 23:16:37 +00001555 struct ethtool_coalesce coal;
1556
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001557 coal.rx_coalesce_usecs = cpsw->coal_intvl;
Mugunthan V Nff5b8ef2013-03-11 23:16:37 +00001558 cpsw_set_coalesce(ndev, &coal);
1559 }
1560
Ivan Khoronzhuk2c836bd2016-08-10 02:22:40 +03001561 cpdma_ctlr_start(cpsw->dma);
1562 cpsw_intr_enable(cpsw);
Mugunthan V Nf63a9752014-04-10 14:23:24 +05301563
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03001564 if (cpsw->data.dual_emac)
1565 cpsw->slaves[priv->emac_port].open_stat = true;
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03001566
Mugunthan V Ndf828592012-03-18 20:17:54 +00001567 return 0;
Mugunthan V Ndf828592012-03-18 20:17:54 +00001568
Sebastian Siewioraacebbf2013-04-23 07:31:36 +00001569err_cleanup:
Ivan Khoronzhuk2c836bd2016-08-10 02:22:40 +03001570 cpdma_ctlr_stop(cpsw->dma);
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001571 for_each_slave(priv, cpsw_slave_stop, cpsw);
Ivan Khoronzhuk56e31bd2016-08-10 02:22:38 +03001572 pm_runtime_put_sync(cpsw->dev);
Sebastian Siewioraacebbf2013-04-23 07:31:36 +00001573 netif_carrier_off(priv->ndev);
1574 return ret;
Mugunthan V Ndf828592012-03-18 20:17:54 +00001575}
1576
1577static int cpsw_ndo_stop(struct net_device *ndev)
1578{
1579 struct cpsw_priv *priv = netdev_priv(ndev);
Ivan Khoronzhuk649a1682016-08-10 02:22:37 +03001580 struct cpsw_common *cpsw = priv->cpsw;
Mugunthan V Ndf828592012-03-18 20:17:54 +00001581
1582 cpsw_info(priv, ifdown, "shutting down cpsw device\n");
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03001583 netif_tx_stop_all_queues(priv->ndev);
Mugunthan V Ndf828592012-03-18 20:17:54 +00001584 netif_carrier_off(priv->ndev);
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00001585
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03001586 if (cpsw_common_res_usage_state(cpsw) <= 1) {
Ivan Khoronzhukdbc4ec52016-08-10 02:22:43 +03001587 napi_disable(&cpsw->napi_rx);
1588 napi_disable(&cpsw->napi_tx);
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001589 cpts_unregister(cpsw->cpts);
Ivan Khoronzhuk2c836bd2016-08-10 02:22:40 +03001590 cpsw_intr_disable(cpsw);
1591 cpdma_ctlr_stop(cpsw->dma);
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001592 cpsw_ale_stop(cpsw->ale);
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00001593 }
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001594 for_each_slave(priv, cpsw_slave_stop, cpsw);
Ivan Khoronzhuk0be01b82016-12-10 14:23:49 +02001595
1596 if (cpsw_need_resplit(cpsw))
1597 cpsw_split_res(ndev);
1598
Ivan Khoronzhuk56e31bd2016-08-10 02:22:38 +03001599 pm_runtime_put_sync(cpsw->dev);
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03001600 if (cpsw->data.dual_emac)
1601 cpsw->slaves[priv->emac_port].open_stat = false;
Mugunthan V Ndf828592012-03-18 20:17:54 +00001602 return 0;
1603}
1604
1605static netdev_tx_t cpsw_ndo_start_xmit(struct sk_buff *skb,
1606 struct net_device *ndev)
1607{
1608 struct cpsw_priv *priv = netdev_priv(ndev);
Ivan Khoronzhuk2c836bd2016-08-10 02:22:40 +03001609 struct cpsw_common *cpsw = priv->cpsw;
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03001610 struct netdev_queue *txq;
1611 struct cpdma_chan *txch;
1612 int ret, q_idx;
Mugunthan V Ndf828592012-03-18 20:17:54 +00001613
Florian Westphal860e9532016-05-03 16:33:13 +02001614 netif_trans_update(ndev);
Mugunthan V Ndf828592012-03-18 20:17:54 +00001615
1616 if (skb_padto(skb, CPSW_MIN_PACKET_SIZE)) {
1617 cpsw_err(priv, tx_err, "packet pad failed\n");
Tobias Klauser8dc43dd2014-03-10 13:12:23 +01001618 ndev->stats.tx_dropped++;
Mugunthan V Ndf828592012-03-18 20:17:54 +00001619 return NETDEV_TX_OK;
1620 }
1621
Mugunthan V N9232b162013-02-11 09:52:19 +00001622 if (skb_shinfo(skb)->tx_flags & SKBTX_HW_TSTAMP &&
Grygorii Strashkob63ba582016-12-06 18:00:35 -06001623 cpts_is_tx_enabled(cpsw->cpts))
Richard Cochran2e5b38a2012-10-29 08:45:20 +00001624 skb_shinfo(skb)->tx_flags |= SKBTX_IN_PROGRESS;
1625
1626 skb_tx_timestamp(skb);
1627
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03001628 q_idx = skb_get_queue_mapping(skb);
1629 if (q_idx >= cpsw->tx_ch_num)
1630 q_idx = q_idx % cpsw->tx_ch_num;
1631
Ivan Khoronzhuk8feb0a12016-11-29 17:00:51 +02001632 txch = cpsw->txv[q_idx].ch;
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03001633 ret = cpsw_tx_packet_submit(priv, skb, txch);
Mugunthan V Ndf828592012-03-18 20:17:54 +00001634 if (unlikely(ret != 0)) {
1635 cpsw_err(priv, tx_err, "desc submit failed\n");
1636 goto fail;
1637 }
1638
Mugunthan V Nfae50822013-01-17 06:31:34 +00001639 /* If there is no more tx desc left free then we need to
1640 * tell the kernel to stop sending us tx frames.
1641 */
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03001642 if (unlikely(!cpdma_check_free_tx_desc(txch))) {
1643 txq = netdev_get_tx_queue(ndev, q_idx);
1644 netif_tx_stop_queue(txq);
1645 }
Mugunthan V Nfae50822013-01-17 06:31:34 +00001646
Mugunthan V Ndf828592012-03-18 20:17:54 +00001647 return NETDEV_TX_OK;
1648fail:
Tobias Klauser8dc43dd2014-03-10 13:12:23 +01001649 ndev->stats.tx_dropped++;
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03001650 txq = netdev_get_tx_queue(ndev, skb_get_queue_mapping(skb));
1651 netif_tx_stop_queue(txq);
Mugunthan V Ndf828592012-03-18 20:17:54 +00001652 return NETDEV_TX_BUSY;
1653}
1654
Grygorii Strashkoc8395d42016-12-06 18:00:34 -06001655#if IS_ENABLED(CONFIG_TI_CPTS)
Richard Cochran2e5b38a2012-10-29 08:45:20 +00001656
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001657static void cpsw_hwtstamp_v1(struct cpsw_common *cpsw)
Richard Cochran2e5b38a2012-10-29 08:45:20 +00001658{
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03001659 struct cpsw_slave *slave = &cpsw->slaves[cpsw->data.active_slave];
Richard Cochran2e5b38a2012-10-29 08:45:20 +00001660 u32 ts_en, seq_id;
1661
Grygorii Strashkob63ba582016-12-06 18:00:35 -06001662 if (!cpts_is_tx_enabled(cpsw->cpts) &&
1663 !cpts_is_rx_enabled(cpsw->cpts)) {
Richard Cochran2e5b38a2012-10-29 08:45:20 +00001664 slave_write(slave, 0, CPSW1_TS_CTL);
1665 return;
1666 }
1667
1668 seq_id = (30 << CPSW_V1_SEQ_ID_OFS_SHIFT) | ETH_P_1588;
1669 ts_en = EVENT_MSG_BITS << CPSW_V1_MSG_TYPE_OFS;
1670
Grygorii Strashkob63ba582016-12-06 18:00:35 -06001671 if (cpts_is_tx_enabled(cpsw->cpts))
Richard Cochran2e5b38a2012-10-29 08:45:20 +00001672 ts_en |= CPSW_V1_TS_TX_EN;
1673
Grygorii Strashkob63ba582016-12-06 18:00:35 -06001674 if (cpts_is_rx_enabled(cpsw->cpts))
Richard Cochran2e5b38a2012-10-29 08:45:20 +00001675 ts_en |= CPSW_V1_TS_RX_EN;
1676
1677 slave_write(slave, ts_en, CPSW1_TS_CTL);
1678 slave_write(slave, seq_id, CPSW1_TS_SEQ_LTYPE);
1679}
1680
1681static void cpsw_hwtstamp_v2(struct cpsw_priv *priv)
1682{
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00001683 struct cpsw_slave *slave;
Ivan Khoronzhuk5d8d0d42016-08-10 02:22:39 +03001684 struct cpsw_common *cpsw = priv->cpsw;
Richard Cochran2e5b38a2012-10-29 08:45:20 +00001685 u32 ctrl, mtype;
1686
Ivan Khoronzhukcb7d78d02016-12-10 14:23:46 +02001687 slave = &cpsw->slaves[cpsw_slave_index(cpsw, priv)];
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00001688
Richard Cochran2e5b38a2012-10-29 08:45:20 +00001689 ctrl = slave_read(slave, CPSW2_CONTROL);
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001690 switch (cpsw->version) {
George Cherian09c55372014-05-02 12:02:02 +05301691 case CPSW_VERSION_2:
1692 ctrl &= ~CTRL_V2_ALL_TS_MASK;
Richard Cochran2e5b38a2012-10-29 08:45:20 +00001693
Grygorii Strashkob63ba582016-12-06 18:00:35 -06001694 if (cpts_is_tx_enabled(cpsw->cpts))
George Cherian09c55372014-05-02 12:02:02 +05301695 ctrl |= CTRL_V2_TX_TS_BITS;
Richard Cochran2e5b38a2012-10-29 08:45:20 +00001696
Grygorii Strashkob63ba582016-12-06 18:00:35 -06001697 if (cpts_is_rx_enabled(cpsw->cpts))
George Cherian09c55372014-05-02 12:02:02 +05301698 ctrl |= CTRL_V2_RX_TS_BITS;
Richard Cochran26fe7eb2015-05-25 11:02:13 +02001699 break;
George Cherian09c55372014-05-02 12:02:02 +05301700 case CPSW_VERSION_3:
1701 default:
1702 ctrl &= ~CTRL_V3_ALL_TS_MASK;
1703
Grygorii Strashkob63ba582016-12-06 18:00:35 -06001704 if (cpts_is_tx_enabled(cpsw->cpts))
George Cherian09c55372014-05-02 12:02:02 +05301705 ctrl |= CTRL_V3_TX_TS_BITS;
1706
Grygorii Strashkob63ba582016-12-06 18:00:35 -06001707 if (cpts_is_rx_enabled(cpsw->cpts))
George Cherian09c55372014-05-02 12:02:02 +05301708 ctrl |= CTRL_V3_RX_TS_BITS;
Richard Cochran26fe7eb2015-05-25 11:02:13 +02001709 break;
George Cherian09c55372014-05-02 12:02:02 +05301710 }
Richard Cochran2e5b38a2012-10-29 08:45:20 +00001711
1712 mtype = (30 << TS_SEQ_ID_OFFSET_SHIFT) | EVENT_MSG_BITS;
1713
1714 slave_write(slave, mtype, CPSW2_TS_SEQ_MTYPE);
1715 slave_write(slave, ctrl, CPSW2_CONTROL);
Ivan Khoronzhuk5d8d0d42016-08-10 02:22:39 +03001716 __raw_writel(ETH_P_1588, &cpsw->regs->ts_ltype);
Richard Cochran2e5b38a2012-10-29 08:45:20 +00001717}
1718
Ben Hutchingsa5b41452013-11-18 23:23:40 +00001719static int cpsw_hwtstamp_set(struct net_device *dev, struct ifreq *ifr)
Richard Cochran2e5b38a2012-10-29 08:45:20 +00001720{
Mugunthan V N3177bf62012-11-27 07:53:40 +00001721 struct cpsw_priv *priv = netdev_priv(dev);
Richard Cochran2e5b38a2012-10-29 08:45:20 +00001722 struct hwtstamp_config cfg;
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001723 struct cpsw_common *cpsw = priv->cpsw;
1724 struct cpts *cpts = cpsw->cpts;
Richard Cochran2e5b38a2012-10-29 08:45:20 +00001725
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001726 if (cpsw->version != CPSW_VERSION_1 &&
1727 cpsw->version != CPSW_VERSION_2 &&
1728 cpsw->version != CPSW_VERSION_3)
Ben Hutchings2ee91e52013-11-14 00:47:36 +00001729 return -EOPNOTSUPP;
1730
Richard Cochran2e5b38a2012-10-29 08:45:20 +00001731 if (copy_from_user(&cfg, ifr->ifr_data, sizeof(cfg)))
1732 return -EFAULT;
1733
1734 /* reserved for future extensions */
1735 if (cfg.flags)
1736 return -EINVAL;
1737
Ben Hutchings2ee91e52013-11-14 00:47:36 +00001738 if (cfg.tx_type != HWTSTAMP_TX_OFF && cfg.tx_type != HWTSTAMP_TX_ON)
Richard Cochran2e5b38a2012-10-29 08:45:20 +00001739 return -ERANGE;
Richard Cochran2e5b38a2012-10-29 08:45:20 +00001740
1741 switch (cfg.rx_filter) {
1742 case HWTSTAMP_FILTER_NONE:
Grygorii Strashkob63ba582016-12-06 18:00:35 -06001743 cpts_rx_enable(cpts, 0);
Richard Cochran2e5b38a2012-10-29 08:45:20 +00001744 break;
1745 case HWTSTAMP_FILTER_ALL:
1746 case HWTSTAMP_FILTER_PTP_V1_L4_EVENT:
1747 case HWTSTAMP_FILTER_PTP_V1_L4_SYNC:
1748 case HWTSTAMP_FILTER_PTP_V1_L4_DELAY_REQ:
1749 return -ERANGE;
1750 case HWTSTAMP_FILTER_PTP_V2_L4_EVENT:
1751 case HWTSTAMP_FILTER_PTP_V2_L4_SYNC:
1752 case HWTSTAMP_FILTER_PTP_V2_L4_DELAY_REQ:
1753 case HWTSTAMP_FILTER_PTP_V2_L2_EVENT:
1754 case HWTSTAMP_FILTER_PTP_V2_L2_SYNC:
1755 case HWTSTAMP_FILTER_PTP_V2_L2_DELAY_REQ:
1756 case HWTSTAMP_FILTER_PTP_V2_EVENT:
1757 case HWTSTAMP_FILTER_PTP_V2_SYNC:
1758 case HWTSTAMP_FILTER_PTP_V2_DELAY_REQ:
Grygorii Strashkob63ba582016-12-06 18:00:35 -06001759 cpts_rx_enable(cpts, 1);
Richard Cochran2e5b38a2012-10-29 08:45:20 +00001760 cfg.rx_filter = HWTSTAMP_FILTER_PTP_V2_EVENT;
1761 break;
1762 default:
1763 return -ERANGE;
1764 }
1765
Grygorii Strashkob63ba582016-12-06 18:00:35 -06001766 cpts_tx_enable(cpts, cfg.tx_type == HWTSTAMP_TX_ON);
Ben Hutchings2ee91e52013-11-14 00:47:36 +00001767
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001768 switch (cpsw->version) {
Richard Cochran2e5b38a2012-10-29 08:45:20 +00001769 case CPSW_VERSION_1:
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001770 cpsw_hwtstamp_v1(cpsw);
Richard Cochran2e5b38a2012-10-29 08:45:20 +00001771 break;
1772 case CPSW_VERSION_2:
George Cherianf7d403c2014-05-02 12:02:01 +05301773 case CPSW_VERSION_3:
Richard Cochran2e5b38a2012-10-29 08:45:20 +00001774 cpsw_hwtstamp_v2(priv);
1775 break;
1776 default:
Ben Hutchings2ee91e52013-11-14 00:47:36 +00001777 WARN_ON(1);
Richard Cochran2e5b38a2012-10-29 08:45:20 +00001778 }
1779
1780 return copy_to_user(ifr->ifr_data, &cfg, sizeof(cfg)) ? -EFAULT : 0;
1781}
1782
Ben Hutchingsa5b41452013-11-18 23:23:40 +00001783static int cpsw_hwtstamp_get(struct net_device *dev, struct ifreq *ifr)
1784{
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001785 struct cpsw_common *cpsw = ndev_to_cpsw(dev);
1786 struct cpts *cpts = cpsw->cpts;
Ben Hutchingsa5b41452013-11-18 23:23:40 +00001787 struct hwtstamp_config cfg;
1788
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001789 if (cpsw->version != CPSW_VERSION_1 &&
1790 cpsw->version != CPSW_VERSION_2 &&
1791 cpsw->version != CPSW_VERSION_3)
Ben Hutchingsa5b41452013-11-18 23:23:40 +00001792 return -EOPNOTSUPP;
1793
1794 cfg.flags = 0;
Grygorii Strashkob63ba582016-12-06 18:00:35 -06001795 cfg.tx_type = cpts_is_tx_enabled(cpts) ?
1796 HWTSTAMP_TX_ON : HWTSTAMP_TX_OFF;
1797 cfg.rx_filter = (cpts_is_rx_enabled(cpts) ?
Ben Hutchingsa5b41452013-11-18 23:23:40 +00001798 HWTSTAMP_FILTER_PTP_V2_EVENT : HWTSTAMP_FILTER_NONE);
1799
1800 return copy_to_user(ifr->ifr_data, &cfg, sizeof(cfg)) ? -EFAULT : 0;
1801}
Grygorii Strashkoc8395d42016-12-06 18:00:34 -06001802#else
1803static int cpsw_hwtstamp_get(struct net_device *dev, struct ifreq *ifr)
1804{
1805 return -EOPNOTSUPP;
1806}
Ben Hutchingsa5b41452013-11-18 23:23:40 +00001807
Grygorii Strashkoc8395d42016-12-06 18:00:34 -06001808static int cpsw_hwtstamp_set(struct net_device *dev, struct ifreq *ifr)
1809{
1810 return -EOPNOTSUPP;
1811}
Richard Cochran2e5b38a2012-10-29 08:45:20 +00001812#endif /*CONFIG_TI_CPTS*/
1813
1814static int cpsw_ndo_ioctl(struct net_device *dev, struct ifreq *req, int cmd)
1815{
Mugunthan V N11f2c982013-03-11 23:16:38 +00001816 struct cpsw_priv *priv = netdev_priv(dev);
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03001817 struct cpsw_common *cpsw = priv->cpsw;
1818 int slave_no = cpsw_slave_index(cpsw, priv);
Mugunthan V N11f2c982013-03-11 23:16:38 +00001819
Richard Cochran2e5b38a2012-10-29 08:45:20 +00001820 if (!netif_running(dev))
1821 return -EINVAL;
1822
Mugunthan V N11f2c982013-03-11 23:16:38 +00001823 switch (cmd) {
Mugunthan V N11f2c982013-03-11 23:16:38 +00001824 case SIOCSHWTSTAMP:
Ben Hutchingsa5b41452013-11-18 23:23:40 +00001825 return cpsw_hwtstamp_set(dev, req);
1826 case SIOCGHWTSTAMP:
1827 return cpsw_hwtstamp_get(dev, req);
Mugunthan V N11f2c982013-03-11 23:16:38 +00001828 }
1829
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03001830 if (!cpsw->slaves[slave_no].phy)
Stefan Sørensenc1b59942014-02-16 14:54:25 +01001831 return -EOPNOTSUPP;
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03001832 return phy_mii_ioctl(cpsw->slaves[slave_no].phy, req, cmd);
Richard Cochran2e5b38a2012-10-29 08:45:20 +00001833}
1834
Mugunthan V Ndf828592012-03-18 20:17:54 +00001835static void cpsw_ndo_tx_timeout(struct net_device *ndev)
1836{
1837 struct cpsw_priv *priv = netdev_priv(ndev);
Ivan Khoronzhuk2c836bd2016-08-10 02:22:40 +03001838 struct cpsw_common *cpsw = priv->cpsw;
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03001839 int ch;
Mugunthan V Ndf828592012-03-18 20:17:54 +00001840
1841 cpsw_err(priv, tx_err, "transmit timeout, restarting dma\n");
Tobias Klauser8dc43dd2014-03-10 13:12:23 +01001842 ndev->stats.tx_errors++;
Ivan Khoronzhuk2c836bd2016-08-10 02:22:40 +03001843 cpsw_intr_disable(cpsw);
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03001844 for (ch = 0; ch < cpsw->tx_ch_num; ch++) {
Ivan Khoronzhuk8feb0a12016-11-29 17:00:51 +02001845 cpdma_chan_stop(cpsw->txv[ch].ch);
1846 cpdma_chan_start(cpsw->txv[ch].ch);
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03001847 }
1848
Ivan Khoronzhuk2c836bd2016-08-10 02:22:40 +03001849 cpsw_intr_enable(cpsw);
Mugunthan V Ndf828592012-03-18 20:17:54 +00001850}
1851
Mugunthan V Ndcfd8d52013-07-25 23:44:01 +05301852static int cpsw_ndo_set_mac_address(struct net_device *ndev, void *p)
1853{
1854 struct cpsw_priv *priv = netdev_priv(ndev);
1855 struct sockaddr *addr = (struct sockaddr *)p;
Ivan Khoronzhuk649a1682016-08-10 02:22:37 +03001856 struct cpsw_common *cpsw = priv->cpsw;
Mugunthan V Ndcfd8d52013-07-25 23:44:01 +05301857 int flags = 0;
1858 u16 vid = 0;
Grygorii Strashkoa6c5d142016-06-24 21:23:45 +03001859 int ret;
Mugunthan V Ndcfd8d52013-07-25 23:44:01 +05301860
1861 if (!is_valid_ether_addr(addr->sa_data))
1862 return -EADDRNOTAVAIL;
1863
Ivan Khoronzhuk56e31bd2016-08-10 02:22:38 +03001864 ret = pm_runtime_get_sync(cpsw->dev);
Grygorii Strashkoa6c5d142016-06-24 21:23:45 +03001865 if (ret < 0) {
Ivan Khoronzhuk56e31bd2016-08-10 02:22:38 +03001866 pm_runtime_put_noidle(cpsw->dev);
Grygorii Strashkoa6c5d142016-06-24 21:23:45 +03001867 return ret;
1868 }
1869
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03001870 if (cpsw->data.dual_emac) {
1871 vid = cpsw->slaves[priv->emac_port].port_vlan;
Mugunthan V Ndcfd8d52013-07-25 23:44:01 +05301872 flags = ALE_VLAN;
1873 }
1874
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001875 cpsw_ale_del_ucast(cpsw->ale, priv->mac_addr, HOST_PORT_NUM,
Mugunthan V Ndcfd8d52013-07-25 23:44:01 +05301876 flags, vid);
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001877 cpsw_ale_add_ucast(cpsw->ale, addr->sa_data, HOST_PORT_NUM,
Mugunthan V Ndcfd8d52013-07-25 23:44:01 +05301878 flags, vid);
1879
1880 memcpy(priv->mac_addr, addr->sa_data, ETH_ALEN);
1881 memcpy(ndev->dev_addr, priv->mac_addr, ETH_ALEN);
1882 for_each_slave(priv, cpsw_set_slave_mac, priv);
1883
Ivan Khoronzhuk56e31bd2016-08-10 02:22:38 +03001884 pm_runtime_put(cpsw->dev);
Grygorii Strashkoa6c5d142016-06-24 21:23:45 +03001885
Mugunthan V Ndcfd8d52013-07-25 23:44:01 +05301886 return 0;
1887}
1888
Mugunthan V Ndf828592012-03-18 20:17:54 +00001889#ifdef CONFIG_NET_POLL_CONTROLLER
1890static void cpsw_ndo_poll_controller(struct net_device *ndev)
1891{
Ivan Khoronzhukdbc4ec52016-08-10 02:22:43 +03001892 struct cpsw_common *cpsw = ndev_to_cpsw(ndev);
Mugunthan V Ndf828592012-03-18 20:17:54 +00001893
Ivan Khoronzhukdbc4ec52016-08-10 02:22:43 +03001894 cpsw_intr_disable(cpsw);
1895 cpsw_rx_interrupt(cpsw->irqs_table[0], cpsw);
1896 cpsw_tx_interrupt(cpsw->irqs_table[1], cpsw);
1897 cpsw_intr_enable(cpsw);
Mugunthan V Ndf828592012-03-18 20:17:54 +00001898}
1899#endif
1900
Mugunthan V N3b72c2f2013-02-05 08:26:48 +00001901static inline int cpsw_add_vlan_ale_entry(struct cpsw_priv *priv,
1902 unsigned short vid)
1903{
1904 int ret;
Mugunthan V N9f6bd8f2015-01-15 14:59:28 +05301905 int unreg_mcast_mask = 0;
1906 u32 port_mask;
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03001907 struct cpsw_common *cpsw = priv->cpsw;
Lennart Sorensen1e5c4bc2014-10-31 13:38:52 -04001908
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03001909 if (cpsw->data.dual_emac) {
Mugunthan V N9f6bd8f2015-01-15 14:59:28 +05301910 port_mask = (1 << (priv->emac_port + 1)) | ALE_PORT_HOST;
Mugunthan V N3b72c2f2013-02-05 08:26:48 +00001911
Mugunthan V N9f6bd8f2015-01-15 14:59:28 +05301912 if (priv->ndev->flags & IFF_ALLMULTI)
1913 unreg_mcast_mask = port_mask;
1914 } else {
1915 port_mask = ALE_ALL_PORTS;
1916
1917 if (priv->ndev->flags & IFF_ALLMULTI)
1918 unreg_mcast_mask = ALE_ALL_PORTS;
1919 else
1920 unreg_mcast_mask = ALE_PORT_1 | ALE_PORT_2;
1921 }
1922
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001923 ret = cpsw_ale_add_vlan(cpsw->ale, vid, port_mask, 0, port_mask,
Grygorii Strashko61f1cef2016-04-07 15:16:43 +03001924 unreg_mcast_mask);
Mugunthan V N3b72c2f2013-02-05 08:26:48 +00001925 if (ret != 0)
1926 return ret;
1927
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001928 ret = cpsw_ale_add_ucast(cpsw->ale, priv->mac_addr,
Grygorii Strashko71a2cbb2016-04-07 15:16:44 +03001929 HOST_PORT_NUM, ALE_VLAN, vid);
Mugunthan V N3b72c2f2013-02-05 08:26:48 +00001930 if (ret != 0)
1931 goto clean_vid;
1932
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001933 ret = cpsw_ale_add_mcast(cpsw->ale, priv->ndev->broadcast,
Mugunthan V N9f6bd8f2015-01-15 14:59:28 +05301934 port_mask, ALE_VLAN, vid, 0);
Mugunthan V N3b72c2f2013-02-05 08:26:48 +00001935 if (ret != 0)
1936 goto clean_vlan_ucast;
1937 return 0;
1938
1939clean_vlan_ucast:
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001940 cpsw_ale_del_ucast(cpsw->ale, priv->mac_addr,
Grygorii Strashko71a2cbb2016-04-07 15:16:44 +03001941 HOST_PORT_NUM, ALE_VLAN, vid);
Mugunthan V N3b72c2f2013-02-05 08:26:48 +00001942clean_vid:
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03001943 cpsw_ale_del_vlan(cpsw->ale, vid, 0);
Mugunthan V N3b72c2f2013-02-05 08:26:48 +00001944 return ret;
1945}
1946
1947static int cpsw_ndo_vlan_rx_add_vid(struct net_device *ndev,
Patrick McHardy80d5c362013-04-19 02:04:28 +00001948 __be16 proto, u16 vid)
Mugunthan V N3b72c2f2013-02-05 08:26:48 +00001949{
1950 struct cpsw_priv *priv = netdev_priv(ndev);
Ivan Khoronzhuk649a1682016-08-10 02:22:37 +03001951 struct cpsw_common *cpsw = priv->cpsw;
Grygorii Strashkoa6c5d142016-06-24 21:23:45 +03001952 int ret;
Mugunthan V N3b72c2f2013-02-05 08:26:48 +00001953
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03001954 if (vid == cpsw->data.default_vlan)
Mugunthan V N3b72c2f2013-02-05 08:26:48 +00001955 return 0;
1956
Ivan Khoronzhuk56e31bd2016-08-10 02:22:38 +03001957 ret = pm_runtime_get_sync(cpsw->dev);
Grygorii Strashkoa6c5d142016-06-24 21:23:45 +03001958 if (ret < 0) {
Ivan Khoronzhuk56e31bd2016-08-10 02:22:38 +03001959 pm_runtime_put_noidle(cpsw->dev);
Grygorii Strashkoa6c5d142016-06-24 21:23:45 +03001960 return ret;
1961 }
1962
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03001963 if (cpsw->data.dual_emac) {
Mugunthan V N02a54162015-01-22 15:19:22 +05301964 /* In dual EMAC, reserved VLAN id should not be used for
1965 * creating VLAN interfaces as this can break the dual
1966 * EMAC port separation
1967 */
1968 int i;
1969
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03001970 for (i = 0; i < cpsw->data.slaves; i++) {
1971 if (vid == cpsw->slaves[i].port_vlan)
Mugunthan V N02a54162015-01-22 15:19:22 +05301972 return -EINVAL;
1973 }
1974 }
1975
Mugunthan V N3b72c2f2013-02-05 08:26:48 +00001976 dev_info(priv->dev, "Adding vlanid %d to vlan filter\n", vid);
Grygorii Strashkoa6c5d142016-06-24 21:23:45 +03001977 ret = cpsw_add_vlan_ale_entry(priv, vid);
1978
Ivan Khoronzhuk56e31bd2016-08-10 02:22:38 +03001979 pm_runtime_put(cpsw->dev);
Grygorii Strashkoa6c5d142016-06-24 21:23:45 +03001980 return ret;
Mugunthan V N3b72c2f2013-02-05 08:26:48 +00001981}
1982
1983static int cpsw_ndo_vlan_rx_kill_vid(struct net_device *ndev,
Patrick McHardy80d5c362013-04-19 02:04:28 +00001984 __be16 proto, u16 vid)
Mugunthan V N3b72c2f2013-02-05 08:26:48 +00001985{
1986 struct cpsw_priv *priv = netdev_priv(ndev);
Ivan Khoronzhuk649a1682016-08-10 02:22:37 +03001987 struct cpsw_common *cpsw = priv->cpsw;
Mugunthan V N3b72c2f2013-02-05 08:26:48 +00001988 int ret;
1989
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03001990 if (vid == cpsw->data.default_vlan)
Mugunthan V N3b72c2f2013-02-05 08:26:48 +00001991 return 0;
1992
Ivan Khoronzhuk56e31bd2016-08-10 02:22:38 +03001993 ret = pm_runtime_get_sync(cpsw->dev);
Grygorii Strashkoa6c5d142016-06-24 21:23:45 +03001994 if (ret < 0) {
Ivan Khoronzhuk56e31bd2016-08-10 02:22:38 +03001995 pm_runtime_put_noidle(cpsw->dev);
Grygorii Strashkoa6c5d142016-06-24 21:23:45 +03001996 return ret;
1997 }
1998
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03001999 if (cpsw->data.dual_emac) {
Mugunthan V N02a54162015-01-22 15:19:22 +05302000 int i;
2001
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03002002 for (i = 0; i < cpsw->data.slaves; i++) {
2003 if (vid == cpsw->slaves[i].port_vlan)
Mugunthan V N02a54162015-01-22 15:19:22 +05302004 return -EINVAL;
2005 }
2006 }
2007
Mugunthan V N3b72c2f2013-02-05 08:26:48 +00002008 dev_info(priv->dev, "removing vlanid %d from vlan filter\n", vid);
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03002009 ret = cpsw_ale_del_vlan(cpsw->ale, vid, 0);
Mugunthan V N3b72c2f2013-02-05 08:26:48 +00002010 if (ret != 0)
2011 return ret;
2012
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03002013 ret = cpsw_ale_del_ucast(cpsw->ale, priv->mac_addr,
Grygorii Strashko61f1cef2016-04-07 15:16:43 +03002014 HOST_PORT_NUM, ALE_VLAN, vid);
Mugunthan V N3b72c2f2013-02-05 08:26:48 +00002015 if (ret != 0)
2016 return ret;
2017
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03002018 ret = cpsw_ale_del_mcast(cpsw->ale, priv->ndev->broadcast,
Grygorii Strashkoa6c5d142016-06-24 21:23:45 +03002019 0, ALE_VLAN, vid);
Ivan Khoronzhuk56e31bd2016-08-10 02:22:38 +03002020 pm_runtime_put(cpsw->dev);
Grygorii Strashkoa6c5d142016-06-24 21:23:45 +03002021 return ret;
Mugunthan V N3b72c2f2013-02-05 08:26:48 +00002022}
2023
Ivan Khoronzhuk83fcad02016-11-29 17:00:49 +02002024static int cpsw_ndo_set_tx_maxrate(struct net_device *ndev, int queue, u32 rate)
2025{
2026 struct cpsw_priv *priv = netdev_priv(ndev);
Ivan Khoronzhuk83fcad02016-11-29 17:00:49 +02002027 struct cpsw_common *cpsw = priv->cpsw;
Ivan Khoronzhuk52986a22016-12-10 14:23:50 +02002028 struct cpsw_slave *slave;
Ivan Khoronzhuk32b78d82016-12-10 14:23:48 +02002029 u32 min_rate;
Ivan Khoronzhuk83fcad02016-11-29 17:00:49 +02002030 u32 ch_rate;
Ivan Khoronzhuk52986a22016-12-10 14:23:50 +02002031 int i, ret;
Ivan Khoronzhuk83fcad02016-11-29 17:00:49 +02002032
2033 ch_rate = netdev_get_tx_queue(ndev, queue)->tx_maxrate;
2034 if (ch_rate == rate)
2035 return 0;
2036
Ivan Khoronzhuk32b78d82016-12-10 14:23:48 +02002037 ch_rate = rate * 1000;
2038 min_rate = cpdma_chan_get_min_rate(cpsw->dma);
2039 if ((ch_rate < min_rate && ch_rate)) {
2040 dev_err(priv->dev, "The channel rate cannot be less than %dMbps",
2041 min_rate);
Ivan Khoronzhuk83fcad02016-11-29 17:00:49 +02002042 return -EINVAL;
2043 }
2044
Ivan Khoronzhuk0be01b82016-12-10 14:23:49 +02002045 if (rate > cpsw->speed) {
Ivan Khoronzhuk32b78d82016-12-10 14:23:48 +02002046 dev_err(priv->dev, "The channel rate cannot be more than 2Gbps");
Ivan Khoronzhuk83fcad02016-11-29 17:00:49 +02002047 return -EINVAL;
2048 }
2049
2050 ret = pm_runtime_get_sync(cpsw->dev);
2051 if (ret < 0) {
2052 pm_runtime_put_noidle(cpsw->dev);
2053 return ret;
2054 }
2055
Ivan Khoronzhuk32b78d82016-12-10 14:23:48 +02002056 ret = cpdma_chan_set_rate(cpsw->txv[queue].ch, ch_rate);
Ivan Khoronzhuk83fcad02016-11-29 17:00:49 +02002057 pm_runtime_put(cpsw->dev);
Ivan Khoronzhuk32b78d82016-12-10 14:23:48 +02002058
2059 if (ret)
2060 return ret;
2061
Ivan Khoronzhuk52986a22016-12-10 14:23:50 +02002062 /* update rates for slaves tx queues */
2063 for (i = 0; i < cpsw->data.slaves; i++) {
2064 slave = &cpsw->slaves[i];
2065 if (!slave->ndev)
2066 continue;
2067
2068 netdev_get_tx_queue(slave->ndev, queue)->tx_maxrate = rate;
2069 }
2070
Ivan Khoronzhuk32b78d82016-12-10 14:23:48 +02002071 cpsw_split_res(ndev);
Ivan Khoronzhuk83fcad02016-11-29 17:00:49 +02002072 return ret;
2073}
2074
Mugunthan V Ndf828592012-03-18 20:17:54 +00002075static const struct net_device_ops cpsw_netdev_ops = {
2076 .ndo_open = cpsw_ndo_open,
2077 .ndo_stop = cpsw_ndo_stop,
2078 .ndo_start_xmit = cpsw_ndo_start_xmit,
Mugunthan V Ndcfd8d52013-07-25 23:44:01 +05302079 .ndo_set_mac_address = cpsw_ndo_set_mac_address,
Richard Cochran2e5b38a2012-10-29 08:45:20 +00002080 .ndo_do_ioctl = cpsw_ndo_ioctl,
Mugunthan V Ndf828592012-03-18 20:17:54 +00002081 .ndo_validate_addr = eth_validate_addr,
Mugunthan V Ndf828592012-03-18 20:17:54 +00002082 .ndo_tx_timeout = cpsw_ndo_tx_timeout,
Mugunthan V N5c50a852012-10-29 08:45:11 +00002083 .ndo_set_rx_mode = cpsw_ndo_set_rx_mode,
Ivan Khoronzhuk83fcad02016-11-29 17:00:49 +02002084 .ndo_set_tx_maxrate = cpsw_ndo_set_tx_maxrate,
Mugunthan V Ndf828592012-03-18 20:17:54 +00002085#ifdef CONFIG_NET_POLL_CONTROLLER
2086 .ndo_poll_controller = cpsw_ndo_poll_controller,
2087#endif
Mugunthan V N3b72c2f2013-02-05 08:26:48 +00002088 .ndo_vlan_rx_add_vid = cpsw_ndo_vlan_rx_add_vid,
2089 .ndo_vlan_rx_kill_vid = cpsw_ndo_vlan_rx_kill_vid,
Mugunthan V Ndf828592012-03-18 20:17:54 +00002090};
2091
Mugunthan V N52c4f0e2014-07-22 23:25:07 +05302092static int cpsw_get_regs_len(struct net_device *ndev)
2093{
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03002094 struct cpsw_common *cpsw = ndev_to_cpsw(ndev);
Mugunthan V N52c4f0e2014-07-22 23:25:07 +05302095
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03002096 return cpsw->data.ale_entries * ALE_ENTRY_WORDS * sizeof(u32);
Mugunthan V N52c4f0e2014-07-22 23:25:07 +05302097}
2098
2099static void cpsw_get_regs(struct net_device *ndev,
2100 struct ethtool_regs *regs, void *p)
2101{
Mugunthan V N52c4f0e2014-07-22 23:25:07 +05302102 u32 *reg = p;
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03002103 struct cpsw_common *cpsw = ndev_to_cpsw(ndev);
Mugunthan V N52c4f0e2014-07-22 23:25:07 +05302104
2105 /* update CPSW IP version */
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03002106 regs->version = cpsw->version;
Mugunthan V N52c4f0e2014-07-22 23:25:07 +05302107
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03002108 cpsw_ale_dump(cpsw->ale, reg);
Mugunthan V N52c4f0e2014-07-22 23:25:07 +05302109}
2110
Mugunthan V Ndf828592012-03-18 20:17:54 +00002111static void cpsw_get_drvinfo(struct net_device *ndev,
2112 struct ethtool_drvinfo *info)
2113{
Ivan Khoronzhuk649a1682016-08-10 02:22:37 +03002114 struct cpsw_common *cpsw = ndev_to_cpsw(ndev);
Ivan Khoronzhuk56e31bd2016-08-10 02:22:38 +03002115 struct platform_device *pdev = to_platform_device(cpsw->dev);
Jiri Pirko7826d432013-01-06 00:44:26 +00002116
Mugunthan V N52c4f0e2014-07-22 23:25:07 +05302117 strlcpy(info->driver, "cpsw", sizeof(info->driver));
Jiri Pirko7826d432013-01-06 00:44:26 +00002118 strlcpy(info->version, "1.0", sizeof(info->version));
Ivan Khoronzhuk56e31bd2016-08-10 02:22:38 +03002119 strlcpy(info->bus_info, pdev->name, sizeof(info->bus_info));
Mugunthan V Ndf828592012-03-18 20:17:54 +00002120}
2121
2122static u32 cpsw_get_msglevel(struct net_device *ndev)
2123{
2124 struct cpsw_priv *priv = netdev_priv(ndev);
2125 return priv->msg_enable;
2126}
2127
2128static void cpsw_set_msglevel(struct net_device *ndev, u32 value)
2129{
2130 struct cpsw_priv *priv = netdev_priv(ndev);
2131 priv->msg_enable = value;
2132}
2133
Grygorii Strashkoc8395d42016-12-06 18:00:34 -06002134#if IS_ENABLED(CONFIG_TI_CPTS)
Richard Cochran2e5b38a2012-10-29 08:45:20 +00002135static int cpsw_get_ts_info(struct net_device *ndev,
2136 struct ethtool_ts_info *info)
2137{
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03002138 struct cpsw_common *cpsw = ndev_to_cpsw(ndev);
Richard Cochran2e5b38a2012-10-29 08:45:20 +00002139
2140 info->so_timestamping =
2141 SOF_TIMESTAMPING_TX_HARDWARE |
2142 SOF_TIMESTAMPING_TX_SOFTWARE |
2143 SOF_TIMESTAMPING_RX_HARDWARE |
2144 SOF_TIMESTAMPING_RX_SOFTWARE |
2145 SOF_TIMESTAMPING_SOFTWARE |
2146 SOF_TIMESTAMPING_RAW_HARDWARE;
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03002147 info->phc_index = cpsw->cpts->phc_index;
Richard Cochran2e5b38a2012-10-29 08:45:20 +00002148 info->tx_types =
2149 (1 << HWTSTAMP_TX_OFF) |
2150 (1 << HWTSTAMP_TX_ON);
2151 info->rx_filters =
2152 (1 << HWTSTAMP_FILTER_NONE) |
2153 (1 << HWTSTAMP_FILTER_PTP_V2_EVENT);
Grygorii Strashkoc8395d42016-12-06 18:00:34 -06002154 return 0;
2155}
Richard Cochran2e5b38a2012-10-29 08:45:20 +00002156#else
Grygorii Strashkoc8395d42016-12-06 18:00:34 -06002157static int cpsw_get_ts_info(struct net_device *ndev,
2158 struct ethtool_ts_info *info)
2159{
Richard Cochran2e5b38a2012-10-29 08:45:20 +00002160 info->so_timestamping =
2161 SOF_TIMESTAMPING_TX_SOFTWARE |
2162 SOF_TIMESTAMPING_RX_SOFTWARE |
2163 SOF_TIMESTAMPING_SOFTWARE;
2164 info->phc_index = -1;
2165 info->tx_types = 0;
2166 info->rx_filters = 0;
Richard Cochran2e5b38a2012-10-29 08:45:20 +00002167 return 0;
2168}
Grygorii Strashkoc8395d42016-12-06 18:00:34 -06002169#endif
Richard Cochran2e5b38a2012-10-29 08:45:20 +00002170
Philippe Reynes24798762016-10-08 17:46:15 +02002171static int cpsw_get_link_ksettings(struct net_device *ndev,
2172 struct ethtool_link_ksettings *ecmd)
Mugunthan V Nd3bb9c52013-03-11 23:16:36 +00002173{
2174 struct cpsw_priv *priv = netdev_priv(ndev);
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03002175 struct cpsw_common *cpsw = priv->cpsw;
2176 int slave_no = cpsw_slave_index(cpsw, priv);
Mugunthan V Nd3bb9c52013-03-11 23:16:36 +00002177
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03002178 if (cpsw->slaves[slave_no].phy)
Philippe Reynes24798762016-10-08 17:46:15 +02002179 return phy_ethtool_ksettings_get(cpsw->slaves[slave_no].phy,
2180 ecmd);
Mugunthan V Nd3bb9c52013-03-11 23:16:36 +00002181 else
2182 return -EOPNOTSUPP;
2183}
2184
Philippe Reynes24798762016-10-08 17:46:15 +02002185static int cpsw_set_link_ksettings(struct net_device *ndev,
2186 const struct ethtool_link_ksettings *ecmd)
Mugunthan V Nd3bb9c52013-03-11 23:16:36 +00002187{
2188 struct cpsw_priv *priv = netdev_priv(ndev);
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03002189 struct cpsw_common *cpsw = priv->cpsw;
2190 int slave_no = cpsw_slave_index(cpsw, priv);
Mugunthan V Nd3bb9c52013-03-11 23:16:36 +00002191
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03002192 if (cpsw->slaves[slave_no].phy)
Philippe Reynes24798762016-10-08 17:46:15 +02002193 return phy_ethtool_ksettings_set(cpsw->slaves[slave_no].phy,
2194 ecmd);
Mugunthan V Nd3bb9c52013-03-11 23:16:36 +00002195 else
2196 return -EOPNOTSUPP;
2197}
2198
Matus Ujhelyid8a64422013-08-20 07:59:38 +02002199static void cpsw_get_wol(struct net_device *ndev, struct ethtool_wolinfo *wol)
2200{
2201 struct cpsw_priv *priv = netdev_priv(ndev);
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03002202 struct cpsw_common *cpsw = priv->cpsw;
2203 int slave_no = cpsw_slave_index(cpsw, priv);
Matus Ujhelyid8a64422013-08-20 07:59:38 +02002204
2205 wol->supported = 0;
2206 wol->wolopts = 0;
2207
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03002208 if (cpsw->slaves[slave_no].phy)
2209 phy_ethtool_get_wol(cpsw->slaves[slave_no].phy, wol);
Matus Ujhelyid8a64422013-08-20 07:59:38 +02002210}
2211
2212static int cpsw_set_wol(struct net_device *ndev, struct ethtool_wolinfo *wol)
2213{
2214 struct cpsw_priv *priv = netdev_priv(ndev);
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03002215 struct cpsw_common *cpsw = priv->cpsw;
2216 int slave_no = cpsw_slave_index(cpsw, priv);
Matus Ujhelyid8a64422013-08-20 07:59:38 +02002217
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03002218 if (cpsw->slaves[slave_no].phy)
2219 return phy_ethtool_set_wol(cpsw->slaves[slave_no].phy, wol);
Matus Ujhelyid8a64422013-08-20 07:59:38 +02002220 else
2221 return -EOPNOTSUPP;
2222}
2223
Mugunthan V N1923d6e2014-09-08 22:54:02 +05302224static void cpsw_get_pauseparam(struct net_device *ndev,
2225 struct ethtool_pauseparam *pause)
2226{
2227 struct cpsw_priv *priv = netdev_priv(ndev);
2228
2229 pause->autoneg = AUTONEG_DISABLE;
2230 pause->rx_pause = priv->rx_pause ? true : false;
2231 pause->tx_pause = priv->tx_pause ? true : false;
2232}
2233
2234static int cpsw_set_pauseparam(struct net_device *ndev,
2235 struct ethtool_pauseparam *pause)
2236{
2237 struct cpsw_priv *priv = netdev_priv(ndev);
2238 bool link;
2239
2240 priv->rx_pause = pause->rx_pause ? true : false;
2241 priv->tx_pause = pause->tx_pause ? true : false;
2242
2243 for_each_slave(priv, _cpsw_adjust_link, priv, &link);
Mugunthan V N1923d6e2014-09-08 22:54:02 +05302244 return 0;
2245}
2246
Grygorii Strashko7898b1d2016-06-24 21:23:44 +03002247static int cpsw_ethtool_op_begin(struct net_device *ndev)
2248{
2249 struct cpsw_priv *priv = netdev_priv(ndev);
Ivan Khoronzhuk649a1682016-08-10 02:22:37 +03002250 struct cpsw_common *cpsw = priv->cpsw;
Grygorii Strashko7898b1d2016-06-24 21:23:44 +03002251 int ret;
2252
Ivan Khoronzhuk56e31bd2016-08-10 02:22:38 +03002253 ret = pm_runtime_get_sync(cpsw->dev);
Grygorii Strashko7898b1d2016-06-24 21:23:44 +03002254 if (ret < 0) {
2255 cpsw_err(priv, drv, "ethtool begin failed %d\n", ret);
Ivan Khoronzhuk56e31bd2016-08-10 02:22:38 +03002256 pm_runtime_put_noidle(cpsw->dev);
Grygorii Strashko7898b1d2016-06-24 21:23:44 +03002257 }
2258
2259 return ret;
2260}
2261
2262static void cpsw_ethtool_op_complete(struct net_device *ndev)
2263{
2264 struct cpsw_priv *priv = netdev_priv(ndev);
2265 int ret;
2266
Ivan Khoronzhuk56e31bd2016-08-10 02:22:38 +03002267 ret = pm_runtime_put(priv->cpsw->dev);
Grygorii Strashko7898b1d2016-06-24 21:23:44 +03002268 if (ret < 0)
2269 cpsw_err(priv, drv, "ethtool complete failed %d\n", ret);
2270}
2271
Ivan Khoronzhukce52c742016-08-22 21:18:28 +03002272static void cpsw_get_channels(struct net_device *ndev,
2273 struct ethtool_channels *ch)
2274{
2275 struct cpsw_common *cpsw = ndev_to_cpsw(ndev);
2276
2277 ch->max_combined = 0;
2278 ch->max_rx = CPSW_MAX_QUEUES;
2279 ch->max_tx = CPSW_MAX_QUEUES;
2280 ch->max_other = 0;
2281 ch->other_count = 0;
2282 ch->rx_count = cpsw->rx_ch_num;
2283 ch->tx_count = cpsw->tx_ch_num;
2284 ch->combined_count = 0;
2285}
2286
2287static int cpsw_check_ch_settings(struct cpsw_common *cpsw,
2288 struct ethtool_channels *ch)
2289{
2290 if (ch->combined_count)
2291 return -EINVAL;
2292
2293 /* verify we have at least one channel in each direction */
2294 if (!ch->rx_count || !ch->tx_count)
2295 return -EINVAL;
2296
2297 if (ch->rx_count > cpsw->data.channels ||
2298 ch->tx_count > cpsw->data.channels)
2299 return -EINVAL;
2300
2301 return 0;
2302}
2303
2304static int cpsw_update_channels_res(struct cpsw_priv *priv, int ch_num, int rx)
2305{
2306 int (*poll)(struct napi_struct *, int);
2307 struct cpsw_common *cpsw = priv->cpsw;
2308 void (*handler)(void *, int, int);
Ivan Khoronzhuk83fcad02016-11-29 17:00:49 +02002309 struct netdev_queue *queue;
Ivan Khoronzhuk8feb0a12016-11-29 17:00:51 +02002310 struct cpsw_vector *vec;
Ivan Khoronzhukce52c742016-08-22 21:18:28 +03002311 int ret, *ch;
2312
2313 if (rx) {
2314 ch = &cpsw->rx_ch_num;
Ivan Khoronzhuk8feb0a12016-11-29 17:00:51 +02002315 vec = cpsw->rxv;
Ivan Khoronzhukce52c742016-08-22 21:18:28 +03002316 handler = cpsw_rx_handler;
2317 poll = cpsw_rx_poll;
2318 } else {
2319 ch = &cpsw->tx_ch_num;
Ivan Khoronzhuk8feb0a12016-11-29 17:00:51 +02002320 vec = cpsw->txv;
Ivan Khoronzhukce52c742016-08-22 21:18:28 +03002321 handler = cpsw_tx_handler;
2322 poll = cpsw_tx_poll;
2323 }
2324
2325 while (*ch < ch_num) {
Ivan Khoronzhuk8feb0a12016-11-29 17:00:51 +02002326 vec[*ch].ch = cpdma_chan_create(cpsw->dma, *ch, handler, rx);
Ivan Khoronzhuk83fcad02016-11-29 17:00:49 +02002327 queue = netdev_get_tx_queue(priv->ndev, *ch);
2328 queue->tx_maxrate = 0;
Ivan Khoronzhukce52c742016-08-22 21:18:28 +03002329
Ivan Khoronzhuk8feb0a12016-11-29 17:00:51 +02002330 if (IS_ERR(vec[*ch].ch))
2331 return PTR_ERR(vec[*ch].ch);
Ivan Khoronzhukce52c742016-08-22 21:18:28 +03002332
Ivan Khoronzhuk8feb0a12016-11-29 17:00:51 +02002333 if (!vec[*ch].ch)
Ivan Khoronzhukce52c742016-08-22 21:18:28 +03002334 return -EINVAL;
2335
2336 cpsw_info(priv, ifup, "created new %d %s channel\n", *ch,
2337 (rx ? "rx" : "tx"));
2338 (*ch)++;
2339 }
2340
2341 while (*ch > ch_num) {
2342 (*ch)--;
2343
Ivan Khoronzhuk8feb0a12016-11-29 17:00:51 +02002344 ret = cpdma_chan_destroy(vec[*ch].ch);
Ivan Khoronzhukce52c742016-08-22 21:18:28 +03002345 if (ret)
2346 return ret;
2347
2348 cpsw_info(priv, ifup, "destroyed %d %s channel\n", *ch,
2349 (rx ? "rx" : "tx"));
2350 }
2351
2352 return 0;
2353}
2354
2355static int cpsw_update_channels(struct cpsw_priv *priv,
2356 struct ethtool_channels *ch)
2357{
2358 int ret;
2359
2360 ret = cpsw_update_channels_res(priv, ch->rx_count, 1);
2361 if (ret)
2362 return ret;
2363
2364 ret = cpsw_update_channels_res(priv, ch->tx_count, 0);
2365 if (ret)
2366 return ret;
2367
2368 return 0;
2369}
2370
2371static int cpsw_set_channels(struct net_device *ndev,
2372 struct ethtool_channels *chs)
2373{
2374 struct cpsw_priv *priv = netdev_priv(ndev);
2375 struct cpsw_common *cpsw = priv->cpsw;
2376 struct cpsw_slave *slave;
2377 int i, ret;
2378
2379 ret = cpsw_check_ch_settings(cpsw, chs);
2380 if (ret < 0)
2381 return ret;
2382
2383 /* Disable NAPI scheduling */
2384 cpsw_intr_disable(cpsw);
2385
2386 /* Stop all transmit queues for every network device.
2387 * Disable re-using rx descriptors with dormant_on.
2388 */
2389 for (i = cpsw->data.slaves, slave = cpsw->slaves; i; i--, slave++) {
2390 if (!(slave->ndev && netif_running(slave->ndev)))
2391 continue;
2392
2393 netif_tx_stop_all_queues(slave->ndev);
2394 netif_dormant_on(slave->ndev);
2395 }
2396
2397 /* Handle rest of tx packets and stop cpdma channels */
2398 cpdma_ctlr_stop(cpsw->dma);
2399 ret = cpsw_update_channels(priv, chs);
2400 if (ret)
2401 goto err;
2402
2403 for (i = cpsw->data.slaves, slave = cpsw->slaves; i; i--, slave++) {
2404 if (!(slave->ndev && netif_running(slave->ndev)))
2405 continue;
2406
2407 /* Inform stack about new count of queues */
2408 ret = netif_set_real_num_tx_queues(slave->ndev,
2409 cpsw->tx_ch_num);
2410 if (ret) {
2411 dev_err(priv->dev, "cannot set real number of tx queues\n");
2412 goto err;
2413 }
2414
2415 ret = netif_set_real_num_rx_queues(slave->ndev,
2416 cpsw->rx_ch_num);
2417 if (ret) {
2418 dev_err(priv->dev, "cannot set real number of rx queues\n");
2419 goto err;
2420 }
2421
2422 /* Enable rx packets handling */
2423 netif_dormant_off(slave->ndev);
2424 }
2425
2426 if (cpsw_common_res_usage_state(cpsw)) {
Wei Yongjune19ac152016-08-26 14:35:43 +00002427 ret = cpsw_fill_rx_channels(priv);
2428 if (ret)
Ivan Khoronzhukce52c742016-08-22 21:18:28 +03002429 goto err;
2430
Ivan Khoronzhuk32b78d82016-12-10 14:23:48 +02002431 cpsw_split_res(ndev);
Ivan Khoronzhuk8feb0a12016-11-29 17:00:51 +02002432
Ivan Khoronzhukce52c742016-08-22 21:18:28 +03002433 /* After this receive is started */
2434 cpdma_ctlr_start(cpsw->dma);
2435 cpsw_intr_enable(cpsw);
2436 }
2437
2438 /* Resume transmit for every affected interface */
2439 for (i = cpsw->data.slaves, slave = cpsw->slaves; i; i--, slave++) {
2440 if (!(slave->ndev && netif_running(slave->ndev)))
2441 continue;
2442 netif_tx_start_all_queues(slave->ndev);
2443 }
2444 return 0;
2445err:
2446 dev_err(priv->dev, "cannot update channels number, closing device\n");
2447 dev_close(ndev);
2448 return ret;
2449}
2450
Yegor Yefremova0909942016-11-28 09:41:33 +01002451static int cpsw_get_eee(struct net_device *ndev, struct ethtool_eee *edata)
2452{
2453 struct cpsw_priv *priv = netdev_priv(ndev);
2454 struct cpsw_common *cpsw = priv->cpsw;
2455 int slave_no = cpsw_slave_index(cpsw, priv);
2456
2457 if (cpsw->slaves[slave_no].phy)
2458 return phy_ethtool_get_eee(cpsw->slaves[slave_no].phy, edata);
2459 else
2460 return -EOPNOTSUPP;
2461}
2462
2463static int cpsw_set_eee(struct net_device *ndev, struct ethtool_eee *edata)
2464{
2465 struct cpsw_priv *priv = netdev_priv(ndev);
2466 struct cpsw_common *cpsw = priv->cpsw;
2467 int slave_no = cpsw_slave_index(cpsw, priv);
2468
2469 if (cpsw->slaves[slave_no].phy)
2470 return phy_ethtool_set_eee(cpsw->slaves[slave_no].phy, edata);
2471 else
2472 return -EOPNOTSUPP;
2473}
2474
Yegor Yefremov6bb10c22016-11-28 10:47:52 +01002475static int cpsw_nway_reset(struct net_device *ndev)
2476{
2477 struct cpsw_priv *priv = netdev_priv(ndev);
2478 struct cpsw_common *cpsw = priv->cpsw;
2479 int slave_no = cpsw_slave_index(cpsw, priv);
2480
2481 if (cpsw->slaves[slave_no].phy)
2482 return genphy_restart_aneg(cpsw->slaves[slave_no].phy);
2483 else
2484 return -EOPNOTSUPP;
2485}
2486
Mugunthan V Ndf828592012-03-18 20:17:54 +00002487static const struct ethtool_ops cpsw_ethtool_ops = {
2488 .get_drvinfo = cpsw_get_drvinfo,
2489 .get_msglevel = cpsw_get_msglevel,
2490 .set_msglevel = cpsw_set_msglevel,
2491 .get_link = ethtool_op_get_link,
Richard Cochran2e5b38a2012-10-29 08:45:20 +00002492 .get_ts_info = cpsw_get_ts_info,
Mugunthan V Nff5b8ef2013-03-11 23:16:37 +00002493 .get_coalesce = cpsw_get_coalesce,
2494 .set_coalesce = cpsw_set_coalesce,
Mugunthan V Nd9718542013-07-23 15:38:17 +05302495 .get_sset_count = cpsw_get_sset_count,
2496 .get_strings = cpsw_get_strings,
2497 .get_ethtool_stats = cpsw_get_ethtool_stats,
Mugunthan V N1923d6e2014-09-08 22:54:02 +05302498 .get_pauseparam = cpsw_get_pauseparam,
2499 .set_pauseparam = cpsw_set_pauseparam,
Matus Ujhelyid8a64422013-08-20 07:59:38 +02002500 .get_wol = cpsw_get_wol,
2501 .set_wol = cpsw_set_wol,
Mugunthan V N52c4f0e2014-07-22 23:25:07 +05302502 .get_regs_len = cpsw_get_regs_len,
2503 .get_regs = cpsw_get_regs,
Grygorii Strashko7898b1d2016-06-24 21:23:44 +03002504 .begin = cpsw_ethtool_op_begin,
2505 .complete = cpsw_ethtool_op_complete,
Ivan Khoronzhukce52c742016-08-22 21:18:28 +03002506 .get_channels = cpsw_get_channels,
2507 .set_channels = cpsw_set_channels,
Philippe Reynes24798762016-10-08 17:46:15 +02002508 .get_link_ksettings = cpsw_get_link_ksettings,
2509 .set_link_ksettings = cpsw_set_link_ksettings,
Yegor Yefremova0909942016-11-28 09:41:33 +01002510 .get_eee = cpsw_get_eee,
2511 .set_eee = cpsw_set_eee,
Yegor Yefremov6bb10c22016-11-28 10:47:52 +01002512 .nway_reset = cpsw_nway_reset,
Mugunthan V Ndf828592012-03-18 20:17:54 +00002513};
2514
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03002515static void cpsw_slave_init(struct cpsw_slave *slave, struct cpsw_common *cpsw,
Richard Cochran549985e2012-11-14 09:07:56 +00002516 u32 slave_reg_ofs, u32 sliver_reg_ofs)
Mugunthan V Ndf828592012-03-18 20:17:54 +00002517{
Ivan Khoronzhuk5d8d0d42016-08-10 02:22:39 +03002518 void __iomem *regs = cpsw->regs;
Mugunthan V Ndf828592012-03-18 20:17:54 +00002519 int slave_num = slave->slave_num;
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03002520 struct cpsw_slave_data *data = cpsw->data.slave_data + slave_num;
Mugunthan V Ndf828592012-03-18 20:17:54 +00002521
2522 slave->data = data;
Richard Cochran549985e2012-11-14 09:07:56 +00002523 slave->regs = regs + slave_reg_ofs;
2524 slave->sliver = regs + sliver_reg_ofs;
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00002525 slave->port_vlan = data->dual_emac_res_vlan;
Mugunthan V Ndf828592012-03-18 20:17:54 +00002526}
2527
David Rivshin552165b2016-04-27 21:25:25 -04002528static int cpsw_probe_dt(struct cpsw_platform_data *data,
Mugunthan V N2eb32b02012-07-30 10:17:14 +00002529 struct platform_device *pdev)
2530{
2531 struct device_node *node = pdev->dev.of_node;
2532 struct device_node *slave_node;
2533 int i = 0, ret;
2534 u32 prop;
2535
2536 if (!node)
2537 return -EINVAL;
2538
2539 if (of_property_read_u32(node, "slaves", &prop)) {
George Cherian88c99ff2014-05-12 10:21:19 +05302540 dev_err(&pdev->dev, "Missing slaves property in the DT.\n");
Mugunthan V N2eb32b02012-07-30 10:17:14 +00002541 return -EINVAL;
2542 }
2543 data->slaves = prop;
2544
Mugunthan V Ne86ac132013-03-11 23:16:35 +00002545 if (of_property_read_u32(node, "active_slave", &prop)) {
George Cherian88c99ff2014-05-12 10:21:19 +05302546 dev_err(&pdev->dev, "Missing active_slave property in the DT.\n");
Daniel Mackaa1a15e2013-09-21 00:50:38 +05302547 return -EINVAL;
Richard Cochran78ca0b22012-10-29 08:45:18 +00002548 }
Mugunthan V Ne86ac132013-03-11 23:16:35 +00002549 data->active_slave = prop;
Richard Cochran78ca0b22012-10-29 08:45:18 +00002550
Daniel Mackaa1a15e2013-09-21 00:50:38 +05302551 data->slave_data = devm_kzalloc(&pdev->dev, data->slaves
2552 * sizeof(struct cpsw_slave_data),
2553 GFP_KERNEL);
Joe Perchesb2adaca2013-02-03 17:43:58 +00002554 if (!data->slave_data)
Daniel Mackaa1a15e2013-09-21 00:50:38 +05302555 return -ENOMEM;
Mugunthan V N2eb32b02012-07-30 10:17:14 +00002556
Mugunthan V N2eb32b02012-07-30 10:17:14 +00002557 if (of_property_read_u32(node, "cpdma_channels", &prop)) {
George Cherian88c99ff2014-05-12 10:21:19 +05302558 dev_err(&pdev->dev, "Missing cpdma_channels property in the DT.\n");
Daniel Mackaa1a15e2013-09-21 00:50:38 +05302559 return -EINVAL;
Mugunthan V N2eb32b02012-07-30 10:17:14 +00002560 }
2561 data->channels = prop;
2562
Mugunthan V N2eb32b02012-07-30 10:17:14 +00002563 if (of_property_read_u32(node, "ale_entries", &prop)) {
George Cherian88c99ff2014-05-12 10:21:19 +05302564 dev_err(&pdev->dev, "Missing ale_entries property in the DT.\n");
Daniel Mackaa1a15e2013-09-21 00:50:38 +05302565 return -EINVAL;
Mugunthan V N2eb32b02012-07-30 10:17:14 +00002566 }
2567 data->ale_entries = prop;
2568
Mugunthan V N2eb32b02012-07-30 10:17:14 +00002569 if (of_property_read_u32(node, "bd_ram_size", &prop)) {
George Cherian88c99ff2014-05-12 10:21:19 +05302570 dev_err(&pdev->dev, "Missing bd_ram_size property in the DT.\n");
Daniel Mackaa1a15e2013-09-21 00:50:38 +05302571 return -EINVAL;
Mugunthan V N2eb32b02012-07-30 10:17:14 +00002572 }
2573 data->bd_ram_size = prop;
2574
Mugunthan V N2eb32b02012-07-30 10:17:14 +00002575 if (of_property_read_u32(node, "mac_control", &prop)) {
George Cherian88c99ff2014-05-12 10:21:19 +05302576 dev_err(&pdev->dev, "Missing mac_control property in the DT.\n");
Daniel Mackaa1a15e2013-09-21 00:50:38 +05302577 return -EINVAL;
Mugunthan V N2eb32b02012-07-30 10:17:14 +00002578 }
2579 data->mac_control = prop;
2580
Markus Pargmann281abd92013-10-04 14:44:40 +02002581 if (of_property_read_bool(node, "dual_emac"))
2582 data->dual_emac = 1;
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00002583
Vaibhav Hiremath1fb19aa2012-11-14 09:07:55 +00002584 /*
2585 * Populate all the child nodes here...
2586 */
2587 ret = of_platform_populate(node, NULL, NULL, &pdev->dev);
2588 /* We do not want to force this, as in some cases may not have child */
2589 if (ret)
George Cherian88c99ff2014-05-12 10:21:19 +05302590 dev_warn(&pdev->dev, "Doesn't have any child node\n");
Vaibhav Hiremath1fb19aa2012-11-14 09:07:55 +00002591
Ben Hutchings8658aaf2016-06-21 01:16:31 +01002592 for_each_available_child_of_node(node, slave_node) {
Richard Cochran549985e2012-11-14 09:07:56 +00002593 struct cpsw_slave_data *slave_data = data->slave_data + i;
2594 const void *mac_addr = NULL;
Richard Cochran549985e2012-11-14 09:07:56 +00002595 int lenp;
2596 const __be32 *parp;
Richard Cochran549985e2012-11-14 09:07:56 +00002597
Markus Pargmannf468b102013-10-04 14:44:39 +02002598 /* This is no slave child node, continue */
2599 if (strcmp(slave_node->name, "slave"))
2600 continue;
2601
David Rivshin552165b2016-04-27 21:25:25 -04002602 slave_data->phy_node = of_parse_phandle(slave_node,
2603 "phy-handle", 0);
David Rivshinf1eea5c2015-12-16 23:02:10 -05002604 parp = of_get_property(slave_node, "phy_id", &lenp);
David Rivshinae092b52016-04-27 21:38:26 -04002605 if (slave_data->phy_node) {
2606 dev_dbg(&pdev->dev,
2607 "slave[%d] using phy-handle=\"%s\"\n",
2608 i, slave_data->phy_node->full_name);
2609 } else if (of_phy_is_fixed_link(slave_node)) {
David Rivshindfc0a6d2015-12-16 23:02:11 -05002610 /* In the case of a fixed PHY, the DT node associated
2611 * to the PHY is the Ethernet MAC DT node.
2612 */
Markus Brunner1f71e8c2015-11-03 22:09:51 +01002613 ret = of_phy_register_fixed_link(slave_node);
Johan Hovold23a09872016-11-17 17:40:04 +01002614 if (ret) {
2615 if (ret != -EPROBE_DEFER)
2616 dev_err(&pdev->dev, "failed to register fixed-link phy: %d\n", ret);
Markus Brunner1f71e8c2015-11-03 22:09:51 +01002617 return ret;
Johan Hovold23a09872016-11-17 17:40:04 +01002618 }
David Rivshin06cd6d62016-04-27 21:45:45 -04002619 slave_data->phy_node = of_node_get(slave_node);
David Rivshinf1eea5c2015-12-16 23:02:10 -05002620 } else if (parp) {
2621 u32 phyid;
2622 struct device_node *mdio_node;
2623 struct platform_device *mdio;
2624
2625 if (lenp != (sizeof(__be32) * 2)) {
2626 dev_err(&pdev->dev, "Invalid slave[%d] phy_id property\n", i);
2627 goto no_phy_slave;
2628 }
2629 mdio_node = of_find_node_by_phandle(be32_to_cpup(parp));
2630 phyid = be32_to_cpup(parp+1);
2631 mdio = of_find_device_by_node(mdio_node);
2632 of_node_put(mdio_node);
2633 if (!mdio) {
2634 dev_err(&pdev->dev, "Missing mdio platform device\n");
2635 return -EINVAL;
2636 }
2637 snprintf(slave_data->phy_id, sizeof(slave_data->phy_id),
2638 PHY_ID_FMT, mdio->name, phyid);
Johan Hovold86e1d5a2016-11-17 17:39:59 +01002639 put_device(&mdio->dev);
David Rivshinf1eea5c2015-12-16 23:02:10 -05002640 } else {
David Rivshinae092b52016-04-27 21:38:26 -04002641 dev_err(&pdev->dev,
2642 "No slave[%d] phy_id, phy-handle, or fixed-link property\n",
2643 i);
Markus Brunner1f71e8c2015-11-03 22:09:51 +01002644 goto no_phy_slave;
2645 }
Mugunthan V N47276fc2014-10-24 18:51:33 +05302646 slave_data->phy_if = of_get_phy_mode(slave_node);
2647 if (slave_data->phy_if < 0) {
2648 dev_err(&pdev->dev, "Missing or malformed slave[%d] phy-mode property\n",
2649 i);
2650 return slave_data->phy_if;
2651 }
2652
2653no_phy_slave:
Richard Cochran549985e2012-11-14 09:07:56 +00002654 mac_addr = of_get_mac_address(slave_node);
Markus Pargmann0ba517b2014-09-29 08:53:17 +02002655 if (mac_addr) {
Richard Cochran549985e2012-11-14 09:07:56 +00002656 memcpy(slave_data->mac_addr, mac_addr, ETH_ALEN);
Markus Pargmann0ba517b2014-09-29 08:53:17 +02002657 } else {
Mugunthan V Nb6745f62015-09-21 15:56:50 +05302658 ret = ti_cm_get_macid(&pdev->dev, i,
2659 slave_data->mac_addr);
2660 if (ret)
2661 return ret;
Markus Pargmann0ba517b2014-09-29 08:53:17 +02002662 }
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00002663 if (data->dual_emac) {
Mugunthan V N91c41662013-04-15 07:31:28 +00002664 if (of_property_read_u32(slave_node, "dual_emac_res_vlan",
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00002665 &prop)) {
George Cherian88c99ff2014-05-12 10:21:19 +05302666 dev_err(&pdev->dev, "Missing dual_emac_res_vlan in DT.\n");
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00002667 slave_data->dual_emac_res_vlan = i+1;
George Cherian88c99ff2014-05-12 10:21:19 +05302668 dev_err(&pdev->dev, "Using %d as Reserved VLAN for %d slave\n",
2669 slave_data->dual_emac_res_vlan, i);
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00002670 } else {
2671 slave_data->dual_emac_res_vlan = prop;
2672 }
2673 }
2674
Richard Cochran549985e2012-11-14 09:07:56 +00002675 i++;
Mugunthan V N3a27bfa2013-12-02 12:53:39 +05302676 if (i == data->slaves)
2677 break;
Richard Cochran549985e2012-11-14 09:07:56 +00002678 }
2679
Mugunthan V N2eb32b02012-07-30 10:17:14 +00002680 return 0;
Mugunthan V N2eb32b02012-07-30 10:17:14 +00002681}
2682
Johan Hovolda4e32b02016-11-17 17:40:00 +01002683static void cpsw_remove_dt(struct platform_device *pdev)
2684{
Johan Hovold8cbcc462016-11-17 17:40:01 +01002685 struct net_device *ndev = platform_get_drvdata(pdev);
2686 struct cpsw_common *cpsw = ndev_to_cpsw(ndev);
2687 struct cpsw_platform_data *data = &cpsw->data;
2688 struct device_node *node = pdev->dev.of_node;
2689 struct device_node *slave_node;
2690 int i = 0;
2691
2692 for_each_available_child_of_node(node, slave_node) {
2693 struct cpsw_slave_data *slave_data = &data->slave_data[i];
2694
2695 if (strcmp(slave_node->name, "slave"))
2696 continue;
2697
Johan Hovold3f650472016-11-28 19:24:55 +01002698 if (of_phy_is_fixed_link(slave_node))
2699 of_phy_deregister_fixed_link(slave_node);
Johan Hovold8cbcc462016-11-17 17:40:01 +01002700
2701 of_node_put(slave_data->phy_node);
2702
2703 i++;
2704 if (i == data->slaves)
2705 break;
2706 }
2707
Johan Hovolda4e32b02016-11-17 17:40:00 +01002708 of_platform_depopulate(&pdev->dev);
2709}
2710
Ivan Khoronzhuk56e31bd2016-08-10 02:22:38 +03002711static int cpsw_probe_dual_emac(struct cpsw_priv *priv)
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00002712{
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03002713 struct cpsw_common *cpsw = priv->cpsw;
2714 struct cpsw_platform_data *data = &cpsw->data;
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00002715 struct net_device *ndev;
2716 struct cpsw_priv *priv_sl2;
Ivan Khoronzhuke38b5a32016-08-10 02:22:41 +03002717 int ret = 0;
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00002718
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03002719 ndev = alloc_etherdev_mq(sizeof(struct cpsw_priv), CPSW_MAX_QUEUES);
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00002720 if (!ndev) {
Ivan Khoronzhuk56e31bd2016-08-10 02:22:38 +03002721 dev_err(cpsw->dev, "cpsw: error allocating net_device\n");
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00002722 return -ENOMEM;
2723 }
2724
2725 priv_sl2 = netdev_priv(ndev);
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03002726 priv_sl2->cpsw = cpsw;
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00002727 priv_sl2->ndev = ndev;
2728 priv_sl2->dev = &ndev->dev;
2729 priv_sl2->msg_enable = netif_msg_init(debug_level, CPSW_DEBUG);
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00002730
2731 if (is_valid_ether_addr(data->slave_data[1].mac_addr)) {
2732 memcpy(priv_sl2->mac_addr, data->slave_data[1].mac_addr,
2733 ETH_ALEN);
Ivan Khoronzhuk56e31bd2016-08-10 02:22:38 +03002734 dev_info(cpsw->dev, "cpsw: Detected MACID = %pM\n",
2735 priv_sl2->mac_addr);
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00002736 } else {
2737 random_ether_addr(priv_sl2->mac_addr);
Ivan Khoronzhuk56e31bd2016-08-10 02:22:38 +03002738 dev_info(cpsw->dev, "cpsw: Random MACID = %pM\n",
2739 priv_sl2->mac_addr);
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00002740 }
2741 memcpy(ndev->dev_addr, priv_sl2->mac_addr, ETH_ALEN);
2742
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00002743 priv_sl2->emac_port = 1;
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03002744 cpsw->slaves[1].ndev = ndev;
Patrick McHardyf6469682013-04-19 02:04:27 +00002745 ndev->features |= NETIF_F_HW_VLAN_CTAG_FILTER;
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00002746
2747 ndev->netdev_ops = &cpsw_netdev_ops;
Wilfried Klaebe7ad24ea2014-05-11 00:12:32 +00002748 ndev->ethtool_ops = &cpsw_ethtool_ops;
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00002749
2750 /* register the network device */
Ivan Khoronzhuk56e31bd2016-08-10 02:22:38 +03002751 SET_NETDEV_DEV(ndev, cpsw->dev);
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00002752 ret = register_netdev(ndev);
2753 if (ret) {
Ivan Khoronzhuk56e31bd2016-08-10 02:22:38 +03002754 dev_err(cpsw->dev, "cpsw: error registering net device\n");
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00002755 free_netdev(ndev);
2756 ret = -ENODEV;
2757 }
2758
2759 return ret;
2760}
2761
Mugunthan V N7da11602015-08-12 15:22:53 +05302762#define CPSW_QUIRK_IRQ BIT(0)
2763
2764static struct platform_device_id cpsw_devtype[] = {
2765 {
2766 /* keep it for existing comaptibles */
2767 .name = "cpsw",
2768 .driver_data = CPSW_QUIRK_IRQ,
2769 }, {
2770 .name = "am335x-cpsw",
2771 .driver_data = CPSW_QUIRK_IRQ,
2772 }, {
2773 .name = "am4372-cpsw",
2774 .driver_data = 0,
2775 }, {
2776 .name = "dra7-cpsw",
2777 .driver_data = 0,
2778 }, {
2779 /* sentinel */
2780 }
2781};
2782MODULE_DEVICE_TABLE(platform, cpsw_devtype);
2783
2784enum ti_cpsw_type {
2785 CPSW = 0,
2786 AM335X_CPSW,
2787 AM4372_CPSW,
2788 DRA7_CPSW,
2789};
2790
2791static const struct of_device_id cpsw_of_mtable[] = {
2792 { .compatible = "ti,cpsw", .data = &cpsw_devtype[CPSW], },
2793 { .compatible = "ti,am335x-cpsw", .data = &cpsw_devtype[AM335X_CPSW], },
2794 { .compatible = "ti,am4372-cpsw", .data = &cpsw_devtype[AM4372_CPSW], },
2795 { .compatible = "ti,dra7-cpsw", .data = &cpsw_devtype[DRA7_CPSW], },
2796 { /* sentinel */ },
2797};
2798MODULE_DEVICE_TABLE(of, cpsw_of_mtable);
2799
Bill Pemberton663e12e2012-12-03 09:23:45 -05002800static int cpsw_probe(struct platform_device *pdev)
Mugunthan V Ndf828592012-03-18 20:17:54 +00002801{
Ivan Khoronzhukef4183a2016-08-10 02:22:35 +03002802 struct clk *clk;
Sebastian Siewiord1bd9ac2013-04-24 08:48:23 +00002803 struct cpsw_platform_data *data;
Mugunthan V Ndf828592012-03-18 20:17:54 +00002804 struct net_device *ndev;
2805 struct cpsw_priv *priv;
2806 struct cpdma_params dma_params;
2807 struct cpsw_ale_params ale_params;
Daniel Mackaa1a15e2013-09-21 00:50:38 +05302808 void __iomem *ss_regs;
Grygorii Strashko8a2c9a52016-12-06 18:00:41 -06002809 void __iomem *cpts_regs;
Daniel Mackaa1a15e2013-09-21 00:50:38 +05302810 struct resource *res, *ss_res;
Mugunthan V N7da11602015-08-12 15:22:53 +05302811 const struct of_device_id *of_id;
Mugunthan V N1d147cc2015-09-07 15:16:44 +05302812 struct gpio_descs *mode;
Richard Cochran549985e2012-11-14 09:07:56 +00002813 u32 slave_offset, sliver_offset, slave_size;
Ivan Khoronzhuk649a1682016-08-10 02:22:37 +03002814 struct cpsw_common *cpsw;
Felipe Balbi5087b912015-01-16 10:11:11 -06002815 int ret = 0, i;
2816 int irq;
Mugunthan V Ndf828592012-03-18 20:17:54 +00002817
Ivan Khoronzhuk649a1682016-08-10 02:22:37 +03002818 cpsw = devm_kzalloc(&pdev->dev, sizeof(struct cpsw_common), GFP_KERNEL);
Johan Hovold3420ea82016-11-17 17:40:03 +01002819 if (!cpsw)
2820 return -ENOMEM;
2821
Ivan Khoronzhuk56e31bd2016-08-10 02:22:38 +03002822 cpsw->dev = &pdev->dev;
Ivan Khoronzhuk649a1682016-08-10 02:22:37 +03002823
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03002824 ndev = alloc_etherdev_mq(sizeof(struct cpsw_priv), CPSW_MAX_QUEUES);
Mugunthan V Ndf828592012-03-18 20:17:54 +00002825 if (!ndev) {
George Cherian88c99ff2014-05-12 10:21:19 +05302826 dev_err(&pdev->dev, "error allocating net_device\n");
Mugunthan V Ndf828592012-03-18 20:17:54 +00002827 return -ENOMEM;
2828 }
2829
2830 platform_set_drvdata(pdev, ndev);
2831 priv = netdev_priv(ndev);
Ivan Khoronzhuk649a1682016-08-10 02:22:37 +03002832 priv->cpsw = cpsw;
Mugunthan V Ndf828592012-03-18 20:17:54 +00002833 priv->ndev = ndev;
2834 priv->dev = &ndev->dev;
2835 priv->msg_enable = netif_msg_init(debug_level, CPSW_DEBUG);
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03002836 cpsw->rx_packet_max = max(rx_packet_max, 128);
Mugunthan V Ndf828592012-03-18 20:17:54 +00002837
Mugunthan V N1d147cc2015-09-07 15:16:44 +05302838 mode = devm_gpiod_get_array_optional(&pdev->dev, "mode", GPIOD_OUT_LOW);
2839 if (IS_ERR(mode)) {
2840 ret = PTR_ERR(mode);
2841 dev_err(&pdev->dev, "gpio request failed, ret %d\n", ret);
2842 goto clean_ndev_ret;
2843 }
2844
Vaibhav Hiremath1fb19aa2012-11-14 09:07:55 +00002845 /*
2846 * This may be required here for child devices.
2847 */
2848 pm_runtime_enable(&pdev->dev);
2849
Mugunthan V N739683b2013-06-06 23:45:14 +05302850 /* Select default pin state */
2851 pinctrl_pm_select_default_state(&pdev->dev);
2852
Johan Hovolda4e32b02016-11-17 17:40:00 +01002853 /* Need to enable clocks with runtime PM api to access module
2854 * registers
2855 */
2856 ret = pm_runtime_get_sync(&pdev->dev);
2857 if (ret < 0) {
2858 pm_runtime_put_noidle(&pdev->dev);
Daniel Mackaa1a15e2013-09-21 00:50:38 +05302859 goto clean_runtime_disable_ret;
Mugunthan V N2eb32b02012-07-30 10:17:14 +00002860 }
Johan Hovolda4e32b02016-11-17 17:40:00 +01002861
Johan Hovold23a09872016-11-17 17:40:04 +01002862 ret = cpsw_probe_dt(&cpsw->data, pdev);
2863 if (ret)
Johan Hovolda4e32b02016-11-17 17:40:00 +01002864 goto clean_dt_ret;
Johan Hovold23a09872016-11-17 17:40:04 +01002865
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03002866 data = &cpsw->data;
Ivan Khoronzhuke05107e2016-08-22 21:18:26 +03002867 cpsw->rx_ch_num = 1;
2868 cpsw->tx_ch_num = 1;
Mugunthan V N2eb32b02012-07-30 10:17:14 +00002869
Mugunthan V Ndf828592012-03-18 20:17:54 +00002870 if (is_valid_ether_addr(data->slave_data[0].mac_addr)) {
2871 memcpy(priv->mac_addr, data->slave_data[0].mac_addr, ETH_ALEN);
George Cherian88c99ff2014-05-12 10:21:19 +05302872 dev_info(&pdev->dev, "Detected MACID = %pM\n", priv->mac_addr);
Mugunthan V Ndf828592012-03-18 20:17:54 +00002873 } else {
Joe Perches7efd26d2012-07-12 19:33:06 +00002874 eth_random_addr(priv->mac_addr);
George Cherian88c99ff2014-05-12 10:21:19 +05302875 dev_info(&pdev->dev, "Random MACID = %pM\n", priv->mac_addr);
Mugunthan V Ndf828592012-03-18 20:17:54 +00002876 }
2877
2878 memcpy(ndev->dev_addr, priv->mac_addr, ETH_ALEN);
2879
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03002880 cpsw->slaves = devm_kzalloc(&pdev->dev,
Daniel Mackaa1a15e2013-09-21 00:50:38 +05302881 sizeof(struct cpsw_slave) * data->slaves,
2882 GFP_KERNEL);
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03002883 if (!cpsw->slaves) {
Daniel Mackaa1a15e2013-09-21 00:50:38 +05302884 ret = -ENOMEM;
Johan Hovolda4e32b02016-11-17 17:40:00 +01002885 goto clean_dt_ret;
Mugunthan V Ndf828592012-03-18 20:17:54 +00002886 }
2887 for (i = 0; i < data->slaves; i++)
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03002888 cpsw->slaves[i].slave_num = i;
Mugunthan V Ndf828592012-03-18 20:17:54 +00002889
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03002890 cpsw->slaves[0].ndev = ndev;
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00002891 priv->emac_port = 0;
2892
Ivan Khoronzhukef4183a2016-08-10 02:22:35 +03002893 clk = devm_clk_get(&pdev->dev, "fck");
2894 if (IS_ERR(clk)) {
Daniel Mackaa1a15e2013-09-21 00:50:38 +05302895 dev_err(priv->dev, "fck is not found\n");
Mugunthan V Nf150bd72012-07-17 08:09:50 +00002896 ret = -ENODEV;
Johan Hovolda4e32b02016-11-17 17:40:00 +01002897 goto clean_dt_ret;
Mugunthan V Ndf828592012-03-18 20:17:54 +00002898 }
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03002899 cpsw->bus_freq_mhz = clk_get_rate(clk) / 1000000;
Mugunthan V Ndf828592012-03-18 20:17:54 +00002900
Daniel Mackaa1a15e2013-09-21 00:50:38 +05302901 ss_res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
2902 ss_regs = devm_ioremap_resource(&pdev->dev, ss_res);
2903 if (IS_ERR(ss_regs)) {
2904 ret = PTR_ERR(ss_regs);
Johan Hovolda4e32b02016-11-17 17:40:00 +01002905 goto clean_dt_ret;
Mugunthan V Ndf828592012-03-18 20:17:54 +00002906 }
Ivan Khoronzhuk5d8d0d42016-08-10 02:22:39 +03002907 cpsw->regs = ss_regs;
Mugunthan V Ndf828592012-03-18 20:17:54 +00002908
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03002909 cpsw->version = readl(&cpsw->regs->id_ver);
Mugunthan V Nf280e892013-12-11 22:09:05 -06002910
Daniel Mackaa1a15e2013-09-21 00:50:38 +05302911 res = platform_get_resource(pdev, IORESOURCE_MEM, 1);
Ivan Khoronzhuk5d8d0d42016-08-10 02:22:39 +03002912 cpsw->wr_regs = devm_ioremap_resource(&pdev->dev, res);
2913 if (IS_ERR(cpsw->wr_regs)) {
2914 ret = PTR_ERR(cpsw->wr_regs);
Johan Hovolda4e32b02016-11-17 17:40:00 +01002915 goto clean_dt_ret;
Mugunthan V Ndf828592012-03-18 20:17:54 +00002916 }
Mugunthan V Ndf828592012-03-18 20:17:54 +00002917
2918 memset(&dma_params, 0, sizeof(dma_params));
Richard Cochran549985e2012-11-14 09:07:56 +00002919 memset(&ale_params, 0, sizeof(ale_params));
2920
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03002921 switch (cpsw->version) {
Richard Cochran549985e2012-11-14 09:07:56 +00002922 case CPSW_VERSION_1:
Ivan Khoronzhuk5d8d0d42016-08-10 02:22:39 +03002923 cpsw->host_port_regs = ss_regs + CPSW1_HOST_PORT_OFFSET;
Grygorii Strashko8a2c9a52016-12-06 18:00:41 -06002924 cpts_regs = ss_regs + CPSW1_CPTS_OFFSET;
Ivan Khoronzhuk5d8d0d42016-08-10 02:22:39 +03002925 cpsw->hw_stats = ss_regs + CPSW1_HW_STATS;
Richard Cochran549985e2012-11-14 09:07:56 +00002926 dma_params.dmaregs = ss_regs + CPSW1_CPDMA_OFFSET;
2927 dma_params.txhdp = ss_regs + CPSW1_STATERAM_OFFSET;
2928 ale_params.ale_regs = ss_regs + CPSW1_ALE_OFFSET;
2929 slave_offset = CPSW1_SLAVE_OFFSET;
2930 slave_size = CPSW1_SLAVE_SIZE;
2931 sliver_offset = CPSW1_SLIVER_OFFSET;
2932 dma_params.desc_mem_phys = 0;
2933 break;
2934 case CPSW_VERSION_2:
Mugunthan V Nc193f362013-08-05 17:30:05 +05302935 case CPSW_VERSION_3:
Mugunthan V N926489b2013-08-12 17:11:15 +05302936 case CPSW_VERSION_4:
Ivan Khoronzhuk5d8d0d42016-08-10 02:22:39 +03002937 cpsw->host_port_regs = ss_regs + CPSW2_HOST_PORT_OFFSET;
Grygorii Strashko8a2c9a52016-12-06 18:00:41 -06002938 cpts_regs = ss_regs + CPSW2_CPTS_OFFSET;
Ivan Khoronzhuk5d8d0d42016-08-10 02:22:39 +03002939 cpsw->hw_stats = ss_regs + CPSW2_HW_STATS;
Richard Cochran549985e2012-11-14 09:07:56 +00002940 dma_params.dmaregs = ss_regs + CPSW2_CPDMA_OFFSET;
2941 dma_params.txhdp = ss_regs + CPSW2_STATERAM_OFFSET;
2942 ale_params.ale_regs = ss_regs + CPSW2_ALE_OFFSET;
2943 slave_offset = CPSW2_SLAVE_OFFSET;
2944 slave_size = CPSW2_SLAVE_SIZE;
2945 sliver_offset = CPSW2_SLIVER_OFFSET;
2946 dma_params.desc_mem_phys =
Daniel Mackaa1a15e2013-09-21 00:50:38 +05302947 (u32 __force) ss_res->start + CPSW2_BD_OFFSET;
Richard Cochran549985e2012-11-14 09:07:56 +00002948 break;
2949 default:
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03002950 dev_err(priv->dev, "unknown version 0x%08x\n", cpsw->version);
Richard Cochran549985e2012-11-14 09:07:56 +00002951 ret = -ENODEV;
Johan Hovolda4e32b02016-11-17 17:40:00 +01002952 goto clean_dt_ret;
Richard Cochran549985e2012-11-14 09:07:56 +00002953 }
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03002954 for (i = 0; i < cpsw->data.slaves; i++) {
2955 struct cpsw_slave *slave = &cpsw->slaves[i];
2956
2957 cpsw_slave_init(slave, cpsw, slave_offset, sliver_offset);
Richard Cochran549985e2012-11-14 09:07:56 +00002958 slave_offset += slave_size;
2959 sliver_offset += SLIVER_SIZE;
2960 }
2961
Mugunthan V Ndf828592012-03-18 20:17:54 +00002962 dma_params.dev = &pdev->dev;
Richard Cochran549985e2012-11-14 09:07:56 +00002963 dma_params.rxthresh = dma_params.dmaregs + CPDMA_RXTHRESH;
2964 dma_params.rxfree = dma_params.dmaregs + CPDMA_RXFREE;
2965 dma_params.rxhdp = dma_params.txhdp + CPDMA_RXHDP;
2966 dma_params.txcp = dma_params.txhdp + CPDMA_TXCP;
2967 dma_params.rxcp = dma_params.txhdp + CPDMA_RXCP;
Mugunthan V Ndf828592012-03-18 20:17:54 +00002968
2969 dma_params.num_chan = data->channels;
2970 dma_params.has_soft_reset = true;
2971 dma_params.min_packet_size = CPSW_MIN_PACKET_SIZE;
2972 dma_params.desc_mem_size = data->bd_ram_size;
2973 dma_params.desc_align = 16;
2974 dma_params.has_ext_regs = true;
Richard Cochran549985e2012-11-14 09:07:56 +00002975 dma_params.desc_hw_addr = dma_params.desc_mem_phys;
Ivan Khoronzhuk83fcad02016-11-29 17:00:49 +02002976 dma_params.bus_freq_mhz = cpsw->bus_freq_mhz;
Grygorii Strashko90225bf2017-01-06 14:07:33 -06002977 dma_params.descs_pool_size = descs_pool_size;
Mugunthan V Ndf828592012-03-18 20:17:54 +00002978
Ivan Khoronzhuk2c836bd2016-08-10 02:22:40 +03002979 cpsw->dma = cpdma_ctlr_create(&dma_params);
2980 if (!cpsw->dma) {
Mugunthan V Ndf828592012-03-18 20:17:54 +00002981 dev_err(priv->dev, "error initializing dma\n");
2982 ret = -ENOMEM;
Johan Hovolda4e32b02016-11-17 17:40:00 +01002983 goto clean_dt_ret;
Mugunthan V Ndf828592012-03-18 20:17:54 +00002984 }
2985
Ivan Khoronzhuk8feb0a12016-11-29 17:00:51 +02002986 cpsw->txv[0].ch = cpdma_chan_create(cpsw->dma, 0, cpsw_tx_handler, 0);
2987 cpsw->rxv[0].ch = cpdma_chan_create(cpsw->dma, 0, cpsw_rx_handler, 1);
2988 if (WARN_ON(!cpsw->rxv[0].ch || !cpsw->txv[0].ch)) {
Mugunthan V Ndf828592012-03-18 20:17:54 +00002989 dev_err(priv->dev, "error initializing dma channels\n");
2990 ret = -ENOMEM;
2991 goto clean_dma_ret;
2992 }
2993
Mugunthan V Ndf828592012-03-18 20:17:54 +00002994 ale_params.dev = &ndev->dev;
Mugunthan V Ndf828592012-03-18 20:17:54 +00002995 ale_params.ale_ageout = ale_ageout;
2996 ale_params.ale_entries = data->ale_entries;
2997 ale_params.ale_ports = data->slaves;
2998
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03002999 cpsw->ale = cpsw_ale_create(&ale_params);
3000 if (!cpsw->ale) {
Mugunthan V Ndf828592012-03-18 20:17:54 +00003001 dev_err(priv->dev, "error initializing ale engine\n");
3002 ret = -ENODEV;
3003 goto clean_dma_ret;
3004 }
3005
Grygorii Strashko4a88fb92016-12-06 18:00:42 -06003006 cpsw->cpts = cpts_create(cpsw->dev, cpts_regs, cpsw->dev->of_node);
Grygorii Strashko8a2c9a52016-12-06 18:00:41 -06003007 if (IS_ERR(cpsw->cpts)) {
3008 ret = PTR_ERR(cpsw->cpts);
3009 goto clean_ale_ret;
3010 }
3011
Felipe Balbic03abd82015-01-16 10:11:12 -06003012 ndev->irq = platform_get_irq(pdev, 1);
Mugunthan V Ndf828592012-03-18 20:17:54 +00003013 if (ndev->irq < 0) {
3014 dev_err(priv->dev, "error getting irq resource\n");
Julia Lawallc1e33342015-12-26 20:12:13 +01003015 ret = ndev->irq;
Mugunthan V Ndf828592012-03-18 20:17:54 +00003016 goto clean_ale_ret;
3017 }
3018
Mugunthan V N7da11602015-08-12 15:22:53 +05303019 of_id = of_match_device(cpsw_of_mtable, &pdev->dev);
3020 if (of_id) {
3021 pdev->id_entry = of_id->data;
3022 if (pdev->id_entry->driver_data)
Ivan Khoronzhuke38b5a32016-08-10 02:22:41 +03003023 cpsw->quirk_irq = true;
Mugunthan V N7da11602015-08-12 15:22:53 +05303024 }
3025
Felipe Balbic03abd82015-01-16 10:11:12 -06003026 /* Grab RX and TX IRQs. Note that we also have RX_THRESHOLD and
3027 * MISC IRQs which are always kept disabled with this driver so
3028 * we will not request them.
3029 *
3030 * If anyone wants to implement support for those, make sure to
3031 * first request and append them to irqs_table array.
3032 */
Daniel Mackc2b32e52014-09-04 09:00:23 +02003033
Felipe Balbic03abd82015-01-16 10:11:12 -06003034 /* RX IRQ */
Felipe Balbi5087b912015-01-16 10:11:11 -06003035 irq = platform_get_irq(pdev, 1);
Julia Lawallc1e33342015-12-26 20:12:13 +01003036 if (irq < 0) {
3037 ret = irq;
Felipe Balbi5087b912015-01-16 10:11:11 -06003038 goto clean_ale_ret;
Julia Lawallc1e33342015-12-26 20:12:13 +01003039 }
Felipe Balbi5087b912015-01-16 10:11:11 -06003040
Ivan Khoronzhuke38b5a32016-08-10 02:22:41 +03003041 cpsw->irqs_table[0] = irq;
Felipe Balbic03abd82015-01-16 10:11:12 -06003042 ret = devm_request_irq(&pdev->dev, irq, cpsw_rx_interrupt,
Ivan Khoronzhukdbc4ec52016-08-10 02:22:43 +03003043 0, dev_name(&pdev->dev), cpsw);
Felipe Balbi5087b912015-01-16 10:11:11 -06003044 if (ret < 0) {
3045 dev_err(priv->dev, "error attaching irq (%d)\n", ret);
3046 goto clean_ale_ret;
3047 }
3048
Felipe Balbic03abd82015-01-16 10:11:12 -06003049 /* TX IRQ */
Felipe Balbi5087b912015-01-16 10:11:11 -06003050 irq = platform_get_irq(pdev, 2);
Julia Lawallc1e33342015-12-26 20:12:13 +01003051 if (irq < 0) {
3052 ret = irq;
Felipe Balbi5087b912015-01-16 10:11:11 -06003053 goto clean_ale_ret;
Julia Lawallc1e33342015-12-26 20:12:13 +01003054 }
Felipe Balbi5087b912015-01-16 10:11:11 -06003055
Ivan Khoronzhuke38b5a32016-08-10 02:22:41 +03003056 cpsw->irqs_table[1] = irq;
Felipe Balbic03abd82015-01-16 10:11:12 -06003057 ret = devm_request_irq(&pdev->dev, irq, cpsw_tx_interrupt,
Ivan Khoronzhukdbc4ec52016-08-10 02:22:43 +03003058 0, dev_name(&pdev->dev), cpsw);
Felipe Balbi5087b912015-01-16 10:11:11 -06003059 if (ret < 0) {
3060 dev_err(priv->dev, "error attaching irq (%d)\n", ret);
3061 goto clean_ale_ret;
3062 }
Daniel Mackc2b32e52014-09-04 09:00:23 +02003063
Patrick McHardyf6469682013-04-19 02:04:27 +00003064 ndev->features |= NETIF_F_HW_VLAN_CTAG_FILTER;
Mugunthan V Ndf828592012-03-18 20:17:54 +00003065
3066 ndev->netdev_ops = &cpsw_netdev_ops;
Wilfried Klaebe7ad24ea2014-05-11 00:12:32 +00003067 ndev->ethtool_ops = &cpsw_ethtool_ops;
Ivan Khoronzhukdbc4ec52016-08-10 02:22:43 +03003068 netif_napi_add(ndev, &cpsw->napi_rx, cpsw_rx_poll, CPSW_POLL_WEIGHT);
3069 netif_tx_napi_add(ndev, &cpsw->napi_tx, cpsw_tx_poll, CPSW_POLL_WEIGHT);
Ivan Khoronzhuk0be01b82016-12-10 14:23:49 +02003070 cpsw_split_res(ndev);
Mugunthan V Ndf828592012-03-18 20:17:54 +00003071
3072 /* register the network device */
3073 SET_NETDEV_DEV(ndev, &pdev->dev);
3074 ret = register_netdev(ndev);
3075 if (ret) {
3076 dev_err(priv->dev, "error registering net device\n");
3077 ret = -ENODEV;
Daniel Mackaa1a15e2013-09-21 00:50:38 +05303078 goto clean_ale_ret;
Mugunthan V Ndf828592012-03-18 20:17:54 +00003079 }
3080
Grygorii Strashko90225bf2017-01-06 14:07:33 -06003081 cpsw_notice(priv, probe,
3082 "initialized device (regs %pa, irq %d, pool size %d)\n",
3083 &ss_res->start, ndev->irq, dma_params.descs_pool_size);
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03003084 if (cpsw->data.dual_emac) {
Ivan Khoronzhuk56e31bd2016-08-10 02:22:38 +03003085 ret = cpsw_probe_dual_emac(priv);
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00003086 if (ret) {
3087 cpsw_err(priv, probe, "error probe slave 2 emac interface\n");
Johan Hovolda7fe9d42016-11-17 17:40:02 +01003088 goto clean_unregister_netdev_ret;
Mugunthan V Nd9ba8f92013-02-11 09:52:20 +00003089 }
3090 }
3091
Johan Hovoldc46ab7e2016-11-17 17:39:58 +01003092 pm_runtime_put(&pdev->dev);
3093
Mugunthan V Ndf828592012-03-18 20:17:54 +00003094 return 0;
3095
Johan Hovolda7fe9d42016-11-17 17:40:02 +01003096clean_unregister_netdev_ret:
3097 unregister_netdev(ndev);
Mugunthan V Ndf828592012-03-18 20:17:54 +00003098clean_ale_ret:
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03003099 cpsw_ale_destroy(cpsw->ale);
Mugunthan V Ndf828592012-03-18 20:17:54 +00003100clean_dma_ret:
Ivan Khoronzhuk2c836bd2016-08-10 02:22:40 +03003101 cpdma_ctlr_destroy(cpsw->dma);
Johan Hovolda4e32b02016-11-17 17:40:00 +01003102clean_dt_ret:
3103 cpsw_remove_dt(pdev);
Johan Hovoldc46ab7e2016-11-17 17:39:58 +01003104 pm_runtime_put_sync(&pdev->dev);
Daniel Mackaa1a15e2013-09-21 00:50:38 +05303105clean_runtime_disable_ret:
Mugunthan V Nf150bd72012-07-17 08:09:50 +00003106 pm_runtime_disable(&pdev->dev);
Mugunthan V Ndf828592012-03-18 20:17:54 +00003107clean_ndev_ret:
Sebastian Siewiord1bd9ac2013-04-24 08:48:23 +00003108 free_netdev(priv->ndev);
Mugunthan V Ndf828592012-03-18 20:17:54 +00003109 return ret;
3110}
3111
Bill Pemberton663e12e2012-12-03 09:23:45 -05003112static int cpsw_remove(struct platform_device *pdev)
Mugunthan V Ndf828592012-03-18 20:17:54 +00003113{
3114 struct net_device *ndev = platform_get_drvdata(pdev);
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03003115 struct cpsw_common *cpsw = ndev_to_cpsw(ndev);
Grygorii Strashko8a0b6dc2016-07-28 20:50:35 +03003116 int ret;
3117
3118 ret = pm_runtime_get_sync(&pdev->dev);
3119 if (ret < 0) {
3120 pm_runtime_put_noidle(&pdev->dev);
3121 return ret;
3122 }
Mugunthan V Ndf828592012-03-18 20:17:54 +00003123
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03003124 if (cpsw->data.dual_emac)
3125 unregister_netdev(cpsw->slaves[1].ndev);
Sebastian Siewiord1bd9ac2013-04-24 08:48:23 +00003126 unregister_netdev(ndev);
Mugunthan V Ndf828592012-03-18 20:17:54 +00003127
Grygorii Strashko8a2c9a52016-12-06 18:00:41 -06003128 cpts_release(cpsw->cpts);
Ivan Khoronzhuk2a05a622016-08-10 02:22:44 +03003129 cpsw_ale_destroy(cpsw->ale);
Ivan Khoronzhuk2c836bd2016-08-10 02:22:40 +03003130 cpdma_ctlr_destroy(cpsw->dma);
Johan Hovolda4e32b02016-11-17 17:40:00 +01003131 cpsw_remove_dt(pdev);
Grygorii Strashko8a0b6dc2016-07-28 20:50:35 +03003132 pm_runtime_put_sync(&pdev->dev);
3133 pm_runtime_disable(&pdev->dev);
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03003134 if (cpsw->data.dual_emac)
3135 free_netdev(cpsw->slaves[1].ndev);
Mugunthan V Ndf828592012-03-18 20:17:54 +00003136 free_netdev(ndev);
Mugunthan V Ndf828592012-03-18 20:17:54 +00003137 return 0;
3138}
3139
Grygorii Strashko8963a502015-02-27 13:19:45 +02003140#ifdef CONFIG_PM_SLEEP
Mugunthan V Ndf828592012-03-18 20:17:54 +00003141static int cpsw_suspend(struct device *dev)
3142{
3143 struct platform_device *pdev = to_platform_device(dev);
3144 struct net_device *ndev = platform_get_drvdata(pdev);
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03003145 struct cpsw_common *cpsw = ndev_to_cpsw(ndev);
Mugunthan V Ndf828592012-03-18 20:17:54 +00003146
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03003147 if (cpsw->data.dual_emac) {
Mugunthan V N618073e2014-09-11 22:52:38 +05303148 int i;
Daniel Mack1e7a2e22013-11-15 08:29:16 +01003149
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03003150 for (i = 0; i < cpsw->data.slaves; i++) {
3151 if (netif_running(cpsw->slaves[i].ndev))
3152 cpsw_ndo_stop(cpsw->slaves[i].ndev);
Mugunthan V N618073e2014-09-11 22:52:38 +05303153 }
3154 } else {
3155 if (netif_running(ndev))
3156 cpsw_ndo_stop(ndev);
Mugunthan V N618073e2014-09-11 22:52:38 +05303157 }
Daniel Mack1e7a2e22013-11-15 08:29:16 +01003158
Mugunthan V N739683b2013-06-06 23:45:14 +05303159 /* Select sleep pin state */
Ivan Khoronzhuk56e31bd2016-08-10 02:22:38 +03003160 pinctrl_pm_select_sleep_state(dev);
Mugunthan V N739683b2013-06-06 23:45:14 +05303161
Mugunthan V Ndf828592012-03-18 20:17:54 +00003162 return 0;
3163}
3164
3165static int cpsw_resume(struct device *dev)
3166{
3167 struct platform_device *pdev = to_platform_device(dev);
3168 struct net_device *ndev = platform_get_drvdata(pdev);
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03003169 struct cpsw_common *cpsw = netdev_priv(ndev);
Mugunthan V Ndf828592012-03-18 20:17:54 +00003170
Mugunthan V N739683b2013-06-06 23:45:14 +05303171 /* Select default pin state */
Ivan Khoronzhuk56e31bd2016-08-10 02:22:38 +03003172 pinctrl_pm_select_default_state(dev);
Mugunthan V N739683b2013-06-06 23:45:14 +05303173
Grygorii Strashko4ccfd632016-11-29 16:27:03 -06003174 /* shut up ASSERT_RTNL() warning in netif_set_real_num_tx/rx_queues */
3175 rtnl_lock();
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03003176 if (cpsw->data.dual_emac) {
Mugunthan V N618073e2014-09-11 22:52:38 +05303177 int i;
3178
Ivan Khoronzhuk606f3992016-08-10 02:22:42 +03003179 for (i = 0; i < cpsw->data.slaves; i++) {
3180 if (netif_running(cpsw->slaves[i].ndev))
3181 cpsw_ndo_open(cpsw->slaves[i].ndev);
Mugunthan V N618073e2014-09-11 22:52:38 +05303182 }
3183 } else {
3184 if (netif_running(ndev))
3185 cpsw_ndo_open(ndev);
3186 }
Grygorii Strashko4ccfd632016-11-29 16:27:03 -06003187 rtnl_unlock();
3188
Mugunthan V Ndf828592012-03-18 20:17:54 +00003189 return 0;
3190}
Grygorii Strashko8963a502015-02-27 13:19:45 +02003191#endif
Mugunthan V Ndf828592012-03-18 20:17:54 +00003192
Grygorii Strashko8963a502015-02-27 13:19:45 +02003193static SIMPLE_DEV_PM_OPS(cpsw_pm_ops, cpsw_suspend, cpsw_resume);
Mugunthan V Ndf828592012-03-18 20:17:54 +00003194
3195static struct platform_driver cpsw_driver = {
3196 .driver = {
3197 .name = "cpsw",
Mugunthan V Ndf828592012-03-18 20:17:54 +00003198 .pm = &cpsw_pm_ops,
Sachin Kamat1e5c76d2013-09-30 09:55:12 +05303199 .of_match_table = cpsw_of_mtable,
Mugunthan V Ndf828592012-03-18 20:17:54 +00003200 },
3201 .probe = cpsw_probe,
Bill Pemberton663e12e2012-12-03 09:23:45 -05003202 .remove = cpsw_remove,
Mugunthan V Ndf828592012-03-18 20:17:54 +00003203};
3204
Grygorii Strashko6fb3b6b52015-10-23 14:41:12 +03003205module_platform_driver(cpsw_driver);
Mugunthan V Ndf828592012-03-18 20:17:54 +00003206
3207MODULE_LICENSE("GPL");
3208MODULE_AUTHOR("Cyril Chemparathy <cyril@ti.com>");
3209MODULE_AUTHOR("Mugunthan V N <mugunthanvnm@ti.com>");
3210MODULE_DESCRIPTION("TI CPSW Ethernet driver");