Merge tag 'cxl-for-6.3' of git://git.kernel.org/pub/scm/linux/kernel/git/cxl/cxl
[platform/kernel/linux-rpi.git] / drivers / net / dsa / rzn1_a5psw.c
1 // SPDX-License-Identifier: GPL-2.0-only
2 /*
3  * Copyright (C) 2022 Schneider-Electric
4  *
5  * Clément Léger <clement.leger@bootlin.com>
6  */
7
8 #include <linux/clk.h>
9 #include <linux/etherdevice.h>
10 #include <linux/if_bridge.h>
11 #include <linux/if_ether.h>
12 #include <linux/kernel.h>
13 #include <linux/module.h>
14 #include <linux/of.h>
15 #include <linux/of_mdio.h>
16 #include <net/dsa.h>
17
18 #include "rzn1_a5psw.h"
19
20 struct a5psw_stats {
21         u16 offset;
22         const char name[ETH_GSTRING_LEN];
23 };
24
25 #define STAT_DESC(_offset) {    \
26         .offset = A5PSW_##_offset,      \
27         .name = __stringify(_offset),   \
28 }
29
30 static const struct a5psw_stats a5psw_stats[] = {
31         STAT_DESC(aFramesTransmittedOK),
32         STAT_DESC(aFramesReceivedOK),
33         STAT_DESC(aFrameCheckSequenceErrors),
34         STAT_DESC(aAlignmentErrors),
35         STAT_DESC(aOctetsTransmittedOK),
36         STAT_DESC(aOctetsReceivedOK),
37         STAT_DESC(aTxPAUSEMACCtrlFrames),
38         STAT_DESC(aRxPAUSEMACCtrlFrames),
39         STAT_DESC(ifInErrors),
40         STAT_DESC(ifOutErrors),
41         STAT_DESC(ifInUcastPkts),
42         STAT_DESC(ifInMulticastPkts),
43         STAT_DESC(ifInBroadcastPkts),
44         STAT_DESC(ifOutDiscards),
45         STAT_DESC(ifOutUcastPkts),
46         STAT_DESC(ifOutMulticastPkts),
47         STAT_DESC(ifOutBroadcastPkts),
48         STAT_DESC(etherStatsDropEvents),
49         STAT_DESC(etherStatsOctets),
50         STAT_DESC(etherStatsPkts),
51         STAT_DESC(etherStatsUndersizePkts),
52         STAT_DESC(etherStatsOversizePkts),
53         STAT_DESC(etherStatsPkts64Octets),
54         STAT_DESC(etherStatsPkts65to127Octets),
55         STAT_DESC(etherStatsPkts128to255Octets),
56         STAT_DESC(etherStatsPkts256to511Octets),
57         STAT_DESC(etherStatsPkts1024to1518Octets),
58         STAT_DESC(etherStatsPkts1519toXOctets),
59         STAT_DESC(etherStatsJabbers),
60         STAT_DESC(etherStatsFragments),
61         STAT_DESC(VLANReceived),
62         STAT_DESC(VLANTransmitted),
63         STAT_DESC(aDeferred),
64         STAT_DESC(aMultipleCollisions),
65         STAT_DESC(aSingleCollisions),
66         STAT_DESC(aLateCollisions),
67         STAT_DESC(aExcessiveCollisions),
68         STAT_DESC(aCarrierSenseErrors),
69 };
70
71 static void a5psw_reg_writel(struct a5psw *a5psw, int offset, u32 value)
72 {
73         writel(value, a5psw->base + offset);
74 }
75
76 static u32 a5psw_reg_readl(struct a5psw *a5psw, int offset)
77 {
78         return readl(a5psw->base + offset);
79 }
80
81 static void a5psw_reg_rmw(struct a5psw *a5psw, int offset, u32 mask, u32 val)
82 {
83         u32 reg;
84
85         spin_lock(&a5psw->reg_lock);
86
87         reg = a5psw_reg_readl(a5psw, offset);
88         reg &= ~mask;
89         reg |= val;
90         a5psw_reg_writel(a5psw, offset, reg);
91
92         spin_unlock(&a5psw->reg_lock);
93 }
94
95 static enum dsa_tag_protocol a5psw_get_tag_protocol(struct dsa_switch *ds,
96                                                     int port,
97                                                     enum dsa_tag_protocol mp)
98 {
99         return DSA_TAG_PROTO_RZN1_A5PSW;
100 }
101
102 static void a5psw_port_pattern_set(struct a5psw *a5psw, int port, int pattern,
103                                    bool enable)
104 {
105         u32 rx_match = 0;
106
107         if (enable)
108                 rx_match |= A5PSW_RXMATCH_CONFIG_PATTERN(pattern);
109
110         a5psw_reg_rmw(a5psw, A5PSW_RXMATCH_CONFIG(port),
111                       A5PSW_RXMATCH_CONFIG_PATTERN(pattern), rx_match);
112 }
113
114 static void a5psw_port_mgmtfwd_set(struct a5psw *a5psw, int port, bool enable)
115 {
116         /* Enable "management forward" pattern matching, this will forward
117          * packets from this port only towards the management port and thus
118          * isolate the port.
119          */
120         a5psw_port_pattern_set(a5psw, port, A5PSW_PATTERN_MGMTFWD, enable);
121 }
122
123 static void a5psw_port_enable_set(struct a5psw *a5psw, int port, bool enable)
124 {
125         u32 port_ena = 0;
126
127         if (enable)
128                 port_ena |= A5PSW_PORT_ENA_TX_RX(port);
129
130         a5psw_reg_rmw(a5psw, A5PSW_PORT_ENA, A5PSW_PORT_ENA_TX_RX(port),
131                       port_ena);
132 }
133
134 static int a5psw_lk_execute_ctrl(struct a5psw *a5psw, u32 *ctrl)
135 {
136         int ret;
137
138         a5psw_reg_writel(a5psw, A5PSW_LK_ADDR_CTRL, *ctrl);
139
140         ret = readl_poll_timeout(a5psw->base + A5PSW_LK_ADDR_CTRL, *ctrl,
141                                  !(*ctrl & A5PSW_LK_ADDR_CTRL_BUSY),
142                                  A5PSW_LK_BUSY_USEC_POLL, A5PSW_CTRL_TIMEOUT);
143         if (ret)
144                 dev_err(a5psw->dev, "LK_CTRL timeout waiting for BUSY bit\n");
145
146         return ret;
147 }
148
149 static void a5psw_port_fdb_flush(struct a5psw *a5psw, int port)
150 {
151         u32 ctrl = A5PSW_LK_ADDR_CTRL_DELETE_PORT | BIT(port);
152
153         mutex_lock(&a5psw->lk_lock);
154         a5psw_lk_execute_ctrl(a5psw, &ctrl);
155         mutex_unlock(&a5psw->lk_lock);
156 }
157
158 static void a5psw_port_authorize_set(struct a5psw *a5psw, int port,
159                                      bool authorize)
160 {
161         u32 reg = a5psw_reg_readl(a5psw, A5PSW_AUTH_PORT(port));
162
163         if (authorize)
164                 reg |= A5PSW_AUTH_PORT_AUTHORIZED;
165         else
166                 reg &= ~A5PSW_AUTH_PORT_AUTHORIZED;
167
168         a5psw_reg_writel(a5psw, A5PSW_AUTH_PORT(port), reg);
169 }
170
171 static void a5psw_port_disable(struct dsa_switch *ds, int port)
172 {
173         struct a5psw *a5psw = ds->priv;
174
175         a5psw_port_authorize_set(a5psw, port, false);
176         a5psw_port_enable_set(a5psw, port, false);
177 }
178
179 static int a5psw_port_enable(struct dsa_switch *ds, int port,
180                              struct phy_device *phy)
181 {
182         struct a5psw *a5psw = ds->priv;
183
184         a5psw_port_authorize_set(a5psw, port, true);
185         a5psw_port_enable_set(a5psw, port, true);
186
187         return 0;
188 }
189
190 static int a5psw_port_change_mtu(struct dsa_switch *ds, int port, int new_mtu)
191 {
192         struct a5psw *a5psw = ds->priv;
193
194         new_mtu += ETH_HLEN + A5PSW_EXTRA_MTU_LEN + ETH_FCS_LEN;
195         a5psw_reg_writel(a5psw, A5PSW_FRM_LENGTH(port), new_mtu);
196
197         return 0;
198 }
199
200 static int a5psw_port_max_mtu(struct dsa_switch *ds, int port)
201 {
202         return A5PSW_MAX_MTU;
203 }
204
205 static void a5psw_phylink_get_caps(struct dsa_switch *ds, int port,
206                                    struct phylink_config *config)
207 {
208         unsigned long *intf = config->supported_interfaces;
209
210         config->mac_capabilities = MAC_1000FD;
211
212         if (dsa_is_cpu_port(ds, port)) {
213                 /* GMII is used internally and GMAC2 is connected to the switch
214                  * using 1000Mbps Full-Duplex mode only (cf ethernet manual)
215                  */
216                 __set_bit(PHY_INTERFACE_MODE_GMII, intf);
217         } else {
218                 config->mac_capabilities |= MAC_100 | MAC_10;
219                 phy_interface_set_rgmii(intf);
220                 __set_bit(PHY_INTERFACE_MODE_RMII, intf);
221                 __set_bit(PHY_INTERFACE_MODE_MII, intf);
222         }
223 }
224
225 static struct phylink_pcs *
226 a5psw_phylink_mac_select_pcs(struct dsa_switch *ds, int port,
227                              phy_interface_t interface)
228 {
229         struct dsa_port *dp = dsa_to_port(ds, port);
230         struct a5psw *a5psw = ds->priv;
231
232         if (!dsa_port_is_cpu(dp) && a5psw->pcs[port])
233                 return a5psw->pcs[port];
234
235         return NULL;
236 }
237
238 static void a5psw_phylink_mac_link_down(struct dsa_switch *ds, int port,
239                                         unsigned int mode,
240                                         phy_interface_t interface)
241 {
242         struct a5psw *a5psw = ds->priv;
243         u32 cmd_cfg;
244
245         cmd_cfg = a5psw_reg_readl(a5psw, A5PSW_CMD_CFG(port));
246         cmd_cfg &= ~(A5PSW_CMD_CFG_RX_ENA | A5PSW_CMD_CFG_TX_ENA);
247         a5psw_reg_writel(a5psw, A5PSW_CMD_CFG(port), cmd_cfg);
248 }
249
250 static void a5psw_phylink_mac_link_up(struct dsa_switch *ds, int port,
251                                       unsigned int mode,
252                                       phy_interface_t interface,
253                                       struct phy_device *phydev, int speed,
254                                       int duplex, bool tx_pause, bool rx_pause)
255 {
256         u32 cmd_cfg = A5PSW_CMD_CFG_RX_ENA | A5PSW_CMD_CFG_TX_ENA |
257                       A5PSW_CMD_CFG_TX_CRC_APPEND;
258         struct a5psw *a5psw = ds->priv;
259
260         if (speed == SPEED_1000)
261                 cmd_cfg |= A5PSW_CMD_CFG_ETH_SPEED;
262
263         if (duplex == DUPLEX_HALF)
264                 cmd_cfg |= A5PSW_CMD_CFG_HD_ENA;
265
266         cmd_cfg |= A5PSW_CMD_CFG_CNTL_FRM_ENA;
267
268         if (!rx_pause)
269                 cmd_cfg &= ~A5PSW_CMD_CFG_PAUSE_IGNORE;
270
271         a5psw_reg_writel(a5psw, A5PSW_CMD_CFG(port), cmd_cfg);
272 }
273
274 static int a5psw_set_ageing_time(struct dsa_switch *ds, unsigned int msecs)
275 {
276         struct a5psw *a5psw = ds->priv;
277         unsigned long rate;
278         u64 max, tmp;
279         u32 agetime;
280
281         rate = clk_get_rate(a5psw->clk);
282         max = div64_ul(((u64)A5PSW_LK_AGETIME_MASK * A5PSW_TABLE_ENTRIES * 1024),
283                        rate) * 1000;
284         if (msecs > max)
285                 return -EINVAL;
286
287         tmp = div_u64(rate, MSEC_PER_SEC);
288         agetime = div_u64(msecs * tmp, 1024 * A5PSW_TABLE_ENTRIES);
289
290         a5psw_reg_writel(a5psw, A5PSW_LK_AGETIME, agetime);
291
292         return 0;
293 }
294
295 static void a5psw_flooding_set_resolution(struct a5psw *a5psw, int port,
296                                           bool set)
297 {
298         u8 offsets[] = {A5PSW_UCAST_DEF_MASK, A5PSW_BCAST_DEF_MASK,
299                         A5PSW_MCAST_DEF_MASK};
300         int i;
301
302         if (set)
303                 a5psw->bridged_ports |= BIT(port);
304         else
305                 a5psw->bridged_ports &= ~BIT(port);
306
307         for (i = 0; i < ARRAY_SIZE(offsets); i++)
308                 a5psw_reg_writel(a5psw, offsets[i], a5psw->bridged_ports);
309 }
310
311 static int a5psw_port_bridge_join(struct dsa_switch *ds, int port,
312                                   struct dsa_bridge bridge,
313                                   bool *tx_fwd_offload,
314                                   struct netlink_ext_ack *extack)
315 {
316         struct a5psw *a5psw = ds->priv;
317
318         /* We only support 1 bridge device */
319         if (a5psw->br_dev && bridge.dev != a5psw->br_dev) {
320                 NL_SET_ERR_MSG_MOD(extack,
321                                    "Forwarding offload supported for a single bridge");
322                 return -EOPNOTSUPP;
323         }
324
325         a5psw->br_dev = bridge.dev;
326         a5psw_flooding_set_resolution(a5psw, port, true);
327         a5psw_port_mgmtfwd_set(a5psw, port, false);
328
329         return 0;
330 }
331
332 static void a5psw_port_bridge_leave(struct dsa_switch *ds, int port,
333                                     struct dsa_bridge bridge)
334 {
335         struct a5psw *a5psw = ds->priv;
336
337         a5psw_flooding_set_resolution(a5psw, port, false);
338         a5psw_port_mgmtfwd_set(a5psw, port, true);
339
340         /* No more ports bridged */
341         if (a5psw->bridged_ports == BIT(A5PSW_CPU_PORT))
342                 a5psw->br_dev = NULL;
343 }
344
345 static void a5psw_port_stp_state_set(struct dsa_switch *ds, int port, u8 state)
346 {
347         u32 mask = A5PSW_INPUT_LEARN_DIS(port) | A5PSW_INPUT_LEARN_BLOCK(port);
348         struct a5psw *a5psw = ds->priv;
349         u32 reg = 0;
350
351         switch (state) {
352         case BR_STATE_DISABLED:
353         case BR_STATE_BLOCKING:
354                 reg |= A5PSW_INPUT_LEARN_DIS(port);
355                 reg |= A5PSW_INPUT_LEARN_BLOCK(port);
356                 break;
357         case BR_STATE_LISTENING:
358                 reg |= A5PSW_INPUT_LEARN_DIS(port);
359                 break;
360         case BR_STATE_LEARNING:
361                 reg |= A5PSW_INPUT_LEARN_BLOCK(port);
362                 break;
363         case BR_STATE_FORWARDING:
364         default:
365                 break;
366         }
367
368         a5psw_reg_rmw(a5psw, A5PSW_INPUT_LEARN, mask, reg);
369 }
370
371 static void a5psw_port_fast_age(struct dsa_switch *ds, int port)
372 {
373         struct a5psw *a5psw = ds->priv;
374
375         a5psw_port_fdb_flush(a5psw, port);
376 }
377
378 static int a5psw_lk_execute_lookup(struct a5psw *a5psw, union lk_data *lk_data,
379                                    u16 *entry)
380 {
381         u32 ctrl;
382         int ret;
383
384         a5psw_reg_writel(a5psw, A5PSW_LK_DATA_LO, lk_data->lo);
385         a5psw_reg_writel(a5psw, A5PSW_LK_DATA_HI, lk_data->hi);
386
387         ctrl = A5PSW_LK_ADDR_CTRL_LOOKUP;
388         ret = a5psw_lk_execute_ctrl(a5psw, &ctrl);
389         if (ret)
390                 return ret;
391
392         *entry = ctrl & A5PSW_LK_ADDR_CTRL_ADDRESS;
393
394         return 0;
395 }
396
397 static int a5psw_port_fdb_add(struct dsa_switch *ds, int port,
398                               const unsigned char *addr, u16 vid,
399                               struct dsa_db db)
400 {
401         struct a5psw *a5psw = ds->priv;
402         union lk_data lk_data = {0};
403         bool inc_learncount = false;
404         int ret = 0;
405         u16 entry;
406         u32 reg;
407
408         ether_addr_copy(lk_data.entry.mac, addr);
409         lk_data.entry.port_mask = BIT(port);
410
411         mutex_lock(&a5psw->lk_lock);
412
413         /* Set the value to be written in the lookup table */
414         ret = a5psw_lk_execute_lookup(a5psw, &lk_data, &entry);
415         if (ret)
416                 goto lk_unlock;
417
418         lk_data.hi = a5psw_reg_readl(a5psw, A5PSW_LK_DATA_HI);
419         if (!lk_data.entry.valid) {
420                 inc_learncount = true;
421                 /* port_mask set to 0x1f when entry is not valid, clear it */
422                 lk_data.entry.port_mask = 0;
423                 lk_data.entry.prio = 0;
424         }
425
426         lk_data.entry.port_mask |= BIT(port);
427         lk_data.entry.is_static = 1;
428         lk_data.entry.valid = 1;
429
430         a5psw_reg_writel(a5psw, A5PSW_LK_DATA_HI, lk_data.hi);
431
432         reg = A5PSW_LK_ADDR_CTRL_WRITE | entry;
433         ret = a5psw_lk_execute_ctrl(a5psw, &reg);
434         if (ret)
435                 goto lk_unlock;
436
437         if (inc_learncount) {
438                 reg = A5PSW_LK_LEARNCOUNT_MODE_INC;
439                 a5psw_reg_writel(a5psw, A5PSW_LK_LEARNCOUNT, reg);
440         }
441
442 lk_unlock:
443         mutex_unlock(&a5psw->lk_lock);
444
445         return ret;
446 }
447
448 static int a5psw_port_fdb_del(struct dsa_switch *ds, int port,
449                               const unsigned char *addr, u16 vid,
450                               struct dsa_db db)
451 {
452         struct a5psw *a5psw = ds->priv;
453         union lk_data lk_data = {0};
454         bool clear = false;
455         u16 entry;
456         u32 reg;
457         int ret;
458
459         ether_addr_copy(lk_data.entry.mac, addr);
460
461         mutex_lock(&a5psw->lk_lock);
462
463         ret = a5psw_lk_execute_lookup(a5psw, &lk_data, &entry);
464         if (ret)
465                 goto lk_unlock;
466
467         lk_data.hi = a5psw_reg_readl(a5psw, A5PSW_LK_DATA_HI);
468
469         /* Our hardware does not associate any VID to the FDB entries so this
470          * means that if two entries were added for the same mac but for
471          * different VID, then, on the deletion of the first one, we would also
472          * delete the second one. Since there is unfortunately nothing we can do
473          * about that, do not return an error...
474          */
475         if (!lk_data.entry.valid)
476                 goto lk_unlock;
477
478         lk_data.entry.port_mask &= ~BIT(port);
479         /* If there is no more port in the mask, clear the entry */
480         if (lk_data.entry.port_mask == 0)
481                 clear = true;
482
483         a5psw_reg_writel(a5psw, A5PSW_LK_DATA_HI, lk_data.hi);
484
485         reg = entry;
486         if (clear)
487                 reg |= A5PSW_LK_ADDR_CTRL_CLEAR;
488         else
489                 reg |= A5PSW_LK_ADDR_CTRL_WRITE;
490
491         ret = a5psw_lk_execute_ctrl(a5psw, &reg);
492         if (ret)
493                 goto lk_unlock;
494
495         /* Decrement LEARNCOUNT */
496         if (clear) {
497                 reg = A5PSW_LK_LEARNCOUNT_MODE_DEC;
498                 a5psw_reg_writel(a5psw, A5PSW_LK_LEARNCOUNT, reg);
499         }
500
501 lk_unlock:
502         mutex_unlock(&a5psw->lk_lock);
503
504         return ret;
505 }
506
507 static int a5psw_port_fdb_dump(struct dsa_switch *ds, int port,
508                                dsa_fdb_dump_cb_t *cb, void *data)
509 {
510         struct a5psw *a5psw = ds->priv;
511         union lk_data lk_data;
512         int i = 0, ret = 0;
513         u32 reg;
514
515         mutex_lock(&a5psw->lk_lock);
516
517         for (i = 0; i < A5PSW_TABLE_ENTRIES; i++) {
518                 reg = A5PSW_LK_ADDR_CTRL_READ | A5PSW_LK_ADDR_CTRL_WAIT | i;
519
520                 ret = a5psw_lk_execute_ctrl(a5psw, &reg);
521                 if (ret)
522                         goto out_unlock;
523
524                 lk_data.hi = a5psw_reg_readl(a5psw, A5PSW_LK_DATA_HI);
525                 /* If entry is not valid or does not contain the port, skip */
526                 if (!lk_data.entry.valid ||
527                     !(lk_data.entry.port_mask & BIT(port)))
528                         continue;
529
530                 lk_data.lo = a5psw_reg_readl(a5psw, A5PSW_LK_DATA_LO);
531
532                 ret = cb(lk_data.entry.mac, 0, lk_data.entry.is_static, data);
533                 if (ret)
534                         goto out_unlock;
535         }
536
537 out_unlock:
538         mutex_unlock(&a5psw->lk_lock);
539
540         return ret;
541 }
542
543 static u64 a5psw_read_stat(struct a5psw *a5psw, u32 offset, int port)
544 {
545         u32 reg_lo, reg_hi;
546
547         reg_lo = a5psw_reg_readl(a5psw, offset + A5PSW_PORT_OFFSET(port));
548         /* A5PSW_STATS_HIWORD is latched on stat read */
549         reg_hi = a5psw_reg_readl(a5psw, A5PSW_STATS_HIWORD);
550
551         return ((u64)reg_hi << 32) | reg_lo;
552 }
553
554 static void a5psw_get_strings(struct dsa_switch *ds, int port, u32 stringset,
555                               uint8_t *data)
556 {
557         unsigned int u;
558
559         if (stringset != ETH_SS_STATS)
560                 return;
561
562         for (u = 0; u < ARRAY_SIZE(a5psw_stats); u++) {
563                 memcpy(data + u * ETH_GSTRING_LEN, a5psw_stats[u].name,
564                        ETH_GSTRING_LEN);
565         }
566 }
567
568 static void a5psw_get_ethtool_stats(struct dsa_switch *ds, int port,
569                                     uint64_t *data)
570 {
571         struct a5psw *a5psw = ds->priv;
572         unsigned int u;
573
574         for (u = 0; u < ARRAY_SIZE(a5psw_stats); u++)
575                 data[u] = a5psw_read_stat(a5psw, a5psw_stats[u].offset, port);
576 }
577
578 static int a5psw_get_sset_count(struct dsa_switch *ds, int port, int sset)
579 {
580         if (sset != ETH_SS_STATS)
581                 return 0;
582
583         return ARRAY_SIZE(a5psw_stats);
584 }
585
586 static void a5psw_get_eth_mac_stats(struct dsa_switch *ds, int port,
587                                     struct ethtool_eth_mac_stats *mac_stats)
588 {
589         struct a5psw *a5psw = ds->priv;
590
591 #define RD(name) a5psw_read_stat(a5psw, A5PSW_##name, port)
592         mac_stats->FramesTransmittedOK = RD(aFramesTransmittedOK);
593         mac_stats->SingleCollisionFrames = RD(aSingleCollisions);
594         mac_stats->MultipleCollisionFrames = RD(aMultipleCollisions);
595         mac_stats->FramesReceivedOK = RD(aFramesReceivedOK);
596         mac_stats->FrameCheckSequenceErrors = RD(aFrameCheckSequenceErrors);
597         mac_stats->AlignmentErrors = RD(aAlignmentErrors);
598         mac_stats->OctetsTransmittedOK = RD(aOctetsTransmittedOK);
599         mac_stats->FramesWithDeferredXmissions = RD(aDeferred);
600         mac_stats->LateCollisions = RD(aLateCollisions);
601         mac_stats->FramesAbortedDueToXSColls = RD(aExcessiveCollisions);
602         mac_stats->FramesLostDueToIntMACXmitError = RD(ifOutErrors);
603         mac_stats->CarrierSenseErrors = RD(aCarrierSenseErrors);
604         mac_stats->OctetsReceivedOK = RD(aOctetsReceivedOK);
605         mac_stats->FramesLostDueToIntMACRcvError = RD(ifInErrors);
606         mac_stats->MulticastFramesXmittedOK = RD(ifOutMulticastPkts);
607         mac_stats->BroadcastFramesXmittedOK = RD(ifOutBroadcastPkts);
608         mac_stats->FramesWithExcessiveDeferral = RD(aDeferred);
609         mac_stats->MulticastFramesReceivedOK = RD(ifInMulticastPkts);
610         mac_stats->BroadcastFramesReceivedOK = RD(ifInBroadcastPkts);
611 #undef RD
612 }
613
614 static const struct ethtool_rmon_hist_range a5psw_rmon_ranges[] = {
615         { 0, 64 },
616         { 65, 127 },
617         { 128, 255 },
618         { 256, 511 },
619         { 512, 1023 },
620         { 1024, 1518 },
621         { 1519, A5PSW_MAX_MTU },
622         {}
623 };
624
625 static void a5psw_get_rmon_stats(struct dsa_switch *ds, int port,
626                                  struct ethtool_rmon_stats *rmon_stats,
627                                  const struct ethtool_rmon_hist_range **ranges)
628 {
629         struct a5psw *a5psw = ds->priv;
630
631 #define RD(name) a5psw_read_stat(a5psw, A5PSW_##name, port)
632         rmon_stats->undersize_pkts = RD(etherStatsUndersizePkts);
633         rmon_stats->oversize_pkts = RD(etherStatsOversizePkts);
634         rmon_stats->fragments = RD(etherStatsFragments);
635         rmon_stats->jabbers = RD(etherStatsJabbers);
636         rmon_stats->hist[0] = RD(etherStatsPkts64Octets);
637         rmon_stats->hist[1] = RD(etherStatsPkts65to127Octets);
638         rmon_stats->hist[2] = RD(etherStatsPkts128to255Octets);
639         rmon_stats->hist[3] = RD(etherStatsPkts256to511Octets);
640         rmon_stats->hist[4] = RD(etherStatsPkts512to1023Octets);
641         rmon_stats->hist[5] = RD(etherStatsPkts1024to1518Octets);
642         rmon_stats->hist[6] = RD(etherStatsPkts1519toXOctets);
643 #undef RD
644
645         *ranges = a5psw_rmon_ranges;
646 }
647
648 static void a5psw_get_eth_ctrl_stats(struct dsa_switch *ds, int port,
649                                      struct ethtool_eth_ctrl_stats *ctrl_stats)
650 {
651         struct a5psw *a5psw = ds->priv;
652         u64 stat;
653
654         stat = a5psw_read_stat(a5psw, A5PSW_aTxPAUSEMACCtrlFrames, port);
655         ctrl_stats->MACControlFramesTransmitted = stat;
656         stat = a5psw_read_stat(a5psw, A5PSW_aRxPAUSEMACCtrlFrames, port);
657         ctrl_stats->MACControlFramesReceived = stat;
658 }
659
660 static int a5psw_setup(struct dsa_switch *ds)
661 {
662         struct a5psw *a5psw = ds->priv;
663         int port, vlan, ret;
664         struct dsa_port *dp;
665         u32 reg;
666
667         /* Validate that there is only 1 CPU port with index A5PSW_CPU_PORT */
668         dsa_switch_for_each_cpu_port(dp, ds) {
669                 if (dp->index != A5PSW_CPU_PORT) {
670                         dev_err(a5psw->dev, "Invalid CPU port\n");
671                         return -EINVAL;
672                 }
673         }
674
675         /* Configure management port */
676         reg = A5PSW_CPU_PORT | A5PSW_MGMT_CFG_DISCARD;
677         a5psw_reg_writel(a5psw, A5PSW_MGMT_CFG, reg);
678
679         /* Set pattern 0 to forward all frame to mgmt port */
680         a5psw_reg_writel(a5psw, A5PSW_PATTERN_CTRL(A5PSW_PATTERN_MGMTFWD),
681                          A5PSW_PATTERN_CTRL_MGMTFWD);
682
683         /* Enable port tagging */
684         reg = FIELD_PREP(A5PSW_MGMT_TAG_CFG_TAGFIELD, ETH_P_DSA_A5PSW);
685         reg |= A5PSW_MGMT_TAG_CFG_ENABLE | A5PSW_MGMT_TAG_CFG_ALL_FRAMES;
686         a5psw_reg_writel(a5psw, A5PSW_MGMT_TAG_CFG, reg);
687
688         /* Enable normal switch operation */
689         reg = A5PSW_LK_ADDR_CTRL_BLOCKING | A5PSW_LK_ADDR_CTRL_LEARNING |
690               A5PSW_LK_ADDR_CTRL_AGEING | A5PSW_LK_ADDR_CTRL_ALLOW_MIGR |
691               A5PSW_LK_ADDR_CTRL_CLEAR_TABLE;
692         a5psw_reg_writel(a5psw, A5PSW_LK_CTRL, reg);
693
694         ret = readl_poll_timeout(a5psw->base + A5PSW_LK_CTRL, reg,
695                                  !(reg & A5PSW_LK_ADDR_CTRL_CLEAR_TABLE),
696                                  A5PSW_LK_BUSY_USEC_POLL, A5PSW_CTRL_TIMEOUT);
697         if (ret) {
698                 dev_err(a5psw->dev, "Failed to clear lookup table\n");
699                 return ret;
700         }
701
702         /* Reset learn count to 0 */
703         reg = A5PSW_LK_LEARNCOUNT_MODE_SET;
704         a5psw_reg_writel(a5psw, A5PSW_LK_LEARNCOUNT, reg);
705
706         /* Clear VLAN resource table */
707         reg = A5PSW_VLAN_RES_WR_PORTMASK | A5PSW_VLAN_RES_WR_TAGMASK;
708         for (vlan = 0; vlan < A5PSW_VLAN_COUNT; vlan++)
709                 a5psw_reg_writel(a5psw, A5PSW_VLAN_RES(vlan), reg);
710
711         /* Reset all ports */
712         dsa_switch_for_each_port(dp, ds) {
713                 port = dp->index;
714
715                 /* Reset the port */
716                 a5psw_reg_writel(a5psw, A5PSW_CMD_CFG(port),
717                                  A5PSW_CMD_CFG_SW_RESET);
718
719                 /* Enable only CPU port */
720                 a5psw_port_enable_set(a5psw, port, dsa_port_is_cpu(dp));
721
722                 if (dsa_port_is_unused(dp))
723                         continue;
724
725                 /* Enable egress flooding for CPU port */
726                 if (dsa_port_is_cpu(dp))
727                         a5psw_flooding_set_resolution(a5psw, port, true);
728
729                 /* Enable management forward only for user ports */
730                 if (dsa_port_is_user(dp))
731                         a5psw_port_mgmtfwd_set(a5psw, port, true);
732         }
733
734         return 0;
735 }
736
737 static const struct dsa_switch_ops a5psw_switch_ops = {
738         .get_tag_protocol = a5psw_get_tag_protocol,
739         .setup = a5psw_setup,
740         .port_disable = a5psw_port_disable,
741         .port_enable = a5psw_port_enable,
742         .phylink_get_caps = a5psw_phylink_get_caps,
743         .phylink_mac_select_pcs = a5psw_phylink_mac_select_pcs,
744         .phylink_mac_link_down = a5psw_phylink_mac_link_down,
745         .phylink_mac_link_up = a5psw_phylink_mac_link_up,
746         .port_change_mtu = a5psw_port_change_mtu,
747         .port_max_mtu = a5psw_port_max_mtu,
748         .get_sset_count = a5psw_get_sset_count,
749         .get_strings = a5psw_get_strings,
750         .get_ethtool_stats = a5psw_get_ethtool_stats,
751         .get_eth_mac_stats = a5psw_get_eth_mac_stats,
752         .get_eth_ctrl_stats = a5psw_get_eth_ctrl_stats,
753         .get_rmon_stats = a5psw_get_rmon_stats,
754         .set_ageing_time = a5psw_set_ageing_time,
755         .port_bridge_join = a5psw_port_bridge_join,
756         .port_bridge_leave = a5psw_port_bridge_leave,
757         .port_stp_state_set = a5psw_port_stp_state_set,
758         .port_fast_age = a5psw_port_fast_age,
759         .port_fdb_add = a5psw_port_fdb_add,
760         .port_fdb_del = a5psw_port_fdb_del,
761         .port_fdb_dump = a5psw_port_fdb_dump,
762 };
763
764 static int a5psw_mdio_wait_busy(struct a5psw *a5psw)
765 {
766         u32 status;
767         int err;
768
769         err = readl_poll_timeout(a5psw->base + A5PSW_MDIO_CFG_STATUS, status,
770                                  !(status & A5PSW_MDIO_CFG_STATUS_BUSY), 10,
771                                  1000 * USEC_PER_MSEC);
772         if (err)
773                 dev_err(a5psw->dev, "MDIO command timeout\n");
774
775         return err;
776 }
777
778 static int a5psw_mdio_read(struct mii_bus *bus, int phy_id, int phy_reg)
779 {
780         struct a5psw *a5psw = bus->priv;
781         u32 cmd, status;
782         int ret;
783
784         cmd = A5PSW_MDIO_COMMAND_READ;
785         cmd |= FIELD_PREP(A5PSW_MDIO_COMMAND_REG_ADDR, phy_reg);
786         cmd |= FIELD_PREP(A5PSW_MDIO_COMMAND_PHY_ADDR, phy_id);
787
788         a5psw_reg_writel(a5psw, A5PSW_MDIO_COMMAND, cmd);
789
790         ret = a5psw_mdio_wait_busy(a5psw);
791         if (ret)
792                 return ret;
793
794         ret = a5psw_reg_readl(a5psw, A5PSW_MDIO_DATA) & A5PSW_MDIO_DATA_MASK;
795
796         status = a5psw_reg_readl(a5psw, A5PSW_MDIO_CFG_STATUS);
797         if (status & A5PSW_MDIO_CFG_STATUS_READERR)
798                 return -EIO;
799
800         return ret;
801 }
802
803 static int a5psw_mdio_write(struct mii_bus *bus, int phy_id, int phy_reg,
804                             u16 phy_data)
805 {
806         struct a5psw *a5psw = bus->priv;
807         u32 cmd;
808
809         cmd = FIELD_PREP(A5PSW_MDIO_COMMAND_REG_ADDR, phy_reg);
810         cmd |= FIELD_PREP(A5PSW_MDIO_COMMAND_PHY_ADDR, phy_id);
811
812         a5psw_reg_writel(a5psw, A5PSW_MDIO_COMMAND, cmd);
813         a5psw_reg_writel(a5psw, A5PSW_MDIO_DATA, phy_data);
814
815         return a5psw_mdio_wait_busy(a5psw);
816 }
817
818 static int a5psw_mdio_config(struct a5psw *a5psw, u32 mdio_freq)
819 {
820         unsigned long rate;
821         unsigned long div;
822         u32 cfgstatus;
823
824         rate = clk_get_rate(a5psw->hclk);
825         div = ((rate / mdio_freq) / 2);
826         if (div > FIELD_MAX(A5PSW_MDIO_CFG_STATUS_CLKDIV) ||
827             div < A5PSW_MDIO_CLK_DIV_MIN) {
828                 dev_err(a5psw->dev, "MDIO clock div %ld out of range\n", div);
829                 return -ERANGE;
830         }
831
832         cfgstatus = FIELD_PREP(A5PSW_MDIO_CFG_STATUS_CLKDIV, div);
833
834         a5psw_reg_writel(a5psw, A5PSW_MDIO_CFG_STATUS, cfgstatus);
835
836         return 0;
837 }
838
839 static int a5psw_probe_mdio(struct a5psw *a5psw, struct device_node *node)
840 {
841         struct device *dev = a5psw->dev;
842         struct mii_bus *bus;
843         u32 mdio_freq;
844         int ret;
845
846         if (of_property_read_u32(node, "clock-frequency", &mdio_freq))
847                 mdio_freq = A5PSW_MDIO_DEF_FREQ;
848
849         ret = a5psw_mdio_config(a5psw, mdio_freq);
850         if (ret)
851                 return ret;
852
853         bus = devm_mdiobus_alloc(dev);
854         if (!bus)
855                 return -ENOMEM;
856
857         bus->name = "a5psw_mdio";
858         bus->read = a5psw_mdio_read;
859         bus->write = a5psw_mdio_write;
860         bus->priv = a5psw;
861         bus->parent = dev;
862         snprintf(bus->id, MII_BUS_ID_SIZE, "%s", dev_name(dev));
863
864         a5psw->mii_bus = bus;
865
866         return devm_of_mdiobus_register(dev, bus, node);
867 }
868
869 static void a5psw_pcs_free(struct a5psw *a5psw)
870 {
871         int i;
872
873         for (i = 0; i < ARRAY_SIZE(a5psw->pcs); i++) {
874                 if (a5psw->pcs[i])
875                         miic_destroy(a5psw->pcs[i]);
876         }
877 }
878
879 static int a5psw_pcs_get(struct a5psw *a5psw)
880 {
881         struct device_node *ports, *port, *pcs_node;
882         struct phylink_pcs *pcs;
883         int ret;
884         u32 reg;
885
886         ports = of_get_child_by_name(a5psw->dev->of_node, "ethernet-ports");
887         if (!ports)
888                 return -EINVAL;
889
890         for_each_available_child_of_node(ports, port) {
891                 pcs_node = of_parse_phandle(port, "pcs-handle", 0);
892                 if (!pcs_node)
893                         continue;
894
895                 if (of_property_read_u32(port, "reg", &reg)) {
896                         ret = -EINVAL;
897                         goto free_pcs;
898                 }
899
900                 if (reg >= ARRAY_SIZE(a5psw->pcs)) {
901                         ret = -ENODEV;
902                         goto free_pcs;
903                 }
904
905                 pcs = miic_create(a5psw->dev, pcs_node);
906                 if (IS_ERR(pcs)) {
907                         dev_err(a5psw->dev, "Failed to create PCS for port %d\n",
908                                 reg);
909                         ret = PTR_ERR(pcs);
910                         goto free_pcs;
911                 }
912
913                 a5psw->pcs[reg] = pcs;
914                 of_node_put(pcs_node);
915         }
916         of_node_put(ports);
917
918         return 0;
919
920 free_pcs:
921         of_node_put(pcs_node);
922         of_node_put(port);
923         of_node_put(ports);
924         a5psw_pcs_free(a5psw);
925
926         return ret;
927 }
928
929 static int a5psw_probe(struct platform_device *pdev)
930 {
931         struct device *dev = &pdev->dev;
932         struct device_node *mdio;
933         struct dsa_switch *ds;
934         struct a5psw *a5psw;
935         int ret;
936
937         a5psw = devm_kzalloc(dev, sizeof(*a5psw), GFP_KERNEL);
938         if (!a5psw)
939                 return -ENOMEM;
940
941         a5psw->dev = dev;
942         mutex_init(&a5psw->lk_lock);
943         spin_lock_init(&a5psw->reg_lock);
944         a5psw->base = devm_platform_ioremap_resource(pdev, 0);
945         if (IS_ERR(a5psw->base))
946                 return PTR_ERR(a5psw->base);
947
948         ret = a5psw_pcs_get(a5psw);
949         if (ret)
950                 return ret;
951
952         a5psw->hclk = devm_clk_get(dev, "hclk");
953         if (IS_ERR(a5psw->hclk)) {
954                 dev_err(dev, "failed get hclk clock\n");
955                 ret = PTR_ERR(a5psw->hclk);
956                 goto free_pcs;
957         }
958
959         a5psw->clk = devm_clk_get(dev, "clk");
960         if (IS_ERR(a5psw->clk)) {
961                 dev_err(dev, "failed get clk_switch clock\n");
962                 ret = PTR_ERR(a5psw->clk);
963                 goto free_pcs;
964         }
965
966         ret = clk_prepare_enable(a5psw->clk);
967         if (ret)
968                 goto free_pcs;
969
970         ret = clk_prepare_enable(a5psw->hclk);
971         if (ret)
972                 goto clk_disable;
973
974         mdio = of_get_child_by_name(dev->of_node, "mdio");
975         if (of_device_is_available(mdio)) {
976                 ret = a5psw_probe_mdio(a5psw, mdio);
977                 if (ret) {
978                         of_node_put(mdio);
979                         dev_err(dev, "Failed to register MDIO: %d\n", ret);
980                         goto hclk_disable;
981                 }
982         }
983
984         of_node_put(mdio);
985
986         ds = &a5psw->ds;
987         ds->dev = dev;
988         ds->num_ports = A5PSW_PORTS_NUM;
989         ds->ops = &a5psw_switch_ops;
990         ds->priv = a5psw;
991
992         ret = dsa_register_switch(ds);
993         if (ret) {
994                 dev_err(dev, "Failed to register DSA switch: %d\n", ret);
995                 goto hclk_disable;
996         }
997
998         return 0;
999
1000 hclk_disable:
1001         clk_disable_unprepare(a5psw->hclk);
1002 clk_disable:
1003         clk_disable_unprepare(a5psw->clk);
1004 free_pcs:
1005         a5psw_pcs_free(a5psw);
1006
1007         return ret;
1008 }
1009
1010 static int a5psw_remove(struct platform_device *pdev)
1011 {
1012         struct a5psw *a5psw = platform_get_drvdata(pdev);
1013
1014         if (!a5psw)
1015                 return 0;
1016
1017         dsa_unregister_switch(&a5psw->ds);
1018         a5psw_pcs_free(a5psw);
1019         clk_disable_unprepare(a5psw->hclk);
1020         clk_disable_unprepare(a5psw->clk);
1021
1022         return 0;
1023 }
1024
1025 static void a5psw_shutdown(struct platform_device *pdev)
1026 {
1027         struct a5psw *a5psw = platform_get_drvdata(pdev);
1028
1029         if (!a5psw)
1030                 return;
1031
1032         dsa_switch_shutdown(&a5psw->ds);
1033
1034         platform_set_drvdata(pdev, NULL);
1035 }
1036
1037 static const struct of_device_id a5psw_of_mtable[] = {
1038         { .compatible = "renesas,rzn1-a5psw", },
1039         { /* sentinel */ },
1040 };
1041 MODULE_DEVICE_TABLE(of, a5psw_of_mtable);
1042
1043 static struct platform_driver a5psw_driver = {
1044         .driver = {
1045                 .name    = "rzn1_a5psw",
1046                 .of_match_table = of_match_ptr(a5psw_of_mtable),
1047         },
1048         .probe = a5psw_probe,
1049         .remove = a5psw_remove,
1050         .shutdown = a5psw_shutdown,
1051 };
1052 module_platform_driver(a5psw_driver);
1053
1054 MODULE_LICENSE("GPL");
1055 MODULE_DESCRIPTION("Renesas RZ/N1 Advanced 5-port Switch driver");
1056 MODULE_AUTHOR("Clément Léger <clement.leger@bootlin.com>");