smsc911x driver

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Attached is a driver patch for SMSC911x family of ethernet chips,
generated against 2.6.18-rc1 sources. There's a similar driver in the
tree; this one has been tested by SMSC on all flavors of the chip and
claimed to be efficient.

Thanks,
Bahadir

diff --git a/drivers/net/Kconfig b/drivers/net/Kconfig
index 3918990..d158e05 100644
--- a/drivers/net/Kconfig
+++ b/drivers/net/Kconfig
@@ -865,6 +865,13 @@ config NET_NETX
	  <file:Documentation/networking/net-modules.txt>. The module
	  will be called netx-eth.

+config SMSC911X
+	tristate "SMSC 911x family of embedded ethernet support"
+	depends on NET_ETHERNET
+	---help---
+	  Say Y here if you want support for SMSC LAN911x family
+	  of ethernet chips.
+
config DM9000
	tristate "DM9000 support"
	depends on (ARM || MIPS) && NET_ETHERNET
diff --git a/drivers/net/Makefile b/drivers/net/Makefile
index c91e951..51f680b 100644
--- a/drivers/net/Makefile
+++ b/drivers/net/Makefile
@@ -196,6 +196,7 @@ obj-$(CONFIG_S2IO) += s2io.o
obj-$(CONFIG_MYRI10GE) += myri10ge/
obj-$(CONFIG_SMC91X) += smc91x.o
obj-$(CONFIG_SMC911X) += smc911x.o
+obj-$(CONFIG_SMSC911X) += smsc911x.o
obj-$(CONFIG_DM9000) += dm9000.o
obj-$(CONFIG_FEC_8XX) += fec_8xx/

