Commit 8c99e7b0 authored by Yaniv Rosner's avatar Yaniv Rosner Committed by David S. Miller

bnx2x: Pause settings

Pause settings
- 1G pause was not working due to missing write to the emac block
  (TX_MODE_FLOW_EN)
- The flow control should use the negotiated result (after autoneg) so
  we should save both the requested autoneg and the result
- The HW credits with flow control at 1G speed were not optimized and
  caused low throughput
- It is recommended to turn off flow control if the MTU is bigger than
  5000B due to internal buffers size
Signed-off-by: default avatarYaniv Rosner <yanivr@broadcom.com>
Signed-off-by: default avatarEilon Greenstein <eilong@broadcom.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 57963ed9
...@@ -38,7 +38,6 @@ ...@@ -38,7 +38,6 @@
#define ETH_MAX_JUMBO_PACKET_SIZE 9600 #define ETH_MAX_JUMBO_PACKET_SIZE 9600
#define MDIO_ACCESS_TIMEOUT 1000 #define MDIO_ACCESS_TIMEOUT 1000
#define BMAC_CONTROL_RX_ENABLE 2 #define BMAC_CONTROL_RX_ENABLE 2
#define MAX_MTU_SIZE 5000
/***********************************************************/ /***********************************************************/
/* Shortcut definitions */ /* Shortcut definitions */
...@@ -298,11 +297,13 @@ static u8 bnx2x_emac_enable(struct link_params *params, ...@@ -298,11 +297,13 @@ static u8 bnx2x_emac_enable(struct link_params *params,
EMAC_RX_MODE_FLOW_EN); EMAC_RX_MODE_FLOW_EN);
bnx2x_bits_dis(bp, emac_base + EMAC_REG_EMAC_TX_MODE, bnx2x_bits_dis(bp, emac_base + EMAC_REG_EMAC_TX_MODE,
EMAC_TX_MODE_EXT_PAUSE_EN); (EMAC_TX_MODE_EXT_PAUSE_EN |
EMAC_TX_MODE_FLOW_EN));
if (vars->flow_ctrl & FLOW_CTRL_TX) if (vars->flow_ctrl & FLOW_CTRL_TX)
bnx2x_bits_en(bp, emac_base + bnx2x_bits_en(bp, emac_base +
EMAC_REG_EMAC_TX_MODE, EMAC_REG_EMAC_TX_MODE,
EMAC_TX_MODE_EXT_PAUSE_EN); (EMAC_TX_MODE_EXT_PAUSE_EN |
EMAC_TX_MODE_FLOW_EN));
} }
/* KEEP_VLAN_TAG, promiscuous */ /* KEEP_VLAN_TAG, promiscuous */
...@@ -591,9 +592,9 @@ void bnx2x_link_status_update(struct link_params *params, ...@@ -591,9 +592,9 @@ void bnx2x_link_status_update(struct link_params *params,
vars->flow_ctrl &= ~FLOW_CTRL_RX; vars->flow_ctrl &= ~FLOW_CTRL_RX;
if (vars->phy_flags & PHY_XGXS_FLAG) { if (vars->phy_flags & PHY_XGXS_FLAG) {
if (params->req_line_speed && if (vars->line_speed &&
((params->req_line_speed == SPEED_10) || ((vars->line_speed == SPEED_10) ||
(params->req_line_speed == SPEED_100))) { (vars->line_speed == SPEED_100))) {
vars->phy_flags |= PHY_SGMII_FLAG; vars->phy_flags |= PHY_SGMII_FLAG;
} else { } else {
vars->phy_flags &= ~PHY_SGMII_FLAG; vars->phy_flags &= ~PHY_SGMII_FLAG;
...@@ -670,7 +671,6 @@ static u8 bnx2x_pbf_update(struct link_params *params, u32 flow_ctrl, ...@@ -670,7 +671,6 @@ static u8 bnx2x_pbf_update(struct link_params *params, u32 flow_ctrl,
u8 port = params->port; u8 port = params->port;
u32 init_crd, crd; u32 init_crd, crd;
u32 count = 1000; u32 count = 1000;
u32 pause = 0;
/* disable port */ /* disable port */
REG_WR(bp, PBF_REG_DISABLE_NEW_TASK_PROC_P0 + port*4, 0x1); REG_WR(bp, PBF_REG_DISABLE_NEW_TASK_PROC_P0 + port*4, 0x1);
...@@ -693,33 +693,25 @@ static u8 bnx2x_pbf_update(struct link_params *params, u32 flow_ctrl, ...@@ -693,33 +693,25 @@ static u8 bnx2x_pbf_update(struct link_params *params, u32 flow_ctrl,
return -EINVAL; return -EINVAL;
} }
if (flow_ctrl & FLOW_CTRL_RX) if (flow_ctrl & FLOW_CTRL_RX ||
pause = 1; line_speed == SPEED_10 ||
REG_WR(bp, PBF_REG_P0_PAUSE_ENABLE + port*4, pause); line_speed == SPEED_100 ||
if (pause) { line_speed == SPEED_1000 ||
line_speed == SPEED_2500) {
REG_WR(bp, PBF_REG_P0_PAUSE_ENABLE + port*4, 1);
/* update threshold */ /* update threshold */
REG_WR(bp, PBF_REG_P0_ARB_THRSH + port*4, 0); REG_WR(bp, PBF_REG_P0_ARB_THRSH + port*4, 0);
/* update init credit */ /* update init credit */
init_crd = 778; /* (800-18-4) */ init_crd = 778; /* (800-18-4) */
} else { } else {
u32 thresh = (ETH_MAX_JUMBO_PACKET_SIZE + u32 thresh = (ETH_MAX_JUMBO_PACKET_SIZE +
ETH_OVREHEAD)/16; ETH_OVREHEAD)/16;
REG_WR(bp, PBF_REG_P0_PAUSE_ENABLE + port*4, 0);
/* update threshold */ /* update threshold */
REG_WR(bp, PBF_REG_P0_ARB_THRSH + port*4, thresh); REG_WR(bp, PBF_REG_P0_ARB_THRSH + port*4, thresh);
/* update init credit */ /* update init credit */
switch (line_speed) { switch (line_speed) {
case SPEED_10:
case SPEED_100:
case SPEED_1000:
init_crd = thresh + 55 - 22;
break;
case SPEED_2500:
init_crd = thresh + 138 - 22;
break;
case SPEED_10000: case SPEED_10000:
init_crd = thresh + 553 - 22; init_crd = thresh + 553 - 22;
break; break;
...@@ -1114,7 +1106,7 @@ static void bnx2x_set_autoneg(struct link_params *params, ...@@ -1114,7 +1106,7 @@ static void bnx2x_set_autoneg(struct link_params *params,
MDIO_COMBO_IEEE0_MII_CONTROL, &reg_val); MDIO_COMBO_IEEE0_MII_CONTROL, &reg_val);
/* CL37 Autoneg Enabled */ /* CL37 Autoneg Enabled */
if (params->req_line_speed == SPEED_AUTO_NEG) if (vars->line_speed == SPEED_AUTO_NEG)
reg_val |= MDIO_COMBO_IEEO_MII_CONTROL_AN_EN; reg_val |= MDIO_COMBO_IEEO_MII_CONTROL_AN_EN;
else /* CL37 Autoneg Disabled */ else /* CL37 Autoneg Disabled */
reg_val &= ~(MDIO_COMBO_IEEO_MII_CONTROL_AN_EN | reg_val &= ~(MDIO_COMBO_IEEO_MII_CONTROL_AN_EN |
...@@ -1132,7 +1124,7 @@ static void bnx2x_set_autoneg(struct link_params *params, ...@@ -1132,7 +1124,7 @@ static void bnx2x_set_autoneg(struct link_params *params,
MDIO_REG_BANK_SERDES_DIGITAL, MDIO_REG_BANK_SERDES_DIGITAL,
MDIO_SERDES_DIGITAL_A_1000X_CONTROL1, &reg_val); MDIO_SERDES_DIGITAL_A_1000X_CONTROL1, &reg_val);
reg_val &= ~MDIO_SERDES_DIGITAL_A_1000X_CONTROL1_SIGNAL_DETECT_EN; reg_val &= ~MDIO_SERDES_DIGITAL_A_1000X_CONTROL1_SIGNAL_DETECT_EN;
if (params->req_line_speed == SPEED_AUTO_NEG) if (vars->line_speed == SPEED_AUTO_NEG)
reg_val |= MDIO_SERDES_DIGITAL_A_1000X_CONTROL1_AUTODET; reg_val |= MDIO_SERDES_DIGITAL_A_1000X_CONTROL1_AUTODET;
else else
reg_val &= ~MDIO_SERDES_DIGITAL_A_1000X_CONTROL1_AUTODET; reg_val &= ~MDIO_SERDES_DIGITAL_A_1000X_CONTROL1_AUTODET;
...@@ -1148,7 +1140,7 @@ static void bnx2x_set_autoneg(struct link_params *params, ...@@ -1148,7 +1140,7 @@ static void bnx2x_set_autoneg(struct link_params *params,
MDIO_REG_BANK_BAM_NEXT_PAGE, MDIO_REG_BANK_BAM_NEXT_PAGE,
MDIO_BAM_NEXT_PAGE_MP5_NEXT_PAGE_CTRL, MDIO_BAM_NEXT_PAGE_MP5_NEXT_PAGE_CTRL,
&reg_val); &reg_val);
if (params->req_line_speed == SPEED_AUTO_NEG) { if (vars->line_speed == SPEED_AUTO_NEG) {
/* Enable BAM aneg Mode and TetonII aneg Mode */ /* Enable BAM aneg Mode and TetonII aneg Mode */
reg_val |= (MDIO_BAM_NEXT_PAGE_MP5_NEXT_PAGE_CTRL_BAM_MODE | reg_val |= (MDIO_BAM_NEXT_PAGE_MP5_NEXT_PAGE_CTRL_BAM_MODE |
MDIO_BAM_NEXT_PAGE_MP5_NEXT_PAGE_CTRL_TETON_AN); MDIO_BAM_NEXT_PAGE_MP5_NEXT_PAGE_CTRL_TETON_AN);
...@@ -1164,7 +1156,7 @@ static void bnx2x_set_autoneg(struct link_params *params, ...@@ -1164,7 +1156,7 @@ static void bnx2x_set_autoneg(struct link_params *params,
reg_val); reg_val);
/* Enable Clause 73 Aneg */ /* Enable Clause 73 Aneg */
if ((params->req_line_speed == SPEED_AUTO_NEG) && if ((vars->line_speed == SPEED_AUTO_NEG) &&
(SUPPORT_CL73)) { (SUPPORT_CL73)) {
/* Enable BAM Station Manager */ /* Enable BAM Station Manager */
...@@ -1226,7 +1218,8 @@ static void bnx2x_set_autoneg(struct link_params *params, ...@@ -1226,7 +1218,8 @@ static void bnx2x_set_autoneg(struct link_params *params,
} }
/* program SerDes, forced speed */ /* program SerDes, forced speed */
static void bnx2x_program_serdes(struct link_params *params) static void bnx2x_program_serdes(struct link_params *params,
struct link_vars *vars)
{ {
struct bnx2x *bp = params->bp; struct bnx2x *bp = params->bp;
u16 reg_val; u16 reg_val;
...@@ -1248,28 +1241,35 @@ static void bnx2x_program_serdes(struct link_params *params) ...@@ -1248,28 +1241,35 @@ static void bnx2x_program_serdes(struct link_params *params)
/* program speed /* program speed
- needed only if the speed is greater than 1G (2.5G or 10G) */ - needed only if the speed is greater than 1G (2.5G or 10G) */
if (!((params->req_line_speed == SPEED_1000) || CL45_RD_OVER_CL22(bp, params->port,
(params->req_line_speed == SPEED_100) ||
(params->req_line_speed == SPEED_10))) {
CL45_RD_OVER_CL22(bp, params->port,
params->phy_addr, params->phy_addr,
MDIO_REG_BANK_SERDES_DIGITAL, MDIO_REG_BANK_SERDES_DIGITAL,
MDIO_SERDES_DIGITAL_MISC1, &reg_val); MDIO_SERDES_DIGITAL_MISC1, &reg_val);
/* clearing the speed value before setting the right speed */ /* clearing the speed value before setting the right speed */
reg_val &= ~MDIO_SERDES_DIGITAL_MISC1_FORCE_SPEED_MASK; DP(NETIF_MSG_LINK, "MDIO_REG_BANK_SERDES_DIGITAL = 0x%x\n", reg_val);
reg_val &= ~(MDIO_SERDES_DIGITAL_MISC1_FORCE_SPEED_MASK |
MDIO_SERDES_DIGITAL_MISC1_FORCE_SPEED_SEL);
if (!((vars->line_speed == SPEED_1000) ||
(vars->line_speed == SPEED_100) ||
(vars->line_speed == SPEED_10))) {
reg_val |= (MDIO_SERDES_DIGITAL_MISC1_REFCLK_SEL_156_25M | reg_val |= (MDIO_SERDES_DIGITAL_MISC1_REFCLK_SEL_156_25M |
MDIO_SERDES_DIGITAL_MISC1_FORCE_SPEED_SEL); MDIO_SERDES_DIGITAL_MISC1_FORCE_SPEED_SEL);
if (params->req_line_speed == SPEED_10000) if (vars->line_speed == SPEED_10000)
reg_val |= reg_val |=
MDIO_SERDES_DIGITAL_MISC1_FORCE_SPEED_10G_CX4; MDIO_SERDES_DIGITAL_MISC1_FORCE_SPEED_10G_CX4;
if (params->req_line_speed == SPEED_13000) if (vars->line_speed == SPEED_13000)
reg_val |= reg_val |=
MDIO_SERDES_DIGITAL_MISC1_FORCE_SPEED_13G; MDIO_SERDES_DIGITAL_MISC1_FORCE_SPEED_13G;
CL45_WR_OVER_CL22(bp, params->port, }
CL45_WR_OVER_CL22(bp, params->port,
params->phy_addr, params->phy_addr,
MDIO_REG_BANK_SERDES_DIGITAL, MDIO_REG_BANK_SERDES_DIGITAL,
MDIO_SERDES_DIGITAL_MISC1, reg_val); MDIO_SERDES_DIGITAL_MISC1, reg_val);
}
} }
static void bnx2x_set_brcm_cl37_advertisment(struct link_params *params) static void bnx2x_set_brcm_cl37_advertisment(struct link_params *params)
...@@ -1295,48 +1295,49 @@ static void bnx2x_set_brcm_cl37_advertisment(struct link_params *params) ...@@ -1295,48 +1295,49 @@ static void bnx2x_set_brcm_cl37_advertisment(struct link_params *params)
MDIO_OVER_1G_UP3, 0); MDIO_OVER_1G_UP3, 0);
} }
static void bnx2x_set_ieee_aneg_advertisment(struct link_params *params, static void bnx2x_calc_ieee_aneg_adv(struct link_params *params, u32 *ieee_fc)
u32 *ieee_fc)
{ {
struct bnx2x *bp = params->bp; *ieee_fc = MDIO_COMBO_IEEE0_AUTO_NEG_ADV_FULL_DUPLEX;
/* for AN, we are always publishing full duplex */
u16 an_adv = MDIO_COMBO_IEEE0_AUTO_NEG_ADV_FULL_DUPLEX;
/* resolve pause mode and advertisement /* resolve pause mode and advertisement
* Please refer to Table 28B-3 of the 802.3ab-1999 spec */ * Please refer to Table 28B-3 of the 802.3ab-1999 spec */
switch (params->req_flow_ctrl) { switch (params->req_flow_ctrl) {
case FLOW_CTRL_AUTO: case FLOW_CTRL_AUTO:
if (params->mtu <= MAX_MTU_SIZE) { if (params->req_fc_auto_adv == FLOW_CTRL_BOTH) {
an_adv |= *ieee_fc |=
MDIO_COMBO_IEEE0_AUTO_NEG_ADV_PAUSE_BOTH; MDIO_COMBO_IEEE0_AUTO_NEG_ADV_PAUSE_BOTH;
} else { } else {
an_adv |= *ieee_fc |=
MDIO_COMBO_IEEE0_AUTO_NEG_ADV_PAUSE_ASYMMETRIC; MDIO_COMBO_IEEE0_AUTO_NEG_ADV_PAUSE_ASYMMETRIC;
} }
break; break;
case FLOW_CTRL_TX: case FLOW_CTRL_TX:
an_adv |= *ieee_fc |=
MDIO_COMBO_IEEE0_AUTO_NEG_ADV_PAUSE_ASYMMETRIC; MDIO_COMBO_IEEE0_AUTO_NEG_ADV_PAUSE_ASYMMETRIC;
break; break;
case FLOW_CTRL_RX: case FLOW_CTRL_RX:
case FLOW_CTRL_BOTH: case FLOW_CTRL_BOTH:
an_adv |= MDIO_COMBO_IEEE0_AUTO_NEG_ADV_PAUSE_BOTH; *ieee_fc |= MDIO_COMBO_IEEE0_AUTO_NEG_ADV_PAUSE_BOTH;
break; break;
case FLOW_CTRL_NONE: case FLOW_CTRL_NONE:
default: default:
an_adv |= MDIO_COMBO_IEEE0_AUTO_NEG_ADV_PAUSE_NONE; *ieee_fc |= MDIO_COMBO_IEEE0_AUTO_NEG_ADV_PAUSE_NONE;
break; break;
} }
}
*ieee_fc = an_adv; static void bnx2x_set_ieee_aneg_advertisment(struct link_params *params,
u32 ieee_fc)
{
struct bnx2x *bp = params->bp;
/* for AN, we are always publishing full duplex */
CL45_WR_OVER_CL22(bp, params->port, CL45_WR_OVER_CL22(bp, params->port,
params->phy_addr, params->phy_addr,
MDIO_REG_BANK_COMBO_IEEE0, MDIO_REG_BANK_COMBO_IEEE0,
MDIO_COMBO_IEEE0_AUTO_NEG_ADV, an_adv); MDIO_COMBO_IEEE0_AUTO_NEG_ADV, (u16)ieee_fc);
} }
static void bnx2x_restart_autoneg(struct link_params *params) static void bnx2x_restart_autoneg(struct link_params *params)
...@@ -1382,7 +1383,8 @@ static void bnx2x_restart_autoneg(struct link_params *params) ...@@ -1382,7 +1383,8 @@ static void bnx2x_restart_autoneg(struct link_params *params)
} }
} }
static void bnx2x_initialize_sgmii_process(struct link_params *params) static void bnx2x_initialize_sgmii_process(struct link_params *params,
struct link_vars *vars)
{ {
struct bnx2x *bp = params->bp; struct bnx2x *bp = params->bp;
u16 control1; u16 control1;
...@@ -1406,7 +1408,7 @@ static void bnx2x_initialize_sgmii_process(struct link_params *params) ...@@ -1406,7 +1408,7 @@ static void bnx2x_initialize_sgmii_process(struct link_params *params)
control1); control1);
/* if forced speed */ /* if forced speed */
if (!(params->req_line_speed == SPEED_AUTO_NEG)) { if (!(vars->line_speed == SPEED_AUTO_NEG)) {
/* set speed, disable autoneg */ /* set speed, disable autoneg */
u16 mii_control; u16 mii_control;
...@@ -1419,7 +1421,7 @@ static void bnx2x_initialize_sgmii_process(struct link_params *params) ...@@ -1419,7 +1421,7 @@ static void bnx2x_initialize_sgmii_process(struct link_params *params)
MDIO_COMBO_IEEO_MII_CONTROL_MAN_SGMII_SP_MASK| MDIO_COMBO_IEEO_MII_CONTROL_MAN_SGMII_SP_MASK|
MDIO_COMBO_IEEO_MII_CONTROL_FULL_DUPLEX); MDIO_COMBO_IEEO_MII_CONTROL_FULL_DUPLEX);
switch (params->req_line_speed) { switch (vars->line_speed) {
case SPEED_100: case SPEED_100:
mii_control |= mii_control |=
MDIO_COMBO_IEEO_MII_CONTROL_MAN_SGMII_SP_100; MDIO_COMBO_IEEO_MII_CONTROL_MAN_SGMII_SP_100;
...@@ -1433,8 +1435,8 @@ static void bnx2x_initialize_sgmii_process(struct link_params *params) ...@@ -1433,8 +1435,8 @@ static void bnx2x_initialize_sgmii_process(struct link_params *params)
break; break;
default: default:
/* invalid speed for SGMII */ /* invalid speed for SGMII */
DP(NETIF_MSG_LINK, "Invalid req_line_speed 0x%x\n", DP(NETIF_MSG_LINK, "Invalid line_speed 0x%x\n",
params->req_line_speed); vars->line_speed);
break; break;
} }
...@@ -1460,20 +1462,20 @@ static void bnx2x_initialize_sgmii_process(struct link_params *params) ...@@ -1460,20 +1462,20 @@ static void bnx2x_initialize_sgmii_process(struct link_params *params)
*/ */
static void bnx2x_pause_resolve(struct link_vars *vars, u32 pause_result) static void bnx2x_pause_resolve(struct link_vars *vars, u32 pause_result)
{ { /* LD LP */
switch (pause_result) { /* ASYM P ASYM P */ switch (pause_result) { /* ASYM P ASYM P */
case 0xb: /* 1 0 1 1 */ case 0xb: /* 1 0 1 1 */
vars->flow_ctrl = FLOW_CTRL_TX; vars->flow_ctrl = FLOW_CTRL_TX;
break; break;
case 0xe: /* 1 1 1 0 */ case 0xe: /* 1 1 1 0 */
vars->flow_ctrl = FLOW_CTRL_RX; vars->flow_ctrl = FLOW_CTRL_RX;
break; break;
case 0x5: /* 0 1 0 1 */ case 0x5: /* 0 1 0 1 */
case 0x7: /* 0 1 1 1 */ case 0x7: /* 0 1 1 1 */
case 0xd: /* 1 1 0 1 */ case 0xd: /* 1 1 0 1 */
case 0xf: /* 1 1 1 1 */ case 0xf: /* 1 1 1 1 */
vars->flow_ctrl = FLOW_CTRL_BOTH; vars->flow_ctrl = FLOW_CTRL_BOTH;
break; break;
...@@ -1487,8 +1489,8 @@ static u8 bnx2x_ext_phy_resove_fc(struct link_params *params, ...@@ -1487,8 +1489,8 @@ static u8 bnx2x_ext_phy_resove_fc(struct link_params *params,
{ {
struct bnx2x *bp = params->bp; struct bnx2x *bp = params->bp;
u8 ext_phy_addr; u8 ext_phy_addr;
u16 ld_pause; /* local */ u16 ld_pause; /* local */
u16 lp_pause; /* link partner */ u16 lp_pause; /* link partner */
u16 an_complete; /* AN complete */ u16 an_complete; /* AN complete */
u16 pause_result; u16 pause_result;
u8 ret = 0; u8 ret = 0;
...@@ -1531,6 +1533,28 @@ static u8 bnx2x_ext_phy_resove_fc(struct link_params *params, ...@@ -1531,6 +1533,28 @@ static u8 bnx2x_ext_phy_resove_fc(struct link_params *params,
DP(NETIF_MSG_LINK, "Ext PHY pause result 0x%x \n", DP(NETIF_MSG_LINK, "Ext PHY pause result 0x%x \n",
pause_result); pause_result);
bnx2x_pause_resolve(vars, pause_result); bnx2x_pause_resolve(vars, pause_result);
if (vars->flow_ctrl == FLOW_CTRL_NONE &&
ext_phy_type == PORT_HW_CFG_XGXS_EXT_PHY_TYPE_BCM8073) {
bnx2x_cl45_read(bp, port,
ext_phy_type,
ext_phy_addr,
MDIO_AN_DEVAD,
MDIO_AN_REG_CL37_FC_LD, &ld_pause);
bnx2x_cl45_read(bp, port,
ext_phy_type,
ext_phy_addr,
MDIO_AN_DEVAD,
MDIO_AN_REG_CL37_FC_LP, &lp_pause);
pause_result = (ld_pause &
MDIO_COMBO_IEEE0_AUTO_NEG_ADV_PAUSE_BOTH) >> 5;
pause_result |= (lp_pause &
MDIO_COMBO_IEEE0_AUTO_NEG_ADV_PAUSE_BOTH) >> 7;
bnx2x_pause_resolve(vars, pause_result);
DP(NETIF_MSG_LINK, "Ext PHY CL37 pause result 0x%x \n",
pause_result);
}
} }
return ret; return ret;
} }
...@@ -1573,13 +1597,10 @@ static void bnx2x_flow_ctrl_resolve(struct link_params *params, ...@@ -1573,13 +1597,10 @@ static void bnx2x_flow_ctrl_resolve(struct link_params *params,
(bnx2x_ext_phy_resove_fc(params, vars))) { (bnx2x_ext_phy_resove_fc(params, vars))) {
return; return;
} else { } else {
vars->flow_ctrl = params->req_flow_ctrl; if (params->req_flow_ctrl == FLOW_CTRL_AUTO)
if (vars->flow_ctrl == FLOW_CTRL_AUTO) { vars->flow_ctrl = params->req_fc_auto_adv;
if (params->mtu <= MAX_MTU_SIZE) else
vars->flow_ctrl = FLOW_CTRL_BOTH; vars->flow_ctrl = params->req_flow_ctrl;
else
vars->flow_ctrl = FLOW_CTRL_TX;
}
} }
DP(NETIF_MSG_LINK, "flow_ctrl 0x%x\n", vars->flow_ctrl); DP(NETIF_MSG_LINK, "flow_ctrl 0x%x\n", vars->flow_ctrl);
} }
...@@ -1709,12 +1730,12 @@ static u8 bnx2x_link_settings_status(struct link_params *params, ...@@ -1709,12 +1730,12 @@ static u8 bnx2x_link_settings_status(struct link_params *params,
} }
if (vars->flow_ctrl & FLOW_CTRL_TX) if (vars->flow_ctrl & FLOW_CTRL_TX)
vars->link_status |= vars->link_status |=
LINK_STATUS_TX_FLOW_CONTROL_ENABLED; LINK_STATUS_TX_FLOW_CONTROL_ENABLED;
if (vars->flow_ctrl & FLOW_CTRL_RX) if (vars->flow_ctrl & FLOW_CTRL_RX)
vars->link_status |= vars->link_status |=
LINK_STATUS_RX_FLOW_CONTROL_ENABLED; LINK_STATUS_RX_FLOW_CONTROL_ENABLED;
} else { /* link_down */ } else { /* link_down */
DP(NETIF_MSG_LINK, "phy link down\n"); DP(NETIF_MSG_LINK, "phy link down\n");
...@@ -2286,13 +2307,16 @@ static void bnx2x_ext_phy_set_pause(struct link_params *params, ...@@ -2286,13 +2307,16 @@ static void bnx2x_ext_phy_set_pause(struct link_params *params,
MDIO_AN_REG_ADV_PAUSE, &val); MDIO_AN_REG_ADV_PAUSE, &val);
val &= ~MDIO_AN_REG_ADV_PAUSE_BOTH; val &= ~MDIO_AN_REG_ADV_PAUSE_BOTH;
/* Please refer to Table 28B-3 of 802.3ab-1999 spec. */ /* Please refer to Table 28B-3 of 802.3ab-1999 spec. */
if (vars->ieee_fc & if ((vars->ieee_fc &
MDIO_COMBO_IEEE0_AUTO_NEG_ADV_PAUSE_ASYMMETRIC) ==
MDIO_COMBO_IEEE0_AUTO_NEG_ADV_PAUSE_ASYMMETRIC) { MDIO_COMBO_IEEE0_AUTO_NEG_ADV_PAUSE_ASYMMETRIC) {
val |= MDIO_AN_REG_ADV_PAUSE_ASYMMETRIC; val |= MDIO_AN_REG_ADV_PAUSE_ASYMMETRIC;
} }
if (vars->ieee_fc & if ((vars->ieee_fc &
MDIO_COMBO_IEEE0_AUTO_NEG_ADV_PAUSE_BOTH) ==
MDIO_COMBO_IEEE0_AUTO_NEG_ADV_PAUSE_BOTH) { MDIO_COMBO_IEEE0_AUTO_NEG_ADV_PAUSE_BOTH) {
val |= val |=
MDIO_AN_REG_ADV_PAUSE_PAUSE; MDIO_AN_REG_ADV_PAUSE_PAUSE;
...@@ -2339,7 +2363,7 @@ static void bnx2x_init_internal_phy(struct link_params *params, ...@@ -2339,7 +2363,7 @@ static void bnx2x_init_internal_phy(struct link_params *params,
bnx2x_set_autoneg(params, vars); bnx2x_set_autoneg(params, vars);
/* program speed and duplex */ /* program speed and duplex */
bnx2x_program_serdes(params); bnx2x_program_serdes(params, vars);
} else { /* AN_mode */ } else { /* AN_mode */
DP(NETIF_MSG_LINK, "not SGMII, AN\n"); DP(NETIF_MSG_LINK, "not SGMII, AN\n");
...@@ -2349,7 +2373,7 @@ static void bnx2x_init_internal_phy(struct link_params *params, ...@@ -2349,7 +2373,7 @@ static void bnx2x_init_internal_phy(struct link_params *params,
/* program duplex & pause advertisement (for aneg) */ /* program duplex & pause advertisement (for aneg) */
bnx2x_set_ieee_aneg_advertisment(params, bnx2x_set_ieee_aneg_advertisment(params,
&vars->ieee_fc); vars->ieee_fc);
/* enable autoneg */ /* enable autoneg */
bnx2x_set_autoneg(params, vars); bnx2x_set_autoneg(params, vars);
...@@ -2361,7 +2385,7 @@ static void bnx2x_init_internal_phy(struct link_params *params, ...@@ -2361,7 +2385,7 @@ static void bnx2x_init_internal_phy(struct link_params *params,
} else { /* SGMII mode */ } else { /* SGMII mode */
DP(NETIF_MSG_LINK, "SGMII\n"); DP(NETIF_MSG_LINK, "SGMII\n");
bnx2x_initialize_sgmii_process(params); bnx2x_initialize_sgmii_process(params, vars);
} }
} }
...@@ -2481,7 +2505,7 @@ static u8 bnx2x_ext_phy_init(struct link_params *params, struct link_vars *vars) ...@@ -2481,7 +2505,7 @@ static u8 bnx2x_ext_phy_init(struct link_params *params, struct link_vars *vars)
ext_phy_type, ext_phy_type,
ext_phy_addr, ext_phy_addr,
MDIO_AN_DEVAD, MDIO_AN_DEVAD,
MDIO_AN_REG_CL37_FD, MDIO_AN_REG_CL37_FC_LP,
0x0020); 0x0020);
/* Enable CL37 AN */ /* Enable CL37 AN */
bnx2x_cl45_write(bp, params->port, bnx2x_cl45_write(bp, params->port,
...@@ -2657,13 +2681,13 @@ static u8 bnx2x_ext_phy_init(struct link_params *params, struct link_vars *vars) ...@@ -2657,13 +2681,13 @@ static u8 bnx2x_ext_phy_init(struct link_params *params, struct link_vars *vars)
ext_phy_type, ext_phy_type,
ext_phy_addr, ext_phy_addr,
MDIO_AN_DEVAD, MDIO_AN_DEVAD,
MDIO_AN_REG_CL37_CL73, 0x040c); MDIO_AN_REG_CL37_FC_LD, 0x040c);
/* Add support for CL37 (passive mode) II */ /* Add support for CL37 (passive mode) II */
bnx2x_cl45_write(bp, params->port, bnx2x_cl45_write(bp, params->port,
ext_phy_type, ext_phy_type,
ext_phy_addr, ext_phy_addr,
MDIO_AN_DEVAD, MDIO_AN_DEVAD,
MDIO_AN_REG_CL37_FD, 0x20); MDIO_AN_REG_CL37_FC_LD, 0x20);
/* Add support for CL37 (passive mode) III */ /* Add support for CL37 (passive mode) III */
bnx2x_cl45_write(bp, params->port, bnx2x_cl45_write(bp, params->port,
ext_phy_type, ext_phy_type,
...@@ -2785,7 +2809,7 @@ static u8 bnx2x_ext_phy_init(struct link_params *params, struct link_vars *vars) ...@@ -2785,7 +2809,7 @@ static u8 bnx2x_ext_phy_init(struct link_params *params, struct link_vars *vars)
static u8 bnx2x_ext_phy_is_link_up(struct link_params *params, static u8 bnx2x_ext_phy_is_link_up(struct link_params *params,
struct link_vars *vars) struct link_vars *vars)
{ {
struct bnx2x *bp = params->bp; struct bnx2x *bp = params->bp;
u32 ext_phy_type; u32 ext_phy_type;
...@@ -2826,6 +2850,8 @@ static u8 bnx2x_ext_phy_is_link_up(struct link_params *params, ...@@ -2826,6 +2850,8 @@ static u8 bnx2x_ext_phy_is_link_up(struct link_params *params,
MDIO_PMA_REG_RX_SD, &rx_sd); MDIO_PMA_REG_RX_SD, &rx_sd);
DP(NETIF_MSG_LINK, "8705 rx_sd 0x%x\n", rx_sd); DP(NETIF_MSG_LINK, "8705 rx_sd 0x%x\n", rx_sd);
ext_phy_link_up = (rx_sd & 0x1); ext_phy_link_up = (rx_sd & 0x1);
if (ext_phy_link_up)
vars->line_speed = SPEED_10000;
break; break;
case PORT_HW_CFG_XGXS_EXT_PHY_TYPE_BCM8706: case PORT_HW_CFG_XGXS_EXT_PHY_TYPE_BCM8706:
...@@ -3734,7 +3760,7 @@ static u8 bnx2x_link_initialize(struct link_params *params, ...@@ -3734,7 +3760,7 @@ static u8 bnx2x_link_initialize(struct link_params *params,
req_line_speed*/ req_line_speed*/
vars->line_speed = params->req_line_speed; vars->line_speed = params->req_line_speed;
bnx2x_set_ieee_aneg_advertisment(params, &vars->ieee_fc); bnx2x_calc_ieee_aneg_adv(params, &vars->ieee_fc);
/* init ext phy and enable link state int */ /* init ext phy and enable link state int */
non_ext_phy = ((ext_phy_type == PORT_HW_CFG_XGXS_EXT_PHY_TYPE_DIRECT) || non_ext_phy = ((ext_phy_type == PORT_HW_CFG_XGXS_EXT_PHY_TYPE_DIRECT) ||
......
...@@ -58,11 +58,13 @@ struct link_params { ...@@ -58,11 +58,13 @@ struct link_params {
u16 req_duplex; u16 req_duplex;
u16 req_flow_ctrl; u16 req_flow_ctrl;
u16 req_fc_auto_adv; /* Should be set to TX / BOTH when
req_flow_ctrl is set to AUTO */
u16 req_line_speed; /* Also determine AutoNeg */ u16 req_line_speed; /* Also determine AutoNeg */
/* Device parameters */ /* Device parameters */
u8 mac_addr[6]; u8 mac_addr[6];
u16 mtu;
/* shmem parameters */ /* shmem parameters */
......
...@@ -1946,7 +1946,14 @@ static u8 bnx2x_initial_phy_init(struct bnx2x *bp) ...@@ -1946,7 +1946,14 @@ static u8 bnx2x_initial_phy_init(struct bnx2x *bp)
u8 rc; u8 rc;
/* Initialize link parameters structure variables */ /* Initialize link parameters structure variables */
bp->link_params.mtu = bp->dev->mtu; /* It is recommended to turn off RX FC for jumbo frames
for better performance */
if (IS_E1HMF(bp))
bp->link_params.req_fc_auto_adv = FLOW_CTRL_BOTH;
else if (bp->dev->mtu > 5000)
bp->link_params.req_fc_auto_adv = FLOW_CTRL_TX;
else
bp->link_params.req_fc_auto_adv = FLOW_CTRL_BOTH;
bnx2x_acquire_phy_lock(bp); bnx2x_acquire_phy_lock(bp);
rc = bnx2x_phy_init(&bp->link_params, &bp->link_vars); rc = bnx2x_phy_init(&bp->link_params, &bp->link_vars);
......
...@@ -4966,6 +4966,7 @@ ...@@ -4966,6 +4966,7 @@
#define EMAC_RX_MODE_PROMISCUOUS (1L<<8) #define EMAC_RX_MODE_PROMISCUOUS (1L<<8)
#define EMAC_RX_MTU_SIZE_JUMBO_ENA (1L<<31) #define EMAC_RX_MTU_SIZE_JUMBO_ENA (1L<<31)
#define EMAC_TX_MODE_EXT_PAUSE_EN (1L<<3) #define EMAC_TX_MODE_EXT_PAUSE_EN (1L<<3)
#define EMAC_TX_MODE_FLOW_EN (1L<<4)
#define MISC_REGISTERS_GPIO_0 0 #define MISC_REGISTERS_GPIO_0 0
#define MISC_REGISTERS_GPIO_1 1 #define MISC_REGISTERS_GPIO_1 1
#define MISC_REGISTERS_GPIO_2 2 #define MISC_REGISTERS_GPIO_2 2
...@@ -5615,7 +5616,8 @@ Theotherbitsarereservedandshouldbezero*/ ...@@ -5615,7 +5616,8 @@ Theotherbitsarereservedandshouldbezero*/
#define MDIO_AN_REG_LINK_STATUS 0x8304 #define MDIO_AN_REG_LINK_STATUS 0x8304
#define MDIO_AN_REG_CL37_CL73 0x8370 #define MDIO_AN_REG_CL37_CL73 0x8370
#define MDIO_AN_REG_CL37_AN 0xffe0 #define MDIO_AN_REG_CL37_AN 0xffe0
#define MDIO_AN_REG_CL37_FD 0xffe4 #define MDIO_AN_REG_CL37_FC_LD 0xffe4
#define MDIO_AN_REG_CL37_FC_LP 0xffe5
#define IGU_FUNC_BASE 0x0400 #define IGU_FUNC_BASE 0x0400
......
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