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
f9c57480
Commit
f9c57480
authored
Mar 11, 2002
by
Jeff Garzik
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
e100 net driver updates from Intel:
* fix zerocopy * add suspend/resume
parent
50b1b006
Changes
3
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
293 additions
and
45 deletions
+293
-45
drivers/net/e100/e100.h
drivers/net/e100/e100.h
+12
-0
drivers/net/e100/e100_main.c
drivers/net/e100/e100_main.c
+280
-44
drivers/net/e100/e100_phy.c
drivers/net/e100/e100_phy.c
+1
-1
No files found.
drivers/net/e100/e100.h
View file @
f9c57480
...
...
@@ -766,6 +766,8 @@ typedef enum _non_tx_cmd_state_t {
#define IPCB_INSERTVLAN_ENABLE BIT_1
#define IPCB_IP_ACTIVATION_DEFAULT IPCB_HARDWAREPARSING_ENABLE
#define FOLD_CSUM(_XSUM) ((((_XSUM << 16) | (_XSUM >> 16)) + _XSUM) >> 16)
/* Transmit Buffer Descriptor (TBD)*/
typedef
struct
_tbd_t
{
u32
tbd_buf_addr
;
/* Physical Transmit Buffer Address */
...
...
@@ -1008,6 +1010,11 @@ struct e100_private {
u32
wolopts
;
u16
ip_lbytes
;
#endif
#ifdef CONFIG_PM
u32
pci_state
[
16
];
#endif
};
#define E100_AUTONEG 0
...
...
@@ -1030,4 +1037,9 @@ extern unsigned char e100_selftest(struct e100_private *bdp, u32 *st_timeout,
extern
unsigned
char
e100_get_link_state
(
struct
e100_private
*
bdp
);
extern
unsigned
char
e100_wait_scb
(
struct
e100_private
*
bdp
);
extern
void
e100_deisolate_driver
(
struct
e100_private
*
bdp
,
u8
recover
,
u8
full_reset
);
extern
unsigned
char
e100_hw_reset_recover
(
struct
e100_private
*
bdp
,
u32
reset_cmd
);
#endif
drivers/net/e100/e100_main.c
View file @
f9c57480
...
...
@@ -122,7 +122,6 @@ static int e100_do_ethtool_ioctl(struct net_device *, struct ifreq *);
static
void
e100_get_speed_duplex_caps
(
struct
e100_private
*
);
static
int
e100_ethtool_get_settings
(
struct
net_device
*
,
struct
ifreq
*
);
static
int
e100_ethtool_set_settings
(
struct
net_device
*
,
struct
ifreq
*
);
static
void
e100_set_speed_duplex
(
struct
e100_private
*
);
#ifdef ETHTOOL_GDRVINFO
static
int
e100_ethtool_get_drvinfo
(
struct
net_device
*
,
struct
ifreq
*
);
...
...
@@ -163,7 +162,8 @@ static void e100_non_tx_background(unsigned long);
/* Global Data structures and variables */
char
e100_copyright
[]
__devinitdata
=
"Copyright (c) 2002 Intel Corporation"
;
#define E100_VERSION "2.0.20-pre1"
#define E100_VERSION "2.0.21"
#define E100_FULL_DRIVER_NAME "Intel(R) PRO/100 Fast Ethernet Adapter - Loadable driver, ver "
const
char
*
e100_version
=
E100_VERSION
;
...
...
@@ -171,6 +171,13 @@ const char *e100_full_driver_name = E100_FULL_DRIVER_NAME E100_VERSION;
char
*
e100_short_driver_name
=
"e100"
;
static
int
e100nics
=
0
;
#ifdef CONFIG_PM
static
int
e100_save_state
(
struct
pci_dev
*
pcid
,
u32
state
);
static
int
e100_suspend
(
struct
pci_dev
*
pcid
,
u32
state
);
static
int
e100_enable_wake
(
struct
pci_dev
*
pcid
,
u32
state
,
int
enable
);
static
int
e100_resume
(
struct
pci_dev
*
pcid
);
#endif
/*********************************************************************/
/*! This is a GCC extension to ANSI C.
* See the item "Labeled Elements in Initializers" in the section
...
...
@@ -203,6 +210,7 @@ struct net_device_stats *e100_get_stats(struct net_device *);
static
void
e100intr
(
int
,
void
*
,
struct
pt_regs
*
);
static
void
e100_print_brd_conf
(
struct
e100_private
*
);
static
void
e100_set_multi
(
struct
net_device
*
);
void
e100_set_speed_duplex
(
struct
e100_private
*
);
char
*
e100_get_brand_msg
(
struct
e100_private
*
);
static
u8
e100_pci_setup
(
struct
pci_dev
*
,
struct
e100_private
*
);
...
...
@@ -728,8 +736,15 @@ e100_remove1(struct pci_dev *pcid)
}
#ifdef ETHTOOL_GWOL
/* Set up wol options and enable PME */
e100_do_wol
(
pcid
,
bdp
);
/* Set up wol options and enable PME if wol is enabled */
if
(
bdp
->
wolopts
)
{
e100_do_wol
(
pcid
,
bdp
);
/* Enable PME for power state D3 */
pci_enable_wake
(
pcid
,
3
,
1
);
/* Set power state to D1 in case driver is RELOADED */
/* If system powers down, device is switched from D1 to D3 */
pci_set_power_state
(
pcid
,
1
);
}
#endif
e100_clear_structs
(
dev
);
...
...
@@ -744,8 +759,15 @@ static struct pci_driver e100_driver = {
id_table:
e100_id_table
,
probe:
e100_found1
,
remove:
__devexit_p
(
e100_remove1
),
#ifdef CONFIG_PM
suspend:
e100_suspend
,
resume:
e100_resume
,
save_state:
e100_save_state
,
enable_wake:
e100_enable_wake
,
#else
suspend:
NULL
,
resume:
NULL
,
#endif
};
static
int
__init
...
...
@@ -1194,7 +1216,7 @@ e100_set_multi(struct net_device *dev)
/* reconfigure the chip if something has changed in its config space */
e100_config
(
bdp
);
if
(
(
promisc_enbl
)
||
(
mulcast_enbl
)
)
{
if
(
promisc_enbl
||
mulcast_enbl
)
{
goto
exit
;
/* no need for Multicast Cmd */
}
...
...
@@ -1996,8 +2018,9 @@ e100_rx_srv(struct e100_private *bdp, u32 max_number_of_rfds,
if
(
max_number_of_rfds
&&
(
rfd_cnt
>=
max_number_of_rfds
))
{
break
;
}
if
(
list_empty
(
&
(
bdp
->
active_rx_list
)))
if
(
list_empty
(
&
(
bdp
->
active_rx_list
)))
{
break
;
}
rx_struct
=
list_entry
(
bdp
->
active_rx_list
.
next
,
struct
rx_list_elem
,
list_elem
);
...
...
@@ -2146,6 +2169,34 @@ e100_refresh_txthld(struct e100_private *bdp)
}
/* end underrun check */
}
#ifdef E100_ZEROCOPY
/**
* e100_pseudo_hdr_csum - compute IP pseudo-header checksum
* @ip: points to the header of the IP packet
*
* Return the 16 bit checksum of the IP pseudo-header.,which is computed
* on the fields: IP src, IP dst, next protocol, payload length.
* The checksum vaule is returned in network byte order.
*/
static
inline
u16
e100_pseudo_hdr_csum
(
const
struct
iphdr
*
ip
)
{
u32
pseudo
=
0
;
u32
payload_len
=
0
;
payload_len
=
ntohs
(
ip
->
tot_len
)
-
(
ip
->
ihl
*
4
);
pseudo
+=
htons
(
payload_len
);
pseudo
+=
(
ip
->
protocol
<<
8
);
pseudo
+=
ip
->
saddr
&
0x0000ffff
;
pseudo
+=
(
ip
->
saddr
&
0xffff0000
)
>>
16
;
pseudo
+=
ip
->
daddr
&
0x0000ffff
;
pseudo
+=
(
ip
->
daddr
&
0xffff0000
)
>>
16
;
return
FOLD_CSUM
(
pseudo
);
}
#endif
/* E100_ZEROCOPY */
/**
* e100_prepare_xmit_buff - prepare a buffer for transmission
* @bdp: atapter's private data struct
...
...
@@ -2204,10 +2255,9 @@ e100_prepare_xmit_buff(struct e100_private *bdp, struct sk_buff *skb)
chksum
=
&
(
udp
->
check
);
}
*
chksum
=
csum_tcpudp_magic
(
ip
->
daddr
,
ip
->
saddr
,
sizeof
(
struct
tcphdr
),
ip
->
protocol
,
0
);
*
chksum
=
e100_pseudo_hdr_csum
(
ip
);
}
}
else
{
if
(
bdp
->
flags
&
USE_IPCB
)
{
tcb
->
tcbu
.
ipcb
.
ip_activation_high
=
...
...
@@ -2319,9 +2369,8 @@ e100_start_cu(struct e100_private *bdp, tcb_t *tcb)
if
(
!
e100_wait_cus_idle
(
bdp
))
printk
(
"%s cu_start: timeout waiting for cu
\n
"
,
bdp
->
device
->
name
);
if
(
!
e100_wait_exec_cmplx
(
bdp
,
(
u32
)
(
tcb
->
tcb_phys
),
SCB_CUC_START
))
{
SCB_CUC_START
))
{
printk
(
"%s cu_start: timeout waiting for scb
\n
"
,
bdp
->
device
->
name
);
e100_exec_cmplx
(
bdp
,
(
u32
)
(
tcb
->
tcb_phys
),
...
...
@@ -2681,7 +2730,7 @@ e100_exec_non_cu_cmd(struct e100_private *bdp, nxmit_cb_entry_t *command)
wmb
();
if
(
in_interrupt
())
if
(
in_interrupt
()
||
netif_running
(
bdp
->
device
)
)
return
e100_delayed_exec_non_cu_cmd
(
bdp
,
command
);
spin_lock_bh
(
&
(
bdp
->
bd_non_tx_lock
));
...
...
@@ -2711,7 +2760,7 @@ e100_exec_non_cu_cmd(struct e100_private *bdp, nxmit_cb_entry_t *command)
bdp
->
next_cu_cmd
=
START_WAIT
;
spin_unlock_irqrestore
(
&
(
bdp
->
bd_lock
),
lock_flag
);
/* now wait for completion of non-cu CB up to 20 msec*/
/* now wait for completion of non-cu CB up to 20 msec
*/
expiration_time
=
jiffies
+
HZ
/
50
+
1
;
while
(
time_before
(
jiffies
,
expiration_time
))
{
rmb
();
...
...
@@ -3086,13 +3135,140 @@ e100_isolate_driver(struct e100_private *bdp)
del_timer_sync
(
&
bdp
->
watchdog_timer
);
netif_stop_queue
(
bdp
->
device
);
if
(
netif_running
(
bdp
->
device
))
netif_stop_queue
(
bdp
->
device
);
bdp
->
last_tcb
=
NULL
;
e100_sw_reset
(
bdp
,
PORT_SELECTIVE_RESET
);
}
void
e100_set_speed_duplex
(
struct
e100_private
*
bdp
)
{
e100_phy_set_speed_duplex
(
bdp
,
true
);
e100_config_fc
(
bdp
);
/* re-config flow-control if necessary */
e100_config
(
bdp
);
}
static
void
e100_tcb_add_C_bit
(
struct
e100_private
*
bdp
)
{
tcb_t
*
tcb
=
(
tcb_t
*
)
bdp
->
tcb_pool
.
data
;
int
i
;
for
(
i
=
0
;
i
<
bdp
->
params
.
TxDescriptors
;
i
++
,
tcb
++
)
{
tcb
->
tcb_hdr
.
cb_status
|=
cpu_to_le16
(
CB_STATUS_COMPLETE
);
}
}
/*
* Procedure: e100_hw_reset_recover
*
* Description: This routine will recover the hw after reset.
*
* Arguments:
* bdp - Ptr to this card's e100_bdconfig structure
* reset_cmd - s/w reset or selective reset.
*
* Returns:
* true upon success
* false upon failure
*/
unsigned
char
e100_hw_reset_recover
(
struct
e100_private
*
bdp
,
u32
reset_cmd
)
{
bdp
->
last_tcb
=
NULL
;
if
(
reset_cmd
==
PORT_SOFTWARE_RESET
)
{
/*load CU & RU base */
if
(
!
e100_wait_exec_cmplx
(
bdp
,
0
,
SCB_CUC_LOAD_BASE
))
{
return
false
;
}
if
(
e100_load_microcode
(
bdp
))
{
bdp
->
flags
|=
DF_UCODE_LOADED
;
}
if
(
!
e100_wait_exec_cmplx
(
bdp
,
0
,
SCB_RUC_LOAD_BASE
))
{
return
false
;
}
/* Issue the load dump counters address command */
if
(
!
e100_wait_exec_cmplx
(
bdp
,
bdp
->
stat_cnt_phys
,
SCB_CUC_DUMP_ADDR
))
{
return
false
;
}
if
(
!
e100_setup_iaaddr
(
bdp
,
bdp
->
device
->
dev_addr
))
{
printk
(
KERN_ERR
"e100_hw_reset_recover: setup iaaddr failed
\n
"
);
return
false
;
}
e100_set_multi_exec
(
bdp
->
device
);
/* Change for 82558 enhancement */
/* If 82558/9 and if the user has enabled flow control, set up * the
* Flow Control Reg. in the CSR */
if
((
bdp
->
flags
&
IS_BACHELOR
)
&&
(
bdp
->
params
.
b_params
&
PRM_FC
))
{
writeb
(
DFLT_FC_THLD
,
&
bdp
->
scb
->
scb_ext
.
d101_scb
.
scb_fc_thld
);
writeb
(
DFLT_FC_CMD
,
&
bdp
->
scb
->
scb_ext
.
d101_scb
.
scb_fc_xon_xoff
);
}
}
e100_force_config
(
bdp
);
return
true
;
}
void
e100_deisolate_driver
(
struct
e100_private
*
bdp
,
u8
recover
,
u8
full_init
)
{
if
(
full_init
)
{
e100_sw_reset
(
bdp
,
PORT_SOFTWARE_RESET
);
if
(
!
e100_hw_reset_recover
(
bdp
,
PORT_SOFTWARE_RESET
))
printk
(
KERN_ERR
"e100_deisolate_driver:"
" HW SOFTWARE reset recover failed
\n
"
);
}
if
(
recover
)
{
bdp
->
next_cu_cmd
=
START_WAIT
;
bdp
->
last_tcb
=
NULL
;
/* lets reset the chip */
if
(
!
full_init
)
{
e100_sw_reset
(
bdp
,
PORT_SELECTIVE_RESET
);
if
(
!
e100_hw_reset_recover
(
bdp
,
PORT_SELECTIVE_RESET
))
{
printk
(
KERN_ERR
"e100_deisolate_driver:"
" HW reset recover failed
\n
"
);
}
}
e100_start_ru
(
bdp
);
/* relaunch watchdog timer in 2 sec */
mod_timer
(
&
(
bdp
->
watchdog_timer
),
jiffies
+
(
2
*
HZ
));
// we must clear tcbs since we may have lost Tx intrrupt
// or have unsent frames on the tcb chain
e100_tcb_add_C_bit
(
bdp
);
e100_tx_srv
(
bdp
);
e100_set_intr_mask
(
bdp
);
if
(
netif_running
(
bdp
->
device
))
netif_wake_queue
(
bdp
->
device
);
}
bdp
->
driver_isolated
=
false
;
}
#ifdef E100_ETHTOOL_IOCTL
static
int
e100_do_ethtool_ioctl
(
struct
net_device
*
dev
,
struct
ifreq
*
ifr
)
...
...
@@ -3458,14 +3634,6 @@ e100_get_speed_duplex_caps(struct e100_private *bdp)
}
static
void
e100_set_speed_duplex
(
struct
e100_private
*
bdp
)
{
e100_phy_set_speed_duplex
(
bdp
,
true
);
e100_config_fc
(
bdp
);
/* re-config flow-control if necessary */
e100_config
(
bdp
);
}
#ifdef ETHTOOL_GWOL
static
unsigned
char
e100_setup_filter
(
struct
e100_private
*
bdp
)
...
...
@@ -3510,32 +3678,19 @@ e100_setup_filter(struct e100_private *bdp)
static
void
e100_do_wol
(
struct
pci_dev
*
pcid
,
struct
e100_private
*
bdp
)
{
int
enable
=
0
;
u32
state
=
0
;
if
(
bdp
->
wolopts
)
{
e100_config_wol
(
bdp
);
if
(
!
e100_config
(
bdp
))
{
printk
(
"e100_config WOL options failed
\n
"
);
goto
exit
;
}
e100_config_wol
(
bdp
);
if
(
bdp
->
wolopts
&
(
WAKE_UCAST
|
WAKE_ARP
))
{
if
(
!
e100_setup_filter
(
bdp
))
{
printk
(
"e100_config WOL options failed
\n
"
);
goto
exit
;
}
state
=
1
;
pci_set_power_state
(
pcid
,
state
);
}
enable
=
1
;
if
(
e100_config
(
bdp
))
{
if
(
bdp
->
wolopts
&
(
WAKE_UCAST
|
WAKE_ARP
))
if
(
!
e100_setup_filter
(
bdp
))
printk
(
KERN_ERR
"e100_config WOL options failed
\n
"
);
}
else
{
printk
(
KERN_ERR
"e100_config WOL failed
\n
"
);
}
exit:
pci_enable_wake
(
pcid
,
state
,
enable
);
}
static
u16
static
u16
e100_get_ip_lbytes
(
struct
net_device
*
dev
)
{
struct
in_ifaddr
*
ifa
;
...
...
@@ -3795,3 +3950,84 @@ e100_non_tx_background(unsigned long ptr)
}
spin_unlock_bh
(
&
(
bdp
->
bd_non_tx_lock
));
}
#ifdef CONFIG_PM
static
int
e100_save_state
(
struct
pci_dev
*
pcid
,
u32
state
)
{
struct
net_device
*
dev
;
struct
e100_private
*
bdp
;
/* Actually, PCI PM does NOT call this entry */
if
(
!
(
dev
=
(
struct
net_device
*
)
pci_get_drvdata
(
pcid
)))
return
-
1
;
bdp
=
dev
->
priv
;
pci_save_state
(
pcid
,
bdp
->
pci_state
);
return
0
;
}
static
int
e100_suspend
(
struct
pci_dev
*
pcid
,
u32
state
)
{
struct
net_device
*
netdev
=
pci_get_drvdata
(
pcid
);
struct
e100_private
*
bdp
=
netdev
->
priv
;
e100_isolate_driver
(
bdp
);
e100_save_state
(
pcid
,
state
);
/* If wol is enabled */
#ifdef ETHTOOL_GWOL
if
(
bdp
->
wolopts
)
{
bdp
->
ip_lbytes
=
e100_get_ip_lbytes
(
netdev
);
e100_do_wol
(
pcid
,
bdp
);
pci_enable_wake
(
pcid
,
3
,
1
);
/* Enable PME for power state D3 */
pci_set_power_state
(
pcid
,
3
);
/* Set power state to D3. */
}
else
{
/* Disable bus mastering */
pci_disable_device
(
pcid
);
pci_set_power_state
(
pcid
,
state
);
}
#else
pci_disable_device
(
pcid
);
pci_set_power_state
(
pcid
,
state
);
#endif
return
0
;
}
static
int
e100_resume
(
struct
pci_dev
*
pcid
)
{
struct
net_device
*
netdev
=
pci_get_drvdata
(
pcid
);
struct
e100_private
*
bdp
=
netdev
->
priv
;
u8
recover
=
false
;
u8
full_init
=
false
;
pci_set_power_state
(
pcid
,
0
);
pci_enable_wake
(
pcid
,
0
,
0
);
/* Clear PME status and disable PME */
pci_restore_state
(
pcid
,
bdp
->
pci_state
);
if
(
netif_running
(
netdev
))
{
recover
=
true
;
}
#ifdef ETHTOOL_GWOL
if
(
bdp
->
wolopts
&
(
WAKE_UCAST
|
WAKE_ARP
))
{
full_init
=
true
;
}
#endif
e100_deisolate_driver
(
bdp
,
recover
,
full_init
);
return
0
;
}
static
int
e100_enable_wake
(
struct
pci_dev
*
pcid
,
u32
state
,
int
enable
)
{
/* Driver doesn't need to do anything because it will enable */
/* wol when suspended. */
/* Actually, PCI PM does NOT call this entry. */
return
0
;
}
#endif
/* CONFIG_PM */
drivers/net/e100/e100_phy.c
View file @
f9c57480
...
...
@@ -769,7 +769,7 @@ e100_set_fc(struct e100_private *bdp)
* Arguments: bdp - Pointer to the e100_private structure for the board
*
* Returns: true if link state was changed
*
B_FLASE
otherwise
*
false
otherwise
*
*/
unsigned
char
...
...
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