Commit e85c9d11 authored by Andrew Morton's avatar Andrew Morton Committed by Linus Torvalds

[PATCH] sis900-fix-phy-transceiver-detection.patch

From: Daniele Venzano <webvenza@libero.it>

Fix PHY transceiver detection code to fall back to known PHY and not to the
last detected.

The code checks every transceiver detected for link status and type, but fails
when ghost transceivers are detected, deciding to use the last one detected.

With this patch the driver should choose the correct transceiver even when
some ghosts are detected by checking for the type of the tranceiver it is
going to use.
parent 29a54a41
...@@ -116,6 +116,7 @@ static struct mii_chip_info { ...@@ -116,6 +116,7 @@ static struct mii_chip_info {
#define HOME 0x0001 #define HOME 0x0001
#define LAN 0x0002 #define LAN 0x0002
#define MIX 0x0003 #define MIX 0x0003
#define UNKNOWN 0x0
} mii_chip_table[] = { } mii_chip_table[] = {
{ "SiS 900 Internal MII PHY", 0x001d, 0x8000, LAN }, { "SiS 900 Internal MII PHY", 0x001d, 0x8000, LAN },
{ "SiS 7014 Physical Layer Solution", 0x0016, 0xf830, LAN }, { "SiS 7014 Physical Layer Solution", 0x0016, 0xf830, LAN },
...@@ -577,9 +578,11 @@ static int __init sis900_mii_probe (struct net_device * net_dev) ...@@ -577,9 +578,11 @@ static int __init sis900_mii_probe (struct net_device * net_dev)
break; break;
} }
if( !mii_chip_table[i].phy_id1 ) if( !mii_chip_table[i].phy_id1 ) {
printk(KERN_INFO "%s: Unknown PHY transceiver found at address %d.\n", printk(KERN_INFO "%s: Unknown PHY transceiver found at address %d.\n",
net_dev->name, phy_addr); net_dev->name, phy_addr);
mii_phy->phy_types = UNKNOWN;
}
} }
if (sis_priv->mii == NULL) { if (sis_priv->mii == NULL) {
...@@ -644,15 +647,15 @@ static int __init sis900_mii_probe (struct net_device * net_dev) ...@@ -644,15 +647,15 @@ static int __init sis900_mii_probe (struct net_device * net_dev)
static u16 sis900_default_phy(struct net_device * net_dev) static u16 sis900_default_phy(struct net_device * net_dev)
{ {
struct sis900_private * sis_priv = net_dev->priv; struct sis900_private * sis_priv = net_dev->priv;
struct mii_phy *phy = NULL, *phy_home = NULL, *default_phy = NULL; struct mii_phy *phy = NULL, *phy_home = NULL, *default_phy = NULL, *phy_lan = NULL;
u16 status; u16 status;
for( phy=sis_priv->first_mii; phy; phy=phy->next ){ for( phy=sis_priv->first_mii; phy; phy=phy->next ){
status = mdio_read(net_dev, phy->phy_addr, MII_STATUS); status = mdio_read(net_dev, phy->phy_addr, MII_STATUS);
status = mdio_read(net_dev, phy->phy_addr, MII_STATUS); status = mdio_read(net_dev, phy->phy_addr, MII_STATUS);
/* Link ON & Not select deafalut PHY */ /* Link ON & Not select default PHY & not ghost PHY */
if ( (status & MII_STAT_LINK) && !(default_phy) ) if ( (status & MII_STAT_LINK) && !default_phy && (phy->phy_types != UNKNOWN) )
default_phy = phy; default_phy = phy;
else{ else{
status = mdio_read(net_dev, phy->phy_addr, MII_CONTROL); status = mdio_read(net_dev, phy->phy_addr, MII_CONTROL);
...@@ -660,12 +663,16 @@ static u16 sis900_default_phy(struct net_device * net_dev) ...@@ -660,12 +663,16 @@ static u16 sis900_default_phy(struct net_device * net_dev)
status | MII_CNTL_AUTO | MII_CNTL_ISOLATE); status | MII_CNTL_AUTO | MII_CNTL_ISOLATE);
if( phy->phy_types == HOME ) if( phy->phy_types == HOME )
phy_home = phy; phy_home = phy;
else if (phy->phy_types == LAN)
phy_lan = phy;
} }
} }
if( (!default_phy) && phy_home ) if( !default_phy && phy_home )
default_phy = phy_home; default_phy = phy_home;
else if(!default_phy) else if( !default_phy && phy_lan )
default_phy = phy_lan;
else if ( !default_phy )
default_phy = sis_priv->first_mii; default_phy = sis_priv->first_mii;
if( sis_priv->mii != default_phy ){ if( sis_priv->mii != default_phy ){
......
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