Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
L
linux
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
nexedi
linux
Commits
0d944a2e
Commit
0d944a2e
authored
Sep 30, 2002
by
Jeff Garzik
Browse files
Options
Browse Files
Download
Plain Diff
Merge mandrakesoft.com:/home/jgarzik/repo/linus-2.5
into mandrakesoft.com:/home/jgarzik/repo/net-drivers-2.5
parents
1a008d0e
1024362a
Changes
1
Hide whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
430 additions
and
328 deletions
+430
-328
drivers/net/natsemi.c
drivers/net/natsemi.c
+430
-328
No files found.
drivers/net/natsemi.c
View file @
0d944a2e
/* natsemi.c: A Linux PCI Ethernet driver for the NatSemi DP8381x series. */
/* natsemi.c: A Linux PCI Ethernet driver for the NatSemi DP8381x series. */
/*
/*
Written/copyright 1999-2001 by Donald Becker.
Written/copyright 1999-2001 by Donald Becker.
Portions copyright (c) 2001 Sun Microsystems (thockin@sun.com)
Portions copyright (c) 2001,2002 Sun Microsystems (thockin@sun.com)
Portions copyright 2001,2002 Manfred Spraul (manfred@colorfullife.com)
This software may be used and distributed according to the terms of
This software may be used and distributed according to the terms of
the GNU General Public License (GPL), incorporated herein by reference.
the GNU General Public License (GPL), incorporated herein by reference.
...
@@ -86,7 +87,7 @@
...
@@ -86,7 +87,7 @@
* use long for ee_addr (various)
* use long for ee_addr (various)
* print pointers properly (DaveM)
* print pointers properly (DaveM)
* include asm/irq.h (?)
* include asm/irq.h (?)
version 1.0.11:
version 1.0.11:
* check and reset if PHY errors appear (Adrian Sun)
* check and reset if PHY errors appear (Adrian Sun)
* WoL cleanup (Tim Hockin)
* WoL cleanup (Tim Hockin)
...
@@ -119,10 +120,23 @@
...
@@ -119,10 +120,23 @@
initialized
initialized
* enable only the WoL and PHY interrupts in wol mode
* enable only the WoL and PHY interrupts in wol mode
version 1.0.17:
* only do cable_magic on 83815 and early 83816 (Tim Hockin)
* create a function for rx refill (Manfred Spraul)
* combine drain_ring and init_ring (Manfred Spraul)
* oom handling (Manfred Spraul)
* hands_off instead of playing with netif_device_{de,a}ttach
(Manfred Spraul)
* be sure to write the MAC back to the chip (Manfred Spraul)
* lengthen EEPROM timeout, and always warn about timeouts
(Manfred Spraul)
* comments update (Manfred)
* do the right thing on a phy-reset (Manfred and Tim)
TODO:
TODO:
* big endian support with CFG:BEM instead of cpu_to_le32
* big endian support with CFG:BEM instead of cpu_to_le32
* support for an external PHY
* support for an external PHY
*
flow control
*
NAPI
*/
*/
#if !defined(__OPTIMIZE__)
#if !defined(__OPTIMIZE__)
...
@@ -157,23 +171,11 @@
...
@@ -157,23 +171,11 @@
#include <asm/uaccess.h>
#include <asm/uaccess.h>
#define DRV_NAME "natsemi"
#define DRV_NAME "natsemi"
#define DRV_VERSION "1.07+LK1.0.1
6
"
#define DRV_VERSION "1.07+LK1.0.1
7
"
#define DRV_RELDATE "
Aug 28
, 2002"
#define DRV_RELDATE "
Sep 27
, 2002"
/* Updated to recommendations in pci-skeleton v2.03. */
/* Updated to recommendations in pci-skeleton v2.03. */
/* Automatically extracted configuration info:
probe-func: natsemi_probe
config-in: tristate 'National Semiconductor DP8381x series PCI Ethernet support' CONFIG_NATSEMI
c-help-name: National Semiconductor DP8381x series PCI Ethernet support
c-help-symbol: CONFIG_NATSEMI
c-help: This driver is for the National Semiconductor DP8381x series,
c-help: including the 8381[56] chips.
c-help: More specific information and updates are available from
c-help: http://www.scyld.com/network/natsemi.html
*/
/* The user-configurable values.
/* The user-configurable values.
These may be modified when a driver module is loaded.*/
These may be modified when a driver module is loaded.*/
...
@@ -235,9 +237,11 @@ static int full_duplex[MAX_UNITS];
...
@@ -235,9 +237,11 @@ static int full_duplex[MAX_UNITS];
/* These identify the driver base version and may not be removed. */
/* These identify the driver base version and may not be removed. */
static
char
version
[]
__devinitdata
=
static
char
version
[]
__devinitdata
=
KERN_INFO
DRV_NAME
".c:v1.07 1/9/2001 Written by Donald Becker <becker@scyld.com>
\n
"
KERN_INFO
DRV_NAME
" dp8381x driver, version "
KERN_INFO
" http://www.scyld.com/network/natsemi.html
\n
"
DRV_VERSION
", "
DRV_RELDATE
"
\n
"
KERN_INFO
" (unofficial 2.4.x kernel port, version "
DRV_VERSION
", "
DRV_RELDATE
" Jeff Garzik, Tjeerd Mulder)
\n
"
;
KERN_INFO
" originally by Donald Becker <becker@scyld.com>
\n
"
KERN_INFO
" http://www.scyld.com/network/natsemi.html
\n
"
KERN_INFO
" 2.4.x kernel port by Jeff Garzik, Tjeerd Mulder
\n
"
;
MODULE_AUTHOR
(
"Donald Becker <becker@scyld.com>"
);
MODULE_AUTHOR
(
"Donald Becker <becker@scyld.com>"
);
MODULE_DESCRIPTION
(
"National Semiconductor DP8381x series PCI Ethernet driver"
);
MODULE_DESCRIPTION
(
"National Semiconductor DP8381x series PCI Ethernet driver"
);
...
@@ -249,11 +253,14 @@ MODULE_PARM(debug, "i");
...
@@ -249,11 +253,14 @@ MODULE_PARM(debug, "i");
MODULE_PARM
(
rx_copybreak
,
"i"
);
MODULE_PARM
(
rx_copybreak
,
"i"
);
MODULE_PARM
(
options
,
"1-"
__MODULE_STRING
(
MAX_UNITS
)
"i"
);
MODULE_PARM
(
options
,
"1-"
__MODULE_STRING
(
MAX_UNITS
)
"i"
);
MODULE_PARM
(
full_duplex
,
"1-"
__MODULE_STRING
(
MAX_UNITS
)
"i"
);
MODULE_PARM
(
full_duplex
,
"1-"
__MODULE_STRING
(
MAX_UNITS
)
"i"
);
MODULE_PARM_DESC
(
max_interrupt_work
,
"DP8381x maximum events handled per interrupt"
);
MODULE_PARM_DESC
(
max_interrupt_work
,
"DP8381x maximum events handled per interrupt"
);
MODULE_PARM_DESC
(
mtu
,
"DP8381x MTU (all boards)"
);
MODULE_PARM_DESC
(
mtu
,
"DP8381x MTU (all boards)"
);
MODULE_PARM_DESC
(
debug
,
"DP8381x default debug bitmask"
);
MODULE_PARM_DESC
(
debug
,
"DP8381x default debug bitmask"
);
MODULE_PARM_DESC
(
rx_copybreak
,
"DP8381x copy breakpoint for copy-only-tiny-frames"
);
MODULE_PARM_DESC
(
rx_copybreak
,
MODULE_PARM_DESC
(
options
,
"DP8381x: Bits 0-3: media type, bit 17: full duplex"
);
"DP8381x copy breakpoint for copy-only-tiny-frames"
);
MODULE_PARM_DESC
(
options
,
"DP8381x: Bits 0-3: media type, bit 17: full duplex"
);
MODULE_PARM_DESC
(
full_duplex
,
"DP8381x full duplex setting(s) (1)"
);
MODULE_PARM_DESC
(
full_duplex
,
"DP8381x full duplex setting(s) (1)"
);
/*
/*
...
@@ -267,7 +274,7 @@ It also works with other chips in in the DP83810 series.
...
@@ -267,7 +274,7 @@ It also works with other chips in in the DP83810 series.
II. Board-specific settings
II. Board-specific settings
This driver requires the PCI interrupt line to be valid.
This driver requires the PCI interrupt line to be valid.
It honors the EEPROM-set values.
It honors the EEPROM-set values.
III. Driver operation
III. Driver operation
...
@@ -277,7 +284,7 @@ This driver uses two statically allocated fixed-size descriptor lists
...
@@ -277,7 +284,7 @@ This driver uses two statically allocated fixed-size descriptor lists
formed into rings by a branch from the final descriptor to the beginning of
formed into rings by a branch from the final descriptor to the beginning of
the list. The ring sizes are set at compile time by RX/TX_RING_SIZE.
the list. The ring sizes are set at compile time by RX/TX_RING_SIZE.
The NatSemi design uses a 'next descriptor' pointer that the driver forms
The NatSemi design uses a 'next descriptor' pointer that the driver forms
into a list.
into a list.
IIIb/c. Transmit/Receive Structure
IIIb/c. Transmit/Receive Structure
...
@@ -306,20 +313,19 @@ skbuff at an offset of "+2", 16-byte aligning the IP header.
...
@@ -306,20 +313,19 @@ skbuff at an offset of "+2", 16-byte aligning the IP header.
IIId. Synchronization
IIId. Synchronization
The driver runs as two independent, single-threaded flows of control. One
Most operations are synchronized on the np->lock irq spinlock, except the
is the send-packet routine, which enforces single-threaded use by the
performance critical codepaths:
dev->tbusy flag. The other thread is the interrupt handler, which is single
threaded by the hardware and interrupt handling software.
The send packet thread has partial control over the Tx ring and 'dev->tbusy'
The rx process only runs in the interrupt handler. Access from outside
flag. It sets the tbusy flag whenever it's queuing a Tx packet. If the next
the interrupt handler is only permitted after disable_irq().
queue slot is empty, it clears the tbusy flag when finished otherwise it sets
the 'lp->tx_full' flag.
The interrupt handler has exclusive control over the Rx ring and records stats
The rx process usually runs under the dev->xmit_lock. If np->intr_tx_reap
from the Tx ring. After reaping the stats, it marks the Tx queue entry as
is set, then access is permitted under spin_lock_irq(&np->lock).
empty by incrementing the dirty_tx mark. Iff the 'lp->tx_full' flag is set, it
clears both the tx_full and tbusy flags.
Thus configuration functions that want to access everything must call
disable_irq(dev->irq);
spin_lock_bh(dev->xmit_lock);
spin_lock_irq(&np->lock);
IV. Notes
IV. Notes
...
@@ -337,7 +343,7 @@ IVc. Errata
...
@@ -337,7 +343,7 @@ IVc. Errata
None characterised.
None characterised.
*/
*/
enum
pcistuff
{
enum
pcistuff
{
PCI_USES_IO
=
0x01
,
PCI_USES_IO
=
0x01
,
...
@@ -378,6 +384,7 @@ enum register_offsets {
...
@@ -378,6 +384,7 @@ enum register_offsets {
IntrStatus
=
0x10
,
IntrStatus
=
0x10
,
IntrMask
=
0x14
,
IntrMask
=
0x14
,
IntrEnable
=
0x18
,
IntrEnable
=
0x18
,
IntrHoldoff
=
0x16
,
/* DP83816 only */
TxRingPtr
=
0x20
,
TxRingPtr
=
0x20
,
TxConfig
=
0x24
,
TxConfig
=
0x24
,
RxRingPtr
=
0x30
,
RxRingPtr
=
0x30
,
...
@@ -486,11 +493,11 @@ enum IntrStatus_bits {
...
@@ -486,11 +493,11 @@ enum IntrStatus_bits {
/*
/*
* Default Interrupts:
* Default Interrupts:
* Rx OK, Rx Packet Error, Rx Overrun,
* Rx OK, Rx Packet Error, Rx Overrun,
* Tx OK, Tx Packet Error, Tx Underrun,
* Tx OK, Tx Packet Error, Tx Underrun,
* MIB Service, Phy Interrupt, High Bits,
* MIB Service, Phy Interrupt, High Bits,
* Rx Status FIFO overrun,
* Rx Status FIFO overrun,
* Received Target Abort, Received Master Abort,
* Received Target Abort, Received Master Abort,
* Signalled System Error, Received Parity Error
* Signalled System Error, Received Parity Error
*/
*/
#define DEFAULT_INTR 0x00f1cd65
#define DEFAULT_INTR 0x00f1cd65
...
@@ -587,8 +594,11 @@ enum PhyCtrl_bits {
...
@@ -587,8 +594,11 @@ enum PhyCtrl_bits {
PhyAddrMask
=
0xf
,
PhyAddrMask
=
0xf
,
};
};
#define SRR_REV_C 0x0302
/* values we might find in the silicon revision register */
#define SRR_REV_D 0x0403
#define SRR_DP83815_C 0x0302
#define SRR_DP83815_D 0x0403
#define SRR_DP83816_A4 0x0504
#define SRR_DP83816_A5 0x0505
/* The Rx and Tx buffer descriptors. */
/* The Rx and Tx buffer descriptors. */
/* Note that using only 32 bit fields simplifies conversion to big-endian
/* Note that using only 32 bit fields simplifies conversion to big-endian
...
@@ -603,14 +613,14 @@ struct netdev_desc {
...
@@ -603,14 +613,14 @@ struct netdev_desc {
/* Bits in network_desc.status */
/* Bits in network_desc.status */
enum
desc_status_bits
{
enum
desc_status_bits
{
DescOwn
=
0x80000000
,
DescMore
=
0x40000000
,
DescIntr
=
0x20000000
,
DescOwn
=
0x80000000
,
DescMore
=
0x40000000
,
DescIntr
=
0x20000000
,
DescNoCRC
=
0x10000000
,
DescPktOK
=
0x08000000
,
DescNoCRC
=
0x10000000
,
DescPktOK
=
0x08000000
,
DescSizeMask
=
0xfff
,
DescSizeMask
=
0xfff
,
DescTxAbort
=
0x04000000
,
DescTxFIFO
=
0x02000000
,
DescTxAbort
=
0x04000000
,
DescTxFIFO
=
0x02000000
,
DescTxCarrier
=
0x01000000
,
DescTxDefer
=
0x00800000
,
DescTxCarrier
=
0x01000000
,
DescTxDefer
=
0x00800000
,
DescTxExcDefer
=
0x00400000
,
DescTxOOWCol
=
0x00200000
,
DescTxExcDefer
=
0x00400000
,
DescTxOOWCol
=
0x00200000
,
DescTxExcColl
=
0x00100000
,
DescTxCollCount
=
0x000f0000
,
DescTxExcColl
=
0x00100000
,
DescTxCollCount
=
0x000f0000
,
DescRxAbort
=
0x04000000
,
DescRxOver
=
0x02000000
,
DescRxAbort
=
0x04000000
,
DescRxOver
=
0x02000000
,
DescRxDest
=
0x01800000
,
DescRxLong
=
0x00400000
,
DescRxDest
=
0x01800000
,
DescRxLong
=
0x00400000
,
DescRxRunt
=
0x00200000
,
DescRxInvalid
=
0x00100000
,
DescRxRunt
=
0x00200000
,
DescRxInvalid
=
0x00100000
,
...
@@ -619,30 +629,36 @@ enum desc_status_bits {
...
@@ -619,30 +629,36 @@ enum desc_status_bits {
};
};
struct
netdev_private
{
struct
netdev_private
{
/* Descriptor rings first for alignment
.
*/
/* Descriptor rings first for alignment */
dma_addr_t
ring_dma
;
dma_addr_t
ring_dma
;
struct
netdev_desc
*
rx_ring
;
struct
netdev_desc
*
rx_ring
;
struct
netdev_desc
*
tx_ring
;
struct
netdev_desc
*
tx_ring
;
/* The addresses of receive-in-place skbuffs
.
*/
/* The addresses of receive-in-place skbuffs */
struct
sk_buff
*
rx_skbuff
[
RX_RING_SIZE
];
struct
sk_buff
*
rx_skbuff
[
RX_RING_SIZE
];
dma_addr_t
rx_dma
[
RX_RING_SIZE
];
dma_addr_t
rx_dma
[
RX_RING_SIZE
];
/*
The saved address of a sent-in-place packet/buffer, for later free().
*/
/*
address of a sent-in-place packet/buffer, for later free()
*/
struct
sk_buff
*
tx_skbuff
[
TX_RING_SIZE
];
struct
sk_buff
*
tx_skbuff
[
TX_RING_SIZE
];
dma_addr_t
tx_dma
[
TX_RING_SIZE
];
dma_addr_t
tx_dma
[
TX_RING_SIZE
];
struct
net_device_stats
stats
;
struct
net_device_stats
stats
;
struct
timer_list
timer
;
/* Media monitoring timer. */
/* Media monitoring timer */
/* Frequently used values: keep some adjacent for cache effect. */
struct
timer_list
timer
;
/* Frequently used values: keep some adjacent for cache effect */
struct
pci_dev
*
pci_dev
;
struct
pci_dev
*
pci_dev
;
struct
netdev_desc
*
rx_head_desc
;
struct
netdev_desc
*
rx_head_desc
;
unsigned
int
cur_rx
,
dirty_rx
;
/* Producer/consumer ring indices */
/* Producer/consumer ring indices */
unsigned
int
cur_rx
,
dirty_rx
;
unsigned
int
cur_tx
,
dirty_tx
;
unsigned
int
cur_tx
,
dirty_tx
;
unsigned
int
rx_buf_sz
;
/* Based on MTU+slack. */
/* Based on MTU+slack. */
/* These values are keep track of the transceiver/media in use. */
unsigned
int
rx_buf_sz
;
int
oom
;
/* Do not touch the nic registers */
int
hands_off
;
/* These values are keep track of the transceiver/media in use */
unsigned
int
full_duplex
;
unsigned
int
full_duplex
;
/* Rx filter
.
*/
/* Rx filter */
u32
cur_rx_mode
;
u32
cur_rx_mode
;
u32
rx_filter
[
16
];
u32
rx_filter
[
16
];
/* FIFO and PCI burst thresholds
.
*/
/* FIFO and PCI burst thresholds */
u32
tx_config
,
rx_config
;
u32
tx_config
,
rx_config
;
/* original contents of ClkRun register */
/* original contents of ClkRun register */
u32
SavedClkRun
;
u32
SavedClkRun
;
...
@@ -650,8 +666,8 @@ struct netdev_private {
...
@@ -650,8 +666,8 @@ struct netdev_private {
u32
srr
;
u32
srr
;
/* expected DSPCFG value */
/* expected DSPCFG value */
u16
dspcfg
;
u16
dspcfg
;
/* MII transceiver section
.
*/
/* MII transceiver section */
u16
advertising
;
/* NWay media advertisement */
u16
advertising
;
unsigned
int
iosize
;
unsigned
int
iosize
;
spinlock_t
lock
;
spinlock_t
lock
;
u32
msg_enable
;
u32
msg_enable
;
...
@@ -668,11 +684,15 @@ static void do_cable_magic(struct net_device *dev);
...
@@ -668,11 +684,15 @@ static void do_cable_magic(struct net_device *dev);
static
void
undo_cable_magic
(
struct
net_device
*
dev
);
static
void
undo_cable_magic
(
struct
net_device
*
dev
);
static
void
check_link
(
struct
net_device
*
dev
);
static
void
check_link
(
struct
net_device
*
dev
);
static
void
netdev_timer
(
unsigned
long
data
);
static
void
netdev_timer
(
unsigned
long
data
);
static
void
dump_ring
(
struct
net_device
*
dev
);
static
void
tx_timeout
(
struct
net_device
*
dev
);
static
void
tx_timeout
(
struct
net_device
*
dev
);
static
int
alloc_ring
(
struct
net_device
*
dev
);
static
int
alloc_ring
(
struct
net_device
*
dev
);
static
void
refill_rx
(
struct
net_device
*
dev
);
static
void
init_ring
(
struct
net_device
*
dev
);
static
void
init_ring
(
struct
net_device
*
dev
);
static
void
drain_tx
(
struct
net_device
*
dev
);
static
void
drain_ring
(
struct
net_device
*
dev
);
static
void
drain_ring
(
struct
net_device
*
dev
);
static
void
free_ring
(
struct
net_device
*
dev
);
static
void
free_ring
(
struct
net_device
*
dev
);
static
void
reinit_ring
(
struct
net_device
*
dev
);
static
void
init_registers
(
struct
net_device
*
dev
);
static
void
init_registers
(
struct
net_device
*
dev
);
static
int
start_tx
(
struct
sk_buff
*
skb
,
struct
net_device
*
dev
);
static
int
start_tx
(
struct
sk_buff
*
skb
,
struct
net_device
*
dev
);
static
void
intr_handler
(
int
irq
,
void
*
dev_instance
,
struct
pt_regs
*
regs
);
static
void
intr_handler
(
int
irq
,
void
*
dev_instance
,
struct
pt_regs
*
regs
);
...
@@ -695,9 +715,9 @@ static int netdev_close(struct net_device *dev);
...
@@ -695,9 +715,9 @@ static int netdev_close(struct net_device *dev);
static
int
netdev_get_regs
(
struct
net_device
*
dev
,
u8
*
buf
);
static
int
netdev_get_regs
(
struct
net_device
*
dev
,
u8
*
buf
);
static
int
netdev_get_eeprom
(
struct
net_device
*
dev
,
u8
*
buf
);
static
int
netdev_get_eeprom
(
struct
net_device
*
dev
,
u8
*
buf
);
static
int
__devinit
natsemi_probe1
(
struct
pci_dev
*
pdev
,
static
int
__devinit
natsemi_probe1
(
struct
pci_dev
*
pdev
,
const
struct
pci_device_id
*
ent
)
const
struct
pci_device_id
*
ent
)
{
{
struct
net_device
*
dev
;
struct
net_device
*
dev
;
struct
netdev_private
*
np
;
struct
netdev_private
*
np
;
...
@@ -777,6 +797,7 @@ static int __devinit natsemi_probe1 (struct pci_dev *pdev,
...
@@ -777,6 +797,7 @@ static int __devinit natsemi_probe1 (struct pci_dev *pdev,
np
->
iosize
=
iosize
;
np
->
iosize
=
iosize
;
spin_lock_init
(
&
np
->
lock
);
spin_lock_init
(
&
np
->
lock
);
np
->
msg_enable
=
debug
;
np
->
msg_enable
=
debug
;
np
->
hands_off
=
0
;
/* Reset the chip to erase previous misconfiguration. */
/* Reset the chip to erase previous misconfiguration. */
natsemi_reload_eeprom
(
dev
);
natsemi_reload_eeprom
(
dev
);
...
@@ -791,7 +812,7 @@ static int __devinit natsemi_probe1 (struct pci_dev *pdev,
...
@@ -791,7 +812,7 @@ static int __devinit natsemi_probe1 (struct pci_dev *pdev,
if
(
option
&
0x200
)
if
(
option
&
0x200
)
np
->
full_duplex
=
1
;
np
->
full_duplex
=
1
;
if
(
option
&
15
)
if
(
option
&
15
)
printk
(
KERN_INFO
printk
(
KERN_INFO
"%s: ignoring user supplied media type %d"
,
"%s: ignoring user supplied media type %d"
,
dev
->
name
,
option
&
15
);
dev
->
name
,
option
&
15
);
}
}
...
@@ -823,27 +844,28 @@ static int __devinit natsemi_probe1 (struct pci_dev *pdev,
...
@@ -823,27 +844,28 @@ static int __devinit natsemi_probe1 (struct pci_dev *pdev,
if
(
netif_msg_drv
(
np
))
{
if
(
netif_msg_drv
(
np
))
{
printk
(
KERN_INFO
"%s: %s at %#08lx, "
,
printk
(
KERN_INFO
"%s: %s at %#08lx, "
,
dev
->
name
,
natsemi_pci_info
[
chip_idx
].
name
,
ioaddr
);
dev
->
name
,
natsemi_pci_info
[
chip_idx
].
name
,
ioaddr
);
for
(
i
=
0
;
i
<
ETH_ALEN
-
1
;
i
++
)
for
(
i
=
0
;
i
<
ETH_ALEN
-
1
;
i
++
)
printk
(
"%02x:"
,
dev
->
dev_addr
[
i
]);
printk
(
"%02x:"
,
dev
->
dev_addr
[
i
]);
printk
(
"%02x, IRQ %d.
\n
"
,
dev
->
dev_addr
[
i
],
irq
);
printk
(
"%02x, IRQ %d.
\n
"
,
dev
->
dev_addr
[
i
],
irq
);
}
}
np
->
advertising
=
mdio_read
(
dev
,
1
,
MII_ADVERTISE
);
np
->
advertising
=
mdio_read
(
dev
,
1
,
MII_ADVERTISE
);
if
((
readl
(
ioaddr
+
ChipConfig
)
&
0xe000
)
!=
0xe000
if
((
readl
(
ioaddr
+
ChipConfig
)
&
0xe000
)
!=
0xe000
&&
netif_msg_probe
(
np
))
{
&&
netif_msg_probe
(
np
))
{
u32
chip_config
=
readl
(
ioaddr
+
ChipConfig
);
u32
chip_config
=
readl
(
ioaddr
+
ChipConfig
);
printk
(
KERN_INFO
"%s: Transceiver default autonegotiation %s "
printk
(
KERN_INFO
"%s: Transceiver default autonegotiation %s "
"10%s %s duplex.
\n
"
,
"10%s %s duplex.
\n
"
,
dev
->
name
,
dev
->
name
,
chip_config
&
CfgAnegEnable
?
"enabled, advertise"
:
"disabled, force"
,
chip_config
&
CfgAnegEnable
?
chip_config
&
CfgAneg100
?
"0"
:
""
,
"enabled, advertise"
:
"disabled, force"
,
chip_config
&
CfgAnegFull
?
"full"
:
"half"
);
chip_config
&
CfgAneg100
?
"0"
:
""
,
chip_config
&
CfgAnegFull
?
"full"
:
"half"
);
}
}
if
(
netif_msg_probe
(
np
))
if
(
netif_msg_probe
(
np
))
printk
(
KERN_INFO
printk
(
KERN_INFO
"%s: Transceiver status %#04x advertising %#04x.
\n
"
,
"%s: Transceiver status %#04x advertising %#04x.
\n
"
,
dev
->
name
,
mdio_read
(
dev
,
1
,
MII_BMSR
),
dev
->
name
,
mdio_read
(
dev
,
1
,
MII_BMSR
),
np
->
advertising
);
np
->
advertising
);
/* save the silicon revision for later querying */
/* save the silicon revision for later querying */
...
@@ -856,7 +878,7 @@ static int __devinit natsemi_probe1 (struct pci_dev *pdev,
...
@@ -856,7 +878,7 @@ static int __devinit natsemi_probe1 (struct pci_dev *pdev,
return
0
;
return
0
;
}
}
/* Read the EEPROM and MII Management Data I/O (MDIO) interfaces.
/* Read the EEPROM and MII Management Data I/O (MDIO) interfaces.
The EEPROM code is for the common 93c06/46 EEPROMs with 6 bit addresses. */
The EEPROM code is for the common 93c06/46 EEPROMs with 6 bit addresses. */
...
@@ -910,9 +932,9 @@ static int eeprom_read(long addr, int location)
...
@@ -910,9 +932,9 @@ static int eeprom_read(long addr, int location)
return
retval
;
return
retval
;
}
}
/*
MII transceiver control section.
/* MII transceiver control section.
The 83815 series has an internal transceiver, and we present the
*
The 83815 series has an internal transceiver, and we present the
management registers as if they were MII connected. */
*
management registers as if they were MII connected. */
static
int
mdio_read
(
struct
net_device
*
dev
,
int
phy_id
,
int
reg
)
static
int
mdio_read
(
struct
net_device
*
dev
,
int
phy_id
,
int
reg
)
{
{
...
@@ -950,11 +972,11 @@ static void natsemi_reset(struct net_device *dev)
...
@@ -950,11 +972,11 @@ static void natsemi_reset(struct net_device *dev)
u16
sopass
[
3
];
u16
sopass
[
3
];
struct
netdev_private
*
np
=
dev
->
priv
;
struct
netdev_private
*
np
=
dev
->
priv
;
/*
/*
* Resetting the chip causes some registers to be lost.
* Resetting the chip causes some registers to be lost.
* Natsemi suggests NOT reloading the EEPROM while live, so instead
* Natsemi suggests NOT reloading the EEPROM while live, so instead
* we save the state that would have been loaded from EEPROM
* we save the state that would have been loaded from EEPROM
* on a normal power-up (see the spec EEPROM map). This assumes
* on a normal power-up (see the spec EEPROM map). This assumes
* whoever calls this will follow up with init_registers() eventually.
* whoever calls this will follow up with init_registers() eventually.
*/
*/
...
@@ -982,12 +1004,12 @@ static void natsemi_reset(struct net_device *dev)
...
@@ -982,12 +1004,12 @@ static void natsemi_reset(struct net_device *dev)
break
;
break
;
udelay
(
5
);
udelay
(
5
);
}
}
if
(
i
==
NATSEMI_HW_TIMEOUT
&&
netif_msg_hw
(
np
)
)
{
if
(
i
==
NATSEMI_HW_TIMEOUT
)
{
printk
(
KERN_
INFO
"%s: reset did not complete in %d usec.
\n
"
,
printk
(
KERN_
WARN
"%s: reset did not complete in %d usec.
\n
"
,
dev
->
name
,
i
*
5
);
dev
->
name
,
i
*
5
);
}
else
if
(
netif_msg_hw
(
np
))
{
}
else
if
(
netif_msg_hw
(
np
))
{
printk
(
KERN_DEBUG
"%s: reset completed in %d usec.
\n
"
,
printk
(
KERN_DEBUG
"%s: reset completed in %d usec.
\n
"
,
dev
->
name
,
i
*
5
);
dev
->
name
,
i
*
5
);
}
}
/* restore CFG */
/* restore CFG */
...
@@ -998,7 +1020,7 @@ static void natsemi_reset(struct net_device *dev)
...
@@ -998,7 +1020,7 @@ static void natsemi_reset(struct net_device *dev)
writel
(
wcsr
,
dev
->
base_addr
+
WOLCmd
);
writel
(
wcsr
,
dev
->
base_addr
+
WOLCmd
);
/* read RFCR */
/* read RFCR */
rfcr
|=
readl
(
dev
->
base_addr
+
RxFilterAddr
)
&
~
RFCR_RESET_SAVE
;
rfcr
|=
readl
(
dev
->
base_addr
+
RxFilterAddr
)
&
~
RFCR_RESET_SAVE
;
/* restore PMATCH */
/* restore PMATCH */
for
(
i
=
0
;
i
<
3
;
i
++
)
{
for
(
i
=
0
;
i
<
3
;
i
++
)
{
writel
(
i
*
2
,
dev
->
base_addr
+
RxFilterAddr
);
writel
(
i
*
2
,
dev
->
base_addr
+
RxFilterAddr
);
writew
(
pmatch
[
i
],
dev
->
base_addr
+
RxFilterData
);
writew
(
pmatch
[
i
],
dev
->
base_addr
+
RxFilterData
);
...
@@ -1009,7 +1031,6 @@ static void natsemi_reset(struct net_device *dev)
...
@@ -1009,7 +1031,6 @@ static void natsemi_reset(struct net_device *dev)
}
}
/* restore RFCR */
/* restore RFCR */
writel
(
rfcr
,
dev
->
base_addr
+
RxFilterAddr
);
writel
(
rfcr
,
dev
->
base_addr
+
RxFilterAddr
);
}
}
static
void
natsemi_reload_eeprom
(
struct
net_device
*
dev
)
static
void
natsemi_reload_eeprom
(
struct
net_device
*
dev
)
...
@@ -1019,16 +1040,16 @@ static void natsemi_reload_eeprom(struct net_device *dev)
...
@@ -1019,16 +1040,16 @@ static void natsemi_reload_eeprom(struct net_device *dev)
writel
(
EepromReload
,
dev
->
base_addr
+
PCIBusCfg
);
writel
(
EepromReload
,
dev
->
base_addr
+
PCIBusCfg
);
for
(
i
=
0
;
i
<
NATSEMI_HW_TIMEOUT
;
i
++
)
{
for
(
i
=
0
;
i
<
NATSEMI_HW_TIMEOUT
;
i
++
)
{
udelay
(
50
);
if
(
!
(
readl
(
dev
->
base_addr
+
PCIBusCfg
)
&
EepromReload
))
if
(
!
(
readl
(
dev
->
base_addr
+
PCIBusCfg
)
&
EepromReload
))
break
;
break
;
udelay
(
5
);
}
}
if
(
i
==
NATSEMI_HW_TIMEOUT
&&
netif_msg_hw
(
np
)
)
{
if
(
i
==
NATSEMI_HW_TIMEOUT
)
{
printk
(
KERN_
INFO
"%s: EEPROM did not reload in %d usec.
\n
"
,
printk
(
KERN_
WARN
"%s: EEPROM did not reload in %d usec.
\n
"
,
dev
->
name
,
i
*
5
);
dev
->
name
,
i
*
50
);
}
else
if
(
netif_msg_hw
(
np
))
{
}
else
if
(
netif_msg_hw
(
np
))
{
printk
(
KERN_DEBUG
"%s: EEPROM reloaded in %d usec.
\n
"
,
printk
(
KERN_DEBUG
"%s: EEPROM reloaded in %d usec.
\n
"
,
dev
->
name
,
i
*
5
);
dev
->
name
,
i
*
50
);
}
}
}
}
...
@@ -1044,12 +1065,12 @@ static void natsemi_stop_rxtx(struct net_device *dev)
...
@@ -1044,12 +1065,12 @@ static void natsemi_stop_rxtx(struct net_device *dev)
break
;
break
;
udelay
(
5
);
udelay
(
5
);
}
}
if
(
i
==
NATSEMI_HW_TIMEOUT
&&
netif_msg_hw
(
np
)
)
{
if
(
i
==
NATSEMI_HW_TIMEOUT
)
{
printk
(
KERN_
INFO
"%s: Tx/Rx process did not stop in %d usec.
\n
"
,
printk
(
KERN_
WARN
"%s: Tx/Rx process did not stop in %d usec.
\n
"
,
dev
->
name
,
i
*
5
);
dev
->
name
,
i
*
5
);
}
else
if
(
netif_msg_hw
(
np
))
{
}
else
if
(
netif_msg_hw
(
np
))
{
printk
(
KERN_DEBUG
"%s: Tx/Rx process stopped in %d usec.
\n
"
,
printk
(
KERN_DEBUG
"%s: Tx/Rx process stopped in %d usec.
\n
"
,
dev
->
name
,
i
*
5
);
dev
->
name
,
i
*
5
);
}
}
}
}
...
@@ -1067,7 +1088,7 @@ static int netdev_open(struct net_device *dev)
...
@@ -1067,7 +1088,7 @@ static int netdev_open(struct net_device *dev)
if
(
netif_msg_ifup
(
np
))
if
(
netif_msg_ifup
(
np
))
printk
(
KERN_DEBUG
"%s: netdev_open() irq %d.
\n
"
,
printk
(
KERN_DEBUG
"%s: netdev_open() irq %d.
\n
"
,
dev
->
name
,
dev
->
irq
);
dev
->
name
,
dev
->
irq
);
i
=
alloc_ring
(
dev
);
i
=
alloc_ring
(
dev
);
if
(
i
<
0
)
{
if
(
i
<
0
)
{
free_irq
(
dev
->
irq
,
dev
);
free_irq
(
dev
->
irq
,
dev
);
...
@@ -1076,13 +1097,21 @@ static int netdev_open(struct net_device *dev)
...
@@ -1076,13 +1097,21 @@ static int netdev_open(struct net_device *dev)
init_ring
(
dev
);
init_ring
(
dev
);
spin_lock_irq
(
&
np
->
lock
);
spin_lock_irq
(
&
np
->
lock
);
init_registers
(
dev
);
init_registers
(
dev
);
/* now set the MAC address according to dev->dev_addr */
for
(
i
=
0
;
i
<
3
;
i
++
)
{
u16
mac
=
(
dev
->
dev_addr
[
2
*
i
+
1
]
<<
8
)
+
dev
->
dev_addr
[
2
*
i
];
writel
(
i
*
2
,
ioaddr
+
RxFilterAddr
);
writew
(
mac
,
ioaddr
+
RxFilterData
);
}
writel
(
np
->
cur_rx_mode
,
ioaddr
+
RxFilterAddr
);
spin_unlock_irq
(
&
np
->
lock
);
spin_unlock_irq
(
&
np
->
lock
);
netif_start_queue
(
dev
);
netif_start_queue
(
dev
);
if
(
netif_msg_ifup
(
np
))
if
(
netif_msg_ifup
(
np
))
printk
(
KERN_DEBUG
"%s: Done netdev_open(), status: %#08x.
\n
"
,
printk
(
KERN_DEBUG
"%s: Done netdev_open(), status: %#08x.
\n
"
,
dev
->
name
,
(
int
)
readl
(
ioaddr
+
ChipCmd
));
dev
->
name
,
(
int
)
readl
(
ioaddr
+
ChipCmd
));
/* Set the timer to check for link beat. */
/* Set the timer to check for link beat. */
init_timer
(
&
np
->
timer
);
init_timer
(
&
np
->
timer
);
...
@@ -1096,6 +1125,11 @@ static int netdev_open(struct net_device *dev)
...
@@ -1096,6 +1125,11 @@ static int netdev_open(struct net_device *dev)
static
void
do_cable_magic
(
struct
net_device
*
dev
)
static
void
do_cable_magic
(
struct
net_device
*
dev
)
{
{
struct
netdev_private
*
np
=
dev
->
priv
;
if
(
np
->
srr
>=
SRR_DP83816_A5
)
return
;
/*
/*
* 100 MBit links with short cables can trip an issue with the chip.
* 100 MBit links with short cables can trip an issue with the chip.
* The problem manifests as lots of CRC errors and/or flickering
* The problem manifests as lots of CRC errors and/or flickering
...
@@ -1134,6 +1168,9 @@ static void undo_cable_magic(struct net_device *dev)
...
@@ -1134,6 +1168,9 @@ static void undo_cable_magic(struct net_device *dev)
u16
data
;
u16
data
;
struct
netdev_private
*
np
=
dev
->
priv
;
struct
netdev_private
*
np
=
dev
->
priv
;
if
(
np
->
srr
>=
SRR_DP83816_A5
)
return
;
writew
(
1
,
dev
->
base_addr
+
PGSEL
);
writew
(
1
,
dev
->
base_addr
+
PGSEL
);
/* make sure the lock bit is clear */
/* make sure the lock bit is clear */
data
=
readw
(
dev
->
base_addr
+
DSPCFG
);
data
=
readw
(
dev
->
base_addr
+
DSPCFG
);
...
@@ -1152,7 +1189,7 @@ static void check_link(struct net_device *dev)
...
@@ -1152,7 +1189,7 @@ static void check_link(struct net_device *dev)
if
(
!
(
chipcfg
&
CfgLink
))
{
if
(
!
(
chipcfg
&
CfgLink
))
{
if
(
netif_carrier_ok
(
dev
))
{
if
(
netif_carrier_ok
(
dev
))
{
if
(
netif_msg_link
(
np
))
if
(
netif_msg_link
(
np
))
printk
(
KERN_NOTICE
"%s: link down.
\n
"
,
printk
(
KERN_NOTICE
"%s: link down.
\n
"
,
dev
->
name
);
dev
->
name
);
netif_carrier_off
(
dev
);
netif_carrier_off
(
dev
);
undo_cable_magic
(
dev
);
undo_cable_magic
(
dev
);
...
@@ -1171,7 +1208,7 @@ static void check_link(struct net_device *dev)
...
@@ -1171,7 +1208,7 @@ static void check_link(struct net_device *dev)
/* if duplex is set then bit 28 must be set, too */
/* if duplex is set then bit 28 must be set, too */
if
(
duplex
^
!!
(
np
->
rx_config
&
RxAcceptTx
))
{
if
(
duplex
^
!!
(
np
->
rx_config
&
RxAcceptTx
))
{
if
(
netif_msg_link
(
np
))
if
(
netif_msg_link
(
np
))
printk
(
KERN_INFO
printk
(
KERN_INFO
"%s: Setting %s-duplex based on negotiated "
"%s: Setting %s-duplex based on negotiated "
"link capability.
\n
"
,
dev
->
name
,
"link capability.
\n
"
,
dev
->
name
,
duplex
?
"full"
:
"half"
);
duplex
?
"full"
:
"half"
);
...
@@ -1199,16 +1236,16 @@ static void init_registers(struct net_device *dev)
...
@@ -1199,16 +1236,16 @@ static void init_registers(struct net_device *dev)
udelay
(
10
);
udelay
(
10
);
}
}
if
(
i
==
NATSEMI_HW_TIMEOUT
&&
netif_msg_link
(
np
))
{
if
(
i
==
NATSEMI_HW_TIMEOUT
&&
netif_msg_link
(
np
))
{
printk
(
KERN_INFO
printk
(
KERN_INFO
"%s: autonegotiation did not complete in %d usec.
\n
"
,
"%s: autonegotiation did not complete in %d usec.
\n
"
,
dev
->
name
,
i
*
10
);
dev
->
name
,
i
*
10
);
}
}
/* On page 78 of the spec, they recommend some settings for "optimum
/* On page 78 of the spec, they recommend some settings for "optimum
performance" to be done in sequence. These settings optimize some
performance" to be done in sequence. These settings optimize some
of the 100Mbit autodetection circuitry. They say we only want to
of the 100Mbit autodetection circuitry. They say we only want to
do this for rev C of the chip, but engineers at NSC (Bradley
do this for rev C of the chip, but engineers at NSC (Bradley
Kennedy) recommends always setting them. If you don't, you get
Kennedy) recommends always setting them. If you don't, you get
errors on some autonegotiations that make the device unusable.
errors on some autonegotiations that make the device unusable.
*/
*/
writew
(
1
,
ioaddr
+
PGSEL
);
writew
(
1
,
ioaddr
+
PGSEL
);
...
@@ -1230,14 +1267,14 @@ static void init_registers(struct net_device *dev)
...
@@ -1230,14 +1267,14 @@ static void init_registers(struct net_device *dev)
readl
(
ioaddr
+
IntrStatus
);
readl
(
ioaddr
+
IntrStatus
);
writel
(
np
->
ring_dma
,
ioaddr
+
RxRingPtr
);
writel
(
np
->
ring_dma
,
ioaddr
+
RxRingPtr
);
writel
(
np
->
ring_dma
+
RX_RING_SIZE
*
sizeof
(
struct
netdev_desc
),
writel
(
np
->
ring_dma
+
RX_RING_SIZE
*
sizeof
(
struct
netdev_desc
),
ioaddr
+
TxRingPtr
);
ioaddr
+
TxRingPtr
);
/* Initialize other registers.
/* Initialize other registers.
* Configure the PCI bus bursts and FIFO thresholds.
* Configure the PCI bus bursts and FIFO thresholds.
* Configure for standard, in-spec Ethernet.
* Configure for standard, in-spec Ethernet.
* Start with half-duplex. check_link will update
* Start with half-duplex. check_link will update
* to the correct settings.
* to the correct settings.
*/
*/
/* DRTH: 2: start tx if 64 bytes are in the fifo
/* DRTH: 2: start tx if 64 bytes are in the fifo
...
@@ -1259,13 +1296,13 @@ static void init_registers(struct net_device *dev)
...
@@ -1259,13 +1296,13 @@ static void init_registers(struct net_device *dev)
/* Disable PME:
/* Disable PME:
* The PME bit is initialized from the EEPROM contents.
* The PME bit is initialized from the EEPROM contents.
* PCI cards probably have PME disabled, but motherboard
* PCI cards probably have PME disabled, but motherboard
* implementations may have PME set to enable WakeOnLan.
* implementations may have PME set to enable WakeOnLan.
* With PME set the chip will scan incoming packets but
* With PME set the chip will scan incoming packets but
* nothing will be written to memory. */
* nothing will be written to memory. */
np
->
SavedClkRun
=
readl
(
ioaddr
+
ClkRun
);
np
->
SavedClkRun
=
readl
(
ioaddr
+
ClkRun
);
writel
(
np
->
SavedClkRun
&
~
PMEEnable
,
ioaddr
+
ClkRun
);
writel
(
np
->
SavedClkRun
&
~
PMEEnable
,
ioaddr
+
ClkRun
);
if
(
np
->
SavedClkRun
&
PMEStatus
&&
netif_msg_wol
(
np
))
{
if
(
np
->
SavedClkRun
&
PMEStatus
&&
netif_msg_wol
(
np
))
{
printk
(
KERN_NOTICE
"%s: Wake-up event %#08x
\n
"
,
printk
(
KERN_NOTICE
"%s: Wake-up event %#08x
\n
"
,
dev
->
name
,
readl
(
ioaddr
+
WOLCmd
));
dev
->
name
,
readl
(
ioaddr
+
WOLCmd
));
}
}
...
@@ -1281,13 +1318,16 @@ static void init_registers(struct net_device *dev)
...
@@ -1281,13 +1318,16 @@ static void init_registers(struct net_device *dev)
}
}
/*
/*
* netdev_timer:
* Purpose:
* Purpose:
* check for sudden death of the NIC:
* 1) check for link changes. Usually they are handled by the MII interrupt
*
* but it doesn't hurt to check twice.
* It seems that a reference set for this chip went out with incorrect info,
* 2) check for sudden death of the NIC:
* and there exist boards that aren't quite right. An unexpected voltage drop
* It seems that a reference set for this chip went out with incorrect info,
* can cause the PHY to get itself in a weird state (basically reset..).
* and there exist boards that aren't quite right. An unexpected voltage
* NOTE: this only seems to affect revC chips.
* drop can cause the PHY to get itself in a weird state (basically reset).
* NOTE: this only seems to affect revC chips.
* 3) check of death of the RX path due to OOM
*/
*/
static
void
netdev_timer
(
unsigned
long
data
)
static
void
netdev_timer
(
unsigned
long
data
)
{
{
...
@@ -1298,11 +1338,11 @@ static void netdev_timer(unsigned long data)
...
@@ -1298,11 +1338,11 @@ static void netdev_timer(unsigned long data)
u16
dspcfg
;
u16
dspcfg
;
if
(
netif_msg_timer
(
np
))
{
if
(
netif_msg_timer
(
np
))
{
/* DO NOT read the IntrStatus register,
/* DO NOT read the IntrStatus register,
* a read clears any pending interrupts.
* a read clears any pending interrupts.
*/
*/
printk
(
KERN_DEBUG
"%s: Media selection timer tick.
\n
"
,
printk
(
KERN_DEBUG
"%s: Media selection timer tick.
\n
"
,
dev
->
name
);
dev
->
name
);
}
}
spin_lock_irq
(
&
np
->
lock
);
spin_lock_irq
(
&
np
->
lock
);
...
@@ -1319,6 +1359,9 @@ static void netdev_timer(unsigned long data)
...
@@ -1319,6 +1359,9 @@ static void netdev_timer(unsigned long data)
"re-initializing
\n
"
,
dev
->
name
);
"re-initializing
\n
"
,
dev
->
name
);
disable_irq
(
dev
->
irq
);
disable_irq
(
dev
->
irq
);
spin_lock_irq
(
&
np
->
lock
);
spin_lock_irq
(
&
np
->
lock
);
natsemi_stop_rxtx
(
dev
);
dump_ring
(
dev
);
reinit_ring
(
dev
);
init_registers
(
dev
);
init_registers
(
dev
);
spin_unlock_irq
(
&
np
->
lock
);
spin_unlock_irq
(
&
np
->
lock
);
enable_irq
(
dev
->
irq
);
enable_irq
(
dev
->
irq
);
...
@@ -1332,6 +1375,17 @@ static void netdev_timer(unsigned long data)
...
@@ -1332,6 +1375,17 @@ static void netdev_timer(unsigned long data)
check_link
(
dev
);
check_link
(
dev
);
spin_unlock_irq
(
&
np
->
lock
);
spin_unlock_irq
(
&
np
->
lock
);
}
}
if
(
np
->
oom
)
{
disable_irq
(
dev
->
irq
);
np
->
oom
=
0
;
refill_rx
(
dev
);
enable_irq
(
dev
->
irq
);
if
(
!
np
->
oom
)
{
writel
(
RxOn
,
dev
->
base_addr
+
ChipCmd
);
}
else
{
next_tick
=
1
;
}
}
mod_timer
(
&
np
->
timer
,
jiffies
+
next_tick
);
mod_timer
(
&
np
->
timer
,
jiffies
+
next_tick
);
}
}
...
@@ -1344,16 +1398,16 @@ static void dump_ring(struct net_device *dev)
...
@@ -1344,16 +1398,16 @@ static void dump_ring(struct net_device *dev)
printk
(
KERN_DEBUG
" Tx ring at %p:
\n
"
,
np
->
tx_ring
);
printk
(
KERN_DEBUG
" Tx ring at %p:
\n
"
,
np
->
tx_ring
);
for
(
i
=
0
;
i
<
TX_RING_SIZE
;
i
++
)
{
for
(
i
=
0
;
i
<
TX_RING_SIZE
;
i
++
)
{
printk
(
KERN_DEBUG
" #%d desc. %#08x %#08x %#08x.
\n
"
,
printk
(
KERN_DEBUG
" #%d desc. %#08x %#08x %#08x.
\n
"
,
i
,
np
->
tx_ring
[
i
].
next_desc
,
i
,
np
->
tx_ring
[
i
].
next_desc
,
np
->
tx_ring
[
i
].
cmd_status
,
np
->
tx_ring
[
i
].
cmd_status
,
np
->
tx_ring
[
i
].
addr
);
np
->
tx_ring
[
i
].
addr
);
}
}
printk
(
KERN_DEBUG
" Rx ring %p:
\n
"
,
np
->
rx_ring
);
printk
(
KERN_DEBUG
" Rx ring %p:
\n
"
,
np
->
rx_ring
);
for
(
i
=
0
;
i
<
RX_RING_SIZE
;
i
++
)
{
for
(
i
=
0
;
i
<
RX_RING_SIZE
;
i
++
)
{
printk
(
KERN_DEBUG
" #%d desc. %#08x %#08x %#08x.
\n
"
,
printk
(
KERN_DEBUG
" #%d desc. %#08x %#08x %#08x.
\n
"
,
i
,
np
->
rx_ring
[
i
].
next_desc
,
i
,
np
->
rx_ring
[
i
].
next_desc
,
np
->
rx_ring
[
i
].
cmd_status
,
np
->
rx_ring
[
i
].
cmd_status
,
np
->
rx_ring
[
i
].
addr
);
np
->
rx_ring
[
i
].
addr
);
}
}
}
}
}
}
...
@@ -1365,22 +1419,21 @@ static void tx_timeout(struct net_device *dev)
...
@@ -1365,22 +1419,21 @@ static void tx_timeout(struct net_device *dev)
disable_irq
(
dev
->
irq
);
disable_irq
(
dev
->
irq
);
spin_lock_irq
(
&
np
->
lock
);
spin_lock_irq
(
&
np
->
lock
);
if
(
netif_device_present
(
dev
)
)
{
if
(
!
np
->
hands_off
)
{
if
(
netif_msg_tx_err
(
np
))
if
(
netif_msg_tx_err
(
np
))
printk
(
KERN_WARNING
printk
(
KERN_WARNING
"%s: Transmit timed out, status %#08x,"
"%s: Transmit timed out, status %#08x,"
" resetting...
\n
"
,
" resetting...
\n
"
,
dev
->
name
,
readl
(
ioaddr
+
IntrStatus
));
dev
->
name
,
readl
(
ioaddr
+
IntrStatus
));
dump_ring
(
dev
);
dump_ring
(
dev
);
natsemi_reset
(
dev
);
natsemi_reset
(
dev
);
drain_ring
(
dev
);
reinit_ring
(
dev
);
init_ring
(
dev
);
init_registers
(
dev
);
init_registers
(
dev
);
}
else
{
}
else
{
printk
(
KERN_WARNING
printk
(
KERN_WARNING
"%s: tx_timeout while in
suspended
state?
\n
"
,
"%s: tx_timeout while in
hands_off
state?
\n
"
,
dev
->
name
);
dev
->
name
);
}
}
spin_unlock_irq
(
&
np
->
lock
);
spin_unlock_irq
(
&
np
->
lock
);
enable_irq
(
dev
->
irq
);
enable_irq
(
dev
->
irq
);
...
@@ -1394,24 +1447,62 @@ static int alloc_ring(struct net_device *dev)
...
@@ -1394,24 +1447,62 @@ static int alloc_ring(struct net_device *dev)
{
{
struct
netdev_private
*
np
=
dev
->
priv
;
struct
netdev_private
*
np
=
dev
->
priv
;
np
->
rx_ring
=
pci_alloc_consistent
(
np
->
pci_dev
,
np
->
rx_ring
=
pci_alloc_consistent
(
np
->
pci_dev
,
sizeof
(
struct
netdev_desc
)
*
(
RX_RING_SIZE
+
TX_RING_SIZE
),
sizeof
(
struct
netdev_desc
)
*
(
RX_RING_SIZE
+
TX_RING_SIZE
),
&
np
->
ring_dma
);
&
np
->
ring_dma
);
if
(
!
np
->
rx_ring
)
if
(
!
np
->
rx_ring
)
return
-
ENOMEM
;
return
-
ENOMEM
;
np
->
tx_ring
=
&
np
->
rx_ring
[
RX_RING_SIZE
];
np
->
tx_ring
=
&
np
->
rx_ring
[
RX_RING_SIZE
];
return
0
;
return
0
;
}
}
static
void
refill_rx
(
struct
net_device
*
dev
)
{
struct
netdev_private
*
np
=
dev
->
priv
;
/* Refill the Rx ring buffers. */
for
(;
np
->
cur_rx
-
np
->
dirty_rx
>
0
;
np
->
dirty_rx
++
)
{
struct
sk_buff
*
skb
;
int
entry
=
np
->
dirty_rx
%
RX_RING_SIZE
;
if
(
np
->
rx_skbuff
[
entry
]
==
NULL
)
{
skb
=
dev_alloc_skb
(
np
->
rx_buf_sz
);
np
->
rx_skbuff
[
entry
]
=
skb
;
if
(
skb
==
NULL
)
break
;
/* Better luck next round. */
skb
->
dev
=
dev
;
/* Mark as being used by this device. */
np
->
rx_dma
[
entry
]
=
pci_map_single
(
np
->
pci_dev
,
skb
->
data
,
skb
->
len
,
PCI_DMA_FROMDEVICE
);
np
->
rx_ring
[
entry
].
addr
=
cpu_to_le32
(
np
->
rx_dma
[
entry
]);
}
np
->
rx_ring
[
entry
].
cmd_status
=
cpu_to_le32
(
np
->
rx_buf_sz
);
}
if
(
np
->
cur_rx
-
np
->
dirty_rx
==
RX_RING_SIZE
)
{
if
(
netif_msg_rx_err
(
np
))
printk
(
KERN_WARN
"%s: going OOM.
\n
"
,
dev
->
name
);
np
->
oom
=
1
;
}
}
/* Initialize the Rx and Tx rings, along with various 'dev' bits. */
/* Initialize the Rx and Tx rings, along with various 'dev' bits. */
static
void
init_ring
(
struct
net_device
*
dev
)
static
void
init_ring
(
struct
net_device
*
dev
)
{
{
struct
netdev_private
*
np
=
dev
->
priv
;
struct
netdev_private
*
np
=
dev
->
priv
;
int
i
;
int
i
;
np
->
cur_rx
=
np
->
cur_tx
=
0
;
/* 1) TX ring */
np
->
dirty_rx
=
np
->
dirty_tx
=
0
;
np
->
dirty_tx
=
np
->
cur_tx
=
0
;
for
(
i
=
0
;
i
<
TX_RING_SIZE
;
i
++
)
{
np
->
tx_skbuff
[
i
]
=
NULL
;
np
->
tx_ring
[
i
].
next_desc
=
cpu_to_le32
(
np
->
ring_dma
+
sizeof
(
struct
netdev_desc
)
*
((
i
+
1
)
%
TX_RING_SIZE
+
RX_RING_SIZE
));
np
->
tx_ring
[
i
].
cmd_status
=
0
;
}
/* 2) RX ring */
np
->
dirty_rx
=
0
;
np
->
cur_rx
=
RX_RING_SIZE
;
np
->
rx_buf_sz
=
(
dev
->
mtu
<=
1500
?
PKT_BUF_SZ
:
dev
->
mtu
+
32
);
np
->
rx_buf_sz
=
(
dev
->
mtu
<=
1500
?
PKT_BUF_SZ
:
dev
->
mtu
+
32
);
np
->
oom
=
0
;
np
->
rx_head_desc
=
&
np
->
rx_ring
[
0
];
np
->
rx_head_desc
=
&
np
->
rx_ring
[
0
];
/* Please be carefull before changing this loop - at least gcc-2.95.1
/* Please be carefull before changing this loop - at least gcc-2.95.1
...
@@ -1425,29 +1516,25 @@ static void init_ring(struct net_device *dev)
...
@@ -1425,29 +1516,25 @@ static void init_ring(struct net_device *dev)
np
->
rx_ring
[
i
].
cmd_status
=
cpu_to_le32
(
DescOwn
);
np
->
rx_ring
[
i
].
cmd_status
=
cpu_to_le32
(
DescOwn
);
np
->
rx_skbuff
[
i
]
=
NULL
;
np
->
rx_skbuff
[
i
]
=
NULL
;
}
}
refill_rx
(
dev
);
dump_ring
(
dev
);
}
/* Fill in the Rx buffers. Handle allocation failure gracefully. */
static
void
drain_tx
(
struct
net_device
*
dev
)
for
(
i
=
0
;
i
<
RX_RING_SIZE
;
i
++
)
{
{
struct
sk_buff
*
skb
=
dev_alloc_skb
(
np
->
rx_buf_sz
);
struct
netdev_private
*
np
=
dev
->
priv
;
np
->
rx_skbuff
[
i
]
=
skb
;
int
i
;
if
(
skb
==
NULL
)
break
;
skb
->
dev
=
dev
;
/* Mark as being used by this device. */
np
->
rx_dma
[
i
]
=
pci_map_single
(
np
->
pci_dev
,
skb
->
data
,
skb
->
len
,
PCI_DMA_FROMDEVICE
);
np
->
rx_ring
[
i
].
addr
=
cpu_to_le32
(
np
->
rx_dma
[
i
]);
np
->
rx_ring
[
i
].
cmd_status
=
cpu_to_le32
(
np
->
rx_buf_sz
);
}
np
->
dirty_rx
=
(
unsigned
int
)(
i
-
RX_RING_SIZE
);
for
(
i
=
0
;
i
<
TX_RING_SIZE
;
i
++
)
{
for
(
i
=
0
;
i
<
TX_RING_SIZE
;
i
++
)
{
if
(
np
->
tx_skbuff
[
i
])
{
pci_unmap_single
(
np
->
pci_dev
,
np
->
rx_dma
[
i
],
np
->
rx_skbuff
[
i
]
->
len
,
PCI_DMA_TODEVICE
);
dev_kfree_skb
(
np
->
tx_skbuff
[
i
]);
np
->
stats
.
tx_dropped
++
;
}
np
->
tx_skbuff
[
i
]
=
NULL
;
np
->
tx_skbuff
[
i
]
=
NULL
;
np
->
tx_ring
[
i
].
next_desc
=
cpu_to_le32
(
np
->
ring_dma
+
sizeof
(
struct
netdev_desc
)
*
((
i
+
1
)
%
TX_RING_SIZE
+
RX_RING_SIZE
));
np
->
tx_ring
[
i
].
cmd_status
=
0
;
}
}
dump_ring
(
dev
);
}
}
static
void
drain_ring
(
struct
net_device
*
dev
)
static
void
drain_ring
(
struct
net_device
*
dev
)
...
@@ -1461,31 +1548,43 @@ static void drain_ring(struct net_device *dev)
...
@@ -1461,31 +1548,43 @@ static void drain_ring(struct net_device *dev)
np
->
rx_ring
[
i
].
addr
=
0xBADF00D0
;
/* An invalid address. */
np
->
rx_ring
[
i
].
addr
=
0xBADF00D0
;
/* An invalid address. */
if
(
np
->
rx_skbuff
[
i
])
{
if
(
np
->
rx_skbuff
[
i
])
{
pci_unmap_single
(
np
->
pci_dev
,
pci_unmap_single
(
np
->
pci_dev
,
np
->
rx_dma
[
i
],
np
->
rx_dma
[
i
],
np
->
rx_skbuff
[
i
]
->
len
,
np
->
rx_skbuff
[
i
]
->
len
,
PCI_DMA_FROMDEVICE
);
PCI_DMA_FROMDEVICE
);
dev_kfree_skb
(
np
->
rx_skbuff
[
i
]);
dev_kfree_skb
(
np
->
rx_skbuff
[
i
]);
}
}
np
->
rx_skbuff
[
i
]
=
NULL
;
np
->
rx_skbuff
[
i
]
=
NULL
;
}
}
for
(
i
=
0
;
i
<
TX_RING_SIZE
;
i
++
)
{
drain_tx
(
dev
);
if
(
np
->
tx_skbuff
[
i
])
{
pci_unmap_single
(
np
->
pci_dev
,
np
->
rx_dma
[
i
],
np
->
rx_skbuff
[
i
]
->
len
,
PCI_DMA_TODEVICE
);
dev_kfree_skb
(
np
->
tx_skbuff
[
i
]);
}
np
->
tx_skbuff
[
i
]
=
NULL
;
}
}
}
static
void
free_ring
(
struct
net_device
*
dev
)
static
void
free_ring
(
struct
net_device
*
dev
)
{
{
struct
netdev_private
*
np
=
dev
->
priv
;
struct
netdev_private
*
np
=
dev
->
priv
;
pci_free_consistent
(
np
->
pci_dev
,
pci_free_consistent
(
np
->
pci_dev
,
sizeof
(
struct
netdev_desc
)
*
(
RX_RING_SIZE
+
TX_RING_SIZE
),
sizeof
(
struct
netdev_desc
)
*
(
RX_RING_SIZE
+
TX_RING_SIZE
),
np
->
rx_ring
,
np
->
ring_dma
);
np
->
rx_ring
,
np
->
ring_dma
);
}
static
void
reinit_ring
(
struct
net_device
*
dev
)
{
struct
netdev_private
*
np
=
dev
->
priv
;
int
i
;
/* drain TX ring */
drain_tx
(
dev
);
np
->
dirty_tx
=
np
->
cur_tx
=
0
;
for
(
i
=
0
;
i
<
TX_RING_SIZE
;
i
++
)
np
->
tx_ring
[
i
].
cmd_status
=
0
;
/* RX Ring */
np
->
dirty_rx
=
0
;
np
->
cur_rx
=
RX_RING_SIZE
;
np
->
rx_head_desc
=
&
np
->
rx_ring
[
0
];
/* Initialize all Rx descriptors. */
for
(
i
=
0
;
i
<
RX_RING_SIZE
;
i
++
)
np
->
rx_ring
[
i
].
cmd_status
=
cpu_to_le32
(
DescOwn
);
refill_rx
(
dev
);
}
}
static
int
start_tx
(
struct
sk_buff
*
skb
,
struct
net_device
*
dev
)
static
int
start_tx
(
struct
sk_buff
*
skb
,
struct
net_device
*
dev
)
...
@@ -1506,10 +1605,10 @@ static int start_tx(struct sk_buff *skb, struct net_device *dev)
...
@@ -1506,10 +1605,10 @@ static int start_tx(struct sk_buff *skb, struct net_device *dev)
np
->
tx_ring
[
entry
].
addr
=
cpu_to_le32
(
np
->
tx_dma
[
entry
]);
np
->
tx_ring
[
entry
].
addr
=
cpu_to_le32
(
np
->
tx_dma
[
entry
]);
spin_lock_irq
(
&
np
->
lock
);
spin_lock_irq
(
&
np
->
lock
);
if
(
netif_device_present
(
dev
)
)
{
if
(
!
np
->
hands_off
)
{
np
->
tx_ring
[
entry
].
cmd_status
=
cpu_to_le32
(
DescOwn
|
skb
->
len
);
np
->
tx_ring
[
entry
].
cmd_status
=
cpu_to_le32
(
DescOwn
|
skb
->
len
);
/* StrongARM: Explicitly cache flush np->tx_ring and
/* StrongARM: Explicitly cache flush np->tx_ring and
* skb->data,skb->len. */
* skb->data,skb->len. */
wmb
();
wmb
();
np
->
cur_tx
++
;
np
->
cur_tx
++
;
...
@@ -1530,7 +1629,7 @@ static int start_tx(struct sk_buff *skb, struct net_device *dev)
...
@@ -1530,7 +1629,7 @@ static int start_tx(struct sk_buff *skb, struct net_device *dev)
if
(
netif_msg_tx_queued
(
np
))
{
if
(
netif_msg_tx_queued
(
np
))
{
printk
(
KERN_DEBUG
"%s: Transmit frame #%d queued in slot %d.
\n
"
,
printk
(
KERN_DEBUG
"%s: Transmit frame #%d queued in slot %d.
\n
"
,
dev
->
name
,
np
->
cur_tx
,
entry
);
dev
->
name
,
np
->
cur_tx
,
entry
);
}
}
return
0
;
return
0
;
}
}
...
@@ -1544,7 +1643,7 @@ static void netdev_tx_done(struct net_device *dev)
...
@@ -1544,7 +1643,7 @@ static void netdev_tx_done(struct net_device *dev)
if
(
np
->
tx_ring
[
entry
].
cmd_status
&
cpu_to_le32
(
DescOwn
))
if
(
np
->
tx_ring
[
entry
].
cmd_status
&
cpu_to_le32
(
DescOwn
))
break
;
break
;
if
(
netif_msg_tx_done
(
np
))
if
(
netif_msg_tx_done
(
np
))
printk
(
KERN_DEBUG
printk
(
KERN_DEBUG
"%s: tx frame #%d finished, status %#08x.
\n
"
,
"%s: tx frame #%d finished, status %#08x.
\n
"
,
dev
->
name
,
np
->
dirty_tx
,
dev
->
name
,
np
->
dirty_tx
,
le32_to_cpu
(
np
->
tx_ring
[
entry
].
cmd_status
));
le32_to_cpu
(
np
->
tx_ring
[
entry
].
cmd_status
));
...
@@ -1552,14 +1651,15 @@ static void netdev_tx_done(struct net_device *dev)
...
@@ -1552,14 +1651,15 @@ static void netdev_tx_done(struct net_device *dev)
np
->
stats
.
tx_packets
++
;
np
->
stats
.
tx_packets
++
;
np
->
stats
.
tx_bytes
+=
np
->
tx_skbuff
[
entry
]
->
len
;
np
->
stats
.
tx_bytes
+=
np
->
tx_skbuff
[
entry
]
->
len
;
}
else
{
/* Various Tx errors */
}
else
{
/* Various Tx errors */
int
tx_status
=
le32_to_cpu
(
np
->
tx_ring
[
entry
].
cmd_status
);
int
tx_status
=
if
(
tx_status
&
(
DescTxAbort
|
DescTxExcColl
))
le32_to_cpu
(
np
->
tx_ring
[
entry
].
cmd_status
);
if
(
tx_status
&
(
DescTxAbort
|
DescTxExcColl
))
np
->
stats
.
tx_aborted_errors
++
;
np
->
stats
.
tx_aborted_errors
++
;
if
(
tx_status
&
DescTxFIFO
)
if
(
tx_status
&
DescTxFIFO
)
np
->
stats
.
tx_fifo_errors
++
;
np
->
stats
.
tx_fifo_errors
++
;
if
(
tx_status
&
DescTxCarrier
)
if
(
tx_status
&
DescTxCarrier
)
np
->
stats
.
tx_carrier_errors
++
;
np
->
stats
.
tx_carrier_errors
++
;
if
(
tx_status
&
DescTxOOWCol
)
if
(
tx_status
&
DescTxOOWCol
)
np
->
stats
.
tx_window_errors
++
;
np
->
stats
.
tx_window_errors
++
;
np
->
stats
.
tx_errors
++
;
np
->
stats
.
tx_errors
++
;
}
}
...
@@ -1586,15 +1686,17 @@ static void intr_handler(int irq, void *dev_instance, struct pt_regs *rgs)
...
@@ -1586,15 +1686,17 @@ static void intr_handler(int irq, void *dev_instance, struct pt_regs *rgs)
long
ioaddr
=
dev
->
base_addr
;
long
ioaddr
=
dev
->
base_addr
;
int
boguscnt
=
max_interrupt_work
;
int
boguscnt
=
max_interrupt_work
;
if
(
!
netif_device_present
(
dev
)
)
if
(
np
->
hands_off
)
return
;
return
;
do
{
do
{
/* Reading automatically acknowledges all int sources. */
/* Reading automatically acknowledges all int sources. */
u32
intr_status
=
readl
(
ioaddr
+
IntrStatus
);
u32
intr_status
=
readl
(
ioaddr
+
IntrStatus
);
if
(
netif_msg_intr
(
np
))
if
(
netif_msg_intr
(
np
))
printk
(
KERN_DEBUG
"%s: Interrupt, status %#08x.
\n
"
,
printk
(
KERN_DEBUG
dev
->
name
,
intr_status
);
"%s: Interrupt, status %#08x, mask %#08x.
\n
"
,
dev
->
name
,
intr_status
,
readl
(
ioaddr
+
IntrMask
));
if
(
intr_status
==
0
)
if
(
intr_status
==
0
)
break
;
break
;
...
@@ -1618,9 +1720,10 @@ static void intr_handler(int irq, void *dev_instance, struct pt_regs *rgs)
...
@@ -1618,9 +1720,10 @@ static void intr_handler(int irq, void *dev_instance, struct pt_regs *rgs)
if
(
--
boguscnt
<
0
)
{
if
(
--
boguscnt
<
0
)
{
if
(
netif_msg_intr
(
np
))
if
(
netif_msg_intr
(
np
))
printk
(
KERN_WARNING
printk
(
KERN_WARNING
"%s: Too much work at interrupt, "
"%s: Too much work at interrupt, "
"status=%#08x.
\n
"
,
dev
->
name
,
intr_status
);
"status=%#08x.
\n
"
,
dev
->
name
,
intr_status
);
break
;
break
;
}
}
}
while
(
1
);
}
while
(
1
);
...
@@ -1639,9 +1742,9 @@ static void netdev_rx(struct net_device *dev)
...
@@ -1639,9 +1742,9 @@ static void netdev_rx(struct net_device *dev)
s32
desc_status
=
le32_to_cpu
(
np
->
rx_head_desc
->
cmd_status
);
s32
desc_status
=
le32_to_cpu
(
np
->
rx_head_desc
->
cmd_status
);
/* If the driver owns the next entry it's a new packet. Send it up. */
/* If the driver owns the next entry it's a new packet. Send it up. */
while
(
desc_status
<
0
)
{
/* e.g. & DescOwn */
while
(
desc_status
<
0
)
{
/* e.g. & DescOwn */
if
(
netif_msg_rx_status
(
np
))
if
(
netif_msg_rx_status
(
np
))
printk
(
KERN_DEBUG
printk
(
KERN_DEBUG
" netdev_rx() entry %d status was %#08x.
\n
"
,
" netdev_rx() entry %d status was %#08x.
\n
"
,
entry
,
desc_status
);
entry
,
desc_status
);
if
(
--
boguscnt
<
0
)
if
(
--
boguscnt
<
0
)
...
@@ -1649,54 +1752,56 @@ static void netdev_rx(struct net_device *dev)
...
@@ -1649,54 +1752,56 @@ static void netdev_rx(struct net_device *dev)
if
((
desc_status
&
(
DescMore
|
DescPktOK
|
DescRxLong
))
!=
DescPktOK
){
if
((
desc_status
&
(
DescMore
|
DescPktOK
|
DescRxLong
))
!=
DescPktOK
){
if
(
desc_status
&
DescMore
)
{
if
(
desc_status
&
DescMore
)
{
if
(
netif_msg_rx_err
(
np
))
if
(
netif_msg_rx_err
(
np
))
printk
(
KERN_WARNING
printk
(
KERN_WARNING
"%s: Oversized(?) Ethernet "
"%s: Oversized(?) Ethernet "
"frame spanned multiple "
"frame spanned multiple "
"buffers, entry %#08x "
"buffers, entry %#08x "
"status %#08x.
\n
"
,
dev
->
name
,
"status %#08x.
\n
"
,
dev
->
name
,
np
->
cur_rx
,
desc_status
);
np
->
cur_rx
,
desc_status
);
np
->
stats
.
rx_length_errors
++
;
np
->
stats
.
rx_length_errors
++
;
}
else
{
}
else
{
/* There was an error. */
/* There was an error. */
np
->
stats
.
rx_errors
++
;
np
->
stats
.
rx_errors
++
;
if
(
desc_status
&
(
DescRxAbort
|
DescRxOver
))
if
(
desc_status
&
(
DescRxAbort
|
DescRxOver
))
np
->
stats
.
rx_over_errors
++
;
np
->
stats
.
rx_over_errors
++
;
if
(
desc_status
&
(
DescRxLong
|
DescRxRunt
))
if
(
desc_status
&
(
DescRxLong
|
DescRxRunt
))
np
->
stats
.
rx_length_errors
++
;
np
->
stats
.
rx_length_errors
++
;
if
(
desc_status
&
(
DescRxInvalid
|
DescRxAlign
))
if
(
desc_status
&
(
DescRxInvalid
|
DescRxAlign
))
np
->
stats
.
rx_frame_errors
++
;
np
->
stats
.
rx_frame_errors
++
;
if
(
desc_status
&
DescRxCRC
)
if
(
desc_status
&
DescRxCRC
)
np
->
stats
.
rx_crc_errors
++
;
np
->
stats
.
rx_crc_errors
++
;
}
}
}
else
{
}
else
{
struct
sk_buff
*
skb
;
struct
sk_buff
*
skb
;
/* Omit CRC size. */
/* Omit CRC size. */
int
pkt_len
=
(
desc_status
&
DescSizeMask
)
-
4
;
int
pkt_len
=
(
desc_status
&
DescSizeMask
)
-
4
;
/* Check if the packet is long enough to accept
/* Check if the packet is long enough to accept
* without copying to a minimally-sized skbuff. */
* without copying to a minimally-sized skbuff. */
if
(
pkt_len
<
rx_copybreak
if
(
pkt_len
<
rx_copybreak
&&
(
skb
=
dev_alloc_skb
(
pkt_len
+
2
))
!=
NULL
)
{
&&
(
skb
=
dev_alloc_skb
(
pkt_len
+
2
))
!=
NULL
)
{
skb
->
dev
=
dev
;
skb
->
dev
=
dev
;
skb_reserve
(
skb
,
2
);
/* 16 byte align the IP header */
/* 16 byte align the IP header */
pci_dma_sync_single
(
np
->
pci_dev
,
np
->
rx_dma
[
entry
],
skb_reserve
(
skb
,
2
);
np
->
rx_skbuff
[
entry
]
->
len
,
pci_dma_sync_single
(
np
->
pci_dev
,
PCI_DMA_FROMDEVICE
);
np
->
rx_dma
[
entry
],
np
->
rx_skbuff
[
entry
]
->
len
,
PCI_DMA_FROMDEVICE
);
#if HAS_IP_COPYSUM
#if HAS_IP_COPYSUM
eth_copy_and_sum
(
skb
,
np
->
rx_skbuff
[
entry
]
->
tail
,
pkt_len
,
0
);
eth_copy_and_sum
(
skb
,
np
->
rx_skbuff
[
entry
]
->
tail
,
pkt_len
,
0
);
skb_put
(
skb
,
pkt_len
);
skb_put
(
skb
,
pkt_len
);
#else
#else
memcpy
(
skb_put
(
skb
,
pkt_len
),
np
->
rx_skbuff
[
entry
]
->
tail
,
memcpy
(
skb_put
(
skb
,
pkt_len
),
pkt_len
);
np
->
rx_skbuff
[
entry
]
->
tail
,
pkt_len
);
#endif
#endif
}
else
{
}
else
{
pci_unmap_single
(
np
->
pci_dev
,
np
->
rx_dma
[
entry
],
pci_unmap_single
(
np
->
pci_dev
,
np
->
rx_dma
[
entry
],
np
->
rx_skbuff
[
entry
]
->
len
,
np
->
rx_skbuff
[
entry
]
->
len
,
PCI_DMA_FROMDEVICE
);
PCI_DMA_FROMDEVICE
);
skb_put
(
skb
=
np
->
rx_skbuff
[
entry
],
pkt_len
);
skb_put
(
skb
=
np
->
rx_skbuff
[
entry
],
pkt_len
);
np
->
rx_skbuff
[
entry
]
=
NULL
;
np
->
rx_skbuff
[
entry
]
=
NULL
;
}
}
skb
->
protocol
=
eth_type_trans
(
skb
,
dev
);
skb
->
protocol
=
eth_type_trans
(
skb
,
dev
);
/* W/ hardware checksum: skb->ip_summed = CHECKSUM_UNNECESSARY; */
netif_rx
(
skb
);
netif_rx
(
skb
);
dev
->
last_rx
=
jiffies
;
dev
->
last_rx
=
jiffies
;
np
->
stats
.
rx_packets
++
;
np
->
stats
.
rx_packets
++
;
...
@@ -1706,27 +1811,13 @@ static void netdev_rx(struct net_device *dev)
...
@@ -1706,27 +1811,13 @@ static void netdev_rx(struct net_device *dev)
np
->
rx_head_desc
=
&
np
->
rx_ring
[
entry
];
np
->
rx_head_desc
=
&
np
->
rx_ring
[
entry
];
desc_status
=
le32_to_cpu
(
np
->
rx_head_desc
->
cmd_status
);
desc_status
=
le32_to_cpu
(
np
->
rx_head_desc
->
cmd_status
);
}
}
refill_rx
(
dev
);
/* Refill the Rx ring buffers. */
for
(;
np
->
cur_rx
-
np
->
dirty_rx
>
0
;
np
->
dirty_rx
++
)
{
struct
sk_buff
*
skb
;
entry
=
np
->
dirty_rx
%
RX_RING_SIZE
;
if
(
np
->
rx_skbuff
[
entry
]
==
NULL
)
{
skb
=
dev_alloc_skb
(
np
->
rx_buf_sz
);
np
->
rx_skbuff
[
entry
]
=
skb
;
if
(
skb
==
NULL
)
break
;
/* Better luck next round. */
skb
->
dev
=
dev
;
/* Mark as being used by this device. */
np
->
rx_dma
[
entry
]
=
pci_map_single
(
np
->
pci_dev
,
skb
->
data
,
skb
->
len
,
PCI_DMA_FROMDEVICE
);
np
->
rx_ring
[
entry
].
addr
=
cpu_to_le32
(
np
->
rx_dma
[
entry
]);
}
np
->
rx_ring
[
entry
].
cmd_status
=
cpu_to_le32
(
np
->
rx_buf_sz
);
}
/* Restart Rx engine if stopped. */
/* Restart Rx engine if stopped. */
writel
(
RxOn
,
dev
->
base_addr
+
ChipCmd
);
if
(
np
->
oom
)
mod_timer
(
&
np
->
timer
,
jiffies
+
1
);
else
writel
(
RxOn
,
dev
->
base_addr
+
ChipCmd
);
}
}
static
void
netdev_error
(
struct
net_device
*
dev
,
int
intr_status
)
static
void
netdev_error
(
struct
net_device
*
dev
,
int
intr_status
)
...
@@ -1738,9 +1829,9 @@ static void netdev_error(struct net_device *dev, int intr_status)
...
@@ -1738,9 +1829,9 @@ static void netdev_error(struct net_device *dev, int intr_status)
if
(
intr_status
&
LinkChange
)
{
if
(
intr_status
&
LinkChange
)
{
u16
adv
=
mdio_read
(
dev
,
1
,
MII_ADVERTISE
);
u16
adv
=
mdio_read
(
dev
,
1
,
MII_ADVERTISE
);
u16
lpa
=
mdio_read
(
dev
,
1
,
MII_LPA
);
u16
lpa
=
mdio_read
(
dev
,
1
,
MII_LPA
);
if
(
mdio_read
(
dev
,
1
,
MII_BMCR
)
&
BMCR_ANENABLE
if
(
mdio_read
(
dev
,
1
,
MII_BMCR
)
&
BMCR_ANENABLE
&&
netif_msg_link
(
np
))
{
&&
netif_msg_link
(
np
))
{
printk
(
KERN_INFO
printk
(
KERN_INFO
"%s: Autonegotiation advertising"
"%s: Autonegotiation advertising"
" %#04x partner %#04x.
\n
"
,
dev
->
name
,
" %#04x partner %#04x.
\n
"
,
dev
->
name
,
adv
,
lpa
);
adv
,
lpa
);
...
@@ -1765,7 +1856,7 @@ static void netdev_error(struct net_device *dev, int intr_status)
...
@@ -1765,7 +1856,7 @@ static void netdev_error(struct net_device *dev, int intr_status)
if
(
intr_status
&
WOLPkt
&&
netif_msg_wol
(
np
))
{
if
(
intr_status
&
WOLPkt
&&
netif_msg_wol
(
np
))
{
int
wol_status
=
readl
(
ioaddr
+
WOLCmd
);
int
wol_status
=
readl
(
ioaddr
+
WOLCmd
);
printk
(
KERN_NOTICE
"%s: Link wake-up event %#08x
\n
"
,
printk
(
KERN_NOTICE
"%s: Link wake-up event %#08x
\n
"
,
dev
->
name
,
wol_status
);
dev
->
name
,
wol_status
);
}
}
if
(
intr_status
&
RxStatusFIFOOver
)
{
if
(
intr_status
&
RxStatusFIFOOver
)
{
if
(
netif_msg_rx_err
(
np
)
&&
netif_msg_intr
(
np
))
{
if
(
netif_msg_rx_err
(
np
)
&&
netif_msg_intr
(
np
))
{
...
@@ -1800,8 +1891,8 @@ static struct net_device_stats *get_stats(struct net_device *dev)
...
@@ -1800,8 +1891,8 @@ static struct net_device_stats *get_stats(struct net_device *dev)
/* The chip only need report frame silently dropped. */
/* The chip only need report frame silently dropped. */
spin_lock_irq
(
&
np
->
lock
);
spin_lock_irq
(
&
np
->
lock
);
if
(
netif_running
(
dev
)
&&
netif_device_present
(
dev
)
)
if
(
netif_running
(
dev
)
&&
!
np
->
hands_off
)
__get_stats
(
dev
);
__get_stats
(
dev
);
spin_unlock_irq
(
&
np
->
lock
);
spin_unlock_irq
(
&
np
->
lock
);
return
&
np
->
stats
;
return
&
np
->
stats
;
...
@@ -1817,27 +1908,27 @@ static struct net_device_stats *get_stats(struct net_device *dev)
...
@@ -1817,27 +1908,27 @@ static struct net_device_stats *get_stats(struct net_device *dev)
#define DP_POLYNOMIAL 0x04C11DB7
#define DP_POLYNOMIAL 0x04C11DB7
static
unsigned
dp83815_crc
(
int
length
,
unsigned
char
*
data
)
static
unsigned
dp83815_crc
(
int
length
,
unsigned
char
*
data
)
{
{
u32
crc
;
u32
crc
;
u8
cur_byte
;
u8
cur_byte
;
u8
msb
;
u8
msb
;
u8
byte
,
bit
;
u8
byte
,
bit
;
crc
=
~
0
;
crc
=
~
0
;
for
(
byte
=
0
;
byte
<
length
;
byte
++
)
{
for
(
byte
=
0
;
byte
<
length
;
byte
++
)
{
cur_byte
=
*
data
++
;
cur_byte
=
*
data
++
;
for
(
bit
=
0
;
bit
<
8
;
bit
++
)
{
for
(
bit
=
0
;
bit
<
8
;
bit
++
)
{
msb
=
crc
>>
31
;
msb
=
crc
>>
31
;
crc
<<=
1
;
crc
<<=
1
;
if
(
msb
^
(
cur_byte
&
1
))
{
if
(
msb
^
(
cur_byte
&
1
))
{
crc
^=
DP_POLYNOMIAL
;
crc
^=
DP_POLYNOMIAL
;
crc
|=
1
;
crc
|=
1
;
}
}
cur_byte
>>=
1
;
cur_byte
>>=
1
;
}
}
}
}
crc
>>=
23
;
crc
>>=
23
;
return
(
crc
);
return
(
crc
);
}
}
...
@@ -1850,18 +1941,18 @@ static void __set_rx_mode(struct net_device *dev)
...
@@ -1850,18 +1941,18 @@ static void __set_rx_mode(struct net_device *dev)
{
{
long
ioaddr
=
dev
->
base_addr
;
long
ioaddr
=
dev
->
base_addr
;
struct
netdev_private
*
np
=
dev
->
priv
;
struct
netdev_private
*
np
=
dev
->
priv
;
u8
mc_filter
[
64
];
/* Multicast hash filter */
u8
mc_filter
[
64
];
/* Multicast hash filter */
u32
rx_mode
;
u32
rx_mode
;
if
(
dev
->
flags
&
IFF_PROMISC
)
{
/* Set promiscuous. */
if
(
dev
->
flags
&
IFF_PROMISC
)
{
/* Set promiscuous. */
/* Unconditionally log net taps. */
/* Unconditionally log net taps. */
printk
(
KERN_NOTICE
"%s: Promiscuous mode enabled.
\n
"
,
printk
(
KERN_NOTICE
"%s: Promiscuous mode enabled.
\n
"
,
dev
->
name
);
dev
->
name
);
rx_mode
=
RxFilterEnable
|
AcceptBroadcast
rx_mode
=
RxFilterEnable
|
AcceptBroadcast
|
AcceptAllMulticast
|
AcceptAllPhys
|
AcceptMyPhys
;
|
AcceptAllMulticast
|
AcceptAllPhys
|
AcceptMyPhys
;
}
else
if
((
dev
->
mc_count
>
multicast_filter_limit
)
}
else
if
((
dev
->
mc_count
>
multicast_filter_limit
)
||
(
dev
->
flags
&
IFF_ALLMULTI
))
{
||
(
dev
->
flags
&
IFF_ALLMULTI
))
{
rx_mode
=
RxFilterEnable
|
AcceptBroadcast
rx_mode
=
RxFilterEnable
|
AcceptBroadcast
|
AcceptAllMulticast
|
AcceptMyPhys
;
|
AcceptAllMulticast
|
AcceptMyPhys
;
}
else
{
}
else
{
struct
dev_mc_list
*
mclist
;
struct
dev_mc_list
*
mclist
;
...
@@ -1869,14 +1960,15 @@ static void __set_rx_mode(struct net_device *dev)
...
@@ -1869,14 +1960,15 @@ static void __set_rx_mode(struct net_device *dev)
memset
(
mc_filter
,
0
,
sizeof
(
mc_filter
));
memset
(
mc_filter
,
0
,
sizeof
(
mc_filter
));
for
(
i
=
0
,
mclist
=
dev
->
mc_list
;
mclist
&&
i
<
dev
->
mc_count
;
for
(
i
=
0
,
mclist
=
dev
->
mc_list
;
mclist
&&
i
<
dev
->
mc_count
;
i
++
,
mclist
=
mclist
->
next
)
{
i
++
,
mclist
=
mclist
->
next
)
{
set_bit_le
(
dp83815_crc
(
ETH_ALEN
,
mclist
->
dmi_addr
)
&
0x1ff
,
set_bit_le
(
mc_filter
);
dp83815_crc
(
ETH_ALEN
,
mclist
->
dmi_addr
)
&
0x1ff
,
mc_filter
);
}
}
rx_mode
=
RxFilterEnable
|
AcceptBroadcast
rx_mode
=
RxFilterEnable
|
AcceptBroadcast
|
AcceptMulticast
|
AcceptMyPhys
;
|
AcceptMulticast
|
AcceptMyPhys
;
for
(
i
=
0
;
i
<
64
;
i
+=
2
)
{
for
(
i
=
0
;
i
<
64
;
i
+=
2
)
{
writew
(
HASH_TABLE
+
i
,
ioaddr
+
RxFilterAddr
);
writew
(
HASH_TABLE
+
i
,
ioaddr
+
RxFilterAddr
);
writew
((
mc_filter
[
i
+
1
]
<<
8
)
+
mc_filter
[
i
],
writew
((
mc_filter
[
i
+
1
]
<<
8
)
+
mc_filter
[
i
],
ioaddr
+
RxFilterData
);
ioaddr
+
RxFilterData
);
}
}
}
}
...
@@ -1888,7 +1980,7 @@ static void set_rx_mode(struct net_device *dev)
...
@@ -1888,7 +1980,7 @@ static void set_rx_mode(struct net_device *dev)
{
{
struct
netdev_private
*
np
=
dev
->
priv
;
struct
netdev_private
*
np
=
dev
->
priv
;
spin_lock_irq
(
&
np
->
lock
);
spin_lock_irq
(
&
np
->
lock
);
if
(
netif_device_present
(
dev
)
)
if
(
!
np
->
hands_off
)
__set_rx_mode
(
dev
);
__set_rx_mode
(
dev
);
spin_unlock_irq
(
&
np
->
lock
);
spin_unlock_irq
(
&
np
->
lock
);
}
}
...
@@ -1897,18 +1989,18 @@ static int netdev_ethtool_ioctl(struct net_device *dev, void *useraddr)
...
@@ -1897,18 +1989,18 @@ static int netdev_ethtool_ioctl(struct net_device *dev, void *useraddr)
{
{
struct
netdev_private
*
np
=
dev
->
priv
;
struct
netdev_private
*
np
=
dev
->
priv
;
u32
cmd
;
u32
cmd
;
if
(
get_user
(
cmd
,
(
u32
*
)
useraddr
))
if
(
get_user
(
cmd
,
(
u32
*
)
useraddr
))
return
-
EFAULT
;
return
-
EFAULT
;
switch
(
cmd
)
{
switch
(
cmd
)
{
/* get driver info */
/* get driver info */
case
ETHTOOL_GDRVINFO
:
{
case
ETHTOOL_GDRVINFO
:
{
struct
ethtool_drvinfo
info
=
{
ETHTOOL_GDRVINFO
};
struct
ethtool_drvinfo
info
=
{
ETHTOOL_GDRVINFO
};
strncpy
(
info
.
driver
,
DRV_NAME
,
ETHTOOL_BUSINFO_LEN
);
strncpy
(
info
.
driver
,
DRV_NAME
,
ETHTOOL_BUSINFO_LEN
);
strncpy
(
info
.
version
,
DRV_VERSION
,
ETHTOOL_BUSINFO_LEN
);
strncpy
(
info
.
version
,
DRV_VERSION
,
ETHTOOL_BUSINFO_LEN
);
info
.
fw_version
[
0
]
=
'\0'
;
info
.
fw_version
[
0
]
=
'\0'
;
strncpy
(
info
.
bus_info
,
np
->
pci_dev
->
slot_name
,
strncpy
(
info
.
bus_info
,
np
->
pci_dev
->
slot_name
,
ETHTOOL_BUSINFO_LEN
);
ETHTOOL_BUSINFO_LEN
);
info
.
eedump_len
=
NATSEMI_EEPROM_SIZE
;
info
.
eedump_len
=
NATSEMI_EEPROM_SIZE
;
info
.
regdump_len
=
NATSEMI_REGS_SIZE
;
info
.
regdump_len
=
NATSEMI_REGS_SIZE
;
...
@@ -1968,7 +2060,7 @@ static int netdev_ethtool_ioctl(struct net_device *dev, void *useraddr)
...
@@ -1968,7 +2060,7 @@ static int netdev_ethtool_ioctl(struct net_device *dev, void *useraddr)
if
(
copy_from_user
(
&
regs
,
useraddr
,
sizeof
(
regs
)))
if
(
copy_from_user
(
&
regs
,
useraddr
,
sizeof
(
regs
)))
return
-
EFAULT
;
return
-
EFAULT
;
if
(
regs
.
len
>
NATSEMI_REGS_SIZE
)
{
if
(
regs
.
len
>
NATSEMI_REGS_SIZE
)
{
regs
.
len
=
NATSEMI_REGS_SIZE
;
regs
.
len
=
NATSEMI_REGS_SIZE
;
}
}
...
@@ -2035,7 +2127,7 @@ static int netdev_ethtool_ioctl(struct net_device *dev, void *useraddr)
...
@@ -2035,7 +2127,7 @@ static int netdev_ethtool_ioctl(struct net_device *dev, void *useraddr)
if
(
copy_from_user
(
&
eeprom
,
useraddr
,
sizeof
(
eeprom
)))
if
(
copy_from_user
(
&
eeprom
,
useraddr
,
sizeof
(
eeprom
)))
return
-
EFAULT
;
return
-
EFAULT
;
if
(
eeprom
.
offset
>
eeprom
.
offset
+
eeprom
.
len
)
if
(
eeprom
.
offset
>
eeprom
.
offset
+
eeprom
.
len
)
return
-
EINVAL
;
return
-
EINVAL
;
...
@@ -2059,8 +2151,8 @@ static int netdev_ethtool_ioctl(struct net_device *dev, void *useraddr)
...
@@ -2059,8 +2151,8 @@ static int netdev_ethtool_ioctl(struct net_device *dev, void *useraddr)
return
0
;
return
0
;
}
}
}
}
return
-
EOPNOTSUPP
;
return
-
EOPNOTSUPP
;
}
}
...
@@ -2082,7 +2174,7 @@ static int netdev_set_wol(struct net_device *dev, u32 newval)
...
@@ -2082,7 +2174,7 @@ static int netdev_set_wol(struct net_device *dev, u32 newval)
data
|=
WakeArp
;
data
|=
WakeArp
;
if
(
newval
&
WAKE_MAGIC
)
if
(
newval
&
WAKE_MAGIC
)
data
|=
WakeMagic
;
data
|=
WakeMagic
;
if
(
np
->
srr
>=
SRR_
REV
_D
)
{
if
(
np
->
srr
>=
SRR_
DP83815
_D
)
{
if
(
newval
&
WAKE_MAGICSECURE
)
{
if
(
newval
&
WAKE_MAGICSECURE
)
{
data
|=
WakeMagicSecure
;
data
|=
WakeMagicSecure
;
}
}
...
@@ -2098,10 +2190,10 @@ static int netdev_get_wol(struct net_device *dev, u32 *supported, u32 *cur)
...
@@ -2098,10 +2190,10 @@ static int netdev_get_wol(struct net_device *dev, u32 *supported, u32 *cur)
struct
netdev_private
*
np
=
dev
->
priv
;
struct
netdev_private
*
np
=
dev
->
priv
;
u32
regval
=
readl
(
dev
->
base_addr
+
WOLCmd
);
u32
regval
=
readl
(
dev
->
base_addr
+
WOLCmd
);
*
supported
=
(
WAKE_PHY
|
WAKE_UCAST
|
WAKE_MCAST
|
WAKE_BCAST
*
supported
=
(
WAKE_PHY
|
WAKE_UCAST
|
WAKE_MCAST
|
WAKE_BCAST
|
WAKE_ARP
|
WAKE_MAGIC
);
|
WAKE_ARP
|
WAKE_MAGIC
);
if
(
np
->
srr
>=
SRR_
REV
_D
)
{
if
(
np
->
srr
>=
SRR_
DP83815
_D
)
{
/* SOPASS works on revD and higher */
/* SOPASS works on revD and higher */
*
supported
|=
WAKE_MAGICSECURE
;
*
supported
|=
WAKE_MAGICSECURE
;
}
}
...
@@ -2133,8 +2225,8 @@ static int netdev_set_sopass(struct net_device *dev, u8 *newval)
...
@@ -2133,8 +2225,8 @@ static int netdev_set_sopass(struct net_device *dev, u8 *newval)
struct
netdev_private
*
np
=
dev
->
priv
;
struct
netdev_private
*
np
=
dev
->
priv
;
u16
*
sval
=
(
u16
*
)
newval
;
u16
*
sval
=
(
u16
*
)
newval
;
u32
addr
;
u32
addr
;
if
(
np
->
srr
<
SRR_
REV
_D
)
{
if
(
np
->
srr
<
SRR_
DP83815
_D
)
{
return
0
;
return
0
;
}
}
...
@@ -2149,10 +2241,10 @@ static int netdev_set_sopass(struct net_device *dev, u8 *newval)
...
@@ -2149,10 +2241,10 @@ static int netdev_set_sopass(struct net_device *dev, u8 *newval)
writel
(
addr
|
0xc
,
dev
->
base_addr
+
RxFilterAddr
);
writel
(
addr
|
0xc
,
dev
->
base_addr
+
RxFilterAddr
);
writew
(
sval
[
1
],
dev
->
base_addr
+
RxFilterData
);
writew
(
sval
[
1
],
dev
->
base_addr
+
RxFilterData
);
writel
(
addr
|
0xe
,
dev
->
base_addr
+
RxFilterAddr
);
writel
(
addr
|
0xe
,
dev
->
base_addr
+
RxFilterAddr
);
writew
(
sval
[
2
],
dev
->
base_addr
+
RxFilterData
);
writew
(
sval
[
2
],
dev
->
base_addr
+
RxFilterData
);
/* re-enable the RX filter */
/* re-enable the RX filter */
writel
(
addr
|
RxFilterEnable
,
dev
->
base_addr
+
RxFilterAddr
);
writel
(
addr
|
RxFilterEnable
,
dev
->
base_addr
+
RxFilterAddr
);
...
@@ -2165,7 +2257,7 @@ static int netdev_get_sopass(struct net_device *dev, u8 *data)
...
@@ -2165,7 +2257,7 @@ static int netdev_get_sopass(struct net_device *dev, u8 *data)
u16
*
sval
=
(
u16
*
)
data
;
u16
*
sval
=
(
u16
*
)
data
;
u32
addr
;
u32
addr
;
if
(
np
->
srr
<
SRR_
REV
_D
)
{
if
(
np
->
srr
<
SRR_
DP83815
_D
)
{
sval
[
0
]
=
sval
[
1
]
=
sval
[
2
]
=
0
;
sval
[
0
]
=
sval
[
1
]
=
sval
[
2
]
=
0
;
return
0
;
return
0
;
}
}
...
@@ -2178,10 +2270,10 @@ static int netdev_get_sopass(struct net_device *dev, u8 *data)
...
@@ -2178,10 +2270,10 @@ static int netdev_get_sopass(struct net_device *dev, u8 *data)
writel
(
addr
|
0xc
,
dev
->
base_addr
+
RxFilterAddr
);
writel
(
addr
|
0xc
,
dev
->
base_addr
+
RxFilterAddr
);
sval
[
1
]
=
readw
(
dev
->
base_addr
+
RxFilterData
);
sval
[
1
]
=
readw
(
dev
->
base_addr
+
RxFilterData
);
writel
(
addr
|
0xe
,
dev
->
base_addr
+
RxFilterAddr
);
writel
(
addr
|
0xe
,
dev
->
base_addr
+
RxFilterAddr
);
sval
[
2
]
=
readw
(
dev
->
base_addr
+
RxFilterData
);
sval
[
2
]
=
readw
(
dev
->
base_addr
+
RxFilterData
);
writel
(
addr
,
dev
->
base_addr
+
RxFilterAddr
);
writel
(
addr
,
dev
->
base_addr
+
RxFilterAddr
);
return
0
;
return
0
;
...
@@ -2191,11 +2283,11 @@ static int netdev_get_ecmd(struct net_device *dev, struct ethtool_cmd *ecmd)
...
@@ -2191,11 +2283,11 @@ static int netdev_get_ecmd(struct net_device *dev, struct ethtool_cmd *ecmd)
{
{
u32
tmp
;
u32
tmp
;
ecmd
->
supported
=
ecmd
->
supported
=
(
SUPPORTED_10baseT_Half
|
SUPPORTED_10baseT_Full
|
(
SUPPORTED_10baseT_Half
|
SUPPORTED_10baseT_Full
|
SUPPORTED_100baseT_Half
|
SUPPORTED_100baseT_Full
|
SUPPORTED_100baseT_Half
|
SUPPORTED_100baseT_Full
|
SUPPORTED_Autoneg
|
SUPPORTED_TP
|
SUPPORTED_MII
);
SUPPORTED_Autoneg
|
SUPPORTED_TP
|
SUPPORTED_MII
);
/* only supports twisted-pair or MII */
/* only supports twisted-pair or MII */
tmp
=
readl
(
dev
->
base_addr
+
ChipConfig
);
tmp
=
readl
(
dev
->
base_addr
+
ChipConfig
);
if
(
tmp
&
CfgExtPhy
)
if
(
tmp
&
CfgExtPhy
)
...
@@ -2234,7 +2326,7 @@ static int netdev_get_ecmd(struct net_device *dev, struct ethtool_cmd *ecmd)
...
@@ -2234,7 +2326,7 @@ static int netdev_get_ecmd(struct net_device *dev, struct ethtool_cmd *ecmd)
}
else
{
}
else
{
ecmd
->
speed
=
SPEED_10
;
ecmd
->
speed
=
SPEED_10
;
}
}
if
(
tmp
&
CfgFullDuplex
)
{
if
(
tmp
&
CfgFullDuplex
)
{
ecmd
->
duplex
=
DUPLEX_FULL
;
ecmd
->
duplex
=
DUPLEX_FULL
;
}
else
{
}
else
{
...
@@ -2262,9 +2354,9 @@ static int netdev_set_ecmd(struct net_device *dev, struct ethtool_cmd *ecmd)
...
@@ -2262,9 +2354,9 @@ static int netdev_set_ecmd(struct net_device *dev, struct ethtool_cmd *ecmd)
if
(
ecmd
->
autoneg
!=
AUTONEG_DISABLE
&&
ecmd
->
autoneg
!=
AUTONEG_ENABLE
)
if
(
ecmd
->
autoneg
!=
AUTONEG_DISABLE
&&
ecmd
->
autoneg
!=
AUTONEG_ENABLE
)
return
-
EINVAL
;
return
-
EINVAL
;
/* ignore phy_address, maxtxpkt, maxrxpkt for now */
/* ignore phy_address, maxtxpkt, maxrxpkt for now */
/* WHEW! now lets bang some bits */
/* WHEW! now lets bang some bits */
tmp
=
mdio_read
(
dev
,
1
,
MII_BMCR
);
tmp
=
mdio_read
(
dev
,
1
,
MII_BMCR
);
if
(
ecmd
->
autoneg
==
AUTONEG_ENABLE
)
{
if
(
ecmd
->
autoneg
==
AUTONEG_ENABLE
)
{
/* turn on autonegotiation */
/* turn on autonegotiation */
...
@@ -2290,7 +2382,7 @@ static int netdev_get_regs(struct net_device *dev, u8 *buf)
...
@@ -2290,7 +2382,7 @@ static int netdev_get_regs(struct net_device *dev, u8 *buf)
int
j
;
int
j
;
u32
rfcr
;
u32
rfcr
;
u32
*
rbuf
=
(
u32
*
)
buf
;
u32
*
rbuf
=
(
u32
*
)
buf
;
/* read all of page 0 of registers */
/* read all of page 0 of registers */
for
(
i
=
0
;
i
<
NATSEMI_PG0_NREGS
;
i
++
)
{
for
(
i
=
0
;
i
<
NATSEMI_PG0_NREGS
;
i
++
)
{
rbuf
[
i
]
=
readl
(
dev
->
base_addr
+
i
*
4
);
rbuf
[
i
]
=
readl
(
dev
->
base_addr
+
i
*
4
);
...
@@ -2314,8 +2406,8 @@ static int netdev_get_regs(struct net_device *dev, u8 *buf)
...
@@ -2314,8 +2406,8 @@ static int netdev_get_regs(struct net_device *dev, u8 *buf)
/* the interrupt status is clear-on-read - see if we missed any */
/* the interrupt status is clear-on-read - see if we missed any */
if
(
rbuf
[
4
]
&
rbuf
[
5
])
{
if
(
rbuf
[
4
]
&
rbuf
[
5
])
{
printk
(
KERN_WARNING
printk
(
KERN_WARNING
"%s: shoot, we dropped an interrupt (%#08x)
\n
"
,
"%s: shoot, we dropped an interrupt (%#08x)
\n
"
,
dev
->
name
,
rbuf
[
4
]
&
rbuf
[
5
]);
dev
->
name
,
rbuf
[
4
]
&
rbuf
[
5
]);
}
}
...
@@ -2339,7 +2431,7 @@ static int netdev_get_eeprom(struct net_device *dev, u8 *buf)
...
@@ -2339,7 +2431,7 @@ static int netdev_get_eeprom(struct net_device *dev, u8 *buf)
/* eeprom_read reads 16 bits, and indexes by 16 bits */
/* eeprom_read reads 16 bits, and indexes by 16 bits */
for
(
i
=
0
;
i
<
NATSEMI_EEPROM_SIZE
/
2
;
i
++
)
{
for
(
i
=
0
;
i
<
NATSEMI_EEPROM_SIZE
/
2
;
i
++
)
{
ebuf
[
i
]
=
eeprom_read
(
dev
->
base_addr
,
i
);
ebuf
[
i
]
=
eeprom_read
(
dev
->
base_addr
,
i
);
/* The EEPROM itself stores data bit-swapped, but eeprom_read
/* The EEPROM itself stores data bit-swapped, but eeprom_read
* reads it back "sanely". So we swap it back here in order to
* reads it back "sanely". So we swap it back here in order to
* present it to userland as it is stored. */
* present it to userland as it is stored. */
ebuf
[
i
]
=
SWAP_BITS
(
ebuf
[
i
]);
ebuf
[
i
]
=
SWAP_BITS
(
ebuf
[
i
]);
...
@@ -2361,7 +2453,7 @@ static int netdev_ioctl(struct net_device *dev, struct ifreq *rq, int cmd)
...
@@ -2361,7 +2453,7 @@ static int netdev_ioctl(struct net_device *dev, struct ifreq *rq, int cmd)
case
SIOCGMIIREG
:
/* Read MII PHY register. */
case
SIOCGMIIREG
:
/* Read MII PHY register. */
case
SIOCDEVPRIVATE
+
1
:
/* for binary compat, remove in 2.5 */
case
SIOCDEVPRIVATE
+
1
:
/* for binary compat, remove in 2.5 */
data
->
val_out
=
mdio_read
(
dev
,
data
->
phy_id
&
0x1f
,
data
->
val_out
=
mdio_read
(
dev
,
data
->
phy_id
&
0x1f
,
data
->
reg_num
&
0x1f
);
data
->
reg_num
&
0x1f
);
return
0
;
return
0
;
...
@@ -2369,7 +2461,7 @@ static int netdev_ioctl(struct net_device *dev, struct ifreq *rq, int cmd)
...
@@ -2369,7 +2461,7 @@ static int netdev_ioctl(struct net_device *dev, struct ifreq *rq, int cmd)
case
SIOCDEVPRIVATE
+
2
:
/* for binary compat, remove in 2.5 */
case
SIOCDEVPRIVATE
+
2
:
/* for binary compat, remove in 2.5 */
if
(
!
capable
(
CAP_NET_ADMIN
))
if
(
!
capable
(
CAP_NET_ADMIN
))
return
-
EPERM
;
return
-
EPERM
;
mdio_write
(
dev
,
data
->
phy_id
&
0x1f
,
data
->
reg_num
&
0x1f
,
mdio_write
(
dev
,
data
->
phy_id
&
0x1f
,
data
->
reg_num
&
0x1f
,
data
->
val_in
);
data
->
val_in
);
return
0
;
return
0
;
default:
default:
...
@@ -2383,7 +2475,7 @@ static void enable_wol_mode(struct net_device *dev, int enable_intr)
...
@@ -2383,7 +2475,7 @@ static void enable_wol_mode(struct net_device *dev, int enable_intr)
struct
netdev_private
*
np
=
dev
->
priv
;
struct
netdev_private
*
np
=
dev
->
priv
;
if
(
netif_msg_wol
(
np
))
if
(
netif_msg_wol
(
np
))
printk
(
KERN_INFO
"%s: remaining active for wake-on-lan
\n
"
,
printk
(
KERN_INFO
"%s: remaining active for wake-on-lan
\n
"
,
dev
->
name
);
dev
->
name
);
/* For WOL we must restart the rx process in silent mode.
/* For WOL we must restart the rx process in silent mode.
...
@@ -2415,51 +2507,56 @@ static int netdev_close(struct net_device *dev)
...
@@ -2415,51 +2507,56 @@ static int netdev_close(struct net_device *dev)
long
ioaddr
=
dev
->
base_addr
;
long
ioaddr
=
dev
->
base_addr
;
struct
netdev_private
*
np
=
dev
->
priv
;
struct
netdev_private
*
np
=
dev
->
priv
;
netif_stop_queue
(
dev
);
netif_carrier_off
(
dev
);
if
(
netif_msg_ifdown
(
np
))
if
(
netif_msg_ifdown
(
np
))
printk
(
KERN_DEBUG
printk
(
KERN_DEBUG
"%s: Shutting down ethercard, status was %#04x.
\n
"
,
"%s: Shutting down ethercard, status was %#04x.
\n
"
,
dev
->
name
,
(
int
)
readl
(
ioaddr
+
ChipCmd
));
dev
->
name
,
(
int
)
readl
(
ioaddr
+
ChipCmd
));
if
(
netif_msg_pktdata
(
np
))
if
(
netif_msg_pktdata
(
np
))
printk
(
KERN_DEBUG
printk
(
KERN_DEBUG
"%s: Queue pointers were Tx %d / %d, Rx %d / %d.
\n
"
,
"%s: Queue pointers were Tx %d / %d, Rx %d / %d.
\n
"
,
dev
->
name
,
np
->
cur_tx
,
np
->
dirty_tx
,
dev
->
name
,
np
->
cur_tx
,
np
->
dirty_tx
,
np
->
cur_rx
,
np
->
dirty_rx
);
np
->
cur_rx
,
np
->
dirty_rx
);
del_timer_sync
(
&
np
->
timer
);
/*
* FIXME: what if someone tries to close a device
* that is suspended?
* Should we reenable the nic to switch to
* the final WOL settings?
*/
del_timer_sync
(
&
np
->
timer
);
disable_irq
(
dev
->
irq
);
disable_irq
(
dev
->
irq
);
spin_lock_irq
(
&
np
->
lock
);
spin_lock_irq
(
&
np
->
lock
);
/* Disable interrupts, and flush posted writes */
/* Disable and clear interrupts */
writel
(
0
,
ioaddr
+
IntrEnable
);
writel
(
0
,
ioaddr
+
IntrEnable
);
readl
(
ioaddr
+
IntrEnable
);
np
->
hands_off
=
1
;
spin_unlock_irq
(
&
np
->
lock
);
enable_irq
(
dev
->
irq
);
free_irq
(
dev
->
irq
,
dev
);
/* Interrupt disabled, interrupt handler released,
* queue stopped, timer deleted, rtnl_lock held
* All async codepaths that access the driver are disabled.
*/
spin_lock_irq
(
&
np
->
lock
);
np
->
hands_off
=
0
;
readl
(
ioaddr
+
IntrMask
);
readl
(
ioaddr
+
IntrMask
);
readw
(
ioaddr
+
MIntrStatus
);
readw
(
ioaddr
+
MIntrStatus
);
/* Freeze Stats */
/* Freeze Stats */
writel
(
StatsFreeze
,
ioaddr
+
StatsCtrl
);
writel
(
StatsFreeze
,
ioaddr
+
StatsCtrl
);
/* Stop the chip's Tx and Rx processes. */
/* Stop the chip's Tx and Rx processes. */
natsemi_stop_rxtx
(
dev
);
natsemi_stop_rxtx
(
dev
);
__get_stats
(
dev
);
__get_stats
(
dev
);
spin_unlock_irq
(
&
np
->
lock
);
spin_unlock_irq
(
&
np
->
lock
);
/* race: shared irq and as most nics the DP83815
* reports _all_ interrupt conditions in IntrStatus, even
* disabled ones.
* packet received after disable_irq, but before stop_rxtx
* --> race. intr_handler would restart the rx process.
* netif_device_{de,a}tach around {enable,free}_irq.
*/
netif_device_detach
(
dev
);
enable_irq
(
dev
->
irq
);
free_irq
(
dev
->
irq
,
dev
);
netif_device_attach
(
dev
);
/* clear the carrier last - an interrupt could reenable it otherwise */
/* clear the carrier last - an interrupt could reenable it otherwise */
netif_carrier_off
(
dev
);
netif_carrier_off
(
dev
);
netif_stop_queue
(
dev
);
dump_ring
(
dev
);
dump_ring
(
dev
);
drain_ring
(
dev
);
drain_ring
(
dev
);
...
@@ -2480,7 +2577,7 @@ static int netdev_close(struct net_device *dev)
...
@@ -2480,7 +2577,7 @@ static int netdev_close(struct net_device *dev)
return
0
;
return
0
;
}
}
static
void
__devexit
natsemi_remove1
(
struct
pci_dev
*
pdev
)
static
void
__devexit
natsemi_remove1
(
struct
pci_dev
*
pdev
)
{
{
struct
net_device
*
dev
=
pci_get_drvdata
(
pdev
);
struct
net_device
*
dev
=
pci_get_drvdata
(
pdev
);
...
@@ -2495,23 +2592,26 @@ static void __devexit natsemi_remove1 (struct pci_dev *pdev)
...
@@ -2495,23 +2592,26 @@ static void __devexit natsemi_remove1 (struct pci_dev *pdev)
#ifdef CONFIG_PM
#ifdef CONFIG_PM
/*
/*
* The ns83815 chip doesn't have explicit RxStop bits.
* Kicking the Rx or Tx process for a new packet reenables the Rx process
* of the nic, thus this function must be very careful:
*
* suspend/resume synchronization:
* suspend/resume synchronization:
* entry points:
* entry points:
* netdev_open, netdev_close, netdev_ioctl, set_rx_mode, intr_handler,
* netdev_open, netdev_close, netdev_ioctl, set_rx_mode, intr_handler,
* start_tx, tx_timeout
* start_tx, tx_timeout
*
Reading from some registers can restart the nic!
*
* No function accesses the hardware without checking n
etif_device_present()
.
* No function accesses the hardware without checking n
p->hands_off
.
*
the check occurs under spin_lock_irq(&np->lock);
* the check occurs under spin_lock_irq(&np->lock);
* exceptions:
* exceptions:
*
* netdev_ioctl, netdev_open
.
*
* netdev_ioctl: noncritical access
.
*
net/core checks netif_device_present() before calling them.
*
* netdev_open: cannot happen due to the device_detach
*
* netdev_close: doesn't hurt.
* * netdev_close: doesn't hurt.
* * netdev_timer: timer stopped by natsemi_suspend.
* * netdev_timer: timer stopped by natsemi_suspend.
* * intr_handler: doesn't acquire the spinlock. suspend calls
* * intr_handler: doesn't acquire the spinlock. suspend calls
* disable_irq() to enforce synchronization.
* disable_irq() to enforce synchronization.
*
*
* netif_device_detach must occur under spin_unlock_irq(), interrupts from a
* Interrupts must be disabled, otherwise hands_off can cause irq storms.
* detached device would cause an irq storm.
*/
*/
static
int
natsemi_suspend
(
struct
pci_dev
*
pdev
,
u32
state
)
static
int
natsemi_suspend
(
struct
pci_dev
*
pdev
,
u32
state
)
...
@@ -2528,13 +2628,13 @@ static int natsemi_suspend (struct pci_dev *pdev, u32 state)
...
@@ -2528,13 +2628,13 @@ static int natsemi_suspend (struct pci_dev *pdev, u32 state)
spin_lock_irq
(
&
np
->
lock
);
spin_lock_irq
(
&
np
->
lock
);
writel
(
0
,
ioaddr
+
IntrEnable
);
writel
(
0
,
ioaddr
+
IntrEnable
);
np
->
hands_off
=
1
;
natsemi_stop_rxtx
(
dev
);
natsemi_stop_rxtx
(
dev
);
netif_stop_queue
(
dev
);
netif_stop_queue
(
dev
);
netif_device_detach
(
dev
);
spin_unlock_irq
(
&
np
->
lock
);
spin_unlock_irq
(
&
np
->
lock
);
enable_irq
(
dev
->
irq
);
enable_irq
(
dev
->
irq
);
/* Update the error counts. */
/* Update the error counts. */
__get_stats
(
dev
);
__get_stats
(
dev
);
...
@@ -2546,7 +2646,7 @@ static int natsemi_suspend (struct pci_dev *pdev, u32 state)
...
@@ -2546,7 +2646,7 @@ static int natsemi_suspend (struct pci_dev *pdev, u32 state)
if
(
wol
)
{
if
(
wol
)
{
/* restart the NIC in WOL mode.
/* restart the NIC in WOL mode.
* The nic must be stopped for this.
* The nic must be stopped for this.
* FIXME: use the WOL interupt
* FIXME: use the WOL interupt
*/
*/
enable_wol_mode
(
dev
,
0
);
enable_wol_mode
(
dev
,
0
);
}
else
{
}
else
{
...
@@ -2554,9 +2654,8 @@ static int natsemi_suspend (struct pci_dev *pdev, u32 state)
...
@@ -2554,9 +2654,8 @@ static int natsemi_suspend (struct pci_dev *pdev, u32 state)
writel
(
np
->
SavedClkRun
,
ioaddr
+
ClkRun
);
writel
(
np
->
SavedClkRun
,
ioaddr
+
ClkRun
);
}
}
}
}
}
else
{
netif_device_detach
(
dev
);
}
}
netif_device_detach
(
dev
);
rtnl_unlock
();
rtnl_unlock
();
return
0
;
return
0
;
}
}
...
@@ -2571,20 +2670,23 @@ static int natsemi_resume (struct pci_dev *pdev)
...
@@ -2571,20 +2670,23 @@ static int natsemi_resume (struct pci_dev *pdev)
if
(
netif_device_present
(
dev
))
if
(
netif_device_present
(
dev
))
goto
out
;
goto
out
;
if
(
netif_running
(
dev
))
{
if
(
netif_running
(
dev
))
{
BUG_ON
(
!
np
->
hands_off
);
pci_enable_device
(
pdev
);
pci_enable_device
(
pdev
);
/* pci_power_on(pdev); */
/* pci_power_on(pdev); */
natsemi_reset
(
dev
);
natsemi_reset
(
dev
);
init_ring
(
dev
);
init_ring
(
dev
);
disable_irq
(
dev
->
irq
);
spin_lock_irq
(
&
np
->
lock
);
spin_lock_irq
(
&
np
->
lock
);
np
->
hands_off
=
0
;
init_registers
(
dev
);
init_registers
(
dev
);
netif_device_attach
(
dev
);
netif_device_attach
(
dev
);
spin_unlock_irq
(
&
np
->
lock
);
spin_unlock_irq
(
&
np
->
lock
);
enable_irq
(
dev
->
irq
);
mod_timer
(
&
np
->
timer
,
jiffies
+
1
*
HZ
);
mod_timer
(
&
np
->
timer
,
jiffies
+
1
*
HZ
);
}
else
{
netif_device_attach
(
dev
);
}
}
netif_device_attach
(
dev
);
out:
out:
rtnl_unlock
();
rtnl_unlock
();
return
0
;
return
0
;
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment