devices/e1000/e1000_ethtool-3.0-ethercat.c
author Knud Baastrup <kba@deif.com>
Tue, 14 Apr 2015 10:12:55 -0400
changeset 2625 e25af8bd3957
parent 2395 f7451c2c274f
permissions -rw-r--r--
Eoe mac address now derived from unique mac.
The EoE MAC address is now derived from the NIC part of the first global
unique MAC address of the linked list of available network interfaces or
otherwise the MAC address used by the EtherCAT master. The EoE MAC address
will get the format 02:NIC:NIC:NIC:RP:RP where NIC comes from the unique MAC
address (if available) and RP is the ring position of the EoE slave.
2395
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
     1
/*******************************************************************************
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
     2
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
     3
  Intel PRO/1000 Linux driver
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
     4
  Copyright(c) 1999 - 2006 Intel Corporation.
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
     5
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
     6
  This program is free software; you can redistribute it and/or modify it
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
     7
  under the terms and conditions of the GNU General Public License,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
     8
  version 2, as published by the Free Software Foundation.
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
     9
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    10
  This program is distributed in the hope it will be useful, but WITHOUT
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    11
  ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    12
  FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    13
  more details.
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    14
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    15
  You should have received a copy of the GNU General Public License along with
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    16
  this program; if not, write to the Free Software Foundation, Inc.,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    17
  51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA.
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    18
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    19
  The full GNU General Public License is included in this distribution in
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    20
  the file called "COPYING".
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    21
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    22
  Contact Information:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    23
  Linux NICS <linux.nics@intel.com>
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    24
  e1000-devel Mailing List <e1000-devel@lists.sourceforge.net>
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    25
  Intel Corporation, 5200 N.E. Elam Young Parkway, Hillsboro, OR 97124-6497
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    26
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    27
*******************************************************************************/
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    28
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    29
/* ethtool support for e1000 */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    30
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    31
#include "e1000-3.0-ethercat.h"
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    32
#include <asm/uaccess.h>
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    33
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    34
enum {NETDEV_STATS, E1000_STATS};
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    35
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    36
struct e1000_stats {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    37
	char stat_string[ETH_GSTRING_LEN];
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    38
	int type;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    39
	int sizeof_stat;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    40
	int stat_offset;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    41
};
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    42
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    43
#define E1000_STAT(m)		E1000_STATS, \
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    44
				sizeof(((struct e1000_adapter *)0)->m), \
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    45
		      		offsetof(struct e1000_adapter, m)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    46
