Commit 62b29a5d authored by Yaniv Rosner's avatar Yaniv Rosner Committed by David S. Miller

bnx2x: Adjust alignment of split PHY functions

In previous patch, common PHY functions were split in simple way to
allow easy diff. This patch comes to align code in the new functions.
In addition, the non-production BCM8072 PHY was removed.
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 b7737c9b
...@@ -2133,30 +2133,14 @@ static void bnx2x_save_8481_spirom_version(struct bnx2x_phy *phy, ...@@ -2133,30 +2133,14 @@ static void bnx2x_save_8481_spirom_version(struct bnx2x_phy *phy,
/* For the 32 bits registers in 848xx, access via MDIO2ARM interface.*/ /* For the 32 bits registers in 848xx, access via MDIO2ARM interface.*/
/* (1) set register 0xc200_0014(SPI_BRIDGE_CTRL_2) to 0x03000000 */ /* (1) set register 0xc200_0014(SPI_BRIDGE_CTRL_2) to 0x03000000 */
bnx2x_cl45_write(bp, phy, MDIO_PMA_DEVAD, bnx2x_cl45_write(bp, phy, MDIO_PMA_DEVAD, 0xA819, 0x0014);
0xA819, 0x0014); bnx2x_cl45_write(bp, phy, MDIO_PMA_DEVAD, 0xA81A, 0xc200);
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy, MDIO_PMA_DEVAD, 0xA81B, 0x0000);
MDIO_PMA_DEVAD, bnx2x_cl45_write(bp, phy, MDIO_PMA_DEVAD, 0xA81C, 0x0300);
0xA81A, bnx2x_cl45_write(bp, phy, MDIO_PMA_DEVAD, 0xA817, 0x0009);
0xc200);
bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD,
0xA81B,
0x0000);
bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD,
0xA81C,
0x0300);
bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD,
0xA817,
0x0009);
for (cnt = 0; cnt < 100; cnt++) { for (cnt = 0; cnt < 100; cnt++) {
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy, MDIO_PMA_DEVAD, 0xA818, &val);
MDIO_PMA_DEVAD,
0xA818,
&val);
if (val & 1) if (val & 1)
break; break;
udelay(5); udelay(5);
...@@ -2170,86 +2154,29 @@ static void bnx2x_save_8481_spirom_version(struct bnx2x_phy *phy, ...@@ -2170,86 +2154,29 @@ static void bnx2x_save_8481_spirom_version(struct bnx2x_phy *phy,
/* 2) read register 0xc200_0000 (SPI_FW_STATUS) */ /* 2) read register 0xc200_0000 (SPI_FW_STATUS) */
bnx2x_cl45_write(bp, phy, MDIO_PMA_DEVAD, bnx2x_cl45_write(bp, phy, MDIO_PMA_DEVAD, 0xA819, 0x0000);
0xA819, 0x0000); bnx2x_cl45_write(bp, phy, MDIO_PMA_DEVAD, 0xA81A, 0xc200);
bnx2x_cl45_write(bp, phy, MDIO_PMA_DEVAD, bnx2x_cl45_write(bp, phy, MDIO_PMA_DEVAD, 0xA817, 0x000A);
0xA81A, 0xc200);
bnx2x_cl45_write(bp, phy, MDIO_PMA_DEVAD,
0xA817, 0x000A);
for (cnt = 0; cnt < 100; cnt++) { for (cnt = 0; cnt < 100; cnt++) {
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy, MDIO_PMA_DEVAD, 0xA818, &val);
MDIO_PMA_DEVAD,
0xA818,
&val);
if (val & 1) if (val & 1)
break; break;
udelay(5); udelay(5);
} }
if (cnt == 100) { if (cnt == 100) {
DP(NETIF_MSG_LINK, "Unable to read 8481 phy fw version(2)\n"); DP(NETIF_MSG_LINK, "Unable to read 848xx phy fw version(2)\n");
bnx2x_save_spirom_version(bp, params->port, bnx2x_save_spirom_version(bp, params->port, 0,
shmem_base, 0); phy->ver_addr);
return; return;
} }
/* lower 16 bits of the register SPI_FW_STATUS */ /* lower 16 bits of the register SPI_FW_STATUS */
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy, MDIO_PMA_DEVAD, 0xA81B, &fw_ver1);
MDIO_PMA_DEVAD,
0xA81B,
&fw_ver1);
/* upper 16 bits of register SPI_FW_STATUS */ /* upper 16 bits of register SPI_FW_STATUS */
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy, MDIO_PMA_DEVAD, 0xA81C, &fw_ver2);
MDIO_PMA_DEVAD,
0xA81C,
&fw_ver2);
bnx2x_save_spirom_version(bp, params->port, bnx2x_save_spirom_version(bp, params->port, (fw_ver2<<16) | fw_ver1,
shmem_base, (fw_ver2<<16) | fw_ver1); phy->ver_addr);
}
static void bnx2x_bcm8072_external_rom_boot(struct bnx2x_phy *phy,
struct link_params *params)
{
struct bnx2x *bp = params->bp;
u8 port = params->port;
/* Need to wait 200ms after reset */
msleep(200);
/* Boot port from external ROM
* Set ser_boot_ctl bit in the MISC_CTRL1 register
*/
bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD,
MDIO_PMA_REG_MISC_CTRL1, 0x0001);
/* Reset internal microprocessor */
bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD,
MDIO_PMA_REG_GEN_CTRL,
MDIO_PMA_REG_GEN_CTRL_ROM_RESET_INTERNAL_MP);
/* set micro reset = 0 */
bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD,
MDIO_PMA_REG_GEN_CTRL,
MDIO_PMA_REG_GEN_CTRL_ROM_MICRO_RESET);
/* Reset internal microprocessor */
bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD,
MDIO_PMA_REG_GEN_CTRL,
MDIO_PMA_REG_GEN_CTRL_ROM_RESET_INTERNAL_MP);
/* wait for 100ms for code download via SPI port */
msleep(100);
/* Clear ser_boot_ctl bit */
bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD,
MDIO_PMA_REG_MISC_CTRL1, 0x0000);
/* Wait 100ms */
msleep(100);
bnx2x_save_bcm_spirom_ver(bp, port,
phy,
params->shmem_base);
} }
static u8 bnx2x_8073_is_snr_needed(struct bnx2x *bp, struct bnx2x_phy *phy) static u8 bnx2x_8073_is_snr_needed(struct bnx2x *bp, struct bnx2x_phy *phy)
...@@ -2382,9 +2309,7 @@ static void bnx2x_8726_external_rom_boot(struct bnx2x_phy *phy, ...@@ -2382,9 +2309,7 @@ static void bnx2x_8726_external_rom_boot(struct bnx2x_phy *phy,
/* Micro controller re-boot */ /* Micro controller re-boot */
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_GEN_CTRL, 0x018B);
MDIO_PMA_REG_GEN_CTRL,
0x018B);
/* Set soft reset */ /* Set soft reset */
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
...@@ -2614,7 +2539,8 @@ static u8 bnx2x_get_edc_mode(struct bnx2x_phy *phy, ...@@ -2614,7 +2539,8 @@ static u8 bnx2x_get_edc_mode(struct bnx2x_phy *phy,
*edc_mode = EDC_MODE_LIMITING; *edc_mode = EDC_MODE_LIMITING;
/* First check for copper cable */ /* First check for copper cable */
if (bnx2x_read_sfp_module_eeprom(phy, params, if (bnx2x_read_sfp_module_eeprom(phy,
params,
SFP_EEPROM_CON_TYPE_ADDR, SFP_EEPROM_CON_TYPE_ADDR,
1, 1,
&val) != 0) { &val) != 0) {
...@@ -2629,7 +2555,8 @@ static u8 bnx2x_get_edc_mode(struct bnx2x_phy *phy, ...@@ -2629,7 +2555,8 @@ static u8 bnx2x_get_edc_mode(struct bnx2x_phy *phy,
/* Check if its active cable( includes SFP+ module) /* Check if its active cable( includes SFP+ module)
of passive cable*/ of passive cable*/
if (bnx2x_read_sfp_module_eeprom(phy, params, if (bnx2x_read_sfp_module_eeprom(phy,
params,
SFP_EEPROM_FC_TX_TECH_ADDR, SFP_EEPROM_FC_TX_TECH_ADDR,
1, 1,
&copper_module_type) != &copper_module_type) !=
...@@ -2669,7 +2596,8 @@ static u8 bnx2x_get_edc_mode(struct bnx2x_phy *phy, ...@@ -2669,7 +2596,8 @@ static u8 bnx2x_get_edc_mode(struct bnx2x_phy *phy,
if (check_limiting_mode) { if (check_limiting_mode) {
u8 options[SFP_EEPROM_OPTIONS_SIZE]; u8 options[SFP_EEPROM_OPTIONS_SIZE];
if (bnx2x_read_sfp_module_eeprom(phy, params, if (bnx2x_read_sfp_module_eeprom(phy,
params,
SFP_EEPROM_OPTIONS_ADDR, SFP_EEPROM_OPTIONS_ADDR,
SFP_EEPROM_OPTIONS_SIZE, SFP_EEPROM_OPTIONS_SIZE,
options) != 0) { options) != 0) {
...@@ -2720,14 +2648,16 @@ static u8 bnx2x_verify_sfp_module(struct bnx2x_phy *phy, ...@@ -2720,14 +2648,16 @@ static u8 bnx2x_verify_sfp_module(struct bnx2x_phy *phy,
} }
/* format the warning message */ /* format the warning message */
if (bnx2x_read_sfp_module_eeprom(phy, params, if (bnx2x_read_sfp_module_eeprom(phy,
params,
SFP_EEPROM_VENDOR_NAME_ADDR, SFP_EEPROM_VENDOR_NAME_ADDR,
SFP_EEPROM_VENDOR_NAME_SIZE, SFP_EEPROM_VENDOR_NAME_SIZE,
(u8 *)vendor_name)) (u8 *)vendor_name))
vendor_name[0] = '\0'; vendor_name[0] = '\0';
else else
vendor_name[SFP_EEPROM_VENDOR_NAME_SIZE] = '\0'; vendor_name[SFP_EEPROM_VENDOR_NAME_SIZE] = '\0';
if (bnx2x_read_sfp_module_eeprom(phy, params, if (bnx2x_read_sfp_module_eeprom(phy,
params,
SFP_EEPROM_PART_NO_ADDR, SFP_EEPROM_PART_NO_ADDR,
SFP_EEPROM_PART_NO_SIZE, SFP_EEPROM_PART_NO_SIZE,
(u8 *)vendor_pn)) (u8 *)vendor_pn))
...@@ -2848,7 +2778,6 @@ static u8 bnx2x_wait_for_sfp_module_initialized(struct bnx2x_phy *phy, ...@@ -2848,7 +2778,6 @@ static u8 bnx2x_wait_for_sfp_module_initialized(struct bnx2x_phy *phy,
} }
static void bnx2x_8727_power_module(struct bnx2x *bp, static void bnx2x_8727_power_module(struct bnx2x *bp,
struct link_params *params,
struct bnx2x_phy *phy, struct bnx2x_phy *phy,
u8 is_power_up) { u8 is_power_up) {
/* Make sure GPIOs are not using for LED mode */ /* Make sure GPIOs are not using for LED mode */
...@@ -2916,7 +2845,7 @@ static u8 bnx2x_sfp_module_detection(struct bnx2x_phy *phy, ...@@ -2916,7 +2845,7 @@ static u8 bnx2x_sfp_module_detection(struct bnx2x_phy *phy,
PORT_FEAT_CFG_OPT_MDL_ENFRCMNT_POWER_DOWN)) { PORT_FEAT_CFG_OPT_MDL_ENFRCMNT_POWER_DOWN)) {
/* Shutdown SFP+ module */ /* Shutdown SFP+ module */
DP(NETIF_MSG_LINK, "Shutdown SFP+ module!!\n"); DP(NETIF_MSG_LINK, "Shutdown SFP+ module!!\n");
bnx2x_8727_power_module(bp, params, phy, 0); bnx2x_8727_power_module(bp, phy, 0);
return rc; return rc;
} }
} else { } else {
...@@ -2929,7 +2858,7 @@ static u8 bnx2x_sfp_module_detection(struct bnx2x_phy *phy, ...@@ -2929,7 +2858,7 @@ static u8 bnx2x_sfp_module_detection(struct bnx2x_phy *phy,
/* power up the SFP module */ /* power up the SFP module */
if (phy->type == PORT_HW_CFG_XGXS_EXT_PHY_TYPE_BCM8727) if (phy->type == PORT_HW_CFG_XGXS_EXT_PHY_TYPE_BCM8727)
bnx2x_8727_power_module(bp, params, phy, 1); bnx2x_8727_power_module(bp, phy, 1);
/* Check and set limiting mode / LRM mode on 8726. /* Check and set limiting mode / LRM mode on 8726.
On 8727 it is done automatically */ On 8727 it is done automatically */
...@@ -2998,21 +2927,13 @@ static void bnx2x_807x_force_10G(struct bnx2x *bp, struct bnx2x_phy *phy) ...@@ -2998,21 +2927,13 @@ static void bnx2x_807x_force_10G(struct bnx2x *bp, struct bnx2x_phy *phy)
{ {
/* Force KR or KX */ /* Force KR or KX */
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_CTRL, 0x2040);
MDIO_PMA_REG_CTRL,
0x2040);
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_10G_CTRL2, 0x000b);
MDIO_PMA_REG_10G_CTRL2,
0x000b);
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_BCM_CTRL, 0x0000);
MDIO_PMA_REG_BCM_CTRL,
0x0000);
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_AN_DEVAD, MDIO_AN_DEVAD, MDIO_AN_REG_CTRL, 0x0000);
MDIO_AN_REG_CTRL,
0x0000);
} }
static void bnx2x_8073_set_xaui_low_power_mode(struct bnx2x *bp, static void bnx2x_8073_set_xaui_low_power_mode(struct bnx2x *bp,
...@@ -3020,8 +2941,7 @@ static void bnx2x_8073_set_xaui_low_power_mode(struct bnx2x *bp, ...@@ -3020,8 +2941,7 @@ static void bnx2x_8073_set_xaui_low_power_mode(struct bnx2x *bp,
{ {
u16 val; u16 val;
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_8073_CHIP_REV, &val);
MDIO_PMA_REG_8073_CHIP_REV, &val);
if (val == 0) { if (val == 0) {
/* Mustn't set low power mode in 8073 A0 */ /* Mustn't set low power mode in 8073 A0 */
...@@ -3030,47 +2950,32 @@ static void bnx2x_8073_set_xaui_low_power_mode(struct bnx2x *bp, ...@@ -3030,47 +2950,32 @@ static void bnx2x_8073_set_xaui_low_power_mode(struct bnx2x *bp,
/* Disable PLL sequencer (use read-modify-write to clear bit 13) */ /* Disable PLL sequencer (use read-modify-write to clear bit 13) */
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_XS_DEVAD, MDIO_XS_DEVAD, MDIO_XS_PLL_SEQUENCER, &val);
MDIO_XS_PLL_SEQUENCER, &val);
val &= ~(1<<13); val &= ~(1<<13);
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_XS_DEVAD, MDIO_XS_PLL_SEQUENCER, val); MDIO_XS_DEVAD, MDIO_XS_PLL_SEQUENCER, val);
/* PLL controls */ /* PLL controls */
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy, MDIO_XS_DEVAD, 0x805E, 0x1077);
MDIO_XS_DEVAD, 0x805E, 0x1077); bnx2x_cl45_write(bp, phy, MDIO_XS_DEVAD, 0x805D, 0x0000);
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy, MDIO_XS_DEVAD, 0x805C, 0x030B);
MDIO_XS_DEVAD, 0x805D, 0x0000); bnx2x_cl45_write(bp, phy, MDIO_XS_DEVAD, 0x805B, 0x1240);
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy, MDIO_XS_DEVAD, 0x805A, 0x2490);
MDIO_XS_DEVAD, 0x805C, 0x030B);
bnx2x_cl45_write(bp, phy,
MDIO_XS_DEVAD, 0x805B, 0x1240);
bnx2x_cl45_write(bp, phy,
MDIO_XS_DEVAD, 0x805A, 0x2490);
/* Tx Controls */ /* Tx Controls */
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy, MDIO_XS_DEVAD, 0x80A7, 0x0C74);
MDIO_XS_DEVAD, 0x80A7, 0x0C74); bnx2x_cl45_write(bp, phy, MDIO_XS_DEVAD, 0x80A6, 0x9041);
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy, MDIO_XS_DEVAD, 0x80A5, 0x4640);
MDIO_XS_DEVAD, 0x80A6, 0x9041);
bnx2x_cl45_write(bp, phy,
MDIO_XS_DEVAD, 0x80A5, 0x4640);
/* Rx Controls */ /* Rx Controls */
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy, MDIO_XS_DEVAD, 0x80FE, 0x01C4);
MDIO_XS_DEVAD, 0x80FE, 0x01C4); bnx2x_cl45_write(bp, phy, MDIO_XS_DEVAD, 0x80FD, 0x9249);
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy, MDIO_XS_DEVAD, 0x80FC, 0x2015);
MDIO_XS_DEVAD, 0x80FD, 0x9249);
bnx2x_cl45_write(bp, phy,
MDIO_XS_DEVAD, 0x80FC, 0x2015);
/* Enable PLL sequencer (use read-modify-write to set bit 13) */ /* Enable PLL sequencer (use read-modify-write to set bit 13) */
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy, MDIO_XS_DEVAD, MDIO_XS_PLL_SEQUENCER, &val);
MDIO_XS_DEVAD,
MDIO_XS_PLL_SEQUENCER, &val);
val |= (1<<13); val |= (1<<13);
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy, MDIO_XS_DEVAD, MDIO_XS_PLL_SEQUENCER, val);
MDIO_XS_DEVAD, MDIO_XS_PLL_SEQUENCER, val);
} }
static void bnx2x_8073_set_pause_cl37(struct link_params *params, static void bnx2x_8073_set_pause_cl37(struct link_params *params,
...@@ -3080,8 +2985,7 @@ static void bnx2x_8073_set_pause_cl37(struct link_params *params, ...@@ -3080,8 +2985,7 @@ static void bnx2x_8073_set_pause_cl37(struct link_params *params,
u16 cl37_val; u16 cl37_val;
struct bnx2x *bp = params->bp; struct bnx2x *bp = params->bp;
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_AN_DEVAD, MDIO_AN_DEVAD, MDIO_AN_REG_CL37_FC_LD, &cl37_val);
MDIO_AN_REG_CL37_FC_LD, &cl37_val);
cl37_val &= ~MDIO_COMBO_IEEE0_AUTO_NEG_ADV_PAUSE_BOTH; cl37_val &= ~MDIO_COMBO_IEEE0_AUTO_NEG_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. */
...@@ -3105,8 +3009,7 @@ static void bnx2x_8073_set_pause_cl37(struct link_params *params, ...@@ -3105,8 +3009,7 @@ static void bnx2x_8073_set_pause_cl37(struct link_params *params,
"Ext phy AN advertize cl37 0x%x\n", cl37_val); "Ext phy AN advertize cl37 0x%x\n", cl37_val);
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_AN_DEVAD, MDIO_AN_DEVAD, MDIO_AN_REG_CL37_FC_LD, cl37_val);
MDIO_AN_REG_CL37_FC_LD, cl37_val);
msleep(500); msleep(500);
} }
...@@ -3117,9 +3020,7 @@ static void bnx2x_ext_phy_set_pause(struct link_params *params, ...@@ -3117,9 +3020,7 @@ static void bnx2x_ext_phy_set_pause(struct link_params *params,
u16 val; u16 val;
struct bnx2x *bp = params->bp; struct bnx2x *bp = params->bp;
/* read modify write pause advertizing */ /* read modify write pause advertizing */
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy, MDIO_AN_DEVAD, MDIO_AN_REG_ADV_PAUSE, &val);
MDIO_AN_DEVAD,
MDIO_AN_REG_ADV_PAUSE, &val);
val &= ~MDIO_AN_REG_ADV_PAUSE_BOTH; val &= ~MDIO_AN_REG_ADV_PAUSE_BOTH;
...@@ -3133,14 +3034,10 @@ static void bnx2x_ext_phy_set_pause(struct link_params *params, ...@@ -3133,14 +3034,10 @@ static void bnx2x_ext_phy_set_pause(struct link_params *params,
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) { MDIO_COMBO_IEEE0_AUTO_NEG_ADV_PAUSE_BOTH) {
val |= val |= MDIO_AN_REG_ADV_PAUSE_PAUSE;
MDIO_AN_REG_ADV_PAUSE_PAUSE;
} }
DP(NETIF_MSG_LINK, DP(NETIF_MSG_LINK, "Ext phy AN advertize 0x%x\n", val);
"Ext phy AN advertize 0x%x\n", val); bnx2x_cl45_write(bp, phy, MDIO_AN_DEVAD, MDIO_AN_REG_ADV_PAUSE, val);
bnx2x_cl45_write(bp, phy,
MDIO_AN_DEVAD,
MDIO_AN_REG_ADV_PAUSE, val);
} }
static void bnx2x_set_preemphasis(struct bnx2x_phy *phy, static void bnx2x_set_preemphasis(struct bnx2x_phy *phy,
...@@ -3260,15 +3157,13 @@ static u16 bnx2x_wait_reset_complete(struct bnx2x *bp, ...@@ -3260,15 +3157,13 @@ static u16 bnx2x_wait_reset_complete(struct bnx2x *bp,
/* Wait for soft reset to get cleared upto 1 sec */ /* Wait for soft reset to get cleared upto 1 sec */
for (cnt = 0; cnt < 1000; cnt++) { for (cnt = 0; cnt < 1000; cnt++) {
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_CTRL, &ctrl);
MDIO_PMA_REG_CTRL, &ctrl);
if (!(ctrl & (1<<15))) if (!(ctrl & (1<<15)))
break; break;
msleep(1); msleep(1);
} }
DP(NETIF_MSG_LINK, "control reg 0x%x (after %d ms)\n", DP(NETIF_MSG_LINK, "control reg 0x%x (after %d ms)\n", ctrl, cnt);
ctrl, cnt); return cnt;
return 0;
} }
static u8 bnx2x_8705_config_init(struct bnx2x_phy *phy, static u8 bnx2x_8705_config_init(struct bnx2x_phy *phy,
...@@ -3284,28 +3179,17 @@ static u8 bnx2x_8705_config_init(struct bnx2x_phy *phy, ...@@ -3284,28 +3179,17 @@ static u8 bnx2x_8705_config_init(struct bnx2x_phy *phy,
bnx2x_ext_phy_hw_reset(bp, params->port); bnx2x_ext_phy_hw_reset(bp, params->port);
bnx2x_cl45_write(bp, phy, MDIO_PMA_DEVAD, MDIO_PMA_REG_CTRL, 0xa040); bnx2x_cl45_write(bp, phy, MDIO_PMA_DEVAD, MDIO_PMA_REG_CTRL, 0xa040);
bnx2x_wait_reset_complete(bp, phy); bnx2x_wait_reset_complete(bp, phy);
DP(NETIF_MSG_LINK, "XGXS 8705\n");
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_MISC_CTRL, 0x8288);
MDIO_PMA_REG_MISC_CTRL,
0x8288);
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_PHY_IDENTIFIER, 0x7fbf);
MDIO_PMA_REG_PHY_IDENTIFIER,
0x7fbf);
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_CMU_PLL_BYPASS, 0x0100);
MDIO_PMA_REG_CMU_PLL_BYPASS,
0x0100);
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_WIS_DEVAD, MDIO_WIS_DEVAD, MDIO_WIS_REG_LASI_CNTL, 0x1);
MDIO_WIS_REG_LASI_CNTL, 0x1);
/* BCM8705 doesn't have microcode, hence the 0 */ /* BCM8705 doesn't have microcode, hence the 0 */
bnx2x_save_spirom_version(bp, params->port, bnx2x_save_spirom_version(bp, params->port, params->shmem_base, 0);
params->shmem_base, 0);
return 0; return 0;
} }
...@@ -3322,16 +3206,16 @@ static u8 bnx2x_8706_config_init(struct bnx2x_phy *phy, ...@@ -3322,16 +3206,16 @@ static u8 bnx2x_8706_config_init(struct bnx2x_phy *phy,
bnx2x_cl45_write(bp, phy, MDIO_PMA_DEVAD, MDIO_PMA_REG_CTRL, 0xa040); bnx2x_cl45_write(bp, phy, MDIO_PMA_DEVAD, MDIO_PMA_REG_CTRL, 0xa040);
bnx2x_wait_reset_complete(bp, phy); bnx2x_wait_reset_complete(bp, phy);
/* Wait until fw is loaded */ /* Wait until fw is loaded */
for (cnt = 0; cnt < 100; cnt++) { for (cnt = 0; cnt < 100; cnt++) {
bnx2x_cl45_read(bp, phy, MDIO_PMA_DEVAD, bnx2x_cl45_read(bp, phy,
MDIO_PMA_REG_ROM_VER1, &val); MDIO_PMA_DEVAD, MDIO_PMA_REG_ROM_VER1, &val);
if (val) if (val)
break; break;
msleep(10); msleep(10);
} }
DP(NETIF_MSG_LINK, "XGXS 8706 is initialized " DP(NETIF_MSG_LINK, "XGXS 8706 is initialized after %d ms\n", cnt);
"after %d ms\n", cnt);
if ((params->feature_config_flags & if ((params->feature_config_flags &
FEATURE_CONFIG_OVERRIDE_PREEMPHASIS_ENABLED)) { FEATURE_CONFIG_OVERRIDE_PREEMPHASIS_ENABLED)) {
u8 i; u8 i;
...@@ -3340,76 +3224,54 @@ static u8 bnx2x_8706_config_init(struct bnx2x_phy *phy, ...@@ -3340,76 +3224,54 @@ static u8 bnx2x_8706_config_init(struct bnx2x_phy *phy,
reg = MDIO_XS_8706_REG_BANK_RX0 + reg = MDIO_XS_8706_REG_BANK_RX0 +
i*(MDIO_XS_8706_REG_BANK_RX1 - i*(MDIO_XS_8706_REG_BANK_RX1 -
MDIO_XS_8706_REG_BANK_RX0); MDIO_XS_8706_REG_BANK_RX0);
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy, MDIO_XS_DEVAD, reg, &val);
MDIO_XS_DEVAD,
reg, &val);
/* Clear first 3 bits of the control */ /* Clear first 3 bits of the control */
val &= ~0x7; val &= ~0x7;
/* Set control bits according to /* Set control bits according to configuration */
configuation */ val |= (phy->rx_preemphasis[i] & 0x7);
val |= (phy->rx_preemphasis[i] & DP(NETIF_MSG_LINK, "Setting RX Equalizer to BCM8706"
0x7); " reg 0x%x <-- val 0x%x\n", reg, val);
DP(NETIF_MSG_LINK, "Setting RX" bnx2x_cl45_write(bp, phy, MDIO_XS_DEVAD, reg, val);
"Equalizer to BCM8706 reg 0x%x"
" <-- val 0x%x\n", reg, val);
bnx2x_cl45_write(bp, phy,
MDIO_XS_DEVAD,
reg, val);
} }
} }
/* Force speed */ /* Force speed */
if (params->req_line_speed == SPEED_10000) { if (phy->req_line_speed == SPEED_10000) {
DP(NETIF_MSG_LINK, "XGXS 8706 force 10Gbps\n"); DP(NETIF_MSG_LINK, "XGXS 8706 force 10Gbps\n");
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD,
MDIO_PMA_REG_DIGITAL_CTRL, MDIO_PMA_REG_DIGITAL_CTRL, 0x400);
0x400); bnx2x_cl45_write(bp, phy,
bnx2x_cl45_write(bp, phy, MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_LASI_CTRL, 1);
MDIO_PMA_REG_LASI_CTRL, 1);
} else { } else {
/* Force 1Gbps using autoneg with 1G /* Force 1Gbps using autoneg with 1G advertisment */
advertisment */
/* Allow CL37 through CL73 */ /* Allow CL37 through CL73 */
DP(NETIF_MSG_LINK, "XGXS 8706 AutoNeg\n"); DP(NETIF_MSG_LINK, "XGXS 8706 AutoNeg\n");
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_AN_DEVAD, MDIO_AN_DEVAD, MDIO_AN_REG_CL37_CL73, 0x040c);
MDIO_AN_REG_CL37_CL73,
0x040c);
/* Enable Full-Duplex advertisment on CL37 */ /* Enable Full-Duplex advertisment on CL37 */
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_AN_DEVAD, MDIO_AN_DEVAD, MDIO_AN_REG_CL37_FC_LP, 0x0020);
MDIO_AN_REG_CL37_FC_LP,
0x0020);
/* Enable CL37 AN */ /* Enable CL37 AN */
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_AN_DEVAD, MDIO_AN_DEVAD, MDIO_AN_REG_CL37_AN, 0x1000);
MDIO_AN_REG_CL37_AN,
0x1000);
/* 1G support */ /* 1G support */
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_AN_DEVAD, MDIO_AN_DEVAD, MDIO_AN_REG_ADV, (1<<5));
MDIO_AN_REG_ADV, (1<<5));
/* Enable clause 73 AN */ /* Enable clause 73 AN */
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_AN_DEVAD, MDIO_AN_DEVAD, MDIO_AN_REG_CTRL, 0x1200);
MDIO_AN_REG_CTRL,
0x1200);
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_RX_ALARM_CTRL,
MDIO_PMA_REG_RX_ALARM_CTRL,
0x0400); 0x0400);
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_LASI_CTRL,
MDIO_PMA_REG_LASI_CTRL, 0x0004); 0x0004);
} }
bnx2x_save_bcm_spirom_ver(bp, params->port, bnx2x_save_bcm_spirom_ver(bp, params->port, phy, params->shmem_base);
phy,
params->shmem_base);
return 0; return 0;
} }
...@@ -3429,8 +3291,6 @@ static u8 bnx2x_8726_config_init(struct bnx2x_phy *phy, ...@@ -3429,8 +3291,6 @@ static u8 bnx2x_8726_config_init(struct bnx2x_phy *phy,
bnx2x_cl45_write(bp, phy, MDIO_PMA_DEVAD, MDIO_PMA_REG_CTRL, 1<<15); bnx2x_cl45_write(bp, phy, MDIO_PMA_DEVAD, MDIO_PMA_REG_CTRL, 1<<15);
bnx2x_wait_reset_complete(bp, phy); bnx2x_wait_reset_complete(bp, phy);
bnx2x_wait_reset_complete(bp, phy);
DP(NETIF_MSG_LINK, "Initializing BCM8726\n");
bnx2x_8726_external_rom_boot(phy, params); bnx2x_8726_external_rom_boot(phy, params);
/* Need to call module detected on initialization since /* Need to call module detected on initialization since
...@@ -3440,46 +3300,47 @@ static u8 bnx2x_8726_config_init(struct bnx2x_phy *phy, ...@@ -3440,46 +3300,47 @@ static u8 bnx2x_8726_config_init(struct bnx2x_phy *phy,
transmitter */ transmitter */
bnx2x_sfp_module_detection(phy, params); bnx2x_sfp_module_detection(phy, params);
/* Set Flow control */ if (phy->req_line_speed == SPEED_1000) {
bnx2x_ext_phy_set_pause(params, phy, vars);
if (params->req_line_speed == SPEED_1000) {
DP(NETIF_MSG_LINK, "Setting 1G force\n"); DP(NETIF_MSG_LINK, "Setting 1G force\n");
bnx2x_cl45_write(bp, phy, MDIO_PMA_DEVAD, bnx2x_cl45_write(bp, phy,
MDIO_PMA_REG_CTRL, 0x40); MDIO_PMA_DEVAD, MDIO_PMA_REG_CTRL, 0x40);
bnx2x_cl45_write(bp, phy, MDIO_PMA_DEVAD, bnx2x_cl45_write(bp, phy,
MDIO_PMA_REG_10G_CTRL2, 0xD); MDIO_PMA_DEVAD, MDIO_PMA_REG_10G_CTRL2, 0xD);
bnx2x_cl45_write(bp, phy, MDIO_PMA_DEVAD, bnx2x_cl45_write(bp, phy,
MDIO_PMA_REG_LASI_CTRL, 0x5); MDIO_PMA_DEVAD, MDIO_PMA_REG_LASI_CTRL, 0x5);
bnx2x_cl45_write(bp, phy, MDIO_PMA_DEVAD, bnx2x_cl45_write(bp, phy,
MDIO_PMA_REG_RX_ALARM_CTRL, MDIO_PMA_DEVAD, MDIO_PMA_REG_RX_ALARM_CTRL,
0x400); 0x400);
} else if ((params->req_line_speed == } else if ((phy->req_line_speed == SPEED_AUTO_NEG) &&
SPEED_AUTO_NEG) && (phy->speed_cap_mask &
((params->speed_cap_mask & PORT_HW_CFG_SPEED_CAPABILITY_D0_1G) &&
PORT_HW_CFG_SPEED_CAPABILITY_D0_1G))) { ((phy->speed_cap_mask &
PORT_HW_CFG_SPEED_CAPABILITY_D0_10G) !=
PORT_HW_CFG_SPEED_CAPABILITY_D0_10G)) {
DP(NETIF_MSG_LINK, "Setting 1G clause37\n"); DP(NETIF_MSG_LINK, "Setting 1G clause37\n");
bnx2x_cl45_write(bp, phy, MDIO_AN_DEVAD, /* Set Flow control */
MDIO_AN_REG_ADV, 0x20); bnx2x_ext_phy_set_pause(params, phy, vars);
bnx2x_cl45_write(bp, phy, MDIO_AN_DEVAD, bnx2x_cl45_write(bp, phy,
MDIO_AN_REG_CL37_CL73, 0x040c); MDIO_AN_DEVAD, MDIO_AN_REG_ADV, 0x20);
bnx2x_cl45_write(bp, phy, MDIO_AN_DEVAD, bnx2x_cl45_write(bp, phy,
MDIO_AN_REG_CL37_FC_LD, 0x0020); MDIO_AN_DEVAD, MDIO_AN_REG_CL37_CL73, 0x040c);
bnx2x_cl45_write(bp, phy, MDIO_AN_DEVAD, bnx2x_cl45_write(bp, phy,
MDIO_AN_REG_CL37_AN, 0x1000); MDIO_AN_DEVAD, MDIO_AN_REG_CL37_FC_LD, 0x0020);
bnx2x_cl45_write(bp, phy, MDIO_AN_DEVAD, bnx2x_cl45_write(bp, phy,
MDIO_AN_REG_CTRL, 0x1200); MDIO_AN_DEVAD, MDIO_AN_REG_CL37_AN, 0x1000);
bnx2x_cl45_write(bp, phy,
MDIO_AN_DEVAD, MDIO_AN_REG_CTRL, 0x1200);
/* Enable RX-ALARM control to receive /* Enable RX-ALARM control to receive
interrupt for 1G speed change */ interrupt for 1G speed change */
bnx2x_cl45_write(bp, phy, MDIO_PMA_DEVAD, bnx2x_cl45_write(bp, phy,
MDIO_PMA_REG_LASI_CTRL, 0x4); MDIO_PMA_DEVAD, MDIO_PMA_REG_LASI_CTRL, 0x4);
bnx2x_cl45_write(bp, phy, MDIO_PMA_DEVAD, bnx2x_cl45_write(bp, phy,
MDIO_PMA_REG_RX_ALARM_CTRL, MDIO_PMA_DEVAD, MDIO_PMA_REG_RX_ALARM_CTRL,
0x400); 0x400);
} else { /* Default 10G. Set only LASI control */ } else { /* Default 10G. Set only LASI control */
bnx2x_cl45_write(bp, phy, MDIO_PMA_DEVAD, bnx2x_cl45_write(bp, phy,
MDIO_PMA_REG_LASI_CTRL, 1); MDIO_PMA_DEVAD, MDIO_PMA_REG_LASI_CTRL, 1);
} }
/* Set TX PreEmphasis if needed */ /* Set TX PreEmphasis if needed */
...@@ -3503,12 +3364,12 @@ static u8 bnx2x_8726_config_init(struct bnx2x_phy *phy, ...@@ -3503,12 +3364,12 @@ static u8 bnx2x_8726_config_init(struct bnx2x_phy *phy,
} }
static u8 bnx2x_8072_8073_config_init(struct bnx2x_phy *phy, static u8 bnx2x_8073_config_init(struct bnx2x_phy *phy,
struct link_params *params, struct link_params *params,
struct link_vars *vars) struct link_vars *vars)
{ {
struct bnx2x *bp = params->bp; struct bnx2x *bp = params->bp;
u16 val = 0; u16 val = 0, tmp1;
u8 gpio_port; u8 gpio_port;
DP(NETIF_MSG_LINK, "Init 8073\n"); DP(NETIF_MSG_LINK, "Init 8073\n");
...@@ -3520,112 +3381,75 @@ static u8 bnx2x_8072_8073_config_init(struct bnx2x_phy *phy, ...@@ -3520,112 +3381,75 @@ static u8 bnx2x_8072_8073_config_init(struct bnx2x_phy *phy,
bnx2x_set_gpio(bp, MISC_REGISTERS_GPIO_1, bnx2x_set_gpio(bp, MISC_REGISTERS_GPIO_1,
MISC_REGISTERS_GPIO_OUTPUT_HIGH, gpio_port); MISC_REGISTERS_GPIO_OUTPUT_HIGH, gpio_port);
{
u16 tmp1;
u16 rx_alarm_ctrl_val;
u16 lasi_ctrl_val;
if (phy->type ==
PORT_HW_CFG_XGXS_EXT_PHY_TYPE_BCM8072) {
rx_alarm_ctrl_val = 0x400;
lasi_ctrl_val = 0x0004;
} else {
rx_alarm_ctrl_val = (1<<2);
lasi_ctrl_val = 0x0004;
}
/* enable LASI */ /* enable LASI */
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_RX_ALARM_CTRL, (1<<2));
MDIO_PMA_REG_RX_ALARM_CTRL,
rx_alarm_ctrl_val);
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_LASI_CTRL, 0x0004);
MDIO_PMA_REG_LASI_CTRL,
lasi_ctrl_val);
bnx2x_8073_set_pause_cl37(params, phy, vars); bnx2x_8073_set_pause_cl37(params, phy, vars);
if (phy->type ==
PORT_HW_CFG_XGXS_EXT_PHY_TYPE_BCM8072)
bnx2x_bcm8072_external_rom_boot(phy, params);
else
/* In case of 8073 with long xaui lines,
don't set the 8073 xaui low power*/
bnx2x_8073_set_xaui_low_power_mode(bp, phy); bnx2x_8073_set_xaui_low_power_mode(bp, phy);
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_M8051_MSGOUT_REG, &tmp1);
MDIO_PMA_REG_M8051_MSGOUT_REG,
&tmp1);
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_RX_ALARM, &tmp1);
MDIO_PMA_REG_RX_ALARM, &tmp1);
DP(NETIF_MSG_LINK, "Before rom RX_ALARM(port1):" DP(NETIF_MSG_LINK, "Before rom RX_ALARM(port1): 0x%x\n", tmp1);
"0x%x\n", tmp1);
/* Enable CL37 BAM */
bnx2x_cl45_read(bp, phy,
MDIO_AN_DEVAD,
MDIO_AN_REG_8073_BAM, &val);
bnx2x_cl45_write(bp, phy,
MDIO_AN_DEVAD,
MDIO_AN_REG_8073_BAM, val | 1);
/* If this is forced speed, set to KR or KX
* (all other are not supported)
*/
if (params->loopback_mode == LOOPBACK_EXT) { if (params->loopback_mode == LOOPBACK_EXT) {
bnx2x_807x_force_10G(bp, phy); bnx2x_807x_force_10G(bp, phy);
DP(NETIF_MSG_LINK, DP(NETIF_MSG_LINK, "Forced speed 10G on 807X\n");
"Forced speed 10G on 807X\n");
return 0; return 0;
} else { } else {
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_BCM_CTRL, 0x0002);
MDIO_PMA_REG_BCM_CTRL,
0x0002);
} }
if (params->req_line_speed != SPEED_AUTO_NEG) { if (phy->req_line_speed != SPEED_AUTO_NEG) {
if (params->req_line_speed == SPEED_10000) { if (phy->req_line_speed == SPEED_10000) {
val = (1<<7); val = (1<<7);
} else if (params->req_line_speed == } else if (phy->req_line_speed == SPEED_2500) {
SPEED_2500) {
val = (1<<5); val = (1<<5);
/* Note that 2.5G works only /* Note that 2.5G works only
when used with 1G advertisment */ when used with 1G advertisment */
} else } else
val = (1<<5); val = (1<<5);
} else { } else {
val = 0; val = 0;
if (params->speed_cap_mask & if (phy->speed_cap_mask &
PORT_HW_CFG_SPEED_CAPABILITY_D0_10G) PORT_HW_CFG_SPEED_CAPABILITY_D0_10G)
val |= (1<<7); val |= (1<<7);
/* Note that 2.5G works only when /* Note that 2.5G works only when
used with 1G advertisment */ used with 1G advertisment */
if (params->speed_cap_mask & if (phy->speed_cap_mask &
(PORT_HW_CFG_SPEED_CAPABILITY_D0_1G | (PORT_HW_CFG_SPEED_CAPABILITY_D0_1G |
PORT_HW_CFG_SPEED_CAPABILITY_D0_2_5G)) PORT_HW_CFG_SPEED_CAPABILITY_D0_2_5G))
val |= (1<<5); val |= (1<<5);
DP(NETIF_MSG_LINK, DP(NETIF_MSG_LINK, "807x autoneg val = 0x%x\n", val);
"807x autoneg val = 0x%x\n", val);
} }
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy, MDIO_AN_DEVAD, MDIO_AN_REG_ADV, val);
MDIO_AN_DEVAD, bnx2x_cl45_read(bp, phy, MDIO_AN_DEVAD, MDIO_AN_REG_8073_2_5G, &tmp1);
MDIO_AN_REG_ADV, val);
bnx2x_cl45_read(bp, phy, if (((phy->speed_cap_mask & PORT_HW_CFG_SPEED_CAPABILITY_D0_2_5G) &&
MDIO_AN_DEVAD, (phy->req_line_speed == SPEED_AUTO_NEG)) ||
MDIO_AN_REG_8073_2_5G, &tmp1); (phy->req_line_speed == SPEED_2500)) {
if (((params->speed_cap_mask &
PORT_HW_CFG_SPEED_CAPABILITY_D0_2_5G) &&
(params->req_line_speed ==
SPEED_AUTO_NEG)) ||
(params->req_line_speed ==
SPEED_2500)) {
u16 phy_ver; u16 phy_ver;
/* Allow 2.5G for A1 and above */ /* Allow 2.5G for A1 and above */
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_8073_CHIP_REV,
MDIO_PMA_REG_8073_CHIP_REV, &phy_ver); &phy_ver);
DP(NETIF_MSG_LINK, "Add 2.5G\n"); DP(NETIF_MSG_LINK, "Add 2.5G\n");
if (phy_ver > 0) if (phy_ver > 0)
tmp1 |= 1; tmp1 |= 1;
...@@ -3636,67 +3460,38 @@ static u8 bnx2x_8072_8073_config_init(struct bnx2x_phy *phy, ...@@ -3636,67 +3460,38 @@ static u8 bnx2x_8072_8073_config_init(struct bnx2x_phy *phy,
tmp1 &= 0xfffe; tmp1 &= 0xfffe;
} }
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy, MDIO_AN_DEVAD, MDIO_AN_REG_8073_2_5G, tmp1);
MDIO_AN_DEVAD,
MDIO_AN_REG_8073_2_5G, tmp1);
/* Add support for CL37 (passive mode) II */ /* Add support for CL37 (passive mode) II */
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy, MDIO_AN_DEVAD, MDIO_AN_REG_CL37_FC_LD, &tmp1);
MDIO_AN_DEVAD, bnx2x_cl45_write(bp, phy, MDIO_AN_DEVAD, MDIO_AN_REG_CL37_FC_LD,
MDIO_AN_REG_CL37_FC_LD, (tmp1 | ((phy->req_duplex == DUPLEX_FULL) ?
&tmp1);
bnx2x_cl45_write(bp, phy,
MDIO_AN_DEVAD,
MDIO_AN_REG_CL37_FC_LD, (tmp1 |
((params->req_duplex == DUPLEX_FULL) ?
0x20 : 0x40))); 0x20 : 0x40)));
/* Add support for CL37 (passive mode) III */ /* Add support for CL37 (passive mode) III */
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy, MDIO_AN_DEVAD, MDIO_AN_REG_CL37_AN, 0x1000);
MDIO_AN_DEVAD,
MDIO_AN_REG_CL37_AN, 0x1000);
if (phy->type ==
PORT_HW_CFG_XGXS_EXT_PHY_TYPE_BCM8073) {
/* The SNR will improve about 2db by changing /* The SNR will improve about 2db by changing
BW and FEE main tap. Rest commands are executed BW and FEE main tap. Rest commands are executed
after link is up*/ after link is up*/
/*Change FFE main cursor to 5 in EDC register*/
if (bnx2x_8073_is_snr_needed(bp, phy)) if (bnx2x_8073_is_snr_needed(bp, phy))
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_EDC_FFE_MAIN,
MDIO_PMA_REG_EDC_FFE_MAIN,
0xFB0C); 0xFB0C);
/* Enable FEC (Forware Error Correction) /* Enable FEC (Forware Error Correction) Request in the AN */
Request in the AN */ bnx2x_cl45_read(bp, phy, MDIO_AN_DEVAD, MDIO_AN_REG_ADV2, &tmp1);
bnx2x_cl45_read(bp, phy,
MDIO_AN_DEVAD,
MDIO_AN_REG_ADV2, &tmp1);
tmp1 |= (1<<15); tmp1 |= (1<<15);
bnx2x_cl45_write(bp, phy, MDIO_AN_DEVAD, MDIO_AN_REG_ADV2, tmp1);
bnx2x_cl45_write(bp, phy,
MDIO_AN_DEVAD,
MDIO_AN_REG_ADV2, tmp1);
}
bnx2x_ext_phy_set_pause(params, phy, vars); bnx2x_ext_phy_set_pause(params, phy, vars);
/* Restart autoneg */ /* Restart autoneg */
msleep(500); msleep(500);
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy, MDIO_AN_DEVAD, MDIO_AN_REG_CTRL, 0x1200);
MDIO_AN_DEVAD, DP(NETIF_MSG_LINK, "807x Autoneg Restart: Advertise 1G=%x, 10G=%x\n",
MDIO_AN_REG_CTRL, 0x1200); ((val & (1<<5)) > 0), ((val & (1<<7)) > 0));
DP(NETIF_MSG_LINK, "807x Autoneg Restart: "
"Advertise 1G=%x, 10G=%x\n",
((val & (1<<5)) > 0),
((val & (1<<7)) > 0));
return 0; return 0;
}
} }
static u8 bnx2x_8727_config_init(struct bnx2x_phy *phy, static u8 bnx2x_8727_config_init(struct bnx2x_phy *phy,
...@@ -3709,139 +3504,116 @@ static u8 bnx2x_8727_config_init(struct bnx2x_phy *phy, ...@@ -3709,139 +3504,116 @@ static u8 bnx2x_8727_config_init(struct bnx2x_phy *phy,
struct bnx2x *bp = params->bp; struct bnx2x *bp = params->bp;
/* Enable PMD link, MOD_ABS_FLT, and 1G link alarm */ /* Enable PMD link, MOD_ABS_FLT, and 1G link alarm */
/* Enable PMD link, MOD_ABS_FLT, and 1G link alarm */ bnx2x_wait_reset_complete(bp, phy);
rx_alarm_ctrl_val = (1<<2) | (1<<5) ; rx_alarm_ctrl_val = (1<<2) | (1<<5) ;
lasi_ctrl_val = 0x0004; lasi_ctrl_val = 0x0004;
bnx2x_wait_reset_complete(bp, phy);
DP(NETIF_MSG_LINK, "Initializing BCM8727\n"); DP(NETIF_MSG_LINK, "Initializing BCM8727\n");
/* enable LASI */ /* enable LASI */
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_RX_ALARM_CTRL,
MDIO_PMA_REG_RX_ALARM_CTRL,
rx_alarm_ctrl_val); rx_alarm_ctrl_val);
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_LASI_CTRL, lasi_ctrl_val);
MDIO_PMA_REG_LASI_CTRL,
lasi_ctrl_val);
/* Initially configure MOD_ABS to interrupt when /* Initially configure MOD_ABS to interrupt when
module is presence( bit 8) */ module is presence( bit 8) */
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_PHY_IDENTIFIER, &mod_abs);
MDIO_PMA_REG_PHY_IDENTIFIER, &mod_abs);
/* Set EDC off by setting OPTXLOS signal input to low /* Set EDC off by setting OPTXLOS signal input to low
(bit 9). (bit 9).
When the EDC is off it locks onto a reference clock and When the EDC is off it locks onto a reference clock and
avoids becoming 'lost'.*/ avoids becoming 'lost'.*/
mod_abs &= ~((1<<8) | (1<<9)); mod_abs &= ~((1<<8) | (1<<9));
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_PHY_IDENTIFIER, mod_abs);
MDIO_PMA_REG_PHY_IDENTIFIER, mod_abs);
/* Make MOD_ABS give interrupt on change */ /* Make MOD_ABS give interrupt on change */
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy, MDIO_PMA_DEVAD, MDIO_PMA_REG_8727_PCS_OPT_CTRL,
MDIO_PMA_DEVAD,
MDIO_PMA_REG_8727_PCS_OPT_CTRL,
&val); &val);
val |= (1<<12); val |= (1<<12);
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_8727_PCS_OPT_CTRL, val);
MDIO_PMA_REG_8727_PCS_OPT_CTRL,
val);
/* Set 8727 GPIOs to input to allow reading from the /* Set 8727 GPIOs to input to allow reading from the
8727 GPIO0 status which reflect SFP+ module 8727 GPIO0 status which reflect SFP+ module
over-current */ over-current */
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy, MDIO_PMA_DEVAD, MDIO_PMA_REG_8727_PCS_OPT_CTRL,
MDIO_PMA_DEVAD,
MDIO_PMA_REG_8727_PCS_OPT_CTRL,
&val); &val);
val &= 0xff8f; /* Reset bits 4-6 */ val &= 0xff8f; /* Reset bits 4-6 */
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_8727_PCS_OPT_CTRL, val);
MDIO_PMA_REG_8727_PCS_OPT_CTRL,
val);
bnx2x_8727_power_module(bp, params, phy, 1); bnx2x_8727_power_module(bp, phy, 1);
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_M8051_MSGOUT_REG, &tmp1);
MDIO_PMA_REG_M8051_MSGOUT_REG,
&tmp1);
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_RX_ALARM, &tmp1);
MDIO_PMA_REG_RX_ALARM, &tmp1);
/* Set option 1G speed */ /* Set option 1G speed */
if (params->req_line_speed == SPEED_1000) { if (phy->req_line_speed == SPEED_1000) {
DP(NETIF_MSG_LINK, "Setting 1G force\n"); DP(NETIF_MSG_LINK, "Setting 1G force\n");
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_CTRL, 0x40);
MDIO_PMA_REG_CTRL, 0x40);
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_10G_CTRL2, 0xD);
MDIO_PMA_REG_10G_CTRL2, 0xD);
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_10G_CTRL2, &tmp1);
MDIO_PMA_REG_10G_CTRL2, &tmp1);
DP(NETIF_MSG_LINK, "1.7 = 0x%x\n", tmp1); DP(NETIF_MSG_LINK, "1.7 = 0x%x\n", tmp1);
} else if ((phy->req_line_speed == SPEED_AUTO_NEG) &&
} else if ((params->req_line_speed == ((phy->speed_cap_mask &
SPEED_AUTO_NEG) &&
((params->speed_cap_mask &
PORT_HW_CFG_SPEED_CAPABILITY_D0_1G)) && PORT_HW_CFG_SPEED_CAPABILITY_D0_1G)) &&
((params->speed_cap_mask & ((phy->speed_cap_mask &
PORT_HW_CFG_SPEED_CAPABILITY_D0_10G) != PORT_HW_CFG_SPEED_CAPABILITY_D0_10G) !=
PORT_HW_CFG_SPEED_CAPABILITY_D0_10G)) { PORT_HW_CFG_SPEED_CAPABILITY_D0_10G)) {
DP(NETIF_MSG_LINK, "Setting 1G clause37\n"); DP(NETIF_MSG_LINK, "Setting 1G clause37\n");
bnx2x_cl45_write(bp, phy, MDIO_AN_DEVAD, bnx2x_cl45_write(bp, phy,
MDIO_AN_REG_8727_MISC_CTRL, 0); MDIO_AN_DEVAD, MDIO_AN_REG_8727_MISC_CTRL, 0);
bnx2x_cl45_write(bp, phy, MDIO_AN_DEVAD, bnx2x_cl45_write(bp, phy,
MDIO_AN_REG_CL37_AN, 0x1300); MDIO_AN_DEVAD, MDIO_AN_REG_CL37_AN, 0x1300);
} else { } else {
/* Since the 8727 has only single reset pin, /**
need to set the 10G registers although it is * Since the 8727 has only single reset pin, need to set the 10G
default */ * registers although it is default
bnx2x_cl45_write(bp, phy, MDIO_AN_DEVAD, */
MDIO_AN_REG_8727_MISC_CTRL, bnx2x_cl45_write(bp, phy,
MDIO_AN_DEVAD, MDIO_AN_REG_8727_MISC_CTRL,
0x0020); 0x0020);
bnx2x_cl45_write(bp, phy, MDIO_AN_DEVAD, bnx2x_cl45_write(bp, phy,
MDIO_AN_REG_CL37_AN, 0x0100); MDIO_AN_DEVAD, MDIO_AN_REG_CL37_AN, 0x0100);
bnx2x_cl45_write(bp, phy, MDIO_PMA_DEVAD, bnx2x_cl45_write(bp, phy,
MDIO_PMA_REG_CTRL, 0x2040); MDIO_PMA_DEVAD, MDIO_PMA_REG_CTRL, 0x2040);
bnx2x_cl45_write(bp, phy, MDIO_PMA_DEVAD, bnx2x_cl45_write(bp, phy,
MDIO_PMA_REG_10G_CTRL2, 0x0008); MDIO_PMA_DEVAD, MDIO_PMA_REG_10G_CTRL2,
0x0008);
} }
/* Set 2-wire transfer rate of SFP+ module EEPROM /* Set 2-wire transfer rate of SFP+ module EEPROM
* to 100Khz since some DACs(direct attached cables) do * to 100Khz since some DACs(direct attached cables) do
* not work at 400Khz. * not work at 400Khz.
*/ */
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_8727_TWO_WIRE_SLAVE_ADDR,
MDIO_PMA_REG_8727_TWO_WIRE_SLAVE_ADDR,
0xa001); 0xa001);
/* Set TX PreEmphasis if needed */ /* Set TX PreEmphasis if needed */
if ((params->feature_config_flags & if ((params->feature_config_flags &
FEATURE_CONFIG_OVERRIDE_PREEMPHASIS_ENABLED)) { FEATURE_CONFIG_OVERRIDE_PREEMPHASIS_ENABLED)) {
DP(NETIF_MSG_LINK, "Setting TX_CTRL1 0x%x," DP(NETIF_MSG_LINK, "Setting TX_CTRL1 0x%x, TX_CTRL2 0x%x\n",
"TX_CTRL2 0x%x\n",
phy->tx_preemphasis[0], phy->tx_preemphasis[0],
phy->tx_preemphasis[1]); phy->tx_preemphasis[1]);
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_8727_TX_CTRL1,
MDIO_PMA_REG_8727_TX_CTRL1,
phy->tx_preemphasis[0]); phy->tx_preemphasis[0]);
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_8727_TX_CTRL2,
MDIO_PMA_REG_8727_TX_CTRL2,
phy->tx_preemphasis[1]); phy->tx_preemphasis[1]);
} }
...@@ -3861,39 +3633,30 @@ static u8 bnx2x_7101_config_init(struct bnx2x_phy *phy, ...@@ -3861,39 +3633,30 @@ static u8 bnx2x_7101_config_init(struct bnx2x_phy *phy,
MISC_REGISTERS_GPIO_OUTPUT_HIGH, params->port); MISC_REGISTERS_GPIO_OUTPUT_HIGH, params->port);
/* HW reset */ /* HW reset */
bnx2x_ext_phy_hw_reset(bp, params->port); bnx2x_ext_phy_hw_reset(bp, params->port);
DP(NETIF_MSG_LINK,
"Setting the SFX7101 LASI indication\n");
bnx2x_wait_reset_complete(bp, phy); bnx2x_wait_reset_complete(bp, phy);
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_LASI_CTRL, 0x1);
MDIO_PMA_REG_LASI_CTRL, 0x1); DP(NETIF_MSG_LINK, "Setting the SFX7101 LED to blink on traffic\n");
DP(NETIF_MSG_LINK,
"Setting the SFX7101 LED to blink on traffic\n");
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_7107_LED_CNTL, (1<<3));
MDIO_PMA_REG_7107_LED_CNTL, (1<<3));
bnx2x_ext_phy_set_pause(params, phy, vars); bnx2x_ext_phy_set_pause(params, phy, vars);
/* Restart autoneg */ /* Restart autoneg */
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_AN_DEVAD, MDIO_AN_DEVAD, MDIO_AN_REG_CTRL, &val);
MDIO_AN_REG_CTRL, &val);
val |= 0x200; val |= 0x200;
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_AN_DEVAD, MDIO_AN_DEVAD, MDIO_AN_REG_CTRL, val);
MDIO_AN_REG_CTRL, val);
/* Save spirom version */ /* Save spirom version */
bnx2x_cl45_read(bp, phy, MDIO_PMA_DEVAD, bnx2x_cl45_read(bp, phy,
MDIO_PMA_REG_7101_VER1, &fw_ver1); MDIO_PMA_DEVAD, MDIO_PMA_REG_7101_VER1, &fw_ver1);
bnx2x_cl45_read(bp, phy, MDIO_PMA_DEVAD,
MDIO_PMA_REG_7101_VER2, &fw_ver2);
bnx2x_save_spirom_version(params->bp, params->port, bnx2x_cl45_read(bp, phy,
params->shmem_base, MDIO_PMA_DEVAD, MDIO_PMA_REG_7101_VER2, &fw_ver2);
(u32)(fw_ver1<<16 | fw_ver2)); bnx2x_save_spirom_version(bp, params->port,
(u32)(fw_ver1<<16 | fw_ver2), phy->ver_addr);
return 0; return 0;
} }
...@@ -3902,145 +3665,121 @@ static u8 bnx2x_848xx_cmn_config_init(struct bnx2x_phy *phy, ...@@ -3902,145 +3665,121 @@ static u8 bnx2x_848xx_cmn_config_init(struct bnx2x_phy *phy,
struct link_vars *vars) struct link_vars *vars)
{ {
struct bnx2x *bp = params->bp; struct bnx2x *bp = params->bp;
{ u16 autoneg_val, an_1000_val, an_10_100_val;
/* This phy uses the NIG latch mechanism since link
indication arrives through its LED4 and not via
its LASI signal, so we get steady signal
instead of clear on read */
u16 autoneg_val, an_1000_val, an_10_100_val, temp;
temp = vars->line_speed;
vars->line_speed = SPEED_10000;
bnx2x_wait_reset_complete(bp, phy); bnx2x_wait_reset_complete(bp, phy);
bnx2x_set_autoneg(phy, params, vars, 0);
bnx2x_program_serdes(phy, params, vars);
vars->line_speed = temp;
bnx2x_bits_en(bp, NIG_REG_LATCH_BC_0 + params->port*4, bnx2x_bits_en(bp, NIG_REG_LATCH_BC_0 + params->port*4,
1 << NIG_LATCH_BC_ENABLE_MI_INT); 1 << NIG_LATCH_BC_ENABLE_MI_INT);
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_CTRL, 0x0000);
MDIO_PMA_REG_CTRL, 0x0000);
bnx2x_8481_set_led(bp, phy); bnx2x_8481_set_led(bp, phy);
/* set 1000 speed advertisement */
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_AN_DEVAD, MDIO_AN_DEVAD, MDIO_AN_REG_8481_1000T_CTRL,
MDIO_AN_REG_8481_1000T_CTRL,
&an_1000_val); &an_1000_val);
bnx2x_ext_phy_set_pause(params, phy, vars); bnx2x_ext_phy_set_pause(params, phy, vars);
bnx2x_cl45_read(bp, phy, MDIO_AN_DEVAD, bnx2x_cl45_read(bp, phy,
MDIO_AN_DEVAD,
MDIO_AN_REG_8481_LEGACY_AN_ADV, MDIO_AN_REG_8481_LEGACY_AN_ADV,
&an_10_100_val); &an_10_100_val);
bnx2x_cl45_read(bp, phy, MDIO_AN_DEVAD, bnx2x_cl45_read(bp, phy,
MDIO_AN_REG_8481_LEGACY_MII_CTRL, MDIO_AN_DEVAD, MDIO_AN_REG_8481_LEGACY_MII_CTRL,
&autoneg_val); &autoneg_val);
/* Disable forced speed */ /* Disable forced speed */
autoneg_val &= ~((1<<6) | (1<<8) | (1<<9) | (1<<12) | autoneg_val &= ~((1<<6) | (1<<8) | (1<<9) | (1<<12) | (1<<13));
(1<<13));
an_10_100_val &= ~((1<<5) | (1<<6) | (1<<7) | (1<<8)); an_10_100_val &= ~((1<<5) | (1<<6) | (1<<7) | (1<<8));
if (((params->req_line_speed == SPEED_AUTO_NEG) && if (((phy->req_line_speed == SPEED_AUTO_NEG) &&
(params->speed_cap_mask & (phy->speed_cap_mask &
PORT_HW_CFG_SPEED_CAPABILITY_D0_1G)) || PORT_HW_CFG_SPEED_CAPABILITY_D0_1G)) ||
(params->req_line_speed == SPEED_1000)) { (phy->req_line_speed == SPEED_1000)) {
an_1000_val |= (1<<8); an_1000_val |= (1<<8);
autoneg_val |= (1<<9 | 1<<12); autoneg_val |= (1<<9 | 1<<12);
if (params->req_duplex == DUPLEX_FULL) if (phy->req_duplex == DUPLEX_FULL)
an_1000_val |= (1<<9); an_1000_val |= (1<<9);
DP(NETIF_MSG_LINK, "Advertising 1G\n"); DP(NETIF_MSG_LINK, "Advertising 1G\n");
} else } else
an_1000_val &= ~((1<<8) | (1<<9)); an_1000_val &= ~((1<<8) | (1<<9));
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_AN_DEVAD, MDIO_AN_DEVAD, MDIO_AN_REG_8481_1000T_CTRL,
MDIO_AN_REG_8481_1000T_CTRL,
an_1000_val); an_1000_val);
/* set 10 speed advertisement */ /* set 10 speed advertisement */
if (((params->req_line_speed == SPEED_AUTO_NEG) && if (((phy->req_line_speed == SPEED_AUTO_NEG) &&
(params->speed_cap_mask & (phy->speed_cap_mask &
(PORT_HW_CFG_SPEED_CAPABILITY_D0_100M_FULL | (PORT_HW_CFG_SPEED_CAPABILITY_D0_100M_FULL |
PORT_HW_CFG_SPEED_CAPABILITY_D0_100M_HALF)))) { PORT_HW_CFG_SPEED_CAPABILITY_D0_100M_HALF)))) {
an_10_100_val |= (1<<7); an_10_100_val |= (1<<7);
/* /* Enable autoneg and restart autoneg for legacy speeds */
* Enable autoneg and restart autoneg for
* legacy speeds
*/
autoneg_val |= (1<<9 | 1<<12); autoneg_val |= (1<<9 | 1<<12);
if (params->req_duplex == DUPLEX_FULL) if (phy->req_duplex == DUPLEX_FULL)
an_10_100_val |= (1<<8); an_10_100_val |= (1<<8);
DP(NETIF_MSG_LINK, "Advertising 100M\n"); DP(NETIF_MSG_LINK, "Advertising 100M\n");
} }
/* set 10 speed advertisement */ /* set 10 speed advertisement */
if (((params->req_line_speed == SPEED_AUTO_NEG) && if (((phy->req_line_speed == SPEED_AUTO_NEG) &&
(params->speed_cap_mask & (phy->speed_cap_mask &
(PORT_HW_CFG_SPEED_CAPABILITY_D0_10M_FULL | (PORT_HW_CFG_SPEED_CAPABILITY_D0_10M_FULL |
PORT_HW_CFG_SPEED_CAPABILITY_D0_10M_HALF)))) { PORT_HW_CFG_SPEED_CAPABILITY_D0_10M_HALF)))) {
an_10_100_val |= (1<<5); an_10_100_val |= (1<<5);
autoneg_val |= (1<<9 | 1<<12); autoneg_val |= (1<<9 | 1<<12);
if (params->req_duplex == DUPLEX_FULL) if (phy->req_duplex == DUPLEX_FULL)
an_10_100_val |= (1<<6); an_10_100_val |= (1<<6);
DP(NETIF_MSG_LINK, "Advertising 10M\n"); DP(NETIF_MSG_LINK, "Advertising 10M\n");
} }
/* Only 10/100 are allowed to work in FORCE mode */ /* Only 10/100 are allowed to work in FORCE mode */
if (params->req_line_speed == SPEED_100) { if (phy->req_line_speed == SPEED_100) {
autoneg_val |= (1<<13); autoneg_val |= (1<<13);
/* Enabled AUTO-MDIX when autoneg is disabled */ /* Enabled AUTO-MDIX when autoneg is disabled */
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_AN_DEVAD, MDIO_AN_DEVAD, MDIO_AN_REG_8481_AUX_CTRL,
MDIO_AN_REG_8481_AUX_CTRL,
(1<<15 | 1<<9 | 7<<0)); (1<<15 | 1<<9 | 7<<0));
DP(NETIF_MSG_LINK, "Setting 100M force\n"); DP(NETIF_MSG_LINK, "Setting 100M force\n");
} }
if (params->req_line_speed == SPEED_10) { if (phy->req_line_speed == SPEED_10) {
/* Enabled AUTO-MDIX when autoneg is disabled */ /* Enabled AUTO-MDIX when autoneg is disabled */
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_AN_DEVAD, MDIO_AN_DEVAD, MDIO_AN_REG_8481_AUX_CTRL,
MDIO_AN_REG_8481_AUX_CTRL,
(1<<15 | 1<<9 | 7<<0)); (1<<15 | 1<<9 | 7<<0));
DP(NETIF_MSG_LINK, "Setting 10M force\n"); DP(NETIF_MSG_LINK, "Setting 10M force\n");
} }
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_AN_DEVAD, MDIO_AN_DEVAD, MDIO_AN_REG_8481_LEGACY_AN_ADV,
MDIO_AN_REG_8481_LEGACY_AN_ADV,
an_10_100_val); an_10_100_val);
if (params->req_duplex == DUPLEX_FULL) if (phy->req_duplex == DUPLEX_FULL)
autoneg_val |= (1<<8); autoneg_val |= (1<<8);
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_AN_DEVAD, MDIO_AN_DEVAD,
MDIO_AN_REG_8481_LEGACY_MII_CTRL, MDIO_AN_REG_8481_LEGACY_MII_CTRL, autoneg_val);
autoneg_val);
if (((params->req_line_speed == SPEED_AUTO_NEG) && if (((phy->req_line_speed == SPEED_AUTO_NEG) &&
(params->speed_cap_mask & (phy->speed_cap_mask &
PORT_HW_CFG_SPEED_CAPABILITY_D0_10G)) || PORT_HW_CFG_SPEED_CAPABILITY_D0_10G)) ||
(params->req_line_speed == SPEED_10000)) { (phy->req_line_speed == SPEED_10000)) {
DP(NETIF_MSG_LINK, "Advertising 10G\n"); DP(NETIF_MSG_LINK, "Advertising 10G\n");
/* Restart autoneg for 10G*/ /* Restart autoneg for 10G*/
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_AN_DEVAD, MDIO_AN_DEVAD, MDIO_AN_REG_CTRL,
MDIO_AN_REG_CTRL,
0x3200); 0x3200);
} else if (phy->req_line_speed != SPEED_10 &&
} else if (params->req_line_speed != SPEED_10 && phy->req_line_speed != SPEED_100) {
params->req_line_speed != SPEED_100)
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_AN_DEVAD, MDIO_AN_DEVAD,
MDIO_AN_REG_8481_10GBASE_T_AN_CTRL, MDIO_AN_REG_8481_10GBASE_T_AN_CTRL,
1); 1);
}
/* Save spirom version */ /* Save spirom version */
bnx2x_save_8481_spirom_version(phy, params, bnx2x_save_8481_spirom_version(phy, params, params->shmem_base);
params->shmem_base);
return 0; return 0;
}
} }
static u8 bnx2x_848x3_config_init(struct bnx2x_phy *phy, static u8 bnx2x_848x3_config_init(struct bnx2x_phy *phy,
...@@ -4164,37 +3903,33 @@ static u8 bnx2x_8705_read_status(struct bnx2x_phy *phy, ...@@ -4164,37 +3903,33 @@ static u8 bnx2x_8705_read_status(struct bnx2x_phy *phy,
struct link_params *params, struct link_params *params,
struct link_vars *vars) struct link_vars *vars)
{ {
u8 ext_phy_link_up = 0; u8 link_up = 0;
u16 val1, rx_sd; u16 val1, rx_sd;
struct bnx2x *bp = params->bp; struct bnx2x *bp = params->bp;
DP(NETIF_MSG_LINK, "XGXS 8705\n"); DP(NETIF_MSG_LINK, "read status 8705\n");
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_WIS_DEVAD, MDIO_WIS_DEVAD, MDIO_WIS_REG_LASI_STATUS, &val1);
MDIO_WIS_REG_LASI_STATUS, &val1);
DP(NETIF_MSG_LINK, "8705 LASI status 0x%x\n", val1); DP(NETIF_MSG_LINK, "8705 LASI status 0x%x\n", val1);
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_WIS_DEVAD, MDIO_WIS_DEVAD, MDIO_WIS_REG_LASI_STATUS, &val1);
MDIO_WIS_REG_LASI_STATUS, &val1);
DP(NETIF_MSG_LINK, "8705 LASI status 0x%x\n", val1); DP(NETIF_MSG_LINK, "8705 LASI status 0x%x\n", val1);
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_RX_SD, &rx_sd);
MDIO_PMA_REG_RX_SD, &rx_sd);
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
1, MDIO_PMA_DEVAD, 0xc809, &val1);
0xc809, &val1);
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
1, MDIO_PMA_DEVAD, 0xc809, &val1);
0xc809, &val1);
DP(NETIF_MSG_LINK, "8705 1.c809 val=0x%x\n", val1); DP(NETIF_MSG_LINK, "8705 1.c809 val=0x%x\n", val1);
ext_phy_link_up = ((rx_sd & 0x1) && (val1 & (1<<9)) && link_up = ((rx_sd & 0x1) && (val1 & (1<<9)) && ((val1 & (1<<8)) == 0));
((val1 & (1<<8)) == 0)); if (link_up) {
if (ext_phy_link_up)
vars->line_speed = SPEED_10000; vars->line_speed = SPEED_10000;
return ext_phy_link_up; bnx2x_ext_phy_resolve_fc(phy, params, vars);
}
return link_up;
} }
static u8 bnx2x_8706_8726_read_status(struct bnx2x_phy *phy, static u8 bnx2x_8706_8726_read_status(struct bnx2x_phy *phy,
...@@ -4207,49 +3942,38 @@ static u8 bnx2x_8706_8726_read_status(struct bnx2x_phy *phy, ...@@ -4207,49 +3942,38 @@ static u8 bnx2x_8706_8726_read_status(struct bnx2x_phy *phy,
DP(NETIF_MSG_LINK, "XGXS 8706/8726\n"); DP(NETIF_MSG_LINK, "XGXS 8706/8726\n");
/* Clear RX Alarm*/ /* Clear RX Alarm*/
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_REG_RX_ALARM, MDIO_PMA_DEVAD, MDIO_PMA_REG_RX_ALARM, &val2);
&val2);
/* clear LASI indication*/ /* clear LASI indication*/
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_REG_LASI_STATUS, MDIO_PMA_DEVAD, MDIO_PMA_REG_LASI_STATUS, &val1);
&val1);
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_REG_LASI_STATUS, MDIO_PMA_DEVAD, MDIO_PMA_REG_LASI_STATUS, &val2);
&val2); DP(NETIF_MSG_LINK, "8706/8726 LASI status 0x%x--> 0x%x\n", val1, val2);
DP(NETIF_MSG_LINK, "8706/8726 LASI status 0x%x-->"
"0x%x\n", val1, val2);
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_REG_RX_SD, MDIO_PMA_DEVAD, MDIO_PMA_REG_RX_SD, &rx_sd);
&rx_sd);
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PCS_DEVAD, MDIO_PCS_REG_STATUS, MDIO_PCS_DEVAD, MDIO_PCS_REG_STATUS, &pcs_status);
&pcs_status);
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_AN_DEVAD, MDIO_AN_REG_LINK_STATUS, MDIO_AN_DEVAD, MDIO_AN_REG_LINK_STATUS, &val2);
&val2);
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_AN_DEVAD, MDIO_AN_REG_LINK_STATUS, MDIO_AN_DEVAD, MDIO_AN_REG_LINK_STATUS, &val2);
&val2);
DP(NETIF_MSG_LINK, "8706/8726 rx_sd 0x%x" DP(NETIF_MSG_LINK, "8706/8726 rx_sd 0x%x pcs_status 0x%x 1Gbps"
" pcs_status 0x%x 1Gbps link_status 0x%x\n", " link_status 0x%x\n", rx_sd, pcs_status, val2);
rx_sd, pcs_status, val2);
/* link is up if both bit 0 of pmd_rx_sd and /* link is up if both bit 0 of pmd_rx_sd and
* bit 0 of pcs_status are set, or if the autoneg bit * bit 0 of pcs_status are set, or if the autoneg bit
1 is set * 1 is set
*/ */
link_up = ((rx_sd & pcs_status & 0x1) || link_up = ((rx_sd & pcs_status & 0x1) || (val2 & (1<<1)));
(val2 & (1<<1)));
if (link_up) { if (link_up) {
if (val2 & (1<<1)) if (val2 & (1<<1))
vars->line_speed = SPEED_1000; vars->line_speed = SPEED_1000;
else else
vars->line_speed = SPEED_10000; vars->line_speed = SPEED_10000;
bnx2x_ext_phy_resolve_fc(phy, params, vars); bnx2x_ext_phy_resolve_fc(phy, params, vars);
return link_up;
} }
return 0; return link_up;
} }
static u8 bnx2x_8706_read_status(struct bnx2x_phy *phy, static u8 bnx2x_8706_read_status(struct bnx2x_phy *phy,
...@@ -4284,75 +4008,68 @@ static u8 bnx2x_8727_read_status(struct bnx2x_phy *phy, ...@@ -4284,75 +4008,68 @@ static u8 bnx2x_8727_read_status(struct bnx2x_phy *phy,
{ {
struct bnx2x *bp = params->bp; struct bnx2x *bp = params->bp;
u8 ext_phy_link_up = 0; u8 link_up = 0;
u16 link_status = 0; u16 link_status = 0;
u16 rx_alarm_status, val1; u16 rx_alarm_status, val1;
/* Check the LASI */ /* Check the LASI */
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_RX_ALARM,
MDIO_PMA_REG_RX_ALARM, &rx_alarm_status); &rx_alarm_status);
vars->line_speed = 0;
DP(NETIF_MSG_LINK, "8727 RX_ALARM_STATUS 0x%x\n", DP(NETIF_MSG_LINK, "8727 RX_ALARM_STATUS 0x%x\n", rx_alarm_status);
rx_alarm_status);
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_LASI_STATUS, &val1);
MDIO_PMA_REG_LASI_STATUS, &val1);
DP(NETIF_MSG_LINK, DP(NETIF_MSG_LINK, "8727 LASI status 0x%x\n", val1);
"8727 LASI status 0x%x\n",
val1);
/* Clear MSG-OUT */ /* Clear MSG-OUT */
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_M8051_MSGOUT_REG, &val1);
MDIO_PMA_REG_M8051_MSGOUT_REG,
&val1);
/* /**
* If a module is present and there is need to check * If a module is present and there is need to check
* for over current * for over current
*/ */
if (!(phy->flags & FLAGS_NOC) && if (!(phy->flags & FLAGS_NOC) && !(rx_alarm_status & (1<<5))) {
!(rx_alarm_status & (1<<5))) {
/* Check over-current using 8727 GPIO0 input*/ /* Check over-current using 8727 GPIO0 input*/
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_8727_GPIO_CTRL,
MDIO_PMA_REG_8727_GPIO_CTRL,
&val1); &val1);
if ((val1 & (1<<8)) == 0) { if ((val1 & (1<<8)) == 0) {
DP(NETIF_MSG_LINK, "8727 Power fault" DP(NETIF_MSG_LINK, "8727 Power fault has been detected"
" has been detected on " " on port %d\n", params->port);
"port %d\n", netdev_err(bp->dev, "Error: Power fault on Port %d has"
params->port); " been detected and the power to "
netdev_err(bp->dev, "Error: Power fault on Port %d has been detected and the power to that SFP+ module has been removed to prevent failure of the card. Please remove the SFP+ module and restart the system to clear this error.\n", "that SFP+ module has been removed"
" to prevent failure of the card."
" Please remove the SFP+ module and"
" restart the system to clear this"
" error.\n",
params->port); params->port);
/* /*
* Disable all RX_ALARMs except for * Disable all RX_ALARMs except for
* mod_abs * mod_abs
*/ */
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD,
MDIO_PMA_REG_RX_ALARM_CTRL, MDIO_PMA_REG_RX_ALARM_CTRL, (1<<5));
(1<<5));
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD,
MDIO_PMA_REG_PHY_IDENTIFIER, MDIO_PMA_REG_PHY_IDENTIFIER, &val1);
&val1);
/* Wait for module_absent_event */ /* Wait for module_absent_event */
val1 |= (1<<8); val1 |= (1<<8);
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD,
MDIO_PMA_REG_PHY_IDENTIFIER, MDIO_PMA_REG_PHY_IDENTIFIER, val1);
val1);
/* Clear RX alarm */ /* Clear RX alarm */
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD,
MDIO_PMA_REG_RX_ALARM, MDIO_PMA_REG_RX_ALARM, &rx_alarm_status);
&rx_alarm_status); return 0;
return ext_phy_link_up;
} }
} /* Over current check */ } /* Over current check */
...@@ -4361,235 +4078,147 @@ static u8 bnx2x_8727_read_status(struct bnx2x_phy *phy, ...@@ -4361,235 +4078,147 @@ static u8 bnx2x_8727_read_status(struct bnx2x_phy *phy,
bnx2x_8727_handle_mod_abs(phy, params); bnx2x_8727_handle_mod_abs(phy, params);
/* Enable all mod_abs and link detection bits */ /* Enable all mod_abs and link detection bits */
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_RX_ALARM_CTRL,
MDIO_PMA_REG_RX_ALARM_CTRL,
((1<<5) | (1<<2))); ((1<<5) | (1<<2)));
} }
/* If transmitter is disabled, /* If transmitter is disabled, ignore false link up indication */
ignore false link up indication */
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_PHY_IDENTIFIER, &val1);
MDIO_PMA_REG_PHY_IDENTIFIER,
&val1);
if (val1 & (1<<15)) { if (val1 & (1<<15)) {
DP(NETIF_MSG_LINK, "Tx is disabled\n"); DP(NETIF_MSG_LINK, "Tx is disabled\n");
ext_phy_link_up = 0; return 0;
return ext_phy_link_up;
} }
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD,
MDIO_PMA_REG_8073_SPEED_LINK_STATUS, MDIO_PMA_REG_8073_SPEED_LINK_STATUS, &link_status);
&link_status);
/* Bits 0..2 --> speed detected, /* Bits 0..2 --> speed detected,
bits 13..15--> link is down */ bits 13..15--> link is down */
if ((link_status & (1<<2)) && if ((link_status & (1<<2)) && (!(link_status & (1<<15)))) {
(!(link_status & (1<<15)))) { link_up = 1;
ext_phy_link_up = 1;
vars->line_speed = SPEED_10000; vars->line_speed = SPEED_10000;
} else if ((link_status & (1<<0)) && } else if ((link_status & (1<<0)) && (!(link_status & (1<<13)))) {
(!(link_status & (1<<13)))) { link_up = 1;
ext_phy_link_up = 1;
vars->line_speed = SPEED_1000; vars->line_speed = SPEED_1000;
DP(NETIF_MSG_LINK, DP(NETIF_MSG_LINK, "port %x: External link up in 1G\n",
"port %x: External link" params->port);
" up in 1G\n", params->port);
} else { } else {
ext_phy_link_up = 0; link_up = 0;
DP(NETIF_MSG_LINK, DP(NETIF_MSG_LINK, "port %x: External link is down\n",
"port %x: External link" params->port);
" is down\n", params->port);
} }
return ext_phy_link_up; if (link_up)
bnx2x_ext_phy_resolve_fc(phy, params, vars);
return link_up;
} }
static u8 bnx2x_8073_read_status(struct bnx2x_phy *phy, static u8 bnx2x_8073_read_status(struct bnx2x_phy *phy,
struct link_params *params, struct link_params *params,
struct link_vars *vars) struct link_vars *vars)
{ {
struct bnx2x *bp = params->bp; struct bnx2x *bp = params->bp;
u8 ext_phy_link_up = 0; u8 link_up = 0;
u16 val1, val2; u16 val1, val2;
u16 link_status = 0; u16 link_status = 0;
u16 an1000_status = 0; u16 an1000_status = 0;
if (phy->type ==
PORT_HW_CFG_XGXS_EXT_PHY_TYPE_BCM8072) {
bnx2x_cl45_read(bp, phy,
MDIO_PCS_DEVAD,
MDIO_PCS_REG_LASI_STATUS, &val1);
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PCS_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_LASI_STATUS, &val1);
MDIO_PCS_REG_LASI_STATUS, &val2);
DP(NETIF_MSG_LINK,
"870x LASI status 0x%x->0x%x\n",
val1, val2);
} else {
/* In 8073, port1 is directed through emac0 and
* port0 is directed through emac1
*/
bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD,
MDIO_PMA_REG_LASI_STATUS, &val1);
DP(NETIF_MSG_LINK, DP(NETIF_MSG_LINK, "8703 LASI status 0x%x\n", val1);
"8703 LASI status 0x%x\n",
val1);
}
/* clear the interrupt LASI status register */ /* clear the interrupt LASI status register */
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PCS_DEVAD, MDIO_PCS_DEVAD, MDIO_PCS_REG_STATUS, &val2);
MDIO_PCS_REG_STATUS, &val2);
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PCS_DEVAD, MDIO_PCS_DEVAD, MDIO_PCS_REG_STATUS, &val1);
MDIO_PCS_REG_STATUS, &val1); DP(NETIF_MSG_LINK, "807x PCS status 0x%x->0x%x\n", val2, val1);
DP(NETIF_MSG_LINK, "807x PCS status 0x%x->0x%x\n",
val2, val1);
/* Clear MSG-OUT */ /* Clear MSG-OUT */
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_M8051_MSGOUT_REG, &val1);
MDIO_PMA_REG_M8051_MSGOUT_REG,
&val1);
/* Check the LASI */ /* Check the LASI */
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_RX_ALARM, &val2);
MDIO_PMA_REG_RX_ALARM, &val2);
DP(NETIF_MSG_LINK, "KR 0x9003 0x%x\n", val2); DP(NETIF_MSG_LINK, "KR 0x9003 0x%x\n", val2);
/* Check the link status */ /* Check the link status */
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PCS_DEVAD, MDIO_PCS_DEVAD, MDIO_PCS_REG_STATUS, &val2);
MDIO_PCS_REG_STATUS, &val2);
DP(NETIF_MSG_LINK, "KR PCS status 0x%x\n", val2); DP(NETIF_MSG_LINK, "KR PCS status 0x%x\n", val2);
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_STATUS, &val2);
MDIO_PMA_REG_STATUS, &val2);
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_STATUS, &val1);
MDIO_PMA_REG_STATUS, &val1); link_up = ((val1 & 4) == 4);
ext_phy_link_up = ((val1 & 4) == 4);
DP(NETIF_MSG_LINK, "PMA_REG_STATUS=0x%x\n", val1); DP(NETIF_MSG_LINK, "PMA_REG_STATUS=0x%x\n", val1);
if (phy->type ==
PORT_HW_CFG_XGXS_EXT_PHY_TYPE_BCM8073) {
if (ext_phy_link_up && if (link_up &&
((params->req_line_speed != ((phy->req_line_speed != SPEED_10000))) {
SPEED_10000))) { if (bnx2x_8073_xaui_wa(bp, phy) != 0)
if (bnx2x_8073_xaui_wa(bp, phy) return 0;
!= 0) {
ext_phy_link_up = 0;
return ext_phy_link_up;
}
} }
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_AN_DEVAD, MDIO_AN_DEVAD, MDIO_AN_REG_LINK_STATUS, &an1000_status);
MDIO_AN_REG_LINK_STATUS,
&an1000_status);
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_AN_DEVAD, MDIO_AN_DEVAD, MDIO_AN_REG_LINK_STATUS, &an1000_status);
MDIO_AN_REG_LINK_STATUS,
&an1000_status);
/* Check the link status on 1.1.2 */ /* Check the link status on 1.1.2 */
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_STATUS, &val2);
MDIO_PMA_REG_STATUS, &val2);
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_STATUS, &val1);
MDIO_PMA_REG_STATUS, &val1);
DP(NETIF_MSG_LINK, "KR PMA status 0x%x->0x%x," DP(NETIF_MSG_LINK, "KR PMA status 0x%x->0x%x,"
"an_link_status=0x%x\n", "an_link_status=0x%x\n", val2, val1, an1000_status);
val2, val1, an1000_status);
ext_phy_link_up = (((val1 & 4) == 4) || link_up = (((val1 & 4) == 4) || (an1000_status & (1<<1)));
(an1000_status & (1<<1))); if (link_up && bnx2x_8073_is_snr_needed(bp, phy)) {
if (ext_phy_link_up &&
bnx2x_8073_is_snr_needed(bp, phy)) {
/* The SNR will improve about 2dbby /* The SNR will improve about 2dbby
changing the BW and FEE main tap.*/ changing the BW and FEE main tap.*/
/* The 1st write to change FFE main /* The 1st write to change FFE main
tap is set before restart AN */ tap is set before restart AN */
/* Change PLL Bandwidth in EDC /* Change PLL Bandwidth in EDC
register */ register */
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_PLL_BANDWIDTH,
MDIO_PMA_REG_PLL_BANDWIDTH,
0x26BC); 0x26BC);
/* Change CDR Bandwidth in EDC /* Change CDR Bandwidth in EDC register */
register */
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_CDR_BANDWIDTH,
MDIO_PMA_REG_CDR_BANDWIDTH,
0x0333); 0x0333);
} }
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_8073_SPEED_LINK_STATUS,
MDIO_PMA_REG_8073_SPEED_LINK_STATUS,
&link_status); &link_status);
/* Bits 0..2 --> speed detected, /* Bits 0..2 --> speed detected, bits 13..15--> link is down */
bits 13..15--> link is down */ if ((link_status & (1<<2)) && (!(link_status & (1<<15)))) {
if ((link_status & (1<<2)) && link_up = 1;
(!(link_status & (1<<15)))) {
ext_phy_link_up = 1;
vars->line_speed = SPEED_10000; vars->line_speed = SPEED_10000;
DP(NETIF_MSG_LINK, DP(NETIF_MSG_LINK, "port %x: External link up in 10G\n",
"port %x: External link" params->port);
" up in 10G\n", params->port); } else if ((link_status & (1<<1)) && (!(link_status & (1<<14)))) {
} else if ((link_status & (1<<1)) && link_up = 1;
(!(link_status & (1<<14)))) {
ext_phy_link_up = 1;
vars->line_speed = SPEED_2500; vars->line_speed = SPEED_2500;
DP(NETIF_MSG_LINK, DP(NETIF_MSG_LINK, "port %x: External link up in 2.5G\n",
"port %x: External link" params->port);
" up in 2.5G\n", params->port); } else if ((link_status & (1<<0)) && (!(link_status & (1<<13)))) {
} else if ((link_status & (1<<0)) && link_up = 1;
(!(link_status & (1<<13)))) {
ext_phy_link_up = 1;
vars->line_speed = SPEED_1000; vars->line_speed = SPEED_1000;
DP(NETIF_MSG_LINK, DP(NETIF_MSG_LINK, "port %x: External link up in 1G\n",
"port %x: External link" params->port);
" up in 1G\n", params->port);
} else {
ext_phy_link_up = 0;
DP(NETIF_MSG_LINK,
"port %x: External link"
" is down\n", params->port);
}
} else { } else {
/* See if 1G link is up for the 8072 */ link_up = 0;
bnx2x_cl45_read(bp, phy, DP(NETIF_MSG_LINK, "port %x: External link is down\n",
MDIO_AN_DEVAD, params->port);
MDIO_AN_REG_LINK_STATUS,
&an1000_status);
bnx2x_cl45_read(bp, phy,
MDIO_AN_DEVAD,
MDIO_AN_REG_LINK_STATUS,
&an1000_status);
if (an1000_status & (1<<1)) {
ext_phy_link_up = 1;
vars->line_speed = SPEED_1000;
DP(NETIF_MSG_LINK,
"port %x: External link"
" up in 1G\n", params->port);
} else if (ext_phy_link_up) {
ext_phy_link_up = 1;
vars->line_speed = SPEED_10000;
DP(NETIF_MSG_LINK,
"port %x: External link"
" up in 10G\n", params->port);
}
} }
return ext_phy_link_up; return link_up;
} }
static u8 bnx2x_7101_read_status(struct bnx2x_phy *phy, static u8 bnx2x_7101_read_status(struct bnx2x_phy *phy,
...@@ -4597,42 +4226,33 @@ static u8 bnx2x_7101_read_status(struct bnx2x_phy *phy, ...@@ -4597,42 +4226,33 @@ static u8 bnx2x_7101_read_status(struct bnx2x_phy *phy,
struct link_vars *vars) struct link_vars *vars)
{ {
struct bnx2x *bp = params->bp; struct bnx2x *bp = params->bp;
u8 ext_phy_link_up; u8 link_up;
u16 val1, val2; u16 val1, val2;
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_LASI_STATUS, &val2);
MDIO_PMA_REG_LASI_STATUS, &val2);
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_LASI_STATUS, &val1);
MDIO_PMA_REG_LASI_STATUS, &val1); DP(NETIF_MSG_LINK, "10G-base-T LASI status 0x%x->0x%x\n",
DP(NETIF_MSG_LINK,
"10G-base-T LASI status 0x%x->0x%x\n",
val2, val1); val2, val1);
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_STATUS, &val2);
MDIO_PMA_REG_STATUS, &val2);
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_STATUS, &val1);
MDIO_PMA_REG_STATUS, &val1); DP(NETIF_MSG_LINK, "10G-base-T PMA status 0x%x->0x%x\n",
DP(NETIF_MSG_LINK,
"10G-base-T PMA status 0x%x->0x%x\n",
val2, val1); val2, val1);
ext_phy_link_up = ((val1 & 4) == 4); link_up = ((val1 & 4) == 4);
/* if link is up /* if link is up
* print the AN outcome of the SFX7101 PHY * print the AN outcome of the SFX7101 PHY
*/ */
if (ext_phy_link_up) { if (link_up) {
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_AN_DEVAD, MDIO_AN_DEVAD, MDIO_AN_REG_MASTER_STATUS,
MDIO_AN_REG_MASTER_STATUS,
&val2); &val2);
vars->line_speed = SPEED_10000; vars->line_speed = SPEED_10000;
DP(NETIF_MSG_LINK, DP(NETIF_MSG_LINK, "SFX7101 AN status 0x%x->Master=%x\n",
"SFX7101 AN status 0x%x->Master=%x\n", val2, (val2 & (1<<14)));
val2,
(val2 & (1<<14)));
} }
return ext_phy_link_up; return link_up;
} }
static u8 bnx2x_848xx_read_status(struct bnx2x_phy *phy, static u8 bnx2x_848xx_read_status(struct bnx2x_phy *phy,
...@@ -4640,34 +4260,29 @@ static u8 bnx2x_848xx_read_status(struct bnx2x_phy *phy, ...@@ -4640,34 +4260,29 @@ static u8 bnx2x_848xx_read_status(struct bnx2x_phy *phy,
struct link_vars *vars) struct link_vars *vars)
{ {
struct bnx2x *bp = params->bp; struct bnx2x *bp = params->bp;
u16 val1, val2; u16 val, val1, val2;
u8 ext_phy_link_up = 0; u8 link_up = 0;
/* Check 10G-BaseT link status */ /* Check 10G-BaseT link status */
/* Check PMD signal ok */ /* Check PMD signal ok */
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_AN_DEVAD, MDIO_AN_DEVAD, 0xFFFA, &val1);
0xFFFA,
&val1);
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
MDIO_PMA_DEVAD, MDIO_PMA_DEVAD, MDIO_PMA_REG_8481_PMD_SIGNAL,
MDIO_PMA_REG_8481_PMD_SIGNAL,
&val2); &val2);
DP(NETIF_MSG_LINK, "PMD_SIGNAL 1.a811 = 0x%x\n", val2); DP(NETIF_MSG_LINK, "BCM848xx: PMD_SIGNAL 1.a811 = 0x%x\n", val2);
/* Check link 10G */ /* Check link 10G */
if (val2 & (1<<11)) { if (val2 & (1<<11)) {
vars->line_speed = SPEED_10000; vars->line_speed = SPEED_10000;
ext_phy_link_up = 1; link_up = 1;
} else { /* Check Legacy speed link */ } else { /* Check Legacy speed link */
u16 legacy_status, legacy_speed; u16 legacy_status, legacy_speed;
/* Enable expansion register 0x42 /* Enable expansion register 0x42 (Operation mode status) */
(Operation mode status) */
bnx2x_cl45_write(bp, phy, bnx2x_cl45_write(bp, phy,
MDIO_AN_DEVAD, MDIO_AN_DEVAD,
MDIO_AN_REG_8481_EXPANSION_REG_ACCESS, MDIO_AN_REG_8481_EXPANSION_REG_ACCESS, 0xf42);
0xf42);
/* Get legacy speed operation status */ /* Get legacy speed operation status */
bnx2x_cl45_read(bp, phy, bnx2x_cl45_read(bp, phy,
...@@ -4677,18 +4292,15 @@ static u8 bnx2x_848xx_read_status(struct bnx2x_phy *phy, ...@@ -4677,18 +4292,15 @@ static u8 bnx2x_848xx_read_status(struct bnx2x_phy *phy,
DP(NETIF_MSG_LINK, "Legacy speed status" DP(NETIF_MSG_LINK, "Legacy speed status"
" = 0x%x\n", legacy_status); " = 0x%x\n", legacy_status);
ext_phy_link_up = ((legacy_status & (1<<11)) link_up = ((legacy_status & (1<<11)) == (1<<11));
== (1<<11)); if (link_up) {
if (ext_phy_link_up) {
legacy_speed = (legacy_status & (3<<9)); legacy_speed = (legacy_status & (3<<9));
if (legacy_speed == (0<<9)) if (legacy_speed == (0<<9))
vars->line_speed = SPEED_10; vars->line_speed = SPEED_10;
else if (legacy_speed == (1<<9)) else if (legacy_speed == (1<<9))
vars->line_speed = vars->line_speed = SPEED_100;
SPEED_100;
else if (legacy_speed == (2<<9)) else if (legacy_speed == (2<<9))
vars->line_speed = vars->line_speed = SPEED_1000;
SPEED_1000;
else /* Should not happen */ else /* Should not happen */
vars->line_speed = 0; vars->line_speed = 0;
...@@ -4697,16 +4309,29 @@ static u8 bnx2x_848xx_read_status(struct bnx2x_phy *phy, ...@@ -4697,16 +4309,29 @@ static u8 bnx2x_848xx_read_status(struct bnx2x_phy *phy,
else else
vars->duplex = DUPLEX_HALF; vars->duplex = DUPLEX_HALF;
DP(NETIF_MSG_LINK, "Link is up " DP(NETIF_MSG_LINK, "Link is up in %dMbps,"
"in %dMbps, is_duplex_full" " is_duplex_full= %d\n", vars->line_speed,
"= %d\n",
vars->line_speed,
(vars->duplex == DUPLEX_FULL)); (vars->duplex == DUPLEX_FULL));
/* Check legacy speed AN resolution */
bnx2x_cl45_read(bp, phy,
MDIO_AN_DEVAD,
MDIO_AN_REG_8481_LEGACY_MII_STATUS,
&val);
if (val & (1<<5))
vars->link_status |=
LINK_STATUS_AUTO_NEGOTIATE_COMPLETE;
bnx2x_cl45_read(bp, phy,
MDIO_AN_DEVAD,
MDIO_AN_REG_8481_LEGACY_AN_EXPANSION,
&val);
if ((val & (1<<0)) == 0)
vars->link_status |=
LINK_STATUS_PARALLEL_DETECTION_USED;
} }
} }
return ext_phy_link_up; return link_up;
} }
static void bnx2x_link_int_enable(struct link_params *params) static void bnx2x_link_int_enable(struct link_params *params)
{ {
u8 port = params->port; u8 port = params->port;
...@@ -4847,7 +4472,6 @@ static void bnx2x_link_int_ack(struct link_params *params, ...@@ -4847,7 +4472,6 @@ static void bnx2x_link_int_ack(struct link_params *params,
NIG_STATUS_SERDES0_LINK_STATUS); NIG_STATUS_SERDES0_LINK_STATUS);
} }
} else { /* link_down */
} }
} }
...@@ -4933,13 +4557,12 @@ u8 bnx2x_get_ext_phy_fw_version(struct link_params *params, u8 driver_loaded, ...@@ -4933,13 +4557,12 @@ u8 bnx2x_get_ext_phy_fw_version(struct link_params *params, u8 driver_loaded,
} }
static void bnx2x_set_xgxs_loopback(struct bnx2x_phy *phy, static void bnx2x_set_xgxs_loopback(struct bnx2x_phy *phy,
struct link_params *params, struct link_params *params)
u8 is_10g)
{ {
u8 port = params->port; u8 port = params->port;
struct bnx2x *bp = params->bp; struct bnx2x *bp = params->bp;
if (is_10g) { if (phy->req_line_speed != SPEED_1000) {
u32 md_devad; u32 md_devad;
DP(NETIF_MSG_LINK, "XGXS 10G loopback enable\n"); DP(NETIF_MSG_LINK, "XGXS 10G loopback enable\n");
...@@ -5714,7 +5337,7 @@ static u8 bnx2x_update_link_up(struct link_params *params, ...@@ -5714,7 +5337,7 @@ static u8 bnx2x_update_link_up(struct link_params *params,
* - DUAL_MEDIA - The link between the 577xx and the first * - DUAL_MEDIA - The link between the 577xx and the first
* external phy needs to be up, and at least one of the 2 * external phy needs to be up, and at least one of the 2
* external phy link must be up. * external phy link must be up.
*/ */
u8 bnx2x_link_update(struct link_params *params, struct link_vars *vars) u8 bnx2x_link_update(struct link_params *params, struct link_vars *vars)
{ {
struct bnx2x *bp = params->bp; struct bnx2x *bp = params->bp;
...@@ -6084,7 +5707,7 @@ static struct bnx2x_phy phy_8073 = { ...@@ -6084,7 +5707,7 @@ static struct bnx2x_phy phy_8073 = {
.speed_cap_mask = 0, .speed_cap_mask = 0,
.req_duplex = 0, .req_duplex = 0,
.rsrv = 0, .rsrv = 0,
.config_init = (config_init_t)bnx2x_8072_8073_config_init, .config_init = (config_init_t)bnx2x_8073_config_init,
.read_status = (read_status_t)bnx2x_8073_read_status, .read_status = (read_status_t)bnx2x_8073_read_status,
.link_reset = (link_reset_t)bnx2x_8073_link_reset, .link_reset = (link_reset_t)bnx2x_8073_link_reset,
.config_loopback = (config_loopback_t)NULL, .config_loopback = (config_loopback_t)NULL,
...@@ -6427,6 +6050,7 @@ static u8 bnx2x_populate_ext_phy(struct bnx2x *bp, ...@@ -6427,6 +6050,7 @@ static u8 bnx2x_populate_ext_phy(struct bnx2x *bp,
phy->addr = XGXS_EXT_PHY_ADDR(ext_phy_config); phy->addr = XGXS_EXT_PHY_ADDR(ext_phy_config);
bnx2x_populate_preemphasis(bp, shmem_base, phy, port, phy_index); bnx2x_populate_preemphasis(bp, shmem_base, phy, port, phy_index);
phy->mdio_ctrl = bnx2x_get_emac_base(bp, phy->type, port); phy->mdio_ctrl = bnx2x_get_emac_base(bp, phy->type, port);
return 0; return 0;
} }
......
...@@ -4348,7 +4348,6 @@ static int bnx2x_init_port(struct bnx2x *bp) ...@@ -4348,7 +4348,6 @@ static int bnx2x_init_port(struct bnx2x *bp)
REG_WR(bp, reg_addr, val); REG_WR(bp, reg_addr, val);
} }
break; break;
case PORT_HW_CFG_XGXS_EXT_PHY_TYPE_BCM8072:
case PORT_HW_CFG_XGXS_EXT_PHY_TYPE_BCM8073: case PORT_HW_CFG_XGXS_EXT_PHY_TYPE_BCM8073:
bp->port.need_hw_lock = 1; bp->port.need_hw_lock = 1;
break; break;
......
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