diff options
Diffstat (limited to 'drivers/net/usb/smsc95xx.c')
| -rw-r--r-- | drivers/net/usb/smsc95xx.c | 34 | 
1 files changed, 31 insertions, 3 deletions
diff --git a/drivers/net/usb/smsc95xx.c b/drivers/net/usb/smsc95xx.c index df9179a1c93..3135af63d37 100644 --- a/drivers/net/usb/smsc95xx.c +++ b/drivers/net/usb/smsc95xx.c @@ -28,6 +28,7 @@  #include <linux/usb.h>  #include <linux/crc32.h>  #include <linux/usb/usbnet.h> +#include <linux/slab.h>  #include "smsc95xx.h"  #define SMSC_CHIPNAME			"smsc95xx" @@ -709,6 +710,8 @@ static void smsc95xx_start_rx_path(struct usbnet *dev)  static int smsc95xx_phy_initialize(struct usbnet *dev)  { +	int bmcr, timeout = 0; +  	/* Initialize MII structure */  	dev->mii.dev = dev->net;  	dev->mii.mdio_read = smsc95xx_mdio_read; @@ -717,7 +720,20 @@ static int smsc95xx_phy_initialize(struct usbnet *dev)  	dev->mii.reg_num_mask = 0x1f;  	dev->mii.phy_id = SMSC95XX_INTERNAL_PHY_ID; +	/* reset phy and wait for reset to complete */  	smsc95xx_mdio_write(dev->net, dev->mii.phy_id, MII_BMCR, BMCR_RESET); + +	do { +		msleep(10); +		bmcr = smsc95xx_mdio_read(dev->net, dev->mii.phy_id, MII_BMCR); +		timeout++; +	} while ((bmcr & MII_BMCR) && (timeout < 100)); + +	if (timeout >= 100) { +		netdev_warn(dev->net, "timeout on PHY Reset"); +		return -EIO; +	} +  	smsc95xx_mdio_write(dev->net, dev->mii.phy_id, MII_ADVERTISE,  		ADVERTISE_ALL | ADVERTISE_CSMA | ADVERTISE_PAUSE_CAP |  		ADVERTISE_PAUSE_ASYM); @@ -1174,9 +1190,21 @@ static struct sk_buff *smsc95xx_tx_fixup(struct usbnet *dev,  	}  	if (csum) { -		u32 csum_preamble = smsc95xx_calc_csum_preamble(skb); -		skb_push(skb, 4); -		memcpy(skb->data, &csum_preamble, 4); +		if (skb->len <= 45) { +			/* workaround - hardware tx checksum does not work +			 * properly with extremely small packets */ +			long csstart = skb->csum_start - skb_headroom(skb); +			__wsum calc = csum_partial(skb->data + csstart, +				skb->len - csstart, 0); +			*((__sum16 *)(skb->data + csstart +				+ skb->csum_offset)) = csum_fold(calc); + +			csum = false; +		} else { +			u32 csum_preamble = smsc95xx_calc_csum_preamble(skb); +			skb_push(skb, 4); +			memcpy(skb->data, &csum_preamble, 4); +		}  	}  	skb_push(skb, 4);  |