summaryrefslogtreecommitdiff
path: root/nuttx/arch/arm/src/c5471
diff options
context:
space:
mode:
authorpatacongo <patacongo@42af7a65-404d-4744-a932-0658087f49c3>2007-12-04 17:11:55 +0000
committerpatacongo <patacongo@42af7a65-404d-4744-a932-0658087f49c3>2007-12-04 17:11:55 +0000
commitd9c3513a33419a4cb6de9b3e39084989709584c1 (patch)
treeae164e1b32d440051b1581af871f523d6f34d702 /nuttx/arch/arm/src/c5471
parent06e5392650035245f70c576a39a46dc7306add2c (diff)
downloadpx4-nuttx-d9c3513a33419a4cb6de9b3e39084989709584c1.tar.gz
px4-nuttx-d9c3513a33419a4cb6de9b3e39084989709584c1.tar.bz2
px4-nuttx-d9c3513a33419a4cb6de9b3e39084989709584c1.zip
Add c5471 Ethernet driver
git-svn-id: svn://svn.code.sf.net/p/nuttx/code/trunk@423 42af7a65-404d-4744-a932-0658087f49c3
Diffstat (limited to 'nuttx/arch/arm/src/c5471')
-rw-r--r--nuttx/arch/arm/src/c5471/Make.defs5
-rw-r--r--nuttx/arch/arm/src/c5471/c5471_ethernet.c1976
-rw-r--r--nuttx/arch/arm/src/c5471/chip.h92
3 files changed, 2051 insertions, 22 deletions
diff --git a/nuttx/arch/arm/src/c5471/Make.defs b/nuttx/arch/arm/src/c5471/Make.defs
index 540edc053..636197f4e 100644
--- a/nuttx/arch/arm/src/c5471/Make.defs
+++ b/nuttx/arch/arm/src/c5471/Make.defs
@@ -37,7 +37,7 @@ HEAD_ASRC = up_nommuhead.S
CMN_ASRCS = up_saveusercontext.S up_fullcontextrestore.S
CMN_CSRCS = up_allocateheap.c up_assert.c up_blocktask.c up_copystate.c \
- up_createstack.c up_dataabort.c up_mdelay.c up_udealy.c up_doirq.c \
+ up_createstack.c up_dataabort.c up_mdelay.c up_udelay.c up_doirq.c \
up_exit.c up_idle.c up_initialize.c up_initialstate.c \
up_interruptcontext.c up_prefetchabort.c up_releasepending.c \
up_releasestack.c up_reprioritizertr.c up_schedulesigaction.c \
@@ -45,4 +45,5 @@ CMN_CSRCS = up_allocateheap.c up_assert.c up_blocktask.c up_copystate.c \
up_undefinedinsn.c up_usestack.c
CHIP_ASRCS = c5471_lowputc.S c5471_vectors.S
-CHIP_CSRCS = c5471_irq.c c5471_serial.c c5471_timerisr.c c5471_watchdog.c
+CHIP_CSRCS = c5471_irq.c c5471_serial.c c5471_timerisr.c c5471_watchdog.c \
+ c5471_ethernet.c
diff --git a/nuttx/arch/arm/src/c5471/c5471_ethernet.c b/nuttx/arch/arm/src/c5471/c5471_ethernet.c
new file mode 100644
index 000000000..133c2fbb1
--- /dev/null
+++ b/nuttx/arch/arm/src/c5471/c5471_ethernet.c
@@ -0,0 +1,1976 @@
+/****************************************************************************
+ * arch/arm/src/c5471/c5471_ethernet.c
+ *
+ * Copyright (C) 2007 Gregory Nutt. All rights reserved.
+ * Author: Gregory Nutt <spudmonkey@racsa.co.cr>
+ *
+ * Based one a C5471 Linux driver and released under this BSD license with
+ * special permisson from the copyright holder of the Linux driver:
+ * Todd Fischer, Cadenux, LLC. Other references: "TMS320VC547x CPU and
+ * Peripherals Reference Guide," TI document spru038.pdf.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in
+ * the documentation and/or other materials provided with the
+ * distribution.
+ * 3. Neither the name NuttX nor the names of its contributors may be
+ * used to endorse or promote products derived from this software
+ * without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+ * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+ * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
+ * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
+ * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
+ * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
+ * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
+ * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
+ * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
+ * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
+ * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
+ * POSSIBILITY OF SUCH DAMAGE.
+ *
+ ****************************************************************************/
+
+/****************************************************************************
+ * Included Files
+ ****************************************************************************/
+
+#include <nuttx/config.h>
+#if defined(CONFIG_NET)
+
+#include <sys/types.h>
+
+#include <time.h>
+#include <string.h>
+#include <debug.h>
+#include <wdog.h>
+#include <errno.h>
+
+#include <nuttx/irq.h>
+#include <nuttx/arch.h>
+
+#include <net/uip/uip.h>
+#include <net/uip/uip-arp.h>
+#include <net/uip/uip-arch.h>
+
+#include "chip.h"
+#include "up_internal.h"
+
+/****************************************************************************
+ * Definitions
+ ****************************************************************************/
+
+/* Configuration ************************************************************/
+/* CONFIG_C5471_NET_NINTERFACES determines the number of physical interfaces
+ * that will be supported.
+ */
+
+#ifndef CONFIG_C5471_NET_NINTERFACES
+# define CONFIG_C5471_NET_NINTERFACES 1
+#endif
+
+/* Timing values ************************************************************/
+/* TX poll deley = 1 seconds. CLK_TCK=number of clock ticks per second */
+
+#define C5471_WDDELAY (1*CLK_TCK)
+#define C5471_POLLHSEC (1*2)
+
+/* TX timeout = 1 minute */
+
+#define C5471_TXTIMEOUT (60*CLK_TCK)
+
+/* Ethernet GPIO bit settings ***********************************************/
+
+#define GPIO_CIO_MDIO 0x00004000
+#define GPIO_IO_MDCLK 0x00008000
+
+/* Ethernet interface bit settings ******************************************/
+
+/* TX descriptor, word #0 */
+
+#define EIM_TXDESC_OWN_HOST 0x80000000 /* Bit 15: Ownership bit */
+#define EIM_TXDESC_OWN_ENET 0x00000000
+#define EIM_TXDESC_WRAP_NEXT 0x40000000 /* Bit 14: Descriptor chain wrap */
+#define EIM_TXDESC_WRAP_FIRST 0x00000000
+#define EIM_TXDESC_FIF 0x20000000 /* Bit 13: First in frame */
+#define EIM_TXDESC_LIF 0x10000000 /* Bit 12: Last in frame */
+ /* Bits 8-11: Retry count status */
+#define EIM_TXDESC_INTRE 0x00800000 /* Bit 7: TX_IRQ int enable */
+#define EIM_TXDESC_STATUSMASK 0x007f0000 /* Bits 0-6: Status */
+#define EIM_TXDESC_RETRYERROR 0x00400000 /* Exceed retry error */
+#define EIM_TXDESC_HEARTBEAT 0x00200000 /* Heartbeat (SQE) */
+#define EIM_TXDESC_LCOLLISON 0x00100000 /* Late collision error */
+#define EIM_TXDESC_COLLISION 0x00080000 /* Collision */
+#define EIM_TXDESC_CRCERROR 0x00040000 /* CRC error */
+#define EIM_TXDESC_UNDERRUN 0x00020000 /* Underrun error */
+#define EIM_TXDESC_LOC 0x00010000 /* Loss of carrier */
+
+/* Packet bytes value used for both TX and RX descriptors */
+
+#define EIM_PACKET_BYTES 0x00000040
+
+/* Count of descriptors */
+
+#define NUM_DESC_TX 32
+#define NUM_DESC_RX 64
+
+/* TX descriptor, word #1 */
+ /* Bit 15: reserved */
+#define EIM_TXDESC_PADCRC 0x00004000 /* Bit 14: Enable padding small frames */
+ /* Bits 11-13: reserved */
+#define EIM_TXDESC_BYTEMASK 0x000007ff /* Bits 0-10: Descriptor byte count */
+
+/* RX descriptor, word #0 */
+
+#define EIM_RXDESC_OWN_HOST 0x80000000 /* Bit 15: Ownership bit */
+#define EIM_RXDESC_OWN_ENET 0x00000000
+#define EIM_RXDESC_WRAP_NEXT 0x40000000 /* Bit 14: Descriptor chain wrap */
+#define EIM_RXDESC_WRAP_FIRST 0x00000000
+#define EIM_RXDESC_FIF 0x20000000 /* Bit 13: First in frame */
+#define EIM_RXDESC_LIF 0x10000000 /* Bit 12: Last in frame */
+ /* Bits 8-11: reserved */
+#define EIM_RXDESC_INTRE 0x00800000 /* Bit 7: RX_IRQ int enable */
+#define EIM_RXDESC_STATUSMASK 0x007f0000 /* Bits 0-6: Status */
+#define EIM_RXDESC_MISS 0x00400000 /* Miss */
+#define EIM_RXDESC_VLAN 0x00200000 /* VLAN */
+#define EIM_RXDESC_LFRAME 0x00100000 /* Long frame error */
+#define EIM_RXDESC_SFRAME 0x00080000 /* Short frame error */
+#define EIM_RXDESC_CRCERROR 0x00040000 /* CRC error */
+#define EIM_RXDESC_OVERRUN 0x00020000 /* Overrun error */
+#define EIM_RXDESC_ALIGN 0x00010000 /* Non-octect align error */
+
+#define EIM_RXDESC_PADCRC 0x00004000 /* Enable padding for small frames */
+
+/* RX descriptor, word #1 */
+ /* Bits 11-15: reserved */
+#define EIM_RXDESC_BYTEMASK 0x000007ff /* Bits 0-10: Descriptor byte count */
+
+/* EIM_CPU_FILTER bit settings */
+ /* Bits 5-31: reserved */
+#define EIM_FILTER_MACLA 0x00000010 /* Bit 4: Enable logical address+multicast filtering */
+#define EIM_FILTER_LOGICAL 0x00000008 /* Bit 3: Enable ENET logical filtering */
+#define EIM_FILTER_MULTICAST 0x00000004 /* Bit 2: Enable multicast filtering */
+#define EIM_FILTER_BROADCAST 0x00000002 /* Bit 1: Enable broadcast matching */
+#define EIM_FILTER_UNICAST 0x00000001 /* Bit 0: Enable dest CPU address matching */
+
+/* EIM_CTRL bit settings */
+ /* Bits 16-31: Reserved */
+#define EIM_CTRL_ESM_EN 0x00008000 /* Bit 15: Ethernet state machine enable */
+ /* Bits 9-14: reserved */
+#define EIM_CTRL_ENET0_EN 0x00000100 /* Bit 8: Enable routing of RX packets CPU->ENET0 */
+ /* Bit 7: reserved */
+#define EIM_CTRL_ENET0_FLW 0x00000040 /* Bit 6: Enable ENET0 flow control RX threshold */
+#define EIM_CTRL_RXENET0_EN 0x00000020 /* Bit 5: Enable processing of ENET0 RX queue */
+#define EIM_CTRL_TXENET0_EN 0x00000010 /* Bit 4: Enable processing of ENET0 TX queue */
+ /* Bits 2-3: reserved */
+#define EIM_CTRL_RXCPU_EN 0x00000002 /* Bit 1: Enable processing of CPU RX queue */
+#define EIM_CTRL_TXCPU_EN 0x00000001 /* Bit 0: Enable processing of CPU TX queue */
+
+/* EIM_STATUS bit settings */
+ /* Bits 10-31: reserved */
+#define EIM_STATUS_CPU_TXLIF 0x00000200 /* Bit 9: Last descriptor of TX packet filled */
+#define EIM_STATUS_CPU_RXLIF 0x00000100 /* Bit 8: Last descriptor of RX queue processed */
+#define EIM_STATUS_CPU_TX 0x00000080 /* Bit 7: Descriptor filled in TX queue */
+#define EIM_STATUS_CPU_RX 0x00000040 /* Bit 6: Descriptor filled in RX queue */
+ /* Bits 3-5: reserved */
+#define EIM_STATUS_ENET0_ERR 0x00000004 /* Bit 2: ENET0 error interrupt */
+#define EIM_STATUS_ENET0_TX 0x00000002 /* Bit 1: ENET0 TX interrupt */
+#define EIM_STATUS_ENET0_RX 0x00000001 /* Bit 0" ENET0 RX interrupt */
+
+/* EIM_INTEN bit settings */
+
+#define EIM_INTEN_CPU_TXLIF 0x00000200 /* Bit 9: Last descriptor of TX packet filled */
+#define EIM_INTEN_CPU_RXLIF 0x00000100 /* Bit 8: Last descriptor of RX queue processed */
+#define EIM_INTEN_CPU_TX 0x00000080 /* Bit 7: Descriptor filled in TX queue */
+#define EIM_INTEN_CPU_RX 0x00000040 /* Bit 6: Descriptor filled in RX queue */
+ /* Bits 3-5: reserved */
+#define EIM_INTEN_ENET0_ERR 0x00000004 /* Bit 2: ENET0 error interrupt */
+#define EIM_INTEN_ENET0_TX 0x00000002 /* Bit 1: ENET0 TX interrupt */
+#define EIM_INTEN_ENET0_RX 0x00000001 /* Bit 0" ENET0 RX interrupt */
+
+/* ENET0_ADRMODE_EN bit settings */
+
+#define ENET_ADR_PROMISCUOUS 0x00000008 /* Bit 3: Enable snoop address comparison */
+#define ENET_ADR_BROADCAST 0x00000004 /* Bit 2: Enable broadcast address comparison */
+#define ENET_ADDR_LCOMPARE 0x00000001 /* Bit 1: Enable logical address comparison */
+#define ENET_ADDR_PCOMPARE 0x00000001 /* Bit 0: Enable physical address comparison */
+
+/* ENET0_MODE bit settings */
+ /* Bits 16-31: reserved */
+#define ENET_MODO_FIFO_EN 0x00008000 /* Bit 15: Fifo enable */
+ /* Bits 8-14: reserved */
+#define ENET_MODE_RJCT_SFE 0x00000080 /* Bit 7: Reject short frames durig receive */
+#define ENET_MODE_DPNET 0x00000040 /* Bit 6: Demand priority networkd vs CSMA/CD */
+#define ENET_MODE_MWIDTH 0x00000020 /* Bit 5: Select nibble mode MII port */
+#define ENET_MODE_WRAP 0x00000010 /* Bit 4: Internal MAC loopback */
+#define ENET_MODE_FDWRAP 0x00000008 /* Bit 3: Full duplex wrap */
+#define ENET_MODE_FULLDUPLEX 0x00000004 /* Bit 2: 1:Full duplex */
+#define ENET_MODE_HALFDUPLEX 0x00000000 /* 0:Half duplex */
+ /* Bit 1: reserved */
+#define ENET_MODE_ENABLE 0x00000001 /* Bit 0: Port enable */
+
+/* PHY registers */
+
+#define MD_PHY_CONTROL_REG 0x00
+#define MD_PHY_MSB_REG 0x02
+#define MD_PHY_LSB_REG 0x03
+#define MD_PHY_CTRL_STAT_REG 0x17
+
+/* Lucent LU3X31T-T64 transeiver ID */
+
+#define LU3X31_T64_PHYID 0x00437421
+
+/* PHY control register bit settings */
+
+#define MODE_AUTONEG 0x1000
+#define MODE_10MBIT_HALFDUP 0x0000
+#define MODE_10MBIT_FULLDUP 0x0100
+#define MODE_100MBIT_FULLDUP 0x2100
+#define MODE_100MBIT_HALFDUP 0x2000
+
+/* Inserts an ARM "nop" instruction */
+
+#define nop() asm(" nop");
+
+/* This is a helper pointer for accessing the contents of the Ethernet header */
+
+#define BUF ((struct uip_eth_hdr *)c5471->c_dev.d_buf)
+
+/****************************************************************************
+ * Private Types
+ ****************************************************************************/
+
+/* The c5471_driver_s encapsulates all state information for a single c5471
+ * hardware interface
+ */
+
+struct c5471_driver_s
+{
+ boolean c_bifup; /* TRUE:ifup FALSE:ifdown */
+ WDOG_ID c_txpoll; /* TX poll timer */
+ WDOG_ID c_txtimeout; /* TX timeout timer */
+
+ /* Note: According to the C547x documentation: "The software has to maintain
+ * two pointers to the current RX-CPU and TX-CPU descriptors. At init time,
+ * they have to be set to the first descriptors of each queue, and they have
+ * to be incremented each time a descriptor ownership is give to the SWITCH".
+ */
+
+ volatile uint32 c_txcpudesc;
+ volatile uint32 c_rxcpudesc;
+
+ /* Last TX descriptor saved for error handling */
+
+ uint32 c_lastdescstart;
+ uint32 c_lastdescend;
+
+ /* Shadowed registers */
+
+ uint32 c_eimstatus;
+
+#ifdef CONFIG_C5471_NET_STATS
+ /* TX statistics */
+
+ uint32 c_txpackets; /* Number of packets sent */
+ uint32 c_txmiss; /* Miss */
+ uint32 c_txvlan; /* VLAN */
+ uint32 c_txlframe; /* Long frame errors */
+ uint32 c_txsframe; /* Short frame errors */
+ uint32 c_txcrc; /* CRC errors */
+ uint32 c_txoverrun; /* Overrun errors */
+ uint32 c_txalign; /* Non-octect align errors */
+ uint32 c_txtimeouts; /* TX timeouts */
+
+ uint32 c_rxpackets; /* Number of packets received */
+ uint32 c_rxretries; /* Exceed retry errors */
+ uint32 c_rxheartbeat; /* Heartbeat (SQE) */
+ uint32 c_rxlcollision; /* Late collision errors */
+ uint32 c_rxcollision; /* Collision */
+ uint32 c_rxcrc; /* CRC errors */
+ uint32 c_rxunderrun; /* Underrun errors */
+ uint32 c_rxloc; /* Loss of carrier */
+#endif
+
+ /* This holds the information visible to uIP/NuttX */
+
+ struct uip_driver_s c_dev; /* Interface understood by uIP */
+};
+
+/****************************************************************************
+ * Private Data
+ ****************************************************************************/
+
+static struct c5471_driver_s g_c5471[CONFIG_C5471_NET_NINTERFACES];
+
+/****************************************************************************
+ * Private Function Prototypes
+ ****************************************************************************/
+
+/* Transceiver interface */
+
+static void c5471_mdtxbit (int bit_state);
+static int c5471_mdrxbit (void);
+static void c5471_mdwrite (int adr, int reg, int data);
+static int c5471_mdread (int adr, int reg);
+static int c5471_phyinit (void);
+
+/* Support logic */
+
+static void c5471_inctxcpu(void);
+static void c5471_incrxcpu(void);
+
+/* Common TX logic */
+
+static int c5471_transmit(struct c5471_driver_s *c5471);
+static int c5471_uiptxpoll(struct uip_driver_s *dev);
+
+/* Interrupt handling */
+
+static void c5471_receive(struct c5471_driver_s *c5471);
+static void c5471_txdone(struct c5471_driver_s *c5471);
+static int c5471_interrupt(int irq, FAR void *context);
+
+/* Watchdog timer expirations */
+
+static void c5471_polltimer(int argc, uint32 arg, ...);
+static void c5471_txtimeout(int argc, uint32 arg, ...);
+
+/* NuttX callback functions */
+
+static int c5471_ifup(struct uip_driver_s *dev);
+static int c5471_ifdown(struct uip_driver_s *dev);
+static int c5471_txavail(struct uip_driver_s *dev);
+
+/* Initialization functions */
+
+static void c5471_eimreset (struct c5471_driver_s *c5471);
+static void c5471_eimconfig(void);
+static void c5471_reset(struct c5471_driver_s *c5471);
+static void c5471_macassign(struct c5471_driver_s *c5471);
+
+/****************************************************************************
+ * Private Functions
+ ****************************************************************************/
+
+/****************************************************************************
+ * Name: c5471_mdtxbit
+ *
+ * Description
+ * A helper routine used when serially communicating with the c547X's
+ * external ethernet transeiver device. GPIO pins are connected to the
+ * transeiver's MDCLK and MDIO pins and are used to accomplish the serial
+ * comm.
+ *
+ * protocol:
+ * ___________
+ * MDCLK ________/ \_
+ * ________:____
+ * MDIO <________:____>--------
+ * :
+ * ^
+ * Pin state internalized
+ *
+ ****************************************************************************/
+
+static void c5471_mdtxbit (int bit_state)
+{
+ /* Note: any non-zero "bit_state" supplied by the caller means we should clk a "1"
+ * out the MDIO pin.
+ */
+
+ /* config MDIO as output pin. */
+
+ putreg32((getreg32(GPIO_CIO) & ~GPIO_CIO_MDIO), GPIO_CIO);
+ if (bit_state)
+ {
+ /* set MDIO state high. */
+
+ putreg32(getreg32(GPIO_IO) | GPIO_CIO_MDIO), GPIO_IO);
+ }
+ else
+ {
+ /* set MDIO state low. */
+
+ putreg32((getreg32(GPIO_IO) & ~GPIO_CIO_MDIO), GPIO_IO);
+ }
+
+ nop();
+ nop();
+ nop();
+ nop();
+
+ /* MDCLK rising edge */
+
+ putreg32((getreg32(GPIO_IO) | GPIO_IO_MDCLK), GPIO_IO);
+ nop();
+ nop();
+
+ /* release MDIO */
+
+ putreg32((getreg32(GPIO_CIO)|GPIO_CIO_MDIO), GPIO_CIO);
+ nop();
+ nop();
+
+ /* MDCLK falling edge. */
+
+ putreg32((getreg32(GPIO_IO) & ~GPIO_IO_MDCLK), GPIO_IO);
+}
+
+/****************************************************************************
+ * Name: c5471_mdrxbit
+ *
+ * Description
+ * A helper routine used when serially communicating with the c547X's
+ * external ethernet transeiver device. GPIO pins are connected to the
+ * transeiver's MDCLK and MDIO pins and are used to accomplish the serial
+ * comm.
+ *
+ * protocol:
+ * ___________
+ * MDCLK ________/ \_
+ * _______:_____
+ * MDIO _______:_____>--------
+ * :
+ * ^
+ * pin state sample point
+ *
+ ****************************************************************************/
+
+static int c5471_mdrxbit (void)
+{
+ register volatile uint32 bit_state;
+
+ /* config MDIO as input pin. */
+
+ putreg32((getreg32(GPIO_CIO) | GPIO_CIO_MDIO), GPIO_CIO);
+
+ /* Make sure the MDCLK is low */
+
+ putreg32((getreg32(GPIO_IO) & ~GPIO_IO_MDCLK), GPIO_IO);
+ nop();
+ nop();
+ nop();
+ nop();
+
+ /* Sample MDIO */
+
+ bit_state = getreg32(GPIO_IO) & GPIO_CIO_MDIO;
+
+ /* MDCLK rising edge */
+
+ putreg32((getreg32(GPIO_IO)|GPIO_IO_MDCLK), GPIO_IO);
+ nop();
+ nop();
+ nop();
+ nop();
+
+ /* MDCLK falling edge. */
+
+ putreg32((getreg32(GPIO_IO)&~GPIO_IO_MDCLK), GPIO_IO); /* MDCLK falling edge */
+ if (bit_state)
+ {
+ return 1;
+ }
+ else
+ {
+ return 0;
+ }
+}
+
+/****************************************************************************
+ * Name: c5471_mdwrite
+ *
+ * Description
+ * A helper routine used when serially communicating with the c547X's
+ * external ethernet transeiver device. GPIO pins are connected to the
+ * transeiver's MDCLK and MDIO pins and are used to accomplish the serial
+ * comm.
+ *
+ ****************************************************************************/
+
+static void c5471_mdwrite (int adr, int reg, int data)
+{
+ int i;
+
+ /* preamble: 11111111111111111111111111111111 */
+
+ for (i = 0; i < 32; i++)
+ c5471_mdtxbit(1);
+
+ /* start of frame: 01 */
+
+ c5471_mdtxbit(0);
+ c5471_mdtxbit(1);
+
+ /* operation code: 01 - write */
+
+ c5471_mdtxbit(0);
+ c5471_mdtxbit(1);
+
+ /* PHY device address: AAAAA, msb first */
+
+ for (i = 0; i < 5; i++)
+ {
+ c5471_mdtxbit(adr & 0x10);
+ adr = adr << 1;
+ }
+
+ /* MII register address: RRRRR, msb first */
+
+ for (i = 0; i < 5; i++)
+ {
+ c5471_mdtxbit(reg & 0x10);
+ reg = reg << 1;
+ }
+
+ /* turnaround time: ZZ */
+
+ c5471_mdtxbit(1);
+ c5471_mdtxbit(0);
+
+ /* data: DDDDDDDDDDDDDDDD, msb first */
+
+ for (i = 0; i < 16; i++)
+ {
+ c5471_mdtxbit(data & 0x8000);
+ data = data << 1;
+ }
+}
+
+/****************************************************************************
+ * Name: c5471_mdread
+ *
+ * Description
+ * A helper routine used when serially communicating with the c547X's
+ * external ethernet transeiver device. GPIO pins are connected to the
+ * transeiver's MDCLK and MDIO pins and are used to accomplish the serial
+ * comm.
+ *
+ ****************************************************************************/
+
+static int c5471_mdread (int adr, int reg)
+{
+ int i;
+ int data = 0;
+
+ /* preamble: 11111111111111111111111111111111 */
+
+ for (i = 0; i < 32; i++)
+ c5471_mdtxbit(1);
+
+ /* start of frame: 01 */
+
+ c5471_mdtxbit(0);
+ c5471_mdtxbit(1);
+
+ /* operation code: 10 - read */
+
+ c5471_mdtxbit(1);
+ c5471_mdtxbit(0);
+
+ /* PHY device address: AAAAA, msb first */
+
+ for (i = 0; i < 5; i++)
+ {
+ c5471_mdtxbit(adr & 0x10);
+ adr = adr << 1;
+ }
+
+ /* MII register address: RRRRR, msb first */
+
+ for (i = 0; i < 5; i++)
+ {
+ c5471_mdtxbit(reg & 0x10);
+ reg = reg << 1;
+ }
+
+ /* turnaround time: ZZ */
+
+ c5471_mdrxbit();
+ c5471_mdrxbit(); /* PHY should drive a 0 */
+
+ /* data: DDDDDDDDDDDDDDDD, msb first */
+
+ for (i = 0; i < 16; i++)
+ {
+ data = data << 1;
+ data |= c5471_mdrxbit();
+ }
+
+ return data;
+}
+
+/****************************************************************************
+ * Name: c5471_phyinit
+ *
+ * Description
+ * The c547X EVM board uses a Lucent LU3X31T-T64 transeiver device to
+ * handle the physical layer (PHY). It's a h/w block that on the one end
+ * offers a Media Independent Interface (MII) which is connected to the
+ * Ethernet Interface Module (EIM) internal to the C547x and on the other
+ * end offers either the 10baseT or 100baseT electrical interface connecting
+ * to an RJ45 onboard network connector. The PHY transeiver has several
+ * internal registers allowing host configuration and status access. These
+ * internal registers are accessable by clocking serial data in/out of the
+ * MDIO pin of the LU3X31T-T64 chip. For c547X, the MDC and the MDIO pins
+ * are connected to the C547x GPIO15 and GPIO14 pins respectivley. Host
+ * software twiddles the GPIO pins appropriately to get data serially into
+ * and out of the chip. This is typically a one time operation at boot and
+ * normal operation of the transeiver involves EIM/Transeiver interaction at
+ * the other pins of the transeiver chip and doesn't require host intervention
+ * at the MDC and MDIO pins.
+ *
+ ****************************************************************************/
+
+#if (CONFIG_C5471_ETHERNET_PHY == ETHERNET_PHY_LU3X31T_T64)
+static int c5471_phyinit (void)
+{
+ int phy_id;
+ int status;
+
+ /* Next, Setup GPIO pins to talk serially to the Lucent transeiver chip */
+
+ /* enable gpio bits 15,14 */
+
+ putreg32((getreg32(GPIO_EN) | 0x0000C000), GPIO_EN);
+
+ /* config gpio(15); out -> MDCLK */
+
+ putreg32((getreg32(GPIO_CIO) & ~0x00008000), GPIO_CIO);
+
+ /* config gpio(14); in <- MDIO */
+
+ putreg32((getreg32(GPIO_CIO) | 0x00004000), GPIO_CIO);
+
+ /* initial pin state; MDCLK = 0 */
+
+ putreg32((getreg32(GPIO_IO) & 0x000F3FFF), GPIO_IO);
+
+ /* Next, request a chip reset */
+
+ c5471_mdwrite(0, MD_PHY_CONTROL_REG, 0x8000);
+ while (c5471_mdread(0, MD_PHY_CONTROL_REG) & 0x8000)
+ {
+ /* wait for chip reset to complete */
+ }
+
+ /* Next, Read out the chip ID */
+
+ phy_id = (c5471_mdread(0, MD_PHY_MSB_REG) << 16) | c5471_mdread(0, MD_PHY_LSB_REG);
+ if (phy_id != LU3X31_T64_PHYID)
+ {
+ return -1;
+ }
+
+ /* Next, Set desired network rate, 10BaseT, 100BaseT, or auto. */
+
+#ifdef CONFIG_NET_C5471_AUTONEGOTIATION
+ ndbg("Setting PHY Transceiver for Autonegotiation. (per rrload Makefile).\n");
+ c5471_mdwrite(0, MD_PHY_CONTROL_REG, MODE_AUTONEG);
+#endif
+#ifdef CONFIG_NET_C5471_BASET100
+ ndbg("Setting PHY Transceiver for 100BaseT FullDuplex. (per rrload Makefile).\n");
+ c5471_mdwrite(0, MD_PHY_CONTROL_REG, MODE_100MBIT_FULLDUP);
+#endif
+#ifdef CONFIG_NET_C5471_BASET10
+ ndbg("Setting PHY Transceiver for 10BaseT FullDuplex. (per rrload Makefile).\n");
+ c5471_mdwrite(0, MD_PHY_CONTROL_REG, MODE_10MBIT_FULLDUP);
+#endif
+
+ status = c5471_mdread(0, MD_PHY_CTRL_STAT_REG);
+ return status;
+}
+
+#elif (CONFIG_C5471_ETHERNET_PHY == ETHERNET_PHY_AC101L)
+
+static int c5471_phyinit (void)
+{
+ int phy_id;
+ int status;
+
+ /* Next, Setup GPIO pins to talk serially to the Lucent transeiver chip */
+
+ putreg32((getreg32(GPIO_EN)|0x0000C000), GPIO_EN); /* enable gpio bits 15,14 */
+ putreg32((getreg32(GPIO_CIO)&~0x00008000), GPIO_CIO); /* config gpio(15); out -> MDCLK */
+ putreg32((getreg32(GPIO_CIO)|0x00004000), GPIO_CIO); /* config gpio(14); in <- MDIO */
+ putreg32((getreg32(GPIO_IO)&0x000F3FFF), GPIO_IO); /* initial pin state; MDCLK = 0 */
+
+ return 1;
+}
+
+#else
+# define c5471_phyinit()
+# if !defined(CONFIG_C5471_ETHERNET_PHY)
+# warning "CONFIG_C5471_ETHERNET_PHY not defined -- assumed NONE"
+# endif
+#endif
+
+/****************************************************************************
+ * Name: c5471_inctxcpu
+ *
+ * Description
+ *
+ ****************************************************************************/
+
+static void c5471_inctxcpu(void)
+{
+ if (EIM_TXDESC_WRAP_NEXT & getreg32(c5471->c_txcpudesc))
+ {
+ /* Loop back around to base of descriptor queue */
+
+ c5471->c_txcpudesc = (volatile uint32 *)(getreg32(EIM_CPU_TXBA) + EIM_RAM_START);
+ }
+ else
+ {
+ c5471->c_txcpudesc += 2*sizeof(uint32);
+ }
+}
+
+/****************************************************************************
+ * Name: c5471_incrxcpu
+ *
+ * Description
+ *
+ ****************************************************************************/
+
+static void c5471_incrxcpu(void)
+{
+ if (EIM_RXDESC_WRAP_NEXT & getreg32(c5471->c_rxcpudesc))
+ {
+ /* Loop back around to base of descriptor queue */
+
+ c5471->c_rxcpudesc = (volatile uint32 *)(getreg32(EIM_CPU_RXBA) + EIM_RAM_START);
+ }
+ else
+ {
+ c5471->c_rxcpudesc += 2*sizeof(uint32);
+ }
+}
+
+/****************************************************************************
+ * Function: c5471_transmit
+ *
+ * Description:
+ * Start hardware transmission. Called either from the txdone interrupt
+ * handling or from watchdog based polling.
+ *
+ * Parameters:
+ * c5471 - Reference to the driver state structure
+ *
+ * Returned Value:
+ * OK on success; a negated errno on failure
+ *
+ * Assumptions:
+ *
+ ****************************************************************************/
+
+static int c5471_transmit(struct c5471_driver_s *c5471)
+{
+ volatile uint32 *packetmem;
+ volatile uint32 *ptr;
+ uint16 bytelen;
+ uint16 longlen;
+ boolean bfirstframe;
+ int nshorts;
+ unsigned int i;
+ unsigned int j;
+
+ j = 0;
+ bfirstframe = TRUE;
+ c5471->c_lastdescstart = c5471->c_rxcpudesc;
+
+ while (nbytes)
+ {
+ /* Words #0 and #1 of descriptor */
+
+ while (EIM_TXDESC_OWN_HOST & getreg32(c5471->c_rxcpudesc))
+ {
+ /* Loop until the SWITCH lets go of the descriptor giving us access
+ * rights to submit our new ether frame to it.
+ */
+ }
+
+ if (bfirstframe)
+ {
+ putreg32((getreg32(c5471->c_rxcpudesc) | EIM_RXDESC_FIF), c5471->c_rxcpudesc);
+ }
+ else
+ {
+ putreg32((getreg32(c5471->c_rxcpudesc) & ~EIM_RXDESC_FIF), c5471->c_rxcpudesc);
+ }
+
+ putreg32((getreg32(c5471->c_rxcpudesc) & ~EIM_RXDESC_PADCRC), c5471->c_rxcpudesc);
+
+ if (bfirstframe)
+ {
+ putreg32((getreg32(c5471->c_rxcpudesc)|EIM_RXDESC_PADCRC), c5471->c_rxcpudesc);
+ }
+
+ if (nbytes >= EIM_PACKET_BYTES)
+ {
+ bytelen = EIM_PACKET_BYTES;
+ }
+ else
+ {
+ bytelen = nbytes;
+ }
+
+ /* Next, submit ether frame bytes to the C547x Ether Module packet memory space */
+
+ for (i = 0; i < longlen; i++, j++)
+ {
+ }
+
+ /* Next, submit ether frame bytes to the C5472 Ether Module packet
+ * memory space.
+ */
+
+ /* divide by 2 with round up. */
+
+ nshorts = (bytelen+1)>>1;
+
+ /* Words #2 and #3 of descriptor */
+
+ packetmem = (uint32 *)getreg32(c5471->c_rxcpudesc + sizeof(uint32));
+ for (i = 0; i < nshorts; i++, j++)
+ {
+ /* 16-bits at a time. */
+
+ packetmem[i] = __swap_16(((uint16*)buf)[j]);
+ }
+
+ putreg32(((getreg32(c5471->c_rxcpudesc) & ~EIM_RXDESC_BYTEMASK) | bytelen), c5471->c_rxcpudesc);
+ nbytes -= bytelen;
+ if (0 == nbytes)
+ {
+ putreg32((getreg32(c5471->c_rxcpudesc) | EIM_RXDESC_LIF), c5471->c_rxcpudesc);
+ }
+ else
+ {
+ putreg32((getreg32(c5471->c_rxcpudesc) & ~EIM_RXDESC_LIF), c5471->c_rxcpudesc);
+ }
+
+ /* We're done with that descriptor; give access rights back to h/w */
+
+ putreg32((getreg32(c5471->c_rxcpudesc) | EIM_RXDESC_OWN_HOST), c5471->c_rxcpudesc);
+
+ /* Next, tell Ether Module that those submitted bytes are ready for the wire */
+
+ putreg32(0x00000001, EIM_CPU_RXREADY);
+ c5471->c_lastdescend = c5471->c_rxcpudesc;
+
+ /* Advance to the next free descriptor */
+
+ c5471_incrxcpu();
+ bfirstframe = FALSE;
+ }
+ return;
+
+ /* Verify that the hardware is ready to send another packet */
+
+ /* Increment statistics */
+
+ /* Disable Ethernet interrupts */
+
+ /* Send the packet: address=c5471->c_dev.d_buf, length=c5471->c_dev.d_len */
+
+ /* Restore Ethernet interrupts */
+
+ /* Setup the TX timeout watchdog (perhaps restarting the timer) */
+
+ (void)wd_start(c5471->c_txtimeout, C5471_TXTIMEOUT, c5471_txtimeout, 1, (uint32)c5471);
+ return OK;
+}
+
+/****************************************************************************
+ * Function: c5471_uiptxpoll
+ *
+ * Description:
+ * The transmitter is available, check if uIP has any outgoing packets ready
+ * to send. This is a callback from uip_poll(). uip_poll() may be called:
+ *
+ * 1. When the preceding TX packet send is complete,
+ * 2. When the preceding TX packet send timesout and the interface is reset
+ * 3. During normal TX polling
+ *
+ * Parameters:
+ * dev - Reference to the NuttX driver state structure
+ *
+ * Returned Value:
+ * OK on success; a negated errno on failure
+ *
+ * Assumptions:
+ *
+ ****************************************************************************/
+
+static int c5471_uiptxpoll(struct uip_driver_s *dev)
+{
+ struct c5471_driver_s *c5471 = (struct c5471_driver_s *)dev->d_private;
+
+ /* If the polling resulted in data that should be sent out on the network,
+ * the field d_len is set to a value > 0.
+ */
+
+ if (c5471->c_dev.d_len > 0)
+ {
+ uip_arp_out(&c5471->c_dev);
+ c5471_transmit(c5471);
+
+ /* Check if the ESM has let go of the RX descriptor giving us access
+ * rights to submit another Ethernet frame.
+ */
+
+ if (EIM_TXDESC_OWN_HOST & getreg32(c5471->c_rxcpudesc) != 0)
+ {
+ /* No, then return non-zero to terminate the poll */
+
+ return 1;
+ }
+ }
+
+ /* If zero is returned, the polling will continue until all connections have
+ * been examined.
+ */
+
+ return 0;
+}
+
+/****************************************************************************
+ * Function: c5471_rxstatus
+ *
+ * Description:
+ * An interrupt was received indicating that the last RX packet(s) is done
+ *
+ * Parameters:
+ * c5471 - Reference to the driver state structure
+ *
+ * Returned Value:
+ * None
+ *
+ * Assumptions:
+ *
+ ****************************************************************************/
+
+#ifdef CONFIG_C5471_NET_STATS
+void c5471_rxstatus(int *numbytes)
+{
+ uint32 *pdesc = c5471->c_txcpudesc;
+ uint32 rxstatus;
+
+ /* Walk that last packet we just received to collect xmit status bits. */
+
+ rxstatus = 0;
+ for (;;)
+ {
+ if (EIM_TXDESC_OWN_HOST & getreg32(pdesc))
+ {
+ /* The incoming packe queue is empty. */
+
+ break;
+ }
+
+ rxstatus |= (getreg32(pdesc) & EIM_TXDESC_STATUSMASK);
+
+ if ((getreg32(pdesc) & EIM_TXDESC_LIF) != 0)
+ {
+ break;
+ }
+
+ /* This packet is made up of several descriptors, find next one in chain. */
+
+ if (EIM_TXDESC_WRAP_NEXT & getreg32(pdesc))
+ {
+ /* Loop back around to base of descriptor queue. */
+
+ pdesc = (uint32*)(getreg32(EIM_CPU_RXBA) + EIM_RAM_START);
+ }
+ else
+ {
+ pdesc += 2;
+ }
+ }
+
+ if (rxstatus != 0)
+ {
+ if ((rxstatus & EIM_TXDESC_RETRYERROR) != 0)
+ {
+ c5471->c_rxretries++;
+ }
+
+ if ((rxstatus & EIM_TXDESC_HEARTBEAT) != 0)
+ {
+ c5471->c_rxheartbeat++;
+ }
+
+ if ((rxstatus & EIM_TXDESC_LCOLLISON) != 0)
+ {
+ c5471->c_rxlcollision++;
+ }
+
+ if ((rxstatus & EIM_TXDESC_COLLISION) != 0)
+ {
+ c5471->c_rxcollision++;
+ }
+
+ if ((rxstatus & EIM_TXDESC_CRCERROR) != 0)
+ {
+ c5471->c_rxcrc++;
+ }
+
+ if ((rxstatus & EIM_TXDESC_UNDERRUN) != 0)
+ {
+ c5471->c_rxunderrun++;
+ }
+
+ if ((rxstatus & EIM_TXDESC_LOC) != 0)
+ {
+ c5471->c_rxloc++;
+ }
+ }
+}
+#endif
+
+/****************************************************************************
+ * Function: c5471_receive
+ *
+ * Description:
+ * An interrupt was received indicating the availability of a new RX packet
+ *
+ * Parameters:
+ * c5471 - Reference to the driver state structure
+ *
+ * Returned Value:
+ * None
+ *
+ * Assumptions:
+ *
+ ****************************************************************************/
+
+static void c5471_receive(struct c5471_driver_s *c5471)
+{
+ struct uip_driver_s *dev = c5471->c_dev;
+ uint16 *packetmem = NULL;
+ boolean bmore = TRUE;
+ int packetlen;
+ int bytelen = 0;
+ int nshorts;
+ int i;
+ int j = 0;
+
+ /* Walk the newly received packet contained within the EIM and transfer
+ * its contents to the uIP buffer. This frees up the memory contained within
+ * the EIM for additional packets that might be received later from the network.
+ */
+
+ while (bmore)
+ {
+ /* Words #0 and #1 of descriptor */
+
+ if (EIM_TXDESC_OWN_HOST & getreg32(c5471->c_txcpudesc))
+ {
+ /* No further packets to receive. */
+
+ break;
+ }
+
+ bytelen = (getreg32(c5471->c_txcpudesc) & EIM_TXDESC_BYTEMASK);
+ packetlen += bytelen;
+
+ /* Words #2 and #3 of descriptor */
+
+ packetmem = (uint16*)getreg32(c5471->c_txcpudesc+1);
+
+ /* Divide by 2 with round up to get the number of 16-bit words. */
+
+ nshorts = (bytelen + 1) >> 1;
+ for (i = 0 ; i < nshorts; i++, j++)
+ {
+ /* Check if the received packet will fit without the uIP packet buffer */
+
+ if (packelen < (CONFIG_NET_BUFSIZE - 4))
+ {
+ /* Copy the data data from the hardware to c5471->c_dev.d_buf 16-bits at
+ * a time.
+ */
+
+ ((uint16*)dev->d_buf)[j] = __swap_16(packetmem[i]);
+ }
+ }
+
+ if (getreg32(c5471->c_txcpudesc) & EIM_TXDESC_LIF)
+ {
+ bmore = FALSE;
+ }
+
+ /* Next, Clear all bits of words0/1 of the emptied descriptor except preserve
+ * the settings of a select few. Can leave descriptor words 2/3 alone.
+ */
+
+ putreg32((getreg32(c5471->c_txcpudesc) & (EIM_TXDESC_WRAP_NEXT|EIM_TXDESC_INTRE)),
+ c5471->c_txcpudesc);
+
+ /* Next, Give ownership of now emptied descriptor back to the Ether Module's SWITCH */
+
+ putreg32((getreg32(c5471->c_txcpudesc) | EIM_TXDESC_OWN_HOST), c5471->c_txcpudesc);
+
+ /* Advance to the next data buffer */
+
+ c5471_inctxcpu();
+ }
+
+ /* Adjust the packet length to remove the CRC bytes that uIP doesn't care about. */
+
+ packetlen -= 4;
+
+
+ /* If we successfully transferred the data into the uIP buffer, then pass it on
+ * to uIP for processing.
+ */
+
+ if (packetlen > 0 && packetlen < (CONFIG_NET_BUFSIZE / sizeof(uint16)))
+ {
+ /* Set amount of data in c5471->c_dev.d_len. */
+
+ dev->d_len = packetlen;
+
+ /* We only accept IP packets of the configured type and ARP packets */
+
+#ifdef CONFIG_NET_IPv6
+ if (BUF->type == HTONS(UIP_ETHTYPE_IP6))
+#else
+ if (BUF->type == HTONS(UIP_ETHTYPE_IP))
+#endif
+ {
+ uip_arp_ipin();
+ uip_input(&c5471->c_dev);
+
+ /* If the above function invocation resulted in data that should be
+ * sent out on the network, the field d_len will set to a value > 0.
+ * Send that data now if ESM has let go of the RX descriptor giving us
+ * access rights to submit another Ethernet frame.
+ */
+
+ if (c5471->c_dev.d_len > 0 &&
+ (EIM_TXDESC_OWN_HOST & getreg32(c5471->c_rxcpudesc)) == 0)
+ {
+ uip_arp_out(&c5471->c_dev);
+ c5471_transmit(c5471);
+ }
+ }
+ else if (BUF->type == htons(UIP_ETHTYPE_ARP))
+ {
+ uip_arp_arpin(&c5471->c_dev);
+
+ /* If the above function invocation resulted in data that should be
+ * sent out on the network, the field d_len will set to a value > 0.
+ * Send that data now if ESM has let go of the RX descriptor giving us
+ * access rights to submit another Ethernet frame.
+ */
+
+ if (c5471->c_dev.d_len > 0 &&
+ (EIM_TXDESC_OWN_HOST & getreg32(c5471->c_rxcpudesc)) == 0)
+ {
+ c5471_transmit(c5471);
+ }
+ }
+ }
+}
+
+/****************************************************************************
+ * Function: c5471_txstatus
+ *
+ * Description:
+ * An interrupt was received indicating that the last TX packet(s) is done
+ *
+ * Parameters:
+ * c5471 - Reference to the driver state structure
+ *
+ * Returned Value:
+ * None
+ *
+ * Assumptions:
+ *
+ ****************************************************************************/
+
+#ifdef CONFIG_C5471_NET_STATS
+static inline void c5471_txstatus(void)
+{
+ uint32 *pdesc = c5471->c_lastdescstart;
+ uint32 txstatus;
+
+ /* Walk that last packet we just sent to collect xmit status bits. */
+
+ txstatus = 0;
+ if (c5471->c_lastdescstart && c5471->c_lastdescend)
+ {
+ for (;;)
+ {
+ txstatus |= (getreg32(pdesc) & EIM_TXDESC_STATUSMASK);
+ if (pdesc == c5471->c_lastdescend)
+ {
+ break;
+ }
+
+ /* This packet is made up of several descriptors, find next one in chain. */
+
+ if (EIM_RXDESC_WRAP_NEXT & getreg32(c5471->c_rxcpudesc))
+ {
+ /* Loop back around to base of descriptor queue. */
+
+ pdesc = (uint32*)(getreg32(EIM_CPU_RXBA) + EIM_RAM_START);
+ }
+ else
+ {
+ pdesc += 2;
+ }
+ }
+ }
+
+ if (txstatus)
+ {
+ if ((txstatus & EIM_RXDESC_MISS) != 0)
+ {
+ c5471->c_txmiss++;
+ }
+
+ if ((txstatus & EIM_RXDESC_VLAN) != 0)
+ {
+ c5471->c_txvlan++;
+ }
+
+ if ((txstatus & EIM_RXDESC_LFRAME) != 0)
+ {
+ c5471->c_txlframe++;
+ }
+
+ if ((txstatus & EIM_RXDESC_SFRAME) != 0)
+ {
+ c5471->c_txsframe++;
+ }
+
+ if ((txstatus & EIM_RXDESC_CRCERROR) != 0)
+ {
+ c5471->c_txcrc++;
+ }
+
+ if ((txstatus & EIM_RXDESC_OVERRUN) != 0)
+ {
+ c5471->c_txoverrun++;
+ }
+
+ if ((txstatus & EIM_RXDESC_OVERRUN) != 0)
+ {
+ c5471->c_txalign++;
+ }
+}
+#endif
+
+/****************************************************************************
+ * Function: c5471_txdone
+ *
+ * Description:
+ * An interrupt was received indicating that the last TX packet(s) is done
+ *
+ * Parameters:
+ * c5471 - Reference to the driver state structure
+ *
+ * Returned Value:
+ * None
+ *
+ * Assumptions:
+ *
+ ****************************************************************************/
+
+static void c5471_txdone(struct c5471_driver_s *c5471)
+{
+ /* Update statistics */
+
+#ifdef CONFIG_C5471_NET_STATS
+ c5471->c_txpackets++;
+
+ /* Check for TX errors */
+
+ c5471_txstatus(c5471);
+#endif
+
+ /* If no further xmits are pending, then cancel the TX timeout */
+
+ wd_cancel(c5471->c_txtimeout);
+
+ /* Then poll uIP for new XMIT data */
+
+ (void)uip_poll(&c5471->c_dev, c5471_uiptxpoll);
+}
+
+/****************************************************************************
+ * Function: c5471_interrupt
+ *
+ * Description:
+ * Hardware interrupt handler
+ *
+ * Parameters:
+ * irq - Number of the IRQ that generated the interrupt
+ * context - Interrupt register state save info (architecture-specific)
+ *
+ * Returned Value:
+ * OK on success
+ *
+ * Assumptions:
+ *
+ ****************************************************************************/
+
+static int c5471_interrupt(int irq, FAR void *context)
+{
+ /* Get and clear interrupt status bits */
+
+ /* Handle interrupts according to status bit settings */
+
+ /* Check if we received an incoming packet, if so, call c5471_receive() */
+
+ c5471->c_eimstatus = getreg32(EIM_STATUS);
+ if (EIM_STATUS_CPU_TX & c5471->c_eimstatus)
+ {
+ /* An incoming packet has been received by the EIM from the network and
+ * the interrupt associated with EIM's CPU TX queue has been asserted. It
+ * is the EIM's CPU TX queue that we need to read from to get those
+ * packets. We use this terminology to stay consistent with the Orion
+ * documentation.
+ */
+
+ #ifdef CONFIG_C5471_NET_STATS
+ /* Increment the count of received packets */
+
+ c5471->c_rxpackets++;
+
+ /* Check for RX errors */
+
+ c5471_rxstatus(c5471);
+#endif
+
+ /* Process the received packet */
+
+ c5471_receive(c5471);
+ }
+
+ /* Check is a packet transmission just completed. If so, call c5471_txdone */
+
+ if (EIM_STATUS_CPU_RX & c5471->c_eimstatus)
+ {
+ /* An outgoing packet has been processed by the EIM and the interrupt
+ * associated with EIM's CPU RX que has been asserted. It is the EIM's
+ * CPU RX queue that we put packets on to send them *out*. TWe use this
+ * terminology to stay consistent with the Orion documentation.
+ */
+
+ c5471_txdone(c5471);
+ }
+
+ /* Enable Ethernet interrupts (perhaps excluding the TX done interrupt if
+ * there are no pending transmissions.
+ */
+
+ return OK;
+}
+
+/****************************************************************************
+ * Function: c5471_txtimeout
+ *
+ * Description:
+ * Our TX watchdog timed out. Called from the timer interrupt handler.
+ * The last TX never completed. Reset the hardware and start again.
+ *
+ * Parameters:
+ * argc - The number of available arguments
+ * arg - The first argument
+ *
+ * Returned Value:
+ * None
+ *
+ * Assumptions:
+ *
+ ****************************************************************************/
+
+static void c5471_txtimeout(int argc, uint32 arg, ...)
+{
+ struct c5471_driver_s *c5471 = (struct c5471_driver_s *)arg;
+
+ /* Increment statistics */
+
+#ifdef CONFIG_C5471_NET_STATS
+ c5471->c_txtimeouts++;
+#endif
+
+ /* Then try to restart the hardware */
+
+ c5471_ifdown(&c5471->c_dev);
+ c5471_ifup(&c5471->c_dev);
+
+ /* Then poll uIP for new XMIT data */
+
+ (void)uip_poll(&c5471->c_dev, c5471_uiptxpoll);
+}
+
+/****************************************************************************
+ * Function: c5471_polltimer
+ *
+ * Description:
+ * Periodic timer handler. Called from the timer interrupt handler.
+ *
+ * Parameters:
+ * argc - The number of available arguments
+ * arg - The first argument
+ *
+ * Returned Value:
+ * None
+ *
+ * Assumptions:
+ *
+ ****************************************************************************/
+
+static void c5471_polltimer(int argc, uint32 arg, ...)
+{
+ struct c5471_driver_s *c5471 = (struct c5471_driver_s *)arg;
+
+ /* Check if the ESM has let go of the RX descriptor giving us access rights
+ * to submit another Ethernet frame.
+ */
+
+ if (EIM_TXDESC_OWN_HOST & getreg32(c5471->c_rxcpudesc) == 0)
+ {
+ /* If so, update TCP timing states and poll uIP for new XMIT data */
+
+ (void)uip_timer(&c5471->c_dev, c5471_uiptxpoll, C5471_POLLHSEC);
+ }
+
+ /* Setup the watchdog poll timer again */
+
+ (void)wd_start(c5471->c_txpoll, C5471_WDDELAY, c5471_polltimer, 1, arg);
+}
+
+/****************************************************************************
+ * Function: c5471_ifup
+ *
+ * Description:
+ * NuttX Callback: Bring up the Ethernet interface when an IP address is
+ * provided
+ *
+ * Parameters:
+ * dev - Reference to the NuttX driver state structure
+ *
+ * Returned Value:
+ * None
+ *
+ * Assumptions:
+ * The user has assigned a MAC to the driver
+ *
+ ****************************************************************************/
+
+static int c5471_ifup(struct uip_driver_s *dev)
+{
+ struct c5471_driver_s *c5471 = (struct c5471_driver_s *)dev->d_private;
+ volatile uint32 clearbits;
+
+ ndbg("Bringing up: %d.%d.%d.%d\n",
+ dev->d_ipaddr & 0xff, (dev->d_ipaddr >> 8) & 0xff,
+ (dev->d_ipaddr >> 16) & 0xff, dev->d_ipaddr >> 24 );
+
+ /* Initilize Ethernet interface */
+
+ c5471_reset();
+
+ /* Assign the MAC to the device */
+
+ c5471_macassign(c5471);
+
+ /* Enable interrupts going from EIM Module to Interrupt Module. */
+
+ clearbits = eth_in32(EIM_STATUS);
+ putreg32((getreg32(EIM_INTEN) | EIM_INTEN_CPU_TX|EIM_INTEN_CPU_RX)), EIM_INTEN);
+
+ /* Next, go on-line. According to the C547X documentation the enables have to
+ * occur in this order to insure proper operation; ESM first then the ENET.
+ */
+
+ putreg32((getreg32(EIM_CTRL) | EIM_CTRL_ESM_EN), EIM_CTRL); /* enable ESM */
+ putreg32((getreg32(ENET0_MODE) | ENET_MODE_ENABLE), ENET0_MODE); /* enable ENET */
+ up_mdelay(100);
+
+ /* Set and activate a timer process */
+
+ (void)wd_start(c5471->c_txpoll, C5471_WDDELAY, c5471_polltimer, 1, (uint32)c5471);
+
+ /* Enable the Ethernet interrupt */
+
+ c5471->c_bifup = TRUE;
+ up_enable_irq(C5471_IRQ_ETHER);
+ return OK;
+}
+
+/****************************************************************************
+ * Function: c5471_ifdown
+ *
+ * Description:
+ * NuttX Callback: Stop the interface.
+ *
+ * Parameters:
+ * dev - Reference to the NuttX driver state structure
+ *
+ * Returned Value:
+ * None
+ *
+ * Assumptions:
+ *
+ ****************************************************************************/
+
+static int c5471_ifdown(struct uip_driver_s *dev)
+{
+ struct c5471_driver_s *c5471 = (struct c5471_driver_s *)dev->d_private;
+ irqstate_t flags;
+
+ /* Disable the Ethernet interrupt */
+
+ flags = irqsave();
+ up_disable_irq(C5471_IRQ_ETHER);
+
+ /* Disable interrupts going from EIM Module to Interrupt Module. */
+
+ putreg32((getreg32(EIM_INTEN) & ~(EIM_INTEN_CPU_TX|EIM_INTEN_CPU_RX)), EIM_INTEN);
+
+ /* Disable ENET */
+
+ putreg32((getreg32(ENET0_MODE) & ~ENET_MODE_ENABLE), ENET0_MODE); /* disable ENET */
+
+ /* Disable ESM */
+
+ putreg32((getreg32(EIM_CTRL) & ~EIM_CTRL_ESM_EN), EIM_CTRL); /* disable ESM */
+
+ /* Cancel the TX poll timer and TX timeout timers */
+
+ wd_cancel(c5471->c_txpoll);
+ wd_cancel(c5471->c_txtimeout);
+
+ /* Reset the device */
+
+ c5471->c_bifup = FALSE;
+ irqrestore(flags);
+ return OK;
+}
+
+/****************************************************************************
+ * Function: c5471_txavail
+ *
+ * Description:
+ * Driver callback invoked when new TX data is available. This is a
+ * stimulus perform an out-of-cycle poll and, thereby, reduce the TX
+ * latency.
+ *
+ * Parameters:
+ * dev - Reference to the NuttX driver state structure
+ *
+ * Returned Value:
+ * None
+ *
+ * Assumptions:
+ * Called in normal user mode
+ *
+ ****************************************************************************/
+
+static int c5471_txavail(struct uip_driver_s *dev)
+{
+ struct c5471_driver_s *c5471 = (struct c5471_driver_s *)dev->d_private;
+ irqstate_t flags;
+
+ flags = irqsave();
+
+ /* Ignore the notification if the interface is not yet up */
+
+ if (c5471->c_bifup)
+ {
+ /* Check if the ESM has let go of the RX descriptor giving us access
+ * rights to submit another Ethernet frame.
+ */
+
+ if (EIM_TXDESC_OWN_HOST & getreg32(c5471->c_rxcpudesc) == 0)
+ {
+ /* If so, then poll uIP for new XMIT data */
+
+ (void)uip_poll(&c5471->c_dev, c5471_uiptxpoll);
+ }
+ }
+
+ irqrestore(flags);
+ return OK;
+}
+
+/****************************************************************************
+ * Name: c5471_eimreset
+ *
+ * Description
+ * The C547x docs states that a module should generally be reset according
+ * to the following algorithm:
+ *
+ * 1. Put the module in reset.
+ * 2. Switch on the module clock.
+ * 3. Wait for eight clock cycles.
+ * 4. Release the reset.
+ *
+ ****************************************************************************/
+
+static void c5471_eimreset (struct c5471_driver_s *c5471)
+{
+ /* Stop the EIM module clock */
+
+ putreg32((getreg32(CLKM) | CLKM_EIM_CLK_STOP), CLKM);
+
+ /* Put EIM module in reset */
+
+ putreg32((getreg32(CLKM_RESET) & ~CLKM_RESET_EIM), CLKM_RESET);
+
+ /* Start the EIM module clock */
+
+ putreg32((getreg32(CLKM) & ~CLKM_EIM_CLK_STOP), CLKM);
+
+ /* Assert nRESET to reset the board's PHY0/1 chips */
+
+ putreg32((CLKM_CTL_RST_EXT_RESET|CLKM_CTL_RST_LEAD_RESET), CLKM_CTL_RST);
+ up_mdelay(2);
+
+ /* Release the peripheral nRESET signal */
+
+ putreg32(CLKM_CTL_RST_LEAD_RESET, CLKM_CTL_RST);
+
+ /* Release EIM module reset */
+
+ putreg32((getreg32(CLKM_RESET) | CLKM_RESET_EIM), CLKM_RESET);
+
+ /* All EIM register should now be in there power-up default states */
+
+ c5471->c_lastdescstart = NULL;
+ c5471->c_lastdescend = NULL;
+}
+
+/****************************************************************************
+ * Name: c5471_eimconfig
+ *
+ * Description
+ * Assumes that all registers are currently in the power-up reset state.
+ * This routine then modifies that state to provide our specific ethernet
+ * configuration.
+ *
+ ****************************************************************************/
+
+static void c5471_eimconfig(void)
+{
+ volatile uint8 *pbuf;
+ volatile uint32 *pdesc;
+ volatile uint32 *val;
+ int i;
+
+ pdesc = EIM_RAM_START;
+ pbuf = (uint8*)EIM_RAM_START + 0x6C0;
+
+ /* TX ENET 0 */
+
+ putreg32((((uint32)pdesc) & 0x0000ffff), ENET0_TDBA); /* 16bit offset address */
+ for (i = NUM_DESC_TX-1; i >= 0; i--)
+ {
+ if (i == 0)
+ val = EIM_TXDESC_WRAP_NEXT;
+ else
+ val = EIM_TXDESC_WRAP_FIRST;
+ val |= EIM_TXDESC_OWN_HOST|EIM_TXDESC_INTRE|EIM_TXDESC_PADCRC|EIM_PACKET_BYTES;
+ putreg32(val, pdesc);
+
+ pdesc++;
+ putreg32((uint32)pbuf, pdesc);
+
+ pdesc++;
+ *((volatile uint32 *)pbuf) = 0x00000000;
+
+ pbuf += EIM_PACKET_BYTES;
+ *((volatile uint32 *)pbuf) = 0x00000000;
+
+ /* As per c547X doc; this space for Ether Module's "Buffer Usage Word" */
+
+ pbuf += 4;
+ }
+
+ /* RX ENET 0 */
+
+ putreg32((((uint32)pdesc) & 0x0000ffff), ENET0_RDBA); /* 16bit offset address */
+ for (i = NUM_DESC_RX-1; i >= 0; i--)
+ {
+ if (i == 0)
+ val = EIM_RXDESC_WRAP_NEXT;
+ else
+ val = EIM_RXDESC_WRAP_FIRST;
+ val |= EIM_RXDESC_OWN_ENET|EIM_RXDESC_INTRE|EIM_RXDESC_PADCRC|EIM_PACKET_BYTES;
+ putreg32(val, pdesc);
+
+ pdesc++;
+ putreg32((uint32)pbuf, pdesc);
+
+ pdesc++;
+ *((uint32 *)pbuf) = 0x00000000;
+
+ pbuf += EIM_PACKET_BYTES;
+ *((uint32 *)pbuf) = 0x00000000;
+
+ /* As per c547X doc; this space for Ether Module's "Buffer Usage Word" */
+
+ pbuf += 4;
+ }
+
+ /* TX CPU */
+
+ c5471->c_txcpudesc = pdesc;
+ putreg32((((uint32)pdesc) & 0x0000ffff), EIM_CPU_TXBA); /* 16bit offset address */
+ for (i = NUM_DESC_TX-1; i >= 0; i--)
+ {
+ /* Set words 1+2 of the TXDESC */
+
+ if (i == 0)
+ val = EIM_TXDESC_WRAP_NEXT;
+ else
+ val = EIM_TXDESC_WRAP_FIRST;
+ val |= EIM_TXDESC_OWN_HOST|EIM_TXDESC_INTRE|EIM_TXDESC_PADCRC|EIM_PACKET_BYTES;
+ putreg32(val, pdesc);
+
+ pdesc++;
+ putreg32((uint32)pbuf, pdesc);
+
+ pdesc++;
+ *((uint32 *)pbuf) = 0x00000000;
+
+ pbuf += EIM_PACKET_BYTES;
+ *((uint32 *)pbuf) = 0x00000000;
+
+ /* As per C547X doc; this space for Ether Module's "Buffer Usage Word" */
+
+ pbuf += 4;
+ }
+
+ /* RX CPU */
+
+ c5471->c_rxcpudesc = pdesc;
+ putreg32((uint32)((uint32)(pdesc)&0x0000ffff), EIM_CPU_RXBA); /* 16bit offset address */
+ for (i = NUM_DESC_RX-1; i >= 0; i--)
+ {
+ /* Set words 1+2 of the RXDESC */
+
+ if (i == 0)
+ val = EIM_RXDESC_WRAP_NEXT;
+ else
+ val = EIM_RXDESC_WRAP_FIRST;
+ val |= EIM_RXDESC_OWN_ENET|EIM_RXDESC_INTRE|EIM_RXDESC_PADCRC|EIM_PACKET_BYTES;
+ putreg32(val, pdesc);
+
+ pdesc++;
+ putreg32((uint32)pbuf, pdesc);
+
+ pdesc++;
+ *((uint32 *)pbuf) = 0x0000;
+
+ pbuf += EIM_PACKET_BYTES;
+ *((uint32 *)pbuf) = 0x0000;
+
+ /* As per c547X doc; this space for Ether Module's "Buffer Usage Word" */
+
+ pbuf += 4;
+ }
+
+ /* Save the descriptor packet size */
+
+ putreg32(EIM_PACKET_BYTES, EIM_BUFSIZE);
+
+ /* Set the filter mode */
+
+#if 0
+ putreg32(EIM_FILTER_UNICAST, EIM_CPU_FILTER);
+#else
+// putreg32(EIM_FILTER_LOGICAL|EIM_FILTER_UNICAST|EIM_FILTER_MULTICAST|
+// EIM_FILTER_BROADCAST, EIM_CPU_FILTER);
+ putreg32(EIM_FILTER_UNICAST|EIM_FILTER_MULTICAST|EIM_FILTER_BROADCAST, EIM_CPU_FILTER);
+#endif
+
+ /* Disable all Ethernet interrupts */
+
+ putreg32(0x00000000, EIM_INTEN);
+
+ /* Setup the EIM control register */
+
+#if 1
+ putreg32(EIM_CTRL_ENET0_EN|EIM_CTRL_RXENET0_EN|EIM_CTRL_TXENET0_EN|
+ EIM_CTRL_RXCPU_EN|EIM_CTRL_TXCPU_EN, EIM_CTRL);
+#else
+ putreg32(EIM_CTRL_ENET0_EN|EIM_CTRL_ENET0_FLW|EIM_CTRL_RXENET0_EN|
+ EIM_CTRL_TXENET0_EN|EIM_CTRL_RXCPU_EN|EIM_CTRL_TXCPU_EN, EIM_CTRL);
+#endif
+
+#if 1
+ putreg32(0x00000000, EIM_MFVHI);
+#else
+ putreg32(0x0000ff00, EIM_MFVHI);
+#endif
+
+ putreg32(0x00000000, EIM_MFVLO);
+ putreg32(0x00000000, EIM_MFMHI);
+ putreg32(0x00000000, EIM_MFMLO);
+ putreg32(0x00000018, EIM_RXTH);
+ putreg32(0x00000000, EIM_CPU_RXREADY);
+
+ /* Setup the ENET0 mode register */
+
+#if 1
+ putreg32(ENET_MODE_RJCT_SFE|ENET_MODE_MWIDTH|ENET_MODE_FULLDUPLEX, ENET0_MODE);
+#else
+ putreg32(ENET_MODE_RJCT_SFE|ENET_MODE_MWIDTH|ENET_MODE_HALFDUPLEX, ENET0_MODE);
+#endif
+
+ putreg32(0x00000000, ENET0_BOFFSEED);
+ putreg32(0x00000000, ENET0_FLWPAUSE);
+ putreg32(0x00000000, ENET0_FLWCONTROL);
+ putreg32(0x00000000, ENET0_VTYPE);
+
+#if 0
+ putreg32(ENET_ADR_BROADCAST|ENET_ADR_PROMISCUOUS, ENET0_ADRMODE_EN);
+#else
+ /* The CPU port is not PROMISCUOUS, it wants a no-promiscuous address
+ * match yet the the SWITCH receives packets from the PROMISCUOUS ENET0
+ * which routes all packets for filter matching at the CPU port which
+ * then allows the s/w to see the new incoming packetes that passed
+ * the filter. Here we are setting the main SWITCH closest the ether
+ * wire.
+ */
+
+ putreg32(ENET_ADR_PROMISCUOUS, ENET0_ADRMODE_EN);
+#endif
+
+ putreg32(0x00000000, ENET0_DRP);
+ up_mdelay(500);
+}
+
+/****************************************************************************
+ * Name: c5471_reset
+ *
+ * Description
+ *
+ ****************************************************************************/
+
+static void c5471_reset(struct c5471_driver_s *c5471)
+{
+#if (CONFIG_C5471_ETHERNET_PHY == ETHERNET_PHY_LU3X31T_T64)
+ c5471_eimreset(c5471);
+#endif
+ c5471_phyinit();
+ c5471_eimconfig();
+}
+
+/****************************************************************************
+ * Name: c5471_macassign
+ *
+ * Description
+ * Set the mac address of our CPU ether port so that when the SWITCH
+ * receives packets from the PROMISCUOUS ENET0 it will switch them to the
+ * CPU port and cause a packet arrival event on the Switch's CPU TX queue
+ * when an address match occurs. The CPU port is not PROMISCUOUS and wants
+ * to see only packets specifically addressed to this device.
+ *
+ ****************************************************************************/
+
+static void c5471_macassign(struct c5471_driver_s *c5471)
+{
+ struct uip_driver_s *dev = &c5471->s_dev;
+ register uint32 tmp;
+
+ /* Set CPU port MAC address. S/W will only see incoming packets that match
+ * this destination address.
+ */
+
+ tmp = (((uint32)dev->d_mac[0]) << 8) | ((uint32)dev->d_mac[1]);
+ putreg32(tmp, EIM_CPU_DAHI);
+
+ tmp = (((uint32)dev->d_mac[2]) << 24) | (((uint32)dev->d_mac[3]) << 16) |
+ (((uint32)dev->d_mac[4]) << 8) | ((uint32)dev->d_mac[5])
+ putreg32(tmp, EIM_CPU_DALO);
+
+#if 0
+ /* Set the ENET MAC address */
+
+ putreg32(getreg32(EIM_CPU_DAHI), ENET0_PARHI);
+ putreg32(getreg32(EIM_CPU_DALO), ENET0_PARLO);
+ putreg32(getreg32(EIM_CPU_DAHI), ENET0_LARHI);
+ putreg32(getreg32(EIM_CPU_DALO), ENET0_LARLO);
+
+#else
+ /* ENET MAC assignment not needed for its PROMISCUOUS mode */
+
+ putreg32(0x00000000, ENET0_PARHI);
+ putreg32(0x00000000, ENET0_PARLO);
+ putreg32(0x00000000, ENET0_LARHI);
+ putreg32(0x00000000, ENET0_LARLO);
+
+#endif
+}
+
+/****************************************************************************
+ * Public Functions
+ ****************************************************************************/
+
+/****************************************************************************
+ * Function: c5471_initialize
+ *
+ * Description:
+ * Initialize the Ethernet driver
+ *
+ * Parameters:
+ * None
+ *
+ * Returned Value:
+ * OK on success; Negated errno on failure.
+ *
+ * Assumptions:
+ *
+ ****************************************************************************/
+
+/* Initialize the DM90x0 chip and driver */
+
+int up_netinitialize(void)
+{
+ /* Attach the IRQ to the driver */
+
+ if (irq_attach(C5471_IRQ_ETHER, c5471_interrupt))
+ {
+ /* We could not attach the ISR to the ISR */
+
+ return -EAGAIN;
+ }
+
+ /* Initialize the driver structure */
+
+ memset(g_c5471, 0, CONFIG_C5471_NET_NINTERFACES*sizeof(struct c5471_driver_s));
+ g_c5471[0].c_dev.d_ifup = c5471_ifup; /* I/F down callback */
+ g_c5471[0].c_dev.d_ifdown = c5471_ifdown; /* I/F up (new IP address) callback */
+ g_c5471[0].c_dev.d_txavail = c5471_txavail; /* New TX data callback */
+ g_c5471[0].c_dev.d_private = (void*)g_c5471; /* Used to recover private state from dev */
+
+ /* Create a watchdog for timing polling for and timing of transmisstions */
+
+ g_c5471[0].c_txpoll = wd_create(); /* Create periodic poll timer */
+ g_c5471[0].c_txtimeout = wd_create(); /* Create TX timeout timer */
+
+ /* Register the device with the OS so that socket IOCTLs can be performed */
+
+ (void)netdev_register(&g_c5471[0].c_dev);
+ return OK;
+}
+
+#endif /* CONFIG_NET */
+
diff --git a/nuttx/arch/arm/src/c5471/chip.h b/nuttx/arch/arm/src/c5471/chip.h
index 6720893c4..7121fd220 100644
--- a/nuttx/arch/arm/src/c5471/chip.h
+++ b/nuttx/arch/arm/src/c5471/chip.h
@@ -1,4 +1,4 @@
-/************************************************************
+/****************************************************************************
* c5471/chip.h
*
* Copyright (C) 2007 Gregory Nutt. All rights reserved.
@@ -31,20 +31,61 @@
* ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*
- ************************************************************/
+ ****************************************************************************/
#ifndef __C5471_CHIP_H
#define __C5471_CHIP_H
-/************************************************************
+/****************************************************************************
* Included Files
- ************************************************************/
+ ****************************************************************************/
-/************************************************************
+/****************************************************************************
* Definitions
- ************************************************************/
-
-/* UARTs ****************************************************/
+ ****************************************************************************/
+
+#define EIM_RAM_START 0xffd00000
+
+/* Ethernet Interface Module (EIM) ******************************************/
+
+#define EIM_CTRL 0xffff0000 /* ESM control register */
+#define EIM_STATUS 0xffff0004 /* ESM status register */
+#define EIM_CPU_TXBA 0xffff0008 /* CPU TX descriptors base address */
+#define EIM_CPU_RXBA 0xffff000c /* CPU RX descriptors base address */
+#define EIM_BUFSIZE 0xffff0010 /* Packet buffer size register */
+#define EIM_CPU_FILTER 0xffff0014 /* CPU filtering contol registers */
+#define EIM_CPU_DAHI 0xffff0018 /* CPU destination address (HI) */
+#define EIM_CPU_DALO 0xffff001c /* CPU destination address (LO) */
+#define EIM_MFVHI 0xffff0020 /* Multicast filter valid (HI) */
+#define EIM_MFVLO 0xffff0024 /* Multicast filter valid (LO) */
+#define EIM_MFMHI 0xffff0028 /* Multicast filter mask (HI) */
+#define EIM_MFMLO 0xffff002c /* Multicast filter mask (LO) */
+#define EIM_RXTH 0xffff0030 /* RX threshold register */
+#define EIM_CPU_RXREADY 0xffff0034 /* CPU RX ready register */
+#define EIM_INTEN 0xffff0038 /* ESM interrupt enable register */
+#define EIM_ENET0_TXDESC 0xffff0040 /* ENET0 TX Queue pointer */
+#define EIM_ENET0_RXDESC 0xffff0044 /* ENET0 RX Queue pointer */
+#define EIM_CPU_TXDESC 0xffff0050 /* CPU TX Queue pointer */
+#define EIM_CPU_RXDESC 0xffff0054 /* CPU RX Queue pointer */
+
+#define ENET0_MODE 0xffff0100 /* Mode register */
+#define ENET0_BOFFSEED 0xffff0104 /* Backoff seed register */
+#define ENET0_BCOUNT 0xffff0108 /* Backoff count register */
+#define ENET0_FLWPAUSE 0xffff010c /* TX flow pause count register */
+#define ENET0_FLWCONTROL 0xffff0110 /* Flow control register */
+#define ENET0_VTYPE 0xffff0114 /* VTYPE tag register */
+#define ENET0_SEISR 0xffff0118 /* System error int status register */
+#define ENET0_TXBUFRDY 0xffff011c /* TX descripter buffer ready */
+#define ENET0_TDBA 0xffff0120 /* TX descriptor base address */
+#define ENET0_RDBA 0xffff0124 /* RX descriptor base address */
+#define ENET0_PARHI 0xffff0128 /* Dest phys address match (HI) */
+#define ENET0_PARLO 0xffff012c /* Dest phys address match (LO) */
+#define ENET0_LARHI 0xffff0130 /* Log address hash filter (HI) */
+#define ENET0_LARLO 0xffff0134 /* Log address hash filter (LO) */
+#define ENET0_ADRMODE_EN 0xffff0138 /* Address mode enable register */
+#define ENET0_DRP 0xffff013c /* Desc ring poll interval count */
+
+/* UARTs ********************************************************************/
#define UART_IRDA_BASE 0xffff0800
#define UART_MODEM_BASE 0xffff1000
@@ -110,7 +151,7 @@
#define UART_MODEM_RDPTR_UTX 0xffff1064 /* TX FIFO Read Pointer Register */
#define UART_MODEM_WRPTR_UTX 0xffff1068 /* TX FIFO Write Pointer Register */
-/* UART Settings ********************************************/
+/* UART Settings ************************************************************/
/* Miscellaneous UART settings. */
@@ -164,13 +205,13 @@
#define MDR_AUTOBAUDING_MODE 0x00000002 /* Modem UART only */
#define MDR_RESET_MODE 0x00000007 /* Both IrDA and Modem UARTs */
-/* SPI ******************************************************/
+/* SPI **********************************************************************/
#define MAX_SPI 3
#define SPI_REGISTER_BASE 0xffff2000
-/* GIO ******************************************************/
+/* GIO **********************************************************************/
#define MAX_GIO (35)
@@ -211,7 +252,7 @@
#define KBGPIO_EN 0xffff2914 /* Selects register for muxed
* KBGPIOs */
-/* Timers ***************************************************/
+/* Timers *******************************************************************/
#define C5471_TIMER0_CTRL 0xffff2a00
#define C5471_TIMER0_CNT 0xffff2a04
@@ -241,8 +282,8 @@
#define ILR_IRQ3_REG 0xffff2d28 /* 3-GPIO0 */
#define ILR_IRQ4_REG 0xffff2d2c /* 4-Ethernet */
#define ILR_IRQ5_REG 0xffff2d30 /* 5-KBGPIO[7:0] */
-#define ILR_IRQ6_REG 0xffff2d34 /* 6-Uart serial */
-#define ILR_IRQ7_REG 0xffff2d38 /* 7-Uart IRDA */
+#define ILR_IRQ6_REG 0xffff2d34 /* 6-Uart serial */
+#define ILR_IRQ7_REG 0xffff2d38 /* 7-Uart IRDA */
#define ILR_IRQ8_REG 0xffff2d3c /* 8-KBGPIO[15:8] */
#define ILR_IRQ9_REG 0xffff2d40 /* 9-GPIO3 */
#define ILR_IRQ10_REG 0xffff2d44 /* 10-GPIO2 */
@@ -252,11 +293,22 @@
#define ILR_IRQ14_REG 0xffff2d54 /* 14-GPIO[19:4] */
#define ILR_IRQ15_REG 0xffff2d58 /* 15-API */
-/* I2C ******************************************************/
+/* CLKM *********************************************************************/
+
+#define CLKM 0xffff2f00
+#define CLKM_CTL_RST 0xffff2f10
+#define CLKM_RESET 0xffff2f18
+
+#define CLKM_RESET_EIM 0x00000008
+#define CLKM_EIM_CLK_STOP 0x00000010
+#define CLKM_CTL_RST_LEAD_RESET 0x00000000
+#define CLKM_CTL_RST_EXT_RESET 0x00000002
+
+/* I2C **********************************************************************/
#define MAX_I2C 1
-/* API ******************************************************/
+/* API **********************************************************************/
#define DSPRAM_BASE 0xffe00000 /* DSPRAM base address */
#define DSPRAM_END 0xffe03fff
@@ -308,12 +360,12 @@
#define DSPMEM_ARM_TO_DSP(addr) \
((((__u32)(addr) - DSPMEM_ARM_START) >> 1) + DSPMEM_DSP_START)
-/************************************************************
+/****************************************************************************
* Inline Functions
- ************************************************************/
+ ****************************************************************************/
-/************************************************************
+/****************************************************************************
* Public Function Prototypes
- ************************************************************/
+ ****************************************************************************/
#endif /* __C5471_CHIP_H */