2007-11-12 21:31:11 -08:00
/*
* RDC R6040 Fast Ethernet MAC support
*
* Copyright ( C ) 2004 Sten Wang < sten . wang @ rdc . com . tw >
* Copyright ( C ) 2007
2007-11-28 23:02:33 +01:00
* Daniel Gimpelevich < daniel @ gimpelevich . san - francisco . ca . us >
2007-11-12 21:31:11 -08:00
* Florian Fainelli < florian @ openwrt . org >
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation ; either version 2
* of the License , or ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the
* Free Software Foundation , Inc . , 51 Franklin Street , Fifth Floor ,
* Boston , MA 02110 - 1301 , USA .
*/
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/moduleparam.h>
# include <linux/string.h>
# include <linux/timer.h>
# include <linux/errno.h>
# include <linux/ioport.h>
# include <linux/interrupt.h>
# include <linux/pci.h>
# include <linux/netdevice.h>
# include <linux/etherdevice.h>
# include <linux/skbuff.h>
# include <linux/init.h>
# include <linux/delay.h>
# include <linux/mii.h>
# include <linux/ethtool.h>
# include <linux/crc32.h>
# include <linux/spinlock.h>
2007-11-23 21:49:27 -05:00
# include <linux/bitops.h>
# include <linux/io.h>
# include <linux/irq.h>
# include <linux/uaccess.h>
2010-05-31 09:18:57 +00:00
# include <linux/phy.h>
2007-11-12 21:31:11 -08:00
# include <asm/processor.h>
# define DRV_NAME "r6040"
2010-05-31 09:19:04 +00:00
# define DRV_VERSION "0.26"
# define DRV_RELDATE "30May2010"
2007-11-12 21:31:11 -08:00
/* PHY CHIP Address */
# define PHY1_ADDR 1 /* For MAC1 */
2009-03-24 23:34:35 +00:00
# define PHY2_ADDR 3 /* For MAC2 */
2007-11-12 21:31:11 -08:00
# define PHY_MODE 0x3100 /* PHY CHIP Register 0 */
# define PHY_CAP 0x01E1 /* PHY CHIP Register 4 */
/* Time in jiffies before concluding the transmitter is hung. */
2007-11-28 23:02:33 +01:00
# define TX_TIMEOUT (6000 * HZ / 1000)
2007-11-12 21:31:11 -08:00
/* RDC MAC I/O Size */
# define R6040_IO_SIZE 256
/* MAX RDC MAC */
# define MAX_MAC 2
/* MAC registers */
# define MCR0 0x00 /* Control register 0 */
# define MCR1 0x04 /* Control register 1 */
# define MAC_RST 0x0001 /* Reset the MAC */
# define MBCR 0x08 /* Bus control */
# define MT_ICR 0x0C /* TX interrupt control */
# define MR_ICR 0x10 /* RX interrupt control */
# define MTPR 0x14 /* TX poll command register */
# define MR_BSR 0x18 /* RX buffer size */
# define MR_DCR 0x1A /* RX descriptor control */
# define MLSR 0x1C /* Last status */
# define MMDIO 0x20 /* MDIO control register */
# define MDIO_WRITE 0x4000 /* MDIO write */
# define MDIO_READ 0x2000 /* MDIO read */
# define MMRD 0x24 /* MDIO read data register */
# define MMWD 0x28 /* MDIO write data register */
# define MTD_SA0 0x2C /* TX descriptor start address 0 */
# define MTD_SA1 0x30 /* TX descriptor start address 1 */
# define MRD_SA0 0x34 /* RX descriptor start address 0 */
# define MRD_SA1 0x38 /* RX descriptor start address 1 */
# define MISR 0x3C /* Status register */
# define MIER 0x40 /* INT enable register */
# define MSK_INT 0x0000 /* Mask off interrupts */
2008-07-13 14:28:27 +02:00
# define RX_FINISH 0x0001 /* RX finished */
# define RX_NO_DESC 0x0002 /* No RX descriptor available */
# define RX_FIFO_FULL 0x0004 /* RX FIFO full */
# define RX_EARLY 0x0008 /* RX early */
# define TX_FINISH 0x0010 /* TX finished */
# define TX_EARLY 0x0080 /* TX early */
# define EVENT_OVRFL 0x0100 /* Event counter overflow */
# define LINK_CHANGED 0x0200 /* PHY link changed */
2007-11-12 21:31:11 -08:00
# define ME_CISR 0x44 /* Event counter INT status */
# define ME_CIER 0x48 /* Event counter INT enable */
# define MR_CNT 0x50 /* Successfully received packet counter */
# define ME_CNT0 0x52 /* Event counter 0 */
# define ME_CNT1 0x54 /* Event counter 1 */
# define ME_CNT2 0x56 /* Event counter 2 */
# define ME_CNT3 0x58 /* Event counter 3 */
# define MT_CNT 0x5A /* Successfully transmit packet counter */
# define ME_CNT4 0x5C /* Event counter 4 */
# define MP_CNT 0x5E /* Pause frame counter register */
# define MAR0 0x60 /* Hash table 0 */
# define MAR1 0x62 /* Hash table 1 */
# define MAR2 0x64 /* Hash table 2 */
# define MAR3 0x66 /* Hash table 3 */
# define MID_0L 0x68 /* Multicast address MID0 Low */
# define MID_0M 0x6A /* Multicast address MID0 Medium */
# define MID_0H 0x6C /* Multicast address MID0 High */
# define MID_1L 0x70 /* MID1 Low */
# define MID_1M 0x72 /* MID1 Medium */
# define MID_1H 0x74 /* MID1 High */
# define MID_2L 0x78 /* MID2 Low */
# define MID_2M 0x7A /* MID2 Medium */
# define MID_2H 0x7C /* MID2 High */
# define MID_3L 0x80 /* MID3 Low */
# define MID_3M 0x82 /* MID3 Medium */
# define MID_3H 0x84 /* MID3 High */
# define PHY_CC 0x88 /* PHY status change configuration register */
# define PHY_ST 0x8A /* PHY status register */
# define MAC_SM 0xAC /* MAC status machine */
# define MAC_ID 0xBE /* Identifier register */
# define TX_DCNT 0x80 /* TX descriptor count */
# define RX_DCNT 0x80 /* RX descriptor count */
# define MAX_BUF_SIZE 0x600
2007-11-28 22:31:00 +01:00
# define RX_DESC_SIZE (RX_DCNT * sizeof(struct r6040_descriptor))
# define TX_DESC_SIZE (TX_DCNT * sizeof(struct r6040_descriptor))
2007-11-12 21:31:11 -08:00
# define MBCR_DEFAULT 0x012A /* MAC Bus Control Register */
2010-04-07 16:50:58 -07:00
# define MCAST_MAX 3 /* Max number multicast addresses to filter */
2007-11-12 21:31:11 -08:00
2008-07-13 14:34:15 +02:00
/* Descriptor status */
# define DSC_OWNER_MAC 0x8000 /* MAC is the owner of this descriptor */
# define DSC_RX_OK 0x4000 /* RX was successful */
# define DSC_RX_ERR 0x0800 /* RX PHY error */
# define DSC_RX_ERR_DRI 0x0400 /* RX dribble packet */
# define DSC_RX_ERR_BUF 0x0200 /* RX length exceeds buffer size */
# define DSC_RX_ERR_LONG 0x0100 /* RX length > maximum packet length */
# define DSC_RX_ERR_RUNT 0x0080 /* RX packet length < 64 byte */
# define DSC_RX_ERR_CRC 0x0040 /* RX CRC error */
# define DSC_RX_BCAST 0x0020 /* RX broadcast (no error) */
# define DSC_RX_MCAST 0x0010 /* RX multicast (no error) */
# define DSC_RX_MCH_HIT 0x0008 /* RX multicast hit in hash table (no error) */
# define DSC_RX_MIDH_HIT 0x0004 /* RX MID table hit (no error) */
# define DSC_RX_IDX_MID_MASK 3 /* RX mask for the index of matched MIDx */
2007-11-12 21:31:11 -08:00
/* PHY settings */
# define ICPLUS_PHY_ID 0x0243
MODULE_AUTHOR ( " Sten Wang <sten.wang@rdc.com.tw>, "
" Daniel Gimpelevich <daniel@gimpelevich.san-francisco.ca.us>, "
" Florian Fainelli <florian@openwrt.org> " ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_DESCRIPTION ( " RDC R6040 NAPI PCI FastEthernet driver " ) ;
2009-04-08 15:50:43 -07:00
MODULE_VERSION ( DRV_VERSION " " DRV_RELDATE ) ;
2007-11-12 21:31:11 -08:00
2008-07-13 14:28:27 +02:00
/* RX and TX interrupts that we handle */
2008-07-13 14:35:32 +02:00
# define RX_INTS (RX_FIFO_FULL | RX_NO_DESC | RX_FINISH)
# define TX_INTS (TX_FINISH)
# define INT_MASK (RX_INTS | TX_INTS)
2007-11-12 21:31:11 -08:00
struct r6040_descriptor {
u16 status , len ; /* 0-3 */
__le32 buf ; /* 4-7 */
__le32 ndesc ; /* 8-B */
u32 rev1 ; /* C-F */
char * vbufp ; /* 10-13 */
struct r6040_descriptor * vndescp ; /* 14-17 */
struct sk_buff * skb_ptr ; /* 18-1B */
u32 rev2 ; /* 1C-1F */
} __attribute__ ( ( aligned ( 32 ) ) ) ;
struct r6040_private {
spinlock_t lock ; /* driver lock */
struct pci_dev * pdev ;
struct r6040_descriptor * rx_insert_ptr ;
struct r6040_descriptor * rx_remove_ptr ;
struct r6040_descriptor * tx_insert_ptr ;
struct r6040_descriptor * tx_remove_ptr ;
2007-11-28 22:31:00 +01:00
struct r6040_descriptor * rx_ring ;
struct r6040_descriptor * tx_ring ;
dma_addr_t rx_ring_dma ;
dma_addr_t tx_ring_dma ;
2010-05-31 09:18:57 +00:00
u16 tx_free_desc , phy_addr ;
2007-11-12 21:31:11 -08:00
u16 mcr0 , mcr1 ;
struct net_device * dev ;
2010-05-31 09:18:57 +00:00
struct mii_bus * mii_bus ;
2007-11-12 21:31:11 -08:00
struct napi_struct napi ;
void __iomem * base ;
2010-05-31 09:18:57 +00:00
struct phy_device * phydev ;
int old_link ;
int old_duplex ;
2007-11-12 21:31:11 -08:00
} ;
2010-08-08 10:08:44 +00:00
static char version [ ] __devinitdata = DRV_NAME
2007-11-12 21:31:11 -08:00
" : RDC R6040 NAPI net driver, "
2009-01-08 11:00:52 -08:00
" version " DRV_VERSION " ( " DRV_RELDATE " ) " ;
2007-11-12 21:31:11 -08:00
2007-11-23 21:49:27 -05:00
static int phy_table [ ] = { PHY1_ADDR , PHY2_ADDR } ;
2007-11-12 21:31:11 -08:00
/* Read a word data from PHY Chip */
2008-07-13 13:39:32 +02:00
static int r6040_phy_read ( void __iomem * ioaddr , int phy_addr , int reg )
2007-11-12 21:31:11 -08:00
{
int limit = 2048 ;
u16 cmd ;
iowrite16 ( MDIO_READ + reg + ( phy_addr < < 8 ) , ioaddr + MMDIO ) ;
/* Wait for the read bit to be cleared */
while ( limit - - ) {
cmd = ioread16 ( ioaddr + MMDIO ) ;
2008-12-22 19:38:17 -08:00
if ( ! ( cmd & MDIO_READ ) )
2007-11-12 21:31:11 -08:00
break ;
}
return ioread16 ( ioaddr + MMRD ) ;
}
/* Write a word data from PHY Chip */
2010-08-08 10:08:44 +00:00
static void r6040_phy_write ( void __iomem * ioaddr ,
int phy_addr , int reg , u16 val )
2007-11-12 21:31:11 -08:00
{
int limit = 2048 ;
u16 cmd ;
iowrite16 ( val , ioaddr + MMWD ) ;
/* Write the command to the MDIO bus */
iowrite16 ( MDIO_WRITE + reg + ( phy_addr < < 8 ) , ioaddr + MMDIO ) ;
/* Wait for the write bit to be cleared */
while ( limit - - ) {
cmd = ioread16 ( ioaddr + MMDIO ) ;
2008-12-22 19:38:17 -08:00
if ( ! ( cmd & MDIO_WRITE ) )
2007-11-12 21:31:11 -08:00
break ;
}
}
2010-05-31 09:18:57 +00:00
static int r6040_mdiobus_read ( struct mii_bus * bus , int phy_addr , int reg )
2007-11-12 21:31:11 -08:00
{
2010-05-31 09:18:57 +00:00
struct net_device * dev = bus - > priv ;
2007-11-12 21:31:11 -08:00
struct r6040_private * lp = netdev_priv ( dev ) ;
void __iomem * ioaddr = lp - > base ;
2010-05-31 09:18:57 +00:00
return r6040_phy_read ( ioaddr , phy_addr , reg ) ;
2007-11-12 21:31:11 -08:00
}
2010-05-31 09:18:57 +00:00
static int r6040_mdiobus_write ( struct mii_bus * bus , int phy_addr ,
int reg , u16 value )
2007-11-12 21:31:11 -08:00
{
2010-05-31 09:18:57 +00:00
struct net_device * dev = bus - > priv ;
2007-11-12 21:31:11 -08:00
struct r6040_private * lp = netdev_priv ( dev ) ;
void __iomem * ioaddr = lp - > base ;
2010-05-31 09:18:57 +00:00
r6040_phy_write ( ioaddr , phy_addr , reg , value ) ;
return 0 ;
}
static int r6040_mdiobus_reset ( struct mii_bus * bus )
{
return 0 ;
2007-11-12 21:31:11 -08:00
}
2007-12-12 22:55:34 +01:00
static void r6040_free_txbufs ( struct net_device * dev )
{
struct r6040_private * lp = netdev_priv ( dev ) ;
int i ;
for ( i = 0 ; i < TX_DCNT ; i + + ) {
if ( lp - > tx_insert_ptr - > skb_ptr ) {
2008-03-16 22:43:06 +00:00
pci_unmap_single ( lp - > pdev ,
le32_to_cpu ( lp - > tx_insert_ptr - > buf ) ,
2007-12-12 22:55:34 +01:00
MAX_BUF_SIZE , PCI_DMA_TODEVICE ) ;
dev_kfree_skb ( lp - > tx_insert_ptr - > skb_ptr ) ;
2008-09-24 21:16:40 +02:00
lp - > tx_insert_ptr - > skb_ptr = NULL ;
2007-12-12 22:55:34 +01:00
}
lp - > tx_insert_ptr = lp - > tx_insert_ptr - > vndescp ;
}
}
static void r6040_free_rxbufs ( struct net_device * dev )
{
struct r6040_private * lp = netdev_priv ( dev ) ;
int i ;
for ( i = 0 ; i < RX_DCNT ; i + + ) {
if ( lp - > rx_insert_ptr - > skb_ptr ) {
2008-03-16 22:43:06 +00:00
pci_unmap_single ( lp - > pdev ,
le32_to_cpu ( lp - > rx_insert_ptr - > buf ) ,
2007-12-12 22:55:34 +01:00
MAX_BUF_SIZE , PCI_DMA_FROMDEVICE ) ;
dev_kfree_skb ( lp - > rx_insert_ptr - > skb_ptr ) ;
lp - > rx_insert_ptr - > skb_ptr = NULL ;
}
lp - > rx_insert_ptr = lp - > rx_insert_ptr - > vndescp ;
}
}
static void r6040_init_ring_desc ( struct r6040_descriptor * desc_ring ,
dma_addr_t desc_dma , int size )
{
struct r6040_descriptor * desc = desc_ring ;
dma_addr_t mapping = desc_dma ;
while ( size - - > 0 ) {
2008-06-23 23:12:31 +02:00
mapping + = sizeof ( * desc ) ;
2007-12-12 22:55:34 +01:00
desc - > ndesc = cpu_to_le32 ( mapping ) ;
desc - > vndescp = desc + 1 ;
desc + + ;
}
desc - - ;
desc - > ndesc = cpu_to_le32 ( desc_dma ) ;
desc - > vndescp = desc_ring ;
}
2008-07-13 14:32:18 +02:00
static void r6040_init_txbufs ( struct net_device * dev )
2007-12-12 22:55:34 +01:00
{
struct r6040_private * lp = netdev_priv ( dev ) ;
lp - > tx_free_desc = TX_DCNT ;
lp - > tx_remove_ptr = lp - > tx_insert_ptr = lp - > tx_ring ;
r6040_init_ring_desc ( lp - > tx_ring , lp - > tx_ring_dma , TX_DCNT ) ;
}
2008-07-13 14:32:18 +02:00
static int r6040_alloc_rxbufs ( struct net_device * dev )
2007-12-12 22:55:34 +01:00
{
struct r6040_private * lp = netdev_priv ( dev ) ;
2008-07-13 14:32:18 +02:00
struct r6040_descriptor * desc ;
struct sk_buff * skb ;
int rc ;
2007-12-12 22:55:34 +01:00
lp - > rx_remove_ptr = lp - > rx_insert_ptr = lp - > rx_ring ;
r6040_init_ring_desc ( lp - > rx_ring , lp - > rx_ring_dma , RX_DCNT ) ;
2008-07-13 14:32:18 +02:00
/* Allocate skbs for the rx descriptors */
desc = lp - > rx_ring ;
do {
skb = netdev_alloc_skb ( dev , MAX_BUF_SIZE ) ;
if ( ! skb ) {
2010-04-07 21:39:27 +00:00
netdev_err ( dev , " failed to alloc skb for rx \n " ) ;
2008-07-13 14:32:18 +02:00
rc = - ENOMEM ;
goto err_exit ;
}
desc - > skb_ptr = skb ;
desc - > buf = cpu_to_le32 ( pci_map_single ( lp - > pdev ,
2010-08-08 10:08:44 +00:00
desc - > skb_ptr - > data ,
MAX_BUF_SIZE , PCI_DMA_FROMDEVICE ) ) ;
2008-07-13 14:34:15 +02:00
desc - > status = DSC_OWNER_MAC ;
2008-07-13 14:32:18 +02:00
desc = desc - > vndescp ;
} while ( desc ! = lp - > rx_ring ) ;
return 0 ;
err_exit :
/* Deallocate all previously allocated skbs */
r6040_free_rxbufs ( dev ) ;
return rc ;
2008-07-13 14:29:20 +02:00
}
static void r6040_init_mac_regs ( struct net_device * dev )
{
struct r6040_private * lp = netdev_priv ( dev ) ;
void __iomem * ioaddr = lp - > base ;
int limit = 2048 ;
u16 cmd ;
/* Mask Off Interrupt */
iowrite16 ( MSK_INT , ioaddr + MIER ) ;
/* Reset RDC MAC */
iowrite16 ( MAC_RST , ioaddr + MCR1 ) ;
while ( limit - - ) {
cmd = ioread16 ( ioaddr + MCR1 ) ;
if ( cmd & 0x1 )
break ;
}
/* Reset internal state machine */
iowrite16 ( 2 , ioaddr + MAC_SM ) ;
iowrite16 ( 0 , ioaddr + MAC_SM ) ;
2008-09-03 16:50:03 +02:00
mdelay ( 5 ) ;
2008-07-13 14:29:20 +02:00
/* MAC Bus Control Register */
iowrite16 ( MBCR_DEFAULT , ioaddr + MBCR ) ;
/* Buffer Size Register */
iowrite16 ( MAX_BUF_SIZE , ioaddr + MR_BSR ) ;
/* Write TX ring start address */
iowrite16 ( lp - > tx_ring_dma , ioaddr + MTD_SA0 ) ;
iowrite16 ( lp - > tx_ring_dma > > 16 , ioaddr + MTD_SA1 ) ;
2007-12-12 22:55:34 +01:00
2008-07-13 14:29:20 +02:00
/* Write RX ring start address */
2007-12-12 22:55:34 +01:00
iowrite16 ( lp - > rx_ring_dma , ioaddr + MRD_SA0 ) ;
iowrite16 ( lp - > rx_ring_dma > > 16 , ioaddr + MRD_SA1 ) ;
2008-07-13 14:29:20 +02:00
/* Set interrupt waiting time and packet numbers */
2008-07-13 14:35:00 +02:00
iowrite16 ( 0 , ioaddr + MT_ICR ) ;
iowrite16 ( 0 , ioaddr + MR_ICR ) ;
2008-07-13 14:29:20 +02:00
/* Enable interrupts */
iowrite16 ( INT_MASK , ioaddr + MIER ) ;
/* Enable TX and RX */
iowrite16 ( lp - > mcr0 | 0x0002 , ioaddr ) ;
/* Let TX poll the descriptors
* we may got called by r6040_tx_timeout which has left
* some unsent tx buffers */
iowrite16 ( 0x01 , ioaddr + MTPR ) ;
2007-12-12 22:55:34 +01:00
}
2007-11-12 21:31:11 -08:00
2007-12-12 23:01:33 +01:00
static void r6040_tx_timeout ( struct net_device * dev )
{
struct r6040_private * priv = netdev_priv ( dev ) ;
void __iomem * ioaddr = priv - > base ;
2010-04-07 21:39:27 +00:00
netdev_warn ( dev , " transmit timed out, int enable %4.4x "
2010-05-31 09:18:57 +00:00
" status %4.4x \n " ,
2010-04-07 21:39:27 +00:00
ioread16 ( ioaddr + MIER ) ,
2010-05-31 09:18:57 +00:00
ioread16 ( ioaddr + MISR ) ) ;
2007-12-12 23:01:33 +01:00
dev - > stats . tx_errors + + ;
2008-07-13 14:29:20 +02:00
/* Reset MAC and re-init all registers */
r6040_init_mac_regs ( dev ) ;
2007-12-12 23:01:33 +01:00
}
2007-11-12 21:31:11 -08:00
static struct net_device_stats * r6040_get_stats ( struct net_device * dev )
{
struct r6040_private * priv = netdev_priv ( dev ) ;
void __iomem * ioaddr = priv - > base ;
unsigned long flags ;
spin_lock_irqsave ( & priv - > lock , flags ) ;
2007-12-12 22:34:55 +01:00
dev - > stats . rx_crc_errors + = ioread8 ( ioaddr + ME_CNT1 ) ;
dev - > stats . multicast + = ioread8 ( ioaddr + ME_CNT0 ) ;
2007-11-12 21:31:11 -08:00
spin_unlock_irqrestore ( & priv - > lock , flags ) ;
2007-12-12 22:34:55 +01:00
return & dev - > stats ;
2007-11-12 21:31:11 -08:00
}
/* Stop RDC MAC and Free the allocated resource */
static void r6040_down ( struct net_device * dev )
{
struct r6040_private * lp = netdev_priv ( dev ) ;
void __iomem * ioaddr = lp - > base ;
int limit = 2048 ;
u16 * adrp ;
u16 cmd ;
/* Stop MAC */
iowrite16 ( MSK_INT , ioaddr + MIER ) ; /* Mask Off Interrupt */
iowrite16 ( MAC_RST , ioaddr + MCR1 ) ; /* Reset RDC MAC */
while ( limit - - ) {
cmd = ioread16 ( ioaddr + MCR1 ) ;
if ( cmd & 0x1 )
break ;
}
/* Restore MAC Address to MIDx */
adrp = ( u16 * ) dev - > dev_addr ;
iowrite16 ( adrp [ 0 ] , ioaddr + MID_0L ) ;
iowrite16 ( adrp [ 1 ] , ioaddr + MID_0M ) ;
iowrite16 ( adrp [ 2 ] , ioaddr + MID_0H ) ;
}
2007-11-28 23:02:33 +01:00
static int r6040_close ( struct net_device * dev )
2007-11-12 21:31:11 -08:00
{
struct r6040_private * lp = netdev_priv ( dev ) ;
2009-01-09 23:19:26 -08:00
struct pci_dev * pdev = lp - > pdev ;
2007-11-12 21:31:11 -08:00
spin_lock_irq ( & lp - > lock ) ;
2008-07-13 14:32:45 +02:00
napi_disable ( & lp - > napi ) ;
2007-11-12 21:31:11 -08:00
netif_stop_queue ( dev ) ;
r6040_down ( dev ) ;
2009-01-09 23:19:26 -08:00
free_irq ( dev - > irq , dev ) ;
/* Free RX buffer */
r6040_free_rxbufs ( dev ) ;
/* Free TX buffer */
r6040_free_txbufs ( dev ) ;
2007-11-12 21:31:11 -08:00
spin_unlock_irq ( & lp - > lock ) ;
2009-01-09 23:19:26 -08:00
/* Free Descriptor memory */
if ( lp - > rx_ring ) {
2010-08-08 10:08:44 +00:00
pci_free_consistent ( pdev ,
RX_DESC_SIZE , lp - > rx_ring , lp - > rx_ring_dma ) ;
2009-02-14 11:14:04 +00:00
lp - > rx_ring = NULL ;
2009-01-09 23:19:26 -08:00
}
if ( lp - > tx_ring ) {
2010-08-08 10:08:44 +00:00
pci_free_consistent ( pdev ,
TX_DESC_SIZE , lp - > tx_ring , lp - > tx_ring_dma ) ;
2009-02-14 11:14:04 +00:00
lp - > tx_ring = NULL ;
2009-01-09 23:19:26 -08:00
}
2007-11-12 21:31:11 -08:00
return 0 ;
}
static int r6040_ioctl ( struct net_device * dev , struct ifreq * rq , int cmd )
{
struct r6040_private * lp = netdev_priv ( dev ) ;
2010-05-31 09:18:57 +00:00
if ( ! lp - > phydev )
2007-11-12 21:31:11 -08:00
return - EINVAL ;
2010-05-31 09:18:57 +00:00
2010-07-21 21:10:49 -07:00
return phy_mii_ioctl ( lp - > phydev , rq , cmd ) ;
2007-11-12 21:31:11 -08:00
}
static int r6040_rx ( struct net_device * dev , int limit )
{
struct r6040_private * priv = netdev_priv ( dev ) ;
2008-07-13 14:33:36 +02:00
struct r6040_descriptor * descptr = priv - > rx_remove_ptr ;
struct sk_buff * skb_ptr , * new_skb ;
int count = 0 ;
2007-11-12 21:31:11 -08:00
u16 err ;
2008-07-13 14:33:36 +02:00
/* Limit not reached and the descriptor belongs to the CPU */
2008-07-13 14:34:15 +02:00
while ( count < limit & & ! ( descptr - > status & DSC_OWNER_MAC ) ) {
2008-07-13 14:33:36 +02:00
/* Read the descriptor status */
err = descptr - > status ;
/* Global error status set */
2008-07-13 14:34:15 +02:00
if ( err & DSC_RX_ERR ) {
2008-07-13 14:33:36 +02:00
/* RX dribble */
2008-07-13 14:34:15 +02:00
if ( err & DSC_RX_ERR_DRI )
2008-07-13 14:33:36 +02:00
dev - > stats . rx_frame_errors + + ;
/* Buffer lenght exceeded */
2008-07-13 14:34:15 +02:00
if ( err & DSC_RX_ERR_BUF )
2008-07-13 14:33:36 +02:00
dev - > stats . rx_length_errors + + ;
/* Packet too long */
2008-07-13 14:34:15 +02:00
if ( err & DSC_RX_ERR_LONG )
2008-07-13 14:33:36 +02:00
dev - > stats . rx_length_errors + + ;
/* Packet < 64 bytes */
2008-07-13 14:34:15 +02:00
if ( err & DSC_RX_ERR_RUNT )
2008-07-13 14:33:36 +02:00
dev - > stats . rx_length_errors + + ;
/* CRC error */
2008-07-13 14:34:15 +02:00
if ( err & DSC_RX_ERR_CRC ) {
2008-07-13 14:33:36 +02:00
spin_lock ( & priv - > lock ) ;
dev - > stats . rx_crc_errors + + ;
spin_unlock ( & priv - > lock ) ;
2007-11-12 21:31:11 -08:00
}
2008-07-13 14:33:36 +02:00
goto next_descr ;
}
2010-08-08 10:08:44 +00:00
2008-07-13 14:33:36 +02:00
/* Packet successfully received */
new_skb = netdev_alloc_skb ( dev , MAX_BUF_SIZE ) ;
if ( ! new_skb ) {
dev - > stats . rx_dropped + + ;
goto next_descr ;
2007-11-12 21:31:11 -08:00
}
2008-07-13 14:33:36 +02:00
skb_ptr = descptr - > skb_ptr ;
skb_ptr - > dev = priv - > dev ;
2010-08-08 10:08:44 +00:00
2008-07-13 14:33:36 +02:00
/* Do not count the CRC */
skb_put ( skb_ptr , descptr - > len - 4 ) ;
pci_unmap_single ( priv - > pdev , le32_to_cpu ( descptr - > buf ) ,
MAX_BUF_SIZE , PCI_DMA_FROMDEVICE ) ;
skb_ptr - > protocol = eth_type_trans ( skb_ptr , priv - > dev ) ;
2010-08-08 10:08:44 +00:00
2008-07-13 14:33:36 +02:00
/* Send to upper layer */
netif_receive_skb ( skb_ptr ) ;
dev - > stats . rx_packets + + ;
dev - > stats . rx_bytes + = descptr - > len - 4 ;
/* put new skb into descriptor */
descptr - > skb_ptr = new_skb ;
descptr - > buf = cpu_to_le32 ( pci_map_single ( priv - > pdev ,
descptr - > skb_ptr - > data ,
MAX_BUF_SIZE , PCI_DMA_FROMDEVICE ) ) ;
next_descr :
/* put the descriptor back to the MAC */
2008-07-13 14:34:15 +02:00
descptr - > status = DSC_OWNER_MAC ;
2008-07-13 14:33:36 +02:00
descptr = descptr - > vndescp ;
count + + ;
2007-11-12 21:31:11 -08:00
}
2008-07-13 14:33:36 +02:00
priv - > rx_remove_ptr = descptr ;
2007-11-12 21:31:11 -08:00
return count ;
}
static void r6040_tx ( struct net_device * dev )
{
struct r6040_private * priv = netdev_priv ( dev ) ;
struct r6040_descriptor * descptr ;
void __iomem * ioaddr = priv - > base ;
struct sk_buff * skb_ptr ;
u16 err ;
spin_lock ( & priv - > lock ) ;
descptr = priv - > tx_remove_ptr ;
while ( priv - > tx_free_desc < TX_DCNT ) {
/* Check for errors */
err = ioread16 ( ioaddr + MLSR ) ;
2007-12-12 22:34:55 +01:00
if ( err & 0x0200 )
dev - > stats . rx_fifo_errors + + ;
if ( err & ( 0x2000 | 0x4000 ) )
dev - > stats . tx_carrier_errors + + ;
2007-11-12 21:31:11 -08:00
2008-07-13 14:34:15 +02:00
if ( descptr - > status & DSC_OWNER_MAC )
2007-12-12 23:13:15 +01:00
break ; /* Not complete */
2007-11-12 21:31:11 -08:00
skb_ptr = descptr - > skb_ptr ;
2008-03-16 22:43:06 +00:00
pci_unmap_single ( priv - > pdev , le32_to_cpu ( descptr - > buf ) ,
2007-11-12 21:31:11 -08:00
skb_ptr - > len , PCI_DMA_TODEVICE ) ;
/* Free buffer */
dev_kfree_skb_irq ( skb_ptr ) ;
descptr - > skb_ptr = NULL ;
/* To next descriptor */
descptr = descptr - > vndescp ;
priv - > tx_free_desc + + ;
}
priv - > tx_remove_ptr = descptr ;
if ( priv - > tx_free_desc )
netif_wake_queue ( dev ) ;
spin_unlock ( & priv - > lock ) ;
}
static int r6040_poll ( struct napi_struct * napi , int budget )
{
struct r6040_private * priv =
container_of ( napi , struct r6040_private , napi ) ;
struct net_device * dev = priv - > dev ;
void __iomem * ioaddr = priv - > base ;
int work_done ;
work_done = r6040_rx ( dev , budget ) ;
if ( work_done < budget ) {
2009-01-19 16:43:59 -08:00
napi_complete ( napi ) ;
2007-11-12 21:31:11 -08:00
/* Enable RX interrupt */
2008-07-13 14:35:32 +02:00
iowrite16 ( ioread16 ( ioaddr + MIER ) | RX_INTS , ioaddr + MIER ) ;
2007-11-12 21:31:11 -08:00
}
return work_done ;
}
/* The RDC interrupt handler. */
static irqreturn_t r6040_interrupt ( int irq , void * dev_id )
{
struct net_device * dev = dev_id ;
struct r6040_private * lp = netdev_priv ( dev ) ;
void __iomem * ioaddr = lp - > base ;
2008-12-22 19:40:02 -08:00
u16 misr , status ;
2007-11-12 21:31:11 -08:00
2008-12-22 19:40:02 -08:00
/* Save MIER */
misr = ioread16 ( ioaddr + MIER ) ;
2007-11-12 21:31:11 -08:00
/* Mask off RDC MAC interrupt */
iowrite16 ( MSK_INT , ioaddr + MIER ) ;
/* Read MISR status and clear */
status = ioread16 ( ioaddr + MISR ) ;
2009-07-08 03:05:14 +00:00
if ( status = = 0x0000 | | status = = 0xffff ) {
/* Restore RDC MAC interrupt */
iowrite16 ( misr , ioaddr + MIER ) ;
2007-11-12 21:31:11 -08:00
return IRQ_NONE ;
2009-07-08 03:05:14 +00:00
}
2007-11-12 21:31:11 -08:00
/* RX interrupt request */
2008-07-13 14:35:32 +02:00
if ( status & RX_INTS ) {
if ( status & RX_NO_DESC ) {
/* RX descriptor unavailable */
dev - > stats . rx_dropped + + ;
dev - > stats . rx_missed_errors + + ;
}
if ( status & RX_FIFO_FULL )
dev - > stats . rx_fifo_errors + + ;
2008-07-13 14:28:27 +02:00
/* Mask off RX interrupt */
2008-12-22 19:40:02 -08:00
misr & = ~ RX_INTS ;
2009-01-19 16:43:59 -08:00
napi_schedule ( & lp - > napi ) ;
2007-11-12 21:31:11 -08:00
}
/* TX interrupt request */
2008-07-13 14:35:32 +02:00
if ( status & TX_INTS )
2007-11-12 21:31:11 -08:00
r6040_tx ( dev ) ;
2008-12-22 19:40:02 -08:00
/* Restore RDC MAC interrupt */
iowrite16 ( misr , ioaddr + MIER ) ;
2007-12-12 23:13:15 +01:00
return IRQ_HANDLED ;
2007-11-12 21:31:11 -08:00
}
# ifdef CONFIG_NET_POLL_CONTROLLER
static void r6040_poll_controller ( struct net_device * dev )
{
disable_irq ( dev - > irq ) ;
2007-11-28 23:02:33 +01:00
r6040_interrupt ( dev - > irq , dev ) ;
2007-11-12 21:31:11 -08:00
enable_irq ( dev - > irq ) ;
}
# endif
/* Init RDC MAC */
2008-07-13 14:32:18 +02:00
static int r6040_up ( struct net_device * dev )
2007-11-12 21:31:11 -08:00
{
struct r6040_private * lp = netdev_priv ( dev ) ;
void __iomem * ioaddr = lp - > base ;
2008-07-13 14:32:18 +02:00
int ret ;
2007-11-12 21:31:11 -08:00
2007-12-12 22:55:34 +01:00
/* Initialise and alloc RX/TX buffers */
2008-07-13 14:32:18 +02:00
r6040_init_txbufs ( dev ) ;
ret = r6040_alloc_rxbufs ( dev ) ;
if ( ret )
return ret ;
2007-11-12 21:31:11 -08:00
/* improve performance (by RDC guys) */
2010-08-08 10:08:44 +00:00
r6040_phy_write ( ioaddr , 30 , 17 ,
( r6040_phy_read ( ioaddr , 30 , 17 ) | 0x4000 ) ) ;
r6040_phy_write ( ioaddr , 30 , 17 ,
~ ( ( ~ r6040_phy_read ( ioaddr , 30 , 17 ) ) | 0x2000 ) ) ;
2008-07-13 13:39:32 +02:00
r6040_phy_write ( ioaddr , 0 , 19 , 0x0000 ) ;
r6040_phy_write ( ioaddr , 0 , 30 , 0x01F0 ) ;
2007-11-12 21:31:11 -08:00
2008-07-13 14:29:20 +02:00
/* Initialize all MAC registers */
r6040_init_mac_regs ( dev ) ;
2008-07-13 14:32:18 +02:00
return 0 ;
2007-11-12 21:31:11 -08:00
}
/* Read/set MAC address routines */
static void r6040_mac_address ( struct net_device * dev )
{
struct r6040_private * lp = netdev_priv ( dev ) ;
void __iomem * ioaddr = lp - > base ;
u16 * adrp ;
/* MAC operation register */
iowrite16 ( 0x01 , ioaddr + MCR1 ) ; /* Reset MAC */
iowrite16 ( 2 , ioaddr + MAC_SM ) ; /* Reset internal state machine */
iowrite16 ( 0 , ioaddr + MAC_SM ) ;
2008-09-03 16:50:03 +02:00
mdelay ( 5 ) ;
2007-11-12 21:31:11 -08:00
/* Restore MAC Address */
adrp = ( u16 * ) dev - > dev_addr ;
iowrite16 ( adrp [ 0 ] , ioaddr + MID_0L ) ;
iowrite16 ( adrp [ 1 ] , ioaddr + MID_0M ) ;
iowrite16 ( adrp [ 2 ] , ioaddr + MID_0H ) ;
2010-09-26 19:58:07 -07:00
/* Store MAC Address in perm_addr */
memcpy ( dev - > perm_addr , dev - > dev_addr , ETH_ALEN ) ;
2007-11-12 21:31:11 -08:00
}
2007-11-28 23:02:33 +01:00
static int r6040_open ( struct net_device * dev )
2007-11-12 21:31:11 -08:00
{
2007-11-28 23:02:33 +01:00
struct r6040_private * lp = netdev_priv ( dev ) ;
2007-11-12 21:31:11 -08:00
int ret ;
/* Request IRQ and Register interrupt handler */
2009-11-18 08:23:00 +00:00
ret = request_irq ( dev - > irq , r6040_interrupt ,
2007-11-12 21:31:11 -08:00
IRQF_SHARED , dev - > name , dev ) ;
if ( ret )
2010-08-24 23:57:55 +00:00
goto out ;
2007-11-12 21:31:11 -08:00
/* Set MAC address */
r6040_mac_address ( dev ) ;
/* Allocate Descriptor memory */
2007-11-28 22:31:00 +01:00
lp - > rx_ring =
pci_alloc_consistent ( lp - > pdev , RX_DESC_SIZE , & lp - > rx_ring_dma ) ;
2010-08-24 23:57:55 +00:00
if ( ! lp - > rx_ring ) {
ret = - ENOMEM ;
goto err_free_irq ;
}
2007-11-12 21:31:11 -08:00
2007-11-28 22:31:00 +01:00
lp - > tx_ring =
pci_alloc_consistent ( lp - > pdev , TX_DESC_SIZE , & lp - > tx_ring_dma ) ;
if ( ! lp - > tx_ring ) {
2010-08-24 23:57:55 +00:00
ret = - ENOMEM ;
goto err_free_rx_ring ;
2007-11-28 22:31:00 +01:00
}
2008-07-13 14:32:18 +02:00
ret = r6040_up ( dev ) ;
2010-08-24 23:57:55 +00:00
if ( ret )
goto err_free_tx_ring ;
2007-11-12 21:31:11 -08:00
napi_enable ( & lp - > napi ) ;
netif_start_queue ( dev ) ;
return 0 ;
2010-08-24 23:57:55 +00:00
err_free_tx_ring :
pci_free_consistent ( lp - > pdev , TX_DESC_SIZE , lp - > tx_ring ,
lp - > tx_ring_dma ) ;
err_free_rx_ring :
pci_free_consistent ( lp - > pdev , RX_DESC_SIZE , lp - > rx_ring ,
lp - > rx_ring_dma ) ;
err_free_irq :
free_irq ( dev - > irq , dev ) ;
out :
return ret ;
2007-11-12 21:31:11 -08:00
}
2009-08-31 19:50:58 +00:00
static netdev_tx_t r6040_start_xmit ( struct sk_buff * skb ,
struct net_device * dev )
2007-11-12 21:31:11 -08:00
{
struct r6040_private * lp = netdev_priv ( dev ) ;
struct r6040_descriptor * descptr ;
void __iomem * ioaddr = lp - > base ;
unsigned long flags ;
/* Critical Section */
spin_lock_irqsave ( & lp - > lock , flags ) ;
/* TX resource check */
if ( ! lp - > tx_free_desc ) {
spin_unlock_irqrestore ( & lp - > lock , flags ) ;
2007-11-23 21:49:27 -05:00
netif_stop_queue ( dev ) ;
2010-04-07 21:39:27 +00:00
netdev_err ( dev , " : no tx descriptor \n " ) ;
2009-08-31 19:50:58 +00:00
return NETDEV_TX_BUSY ;
2007-11-12 21:31:11 -08:00
}
/* Statistic Counter */
dev - > stats . tx_packets + + ;
dev - > stats . tx_bytes + = skb - > len ;
/* Set TX descriptor & Transmit it */
lp - > tx_free_desc - - ;
descptr = lp - > tx_insert_ptr ;
if ( skb - > len < MISR )
descptr - > len = MISR ;
else
descptr - > len = skb - > len ;
descptr - > skb_ptr = skb ;
descptr - > buf = cpu_to_le32 ( pci_map_single ( lp - > pdev ,
skb - > data , skb - > len , PCI_DMA_TODEVICE ) ) ;
2008-07-13 14:34:15 +02:00
descptr - > status = DSC_OWNER_MAC ;
2007-11-12 21:31:11 -08:00
/* Trigger the MAC to check the TX descriptor */
iowrite16 ( 0x01 , ioaddr + MTPR ) ;
lp - > tx_insert_ptr = descptr - > vndescp ;
/* If no tx resource, stop */
if ( ! lp - > tx_free_desc )
netif_stop_queue ( dev ) ;
spin_unlock_irqrestore ( & lp - > lock , flags ) ;
2009-08-31 19:50:58 +00:00
return NETDEV_TX_OK ;
2007-11-12 21:31:11 -08:00
}
2007-11-28 23:02:33 +01:00
static void r6040_multicast_list ( struct net_device * dev )
2007-11-12 21:31:11 -08:00
{
struct r6040_private * lp = netdev_priv ( dev ) ;
void __iomem * ioaddr = lp - > base ;
u16 * adrp ;
u16 reg ;
unsigned long flags ;
2010-04-01 21:22:57 +00:00
struct netdev_hw_addr * ha ;
2007-11-12 21:31:11 -08:00
int i ;
/* MAC Address */
adrp = ( u16 * ) dev - > dev_addr ;
iowrite16 ( adrp [ 0 ] , ioaddr + MID_0L ) ;
iowrite16 ( adrp [ 1 ] , ioaddr + MID_0M ) ;
iowrite16 ( adrp [ 2 ] , ioaddr + MID_0H ) ;
/* Promiscous Mode */
spin_lock_irqsave ( & lp - > lock , flags ) ;
/* Clear AMCP & PROM bits */
reg = ioread16 ( ioaddr ) & ~ 0x0120 ;
if ( dev - > flags & IFF_PROMISC ) {
reg | = 0x0020 ;
lp - > mcr0 | = 0x0020 ;
}
/* Too many multicast addresses
* accept all traffic */
2010-02-08 04:30:35 +00:00
else if ( ( netdev_mc_count ( dev ) > MCAST_MAX ) | |
( dev - > flags & IFF_ALLMULTI ) )
2007-11-12 21:31:11 -08:00
reg | = 0x0020 ;
iowrite16 ( reg , ioaddr ) ;
spin_unlock_irqrestore ( & lp - > lock , flags ) ;
/* Build the hash table */
2010-02-08 04:30:35 +00:00
if ( netdev_mc_count ( dev ) > MCAST_MAX ) {
2007-11-12 21:31:11 -08:00
u16 hash_table [ 4 ] ;
u32 crc ;
for ( i = 0 ; i < 4 ; i + + )
hash_table [ i ] = 0 ;
2010-04-01 21:22:57 +00:00
netdev_for_each_mc_addr ( ha , dev ) {
char * addrs = ha - > addr ;
2007-11-12 21:31:11 -08:00
if ( ! ( * addrs & 1 ) )
continue ;
crc = ether_crc_le ( 6 , addrs ) ;
crc > > = 26 ;
hash_table [ crc > > 4 ] | = 1 < < ( 15 - ( crc & 0xf ) ) ;
}
/* Fill the MAC hash tables with their values */
iowrite16 ( hash_table [ 0 ] , ioaddr + MAR0 ) ;
iowrite16 ( hash_table [ 1 ] , ioaddr + MAR1 ) ;
iowrite16 ( hash_table [ 2 ] , ioaddr + MAR2 ) ;
iowrite16 ( hash_table [ 3 ] , ioaddr + MAR3 ) ;
}
/* Multicast Address 1~4 case */
2010-02-23 09:19:49 +00:00
i = 0 ;
2010-04-01 21:22:57 +00:00
netdev_for_each_mc_addr ( ha , dev ) {
r6040: Fix multicast filter some more
This code has been broken forever, but in several different and
creative ways.
So far as I can work out, the R6040 MAC filter has 4 exact-match
entries, the first of which the driver uses for its assigned unicast
address, plus a 64-entry hash-based filter for multicast addresses
(maybe unicast as well?).
The original version of this code would write the first 4 multicast
addresses as exact-match entries from offset 1 (bug #1: there is no
entry 4 so this could write to some PHY registers). It would fill the
remainder of the exact-match entries with the broadcast address (bug #2:
this would overwrite the last used entry). If more than 4 multicast
addresses were configured, it would set up the hash table, write some
random crap to the MAC control register (bug #3) and finally walk off
the end of the list when filling the exact-match entries (bug #4).
All of this seems to be pointless, since it sets the promiscuous bit
when the interface is made promiscuous or if >4 multicast addresses
are enabled, and never clears it (bug #5, masking bug #2).
The recent(ish) changes to the multicast list fixed bug #4, but
completely removed the limit on iteration over the exact-match entries
(bug #6).
Bug #4 was reported as
<https://bugzilla.kernel.org/show_bug.cgi?id=15355> and more recently
as <http://bugs.debian.org/600155>. Florian Fainelli attempted to fix
these in commit 3bcf8229a8c49769e48d3e0bd1e20d8e003f8106, but that
actually dealt with bugs #1-3, bug #4 having been fixed in mainline at
that point.
That commit fixes the most important current bug #6.
Signed-off-by: Ben Hutchings <ben@decadent.org.uk>
Cc: stable@kernel.org [2.6.35 only]
Signed-off-by: David S. Miller <davem@davemloft.net>
2010-10-14 17:41:53 +00:00
if ( i > = MCAST_MAX )
break ;
adrp = ( u16 * ) ha - > addr ;
iowrite16 ( adrp [ 0 ] , ioaddr + MID_1L + 8 * i ) ;
iowrite16 ( adrp [ 1 ] , ioaddr + MID_1M + 8 * i ) ;
iowrite16 ( adrp [ 2 ] , ioaddr + MID_1H + 8 * i ) ;
i + + ;
}
while ( i < MCAST_MAX ) {
iowrite16 ( 0xffff , ioaddr + MID_1L + 8 * i ) ;
iowrite16 ( 0xffff , ioaddr + MID_1M + 8 * i ) ;
iowrite16 ( 0xffff , ioaddr + MID_1H + 8 * i ) ;
2010-02-23 09:19:49 +00:00
i + + ;
2007-11-12 21:31:11 -08:00
}
}
static void netdev_get_drvinfo ( struct net_device * dev ,
struct ethtool_drvinfo * info )
{
struct r6040_private * rp = netdev_priv ( dev ) ;
strcpy ( info - > driver , DRV_NAME ) ;
strcpy ( info - > version , DRV_VERSION ) ;
strcpy ( info - > bus_info , pci_name ( rp - > pdev ) ) ;
}
static int netdev_get_settings ( struct net_device * dev , struct ethtool_cmd * cmd )
{
struct r6040_private * rp = netdev_priv ( dev ) ;
2010-05-31 09:18:57 +00:00
return phy_ethtool_gset ( rp - > phydev , cmd ) ;
2007-11-12 21:31:11 -08:00
}
static int netdev_set_settings ( struct net_device * dev , struct ethtool_cmd * cmd )
{
struct r6040_private * rp = netdev_priv ( dev ) ;
2010-05-31 09:18:57 +00:00
return phy_ethtool_sset ( rp - > phydev , cmd ) ;
2007-11-12 21:31:11 -08:00
}
2008-11-21 17:34:56 -08:00
static const struct ethtool_ops netdev_ethtool_ops = {
2007-11-12 21:31:11 -08:00
. get_drvinfo = netdev_get_drvinfo ,
. get_settings = netdev_get_settings ,
. set_settings = netdev_set_settings ,
2010-05-31 09:18:57 +00:00
. get_link = ethtool_op_get_link ,
2007-11-12 21:31:11 -08:00
} ;
2008-11-21 17:34:56 -08:00
static const struct net_device_ops r6040_netdev_ops = {
. ndo_open = r6040_open ,
. ndo_stop = r6040_close ,
. ndo_start_xmit = r6040_start_xmit ,
. ndo_get_stats = r6040_get_stats ,
. ndo_set_multicast_list = r6040_multicast_list ,
. ndo_change_mtu = eth_change_mtu ,
. ndo_validate_addr = eth_validate_addr ,
2010-08-08 10:08:44 +00:00
. ndo_set_mac_address = eth_mac_addr ,
2008-11-21 17:34:56 -08:00
. ndo_do_ioctl = r6040_ioctl ,
. ndo_tx_timeout = r6040_tx_timeout ,
# ifdef CONFIG_NET_POLL_CONTROLLER
. ndo_poll_controller = r6040_poll_controller ,
# endif
} ;
2010-05-31 09:18:57 +00:00
static void r6040_adjust_link ( struct net_device * dev )
{
struct r6040_private * lp = netdev_priv ( dev ) ;
struct phy_device * phydev = lp - > phydev ;
int status_changed = 0 ;
void __iomem * ioaddr = lp - > base ;
BUG_ON ( ! phydev ) ;
if ( lp - > old_link ! = phydev - > link ) {
status_changed = 1 ;
lp - > old_link = phydev - > link ;
}
/* reflect duplex change */
if ( phydev - > link & & ( lp - > old_duplex ! = phydev - > duplex ) ) {
lp - > mcr0 | = ( phydev - > duplex = = DUPLEX_FULL ? 0x8000 : 0 ) ;
iowrite16 ( lp - > mcr0 , ioaddr ) ;
status_changed = 1 ;
lp - > old_duplex = phydev - > duplex ;
}
if ( status_changed ) {
pr_info ( " %s: link %s " , dev - > name , phydev - > link ?
" UP " : " DOWN " ) ;
if ( phydev - > link )
pr_cont ( " - %d/%s " , phydev - > speed ,
DUPLEX_FULL = = phydev - > duplex ? " full " : " half " ) ;
pr_cont ( " \n " ) ;
}
}
static int r6040_mii_probe ( struct net_device * dev )
{
struct r6040_private * lp = netdev_priv ( dev ) ;
struct phy_device * phydev = NULL ;
phydev = phy_find_first ( lp - > mii_bus ) ;
if ( ! phydev ) {
dev_err ( & lp - > pdev - > dev , " no PHY found \n " ) ;
return - ENODEV ;
}
phydev = phy_connect ( dev , dev_name ( & phydev - > dev ) , & r6040_adjust_link ,
0 , PHY_INTERFACE_MODE_MII ) ;
if ( IS_ERR ( phydev ) ) {
dev_err ( & lp - > pdev - > dev , " could not attach to PHY \n " ) ;
return PTR_ERR ( phydev ) ;
}
/* mask with MAC supported features */
phydev - > supported & = ( SUPPORTED_10baseT_Half
| SUPPORTED_10baseT_Full
| SUPPORTED_100baseT_Half
| SUPPORTED_100baseT_Full
| SUPPORTED_Autoneg
| SUPPORTED_MII
| SUPPORTED_TP ) ;
phydev - > advertising = phydev - > supported ;
lp - > phydev = phydev ;
lp - > old_link = 0 ;
lp - > old_duplex = - 1 ;
dev_info ( & lp - > pdev - > dev , " attached PHY driver [%s] "
" (mii_bus:phy_addr=%s) \n " ,
phydev - > drv - > name , dev_name ( & phydev - > dev ) ) ;
return 0 ;
}
2007-11-12 21:31:11 -08:00
static int __devinit r6040_init_one ( struct pci_dev * pdev ,
const struct pci_device_id * ent )
{
struct net_device * dev ;
struct r6040_private * lp ;
void __iomem * ioaddr ;
int err , io_size = R6040_IO_SIZE ;
static int card_idx = - 1 ;
int bar = 0 ;
u16 * adrp ;
2010-05-31 09:18:57 +00:00
int i ;
2007-11-12 21:31:11 -08:00
2010-08-08 10:08:44 +00:00
pr_info ( " %s \n " , version ) ;
2007-11-12 21:31:11 -08:00
err = pci_enable_device ( pdev ) ;
if ( err )
2008-07-21 12:32:29 +02:00
goto err_out ;
2007-11-12 21:31:11 -08:00
/* this should always be supported */
2009-04-06 19:01:15 -07:00
err = pci_set_dma_mask ( pdev , DMA_BIT_MASK ( 32 ) ) ;
2008-07-21 12:32:29 +02:00
if ( err ) {
2010-04-07 21:39:27 +00:00
dev_err ( & pdev - > dev , " 32-bit PCI DMA addresses "
2007-11-12 21:31:11 -08:00
" not supported by the card \n " ) ;
2008-07-21 12:32:29 +02:00
goto err_out ;
2007-11-12 21:31:11 -08:00
}
2009-04-06 19:01:15 -07:00
err = pci_set_consistent_dma_mask ( pdev , DMA_BIT_MASK ( 32 ) ) ;
2008-07-21 12:32:29 +02:00
if ( err ) {
2010-04-07 21:39:27 +00:00
dev_err ( & pdev - > dev , " 32-bit PCI DMA addresses "
2007-11-23 21:49:27 -05:00
" not supported by the card \n " ) ;
2008-07-21 12:32:29 +02:00
goto err_out ;
2007-11-23 21:49:27 -05:00
}
2007-11-12 21:31:11 -08:00
/* IO Size check */
2009-06-24 21:05:09 +00:00
if ( pci_resource_len ( pdev , bar ) < io_size ) {
2010-04-07 21:39:27 +00:00
dev_err ( & pdev - > dev , " Insufficient PCI resources, aborting \n " ) ;
2008-07-21 12:32:29 +02:00
err = - EIO ;
goto err_out ;
2007-11-12 21:31:11 -08:00
}
pci_set_master ( pdev ) ;
dev = alloc_etherdev ( sizeof ( struct r6040_private ) ) ;
if ( ! dev ) {
2010-04-07 21:39:27 +00:00
dev_err ( & pdev - > dev , " Failed to allocate etherdev \n " ) ;
2008-07-21 12:32:29 +02:00
err = - ENOMEM ;
goto err_out ;
2007-11-12 21:31:11 -08:00
}
SET_NETDEV_DEV ( dev , & pdev - > dev ) ;
lp = netdev_priv ( dev ) ;
2008-07-21 12:32:29 +02:00
err = pci_request_regions ( pdev , DRV_NAME ) ;
if ( err ) {
2010-04-07 21:39:27 +00:00
dev_err ( & pdev - > dev , " Failed to request PCI regions \n " ) ;
2008-07-21 12:32:29 +02:00
goto err_out_free_dev ;
2007-11-12 21:31:11 -08:00
}
ioaddr = pci_iomap ( pdev , bar , io_size ) ;
if ( ! ioaddr ) {
2010-04-07 21:39:27 +00:00
dev_err ( & pdev - > dev , " ioremap failed for device \n " ) ;
2008-07-21 12:32:29 +02:00
err = - EIO ;
goto err_out_free_res ;
2007-11-12 21:31:11 -08:00
}
2009-01-08 11:01:58 -08:00
/* If PHY status change register is still set to zero it means the
* bootloader didn ' t initialize it */
if ( ioread16 ( ioaddr + PHY_CC ) = = 0 )
iowrite16 ( 0x9f07 , ioaddr + PHY_CC ) ;
2007-11-12 21:31:11 -08:00
/* Init system & device */
lp - > base = ioaddr ;
dev - > irq = pdev - > irq ;
spin_lock_init ( & lp - > lock ) ;
pci_set_drvdata ( pdev , dev ) ;
/* Set MAC address */
card_idx + + ;
adrp = ( u16 * ) dev - > dev_addr ;
adrp [ 0 ] = ioread16 ( ioaddr + MID_0L ) ;
adrp [ 1 ] = ioread16 ( ioaddr + MID_0M ) ;
adrp [ 2 ] = ioread16 ( ioaddr + MID_0H ) ;
2009-01-08 11:02:30 -08:00
/* Some bootloader/BIOSes do not initialize
* MAC address , warn about that */
2009-01-08 15:04:45 +00:00
if ( ! ( adrp [ 0 ] | | adrp [ 1 ] | | adrp [ 2 ] ) ) {
2010-08-08 10:08:44 +00:00
netdev_warn ( dev , " MAC address not initialized, "
" generating random \n " ) ;
2009-01-08 15:04:45 +00:00
random_ether_addr ( dev - > dev_addr ) ;
}
2009-01-08 11:02:30 -08:00
2007-11-12 21:31:11 -08:00
/* Link new device into r6040_root_dev */
lp - > pdev = pdev ;
2008-07-13 14:32:45 +02:00
lp - > dev = dev ;
2007-11-12 21:31:11 -08:00
/* Init RDC private data */
lp - > mcr0 = 0x1002 ;
lp - > phy_addr = phy_table [ card_idx ] ;
/* The RDC-specific entries in the device structure. */
2008-11-21 17:34:56 -08:00
dev - > netdev_ops = & r6040_netdev_ops ;
2007-11-12 21:31:11 -08:00
dev - > ethtool_ops = & netdev_ethtool_ops ;
dev - > watchdog_timeo = TX_TIMEOUT ;
2008-11-21 17:34:56 -08:00
2007-11-12 21:31:11 -08:00
netif_napi_add ( dev , & lp - > napi , r6040_poll , 64 ) ;
2010-05-31 09:18:57 +00:00
lp - > mii_bus = mdiobus_alloc ( ) ;
if ( ! lp - > mii_bus ) {
dev_err ( & pdev - > dev , " mdiobus_alloc() failed \n " ) ;
2009-08-20 01:26:20 +00:00
goto err_out_unmap ;
}
2010-05-31 09:18:57 +00:00
lp - > mii_bus - > priv = dev ;
lp - > mii_bus - > read = r6040_mdiobus_read ;
lp - > mii_bus - > write = r6040_mdiobus_write ;
lp - > mii_bus - > reset = r6040_mdiobus_reset ;
lp - > mii_bus - > name = " r6040_eth_mii " ;
snprintf ( lp - > mii_bus - > id , MII_BUS_ID_SIZE , " %x " , card_idx ) ;
lp - > mii_bus - > irq = kmalloc ( sizeof ( int ) * PHY_MAX_ADDR , GFP_KERNEL ) ;
if ( ! lp - > mii_bus - > irq ) {
dev_err ( & pdev - > dev , " mii_bus irq allocation failed \n " ) ;
goto err_out_mdio ;
}
for ( i = 0 ; i < PHY_MAX_ADDR ; i + + )
lp - > mii_bus - > irq [ i ] = PHY_POLL ;
err = mdiobus_register ( lp - > mii_bus ) ;
if ( err ) {
dev_err ( & pdev - > dev , " failed to register MII bus \n " ) ;
goto err_out_mdio_irq ;
}
err = r6040_mii_probe ( dev ) ;
if ( err ) {
dev_err ( & pdev - > dev , " failed to probe MII bus \n " ) ;
goto err_out_mdio_unregister ;
}
2007-11-12 21:31:11 -08:00
/* Register net device. After this dev->name assign */
err = register_netdev ( dev ) ;
if ( err ) {
2010-04-07 21:39:27 +00:00
dev_err ( & pdev - > dev , " Failed to register net device \n " ) ;
2010-05-31 09:18:57 +00:00
goto err_out_mdio_unregister ;
2007-11-12 21:31:11 -08:00
}
return 0 ;
2010-05-31 09:18:57 +00:00
err_out_mdio_unregister :
mdiobus_unregister ( lp - > mii_bus ) ;
err_out_mdio_irq :
kfree ( lp - > mii_bus - > irq ) ;
err_out_mdio :
mdiobus_free ( lp - > mii_bus ) ;
2008-07-21 12:32:29 +02:00
err_out_unmap :
pci_iounmap ( pdev , ioaddr ) ;
err_out_free_res :
2007-11-12 21:31:11 -08:00
pci_release_regions ( pdev ) ;
2008-07-21 12:32:29 +02:00
err_out_free_dev :
2007-11-12 21:31:11 -08:00
free_netdev ( dev ) ;
2008-07-21 12:32:29 +02:00
err_out :
2007-11-12 21:31:11 -08:00
return err ;
}
static void __devexit r6040_remove_one ( struct pci_dev * pdev )
{
struct net_device * dev = pci_get_drvdata ( pdev ) ;
2010-05-31 09:18:57 +00:00
struct r6040_private * lp = netdev_priv ( dev ) ;
2007-11-12 21:31:11 -08:00
unregister_netdev ( dev ) ;
2010-05-31 09:18:57 +00:00
mdiobus_unregister ( lp - > mii_bus ) ;
kfree ( lp - > mii_bus - > irq ) ;
mdiobus_free ( lp - > mii_bus ) ;
2007-11-12 21:31:11 -08:00
pci_release_regions ( pdev ) ;
free_netdev ( dev ) ;
pci_disable_device ( pdev ) ;
pci_set_drvdata ( pdev , NULL ) ;
}
2010-01-07 11:58:11 +00:00
static DEFINE_PCI_DEVICE_TABLE ( r6040_pci_tbl ) = {
2007-11-28 23:02:33 +01:00
{ PCI_DEVICE ( PCI_VENDOR_ID_RDC , 0x6040 ) } ,
{ 0 }
2007-11-12 21:31:11 -08:00
} ;
MODULE_DEVICE_TABLE ( pci , r6040_pci_tbl ) ;
static struct pci_driver r6040_driver = {
2007-11-28 23:02:33 +01:00
. name = DRV_NAME ,
2007-11-12 21:31:11 -08:00
. id_table = r6040_pci_tbl ,
. probe = r6040_init_one ,
. remove = __devexit_p ( r6040_remove_one ) ,
} ;
static int __init r6040_init ( void )
{
return pci_register_driver ( & r6040_driver ) ;
}
static void __exit r6040_cleanup ( void )
{
pci_unregister_driver ( & r6040_driver ) ;
}
module_init ( r6040_init ) ;
module_exit ( r6040_cleanup ) ;