1fd92db83d
Update the naming convention used in the network stack functions and variables that Ethernet drivers use to interact with it. This cleans up the temporary hacks that were added to this interface along with the DM support. This patch has a few remaining checkpatch.pl failures that would be out of the scope of this patch to fix (drivers that are in gross violation of checkpatch.pl). Signed-off-by: Joe Hershberger <joe.hershberger@ni.com> Acked-by: Simon Glass <sjg@chromium.org>
393 lines
9.9 KiB
C
393 lines
9.9 KiB
C
/*
|
|
* (C) Copyright 2007-2009 Michal Simek
|
|
* (C) Copyright 2003 Xilinx Inc.
|
|
*
|
|
* Michal SIMEK <monstr@monstr.eu>
|
|
*
|
|
* SPDX-License-Identifier: GPL-2.0+
|
|
*/
|
|
|
|
#include <common.h>
|
|
#include <net.h>
|
|
#include <config.h>
|
|
#include <malloc.h>
|
|
#include <asm/io.h>
|
|
#include <fdtdec.h>
|
|
|
|
#undef DEBUG
|
|
|
|
#define ENET_ADDR_LENGTH 6
|
|
|
|
/* EmacLite constants */
|
|
#define XEL_BUFFER_OFFSET 0x0800 /* Next buffer's offset */
|
|
#define XEL_TPLR_OFFSET 0x07F4 /* Tx packet length */
|
|
#define XEL_TSR_OFFSET 0x07FC /* Tx status */
|
|
#define XEL_RSR_OFFSET 0x17FC /* Rx status */
|
|
#define XEL_RXBUFF_OFFSET 0x1000 /* Receive Buffer */
|
|
|
|
/* Xmit complete */
|
|
#define XEL_TSR_XMIT_BUSY_MASK 0x00000001UL
|
|
/* Xmit interrupt enable bit */
|
|
#define XEL_TSR_XMIT_IE_MASK 0x00000008UL
|
|
/* Buffer is active, SW bit only */
|
|
#define XEL_TSR_XMIT_ACTIVE_MASK 0x80000000UL
|
|
/* Program the MAC address */
|
|
#define XEL_TSR_PROGRAM_MASK 0x00000002UL
|
|
/* define for programming the MAC address into the EMAC Lite */
|
|
#define XEL_TSR_PROG_MAC_ADDR (XEL_TSR_XMIT_BUSY_MASK | XEL_TSR_PROGRAM_MASK)
|
|
|
|
/* Transmit packet length upper byte */
|
|
#define XEL_TPLR_LENGTH_MASK_HI 0x0000FF00UL
|
|
/* Transmit packet length lower byte */
|
|
#define XEL_TPLR_LENGTH_MASK_LO 0x000000FFUL
|
|
|
|
/* Recv complete */
|
|
#define XEL_RSR_RECV_DONE_MASK 0x00000001UL
|
|
/* Recv interrupt enable bit */
|
|
#define XEL_RSR_RECV_IE_MASK 0x00000008UL
|
|
|
|
struct xemaclite {
|
|
u32 nexttxbuffertouse; /* Next TX buffer to write to */
|
|
u32 nextrxbuffertouse; /* Next RX buffer to read from */
|
|
u32 txpp; /* TX ping pong buffer */
|
|
u32 rxpp; /* RX ping pong buffer */
|
|
};
|
|
|
|
static u32 etherrxbuff[PKTSIZE_ALIGN/4]; /* Receive buffer */
|
|
|
|
static void xemaclite_alignedread(u32 *srcptr, void *destptr, u32 bytecount)
|
|
{
|
|
u32 i;
|
|
u32 alignbuffer;
|
|
u32 *to32ptr;
|
|
u32 *from32ptr;
|
|
u8 *to8ptr;
|
|
u8 *from8ptr;
|
|
|
|
from32ptr = (u32 *) srcptr;
|
|
|
|
/* Word aligned buffer, no correction needed. */
|
|
to32ptr = (u32 *) destptr;
|
|
while (bytecount > 3) {
|
|
*to32ptr++ = *from32ptr++;
|
|
bytecount -= 4;
|
|
}
|
|
to8ptr = (u8 *) to32ptr;
|
|
|
|
alignbuffer = *from32ptr++;
|
|
from8ptr = (u8 *) &alignbuffer;
|
|
|
|
for (i = 0; i < bytecount; i++)
|
|
*to8ptr++ = *from8ptr++;
|
|
}
|
|
|
|
static void xemaclite_alignedwrite(void *srcptr, u32 destptr, u32 bytecount)
|
|
{
|
|
u32 i;
|
|
u32 alignbuffer;
|
|
u32 *to32ptr = (u32 *) destptr;
|
|
u32 *from32ptr;
|
|
u8 *to8ptr;
|
|
u8 *from8ptr;
|
|
|
|
from32ptr = (u32 *) srcptr;
|
|
while (bytecount > 3) {
|
|
|
|
*to32ptr++ = *from32ptr++;
|
|
bytecount -= 4;
|
|
}
|
|
|
|
alignbuffer = 0;
|
|
to8ptr = (u8 *) &alignbuffer;
|
|
from8ptr = (u8 *) from32ptr;
|
|
|
|
for (i = 0; i < bytecount; i++)
|
|
*to8ptr++ = *from8ptr++;
|
|
|
|
*to32ptr++ = alignbuffer;
|
|
}
|
|
|
|
static void emaclite_halt(struct eth_device *dev)
|
|
{
|
|
debug("eth_halt\n");
|
|
}
|
|
|
|
static int emaclite_init(struct eth_device *dev, bd_t *bis)
|
|
{
|
|
struct xemaclite *emaclite = dev->priv;
|
|
debug("EmacLite Initialization Started\n");
|
|
|
|
/*
|
|
* TX - TX_PING & TX_PONG initialization
|
|
*/
|
|
/* Restart PING TX */
|
|
out_be32 (dev->iobase + XEL_TSR_OFFSET, 0);
|
|
/* Copy MAC address */
|
|
xemaclite_alignedwrite(dev->enetaddr, dev->iobase, ENET_ADDR_LENGTH);
|
|
/* Set the length */
|
|
out_be32 (dev->iobase + XEL_TPLR_OFFSET, ENET_ADDR_LENGTH);
|
|
/* Update the MAC address in the EMAC Lite */
|
|
out_be32 (dev->iobase + XEL_TSR_OFFSET, XEL_TSR_PROG_MAC_ADDR);
|
|
/* Wait for EMAC Lite to finish with the MAC address update */
|
|
while ((in_be32 (dev->iobase + XEL_TSR_OFFSET) &
|
|
XEL_TSR_PROG_MAC_ADDR) != 0)
|
|
;
|
|
|
|
if (emaclite->txpp) {
|
|
/* The same operation with PONG TX */
|
|
out_be32 (dev->iobase + XEL_TSR_OFFSET + XEL_BUFFER_OFFSET, 0);
|
|
xemaclite_alignedwrite(dev->enetaddr, dev->iobase +
|
|
XEL_BUFFER_OFFSET, ENET_ADDR_LENGTH);
|
|
out_be32 (dev->iobase + XEL_TPLR_OFFSET, ENET_ADDR_LENGTH);
|
|
out_be32 (dev->iobase + XEL_TSR_OFFSET + XEL_BUFFER_OFFSET,
|
|
XEL_TSR_PROG_MAC_ADDR);
|
|
while ((in_be32 (dev->iobase + XEL_TSR_OFFSET +
|
|
XEL_BUFFER_OFFSET) & XEL_TSR_PROG_MAC_ADDR) != 0)
|
|
;
|
|
}
|
|
|
|
/*
|
|
* RX - RX_PING & RX_PONG initialization
|
|
*/
|
|
/* Write out the value to flush the RX buffer */
|
|
out_be32 (dev->iobase + XEL_RSR_OFFSET, XEL_RSR_RECV_IE_MASK);
|
|
|
|
if (emaclite->rxpp)
|
|
out_be32 (dev->iobase + XEL_RSR_OFFSET + XEL_BUFFER_OFFSET,
|
|
XEL_RSR_RECV_IE_MASK);
|
|
|
|
debug("EmacLite Initialization complete\n");
|
|
return 0;
|
|
}
|
|
|
|
static int xemaclite_txbufferavailable(struct eth_device *dev)
|
|
{
|
|
u32 reg;
|
|
u32 txpingbusy;
|
|
u32 txpongbusy;
|
|
struct xemaclite *emaclite = dev->priv;
|
|
|
|
/*
|
|
* Read the other buffer register
|
|
* and determine if the other buffer is available
|
|
*/
|
|
reg = in_be32 (dev->iobase +
|
|
emaclite->nexttxbuffertouse + 0);
|
|
txpingbusy = ((reg & XEL_TSR_XMIT_BUSY_MASK) ==
|
|
XEL_TSR_XMIT_BUSY_MASK);
|
|
|
|
reg = in_be32 (dev->iobase +
|
|
(emaclite->nexttxbuffertouse ^ XEL_TSR_OFFSET) + 0);
|
|
txpongbusy = ((reg & XEL_TSR_XMIT_BUSY_MASK) ==
|
|
XEL_TSR_XMIT_BUSY_MASK);
|
|
|
|
return !(txpingbusy && txpongbusy);
|
|
}
|
|
|
|
static int emaclite_send(struct eth_device *dev, void *ptr, int len)
|
|
{
|
|
u32 reg;
|
|
u32 baseaddress;
|
|
struct xemaclite *emaclite = dev->priv;
|
|
|
|
u32 maxtry = 1000;
|
|
|
|
if (len > PKTSIZE)
|
|
len = PKTSIZE;
|
|
|
|
while (!xemaclite_txbufferavailable(dev) && maxtry) {
|
|
udelay(10);
|
|
maxtry--;
|
|
}
|
|
|
|
if (!maxtry) {
|
|
printf("Error: Timeout waiting for ethernet TX buffer\n");
|
|
/* Restart PING TX */
|
|
out_be32 (dev->iobase + XEL_TSR_OFFSET, 0);
|
|
if (emaclite->txpp) {
|
|
out_be32 (dev->iobase + XEL_TSR_OFFSET +
|
|
XEL_BUFFER_OFFSET, 0);
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
/* Determine the expected TX buffer address */
|
|
baseaddress = (dev->iobase + emaclite->nexttxbuffertouse);
|
|
|
|
/* Determine if the expected buffer address is empty */
|
|
reg = in_be32 (baseaddress + XEL_TSR_OFFSET);
|
|
if (((reg & XEL_TSR_XMIT_BUSY_MASK) == 0)
|
|
&& ((in_be32 ((baseaddress) + XEL_TSR_OFFSET)
|
|
& XEL_TSR_XMIT_ACTIVE_MASK) == 0)) {
|
|
|
|
if (emaclite->txpp)
|
|
emaclite->nexttxbuffertouse ^= XEL_BUFFER_OFFSET;
|
|
|
|
debug("Send packet from 0x%x\n", baseaddress);
|
|
/* Write the frame to the buffer */
|
|
xemaclite_alignedwrite(ptr, baseaddress, len);
|
|
out_be32 (baseaddress + XEL_TPLR_OFFSET,(len &
|
|
(XEL_TPLR_LENGTH_MASK_HI | XEL_TPLR_LENGTH_MASK_LO)));
|
|
reg = in_be32 (baseaddress + XEL_TSR_OFFSET);
|
|
reg |= XEL_TSR_XMIT_BUSY_MASK;
|
|
if ((reg & XEL_TSR_XMIT_IE_MASK) != 0)
|
|
reg |= XEL_TSR_XMIT_ACTIVE_MASK;
|
|
out_be32 (baseaddress + XEL_TSR_OFFSET, reg);
|
|
return 0;
|
|
}
|
|
|
|
if (emaclite->txpp) {
|
|
/* Switch to second buffer */
|
|
baseaddress ^= XEL_BUFFER_OFFSET;
|
|
/* Determine if the expected buffer address is empty */
|
|
reg = in_be32 (baseaddress + XEL_TSR_OFFSET);
|
|
if (((reg & XEL_TSR_XMIT_BUSY_MASK) == 0)
|
|
&& ((in_be32 ((baseaddress) + XEL_TSR_OFFSET)
|
|
& XEL_TSR_XMIT_ACTIVE_MASK) == 0)) {
|
|
debug("Send packet from 0x%x\n", baseaddress);
|
|
/* Write the frame to the buffer */
|
|
xemaclite_alignedwrite(ptr, baseaddress, len);
|
|
out_be32 (baseaddress + XEL_TPLR_OFFSET, (len &
|
|
(XEL_TPLR_LENGTH_MASK_HI |
|
|
XEL_TPLR_LENGTH_MASK_LO)));
|
|
reg = in_be32 (baseaddress + XEL_TSR_OFFSET);
|
|
reg |= XEL_TSR_XMIT_BUSY_MASK;
|
|
if ((reg & XEL_TSR_XMIT_IE_MASK) != 0)
|
|
reg |= XEL_TSR_XMIT_ACTIVE_MASK;
|
|
out_be32 (baseaddress + XEL_TSR_OFFSET, reg);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
puts("Error while sending frame\n");
|
|
return -1;
|
|
}
|
|
|
|
static int emaclite_recv(struct eth_device *dev)
|
|
{
|
|
u32 length;
|
|
u32 reg;
|
|
u32 baseaddress;
|
|
struct xemaclite *emaclite = dev->priv;
|
|
|
|
baseaddress = dev->iobase + emaclite->nextrxbuffertouse;
|
|
reg = in_be32 (baseaddress + XEL_RSR_OFFSET);
|
|
debug("Testing data at address 0x%x\n", baseaddress);
|
|
if ((reg & XEL_RSR_RECV_DONE_MASK) == XEL_RSR_RECV_DONE_MASK) {
|
|
if (emaclite->rxpp)
|
|
emaclite->nextrxbuffertouse ^= XEL_BUFFER_OFFSET;
|
|
} else {
|
|
|
|
if (!emaclite->rxpp) {
|
|
debug("No data was available - address 0x%x\n",
|
|
baseaddress);
|
|
return 0;
|
|
} else {
|
|
baseaddress ^= XEL_BUFFER_OFFSET;
|
|
reg = in_be32 (baseaddress + XEL_RSR_OFFSET);
|
|
if ((reg & XEL_RSR_RECV_DONE_MASK) !=
|
|
XEL_RSR_RECV_DONE_MASK) {
|
|
debug("No data was available - address 0x%x\n",
|
|
baseaddress);
|
|
return 0;
|
|
}
|
|
}
|
|
}
|
|
/* Get the length of the frame that arrived */
|
|
switch(((ntohl(in_be32 (baseaddress + XEL_RXBUFF_OFFSET + 0xC))) &
|
|
0xFFFF0000 ) >> 16) {
|
|
case 0x806:
|
|
length = 42 + 20; /* FIXME size of ARP */
|
|
debug("ARP Packet\n");
|
|
break;
|
|
case 0x800:
|
|
length = 14 + 14 +
|
|
(((ntohl(in_be32 (baseaddress + XEL_RXBUFF_OFFSET +
|
|
0x10))) & 0xFFFF0000) >> 16);
|
|
/* FIXME size of IP packet */
|
|
debug ("IP Packet\n");
|
|
break;
|
|
default:
|
|
debug("Other Packet\n");
|
|
length = PKTSIZE;
|
|
break;
|
|
}
|
|
|
|
xemaclite_alignedread((u32 *) (baseaddress + XEL_RXBUFF_OFFSET),
|
|
etherrxbuff, length);
|
|
|
|
/* Acknowledge the frame */
|
|
reg = in_be32 (baseaddress + XEL_RSR_OFFSET);
|
|
reg &= ~XEL_RSR_RECV_DONE_MASK;
|
|
out_be32 (baseaddress + XEL_RSR_OFFSET, reg);
|
|
|
|
debug("Packet receive from 0x%x, length %dB\n", baseaddress, length);
|
|
net_process_received_packet((uchar *)etherrxbuff, length);
|
|
return length;
|
|
|
|
}
|
|
|
|
int xilinx_emaclite_initialize(bd_t *bis, unsigned long base_addr,
|
|
int txpp, int rxpp)
|
|
{
|
|
struct eth_device *dev;
|
|
struct xemaclite *emaclite;
|
|
|
|
dev = calloc(1, sizeof(*dev));
|
|
if (dev == NULL)
|
|
return -1;
|
|
|
|
emaclite = calloc(1, sizeof(struct xemaclite));
|
|
if (emaclite == NULL) {
|
|
free(dev);
|
|
return -1;
|
|
}
|
|
|
|
dev->priv = emaclite;
|
|
|
|
emaclite->txpp = txpp;
|
|
emaclite->rxpp = rxpp;
|
|
|
|
sprintf(dev->name, "Xelite.%lx", base_addr);
|
|
|
|
dev->iobase = base_addr;
|
|
dev->init = emaclite_init;
|
|
dev->halt = emaclite_halt;
|
|
dev->send = emaclite_send;
|
|
dev->recv = emaclite_recv;
|
|
|
|
eth_register(dev);
|
|
|
|
return 1;
|
|
}
|
|
|
|
#ifdef CONFIG_OF_CONTROL
|
|
int xilinx_emaclite_of_init(const void *blob)
|
|
{
|
|
int offset = 0;
|
|
u32 ret = 0;
|
|
u32 reg;
|
|
|
|
do {
|
|
offset = fdt_node_offset_by_compatible(blob, offset,
|
|
"xlnx,xps-ethernetlite-1.00.a");
|
|
if (offset != -1) {
|
|
reg = fdtdec_get_addr(blob, offset, "reg");
|
|
if (reg != FDT_ADDR_T_NONE) {
|
|
u32 rxpp = fdtdec_get_int(blob, offset,
|
|
"xlnx,rx-ping-pong", 0);
|
|
u32 txpp = fdtdec_get_int(blob, offset,
|
|
"xlnx,tx-ping-pong", 0);
|
|
ret |= xilinx_emaclite_initialize(NULL, reg,
|
|
txpp, rxpp);
|
|
} else {
|
|
debug("EMACLITE: Can't get base address\n");
|
|
return -1;
|
|
}
|
|
}
|
|
} while (offset != -1);
|
|
|
|
return ret;
|
|
}
|
|
#endif
|