Commit f9c57480 authored by Jeff Garzik's avatar Jeff Garzik

e100 net driver updates from Intel:

* fix zerocopy
* add suspend/resume
parent 50b1b006
...@@ -766,6 +766,8 @@ typedef enum _non_tx_cmd_state_t { ...@@ -766,6 +766,8 @@ typedef enum _non_tx_cmd_state_t {
#define IPCB_INSERTVLAN_ENABLE BIT_1 #define IPCB_INSERTVLAN_ENABLE BIT_1
#define IPCB_IP_ACTIVATION_DEFAULT IPCB_HARDWAREPARSING_ENABLE #define IPCB_IP_ACTIVATION_DEFAULT IPCB_HARDWAREPARSING_ENABLE
#define FOLD_CSUM(_XSUM) ((((_XSUM << 16) | (_XSUM >> 16)) + _XSUM) >> 16)
/* Transmit Buffer Descriptor (TBD)*/ /* Transmit Buffer Descriptor (TBD)*/
typedef struct _tbd_t { typedef struct _tbd_t {
u32 tbd_buf_addr; /* Physical Transmit Buffer Address */ u32 tbd_buf_addr; /* Physical Transmit Buffer Address */
...@@ -1008,6 +1010,11 @@ struct e100_private { ...@@ -1008,6 +1010,11 @@ struct e100_private {
u32 wolopts; u32 wolopts;
u16 ip_lbytes; u16 ip_lbytes;
#endif #endif
#ifdef CONFIG_PM
u32 pci_state[16];
#endif
}; };
#define E100_AUTONEG 0 #define E100_AUTONEG 0
...@@ -1030,4 +1037,9 @@ extern unsigned char e100_selftest(struct e100_private *bdp, u32 *st_timeout, ...@@ -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_get_link_state(struct e100_private *bdp);
extern unsigned char e100_wait_scb(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 #endif
...@@ -122,7 +122,6 @@ static int e100_do_ethtool_ioctl(struct net_device *, struct ifreq *); ...@@ -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 void e100_get_speed_duplex_caps(struct e100_private *);
static int e100_ethtool_get_settings(struct net_device *, struct ifreq *); static int e100_ethtool_get_settings(struct net_device *, struct ifreq *);
static int e100_ethtool_set_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 #ifdef ETHTOOL_GDRVINFO
static int e100_ethtool_get_drvinfo(struct net_device *, struct ifreq *); static int e100_ethtool_get_drvinfo(struct net_device *, struct ifreq *);
...@@ -163,7 +162,8 @@ static void e100_non_tx_background(unsigned long); ...@@ -163,7 +162,8 @@ static void e100_non_tx_background(unsigned long);
/* Global Data structures and variables */ /* Global Data structures and variables */
char e100_copyright[] __devinitdata = "Copyright (c) 2002 Intel Corporation"; 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 " #define E100_FULL_DRIVER_NAME "Intel(R) PRO/100 Fast Ethernet Adapter - Loadable driver, ver "
const char *e100_version = E100_VERSION; const char *e100_version = E100_VERSION;
...@@ -171,6 +171,13 @@ const char *e100_full_driver_name = E100_FULL_DRIVER_NAME E100_VERSION; ...@@ -171,6 +171,13 @@ const char *e100_full_driver_name = E100_FULL_DRIVER_NAME E100_VERSION;
char *e100_short_driver_name = "e100"; char *e100_short_driver_name = "e100";
static int e100nics = 0; 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. /*! This is a GCC extension to ANSI C.
* See the item "Labeled Elements in Initializers" in the section * See the item "Labeled Elements in Initializers" in the section
...@@ -203,6 +210,7 @@ struct net_device_stats *e100_get_stats(struct net_device *); ...@@ -203,6 +210,7 @@ struct net_device_stats *e100_get_stats(struct net_device *);
static void e100intr(int, void *, struct pt_regs *); static void e100intr(int, void *, struct pt_regs *);
static void e100_print_brd_conf(struct e100_private *); static void e100_print_brd_conf(struct e100_private *);
static void e100_set_multi(struct net_device *); static void e100_set_multi(struct net_device *);
void e100_set_speed_duplex(struct e100_private *);
char *e100_get_brand_msg(struct e100_private *); char *e100_get_brand_msg(struct e100_private *);
static u8 e100_pci_setup(struct pci_dev *, struct e100_private *); static u8 e100_pci_setup(struct pci_dev *, struct e100_private *);
...@@ -728,8 +736,15 @@ e100_remove1(struct pci_dev *pcid) ...@@ -728,8 +736,15 @@ e100_remove1(struct pci_dev *pcid)
} }
#ifdef ETHTOOL_GWOL #ifdef ETHTOOL_GWOL
/* Set up wol options and enable PME */ /* Set up wol options and enable PME if wol is enabled */
if (bdp->wolopts) {
e100_do_wol(pcid, bdp); 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 #endif
e100_clear_structs(dev); e100_clear_structs(dev);
...@@ -744,8 +759,15 @@ static struct pci_driver e100_driver = { ...@@ -744,8 +759,15 @@ static struct pci_driver e100_driver = {
id_table: e100_id_table, id_table: e100_id_table,
probe: e100_found1, probe: e100_found1,
remove: __devexit_p(e100_remove1), 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, suspend: NULL,
resume: NULL, resume: NULL,
#endif
}; };
static int __init static int __init
...@@ -1194,7 +1216,7 @@ e100_set_multi(struct net_device *dev) ...@@ -1194,7 +1216,7 @@ e100_set_multi(struct net_device *dev)
/* reconfigure the chip if something has changed in its config space */ /* reconfigure the chip if something has changed in its config space */
e100_config(bdp); e100_config(bdp);
if ((promisc_enbl) || (mulcast_enbl)) { if (promisc_enbl || mulcast_enbl) {
goto exit; /* no need for Multicast Cmd */ goto exit; /* no need for Multicast Cmd */
} }
...@@ -1996,8 +2018,9 @@ e100_rx_srv(struct e100_private *bdp, u32 max_number_of_rfds, ...@@ -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)) { if (max_number_of_rfds && (rfd_cnt >= max_number_of_rfds)) {
break; break;
} }
if (list_empty(&(bdp->active_rx_list))) if (list_empty(&(bdp->active_rx_list))) {
break; break;
}
rx_struct = list_entry(bdp->active_rx_list.next, rx_struct = list_entry(bdp->active_rx_list.next,
struct rx_list_elem, list_elem); struct rx_list_elem, list_elem);
...@@ -2146,6 +2169,34 @@ e100_refresh_txthld(struct e100_private *bdp) ...@@ -2146,6 +2169,34 @@ e100_refresh_txthld(struct e100_private *bdp)
} /* end underrun check */ } /* 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 * e100_prepare_xmit_buff - prepare a buffer for transmission
* @bdp: atapter's private data struct * @bdp: atapter's private data struct
...@@ -2204,10 +2255,9 @@ e100_prepare_xmit_buff(struct e100_private *bdp, struct sk_buff *skb) ...@@ -2204,10 +2255,9 @@ e100_prepare_xmit_buff(struct e100_private *bdp, struct sk_buff *skb)
chksum = &(udp->check); chksum = &(udp->check);
} }
*chksum = csum_tcpudp_magic(ip->daddr, ip->saddr, *chksum = e100_pseudo_hdr_csum(ip);
sizeof (struct tcphdr),
ip->protocol, 0);
} }
} else { } else {
if (bdp->flags & USE_IPCB) { if (bdp->flags & USE_IPCB) {
tcb->tcbu.ipcb.ip_activation_high = tcb->tcbu.ipcb.ip_activation_high =
...@@ -2319,7 +2369,6 @@ e100_start_cu(struct e100_private *bdp, tcb_t *tcb) ...@@ -2319,7 +2369,6 @@ e100_start_cu(struct e100_private *bdp, tcb_t *tcb)
if (!e100_wait_cus_idle(bdp)) if (!e100_wait_cus_idle(bdp))
printk("%s cu_start: timeout waiting for cu\n", printk("%s cu_start: timeout waiting for cu\n",
bdp->device->name); bdp->device->name);
if (!e100_wait_exec_cmplx(bdp, (u32) (tcb->tcb_phys), 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", printk("%s cu_start: timeout waiting for scb\n",
...@@ -2681,7 +2730,7 @@ e100_exec_non_cu_cmd(struct e100_private *bdp, nxmit_cb_entry_t *command) ...@@ -2681,7 +2730,7 @@ e100_exec_non_cu_cmd(struct e100_private *bdp, nxmit_cb_entry_t *command)
wmb(); wmb();
if (in_interrupt()) if (in_interrupt() || netif_running(bdp->device))
return e100_delayed_exec_non_cu_cmd(bdp, command); return e100_delayed_exec_non_cu_cmd(bdp, command);
spin_lock_bh(&(bdp->bd_non_tx_lock)); 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) ...@@ -2711,7 +2760,7 @@ e100_exec_non_cu_cmd(struct e100_private *bdp, nxmit_cb_entry_t *command)
bdp->next_cu_cmd = START_WAIT; bdp->next_cu_cmd = START_WAIT;
spin_unlock_irqrestore(&(bdp->bd_lock), lock_flag); 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; expiration_time = jiffies + HZ / 50 + 1;
while (time_before(jiffies, expiration_time)) { while (time_before(jiffies, expiration_time)) {
rmb(); rmb();
...@@ -3086,6 +3135,7 @@ e100_isolate_driver(struct e100_private *bdp) ...@@ -3086,6 +3135,7 @@ e100_isolate_driver(struct e100_private *bdp)
del_timer_sync(&bdp->watchdog_timer); del_timer_sync(&bdp->watchdog_timer);
if (netif_running(bdp->device))
netif_stop_queue(bdp->device); netif_stop_queue(bdp->device);
bdp->last_tcb = NULL; bdp->last_tcb = NULL;
...@@ -3093,6 +3143,132 @@ e100_isolate_driver(struct e100_private *bdp) ...@@ -3093,6 +3143,132 @@ e100_isolate_driver(struct e100_private *bdp)
e100_sw_reset(bdp, PORT_SELECTIVE_RESET); 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 #ifdef E100_ETHTOOL_IOCTL
static int static int
e100_do_ethtool_ioctl(struct net_device *dev, struct ifreq *ifr) e100_do_ethtool_ioctl(struct net_device *dev, struct ifreq *ifr)
...@@ -3458,14 +3634,6 @@ e100_get_speed_duplex_caps(struct e100_private *bdp) ...@@ -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 #ifdef ETHTOOL_GWOL
static unsigned char static unsigned char
e100_setup_filter(struct e100_private *bdp) e100_setup_filter(struct e100_private *bdp)
...@@ -3510,29 +3678,16 @@ e100_setup_filter(struct e100_private *bdp) ...@@ -3510,29 +3678,16 @@ e100_setup_filter(struct e100_private *bdp)
static void static void
e100_do_wol(struct pci_dev *pcid, struct e100_private *bdp) e100_do_wol(struct pci_dev *pcid, struct e100_private *bdp)
{ {
int enable = 0;
u32 state = 0;
if (bdp->wolopts) {
e100_config_wol(bdp); e100_config_wol(bdp);
if (!e100_config(bdp)) { if (e100_config(bdp)) {
printk("e100_config WOL options failed\n"); if (bdp->wolopts & (WAKE_UCAST | WAKE_ARP))
goto exit; if (!e100_setup_filter(bdp))
} printk(KERN_ERR
"e100_config WOL options failed\n");
if (bdp->wolopts & (WAKE_UCAST | WAKE_ARP)) { } else {
if (!e100_setup_filter(bdp)) { printk(KERN_ERR "e100_config WOL failed\n");
printk("e100_config WOL options failed\n");
goto exit;
}
state = 1;
pci_set_power_state(pcid, state);
}
enable = 1;
} }
exit:
pci_enable_wake(pcid, state, enable);
} }
static u16 static u16
...@@ -3795,3 +3950,84 @@ e100_non_tx_background(unsigned long ptr) ...@@ -3795,3 +3950,84 @@ e100_non_tx_background(unsigned long ptr)
} }
spin_unlock_bh(&(bdp->bd_non_tx_lock)); 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 */
...@@ -769,7 +769,7 @@ e100_set_fc(struct e100_private *bdp) ...@@ -769,7 +769,7 @@ e100_set_fc(struct e100_private *bdp)
* Arguments: bdp - Pointer to the e100_private structure for the board * Arguments: bdp - Pointer to the e100_private structure for the board
* *
* Returns: true if link state was changed * Returns: true if link state was changed
* B_FLASE otherwise * false otherwise
* *
*/ */
unsigned char unsigned char
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment