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
Kirill Smelkov
linux
Commits
d0051184
Commit
d0051184
authored
Nov 07, 2002
by
Jeff Garzik
Browse files
Options
Browse Files
Download
Plain Diff
Merge redhat.com:/home/jgarzik/repo/linus-2.5
into redhat.com:/home/jgarzik/repo/net-drivers-2.5
parents
56c32f41
51449c8c
Changes
3
Show whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
370 additions
and
120 deletions
+370
-120
drivers/net/dl2k.c
drivers/net/dl2k.c
+200
-38
drivers/net/dl2k.h
drivers/net/dl2k.h
+2
-0
drivers/net/sundance.c
drivers/net/sundance.c
+168
-82
No files found.
drivers/net/dl2k.c
View file @
d0051184
...
...
@@ -30,17 +30,32 @@
1.09 2002/03/07 Move rx-poll-now to re-fill loop.
Added rio_timer() to watch rx buffers.
1.10 2002/04/16 Fixed miscount of carrier error.
1.11 2002/05/23 Added ISR schedule scheme
.
1.11 2002/05/23 Added ISR schedule scheme
Fixed miscount of rx frame error for DGE-550SX.
Fixed VLAN bug.
1.12 2002/06/13 Lock tx_coalesce=1 on 10/100Mbps mode.
*/
1.13 2002/08/13 1. Fix disconnection (many tx:carrier/rx:frame
errs) with some mainboards.
2. Use definition "DRV_NAME" "DRV_VERSION"
"DRV_RELDATE" for flexibility.
1.14 2002/08/14 Support ethtool.
1.15 2002/08/27 Changed the default media to Auto-Negotiation
for the fiber devices.
1.16 2002/09/04 More power down time for fiber devices auto-
negotiation.
Fix disconnect bug after ifup and ifdown.
1.17 2002/10/03 Fix RMON statistics overflow.
Always use I/O mapping to access eeprom,
avoid system freezing with some chipsets.
*/
#define DRV_NAME "D-Link DL2000-based linux driver"
#define DRV_VERSION "v1.17"
#define DRV_RELDATE "2002/10/04"
#include "dl2k.h"
static
char
version
[]
__devinitdata
=
KERN_INFO
"D-Link DL2000-based linux driver v1.12 2002/06/13
\n
"
;
KERN_INFO
DRV_NAME
" "
DRV_VERSION
" "
DRV_RELDATE
"
\n
"
;
#define MAX_UNITS 8
static
int
mtu
[
MAX_UNITS
];
static
int
vlan
[
MAX_UNITS
];
...
...
@@ -92,6 +107,7 @@ static int change_mtu (struct net_device *dev, int new_mtu);
static
void
set_multicast
(
struct
net_device
*
dev
);
static
struct
net_device_stats
*
get_stats
(
struct
net_device
*
dev
);
static
int
clear_stats
(
struct
net_device
*
dev
);
static
int
rio_ethtool_ioctl
(
struct
net_device
*
dev
,
void
*
useraddr
);
static
int
rio_ioctl
(
struct
net_device
*
dev
,
struct
ifreq
*
rq
,
int
cmd
);
static
int
rio_close
(
struct
net_device
*
dev
);
static
int
find_miiphy
(
struct
net_device
*
dev
);
...
...
@@ -257,12 +273,8 @@ rio_probe1 (struct pci_dev *pdev, const struct pci_device_id *ent)
np
->
link_status
=
0
;
/* Set media and reset PHY */
if
(
np
->
phy_media
)
{
/* default 1000mbps_fd for fiber deivices */
if
(
np
->
an_enable
==
1
)
{
np
->
an_enable
=
0
;
np
->
speed
=
1000
;
np
->
full_duplex
=
1
;
}
else
if
(
np
->
an_enable
==
2
)
{
/* default Auto-Negotiation for fiber deivices */
if
(
np
->
an_enable
==
2
)
{
np
->
an_enable
=
1
;
}
mii_set_media_pcs
(
dev
);
...
...
@@ -275,10 +287,6 @@ rio_probe1 (struct pci_dev *pdev, const struct pci_device_id *ent)
}
pci_read_config_byte
(
pdev
,
PCI_REVISION_ID
,
&
np
->
pci_rev_id
);
/* Reset all logic functions */
writew
(
GlobalReset
|
DMAReset
|
FIFOReset
|
NetworkReset
|
HostReset
,
ioaddr
+
ASICCtrl
+
2
);
err
=
register_netdev
(
dev
);
if
(
err
)
goto
err_out_unmap_rx
;
...
...
@@ -357,11 +365,16 @@ parse_eeprom (struct net_device *dev)
int
cid
,
next
;
#ifdef MEM_MAPPING
ioaddr
=
pci_resource_start
(
np
->
pdev
,
0
);
#endif
/* Read eeprom */
for
(
i
=
0
;
i
<
128
;
i
++
)
{
((
u16
*
)
sromdata
)[
i
]
=
le16_to_cpu
(
read_eeprom
(
ioaddr
,
i
));
}
#ifdef MEM_MAPPING
ioaddr
=
dev
->
base_addr
;
#endif
/* Check CRC */
crc
=
~
ether_crc_le
(
256
-
4
,
sromdata
);
if
(
psrom
->
crc
!=
crc
)
{
...
...
@@ -421,10 +434,17 @@ rio_open (struct net_device *dev)
struct
netdev_private
*
np
=
dev
->
priv
;
long
ioaddr
=
dev
->
base_addr
;
int
i
;
u16
macctrl
;
i
=
request_irq
(
dev
->
irq
,
&
rio_interrupt
,
SA_SHIRQ
,
dev
->
name
,
dev
);
if
(
i
)
return
i
;
/* Reset all logic functions */
writew
(
GlobalReset
|
DMAReset
|
FIFOReset
|
NetworkReset
|
HostReset
,
ioaddr
+
ASICCtrl
+
2
);
mdelay
(
10
);
/* DebugCtrl bit 4, 5, 9 must set */
writel
(
readl
(
ioaddr
+
DebugCtrl
)
|
0x0230
,
ioaddr
+
DebugCtrl
);
...
...
@@ -448,7 +468,7 @@ rio_open (struct net_device *dev)
writeb
(
0xff
,
ioaddr
+
TxDMAPollPeriod
);
writeb
(
0x30
,
ioaddr
+
RxDMABurstThresh
);
writeb
(
0x30
,
ioaddr
+
RxDMAUrgentThresh
);
writel
(
0x0007ffff
,
ioaddr
+
RmonStatMask
);
/* clear statistics */
clear_stats
(
dev
);
...
...
@@ -467,9 +487,6 @@ rio_open (struct net_device *dev)
ioaddr
+
MACCtrl
);
}
/* Enable default interrupts */
EnableInt
();
init_timer
(
&
np
->
timer
);
np
->
timer
.
expires
=
jiffies
+
1
*
HZ
;
np
->
timer
.
data
=
(
unsigned
long
)
dev
;
...
...
@@ -480,7 +497,17 @@ rio_open (struct net_device *dev)
writel
(
readl
(
ioaddr
+
MACCtrl
)
|
StatsEnable
|
RxEnable
|
TxEnable
,
ioaddr
+
MACCtrl
);
macctrl
=
0
;
macctrl
|=
(
np
->
vlan
)
?
AutoVLANuntagging
:
0
;
macctrl
|=
(
np
->
full_duplex
)
?
DuplexSelect
:
0
;
macctrl
|=
(
np
->
tx_flow
)
?
TxFlowControlEnable
:
0
;
macctrl
|=
(
np
->
rx_flow
)
?
RxFlowControlEnable
:
0
;
writew
(
macctrl
,
ioaddr
+
MACCtrl
);
netif_start_queue
(
dev
);
/* Enable default interrupts */
EnableInt
();
return
0
;
}
...
...
@@ -626,7 +653,7 @@ start_xmit (struct sk_buff *skb, struct net_device *dev)
}
#endif
if
(
np
->
vlan
)
{
t
xdesc
->
status
|
=
t
fc_vlan_tag
=
cpu_to_le64
(
VLANTagInsert
)
|
(
cpu_to_le64
(
np
->
vlan
)
<<
32
)
|
(
cpu_to_le64
(
skb
->
priority
)
<<
45
);
...
...
@@ -720,9 +747,10 @@ rio_free_tx (struct net_device *dev, int irq)
long
flag
=
0
;
if
(
irq
)
spin_lock_irqsave
(
&
np
->
tx_lock
,
flag
);
else
spin_lock
(
&
np
->
tx_lock
);
else
spin_lock_irqsave
(
&
np
->
tx_lock
,
flag
);
/* Free used tx skbuffs */
while
(
entry
!=
np
->
cur_tx
)
{
struct
sk_buff
*
skb
;
...
...
@@ -743,9 +771,9 @@ rio_free_tx (struct net_device *dev, int irq)
tx_use
++
;
}
if
(
irq
)
spin_unlock_irqrestore
(
&
np
->
tx_lock
,
flag
);
else
spin_unlock
(
&
np
->
tx_lock
);
else
spin_unlock_irqrestore
(
&
np
->
tx_lock
,
flag
);
np
->
old_tx
=
entry
;
/* If the ring is no longer full, clear tx_full and
...
...
@@ -809,8 +837,13 @@ tx_error (struct net_device *dev, int tx_status)
/* Let TxStartThresh stay default value */
}
/* Maximum Collisions */
#ifdef ETHER_STATS
if
(
tx_status
&
0x08
)
np
->
stats
.
collisions16
++
;
#else
if
(
tx_status
&
0x08
)
np
->
stats
.
collisions
++
;
#endif
/* Restart the Tx */
writel
(
readw
(
dev
->
base_addr
+
MACCtrl
)
|
TxEnable
,
ioaddr
+
MACCtrl
);
}
...
...
@@ -1103,6 +1136,7 @@ set_multicast (struct net_device *dev)
u16
rx_mode
=
0
;
int
i
;
int
bit
;
int
index
,
crc
;
struct
dev_mc_list
*
mclist
;
struct
netdev_private
*
np
=
dev
->
priv
;
...
...
@@ -1122,17 +1156,16 @@ set_multicast (struct net_device *dev)
rx_mode
=
ReceiveBroadcast
|
ReceiveMulticastHash
|
ReceiveUnicast
;
for
(
i
=
0
,
mclist
=
dev
->
mc_list
;
mclist
&&
i
<
dev
->
mc_count
;
i
++
,
mclist
=
mclist
->
next
)
{
int
index
=
0
;
int
crc
=
ether_crc_le
(
ETH_ALEN
,
mclist
->
dmi_addr
);
i
++
,
mclist
=
mclist
->
next
)
{
crc
=
ether_crc_le
(
ETH_ALEN
,
mclist
->
dmi_addr
);
/* The inverted high significant 6 bits of CRC are
used as an index to hashtable */
for
(
bit
=
0
;
bit
<
6
;
bit
++
)
if
(
crc
&
(
1
<<
(
31
-
bit
)))
index
|=
(
1
<<
bit
);
for
(
index
=
0
,
bit
=
0
;
bit
<
6
;
bit
++
)
{
if
(
test_bit
(
31
-
bit
,
&
crc
))
{
set_bit
(
bit
,
&
index
);
}
}
hash_table
[
index
/
32
]
|=
(
1
<<
(
index
%
32
));
}
}
else
{
...
...
@@ -1148,6 +1181,132 @@ set_multicast (struct net_device *dev)
writew
(
rx_mode
,
ioaddr
+
ReceiveMode
);
}
static
int
rio_ethtool_ioctl
(
struct
net_device
*
dev
,
void
*
useraddr
)
{
struct
netdev_private
*
np
=
dev
->
priv
;
u32
ethcmd
;
if
(
copy_from_user
(
&
ethcmd
,
useraddr
,
sizeof
(
ethcmd
)))
return
-
EFAULT
;
switch
(
ethcmd
)
{
case
ETHTOOL_GDRVINFO
:
{
struct
ethtool_drvinfo
info
=
{
ETHTOOL_GDRVINFO
};
strcpy
(
info
.
driver
,
"DL2K"
);
strcpy
(
info
.
version
,
DRV_VERSION
);
strcpy
(
info
.
bus_info
,
np
->
pdev
->
slot_name
);
memset
(
&
info
.
fw_version
,
0
,
sizeof
(
info
.
fw_version
));
if
(
copy_to_user
(
useraddr
,
&
info
,
sizeof
(
info
)))
return
-
EFAULT
;
return
0
;
}
case
ETHTOOL_GSET
:
{
struct
ethtool_cmd
cmd
=
{
ETHTOOL_GSET
};
if
(
np
->
phy_media
)
{
/* fiber device */
cmd
.
supported
=
SUPPORTED_Autoneg
|
SUPPORTED_FIBRE
;
cmd
.
advertising
=
ADVERTISED_Autoneg
|
ADVERTISED_FIBRE
;
cmd
.
port
=
PORT_FIBRE
;
cmd
.
transceiver
=
XCVR_INTERNAL
;
}
else
{
/* copper device */
cmd
.
supported
=
SUPPORTED_10baseT_Half
|
SUPPORTED_10baseT_Full
|
SUPPORTED_100baseT_Half
|
SUPPORTED_100baseT_Full
|
SUPPORTED_1000baseT_Full
|
SUPPORTED_Autoneg
|
SUPPORTED_MII
;
cmd
.
advertising
=
ADVERTISED_10baseT_Half
|
ADVERTISED_10baseT_Full
|
ADVERTISED_100baseT_Half
|
ADVERTISED_100baseT_Full
|
ADVERTISED_1000baseT_Full
|
ADVERTISED_Autoneg
|
ADVERTISED_MII
;
cmd
.
port
=
PORT_MII
;
cmd
.
transceiver
=
XCVR_INTERNAL
;
}
if
(
np
->
link_status
)
{
cmd
.
speed
=
np
->
speed
;
cmd
.
duplex
=
np
->
full_duplex
?
DUPLEX_FULL
:
DUPLEX_HALF
;
}
else
{
cmd
.
speed
=
-
1
;
cmd
.
duplex
=
-
1
;
}
if
(
np
->
an_enable
)
cmd
.
autoneg
=
AUTONEG_ENABLE
;
else
cmd
.
autoneg
=
AUTONEG_DISABLE
;
cmd
.
phy_address
=
np
->
phy_addr
;
if
(
copy_to_user
(
useraddr
,
&
cmd
,
sizeof
(
cmd
)))
return
-
EFAULT
;
return
0
;
}
case
ETHTOOL_SSET
:
{
struct
ethtool_cmd
cmd
;
if
(
copy_from_user
(
&
cmd
,
useraddr
,
sizeof
(
cmd
)))
return
-
EFAULT
;
netif_carrier_off
(
dev
);
if
(
cmd
.
autoneg
==
AUTONEG_ENABLE
)
{
if
(
np
->
an_enable
)
return
0
;
else
{
np
->
an_enable
=
1
;
mii_set_media
(
dev
);
return
0
;
}
}
else
{
np
->
an_enable
=
0
;
if
(
np
->
speed
==
1000
){
cmd
.
speed
=
SPEED_100
;
cmd
.
duplex
=
DUPLEX_FULL
;
printk
(
"Warning!! Can't disable Auto negotiation in 1000Mbps, change to Manul 100Mbps, Full duplex.
\n
"
);
}
switch
(
cmd
.
speed
+
cmd
.
duplex
){
case
SPEED_10
+
DUPLEX_HALF
:
np
->
speed
=
10
;
np
->
full_duplex
=
0
;
break
;
case
SPEED_10
+
DUPLEX_FULL
:
np
->
speed
=
10
;
np
->
full_duplex
=
1
;
break
;
case
SPEED_100
+
DUPLEX_HALF
:
np
->
speed
=
100
;
np
->
full_duplex
=
0
;
break
;
case
SPEED_100
+
DUPLEX_FULL
:
np
->
speed
=
100
;
np
->
full_duplex
=
1
;
break
;
case
SPEED_1000
+
DUPLEX_HALF
:
/* not supported */
case
SPEED_1000
+
DUPLEX_FULL
:
/* not supported */
default:
return
-
EINVAL
;
}
mii_set_media
(
dev
);
}
return
0
;
}
#ifdef ETHTOOL_GLINK
case
ETHTOOL_GLINK
:{
struct
ethtool_value
link
=
{
ETHTOOL_GLINK
};
link
.
data
=
np
->
link_status
;
if
(
copy_to_user
(
useraddr
,
&
link
,
sizeof
(
link
)))
return
-
EFAULT
;
return
0
;
}
#endif
default:
return
-
EOPNOTSUPP
;
}
}
static
int
rio_ioctl
(
struct
net_device
*
dev
,
struct
ifreq
*
rq
,
int
cmd
)
{
...
...
@@ -1160,6 +1319,8 @@ rio_ioctl (struct net_device *dev, struct ifreq *rq, int cmd)
phy_addr
=
np
->
phy_addr
;
switch
(
cmd
)
{
case
SIOCETHTOOL
:
return
rio_ethtool_ioctl
(
dev
,
(
void
*
)
rq
->
ifr_data
);
case
SIOCDEVPRIVATE
:
break
;
...
...
@@ -1210,14 +1371,15 @@ rio_ioctl (struct net_device *dev, struct ifreq *rq, int cmd)
#define EEP_READ 0x0200
#define EEP_BUSY 0x8000
/* Read the EEPROM word */
/* We use I/O instruction to read/write eeprom to avoid fail on some machines */
int
read_eeprom
(
long
ioaddr
,
int
eep_addr
)
{
int
i
=
1000
;
write
w
(
EEP_READ
|
(
eep_addr
&
0xff
),
ioaddr
+
EepromCtrl
);
out
w
(
EEP_READ
|
(
eep_addr
&
0xff
),
ioaddr
+
EepromCtrl
);
while
(
i
--
>
0
)
{
if
(
!
(
read
w
(
ioaddr
+
EepromCtrl
)
&
EEP_BUSY
))
{
return
read
w
(
ioaddr
+
EepromData
);
if
(
!
(
in
w
(
ioaddr
+
EepromCtrl
)
&
EEP_BUSY
))
{
return
in
w
(
ioaddr
+
EepromData
);
}
}
return
0
;
...
...
@@ -1464,7 +1626,7 @@ mii_set_media (struct net_device *dev)
/* 3) Power Down */
bmcr
.
image
=
0x1940
;
/* must be 0x1940 */
mii_write
(
dev
,
phy_addr
,
MII_BMCR
,
bmcr
.
image
);
mdelay
(
10
);
/* wait a certain time */
mdelay
(
10
0
);
/* wait a certain time */
/* 4) Advertise nothing */
mii_write
(
dev
,
phy_addr
,
MII_ANAR
,
0
);
...
...
drivers/net/dl2k.h
View file @
d0051184
...
...
@@ -27,9 +27,11 @@
#include <linux/skbuff.h>
#include <linux/init.h>
#include <linux/crc32.h>
#include <linux/ethtool.h>
#include <asm/processor.h>
/* Processor type for cache alignment. */
#include <asm/bitops.h>
#include <asm/io.h>
#include <asm/uaccess.h>
#include <linux/delay.h>
#include <linux/spinlock.h>
#include <linux/time.h>
...
...
drivers/net/sundance.c
View file @
d0051184
...
...
@@ -63,14 +63,20 @@
- Better rx buf size calculation (Donald Becker)
Version LK1.05 (D-Link):
- fix DFE-580TX packet drop issue
- fix reset_tx logic
- Fix DFE-580TX packet drop issue (for DL10050C)
- Fix reset_tx logic
Version LK1.06 (D-Link):
- Fix crash while unloading driver
Versin LK1.06b (D-Link):
- New tx scheme, adaptive tx_coalesce
*/
#define DRV_NAME "sundance"
#define DRV_VERSION "1.01+LK1.0
5
"
#define DRV_RELDATE "
28-Sep
-2002"
#define DRV_VERSION "1.01+LK1.0
6b
"
#define DRV_RELDATE "
6-Nov
-2002"
/* The user-configurable values.
...
...
@@ -87,7 +93,6 @@ static int multicast_filter_limit = 32;
This chip can receive into offset buffers, so the Alpha does not
need a copy-align. */
static
int
rx_copybreak
;
static
int
tx_coalesce
=
1
;
static
int
flowctrl
=
1
;
/* media[] specifies the media type the NIC operates at.
...
...
@@ -114,7 +119,7 @@ static char *media[MAX_UNITS];
bonding and packet priority, and more than 128 requires modifying the
Tx error recovery.
Large receive rings merely waste memory. */
#define TX_RING_SIZE
64
#define TX_RING_SIZE
32
#define TX_QUEUE_LEN (TX_RING_SIZE - 1)
/* Limit ring entries actually used. */
#define RX_RING_SIZE 64
#define RX_BUDGET 32
...
...
@@ -459,7 +464,9 @@ struct netdev_private {
unsigned
int
an_enable
:
1
;
unsigned
int
speed
;
struct
tasklet_struct
rx_tasklet
;
struct
tasklet_struct
tx_tasklet
;
int
budget
;
int
cur_task
;
/* Multicast and receive mode. */
spinlock_t
mcastlock
;
/* SMP lock multicast updates. */
u16
mcast_filter
[
4
];
...
...
@@ -468,6 +475,7 @@ struct netdev_private {
int
mii_preamble_required
;
unsigned
char
phys
[
MII_CNT
];
/* MII device addresses, only first one used. */
struct
pci_dev
*
pci_dev
;
unsigned
char
pci_rev_id
;
};
/* The station address location in the EEPROM. */
...
...
@@ -489,6 +497,7 @@ static int start_tx(struct sk_buff *skb, struct net_device *dev);
static
int
reset_tx
(
struct
net_device
*
dev
);
static
void
intr_handler
(
int
irq
,
void
*
dev_instance
,
struct
pt_regs
*
regs
);
static
void
rx_poll
(
unsigned
long
data
);
static
void
tx_poll
(
unsigned
long
data
);
static
void
refill_rx
(
struct
net_device
*
dev
);
static
void
netdev_error
(
struct
net_device
*
dev
,
int
intr_status
);
static
void
netdev_error
(
struct
net_device
*
dev
,
int
intr_status
);
...
...
@@ -557,6 +566,7 @@ static int __devinit sundance_probe1 (struct pci_dev *pdev,
np
->
msg_enable
=
(
1
<<
debug
)
-
1
;
spin_lock_init
(
&
np
->
lock
);
tasklet_init
(
&
np
->
rx_tasklet
,
rx_poll
,
(
unsigned
long
)
dev
);
tasklet_init
(
&
np
->
tx_tasklet
,
tx_poll
,
(
unsigned
long
)
dev
);
ring_space
=
pci_alloc_consistent
(
pdev
,
TX_TOTAL_SIZE
,
&
ring_dma
);
if
(
!
ring_space
)
...
...
@@ -588,6 +598,8 @@ static int __devinit sundance_probe1 (struct pci_dev *pdev,
dev
->
change_mtu
=
&
change_mtu
;
pci_set_drvdata
(
pdev
,
dev
);
pci_read_config_byte
(
pdev
,
PCI_REVISION_ID
,
&
np
->
pci_rev_id
);
i
=
register_netdev
(
dev
);
if
(
i
)
goto
err_out_unmap_rx
;
...
...
@@ -650,10 +662,6 @@ static int __devinit sundance_probe1 (struct pci_dev *pdev,
np
->
an_enable
=
1
;
}
}
if
(
tx_coalesce
<
1
)
tx_coalesce
=
1
;
else
if
(
tx_coalesce
>
TX_QUEUE_LEN
-
1
)
tx_coalesce
=
TX_QUEUE_LEN
-
1
;
if
(
flowctrl
==
0
)
np
->
flowctrl
=
0
;
}
...
...
@@ -867,6 +875,7 @@ static int netdev_open(struct net_device *dev)
writeb
(
100
,
ioaddr
+
RxDMAPollPeriod
);
writeb
(
127
,
ioaddr
+
TxDMAPollPeriod
);
/* Fix DFE-580TX packet drop issue */
if
(
np
->
pci_rev_id
>=
0x14
)
writeb
(
0x01
,
ioaddr
+
DebugCtrl1
);
netif_start_queue
(
dev
);
...
...
@@ -944,6 +953,8 @@ static void tx_timeout(struct net_device *dev)
long
ioaddr
=
dev
->
base_addr
;
long
flag
;
netif_stop_queue
(
dev
);
tasklet_disable
(
&
np
->
tx_tasklet
);
writew
(
0
,
ioaddr
+
IntrEnable
);
printk
(
KERN_WARNING
"%s: Transmit timed out, TxStatus %2.2x "
"TxFrameId %2.2x,"
...
...
@@ -952,31 +963,39 @@ static void tx_timeout(struct net_device *dev)
{
int
i
;
printk
(
KERN_DEBUG
" Rx ring %p: "
,
np
->
rx_ring
);
for
(
i
=
0
;
i
<
RX_RING_SIZE
;
i
++
)
printk
(
" %8.8x"
,
(
unsigned
int
)
np
->
rx_ring
[
i
].
status
);
printk
(
"
\n
"
KERN_DEBUG
" Tx ring %p: "
,
np
->
tx_ring
);
for
(
i
=
0
;
i
<
TX_RING_SIZE
;
i
++
)
printk
(
" %8.8x"
,
np
->
tx_ring
[
i
].
status
);
printk
(
"
\n
"
);
printk
(
KERN_DEBUG
"cur_tx=%d dirty_tx=%d
\n
"
,
np
->
cur_tx
,
np
->
dirty_tx
);
for
(
i
=
0
;
i
<
TX_RING_SIZE
;
i
++
)
{
printk
(
KERN_DEBUG
"%02x %08x %08x %08x(%02x) %08x %08x
\n
"
,
i
,
np
->
tx_ring_dma
+
i
*
sizeof
(
*
np
->
tx_ring
),
np
->
tx_ring
[
i
].
next_desc
,
np
->
tx_ring
[
i
].
status
,
(
np
->
tx_ring
[
i
].
status
>>
2
)
&
0xff
,
np
->
tx_ring
[
i
].
frag
[
0
].
addr
,
np
->
tx_ring
[
i
].
frag
[
0
].
length
);
}
printk
(
KERN_DEBUG
"TxListPtr=%08x netif_queue_stopped=%d
\n
"
,
readl
(
dev
->
base_addr
+
TxListPtr
),
netif_queue_stopped
(
dev
));
printk
(
KERN_DEBUG
"cur_tx=%d(%02x) dirty_tx=%d(%02x)
\n
"
,
np
->
cur_tx
,
np
->
cur_tx
%
TX_RING_SIZE
,
np
->
dirty_tx
,
np
->
dirty_tx
%
TX_RING_SIZE
);
printk
(
KERN_DEBUG
"cur_rx=%d dirty_rx=%d
\n
"
,
np
->
cur_rx
,
np
->
dirty_rx
);
printk
(
KERN_DEBUG
"cur_task=%d
\n
"
,
np
->
cur_task
);
}
spin_lock_irqsave
(
&
np
->
lock
,
flag
);
/* Stop and restart the chip's Tx processes . */
reset_tx
(
dev
);
spin_unlock_irqrestore
(
&
np
->
lock
,
flag
);
/* Perhaps we should reinitialize the hardware here. */
dev
->
if_port
=
0
;
/* Stop and restart the chip's Tx processes . */
/* Trigger an immediate transmit demand. */
writew
(
DEFAULT_INTR
,
ioaddr
+
IntrEnable
);
dev
->
trans_start
=
jiffies
;
np
->
stats
.
tx_errors
++
;
if
(
!
netif_queue_stopped
(
dev
))
if
(
np
->
cur_tx
-
np
->
dirty_tx
<
TX_QUEUE_LEN
-
4
)
{
netif_wake_queue
(
dev
);
}
writew
(
DEFAULT_INTR
,
ioaddr
+
IntrEnable
);
tasklet_enable
(
&
np
->
tx_tasklet
);
}
...
...
@@ -988,6 +1007,7 @@ static void init_ring(struct net_device *dev)
np
->
cur_rx
=
np
->
cur_tx
=
0
;
np
->
dirty_rx
=
np
->
dirty_tx
=
0
;
np
->
cur_task
=
0
;
np
->
rx_buf_sz
=
(
dev
->
mtu
<=
1520
?
PKT_BUF_SZ
:
dev
->
mtu
+
16
);
...
...
@@ -1022,39 +1042,57 @@ static void init_ring(struct net_device *dev)
return
;
}
static
void
tx_poll
(
unsigned
long
data
)
{
struct
net_device
*
dev
=
(
struct
net_device
*
)
data
;
struct
netdev_private
*
np
=
dev
->
priv
;
unsigned
head
=
np
->
cur_task
%
TX_RING_SIZE
;
struct
netdev_desc
*
txdesc
=
&
np
->
tx_ring
[(
np
->
cur_tx
-
1
)
%
TX_RING_SIZE
];
/* Chain the next pointer */
for
(;
np
->
cur_tx
-
np
->
cur_task
>
0
;
np
->
cur_task
++
)
{
int
entry
=
np
->
cur_task
%
TX_RING_SIZE
;
txdesc
=
&
np
->
tx_ring
[
entry
];
if
(
np
->
last_tx
)
{
np
->
last_tx
->
next_desc
=
cpu_to_le32
(
np
->
tx_ring_dma
+
entry
*
sizeof
(
struct
netdev_desc
));
}
np
->
last_tx
=
txdesc
;
}
/* Indicate the latest descriptor of tx ring */
txdesc
->
status
|=
cpu_to_le32
(
DescIntrOnTx
);
if
(
readl
(
dev
->
base_addr
+
TxListPtr
)
==
0
)
writel
(
np
->
tx_ring_dma
+
head
*
sizeof
(
struct
netdev_desc
),
dev
->
base_addr
+
TxListPtr
);
return
;
}
static
int
start_tx
(
struct
sk_buff
*
skb
,
struct
net_device
*
dev
)
{
struct
netdev_private
*
np
=
dev
->
priv
;
struct
netdev_desc
*
txdesc
;
unsigned
entry
;
long
ioaddr
=
dev
->
base_addr
;
/* Note: Ordering is important here, set the field with the
"ownership" bit last, and only then increment cur_tx. */
/* Calculate the next Tx descriptor entry. */
entry
=
np
->
cur_tx
%
TX_RING_SIZE
;
np
->
tx_skbuff
[
entry
]
=
skb
;
txdesc
=
&
np
->
tx_ring
[
entry
];
txdesc
->
next_desc
=
0
;
/* Note: disable the interrupt generation here before releasing. */
if
(
entry
%
tx_coalesce
==
0
)
{
txdesc
->
status
=
cpu_to_le32
((
entry
<<
2
)
|
DescIntrOnTx
|
DisableAlign
);
}
else
{
txdesc
->
status
=
cpu_to_le32
((
entry
<<
2
)
|
DisableAlign
);
}
txdesc
->
frag
[
0
].
addr
=
cpu_to_le32
(
pci_map_single
(
np
->
pci_dev
,
skb
->
data
,
skb
->
len
,
PCI_DMA_TODEVICE
));
txdesc
->
frag
[
0
].
length
=
cpu_to_le32
(
skb
->
len
|
LastFrag
);
if
(
np
->
last_tx
)
np
->
last_tx
->
next_desc
=
cpu_to_le32
(
np
->
tx_ring_dma
+
entry
*
sizeof
(
struct
netdev_desc
));
np
->
last_tx
=
txdesc
;
/* Increment cur_tx before tasklet_schedule() */
np
->
cur_tx
++
;
mb
();
/* Schedule a tx_poll() task */
tasklet_schedule
(
&
np
->
tx_tasklet
);
/* On some architectures: explicitly flush cache lines here. */
if
(
np
->
cur_tx
-
np
->
dirty_tx
<
TX_QUEUE_LEN
-
1
...
...
@@ -1063,23 +1101,16 @@ start_tx (struct sk_buff *skb, struct net_device *dev)
}
else
{
netif_stop_queue
(
dev
);
}
/* Side effect: The read wakes the potentially-idle transmit channel. */
if
(
readl
(
dev
->
base_addr
+
TxListPtr
)
==
0
)
writel
(
np
->
tx_ring_dma
+
entry
*
sizeof
(
*
np
->
tx_ring
),
dev
->
base_addr
+
TxListPtr
);
dev
->
trans_start
=
jiffies
;
if
(
netif_msg_tx_queued
(
np
))
{
printk
(
KERN_DEBUG
"%s: Transmit frame #%d queued in slot %d.
\n
"
,
dev
->
name
,
np
->
cur_tx
,
entry
);
}
if
(
tx_coalesce
>
1
)
writel
(
1000
,
ioaddr
+
DownCounter
);
return
0
;
}
/* Reset hardware tx and reset TxListPtr to TxFrameId */
/* Reset hardware tx and free all of tx buffers */
static
int
reset_tx
(
struct
net_device
*
dev
)
{
...
...
@@ -1089,8 +1120,8 @@ reset_tx (struct net_device *dev)
int
i
;
int
irq
=
in_interrupt
();
/*
reset tx logic
*/
write
l
(
0
,
dev
->
base_addr
+
TxListPtr
);
/*
Reset tx logic, TxListPtr will be cleaned
*/
write
w
(
TxDisable
,
ioaddr
+
MACCtrl1
);
writew
(
TxReset
|
DMAReset
|
FIFOReset
|
NetworkReset
,
ioaddr
+
ASICCtrl
+
2
);
for
(
i
=
50
;
i
>
0
;
i
--
)
{
...
...
@@ -1114,11 +1145,13 @@ reset_tx (struct net_device *dev)
}
}
np
->
cur_tx
=
np
->
dirty_tx
=
0
;
np
->
cur_task
=
0
;
writew
(
StatsEnable
|
RxEnable
|
TxEnable
,
ioaddr
+
MACCtrl1
);
return
0
;
}
/* The interrupt handler
does all of the Rx thread work and cleans up
a
fter the Tx thread.
*/
/* The interrupt handler
cleans up after the Tx thread,
a
nd schedule a Rx thread work
*/
static
void
intr_handler
(
int
irq
,
void
*
dev_instance
,
struct
pt_regs
*
rgs
)
{
struct
net_device
*
dev
=
(
struct
net_device
*
)
dev_instance
;
...
...
@@ -1126,6 +1159,8 @@ static void intr_handler(int irq, void *dev_instance, struct pt_regs *rgs)
long
ioaddr
;
int
boguscnt
=
max_interrupt_work
;
int
hw_frame_id
;
int
tx_cnt
;
int
tx_status
;
ioaddr
=
dev
->
base_addr
;
np
=
dev
->
priv
;
...
...
@@ -1148,11 +1183,9 @@ static void intr_handler(int irq, void *dev_instance, struct pt_regs *rgs)
np
->
budget
=
RX_BUDGET
;
tasklet_schedule
(
&
np
->
rx_tasklet
);
}
if
(
intr_status
&
(
IntrTxDone
|
IntrDrvRqst
))
{
int
boguscnt
=
32
;
int
tx_status
=
readw
(
ioaddr
+
TxStatus
);
while
(
tx_status
&
0x80
)
{
tx_status
=
readw
(
ioaddr
+
TxStatus
);
for
(
tx_cnt
=
32
;
tx_status
&
0x80
;
--
tx_cnt
)
{
if
(
netif_msg_tx_done
(
np
))
printk
(
"%s: Transmit status is %2.2x.
\n
"
,
...
...
@@ -1179,19 +1212,43 @@ static void intr_handler(int irq, void *dev_instance, struct pt_regs *rgs)
/* Yup, this is a documentation bug. It cost me *hours*. */
writew
(
0
,
ioaddr
+
TxStatus
);
tx_status
=
readw
(
ioaddr
+
TxStatus
);
if
(
--
bogus
cnt
<
0
)
if
(
tx_
cnt
<
0
)
break
;
}
hw_frame_id
=
(
tx_status
>>
8
)
&
0xff
;
}
else
{
hw_frame_id
=
readb
(
ioaddr
+
TxFrameId
);
}
if
(
np
->
pci_rev_id
>=
0x14
)
{
spin_lock
(
&
np
->
lock
);
hw_frame_id
=
readb
(
ioaddr
+
TxFrameId
);
for
(;
np
->
cur_tx
-
np
->
dirty_tx
>
0
;
np
->
dirty_tx
++
)
{
int
entry
=
np
->
dirty_tx
%
TX_RING_SIZE
;
struct
sk_buff
*
skb
;
int
sw_frame_id
;
sw_frame_id
=
(
np
->
tx_ring
[
entry
].
status
>>
2
)
&
0xff
;
if
(
sw_frame_id
==
hw_frame_id
)
if
(
sw_frame_id
==
hw_frame_id
&&
!
(
np
->
tx_ring
[
entry
].
status
&
0x00010000
))
break
;
if
(
sw_frame_id
==
(
hw_frame_id
+
1
)
%
TX_RING_SIZE
)
break
;
skb
=
np
->
tx_skbuff
[
entry
];
/* Free the original skb. */
pci_unmap_single
(
np
->
pci_dev
,
np
->
tx_ring
[
entry
].
frag
[
0
].
addr
,
skb
->
len
,
PCI_DMA_TODEVICE
);
dev_kfree_skb_irq
(
np
->
tx_skbuff
[
entry
]);
np
->
tx_skbuff
[
entry
]
=
0
;
np
->
tx_ring
[
entry
].
frag
[
0
].
addr
=
0
;
np
->
tx_ring
[
entry
].
frag
[
0
].
length
=
0
;
}
spin_unlock
(
&
np
->
lock
);
}
else
{
spin_lock
(
&
np
->
lock
);
for
(;
np
->
cur_tx
-
np
->
dirty_tx
>
0
;
np
->
dirty_tx
++
)
{
int
entry
=
np
->
dirty_tx
%
TX_RING_SIZE
;
struct
sk_buff
*
skb
;
if
(
!
(
np
->
tx_ring
[
entry
].
status
&
0x00010000
))
break
;
skb
=
np
->
tx_skbuff
[
entry
];
/* Free the original skb. */
...
...
@@ -1200,14 +1257,17 @@ static void intr_handler(int irq, void *dev_instance, struct pt_regs *rgs)
skb
->
len
,
PCI_DMA_TODEVICE
);
dev_kfree_skb_irq
(
np
->
tx_skbuff
[
entry
]);
np
->
tx_skbuff
[
entry
]
=
0
;
np
->
tx_ring
[
entry
].
frag
[
0
].
addr
=
0
;
np
->
tx_ring
[
entry
].
frag
[
0
].
length
=
0
;
}
spin_unlock
(
&
np
->
lock
);
}
if
(
netif_queue_stopped
(
dev
)
&&
np
->
cur_tx
-
np
->
dirty_tx
<
TX_QUEUE_LEN
-
4
)
{
/* The ring is no longer full, clear
tbusy
. */
/* The ring is no longer full, clear
busy flag
. */
netif_wake_queue
(
dev
);
}
/* Abnormal error summary/uncommon events handlers. */
if
(
intr_status
&
(
IntrPCIErr
|
LinkChange
|
StatsMax
))
netdev_error
(
dev
,
intr_status
);
...
...
@@ -1223,8 +1283,7 @@ static void intr_handler(int irq, void *dev_instance, struct pt_regs *rgs)
if
(
netif_msg_intr
(
np
))
printk
(
KERN_DEBUG
"%s: exiting interrupt, status=%#4.4x.
\n
"
,
dev
->
name
,
readw
(
ioaddr
+
IntrStatus
));
if
(
np
->
cur_tx
-
np
->
dirty_tx
>
0
&&
tx_coalesce
>
1
)
writel
(
100
,
ioaddr
+
DownCounter
);
writel
(
5000
,
ioaddr
+
DownCounter
);
}
...
...
@@ -1246,7 +1305,7 @@ static void rx_poll(unsigned long data)
if
(
--
boguscnt
<
0
)
{
goto
not_done
;
}
if
(
!
(
desc
->
status
&
DescOwn
))
if
(
!
(
frame_
status
&
DescOwn
))
break
;
pkt_len
=
frame_status
&
0x1fff
;
/* Chip omits the CRC. */
if
(
netif_msg_rx_status
(
np
))
...
...
@@ -1555,6 +1614,7 @@ static int netdev_ioctl(struct net_device *dev, struct ifreq *rq, int cmd)
struct
netdev_private
*
np
=
dev
->
priv
;
struct
mii_ioctl_data
*
data
=
(
struct
mii_ioctl_data
*
)
&
rq
->
ifr_data
;
int
rc
;
int
i
;
if
(
!
netif_running
(
dev
))
return
-
EINVAL
;
...
...
@@ -1567,6 +1627,28 @@ static int netdev_ioctl(struct net_device *dev, struct ifreq *rq, int cmd)
rc
=
generic_mii_ioctl
(
&
np
->
mii_if
,
data
,
cmd
,
NULL
);
spin_unlock_irq
(
&
np
->
lock
);
}
switch
(
cmd
)
{
case
SIOCDEVPRIVATE
:
for
(
i
=
0
;
i
<
TX_RING_SIZE
;
i
++
)
{
printk
(
KERN_DEBUG
"%02x %08x %08x %08x(%02x) %08x %08x
\n
"
,
i
,
np
->
tx_ring_dma
+
i
*
sizeof
(
*
np
->
tx_ring
),
np
->
tx_ring
[
i
].
next_desc
,
np
->
tx_ring
[
i
].
status
,
(
np
->
tx_ring
[
i
].
status
>>
2
)
&
0xff
,
np
->
tx_ring
[
i
].
frag
[
0
].
addr
,
np
->
tx_ring
[
i
].
frag
[
0
].
length
);
}
printk
(
KERN_DEBUG
"TxListPtr=%08x netif_queue_stopped=%d
\n
"
,
readl
(
dev
->
base_addr
+
TxListPtr
),
netif_queue_stopped
(
dev
));
printk
(
KERN_DEBUG
"cur_tx=%d(%02x) dirty_tx=%d(%02x)
\n
"
,
np
->
cur_tx
,
np
->
cur_tx
%
TX_RING_SIZE
,
np
->
dirty_tx
,
np
->
dirty_tx
%
TX_RING_SIZE
);
printk
(
KERN_DEBUG
"cur_rx=%d dirty_rx=%d
\n
"
,
np
->
cur_rx
,
np
->
dirty_rx
);
printk
(
KERN_DEBUG
"cur_task=%d
\n
"
,
np
->
cur_task
);
return
0
;
}
return
rc
;
}
...
...
@@ -1595,6 +1677,10 @@ static int netdev_close(struct net_device *dev)
/* Stop the chip's Tx and Rx processes. */
writew
(
TxDisable
|
RxDisable
|
StatsDisable
,
ioaddr
+
MACCtrl1
);
/* Wait and kill tasklet */
tasklet_kill
(
&
np
->
rx_tasklet
);
tasklet_kill
(
&
np
->
tx_tasklet
);
#ifdef __i386__
if
(
netif_msg_hw
(
np
))
{
printk
(
"
\n
"
KERN_DEBUG
" Tx ring at %8.8x:
\n
"
,
...
...
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