1f6569884SThomas Chou /* 2f6569884SThomas Chou * Opencore 10/100 ethernet mac driver 3f6569884SThomas Chou * 4f6569884SThomas Chou * Copyright (C) 2007-2008 Avionic Design Development GmbH 5f6569884SThomas Chou * Copyright (C) 2008-2009 Avionic Design GmbH 6f6569884SThomas Chou * Thierry Reding <thierry.reding@avionic-design.de> 7f6569884SThomas Chou * Copyright (C) 2010 Thomas Chou <thomas@wytron.com.tw> 85d43feabSMax Filippov * Copyright (C) 2016 Cadence Design Systems Inc. 9f6569884SThomas Chou * 105d43feabSMax Filippov * SPDX-License-Identifier: GPL-2.0 11f6569884SThomas Chou */ 12f6569884SThomas Chou 13f6569884SThomas Chou #include <common.h> 145d43feabSMax Filippov #include <dm/device.h> 155d43feabSMax Filippov #include <dm/platform_data/net_ethoc.h> 16a84a757aSMax Filippov #include <linux/io.h> 17f6569884SThomas Chou #include <malloc.h> 18f6569884SThomas Chou #include <net.h> 19f6569884SThomas Chou #include <miiphy.h> 20f6569884SThomas Chou #include <asm/cache.h> 21f6569884SThomas Chou 22f6569884SThomas Chou /* register offsets */ 23f6569884SThomas Chou #define MODER 0x00 24f6569884SThomas Chou #define INT_SOURCE 0x04 25f6569884SThomas Chou #define INT_MASK 0x08 26f6569884SThomas Chou #define IPGT 0x0c 27f6569884SThomas Chou #define IPGR1 0x10 28f6569884SThomas Chou #define IPGR2 0x14 29f6569884SThomas Chou #define PACKETLEN 0x18 30f6569884SThomas Chou #define COLLCONF 0x1c 31f6569884SThomas Chou #define TX_BD_NUM 0x20 32f6569884SThomas Chou #define CTRLMODER 0x24 33f6569884SThomas Chou #define MIIMODER 0x28 34f6569884SThomas Chou #define MIICOMMAND 0x2c 35f6569884SThomas Chou #define MIIADDRESS 0x30 36f6569884SThomas Chou #define MIITX_DATA 0x34 37f6569884SThomas Chou #define MIIRX_DATA 0x38 38f6569884SThomas Chou #define MIISTATUS 0x3c 39f6569884SThomas Chou #define MAC_ADDR0 0x40 40f6569884SThomas Chou #define MAC_ADDR1 0x44 41f6569884SThomas Chou #define ETH_HASH0 0x48 42f6569884SThomas Chou #define ETH_HASH1 0x4c 43f6569884SThomas Chou #define ETH_TXCTRL 0x50 44f6569884SThomas Chou 45f6569884SThomas Chou /* mode register */ 46f6569884SThomas Chou #define MODER_RXEN (1 << 0) /* receive enable */ 47f6569884SThomas Chou #define MODER_TXEN (1 << 1) /* transmit enable */ 48f6569884SThomas Chou #define MODER_NOPRE (1 << 2) /* no preamble */ 49f6569884SThomas Chou #define MODER_BRO (1 << 3) /* broadcast address */ 50f6569884SThomas Chou #define MODER_IAM (1 << 4) /* individual address mode */ 51f6569884SThomas Chou #define MODER_PRO (1 << 5) /* promiscuous mode */ 52f6569884SThomas Chou #define MODER_IFG (1 << 6) /* interframe gap for incoming frames */ 53f6569884SThomas Chou #define MODER_LOOP (1 << 7) /* loopback */ 54f6569884SThomas Chou #define MODER_NBO (1 << 8) /* no back-off */ 55f6569884SThomas Chou #define MODER_EDE (1 << 9) /* excess defer enable */ 56f6569884SThomas Chou #define MODER_FULLD (1 << 10) /* full duplex */ 57f6569884SThomas Chou #define MODER_RESET (1 << 11) /* FIXME: reset (undocumented) */ 58f6569884SThomas Chou #define MODER_DCRC (1 << 12) /* delayed CRC enable */ 59f6569884SThomas Chou #define MODER_CRC (1 << 13) /* CRC enable */ 60f6569884SThomas Chou #define MODER_HUGE (1 << 14) /* huge packets enable */ 61f6569884SThomas Chou #define MODER_PAD (1 << 15) /* padding enabled */ 62f6569884SThomas Chou #define MODER_RSM (1 << 16) /* receive small packets */ 63f6569884SThomas Chou 64f6569884SThomas Chou /* interrupt source and mask registers */ 65f6569884SThomas Chou #define INT_MASK_TXF (1 << 0) /* transmit frame */ 66f6569884SThomas Chou #define INT_MASK_TXE (1 << 1) /* transmit error */ 67f6569884SThomas Chou #define INT_MASK_RXF (1 << 2) /* receive frame */ 68f6569884SThomas Chou #define INT_MASK_RXE (1 << 3) /* receive error */ 69f6569884SThomas Chou #define INT_MASK_BUSY (1 << 4) 70f6569884SThomas Chou #define INT_MASK_TXC (1 << 5) /* transmit control frame */ 71f6569884SThomas Chou #define INT_MASK_RXC (1 << 6) /* receive control frame */ 72f6569884SThomas Chou 73f6569884SThomas Chou #define INT_MASK_TX (INT_MASK_TXF | INT_MASK_TXE) 74f6569884SThomas Chou #define INT_MASK_RX (INT_MASK_RXF | INT_MASK_RXE) 75f6569884SThomas Chou 76f6569884SThomas Chou #define INT_MASK_ALL ( \ 77f6569884SThomas Chou INT_MASK_TXF | INT_MASK_TXE | \ 78f6569884SThomas Chou INT_MASK_RXF | INT_MASK_RXE | \ 79f6569884SThomas Chou INT_MASK_TXC | INT_MASK_RXC | \ 80f6569884SThomas Chou INT_MASK_BUSY \ 81f6569884SThomas Chou ) 82f6569884SThomas Chou 83f6569884SThomas Chou /* packet length register */ 84f6569884SThomas Chou #define PACKETLEN_MIN(min) (((min) & 0xffff) << 16) 85f6569884SThomas Chou #define PACKETLEN_MAX(max) (((max) & 0xffff) << 0) 86f6569884SThomas Chou #define PACKETLEN_MIN_MAX(min, max) (PACKETLEN_MIN(min) | \ 87f6569884SThomas Chou PACKETLEN_MAX(max)) 88f6569884SThomas Chou 89f6569884SThomas Chou /* transmit buffer number register */ 90f6569884SThomas Chou #define TX_BD_NUM_VAL(x) (((x) <= 0x80) ? (x) : 0x80) 91f6569884SThomas Chou 92f6569884SThomas Chou /* control module mode register */ 93f6569884SThomas Chou #define CTRLMODER_PASSALL (1 << 0) /* pass all receive frames */ 94f6569884SThomas Chou #define CTRLMODER_RXFLOW (1 << 1) /* receive control flow */ 95f6569884SThomas Chou #define CTRLMODER_TXFLOW (1 << 2) /* transmit control flow */ 96f6569884SThomas Chou 97f6569884SThomas Chou /* MII mode register */ 98f6569884SThomas Chou #define MIIMODER_CLKDIV(x) ((x) & 0xfe) /* needs to be an even number */ 99f6569884SThomas Chou #define MIIMODER_NOPRE (1 << 8) /* no preamble */ 100f6569884SThomas Chou 101f6569884SThomas Chou /* MII command register */ 102f6569884SThomas Chou #define MIICOMMAND_SCAN (1 << 0) /* scan status */ 103f6569884SThomas Chou #define MIICOMMAND_READ (1 << 1) /* read status */ 104f6569884SThomas Chou #define MIICOMMAND_WRITE (1 << 2) /* write control data */ 105f6569884SThomas Chou 106f6569884SThomas Chou /* MII address register */ 107f6569884SThomas Chou #define MIIADDRESS_FIAD(x) (((x) & 0x1f) << 0) 108f6569884SThomas Chou #define MIIADDRESS_RGAD(x) (((x) & 0x1f) << 8) 109f6569884SThomas Chou #define MIIADDRESS_ADDR(phy, reg) (MIIADDRESS_FIAD(phy) | \ 110f6569884SThomas Chou MIIADDRESS_RGAD(reg)) 111f6569884SThomas Chou 112f6569884SThomas Chou /* MII transmit data register */ 113f6569884SThomas Chou #define MIITX_DATA_VAL(x) ((x) & 0xffff) 114f6569884SThomas Chou 115f6569884SThomas Chou /* MII receive data register */ 116f6569884SThomas Chou #define MIIRX_DATA_VAL(x) ((x) & 0xffff) 117f6569884SThomas Chou 118f6569884SThomas Chou /* MII status register */ 119f6569884SThomas Chou #define MIISTATUS_LINKFAIL (1 << 0) 120f6569884SThomas Chou #define MIISTATUS_BUSY (1 << 1) 121f6569884SThomas Chou #define MIISTATUS_INVALID (1 << 2) 122f6569884SThomas Chou 123f6569884SThomas Chou /* TX buffer descriptor */ 124f6569884SThomas Chou #define TX_BD_CS (1 << 0) /* carrier sense lost */ 125f6569884SThomas Chou #define TX_BD_DF (1 << 1) /* defer indication */ 126f6569884SThomas Chou #define TX_BD_LC (1 << 2) /* late collision */ 127f6569884SThomas Chou #define TX_BD_RL (1 << 3) /* retransmission limit */ 128f6569884SThomas Chou #define TX_BD_RETRY_MASK (0x00f0) 129f6569884SThomas Chou #define TX_BD_RETRY(x) (((x) & 0x00f0) >> 4) 130f6569884SThomas Chou #define TX_BD_UR (1 << 8) /* transmitter underrun */ 131f6569884SThomas Chou #define TX_BD_CRC (1 << 11) /* TX CRC enable */ 132f6569884SThomas Chou #define TX_BD_PAD (1 << 12) /* pad enable */ 133f6569884SThomas Chou #define TX_BD_WRAP (1 << 13) 134f6569884SThomas Chou #define TX_BD_IRQ (1 << 14) /* interrupt request enable */ 135f6569884SThomas Chou #define TX_BD_READY (1 << 15) /* TX buffer ready */ 136f6569884SThomas Chou #define TX_BD_LEN(x) (((x) & 0xffff) << 16) 137f6569884SThomas Chou #define TX_BD_LEN_MASK (0xffff << 16) 138f6569884SThomas Chou 139f6569884SThomas Chou #define TX_BD_STATS (TX_BD_CS | TX_BD_DF | TX_BD_LC | \ 140f6569884SThomas Chou TX_BD_RL | TX_BD_RETRY_MASK | TX_BD_UR) 141f6569884SThomas Chou 142f6569884SThomas Chou /* RX buffer descriptor */ 143f6569884SThomas Chou #define RX_BD_LC (1 << 0) /* late collision */ 144f6569884SThomas Chou #define RX_BD_CRC (1 << 1) /* RX CRC error */ 145f6569884SThomas Chou #define RX_BD_SF (1 << 2) /* short frame */ 146f6569884SThomas Chou #define RX_BD_TL (1 << 3) /* too long */ 147f6569884SThomas Chou #define RX_BD_DN (1 << 4) /* dribble nibble */ 148f6569884SThomas Chou #define RX_BD_IS (1 << 5) /* invalid symbol */ 149f6569884SThomas Chou #define RX_BD_OR (1 << 6) /* receiver overrun */ 150f6569884SThomas Chou #define RX_BD_MISS (1 << 7) 151f6569884SThomas Chou #define RX_BD_CF (1 << 8) /* control frame */ 152f6569884SThomas Chou #define RX_BD_WRAP (1 << 13) 153f6569884SThomas Chou #define RX_BD_IRQ (1 << 14) /* interrupt request enable */ 154f6569884SThomas Chou #define RX_BD_EMPTY (1 << 15) 155f6569884SThomas Chou #define RX_BD_LEN(x) (((x) & 0xffff) << 16) 156f6569884SThomas Chou 157f6569884SThomas Chou #define RX_BD_STATS (RX_BD_LC | RX_BD_CRC | RX_BD_SF | RX_BD_TL | \ 158f6569884SThomas Chou RX_BD_DN | RX_BD_IS | RX_BD_OR | RX_BD_MISS) 159f6569884SThomas Chou 160f6569884SThomas Chou #define ETHOC_BUFSIZ 1536 161f6569884SThomas Chou #define ETHOC_ZLEN 64 162f6569884SThomas Chou #define ETHOC_BD_BASE 0x400 163f6569884SThomas Chou #define ETHOC_TIMEOUT (HZ / 2) 164f6569884SThomas Chou #define ETHOC_MII_TIMEOUT (1 + (HZ / 5)) 165a84a757aSMax Filippov #define ETHOC_IOSIZE 0x54 166f6569884SThomas Chou 167f6569884SThomas Chou /** 168f6569884SThomas Chou * struct ethoc - driver-private device structure 169f6569884SThomas Chou * @num_tx: number of send buffers 170f6569884SThomas Chou * @cur_tx: last send buffer written 171f6569884SThomas Chou * @dty_tx: last buffer actually sent 172f6569884SThomas Chou * @num_rx: number of receive buffers 173f6569884SThomas Chou * @cur_rx: current receive buffer 174f6569884SThomas Chou */ 175f6569884SThomas Chou struct ethoc { 176f6569884SThomas Chou u32 num_tx; 177f6569884SThomas Chou u32 cur_tx; 178f6569884SThomas Chou u32 dty_tx; 179f6569884SThomas Chou u32 num_rx; 180f6569884SThomas Chou u32 cur_rx; 181a84a757aSMax Filippov void __iomem *iobase; 182*59b7dfa0SMax Filippov void __iomem *packet; 183*59b7dfa0SMax Filippov phys_addr_t packet_phys; 184f6569884SThomas Chou }; 185f6569884SThomas Chou 186f6569884SThomas Chou /** 187f6569884SThomas Chou * struct ethoc_bd - buffer descriptor 188f6569884SThomas Chou * @stat: buffer statistics 189f6569884SThomas Chou * @addr: physical memory address 190f6569884SThomas Chou */ 191f6569884SThomas Chou struct ethoc_bd { 192f6569884SThomas Chou u32 stat; 193f6569884SThomas Chou u32 addr; 194f6569884SThomas Chou }; 195f6569884SThomas Chou 196a84a757aSMax Filippov static inline u32 ethoc_read(struct ethoc *priv, size_t offset) 197f6569884SThomas Chou { 198a84a757aSMax Filippov return readl(priv->iobase + offset); 199f6569884SThomas Chou } 200f6569884SThomas Chou 201a84a757aSMax Filippov static inline void ethoc_write(struct ethoc *priv, size_t offset, u32 data) 202f6569884SThomas Chou { 203a84a757aSMax Filippov writel(data, priv->iobase + offset); 204f6569884SThomas Chou } 205f6569884SThomas Chou 206a84a757aSMax Filippov static inline void ethoc_read_bd(struct ethoc *priv, int index, 207f6569884SThomas Chou struct ethoc_bd *bd) 208f6569884SThomas Chou { 2099f680d2dSVasili Galka size_t offset = ETHOC_BD_BASE + (index * sizeof(struct ethoc_bd)); 210a84a757aSMax Filippov bd->stat = ethoc_read(priv, offset + 0); 211a84a757aSMax Filippov bd->addr = ethoc_read(priv, offset + 4); 212f6569884SThomas Chou } 213f6569884SThomas Chou 214a84a757aSMax Filippov static inline void ethoc_write_bd(struct ethoc *priv, int index, 215f6569884SThomas Chou const struct ethoc_bd *bd) 216f6569884SThomas Chou { 2179f680d2dSVasili Galka size_t offset = ETHOC_BD_BASE + (index * sizeof(struct ethoc_bd)); 218a84a757aSMax Filippov ethoc_write(priv, offset + 0, bd->stat); 219a84a757aSMax Filippov ethoc_write(priv, offset + 4, bd->addr); 220f6569884SThomas Chou } 221f6569884SThomas Chou 2225d43feabSMax Filippov static int ethoc_write_hwaddr_common(struct ethoc *priv, u8 *mac) 223f6569884SThomas Chou { 224a84a757aSMax Filippov ethoc_write(priv, MAC_ADDR0, (mac[2] << 24) | (mac[3] << 16) | 225f6569884SThomas Chou (mac[4] << 8) | (mac[5] << 0)); 226a84a757aSMax Filippov ethoc_write(priv, MAC_ADDR1, (mac[0] << 8) | (mac[1] << 0)); 2273ac9d6c6SThomas Chou return 0; 228f6569884SThomas Chou } 229f6569884SThomas Chou 230a84a757aSMax Filippov static inline void ethoc_ack_irq(struct ethoc *priv, u32 mask) 231f6569884SThomas Chou { 232a84a757aSMax Filippov ethoc_write(priv, INT_SOURCE, mask); 233f6569884SThomas Chou } 234f6569884SThomas Chou 235a84a757aSMax Filippov static inline void ethoc_enable_rx_and_tx(struct ethoc *priv) 236f6569884SThomas Chou { 237a84a757aSMax Filippov u32 mode = ethoc_read(priv, MODER); 238f6569884SThomas Chou mode |= MODER_RXEN | MODER_TXEN; 239a84a757aSMax Filippov ethoc_write(priv, MODER, mode); 240f6569884SThomas Chou } 241f6569884SThomas Chou 242a84a757aSMax Filippov static inline void ethoc_disable_rx_and_tx(struct ethoc *priv) 243f6569884SThomas Chou { 244a84a757aSMax Filippov u32 mode = ethoc_read(priv, MODER); 245f6569884SThomas Chou mode &= ~(MODER_RXEN | MODER_TXEN); 246a84a757aSMax Filippov ethoc_write(priv, MODER, mode); 247f6569884SThomas Chou } 248f6569884SThomas Chou 249a84a757aSMax Filippov static int ethoc_init_ring(struct ethoc *priv) 250f6569884SThomas Chou { 251f6569884SThomas Chou struct ethoc_bd bd; 252*59b7dfa0SMax Filippov phys_addr_t addr = priv->packet_phys; 253f6569884SThomas Chou int i; 254f6569884SThomas Chou 255f6569884SThomas Chou priv->cur_tx = 0; 256f6569884SThomas Chou priv->dty_tx = 0; 257f6569884SThomas Chou priv->cur_rx = 0; 258f6569884SThomas Chou 259f6569884SThomas Chou /* setup transmission buffers */ 260f6569884SThomas Chou bd.stat = TX_BD_IRQ | TX_BD_CRC; 26102a888b5SMax Filippov bd.addr = 0; 262f6569884SThomas Chou 263f6569884SThomas Chou for (i = 0; i < priv->num_tx; i++) { 264*59b7dfa0SMax Filippov if (addr) { 265*59b7dfa0SMax Filippov bd.addr = addr; 266*59b7dfa0SMax Filippov addr += PKTSIZE_ALIGN; 267*59b7dfa0SMax Filippov } 268f6569884SThomas Chou if (i == priv->num_tx - 1) 269f6569884SThomas Chou bd.stat |= TX_BD_WRAP; 270f6569884SThomas Chou 271a84a757aSMax Filippov ethoc_write_bd(priv, i, &bd); 272f6569884SThomas Chou } 273f6569884SThomas Chou 274f6569884SThomas Chou bd.stat = RX_BD_EMPTY | RX_BD_IRQ; 275f6569884SThomas Chou 276f6569884SThomas Chou for (i = 0; i < priv->num_rx; i++) { 277*59b7dfa0SMax Filippov if (addr) { 278*59b7dfa0SMax Filippov bd.addr = addr; 279*59b7dfa0SMax Filippov addr += PKTSIZE_ALIGN; 280*59b7dfa0SMax Filippov } else { 28102a888b5SMax Filippov bd.addr = virt_to_phys(net_rx_packets[i]); 282*59b7dfa0SMax Filippov } 283f6569884SThomas Chou if (i == priv->num_rx - 1) 284f6569884SThomas Chou bd.stat |= RX_BD_WRAP; 285f6569884SThomas Chou 28602a888b5SMax Filippov flush_dcache_range((ulong)net_rx_packets[i], 28702a888b5SMax Filippov (ulong)net_rx_packets[i] + PKTSIZE_ALIGN); 288a84a757aSMax Filippov ethoc_write_bd(priv, priv->num_tx + i, &bd); 289f6569884SThomas Chou } 290f6569884SThomas Chou 291f6569884SThomas Chou return 0; 292f6569884SThomas Chou } 293f6569884SThomas Chou 294a84a757aSMax Filippov static int ethoc_reset(struct ethoc *priv) 295f6569884SThomas Chou { 296f6569884SThomas Chou u32 mode; 297f6569884SThomas Chou 298f6569884SThomas Chou /* TODO: reset controller? */ 299f6569884SThomas Chou 300a84a757aSMax Filippov ethoc_disable_rx_and_tx(priv); 301f6569884SThomas Chou 302f6569884SThomas Chou /* TODO: setup registers */ 303f6569884SThomas Chou 304f6569884SThomas Chou /* enable FCS generation and automatic padding */ 305a84a757aSMax Filippov mode = ethoc_read(priv, MODER); 306f6569884SThomas Chou mode |= MODER_CRC | MODER_PAD; 307a84a757aSMax Filippov ethoc_write(priv, MODER, mode); 308f6569884SThomas Chou 309f6569884SThomas Chou /* set full-duplex mode */ 310a84a757aSMax Filippov mode = ethoc_read(priv, MODER); 311f6569884SThomas Chou mode |= MODER_FULLD; 312a84a757aSMax Filippov ethoc_write(priv, MODER, mode); 313a84a757aSMax Filippov ethoc_write(priv, IPGT, 0x15); 314f6569884SThomas Chou 315a84a757aSMax Filippov ethoc_ack_irq(priv, INT_MASK_ALL); 316a84a757aSMax Filippov ethoc_enable_rx_and_tx(priv); 317f6569884SThomas Chou return 0; 318f6569884SThomas Chou } 319f6569884SThomas Chou 3205d43feabSMax Filippov static int ethoc_init_common(struct ethoc *priv) 321f6569884SThomas Chou { 322f6569884SThomas Chou priv->num_tx = 1; 323f6569884SThomas Chou priv->num_rx = PKTBUFSRX; 324a84a757aSMax Filippov ethoc_write(priv, TX_BD_NUM, priv->num_tx); 325a84a757aSMax Filippov ethoc_init_ring(priv); 326a84a757aSMax Filippov ethoc_reset(priv); 327f6569884SThomas Chou 328f6569884SThomas Chou return 0; 329f6569884SThomas Chou } 330f6569884SThomas Chou 331f6569884SThomas Chou static int ethoc_update_rx_stats(struct ethoc_bd *bd) 332f6569884SThomas Chou { 333f6569884SThomas Chou int ret = 0; 334f6569884SThomas Chou 335f6569884SThomas Chou if (bd->stat & RX_BD_TL) { 336f6569884SThomas Chou debug("ETHOC: " "RX: frame too long\n"); 337f6569884SThomas Chou ret++; 338f6569884SThomas Chou } 339f6569884SThomas Chou 340f6569884SThomas Chou if (bd->stat & RX_BD_SF) { 341f6569884SThomas Chou debug("ETHOC: " "RX: frame too short\n"); 342f6569884SThomas Chou ret++; 343f6569884SThomas Chou } 344f6569884SThomas Chou 345f6569884SThomas Chou if (bd->stat & RX_BD_DN) 346f6569884SThomas Chou debug("ETHOC: " "RX: dribble nibble\n"); 347f6569884SThomas Chou 348f6569884SThomas Chou if (bd->stat & RX_BD_CRC) { 349f6569884SThomas Chou debug("ETHOC: " "RX: wrong CRC\n"); 350f6569884SThomas Chou ret++; 351f6569884SThomas Chou } 352f6569884SThomas Chou 353f6569884SThomas Chou if (bd->stat & RX_BD_OR) { 354f6569884SThomas Chou debug("ETHOC: " "RX: overrun\n"); 355f6569884SThomas Chou ret++; 356f6569884SThomas Chou } 357f6569884SThomas Chou 358f6569884SThomas Chou if (bd->stat & RX_BD_LC) { 359f6569884SThomas Chou debug("ETHOC: " "RX: late collision\n"); 360f6569884SThomas Chou ret++; 361f6569884SThomas Chou } 362f6569884SThomas Chou 363f6569884SThomas Chou return ret; 364f6569884SThomas Chou } 365f6569884SThomas Chou 3665d43feabSMax Filippov static int ethoc_rx_common(struct ethoc *priv, uchar **packetp) 367f6569884SThomas Chou { 368f6569884SThomas Chou struct ethoc_bd bd; 36902a888b5SMax Filippov u32 i = priv->cur_rx % priv->num_rx; 37002a888b5SMax Filippov u32 entry = priv->num_tx + i; 371f6569884SThomas Chou 372a84a757aSMax Filippov ethoc_read_bd(priv, entry, &bd); 373f6569884SThomas Chou if (bd.stat & RX_BD_EMPTY) 3745d43feabSMax Filippov return -EAGAIN; 375f6569884SThomas Chou 376f6569884SThomas Chou debug("%s(): RX buffer %d, %x received\n", 377f6569884SThomas Chou __func__, priv->cur_rx, bd.stat); 378f6569884SThomas Chou if (ethoc_update_rx_stats(&bd) == 0) { 379f6569884SThomas Chou int size = bd.stat >> 16; 3805d43feabSMax Filippov 381f6569884SThomas Chou size -= 4; /* strip the CRC */ 382*59b7dfa0SMax Filippov if (priv->packet) 383*59b7dfa0SMax Filippov *packetp = priv->packet + entry * PKTSIZE_ALIGN; 384*59b7dfa0SMax Filippov else 38502a888b5SMax Filippov *packetp = net_rx_packets[i]; 3865d43feabSMax Filippov return size; 3875d43feabSMax Filippov } else { 3885d43feabSMax Filippov return 0; 3895d43feabSMax Filippov } 390f6569884SThomas Chou } 391f6569884SThomas Chou 3925d43feabSMax Filippov static int ethoc_is_new_packet_received(struct ethoc *priv) 3935d43feabSMax Filippov { 3945d43feabSMax Filippov u32 pending; 3955d43feabSMax Filippov 3965d43feabSMax Filippov pending = ethoc_read(priv, INT_SOURCE); 3975d43feabSMax Filippov ethoc_ack_irq(priv, pending); 3985d43feabSMax Filippov if (pending & INT_MASK_BUSY) 3995d43feabSMax Filippov debug("%s(): packet dropped\n", __func__); 4005d43feabSMax Filippov if (pending & INT_MASK_RX) { 4015d43feabSMax Filippov debug("%s(): rx irq\n", __func__); 4025d43feabSMax Filippov return 1; 403f6569884SThomas Chou } 404f6569884SThomas Chou 4055d43feabSMax Filippov return 0; 406f6569884SThomas Chou } 407f6569884SThomas Chou 408f6569884SThomas Chou static int ethoc_update_tx_stats(struct ethoc_bd *bd) 409f6569884SThomas Chou { 410f6569884SThomas Chou if (bd->stat & TX_BD_LC) 411f6569884SThomas Chou debug("ETHOC: " "TX: late collision\n"); 412f6569884SThomas Chou 413f6569884SThomas Chou if (bd->stat & TX_BD_RL) 414f6569884SThomas Chou debug("ETHOC: " "TX: retransmit limit\n"); 415f6569884SThomas Chou 416f6569884SThomas Chou if (bd->stat & TX_BD_UR) 417f6569884SThomas Chou debug("ETHOC: " "TX: underrun\n"); 418f6569884SThomas Chou 419f6569884SThomas Chou if (bd->stat & TX_BD_CS) 420f6569884SThomas Chou debug("ETHOC: " "TX: carrier sense lost\n"); 421f6569884SThomas Chou 422f6569884SThomas Chou return 0; 423f6569884SThomas Chou } 424f6569884SThomas Chou 425a84a757aSMax Filippov static void ethoc_tx(struct ethoc *priv) 426f6569884SThomas Chou { 427f6569884SThomas Chou u32 entry = priv->dty_tx % priv->num_tx; 428f6569884SThomas Chou struct ethoc_bd bd; 429f6569884SThomas Chou 430a84a757aSMax Filippov ethoc_read_bd(priv, entry, &bd); 431f6569884SThomas Chou if ((bd.stat & TX_BD_READY) == 0) 432f6569884SThomas Chou (void)ethoc_update_tx_stats(&bd); 433f6569884SThomas Chou } 434f6569884SThomas Chou 4355d43feabSMax Filippov static int ethoc_send_common(struct ethoc *priv, void *packet, int length) 436f6569884SThomas Chou { 437f6569884SThomas Chou struct ethoc_bd bd; 438f6569884SThomas Chou u32 entry; 439f6569884SThomas Chou u32 pending; 440f6569884SThomas Chou int tmo; 441f6569884SThomas Chou 442f6569884SThomas Chou entry = priv->cur_tx % priv->num_tx; 443a84a757aSMax Filippov ethoc_read_bd(priv, entry, &bd); 444f6569884SThomas Chou if (unlikely(length < ETHOC_ZLEN)) 445f6569884SThomas Chou bd.stat |= TX_BD_PAD; 446f6569884SThomas Chou else 447f6569884SThomas Chou bd.stat &= ~TX_BD_PAD; 448f6569884SThomas Chou 449*59b7dfa0SMax Filippov if (priv->packet) { 450*59b7dfa0SMax Filippov void *p = priv->packet + entry * PKTSIZE_ALIGN; 451*59b7dfa0SMax Filippov 452*59b7dfa0SMax Filippov memcpy(p, packet, length); 453*59b7dfa0SMax Filippov packet = p; 454*59b7dfa0SMax Filippov } else { 455*59b7dfa0SMax Filippov bd.addr = virt_to_phys(packet); 456*59b7dfa0SMax Filippov } 45702a888b5SMax Filippov flush_dcache_range((ulong)packet, (ulong)packet + length); 458f6569884SThomas Chou bd.stat &= ~(TX_BD_STATS | TX_BD_LEN_MASK); 459f6569884SThomas Chou bd.stat |= TX_BD_LEN(length); 460a84a757aSMax Filippov ethoc_write_bd(priv, entry, &bd); 461f6569884SThomas Chou 462f6569884SThomas Chou /* start transmit */ 463f6569884SThomas Chou bd.stat |= TX_BD_READY; 464a84a757aSMax Filippov ethoc_write_bd(priv, entry, &bd); 465f6569884SThomas Chou 466f6569884SThomas Chou /* wait for transfer to succeed */ 467f6569884SThomas Chou tmo = get_timer(0) + 5 * CONFIG_SYS_HZ; 468f6569884SThomas Chou while (1) { 469a84a757aSMax Filippov pending = ethoc_read(priv, INT_SOURCE); 470a84a757aSMax Filippov ethoc_ack_irq(priv, pending & ~INT_MASK_RX); 471f6569884SThomas Chou if (pending & INT_MASK_BUSY) 472f6569884SThomas Chou debug("%s(): packet dropped\n", __func__); 473f6569884SThomas Chou 474f6569884SThomas Chou if (pending & INT_MASK_TX) { 475a84a757aSMax Filippov ethoc_tx(priv); 476f6569884SThomas Chou break; 477f6569884SThomas Chou } 478f6569884SThomas Chou if (get_timer(0) >= tmo) { 479f6569884SThomas Chou debug("%s(): timed out\n", __func__); 480f6569884SThomas Chou return -1; 481f6569884SThomas Chou } 482f6569884SThomas Chou } 483f6569884SThomas Chou 484f6569884SThomas Chou debug("%s(): packet sent\n", __func__); 485f6569884SThomas Chou return 0; 486f6569884SThomas Chou } 487f6569884SThomas Chou 4885d43feabSMax Filippov static int ethoc_free_pkt_common(struct ethoc *priv) 4895d43feabSMax Filippov { 4905d43feabSMax Filippov struct ethoc_bd bd; 49102a888b5SMax Filippov u32 i = priv->cur_rx % priv->num_rx; 49202a888b5SMax Filippov u32 entry = priv->num_tx + i; 493*59b7dfa0SMax Filippov void *src; 4945d43feabSMax Filippov 4955d43feabSMax Filippov ethoc_read_bd(priv, entry, &bd); 4965d43feabSMax Filippov 497*59b7dfa0SMax Filippov if (priv->packet) 498*59b7dfa0SMax Filippov src = priv->packet + entry * PKTSIZE_ALIGN; 499*59b7dfa0SMax Filippov else 500*59b7dfa0SMax Filippov src = net_rx_packets[i]; 5015d43feabSMax Filippov /* clear the buffer descriptor so it can be reused */ 502*59b7dfa0SMax Filippov flush_dcache_range((ulong)src, 503*59b7dfa0SMax Filippov (ulong)src + PKTSIZE_ALIGN); 5045d43feabSMax Filippov bd.stat &= ~RX_BD_STATS; 5055d43feabSMax Filippov bd.stat |= RX_BD_EMPTY; 5065d43feabSMax Filippov ethoc_write_bd(priv, entry, &bd); 5075d43feabSMax Filippov priv->cur_rx++; 5085d43feabSMax Filippov 5095d43feabSMax Filippov return 0; 5105d43feabSMax Filippov } 5115d43feabSMax Filippov 5125d43feabSMax Filippov #ifdef CONFIG_DM_ETH 5135d43feabSMax Filippov 5145d43feabSMax Filippov static int ethoc_write_hwaddr(struct udevice *dev) 5155d43feabSMax Filippov { 5165d43feabSMax Filippov struct ethoc_eth_pdata *pdata = dev_get_platdata(dev); 5175d43feabSMax Filippov struct ethoc *priv = dev_get_priv(dev); 5185d43feabSMax Filippov u8 *mac = pdata->eth_pdata.enetaddr; 5195d43feabSMax Filippov 5205d43feabSMax Filippov return ethoc_write_hwaddr_common(priv, mac); 5215d43feabSMax Filippov } 5225d43feabSMax Filippov 5235d43feabSMax Filippov static int ethoc_send(struct udevice *dev, void *packet, int length) 5245d43feabSMax Filippov { 5255d43feabSMax Filippov return ethoc_send_common(dev_get_priv(dev), packet, length); 5265d43feabSMax Filippov } 5275d43feabSMax Filippov 5285d43feabSMax Filippov static int ethoc_free_pkt(struct udevice *dev, uchar *packet, int length) 5295d43feabSMax Filippov { 5305d43feabSMax Filippov return ethoc_free_pkt_common(dev_get_priv(dev)); 5315d43feabSMax Filippov } 5325d43feabSMax Filippov 5335d43feabSMax Filippov static int ethoc_recv(struct udevice *dev, int flags, uchar **packetp) 5345d43feabSMax Filippov { 5355d43feabSMax Filippov struct ethoc *priv = dev_get_priv(dev); 5365d43feabSMax Filippov 5375d43feabSMax Filippov if (flags & ETH_RECV_CHECK_DEVICE) 5385d43feabSMax Filippov if (!ethoc_is_new_packet_received(priv)) 5395d43feabSMax Filippov return -EAGAIN; 5405d43feabSMax Filippov 5415d43feabSMax Filippov return ethoc_rx_common(priv, packetp); 5425d43feabSMax Filippov } 5435d43feabSMax Filippov 5445d43feabSMax Filippov static int ethoc_start(struct udevice *dev) 5455d43feabSMax Filippov { 5465d43feabSMax Filippov return ethoc_init_common(dev_get_priv(dev)); 5475d43feabSMax Filippov } 5485d43feabSMax Filippov 5495d43feabSMax Filippov static void ethoc_stop(struct udevice *dev) 5505d43feabSMax Filippov { 5515d43feabSMax Filippov struct ethoc *priv = dev_get_priv(dev); 5525d43feabSMax Filippov 5535d43feabSMax Filippov ethoc_disable_rx_and_tx(priv); 5545d43feabSMax Filippov } 5555d43feabSMax Filippov 5562de18c8dSMax Filippov static int ethoc_ofdata_to_platdata(struct udevice *dev) 5572de18c8dSMax Filippov { 5582de18c8dSMax Filippov struct ethoc_eth_pdata *pdata = dev_get_platdata(dev); 559*59b7dfa0SMax Filippov fdt_addr_t addr; 5602de18c8dSMax Filippov 5612de18c8dSMax Filippov pdata->eth_pdata.iobase = dev_get_addr(dev); 562*59b7dfa0SMax Filippov addr = dev_get_addr_index(dev, 1); 563*59b7dfa0SMax Filippov if (addr != FDT_ADDR_T_NONE) 564*59b7dfa0SMax Filippov pdata->packet_base = addr; 5652de18c8dSMax Filippov return 0; 5662de18c8dSMax Filippov } 5672de18c8dSMax Filippov 5685d43feabSMax Filippov static int ethoc_probe(struct udevice *dev) 5695d43feabSMax Filippov { 5705d43feabSMax Filippov struct ethoc_eth_pdata *pdata = dev_get_platdata(dev); 5715d43feabSMax Filippov struct ethoc *priv = dev_get_priv(dev); 5725d43feabSMax Filippov 5735d43feabSMax Filippov priv->iobase = ioremap(pdata->eth_pdata.iobase, ETHOC_IOSIZE); 574*59b7dfa0SMax Filippov if (pdata->packet_base) { 575*59b7dfa0SMax Filippov priv->packet_phys = pdata->packet_base; 576*59b7dfa0SMax Filippov priv->packet = ioremap(pdata->packet_base, 577*59b7dfa0SMax Filippov (1 + PKTBUFSRX) * PKTSIZE_ALIGN); 578*59b7dfa0SMax Filippov } 5795d43feabSMax Filippov return 0; 5805d43feabSMax Filippov } 5815d43feabSMax Filippov 5825d43feabSMax Filippov static int ethoc_remove(struct udevice *dev) 5835d43feabSMax Filippov { 5845d43feabSMax Filippov struct ethoc *priv = dev_get_priv(dev); 5855d43feabSMax Filippov 5865d43feabSMax Filippov iounmap(priv->iobase); 5875d43feabSMax Filippov return 0; 5885d43feabSMax Filippov } 5895d43feabSMax Filippov 5905d43feabSMax Filippov static const struct eth_ops ethoc_ops = { 5915d43feabSMax Filippov .start = ethoc_start, 5925d43feabSMax Filippov .stop = ethoc_stop, 5935d43feabSMax Filippov .send = ethoc_send, 5945d43feabSMax Filippov .recv = ethoc_recv, 5955d43feabSMax Filippov .free_pkt = ethoc_free_pkt, 5965d43feabSMax Filippov .write_hwaddr = ethoc_write_hwaddr, 5975d43feabSMax Filippov }; 5985d43feabSMax Filippov 5992de18c8dSMax Filippov static const struct udevice_id ethoc_ids[] = { 6002de18c8dSMax Filippov { .compatible = "opencores,ethoc" }, 6012de18c8dSMax Filippov { } 6022de18c8dSMax Filippov }; 6032de18c8dSMax Filippov 6045d43feabSMax Filippov U_BOOT_DRIVER(ethoc) = { 6055d43feabSMax Filippov .name = "ethoc", 6065d43feabSMax Filippov .id = UCLASS_ETH, 6072de18c8dSMax Filippov .of_match = ethoc_ids, 6082de18c8dSMax Filippov .ofdata_to_platdata = ethoc_ofdata_to_platdata, 6095d43feabSMax Filippov .probe = ethoc_probe, 6105d43feabSMax Filippov .remove = ethoc_remove, 6115d43feabSMax Filippov .ops = ðoc_ops, 6125d43feabSMax Filippov .priv_auto_alloc_size = sizeof(struct ethoc), 6135d43feabSMax Filippov .platdata_auto_alloc_size = sizeof(struct ethoc_eth_pdata), 6145d43feabSMax Filippov }; 6155d43feabSMax Filippov 6165d43feabSMax Filippov #else 6175d43feabSMax Filippov 6185d43feabSMax Filippov static int ethoc_init(struct eth_device *dev, bd_t *bd) 6195d43feabSMax Filippov { 6205d43feabSMax Filippov struct ethoc *priv = (struct ethoc *)dev->priv; 6215d43feabSMax Filippov 6225d43feabSMax Filippov return ethoc_init_common(priv); 6235d43feabSMax Filippov } 6245d43feabSMax Filippov 6255d43feabSMax Filippov static int ethoc_write_hwaddr(struct eth_device *dev) 6265d43feabSMax Filippov { 6275d43feabSMax Filippov struct ethoc *priv = (struct ethoc *)dev->priv; 6285d43feabSMax Filippov u8 *mac = dev->enetaddr; 6295d43feabSMax Filippov 6305d43feabSMax Filippov return ethoc_write_hwaddr_common(priv, mac); 6315d43feabSMax Filippov } 6325d43feabSMax Filippov 6335d43feabSMax Filippov static int ethoc_send(struct eth_device *dev, void *packet, int length) 6345d43feabSMax Filippov { 6355d43feabSMax Filippov return ethoc_send_common(dev->priv, packet, length); 6365d43feabSMax Filippov } 6375d43feabSMax Filippov 638f6569884SThomas Chou static void ethoc_halt(struct eth_device *dev) 639f6569884SThomas Chou { 640a84a757aSMax Filippov ethoc_disable_rx_and_tx(dev->priv); 641f6569884SThomas Chou } 642f6569884SThomas Chou 643f6569884SThomas Chou static int ethoc_recv(struct eth_device *dev) 644f6569884SThomas Chou { 645a84a757aSMax Filippov struct ethoc *priv = (struct ethoc *)dev->priv; 6465d43feabSMax Filippov int count; 647f6569884SThomas Chou 6485d43feabSMax Filippov if (!ethoc_is_new_packet_received(priv)) 6495d43feabSMax Filippov return 0; 6505d43feabSMax Filippov 6515d43feabSMax Filippov for (count = 0; count < PKTBUFSRX; ++count) { 6525d43feabSMax Filippov uchar *packetp; 6535d43feabSMax Filippov int size = ethoc_rx_common(priv, &packetp); 6545d43feabSMax Filippov 6555d43feabSMax Filippov if (size < 0) 6565d43feabSMax Filippov break; 6575d43feabSMax Filippov if (size > 0) 6585d43feabSMax Filippov net_process_received_packet(packetp, size); 6595d43feabSMax Filippov ethoc_free_pkt_common(priv); 660f6569884SThomas Chou } 661f6569884SThomas Chou return 0; 662f6569884SThomas Chou } 663f6569884SThomas Chou 664f6569884SThomas Chou int ethoc_initialize(u8 dev_num, int base_addr) 665f6569884SThomas Chou { 666f6569884SThomas Chou struct ethoc *priv; 667f6569884SThomas Chou struct eth_device *dev; 668f6569884SThomas Chou 669f6569884SThomas Chou priv = malloc(sizeof(*priv)); 670f6569884SThomas Chou if (!priv) 671f6569884SThomas Chou return 0; 672f6569884SThomas Chou dev = malloc(sizeof(*dev)); 673f6569884SThomas Chou if (!dev) { 674f6569884SThomas Chou free(priv); 675f6569884SThomas Chou return 0; 676f6569884SThomas Chou } 677f6569884SThomas Chou 678f6569884SThomas Chou memset(dev, 0, sizeof(*dev)); 679f6569884SThomas Chou dev->priv = priv; 680f6569884SThomas Chou dev->iobase = base_addr; 681f6569884SThomas Chou dev->init = ethoc_init; 682f6569884SThomas Chou dev->halt = ethoc_halt; 683f6569884SThomas Chou dev->send = ethoc_send; 684f6569884SThomas Chou dev->recv = ethoc_recv; 6855d43feabSMax Filippov dev->write_hwaddr = ethoc_write_hwaddr; 686f6569884SThomas Chou sprintf(dev->name, "%s-%hu", "ETHOC", dev_num); 687a84a757aSMax Filippov priv->iobase = ioremap(dev->iobase, ETHOC_IOSIZE); 688f6569884SThomas Chou 689f6569884SThomas Chou eth_register(dev); 690f6569884SThomas Chou return 1; 691f6569884SThomas Chou } 6925d43feabSMax Filippov 6935d43feabSMax Filippov #endif 694