diff options
Diffstat (limited to 'drivers/net/sfc/mdio_10g.c')
-rw-r--r-- | drivers/net/sfc/mdio_10g.c | 71 |
1 files changed, 13 insertions, 58 deletions
diff --git a/drivers/net/sfc/mdio_10g.c b/drivers/net/sfc/mdio_10g.c index eeaf0bd64bd3..7ab385c8136d 100644 --- a/drivers/net/sfc/mdio_10g.c +++ b/drivers/net/sfc/mdio_10g.c | |||
@@ -1,6 +1,6 @@ | |||
1 | /**************************************************************************** | 1 | /**************************************************************************** |
2 | * Driver for Solarflare Solarstorm network controllers and boards | 2 | * Driver for Solarflare Solarstorm network controllers and boards |
3 | * Copyright 2006-2009 Solarflare Communications Inc. | 3 | * Copyright 2006-2011 Solarflare Communications Inc. |
4 | * | 4 | * |
5 | * This program is free software; you can redistribute it and/or modify it | 5 | * This program is free software; you can redistribute it and/or modify it |
6 | * under the terms of the GNU General Public License version 2 as published | 6 | * under the terms of the GNU General Public License version 2 as published |
@@ -15,7 +15,6 @@ | |||
15 | #include "net_driver.h" | 15 | #include "net_driver.h" |
16 | #include "mdio_10g.h" | 16 | #include "mdio_10g.h" |
17 | #include "workarounds.h" | 17 | #include "workarounds.h" |
18 | #include "nic.h" | ||
19 | 18 | ||
20 | unsigned efx_mdio_id_oui(u32 id) | 19 | unsigned efx_mdio_id_oui(u32 id) |
21 | { | 20 | { |
@@ -52,13 +51,10 @@ int efx_mdio_reset_mmd(struct efx_nic *port, int mmd, | |||
52 | return spins ? spins : -ETIMEDOUT; | 51 | return spins ? spins : -ETIMEDOUT; |
53 | } | 52 | } |
54 | 53 | ||
55 | static int efx_mdio_check_mmd(struct efx_nic *efx, int mmd, int fault_fatal) | 54 | static int efx_mdio_check_mmd(struct efx_nic *efx, int mmd) |
56 | { | 55 | { |
57 | int status; | 56 | int status; |
58 | 57 | ||
59 | if (LOOPBACK_INTERNAL(efx)) | ||
60 | return 0; | ||
61 | |||
62 | if (mmd != MDIO_MMD_AN) { | 58 | if (mmd != MDIO_MMD_AN) { |
63 | /* Read MMD STATUS2 to check it is responding. */ | 59 | /* Read MMD STATUS2 to check it is responding. */ |
64 | status = efx_mdio_read(efx, mmd, MDIO_STAT2); | 60 | status = efx_mdio_read(efx, mmd, MDIO_STAT2); |
@@ -69,20 +65,6 @@ static int efx_mdio_check_mmd(struct efx_nic *efx, int mmd, int fault_fatal) | |||
69 | } | 65 | } |
70 | } | 66 | } |
71 | 67 | ||
72 | /* Read MMD STATUS 1 to check for fault. */ | ||
73 | status = efx_mdio_read(efx, mmd, MDIO_STAT1); | ||
74 | if (status & MDIO_STAT1_FAULT) { | ||
75 | if (fault_fatal) { | ||
76 | netif_err(efx, hw, efx->net_dev, | ||
77 | "PHY MMD %d reporting fatal" | ||
78 | " fault: status %x\n", mmd, status); | ||
79 | return -EIO; | ||
80 | } else { | ||
81 | netif_dbg(efx, hw, efx->net_dev, | ||
82 | "PHY MMD %d reporting status" | ||
83 | " %x (expected)\n", mmd, status); | ||
84 | } | ||
85 | } | ||
86 | return 0; | 68 | return 0; |
87 | } | 69 | } |
88 | 70 | ||
@@ -131,8 +113,7 @@ int efx_mdio_wait_reset_mmds(struct efx_nic *efx, unsigned int mmd_mask) | |||
131 | return rc; | 113 | return rc; |
132 | } | 114 | } |
133 | 115 | ||
134 | int efx_mdio_check_mmds(struct efx_nic *efx, | 116 | int efx_mdio_check_mmds(struct efx_nic *efx, unsigned int mmd_mask) |
135 | unsigned int mmd_mask, unsigned int fatal_mask) | ||
136 | { | 117 | { |
137 | int mmd = 0, probe_mmd, devs1, devs2; | 118 | int mmd = 0, probe_mmd, devs1, devs2; |
138 | u32 devices; | 119 | u32 devices; |
@@ -162,13 +143,9 @@ int efx_mdio_check_mmds(struct efx_nic *efx, | |||
162 | 143 | ||
163 | /* Check all required MMDs are responding and happy. */ | 144 | /* Check all required MMDs are responding and happy. */ |
164 | while (mmd_mask) { | 145 | while (mmd_mask) { |
165 | if (mmd_mask & 1) { | 146 | if ((mmd_mask & 1) && efx_mdio_check_mmd(efx, mmd)) |
166 | int fault_fatal = fatal_mask & 1; | 147 | return -EIO; |
167 | if (efx_mdio_check_mmd(efx, mmd, fault_fatal)) | ||
168 | return -EIO; | ||
169 | } | ||
170 | mmd_mask = mmd_mask >> 1; | 148 | mmd_mask = mmd_mask >> 1; |
171 | fatal_mask = fatal_mask >> 1; | ||
172 | mmd++; | 149 | mmd++; |
173 | } | 150 | } |
174 | 151 | ||
@@ -255,12 +232,12 @@ void efx_mdio_set_mmds_lpower(struct efx_nic *efx, | |||
255 | */ | 232 | */ |
256 | int efx_mdio_set_settings(struct efx_nic *efx, struct ethtool_cmd *ecmd) | 233 | int efx_mdio_set_settings(struct efx_nic *efx, struct ethtool_cmd *ecmd) |
257 | { | 234 | { |
258 | struct ethtool_cmd prev; | 235 | struct ethtool_cmd prev = { .cmd = ETHTOOL_GSET }; |
259 | 236 | ||
260 | efx->phy_op->get_settings(efx, &prev); | 237 | efx->phy_op->get_settings(efx, &prev); |
261 | 238 | ||
262 | if (ecmd->advertising == prev.advertising && | 239 | if (ecmd->advertising == prev.advertising && |
263 | ecmd->speed == prev.speed && | 240 | ethtool_cmd_speed(ecmd) == ethtool_cmd_speed(&prev) && |
264 | ecmd->duplex == prev.duplex && | 241 | ecmd->duplex == prev.duplex && |
265 | ecmd->port == prev.port && | 242 | ecmd->port == prev.port && |
266 | ecmd->autoneg == prev.autoneg) | 243 | ecmd->autoneg == prev.autoneg) |
@@ -286,50 +263,28 @@ int efx_mdio_set_settings(struct efx_nic *efx, struct ethtool_cmd *ecmd) | |||
286 | */ | 263 | */ |
287 | void efx_mdio_an_reconfigure(struct efx_nic *efx) | 264 | void efx_mdio_an_reconfigure(struct efx_nic *efx) |
288 | { | 265 | { |
289 | bool xnp = (efx->link_advertising & ADVERTISED_10000baseT_Full | ||
290 | || EFX_WORKAROUND_13204(efx)); | ||
291 | int reg; | 266 | int reg; |
292 | 267 | ||
293 | WARN_ON(!(efx->mdio.mmds & MDIO_DEVS_AN)); | 268 | WARN_ON(!(efx->mdio.mmds & MDIO_DEVS_AN)); |
294 | 269 | ||
295 | /* Set up the base page */ | 270 | /* Set up the base page */ |
296 | reg = ADVERTISE_CSMA; | 271 | reg = ADVERTISE_CSMA | ADVERTISE_RESV; |
297 | if (efx->link_advertising & ADVERTISED_10baseT_Half) | ||
298 | reg |= ADVERTISE_10HALF; | ||
299 | if (efx->link_advertising & ADVERTISED_10baseT_Full) | ||
300 | reg |= ADVERTISE_10FULL; | ||
301 | if (efx->link_advertising & ADVERTISED_100baseT_Half) | ||
302 | reg |= ADVERTISE_100HALF; | ||
303 | if (efx->link_advertising & ADVERTISED_100baseT_Full) | ||
304 | reg |= ADVERTISE_100FULL; | ||
305 | if (xnp) | ||
306 | reg |= ADVERTISE_RESV; | ||
307 | else if (efx->link_advertising & (ADVERTISED_1000baseT_Half | | ||
308 | ADVERTISED_1000baseT_Full)) | ||
309 | reg |= ADVERTISE_NPAGE; | ||
310 | if (efx->link_advertising & ADVERTISED_Pause) | 272 | if (efx->link_advertising & ADVERTISED_Pause) |
311 | reg |= ADVERTISE_PAUSE_CAP; | 273 | reg |= ADVERTISE_PAUSE_CAP; |
312 | if (efx->link_advertising & ADVERTISED_Asym_Pause) | 274 | if (efx->link_advertising & ADVERTISED_Asym_Pause) |
313 | reg |= ADVERTISE_PAUSE_ASYM; | 275 | reg |= ADVERTISE_PAUSE_ASYM; |
314 | efx_mdio_write(efx, MDIO_MMD_AN, MDIO_AN_ADVERTISE, reg); | 276 | efx_mdio_write(efx, MDIO_MMD_AN, MDIO_AN_ADVERTISE, reg); |
315 | 277 | ||
316 | /* Set up the (extended) next page if necessary */ | 278 | /* Set up the (extended) next page */ |
317 | if (efx->phy_op->set_npage_adv) | 279 | efx->phy_op->set_npage_adv(efx, efx->link_advertising); |
318 | efx->phy_op->set_npage_adv(efx, efx->link_advertising); | ||
319 | 280 | ||
320 | /* Enable and restart AN */ | 281 | /* Enable and restart AN */ |
321 | reg = efx_mdio_read(efx, MDIO_MMD_AN, MDIO_CTRL1); | 282 | reg = efx_mdio_read(efx, MDIO_MMD_AN, MDIO_CTRL1); |
322 | reg |= MDIO_AN_CTRL1_ENABLE; | 283 | reg |= MDIO_AN_CTRL1_ENABLE | MDIO_AN_CTRL1_RESTART | MDIO_AN_CTRL1_XNP; |
323 | if (!(EFX_WORKAROUND_15195(efx) && LOOPBACK_EXTERNAL(efx))) | ||
324 | reg |= MDIO_AN_CTRL1_RESTART; | ||
325 | if (xnp) | ||
326 | reg |= MDIO_AN_CTRL1_XNP; | ||
327 | else | ||
328 | reg &= ~MDIO_AN_CTRL1_XNP; | ||
329 | efx_mdio_write(efx, MDIO_MMD_AN, MDIO_CTRL1, reg); | 284 | efx_mdio_write(efx, MDIO_MMD_AN, MDIO_CTRL1, reg); |
330 | } | 285 | } |
331 | 286 | ||
332 | enum efx_fc_type efx_mdio_get_pause(struct efx_nic *efx) | 287 | u8 efx_mdio_get_pause(struct efx_nic *efx) |
333 | { | 288 | { |
334 | BUILD_BUG_ON(EFX_FC_AUTO & (EFX_FC_RX | EFX_FC_TX)); | 289 | BUILD_BUG_ON(EFX_FC_AUTO & (EFX_FC_RX | EFX_FC_TX)); |
335 | 290 | ||
@@ -360,7 +315,7 @@ int efx_mdio_test_alive(struct efx_nic *efx) | |||
360 | "no MDIO PHY present with ID %d\n", efx->mdio.prtad); | 315 | "no MDIO PHY present with ID %d\n", efx->mdio.prtad); |
361 | rc = -EINVAL; | 316 | rc = -EINVAL; |
362 | } else { | 317 | } else { |
363 | rc = efx_mdio_check_mmds(efx, efx->mdio.mmds, 0); | 318 | rc = efx_mdio_check_mmds(efx, efx->mdio.mmds); |
364 | } | 319 | } |
365 | 320 | ||
366 | mutex_unlock(&efx->mac_lock); | 321 | mutex_unlock(&efx->mac_lock); |