12dc95a4dSJoseph CHAMG // SPDX-License-Identifier: GPL-2.0-only
22dc95a4dSJoseph CHAMG /*
32dc95a4dSJoseph CHAMG  * Copyright (c) 2022 Davicom Semiconductor,Inc.
42dc95a4dSJoseph CHAMG  * Davicom DM9051 SPI Fast Ethernet Linux driver
52dc95a4dSJoseph CHAMG  */
62dc95a4dSJoseph CHAMG 
72dc95a4dSJoseph CHAMG #include <linux/etherdevice.h>
82dc95a4dSJoseph CHAMG #include <linux/ethtool.h>
92dc95a4dSJoseph CHAMG #include <linux/interrupt.h>
102dc95a4dSJoseph CHAMG #include <linux/iopoll.h>
112dc95a4dSJoseph CHAMG #include <linux/irq.h>
122dc95a4dSJoseph CHAMG #include <linux/mii.h>
132dc95a4dSJoseph CHAMG #include <linux/module.h>
142dc95a4dSJoseph CHAMG #include <linux/netdevice.h>
152dc95a4dSJoseph CHAMG #include <linux/phy.h>
162dc95a4dSJoseph CHAMG #include <linux/regmap.h>
172dc95a4dSJoseph CHAMG #include <linux/skbuff.h>
182dc95a4dSJoseph CHAMG #include <linux/spinlock.h>
192dc95a4dSJoseph CHAMG #include <linux/spi/spi.h>
202dc95a4dSJoseph CHAMG #include <linux/types.h>
212dc95a4dSJoseph CHAMG 
222dc95a4dSJoseph CHAMG #include "dm9051.h"
232dc95a4dSJoseph CHAMG 
242dc95a4dSJoseph CHAMG #define DRVNAME_9051	"dm9051"
252dc95a4dSJoseph CHAMG 
262dc95a4dSJoseph CHAMG /**
272dc95a4dSJoseph CHAMG  * struct rx_ctl_mach - rx activities record
282dc95a4dSJoseph CHAMG  * @status_err_counter: rx status error counter
292dc95a4dSJoseph CHAMG  * @large_err_counter: rx get large packet length error counter
302dc95a4dSJoseph CHAMG  * @rx_err_counter: receive packet error counter
312dc95a4dSJoseph CHAMG  * @tx_err_counter: transmit packet error counter
322dc95a4dSJoseph CHAMG  * @fifo_rst_counter: reset operation counter
332dc95a4dSJoseph CHAMG  *
342dc95a4dSJoseph CHAMG  * To keep track for the driver operation statistics
352dc95a4dSJoseph CHAMG  */
362dc95a4dSJoseph CHAMG struct rx_ctl_mach {
372dc95a4dSJoseph CHAMG 	u16				status_err_counter;
382dc95a4dSJoseph CHAMG 	u16				large_err_counter;
392dc95a4dSJoseph CHAMG 	u16				rx_err_counter;
402dc95a4dSJoseph CHAMG 	u16				tx_err_counter;
412dc95a4dSJoseph CHAMG 	u16				fifo_rst_counter;
422dc95a4dSJoseph CHAMG };
432dc95a4dSJoseph CHAMG 
442dc95a4dSJoseph CHAMG /**
452dc95a4dSJoseph CHAMG  * struct dm9051_rxctrl - dm9051 driver rx control
462dc95a4dSJoseph CHAMG  * @hash_table: Multicast hash-table data
472dc95a4dSJoseph CHAMG  * @rcr_all: KS_RXCR1 register setting
482dc95a4dSJoseph CHAMG  *
492dc95a4dSJoseph CHAMG  * The settings needs to control the receive filtering
502dc95a4dSJoseph CHAMG  * such as the multicast hash-filter and the receive register settings
512dc95a4dSJoseph CHAMG  */
522dc95a4dSJoseph CHAMG struct dm9051_rxctrl {
532dc95a4dSJoseph CHAMG 	u16				hash_table[4];
542dc95a4dSJoseph CHAMG 	u8				rcr_all;
552dc95a4dSJoseph CHAMG };
562dc95a4dSJoseph CHAMG 
572dc95a4dSJoseph CHAMG /**
582dc95a4dSJoseph CHAMG  * struct dm9051_rxhdr - rx packet data header
592dc95a4dSJoseph CHAMG  * @headbyte: lead byte equal to 0x01 notifies a valid packet
602dc95a4dSJoseph CHAMG  * @status: status bits for the received packet
612dc95a4dSJoseph CHAMG  * @rxlen: packet length
622dc95a4dSJoseph CHAMG  *
632dc95a4dSJoseph CHAMG  * The Rx packed, entered into the FIFO memory, start with these
642dc95a4dSJoseph CHAMG  * four bytes which is the Rx header, followed by the ethernet
652dc95a4dSJoseph CHAMG  * packet data and ends with an appended 4-byte CRC data.
662dc95a4dSJoseph CHAMG  * Both Rx packet and CRC data are for check purpose and finally
672dc95a4dSJoseph CHAMG  * are dropped by this driver
682dc95a4dSJoseph CHAMG  */
692dc95a4dSJoseph CHAMG struct dm9051_rxhdr {
702dc95a4dSJoseph CHAMG 	u8				headbyte;
712dc95a4dSJoseph CHAMG 	u8				status;
722dc95a4dSJoseph CHAMG 	__le16				rxlen;
732dc95a4dSJoseph CHAMG };
742dc95a4dSJoseph CHAMG 
752dc95a4dSJoseph CHAMG /**
762dc95a4dSJoseph CHAMG  * struct board_info - maintain the saved data
772dc95a4dSJoseph CHAMG  * @spidev: spi device structure
782dc95a4dSJoseph CHAMG  * @ndev: net device structure
792dc95a4dSJoseph CHAMG  * @mdiobus: mii bus structure
802dc95a4dSJoseph CHAMG  * @phydev: phy device structure
812dc95a4dSJoseph CHAMG  * @txq: tx queue structure
822dc95a4dSJoseph CHAMG  * @regmap_dm: regmap for register read/write
832dc95a4dSJoseph CHAMG  * @regmap_dmbulk: extra regmap for bulk read/write
842dc95a4dSJoseph CHAMG  * @rxctrl_work: Work queue for updating RX mode and multicast lists
852dc95a4dSJoseph CHAMG  * @tx_work: Work queue for tx packets
862dc95a4dSJoseph CHAMG  * @pause: ethtool pause parameter structure
872dc95a4dSJoseph CHAMG  * @spi_lockm: between threads lock structure
882dc95a4dSJoseph CHAMG  * @reg_mutex: regmap access lock structure
892dc95a4dSJoseph CHAMG  * @bc: rx control statistics structure
902dc95a4dSJoseph CHAMG  * @rxhdr: rx header structure
912dc95a4dSJoseph CHAMG  * @rctl: rx control setting structure
922dc95a4dSJoseph CHAMG  * @msg_enable: message level value
932dc95a4dSJoseph CHAMG  * @imr_all: to store operating imr value for register DM9051_IMR
942dc95a4dSJoseph CHAMG  * @lcr_all: to store operating rcr value for register DM9051_LMCR
952dc95a4dSJoseph CHAMG  *
962dc95a4dSJoseph CHAMG  * The saved data variables, keep up to date for retrieval back to use
972dc95a4dSJoseph CHAMG  */
982dc95a4dSJoseph CHAMG struct board_info {
992dc95a4dSJoseph CHAMG 	u32				msg_enable;
1002dc95a4dSJoseph CHAMG 	struct spi_device		*spidev;
1012dc95a4dSJoseph CHAMG 	struct net_device		*ndev;
1022dc95a4dSJoseph CHAMG 	struct mii_bus			*mdiobus;
1032dc95a4dSJoseph CHAMG 	struct phy_device		*phydev;
1042dc95a4dSJoseph CHAMG 	struct sk_buff_head		txq;
1052dc95a4dSJoseph CHAMG 	struct regmap			*regmap_dm;
1062dc95a4dSJoseph CHAMG 	struct regmap			*regmap_dmbulk;
1072dc95a4dSJoseph CHAMG 	struct work_struct		rxctrl_work;
1082dc95a4dSJoseph CHAMG 	struct work_struct		tx_work;
1092dc95a4dSJoseph CHAMG 	struct ethtool_pauseparam	pause;
1102dc95a4dSJoseph CHAMG 	struct mutex			spi_lockm;
1112dc95a4dSJoseph CHAMG 	struct mutex			reg_mutex;
1122dc95a4dSJoseph CHAMG 	struct rx_ctl_mach		bc;
1132dc95a4dSJoseph CHAMG 	struct dm9051_rxhdr		rxhdr;
1142dc95a4dSJoseph CHAMG 	struct dm9051_rxctrl		rctl;
1152dc95a4dSJoseph CHAMG 	u8				imr_all;
1162dc95a4dSJoseph CHAMG 	u8				lcr_all;
1172dc95a4dSJoseph CHAMG };
1182dc95a4dSJoseph CHAMG 
dm9051_set_reg(struct board_info * db,unsigned int reg,unsigned int val)1192dc95a4dSJoseph CHAMG static int dm9051_set_reg(struct board_info *db, unsigned int reg, unsigned int val)
1202dc95a4dSJoseph CHAMG {
1212dc95a4dSJoseph CHAMG 	int ret;
1222dc95a4dSJoseph CHAMG 
1232dc95a4dSJoseph CHAMG 	ret = regmap_write(db->regmap_dm, reg, val);
1242dc95a4dSJoseph CHAMG 	if (ret < 0)
1252dc95a4dSJoseph CHAMG 		netif_err(db, drv, db->ndev, "%s: error %d set reg %02x\n",
1262dc95a4dSJoseph CHAMG 			  __func__, ret, reg);
1272dc95a4dSJoseph CHAMG 	return ret;
1282dc95a4dSJoseph CHAMG }
1292dc95a4dSJoseph CHAMG 
dm9051_update_bits(struct board_info * db,unsigned int reg,unsigned int mask,unsigned int val)1302dc95a4dSJoseph CHAMG static int dm9051_update_bits(struct board_info *db, unsigned int reg, unsigned int mask,
1312dc95a4dSJoseph CHAMG 			      unsigned int val)
1322dc95a4dSJoseph CHAMG {
1332dc95a4dSJoseph CHAMG 	int ret;
1342dc95a4dSJoseph CHAMG 
1352dc95a4dSJoseph CHAMG 	ret = regmap_update_bits(db->regmap_dm, reg, mask, val);
1362dc95a4dSJoseph CHAMG 	if (ret < 0)
1372dc95a4dSJoseph CHAMG 		netif_err(db, drv, db->ndev, "%s: error %d update bits reg %02x\n",
1382dc95a4dSJoseph CHAMG 			  __func__, ret, reg);
1392dc95a4dSJoseph CHAMG 	return ret;
1402dc95a4dSJoseph CHAMG }
1412dc95a4dSJoseph CHAMG 
1422dc95a4dSJoseph CHAMG /* skb buffer exhausted, just discard the received data
1432dc95a4dSJoseph CHAMG  */
dm9051_dumpblk(struct board_info * db,u8 reg,size_t count)1442dc95a4dSJoseph CHAMG static int dm9051_dumpblk(struct board_info *db, u8 reg, size_t count)
1452dc95a4dSJoseph CHAMG {
1462dc95a4dSJoseph CHAMG 	struct net_device *ndev = db->ndev;
1472dc95a4dSJoseph CHAMG 	unsigned int rb;
1482dc95a4dSJoseph CHAMG 	int ret;
1492dc95a4dSJoseph CHAMG 
1502dc95a4dSJoseph CHAMG 	/* no skb buffer,
1512dc95a4dSJoseph CHAMG 	 * both reg and &rb must be noinc,
1522dc95a4dSJoseph CHAMG 	 * read once one byte via regmap_read
1532dc95a4dSJoseph CHAMG 	 */
1542dc95a4dSJoseph CHAMG 	do {
1552dc95a4dSJoseph CHAMG 		ret = regmap_read(db->regmap_dm, reg, &rb);
1562dc95a4dSJoseph CHAMG 		if (ret < 0) {
1572dc95a4dSJoseph CHAMG 			netif_err(db, drv, ndev, "%s: error %d dumping read reg %02x\n",
1582dc95a4dSJoseph CHAMG 				  __func__, ret, reg);
1592dc95a4dSJoseph CHAMG 			break;
1602dc95a4dSJoseph CHAMG 		}
1612dc95a4dSJoseph CHAMG 	} while (--count);
1622dc95a4dSJoseph CHAMG 
1632dc95a4dSJoseph CHAMG 	return ret;
1642dc95a4dSJoseph CHAMG }
1652dc95a4dSJoseph CHAMG 
dm9051_set_regs(struct board_info * db,unsigned int reg,const void * val,size_t val_count)1662dc95a4dSJoseph CHAMG static int dm9051_set_regs(struct board_info *db, unsigned int reg, const void *val,
1672dc95a4dSJoseph CHAMG 			   size_t val_count)
1682dc95a4dSJoseph CHAMG {
1692dc95a4dSJoseph CHAMG 	int ret;
1702dc95a4dSJoseph CHAMG 
1712dc95a4dSJoseph CHAMG 	ret = regmap_bulk_write(db->regmap_dmbulk, reg, val, val_count);
1722dc95a4dSJoseph CHAMG 	if (ret < 0)
1732dc95a4dSJoseph CHAMG 		netif_err(db, drv, db->ndev, "%s: error %d bulk writing regs %02x\n",
1742dc95a4dSJoseph CHAMG 			  __func__, ret, reg);
1752dc95a4dSJoseph CHAMG 	return ret;
1762dc95a4dSJoseph CHAMG }
1772dc95a4dSJoseph CHAMG 
dm9051_get_regs(struct board_info * db,unsigned int reg,void * val,size_t val_count)1782dc95a4dSJoseph CHAMG static int dm9051_get_regs(struct board_info *db, unsigned int reg, void *val,
1792dc95a4dSJoseph CHAMG 			   size_t val_count)
1802dc95a4dSJoseph CHAMG {
1812dc95a4dSJoseph CHAMG 	int ret;
1822dc95a4dSJoseph CHAMG 
1832dc95a4dSJoseph CHAMG 	ret = regmap_bulk_read(db->regmap_dmbulk, reg, val, val_count);
1842dc95a4dSJoseph CHAMG 	if (ret < 0)
1852dc95a4dSJoseph CHAMG 		netif_err(db, drv, db->ndev, "%s: error %d bulk reading regs %02x\n",
1862dc95a4dSJoseph CHAMG 			  __func__, ret, reg);
1872dc95a4dSJoseph CHAMG 	return ret;
1882dc95a4dSJoseph CHAMG }
1892dc95a4dSJoseph CHAMG 
dm9051_write_mem(struct board_info * db,unsigned int reg,const void * buff,size_t len)1902dc95a4dSJoseph CHAMG static int dm9051_write_mem(struct board_info *db, unsigned int reg, const void *buff,
1912dc95a4dSJoseph CHAMG 			    size_t len)
1922dc95a4dSJoseph CHAMG {
1932dc95a4dSJoseph CHAMG 	int ret;
1942dc95a4dSJoseph CHAMG 
1952dc95a4dSJoseph CHAMG 	ret = regmap_noinc_write(db->regmap_dm, reg, buff, len);
1962dc95a4dSJoseph CHAMG 	if (ret < 0)
1972dc95a4dSJoseph CHAMG 		netif_err(db, drv, db->ndev, "%s: error %d noinc writing regs %02x\n",
1982dc95a4dSJoseph CHAMG 			  __func__, ret, reg);
1992dc95a4dSJoseph CHAMG 	return ret;
2002dc95a4dSJoseph CHAMG }
2012dc95a4dSJoseph CHAMG 
dm9051_read_mem(struct board_info * db,unsigned int reg,void * buff,size_t len)2022dc95a4dSJoseph CHAMG static int dm9051_read_mem(struct board_info *db, unsigned int reg, void *buff,
2032dc95a4dSJoseph CHAMG 			   size_t len)
2042dc95a4dSJoseph CHAMG {
2052dc95a4dSJoseph CHAMG 	int ret;
2062dc95a4dSJoseph CHAMG 
2072dc95a4dSJoseph CHAMG 	ret = regmap_noinc_read(db->regmap_dm, reg, buff, len);
2082dc95a4dSJoseph CHAMG 	if (ret < 0)
2092dc95a4dSJoseph CHAMG 		netif_err(db, drv, db->ndev, "%s: error %d noinc reading regs %02x\n",
2102dc95a4dSJoseph CHAMG 			  __func__, ret, reg);
2112dc95a4dSJoseph CHAMG 	return ret;
2122dc95a4dSJoseph CHAMG }
2132dc95a4dSJoseph CHAMG 
2142dc95a4dSJoseph CHAMG /* waiting tx-end rather than tx-req
2152dc95a4dSJoseph CHAMG  * got faster
2162dc95a4dSJoseph CHAMG  */
dm9051_nsr_poll(struct board_info * db)2172dc95a4dSJoseph CHAMG static int dm9051_nsr_poll(struct board_info *db)
2182dc95a4dSJoseph CHAMG {
2192dc95a4dSJoseph CHAMG 	unsigned int mval;
2202dc95a4dSJoseph CHAMG 	int ret;
2212dc95a4dSJoseph CHAMG 
2222dc95a4dSJoseph CHAMG 	ret = regmap_read_poll_timeout(db->regmap_dm, DM9051_NSR, mval,
2232dc95a4dSJoseph CHAMG 				       mval & (NSR_TX2END | NSR_TX1END), 1, 20);
2242dc95a4dSJoseph CHAMG 	if (ret == -ETIMEDOUT)
2252dc95a4dSJoseph CHAMG 		netdev_err(db->ndev, "timeout in checking for tx end\n");
2262dc95a4dSJoseph CHAMG 	return ret;
2272dc95a4dSJoseph CHAMG }
2282dc95a4dSJoseph CHAMG 
dm9051_epcr_poll(struct board_info * db)2292dc95a4dSJoseph CHAMG static int dm9051_epcr_poll(struct board_info *db)
2302dc95a4dSJoseph CHAMG {
2312dc95a4dSJoseph CHAMG 	unsigned int mval;
2322dc95a4dSJoseph CHAMG 	int ret;
2332dc95a4dSJoseph CHAMG 
2342dc95a4dSJoseph CHAMG 	ret = regmap_read_poll_timeout(db->regmap_dm, DM9051_EPCR, mval,
2352dc95a4dSJoseph CHAMG 				       !(mval & EPCR_ERRE), 100, 10000);
2362dc95a4dSJoseph CHAMG 	if (ret == -ETIMEDOUT)
2372dc95a4dSJoseph CHAMG 		netdev_err(db->ndev, "eeprom/phy in processing get timeout\n");
2382dc95a4dSJoseph CHAMG 	return ret;
2392dc95a4dSJoseph CHAMG }
2402dc95a4dSJoseph CHAMG 
dm9051_irq_flag(struct board_info * db)2412dc95a4dSJoseph CHAMG static int dm9051_irq_flag(struct board_info *db)
2422dc95a4dSJoseph CHAMG {
2432dc95a4dSJoseph CHAMG 	struct spi_device *spi = db->spidev;
2442dc95a4dSJoseph CHAMG 	int irq_type = irq_get_trigger_type(spi->irq);
2452dc95a4dSJoseph CHAMG 
2462dc95a4dSJoseph CHAMG 	if (irq_type)
2472dc95a4dSJoseph CHAMG 		return irq_type;
2482dc95a4dSJoseph CHAMG 
2492dc95a4dSJoseph CHAMG 	return IRQF_TRIGGER_LOW;
2502dc95a4dSJoseph CHAMG }
2512dc95a4dSJoseph CHAMG 
dm9051_intcr_value(struct board_info * db)2522dc95a4dSJoseph CHAMG static unsigned int dm9051_intcr_value(struct board_info *db)
2532dc95a4dSJoseph CHAMG {
2542dc95a4dSJoseph CHAMG 	return (dm9051_irq_flag(db) == IRQF_TRIGGER_LOW) ?
2552dc95a4dSJoseph CHAMG 		INTCR_POL_LOW : INTCR_POL_HIGH;
2562dc95a4dSJoseph CHAMG }
2572dc95a4dSJoseph CHAMG 
dm9051_set_fcr(struct board_info * db)2582dc95a4dSJoseph CHAMG static int dm9051_set_fcr(struct board_info *db)
2592dc95a4dSJoseph CHAMG {
2602dc95a4dSJoseph CHAMG 	u8 fcr = 0;
2612dc95a4dSJoseph CHAMG 
2622dc95a4dSJoseph CHAMG 	if (db->pause.rx_pause)
2632dc95a4dSJoseph CHAMG 		fcr |= FCR_BKPM | FCR_FLCE;
2642dc95a4dSJoseph CHAMG 	if (db->pause.tx_pause)
2652dc95a4dSJoseph CHAMG 		fcr |= FCR_TXPEN;
2662dc95a4dSJoseph CHAMG 
2672dc95a4dSJoseph CHAMG 	return dm9051_set_reg(db, DM9051_FCR, fcr);
2682dc95a4dSJoseph CHAMG }
2692dc95a4dSJoseph CHAMG 
dm9051_set_recv(struct board_info * db)2702dc95a4dSJoseph CHAMG static int dm9051_set_recv(struct board_info *db)
2712dc95a4dSJoseph CHAMG {
2722dc95a4dSJoseph CHAMG 	int ret;
2732dc95a4dSJoseph CHAMG 
2742dc95a4dSJoseph CHAMG 	ret = dm9051_set_regs(db, DM9051_MAR, db->rctl.hash_table, sizeof(db->rctl.hash_table));
2752dc95a4dSJoseph CHAMG 	if (ret)
2762dc95a4dSJoseph CHAMG 		return ret;
2772dc95a4dSJoseph CHAMG 
2782dc95a4dSJoseph CHAMG 	return dm9051_set_reg(db, DM9051_RCR, db->rctl.rcr_all); /* enable rx */
2792dc95a4dSJoseph CHAMG }
2802dc95a4dSJoseph CHAMG 
dm9051_core_reset(struct board_info * db)2812dc95a4dSJoseph CHAMG static int dm9051_core_reset(struct board_info *db)
2822dc95a4dSJoseph CHAMG {
2832dc95a4dSJoseph CHAMG 	int ret;
2842dc95a4dSJoseph CHAMG 
2852dc95a4dSJoseph CHAMG 	db->bc.fifo_rst_counter++;
2862dc95a4dSJoseph CHAMG 
2872dc95a4dSJoseph CHAMG 	ret = regmap_write(db->regmap_dm, DM9051_NCR, NCR_RST); /* NCR reset */
2882dc95a4dSJoseph CHAMG 	if (ret)
2892dc95a4dSJoseph CHAMG 		return ret;
2902dc95a4dSJoseph CHAMG 	ret = regmap_write(db->regmap_dm, DM9051_MBNDRY, MBNDRY_BYTE); /* MemBound */
2912dc95a4dSJoseph CHAMG 	if (ret)
2922dc95a4dSJoseph CHAMG 		return ret;
2932dc95a4dSJoseph CHAMG 	ret = regmap_write(db->regmap_dm, DM9051_PPCR, PPCR_PAUSE_COUNT); /* Pause Count */
2942dc95a4dSJoseph CHAMG 	if (ret)
2952dc95a4dSJoseph CHAMG 		return ret;
2962dc95a4dSJoseph CHAMG 	ret = regmap_write(db->regmap_dm, DM9051_LMCR, db->lcr_all); /* LEDMode1 */
2972dc95a4dSJoseph CHAMG 	if (ret)
2982dc95a4dSJoseph CHAMG 		return ret;
2992dc95a4dSJoseph CHAMG 
3002dc95a4dSJoseph CHAMG 	return dm9051_set_reg(db, DM9051_INTCR, dm9051_intcr_value(db));
3012dc95a4dSJoseph CHAMG }
3022dc95a4dSJoseph CHAMG 
dm9051_update_fcr(struct board_info * db)3032dc95a4dSJoseph CHAMG static int dm9051_update_fcr(struct board_info *db)
3042dc95a4dSJoseph CHAMG {
3052dc95a4dSJoseph CHAMG 	u8 fcr = 0;
3062dc95a4dSJoseph CHAMG 
3072dc95a4dSJoseph CHAMG 	if (db->pause.rx_pause)
3082dc95a4dSJoseph CHAMG 		fcr |= FCR_BKPM | FCR_FLCE;
3092dc95a4dSJoseph CHAMG 	if (db->pause.tx_pause)
3102dc95a4dSJoseph CHAMG 		fcr |= FCR_TXPEN;
3112dc95a4dSJoseph CHAMG 
3122dc95a4dSJoseph CHAMG 	return dm9051_update_bits(db, DM9051_FCR, FCR_RXTX_BITS, fcr);
3132dc95a4dSJoseph CHAMG }
3142dc95a4dSJoseph CHAMG 
dm9051_disable_interrupt(struct board_info * db)3152dc95a4dSJoseph CHAMG static int dm9051_disable_interrupt(struct board_info *db)
3162dc95a4dSJoseph CHAMG {
3172dc95a4dSJoseph CHAMG 	return dm9051_set_reg(db, DM9051_IMR, IMR_PAR); /* disable int */
3182dc95a4dSJoseph CHAMG }
3192dc95a4dSJoseph CHAMG 
dm9051_enable_interrupt(struct board_info * db)3202dc95a4dSJoseph CHAMG static int dm9051_enable_interrupt(struct board_info *db)
3212dc95a4dSJoseph CHAMG {
3222dc95a4dSJoseph CHAMG 	return dm9051_set_reg(db, DM9051_IMR, db->imr_all); /* enable int */
3232dc95a4dSJoseph CHAMG }
3242dc95a4dSJoseph CHAMG 
dm9051_stop_mrcmd(struct board_info * db)3252dc95a4dSJoseph CHAMG static int dm9051_stop_mrcmd(struct board_info *db)
3262dc95a4dSJoseph CHAMG {
3272dc95a4dSJoseph CHAMG 	return dm9051_set_reg(db, DM9051_ISR, ISR_STOP_MRCMD); /* to stop mrcmd */
3282dc95a4dSJoseph CHAMG }
3292dc95a4dSJoseph CHAMG 
dm9051_clear_interrupt(struct board_info * db)3302dc95a4dSJoseph CHAMG static int dm9051_clear_interrupt(struct board_info *db)
3312dc95a4dSJoseph CHAMG {
3322dc95a4dSJoseph CHAMG 	return dm9051_update_bits(db, DM9051_ISR, ISR_CLR_INT, ISR_CLR_INT);
3332dc95a4dSJoseph CHAMG }
3342dc95a4dSJoseph CHAMG 
dm9051_eeprom_read(struct board_info * db,int offset,u8 * to)3352dc95a4dSJoseph CHAMG static int dm9051_eeprom_read(struct board_info *db, int offset, u8 *to)
3362dc95a4dSJoseph CHAMG {
3372dc95a4dSJoseph CHAMG 	int ret;
3382dc95a4dSJoseph CHAMG 
3392dc95a4dSJoseph CHAMG 	ret = regmap_write(db->regmap_dm, DM9051_EPAR, offset);
3402dc95a4dSJoseph CHAMG 	if (ret)
3412dc95a4dSJoseph CHAMG 		return ret;
3422dc95a4dSJoseph CHAMG 
3432dc95a4dSJoseph CHAMG 	ret = regmap_write(db->regmap_dm, DM9051_EPCR, EPCR_ERPRR);
3442dc95a4dSJoseph CHAMG 	if (ret)
3452dc95a4dSJoseph CHAMG 		return ret;
3462dc95a4dSJoseph CHAMG 
3472dc95a4dSJoseph CHAMG 	ret = dm9051_epcr_poll(db);
3482dc95a4dSJoseph CHAMG 	if (ret)
3492dc95a4dSJoseph CHAMG 		return ret;
3502dc95a4dSJoseph CHAMG 
3512dc95a4dSJoseph CHAMG 	ret = regmap_write(db->regmap_dm, DM9051_EPCR, 0);
3522dc95a4dSJoseph CHAMG 	if (ret)
3532dc95a4dSJoseph CHAMG 		return ret;
3542dc95a4dSJoseph CHAMG 
3552dc95a4dSJoseph CHAMG 	return regmap_bulk_read(db->regmap_dmbulk, DM9051_EPDRL, to, 2);
3562dc95a4dSJoseph CHAMG }
3572dc95a4dSJoseph CHAMG 
dm9051_eeprom_write(struct board_info * db,int offset,u8 * data)3582dc95a4dSJoseph CHAMG static int dm9051_eeprom_write(struct board_info *db, int offset, u8 *data)
3592dc95a4dSJoseph CHAMG {
3602dc95a4dSJoseph CHAMG 	int ret;
3612dc95a4dSJoseph CHAMG 
3622dc95a4dSJoseph CHAMG 	ret = regmap_write(db->regmap_dm, DM9051_EPAR, offset);
3632dc95a4dSJoseph CHAMG 	if (ret)
3642dc95a4dSJoseph CHAMG 		return ret;
3652dc95a4dSJoseph CHAMG 
3662dc95a4dSJoseph CHAMG 	ret = regmap_bulk_write(db->regmap_dmbulk, DM9051_EPDRL, data, 2);
3672dc95a4dSJoseph CHAMG 	if (ret < 0)
3682dc95a4dSJoseph CHAMG 		return ret;
3692dc95a4dSJoseph CHAMG 
3702dc95a4dSJoseph CHAMG 	ret = regmap_write(db->regmap_dm, DM9051_EPCR, EPCR_WEP | EPCR_ERPRW);
3712dc95a4dSJoseph CHAMG 	if (ret)
3722dc95a4dSJoseph CHAMG 		return ret;
3732dc95a4dSJoseph CHAMG 
3742dc95a4dSJoseph CHAMG 	ret = dm9051_epcr_poll(db);
3752dc95a4dSJoseph CHAMG 	if (ret)
3762dc95a4dSJoseph CHAMG 		return ret;
3772dc95a4dSJoseph CHAMG 
3782dc95a4dSJoseph CHAMG 	return regmap_write(db->regmap_dm, DM9051_EPCR, 0);
3792dc95a4dSJoseph CHAMG }
3802dc95a4dSJoseph CHAMG 
dm9051_phyread(void * context,unsigned int reg,unsigned int * val)3812dc95a4dSJoseph CHAMG static int dm9051_phyread(void *context, unsigned int reg, unsigned int *val)
3822dc95a4dSJoseph CHAMG {
3832dc95a4dSJoseph CHAMG 	struct board_info *db = context;
3842dc95a4dSJoseph CHAMG 	int ret;
3852dc95a4dSJoseph CHAMG 
3862dc95a4dSJoseph CHAMG 	ret = regmap_write(db->regmap_dm, DM9051_EPAR, DM9051_PHY | reg);
3872dc95a4dSJoseph CHAMG 	if (ret)
3882dc95a4dSJoseph CHAMG 		return ret;
3892dc95a4dSJoseph CHAMG 
3902dc95a4dSJoseph CHAMG 	ret = regmap_write(db->regmap_dm, DM9051_EPCR, EPCR_ERPRR | EPCR_EPOS);
3912dc95a4dSJoseph CHAMG 	if (ret)
3922dc95a4dSJoseph CHAMG 		return ret;
3932dc95a4dSJoseph CHAMG 
3942dc95a4dSJoseph CHAMG 	ret = dm9051_epcr_poll(db);
3952dc95a4dSJoseph CHAMG 	if (ret)
3962dc95a4dSJoseph CHAMG 		return ret;
3972dc95a4dSJoseph CHAMG 
3982dc95a4dSJoseph CHAMG 	ret = regmap_write(db->regmap_dm, DM9051_EPCR, 0);
3992dc95a4dSJoseph CHAMG 	if (ret)
4002dc95a4dSJoseph CHAMG 		return ret;
4012dc95a4dSJoseph CHAMG 
4022dc95a4dSJoseph CHAMG 	/* this is a 4 bytes data, clear to zero since following regmap_bulk_read
4032dc95a4dSJoseph CHAMG 	 * only fill lower 2 bytes
4042dc95a4dSJoseph CHAMG 	 */
4052dc95a4dSJoseph CHAMG 	*val = 0;
4062dc95a4dSJoseph CHAMG 	return regmap_bulk_read(db->regmap_dmbulk, DM9051_EPDRL, val, 2);
4072dc95a4dSJoseph CHAMG }
4082dc95a4dSJoseph CHAMG 
dm9051_phywrite(void * context,unsigned int reg,unsigned int val)4092dc95a4dSJoseph CHAMG static int dm9051_phywrite(void *context, unsigned int reg, unsigned int val)
4102dc95a4dSJoseph CHAMG {
4112dc95a4dSJoseph CHAMG 	struct board_info *db = context;
4122dc95a4dSJoseph CHAMG 	int ret;
4132dc95a4dSJoseph CHAMG 
4142dc95a4dSJoseph CHAMG 	ret = regmap_write(db->regmap_dm, DM9051_EPAR, DM9051_PHY | reg);
4152dc95a4dSJoseph CHAMG 	if (ret)
4162dc95a4dSJoseph CHAMG 		return ret;
4172dc95a4dSJoseph CHAMG 
4182dc95a4dSJoseph CHAMG 	ret = regmap_bulk_write(db->regmap_dmbulk, DM9051_EPDRL, &val, 2);
4192dc95a4dSJoseph CHAMG 	if (ret < 0)
4202dc95a4dSJoseph CHAMG 		return ret;
4212dc95a4dSJoseph CHAMG 
4222dc95a4dSJoseph CHAMG 	ret = regmap_write(db->regmap_dm, DM9051_EPCR, EPCR_EPOS | EPCR_ERPRW);
4232dc95a4dSJoseph CHAMG 	if (ret)
4242dc95a4dSJoseph CHAMG 		return ret;
4252dc95a4dSJoseph CHAMG 
4262dc95a4dSJoseph CHAMG 	ret = dm9051_epcr_poll(db);
4272dc95a4dSJoseph CHAMG 	if (ret)
4282dc95a4dSJoseph CHAMG 		return ret;
4292dc95a4dSJoseph CHAMG 
4302dc95a4dSJoseph CHAMG 	return regmap_write(db->regmap_dm, DM9051_EPCR, 0);
4312dc95a4dSJoseph CHAMG }
4322dc95a4dSJoseph CHAMG 
dm9051_mdio_read(struct mii_bus * bus,int addr,int regnum)4332dc95a4dSJoseph CHAMG static int dm9051_mdio_read(struct mii_bus *bus, int addr, int regnum)
4342dc95a4dSJoseph CHAMG {
4352dc95a4dSJoseph CHAMG 	struct board_info *db = bus->priv;
4362dc95a4dSJoseph CHAMG 	unsigned int val = 0xffff;
4372dc95a4dSJoseph CHAMG 	int ret;
4382dc95a4dSJoseph CHAMG 
4392dc95a4dSJoseph CHAMG 	if (addr == DM9051_PHY_ADDR) {
4402dc95a4dSJoseph CHAMG 		ret = dm9051_phyread(db, regnum, &val);
4412dc95a4dSJoseph CHAMG 		if (ret)
4422dc95a4dSJoseph CHAMG 			return ret;
4432dc95a4dSJoseph CHAMG 	}
4442dc95a4dSJoseph CHAMG 
4452dc95a4dSJoseph CHAMG 	return val;
4462dc95a4dSJoseph CHAMG }
4472dc95a4dSJoseph CHAMG 
dm9051_mdio_write(struct mii_bus * bus,int addr,int regnum,u16 val)4482dc95a4dSJoseph CHAMG static int dm9051_mdio_write(struct mii_bus *bus, int addr, int regnum, u16 val)
4492dc95a4dSJoseph CHAMG {
4502dc95a4dSJoseph CHAMG 	struct board_info *db = bus->priv;
4512dc95a4dSJoseph CHAMG 
4522dc95a4dSJoseph CHAMG 	if (addr == DM9051_PHY_ADDR)
4532dc95a4dSJoseph CHAMG 		return dm9051_phywrite(db, regnum, val);
4542dc95a4dSJoseph CHAMG 
4552dc95a4dSJoseph CHAMG 	return -ENODEV;
4562dc95a4dSJoseph CHAMG }
4572dc95a4dSJoseph CHAMG 
dm9051_reg_lock_mutex(void * dbcontext)4582dc95a4dSJoseph CHAMG static void dm9051_reg_lock_mutex(void *dbcontext)
4592dc95a4dSJoseph CHAMG {
4602dc95a4dSJoseph CHAMG 	struct board_info *db = dbcontext;
4612dc95a4dSJoseph CHAMG 
4622dc95a4dSJoseph CHAMG 	mutex_lock(&db->reg_mutex);
4632dc95a4dSJoseph CHAMG }
4642dc95a4dSJoseph CHAMG 
dm9051_reg_unlock_mutex(void * dbcontext)4652dc95a4dSJoseph CHAMG static void dm9051_reg_unlock_mutex(void *dbcontext)
4662dc95a4dSJoseph CHAMG {
4672dc95a4dSJoseph CHAMG 	struct board_info *db = dbcontext;
4682dc95a4dSJoseph CHAMG 
4692dc95a4dSJoseph CHAMG 	mutex_unlock(&db->reg_mutex);
4702dc95a4dSJoseph CHAMG }
4712dc95a4dSJoseph CHAMG 
4722dc95a4dSJoseph CHAMG static struct regmap_config regconfigdm = {
4732dc95a4dSJoseph CHAMG 	.reg_bits = 8,
4742dc95a4dSJoseph CHAMG 	.val_bits = 8,
4752dc95a4dSJoseph CHAMG 	.max_register = 0xff,
4762dc95a4dSJoseph CHAMG 	.reg_stride = 1,
4772dc95a4dSJoseph CHAMG 	.cache_type = REGCACHE_NONE,
4782dc95a4dSJoseph CHAMG 	.read_flag_mask = 0,
4792dc95a4dSJoseph CHAMG 	.write_flag_mask = DM_SPI_WR,
4802dc95a4dSJoseph CHAMG 	.val_format_endian = REGMAP_ENDIAN_LITTLE,
4812dc95a4dSJoseph CHAMG 	.lock = dm9051_reg_lock_mutex,
4822dc95a4dSJoseph CHAMG 	.unlock = dm9051_reg_unlock_mutex,
4832dc95a4dSJoseph CHAMG };
4842dc95a4dSJoseph CHAMG 
4852dc95a4dSJoseph CHAMG static struct regmap_config regconfigdmbulk = {
4862dc95a4dSJoseph CHAMG 	.reg_bits = 8,
4872dc95a4dSJoseph CHAMG 	.val_bits = 8,
4882dc95a4dSJoseph CHAMG 	.max_register = 0xff,
4892dc95a4dSJoseph CHAMG 	.reg_stride = 1,
4902dc95a4dSJoseph CHAMG 	.cache_type = REGCACHE_NONE,
4912dc95a4dSJoseph CHAMG 	.read_flag_mask = 0,
4922dc95a4dSJoseph CHAMG 	.write_flag_mask = DM_SPI_WR,
4932dc95a4dSJoseph CHAMG 	.val_format_endian = REGMAP_ENDIAN_LITTLE,
4942dc95a4dSJoseph CHAMG 	.lock = dm9051_reg_lock_mutex,
4952dc95a4dSJoseph CHAMG 	.unlock = dm9051_reg_unlock_mutex,
4962dc95a4dSJoseph CHAMG 	.use_single_read = true,
4972dc95a4dSJoseph CHAMG 	.use_single_write = true,
4982dc95a4dSJoseph CHAMG };
4992dc95a4dSJoseph CHAMG 
dm9051_map_init(struct spi_device * spi,struct board_info * db)5002dc95a4dSJoseph CHAMG static int dm9051_map_init(struct spi_device *spi, struct board_info *db)
5012dc95a4dSJoseph CHAMG {
5022dc95a4dSJoseph CHAMG 	/* create two regmap instances,
5032dc95a4dSJoseph CHAMG 	 * split read/write and bulk_read/bulk_write to individual regmap
5042dc95a4dSJoseph CHAMG 	 * to resolve regmap execution confliction problem
5052dc95a4dSJoseph CHAMG 	 */
5062dc95a4dSJoseph CHAMG 	regconfigdm.lock_arg = db;
5072dc95a4dSJoseph CHAMG 	db->regmap_dm = devm_regmap_init_spi(db->spidev, &regconfigdm);
5082dc95a4dSJoseph CHAMG 	if (IS_ERR(db->regmap_dm))
5092dc95a4dSJoseph CHAMG 		return PTR_ERR(db->regmap_dm);
5102dc95a4dSJoseph CHAMG 
5112dc95a4dSJoseph CHAMG 	regconfigdmbulk.lock_arg = db;
5122dc95a4dSJoseph CHAMG 	db->regmap_dmbulk = devm_regmap_init_spi(db->spidev, &regconfigdmbulk);
513*829b3357SRuan Jinjie 	return PTR_ERR_OR_ZERO(db->regmap_dmbulk);
5142dc95a4dSJoseph CHAMG }
5152dc95a4dSJoseph CHAMG 
dm9051_map_chipid(struct board_info * db)5162dc95a4dSJoseph CHAMG static int dm9051_map_chipid(struct board_info *db)
5172dc95a4dSJoseph CHAMG {
5182dc95a4dSJoseph CHAMG 	struct device *dev = &db->spidev->dev;
5192dc95a4dSJoseph CHAMG 	unsigned short wid;
5202dc95a4dSJoseph CHAMG 	u8 buff[6];
5218c16baa5SYang Li 	int ret;
5222dc95a4dSJoseph CHAMG 
5232dc95a4dSJoseph CHAMG 	ret = dm9051_get_regs(db, DM9051_VIDL, buff, sizeof(buff));
5242dc95a4dSJoseph CHAMG 	if (ret < 0)
5252dc95a4dSJoseph CHAMG 		return ret;
5262dc95a4dSJoseph CHAMG 
5272dc95a4dSJoseph CHAMG 	wid = get_unaligned_le16(buff + 2);
5282dc95a4dSJoseph CHAMG 	if (wid != DM9051_ID) {
5292dc95a4dSJoseph CHAMG 		dev_err(dev, "chipid error as %04x !\n", wid);
5302dc95a4dSJoseph CHAMG 		return -ENODEV;
5312dc95a4dSJoseph CHAMG 	}
5322dc95a4dSJoseph CHAMG 
5332dc95a4dSJoseph CHAMG 	dev_info(dev, "chip %04x found\n", wid);
5342dc95a4dSJoseph CHAMG 	return 0;
5352dc95a4dSJoseph CHAMG }
5362dc95a4dSJoseph CHAMG 
5372dc95a4dSJoseph CHAMG /* Read DM9051_PAR registers which is the mac address loaded from EEPROM while power-on
5382dc95a4dSJoseph CHAMG  */
dm9051_map_etherdev_par(struct net_device * ndev,struct board_info * db)5392dc95a4dSJoseph CHAMG static int dm9051_map_etherdev_par(struct net_device *ndev, struct board_info *db)
5402dc95a4dSJoseph CHAMG {
5412dc95a4dSJoseph CHAMG 	u8 addr[ETH_ALEN];
5422dc95a4dSJoseph CHAMG 	int ret;
5432dc95a4dSJoseph CHAMG 
5442dc95a4dSJoseph CHAMG 	ret = dm9051_get_regs(db, DM9051_PAR, addr, sizeof(addr));
5452dc95a4dSJoseph CHAMG 	if (ret < 0)
5462dc95a4dSJoseph CHAMG 		return ret;
5472dc95a4dSJoseph CHAMG 
5482dc95a4dSJoseph CHAMG 	if (!is_valid_ether_addr(addr)) {
5492dc95a4dSJoseph CHAMG 		eth_hw_addr_random(ndev);
5502dc95a4dSJoseph CHAMG 
5512dc95a4dSJoseph CHAMG 		ret = dm9051_set_regs(db, DM9051_PAR, ndev->dev_addr, sizeof(ndev->dev_addr));
5522dc95a4dSJoseph CHAMG 		if (ret < 0)
5532dc95a4dSJoseph CHAMG 			return ret;
5542dc95a4dSJoseph CHAMG 
5552dc95a4dSJoseph CHAMG 		dev_dbg(&db->spidev->dev, "Use random MAC address\n");
5562dc95a4dSJoseph CHAMG 		return 0;
5572dc95a4dSJoseph CHAMG 	}
5582dc95a4dSJoseph CHAMG 
5592dc95a4dSJoseph CHAMG 	eth_hw_addr_set(ndev, addr);
5602dc95a4dSJoseph CHAMG 	return 0;
5612dc95a4dSJoseph CHAMG }
5622dc95a4dSJoseph CHAMG 
5632dc95a4dSJoseph CHAMG /* ethtool-ops
5642dc95a4dSJoseph CHAMG  */
dm9051_get_drvinfo(struct net_device * dev,struct ethtool_drvinfo * info)5652dc95a4dSJoseph CHAMG static void dm9051_get_drvinfo(struct net_device *dev, struct ethtool_drvinfo *info)
5662dc95a4dSJoseph CHAMG {
5672dc95a4dSJoseph CHAMG 	strscpy(info->driver, DRVNAME_9051, sizeof(info->driver));
5682dc95a4dSJoseph CHAMG }
5692dc95a4dSJoseph CHAMG 
dm9051_set_msglevel(struct net_device * ndev,u32 value)5702dc95a4dSJoseph CHAMG static void dm9051_set_msglevel(struct net_device *ndev, u32 value)
5712dc95a4dSJoseph CHAMG {
5722dc95a4dSJoseph CHAMG 	struct board_info *db = to_dm9051_board(ndev);
5732dc95a4dSJoseph CHAMG 
5742dc95a4dSJoseph CHAMG 	db->msg_enable = value;
5752dc95a4dSJoseph CHAMG }
5762dc95a4dSJoseph CHAMG 
dm9051_get_msglevel(struct net_device * ndev)5772dc95a4dSJoseph CHAMG static u32 dm9051_get_msglevel(struct net_device *ndev)
5782dc95a4dSJoseph CHAMG {
5792dc95a4dSJoseph CHAMG 	struct board_info *db = to_dm9051_board(ndev);
5802dc95a4dSJoseph CHAMG 
5812dc95a4dSJoseph CHAMG 	return db->msg_enable;
5822dc95a4dSJoseph CHAMG }
5832dc95a4dSJoseph CHAMG 
dm9051_get_eeprom_len(struct net_device * dev)5842dc95a4dSJoseph CHAMG static int dm9051_get_eeprom_len(struct net_device *dev)
5852dc95a4dSJoseph CHAMG {
5862dc95a4dSJoseph CHAMG 	return 128;
5872dc95a4dSJoseph CHAMG }
5882dc95a4dSJoseph CHAMG 
dm9051_get_eeprom(struct net_device * ndev,struct ethtool_eeprom * ee,u8 * data)5892dc95a4dSJoseph CHAMG static int dm9051_get_eeprom(struct net_device *ndev,
5902dc95a4dSJoseph CHAMG 			     struct ethtool_eeprom *ee, u8 *data)
5912dc95a4dSJoseph CHAMG {
5922dc95a4dSJoseph CHAMG 	struct board_info *db = to_dm9051_board(ndev);
5932dc95a4dSJoseph CHAMG 	int offset = ee->offset;
5942dc95a4dSJoseph CHAMG 	int len = ee->len;
5952dc95a4dSJoseph CHAMG 	int i, ret;
5962dc95a4dSJoseph CHAMG 
5972dc95a4dSJoseph CHAMG 	if ((len | offset) & 1)
5982dc95a4dSJoseph CHAMG 		return -EINVAL;
5992dc95a4dSJoseph CHAMG 
6002dc95a4dSJoseph CHAMG 	ee->magic = DM_EEPROM_MAGIC;
6012dc95a4dSJoseph CHAMG 
6022dc95a4dSJoseph CHAMG 	for (i = 0; i < len; i += 2) {
6032dc95a4dSJoseph CHAMG 		ret = dm9051_eeprom_read(db, (offset + i) / 2, data + i);
6042dc95a4dSJoseph CHAMG 		if (ret)
6052dc95a4dSJoseph CHAMG 			break;
6062dc95a4dSJoseph CHAMG 	}
6072dc95a4dSJoseph CHAMG 	return ret;
6082dc95a4dSJoseph CHAMG }
6092dc95a4dSJoseph CHAMG 
dm9051_set_eeprom(struct net_device * ndev,struct ethtool_eeprom * ee,u8 * data)6102dc95a4dSJoseph CHAMG static int dm9051_set_eeprom(struct net_device *ndev,
6112dc95a4dSJoseph CHAMG 			     struct ethtool_eeprom *ee, u8 *data)
6122dc95a4dSJoseph CHAMG {
6132dc95a4dSJoseph CHAMG 	struct board_info *db = to_dm9051_board(ndev);
6142dc95a4dSJoseph CHAMG 	int offset = ee->offset;
6152dc95a4dSJoseph CHAMG 	int len = ee->len;
6162dc95a4dSJoseph CHAMG 	int i, ret;
6172dc95a4dSJoseph CHAMG 
6182dc95a4dSJoseph CHAMG 	if ((len | offset) & 1)
6192dc95a4dSJoseph CHAMG 		return -EINVAL;
6202dc95a4dSJoseph CHAMG 
6212dc95a4dSJoseph CHAMG 	if (ee->magic != DM_EEPROM_MAGIC)
6222dc95a4dSJoseph CHAMG 		return -EINVAL;
6232dc95a4dSJoseph CHAMG 
6242dc95a4dSJoseph CHAMG 	for (i = 0; i < len; i += 2) {
6252dc95a4dSJoseph CHAMG 		ret = dm9051_eeprom_write(db, (offset + i) / 2, data + i);
6262dc95a4dSJoseph CHAMG 		if (ret)
6272dc95a4dSJoseph CHAMG 			break;
6282dc95a4dSJoseph CHAMG 	}
6292dc95a4dSJoseph CHAMG 	return ret;
6302dc95a4dSJoseph CHAMG }
6312dc95a4dSJoseph CHAMG 
dm9051_get_pauseparam(struct net_device * ndev,struct ethtool_pauseparam * pause)6322dc95a4dSJoseph CHAMG static void dm9051_get_pauseparam(struct net_device *ndev,
6332dc95a4dSJoseph CHAMG 				  struct ethtool_pauseparam *pause)
6342dc95a4dSJoseph CHAMG {
6352dc95a4dSJoseph CHAMG 	struct board_info *db = to_dm9051_board(ndev);
6362dc95a4dSJoseph CHAMG 
6372dc95a4dSJoseph CHAMG 	*pause = db->pause;
6382dc95a4dSJoseph CHAMG }
6392dc95a4dSJoseph CHAMG 
dm9051_set_pauseparam(struct net_device * ndev,struct ethtool_pauseparam * pause)6402dc95a4dSJoseph CHAMG static int dm9051_set_pauseparam(struct net_device *ndev,
6412dc95a4dSJoseph CHAMG 				 struct ethtool_pauseparam *pause)
6422dc95a4dSJoseph CHAMG {
6432dc95a4dSJoseph CHAMG 	struct board_info *db = to_dm9051_board(ndev);
6442dc95a4dSJoseph CHAMG 
6452dc95a4dSJoseph CHAMG 	db->pause = *pause;
6462dc95a4dSJoseph CHAMG 
6472dc95a4dSJoseph CHAMG 	if (pause->autoneg == AUTONEG_DISABLE)
6482dc95a4dSJoseph CHAMG 		return dm9051_update_fcr(db);
6492dc95a4dSJoseph CHAMG 
6502dc95a4dSJoseph CHAMG 	phy_set_sym_pause(db->phydev, pause->rx_pause, pause->tx_pause,
6512dc95a4dSJoseph CHAMG 			  pause->autoneg);
6522dc95a4dSJoseph CHAMG 	phy_start_aneg(db->phydev);
6532dc95a4dSJoseph CHAMG 	return 0;
6542dc95a4dSJoseph CHAMG }
6552dc95a4dSJoseph CHAMG 
6562dc95a4dSJoseph CHAMG static const struct ethtool_ops dm9051_ethtool_ops = {
6572dc95a4dSJoseph CHAMG 	.get_drvinfo = dm9051_get_drvinfo,
6582dc95a4dSJoseph CHAMG 	.get_link_ksettings = phy_ethtool_get_link_ksettings,
6592dc95a4dSJoseph CHAMG 	.set_link_ksettings = phy_ethtool_set_link_ksettings,
6602dc95a4dSJoseph CHAMG 	.get_msglevel = dm9051_get_msglevel,
6612dc95a4dSJoseph CHAMG 	.set_msglevel = dm9051_set_msglevel,
6622dc95a4dSJoseph CHAMG 	.nway_reset = phy_ethtool_nway_reset,
6632dc95a4dSJoseph CHAMG 	.get_link = ethtool_op_get_link,
6642dc95a4dSJoseph CHAMG 	.get_eeprom_len = dm9051_get_eeprom_len,
6652dc95a4dSJoseph CHAMG 	.get_eeprom = dm9051_get_eeprom,
6662dc95a4dSJoseph CHAMG 	.set_eeprom = dm9051_set_eeprom,
6672dc95a4dSJoseph CHAMG 	.get_pauseparam = dm9051_get_pauseparam,
6682dc95a4dSJoseph CHAMG 	.set_pauseparam = dm9051_set_pauseparam,
6692dc95a4dSJoseph CHAMG };
6702dc95a4dSJoseph CHAMG 
dm9051_all_start(struct board_info * db)6712dc95a4dSJoseph CHAMG static int dm9051_all_start(struct board_info *db)
6722dc95a4dSJoseph CHAMG {
6732dc95a4dSJoseph CHAMG 	int ret;
6742dc95a4dSJoseph CHAMG 
6752dc95a4dSJoseph CHAMG 	/* GPR power on of the internal phy
6762dc95a4dSJoseph CHAMG 	 */
6772dc95a4dSJoseph CHAMG 	ret = dm9051_set_reg(db, DM9051_GPR, 0);
6782dc95a4dSJoseph CHAMG 	if (ret)
6792dc95a4dSJoseph CHAMG 		return ret;
6802dc95a4dSJoseph CHAMG 
6812dc95a4dSJoseph CHAMG 	/* dm9051 chip registers could not be accessed within 1 ms
6822dc95a4dSJoseph CHAMG 	 * after GPR power on, delay 1 ms is essential
6832dc95a4dSJoseph CHAMG 	 */
6842dc95a4dSJoseph CHAMG 	msleep(1);
6852dc95a4dSJoseph CHAMG 
6862dc95a4dSJoseph CHAMG 	ret = dm9051_core_reset(db);
6872dc95a4dSJoseph CHAMG 	if (ret)
6882dc95a4dSJoseph CHAMG 		return ret;
6892dc95a4dSJoseph CHAMG 
6902dc95a4dSJoseph CHAMG 	return dm9051_enable_interrupt(db);
6912dc95a4dSJoseph CHAMG }
6922dc95a4dSJoseph CHAMG 
dm9051_all_stop(struct board_info * db)6932dc95a4dSJoseph CHAMG static int dm9051_all_stop(struct board_info *db)
6942dc95a4dSJoseph CHAMG {
6952dc95a4dSJoseph CHAMG 	int ret;
6962dc95a4dSJoseph CHAMG 
6972dc95a4dSJoseph CHAMG 	/* GPR power off of the internal phy,
6982dc95a4dSJoseph CHAMG 	 * The internal phy still could be accessed after this GPR power off control
6992dc95a4dSJoseph CHAMG 	 */
7002dc95a4dSJoseph CHAMG 	ret = dm9051_set_reg(db, DM9051_GPR, GPR_PHY_OFF);
7012dc95a4dSJoseph CHAMG 	if (ret)
7022dc95a4dSJoseph CHAMG 		return ret;
7032dc95a4dSJoseph CHAMG 
7042dc95a4dSJoseph CHAMG 	return dm9051_set_reg(db, DM9051_RCR, RCR_RX_DISABLE);
7052dc95a4dSJoseph CHAMG }
7062dc95a4dSJoseph CHAMG 
7072dc95a4dSJoseph CHAMG /* fifo reset while rx error found
7082dc95a4dSJoseph CHAMG  */
dm9051_all_restart(struct board_info * db)7092dc95a4dSJoseph CHAMG static int dm9051_all_restart(struct board_info *db)
7102dc95a4dSJoseph CHAMG {
7112dc95a4dSJoseph CHAMG 	struct net_device *ndev = db->ndev;
7122dc95a4dSJoseph CHAMG 	int ret;
7132dc95a4dSJoseph CHAMG 
7142dc95a4dSJoseph CHAMG 	ret = dm9051_core_reset(db);
7152dc95a4dSJoseph CHAMG 	if (ret)
7162dc95a4dSJoseph CHAMG 		return ret;
7172dc95a4dSJoseph CHAMG 
7182dc95a4dSJoseph CHAMG 	ret = dm9051_enable_interrupt(db);
7192dc95a4dSJoseph CHAMG 	if (ret)
7202dc95a4dSJoseph CHAMG 		return ret;
7212dc95a4dSJoseph CHAMG 
7222dc95a4dSJoseph CHAMG 	netdev_dbg(ndev, " rxstatus_Er & rxlen_Er %d, RST_c %d\n",
7232dc95a4dSJoseph CHAMG 		   db->bc.status_err_counter + db->bc.large_err_counter,
7242dc95a4dSJoseph CHAMG 		   db->bc.fifo_rst_counter);
7252dc95a4dSJoseph CHAMG 
7262dc95a4dSJoseph CHAMG 	ret = dm9051_set_recv(db);
7272dc95a4dSJoseph CHAMG 	if (ret)
7282dc95a4dSJoseph CHAMG 		return ret;
7292dc95a4dSJoseph CHAMG 
7302dc95a4dSJoseph CHAMG 	return dm9051_set_fcr(db);
7312dc95a4dSJoseph CHAMG }
7322dc95a4dSJoseph CHAMG 
7332dc95a4dSJoseph CHAMG /* read packets from the fifo memory
7342dc95a4dSJoseph CHAMG  * return value,
7352dc95a4dSJoseph CHAMG  *  > 0 - read packet number, caller can repeat the rx operation
7362dc95a4dSJoseph CHAMG  *    0 - no error, caller need stop further rx operation
7372dc95a4dSJoseph CHAMG  *  -EBUSY - read data error, caller escape from rx operation
7382dc95a4dSJoseph CHAMG  */
dm9051_loop_rx(struct board_info * db)7392dc95a4dSJoseph CHAMG static int dm9051_loop_rx(struct board_info *db)
7402dc95a4dSJoseph CHAMG {
7412dc95a4dSJoseph CHAMG 	struct net_device *ndev = db->ndev;
7422dc95a4dSJoseph CHAMG 	unsigned int rxbyte;
7432dc95a4dSJoseph CHAMG 	int ret, rxlen;
7442dc95a4dSJoseph CHAMG 	struct sk_buff *skb;
7452dc95a4dSJoseph CHAMG 	u8 *rdptr;
7462dc95a4dSJoseph CHAMG 	int scanrr = 0;
7472dc95a4dSJoseph CHAMG 
7482dc95a4dSJoseph CHAMG 	do {
7492dc95a4dSJoseph CHAMG 		ret = dm9051_read_mem(db, DM_SPI_MRCMDX, &rxbyte, 2);
7502dc95a4dSJoseph CHAMG 		if (ret)
7512dc95a4dSJoseph CHAMG 			return ret;
7522dc95a4dSJoseph CHAMG 
7532dc95a4dSJoseph CHAMG 		if ((rxbyte & GENMASK(7, 0)) != DM9051_PKT_RDY)
7542dc95a4dSJoseph CHAMG 			break; /* exhaust-empty */
7552dc95a4dSJoseph CHAMG 
7562dc95a4dSJoseph CHAMG 		ret = dm9051_read_mem(db, DM_SPI_MRCMD, &db->rxhdr, DM_RXHDR_SIZE);
7572dc95a4dSJoseph CHAMG 		if (ret)
7582dc95a4dSJoseph CHAMG 			return ret;
7592dc95a4dSJoseph CHAMG 
7602dc95a4dSJoseph CHAMG 		ret = dm9051_stop_mrcmd(db);
7612dc95a4dSJoseph CHAMG 		if (ret)
7622dc95a4dSJoseph CHAMG 			return ret;
7632dc95a4dSJoseph CHAMG 
7642dc95a4dSJoseph CHAMG 		rxlen = le16_to_cpu(db->rxhdr.rxlen);
7652dc95a4dSJoseph CHAMG 		if (db->rxhdr.status & RSR_ERR_BITS || rxlen > DM9051_PKT_MAX) {
7662dc95a4dSJoseph CHAMG 			netdev_dbg(ndev, "rxhdr-byte (%02x)\n",
7672dc95a4dSJoseph CHAMG 				   db->rxhdr.headbyte);
7682dc95a4dSJoseph CHAMG 
7692dc95a4dSJoseph CHAMG 			if (db->rxhdr.status & RSR_ERR_BITS) {
7702dc95a4dSJoseph CHAMG 				db->bc.status_err_counter++;
7712c955856SColin Ian King 				netdev_dbg(ndev, "check rxstatus-error (%02x)\n",
7722dc95a4dSJoseph CHAMG 					   db->rxhdr.status);
7732dc95a4dSJoseph CHAMG 			} else {
7742dc95a4dSJoseph CHAMG 				db->bc.large_err_counter++;
7752c955856SColin Ian King 				netdev_dbg(ndev, "check rxlen large-error (%d > %d)\n",
7762dc95a4dSJoseph CHAMG 					   rxlen, DM9051_PKT_MAX);
7772dc95a4dSJoseph CHAMG 			}
7782dc95a4dSJoseph CHAMG 			return dm9051_all_restart(db);
7792dc95a4dSJoseph CHAMG 		}
7802dc95a4dSJoseph CHAMG 
7812dc95a4dSJoseph CHAMG 		skb = dev_alloc_skb(rxlen);
7822dc95a4dSJoseph CHAMG 		if (!skb) {
7832dc95a4dSJoseph CHAMG 			ret = dm9051_dumpblk(db, DM_SPI_MRCMD, rxlen);
7842dc95a4dSJoseph CHAMG 			if (ret)
7852dc95a4dSJoseph CHAMG 				return ret;
7862dc95a4dSJoseph CHAMG 			return scanrr;
7872dc95a4dSJoseph CHAMG 		}
7882dc95a4dSJoseph CHAMG 
7892dc95a4dSJoseph CHAMG 		rdptr = skb_put(skb, rxlen - 4);
7902dc95a4dSJoseph CHAMG 		ret = dm9051_read_mem(db, DM_SPI_MRCMD, rdptr, rxlen);
7912dc95a4dSJoseph CHAMG 		if (ret) {
7922dc95a4dSJoseph CHAMG 			db->bc.rx_err_counter++;
7932dc95a4dSJoseph CHAMG 			dev_kfree_skb(skb);
7942dc95a4dSJoseph CHAMG 			return ret;
7952dc95a4dSJoseph CHAMG 		}
7962dc95a4dSJoseph CHAMG 
7972dc95a4dSJoseph CHAMG 		ret = dm9051_stop_mrcmd(db);
798bac81f40SYuan Can 		if (ret) {
799bac81f40SYuan Can 			dev_kfree_skb(skb);
8002dc95a4dSJoseph CHAMG 			return ret;
801bac81f40SYuan Can 		}
8022dc95a4dSJoseph CHAMG 
8032dc95a4dSJoseph CHAMG 		skb->protocol = eth_type_trans(skb, db->ndev);
8042dc95a4dSJoseph CHAMG 		if (db->ndev->features & NETIF_F_RXCSUM)
8052dc95a4dSJoseph CHAMG 			skb_checksum_none_assert(skb);
80690f77c1cSSebastian Andrzej Siewior 		netif_rx(skb);
8072dc95a4dSJoseph CHAMG 		db->ndev->stats.rx_bytes += rxlen;
8082dc95a4dSJoseph CHAMG 		db->ndev->stats.rx_packets++;
8092dc95a4dSJoseph CHAMG 		scanrr++;
8102dc95a4dSJoseph CHAMG 	} while (!ret);
8112dc95a4dSJoseph CHAMG 
8122dc95a4dSJoseph CHAMG 	return scanrr;
8132dc95a4dSJoseph CHAMG }
8142dc95a4dSJoseph CHAMG 
8152dc95a4dSJoseph CHAMG /* transmit a packet,
8162dc95a4dSJoseph CHAMG  * return value,
8172dc95a4dSJoseph CHAMG  *   0 - succeed
8182dc95a4dSJoseph CHAMG  *  -ETIMEDOUT - timeout error
8192dc95a4dSJoseph CHAMG  */
dm9051_single_tx(struct board_info * db,u8 * buff,unsigned int len)8202dc95a4dSJoseph CHAMG static int dm9051_single_tx(struct board_info *db, u8 *buff, unsigned int len)
8212dc95a4dSJoseph CHAMG {
8222dc95a4dSJoseph CHAMG 	int ret;
8232dc95a4dSJoseph CHAMG 
8242dc95a4dSJoseph CHAMG 	ret = dm9051_nsr_poll(db);
8252dc95a4dSJoseph CHAMG 	if (ret)
8262dc95a4dSJoseph CHAMG 		return ret;
8272dc95a4dSJoseph CHAMG 
8282dc95a4dSJoseph CHAMG 	ret = dm9051_write_mem(db, DM_SPI_MWCMD, buff, len);
8292dc95a4dSJoseph CHAMG 	if (ret)
8302dc95a4dSJoseph CHAMG 		return ret;
8312dc95a4dSJoseph CHAMG 
8322dc95a4dSJoseph CHAMG 	ret = dm9051_set_regs(db, DM9051_TXPLL, &len, 2);
8332dc95a4dSJoseph CHAMG 	if (ret < 0)
8342dc95a4dSJoseph CHAMG 		return ret;
8352dc95a4dSJoseph CHAMG 
8362dc95a4dSJoseph CHAMG 	return dm9051_set_reg(db, DM9051_TCR, TCR_TXREQ);
8372dc95a4dSJoseph CHAMG }
8382dc95a4dSJoseph CHAMG 
dm9051_loop_tx(struct board_info * db)8392dc95a4dSJoseph CHAMG static int dm9051_loop_tx(struct board_info *db)
8402dc95a4dSJoseph CHAMG {
8412dc95a4dSJoseph CHAMG 	struct net_device *ndev = db->ndev;
8422dc95a4dSJoseph CHAMG 	int ntx = 0;
8432dc95a4dSJoseph CHAMG 	int ret;
8442dc95a4dSJoseph CHAMG 
8452dc95a4dSJoseph CHAMG 	while (!skb_queue_empty(&db->txq)) {
8462dc95a4dSJoseph CHAMG 		struct sk_buff *skb;
847b6553c71SDan Carpenter 		unsigned int len;
8482dc95a4dSJoseph CHAMG 
8492dc95a4dSJoseph CHAMG 		skb = skb_dequeue(&db->txq);
8502dc95a4dSJoseph CHAMG 		if (skb) {
8512dc95a4dSJoseph CHAMG 			ntx++;
8522dc95a4dSJoseph CHAMG 			ret = dm9051_single_tx(db, skb->data, skb->len);
853b6553c71SDan Carpenter 			len = skb->len;
8542dc95a4dSJoseph CHAMG 			dev_kfree_skb(skb);
8552dc95a4dSJoseph CHAMG 			if (ret < 0) {
8562dc95a4dSJoseph CHAMG 				db->bc.tx_err_counter++;
8572dc95a4dSJoseph CHAMG 				return 0;
8582dc95a4dSJoseph CHAMG 			}
859b6553c71SDan Carpenter 			ndev->stats.tx_bytes += len;
8602dc95a4dSJoseph CHAMG 			ndev->stats.tx_packets++;
8612dc95a4dSJoseph CHAMG 		}
8622dc95a4dSJoseph CHAMG 
8632dc95a4dSJoseph CHAMG 		if (netif_queue_stopped(ndev) &&
8642dc95a4dSJoseph CHAMG 		    (skb_queue_len(&db->txq) < DM9051_TX_QUE_LO_WATER))
8652dc95a4dSJoseph CHAMG 			netif_wake_queue(ndev);
8662dc95a4dSJoseph CHAMG 	}
8672dc95a4dSJoseph CHAMG 
8682dc95a4dSJoseph CHAMG 	return ntx;
8692dc95a4dSJoseph CHAMG }
8702dc95a4dSJoseph CHAMG 
dm9051_rx_threaded_irq(int irq,void * pw)8712dc95a4dSJoseph CHAMG static irqreturn_t dm9051_rx_threaded_irq(int irq, void *pw)
8722dc95a4dSJoseph CHAMG {
8732dc95a4dSJoseph CHAMG 	struct board_info *db = pw;
8742dc95a4dSJoseph CHAMG 	int result, result_tx;
8752dc95a4dSJoseph CHAMG 
8762dc95a4dSJoseph CHAMG 	mutex_lock(&db->spi_lockm);
8772dc95a4dSJoseph CHAMG 
8782dc95a4dSJoseph CHAMG 	result = dm9051_disable_interrupt(db);
8792dc95a4dSJoseph CHAMG 	if (result)
8802dc95a4dSJoseph CHAMG 		goto out_unlock;
8812dc95a4dSJoseph CHAMG 
8822dc95a4dSJoseph CHAMG 	result = dm9051_clear_interrupt(db);
8832dc95a4dSJoseph CHAMG 	if (result)
8842dc95a4dSJoseph CHAMG 		goto out_unlock;
8852dc95a4dSJoseph CHAMG 
8862dc95a4dSJoseph CHAMG 	do {
8872dc95a4dSJoseph CHAMG 		result = dm9051_loop_rx(db); /* threaded irq rx */
8882dc95a4dSJoseph CHAMG 		if (result < 0)
8892dc95a4dSJoseph CHAMG 			goto out_unlock;
8902dc95a4dSJoseph CHAMG 		result_tx = dm9051_loop_tx(db); /* more tx better performance */
8912dc95a4dSJoseph CHAMG 		if (result_tx < 0)
8922dc95a4dSJoseph CHAMG 			goto out_unlock;
8932dc95a4dSJoseph CHAMG 	} while (result > 0);
8942dc95a4dSJoseph CHAMG 
8952dc95a4dSJoseph CHAMG 	dm9051_enable_interrupt(db);
8962dc95a4dSJoseph CHAMG 
8972dc95a4dSJoseph CHAMG 	/* To exit and has mutex unlock while rx or tx error
8982dc95a4dSJoseph CHAMG 	 */
8992dc95a4dSJoseph CHAMG out_unlock:
9002dc95a4dSJoseph CHAMG 	mutex_unlock(&db->spi_lockm);
9012dc95a4dSJoseph CHAMG 
9022dc95a4dSJoseph CHAMG 	return IRQ_HANDLED;
9032dc95a4dSJoseph CHAMG }
9042dc95a4dSJoseph CHAMG 
dm9051_tx_delay(struct work_struct * work)9052dc95a4dSJoseph CHAMG static void dm9051_tx_delay(struct work_struct *work)
9062dc95a4dSJoseph CHAMG {
9072dc95a4dSJoseph CHAMG 	struct board_info *db = container_of(work, struct board_info, tx_work);
9082dc95a4dSJoseph CHAMG 	int result;
9092dc95a4dSJoseph CHAMG 
9102dc95a4dSJoseph CHAMG 	mutex_lock(&db->spi_lockm);
9112dc95a4dSJoseph CHAMG 
9122dc95a4dSJoseph CHAMG 	result = dm9051_loop_tx(db);
9132dc95a4dSJoseph CHAMG 	if (result < 0)
9142dc95a4dSJoseph CHAMG 		netdev_err(db->ndev, "transmit packet error\n");
9152dc95a4dSJoseph CHAMG 
9162dc95a4dSJoseph CHAMG 	mutex_unlock(&db->spi_lockm);
9172dc95a4dSJoseph CHAMG }
9182dc95a4dSJoseph CHAMG 
dm9051_rxctl_delay(struct work_struct * work)9192dc95a4dSJoseph CHAMG static void dm9051_rxctl_delay(struct work_struct *work)
9202dc95a4dSJoseph CHAMG {
9212dc95a4dSJoseph CHAMG 	struct board_info *db = container_of(work, struct board_info, rxctrl_work);
9222dc95a4dSJoseph CHAMG 	struct net_device *ndev = db->ndev;
9232dc95a4dSJoseph CHAMG 	int result;
9242dc95a4dSJoseph CHAMG 
9252dc95a4dSJoseph CHAMG 	mutex_lock(&db->spi_lockm);
9262dc95a4dSJoseph CHAMG 
9272dc95a4dSJoseph CHAMG 	result = dm9051_set_regs(db, DM9051_PAR, ndev->dev_addr, sizeof(ndev->dev_addr));
9282dc95a4dSJoseph CHAMG 	if (result < 0)
9292dc95a4dSJoseph CHAMG 		goto out_unlock;
9302dc95a4dSJoseph CHAMG 
9312dc95a4dSJoseph CHAMG 	dm9051_set_recv(db);
9322dc95a4dSJoseph CHAMG 
9332dc95a4dSJoseph CHAMG 	/* To has mutex unlock and return from this function if regmap function fail
9342dc95a4dSJoseph CHAMG 	 */
9352dc95a4dSJoseph CHAMG out_unlock:
9362dc95a4dSJoseph CHAMG 	mutex_unlock(&db->spi_lockm);
9372dc95a4dSJoseph CHAMG }
9382dc95a4dSJoseph CHAMG 
9392dc95a4dSJoseph CHAMG /* Open network device
9402dc95a4dSJoseph CHAMG  * Called when the network device is marked active, such as a user executing
9412dc95a4dSJoseph CHAMG  * 'ifconfig up' on the device
9422dc95a4dSJoseph CHAMG  */
dm9051_open(struct net_device * ndev)9432dc95a4dSJoseph CHAMG static int dm9051_open(struct net_device *ndev)
9442dc95a4dSJoseph CHAMG {
9452dc95a4dSJoseph CHAMG 	struct board_info *db = to_dm9051_board(ndev);
9462dc95a4dSJoseph CHAMG 	struct spi_device *spi = db->spidev;
9472dc95a4dSJoseph CHAMG 	int ret;
9482dc95a4dSJoseph CHAMG 
9492dc95a4dSJoseph CHAMG 	db->imr_all = IMR_PAR | IMR_PRM;
9502dc95a4dSJoseph CHAMG 	db->lcr_all = LMCR_MODE1;
9512dc95a4dSJoseph CHAMG 	db->rctl.rcr_all = RCR_DIS_LONG | RCR_DIS_CRC | RCR_RXEN;
9522dc95a4dSJoseph CHAMG 	memset(db->rctl.hash_table, 0, sizeof(db->rctl.hash_table));
9532dc95a4dSJoseph CHAMG 
9542dc95a4dSJoseph CHAMG 	ndev->irq = spi->irq; /* by dts */
9552dc95a4dSJoseph CHAMG 	ret = request_threaded_irq(spi->irq, NULL, dm9051_rx_threaded_irq,
9562dc95a4dSJoseph CHAMG 				   dm9051_irq_flag(db) | IRQF_ONESHOT,
9572dc95a4dSJoseph CHAMG 				   ndev->name, db);
9582dc95a4dSJoseph CHAMG 	if (ret < 0) {
9592dc95a4dSJoseph CHAMG 		netdev_err(ndev, "failed to get irq\n");
9602dc95a4dSJoseph CHAMG 		return ret;
9612dc95a4dSJoseph CHAMG 	}
9622dc95a4dSJoseph CHAMG 
9632dc95a4dSJoseph CHAMG 	phy_support_sym_pause(db->phydev);
9642dc95a4dSJoseph CHAMG 	phy_start(db->phydev);
9652dc95a4dSJoseph CHAMG 
9662dc95a4dSJoseph CHAMG 	/* flow control parameters init */
9672dc95a4dSJoseph CHAMG 	db->pause.rx_pause = true;
9682dc95a4dSJoseph CHAMG 	db->pause.tx_pause = true;
9692dc95a4dSJoseph CHAMG 	db->pause.autoneg = AUTONEG_DISABLE;
9702dc95a4dSJoseph CHAMG 
9712dc95a4dSJoseph CHAMG 	if (db->phydev->autoneg)
9722dc95a4dSJoseph CHAMG 		db->pause.autoneg = AUTONEG_ENABLE;
9732dc95a4dSJoseph CHAMG 
9742dc95a4dSJoseph CHAMG 	ret = dm9051_all_start(db);
9752dc95a4dSJoseph CHAMG 	if (ret) {
9762dc95a4dSJoseph CHAMG 		phy_stop(db->phydev);
9772dc95a4dSJoseph CHAMG 		free_irq(spi->irq, db);
9782dc95a4dSJoseph CHAMG 		return ret;
9792dc95a4dSJoseph CHAMG 	}
9802dc95a4dSJoseph CHAMG 
9812dc95a4dSJoseph CHAMG 	netif_wake_queue(ndev);
9822dc95a4dSJoseph CHAMG 
9832dc95a4dSJoseph CHAMG 	return 0;
9842dc95a4dSJoseph CHAMG }
9852dc95a4dSJoseph CHAMG 
9862dc95a4dSJoseph CHAMG /* Close network device
9872dc95a4dSJoseph CHAMG  * Called to close down a network device which has been active. Cancel any
9882dc95a4dSJoseph CHAMG  * work, shutdown the RX and TX process and then place the chip into a low
9892dc95a4dSJoseph CHAMG  * power state while it is not being used
9902dc95a4dSJoseph CHAMG  */
dm9051_stop(struct net_device * ndev)9912dc95a4dSJoseph CHAMG static int dm9051_stop(struct net_device *ndev)
9922dc95a4dSJoseph CHAMG {
9932dc95a4dSJoseph CHAMG 	struct board_info *db = to_dm9051_board(ndev);
9942dc95a4dSJoseph CHAMG 	int ret;
9952dc95a4dSJoseph CHAMG 
9962dc95a4dSJoseph CHAMG 	ret = dm9051_all_stop(db);
9972dc95a4dSJoseph CHAMG 	if (ret)
9982dc95a4dSJoseph CHAMG 		return ret;
9992dc95a4dSJoseph CHAMG 
10002dc95a4dSJoseph CHAMG 	flush_work(&db->tx_work);
10012dc95a4dSJoseph CHAMG 	flush_work(&db->rxctrl_work);
10022dc95a4dSJoseph CHAMG 
10032dc95a4dSJoseph CHAMG 	phy_stop(db->phydev);
10042dc95a4dSJoseph CHAMG 
10052dc95a4dSJoseph CHAMG 	free_irq(db->spidev->irq, db);
10062dc95a4dSJoseph CHAMG 
10072dc95a4dSJoseph CHAMG 	netif_stop_queue(ndev);
10082dc95a4dSJoseph CHAMG 
10092dc95a4dSJoseph CHAMG 	skb_queue_purge(&db->txq);
10102dc95a4dSJoseph CHAMG 
10112dc95a4dSJoseph CHAMG 	return 0;
10122dc95a4dSJoseph CHAMG }
10132dc95a4dSJoseph CHAMG 
10142dc95a4dSJoseph CHAMG /* event: play a schedule starter in condition
10152dc95a4dSJoseph CHAMG  */
dm9051_start_xmit(struct sk_buff * skb,struct net_device * ndev)10162dc95a4dSJoseph CHAMG static netdev_tx_t dm9051_start_xmit(struct sk_buff *skb, struct net_device *ndev)
10172dc95a4dSJoseph CHAMG {
10182dc95a4dSJoseph CHAMG 	struct board_info *db = to_dm9051_board(ndev);
10192dc95a4dSJoseph CHAMG 
10202dc95a4dSJoseph CHAMG 	skb_queue_tail(&db->txq, skb);
10212dc95a4dSJoseph CHAMG 	if (skb_queue_len(&db->txq) > DM9051_TX_QUE_HI_WATER)
10222dc95a4dSJoseph CHAMG 		netif_stop_queue(ndev); /* enforce limit queue size */
10232dc95a4dSJoseph CHAMG 
10242dc95a4dSJoseph CHAMG 	schedule_work(&db->tx_work);
10252dc95a4dSJoseph CHAMG 
10262dc95a4dSJoseph CHAMG 	return NETDEV_TX_OK;
10272dc95a4dSJoseph CHAMG }
10282dc95a4dSJoseph CHAMG 
10292dc95a4dSJoseph CHAMG /* event: play with a schedule starter
10302dc95a4dSJoseph CHAMG  */
dm9051_set_rx_mode(struct net_device * ndev)10312dc95a4dSJoseph CHAMG static void dm9051_set_rx_mode(struct net_device *ndev)
10322dc95a4dSJoseph CHAMG {
10332dc95a4dSJoseph CHAMG 	struct board_info *db = to_dm9051_board(ndev);
10342dc95a4dSJoseph CHAMG 	struct dm9051_rxctrl rxctrl;
10352dc95a4dSJoseph CHAMG 	struct netdev_hw_addr *ha;
10362dc95a4dSJoseph CHAMG 	u8 rcr = RCR_DIS_LONG | RCR_DIS_CRC | RCR_RXEN;
10372dc95a4dSJoseph CHAMG 	u32 hash_val;
10382dc95a4dSJoseph CHAMG 
10392dc95a4dSJoseph CHAMG 	memset(&rxctrl, 0, sizeof(rxctrl));
10402dc95a4dSJoseph CHAMG 
10412dc95a4dSJoseph CHAMG 	/* rx control */
10422dc95a4dSJoseph CHAMG 	if (ndev->flags & IFF_PROMISC) {
10432dc95a4dSJoseph CHAMG 		rcr |= RCR_PRMSC;
10442dc95a4dSJoseph CHAMG 		netdev_dbg(ndev, "set_multicast rcr |= RCR_PRMSC, rcr= %02x\n", rcr);
10452dc95a4dSJoseph CHAMG 	}
10462dc95a4dSJoseph CHAMG 
10472dc95a4dSJoseph CHAMG 	if (ndev->flags & IFF_ALLMULTI) {
10482dc95a4dSJoseph CHAMG 		rcr |= RCR_ALL;
10492dc95a4dSJoseph CHAMG 		netdev_dbg(ndev, "set_multicast rcr |= RCR_ALLMULTI, rcr= %02x\n", rcr);
10502dc95a4dSJoseph CHAMG 	}
10512dc95a4dSJoseph CHAMG 
10522dc95a4dSJoseph CHAMG 	rxctrl.rcr_all = rcr;
10532dc95a4dSJoseph CHAMG 
10542dc95a4dSJoseph CHAMG 	/* broadcast address */
10552dc95a4dSJoseph CHAMG 	rxctrl.hash_table[0] = 0;
10562dc95a4dSJoseph CHAMG 	rxctrl.hash_table[1] = 0;
10572dc95a4dSJoseph CHAMG 	rxctrl.hash_table[2] = 0;
10582dc95a4dSJoseph CHAMG 	rxctrl.hash_table[3] = 0x8000;
10592dc95a4dSJoseph CHAMG 
10602dc95a4dSJoseph CHAMG 	/* the multicast address in Hash Table : 64 bits */
10612dc95a4dSJoseph CHAMG 	netdev_for_each_mc_addr(ha, ndev) {
10622dc95a4dSJoseph CHAMG 		hash_val = ether_crc_le(ETH_ALEN, ha->addr) & GENMASK(5, 0);
10632dc95a4dSJoseph CHAMG 		rxctrl.hash_table[hash_val / 16] |= BIT(0) << (hash_val % 16);
10642dc95a4dSJoseph CHAMG 	}
10652dc95a4dSJoseph CHAMG 
10662dc95a4dSJoseph CHAMG 	/* schedule work to do the actual set of the data if needed */
10672dc95a4dSJoseph CHAMG 
10682dc95a4dSJoseph CHAMG 	if (memcmp(&db->rctl, &rxctrl, sizeof(rxctrl))) {
10692dc95a4dSJoseph CHAMG 		memcpy(&db->rctl, &rxctrl, sizeof(rxctrl));
10702dc95a4dSJoseph CHAMG 		schedule_work(&db->rxctrl_work);
10712dc95a4dSJoseph CHAMG 	}
10722dc95a4dSJoseph CHAMG }
10732dc95a4dSJoseph CHAMG 
10742dc95a4dSJoseph CHAMG /* event: write into the mac registers and eeprom directly
10752dc95a4dSJoseph CHAMG  */
dm9051_set_mac_address(struct net_device * ndev,void * p)10762dc95a4dSJoseph CHAMG static int dm9051_set_mac_address(struct net_device *ndev, void *p)
10772dc95a4dSJoseph CHAMG {
10782dc95a4dSJoseph CHAMG 	struct board_info *db = to_dm9051_board(ndev);
10792dc95a4dSJoseph CHAMG 	int ret;
10802dc95a4dSJoseph CHAMG 
10812dc95a4dSJoseph CHAMG 	ret = eth_prepare_mac_addr_change(ndev, p);
10822dc95a4dSJoseph CHAMG 	if (ret < 0)
10832dc95a4dSJoseph CHAMG 		return ret;
10842dc95a4dSJoseph CHAMG 
10852dc95a4dSJoseph CHAMG 	eth_commit_mac_addr_change(ndev, p);
10862dc95a4dSJoseph CHAMG 	return dm9051_set_regs(db, DM9051_PAR, ndev->dev_addr, sizeof(ndev->dev_addr));
10872dc95a4dSJoseph CHAMG }
10882dc95a4dSJoseph CHAMG 
10892dc95a4dSJoseph CHAMG static const struct net_device_ops dm9051_netdev_ops = {
10902dc95a4dSJoseph CHAMG 	.ndo_open = dm9051_open,
10912dc95a4dSJoseph CHAMG 	.ndo_stop = dm9051_stop,
10922dc95a4dSJoseph CHAMG 	.ndo_start_xmit = dm9051_start_xmit,
10932dc95a4dSJoseph CHAMG 	.ndo_set_rx_mode = dm9051_set_rx_mode,
10942dc95a4dSJoseph CHAMG 	.ndo_validate_addr = eth_validate_addr,
10952dc95a4dSJoseph CHAMG 	.ndo_set_mac_address = dm9051_set_mac_address,
10962dc95a4dSJoseph CHAMG };
10972dc95a4dSJoseph CHAMG 
dm9051_operation_clear(struct board_info * db)10982dc95a4dSJoseph CHAMG static void dm9051_operation_clear(struct board_info *db)
10992dc95a4dSJoseph CHAMG {
11002dc95a4dSJoseph CHAMG 	db->bc.status_err_counter = 0;
11012dc95a4dSJoseph CHAMG 	db->bc.large_err_counter = 0;
11022dc95a4dSJoseph CHAMG 	db->bc.rx_err_counter = 0;
11032dc95a4dSJoseph CHAMG 	db->bc.tx_err_counter = 0;
11042dc95a4dSJoseph CHAMG 	db->bc.fifo_rst_counter = 0;
11052dc95a4dSJoseph CHAMG }
11062dc95a4dSJoseph CHAMG 
dm9051_mdio_register(struct board_info * db)11072dc95a4dSJoseph CHAMG static int dm9051_mdio_register(struct board_info *db)
11082dc95a4dSJoseph CHAMG {
11092dc95a4dSJoseph CHAMG 	struct spi_device *spi = db->spidev;
11102dc95a4dSJoseph CHAMG 	int ret;
11112dc95a4dSJoseph CHAMG 
11122dc95a4dSJoseph CHAMG 	db->mdiobus = devm_mdiobus_alloc(&spi->dev);
11132dc95a4dSJoseph CHAMG 	if (!db->mdiobus)
11142dc95a4dSJoseph CHAMG 		return -ENOMEM;
11152dc95a4dSJoseph CHAMG 
11162dc95a4dSJoseph CHAMG 	db->mdiobus->priv = db;
11172dc95a4dSJoseph CHAMG 	db->mdiobus->read = dm9051_mdio_read;
11182dc95a4dSJoseph CHAMG 	db->mdiobus->write = dm9051_mdio_write;
11192dc95a4dSJoseph CHAMG 	db->mdiobus->name = "dm9051-mdiobus";
11202dc95a4dSJoseph CHAMG 	db->mdiobus->phy_mask = (u32)~BIT(1);
11212dc95a4dSJoseph CHAMG 	db->mdiobus->parent = &spi->dev;
11222dc95a4dSJoseph CHAMG 	snprintf(db->mdiobus->id, MII_BUS_ID_SIZE,
112325fd0550SAmit Kumar Mahapatra 		 "dm9051-%s.%u", dev_name(&spi->dev), spi_get_chipselect(spi, 0));
11242dc95a4dSJoseph CHAMG 
11252dc95a4dSJoseph CHAMG 	ret = devm_mdiobus_register(&spi->dev, db->mdiobus);
11262dc95a4dSJoseph CHAMG 	if (ret)
11272dc95a4dSJoseph CHAMG 		dev_err(&spi->dev, "Could not register MDIO bus\n");
11282dc95a4dSJoseph CHAMG 
11292dc95a4dSJoseph CHAMG 	return ret;
11302dc95a4dSJoseph CHAMG }
11312dc95a4dSJoseph CHAMG 
dm9051_handle_link_change(struct net_device * ndev)11322dc95a4dSJoseph CHAMG static void dm9051_handle_link_change(struct net_device *ndev)
11332dc95a4dSJoseph CHAMG {
11342dc95a4dSJoseph CHAMG 	struct board_info *db = to_dm9051_board(ndev);
11352dc95a4dSJoseph CHAMG 
11362dc95a4dSJoseph CHAMG 	phy_print_status(db->phydev);
11372dc95a4dSJoseph CHAMG 
11382dc95a4dSJoseph CHAMG 	/* only write pause settings to mac. since mac and phy are integrated
11392dc95a4dSJoseph CHAMG 	 * together, such as link state, speed and duplex are sync already
11402dc95a4dSJoseph CHAMG 	 */
11412dc95a4dSJoseph CHAMG 	if (db->phydev->link) {
11422dc95a4dSJoseph CHAMG 		if (db->phydev->pause) {
11432dc95a4dSJoseph CHAMG 			db->pause.rx_pause = true;
11442dc95a4dSJoseph CHAMG 			db->pause.tx_pause = true;
11452dc95a4dSJoseph CHAMG 		}
11462dc95a4dSJoseph CHAMG 		dm9051_update_fcr(db);
11472dc95a4dSJoseph CHAMG 	}
11482dc95a4dSJoseph CHAMG }
11492dc95a4dSJoseph CHAMG 
11502dc95a4dSJoseph CHAMG /* phy connect as poll mode
11512dc95a4dSJoseph CHAMG  */
dm9051_phy_connect(struct board_info * db)11522dc95a4dSJoseph CHAMG static int dm9051_phy_connect(struct board_info *db)
11532dc95a4dSJoseph CHAMG {
11542dc95a4dSJoseph CHAMG 	char phy_id[MII_BUS_ID_SIZE + 3];
11552dc95a4dSJoseph CHAMG 
11562dc95a4dSJoseph CHAMG 	snprintf(phy_id, sizeof(phy_id), PHY_ID_FMT,
11572dc95a4dSJoseph CHAMG 		 db->mdiobus->id, DM9051_PHY_ADDR);
11582dc95a4dSJoseph CHAMG 
11592dc95a4dSJoseph CHAMG 	db->phydev = phy_connect(db->ndev, phy_id, dm9051_handle_link_change,
11602dc95a4dSJoseph CHAMG 				 PHY_INTERFACE_MODE_MII);
11612dc95a4dSJoseph CHAMG 	return PTR_ERR_OR_ZERO(db->phydev);
11622dc95a4dSJoseph CHAMG }
11632dc95a4dSJoseph CHAMG 
dm9051_probe(struct spi_device * spi)11642dc95a4dSJoseph CHAMG static int dm9051_probe(struct spi_device *spi)
11652dc95a4dSJoseph CHAMG {
11662dc95a4dSJoseph CHAMG 	struct device *dev = &spi->dev;
11672dc95a4dSJoseph CHAMG 	struct net_device *ndev;
11682dc95a4dSJoseph CHAMG 	struct board_info *db;
11692dc95a4dSJoseph CHAMG 	int ret;
11702dc95a4dSJoseph CHAMG 
11712dc95a4dSJoseph CHAMG 	ndev = devm_alloc_etherdev(dev, sizeof(struct board_info));
11722dc95a4dSJoseph CHAMG 	if (!ndev)
11732dc95a4dSJoseph CHAMG 		return -ENOMEM;
11742dc95a4dSJoseph CHAMG 
11752dc95a4dSJoseph CHAMG 	SET_NETDEV_DEV(ndev, dev);
11762dc95a4dSJoseph CHAMG 	dev_set_drvdata(dev, ndev);
11772dc95a4dSJoseph CHAMG 
11782dc95a4dSJoseph CHAMG 	db = netdev_priv(ndev);
11792dc95a4dSJoseph CHAMG 
11802dc95a4dSJoseph CHAMG 	db->msg_enable = 0;
11812dc95a4dSJoseph CHAMG 	db->spidev = spi;
11822dc95a4dSJoseph CHAMG 	db->ndev = ndev;
11832dc95a4dSJoseph CHAMG 
11842dc95a4dSJoseph CHAMG 	ndev->netdev_ops = &dm9051_netdev_ops;
11852dc95a4dSJoseph CHAMG 	ndev->ethtool_ops = &dm9051_ethtool_ops;
11862dc95a4dSJoseph CHAMG 
11872dc95a4dSJoseph CHAMG 	mutex_init(&db->spi_lockm);
11882dc95a4dSJoseph CHAMG 	mutex_init(&db->reg_mutex);
11892dc95a4dSJoseph CHAMG 
11902dc95a4dSJoseph CHAMG 	INIT_WORK(&db->rxctrl_work, dm9051_rxctl_delay);
11912dc95a4dSJoseph CHAMG 	INIT_WORK(&db->tx_work, dm9051_tx_delay);
11922dc95a4dSJoseph CHAMG 
11932dc95a4dSJoseph CHAMG 	ret = dm9051_map_init(spi, db);
11942dc95a4dSJoseph CHAMG 	if (ret)
11952dc95a4dSJoseph CHAMG 		return ret;
11962dc95a4dSJoseph CHAMG 
11972dc95a4dSJoseph CHAMG 	ret = dm9051_map_chipid(db);
11982dc95a4dSJoseph CHAMG 	if (ret)
11992dc95a4dSJoseph CHAMG 		return ret;
12002dc95a4dSJoseph CHAMG 
12012dc95a4dSJoseph CHAMG 	ret = dm9051_map_etherdev_par(ndev, db);
12022dc95a4dSJoseph CHAMG 	if (ret < 0)
12032dc95a4dSJoseph CHAMG 		return ret;
12042dc95a4dSJoseph CHAMG 
12052dc95a4dSJoseph CHAMG 	ret = dm9051_mdio_register(db);
12062dc95a4dSJoseph CHAMG 	if (ret)
12072dc95a4dSJoseph CHAMG 		return ret;
12082dc95a4dSJoseph CHAMG 
12092dc95a4dSJoseph CHAMG 	ret = dm9051_phy_connect(db);
12102dc95a4dSJoseph CHAMG 	if (ret)
12112dc95a4dSJoseph CHAMG 		return ret;
12122dc95a4dSJoseph CHAMG 
12132dc95a4dSJoseph CHAMG 	dm9051_operation_clear(db);
12142dc95a4dSJoseph CHAMG 	skb_queue_head_init(&db->txq);
12152dc95a4dSJoseph CHAMG 
12162dc95a4dSJoseph CHAMG 	ret = devm_register_netdev(dev, ndev);
12172dc95a4dSJoseph CHAMG 	if (ret) {
12182dc95a4dSJoseph CHAMG 		phy_disconnect(db->phydev);
12192dc95a4dSJoseph CHAMG 		return dev_err_probe(dev, ret, "device register failed");
12202dc95a4dSJoseph CHAMG 	}
12212dc95a4dSJoseph CHAMG 
12222dc95a4dSJoseph CHAMG 	return 0;
12232dc95a4dSJoseph CHAMG }
12242dc95a4dSJoseph CHAMG 
dm9051_drv_remove(struct spi_device * spi)12250b9e69e1SStephen Rothwell static void dm9051_drv_remove(struct spi_device *spi)
12262dc95a4dSJoseph CHAMG {
12272dc95a4dSJoseph CHAMG 	struct device *dev = &spi->dev;
12282dc95a4dSJoseph CHAMG 	struct net_device *ndev = dev_get_drvdata(dev);
12292dc95a4dSJoseph CHAMG 	struct board_info *db = to_dm9051_board(ndev);
12302dc95a4dSJoseph CHAMG 
12312dc95a4dSJoseph CHAMG 	phy_disconnect(db->phydev);
12322dc95a4dSJoseph CHAMG }
12332dc95a4dSJoseph CHAMG 
12342dc95a4dSJoseph CHAMG static const struct of_device_id dm9051_match_table[] = {
12352dc95a4dSJoseph CHAMG 	{ .compatible = "davicom,dm9051" },
12362dc95a4dSJoseph CHAMG 	{}
12372dc95a4dSJoseph CHAMG };
12382dc95a4dSJoseph CHAMG 
12392dc95a4dSJoseph CHAMG static const struct spi_device_id dm9051_id_table[] = {
12402dc95a4dSJoseph CHAMG 	{ "dm9051", 0 },
12412dc95a4dSJoseph CHAMG 	{}
12422dc95a4dSJoseph CHAMG };
12432dc95a4dSJoseph CHAMG 
12442dc95a4dSJoseph CHAMG static struct spi_driver dm9051_driver = {
12452dc95a4dSJoseph CHAMG 	.driver = {
12462dc95a4dSJoseph CHAMG 		.name = DRVNAME_9051,
12472dc95a4dSJoseph CHAMG 		.of_match_table = dm9051_match_table,
12482dc95a4dSJoseph CHAMG 	},
12492dc95a4dSJoseph CHAMG 	.probe = dm9051_probe,
12502dc95a4dSJoseph CHAMG 	.remove = dm9051_drv_remove,
12512dc95a4dSJoseph CHAMG 	.id_table = dm9051_id_table,
12522dc95a4dSJoseph CHAMG };
12532dc95a4dSJoseph CHAMG module_spi_driver(dm9051_driver);
12542dc95a4dSJoseph CHAMG 
12552dc95a4dSJoseph CHAMG MODULE_AUTHOR("Joseph CHANG <joseph_chang@davicom.com.tw>");
12562dc95a4dSJoseph CHAMG MODULE_DESCRIPTION("Davicom DM9051 network SPI driver");
12572dc95a4dSJoseph CHAMG MODULE_LICENSE("GPL");
1258