r8169: link speed selection timer rework.
The implementation was a bit krusty. The 10s rtl8169_phy_timer timer has been (was ?) required with older 8169 for adequate phy operation when full gigabit is advertised in autonegotiated mode. The timer does nothing if the link is up. Otherwise it keeps resetting the phy until things improve. - the device private data field phy_1000_ctrl_reg was used to schedule the timer. Avoid it and save a few bytes. - rtl8169_set_settings pending timer is disabled before changing the link settings as rtl8169_phy_timer is not always needed (see the removed test in rtl8169_phy_timer). - rtl8169_set_speed the requested link parameters may not match the chipset : bail out early on failure. - rtl8169_open Calling rtl8169_request_timer is redundant with -> rtl8169_open -> rtl8169_init_phy -> rtl8169_set_speed -> mod_timer The latter always enables the phy timer whereas the former did not for RTL_GIGA_MAC_VER_01. It should not make things worse but only time will tell if reality agrees. - rtl8169_request_timer : unused yet. Removed. - rtl8169_delete_timer : useless. Bloat. Removed. Side effect : the timer may kick in if the TBI is enabled. I do not know if the TBI has ever been used in real life. Signed-off-by: Francois Romieu <romieu@fr.zoreil.com> Cc: Realtek linux nic maintainers <nic_swsd@realtek.com>
This commit is contained in:
parent
826e6cbdad
commit
4876cc1e49
|
@ -616,7 +616,6 @@ struct rtl8169_private {
|
||||||
u16 intr_event;
|
u16 intr_event;
|
||||||
u16 napi_event;
|
u16 napi_event;
|
||||||
u16 intr_mask;
|
u16 intr_mask;
|
||||||
int phy_1000_ctrl_reg;
|
|
||||||
|
|
||||||
struct mdio_ops {
|
struct mdio_ops {
|
||||||
void (*write)(void __iomem *, int, int);
|
void (*write)(void __iomem *, int, int);
|
||||||
|
@ -1288,8 +1287,6 @@ static int rtl8169_set_speed_xmii(struct net_device *dev,
|
||||||
bmcr |= BMCR_FULLDPLX;
|
bmcr |= BMCR_FULLDPLX;
|
||||||
}
|
}
|
||||||
|
|
||||||
tp->phy_1000_ctrl_reg = giga_ctrl;
|
|
||||||
|
|
||||||
rtl_writephy(tp, MII_BMCR, bmcr);
|
rtl_writephy(tp, MII_BMCR, bmcr);
|
||||||
|
|
||||||
if (tp->mac_version == RTL_GIGA_MAC_VER_02 ||
|
if (tp->mac_version == RTL_GIGA_MAC_VER_02 ||
|
||||||
|
@ -1315,10 +1312,14 @@ static int rtl8169_set_speed(struct net_device *dev,
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
ret = tp->set_speed(dev, autoneg, speed, duplex, advertising);
|
ret = tp->set_speed(dev, autoneg, speed, duplex, advertising);
|
||||||
|
if (ret < 0)
|
||||||
|
goto out;
|
||||||
|
|
||||||
if (netif_running(dev) && (tp->phy_1000_ctrl_reg & ADVERTISE_1000FULL))
|
if (netif_running(dev) && (autoneg == AUTONEG_ENABLE) &&
|
||||||
|
(advertising & ADVERTISED_1000baseT_Full)) {
|
||||||
mod_timer(&tp->timer, jiffies + RTL8169_PHY_TIMEOUT);
|
mod_timer(&tp->timer, jiffies + RTL8169_PHY_TIMEOUT);
|
||||||
|
}
|
||||||
|
out:
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1328,6 +1329,8 @@ static int rtl8169_set_settings(struct net_device *dev, struct ethtool_cmd *cmd)
|
||||||
unsigned long flags;
|
unsigned long flags;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
|
del_timer_sync(&tp->timer);
|
||||||
|
|
||||||
spin_lock_irqsave(&tp->lock, flags);
|
spin_lock_irqsave(&tp->lock, flags);
|
||||||
ret = rtl8169_set_speed(dev, cmd->autoneg, ethtool_cmd_speed(cmd),
|
ret = rtl8169_set_speed(dev, cmd->autoneg, ethtool_cmd_speed(cmd),
|
||||||
cmd->duplex, cmd->advertising);
|
cmd->duplex, cmd->advertising);
|
||||||
|
@ -2691,9 +2694,6 @@ static void rtl8169_phy_timer(unsigned long __opaque)
|
||||||
|
|
||||||
assert(tp->mac_version > RTL_GIGA_MAC_VER_01);
|
assert(tp->mac_version > RTL_GIGA_MAC_VER_01);
|
||||||
|
|
||||||
if (!(tp->phy_1000_ctrl_reg & ADVERTISE_1000FULL))
|
|
||||||
return;
|
|
||||||
|
|
||||||
spin_lock_irq(&tp->lock);
|
spin_lock_irq(&tp->lock);
|
||||||
|
|
||||||
if (tp->phy_reset_pending(tp)) {
|
if (tp->phy_reset_pending(tp)) {
|
||||||
|
@ -2718,28 +2718,6 @@ out_unlock:
|
||||||
spin_unlock_irq(&tp->lock);
|
spin_unlock_irq(&tp->lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void rtl8169_delete_timer(struct net_device *dev)
|
|
||||||
{
|
|
||||||
struct rtl8169_private *tp = netdev_priv(dev);
|
|
||||||
struct timer_list *timer = &tp->timer;
|
|
||||||
|
|
||||||
if (tp->mac_version <= RTL_GIGA_MAC_VER_01)
|
|
||||||
return;
|
|
||||||
|
|
||||||
del_timer_sync(timer);
|
|
||||||
}
|
|
||||||
|
|
||||||
static inline void rtl8169_request_timer(struct net_device *dev)
|
|
||||||
{
|
|
||||||
struct rtl8169_private *tp = netdev_priv(dev);
|
|
||||||
struct timer_list *timer = &tp->timer;
|
|
||||||
|
|
||||||
if (tp->mac_version <= RTL_GIGA_MAC_VER_01)
|
|
||||||
return;
|
|
||||||
|
|
||||||
mod_timer(timer, jiffies + RTL8169_PHY_TIMEOUT);
|
|
||||||
}
|
|
||||||
|
|
||||||
#ifdef CONFIG_NET_POLL_CONTROLLER
|
#ifdef CONFIG_NET_POLL_CONTROLLER
|
||||||
/*
|
/*
|
||||||
* Polling 'interrupt' - used by things like netconsole to send skbs
|
* Polling 'interrupt' - used by things like netconsole to send skbs
|
||||||
|
@ -3396,8 +3374,6 @@ rtl8169_init_one(struct pci_dev *pdev, const struct pci_device_id *ent)
|
||||||
tp->phy_reset_pending = rtl8169_tbi_reset_pending;
|
tp->phy_reset_pending = rtl8169_tbi_reset_pending;
|
||||||
tp->link_ok = rtl8169_tbi_link_ok;
|
tp->link_ok = rtl8169_tbi_link_ok;
|
||||||
tp->do_ioctl = rtl_tbi_ioctl;
|
tp->do_ioctl = rtl_tbi_ioctl;
|
||||||
|
|
||||||
tp->phy_1000_ctrl_reg = ADVERTISE_1000FULL; /* Implied by TBI */
|
|
||||||
} else {
|
} else {
|
||||||
tp->set_speed = rtl8169_set_speed_xmii;
|
tp->set_speed = rtl8169_set_speed_xmii;
|
||||||
tp->get_settings = rtl8169_gset_xmii;
|
tp->get_settings = rtl8169_gset_xmii;
|
||||||
|
@ -3593,8 +3569,6 @@ static int rtl8169_open(struct net_device *dev)
|
||||||
|
|
||||||
rtl_hw_start(dev);
|
rtl_hw_start(dev);
|
||||||
|
|
||||||
rtl8169_request_timer(dev);
|
|
||||||
|
|
||||||
tp->saved_wolopts = 0;
|
tp->saved_wolopts = 0;
|
||||||
pm_runtime_put_noidle(&pdev->dev);
|
pm_runtime_put_noidle(&pdev->dev);
|
||||||
|
|
||||||
|
@ -5147,7 +5121,7 @@ static void rtl8169_down(struct net_device *dev)
|
||||||
struct rtl8169_private *tp = netdev_priv(dev);
|
struct rtl8169_private *tp = netdev_priv(dev);
|
||||||
void __iomem *ioaddr = tp->mmio_addr;
|
void __iomem *ioaddr = tp->mmio_addr;
|
||||||
|
|
||||||
rtl8169_delete_timer(dev);
|
del_timer_sync(&tp->timer);
|
||||||
|
|
||||||
netif_stop_queue(dev);
|
netif_stop_queue(dev);
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue