2007-07-17 10:43:44 +04:00
/*
2008-01-30 11:52:24 +03:00
* Blackfin On - Chip MAC Driver
2007-07-17 10:43:44 +04:00
*
2008-01-30 11:52:24 +03:00
* Copyright 2004 - 2007 Analog Devices Inc .
2007-07-17 10:43:44 +04:00
*
2008-01-30 11:52:24 +03:00
* Enter bugs at http : //blackfin.uclinux.org/
2007-07-17 10:43:44 +04:00
*
2008-01-30 11:52:24 +03:00
* Licensed under the GPL - 2 or later .
2007-07-17 10:43:44 +04:00
*/
# include <linux/init.h>
# include <linux/module.h>
# include <linux/kernel.h>
# include <linux/sched.h>
# include <linux/slab.h>
# include <linux/delay.h>
# include <linux/timer.h>
# include <linux/errno.h>
# include <linux/irq.h>
# include <linux/io.h>
# include <linux/ioport.h>
# include <linux/crc32.h>
# include <linux/device.h>
# include <linux/spinlock.h>
# include <linux/mii.h>
2007-09-19 19:37:36 +04:00
# include <linux/phy.h>
2007-07-17 10:43:44 +04:00
# include <linux/netdevice.h>
# include <linux/etherdevice.h>
2008-04-25 07:53:11 +04:00
# include <linux/ethtool.h>
2007-07-17 10:43:44 +04:00
# include <linux/skbuff.h>
# include <linux/platform_device.h>
# include <asm/dma.h>
# include <linux/dma-mapping.h>
# include <asm/blackfin.h>
# include <asm/cacheflush.h>
# include <asm/portmux.h>
# include "bfin_mac.h"
# define DRV_NAME "bfin_mac"
# define DRV_VERSION "1.1"
# define DRV_AUTHOR "Bryan Wu, Luke Yang"
2008-04-25 07:53:10 +04:00
# define DRV_DESC "Blackfin on-chip Ethernet MAC driver"
2007-07-17 10:43:44 +04:00
MODULE_AUTHOR ( DRV_AUTHOR ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_DESCRIPTION ( DRV_DESC ) ;
2008-04-19 00:50:44 +04:00
MODULE_ALIAS ( " platform:bfin_mac " ) ;
2007-07-17 10:43:44 +04:00
# if defined(CONFIG_BFIN_MAC_USE_L1)
# define bfin_mac_alloc(dma_handle, size) l1_data_sram_zalloc(size)
# define bfin_mac_free(dma_handle, ptr) l1_data_sram_free(ptr)
# else
# define bfin_mac_alloc(dma_handle, size) \
dma_alloc_coherent ( NULL , size , dma_handle , GFP_KERNEL )
# define bfin_mac_free(dma_handle, ptr) \
dma_free_coherent ( NULL , sizeof ( * ptr ) , ptr , dma_handle )
# endif
# define PKT_BUF_SZ 1580
# define MAX_TIMEOUT_CNT 500
/* pointers to maintain transmit list */
static struct net_dma_desc_tx * tx_list_head ;
static struct net_dma_desc_tx * tx_list_tail ;
static struct net_dma_desc_rx * rx_list_head ;
static struct net_dma_desc_rx * rx_list_tail ;
static struct net_dma_desc_rx * current_rx_ptr ;
static struct net_dma_desc_tx * current_tx_ptr ;
static struct net_dma_desc_tx * tx_desc ;
static struct net_dma_desc_rx * rx_desc ;
2008-04-25 07:53:10 +04:00
# if defined(CONFIG_BFIN_MAC_RMII)
static u16 pin_req [ ] = P_RMII0 ;
# else
static u16 pin_req [ ] = P_MII0 ;
# endif
static void bfin_mac_disable ( void ) ;
static void bfin_mac_enable ( void ) ;
2007-09-19 19:37:36 +04:00
2007-07-17 10:43:44 +04:00
static void desc_list_free ( void )
{
struct net_dma_desc_rx * r ;
struct net_dma_desc_tx * t ;
int i ;
# if !defined(CONFIG_BFIN_MAC_USE_L1)
dma_addr_t dma_handle = 0 ;
# endif
if ( tx_desc ) {
t = tx_list_head ;
for ( i = 0 ; i < CONFIG_BFIN_TX_DESC_NUM ; i + + ) {
if ( t ) {
if ( t - > skb ) {
dev_kfree_skb ( t - > skb ) ;
t - > skb = NULL ;
}
t = t - > next ;
}
}
bfin_mac_free ( dma_handle , tx_desc ) ;
}
if ( rx_desc ) {
r = rx_list_head ;
for ( i = 0 ; i < CONFIG_BFIN_RX_DESC_NUM ; i + + ) {
if ( r ) {
if ( r - > skb ) {
dev_kfree_skb ( r - > skb ) ;
r - > skb = NULL ;
}
r = r - > next ;
}
}
bfin_mac_free ( dma_handle , rx_desc ) ;
}
}
static int desc_list_init ( void )
{
int i ;
struct sk_buff * new_skb ;
# if !defined(CONFIG_BFIN_MAC_USE_L1)
/*
* This dma_handle is useless in Blackfin dma_alloc_coherent ( ) .
* The real dma handler is the return value of dma_alloc_coherent ( ) .
*/
dma_addr_t dma_handle ;
# endif
tx_desc = bfin_mac_alloc ( & dma_handle ,
sizeof ( struct net_dma_desc_tx ) *
CONFIG_BFIN_TX_DESC_NUM ) ;
if ( tx_desc = = NULL )
goto init_error ;
rx_desc = bfin_mac_alloc ( & dma_handle ,
sizeof ( struct net_dma_desc_rx ) *
CONFIG_BFIN_RX_DESC_NUM ) ;
if ( rx_desc = = NULL )
goto init_error ;
/* init tx_list */
tx_list_head = tx_list_tail = tx_desc ;
for ( i = 0 ; i < CONFIG_BFIN_TX_DESC_NUM ; i + + ) {
struct net_dma_desc_tx * t = tx_desc + i ;
struct dma_descriptor * a = & ( t - > desc_a ) ;
struct dma_descriptor * b = & ( t - > desc_b ) ;
/*
* disable DMA
* read from memory WNR = 0
* wordsize is 32 bits
* 6 half words is desc size
* large desc flow
*/
a - > config = WDSIZE_32 | NDSIZE_6 | DMAFLOW_LARGE ;
a - > start_addr = ( unsigned long ) t - > packet ;
a - > x_count = 0 ;
a - > next_dma_desc = b ;
/*
* enabled DMA
* write to memory WNR = 1
* wordsize is 32 bits
* disable interrupt
* 6 half words is desc size
* large desc flow
*/
b - > config = DMAEN | WNR | WDSIZE_32 | NDSIZE_6 | DMAFLOW_LARGE ;
b - > start_addr = ( unsigned long ) ( & ( t - > status ) ) ;
b - > x_count = 0 ;
t - > skb = NULL ;
tx_list_tail - > desc_b . next_dma_desc = a ;
tx_list_tail - > next = t ;
tx_list_tail = t ;
}
tx_list_tail - > next = tx_list_head ; /* tx_list is a circle */
tx_list_tail - > desc_b . next_dma_desc = & ( tx_list_head - > desc_a ) ;
current_tx_ptr = tx_list_head ;
/* init rx_list */
rx_list_head = rx_list_tail = rx_desc ;
for ( i = 0 ; i < CONFIG_BFIN_RX_DESC_NUM ; i + + ) {
struct net_dma_desc_rx * r = rx_desc + i ;
struct dma_descriptor * a = & ( r - > desc_a ) ;
struct dma_descriptor * b = & ( r - > desc_b ) ;
/* allocate a new skb for next time receive */
new_skb = dev_alloc_skb ( PKT_BUF_SZ + 2 ) ;
if ( ! new_skb ) {
printk ( KERN_NOTICE DRV_NAME
" : init: low on mem - packet dropped \n " ) ;
goto init_error ;
}
skb_reserve ( new_skb , 2 ) ;
r - > skb = new_skb ;
/*
* enabled DMA
* write to memory WNR = 1
* wordsize is 32 bits
* disable interrupt
* 6 half words is desc size
* large desc flow
*/
a - > config = DMAEN | WNR | WDSIZE_32 | NDSIZE_6 | DMAFLOW_LARGE ;
/* since RXDWA is enabled */
a - > start_addr = ( unsigned long ) new_skb - > data - 2 ;
a - > x_count = 0 ;
a - > next_dma_desc = b ;
/*
* enabled DMA
* write to memory WNR = 1
* wordsize is 32 bits
* enable interrupt
* 6 half words is desc size
* large desc flow
*/
b - > config = DMAEN | WNR | WDSIZE_32 | DI_EN |
NDSIZE_6 | DMAFLOW_LARGE ;
b - > start_addr = ( unsigned long ) ( & ( r - > status ) ) ;
b - > x_count = 0 ;
rx_list_tail - > desc_b . next_dma_desc = a ;
rx_list_tail - > next = r ;
rx_list_tail = r ;
}
rx_list_tail - > next = rx_list_head ; /* rx_list is a circle */
rx_list_tail - > desc_b . next_dma_desc = & ( rx_list_head - > desc_a ) ;
current_rx_ptr = rx_list_head ;
return 0 ;
init_error :
desc_list_free ( ) ;
printk ( KERN_ERR DRV_NAME " : kmalloc failed \n " ) ;
return - ENOMEM ;
}
/*---PHY CONTROL AND CONFIGURATION-----------------------------------------*/
2007-09-19 19:37:36 +04:00
/*
* MII operations
*/
2007-07-17 10:43:44 +04:00
/* Wait until the previous MDC/MDIO transaction has completed */
2007-09-19 19:37:36 +04:00
static void mdio_poll ( void )
2007-07-17 10:43:44 +04:00
{
int timeout_cnt = MAX_TIMEOUT_CNT ;
/* poll the STABUSY bit */
while ( ( bfin_read_EMAC_STAADD ( ) ) & STABUSY ) {
2008-01-30 11:52:21 +03:00
udelay ( 1 ) ;
2007-07-17 10:43:44 +04:00
if ( timeout_cnt - - < 0 ) {
printk ( KERN_ERR DRV_NAME
" : wait MDC/MDIO transaction to complete timeout \n " ) ;
break ;
}
}
}
/* Read an off-chip register in a PHY through the MDC/MDIO port */
2007-09-19 19:37:36 +04:00
static int mdiobus_read ( struct mii_bus * bus , int phy_addr , int regnum )
2007-07-17 10:43:44 +04:00
{
2007-09-19 19:37:36 +04:00
mdio_poll ( ) ;
2007-07-17 10:43:44 +04:00
/* read mode */
2007-09-19 19:37:36 +04:00
bfin_write_EMAC_STAADD ( SET_PHYAD ( ( u16 ) phy_addr ) |
SET_REGAD ( ( u16 ) regnum ) |
2007-07-17 10:43:44 +04:00
STABUSY ) ;
2007-09-19 19:37:36 +04:00
mdio_poll ( ) ;
return ( int ) bfin_read_EMAC_STADAT ( ) ;
2007-07-17 10:43:44 +04:00
}
/* Write an off-chip register in a PHY through the MDC/MDIO port */
2007-09-19 19:37:36 +04:00
static int mdiobus_write ( struct mii_bus * bus , int phy_addr , int regnum ,
u16 value )
2007-07-17 10:43:44 +04:00
{
2007-09-19 19:37:36 +04:00
mdio_poll ( ) ;
bfin_write_EMAC_STADAT ( ( u32 ) value ) ;
2007-07-17 10:43:44 +04:00
/* write mode */
2007-09-19 19:37:36 +04:00
bfin_write_EMAC_STAADD ( SET_PHYAD ( ( u16 ) phy_addr ) |
SET_REGAD ( ( u16 ) regnum ) |
2007-07-17 10:43:44 +04:00
STAOP |
STABUSY ) ;
2007-09-19 19:37:36 +04:00
mdio_poll ( ) ;
return 0 ;
2007-07-17 10:43:44 +04:00
}
2007-09-19 19:37:36 +04:00
static int mdiobus_reset ( struct mii_bus * bus )
2007-07-17 10:43:44 +04:00
{
2007-09-19 19:37:36 +04:00
return 0 ;
2007-07-17 10:43:44 +04:00
}
2008-04-25 07:53:10 +04:00
static void bfin_mac_adjust_link ( struct net_device * dev )
2007-07-17 10:43:44 +04:00
{
2008-04-25 07:53:10 +04:00
struct bfin_mac_local * lp = netdev_priv ( dev ) ;
2007-09-19 19:37:36 +04:00
struct phy_device * phydev = lp - > phydev ;
unsigned long flags ;
int new_state = 0 ;
spin_lock_irqsave ( & lp - > lock , flags ) ;
if ( phydev - > link ) {
/* Now we make sure that we can be in full duplex mode.
* If not , we operate in half - duplex mode . */
if ( phydev - > duplex ! = lp - > old_duplex ) {
u32 opmode = bfin_read_EMAC_OPMODE ( ) ;
new_state = 1 ;
if ( phydev - > duplex )
opmode | = FDMODE ;
else
opmode & = ~ ( FDMODE ) ;
bfin_write_EMAC_OPMODE ( opmode ) ;
lp - > old_duplex = phydev - > duplex ;
}
2007-07-17 10:43:44 +04:00
2007-09-19 19:37:36 +04:00
if ( phydev - > speed ! = lp - > old_speed ) {
# if defined(CONFIG_BFIN_MAC_RMII)
u32 opmode = bfin_read_EMAC_OPMODE ( ) ;
switch ( phydev - > speed ) {
case 10 :
opmode | = RMII_10 ;
break ;
case 100 :
opmode & = ~ ( RMII_10 ) ;
break ;
default :
printk ( KERN_WARNING
" %s: Ack! Speed (%d) is not 10/100! \n " ,
DRV_NAME , phydev - > speed ) ;
break ;
}
bfin_write_EMAC_OPMODE ( opmode ) ;
# endif
2007-07-17 10:43:44 +04:00
2007-09-19 19:37:36 +04:00
new_state = 1 ;
lp - > old_speed = phydev - > speed ;
}
2007-07-17 10:43:44 +04:00
2007-09-19 19:37:36 +04:00
if ( ! lp - > old_link ) {
new_state = 1 ;
lp - > old_link = 1 ;
}
} else if ( lp - > old_link ) {
new_state = 1 ;
lp - > old_link = 0 ;
lp - > old_speed = 0 ;
lp - > old_duplex = - 1 ;
2007-07-17 10:43:44 +04:00
}
2007-09-19 19:37:36 +04:00
if ( new_state ) {
u32 opmode = bfin_read_EMAC_OPMODE ( ) ;
phy_print_status ( phydev ) ;
pr_debug ( " EMAC_OPMODE = 0x%08x \n " , opmode ) ;
2007-07-17 10:43:44 +04:00
}
2007-09-19 19:37:36 +04:00
spin_unlock_irqrestore ( & lp - > lock , flags ) ;
2007-07-17 10:43:44 +04:00
}
2008-01-30 11:52:22 +03:00
/* MDC = 2.5 MHz */
# define MDC_CLK 2500000
2007-09-19 19:37:36 +04:00
static int mii_probe ( struct net_device * dev )
2007-07-17 10:43:44 +04:00
{
2008-04-25 07:53:10 +04:00
struct bfin_mac_local * lp = netdev_priv ( dev ) ;
2007-09-19 19:37:36 +04:00
struct phy_device * phydev = NULL ;
unsigned short sysctl ;
int i ;
2008-01-30 11:52:22 +03:00
u32 sclk , mdc_div ;
2007-07-17 10:43:44 +04:00
2007-09-19 19:37:36 +04:00
/* Enable PHY output early */
2007-07-17 10:43:44 +04:00
if ( ! ( bfin_read_VR_CTL ( ) & PHYCLKOE ) )
bfin_write_VR_CTL ( bfin_read_VR_CTL ( ) | PHYCLKOE ) ;
2008-01-30 11:52:22 +03:00
sclk = get_sclk ( ) ;
mdc_div = ( ( sclk / MDC_CLK ) / 2 ) - 1 ;
2007-09-19 19:37:36 +04:00
sysctl = bfin_read_EMAC_SYSCTL ( ) ;
2008-01-30 11:52:28 +03:00
sysctl = ( sysctl & ~ MDCDIV ) | SET_MDCDIV ( mdc_div ) ;
2007-07-17 10:43:44 +04:00
bfin_write_EMAC_SYSCTL ( sysctl ) ;
2007-09-19 19:37:36 +04:00
/* search for connect PHY device */
for ( i = 0 ; i < PHY_MAX_ADDR ; i + + ) {
struct phy_device * const tmp_phydev = lp - > mii_bus . phy_map [ i ] ;
2007-07-17 10:43:44 +04:00
2007-09-19 19:37:36 +04:00
if ( ! tmp_phydev )
continue ; /* no PHY here... */
2007-07-17 10:43:44 +04:00
2007-09-19 19:37:36 +04:00
phydev = tmp_phydev ;
break ; /* found it */
}
/* now we are supposed to have a proper phydev, to attach to... */
if ( ! phydev ) {
printk ( KERN_INFO " %s: Don't found any phy device at all \n " ,
dev - > name ) ;
return - ENODEV ;
2007-07-17 10:43:44 +04:00
}
# if defined(CONFIG_BFIN_MAC_RMII)
2008-04-25 07:53:10 +04:00
phydev = phy_connect ( dev , phydev - > dev . bus_id , & bfin_mac_adjust_link , 0 ,
2007-09-19 19:37:36 +04:00
PHY_INTERFACE_MODE_RMII ) ;
# else
2008-04-25 07:53:10 +04:00
phydev = phy_connect ( dev , phydev - > dev . bus_id , & bfin_mac_adjust_link , 0 ,
2007-09-19 19:37:36 +04:00
PHY_INTERFACE_MODE_MII ) ;
2007-07-17 10:43:44 +04:00
# endif
2007-09-19 19:37:36 +04:00
if ( IS_ERR ( phydev ) ) {
printk ( KERN_ERR " %s: Could not attach to PHY \n " , dev - > name ) ;
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_Pause | SUPPORTED_Asym_Pause
| SUPPORTED_MII
| SUPPORTED_TP ) ;
phydev - > advertising = phydev - > supported ;
lp - > old_link = 0 ;
lp - > old_speed = 0 ;
lp - > old_duplex = - 1 ;
lp - > phydev = phydev ;
printk ( KERN_INFO " %s: attached PHY driver [%s] "
2008-01-30 11:52:22 +03:00
" (mii_bus:phy_addr=%s, irq=%d, mdc_clk=%dHz(mdc_div=%d) "
" @sclk=%dMHz) \n " ,
DRV_NAME , phydev - > drv - > name , phydev - > dev . bus_id , phydev - > irq ,
MDC_CLK , mdc_div , sclk / 1000000 ) ;
2007-09-19 19:37:36 +04:00
return 0 ;
}
2008-04-25 07:53:11 +04:00
/*
* Ethtool support
*/
static int
bfin_mac_ethtool_getsettings ( struct net_device * dev , struct ethtool_cmd * cmd )
{
struct bfin_mac_local * lp = netdev_priv ( dev ) ;
if ( lp - > phydev )
return phy_ethtool_gset ( lp - > phydev , cmd ) ;
return - EINVAL ;
}
static int
bfin_mac_ethtool_setsettings ( struct net_device * dev , struct ethtool_cmd * cmd )
{
struct bfin_mac_local * lp = netdev_priv ( dev ) ;
if ( ! capable ( CAP_NET_ADMIN ) )
return - EPERM ;
if ( lp - > phydev )
return phy_ethtool_sset ( lp - > phydev , cmd ) ;
return - EINVAL ;
}
static void bfin_mac_ethtool_getdrvinfo ( struct net_device * dev ,
struct ethtool_drvinfo * info )
{
strcpy ( info - > driver , DRV_NAME ) ;
strcpy ( info - > version , DRV_VERSION ) ;
strcpy ( info - > fw_version , " N/A " ) ;
strcpy ( info - > bus_info , dev - > dev . bus_id ) ;
}
static struct ethtool_ops bfin_mac_ethtool_ops = {
. get_settings = bfin_mac_ethtool_getsettings ,
. set_settings = bfin_mac_ethtool_setsettings ,
. get_link = ethtool_op_get_link ,
. get_drvinfo = bfin_mac_ethtool_getdrvinfo ,
} ;
2007-09-19 19:37:36 +04:00
/**************************************************************************/
void setup_system_regs ( struct net_device * dev )
{
unsigned short sysctl ;
/*
* Odd word alignment for Receive Frame DMA word
* Configure checksum support and rcve frame word alignment
*/
sysctl = bfin_read_EMAC_SYSCTL ( ) ;
# if defined(BFIN_MAC_CSUM_OFFLOAD)
sysctl | = RXDWA | RXCKS ;
# else
sysctl | = RXDWA ;
# endif
bfin_write_EMAC_SYSCTL ( sysctl ) ;
2007-07-17 10:43:44 +04:00
bfin_write_EMAC_MMC_CTL ( RSTC | CROLL ) ;
/* Initialize the TX DMA channel registers */
bfin_write_DMA2_X_COUNT ( 0 ) ;
bfin_write_DMA2_X_MODIFY ( 4 ) ;
bfin_write_DMA2_Y_COUNT ( 0 ) ;
bfin_write_DMA2_Y_MODIFY ( 0 ) ;
/* Initialize the RX DMA channel registers */
bfin_write_DMA1_X_COUNT ( 0 ) ;
bfin_write_DMA1_X_MODIFY ( 4 ) ;
bfin_write_DMA1_Y_COUNT ( 0 ) ;
bfin_write_DMA1_Y_MODIFY ( 0 ) ;
}
2007-09-19 19:14:18 +04:00
static void setup_mac_addr ( u8 * mac_addr )
2007-07-17 10:43:44 +04:00
{
u32 addr_low = le32_to_cpu ( * ( __le32 * ) & mac_addr [ 0 ] ) ;
u16 addr_hi = le16_to_cpu ( * ( __le16 * ) & mac_addr [ 4 ] ) ;
/* this depends on a little-endian machine */
bfin_write_EMAC_ADDRLO ( addr_low ) ;
bfin_write_EMAC_ADDRHI ( addr_hi ) ;
}
2008-04-25 07:53:10 +04:00
static int bfin_mac_set_mac_address ( struct net_device * dev , void * p )
2007-09-19 19:14:18 +04:00
{
struct sockaddr * addr = p ;
if ( netif_running ( dev ) )
return - EBUSY ;
memcpy ( dev - > dev_addr , addr - > sa_data , dev - > addr_len ) ;
setup_mac_addr ( dev - > dev_addr ) ;
return 0 ;
}
2007-07-17 10:43:44 +04:00
static void adjust_tx_list ( void )
{
int timeout_cnt = MAX_TIMEOUT_CNT ;
if ( tx_list_head - > status . status_word ! = 0
& & current_tx_ptr ! = tx_list_head ) {
goto adjust_head ; /* released something, just return; */
}
/*
* if nothing released , check wait condition
* current ' s next can not be the head ,
* otherwise the dma will not stop as we want
*/
if ( current_tx_ptr - > next - > next = = tx_list_head ) {
while ( tx_list_head - > status . status_word = = 0 ) {
2008-01-30 11:52:21 +03:00
mdelay ( 1 ) ;
2007-07-17 10:43:44 +04:00
if ( tx_list_head - > status . status_word ! = 0
| | ! ( bfin_read_DMA2_IRQ_STATUS ( ) & 0x08 ) ) {
goto adjust_head ;
}
if ( timeout_cnt - - < 0 ) {
printk ( KERN_ERR DRV_NAME
" : wait for adjust tx list head timeout \n " ) ;
break ;
}
}
if ( tx_list_head - > status . status_word ! = 0 ) {
goto adjust_head ;
}
}
return ;
adjust_head :
do {
tx_list_head - > desc_a . config & = ~ DMAEN ;
tx_list_head - > status . status_word = 0 ;
if ( tx_list_head - > skb ) {
dev_kfree_skb ( tx_list_head - > skb ) ;
tx_list_head - > skb = NULL ;
} else {
printk ( KERN_ERR DRV_NAME
" : no sk_buff in a transmitted frame! \n " ) ;
}
tx_list_head = tx_list_head - > next ;
} while ( tx_list_head - > status . status_word ! = 0
& & current_tx_ptr ! = tx_list_head ) ;
return ;
}
2008-04-25 07:53:10 +04:00
static int bfin_mac_hard_start_xmit ( struct sk_buff * skb ,
2007-07-17 10:43:44 +04:00
struct net_device * dev )
{
2008-07-27 18:45:04 +04:00
u16 * data ;
2007-07-17 10:43:44 +04:00
current_tx_ptr - > skb = skb ;
2008-07-27 18:45:04 +04:00
if ( ANOMALY_05000285 ) {
/*
* TXDWA feature is not avaible to older revision < 0.3 silicon
* of BF537
*
* Only if data buffer is ODD WORD alignment , we do not
* need to memcpy
*/
u32 data_align = ( u32 ) ( skb - > data ) & 0x3 ;
if ( data_align = = 0x2 ) {
/* move skb->data to current_tx_ptr payload */
data = ( u16 * ) ( skb - > data ) - 1 ;
* data = ( u16 ) ( skb - > len ) ;
current_tx_ptr - > desc_a . start_addr = ( u32 ) data ;
/* this is important! */
blackfin_dcache_flush_range ( ( u32 ) data ,
( u32 ) ( ( u8 * ) data + skb - > len + 4 ) ) ;
} else {
* ( ( u16 * ) ( current_tx_ptr - > packet ) ) = ( u16 ) ( skb - > len ) ;
memcpy ( ( u8 * ) ( current_tx_ptr - > packet + 2 ) , skb - > data ,
skb - > len ) ;
current_tx_ptr - > desc_a . start_addr =
( u32 ) current_tx_ptr - > packet ;
if ( current_tx_ptr - > status . status_word ! = 0 )
current_tx_ptr - > status . status_word = 0 ;
blackfin_dcache_flush_range (
( u32 ) current_tx_ptr - > packet ,
( u32 ) ( current_tx_ptr - > packet + skb - > len + 2 ) ) ;
}
2007-07-17 10:43:44 +04:00
} else {
2008-07-27 18:45:04 +04:00
/*
* TXDWA feature is avaible to revision < 0.3 silicon of
* BF537 and always avaible to BF52x
*/
u32 data_align = ( u32 ) ( skb - > data ) & 0x3 ;
if ( data_align = = 0x0 ) {
u16 sysctl = bfin_read_EMAC_SYSCTL ( ) ;
sysctl | = TXDWA ;
bfin_write_EMAC_SYSCTL ( sysctl ) ;
/* move skb->data to current_tx_ptr payload */
data = ( u16 * ) ( skb - > data ) - 2 ;
* data = ( u16 ) ( skb - > len ) ;
current_tx_ptr - > desc_a . start_addr = ( u32 ) data ;
/* this is important! */
blackfin_dcache_flush_range (
( u32 ) data ,
( u32 ) ( ( u8 * ) data + skb - > len + 4 ) ) ;
} else if ( data_align = = 0x2 ) {
u16 sysctl = bfin_read_EMAC_SYSCTL ( ) ;
sysctl & = ~ TXDWA ;
bfin_write_EMAC_SYSCTL ( sysctl ) ;
/* move skb->data to current_tx_ptr payload */
data = ( u16 * ) ( skb - > data ) - 1 ;
* data = ( u16 ) ( skb - > len ) ;
current_tx_ptr - > desc_a . start_addr = ( u32 ) data ;
/* this is important! */
blackfin_dcache_flush_range (
( u32 ) data ,
( u32 ) ( ( u8 * ) data + skb - > len + 4 ) ) ;
} else {
u16 sysctl = bfin_read_EMAC_SYSCTL ( ) ;
sysctl & = ~ TXDWA ;
bfin_write_EMAC_SYSCTL ( sysctl ) ;
* ( ( u16 * ) ( current_tx_ptr - > packet ) ) = ( u16 ) ( skb - > len ) ;
memcpy ( ( u8 * ) ( current_tx_ptr - > packet + 2 ) , skb - > data ,
skb - > len ) ;
current_tx_ptr - > desc_a . start_addr =
( u32 ) current_tx_ptr - > packet ;
if ( current_tx_ptr - > status . status_word ! = 0 )
current_tx_ptr - > status . status_word = 0 ;
blackfin_dcache_flush_range (
( u32 ) current_tx_ptr - > packet ,
( u32 ) ( current_tx_ptr - > packet + skb - > len + 2 ) ) ;
}
2007-07-17 10:43:44 +04:00
}
/* enable this packet's dma */
current_tx_ptr - > desc_a . config | = DMAEN ;
/* tx dma is running, just return */
if ( bfin_read_DMA2_IRQ_STATUS ( ) & 0x08 )
goto out ;
/* tx dma is not running */
bfin_write_DMA2_NEXT_DESC_PTR ( & ( current_tx_ptr - > desc_a ) ) ;
/* dma enabled, read from memory, size is 6 */
bfin_write_DMA2_CONFIG ( current_tx_ptr - > desc_a . config ) ;
/* Turn on the EMAC tx */
bfin_write_EMAC_OPMODE ( bfin_read_EMAC_OPMODE ( ) | TE ) ;
out :
adjust_tx_list ( ) ;
current_tx_ptr = current_tx_ptr - > next ;
dev - > trans_start = jiffies ;
2007-10-04 04:41:50 +04:00
dev - > stats . tx_packets + + ;
dev - > stats . tx_bytes + = ( skb - > len ) ;
2007-07-17 10:43:44 +04:00
return 0 ;
}
2008-04-25 07:53:10 +04:00
static void bfin_mac_rx ( struct net_device * dev )
2007-07-17 10:43:44 +04:00
{
struct sk_buff * skb , * new_skb ;
unsigned short len ;
/* allocate a new skb for next time receive */
skb = current_rx_ptr - > skb ;
new_skb = dev_alloc_skb ( PKT_BUF_SZ + 2 ) ;
if ( ! new_skb ) {
printk ( KERN_NOTICE DRV_NAME
" : rx: low on mem - packet dropped \n " ) ;
2007-10-04 04:41:50 +04:00
dev - > stats . rx_dropped + + ;
2007-07-17 10:43:44 +04:00
goto out ;
}
/* reserve 2 bytes for RXDWA padding */
skb_reserve ( new_skb , 2 ) ;
current_rx_ptr - > skb = new_skb ;
current_rx_ptr - > desc_a . start_addr = ( unsigned long ) new_skb - > data - 2 ;
2008-01-30 11:52:27 +03:00
/* Invidate the data cache of skb->data range when it is write back
* cache . It will prevent overwritting the new data from DMA
*/
blackfin_dcache_invalidate_range ( ( unsigned long ) new_skb - > head ,
( unsigned long ) new_skb - > end ) ;
2007-07-17 10:43:44 +04:00
len = ( unsigned short ) ( ( current_rx_ptr - > status . status_word ) & RX_FRLEN ) ;
skb_put ( skb , len ) ;
blackfin_dcache_invalidate_range ( ( unsigned long ) skb - > head ,
( unsigned long ) skb - > tail ) ;
dev - > last_rx = jiffies ;
skb - > protocol = eth_type_trans ( skb , dev ) ;
# if defined(BFIN_MAC_CSUM_OFFLOAD)
skb - > csum = current_rx_ptr - > status . ip_payload_csum ;
2007-11-23 12:55:51 +03:00
skb - > ip_summed = CHECKSUM_COMPLETE ;
2007-07-17 10:43:44 +04:00
# endif
netif_rx ( skb ) ;
2007-10-04 04:41:50 +04:00
dev - > stats . rx_packets + + ;
dev - > stats . rx_bytes + = len ;
2007-07-17 10:43:44 +04:00
current_rx_ptr - > status . status_word = 0x00000000 ;
current_rx_ptr = current_rx_ptr - > next ;
out :
return ;
}
/* interrupt routine to handle rx and error signal */
2008-04-25 07:53:10 +04:00
static irqreturn_t bfin_mac_interrupt ( int irq , void * dev_id )
2007-07-17 10:43:44 +04:00
{
struct net_device * dev = dev_id ;
int number = 0 ;
get_one_packet :
if ( current_rx_ptr - > status . status_word = = 0 ) {
/* no more new packet received */
if ( number = = 0 ) {
if ( current_rx_ptr - > next - > status . status_word ! = 0 ) {
current_rx_ptr = current_rx_ptr - > next ;
goto real_rx ;
}
}
bfin_write_DMA1_IRQ_STATUS ( bfin_read_DMA1_IRQ_STATUS ( ) |
DMA_DONE | DMA_ERR ) ;
return IRQ_HANDLED ;
}
real_rx :
2008-04-25 07:53:10 +04:00
bfin_mac_rx ( dev ) ;
2007-07-17 10:43:44 +04:00
number + + ;
goto get_one_packet ;
}
# ifdef CONFIG_NET_POLL_CONTROLLER
2008-04-25 07:53:10 +04:00
static void bfin_mac_poll ( struct net_device * dev )
2007-07-17 10:43:44 +04:00
{
disable_irq ( IRQ_MAC_RX ) ;
2008-04-25 07:53:10 +04:00
bfin_mac_interrupt ( IRQ_MAC_RX , dev ) ;
2007-07-17 10:43:44 +04:00
enable_irq ( IRQ_MAC_RX ) ;
}
# endif /* CONFIG_NET_POLL_CONTROLLER */
2008-04-25 07:53:10 +04:00
static void bfin_mac_disable ( void )
2007-07-17 10:43:44 +04:00
{
unsigned int opmode ;
opmode = bfin_read_EMAC_OPMODE ( ) ;
opmode & = ( ~ RE ) ;
opmode & = ( ~ TE ) ;
/* Turn off the EMAC */
bfin_write_EMAC_OPMODE ( opmode ) ;
}
/*
* Enable Interrupts , Receive , and Transmit
*/
2008-04-25 07:53:10 +04:00
static void bfin_mac_enable ( void )
2007-07-17 10:43:44 +04:00
{
u32 opmode ;
2007-09-19 19:37:36 +04:00
pr_debug ( " %s: %s \n " , DRV_NAME , __FUNCTION__ ) ;
2007-07-17 10:43:44 +04:00
/* Set RX DMA */
bfin_write_DMA1_NEXT_DESC_PTR ( & ( rx_list_head - > desc_a ) ) ;
bfin_write_DMA1_CONFIG ( rx_list_head - > desc_a . config ) ;
/* Wait MII done */
2007-09-19 19:37:36 +04:00
mdio_poll ( ) ;
2007-07-17 10:43:44 +04:00
/* We enable only RX here */
/* ASTP : Enable Automatic Pad Stripping
PR : Promiscuous Mode for test
PSF : Receive frames with total length less than 64 bytes .
FDMODE : Full Duplex Mode
LB : Internal Loopback for test
RE : Receiver Enable */
opmode = bfin_read_EMAC_OPMODE ( ) ;
if ( opmode & FDMODE )
opmode | = PSF ;
else
opmode | = DRO | DC | PSF ;
opmode | = RE ;
# if defined(CONFIG_BFIN_MAC_RMII)
opmode | = RMII ; /* For Now only 100MBit are supported */
2008-01-30 11:52:25 +03:00
# if (defined(CONFIG_BF537) || defined(CONFIG_BF536)) && CONFIG_BF_REV_0_2
2007-07-17 10:43:44 +04:00
opmode | = TE ;
# endif
# endif
/* Turn on the EMAC rx */
bfin_write_EMAC_OPMODE ( opmode ) ;
}
/* Our watchdog timed out. Called by the networking layer */
2008-04-25 07:53:10 +04:00
static void bfin_mac_timeout ( struct net_device * dev )
2007-07-17 10:43:44 +04:00
{
pr_debug ( " %s: %s \n " , dev - > name , __FUNCTION__ ) ;
2008-04-25 07:53:10 +04:00
bfin_mac_disable ( ) ;
2007-07-17 10:43:44 +04:00
/* reset tx queue */
tx_list_tail = tx_list_head - > next ;
2008-04-25 07:53:10 +04:00
bfin_mac_enable ( ) ;
2007-07-17 10:43:44 +04:00
/* We can accept TX packets again */
dev - > trans_start = jiffies ;
netif_wake_queue ( dev ) ;
}
2008-04-25 07:53:10 +04:00
static void bfin_mac_multicast_hash ( struct net_device * dev )
2008-01-30 11:52:23 +03:00
{
u32 emac_hashhi , emac_hashlo ;
struct dev_mc_list * dmi = dev - > mc_list ;
char * addrs ;
int i ;
u32 crc ;
emac_hashhi = emac_hashlo = 0 ;
for ( i = 0 ; i < dev - > mc_count ; i + + ) {
addrs = dmi - > dmi_addr ;
dmi = dmi - > next ;
/* skip non-multicast addresses */
if ( ! ( * addrs & 1 ) )
continue ;
crc = ether_crc ( ETH_ALEN , addrs ) ;
crc > > = 26 ;
if ( crc & 0x20 )
emac_hashhi | = 1 < < ( crc & 0x1f ) ;
else
emac_hashlo | = 1 < < ( crc & 0x1f ) ;
}
bfin_write_EMAC_HASHHI ( emac_hashhi ) ;
bfin_write_EMAC_HASHLO ( emac_hashlo ) ;
return ;
}
2007-07-17 10:43:44 +04:00
/*
* This routine will , depending on the values passed to it ,
* either make it accept multicast packets , go into
* promiscuous mode ( for TCPDUMP and cousins ) or accept
* a select set of multicast packets
*/
2008-04-25 07:53:10 +04:00
static void bfin_mac_set_multicast_list ( struct net_device * dev )
2007-07-17 10:43:44 +04:00
{
u32 sysctl ;
if ( dev - > flags & IFF_PROMISC ) {
printk ( KERN_INFO " %s: set to promisc mode \n " , dev - > name ) ;
sysctl = bfin_read_EMAC_OPMODE ( ) ;
sysctl | = RAF ;
bfin_write_EMAC_OPMODE ( sysctl ) ;
2008-01-30 11:52:23 +03:00
} else if ( dev - > flags & IFF_ALLMULTI ) {
2007-07-17 10:43:44 +04:00
/* accept all multicast */
sysctl = bfin_read_EMAC_OPMODE ( ) ;
sysctl | = PAM ;
bfin_write_EMAC_OPMODE ( sysctl ) ;
2008-01-30 11:52:23 +03:00
} else if ( dev - > mc_count ) {
/* set up multicast hash table */
sysctl = bfin_read_EMAC_OPMODE ( ) ;
sysctl | = HM ;
bfin_write_EMAC_OPMODE ( sysctl ) ;
2008-04-25 07:53:10 +04:00
bfin_mac_multicast_hash ( dev ) ;
2007-07-17 10:43:44 +04:00
} else {
/* clear promisc or multicast mode */
sysctl = bfin_read_EMAC_OPMODE ( ) ;
sysctl & = ~ ( RAF | PAM ) ;
bfin_write_EMAC_OPMODE ( sysctl ) ;
}
}
/*
* this puts the device in an inactive state
*/
2008-04-25 07:53:10 +04:00
static void bfin_mac_shutdown ( struct net_device * dev )
2007-07-17 10:43:44 +04:00
{
/* Turn off the EMAC */
bfin_write_EMAC_OPMODE ( 0x00000000 ) ;
/* Turn off the EMAC RX DMA */
bfin_write_DMA1_CONFIG ( 0x0000 ) ;
bfin_write_DMA2_CONFIG ( 0x0000 ) ;
}
/*
* Open and Initialize the interface
*
* Set up everything , reset the card , etc . .
*/
2008-04-25 07:53:10 +04:00
static int bfin_mac_open ( struct net_device * dev )
2007-07-17 10:43:44 +04:00
{
2008-04-25 07:53:10 +04:00
struct bfin_mac_local * lp = netdev_priv ( dev ) ;
2007-07-25 10:09:54 +04:00
int retval ;
2007-07-17 10:43:44 +04:00
pr_debug ( " %s: %s \n " , dev - > name , __FUNCTION__ ) ;
/*
* Check that the address is valid . If its not , refuse
* to bring the device up . The user must specify an
* address using ifconfig eth0 hw ether xx : xx : xx : xx : xx : xx
*/
if ( ! is_valid_ether_addr ( dev - > dev_addr ) ) {
printk ( KERN_WARNING DRV_NAME " : no valid ethernet hw addr \n " ) ;
return - EINVAL ;
}
/* initial rx and tx list */
2007-07-25 10:09:54 +04:00
retval = desc_list_init ( ) ;
if ( retval )
return retval ;
2007-07-17 10:43:44 +04:00
2007-09-19 19:37:36 +04:00
phy_start ( lp - > phydev ) ;
2008-01-30 11:52:26 +03:00
phy_write ( lp - > phydev , MII_BMCR , BMCR_RESET ) ;
2007-07-17 10:43:44 +04:00
setup_system_regs ( dev ) ;
2008-07-27 18:45:05 +04:00
setup_mac_addr ( dev - > dev_addr ) ;
2008-04-25 07:53:10 +04:00
bfin_mac_disable ( ) ;
bfin_mac_enable ( ) ;
2007-07-17 10:43:44 +04:00
pr_debug ( " hardware init finished \n " ) ;
netif_start_queue ( dev ) ;
netif_carrier_on ( dev ) ;
return 0 ;
}
/*
*
* this makes the board clean up everything that it can
* and not talk to the outside world . Caused by
* an ' ifconfig ethX down '
*/
2008-04-25 07:53:10 +04:00
static int bfin_mac_close ( struct net_device * dev )
2007-07-17 10:43:44 +04:00
{
2008-04-25 07:53:10 +04:00
struct bfin_mac_local * lp = netdev_priv ( dev ) ;
2007-07-17 10:43:44 +04:00
pr_debug ( " %s: %s \n " , dev - > name , __FUNCTION__ ) ;
netif_stop_queue ( dev ) ;
netif_carrier_off ( dev ) ;
2007-09-19 19:37:36 +04:00
phy_stop ( lp - > phydev ) ;
2008-01-30 11:52:26 +03:00
phy_write ( lp - > phydev , MII_BMCR , BMCR_PDOWN ) ;
2007-09-19 19:37:36 +04:00
2007-07-17 10:43:44 +04:00
/* clear everything */
2008-04-25 07:53:10 +04:00
bfin_mac_shutdown ( dev ) ;
2007-07-17 10:43:44 +04:00
/* free the rx/tx buffers */
desc_list_free ( ) ;
return 0 ;
}
2008-07-27 18:45:03 +04:00
static int __devinit bfin_mac_probe ( struct platform_device * pdev )
2007-07-17 10:43:44 +04:00
{
2008-04-25 07:53:10 +04:00
struct net_device * ndev ;
struct bfin_mac_local * lp ;
int rc , i ;
ndev = alloc_etherdev ( sizeof ( struct bfin_mac_local ) ) ;
if ( ! ndev ) {
dev_err ( & pdev - > dev , " Cannot allocate net device! \n " ) ;
return - ENOMEM ;
}
SET_NETDEV_DEV ( ndev , & pdev - > dev ) ;
platform_set_drvdata ( pdev , ndev ) ;
lp = netdev_priv ( ndev ) ;
2007-07-17 10:43:44 +04:00
/* Grab the MAC address in the MAC */
2008-04-25 07:53:10 +04:00
* ( __le32 * ) ( & ( ndev - > dev_addr [ 0 ] ) ) = cpu_to_le32 ( bfin_read_EMAC_ADDRLO ( ) ) ;
* ( __le16 * ) ( & ( ndev - > dev_addr [ 4 ] ) ) = cpu_to_le16 ( ( u16 ) bfin_read_EMAC_ADDRHI ( ) ) ;
2007-07-17 10:43:44 +04:00
/* probe mac */
/*todo: how to proble? which is revision_register */
bfin_write_EMAC_ADDRLO ( 0x12345678 ) ;
if ( bfin_read_EMAC_ADDRLO ( ) ! = 0x12345678 ) {
2008-04-25 07:53:10 +04:00
dev_err ( & pdev - > dev , " Cannot detect Blackfin on-chip ethernet MAC controller! \n " ) ;
rc = - ENODEV ;
goto out_err_probe_mac ;
2007-07-17 10:43:44 +04:00
}
/* set the GPIO pins to Ethernet mode */
2008-04-25 07:53:10 +04:00
rc = peripheral_request_list ( pin_req , DRV_NAME ) ;
if ( rc ) {
dev_err ( & pdev - > dev , " Requesting peripherals failed! \n " ) ;
rc = - EFAULT ;
goto out_err_setup_pin_mux ;
2007-07-17 10:43:44 +04:00
}
2008-04-25 07:53:10 +04:00
/*
* Is it valid ? ( Did bootloader initialize it ? )
* Grab the MAC from the board somehow
* this is done in the arch / blackfin / mach - bfxxx / boards / eth_mac . c
*/
if ( ! is_valid_ether_addr ( ndev - > dev_addr ) )
bfin_get_ether_addr ( ndev - > dev_addr ) ;
2007-07-17 10:43:44 +04:00
/* If still not valid, get a random one */
2008-04-25 07:53:10 +04:00
if ( ! is_valid_ether_addr ( ndev - > dev_addr ) )
random_ether_addr ( ndev - > dev_addr ) ;
2007-07-17 10:43:44 +04:00
2008-04-25 07:53:10 +04:00
setup_mac_addr ( ndev - > dev_addr ) ;
2007-07-17 10:43:44 +04:00
2007-09-19 19:37:36 +04:00
/* MDIO bus initial */
2008-04-25 07:53:10 +04:00
lp - > mii_bus . priv = ndev ;
2007-09-19 19:37:36 +04:00
lp - > mii_bus . read = mdiobus_read ;
lp - > mii_bus . write = mdiobus_write ;
lp - > mii_bus . reset = mdiobus_reset ;
lp - > mii_bus . name = " bfin_mac_mdio " ;
2008-04-10 04:38:13 +04:00
snprintf ( lp - > mii_bus . id , MII_BUS_ID_SIZE , " 0 " ) ;
2007-09-19 19:37:36 +04:00
lp - > mii_bus . irq = kmalloc ( sizeof ( int ) * PHY_MAX_ADDR , GFP_KERNEL ) ;
for ( i = 0 ; i < PHY_MAX_ADDR ; + + i )
lp - > mii_bus . irq [ i ] = PHY_POLL ;
2008-04-25 07:53:10 +04:00
rc = mdiobus_register ( & lp - > mii_bus ) ;
if ( rc ) {
dev_err ( & pdev - > dev , " Cannot register MDIO bus! \n " ) ;
goto out_err_mdiobus_register ;
}
2007-09-19 19:37:36 +04:00
2008-04-25 07:53:10 +04:00
rc = mii_probe ( ndev ) ;
if ( rc ) {
dev_err ( & pdev - > dev , " MII Probe failed! \n " ) ;
goto out_err_mii_probe ;
}
2007-09-19 19:37:36 +04:00
2007-07-17 10:43:44 +04:00
/* Fill in the fields of the device structure with ethernet values. */
2008-04-25 07:53:10 +04:00
ether_setup ( ndev ) ;
ndev - > open = bfin_mac_open ;
ndev - > stop = bfin_mac_close ;
ndev - > hard_start_xmit = bfin_mac_hard_start_xmit ;
ndev - > set_mac_address = bfin_mac_set_mac_address ;
ndev - > tx_timeout = bfin_mac_timeout ;
ndev - > set_multicast_list = bfin_mac_set_multicast_list ;
2007-07-17 10:43:44 +04:00
# ifdef CONFIG_NET_POLL_CONTROLLER
2008-04-25 07:53:10 +04:00
ndev - > poll_controller = bfin_mac_poll ;
2007-07-17 10:43:44 +04:00
# endif
2008-04-25 07:53:11 +04:00
ndev - > ethtool_ops = & bfin_mac_ethtool_ops ;
2007-07-17 10:43:44 +04:00
spin_lock_init ( & lp - > lock ) ;
/* now, enable interrupts */
/* register irq handler */
2008-04-25 07:53:10 +04:00
rc = request_irq ( IRQ_MAC_RX , bfin_mac_interrupt ,
IRQF_DISABLED | IRQF_SHARED , " EMAC_RX " , ndev ) ;
if ( rc ) {
dev_err ( & pdev - > dev , " Cannot request Blackfin MAC RX IRQ! \n " ) ;
rc = - EBUSY ;
goto out_err_request_irq ;
2007-07-17 10:43:44 +04:00
}
2008-04-25 07:53:10 +04:00
rc = register_netdev ( ndev ) ;
if ( rc ) {
dev_err ( & pdev - > dev , " Cannot register net device! \n " ) ;
goto out_err_reg_ndev ;
2007-07-17 10:43:44 +04:00
}
2008-04-25 07:53:10 +04:00
/* now, print out the card info, in a short format.. */
dev_info ( & pdev - > dev , " %s, Version %s \n " , DRV_DESC , DRV_VERSION ) ;
2007-07-17 10:43:44 +04:00
2008-04-25 07:53:10 +04:00
return 0 ;
2007-07-17 10:43:44 +04:00
2008-04-25 07:53:10 +04:00
out_err_reg_ndev :
free_irq ( IRQ_MAC_RX , ndev ) ;
out_err_request_irq :
out_err_mii_probe :
mdiobus_unregister ( & lp - > mii_bus ) ;
out_err_mdiobus_register :
peripheral_free_list ( pin_req ) ;
out_err_setup_pin_mux :
out_err_probe_mac :
platform_set_drvdata ( pdev , NULL ) ;
free_netdev ( ndev ) ;
2007-07-17 10:43:44 +04:00
2008-04-25 07:53:10 +04:00
return rc ;
2007-07-17 10:43:44 +04:00
}
2008-07-27 18:45:03 +04:00
static int __devexit bfin_mac_remove ( struct platform_device * pdev )
2007-07-17 10:43:44 +04:00
{
struct net_device * ndev = platform_get_drvdata ( pdev ) ;
2008-04-25 07:53:10 +04:00
struct bfin_mac_local * lp = netdev_priv ( ndev ) ;
2007-07-17 10:43:44 +04:00
platform_set_drvdata ( pdev , NULL ) ;
2008-04-25 07:53:10 +04:00
mdiobus_unregister ( & lp - > mii_bus ) ;
2007-07-17 10:43:44 +04:00
unregister_netdev ( ndev ) ;
free_irq ( IRQ_MAC_RX , ndev ) ;
free_netdev ( ndev ) ;
2008-04-25 07:53:10 +04:00
peripheral_free_list ( pin_req ) ;
2007-07-17 10:43:44 +04:00
return 0 ;
}
2007-09-19 19:37:14 +04:00
# ifdef CONFIG_PM
static int bfin_mac_suspend ( struct platform_device * pdev , pm_message_t mesg )
2007-07-17 10:43:44 +04:00
{
2007-09-19 19:37:14 +04:00
struct net_device * net_dev = platform_get_drvdata ( pdev ) ;
if ( netif_running ( net_dev ) )
2008-04-25 07:53:10 +04:00
bfin_mac_close ( net_dev ) ;
2007-09-19 19:37:14 +04:00
2007-07-17 10:43:44 +04:00
return 0 ;
}
static int bfin_mac_resume ( struct platform_device * pdev )
{
2007-09-19 19:37:14 +04:00
struct net_device * net_dev = platform_get_drvdata ( pdev ) ;
if ( netif_running ( net_dev ) )
2008-04-25 07:53:10 +04:00
bfin_mac_open ( net_dev ) ;
2007-09-19 19:37:14 +04:00
2007-07-17 10:43:44 +04:00
return 0 ;
}
2007-09-19 19:37:14 +04:00
# else
# define bfin_mac_suspend NULL
# define bfin_mac_resume NULL
# endif /* CONFIG_PM */
2007-07-17 10:43:44 +04:00
static struct platform_driver bfin_mac_driver = {
. probe = bfin_mac_probe ,
2008-07-27 18:45:03 +04:00
. remove = __devexit_p ( bfin_mac_remove ) ,
2007-07-17 10:43:44 +04:00
. resume = bfin_mac_resume ,
. suspend = bfin_mac_suspend ,
. driver = {
2008-04-19 00:50:44 +04:00
. name = DRV_NAME ,
. owner = THIS_MODULE ,
} ,
2007-07-17 10:43:44 +04:00
} ;
static int __init bfin_mac_init ( void )
{
return platform_driver_register ( & bfin_mac_driver ) ;
}
module_init ( bfin_mac_init ) ;
static void __exit bfin_mac_cleanup ( void )
{
platform_driver_unregister ( & bfin_mac_driver ) ;
}
module_exit ( bfin_mac_cleanup ) ;
2008-04-19 00:50:44 +04:00