204f3f5987
The patch, which adds multiphy support, adds new path for non-standard PHYs (e.g. MV88E6060 switch IC) to avoid using kernel phy framework. All work well except the link status traking (Duplex and Speed), which is reseted as soon as PHY connection procedure is done. This leads to lost of the link status of non-standard PHY, which is configured exactly in the ixp4xx_phy_connect() function. Move the generic reset of a link state to the ixp4xx_phy_connect() function to the code path, which is intended for handling of a normal PHY. Reported-by: Nerijus Baliunas <nerijus@users.sourceforge.net> Tested-by: Nerijus Baliunas <nerijus@users.sourceforge.net> Signed-off-by: Sergey Ryazanov <ryazanov.s.a@gmail.com>
168 lines
3.8 KiB
Diff
168 lines
3.8 KiB
Diff
TODO: take care of additional PHYs through the PHY abstraction layer
|
|
|
|
--- a/arch/arm/mach-ixp4xx/include/mach/platform.h
|
|
+++ b/arch/arm/mach-ixp4xx/include/mach/platform.h
|
|
@@ -95,12 +95,23 @@ struct ixp4xx_pata_data {
|
|
#define IXP4XX_ETH_NPEB 0x10
|
|
#define IXP4XX_ETH_NPEC 0x20
|
|
|
|
+#define IXP4XX_ETH_PHY_MAX_ADDR 32
|
|
+
|
|
/* Information about built-in Ethernet MAC interfaces */
|
|
struct eth_plat_info {
|
|
u8 phy; /* MII PHY ID, 0 - 31 */
|
|
u8 rxq; /* configurable, currently 0 - 31 only */
|
|
u8 txreadyq;
|
|
u8 hwaddr[6];
|
|
+
|
|
+ u32 phy_mask;
|
|
+#if 0
|
|
+ int speed;
|
|
+ int duplex;
|
|
+#else
|
|
+ int speed_10;
|
|
+ int half_duplex;
|
|
+#endif
|
|
};
|
|
|
|
/* Information about built-in HSS (synchronous serial) interfaces */
|
|
--- a/drivers/net/ethernet/xscale/ixp4xx_eth.c
|
|
+++ b/drivers/net/ethernet/xscale/ixp4xx_eth.c
|
|
@@ -610,6 +610,37 @@ static int ixp4xx_phy_connect(struct net
|
|
struct eth_plat_info *plat = port->plat;
|
|
char phy_id[MII_BUS_ID_SIZE + 3];
|
|
|
|
+ if (plat->phy == IXP4XX_ETH_PHY_MAX_ADDR) {
|
|
+#if 0
|
|
+ switch (plat->speed) {
|
|
+ case SPEED_10:
|
|
+ case SPEED_100:
|
|
+ break;
|
|
+ default:
|
|
+ printk(KERN_ERR "%s: invalid speed (%d)\n",
|
|
+ dev->name, plat->speed);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+
|
|
+ switch (plat->duplex) {
|
|
+ case DUPLEX_HALF:
|
|
+ case DUPLEX_FULL:
|
|
+ break;
|
|
+ default:
|
|
+ printk(KERN_ERR "%s: invalid duplex mode (%d)\n",
|
|
+ dev->name, plat->duplex);
|
|
+ return -EINVAL;
|
|
+ }
|
|
+ port->speed = plat->speed;
|
|
+ port->duplex = plat->duplex;
|
|
+#else
|
|
+ port->speed = plat->speed_10 ? SPEED_10 : SPEED_100;
|
|
+ port->duplex = plat->half_duplex ? DUPLEX_HALF : DUPLEX_FULL;
|
|
+#endif
|
|
+
|
|
+ return 0;
|
|
+ }
|
|
+
|
|
snprintf(phy_id, MII_BUS_ID_SIZE + 3, PHY_ID_FMT,
|
|
mdio_bus->id, plat->phy);
|
|
port->phydev = phy_connect(dev, phy_id, &ixp4xx_adjust_link,
|
|
@@ -625,6 +656,10 @@ static int ixp4xx_phy_connect(struct net
|
|
|
|
port->phydev->irq = PHY_POLL;
|
|
|
|
+ port->link = 0;
|
|
+ port->speed = 0;
|
|
+ port->duplex = -1;
|
|
+
|
|
return 0;
|
|
}
|
|
|
|
@@ -632,21 +667,32 @@ static void ixp4xx_phy_disconnect(struct
|
|
{
|
|
struct port *port = netdev_priv(dev);
|
|
|
|
- phy_disconnect(port->phydev);
|
|
+ if (port->phydev)
|
|
+ phy_disconnect(port->phydev);
|
|
}
|
|
|
|
static void ixp4xx_phy_start(struct net_device *dev)
|
|
{
|
|
struct port *port = netdev_priv(dev);
|
|
|
|
- phy_start(port->phydev);
|
|
+ if (port->phydev) {
|
|
+ phy_start(port->phydev);
|
|
+ } else {
|
|
+ port->link = 1;
|
|
+ ixp4xx_update_link(dev);
|
|
+ }
|
|
}
|
|
|
|
static void ixp4xx_phy_stop(struct net_device *dev)
|
|
{
|
|
struct port *port = netdev_priv(dev);
|
|
|
|
- phy_stop(port->phydev);
|
|
+ if (port->phydev) {
|
|
+ phy_stop(port->phydev);
|
|
+ } else {
|
|
+ port->link = 0;
|
|
+ ixp4xx_update_link(dev);
|
|
+ }
|
|
}
|
|
|
|
static inline void debug_pkt(struct net_device *dev, const char *func,
|
|
@@ -1048,6 +1094,9 @@ static int eth_ioctl(struct net_device *
|
|
return hwtstamp_get(dev, req);
|
|
}
|
|
|
|
+ if (!port->phydev)
|
|
+ return -EOPNOTSUPP;
|
|
+
|
|
return phy_mii_ioctl(port->phydev, req, cmd);
|
|
}
|
|
|
|
@@ -1068,18 +1117,30 @@ static void ixp4xx_get_drvinfo(struct ne
|
|
static int ixp4xx_get_settings(struct net_device *dev, struct ethtool_cmd *cmd)
|
|
{
|
|
struct port *port = netdev_priv(dev);
|
|
+
|
|
+ if (!port->phydev)
|
|
+ return -EOPNOTSUPP;
|
|
+
|
|
return phy_ethtool_gset(port->phydev, cmd);
|
|
}
|
|
|
|
static int ixp4xx_set_settings(struct net_device *dev, struct ethtool_cmd *cmd)
|
|
{
|
|
struct port *port = netdev_priv(dev);
|
|
+
|
|
+ if (!port->phydev)
|
|
+ return -EOPNOTSUPP;
|
|
+
|
|
return phy_ethtool_sset(port->phydev, cmd);
|
|
}
|
|
|
|
static int ixp4xx_nway_reset(struct net_device *dev)
|
|
{
|
|
struct port *port = netdev_priv(dev);
|
|
+
|
|
+ if (!port->phydev)
|
|
+ return -EOPNOTSUPP;
|
|
+
|
|
return phy_start_aneg(port->phydev);
|
|
}
|
|
|
|
@@ -1529,10 +1590,6 @@ static int eth_init_one(struct platform_
|
|
if ((err = register_netdev(dev)))
|
|
goto err_phy_dis;
|
|
|
|
- port->link = 0;
|
|
- port->speed = 0;
|
|
- port->duplex = -1;
|
|
-
|
|
printk(KERN_INFO "%s: MII PHY %i on %s\n", dev->name, plat->phy,
|
|
npe_name(port->npe));
|
|
|