Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux

net: dsa: b53: Add SerDes support

Add support for the Northstar Plus SerDes which is accessed through a
special page of the switch. Since this is something that most people
probably will not want to use, make it a configurable option with a
default on ARCH_BCM_NSP where it is the most useful currently.

The SerDes supports both SGMII and 1000baseX modes for both lanes, and
2500baseX for one of the lanes, and is internally looking like a
seemingly standard MII PHY, except for the few bits that got repurposed.

Signed-off-by: Florian Fainelli <f.fainelli@gmail.com>
Reviewed-by: Andrew Lunn <andrew@lunn.ch>
Signed-off-by: David S. Miller <davem@davemloft.net>

authored by

Florian Fainelli and committed by
David S. Miller
0e01491d a8e8b985

+490
+7
drivers/net/dsa/b53/Kconfig
··· 35 35 help 36 36 Select to enable support for memory-mapped Switch Register Access 37 37 Bridge Registers (SRAB) like it is found on the BCM53010 38 + 39 + config B53_SERDES 40 + tristate "B53 SerDes support" 41 + depends on B53 42 + default ARCH_BCM_NSP 43 + help 44 + Select to enable support for SerDes on e.g: Northstar Plus SoCs.
+1
drivers/net/dsa/b53/Makefile
··· 5 5 obj-$(CONFIG_B53_MDIO_DRIVER) += b53_mdio.o 6 6 obj-$(CONFIG_B53_MMAP_DRIVER) += b53_mmap.o 7 7 obj-$(CONFIG_B53_SRAB_DRIVER) += b53_srab.o 8 + obj-$(CONFIG_B53_SERDES) += b53_serdes.o
+26
drivers/net/dsa/b53/b53_common.c
··· 765 765 memset(priv->vlans, 0, sizeof(*priv->vlans) * priv->num_vlans); 766 766 memset(priv->ports, 0, sizeof(*priv->ports) * priv->num_ports); 767 767 768 + priv->serdes_lane = B53_INVALID_LANE; 769 + 768 770 return b53_switch_reset(priv); 769 771 } 770 772 ··· 1130 1128 struct b53_device *dev = ds->priv; 1131 1129 __ETHTOOL_DECLARE_LINK_MODE_MASK(mask) = { 0, }; 1132 1130 1131 + if (dev->ops->serdes_phylink_validate) 1132 + dev->ops->serdes_phylink_validate(dev, port, mask, state); 1133 + 1133 1134 /* Allow all the expected bits */ 1134 1135 phylink_set(mask, Autoneg); 1135 1136 phylink_set_port_modes(mask); ··· 1169 1164 int b53_phylink_mac_link_state(struct dsa_switch *ds, int port, 1170 1165 struct phylink_link_state *state) 1171 1166 { 1167 + struct b53_device *dev = ds->priv; 1172 1168 int ret = -EOPNOTSUPP; 1169 + 1170 + if (phy_interface_mode_is_8023z(state->interface) && 1171 + dev->ops->serdes_link_state) 1172 + ret = dev->ops->serdes_link_state(dev, port, state); 1173 1173 1174 1174 return ret; 1175 1175 } ··· 1194 1184 state->duplex, state->pause); 1195 1185 return; 1196 1186 } 1187 + 1188 + if (phy_interface_mode_is_8023z(state->interface) && 1189 + dev->ops->serdes_config) 1190 + dev->ops->serdes_config(dev, port, mode, state); 1197 1191 } 1198 1192 EXPORT_SYMBOL(b53_phylink_mac_config); 1199 1193 1200 1194 void b53_phylink_mac_an_restart(struct dsa_switch *ds, int port) 1201 1195 { 1196 + struct b53_device *dev = ds->priv; 1197 + 1198 + if (dev->ops->serdes_an_restart) 1199 + dev->ops->serdes_an_restart(dev, port); 1202 1200 } 1203 1201 EXPORT_SYMBOL(b53_phylink_mac_an_restart); 1204 1202 ··· 1223 1205 b53_force_link(dev, port, false); 1224 1206 return; 1225 1207 } 1208 + 1209 + if (phy_interface_mode_is_8023z(interface) && 1210 + dev->ops->serdes_link_set) 1211 + dev->ops->serdes_link_set(dev, port, mode, interface, false); 1226 1212 } 1227 1213 EXPORT_SYMBOL(b53_phylink_mac_link_down); 1228 1214 ··· 1244 1222 b53_force_link(dev, port, true); 1245 1223 return; 1246 1224 } 1225 + 1226 + if (phy_interface_mode_is_8023z(interface) && 1227 + dev->ops->serdes_link_set) 1228 + dev->ops->serdes_link_set(dev, port, mode, interface, true); 1247 1229 } 1248 1230 EXPORT_SYMBOL(b53_phylink_mac_link_up); 1249 1231
+17
drivers/net/dsa/b53/b53_priv.h
··· 29 29 30 30 struct b53_device; 31 31 struct net_device; 32 + struct phylink_link_state; 32 33 33 34 struct b53_io_ops { 34 35 int (*read8)(struct b53_device *dev, u8 page, u8 reg, u8 *value); ··· 46 45 int (*phy_write16)(struct b53_device *dev, int addr, int reg, u16 value); 47 46 int (*irq_enable)(struct b53_device *dev, int port); 48 47 void (*irq_disable)(struct b53_device *dev, int port); 48 + u8 (*serdes_map_lane)(struct b53_device *dev, int port); 49 + int (*serdes_link_state)(struct b53_device *dev, int port, 50 + struct phylink_link_state *state); 51 + void (*serdes_config)(struct b53_device *dev, int port, 52 + unsigned int mode, 53 + const struct phylink_link_state *state); 54 + void (*serdes_an_restart)(struct b53_device *dev, int port); 55 + void (*serdes_link_set)(struct b53_device *dev, int port, 56 + unsigned int mode, phy_interface_t interface, 57 + bool link_up); 58 + void (*serdes_phylink_validate)(struct b53_device *dev, int port, 59 + unsigned long *supported, 60 + struct phylink_link_state *state); 49 61 }; 62 + 63 + #define B53_INVALID_LANE 0xff 50 64 51 65 enum { 52 66 BCM5325_DEVICE_ID = 0x25, ··· 125 109 /* connect specific data */ 126 110 u8 current_page; 127 111 struct device *dev; 112 + u8 serdes_lane; 128 113 129 114 /* Master MDIO bus we got probed from */ 130 115 struct mii_bus *bus;
+217
drivers/net/dsa/b53/b53_serdes.c
··· 1 + // SPDX-License-Identifier: GPL-2.0 or BSD-3-Clause 2 + /* 3 + * Northstar Plus switch SerDes/SGMII PHY main logic 4 + * 5 + * Copyright (C) 2018 Florian Fainelli <f.fainelli@gmail.com> 6 + */ 7 + 8 + #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt 9 + 10 + #include <linux/delay.h> 11 + #include <linux/kernel.h> 12 + #include <linux/phy.h> 13 + #include <linux/phylink.h> 14 + #include <net/dsa.h> 15 + 16 + #include "b53_priv.h" 17 + #include "b53_serdes.h" 18 + #include "b53_regs.h" 19 + 20 + static void b53_serdes_write_blk(struct b53_device *dev, u8 offset, u16 block, 21 + u16 value) 22 + { 23 + b53_write16(dev, B53_SERDES_PAGE, B53_SERDES_BLKADDR, block); 24 + b53_write16(dev, B53_SERDES_PAGE, offset, value); 25 + } 26 + 27 + static u16 b53_serdes_read_blk(struct b53_device *dev, u8 offset, u16 block) 28 + { 29 + u16 value; 30 + 31 + b53_write16(dev, B53_SERDES_PAGE, B53_SERDES_BLKADDR, block); 32 + b53_read16(dev, B53_SERDES_PAGE, offset, &value); 33 + 34 + return value; 35 + } 36 + 37 + static void b53_serdes_set_lane(struct b53_device *dev, u8 lane) 38 + { 39 + if (dev->serdes_lane == lane) 40 + return; 41 + 42 + WARN_ON(lane > 1); 43 + 44 + b53_serdes_write_blk(dev, B53_SERDES_LANE, 45 + SERDES_XGXSBLK0_BLOCKADDRESS, lane); 46 + dev->serdes_lane = lane; 47 + } 48 + 49 + static void b53_serdes_write(struct b53_device *dev, u8 lane, 50 + u8 offset, u16 block, u16 value) 51 + { 52 + b53_serdes_set_lane(dev, lane); 53 + b53_serdes_write_blk(dev, offset, block, value); 54 + } 55 + 56 + static u16 b53_serdes_read(struct b53_device *dev, u8 lane, 57 + u8 offset, u16 block) 58 + { 59 + b53_serdes_set_lane(dev, lane); 60 + return b53_serdes_read_blk(dev, offset, block); 61 + } 62 + 63 + void b53_serdes_config(struct b53_device *dev, int port, unsigned int mode, 64 + const struct phylink_link_state *state) 65 + { 66 + u8 lane = b53_serdes_map_lane(dev, port); 67 + u16 reg; 68 + 69 + if (lane == B53_INVALID_LANE) 70 + return; 71 + 72 + reg = b53_serdes_read(dev, lane, B53_SERDES_DIGITAL_CONTROL(1), 73 + SERDES_DIGITAL_BLK); 74 + if (state->interface == PHY_INTERFACE_MODE_1000BASEX) 75 + reg |= FIBER_MODE_1000X; 76 + else 77 + reg &= ~FIBER_MODE_1000X; 78 + b53_serdes_write(dev, lane, B53_SERDES_DIGITAL_CONTROL(1), 79 + SERDES_DIGITAL_BLK, reg); 80 + } 81 + EXPORT_SYMBOL(b53_serdes_config); 82 + 83 + void b53_serdes_an_restart(struct b53_device *dev, int port) 84 + { 85 + u8 lane = b53_serdes_map_lane(dev, port); 86 + u16 reg; 87 + 88 + if (lane == B53_INVALID_LANE) 89 + return; 90 + 91 + reg = b53_serdes_read(dev, lane, B53_SERDES_MII_REG(MII_BMCR), 92 + SERDES_MII_BLK); 93 + reg |= BMCR_ANRESTART; 94 + b53_serdes_write(dev, lane, B53_SERDES_MII_REG(MII_BMCR), 95 + SERDES_MII_BLK, reg); 96 + } 97 + EXPORT_SYMBOL(b53_serdes_an_restart); 98 + 99 + int b53_serdes_link_state(struct b53_device *dev, int port, 100 + struct phylink_link_state *state) 101 + { 102 + u8 lane = b53_serdes_map_lane(dev, port); 103 + u16 dig, bmcr, bmsr; 104 + 105 + if (lane == B53_INVALID_LANE) 106 + return 1; 107 + 108 + dig = b53_serdes_read(dev, lane, B53_SERDES_DIGITAL_STATUS, 109 + SERDES_DIGITAL_BLK); 110 + bmcr = b53_serdes_read(dev, lane, B53_SERDES_MII_REG(MII_BMCR), 111 + SERDES_MII_BLK); 112 + bmsr = b53_serdes_read(dev, lane, B53_SERDES_MII_REG(MII_BMSR), 113 + SERDES_MII_BLK); 114 + 115 + switch ((dig >> SPEED_STATUS_SHIFT) & SPEED_STATUS_MASK) { 116 + case SPEED_STATUS_10: 117 + state->speed = SPEED_10; 118 + break; 119 + case SPEED_STATUS_100: 120 + state->speed = SPEED_100; 121 + break; 122 + case SPEED_STATUS_1000: 123 + state->speed = SPEED_1000; 124 + break; 125 + default: 126 + case SPEED_STATUS_2500: 127 + state->speed = SPEED_2500; 128 + break; 129 + } 130 + 131 + state->duplex = dig & DUPLEX_STATUS ? DUPLEX_FULL : DUPLEX_HALF; 132 + state->an_enabled = !!(bmcr & BMCR_ANENABLE); 133 + state->an_complete = !!(bmsr & BMSR_ANEGCOMPLETE); 134 + state->link = !!(dig & LINK_STATUS); 135 + if (dig & PAUSE_RESOLUTION_RX_SIDE) 136 + state->pause |= MLO_PAUSE_RX; 137 + if (dig & PAUSE_RESOLUTION_TX_SIDE) 138 + state->pause |= MLO_PAUSE_TX; 139 + 140 + return 0; 141 + } 142 + EXPORT_SYMBOL(b53_serdes_link_state); 143 + 144 + void b53_serdes_link_set(struct b53_device *dev, int port, unsigned int mode, 145 + phy_interface_t interface, bool link_up) 146 + { 147 + u8 lane = b53_serdes_map_lane(dev, port); 148 + u16 reg; 149 + 150 + if (lane == B53_INVALID_LANE) 151 + return; 152 + 153 + reg = b53_serdes_read(dev, lane, B53_SERDES_MII_REG(MII_BMCR), 154 + SERDES_MII_BLK); 155 + if (link_up) 156 + reg &= ~BMCR_PDOWN; 157 + else 158 + reg |= BMCR_PDOWN; 159 + b53_serdes_write(dev, lane, B53_SERDES_MII_REG(MII_BMCR), 160 + SERDES_MII_BLK, reg); 161 + } 162 + EXPORT_SYMBOL(b53_serdes_link_set); 163 + 164 + void b53_serdes_phylink_validate(struct b53_device *dev, int port, 165 + unsigned long *supported, 166 + struct phylink_link_state *state) 167 + { 168 + u8 lane = b53_serdes_map_lane(dev, port); 169 + 170 + if (lane == B53_INVALID_LANE) 171 + return; 172 + 173 + switch (lane) { 174 + case 0: 175 + phylink_set(supported, 2500baseX_Full); 176 + /* fallthrough */ 177 + case 1: 178 + phylink_set(supported, 1000baseX_Full); 179 + break; 180 + default: 181 + break; 182 + } 183 + } 184 + EXPORT_SYMBOL(b53_serdes_phylink_validate); 185 + 186 + int b53_serdes_init(struct b53_device *dev, int port) 187 + { 188 + u8 lane = b53_serdes_map_lane(dev, port); 189 + u16 id0, msb, lsb; 190 + 191 + if (lane == B53_INVALID_LANE) 192 + return -EINVAL; 193 + 194 + id0 = b53_serdes_read(dev, lane, B53_SERDES_ID0, SERDES_ID0); 195 + msb = b53_serdes_read(dev, lane, B53_SERDES_MII_REG(MII_PHYSID1), 196 + SERDES_MII_BLK); 197 + lsb = b53_serdes_read(dev, lane, B53_SERDES_MII_REG(MII_PHYSID2), 198 + SERDES_MII_BLK); 199 + if (id0 == 0 || id0 == 0xffff) { 200 + dev_err(dev->dev, "SerDes not initialized, check settings\n"); 201 + return -ENODEV; 202 + } 203 + 204 + dev_info(dev->dev, 205 + "SerDes lane %d, model: %d, rev %c%d (OUI: 0x%08x)\n", 206 + lane, id0 & SERDES_ID0_MODEL_MASK, 207 + (id0 >> SERDES_ID0_REV_LETTER_SHIFT) + 0x41, 208 + (id0 >> SERDES_ID0_REV_NUM_SHIFT) & SERDES_ID0_REV_NUM_MASK, 209 + (u32)msb << 16 | lsb); 210 + 211 + return 0; 212 + } 213 + EXPORT_SYMBOL(b53_serdes_init); 214 + 215 + MODULE_AUTHOR("Florian Fainelli <f.fainelli@gmail.com>"); 216 + MODULE_DESCRIPTION("B53 Switch SerDes driver"); 217 + MODULE_LICENSE("Dual BSD/GPL");
+121
drivers/net/dsa/b53/b53_serdes.h
··· 1 + /* SPDX-License-Identifier: GPL-2.0 or BSD-3-Clause 2 + * 3 + * Northstar Plus switch SerDes/SGMII PHY definitions 4 + * 5 + * Copyright (C) 2018 Florian Fainelli <f.fainelli@gmail.com> 6 + */ 7 + 8 + #include <linux/phy.h> 9 + #include <linux/types.h> 10 + 11 + /* Non-standard page used to access SerDes PHY registers on NorthStar Plus */ 12 + #define B53_SERDES_PAGE 0x16 13 + #define B53_SERDES_BLKADDR 0x3e 14 + #define B53_SERDES_LANE 0x3c 15 + 16 + #define B53_SERDES_ID0 0x20 17 + #define SERDES_ID0_MODEL_MASK 0x3f 18 + #define SERDES_ID0_REV_NUM_SHIFT 11 19 + #define SERDES_ID0_REV_NUM_MASK 0x7 20 + #define SERDES_ID0_REV_LETTER_SHIFT 14 21 + 22 + #define B53_SERDES_MII_REG(x) (0x20 + (x) * 2) 23 + #define B53_SERDES_DIGITAL_CONTROL(x) (0x18 + (x) * 2) 24 + #define B53_SERDES_DIGITAL_STATUS 0x28 25 + 26 + /* SERDES_DIGITAL_CONTROL1 */ 27 + #define FIBER_MODE_1000X BIT(0) 28 + #define TBI_INTERFACE BIT(1) 29 + #define SIGNAL_DETECT_EN BIT(2) 30 + #define INVERT_SIGNAL_DETECT BIT(3) 31 + #define AUTODET_EN BIT(4) 32 + #define SGMII_MASTER_MODE BIT(5) 33 + #define DISABLE_DLL_PWRDOWN BIT(6) 34 + #define CRC_CHECKER_DIS BIT(7) 35 + #define COMMA_DET_EN BIT(8) 36 + #define ZERO_COMMA_DET_EN BIT(9) 37 + #define REMOTE_LOOPBACK BIT(10) 38 + #define SEL_RX_PKTS_FOR_CNTR BIT(11) 39 + #define MASTER_MDIO_PHY_SEL BIT(13) 40 + #define DISABLE_SIGNAL_DETECT_FLT BIT(14) 41 + 42 + /* SERDES_DIGITAL_CONTROL2 */ 43 + #define EN_PARALLEL_DET BIT(0) 44 + #define DIS_FALSE_LINK BIT(1) 45 + #define FLT_FORCE_LINK BIT(2) 46 + #define EN_AUTONEG_ERR_TIMER BIT(3) 47 + #define DIS_REMOTE_FAULT_SENSING BIT(4) 48 + #define FORCE_XMIT_DATA BIT(5) 49 + #define AUTONEG_FAST_TIMERS BIT(6) 50 + #define DIS_CARRIER_EXTEND BIT(7) 51 + #define DIS_TRRR_GENERATION BIT(8) 52 + #define BYPASS_PCS_RX BIT(9) 53 + #define BYPASS_PCS_TX BIT(10) 54 + #define TEST_CNTR_EN BIT(11) 55 + #define TX_PACKET_SEQ_TEST BIT(12) 56 + #define TX_IDLE_JAM_SEQ_TEST BIT(13) 57 + #define CLR_BER_CNTR BIT(14) 58 + 59 + /* SERDES_DIGITAL_CONTROL3 */ 60 + #define TX_FIFO_RST BIT(0) 61 + #define FIFO_ELAST_TX_RX_SHIFT 1 62 + #define FIFO_ELAST_TX_RX_5K 0 63 + #define FIFO_ELAST_TX_RX_10K 1 64 + #define FIFO_ELAST_TX_RX_13_5K 2 65 + #define FIFO_ELAST_TX_RX_18_5K 3 66 + #define BLOCK_TXEN_MODE BIT(9) 67 + #define JAM_FALSE_CARRIER_MODE BIT(10) 68 + #define EXT_PHY_CRS_MODE BIT(11) 69 + #define INVERT_EXT_PHY_CRS BIT(12) 70 + #define DISABLE_TX_CRS BIT(13) 71 + 72 + /* SERDES_DIGITAL_STATUS */ 73 + #define SGMII_MODE BIT(0) 74 + #define LINK_STATUS BIT(1) 75 + #define DUPLEX_STATUS BIT(2) 76 + #define SPEED_STATUS_SHIFT 3 77 + #define SPEED_STATUS_10 0 78 + #define SPEED_STATUS_100 1 79 + #define SPEED_STATUS_1000 2 80 + #define SPEED_STATUS_2500 3 81 + #define SPEED_STATUS_MASK SPEED_STATUS_2500 82 + #define PAUSE_RESOLUTION_TX_SIDE BIT(5) 83 + #define PAUSE_RESOLUTION_RX_SIDE BIT(6) 84 + #define LINK_STATUS_CHANGE BIT(7) 85 + #define EARLY_END_EXT_DET BIT(8) 86 + #define CARRIER_EXT_ERR_DET BIT(9) 87 + #define RX_ERR_DET BIT(10) 88 + #define TX_ERR_DET BIT(11) 89 + #define CRC_ERR_DET BIT(12) 90 + #define FALSE_CARRIER_ERR_DET BIT(13) 91 + #define RXFIFO_ERR_DET BIT(14) 92 + #define TXFIFO_ERR_DET BIT(15) 93 + 94 + /* Block offsets */ 95 + #define SERDES_DIGITAL_BLK 0x8300 96 + #define SERDES_ID0 0x8310 97 + #define SERDES_MII_BLK 0xffe0 98 + #define SERDES_XGXSBLK0_BLOCKADDRESS 0xffd0 99 + 100 + struct phylink_link_state; 101 + 102 + static inline u8 b53_serdes_map_lane(struct b53_device *dev, int port) 103 + { 104 + if (!dev->ops->serdes_map_lane) 105 + return B53_INVALID_LANE; 106 + 107 + return dev->ops->serdes_map_lane(dev, port); 108 + } 109 + 110 + int b53_serdes_get_link(struct b53_device *dev, int port); 111 + int b53_serdes_link_state(struct b53_device *dev, int port, 112 + struct phylink_link_state *state); 113 + void b53_serdes_config(struct b53_device *dev, int port, unsigned int mode, 114 + const struct phylink_link_state *state); 115 + void b53_serdes_an_restart(struct b53_device *dev, int port); 116 + void b53_serdes_link_set(struct b53_device *dev, int port, unsigned int mode, 117 + phy_interface_t interface, bool link_up); 118 + void b53_serdes_phylink_validate(struct b53_device *dev, int port, 119 + unsigned long *supported, 120 + struct phylink_link_state *state); 121 + int b53_serdes_init(struct b53_device *dev, int port);
+101
drivers/net/dsa/b53/b53_srab.c
··· 25 25 #include <linux/of.h> 26 26 27 27 #include "b53_priv.h" 28 + #include "b53_serdes.h" 28 29 29 30 /* command and status register of the SRAB */ 30 31 #define B53_SRAB_CMDSTAT 0x2c ··· 63 62 #define B53_SRAB_P7_SLEEP_TIMER BIT(11) 64 63 #define B53_SRAB_IMP0_SLEEP_TIMER BIT(12) 65 64 65 + /* Port mux configuration registers */ 66 + #define B53_MUX_CONFIG_P5 0x00 67 + #define MUX_CONFIG_SGMII 0 68 + #define MUX_CONFIG_MII_LITE 1 69 + #define MUX_CONFIG_RGMII 2 70 + #define MUX_CONFIG_GMII 3 71 + #define MUX_CONFIG_GPHY 4 72 + #define MUX_CONFIG_INTERNAL 5 73 + #define MUX_CONFIG_MASK 0x7 74 + #define B53_MUX_CONFIG_P4 0x04 75 + 66 76 struct b53_srab_port_priv { 67 77 int irq; 68 78 bool irq_enabled; 69 79 struct b53_device *dev; 70 80 unsigned int num; 81 + phy_interface_t mode; 71 82 }; 72 83 73 84 struct b53_srab_priv { 74 85 void __iomem *regs; 86 + void __iomem *mux_config; 75 87 struct b53_srab_port_priv port_intrs[B53_N_PORTS]; 76 88 }; 77 89 ··· 370 356 371 357 static irqreturn_t b53_srab_port_thread(int irq, void *dev_id) 372 358 { 359 + struct b53_srab_port_priv *port = dev_id; 360 + struct b53_device *dev = port->dev; 361 + 362 + b53_port_event(dev->ds, port->num); 363 + 373 364 return IRQ_HANDLED; 374 365 } 375 366 ··· 388 369 writel(BIT(port->num), priv->regs + B53_SRAB_INTR); 389 370 390 371 return IRQ_WAKE_THREAD; 372 + } 373 + 374 + static u8 b53_srab_serdes_map_lane(struct b53_device *dev, int port) 375 + { 376 + struct b53_srab_priv *priv = dev->priv; 377 + struct b53_srab_port_priv *p = &priv->port_intrs[port]; 378 + 379 + if (p->mode != PHY_INTERFACE_MODE_SGMII) 380 + return B53_INVALID_LANE; 381 + 382 + switch (port) { 383 + case 5: 384 + return 0; 385 + case 4: 386 + return 1; 387 + default: 388 + return B53_INVALID_LANE; 389 + } 391 390 } 392 391 393 392 static int b53_srab_irq_enable(struct b53_device *dev, int port) ··· 447 410 .write64 = b53_srab_write64, 448 411 .irq_enable = b53_srab_irq_enable, 449 412 .irq_disable = b53_srab_irq_disable, 413 + #if IS_ENABLED(CONFIG_B53_SERDES) 414 + .serdes_map_lane = b53_srab_serdes_map_lane, 415 + .serdes_link_state = b53_serdes_link_state, 416 + .serdes_config = b53_serdes_config, 417 + .serdes_an_restart = b53_serdes_an_restart, 418 + .serdes_link_set = b53_serdes_link_set, 419 + .serdes_phylink_validate = b53_serdes_phylink_validate, 420 + #endif 450 421 }; 451 422 452 423 static const struct of_device_id b53_srab_of_match[] = { ··· 525 480 b53_srab_intr_set(priv, true); 526 481 } 527 482 483 + static void b53_srab_mux_init(struct platform_device *pdev) 484 + { 485 + struct b53_device *dev = platform_get_drvdata(pdev); 486 + struct b53_srab_priv *priv = dev->priv; 487 + struct b53_srab_port_priv *p; 488 + struct resource *r; 489 + unsigned int port; 490 + u32 reg, off = 0; 491 + int ret; 492 + 493 + if (dev->pdata && dev->pdata->chip_id != BCM58XX_DEVICE_ID) 494 + return; 495 + 496 + r = platform_get_resource(pdev, IORESOURCE_MEM, 1); 497 + priv->mux_config = devm_ioremap_resource(&pdev->dev, r); 498 + if (IS_ERR(priv->mux_config)) 499 + return; 500 + 501 + /* Obtain the port mux configuration so we know which lanes 502 + * actually map to SerDes lanes 503 + */ 504 + for (port = 5; port > 3; port--, off += 4) { 505 + p = &priv->port_intrs[port]; 506 + 507 + reg = readl(priv->mux_config + B53_MUX_CONFIG_P5 + off); 508 + switch (reg & MUX_CONFIG_MASK) { 509 + case MUX_CONFIG_SGMII: 510 + p->mode = PHY_INTERFACE_MODE_SGMII; 511 + ret = b53_serdes_init(dev, port); 512 + if (ret) 513 + continue; 514 + break; 515 + case MUX_CONFIG_MII_LITE: 516 + p->mode = PHY_INTERFACE_MODE_MII; 517 + break; 518 + case MUX_CONFIG_GMII: 519 + p->mode = PHY_INTERFACE_MODE_GMII; 520 + break; 521 + case MUX_CONFIG_RGMII: 522 + p->mode = PHY_INTERFACE_MODE_RGMII; 523 + break; 524 + case MUX_CONFIG_INTERNAL: 525 + p->mode = PHY_INTERFACE_MODE_INTERNAL; 526 + break; 527 + default: 528 + p->mode = PHY_INTERFACE_MODE_NA; 529 + break; 530 + } 531 + 532 + if (p->mode != PHY_INTERFACE_MODE_NA) 533 + dev_info(&pdev->dev, "Port %d mode: %s\n", 534 + port, phy_modes(p->mode)); 535 + } 536 + } 537 + 528 538 static int b53_srab_probe(struct platform_device *pdev) 529 539 { 530 540 struct b53_platform_data *pdata = pdev->dev.platform_data; ··· 619 519 platform_set_drvdata(pdev, dev); 620 520 621 521 b53_srab_prepare_irq(pdev); 522 + b53_srab_mux_init(pdev); 622 523 623 524 return b53_switch_register(dev); 624 525 }