diff options
author | Yaniv Rosner <yanivr@broadcom.com> | 2011-05-31 17:27:06 -0400 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2011-06-01 16:10:56 -0400 |
commit | c688fe2fc0cab3a5d266f7f6fcb21f14e4ac39ba (patch) | |
tree | 6004092cce67ac60b919c5c4ec324d83c8f3d8ba /drivers/net/bnx2x/bnx2x_link.c | |
parent | fcf5b650832996bd857bb8f0b0b42097218f7fb8 (diff) |
bnx2x: Add TX fault check for fiber PHYs
In case TX fault is detected on Fiber PHYs, declare the link as down until TX fault is gone.
Signed-off-by: Yaniv Rosner <yanivr@broadcom.com>
Signed-off-by: Eilon Greenstein <eilong@broadcom.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'drivers/net/bnx2x/bnx2x_link.c')
-rw-r--r-- | drivers/net/bnx2x/bnx2x_link.c | 72 |
1 files changed, 68 insertions, 4 deletions
diff --git a/drivers/net/bnx2x/bnx2x_link.c b/drivers/net/bnx2x/bnx2x_link.c index 7705e79945ec..87c5d9ff8163 100644 --- a/drivers/net/bnx2x/bnx2x_link.c +++ b/drivers/net/bnx2x/bnx2x_link.c | |||
@@ -3495,6 +3495,7 @@ int bnx2x_link_update(struct link_params *params, struct link_vars *vars) | |||
3495 | phy_vars[phy_index].duplex = DUPLEX_FULL; | 3495 | phy_vars[phy_index].duplex = DUPLEX_FULL; |
3496 | phy_vars[phy_index].phy_link_up = 0; | 3496 | phy_vars[phy_index].phy_link_up = 0; |
3497 | phy_vars[phy_index].link_up = 0; | 3497 | phy_vars[phy_index].link_up = 0; |
3498 | phy_vars[phy_index].fault_detected = 0; | ||
3498 | } | 3499 | } |
3499 | 3500 | ||
3500 | DP(NETIF_MSG_LINK, "port %x, XGXS?%x, int_status 0x%x\n", | 3501 | DP(NETIF_MSG_LINK, "port %x, XGXS?%x, int_status 0x%x\n", |
@@ -3707,7 +3708,8 @@ int bnx2x_link_update(struct link_params *params, struct link_vars *vars) | |||
3707 | */ | 3708 | */ |
3708 | vars->link_up = (vars->phy_link_up && | 3709 | vars->link_up = (vars->phy_link_up && |
3709 | (ext_phy_link_up || | 3710 | (ext_phy_link_up || |
3710 | SINGLE_MEDIA_DIRECT(params))); | 3711 | SINGLE_MEDIA_DIRECT(params)) && |
3712 | (phy_vars[active_external_phy].fault_detected == 0)); | ||
3711 | 3713 | ||
3712 | if (vars->link_up) | 3714 | if (vars->link_up) |
3713 | rc = bnx2x_update_link_up(params, vars, link_10g); | 3715 | rc = bnx2x_update_link_up(params, vars, link_10g); |
@@ -5205,6 +5207,29 @@ void bnx2x_handle_module_detect_int(struct link_params *params) | |||
5205 | } | 5207 | } |
5206 | 5208 | ||
5207 | /******************************************************************/ | 5209 | /******************************************************************/ |
5210 | /* Used by 8706 and 8727 */ | ||
5211 | /******************************************************************/ | ||
5212 | static void bnx2x_sfp_mask_fault(struct bnx2x *bp, | ||
5213 | struct bnx2x_phy *phy, | ||
5214 | u16 alarm_status_offset, | ||
5215 | u16 alarm_ctrl_offset) | ||
5216 | { | ||
5217 | u16 alarm_status, val; | ||
5218 | bnx2x_cl45_read(bp, phy, | ||
5219 | MDIO_PMA_DEVAD, alarm_status_offset, | ||
5220 | &alarm_status); | ||
5221 | bnx2x_cl45_read(bp, phy, | ||
5222 | MDIO_PMA_DEVAD, alarm_status_offset, | ||
5223 | &alarm_status); | ||
5224 | /* Mask or enable the fault event. */ | ||
5225 | bnx2x_cl45_read(bp, phy, MDIO_PMA_DEVAD, alarm_ctrl_offset, &val); | ||
5226 | if (alarm_status & (1<<0)) | ||
5227 | val &= ~(1<<0); | ||
5228 | else | ||
5229 | val |= (1<<0); | ||
5230 | bnx2x_cl45_write(bp, phy, MDIO_PMA_DEVAD, alarm_ctrl_offset, val); | ||
5231 | } | ||
5232 | /******************************************************************/ | ||
5208 | /* common BCM8706/BCM8726 PHY SECTION */ | 5233 | /* common BCM8706/BCM8726 PHY SECTION */ |
5209 | /******************************************************************/ | 5234 | /******************************************************************/ |
5210 | static u8 bnx2x_8706_8726_read_status(struct bnx2x_phy *phy, | 5235 | static u8 bnx2x_8706_8726_read_status(struct bnx2x_phy *phy, |
@@ -5218,6 +5243,10 @@ static u8 bnx2x_8706_8726_read_status(struct bnx2x_phy *phy, | |||
5218 | /* Clear RX Alarm*/ | 5243 | /* Clear RX Alarm*/ |
5219 | bnx2x_cl45_read(bp, phy, | 5244 | bnx2x_cl45_read(bp, phy, |
5220 | MDIO_PMA_DEVAD, MDIO_PMA_REG_RX_ALARM, &val2); | 5245 | MDIO_PMA_DEVAD, MDIO_PMA_REG_RX_ALARM, &val2); |
5246 | |||
5247 | bnx2x_sfp_mask_fault(bp, phy, MDIO_PMA_REG_TX_ALARM, | ||
5248 | MDIO_PMA_REG_TX_ALARM_CTRL); | ||
5249 | |||
5221 | /* clear LASI indication*/ | 5250 | /* clear LASI indication*/ |
5222 | bnx2x_cl45_read(bp, phy, | 5251 | bnx2x_cl45_read(bp, phy, |
5223 | MDIO_PMA_DEVAD, MDIO_PMA_REG_LASI_STATUS, &val1); | 5252 | MDIO_PMA_DEVAD, MDIO_PMA_REG_LASI_STATUS, &val1); |
@@ -5249,6 +5278,17 @@ static u8 bnx2x_8706_8726_read_status(struct bnx2x_phy *phy, | |||
5249 | bnx2x_ext_phy_resolve_fc(phy, params, vars); | 5278 | bnx2x_ext_phy_resolve_fc(phy, params, vars); |
5250 | vars->duplex = DUPLEX_FULL; | 5279 | vars->duplex = DUPLEX_FULL; |
5251 | } | 5280 | } |
5281 | |||
5282 | /* Capture 10G link fault. Read twice to clear stale value. */ | ||
5283 | if (vars->line_speed == SPEED_10000) { | ||
5284 | bnx2x_cl45_read(bp, phy, MDIO_PMA_DEVAD, | ||
5285 | MDIO_PMA_REG_TX_ALARM, &val1); | ||
5286 | bnx2x_cl45_read(bp, phy, MDIO_PMA_DEVAD, | ||
5287 | MDIO_PMA_REG_TX_ALARM, &val1); | ||
5288 | if (val1 & (1<<0)) | ||
5289 | vars->fault_detected = 1; | ||
5290 | } | ||
5291 | |||
5252 | return link_up; | 5292 | return link_up; |
5253 | } | 5293 | } |
5254 | 5294 | ||
@@ -5304,7 +5344,11 @@ static u8 bnx2x_8706_config_init(struct bnx2x_phy *phy, | |||
5304 | MDIO_PMA_DEVAD, | 5344 | MDIO_PMA_DEVAD, |
5305 | MDIO_PMA_REG_DIGITAL_CTRL, 0x400); | 5345 | MDIO_PMA_REG_DIGITAL_CTRL, 0x400); |
5306 | bnx2x_cl45_write(bp, phy, | 5346 | bnx2x_cl45_write(bp, phy, |
5307 | MDIO_PMA_DEVAD, MDIO_PMA_REG_LASI_CTRL, 1); | 5347 | MDIO_PMA_DEVAD, MDIO_PMA_REG_TX_ALARM_CTRL, |
5348 | 0); | ||
5349 | /* Arm LASI for link and Tx fault. */ | ||
5350 | bnx2x_cl45_write(bp, phy, | ||
5351 | MDIO_PMA_DEVAD, MDIO_PMA_REG_LASI_CTRL, 3); | ||
5308 | } else { | 5352 | } else { |
5309 | /* Force 1Gbps using autoneg with 1G advertisement */ | 5353 | /* Force 1Gbps using autoneg with 1G advertisement */ |
5310 | 5354 | ||
@@ -5637,14 +5681,17 @@ static int bnx2x_8727_config_init(struct bnx2x_phy *phy, | |||
5637 | 5681 | ||
5638 | bnx2x_wait_reset_complete(bp, phy, params); | 5682 | bnx2x_wait_reset_complete(bp, phy, params); |
5639 | rx_alarm_ctrl_val = (1<<2) | (1<<5) ; | 5683 | rx_alarm_ctrl_val = (1<<2) | (1<<5) ; |
5640 | lasi_ctrl_val = 0x0004; | 5684 | /* Should be 0x6 to enable XS on Tx side. */ |
5685 | lasi_ctrl_val = 0x0006; | ||
5641 | 5686 | ||
5642 | DP(NETIF_MSG_LINK, "Initializing BCM8727\n"); | 5687 | DP(NETIF_MSG_LINK, "Initializing BCM8727\n"); |
5643 | /* enable LASI */ | 5688 | /* enable LASI */ |
5644 | bnx2x_cl45_write(bp, phy, | 5689 | bnx2x_cl45_write(bp, phy, |
5645 | MDIO_PMA_DEVAD, MDIO_PMA_REG_RX_ALARM_CTRL, | 5690 | MDIO_PMA_DEVAD, MDIO_PMA_REG_RX_ALARM_CTRL, |
5646 | rx_alarm_ctrl_val); | 5691 | rx_alarm_ctrl_val); |
5647 | 5692 | bnx2x_cl45_write(bp, phy, | |
5693 | MDIO_PMA_DEVAD, MDIO_PMA_REG_TX_ALARM_CTRL, | ||
5694 | 0); | ||
5648 | bnx2x_cl45_write(bp, phy, | 5695 | bnx2x_cl45_write(bp, phy, |
5649 | MDIO_PMA_DEVAD, MDIO_PMA_REG_LASI_CTRL, lasi_ctrl_val); | 5696 | MDIO_PMA_DEVAD, MDIO_PMA_REG_LASI_CTRL, lasi_ctrl_val); |
5650 | 5697 | ||
@@ -5899,6 +5946,9 @@ static u8 bnx2x_8727_read_status(struct bnx2x_phy *phy, | |||
5899 | vars->line_speed = 0; | 5946 | vars->line_speed = 0; |
5900 | DP(NETIF_MSG_LINK, "8727 RX_ALARM_STATUS 0x%x\n", rx_alarm_status); | 5947 | DP(NETIF_MSG_LINK, "8727 RX_ALARM_STATUS 0x%x\n", rx_alarm_status); |
5901 | 5948 | ||
5949 | bnx2x_sfp_mask_fault(bp, phy, MDIO_PMA_REG_TX_ALARM, | ||
5950 | MDIO_PMA_REG_TX_ALARM_CTRL); | ||
5951 | |||
5902 | bnx2x_cl45_read(bp, phy, | 5952 | bnx2x_cl45_read(bp, phy, |
5903 | MDIO_PMA_DEVAD, MDIO_PMA_REG_LASI_STATUS, &val1); | 5953 | MDIO_PMA_DEVAD, MDIO_PMA_REG_LASI_STATUS, &val1); |
5904 | 5954 | ||
@@ -5991,6 +6041,20 @@ static u8 bnx2x_8727_read_status(struct bnx2x_phy *phy, | |||
5991 | DP(NETIF_MSG_LINK, "port %x: External link is down\n", | 6041 | DP(NETIF_MSG_LINK, "port %x: External link is down\n", |
5992 | params->port); | 6042 | params->port); |
5993 | } | 6043 | } |
6044 | |||
6045 | /* Capture 10G link fault. */ | ||
6046 | if (vars->line_speed == SPEED_10000) { | ||
6047 | bnx2x_cl45_read(bp, phy, MDIO_PMA_DEVAD, | ||
6048 | MDIO_PMA_REG_TX_ALARM, &val1); | ||
6049 | |||
6050 | bnx2x_cl45_read(bp, phy, MDIO_PMA_DEVAD, | ||
6051 | MDIO_PMA_REG_TX_ALARM, &val1); | ||
6052 | |||
6053 | if (val1 & (1<<0)) { | ||
6054 | vars->fault_detected = 1; | ||
6055 | } | ||
6056 | } | ||
6057 | |||
5994 | if (link_up) { | 6058 | if (link_up) { |
5995 | bnx2x_ext_phy_resolve_fc(phy, params, vars); | 6059 | bnx2x_ext_phy_resolve_fc(phy, params, vars); |
5996 | vars->duplex = DUPLEX_FULL; | 6060 | vars->duplex = DUPLEX_FULL; |