#define E1000_NETDEV_STAT(m)	NETDEV_STATS, \
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    47
				sizeof(((struct net_device *)0)->m), \
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    48
				offsetof(struct net_device, m)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    49
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    50
static const struct e1000_stats e1000_gstrings_stats[] = {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    51
	{ "rx_packets", E1000_STAT(stats.gprc) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    52
	{ "tx_packets", E1000_STAT(stats.gptc) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    53
	{ "rx_bytes", E1000_STAT(stats.gorcl) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    54
	{ "tx_bytes", E1000_STAT(stats.gotcl) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    55
	{ "rx_broadcast", E1000_STAT(stats.bprc) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    56
	{ "tx_broadcast", E1000_STAT(stats.bptc) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    57
	{ "rx_multicast", E1000_STAT(stats.mprc) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    58
	{ "tx_multicast", E1000_STAT(stats.mptc) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    59
	{ "rx_errors", E1000_STAT(stats.rxerrc) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    60
	{ "tx_errors", E1000_STAT(stats.txerrc) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    61
	{ "tx_dropped", E1000_NETDEV_STAT(stats.tx_dropped) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    62
	{ "multicast", E1000_STAT(stats.mprc) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    63
	{ "collisions", E1000_STAT(stats.colc) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    64
	{ "rx_length_errors", E1000_STAT(stats.rlerrc) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    65
	{ "rx_over_errors", E1000_NETDEV_STAT(stats.rx_over_errors) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    66
	{ "rx_crc_errors", E1000_STAT(stats.crcerrs) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    67
	{ "rx_frame_errors", E1000_NETDEV_STAT(stats.rx_frame_errors) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    68
	{ "rx_no_buffer_count", E1000_STAT(stats.rnbc) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    69
	{ "rx_missed_errors", E1000_STAT(stats.mpc) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    70
	{ "tx_aborted_errors", E1000_STAT(stats.ecol) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    71
	{ "tx_carrier_errors", E1000_STAT(stats.tncrs) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    72
	{ "tx_fifo_errors", E1000_NETDEV_STAT(stats.tx_fifo_errors) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    73
	{ "tx_heartbeat_errors", E1000_NETDEV_STAT(stats.tx_heartbeat_errors) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    74
	{ "tx_window_errors", E1000_STAT(stats.latecol) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    75
	{ "tx_abort_late_coll", E1000_STAT(stats.latecol) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    76
	{ "tx_deferred_ok", E1000_STAT(stats.dc) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    77
	{ "tx_single_coll_ok", E1000_STAT(stats.scc) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    78
	{ "tx_multi_coll_ok", E1000_STAT(stats.mcc) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    79
	{ "tx_timeout_count", E1000_STAT(tx_timeout_count) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    80
	{ "tx_restart_queue", E1000_STAT(restart_queue) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    81
	{ "rx_long_length_errors", E1000_STAT(stats.roc) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    82
	{ "rx_short_length_errors", E1000_STAT(stats.ruc) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    83
	{ "rx_align_errors", E1000_STAT(stats.algnerrc) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    84
	{ "tx_tcp_seg_good", E1000_STAT(stats.tsctc) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    85
	{ "tx_tcp_seg_failed", E1000_STAT(stats.tsctfc) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    86
	{ "rx_flow_control_xon", E1000_STAT(stats.xonrxc) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    87
	{ "rx_flow_control_xoff", E1000_STAT(stats.xoffrxc) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    88
	{ "tx_flow_control_xon", E1000_STAT(stats.xontxc) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    89
	{ "tx_flow_control_xoff", E1000_STAT(stats.xofftxc) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    90
	{ "rx_long_byte_count", E1000_STAT(stats.gorcl) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    91
	{ "rx_csum_offload_good", E1000_STAT(hw_csum_good) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    92
	{ "rx_csum_offload_errors", E1000_STAT(hw_csum_err) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    93
	{ "alloc_rx_buff_failed", E1000_STAT(alloc_rx_buff_failed) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    94
	{ "tx_smbus", E1000_STAT(stats.mgptc) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    95
	{ "rx_smbus", E1000_STAT(stats.mgprc) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    96
	{ "dropped_smbus", E1000_STAT(stats.mgpdc) },
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    97
};
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    98
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
    99
#define E1000_QUEUE_STATS_LEN 0
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   100
#define E1000_GLOBAL_STATS_LEN ARRAY_SIZE(e1000_gstrings_stats)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   101
#define E1000_STATS_LEN (E1000_GLOBAL_STATS_LEN + E1000_QUEUE_STATS_LEN)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   102
static const char e1000_gstrings_test[][ETH_GSTRING_LEN] = {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   103
	"Register test  (offline)", "Eeprom test    (offline)",
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   104
	"Interrupt test (offline)", "Loopback test  (offline)",
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   105
	"Link test   (on/offline)"
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   106
};
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   107
#define E1000_TEST_LEN	ARRAY_SIZE(e1000_gstrings_test)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   108
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   109
static int e1000_get_settings(struct net_device *netdev,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   110
			      struct ethtool_cmd *ecmd)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   111
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   112
	struct e1000_adapter *adapter = netdev_priv(netdev);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   113
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   114
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   115
	if (hw->media_type == e1000_media_type_copper) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   116
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   117
		ecmd->supported = (SUPPORTED_10baseT_Half |
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   118
		                   SUPPORTED_10baseT_Full |
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   119
		                   SUPPORTED_100baseT_Half |
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   120
		                   SUPPORTED_100baseT_Full |
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   121
		                   SUPPORTED_1000baseT_Full|
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   122
		                   SUPPORTED_Autoneg |
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   123
		                   SUPPORTED_TP);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   124
		ecmd->advertising = ADVERTISED_TP;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   125
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   126
		if (hw->autoneg == 1) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   127
			ecmd->advertising |= ADVERTISED_Autoneg;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   128
			/* the e1000 autoneg seems to match ethtool nicely */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   129
			ecmd->advertising |= hw->autoneg_advertised;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   130
		}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   131
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   132
		ecmd->port = PORT_TP;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   133
		ecmd->phy_address = hw->phy_addr;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   134
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   135
		if (hw->mac_type == e1000_82543)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   136
			ecmd->transceiver = XCVR_EXTERNAL;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   137
		else
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   138
			ecmd->transceiver = XCVR_INTERNAL;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   139
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   140
	} else {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   141
		ecmd->supported   = (SUPPORTED_1000baseT_Full |
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   142
				     SUPPORTED_FIBRE |
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   143
				     SUPPORTED_Autoneg);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   144
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   145
		ecmd->advertising = (ADVERTISED_1000baseT_Full |
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   146
				     ADVERTISED_FIBRE |
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   147
				     ADVERTISED_Autoneg);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   148
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   149
		ecmd->port = PORT_FIBRE;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   150
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   151
		if (hw->mac_type >= e1000_82545)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   152
			ecmd->transceiver = XCVR_INTERNAL;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   153
		else
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   154
			ecmd->transceiver = XCVR_EXTERNAL;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   155
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   156
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   157
	if (er32(STATUS) & E1000_STATUS_LU) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   158
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   159
		e1000_get_speed_and_duplex(hw, &adapter->link_speed,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   160
		                                   &adapter->link_duplex);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   161
		ethtool_cmd_speed_set(ecmd, adapter->link_speed);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   162
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   163
		/* unfortunately FULL_DUPLEX != DUPLEX_FULL
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   164
		 *          and HALF_DUPLEX != DUPLEX_HALF */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   165
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   166
		if (adapter->link_duplex == FULL_DUPLEX)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   167
			ecmd->duplex = DUPLEX_FULL;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   168
		else
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   169
			ecmd->duplex = DUPLEX_HALF;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   170
	} else {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   171
		ethtool_cmd_speed_set(ecmd, -1);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   172
		ecmd->duplex = -1;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   173
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   174
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   175
	ecmd->autoneg = ((hw->media_type == e1000_media_type_fiber) ||
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   176
			 hw->autoneg) ? AUTONEG_ENABLE : AUTONEG_DISABLE;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   177
	return 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   178
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   179
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   180
static int e1000_set_settings(struct net_device *netdev,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   181
			      struct ethtool_cmd *ecmd)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   182
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   183
	struct e1000_adapter *adapter = netdev_priv(netdev);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   184
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   185
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   186
	if (adapter->ecdev)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   187
		return -EBUSY;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   188
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   189
	while (test_and_set_bit(__E1000_RESETTING, &adapter->flags))
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   190
		msleep(1);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   191
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   192
	if (ecmd->autoneg == AUTONEG_ENABLE) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   193
		hw->autoneg = 1;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   194
		if (hw->media_type == e1000_media_type_fiber)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   195
			hw->autoneg_advertised = ADVERTISED_1000baseT_Full |
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   196
				     ADVERTISED_FIBRE |
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   197
				     ADVERTISED_Autoneg;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   198
		else
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   199
			hw->autoneg_advertised = ecmd->advertising |
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   200
			                         ADVERTISED_TP |
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   201
			                         ADVERTISED_Autoneg;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   202
		ecmd->advertising = hw->autoneg_advertised;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   203
	} else {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   204
		u32 speed = ethtool_cmd_speed(ecmd);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   205
		if (e1000_set_spd_dplx(adapter, speed, ecmd->duplex)) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   206
			clear_bit(__E1000_RESETTING, &adapter->flags);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   207
			return -EINVAL;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   208
		}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   209
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   210
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   211
	/* reset the link */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   212
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   213
	if (netif_running(adapter->netdev)) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   214
		e1000_down(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   215
		e1000_up(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   216
	} else
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   217
		e1000_reset(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   218
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   219
	clear_bit(__E1000_RESETTING, &adapter->flags);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   220
	return 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   221
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   222
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   223
static u32 e1000_get_link(struct net_device *netdev)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   224
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   225
	struct e1000_adapter *adapter = netdev_priv(netdev);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   226
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   227
	/*
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   228
	 * If the link is not reported up to netdev, interrupts are disabled,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   229
	 * and so the physical link state may have changed since we last
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   230
	 * looked. Set get_link_status to make sure that the true link
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   231
	 * state is interrogated, rather than pulling a cached and possibly
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   232
	 * stale link state from the driver.
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   233
	 */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   234
	if (!netif_carrier_ok(netdev))
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   235
		adapter->hw.get_link_status = 1;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   236
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   237
	return e1000_has_link(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   238
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   239
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   240
static void e1000_get_pauseparam(struct net_device *netdev,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   241
				 struct ethtool_pauseparam *pause)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   242
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   243
	struct e1000_adapter *adapter = netdev_priv(netdev);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   244
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   245
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   246
	pause->autoneg =
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   247
		(adapter->fc_autoneg ? AUTONEG_ENABLE : AUTONEG_DISABLE);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   248
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   249
	if (hw->fc == E1000_FC_RX_PAUSE)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   250
		pause->rx_pause = 1;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   251
	else if (hw->fc == E1000_FC_TX_PAUSE)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   252
		pause->tx_pause = 1;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   253
	else if (hw->fc == E1000_FC_FULL) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   254
		pause->rx_pause = 1;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   255
		pause->tx_pause = 1;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   256
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   257
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   258
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   259
static int e1000_set_pauseparam(struct net_device *netdev,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   260
				struct ethtool_pauseparam *pause)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   261
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   262
	struct e1000_adapter *adapter = netdev_priv(netdev);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   263
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   264
	int retval = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   265
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   266
	if (adapter->ecdev)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   267
		return -EBUSY;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   268
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   269
	adapter->fc_autoneg = pause->autoneg;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   270
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   271
	while (test_and_set_bit(__E1000_RESETTING, &adapter->flags))
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   272
		msleep(1);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   273
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   274
	if (pause->rx_pause && pause->tx_pause)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   275
		hw->fc = E1000_FC_FULL;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   276
	else if (pause->rx_pause && !pause->tx_pause)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   277
		hw->fc = E1000_FC_RX_PAUSE;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   278
	else if (!pause->rx_pause && pause->tx_pause)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   279
		hw->fc = E1000_FC_TX_PAUSE;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   280
	else if (!pause->rx_pause && !pause->tx_pause)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   281
		hw->fc = E1000_FC_NONE;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   282
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   283
	hw->original_fc = hw->fc;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   284
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   285
	if (adapter->fc_autoneg == AUTONEG_ENABLE) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   286
		if (netif_running(adapter->netdev)) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   287
			e1000_down(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   288
			e1000_up(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   289
		} else
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   290
			e1000_reset(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   291
	} else
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   292
		retval = ((hw->media_type == e1000_media_type_fiber) ?
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   293
			  e1000_setup_link(hw) : e1000_force_mac_fc(hw));
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   294
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   295
	clear_bit(__E1000_RESETTING, &adapter->flags);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   296
	return retval;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   297
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   298
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   299
static u32 e1000_get_rx_csum(struct net_device *netdev)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   300
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   301
	struct e1000_adapter *adapter = netdev_priv(netdev);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   302
	return adapter->rx_csum;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   303
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   304
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   305
static int e1000_set_rx_csum(struct net_device *netdev, u32 data)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   306
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   307
	struct e1000_adapter *adapter = netdev_priv(netdev);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   308
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   309
	if (adapter->ecdev)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   310
		return -EBUSY;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   311
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   312
	adapter->rx_csum = data;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   313
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   314
	if (netif_running(netdev))
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   315
		e1000_reinit_locked(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   316
	else
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   317
		e1000_reset(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   318
	return 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   319
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   320
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   321
static u32 e1000_get_tx_csum(struct net_device *netdev)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   322
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   323
	return (netdev->features & NETIF_F_HW_CSUM) != 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   324
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   325
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   326
static int e1000_set_tx_csum(struct net_device *netdev, u32 data)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   327
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   328
	struct e1000_adapter *adapter = netdev_priv(netdev);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   329
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   330
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   331
	if (hw->mac_type < e1000_82543) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   332
		if (!data)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   333
			return -EINVAL;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   334
		return 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   335
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   336
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   337
	if (data)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   338
		netdev->features |= NETIF_F_HW_CSUM;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   339
	else
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   340
		netdev->features &= ~NETIF_F_HW_CSUM;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   341
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   342
	return 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   343
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   344
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   345
static int e1000_set_tso(struct net_device *netdev, u32 data)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   346
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   347
	struct e1000_adapter *adapter = netdev_priv(netdev);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   348
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   349
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   350
	if ((hw->mac_type < e1000_82544) ||
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   351
	    (hw->mac_type == e1000_82547))
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   352
		return data ? -EINVAL : 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   353
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   354
	if (data)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   355
		netdev->features |= NETIF_F_TSO;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   356
	else
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   357
		netdev->features &= ~NETIF_F_TSO;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   358
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   359
	netdev->features &= ~NETIF_F_TSO6;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   360
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   361
	e_info(probe, "TSO is %s\n", data ? "Enabled" : "Disabled");
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   362
	adapter->tso_force = true;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   363
	return 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   364
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   365
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   366
static u32 e1000_get_msglevel(struct net_device *netdev)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   367
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   368
	struct e1000_adapter *adapter = netdev_priv(netdev);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   369
	return adapter->msg_enable;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   370
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   371
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   372
static void e1000_set_msglevel(struct net_device *netdev, u32 data)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   373
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   374
	struct e1000_adapter *adapter = netdev_priv(netdev);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   375
	adapter->msg_enable = data;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   376
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   377
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   378
static int e1000_get_regs_len(struct net_device *netdev)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   379
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   380
#define E1000_REGS_LEN 32
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   381
	return E1000_REGS_LEN * sizeof(u32);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   382
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   383
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   384
static void e1000_get_regs(struct net_device *netdev, struct ethtool_regs *regs,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   385
			   void *p)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   386
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   387
	struct e1000_adapter *adapter = netdev_priv(netdev);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   388
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   389
	u32 *regs_buff = p;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   390
	u16 phy_data;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   391
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   392
	memset(p, 0, E1000_REGS_LEN * sizeof(u32));
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   393
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   394
	regs->version = (1 << 24) | (hw->revision_id << 16) | hw->device_id;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   395
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   396
	regs_buff[0]  = er32(CTRL);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   397
	regs_buff[1]  = er32(STATUS);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   398
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   399
	regs_buff[2]  = er32(RCTL);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   400
	regs_buff[3]  = er32(RDLEN);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   401
	regs_buff[4]  = er32(RDH);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   402
	regs_buff[5]  = er32(RDT);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   403
	regs_buff[6]  = er32(RDTR);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   404
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   405
	regs_buff[7]  = er32(TCTL);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   406
	regs_buff[8]  = er32(TDLEN);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   407
	regs_buff[9]  = er32(TDH);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   408
	regs_buff[10] = er32(TDT);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   409
	regs_buff[11] = er32(TIDV);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   410
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   411
	regs_buff[12] = hw->phy_type;  /* PHY type (IGP=1, M88=0) */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   412
	if (hw->phy_type == e1000_phy_igp) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   413
		e1000_write_phy_reg(hw, IGP01E1000_PHY_PAGE_SELECT,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   414
				    IGP01E1000_PHY_AGC_A);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   415
		e1000_read_phy_reg(hw, IGP01E1000_PHY_AGC_A &
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   416
				   IGP01E1000_PHY_PAGE_SELECT, &phy_data);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   417
		regs_buff[13] = (u32)phy_data; /* cable length */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   418
		e1000_write_phy_reg(hw, IGP01E1000_PHY_PAGE_SELECT,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   419
				    IGP01E1000_PHY_AGC_B);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   420
		e1000_read_phy_reg(hw, IGP01E1000_PHY_AGC_B &
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   421
				   IGP01E1000_PHY_PAGE_SELECT, &phy_data);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   422
		regs_buff[14] = (u32)phy_data; /* cable length */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   423
		e1000_write_phy_reg(hw, IGP01E1000_PHY_PAGE_SELECT,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   424
				    IGP01E1000_PHY_AGC_C);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   425
		e1000_read_phy_reg(hw, IGP01E1000_PHY_AGC_C &
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   426
				   IGP01E1000_PHY_PAGE_SELECT, &phy_data);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   427
		regs_buff[15] = (u32)phy_data; /* cable length */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   428
		e1000_write_phy_reg(hw, IGP01E1000_PHY_PAGE_SELECT,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   429
				    IGP01E1000_PHY_AGC_D);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   430
		e1000_read_phy_reg(hw, IGP01E1000_PHY_AGC_D &
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   431
				   IGP01E1000_PHY_PAGE_SELECT, &phy_data);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   432
		regs_buff[16] = (u32)phy_data; /* cable length */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   433
		regs_buff[17] = 0; /* extended 10bt distance (not needed) */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   434
		e1000_write_phy_reg(hw, IGP01E1000_PHY_PAGE_SELECT, 0x0);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   435
		e1000_read_phy_reg(hw, IGP01E1000_PHY_PORT_STATUS &
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   436
				   IGP01E1000_PHY_PAGE_SELECT, &phy_data);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   437
		regs_buff[18] = (u32)phy_data; /* cable polarity */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   438
		e1000_write_phy_reg(hw, IGP01E1000_PHY_PAGE_SELECT,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   439
				    IGP01E1000_PHY_PCS_INIT_REG);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   440
		e1000_read_phy_reg(hw, IGP01E1000_PHY_PCS_INIT_REG &
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   441
				   IGP01E1000_PHY_PAGE_SELECT, &phy_data);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   442
		regs_buff[19] = (u32)phy_data; /* cable polarity */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   443
		regs_buff[20] = 0; /* polarity correction enabled (always) */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   444
		regs_buff[22] = 0; /* phy receive errors (unavailable) */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   445
		regs_buff[23] = regs_buff[18]; /* mdix mode */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   446
		e1000_write_phy_reg(hw, IGP01E1000_PHY_PAGE_SELECT, 0x0);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   447
	} else {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   448
		e1000_read_phy_reg(hw, M88E1000_PHY_SPEC_STATUS, &phy_data);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   449
		regs_buff[13] = (u32)phy_data; /* cable length */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   450
		regs_buff[14] = 0;  /* Dummy (to align w/ IGP phy reg dump) */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   451
		regs_buff[15] = 0;  /* Dummy (to align w/ IGP phy reg dump) */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   452
		regs_buff[16] = 0;  /* Dummy (to align w/ IGP phy reg dump) */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   453
		e1000_read_phy_reg(hw, M88E1000_PHY_SPEC_CTRL, &phy_data);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   454
		regs_buff[17] = (u32)phy_data; /* extended 10bt distance */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   455
		regs_buff[18] = regs_buff[13]; /* cable polarity */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   456
		regs_buff[19] = 0;  /* Dummy (to align w/ IGP phy reg dump) */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   457
		regs_buff[20] = regs_buff[17]; /* polarity correction */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   458
		/* phy receive errors */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   459
		regs_buff[22] = adapter->phy_stats.receive_errors;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   460
		regs_buff[23] = regs_buff[13]; /* mdix mode */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   461
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   462
	regs_buff[21] = adapter->phy_stats.idle_errors;  /* phy idle errors */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   463
	e1000_read_phy_reg(hw, PHY_1000T_STATUS, &phy_data);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   464
	regs_buff[24] = (u32)phy_data;  /* phy local receiver status */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   465
	regs_buff[25] = regs_buff[24];  /* phy remote receiver status */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   466
	if (hw->mac_type >= e1000_82540 &&
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   467
	    hw->media_type == e1000_media_type_copper) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   468
		regs_buff[26] = er32(MANC);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   469
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   470
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   471
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   472
static int e1000_get_eeprom_len(struct net_device *netdev)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   473
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   474
	struct e1000_adapter *adapter = netdev_priv(netdev);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   475
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   476
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   477
	return hw->eeprom.word_size * 2;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   478
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   479
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   480
static int e1000_get_eeprom(struct net_device *netdev,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   481
			    struct ethtool_eeprom *eeprom, u8 *bytes)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   482
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   483
	struct e1000_adapter *adapter = netdev_priv(netdev);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   484
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   485
	u16 *eeprom_buff;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   486
	int first_word, last_word;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   487
	int ret_val = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   488
	u16 i;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   489
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   490
	if (eeprom->len == 0)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   491
		return -EINVAL;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   492
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   493
	eeprom->magic = hw->vendor_id | (hw->device_id << 16);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   494
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   495
	first_word = eeprom->offset >> 1;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   496
	last_word = (eeprom->offset + eeprom->len - 1) >> 1;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   497
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   498
	eeprom_buff = kmalloc(sizeof(u16) *
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   499
			(last_word - first_word + 1), GFP_KERNEL);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   500
	if (!eeprom_buff)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   501
		return -ENOMEM;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   502
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   503
	if (hw->eeprom.type == e1000_eeprom_spi)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   504
		ret_val = e1000_read_eeprom(hw, first_word,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   505
					    last_word - first_word + 1,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   506
					    eeprom_buff);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   507
	else {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   508
		for (i = 0; i < last_word - first_word + 1; i++) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   509
			ret_val = e1000_read_eeprom(hw, first_word + i, 1,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   510
						    &eeprom_buff[i]);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   511
			if (ret_val)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   512
				break;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   513
		}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   514
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   515
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   516
	/* Device's eeprom is always little-endian, word addressable */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   517
	for (i = 0; i < last_word - first_word + 1; i++)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   518
		le16_to_cpus(&eeprom_buff[i]);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   519
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   520
	memcpy(bytes, (u8 *)eeprom_buff + (eeprom->offset & 1),
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   521
			eeprom->len);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   522
	kfree(eeprom_buff);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   523
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   524
	return ret_val;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   525
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   526
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   527
static int e1000_set_eeprom(struct net_device *netdev,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   528
			    struct ethtool_eeprom *eeprom, u8 *bytes)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   529
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   530
	struct e1000_adapter *adapter = netdev_priv(netdev);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   531
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   532
	u16 *eeprom_buff;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   533
	void *ptr;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   534
	int max_len, first_word, last_word, ret_val = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   535
	u16 i;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   536
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   537
	if (eeprom->len == 0)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   538
		return -EOPNOTSUPP;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   539
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   540
	if (eeprom->magic != (hw->vendor_id | (hw->device_id << 16)))
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   541
		return -EFAULT;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   542
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   543
	max_len = hw->eeprom.word_size * 2;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   544
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   545
	first_word = eeprom->offset >> 1;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   546
	last_word = (eeprom->offset + eeprom->len - 1) >> 1;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   547
	eeprom_buff = kmalloc(max_len, GFP_KERNEL);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   548
	if (!eeprom_buff)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   549
		return -ENOMEM;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   550
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   551
	ptr = (void *)eeprom_buff;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   552
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   553
	if (eeprom->offset & 1) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   554
		/* need read/modify/write of first changed EEPROM word */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   555
		/* only the second byte of the word is being modified */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   556
		ret_val = e1000_read_eeprom(hw, first_word, 1,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   557
					    &eeprom_buff[0]);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   558
		ptr++;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   559
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   560
	if (((eeprom->offset + eeprom->len) & 1) && (ret_val == 0)) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   561
		/* need read/modify/write of last changed EEPROM word */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   562
		/* only the first byte of the word is being modified */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   563
		ret_val = e1000_read_eeprom(hw, last_word, 1,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   564
		                  &eeprom_buff[last_word - first_word]);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   565
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   566
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   567
	/* Device's eeprom is always little-endian, word addressable */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   568
	for (i = 0; i < last_word - first_word + 1; i++)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   569
		le16_to_cpus(&eeprom_buff[i]);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   570
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   571
	memcpy(ptr, bytes, eeprom->len);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   572
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   573
	for (i = 0; i < last_word - first_word + 1; i++)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   574
		eeprom_buff[i] = cpu_to_le16(eeprom_buff[i]);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   575
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   576
	ret_val = e1000_write_eeprom(hw, first_word,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   577
				     last_word - first_word + 1, eeprom_buff);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   578
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   579
	/* Update the checksum over the first part of the EEPROM if needed */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   580
	if ((ret_val == 0) && (first_word <= EEPROM_CHECKSUM_REG))
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   581
		e1000_update_eeprom_checksum(hw);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   582
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   583
	kfree(eeprom_buff);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   584
	return ret_val;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   585
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   586
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   587
static void e1000_get_drvinfo(struct net_device *netdev,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   588
			      struct ethtool_drvinfo *drvinfo)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   589
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   590
	struct e1000_adapter *adapter = netdev_priv(netdev);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   591
	char firmware_version[32];
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   592
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   593
	strncpy(drvinfo->driver,  e1000_driver_name, 32);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   594
	strncpy(drvinfo->version, e1000_driver_version, 32);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   595
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   596
	sprintf(firmware_version, "N/A");
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   597
	strncpy(drvinfo->fw_version, firmware_version, 32);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   598
	strncpy(drvinfo->bus_info, pci_name(adapter->pdev), 32);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   599
	drvinfo->regdump_len = e1000_get_regs_len(netdev);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   600
	drvinfo->eedump_len = e1000_get_eeprom_len(netdev);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   601
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   602
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   603
static void e1000_get_ringparam(struct net_device *netdev,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   604
				struct ethtool_ringparam *ring)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   605
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   606
	struct e1000_adapter *adapter = netdev_priv(netdev);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   607
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   608
	e1000_mac_type mac_type = hw->mac_type;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   609
	struct e1000_tx_ring *txdr = adapter->tx_ring;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   610
	struct e1000_rx_ring *rxdr = adapter->rx_ring;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   611
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   612
	ring->rx_max_pending = (mac_type < e1000_82544) ? E1000_MAX_RXD :
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   613
		E1000_MAX_82544_RXD;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   614
	ring->tx_max_pending = (mac_type < e1000_82544) ? E1000_MAX_TXD :
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   615
		E1000_MAX_82544_TXD;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   616
	ring->rx_mini_max_pending = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   617
	ring->rx_jumbo_max_pending = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   618
	ring->rx_pending = rxdr->count;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   619
	ring->tx_pending = txdr->count;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   620
	ring->rx_mini_pending = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   621
	ring->rx_jumbo_pending = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   622
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   623
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   624
static int e1000_set_ringparam(struct net_device *netdev,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   625
			       struct ethtool_ringparam *ring)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   626
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   627
	struct e1000_adapter *adapter = netdev_priv(netdev);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   628
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   629
	e1000_mac_type mac_type = hw->mac_type;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   630
	struct e1000_tx_ring *txdr, *tx_old;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   631
	struct e1000_rx_ring *rxdr, *rx_old;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   632
	int i, err;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   633
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   634
	if (adapter->ecdev)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   635
		return -EBUSY;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   636
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   637
	if ((ring->rx_mini_pending) || (ring->rx_jumbo_pending))
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   638
		return -EINVAL;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   639
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   640
	while (test_and_set_bit(__E1000_RESETTING, &adapter->flags))
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   641
		msleep(1);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   642
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   643
	if (netif_running(adapter->netdev))
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   644
		e1000_down(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   645
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   646
	tx_old = adapter->tx_ring;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   647
	rx_old = adapter->rx_ring;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   648
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   649
	err = -ENOMEM;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   650
	txdr = kcalloc(adapter->num_tx_queues, sizeof(struct e1000_tx_ring), GFP_KERNEL);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   651
	if (!txdr)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   652
		goto err_alloc_tx;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   653
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   654
	rxdr = kcalloc(adapter->num_rx_queues, sizeof(struct e1000_rx_ring), GFP_KERNEL);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   655
	if (!rxdr)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   656
		goto err_alloc_rx;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   657
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   658
	adapter->tx_ring = txdr;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   659
	adapter->rx_ring = rxdr;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   660
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   661
	rxdr->count = max(ring->rx_pending,(u32)E1000_MIN_RXD);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   662
	rxdr->count = min(rxdr->count,(u32)(mac_type < e1000_82544 ?
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   663
		E1000_MAX_RXD : E1000_MAX_82544_RXD));
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   664
	rxdr->count = ALIGN(rxdr->count, REQ_RX_DESCRIPTOR_MULTIPLE);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   665
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   666
	txdr->count = max(ring->tx_pending,(u32)E1000_MIN_TXD);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   667
	txdr->count = min(txdr->count,(u32)(mac_type < e1000_82544 ?
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   668
		E1000_MAX_TXD : E1000_MAX_82544_TXD));
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   669
	txdr->count = ALIGN(txdr->count, REQ_TX_DESCRIPTOR_MULTIPLE);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   670
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   671
	for (i = 0; i < adapter->num_tx_queues; i++)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   672
		txdr[i].count = txdr->count;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   673
	for (i = 0; i < adapter->num_rx_queues; i++)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   674
		rxdr[i].count = rxdr->count;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   675
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   676
	if (netif_running(adapter->netdev)) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   677
		/* Try to get new resources before deleting old */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   678
		err = e1000_setup_all_rx_resources(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   679
		if (err)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   680
			goto err_setup_rx;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   681
		err = e1000_setup_all_tx_resources(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   682
		if (err)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   683
			goto err_setup_tx;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   684
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   685
		/* save the new, restore the old in order to free it,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   686
		 * then restore the new back again */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   687
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   688
		adapter->rx_ring = rx_old;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   689
		adapter->tx_ring = tx_old;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   690
		e1000_free_all_rx_resources(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   691
		e1000_free_all_tx_resources(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   692
		kfree(tx_old);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   693
		kfree(rx_old);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   694
		adapter->rx_ring = rxdr;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   695
		adapter->tx_ring = txdr;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   696
		err = e1000_up(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   697
		if (err)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   698
			goto err_setup;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   699
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   700
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   701
	clear_bit(__E1000_RESETTING, &adapter->flags);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   702
	return 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   703
err_setup_tx:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   704
	e1000_free_all_rx_resources(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   705
err_setup_rx:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   706
	adapter->rx_ring = rx_old;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   707
	adapter->tx_ring = tx_old;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   708
	kfree(rxdr);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   709
err_alloc_rx:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   710
	kfree(txdr);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   711
err_alloc_tx:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   712
	e1000_up(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   713
err_setup:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   714
	clear_bit(__E1000_RESETTING, &adapter->flags);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   715
	return err;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   716
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   717
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   718
static bool reg_pattern_test(struct e1000_adapter *adapter, u64 *data, int reg,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   719
			     u32 mask, u32 write)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   720
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   721
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   722
	static const u32 test[] =
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   723
		{0x5A5A5A5A, 0xA5A5A5A5, 0x00000000, 0xFFFFFFFF};
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   724
	u8 __iomem *address = hw->hw_addr + reg;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   725
	u32 read;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   726
	int i;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   727
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   728
	for (i = 0; i < ARRAY_SIZE(test); i++) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   729
		writel(write & test[i], address);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   730
		read = readl(address);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   731
		if (read != (write & test[i] & mask)) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   732
			e_err(drv, "pattern test reg %04X failed: "
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   733
			      "got 0x%08X expected 0x%08X\n",
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   734
			      reg, read, (write & test[i] & mask));
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   735
			*data = reg;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   736
			return true;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   737
		}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   738
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   739
	return false;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   740
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   741
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   742
static bool reg_set_and_check(struct e1000_adapter *adapter, u64 *data, int reg,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   743
			      u32 mask, u32 write)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   744
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   745
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   746
	u8 __iomem *address = hw->hw_addr + reg;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   747
	u32 read;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   748
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   749
	writel(write & mask, address);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   750
	read = readl(address);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   751
	if ((read & mask) != (write & mask)) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   752
		e_err(drv, "set/check reg %04X test failed: "
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   753
		      "got 0x%08X expected 0x%08X\n",
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   754
		      reg, (read & mask), (write & mask));
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   755
		*data = reg;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   756
		return true;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   757
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   758
	return false;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   759
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   760
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   761
#define REG_PATTERN_TEST(reg, mask, write)			     \
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   762
	do {							     \
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   763
		if (reg_pattern_test(adapter, data,		     \
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   764
			     (hw->mac_type >= e1000_82543)   \
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   765
			     ? E1000_##reg : E1000_82542_##reg,	     \
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   766
			     mask, write))			     \
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   767
			return 1;				     \
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   768
	} while (0)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   769
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   770
#define REG_SET_AND_CHECK(reg, mask, write)			     \
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   771
	do {							     \
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   772
		if (reg_set_and_check(adapter, data,		     \
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   773
			      (hw->mac_type >= e1000_82543)  \
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   774
			      ? E1000_##reg : E1000_82542_##reg,     \
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   775
			      mask, write))			     \
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   776
			return 1;				     \
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   777
	} while (0)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   778
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   779
static int e1000_reg_test(struct e1000_adapter *adapter, u64 *data)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   780
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   781
	u32 value, before, after;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   782
	u32 i, toggle;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   783
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   784
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   785
	/* The status register is Read Only, so a write should fail.
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   786
	 * Some bits that get toggled are ignored.
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   787
	 */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   788
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   789
	/* there are several bits on newer hardware that are r/w */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   790
	toggle = 0xFFFFF833;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   791
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   792
	before = er32(STATUS);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   793
	value = (er32(STATUS) & toggle);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   794
	ew32(STATUS, toggle);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   795
	after = er32(STATUS) & toggle;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   796
	if (value != after) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   797
		e_err(drv, "failed STATUS register test got: "
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   798
		      "0x%08X expected: 0x%08X\n", after, value);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   799
		*data = 1;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   800
		return 1;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   801
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   802
	/* restore previous status */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   803
	ew32(STATUS, before);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   804
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   805
	REG_PATTERN_TEST(FCAL, 0xFFFFFFFF, 0xFFFFFFFF);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   806
	REG_PATTERN_TEST(FCAH, 0x0000FFFF, 0xFFFFFFFF);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   807
	REG_PATTERN_TEST(FCT, 0x0000FFFF, 0xFFFFFFFF);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   808
	REG_PATTERN_TEST(VET, 0x0000FFFF, 0xFFFFFFFF);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   809
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   810
	REG_PATTERN_TEST(RDTR, 0x0000FFFF, 0xFFFFFFFF);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   811
	REG_PATTERN_TEST(RDBAH, 0xFFFFFFFF, 0xFFFFFFFF);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   812
	REG_PATTERN_TEST(RDLEN, 0x000FFF80, 0x000FFFFF);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   813
	REG_PATTERN_TEST(RDH, 0x0000FFFF, 0x0000FFFF);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   814
	REG_PATTERN_TEST(RDT, 0x0000FFFF, 0x0000FFFF);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   815
	REG_PATTERN_TEST(FCRTH, 0x0000FFF8, 0x0000FFF8);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   816
	REG_PATTERN_TEST(FCTTV, 0x0000FFFF, 0x0000FFFF);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   817
	REG_PATTERN_TEST(TIPG, 0x3FFFFFFF, 0x3FFFFFFF);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   818
	REG_PATTERN_TEST(TDBAH, 0xFFFFFFFF, 0xFFFFFFFF);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   819
	REG_PATTERN_TEST(TDLEN, 0x000FFF80, 0x000FFFFF);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   820
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   821
	REG_SET_AND_CHECK(RCTL, 0xFFFFFFFF, 0x00000000);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   822
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   823
	before = 0x06DFB3FE;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   824
	REG_SET_AND_CHECK(RCTL, before, 0x003FFFFB);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   825
	REG_SET_AND_CHECK(TCTL, 0xFFFFFFFF, 0x00000000);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   826
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   827
	if (hw->mac_type >= e1000_82543) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   828
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   829
		REG_SET_AND_CHECK(RCTL, before, 0xFFFFFFFF);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   830
		REG_PATTERN_TEST(RDBAL, 0xFFFFFFF0, 0xFFFFFFFF);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   831
		REG_PATTERN_TEST(TXCW, 0xC000FFFF, 0x0000FFFF);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   832
		REG_PATTERN_TEST(TDBAL, 0xFFFFFFF0, 0xFFFFFFFF);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   833
		REG_PATTERN_TEST(TIDV, 0x0000FFFF, 0x0000FFFF);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   834
		value = E1000_RAR_ENTRIES;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   835
		for (i = 0; i < value; i++) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   836
			REG_PATTERN_TEST(RA + (((i << 1) + 1) << 2), 0x8003FFFF,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   837
			                 0xFFFFFFFF);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   838
		}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   839
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   840
	} else {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   841
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   842
		REG_SET_AND_CHECK(RCTL, 0xFFFFFFFF, 0x01FFFFFF);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   843
		REG_PATTERN_TEST(RDBAL, 0xFFFFF000, 0xFFFFFFFF);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   844
		REG_PATTERN_TEST(TXCW, 0x0000FFFF, 0x0000FFFF);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   845
		REG_PATTERN_TEST(TDBAL, 0xFFFFF000, 0xFFFFFFFF);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   846
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   847
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   848
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   849
	value = E1000_MC_TBL_SIZE;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   850
	for (i = 0; i < value; i++)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   851
		REG_PATTERN_TEST(MTA + (i << 2), 0xFFFFFFFF, 0xFFFFFFFF);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   852
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   853
	*data = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   854
	return 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   855
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   856
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   857
static int e1000_eeprom_test(struct e1000_adapter *adapter, u64 *data)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   858
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   859
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   860
	u16 temp;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   861
	u16 checksum = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   862
	u16 i;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   863
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   864
	*data = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   865
	/* Read and add up the contents of the EEPROM */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   866
	for (i = 0; i < (EEPROM_CHECKSUM_REG + 1); i++) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   867
		if ((e1000_read_eeprom(hw, i, 1, &temp)) < 0) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   868
			*data = 1;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   869
			break;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   870
		}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   871
		checksum += temp;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   872
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   873
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   874
	/* If Checksum is not Correct return error else test passed */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   875
	if ((checksum != (u16)EEPROM_SUM) && !(*data))
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   876
		*data = 2;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   877
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   878
	return *data;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   879
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   880
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   881
static irqreturn_t e1000_test_intr(int irq, void *data)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   882
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   883
	struct net_device *netdev = (struct net_device *)data;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   884
	struct e1000_adapter *adapter = netdev_priv(netdev);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   885
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   886
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   887
	adapter->test_icr |= er32(ICR);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   888
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   889
	return IRQ_HANDLED;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   890
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   891
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   892
static int e1000_intr_test(struct e1000_adapter *adapter, u64 *data)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   893
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   894
	struct net_device *netdev = adapter->netdev;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   895
	u32 mask, i = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   896
	bool shared_int = true;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   897
	u32 irq = adapter->pdev->irq;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   898
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   899
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   900
	*data = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   901
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   902
	/* NOTE: we don't test MSI interrupts here, yet */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   903
	/* Hook up test interrupt handler just for this test */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   904
	if (!request_irq(irq, e1000_test_intr, IRQF_PROBE_SHARED, netdev->name,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   905
	                 netdev))
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   906
		shared_int = false;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   907
	else if (request_irq(irq, e1000_test_intr, IRQF_SHARED,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   908
	         netdev->name, netdev)) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   909
		*data = 1;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   910
		return -1;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   911
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   912
	e_info(hw, "testing %s interrupt\n", (shared_int ?
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   913
	       "shared" : "unshared"));
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   914
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   915
	/* Disable all the interrupts */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   916
	ew32(IMC, 0xFFFFFFFF);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   917
	msleep(10);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   918
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   919
	/* Test each interrupt */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   920
	for (; i < 10; i++) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   921
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   922
		/* Interrupt to test */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   923
		mask = 1 << i;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   924
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   925
		if (!shared_int) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   926
			/* Disable the interrupt to be reported in
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   927
			 * the cause register and then force the same
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   928
			 * interrupt and see if one gets posted.  If
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   929
			 * an interrupt was posted to the bus, the
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   930
			 * test failed.
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   931
			 */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   932
			adapter->test_icr = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   933
			ew32(IMC, mask);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   934
			ew32(ICS, mask);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   935
			msleep(10);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   936
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   937
			if (adapter->test_icr & mask) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   938
				*data = 3;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   939
				break;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   940
			}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   941
		}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   942
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   943
		/* Enable the interrupt to be reported in
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   944
		 * the cause register and then force the same
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   945
		 * interrupt and see if one gets posted.  If
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   946
		 * an interrupt was not posted to the bus, the
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   947
		 * test failed.
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   948
		 */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   949
		adapter->test_icr = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   950
		ew32(IMS, mask);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   951
		ew32(ICS, mask);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   952
		msleep(10);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   953
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   954
		if (!(adapter->test_icr & mask)) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   955
			*data = 4;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   956
			break;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   957
		}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   958
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   959
		if (!shared_int) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   960
			/* Disable the other interrupts to be reported in
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   961
			 * the cause register and then force the other
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   962
			 * interrupts and see if any get posted.  If
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   963
			 * an interrupt was posted to the bus, the
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   964
			 * test failed.
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   965
			 */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   966
			adapter->test_icr = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   967
			ew32(IMC, ~mask & 0x00007FFF);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   968
			ew32(ICS, ~mask & 0x00007FFF);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   969
			msleep(10);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   970
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   971
			if (adapter->test_icr) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   972
				*data = 5;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   973
				break;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   974
			}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   975
		}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   976
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   977
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   978
	/* Disable all the interrupts */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   979
	ew32(IMC, 0xFFFFFFFF);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   980
	msleep(10);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   981
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   982
	/* Unhook test interrupt handler */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   983
	free_irq(irq, netdev);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   984
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   985
	return *data;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   986
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   987
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   988
static void e1000_free_desc_rings(struct e1000_adapter *adapter)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   989
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   990
	struct e1000_tx_ring *txdr = &adapter->test_tx_ring;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   991
	struct e1000_rx_ring *rxdr = &adapter->test_rx_ring;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   992
	struct pci_dev *pdev = adapter->pdev;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   993
	int i;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   994
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   995
	if (txdr->desc && txdr->buffer_info) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   996
		for (i = 0; i < txdr->count; i++) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   997
			if (txdr->buffer_info[i].dma)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   998
				dma_unmap_single(&pdev->dev,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
   999
						 txdr->buffer_info[i].dma,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1000
						 txdr->buffer_info[i].length,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1001
						 DMA_TO_DEVICE);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1002
			if (txdr->buffer_info[i].skb)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1003
				dev_kfree_skb(txdr->buffer_info[i].skb);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1004
		}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1005
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1006
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1007
	if (rxdr->desc && rxdr->buffer_info) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1008
		for (i = 0; i < rxdr->count; i++) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1009
			if (rxdr->buffer_info[i].dma)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1010
				dma_unmap_single(&pdev->dev,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1011
						 rxdr->buffer_info[i].dma,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1012
						 rxdr->buffer_info[i].length,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1013
						 DMA_FROM_DEVICE);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1014
			if (rxdr->buffer_info[i].skb)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1015
				dev_kfree_skb(rxdr->buffer_info[i].skb);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1016
		}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1017
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1018
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1019
	if (txdr->desc) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1020
		dma_free_coherent(&pdev->dev, txdr->size, txdr->desc,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1021
				  txdr->dma);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1022
		txdr->desc = NULL;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1023
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1024
	if (rxdr->desc) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1025
		dma_free_coherent(&pdev->dev, rxdr->size, rxdr->desc,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1026
				  rxdr->dma);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1027
		rxdr->desc = NULL;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1028
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1029
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1030
	kfree(txdr->buffer_info);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1031
	txdr->buffer_info = NULL;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1032
	kfree(rxdr->buffer_info);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1033
	rxdr->buffer_info = NULL;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1034
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1035
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1036
static int e1000_setup_desc_rings(struct e1000_adapter *adapter)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1037
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1038
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1039
	struct e1000_tx_ring *txdr = &adapter->test_tx_ring;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1040
	struct e1000_rx_ring *rxdr = &adapter->test_rx_ring;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1041
	struct pci_dev *pdev = adapter->pdev;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1042
	u32 rctl;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1043
	int i, ret_val;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1044
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1045
	/* Setup Tx descriptor ring and Tx buffers */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1046
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1047
	if (!txdr->count)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1048
		txdr->count = E1000_DEFAULT_TXD;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1049
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1050
	txdr->buffer_info = kcalloc(txdr->count, sizeof(struct e1000_buffer),
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1051
				    GFP_KERNEL);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1052
	if (!txdr->buffer_info) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1053
		ret_val = 1;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1054
		goto err_nomem;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1055
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1056
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1057
	txdr->size = txdr->count * sizeof(struct e1000_tx_desc);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1058
	txdr->size = ALIGN(txdr->size, 4096);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1059
	txdr->desc = dma_alloc_coherent(&pdev->dev, txdr->size, &txdr->dma,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1060
					GFP_KERNEL);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1061
	if (!txdr->desc) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1062
		ret_val = 2;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1063
		goto err_nomem;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1064
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1065
	memset(txdr->desc, 0, txdr->size);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1066
	txdr->next_to_use = txdr->next_to_clean = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1067
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1068
	ew32(TDBAL, ((u64)txdr->dma & 0x00000000FFFFFFFF));
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1069
	ew32(TDBAH, ((u64)txdr->dma >> 32));
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1070
	ew32(TDLEN, txdr->count * sizeof(struct e1000_tx_desc));
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1071
	ew32(TDH, 0);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1072
	ew32(TDT, 0);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1073
	ew32(TCTL, E1000_TCTL_PSP | E1000_TCTL_EN |
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1074
	     E1000_COLLISION_THRESHOLD << E1000_CT_SHIFT |
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1075
	     E1000_FDX_COLLISION_DISTANCE << E1000_COLD_SHIFT);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1076
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1077
	for (i = 0; i < txdr->count; i++) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1078
		struct e1000_tx_desc *tx_desc = E1000_TX_DESC(*txdr, i);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1079
		struct sk_buff *skb;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1080
		unsigned int size = 1024;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1081
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1082
		skb = alloc_skb(size, GFP_KERNEL);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1083
		if (!skb) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1084
			ret_val = 3;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1085
			goto err_nomem;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1086
		}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1087
		skb_put(skb, size);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1088
		txdr->buffer_info[i].skb = skb;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1089
		txdr->buffer_info[i].length = skb->len;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1090
		txdr->buffer_info[i].dma =
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1091
			dma_map_single(&pdev->dev, skb->data, skb->len,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1092
				       DMA_TO_DEVICE);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1093
		tx_desc->buffer_addr = cpu_to_le64(txdr->buffer_info[i].dma);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1094
		tx_desc->lower.data = cpu_to_le32(skb->len);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1095
		tx_desc->lower.data |= cpu_to_le32(E1000_TXD_CMD_EOP |
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1096
						   E1000_TXD_CMD_IFCS |
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1097
						   E1000_TXD_CMD_RPS);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1098
		tx_desc->upper.data = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1099
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1100
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1101
	/* Setup Rx descriptor ring and Rx buffers */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1102
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1103
	if (!rxdr->count)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1104
		rxdr->count = E1000_DEFAULT_RXD;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1105
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1106
	rxdr->buffer_info = kcalloc(rxdr->count, sizeof(struct e1000_buffer),
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1107
				    GFP_KERNEL);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1108
	if (!rxdr->buffer_info) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1109
		ret_val = 4;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1110
		goto err_nomem;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1111
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1112
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1113
	rxdr->size = rxdr->count * sizeof(struct e1000_rx_desc);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1114
	rxdr->desc = dma_alloc_coherent(&pdev->dev, rxdr->size, &rxdr->dma,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1115
					GFP_KERNEL);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1116
	if (!rxdr->desc) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1117
		ret_val = 5;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1118
		goto err_nomem;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1119
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1120
	memset(rxdr->desc, 0, rxdr->size);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1121
	rxdr->next_to_use = rxdr->next_to_clean = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1122
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1123
	rctl = er32(RCTL);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1124
	ew32(RCTL, rctl & ~E1000_RCTL_EN);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1125
	ew32(RDBAL, ((u64)rxdr->dma & 0xFFFFFFFF));
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1126
	ew32(RDBAH, ((u64)rxdr->dma >> 32));
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1127
	ew32(RDLEN, rxdr->size);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1128
	ew32(RDH, 0);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1129
	ew32(RDT, 0);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1130
	rctl = E1000_RCTL_EN | E1000_RCTL_BAM | E1000_RCTL_SZ_2048 |
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1131
		E1000_RCTL_LBM_NO | E1000_RCTL_RDMTS_HALF |
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1132
		(hw->mc_filter_type << E1000_RCTL_MO_SHIFT);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1133
	ew32(RCTL, rctl);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1134
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1135
	for (i = 0; i < rxdr->count; i++) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1136
		struct e1000_rx_desc *rx_desc = E1000_RX_DESC(*rxdr, i);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1137
		struct sk_buff *skb;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1138
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1139
		skb = alloc_skb(E1000_RXBUFFER_2048 + NET_IP_ALIGN, GFP_KERNEL);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1140
		if (!skb) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1141
			ret_val = 6;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1142
			goto err_nomem;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1143
		}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1144
		skb_reserve(skb, NET_IP_ALIGN);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1145
		rxdr->buffer_info[i].skb = skb;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1146
		rxdr->buffer_info[i].length = E1000_RXBUFFER_2048;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1147
		rxdr->buffer_info[i].dma =
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1148
			dma_map_single(&pdev->dev, skb->data,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1149
				       E1000_RXBUFFER_2048, DMA_FROM_DEVICE);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1150
		rx_desc->buffer_addr = cpu_to_le64(rxdr->buffer_info[i].dma);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1151
		memset(skb->data, 0x00, skb->len);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1152
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1153
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1154
	return 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1155
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1156
err_nomem:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1157
	e1000_free_desc_rings(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1158
	return ret_val;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1159
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1160
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1161
static void e1000_phy_disable_receiver(struct e1000_adapter *adapter)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1162
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1163
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1164
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1165
	/* Write out to PHY registers 29 and 30 to disable the Receiver. */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1166
	e1000_write_phy_reg(hw, 29, 0x001F);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1167
	e1000_write_phy_reg(hw, 30, 0x8FFC);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1168
	e1000_write_phy_reg(hw, 29, 0x001A);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1169
	e1000_write_phy_reg(hw, 30, 0x8FF0);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1170
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1171
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1172
static void e1000_phy_reset_clk_and_crs(struct e1000_adapter *adapter)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1173
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1174
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1175
	u16 phy_reg;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1176
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1177
	/* Because we reset the PHY above, we need to re-force TX_CLK in the
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1178
	 * Extended PHY Specific Control Register to 25MHz clock.  This
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1179
	 * value defaults back to a 2.5MHz clock when the PHY is reset.
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1180
	 */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1181
	e1000_read_phy_reg(hw, M88E1000_EXT_PHY_SPEC_CTRL, &phy_reg);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1182
	phy_reg |= M88E1000_EPSCR_TX_CLK_25;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1183
	e1000_write_phy_reg(hw,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1184
		M88E1000_EXT_PHY_SPEC_CTRL, phy_reg);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1185
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1186
	/* In addition, because of the s/w reset above, we need to enable
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1187
	 * CRS on TX.  This must be set for both full and half duplex
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1188
	 * operation.
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1189
	 */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1190
	e1000_read_phy_reg(hw, M88E1000_PHY_SPEC_CTRL, &phy_reg);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1191
	phy_reg |= M88E1000_PSCR_ASSERT_CRS_ON_TX;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1192
	e1000_write_phy_reg(hw,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1193
		M88E1000_PHY_SPEC_CTRL, phy_reg);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1194
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1195
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1196
static int e1000_nonintegrated_phy_loopback(struct e1000_adapter *adapter)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1197
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1198
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1199
	u32 ctrl_reg;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1200
	u16 phy_reg;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1201
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1202
	/* Setup the Device Control Register for PHY loopback test. */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1203
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1204
	ctrl_reg = er32(CTRL);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1205
	ctrl_reg |= (E1000_CTRL_ILOS |		/* Invert Loss-Of-Signal */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1206
		     E1000_CTRL_FRCSPD |	/* Set the Force Speed Bit */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1207
		     E1000_CTRL_FRCDPX |	/* Set the Force Duplex Bit */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1208
		     E1000_CTRL_SPD_1000 |	/* Force Speed to 1000 */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1209
		     E1000_CTRL_FD);		/* Force Duplex to FULL */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1210
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1211
	ew32(CTRL, ctrl_reg);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1212
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1213
	/* Read the PHY Specific Control Register (0x10) */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1214
	e1000_read_phy_reg(hw, M88E1000_PHY_SPEC_CTRL, &phy_reg);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1215
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1216
	/* Clear Auto-Crossover bits in PHY Specific Control Register
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1217
	 * (bits 6:5).
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1218
	 */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1219
	phy_reg &= ~M88E1000_PSCR_AUTO_X_MODE;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1220
	e1000_write_phy_reg(hw, M88E1000_PHY_SPEC_CTRL, phy_reg);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1221
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1222
	/* Perform software reset on the PHY */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1223
	e1000_phy_reset(hw);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1224
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1225
	/* Have to setup TX_CLK and TX_CRS after software reset */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1226
	e1000_phy_reset_clk_and_crs(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1227
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1228
	e1000_write_phy_reg(hw, PHY_CTRL, 0x8100);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1229
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1230
	/* Wait for reset to complete. */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1231
	udelay(500);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1232
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1233
	/* Have to setup TX_CLK and TX_CRS after software reset */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1234
	e1000_phy_reset_clk_and_crs(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1235
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1236
	/* Write out to PHY registers 29 and 30 to disable the Receiver. */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1237
	e1000_phy_disable_receiver(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1238
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1239
	/* Set the loopback bit in the PHY control register. */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1240
	e1000_read_phy_reg(hw, PHY_CTRL, &phy_reg);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1241
	phy_reg |= MII_CR_LOOPBACK;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1242
	e1000_write_phy_reg(hw, PHY_CTRL, phy_reg);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1243
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1244
	/* Setup TX_CLK and TX_CRS one more time. */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1245
	e1000_phy_reset_clk_and_crs(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1246
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1247
	/* Check Phy Configuration */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1248
	e1000_read_phy_reg(hw, PHY_CTRL, &phy_reg);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1249
	if (phy_reg != 0x4100)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1250
		 return 9;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1251
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1252
	e1000_read_phy_reg(hw, M88E1000_EXT_PHY_SPEC_CTRL, &phy_reg);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1253
	if (phy_reg != 0x0070)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1254
		return 10;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1255
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1256
	e1000_read_phy_reg(hw, 29, &phy_reg);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1257
	if (phy_reg != 0x001A)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1258
		return 11;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1259
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1260
	return 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1261
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1262
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1263
static int e1000_integrated_phy_loopback(struct e1000_adapter *adapter)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1264
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1265
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1266
	u32 ctrl_reg = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1267
	u32 stat_reg = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1268
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1269
	hw->autoneg = false;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1270
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1271
	if (hw->phy_type == e1000_phy_m88) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1272
		/* Auto-MDI/MDIX Off */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1273
		e1000_write_phy_reg(hw,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1274
				    M88E1000_PHY_SPEC_CTRL, 0x0808);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1275
		/* reset to update Auto-MDI/MDIX */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1276
		e1000_write_phy_reg(hw, PHY_CTRL, 0x9140);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1277
		/* autoneg off */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1278
		e1000_write_phy_reg(hw, PHY_CTRL, 0x8140);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1279
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1280
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1281
	ctrl_reg = er32(CTRL);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1282
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1283
	/* force 1000, set loopback */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1284
	e1000_write_phy_reg(hw, PHY_CTRL, 0x4140);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1285
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1286
	/* Now set up the MAC to the same speed/duplex as the PHY. */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1287
	ctrl_reg = er32(CTRL);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1288
	ctrl_reg &= ~E1000_CTRL_SPD_SEL; /* Clear the speed sel bits */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1289
	ctrl_reg |= (E1000_CTRL_FRCSPD | /* Set the Force Speed Bit */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1290
			E1000_CTRL_FRCDPX | /* Set the Force Duplex Bit */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1291
			E1000_CTRL_SPD_1000 |/* Force Speed to 1000 */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1292
			E1000_CTRL_FD);	 /* Force Duplex to FULL */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1293
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1294
	if (hw->media_type == e1000_media_type_copper &&
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1295
	   hw->phy_type == e1000_phy_m88)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1296
		ctrl_reg |= E1000_CTRL_ILOS; /* Invert Loss of Signal */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1297
	else {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1298
		/* Set the ILOS bit on the fiber Nic is half
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1299
		 * duplex link is detected. */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1300
		stat_reg = er32(STATUS);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1301
		if ((stat_reg & E1000_STATUS_FD) == 0)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1302
			ctrl_reg |= (E1000_CTRL_ILOS | E1000_CTRL_SLU);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1303
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1304
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1305
	ew32(CTRL, ctrl_reg);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1306
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1307
	/* Disable the receiver on the PHY so when a cable is plugged in, the
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1308
	 * PHY does not begin to autoneg when a cable is reconnected to the NIC.
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1309
	 */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1310
	if (hw->phy_type == e1000_phy_m88)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1311
		e1000_phy_disable_receiver(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1312
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1313
	udelay(500);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1314
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1315
	return 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1316
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1317
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1318
static int e1000_set_phy_loopback(struct e1000_adapter *adapter)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1319
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1320
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1321
	u16 phy_reg = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1322
	u16 count = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1323
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1324
	switch (hw->mac_type) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1325
	case e1000_82543:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1326
		if (hw->media_type == e1000_media_type_copper) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1327
			/* Attempt to setup Loopback mode on Non-integrated PHY.
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1328
			 * Some PHY registers get corrupted at random, so
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1329
			 * attempt this 10 times.
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1330
			 */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1331
			while (e1000_nonintegrated_phy_loopback(adapter) &&
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1332
			      count++ < 10);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1333
			if (count < 11)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1334
				return 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1335
		}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1336
		break;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1337
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1338
	case e1000_82544:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1339
	case e1000_82540:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1340
	case e1000_82545:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1341
	case e1000_82545_rev_3:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1342
	case e1000_82546:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1343
	case e1000_82546_rev_3:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1344
	case e1000_82541:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1345
	case e1000_82541_rev_2:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1346
	case e1000_82547:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1347
	case e1000_82547_rev_2:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1348
		return e1000_integrated_phy_loopback(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1349
		break;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1350
	default:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1351
		/* Default PHY loopback work is to read the MII
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1352
		 * control register and assert bit 14 (loopback mode).
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1353
		 */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1354
		e1000_read_phy_reg(hw, PHY_CTRL, &phy_reg);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1355
		phy_reg |= MII_CR_LOOPBACK;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1356
		e1000_write_phy_reg(hw, PHY_CTRL, phy_reg);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1357
		return 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1358
		break;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1359
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1360
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1361
	return 8;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1362
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1363
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1364
static int e1000_setup_loopback_test(struct e1000_adapter *adapter)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1365
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1366
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1367
	u32 rctl;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1368
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1369
	if (hw->media_type == e1000_media_type_fiber ||
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1370
	    hw->media_type == e1000_media_type_internal_serdes) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1371
		switch (hw->mac_type) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1372
		case e1000_82545:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1373
		case e1000_82546:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1374
		case e1000_82545_rev_3:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1375
		case e1000_82546_rev_3:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1376
			return e1000_set_phy_loopback(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1377
			break;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1378
		default:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1379
			rctl = er32(RCTL);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1380
			rctl |= E1000_RCTL_LBM_TCVR;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1381
			ew32(RCTL, rctl);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1382
			return 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1383
		}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1384
	} else if (hw->media_type == e1000_media_type_copper)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1385
		return e1000_set_phy_loopback(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1386
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1387
	return 7;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1388
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1389
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1390
static void e1000_loopback_cleanup(struct e1000_adapter *adapter)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1391
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1392
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1393
	u32 rctl;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1394
	u16 phy_reg;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1395
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1396
	rctl = er32(RCTL);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1397
	rctl &= ~(E1000_RCTL_LBM_TCVR | E1000_RCTL_LBM_MAC);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1398
	ew32(RCTL, rctl);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1399
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1400
	switch (hw->mac_type) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1401
	case e1000_82545:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1402
	case e1000_82546:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1403
	case e1000_82545_rev_3:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1404
	case e1000_82546_rev_3:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1405
	default:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1406
		hw->autoneg = true;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1407
		e1000_read_phy_reg(hw, PHY_CTRL, &phy_reg);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1408
		if (phy_reg & MII_CR_LOOPBACK) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1409
			phy_reg &= ~MII_CR_LOOPBACK;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1410
			e1000_write_phy_reg(hw, PHY_CTRL, phy_reg);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1411
			e1000_phy_reset(hw);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1412
		}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1413
		break;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1414
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1415
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1416
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1417
static void e1000_create_lbtest_frame(struct sk_buff *skb,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1418
				      unsigned int frame_size)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1419
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1420
	memset(skb->data, 0xFF, frame_size);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1421
	frame_size &= ~1;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1422
	memset(&skb->data[frame_size / 2], 0xAA, frame_size / 2 - 1);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1423
	memset(&skb->data[frame_size / 2 + 10], 0xBE, 1);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1424
	memset(&skb->data[frame_size / 2 + 12], 0xAF, 1);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1425
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1426
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1427
static int e1000_check_lbtest_frame(struct sk_buff *skb,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1428
				    unsigned int frame_size)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1429
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1430
	frame_size &= ~1;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1431
	if (*(skb->data + 3) == 0xFF) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1432
		if ((*(skb->data + frame_size / 2 + 10) == 0xBE) &&
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1433
		   (*(skb->data + frame_size / 2 + 12) == 0xAF)) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1434
			return 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1435
		}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1436
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1437
	return 13;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1438
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1439
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1440
static int e1000_run_loopback_test(struct e1000_adapter *adapter)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1441
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1442
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1443
	struct e1000_tx_ring *txdr = &adapter->test_tx_ring;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1444
	struct e1000_rx_ring *rxdr = &adapter->test_rx_ring;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1445
	struct pci_dev *pdev = adapter->pdev;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1446
	int i, j, k, l, lc, good_cnt, ret_val=0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1447
	unsigned long time;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1448
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1449
	ew32(RDT, rxdr->count - 1);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1450
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1451
	/* Calculate the loop count based on the largest descriptor ring
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1452
	 * The idea is to wrap the largest ring a number of times using 64
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1453
	 * send/receive pairs during each loop
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1454
	 */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1455
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1456
	if (rxdr->count <= txdr->count)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1457
		lc = ((txdr->count / 64) * 2) + 1;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1458
	else
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1459
		lc = ((rxdr->count / 64) * 2) + 1;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1460
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1461
	k = l = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1462
	for (j = 0; j <= lc; j++) { /* loop count loop */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1463
		for (i = 0; i < 64; i++) { /* send the packets */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1464
			e1000_create_lbtest_frame(txdr->buffer_info[i].skb,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1465
					1024);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1466
			dma_sync_single_for_device(&pdev->dev,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1467
						   txdr->buffer_info[k].dma,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1468
						   txdr->buffer_info[k].length,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1469
						   DMA_TO_DEVICE);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1470
			if (unlikely(++k == txdr->count)) k = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1471
		}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1472
		ew32(TDT, k);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1473
		msleep(200);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1474
		time = jiffies; /* set the start time for the receive */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1475
		good_cnt = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1476
		do { /* receive the sent packets */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1477
			dma_sync_single_for_cpu(&pdev->dev,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1478
						rxdr->buffer_info[l].dma,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1479
						rxdr->buffer_info[l].length,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1480
						DMA_FROM_DEVICE);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1481
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1482
			ret_val = e1000_check_lbtest_frame(
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1483
					rxdr->buffer_info[l].skb,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1484
				   	1024);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1485
			if (!ret_val)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1486
				good_cnt++;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1487
			if (unlikely(++l == rxdr->count)) l = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1488
			/* time + 20 msecs (200 msecs on 2.4) is more than
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1489
			 * enough time to complete the receives, if it's
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1490
			 * exceeded, break and error off
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1491
			 */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1492
		} while (good_cnt < 64 && jiffies < (time + 20));
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1493
		if (good_cnt != 64) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1494
			ret_val = 13; /* ret_val is the same as mis-compare */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1495
			break;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1496
		}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1497
		if (jiffies >= (time + 2)) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1498
			ret_val = 14; /* error code for time out error */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1499
			break;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1500
		}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1501
	} /* end loop count loop */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1502
	return ret_val;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1503
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1504
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1505
static int e1000_loopback_test(struct e1000_adapter *adapter, u64 *data)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1506
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1507
	*data = e1000_setup_desc_rings(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1508
	if (*data)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1509
		goto out;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1510
	*data = e1000_setup_loopback_test(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1511
	if (*data)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1512
		goto err_loopback;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1513
	*data = e1000_run_loopback_test(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1514
	e1000_loopback_cleanup(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1515
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1516
err_loopback:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1517
	e1000_free_desc_rings(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1518
out:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1519
	return *data;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1520
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1521
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1522
static int e1000_link_test(struct e1000_adapter *adapter, u64 *data)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1523
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1524
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1525
	*data = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1526
	if (hw->media_type == e1000_media_type_internal_serdes) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1527
		int i = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1528
		hw->serdes_has_link = false;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1529
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1530
		/* On some blade server designs, link establishment
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1531
		 * could take as long as 2-3 minutes */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1532
		do {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1533
			e1000_check_for_link(hw);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1534
			if (hw->serdes_has_link)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1535
				return *data;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1536
			msleep(20);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1537
		} while (i++ < 3750);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1538
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1539
		*data = 1;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1540
	} else {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1541
		e1000_check_for_link(hw);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1542
		if (hw->autoneg)  /* if auto_neg is set wait for it */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1543
			msleep(4000);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1544
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1545
		if (!(er32(STATUS) & E1000_STATUS_LU)) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1546
			*data = 1;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1547
		}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1548
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1549
	return *data;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1550
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1551
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1552
static int e1000_get_sset_count(struct net_device *netdev, int sset)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1553
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1554
	switch (sset) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1555
	case ETH_SS_TEST:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1556
		return E1000_TEST_LEN;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1557
	case ETH_SS_STATS:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1558
		return E1000_STATS_LEN;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1559
	default:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1560
		return -EOPNOTSUPP;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1561
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1562
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1563
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1564
static void e1000_diag_test(struct net_device *netdev,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1565
			    struct ethtool_test *eth_test, u64 *data)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1566
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1567
	struct e1000_adapter *adapter = netdev_priv(netdev);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1568
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1569
	bool if_running;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1570
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1571
	if (adapter->ecdev)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1572
		return;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1573
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1574
	if_running = netif_running(netdev);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1575
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1576
	set_bit(__E1000_TESTING, &adapter->flags);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1577
	if (eth_test->flags == ETH_TEST_FL_OFFLINE) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1578
		/* Offline tests */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1579
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1580
		/* save speed, duplex, autoneg settings */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1581
		u16 autoneg_advertised = hw->autoneg_advertised;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1582
		u8 forced_speed_duplex = hw->forced_speed_duplex;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1583
		u8 autoneg = hw->autoneg;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1584
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1585
		e_info(hw, "offline testing starting\n");
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1586
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1587
		/* Link test performed before hardware reset so autoneg doesn't
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1588
		 * interfere with test result */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1589
		if (e1000_link_test(adapter, &data[4]))
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1590
			eth_test->flags |= ETH_TEST_FL_FAILED;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1591
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1592
		if (if_running)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1593
			/* indicate we're in test mode */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1594
			dev_close(netdev);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1595
		else
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1596
			e1000_reset(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1597
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1598
		if (e1000_reg_test(adapter, &data[0]))
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1599
			eth_test->flags |= ETH_TEST_FL_FAILED;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1600
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1601
		e1000_reset(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1602
		if (e1000_eeprom_test(adapter, &data[1]))
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1603
			eth_test->flags |= ETH_TEST_FL_FAILED;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1604
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1605
		e1000_reset(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1606
		if (e1000_intr_test(adapter, &data[2]))
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1607
			eth_test->flags |= ETH_TEST_FL_FAILED;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1608
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1609
		e1000_reset(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1610
		/* make sure the phy is powered up */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1611
		e1000_power_up_phy(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1612
		if (e1000_loopback_test(adapter, &data[3]))
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1613
			eth_test->flags |= ETH_TEST_FL_FAILED;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1614
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1615
		/* restore speed, duplex, autoneg settings */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1616
		hw->autoneg_advertised = autoneg_advertised;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1617
		hw->forced_speed_duplex = forced_speed_duplex;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1618
		hw->autoneg = autoneg;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1619
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1620
		e1000_reset(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1621
		clear_bit(__E1000_TESTING, &adapter->flags);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1622
		if (if_running)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1623
			dev_open(netdev);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1624
	} else {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1625
		e_info(hw, "online testing starting\n");
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1626
		/* Online tests */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1627
		if (e1000_link_test(adapter, &data[4]))
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1628
			eth_test->flags |= ETH_TEST_FL_FAILED;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1629
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1630
		/* Online tests aren't run; pass by default */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1631
		data[0] = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1632
		data[1] = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1633
		data[2] = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1634
		data[3] = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1635
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1636
		clear_bit(__E1000_TESTING, &adapter->flags);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1637
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1638
	msleep_interruptible(4 * 1000);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1639
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1640
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1641
static int e1000_wol_exclusion(struct e1000_adapter *adapter,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1642
			       struct ethtool_wolinfo *wol)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1643
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1644
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1645
	int retval = 1; /* fail by default */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1646
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1647
	switch (hw->device_id) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1648
	case E1000_DEV_ID_82542:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1649
	case E1000_DEV_ID_82543GC_FIBER:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1650
	case E1000_DEV_ID_82543GC_COPPER:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1651
	case E1000_DEV_ID_82544EI_FIBER:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1652
	case E1000_DEV_ID_82546EB_QUAD_COPPER:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1653
	case E1000_DEV_ID_82545EM_FIBER:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1654
	case E1000_DEV_ID_82545EM_COPPER:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1655
	case E1000_DEV_ID_82546GB_QUAD_COPPER:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1656
	case E1000_DEV_ID_82546GB_PCIE:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1657
		/* these don't support WoL at all */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1658
		wol->supported = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1659
		break;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1660
	case E1000_DEV_ID_82546EB_FIBER:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1661
	case E1000_DEV_ID_82546GB_FIBER:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1662
		/* Wake events not supported on port B */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1663
		if (er32(STATUS) & E1000_STATUS_FUNC_1) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1664
			wol->supported = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1665
			break;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1666
		}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1667
		/* return success for non excluded adapter ports */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1668
		retval = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1669
		break;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1670
	case E1000_DEV_ID_82546GB_QUAD_COPPER_KSP3:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1671
		/* quad port adapters only support WoL on port A */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1672
		if (!adapter->quad_port_a) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1673
			wol->supported = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1674
			break;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1675
		}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1676
		/* return success for non excluded adapter ports */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1677
		retval = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1678
		break;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1679
	default:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1680
		/* dual port cards only support WoL on port A from now on
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1681
		 * unless it was enabled in the eeprom for port B
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1682
		 * so exclude FUNC_1 ports from having WoL enabled */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1683
		if (er32(STATUS) & E1000_STATUS_FUNC_1 &&
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1684
		    !adapter->eeprom_wol) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1685
			wol->supported = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1686
			break;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1687
		}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1688
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1689
		retval = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1690
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1691
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1692
	return retval;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1693
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1694
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1695
static void e1000_get_wol(struct net_device *netdev,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1696
			  struct ethtool_wolinfo *wol)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1697
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1698
	struct e1000_adapter *adapter = netdev_priv(netdev);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1699
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1700
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1701
	wol->supported = WAKE_UCAST | WAKE_MCAST |
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1702
	                 WAKE_BCAST | WAKE_MAGIC;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1703
	wol->wolopts = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1704
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1705
	/* this function will set ->supported = 0 and return 1 if wol is not
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1706
	 * supported by this hardware */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1707
	if (e1000_wol_exclusion(adapter, wol) ||
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1708
	    !device_can_wakeup(&adapter->pdev->dev))
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1709
		return;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1710
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1711
	/* apply any specific unsupported masks here */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1712
	switch (hw->device_id) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1713
	case E1000_DEV_ID_82546GB_QUAD_COPPER_KSP3:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1714
		/* KSP3 does not suppport UCAST wake-ups */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1715
		wol->supported &= ~WAKE_UCAST;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1716
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1717
		if (adapter->wol & E1000_WUFC_EX)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1718
			e_err(drv, "Interface does not support directed "
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1719
			      "(unicast) frame wake-up packets\n");
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1720
		break;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1721
	default:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1722
		break;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1723
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1724
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1725
	if (adapter->wol & E1000_WUFC_EX)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1726
		wol->wolopts |= WAKE_UCAST;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1727
	if (adapter->wol & E1000_WUFC_MC)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1728
		wol->wolopts |= WAKE_MCAST;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1729
	if (adapter->wol & E1000_WUFC_BC)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1730
		wol->wolopts |= WAKE_BCAST;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1731
	if (adapter->wol & E1000_WUFC_MAG)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1732
		wol->wolopts |= WAKE_MAGIC;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1733
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1734
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1735
static int e1000_set_wol(struct net_device *netdev, struct ethtool_wolinfo *wol)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1736
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1737
	struct e1000_adapter *adapter = netdev_priv(netdev);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1738
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1739
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1740
	if (wol->wolopts & (WAKE_PHY | WAKE_ARP | WAKE_MAGICSECURE))
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1741
		return -EOPNOTSUPP;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1742
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1743
	if (e1000_wol_exclusion(adapter, wol) ||
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1744
	    !device_can_wakeup(&adapter->pdev->dev))
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1745
		return wol->wolopts ? -EOPNOTSUPP : 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1746
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1747
	switch (hw->device_id) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1748
	case E1000_DEV_ID_82546GB_QUAD_COPPER_KSP3:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1749
		if (wol->wolopts & WAKE_UCAST) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1750
			e_err(drv, "Interface does not support directed "
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1751
			      "(unicast) frame wake-up packets\n");
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1752
			return -EOPNOTSUPP;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1753
		}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1754
		break;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1755
	default:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1756
		break;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1757
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1758
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1759
	/* these settings will always override what we currently have */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1760
	adapter->wol = 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1761
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1762
	if (wol->wolopts & WAKE_UCAST)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1763
		adapter->wol |= E1000_WUFC_EX;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1764
	if (wol->wolopts & WAKE_MCAST)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1765
		adapter->wol |= E1000_WUFC_MC;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1766
	if (wol->wolopts & WAKE_BCAST)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1767
		adapter->wol |= E1000_WUFC_BC;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1768
	if (wol->wolopts & WAKE_MAGIC)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1769
		adapter->wol |= E1000_WUFC_MAG;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1770
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1771
	device_set_wakeup_enable(&adapter->pdev->dev, adapter->wol);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1772
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1773
	return 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1774
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1775
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1776
static int e1000_set_phys_id(struct net_device *netdev,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1777
			     enum ethtool_phys_id_state state)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1778
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1779
	struct e1000_adapter *adapter = netdev_priv(netdev);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1780
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1781
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1782
	switch (state) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1783
	case ETHTOOL_ID_ACTIVE:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1784
		e1000_setup_led(hw);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1785
		return 2;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1786
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1787
	case ETHTOOL_ID_ON:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1788
		e1000_led_on(hw);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1789
		break;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1790
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1791
	case ETHTOOL_ID_OFF:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1792
		e1000_led_off(hw);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1793
		break;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1794
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1795
	case ETHTOOL_ID_INACTIVE:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1796
		e1000_cleanup_led(hw);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1797
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1798
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1799
	return 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1800
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1801
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1802
static int e1000_get_coalesce(struct net_device *netdev,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1803
			      struct ethtool_coalesce *ec)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1804
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1805
	struct e1000_adapter *adapter = netdev_priv(netdev);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1806
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1807
	if (adapter->hw.mac_type < e1000_82545)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1808
		return -EOPNOTSUPP;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1809
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1810
	if (adapter->itr_setting <= 4)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1811
		ec->rx_coalesce_usecs = adapter->itr_setting;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1812
	else
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1813
		ec->rx_coalesce_usecs = 1000000 / adapter->itr_setting;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1814
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1815
	return 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1816
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1817
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1818
static int e1000_set_coalesce(struct net_device *netdev,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1819
			      struct ethtool_coalesce *ec)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1820
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1821
	struct e1000_adapter *adapter = netdev_priv(netdev);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1822
	struct e1000_hw *hw = &adapter->hw;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1823
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1824
	if (hw->mac_type < e1000_82545)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1825
		return -EOPNOTSUPP;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1826
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1827
	if ((ec->rx_coalesce_usecs > E1000_MAX_ITR_USECS) ||
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1828
	    ((ec->rx_coalesce_usecs > 4) &&
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1829
	     (ec->rx_coalesce_usecs < E1000_MIN_ITR_USECS)) ||
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1830
	    (ec->rx_coalesce_usecs == 2))
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1831
		return -EINVAL;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1832
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1833
	if (ec->rx_coalesce_usecs == 4) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1834
		adapter->itr = adapter->itr_setting = 4;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1835
	} else if (ec->rx_coalesce_usecs <= 3) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1836
		adapter->itr = 20000;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1837
		adapter->itr_setting = ec->rx_coalesce_usecs;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1838
	} else {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1839
		adapter->itr = (1000000 / ec->rx_coalesce_usecs);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1840
		adapter->itr_setting = adapter->itr & ~3;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1841
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1842
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1843
	if (adapter->itr_setting != 0)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1844
		ew32(ITR, 1000000000 / (adapter->itr * 256));
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1845
	else
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1846
		ew32(ITR, 0);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1847
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1848
	return 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1849
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1850
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1851
static int e1000_nway_reset(struct net_device *netdev)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1852
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1853
	struct e1000_adapter *adapter = netdev_priv(netdev);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1854
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1855
	if (adapter->ecdev)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1856
		return -EBUSY;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1857
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1858
	if (netif_running(netdev))
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1859
		e1000_reinit_locked(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1860
	return 0;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1861
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1862
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1863
static void e1000_get_ethtool_stats(struct net_device *netdev,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1864
				    struct ethtool_stats *stats, u64 *data)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1865
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1866
	struct e1000_adapter *adapter = netdev_priv(netdev);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1867
	int i;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1868
	char *p = NULL;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1869
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1870
	e1000_update_stats(adapter);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1871
	for (i = 0; i < E1000_GLOBAL_STATS_LEN; i++) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1872
		switch (e1000_gstrings_stats[i].type) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1873
		case NETDEV_STATS:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1874
			p = (char *) netdev +
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1875
					e1000_gstrings_stats[i].stat_offset;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1876
			break;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1877
		case E1000_STATS:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1878
			p = (char *) adapter +
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1879
					e1000_gstrings_stats[i].stat_offset;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1880
			break;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1881
		}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1882
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1883
		data[i] = (e1000_gstrings_stats[i].sizeof_stat ==
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1884
			sizeof(u64)) ? *(u64 *)p : *(u32 *)p;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1885
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1886
/*	BUG_ON(i != E1000_STATS_LEN); */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1887
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1888
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1889
static void e1000_get_strings(struct net_device *netdev, u32 stringset,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1890
			      u8 *data)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1891
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1892
	u8 *p = data;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1893
	int i;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1894
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1895
	switch (stringset) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1896
	case ETH_SS_TEST:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1897
		memcpy(data, *e1000_gstrings_test,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1898
			sizeof(e1000_gstrings_test));
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1899
		break;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1900
	case ETH_SS_STATS:
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1901
		for (i = 0; i < E1000_GLOBAL_STATS_LEN; i++) {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1902
			memcpy(p, e1000_gstrings_stats[i].stat_string,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1903
			       ETH_GSTRING_LEN);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1904
			p += ETH_GSTRING_LEN;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1905
		}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1906
/*		BUG_ON(p - data != E1000_STATS_LEN * ETH_GSTRING_LEN); */
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1907
		break;
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1908
	}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1909
}
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1910
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1911
static const struct ethtool_ops e1000_ethtool_ops = {
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1912
	.get_settings           = e1000_get_settings,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1913
	.set_settings           = e1000_set_settings,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1914
	.get_drvinfo            = e1000_get_drvinfo,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1915
	.get_regs_len           = e1000_get_regs_len,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1916
	.get_regs               = e1000_get_regs,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1917
	.get_wol                = e1000_get_wol,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1918
	.set_wol                = e1000_set_wol,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1919
	.get_msglevel           = e1000_get_msglevel,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1920
	.set_msglevel           = e1000_set_msglevel,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1921
	.nway_reset             = e1000_nway_reset,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1922
	.get_link               = e1000_get_link,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1923
	.get_eeprom_len         = e1000_get_eeprom_len,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1924
	.get_eeprom             = e1000_get_eeprom,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1925
	.set_eeprom             = e1000_set_eeprom,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1926
	.get_ringparam          = e1000_get_ringparam,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1927
	.set_ringparam          = e1000_set_ringparam,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1928
	.get_pauseparam         = e1000_get_pauseparam,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1929
	.set_pauseparam         = e1000_set_pauseparam,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1930
	.get_rx_csum            = e1000_get_rx_csum,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1931
	.set_rx_csum            = e1000_set_rx_csum,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1932
	.get_tx_csum            = e1000_get_tx_csum,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1933
	.set_tx_csum            = e1000_set_tx_csum,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1934
	.set_sg                 = ethtool_op_set_sg,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1935
	.set_tso                = e1000_set_tso,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1936
	.self_test              = e1000_diag_test,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1937
	.get_strings            = e1000_get_strings,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1938
	.set_phys_id            = e1000_set_phys_id,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1939
	.get_ethtool_stats      = e1000_get_ethtool_stats,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1940
	.get_sset_count         = e1000_get_sset_count,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1941
	.get_coalesce           = e1000_get_coalesce,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1942
	.set_coalesce           = e1000_set_coalesce,
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1943
};
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1944
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1945
void e1000_set_ethtool_ops(struct net_device *netdev)
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1946
{
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1947
	SET_ETHTOOL_OPS(netdev, &e1000_ethtool_ops);
f7451c2c274f Added e1000 driver for kernel 3.0.
Florian Pose <fp@igh-essen.com>
parents:
diff changeset
  1948
}