162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * Ethernet driver for the WIZnet W5300 chip. 462306a36Sopenharmony_ci * 562306a36Sopenharmony_ci * Copyright (C) 2008-2009 WIZnet Co.,Ltd. 662306a36Sopenharmony_ci * Copyright (C) 2011 Taehun Kim <kth3321 <at> gmail.com> 762306a36Sopenharmony_ci * Copyright (C) 2012 Mike Sinkovsky <msink@permonline.ru> 862306a36Sopenharmony_ci */ 962306a36Sopenharmony_ci 1062306a36Sopenharmony_ci#include <linux/kernel.h> 1162306a36Sopenharmony_ci#include <linux/module.h> 1262306a36Sopenharmony_ci#include <linux/netdevice.h> 1362306a36Sopenharmony_ci#include <linux/etherdevice.h> 1462306a36Sopenharmony_ci#include <linux/platform_device.h> 1562306a36Sopenharmony_ci#include <linux/platform_data/wiznet.h> 1662306a36Sopenharmony_ci#include <linux/ethtool.h> 1762306a36Sopenharmony_ci#include <linux/skbuff.h> 1862306a36Sopenharmony_ci#include <linux/types.h> 1962306a36Sopenharmony_ci#include <linux/errno.h> 2062306a36Sopenharmony_ci#include <linux/delay.h> 2162306a36Sopenharmony_ci#include <linux/slab.h> 2262306a36Sopenharmony_ci#include <linux/spinlock.h> 2362306a36Sopenharmony_ci#include <linux/io.h> 2462306a36Sopenharmony_ci#include <linux/ioport.h> 2562306a36Sopenharmony_ci#include <linux/interrupt.h> 2662306a36Sopenharmony_ci#include <linux/irq.h> 2762306a36Sopenharmony_ci#include <linux/gpio.h> 2862306a36Sopenharmony_ci 2962306a36Sopenharmony_ci#define DRV_NAME "w5300" 3062306a36Sopenharmony_ci#define DRV_VERSION "2012-04-04" 3162306a36Sopenharmony_ci 3262306a36Sopenharmony_ciMODULE_DESCRIPTION("WIZnet W5300 Ethernet driver v"DRV_VERSION); 3362306a36Sopenharmony_ciMODULE_AUTHOR("Mike Sinkovsky <msink@permonline.ru>"); 3462306a36Sopenharmony_ciMODULE_ALIAS("platform:"DRV_NAME); 3562306a36Sopenharmony_ciMODULE_LICENSE("GPL"); 3662306a36Sopenharmony_ci 3762306a36Sopenharmony_ci/* 3862306a36Sopenharmony_ci * Registers 3962306a36Sopenharmony_ci */ 4062306a36Sopenharmony_ci#define W5300_MR 0x0000 /* Mode Register */ 4162306a36Sopenharmony_ci#define MR_DBW (1 << 15) /* Data bus width */ 4262306a36Sopenharmony_ci#define MR_MPF (1 << 14) /* Mac layer pause frame */ 4362306a36Sopenharmony_ci#define MR_WDF(n) (((n)&7)<<11) /* Write data fetch time */ 4462306a36Sopenharmony_ci#define MR_RDH (1 << 10) /* Read data hold time */ 4562306a36Sopenharmony_ci#define MR_FS (1 << 8) /* FIFO swap */ 4662306a36Sopenharmony_ci#define MR_RST (1 << 7) /* S/W reset */ 4762306a36Sopenharmony_ci#define MR_PB (1 << 4) /* Ping block */ 4862306a36Sopenharmony_ci#define MR_DBS (1 << 2) /* Data bus swap */ 4962306a36Sopenharmony_ci#define MR_IND (1 << 0) /* Indirect mode */ 5062306a36Sopenharmony_ci#define W5300_IR 0x0002 /* Interrupt Register */ 5162306a36Sopenharmony_ci#define W5300_IMR 0x0004 /* Interrupt Mask Register */ 5262306a36Sopenharmony_ci#define IR_S0 0x0001 /* S0 interrupt */ 5362306a36Sopenharmony_ci#define W5300_SHARL 0x0008 /* Source MAC address (0123) */ 5462306a36Sopenharmony_ci#define W5300_SHARH 0x000c /* Source MAC address (45) */ 5562306a36Sopenharmony_ci#define W5300_TMSRL 0x0020 /* Transmit Memory Size (0123) */ 5662306a36Sopenharmony_ci#define W5300_TMSRH 0x0024 /* Transmit Memory Size (4567) */ 5762306a36Sopenharmony_ci#define W5300_RMSRL 0x0028 /* Receive Memory Size (0123) */ 5862306a36Sopenharmony_ci#define W5300_RMSRH 0x002c /* Receive Memory Size (4567) */ 5962306a36Sopenharmony_ci#define W5300_MTYPE 0x0030 /* Memory Type */ 6062306a36Sopenharmony_ci#define W5300_IDR 0x00fe /* Chip ID register */ 6162306a36Sopenharmony_ci#define IDR_W5300 0x5300 /* =0x5300 for WIZnet W5300 */ 6262306a36Sopenharmony_ci#define W5300_S0_MR 0x0200 /* S0 Mode Register */ 6362306a36Sopenharmony_ci#define S0_MR_CLOSED 0x0000 /* Close mode */ 6462306a36Sopenharmony_ci#define S0_MR_MACRAW 0x0004 /* MAC RAW mode (promiscuous) */ 6562306a36Sopenharmony_ci#define S0_MR_MACRAW_MF 0x0044 /* MAC RAW mode (filtered) */ 6662306a36Sopenharmony_ci#define W5300_S0_CR 0x0202 /* S0 Command Register */ 6762306a36Sopenharmony_ci#define S0_CR_OPEN 0x0001 /* OPEN command */ 6862306a36Sopenharmony_ci#define S0_CR_CLOSE 0x0010 /* CLOSE command */ 6962306a36Sopenharmony_ci#define S0_CR_SEND 0x0020 /* SEND command */ 7062306a36Sopenharmony_ci#define S0_CR_RECV 0x0040 /* RECV command */ 7162306a36Sopenharmony_ci#define W5300_S0_IMR 0x0204 /* S0 Interrupt Mask Register */ 7262306a36Sopenharmony_ci#define W5300_S0_IR 0x0206 /* S0 Interrupt Register */ 7362306a36Sopenharmony_ci#define S0_IR_RECV 0x0004 /* Receive interrupt */ 7462306a36Sopenharmony_ci#define S0_IR_SENDOK 0x0010 /* Send OK interrupt */ 7562306a36Sopenharmony_ci#define W5300_S0_SSR 0x0208 /* S0 Socket Status Register */ 7662306a36Sopenharmony_ci#define W5300_S0_TX_WRSR 0x0220 /* S0 TX Write Size Register */ 7762306a36Sopenharmony_ci#define W5300_S0_TX_FSR 0x0224 /* S0 TX Free Size Register */ 7862306a36Sopenharmony_ci#define W5300_S0_RX_RSR 0x0228 /* S0 Received data Size */ 7962306a36Sopenharmony_ci#define W5300_S0_TX_FIFO 0x022e /* S0 Transmit FIFO */ 8062306a36Sopenharmony_ci#define W5300_S0_RX_FIFO 0x0230 /* S0 Receive FIFO */ 8162306a36Sopenharmony_ci#define W5300_REGS_LEN 0x0400 8262306a36Sopenharmony_ci 8362306a36Sopenharmony_ci/* 8462306a36Sopenharmony_ci * Device driver private data structure 8562306a36Sopenharmony_ci */ 8662306a36Sopenharmony_cistruct w5300_priv { 8762306a36Sopenharmony_ci void __iomem *base; 8862306a36Sopenharmony_ci spinlock_t reg_lock; 8962306a36Sopenharmony_ci bool indirect; 9062306a36Sopenharmony_ci u16 (*read) (struct w5300_priv *priv, u16 addr); 9162306a36Sopenharmony_ci void (*write)(struct w5300_priv *priv, u16 addr, u16 data); 9262306a36Sopenharmony_ci int irq; 9362306a36Sopenharmony_ci int link_irq; 9462306a36Sopenharmony_ci int link_gpio; 9562306a36Sopenharmony_ci 9662306a36Sopenharmony_ci struct napi_struct napi; 9762306a36Sopenharmony_ci struct net_device *ndev; 9862306a36Sopenharmony_ci bool promisc; 9962306a36Sopenharmony_ci u32 msg_enable; 10062306a36Sopenharmony_ci}; 10162306a36Sopenharmony_ci 10262306a36Sopenharmony_ci/************************************************************************ 10362306a36Sopenharmony_ci * 10462306a36Sopenharmony_ci * Lowlevel I/O functions 10562306a36Sopenharmony_ci * 10662306a36Sopenharmony_ci ***********************************************************************/ 10762306a36Sopenharmony_ci 10862306a36Sopenharmony_ci/* 10962306a36Sopenharmony_ci * In direct address mode host system can directly access W5300 registers 11062306a36Sopenharmony_ci * after mapping to Memory-Mapped I/O space. 11162306a36Sopenharmony_ci * 11262306a36Sopenharmony_ci * 0x400 bytes are required for memory space. 11362306a36Sopenharmony_ci */ 11462306a36Sopenharmony_cistatic inline u16 w5300_read_direct(struct w5300_priv *priv, u16 addr) 11562306a36Sopenharmony_ci{ 11662306a36Sopenharmony_ci return ioread16(priv->base + (addr << CONFIG_WIZNET_BUS_SHIFT)); 11762306a36Sopenharmony_ci} 11862306a36Sopenharmony_ci 11962306a36Sopenharmony_cistatic inline void w5300_write_direct(struct w5300_priv *priv, 12062306a36Sopenharmony_ci u16 addr, u16 data) 12162306a36Sopenharmony_ci{ 12262306a36Sopenharmony_ci iowrite16(data, priv->base + (addr << CONFIG_WIZNET_BUS_SHIFT)); 12362306a36Sopenharmony_ci} 12462306a36Sopenharmony_ci 12562306a36Sopenharmony_ci/* 12662306a36Sopenharmony_ci * In indirect address mode host system indirectly accesses registers by 12762306a36Sopenharmony_ci * using Indirect Mode Address Register (IDM_AR) and Indirect Mode Data 12862306a36Sopenharmony_ci * Register (IDM_DR), which are directly mapped to Memory-Mapped I/O space. 12962306a36Sopenharmony_ci * Mode Register (MR) is directly accessible. 13062306a36Sopenharmony_ci * 13162306a36Sopenharmony_ci * Only 0x06 bytes are required for memory space. 13262306a36Sopenharmony_ci */ 13362306a36Sopenharmony_ci#define W5300_IDM_AR 0x0002 /* Indirect Mode Address */ 13462306a36Sopenharmony_ci#define W5300_IDM_DR 0x0004 /* Indirect Mode Data */ 13562306a36Sopenharmony_ci 13662306a36Sopenharmony_cistatic u16 w5300_read_indirect(struct w5300_priv *priv, u16 addr) 13762306a36Sopenharmony_ci{ 13862306a36Sopenharmony_ci unsigned long flags; 13962306a36Sopenharmony_ci u16 data; 14062306a36Sopenharmony_ci 14162306a36Sopenharmony_ci spin_lock_irqsave(&priv->reg_lock, flags); 14262306a36Sopenharmony_ci w5300_write_direct(priv, W5300_IDM_AR, addr); 14362306a36Sopenharmony_ci data = w5300_read_direct(priv, W5300_IDM_DR); 14462306a36Sopenharmony_ci spin_unlock_irqrestore(&priv->reg_lock, flags); 14562306a36Sopenharmony_ci 14662306a36Sopenharmony_ci return data; 14762306a36Sopenharmony_ci} 14862306a36Sopenharmony_ci 14962306a36Sopenharmony_cistatic void w5300_write_indirect(struct w5300_priv *priv, u16 addr, u16 data) 15062306a36Sopenharmony_ci{ 15162306a36Sopenharmony_ci unsigned long flags; 15262306a36Sopenharmony_ci 15362306a36Sopenharmony_ci spin_lock_irqsave(&priv->reg_lock, flags); 15462306a36Sopenharmony_ci w5300_write_direct(priv, W5300_IDM_AR, addr); 15562306a36Sopenharmony_ci w5300_write_direct(priv, W5300_IDM_DR, data); 15662306a36Sopenharmony_ci spin_unlock_irqrestore(&priv->reg_lock, flags); 15762306a36Sopenharmony_ci} 15862306a36Sopenharmony_ci 15962306a36Sopenharmony_ci#if defined(CONFIG_WIZNET_BUS_DIRECT) 16062306a36Sopenharmony_ci#define w5300_read w5300_read_direct 16162306a36Sopenharmony_ci#define w5300_write w5300_write_direct 16262306a36Sopenharmony_ci 16362306a36Sopenharmony_ci#elif defined(CONFIG_WIZNET_BUS_INDIRECT) 16462306a36Sopenharmony_ci#define w5300_read w5300_read_indirect 16562306a36Sopenharmony_ci#define w5300_write w5300_write_indirect 16662306a36Sopenharmony_ci 16762306a36Sopenharmony_ci#else /* CONFIG_WIZNET_BUS_ANY */ 16862306a36Sopenharmony_ci#define w5300_read priv->read 16962306a36Sopenharmony_ci#define w5300_write priv->write 17062306a36Sopenharmony_ci#endif 17162306a36Sopenharmony_ci 17262306a36Sopenharmony_cistatic u32 w5300_read32(struct w5300_priv *priv, u16 addr) 17362306a36Sopenharmony_ci{ 17462306a36Sopenharmony_ci u32 data; 17562306a36Sopenharmony_ci data = w5300_read(priv, addr) << 16; 17662306a36Sopenharmony_ci data |= w5300_read(priv, addr + 2); 17762306a36Sopenharmony_ci return data; 17862306a36Sopenharmony_ci} 17962306a36Sopenharmony_ci 18062306a36Sopenharmony_cistatic void w5300_write32(struct w5300_priv *priv, u16 addr, u32 data) 18162306a36Sopenharmony_ci{ 18262306a36Sopenharmony_ci w5300_write(priv, addr, data >> 16); 18362306a36Sopenharmony_ci w5300_write(priv, addr + 2, data); 18462306a36Sopenharmony_ci} 18562306a36Sopenharmony_ci 18662306a36Sopenharmony_cistatic int w5300_command(struct w5300_priv *priv, u16 cmd) 18762306a36Sopenharmony_ci{ 18862306a36Sopenharmony_ci unsigned long timeout = jiffies + msecs_to_jiffies(100); 18962306a36Sopenharmony_ci 19062306a36Sopenharmony_ci w5300_write(priv, W5300_S0_CR, cmd); 19162306a36Sopenharmony_ci 19262306a36Sopenharmony_ci while (w5300_read(priv, W5300_S0_CR) != 0) { 19362306a36Sopenharmony_ci if (time_after(jiffies, timeout)) 19462306a36Sopenharmony_ci return -EIO; 19562306a36Sopenharmony_ci cpu_relax(); 19662306a36Sopenharmony_ci } 19762306a36Sopenharmony_ci 19862306a36Sopenharmony_ci return 0; 19962306a36Sopenharmony_ci} 20062306a36Sopenharmony_ci 20162306a36Sopenharmony_cistatic void w5300_read_frame(struct w5300_priv *priv, u8 *buf, int len) 20262306a36Sopenharmony_ci{ 20362306a36Sopenharmony_ci u16 fifo; 20462306a36Sopenharmony_ci int i; 20562306a36Sopenharmony_ci 20662306a36Sopenharmony_ci for (i = 0; i < len; i += 2) { 20762306a36Sopenharmony_ci fifo = w5300_read(priv, W5300_S0_RX_FIFO); 20862306a36Sopenharmony_ci *buf++ = fifo >> 8; 20962306a36Sopenharmony_ci *buf++ = fifo; 21062306a36Sopenharmony_ci } 21162306a36Sopenharmony_ci fifo = w5300_read(priv, W5300_S0_RX_FIFO); 21262306a36Sopenharmony_ci fifo = w5300_read(priv, W5300_S0_RX_FIFO); 21362306a36Sopenharmony_ci} 21462306a36Sopenharmony_ci 21562306a36Sopenharmony_cistatic void w5300_write_frame(struct w5300_priv *priv, u8 *buf, int len) 21662306a36Sopenharmony_ci{ 21762306a36Sopenharmony_ci u16 fifo; 21862306a36Sopenharmony_ci int i; 21962306a36Sopenharmony_ci 22062306a36Sopenharmony_ci for (i = 0; i < len; i += 2) { 22162306a36Sopenharmony_ci fifo = *buf++ << 8; 22262306a36Sopenharmony_ci fifo |= *buf++; 22362306a36Sopenharmony_ci w5300_write(priv, W5300_S0_TX_FIFO, fifo); 22462306a36Sopenharmony_ci } 22562306a36Sopenharmony_ci w5300_write32(priv, W5300_S0_TX_WRSR, len); 22662306a36Sopenharmony_ci} 22762306a36Sopenharmony_ci 22862306a36Sopenharmony_cistatic void w5300_write_macaddr(struct w5300_priv *priv) 22962306a36Sopenharmony_ci{ 23062306a36Sopenharmony_ci struct net_device *ndev = priv->ndev; 23162306a36Sopenharmony_ci w5300_write32(priv, W5300_SHARL, 23262306a36Sopenharmony_ci ndev->dev_addr[0] << 24 | 23362306a36Sopenharmony_ci ndev->dev_addr[1] << 16 | 23462306a36Sopenharmony_ci ndev->dev_addr[2] << 8 | 23562306a36Sopenharmony_ci ndev->dev_addr[3]); 23662306a36Sopenharmony_ci w5300_write(priv, W5300_SHARH, 23762306a36Sopenharmony_ci ndev->dev_addr[4] << 8 | 23862306a36Sopenharmony_ci ndev->dev_addr[5]); 23962306a36Sopenharmony_ci} 24062306a36Sopenharmony_ci 24162306a36Sopenharmony_cistatic void w5300_hw_reset(struct w5300_priv *priv) 24262306a36Sopenharmony_ci{ 24362306a36Sopenharmony_ci w5300_write_direct(priv, W5300_MR, MR_RST); 24462306a36Sopenharmony_ci mdelay(5); 24562306a36Sopenharmony_ci w5300_write_direct(priv, W5300_MR, priv->indirect ? 24662306a36Sopenharmony_ci MR_WDF(7) | MR_PB | MR_IND : 24762306a36Sopenharmony_ci MR_WDF(7) | MR_PB); 24862306a36Sopenharmony_ci w5300_write(priv, W5300_IMR, 0); 24962306a36Sopenharmony_ci w5300_write_macaddr(priv); 25062306a36Sopenharmony_ci 25162306a36Sopenharmony_ci /* Configure 128K of internal memory 25262306a36Sopenharmony_ci * as 64K RX fifo and 64K TX fifo 25362306a36Sopenharmony_ci */ 25462306a36Sopenharmony_ci w5300_write32(priv, W5300_RMSRL, 64 << 24); 25562306a36Sopenharmony_ci w5300_write32(priv, W5300_RMSRH, 0); 25662306a36Sopenharmony_ci w5300_write32(priv, W5300_TMSRL, 64 << 24); 25762306a36Sopenharmony_ci w5300_write32(priv, W5300_TMSRH, 0); 25862306a36Sopenharmony_ci w5300_write(priv, W5300_MTYPE, 0x00ff); 25962306a36Sopenharmony_ci} 26062306a36Sopenharmony_ci 26162306a36Sopenharmony_cistatic void w5300_hw_start(struct w5300_priv *priv) 26262306a36Sopenharmony_ci{ 26362306a36Sopenharmony_ci w5300_write(priv, W5300_S0_MR, priv->promisc ? 26462306a36Sopenharmony_ci S0_MR_MACRAW : S0_MR_MACRAW_MF); 26562306a36Sopenharmony_ci w5300_command(priv, S0_CR_OPEN); 26662306a36Sopenharmony_ci w5300_write(priv, W5300_S0_IMR, S0_IR_RECV | S0_IR_SENDOK); 26762306a36Sopenharmony_ci w5300_write(priv, W5300_IMR, IR_S0); 26862306a36Sopenharmony_ci} 26962306a36Sopenharmony_ci 27062306a36Sopenharmony_cistatic void w5300_hw_close(struct w5300_priv *priv) 27162306a36Sopenharmony_ci{ 27262306a36Sopenharmony_ci w5300_write(priv, W5300_IMR, 0); 27362306a36Sopenharmony_ci w5300_command(priv, S0_CR_CLOSE); 27462306a36Sopenharmony_ci} 27562306a36Sopenharmony_ci 27662306a36Sopenharmony_ci/*********************************************************************** 27762306a36Sopenharmony_ci * 27862306a36Sopenharmony_ci * Device driver functions / callbacks 27962306a36Sopenharmony_ci * 28062306a36Sopenharmony_ci ***********************************************************************/ 28162306a36Sopenharmony_ci 28262306a36Sopenharmony_cistatic void w5300_get_drvinfo(struct net_device *ndev, 28362306a36Sopenharmony_ci struct ethtool_drvinfo *info) 28462306a36Sopenharmony_ci{ 28562306a36Sopenharmony_ci strscpy(info->driver, DRV_NAME, sizeof(info->driver)); 28662306a36Sopenharmony_ci strscpy(info->version, DRV_VERSION, sizeof(info->version)); 28762306a36Sopenharmony_ci strscpy(info->bus_info, dev_name(ndev->dev.parent), 28862306a36Sopenharmony_ci sizeof(info->bus_info)); 28962306a36Sopenharmony_ci} 29062306a36Sopenharmony_ci 29162306a36Sopenharmony_cistatic u32 w5300_get_link(struct net_device *ndev) 29262306a36Sopenharmony_ci{ 29362306a36Sopenharmony_ci struct w5300_priv *priv = netdev_priv(ndev); 29462306a36Sopenharmony_ci 29562306a36Sopenharmony_ci if (gpio_is_valid(priv->link_gpio)) 29662306a36Sopenharmony_ci return !!gpio_get_value(priv->link_gpio); 29762306a36Sopenharmony_ci 29862306a36Sopenharmony_ci return 1; 29962306a36Sopenharmony_ci} 30062306a36Sopenharmony_ci 30162306a36Sopenharmony_cistatic u32 w5300_get_msglevel(struct net_device *ndev) 30262306a36Sopenharmony_ci{ 30362306a36Sopenharmony_ci struct w5300_priv *priv = netdev_priv(ndev); 30462306a36Sopenharmony_ci 30562306a36Sopenharmony_ci return priv->msg_enable; 30662306a36Sopenharmony_ci} 30762306a36Sopenharmony_ci 30862306a36Sopenharmony_cistatic void w5300_set_msglevel(struct net_device *ndev, u32 value) 30962306a36Sopenharmony_ci{ 31062306a36Sopenharmony_ci struct w5300_priv *priv = netdev_priv(ndev); 31162306a36Sopenharmony_ci 31262306a36Sopenharmony_ci priv->msg_enable = value; 31362306a36Sopenharmony_ci} 31462306a36Sopenharmony_ci 31562306a36Sopenharmony_cistatic int w5300_get_regs_len(struct net_device *ndev) 31662306a36Sopenharmony_ci{ 31762306a36Sopenharmony_ci return W5300_REGS_LEN; 31862306a36Sopenharmony_ci} 31962306a36Sopenharmony_ci 32062306a36Sopenharmony_cistatic void w5300_get_regs(struct net_device *ndev, 32162306a36Sopenharmony_ci struct ethtool_regs *regs, void *_buf) 32262306a36Sopenharmony_ci{ 32362306a36Sopenharmony_ci struct w5300_priv *priv = netdev_priv(ndev); 32462306a36Sopenharmony_ci u8 *buf = _buf; 32562306a36Sopenharmony_ci u16 addr; 32662306a36Sopenharmony_ci u16 data; 32762306a36Sopenharmony_ci 32862306a36Sopenharmony_ci regs->version = 1; 32962306a36Sopenharmony_ci for (addr = 0; addr < W5300_REGS_LEN; addr += 2) { 33062306a36Sopenharmony_ci switch (addr & 0x23f) { 33162306a36Sopenharmony_ci case W5300_S0_TX_FIFO: /* cannot read TX_FIFO */ 33262306a36Sopenharmony_ci case W5300_S0_RX_FIFO: /* cannot read RX_FIFO */ 33362306a36Sopenharmony_ci data = 0xffff; 33462306a36Sopenharmony_ci break; 33562306a36Sopenharmony_ci default: 33662306a36Sopenharmony_ci data = w5300_read(priv, addr); 33762306a36Sopenharmony_ci break; 33862306a36Sopenharmony_ci } 33962306a36Sopenharmony_ci *buf++ = data >> 8; 34062306a36Sopenharmony_ci *buf++ = data; 34162306a36Sopenharmony_ci } 34262306a36Sopenharmony_ci} 34362306a36Sopenharmony_ci 34462306a36Sopenharmony_cistatic void w5300_tx_timeout(struct net_device *ndev, unsigned int txqueue) 34562306a36Sopenharmony_ci{ 34662306a36Sopenharmony_ci struct w5300_priv *priv = netdev_priv(ndev); 34762306a36Sopenharmony_ci 34862306a36Sopenharmony_ci netif_stop_queue(ndev); 34962306a36Sopenharmony_ci w5300_hw_reset(priv); 35062306a36Sopenharmony_ci w5300_hw_start(priv); 35162306a36Sopenharmony_ci ndev->stats.tx_errors++; 35262306a36Sopenharmony_ci netif_trans_update(ndev); 35362306a36Sopenharmony_ci netif_wake_queue(ndev); 35462306a36Sopenharmony_ci} 35562306a36Sopenharmony_ci 35662306a36Sopenharmony_cistatic netdev_tx_t w5300_start_tx(struct sk_buff *skb, struct net_device *ndev) 35762306a36Sopenharmony_ci{ 35862306a36Sopenharmony_ci struct w5300_priv *priv = netdev_priv(ndev); 35962306a36Sopenharmony_ci 36062306a36Sopenharmony_ci netif_stop_queue(ndev); 36162306a36Sopenharmony_ci 36262306a36Sopenharmony_ci w5300_write_frame(priv, skb->data, skb->len); 36362306a36Sopenharmony_ci ndev->stats.tx_packets++; 36462306a36Sopenharmony_ci ndev->stats.tx_bytes += skb->len; 36562306a36Sopenharmony_ci dev_kfree_skb(skb); 36662306a36Sopenharmony_ci netif_dbg(priv, tx_queued, ndev, "tx queued\n"); 36762306a36Sopenharmony_ci 36862306a36Sopenharmony_ci w5300_command(priv, S0_CR_SEND); 36962306a36Sopenharmony_ci 37062306a36Sopenharmony_ci return NETDEV_TX_OK; 37162306a36Sopenharmony_ci} 37262306a36Sopenharmony_ci 37362306a36Sopenharmony_cistatic int w5300_napi_poll(struct napi_struct *napi, int budget) 37462306a36Sopenharmony_ci{ 37562306a36Sopenharmony_ci struct w5300_priv *priv = container_of(napi, struct w5300_priv, napi); 37662306a36Sopenharmony_ci struct net_device *ndev = priv->ndev; 37762306a36Sopenharmony_ci struct sk_buff *skb; 37862306a36Sopenharmony_ci int rx_count; 37962306a36Sopenharmony_ci u16 rx_len; 38062306a36Sopenharmony_ci 38162306a36Sopenharmony_ci for (rx_count = 0; rx_count < budget; rx_count++) { 38262306a36Sopenharmony_ci u32 rx_fifo_len = w5300_read32(priv, W5300_S0_RX_RSR); 38362306a36Sopenharmony_ci if (rx_fifo_len == 0) 38462306a36Sopenharmony_ci break; 38562306a36Sopenharmony_ci 38662306a36Sopenharmony_ci rx_len = w5300_read(priv, W5300_S0_RX_FIFO); 38762306a36Sopenharmony_ci 38862306a36Sopenharmony_ci skb = netdev_alloc_skb_ip_align(ndev, roundup(rx_len, 2)); 38962306a36Sopenharmony_ci if (unlikely(!skb)) { 39062306a36Sopenharmony_ci u32 i; 39162306a36Sopenharmony_ci for (i = 0; i < rx_fifo_len; i += 2) 39262306a36Sopenharmony_ci w5300_read(priv, W5300_S0_RX_FIFO); 39362306a36Sopenharmony_ci ndev->stats.rx_dropped++; 39462306a36Sopenharmony_ci return -ENOMEM; 39562306a36Sopenharmony_ci } 39662306a36Sopenharmony_ci 39762306a36Sopenharmony_ci skb_put(skb, rx_len); 39862306a36Sopenharmony_ci w5300_read_frame(priv, skb->data, rx_len); 39962306a36Sopenharmony_ci skb->protocol = eth_type_trans(skb, ndev); 40062306a36Sopenharmony_ci 40162306a36Sopenharmony_ci netif_receive_skb(skb); 40262306a36Sopenharmony_ci ndev->stats.rx_packets++; 40362306a36Sopenharmony_ci ndev->stats.rx_bytes += rx_len; 40462306a36Sopenharmony_ci } 40562306a36Sopenharmony_ci 40662306a36Sopenharmony_ci if (rx_count < budget) { 40762306a36Sopenharmony_ci napi_complete_done(napi, rx_count); 40862306a36Sopenharmony_ci w5300_write(priv, W5300_IMR, IR_S0); 40962306a36Sopenharmony_ci } 41062306a36Sopenharmony_ci 41162306a36Sopenharmony_ci return rx_count; 41262306a36Sopenharmony_ci} 41362306a36Sopenharmony_ci 41462306a36Sopenharmony_cistatic irqreturn_t w5300_interrupt(int irq, void *ndev_instance) 41562306a36Sopenharmony_ci{ 41662306a36Sopenharmony_ci struct net_device *ndev = ndev_instance; 41762306a36Sopenharmony_ci struct w5300_priv *priv = netdev_priv(ndev); 41862306a36Sopenharmony_ci 41962306a36Sopenharmony_ci int ir = w5300_read(priv, W5300_S0_IR); 42062306a36Sopenharmony_ci if (!ir) 42162306a36Sopenharmony_ci return IRQ_NONE; 42262306a36Sopenharmony_ci w5300_write(priv, W5300_S0_IR, ir); 42362306a36Sopenharmony_ci 42462306a36Sopenharmony_ci if (ir & S0_IR_SENDOK) { 42562306a36Sopenharmony_ci netif_dbg(priv, tx_done, ndev, "tx done\n"); 42662306a36Sopenharmony_ci netif_wake_queue(ndev); 42762306a36Sopenharmony_ci } 42862306a36Sopenharmony_ci 42962306a36Sopenharmony_ci if (ir & S0_IR_RECV) { 43062306a36Sopenharmony_ci if (napi_schedule_prep(&priv->napi)) { 43162306a36Sopenharmony_ci w5300_write(priv, W5300_IMR, 0); 43262306a36Sopenharmony_ci __napi_schedule(&priv->napi); 43362306a36Sopenharmony_ci } 43462306a36Sopenharmony_ci } 43562306a36Sopenharmony_ci 43662306a36Sopenharmony_ci return IRQ_HANDLED; 43762306a36Sopenharmony_ci} 43862306a36Sopenharmony_ci 43962306a36Sopenharmony_cistatic irqreturn_t w5300_detect_link(int irq, void *ndev_instance) 44062306a36Sopenharmony_ci{ 44162306a36Sopenharmony_ci struct net_device *ndev = ndev_instance; 44262306a36Sopenharmony_ci struct w5300_priv *priv = netdev_priv(ndev); 44362306a36Sopenharmony_ci 44462306a36Sopenharmony_ci if (netif_running(ndev)) { 44562306a36Sopenharmony_ci if (gpio_get_value(priv->link_gpio) != 0) { 44662306a36Sopenharmony_ci netif_info(priv, link, ndev, "link is up\n"); 44762306a36Sopenharmony_ci netif_carrier_on(ndev); 44862306a36Sopenharmony_ci } else { 44962306a36Sopenharmony_ci netif_info(priv, link, ndev, "link is down\n"); 45062306a36Sopenharmony_ci netif_carrier_off(ndev); 45162306a36Sopenharmony_ci } 45262306a36Sopenharmony_ci } 45362306a36Sopenharmony_ci 45462306a36Sopenharmony_ci return IRQ_HANDLED; 45562306a36Sopenharmony_ci} 45662306a36Sopenharmony_ci 45762306a36Sopenharmony_cistatic void w5300_set_rx_mode(struct net_device *ndev) 45862306a36Sopenharmony_ci{ 45962306a36Sopenharmony_ci struct w5300_priv *priv = netdev_priv(ndev); 46062306a36Sopenharmony_ci bool set_promisc = (ndev->flags & IFF_PROMISC) != 0; 46162306a36Sopenharmony_ci 46262306a36Sopenharmony_ci if (priv->promisc != set_promisc) { 46362306a36Sopenharmony_ci priv->promisc = set_promisc; 46462306a36Sopenharmony_ci w5300_hw_start(priv); 46562306a36Sopenharmony_ci } 46662306a36Sopenharmony_ci} 46762306a36Sopenharmony_ci 46862306a36Sopenharmony_cistatic int w5300_set_macaddr(struct net_device *ndev, void *addr) 46962306a36Sopenharmony_ci{ 47062306a36Sopenharmony_ci struct w5300_priv *priv = netdev_priv(ndev); 47162306a36Sopenharmony_ci struct sockaddr *sock_addr = addr; 47262306a36Sopenharmony_ci 47362306a36Sopenharmony_ci if (!is_valid_ether_addr(sock_addr->sa_data)) 47462306a36Sopenharmony_ci return -EADDRNOTAVAIL; 47562306a36Sopenharmony_ci eth_hw_addr_set(ndev, sock_addr->sa_data); 47662306a36Sopenharmony_ci w5300_write_macaddr(priv); 47762306a36Sopenharmony_ci return 0; 47862306a36Sopenharmony_ci} 47962306a36Sopenharmony_ci 48062306a36Sopenharmony_cistatic int w5300_open(struct net_device *ndev) 48162306a36Sopenharmony_ci{ 48262306a36Sopenharmony_ci struct w5300_priv *priv = netdev_priv(ndev); 48362306a36Sopenharmony_ci 48462306a36Sopenharmony_ci netif_info(priv, ifup, ndev, "enabling\n"); 48562306a36Sopenharmony_ci w5300_hw_start(priv); 48662306a36Sopenharmony_ci napi_enable(&priv->napi); 48762306a36Sopenharmony_ci netif_start_queue(ndev); 48862306a36Sopenharmony_ci if (!gpio_is_valid(priv->link_gpio) || 48962306a36Sopenharmony_ci gpio_get_value(priv->link_gpio) != 0) 49062306a36Sopenharmony_ci netif_carrier_on(ndev); 49162306a36Sopenharmony_ci return 0; 49262306a36Sopenharmony_ci} 49362306a36Sopenharmony_ci 49462306a36Sopenharmony_cistatic int w5300_stop(struct net_device *ndev) 49562306a36Sopenharmony_ci{ 49662306a36Sopenharmony_ci struct w5300_priv *priv = netdev_priv(ndev); 49762306a36Sopenharmony_ci 49862306a36Sopenharmony_ci netif_info(priv, ifdown, ndev, "shutting down\n"); 49962306a36Sopenharmony_ci w5300_hw_close(priv); 50062306a36Sopenharmony_ci netif_carrier_off(ndev); 50162306a36Sopenharmony_ci netif_stop_queue(ndev); 50262306a36Sopenharmony_ci napi_disable(&priv->napi); 50362306a36Sopenharmony_ci return 0; 50462306a36Sopenharmony_ci} 50562306a36Sopenharmony_ci 50662306a36Sopenharmony_cistatic const struct ethtool_ops w5300_ethtool_ops = { 50762306a36Sopenharmony_ci .get_drvinfo = w5300_get_drvinfo, 50862306a36Sopenharmony_ci .get_msglevel = w5300_get_msglevel, 50962306a36Sopenharmony_ci .set_msglevel = w5300_set_msglevel, 51062306a36Sopenharmony_ci .get_link = w5300_get_link, 51162306a36Sopenharmony_ci .get_regs_len = w5300_get_regs_len, 51262306a36Sopenharmony_ci .get_regs = w5300_get_regs, 51362306a36Sopenharmony_ci}; 51462306a36Sopenharmony_ci 51562306a36Sopenharmony_cistatic const struct net_device_ops w5300_netdev_ops = { 51662306a36Sopenharmony_ci .ndo_open = w5300_open, 51762306a36Sopenharmony_ci .ndo_stop = w5300_stop, 51862306a36Sopenharmony_ci .ndo_start_xmit = w5300_start_tx, 51962306a36Sopenharmony_ci .ndo_tx_timeout = w5300_tx_timeout, 52062306a36Sopenharmony_ci .ndo_set_rx_mode = w5300_set_rx_mode, 52162306a36Sopenharmony_ci .ndo_set_mac_address = w5300_set_macaddr, 52262306a36Sopenharmony_ci .ndo_validate_addr = eth_validate_addr, 52362306a36Sopenharmony_ci}; 52462306a36Sopenharmony_ci 52562306a36Sopenharmony_cistatic int w5300_hw_probe(struct platform_device *pdev) 52662306a36Sopenharmony_ci{ 52762306a36Sopenharmony_ci struct wiznet_platform_data *data = dev_get_platdata(&pdev->dev); 52862306a36Sopenharmony_ci struct net_device *ndev = platform_get_drvdata(pdev); 52962306a36Sopenharmony_ci struct w5300_priv *priv = netdev_priv(ndev); 53062306a36Sopenharmony_ci const char *name = netdev_name(ndev); 53162306a36Sopenharmony_ci struct resource *mem; 53262306a36Sopenharmony_ci int mem_size; 53362306a36Sopenharmony_ci int irq; 53462306a36Sopenharmony_ci int ret; 53562306a36Sopenharmony_ci 53662306a36Sopenharmony_ci if (data && is_valid_ether_addr(data->mac_addr)) { 53762306a36Sopenharmony_ci eth_hw_addr_set(ndev, data->mac_addr); 53862306a36Sopenharmony_ci } else { 53962306a36Sopenharmony_ci eth_hw_addr_random(ndev); 54062306a36Sopenharmony_ci } 54162306a36Sopenharmony_ci 54262306a36Sopenharmony_ci mem = platform_get_resource(pdev, IORESOURCE_MEM, 0); 54362306a36Sopenharmony_ci priv->base = devm_ioremap_resource(&pdev->dev, mem); 54462306a36Sopenharmony_ci if (IS_ERR(priv->base)) 54562306a36Sopenharmony_ci return PTR_ERR(priv->base); 54662306a36Sopenharmony_ci 54762306a36Sopenharmony_ci mem_size = resource_size(mem); 54862306a36Sopenharmony_ci 54962306a36Sopenharmony_ci spin_lock_init(&priv->reg_lock); 55062306a36Sopenharmony_ci priv->indirect = mem_size < W5300_BUS_DIRECT_SIZE; 55162306a36Sopenharmony_ci if (priv->indirect) { 55262306a36Sopenharmony_ci priv->read = w5300_read_indirect; 55362306a36Sopenharmony_ci priv->write = w5300_write_indirect; 55462306a36Sopenharmony_ci } else { 55562306a36Sopenharmony_ci priv->read = w5300_read_direct; 55662306a36Sopenharmony_ci priv->write = w5300_write_direct; 55762306a36Sopenharmony_ci } 55862306a36Sopenharmony_ci 55962306a36Sopenharmony_ci w5300_hw_reset(priv); 56062306a36Sopenharmony_ci if (w5300_read(priv, W5300_IDR) != IDR_W5300) 56162306a36Sopenharmony_ci return -ENODEV; 56262306a36Sopenharmony_ci 56362306a36Sopenharmony_ci irq = platform_get_irq(pdev, 0); 56462306a36Sopenharmony_ci if (irq < 0) 56562306a36Sopenharmony_ci return irq; 56662306a36Sopenharmony_ci ret = request_irq(irq, w5300_interrupt, 56762306a36Sopenharmony_ci IRQ_TYPE_LEVEL_LOW, name, ndev); 56862306a36Sopenharmony_ci if (ret < 0) 56962306a36Sopenharmony_ci return ret; 57062306a36Sopenharmony_ci priv->irq = irq; 57162306a36Sopenharmony_ci 57262306a36Sopenharmony_ci priv->link_gpio = data ? data->link_gpio : -EINVAL; 57362306a36Sopenharmony_ci if (gpio_is_valid(priv->link_gpio)) { 57462306a36Sopenharmony_ci char *link_name = devm_kzalloc(&pdev->dev, 16, GFP_KERNEL); 57562306a36Sopenharmony_ci if (!link_name) 57662306a36Sopenharmony_ci return -ENOMEM; 57762306a36Sopenharmony_ci snprintf(link_name, 16, "%s-link", name); 57862306a36Sopenharmony_ci priv->link_irq = gpio_to_irq(priv->link_gpio); 57962306a36Sopenharmony_ci if (request_any_context_irq(priv->link_irq, w5300_detect_link, 58062306a36Sopenharmony_ci IRQF_TRIGGER_RISING | IRQF_TRIGGER_FALLING, 58162306a36Sopenharmony_ci link_name, priv->ndev) < 0) 58262306a36Sopenharmony_ci priv->link_gpio = -EINVAL; 58362306a36Sopenharmony_ci } 58462306a36Sopenharmony_ci 58562306a36Sopenharmony_ci netdev_info(ndev, "at 0x%llx irq %d\n", (u64)mem->start, irq); 58662306a36Sopenharmony_ci return 0; 58762306a36Sopenharmony_ci} 58862306a36Sopenharmony_ci 58962306a36Sopenharmony_cistatic int w5300_probe(struct platform_device *pdev) 59062306a36Sopenharmony_ci{ 59162306a36Sopenharmony_ci struct w5300_priv *priv; 59262306a36Sopenharmony_ci struct net_device *ndev; 59362306a36Sopenharmony_ci int err; 59462306a36Sopenharmony_ci 59562306a36Sopenharmony_ci ndev = alloc_etherdev(sizeof(*priv)); 59662306a36Sopenharmony_ci if (!ndev) 59762306a36Sopenharmony_ci return -ENOMEM; 59862306a36Sopenharmony_ci SET_NETDEV_DEV(ndev, &pdev->dev); 59962306a36Sopenharmony_ci platform_set_drvdata(pdev, ndev); 60062306a36Sopenharmony_ci priv = netdev_priv(ndev); 60162306a36Sopenharmony_ci priv->ndev = ndev; 60262306a36Sopenharmony_ci 60362306a36Sopenharmony_ci ndev->netdev_ops = &w5300_netdev_ops; 60462306a36Sopenharmony_ci ndev->ethtool_ops = &w5300_ethtool_ops; 60562306a36Sopenharmony_ci ndev->watchdog_timeo = HZ; 60662306a36Sopenharmony_ci netif_napi_add_weight(ndev, &priv->napi, w5300_napi_poll, 16); 60762306a36Sopenharmony_ci 60862306a36Sopenharmony_ci /* This chip doesn't support VLAN packets with normal MTU, 60962306a36Sopenharmony_ci * so disable VLAN for this device. 61062306a36Sopenharmony_ci */ 61162306a36Sopenharmony_ci ndev->features |= NETIF_F_VLAN_CHALLENGED; 61262306a36Sopenharmony_ci 61362306a36Sopenharmony_ci err = register_netdev(ndev); 61462306a36Sopenharmony_ci if (err < 0) 61562306a36Sopenharmony_ci goto err_register; 61662306a36Sopenharmony_ci 61762306a36Sopenharmony_ci err = w5300_hw_probe(pdev); 61862306a36Sopenharmony_ci if (err < 0) 61962306a36Sopenharmony_ci goto err_hw_probe; 62062306a36Sopenharmony_ci 62162306a36Sopenharmony_ci return 0; 62262306a36Sopenharmony_ci 62362306a36Sopenharmony_cierr_hw_probe: 62462306a36Sopenharmony_ci unregister_netdev(ndev); 62562306a36Sopenharmony_cierr_register: 62662306a36Sopenharmony_ci free_netdev(ndev); 62762306a36Sopenharmony_ci return err; 62862306a36Sopenharmony_ci} 62962306a36Sopenharmony_ci 63062306a36Sopenharmony_cistatic int w5300_remove(struct platform_device *pdev) 63162306a36Sopenharmony_ci{ 63262306a36Sopenharmony_ci struct net_device *ndev = platform_get_drvdata(pdev); 63362306a36Sopenharmony_ci struct w5300_priv *priv = netdev_priv(ndev); 63462306a36Sopenharmony_ci 63562306a36Sopenharmony_ci w5300_hw_reset(priv); 63662306a36Sopenharmony_ci free_irq(priv->irq, ndev); 63762306a36Sopenharmony_ci if (gpio_is_valid(priv->link_gpio)) 63862306a36Sopenharmony_ci free_irq(priv->link_irq, ndev); 63962306a36Sopenharmony_ci 64062306a36Sopenharmony_ci unregister_netdev(ndev); 64162306a36Sopenharmony_ci free_netdev(ndev); 64262306a36Sopenharmony_ci return 0; 64362306a36Sopenharmony_ci} 64462306a36Sopenharmony_ci 64562306a36Sopenharmony_ci#ifdef CONFIG_PM_SLEEP 64662306a36Sopenharmony_cistatic int w5300_suspend(struct device *dev) 64762306a36Sopenharmony_ci{ 64862306a36Sopenharmony_ci struct net_device *ndev = dev_get_drvdata(dev); 64962306a36Sopenharmony_ci struct w5300_priv *priv = netdev_priv(ndev); 65062306a36Sopenharmony_ci 65162306a36Sopenharmony_ci if (netif_running(ndev)) { 65262306a36Sopenharmony_ci netif_carrier_off(ndev); 65362306a36Sopenharmony_ci netif_device_detach(ndev); 65462306a36Sopenharmony_ci 65562306a36Sopenharmony_ci w5300_hw_close(priv); 65662306a36Sopenharmony_ci } 65762306a36Sopenharmony_ci return 0; 65862306a36Sopenharmony_ci} 65962306a36Sopenharmony_ci 66062306a36Sopenharmony_cistatic int w5300_resume(struct device *dev) 66162306a36Sopenharmony_ci{ 66262306a36Sopenharmony_ci struct net_device *ndev = dev_get_drvdata(dev); 66362306a36Sopenharmony_ci struct w5300_priv *priv = netdev_priv(ndev); 66462306a36Sopenharmony_ci 66562306a36Sopenharmony_ci if (!netif_running(ndev)) { 66662306a36Sopenharmony_ci w5300_hw_reset(priv); 66762306a36Sopenharmony_ci w5300_hw_start(priv); 66862306a36Sopenharmony_ci 66962306a36Sopenharmony_ci netif_device_attach(ndev); 67062306a36Sopenharmony_ci if (!gpio_is_valid(priv->link_gpio) || 67162306a36Sopenharmony_ci gpio_get_value(priv->link_gpio) != 0) 67262306a36Sopenharmony_ci netif_carrier_on(ndev); 67362306a36Sopenharmony_ci } 67462306a36Sopenharmony_ci return 0; 67562306a36Sopenharmony_ci} 67662306a36Sopenharmony_ci#endif /* CONFIG_PM_SLEEP */ 67762306a36Sopenharmony_ci 67862306a36Sopenharmony_cistatic SIMPLE_DEV_PM_OPS(w5300_pm_ops, w5300_suspend, w5300_resume); 67962306a36Sopenharmony_ci 68062306a36Sopenharmony_cistatic struct platform_driver w5300_driver = { 68162306a36Sopenharmony_ci .driver = { 68262306a36Sopenharmony_ci .name = DRV_NAME, 68362306a36Sopenharmony_ci .pm = &w5300_pm_ops, 68462306a36Sopenharmony_ci }, 68562306a36Sopenharmony_ci .probe = w5300_probe, 68662306a36Sopenharmony_ci .remove = w5300_remove, 68762306a36Sopenharmony_ci}; 68862306a36Sopenharmony_ci 68962306a36Sopenharmony_cimodule_platform_driver(w5300_driver); 690