aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/net/sfc/mdio_10g.c
diff options
context:
space:
mode:
authorGlenn Elliott <gelliott@cs.unc.edu>2012-03-04 19:47:13 -0500
committerGlenn Elliott <gelliott@cs.unc.edu>2012-03-04 19:47:13 -0500
commitc71c03bda1e86c9d5198c5d83f712e695c4f2a1e (patch)
treeecb166cb3e2b7e2adb3b5e292245fefd23381ac8 /drivers/net/sfc/mdio_10g.c
parentea53c912f8a86a8567697115b6a0d8152beee5c8 (diff)
parent6a00f206debf8a5c8899055726ad127dbeeed098 (diff)
Merge branch 'mpi-master' into wip-k-fmlpwip-k-fmlp
Conflicts: litmus/sched_cedf.c
Diffstat (limited to 'drivers/net/sfc/mdio_10g.c')
-rw-r--r--drivers/net/sfc/mdio_10g.c71
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
20unsigned efx_mdio_id_oui(u32 id) 19unsigned 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
55static int efx_mdio_check_mmd(struct efx_nic *efx, int mmd, int fault_fatal) 54static 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
134int efx_mdio_check_mmds(struct efx_nic *efx, 116int 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 */
256int efx_mdio_set_settings(struct efx_nic *efx, struct ethtool_cmd *ecmd) 233int 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 */
287void efx_mdio_an_reconfigure(struct efx_nic *efx) 264void 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
332enum efx_fc_type efx_mdio_get_pause(struct efx_nic *efx) 287u8 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);