Commit 70d92f86 authored by Alexander Duyck's avatar Alexander Duyck Committed by David S. Miller

igb: update comments for serdes config and update to handle duplex

This update corrects the driver so that it handles duplex for serdes links
correctly instead of just forcing full duplex always.
Signed-off-by: default avatarAlexander Duyck <alexander.h.duyck@intel.com>
Signed-off-by: default avatarJeff Kirsher <jeffrey.t.kirsher@intel.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent cc9073bb
...@@ -706,9 +706,7 @@ static s32 igb_check_for_link_82575(struct e1000_hw *hw) ...@@ -706,9 +706,7 @@ static s32 igb_check_for_link_82575(struct e1000_hw *hw)
s32 ret_val; s32 ret_val;
u16 speed, duplex; u16 speed, duplex;
/* SGMII link check is done through the PCS register. */ if (hw->phy.media_type != e1000_media_type_copper) {
if ((hw->phy.media_type != e1000_media_type_copper) ||
(igb_sgmii_active_82575(hw))) {
ret_val = igb_get_pcs_speed_and_duplex_82575(hw, &speed, ret_val = igb_get_pcs_speed_and_duplex_82575(hw, &speed,
&duplex); &duplex);
/* /*
...@@ -723,6 +721,7 @@ static s32 igb_check_for_link_82575(struct e1000_hw *hw) ...@@ -723,6 +721,7 @@ static s32 igb_check_for_link_82575(struct e1000_hw *hw)
return ret_val; return ret_val;
} }
/** /**
* igb_get_pcs_speed_and_duplex_82575 - Retrieve current speed/duplex * igb_get_pcs_speed_and_duplex_82575 - Retrieve current speed/duplex
* @hw: pointer to the HW structure * @hw: pointer to the HW structure
...@@ -788,13 +787,23 @@ static s32 igb_get_pcs_speed_and_duplex_82575(struct e1000_hw *hw, u16 *speed, ...@@ -788,13 +787,23 @@ static s32 igb_get_pcs_speed_and_duplex_82575(struct e1000_hw *hw, u16 *speed,
void igb_shutdown_serdes_link_82575(struct e1000_hw *hw) void igb_shutdown_serdes_link_82575(struct e1000_hw *hw)
{ {
u32 reg; u32 reg;
u16 eeprom_data = 0;
if (hw->phy.media_type != e1000_media_type_internal_serdes || if (hw->phy.media_type != e1000_media_type_internal_serdes ||
igb_sgmii_active_82575(hw)) igb_sgmii_active_82575(hw))
return; return;
/* if the management interface is not enabled, then power down */ if (hw->bus.func == E1000_FUNC_0)
if (!igb_enable_mng_pass_thru(hw)) { hw->nvm.ops.read(hw, NVM_INIT_CONTROL3_PORT_A, 1, &eeprom_data);
else if (hw->bus.func == E1000_FUNC_1)
hw->nvm.ops.read(hw, NVM_INIT_CONTROL3_PORT_B, 1, &eeprom_data);
/*
* If APM is not enabled in the EEPROM and management interface is
* not enabled, then power down.
*/
if (!(eeprom_data & E1000_NVM_APME_82575) &&
!igb_enable_mng_pass_thru(hw)) {
/* Disable PCS to turn off link */ /* Disable PCS to turn off link */
reg = rd32(E1000_PCS_CFG0); reg = rd32(E1000_PCS_CFG0);
reg &= ~E1000_PCS_CFG_PCS_EN; reg &= ~E1000_PCS_CFG_PCS_EN;
...@@ -1010,10 +1019,13 @@ static s32 igb_setup_copper_link_82575(struct e1000_hw *hw) ...@@ -1010,10 +1019,13 @@ static s32 igb_setup_copper_link_82575(struct e1000_hw *hw)
} }
/** /**
* igb_setup_serdes_link_82575 - Setup link for fiber/serdes * igb_setup_serdes_link_82575 - Setup link for serdes
* @hw: pointer to the HW structure * @hw: pointer to the HW structure
* *
* Configures speed and duplex for fiber and serdes links. * Configure the physical coding sub-layer (PCS) link. The PCS link is
* used on copper connections where the serialized gigabit media independent
* interface (sgmii), or serdes fiber is being used. Configures the link
* for auto-negotiation or forces speed/duplex.
**/ **/
static s32 igb_setup_serdes_link_82575(struct e1000_hw *hw) static s32 igb_setup_serdes_link_82575(struct e1000_hw *hw)
{ {
...@@ -1086,18 +1098,27 @@ static s32 igb_setup_serdes_link_82575(struct e1000_hw *hw) ...@@ -1086,18 +1098,27 @@ static s32 igb_setup_serdes_link_82575(struct e1000_hw *hw)
*/ */
if (hw->mac.autoneg || igb_sgmii_active_82575(hw)) { if (hw->mac.autoneg || igb_sgmii_active_82575(hw)) {
/* Set PCS register for autoneg */ /* Set PCS register for autoneg */
reg |= E1000_PCS_LCTL_FSV_1000 | /* Force 1000 */ reg |= E1000_PCS_LCTL_FSV_1000 | /* Force 1000 */
E1000_PCS_LCTL_FDV_FULL | /* SerDes Full duplex */ E1000_PCS_LCTL_FDV_FULL | /* SerDes Full dplx */
E1000_PCS_LCTL_AN_ENABLE | /* Enable Autoneg */ E1000_PCS_LCTL_AN_ENABLE | /* Enable Autoneg */
E1000_PCS_LCTL_AN_RESTART; /* Restart autoneg */ E1000_PCS_LCTL_AN_RESTART; /* Restart autoneg */
hw_dbg("Configuring Autoneg; PCS_LCTL = 0x%08X\n", reg); hw_dbg("Configuring Autoneg; PCS_LCTL = 0x%08X\n", reg);
} else { } else {
/* Set PCS register for forced speed */ /* Check for duplex first */
reg |= E1000_PCS_LCTL_FLV_LINK_UP | /* Force link up */ if (hw->mac.forced_speed_duplex & E1000_ALL_FULL_DUPLEX)
E1000_PCS_LCTL_FSV_1000 | /* Force 1000 */ reg |= E1000_PCS_LCTL_FDV_FULL;
E1000_PCS_LCTL_FDV_FULL | /* SerDes Full duplex */
E1000_PCS_LCTL_FSD | /* Force Speed */ /* No need to check for 1000/full since the spec states that
E1000_PCS_LCTL_FORCE_LINK; /* Force Link */ * it requires autoneg to be enabled */
/* Now set speed */
if (hw->mac.forced_speed_duplex & E1000_ALL_100_SPEED)
reg |= E1000_PCS_LCTL_FSV_100;
/* Force speed and force link */
reg |= E1000_PCS_LCTL_FSD |
E1000_PCS_LCTL_FORCE_LINK |
E1000_PCS_LCTL_FLV_LINK_UP;
hw_dbg("Configuring Forced Link; PCS_LCTL = 0x%08X\n", reg); hw_dbg("Configuring Forced Link; PCS_LCTL = 0x%08X\n", reg);
} }
......
...@@ -167,6 +167,7 @@ struct e1000_adv_tx_context_desc { ...@@ -167,6 +167,7 @@ struct e1000_adv_tx_context_desc {
#define E1000_DCA_TXCTRL_CPUID_SHIFT 24 /* Tx CPUID now in the last byte */ #define E1000_DCA_TXCTRL_CPUID_SHIFT 24 /* Tx CPUID now in the last byte */
#define E1000_DCA_RXCTRL_CPUID_SHIFT 24 /* Rx CPUID now in the last byte */ #define E1000_DCA_RXCTRL_CPUID_SHIFT 24 /* Rx CPUID now in the last byte */
#define E1000_NVM_APME_82575 0x0400
#define MAX_NUM_VFS 8 #define MAX_NUM_VFS 8
#define E1000_DTXSWC_VMDQ_LOOPBACK_EN (1 << 31) /* global VF LB enable */ #define E1000_DTXSWC_VMDQ_LOOPBACK_EN (1 << 31) /* global VF LB enable */
......
...@@ -50,6 +50,7 @@ struct e1000_hw; ...@@ -50,6 +50,7 @@ struct e1000_hw;
#define E1000_REVISION_2 2 #define E1000_REVISION_2 2
#define E1000_REVISION_4 4 #define E1000_REVISION_4 4
#define E1000_FUNC_0 0
#define E1000_FUNC_1 1 #define E1000_FUNC_1 1
enum e1000_mac_type { enum e1000_mac_type {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment