162306a36Sopenharmony_ci/* 262306a36Sopenharmony_ci Written 1997-1998 by Donald Becker. 362306a36Sopenharmony_ci 462306a36Sopenharmony_ci This software may be used and distributed according to the terms 562306a36Sopenharmony_ci of the GNU General Public License, incorporated herein by reference. 662306a36Sopenharmony_ci 762306a36Sopenharmony_ci This driver is for the 3Com ISA EtherLink XL "Corkscrew" 3c515 ethercard. 862306a36Sopenharmony_ci 962306a36Sopenharmony_ci The author may be reached as becker@scyld.com, or C/O 1062306a36Sopenharmony_ci Scyld Computing Corporation 1162306a36Sopenharmony_ci 410 Severn Ave., Suite 210 1262306a36Sopenharmony_ci Annapolis MD 21403 1362306a36Sopenharmony_ci 1462306a36Sopenharmony_ci 1562306a36Sopenharmony_ci 2000/2/2- Added support for kernel-level ISAPnP 1662306a36Sopenharmony_ci by Stephen Frost <sfrost@snowman.net> and Alessandro Zummo 1762306a36Sopenharmony_ci Cleaned up for 2.3.x/softnet by Jeff Garzik and Alan Cox. 1862306a36Sopenharmony_ci 1962306a36Sopenharmony_ci 2001/11/17 - Added ethtool support (jgarzik) 2062306a36Sopenharmony_ci 2162306a36Sopenharmony_ci 2002/10/28 - Locking updates for 2.5 (alan@lxorguk.ukuu.org.uk) 2262306a36Sopenharmony_ci 2362306a36Sopenharmony_ci*/ 2462306a36Sopenharmony_ci 2562306a36Sopenharmony_ci#define DRV_NAME "3c515" 2662306a36Sopenharmony_ci 2762306a36Sopenharmony_ci#define CORKSCREW 1 2862306a36Sopenharmony_ci 2962306a36Sopenharmony_ci/* "Knobs" that adjust features and parameters. */ 3062306a36Sopenharmony_ci/* Set the copy breakpoint for the copy-only-tiny-frames scheme. 3162306a36Sopenharmony_ci Setting to > 1512 effectively disables this feature. */ 3262306a36Sopenharmony_cistatic int rx_copybreak = 200; 3362306a36Sopenharmony_ci 3462306a36Sopenharmony_ci/* Allow setting MTU to a larger size, bypassing the normal ethernet setup. */ 3562306a36Sopenharmony_cistatic const int mtu = 1500; 3662306a36Sopenharmony_ci 3762306a36Sopenharmony_ci/* Maximum events (Rx packets, etc.) to handle at each interrupt. */ 3862306a36Sopenharmony_cistatic int max_interrupt_work = 20; 3962306a36Sopenharmony_ci 4062306a36Sopenharmony_ci/* Enable the automatic media selection code -- usually set. */ 4162306a36Sopenharmony_ci#define AUTOMEDIA 1 4262306a36Sopenharmony_ci 4362306a36Sopenharmony_ci/* Allow the use of fragment bus master transfers instead of only 4462306a36Sopenharmony_ci programmed-I/O for Vortex cards. Full-bus-master transfers are always 4562306a36Sopenharmony_ci enabled by default on Boomerang cards. If VORTEX_BUS_MASTER is defined, 4662306a36Sopenharmony_ci the feature may be turned on using 'options'. */ 4762306a36Sopenharmony_ci#define VORTEX_BUS_MASTER 4862306a36Sopenharmony_ci 4962306a36Sopenharmony_ci/* A few values that may be tweaked. */ 5062306a36Sopenharmony_ci/* Keep the ring sizes a power of two for efficiency. */ 5162306a36Sopenharmony_ci#define TX_RING_SIZE 16 5262306a36Sopenharmony_ci#define RX_RING_SIZE 16 5362306a36Sopenharmony_ci#define PKT_BUF_SZ 1536 /* Size of each temporary Rx buffer. */ 5462306a36Sopenharmony_ci 5562306a36Sopenharmony_ci#include <linux/module.h> 5662306a36Sopenharmony_ci#include <linux/isapnp.h> 5762306a36Sopenharmony_ci#include <linux/kernel.h> 5862306a36Sopenharmony_ci#include <linux/netdevice.h> 5962306a36Sopenharmony_ci#include <linux/string.h> 6062306a36Sopenharmony_ci#include <linux/errno.h> 6162306a36Sopenharmony_ci#include <linux/in.h> 6262306a36Sopenharmony_ci#include <linux/ioport.h> 6362306a36Sopenharmony_ci#include <linux/skbuff.h> 6462306a36Sopenharmony_ci#include <linux/etherdevice.h> 6562306a36Sopenharmony_ci#include <linux/interrupt.h> 6662306a36Sopenharmony_ci#include <linux/timer.h> 6762306a36Sopenharmony_ci#include <linux/ethtool.h> 6862306a36Sopenharmony_ci#include <linux/bitops.h> 6962306a36Sopenharmony_ci#include <linux/uaccess.h> 7062306a36Sopenharmony_ci 7162306a36Sopenharmony_ci#include <net/Space.h> 7262306a36Sopenharmony_ci 7362306a36Sopenharmony_ci#include <asm/io.h> 7462306a36Sopenharmony_ci#include <asm/dma.h> 7562306a36Sopenharmony_ci 7662306a36Sopenharmony_ci#define NEW_MULTICAST 7762306a36Sopenharmony_ci#include <linux/delay.h> 7862306a36Sopenharmony_ci 7962306a36Sopenharmony_ci#define MAX_UNITS 8 8062306a36Sopenharmony_ci 8162306a36Sopenharmony_ciMODULE_AUTHOR("Donald Becker <becker@scyld.com>"); 8262306a36Sopenharmony_ciMODULE_DESCRIPTION("3Com 3c515 Corkscrew driver"); 8362306a36Sopenharmony_ciMODULE_LICENSE("GPL"); 8462306a36Sopenharmony_ci 8562306a36Sopenharmony_ci/* "Knobs" for adjusting internal parameters. */ 8662306a36Sopenharmony_ci/* Put out somewhat more debugging messages. (0 - no msg, 1 minimal msgs). */ 8762306a36Sopenharmony_ci#define DRIVER_DEBUG 1 8862306a36Sopenharmony_ci/* Some values here only for performance evaluation and path-coverage 8962306a36Sopenharmony_ci debugging. */ 9062306a36Sopenharmony_cistatic int rx_nocopy, rx_copy, queued_packet; 9162306a36Sopenharmony_ci 9262306a36Sopenharmony_ci/* Number of times to check to see if the Tx FIFO has space, used in some 9362306a36Sopenharmony_ci limited cases. */ 9462306a36Sopenharmony_ci#define WAIT_TX_AVAIL 200 9562306a36Sopenharmony_ci 9662306a36Sopenharmony_ci/* Operational parameter that usually are not changed. */ 9762306a36Sopenharmony_ci#define TX_TIMEOUT ((4*HZ)/10) /* Time in jiffies before concluding Tx hung */ 9862306a36Sopenharmony_ci 9962306a36Sopenharmony_ci/* The size here is somewhat misleading: the Corkscrew also uses the ISA 10062306a36Sopenharmony_ci aliased registers at <base>+0x400. 10162306a36Sopenharmony_ci */ 10262306a36Sopenharmony_ci#define CORKSCREW_TOTAL_SIZE 0x20 10362306a36Sopenharmony_ci 10462306a36Sopenharmony_ci#ifdef DRIVER_DEBUG 10562306a36Sopenharmony_cistatic int corkscrew_debug = DRIVER_DEBUG; 10662306a36Sopenharmony_ci#else 10762306a36Sopenharmony_cistatic int corkscrew_debug = 1; 10862306a36Sopenharmony_ci#endif 10962306a36Sopenharmony_ci 11062306a36Sopenharmony_ci#define CORKSCREW_ID 10 11162306a36Sopenharmony_ci 11262306a36Sopenharmony_ci/* 11362306a36Sopenharmony_ci Theory of Operation 11462306a36Sopenharmony_ci 11562306a36Sopenharmony_ciI. Board Compatibility 11662306a36Sopenharmony_ci 11762306a36Sopenharmony_ciThis device driver is designed for the 3Com 3c515 ISA Fast EtherLink XL, 11862306a36Sopenharmony_ci3Com's ISA bus adapter for Fast Ethernet. Due to the unique I/O port layout, 11962306a36Sopenharmony_ciit's not practical to integrate this driver with the other EtherLink drivers. 12062306a36Sopenharmony_ci 12162306a36Sopenharmony_ciII. Board-specific settings 12262306a36Sopenharmony_ci 12362306a36Sopenharmony_ciThe Corkscrew has an EEPROM for configuration, but no special settings are 12462306a36Sopenharmony_cineeded for Linux. 12562306a36Sopenharmony_ci 12662306a36Sopenharmony_ciIII. Driver operation 12762306a36Sopenharmony_ci 12862306a36Sopenharmony_ciThe 3c515 series use an interface that's very similar to the 3c900 "Boomerang" 12962306a36Sopenharmony_ciPCI cards, with the bus master interface extensively modified to work with 13062306a36Sopenharmony_cithe ISA bus. 13162306a36Sopenharmony_ci 13262306a36Sopenharmony_ciThe card is capable of full-bus-master transfers with separate 13362306a36Sopenharmony_cilists of transmit and receive descriptors, similar to the AMD LANCE/PCnet, 13462306a36Sopenharmony_ciDEC Tulip and Intel Speedo3. 13562306a36Sopenharmony_ci 13662306a36Sopenharmony_ciThis driver uses a "RX_COPYBREAK" scheme rather than a fixed intermediate 13762306a36Sopenharmony_cireceive buffer. This scheme allocates full-sized skbuffs as receive 13862306a36Sopenharmony_cibuffers. The value RX_COPYBREAK is used as the copying breakpoint: it is 13962306a36Sopenharmony_cichosen to trade-off the memory wasted by passing the full-sized skbuff to 14062306a36Sopenharmony_cithe queue layer for all frames vs. the copying cost of copying a frame to a 14162306a36Sopenharmony_cicorrectly-sized skbuff. 14262306a36Sopenharmony_ci 14362306a36Sopenharmony_ci 14462306a36Sopenharmony_ciIIIC. Synchronization 14562306a36Sopenharmony_ciThe driver runs as two independent, single-threaded flows of control. One 14662306a36Sopenharmony_ciis the send-packet routine, which enforces single-threaded use by the netif 14762306a36Sopenharmony_cilayer. The other thread is the interrupt handler, which is single 14862306a36Sopenharmony_cithreaded by the hardware and other software. 14962306a36Sopenharmony_ci 15062306a36Sopenharmony_ciIV. Notes 15162306a36Sopenharmony_ci 15262306a36Sopenharmony_ciThanks to Terry Murphy of 3Com for providing documentation and a development 15362306a36Sopenharmony_ciboard. 15462306a36Sopenharmony_ci 15562306a36Sopenharmony_ciThe names "Vortex", "Boomerang" and "Corkscrew" are the internal 3Com 15662306a36Sopenharmony_ciproject names. I use these names to eliminate confusion -- 3Com product 15762306a36Sopenharmony_cinumbers and names are very similar and often confused. 15862306a36Sopenharmony_ci 15962306a36Sopenharmony_ciThe new chips support both ethernet (1.5K) and FDDI (4.5K) frame sizes! 16062306a36Sopenharmony_ciThis driver only supports ethernet frames because of the recent MTU limit 16162306a36Sopenharmony_ciof 1.5K, but the changes to support 4.5K are minimal. 16262306a36Sopenharmony_ci*/ 16362306a36Sopenharmony_ci 16462306a36Sopenharmony_ci/* Operational definitions. 16562306a36Sopenharmony_ci These are not used by other compilation units and thus are not 16662306a36Sopenharmony_ci exported in a ".h" file. 16762306a36Sopenharmony_ci 16862306a36Sopenharmony_ci First the windows. There are eight register windows, with the command 16962306a36Sopenharmony_ci and status registers available in each. 17062306a36Sopenharmony_ci */ 17162306a36Sopenharmony_ci#define EL3WINDOW(win_num) outw(SelectWindow + (win_num), ioaddr + EL3_CMD) 17262306a36Sopenharmony_ci#define EL3_CMD 0x0e 17362306a36Sopenharmony_ci#define EL3_STATUS 0x0e 17462306a36Sopenharmony_ci 17562306a36Sopenharmony_ci/* The top five bits written to EL3_CMD are a command, the lower 17662306a36Sopenharmony_ci 11 bits are the parameter, if applicable. 17762306a36Sopenharmony_ci Note that 11 parameters bits was fine for ethernet, but the new chips 17862306a36Sopenharmony_ci can handle FDDI length frames (~4500 octets) and now parameters count 17962306a36Sopenharmony_ci 32-bit 'Dwords' rather than octets. */ 18062306a36Sopenharmony_ci 18162306a36Sopenharmony_cienum corkscrew_cmd { 18262306a36Sopenharmony_ci TotalReset = 0 << 11, SelectWindow = 1 << 11, StartCoax = 2 << 11, 18362306a36Sopenharmony_ci RxDisable = 3 << 11, RxEnable = 4 << 11, RxReset = 5 << 11, 18462306a36Sopenharmony_ci UpStall = 6 << 11, UpUnstall = (6 << 11) + 1, DownStall = (6 << 11) + 2, 18562306a36Sopenharmony_ci DownUnstall = (6 << 11) + 3, RxDiscard = 8 << 11, TxEnable = 9 << 11, 18662306a36Sopenharmony_ci TxDisable = 10 << 11, TxReset = 11 << 11, FakeIntr = 12 << 11, 18762306a36Sopenharmony_ci AckIntr = 13 << 11, SetIntrEnb = 14 << 11, SetStatusEnb = 15 << 11, 18862306a36Sopenharmony_ci SetRxFilter = 16 << 11, SetRxThreshold = 17 << 11, 18962306a36Sopenharmony_ci SetTxThreshold = 18 << 11, SetTxStart = 19 << 11, StartDMAUp = 20 << 11, 19062306a36Sopenharmony_ci StartDMADown = (20 << 11) + 1, StatsEnable = 21 << 11, 19162306a36Sopenharmony_ci StatsDisable = 22 << 11, StopCoax = 23 << 11, 19262306a36Sopenharmony_ci}; 19362306a36Sopenharmony_ci 19462306a36Sopenharmony_ci/* The SetRxFilter command accepts the following classes: */ 19562306a36Sopenharmony_cienum RxFilter { 19662306a36Sopenharmony_ci RxStation = 1, RxMulticast = 2, RxBroadcast = 4, RxProm = 8 19762306a36Sopenharmony_ci}; 19862306a36Sopenharmony_ci 19962306a36Sopenharmony_ci/* Bits in the general status register. */ 20062306a36Sopenharmony_cienum corkscrew_status { 20162306a36Sopenharmony_ci IntLatch = 0x0001, AdapterFailure = 0x0002, TxComplete = 0x0004, 20262306a36Sopenharmony_ci TxAvailable = 0x0008, RxComplete = 0x0010, RxEarly = 0x0020, 20362306a36Sopenharmony_ci IntReq = 0x0040, StatsFull = 0x0080, 20462306a36Sopenharmony_ci DMADone = 1 << 8, DownComplete = 1 << 9, UpComplete = 1 << 10, 20562306a36Sopenharmony_ci DMAInProgress = 1 << 11, /* DMA controller is still busy. */ 20662306a36Sopenharmony_ci CmdInProgress = 1 << 12, /* EL3_CMD is still busy. */ 20762306a36Sopenharmony_ci}; 20862306a36Sopenharmony_ci 20962306a36Sopenharmony_ci/* Register window 1 offsets, the window used in normal operation. 21062306a36Sopenharmony_ci On the Corkscrew this window is always mapped at offsets 0x10-0x1f. */ 21162306a36Sopenharmony_cienum Window1 { 21262306a36Sopenharmony_ci TX_FIFO = 0x10, RX_FIFO = 0x10, RxErrors = 0x14, 21362306a36Sopenharmony_ci RxStatus = 0x18, Timer = 0x1A, TxStatus = 0x1B, 21462306a36Sopenharmony_ci TxFree = 0x1C, /* Remaining free bytes in Tx buffer. */ 21562306a36Sopenharmony_ci}; 21662306a36Sopenharmony_cienum Window0 { 21762306a36Sopenharmony_ci Wn0IRQ = 0x08, 21862306a36Sopenharmony_ci#if defined(CORKSCREW) 21962306a36Sopenharmony_ci Wn0EepromCmd = 0x200A, /* Corkscrew EEPROM command register. */ 22062306a36Sopenharmony_ci Wn0EepromData = 0x200C, /* Corkscrew EEPROM results register. */ 22162306a36Sopenharmony_ci#else 22262306a36Sopenharmony_ci Wn0EepromCmd = 10, /* Window 0: EEPROM command register. */ 22362306a36Sopenharmony_ci Wn0EepromData = 12, /* Window 0: EEPROM results register. */ 22462306a36Sopenharmony_ci#endif 22562306a36Sopenharmony_ci}; 22662306a36Sopenharmony_cienum Win0_EEPROM_bits { 22762306a36Sopenharmony_ci EEPROM_Read = 0x80, EEPROM_WRITE = 0x40, EEPROM_ERASE = 0xC0, 22862306a36Sopenharmony_ci EEPROM_EWENB = 0x30, /* Enable erasing/writing for 10 msec. */ 22962306a36Sopenharmony_ci EEPROM_EWDIS = 0x00, /* Disable EWENB before 10 msec timeout. */ 23062306a36Sopenharmony_ci}; 23162306a36Sopenharmony_ci 23262306a36Sopenharmony_ci/* EEPROM locations. */ 23362306a36Sopenharmony_cienum eeprom_offset { 23462306a36Sopenharmony_ci PhysAddr01 = 0, PhysAddr23 = 1, PhysAddr45 = 2, ModelID = 3, 23562306a36Sopenharmony_ci EtherLink3ID = 7, 23662306a36Sopenharmony_ci}; 23762306a36Sopenharmony_ci 23862306a36Sopenharmony_cienum Window3 { /* Window 3: MAC/config bits. */ 23962306a36Sopenharmony_ci Wn3_Config = 0, Wn3_MAC_Ctrl = 6, Wn3_Options = 8, 24062306a36Sopenharmony_ci}; 24162306a36Sopenharmony_cienum wn3_config { 24262306a36Sopenharmony_ci Ram_size = 7, 24362306a36Sopenharmony_ci Ram_width = 8, 24462306a36Sopenharmony_ci Ram_speed = 0x30, 24562306a36Sopenharmony_ci Rom_size = 0xc0, 24662306a36Sopenharmony_ci Ram_split_shift = 16, 24762306a36Sopenharmony_ci Ram_split = 3 << Ram_split_shift, 24862306a36Sopenharmony_ci Xcvr_shift = 20, 24962306a36Sopenharmony_ci Xcvr = 7 << Xcvr_shift, 25062306a36Sopenharmony_ci Autoselect = 0x1000000, 25162306a36Sopenharmony_ci}; 25262306a36Sopenharmony_ci 25362306a36Sopenharmony_cienum Window4 { 25462306a36Sopenharmony_ci Wn4_NetDiag = 6, Wn4_Media = 10, /* Window 4: Xcvr/media bits. */ 25562306a36Sopenharmony_ci}; 25662306a36Sopenharmony_cienum Win4_Media_bits { 25762306a36Sopenharmony_ci Media_SQE = 0x0008, /* Enable SQE error counting for AUI. */ 25862306a36Sopenharmony_ci Media_10TP = 0x00C0, /* Enable link beat and jabber for 10baseT. */ 25962306a36Sopenharmony_ci Media_Lnk = 0x0080, /* Enable just link beat for 100TX/100FX. */ 26062306a36Sopenharmony_ci Media_LnkBeat = 0x0800, 26162306a36Sopenharmony_ci}; 26262306a36Sopenharmony_cienum Window7 { /* Window 7: Bus Master control. */ 26362306a36Sopenharmony_ci Wn7_MasterAddr = 0, Wn7_MasterLen = 6, Wn7_MasterStatus = 12, 26462306a36Sopenharmony_ci}; 26562306a36Sopenharmony_ci 26662306a36Sopenharmony_ci/* Boomerang-style bus master control registers. Note ISA aliases! */ 26762306a36Sopenharmony_cienum MasterCtrl { 26862306a36Sopenharmony_ci PktStatus = 0x400, DownListPtr = 0x404, FragAddr = 0x408, FragLen = 26962306a36Sopenharmony_ci 0x40c, 27062306a36Sopenharmony_ci TxFreeThreshold = 0x40f, UpPktStatus = 0x410, UpListPtr = 0x418, 27162306a36Sopenharmony_ci}; 27262306a36Sopenharmony_ci 27362306a36Sopenharmony_ci/* The Rx and Tx descriptor lists. 27462306a36Sopenharmony_ci Caution Alpha hackers: these types are 32 bits! Note also the 8 byte 27562306a36Sopenharmony_ci alignment contraint on tx_ring[] and rx_ring[]. */ 27662306a36Sopenharmony_cistruct boom_rx_desc { 27762306a36Sopenharmony_ci u32 next; 27862306a36Sopenharmony_ci s32 status; 27962306a36Sopenharmony_ci u32 addr; 28062306a36Sopenharmony_ci s32 length; 28162306a36Sopenharmony_ci}; 28262306a36Sopenharmony_ci 28362306a36Sopenharmony_ci/* Values for the Rx status entry. */ 28462306a36Sopenharmony_cienum rx_desc_status { 28562306a36Sopenharmony_ci RxDComplete = 0x00008000, RxDError = 0x4000, 28662306a36Sopenharmony_ci /* See boomerang_rx() for actual error bits */ 28762306a36Sopenharmony_ci}; 28862306a36Sopenharmony_ci 28962306a36Sopenharmony_cistruct boom_tx_desc { 29062306a36Sopenharmony_ci u32 next; 29162306a36Sopenharmony_ci s32 status; 29262306a36Sopenharmony_ci u32 addr; 29362306a36Sopenharmony_ci s32 length; 29462306a36Sopenharmony_ci}; 29562306a36Sopenharmony_ci 29662306a36Sopenharmony_cistruct corkscrew_private { 29762306a36Sopenharmony_ci const char *product_name; 29862306a36Sopenharmony_ci struct list_head list; 29962306a36Sopenharmony_ci struct net_device *our_dev; 30062306a36Sopenharmony_ci /* The Rx and Tx rings are here to keep them quad-word-aligned. */ 30162306a36Sopenharmony_ci struct boom_rx_desc rx_ring[RX_RING_SIZE]; 30262306a36Sopenharmony_ci struct boom_tx_desc tx_ring[TX_RING_SIZE]; 30362306a36Sopenharmony_ci /* The addresses of transmit- and receive-in-place skbuffs. */ 30462306a36Sopenharmony_ci struct sk_buff *rx_skbuff[RX_RING_SIZE]; 30562306a36Sopenharmony_ci struct sk_buff *tx_skbuff[TX_RING_SIZE]; 30662306a36Sopenharmony_ci unsigned int cur_rx, cur_tx; /* The next free ring entry */ 30762306a36Sopenharmony_ci unsigned int dirty_rx, dirty_tx;/* The ring entries to be free()ed. */ 30862306a36Sopenharmony_ci struct sk_buff *tx_skb; /* Packet being eaten by bus master ctrl. */ 30962306a36Sopenharmony_ci struct timer_list timer; /* Media selection timer. */ 31062306a36Sopenharmony_ci int capabilities ; /* Adapter capabilities word. */ 31162306a36Sopenharmony_ci int options; /* User-settable misc. driver options. */ 31262306a36Sopenharmony_ci int last_rx_packets; /* For media autoselection. */ 31362306a36Sopenharmony_ci unsigned int available_media:8, /* From Wn3_Options */ 31462306a36Sopenharmony_ci media_override:3, /* Passed-in media type. */ 31562306a36Sopenharmony_ci default_media:3, /* Read from the EEPROM. */ 31662306a36Sopenharmony_ci full_duplex:1, autoselect:1, bus_master:1, /* Vortex can only do a fragment bus-m. */ 31762306a36Sopenharmony_ci full_bus_master_tx:1, full_bus_master_rx:1, /* Boomerang */ 31862306a36Sopenharmony_ci tx_full:1; 31962306a36Sopenharmony_ci spinlock_t lock; 32062306a36Sopenharmony_ci struct device *dev; 32162306a36Sopenharmony_ci}; 32262306a36Sopenharmony_ci 32362306a36Sopenharmony_ci/* The action to take with a media selection timer tick. 32462306a36Sopenharmony_ci Note that we deviate from the 3Com order by checking 10base2 before AUI. 32562306a36Sopenharmony_ci */ 32662306a36Sopenharmony_cienum xcvr_types { 32762306a36Sopenharmony_ci XCVR_10baseT = 0, XCVR_AUI, XCVR_10baseTOnly, XCVR_10base2, XCVR_100baseTx, 32862306a36Sopenharmony_ci XCVR_100baseFx, XCVR_MII = 6, XCVR_Default = 8, 32962306a36Sopenharmony_ci}; 33062306a36Sopenharmony_ci 33162306a36Sopenharmony_cistatic struct media_table { 33262306a36Sopenharmony_ci char *name; 33362306a36Sopenharmony_ci unsigned int media_bits:16, /* Bits to set in Wn4_Media register. */ 33462306a36Sopenharmony_ci mask:8, /* The transceiver-present bit in Wn3_Config. */ 33562306a36Sopenharmony_ci next:8; /* The media type to try next. */ 33662306a36Sopenharmony_ci short wait; /* Time before we check media status. */ 33762306a36Sopenharmony_ci} media_tbl[] = { 33862306a36Sopenharmony_ci { "10baseT", Media_10TP, 0x08, XCVR_10base2, (14 * HZ) / 10 }, 33962306a36Sopenharmony_ci { "10Mbs AUI", Media_SQE, 0x20, XCVR_Default, (1 * HZ) / 10}, 34062306a36Sopenharmony_ci { "undefined", 0, 0x80, XCVR_10baseT, 10000}, 34162306a36Sopenharmony_ci { "10base2", 0, 0x10, XCVR_AUI, (1 * HZ) / 10}, 34262306a36Sopenharmony_ci { "100baseTX", Media_Lnk, 0x02, XCVR_100baseFx, (14 * HZ) / 10}, 34362306a36Sopenharmony_ci { "100baseFX", Media_Lnk, 0x04, XCVR_MII, (14 * HZ) / 10}, 34462306a36Sopenharmony_ci { "MII", 0, 0x40, XCVR_10baseT, 3 * HZ}, 34562306a36Sopenharmony_ci { "undefined", 0, 0x01, XCVR_10baseT, 10000}, 34662306a36Sopenharmony_ci { "Default", 0, 0xFF, XCVR_10baseT, 10000}, 34762306a36Sopenharmony_ci}; 34862306a36Sopenharmony_ci 34962306a36Sopenharmony_ci#ifdef __ISAPNP__ 35062306a36Sopenharmony_cistatic struct isapnp_device_id corkscrew_isapnp_adapters[] = { 35162306a36Sopenharmony_ci { ISAPNP_ANY_ID, ISAPNP_ANY_ID, 35262306a36Sopenharmony_ci ISAPNP_VENDOR('T', 'C', 'M'), ISAPNP_FUNCTION(0x5051), 35362306a36Sopenharmony_ci (long) "3Com Fast EtherLink ISA" }, 35462306a36Sopenharmony_ci { } /* terminate list */ 35562306a36Sopenharmony_ci}; 35662306a36Sopenharmony_ci 35762306a36Sopenharmony_ciMODULE_DEVICE_TABLE(isapnp, corkscrew_isapnp_adapters); 35862306a36Sopenharmony_ci 35962306a36Sopenharmony_cistatic int nopnp; 36062306a36Sopenharmony_ci#endif /* __ISAPNP__ */ 36162306a36Sopenharmony_ci 36262306a36Sopenharmony_cistatic struct net_device *corkscrew_scan(int unit); 36362306a36Sopenharmony_cistatic int corkscrew_setup(struct net_device *dev, int ioaddr, 36462306a36Sopenharmony_ci struct pnp_dev *idev, int card_number); 36562306a36Sopenharmony_cistatic int corkscrew_open(struct net_device *dev); 36662306a36Sopenharmony_cistatic void corkscrew_timer(struct timer_list *t); 36762306a36Sopenharmony_cistatic netdev_tx_t corkscrew_start_xmit(struct sk_buff *skb, 36862306a36Sopenharmony_ci struct net_device *dev); 36962306a36Sopenharmony_cistatic int corkscrew_rx(struct net_device *dev); 37062306a36Sopenharmony_cistatic void corkscrew_timeout(struct net_device *dev, unsigned int txqueue); 37162306a36Sopenharmony_cistatic int boomerang_rx(struct net_device *dev); 37262306a36Sopenharmony_cistatic irqreturn_t corkscrew_interrupt(int irq, void *dev_id); 37362306a36Sopenharmony_cistatic int corkscrew_close(struct net_device *dev); 37462306a36Sopenharmony_cistatic void update_stats(int addr, struct net_device *dev); 37562306a36Sopenharmony_cistatic struct net_device_stats *corkscrew_get_stats(struct net_device *dev); 37662306a36Sopenharmony_cistatic void set_rx_mode(struct net_device *dev); 37762306a36Sopenharmony_cistatic const struct ethtool_ops netdev_ethtool_ops; 37862306a36Sopenharmony_ci 37962306a36Sopenharmony_ci 38062306a36Sopenharmony_ci/* 38162306a36Sopenharmony_ci Unfortunately maximizing the shared code between the integrated and 38262306a36Sopenharmony_ci module version of the driver results in a complicated set of initialization 38362306a36Sopenharmony_ci procedures. 38462306a36Sopenharmony_ci init_module() -- modules / tc59x_init() -- built-in 38562306a36Sopenharmony_ci The wrappers for corkscrew_scan() 38662306a36Sopenharmony_ci corkscrew_scan() The common routine that scans for PCI and EISA cards 38762306a36Sopenharmony_ci corkscrew_found_device() Allocate a device structure when we find a card. 38862306a36Sopenharmony_ci Different versions exist for modules and built-in. 38962306a36Sopenharmony_ci corkscrew_probe1() Fill in the device structure -- this is separated 39062306a36Sopenharmony_ci so that the modules code can put it in dev->init. 39162306a36Sopenharmony_ci*/ 39262306a36Sopenharmony_ci/* This driver uses 'options' to pass the media type, full-duplex flag, etc. */ 39362306a36Sopenharmony_ci/* Note: this is the only limit on the number of cards supported!! */ 39462306a36Sopenharmony_cistatic int options[MAX_UNITS] = { -1, -1, -1, -1, -1, -1, -1, -1, }; 39562306a36Sopenharmony_ci 39662306a36Sopenharmony_ci#ifdef MODULE 39762306a36Sopenharmony_cistatic int debug = -1; 39862306a36Sopenharmony_ci 39962306a36Sopenharmony_cimodule_param(debug, int, 0); 40062306a36Sopenharmony_cimodule_param_array(options, int, NULL, 0); 40162306a36Sopenharmony_cimodule_param(rx_copybreak, int, 0); 40262306a36Sopenharmony_cimodule_param(max_interrupt_work, int, 0); 40362306a36Sopenharmony_ciMODULE_PARM_DESC(debug, "3c515 debug level (0-6)"); 40462306a36Sopenharmony_ciMODULE_PARM_DESC(options, "3c515: Bits 0-2: media type, bit 3: full duplex, bit 4: bus mastering"); 40562306a36Sopenharmony_ciMODULE_PARM_DESC(rx_copybreak, "3c515 copy breakpoint for copy-only-tiny-frames"); 40662306a36Sopenharmony_ciMODULE_PARM_DESC(max_interrupt_work, "3c515 maximum events handled per interrupt"); 40762306a36Sopenharmony_ci 40862306a36Sopenharmony_ci/* A list of all installed Vortex devices, for removing the driver module. */ 40962306a36Sopenharmony_ci/* we will need locking (and refcounting) if we ever use it for more */ 41062306a36Sopenharmony_cistatic LIST_HEAD(root_corkscrew_dev); 41162306a36Sopenharmony_ci 41262306a36Sopenharmony_cistatic int corkscrew_init_module(void) 41362306a36Sopenharmony_ci{ 41462306a36Sopenharmony_ci int found = 0; 41562306a36Sopenharmony_ci if (debug >= 0) 41662306a36Sopenharmony_ci corkscrew_debug = debug; 41762306a36Sopenharmony_ci while (corkscrew_scan(-1)) 41862306a36Sopenharmony_ci found++; 41962306a36Sopenharmony_ci return found ? 0 : -ENODEV; 42062306a36Sopenharmony_ci} 42162306a36Sopenharmony_cimodule_init(corkscrew_init_module); 42262306a36Sopenharmony_ci 42362306a36Sopenharmony_ci#else 42462306a36Sopenharmony_cistruct net_device *tc515_probe(int unit) 42562306a36Sopenharmony_ci{ 42662306a36Sopenharmony_ci struct net_device *dev = corkscrew_scan(unit); 42762306a36Sopenharmony_ci 42862306a36Sopenharmony_ci if (!dev) 42962306a36Sopenharmony_ci return ERR_PTR(-ENODEV); 43062306a36Sopenharmony_ci 43162306a36Sopenharmony_ci return dev; 43262306a36Sopenharmony_ci} 43362306a36Sopenharmony_ci#endif /* not MODULE */ 43462306a36Sopenharmony_ci 43562306a36Sopenharmony_cistatic int check_device(unsigned ioaddr) 43662306a36Sopenharmony_ci{ 43762306a36Sopenharmony_ci int timer; 43862306a36Sopenharmony_ci 43962306a36Sopenharmony_ci if (!request_region(ioaddr, CORKSCREW_TOTAL_SIZE, "3c515")) 44062306a36Sopenharmony_ci return 0; 44162306a36Sopenharmony_ci /* Check the resource configuration for a matching ioaddr. */ 44262306a36Sopenharmony_ci if ((inw(ioaddr + 0x2002) & 0x1f0) != (ioaddr & 0x1f0)) { 44362306a36Sopenharmony_ci release_region(ioaddr, CORKSCREW_TOTAL_SIZE); 44462306a36Sopenharmony_ci return 0; 44562306a36Sopenharmony_ci } 44662306a36Sopenharmony_ci /* Verify by reading the device ID from the EEPROM. */ 44762306a36Sopenharmony_ci outw(EEPROM_Read + 7, ioaddr + Wn0EepromCmd); 44862306a36Sopenharmony_ci /* Pause for at least 162 us. for the read to take place. */ 44962306a36Sopenharmony_ci for (timer = 4; timer >= 0; timer--) { 45062306a36Sopenharmony_ci udelay(162); 45162306a36Sopenharmony_ci if ((inw(ioaddr + Wn0EepromCmd) & 0x0200) == 0) 45262306a36Sopenharmony_ci break; 45362306a36Sopenharmony_ci } 45462306a36Sopenharmony_ci if (inw(ioaddr + Wn0EepromData) != 0x6d50) { 45562306a36Sopenharmony_ci release_region(ioaddr, CORKSCREW_TOTAL_SIZE); 45662306a36Sopenharmony_ci return 0; 45762306a36Sopenharmony_ci } 45862306a36Sopenharmony_ci return 1; 45962306a36Sopenharmony_ci} 46062306a36Sopenharmony_ci 46162306a36Sopenharmony_cistatic void cleanup_card(struct net_device *dev) 46262306a36Sopenharmony_ci{ 46362306a36Sopenharmony_ci struct corkscrew_private *vp = netdev_priv(dev); 46462306a36Sopenharmony_ci list_del_init(&vp->list); 46562306a36Sopenharmony_ci if (dev->dma) 46662306a36Sopenharmony_ci free_dma(dev->dma); 46762306a36Sopenharmony_ci outw(TotalReset, dev->base_addr + EL3_CMD); 46862306a36Sopenharmony_ci release_region(dev->base_addr, CORKSCREW_TOTAL_SIZE); 46962306a36Sopenharmony_ci if (vp->dev) 47062306a36Sopenharmony_ci pnp_device_detach(to_pnp_dev(vp->dev)); 47162306a36Sopenharmony_ci} 47262306a36Sopenharmony_ci 47362306a36Sopenharmony_cistatic struct net_device *corkscrew_scan(int unit) 47462306a36Sopenharmony_ci{ 47562306a36Sopenharmony_ci struct net_device *dev; 47662306a36Sopenharmony_ci static int cards_found = 0; 47762306a36Sopenharmony_ci static int ioaddr; 47862306a36Sopenharmony_ci int err; 47962306a36Sopenharmony_ci#ifdef __ISAPNP__ 48062306a36Sopenharmony_ci short i; 48162306a36Sopenharmony_ci static int pnp_cards; 48262306a36Sopenharmony_ci#endif 48362306a36Sopenharmony_ci 48462306a36Sopenharmony_ci dev = alloc_etherdev(sizeof(struct corkscrew_private)); 48562306a36Sopenharmony_ci if (!dev) 48662306a36Sopenharmony_ci return ERR_PTR(-ENOMEM); 48762306a36Sopenharmony_ci 48862306a36Sopenharmony_ci if (unit >= 0) { 48962306a36Sopenharmony_ci sprintf(dev->name, "eth%d", unit); 49062306a36Sopenharmony_ci netdev_boot_setup_check(dev); 49162306a36Sopenharmony_ci } 49262306a36Sopenharmony_ci 49362306a36Sopenharmony_ci#ifdef __ISAPNP__ 49462306a36Sopenharmony_ci if(nopnp == 1) 49562306a36Sopenharmony_ci goto no_pnp; 49662306a36Sopenharmony_ci for(i=0; corkscrew_isapnp_adapters[i].vendor != 0; i++) { 49762306a36Sopenharmony_ci struct pnp_dev *idev = NULL; 49862306a36Sopenharmony_ci int irq; 49962306a36Sopenharmony_ci while((idev = pnp_find_dev(NULL, 50062306a36Sopenharmony_ci corkscrew_isapnp_adapters[i].vendor, 50162306a36Sopenharmony_ci corkscrew_isapnp_adapters[i].function, 50262306a36Sopenharmony_ci idev))) { 50362306a36Sopenharmony_ci 50462306a36Sopenharmony_ci if (pnp_device_attach(idev) < 0) 50562306a36Sopenharmony_ci continue; 50662306a36Sopenharmony_ci if (pnp_activate_dev(idev) < 0) { 50762306a36Sopenharmony_ci pr_warn("pnp activate failed (out of resources?)\n"); 50862306a36Sopenharmony_ci pnp_device_detach(idev); 50962306a36Sopenharmony_ci continue; 51062306a36Sopenharmony_ci } 51162306a36Sopenharmony_ci if (!pnp_port_valid(idev, 0) || !pnp_irq_valid(idev, 0)) { 51262306a36Sopenharmony_ci pnp_device_detach(idev); 51362306a36Sopenharmony_ci continue; 51462306a36Sopenharmony_ci } 51562306a36Sopenharmony_ci ioaddr = pnp_port_start(idev, 0); 51662306a36Sopenharmony_ci irq = pnp_irq(idev, 0); 51762306a36Sopenharmony_ci if (!check_device(ioaddr)) { 51862306a36Sopenharmony_ci pnp_device_detach(idev); 51962306a36Sopenharmony_ci continue; 52062306a36Sopenharmony_ci } 52162306a36Sopenharmony_ci if(corkscrew_debug) 52262306a36Sopenharmony_ci pr_debug("ISAPNP reports %s at i/o 0x%x, irq %d\n", 52362306a36Sopenharmony_ci (char*) corkscrew_isapnp_adapters[i].driver_data, ioaddr, irq); 52462306a36Sopenharmony_ci pr_info("3c515 Resource configuration register %#4.4x, DCR %4.4x.\n", 52562306a36Sopenharmony_ci inl(ioaddr + 0x2002), inw(ioaddr + 0x2000)); 52662306a36Sopenharmony_ci /* irq = inw(ioaddr + 0x2002) & 15; */ /* Use the irq from isapnp */ 52762306a36Sopenharmony_ci SET_NETDEV_DEV(dev, &idev->dev); 52862306a36Sopenharmony_ci pnp_cards++; 52962306a36Sopenharmony_ci err = corkscrew_setup(dev, ioaddr, idev, cards_found++); 53062306a36Sopenharmony_ci if (!err) 53162306a36Sopenharmony_ci return dev; 53262306a36Sopenharmony_ci cleanup_card(dev); 53362306a36Sopenharmony_ci } 53462306a36Sopenharmony_ci } 53562306a36Sopenharmony_cino_pnp: 53662306a36Sopenharmony_ci#endif /* __ISAPNP__ */ 53762306a36Sopenharmony_ci 53862306a36Sopenharmony_ci /* Check all locations on the ISA bus -- evil! */ 53962306a36Sopenharmony_ci for (ioaddr = 0x100; ioaddr < 0x400; ioaddr += 0x20) { 54062306a36Sopenharmony_ci if (!check_device(ioaddr)) 54162306a36Sopenharmony_ci continue; 54262306a36Sopenharmony_ci 54362306a36Sopenharmony_ci pr_info("3c515 Resource configuration register %#4.4x, DCR %4.4x.\n", 54462306a36Sopenharmony_ci inl(ioaddr + 0x2002), inw(ioaddr + 0x2000)); 54562306a36Sopenharmony_ci err = corkscrew_setup(dev, ioaddr, NULL, cards_found++); 54662306a36Sopenharmony_ci if (!err) 54762306a36Sopenharmony_ci return dev; 54862306a36Sopenharmony_ci cleanup_card(dev); 54962306a36Sopenharmony_ci } 55062306a36Sopenharmony_ci free_netdev(dev); 55162306a36Sopenharmony_ci return NULL; 55262306a36Sopenharmony_ci} 55362306a36Sopenharmony_ci 55462306a36Sopenharmony_ci 55562306a36Sopenharmony_cistatic const struct net_device_ops netdev_ops = { 55662306a36Sopenharmony_ci .ndo_open = corkscrew_open, 55762306a36Sopenharmony_ci .ndo_stop = corkscrew_close, 55862306a36Sopenharmony_ci .ndo_start_xmit = corkscrew_start_xmit, 55962306a36Sopenharmony_ci .ndo_tx_timeout = corkscrew_timeout, 56062306a36Sopenharmony_ci .ndo_get_stats = corkscrew_get_stats, 56162306a36Sopenharmony_ci .ndo_set_rx_mode = set_rx_mode, 56262306a36Sopenharmony_ci .ndo_set_mac_address = eth_mac_addr, 56362306a36Sopenharmony_ci .ndo_validate_addr = eth_validate_addr, 56462306a36Sopenharmony_ci}; 56562306a36Sopenharmony_ci 56662306a36Sopenharmony_ci 56762306a36Sopenharmony_cistatic int corkscrew_setup(struct net_device *dev, int ioaddr, 56862306a36Sopenharmony_ci struct pnp_dev *idev, int card_number) 56962306a36Sopenharmony_ci{ 57062306a36Sopenharmony_ci struct corkscrew_private *vp = netdev_priv(dev); 57162306a36Sopenharmony_ci unsigned int eeprom[0x40], checksum = 0; /* EEPROM contents */ 57262306a36Sopenharmony_ci __be16 addr[ETH_ALEN / 2]; 57362306a36Sopenharmony_ci int i; 57462306a36Sopenharmony_ci int irq; 57562306a36Sopenharmony_ci 57662306a36Sopenharmony_ci#ifdef __ISAPNP__ 57762306a36Sopenharmony_ci if (idev) { 57862306a36Sopenharmony_ci irq = pnp_irq(idev, 0); 57962306a36Sopenharmony_ci vp->dev = &idev->dev; 58062306a36Sopenharmony_ci } else { 58162306a36Sopenharmony_ci irq = inw(ioaddr + 0x2002) & 15; 58262306a36Sopenharmony_ci } 58362306a36Sopenharmony_ci#else 58462306a36Sopenharmony_ci irq = inw(ioaddr + 0x2002) & 15; 58562306a36Sopenharmony_ci#endif 58662306a36Sopenharmony_ci 58762306a36Sopenharmony_ci dev->base_addr = ioaddr; 58862306a36Sopenharmony_ci dev->irq = irq; 58962306a36Sopenharmony_ci dev->dma = inw(ioaddr + 0x2000) & 7; 59062306a36Sopenharmony_ci vp->product_name = "3c515"; 59162306a36Sopenharmony_ci vp->options = dev->mem_start; 59262306a36Sopenharmony_ci vp->our_dev = dev; 59362306a36Sopenharmony_ci 59462306a36Sopenharmony_ci if (!vp->options) { 59562306a36Sopenharmony_ci if (card_number >= MAX_UNITS) 59662306a36Sopenharmony_ci vp->options = -1; 59762306a36Sopenharmony_ci else 59862306a36Sopenharmony_ci vp->options = options[card_number]; 59962306a36Sopenharmony_ci } 60062306a36Sopenharmony_ci 60162306a36Sopenharmony_ci if (vp->options >= 0) { 60262306a36Sopenharmony_ci vp->media_override = vp->options & 7; 60362306a36Sopenharmony_ci if (vp->media_override == 2) 60462306a36Sopenharmony_ci vp->media_override = 0; 60562306a36Sopenharmony_ci vp->full_duplex = (vp->options & 8) ? 1 : 0; 60662306a36Sopenharmony_ci vp->bus_master = (vp->options & 16) ? 1 : 0; 60762306a36Sopenharmony_ci } else { 60862306a36Sopenharmony_ci vp->media_override = 7; 60962306a36Sopenharmony_ci vp->full_duplex = 0; 61062306a36Sopenharmony_ci vp->bus_master = 0; 61162306a36Sopenharmony_ci } 61262306a36Sopenharmony_ci#ifdef MODULE 61362306a36Sopenharmony_ci list_add(&vp->list, &root_corkscrew_dev); 61462306a36Sopenharmony_ci#endif 61562306a36Sopenharmony_ci 61662306a36Sopenharmony_ci pr_info("%s: 3Com %s at %#3x,", dev->name, vp->product_name, ioaddr); 61762306a36Sopenharmony_ci 61862306a36Sopenharmony_ci spin_lock_init(&vp->lock); 61962306a36Sopenharmony_ci 62062306a36Sopenharmony_ci timer_setup(&vp->timer, corkscrew_timer, 0); 62162306a36Sopenharmony_ci 62262306a36Sopenharmony_ci /* Read the station address from the EEPROM. */ 62362306a36Sopenharmony_ci EL3WINDOW(0); 62462306a36Sopenharmony_ci for (i = 0; i < 0x18; i++) { 62562306a36Sopenharmony_ci int timer; 62662306a36Sopenharmony_ci outw(EEPROM_Read + i, ioaddr + Wn0EepromCmd); 62762306a36Sopenharmony_ci /* Pause for at least 162 us. for the read to take place. */ 62862306a36Sopenharmony_ci for (timer = 4; timer >= 0; timer--) { 62962306a36Sopenharmony_ci udelay(162); 63062306a36Sopenharmony_ci if ((inw(ioaddr + Wn0EepromCmd) & 0x0200) == 0) 63162306a36Sopenharmony_ci break; 63262306a36Sopenharmony_ci } 63362306a36Sopenharmony_ci eeprom[i] = inw(ioaddr + Wn0EepromData); 63462306a36Sopenharmony_ci checksum ^= eeprom[i]; 63562306a36Sopenharmony_ci if (i < 3) 63662306a36Sopenharmony_ci addr[i] = htons(eeprom[i]); 63762306a36Sopenharmony_ci } 63862306a36Sopenharmony_ci eth_hw_addr_set(dev, (u8 *)addr); 63962306a36Sopenharmony_ci checksum = (checksum ^ (checksum >> 8)) & 0xff; 64062306a36Sopenharmony_ci if (checksum != 0x00) 64162306a36Sopenharmony_ci pr_cont(" ***INVALID CHECKSUM %4.4x*** ", checksum); 64262306a36Sopenharmony_ci pr_cont(" %pM", dev->dev_addr); 64362306a36Sopenharmony_ci if (eeprom[16] == 0x11c7) { /* Corkscrew */ 64462306a36Sopenharmony_ci if (request_dma(dev->dma, "3c515")) { 64562306a36Sopenharmony_ci pr_cont(", DMA %d allocation failed", dev->dma); 64662306a36Sopenharmony_ci dev->dma = 0; 64762306a36Sopenharmony_ci } else 64862306a36Sopenharmony_ci pr_cont(", DMA %d", dev->dma); 64962306a36Sopenharmony_ci } 65062306a36Sopenharmony_ci pr_cont(", IRQ %d\n", dev->irq); 65162306a36Sopenharmony_ci /* Tell them about an invalid IRQ. */ 65262306a36Sopenharmony_ci if (corkscrew_debug && (dev->irq <= 0 || dev->irq > 15)) 65362306a36Sopenharmony_ci pr_warn(" *** Warning: this IRQ is unlikely to work! ***\n"); 65462306a36Sopenharmony_ci 65562306a36Sopenharmony_ci { 65662306a36Sopenharmony_ci static const char * const ram_split[] = { 65762306a36Sopenharmony_ci "5:3", "3:1", "1:1", "3:5" 65862306a36Sopenharmony_ci }; 65962306a36Sopenharmony_ci __u32 config; 66062306a36Sopenharmony_ci EL3WINDOW(3); 66162306a36Sopenharmony_ci vp->available_media = inw(ioaddr + Wn3_Options); 66262306a36Sopenharmony_ci config = inl(ioaddr + Wn3_Config); 66362306a36Sopenharmony_ci if (corkscrew_debug > 1) 66462306a36Sopenharmony_ci pr_info(" Internal config register is %4.4x, transceivers %#x.\n", 66562306a36Sopenharmony_ci config, inw(ioaddr + Wn3_Options)); 66662306a36Sopenharmony_ci pr_info(" %dK %s-wide RAM %s Rx:Tx split, %s%s interface.\n", 66762306a36Sopenharmony_ci 8 << config & Ram_size, 66862306a36Sopenharmony_ci config & Ram_width ? "word" : "byte", 66962306a36Sopenharmony_ci ram_split[(config & Ram_split) >> Ram_split_shift], 67062306a36Sopenharmony_ci config & Autoselect ? "autoselect/" : "", 67162306a36Sopenharmony_ci media_tbl[(config & Xcvr) >> Xcvr_shift].name); 67262306a36Sopenharmony_ci vp->default_media = (config & Xcvr) >> Xcvr_shift; 67362306a36Sopenharmony_ci vp->autoselect = config & Autoselect ? 1 : 0; 67462306a36Sopenharmony_ci dev->if_port = vp->default_media; 67562306a36Sopenharmony_ci } 67662306a36Sopenharmony_ci if (vp->media_override != 7) { 67762306a36Sopenharmony_ci pr_info(" Media override to transceiver type %d (%s).\n", 67862306a36Sopenharmony_ci vp->media_override, 67962306a36Sopenharmony_ci media_tbl[vp->media_override].name); 68062306a36Sopenharmony_ci dev->if_port = vp->media_override; 68162306a36Sopenharmony_ci } 68262306a36Sopenharmony_ci 68362306a36Sopenharmony_ci vp->capabilities = eeprom[16]; 68462306a36Sopenharmony_ci vp->full_bus_master_tx = (vp->capabilities & 0x20) ? 1 : 0; 68562306a36Sopenharmony_ci /* Rx is broken at 10mbps, so we always disable it. */ 68662306a36Sopenharmony_ci /* vp->full_bus_master_rx = 0; */ 68762306a36Sopenharmony_ci vp->full_bus_master_rx = (vp->capabilities & 0x20) ? 1 : 0; 68862306a36Sopenharmony_ci 68962306a36Sopenharmony_ci /* The 3c51x-specific entries in the device structure. */ 69062306a36Sopenharmony_ci dev->netdev_ops = &netdev_ops; 69162306a36Sopenharmony_ci dev->watchdog_timeo = (400 * HZ) / 1000; 69262306a36Sopenharmony_ci dev->ethtool_ops = &netdev_ethtool_ops; 69362306a36Sopenharmony_ci 69462306a36Sopenharmony_ci return register_netdev(dev); 69562306a36Sopenharmony_ci} 69662306a36Sopenharmony_ci 69762306a36Sopenharmony_ci 69862306a36Sopenharmony_cistatic int corkscrew_open(struct net_device *dev) 69962306a36Sopenharmony_ci{ 70062306a36Sopenharmony_ci int ioaddr = dev->base_addr; 70162306a36Sopenharmony_ci struct corkscrew_private *vp = netdev_priv(dev); 70262306a36Sopenharmony_ci bool armtimer = false; 70362306a36Sopenharmony_ci __u32 config; 70462306a36Sopenharmony_ci int i; 70562306a36Sopenharmony_ci 70662306a36Sopenharmony_ci /* Before initializing select the active media port. */ 70762306a36Sopenharmony_ci EL3WINDOW(3); 70862306a36Sopenharmony_ci if (vp->full_duplex) 70962306a36Sopenharmony_ci outb(0x20, ioaddr + Wn3_MAC_Ctrl); /* Set the full-duplex bit. */ 71062306a36Sopenharmony_ci config = inl(ioaddr + Wn3_Config); 71162306a36Sopenharmony_ci 71262306a36Sopenharmony_ci if (vp->media_override != 7) { 71362306a36Sopenharmony_ci if (corkscrew_debug > 1) 71462306a36Sopenharmony_ci pr_info("%s: Media override to transceiver %d (%s).\n", 71562306a36Sopenharmony_ci dev->name, vp->media_override, 71662306a36Sopenharmony_ci media_tbl[vp->media_override].name); 71762306a36Sopenharmony_ci dev->if_port = vp->media_override; 71862306a36Sopenharmony_ci } else if (vp->autoselect) { 71962306a36Sopenharmony_ci /* Find first available media type, starting with 100baseTx. */ 72062306a36Sopenharmony_ci dev->if_port = 4; 72162306a36Sopenharmony_ci while (!(vp->available_media & media_tbl[dev->if_port].mask)) 72262306a36Sopenharmony_ci dev->if_port = media_tbl[dev->if_port].next; 72362306a36Sopenharmony_ci 72462306a36Sopenharmony_ci if (corkscrew_debug > 1) 72562306a36Sopenharmony_ci pr_debug("%s: Initial media type %s.\n", 72662306a36Sopenharmony_ci dev->name, media_tbl[dev->if_port].name); 72762306a36Sopenharmony_ci armtimer = true; 72862306a36Sopenharmony_ci } else 72962306a36Sopenharmony_ci dev->if_port = vp->default_media; 73062306a36Sopenharmony_ci 73162306a36Sopenharmony_ci config = (config & ~Xcvr) | (dev->if_port << Xcvr_shift); 73262306a36Sopenharmony_ci outl(config, ioaddr + Wn3_Config); 73362306a36Sopenharmony_ci 73462306a36Sopenharmony_ci if (corkscrew_debug > 1) { 73562306a36Sopenharmony_ci pr_debug("%s: corkscrew_open() InternalConfig %8.8x.\n", 73662306a36Sopenharmony_ci dev->name, config); 73762306a36Sopenharmony_ci } 73862306a36Sopenharmony_ci 73962306a36Sopenharmony_ci outw(TxReset, ioaddr + EL3_CMD); 74062306a36Sopenharmony_ci for (i = 20; i >= 0; i--) 74162306a36Sopenharmony_ci if (!(inw(ioaddr + EL3_STATUS) & CmdInProgress)) 74262306a36Sopenharmony_ci break; 74362306a36Sopenharmony_ci 74462306a36Sopenharmony_ci outw(RxReset, ioaddr + EL3_CMD); 74562306a36Sopenharmony_ci /* Wait a few ticks for the RxReset command to complete. */ 74662306a36Sopenharmony_ci for (i = 20; i >= 0; i--) 74762306a36Sopenharmony_ci if (!(inw(ioaddr + EL3_STATUS) & CmdInProgress)) 74862306a36Sopenharmony_ci break; 74962306a36Sopenharmony_ci 75062306a36Sopenharmony_ci outw(SetStatusEnb | 0x00, ioaddr + EL3_CMD); 75162306a36Sopenharmony_ci 75262306a36Sopenharmony_ci /* Use the now-standard shared IRQ implementation. */ 75362306a36Sopenharmony_ci if (vp->capabilities == 0x11c7) { 75462306a36Sopenharmony_ci /* Corkscrew: Cannot share ISA resources. */ 75562306a36Sopenharmony_ci if (dev->irq == 0 || 75662306a36Sopenharmony_ci dev->dma == 0 || 75762306a36Sopenharmony_ci request_irq(dev->irq, corkscrew_interrupt, 0, 75862306a36Sopenharmony_ci vp->product_name, dev)) 75962306a36Sopenharmony_ci return -EAGAIN; 76062306a36Sopenharmony_ci enable_dma(dev->dma); 76162306a36Sopenharmony_ci set_dma_mode(dev->dma, DMA_MODE_CASCADE); 76262306a36Sopenharmony_ci } else if (request_irq(dev->irq, corkscrew_interrupt, IRQF_SHARED, 76362306a36Sopenharmony_ci vp->product_name, dev)) { 76462306a36Sopenharmony_ci return -EAGAIN; 76562306a36Sopenharmony_ci } 76662306a36Sopenharmony_ci 76762306a36Sopenharmony_ci if (armtimer) 76862306a36Sopenharmony_ci mod_timer(&vp->timer, jiffies + media_tbl[dev->if_port].wait); 76962306a36Sopenharmony_ci 77062306a36Sopenharmony_ci if (corkscrew_debug > 1) { 77162306a36Sopenharmony_ci EL3WINDOW(4); 77262306a36Sopenharmony_ci pr_debug("%s: corkscrew_open() irq %d media status %4.4x.\n", 77362306a36Sopenharmony_ci dev->name, dev->irq, inw(ioaddr + Wn4_Media)); 77462306a36Sopenharmony_ci } 77562306a36Sopenharmony_ci 77662306a36Sopenharmony_ci /* Set the station address and mask in window 2 each time opened. */ 77762306a36Sopenharmony_ci EL3WINDOW(2); 77862306a36Sopenharmony_ci for (i = 0; i < 6; i++) 77962306a36Sopenharmony_ci outb(dev->dev_addr[i], ioaddr + i); 78062306a36Sopenharmony_ci for (; i < 12; i += 2) 78162306a36Sopenharmony_ci outw(0, ioaddr + i); 78262306a36Sopenharmony_ci 78362306a36Sopenharmony_ci if (dev->if_port == 3) 78462306a36Sopenharmony_ci /* Start the thinnet transceiver. We should really wait 50ms... */ 78562306a36Sopenharmony_ci outw(StartCoax, ioaddr + EL3_CMD); 78662306a36Sopenharmony_ci EL3WINDOW(4); 78762306a36Sopenharmony_ci outw((inw(ioaddr + Wn4_Media) & ~(Media_10TP | Media_SQE)) | 78862306a36Sopenharmony_ci media_tbl[dev->if_port].media_bits, ioaddr + Wn4_Media); 78962306a36Sopenharmony_ci 79062306a36Sopenharmony_ci /* Switch to the stats window, and clear all stats by reading. */ 79162306a36Sopenharmony_ci outw(StatsDisable, ioaddr + EL3_CMD); 79262306a36Sopenharmony_ci EL3WINDOW(6); 79362306a36Sopenharmony_ci for (i = 0; i < 10; i++) 79462306a36Sopenharmony_ci inb(ioaddr + i); 79562306a36Sopenharmony_ci inw(ioaddr + 10); 79662306a36Sopenharmony_ci inw(ioaddr + 12); 79762306a36Sopenharmony_ci /* New: On the Vortex we must also clear the BadSSD counter. */ 79862306a36Sopenharmony_ci EL3WINDOW(4); 79962306a36Sopenharmony_ci inb(ioaddr + 12); 80062306a36Sopenharmony_ci /* ..and on the Boomerang we enable the extra statistics bits. */ 80162306a36Sopenharmony_ci outw(0x0040, ioaddr + Wn4_NetDiag); 80262306a36Sopenharmony_ci 80362306a36Sopenharmony_ci /* Switch to register set 7 for normal use. */ 80462306a36Sopenharmony_ci EL3WINDOW(7); 80562306a36Sopenharmony_ci 80662306a36Sopenharmony_ci if (vp->full_bus_master_rx) { /* Boomerang bus master. */ 80762306a36Sopenharmony_ci vp->cur_rx = vp->dirty_rx = 0; 80862306a36Sopenharmony_ci if (corkscrew_debug > 2) 80962306a36Sopenharmony_ci pr_debug("%s: Filling in the Rx ring.\n", dev->name); 81062306a36Sopenharmony_ci for (i = 0; i < RX_RING_SIZE; i++) { 81162306a36Sopenharmony_ci struct sk_buff *skb; 81262306a36Sopenharmony_ci if (i < (RX_RING_SIZE - 1)) 81362306a36Sopenharmony_ci vp->rx_ring[i].next = 81462306a36Sopenharmony_ci isa_virt_to_bus(&vp->rx_ring[i + 1]); 81562306a36Sopenharmony_ci else 81662306a36Sopenharmony_ci vp->rx_ring[i].next = 0; 81762306a36Sopenharmony_ci vp->rx_ring[i].status = 0; /* Clear complete bit. */ 81862306a36Sopenharmony_ci vp->rx_ring[i].length = PKT_BUF_SZ | 0x80000000; 81962306a36Sopenharmony_ci skb = netdev_alloc_skb(dev, PKT_BUF_SZ); 82062306a36Sopenharmony_ci vp->rx_skbuff[i] = skb; 82162306a36Sopenharmony_ci if (skb == NULL) 82262306a36Sopenharmony_ci break; /* Bad news! */ 82362306a36Sopenharmony_ci skb_reserve(skb, 2); /* Align IP on 16 byte boundaries */ 82462306a36Sopenharmony_ci vp->rx_ring[i].addr = isa_virt_to_bus(skb->data); 82562306a36Sopenharmony_ci } 82662306a36Sopenharmony_ci if (i != 0) 82762306a36Sopenharmony_ci vp->rx_ring[i - 1].next = 82862306a36Sopenharmony_ci isa_virt_to_bus(&vp->rx_ring[0]); /* Wrap the ring. */ 82962306a36Sopenharmony_ci outl(isa_virt_to_bus(&vp->rx_ring[0]), ioaddr + UpListPtr); 83062306a36Sopenharmony_ci } 83162306a36Sopenharmony_ci if (vp->full_bus_master_tx) { /* Boomerang bus master Tx. */ 83262306a36Sopenharmony_ci vp->cur_tx = vp->dirty_tx = 0; 83362306a36Sopenharmony_ci outb(PKT_BUF_SZ >> 8, ioaddr + TxFreeThreshold); /* Room for a packet. */ 83462306a36Sopenharmony_ci /* Clear the Tx ring. */ 83562306a36Sopenharmony_ci for (i = 0; i < TX_RING_SIZE; i++) 83662306a36Sopenharmony_ci vp->tx_skbuff[i] = NULL; 83762306a36Sopenharmony_ci outl(0, ioaddr + DownListPtr); 83862306a36Sopenharmony_ci } 83962306a36Sopenharmony_ci /* Set receiver mode: presumably accept b-case and phys addr only. */ 84062306a36Sopenharmony_ci set_rx_mode(dev); 84162306a36Sopenharmony_ci outw(StatsEnable, ioaddr + EL3_CMD); /* Turn on statistics. */ 84262306a36Sopenharmony_ci 84362306a36Sopenharmony_ci netif_start_queue(dev); 84462306a36Sopenharmony_ci 84562306a36Sopenharmony_ci outw(RxEnable, ioaddr + EL3_CMD); /* Enable the receiver. */ 84662306a36Sopenharmony_ci outw(TxEnable, ioaddr + EL3_CMD); /* Enable transmitter. */ 84762306a36Sopenharmony_ci /* Allow status bits to be seen. */ 84862306a36Sopenharmony_ci outw(SetStatusEnb | AdapterFailure | IntReq | StatsFull | 84962306a36Sopenharmony_ci (vp->full_bus_master_tx ? DownComplete : TxAvailable) | 85062306a36Sopenharmony_ci (vp->full_bus_master_rx ? UpComplete : RxComplete) | 85162306a36Sopenharmony_ci (vp->bus_master ? DMADone : 0), ioaddr + EL3_CMD); 85262306a36Sopenharmony_ci /* Ack all pending events, and set active indicator mask. */ 85362306a36Sopenharmony_ci outw(AckIntr | IntLatch | TxAvailable | RxEarly | IntReq, 85462306a36Sopenharmony_ci ioaddr + EL3_CMD); 85562306a36Sopenharmony_ci outw(SetIntrEnb | IntLatch | TxAvailable | RxComplete | StatsFull 85662306a36Sopenharmony_ci | (vp->bus_master ? DMADone : 0) | UpComplete | DownComplete, 85762306a36Sopenharmony_ci ioaddr + EL3_CMD); 85862306a36Sopenharmony_ci 85962306a36Sopenharmony_ci return 0; 86062306a36Sopenharmony_ci} 86162306a36Sopenharmony_ci 86262306a36Sopenharmony_cistatic void corkscrew_timer(struct timer_list *t) 86362306a36Sopenharmony_ci{ 86462306a36Sopenharmony_ci#ifdef AUTOMEDIA 86562306a36Sopenharmony_ci struct corkscrew_private *vp = from_timer(vp, t, timer); 86662306a36Sopenharmony_ci struct net_device *dev = vp->our_dev; 86762306a36Sopenharmony_ci int ioaddr = dev->base_addr; 86862306a36Sopenharmony_ci unsigned long flags; 86962306a36Sopenharmony_ci int ok = 0; 87062306a36Sopenharmony_ci 87162306a36Sopenharmony_ci if (corkscrew_debug > 1) 87262306a36Sopenharmony_ci pr_debug("%s: Media selection timer tick happened, %s.\n", 87362306a36Sopenharmony_ci dev->name, media_tbl[dev->if_port].name); 87462306a36Sopenharmony_ci 87562306a36Sopenharmony_ci spin_lock_irqsave(&vp->lock, flags); 87662306a36Sopenharmony_ci 87762306a36Sopenharmony_ci { 87862306a36Sopenharmony_ci int old_window = inw(ioaddr + EL3_CMD) >> 13; 87962306a36Sopenharmony_ci int media_status; 88062306a36Sopenharmony_ci EL3WINDOW(4); 88162306a36Sopenharmony_ci media_status = inw(ioaddr + Wn4_Media); 88262306a36Sopenharmony_ci switch (dev->if_port) { 88362306a36Sopenharmony_ci case 0: 88462306a36Sopenharmony_ci case 4: 88562306a36Sopenharmony_ci case 5: /* 10baseT, 100baseTX, 100baseFX */ 88662306a36Sopenharmony_ci if (media_status & Media_LnkBeat) { 88762306a36Sopenharmony_ci ok = 1; 88862306a36Sopenharmony_ci if (corkscrew_debug > 1) 88962306a36Sopenharmony_ci pr_debug("%s: Media %s has link beat, %x.\n", 89062306a36Sopenharmony_ci dev->name, 89162306a36Sopenharmony_ci media_tbl[dev->if_port].name, 89262306a36Sopenharmony_ci media_status); 89362306a36Sopenharmony_ci } else if (corkscrew_debug > 1) 89462306a36Sopenharmony_ci pr_debug("%s: Media %s is has no link beat, %x.\n", 89562306a36Sopenharmony_ci dev->name, 89662306a36Sopenharmony_ci media_tbl[dev->if_port].name, 89762306a36Sopenharmony_ci media_status); 89862306a36Sopenharmony_ci 89962306a36Sopenharmony_ci break; 90062306a36Sopenharmony_ci default: /* Other media types handled by Tx timeouts. */ 90162306a36Sopenharmony_ci if (corkscrew_debug > 1) 90262306a36Sopenharmony_ci pr_debug("%s: Media %s is has no indication, %x.\n", 90362306a36Sopenharmony_ci dev->name, 90462306a36Sopenharmony_ci media_tbl[dev->if_port].name, 90562306a36Sopenharmony_ci media_status); 90662306a36Sopenharmony_ci ok = 1; 90762306a36Sopenharmony_ci } 90862306a36Sopenharmony_ci if (!ok) { 90962306a36Sopenharmony_ci __u32 config; 91062306a36Sopenharmony_ci 91162306a36Sopenharmony_ci do { 91262306a36Sopenharmony_ci dev->if_port = 91362306a36Sopenharmony_ci media_tbl[dev->if_port].next; 91462306a36Sopenharmony_ci } 91562306a36Sopenharmony_ci while (!(vp->available_media & media_tbl[dev->if_port].mask)); 91662306a36Sopenharmony_ci 91762306a36Sopenharmony_ci if (dev->if_port == 8) { /* Go back to default. */ 91862306a36Sopenharmony_ci dev->if_port = vp->default_media; 91962306a36Sopenharmony_ci if (corkscrew_debug > 1) 92062306a36Sopenharmony_ci pr_debug("%s: Media selection failing, using default %s port.\n", 92162306a36Sopenharmony_ci dev->name, 92262306a36Sopenharmony_ci media_tbl[dev->if_port].name); 92362306a36Sopenharmony_ci } else { 92462306a36Sopenharmony_ci if (corkscrew_debug > 1) 92562306a36Sopenharmony_ci pr_debug("%s: Media selection failed, now trying %s port.\n", 92662306a36Sopenharmony_ci dev->name, 92762306a36Sopenharmony_ci media_tbl[dev->if_port].name); 92862306a36Sopenharmony_ci vp->timer.expires = jiffies + media_tbl[dev->if_port].wait; 92962306a36Sopenharmony_ci add_timer(&vp->timer); 93062306a36Sopenharmony_ci } 93162306a36Sopenharmony_ci outw((media_status & ~(Media_10TP | Media_SQE)) | 93262306a36Sopenharmony_ci media_tbl[dev->if_port].media_bits, 93362306a36Sopenharmony_ci ioaddr + Wn4_Media); 93462306a36Sopenharmony_ci 93562306a36Sopenharmony_ci EL3WINDOW(3); 93662306a36Sopenharmony_ci config = inl(ioaddr + Wn3_Config); 93762306a36Sopenharmony_ci config = (config & ~Xcvr) | (dev->if_port << Xcvr_shift); 93862306a36Sopenharmony_ci outl(config, ioaddr + Wn3_Config); 93962306a36Sopenharmony_ci 94062306a36Sopenharmony_ci outw(dev->if_port == 3 ? StartCoax : StopCoax, 94162306a36Sopenharmony_ci ioaddr + EL3_CMD); 94262306a36Sopenharmony_ci } 94362306a36Sopenharmony_ci EL3WINDOW(old_window); 94462306a36Sopenharmony_ci } 94562306a36Sopenharmony_ci 94662306a36Sopenharmony_ci spin_unlock_irqrestore(&vp->lock, flags); 94762306a36Sopenharmony_ci if (corkscrew_debug > 1) 94862306a36Sopenharmony_ci pr_debug("%s: Media selection timer finished, %s.\n", 94962306a36Sopenharmony_ci dev->name, media_tbl[dev->if_port].name); 95062306a36Sopenharmony_ci 95162306a36Sopenharmony_ci#endif /* AUTOMEDIA */ 95262306a36Sopenharmony_ci} 95362306a36Sopenharmony_ci 95462306a36Sopenharmony_cistatic void corkscrew_timeout(struct net_device *dev, unsigned int txqueue) 95562306a36Sopenharmony_ci{ 95662306a36Sopenharmony_ci int i; 95762306a36Sopenharmony_ci struct corkscrew_private *vp = netdev_priv(dev); 95862306a36Sopenharmony_ci int ioaddr = dev->base_addr; 95962306a36Sopenharmony_ci 96062306a36Sopenharmony_ci pr_warn("%s: transmit timed out, tx_status %2.2x status %4.4x\n", 96162306a36Sopenharmony_ci dev->name, inb(ioaddr + TxStatus), 96262306a36Sopenharmony_ci inw(ioaddr + EL3_STATUS)); 96362306a36Sopenharmony_ci /* Slight code bloat to be user friendly. */ 96462306a36Sopenharmony_ci if ((inb(ioaddr + TxStatus) & 0x88) == 0x88) 96562306a36Sopenharmony_ci pr_warn("%s: Transmitter encountered 16 collisions -- network cable problem?\n", 96662306a36Sopenharmony_ci dev->name); 96762306a36Sopenharmony_ci#ifndef final_version 96862306a36Sopenharmony_ci pr_debug(" Flags; bus-master %d, full %d; dirty %d current %d.\n", 96962306a36Sopenharmony_ci vp->full_bus_master_tx, vp->tx_full, vp->dirty_tx, 97062306a36Sopenharmony_ci vp->cur_tx); 97162306a36Sopenharmony_ci pr_debug(" Down list %8.8x vs. %p.\n", inl(ioaddr + DownListPtr), 97262306a36Sopenharmony_ci &vp->tx_ring[0]); 97362306a36Sopenharmony_ci for (i = 0; i < TX_RING_SIZE; i++) { 97462306a36Sopenharmony_ci pr_debug(" %d: %p length %8.8x status %8.8x\n", i, 97562306a36Sopenharmony_ci &vp->tx_ring[i], 97662306a36Sopenharmony_ci vp->tx_ring[i].length, vp->tx_ring[i].status); 97762306a36Sopenharmony_ci } 97862306a36Sopenharmony_ci#endif 97962306a36Sopenharmony_ci /* Issue TX_RESET and TX_START commands. */ 98062306a36Sopenharmony_ci outw(TxReset, ioaddr + EL3_CMD); 98162306a36Sopenharmony_ci for (i = 20; i >= 0; i--) 98262306a36Sopenharmony_ci if (!(inw(ioaddr + EL3_STATUS) & CmdInProgress)) 98362306a36Sopenharmony_ci break; 98462306a36Sopenharmony_ci outw(TxEnable, ioaddr + EL3_CMD); 98562306a36Sopenharmony_ci netif_trans_update(dev); /* prevent tx timeout */ 98662306a36Sopenharmony_ci dev->stats.tx_errors++; 98762306a36Sopenharmony_ci dev->stats.tx_dropped++; 98862306a36Sopenharmony_ci netif_wake_queue(dev); 98962306a36Sopenharmony_ci} 99062306a36Sopenharmony_ci 99162306a36Sopenharmony_cistatic netdev_tx_t corkscrew_start_xmit(struct sk_buff *skb, 99262306a36Sopenharmony_ci struct net_device *dev) 99362306a36Sopenharmony_ci{ 99462306a36Sopenharmony_ci struct corkscrew_private *vp = netdev_priv(dev); 99562306a36Sopenharmony_ci int ioaddr = dev->base_addr; 99662306a36Sopenharmony_ci 99762306a36Sopenharmony_ci /* Block a timer-based transmit from overlapping. */ 99862306a36Sopenharmony_ci 99962306a36Sopenharmony_ci netif_stop_queue(dev); 100062306a36Sopenharmony_ci 100162306a36Sopenharmony_ci if (vp->full_bus_master_tx) { /* BOOMERANG bus-master */ 100262306a36Sopenharmony_ci /* Calculate the next Tx descriptor entry. */ 100362306a36Sopenharmony_ci int entry = vp->cur_tx % TX_RING_SIZE; 100462306a36Sopenharmony_ci struct boom_tx_desc *prev_entry; 100562306a36Sopenharmony_ci unsigned long flags; 100662306a36Sopenharmony_ci int i; 100762306a36Sopenharmony_ci 100862306a36Sopenharmony_ci if (vp->tx_full) /* No room to transmit with */ 100962306a36Sopenharmony_ci return NETDEV_TX_BUSY; 101062306a36Sopenharmony_ci if (vp->cur_tx != 0) 101162306a36Sopenharmony_ci prev_entry = &vp->tx_ring[(vp->cur_tx - 1) % TX_RING_SIZE]; 101262306a36Sopenharmony_ci else 101362306a36Sopenharmony_ci prev_entry = NULL; 101462306a36Sopenharmony_ci if (corkscrew_debug > 3) 101562306a36Sopenharmony_ci pr_debug("%s: Trying to send a packet, Tx index %d.\n", 101662306a36Sopenharmony_ci dev->name, vp->cur_tx); 101762306a36Sopenharmony_ci /* vp->tx_full = 1; */ 101862306a36Sopenharmony_ci vp->tx_skbuff[entry] = skb; 101962306a36Sopenharmony_ci vp->tx_ring[entry].next = 0; 102062306a36Sopenharmony_ci vp->tx_ring[entry].addr = isa_virt_to_bus(skb->data); 102162306a36Sopenharmony_ci vp->tx_ring[entry].length = skb->len | 0x80000000; 102262306a36Sopenharmony_ci vp->tx_ring[entry].status = skb->len | 0x80000000; 102362306a36Sopenharmony_ci 102462306a36Sopenharmony_ci spin_lock_irqsave(&vp->lock, flags); 102562306a36Sopenharmony_ci outw(DownStall, ioaddr + EL3_CMD); 102662306a36Sopenharmony_ci /* Wait for the stall to complete. */ 102762306a36Sopenharmony_ci for (i = 20; i >= 0; i--) 102862306a36Sopenharmony_ci if ((inw(ioaddr + EL3_STATUS) & CmdInProgress) == 0) 102962306a36Sopenharmony_ci break; 103062306a36Sopenharmony_ci if (prev_entry) 103162306a36Sopenharmony_ci prev_entry->next = isa_virt_to_bus(&vp->tx_ring[entry]); 103262306a36Sopenharmony_ci if (inl(ioaddr + DownListPtr) == 0) { 103362306a36Sopenharmony_ci outl(isa_virt_to_bus(&vp->tx_ring[entry]), 103462306a36Sopenharmony_ci ioaddr + DownListPtr); 103562306a36Sopenharmony_ci queued_packet++; 103662306a36Sopenharmony_ci } 103762306a36Sopenharmony_ci outw(DownUnstall, ioaddr + EL3_CMD); 103862306a36Sopenharmony_ci spin_unlock_irqrestore(&vp->lock, flags); 103962306a36Sopenharmony_ci 104062306a36Sopenharmony_ci vp->cur_tx++; 104162306a36Sopenharmony_ci if (vp->cur_tx - vp->dirty_tx > TX_RING_SIZE - 1) 104262306a36Sopenharmony_ci vp->tx_full = 1; 104362306a36Sopenharmony_ci else { /* Clear previous interrupt enable. */ 104462306a36Sopenharmony_ci if (prev_entry) 104562306a36Sopenharmony_ci prev_entry->status &= ~0x80000000; 104662306a36Sopenharmony_ci netif_wake_queue(dev); 104762306a36Sopenharmony_ci } 104862306a36Sopenharmony_ci return NETDEV_TX_OK; 104962306a36Sopenharmony_ci } 105062306a36Sopenharmony_ci /* Put out the doubleword header... */ 105162306a36Sopenharmony_ci outl(skb->len, ioaddr + TX_FIFO); 105262306a36Sopenharmony_ci dev->stats.tx_bytes += skb->len; 105362306a36Sopenharmony_ci#ifdef VORTEX_BUS_MASTER 105462306a36Sopenharmony_ci if (vp->bus_master) { 105562306a36Sopenharmony_ci /* Set the bus-master controller to transfer the packet. */ 105662306a36Sopenharmony_ci outl(isa_virt_to_bus(skb->data), ioaddr + Wn7_MasterAddr); 105762306a36Sopenharmony_ci outw((skb->len + 3) & ~3, ioaddr + Wn7_MasterLen); 105862306a36Sopenharmony_ci vp->tx_skb = skb; 105962306a36Sopenharmony_ci outw(StartDMADown, ioaddr + EL3_CMD); 106062306a36Sopenharmony_ci /* queue will be woken at the DMADone interrupt. */ 106162306a36Sopenharmony_ci } else { 106262306a36Sopenharmony_ci /* ... and the packet rounded to a doubleword. */ 106362306a36Sopenharmony_ci outsl(ioaddr + TX_FIFO, skb->data, (skb->len + 3) >> 2); 106462306a36Sopenharmony_ci dev_kfree_skb(skb); 106562306a36Sopenharmony_ci if (inw(ioaddr + TxFree) > 1536) { 106662306a36Sopenharmony_ci netif_wake_queue(dev); 106762306a36Sopenharmony_ci } else 106862306a36Sopenharmony_ci /* Interrupt us when the FIFO has room for max-sized packet. */ 106962306a36Sopenharmony_ci outw(SetTxThreshold + (1536 >> 2), 107062306a36Sopenharmony_ci ioaddr + EL3_CMD); 107162306a36Sopenharmony_ci } 107262306a36Sopenharmony_ci#else 107362306a36Sopenharmony_ci /* ... and the packet rounded to a doubleword. */ 107462306a36Sopenharmony_ci outsl(ioaddr + TX_FIFO, skb->data, (skb->len + 3) >> 2); 107562306a36Sopenharmony_ci dev_kfree_skb(skb); 107662306a36Sopenharmony_ci if (inw(ioaddr + TxFree) > 1536) { 107762306a36Sopenharmony_ci netif_wake_queue(dev); 107862306a36Sopenharmony_ci } else 107962306a36Sopenharmony_ci /* Interrupt us when the FIFO has room for max-sized packet. */ 108062306a36Sopenharmony_ci outw(SetTxThreshold + (1536 >> 2), ioaddr + EL3_CMD); 108162306a36Sopenharmony_ci#endif /* bus master */ 108262306a36Sopenharmony_ci 108362306a36Sopenharmony_ci 108462306a36Sopenharmony_ci /* Clear the Tx status stack. */ 108562306a36Sopenharmony_ci { 108662306a36Sopenharmony_ci short tx_status; 108762306a36Sopenharmony_ci int i = 4; 108862306a36Sopenharmony_ci 108962306a36Sopenharmony_ci while (--i > 0 && (tx_status = inb(ioaddr + TxStatus)) > 0) { 109062306a36Sopenharmony_ci if (tx_status & 0x3C) { /* A Tx-disabling error occurred. */ 109162306a36Sopenharmony_ci if (corkscrew_debug > 2) 109262306a36Sopenharmony_ci pr_debug("%s: Tx error, status %2.2x.\n", 109362306a36Sopenharmony_ci dev->name, tx_status); 109462306a36Sopenharmony_ci if (tx_status & 0x04) 109562306a36Sopenharmony_ci dev->stats.tx_fifo_errors++; 109662306a36Sopenharmony_ci if (tx_status & 0x38) 109762306a36Sopenharmony_ci dev->stats.tx_aborted_errors++; 109862306a36Sopenharmony_ci if (tx_status & 0x30) { 109962306a36Sopenharmony_ci int j; 110062306a36Sopenharmony_ci outw(TxReset, ioaddr + EL3_CMD); 110162306a36Sopenharmony_ci for (j = 20; j >= 0; j--) 110262306a36Sopenharmony_ci if (!(inw(ioaddr + EL3_STATUS) & CmdInProgress)) 110362306a36Sopenharmony_ci break; 110462306a36Sopenharmony_ci } 110562306a36Sopenharmony_ci outw(TxEnable, ioaddr + EL3_CMD); 110662306a36Sopenharmony_ci } 110762306a36Sopenharmony_ci outb(0x00, ioaddr + TxStatus); /* Pop the status stack. */ 110862306a36Sopenharmony_ci } 110962306a36Sopenharmony_ci } 111062306a36Sopenharmony_ci return NETDEV_TX_OK; 111162306a36Sopenharmony_ci} 111262306a36Sopenharmony_ci 111362306a36Sopenharmony_ci/* The interrupt handler does all of the Rx thread work and cleans up 111462306a36Sopenharmony_ci after the Tx thread. */ 111562306a36Sopenharmony_ci 111662306a36Sopenharmony_cistatic irqreturn_t corkscrew_interrupt(int irq, void *dev_id) 111762306a36Sopenharmony_ci{ 111862306a36Sopenharmony_ci /* Use the now-standard shared IRQ implementation. */ 111962306a36Sopenharmony_ci struct net_device *dev = dev_id; 112062306a36Sopenharmony_ci struct corkscrew_private *lp = netdev_priv(dev); 112162306a36Sopenharmony_ci int ioaddr, status; 112262306a36Sopenharmony_ci int latency; 112362306a36Sopenharmony_ci int i = max_interrupt_work; 112462306a36Sopenharmony_ci 112562306a36Sopenharmony_ci ioaddr = dev->base_addr; 112662306a36Sopenharmony_ci latency = inb(ioaddr + Timer); 112762306a36Sopenharmony_ci 112862306a36Sopenharmony_ci spin_lock(&lp->lock); 112962306a36Sopenharmony_ci 113062306a36Sopenharmony_ci status = inw(ioaddr + EL3_STATUS); 113162306a36Sopenharmony_ci 113262306a36Sopenharmony_ci if (corkscrew_debug > 4) 113362306a36Sopenharmony_ci pr_debug("%s: interrupt, status %4.4x, timer %d.\n", 113462306a36Sopenharmony_ci dev->name, status, latency); 113562306a36Sopenharmony_ci if ((status & 0xE000) != 0xE000) { 113662306a36Sopenharmony_ci static int donedidthis; 113762306a36Sopenharmony_ci /* Some interrupt controllers store a bogus interrupt from boot-time. 113862306a36Sopenharmony_ci Ignore a single early interrupt, but don't hang the machine for 113962306a36Sopenharmony_ci other interrupt problems. */ 114062306a36Sopenharmony_ci if (donedidthis++ > 100) { 114162306a36Sopenharmony_ci pr_err("%s: Bogus interrupt, bailing. Status %4.4x, start=%d.\n", 114262306a36Sopenharmony_ci dev->name, status, netif_running(dev)); 114362306a36Sopenharmony_ci free_irq(dev->irq, dev); 114462306a36Sopenharmony_ci dev->irq = -1; 114562306a36Sopenharmony_ci } 114662306a36Sopenharmony_ci } 114762306a36Sopenharmony_ci 114862306a36Sopenharmony_ci do { 114962306a36Sopenharmony_ci if (corkscrew_debug > 5) 115062306a36Sopenharmony_ci pr_debug("%s: In interrupt loop, status %4.4x.\n", 115162306a36Sopenharmony_ci dev->name, status); 115262306a36Sopenharmony_ci if (status & RxComplete) 115362306a36Sopenharmony_ci corkscrew_rx(dev); 115462306a36Sopenharmony_ci 115562306a36Sopenharmony_ci if (status & TxAvailable) { 115662306a36Sopenharmony_ci if (corkscrew_debug > 5) 115762306a36Sopenharmony_ci pr_debug(" TX room bit was handled.\n"); 115862306a36Sopenharmony_ci /* There's room in the FIFO for a full-sized packet. */ 115962306a36Sopenharmony_ci outw(AckIntr | TxAvailable, ioaddr + EL3_CMD); 116062306a36Sopenharmony_ci netif_wake_queue(dev); 116162306a36Sopenharmony_ci } 116262306a36Sopenharmony_ci if (status & DownComplete) { 116362306a36Sopenharmony_ci unsigned int dirty_tx = lp->dirty_tx; 116462306a36Sopenharmony_ci 116562306a36Sopenharmony_ci while (lp->cur_tx - dirty_tx > 0) { 116662306a36Sopenharmony_ci int entry = dirty_tx % TX_RING_SIZE; 116762306a36Sopenharmony_ci if (inl(ioaddr + DownListPtr) == isa_virt_to_bus(&lp->tx_ring[entry])) 116862306a36Sopenharmony_ci break; /* It still hasn't been processed. */ 116962306a36Sopenharmony_ci if (lp->tx_skbuff[entry]) { 117062306a36Sopenharmony_ci dev_consume_skb_irq(lp->tx_skbuff[entry]); 117162306a36Sopenharmony_ci lp->tx_skbuff[entry] = NULL; 117262306a36Sopenharmony_ci } 117362306a36Sopenharmony_ci dirty_tx++; 117462306a36Sopenharmony_ci } 117562306a36Sopenharmony_ci lp->dirty_tx = dirty_tx; 117662306a36Sopenharmony_ci outw(AckIntr | DownComplete, ioaddr + EL3_CMD); 117762306a36Sopenharmony_ci if (lp->tx_full && (lp->cur_tx - dirty_tx <= TX_RING_SIZE - 1)) { 117862306a36Sopenharmony_ci lp->tx_full = 0; 117962306a36Sopenharmony_ci netif_wake_queue(dev); 118062306a36Sopenharmony_ci } 118162306a36Sopenharmony_ci } 118262306a36Sopenharmony_ci#ifdef VORTEX_BUS_MASTER 118362306a36Sopenharmony_ci if (status & DMADone) { 118462306a36Sopenharmony_ci outw(0x1000, ioaddr + Wn7_MasterStatus); /* Ack the event. */ 118562306a36Sopenharmony_ci dev_consume_skb_irq(lp->tx_skb); /* Release the transferred buffer */ 118662306a36Sopenharmony_ci netif_wake_queue(dev); 118762306a36Sopenharmony_ci } 118862306a36Sopenharmony_ci#endif 118962306a36Sopenharmony_ci if (status & UpComplete) { 119062306a36Sopenharmony_ci boomerang_rx(dev); 119162306a36Sopenharmony_ci outw(AckIntr | UpComplete, ioaddr + EL3_CMD); 119262306a36Sopenharmony_ci } 119362306a36Sopenharmony_ci if (status & (AdapterFailure | RxEarly | StatsFull)) { 119462306a36Sopenharmony_ci /* Handle all uncommon interrupts at once. */ 119562306a36Sopenharmony_ci if (status & RxEarly) { /* Rx early is unused. */ 119662306a36Sopenharmony_ci corkscrew_rx(dev); 119762306a36Sopenharmony_ci outw(AckIntr | RxEarly, ioaddr + EL3_CMD); 119862306a36Sopenharmony_ci } 119962306a36Sopenharmony_ci if (status & StatsFull) { /* Empty statistics. */ 120062306a36Sopenharmony_ci static int DoneDidThat; 120162306a36Sopenharmony_ci if (corkscrew_debug > 4) 120262306a36Sopenharmony_ci pr_debug("%s: Updating stats.\n", dev->name); 120362306a36Sopenharmony_ci update_stats(ioaddr, dev); 120462306a36Sopenharmony_ci /* DEBUG HACK: Disable statistics as an interrupt source. */ 120562306a36Sopenharmony_ci /* This occurs when we have the wrong media type! */ 120662306a36Sopenharmony_ci if (DoneDidThat == 0 && inw(ioaddr + EL3_STATUS) & StatsFull) { 120762306a36Sopenharmony_ci int win, reg; 120862306a36Sopenharmony_ci pr_notice("%s: Updating stats failed, disabling stats as an interrupt source.\n", 120962306a36Sopenharmony_ci dev->name); 121062306a36Sopenharmony_ci for (win = 0; win < 8; win++) { 121162306a36Sopenharmony_ci EL3WINDOW(win); 121262306a36Sopenharmony_ci pr_notice("Vortex window %d:", win); 121362306a36Sopenharmony_ci for (reg = 0; reg < 16; reg++) 121462306a36Sopenharmony_ci pr_cont(" %2.2x", inb(ioaddr + reg)); 121562306a36Sopenharmony_ci pr_cont("\n"); 121662306a36Sopenharmony_ci } 121762306a36Sopenharmony_ci EL3WINDOW(7); 121862306a36Sopenharmony_ci outw(SetIntrEnb | TxAvailable | 121962306a36Sopenharmony_ci RxComplete | AdapterFailure | 122062306a36Sopenharmony_ci UpComplete | DownComplete | 122162306a36Sopenharmony_ci TxComplete, ioaddr + EL3_CMD); 122262306a36Sopenharmony_ci DoneDidThat++; 122362306a36Sopenharmony_ci } 122462306a36Sopenharmony_ci } 122562306a36Sopenharmony_ci if (status & AdapterFailure) { 122662306a36Sopenharmony_ci /* Adapter failure requires Rx reset and reinit. */ 122762306a36Sopenharmony_ci outw(RxReset, ioaddr + EL3_CMD); 122862306a36Sopenharmony_ci /* Set the Rx filter to the current state. */ 122962306a36Sopenharmony_ci set_rx_mode(dev); 123062306a36Sopenharmony_ci outw(RxEnable, ioaddr + EL3_CMD); /* Re-enable the receiver. */ 123162306a36Sopenharmony_ci outw(AckIntr | AdapterFailure, 123262306a36Sopenharmony_ci ioaddr + EL3_CMD); 123362306a36Sopenharmony_ci } 123462306a36Sopenharmony_ci } 123562306a36Sopenharmony_ci 123662306a36Sopenharmony_ci if (--i < 0) { 123762306a36Sopenharmony_ci pr_err("%s: Too much work in interrupt, status %4.4x. Disabling functions (%4.4x).\n", 123862306a36Sopenharmony_ci dev->name, status, SetStatusEnb | ((~status) & 0x7FE)); 123962306a36Sopenharmony_ci /* Disable all pending interrupts. */ 124062306a36Sopenharmony_ci outw(SetStatusEnb | ((~status) & 0x7FE), ioaddr + EL3_CMD); 124162306a36Sopenharmony_ci outw(AckIntr | 0x7FF, ioaddr + EL3_CMD); 124262306a36Sopenharmony_ci break; 124362306a36Sopenharmony_ci } 124462306a36Sopenharmony_ci /* Acknowledge the IRQ. */ 124562306a36Sopenharmony_ci outw(AckIntr | IntReq | IntLatch, ioaddr + EL3_CMD); 124662306a36Sopenharmony_ci 124762306a36Sopenharmony_ci } while ((status = inw(ioaddr + EL3_STATUS)) & (IntLatch | RxComplete)); 124862306a36Sopenharmony_ci 124962306a36Sopenharmony_ci spin_unlock(&lp->lock); 125062306a36Sopenharmony_ci 125162306a36Sopenharmony_ci if (corkscrew_debug > 4) 125262306a36Sopenharmony_ci pr_debug("%s: exiting interrupt, status %4.4x.\n", dev->name, status); 125362306a36Sopenharmony_ci return IRQ_HANDLED; 125462306a36Sopenharmony_ci} 125562306a36Sopenharmony_ci 125662306a36Sopenharmony_cistatic int corkscrew_rx(struct net_device *dev) 125762306a36Sopenharmony_ci{ 125862306a36Sopenharmony_ci int ioaddr = dev->base_addr; 125962306a36Sopenharmony_ci int i; 126062306a36Sopenharmony_ci short rx_status; 126162306a36Sopenharmony_ci 126262306a36Sopenharmony_ci if (corkscrew_debug > 5) 126362306a36Sopenharmony_ci pr_debug(" In rx_packet(), status %4.4x, rx_status %4.4x.\n", 126462306a36Sopenharmony_ci inw(ioaddr + EL3_STATUS), inw(ioaddr + RxStatus)); 126562306a36Sopenharmony_ci while ((rx_status = inw(ioaddr + RxStatus)) > 0) { 126662306a36Sopenharmony_ci if (rx_status & 0x4000) { /* Error, update stats. */ 126762306a36Sopenharmony_ci unsigned char rx_error = inb(ioaddr + RxErrors); 126862306a36Sopenharmony_ci if (corkscrew_debug > 2) 126962306a36Sopenharmony_ci pr_debug(" Rx error: status %2.2x.\n", 127062306a36Sopenharmony_ci rx_error); 127162306a36Sopenharmony_ci dev->stats.rx_errors++; 127262306a36Sopenharmony_ci if (rx_error & 0x01) 127362306a36Sopenharmony_ci dev->stats.rx_over_errors++; 127462306a36Sopenharmony_ci if (rx_error & 0x02) 127562306a36Sopenharmony_ci dev->stats.rx_length_errors++; 127662306a36Sopenharmony_ci if (rx_error & 0x04) 127762306a36Sopenharmony_ci dev->stats.rx_frame_errors++; 127862306a36Sopenharmony_ci if (rx_error & 0x08) 127962306a36Sopenharmony_ci dev->stats.rx_crc_errors++; 128062306a36Sopenharmony_ci if (rx_error & 0x10) 128162306a36Sopenharmony_ci dev->stats.rx_length_errors++; 128262306a36Sopenharmony_ci } else { 128362306a36Sopenharmony_ci /* The packet length: up to 4.5K!. */ 128462306a36Sopenharmony_ci short pkt_len = rx_status & 0x1fff; 128562306a36Sopenharmony_ci struct sk_buff *skb; 128662306a36Sopenharmony_ci 128762306a36Sopenharmony_ci skb = netdev_alloc_skb(dev, pkt_len + 5 + 2); 128862306a36Sopenharmony_ci if (corkscrew_debug > 4) 128962306a36Sopenharmony_ci pr_debug("Receiving packet size %d status %4.4x.\n", 129062306a36Sopenharmony_ci pkt_len, rx_status); 129162306a36Sopenharmony_ci if (skb != NULL) { 129262306a36Sopenharmony_ci skb_reserve(skb, 2); /* Align IP on 16 byte boundaries */ 129362306a36Sopenharmony_ci /* 'skb_put()' points to the start of sk_buff data area. */ 129462306a36Sopenharmony_ci insl(ioaddr + RX_FIFO, 129562306a36Sopenharmony_ci skb_put(skb, pkt_len), 129662306a36Sopenharmony_ci (pkt_len + 3) >> 2); 129762306a36Sopenharmony_ci outw(RxDiscard, ioaddr + EL3_CMD); /* Pop top Rx packet. */ 129862306a36Sopenharmony_ci skb->protocol = eth_type_trans(skb, dev); 129962306a36Sopenharmony_ci netif_rx(skb); 130062306a36Sopenharmony_ci dev->stats.rx_packets++; 130162306a36Sopenharmony_ci dev->stats.rx_bytes += pkt_len; 130262306a36Sopenharmony_ci /* Wait a limited time to go to next packet. */ 130362306a36Sopenharmony_ci for (i = 200; i >= 0; i--) 130462306a36Sopenharmony_ci if (! (inw(ioaddr + EL3_STATUS) & CmdInProgress)) 130562306a36Sopenharmony_ci break; 130662306a36Sopenharmony_ci continue; 130762306a36Sopenharmony_ci } else if (corkscrew_debug) 130862306a36Sopenharmony_ci pr_debug("%s: Couldn't allocate a sk_buff of size %d.\n", dev->name, pkt_len); 130962306a36Sopenharmony_ci } 131062306a36Sopenharmony_ci outw(RxDiscard, ioaddr + EL3_CMD); 131162306a36Sopenharmony_ci dev->stats.rx_dropped++; 131262306a36Sopenharmony_ci /* Wait a limited time to skip this packet. */ 131362306a36Sopenharmony_ci for (i = 200; i >= 0; i--) 131462306a36Sopenharmony_ci if (!(inw(ioaddr + EL3_STATUS) & CmdInProgress)) 131562306a36Sopenharmony_ci break; 131662306a36Sopenharmony_ci } 131762306a36Sopenharmony_ci return 0; 131862306a36Sopenharmony_ci} 131962306a36Sopenharmony_ci 132062306a36Sopenharmony_cistatic int boomerang_rx(struct net_device *dev) 132162306a36Sopenharmony_ci{ 132262306a36Sopenharmony_ci struct corkscrew_private *vp = netdev_priv(dev); 132362306a36Sopenharmony_ci int entry = vp->cur_rx % RX_RING_SIZE; 132462306a36Sopenharmony_ci int ioaddr = dev->base_addr; 132562306a36Sopenharmony_ci int rx_status; 132662306a36Sopenharmony_ci 132762306a36Sopenharmony_ci if (corkscrew_debug > 5) 132862306a36Sopenharmony_ci pr_debug(" In boomerang_rx(), status %4.4x, rx_status %4.4x.\n", 132962306a36Sopenharmony_ci inw(ioaddr + EL3_STATUS), inw(ioaddr + RxStatus)); 133062306a36Sopenharmony_ci while ((rx_status = vp->rx_ring[entry].status) & RxDComplete) { 133162306a36Sopenharmony_ci if (rx_status & RxDError) { /* Error, update stats. */ 133262306a36Sopenharmony_ci unsigned char rx_error = rx_status >> 16; 133362306a36Sopenharmony_ci if (corkscrew_debug > 2) 133462306a36Sopenharmony_ci pr_debug(" Rx error: status %2.2x.\n", 133562306a36Sopenharmony_ci rx_error); 133662306a36Sopenharmony_ci dev->stats.rx_errors++; 133762306a36Sopenharmony_ci if (rx_error & 0x01) 133862306a36Sopenharmony_ci dev->stats.rx_over_errors++; 133962306a36Sopenharmony_ci if (rx_error & 0x02) 134062306a36Sopenharmony_ci dev->stats.rx_length_errors++; 134162306a36Sopenharmony_ci if (rx_error & 0x04) 134262306a36Sopenharmony_ci dev->stats.rx_frame_errors++; 134362306a36Sopenharmony_ci if (rx_error & 0x08) 134462306a36Sopenharmony_ci dev->stats.rx_crc_errors++; 134562306a36Sopenharmony_ci if (rx_error & 0x10) 134662306a36Sopenharmony_ci dev->stats.rx_length_errors++; 134762306a36Sopenharmony_ci } else { 134862306a36Sopenharmony_ci /* The packet length: up to 4.5K!. */ 134962306a36Sopenharmony_ci short pkt_len = rx_status & 0x1fff; 135062306a36Sopenharmony_ci struct sk_buff *skb; 135162306a36Sopenharmony_ci 135262306a36Sopenharmony_ci dev->stats.rx_bytes += pkt_len; 135362306a36Sopenharmony_ci if (corkscrew_debug > 4) 135462306a36Sopenharmony_ci pr_debug("Receiving packet size %d status %4.4x.\n", 135562306a36Sopenharmony_ci pkt_len, rx_status); 135662306a36Sopenharmony_ci 135762306a36Sopenharmony_ci /* Check if the packet is long enough to just accept without 135862306a36Sopenharmony_ci copying to a properly sized skbuff. */ 135962306a36Sopenharmony_ci if (pkt_len < rx_copybreak && 136062306a36Sopenharmony_ci (skb = netdev_alloc_skb(dev, pkt_len + 4)) != NULL) { 136162306a36Sopenharmony_ci skb_reserve(skb, 2); /* Align IP on 16 byte boundaries */ 136262306a36Sopenharmony_ci /* 'skb_put()' points to the start of sk_buff data area. */ 136362306a36Sopenharmony_ci skb_put_data(skb, 136462306a36Sopenharmony_ci isa_bus_to_virt(vp->rx_ring[entry].addr), 136562306a36Sopenharmony_ci pkt_len); 136662306a36Sopenharmony_ci rx_copy++; 136762306a36Sopenharmony_ci } else { 136862306a36Sopenharmony_ci void *temp; 136962306a36Sopenharmony_ci /* Pass up the skbuff already on the Rx ring. */ 137062306a36Sopenharmony_ci skb = vp->rx_skbuff[entry]; 137162306a36Sopenharmony_ci vp->rx_skbuff[entry] = NULL; 137262306a36Sopenharmony_ci temp = skb_put(skb, pkt_len); 137362306a36Sopenharmony_ci /* Remove this checking code for final release. */ 137462306a36Sopenharmony_ci if (isa_bus_to_virt(vp->rx_ring[entry].addr) != temp) 137562306a36Sopenharmony_ci pr_warn("%s: Warning -- the skbuff addresses do not match in boomerang_rx: %p vs. %p / %p\n", 137662306a36Sopenharmony_ci dev->name, 137762306a36Sopenharmony_ci isa_bus_to_virt(vp->rx_ring[entry].addr), 137862306a36Sopenharmony_ci skb->head, temp); 137962306a36Sopenharmony_ci rx_nocopy++; 138062306a36Sopenharmony_ci } 138162306a36Sopenharmony_ci skb->protocol = eth_type_trans(skb, dev); 138262306a36Sopenharmony_ci netif_rx(skb); 138362306a36Sopenharmony_ci dev->stats.rx_packets++; 138462306a36Sopenharmony_ci } 138562306a36Sopenharmony_ci entry = (++vp->cur_rx) % RX_RING_SIZE; 138662306a36Sopenharmony_ci } 138762306a36Sopenharmony_ci /* Refill the Rx ring buffers. */ 138862306a36Sopenharmony_ci for (; vp->cur_rx - vp->dirty_rx > 0; vp->dirty_rx++) { 138962306a36Sopenharmony_ci struct sk_buff *skb; 139062306a36Sopenharmony_ci entry = vp->dirty_rx % RX_RING_SIZE; 139162306a36Sopenharmony_ci if (vp->rx_skbuff[entry] == NULL) { 139262306a36Sopenharmony_ci skb = netdev_alloc_skb(dev, PKT_BUF_SZ); 139362306a36Sopenharmony_ci if (skb == NULL) 139462306a36Sopenharmony_ci break; /* Bad news! */ 139562306a36Sopenharmony_ci skb_reserve(skb, 2); /* Align IP on 16 byte boundaries */ 139662306a36Sopenharmony_ci vp->rx_ring[entry].addr = isa_virt_to_bus(skb->data); 139762306a36Sopenharmony_ci vp->rx_skbuff[entry] = skb; 139862306a36Sopenharmony_ci } 139962306a36Sopenharmony_ci vp->rx_ring[entry].status = 0; /* Clear complete bit. */ 140062306a36Sopenharmony_ci } 140162306a36Sopenharmony_ci return 0; 140262306a36Sopenharmony_ci} 140362306a36Sopenharmony_ci 140462306a36Sopenharmony_cistatic int corkscrew_close(struct net_device *dev) 140562306a36Sopenharmony_ci{ 140662306a36Sopenharmony_ci struct corkscrew_private *vp = netdev_priv(dev); 140762306a36Sopenharmony_ci int ioaddr = dev->base_addr; 140862306a36Sopenharmony_ci int i; 140962306a36Sopenharmony_ci 141062306a36Sopenharmony_ci netif_stop_queue(dev); 141162306a36Sopenharmony_ci 141262306a36Sopenharmony_ci if (corkscrew_debug > 1) { 141362306a36Sopenharmony_ci pr_debug("%s: corkscrew_close() status %4.4x, Tx status %2.2x.\n", 141462306a36Sopenharmony_ci dev->name, inw(ioaddr + EL3_STATUS), 141562306a36Sopenharmony_ci inb(ioaddr + TxStatus)); 141662306a36Sopenharmony_ci pr_debug("%s: corkscrew close stats: rx_nocopy %d rx_copy %d tx_queued %d.\n", 141762306a36Sopenharmony_ci dev->name, rx_nocopy, rx_copy, queued_packet); 141862306a36Sopenharmony_ci } 141962306a36Sopenharmony_ci 142062306a36Sopenharmony_ci del_timer_sync(&vp->timer); 142162306a36Sopenharmony_ci 142262306a36Sopenharmony_ci /* Turn off statistics ASAP. We update lp->stats below. */ 142362306a36Sopenharmony_ci outw(StatsDisable, ioaddr + EL3_CMD); 142462306a36Sopenharmony_ci 142562306a36Sopenharmony_ci /* Disable the receiver and transmitter. */ 142662306a36Sopenharmony_ci outw(RxDisable, ioaddr + EL3_CMD); 142762306a36Sopenharmony_ci outw(TxDisable, ioaddr + EL3_CMD); 142862306a36Sopenharmony_ci 142962306a36Sopenharmony_ci if (dev->if_port == XCVR_10base2) 143062306a36Sopenharmony_ci /* Turn off thinnet power. Green! */ 143162306a36Sopenharmony_ci outw(StopCoax, ioaddr + EL3_CMD); 143262306a36Sopenharmony_ci 143362306a36Sopenharmony_ci free_irq(dev->irq, dev); 143462306a36Sopenharmony_ci 143562306a36Sopenharmony_ci outw(SetIntrEnb | 0x0000, ioaddr + EL3_CMD); 143662306a36Sopenharmony_ci 143762306a36Sopenharmony_ci update_stats(ioaddr, dev); 143862306a36Sopenharmony_ci if (vp->full_bus_master_rx) { /* Free Boomerang bus master Rx buffers. */ 143962306a36Sopenharmony_ci outl(0, ioaddr + UpListPtr); 144062306a36Sopenharmony_ci for (i = 0; i < RX_RING_SIZE; i++) 144162306a36Sopenharmony_ci if (vp->rx_skbuff[i]) { 144262306a36Sopenharmony_ci dev_kfree_skb(vp->rx_skbuff[i]); 144362306a36Sopenharmony_ci vp->rx_skbuff[i] = NULL; 144462306a36Sopenharmony_ci } 144562306a36Sopenharmony_ci } 144662306a36Sopenharmony_ci if (vp->full_bus_master_tx) { /* Free Boomerang bus master Tx buffers. */ 144762306a36Sopenharmony_ci outl(0, ioaddr + DownListPtr); 144862306a36Sopenharmony_ci for (i = 0; i < TX_RING_SIZE; i++) 144962306a36Sopenharmony_ci if (vp->tx_skbuff[i]) { 145062306a36Sopenharmony_ci dev_kfree_skb(vp->tx_skbuff[i]); 145162306a36Sopenharmony_ci vp->tx_skbuff[i] = NULL; 145262306a36Sopenharmony_ci } 145362306a36Sopenharmony_ci } 145462306a36Sopenharmony_ci 145562306a36Sopenharmony_ci return 0; 145662306a36Sopenharmony_ci} 145762306a36Sopenharmony_ci 145862306a36Sopenharmony_cistatic struct net_device_stats *corkscrew_get_stats(struct net_device *dev) 145962306a36Sopenharmony_ci{ 146062306a36Sopenharmony_ci struct corkscrew_private *vp = netdev_priv(dev); 146162306a36Sopenharmony_ci unsigned long flags; 146262306a36Sopenharmony_ci 146362306a36Sopenharmony_ci if (netif_running(dev)) { 146462306a36Sopenharmony_ci spin_lock_irqsave(&vp->lock, flags); 146562306a36Sopenharmony_ci update_stats(dev->base_addr, dev); 146662306a36Sopenharmony_ci spin_unlock_irqrestore(&vp->lock, flags); 146762306a36Sopenharmony_ci } 146862306a36Sopenharmony_ci return &dev->stats; 146962306a36Sopenharmony_ci} 147062306a36Sopenharmony_ci 147162306a36Sopenharmony_ci/* Update statistics. 147262306a36Sopenharmony_ci Unlike with the EL3 we need not worry about interrupts changing 147362306a36Sopenharmony_ci the window setting from underneath us, but we must still guard 147462306a36Sopenharmony_ci against a race condition with a StatsUpdate interrupt updating the 147562306a36Sopenharmony_ci table. This is done by checking that the ASM (!) code generated uses 147662306a36Sopenharmony_ci atomic updates with '+='. 147762306a36Sopenharmony_ci */ 147862306a36Sopenharmony_cistatic void update_stats(int ioaddr, struct net_device *dev) 147962306a36Sopenharmony_ci{ 148062306a36Sopenharmony_ci /* Unlike the 3c5x9 we need not turn off stats updates while reading. */ 148162306a36Sopenharmony_ci /* Switch to the stats window, and read everything. */ 148262306a36Sopenharmony_ci EL3WINDOW(6); 148362306a36Sopenharmony_ci dev->stats.tx_carrier_errors += inb(ioaddr + 0); 148462306a36Sopenharmony_ci dev->stats.tx_heartbeat_errors += inb(ioaddr + 1); 148562306a36Sopenharmony_ci /* Multiple collisions. */ inb(ioaddr + 2); 148662306a36Sopenharmony_ci dev->stats.collisions += inb(ioaddr + 3); 148762306a36Sopenharmony_ci dev->stats.tx_window_errors += inb(ioaddr + 4); 148862306a36Sopenharmony_ci dev->stats.rx_fifo_errors += inb(ioaddr + 5); 148962306a36Sopenharmony_ci dev->stats.tx_packets += inb(ioaddr + 6); 149062306a36Sopenharmony_ci dev->stats.tx_packets += (inb(ioaddr + 9) & 0x30) << 4; 149162306a36Sopenharmony_ci /* Rx packets */ inb(ioaddr + 7); 149262306a36Sopenharmony_ci /* Must read to clear */ 149362306a36Sopenharmony_ci /* Tx deferrals */ inb(ioaddr + 8); 149462306a36Sopenharmony_ci /* Don't bother with register 9, an extension of registers 6&7. 149562306a36Sopenharmony_ci If we do use the 6&7 values the atomic update assumption above 149662306a36Sopenharmony_ci is invalid. */ 149762306a36Sopenharmony_ci inw(ioaddr + 10); /* Total Rx and Tx octets. */ 149862306a36Sopenharmony_ci inw(ioaddr + 12); 149962306a36Sopenharmony_ci /* New: On the Vortex we must also clear the BadSSD counter. */ 150062306a36Sopenharmony_ci EL3WINDOW(4); 150162306a36Sopenharmony_ci inb(ioaddr + 12); 150262306a36Sopenharmony_ci 150362306a36Sopenharmony_ci /* We change back to window 7 (not 1) with the Vortex. */ 150462306a36Sopenharmony_ci EL3WINDOW(7); 150562306a36Sopenharmony_ci} 150662306a36Sopenharmony_ci 150762306a36Sopenharmony_ci/* This new version of set_rx_mode() supports v1.4 kernels. 150862306a36Sopenharmony_ci The Vortex chip has no documented multicast filter, so the only 150962306a36Sopenharmony_ci multicast setting is to receive all multicast frames. At least 151062306a36Sopenharmony_ci the chip has a very clean way to set the mode, unlike many others. */ 151162306a36Sopenharmony_cistatic void set_rx_mode(struct net_device *dev) 151262306a36Sopenharmony_ci{ 151362306a36Sopenharmony_ci int ioaddr = dev->base_addr; 151462306a36Sopenharmony_ci unsigned short new_mode; 151562306a36Sopenharmony_ci 151662306a36Sopenharmony_ci if (dev->flags & IFF_PROMISC) { 151762306a36Sopenharmony_ci if (corkscrew_debug > 3) 151862306a36Sopenharmony_ci pr_debug("%s: Setting promiscuous mode.\n", 151962306a36Sopenharmony_ci dev->name); 152062306a36Sopenharmony_ci new_mode = SetRxFilter | RxStation | RxMulticast | RxBroadcast | RxProm; 152162306a36Sopenharmony_ci } else if (!netdev_mc_empty(dev) || dev->flags & IFF_ALLMULTI) { 152262306a36Sopenharmony_ci new_mode = SetRxFilter | RxStation | RxMulticast | RxBroadcast; 152362306a36Sopenharmony_ci } else 152462306a36Sopenharmony_ci new_mode = SetRxFilter | RxStation | RxBroadcast; 152562306a36Sopenharmony_ci 152662306a36Sopenharmony_ci outw(new_mode, ioaddr + EL3_CMD); 152762306a36Sopenharmony_ci} 152862306a36Sopenharmony_ci 152962306a36Sopenharmony_cistatic void netdev_get_drvinfo(struct net_device *dev, 153062306a36Sopenharmony_ci struct ethtool_drvinfo *info) 153162306a36Sopenharmony_ci{ 153262306a36Sopenharmony_ci strscpy(info->driver, DRV_NAME, sizeof(info->driver)); 153362306a36Sopenharmony_ci snprintf(info->bus_info, sizeof(info->bus_info), "ISA 0x%lx", 153462306a36Sopenharmony_ci dev->base_addr); 153562306a36Sopenharmony_ci} 153662306a36Sopenharmony_ci 153762306a36Sopenharmony_cistatic u32 netdev_get_msglevel(struct net_device *dev) 153862306a36Sopenharmony_ci{ 153962306a36Sopenharmony_ci return corkscrew_debug; 154062306a36Sopenharmony_ci} 154162306a36Sopenharmony_ci 154262306a36Sopenharmony_cistatic void netdev_set_msglevel(struct net_device *dev, u32 level) 154362306a36Sopenharmony_ci{ 154462306a36Sopenharmony_ci corkscrew_debug = level; 154562306a36Sopenharmony_ci} 154662306a36Sopenharmony_ci 154762306a36Sopenharmony_cistatic const struct ethtool_ops netdev_ethtool_ops = { 154862306a36Sopenharmony_ci .get_drvinfo = netdev_get_drvinfo, 154962306a36Sopenharmony_ci .get_msglevel = netdev_get_msglevel, 155062306a36Sopenharmony_ci .set_msglevel = netdev_set_msglevel, 155162306a36Sopenharmony_ci}; 155262306a36Sopenharmony_ci 155362306a36Sopenharmony_ci 155462306a36Sopenharmony_ci#ifdef MODULE 155562306a36Sopenharmony_civoid cleanup_module(void) 155662306a36Sopenharmony_ci{ 155762306a36Sopenharmony_ci while (!list_empty(&root_corkscrew_dev)) { 155862306a36Sopenharmony_ci struct net_device *dev; 155962306a36Sopenharmony_ci struct corkscrew_private *vp; 156062306a36Sopenharmony_ci 156162306a36Sopenharmony_ci vp = list_entry(root_corkscrew_dev.next, 156262306a36Sopenharmony_ci struct corkscrew_private, list); 156362306a36Sopenharmony_ci dev = vp->our_dev; 156462306a36Sopenharmony_ci unregister_netdev(dev); 156562306a36Sopenharmony_ci cleanup_card(dev); 156662306a36Sopenharmony_ci free_netdev(dev); 156762306a36Sopenharmony_ci } 156862306a36Sopenharmony_ci} 156962306a36Sopenharmony_ci#endif /* MODULE */ 1570