diff --git a/drivers/net/smsc911x.c b/drivers/net/smsc911x.c
new file mode 100644
index 0000000..e2f9adb
--- /dev/null
+++ b/drivers/net/smsc911x.c
@@ -0,0 +1,2105 @@
+/***************************************************************************
+ *
+ * Copyright (C) 2004-2005  SMSC
+ * Copyright (C) 2005 ARM
+ *
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version 2
+ * of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
+ *
+ ***************************************************************************
+ * Rewritten, heavily based on smsc911x simple driver by SMSC.
+ * Partly uses io macros from smc91x.c by Nicolas Pitre
+ */
+
+#include <linux/config.h>
+#include <linux/delay.h>
+#include <linux/errno.h>
+#include <linux/etherdevice.h>
+#include <linux/init.h>
+#include <linux/ioport.h>
+#include <linux/kernel.h>
+#include <linux/mii.h>
+#include <linux/module.h>
+#include <linux/netdevice.h>
+#include <linux/platform_device.h>
+#include <linux/sched.h>
+#include <linux/slab.h>
+#include <linux/timer.h>
+#include <linux/version.h>
+#include <asm/bug.h>
+#include <asm/bitops.h>
+#include <asm/dma.h>
+#include <asm/irq.h>
+#include <asm/io.h>
+#include "smsc911x.h"
+
+#define SMSC_CHIPNAME	"smsc911x"
+
+/* Tasklet declarations */
+static unsigned long rx_tasklet_parameter;
+static void smsc911x_rx_tasklet(unsigned long data);
+
+DECLARE_TASKLET(rx_tasklet, smsc911x_rx_tasklet, 0);
+MODULE_LICENSE("GPL");
+
+/* MAC */
+static void smsc911x_mac_initialise(struct smsc911x_data *pdata);
+static int smsc911x_mac_notbusy(struct smsc911x_data *pdata);
+
+static unsigned int smsc911x_mac_read(struct smsc911x_data *pdata,
+				      unsigned int offset);
+
+static void smsc911x_mac_write(struct smsc911x_data *pdata,
+			       unsigned int offset,
+			       unsigned int value);
+
+/* PHY layer */
+static int smsc911x_phy_reset(struct smsc911x_data * pdata);
+static int smsc911x_phy_initialise_external(struct smsc911x_data * pdata);
+static int smsc911x_phy_initialise(struct smsc911x_data *pdata);
+static void smsc911x_phy_setlink(struct smsc911x_data *pdata);
+static unsigned int smsc911x_phy_read(struct smsc911x_data *pdata,
+				      unsigned int offset);
+
+static void smsc911x_phy_write(struct smsc911x_data *pdata,
+			       unsigned int offset,
+			       unsigned int value);
+
+static void smsc911x_phy_upd_lmode_fd(struct smsc911x_data *pdata,
+				      unsigned int, unsigned int);
+
+static void smsc911x_phy_update_linkmode(struct smsc911x_data *pdata);
+static void smsc911x_phy_getlinkmode(struct smsc911x_data *pdata);
+static void smsc911x_phy_checklink(unsigned long);
+static int smsc911x_phy_check_loopbackpkt(struct smsc911x_data * pdata);
+static int smsc911x_phy_loopbacktest(struct smsc911x_data * pdata);
+
+/* Transmit */
+static void smsc911x_tx_initialise(struct smsc911x_data *pdata);
+static void smsc911x_tx_writefifo(struct smsc911x_data *pdata,
+				  unsigned int *bufp,
+				  unsigned int size);
+
+static unsigned int smsc911x_tx_get_txstatcount(struct smsc911x_data *pdata);
+static unsigned int smsc911x_tx_completetx(struct smsc911x_data *pdata);
+static void smsc911x_tx_sendskb(struct smsc911x_data *pdata,
+				struct sk_buff *skb);
+
+static int smsc911x_tx_handleirq(struct smsc911x_data *pdata,
+				 unsigned int irqstatus);
+
+static void smsc911x_tx_update_txcounters(struct smsc911x_data *pdata);
+
+/* Receive */
+static void smsc911x_rx_initialise(struct smsc911x_data *pdata);
+static void smsc911x_rx_readfifo(struct smsc911x_data *pdata,
+				 unsigned int *bufp,
+				 unsigned int size);
+
+static void smsc911x_rx_handoffskb(struct smsc911x_data *pdata,
+				   struct sk_buff *skb);
+
+static unsigned int smsc911x_rx_pop_rxstatus(struct smsc911x_data *pdata);
+static void smsc911x_rx_counterrors(struct smsc911x_data *pdata,
+				    unsigned int rxstatus);
+
+static void smsc911x_rx_fastforward(struct smsc911x_data *pdata,
+				    unsigned int size);
+
+static void smsc911x_rx_processpackets(struct smsc911x_data *pdata);
+static int smsc911x_rx_handleirq(struct smsc911x_data *pdata,
+				 unsigned int intstat);
+
+static unsigned int smsc911x_hash(char addr[]);
+static void smsc911x_rx_set_multicastlist(struct net_device *dev);
+
+/* LAN initialisation */
+static int smsc911x_lan_initialise(struct smsc911x_data *pdata,
+				   unsigned int intcfg);
+
+/* Linux network device interface */
+static int smsc911x_init(struct net_device *dev);
+static int smsc911x_open(struct net_device *dev);
+static int smsc911x_stop(struct net_device *dev);
+static int smsc911x_hard_start_xmit(struct sk_buff *skb,
+				    struct net_device *dev);
+
+static struct net_device_stats * smsc911x_get_stats(struct net_device *dev);
+static void smsc911x_set_multicast_list(struct net_device *dev);
+static irqreturn_t smsc911x_irqhandler(int irq, void *dev_id,
+				       struct pt_regs *regs);
+
+/* Module interface */
+static int smsc911x_init_module(void);
+static void smsc911x_cleanup_module(void);
+
+/* Driver interface */
+static int smsc911x_drv_probe(struct platform_device * pdev);
+static int smsc911x_drv_remove(struct platform_device *pdev);
+
+/* Entry point for starting/opening the interface */
+static int smsc911x_open(struct net_device *dev)
+{
+	struct smsc911x_data *pdata;
+	unsigned char *mac_byte;
+	unsigned int mac_high16;
+	unsigned int mac_low32;
+	unsigned int timeout;
+	unsigned int intcfg;
+	int result;
+
+	/* Set interrupt deassertion to 220uS */
+	intcfg  = 22 << 24;
+	timeout = 1000;
+	result  = -ENODEV;
+	pdata   = 0;
+
+	BUG_ON(!dev);
+	pdata = netdev_priv(dev);
+	BUG_ON(!pdata);
+
+	/* Initialise smsc911x */
+
+	/*
+	 * dwIntCfg|=INT_CFG_IRQ_POL_;  use this to set IRQ_POL bit
+	 * dwIntCfg|=INT_CFG_IRQ_TYPE_;  use this to set IRQ_TYPE bit
+	 */
+	if (!smsc911x_lan_initialise(pdata, intcfg))
+		goto done;
+
+	SMSC_TRACE("Testing irq handler using IRQ %d", dev->irq);
+	pdata->request_irq_disable = 0;
+	pdata->software_irq_signal = 0;
+	SMC_regwrite((SMC_regread(pdata, INT_EN) | INT_EN_SW_INT_EN_),
+		     pdata, INT_EN);
+	do {
+		udelay(10);
+		timeout--;
+	} while (timeout && (!pdata->software_irq_signal));
+
+	if (!pdata->software_irq_signal) {
+		printk("<1>ISR failed signaling test.");
+		result = -ENODEV;
+		goto done;
+	}
+	SMSC_TRACE("IRQ handler passed test using IRQ %d", dev->irq);
+
+	printk("%s: SMSC9%3x identified at %#08x, IRQ: %d\n", dev->name,
+	       (u32)((pdata->idrev >> 16) & 0xFFFF), pdata->base, dev->irq);
+
+	smsc911x_mac_initialise(pdata);
+
+	/* Read mac address from EEPROM */
+	mac_high16 = smsc911x_mac_read(pdata, ADDRH);
+	mac_low32 = smsc911x_mac_read(pdata, ADDRL);
+
+	if ((mac_high16 == 0x0000FFFF) && (mac_low32 == 0xFFFFFFFF)) {
+		/* Use default MAC addresses if eeprom values are invalid */
+		mac_high16 = 0x00000070;
+		mac_low32 = 0x110F8000;
+
+		smsc911x_mac_write(pdata, ADDRH, mac_high16);
+		smsc911x_mac_write(pdata, ADDRL, mac_low32);
+		SMSC_TRACE("MAC Address is set by default to 0x%04X%08X",
+			   mac_high16, mac_low32);
+	} else {
+		SMSC_TRACE("Mac Address is read from LAN911x as 0x%04X%08X",
+			   mac_high16, mac_low32);
+	}
+	mac_byte = (unsigned char *)&mac_low32;
+	dev->dev_addr[0] = mac_byte[0];
+	dev->dev_addr[1] = mac_byte[1];
+	dev->dev_addr[2] = mac_byte[2];
+	dev->dev_addr[3] = mac_byte[3];
+
+	mac_byte = (unsigned char *)&mac_high16;
+	dev->dev_addr[4] = mac_byte[0];
+	dev->dev_addr[5] = mac_byte[1];
+
+	printk("%s: SMSC9%3x MAC Address: %02x:%02x:%02x:%02x:%02x:%02x\n",
+	       dev->name, (u32)((pdata->idrev >> 16) & 0xFFFF),
+	       dev->dev_addr[5], dev->dev_addr[4], dev->dev_addr[3],
+	       dev->dev_addr[2], dev->dev_addr[1], dev->dev_addr[0]);
+
+	netif_carrier_off(dev);
+	if (!smsc911x_phy_initialise(pdata)) {
+		SMSC_WARNING("Failed to initialize PHY");
+		result = -ENODEV;
+		goto done;
+	}
+
+	smsc911x_tx_initialise(pdata);
+	smsc911x_rx_initialise(pdata);
+	netif_start_queue(dev);
+
+	result = 0;
+
+done:
+	SMSC_TRACE("result: %d", result);
+	return result;
+}
+
+/* Entry point for stopping the interface */
+static int smsc911x_stop(struct net_device *dev)
+{
+	unsigned long flags;
+	unsigned long base;
+	unsigned long idrev;
+	struct smsc911x_data *pdata;
+	struct net_device *pnetdev;
+
+	BUG_ON(!dev);
+	pdata = netdev_priv(dev);
+	BUG_ON(!pdata);
+
+	pdata->stop_link_poll = 1;
+	del_timer_sync(&pdata->link_poll_timer);
+
+	spin_lock_irqsave(&dev->_xmit_lock, flags);
+	SMC_regwrite((SMC_regread(pdata, INT_CFG) & (~INT_CFG_IRQ_EN_)),
+		     pdata, INT_CFG);
+	netif_stop_queue(pdata->dev);
+	spin_unlock_irqrestore(&dev->_xmit_lock, flags);
+
+	/* At this point all Rx and Tx activity is stopped */
+	pdata->stats.rx_dropped += SMC_regread(pdata, RX_DROP);
+	smsc911x_tx_update_txcounters(pdata);
+
+	/* Preserve important fields */
+	base = pdata->base;
+	idrev = pdata->idrev;
+	pnetdev = pdata->dev;
+
+	/* Clear all structure */
+	memset((void *)pdata, 0, sizeof(struct smsc911x_data));
+
+	/* Reassign important fields */
+	pdata->base = base;
+	pdata->idrev = idrev;
+	pdata->dev = pnetdev;
+
+	SMSC_TRACE("<--Simp911x_stop");
+	return 0;
+}
+
+/* Entry point for transmitting a packet */
+static int
+smsc911x_hard_start_xmit(struct sk_buff *skb, struct net_device *dev)
+{
+	struct smsc911x_data *pdata;
+
+	BUG_ON(!skb);
+	BUG_ON(!dev);
+	BUG_ON(!netdev_priv(dev));
+
+	pdata = netdev_priv(dev);
+	smsc911x_tx_sendskb(pdata, skb);
+	return 0;
+}
+
+/* Entry point for getting status counters */
+static struct net_device_stats * smsc911x_get_stats(struct net_device *dev)
+{
+	struct smsc911x_data *pdata;
+
+	pdata = netdev_priv(dev);
+	BUG_ON(!netdev_priv(dev));
+
+	return &pdata->stats;
+}
+
+/* Entry point for setting addressing modes */
+static void smsc911x_set_multicast_list(struct net_device *dev)
+{
+	BUG_ON(!dev);
+	smsc911x_rx_set_multicastlist(dev);
+}
+
+
+static irqreturn_t
+smsc911x_irqhandler(int irq, void *dev_id, struct pt_regs *regs)
+{
+	unsigned int intcfg;
+	unsigned int intsts;
+	int serviced;
+	unsigned int reserved_bits;
+	struct smsc911x_data *pdata;
+
+	intcfg = 0;
+	intsts = 0;
+	serviced = IRQ_NONE;
+	reserved_bits = 0x00FFCEEE;
+
+	pdata = dev_id;
+	BUG_ON(!pdata);
+	intcfg = SMC_regread(pdata, INT_CFG);
+
+	if ((intcfg & 0x00001100) != 0x00001100) {
+		SMSC_TRACE("Spurious interrupt, intcfg: 0x%08X", intcfg);
+		goto done;
+	}
+
+	/* this could mean surprise removal */
+	if(intcfg & reserved_bits) {
+		SMSC_WARNING("SMSC911x irq handler, reserved bits are high.");
+		goto done;
+	}
+
+	intsts = SMC_regread(pdata, INT_STS);
+	if (intsts & INT_STS_SW_INT_) {
+		SMC_regwrite((SMC_regread(pdata, INT_EN)
+			     & (~INT_EN_SW_INT_EN_)), pdata, INT_EN);
+		SMC_regwrite(INT_STS_SW_INT_, pdata, INT_STS);
+		pdata->software_irq_signal = 1;
+		serviced = IRQ_HANDLED;
+		if (pdata->request_irq_disable) {
+			SMC_regwrite((SMC_regread(pdata, INT_CFG)
+				     & (~INT_CFG_IRQ_EN_)), pdata, INT_CFG);
+			/* Prevent irqs from being handled */
+			intsts = 0;
+		}
+	}
+
+	if (smsc911x_tx_handleirq(pdata, intsts))
+		serviced = IRQ_HANDLED;
+
+	if (smsc911x_rx_handleirq(pdata, intsts))
+		serviced = IRQ_HANDLED;
+
+	if (!serviced)
+		SMSC_WARNING("Unserviced irq. intcfg: 0x%08X, intsts: "
+			     "0x%08X, int_en: 0x%08X", intcfg, intsts,
+			     SMC_regread(pdata, INT_EN));
+done:
+	return serviced;
+}
+
+/* Autodetects and initialises external phy for SMSC9115 and SMSC9117 flavors.
+ * If something goes wrong, returns -ENODEV to revert back to internal phy. */
+static int smsc911x_phy_initialise_external(struct smsc911x_data *pdata)
+{
+	unsigned int address;
+	unsigned int hwcfg;
+	unsigned int phyid1;
+	unsigned int phyid2;
+
+	hwcfg = SMC_regread(pdata, HW_CFG);
+
+	/* External phy is requested, supported, and detected */
+	if (hwcfg & HW_CFG_EXT_PHY_DET_) {
+
+		/* Attempt to switch to external phy for auto-detecting
+		 * its address. Assuming tx and rx are stopped because
+		 * smsc911x_phy_initialise is called before
+		 * smsc911x_rx_initialise and tx_initialise.
+		 */
+
+		/* Disable phy clocks to the MAC */
+		hwcfg &= (~HW_CFG_PHY_CLK_SEL_);
+		hwcfg |= HW_CFG_PHY_CLK_SEL_CLK_DIS_;
+		SMC_regwrite(hwcfg, pdata, HW_CFG);
+		udelay(10); /* Enough time for clocks to stop */
+
+		/* Switch to external phy */
+		hwcfg |= HW_CFG_EXT_PHY_EN_;
+		SMC_regwrite(hwcfg, pdata, HW_CFG);
+
+		/* Enable phy clocks to the MAC */
+		hwcfg &= (~HW_CFG_PHY_CLK_SEL_);
+		hwcfg |= HW_CFG_PHY_CLK_SEL_EXT_PHY_;
+		SMC_regwrite(hwcfg, pdata, HW_CFG);
+		udelay(10); /* Enough time for clocks to restart */
+
+		hwcfg |= HW_CFG_SMI_SEL_;
+		SMC_regwrite(hwcfg, pdata, HW_CFG);
+
+		/* As far as a single threaded initialisation of this driver
+		 * is concerned, there is no need to acquire phy_lock.
+		 */
+
+		/* Auto-detect PHY */
+		for (address = 0; address <= 31; address++) {
+			pdata->phy_address = address;
+			phyid1 = smsc911x_phy_read(pdata, PHY_ID_1);
+			phyid2 = smsc911x_phy_read(pdata, PHY_ID_2);
+			if ((phyid1 != 0xFFFFU) || (phyid2 != 0xFFFFU)) {
+				SMSC_TRACE("Detected PHY at address = "
+					   "0x%02X = %d", address, address);
+				break;
+			}
+		}
+
+		if ((phyid1 == 0xFFFFU) && (phyid2 == 0xFFFFU)) {
+			SMSC_WARNING("External PHY is not accessable, "
+				     "using internal PHY instead");
+			/* Revert back to interal phy settings. */
+
+			/* Disable phy clocks to the MAC */
+			hwcfg &= (~HW_CFG_PHY_CLK_SEL_);
+			hwcfg |= HW_CFG_PHY_CLK_SEL_CLK_DIS_;
+			SMC_regwrite(hwcfg, pdata, HW_CFG);
+			udelay(10); /* Enough time for clocks to stop */
+
+			/* Switch to internal phy */
+			hwcfg &= (~HW_CFG_EXT_PHY_EN_);
+			SMC_regwrite(hwcfg, pdata, HW_CFG);
+
+			/* Enable phy clocks to the MAC */
+			hwcfg &= (~HW_CFG_PHY_CLK_SEL_);
+			hwcfg |= HW_CFG_PHY_CLK_SEL_INT_PHY_;
+			SMC_regwrite(hwcfg, pdata, HW_CFG);
+			udelay(10); /* Enough time for clocks to restart */
+
+			hwcfg &= (~HW_CFG_SMI_SEL_);
+			SMC_regwrite(hwcfg, pdata, HW_CFG);
+			/* Use internal phy */
+			return -ENODEV;
+		} else {
+			SMSC_TRACE("Successfully switched to external PHY");
+#ifdef USE_LED1_WORK_AROUND
+			pdata->not_using_extphy = 0;
+#endif
+		}
+	} else {
+		SMSC_WARNING("No external PHY detected.");
+		SMSC_WARNING("Using internal PHY instead.");
+		/* Use internal phy */
+		return -ENODEV;
+	}
+	return 0;
+}
+
+/* Initialises the PHY layer */
+static int smsc911x_phy_initialise(struct smsc911x_data *pdata)
+{
+	int result = 0;
+	unsigned int phyid1 = 0;
+	unsigned int phyid2 = 0;
+
+#ifndef USE_PHY_WORK_AROUND
+	unsigned int temp = 0;
+#endif
+#ifndef USE_PHY_WORK_AROUND
+	unsigned int loopcount = 0;
+#endif
+
+	BUG_ON(!pdata);
+	BUG_ON(!pdata->base);
+
+	switch (pdata->idrev & 0xFFFF0000) {
+		case 0x01170000:
+		case 0x01150000:
+			if (smsc911x_phy_initialise_external(pdata) < 0)
+				goto use_internal_phy;
+			break;
+		default:
+			SMSC_TRACE("External PHY is not supported");
+			SMSC_TRACE("Using internal PHY instead");
+			goto use_internal_phy;
+	}
+use_internal_phy:
+		pdata->phy_address = 1;
+#ifdef USE_LED1_WORK_AROUND
+		pdata->not_using_extphy = 1;
+#endif
+
+	/* Single threaded initialisation so no need to grab phy_lock. */
+	phyid1 = smsc911x_phy_read(pdata, PHY_ID_1);
+	phyid2 = smsc911x_phy_read(pdata, PHY_ID_2);
+
+	if ((phyid1 == 0xFFFF) && (phyid2 == 0xFFFF)) {
+		SMSC_WARNING("Internal PHY not detected!");
+		goto done;
+	}
+
+	pdata->link_speed = LINK_OFF;
+	pdata->link_settings = LINK_OFF;
+
+	/* Reset the phy */
+#ifdef USE_PHY_WORK_AROUND
+	smsc911x_phy_reset(pdata);
+	if (!smsc911x_phy_loopbacktest(pdata)) {
+		SMSC_WARNING("Failed Loop Back Test");
+		goto done;
+	} else {
+		SMSC_TRACE("Passed Loop Back Test");
+	}
+#else
+	smsc911x_phy_write(pdata, PHY_BCR, PHY_BCR_RESET_);
+    	loopcount = 100000;
+	do {
+		udelay(10);
+		temp = smsc911x_phy_read(pdata, PHY_BCR);
+		loopcount--;
+	} while ((loopcount > 0) && (temp & PHY_BCR_RESET_));
+	if (temp & PHY_BCR_RESET_) {
+		SMSC_WARNING("PHY reset failed to complete.");
+		goto done;
+	}
+#endif /* not USE_PHY_WORK_AROUND */
+
+	smsc911x_phy_setlink(pdata);
+	init_timer(&pdata->link_poll_timer);
+	pdata->link_poll_timer.function = smsc911x_phy_checklink;
+	pdata->link_poll_timer.data = (unsigned long) pdata;
+	pdata->link_poll_timer.expires = jiffies + HZ;
+	add_timer(&pdata->link_poll_timer);
+	result = 1;
+done:
+	SMSC_TRACE("smsc911x_phy_initialise, result: %s", result ?
+		   "SUCCESS" : "FAILURE");
+	return result;
+}
+
+#ifdef USE_PHY_WORK_AROUND
+static int smsc911x_phy_reset(struct smsc911x_data *pdata)
+{
+	int result = 0;
+	unsigned int temp = 0;
+	unsigned int lcount = 100000;
+
+	SMSC_TRACE("Performing PHY BCR Reset");
+	smsc911x_phy_write(pdata, PHY_BCR, PHY_BCR_RESET_);
+	do {
+		udelay(10);
+		temp = smsc911x_phy_read(pdata, PHY_BCR);
+		lcount--;
+	} while ((lcount > 0) && (temp & PHY_BCR_RESET_));
+
+	if (temp & PHY_BCR_RESET_) {
+		SMSC_WARNING("PHY reset failed to complete.");
+		goto done;
+	}
+	/* Extra delay required because the phy may not be completed with
+	 * its reset when PHY_BCR_RESET_ is cleared. Specs say 256 uS is
+	 * enough delay but using 500 here to be safe
+	 */
+	udelay(500);
+	result = 1;
+done:
+	return result;
+}
+
+static unsigned int
+smsc911x_phy_lbt_gettxstatus(struct smsc911x_data *pdata)
+{
+	unsigned int result = SMC_regread(pdata, TX_FIFO_INF);
+
+	result &= TX_FIFO_INF_TSUSED_;
+	if (result != 0)
+		result = SMC_regread(pdata, TX_STATUS_FIFO);
+	else
+		result = 0;
+	return result;
+}
+
+static unsigned int
+smsc911x_phy_lbt_getrxstatus(struct smsc911x_data *pdata)
+{
+	unsigned int result = SMC_regread(pdata, RX_FIFO_INF);
+	if (result & 0x00FF0000)
+		/* Rx status is available, read it */
+		result = SMC_regread(pdata, RX_STATUS_FIFO);
+	else
+		result = 0;
+
+	return result;
+}
+
+static int
+smsc911x_phy_check_loopbackpkt(struct smsc911x_data *pdata)
+{
+	int result = 0;
+	unsigned int tries = 0;
+	unsigned int lcount = 0;
+	u32 wrsz;
+	u32 rdsz;
+	u32 bufp;
+
+	for (tries = 0; tries < 10; tries++) {
+		unsigned int txcmd_a	= 0;
+		unsigned int txcmd_b	= 0;
+		unsigned int status	= 0;
+		unsigned int pktlength	= 0;
+
+		/* Zero-out rx packet memory */
+		memset(pdata->loopback_rx_pkt, 0, MIN_PACKET_SIZE);
+
+		/* Write tx packet to 118 */
+		txcmd_a = (((unsigned int)pdata->loopback_tx_pkt)
+			   & 0x03) << 16;
+		txcmd_a |= TX_CMD_A_FIRST_SEG_ | TX_CMD_A_LAST_SEG_;
+		txcmd_a |= MIN_PACKET_SIZE;
+
+		txcmd_b = MIN_PACKET_SIZE << 16  | MIN_PACKET_SIZE;
+
+		SMC_regwrite(txcmd_a, pdata, TX_DATA_FIFO);
+		SMC_regwrite(txcmd_b, pdata, TX_DATA_FIFO);
+
+		bufp = ((u32)pdata->loopback_tx_pkt) & 0xFFFFFFFC;
+		wrsz = MIN_PACKET_SIZE + 3;
+		wrsz += (((u32)pdata->loopback_tx_pkt) & 0x3);
+		wrsz >>= 2;
+
+		smsc911x_tx_writefifo(pdata, (unsigned int *)bufp, wrsz);
+
+		/* Wait till transmit is done */
+		lcount = 60;
+		do {
+			udelay(5);
+			lcount--;
+			status = smsc911x_phy_lbt_gettxstatus(pdata);
+		} while ((lcount > 0) && (status == 0));
+
+		if (status == 0) {
+			SMSC_WARNING("Failed to transmit during "
+				     "loopback test");
+			continue;
+		}
+		if (status & 0x00008000) {
+			SMSC_WARNING("Transmit encountered errors "
+				     "during loopback test");
+			continue;
+		}
+
+		/* Wait till receive is done */
+		lcount = 60;
+		do {
+			udelay(5);
+			lcount--;
+			status = smsc911x_phy_lbt_getrxstatus(pdata);
+		} while ((lcount > 0) && (status == 0));
+
+		if (status == 0) {
+			SMSC_WARNING("Failed to receive during "
+				     "loopback test");
+			continue;
+		}
+		if (status & RX_STS_ES_) {
+			SMSC_WARNING("Receive encountered errors "
+				     "during loopback test");
+			continue;
+		}
+
+		pktlength = ((status & 0x3FFF0000UL) >> 16);
+
+		bufp = (u32)pdata->loopback_rx_pkt;
+		rdsz = pktlength + 3;
+		rdsz += ((u32)pdata->loopback_rx_pkt) & 0x3;
+		rdsz >>= 2;
+
+		smsc911x_rx_readfifo(pdata, (unsigned int *)bufp, rdsz);
+
+		if (pktlength != (MIN_PACKET_SIZE + 4)) {
+			SMSC_WARNING("Unexpected packet size during "
+				     "loop back test, size=%d, "
+				     "will retry", pktlength);
+		} else {
+			unsigned int index = 0;
+			int mismatch = 0;
+			for (index = 0; index < MIN_PACKET_SIZE; index++) {
+				if (pdata->loopback_tx_pkt[index]
+				    != pdata->loopback_rx_pkt[index]) {
+					mismatch = 1;
+					break;
+				}
+			}
+			if (!mismatch) {
+				SMSC_TRACE("Successfully verified "
+					   "loopback packet");
+				result = 1;
+				goto done;
+			} else {
+				SMSC_WARNING("Data miss match during "
+					     "loop back test, will retry.");
+			}
+		}
+	}
+done:
+	return result;
+}
+
+static int smsc911x_phy_loopbacktest(struct smsc911x_data *pdata)
+{
+	int result = 0;
+	unsigned int index = 0;
+	unsigned int tries = 0;
+	unsigned int val;
+
+	/* Initialise tx packet */
+	for (index = 0; index < 6; index++) {
+		/* Use broadcast destination address */
+		pdata->loopback_tx_pkt[index] = (char)0xFF;
+	}
+
+	for (index = 6; index < 12; index++) {
+		/* Use incrementing source address */
+		pdata->loopback_tx_pkt[index] = (char)index;
+	}
+
+	/* Set length type field */
+	pdata->loopback_tx_pkt[12] = 0x00;
+	pdata->loopback_tx_pkt[13] = 0x00;
+	for (index = 14; index < MIN_PACKET_SIZE; index++) {
+		pdata->loopback_tx_pkt[index] = (char)index;
+	}
+
+	val = SMC_regread(pdata, HW_CFG);
+	val &= HW_CFG_TX_FIF_SZ_;
+	val |= HW_CFG_SF_;
+	SMC_regwrite(val, pdata, HW_CFG);
+
+	SMC_regwrite(TX_CFG_TX_ON_, pdata, TX_CFG);
+	SMC_regwrite((((unsigned int)(pdata->loopback_rx_pkt)) & 0x03) << 8,
+		     pdata, RX_CFG);
+
+	/* Set PHY to 10/FD, no ANEG, */
+	smsc911x_phy_write(pdata, PHY_BCR, 0x0100);
+
+	/* Enable MAC tx/rx, FD */
+	smsc911x_mac_write(pdata, MAC_CR, MAC_CR_FDPX_
+			   | MAC_CR_TXEN_ | MAC_CR_RXEN_);
+
+	/* Set PHY to loopback mode */
+	smsc911x_phy_write(pdata, PHY_BCR, 0x4100);
+
+	for (tries = 0; tries < 10; tries++) {
+		if (smsc911x_phy_check_loopbackpkt(pdata)) {
+			result = 1;
+			goto done;
+		}
+		pdata->resetcount++;
+		/* Disable MAC rx */
+		smsc911x_mac_write(pdata, MAC_CR, 0);
+		smsc911x_phy_reset(pdata);
+
+		/* Set PHY to 10/FD, no ANEG, and loopback mode */
+		smsc911x_phy_write(pdata, PHY_BCR, 0x4100);
+
+		/* Enable MAC tx/rx, FD */
+		smsc911x_mac_write(pdata, MAC_CR,MAC_CR_FDPX_
+				   | MAC_CR_TXEN_ | MAC_CR_RXEN_);
+	}
+done:
+	/* Disable MAC */
+	smsc911x_mac_write(pdata, MAC_CR, 0);
+
+	/* Cancel PHY loopback mode */
+	smsc911x_phy_write(pdata, PHY_BCR, 0);
+
+	SMC_regwrite(0, pdata, TX_CFG);
+	SMC_regwrite(0, pdata, RX_CFG);
+
+	return result;
+}
+#endif /* USE_PHY_WORK_AROUND */
+
+/* Sets the link mode */
+static void smsc911x_phy_setlink(struct smsc911x_data *pdata)
+{
+	unsigned int temp;
+
+	/* Because this is part of the single threaded initialization
+	 * path there is no need to acquire the phy_lock
+	 */
+	temp = smsc911x_phy_read(pdata, PHY_ANEG_ADV);
+
+	/* Advertise all speeds and pause capabilities */
+	temp |= (PHY_ANEG_ADV_PAUSE_ | PHY_ANEG_ADV_SPEED_);
+	smsc911x_phy_write(pdata, PHY_ANEG_ADV, temp);
+
+	/* begin to establish link */
+	smsc911x_phy_write(pdata, PHY_BCR, PHY_BCR_AUTO_NEG_ENABLE_ |
+			   PHY_BCR_RESTART_AUTO_NEG_);
+	return;
+}
+
+/* NOTE: Assuming phy_lock has already been acquired.
+ * Gets the current link mode
+ */
+static void smsc911x_phy_getlinkmode(struct smsc911x_data *pdata)
+{
+	unsigned int result = LINK_OFF;
+	unsigned int phy_reg = 0;
+	unsigned int phy_bsr = 0;
+
+	phy_bsr = smsc911x_phy_read(pdata, PHY_BSR);
+
+	pdata->link_settings = LINK_OFF;
+	if (phy_bsr & PHY_BSR_LINK_STATUS_) {
+		phy_reg = smsc911x_phy_read(pdata, PHY_BCR);
+		if (phy_reg & PHY_BCR_AUTO_NEG_ENABLE_) {
+			unsigned int linksettings = LINK_AUTO_NEGOTIATE;
+			unsigned int phy_adv =
+				smsc911x_phy_read(pdata, PHY_ANEG_ADV);
+			unsigned int phy_lpa =
+				smsc911x_phy_read(pdata, PHY_ANEG_LPA);
+			if (phy_adv & PHY_ANEG_ADV_ASYMP_)
+				linksettings |= LINK_ASYMMETRIC_PAUSE;
+			if (phy_adv & PHY_ANEG_ADV_SYMP_)
+				linksettings |= LINK_SYMMETRIC_PAUSE;
+			if (phy_adv & PHY_ANEG_LPA_100FDX_)
+				linksettings |= LINK_SPEED_100FD;
+			if (phy_adv & PHY_ANEG_LPA_100HDX_)
+				linksettings |= LINK_SPEED_100HD;
+			if (phy_adv & PHY_ANEG_LPA_10FDX_)
+				linksettings |= LINK_SPEED_10FD;
+			if (phy_adv & PHY_ANEG_LPA_10HDX_)
+				linksettings |= LINK_SPEED_10HD;
+
+			pdata->link_settings = linksettings;
+			phy_lpa &= phy_adv;
+
+			if (phy_lpa & PHY_ANEG_LPA_100FDX_)
+				result = LINK_SPEED_100FD;
+			else if (phy_lpa & PHY_ANEG_LPA_100HDX_)
+				result = LINK_SPEED_100HD;
+			else if (phy_lpa & PHY_ANEG_LPA_10FDX_)
+				result = LINK_SPEED_10FD;
+			else if (phy_lpa & PHY_ANEG_LPA_10HDX_)
+				result = LINK_SPEED_10HD;
+		} else {
+			if (phy_reg & PHY_BCR_SPEED_SELECT_) {
+				if (phy_reg & PHY_BCR_DUPLEX_MODE_) {
+					result = LINK_SPEED_100FD;
+					pdata->link_settings = result;
+				} else {
+					result = LINK_SPEED_100HD;
+					pdata->link_settings = result;
+				}
+			} else {
+				if(phy_reg & PHY_BCR_DUPLEX_MODE_) {
+					result = LINK_SPEED_10FD;
+					pdata->link_settings = result;
+				} else {
+					result = LINK_SPEED_10HD;
+					pdata->link_settings = result;
+				}
+			}
+		}
+	}
+	pdata->link_speed = result;
+	return;
+}
+
+/* Updates link mode full-duplex */
+static void
+smsc911x_phy_upd_lmode_fd(struct smsc911x_data *pdata,
+			  unsigned int locallink,
+			  unsigned int linkpartner)
+{
+	unsigned int temp;
+
+	if (((locallink & linkpartner) & 0x0400) != 0 ) {
+		/* Enable PAUSE receive and transmit */
+		smsc911x_mac_write(pdata, FLOW, 0xFFFF0002);
+		temp = SMC_regread(pdata, AFC_CFG);
+		temp |= 0xF;
+		SMC_regwrite(temp, pdata, AFC_CFG);
+
+	} else if (((locallink & 0x0C00) == 0x0C00) &&
+		   ((linkpartner & 0x0C00) == 0x0800)) {
+		/* Enable PAUSE receive, disable PAUSE transmit */
+		smsc911x_mac_write(pdata, FLOW, 0xFFFF0002);
+		temp = SMC_regread(pdata, AFC_CFG);
+		temp &= ~0xF;
+		SMC_regwrite(temp, pdata, AFC_CFG);
+
+	} else {
+		/* Disable PAUSE receive and transmit */
+		smsc911x_mac_write(pdata, FLOW, 0);
+		temp = SMC_regread(pdata, AFC_CFG);
+		temp &= ~0xF;
+		SMC_regwrite(temp, pdata, AFC_CFG);
+	}
+	return;
+}
+
+static void smsc911x_phy_print_linkmode(unsigned int locallink,
+					unsigned int linkpartner)
+{
+	SMSC_TRACE("LAN911x: %s,%s,%s,%s,%s,%s",
+		(locallink & PHY_ANEG_ADV_ASYMP_) ? "ASYMP" : "     ",
+		(locallink & PHY_ANEG_ADV_SYMP_)  ? "SYMP " : "     ",
+		(locallink & PHY_ANEG_ADV_100F_)  ? "100FD" : "     ",
+		(locallink & PHY_ANEG_ADV_100H_)  ? "100HD" : "     ",
+		(locallink & PHY_ANEG_ADV_10F_)   ? "10FD " : "     ",
+		(locallink & PHY_ANEG_ADV_10H_)   ? "10HD " : "     ");
+
+	SMSC_TRACE("Partner: %s,%s,%s,%s,%s,%s",
+		(linkpartner & PHY_ANEG_LPA_ASYMP_)  ? "ASYMP" : "     ",
+		(linkpartner & PHY_ANEG_LPA_SYMP_)   ? "SYMP " : "     ",
+		(linkpartner & PHY_ANEG_LPA_100FDX_) ? "100FD" : "     ",
+		(linkpartner & PHY_ANEG_LPA_100HDX_) ? "100HD" : "     ",
+		(linkpartner & PHY_ANEG_LPA_10FDX_)  ? "10FD " : "     ",
+		(linkpartner & PHY_ANEG_LPA_10HDX_)  ? "10HD " : "     ");
+	return;
+}
+
+/* Update link mode if any thing has changed */
+static void smsc911x_phy_update_linkmode(struct smsc911x_data *pdata)
+{
+	unsigned int old_link_speed = pdata->link_speed;
+	unsigned int temp;
+
+	spin_lock(&pdata->phy_lock);
+	smsc911x_phy_getlinkmode(pdata);
+
+	if (old_link_speed != pdata->link_speed) {
+		if (pdata->link_speed != LINK_OFF) {
+			unsigned int phy_reg = 0;
+			switch (pdata->link_speed) {
+			case LINK_SPEED_10HD:
+				SMSC_TRACE("Link is now UP at 10Mbps HD");
+				break;
+			case LINK_SPEED_10FD:
+				SMSC_TRACE("Link is now UP at 10Mbps FD");
+				break;
+			case LINK_SPEED_100HD:
+				SMSC_TRACE("Link is now UP at 100Mbps HD");
+				break;
+			case LINK_SPEED_100FD:
+				SMSC_TRACE("Link is now UP at 100Mbps FD");
+				break;
+			default:
+				SMSC_WARNING("Link is now UP at unknown link "
+					     "speed: 0x%08X",
+					     pdata->link_speed);
+				break;
+			}
+			phy_reg = smsc911x_mac_read(pdata, MAC_CR);
+			phy_reg &= ~(MAC_CR_FDPX_ | MAC_CR_RCVOWN_);
+
+			switch (pdata->link_speed) {
+			case LINK_SPEED_10HD:
+			case LINK_SPEED_100HD:
+				phy_reg |= MAC_CR_RCVOWN_;
+				break;
+
+			case LINK_SPEED_10FD:
+			case LINK_SPEED_100FD:
+				phy_reg |= MAC_CR_FDPX_;
+				break;
+
+			default:
+				SMSC_WARNING("Unknown link speed: 0x%08X",
+					     pdata->link_speed);
+				break;
+			}
+
+			smsc911x_mac_write(pdata, MAC_CR, phy_reg);
+
+			if (pdata->link_settings & LINK_AUTO_NEGOTIATE) {
+				unsigned int linkpartner = 0;
+				unsigned int locallink = 0;
+				locallink = smsc911x_phy_read(pdata, 4);
+				linkpartner= smsc911x_phy_read(pdata,5);
+				switch(pdata->link_speed) {
+				case LINK_SPEED_10FD:
+				case LINK_SPEED_100FD:
+					smsc911x_phy_upd_lmode_fd(pdata,
+								  locallink,
+								  linkpartner);
+					break;
+
+				case LINK_SPEED_10HD:
+				case LINK_SPEED_100HD:
+					smsc911x_mac_write(pdata, FLOW, 0);
+					temp = SMC_regread(pdata, AFC_CFG);
+					temp |= 0xF;
+					SMC_regwrite(temp, pdata, AFC_CFG);
+					break;
+
+				default:
+					SMSC_WARNING("Unknown link speed: "
+						     "0x%08X\n",
+						     pdata->link_speed);
+					break;
+				}
+				smsc911x_phy_print_linkmode(locallink,
+							    linkpartner);
+			} else {
+				switch(pdata->link_speed) {
+				case LINK_SPEED_10HD:
+				case LINK_SPEED_100HD:
+					smsc911x_mac_write(pdata,
+							   FLOW, 0);
+					SMC_regwrite(0x0000000F,
+						     pdata, AFC_CFG);
+					break;
+				default:
+					smsc911x_mac_write(pdata, FLOW, 0);
+					temp = SMC_regread(pdata, AFC_CFG);
+					temp &= ~0xF;
+					SMC_regwrite(temp, pdata, AFC_CFG);
+					break;
+				}
+			}
+			netif_carrier_on(pdata->dev);
+#ifdef USE_LED1_WORK_AROUND
+			if ((pdata->gpio_orig_setting & GPIO_CFG_LED1_EN_) &&
+			    pdata->not_using_extphy) {
+				/* Restore orginal GPIO configuration */
+				pdata->gpio_setting = pdata->gpio_orig_setting;
+				SMC_regwrite(pdata->gpio_setting, pdata,
+					     GPIO_CFG);
+			}
+#endif /* USE_LED1_WORK_AROUND */
+		} else {
+			SMSC_TRACE("Link is now down");
+			netif_carrier_off(pdata->dev);
+			smsc911x_mac_write(pdata, FLOW, 0);
+
+			temp = SMC_regread(pdata, AFC_CFG);
+			temp &= ~0xF;
+			SMC_regwrite(temp, pdata, AFC_CFG);
+
+#ifdef USE_LED1_WORK_AROUND
+			/* Check global setting that LED1
+			 * usage is 10/100 indicator */
+			pdata->gpio_setting = SMC_regread(pdata, GPIO_CFG);
+			if ((pdata->gpio_setting & GPIO_CFG_LED1_EN_) &&
+			    pdata->not_using_extphy) {
+				/* Force 10/100 LED off, after saving
+				 * orginal GPIO configuration */
+				pdata->gpio_orig_setting = pdata->gpio_setting;
+
+				pdata->gpio_setting &= ~GPIO_CFG_LED1_EN_;
+				pdata->gpio_setting |= (GPIO_CFG_GPIOBUF0_
+							| GPIO_CFG_GPIODIR0_
+							| GPIO_CFG_GPIOD0_);
+				SMC_regwrite(pdata->gpio_setting, pdata,
+					     GPIO_CFG);
+			}
+#endif /* USE_LED1_WORK_AROUND */
+		}
+	}
+	spin_unlock(&pdata->phy_lock);
+	return;
+}
+
+/* Gets a phy register */
+static unsigned int
+smsc911x_phy_read(struct smsc911x_data * pdata, unsigned int index)
+{
+	unsigned int addr = 0;
+	unsigned int result = 0xFFFF;
+	int i;
+
+	BUG_ON(!pdata);
+
+	/* Confirm MII not busy */
+	if ((smsc911x_mac_read(pdata, MII_ACC) & MII_ACC_MII_BUSY_) != 0) {
+		SMSC_WARNING("MII is busy in smsc911x_phy_read???");
+		result = 0;
+		goto done;
+	}
+
+	/* Set the address, index & direction (read from PHY) */
+	addr = (((pdata->phy_address) & 0x1F) << 11)
+		| ((index & 0x1F) << 6);
+	smsc911x_mac_write(pdata, MII_ACC, addr);
+
+	/* Wait for read to complete w/ timeout */
+	for (i = 0; i < 100; i++) {
+		/* See if MII is finished yet */
+		if ((smsc911x_mac_read(pdata, MII_ACC)
+		    & MII_ACC_MII_BUSY_) == 0) {
+			result = (unsigned int)smsc911x_mac_read(pdata,
+								 MII_DATA);
+			goto done;
+		}
+	}
+	SMSC_WARNING("Timed out waiting for MII write to finish");
+
+done:
+	return result;
+}
+
+/* Sets a phy register */
+static void smsc911x_phy_write(struct smsc911x_data *pdata,
+			       unsigned int index, unsigned int val)
+{
+	unsigned int addr = 0;
+	int i = 0;
+
+	BUG_ON(!pdata);
+
+	/* Confirm MII not busy */
+	if ((smsc911x_mac_read(pdata, MII_ACC) & MII_ACC_MII_BUSY_) != 0) {
+		SMSC_WARNING("MII is busy in smsc911x_write_phy???");
+		goto done;
+	}
+
+	/* Put the data to write in the MAC */
+	smsc911x_mac_write(pdata, MII_DATA, val);
+
+	/* Set the address, index & direction (write to PHY) */
+	addr = (((pdata->phy_address) & 0x1F) << 11) |
+		((index & 0x1F) << 6) | MII_ACC_MII_WRITE_;
+	smsc911x_mac_write(pdata, MII_ACC, addr);
+
+	/* Wait for write to complete w/ timeout */
+	for (i = 0; i < 100; i++) {
+		/* See if MII is finished yet */
+		if ((smsc911x_mac_read(pdata, MII_ACC)
+		    & MII_ACC_MII_BUSY_) == 0)
+			goto done;
+	}
+	SMSC_WARNING("Timed out waiting for MII write to finish");
+done:
+	return;
+}
+
+/* Entry point for the link poller */
+static void smsc911x_phy_checklink(unsigned long ptr)
+{
+	struct smsc911x_data * pdata;
+
+	pdata = (struct smsc911x_data *)ptr;
+	BUG_ON(!pdata);
+
+	/* Must call this twice */
+	smsc911x_phy_update_linkmode(pdata);
+	smsc911x_phy_update_linkmode(pdata);
+
+	if (!(pdata->stop_link_poll)) {
+		pdata->link_poll_timer.expires= jiffies + HZ;
+		add_timer(&(pdata->link_poll_timer));
+	}
+	return;
+}
+
+
+
+/* Initalise the MAC data elements */
+static void smsc911x_mac_initialise(struct smsc911x_data *pdata)
+{
+	BUG_ON(!pdata);
+
+	/* Does nothing but kept as a place holder. */
+}
+
+/* Polls for not busy for a limited time */
+static int smsc911x_mac_notbusy(struct smsc911x_data *pdata)
+{
+	int i = 0;
+	/* Assuming MacPhyAccessLock has already been acquired */
+
+	/* wait for MAC not busy, w/ timeout */
+	for (i = 0; i < 40; i++) {
+		if ((SMC_regread(pdata, MAC_CSR_CMD)
+		    & MAC_CSR_CMD_CSR_BUSY_) == 0) {
+			return 1;
+		}
+	}
+	SMSC_WARNING("Timed out waiting for MAC not BUSY. "
+		"MAC_CSR_CMD: 0x%08X", SMC_regread(pdata, MAC_CSR_CMD));
+	return 0;
+}
+
+/* Fetches a MAC register value. Assumes phy_lock is acquired */
+static unsigned int
+smsc911x_mac_read(struct smsc911x_data * pdata, unsigned int offset)
+{
+	unsigned int result = 0xFFFFFFFF;
+	volatile unsigned int temp = 0;
+
+	BUG_ON(!pdata);
+	BUG_ON(!pdata->base);
+
+	/* Wait until not busy */
+	if (SMC_regread(pdata, MAC_CSR_CMD) & MAC_CSR_CMD_CSR_BUSY_) {
+		SMSC_WARNING("smsc911x_mac_read failed, "
+			     "MAC already busy at entry");
+		goto done;
+	}
+
+	/* Send the MAC cmd */
+	SMC_regwrite(((offset & 0x000000FF) | MAC_CSR_CMD_CSR_BUSY_
+		      | MAC_CSR_CMD_R_NOT_W_), pdata, MAC_CSR_CMD);
+
+	temp = SMC_regread(pdata, BYTE_TEST); /* To flush previous write */
+
+	/* Wait for the read to happen */
+	if (!smsc911x_mac_notbusy(pdata)) {
+		SMSC_WARNING("smsc911x_mac_read failed, "
+			     "waiting for MAC not busy after read");
+		goto done;
+	} else {
+		/* Read the data */
+		result = SMC_regread(pdata, MAC_CSR_DATA);
+	}
+done:
+	return result;
+}
+
+/* Set a mac register, phy_lock must be acquired before calling. */
+static void smsc911x_mac_write(struct smsc911x_data *pdata,
+			       unsigned int offset, unsigned int val)
+{
+	volatile unsigned int temp = 0;
+
+	BUG_ON(!pdata);
+	BUG_ON(!pdata->base);
+
+	if (SMC_regread(pdata, MAC_CSR_CMD) & MAC_CSR_CMD_CSR_BUSY_) {
+		SMSC_WARNING("smsc911x_mac_write failed, "
+			     "MAC already busy at entry");
+		goto done;
+	}
+
+	/* Send data to write */
+	SMC_regwrite(val, pdata, MAC_CSR_DATA);
+
+	/* Write the actual data */
+	SMC_regwrite(((offset & 0x000000FF) | MAC_CSR_CMD_CSR_BUSY_),
+		     pdata, MAC_CSR_CMD);
+	/* Force flush of previous write */
+	temp = SMC_regread(pdata, BYTE_TEST);
+
+	/* Wait for the write to complete */
+	if (!smsc911x_mac_notbusy(pdata))
+		SMSC_WARNING("smsc911x_mac_write failed, "
+			     "waiting for MAC not busy after write");
+done:
+	return;
+}
+
+#define TX_FIFO_LOW_THRESHOLD	(1600)
+
+/* Initialise the transmitter, it is called during driver
+ * initialisation, thus no need to acquire phy_lock */
+static void smsc911x_tx_initialise(struct smsc911x_data *pdata)
+{
+	unsigned int mac_cr;
+	unsigned int val;
+
+	val = 0;
+	BUG_ON(!pdata);
+	BUG_ON(!pdata->base);
+
+	val = SMC_regread(pdata, HW_CFG);
+	val &= HW_CFG_TX_FIF_SZ_;
+	val |= HW_CFG_SF_;
+	SMC_regwrite(val, pdata, HW_CFG);
+
+	SMC_regwrite(SMC_regread(pdata, FIFO_INT) | 0xFF000000,
+		     pdata, FIFO_INT);
+	SMC_regwrite(SMC_regread(pdata, INT_EN) | INT_EN_TDFA_EN_,
+		     pdata, INT_EN);
+
+	mac_cr = smsc911x_mac_read(pdata, MAC_CR);
+	mac_cr |= (MAC_CR_TXEN_|MAC_CR_HBDIS_);
+	smsc911x_mac_write(pdata, MAC_CR, mac_cr);
+	SMC_regwrite(TX_CFG_TX_ON_, pdata, TX_CFG);
+	return;
+}
+
+/* Reenables the transmitter */
+static int
+smsc911x_tx_handleirq(struct smsc911x_data *pdata, unsigned int intsts)
+{
+	BUG_ON(!pdata);
+	if (intsts & INT_STS_TDFA_) {
+		SMC_regwrite(SMC_regread(pdata, FIFO_INT) | 0xFF000000,
+			     pdata, FIFO_INT);
+		SMC_regwrite(INT_STS_TDFA_, pdata, INT_STS);
+		netif_wake_queue(pdata->dev);
+		return 1;
+	}
+	return 0;
+}
+
+/* Writes a packet to the TX_DATA_FIFO */
+static void
+smsc911x_tx_writefifo(struct smsc911x_data *pdata, unsigned int *buf,
+				  unsigned int wordcount)
+{
+	while (wordcount) {
+		SMC_regwrite(*buf++, pdata, TX_DATA_FIFO);
+		wordcount--;
+	}
+}
+
+/* Gets the number of tx statuses in the fifo */
+static unsigned int
+smsc911x_tx_get_txstatcount(struct smsc911x_data *pdata)
+{
+	unsigned int result = 0;
+	BUG_ON(!pdata);
+	BUG_ON(!pdata->base);
+	result = SMC_regread(pdata, TX_FIFO_INF);
+	result &= TX_FIFO_INF_TSUSED_;
+	result >>= 16;
+	return result;
+}
+
+/* Transmit a packet */
+static void
+smsc911x_tx_sendskb(struct smsc911x_data * pdata, struct sk_buff *skb)
+{
+	unsigned int freespace;
+	unsigned int tx_cmd_a;
+	unsigned int tx_cmd_b;
+	unsigned int temp;
+	u32 wrsz;
+	u32 bufp;
+
+	BUG_ON(!pdata);
+	BUG_ON(!pdata->base);
+
+	freespace = SMC_regread(pdata, TX_FIFO_INF);
+	freespace &= TX_FIFO_INF_TDFREE_;
+
+	if (freespace < TX_FIFO_LOW_THRESHOLD)
+		SMSC_WARNING("Tx data fifo low, space available: %d",
+			     freespace);
+
+	/* Word alignment adjustment */
+	tx_cmd_a = ((((unsigned int)(skb->data)) & 0x03) << 16);
+	tx_cmd_a |= TX_CMD_A_FIRST_SEG_ | TX_CMD_A_LAST_SEG_;
+	tx_cmd_a |= (unsigned int)skb->len;
+
+	tx_cmd_b = ((unsigned int)skb->len) << 16;
+	tx_cmd_b |= (unsigned int)skb->len;
+
+	SMC_regwrite(tx_cmd_a, pdata, TX_DATA_FIFO);
+	SMC_regwrite(tx_cmd_b, pdata, TX_DATA_FIFO);
+
+	bufp = ((u32)skb->data) & 0xFFFFFFFC;
+	wrsz = (u32)skb->len + 3;
+	wrsz += ((u32)skb->data) & 0x3;
+	wrsz >>= 2;
+
+	smsc911x_tx_writefifo(pdata, (unsigned int *)bufp, wrsz);
+
+	freespace -= (skb->len + 32);
+	dev_kfree_skb(skb);
+
+	if (smsc911x_tx_get_txstatcount(pdata) >= 30)
+		smsc911x_tx_update_txcounters(pdata);
+	if (freespace < TX_FIFO_LOW_THRESHOLD) {
+		netif_stop_queue(pdata->dev);
+		temp = SMC_regread(pdata, FIFO_INT);
+		temp &= 0x00FFFFFF;
+		temp |= 0x32000000;
+		SMC_regwrite(temp, pdata, FIFO_INT);
+	}
+}
+
+/* Fetches a tx status out of the status fifo */
+static unsigned int smsc911x_tx_completetx(struct smsc911x_data *pdata)
+{
+	unsigned int result;
+
+	BUG_ON(!pdata);
+	BUG_ON(!pdata->base);
+
+	result = SMC_regread(pdata, TX_FIFO_INF);
+	result &= TX_FIFO_INF_TSUSED_;
+
+	if (result)
+		result = SMC_regread(pdata, TX_STATUS_FIFO);
+	else
+		result = 0;
+	return result;
+}
+
+/* Reads tx statuses and increments counters where necessary */
+static void
+smsc911x_tx_update_txcounters(struct smsc911x_data *pdata)
+{
+	unsigned int tx_stat;
+
+	BUG_ON(!pdata);
+
+	while ((tx_stat = smsc911x_tx_completetx(pdata)) != 0) {
+		if (tx_stat & 0x80000000) {
+	/* In this driver the packet tag is used as the packet length.
+	 * Since a packet length can never reach the size of 0x8000,
+	 * this bit is reserved so that if packet tracking tags were
+	 * ever used, then those tracking tags would set the reserved bit.
+	 * Accordingly, this control path would be used to look up the
+	 * packet and perhaps free it. It is worth noting that the
+	 * "reserved bit" in the warning above does not reference a
+	 * hardware defined reserved bit but rather a driver defined one.
+	 */
+			SMSC_WARNING("Packet tag reserved bit is high");
+		} else {
+			if (tx_stat & 0x00008000) {
+				pdata->stats.tx_errors++;
+			} else {
+				pdata->stats.tx_packets++;
+				pdata->stats.tx_bytes += (tx_stat >> 16);
+			}
+			if (tx_stat & 0x00000100) {
+				pdata->stats.collisions += 16;
+				pdata->stats.tx_aborted_errors += 1;
+			} else {
+				pdata->stats.collisions +=
+					((tx_stat >> 3) & 0xF);
+			}
+			if (tx_stat & 0x00000800)
+				pdata->stats.tx_carrier_errors += 1;
+			if (tx_stat & 0x00000200) {
+				pdata->stats.collisions++;
+				pdata->stats.tx_aborted_errors++;
+			}
+		}
+	}
+	return;
+}
+
+/* Initializes the receiver, called single-threadedly during
+ * driver initialisation, it doesn't need phy_lock */
+static void
+smsc911x_rx_initialise(struct smsc911x_data *pdata)
+{
+	unsigned int mac_cr;
+
+	BUG_ON(!pdata);
+	SMC_regwrite(0x00000200, pdata, RX_CFG);
+	mac_cr = smsc911x_mac_read(pdata, MAC_CR);
+	mac_cr |= MAC_CR_RXEN_;
+	smsc911x_mac_write(pdata, MAC_CR, mac_cr);
+
+	SMC_regwrite(SMC_regread(pdata, FIFO_INT) & ~(0xFF), pdata, FIFO_INT);
+	SMC_regwrite(SMC_regread(pdata, INT_EN) | INT_EN_RSFL_EN_,
+		     pdata, INT_EN);
+	return;
+}
+
+/* Reads a packet out of the RX_DATA_FIFO */
+static void
+smsc911x_rx_readfifo(struct smsc911x_data *pdata, unsigned int *buf,
+		     unsigned int count)
+{
+	while (count) {
+		*buf++ = SMC_regread(pdata, RX_DATA_FIFO);
+		count--;
+	}
+	return;
+}
+
+/* Passes a packet to linux */
+static void smsc911x_rx_handoffskb(struct smsc911x_data *pdata,
+	struct sk_buff *skb)
+{
+	int result;
+
+	skb->dev = pdata->dev;
+	skb->protocol = eth_type_trans(skb, pdata->dev);
+	skb->ip_summed = CHECKSUM_NONE;
+
+	result = netif_rx(skb);
+
+	switch (result) {
+		case NET_RX_SUCCESS:
+			break;
+		case NET_RX_CN_LOW:
+		case NET_RX_CN_MOD:
+		case NET_RX_CN_HIGH:
+		case NET_RX_DROP:
+			pdata->rx_congested = 1;
+			break;
+		default:
+			pdata->rx_congested = 1;
+			SMSC_WARNING("Unknown return value from "
+				     "netif_rx, result: %d", result);
+			break;
+	}
+	return;
+}
+
+/* Fetches the next rx status */
+static unsigned int smsc911x_rx_pop_rxstatus(struct smsc911x_data *pdata)
+{
+	unsigned int result;
+
+	result = SMC_regread(pdata, RX_FIFO_INF);
+	if ((pdata->rx_congested == 0) ||
+	    ((pdata->rx_congested == 1) &&
+	     ((result & 0x00FF0000) == 0))) {
+		/* Rx status is available, read it */
+		if (result & 0x00FF0000)
+			result = SMC_regread(pdata, RX_STATUS_FIFO);
+		else
+			result = 0;
+	} else {
+		/*  Initiate the interrupt deassertion interval */
+		SMC_regwrite(SMC_regread(pdata, INT_CFG)
+			     | INT_CFG_INT_DEAS_CLR_, pdata, INT_CFG);
+		result = 0;
+	}
+	return result;
+}
+
+/* Increments the Rx error counters */
+static void
+smsc911x_rx_counterrors(struct smsc911x_data *pdata, unsigned int rxstat)
+{
+	int crc_err;
+
+	crc_err = 0;
+	if (rxstat & 0x00008000) {
+		pdata->stats.rx_errors++;
+		if (rxstat & 0x00000002) {
+			pdata->stats.rx_crc_errors++;
+			crc_err = 1;
+		}
+	}
+	if(!crc_err) {
+		if ((rxstat & 0x00001020) == 0x00001020) {
+			/* Frame type indicates length,
+			 * and length error is set */
+			pdata->stats.rx_length_errors++;
+		}
+		if (rxstat & RX_STS_MCAST_)
+			pdata->stats.multicast++;
+	}
+	return;
+}
+
+/* Quickly dumps bad packets */
+static void
+smsc911x_rx_fastforward(struct smsc911x_data *pdata, unsigned int count)
+{
+	if (count >= 4) {
+		unsigned int timeout = 500;
+		SMC_regwrite(RX_DP_CTRL_RX_FFWD_, pdata, RX_DP_CTRL);
+		while (timeout && (SMC_regread(pdata, RX_DP_CTRL)
+		       & RX_DP_CTRL_RX_FFWD_)) {
+			udelay(1);
+			timeout--;
+		}
+		if (timeout == 0) {
+			SMSC_WARNING("Timed out waiting for RX FFWD "
+				     "to finish, RX_DP_CTRL: 0x%08X",
+				     SMC_regread(pdata, RX_DP_CTRL));
+		}
+	} else {
+		while (count) {
+			volatile unsigned int temp;
+			temp = SMC_regread(pdata, RX_DATA_FIFO);
+			count--;
+		}
+	}
+	return;
+}
+
+/* Main function for reading packets out of the LAN911x */
+static void smsc911x_rx_processpackets(struct smsc911x_data *pdata)
+{
+	unsigned int rxstat;
+	unsigned int *bufptr;
+
+	pdata->rx_congested = 0;
+	while ((rxstat = smsc911x_rx_pop_rxstatus(pdata)) != 0) {
+		unsigned int pktlength = ((rxstat & 0x3FFF0000) >> 16);
+		smsc911x_rx_counterrors(pdata,rxstat);
+
+		if ((rxstat & RX_STS_ES_) == 0) {
+			struct sk_buff *skb = NULL;
+			skb = dev_alloc_skb(pktlength + 2);
+			if (skb) {
+				skb->data = skb->head;
+				skb->tail = skb->head;
+				/* Align IP on 16B boundary */
+				skb_reserve(skb, 2);
+				skb_put(skb, pktlength - 4);
+
+				/* Update counters */
+				pdata->stats.rx_packets++;
+				pdata->stats.rx_bytes += (pktlength - 4);
+				bufptr = (unsigned int *)skb->head;
+				smsc911x_rx_readfifo(pdata, bufptr,
+						     (pktlength + 2 + 3) >> 2);
+
+				smsc911x_rx_handoffskb(pdata, skb);
+				continue;
+			} else {
+				SMSC_WARNING("Unable to allocate sk_buff "
+					     "for rx packet, in PIO path");
+				pdata->stats.rx_dropped++;
+			}
+		}
+		/* At this point, the packet is to be read out
+		 * of the fifo and discarded */
+		pktlength += 2 + 3;
+		pktlength >>= 2;
+		smsc911x_rx_fastforward(pdata,pktlength);
+	}
+	pdata->stats.rx_dropped += SMC_regread(pdata, RX_DROP);
+	SMC_regwrite(INT_STS_RSFL_, pdata, INT_STS);
+	return;
+}
+
+
+/* Receive tasklet */
+static void smsc911x_rx_tasklet(unsigned long data)
+{
+	struct net_device *netdev;
+	struct smsc911x_data *pdata;
+
+	netdev = (struct net_device *)rx_tasklet_parameter;
+	pdata = netdev_priv(netdev);
+
+	BUG_ON(!pdata);
+	smsc911x_rx_processpackets(pdata);
+	SMC_regwrite(SMC_regread(pdata, INT_CFG) | INT_CFG_IRQ_EN_,
+		     pdata, INT_CFG);
+	return;
+}
+
+/* Receiver irq handler */
+static int smsc911x_rx_handleirq(struct smsc911x_data *pdata,
+				unsigned int intsts)
+{
+	int result;
+
+	result = 0;
+	BUG_ON(!pdata);
+	if (intsts & INT_STS_RXE_) {
+		SMC_regwrite(INT_STS_RXE_, pdata, INT_STS);
+		result = 1;
+	}
+	if (!(intsts & INT_STS_RSFL_))
+		return result;
+
+	result = 1;
+	SMC_regwrite(SMC_regread(pdata, INT_CFG) & (~INT_CFG_IRQ_EN_),
+		     pdata, INT_CFG);
+	tasklet_schedule(&rx_tasklet);
+	return result;
+}
+
+/* Returns hash bit number for given MAC address
+ * Example:
+ * 01 00 5E 00 00 01 -> returns bit number 31 */
+static unsigned int smsc911x_hash(char addr[6])
+{
+	int i;
+	int bit;
+	unsigned int crc;
+	unsigned int poly;
+	unsigned int result;
+	unsigned int data;
+
+	crc 	= 0xFFFFFFFF;
+	poly 	= 0xEDB88320;
+	result 	= 0;
+
+	for (i = 0; i < 6; i++) {
+		data = (unsigned int)addr[i];
+		for (bit = 0; bit < 8; bit++) {
+			unsigned int p = (crc ^ ((unsigned int) data)) & 1;
+			crc >>= 1;
+			if (p != 0) crc ^= poly;
+			data >>= 1;
+		}
+	}
+	result = ((crc & 0x01) << 5) | ((crc & 0x02) << 3) |
+		 ((crc & 0x04) << 1) | ((crc & 0x08) >> 1) |
+		 ((crc & 0x10) >> 3) | ((crc & 0x20) >> 5);
+	return result;
+}
+
+/* Sets addressing modes */
+static void smsc911x_rx_set_multicastlist(struct net_device *dev)
+{
+	struct smsc911x_data *pdata;
+
+	BUG_ON(!dev);
+	pdata = netdev_priv(dev);
+	BUG_ON(!pdata);
+	spin_lock(&pdata->phy_lock);
+	if (dev->flags & IFF_PROMISC) {
+		/* Enabling promiscuous mode */
+		unsigned int mac_cr = 0;
+		mac_cr = smsc911x_mac_read(pdata, MAC_CR);
+		mac_cr |= MAC_CR_PRMS_;
+		mac_cr &= (~MAC_CR_MCPAS_);
+		mac_cr &= (~MAC_CR_HPFILT_);
+		smsc911x_mac_write(pdata, MAC_CR, mac_cr);
+		return;
+	}
+	if (dev->flags & IFF_ALLMULTI) {
+		/* Enabling all multicast mode */
+		unsigned int mac_cr = 0;
+		mac_cr = smsc911x_mac_read(pdata, MAC_CR);
+		mac_cr &= (~MAC_CR_PRMS_);
+		mac_cr |= MAC_CR_MCPAS_;
+		mac_cr &= (~MAC_CR_HPFILT_);
+		smsc911x_mac_write(pdata, MAC_CR, mac_cr);
+		return;
+	}
+	if (dev->mc_count > 0) {
+		/* Enabling specific multicast addresses */
+		unsigned int hash_high = 0;
+		unsigned int hash_low = 0;
+		unsigned int count = 0;
+		unsigned int mac_cr = 0;
+		struct dev_mc_list *mc_list = dev->mc_list;
+		mac_cr = smsc911x_mac_read(pdata, MAC_CR);
+		mac_cr &= ~MAC_CR_PRMS_;
+		mac_cr &= ~MAC_CR_MCPAS_;
+		mac_cr |= MAC_CR_HPFILT_;
+
+		while (mc_list) {
+			count++;
+			if ((mc_list->dmi_addrlen) == 6) {
+				unsigned int mask = 0x01;
+				unsigned int bitnum;
+				bitnum = smsc911x_hash(mc_list->dmi_addr);
+				mask <<= (bitnum & 0x1F);
+				if(bitnum & 0x20) {
+					hash_high |= mask;
+				} else {
+					hash_low |= mask;
+				}
+			} else {
+				SMSC_WARNING("dmi_addrlen != 6");
+			}
+			mc_list = mc_list->next;
+		}
+		if (count != (unsigned int)dev->mc_count)
+			SMSC_WARNING("mc_count != dev->mc_count");
+		smsc911x_mac_write(pdata, HASHH, hash_high);
+		smsc911x_mac_write(pdata, HASHL, hash_low);
+		smsc911x_mac_write(pdata, MAC_CR, mac_cr);
+	}
+	else {
+		/* Enabling local MAC address only */
+		unsigned int mac_cr = 0;
+		mac_cr = smsc911x_mac_read(pdata, MAC_CR);
+		mac_cr &= ~MAC_CR_PRMS_;
+		mac_cr &= ~MAC_CR_MCPAS_;
+		mac_cr &= ~MAC_CR_HPFILT_;
+		smsc911x_mac_write(pdata, HASHH, 0);
+		smsc911x_mac_write(pdata, HASHL, 0);
+		smsc911x_mac_write(pdata, MAC_CR, mac_cr);
+	}
+	spin_unlock(&pdata->phy_lock);
+	return;
+}
+
+
+
+static int
+smsc911x_lan_initialise(struct smsc911x_data *pdata, unsigned int intcfg)
+{
+	int result;
+	unsigned int timeout;
+	unsigned int temp;
+
+	result 	= 0;
+	timeout = 0;
+	temp 	= 0;
+	SMSC_TRACE("LAN Initialise, int_cfg: 0x%08X", intcfg);
+	BUG_ON(!pdata);
+	spin_lock_init(&pdata->phy_lock);
+
+	/* Reset the LAN911x */
+	SMC_regwrite(HW_CFG_SRST_, pdata, HW_CFG);
+	timeout = 10;
+
+	do {
+		udelay(10);
+		temp = SMC_regread(pdata, HW_CFG);
+		timeout--;
+	} while((timeout > 0) && (temp & HW_CFG_SRST_));
+
+	if (temp & HW_CFG_SRST_) {
+		SMSC_WARNING("Failed to complete reset");
+		goto done;
+	}
+
+	SMC_regwrite(0x00050000, pdata, HW_CFG);
+	SMC_regwrite(0x006E3740, pdata, AFC_CFG);
+
+	/* Make sure EEPROM has finished loading before setting GPIO_CFG */
+	timeout = 50;
+	while ((timeout > 0) &&
+		(SMC_regread(pdata, E2P_CMD) & E2P_CMD_EPC_BUSY_)) {
+		udelay(10);
+		timeout--;
+	}
+	if (timeout == 0)
+		SMSC_WARNING("Timed out waiting for EEPROM "
+			     "busy bit to clear\n");
+
+#if USE_DEBUG >= 1
+	SMC_regwrite(0x00670700, pdata, GPIO_CFG);
+#else
+	SMC_regwrite(0x70070000, pdata, GPIO_CFG);
+#endif
+
+	/* Initialise irqs */
+	SMC_regwrite(0, pdata, INT_EN);
+	SMC_regwrite(0xFFFFFFFF, pdata, INT_STS);
+	intcfg |= INT_CFG_IRQ_EN_;
+	SMC_regwrite(intcfg, pdata, INT_CFG);
+	result = 1;
+done:
+	return result;
+}
+
+
+/* Initializing private device structures */
+static int smsc911x_init(struct net_device *netdev)
+{
+	int result;
+	unsigned long idrev;
+	unsigned int mac_high16;
+	unsigned int mac_low32;
+	struct smsc911x_data * pdata;
+
+	idrev  = 0;
+	pdata  = 0;
+	result = -ENODEV;
+
+	BUG_ON(!netdev);
+
+	SMSC_TRACE("Driver Parameters:");
+	SMSC_TRACE("LAN base: 0x%08lX", netdev->base_addr);
+	SMSC_TRACE("IRQ: %d", netdev->irq);
+	SMSC_TRACE("PHY will be autodetected.");
+	BUG_ON(!netdev_priv(netdev));
+
+	memset(netdev_priv(netdev), 0, sizeof(struct smsc911x_data));
+	pdata = netdev_priv(netdev);
+
+	if (netdev->base_addr == 0) {
+		SMSC_WARNING("netdev->base_addr: 0x00000000");
+		result = -ENODEV;
+		goto done;
+	}
+	/* Initialise internal base field */
+	pdata->base = netdev->base_addr;
+
+	/* Read mac address assigned as from EEPROM */
+	mac_high16 = smsc911x_mac_read(pdata, ADDRH);
+	mac_low32 = smsc911x_mac_read(pdata, ADDRL);
+
+	if ((mac_high16 == 0x0000FFFF)
+	    && (mac_low32 == 0xFFFFFFFF)) {
+
+		/* Use default MAC addresses if eeprom values are invalid */
+		mac_high16 = 0x00000070;
+		mac_low32 = 0x110F8000;
+		smsc911x_mac_write(pdata, ADDRH, mac_high16);
+		smsc911x_mac_write(pdata, ADDRL, mac_low32);
+		SMSC_TRACE("MAC Address is set by default to 0x%04X%08X",
+			   mac_high16, mac_low32);
+	} else {
+		SMSC_TRACE("MAC Address is read from LAN911x as 0x%04X%08X",
+			   mac_high16, mac_low32);
+	}
+
+	idrev = SMC_regread(pdata, ID_REV);
+	if (((idrev >> 16) & 0xFFFF) == (idrev & 0xFFFF)) {
+		/* this may mean the chip is set for 32 bit
+		 * while the bus is reading as 16 bit
+		 */
+unknown_chip:
+		SMSC_WARNING("LAN911x not identified, idrev: 0x%08lX", idrev);
+		result= -ENODEV;
+		goto done;
+	}
+	switch (idrev & 0xFFFF0000) {
+		case 0x01180000:
+			SMSC_TRACE("LAN9118 identified, idrev: 0x%08lX",idrev);
+			break;
+
+		case 0x01170000:
+			SMSC_TRACE("LAN9117 identified, idrev: 0x%08lX",idrev);
+			break;
+
+		case 0x01160000:
+			SMSC_TRACE("LAN9116 identified, idrev: 0x%08lX",idrev);
+			break;
+
+		case 0x01150000:
+			SMSC_TRACE("LAN9115 identified, idrev: 0x%08lX",idrev);
+			break;
+
+		case 0x01120000:
+			SMSC_TRACE("LAN9112 identified, idrev: 0x%08lX",idrev);
+			break;
+
+		default:
+			goto unknown_chip;
+	}
+	pdata->idrev = idrev;
+
+	if ((idrev & 0x0000FFFF) == 0)
+		SMSC_WARNING("This driver is not intended "
+			     "for this chip revision");
+
+	ether_setup(netdev);
+	netdev->open = smsc911x_open;
+	netdev->stop = smsc911x_stop;
+	netdev->hard_start_xmit = smsc911x_hard_start_xmit;
+	netdev->get_stats = smsc911x_get_stats;
+	netdev->set_multicast_list = smsc911x_set_multicast_list;
+	netdev->flags |= IFF_MULTICAST;
+	pdata->dev = netdev;
+
+	result = 0;
+
+done:
+	return result;
+}
+
+static int smsc911x_drv_remove(struct platform_device *pdev)
+{
+	struct net_device *netdev;
+	struct smsc911x_data * pdata;
+	struct resource * res;
+	unsigned int base;
+
+	netdev = platform_get_drvdata(pdev);
+	BUG_ON(!netdev);
+	pdata = netdev_priv(netdev);
+	BUG_ON(!pdata);
+	base = pdata->base;
+	BUG_ON(!base);
+
+	SMSC_TRACE("Stopping driver.");
+	platform_set_drvdata(pdev, NULL);
+	unregister_netdev(netdev);
+	free_irq(netdev->irq, netdev);
+	res = platform_get_resource_byname(pdev, IORESOURCE_MEM,
+					   "smsc911x-memory");
+	if (!res)
+		platform_get_resource(pdev, IORESOURCE_MEM, 0);
+
+	release_mem_region(res->start, res->end - res->start);
+	/* Clear global ptr to netdev for tasklets */
+	rx_tasklet_parameter = 0;
+	free_netdev(netdev);
+	iounmap((void *)base);
+	return 0;
+}
+
+static int smsc911x_drv_probe(struct platform_device * pdev)
+{
+	struct net_device *netdev;
+	struct resource *res;
+	unsigned int base;
+	int res_size;
+	int retval;
+	res = platform_get_resource_byname(pdev, IORESOURCE_MEM,
+					   "smsc911x-memory");
+	if (!res)
+		res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
+	if (!res) {
+		retval = -ENODEV;
+		goto out;
+	}
+	res_size = res->end - res->start;
+
+	if (!request_mem_region(res->start, res_size, SMSC_CHIPNAME)) {
+		retval = -EBUSY;
+		goto out;
+	}
+
+	netdev = alloc_etherdev(sizeof(struct smsc911x_data));
+	if (!netdev) {
+		printk("%s: Could not allocate device.\n", SMSC_CHIPNAME);
+		retval = -ENOMEM;
+		goto out_release_io;
+	}
+	SET_MODULE_OWNER(netdev);
+	SET_NETDEV_DEV(netdev, &pdev->dev);
+
+	/* Global pointer to netdev, used by tasklets */
+	rx_tasklet_parameter = (unsigned long) netdev;
+	netdev->irq = platform_get_irq(pdev, 0);
+	netdev->base_addr = (unsigned int) ioremap_nocache(res->start,
+							   res_size);
+	base = netdev->base_addr;
+	if (netdev->base_addr == 0) {
+		SMSC_WARNING("Error smsc911x base address invalid");
+		retval = -ENOMEM;
+		goto out_free_netdev;
+	}
+
+	platform_set_drvdata(pdev, netdev);
+	strcpy(netdev->name, "eth%d");
+	retval = register_netdev(netdev);
+
+	if (retval) {
+		SMSC_WARNING("Error %i registering device", retval);
+		retval = -ENODEV;
+		goto out_unmap_io;
+	} else {
+		SMSC_TRACE("Network interface: \"%s\"",netdev->name);
+	}
+
+	if ((retval = smsc911x_init(netdev)) < 0)
+		goto out_deregister_netdev;
+
+	if (request_irq(netdev->irq, smsc911x_irqhandler,
+	    SA_INTERRUPT, SMSC_CHIPNAME, netdev_priv(netdev)) != 0) {
+		SMSC_WARNING("Unable to claim requested irq: %d", netdev->irq);
+		retval = -ENODEV;
+		goto out_deregister_netdev;
+	}
+	return 0;
+
+out_deregister_netdev:
+	unregister_netdev(netdev);
+out_unmap_io:
+	platform_set_drvdata(pdev, NULL);
+	iounmap((void *) base);
+out_free_netdev:
+	rx_tasklet_parameter = 0;
+	free_netdev(netdev);
+out_release_io:
+	release_mem_region(res->start, res->end - res->start);
+out:
+	return retval;
+}
+
+static struct platform_driver smsc911x_driver = {
+	.probe		= smsc911x_drv_probe,
+	.remove		= smsc911x_drv_remove,
+	.suspend	= 0, /* TODO: Add suspend routine */
+	.resume		= 0, /* TODO: Add resume routeine */
+	.driver		= {
+		.name	= SMSC_CHIPNAME,
+	},
+};
+
+/* Entry point for loading the module ==> All below must go
+ * into drv_probe */
+static int __init smsc911x_init_module(void)
+{
+	platform_driver_register(&smsc911x_driver);
+	return 0;
+}
+
+/* entry point for unloading the module */
+static void __exit smsc911x_cleanup_module(void)
+{
+	platform_driver_unregister(&smsc911x_driver);
+}
+
+module_init(smsc911x_init_module);
+module_exit(smsc911x_cleanup_module);
+
+
diff --git a/drivers/net/smsc911x.h b/drivers/net/smsc911x.h
new file mode 100644
index 0000000..5ab2987
--- /dev/null
+++ b/drivers/net/smsc911x.h
@@ -0,0 +1,526 @@
+#ifndef __SMSC911X_H__
+#define __SMSC911X_H__
+
+#define USE_PHY_WORK_AROUND
+#define USE_LED1_WORK_AROUND	/* 10/100 LED link-state inversion */
+
+/* Debugging */
+#define USE_DEBUG	0
+#if USE_DEBUG >= 1
+#	define SMSC_WARNING(fmt, args...)			\
+		printk(KERN_EMERG "SMSC_WARNING: %s: " fmt "\n",\
+			__FUNCTION__ , ## args)
+#else
+#	define SMSC_WARNING(msg, args...)
+#endif /* USE_DEBUG >= 1 */
+
+#if USE_DEBUG >= 2
+#	define SMSC_TRACE(fmt,args...)				\
+		printk(KERN_EMERG "SMSC_TRACE: %s: " fmt "\n",	\
+			__FUNCTION__ , ## args)
+#else
+#	define SMSC_TRACE(msg,args...)
+#endif /* USE_DEBUG >= 2 */
+
+/* NOTE: If you add any more fields, make sure to decide on whether
+ * to clear the fields or retain their values in smsc911x_stop() */
+struct smsc911x_data {
+	unsigned int base;
+	unsigned int idrev;
+	struct net_device *dev;
+
+	/* This needs to be acquired before calling any of below:
+	 * smsc911x_mac_read(), smsc911x_mac_write()
+	 * smsc911x_phy_read(), smsc911x_phy_write()
+	 */
+	spinlock_t phy_lock;
+
+	int rx_congested;
+	struct net_device_stats stats;
+
+	unsigned int phy_address;
+#ifdef USE_LED1_WORK_AROUND
+	unsigned int gpio_setting;
+	unsigned int gpio_orig_setting;
+	unsigned int not_using_extphy;
+#endif
+	unsigned int link_speed;
+	unsigned int link_settings;
+	struct timer_list link_poll_timer;
+	int stop_link_poll;
+
+	int request_irq_disable;
+	int software_irq_signal;
+
+#ifdef USE_PHY_WORK_AROUND
+	#define MIN_PACKET_SIZE (64)
+	char loopback_tx_pkt[MIN_PACKET_SIZE];
+	char loopback_rx_pkt[MIN_PACKET_SIZE];
+	unsigned int resetcount;
+#endif
+};
+
+/* IO macros for portability */
+
+#if defined(CONFIG_MACH_REALVIEW_EB) || defined(CONFIG_ARCH_VERSATILE_PB)
+
+#define SMC_CAN_USE_8BIT	0
+#define SMC_CAN_USE_16BIT	0
+#define SMC_CAN_USE_32BIT	1
+
+#if SMC_CAN_USE_16BIT
+#define SMC_inb(a, r)		readb((a) + (r))
+#define SMC_inw(a, r)		readw((a) + (r))
+#define SMC_inl(a, r)		((SMC_inw((a) + (r)) & 0xFFFF) 	\
+				 | ((SMC_inw((a) + (r) + 2) & 0xFFFF) << 16))
+#define SMC_outb(v, a, r)	writeb(v, (a) + (r))
+#define SMC_outw(v, a, r)	writew(v, (a) + (r))
+#define SMC_outl(v, a, r)	SMC_outw(v & 0xFFFF, (a) + (r))	\
+				SMC_outw(((v >> 16) & 0xFFFF), (a) + (r) + 2)
+#define SMC_insl(a, r, p, l)	readsl((a) + (r), p, l)
+#define SMC_outsl(a, r, p, l)	writesl((a) + (r), p, l)
+
+#elif SMC_CAN_USE_32BIT
+#define SMC_inb(a, r)		readb((a) + (r))
+#define SMC_inw(a, r)		readw((a) + (r))
+#define SMC_inl(a, r)		readl((a) + (r))
+#define SMC_outb(v, a, r)	writeb(v, (a) + (r))
+#define SMC_outw(v, a, r)	writew(v, (a) + (r))
+#define SMC_outl(v, a, r)	writel(v, (a) + (r))
+#define SMC_insl(a, r, p, l)	readsl((a) + (r), p, l)
+#define SMC_outsl(a, r, p, l)	writesl((a) + (r), p, l)
+#endif /* SMC_CAN_USE_32BIT */
+
+static inline u32
+SMC_regread(struct smsc911x_data *pdata, u32 reg)
+{
+	return SMC_inl(pdata->base, reg);
+}
+
+static inline void
+SMC_regwrite(u32 val, struct smsc911x_data *pdata, u32 reg)
+{
+	SMC_outl(val, pdata->base, reg);
+}
+#else /* Default for all architectures */
+
+#define SMC_CAN_USE_8BIT	0
+#define SMC_CAN_USE_16BIT	0
+#define SMC_CAN_USE_32BIT	1
+
+#if SMC_CAN_USE_16BIT
+#define SMC_inb(a, r)		readb((a) + (r))
+#define SMC_inw(a, r)		readw((a) + (r))
+#define SMC_inl(a, r)		((SMC_inw((a) + (r)) & 0xFFFF) 	\
+				 | ((SMC_inw((a) + (r) + 2) & 0xFFFF) << 16))
+#define SMC_outb(v, a, r)	writeb(v, (a) + (r))
+#define SMC_outw(v, a, r)	writew(v, (a) + (r))
+#define SMC_outl(v, a, r)	SMC_outw(v & 0xFFFF, (a) + (r))	\
+				SMC_outw(((v >> 16) & 0xFFFF), (a) + (r) + 2)
+#define SMC_insl(a, r, p, l)	readsl((a) + (r), p, l)
+#define SMC_outsl(a, r, p, l)	writesl((a) + (r), p, l)
+
+#elif SMC_CAN_USE_32BIT
+#define SMC_inb(a, r)		readb((u8*)((a) + (r)))
+#define SMC_inw(a, r)		readw((u16*)((a) + (r)))
+#define SMC_inl(a, r)		readl((u32*)((a) + (r)))
+#define SMC_outb(v, a, r)	writeb(v, (u8*)((a) + (r)))
+#define SMC_outw(v, a, r)	writew(v, (u16*)((a) + (r)))
+#define SMC_outl(v, a, r)	writel(v, (u32*)((a) + (r)))
+#define SMC_insl(a, r, p, l)	readsl((a) + (r), p, l)
+#define SMC_outsl(a, r, p, l)	writesl((a) + (r), p, l)
+#endif /* SMC_CAN_USE_32BIT */
+
+static inline u32
+SMC_regread(struct smsc911x_data *pdata, u32 reg)
+{
+	return SMC_inl(pdata->base, reg);
+}
+
+static inline void
+SMC_regwrite(u32 val, struct smsc911x_data *pdata, u32 reg)
+{
+	SMC_outl(val, pdata->base, reg);
+}
+#endif /* Default for all architectures */
+
+/* SMSC911x registers and bitfields */
+#define RX_DATA_FIFO			0x00
+
+#define TX_DATA_FIFO			0x20
+#define TX_CMD_A_ON_COMP_		0x80000000
+#define TX_CMD_A_BUF_END_ALGN_		0x03000000
+#define TX_CMD_A_4_BYTE_ALGN_		0x00000000
+#define TX_CMD_A_16_BYTE_ALGN_		0x01000000
+#define TX_CMD_A_32_BYTE_ALGN_		0x02000000
+#define TX_CMD_A_DATA_OFFSET_		0x001F0000
+#define TX_CMD_A_FIRST_SEG_		0x00002000
+#define TX_CMD_A_LAST_SEG_		0x00001000
+#define TX_CMD_A_BUF_SIZE_		0x000007FF
+#define TX_CMD_B_PKT_TAG_		0xFFFF0000
+#define TX_CMD_B_ADD_CRC_DISABLE_	0x00002000
+#define TX_CMD_B_DISABLE_PADDING_	0x00001000
+#define TX_CMD_B_PKT_BYTE_LENGTH_	0x000007FF
+
+#define RX_STATUS_FIFO			0x40
+#define RX_STS_ES_			0x00008000
+#define RX_STS_MCAST_			0x00000400
+#define RX_STATUS_FIFO_PEEK		0x44
+#define TX_STATUS_FIFO			0x48
+#define TX_STATUS_FIFO_PEEK		0x4C
+#define ID_REV				0x50
+#define ID_REV_CHIP_ID_			0xFFFF0000 /* RO */
+#define ID_REV_REV_ID_			0x0000FFFF /* RO */
+
+#define INT_CFG				0x54
+#define INT_CFG_INT_DEAS_		0xFF000000 /* R/W */
+#define INT_CFG_INT_DEAS_CLR_		0x00004000 /* SC */
+#define INT_CFG_INT_DEAS_STS_		0x00002000 /* SC */
+#define INT_CFG_IRQ_INT_		0x00001000 /* RO */
+#define INT_CFG_IRQ_EN_			0x00000100 /* R/W */
+#define INT_CFG_IRQ_POL_		0x00000010 /* R/W Not Affected by SW Reset */
+#define INT_CFG_IRQ_TYPE_		0x00000001 /* R/W Not Affected by SW Reset */
+
+#define INT_STS				0x58
+#define INT_STS_SW_INT_			0x80000000 /* R/WC */
+#define INT_STS_TXSTOP_INT_		0x02000000 /* R/WC */
+#define INT_STS_RXSTOP_INT_		0x01000000 /* R/WC */
+#define INT_STS_RXDFH_INT_		0x00800000 /* R/WC */
+#define INT_STS_RXDF_INT_		0x00400000 /* R/WC */
+#define INT_STS_TX_IOC_			0x00200000 /* R/WC */
+#define INT_STS_RXD_INT_		0x00100000 /* R/WC */
+#define INT_STS_GPT_INT_		0x00080000 /* R/WC */
+#define INT_STS_PHY_INT_		0x00040000 /* RO */
+#define INT_STS_PME_INT_		0x00020000 /* R/WC */
+#define INT_STS_TXSO_			0x00010000 /* R/WC */
+#define INT_STS_RWT_			0x00008000 /* R/WC */
+#define INT_STS_RXE_			0x00004000 /* R/WC */
+#define INT_STS_TXE_			0x00002000 /* R/WC */
+#define INT_STS_TDFU_			0x00000800 /* R/WC */
+#define INT_STS_TDFO_			0x00000400 /* R/WC */
+#define INT_STS_TDFA_			0x00000200 /* R/WC */
+#define INT_STS_TSFF_			0x00000100 /* R/WC */
+#define INT_STS_TSFL_			0x00000080 /* R/WC */
+#define INT_STS_RXDF_			0x00000040 /* R/WC */
+#define INT_STS_RDFL_			0x00000020 /* R/WC */
+#define INT_STS_RSFF_			0x00000010 /* R/WC */
+#define INT_STS_RSFL_			0x00000008 /* R/WC */
+#define INT_STS_GPIO2_INT_		0x00000004 /* R/WC */
+#define INT_STS_GPIO1_INT_		0x00000002 /* R/WC */
+#define INT_STS_GPIO0_INT_		0x00000001 /* R/WC */
+
+#define INT_EN				0x5C
+#define INT_EN_SW_INT_EN_		0x80000000 /* R/W */
+#define INT_EN_TXSTOP_INT_EN_		0x02000000 /* R/W */
+#define INT_EN_RXSTOP_INT_EN_		0x01000000 /* R/W */
+#define INT_EN_RXDFH_INT_EN_		0x00800000 /* R/W */
+#define INT_EN_TIOC_INT_EN_		0x00200000 /* R/W */
+#define INT_EN_RXD_INT_EN_		0x00100000 /* R/W */
+#define INT_EN_GPT_INT_EN_		0x00080000 /* R/W */
+#define INT_EN_PHY_INT_EN_		0x00040000 /* R/W */
+#define INT_EN_PME_INT_EN_		0x00020000 /* R/W */
+#define INT_EN_TXSO_EN_			0x00010000 /* R/W */
+#define INT_EN_RWT_EN_			0x00008000 /* R/W */
+#define INT_EN_RXE_EN_			0x00004000 /* R/W */
+#define INT_EN_TXE_EN_			0x00002000 /* R/W */
+#define INT_EN_TDFU_EN_			0x00000800 /* R/W */
+#define INT_EN_TDFO_EN_			0x00000400 /* R/W */
+#define INT_EN_TDFA_EN_			0x00000200 /* R/W */
+#define INT_EN_TSFF_EN_			0x00000100 /* R/W */
+#define INT_EN_TSFL_EN_			0x00000080 /* R/W */
+#define INT_EN_RXDF_EN_			0x00000040 /* R/W */
+#define INT_EN_RDFL_EN_			0x00000020 /* R/W */
+#define INT_EN_RSFF_EN_			0x00000010 /* R/W */
+#define INT_EN_RSFL_EN_			0x00000008 /* R/W */
+#define INT_EN_GPIO2_INT_		0x00000004 /* R/W */
+#define INT_EN_GPIO1_INT_		0x00000002 /* R/W */
+#define INT_EN_GPIO0_INT_		0x00000001 /* R/W */
+
+#define BYTE_TEST			0x64
+#define FIFO_INT			0x68
+#define FIFO_INT_TX_AVAIL_LEVEL_	0xFF000000 /* R/W */
+#define FIFO_INT_TX_STS_LEVEL_		0x00FF0000 /* R/W */
+#define FIFO_INT_RX_AVAIL_LEVEL_	0x0000FF00 /* R/W */
+#define FIFO_INT_RX_STS_LEVEL_		0x000000FF /* R/W */
+
+#define RX_CFG				0x6C
+#define RX_CFG_RX_END_ALGN_		0xC0000000 /* R/W */
+#define RX_CFG_RX_END_ALGN4_		0x00000000 /* R/W */
+#define RX_CFG_RX_END_ALGN16_		0x40000000 /* R/W */
+#define RX_CFG_RX_END_ALGN32_		0x80000000 /* R/W */
+#define RX_CFG_RX_DMA_CNT_		0x0FFF0000 /* R/W */
+#define RX_CFG_RX_DUMP_			0x00008000 /* R/W */
+#define RX_CFG_RXDOFF_			0x00001F00 /* R/W */
+
+#define TX_CFG				0x70
+#define TX_CFG_TXS_DUMP_		0x00008000 /* Self Clearing */
+#define TX_CFG_TXD_DUMP_		0x00004000 /* Self Clearing */
+#define TX_CFG_TXSAO_			0x00000004 /* R/W */
+#define TX_CFG_TX_ON_			0x00000002 /* R/W */
+#define TX_CFG_STOP_TX_			0x00000001 /* Self Clearing */
+
+#define HW_CFG				0x74
+#define HW_CFG_TTM_			0x00200000 /* R/W */
+#define HW_CFG_SF_			0x00100000 /* R/W */
+#define HW_CFG_TX_FIF_SZ_		0x000F0000 /* R/W */
+#define HW_CFG_TR_			0x00003000 /* R/W */
+
+/* R/W only available on 115/117 */
+#define HW_CFG_PHY_CLK_SEL_		0x00000060
+
+/* R/W only available on 115/117 */
+#define HW_CFG_PHY_CLK_SEL_INT_PHY_	0x00000000
+
+/* R/W only available on 115/117 */
+#define HW_CFG_PHY_CLK_SEL_EXT_PHY_	0x00000020
+
+/* R/W only available on 115/117 */
+#define HW_CFG_PHY_CLK_SEL_CLK_DIS_	0x00000040
+
+/* R/W only available on 115/117 */
+#define HW_CFG_SMI_SEL_			0x00000010
+
+/* RO only available  on 115/117 */
+#define HW_CFG_EXT_PHY_DET_		0x00000008
+
+/* R/W only available on 115/117 */
+#define HW_CFG_EXT_PHY_EN_		0x00000004
+
+/* RO only available  on 116/118 */
+#define HW_CFG_32_16_BIT_MODE_		0x00000004
+
+/* RO only available  on 115/117 */
+#define HW_CFG_SRST_TO_			0x00000002
+#define HW_CFG_SRST_			0x00000001 /* Self Clearing */
+
+#define RX_DP_CTRL			0x78
+#define RX_DP_CTRL_RX_FFWD_		0x80000000 /* RO */
+
+#define RX_FIFO_INF			0x7C
+#define RX_FIFO_INF_RXSUSED_		0x00FF0000 /* RO */
+#define RX_FIFO_INF_RXDUSED_		0x0000FFFF /* RO */
+
+#define TX_FIFO_INF			0x80
+#define TX_FIFO_INF_TSUSED_		0x00FF0000 /* RO */
+#define TX_FIFO_INF_TDFREE_		0x0000FFFF /* RO */
+
+#define PMT_CTRL			0x84
+#define PMT_CTRL_PM_MODE_		0x00003000 /* Self Clearing */
+#define PMT_CTRL_PM_MODE_D0_		0x00000000 /* Self Clearing */
+#define PMT_CTRL_PM_MODE_D1_		0x00001000 /* Self Clearing */
+#define PMT_CTRL_PM_MODE_D2_		0x00002000 /* Self Clearing */
+#define PMT_CTRL_PM_MODE_D3_		0x00003000 /* Self Clearing */
+#define PMT_CTRL_PHY_RST_		0x00000400 /* Self Clearing */
+#define PMT_CTRL_WOL_EN_		0x00000200 /* R/W */
+#define PMT_CTRL_ED_EN_			0x00000100 /* R/W */
+#define PMT_CTRL_PME_TYPE_		0x00000040 /* R/W
+						    * Not Affected by
+						    * SW Reset */
+#define PMT_CTRL_WUPS_			0x00000030 /* R/WC */
+#define PMT_CTRL_WUPS_NOWAKE_		0x00000000 /* R/WC */
+#define PMT_CTRL_WUPS_ED_		0x00000010 /* R/WC */
+#define PMT_CTRL_WUPS_WOL_		0x00000020 /* R/WC */
+#define PMT_CTRL_WUPS_MULTI_		0x00000030 /* R/WC */
+#define PMT_CTRL_PME_IND_		0x00000008 /* R/W */
+#define PMT_CTRL_PME_POL_		0x00000004 /* R/W */
+#define PMT_CTRL_PME_EN_		0x00000002 /* R/W
+						    * Not Affected by
+						    * SW Reset */
+#define PMT_CTRL_READY_			0x00000001 /* RO */
+
+#define GPIO_CFG			0x88
+#define GPIO_CFG_LED3_EN_		0x40000000 /* R/W */
+#define GPIO_CFG_LED2_EN_		0x20000000 /* R/W */
+#define GPIO_CFG_LED1_EN_		0x10000000 /* R/W */
+#define GPIO_CFG_GPIO2_INT_POL_		0x04000000 /* R/W */
+#define GPIO_CFG_GPIO1_INT_POL_		0x02000000 /* R/W */
+#define GPIO_CFG_GPIO0_INT_POL_		0x01000000 /* R/W */
+#define GPIO_CFG_EEPR_EN_		0x00700000 /* R/W */
+#define GPIO_CFG_GPIOBUF2_		0x00040000 /* R/W */
+#define GPIO_CFG_GPIOBUF1_		0x00020000 /* R/W */
+#define GPIO_CFG_GPIOBUF0_		0x00010000 /* R/W */
+#define GPIO_CFG_GPIODIR2_		0x00000400 /* R/W */
+#define GPIO_CFG_GPIODIR1_		0x00000200 /* R/W */
+#define GPIO_CFG_GPIODIR0_		0x00000100 /* R/W */
+#define GPIO_CFG_GPIOD4_		0x00000020 /* R/W */
+#define GPIO_CFG_GPIOD3_		0x00000010 /* R/W */
+#define GPIO_CFG_GPIOD2_		0x00000004 /* R/W */
+#define GPIO_CFG_GPIOD1_		0x00000002 /* R/W */
+#define GPIO_CFG_GPIOD0_		0x00000001 /* R/W */
+
+#define GPT_CFG				0x8C
+#define GPT_CFG_TIMER_EN_		0x20000000 /* R/W */
+#define GPT_CFG_GPT_LOAD_		0x0000FFFF /* R/W */
+
+#define GPT_CNT				0x90
+#define GPT_CNT_GPT_CNT_		0x0000FFFF /* RO */
+
+#define ENDIAN				0x98
+#define FREE_RUN			0x9C
+#define RX_DROP				0xA0
+#define MAC_CSR_CMD			0xA4
+#define MAC_CSR_CMD_CSR_BUSY_		0x80000000 /* Self Clearing */
+#define MAC_CSR_CMD_R_NOT_W_		0x40000000 /* R/W */
+#define MAC_CSR_CMD_CSR_ADDR_		0x000000FF /* R/W */
+
+#define MAC_CSR_DATA			0xA8
+#define AFC_CFG				0xAC
+#define AFC_CFG_AFC_HI_			0x00FF0000 /* R/W */
+#define AFC_CFG_AFC_LO_			0x0000FF00 /* R/W */
+#define AFC_CFG_BACK_DUR_		0x000000F0 /* R/W */
+#define AFC_CFG_FCMULT_			0x00000008 /* R/W */
+#define AFC_CFG_FCBRD_			0x00000004 /* R/W */
+#define AFC_CFG_FCADD_			0x00000002 /* R/W */
+#define AFC_CFG_FCANY_			0x00000001 /* R/W */
+
+#define E2P_CMD				0xB0
+#define E2P_CMD_EPC_BUSY_		0x80000000 /* Self Clearing */
+#define E2P_CMD_EPC_CMD_		0x70000000 /* R/W */
+#define E2P_CMD_EPC_CMD_READ_		0x00000000 /* R/W */
+#define E2P_CMD_EPC_CMD_EWDS_		0x10000000 /* R/W */
+#define E2P_CMD_EPC_CMD_EWEN_		0x20000000 /* R/W */
+#define E2P_CMD_EPC_CMD_WRITE_		0x30000000 /* R/W */
+#define E2P_CMD_EPC_CMD_WRAL_		0x40000000 /* R/W */
+#define E2P_CMD_EPC_CMD_ERASE_		0x50000000 /* R/W */
+#define E2P_CMD_EPC_CMD_ERAL_		0x60000000 /* R/W */
+#define E2P_CMD_EPC_CMD_RELOAD_		0x70000000 /* R/W */
+#define E2P_CMD_EPC_TIMEOUT_		0x00000200 /* R */
+#define E2P_CMD_MAC_ADDR_LOADED_	0x00000100 /* RO */
+#define E2P_CMD_EPC_ADDR_		0x000000FF /* R/W */
+
+#define E2P_DATA			0xB4
+#define E2P_DATA_EEPROM_DATA_		0x000000FF /* R/W */
+#define LAN_REGISTER_EXTENT		0x00000100
+
+#define LINK_OFF			0x00
+#define LINK_SPEED_10HD			0x01
+#define LINK_SPEED_10FD			0x02
+#define LINK_SPEED_100HD		0x04
+#define LINK_SPEED_100FD		0x08
+#define LINK_SYMMETRIC_PAUSE		0x10
+#define LINK_ASYMMETRIC_PAUSE		0x20
+#define LINK_AUTO_NEGOTIATE		0x40
+
+/*
+ *	MAC Control and Status Register (Indirect Address)
+ *	Offset (through the MAC_CSR CMD and DATA port)
+ */
+#define MAC_CR				0x01 /* R/W */
+
+/* MAC_CR - MAC Control Register */
+#define MAC_CR_RXALL_			0x80000000
+#define MAC_CR_HBDIS_			0x10000000
+#define MAC_CR_RCVOWN_			0x00800000
+#define MAC_CR_LOOPBK_			0x00200000
+#define MAC_CR_FDPX_			0x00100000
+#define MAC_CR_MCPAS_			0x00080000
+#define MAC_CR_PRMS_			0x00040000
+#define MAC_CR_INVFILT_			0x00020000
+#define MAC_CR_PASSBAD_			0x00010000
+#define MAC_CR_HFILT_			0x00008000
+#define MAC_CR_HPFILT_			0x00002000
+#define MAC_CR_LCOLL_			0x00001000
+#define MAC_CR_BCAST_			0x00000800
+#define MAC_CR_DISRTY_			0x00000400
+#define MAC_CR_PADSTR_			0x00000100
+#define MAC_CR_BOLMT_MASK_		0x000000C0
+#define MAC_CR_DFCHK_			0x00000020
+#define MAC_CR_TXEN_			0x00000008
+#define MAC_CR_RXEN_			0x00000004
+
+#define ADDRH				0x02 /* R/W mask 0x0000FFFFUL */
+#define ADDRL				0x03 /* R/W mask 0xFFFFFFFFUL */
+#define HASHH				0x04 /* R/W */
+#define HASHL				0x05 /* R/W */
+
+#define MII_ACC				0x06 /* R/W */
+#define MII_ACC_PHY_ADDR_		0x0000F800
+#define MII_ACC_MIIRINDA_		0x000007C0
+#define MII_ACC_MII_WRITE_		0x00000002
+#define MII_ACC_MII_BUSY_		0x00000001
+
+#define MII_DATA			0x07 /* R/W mask 0x0000FFFFUL */
+
+#define FLOW				0x08 /* R/W */
+#define FLOW_FCPT_			0xFFFF0000
+#define FLOW_FCPASS_			0x00000004
+#define FLOW_FCEN_			0x00000002
+#define FLOW_FCBSY_			0x00000001
+
+#define VLAN1				0x09 /* R/W mask 0x0000FFFFUL */
+#define VLAN2				0x0A /* R/W mask 0x0000FFFFUL */
+
+#define WUFF				0x0B /* WO */
+
+#define WUCSR				0x0C /* R/W */
+#define WUCSR_GUE_			0x00000200
+#define WUCSR_WUFR_			0x00000040
+#define WUCSR_MPR_			0x00000020
+#define WUCSR_WAKE_EN_			0x00000004
+#define WUCSR_MPEN_			0x00000002
+
+
+/*
+ * Phy register offsets and bit definitions
+ */
+#define LAN9118_PHY_ID			0x00C0001C
+
+#define PHY_BCR				0
+#define PHY_BCR_RESET_			0x8000
+#define PHY_BCR_SPEED_SELECT_		0x2000
+#define PHY_BCR_AUTO_NEG_ENABLE_	0x1000
+#define PHY_BCR_RESTART_AUTO_NEG_	0x0200
+#define PHY_BCR_DUPLEX_MODE_		0x0100
+
+#define PHY_BSR				1
+#define PHY_BSR_LINK_STATUS_		0x0004
+#define PHY_BSR_REMOTE_FAULT_		0x0010
+#define PHY_BSR_AUTO_NEG_COMP_		0x0020
+
+#define PHY_ID_1			2
+#define PHY_ID_2			3
+
+#define PHY_ANEG_ADV			4
+#define PHY_ANEG_ADV_PAUSE_		0x0C00
+#define PHY_ANEG_ADV_ASYMP_		0x0800
+#define PHY_ANEG_ADV_SYMP_		0x0400
+#define PHY_ANEG_ADV_10H_		0x020
+#define PHY_ANEG_ADV_10F_		0x040
+#define PHY_ANEG_ADV_100H_		0x080
+#define PHY_ANEG_ADV_100F_		0x100
+#define PHY_ANEG_ADV_SPEED_		0x1E0
+
+#define PHY_ANEG_LPA			5
+#define PHY_ANEG_LPA_ASYMP_		0x0800
+#define PHY_ANEG_LPA_SYMP_		0x0400
+#define PHY_ANEG_LPA_100FDX_		0x0100
+#define PHY_ANEG_LPA_100HDX_		0x0080
+#define PHY_ANEG_LPA_10FDX_		0x0040
+#define PHY_ANEG_LPA_10HDX_		0x0020
+
+/* Mode Control/Status Register */
+#define PHY_MODE_CTRL_STS		17
+#define MODE_CTRL_STS_EDPWRDOWN_	0x2000
+#define MODE_CTRL_STS_ENERGYON_		0x0002
+
+#define PHY_INT_SRC			29
+#define PHY_INT_SRC_ENERGY_ON_		0x0080
+#define PHY_INT_SRC_ANEG_COMP_		0x0040
+#define PHY_INT_SRC_REMOTE_FAULT_	0x0020
+#define PHY_INT_SRC_LINK_DOWN_		0x0010
+
+#define PHY_INT_MASK			30
+#define PHY_INT_MASK_ENERGY_ON_		0x0080
+#define PHY_INT_MASK_ANEG_COMP_		0x0040
+#define PHY_INT_MASK_REMOTE_FAULT_	0x0020
+#define PHY_INT_MASK_LINK_DOWN_		0x0010
+
+#define PHY_SPECIAL			31
+#define PHY_SPECIAL_SPD_		0x001C
+#define PHY_SPECIAL_SPD_10HALF_		0x0004
+#define PHY_SPECIAL_SPD_10FULL_		0x0014
+#define PHY_SPECIAL_SPD_100HALF_	0x0008
+#define PHY_SPECIAL_SPD_100FULL_	0x0018
+
+
+#endif /* __SMSC911X_H__ */
-
: send the line "unsubscribe linux-net" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html

[Index of Archives]     [Netdev]     [Ethernet Bridging]     [Linux 802.1Q VLAN]     [Linux Wireless]     [Kernel Newbies]     [Security]     [Linux for Hams]     [Netfilter]     [Git]     [Bugtraq]     [Yosemite News and Information]     [MIPS Linux]     [ARM Linux]     [Linux RAID]     [Linux PCI]     [Linux Admin]     [Samba]

  Powered by Linux