summaryrefslogtreecommitdiffstats
path: root/patches/barebox-2012.12.0/0019-drivers-net-add-driver-for-TI-CPSW.patch
diff options
context:
space:
mode:
Diffstat (limited to 'patches/barebox-2012.12.0/0019-drivers-net-add-driver-for-TI-CPSW.patch')
-rw-r--r--patches/barebox-2012.12.0/0019-drivers-net-add-driver-for-TI-CPSW.patch1229
1 files changed, 0 insertions, 1229 deletions
diff --git a/patches/barebox-2012.12.0/0019-drivers-net-add-driver-for-TI-CPSW.patch b/patches/barebox-2012.12.0/0019-drivers-net-add-driver-for-TI-CPSW.patch
deleted file mode 100644
index 4332937..0000000
--- a/patches/barebox-2012.12.0/0019-drivers-net-add-driver-for-TI-CPSW.patch
+++ /dev/null
@@ -1,1229 +0,0 @@
-From 8b84f30d8d258e4e2da3d819b588809e231efdeb Mon Sep 17 00:00:00 2001
-From: Jan Luebbe <jlu@pengutronix.de>
-Date: Thu, 13 Dec 2012 17:30:49 +0100
-Subject: [PATCH] drivers: net: add driver for TI CPSW
-
-Signed-off-by: Jan Luebbe <jlu@pengutronix.de>
-Signed-off-by: Sascha Hauer <s.hauer@pengutronix.de>
----
- arch/arm/boards/beaglebone/board.c | 80 +++
- arch/arm/mach-omap/include/mach/cpsw.h | 49 ++
- drivers/net/Kconfig | 4 +
- drivers/net/Makefile | 1 +
- drivers/net/cpsw.c | 1021 ++++++++++++++++++++++++++++++++
- 5 files changed, 1155 insertions(+)
- create mode 100644 arch/arm/mach-omap/include/mach/cpsw.h
- create mode 100644 drivers/net/cpsw.c
-
-diff --git a/arch/arm/boards/beaglebone/board.c b/arch/arm/boards/beaglebone/board.c
-index d139e94..20b4bdd 100644
---- a/arch/arm/boards/beaglebone/board.c
-+++ b/arch/arm/boards/beaglebone/board.c
-@@ -31,6 +31,7 @@
- #include <sizes.h>
- #include <io.h>
- #include <ns16550.h>
-+#include <net.h>
- #include <asm/armlinux.h>
- #include <generated/mach-types.h>
- #include <mach/am33xx-silicon.h>
-@@ -42,6 +43,7 @@
- #include <mach/ehci.h>
- #include <i2c/i2c.h>
- #include <linux/err.h>
-+#include <linux/phy.h>
- #include <usb/ehci.h>
- #include <mach/xload.h>
- #include <mach/am33xx-devices.h>
-@@ -306,6 +308,84 @@ static int beaglebone_console_init(void)
- console_initcall(beaglebone_console_init);
- #endif /* CONFIG_DRIVER_SERIAL_NS16550 */
-
-+#ifdef CONFIG_DRIVER_NET_CPSW
-+static void cpsw_control(int enabled)
-+{
-+ return;
-+}
-+
-+static struct cpsw_slave_data cpsw_slaves[] = {
-+ {
-+ .slave_reg_ofs = 0x208,
-+ .sliver_reg_ofs = 0xd80,
-+ .phy_id = 0,
-+ .phy_if = PHY_INTERFACE_MODE_MII,
-+ },
-+ {
-+ .slave_reg_ofs = 0x308,
-+ .sliver_reg_ofs = 0xdc0,
-+ .phy_id = 1,
-+ .phy_if = PHY_INTERFACE_MODE_MII,
-+ },
-+};
-+
-+static struct cpsw_platform_data cpsw_data = {
-+ .mdio_base = AM335X_CPSW_MDIO_BASE,
-+ .cpsw_base = AM335X_CPSW_BASE,
-+ .mdio_div = 0xff,
-+ .channels = 8,
-+ .cpdma_reg_ofs = 0x800,
-+ .slaves = 2,
-+ .slave_data = cpsw_slaves,
-+ .ale_reg_ofs = 0xd00,
-+ .ale_entries = 1024,
-+ .host_port_reg_ofs = 0x108,
-+ .hw_stats_reg_ofs = 0x900,
-+ .mac_control = (1 << 5) /* MIIEN */,
-+ .control = cpsw_control,
-+ .host_port_num = 0,
-+ .version = CPSW_CTRL_VERSION_2,
-+};
-+
-+static int beaglebone_eth_init(void)
-+{
-+ uint8_t mac_addr[6];
-+ uint32_t mac_hi, mac_lo;
-+
-+ mac_lo = readl(AM33XX_MAC_ID0_LO);
-+ mac_hi = readl(AM33XX_MAC_ID0_HI);
-+ mac_addr[0] = mac_hi & 0xFF;
-+ mac_addr[1] = (mac_hi & 0xFF00) >> 8;
-+ mac_addr[2] = (mac_hi & 0xFF0000) >> 16;
-+ mac_addr[3] = (mac_hi & 0xFF000000) >> 24;
-+ mac_addr[4] = mac_lo & 0xFF;
-+ mac_addr[5] = (mac_lo & 0xFF00) >> 8;
-+ if (is_valid_ether_addr(mac_addr))
-+ eth_register_ethaddr(0, mac_addr);
-+
-+ mac_lo = readl(AM33XX_MAC_ID1_LO);
-+ mac_hi = readl(AM33XX_MAC_ID1_HI);
-+ mac_addr[0] = mac_hi & 0xFF;
-+ mac_addr[1] = (mac_hi & 0xFF00) >> 8;
-+ mac_addr[2] = (mac_hi & 0xFF0000) >> 16;
-+ mac_addr[3] = (mac_hi & 0xFF000000) >> 24;
-+ mac_addr[4] = mac_lo & 0xFF;
-+ mac_addr[5] = (mac_lo & 0xFF00) >> 8;
-+ if (is_valid_ether_addr(mac_addr))
-+ eth_register_ethaddr(1, mac_addr);
-+
-+ writel(0, AM33XX_MAC_MII_SEL);
-+
-+ enable_mii1_pin_mux();
-+
-+ add_generic_device("cpsw", 0, NULL, AM335X_CPSW_BASE, SZ_4K,
-+ IORESOURCE_MEM, &cpsw_data);
-+
-+ return 0;
-+}
-+device_initcall(beaglebone_eth_init);
-+#endif
-+
- static int beaglebone_mem_init(void)
- {
- arm_add_mem_device("ram0", 0x80000000, 256 * 1024 * 1024);
-diff --git a/arch/arm/mach-omap/include/mach/cpsw.h b/arch/arm/mach-omap/include/mach/cpsw.h
-new file mode 100644
-index 0000000..c101c93
---- /dev/null
-+++ b/arch/arm/mach-omap/include/mach/cpsw.h
-@@ -0,0 +1,49 @@
-+/*
-+ * CPSW Ethernet Switch Driver
-+ *
-+ * Copyright (C) 2010 Texas Instruments Incorporated - http://www.ti.com/
-+ *
-+ * 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 version 2.
-+ *
-+ * This program is distributed "as is" WITHOUT ANY WARRANTY of any
-+ * kind, whether express or implied; without even the implied warranty
-+ * of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-+ * GNU General Public License for more details.
-+ */
-+
-+#ifndef _CPSW_H_
-+#define _CPSW_H_
-+
-+struct cpsw_slave_data {
-+ u32 slave_reg_ofs;
-+ u32 sliver_reg_ofs;
-+ int phy_id;
-+ int phy_if;
-+};
-+
-+enum {
-+ CPSW_CTRL_VERSION_1 = 0,
-+ CPSW_CTRL_VERSION_2 /* am33xx like devices */
-+};
-+
-+struct cpsw_platform_data {
-+ u32 mdio_base;
-+ u32 cpsw_base;
-+ int mdio_div;
-+ int channels; /* number of cpdma channels (symmetric) */
-+ u32 cpdma_reg_ofs; /* cpdma register offset */
-+ int slaves; /* number of slave cpgmac ports */
-+ u32 ale_reg_ofs; /* address lookup engine reg offset */
-+ int ale_entries; /* ale table size */
-+ u32 host_port_reg_ofs; /* cpdma host port registers */
-+ u32 hw_stats_reg_ofs; /* cpsw hw stats counters */
-+ u32 mac_control;
-+ struct cpsw_slave_data *slave_data;
-+ void (*control)(int enabled);
-+ u32 host_port_num;
-+ u8 version;
-+};
-+
-+#endif /* _CPSW_H_ */
-diff --git a/drivers/net/Kconfig b/drivers/net/Kconfig
-index 43d5984..cf6ceee 100644
---- a/drivers/net/Kconfig
-+++ b/drivers/net/Kconfig
-@@ -31,6 +31,10 @@ config DRIVER_NET_CS8900
- bool "cs8900 ethernet driver"
- depends on HAS_CS8900
-
-+config DRIVER_NET_CPSW
-+ bool "CPSW ethernet driver"
-+ select PHYLIB
-+
- config DRIVER_NET_SMC911X
- bool "smc911x ethernet driver"
- select PHYLIB
-diff --git a/drivers/net/Makefile b/drivers/net/Makefile
-index 4e6b49b..6e1aeaa 100644
---- a/drivers/net/Makefile
-+++ b/drivers/net/Makefile
-@@ -1,4 +1,5 @@
- obj-$(CONFIG_DRIVER_NET_CS8900) += cs8900.o
-+obj-$(CONFIG_DRIVER_NET_CPSW) += cpsw.o
- obj-$(CONFIG_DRIVER_NET_SMC911X) += smc911x.o
- obj-$(CONFIG_DRIVER_NET_SMC91111) += smc91111.o
- obj-$(CONFIG_DRIVER_NET_DAVINCI_EMAC) += davinci_emac.o
-diff --git a/drivers/net/cpsw.c b/drivers/net/cpsw.c
-new file mode 100644
-index 0000000..69639ff
---- /dev/null
-+++ b/drivers/net/cpsw.c
-@@ -0,0 +1,1021 @@
-+/*
-+ * CPSW Ethernet Switch Driver
-+ *
-+ * See file CREDITS for list of people who contributed to this
-+ * project.
-+ *
-+ * 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., 675 Mass Ave, Cambridge, MA 02139, USA.
-+ */
-+
-+#include <common.h>
-+#include <driver.h>
-+#include <init.h>
-+
-+#include <command.h>
-+#include <net.h>
-+#include <malloc.h>
-+#include <net.h>
-+#include <linux/phy.h>
-+#include <errno.h>
-+#include <io.h>
-+#include <xfuncs.h>
-+#include <asm/mmu.h>
-+#include <asm/system.h>
-+
-+#include <mach/cpsw.h>
-+
-+#define BITMASK(bits) ((1 << (bits)) - 1)
-+#define PHY_REG_MASK 0x1f
-+#define PHY_ID_MASK 0x1f
-+#define NUM_DESCS (PKTBUFSRX * 2)
-+#define PKT_MIN 60
-+#define PKT_MAX (1500 + 14 + 4 + 4)
-+
-+/* DMA Registers */
-+#define CPDMA_TXCONTROL 0x004
-+#define CPDMA_RXCONTROL 0x014
-+#define CPDMA_SOFTRESET 0x01c
-+#define CPDMA_RXFREE 0x0e0
-+#define CPDMA_TXHDP_VER1 0x100
-+#define CPDMA_TXHDP_VER2 0x200
-+#define CPDMA_RXHDP_VER1 0x120
-+#define CPDMA_RXHDP_VER2 0x220
-+#define CPDMA_TXCP_VER1 0x140
-+#define CPDMA_TXCP_VER2 0x240
-+#define CPDMA_RXCP_VER1 0x160
-+#define CPDMA_RXCP_VER2 0x260
-+
-+#define CPDMA_RAM_ADDR 0x4a102000
-+
-+/* Descriptor mode bits */
-+#define CPDMA_DESC_SOP BIT(31)
-+#define CPDMA_DESC_EOP BIT(30)
-+#define CPDMA_DESC_OWNER BIT(29)
-+#define CPDMA_DESC_EOQ BIT(28)
-+
-+struct cpsw_mdio_regs {
-+ u32 version;
-+ u32 control;
-+#define CONTROL_IDLE (1 << 31)
-+#define CONTROL_ENABLE (1 << 30)
-+
-+ u32 alive;
-+ u32 link;
-+ u32 linkintraw;
-+ u32 linkintmasked;
-+ u32 __reserved_0[2];
-+ u32 userintraw;
-+ u32 userintmasked;
-+ u32 userintmaskset;
-+ u32 userintmaskclr;
-+ u32 __reserved_1[20];
-+
-+ struct {
-+ u32 access;
-+ u32 physel;
-+#define USERACCESS_GO (1 << 31)
-+#define USERACCESS_WRITE (1 << 30)
-+#define USERACCESS_ACK (1 << 29)
-+#define USERACCESS_READ (0)
-+#define USERACCESS_DATA (0xffff)
-+ } user[0];
-+};
-+
-+struct cpsw_regs {
-+ u32 id_ver;
-+ u32 control;
-+ u32 soft_reset;
-+ u32 stat_port_en;
-+ u32 ptype;
-+};
-+
-+struct cpsw_slave_regs {
-+ u32 max_blks;
-+ u32 blk_cnt;
-+ u32 flow_thresh;
-+ u32 port_vlan;
-+ u32 tx_pri_map;
-+#ifdef CONFIG_TI814X
-+ u32 ts_ctl;
-+ u32 ts_seq_ltype;
-+ u32 ts_vlan;
-+#endif
-+ u32 sa_lo;
-+ u32 sa_hi;
-+};
-+
-+struct cpsw_host_regs {
-+ u32 max_blks;
-+ u32 blk_cnt;
-+ u32 flow_thresh;
-+ u32 port_vlan;
-+ u32 tx_pri_map;
-+ u32 cpdma_tx_pri_map;
-+ u32 cpdma_rx_chan_map;
-+};
-+
-+struct cpsw_sliver_regs {
-+ u32 id_ver;
-+ u32 mac_control;
-+ u32 mac_status;
-+ u32 soft_reset;
-+ u32 rx_maxlen;
-+ u32 __reserved_0;
-+ u32 rx_pause;
-+ u32 tx_pause;
-+ u32 __reserved_1;
-+ u32 rx_pri_map;
-+};
-+
-+#define ALE_ENTRY_BITS 68
-+#define ALE_ENTRY_WORDS DIV_ROUND_UP(ALE_ENTRY_BITS, 32)
-+
-+/* ALE Registers */
-+#define ALE_CONTROL 0x08
-+#define ALE_UNKNOWNVLAN 0x18
-+#define ALE_TABLE_CONTROL 0x20
-+#define ALE_TABLE 0x34
-+#define ALE_PORTCTL 0x40
-+
-+#define ALE_TABLE_WRITE BIT(31)
-+
-+#define ALE_TYPE_FREE 0
-+#define ALE_TYPE_ADDR 1
-+#define ALE_TYPE_VLAN 2
-+#define ALE_TYPE_VLAN_ADDR 3
-+
-+#define ALE_UCAST_PERSISTANT 0
-+#define ALE_UCAST_UNTOUCHED 1
-+#define ALE_UCAST_OUI 2
-+#define ALE_UCAST_TOUCHED 3
-+
-+#define ALE_MCAST_FWD 0
-+#define ALE_MCAST_BLOCK_LEARN_FWD 1
-+#define ALE_MCAST_FWD_LEARN 2
-+#define ALE_MCAST_FWD_2 3
-+
-+enum cpsw_ale_port_state {
-+ ALE_PORT_STATE_DISABLE = 0x00,
-+ ALE_PORT_STATE_BLOCK = 0x01,
-+ ALE_PORT_STATE_LEARN = 0x02,
-+ ALE_PORT_STATE_FORWARD = 0x03,
-+};
-+
-+/* ALE unicast entry flags - passed into cpsw_ale_add_ucast() */
-+#define ALE_SECURE 1
-+#define ALE_BLOCKED 2
-+
-+struct cpsw_slave {
-+ struct cpsw_slave_regs *regs;
-+ struct cpsw_sliver_regs *sliver;
-+ int slave_num;
-+ u32 mac_control;
-+ struct cpsw_slave_data *data;
-+};
-+
-+struct cpdma_desc {
-+ /* hardware fields */
-+ u32 hw_next;
-+ u32 hw_buffer;
-+ u32 hw_len;
-+ u32 hw_mode;
-+ /* software fields */
-+ u32 sw_buffer;
-+ u32 sw_len;
-+};
-+
-+struct cpdma_chan {
-+ struct cpdma_desc *head, *tail;
-+ void *hdp, *cp, *rxfree;
-+};
-+
-+#define desc_write(desc, fld, val) __raw_writel((u32)(val), &(desc)->fld)
-+#define desc_read(desc, fld) __raw_readl(&(desc)->fld)
-+#define desc_read_ptr(desc, fld) ((void *)__raw_readl(&(desc)->fld))
-+
-+#define chan_write(chan, fld, val) __raw_writel((u32)(val), (chan)->fld)
-+#define chan_read(chan, fld) __raw_readl((chan)->fld)
-+#define chan_read_ptr(chan, fld) ((void *)__raw_readl((chan)->fld))
-+
-+struct cpsw_priv {
-+ struct device_d *dev;
-+ struct eth_device edev;
-+ struct mii_bus miibus;
-+
-+ struct cpsw_platform_data data;
-+ int host_port;
-+ uint8_t mac_addr[6];
-+
-+ struct cpsw_regs *regs;
-+ void *dma_regs;
-+ struct cpsw_host_regs *host_port_regs;
-+ void *ale_regs;
-+
-+ struct cpdma_desc *descs;
-+ struct cpdma_desc *desc_free;
-+ struct cpdma_chan rx_chan, tx_chan;
-+
-+ struct cpsw_slave *slaves;
-+#define for_each_slave(priv, func, arg...) \
-+ do { \
-+ int idx; \
-+ for (idx = 0; idx < (priv)->data.slaves; idx++) \
-+ (func)((priv)->slaves + idx, ##arg); \
-+ } while (0)
-+};
-+
-+static inline int cpsw_ale_get_field(u32 *ale_entry, u32 start, u32 bits)
-+{
-+ int idx;
-+
-+ idx = start / 32;
-+ start -= idx * 32;
-+ idx = 2 - idx; /* flip */
-+ return (ale_entry[idx] >> start) & BITMASK(bits);
-+}
-+
-+static inline void cpsw_ale_set_field(u32 *ale_entry, u32 start, u32 bits,
-+ u32 value)
-+{
-+ int idx;
-+
-+ value &= BITMASK(bits);
-+ idx = start / 32;
-+ start -= idx * 32;
-+ idx = 2 - idx; /* flip */
-+ ale_entry[idx] &= ~(BITMASK(bits) << start);
-+ ale_entry[idx] |= (value << start);
-+}
-+
-+#define DEFINE_ALE_FIELD(name, start, bits) \
-+static inline int cpsw_ale_get_##name(u32 *ale_entry) \
-+{ \
-+ return cpsw_ale_get_field(ale_entry, start, bits); \
-+} \
-+static inline void cpsw_ale_set_##name(u32 *ale_entry, u32 value) \
-+{ \
-+ cpsw_ale_set_field(ale_entry, start, bits, value); \
-+}
-+
-+DEFINE_ALE_FIELD(entry_type, 60, 2)
-+DEFINE_ALE_FIELD(mcast_state, 62, 2)
-+DEFINE_ALE_FIELD(port_mask, 66, 3)
-+DEFINE_ALE_FIELD(ucast_type, 62, 2)
-+DEFINE_ALE_FIELD(port_num, 66, 2)
-+DEFINE_ALE_FIELD(blocked, 65, 1)
-+DEFINE_ALE_FIELD(secure, 64, 1)
-+DEFINE_ALE_FIELD(mcast, 40, 1)
-+
-+char ethbdaddr [6] = { 0xff, 0xff, 0xff, 0xff, 0xff, 0xff };
-+
-+/* The MAC address field in the ALE entry cannot be macroized as above */
-+static inline void cpsw_ale_get_addr(u32 *ale_entry, u8 *addr)
-+{
-+ int i;
-+
-+ for (i = 0; i < 6; i++)
-+ addr[i] = cpsw_ale_get_field(ale_entry, 40 - 8*i, 8);
-+}
-+
-+static inline void cpsw_ale_set_addr(u32 *ale_entry, u8 *addr)
-+{
-+ int i;
-+
-+ for (i = 0; i < 6; i++)
-+ cpsw_ale_set_field(ale_entry, 40 - 8*i, 8, addr[i]);
-+}
-+
-+static int cpsw_ale_read(struct cpsw_priv *priv, int idx, u32 *ale_entry)
-+{
-+ int i;
-+
-+ __raw_writel(idx, priv->ale_regs + ALE_TABLE_CONTROL);
-+
-+ for (i = 0; i < ALE_ENTRY_WORDS; i++)
-+ ale_entry[i] = __raw_readl(priv->ale_regs + ALE_TABLE + 4 * i);
-+
-+ return idx;
-+}
-+
-+static int cpsw_ale_write(struct cpsw_priv *priv, int idx, u32 *ale_entry)
-+{
-+ int i;
-+
-+ for (i = 0; i < ALE_ENTRY_WORDS; i++)
-+ __raw_writel(ale_entry[i], priv->ale_regs + ALE_TABLE + 4 * i);
-+
-+ __raw_writel(idx | ALE_TABLE_WRITE, priv->ale_regs + ALE_TABLE_CONTROL);
-+
-+ return idx;
-+}
-+
-+static int cpsw_ale_match_addr(struct cpsw_priv *priv, u8* addr)
-+{
-+ u32 ale_entry[ALE_ENTRY_WORDS];
-+ int type, idx;
-+
-+ for (idx = 0; idx < priv->data.ale_entries; idx++) {
-+ u8 entry_addr[6];
-+
-+ cpsw_ale_read(priv, idx, ale_entry);
-+ type = cpsw_ale_get_entry_type(ale_entry);
-+ if (type != ALE_TYPE_ADDR && type != ALE_TYPE_VLAN_ADDR)
-+ continue;
-+ cpsw_ale_get_addr(ale_entry, entry_addr);
-+ if (memcmp(entry_addr, addr, 6) == 0)
-+ return idx;
-+ }
-+ return -ENOENT;
-+}
-+
-+static int cpsw_ale_match_free(struct cpsw_priv *priv)
-+{
-+ u32 ale_entry[ALE_ENTRY_WORDS];
-+ int type, idx;
-+
-+ for (idx = 0; idx < priv->data.ale_entries; idx++) {
-+ cpsw_ale_read(priv, idx, ale_entry);
-+ type = cpsw_ale_get_entry_type(ale_entry);
-+ if (type == ALE_TYPE_FREE)
-+ return idx;
-+ }
-+ return -ENOENT;
-+}
-+
-+static int cpsw_ale_find_ageable(struct cpsw_priv *priv)
-+{
-+ u32 ale_entry[ALE_ENTRY_WORDS];
-+ int type, idx;
-+
-+ for (idx = 0; idx < priv->data.ale_entries; idx++) {
-+ cpsw_ale_read(priv, idx, ale_entry);
-+ type = cpsw_ale_get_entry_type(ale_entry);
-+ if (type != ALE_TYPE_ADDR && type != ALE_TYPE_VLAN_ADDR)
-+ continue;
-+ if (cpsw_ale_get_mcast(ale_entry))
-+ continue;
-+ type = cpsw_ale_get_ucast_type(ale_entry);
-+ if (type != ALE_UCAST_PERSISTANT &&
-+ type != ALE_UCAST_OUI)
-+ return idx;
-+ }
-+ return -ENOENT;
-+}
-+
-+static int cpsw_ale_add_ucast(struct cpsw_priv *priv, u8 *addr,
-+ int port, int flags)
-+{
-+ u32 ale_entry[ALE_ENTRY_WORDS] = {0, 0, 0};
-+ int idx;
-+
-+ cpsw_ale_set_entry_type(ale_entry, ALE_TYPE_ADDR);
-+ cpsw_ale_set_addr(ale_entry, addr);
-+ cpsw_ale_set_ucast_type(ale_entry, ALE_UCAST_PERSISTANT);
-+ cpsw_ale_set_secure(ale_entry, (flags & ALE_SECURE) ? 1 : 0);
-+ cpsw_ale_set_blocked(ale_entry, (flags & ALE_BLOCKED) ? 1 : 0);
-+ cpsw_ale_set_port_num(ale_entry, port);
-+
-+ idx = cpsw_ale_match_addr(priv, addr);
-+ if (idx < 0)
-+ idx = cpsw_ale_match_free(priv);
-+ if (idx < 0)
-+ idx = cpsw_ale_find_ageable(priv);
-+ if (idx < 0)
-+ return -ENOMEM;
-+
-+ cpsw_ale_write(priv, idx, ale_entry);
-+ return 0;
-+}
-+
-+static int cpsw_ale_add_mcast(struct cpsw_priv *priv, u8 *addr, int port_mask)
-+{
-+ u32 ale_entry[ALE_ENTRY_WORDS] = {0, 0, 0};
-+ int idx, mask;
-+
-+ idx = cpsw_ale_match_addr(priv, addr);
-+ if (idx >= 0)
-+ cpsw_ale_read(priv, idx, ale_entry);
-+
-+ cpsw_ale_set_entry_type(ale_entry, ALE_TYPE_ADDR);
-+ cpsw_ale_set_addr(ale_entry, addr);
-+ cpsw_ale_set_mcast_state(ale_entry, ALE_MCAST_FWD_2);
-+
-+ mask = cpsw_ale_get_port_mask(ale_entry);
-+ port_mask |= mask;
-+ cpsw_ale_set_port_mask(ale_entry, port_mask);
-+
-+ if (idx < 0)
-+ idx = cpsw_ale_match_free(priv);
-+ if (idx < 0)
-+ idx = cpsw_ale_find_ageable(priv);
-+ if (idx < 0)
-+ return -ENOMEM;
-+
-+ cpsw_ale_write(priv, idx, ale_entry);
-+ return 0;
-+}
-+
-+static inline void cpsw_ale_control(struct cpsw_priv *priv, int bit, int val)
-+{
-+ u32 tmp, mask = BIT(bit);
-+
-+ tmp = __raw_readl(priv->ale_regs + ALE_CONTROL);
-+ tmp &= ~mask;
-+ tmp |= val ? mask : 0;
-+ __raw_writel(tmp, priv->ale_regs + ALE_CONTROL);
-+}
-+
-+#define cpsw_ale_enable(priv, val) cpsw_ale_control(priv, 31, val)
-+#define cpsw_ale_clear(priv, val) cpsw_ale_control(priv, 30, val)
-+#define cpsw_ale_bypass(priv, val) cpsw_ale_control(priv, 4, val)
-+#define cpsw_ale_vlan_aware(priv, val) cpsw_ale_control(priv, 2, val)
-+
-+static inline void cpsw_ale_port_state(struct cpsw_priv *priv, int port,
-+ int val)
-+{
-+ int offset = ALE_PORTCTL + 4 * port;
-+ u32 tmp, mask = 0x3;
-+
-+ tmp = __raw_readl(priv->ale_regs + offset);
-+ tmp &= ~mask;
-+ tmp |= val & 0x3;
-+ __raw_writel(tmp, priv->ale_regs + offset);
-+}
-+
-+static struct cpsw_mdio_regs *mdio_regs;
-+
-+/* wait until hardware is ready for another user access */
-+static inline u32 wait_for_user_access(void)
-+{
-+ u32 tmp;
-+
-+ while ((tmp = __raw_readl(&mdio_regs->user[0].access)) & USERACCESS_GO)
-+ ;
-+
-+ return tmp;
-+}
-+
-+/* wait until hardware state machine is idle */
-+static inline void wait_for_idle(void)
-+{
-+ while ((__raw_readl(&mdio_regs->control) & CONTROL_IDLE) == 0)
-+ ;
-+}
-+
-+static int cpsw_mdio_read(struct mii_bus *bus, int phy_id, int phy_reg)
-+{
-+ u32 tmp;
-+
-+ if (phy_reg & ~PHY_REG_MASK || phy_id & ~PHY_ID_MASK)
-+ return -EINVAL;
-+
-+ wait_for_user_access();
-+ tmp = (USERACCESS_GO | USERACCESS_READ | (phy_reg << 21) |
-+ (phy_id << 16));
-+ __raw_writel(tmp, &mdio_regs->user[0].access);
-+ tmp = wait_for_user_access();
-+
-+ return (tmp & USERACCESS_ACK) ? (tmp & USERACCESS_DATA) : -1;
-+}
-+
-+static int cpsw_mdio_write(struct mii_bus *bus, int phy_id, int phy_reg, u16 value)
-+{
-+ u32 tmp;
-+
-+ if (phy_reg & ~PHY_REG_MASK || phy_id & ~PHY_ID_MASK)
-+ return -EINVAL;
-+
-+ wait_for_user_access();
-+ tmp = (USERACCESS_GO | USERACCESS_WRITE | (phy_reg << 21) |
-+ (phy_id << 16) | (value & USERACCESS_DATA));
-+ __raw_writel(tmp, &mdio_regs->user[0].access);
-+ wait_for_user_access();
-+
-+ return 0;
-+}
-+
-+static inline void soft_reset(void *reg)
-+{
-+ __raw_writel(1, reg);
-+ while (__raw_readl(reg) & 1)
-+ ;
-+}
-+
-+#define mac_hi(mac) (((mac)[0] << 0) | ((mac)[1] << 8) | \
-+ ((mac)[2] << 16) | ((mac)[3] << 24))
-+#define mac_lo(mac) (((mac)[4] << 0) | ((mac)[5] << 8))
-+
-+static void cpsw_set_slave_mac(struct cpsw_slave *slave,
-+ struct cpsw_priv *priv,
-+ unsigned char *mac)
-+{
-+ dev_dbg(priv->dev, "* %s\n", __func__);
-+
-+ __raw_writel(mac_hi(mac), &slave->regs->sa_hi);
-+ __raw_writel(mac_lo(mac), &slave->regs->sa_lo);
-+}
-+
-+static int cpsw_get_hwaddr(struct eth_device *edev, unsigned char *mac)
-+{
-+ struct cpsw_priv *priv = edev->priv;
-+
-+ dev_dbg(priv->dev, "* %s\n", __func__);
-+
-+ return -1;
-+}
-+
-+static int cpsw_set_hwaddr(struct eth_device *edev, unsigned char *mac)
-+{
-+ struct cpsw_priv *priv = edev->priv;
-+
-+ dev_dbg(priv->dev, "* %s\n", __func__);
-+
-+ memcpy(&priv->mac_addr, mac, sizeof(priv->mac_addr));
-+
-+ for_each_slave(priv, cpsw_set_slave_mac, priv, mac);
-+
-+ return 0;
-+}
-+
-+static void cpsw_slave_update_link(struct cpsw_slave *slave,
-+ struct cpsw_priv *priv, int *link)
-+{
-+ struct phy_device *phydev = priv->edev.phydev;
-+ u32 mac_control = 0;
-+
-+ dev_dbg(priv->dev, "* %s\n", __func__);
-+
-+ if (!phydev)
-+ return;
-+
-+ if (slave->slave_num)
-+ return;
-+
-+ if (phydev->link) {
-+ *link = 1;
-+ mac_control = priv->data.mac_control;
-+ if (phydev->speed == SPEED_10)
-+ mac_control |= BIT(18); /* In Band mode */
-+ else if (phydev->speed == SPEED_100)
-+ mac_control |= BIT(15);
-+ else if (phydev->speed == SPEED_1000)
-+ mac_control &= ~BIT(7); /* TODO: Do not enable
-+ * gig support now */
-+ if (phydev->duplex == DUPLEX_FULL)
-+ mac_control |= BIT(0); /* FULLDUPLEXEN */
-+ }
-+
-+ if (mac_control == slave->mac_control)
-+ return;
-+
-+ if (mac_control) {
-+ dev_dbg(priv->dev, "link up on port %d, speed %d, %s duplex\n",
-+ slave->slave_num, phydev->speed,
-+ (phydev->duplex == DUPLEX_FULL) ? "full" : "half");
-+ } else {
-+ dev_dbg(priv->dev, "link down on port %d\n", slave->slave_num);
-+ }
-+
-+ __raw_writel(mac_control, &slave->sliver->mac_control);
-+ slave->mac_control = mac_control;
-+}
-+
-+static int cpsw_update_link(struct cpsw_priv *priv)
-+{
-+ int link = 0;
-+
-+ dev_dbg(priv->dev, "* %s\n", __func__);
-+
-+ for_each_slave(priv, cpsw_slave_update_link, priv, &link);
-+ return link;
-+}
-+
-+static void cpsw_adjust_link(struct eth_device *edev) {
-+ struct cpsw_priv *priv = edev->priv;
-+
-+ dev_dbg(priv->dev, "* %s\n", __func__);
-+
-+ cpsw_update_link(priv);
-+}
-+
-+static inline u32 cpsw_get_slave_port(struct cpsw_priv *priv, u32 slave_num)
-+{
-+ if (priv->host_port == 0)
-+ return slave_num + 1;
-+ else
-+ return slave_num;
-+}
-+
-+static void cpsw_slave_init(struct cpsw_slave *slave, struct cpsw_priv *priv)
-+{
-+ u32 slave_port;
-+
-+ dev_dbg(priv->dev, "* %s\n", __func__);
-+
-+ soft_reset(&slave->sliver->soft_reset);
-+
-+ /* setup priority mapping */
-+ __raw_writel(0x76543210, &slave->sliver->rx_pri_map);
-+ __raw_writel(0x33221100, &slave->regs->tx_pri_map);
-+
-+ /* setup max packet size, and mac address */
-+ __raw_writel(PKT_MAX, &slave->sliver->rx_maxlen);
-+ //cpsw_set_slave_mac(slave, priv);
-+
-+ slave->mac_control = 0; /* no link yet */
-+
-+ /* enable forwarding */
-+ slave_port = cpsw_get_slave_port(priv, slave->slave_num);
-+ cpsw_ale_port_state(priv, slave_port, ALE_PORT_STATE_FORWARD);
-+
-+ /* add broadcast address */
-+ cpsw_ale_add_mcast(priv, ethbdaddr, 1 << slave_port);
-+}
-+
-+static struct cpdma_desc *cpdma_desc_alloc(struct cpsw_priv *priv)
-+{
-+ struct cpdma_desc *desc = priv->desc_free;
-+
-+ if (desc)
-+ priv->desc_free = desc_read_ptr(desc, hw_next);
-+ return desc;
-+}
-+
-+static void cpdma_desc_free(struct cpsw_priv *priv, struct cpdma_desc *desc)
-+{
-+ if (desc) {
-+ desc_write(desc, hw_next, priv->desc_free);
-+ priv->desc_free = desc;
-+ }
-+}
-+
-+static int cpdma_submit(struct cpsw_priv *priv, struct cpdma_chan *chan,
-+ volatile void *buffer, int len)
-+{
-+ struct cpdma_desc *desc, *prev;
-+ u32 mode;
-+
-+ //dev_dbg(priv->dev, "* %s\n", __func__);
-+
-+ desc = cpdma_desc_alloc(priv);
-+ if (!desc)
-+ return -ENOMEM;
-+
-+ if (len < PKT_MIN)
-+ len = PKT_MIN;
-+
-+ mode = CPDMA_DESC_OWNER | CPDMA_DESC_SOP | CPDMA_DESC_EOP;
-+
-+ desc_write(desc, hw_next, 0);
-+ desc_write(desc, hw_buffer, buffer);
-+ desc_write(desc, hw_len, len);
-+ desc_write(desc, hw_mode, mode | len);
-+ desc_write(desc, sw_buffer, buffer);
-+ desc_write(desc, sw_len, len);
-+
-+ dev_dbg(priv->dev, "%s: submit addr=0x%p len=%i\n", __func__, buffer, len);
-+
-+ if (!chan->head) {
-+ /* simple case - first packet enqueued */
-+ chan->head = desc;
-+ chan->tail = desc;
-+ chan_write(chan, hdp, desc);
-+ goto done;
-+ }
-+
-+ /* not the first packet - enqueue at the tail */
-+ prev = chan->tail;
-+ desc_write(prev, hw_next, desc);
-+ chan->tail = desc;
-+
-+ /* next check if EOQ has been triggered already */
-+ if (desc_read(prev, hw_mode) & CPDMA_DESC_EOQ) {
-+ chan_write(chan, hdp, desc);
-+ }
-+
-+done:
-+ if (chan->rxfree)
-+ chan_write(chan, rxfree, 1);
-+ return 0;
-+}
-+
-+static int cpdma_process(struct cpsw_priv *priv, struct cpdma_chan *chan,
-+ void **buffer, int *len)
-+{
-+ struct cpdma_desc *desc = chan->head;
-+ u32 status;
-+
-+ if (!desc)
-+ return -ENOENT;
-+
-+ status = desc_read(desc, hw_mode);
-+
-+ if (len)
-+ *len = status & 0x7ff;
-+
-+ if (buffer)
-+ *buffer = desc_read_ptr(desc, sw_buffer);
-+
-+ if (status & CPDMA_DESC_OWNER)
-+ return -EBUSY;
-+
-+ chan->head = desc_read_ptr(desc, hw_next);
-+ chan_write(chan, cp, desc);
-+
-+ cpdma_desc_free(priv, desc);
-+ return 0;
-+}
-+
-+static int cpsw_init(struct eth_device *edev)
-+{
-+ struct cpsw_priv *priv = edev->priv;
-+
-+ dev_dbg(priv->dev, "* %s\n", __func__);
-+
-+ return 0;
-+}
-+
-+static int cpsw_open(struct eth_device *edev)
-+{
-+ struct cpsw_priv *priv = edev->priv;
-+ struct cpsw_slave_data *slave_data = priv->data.slave_data;
-+ int i, ret;
-+
-+ dev_dbg(priv->dev, "* %s\n", __func__);
-+
-+ ret = phy_device_connect(edev, &priv->miibus, slave_data[0].phy_id,
-+ cpsw_adjust_link, 0, slave_data[0].phy_if);
-+ if (ret)
-+ return ret;
-+
-+ priv->data.control(1);
-+
-+ /* soft reset the controller and initialize priv */
-+ soft_reset(&priv->regs->soft_reset);
-+
-+ /* initialize and reset the address lookup engine */
-+ cpsw_ale_enable(priv, 1);
-+ cpsw_ale_clear(priv, 1);
-+ cpsw_ale_bypass(priv, 0);
-+ cpsw_ale_vlan_aware(priv, 0); /* vlan unaware mode */
-+
-+ /* setup host port priority mapping */
-+ __raw_writel(0x76543210, &priv->host_port_regs->cpdma_tx_pri_map);
-+ __raw_writel(0, &priv->host_port_regs->cpdma_rx_chan_map);
-+
-+ /* disable priority elevation and enable statistics on all ports */
-+ __raw_writel(0, &priv->regs->ptype);
-+
-+ /* enable statistics collection only on the host port */
-+ __raw_writel(BIT(priv->host_port), &priv->regs->stat_port_en);
-+
-+ cpsw_ale_port_state(priv, priv->host_port, ALE_PORT_STATE_FORWARD);
-+
-+ cpsw_ale_add_ucast(priv, priv->mac_addr, priv->host_port,
-+ ALE_SECURE);
-+ cpsw_ale_add_mcast(priv, ethbdaddr, 1 << priv->host_port);
-+
-+ for_each_slave(priv, cpsw_slave_init, priv);
-+
-+ cpsw_update_link(priv);
-+
-+ /* init descriptor pool */
-+ for (i = 0; i < NUM_DESCS; i++) {
-+ desc_write(&priv->descs[i], hw_next,
-+ (i == (NUM_DESCS - 1)) ? 0 : &priv->descs[i+1]);
-+ }
-+ priv->desc_free = &priv->descs[0];
-+
-+ /* initialize channels */
-+ if (priv->data.version == CPSW_CTRL_VERSION_2) {
-+ memset(&priv->rx_chan, 0, sizeof(struct cpdma_chan));
-+ priv->rx_chan.hdp = priv->dma_regs + CPDMA_RXHDP_VER2;
-+ priv->rx_chan.cp = priv->dma_regs + CPDMA_RXCP_VER2;
-+ priv->rx_chan.rxfree = priv->dma_regs + CPDMA_RXFREE;
-+
-+ memset(&priv->tx_chan, 0, sizeof(struct cpdma_chan));
-+ priv->tx_chan.hdp = priv->dma_regs + CPDMA_TXHDP_VER2;
-+ priv->tx_chan.cp = priv->dma_regs + CPDMA_TXCP_VER2;
-+ } else {
-+ memset(&priv->rx_chan, 0, sizeof(struct cpdma_chan));
-+ priv->rx_chan.hdp = priv->dma_regs + CPDMA_RXHDP_VER1;
-+ priv->rx_chan.cp = priv->dma_regs + CPDMA_RXCP_VER1;
-+ priv->rx_chan.rxfree = priv->dma_regs + CPDMA_RXFREE;
-+
-+ memset(&priv->tx_chan, 0, sizeof(struct cpdma_chan));
-+ priv->tx_chan.hdp = priv->dma_regs + CPDMA_TXHDP_VER1;
-+ priv->tx_chan.cp = priv->dma_regs + CPDMA_TXCP_VER1;
-+ }
-+
-+ /* clear dma state */
-+ soft_reset(priv->dma_regs + CPDMA_SOFTRESET);
-+
-+ if (priv->data.version == CPSW_CTRL_VERSION_2) {
-+ for (i = 0; i < priv->data.channels; i++) {
-+ __raw_writel(0, priv->dma_regs + CPDMA_RXHDP_VER2 + 4
-+ * i);
-+ __raw_writel(0, priv->dma_regs + CPDMA_RXFREE + 4
-+ * i);
-+ __raw_writel(0, priv->dma_regs + CPDMA_RXCP_VER2 + 4
-+ * i);
-+ __raw_writel(0, priv->dma_regs + CPDMA_TXHDP_VER2 + 4
-+ * i);
-+ __raw_writel(0, priv->dma_regs + CPDMA_TXCP_VER2 + 4
-+ * i);
-+ }
-+ } else {
-+ for (i = 0; i < priv->data.channels; i++) {
-+ __raw_writel(0, priv->dma_regs + CPDMA_RXHDP_VER1 + 4
-+ * i);
-+ __raw_writel(0, priv->dma_regs + CPDMA_RXFREE + 4
-+ * i);
-+ __raw_writel(0, priv->dma_regs + CPDMA_RXCP_VER1 + 4
-+ * i);
-+ __raw_writel(0, priv->dma_regs + CPDMA_TXHDP_VER1 + 4
-+ * i);
-+ __raw_writel(0, priv->dma_regs + CPDMA_TXCP_VER1 + 4
-+ * i);
-+
-+ }
-+ }
-+
-+ __raw_writel(1, priv->dma_regs + CPDMA_TXCONTROL);
-+ __raw_writel(1, priv->dma_regs + CPDMA_RXCONTROL);
-+
-+ /* submit rx descs */
-+ for (i = 0; i < PKTBUFSRX; i++) {
-+ ret = cpdma_submit(priv, &priv->rx_chan, NetRxPackets[i],
-+ PKTSIZE);
-+ if (ret < 0) {
-+ printf("error %d submitting rx desc\n", ret);
-+ break;
-+ }
-+ }
-+
-+ return 0;
-+}
-+
-+static void cpsw_halt(struct eth_device *dev)
-+{
-+ struct cpsw_priv *priv = dev->priv;
-+ priv->data.control(0);
-+}
-+
-+static int cpsw_send(struct eth_device *dev, void *packet, int length)
-+{
-+ // FIXME *packet was volatile before,
-+ struct cpsw_priv *priv = dev->priv;
-+ void *buffer;
-+ int len;
-+
-+ dev_dbg(priv->dev, "* %s\n", __func__);
-+
-+ if (!cpsw_update_link(priv))
-+ return -EIO;
-+
-+ /* first reap completed packets */
-+ while (cpdma_process(priv, &priv->tx_chan, &buffer, &len) >= 0)
-+ ;
-+
-+ dev_dbg(priv->dev, "%s: %i bytes @ 0x%p\n", __func__, length, packet);
-+ dma_flush_range((ulong) packet, (ulong)packet + length);
-+ return cpdma_submit(priv, &priv->tx_chan, packet, length);
-+}
-+
-+static int cpsw_recv(struct eth_device *dev)
-+{
-+ struct cpsw_priv *priv = dev->priv;
-+ void *buffer;
-+ int len;
-+
-+ //dev_dbg(priv->dev, "* %s\n", __func__);
-+
-+ while (cpdma_process(priv, &priv->rx_chan, &buffer, &len) >= 0) {
-+ dev_dbg(priv->dev, "%s: %i bytes @ 0x%p\n", __func__, len, buffer);
-+ dma_inv_range((ulong)buffer, (ulong)buffer + len);
-+ net_receive(buffer, len);
-+ cpdma_submit(priv, &priv->rx_chan, buffer, PKTSIZE);
-+ }
-+
-+ return 0;
-+}
-+
-+static void cpsw_slave_setup(struct cpsw_slave *slave, int slave_num,
-+ struct cpsw_priv *priv)
-+{
-+ void *regs = priv->regs;
-+ struct cpsw_slave_data *data = priv->data.slave_data + slave_num;
-+
-+ dev_dbg(priv->dev, "* %s\n", __func__);
-+
-+ slave->slave_num = slave_num;
-+ slave->data = data;
-+ slave->regs = regs + data->slave_reg_ofs;
-+ slave->sliver = regs + data->sliver_reg_ofs;
-+}
-+
-+int cpsw_probe(struct device_d *dev)
-+{
-+ struct cpsw_platform_data *data = (struct cpsw_platform_data *)dev->platform_data;
-+ struct cpsw_priv *priv;
-+ void *regs = (void *)data->cpsw_base;
-+ struct eth_device *edev;
-+ uint64_t start;
-+ uint32_t phy_mask;
-+
-+ dev_dbg(dev, "* %s\n", __func__);
-+
-+ priv = xzalloc(sizeof(*priv));
-+ priv->dev = dev;
-+ priv->data = *data;
-+ edev = &priv->edev;
-+
-+ priv->slaves = xzalloc(sizeof(struct cpsw_slave) * data->slaves);
-+
-+ priv->descs = (void *)CPDMA_RAM_ADDR;
-+ priv->host_port = data->host_port_num;
-+ priv->regs = regs;
-+ priv->host_port_regs = regs + data->host_port_reg_ofs;
-+ priv->dma_regs = regs + data->cpdma_reg_ofs;
-+ priv->ale_regs = regs + data->ale_reg_ofs;
-+
-+ for_each_slave(priv, cpsw_slave_setup, idx, priv);
-+
-+ edev->priv = priv;
-+ edev->init = cpsw_init;
-+ edev->open = cpsw_open;
-+ edev->halt = cpsw_halt;
-+ edev->send = cpsw_send;
-+ edev->recv = cpsw_recv;
-+ edev->get_ethaddr = cpsw_get_hwaddr;
-+ edev->set_ethaddr = cpsw_set_hwaddr;
-+ edev->parent = dev;
-+
-+ mdio_regs = (struct cpsw_mdio_regs *)data->mdio_base;
-+ priv->miibus.read = cpsw_mdio_read;
-+ priv->miibus.write = cpsw_mdio_write;
-+ priv->miibus.priv = priv;
-+ priv->miibus.parent = dev;
-+
-+ /* set enable and clock divider */
-+ __raw_writel(data->mdio_div | CONTROL_ENABLE, &mdio_regs->control);
-+
-+ /*
-+ * wait for scan logic to settle:
-+ * the scan time consists of (a) a large fixed component, and (b) a
-+ * small component that varies with the mii bus frequency. These
-+ * were estimated using measurements at 1.1 and 2.2 MHz on tnetv107x
-+ * silicon. Since the effect of (b) was found to be largely
-+ * negligible, we keep things simple here.
-+ */
-+ udelay(1000);
-+
-+ start = get_time_ns();
-+ while (1) {
-+ phy_mask = readl(&mdio_regs->alive);
-+ if (phy_mask) {
-+ dev_info(dev, "detected phy mask 0x%x\n", phy_mask);
-+ phy_mask = ~phy_mask;
-+ break;
-+ }
-+ if (is_timeout(start, 256 * MSECOND)) {
-+ dev_err(dev, "no live phy, scanning all\n");
-+ phy_mask = 0;
-+ break;
-+ }
-+ }
-+
-+ priv->miibus.phy_mask = phy_mask;
-+
-+ mdiobus_register(&priv->miibus);
-+
-+ eth_register(edev);
-+
-+ return 0;
-+}
-+
-+static struct driver_d cpsw_driver = {
-+ .name = "cpsw",
-+ .probe = cpsw_probe,
-+// .remove = cpsw_remove,
-+};
-+
-+static int cpsw_register(void)
-+{
-+ platform_driver_register(&cpsw_driver);
-+ return 0;
-+}
-+
-+device_initcall(cpsw_register);