mirror of
https://mirrors.bfsu.edu.cn/git/linux.git
synced 2024-11-13 23:24:05 +08:00
[BNX2]: Re-structure the 2.5G Serdes code.
Add some common procedures to handle enabling and disabling 2.5G. Add some missing code to resolve flow control. Signed-off-by: Michael Chan <mchan@broadcom.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
ca58c3af99
commit
605a9e20aa
@ -898,6 +898,86 @@ bnx2_set_mac_link(struct bnx2 *bp)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
bnx2_test_and_enable_2g5(struct bnx2 *bp)
|
||||||
|
{
|
||||||
|
u32 up1;
|
||||||
|
int ret = 1;
|
||||||
|
|
||||||
|
if (!(bp->phy_flags & PHY_2_5G_CAPABLE_FLAG))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
if (bp->autoneg & AUTONEG_SPEED)
|
||||||
|
bp->advertising |= ADVERTISED_2500baseX_Full;
|
||||||
|
|
||||||
|
bnx2_read_phy(bp, bp->mii_up1, &up1);
|
||||||
|
if (!(up1 & BCM5708S_UP1_2G5)) {
|
||||||
|
up1 |= BCM5708S_UP1_2G5;
|
||||||
|
bnx2_write_phy(bp, bp->mii_up1, up1);
|
||||||
|
ret = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int
|
||||||
|
bnx2_test_and_disable_2g5(struct bnx2 *bp)
|
||||||
|
{
|
||||||
|
u32 up1;
|
||||||
|
int ret = 0;
|
||||||
|
|
||||||
|
if (!(bp->phy_flags & PHY_2_5G_CAPABLE_FLAG))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
bnx2_read_phy(bp, bp->mii_up1, &up1);
|
||||||
|
if (up1 & BCM5708S_UP1_2G5) {
|
||||||
|
up1 &= ~BCM5708S_UP1_2G5;
|
||||||
|
bnx2_write_phy(bp, bp->mii_up1, up1);
|
||||||
|
ret = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
bnx2_enable_forced_2g5(struct bnx2 *bp)
|
||||||
|
{
|
||||||
|
u32 bmcr;
|
||||||
|
|
||||||
|
if (!(bp->phy_flags & PHY_2_5G_CAPABLE_FLAG))
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (CHIP_NUM(bp) == CHIP_NUM_5708) {
|
||||||
|
bnx2_read_phy(bp, bp->mii_bmcr, &bmcr);
|
||||||
|
bmcr |= BCM5708S_BMCR_FORCE_2500;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (bp->autoneg & AUTONEG_SPEED) {
|
||||||
|
bmcr &= ~BMCR_ANENABLE;
|
||||||
|
if (bp->req_duplex == DUPLEX_FULL)
|
||||||
|
bmcr |= BMCR_FULLDPLX;
|
||||||
|
}
|
||||||
|
bnx2_write_phy(bp, bp->mii_bmcr, bmcr);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
bnx2_disable_forced_2g5(struct bnx2 *bp)
|
||||||
|
{
|
||||||
|
u32 bmcr;
|
||||||
|
|
||||||
|
if (!(bp->phy_flags & PHY_2_5G_CAPABLE_FLAG))
|
||||||
|
return;
|
||||||
|
|
||||||
|
if (CHIP_NUM(bp) == CHIP_NUM_5708) {
|
||||||
|
bnx2_read_phy(bp, bp->mii_bmcr, &bmcr);
|
||||||
|
bmcr &= ~BCM5708S_BMCR_FORCE_2500;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (bp->autoneg & AUTONEG_SPEED)
|
||||||
|
bmcr |= BMCR_SPEED1000 | BMCR_ANENABLE | BMCR_ANRESTART;
|
||||||
|
bnx2_write_phy(bp, bp->mii_bmcr, bmcr);
|
||||||
|
}
|
||||||
|
|
||||||
static int
|
static int
|
||||||
bnx2_set_link(struct bnx2 *bp)
|
bnx2_set_link(struct bnx2 *bp)
|
||||||
{
|
{
|
||||||
@ -941,17 +1021,9 @@ bnx2_set_link(struct bnx2 *bp)
|
|||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
if ((bp->phy_flags & PHY_SERDES_FLAG) &&
|
if ((bp->phy_flags & PHY_SERDES_FLAG) &&
|
||||||
(bp->autoneg & AUTONEG_SPEED)) {
|
(bp->autoneg & AUTONEG_SPEED))
|
||||||
|
bnx2_disable_forced_2g5(bp);
|
||||||
|
|
||||||
u32 bmcr;
|
|
||||||
|
|
||||||
bnx2_read_phy(bp, MII_BMCR, &bmcr);
|
|
||||||
bmcr &= ~BCM5708S_BMCR_FORCE_2500;
|
|
||||||
if (!(bmcr & BMCR_ANENABLE)) {
|
|
||||||
bnx2_write_phy(bp, MII_BMCR, bmcr |
|
|
||||||
BMCR_ANENABLE);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
bp->phy_flags &= ~PHY_PARALLEL_DETECT_FLAG;
|
bp->phy_flags &= ~PHY_PARALLEL_DETECT_FLAG;
|
||||||
bp->link_up = 0;
|
bp->link_up = 0;
|
||||||
}
|
}
|
||||||
@ -1026,34 +1098,32 @@ bnx2_phy_get_pause_adv(struct bnx2 *bp)
|
|||||||
static int
|
static int
|
||||||
bnx2_setup_serdes_phy(struct bnx2 *bp)
|
bnx2_setup_serdes_phy(struct bnx2 *bp)
|
||||||
{
|
{
|
||||||
u32 adv, bmcr, up1;
|
u32 adv, bmcr;
|
||||||
u32 new_adv = 0;
|
u32 new_adv = 0;
|
||||||
|
|
||||||
if (!(bp->autoneg & AUTONEG_SPEED)) {
|
if (!(bp->autoneg & AUTONEG_SPEED)) {
|
||||||
u32 new_bmcr;
|
u32 new_bmcr;
|
||||||
int force_link_down = 0;
|
int force_link_down = 0;
|
||||||
|
|
||||||
|
if (bp->req_line_speed == SPEED_2500) {
|
||||||
|
if (!bnx2_test_and_enable_2g5(bp))
|
||||||
|
force_link_down = 1;
|
||||||
|
} else if (bp->req_line_speed == SPEED_1000) {
|
||||||
|
if (bnx2_test_and_disable_2g5(bp))
|
||||||
|
force_link_down = 1;
|
||||||
|
}
|
||||||
bnx2_read_phy(bp, bp->mii_adv, &adv);
|
bnx2_read_phy(bp, bp->mii_adv, &adv);
|
||||||
adv &= ~(ADVERTISE_1000XFULL | ADVERTISE_1000XHALF);
|
adv &= ~(ADVERTISE_1000XFULL | ADVERTISE_1000XHALF);
|
||||||
|
|
||||||
bnx2_read_phy(bp, bp->mii_bmcr, &bmcr);
|
bnx2_read_phy(bp, bp->mii_bmcr, &bmcr);
|
||||||
new_bmcr = bmcr & ~(BMCR_ANENABLE | BCM5708S_BMCR_FORCE_2500);
|
new_bmcr = bmcr & ~BMCR_ANENABLE;
|
||||||
new_bmcr |= BMCR_SPEED1000;
|
new_bmcr |= BMCR_SPEED1000;
|
||||||
if (bp->req_line_speed == SPEED_2500) {
|
|
||||||
new_bmcr |= BCM5708S_BMCR_FORCE_2500;
|
if (CHIP_NUM(bp) == CHIP_NUM_5708) {
|
||||||
bnx2_read_phy(bp, BCM5708S_UP1, &up1);
|
if (bp->req_line_speed == SPEED_2500)
|
||||||
if (!(up1 & BCM5708S_UP1_2G5)) {
|
new_bmcr |= BCM5708S_BMCR_FORCE_2500;
|
||||||
up1 |= BCM5708S_UP1_2G5;
|
else
|
||||||
bnx2_write_phy(bp, BCM5708S_UP1, up1);
|
new_bmcr = bmcr & ~BCM5708S_BMCR_FORCE_2500;
|
||||||
force_link_down = 1;
|
|
||||||
}
|
|
||||||
} else if (CHIP_NUM(bp) == CHIP_NUM_5708) {
|
|
||||||
bnx2_read_phy(bp, BCM5708S_UP1, &up1);
|
|
||||||
if (up1 & BCM5708S_UP1_2G5) {
|
|
||||||
up1 &= ~BCM5708S_UP1_2G5;
|
|
||||||
bnx2_write_phy(bp, BCM5708S_UP1, up1);
|
|
||||||
force_link_down = 1;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (bp->req_duplex == DUPLEX_FULL) {
|
if (bp->req_duplex == DUPLEX_FULL) {
|
||||||
@ -1080,15 +1150,14 @@ bnx2_setup_serdes_phy(struct bnx2 *bp)
|
|||||||
}
|
}
|
||||||
bnx2_write_phy(bp, bp->mii_adv, adv);
|
bnx2_write_phy(bp, bp->mii_adv, adv);
|
||||||
bnx2_write_phy(bp, bp->mii_bmcr, new_bmcr);
|
bnx2_write_phy(bp, bp->mii_bmcr, new_bmcr);
|
||||||
|
} else {
|
||||||
|
bnx2_resolve_flow_ctrl(bp);
|
||||||
|
bnx2_set_mac_link(bp);
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (bp->phy_flags & PHY_2_5G_CAPABLE_FLAG) {
|
bnx2_test_and_enable_2g5(bp);
|
||||||
bnx2_read_phy(bp, BCM5708S_UP1, &up1);
|
|
||||||
up1 |= BCM5708S_UP1_2G5;
|
|
||||||
bnx2_write_phy(bp, BCM5708S_UP1, up1);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (bp->advertising & ADVERTISED_1000baseT_Full)
|
if (bp->advertising & ADVERTISED_1000baseT_Full)
|
||||||
new_adv |= ADVERTISE_1000XFULL;
|
new_adv |= ADVERTISE_1000XFULL;
|
||||||
@ -1122,6 +1191,9 @@ bnx2_setup_serdes_phy(struct bnx2 *bp)
|
|||||||
bp->current_interval = SERDES_AN_TIMEOUT;
|
bp->current_interval = SERDES_AN_TIMEOUT;
|
||||||
bp->serdes_an_pending = 1;
|
bp->serdes_an_pending = 1;
|
||||||
mod_timer(&bp->timer, jiffies + bp->current_interval);
|
mod_timer(&bp->timer, jiffies + bp->current_interval);
|
||||||
|
} else {
|
||||||
|
bnx2_resolve_flow_ctrl(bp);
|
||||||
|
bnx2_set_mac_link(bp);
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
@ -4300,16 +4372,11 @@ bnx2_5708_serdes_timer(struct bnx2 *bp)
|
|||||||
u32 bmcr;
|
u32 bmcr;
|
||||||
|
|
||||||
bnx2_read_phy(bp, bp->mii_bmcr, &bmcr);
|
bnx2_read_phy(bp, bp->mii_bmcr, &bmcr);
|
||||||
|
|
||||||
if (bmcr & BMCR_ANENABLE) {
|
if (bmcr & BMCR_ANENABLE) {
|
||||||
bmcr &= ~BMCR_ANENABLE;
|
bnx2_enable_forced_2g5(bp);
|
||||||
bmcr |= BMCR_FULLDPLX | BCM5708S_BMCR_FORCE_2500;
|
|
||||||
bnx2_write_phy(bp, MII_BMCR, bmcr);
|
|
||||||
bp->current_interval = SERDES_FORCED_TIMEOUT;
|
bp->current_interval = SERDES_FORCED_TIMEOUT;
|
||||||
} else {
|
} else {
|
||||||
bmcr &= ~(BMCR_FULLDPLX | BCM5708S_BMCR_FORCE_2500);
|
bnx2_disable_forced_2g5(bp);
|
||||||
bmcr |= BMCR_ANENABLE;
|
|
||||||
bnx2_write_phy(bp, MII_BMCR, bmcr);
|
|
||||||
bp->serdes_an_pending = 2;
|
bp->serdes_an_pending = 2;
|
||||||
bp->current_interval = bp->timer_interval;
|
bp->current_interval = bp->timer_interval;
|
||||||
}
|
}
|
||||||
@ -4778,6 +4845,8 @@ bnx2_get_settings(struct net_device *dev, struct ethtool_cmd *cmd)
|
|||||||
if (bp->phy_flags & PHY_SERDES_FLAG) {
|
if (bp->phy_flags & PHY_SERDES_FLAG) {
|
||||||
cmd->supported |= SUPPORTED_1000baseT_Full |
|
cmd->supported |= SUPPORTED_1000baseT_Full |
|
||||||
SUPPORTED_FIBRE;
|
SUPPORTED_FIBRE;
|
||||||
|
if (bp->phy_flags & PHY_2_5G_CAPABLE_FLAG)
|
||||||
|
cmd->supported |= SUPPORTED_2500baseX_Full;
|
||||||
|
|
||||||
cmd->port = PORT_FIBRE;
|
cmd->port = PORT_FIBRE;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user