Commit c69552c2 authored by Jeff Garzik's avatar Jeff Garzik

sis900 net driver update:

* fix eeprom accesses
* fix tx desc overflow
* fix tx timeout bug
* add sis963 support
parent 9b64d5a3
/* sis900.c: A SiS 900/7016 PCI Fast Ethernet driver for Linux. /* sis900.c: A SiS 900/7016 PCI Fast Ethernet driver for Linux.
Copyright 1999 Silicon Integrated System Corporation Copyright 1999 Silicon Integrated System Corporation
Revision: 1.08.04 Apr. 25 2002 Revision: 1.08.06 Sep. 24 2002
Modified from the driver which is originally written by Donald Becker. Modified from the driver which is originally written by Donald Becker.
...@@ -18,6 +18,8 @@ ...@@ -18,6 +18,8 @@
preliminary Rev. 1.0 Jan. 18, 1998 preliminary Rev. 1.0 Jan. 18, 1998
http://www.sis.com.tw/support/databook.htm http://www.sis.com.tw/support/databook.htm
Rev 1.08.06 Sep. 24 2002 Mufasa Yang bug fix for Tx timeout & add SiS963 support
Rev 1.08.05 Jun. 6 2002 Mufasa Yang bug fix for read_eeprom & Tx descriptor over-boundary
Rev 1.08.04 Apr. 25 2002 Mufasa Yang <mufasa@sis.com.tw> added SiS962 support Rev 1.08.04 Apr. 25 2002 Mufasa Yang <mufasa@sis.com.tw> added SiS962 support
Rev 1.08.03 Feb. 1 2002 Matt Domsch <Matt_Domsch@dell.com> update to use library crc32 function Rev 1.08.03 Feb. 1 2002 Matt Domsch <Matt_Domsch@dell.com> update to use library crc32 function
Rev 1.08.02 Nov. 30 2001 Hui-Fen Hsu workaround for EDB & bug fix for dhcp problem Rev 1.08.02 Nov. 30 2001 Hui-Fen Hsu workaround for EDB & bug fix for dhcp problem
...@@ -72,7 +74,7 @@ ...@@ -72,7 +74,7 @@
#include "sis900.h" #include "sis900.h"
#define SIS900_MODULE_NAME "sis900" #define SIS900_MODULE_NAME "sis900"
#define SIS900_DRV_VERSION "v1.08.04 4/25/2002" #define SIS900_DRV_VERSION "v1.08.06 9/24/2002"
static char version[] __devinitdata = static char version[] __devinitdata =
KERN_INFO "sis900.c: " SIS900_DRV_VERSION "\n"; KERN_INFO "sis900.c: " SIS900_DRV_VERSION "\n";
...@@ -309,31 +311,38 @@ static int __devinit sis635_get_mac_addr(struct pci_dev * pci_dev, struct net_de ...@@ -309,31 +311,38 @@ static int __devinit sis635_get_mac_addr(struct pci_dev * pci_dev, struct net_de
} }
/** /**
* sis962_get_mac_addr: - Get MAC address for SiS962 model * sis96x_get_mac_addr: - Get MAC address for SiS962 or SiS963 model
* @pci_dev: the sis900 pci device * @pci_dev: the sis900 pci device
* @net_dev: the net device to get address for * @net_dev: the net device to get address for
* *
* SiS962 model, use EEPROM to store MAC address. And EEPROM is shared by * SiS962 or SiS963 model, use EEPROM to store MAC address. And EEPROM
* is shared by
* LAN and 1394. When access EEPROM, send EEREQ signal to hardware first * LAN and 1394. When access EEPROM, send EEREQ signal to hardware first
* and wait for EEGNT. If EEGNT is ON, EEPROM is permitted to be access * and wait for EEGNT. If EEGNT is ON, EEPROM is permitted to be access
* by LAN, otherwise is not. After MAC address is read from EEPROM, send * by LAN, otherwise is not. After MAC address is read from EEPROM, send
* EEDONE signal to refuse EEPROM access by LAN. * EEDONE signal to refuse EEPROM access by LAN.
* The EEPROM map of SiS962 or SiS963 is different to SiS900.
* The signature field in SiS962 or SiS963 spec is meaningless.
* MAC address is read into @net_dev->dev_addr. * MAC address is read into @net_dev->dev_addr.
*/ */
static int __devinit sis962_get_mac_addr(struct pci_dev * pci_dev, struct net_device *net_dev) static int __devinit sis96x_get_mac_addr(struct pci_dev * pci_dev, struct net_device *net_dev)
{ {
long ioaddr = net_dev->base_addr; long ioaddr = net_dev->base_addr;
long ee_addr = ioaddr + mear; long ee_addr = ioaddr + mear;
u32 waittime = 0; u32 waittime = 0;
int ret = 0; int i;
outl(EEREQ, ee_addr); outl(EEREQ, ee_addr);
while(waittime < 2000) { while(waittime < 2000) {
if(inl(ee_addr) & EEGNT) { if(inl(ee_addr) & EEGNT) {
ret = sis900_get_mac_addr(pci_dev, net_dev);
/* get MAC address from EEPROM */
for (i = 0; i < 3; i++)
((u16 *)(net_dev->dev_addr))[i] = read_eeprom(ioaddr, i+EEPROMMACAddr);
outl(EEDONE, ee_addr); outl(EEDONE, ee_addr);
return(ret); return 1;
} else { } else {
udelay(1); udelay(1);
waittime ++; waittime ++;
...@@ -443,8 +452,8 @@ static int __devinit sis900_probe (struct pci_dev *pci_dev, const struct pci_dev ...@@ -443,8 +452,8 @@ static int __devinit sis900_probe (struct pci_dev *pci_dev, const struct pci_dev
ret = sis630e_get_mac_addr(pci_dev, net_dev); ret = sis630e_get_mac_addr(pci_dev, net_dev);
else if ((revision > 0x81) && (revision <= 0x90) ) else if ((revision > 0x81) && (revision <= 0x90) )
ret = sis635_get_mac_addr(pci_dev, net_dev); ret = sis635_get_mac_addr(pci_dev, net_dev);
else if (revision == SIS962_900_REV) else if (revision == SIS96x_900_REV)
ret = sis962_get_mac_addr(pci_dev, net_dev); ret = sis96x_get_mac_addr(pci_dev, net_dev);
else else
ret = sis900_get_mac_addr(pci_dev, net_dev); ret = sis900_get_mac_addr(pci_dev, net_dev);
...@@ -702,7 +711,7 @@ static u16 __devinit read_eeprom(long ioaddr, int location) ...@@ -702,7 +711,7 @@ static u16 __devinit read_eeprom(long ioaddr, int location)
outl(0, ee_addr); outl(0, ee_addr);
eeprom_delay(); eeprom_delay();
outl(EECLK, ee_addr); outl(EECS, ee_addr);
eeprom_delay(); eeprom_delay();
/* Shift the read command (9) bits out. */ /* Shift the read command (9) bits out. */
...@@ -713,7 +722,7 @@ static u16 __devinit read_eeprom(long ioaddr, int location) ...@@ -713,7 +722,7 @@ static u16 __devinit read_eeprom(long ioaddr, int location)
outl(dataval | EECLK, ee_addr); outl(dataval | EECLK, ee_addr);
eeprom_delay(); eeprom_delay();
} }
outb(EECS, ee_addr); outl(EECS, ee_addr);
eeprom_delay(); eeprom_delay();
/* read the 16-bits data in */ /* read the 16-bits data in */
...@@ -729,7 +738,6 @@ static u16 __devinit read_eeprom(long ioaddr, int location) ...@@ -729,7 +738,6 @@ static u16 __devinit read_eeprom(long ioaddr, int location)
/* Terminate the EEPROM access. */ /* Terminate the EEPROM access. */
outl(0, ee_addr); outl(0, ee_addr);
eeprom_delay(); eeprom_delay();
outl(EECLK, ee_addr);
return (retval); return (retval);
} }
...@@ -1432,8 +1440,8 @@ static void sis900_tx_timeout(struct net_device *net_dev) ...@@ -1432,8 +1440,8 @@ static void sis900_tx_timeout(struct net_device *net_dev)
net_dev->trans_start = jiffies; net_dev->trans_start = jiffies;
/* FIXME: Should we restart the transmission thread here ?? */ /* load Transmit Descriptor Register */
outl(TxENA | inl(ioaddr + cr), ioaddr + cr); outl(sis_priv->tx_ring_dma, ioaddr + txdp);
/* Enable all known interrupts by setting the interrupt mask. */ /* Enable all known interrupts by setting the interrupt mask. */
outl((RxSOVR|RxORN|RxERR|RxOK|TxURN|TxERR|TxIDLE), ioaddr + imr); outl((RxSOVR|RxORN|RxERR|RxOK|TxURN|TxERR|TxIDLE), ioaddr + imr);
...@@ -1457,6 +1465,8 @@ sis900_start_xmit(struct sk_buff *skb, struct net_device *net_dev) ...@@ -1457,6 +1465,8 @@ sis900_start_xmit(struct sk_buff *skb, struct net_device *net_dev)
long ioaddr = net_dev->base_addr; long ioaddr = net_dev->base_addr;
unsigned int entry; unsigned int entry;
unsigned long flags; unsigned long flags;
unsigned int index_cur_tx, index_dirty_tx;
unsigned int count_dirty_tx;
/* Don't transmit data before the complete of auto-negotiation */ /* Don't transmit data before the complete of auto-negotiation */
if(!sis_priv->autong_complete){ if(!sis_priv->autong_complete){
...@@ -1476,7 +1486,18 @@ sis900_start_xmit(struct sk_buff *skb, struct net_device *net_dev) ...@@ -1476,7 +1486,18 @@ sis900_start_xmit(struct sk_buff *skb, struct net_device *net_dev)
sis_priv->tx_ring[entry].cmdsts = (OWN | skb->len); sis_priv->tx_ring[entry].cmdsts = (OWN | skb->len);
outl(TxENA | inl(ioaddr + cr), ioaddr + cr); outl(TxENA | inl(ioaddr + cr), ioaddr + cr);
if (++sis_priv->cur_tx - sis_priv->dirty_tx < NUM_TX_DESC) { sis_priv->cur_tx ++;
index_cur_tx = sis_priv->cur_tx;
index_dirty_tx = sis_priv->dirty_tx;
for (count_dirty_tx = 0; index_cur_tx != index_dirty_tx; index_dirty_tx++)
count_dirty_tx ++;
if (index_cur_tx == index_dirty_tx) {
/* dirty_tx is met in the cycle of cur_tx, buffer full */
sis_priv->tx_full = 1;
netif_stop_queue(net_dev);
} else if (count_dirty_tx < NUM_TX_DESC) {
/* Typical path, tell upper layer that more transmission is possible */ /* Typical path, tell upper layer that more transmission is possible */
netif_start_queue(net_dev); netif_start_queue(net_dev);
} else { } else {
...@@ -1709,7 +1730,7 @@ static void sis900_finish_xmit (struct net_device *net_dev) ...@@ -1709,7 +1730,7 @@ static void sis900_finish_xmit (struct net_device *net_dev)
{ {
struct sis900_private *sis_priv = net_dev->priv; struct sis900_private *sis_priv = net_dev->priv;
for (; sis_priv->dirty_tx < sis_priv->cur_tx; sis_priv->dirty_tx++) { for (; sis_priv->dirty_tx != sis_priv->cur_tx; sis_priv->dirty_tx++) {
struct sk_buff *skb; struct sk_buff *skb;
unsigned int entry; unsigned int entry;
u32 tx_status; u32 tx_status;
......
...@@ -135,8 +135,8 @@ enum sis900_eeprom_command { ...@@ -135,8 +135,8 @@ enum sis900_eeprom_command {
EEaddrMask = 0x013F, EEcmdShift = 16 EEaddrMask = 0x013F, EEcmdShift = 16
}; };
/* For SiS962, request the eeprom software access */ /* For SiS962 or SiS963, request the eeprom software access */
enum sis962_eeprom_command { enum sis96x_eeprom_command {
EEREQ = 0x00000400, EEDONE = 0x00000200, EEGNT = 0x00000100 EEREQ = 0x00000400, EEDONE = 0x00000200, EEGNT = 0x00000100
}; };
...@@ -245,7 +245,7 @@ enum sis900_revision_id { ...@@ -245,7 +245,7 @@ enum sis900_revision_id {
SIS630A_900_REV = 0x80, SIS630E_900_REV = 0x81, SIS630A_900_REV = 0x80, SIS630E_900_REV = 0x81,
SIS630S_900_REV = 0x82, SIS630EA1_900_REV = 0x83, SIS630S_900_REV = 0x82, SIS630EA1_900_REV = 0x83,
SIS630ET_900_REV = 0x84, SIS635A_900_REV = 0x90, SIS630ET_900_REV = 0x84, SIS635A_900_REV = 0x90,
SIS962_900_REV = 0X91, SIS900B_900_REV = 0x03 SIS96x_900_REV = 0X91, SIS900B_900_REV = 0x03
}; };
enum sis630_revision_id { enum sis630_revision_id {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment