2005-04-17 02:20:36 +04:00
/* mac8390.c: New driver for 8390-based Nubus (or Nubus-alike)
Ethernet cards on Linux */
/* Based on the former daynaport.c driver, by Alan Cox. Some code
taken from or inspired by skeleton . c by Donald Becker , acenic . c by
Jes Sorensen , and ne2k - pci . c by Donald Becker and Paul Gortmaker .
This software may be used and distributed according to the terms of
the GNU Public License , incorporated herein by reference . */
2006-09-13 21:24:59 +04:00
/* 2000-02-28: support added for Dayna and Kinetics cards by
2005-04-17 02:20:36 +04:00
A . G . deWijn @ phys . uu . nl */
/* 2000-04-04: support added for Dayna2 by bart@etpmod.phys.tue.nl */
/* 2001-04-18: support for DaynaPort E/LC-M by rayk@knightsmanor.org */
/* 2001-05-15: support for Cabletron ported from old daynaport driver
2006-09-13 21:24:59 +04:00
* and fixed access to Sonic Sys card which masquerades as a Farallon
2005-04-17 02:20:36 +04:00
* by rayk @ knightsmanor . org */
2007-05-02 00:32:54 +04:00
/* 2002-12-30: Try to support more cards, some clues from NetBSD driver */
/* 2003-12-26: Make sure Asante cards always work. */
2005-04-17 02:20:36 +04:00
2010-01-04 14:53:01 +03:00
# define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
2005-04-17 02:20:36 +04:00
# include <linux/module.h>
# include <linux/kernel.h>
# include <linux/types.h>
# include <linux/fcntl.h>
# include <linux/interrupt.h>
# include <linux/ptrace.h>
# include <linux/ioport.h>
# include <linux/nubus.h>
# include <linux/in.h>
# include <linux/string.h>
# include <linux/errno.h>
# include <linux/init.h>
# include <linux/netdevice.h>
# include <linux/etherdevice.h>
# include <linux/skbuff.h>
# include <linux/bitops.h>
2010-01-04 14:53:02 +03:00
# include <linux/io.h>
2005-04-17 02:20:36 +04:00
# include <asm/dma.h>
# include <asm/hwtest.h>
# include <asm/macints.h>
2006-10-10 03:19:36 +04:00
static char version [ ] =
2010-01-04 14:53:01 +03:00
" v0.4 2001-05-15 David Huggins-Daines <dhd@debian.org> and others \n " ;
2006-10-10 03:19:36 +04:00
# define EI_SHIFT(x) (ei_local->reg_offset[x])
2010-01-04 14:53:02 +03:00
# define ei_inb(port) in_8(port)
# define ei_outb(val, port) out_8(port, val)
# define ei_inb_p(port) in_8(port)
# define ei_outb_p(val, port) out_8(port, val)
2006-10-10 03:19:36 +04:00
# include "lib8390.c"
2005-04-17 02:20:36 +04:00
# define WD_START_PG 0x00 /* First page of TX buffer */
# define CABLETRON_RX_START_PG 0x00 /* First page of RX buffer */
# define CABLETRON_RX_STOP_PG 0x30 /* Last page +1 of RX ring */
2010-01-04 14:53:02 +03:00
# define CABLETRON_TX_START_PG CABLETRON_RX_STOP_PG
/* First page of TX buffer */
2005-04-17 02:20:36 +04:00
2010-01-04 14:53:02 +03:00
/*
* Unfortunately it seems we have to hardcode these for the moment
* Shouldn ' t the card know about this ?
* Does anyone know where to read it off the card ?
* Do we trust the data provided by the card ?
*/
2005-04-17 02:20:36 +04:00
# define DAYNA_8390_BASE 0x80000
# define DAYNA_8390_MEM 0x00000
2006-09-13 21:24:59 +04:00
# define CABLETRON_8390_BASE 0x90000
2005-04-17 02:20:36 +04:00
# define CABLETRON_8390_MEM 0x00000
2007-05-02 00:32:54 +04:00
# define INTERLAN_8390_BASE 0xE0000
# define INTERLAN_8390_MEM 0xD0000
2005-04-17 02:20:36 +04:00
enum mac8390_type {
MAC8390_NONE = - 1 ,
MAC8390_APPLE ,
MAC8390_ASANTE ,
2007-05-02 00:32:54 +04:00
MAC8390_FARALLON ,
2005-04-17 02:20:36 +04:00
MAC8390_CABLETRON ,
MAC8390_DAYNA ,
MAC8390_INTERLAN ,
MAC8390_KINETICS ,
} ;
2010-01-04 14:53:02 +03:00
static const char * cardname [ ] = {
2005-04-17 02:20:36 +04:00
" apple " ,
" asante " ,
" farallon " ,
" cabletron " ,
" dayna " ,
" interlan " ,
" kinetics " ,
} ;
2010-01-04 14:53:02 +03:00
static const int word16 [ ] = {
2005-04-17 02:20:36 +04:00
1 , /* apple */
1 , /* asante */
1 , /* farallon */
1 , /* cabletron */
0 , /* dayna */
1 , /* interlan */
0 , /* kinetics */
} ;
/* on which cards do we use NuBus resources? */
2010-01-04 14:53:02 +03:00
static const int useresources [ ] = {
2005-04-17 02:20:36 +04:00
1 , /* apple */
1 , /* asante */
1 , /* farallon */
0 , /* cabletron */
0 , /* dayna */
0 , /* interlan */
0 , /* kinetics */
2007-05-02 00:32:54 +04:00
} ;
enum mac8390_access {
ACCESS_UNKNOWN = 0 ,
ACCESS_32 ,
ACCESS_16 ,
2005-04-17 02:20:36 +04:00
} ;
2010-01-04 14:53:02 +03:00
extern int mac8390_memtest ( struct net_device * dev ) ;
2018-02-19 05:39:17 +03:00
static int mac8390_initdev ( struct net_device * dev , struct nubus_board * board ,
2005-04-17 02:20:36 +04:00
enum mac8390_type type ) ;
2010-01-04 14:53:02 +03:00
static int mac8390_open ( struct net_device * dev ) ;
static int mac8390_close ( struct net_device * dev ) ;
2005-04-17 02:20:36 +04:00
static void mac8390_no_reset ( struct net_device * dev ) ;
2007-05-02 00:32:54 +04:00
static void interlan_reset ( struct net_device * dev ) ;
2005-04-17 02:20:36 +04:00
2007-05-02 00:32:54 +04:00
/* Sane (32-bit chunk memory read/write) - Some Farallon and Apple do this*/
2005-04-17 02:20:36 +04:00
static void sane_get_8390_hdr ( struct net_device * dev ,
struct e8390_pkt_hdr * hdr , int ring_page ) ;
2010-01-04 14:53:02 +03:00
static void sane_block_input ( struct net_device * dev , int count ,
struct sk_buff * skb , int ring_offset ) ;
static void sane_block_output ( struct net_device * dev , int count ,
const unsigned char * buf , const int start_page ) ;
2005-04-17 02:20:36 +04:00
/* dayna_memcpy to and from card */
static void dayna_memcpy_fromcard ( struct net_device * dev , void * to ,
int from , int count ) ;
static void dayna_memcpy_tocard ( struct net_device * dev , int to ,
const void * from , int count ) ;
/* Dayna - Dayna/Kinetics use this */
static void dayna_get_8390_hdr ( struct net_device * dev ,
struct e8390_pkt_hdr * hdr , int ring_page ) ;
static void dayna_block_input ( struct net_device * dev , int count ,
struct sk_buff * skb , int ring_offset ) ;
static void dayna_block_output ( struct net_device * dev , int count ,
const unsigned char * buf , int start_page ) ;
/* Slow Sane (16-bit chunk memory read/write) Cabletron uses this */
static void slow_sane_get_8390_hdr ( struct net_device * dev ,
struct e8390_pkt_hdr * hdr , int ring_page ) ;
static void slow_sane_block_input ( struct net_device * dev , int count ,
struct sk_buff * skb , int ring_offset ) ;
static void slow_sane_block_output ( struct net_device * dev , int count ,
const unsigned char * buf , int start_page ) ;
2010-06-02 11:36:20 +04:00
static void word_memcpy_tocard ( unsigned long tp , const void * fp , int count ) ;
static void word_memcpy_fromcard ( void * tp , unsigned long fp , int count ) ;
2005-04-17 02:20:36 +04:00
2018-02-19 05:39:17 +03:00
static enum mac8390_type mac8390_ident ( struct nubus_rsrc * fres )
2005-04-17 02:20:36 +04:00
{
2018-01-14 01:37:13 +03:00
switch ( fres - > dr_sw ) {
2010-01-04 14:52:59 +03:00
case NUBUS_DRSW_3COM :
2018-01-14 01:37:13 +03:00
switch ( fres - > dr_hw ) {
2010-01-04 14:52:59 +03:00
case NUBUS_DRHW_APPLE_SONIC_NB :
case NUBUS_DRHW_APPLE_SONIC_LC :
case NUBUS_DRHW_SONNET :
return MAC8390_NONE ;
default :
return MAC8390_APPLE ;
}
break ;
2007-05-02 00:32:54 +04:00
2010-01-04 14:52:59 +03:00
case NUBUS_DRSW_APPLE :
2018-01-14 01:37:13 +03:00
switch ( fres - > dr_hw ) {
2010-01-04 14:52:59 +03:00
case NUBUS_DRHW_ASANTE_LC :
return MAC8390_NONE ;
case NUBUS_DRHW_CABLETRON :
return MAC8390_CABLETRON ;
default :
return MAC8390_APPLE ;
}
break ;
2007-05-02 00:32:54 +04:00
2010-01-04 14:52:59 +03:00
case NUBUS_DRSW_ASANTE :
return MAC8390_ASANTE ;
break ;
case NUBUS_DRSW_TECHWORKS :
case NUBUS_DRSW_DAYNA2 :
case NUBUS_DRSW_DAYNA_LC :
2018-01-14 01:37:13 +03:00
if ( fres - > dr_hw = = NUBUS_DRHW_CABLETRON )
2010-01-04 14:52:59 +03:00
return MAC8390_CABLETRON ;
else
return MAC8390_APPLE ;
break ;
case NUBUS_DRSW_FARALLON :
return MAC8390_FARALLON ;
break ;
2007-05-02 00:32:54 +04:00
2010-01-04 14:52:59 +03:00
case NUBUS_DRSW_KINETICS :
2018-01-14 01:37:13 +03:00
switch ( fres - > dr_hw ) {
2010-01-04 14:52:59 +03:00
case NUBUS_DRHW_INTERLAN :
return MAC8390_INTERLAN ;
default :
return MAC8390_KINETICS ;
}
break ;
case NUBUS_DRSW_DAYNA :
2010-01-04 14:53:02 +03:00
/*
* These correspond to Dayna Sonic cards
* which use the macsonic driver
*/
2018-01-14 01:37:13 +03:00
if ( fres - > dr_hw = = NUBUS_DRHW_SMC9194 | |
fres - > dr_hw = = NUBUS_DRHW_INTERLAN )
2010-01-04 14:52:59 +03:00
return MAC8390_NONE ;
else
return MAC8390_DAYNA ;
break ;
2007-05-02 00:32:54 +04:00
}
2005-04-17 02:20:36 +04:00
return MAC8390_NONE ;
}
2018-02-19 05:39:17 +03:00
static enum mac8390_access mac8390_testio ( unsigned long membase )
2007-05-02 00:32:54 +04:00
{
2019-03-16 06:21:19 +03:00
u32 outdata = 0xA5A0B5B0 ;
u32 indata = 0 ;
2007-05-02 00:32:54 +04:00
/* Try writing 32 bits */
2019-03-16 06:21:19 +03:00
nubus_writel ( outdata , membase ) ;
/* Now read it back */
indata = nubus_readl ( membase ) ;
if ( outdata = = indata )
2007-05-02 00:32:54 +04:00
return ACCESS_32 ;
2019-03-16 06:21:19 +03:00
outdata = 0xC5C0D5D0 ;
indata = 0 ;
2007-05-02 00:32:54 +04:00
/* Write 16 bit output */
2010-01-04 14:53:00 +03:00
word_memcpy_tocard ( membase , & outdata , 4 ) ;
2007-05-02 00:32:54 +04:00
/* Now read it back */
2010-01-04 14:53:00 +03:00
word_memcpy_fromcard ( & indata , membase , 4 ) ;
2007-05-02 00:32:54 +04:00
if ( outdata = = indata )
return ACCESS_16 ;
2019-03-16 06:21:19 +03:00
2007-05-02 00:32:54 +04:00
return ACCESS_UNKNOWN ;
}
2018-02-19 05:39:17 +03:00
static int mac8390_memsize ( unsigned long membase )
2005-04-17 02:20:36 +04:00
{
unsigned long flags ;
int i , j ;
2006-09-13 21:24:59 +04:00
2005-04-17 02:20:36 +04:00
local_irq_save ( flags ) ;
/* Check up to 32K in 4K increments */
for ( i = 0 ; i < 8 ; i + + ) {
2010-01-04 14:53:02 +03:00
volatile unsigned short * m = ( unsigned short * ) ( membase + ( i * 0x1000 ) ) ;
2005-04-17 02:20:36 +04:00
/* Unwriteable - we have a fully decoded card and the
RAM end located */
if ( hwreg_present ( m ) = = 0 )
break ;
2006-09-13 21:24:59 +04:00
2005-04-17 02:20:36 +04:00
/* write a distinctive byte */
* m = 0xA5A0 | i ;
/* check that we read back what we wrote */
if ( * m ! = ( 0xA5A0 | i ) )
break ;
/* check for partial decode and wrap */
for ( j = 0 ; j < i ; j + + ) {
2010-01-04 14:53:02 +03:00
volatile unsigned short * p = ( unsigned short * ) ( membase + ( j * 0x1000 ) ) ;
2005-04-17 02:20:36 +04:00
if ( * p ! = ( 0xA5A0 | j ) )
break ;
2010-01-04 14:53:02 +03:00
}
}
2005-04-17 02:20:36 +04:00
local_irq_restore ( flags ) ;
2010-01-04 14:53:02 +03:00
/*
* in any case , we stopped once we tried one block too many ,
* or once we reached 32 K
*/
return i * 0x1000 ;
2005-04-17 02:20:36 +04:00
}
2018-02-19 05:39:17 +03:00
static bool mac8390_rsrc_init ( struct net_device * dev ,
struct nubus_rsrc * fres ,
enum mac8390_type cardtype )
2010-01-04 14:53:03 +03:00
{
2018-02-19 05:39:17 +03:00
struct nubus_board * board = fres - > board ;
2010-01-04 14:53:03 +03:00
struct nubus_dir dir ;
struct nubus_dirent ent ;
int offset ;
volatile unsigned short * i ;
2018-02-19 05:39:17 +03:00
dev - > irq = SLOT2IRQ ( board - > slot ) ;
2010-01-04 14:53:03 +03:00
/* This is getting to be a habit */
2018-02-19 05:39:17 +03:00
dev - > base_addr = board - > slot_addr | ( ( board - > slot & 0xf ) < < 20 ) ;
2010-01-04 14:53:03 +03:00
/*
* Get some Nubus info - we will trust the card ' s idea
* of where its memory and registers are .
*/
2018-02-19 05:39:17 +03:00
if ( nubus_get_func_dir ( fres , & dir ) = = - 1 ) {
2018-02-19 05:39:17 +03:00
dev_err ( & board - > dev ,
" Unable to get Nubus functional directory \n " ) ;
2010-01-04 14:53:03 +03:00
return false ;
}
/* Get the MAC address */
if ( nubus_find_rsrc ( & dir , NUBUS_RESID_MAC_ADDRESS , & ent ) = = - 1 ) {
2018-02-19 05:39:17 +03:00
dev_info ( & board - > dev , " MAC address resource not found \n " ) ;
2010-01-04 14:53:03 +03:00
return false ;
}
nubus_get_rsrc_mem ( dev - > dev_addr , & ent , 6 ) ;
if ( useresources [ cardtype ] = = 1 ) {
nubus_rewinddir ( & dir ) ;
if ( nubus_find_rsrc ( & dir , NUBUS_RESID_MINOR_BASEOS ,
& ent ) = = - 1 ) {
2018-02-19 05:39:17 +03:00
dev_err ( & board - > dev ,
" Memory offset resource not found \n " ) ;
2010-01-04 14:53:03 +03:00
return false ;
}
nubus_get_rsrc_mem ( & offset , & ent , 4 ) ;
dev - > mem_start = dev - > base_addr + offset ;
/* yes, this is how the Apple driver does it */
dev - > base_addr = dev - > mem_start + 0x10000 ;
nubus_rewinddir ( & dir ) ;
if ( nubus_find_rsrc ( & dir , NUBUS_RESID_MINOR_LENGTH ,
& ent ) = = - 1 ) {
2018-02-19 05:39:17 +03:00
dev_info ( & board - > dev ,
" Memory length resource not found, probing \n " ) ;
2010-01-04 14:53:03 +03:00
offset = mac8390_memsize ( dev - > mem_start ) ;
} else {
nubus_get_rsrc_mem ( & offset , & ent , 4 ) ;
}
dev - > mem_end = dev - > mem_start + offset ;
} else {
switch ( cardtype ) {
case MAC8390_KINETICS :
case MAC8390_DAYNA : /* it's the same */
2018-02-19 05:39:17 +03:00
dev - > base_addr = ( int ) ( board - > slot_addr +
2010-01-04 14:53:03 +03:00
DAYNA_8390_BASE ) ;
2018-02-19 05:39:17 +03:00
dev - > mem_start = ( int ) ( board - > slot_addr +
2010-01-04 14:53:03 +03:00
DAYNA_8390_MEM ) ;
dev - > mem_end = dev - > mem_start +
mac8390_memsize ( dev - > mem_start ) ;
break ;
case MAC8390_INTERLAN :
2018-02-19 05:39:17 +03:00
dev - > base_addr = ( int ) ( board - > slot_addr +
2010-01-04 14:53:03 +03:00
INTERLAN_8390_BASE ) ;
2018-02-19 05:39:17 +03:00
dev - > mem_start = ( int ) ( board - > slot_addr +
2010-01-04 14:53:03 +03:00
INTERLAN_8390_MEM ) ;
dev - > mem_end = dev - > mem_start +
mac8390_memsize ( dev - > mem_start ) ;
break ;
case MAC8390_CABLETRON :
2018-02-19 05:39:17 +03:00
dev - > base_addr = ( int ) ( board - > slot_addr +
2010-01-04 14:53:03 +03:00
CABLETRON_8390_BASE ) ;
2018-02-19 05:39:17 +03:00
dev - > mem_start = ( int ) ( board - > slot_addr +
2010-01-04 14:53:03 +03:00
CABLETRON_8390_MEM ) ;
/* The base address is unreadable if 0x00
* has been written to the command register
* Reset the chip by writing E8390_NODMA +
* E8390_PAGE0 + E8390_STOP just to be
* sure
*/
i = ( void * ) dev - > base_addr ;
* i = 0x21 ;
dev - > mem_end = dev - > mem_start +
mac8390_memsize ( dev - > mem_start ) ;
break ;
default :
2018-02-19 05:39:17 +03:00
dev_err ( & board - > dev ,
" No known base address for card type \n " ) ;
2010-01-04 14:53:03 +03:00
return false ;
}
}
return true ;
}
2018-02-19 05:39:17 +03:00
static int mac8390_device_probe ( struct nubus_board * board )
2005-04-17 02:20:36 +04:00
{
struct net_device * dev ;
int err = - ENODEV ;
2018-02-19 05:39:17 +03:00
struct nubus_rsrc * fres ;
enum mac8390_type cardtype = MAC8390_NONE ;
2005-04-17 02:20:36 +04:00
2009-05-26 09:43:49 +04:00
dev = ____alloc_ei_netdev ( 0 ) ;
2005-04-17 02:20:36 +04:00
if ( ! dev )
2018-02-19 05:39:17 +03:00
return - ENOMEM ;
2005-04-17 02:20:36 +04:00
2018-02-19 05:39:17 +03:00
SET_NETDEV_DEV ( dev , & board - > dev ) ;
2005-04-17 02:20:36 +04:00
2018-02-19 05:39:17 +03:00
for_each_board_func_rsrc ( board , fres ) {
if ( fres - > category ! = NUBUS_CAT_NETWORK | |
fres - > type ! = NUBUS_TYPE_ETHERNET )
2018-01-14 01:37:13 +03:00
continue ;
2018-02-19 05:39:17 +03:00
cardtype = mac8390_ident ( fres ) ;
2010-01-04 14:53:02 +03:00
if ( cardtype = = MAC8390_NONE )
2005-04-17 02:20:36 +04:00
continue ;
2018-02-19 05:39:17 +03:00
if ( mac8390_rsrc_init ( dev , fres , cardtype ) )
2005-04-17 02:20:36 +04:00
break ;
}
2018-02-19 05:39:17 +03:00
if ( ! fres )
goto out ;
2005-04-17 02:20:36 +04:00
2018-02-19 05:39:17 +03:00
err = mac8390_initdev ( dev , board , cardtype ) ;
if ( err )
2005-04-17 02:20:36 +04:00
goto out ;
2013-12-12 02:00:59 +04:00
2005-04-17 02:20:36 +04:00
err = register_netdev ( dev ) ;
if ( err )
goto out ;
2018-02-19 05:39:17 +03:00
nubus_set_drvdata ( board , dev ) ;
return 0 ;
2005-04-17 02:20:36 +04:00
out :
free_netdev ( dev ) ;
2018-02-19 05:39:17 +03:00
return err ;
}
static int mac8390_device_remove ( struct nubus_board * board )
{
struct net_device * dev = nubus_get_drvdata ( board ) ;
unregister_netdev ( dev ) ;
free_netdev ( dev ) ;
return 0 ;
2005-04-17 02:20:36 +04:00
}
2018-02-19 05:39:17 +03:00
static struct nubus_driver mac8390_driver = {
. probe = mac8390_device_probe ,
. remove = mac8390_device_remove ,
. driver = {
. name = KBUILD_MODNAME ,
. owner = THIS_MODULE ,
}
} ;
2005-04-17 02:20:36 +04:00
MODULE_AUTHOR ( " David Huggins-Daines <dhd@debian.org> and others " ) ;
MODULE_DESCRIPTION ( " Macintosh NS8390-based Nubus Ethernet driver " ) ;
MODULE_LICENSE ( " GPL " ) ;
2018-02-19 05:39:17 +03:00
static int __init mac8390_init ( void )
2005-04-17 02:20:36 +04:00
{
2018-02-19 05:39:17 +03:00
return nubus_driver_register ( & mac8390_driver ) ;
2005-04-17 02:20:36 +04:00
}
2018-02-19 05:39:17 +03:00
module_init ( mac8390_init ) ;
2005-04-17 02:20:36 +04:00
2018-02-19 05:39:17 +03:00
static void __exit mac8390_exit ( void )
2005-04-17 02:20:36 +04:00
{
2018-02-19 05:39:17 +03:00
nubus_driver_unregister ( & mac8390_driver ) ;
2005-04-17 02:20:36 +04:00
}
2018-02-19 05:39:17 +03:00
module_exit ( mac8390_exit ) ;
2005-04-17 02:20:36 +04:00
2008-12-03 02:00:28 +03:00
static const struct net_device_ops mac8390_netdev_ops = {
. ndo_open = mac8390_open ,
. ndo_stop = mac8390_close ,
2009-05-26 09:43:49 +04:00
. ndo_start_xmit = __ei_start_xmit ,
. ndo_tx_timeout = __ei_tx_timeout ,
. ndo_get_stats = __ei_get_stats ,
2011-08-16 10:29:01 +04:00
. ndo_set_rx_mode = __ei_set_multicast_list ,
2008-12-03 02:00:28 +03:00
. ndo_validate_addr = eth_validate_addr ,
2009-01-09 14:13:14 +03:00
. ndo_set_mac_address = eth_mac_addr ,
2008-12-03 02:00:28 +03:00
# ifdef CONFIG_NET_POLL_CONTROLLER
2009-05-28 06:05:53 +04:00
. ndo_poll_controller = __ei_poll ,
2008-12-03 02:00:28 +03:00
# endif
} ;
2018-02-19 05:39:17 +03:00
static int mac8390_initdev ( struct net_device * dev , struct nubus_board * board ,
enum mac8390_type type )
2005-04-17 02:20:36 +04:00
{
2010-01-04 14:53:02 +03:00
static u32 fwrd4_offsets [ 16 ] = {
2005-04-17 02:20:36 +04:00
0 , 4 , 8 , 12 ,
16 , 20 , 24 , 28 ,
32 , 36 , 40 , 44 ,
48 , 52 , 56 , 60
} ;
2010-01-04 14:53:02 +03:00
static u32 back4_offsets [ 16 ] = {
2005-04-17 02:20:36 +04:00
60 , 56 , 52 , 48 ,
44 , 40 , 36 , 32 ,
28 , 24 , 20 , 16 ,
12 , 8 , 4 , 0
} ;
2010-01-04 14:53:02 +03:00
static u32 fwrd2_offsets [ 16 ] = {
2005-04-17 02:20:36 +04:00
0 , 2 , 4 , 6 ,
8 , 10 , 12 , 14 ,
16 , 18 , 20 , 22 ,
24 , 26 , 28 , 30
} ;
2007-05-02 00:32:54 +04:00
int access_bitmode = 0 ;
2006-09-13 21:24:59 +04:00
2005-04-17 02:20:36 +04:00
/* Now fill in our stuff */
2008-12-03 02:00:28 +03:00
dev - > netdev_ops = & mac8390_netdev_ops ;
2005-04-17 02:20:36 +04:00
/* GAR, ei_status is actually a macro even though it looks global */
ei_status . name = cardname [ type ] ;
ei_status . word16 = word16 [ type ] ;
/* Cabletron's TX/RX buffers are backwards */
if ( type = = MAC8390_CABLETRON ) {
2010-01-04 14:53:02 +03:00
ei_status . tx_start_page = CABLETRON_TX_START_PG ;
ei_status . rx_start_page = CABLETRON_RX_START_PG ;
ei_status . stop_page = CABLETRON_RX_STOP_PG ;
ei_status . rmem_start = dev - > mem_start ;
ei_status . rmem_end = dev - > mem_start + CABLETRON_RX_STOP_PG * 256 ;
2005-04-17 02:20:36 +04:00
} else {
2010-01-04 14:53:02 +03:00
ei_status . tx_start_page = WD_START_PG ;
ei_status . rx_start_page = WD_START_PG + TX_PAGES ;
ei_status . stop_page = ( dev - > mem_end - dev - > mem_start ) / 256 ;
ei_status . rmem_start = dev - > mem_start + TX_PAGES * 256 ;
ei_status . rmem_end = dev - > mem_end ;
2005-04-17 02:20:36 +04:00
}
2006-09-13 21:24:59 +04:00
2005-04-17 02:20:36 +04:00
/* Fill in model-specific information and functions */
2010-01-04 14:52:59 +03:00
switch ( type ) {
2005-04-17 02:20:36 +04:00
case MAC8390_FARALLON :
case MAC8390_APPLE :
2010-01-04 14:52:59 +03:00
switch ( mac8390_testio ( dev - > mem_start ) ) {
case ACCESS_UNKNOWN :
2018-02-19 05:39:17 +03:00
dev_err ( & board - > dev ,
" Don't know how to access card memory \n " ) ;
2010-01-04 14:52:59 +03:00
return - ENODEV ;
2007-05-02 00:32:54 +04:00
2010-01-04 14:52:59 +03:00
case ACCESS_16 :
/* 16 bit card, register map is reversed */
2010-08-23 22:20:03 +04:00
ei_status . reset_8390 = mac8390_no_reset ;
ei_status . block_input = slow_sane_block_input ;
ei_status . block_output = slow_sane_block_output ;
ei_status . get_8390_hdr = slow_sane_get_8390_hdr ;
2010-01-04 14:52:59 +03:00
ei_status . reg_offset = back4_offsets ;
break ;
2007-05-02 00:32:54 +04:00
2010-01-04 14:52:59 +03:00
case ACCESS_32 :
/* 32 bit card, register map is reversed */
2010-08-23 22:20:03 +04:00
ei_status . reset_8390 = mac8390_no_reset ;
ei_status . block_input = sane_block_input ;
ei_status . block_output = sane_block_output ;
ei_status . get_8390_hdr = sane_get_8390_hdr ;
2010-01-04 14:52:59 +03:00
ei_status . reg_offset = back4_offsets ;
access_bitmode = 1 ;
break ;
2007-05-02 00:32:54 +04:00
}
break ;
2005-04-17 02:20:36 +04:00
case MAC8390_ASANTE :
2007-05-02 00:32:54 +04:00
/* Some Asante cards pass the 32 bit test
* but overwrite system memory when run at 32 bit .
* so we run them all at 16 bit .
*/
2010-08-23 22:20:03 +04:00
ei_status . reset_8390 = mac8390_no_reset ;
ei_status . block_input = slow_sane_block_input ;
ei_status . block_output = slow_sane_block_output ;
ei_status . get_8390_hdr = slow_sane_get_8390_hdr ;
2005-04-17 02:20:36 +04:00
ei_status . reg_offset = back4_offsets ;
break ;
2007-05-02 00:32:54 +04:00
2005-04-17 02:20:36 +04:00
case MAC8390_CABLETRON :
/* 16 bit card, register map is short forward */
2010-08-23 22:20:03 +04:00
ei_status . reset_8390 = mac8390_no_reset ;
ei_status . block_input = slow_sane_block_input ;
ei_status . block_output = slow_sane_block_output ;
ei_status . get_8390_hdr = slow_sane_get_8390_hdr ;
2005-04-17 02:20:36 +04:00
ei_status . reg_offset = fwrd2_offsets ;
break ;
2007-05-02 00:32:54 +04:00
2005-04-17 02:20:36 +04:00
case MAC8390_DAYNA :
case MAC8390_KINETICS :
2007-05-02 00:32:54 +04:00
/* 16 bit memory, register map is forward */
2005-04-17 02:20:36 +04:00
/* dayna and similar */
2010-08-23 22:20:03 +04:00
ei_status . reset_8390 = mac8390_no_reset ;
ei_status . block_input = dayna_block_input ;
ei_status . block_output = dayna_block_output ;
ei_status . get_8390_hdr = dayna_get_8390_hdr ;
2005-04-17 02:20:36 +04:00
ei_status . reg_offset = fwrd4_offsets ;
break ;
2007-05-02 00:32:54 +04:00
case MAC8390_INTERLAN :
/* 16 bit memory, register map is forward */
2010-08-23 22:20:03 +04:00
ei_status . reset_8390 = interlan_reset ;
ei_status . block_input = slow_sane_block_input ;
ei_status . block_output = slow_sane_block_output ;
ei_status . get_8390_hdr = slow_sane_get_8390_hdr ;
2010-01-04 14:53:02 +03:00
ei_status . reg_offset = fwrd4_offsets ;
break ;
2007-05-02 00:32:54 +04:00
2005-04-17 02:20:36 +04:00
default :
2018-02-19 05:39:17 +03:00
dev_err ( & board - > dev , " Unsupported card type \n " ) ;
2005-04-17 02:20:36 +04:00
return - ENODEV ;
}
2006-09-13 21:24:59 +04:00
2006-10-10 03:19:36 +04:00
__NS8390_init ( dev , 0 ) ;
2005-04-17 02:20:36 +04:00
/* Good, done, now spit out some messages */
2018-02-19 05:39:17 +03:00
dev_info ( & board - > dev , " %s (type %s) \n " , board - > name , cardname [ type ] ) ;
dev_info ( & board - > dev , " MAC %pM, IRQ %d, %d KB shared memory at %#lx, %d-bit access. \n " ,
dev - > dev_addr , dev - > irq ,
( unsigned int ) ( dev - > mem_end - dev - > mem_start ) > > 10 ,
dev - > mem_start , access_bitmode ? 32 : 16 ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
static int mac8390_open ( struct net_device * dev )
{
2010-06-01 06:18:32 +04:00
int err ;
2006-10-10 03:19:36 +04:00
__ei_open ( dev ) ;
2010-06-01 06:18:32 +04:00
err = request_irq ( dev - > irq , __ei_interrupt , 0 , " 8390 Ethernet " , dev ) ;
if ( err )
2010-06-02 18:06:34 +04:00
pr_err ( " %s: unable to get IRQ %d \n " , dev - > name , dev - > irq ) ;
2010-06-01 06:18:32 +04:00
return err ;
2005-04-17 02:20:36 +04:00
}
static int mac8390_close ( struct net_device * dev )
{
free_irq ( dev - > irq , dev ) ;
2006-10-10 03:19:36 +04:00
__ei_close ( dev ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
static void mac8390_no_reset ( struct net_device * dev )
{
2013-12-12 02:00:59 +04:00
struct ei_device * ei_local = netdev_priv ( dev ) ;
2005-04-17 02:20:36 +04:00
ei_status . txing = 0 ;
2013-12-12 02:00:59 +04:00
netif_info ( ei_local , hw , dev , " reset not supported \n " ) ;
2005-04-17 02:20:36 +04:00
}
2007-05-02 00:32:54 +04:00
static void interlan_reset ( struct net_device * dev )
{
2010-01-04 14:53:02 +03:00
unsigned char * target = nubus_slot_addr ( IRQ2SLOT ( dev - > irq ) ) ;
2013-12-12 02:00:59 +04:00
struct ei_device * ei_local = netdev_priv ( dev ) ;
netif_info ( ei_local , hw , dev , " Need to reset the NS8390 t=%lu... " ,
jiffies ) ;
2007-05-02 00:32:54 +04:00
ei_status . txing = 0 ;
target [ 0xC0000 ] = 0 ;
2013-12-12 02:00:59 +04:00
if ( netif_msg_hw ( ei_local ) )
2010-01-04 14:53:01 +03:00
pr_cont ( " reset complete \n " ) ;
2007-05-02 00:32:54 +04:00
}
2005-04-17 02:20:36 +04:00
/* dayna_memcpy_fromio/dayna_memcpy_toio */
/* directly from daynaport.c by Alan Cox */
2010-01-04 14:53:02 +03:00
static void dayna_memcpy_fromcard ( struct net_device * dev , void * to , int from ,
int count )
2005-04-17 02:20:36 +04:00
{
2006-01-12 12:06:21 +03:00
volatile unsigned char * ptr ;
2010-01-04 14:53:02 +03:00
unsigned char * target = to ;
from < < = 1 ; /* word, skip overhead */
ptr = ( unsigned char * ) ( dev - > mem_start + from ) ;
2005-04-17 02:20:36 +04:00
/* Leading byte? */
2010-01-04 14:53:02 +03:00
if ( from & 2 ) {
2006-01-12 12:06:21 +03:00
* target + + = ptr [ - 1 ] ;
ptr + = 2 ;
2005-04-17 02:20:36 +04:00
count - - ;
}
2010-01-04 14:53:02 +03:00
while ( count > = 2 ) {
2006-01-12 12:06:21 +03:00
* ( unsigned short * ) target = * ( unsigned short volatile * ) ptr ;
ptr + = 4 ; /* skip cruft */
target + = 2 ;
2010-01-04 14:53:02 +03:00
count - = 2 ;
2005-04-17 02:20:36 +04:00
}
/* Trailing byte? */
2010-01-04 14:53:02 +03:00
if ( count )
2006-01-12 12:06:21 +03:00
* target = * ptr ;
2005-04-17 02:20:36 +04:00
}
2010-01-04 14:53:02 +03:00
static void dayna_memcpy_tocard ( struct net_device * dev , int to ,
const void * from , int count )
2005-04-17 02:20:36 +04:00
{
volatile unsigned short * ptr ;
2010-01-04 14:53:02 +03:00
const unsigned char * src = from ;
to < < = 1 ; /* word, skip overhead */
ptr = ( unsigned short * ) ( dev - > mem_start + to ) ;
2005-04-17 02:20:36 +04:00
/* Leading byte? */
2010-01-04 14:53:02 +03:00
if ( to & 2 ) { /* avoid a byte write (stomps on other data) */
2006-01-12 12:06:21 +03:00
ptr [ - 1 ] = ( ptr [ - 1 ] & 0xFF00 ) | * src + + ;
2005-04-17 02:20:36 +04:00
ptr + + ;
count - - ;
}
2010-01-04 14:53:02 +03:00
while ( count > = 2 ) {
* ptr + + = * ( unsigned short * ) src ; /* Copy and */
2005-04-17 02:20:36 +04:00
ptr + + ; /* skip cruft */
2006-01-12 12:06:21 +03:00
src + = 2 ;
2010-01-04 14:53:02 +03:00
count - = 2 ;
2005-04-17 02:20:36 +04:00
}
/* Trailing byte? */
2010-01-04 14:53:02 +03:00
if ( count ) {
2005-04-17 02:20:36 +04:00
/* card doesn't like byte writes */
2010-01-04 14:53:02 +03:00
* ptr = ( * ptr & 0x00FF ) | ( * src < < 8 ) ;
2005-04-17 02:20:36 +04:00
}
}
/* sane block input/output */
static void sane_get_8390_hdr ( struct net_device * dev ,
struct e8390_pkt_hdr * hdr , int ring_page )
{
unsigned long hdr_start = ( ring_page - WD_START_PG ) < < 8 ;
2018-06-26 21:32:22 +03:00
memcpy_fromio ( hdr , ( void __iomem * ) dev - > mem_start + hdr_start , 4 ) ;
2005-04-17 02:20:36 +04:00
/* Fix endianness */
hdr - > count = swab16 ( hdr - > count ) ;
}
static void sane_block_input ( struct net_device * dev , int count ,
struct sk_buff * skb , int ring_offset )
{
unsigned long xfer_base = ring_offset - ( WD_START_PG < < 8 ) ;
unsigned long xfer_start = xfer_base + dev - > mem_start ;
if ( xfer_start + count > ei_status . rmem_end ) {
/* We must wrap the input move. */
int semi_count = ei_status . rmem_end - xfer_start ;
2018-06-26 21:32:22 +03:00
memcpy_fromio ( skb - > data ,
( void __iomem * ) dev - > mem_start + xfer_base ,
2010-01-04 14:53:02 +03:00
semi_count ) ;
2005-04-17 02:20:36 +04:00
count - = semi_count ;
2018-06-26 21:32:22 +03:00
memcpy_fromio ( skb - > data + semi_count ,
( void __iomem * ) ei_status . rmem_start , count ) ;
2010-06-02 11:36:20 +04:00
} else {
2018-06-26 21:32:22 +03:00
memcpy_fromio ( skb - > data ,
( void __iomem * ) dev - > mem_start + xfer_base ,
count ) ;
2005-04-17 02:20:36 +04:00
}
}
static void sane_block_output ( struct net_device * dev , int count ,
const unsigned char * buf , int start_page )
{
long shmem = ( start_page - WD_START_PG ) < < 8 ;
2006-09-13 21:24:59 +04:00
2018-06-26 21:32:22 +03:00
memcpy_toio ( ( void __iomem * ) dev - > mem_start + shmem , buf , count ) ;
2005-04-17 02:20:36 +04:00
}
/* dayna block input/output */
2010-01-04 14:53:02 +03:00
static void dayna_get_8390_hdr ( struct net_device * dev ,
struct e8390_pkt_hdr * hdr , int ring_page )
2005-04-17 02:20:36 +04:00
{
unsigned long hdr_start = ( ring_page - WD_START_PG ) < < 8 ;
2010-01-04 14:53:00 +03:00
dayna_memcpy_fromcard ( dev , hdr , hdr_start , 4 ) ;
2005-04-17 02:20:36 +04:00
/* Fix endianness */
2010-01-04 14:53:02 +03:00
hdr - > count = ( hdr - > count & 0xFF ) < < 8 | ( hdr - > count > > 8 ) ;
2005-04-17 02:20:36 +04:00
}
2010-01-04 14:53:02 +03:00
static void dayna_block_input ( struct net_device * dev , int count ,
struct sk_buff * skb , int ring_offset )
2005-04-17 02:20:36 +04:00
{
unsigned long xfer_base = ring_offset - ( WD_START_PG < < 8 ) ;
unsigned long xfer_start = xfer_base + dev - > mem_start ;
/* Note the offset math is done in card memory space which is word
per long onto our space . */
2010-01-04 14:53:02 +03:00
if ( xfer_start + count > ei_status . rmem_end ) {
2005-04-17 02:20:36 +04:00
/* We must wrap the input move. */
int semi_count = ei_status . rmem_end - xfer_start ;
dayna_memcpy_fromcard ( dev , skb - > data , xfer_base , semi_count ) ;
count - = semi_count ;
dayna_memcpy_fromcard ( dev , skb - > data + semi_count ,
ei_status . rmem_start - dev - > mem_start ,
count ) ;
2010-01-04 14:53:02 +03:00
} else {
2005-04-17 02:20:36 +04:00
dayna_memcpy_fromcard ( dev , skb - > data , xfer_base , count ) ;
}
}
2010-01-04 14:53:02 +03:00
static void dayna_block_output ( struct net_device * dev , int count ,
const unsigned char * buf ,
int start_page )
2005-04-17 02:20:36 +04:00
{
long shmem = ( start_page - WD_START_PG ) < < 8 ;
2006-09-13 21:24:59 +04:00
2005-04-17 02:20:36 +04:00
dayna_memcpy_tocard ( dev , shmem , buf , count ) ;
}
/* Cabletron block I/O */
2010-01-04 14:53:02 +03:00
static void slow_sane_get_8390_hdr ( struct net_device * dev ,
struct e8390_pkt_hdr * hdr ,
int ring_page )
2005-04-17 02:20:36 +04:00
{
unsigned long hdr_start = ( ring_page - WD_START_PG ) < < 8 ;
2010-06-02 11:36:20 +04:00
word_memcpy_fromcard ( hdr , dev - > mem_start + hdr_start , 4 ) ;
2005-04-17 02:20:36 +04:00
/* Register endianism - fix here rather than 8390.c */
hdr - > count = ( hdr - > count & 0xFF ) < < 8 | ( hdr - > count > > 8 ) ;
}
2010-01-04 14:53:02 +03:00
static void slow_sane_block_input ( struct net_device * dev , int count ,
struct sk_buff * skb , int ring_offset )
2005-04-17 02:20:36 +04:00
{
unsigned long xfer_base = ring_offset - ( WD_START_PG < < 8 ) ;
unsigned long xfer_start = xfer_base + dev - > mem_start ;
2010-01-04 14:53:02 +03:00
if ( xfer_start + count > ei_status . rmem_end ) {
2005-04-17 02:20:36 +04:00
/* We must wrap the input move. */
int semi_count = ei_status . rmem_end - xfer_start ;
2010-06-02 11:36:20 +04:00
word_memcpy_fromcard ( skb - > data , dev - > mem_start + xfer_base ,
2010-01-04 14:53:00 +03:00
semi_count ) ;
2005-04-17 02:20:36 +04:00
count - = semi_count ;
word_memcpy_fromcard ( skb - > data + semi_count ,
2010-06-02 11:36:20 +04:00
ei_status . rmem_start , count ) ;
2010-01-04 14:53:02 +03:00
} else {
2010-06-02 11:36:20 +04:00
word_memcpy_fromcard ( skb - > data , dev - > mem_start + xfer_base ,
count ) ;
2005-04-17 02:20:36 +04:00
}
}
2010-01-04 14:53:02 +03:00
static void slow_sane_block_output ( struct net_device * dev , int count ,
const unsigned char * buf , int start_page )
2005-04-17 02:20:36 +04:00
{
long shmem = ( start_page - WD_START_PG ) < < 8 ;
2010-06-02 11:36:20 +04:00
word_memcpy_tocard ( dev - > mem_start + shmem , buf , count ) ;
2005-04-17 02:20:36 +04:00
}
2010-06-02 11:36:20 +04:00
static void word_memcpy_tocard ( unsigned long tp , const void * fp , int count )
2005-04-17 02:20:36 +04:00
{
2010-06-02 11:36:20 +04:00
volatile unsigned short * to = ( void * ) tp ;
2005-04-17 02:20:36 +04:00
const unsigned short * from = fp ;
count + + ;
2010-01-04 14:53:02 +03:00
count / = 2 ;
2005-04-17 02:20:36 +04:00
2010-01-04 14:53:02 +03:00
while ( count - - )
* to + + = * from + + ;
2005-04-17 02:20:36 +04:00
}
2010-06-02 11:36:20 +04:00
static void word_memcpy_fromcard ( void * tp , unsigned long fp , int count )
2005-04-17 02:20:36 +04:00
{
unsigned short * to = tp ;
2010-06-02 11:36:20 +04:00
const volatile unsigned short * from = ( const void * ) fp ;
2005-04-17 02:20:36 +04:00
count + + ;
2010-01-04 14:53:02 +03:00
count / = 2 ;
2005-04-17 02:20:36 +04:00
2010-01-04 14:53:02 +03:00
while ( count - - )
* to + + = * from + + ;
2005-04-17 02:20:36 +04:00
}
2006-09-13 21:24:59 +04:00