blob: c29c5fb3aecf7cae7ba1a7d88077d3e17105ff62 [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
Francois Romieu07d3f512007-02-21 22:40:46 +01002 * r8169.c: RealTek 8169/8168/8101 ethernet driver.
3 *
4 * Copyright (c) 2002 ShuChen <shuchen@realtek.com.tw>
5 * Copyright (c) 2003 - 2007 Francois Romieu <romieu@fr.zoreil.com>
6 * Copyright (c) a lot of people too. Please respect their work.
7 *
8 * See MAINTAINERS file for support contact information.
Linus Torvalds1da177e2005-04-16 15:20:36 -07009 */
10
11#include <linux/module.h>
12#include <linux/moduleparam.h>
13#include <linux/pci.h>
14#include <linux/netdevice.h>
15#include <linux/etherdevice.h>
16#include <linux/delay.h>
17#include <linux/ethtool.h>
18#include <linux/mii.h>
19#include <linux/if_vlan.h>
20#include <linux/crc32.h>
21#include <linux/in.h>
22#include <linux/ip.h>
23#include <linux/tcp.h>
24#include <linux/init.h>
Alexey Dobriyana6b7a402011-06-06 10:43:46 +000025#include <linux/interrupt.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070026#include <linux/dma-mapping.h>
Rafael J. Wysockie1759442010-03-14 14:33:51 +000027#include <linux/pm_runtime.h>
françois romieubca03d52011-01-03 15:07:31 +000028#include <linux/firmware.h>
Stanislaw Gruszkaba04c7c2011-02-22 02:00:11 +000029#include <linux/pci-aspm.h>
Paul Gortmaker70c71602011-05-22 16:47:17 -040030#include <linux/prefetch.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070031
32#include <asm/io.h>
33#include <asm/irq.h>
34
Francois Romieu865c6522008-05-11 14:51:00 +020035#define RTL8169_VERSION "2.3LK-NAPI"
Linus Torvalds1da177e2005-04-16 15:20:36 -070036#define MODULENAME "r8169"
37#define PFX MODULENAME ": "
38
françois romieubca03d52011-01-03 15:07:31 +000039#define FIRMWARE_8168D_1 "rtl_nic/rtl8168d-1.fw"
40#define FIRMWARE_8168D_2 "rtl_nic/rtl8168d-2.fw"
hayeswang01dc7fe2011-03-21 01:50:28 +000041#define FIRMWARE_8168E_1 "rtl_nic/rtl8168e-1.fw"
42#define FIRMWARE_8168E_2 "rtl_nic/rtl8168e-2.fw"
Hayes Wang70090422011-07-06 15:58:06 +080043#define FIRMWARE_8168E_3 "rtl_nic/rtl8168e-3.fw"
Hayes Wangc2218922011-09-06 16:55:18 +080044#define FIRMWARE_8168F_1 "rtl_nic/rtl8168f-1.fw"
45#define FIRMWARE_8168F_2 "rtl_nic/rtl8168f-2.fw"
Hayes Wang5a5e4442011-02-22 17:26:21 +080046#define FIRMWARE_8105E_1 "rtl_nic/rtl8105e-1.fw"
Hayes Wang7e18dca2012-03-30 14:33:02 +080047#define FIRMWARE_8402_1 "rtl_nic/rtl8402-1.fw"
Hayes Wangb3d7b2f2012-03-30 14:48:06 +080048#define FIRMWARE_8411_1 "rtl_nic/rtl8411-1.fw"
Hayes Wang5598bfe2012-07-02 17:23:21 +080049#define FIRMWARE_8106E_1 "rtl_nic/rtl8106e-1.fw"
Hayes Wangc5583862012-07-02 17:23:22 +080050#define FIRMWARE_8168G_1 "rtl_nic/rtl8168g-1.fw"
françois romieubca03d52011-01-03 15:07:31 +000051
Linus Torvalds1da177e2005-04-16 15:20:36 -070052#ifdef RTL8169_DEBUG
53#define assert(expr) \
Francois Romieu5b0384f2006-08-16 16:00:01 +020054 if (!(expr)) { \
55 printk( "Assertion failed! %s,%s,%s,line=%d\n", \
Harvey Harrisonb39d66a2008-08-20 16:52:04 -070056 #expr,__FILE__,__func__,__LINE__); \
Francois Romieu5b0384f2006-08-16 16:00:01 +020057 }
Joe Perches06fa7352007-10-18 21:15:00 +020058#define dprintk(fmt, args...) \
59 do { printk(KERN_DEBUG PFX fmt, ## args); } while (0)
Linus Torvalds1da177e2005-04-16 15:20:36 -070060#else
61#define assert(expr) do {} while (0)
62#define dprintk(fmt, args...) do {} while (0)
63#endif /* RTL8169_DEBUG */
64
Stephen Hemmingerb57b7e52005-05-27 21:11:52 +020065#define R8169_MSG_DEFAULT \
Francois Romieuf0e837d2005-09-30 16:54:02 -070066 (NETIF_MSG_DRV | NETIF_MSG_PROBE | NETIF_MSG_IFUP | NETIF_MSG_IFDOWN)
Stephen Hemmingerb57b7e52005-05-27 21:11:52 +020067
Julien Ducourthial477206a2012-05-09 00:00:06 +020068#define TX_SLOTS_AVAIL(tp) \
69 (tp->dirty_tx + NUM_TX_DESC - tp->cur_tx)
70
71/* A skbuff with nr_frags needs nr_frags+1 entries in the tx queue */
72#define TX_FRAGS_READY_FOR(tp,nr_frags) \
73 (TX_SLOTS_AVAIL(tp) >= (nr_frags + 1))
Linus Torvalds1da177e2005-04-16 15:20:36 -070074
Linus Torvalds1da177e2005-04-16 15:20:36 -070075/* Maximum number of multicast addresses to filter (vs. Rx-all-multicast).
76 The RTL chips use a 64 element hash table based on the Ethernet CRC. */
Arjan van de Venf71e1302006-03-03 21:33:57 -050077static const int multicast_filter_limit = 32;
Linus Torvalds1da177e2005-04-16 15:20:36 -070078
Francois Romieu9c14cea2008-07-05 00:21:15 +020079#define MAX_READ_REQUEST_SHIFT 12
Linus Torvalds1da177e2005-04-16 15:20:36 -070080#define TX_DMA_BURST 6 /* Maximum PCI burst, '6' is 1024 */
Linus Torvalds1da177e2005-04-16 15:20:36 -070081#define SafeMtu 0x1c20 /* ... actually life sucks beyond ~7k */
82#define InterFrameGap 0x03 /* 3 means InterFrameGap = the shortest one */
83
84#define R8169_REGS_SIZE 256
85#define R8169_NAPI_WEIGHT 64
86#define NUM_TX_DESC 64 /* Number of Tx descriptor registers */
87#define NUM_RX_DESC 256 /* Number of Rx descriptor registers */
88#define RX_BUF_SIZE 1536 /* Rx Buffer size */
89#define R8169_TX_RING_BYTES (NUM_TX_DESC * sizeof(struct TxDesc))
90#define R8169_RX_RING_BYTES (NUM_RX_DESC * sizeof(struct RxDesc))
91
92#define RTL8169_TX_TIMEOUT (6*HZ)
93#define RTL8169_PHY_TIMEOUT (10*HZ)
94
françois romieuea8dbdd2009-03-15 01:10:50 +000095#define RTL_EEPROM_SIG cpu_to_le32(0x8129)
96#define RTL_EEPROM_SIG_MASK cpu_to_le32(0xffff)
Francois Romieue1564ec2008-10-16 22:46:13 +020097#define RTL_EEPROM_SIG_ADDR 0x0000
98
Linus Torvalds1da177e2005-04-16 15:20:36 -070099/* write/read MMIO register */
100#define RTL_W8(reg, val8) writeb ((val8), ioaddr + (reg))
101#define RTL_W16(reg, val16) writew ((val16), ioaddr + (reg))
102#define RTL_W32(reg, val32) writel ((val32), ioaddr + (reg))
103#define RTL_R8(reg) readb (ioaddr + (reg))
104#define RTL_R16(reg) readw (ioaddr + (reg))
Junchang Wang06f555f2010-05-30 02:26:07 +0000105#define RTL_R32(reg) readl (ioaddr + (reg))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700106
107enum mac_version {
Francois Romieu85bffe62011-04-27 08:22:39 +0200108 RTL_GIGA_MAC_VER_01 = 0,
109 RTL_GIGA_MAC_VER_02,
110 RTL_GIGA_MAC_VER_03,
111 RTL_GIGA_MAC_VER_04,
112 RTL_GIGA_MAC_VER_05,
113 RTL_GIGA_MAC_VER_06,
114 RTL_GIGA_MAC_VER_07,
115 RTL_GIGA_MAC_VER_08,
116 RTL_GIGA_MAC_VER_09,
117 RTL_GIGA_MAC_VER_10,
118 RTL_GIGA_MAC_VER_11,
119 RTL_GIGA_MAC_VER_12,
120 RTL_GIGA_MAC_VER_13,
121 RTL_GIGA_MAC_VER_14,
122 RTL_GIGA_MAC_VER_15,
123 RTL_GIGA_MAC_VER_16,
124 RTL_GIGA_MAC_VER_17,
125 RTL_GIGA_MAC_VER_18,
126 RTL_GIGA_MAC_VER_19,
127 RTL_GIGA_MAC_VER_20,
128 RTL_GIGA_MAC_VER_21,
129 RTL_GIGA_MAC_VER_22,
130 RTL_GIGA_MAC_VER_23,
131 RTL_GIGA_MAC_VER_24,
132 RTL_GIGA_MAC_VER_25,
133 RTL_GIGA_MAC_VER_26,
134 RTL_GIGA_MAC_VER_27,
135 RTL_GIGA_MAC_VER_28,
136 RTL_GIGA_MAC_VER_29,
137 RTL_GIGA_MAC_VER_30,
138 RTL_GIGA_MAC_VER_31,
139 RTL_GIGA_MAC_VER_32,
140 RTL_GIGA_MAC_VER_33,
Hayes Wang70090422011-07-06 15:58:06 +0800141 RTL_GIGA_MAC_VER_34,
Hayes Wangc2218922011-09-06 16:55:18 +0800142 RTL_GIGA_MAC_VER_35,
143 RTL_GIGA_MAC_VER_36,
Hayes Wang7e18dca2012-03-30 14:33:02 +0800144 RTL_GIGA_MAC_VER_37,
Hayes Wangb3d7b2f2012-03-30 14:48:06 +0800145 RTL_GIGA_MAC_VER_38,
Hayes Wang5598bfe2012-07-02 17:23:21 +0800146 RTL_GIGA_MAC_VER_39,
Hayes Wangc5583862012-07-02 17:23:22 +0800147 RTL_GIGA_MAC_VER_40,
148 RTL_GIGA_MAC_VER_41,
Francois Romieu85bffe62011-04-27 08:22:39 +0200149 RTL_GIGA_MAC_NONE = 0xff,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700150};
151
Francois Romieu2b7b4312011-04-18 22:53:24 -0700152enum rtl_tx_desc_version {
153 RTL_TD_0 = 0,
154 RTL_TD_1 = 1,
155};
156
Francois Romieud58d46b2011-05-03 16:38:29 +0200157#define JUMBO_1K ETH_DATA_LEN
158#define JUMBO_4K (4*1024 - ETH_HLEN - 2)
159#define JUMBO_6K (6*1024 - ETH_HLEN - 2)
160#define JUMBO_7K (7*1024 - ETH_HLEN - 2)
161#define JUMBO_9K (9*1024 - ETH_HLEN - 2)
162
163#define _R(NAME,TD,FW,SZ,B) { \
164 .name = NAME, \
165 .txd_version = TD, \
166 .fw_name = FW, \
167 .jumbo_max = SZ, \
168 .jumbo_tx_csum = B \
169}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700170
Jesper Juhl3c6bee12006-01-09 20:54:01 -0800171static const struct {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700172 const char *name;
Francois Romieu2b7b4312011-04-18 22:53:24 -0700173 enum rtl_tx_desc_version txd_version;
Francois Romieu85bffe62011-04-27 08:22:39 +0200174 const char *fw_name;
Francois Romieud58d46b2011-05-03 16:38:29 +0200175 u16 jumbo_max;
176 bool jumbo_tx_csum;
Francois Romieu85bffe62011-04-27 08:22:39 +0200177} rtl_chip_infos[] = {
178 /* PCI devices. */
179 [RTL_GIGA_MAC_VER_01] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200180 _R("RTL8169", RTL_TD_0, NULL, JUMBO_7K, true),
Francois Romieu85bffe62011-04-27 08:22:39 +0200181 [RTL_GIGA_MAC_VER_02] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200182 _R("RTL8169s", RTL_TD_0, NULL, JUMBO_7K, true),
Francois Romieu85bffe62011-04-27 08:22:39 +0200183 [RTL_GIGA_MAC_VER_03] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200184 _R("RTL8110s", RTL_TD_0, NULL, JUMBO_7K, true),
Francois Romieu85bffe62011-04-27 08:22:39 +0200185 [RTL_GIGA_MAC_VER_04] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200186 _R("RTL8169sb/8110sb", RTL_TD_0, NULL, JUMBO_7K, true),
Francois Romieu85bffe62011-04-27 08:22:39 +0200187 [RTL_GIGA_MAC_VER_05] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200188 _R("RTL8169sc/8110sc", RTL_TD_0, NULL, JUMBO_7K, true),
Francois Romieu85bffe62011-04-27 08:22:39 +0200189 [RTL_GIGA_MAC_VER_06] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200190 _R("RTL8169sc/8110sc", RTL_TD_0, NULL, JUMBO_7K, true),
Francois Romieu85bffe62011-04-27 08:22:39 +0200191 /* PCI-E devices. */
192 [RTL_GIGA_MAC_VER_07] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200193 _R("RTL8102e", RTL_TD_1, NULL, JUMBO_1K, true),
Francois Romieu85bffe62011-04-27 08:22:39 +0200194 [RTL_GIGA_MAC_VER_08] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200195 _R("RTL8102e", RTL_TD_1, NULL, JUMBO_1K, true),
Francois Romieu85bffe62011-04-27 08:22:39 +0200196 [RTL_GIGA_MAC_VER_09] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200197 _R("RTL8102e", RTL_TD_1, NULL, JUMBO_1K, true),
Francois Romieu85bffe62011-04-27 08:22:39 +0200198 [RTL_GIGA_MAC_VER_10] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200199 _R("RTL8101e", RTL_TD_0, NULL, JUMBO_1K, true),
Francois Romieu85bffe62011-04-27 08:22:39 +0200200 [RTL_GIGA_MAC_VER_11] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200201 _R("RTL8168b/8111b", RTL_TD_0, NULL, JUMBO_4K, false),
Francois Romieu85bffe62011-04-27 08:22:39 +0200202 [RTL_GIGA_MAC_VER_12] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200203 _R("RTL8168b/8111b", RTL_TD_0, NULL, JUMBO_4K, false),
Francois Romieu85bffe62011-04-27 08:22:39 +0200204 [RTL_GIGA_MAC_VER_13] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200205 _R("RTL8101e", RTL_TD_0, NULL, JUMBO_1K, true),
Francois Romieu85bffe62011-04-27 08:22:39 +0200206 [RTL_GIGA_MAC_VER_14] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200207 _R("RTL8100e", RTL_TD_0, NULL, JUMBO_1K, true),
Francois Romieu85bffe62011-04-27 08:22:39 +0200208 [RTL_GIGA_MAC_VER_15] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200209 _R("RTL8100e", RTL_TD_0, NULL, JUMBO_1K, true),
Francois Romieu85bffe62011-04-27 08:22:39 +0200210 [RTL_GIGA_MAC_VER_16] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200211 _R("RTL8101e", RTL_TD_0, NULL, JUMBO_1K, true),
Francois Romieu85bffe62011-04-27 08:22:39 +0200212 [RTL_GIGA_MAC_VER_17] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200213 _R("RTL8168b/8111b", RTL_TD_1, NULL, JUMBO_4K, false),
Francois Romieu85bffe62011-04-27 08:22:39 +0200214 [RTL_GIGA_MAC_VER_18] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200215 _R("RTL8168cp/8111cp", RTL_TD_1, NULL, JUMBO_6K, false),
Francois Romieu85bffe62011-04-27 08:22:39 +0200216 [RTL_GIGA_MAC_VER_19] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200217 _R("RTL8168c/8111c", RTL_TD_1, NULL, JUMBO_6K, false),
Francois Romieu85bffe62011-04-27 08:22:39 +0200218 [RTL_GIGA_MAC_VER_20] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200219 _R("RTL8168c/8111c", RTL_TD_1, NULL, JUMBO_6K, false),
Francois Romieu85bffe62011-04-27 08:22:39 +0200220 [RTL_GIGA_MAC_VER_21] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200221 _R("RTL8168c/8111c", RTL_TD_1, NULL, JUMBO_6K, false),
Francois Romieu85bffe62011-04-27 08:22:39 +0200222 [RTL_GIGA_MAC_VER_22] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200223 _R("RTL8168c/8111c", RTL_TD_1, NULL, JUMBO_6K, false),
Francois Romieu85bffe62011-04-27 08:22:39 +0200224 [RTL_GIGA_MAC_VER_23] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200225 _R("RTL8168cp/8111cp", RTL_TD_1, NULL, JUMBO_6K, false),
Francois Romieu85bffe62011-04-27 08:22:39 +0200226 [RTL_GIGA_MAC_VER_24] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200227 _R("RTL8168cp/8111cp", RTL_TD_1, NULL, JUMBO_6K, false),
Francois Romieu85bffe62011-04-27 08:22:39 +0200228 [RTL_GIGA_MAC_VER_25] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200229 _R("RTL8168d/8111d", RTL_TD_1, FIRMWARE_8168D_1,
230 JUMBO_9K, false),
Francois Romieu85bffe62011-04-27 08:22:39 +0200231 [RTL_GIGA_MAC_VER_26] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200232 _R("RTL8168d/8111d", RTL_TD_1, FIRMWARE_8168D_2,
233 JUMBO_9K, false),
Francois Romieu85bffe62011-04-27 08:22:39 +0200234 [RTL_GIGA_MAC_VER_27] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200235 _R("RTL8168dp/8111dp", RTL_TD_1, NULL, JUMBO_9K, false),
Francois Romieu85bffe62011-04-27 08:22:39 +0200236 [RTL_GIGA_MAC_VER_28] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200237 _R("RTL8168dp/8111dp", RTL_TD_1, NULL, JUMBO_9K, false),
Francois Romieu85bffe62011-04-27 08:22:39 +0200238 [RTL_GIGA_MAC_VER_29] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200239 _R("RTL8105e", RTL_TD_1, FIRMWARE_8105E_1,
240 JUMBO_1K, true),
Francois Romieu85bffe62011-04-27 08:22:39 +0200241 [RTL_GIGA_MAC_VER_30] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200242 _R("RTL8105e", RTL_TD_1, FIRMWARE_8105E_1,
243 JUMBO_1K, true),
Francois Romieu85bffe62011-04-27 08:22:39 +0200244 [RTL_GIGA_MAC_VER_31] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200245 _R("RTL8168dp/8111dp", RTL_TD_1, NULL, JUMBO_9K, false),
Francois Romieu85bffe62011-04-27 08:22:39 +0200246 [RTL_GIGA_MAC_VER_32] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200247 _R("RTL8168e/8111e", RTL_TD_1, FIRMWARE_8168E_1,
248 JUMBO_9K, false),
Francois Romieu85bffe62011-04-27 08:22:39 +0200249 [RTL_GIGA_MAC_VER_33] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200250 _R("RTL8168e/8111e", RTL_TD_1, FIRMWARE_8168E_2,
251 JUMBO_9K, false),
Hayes Wang70090422011-07-06 15:58:06 +0800252 [RTL_GIGA_MAC_VER_34] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200253 _R("RTL8168evl/8111evl",RTL_TD_1, FIRMWARE_8168E_3,
254 JUMBO_9K, false),
Hayes Wangc2218922011-09-06 16:55:18 +0800255 [RTL_GIGA_MAC_VER_35] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200256 _R("RTL8168f/8111f", RTL_TD_1, FIRMWARE_8168F_1,
257 JUMBO_9K, false),
Hayes Wangc2218922011-09-06 16:55:18 +0800258 [RTL_GIGA_MAC_VER_36] =
Francois Romieud58d46b2011-05-03 16:38:29 +0200259 _R("RTL8168f/8111f", RTL_TD_1, FIRMWARE_8168F_2,
260 JUMBO_9K, false),
Hayes Wang7e18dca2012-03-30 14:33:02 +0800261 [RTL_GIGA_MAC_VER_37] =
262 _R("RTL8402", RTL_TD_1, FIRMWARE_8402_1,
263 JUMBO_1K, true),
Hayes Wangb3d7b2f2012-03-30 14:48:06 +0800264 [RTL_GIGA_MAC_VER_38] =
265 _R("RTL8411", RTL_TD_1, FIRMWARE_8411_1,
266 JUMBO_9K, false),
Hayes Wang5598bfe2012-07-02 17:23:21 +0800267 [RTL_GIGA_MAC_VER_39] =
268 _R("RTL8106e", RTL_TD_1, FIRMWARE_8106E_1,
269 JUMBO_1K, true),
Hayes Wangc5583862012-07-02 17:23:22 +0800270 [RTL_GIGA_MAC_VER_40] =
271 _R("RTL8168g/8111g", RTL_TD_1, FIRMWARE_8168G_1,
272 JUMBO_9K, false),
273 [RTL_GIGA_MAC_VER_41] =
274 _R("RTL8168g/8111g", RTL_TD_1, NULL, JUMBO_9K, false),
Linus Torvalds1da177e2005-04-16 15:20:36 -0700275};
276#undef _R
277
Francois Romieubcf0bf92006-07-26 23:14:13 +0200278enum cfg_version {
279 RTL_CFG_0 = 0x00,
280 RTL_CFG_1,
281 RTL_CFG_2
282};
283
Alexey Dobriyana3aa1882010-01-07 11:58:11 +0000284static DEFINE_PCI_DEVICE_TABLE(rtl8169_pci_tbl) = {
Francois Romieubcf0bf92006-07-26 23:14:13 +0200285 { PCI_DEVICE(PCI_VENDOR_ID_REALTEK, 0x8129), 0, 0, RTL_CFG_0 },
Francois Romieud2eed8c2006-08-31 22:01:07 +0200286 { PCI_DEVICE(PCI_VENDOR_ID_REALTEK, 0x8136), 0, 0, RTL_CFG_2 },
Francois Romieud81bf552006-09-20 21:31:20 +0200287 { PCI_DEVICE(PCI_VENDOR_ID_REALTEK, 0x8167), 0, 0, RTL_CFG_0 },
Francois Romieu07ce4062007-02-23 23:36:39 +0100288 { PCI_DEVICE(PCI_VENDOR_ID_REALTEK, 0x8168), 0, 0, RTL_CFG_1 },
Francois Romieubcf0bf92006-07-26 23:14:13 +0200289 { PCI_DEVICE(PCI_VENDOR_ID_REALTEK, 0x8169), 0, 0, RTL_CFG_0 },
290 { PCI_DEVICE(PCI_VENDOR_ID_DLINK, 0x4300), 0, 0, RTL_CFG_0 },
Lennart Sorensen93a3aa22011-07-28 13:18:11 +0000291 { PCI_DEVICE(PCI_VENDOR_ID_DLINK, 0x4302), 0, 0, RTL_CFG_0 },
Francois Romieubc1660b2007-10-12 23:58:09 +0200292 { PCI_DEVICE(PCI_VENDOR_ID_AT, 0xc107), 0, 0, RTL_CFG_0 },
Francois Romieubcf0bf92006-07-26 23:14:13 +0200293 { PCI_DEVICE(0x16ec, 0x0116), 0, 0, RTL_CFG_0 },
294 { PCI_VENDOR_ID_LINKSYS, 0x1032,
295 PCI_ANY_ID, 0x0024, 0, 0, RTL_CFG_0 },
Ciaran McCreesh11d2e282007-11-01 22:48:15 +0100296 { 0x0001, 0x8168,
297 PCI_ANY_ID, 0x2410, 0, 0, RTL_CFG_2 },
Linus Torvalds1da177e2005-04-16 15:20:36 -0700298 {0,},
299};
300
301MODULE_DEVICE_TABLE(pci, rtl8169_pci_tbl);
302
Eric Dumazet6f0333b2010-10-11 11:17:47 +0000303static int rx_buf_sz = 16383;
David S. Miller4300e8c2010-03-26 10:23:30 -0700304static int use_dac;
Stephen Hemmingerb57b7e52005-05-27 21:11:52 +0200305static struct {
306 u32 msg_enable;
307} debug = { -1 };
Linus Torvalds1da177e2005-04-16 15:20:36 -0700308
Francois Romieu07d3f512007-02-21 22:40:46 +0100309enum rtl_registers {
310 MAC0 = 0, /* Ethernet hardware address. */
Francois Romieu773d2022007-01-31 23:47:43 +0100311 MAC4 = 4,
Francois Romieu07d3f512007-02-21 22:40:46 +0100312 MAR0 = 8, /* Multicast filter. */
313 CounterAddrLow = 0x10,
314 CounterAddrHigh = 0x14,
315 TxDescStartAddrLow = 0x20,
316 TxDescStartAddrHigh = 0x24,
317 TxHDescStartAddrLow = 0x28,
318 TxHDescStartAddrHigh = 0x2c,
319 FLASH = 0x30,
320 ERSR = 0x36,
321 ChipCmd = 0x37,
322 TxPoll = 0x38,
323 IntrMask = 0x3c,
324 IntrStatus = 0x3e,
Francois Romieu2b7b4312011-04-18 22:53:24 -0700325
Hayes Wang4f6b00e52011-07-06 15:58:02 +0800326 TxConfig = 0x40,
327#define TXCFG_AUTO_FIFO (1 << 7) /* 8111e-vl */
328#define TXCFG_EMPTY (1 << 11) /* 8111e-vl */
329
330 RxConfig = 0x44,
331#define RX128_INT_EN (1 << 15) /* 8111c and later */
332#define RX_MULTI_EN (1 << 14) /* 8111c only */
333#define RXCFG_FIFO_SHIFT 13
334 /* No threshold before first PCI xfer */
335#define RX_FIFO_THRESH (7 << RXCFG_FIFO_SHIFT)
336#define RXCFG_DMA_SHIFT 8
337 /* Unlimited maximum PCI burst. */
338#define RX_DMA_BURST (7 << RXCFG_DMA_SHIFT)
Francois Romieu2b7b4312011-04-18 22:53:24 -0700339
Francois Romieu07d3f512007-02-21 22:40:46 +0100340 RxMissed = 0x4c,
341 Cfg9346 = 0x50,
342 Config0 = 0x51,
343 Config1 = 0x52,
344 Config2 = 0x53,
Francois Romieud387b422012-04-17 11:12:01 +0200345#define PME_SIGNAL (1 << 5) /* 8168c and later */
346
Francois Romieu07d3f512007-02-21 22:40:46 +0100347 Config3 = 0x54,
348 Config4 = 0x55,
349 Config5 = 0x56,
350 MultiIntr = 0x5c,
351 PHYAR = 0x60,
Francois Romieu07d3f512007-02-21 22:40:46 +0100352 PHYstatus = 0x6c,
353 RxMaxSize = 0xda,
354 CPlusCmd = 0xe0,
355 IntrMitigate = 0xe2,
356 RxDescAddrLow = 0xe4,
357 RxDescAddrHigh = 0xe8,
françois romieuf0298f82011-01-03 15:07:42 +0000358 EarlyTxThres = 0xec, /* 8169. Unit of 32 bytes. */
359
360#define NoEarlyTx 0x3f /* Max value : no early transmit. */
361
362 MaxTxPacketSize = 0xec, /* 8101/8168. Unit of 128 bytes. */
363
364#define TxPacketMax (8064 >> 7)
Hayes Wang3090bd92011-09-06 16:55:15 +0800365#define EarlySize 0x27
françois romieuf0298f82011-01-03 15:07:42 +0000366
Francois Romieu07d3f512007-02-21 22:40:46 +0100367 FuncEvent = 0xf0,
368 FuncEventMask = 0xf4,
369 FuncPresetState = 0xf8,
370 FuncForceEvent = 0xfc,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700371};
372
Francois Romieuf162a5d2008-06-01 22:37:49 +0200373enum rtl8110_registers {
374 TBICSR = 0x64,
375 TBI_ANAR = 0x68,
376 TBI_LPAR = 0x6a,
377};
378
379enum rtl8168_8101_registers {
380 CSIDR = 0x64,
381 CSIAR = 0x68,
382#define CSIAR_FLAG 0x80000000
383#define CSIAR_WRITE_CMD 0x80000000
384#define CSIAR_BYTE_ENABLE 0x0f
385#define CSIAR_BYTE_ENABLE_SHIFT 12
386#define CSIAR_ADDR_MASK 0x0fff
Hayes Wang7e18dca2012-03-30 14:33:02 +0800387#define CSIAR_FUNC_CARD 0x00000000
388#define CSIAR_FUNC_SDIO 0x00010000
389#define CSIAR_FUNC_NIC 0x00020000
françois romieu065c27c2011-01-03 15:08:12 +0000390 PMCH = 0x6f,
Francois Romieuf162a5d2008-06-01 22:37:49 +0200391 EPHYAR = 0x80,
392#define EPHYAR_FLAG 0x80000000
393#define EPHYAR_WRITE_CMD 0x80000000
394#define EPHYAR_REG_MASK 0x1f
395#define EPHYAR_REG_SHIFT 16
396#define EPHYAR_DATA_MASK 0xffff
Hayes Wang5a5e4442011-02-22 17:26:21 +0800397 DLLPR = 0xd0,
Hayes Wang4f6b00e52011-07-06 15:58:02 +0800398#define PFM_EN (1 << 6)
Francois Romieuf162a5d2008-06-01 22:37:49 +0200399 DBG_REG = 0xd1,
400#define FIX_NAK_1 (1 << 4)
401#define FIX_NAK_2 (1 << 3)
Hayes Wang5a5e4442011-02-22 17:26:21 +0800402 TWSI = 0xd2,
403 MCU = 0xd3,
Hayes Wang4f6b00e52011-07-06 15:58:02 +0800404#define NOW_IS_OOB (1 << 7)
Hayes Wangc5583862012-07-02 17:23:22 +0800405#define TX_EMPTY (1 << 5)
406#define RX_EMPTY (1 << 4)
407#define RXTX_EMPTY (TX_EMPTY | RX_EMPTY)
Hayes Wang5a5e4442011-02-22 17:26:21 +0800408#define EN_NDP (1 << 3)
409#define EN_OOB_RESET (1 << 2)
Hayes Wangc5583862012-07-02 17:23:22 +0800410#define LINK_LIST_RDY (1 << 1)
françois romieudaf9df62009-10-07 12:44:20 +0000411 EFUSEAR = 0xdc,
412#define EFUSEAR_FLAG 0x80000000
413#define EFUSEAR_WRITE_CMD 0x80000000
414#define EFUSEAR_READ_CMD 0x00000000
415#define EFUSEAR_REG_MASK 0x03ff
416#define EFUSEAR_REG_SHIFT 8
417#define EFUSEAR_DATA_MASK 0xff
Francois Romieuf162a5d2008-06-01 22:37:49 +0200418};
419
françois romieuc0e45c12011-01-03 15:08:04 +0000420enum rtl8168_registers {
Hayes Wang4f6b00e52011-07-06 15:58:02 +0800421 LED_FREQ = 0x1a,
422 EEE_LED = 0x1b,
françois romieub646d902011-01-03 15:08:21 +0000423 ERIDR = 0x70,
424 ERIAR = 0x74,
425#define ERIAR_FLAG 0x80000000
426#define ERIAR_WRITE_CMD 0x80000000
427#define ERIAR_READ_CMD 0x00000000
428#define ERIAR_ADDR_BYTE_ALIGN 4
françois romieub646d902011-01-03 15:08:21 +0000429#define ERIAR_TYPE_SHIFT 16
Hayes Wang4f6b00e52011-07-06 15:58:02 +0800430#define ERIAR_EXGMAC (0x00 << ERIAR_TYPE_SHIFT)
431#define ERIAR_MSIX (0x01 << ERIAR_TYPE_SHIFT)
432#define ERIAR_ASF (0x02 << ERIAR_TYPE_SHIFT)
433#define ERIAR_MASK_SHIFT 12
434#define ERIAR_MASK_0001 (0x1 << ERIAR_MASK_SHIFT)
435#define ERIAR_MASK_0011 (0x3 << ERIAR_MASK_SHIFT)
Hayes Wangc5583862012-07-02 17:23:22 +0800436#define ERIAR_MASK_0101 (0x5 << ERIAR_MASK_SHIFT)
Hayes Wang4f6b00e52011-07-06 15:58:02 +0800437#define ERIAR_MASK_1111 (0xf << ERIAR_MASK_SHIFT)
françois romieuc0e45c12011-01-03 15:08:04 +0000438 EPHY_RXER_NUM = 0x7c,
439 OCPDR = 0xb0, /* OCP GPHY access */
440#define OCPDR_WRITE_CMD 0x80000000
441#define OCPDR_READ_CMD 0x00000000
442#define OCPDR_REG_MASK 0x7f
443#define OCPDR_GPHY_REG_SHIFT 16
444#define OCPDR_DATA_MASK 0xffff
445 OCPAR = 0xb4,
446#define OCPAR_FLAG 0x80000000
447#define OCPAR_GPHY_WRITE_CMD 0x8000f060
448#define OCPAR_GPHY_READ_CMD 0x0000f060
Hayes Wangc5583862012-07-02 17:23:22 +0800449 GPHY_OCP = 0xb8,
hayeswang01dc7fe2011-03-21 01:50:28 +0000450 RDSAR1 = 0xd0, /* 8168c only. Undocumented on 8168dp */
451 MISC = 0xf0, /* 8168e only. */
Francois Romieucecb5fd2011-04-01 10:21:07 +0200452#define TXPLA_RST (1 << 29)
Hayes Wang5598bfe2012-07-02 17:23:21 +0800453#define DISABLE_LAN_EN (1 << 23) /* Enable GPIO pin */
Hayes Wang4f6b00e52011-07-06 15:58:02 +0800454#define PWM_EN (1 << 22)
Hayes Wangc5583862012-07-02 17:23:22 +0800455#define RXDV_GATED_EN (1 << 19)
Hayes Wang5598bfe2012-07-02 17:23:21 +0800456#define EARLY_TALLY_EN (1 << 16)
françois romieuc0e45c12011-01-03 15:08:04 +0000457};
458
Francois Romieu07d3f512007-02-21 22:40:46 +0100459enum rtl_register_content {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700460 /* InterruptStatusBits */
Francois Romieu07d3f512007-02-21 22:40:46 +0100461 SYSErr = 0x8000,
462 PCSTimeout = 0x4000,
463 SWInt = 0x0100,
464 TxDescUnavail = 0x0080,
465 RxFIFOOver = 0x0040,
466 LinkChg = 0x0020,
467 RxOverflow = 0x0010,
468 TxErr = 0x0008,
469 TxOK = 0x0004,
470 RxErr = 0x0002,
471 RxOK = 0x0001,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700472
473 /* RxStatusDesc */
David S. Miller8decf862011-09-22 03:23:13 -0400474 RxBOVF = (1 << 24),
Francois Romieu9dccf612006-05-14 12:31:17 +0200475 RxFOVF = (1 << 23),
476 RxRWT = (1 << 22),
477 RxRES = (1 << 21),
478 RxRUNT = (1 << 20),
479 RxCRC = (1 << 19),
Linus Torvalds1da177e2005-04-16 15:20:36 -0700480
481 /* ChipCmdBits */
Hayes Wang4f6b00e52011-07-06 15:58:02 +0800482 StopReq = 0x80,
Francois Romieu07d3f512007-02-21 22:40:46 +0100483 CmdReset = 0x10,
484 CmdRxEnb = 0x08,
485 CmdTxEnb = 0x04,
486 RxBufEmpty = 0x01,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700487
Francois Romieu275391a2007-02-23 23:50:28 +0100488 /* TXPoll register p.5 */
489 HPQ = 0x80, /* Poll cmd on the high prio queue */
490 NPQ = 0x40, /* Poll cmd on the low prio queue */
491 FSWInt = 0x01, /* Forced software interrupt */
492
Linus Torvalds1da177e2005-04-16 15:20:36 -0700493 /* Cfg9346Bits */
Francois Romieu07d3f512007-02-21 22:40:46 +0100494 Cfg9346_Lock = 0x00,
495 Cfg9346_Unlock = 0xc0,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700496
497 /* rx_mode_bits */
Francois Romieu07d3f512007-02-21 22:40:46 +0100498 AcceptErr = 0x20,
499 AcceptRunt = 0x10,
500 AcceptBroadcast = 0x08,
501 AcceptMulticast = 0x04,
502 AcceptMyPhys = 0x02,
503 AcceptAllPhys = 0x01,
Francois Romieu1687b562011-07-19 17:21:29 +0200504#define RX_CONFIG_ACCEPT_MASK 0x3f
Linus Torvalds1da177e2005-04-16 15:20:36 -0700505
Linus Torvalds1da177e2005-04-16 15:20:36 -0700506 /* TxConfigBits */
507 TxInterFrameGapShift = 24,
508 TxDMAShift = 8, /* DMA burst value (0-7) is shift this many bits */
509
Francois Romieu5d06a992006-02-23 00:47:58 +0100510 /* Config1 register p.24 */
Francois Romieuf162a5d2008-06-01 22:37:49 +0200511 LEDS1 = (1 << 7),
512 LEDS0 = (1 << 6),
Francois Romieuf162a5d2008-06-01 22:37:49 +0200513 Speed_down = (1 << 4),
514 MEMMAP = (1 << 3),
515 IOMAP = (1 << 2),
516 VPD = (1 << 1),
Francois Romieu5d06a992006-02-23 00:47:58 +0100517 PMEnable = (1 << 0), /* Power Management Enable */
518
Francois Romieu6dccd162007-02-13 23:38:05 +0100519 /* Config2 register p. 25 */
françois romieu2ca6cf02011-12-15 08:37:43 +0000520 MSIEnable = (1 << 5), /* 8169 only. Reserved in the 8168. */
Francois Romieu6dccd162007-02-13 23:38:05 +0100521 PCI_Clock_66MHz = 0x01,
522 PCI_Clock_33MHz = 0x00,
523
Francois Romieu61a4dcc2006-02-23 00:55:25 +0100524 /* Config3 register p.25 */
525 MagicPacket = (1 << 5), /* Wake up when receives a Magic Packet */
526 LinkUp = (1 << 4), /* Wake up when the cable connection is re-established */
Francois Romieud58d46b2011-05-03 16:38:29 +0200527 Jumbo_En0 = (1 << 2), /* 8168 only. Reserved in the 8168b */
Francois Romieuf162a5d2008-06-01 22:37:49 +0200528 Beacon_en = (1 << 0), /* 8168 only. Reserved in the 8168b */
Francois Romieu61a4dcc2006-02-23 00:55:25 +0100529
Francois Romieud58d46b2011-05-03 16:38:29 +0200530 /* Config4 register */
531 Jumbo_En1 = (1 << 1), /* 8168 only. Reserved in the 8168b */
532
Francois Romieu5d06a992006-02-23 00:47:58 +0100533 /* Config5 register p.27 */
Francois Romieu61a4dcc2006-02-23 00:55:25 +0100534 BWF = (1 << 6), /* Accept Broadcast wakeup frame */
535 MWF = (1 << 5), /* Accept Multicast wakeup frame */
536 UWF = (1 << 4), /* Accept Unicast wakeup frame */
Francois Romieucecb5fd2011-04-01 10:21:07 +0200537 Spi_en = (1 << 3),
Francois Romieu61a4dcc2006-02-23 00:55:25 +0100538 LanWake = (1 << 1), /* LanWake enable/disable */
Francois Romieu5d06a992006-02-23 00:47:58 +0100539 PMEStatus = (1 << 0), /* PME status can be reset by PCI RST# */
540
Linus Torvalds1da177e2005-04-16 15:20:36 -0700541 /* TBICSR p.28 */
542 TBIReset = 0x80000000,
543 TBILoopback = 0x40000000,
544 TBINwEnable = 0x20000000,
545 TBINwRestart = 0x10000000,
546 TBILinkOk = 0x02000000,
547 TBINwComplete = 0x01000000,
548
549 /* CPlusCmd p.31 */
Francois Romieuf162a5d2008-06-01 22:37:49 +0200550 EnableBist = (1 << 15), // 8168 8101
551 Mac_dbgo_oe = (1 << 14), // 8168 8101
552 Normal_mode = (1 << 13), // unused
553 Force_half_dup = (1 << 12), // 8168 8101
554 Force_rxflow_en = (1 << 11), // 8168 8101
555 Force_txflow_en = (1 << 10), // 8168 8101
556 Cxpl_dbg_sel = (1 << 9), // 8168 8101
557 ASF = (1 << 8), // 8168 8101
558 PktCntrDisable = (1 << 7), // 8168 8101
559 Mac_dbgo_sel = 0x001c, // 8168
Linus Torvalds1da177e2005-04-16 15:20:36 -0700560 RxVlan = (1 << 6),
561 RxChkSum = (1 << 5),
562 PCIDAC = (1 << 4),
563 PCIMulRW = (1 << 3),
Francois Romieu0e485152007-02-20 00:00:26 +0100564 INTT_0 = 0x0000, // 8168
565 INTT_1 = 0x0001, // 8168
566 INTT_2 = 0x0002, // 8168
567 INTT_3 = 0x0003, // 8168
Linus Torvalds1da177e2005-04-16 15:20:36 -0700568
569 /* rtl8169_PHYstatus */
Francois Romieu07d3f512007-02-21 22:40:46 +0100570 TBI_Enable = 0x80,
571 TxFlowCtrl = 0x40,
572 RxFlowCtrl = 0x20,
573 _1000bpsF = 0x10,
574 _100bps = 0x08,
575 _10bps = 0x04,
576 LinkStatus = 0x02,
577 FullDup = 0x01,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700578
Linus Torvalds1da177e2005-04-16 15:20:36 -0700579 /* _TBICSRBit */
Francois Romieu07d3f512007-02-21 22:40:46 +0100580 TBILinkOK = 0x02000000,
Stephen Hemmingerd4a3a0f2005-05-27 21:11:56 +0200581
582 /* DumpCounterCommand */
Francois Romieu07d3f512007-02-21 22:40:46 +0100583 CounterDump = 0x8,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700584};
585
Francois Romieu2b7b4312011-04-18 22:53:24 -0700586enum rtl_desc_bit {
587 /* First doubleword. */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700588 DescOwn = (1 << 31), /* Descriptor is owned by NIC */
589 RingEnd = (1 << 30), /* End of descriptor ring */
590 FirstFrag = (1 << 29), /* First segment of a packet */
591 LastFrag = (1 << 28), /* Final segment of a packet */
Francois Romieu2b7b4312011-04-18 22:53:24 -0700592};
Linus Torvalds1da177e2005-04-16 15:20:36 -0700593
Francois Romieu2b7b4312011-04-18 22:53:24 -0700594/* Generic case. */
595enum rtl_tx_desc_bit {
596 /* First doubleword. */
597 TD_LSO = (1 << 27), /* Large Send Offload */
598#define TD_MSS_MAX 0x07ffu /* MSS value */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700599
Francois Romieu2b7b4312011-04-18 22:53:24 -0700600 /* Second doubleword. */
601 TxVlanTag = (1 << 17), /* Add VLAN tag */
602};
603
604/* 8169, 8168b and 810x except 8102e. */
605enum rtl_tx_desc_bit_0 {
606 /* First doubleword. */
607#define TD0_MSS_SHIFT 16 /* MSS position (11 bits) */
608 TD0_TCP_CS = (1 << 16), /* Calculate TCP/IP checksum */
609 TD0_UDP_CS = (1 << 17), /* Calculate UDP/IP checksum */
610 TD0_IP_CS = (1 << 18), /* Calculate IP checksum */
611};
612
613/* 8102e, 8168c and beyond. */
614enum rtl_tx_desc_bit_1 {
615 /* Second doubleword. */
616#define TD1_MSS_SHIFT 18 /* MSS position (11 bits) */
617 TD1_IP_CS = (1 << 29), /* Calculate IP checksum */
618 TD1_TCP_CS = (1 << 30), /* Calculate TCP/IP checksum */
619 TD1_UDP_CS = (1 << 31), /* Calculate UDP/IP checksum */
620};
621
622static const struct rtl_tx_desc_info {
623 struct {
624 u32 udp;
625 u32 tcp;
626 } checksum;
627 u16 mss_shift;
628 u16 opts_offset;
629} tx_desc_info [] = {
630 [RTL_TD_0] = {
631 .checksum = {
632 .udp = TD0_IP_CS | TD0_UDP_CS,
633 .tcp = TD0_IP_CS | TD0_TCP_CS
634 },
635 .mss_shift = TD0_MSS_SHIFT,
636 .opts_offset = 0
637 },
638 [RTL_TD_1] = {
639 .checksum = {
640 .udp = TD1_IP_CS | TD1_UDP_CS,
641 .tcp = TD1_IP_CS | TD1_TCP_CS
642 },
643 .mss_shift = TD1_MSS_SHIFT,
644 .opts_offset = 1
645 }
646};
647
648enum rtl_rx_desc_bit {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700649 /* Rx private */
650 PID1 = (1 << 18), /* Protocol ID bit 1/2 */
651 PID0 = (1 << 17), /* Protocol ID bit 2/2 */
652
653#define RxProtoUDP (PID1)
654#define RxProtoTCP (PID0)
655#define RxProtoIP (PID1 | PID0)
656#define RxProtoMask RxProtoIP
657
658 IPFail = (1 << 16), /* IP checksum failed */
659 UDPFail = (1 << 15), /* UDP/IP checksum failed */
660 TCPFail = (1 << 14), /* TCP/IP checksum failed */
661 RxVlanTag = (1 << 16), /* VLAN tag available */
662};
663
664#define RsvdMask 0x3fffc000
665
666struct TxDesc {
Rolf Eike Beer6cccd6e2007-05-21 22:11:04 +0200667 __le32 opts1;
668 __le32 opts2;
669 __le64 addr;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700670};
671
672struct RxDesc {
Rolf Eike Beer6cccd6e2007-05-21 22:11:04 +0200673 __le32 opts1;
674 __le32 opts2;
675 __le64 addr;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700676};
677
678struct ring_info {
679 struct sk_buff *skb;
680 u32 len;
681 u8 __pad[sizeof(void *) - sizeof(u32)];
682};
683
Francois Romieuf23e7fd2007-10-04 22:36:14 +0200684enum features {
Francois Romieuccdffb92008-07-26 14:26:06 +0200685 RTL_FEATURE_WOL = (1 << 0),
686 RTL_FEATURE_MSI = (1 << 1),
687 RTL_FEATURE_GMII = (1 << 2),
Francois Romieuf23e7fd2007-10-04 22:36:14 +0200688};
689
Ivan Vecera355423d2009-02-06 21:49:57 -0800690struct rtl8169_counters {
691 __le64 tx_packets;
692 __le64 rx_packets;
693 __le64 tx_errors;
694 __le32 rx_errors;
695 __le16 rx_missed;
696 __le16 align_errors;
697 __le32 tx_one_collision;
698 __le32 tx_multi_collision;
699 __le64 rx_unicast;
700 __le64 rx_broadcast;
701 __le32 rx_multicast;
702 __le16 tx_aborted;
703 __le16 tx_underun;
704};
705
Francois Romieuda78dbf2012-01-26 14:18:23 +0100706enum rtl_flag {
Francois Romieu6c4a70c2012-01-31 10:56:44 +0100707 RTL_FLAG_TASK_ENABLED,
Francois Romieuda78dbf2012-01-26 14:18:23 +0100708 RTL_FLAG_TASK_SLOW_PENDING,
709 RTL_FLAG_TASK_RESET_PENDING,
710 RTL_FLAG_TASK_PHY_PENDING,
711 RTL_FLAG_MAX
712};
713
Junchang Wang8027aa22012-03-04 23:30:32 +0100714struct rtl8169_stats {
715 u64 packets;
716 u64 bytes;
717 struct u64_stats_sync syncp;
718};
719
Linus Torvalds1da177e2005-04-16 15:20:36 -0700720struct rtl8169_private {
721 void __iomem *mmio_addr; /* memory map physical address */
Francois Romieucecb5fd2011-04-01 10:21:07 +0200722 struct pci_dev *pci_dev;
David Howellsc4028952006-11-22 14:57:56 +0000723 struct net_device *dev;
Stephen Hemmingerbea33482007-10-03 16:41:36 -0700724 struct napi_struct napi;
Stephen Hemmingerb57b7e52005-05-27 21:11:52 +0200725 u32 msg_enable;
Francois Romieu2b7b4312011-04-18 22:53:24 -0700726 u16 txd_version;
727 u16 mac_version;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700728 u32 cur_rx; /* Index into the Rx descriptor buffer of next Rx pkt. */
729 u32 cur_tx; /* Index into the Tx descriptor buffer of next Rx pkt. */
730 u32 dirty_rx;
731 u32 dirty_tx;
Junchang Wang8027aa22012-03-04 23:30:32 +0100732 struct rtl8169_stats rx_stats;
733 struct rtl8169_stats tx_stats;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700734 struct TxDesc *TxDescArray; /* 256-aligned Tx descriptor ring */
735 struct RxDesc *RxDescArray; /* 256-aligned Rx descriptor ring */
736 dma_addr_t TxPhyAddr;
737 dma_addr_t RxPhyAddr;
Eric Dumazet6f0333b2010-10-11 11:17:47 +0000738 void *Rx_databuff[NUM_RX_DESC]; /* Rx data buffers */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700739 struct ring_info tx_skb[NUM_TX_DESC]; /* Tx data buffers */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700740 struct timer_list timer;
741 u16 cp_cmd;
Francois Romieuda78dbf2012-01-26 14:18:23 +0100742
743 u16 event_slow;
françois romieuc0e45c12011-01-03 15:08:04 +0000744
745 struct mdio_ops {
Francois Romieu24192212012-07-06 20:19:42 +0200746 void (*write)(struct rtl8169_private *, int, int);
747 int (*read)(struct rtl8169_private *, int);
françois romieuc0e45c12011-01-03 15:08:04 +0000748 } mdio_ops;
749
françois romieu065c27c2011-01-03 15:08:12 +0000750 struct pll_power_ops {
751 void (*down)(struct rtl8169_private *);
752 void (*up)(struct rtl8169_private *);
753 } pll_power_ops;
754
Francois Romieud58d46b2011-05-03 16:38:29 +0200755 struct jumbo_ops {
756 void (*enable)(struct rtl8169_private *);
757 void (*disable)(struct rtl8169_private *);
758 } jumbo_ops;
759
Hayes Wangbeb1fe12012-03-30 14:33:01 +0800760 struct csi_ops {
Francois Romieu52989f02012-07-06 13:37:00 +0200761 void (*write)(struct rtl8169_private *, int, int);
762 u32 (*read)(struct rtl8169_private *, int);
Hayes Wangbeb1fe12012-03-30 14:33:01 +0800763 } csi_ops;
764
Oliver Neukum54405cd2011-01-06 21:55:13 +0100765 int (*set_speed)(struct net_device *, u8 aneg, u16 sp, u8 dpx, u32 adv);
Francois Romieuccdffb92008-07-26 14:26:06 +0200766 int (*get_settings)(struct net_device *, struct ethtool_cmd *);
françois romieu4da19632011-01-03 15:07:55 +0000767 void (*phy_reset_enable)(struct rtl8169_private *tp);
Francois Romieu07ce4062007-02-23 23:36:39 +0100768 void (*hw_start)(struct net_device *);
françois romieu4da19632011-01-03 15:07:55 +0000769 unsigned int (*phy_reset_pending)(struct rtl8169_private *tp);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700770 unsigned int (*link_ok)(void __iomem *);
Francois Romieu8b4ab282008-11-19 22:05:25 -0800771 int (*do_ioctl)(struct rtl8169_private *tp, struct mii_ioctl_data *data, int cmd);
Francois Romieu4422bcd2012-01-26 11:23:32 +0100772
773 struct {
Francois Romieuda78dbf2012-01-26 14:18:23 +0100774 DECLARE_BITMAP(flags, RTL_FLAG_MAX);
775 struct mutex mutex;
Francois Romieu4422bcd2012-01-26 11:23:32 +0100776 struct work_struct work;
777 } wk;
778
Francois Romieuf23e7fd2007-10-04 22:36:14 +0200779 unsigned features;
Francois Romieuccdffb92008-07-26 14:26:06 +0200780
781 struct mii_if_info mii;
Ivan Vecera355423d2009-02-06 21:49:57 -0800782 struct rtl8169_counters counters;
Rafael J. Wysockie1759442010-03-14 14:33:51 +0000783 u32 saved_wolopts;
David S. Miller8decf862011-09-22 03:23:13 -0400784 u32 opts1_mask;
françois romieuf1e02ed2011-01-13 13:07:53 +0000785
Francois Romieub6ffd972011-06-17 17:00:05 +0200786 struct rtl_fw {
787 const struct firmware *fw;
Francois Romieu1c361ef2011-06-17 17:16:24 +0200788
789#define RTL_VER_SIZE 32
790
791 char version[RTL_VER_SIZE];
792
793 struct rtl_fw_phy_action {
794 __le32 *code;
795 size_t size;
796 } phy_action;
Francois Romieub6ffd972011-06-17 17:00:05 +0200797 } *rtl_fw;
Phil Carmody497888c2011-07-14 15:07:13 +0300798#define RTL_FIRMWARE_UNKNOWN ERR_PTR(-EAGAIN)
Hayes Wangc5583862012-07-02 17:23:22 +0800799
800 u32 ocp_base;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700801};
802
Ralf Baechle979b6c12005-06-13 14:30:40 -0700803MODULE_AUTHOR("Realtek and the Linux r8169 crew <netdev@vger.kernel.org>");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700804MODULE_DESCRIPTION("RealTek RTL-8169 Gigabit Ethernet driver");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700805module_param(use_dac, int, 0);
David S. Miller4300e8c2010-03-26 10:23:30 -0700806MODULE_PARM_DESC(use_dac, "Enable PCI DAC. Unsafe on 32 bit PCI slot.");
Stephen Hemmingerb57b7e52005-05-27 21:11:52 +0200807module_param_named(debug, debug.msg_enable, int, 0);
808MODULE_PARM_DESC(debug, "Debug verbosity level (0=none, ..., 16=all)");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700809MODULE_LICENSE("GPL");
810MODULE_VERSION(RTL8169_VERSION);
françois romieubca03d52011-01-03 15:07:31 +0000811MODULE_FIRMWARE(FIRMWARE_8168D_1);
812MODULE_FIRMWARE(FIRMWARE_8168D_2);
hayeswang01dc7fe2011-03-21 01:50:28 +0000813MODULE_FIRMWARE(FIRMWARE_8168E_1);
814MODULE_FIRMWARE(FIRMWARE_8168E_2);
David S. Miller8decf862011-09-22 03:23:13 -0400815MODULE_FIRMWARE(FIRMWARE_8168E_3);
Hayes Wang5a5e4442011-02-22 17:26:21 +0800816MODULE_FIRMWARE(FIRMWARE_8105E_1);
Hayes Wangc2218922011-09-06 16:55:18 +0800817MODULE_FIRMWARE(FIRMWARE_8168F_1);
818MODULE_FIRMWARE(FIRMWARE_8168F_2);
Hayes Wang7e18dca2012-03-30 14:33:02 +0800819MODULE_FIRMWARE(FIRMWARE_8402_1);
Hayes Wangb3d7b2f2012-03-30 14:48:06 +0800820MODULE_FIRMWARE(FIRMWARE_8411_1);
Hayes Wang5598bfe2012-07-02 17:23:21 +0800821MODULE_FIRMWARE(FIRMWARE_8106E_1);
Hayes Wangc5583862012-07-02 17:23:22 +0800822MODULE_FIRMWARE(FIRMWARE_8168G_1);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700823
Francois Romieuda78dbf2012-01-26 14:18:23 +0100824static void rtl_lock_work(struct rtl8169_private *tp)
825{
826 mutex_lock(&tp->wk.mutex);
827}
828
829static void rtl_unlock_work(struct rtl8169_private *tp)
830{
831 mutex_unlock(&tp->wk.mutex);
832}
833
Francois Romieud58d46b2011-05-03 16:38:29 +0200834static void rtl_tx_performance_tweak(struct pci_dev *pdev, u16 force)
835{
836 int cap = pci_pcie_cap(pdev);
837
838 if (cap) {
839 u16 ctl;
840
841 pci_read_config_word(pdev, cap + PCI_EXP_DEVCTL, &ctl);
842 ctl = (ctl & ~PCI_EXP_DEVCTL_READRQ) | force;
843 pci_write_config_word(pdev, cap + PCI_EXP_DEVCTL, ctl);
844 }
845}
846
Francois Romieuffc46952012-07-06 14:19:23 +0200847struct rtl_cond {
848 bool (*check)(struct rtl8169_private *);
849 const char *msg;
850};
851
852static void rtl_udelay(unsigned int d)
853{
854 udelay(d);
855}
856
857static bool rtl_loop_wait(struct rtl8169_private *tp, const struct rtl_cond *c,
858 void (*delay)(unsigned int), unsigned int d, int n,
859 bool high)
860{
861 int i;
862
863 for (i = 0; i < n; i++) {
864 delay(d);
865 if (c->check(tp) == high)
866 return true;
867 }
868 netif_err(tp, drv, tp->dev, c->msg);
869 return false;
870}
871
872static bool rtl_udelay_loop_wait_high(struct rtl8169_private *tp,
873 const struct rtl_cond *c,
874 unsigned int d, int n)
875{
876 return rtl_loop_wait(tp, c, rtl_udelay, d, n, true);
877}
878
879static bool rtl_udelay_loop_wait_low(struct rtl8169_private *tp,
880 const struct rtl_cond *c,
881 unsigned int d, int n)
882{
883 return rtl_loop_wait(tp, c, rtl_udelay, d, n, false);
884}
885
886static bool rtl_msleep_loop_wait_high(struct rtl8169_private *tp,
887 const struct rtl_cond *c,
888 unsigned int d, int n)
889{
890 return rtl_loop_wait(tp, c, msleep, d, n, true);
891}
892
893static bool rtl_msleep_loop_wait_low(struct rtl8169_private *tp,
894 const struct rtl_cond *c,
895 unsigned int d, int n)
896{
897 return rtl_loop_wait(tp, c, msleep, d, n, false);
898}
899
900#define DECLARE_RTL_COND(name) \
901static bool name ## _check(struct rtl8169_private *); \
902 \
903static const struct rtl_cond name = { \
904 .check = name ## _check, \
905 .msg = #name \
906}; \
907 \
908static bool name ## _check(struct rtl8169_private *tp)
909
910DECLARE_RTL_COND(rtl_ocpar_cond)
911{
912 void __iomem *ioaddr = tp->mmio_addr;
913
914 return RTL_R32(OCPAR) & OCPAR_FLAG;
915}
916
françois romieub646d902011-01-03 15:08:21 +0000917static u32 ocp_read(struct rtl8169_private *tp, u8 mask, u16 reg)
918{
919 void __iomem *ioaddr = tp->mmio_addr;
françois romieub646d902011-01-03 15:08:21 +0000920
921 RTL_W32(OCPAR, ((u32)mask & 0x0f) << 12 | (reg & 0x0fff));
Francois Romieuffc46952012-07-06 14:19:23 +0200922
923 return rtl_udelay_loop_wait_high(tp, &rtl_ocpar_cond, 100, 20) ?
924 RTL_R32(OCPDR) : ~0;
françois romieub646d902011-01-03 15:08:21 +0000925}
926
927static void ocp_write(struct rtl8169_private *tp, u8 mask, u16 reg, u32 data)
928{
929 void __iomem *ioaddr = tp->mmio_addr;
françois romieub646d902011-01-03 15:08:21 +0000930
931 RTL_W32(OCPDR, data);
932 RTL_W32(OCPAR, OCPAR_FLAG | ((u32)mask & 0x0f) << 12 | (reg & 0x0fff));
Francois Romieuffc46952012-07-06 14:19:23 +0200933
934 rtl_udelay_loop_wait_low(tp, &rtl_ocpar_cond, 100, 20);
935}
936
937DECLARE_RTL_COND(rtl_eriar_cond)
938{
939 void __iomem *ioaddr = tp->mmio_addr;
940
941 return RTL_R32(ERIAR) & ERIAR_FLAG;
françois romieub646d902011-01-03 15:08:21 +0000942}
943
Hayes Wangfac5b3c2011-02-22 17:26:20 +0800944static void rtl8168_oob_notify(struct rtl8169_private *tp, u8 cmd)
françois romieub646d902011-01-03 15:08:21 +0000945{
Hayes Wangfac5b3c2011-02-22 17:26:20 +0800946 void __iomem *ioaddr = tp->mmio_addr;
françois romieub646d902011-01-03 15:08:21 +0000947
948 RTL_W8(ERIDR, cmd);
949 RTL_W32(ERIAR, 0x800010e8);
950 msleep(2);
Francois Romieuffc46952012-07-06 14:19:23 +0200951
952 if (!rtl_udelay_loop_wait_low(tp, &rtl_eriar_cond, 100, 5))
953 return;
françois romieub646d902011-01-03 15:08:21 +0000954
Hayes Wangfac5b3c2011-02-22 17:26:20 +0800955 ocp_write(tp, 0x1, 0x30, 0x00000001);
françois romieub646d902011-01-03 15:08:21 +0000956}
957
958#define OOB_CMD_RESET 0x00
959#define OOB_CMD_DRIVER_START 0x05
960#define OOB_CMD_DRIVER_STOP 0x06
961
Francois Romieucecb5fd2011-04-01 10:21:07 +0200962static u16 rtl8168_get_ocp_reg(struct rtl8169_private *tp)
963{
964 return (tp->mac_version == RTL_GIGA_MAC_VER_31) ? 0xb8 : 0x10;
965}
966
Francois Romieuffc46952012-07-06 14:19:23 +0200967DECLARE_RTL_COND(rtl_ocp_read_cond)
françois romieub646d902011-01-03 15:08:21 +0000968{
Francois Romieucecb5fd2011-04-01 10:21:07 +0200969 u16 reg;
françois romieub646d902011-01-03 15:08:21 +0000970
Francois Romieucecb5fd2011-04-01 10:21:07 +0200971 reg = rtl8168_get_ocp_reg(tp);
hayeswang4804b3b2011-03-21 01:50:29 +0000972
Francois Romieuffc46952012-07-06 14:19:23 +0200973 return ocp_read(tp, 0x0f, reg) & 0x00000800;
974}
975
976static void rtl8168_driver_start(struct rtl8169_private *tp)
977{
978 rtl8168_oob_notify(tp, OOB_CMD_DRIVER_START);
979
980 rtl_msleep_loop_wait_high(tp, &rtl_ocp_read_cond, 10, 10);
françois romieub646d902011-01-03 15:08:21 +0000981}
982
983static void rtl8168_driver_stop(struct rtl8169_private *tp)
984{
françois romieub646d902011-01-03 15:08:21 +0000985 rtl8168_oob_notify(tp, OOB_CMD_DRIVER_STOP);
986
Francois Romieuffc46952012-07-06 14:19:23 +0200987 rtl_msleep_loop_wait_low(tp, &rtl_ocp_read_cond, 10, 10);
françois romieub646d902011-01-03 15:08:21 +0000988}
989
hayeswang4804b3b2011-03-21 01:50:29 +0000990static int r8168dp_check_dash(struct rtl8169_private *tp)
991{
Francois Romieucecb5fd2011-04-01 10:21:07 +0200992 u16 reg = rtl8168_get_ocp_reg(tp);
hayeswang4804b3b2011-03-21 01:50:29 +0000993
Francois Romieucecb5fd2011-04-01 10:21:07 +0200994 return (ocp_read(tp, 0x0f, reg) & 0x00008000) ? 1 : 0;
hayeswang4804b3b2011-03-21 01:50:29 +0000995}
françois romieub646d902011-01-03 15:08:21 +0000996
Hayes Wangc5583862012-07-02 17:23:22 +0800997static bool rtl_ocp_reg_failure(struct rtl8169_private *tp, u32 reg)
998{
999 if (reg & 0xffff0001) {
1000 netif_err(tp, drv, tp->dev, "Invalid ocp reg %x!\n", reg);
1001 return true;
1002 }
1003 return false;
1004}
1005
1006DECLARE_RTL_COND(rtl_ocp_gphy_cond)
1007{
1008 void __iomem *ioaddr = tp->mmio_addr;
1009
1010 return RTL_R32(GPHY_OCP) & OCPAR_FLAG;
1011}
1012
1013static void r8168_phy_ocp_write(struct rtl8169_private *tp, u32 reg, u32 data)
1014{
1015 void __iomem *ioaddr = tp->mmio_addr;
1016
1017 if (rtl_ocp_reg_failure(tp, reg))
1018 return;
1019
1020 RTL_W32(GPHY_OCP, OCPAR_FLAG | (reg << 15) | data);
1021
1022 rtl_udelay_loop_wait_low(tp, &rtl_ocp_gphy_cond, 25, 10);
1023}
1024
1025static u16 r8168_phy_ocp_read(struct rtl8169_private *tp, u32 reg)
1026{
1027 void __iomem *ioaddr = tp->mmio_addr;
1028
1029 if (rtl_ocp_reg_failure(tp, reg))
1030 return 0;
1031
1032 RTL_W32(GPHY_OCP, reg << 15);
1033
1034 return rtl_udelay_loop_wait_high(tp, &rtl_ocp_gphy_cond, 25, 10) ?
1035 (RTL_R32(GPHY_OCP) & 0xffff) : ~0;
1036}
1037
1038static void rtl_w1w0_phy_ocp(struct rtl8169_private *tp, int reg, int p, int m)
1039{
1040 int val;
1041
1042 val = r8168_phy_ocp_read(tp, reg);
1043 r8168_phy_ocp_write(tp, reg, (val | p) & ~m);
1044}
1045
1046DECLARE_RTL_COND(rtl_ocpdr_cond)
1047{
1048 void __iomem *ioaddr = tp->mmio_addr;
1049
1050 return RTL_R32(OCPDR) & OCPAR_FLAG;
1051}
1052
1053static void r8168_mac_ocp_write(struct rtl8169_private *tp, u32 reg, u32 data)
1054{
1055 void __iomem *ioaddr = tp->mmio_addr;
1056
1057 if (rtl_ocp_reg_failure(tp, reg))
1058 return;
1059
1060 RTL_W32(OCPDR, OCPAR_FLAG | (reg << 15) | data);
1061
1062 rtl_udelay_loop_wait_low(tp, &rtl_ocpdr_cond, 25, 10);
1063}
1064
1065static u16 r8168_mac_ocp_read(struct rtl8169_private *tp, u32 reg)
1066{
1067 void __iomem *ioaddr = tp->mmio_addr;
1068
1069 if (rtl_ocp_reg_failure(tp, reg))
1070 return 0;
1071
1072 RTL_W32(OCPDR, reg << 15);
1073
1074 return rtl_udelay_loop_wait_high(tp, &rtl_ocpdr_cond, 25, 10) ?
1075 RTL_R32(OCPDR) : ~0;
1076}
1077
1078#define OCP_STD_PHY_BASE 0xa400
1079
1080static void r8168g_mdio_write(struct rtl8169_private *tp, int reg, int value)
1081{
1082 if (reg == 0x1f) {
1083 tp->ocp_base = value ? value << 4 : OCP_STD_PHY_BASE;
1084 return;
1085 }
1086
1087 if (tp->ocp_base != OCP_STD_PHY_BASE)
1088 reg -= 0x10;
1089
1090 r8168_phy_ocp_write(tp, tp->ocp_base + reg * 2, value);
1091}
1092
1093static int r8168g_mdio_read(struct rtl8169_private *tp, int reg)
1094{
1095 if (tp->ocp_base != OCP_STD_PHY_BASE)
1096 reg -= 0x10;
1097
1098 return r8168_phy_ocp_read(tp, tp->ocp_base + reg * 2);
1099}
1100
Francois Romieuffc46952012-07-06 14:19:23 +02001101DECLARE_RTL_COND(rtl_phyar_cond)
1102{
1103 void __iomem *ioaddr = tp->mmio_addr;
1104
1105 return RTL_R32(PHYAR) & 0x80000000;
1106}
1107
Francois Romieu24192212012-07-06 20:19:42 +02001108static void r8169_mdio_write(struct rtl8169_private *tp, int reg, int value)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001109{
Francois Romieu24192212012-07-06 20:19:42 +02001110 void __iomem *ioaddr = tp->mmio_addr;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001111
Francois Romieu24192212012-07-06 20:19:42 +02001112 RTL_W32(PHYAR, 0x80000000 | (reg & 0x1f) << 16 | (value & 0xffff));
Linus Torvalds1da177e2005-04-16 15:20:36 -07001113
Francois Romieuffc46952012-07-06 14:19:23 +02001114 rtl_udelay_loop_wait_low(tp, &rtl_phyar_cond, 25, 20);
Timo Teräs024a07b2010-06-06 15:38:47 -07001115 /*
Timo Teräs81a95f02010-06-09 17:31:48 -07001116 * According to hardware specs a 20us delay is required after write
1117 * complete indication, but before sending next command.
Timo Teräs024a07b2010-06-06 15:38:47 -07001118 */
Timo Teräs81a95f02010-06-09 17:31:48 -07001119 udelay(20);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001120}
1121
Francois Romieu24192212012-07-06 20:19:42 +02001122static int r8169_mdio_read(struct rtl8169_private *tp, int reg)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001123{
Francois Romieu24192212012-07-06 20:19:42 +02001124 void __iomem *ioaddr = tp->mmio_addr;
Francois Romieuffc46952012-07-06 14:19:23 +02001125 int value;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001126
Francois Romieu24192212012-07-06 20:19:42 +02001127 RTL_W32(PHYAR, 0x0 | (reg & 0x1f) << 16);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001128
Francois Romieuffc46952012-07-06 14:19:23 +02001129 value = rtl_udelay_loop_wait_high(tp, &rtl_phyar_cond, 25, 20) ?
1130 RTL_R32(PHYAR) & 0xffff : ~0;
1131
Timo Teräs81a95f02010-06-09 17:31:48 -07001132 /*
1133 * According to hardware specs a 20us delay is required after read
1134 * complete indication, but before sending next command.
1135 */
1136 udelay(20);
1137
Linus Torvalds1da177e2005-04-16 15:20:36 -07001138 return value;
1139}
1140
Francois Romieu24192212012-07-06 20:19:42 +02001141static void r8168dp_1_mdio_access(struct rtl8169_private *tp, int reg, u32 data)
françois romieuc0e45c12011-01-03 15:08:04 +00001142{
Francois Romieu24192212012-07-06 20:19:42 +02001143 void __iomem *ioaddr = tp->mmio_addr;
françois romieuc0e45c12011-01-03 15:08:04 +00001144
Francois Romieu24192212012-07-06 20:19:42 +02001145 RTL_W32(OCPDR, data | ((reg & OCPDR_REG_MASK) << OCPDR_GPHY_REG_SHIFT));
françois romieuc0e45c12011-01-03 15:08:04 +00001146 RTL_W32(OCPAR, OCPAR_GPHY_WRITE_CMD);
1147 RTL_W32(EPHY_RXER_NUM, 0);
1148
Francois Romieuffc46952012-07-06 14:19:23 +02001149 rtl_udelay_loop_wait_low(tp, &rtl_ocpar_cond, 1000, 100);
françois romieuc0e45c12011-01-03 15:08:04 +00001150}
1151
Francois Romieu24192212012-07-06 20:19:42 +02001152static void r8168dp_1_mdio_write(struct rtl8169_private *tp, int reg, int value)
françois romieuc0e45c12011-01-03 15:08:04 +00001153{
Francois Romieu24192212012-07-06 20:19:42 +02001154 r8168dp_1_mdio_access(tp, reg,
1155 OCPDR_WRITE_CMD | (value & OCPDR_DATA_MASK));
françois romieuc0e45c12011-01-03 15:08:04 +00001156}
1157
Francois Romieu24192212012-07-06 20:19:42 +02001158static int r8168dp_1_mdio_read(struct rtl8169_private *tp, int reg)
françois romieuc0e45c12011-01-03 15:08:04 +00001159{
Francois Romieu24192212012-07-06 20:19:42 +02001160 void __iomem *ioaddr = tp->mmio_addr;
françois romieuc0e45c12011-01-03 15:08:04 +00001161
Francois Romieu24192212012-07-06 20:19:42 +02001162 r8168dp_1_mdio_access(tp, reg, OCPDR_READ_CMD);
françois romieuc0e45c12011-01-03 15:08:04 +00001163
1164 mdelay(1);
1165 RTL_W32(OCPAR, OCPAR_GPHY_READ_CMD);
1166 RTL_W32(EPHY_RXER_NUM, 0);
1167
Francois Romieuffc46952012-07-06 14:19:23 +02001168 return rtl_udelay_loop_wait_high(tp, &rtl_ocpar_cond, 1000, 100) ?
1169 RTL_R32(OCPDR) & OCPDR_DATA_MASK : ~0;
françois romieuc0e45c12011-01-03 15:08:04 +00001170}
1171
françois romieue6de30d2011-01-03 15:08:37 +00001172#define R8168DP_1_MDIO_ACCESS_BIT 0x00020000
1173
1174static void r8168dp_2_mdio_start(void __iomem *ioaddr)
1175{
1176 RTL_W32(0xd0, RTL_R32(0xd0) & ~R8168DP_1_MDIO_ACCESS_BIT);
1177}
1178
1179static void r8168dp_2_mdio_stop(void __iomem *ioaddr)
1180{
1181 RTL_W32(0xd0, RTL_R32(0xd0) | R8168DP_1_MDIO_ACCESS_BIT);
1182}
1183
Francois Romieu24192212012-07-06 20:19:42 +02001184static void r8168dp_2_mdio_write(struct rtl8169_private *tp, int reg, int value)
françois romieue6de30d2011-01-03 15:08:37 +00001185{
Francois Romieu24192212012-07-06 20:19:42 +02001186 void __iomem *ioaddr = tp->mmio_addr;
1187
françois romieue6de30d2011-01-03 15:08:37 +00001188 r8168dp_2_mdio_start(ioaddr);
1189
Francois Romieu24192212012-07-06 20:19:42 +02001190 r8169_mdio_write(tp, reg, value);
françois romieue6de30d2011-01-03 15:08:37 +00001191
1192 r8168dp_2_mdio_stop(ioaddr);
1193}
1194
Francois Romieu24192212012-07-06 20:19:42 +02001195static int r8168dp_2_mdio_read(struct rtl8169_private *tp, int reg)
françois romieue6de30d2011-01-03 15:08:37 +00001196{
Francois Romieu24192212012-07-06 20:19:42 +02001197 void __iomem *ioaddr = tp->mmio_addr;
françois romieue6de30d2011-01-03 15:08:37 +00001198 int value;
1199
1200 r8168dp_2_mdio_start(ioaddr);
1201
Francois Romieu24192212012-07-06 20:19:42 +02001202 value = r8169_mdio_read(tp, reg);
françois romieue6de30d2011-01-03 15:08:37 +00001203
1204 r8168dp_2_mdio_stop(ioaddr);
1205
1206 return value;
1207}
1208
françois romieu4da19632011-01-03 15:07:55 +00001209static void rtl_writephy(struct rtl8169_private *tp, int location, u32 val)
Francois Romieudacf8152008-08-02 20:44:13 +02001210{
Francois Romieu24192212012-07-06 20:19:42 +02001211 tp->mdio_ops.write(tp, location, val);
Francois Romieudacf8152008-08-02 20:44:13 +02001212}
1213
françois romieu4da19632011-01-03 15:07:55 +00001214static int rtl_readphy(struct rtl8169_private *tp, int location)
1215{
Francois Romieu24192212012-07-06 20:19:42 +02001216 return tp->mdio_ops.read(tp, location);
françois romieu4da19632011-01-03 15:07:55 +00001217}
1218
1219static void rtl_patchphy(struct rtl8169_private *tp, int reg_addr, int value)
1220{
1221 rtl_writephy(tp, reg_addr, rtl_readphy(tp, reg_addr) | value);
1222}
1223
1224static void rtl_w1w0_phy(struct rtl8169_private *tp, int reg_addr, int p, int m)
françois romieudaf9df62009-10-07 12:44:20 +00001225{
1226 int val;
1227
françois romieu4da19632011-01-03 15:07:55 +00001228 val = rtl_readphy(tp, reg_addr);
1229 rtl_writephy(tp, reg_addr, (val | p) & ~m);
françois romieudaf9df62009-10-07 12:44:20 +00001230}
1231
Francois Romieuccdffb92008-07-26 14:26:06 +02001232static void rtl_mdio_write(struct net_device *dev, int phy_id, int location,
1233 int val)
1234{
1235 struct rtl8169_private *tp = netdev_priv(dev);
Francois Romieuccdffb92008-07-26 14:26:06 +02001236
françois romieu4da19632011-01-03 15:07:55 +00001237 rtl_writephy(tp, location, val);
Francois Romieuccdffb92008-07-26 14:26:06 +02001238}
1239
1240static int rtl_mdio_read(struct net_device *dev, int phy_id, int location)
1241{
1242 struct rtl8169_private *tp = netdev_priv(dev);
Francois Romieuccdffb92008-07-26 14:26:06 +02001243
françois romieu4da19632011-01-03 15:07:55 +00001244 return rtl_readphy(tp, location);
Francois Romieuccdffb92008-07-26 14:26:06 +02001245}
1246
Francois Romieuffc46952012-07-06 14:19:23 +02001247DECLARE_RTL_COND(rtl_ephyar_cond)
1248{
1249 void __iomem *ioaddr = tp->mmio_addr;
1250
1251 return RTL_R32(EPHYAR) & EPHYAR_FLAG;
1252}
1253
Francois Romieufdf6fc02012-07-06 22:40:38 +02001254static void rtl_ephy_write(struct rtl8169_private *tp, int reg_addr, int value)
Francois Romieudacf8152008-08-02 20:44:13 +02001255{
Francois Romieufdf6fc02012-07-06 22:40:38 +02001256 void __iomem *ioaddr = tp->mmio_addr;
Francois Romieudacf8152008-08-02 20:44:13 +02001257
1258 RTL_W32(EPHYAR, EPHYAR_WRITE_CMD | (value & EPHYAR_DATA_MASK) |
1259 (reg_addr & EPHYAR_REG_MASK) << EPHYAR_REG_SHIFT);
1260
Francois Romieuffc46952012-07-06 14:19:23 +02001261 rtl_udelay_loop_wait_low(tp, &rtl_ephyar_cond, 10, 100);
1262
1263 udelay(10);
Francois Romieudacf8152008-08-02 20:44:13 +02001264}
1265
Francois Romieufdf6fc02012-07-06 22:40:38 +02001266static u16 rtl_ephy_read(struct rtl8169_private *tp, int reg_addr)
Francois Romieudacf8152008-08-02 20:44:13 +02001267{
Francois Romieufdf6fc02012-07-06 22:40:38 +02001268 void __iomem *ioaddr = tp->mmio_addr;
Francois Romieudacf8152008-08-02 20:44:13 +02001269
1270 RTL_W32(EPHYAR, (reg_addr & EPHYAR_REG_MASK) << EPHYAR_REG_SHIFT);
1271
Francois Romieuffc46952012-07-06 14:19:23 +02001272 return rtl_udelay_loop_wait_high(tp, &rtl_ephyar_cond, 10, 100) ?
1273 RTL_R32(EPHYAR) & EPHYAR_DATA_MASK : ~0;
Francois Romieudacf8152008-08-02 20:44:13 +02001274}
1275
Francois Romieufdf6fc02012-07-06 22:40:38 +02001276static void rtl_eri_write(struct rtl8169_private *tp, int addr, u32 mask,
1277 u32 val, int type)
Hayes Wang133ac402011-07-06 15:58:05 +08001278{
Francois Romieufdf6fc02012-07-06 22:40:38 +02001279 void __iomem *ioaddr = tp->mmio_addr;
Hayes Wang133ac402011-07-06 15:58:05 +08001280
1281 BUG_ON((addr & 3) || (mask == 0));
1282 RTL_W32(ERIDR, val);
1283 RTL_W32(ERIAR, ERIAR_WRITE_CMD | type | mask | addr);
1284
Francois Romieuffc46952012-07-06 14:19:23 +02001285 rtl_udelay_loop_wait_low(tp, &rtl_eriar_cond, 100, 100);
Hayes Wang133ac402011-07-06 15:58:05 +08001286}
1287
Francois Romieufdf6fc02012-07-06 22:40:38 +02001288static u32 rtl_eri_read(struct rtl8169_private *tp, int addr, int type)
Hayes Wang133ac402011-07-06 15:58:05 +08001289{
Francois Romieufdf6fc02012-07-06 22:40:38 +02001290 void __iomem *ioaddr = tp->mmio_addr;
Hayes Wang133ac402011-07-06 15:58:05 +08001291
1292 RTL_W32(ERIAR, ERIAR_READ_CMD | type | ERIAR_MASK_1111 | addr);
1293
Francois Romieuffc46952012-07-06 14:19:23 +02001294 return rtl_udelay_loop_wait_high(tp, &rtl_eriar_cond, 100, 100) ?
1295 RTL_R32(ERIDR) : ~0;
Hayes Wang133ac402011-07-06 15:58:05 +08001296}
1297
Francois Romieufdf6fc02012-07-06 22:40:38 +02001298static void rtl_w1w0_eri(struct rtl8169_private *tp, int addr, u32 mask, u32 p,
1299 u32 m, int type)
Hayes Wang133ac402011-07-06 15:58:05 +08001300{
1301 u32 val;
1302
Francois Romieufdf6fc02012-07-06 22:40:38 +02001303 val = rtl_eri_read(tp, addr, type);
1304 rtl_eri_write(tp, addr, mask, (val & ~m) | p, type);
Hayes Wang133ac402011-07-06 15:58:05 +08001305}
1306
françois romieuc28aa382011-08-02 03:53:43 +00001307struct exgmac_reg {
1308 u16 addr;
1309 u16 mask;
1310 u32 val;
1311};
1312
Francois Romieufdf6fc02012-07-06 22:40:38 +02001313static void rtl_write_exgmac_batch(struct rtl8169_private *tp,
françois romieuc28aa382011-08-02 03:53:43 +00001314 const struct exgmac_reg *r, int len)
1315{
1316 while (len-- > 0) {
Francois Romieufdf6fc02012-07-06 22:40:38 +02001317 rtl_eri_write(tp, r->addr, r->mask, r->val, ERIAR_EXGMAC);
françois romieuc28aa382011-08-02 03:53:43 +00001318 r++;
1319 }
1320}
1321
Francois Romieuffc46952012-07-06 14:19:23 +02001322DECLARE_RTL_COND(rtl_efusear_cond)
1323{
1324 void __iomem *ioaddr = tp->mmio_addr;
1325
1326 return RTL_R32(EFUSEAR) & EFUSEAR_FLAG;
1327}
1328
Francois Romieufdf6fc02012-07-06 22:40:38 +02001329static u8 rtl8168d_efuse_read(struct rtl8169_private *tp, int reg_addr)
françois romieudaf9df62009-10-07 12:44:20 +00001330{
Francois Romieufdf6fc02012-07-06 22:40:38 +02001331 void __iomem *ioaddr = tp->mmio_addr;
françois romieudaf9df62009-10-07 12:44:20 +00001332
1333 RTL_W32(EFUSEAR, (reg_addr & EFUSEAR_REG_MASK) << EFUSEAR_REG_SHIFT);
1334
Francois Romieuffc46952012-07-06 14:19:23 +02001335 return rtl_udelay_loop_wait_high(tp, &rtl_efusear_cond, 100, 300) ?
1336 RTL_R32(EFUSEAR) & EFUSEAR_DATA_MASK : ~0;
françois romieudaf9df62009-10-07 12:44:20 +00001337}
1338
Francois Romieu9085cdfa2012-01-26 12:59:08 +01001339static u16 rtl_get_events(struct rtl8169_private *tp)
1340{
1341 void __iomem *ioaddr = tp->mmio_addr;
1342
1343 return RTL_R16(IntrStatus);
1344}
1345
1346static void rtl_ack_events(struct rtl8169_private *tp, u16 bits)
1347{
1348 void __iomem *ioaddr = tp->mmio_addr;
1349
1350 RTL_W16(IntrStatus, bits);
1351 mmiowb();
1352}
1353
1354static void rtl_irq_disable(struct rtl8169_private *tp)
1355{
1356 void __iomem *ioaddr = tp->mmio_addr;
1357
1358 RTL_W16(IntrMask, 0);
1359 mmiowb();
1360}
1361
Francois Romieu3e990ff2012-01-26 12:50:01 +01001362static void rtl_irq_enable(struct rtl8169_private *tp, u16 bits)
1363{
1364 void __iomem *ioaddr = tp->mmio_addr;
1365
1366 RTL_W16(IntrMask, bits);
1367}
1368
Francois Romieuda78dbf2012-01-26 14:18:23 +01001369#define RTL_EVENT_NAPI_RX (RxOK | RxErr)
1370#define RTL_EVENT_NAPI_TX (TxOK | TxErr)
1371#define RTL_EVENT_NAPI (RTL_EVENT_NAPI_RX | RTL_EVENT_NAPI_TX)
1372
1373static void rtl_irq_enable_all(struct rtl8169_private *tp)
1374{
1375 rtl_irq_enable(tp, RTL_EVENT_NAPI | tp->event_slow);
1376}
1377
françois romieu811fd302011-12-04 20:30:45 +00001378static void rtl8169_irq_mask_and_ack(struct rtl8169_private *tp)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001379{
françois romieu811fd302011-12-04 20:30:45 +00001380 void __iomem *ioaddr = tp->mmio_addr;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001381
Francois Romieu9085cdfa2012-01-26 12:59:08 +01001382 rtl_irq_disable(tp);
Francois Romieuda78dbf2012-01-26 14:18:23 +01001383 rtl_ack_events(tp, RTL_EVENT_NAPI | tp->event_slow);
françois romieu811fd302011-12-04 20:30:45 +00001384 RTL_R8(ChipCmd);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001385}
1386
françois romieu4da19632011-01-03 15:07:55 +00001387static unsigned int rtl8169_tbi_reset_pending(struct rtl8169_private *tp)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001388{
françois romieu4da19632011-01-03 15:07:55 +00001389 void __iomem *ioaddr = tp->mmio_addr;
1390
Linus Torvalds1da177e2005-04-16 15:20:36 -07001391 return RTL_R32(TBICSR) & TBIReset;
1392}
1393
françois romieu4da19632011-01-03 15:07:55 +00001394static unsigned int rtl8169_xmii_reset_pending(struct rtl8169_private *tp)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001395{
françois romieu4da19632011-01-03 15:07:55 +00001396 return rtl_readphy(tp, MII_BMCR) & BMCR_RESET;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001397}
1398
1399static unsigned int rtl8169_tbi_link_ok(void __iomem *ioaddr)
1400{
1401 return RTL_R32(TBICSR) & TBILinkOk;
1402}
1403
1404static unsigned int rtl8169_xmii_link_ok(void __iomem *ioaddr)
1405{
1406 return RTL_R8(PHYstatus) & LinkStatus;
1407}
1408
françois romieu4da19632011-01-03 15:07:55 +00001409static void rtl8169_tbi_reset_enable(struct rtl8169_private *tp)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001410{
françois romieu4da19632011-01-03 15:07:55 +00001411 void __iomem *ioaddr = tp->mmio_addr;
1412
Linus Torvalds1da177e2005-04-16 15:20:36 -07001413 RTL_W32(TBICSR, RTL_R32(TBICSR) | TBIReset);
1414}
1415
françois romieu4da19632011-01-03 15:07:55 +00001416static void rtl8169_xmii_reset_enable(struct rtl8169_private *tp)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001417{
1418 unsigned int val;
1419
françois romieu4da19632011-01-03 15:07:55 +00001420 val = rtl_readphy(tp, MII_BMCR) | BMCR_RESET;
1421 rtl_writephy(tp, MII_BMCR, val & 0xffff);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001422}
1423
Hayes Wang70090422011-07-06 15:58:06 +08001424static void rtl_link_chg_patch(struct rtl8169_private *tp)
1425{
1426 void __iomem *ioaddr = tp->mmio_addr;
1427 struct net_device *dev = tp->dev;
1428
1429 if (!netif_running(dev))
1430 return;
1431
Hayes Wangb3d7b2f2012-03-30 14:48:06 +08001432 if (tp->mac_version == RTL_GIGA_MAC_VER_34 ||
1433 tp->mac_version == RTL_GIGA_MAC_VER_38) {
Hayes Wang70090422011-07-06 15:58:06 +08001434 if (RTL_R8(PHYstatus) & _1000bpsF) {
Francois Romieufdf6fc02012-07-06 22:40:38 +02001435 rtl_eri_write(tp, 0x1bc, ERIAR_MASK_1111, 0x00000011,
1436 ERIAR_EXGMAC);
1437 rtl_eri_write(tp, 0x1dc, ERIAR_MASK_1111, 0x00000005,
1438 ERIAR_EXGMAC);
Hayes Wang70090422011-07-06 15:58:06 +08001439 } else if (RTL_R8(PHYstatus) & _100bps) {
Francois Romieufdf6fc02012-07-06 22:40:38 +02001440 rtl_eri_write(tp, 0x1bc, ERIAR_MASK_1111, 0x0000001f,
1441 ERIAR_EXGMAC);
1442 rtl_eri_write(tp, 0x1dc, ERIAR_MASK_1111, 0x00000005,
1443 ERIAR_EXGMAC);
Hayes Wang70090422011-07-06 15:58:06 +08001444 } else {
Francois Romieufdf6fc02012-07-06 22:40:38 +02001445 rtl_eri_write(tp, 0x1bc, ERIAR_MASK_1111, 0x0000001f,
1446 ERIAR_EXGMAC);
1447 rtl_eri_write(tp, 0x1dc, ERIAR_MASK_1111, 0x0000003f,
1448 ERIAR_EXGMAC);
Hayes Wang70090422011-07-06 15:58:06 +08001449 }
1450 /* Reset packet filter */
Francois Romieufdf6fc02012-07-06 22:40:38 +02001451 rtl_w1w0_eri(tp, 0xdc, ERIAR_MASK_0001, 0x00, 0x01,
Hayes Wang70090422011-07-06 15:58:06 +08001452 ERIAR_EXGMAC);
Francois Romieufdf6fc02012-07-06 22:40:38 +02001453 rtl_w1w0_eri(tp, 0xdc, ERIAR_MASK_0001, 0x01, 0x00,
Hayes Wang70090422011-07-06 15:58:06 +08001454 ERIAR_EXGMAC);
Hayes Wangc2218922011-09-06 16:55:18 +08001455 } else if (tp->mac_version == RTL_GIGA_MAC_VER_35 ||
1456 tp->mac_version == RTL_GIGA_MAC_VER_36) {
1457 if (RTL_R8(PHYstatus) & _1000bpsF) {
Francois Romieufdf6fc02012-07-06 22:40:38 +02001458 rtl_eri_write(tp, 0x1bc, ERIAR_MASK_1111, 0x00000011,
1459 ERIAR_EXGMAC);
1460 rtl_eri_write(tp, 0x1dc, ERIAR_MASK_1111, 0x00000005,
1461 ERIAR_EXGMAC);
Hayes Wangc2218922011-09-06 16:55:18 +08001462 } else {
Francois Romieufdf6fc02012-07-06 22:40:38 +02001463 rtl_eri_write(tp, 0x1bc, ERIAR_MASK_1111, 0x0000001f,
1464 ERIAR_EXGMAC);
1465 rtl_eri_write(tp, 0x1dc, ERIAR_MASK_1111, 0x0000003f,
1466 ERIAR_EXGMAC);
Hayes Wangc2218922011-09-06 16:55:18 +08001467 }
Hayes Wang7e18dca2012-03-30 14:33:02 +08001468 } else if (tp->mac_version == RTL_GIGA_MAC_VER_37) {
1469 if (RTL_R8(PHYstatus) & _10bps) {
Francois Romieufdf6fc02012-07-06 22:40:38 +02001470 rtl_eri_write(tp, 0x1d0, ERIAR_MASK_0011, 0x4d02,
1471 ERIAR_EXGMAC);
1472 rtl_eri_write(tp, 0x1dc, ERIAR_MASK_0011, 0x0060,
1473 ERIAR_EXGMAC);
Hayes Wang7e18dca2012-03-30 14:33:02 +08001474 } else {
Francois Romieufdf6fc02012-07-06 22:40:38 +02001475 rtl_eri_write(tp, 0x1d0, ERIAR_MASK_0011, 0x0000,
1476 ERIAR_EXGMAC);
Hayes Wang7e18dca2012-03-30 14:33:02 +08001477 }
Hayes Wang70090422011-07-06 15:58:06 +08001478 }
1479}
1480
Rafael J. Wysockie4fbce72010-12-08 15:32:14 +00001481static void __rtl8169_check_link_status(struct net_device *dev,
Francois Romieucecb5fd2011-04-01 10:21:07 +02001482 struct rtl8169_private *tp,
1483 void __iomem *ioaddr, bool pm)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001484{
Linus Torvalds1da177e2005-04-16 15:20:36 -07001485 if (tp->link_ok(ioaddr)) {
Hayes Wang70090422011-07-06 15:58:06 +08001486 rtl_link_chg_patch(tp);
Rafael J. Wysockie1759442010-03-14 14:33:51 +00001487 /* This is to cancel a scheduled suspend if there's one. */
Rafael J. Wysockie4fbce72010-12-08 15:32:14 +00001488 if (pm)
1489 pm_request_resume(&tp->pci_dev->dev);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001490 netif_carrier_on(dev);
Francois Romieu1519e572011-02-03 12:02:36 +01001491 if (net_ratelimit())
1492 netif_info(tp, ifup, dev, "link up\n");
Stephen Hemmingerb57b7e52005-05-27 21:11:52 +02001493 } else {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001494 netif_carrier_off(dev);
Joe Perchesbf82c182010-02-09 11:49:50 +00001495 netif_info(tp, ifdown, dev, "link down\n");
Rafael J. Wysockie4fbce72010-12-08 15:32:14 +00001496 if (pm)
hayeswang10953db2011-11-07 20:44:37 +00001497 pm_schedule_suspend(&tp->pci_dev->dev, 5000);
Stephen Hemmingerb57b7e52005-05-27 21:11:52 +02001498 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001499}
1500
Rafael J. Wysockie4fbce72010-12-08 15:32:14 +00001501static void rtl8169_check_link_status(struct net_device *dev,
1502 struct rtl8169_private *tp,
1503 void __iomem *ioaddr)
1504{
1505 __rtl8169_check_link_status(dev, tp, ioaddr, false);
1506}
1507
Rafael J. Wysockie1759442010-03-14 14:33:51 +00001508#define WAKE_ANY (WAKE_PHY | WAKE_MAGIC | WAKE_UCAST | WAKE_BCAST | WAKE_MCAST)
1509
1510static u32 __rtl8169_get_wol(struct rtl8169_private *tp)
1511{
1512 void __iomem *ioaddr = tp->mmio_addr;
1513 u8 options;
1514 u32 wolopts = 0;
1515
1516 options = RTL_R8(Config1);
1517 if (!(options & PMEnable))
1518 return 0;
1519
1520 options = RTL_R8(Config3);
1521 if (options & LinkUp)
1522 wolopts |= WAKE_PHY;
1523 if (options & MagicPacket)
1524 wolopts |= WAKE_MAGIC;
1525
1526 options = RTL_R8(Config5);
1527 if (options & UWF)
1528 wolopts |= WAKE_UCAST;
1529 if (options & BWF)
1530 wolopts |= WAKE_BCAST;
1531 if (options & MWF)
1532 wolopts |= WAKE_MCAST;
1533
1534 return wolopts;
1535}
1536
Francois Romieu61a4dcc2006-02-23 00:55:25 +01001537static void rtl8169_get_wol(struct net_device *dev, struct ethtool_wolinfo *wol)
1538{
1539 struct rtl8169_private *tp = netdev_priv(dev);
Francois Romieu61a4dcc2006-02-23 00:55:25 +01001540
Francois Romieuda78dbf2012-01-26 14:18:23 +01001541 rtl_lock_work(tp);
Francois Romieu61a4dcc2006-02-23 00:55:25 +01001542
Rafael J. Wysockie1759442010-03-14 14:33:51 +00001543 wol->supported = WAKE_ANY;
1544 wol->wolopts = __rtl8169_get_wol(tp);
Francois Romieu61a4dcc2006-02-23 00:55:25 +01001545
Francois Romieuda78dbf2012-01-26 14:18:23 +01001546 rtl_unlock_work(tp);
Francois Romieu61a4dcc2006-02-23 00:55:25 +01001547}
1548
Rafael J. Wysockie1759442010-03-14 14:33:51 +00001549static void __rtl8169_set_wol(struct rtl8169_private *tp, u32 wolopts)
Francois Romieu61a4dcc2006-02-23 00:55:25 +01001550{
Francois Romieu61a4dcc2006-02-23 00:55:25 +01001551 void __iomem *ioaddr = tp->mmio_addr;
Francois Romieu07d3f512007-02-21 22:40:46 +01001552 unsigned int i;
Alexey Dobriyan350f7592009-11-25 15:54:21 -08001553 static const struct {
Francois Romieu61a4dcc2006-02-23 00:55:25 +01001554 u32 opt;
1555 u16 reg;
1556 u8 mask;
1557 } cfg[] = {
Francois Romieu61a4dcc2006-02-23 00:55:25 +01001558 { WAKE_PHY, Config3, LinkUp },
1559 { WAKE_MAGIC, Config3, MagicPacket },
1560 { WAKE_UCAST, Config5, UWF },
1561 { WAKE_BCAST, Config5, BWF },
1562 { WAKE_MCAST, Config5, MWF },
1563 { WAKE_ANY, Config5, LanWake }
1564 };
Francois Romieu851e6022012-04-17 11:10:11 +02001565 u8 options;
Francois Romieu61a4dcc2006-02-23 00:55:25 +01001566
Francois Romieu61a4dcc2006-02-23 00:55:25 +01001567 RTL_W8(Cfg9346, Cfg9346_Unlock);
1568
1569 for (i = 0; i < ARRAY_SIZE(cfg); i++) {
Francois Romieu851e6022012-04-17 11:10:11 +02001570 options = RTL_R8(cfg[i].reg) & ~cfg[i].mask;
Rafael J. Wysockie1759442010-03-14 14:33:51 +00001571 if (wolopts & cfg[i].opt)
Francois Romieu61a4dcc2006-02-23 00:55:25 +01001572 options |= cfg[i].mask;
1573 RTL_W8(cfg[i].reg, options);
1574 }
1575
Francois Romieu851e6022012-04-17 11:10:11 +02001576 switch (tp->mac_version) {
1577 case RTL_GIGA_MAC_VER_01 ... RTL_GIGA_MAC_VER_17:
1578 options = RTL_R8(Config1) & ~PMEnable;
1579 if (wolopts)
1580 options |= PMEnable;
1581 RTL_W8(Config1, options);
1582 break;
1583 default:
Francois Romieud387b422012-04-17 11:12:01 +02001584 options = RTL_R8(Config2) & ~PME_SIGNAL;
1585 if (wolopts)
1586 options |= PME_SIGNAL;
1587 RTL_W8(Config2, options);
Francois Romieu851e6022012-04-17 11:10:11 +02001588 break;
1589 }
1590
Francois Romieu61a4dcc2006-02-23 00:55:25 +01001591 RTL_W8(Cfg9346, Cfg9346_Lock);
Rafael J. Wysockie1759442010-03-14 14:33:51 +00001592}
1593
1594static int rtl8169_set_wol(struct net_device *dev, struct ethtool_wolinfo *wol)
1595{
1596 struct rtl8169_private *tp = netdev_priv(dev);
1597
Francois Romieuda78dbf2012-01-26 14:18:23 +01001598 rtl_lock_work(tp);
Francois Romieu61a4dcc2006-02-23 00:55:25 +01001599
Francois Romieuf23e7fd2007-10-04 22:36:14 +02001600 if (wol->wolopts)
1601 tp->features |= RTL_FEATURE_WOL;
1602 else
1603 tp->features &= ~RTL_FEATURE_WOL;
Rafael J. Wysockie1759442010-03-14 14:33:51 +00001604 __rtl8169_set_wol(tp, wol->wolopts);
Francois Romieuda78dbf2012-01-26 14:18:23 +01001605
1606 rtl_unlock_work(tp);
Francois Romieu61a4dcc2006-02-23 00:55:25 +01001607
françois romieuea809072010-11-08 13:23:58 +00001608 device_set_wakeup_enable(&tp->pci_dev->dev, wol->wolopts);
1609
Francois Romieu61a4dcc2006-02-23 00:55:25 +01001610 return 0;
1611}
1612
Francois Romieu31bd2042011-04-26 18:58:59 +02001613static const char *rtl_lookup_firmware_name(struct rtl8169_private *tp)
1614{
Francois Romieu85bffe62011-04-27 08:22:39 +02001615 return rtl_chip_infos[tp->mac_version].fw_name;
Francois Romieu31bd2042011-04-26 18:58:59 +02001616}
1617
Linus Torvalds1da177e2005-04-16 15:20:36 -07001618static void rtl8169_get_drvinfo(struct net_device *dev,
1619 struct ethtool_drvinfo *info)
1620{
1621 struct rtl8169_private *tp = netdev_priv(dev);
Francois Romieub6ffd972011-06-17 17:00:05 +02001622 struct rtl_fw *rtl_fw = tp->rtl_fw;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001623
Rick Jones68aad782011-11-07 13:29:27 +00001624 strlcpy(info->driver, MODULENAME, sizeof(info->driver));
1625 strlcpy(info->version, RTL8169_VERSION, sizeof(info->version));
1626 strlcpy(info->bus_info, pci_name(tp->pci_dev), sizeof(info->bus_info));
Francois Romieu1c361ef2011-06-17 17:16:24 +02001627 BUILD_BUG_ON(sizeof(info->fw_version) < sizeof(rtl_fw->version));
Rick Jones8ac72d12011-11-22 14:06:26 +00001628 if (!IS_ERR_OR_NULL(rtl_fw))
1629 strlcpy(info->fw_version, rtl_fw->version,
1630 sizeof(info->fw_version));
Linus Torvalds1da177e2005-04-16 15:20:36 -07001631}
1632
1633static int rtl8169_get_regs_len(struct net_device *dev)
1634{
1635 return R8169_REGS_SIZE;
1636}
1637
1638static int rtl8169_set_speed_tbi(struct net_device *dev,
Oliver Neukum54405cd2011-01-06 21:55:13 +01001639 u8 autoneg, u16 speed, u8 duplex, u32 ignored)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001640{
1641 struct rtl8169_private *tp = netdev_priv(dev);
1642 void __iomem *ioaddr = tp->mmio_addr;
1643 int ret = 0;
1644 u32 reg;
1645
1646 reg = RTL_R32(TBICSR);
1647 if ((autoneg == AUTONEG_DISABLE) && (speed == SPEED_1000) &&
1648 (duplex == DUPLEX_FULL)) {
1649 RTL_W32(TBICSR, reg & ~(TBINwEnable | TBINwRestart));
1650 } else if (autoneg == AUTONEG_ENABLE)
1651 RTL_W32(TBICSR, reg | TBINwEnable | TBINwRestart);
1652 else {
Joe Perchesbf82c182010-02-09 11:49:50 +00001653 netif_warn(tp, link, dev,
1654 "incorrect speed setting refused in TBI mode\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07001655 ret = -EOPNOTSUPP;
1656 }
1657
1658 return ret;
1659}
1660
1661static int rtl8169_set_speed_xmii(struct net_device *dev,
Oliver Neukum54405cd2011-01-06 21:55:13 +01001662 u8 autoneg, u16 speed, u8 duplex, u32 adv)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001663{
1664 struct rtl8169_private *tp = netdev_priv(dev);
françois romieu3577aa12009-05-19 10:46:48 +00001665 int giga_ctrl, bmcr;
Oliver Neukum54405cd2011-01-06 21:55:13 +01001666 int rc = -EINVAL;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001667
Hayes Wang716b50a2011-02-22 17:26:18 +08001668 rtl_writephy(tp, 0x1f, 0x0000);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001669
1670 if (autoneg == AUTONEG_ENABLE) {
françois romieu3577aa12009-05-19 10:46:48 +00001671 int auto_nego;
1672
françois romieu4da19632011-01-03 15:07:55 +00001673 auto_nego = rtl_readphy(tp, MII_ADVERTISE);
Oliver Neukum54405cd2011-01-06 21:55:13 +01001674 auto_nego &= ~(ADVERTISE_10HALF | ADVERTISE_10FULL |
1675 ADVERTISE_100HALF | ADVERTISE_100FULL);
1676
1677 if (adv & ADVERTISED_10baseT_Half)
1678 auto_nego |= ADVERTISE_10HALF;
1679 if (adv & ADVERTISED_10baseT_Full)
1680 auto_nego |= ADVERTISE_10FULL;
1681 if (adv & ADVERTISED_100baseT_Half)
1682 auto_nego |= ADVERTISE_100HALF;
1683 if (adv & ADVERTISED_100baseT_Full)
1684 auto_nego |= ADVERTISE_100FULL;
1685
françois romieu3577aa12009-05-19 10:46:48 +00001686 auto_nego |= ADVERTISE_PAUSE_CAP | ADVERTISE_PAUSE_ASYM;
1687
françois romieu4da19632011-01-03 15:07:55 +00001688 giga_ctrl = rtl_readphy(tp, MII_CTRL1000);
françois romieu3577aa12009-05-19 10:46:48 +00001689 giga_ctrl &= ~(ADVERTISE_1000FULL | ADVERTISE_1000HALF);
1690
1691 /* The 8100e/8101e/8102e do Fast Ethernet only. */
Francois Romieu826e6cb2011-03-11 20:30:24 +01001692 if (tp->mii.supports_gmii) {
Oliver Neukum54405cd2011-01-06 21:55:13 +01001693 if (adv & ADVERTISED_1000baseT_Half)
1694 giga_ctrl |= ADVERTISE_1000HALF;
1695 if (adv & ADVERTISED_1000baseT_Full)
1696 giga_ctrl |= ADVERTISE_1000FULL;
1697 } else if (adv & (ADVERTISED_1000baseT_Half |
1698 ADVERTISED_1000baseT_Full)) {
Joe Perchesbf82c182010-02-09 11:49:50 +00001699 netif_info(tp, link, dev,
1700 "PHY does not support 1000Mbps\n");
Oliver Neukum54405cd2011-01-06 21:55:13 +01001701 goto out;
Francois Romieubcf0bf92006-07-26 23:14:13 +02001702 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07001703
françois romieu3577aa12009-05-19 10:46:48 +00001704 bmcr = BMCR_ANENABLE | BMCR_ANRESTART;
Francois Romieu623a1592006-05-14 12:42:14 +02001705
françois romieu4da19632011-01-03 15:07:55 +00001706 rtl_writephy(tp, MII_ADVERTISE, auto_nego);
1707 rtl_writephy(tp, MII_CTRL1000, giga_ctrl);
françois romieu3577aa12009-05-19 10:46:48 +00001708 } else {
1709 giga_ctrl = 0;
1710
1711 if (speed == SPEED_10)
1712 bmcr = 0;
1713 else if (speed == SPEED_100)
1714 bmcr = BMCR_SPEED100;
1715 else
Oliver Neukum54405cd2011-01-06 21:55:13 +01001716 goto out;
françois romieu3577aa12009-05-19 10:46:48 +00001717
1718 if (duplex == DUPLEX_FULL)
1719 bmcr |= BMCR_FULLDPLX;
Roger So2584fbc2007-07-31 23:52:42 +02001720 }
1721
françois romieu4da19632011-01-03 15:07:55 +00001722 rtl_writephy(tp, MII_BMCR, bmcr);
françois romieu3577aa12009-05-19 10:46:48 +00001723
Francois Romieucecb5fd2011-04-01 10:21:07 +02001724 if (tp->mac_version == RTL_GIGA_MAC_VER_02 ||
1725 tp->mac_version == RTL_GIGA_MAC_VER_03) {
françois romieu3577aa12009-05-19 10:46:48 +00001726 if ((speed == SPEED_100) && (autoneg != AUTONEG_ENABLE)) {
françois romieu4da19632011-01-03 15:07:55 +00001727 rtl_writephy(tp, 0x17, 0x2138);
1728 rtl_writephy(tp, 0x0e, 0x0260);
françois romieu3577aa12009-05-19 10:46:48 +00001729 } else {
françois romieu4da19632011-01-03 15:07:55 +00001730 rtl_writephy(tp, 0x17, 0x2108);
1731 rtl_writephy(tp, 0x0e, 0x0000);
françois romieu3577aa12009-05-19 10:46:48 +00001732 }
1733 }
1734
Oliver Neukum54405cd2011-01-06 21:55:13 +01001735 rc = 0;
1736out:
1737 return rc;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001738}
1739
1740static int rtl8169_set_speed(struct net_device *dev,
Oliver Neukum54405cd2011-01-06 21:55:13 +01001741 u8 autoneg, u16 speed, u8 duplex, u32 advertising)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001742{
1743 struct rtl8169_private *tp = netdev_priv(dev);
1744 int ret;
1745
Oliver Neukum54405cd2011-01-06 21:55:13 +01001746 ret = tp->set_speed(dev, autoneg, speed, duplex, advertising);
Francois Romieu4876cc12011-03-11 21:07:11 +01001747 if (ret < 0)
1748 goto out;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001749
Francois Romieu4876cc12011-03-11 21:07:11 +01001750 if (netif_running(dev) && (autoneg == AUTONEG_ENABLE) &&
1751 (advertising & ADVERTISED_1000baseT_Full)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07001752 mod_timer(&tp->timer, jiffies + RTL8169_PHY_TIMEOUT);
Francois Romieu4876cc12011-03-11 21:07:11 +01001753 }
1754out:
Linus Torvalds1da177e2005-04-16 15:20:36 -07001755 return ret;
1756}
1757
1758static int rtl8169_set_settings(struct net_device *dev, struct ethtool_cmd *cmd)
1759{
1760 struct rtl8169_private *tp = netdev_priv(dev);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001761 int ret;
1762
Francois Romieu4876cc12011-03-11 21:07:11 +01001763 del_timer_sync(&tp->timer);
1764
Francois Romieuda78dbf2012-01-26 14:18:23 +01001765 rtl_lock_work(tp);
Francois Romieucecb5fd2011-04-01 10:21:07 +02001766 ret = rtl8169_set_speed(dev, cmd->autoneg, ethtool_cmd_speed(cmd),
David Decotigny25db0332011-04-27 18:32:39 +00001767 cmd->duplex, cmd->advertising);
Francois Romieuda78dbf2012-01-26 14:18:23 +01001768 rtl_unlock_work(tp);
Francois Romieu5b0384f2006-08-16 16:00:01 +02001769
Linus Torvalds1da177e2005-04-16 15:20:36 -07001770 return ret;
1771}
1772
Michał Mirosławc8f44af2011-11-15 15:29:55 +00001773static netdev_features_t rtl8169_fix_features(struct net_device *dev,
1774 netdev_features_t features)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001775{
Francois Romieud58d46b2011-05-03 16:38:29 +02001776 struct rtl8169_private *tp = netdev_priv(dev);
1777
Francois Romieu2b7b4312011-04-18 22:53:24 -07001778 if (dev->mtu > TD_MSS_MAX)
Michał Mirosław350fb322011-04-08 06:35:56 +00001779 features &= ~NETIF_F_ALL_TSO;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001780
Francois Romieud58d46b2011-05-03 16:38:29 +02001781 if (dev->mtu > JUMBO_1K &&
1782 !rtl_chip_infos[tp->mac_version].jumbo_tx_csum)
1783 features &= ~NETIF_F_IP_CSUM;
1784
Michał Mirosław350fb322011-04-08 06:35:56 +00001785 return features;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001786}
1787
Francois Romieuda78dbf2012-01-26 14:18:23 +01001788static void __rtl8169_set_features(struct net_device *dev,
1789 netdev_features_t features)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001790{
1791 struct rtl8169_private *tp = netdev_priv(dev);
Ben Greear6bbe0212012-02-10 15:04:33 +00001792 netdev_features_t changed = features ^ dev->features;
Francois Romieuda78dbf2012-01-26 14:18:23 +01001793 void __iomem *ioaddr = tp->mmio_addr;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001794
Ben Greear6bbe0212012-02-10 15:04:33 +00001795 if (!(changed & (NETIF_F_RXALL | NETIF_F_RXCSUM | NETIF_F_HW_VLAN_RX)))
1796 return;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001797
Ben Greear6bbe0212012-02-10 15:04:33 +00001798 if (changed & (NETIF_F_RXCSUM | NETIF_F_HW_VLAN_RX)) {
1799 if (features & NETIF_F_RXCSUM)
1800 tp->cp_cmd |= RxChkSum;
1801 else
1802 tp->cp_cmd &= ~RxChkSum;
Michał Mirosław350fb322011-04-08 06:35:56 +00001803
Ben Greear6bbe0212012-02-10 15:04:33 +00001804 if (dev->features & NETIF_F_HW_VLAN_RX)
1805 tp->cp_cmd |= RxVlan;
1806 else
1807 tp->cp_cmd &= ~RxVlan;
1808
1809 RTL_W16(CPlusCmd, tp->cp_cmd);
1810 RTL_R16(CPlusCmd);
1811 }
1812 if (changed & NETIF_F_RXALL) {
1813 int tmp = (RTL_R32(RxConfig) & ~(AcceptErr | AcceptRunt));
1814 if (features & NETIF_F_RXALL)
1815 tmp |= (AcceptErr | AcceptRunt);
1816 RTL_W32(RxConfig, tmp);
1817 }
Francois Romieuda78dbf2012-01-26 14:18:23 +01001818}
Linus Torvalds1da177e2005-04-16 15:20:36 -07001819
Francois Romieuda78dbf2012-01-26 14:18:23 +01001820static int rtl8169_set_features(struct net_device *dev,
1821 netdev_features_t features)
1822{
1823 struct rtl8169_private *tp = netdev_priv(dev);
1824
1825 rtl_lock_work(tp);
1826 __rtl8169_set_features(dev, features);
1827 rtl_unlock_work(tp);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001828
1829 return 0;
1830}
1831
Francois Romieuda78dbf2012-01-26 14:18:23 +01001832
Linus Torvalds1da177e2005-04-16 15:20:36 -07001833static inline u32 rtl8169_tx_vlan_tag(struct rtl8169_private *tp,
1834 struct sk_buff *skb)
1835{
Jesse Grosseab6d182010-10-20 13:56:03 +00001836 return (vlan_tx_tag_present(skb)) ?
Linus Torvalds1da177e2005-04-16 15:20:36 -07001837 TxVlanTag | swab16(vlan_tx_tag_get(skb)) : 0x00;
1838}
1839
Francois Romieu7a8fc772011-03-01 17:18:33 +01001840static void rtl8169_rx_vlan_tag(struct RxDesc *desc, struct sk_buff *skb)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001841{
1842 u32 opts2 = le32_to_cpu(desc->opts2);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001843
Francois Romieu7a8fc772011-03-01 17:18:33 +01001844 if (opts2 & RxVlanTag)
1845 __vlan_hwaccel_put_tag(skb, swab16(opts2 & 0xffff));
Eric Dumazet2edae082010-09-06 18:46:39 +00001846
Linus Torvalds1da177e2005-04-16 15:20:36 -07001847 desc->opts2 = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001848}
1849
Francois Romieuccdffb92008-07-26 14:26:06 +02001850static int rtl8169_gset_tbi(struct net_device *dev, struct ethtool_cmd *cmd)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001851{
1852 struct rtl8169_private *tp = netdev_priv(dev);
1853 void __iomem *ioaddr = tp->mmio_addr;
1854 u32 status;
1855
1856 cmd->supported =
1857 SUPPORTED_1000baseT_Full | SUPPORTED_Autoneg | SUPPORTED_FIBRE;
1858 cmd->port = PORT_FIBRE;
1859 cmd->transceiver = XCVR_INTERNAL;
1860
1861 status = RTL_R32(TBICSR);
1862 cmd->advertising = (status & TBINwEnable) ? ADVERTISED_Autoneg : 0;
1863 cmd->autoneg = !!(status & TBINwEnable);
1864
David Decotigny70739492011-04-27 18:32:40 +00001865 ethtool_cmd_speed_set(cmd, SPEED_1000);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001866 cmd->duplex = DUPLEX_FULL; /* Always set */
Francois Romieuccdffb92008-07-26 14:26:06 +02001867
1868 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001869}
1870
Francois Romieuccdffb92008-07-26 14:26:06 +02001871static int rtl8169_gset_xmii(struct net_device *dev, struct ethtool_cmd *cmd)
Linus Torvalds1da177e2005-04-16 15:20:36 -07001872{
1873 struct rtl8169_private *tp = netdev_priv(dev);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001874
Francois Romieuccdffb92008-07-26 14:26:06 +02001875 return mii_ethtool_gset(&tp->mii, cmd);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001876}
1877
1878static int rtl8169_get_settings(struct net_device *dev, struct ethtool_cmd *cmd)
1879{
1880 struct rtl8169_private *tp = netdev_priv(dev);
Francois Romieuccdffb92008-07-26 14:26:06 +02001881 int rc;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001882
Francois Romieuda78dbf2012-01-26 14:18:23 +01001883 rtl_lock_work(tp);
Francois Romieuccdffb92008-07-26 14:26:06 +02001884 rc = tp->get_settings(dev, cmd);
Francois Romieuda78dbf2012-01-26 14:18:23 +01001885 rtl_unlock_work(tp);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001886
Francois Romieuccdffb92008-07-26 14:26:06 +02001887 return rc;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001888}
1889
1890static void rtl8169_get_regs(struct net_device *dev, struct ethtool_regs *regs,
1891 void *p)
1892{
Francois Romieu5b0384f2006-08-16 16:00:01 +02001893 struct rtl8169_private *tp = netdev_priv(dev);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001894
Francois Romieu5b0384f2006-08-16 16:00:01 +02001895 if (regs->len > R8169_REGS_SIZE)
1896 regs->len = R8169_REGS_SIZE;
Linus Torvalds1da177e2005-04-16 15:20:36 -07001897
Francois Romieuda78dbf2012-01-26 14:18:23 +01001898 rtl_lock_work(tp);
Francois Romieu5b0384f2006-08-16 16:00:01 +02001899 memcpy_fromio(p, tp->mmio_addr, regs->len);
Francois Romieuda78dbf2012-01-26 14:18:23 +01001900 rtl_unlock_work(tp);
Linus Torvalds1da177e2005-04-16 15:20:36 -07001901}
1902
Stephen Hemmingerb57b7e52005-05-27 21:11:52 +02001903static u32 rtl8169_get_msglevel(struct net_device *dev)
1904{
1905 struct rtl8169_private *tp = netdev_priv(dev);
1906
1907 return tp->msg_enable;
1908}
1909
1910static void rtl8169_set_msglevel(struct net_device *dev, u32 value)
1911{
1912 struct rtl8169_private *tp = netdev_priv(dev);
1913
1914 tp->msg_enable = value;
1915}
1916
Stephen Hemmingerd4a3a0f2005-05-27 21:11:56 +02001917static const char rtl8169_gstrings[][ETH_GSTRING_LEN] = {
1918 "tx_packets",
1919 "rx_packets",
1920 "tx_errors",
1921 "rx_errors",
1922 "rx_missed",
1923 "align_errors",
1924 "tx_single_collisions",
1925 "tx_multi_collisions",
1926 "unicast",
1927 "broadcast",
1928 "multicast",
1929 "tx_aborted",
1930 "tx_underrun",
1931};
1932
Jeff Garzikb9f2c042007-10-03 18:07:32 -07001933static int rtl8169_get_sset_count(struct net_device *dev, int sset)
Stephen Hemmingerd4a3a0f2005-05-27 21:11:56 +02001934{
Jeff Garzikb9f2c042007-10-03 18:07:32 -07001935 switch (sset) {
1936 case ETH_SS_STATS:
1937 return ARRAY_SIZE(rtl8169_gstrings);
1938 default:
1939 return -EOPNOTSUPP;
1940 }
Stephen Hemmingerd4a3a0f2005-05-27 21:11:56 +02001941}
1942
Francois Romieuffc46952012-07-06 14:19:23 +02001943DECLARE_RTL_COND(rtl_counters_cond)
1944{
1945 void __iomem *ioaddr = tp->mmio_addr;
1946
1947 return RTL_R32(CounterAddrLow) & CounterDump;
1948}
1949
Ivan Vecera355423d2009-02-06 21:49:57 -08001950static void rtl8169_update_counters(struct net_device *dev)
Stephen Hemmingerd4a3a0f2005-05-27 21:11:56 +02001951{
1952 struct rtl8169_private *tp = netdev_priv(dev);
1953 void __iomem *ioaddr = tp->mmio_addr;
Francois Romieucecb5fd2011-04-01 10:21:07 +02001954 struct device *d = &tp->pci_dev->dev;
Stephen Hemmingerd4a3a0f2005-05-27 21:11:56 +02001955 struct rtl8169_counters *counters;
1956 dma_addr_t paddr;
1957 u32 cmd;
1958
Ivan Vecera355423d2009-02-06 21:49:57 -08001959 /*
1960 * Some chips are unable to dump tally counters when the receiver
1961 * is disabled.
1962 */
1963 if ((RTL_R8(ChipCmd) & CmdRxEnb) == 0)
1964 return;
Stephen Hemmingerd4a3a0f2005-05-27 21:11:56 +02001965
Stanislaw Gruszka48addcc2010-10-20 22:25:39 +00001966 counters = dma_alloc_coherent(d, sizeof(*counters), &paddr, GFP_KERNEL);
Stephen Hemmingerd4a3a0f2005-05-27 21:11:56 +02001967 if (!counters)
1968 return;
1969
1970 RTL_W32(CounterAddrHigh, (u64)paddr >> 32);
Yang Hongyang284901a2009-04-06 19:01:15 -07001971 cmd = (u64)paddr & DMA_BIT_MASK(32);
Stephen Hemmingerd4a3a0f2005-05-27 21:11:56 +02001972 RTL_W32(CounterAddrLow, cmd);
1973 RTL_W32(CounterAddrLow, cmd | CounterDump);
1974
Francois Romieuffc46952012-07-06 14:19:23 +02001975 if (rtl_udelay_loop_wait_low(tp, &rtl_counters_cond, 10, 1000))
1976 memcpy(&tp->counters, counters, sizeof(*counters));
Stephen Hemmingerd4a3a0f2005-05-27 21:11:56 +02001977
1978 RTL_W32(CounterAddrLow, 0);
1979 RTL_W32(CounterAddrHigh, 0);
1980
Stanislaw Gruszka48addcc2010-10-20 22:25:39 +00001981 dma_free_coherent(d, sizeof(*counters), counters, paddr);
Stephen Hemmingerd4a3a0f2005-05-27 21:11:56 +02001982}
1983
Ivan Vecera355423d2009-02-06 21:49:57 -08001984static void rtl8169_get_ethtool_stats(struct net_device *dev,
1985 struct ethtool_stats *stats, u64 *data)
1986{
1987 struct rtl8169_private *tp = netdev_priv(dev);
1988
1989 ASSERT_RTNL();
1990
1991 rtl8169_update_counters(dev);
1992
1993 data[0] = le64_to_cpu(tp->counters.tx_packets);
1994 data[1] = le64_to_cpu(tp->counters.rx_packets);
1995 data[2] = le64_to_cpu(tp->counters.tx_errors);
1996 data[3] = le32_to_cpu(tp->counters.rx_errors);
1997 data[4] = le16_to_cpu(tp->counters.rx_missed);
1998 data[5] = le16_to_cpu(tp->counters.align_errors);
1999 data[6] = le32_to_cpu(tp->counters.tx_one_collision);
2000 data[7] = le32_to_cpu(tp->counters.tx_multi_collision);
2001 data[8] = le64_to_cpu(tp->counters.rx_unicast);
2002 data[9] = le64_to_cpu(tp->counters.rx_broadcast);
2003 data[10] = le32_to_cpu(tp->counters.rx_multicast);
2004 data[11] = le16_to_cpu(tp->counters.tx_aborted);
2005 data[12] = le16_to_cpu(tp->counters.tx_underun);
2006}
2007
Stephen Hemmingerd4a3a0f2005-05-27 21:11:56 +02002008static void rtl8169_get_strings(struct net_device *dev, u32 stringset, u8 *data)
2009{
2010 switch(stringset) {
2011 case ETH_SS_STATS:
2012 memcpy(data, *rtl8169_gstrings, sizeof(rtl8169_gstrings));
2013 break;
2014 }
2015}
2016
Jeff Garzik7282d492006-09-13 14:30:00 -04002017static const struct ethtool_ops rtl8169_ethtool_ops = {
Linus Torvalds1da177e2005-04-16 15:20:36 -07002018 .get_drvinfo = rtl8169_get_drvinfo,
2019 .get_regs_len = rtl8169_get_regs_len,
2020 .get_link = ethtool_op_get_link,
2021 .get_settings = rtl8169_get_settings,
2022 .set_settings = rtl8169_set_settings,
Stephen Hemmingerb57b7e52005-05-27 21:11:52 +02002023 .get_msglevel = rtl8169_get_msglevel,
2024 .set_msglevel = rtl8169_set_msglevel,
Linus Torvalds1da177e2005-04-16 15:20:36 -07002025 .get_regs = rtl8169_get_regs,
Francois Romieu61a4dcc2006-02-23 00:55:25 +01002026 .get_wol = rtl8169_get_wol,
2027 .set_wol = rtl8169_set_wol,
Stephen Hemmingerd4a3a0f2005-05-27 21:11:56 +02002028 .get_strings = rtl8169_get_strings,
Jeff Garzikb9f2c042007-10-03 18:07:32 -07002029 .get_sset_count = rtl8169_get_sset_count,
Stephen Hemmingerd4a3a0f2005-05-27 21:11:56 +02002030 .get_ethtool_stats = rtl8169_get_ethtool_stats,
Richard Cochrane1593bb2012-04-03 22:59:35 +00002031 .get_ts_info = ethtool_op_get_ts_info,
Linus Torvalds1da177e2005-04-16 15:20:36 -07002032};
2033
Francois Romieu07d3f512007-02-21 22:40:46 +01002034static void rtl8169_get_mac_version(struct rtl8169_private *tp,
Francois Romieu5d320a22011-05-08 17:47:36 +02002035 struct net_device *dev, u8 default_version)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002036{
Francois Romieu5d320a22011-05-08 17:47:36 +02002037 void __iomem *ioaddr = tp->mmio_addr;
Francois Romieu0e485152007-02-20 00:00:26 +01002038 /*
2039 * The driver currently handles the 8168Bf and the 8168Be identically
2040 * but they can be identified more specifically through the test below
2041 * if needed:
2042 *
2043 * (RTL_R32(TxConfig) & 0x700000) == 0x500000 ? 8168Bf : 8168Be
Francois Romieu01272152007-02-20 22:58:51 +01002044 *
2045 * Same thing for the 8101Eb and the 8101Ec:
2046 *
2047 * (RTL_R32(TxConfig) & 0x700000) == 0x200000 ? 8101Eb : 8101Ec
Francois Romieu0e485152007-02-20 00:00:26 +01002048 */
Francois Romieu37441002011-06-17 22:58:54 +02002049 static const struct rtl_mac_info {
Linus Torvalds1da177e2005-04-16 15:20:36 -07002050 u32 mask;
Francois Romieue3cf0cc2007-08-17 14:55:46 +02002051 u32 val;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002052 int mac_version;
2053 } mac_info[] = {
Hayes Wangc5583862012-07-02 17:23:22 +08002054 /* 8168G family. */
2055 { 0x7cf00000, 0x4c100000, RTL_GIGA_MAC_VER_41 },
2056 { 0x7cf00000, 0x4c000000, RTL_GIGA_MAC_VER_40 },
2057
Hayes Wangc2218922011-09-06 16:55:18 +08002058 /* 8168F family. */
Hayes Wangb3d7b2f2012-03-30 14:48:06 +08002059 { 0x7c800000, 0x48800000, RTL_GIGA_MAC_VER_38 },
Hayes Wangc2218922011-09-06 16:55:18 +08002060 { 0x7cf00000, 0x48100000, RTL_GIGA_MAC_VER_36 },
2061 { 0x7cf00000, 0x48000000, RTL_GIGA_MAC_VER_35 },
2062
hayeswang01dc7fe2011-03-21 01:50:28 +00002063 /* 8168E family. */
Hayes Wang70090422011-07-06 15:58:06 +08002064 { 0x7c800000, 0x2c800000, RTL_GIGA_MAC_VER_34 },
hayeswang01dc7fe2011-03-21 01:50:28 +00002065 { 0x7cf00000, 0x2c200000, RTL_GIGA_MAC_VER_33 },
2066 { 0x7cf00000, 0x2c100000, RTL_GIGA_MAC_VER_32 },
2067 { 0x7c800000, 0x2c000000, RTL_GIGA_MAC_VER_33 },
2068
Francois Romieu5b538df2008-07-20 16:22:45 +02002069 /* 8168D family. */
françois romieudaf9df62009-10-07 12:44:20 +00002070 { 0x7cf00000, 0x28300000, RTL_GIGA_MAC_VER_26 },
2071 { 0x7cf00000, 0x28100000, RTL_GIGA_MAC_VER_25 },
françois romieudaf9df62009-10-07 12:44:20 +00002072 { 0x7c800000, 0x28000000, RTL_GIGA_MAC_VER_26 },
Francois Romieu5b538df2008-07-20 16:22:45 +02002073
françois romieue6de30d2011-01-03 15:08:37 +00002074 /* 8168DP family. */
2075 { 0x7cf00000, 0x28800000, RTL_GIGA_MAC_VER_27 },
2076 { 0x7cf00000, 0x28a00000, RTL_GIGA_MAC_VER_28 },
hayeswang4804b3b2011-03-21 01:50:29 +00002077 { 0x7cf00000, 0x28b00000, RTL_GIGA_MAC_VER_31 },
françois romieue6de30d2011-01-03 15:08:37 +00002078
Francois Romieuef808d52008-06-29 13:10:54 +02002079 /* 8168C family. */
Francois Romieu17c99292010-07-11 17:10:09 -07002080 { 0x7cf00000, 0x3cb00000, RTL_GIGA_MAC_VER_24 },
Francois Romieuef3386f2008-06-29 12:24:30 +02002081 { 0x7cf00000, 0x3c900000, RTL_GIGA_MAC_VER_23 },
Francois Romieuef808d52008-06-29 13:10:54 +02002082 { 0x7cf00000, 0x3c800000, RTL_GIGA_MAC_VER_18 },
Francois Romieu7f3e3d32008-07-20 18:53:20 +02002083 { 0x7c800000, 0x3c800000, RTL_GIGA_MAC_VER_24 },
Francois Romieue3cf0cc2007-08-17 14:55:46 +02002084 { 0x7cf00000, 0x3c000000, RTL_GIGA_MAC_VER_19 },
2085 { 0x7cf00000, 0x3c200000, RTL_GIGA_MAC_VER_20 },
Francois Romieu197ff762008-06-28 13:16:02 +02002086 { 0x7cf00000, 0x3c300000, RTL_GIGA_MAC_VER_21 },
Francois Romieu6fb07052008-06-29 11:54:28 +02002087 { 0x7cf00000, 0x3c400000, RTL_GIGA_MAC_VER_22 },
Francois Romieuef808d52008-06-29 13:10:54 +02002088 { 0x7c800000, 0x3c000000, RTL_GIGA_MAC_VER_22 },
Francois Romieue3cf0cc2007-08-17 14:55:46 +02002089
2090 /* 8168B family. */
2091 { 0x7cf00000, 0x38000000, RTL_GIGA_MAC_VER_12 },
2092 { 0x7cf00000, 0x38500000, RTL_GIGA_MAC_VER_17 },
2093 { 0x7c800000, 0x38000000, RTL_GIGA_MAC_VER_17 },
2094 { 0x7c800000, 0x30000000, RTL_GIGA_MAC_VER_11 },
2095
2096 /* 8101 family. */
Hayes Wang5598bfe2012-07-02 17:23:21 +08002097 { 0x7cf00000, 0x44900000, RTL_GIGA_MAC_VER_39 },
2098 { 0x7c800000, 0x44800000, RTL_GIGA_MAC_VER_39 },
Hayes Wang7e18dca2012-03-30 14:33:02 +08002099 { 0x7c800000, 0x44000000, RTL_GIGA_MAC_VER_37 },
hayeswang36a0e6c2011-03-21 01:50:30 +00002100 { 0x7cf00000, 0x40b00000, RTL_GIGA_MAC_VER_30 },
Hayes Wang5a5e4442011-02-22 17:26:21 +08002101 { 0x7cf00000, 0x40a00000, RTL_GIGA_MAC_VER_30 },
2102 { 0x7cf00000, 0x40900000, RTL_GIGA_MAC_VER_29 },
2103 { 0x7c800000, 0x40800000, RTL_GIGA_MAC_VER_30 },
Francois Romieu2857ffb2008-08-02 21:08:49 +02002104 { 0x7cf00000, 0x34a00000, RTL_GIGA_MAC_VER_09 },
2105 { 0x7cf00000, 0x24a00000, RTL_GIGA_MAC_VER_09 },
2106 { 0x7cf00000, 0x34900000, RTL_GIGA_MAC_VER_08 },
2107 { 0x7cf00000, 0x24900000, RTL_GIGA_MAC_VER_08 },
2108 { 0x7cf00000, 0x34800000, RTL_GIGA_MAC_VER_07 },
2109 { 0x7cf00000, 0x24800000, RTL_GIGA_MAC_VER_07 },
Francois Romieue3cf0cc2007-08-17 14:55:46 +02002110 { 0x7cf00000, 0x34000000, RTL_GIGA_MAC_VER_13 },
Francois Romieu2857ffb2008-08-02 21:08:49 +02002111 { 0x7cf00000, 0x34300000, RTL_GIGA_MAC_VER_10 },
Francois Romieue3cf0cc2007-08-17 14:55:46 +02002112 { 0x7cf00000, 0x34200000, RTL_GIGA_MAC_VER_16 },
Francois Romieu2857ffb2008-08-02 21:08:49 +02002113 { 0x7c800000, 0x34800000, RTL_GIGA_MAC_VER_09 },
2114 { 0x7c800000, 0x24800000, RTL_GIGA_MAC_VER_09 },
Francois Romieue3cf0cc2007-08-17 14:55:46 +02002115 { 0x7c800000, 0x34000000, RTL_GIGA_MAC_VER_16 },
2116 /* FIXME: where did these entries come from ? -- FR */
2117 { 0xfc800000, 0x38800000, RTL_GIGA_MAC_VER_15 },
2118 { 0xfc800000, 0x30800000, RTL_GIGA_MAC_VER_14 },
2119
2120 /* 8110 family. */
2121 { 0xfc800000, 0x98000000, RTL_GIGA_MAC_VER_06 },
2122 { 0xfc800000, 0x18000000, RTL_GIGA_MAC_VER_05 },
2123 { 0xfc800000, 0x10000000, RTL_GIGA_MAC_VER_04 },
2124 { 0xfc800000, 0x04000000, RTL_GIGA_MAC_VER_03 },
2125 { 0xfc800000, 0x00800000, RTL_GIGA_MAC_VER_02 },
2126 { 0xfc800000, 0x00000000, RTL_GIGA_MAC_VER_01 },
2127
Jean Delvaref21b75e2009-05-26 20:54:48 -07002128 /* Catch-all */
2129 { 0x00000000, 0x00000000, RTL_GIGA_MAC_NONE }
Francois Romieu37441002011-06-17 22:58:54 +02002130 };
2131 const struct rtl_mac_info *p = mac_info;
Linus Torvalds1da177e2005-04-16 15:20:36 -07002132 u32 reg;
2133
Francois Romieue3cf0cc2007-08-17 14:55:46 +02002134 reg = RTL_R32(TxConfig);
2135 while ((reg & p->mask) != p->val)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002136 p++;
2137 tp->mac_version = p->mac_version;
Francois Romieu5d320a22011-05-08 17:47:36 +02002138
2139 if (tp->mac_version == RTL_GIGA_MAC_NONE) {
2140 netif_notice(tp, probe, dev,
2141 "unknown MAC, using family default\n");
2142 tp->mac_version = default_version;
2143 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07002144}
2145
2146static void rtl8169_print_mac_version(struct rtl8169_private *tp)
2147{
Francois Romieubcf0bf92006-07-26 23:14:13 +02002148 dprintk("mac_version = 0x%02x\n", tp->mac_version);
Linus Torvalds1da177e2005-04-16 15:20:36 -07002149}
2150
Francois Romieu867763c2007-08-17 18:21:58 +02002151struct phy_reg {
2152 u16 reg;
2153 u16 val;
2154};
2155
françois romieu4da19632011-01-03 15:07:55 +00002156static void rtl_writephy_batch(struct rtl8169_private *tp,
2157 const struct phy_reg *regs, int len)
Francois Romieu867763c2007-08-17 18:21:58 +02002158{
2159 while (len-- > 0) {
françois romieu4da19632011-01-03 15:07:55 +00002160 rtl_writephy(tp, regs->reg, regs->val);
Francois Romieu867763c2007-08-17 18:21:58 +02002161 regs++;
2162 }
2163}
2164
françois romieubca03d52011-01-03 15:07:31 +00002165#define PHY_READ 0x00000000
2166#define PHY_DATA_OR 0x10000000
2167#define PHY_DATA_AND 0x20000000
2168#define PHY_BJMPN 0x30000000
2169#define PHY_READ_EFUSE 0x40000000
2170#define PHY_READ_MAC_BYTE 0x50000000
2171#define PHY_WRITE_MAC_BYTE 0x60000000
2172#define PHY_CLEAR_READCOUNT 0x70000000
2173#define PHY_WRITE 0x80000000
2174#define PHY_READCOUNT_EQ_SKIP 0x90000000
2175#define PHY_COMP_EQ_SKIPN 0xa0000000
2176#define PHY_COMP_NEQ_SKIPN 0xb0000000
2177#define PHY_WRITE_PREVIOUS 0xc0000000
2178#define PHY_SKIPN 0xd0000000
2179#define PHY_DELAY_MS 0xe0000000
2180#define PHY_WRITE_ERI_WORD 0xf0000000
2181
Hayes Wang960aee62011-06-18 11:37:48 +02002182struct fw_info {
2183 u32 magic;
2184 char version[RTL_VER_SIZE];
2185 __le32 fw_start;
2186 __le32 fw_len;
2187 u8 chksum;
2188} __packed;
2189
Francois Romieu1c361ef2011-06-17 17:16:24 +02002190#define FW_OPCODE_SIZE sizeof(typeof(*((struct rtl_fw_phy_action *)0)->code))
2191
2192static bool rtl_fw_format_ok(struct rtl8169_private *tp, struct rtl_fw *rtl_fw)
françois romieubca03d52011-01-03 15:07:31 +00002193{
Francois Romieub6ffd972011-06-17 17:00:05 +02002194 const struct firmware *fw = rtl_fw->fw;
Hayes Wang960aee62011-06-18 11:37:48 +02002195 struct fw_info *fw_info = (struct fw_info *)fw->data;
Francois Romieu1c361ef2011-06-17 17:16:24 +02002196 struct rtl_fw_phy_action *pa = &rtl_fw->phy_action;
2197 char *version = rtl_fw->version;
2198 bool rc = false;
françois romieubca03d52011-01-03 15:07:31 +00002199
Francois Romieu1c361ef2011-06-17 17:16:24 +02002200 if (fw->size < FW_OPCODE_SIZE)
2201 goto out;
Hayes Wang960aee62011-06-18 11:37:48 +02002202
2203 if (!fw_info->magic) {
2204 size_t i, size, start;
2205 u8 checksum = 0;
2206
2207 if (fw->size < sizeof(*fw_info))
2208 goto out;
2209
2210 for (i = 0; i < fw->size; i++)
2211 checksum += fw->data[i];
2212 if (checksum != 0)
2213 goto out;
2214
2215 start = le32_to_cpu(fw_info->fw_start);
2216 if (start > fw->size)
2217 goto out;
2218
2219 size = le32_to_cpu(fw_info->fw_len);
2220 if (size > (fw->size - start) / FW_OPCODE_SIZE)
2221 goto out;
2222
2223 memcpy(version, fw_info->version, RTL_VER_SIZE);
2224
2225 pa->code = (__le32 *)(fw->data + start);
2226 pa->size = size;
2227 } else {
Francois Romieu1c361ef2011-06-17 17:16:24 +02002228 if (fw->size % FW_OPCODE_SIZE)
2229 goto out;
2230
2231 strlcpy(version, rtl_lookup_firmware_name(tp), RTL_VER_SIZE);
2232
2233 pa->code = (__le32 *)fw->data;
2234 pa->size = fw->size / FW_OPCODE_SIZE;
2235 }
2236 version[RTL_VER_SIZE - 1] = 0;
2237
2238 rc = true;
2239out:
2240 return rc;
2241}
2242
Francois Romieufd112f22011-06-18 00:10:29 +02002243static bool rtl_fw_data_ok(struct rtl8169_private *tp, struct net_device *dev,
2244 struct rtl_fw_phy_action *pa)
Francois Romieu1c361ef2011-06-17 17:16:24 +02002245{
Francois Romieufd112f22011-06-18 00:10:29 +02002246 bool rc = false;
Francois Romieu1c361ef2011-06-17 17:16:24 +02002247 size_t index;
2248
Francois Romieu1c361ef2011-06-17 17:16:24 +02002249 for (index = 0; index < pa->size; index++) {
2250 u32 action = le32_to_cpu(pa->code[index]);
hayeswang42b82dc2011-01-10 02:07:25 +00002251 u32 regno = (action & 0x0fff0000) >> 16;
françois romieubca03d52011-01-03 15:07:31 +00002252
hayeswang42b82dc2011-01-10 02:07:25 +00002253 switch(action & 0xf0000000) {
2254 case PHY_READ:
2255 case PHY_DATA_OR:
2256 case PHY_DATA_AND:
2257 case PHY_READ_EFUSE:
2258 case PHY_CLEAR_READCOUNT:
2259 case PHY_WRITE:
2260 case PHY_WRITE_PREVIOUS:
2261 case PHY_DELAY_MS:
françois romieubca03d52011-01-03 15:07:31 +00002262 break;
2263
hayeswang42b82dc2011-01-10 02:07:25 +00002264 case PHY_BJMPN:
2265 if (regno > index) {
Francois Romieufd112f22011-06-18 00:10:29 +02002266 netif_err(tp, ifup, tp->dev,
Francois Romieucecb5fd2011-04-01 10:21:07 +02002267 "Out of range of firmware\n");
Francois Romieufd112f22011-06-18 00:10:29 +02002268 goto out;
hayeswang42b82dc2011-01-10 02:07:25 +00002269 }
2270 break;
2271 case PHY_READCOUNT_EQ_SKIP:
Francois Romieu1c361ef2011-06-17 17:16:24 +02002272 if (index + 2 >= pa->size) {
Francois Romieufd112f22011-06-18 00:10:29 +02002273 netif_err(tp, ifup, tp->dev,
Francois Romieucecb5fd2011-04-01 10:21:07 +02002274 "Out of range of firmware\n");
Francois Romieufd112f22011-06-18 00:10:29 +02002275 goto out;
hayeswang42b82dc2011-01-10 02:07:25 +00002276 }
2277 break;
2278 case PHY_COMP_EQ_SKIPN:
2279 case PHY_COMP_NEQ_SKIPN:
2280 case PHY_SKIPN:
Francois Romieu1c361ef2011-06-17 17:16:24 +02002281 if (index + 1 + regno >= pa->size) {
Francois Romieufd112f22011-06-18 00:10:29 +02002282 netif_err(tp, ifup, tp->dev,
Francois Romieucecb5fd2011-04-01 10:21:07 +02002283 "Out of range of firmware\n");
Francois Romieufd112f22011-06-18 00:10:29 +02002284 goto out;
hayeswang42b82dc2011-01-10 02:07:25 +00002285 }
2286 break;
2287
2288 case PHY_READ_MAC_BYTE:
2289 case PHY_WRITE_MAC_BYTE:
2290 case PHY_WRITE_ERI_WORD:
2291 default:
Francois Romieufd112f22011-06-18 00:10:29 +02002292 netif_err(tp, ifup, tp->dev,
hayeswang42b82dc2011-01-10 02:07:25 +00002293 "Invalid action 0x%08x\n", action);
Francois Romieufd112f22011-06-18 00:10:29 +02002294 goto out;
françois romieubca03d52011-01-03 15:07:31 +00002295 }
2296 }
Francois Romieufd112f22011-06-18 00:10:29 +02002297 rc = true;
2298out:
2299 return rc;
2300}
françois romieubca03d52011-01-03 15:07:31 +00002301
Francois Romieufd112f22011-06-18 00:10:29 +02002302static int rtl_check_firmware(struct rtl8169_private *tp, struct rtl_fw *rtl_fw)
2303{
2304 struct net_device *dev = tp->dev;
2305 int rc = -EINVAL;
2306
2307 if (!rtl_fw_format_ok(tp, rtl_fw)) {
2308 netif_err(tp, ifup, dev, "invalid firwmare\n");
2309 goto out;
2310 }
2311
2312 if (rtl_fw_data_ok(tp, dev, &rtl_fw->phy_action))
2313 rc = 0;
2314out:
2315 return rc;
2316}
2317
2318static void rtl_phy_write_fw(struct rtl8169_private *tp, struct rtl_fw *rtl_fw)
2319{
2320 struct rtl_fw_phy_action *pa = &rtl_fw->phy_action;
2321 u32 predata, count;
2322 size_t index;
2323
2324 predata = count = 0;
hayeswang42b82dc2011-01-10 02:07:25 +00002325
Francois Romieu1c361ef2011-06-17 17:16:24 +02002326 for (index = 0; index < pa->size; ) {
2327 u32 action = le32_to_cpu(pa->code[index]);
françois romieubca03d52011-01-03 15:07:31 +00002328 u32 data = action & 0x0000ffff;
hayeswang42b82dc2011-01-10 02:07:25 +00002329 u32 regno = (action & 0x0fff0000) >> 16;
2330
2331 if (!action)
2332 break;
françois romieubca03d52011-01-03 15:07:31 +00002333
2334 switch(action & 0xf0000000) {
hayeswang42b82dc2011-01-10 02:07:25 +00002335 case PHY_READ:
2336 predata = rtl_readphy(tp, regno);
2337 count++;
2338 index++;
françois romieubca03d52011-01-03 15:07:31 +00002339 break;
hayeswang42b82dc2011-01-10 02:07:25 +00002340 case PHY_DATA_OR:
2341 predata |= data;
2342 index++;
2343 break;
2344 case PHY_DATA_AND:
2345 predata &= data;
2346 index++;
2347 break;
2348 case PHY_BJMPN:
2349 index -= regno;
2350 break;
2351 case PHY_READ_EFUSE:
Francois Romieufdf6fc02012-07-06 22:40:38 +02002352 predata = rtl8168d_efuse_read(tp, regno);
hayeswang42b82dc2011-01-10 02:07:25 +00002353 index++;
2354 break;
2355 case PHY_CLEAR_READCOUNT:
2356 count = 0;
2357 index++;
2358 break;
2359 case PHY_WRITE:
2360 rtl_writephy(tp, regno, data);
2361 index++;
2362 break;
2363 case PHY_READCOUNT_EQ_SKIP:
Francois Romieucecb5fd2011-04-01 10:21:07 +02002364 index += (count == data) ? 2 : 1;
hayeswang42b82dc2011-01-10 02:07:25 +00002365 break;
2366 case PHY_COMP_EQ_SKIPN:
2367 if (predata == data)
2368 index += regno;
2369 index++;
2370 break;
2371 case PHY_COMP_NEQ_SKIPN:
2372 if (predata != data)
2373 index += regno;
2374 index++;
2375 break;
2376 case PHY_WRITE_PREVIOUS:
2377 rtl_writephy(tp, regno, predata);
2378 index++;
2379 break;
2380 case PHY_SKIPN:
2381 index += regno + 1;
2382 break;
2383 case PHY_DELAY_MS:
2384 mdelay(data);
2385 index++;
2386 break;
2387
2388 case PHY_READ_MAC_BYTE:
2389 case PHY_WRITE_MAC_BYTE:
2390 case PHY_WRITE_ERI_WORD:
françois romieubca03d52011-01-03 15:07:31 +00002391 default:
2392 BUG();
2393 }
2394 }
2395}
2396
françois romieuf1e02ed2011-01-13 13:07:53 +00002397static void rtl_release_firmware(struct rtl8169_private *tp)
2398{
Francois Romieub6ffd972011-06-17 17:00:05 +02002399 if (!IS_ERR_OR_NULL(tp->rtl_fw)) {
2400 release_firmware(tp->rtl_fw->fw);
2401 kfree(tp->rtl_fw);
2402 }
2403 tp->rtl_fw = RTL_FIRMWARE_UNKNOWN;
françois romieuf1e02ed2011-01-13 13:07:53 +00002404}
2405
François Romieu953a12c2011-04-24 17:38:48 +02002406static void rtl_apply_firmware(struct rtl8169_private *tp)
françois romieuf1e02ed2011-01-13 13:07:53 +00002407{
Francois Romieub6ffd972011-06-17 17:00:05 +02002408 struct rtl_fw *rtl_fw = tp->rtl_fw;
françois romieuf1e02ed2011-01-13 13:07:53 +00002409
2410 /* TODO: release firmware once rtl_phy_write_fw signals failures. */
Francois Romieub6ffd972011-06-17 17:00:05 +02002411 if (!IS_ERR_OR_NULL(rtl_fw))
2412 rtl_phy_write_fw(tp, rtl_fw);
François Romieu953a12c2011-04-24 17:38:48 +02002413}
2414
2415static void rtl_apply_firmware_cond(struct rtl8169_private *tp, u8 reg, u16 val)
2416{
2417 if (rtl_readphy(tp, reg) != val)
2418 netif_warn(tp, hw, tp->dev, "chipset not ready for firmware\n");
2419 else
2420 rtl_apply_firmware(tp);
françois romieuf1e02ed2011-01-13 13:07:53 +00002421}
2422
françois romieu4da19632011-01-03 15:07:55 +00002423static void rtl8169s_hw_phy_config(struct rtl8169_private *tp)
Linus Torvalds1da177e2005-04-16 15:20:36 -07002424{
Alexey Dobriyan350f7592009-11-25 15:54:21 -08002425 static const struct phy_reg phy_reg_init[] = {
françois romieu0b9b5712009-08-10 19:44:56 +00002426 { 0x1f, 0x0001 },
2427 { 0x06, 0x006e },
2428 { 0x08, 0x0708 },
2429 { 0x15, 0x4000 },
2430 { 0x18, 0x65c7 },
Linus Torvalds1da177e2005-04-16 15:20:36 -07002431
françois romieu0b9b5712009-08-10 19:44:56 +00002432 { 0x1f, 0x0001 },
2433 { 0x03, 0x00a1 },
2434 { 0x02, 0x0008 },
2435 { 0x01, 0x0120 },
2436 { 0x00, 0x1000 },
2437 { 0x04, 0x0800 },
2438 { 0x04, 0x0000 },
Linus Torvalds1da177e2005-04-16 15:20:36 -07002439
françois romieu0b9b5712009-08-10 19:44:56 +00002440 { 0x03, 0xff41 },
2441 { 0x02, 0xdf60 },
2442 { 0x01, 0x0140 },
2443 { 0x00, 0x0077 },
2444 { 0x04, 0x7800 },
2445 { 0x04, 0x7000 },
Linus Torvalds1da177e2005-04-16 15:20:36 -07002446
françois romieu0b9b5712009-08-10 19:44:56 +00002447 { 0x03, 0x802f },
2448 { 0x02, 0x4f02 },
2449 { 0x01, 0x0409 },
2450 { 0x00, 0xf0f9 },
2451 { 0x04, 0x9800 },
2452 { 0x04, 0x9000 },
2453
2454 { 0x03, 0xdf01 },
2455 { 0x02, 0xdf20 },
2456 { 0x01, 0xff95 },
2457 { 0x00, 0xba00 },
2458 { 0x04, 0xa800 },
2459 { 0x04, 0xa000 },
2460
2461 { 0x03, 0xff41 },
2462 { 0x02, 0xdf20 },
2463 { 0x01, 0x0140 },
2464 { 0x00, 0x00bb },
2465 { 0x04, 0xb800 },
2466 { 0x04, 0xb000 },
2467
2468 { 0x03, 0xdf41 },
2469 { 0x02, 0xdc60 },
2470 { 0x01, 0x6340 },
2471 { 0x00, 0x007d },
2472 { 0x04, 0xd800 },
2473 { 0x04, 0xd000 },
2474
2475 { 0x03, 0xdf01 },
2476 { 0x02, 0xdf20 },
2477 { 0x01, 0x100a },
2478 { 0x00, 0xa0ff },
2479 { 0x04, 0xf800 },
2480 { 0x04, 0xf000 },
2481
2482 { 0x1f, 0x0000 },
2483 { 0x0b, 0x0000 },
2484 { 0x00, 0x9200 }
2485 };
2486
françois romieu4da19632011-01-03 15:07:55 +00002487 rtl_writephy_batch(tp, phy_reg_init, ARRAY_SIZE(phy_reg_init));
Linus Torvalds1da177e2005-04-16 15:20:36 -07002488}
2489
françois romieu4da19632011-01-03 15:07:55 +00002490static void rtl8169sb_hw_phy_config(struct rtl8169_private *tp)
Francois Romieu5615d9f2007-08-17 17:50:46 +02002491{
Alexey Dobriyan350f7592009-11-25 15:54:21 -08002492 static const struct phy_reg phy_reg_init[] = {
Francois Romieua441d7b2007-08-17 18:26:35 +02002493 { 0x1f, 0x0002 },
2494 { 0x01, 0x90d0 },
2495 { 0x1f, 0x0000 }
2496 };
2497
françois romieu4da19632011-01-03 15:07:55 +00002498 rtl_writephy_batch(tp, phy_reg_init, ARRAY_SIZE(phy_reg_init));
Francois Romieu5615d9f2007-08-17 17:50:46 +02002499}
2500
françois romieu4da19632011-01-03 15:07:55 +00002501static void rtl8169scd_hw_phy_config_quirk(struct rtl8169_private *tp)
françois romieu2e9558562009-08-10 19:44:19 +00002502{
2503 struct pci_dev *pdev = tp->pci_dev;
françois romieu2e9558562009-08-10 19:44:19 +00002504
Sergei Shtylyovccbae552011-07-22 05:37:24 +00002505 if ((pdev->subsystem_vendor != PCI_VENDOR_ID_GIGABYTE) ||
2506 (pdev->subsystem_device != 0xe000))
françois romieu2e9558562009-08-10 19:44:19 +00002507 return;
2508
françois romieu4da19632011-01-03 15:07:55 +00002509 rtl_writephy(tp, 0x1f, 0x0001);
2510 rtl_writephy(tp, 0x10, 0xf01b);
2511 rtl_writephy(tp, 0x1f, 0x0000);
françois romieu2e9558562009-08-10 19:44:19 +00002512}
2513
françois romieu4da19632011-01-03 15:07:55 +00002514static void rtl8169scd_hw_phy_config(struct rtl8169_private *tp)
françois romieu2e9558562009-08-10 19:44:19 +00002515{
Alexey Dobriyan350f7592009-11-25 15:54:21 -08002516 static const struct phy_reg phy_reg_init[] = {
françois romieu2e9558562009-08-10 19:44:19 +00002517 { 0x1f, 0x0001 },
2518 { 0x04, 0x0000 },
2519 { 0x03, 0x00a1 },
2520 { 0x02, 0x0008 },
2521 { 0x01, 0x0120 },
2522 { 0x00, 0x1000 },
2523 { 0x04, 0x0800 },
2524 { 0x04, 0x9000 },
2525 { 0x03, 0x802f },
2526 { 0x02, 0x4f02 },
2527 { 0x01, 0x0409 },
2528 { 0x00, 0xf099 },
2529 { 0x04, 0x9800 },
2530 { 0x04, 0xa000 },
2531 { 0x03, 0xdf01 },
2532 { 0x02, 0xdf20 },
2533 { 0x01, 0xff95 },
2534 { 0x00, 0xba00 },
2535 { 0x04, 0xa800 },
2536 { 0x04, 0xf000 },
2537 { 0x03, 0xdf01 },
2538 { 0x02, 0xdf20 },
2539 { 0x01, 0x101a },
2540 { 0x00, 0xa0ff },
2541 { 0x04, 0xf800 },
2542 { 0x04, 0x0000 },
2543 { 0x1f, 0x0000 },
2544
2545 { 0x1f, 0x0001 },
2546 { 0x10, 0xf41b },
2547 { 0x14, 0xfb54 },
2548 { 0x18, 0xf5c7 },
2549 { 0x1f, 0x0000 },
2550
2551 { 0x1f, 0x0001 },
2552 { 0x17, 0x0cc0 },
2553 { 0x1f, 0x0000 }
2554 };
2555
françois romieu4da19632011-01-03 15:07:55 +00002556 rtl_writephy_batch(tp, phy_reg_init, ARRAY_SIZE(phy_reg_init));
françois romieu2e9558562009-08-10 19:44:19 +00002557
françois romieu4da19632011-01-03 15:07:55 +00002558 rtl8169scd_hw_phy_config_quirk(tp);
françois romieu2e9558562009-08-10 19:44:19 +00002559}
2560
françois romieu4da19632011-01-03 15:07:55 +00002561static void rtl8169sce_hw_phy_config(struct rtl8169_private *tp)
françois romieu8c7006a2009-08-10 19:43:29 +00002562{
Alexey Dobriyan350f7592009-11-25 15:54:21 -08002563 static const struct phy_reg phy_reg_init[] = {
françois romieu8c7006a2009-08-10 19:43:29 +00002564 { 0x1f, 0x0001 },
2565 { 0x04, 0x0000 },
2566 { 0x03, 0x00a1 },
2567 { 0x02, 0x0008 },
2568 { 0x01, 0x0120 },
2569 { 0x00, 0x1000 },
2570 { 0x04, 0x0800 },
2571 { 0x04, 0x9000 },
2572 { 0x03, 0x802f },
2573 { 0x02, 0x4f02 },
2574 { 0x01, 0x0409 },
2575 { 0x00, 0xf099 },
2576 { 0x04, 0x9800 },
2577 { 0x04, 0xa000 },
2578 { 0x03, 0xdf01 },
2579 { 0x02, 0xdf20 },
2580 { 0x01, 0xff95 },
2581 { 0x00, 0xba00 },
2582 { 0x04, 0xa800 },
2583 { 0x04, 0xf000 },
2584 { 0x03, 0xdf01 },
2585 { 0x02, 0xdf20 },
2586 { 0x01, 0x101a },
2587 { 0x00, 0xa0ff },
2588 { 0x04, 0xf800 },
2589 { 0x04, 0x0000 },
2590 { 0x1f, 0x0000 },
2591
2592 { 0x1f, 0x0001 },
2593 { 0x0b, 0x8480 },
2594 { 0x1f, 0x0000 },
2595
2596 { 0x1f, 0x0001 },
2597 { 0x18, 0x67c7 },
2598 { 0x04, 0x2000 },
2599 { 0x03, 0x002f },
2600 { 0x02, 0x4360 },
2601 { 0x01, 0x0109 },
2602 { 0x00, 0x3022 },
2603 { 0x04, 0x2800 },
2604 { 0x1f, 0x0000 },
2605
2606 { 0x1f, 0x0001 },
2607 { 0x17, 0x0cc0 },
2608 { 0x1f, 0x0000 }
2609 };
2610
françois romieu4da19632011-01-03 15:07:55 +00002611 rtl_writephy_batch(tp, phy_reg_init, ARRAY_SIZE(phy_reg_init));
françois romieu8c7006a2009-08-10 19:43:29 +00002612}
2613
françois romieu4da19632011-01-03 15:07:55 +00002614static void rtl8168bb_hw_phy_config(struct rtl8169_private *tp)
Francois Romieu236b8082008-05-30 16:11:48 +02002615{
Alexey Dobriyan350f7592009-11-25 15:54:21 -08002616 static const struct phy_reg phy_reg_init[] = {
Francois Romieu236b8082008-05-30 16:11:48 +02002617 { 0x10, 0xf41b },
2618 { 0x1f, 0x0000 }
2619 };
2620
françois romieu4da19632011-01-03 15:07:55 +00002621 rtl_writephy(tp, 0x1f, 0x0001);
2622 rtl_patchphy(tp, 0x16, 1 << 0);
Francois Romieu236b8082008-05-30 16:11:48 +02002623
françois romieu4da19632011-01-03 15:07:55 +00002624 rtl_writephy_batch(tp, phy_reg_init, ARRAY_SIZE(phy_reg_init));
Francois Romieu236b8082008-05-30 16:11:48 +02002625}
2626
françois romieu4da19632011-01-03 15:07:55 +00002627static void rtl8168bef_hw_phy_config(struct rtl8169_private *tp)
Francois Romieu236b8082008-05-30 16:11:48 +02002628{
Alexey Dobriyan350f7592009-11-25 15:54:21 -08002629 static const struct phy_reg phy_reg_init[] = {
Francois Romieu236b8082008-05-30 16:11:48 +02002630 { 0x1f, 0x0001 },
2631 { 0x10, 0xf41b },
2632 { 0x1f, 0x0000 }
2633 };
2634
françois romieu4da19632011-01-03 15:07:55 +00002635 rtl_writephy_batch(tp, phy_reg_init, ARRAY_SIZE(phy_reg_init));
Francois Romieu236b8082008-05-30 16:11:48 +02002636}
2637
françois romieu4da19632011-01-03 15:07:55 +00002638static void rtl8168cp_1_hw_phy_config(struct rtl8169_private *tp)
Francois Romieu867763c2007-08-17 18:21:58 +02002639{
Alexey Dobriyan350f7592009-11-25 15:54:21 -08002640 static const struct phy_reg phy_reg_init[] = {
Francois Romieu867763c2007-08-17 18:21:58 +02002641 { 0x1f, 0x0000 },
2642 { 0x1d, 0x0f00 },
2643 { 0x1f, 0x0002 },
2644 { 0x0c, 0x1ec8 },
2645 { 0x1f, 0x0000 }
2646 };
2647
françois romieu4da19632011-01-03 15:07:55 +00002648 rtl_writephy_batch(tp, phy_reg_init, ARRAY_SIZE(phy_reg_init));
Francois Romieu867763c2007-08-17 18:21:58 +02002649}
2650
françois romieu4da19632011-01-03 15:07:55 +00002651static void rtl8168cp_2_hw_phy_config(struct rtl8169_private *tp)
Francois Romieuef3386f2008-06-29 12:24:30 +02002652{
Alexey Dobriyan350f7592009-11-25 15:54:21 -08002653 static const struct phy_reg phy_reg_init[] = {
Francois Romieuef3386f2008-06-29 12:24:30 +02002654 { 0x1f, 0x0001 },
2655 { 0x1d, 0x3d98 },
2656 { 0x1f, 0x0000 }
2657 };
2658
françois romieu4da19632011-01-03 15:07:55 +00002659 rtl_writephy(tp, 0x1f, 0x0000);
2660 rtl_patchphy(tp, 0x14, 1 << 5);
2661 rtl_patchphy(tp, 0x0d, 1 << 5);
Francois Romieuef3386f2008-06-29 12:24:30 +02002662
françois romieu4da19632011-01-03 15:07:55 +00002663 rtl_writephy_batch(tp, phy_reg_init, ARRAY_SIZE(phy_reg_init));
Francois Romieuef3386f2008-06-29 12:24:30 +02002664}
2665
françois romieu4da19632011-01-03 15:07:55 +00002666static void rtl8168c_1_hw_phy_config(struct rtl8169_private *tp)
Francois Romieu867763c2007-08-17 18:21:58 +02002667{
Alexey Dobriyan350f7592009-11-25 15:54:21 -08002668 static const struct phy_reg phy_reg_init[] = {
Francois Romieua3f80672007-10-18 14:35:11 +02002669 { 0x1f, 0x0001 },
2670 { 0x12, 0x2300 },
Francois Romieu867763c2007-08-17 18:21:58 +02002671 { 0x1f, 0x0002 },
2672 { 0x00, 0x88d4 },
2673 { 0x01, 0x82b1 },
2674 { 0x03, 0x7002 },
2675 { 0x08, 0x9e30 },
2676 { 0x09, 0x01f0 },
2677 { 0x0a, 0x5500 },
2678 { 0x0c, 0x00c8 },
2679 { 0x1f, 0x0003 },
2680 { 0x12, 0xc096 },
2681 { 0x16, 0x000a },
Francois Romieuf50d4272008-05-30 16:07:07 +02002682 { 0x1f, 0x0000 },
2683 { 0x1f, 0x0000 },
2684 { 0x09, 0x2000 },
2685 { 0x09, 0x0000 }
Francois Romieu867763c2007-08-17 18:21:58 +02002686 };
2687
françois romieu4da19632011-01-03 15:07:55 +00002688 rtl_writephy_batch(tp, phy_reg_init, ARRAY_SIZE(phy_reg_init));
Francois Romieuf50d4272008-05-30 16:07:07 +02002689
françois romieu4da19632011-01-03 15:07:55 +00002690 rtl_patchphy(tp, 0x14, 1 << 5);
2691 rtl_patchphy(tp, 0x0d, 1 << 5);
2692 rtl_writephy(tp, 0x1f, 0x0000);
Francois Romieu867763c2007-08-17 18:21:58 +02002693}
2694
françois romieu4da19632011-01-03 15:07:55 +00002695static void rtl8168c_2_hw_phy_config(struct rtl8169_private *tp)
Francois Romieu7da97ec2007-10-18 15:20:43 +02002696{
Alexey Dobriyan350f7592009-11-25 15:54:21 -08002697 static const struct phy_reg phy_reg_init[] = {
Francois Romieuf50d4272008-05-30 16:07:07 +02002698 { 0x1f, 0x0001 },
Francois Romieu7da97ec2007-10-18 15:20:43 +02002699 { 0x12, 0x2300 },
Francois Romieuf50d4272008-05-30 16:07:07 +02002700 { 0x03, 0x802f },
2701 { 0x02, 0x4f02 },
2702 { 0x01, 0x0409 },
2703 { 0x00, 0xf099 },
2704 { 0x04, 0x9800 },
2705 { 0x04, 0x9000 },
2706 { 0x1d, 0x3d98 },
Francois Romieu7da97ec2007-10-18 15:20:43 +02002707 { 0x1f, 0x0002 },
2708 { 0x0c, 0x7eb8 },
Francois Romieuf50d4272008-05-30 16:07:07 +02002709 { 0x06, 0x0761 },
2710 { 0x1f, 0x0003 },
2711 { 0x16, 0x0f0a },
Francois Romieu7da97ec2007-10-18 15:20:43 +02002712 { 0x1f, 0x0000 }
2713 };
2714
françois romieu4da19632011-01-03 15:07:55 +00002715 rtl_writephy_batch(tp, phy_reg_init, ARRAY_SIZE(phy_reg_init));
Francois Romieuf50d4272008-05-30 16:07:07 +02002716
françois romieu4da19632011-01-03 15:07:55 +00002717 rtl_patchphy(tp, 0x16, 1 << 0);
2718 rtl_patchphy(tp, 0x14, 1 << 5);
2719 rtl_patchphy(tp, 0x0d, 1 << 5);
2720 rtl_writephy(tp, 0x1f, 0x0000);
Francois Romieu7da97ec2007-10-18 15:20:43 +02002721}
2722
françois romieu4da19632011-01-03 15:07:55 +00002723static void rtl8168c_3_hw_phy_config(struct rtl8169_private *tp)
Francois Romieu197ff762008-06-28 13:16:02 +02002724{
Alexey Dobriyan350f7592009-11-25 15:54:21 -08002725 static const struct phy_reg phy_reg_init[] = {
Francois Romieu197ff762008-06-28 13:16:02 +02002726 { 0x1f, 0x0001 },
2727 { 0x12, 0x2300 },
2728 { 0x1d, 0x3d98 },
2729 { 0x1f, 0x0002 },
2730 { 0x0c, 0x7eb8 },
2731 { 0x06, 0x5461 },
2732 { 0x1f, 0x0003 },
2733 { 0x16, 0x0f0a },
2734 { 0x1f, 0x0000 }
2735 };
2736
françois romieu4da19632011-01-03 15:07:55 +00002737 rtl_writephy_batch(tp, phy_reg_init, ARRAY_SIZE(phy_reg_init));
Francois Romieu197ff762008-06-28 13:16:02 +02002738
françois romieu4da19632011-01-03 15:07:55 +00002739 rtl_patchphy(tp, 0x16, 1 << 0);
2740 rtl_patchphy(tp, 0x14, 1 << 5);
2741 rtl_patchphy(tp, 0x0d, 1 << 5);
2742 rtl_writephy(tp, 0x1f, 0x0000);
Francois Romieu197ff762008-06-28 13:16:02 +02002743}
2744
françois romieu4da19632011-01-03 15:07:55 +00002745static void rtl8168c_4_hw_phy_config(struct rtl8169_private *tp)
Francois Romieu6fb07052008-06-29 11:54:28 +02002746{
françois romieu4da19632011-01-03 15:07:55 +00002747 rtl8168c_3_hw_phy_config(tp);
Francois Romieu6fb07052008-06-29 11:54:28 +02002748}
2749
françois romieubca03d52011-01-03 15:07:31 +00002750static void rtl8168d_1_hw_phy_config(struct rtl8169_private *tp)
Francois Romieu5b538df2008-07-20 16:22:45 +02002751{
Alexey Dobriyan350f7592009-11-25 15:54:21 -08002752 static const struct phy_reg phy_reg_init_0[] = {
françois romieubca03d52011-01-03 15:07:31 +00002753 /* Channel Estimation */
Francois Romieu5b538df2008-07-20 16:22:45 +02002754 { 0x1f, 0x0001 },
françois romieudaf9df62009-10-07 12:44:20 +00002755 { 0x06, 0x4064 },
2756 { 0x07, 0x2863 },
2757 { 0x08, 0x059c },
2758 { 0x09, 0x26b4 },
2759 { 0x0a, 0x6a19 },
2760 { 0x0b, 0xdcc8 },
2761 { 0x10, 0xf06d },
2762 { 0x14, 0x7f68 },
2763 { 0x18, 0x7fd9 },
2764 { 0x1c, 0xf0ff },
2765 { 0x1d, 0x3d9c },
Francois Romieu5b538df2008-07-20 16:22:45 +02002766 { 0x1f, 0x0003 },
françois romieudaf9df62009-10-07 12:44:20 +00002767 { 0x12, 0xf49f },
2768 { 0x13, 0x070b },
2769 { 0x1a, 0x05ad },
françois romieubca03d52011-01-03 15:07:31 +00002770 { 0x14, 0x94c0 },
2771
2772 /*
2773 * Tx Error Issue
Francois Romieucecb5fd2011-04-01 10:21:07 +02002774 * Enhance line driver power
françois romieubca03d52011-01-03 15:07:31 +00002775 */
Francois Romieu5b538df2008-07-20 16:22:45 +02002776 { 0x1f, 0x0002 },
françois romieudaf9df62009-10-07 12:44:20 +00002777 { 0x06, 0x5561 },
Francois Romieu5b538df2008-07-20 16:22:45 +02002778 { 0x1f, 0x0005 },
françois romieudaf9df62009-10-07 12:44:20 +00002779 { 0x05, 0x8332 },
françois romieubca03d52011-01-03 15:07:31 +00002780 { 0x06, 0x5561 },
2781
2782 /*
2783 * Can not link to 1Gbps with bad cable
2784 * Decrease SNR threshold form 21.07dB to 19.04dB
2785 */
2786 { 0x1f, 0x0001 },
2787 { 0x17, 0x0cc0 },
françois romieudaf9df62009-10-07 12:44:20 +00002788
2789 { 0x1f, 0x0000 },
françois romieubca03d52011-01-03 15:07:31 +00002790 { 0x0d, 0xf880 }
Francois Romieu5b538df2008-07-20 16:22:45 +02002791 };
2792
françois romieu4da19632011-01-03 15:07:55 +00002793 rtl_writephy_batch(tp, phy_reg_init_0, ARRAY_SIZE(phy_reg_init_0));
Francois Romieu5b538df2008-07-20 16:22:45 +02002794
françois romieubca03d52011-01-03 15:07:31 +00002795 /*
2796 * Rx Error Issue
2797 * Fine Tune Switching regulator parameter
2798 */
françois romieu4da19632011-01-03 15:07:55 +00002799 rtl_writephy(tp, 0x1f, 0x0002);
2800 rtl_w1w0_phy(tp, 0x0b, 0x0010, 0x00ef);
2801 rtl_w1w0_phy(tp, 0x0c, 0xa200, 0x5d00);
françois romieudaf9df62009-10-07 12:44:20 +00002802
Francois Romieufdf6fc02012-07-06 22:40:38 +02002803 if (rtl8168d_efuse_read(tp, 0x01) == 0xb1) {
Alexey Dobriyan350f7592009-11-25 15:54:21 -08002804 static const struct phy_reg phy_reg_init[] = {
françois romieudaf9df62009-10-07 12:44:20 +00002805 { 0x1f, 0x0002 },
2806 { 0x05, 0x669a },
Francois Romieu5b538df2008-07-20 16:22:45 +02002807 { 0x1f, 0x0005 },
françois romieudaf9df62009-10-07 12:44:20 +00002808 { 0x05, 0x8330 },
2809 { 0x06, 0x669a },
2810 { 0x1f, 0x0002 }
2811 };
2812 int val;
2813
françois romieu4da19632011-01-03 15:07:55 +00002814 rtl_writephy_batch(tp, phy_reg_init, ARRAY_SIZE(phy_reg_init));
françois romieudaf9df62009-10-07 12:44:20 +00002815
françois romieu4da19632011-01-03 15:07:55 +00002816 val = rtl_readphy(tp, 0x0d);
françois romieudaf9df62009-10-07 12:44:20 +00002817
2818 if ((val & 0x00ff) != 0x006c) {
Alexey Dobriyan350f7592009-11-25 15:54:21 -08002819 static const u32 set[] = {
françois romieudaf9df62009-10-07 12:44:20 +00002820 0x0065, 0x0066, 0x0067, 0x0068,
2821 0x0069, 0x006a, 0x006b, 0x006c
2822 };
2823 int i;
2824
françois romieu4da19632011-01-03 15:07:55 +00002825 rtl_writephy(tp, 0x1f, 0x0002);
françois romieudaf9df62009-10-07 12:44:20 +00002826
2827 val &= 0xff00;
2828 for (i = 0; i < ARRAY_SIZE(set); i++)
françois romieu4da19632011-01-03 15:07:55 +00002829 rtl_writephy(tp, 0x0d, val | set[i]);
françois romieudaf9df62009-10-07 12:44:20 +00002830 }
2831 } else {
Alexey Dobriyan350f7592009-11-25 15:54:21 -08002832 static const struct phy_reg phy_reg_init[] = {
françois romieudaf9df62009-10-07 12:44:20 +00002833 { 0x1f, 0x0002 },
2834 { 0x05, 0x6662 },
Francois Romieu5b538df2008-07-20 16:22:45 +02002835 { 0x1f, 0x0005 },
françois romieudaf9df62009-10-07 12:44:20 +00002836 { 0x05, 0x8330 },
2837 { 0x06, 0x6662 }
Francois Romieu5b538df2008-07-20 16:22:45 +02002838 };
2839
françois romieu4da19632011-01-03 15:07:55 +00002840 rtl_writephy_batch(tp, phy_reg_init, ARRAY_SIZE(phy_reg_init));
Francois Romieu5b538df2008-07-20 16:22:45 +02002841 }
2842
françois romieubca03d52011-01-03 15:07:31 +00002843 /* RSET couple improve */
françois romieu4da19632011-01-03 15:07:55 +00002844 rtl_writephy(tp, 0x1f, 0x0002);
2845 rtl_patchphy(tp, 0x0d, 0x0300);
2846 rtl_patchphy(tp, 0x0f, 0x0010);
françois romieudaf9df62009-10-07 12:44:20 +00002847
françois romieubca03d52011-01-03 15:07:31 +00002848 /* Fine tune PLL performance */
françois romieu4da19632011-01-03 15:07:55 +00002849 rtl_writephy(tp, 0x1f, 0x0002);
2850 rtl_w1w0_phy(tp, 0x02, 0x0100, 0x0600);
2851 rtl_w1w0_phy(tp, 0x03, 0x0000, 0xe000);
françois romieudaf9df62009-10-07 12:44:20 +00002852
françois romieu4da19632011-01-03 15:07:55 +00002853 rtl_writephy(tp, 0x1f, 0x0005);
2854 rtl_writephy(tp, 0x05, 0x001b);
François Romieu953a12c2011-04-24 17:38:48 +02002855
2856 rtl_apply_firmware_cond(tp, MII_EXPANSION, 0xbf00);
françois romieubca03d52011-01-03 15:07:31 +00002857
françois romieu4da19632011-01-03 15:07:55 +00002858 rtl_writephy(tp, 0x1f, 0x0000);
françois romieudaf9df62009-10-07 12:44:20 +00002859}
2860
françois romieubca03d52011-01-03 15:07:31 +00002861static void rtl8168d_2_hw_phy_config(struct rtl8169_private *tp)
françois romieudaf9df62009-10-07 12:44:20 +00002862{
Alexey Dobriyan350f7592009-11-25 15:54:21 -08002863 static const struct phy_reg phy_reg_init_0[] = {
françois romieubca03d52011-01-03 15:07:31 +00002864 /* Channel Estimation */
françois romieudaf9df62009-10-07 12:44:20 +00002865 { 0x1f, 0x0001 },
2866 { 0x06, 0x4064 },
2867 { 0x07, 0x2863 },
2868 { 0x08, 0x059c },
2869 { 0x09, 0x26b4 },
2870 { 0x0a, 0x6a19 },
2871 { 0x0b, 0xdcc8 },
2872 { 0x10, 0xf06d },
2873 { 0x14, 0x7f68 },
2874 { 0x18, 0x7fd9 },
2875 { 0x1c, 0xf0ff },
2876 { 0x1d, 0x3d9c },
2877 { 0x1f, 0x0003 },
2878 { 0x12, 0xf49f },
2879 { 0x13, 0x070b },
2880 { 0x1a, 0x05ad },
2881 { 0x14, 0x94c0 },
2882
françois romieubca03d52011-01-03 15:07:31 +00002883 /*
2884 * Tx Error Issue
Francois Romieucecb5fd2011-04-01 10:21:07 +02002885 * Enhance line driver power
françois romieubca03d52011-01-03 15:07:31 +00002886 */
françois romieudaf9df62009-10-07 12:44:20 +00002887 { 0x1f, 0x0002 },
2888 { 0x06, 0x5561 },
2889 { 0x1f, 0x0005 },
2890 { 0x05, 0x8332 },
françois romieubca03d52011-01-03 15:07:31 +00002891 { 0x06, 0x5561 },
2892
2893 /*
2894 * Can not link to 1Gbps with bad cable
2895 * Decrease SNR threshold form 21.07dB to 19.04dB
2896 */
2897 { 0x1f, 0x0001 },
2898 { 0x17, 0x0cc0 },
françois romieudaf9df62009-10-07 12:44:20 +00002899
2900 { 0x1f, 0x0000 },
françois romieubca03d52011-01-03 15:07:31 +00002901 { 0x0d, 0xf880 }
françois romieudaf9df62009-10-07 12:44:20 +00002902 };
2903
françois romieu4da19632011-01-03 15:07:55 +00002904 rtl_writephy_batch(tp, phy_reg_init_0, ARRAY_SIZE(phy_reg_init_0));
françois romieudaf9df62009-10-07 12:44:20 +00002905
Francois Romieufdf6fc02012-07-06 22:40:38 +02002906 if (rtl8168d_efuse_read(tp, 0x01) == 0xb1) {
Alexey Dobriyan350f7592009-11-25 15:54:21 -08002907 static const struct phy_reg phy_reg_init[] = {
françois romieudaf9df62009-10-07 12:44:20 +00002908 { 0x1f, 0x0002 },
2909 { 0x05, 0x669a },
2910 { 0x1f, 0x0005 },
2911 { 0x05, 0x8330 },
2912 { 0x06, 0x669a },
2913
2914 { 0x1f, 0x0002 }
2915 };
2916 int val;
2917
françois romieu4da19632011-01-03 15:07:55 +00002918 rtl_writephy_batch(tp, phy_reg_init, ARRAY_SIZE(phy_reg_init));
françois romieudaf9df62009-10-07 12:44:20 +00002919
françois romieu4da19632011-01-03 15:07:55 +00002920 val = rtl_readphy(tp, 0x0d);
françois romieudaf9df62009-10-07 12:44:20 +00002921 if ((val & 0x00ff) != 0x006c) {
Joe Perchesb6bc7652010-12-21 02:16:08 -08002922 static const u32 set[] = {
françois romieudaf9df62009-10-07 12:44:20 +00002923 0x0065, 0x0066, 0x0067, 0x0068,
2924 0x0069, 0x006a, 0x006b, 0x006c
2925 };
2926 int i;
2927
françois romieu4da19632011-01-03 15:07:55 +00002928 rtl_writephy(tp, 0x1f, 0x0002);
françois romieudaf9df62009-10-07 12:44:20 +00002929
2930 val &= 0xff00;
2931 for (i = 0; i < ARRAY_SIZE(set); i++)
françois romieu4da19632011-01-03 15:07:55 +00002932 rtl_writephy(tp, 0x0d, val | set[i]);
françois romieudaf9df62009-10-07 12:44:20 +00002933 }
2934 } else {
Alexey Dobriyan350f7592009-11-25 15:54:21 -08002935 static const struct phy_reg phy_reg_init[] = {
françois romieudaf9df62009-10-07 12:44:20 +00002936 { 0x1f, 0x0002 },
2937 { 0x05, 0x2642 },
2938 { 0x1f, 0x0005 },
2939 { 0x05, 0x8330 },
2940 { 0x06, 0x2642 }
2941 };
2942
françois romieu4da19632011-01-03 15:07:55 +00002943 rtl_writephy_batch(tp, phy_reg_init, ARRAY_SIZE(phy_reg_init));
françois romieudaf9df62009-10-07 12:44:20 +00002944 }
2945
françois romieubca03d52011-01-03 15:07:31 +00002946 /* Fine tune PLL performance */
françois romieu4da19632011-01-03 15:07:55 +00002947 rtl_writephy(tp, 0x1f, 0x0002);
2948 rtl_w1w0_phy(tp, 0x02, 0x0100, 0x0600);
2949 rtl_w1w0_phy(tp, 0x03, 0x0000, 0xe000);
françois romieudaf9df62009-10-07 12:44:20 +00002950
françois romieubca03d52011-01-03 15:07:31 +00002951 /* Switching regulator Slew rate */
françois romieu4da19632011-01-03 15:07:55 +00002952 rtl_writephy(tp, 0x1f, 0x0002);
2953 rtl_patchphy(tp, 0x0f, 0x0017);
françois romieudaf9df62009-10-07 12:44:20 +00002954
françois romieu4da19632011-01-03 15:07:55 +00002955 rtl_writephy(tp, 0x1f, 0x0005);
2956 rtl_writephy(tp, 0x05, 0x001b);
François Romieu953a12c2011-04-24 17:38:48 +02002957
2958 rtl_apply_firmware_cond(tp, MII_EXPANSION, 0xb300);
françois romieubca03d52011-01-03 15:07:31 +00002959
françois romieu4da19632011-01-03 15:07:55 +00002960 rtl_writephy(tp, 0x1f, 0x0000);
françois romieudaf9df62009-10-07 12:44:20 +00002961}
2962
françois romieu4da19632011-01-03 15:07:55 +00002963static void rtl8168d_3_hw_phy_config(struct rtl8169_private *tp)
françois romieudaf9df62009-10-07 12:44:20 +00002964{
Alexey Dobriyan350f7592009-11-25 15:54:21 -08002965 static const struct phy_reg phy_reg_init[] = {
françois romieudaf9df62009-10-07 12:44:20 +00002966 { 0x1f, 0x0002 },
2967 { 0x10, 0x0008 },
2968 { 0x0d, 0x006c },
2969
2970 { 0x1f, 0x0000 },
2971 { 0x0d, 0xf880 },
2972
2973 { 0x1f, 0x0001 },
2974 { 0x17, 0x0cc0 },
2975
2976 { 0x1f, 0x0001 },
2977 { 0x0b, 0xa4d8 },
2978 { 0x09, 0x281c },
2979 { 0x07, 0x2883 },
2980 { 0x0a, 0x6b35 },
2981 { 0x1d, 0x3da4 },
2982 { 0x1c, 0xeffd },
2983 { 0x14, 0x7f52 },
2984 { 0x18, 0x7fc6 },
2985 { 0x08, 0x0601 },
2986 { 0x06, 0x4063 },
2987 { 0x10, 0xf074 },
2988 { 0x1f, 0x0003 },
2989 { 0x13, 0x0789 },
2990 { 0x12, 0xf4bd },
2991 { 0x1a, 0x04fd },
2992 { 0x14, 0x84b0 },
2993 { 0x1f, 0x0000 },
2994 { 0x00, 0x9200 },
2995
2996 { 0x1f, 0x0005 },
2997 { 0x01, 0x0340 },
2998 { 0x1f, 0x0001 },
2999 { 0x04, 0x4000 },
3000 { 0x03, 0x1d21 },
3001 { 0x02, 0x0c32 },
3002 { 0x01, 0x0200 },
3003 { 0x00, 0x5554 },
3004 { 0x04, 0x4800 },
3005 { 0x04, 0x4000 },
3006 { 0x04, 0xf000 },
3007 { 0x03, 0xdf01 },
3008 { 0x02, 0xdf20 },
3009 { 0x01, 0x101a },
3010 { 0x00, 0xa0ff },
3011 { 0x04, 0xf800 },
3012 { 0x04, 0xf000 },
3013 { 0x1f, 0x0000 },
3014
3015 { 0x1f, 0x0007 },
3016 { 0x1e, 0x0023 },
3017 { 0x16, 0x0000 },
3018 { 0x1f, 0x0000 }
3019 };
3020
françois romieu4da19632011-01-03 15:07:55 +00003021 rtl_writephy_batch(tp, phy_reg_init, ARRAY_SIZE(phy_reg_init));
Francois Romieu5b538df2008-07-20 16:22:45 +02003022}
3023
françois romieue6de30d2011-01-03 15:08:37 +00003024static void rtl8168d_4_hw_phy_config(struct rtl8169_private *tp)
3025{
3026 static const struct phy_reg phy_reg_init[] = {
3027 { 0x1f, 0x0001 },
3028 { 0x17, 0x0cc0 },
3029
3030 { 0x1f, 0x0007 },
3031 { 0x1e, 0x002d },
3032 { 0x18, 0x0040 },
3033 { 0x1f, 0x0000 }
3034 };
3035
3036 rtl_writephy_batch(tp, phy_reg_init, ARRAY_SIZE(phy_reg_init));
3037 rtl_patchphy(tp, 0x0d, 1 << 5);
3038}
3039
Hayes Wang70090422011-07-06 15:58:06 +08003040static void rtl8168e_1_hw_phy_config(struct rtl8169_private *tp)
hayeswang01dc7fe2011-03-21 01:50:28 +00003041{
3042 static const struct phy_reg phy_reg_init[] = {
3043 /* Enable Delay cap */
3044 { 0x1f, 0x0005 },
3045 { 0x05, 0x8b80 },
3046 { 0x06, 0xc896 },
3047 { 0x1f, 0x0000 },
3048
3049 /* Channel estimation fine tune */
3050 { 0x1f, 0x0001 },
3051 { 0x0b, 0x6c20 },
3052 { 0x07, 0x2872 },
3053 { 0x1c, 0xefff },
3054 { 0x1f, 0x0003 },
3055 { 0x14, 0x6420 },
3056 { 0x1f, 0x0000 },
3057
3058 /* Update PFM & 10M TX idle timer */
3059 { 0x1f, 0x0007 },
3060 { 0x1e, 0x002f },
3061 { 0x15, 0x1919 },
3062 { 0x1f, 0x0000 },
3063
3064 { 0x1f, 0x0007 },
3065 { 0x1e, 0x00ac },
3066 { 0x18, 0x0006 },
3067 { 0x1f, 0x0000 }
3068 };
3069
Francois Romieu15ecd032011-04-27 13:52:22 -07003070 rtl_apply_firmware(tp);
3071
hayeswang01dc7fe2011-03-21 01:50:28 +00003072 rtl_writephy_batch(tp, phy_reg_init, ARRAY_SIZE(phy_reg_init));
3073
3074 /* DCO enable for 10M IDLE Power */
3075 rtl_writephy(tp, 0x1f, 0x0007);
3076 rtl_writephy(tp, 0x1e, 0x0023);
3077 rtl_w1w0_phy(tp, 0x17, 0x0006, 0x0000);
3078 rtl_writephy(tp, 0x1f, 0x0000);
3079
3080 /* For impedance matching */
3081 rtl_writephy(tp, 0x1f, 0x0002);
3082 rtl_w1w0_phy(tp, 0x08, 0x8000, 0x7f00);
Francois Romieucecb5fd2011-04-01 10:21:07 +02003083 rtl_writephy(tp, 0x1f, 0x0000);
hayeswang01dc7fe2011-03-21 01:50:28 +00003084
3085 /* PHY auto speed down */
3086 rtl_writephy(tp, 0x1f, 0x0007);
3087 rtl_writephy(tp, 0x1e, 0x002d);
3088 rtl_w1w0_phy(tp, 0x18, 0x0050, 0x0000);
3089 rtl_writephy(tp, 0x1f, 0x0000);
3090 rtl_w1w0_phy(tp, 0x14, 0x8000, 0x0000);
3091
3092 rtl_writephy(tp, 0x1f, 0x0005);
3093 rtl_writephy(tp, 0x05, 0x8b86);
3094 rtl_w1w0_phy(tp, 0x06, 0x0001, 0x0000);
3095 rtl_writephy(tp, 0x1f, 0x0000);
3096
3097 rtl_writephy(tp, 0x1f, 0x0005);
3098 rtl_writephy(tp, 0x05, 0x8b85);
3099 rtl_w1w0_phy(tp, 0x06, 0x0000, 0x2000);
3100 rtl_writephy(tp, 0x1f, 0x0007);
3101 rtl_writephy(tp, 0x1e, 0x0020);
3102 rtl_w1w0_phy(tp, 0x15, 0x0000, 0x1100);
3103 rtl_writephy(tp, 0x1f, 0x0006);
3104 rtl_writephy(tp, 0x00, 0x5a00);
3105 rtl_writephy(tp, 0x1f, 0x0000);
3106 rtl_writephy(tp, 0x0d, 0x0007);
3107 rtl_writephy(tp, 0x0e, 0x003c);
3108 rtl_writephy(tp, 0x0d, 0x4007);
3109 rtl_writephy(tp, 0x0e, 0x0000);
3110 rtl_writephy(tp, 0x0d, 0x0000);
3111}
3112
Hayes Wang70090422011-07-06 15:58:06 +08003113static void rtl8168e_2_hw_phy_config(struct rtl8169_private *tp)
3114{
3115 static const struct phy_reg phy_reg_init[] = {
3116 /* Enable Delay cap */
3117 { 0x1f, 0x0004 },
3118 { 0x1f, 0x0007 },
3119 { 0x1e, 0x00ac },
3120 { 0x18, 0x0006 },
3121 { 0x1f, 0x0002 },
3122 { 0x1f, 0x0000 },
3123 { 0x1f, 0x0000 },
3124
3125 /* Channel estimation fine tune */
3126 { 0x1f, 0x0003 },
3127 { 0x09, 0xa20f },
3128 { 0x1f, 0x0000 },
3129 { 0x1f, 0x0000 },
3130
3131 /* Green Setting */
3132 { 0x1f, 0x0005 },
3133 { 0x05, 0x8b5b },
3134 { 0x06, 0x9222 },
3135 { 0x05, 0x8b6d },
3136 { 0x06, 0x8000 },
3137 { 0x05, 0x8b76 },
3138 { 0x06, 0x8000 },
3139 { 0x1f, 0x0000 }
3140 };
3141
3142 rtl_apply_firmware(tp);
3143
3144 rtl_writephy_batch(tp, phy_reg_init, ARRAY_SIZE(phy_reg_init));
3145
3146 /* For 4-corner performance improve */
3147 rtl_writephy(tp, 0x1f, 0x0005);
3148 rtl_writephy(tp, 0x05, 0x8b80);
3149 rtl_w1w0_phy(tp, 0x17, 0x0006, 0x0000);
3150 rtl_writephy(tp, 0x1f, 0x0000);
3151
3152 /* PHY auto speed down */
3153 rtl_writephy(tp, 0x1f, 0x0004);
3154 rtl_writephy(tp, 0x1f, 0x0007);
3155 rtl_writephy(tp, 0x1e, 0x002d);
3156 rtl_w1w0_phy(tp, 0x18, 0x0010, 0x0000);
3157 rtl_writephy(tp, 0x1f, 0x0002);
3158 rtl_writephy(tp, 0x1f, 0x0000);
3159 rtl_w1w0_phy(tp, 0x14, 0x8000, 0x0000);
3160
3161 /* improve 10M EEE waveform */
3162 rtl_writephy(tp, 0x1f, 0x0005);
3163 rtl_writephy(tp, 0x05, 0x8b86);
3164 rtl_w1w0_phy(tp, 0x06, 0x0001, 0x0000);
3165 rtl_writephy(tp, 0x1f, 0x0000);
3166
3167 /* Improve 2-pair detection performance */
3168 rtl_writephy(tp, 0x1f, 0x0005);
3169 rtl_writephy(tp, 0x05, 0x8b85);
3170 rtl_w1w0_phy(tp, 0x06, 0x4000, 0x0000);
3171 rtl_writephy(tp, 0x1f, 0x0000);
3172
3173 /* EEE setting */
Francois Romieufdf6fc02012-07-06 22:40:38 +02003174 rtl_w1w0_eri(tp, 0x1b0, ERIAR_MASK_1111, 0x0000, 0x0003, ERIAR_EXGMAC);
Hayes Wang70090422011-07-06 15:58:06 +08003175 rtl_writephy(tp, 0x1f, 0x0005);
3176 rtl_writephy(tp, 0x05, 0x8b85);
3177 rtl_w1w0_phy(tp, 0x06, 0x0000, 0x2000);
3178 rtl_writephy(tp, 0x1f, 0x0004);
3179 rtl_writephy(tp, 0x1f, 0x0007);
3180 rtl_writephy(tp, 0x1e, 0x0020);
David S. Miller1805b2f2011-10-24 18:18:09 -04003181 rtl_w1w0_phy(tp, 0x15, 0x0000, 0x0100);
Hayes Wang70090422011-07-06 15:58:06 +08003182 rtl_writephy(tp, 0x1f, 0x0002);
3183 rtl_writephy(tp, 0x1f, 0x0000);
3184 rtl_writephy(tp, 0x0d, 0x0007);
3185 rtl_writephy(tp, 0x0e, 0x003c);
3186 rtl_writephy(tp, 0x0d, 0x4007);
3187 rtl_writephy(tp, 0x0e, 0x0000);
3188 rtl_writephy(tp, 0x0d, 0x0000);
3189
3190 /* Green feature */
3191 rtl_writephy(tp, 0x1f, 0x0003);
3192 rtl_w1w0_phy(tp, 0x19, 0x0000, 0x0001);
3193 rtl_w1w0_phy(tp, 0x10, 0x0000, 0x0400);
3194 rtl_writephy(tp, 0x1f, 0x0000);
3195}
3196
Hayes Wang5f886e02012-03-30 14:33:03 +08003197static void rtl8168f_hw_phy_config(struct rtl8169_private *tp)
3198{
3199 /* For 4-corner performance improve */
3200 rtl_writephy(tp, 0x1f, 0x0005);
3201 rtl_writephy(tp, 0x05, 0x8b80);
3202 rtl_w1w0_phy(tp, 0x06, 0x0006, 0x0000);
3203 rtl_writephy(tp, 0x1f, 0x0000);
3204
3205 /* PHY auto speed down */
3206 rtl_writephy(tp, 0x1f, 0x0007);
3207 rtl_writephy(tp, 0x1e, 0x002d);
3208 rtl_w1w0_phy(tp, 0x18, 0x0010, 0x0000);
3209 rtl_writephy(tp, 0x1f, 0x0000);
3210 rtl_w1w0_phy(tp, 0x14, 0x8000, 0x0000);
3211
3212 /* Improve 10M EEE waveform */
3213 rtl_writephy(tp, 0x1f, 0x0005);
3214 rtl_writephy(tp, 0x05, 0x8b86);
3215 rtl_w1w0_phy(tp, 0x06, 0x0001, 0x0000);
3216 rtl_writephy(tp, 0x1f, 0x0000);
3217}
3218
Hayes Wangc2218922011-09-06 16:55:18 +08003219static void rtl8168f_1_hw_phy_config(struct rtl8169_private *tp)
3220{
3221 static const struct phy_reg phy_reg_init[] = {
3222 /* Channel estimation fine tune */
3223 { 0x1f, 0x0003 },
3224 { 0x09, 0xa20f },
3225 { 0x1f, 0x0000 },
3226
3227 /* Modify green table for giga & fnet */
3228 { 0x1f, 0x0005 },
3229 { 0x05, 0x8b55 },
3230 { 0x06, 0x0000 },
3231 { 0x05, 0x8b5e },
3232 { 0x06, 0x0000 },
3233 { 0x05, 0x8b67 },
3234 { 0x06, 0x0000 },
3235 { 0x05, 0x8b70 },
3236 { 0x06, 0x0000 },
3237 { 0x1f, 0x0000 },
3238 { 0x1f, 0x0007 },
3239 { 0x1e, 0x0078 },
3240 { 0x17, 0x0000 },
3241 { 0x19, 0x00fb },
3242 { 0x1f, 0x0000 },
3243
3244 /* Modify green table for 10M */
3245 { 0x1f, 0x0005 },
3246 { 0x05, 0x8b79 },
3247 { 0x06, 0xaa00 },
3248 { 0x1f, 0x0000 },
3249
3250 /* Disable hiimpedance detection (RTCT) */
3251 { 0x1f, 0x0003 },
3252 { 0x01, 0x328a },
3253 { 0x1f, 0x0000 }
3254 };
3255
3256 rtl_apply_firmware(tp);
3257
3258 rtl_writephy_batch(tp, phy_reg_init, ARRAY_SIZE(phy_reg_init));
3259
Hayes Wang5f886e02012-03-30 14:33:03 +08003260 rtl8168f_hw_phy_config(tp);
Hayes Wangc2218922011-09-06 16:55:18 +08003261
3262 /* Improve 2-pair detection performance */
3263 rtl_writephy(tp, 0x1f, 0x0005);
3264 rtl_writephy(tp, 0x05, 0x8b85);
3265 rtl_w1w0_phy(tp, 0x06, 0x4000, 0x0000);
3266 rtl_writephy(tp, 0x1f, 0x0000);
3267}
3268
3269static void rtl8168f_2_hw_phy_config(struct rtl8169_private *tp)
3270{
3271 rtl_apply_firmware(tp);
3272
Hayes Wang5f886e02012-03-30 14:33:03 +08003273 rtl8168f_hw_phy_config(tp);
Hayes Wangc2218922011-09-06 16:55:18 +08003274}
3275
Hayes Wangb3d7b2f2012-03-30 14:48:06 +08003276static void rtl8411_hw_phy_config(struct rtl8169_private *tp)
3277{
Hayes Wangb3d7b2f2012-03-30 14:48:06 +08003278 static const struct phy_reg phy_reg_init[] = {
3279 /* Channel estimation fine tune */
3280 { 0x1f, 0x0003 },
3281 { 0x09, 0xa20f },
3282 { 0x1f, 0x0000 },
3283
3284 /* Modify green table for giga & fnet */
3285 { 0x1f, 0x0005 },
3286 { 0x05, 0x8b55 },
3287 { 0x06, 0x0000 },
3288 { 0x05, 0x8b5e },
3289 { 0x06, 0x0000 },
3290 { 0x05, 0x8b67 },
3291 { 0x06, 0x0000 },
3292 { 0x05, 0x8b70 },
3293 { 0x06, 0x0000 },
3294 { 0x1f, 0x0000 },
3295 { 0x1f, 0x0007 },
3296 { 0x1e, 0x0078 },
3297 { 0x17, 0x0000 },
3298 { 0x19, 0x00aa },
3299 { 0x1f, 0x0000 },
3300
3301 /* Modify green table for 10M */
3302 { 0x1f, 0x0005 },
3303 { 0x05, 0x8b79 },
3304 { 0x06, 0xaa00 },
3305 { 0x1f, 0x0000 },
3306
3307 /* Disable hiimpedance detection (RTCT) */
3308 { 0x1f, 0x0003 },
3309 { 0x01, 0x328a },
3310 { 0x1f, 0x0000 }
3311 };
3312
3313
3314 rtl_apply_firmware(tp);
3315
3316 rtl8168f_hw_phy_config(tp);
3317
3318 /* Improve 2-pair detection performance */
3319 rtl_writephy(tp, 0x1f, 0x0005);
3320 rtl_writephy(tp, 0x05, 0x8b85);
3321 rtl_w1w0_phy(tp, 0x06, 0x4000, 0x0000);
3322 rtl_writephy(tp, 0x1f, 0x0000);
3323
3324 rtl_writephy_batch(tp, phy_reg_init, ARRAY_SIZE(phy_reg_init));
3325
3326 /* Modify green table for giga */
3327 rtl_writephy(tp, 0x1f, 0x0005);
3328 rtl_writephy(tp, 0x05, 0x8b54);
3329 rtl_w1w0_phy(tp, 0x06, 0x0000, 0x0800);
3330 rtl_writephy(tp, 0x05, 0x8b5d);
3331 rtl_w1w0_phy(tp, 0x06, 0x0000, 0x0800);
3332 rtl_writephy(tp, 0x05, 0x8a7c);
3333 rtl_w1w0_phy(tp, 0x06, 0x0000, 0x0100);
3334 rtl_writephy(tp, 0x05, 0x8a7f);
3335 rtl_w1w0_phy(tp, 0x06, 0x0100, 0x0000);
3336 rtl_writephy(tp, 0x05, 0x8a82);
3337 rtl_w1w0_phy(tp, 0x06, 0x0000, 0x0100);
3338 rtl_writephy(tp, 0x05, 0x8a85);
3339 rtl_w1w0_phy(tp, 0x06, 0x0000, 0x0100);
3340 rtl_writephy(tp, 0x05, 0x8a88);
3341 rtl_w1w0_phy(tp, 0x06, 0x0000, 0x0100);
3342 rtl_writephy(tp, 0x1f, 0x0000);
3343
3344 /* uc same-seed solution */
3345 rtl_writephy(tp, 0x1f, 0x0005);
3346 rtl_writephy(tp, 0x05, 0x8b85);
3347 rtl_w1w0_phy(tp, 0x06, 0x8000, 0x0000);
3348 rtl_writephy(tp, 0x1f, 0x0000);
3349
3350 /* eee setting */
Francois Romieufdf6fc02012-07-06 22:40:38 +02003351 rtl_w1w0_eri(tp, 0x1b0, ERIAR_MASK_0001, 0x00, 0x03, ERIAR_EXGMAC);
Hayes Wangb3d7b2f2012-03-30 14:48:06 +08003352 rtl_writephy(tp, 0x1f, 0x0005);
3353 rtl_writephy(tp, 0x05, 0x8b85);
3354 rtl_w1w0_phy(tp, 0x06, 0x0000, 0x2000);
3355 rtl_writephy(tp, 0x1f, 0x0004);
3356 rtl_writephy(tp, 0x1f, 0x0007);
3357 rtl_writephy(tp, 0x1e, 0x0020);
3358 rtl_w1w0_phy(tp, 0x15, 0x0000, 0x0100);
3359 rtl_writephy(tp, 0x1f, 0x0000);
3360 rtl_writephy(tp, 0x0d, 0x0007);
3361 rtl_writephy(tp, 0x0e, 0x003c);
3362 rtl_writephy(tp, 0x0d, 0x4007);
3363 rtl_writephy(tp, 0x0e, 0x0000);
3364 rtl_writephy(tp, 0x0d, 0x0000);
3365
3366 /* Green feature */
3367 rtl_writephy(tp, 0x1f, 0x0003);
3368 rtl_w1w0_phy(tp, 0x19, 0x0000, 0x0001);
3369 rtl_w1w0_phy(tp, 0x10, 0x0000, 0x0400);
3370 rtl_writephy(tp, 0x1f, 0x0000);
3371}
3372
Hayes Wangc5583862012-07-02 17:23:22 +08003373static void rtl8168g_1_hw_phy_config(struct rtl8169_private *tp)
3374{
3375 static const u16 mac_ocp_patch[] = {
3376 0xe008, 0xe01b, 0xe01d, 0xe01f,
3377 0xe021, 0xe023, 0xe025, 0xe027,
3378 0x49d2, 0xf10d, 0x766c, 0x49e2,
3379 0xf00a, 0x1ec0, 0x8ee1, 0xc60a,
3380
3381 0x77c0, 0x4870, 0x9fc0, 0x1ea0,
3382 0xc707, 0x8ee1, 0x9d6c, 0xc603,
3383 0xbe00, 0xb416, 0x0076, 0xe86c,
3384 0xc602, 0xbe00, 0x0000, 0xc602,
3385
3386 0xbe00, 0x0000, 0xc602, 0xbe00,
3387 0x0000, 0xc602, 0xbe00, 0x0000,
3388 0xc602, 0xbe00, 0x0000, 0xc602,
3389 0xbe00, 0x0000, 0xc602, 0xbe00,
3390
3391 0x0000, 0x0000, 0x0000, 0x0000
3392 };
3393 u32 i;
3394
3395 /* Patch code for GPHY reset */
3396 for (i = 0; i < ARRAY_SIZE(mac_ocp_patch); i++)
3397 r8168_mac_ocp_write(tp, 0xf800 + 2*i, mac_ocp_patch[i]);
3398 r8168_mac_ocp_write(tp, 0xfc26, 0x8000);
3399 r8168_mac_ocp_write(tp, 0xfc28, 0x0075);
3400
3401 rtl_apply_firmware(tp);
3402
3403 if (r8168_phy_ocp_read(tp, 0xa460) & 0x0100)
3404 rtl_w1w0_phy_ocp(tp, 0xbcc4, 0x0000, 0x8000);
3405 else
3406 rtl_w1w0_phy_ocp(tp, 0xbcc4, 0x8000, 0x0000);
3407
3408 if (r8168_phy_ocp_read(tp, 0xa466) & 0x0100)
3409 rtl_w1w0_phy_ocp(tp, 0xc41a, 0x0002, 0x0000);
3410 else
3411 rtl_w1w0_phy_ocp(tp, 0xbcc4, 0x0000, 0x0002);
3412
3413 rtl_w1w0_phy_ocp(tp, 0xa442, 0x000c, 0x0000);
3414 rtl_w1w0_phy_ocp(tp, 0xa4b2, 0x0004, 0x0000);
3415
3416 r8168_phy_ocp_write(tp, 0xa436, 0x8012);
3417 rtl_w1w0_phy_ocp(tp, 0xa438, 0x8000, 0x0000);
3418
3419 rtl_w1w0_phy_ocp(tp, 0xc422, 0x4000, 0x2000);
3420}
3421
françois romieu4da19632011-01-03 15:07:55 +00003422static void rtl8102e_hw_phy_config(struct rtl8169_private *tp)
Francois Romieu2857ffb2008-08-02 21:08:49 +02003423{
Alexey Dobriyan350f7592009-11-25 15:54:21 -08003424 static const struct phy_reg phy_reg_init[] = {
Francois Romieu2857ffb2008-08-02 21:08:49 +02003425 { 0x1f, 0x0003 },
3426 { 0x08, 0x441d },
3427 { 0x01, 0x9100 },
3428 { 0x1f, 0x0000 }
3429 };
3430
françois romieu4da19632011-01-03 15:07:55 +00003431 rtl_writephy(tp, 0x1f, 0x0000);
3432 rtl_patchphy(tp, 0x11, 1 << 12);
3433 rtl_patchphy(tp, 0x19, 1 << 13);
3434 rtl_patchphy(tp, 0x10, 1 << 15);
Francois Romieu2857ffb2008-08-02 21:08:49 +02003435
françois romieu4da19632011-01-03 15:07:55 +00003436 rtl_writephy_batch(tp, phy_reg_init, ARRAY_SIZE(phy_reg_init));
Francois Romieu2857ffb2008-08-02 21:08:49 +02003437}
3438
Hayes Wang5a5e4442011-02-22 17:26:21 +08003439static void rtl8105e_hw_phy_config(struct rtl8169_private *tp)
3440{
3441 static const struct phy_reg phy_reg_init[] = {
3442 { 0x1f, 0x0005 },
3443 { 0x1a, 0x0000 },
3444 { 0x1f, 0x0000 },
3445
3446 { 0x1f, 0x0004 },
3447 { 0x1c, 0x0000 },
3448 { 0x1f, 0x0000 },
3449
3450 { 0x1f, 0x0001 },
3451 { 0x15, 0x7701 },
3452 { 0x1f, 0x0000 }
3453 };
3454
3455 /* Disable ALDPS before ram code */
3456 rtl_writephy(tp, 0x1f, 0x0000);
3457 rtl_writephy(tp, 0x18, 0x0310);
3458 msleep(100);
3459
François Romieu953a12c2011-04-24 17:38:48 +02003460 rtl_apply_firmware(tp);
Hayes Wang5a5e4442011-02-22 17:26:21 +08003461
3462 rtl_writephy_batch(tp, phy_reg_init, ARRAY_SIZE(phy_reg_init));
3463}
3464
Hayes Wang7e18dca2012-03-30 14:33:02 +08003465static void rtl8402_hw_phy_config(struct rtl8169_private *tp)
3466{
Hayes Wang7e18dca2012-03-30 14:33:02 +08003467 /* Disable ALDPS before setting firmware */
3468 rtl_writephy(tp, 0x1f, 0x0000);
3469 rtl_writephy(tp, 0x18, 0x0310);
3470 msleep(20);
3471
3472 rtl_apply_firmware(tp);
3473
3474 /* EEE setting */
Francois Romieufdf6fc02012-07-06 22:40:38 +02003475 rtl_eri_write(tp, 0x1b0, ERIAR_MASK_0011, 0x0000, ERIAR_EXGMAC);
Hayes Wang7e18dca2012-03-30 14:33:02 +08003476 rtl_writephy(tp, 0x1f, 0x0004);
3477 rtl_writephy(tp, 0x10, 0x401f);
3478 rtl_writephy(tp, 0x19, 0x7030);
3479 rtl_writephy(tp, 0x1f, 0x0000);
3480}
3481
Hayes Wang5598bfe2012-07-02 17:23:21 +08003482static void rtl8106e_hw_phy_config(struct rtl8169_private *tp)
3483{
Hayes Wang5598bfe2012-07-02 17:23:21 +08003484 static const struct phy_reg phy_reg_init[] = {
3485 { 0x1f, 0x0004 },
3486 { 0x10, 0xc07f },
3487 { 0x19, 0x7030 },
3488 { 0x1f, 0x0000 }
3489 };
3490
3491 /* Disable ALDPS before ram code */
3492 rtl_writephy(tp, 0x1f, 0x0000);
3493 rtl_writephy(tp, 0x18, 0x0310);
3494 msleep(100);
3495
3496 rtl_apply_firmware(tp);
3497
Francois Romieufdf6fc02012-07-06 22:40:38 +02003498 rtl_eri_write(tp, 0x1b0, ERIAR_MASK_0011, 0x0000, ERIAR_EXGMAC);
Hayes Wang5598bfe2012-07-02 17:23:21 +08003499 rtl_writephy_batch(tp, phy_reg_init, ARRAY_SIZE(phy_reg_init));
3500
Francois Romieufdf6fc02012-07-06 22:40:38 +02003501 rtl_eri_write(tp, 0x1d0, ERIAR_MASK_0011, 0x0000, ERIAR_EXGMAC);
Hayes Wang5598bfe2012-07-02 17:23:21 +08003502}
3503
Francois Romieu5615d9f2007-08-17 17:50:46 +02003504static void rtl_hw_phy_config(struct net_device *dev)
3505{
3506 struct rtl8169_private *tp = netdev_priv(dev);
Francois Romieu5615d9f2007-08-17 17:50:46 +02003507
3508 rtl8169_print_mac_version(tp);
3509
3510 switch (tp->mac_version) {
3511 case RTL_GIGA_MAC_VER_01:
3512 break;
3513 case RTL_GIGA_MAC_VER_02:
3514 case RTL_GIGA_MAC_VER_03:
françois romieu4da19632011-01-03 15:07:55 +00003515 rtl8169s_hw_phy_config(tp);
Francois Romieu5615d9f2007-08-17 17:50:46 +02003516 break;
3517 case RTL_GIGA_MAC_VER_04:
françois romieu4da19632011-01-03 15:07:55 +00003518 rtl8169sb_hw_phy_config(tp);
Francois Romieu5615d9f2007-08-17 17:50:46 +02003519 break;
françois romieu2e9558562009-08-10 19:44:19 +00003520 case RTL_GIGA_MAC_VER_05:
françois romieu4da19632011-01-03 15:07:55 +00003521 rtl8169scd_hw_phy_config(tp);
françois romieu2e9558562009-08-10 19:44:19 +00003522 break;
françois romieu8c7006a2009-08-10 19:43:29 +00003523 case RTL_GIGA_MAC_VER_06:
françois romieu4da19632011-01-03 15:07:55 +00003524 rtl8169sce_hw_phy_config(tp);
françois romieu8c7006a2009-08-10 19:43:29 +00003525 break;
Francois Romieu2857ffb2008-08-02 21:08:49 +02003526 case RTL_GIGA_MAC_VER_07:
3527 case RTL_GIGA_MAC_VER_08:
3528 case RTL_GIGA_MAC_VER_09:
françois romieu4da19632011-01-03 15:07:55 +00003529 rtl8102e_hw_phy_config(tp);
Francois Romieu2857ffb2008-08-02 21:08:49 +02003530 break;
Francois Romieu236b8082008-05-30 16:11:48 +02003531 case RTL_GIGA_MAC_VER_11:
françois romieu4da19632011-01-03 15:07:55 +00003532 rtl8168bb_hw_phy_config(tp);
Francois Romieu236b8082008-05-30 16:11:48 +02003533 break;
3534 case RTL_GIGA_MAC_VER_12:
françois romieu4da19632011-01-03 15:07:55 +00003535 rtl8168bef_hw_phy_config(tp);
Francois Romieu236b8082008-05-30 16:11:48 +02003536 break;
3537 case RTL_GIGA_MAC_VER_17:
françois romieu4da19632011-01-03 15:07:55 +00003538 rtl8168bef_hw_phy_config(tp);
Francois Romieu236b8082008-05-30 16:11:48 +02003539 break;
Francois Romieu867763c2007-08-17 18:21:58 +02003540 case RTL_GIGA_MAC_VER_18:
françois romieu4da19632011-01-03 15:07:55 +00003541 rtl8168cp_1_hw_phy_config(tp);
Francois Romieu867763c2007-08-17 18:21:58 +02003542 break;
3543 case RTL_GIGA_MAC_VER_19:
françois romieu4da19632011-01-03 15:07:55 +00003544 rtl8168c_1_hw_phy_config(tp);
Francois Romieu867763c2007-08-17 18:21:58 +02003545 break;
Francois Romieu7da97ec2007-10-18 15:20:43 +02003546 case RTL_GIGA_MAC_VER_20:
françois romieu4da19632011-01-03 15:07:55 +00003547 rtl8168c_2_hw_phy_config(tp);
Francois Romieu7da97ec2007-10-18 15:20:43 +02003548 break;
Francois Romieu197ff762008-06-28 13:16:02 +02003549 case RTL_GIGA_MAC_VER_21:
françois romieu4da19632011-01-03 15:07:55 +00003550 rtl8168c_3_hw_phy_config(tp);
Francois Romieu197ff762008-06-28 13:16:02 +02003551 break;
Francois Romieu6fb07052008-06-29 11:54:28 +02003552 case RTL_GIGA_MAC_VER_22:
françois romieu4da19632011-01-03 15:07:55 +00003553 rtl8168c_4_hw_phy_config(tp);
Francois Romieu6fb07052008-06-29 11:54:28 +02003554 break;
Francois Romieuef3386f2008-06-29 12:24:30 +02003555 case RTL_GIGA_MAC_VER_23:
Francois Romieu7f3e3d32008-07-20 18:53:20 +02003556 case RTL_GIGA_MAC_VER_24:
françois romieu4da19632011-01-03 15:07:55 +00003557 rtl8168cp_2_hw_phy_config(tp);
Francois Romieuef3386f2008-06-29 12:24:30 +02003558 break;
Francois Romieu5b538df2008-07-20 16:22:45 +02003559 case RTL_GIGA_MAC_VER_25:
françois romieubca03d52011-01-03 15:07:31 +00003560 rtl8168d_1_hw_phy_config(tp);
françois romieudaf9df62009-10-07 12:44:20 +00003561 break;
3562 case RTL_GIGA_MAC_VER_26:
françois romieubca03d52011-01-03 15:07:31 +00003563 rtl8168d_2_hw_phy_config(tp);
françois romieudaf9df62009-10-07 12:44:20 +00003564 break;
3565 case RTL_GIGA_MAC_VER_27:
françois romieu4da19632011-01-03 15:07:55 +00003566 rtl8168d_3_hw_phy_config(tp);
Francois Romieu5b538df2008-07-20 16:22:45 +02003567 break;
françois romieue6de30d2011-01-03 15:08:37 +00003568 case RTL_GIGA_MAC_VER_28:
3569 rtl8168d_4_hw_phy_config(tp);
3570 break;
Hayes Wang5a5e4442011-02-22 17:26:21 +08003571 case RTL_GIGA_MAC_VER_29:
3572 case RTL_GIGA_MAC_VER_30:
3573 rtl8105e_hw_phy_config(tp);
3574 break;
Francois Romieucecb5fd2011-04-01 10:21:07 +02003575 case RTL_GIGA_MAC_VER_31:
3576 /* None. */
3577 break;
hayeswang01dc7fe2011-03-21 01:50:28 +00003578 case RTL_GIGA_MAC_VER_32:
hayeswang01dc7fe2011-03-21 01:50:28 +00003579 case RTL_GIGA_MAC_VER_33:
Hayes Wang70090422011-07-06 15:58:06 +08003580 rtl8168e_1_hw_phy_config(tp);
3581 break;
3582 case RTL_GIGA_MAC_VER_34:
3583 rtl8168e_2_hw_phy_config(tp);
hayeswang01dc7fe2011-03-21 01:50:28 +00003584 break;
Hayes Wangc2218922011-09-06 16:55:18 +08003585 case RTL_GIGA_MAC_VER_35:
3586 rtl8168f_1_hw_phy_config(tp);
3587 break;
3588 case RTL_GIGA_MAC_VER_36:
3589 rtl8168f_2_hw_phy_config(tp);
3590 break;
Francois Romieuef3386f2008-06-29 12:24:30 +02003591
Hayes Wang7e18dca2012-03-30 14:33:02 +08003592 case RTL_GIGA_MAC_VER_37:
3593 rtl8402_hw_phy_config(tp);
3594 break;
3595
Hayes Wangb3d7b2f2012-03-30 14:48:06 +08003596 case RTL_GIGA_MAC_VER_38:
3597 rtl8411_hw_phy_config(tp);
3598 break;
3599
Hayes Wang5598bfe2012-07-02 17:23:21 +08003600 case RTL_GIGA_MAC_VER_39:
3601 rtl8106e_hw_phy_config(tp);
3602 break;
3603
Hayes Wangc5583862012-07-02 17:23:22 +08003604 case RTL_GIGA_MAC_VER_40:
3605 rtl8168g_1_hw_phy_config(tp);
3606 break;
3607
3608 case RTL_GIGA_MAC_VER_41:
Francois Romieu5615d9f2007-08-17 17:50:46 +02003609 default:
3610 break;
3611 }
3612}
3613
Francois Romieuda78dbf2012-01-26 14:18:23 +01003614static void rtl_phy_work(struct rtl8169_private *tp)
Linus Torvalds1da177e2005-04-16 15:20:36 -07003615{
Linus Torvalds1da177e2005-04-16 15:20:36 -07003616 struct timer_list *timer = &tp->timer;
3617 void __iomem *ioaddr = tp->mmio_addr;
3618 unsigned long timeout = RTL8169_PHY_TIMEOUT;
3619
Francois Romieubcf0bf92006-07-26 23:14:13 +02003620 assert(tp->mac_version > RTL_GIGA_MAC_VER_01);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003621
françois romieu4da19632011-01-03 15:07:55 +00003622 if (tp->phy_reset_pending(tp)) {
Francois Romieu5b0384f2006-08-16 16:00:01 +02003623 /*
Linus Torvalds1da177e2005-04-16 15:20:36 -07003624 * A busy loop could burn quite a few cycles on nowadays CPU.
3625 * Let's delay the execution of the timer for a few ticks.
3626 */
3627 timeout = HZ/10;
3628 goto out_mod_timer;
3629 }
3630
3631 if (tp->link_ok(ioaddr))
Francois Romieuda78dbf2012-01-26 14:18:23 +01003632 return;
Linus Torvalds1da177e2005-04-16 15:20:36 -07003633
Francois Romieuda78dbf2012-01-26 14:18:23 +01003634 netif_warn(tp, link, tp->dev, "PHY reset until link up\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07003635
françois romieu4da19632011-01-03 15:07:55 +00003636 tp->phy_reset_enable(tp);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003637
3638out_mod_timer:
3639 mod_timer(timer, jiffies + timeout);
Francois Romieuda78dbf2012-01-26 14:18:23 +01003640}
3641
3642static void rtl_schedule_task(struct rtl8169_private *tp, enum rtl_flag flag)
3643{
Francois Romieuda78dbf2012-01-26 14:18:23 +01003644 if (!test_and_set_bit(flag, tp->wk.flags))
3645 schedule_work(&tp->wk.work);
Francois Romieuda78dbf2012-01-26 14:18:23 +01003646}
3647
3648static void rtl8169_phy_timer(unsigned long __opaque)
3649{
3650 struct net_device *dev = (struct net_device *)__opaque;
3651 struct rtl8169_private *tp = netdev_priv(dev);
3652
Francois Romieu98ddf982012-01-31 10:47:34 +01003653 rtl_schedule_task(tp, RTL_FLAG_TASK_PHY_PENDING);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003654}
3655
Linus Torvalds1da177e2005-04-16 15:20:36 -07003656static void rtl8169_release_board(struct pci_dev *pdev, struct net_device *dev,
3657 void __iomem *ioaddr)
3658{
3659 iounmap(ioaddr);
3660 pci_release_regions(pdev);
françois romieu87aeec72010-04-26 11:42:06 +00003661 pci_clear_mwi(pdev);
Linus Torvalds1da177e2005-04-16 15:20:36 -07003662 pci_disable_device(pdev);
3663 free_netdev(dev);
3664}
3665
Francois Romieuffc46952012-07-06 14:19:23 +02003666DECLARE_RTL_COND(rtl_phy_reset_cond)
3667{
3668 return tp->phy_reset_pending(tp);
3669}
3670
Francois Romieubf793292006-11-01 00:53:05 +01003671static void rtl8169_phy_reset(struct net_device *dev,
3672 struct rtl8169_private *tp)
3673{
françois romieu4da19632011-01-03 15:07:55 +00003674 tp->phy_reset_enable(tp);
Francois Romieuffc46952012-07-06 14:19:23 +02003675 rtl_msleep_loop_wait_low(tp, &rtl_phy_reset_cond, 1, 100);
Francois Romieubf793292006-11-01 00:53:05 +01003676}
3677
David S. Miller8decf862011-09-22 03:23:13 -04003678static bool rtl_tbi_enabled(struct rtl8169_private *tp)
3679{
3680 void __iomem *ioaddr = tp->mmio_addr;
3681
3682 return (tp->mac_version == RTL_GIGA_MAC_VER_01) &&
3683 (RTL_R8(PHYstatus) & TBI_Enable);
3684}
3685
Francois Romieu4ff96fa2006-07-26 22:05:06 +02003686static void rtl8169_init_phy(struct net_device *dev, struct rtl8169_private *tp)
Linus Torvalds1da177e2005-04-16 15:20:36 -07003687{
Francois Romieu4ff96fa2006-07-26 22:05:06 +02003688 void __iomem *ioaddr = tp->mmio_addr;
Francois Romieu4ff96fa2006-07-26 22:05:06 +02003689
Francois Romieu5615d9f2007-08-17 17:50:46 +02003690 rtl_hw_phy_config(dev);
Francois Romieu4ff96fa2006-07-26 22:05:06 +02003691
Marcus Sundberg773328942008-07-10 21:28:08 +02003692 if (tp->mac_version <= RTL_GIGA_MAC_VER_06) {
3693 dprintk("Set MAC Reg C+CR Offset 0x82h = 0x01h\n");
3694 RTL_W8(0x82, 0x01);
3695 }
Francois Romieu4ff96fa2006-07-26 22:05:06 +02003696
Francois Romieu6dccd162007-02-13 23:38:05 +01003697 pci_write_config_byte(tp->pci_dev, PCI_LATENCY_TIMER, 0x40);
3698
3699 if (tp->mac_version <= RTL_GIGA_MAC_VER_06)
3700 pci_write_config_byte(tp->pci_dev, PCI_CACHE_LINE_SIZE, 0x08);
Francois Romieu4ff96fa2006-07-26 22:05:06 +02003701
Francois Romieubcf0bf92006-07-26 23:14:13 +02003702 if (tp->mac_version == RTL_GIGA_MAC_VER_02) {
Francois Romieu4ff96fa2006-07-26 22:05:06 +02003703 dprintk("Set MAC Reg C+CR Offset 0x82h = 0x01h\n");
3704 RTL_W8(0x82, 0x01);
3705 dprintk("Set PHY Reg 0x0bh = 0x00h\n");
françois romieu4da19632011-01-03 15:07:55 +00003706 rtl_writephy(tp, 0x0b, 0x0000); //w 0x0b 15 0 0
Francois Romieu4ff96fa2006-07-26 22:05:06 +02003707 }
3708
Francois Romieubf793292006-11-01 00:53:05 +01003709 rtl8169_phy_reset(dev, tp);
3710
Oliver Neukum54405cd2011-01-06 21:55:13 +01003711 rtl8169_set_speed(dev, AUTONEG_ENABLE, SPEED_1000, DUPLEX_FULL,
Francois Romieucecb5fd2011-04-01 10:21:07 +02003712 ADVERTISED_10baseT_Half | ADVERTISED_10baseT_Full |
3713 ADVERTISED_100baseT_Half | ADVERTISED_100baseT_Full |
3714 (tp->mii.supports_gmii ?
3715 ADVERTISED_1000baseT_Half |
3716 ADVERTISED_1000baseT_Full : 0));
Francois Romieu4ff96fa2006-07-26 22:05:06 +02003717
David S. Miller8decf862011-09-22 03:23:13 -04003718 if (rtl_tbi_enabled(tp))
Joe Perchesbf82c182010-02-09 11:49:50 +00003719 netif_info(tp, link, dev, "TBI auto-negotiating\n");
Francois Romieu4ff96fa2006-07-26 22:05:06 +02003720}
3721
Francois Romieu773d2022007-01-31 23:47:43 +01003722static void rtl_rar_set(struct rtl8169_private *tp, u8 *addr)
3723{
3724 void __iomem *ioaddr = tp->mmio_addr;
3725 u32 high;
3726 u32 low;
3727
3728 low = addr[0] | (addr[1] << 8) | (addr[2] << 16) | (addr[3] << 24);
3729 high = addr[4] | (addr[5] << 8);
3730
Francois Romieuda78dbf2012-01-26 14:18:23 +01003731 rtl_lock_work(tp);
Francois Romieu773d2022007-01-31 23:47:43 +01003732
3733 RTL_W8(Cfg9346, Cfg9346_Unlock);
françois romieu908ba2b2010-04-26 11:42:58 +00003734
Francois Romieu773d2022007-01-31 23:47:43 +01003735 RTL_W32(MAC4, high);
françois romieu908ba2b2010-04-26 11:42:58 +00003736 RTL_R32(MAC4);
3737
Francois Romieu78f1cd02010-03-27 19:35:46 -07003738 RTL_W32(MAC0, low);
françois romieu908ba2b2010-04-26 11:42:58 +00003739 RTL_R32(MAC0);
3740
françois romieuc28aa382011-08-02 03:53:43 +00003741 if (tp->mac_version == RTL_GIGA_MAC_VER_34) {
3742 const struct exgmac_reg e[] = {
3743 { .addr = 0xe0, ERIAR_MASK_1111, .val = low },
3744 { .addr = 0xe4, ERIAR_MASK_1111, .val = high },
3745 { .addr = 0xf0, ERIAR_MASK_1111, .val = low << 16 },
3746 { .addr = 0xf4, ERIAR_MASK_1111, .val = high << 16 |
3747 low >> 16 },
3748 };
3749
Francois Romieufdf6fc02012-07-06 22:40:38 +02003750 rtl_write_exgmac_batch(tp, e, ARRAY_SIZE(e));
françois romieuc28aa382011-08-02 03:53:43 +00003751 }
3752
Francois Romieu773d2022007-01-31 23:47:43 +01003753 RTL_W8(Cfg9346, Cfg9346_Lock);
3754
Francois Romieuda78dbf2012-01-26 14:18:23 +01003755 rtl_unlock_work(tp);
Francois Romieu773d2022007-01-31 23:47:43 +01003756}
3757
3758static int rtl_set_mac_address(struct net_device *dev, void *p)
3759{
3760 struct rtl8169_private *tp = netdev_priv(dev);
3761 struct sockaddr *addr = p;
3762
3763 if (!is_valid_ether_addr(addr->sa_data))
3764 return -EADDRNOTAVAIL;
3765
3766 memcpy(dev->dev_addr, addr->sa_data, dev->addr_len);
3767
3768 rtl_rar_set(tp, dev->dev_addr);
3769
3770 return 0;
3771}
3772
Francois Romieu5f787a12006-08-17 13:02:36 +02003773static int rtl8169_ioctl(struct net_device *dev, struct ifreq *ifr, int cmd)
3774{
3775 struct rtl8169_private *tp = netdev_priv(dev);
3776 struct mii_ioctl_data *data = if_mii(ifr);
3777
Francois Romieu8b4ab282008-11-19 22:05:25 -08003778 return netif_running(dev) ? tp->do_ioctl(tp, data, cmd) : -ENODEV;
3779}
Francois Romieu5f787a12006-08-17 13:02:36 +02003780
Francois Romieucecb5fd2011-04-01 10:21:07 +02003781static int rtl_xmii_ioctl(struct rtl8169_private *tp,
3782 struct mii_ioctl_data *data, int cmd)
Francois Romieu8b4ab282008-11-19 22:05:25 -08003783{
Francois Romieu5f787a12006-08-17 13:02:36 +02003784 switch (cmd) {
3785 case SIOCGMIIPHY:
3786 data->phy_id = 32; /* Internal PHY */
3787 return 0;
3788
3789 case SIOCGMIIREG:
françois romieu4da19632011-01-03 15:07:55 +00003790 data->val_out = rtl_readphy(tp, data->reg_num & 0x1f);
Francois Romieu5f787a12006-08-17 13:02:36 +02003791 return 0;
3792
3793 case SIOCSMIIREG:
françois romieu4da19632011-01-03 15:07:55 +00003794 rtl_writephy(tp, data->reg_num & 0x1f, data->val_in);
Francois Romieu5f787a12006-08-17 13:02:36 +02003795 return 0;
3796 }
3797 return -EOPNOTSUPP;
3798}
3799
Francois Romieu8b4ab282008-11-19 22:05:25 -08003800static int rtl_tbi_ioctl(struct rtl8169_private *tp, struct mii_ioctl_data *data, int cmd)
3801{
3802 return -EOPNOTSUPP;
3803}
3804
Francois Romieufbac58f2007-10-04 22:51:38 +02003805static void rtl_disable_msi(struct pci_dev *pdev, struct rtl8169_private *tp)
3806{
3807 if (tp->features & RTL_FEATURE_MSI) {
3808 pci_disable_msi(pdev);
3809 tp->features &= ~RTL_FEATURE_MSI;
3810 }
3811}
3812
françois romieuc0e45c12011-01-03 15:08:04 +00003813static void __devinit rtl_init_mdio_ops(struct rtl8169_private *tp)
3814{
3815 struct mdio_ops *ops = &tp->mdio_ops;
3816
3817 switch (tp->mac_version) {
3818 case RTL_GIGA_MAC_VER_27:
3819 ops->write = r8168dp_1_mdio_write;
3820 ops->read = r8168dp_1_mdio_read;
3821 break;
françois romieue6de30d2011-01-03 15:08:37 +00003822 case RTL_GIGA_MAC_VER_28:
hayeswang4804b3b2011-03-21 01:50:29 +00003823 case RTL_GIGA_MAC_VER_31:
françois romieue6de30d2011-01-03 15:08:37 +00003824 ops->write = r8168dp_2_mdio_write;
3825 ops->read = r8168dp_2_mdio_read;
3826 break;
Hayes Wangc5583862012-07-02 17:23:22 +08003827 case RTL_GIGA_MAC_VER_40:
3828 case RTL_GIGA_MAC_VER_41:
3829 ops->write = r8168g_mdio_write;
3830 ops->read = r8168g_mdio_read;
3831 break;
françois romieuc0e45c12011-01-03 15:08:04 +00003832 default:
3833 ops->write = r8169_mdio_write;
3834 ops->read = r8169_mdio_read;
3835 break;
3836 }
3837}
3838
David S. Miller1805b2f2011-10-24 18:18:09 -04003839static void rtl_wol_suspend_quirk(struct rtl8169_private *tp)
3840{
3841 void __iomem *ioaddr = tp->mmio_addr;
3842
3843 switch (tp->mac_version) {
3844 case RTL_GIGA_MAC_VER_29:
3845 case RTL_GIGA_MAC_VER_30:
3846 case RTL_GIGA_MAC_VER_32:
3847 case RTL_GIGA_MAC_VER_33:
3848 case RTL_GIGA_MAC_VER_34:
Hayes Wang7e18dca2012-03-30 14:33:02 +08003849 case RTL_GIGA_MAC_VER_37:
Hayes Wangb3d7b2f2012-03-30 14:48:06 +08003850 case RTL_GIGA_MAC_VER_38:
Hayes Wang5598bfe2012-07-02 17:23:21 +08003851 case RTL_GIGA_MAC_VER_39:
Hayes Wangc5583862012-07-02 17:23:22 +08003852 case RTL_GIGA_MAC_VER_40:
3853 case RTL_GIGA_MAC_VER_41:
David S. Miller1805b2f2011-10-24 18:18:09 -04003854 RTL_W32(RxConfig, RTL_R32(RxConfig) |
3855 AcceptBroadcast | AcceptMulticast | AcceptMyPhys);
3856 break;
3857 default:
3858 break;
3859 }
3860}
3861
3862static bool rtl_wol_pll_power_down(struct rtl8169_private *tp)
3863{
3864 if (!(__rtl8169_get_wol(tp) & WAKE_ANY))
3865 return false;
3866
3867 rtl_writephy(tp, 0x1f, 0x0000);
3868 rtl_writephy(tp, MII_BMCR, 0x0000);
3869
3870 rtl_wol_suspend_quirk(tp);
3871
3872 return true;
3873}
3874
françois romieu065c27c2011-01-03 15:08:12 +00003875static void r810x_phy_power_down(struct rtl8169_private *tp)
3876{
3877 rtl_writephy(tp, 0x1f, 0x0000);
3878 rtl_writephy(tp, MII_BMCR, BMCR_PDOWN);
3879}
3880
3881static void r810x_phy_power_up(struct rtl8169_private *tp)
3882{
3883 rtl_writephy(tp, 0x1f, 0x0000);
3884 rtl_writephy(tp, MII_BMCR, BMCR_ANENABLE);
3885}
3886
3887static void r810x_pll_power_down(struct rtl8169_private *tp)
3888{
Hayes Wang00042992012-03-30 14:33:00 +08003889 void __iomem *ioaddr = tp->mmio_addr;
3890
David S. Miller1805b2f2011-10-24 18:18:09 -04003891 if (rtl_wol_pll_power_down(tp))
françois romieu065c27c2011-01-03 15:08:12 +00003892 return;
françois romieu065c27c2011-01-03 15:08:12 +00003893
3894 r810x_phy_power_down(tp);
Hayes Wang00042992012-03-30 14:33:00 +08003895
3896 switch (tp->mac_version) {
3897 case RTL_GIGA_MAC_VER_07:
3898 case RTL_GIGA_MAC_VER_08:
3899 case RTL_GIGA_MAC_VER_09:
3900 case RTL_GIGA_MAC_VER_10:
3901 case RTL_GIGA_MAC_VER_13:
3902 case RTL_GIGA_MAC_VER_16:
3903 break;
3904 default:
3905 RTL_W8(PMCH, RTL_R8(PMCH) & ~0x80);
3906 break;
3907 }
françois romieu065c27c2011-01-03 15:08:12 +00003908}
3909
3910static void r810x_pll_power_up(struct rtl8169_private *tp)
3911{
Hayes Wang00042992012-03-30 14:33:00 +08003912 void __iomem *ioaddr = tp->mmio_addr;
3913
françois romieu065c27c2011-01-03 15:08:12 +00003914 r810x_phy_power_up(tp);
Hayes Wang00042992012-03-30 14:33:00 +08003915
3916 switch (tp->mac_version) {
3917 case RTL_GIGA_MAC_VER_07:
3918 case RTL_GIGA_MAC_VER_08:
3919 case RTL_GIGA_MAC_VER_09:
3920 case RTL_GIGA_MAC_VER_10:
3921 case RTL_GIGA_MAC_VER_13:
3922 case RTL_GIGA_MAC_VER_16:
3923 break;
3924 default:
3925 RTL_W8(PMCH, RTL_R8(PMCH) | 0x80);
3926 break;
3927 }
françois romieu065c27c2011-01-03 15:08:12 +00003928}
3929
3930static void r8168_phy_power_up(struct rtl8169_private *tp)
3931{
3932 rtl_writephy(tp, 0x1f, 0x0000);
hayeswang01dc7fe2011-03-21 01:50:28 +00003933 switch (tp->mac_version) {
3934 case RTL_GIGA_MAC_VER_11:
3935 case RTL_GIGA_MAC_VER_12:
3936 case RTL_GIGA_MAC_VER_17:
3937 case RTL_GIGA_MAC_VER_18:
3938 case RTL_GIGA_MAC_VER_19:
3939 case RTL_GIGA_MAC_VER_20:
3940 case RTL_GIGA_MAC_VER_21:
3941 case RTL_GIGA_MAC_VER_22:
3942 case RTL_GIGA_MAC_VER_23:
3943 case RTL_GIGA_MAC_VER_24:
3944 case RTL_GIGA_MAC_VER_25:
3945 case RTL_GIGA_MAC_VER_26:
3946 case RTL_GIGA_MAC_VER_27:
3947 case RTL_GIGA_MAC_VER_28:
3948 case RTL_GIGA_MAC_VER_31:
3949 rtl_writephy(tp, 0x0e, 0x0000);
3950 break;
3951 default:
3952 break;
3953 }
françois romieu065c27c2011-01-03 15:08:12 +00003954 rtl_writephy(tp, MII_BMCR, BMCR_ANENABLE);
3955}
3956
3957static void r8168_phy_power_down(struct rtl8169_private *tp)
3958{
3959 rtl_writephy(tp, 0x1f, 0x0000);
hayeswang01dc7fe2011-03-21 01:50:28 +00003960 switch (tp->mac_version) {
3961 case RTL_GIGA_MAC_VER_32:
3962 case RTL_GIGA_MAC_VER_33:
3963 rtl_writephy(tp, MII_BMCR, BMCR_ANENABLE | BMCR_PDOWN);
3964 break;
3965
3966 case RTL_GIGA_MAC_VER_11:
3967 case RTL_GIGA_MAC_VER_12:
3968 case RTL_GIGA_MAC_VER_17:
3969 case RTL_GIGA_MAC_VER_18:
3970 case RTL_GIGA_MAC_VER_19:
3971 case RTL_GIGA_MAC_VER_20:
3972 case RTL_GIGA_MAC_VER_21:
3973 case RTL_GIGA_MAC_VER_22:
3974 case RTL_GIGA_MAC_VER_23:
3975 case RTL_GIGA_MAC_VER_24:
3976 case RTL_GIGA_MAC_VER_25:
3977 case RTL_GIGA_MAC_VER_26:
3978 case RTL_GIGA_MAC_VER_27:
3979 case RTL_GIGA_MAC_VER_28:
3980 case RTL_GIGA_MAC_VER_31:
3981 rtl_writephy(tp, 0x0e, 0x0200);
3982 default:
3983 rtl_writephy(tp, MII_BMCR, BMCR_PDOWN);
3984 break;
3985 }
françois romieu065c27c2011-01-03 15:08:12 +00003986}
3987
3988static void r8168_pll_power_down(struct rtl8169_private *tp)
3989{
3990 void __iomem *ioaddr = tp->mmio_addr;
3991
Francois Romieucecb5fd2011-04-01 10:21:07 +02003992 if ((tp->mac_version == RTL_GIGA_MAC_VER_27 ||
3993 tp->mac_version == RTL_GIGA_MAC_VER_28 ||
3994 tp->mac_version == RTL_GIGA_MAC_VER_31) &&
hayeswang4804b3b2011-03-21 01:50:29 +00003995 r8168dp_check_dash(tp)) {
françois romieu065c27c2011-01-03 15:08:12 +00003996 return;
Hayes Wang5d2e1952011-02-22 17:26:22 +08003997 }
françois romieu065c27c2011-01-03 15:08:12 +00003998
Francois Romieucecb5fd2011-04-01 10:21:07 +02003999 if ((tp->mac_version == RTL_GIGA_MAC_VER_23 ||
4000 tp->mac_version == RTL_GIGA_MAC_VER_24) &&
françois romieu065c27c2011-01-03 15:08:12 +00004001 (RTL_R16(CPlusCmd) & ASF)) {
4002 return;
4003 }
4004
hayeswang01dc7fe2011-03-21 01:50:28 +00004005 if (tp->mac_version == RTL_GIGA_MAC_VER_32 ||
4006 tp->mac_version == RTL_GIGA_MAC_VER_33)
Francois Romieufdf6fc02012-07-06 22:40:38 +02004007 rtl_ephy_write(tp, 0x19, 0xff64);
hayeswang01dc7fe2011-03-21 01:50:28 +00004008
David S. Miller1805b2f2011-10-24 18:18:09 -04004009 if (rtl_wol_pll_power_down(tp))
françois romieu065c27c2011-01-03 15:08:12 +00004010 return;
françois romieu065c27c2011-01-03 15:08:12 +00004011
4012 r8168_phy_power_down(tp);
4013
4014 switch (tp->mac_version) {
4015 case RTL_GIGA_MAC_VER_25:
4016 case RTL_GIGA_MAC_VER_26:
Hayes Wang5d2e1952011-02-22 17:26:22 +08004017 case RTL_GIGA_MAC_VER_27:
4018 case RTL_GIGA_MAC_VER_28:
hayeswang4804b3b2011-03-21 01:50:29 +00004019 case RTL_GIGA_MAC_VER_31:
hayeswang01dc7fe2011-03-21 01:50:28 +00004020 case RTL_GIGA_MAC_VER_32:
4021 case RTL_GIGA_MAC_VER_33:
françois romieu065c27c2011-01-03 15:08:12 +00004022 RTL_W8(PMCH, RTL_R8(PMCH) & ~0x80);
4023 break;
4024 }
4025}
4026
4027static void r8168_pll_power_up(struct rtl8169_private *tp)
4028{
4029 void __iomem *ioaddr = tp->mmio_addr;
4030
françois romieu065c27c2011-01-03 15:08:12 +00004031 switch (tp->mac_version) {
4032 case RTL_GIGA_MAC_VER_25:
4033 case RTL_GIGA_MAC_VER_26:
Hayes Wang5d2e1952011-02-22 17:26:22 +08004034 case RTL_GIGA_MAC_VER_27:
4035 case RTL_GIGA_MAC_VER_28:
hayeswang4804b3b2011-03-21 01:50:29 +00004036 case RTL_GIGA_MAC_VER_31:
hayeswang01dc7fe2011-03-21 01:50:28 +00004037 case RTL_GIGA_MAC_VER_32:
4038 case RTL_GIGA_MAC_VER_33:
françois romieu065c27c2011-01-03 15:08:12 +00004039 RTL_W8(PMCH, RTL_R8(PMCH) | 0x80);
4040 break;
4041 }
4042
4043 r8168_phy_power_up(tp);
4044}
4045
Francois Romieud58d46b2011-05-03 16:38:29 +02004046static void rtl_generic_op(struct rtl8169_private *tp,
4047 void (*op)(struct rtl8169_private *))
françois romieu065c27c2011-01-03 15:08:12 +00004048{
4049 if (op)
4050 op(tp);
4051}
4052
4053static void rtl_pll_power_down(struct rtl8169_private *tp)
4054{
Francois Romieud58d46b2011-05-03 16:38:29 +02004055 rtl_generic_op(tp, tp->pll_power_ops.down);
françois romieu065c27c2011-01-03 15:08:12 +00004056}
4057
4058static void rtl_pll_power_up(struct rtl8169_private *tp)
4059{
Francois Romieud58d46b2011-05-03 16:38:29 +02004060 rtl_generic_op(tp, tp->pll_power_ops.up);
françois romieu065c27c2011-01-03 15:08:12 +00004061}
4062
4063static void __devinit rtl_init_pll_power_ops(struct rtl8169_private *tp)
4064{
4065 struct pll_power_ops *ops = &tp->pll_power_ops;
4066
4067 switch (tp->mac_version) {
4068 case RTL_GIGA_MAC_VER_07:
4069 case RTL_GIGA_MAC_VER_08:
4070 case RTL_GIGA_MAC_VER_09:
4071 case RTL_GIGA_MAC_VER_10:
4072 case RTL_GIGA_MAC_VER_16:
Hayes Wang5a5e4442011-02-22 17:26:21 +08004073 case RTL_GIGA_MAC_VER_29:
4074 case RTL_GIGA_MAC_VER_30:
Hayes Wang7e18dca2012-03-30 14:33:02 +08004075 case RTL_GIGA_MAC_VER_37:
Hayes Wang5598bfe2012-07-02 17:23:21 +08004076 case RTL_GIGA_MAC_VER_39:
françois romieu065c27c2011-01-03 15:08:12 +00004077 ops->down = r810x_pll_power_down;
4078 ops->up = r810x_pll_power_up;
4079 break;
4080
4081 case RTL_GIGA_MAC_VER_11:
4082 case RTL_GIGA_MAC_VER_12:
4083 case RTL_GIGA_MAC_VER_17:
4084 case RTL_GIGA_MAC_VER_18:
4085 case RTL_GIGA_MAC_VER_19:
4086 case RTL_GIGA_MAC_VER_20:
4087 case RTL_GIGA_MAC_VER_21:
4088 case RTL_GIGA_MAC_VER_22:
4089 case RTL_GIGA_MAC_VER_23:
4090 case RTL_GIGA_MAC_VER_24:
4091 case RTL_GIGA_MAC_VER_25:
4092 case RTL_GIGA_MAC_VER_26:
4093 case RTL_GIGA_MAC_VER_27:
françois romieue6de30d2011-01-03 15:08:37 +00004094 case RTL_GIGA_MAC_VER_28:
hayeswang4804b3b2011-03-21 01:50:29 +00004095 case RTL_GIGA_MAC_VER_31:
hayeswang01dc7fe2011-03-21 01:50:28 +00004096 case RTL_GIGA_MAC_VER_32:
4097 case RTL_GIGA_MAC_VER_33:
Hayes Wang70090422011-07-06 15:58:06 +08004098 case RTL_GIGA_MAC_VER_34:
Hayes Wangc2218922011-09-06 16:55:18 +08004099 case RTL_GIGA_MAC_VER_35:
4100 case RTL_GIGA_MAC_VER_36:
Hayes Wangb3d7b2f2012-03-30 14:48:06 +08004101 case RTL_GIGA_MAC_VER_38:
Hayes Wangc5583862012-07-02 17:23:22 +08004102 case RTL_GIGA_MAC_VER_40:
4103 case RTL_GIGA_MAC_VER_41:
françois romieu065c27c2011-01-03 15:08:12 +00004104 ops->down = r8168_pll_power_down;
4105 ops->up = r8168_pll_power_up;
4106 break;
4107
4108 default:
4109 ops->down = NULL;
4110 ops->up = NULL;
4111 break;
4112 }
4113}
4114
Hayes Wange542a222011-07-06 15:58:04 +08004115static void rtl_init_rxcfg(struct rtl8169_private *tp)
4116{
4117 void __iomem *ioaddr = tp->mmio_addr;
4118
4119 switch (tp->mac_version) {
4120 case RTL_GIGA_MAC_VER_01:
4121 case RTL_GIGA_MAC_VER_02:
4122 case RTL_GIGA_MAC_VER_03:
4123 case RTL_GIGA_MAC_VER_04:
4124 case RTL_GIGA_MAC_VER_05:
4125 case RTL_GIGA_MAC_VER_06:
4126 case RTL_GIGA_MAC_VER_10:
4127 case RTL_GIGA_MAC_VER_11:
4128 case RTL_GIGA_MAC_VER_12:
4129 case RTL_GIGA_MAC_VER_13:
4130 case RTL_GIGA_MAC_VER_14:
4131 case RTL_GIGA_MAC_VER_15:
4132 case RTL_GIGA_MAC_VER_16:
4133 case RTL_GIGA_MAC_VER_17:
4134 RTL_W32(RxConfig, RX_FIFO_THRESH | RX_DMA_BURST);
4135 break;
4136 case RTL_GIGA_MAC_VER_18:
4137 case RTL_GIGA_MAC_VER_19:
4138 case RTL_GIGA_MAC_VER_20:
4139 case RTL_GIGA_MAC_VER_21:
4140 case RTL_GIGA_MAC_VER_22:
4141 case RTL_GIGA_MAC_VER_23:
4142 case RTL_GIGA_MAC_VER_24:
françois romieueb2dc352012-06-20 12:09:18 +00004143 case RTL_GIGA_MAC_VER_34:
Hayes Wange542a222011-07-06 15:58:04 +08004144 RTL_W32(RxConfig, RX128_INT_EN | RX_MULTI_EN | RX_DMA_BURST);
4145 break;
4146 default:
4147 RTL_W32(RxConfig, RX128_INT_EN | RX_DMA_BURST);
4148 break;
4149 }
4150}
4151
Hayes Wang92fc43b2011-07-06 15:58:03 +08004152static void rtl8169_init_ring_indexes(struct rtl8169_private *tp)
4153{
4154 tp->dirty_tx = tp->dirty_rx = tp->cur_tx = tp->cur_rx = 0;
4155}
4156
Francois Romieud58d46b2011-05-03 16:38:29 +02004157static void rtl_hw_jumbo_enable(struct rtl8169_private *tp)
4158{
françois romieu9c5028e2012-03-02 04:43:14 +00004159 void __iomem *ioaddr = tp->mmio_addr;
4160
4161 RTL_W8(Cfg9346, Cfg9346_Unlock);
Francois Romieud58d46b2011-05-03 16:38:29 +02004162 rtl_generic_op(tp, tp->jumbo_ops.enable);
françois romieu9c5028e2012-03-02 04:43:14 +00004163 RTL_W8(Cfg9346, Cfg9346_Lock);
Francois Romieud58d46b2011-05-03 16:38:29 +02004164}
4165
4166static void rtl_hw_jumbo_disable(struct rtl8169_private *tp)
4167{
françois romieu9c5028e2012-03-02 04:43:14 +00004168 void __iomem *ioaddr = tp->mmio_addr;
4169
4170 RTL_W8(Cfg9346, Cfg9346_Unlock);
Francois Romieud58d46b2011-05-03 16:38:29 +02004171 rtl_generic_op(tp, tp->jumbo_ops.disable);
françois romieu9c5028e2012-03-02 04:43:14 +00004172 RTL_W8(Cfg9346, Cfg9346_Lock);
Francois Romieud58d46b2011-05-03 16:38:29 +02004173}
4174
4175static void r8168c_hw_jumbo_enable(struct rtl8169_private *tp)
4176{
4177 void __iomem *ioaddr = tp->mmio_addr;
4178
4179 RTL_W8(Config3, RTL_R8(Config3) | Jumbo_En0);
4180 RTL_W8(Config4, RTL_R8(Config4) | Jumbo_En1);
4181 rtl_tx_performance_tweak(tp->pci_dev, 0x2 << MAX_READ_REQUEST_SHIFT);
4182}
4183
4184static void r8168c_hw_jumbo_disable(struct rtl8169_private *tp)
4185{
4186 void __iomem *ioaddr = tp->mmio_addr;
4187
4188 RTL_W8(Config3, RTL_R8(Config3) & ~Jumbo_En0);
4189 RTL_W8(Config4, RTL_R8(Config4) & ~Jumbo_En1);
4190 rtl_tx_performance_tweak(tp->pci_dev, 0x5 << MAX_READ_REQUEST_SHIFT);
4191}
4192
4193static void r8168dp_hw_jumbo_enable(struct rtl8169_private *tp)
4194{
4195 void __iomem *ioaddr = tp->mmio_addr;
4196
4197 RTL_W8(Config3, RTL_R8(Config3) | Jumbo_En0);
4198}
4199
4200static void r8168dp_hw_jumbo_disable(struct rtl8169_private *tp)
4201{
4202 void __iomem *ioaddr = tp->mmio_addr;
4203
4204 RTL_W8(Config3, RTL_R8(Config3) & ~Jumbo_En0);
4205}
4206
4207static void r8168e_hw_jumbo_enable(struct rtl8169_private *tp)
4208{
4209 void __iomem *ioaddr = tp->mmio_addr;
Francois Romieud58d46b2011-05-03 16:38:29 +02004210
4211 RTL_W8(MaxTxPacketSize, 0x3f);
4212 RTL_W8(Config3, RTL_R8(Config3) | Jumbo_En0);
4213 RTL_W8(Config4, RTL_R8(Config4) | 0x01);
Francois Romieu4512ff92011-12-22 18:59:37 +01004214 rtl_tx_performance_tweak(tp->pci_dev, 0x2 << MAX_READ_REQUEST_SHIFT);
Francois Romieud58d46b2011-05-03 16:38:29 +02004215}
4216
4217static void r8168e_hw_jumbo_disable(struct rtl8169_private *tp)
4218{
4219 void __iomem *ioaddr = tp->mmio_addr;
Francois Romieud58d46b2011-05-03 16:38:29 +02004220
4221 RTL_W8(MaxTxPacketSize, 0x0c);
4222 RTL_W8(Config3, RTL_R8(Config3) & ~Jumbo_En0);
4223 RTL_W8(Config4, RTL_R8(Config4) & ~0x01);
Francois Romieu4512ff92011-12-22 18:59:37 +01004224 rtl_tx_performance_tweak(tp->pci_dev, 0x5 << MAX_READ_REQUEST_SHIFT);
Francois Romieud58d46b2011-05-03 16:38:29 +02004225}
4226
4227static void r8168b_0_hw_jumbo_enable(struct rtl8169_private *tp)
4228{
4229 rtl_tx_performance_tweak(tp->pci_dev,
4230 (0x2 << MAX_READ_REQUEST_SHIFT) | PCI_EXP_DEVCTL_NOSNOOP_EN);
4231}
4232
4233static void r8168b_0_hw_jumbo_disable(struct rtl8169_private *tp)
4234{
4235 rtl_tx_performance_tweak(tp->pci_dev,
4236 (0x5 << MAX_READ_REQUEST_SHIFT) | PCI_EXP_DEVCTL_NOSNOOP_EN);
4237}
4238
4239static void r8168b_1_hw_jumbo_enable(struct rtl8169_private *tp)
4240{
4241 void __iomem *ioaddr = tp->mmio_addr;
4242
4243 r8168b_0_hw_jumbo_enable(tp);
4244
4245 RTL_W8(Config4, RTL_R8(Config4) | (1 << 0));
4246}
4247
4248static void r8168b_1_hw_jumbo_disable(struct rtl8169_private *tp)
4249{
4250 void __iomem *ioaddr = tp->mmio_addr;
4251
4252 r8168b_0_hw_jumbo_disable(tp);
4253
4254 RTL_W8(Config4, RTL_R8(Config4) & ~(1 << 0));
4255}
4256
4257static void __devinit rtl_init_jumbo_ops(struct rtl8169_private *tp)
4258{
4259 struct jumbo_ops *ops = &tp->jumbo_ops;
4260
4261 switch (tp->mac_version) {
4262 case RTL_GIGA_MAC_VER_11:
4263 ops->disable = r8168b_0_hw_jumbo_disable;
4264 ops->enable = r8168b_0_hw_jumbo_enable;
4265 break;
4266 case RTL_GIGA_MAC_VER_12:
4267 case RTL_GIGA_MAC_VER_17:
4268 ops->disable = r8168b_1_hw_jumbo_disable;
4269 ops->enable = r8168b_1_hw_jumbo_enable;
4270 break;
4271 case RTL_GIGA_MAC_VER_18: /* Wild guess. Needs info from Realtek. */
4272 case RTL_GIGA_MAC_VER_19:
4273 case RTL_GIGA_MAC_VER_20:
4274 case RTL_GIGA_MAC_VER_21: /* Wild guess. Needs info from Realtek. */
4275 case RTL_GIGA_MAC_VER_22:
4276 case RTL_GIGA_MAC_VER_23:
4277 case RTL_GIGA_MAC_VER_24:
4278 case RTL_GIGA_MAC_VER_25:
4279 case RTL_GIGA_MAC_VER_26:
4280 ops->disable = r8168c_hw_jumbo_disable;
4281 ops->enable = r8168c_hw_jumbo_enable;
4282 break;
4283 case RTL_GIGA_MAC_VER_27:
4284 case RTL_GIGA_MAC_VER_28:
4285 ops->disable = r8168dp_hw_jumbo_disable;
4286 ops->enable = r8168dp_hw_jumbo_enable;
4287 break;
4288 case RTL_GIGA_MAC_VER_31: /* Wild guess. Needs info from Realtek. */
4289 case RTL_GIGA_MAC_VER_32:
4290 case RTL_GIGA_MAC_VER_33:
4291 case RTL_GIGA_MAC_VER_34:
4292 ops->disable = r8168e_hw_jumbo_disable;
4293 ops->enable = r8168e_hw_jumbo_enable;
4294 break;
4295
4296 /*
4297 * No action needed for jumbo frames with 8169.
4298 * No jumbo for 810x at all.
4299 */
Hayes Wangc5583862012-07-02 17:23:22 +08004300 case RTL_GIGA_MAC_VER_40:
4301 case RTL_GIGA_MAC_VER_41:
Francois Romieud58d46b2011-05-03 16:38:29 +02004302 default:
4303 ops->disable = NULL;
4304 ops->enable = NULL;
4305 break;
4306 }
4307}
4308
Francois Romieuffc46952012-07-06 14:19:23 +02004309DECLARE_RTL_COND(rtl_chipcmd_cond)
4310{
4311 void __iomem *ioaddr = tp->mmio_addr;
4312
4313 return RTL_R8(ChipCmd) & CmdReset;
4314}
4315
Francois Romieu6f43adc2011-04-29 15:05:51 +02004316static void rtl_hw_reset(struct rtl8169_private *tp)
4317{
4318 void __iomem *ioaddr = tp->mmio_addr;
Francois Romieu6f43adc2011-04-29 15:05:51 +02004319
Francois Romieu6f43adc2011-04-29 15:05:51 +02004320 RTL_W8(ChipCmd, CmdReset);
4321
Francois Romieuffc46952012-07-06 14:19:23 +02004322 rtl_udelay_loop_wait_low(tp, &rtl_chipcmd_cond, 100, 100);
Francois Romieu6f43adc2011-04-29 15:05:51 +02004323}
4324
Francois Romieub6ffd972011-06-17 17:00:05 +02004325static void rtl_request_uncached_firmware(struct rtl8169_private *tp)
4326{
4327 struct rtl_fw *rtl_fw;
4328 const char *name;
4329 int rc = -ENOMEM;
4330
4331 name = rtl_lookup_firmware_name(tp);
4332 if (!name)
4333 goto out_no_firmware;
4334
4335 rtl_fw = kzalloc(sizeof(*rtl_fw), GFP_KERNEL);
4336 if (!rtl_fw)
4337 goto err_warn;
4338
4339 rc = request_firmware(&rtl_fw->fw, name, &tp->pci_dev->dev);
4340 if (rc < 0)
4341 goto err_free;
4342
Francois Romieufd112f22011-06-18 00:10:29 +02004343 rc = rtl_check_firmware(tp, rtl_fw);
4344 if (rc < 0)
4345 goto err_release_firmware;
4346
Francois Romieub6ffd972011-06-17 17:00:05 +02004347 tp->rtl_fw = rtl_fw;
4348out:
4349 return;
4350
Francois Romieufd112f22011-06-18 00:10:29 +02004351err_release_firmware:
4352 release_firmware(rtl_fw->fw);
Francois Romieub6ffd972011-06-17 17:00:05 +02004353err_free:
4354 kfree(rtl_fw);
4355err_warn:
4356 netif_warn(tp, ifup, tp->dev, "unable to load firmware patch %s (%d)\n",
4357 name, rc);
4358out_no_firmware:
4359 tp->rtl_fw = NULL;
4360 goto out;
4361}
4362
François Romieu953a12c2011-04-24 17:38:48 +02004363static void rtl_request_firmware(struct rtl8169_private *tp)
4364{
Francois Romieub6ffd972011-06-17 17:00:05 +02004365 if (IS_ERR(tp->rtl_fw))
4366 rtl_request_uncached_firmware(tp);
François Romieu953a12c2011-04-24 17:38:48 +02004367}
4368
Hayes Wang92fc43b2011-07-06 15:58:03 +08004369static void rtl_rx_close(struct rtl8169_private *tp)
4370{
4371 void __iomem *ioaddr = tp->mmio_addr;
Hayes Wang92fc43b2011-07-06 15:58:03 +08004372
Francois Romieu1687b562011-07-19 17:21:29 +02004373 RTL_W32(RxConfig, RTL_R32(RxConfig) & ~RX_CONFIG_ACCEPT_MASK);
Hayes Wang92fc43b2011-07-06 15:58:03 +08004374}
4375
Francois Romieuffc46952012-07-06 14:19:23 +02004376DECLARE_RTL_COND(rtl_npq_cond)
4377{
4378 void __iomem *ioaddr = tp->mmio_addr;
4379
4380 return RTL_R8(TxPoll) & NPQ;
4381}
4382
4383DECLARE_RTL_COND(rtl_txcfg_empty_cond)
4384{
4385 void __iomem *ioaddr = tp->mmio_addr;
4386
4387 return RTL_R32(TxConfig) & TXCFG_EMPTY;
4388}
4389
françois romieue6de30d2011-01-03 15:08:37 +00004390static void rtl8169_hw_reset(struct rtl8169_private *tp)
Linus Torvalds1da177e2005-04-16 15:20:36 -07004391{
françois romieue6de30d2011-01-03 15:08:37 +00004392 void __iomem *ioaddr = tp->mmio_addr;
4393
Linus Torvalds1da177e2005-04-16 15:20:36 -07004394 /* Disable interrupts */
françois romieu811fd302011-12-04 20:30:45 +00004395 rtl8169_irq_mask_and_ack(tp);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004396
Hayes Wang92fc43b2011-07-06 15:58:03 +08004397 rtl_rx_close(tp);
4398
Hayes Wang5d2e1952011-02-22 17:26:22 +08004399 if (tp->mac_version == RTL_GIGA_MAC_VER_27 ||
hayeswang4804b3b2011-03-21 01:50:29 +00004400 tp->mac_version == RTL_GIGA_MAC_VER_28 ||
4401 tp->mac_version == RTL_GIGA_MAC_VER_31) {
Francois Romieuffc46952012-07-06 14:19:23 +02004402 rtl_udelay_loop_wait_low(tp, &rtl_npq_cond, 20, 42*42);
Hayes Wangc2218922011-09-06 16:55:18 +08004403 } else if (tp->mac_version == RTL_GIGA_MAC_VER_34 ||
4404 tp->mac_version == RTL_GIGA_MAC_VER_35 ||
Hayes Wang7e18dca2012-03-30 14:33:02 +08004405 tp->mac_version == RTL_GIGA_MAC_VER_36 ||
Hayes Wangb3d7b2f2012-03-30 14:48:06 +08004406 tp->mac_version == RTL_GIGA_MAC_VER_37 ||
Hayes Wangc5583862012-07-02 17:23:22 +08004407 tp->mac_version == RTL_GIGA_MAC_VER_40 ||
4408 tp->mac_version == RTL_GIGA_MAC_VER_41 ||
Hayes Wangb3d7b2f2012-03-30 14:48:06 +08004409 tp->mac_version == RTL_GIGA_MAC_VER_38) {
David S. Miller8decf862011-09-22 03:23:13 -04004410 RTL_W8(ChipCmd, RTL_R8(ChipCmd) | StopReq);
Francois Romieuffc46952012-07-06 14:19:23 +02004411 rtl_udelay_loop_wait_high(tp, &rtl_txcfg_empty_cond, 100, 666);
Hayes Wang92fc43b2011-07-06 15:58:03 +08004412 } else {
4413 RTL_W8(ChipCmd, RTL_R8(ChipCmd) | StopReq);
4414 udelay(100);
françois romieue6de30d2011-01-03 15:08:37 +00004415 }
4416
Hayes Wang92fc43b2011-07-06 15:58:03 +08004417 rtl_hw_reset(tp);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004418}
4419
Francois Romieu7f796d832007-06-11 23:04:41 +02004420static void rtl_set_rx_tx_config_registers(struct rtl8169_private *tp)
Francois Romieu9cb427b2006-11-02 00:10:16 +01004421{
4422 void __iomem *ioaddr = tp->mmio_addr;
Francois Romieu9cb427b2006-11-02 00:10:16 +01004423
4424 /* Set DMA burst size and Interframe Gap Time */
4425 RTL_W32(TxConfig, (TX_DMA_BURST << TxDMAShift) |
4426 (InterFrameGap << TxInterFrameGapShift));
4427}
4428
Francois Romieu07ce4062007-02-23 23:36:39 +01004429static void rtl_hw_start(struct net_device *dev)
Linus Torvalds1da177e2005-04-16 15:20:36 -07004430{
4431 struct rtl8169_private *tp = netdev_priv(dev);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004432
Francois Romieu07ce4062007-02-23 23:36:39 +01004433 tp->hw_start(dev);
4434
Francois Romieuda78dbf2012-01-26 14:18:23 +01004435 rtl_irq_enable_all(tp);
Francois Romieu07ce4062007-02-23 23:36:39 +01004436}
4437
Francois Romieu7f796d832007-06-11 23:04:41 +02004438static void rtl_set_rx_tx_desc_registers(struct rtl8169_private *tp,
4439 void __iomem *ioaddr)
4440{
4441 /*
4442 * Magic spell: some iop3xx ARM board needs the TxDescAddrHigh
4443 * register to be written before TxDescAddrLow to work.
4444 * Switching from MMIO to I/O access fixes the issue as well.
4445 */
4446 RTL_W32(TxDescStartAddrHigh, ((u64) tp->TxPhyAddr) >> 32);
Yang Hongyang284901a2009-04-06 19:01:15 -07004447 RTL_W32(TxDescStartAddrLow, ((u64) tp->TxPhyAddr) & DMA_BIT_MASK(32));
Francois Romieu7f796d832007-06-11 23:04:41 +02004448 RTL_W32(RxDescAddrHigh, ((u64) tp->RxPhyAddr) >> 32);
Yang Hongyang284901a2009-04-06 19:01:15 -07004449 RTL_W32(RxDescAddrLow, ((u64) tp->RxPhyAddr) & DMA_BIT_MASK(32));
Francois Romieu7f796d832007-06-11 23:04:41 +02004450}
4451
4452static u16 rtl_rw_cpluscmd(void __iomem *ioaddr)
4453{
4454 u16 cmd;
4455
4456 cmd = RTL_R16(CPlusCmd);
4457 RTL_W16(CPlusCmd, cmd);
4458 return cmd;
4459}
4460
Eric Dumazetfdd7b4c2009-06-09 04:01:02 -07004461static void rtl_set_rx_max_size(void __iomem *ioaddr, unsigned int rx_buf_sz)
Francois Romieu7f796d832007-06-11 23:04:41 +02004462{
4463 /* Low hurts. Let's disable the filtering. */
Raimonds Cicans207d6e872009-10-26 10:52:37 +00004464 RTL_W16(RxMaxSize, rx_buf_sz + 1);
Francois Romieu7f796d832007-06-11 23:04:41 +02004465}
4466
Francois Romieu6dccd162007-02-13 23:38:05 +01004467static void rtl8169_set_magic_reg(void __iomem *ioaddr, unsigned mac_version)
4468{
Francois Romieu37441002011-06-17 22:58:54 +02004469 static const struct rtl_cfg2_info {
Francois Romieu6dccd162007-02-13 23:38:05 +01004470 u32 mac_version;
4471 u32 clk;
4472 u32 val;
4473 } cfg2_info [] = {
4474 { RTL_GIGA_MAC_VER_05, PCI_Clock_33MHz, 0x000fff00 }, // 8110SCd
4475 { RTL_GIGA_MAC_VER_05, PCI_Clock_66MHz, 0x000fffff },
4476 { RTL_GIGA_MAC_VER_06, PCI_Clock_33MHz, 0x00ffff00 }, // 8110SCe
4477 { RTL_GIGA_MAC_VER_06, PCI_Clock_66MHz, 0x00ffffff }
Francois Romieu37441002011-06-17 22:58:54 +02004478 };
4479 const struct rtl_cfg2_info *p = cfg2_info;
Francois Romieu6dccd162007-02-13 23:38:05 +01004480 unsigned int i;
4481 u32 clk;
4482
4483 clk = RTL_R8(Config2) & PCI_Clock_66MHz;
Francois Romieucadf1852008-01-03 23:38:38 +01004484 for (i = 0; i < ARRAY_SIZE(cfg2_info); i++, p++) {
Francois Romieu6dccd162007-02-13 23:38:05 +01004485 if ((p->mac_version == mac_version) && (p->clk == clk)) {
4486 RTL_W32(0x7c, p->val);
4487 break;
4488 }
4489 }
4490}
4491
Francois Romieue6b763e2012-03-08 09:35:39 +01004492static void rtl_set_rx_mode(struct net_device *dev)
4493{
4494 struct rtl8169_private *tp = netdev_priv(dev);
4495 void __iomem *ioaddr = tp->mmio_addr;
4496 u32 mc_filter[2]; /* Multicast hash filter */
4497 int rx_mode;
4498 u32 tmp = 0;
4499
4500 if (dev->flags & IFF_PROMISC) {
4501 /* Unconditionally log net taps. */
4502 netif_notice(tp, link, dev, "Promiscuous mode enabled\n");
4503 rx_mode =
4504 AcceptBroadcast | AcceptMulticast | AcceptMyPhys |
4505 AcceptAllPhys;
4506 mc_filter[1] = mc_filter[0] = 0xffffffff;
4507 } else if ((netdev_mc_count(dev) > multicast_filter_limit) ||
4508 (dev->flags & IFF_ALLMULTI)) {
4509 /* Too many to filter perfectly -- accept all multicasts. */
4510 rx_mode = AcceptBroadcast | AcceptMulticast | AcceptMyPhys;
4511 mc_filter[1] = mc_filter[0] = 0xffffffff;
4512 } else {
4513 struct netdev_hw_addr *ha;
4514
4515 rx_mode = AcceptBroadcast | AcceptMyPhys;
4516 mc_filter[1] = mc_filter[0] = 0;
4517 netdev_for_each_mc_addr(ha, dev) {
4518 int bit_nr = ether_crc(ETH_ALEN, ha->addr) >> 26;
4519 mc_filter[bit_nr >> 5] |= 1 << (bit_nr & 31);
4520 rx_mode |= AcceptMulticast;
4521 }
4522 }
4523
4524 if (dev->features & NETIF_F_RXALL)
4525 rx_mode |= (AcceptErr | AcceptRunt);
4526
4527 tmp = (RTL_R32(RxConfig) & ~RX_CONFIG_ACCEPT_MASK) | rx_mode;
4528
4529 if (tp->mac_version > RTL_GIGA_MAC_VER_06) {
4530 u32 data = mc_filter[0];
4531
4532 mc_filter[0] = swab32(mc_filter[1]);
4533 mc_filter[1] = swab32(data);
4534 }
4535
4536 RTL_W32(MAR0 + 4, mc_filter[1]);
4537 RTL_W32(MAR0 + 0, mc_filter[0]);
4538
4539 RTL_W32(RxConfig, tmp);
4540}
4541
Francois Romieu07ce4062007-02-23 23:36:39 +01004542static void rtl_hw_start_8169(struct net_device *dev)
4543{
4544 struct rtl8169_private *tp = netdev_priv(dev);
4545 void __iomem *ioaddr = tp->mmio_addr;
4546 struct pci_dev *pdev = tp->pci_dev;
Francois Romieu07ce4062007-02-23 23:36:39 +01004547
Francois Romieu9cb427b2006-11-02 00:10:16 +01004548 if (tp->mac_version == RTL_GIGA_MAC_VER_05) {
4549 RTL_W16(CPlusCmd, RTL_R16(CPlusCmd) | PCIMulRW);
4550 pci_write_config_byte(pdev, PCI_CACHE_LINE_SIZE, 0x08);
4551 }
4552
Linus Torvalds1da177e2005-04-16 15:20:36 -07004553 RTL_W8(Cfg9346, Cfg9346_Unlock);
Francois Romieucecb5fd2011-04-01 10:21:07 +02004554 if (tp->mac_version == RTL_GIGA_MAC_VER_01 ||
4555 tp->mac_version == RTL_GIGA_MAC_VER_02 ||
4556 tp->mac_version == RTL_GIGA_MAC_VER_03 ||
4557 tp->mac_version == RTL_GIGA_MAC_VER_04)
Francois Romieu9cb427b2006-11-02 00:10:16 +01004558 RTL_W8(ChipCmd, CmdTxEnb | CmdRxEnb);
4559
Hayes Wange542a222011-07-06 15:58:04 +08004560 rtl_init_rxcfg(tp);
4561
françois romieuf0298f82011-01-03 15:07:42 +00004562 RTL_W8(EarlyTxThres, NoEarlyTx);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004563
Eric Dumazet6f0333b2010-10-11 11:17:47 +00004564 rtl_set_rx_max_size(ioaddr, rx_buf_sz);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004565
Francois Romieucecb5fd2011-04-01 10:21:07 +02004566 if (tp->mac_version == RTL_GIGA_MAC_VER_01 ||
4567 tp->mac_version == RTL_GIGA_MAC_VER_02 ||
4568 tp->mac_version == RTL_GIGA_MAC_VER_03 ||
4569 tp->mac_version == RTL_GIGA_MAC_VER_04)
Francois Romieuc946b302007-10-04 00:42:50 +02004570 rtl_set_rx_tx_config_registers(tp);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004571
Francois Romieu7f796d832007-06-11 23:04:41 +02004572 tp->cp_cmd |= rtl_rw_cpluscmd(ioaddr) | PCIMulRW;
Francois Romieubcf0bf92006-07-26 23:14:13 +02004573
Francois Romieucecb5fd2011-04-01 10:21:07 +02004574 if (tp->mac_version == RTL_GIGA_MAC_VER_02 ||
4575 tp->mac_version == RTL_GIGA_MAC_VER_03) {
Joe Perches06fa7352007-10-18 21:15:00 +02004576 dprintk("Set MAC Reg C+CR Offset 0xE0. "
Linus Torvalds1da177e2005-04-16 15:20:36 -07004577 "Bit-3 and bit-14 MUST be 1\n");
Francois Romieubcf0bf92006-07-26 23:14:13 +02004578 tp->cp_cmd |= (1 << 14);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004579 }
4580
Francois Romieubcf0bf92006-07-26 23:14:13 +02004581 RTL_W16(CPlusCmd, tp->cp_cmd);
4582
Francois Romieu6dccd162007-02-13 23:38:05 +01004583 rtl8169_set_magic_reg(ioaddr, tp->mac_version);
4584
Linus Torvalds1da177e2005-04-16 15:20:36 -07004585 /*
4586 * Undocumented corner. Supposedly:
4587 * (TxTimer << 12) | (TxPackets << 8) | (RxTimer << 4) | RxPackets
4588 */
4589 RTL_W16(IntrMitigate, 0x0000);
4590
Francois Romieu7f796d832007-06-11 23:04:41 +02004591 rtl_set_rx_tx_desc_registers(tp, ioaddr);
Francois Romieu9cb427b2006-11-02 00:10:16 +01004592
Francois Romieucecb5fd2011-04-01 10:21:07 +02004593 if (tp->mac_version != RTL_GIGA_MAC_VER_01 &&
4594 tp->mac_version != RTL_GIGA_MAC_VER_02 &&
4595 tp->mac_version != RTL_GIGA_MAC_VER_03 &&
4596 tp->mac_version != RTL_GIGA_MAC_VER_04) {
Francois Romieuc946b302007-10-04 00:42:50 +02004597 RTL_W8(ChipCmd, CmdTxEnb | CmdRxEnb);
4598 rtl_set_rx_tx_config_registers(tp);
4599 }
4600
Linus Torvalds1da177e2005-04-16 15:20:36 -07004601 RTL_W8(Cfg9346, Cfg9346_Lock);
Francois Romieub518fa82006-08-16 15:23:13 +02004602
4603 /* Initially a 10 us delay. Turned it into a PCI commit. - FR */
4604 RTL_R8(IntrMask);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004605
4606 RTL_W32(RxMissed, 0);
4607
Francois Romieu07ce4062007-02-23 23:36:39 +01004608 rtl_set_rx_mode(dev);
Linus Torvalds1da177e2005-04-16 15:20:36 -07004609
4610 /* no early-rx interrupts */
4611 RTL_W16(MultiIntr, RTL_R16(MultiIntr) & 0xF000);
Francois Romieu07ce4062007-02-23 23:36:39 +01004612}
Linus Torvalds1da177e2005-04-16 15:20:36 -07004613
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004614static void rtl_csi_write(struct rtl8169_private *tp, int addr, int value)
4615{
4616 if (tp->csi_ops.write)
Francois Romieu52989f02012-07-06 13:37:00 +02004617 tp->csi_ops.write(tp, addr, value);
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004618}
4619
4620static u32 rtl_csi_read(struct rtl8169_private *tp, int addr)
4621{
Francois Romieu52989f02012-07-06 13:37:00 +02004622 return tp->csi_ops.read ? tp->csi_ops.read(tp, addr) : ~0;
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004623}
4624
4625static void rtl_csi_access_enable(struct rtl8169_private *tp, u32 bits)
Francois Romieudacf8152008-08-02 20:44:13 +02004626{
4627 u32 csi;
4628
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004629 csi = rtl_csi_read(tp, 0x070c) & 0x00ffffff;
4630 rtl_csi_write(tp, 0x070c, csi | bits);
françois romieu650e8d52011-01-03 15:08:29 +00004631}
4632
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004633static void rtl_csi_access_enable_1(struct rtl8169_private *tp)
françois romieue6de30d2011-01-03 15:08:37 +00004634{
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004635 rtl_csi_access_enable(tp, 0x17000000);
françois romieue6de30d2011-01-03 15:08:37 +00004636}
4637
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004638static void rtl_csi_access_enable_2(struct rtl8169_private *tp)
françois romieu650e8d52011-01-03 15:08:29 +00004639{
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004640 rtl_csi_access_enable(tp, 0x27000000);
4641}
4642
Francois Romieuffc46952012-07-06 14:19:23 +02004643DECLARE_RTL_COND(rtl_csiar_cond)
4644{
4645 void __iomem *ioaddr = tp->mmio_addr;
4646
4647 return RTL_R32(CSIAR) & CSIAR_FLAG;
4648}
4649
Francois Romieu52989f02012-07-06 13:37:00 +02004650static void r8169_csi_write(struct rtl8169_private *tp, int addr, int value)
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004651{
Francois Romieu52989f02012-07-06 13:37:00 +02004652 void __iomem *ioaddr = tp->mmio_addr;
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004653
4654 RTL_W32(CSIDR, value);
4655 RTL_W32(CSIAR, CSIAR_WRITE_CMD | (addr & CSIAR_ADDR_MASK) |
4656 CSIAR_BYTE_ENABLE << CSIAR_BYTE_ENABLE_SHIFT);
4657
Francois Romieuffc46952012-07-06 14:19:23 +02004658 rtl_udelay_loop_wait_low(tp, &rtl_csiar_cond, 10, 100);
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004659}
4660
Francois Romieu52989f02012-07-06 13:37:00 +02004661static u32 r8169_csi_read(struct rtl8169_private *tp, int addr)
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004662{
Francois Romieu52989f02012-07-06 13:37:00 +02004663 void __iomem *ioaddr = tp->mmio_addr;
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004664
4665 RTL_W32(CSIAR, (addr & CSIAR_ADDR_MASK) |
4666 CSIAR_BYTE_ENABLE << CSIAR_BYTE_ENABLE_SHIFT);
4667
Francois Romieuffc46952012-07-06 14:19:23 +02004668 return rtl_udelay_loop_wait_high(tp, &rtl_csiar_cond, 10, 100) ?
4669 RTL_R32(CSIDR) : ~0;
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004670}
4671
Francois Romieu52989f02012-07-06 13:37:00 +02004672static void r8402_csi_write(struct rtl8169_private *tp, int addr, int value)
Hayes Wang7e18dca2012-03-30 14:33:02 +08004673{
Francois Romieu52989f02012-07-06 13:37:00 +02004674 void __iomem *ioaddr = tp->mmio_addr;
Hayes Wang7e18dca2012-03-30 14:33:02 +08004675
4676 RTL_W32(CSIDR, value);
4677 RTL_W32(CSIAR, CSIAR_WRITE_CMD | (addr & CSIAR_ADDR_MASK) |
4678 CSIAR_BYTE_ENABLE << CSIAR_BYTE_ENABLE_SHIFT |
4679 CSIAR_FUNC_NIC);
4680
Francois Romieuffc46952012-07-06 14:19:23 +02004681 rtl_udelay_loop_wait_low(tp, &rtl_csiar_cond, 10, 100);
Hayes Wang7e18dca2012-03-30 14:33:02 +08004682}
4683
Francois Romieu52989f02012-07-06 13:37:00 +02004684static u32 r8402_csi_read(struct rtl8169_private *tp, int addr)
Hayes Wang7e18dca2012-03-30 14:33:02 +08004685{
Francois Romieu52989f02012-07-06 13:37:00 +02004686 void __iomem *ioaddr = tp->mmio_addr;
Hayes Wang7e18dca2012-03-30 14:33:02 +08004687
4688 RTL_W32(CSIAR, (addr & CSIAR_ADDR_MASK) | CSIAR_FUNC_NIC |
4689 CSIAR_BYTE_ENABLE << CSIAR_BYTE_ENABLE_SHIFT);
4690
Francois Romieuffc46952012-07-06 14:19:23 +02004691 return rtl_udelay_loop_wait_high(tp, &rtl_csiar_cond, 10, 100) ?
4692 RTL_R32(CSIDR) : ~0;
Hayes Wang7e18dca2012-03-30 14:33:02 +08004693}
4694
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004695static void __devinit rtl_init_csi_ops(struct rtl8169_private *tp)
4696{
4697 struct csi_ops *ops = &tp->csi_ops;
4698
4699 switch (tp->mac_version) {
4700 case RTL_GIGA_MAC_VER_01:
4701 case RTL_GIGA_MAC_VER_02:
4702 case RTL_GIGA_MAC_VER_03:
4703 case RTL_GIGA_MAC_VER_04:
4704 case RTL_GIGA_MAC_VER_05:
4705 case RTL_GIGA_MAC_VER_06:
4706 case RTL_GIGA_MAC_VER_10:
4707 case RTL_GIGA_MAC_VER_11:
4708 case RTL_GIGA_MAC_VER_12:
4709 case RTL_GIGA_MAC_VER_13:
4710 case RTL_GIGA_MAC_VER_14:
4711 case RTL_GIGA_MAC_VER_15:
4712 case RTL_GIGA_MAC_VER_16:
4713 case RTL_GIGA_MAC_VER_17:
4714 ops->write = NULL;
4715 ops->read = NULL;
4716 break;
4717
Hayes Wang7e18dca2012-03-30 14:33:02 +08004718 case RTL_GIGA_MAC_VER_37:
Hayes Wangb3d7b2f2012-03-30 14:48:06 +08004719 case RTL_GIGA_MAC_VER_38:
Hayes Wang7e18dca2012-03-30 14:33:02 +08004720 ops->write = r8402_csi_write;
4721 ops->read = r8402_csi_read;
4722 break;
4723
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004724 default:
4725 ops->write = r8169_csi_write;
4726 ops->read = r8169_csi_read;
4727 break;
4728 }
Francois Romieudacf8152008-08-02 20:44:13 +02004729}
4730
4731struct ephy_info {
4732 unsigned int offset;
4733 u16 mask;
4734 u16 bits;
4735};
4736
Francois Romieufdf6fc02012-07-06 22:40:38 +02004737static void rtl_ephy_init(struct rtl8169_private *tp, const struct ephy_info *e,
4738 int len)
Francois Romieudacf8152008-08-02 20:44:13 +02004739{
4740 u16 w;
4741
4742 while (len-- > 0) {
Francois Romieufdf6fc02012-07-06 22:40:38 +02004743 w = (rtl_ephy_read(tp, e->offset) & ~e->mask) | e->bits;
4744 rtl_ephy_write(tp, e->offset, w);
Francois Romieudacf8152008-08-02 20:44:13 +02004745 e++;
4746 }
4747}
4748
Francois Romieub726e492008-06-28 12:22:59 +02004749static void rtl_disable_clock_request(struct pci_dev *pdev)
4750{
Jon Masone44daad2011-06-27 07:46:31 +00004751 int cap = pci_pcie_cap(pdev);
Francois Romieub726e492008-06-28 12:22:59 +02004752
4753 if (cap) {
4754 u16 ctl;
4755
4756 pci_read_config_word(pdev, cap + PCI_EXP_LNKCTL, &ctl);
4757 ctl &= ~PCI_EXP_LNKCTL_CLKREQ_EN;
4758 pci_write_config_word(pdev, cap + PCI_EXP_LNKCTL, ctl);
4759 }
4760}
4761
françois romieue6de30d2011-01-03 15:08:37 +00004762static void rtl_enable_clock_request(struct pci_dev *pdev)
4763{
Jon Masone44daad2011-06-27 07:46:31 +00004764 int cap = pci_pcie_cap(pdev);
françois romieue6de30d2011-01-03 15:08:37 +00004765
4766 if (cap) {
4767 u16 ctl;
4768
4769 pci_read_config_word(pdev, cap + PCI_EXP_LNKCTL, &ctl);
4770 ctl |= PCI_EXP_LNKCTL_CLKREQ_EN;
4771 pci_write_config_word(pdev, cap + PCI_EXP_LNKCTL, ctl);
4772 }
4773}
4774
Francois Romieub726e492008-06-28 12:22:59 +02004775#define R8168_CPCMD_QUIRK_MASK (\
4776 EnableBist | \
4777 Mac_dbgo_oe | \
4778 Force_half_dup | \
4779 Force_rxflow_en | \
4780 Force_txflow_en | \
4781 Cxpl_dbg_sel | \
4782 ASF | \
4783 PktCntrDisable | \
4784 Mac_dbgo_sel)
4785
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004786static void rtl_hw_start_8168bb(struct rtl8169_private *tp)
Francois Romieu219a1e92008-06-28 11:58:39 +02004787{
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004788 void __iomem *ioaddr = tp->mmio_addr;
4789 struct pci_dev *pdev = tp->pci_dev;
4790
Francois Romieub726e492008-06-28 12:22:59 +02004791 RTL_W8(Config3, RTL_R8(Config3) & ~Beacon_en);
4792
4793 RTL_W16(CPlusCmd, RTL_R16(CPlusCmd) & ~R8168_CPCMD_QUIRK_MASK);
4794
Francois Romieu2e68ae42008-06-28 12:00:55 +02004795 rtl_tx_performance_tweak(pdev,
4796 (0x5 << MAX_READ_REQUEST_SHIFT) | PCI_EXP_DEVCTL_NOSNOOP_EN);
Francois Romieu219a1e92008-06-28 11:58:39 +02004797}
4798
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004799static void rtl_hw_start_8168bef(struct rtl8169_private *tp)
Francois Romieu219a1e92008-06-28 11:58:39 +02004800{
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004801 void __iomem *ioaddr = tp->mmio_addr;
4802
4803 rtl_hw_start_8168bb(tp);
Francois Romieub726e492008-06-28 12:22:59 +02004804
françois romieuf0298f82011-01-03 15:07:42 +00004805 RTL_W8(MaxTxPacketSize, TxPacketMax);
Francois Romieub726e492008-06-28 12:22:59 +02004806
4807 RTL_W8(Config4, RTL_R8(Config4) & ~(1 << 0));
Francois Romieu219a1e92008-06-28 11:58:39 +02004808}
4809
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004810static void __rtl_hw_start_8168cp(struct rtl8169_private *tp)
Francois Romieu219a1e92008-06-28 11:58:39 +02004811{
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004812 void __iomem *ioaddr = tp->mmio_addr;
4813 struct pci_dev *pdev = tp->pci_dev;
4814
Francois Romieub726e492008-06-28 12:22:59 +02004815 RTL_W8(Config1, RTL_R8(Config1) | Speed_down);
4816
4817 RTL_W8(Config3, RTL_R8(Config3) & ~Beacon_en);
4818
Francois Romieu219a1e92008-06-28 11:58:39 +02004819 rtl_tx_performance_tweak(pdev, 0x5 << MAX_READ_REQUEST_SHIFT);
Francois Romieub726e492008-06-28 12:22:59 +02004820
4821 rtl_disable_clock_request(pdev);
4822
4823 RTL_W16(CPlusCmd, RTL_R16(CPlusCmd) & ~R8168_CPCMD_QUIRK_MASK);
Francois Romieu219a1e92008-06-28 11:58:39 +02004824}
4825
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004826static void rtl_hw_start_8168cp_1(struct rtl8169_private *tp)
Francois Romieu219a1e92008-06-28 11:58:39 +02004827{
Alexey Dobriyan350f7592009-11-25 15:54:21 -08004828 static const struct ephy_info e_info_8168cp[] = {
Francois Romieub726e492008-06-28 12:22:59 +02004829 { 0x01, 0, 0x0001 },
4830 { 0x02, 0x0800, 0x1000 },
4831 { 0x03, 0, 0x0042 },
4832 { 0x06, 0x0080, 0x0000 },
4833 { 0x07, 0, 0x2000 }
4834 };
4835
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004836 rtl_csi_access_enable_2(tp);
Francois Romieub726e492008-06-28 12:22:59 +02004837
Francois Romieufdf6fc02012-07-06 22:40:38 +02004838 rtl_ephy_init(tp, e_info_8168cp, ARRAY_SIZE(e_info_8168cp));
Francois Romieub726e492008-06-28 12:22:59 +02004839
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004840 __rtl_hw_start_8168cp(tp);
Francois Romieu219a1e92008-06-28 11:58:39 +02004841}
4842
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004843static void rtl_hw_start_8168cp_2(struct rtl8169_private *tp)
Francois Romieuef3386f2008-06-29 12:24:30 +02004844{
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004845 void __iomem *ioaddr = tp->mmio_addr;
4846 struct pci_dev *pdev = tp->pci_dev;
4847
4848 rtl_csi_access_enable_2(tp);
Francois Romieuef3386f2008-06-29 12:24:30 +02004849
4850 RTL_W8(Config3, RTL_R8(Config3) & ~Beacon_en);
4851
4852 rtl_tx_performance_tweak(pdev, 0x5 << MAX_READ_REQUEST_SHIFT);
4853
4854 RTL_W16(CPlusCmd, RTL_R16(CPlusCmd) & ~R8168_CPCMD_QUIRK_MASK);
4855}
4856
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004857static void rtl_hw_start_8168cp_3(struct rtl8169_private *tp)
Francois Romieu7f3e3d32008-07-20 18:53:20 +02004858{
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004859 void __iomem *ioaddr = tp->mmio_addr;
4860 struct pci_dev *pdev = tp->pci_dev;
4861
4862 rtl_csi_access_enable_2(tp);
Francois Romieu7f3e3d32008-07-20 18:53:20 +02004863
4864 RTL_W8(Config3, RTL_R8(Config3) & ~Beacon_en);
4865
4866 /* Magic. */
4867 RTL_W8(DBG_REG, 0x20);
4868
françois romieuf0298f82011-01-03 15:07:42 +00004869 RTL_W8(MaxTxPacketSize, TxPacketMax);
Francois Romieu7f3e3d32008-07-20 18:53:20 +02004870
4871 rtl_tx_performance_tweak(pdev, 0x5 << MAX_READ_REQUEST_SHIFT);
4872
4873 RTL_W16(CPlusCmd, RTL_R16(CPlusCmd) & ~R8168_CPCMD_QUIRK_MASK);
4874}
4875
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004876static void rtl_hw_start_8168c_1(struct rtl8169_private *tp)
Francois Romieu219a1e92008-06-28 11:58:39 +02004877{
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004878 void __iomem *ioaddr = tp->mmio_addr;
Alexey Dobriyan350f7592009-11-25 15:54:21 -08004879 static const struct ephy_info e_info_8168c_1[] = {
Francois Romieub726e492008-06-28 12:22:59 +02004880 { 0x02, 0x0800, 0x1000 },
4881 { 0x03, 0, 0x0002 },
4882 { 0x06, 0x0080, 0x0000 }
4883 };
4884
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004885 rtl_csi_access_enable_2(tp);
Francois Romieub726e492008-06-28 12:22:59 +02004886
4887 RTL_W8(DBG_REG, 0x06 | FIX_NAK_1 | FIX_NAK_2);
4888
Francois Romieufdf6fc02012-07-06 22:40:38 +02004889 rtl_ephy_init(tp, e_info_8168c_1, ARRAY_SIZE(e_info_8168c_1));
Francois Romieub726e492008-06-28 12:22:59 +02004890
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004891 __rtl_hw_start_8168cp(tp);
Francois Romieu219a1e92008-06-28 11:58:39 +02004892}
4893
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004894static void rtl_hw_start_8168c_2(struct rtl8169_private *tp)
Francois Romieu219a1e92008-06-28 11:58:39 +02004895{
Alexey Dobriyan350f7592009-11-25 15:54:21 -08004896 static const struct ephy_info e_info_8168c_2[] = {
Francois Romieub726e492008-06-28 12:22:59 +02004897 { 0x01, 0, 0x0001 },
4898 { 0x03, 0x0400, 0x0220 }
4899 };
4900
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004901 rtl_csi_access_enable_2(tp);
Francois Romieub726e492008-06-28 12:22:59 +02004902
Francois Romieufdf6fc02012-07-06 22:40:38 +02004903 rtl_ephy_init(tp, e_info_8168c_2, ARRAY_SIZE(e_info_8168c_2));
Francois Romieub726e492008-06-28 12:22:59 +02004904
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004905 __rtl_hw_start_8168cp(tp);
Francois Romieu219a1e92008-06-28 11:58:39 +02004906}
4907
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004908static void rtl_hw_start_8168c_3(struct rtl8169_private *tp)
Francois Romieu197ff762008-06-28 13:16:02 +02004909{
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004910 rtl_hw_start_8168c_2(tp);
Francois Romieu197ff762008-06-28 13:16:02 +02004911}
4912
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004913static void rtl_hw_start_8168c_4(struct rtl8169_private *tp)
Francois Romieu6fb07052008-06-29 11:54:28 +02004914{
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004915 rtl_csi_access_enable_2(tp);
Francois Romieu6fb07052008-06-29 11:54:28 +02004916
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004917 __rtl_hw_start_8168cp(tp);
Francois Romieu6fb07052008-06-29 11:54:28 +02004918}
4919
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004920static void rtl_hw_start_8168d(struct rtl8169_private *tp)
Francois Romieu5b538df2008-07-20 16:22:45 +02004921{
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004922 void __iomem *ioaddr = tp->mmio_addr;
4923 struct pci_dev *pdev = tp->pci_dev;
4924
4925 rtl_csi_access_enable_2(tp);
Francois Romieu5b538df2008-07-20 16:22:45 +02004926
4927 rtl_disable_clock_request(pdev);
4928
françois romieuf0298f82011-01-03 15:07:42 +00004929 RTL_W8(MaxTxPacketSize, TxPacketMax);
Francois Romieu5b538df2008-07-20 16:22:45 +02004930
4931 rtl_tx_performance_tweak(pdev, 0x5 << MAX_READ_REQUEST_SHIFT);
4932
4933 RTL_W16(CPlusCmd, RTL_R16(CPlusCmd) & ~R8168_CPCMD_QUIRK_MASK);
4934}
4935
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004936static void rtl_hw_start_8168dp(struct rtl8169_private *tp)
hayeswang4804b3b2011-03-21 01:50:29 +00004937{
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004938 void __iomem *ioaddr = tp->mmio_addr;
4939 struct pci_dev *pdev = tp->pci_dev;
4940
4941 rtl_csi_access_enable_1(tp);
hayeswang4804b3b2011-03-21 01:50:29 +00004942
4943 rtl_tx_performance_tweak(pdev, 0x5 << MAX_READ_REQUEST_SHIFT);
4944
4945 RTL_W8(MaxTxPacketSize, TxPacketMax);
4946
4947 rtl_disable_clock_request(pdev);
4948}
4949
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004950static void rtl_hw_start_8168d_4(struct rtl8169_private *tp)
françois romieue6de30d2011-01-03 15:08:37 +00004951{
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004952 void __iomem *ioaddr = tp->mmio_addr;
4953 struct pci_dev *pdev = tp->pci_dev;
françois romieue6de30d2011-01-03 15:08:37 +00004954 static const struct ephy_info e_info_8168d_4[] = {
4955 { 0x0b, ~0, 0x48 },
4956 { 0x19, 0x20, 0x50 },
4957 { 0x0c, ~0, 0x20 }
4958 };
4959 int i;
4960
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004961 rtl_csi_access_enable_1(tp);
françois romieue6de30d2011-01-03 15:08:37 +00004962
4963 rtl_tx_performance_tweak(pdev, 0x5 << MAX_READ_REQUEST_SHIFT);
4964
4965 RTL_W8(MaxTxPacketSize, TxPacketMax);
4966
4967 for (i = 0; i < ARRAY_SIZE(e_info_8168d_4); i++) {
4968 const struct ephy_info *e = e_info_8168d_4 + i;
4969 u16 w;
4970
Francois Romieufdf6fc02012-07-06 22:40:38 +02004971 w = rtl_ephy_read(tp, e->offset);
4972 rtl_ephy_write(tp, 0x03, (w & e->mask) | e->bits);
françois romieue6de30d2011-01-03 15:08:37 +00004973 }
4974
4975 rtl_enable_clock_request(pdev);
4976}
4977
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004978static void rtl_hw_start_8168e_1(struct rtl8169_private *tp)
hayeswang01dc7fe2011-03-21 01:50:28 +00004979{
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004980 void __iomem *ioaddr = tp->mmio_addr;
4981 struct pci_dev *pdev = tp->pci_dev;
Hayes Wang70090422011-07-06 15:58:06 +08004982 static const struct ephy_info e_info_8168e_1[] = {
hayeswang01dc7fe2011-03-21 01:50:28 +00004983 { 0x00, 0x0200, 0x0100 },
4984 { 0x00, 0x0000, 0x0004 },
4985 { 0x06, 0x0002, 0x0001 },
4986 { 0x06, 0x0000, 0x0030 },
4987 { 0x07, 0x0000, 0x2000 },
4988 { 0x00, 0x0000, 0x0020 },
4989 { 0x03, 0x5800, 0x2000 },
4990 { 0x03, 0x0000, 0x0001 },
4991 { 0x01, 0x0800, 0x1000 },
4992 { 0x07, 0x0000, 0x4000 },
4993 { 0x1e, 0x0000, 0x2000 },
4994 { 0x19, 0xffff, 0xfe6c },
4995 { 0x0a, 0x0000, 0x0040 }
4996 };
4997
Hayes Wangbeb1fe12012-03-30 14:33:01 +08004998 rtl_csi_access_enable_2(tp);
hayeswang01dc7fe2011-03-21 01:50:28 +00004999
Francois Romieufdf6fc02012-07-06 22:40:38 +02005000 rtl_ephy_init(tp, e_info_8168e_1, ARRAY_SIZE(e_info_8168e_1));
hayeswang01dc7fe2011-03-21 01:50:28 +00005001
5002 rtl_tx_performance_tweak(pdev, 0x5 << MAX_READ_REQUEST_SHIFT);
5003
5004 RTL_W8(MaxTxPacketSize, TxPacketMax);
5005
5006 rtl_disable_clock_request(pdev);
5007
5008 /* Reset tx FIFO pointer */
Francois Romieucecb5fd2011-04-01 10:21:07 +02005009 RTL_W32(MISC, RTL_R32(MISC) | TXPLA_RST);
5010 RTL_W32(MISC, RTL_R32(MISC) & ~TXPLA_RST);
hayeswang01dc7fe2011-03-21 01:50:28 +00005011
Francois Romieucecb5fd2011-04-01 10:21:07 +02005012 RTL_W8(Config5, RTL_R8(Config5) & ~Spi_en);
hayeswang01dc7fe2011-03-21 01:50:28 +00005013}
5014
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005015static void rtl_hw_start_8168e_2(struct rtl8169_private *tp)
Hayes Wang70090422011-07-06 15:58:06 +08005016{
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005017 void __iomem *ioaddr = tp->mmio_addr;
5018 struct pci_dev *pdev = tp->pci_dev;
Hayes Wang70090422011-07-06 15:58:06 +08005019 static const struct ephy_info e_info_8168e_2[] = {
5020 { 0x09, 0x0000, 0x0080 },
5021 { 0x19, 0x0000, 0x0224 }
5022 };
5023
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005024 rtl_csi_access_enable_1(tp);
Hayes Wang70090422011-07-06 15:58:06 +08005025
Francois Romieufdf6fc02012-07-06 22:40:38 +02005026 rtl_ephy_init(tp, e_info_8168e_2, ARRAY_SIZE(e_info_8168e_2));
Hayes Wang70090422011-07-06 15:58:06 +08005027
5028 rtl_tx_performance_tweak(pdev, 0x5 << MAX_READ_REQUEST_SHIFT);
5029
Francois Romieufdf6fc02012-07-06 22:40:38 +02005030 rtl_eri_write(tp, 0xc0, ERIAR_MASK_0011, 0x0000, ERIAR_EXGMAC);
5031 rtl_eri_write(tp, 0xb8, ERIAR_MASK_0011, 0x0000, ERIAR_EXGMAC);
5032 rtl_eri_write(tp, 0xc8, ERIAR_MASK_1111, 0x00100002, ERIAR_EXGMAC);
5033 rtl_eri_write(tp, 0xe8, ERIAR_MASK_1111, 0x00100006, ERIAR_EXGMAC);
5034 rtl_eri_write(tp, 0xcc, ERIAR_MASK_1111, 0x00000050, ERIAR_EXGMAC);
5035 rtl_eri_write(tp, 0xd0, ERIAR_MASK_1111, 0x07ff0060, ERIAR_EXGMAC);
5036 rtl_w1w0_eri(tp, 0x1b0, ERIAR_MASK_0001, 0x10, 0x00, ERIAR_EXGMAC);
5037 rtl_w1w0_eri(tp, 0x0d4, ERIAR_MASK_0011, 0x0c00, 0xff00, ERIAR_EXGMAC);
Hayes Wang70090422011-07-06 15:58:06 +08005038
Hayes Wang3090bd92011-09-06 16:55:15 +08005039 RTL_W8(MaxTxPacketSize, EarlySize);
Hayes Wang70090422011-07-06 15:58:06 +08005040
5041 rtl_disable_clock_request(pdev);
5042
5043 RTL_W32(TxConfig, RTL_R32(TxConfig) | TXCFG_AUTO_FIFO);
5044 RTL_W8(MCU, RTL_R8(MCU) & ~NOW_IS_OOB);
5045
5046 /* Adjust EEE LED frequency */
5047 RTL_W8(EEE_LED, RTL_R8(EEE_LED) & ~0x07);
5048
5049 RTL_W8(DLLPR, RTL_R8(DLLPR) | PFM_EN);
5050 RTL_W32(MISC, RTL_R32(MISC) | PWM_EN);
5051 RTL_W8(Config5, RTL_R8(Config5) & ~Spi_en);
5052}
5053
Hayes Wang5f886e02012-03-30 14:33:03 +08005054static void rtl_hw_start_8168f(struct rtl8169_private *tp)
Hayes Wangc2218922011-09-06 16:55:18 +08005055{
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005056 void __iomem *ioaddr = tp->mmio_addr;
5057 struct pci_dev *pdev = tp->pci_dev;
Hayes Wangc2218922011-09-06 16:55:18 +08005058
Hayes Wang5f886e02012-03-30 14:33:03 +08005059 rtl_csi_access_enable_2(tp);
Hayes Wangc2218922011-09-06 16:55:18 +08005060
5061 rtl_tx_performance_tweak(pdev, 0x5 << MAX_READ_REQUEST_SHIFT);
5062
Francois Romieufdf6fc02012-07-06 22:40:38 +02005063 rtl_eri_write(tp, 0xc0, ERIAR_MASK_0011, 0x0000, ERIAR_EXGMAC);
5064 rtl_eri_write(tp, 0xb8, ERIAR_MASK_0011, 0x0000, ERIAR_EXGMAC);
5065 rtl_eri_write(tp, 0xc8, ERIAR_MASK_1111, 0x00100002, ERIAR_EXGMAC);
5066 rtl_eri_write(tp, 0xe8, ERIAR_MASK_1111, 0x00100006, ERIAR_EXGMAC);
5067 rtl_w1w0_eri(tp, 0xdc, ERIAR_MASK_0001, 0x00, 0x01, ERIAR_EXGMAC);
5068 rtl_w1w0_eri(tp, 0xdc, ERIAR_MASK_0001, 0x01, 0x00, ERIAR_EXGMAC);
5069 rtl_w1w0_eri(tp, 0x1b0, ERIAR_MASK_0001, 0x10, 0x00, ERIAR_EXGMAC);
5070 rtl_w1w0_eri(tp, 0x1d0, ERIAR_MASK_0001, 0x10, 0x00, ERIAR_EXGMAC);
5071 rtl_eri_write(tp, 0xcc, ERIAR_MASK_1111, 0x00000050, ERIAR_EXGMAC);
5072 rtl_eri_write(tp, 0xd0, ERIAR_MASK_1111, 0x00000060, ERIAR_EXGMAC);
Hayes Wangc2218922011-09-06 16:55:18 +08005073
5074 RTL_W8(MaxTxPacketSize, EarlySize);
5075
5076 rtl_disable_clock_request(pdev);
5077
5078 RTL_W32(TxConfig, RTL_R32(TxConfig) | TXCFG_AUTO_FIFO);
5079 RTL_W8(MCU, RTL_R8(MCU) & ~NOW_IS_OOB);
Hayes Wangc2218922011-09-06 16:55:18 +08005080 RTL_W8(DLLPR, RTL_R8(DLLPR) | PFM_EN);
5081 RTL_W32(MISC, RTL_R32(MISC) | PWM_EN);
5082 RTL_W8(Config5, RTL_R8(Config5) & ~Spi_en);
5083}
5084
Hayes Wang5f886e02012-03-30 14:33:03 +08005085static void rtl_hw_start_8168f_1(struct rtl8169_private *tp)
5086{
5087 void __iomem *ioaddr = tp->mmio_addr;
5088 static const struct ephy_info e_info_8168f_1[] = {
5089 { 0x06, 0x00c0, 0x0020 },
5090 { 0x08, 0x0001, 0x0002 },
5091 { 0x09, 0x0000, 0x0080 },
5092 { 0x19, 0x0000, 0x0224 }
5093 };
5094
5095 rtl_hw_start_8168f(tp);
5096
Francois Romieufdf6fc02012-07-06 22:40:38 +02005097 rtl_ephy_init(tp, e_info_8168f_1, ARRAY_SIZE(e_info_8168f_1));
Hayes Wang5f886e02012-03-30 14:33:03 +08005098
Francois Romieufdf6fc02012-07-06 22:40:38 +02005099 rtl_w1w0_eri(tp, 0x0d4, ERIAR_MASK_0011, 0x0c00, 0xff00, ERIAR_EXGMAC);
Hayes Wang5f886e02012-03-30 14:33:03 +08005100
5101 /* Adjust EEE LED frequency */
5102 RTL_W8(EEE_LED, RTL_R8(EEE_LED) & ~0x07);
5103}
5104
Hayes Wangb3d7b2f2012-03-30 14:48:06 +08005105static void rtl_hw_start_8411(struct rtl8169_private *tp)
5106{
Hayes Wangb3d7b2f2012-03-30 14:48:06 +08005107 static const struct ephy_info e_info_8168f_1[] = {
5108 { 0x06, 0x00c0, 0x0020 },
5109 { 0x0f, 0xffff, 0x5200 },
5110 { 0x1e, 0x0000, 0x4000 },
5111 { 0x19, 0x0000, 0x0224 }
5112 };
5113
5114 rtl_hw_start_8168f(tp);
5115
Francois Romieufdf6fc02012-07-06 22:40:38 +02005116 rtl_ephy_init(tp, e_info_8168f_1, ARRAY_SIZE(e_info_8168f_1));
Hayes Wangb3d7b2f2012-03-30 14:48:06 +08005117
Francois Romieufdf6fc02012-07-06 22:40:38 +02005118 rtl_w1w0_eri(tp, 0x0d4, ERIAR_MASK_0011, 0x0c00, 0x0000, ERIAR_EXGMAC);
Hayes Wangb3d7b2f2012-03-30 14:48:06 +08005119}
5120
Hayes Wangc5583862012-07-02 17:23:22 +08005121static void rtl_hw_start_8168g_1(struct rtl8169_private *tp)
5122{
5123 void __iomem *ioaddr = tp->mmio_addr;
5124 struct pci_dev *pdev = tp->pci_dev;
5125
5126 rtl_eri_write(tp, 0xc8, ERIAR_MASK_0101, 0x080002, ERIAR_EXGMAC);
5127 rtl_eri_write(tp, 0xcc, ERIAR_MASK_0001, 0x38, ERIAR_EXGMAC);
5128 rtl_eri_write(tp, 0xd0, ERIAR_MASK_0001, 0x48, ERIAR_EXGMAC);
5129 rtl_eri_write(tp, 0xe8, ERIAR_MASK_1111, 0x00100006, ERIAR_EXGMAC);
5130
5131 rtl_csi_access_enable_1(tp);
5132
5133 rtl_tx_performance_tweak(pdev, 0x5 << MAX_READ_REQUEST_SHIFT);
5134
5135 rtl_w1w0_eri(tp, 0xdc, ERIAR_MASK_0001, 0x00, 0x01, ERIAR_EXGMAC);
5136 rtl_w1w0_eri(tp, 0xdc, ERIAR_MASK_0001, 0x01, 0x00, ERIAR_EXGMAC);
5137
5138 RTL_W8(ChipCmd, CmdTxEnb | CmdRxEnb);
5139 RTL_W32(MISC, RTL_R32(MISC) & ~RXDV_GATED_EN);
5140 RTL_W8(MaxTxPacketSize, EarlySize);
5141
5142 rtl_eri_write(tp, 0xc0, ERIAR_MASK_0011, 0x0000, ERIAR_EXGMAC);
5143 rtl_eri_write(tp, 0xb8, ERIAR_MASK_0011, 0x0000, ERIAR_EXGMAC);
5144
5145 /* Adjust EEE LED frequency */
5146 RTL_W8(EEE_LED, RTL_R8(EEE_LED) & ~0x07);
5147
5148 rtl_w1w0_eri(tp, 0x2fc, ERIAR_MASK_0001, 0x01, 0x02, ERIAR_EXGMAC);
5149}
5150
Francois Romieu07ce4062007-02-23 23:36:39 +01005151static void rtl_hw_start_8168(struct net_device *dev)
5152{
Francois Romieu2dd99532007-06-11 23:22:52 +02005153 struct rtl8169_private *tp = netdev_priv(dev);
5154 void __iomem *ioaddr = tp->mmio_addr;
5155
5156 RTL_W8(Cfg9346, Cfg9346_Unlock);
5157
françois romieuf0298f82011-01-03 15:07:42 +00005158 RTL_W8(MaxTxPacketSize, TxPacketMax);
Francois Romieu2dd99532007-06-11 23:22:52 +02005159
Eric Dumazet6f0333b2010-10-11 11:17:47 +00005160 rtl_set_rx_max_size(ioaddr, rx_buf_sz);
Francois Romieu2dd99532007-06-11 23:22:52 +02005161
Francois Romieu0e485152007-02-20 00:00:26 +01005162 tp->cp_cmd |= RTL_R16(CPlusCmd) | PktCntrDisable | INTT_1;
Francois Romieu2dd99532007-06-11 23:22:52 +02005163
5164 RTL_W16(CPlusCmd, tp->cp_cmd);
5165
Francois Romieu0e485152007-02-20 00:00:26 +01005166 RTL_W16(IntrMitigate, 0x5151);
5167
5168 /* Work around for RxFIFO overflow. */
françois romieu811fd302011-12-04 20:30:45 +00005169 if (tp->mac_version == RTL_GIGA_MAC_VER_11) {
Francois Romieuda78dbf2012-01-26 14:18:23 +01005170 tp->event_slow |= RxFIFOOver | PCSTimeout;
5171 tp->event_slow &= ~RxOverflow;
Francois Romieu0e485152007-02-20 00:00:26 +01005172 }
Francois Romieu2dd99532007-06-11 23:22:52 +02005173
5174 rtl_set_rx_tx_desc_registers(tp, ioaddr);
5175
Francois Romieub8363902008-06-01 12:31:57 +02005176 rtl_set_rx_mode(dev);
5177
5178 RTL_W32(TxConfig, (TX_DMA_BURST << TxDMAShift) |
5179 (InterFrameGap << TxInterFrameGapShift));
Francois Romieu2dd99532007-06-11 23:22:52 +02005180
5181 RTL_R8(IntrMask);
5182
Francois Romieu219a1e92008-06-28 11:58:39 +02005183 switch (tp->mac_version) {
5184 case RTL_GIGA_MAC_VER_11:
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005185 rtl_hw_start_8168bb(tp);
hayeswang4804b3b2011-03-21 01:50:29 +00005186 break;
Francois Romieu219a1e92008-06-28 11:58:39 +02005187
5188 case RTL_GIGA_MAC_VER_12:
5189 case RTL_GIGA_MAC_VER_17:
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005190 rtl_hw_start_8168bef(tp);
hayeswang4804b3b2011-03-21 01:50:29 +00005191 break;
Francois Romieu219a1e92008-06-28 11:58:39 +02005192
5193 case RTL_GIGA_MAC_VER_18:
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005194 rtl_hw_start_8168cp_1(tp);
hayeswang4804b3b2011-03-21 01:50:29 +00005195 break;
Francois Romieu219a1e92008-06-28 11:58:39 +02005196
5197 case RTL_GIGA_MAC_VER_19:
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005198 rtl_hw_start_8168c_1(tp);
hayeswang4804b3b2011-03-21 01:50:29 +00005199 break;
Francois Romieu219a1e92008-06-28 11:58:39 +02005200
5201 case RTL_GIGA_MAC_VER_20:
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005202 rtl_hw_start_8168c_2(tp);
hayeswang4804b3b2011-03-21 01:50:29 +00005203 break;
Francois Romieu219a1e92008-06-28 11:58:39 +02005204
Francois Romieu197ff762008-06-28 13:16:02 +02005205 case RTL_GIGA_MAC_VER_21:
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005206 rtl_hw_start_8168c_3(tp);
hayeswang4804b3b2011-03-21 01:50:29 +00005207 break;
Francois Romieu197ff762008-06-28 13:16:02 +02005208
Francois Romieu6fb07052008-06-29 11:54:28 +02005209 case RTL_GIGA_MAC_VER_22:
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005210 rtl_hw_start_8168c_4(tp);
hayeswang4804b3b2011-03-21 01:50:29 +00005211 break;
Francois Romieu6fb07052008-06-29 11:54:28 +02005212
Francois Romieuef3386f2008-06-29 12:24:30 +02005213 case RTL_GIGA_MAC_VER_23:
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005214 rtl_hw_start_8168cp_2(tp);
hayeswang4804b3b2011-03-21 01:50:29 +00005215 break;
Francois Romieuef3386f2008-06-29 12:24:30 +02005216
Francois Romieu7f3e3d32008-07-20 18:53:20 +02005217 case RTL_GIGA_MAC_VER_24:
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005218 rtl_hw_start_8168cp_3(tp);
hayeswang4804b3b2011-03-21 01:50:29 +00005219 break;
Francois Romieu7f3e3d32008-07-20 18:53:20 +02005220
Francois Romieu5b538df2008-07-20 16:22:45 +02005221 case RTL_GIGA_MAC_VER_25:
françois romieudaf9df62009-10-07 12:44:20 +00005222 case RTL_GIGA_MAC_VER_26:
5223 case RTL_GIGA_MAC_VER_27:
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005224 rtl_hw_start_8168d(tp);
hayeswang4804b3b2011-03-21 01:50:29 +00005225 break;
Francois Romieu5b538df2008-07-20 16:22:45 +02005226
françois romieue6de30d2011-01-03 15:08:37 +00005227 case RTL_GIGA_MAC_VER_28:
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005228 rtl_hw_start_8168d_4(tp);
hayeswang4804b3b2011-03-21 01:50:29 +00005229 break;
Francois Romieucecb5fd2011-04-01 10:21:07 +02005230
hayeswang4804b3b2011-03-21 01:50:29 +00005231 case RTL_GIGA_MAC_VER_31:
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005232 rtl_hw_start_8168dp(tp);
hayeswang4804b3b2011-03-21 01:50:29 +00005233 break;
5234
hayeswang01dc7fe2011-03-21 01:50:28 +00005235 case RTL_GIGA_MAC_VER_32:
5236 case RTL_GIGA_MAC_VER_33:
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005237 rtl_hw_start_8168e_1(tp);
Hayes Wang70090422011-07-06 15:58:06 +08005238 break;
5239 case RTL_GIGA_MAC_VER_34:
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005240 rtl_hw_start_8168e_2(tp);
hayeswang01dc7fe2011-03-21 01:50:28 +00005241 break;
françois romieue6de30d2011-01-03 15:08:37 +00005242
Hayes Wangc2218922011-09-06 16:55:18 +08005243 case RTL_GIGA_MAC_VER_35:
5244 case RTL_GIGA_MAC_VER_36:
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005245 rtl_hw_start_8168f_1(tp);
Hayes Wangc2218922011-09-06 16:55:18 +08005246 break;
5247
Hayes Wangb3d7b2f2012-03-30 14:48:06 +08005248 case RTL_GIGA_MAC_VER_38:
5249 rtl_hw_start_8411(tp);
5250 break;
5251
Hayes Wangc5583862012-07-02 17:23:22 +08005252 case RTL_GIGA_MAC_VER_40:
5253 case RTL_GIGA_MAC_VER_41:
5254 rtl_hw_start_8168g_1(tp);
5255 break;
5256
Francois Romieu219a1e92008-06-28 11:58:39 +02005257 default:
5258 printk(KERN_ERR PFX "%s: unknown chipset (mac_version = %d).\n",
5259 dev->name, tp->mac_version);
hayeswang4804b3b2011-03-21 01:50:29 +00005260 break;
Francois Romieu219a1e92008-06-28 11:58:39 +02005261 }
Francois Romieu2dd99532007-06-11 23:22:52 +02005262
Francois Romieu0e485152007-02-20 00:00:26 +01005263 RTL_W8(ChipCmd, CmdTxEnb | CmdRxEnb);
5264
Francois Romieub8363902008-06-01 12:31:57 +02005265 RTL_W8(Cfg9346, Cfg9346_Lock);
5266
Francois Romieu2dd99532007-06-11 23:22:52 +02005267 RTL_W16(MultiIntr, RTL_R16(MultiIntr) & 0xF000);
Francois Romieu07ce4062007-02-23 23:36:39 +01005268}
Linus Torvalds1da177e2005-04-16 15:20:36 -07005269
Francois Romieu2857ffb2008-08-02 21:08:49 +02005270#define R810X_CPCMD_QUIRK_MASK (\
5271 EnableBist | \
5272 Mac_dbgo_oe | \
5273 Force_half_dup | \
françois romieu5edcc532009-08-10 19:41:52 +00005274 Force_rxflow_en | \
Francois Romieu2857ffb2008-08-02 21:08:49 +02005275 Force_txflow_en | \
5276 Cxpl_dbg_sel | \
5277 ASF | \
5278 PktCntrDisable | \
Hayes Wangd24e9aa2011-02-22 17:26:19 +08005279 Mac_dbgo_sel)
Francois Romieu2857ffb2008-08-02 21:08:49 +02005280
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005281static void rtl_hw_start_8102e_1(struct rtl8169_private *tp)
Francois Romieu2857ffb2008-08-02 21:08:49 +02005282{
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005283 void __iomem *ioaddr = tp->mmio_addr;
5284 struct pci_dev *pdev = tp->pci_dev;
Alexey Dobriyan350f7592009-11-25 15:54:21 -08005285 static const struct ephy_info e_info_8102e_1[] = {
Francois Romieu2857ffb2008-08-02 21:08:49 +02005286 { 0x01, 0, 0x6e65 },
5287 { 0x02, 0, 0x091f },
5288 { 0x03, 0, 0xc2f9 },
5289 { 0x06, 0, 0xafb5 },
5290 { 0x07, 0, 0x0e00 },
5291 { 0x19, 0, 0xec80 },
5292 { 0x01, 0, 0x2e65 },
5293 { 0x01, 0, 0x6e65 }
5294 };
5295 u8 cfg1;
5296
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005297 rtl_csi_access_enable_2(tp);
Francois Romieu2857ffb2008-08-02 21:08:49 +02005298
5299 RTL_W8(DBG_REG, FIX_NAK_1);
5300
5301 rtl_tx_performance_tweak(pdev, 0x5 << MAX_READ_REQUEST_SHIFT);
5302
5303 RTL_W8(Config1,
5304 LEDS1 | LEDS0 | Speed_down | MEMMAP | IOMAP | VPD | PMEnable);
5305 RTL_W8(Config3, RTL_R8(Config3) & ~Beacon_en);
5306
5307 cfg1 = RTL_R8(Config1);
5308 if ((cfg1 & LEDS0) && (cfg1 & LEDS1))
5309 RTL_W8(Config1, cfg1 & ~LEDS0);
5310
Francois Romieufdf6fc02012-07-06 22:40:38 +02005311 rtl_ephy_init(tp, e_info_8102e_1, ARRAY_SIZE(e_info_8102e_1));
Francois Romieu2857ffb2008-08-02 21:08:49 +02005312}
5313
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005314static void rtl_hw_start_8102e_2(struct rtl8169_private *tp)
Francois Romieu2857ffb2008-08-02 21:08:49 +02005315{
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005316 void __iomem *ioaddr = tp->mmio_addr;
5317 struct pci_dev *pdev = tp->pci_dev;
5318
5319 rtl_csi_access_enable_2(tp);
Francois Romieu2857ffb2008-08-02 21:08:49 +02005320
5321 rtl_tx_performance_tweak(pdev, 0x5 << MAX_READ_REQUEST_SHIFT);
5322
5323 RTL_W8(Config1, MEMMAP | IOMAP | VPD | PMEnable);
5324 RTL_W8(Config3, RTL_R8(Config3) & ~Beacon_en);
Francois Romieu2857ffb2008-08-02 21:08:49 +02005325}
5326
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005327static void rtl_hw_start_8102e_3(struct rtl8169_private *tp)
Francois Romieu2857ffb2008-08-02 21:08:49 +02005328{
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005329 rtl_hw_start_8102e_2(tp);
Francois Romieu2857ffb2008-08-02 21:08:49 +02005330
Francois Romieufdf6fc02012-07-06 22:40:38 +02005331 rtl_ephy_write(tp, 0x03, 0xc2f9);
Francois Romieu2857ffb2008-08-02 21:08:49 +02005332}
5333
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005334static void rtl_hw_start_8105e_1(struct rtl8169_private *tp)
Hayes Wang5a5e4442011-02-22 17:26:21 +08005335{
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005336 void __iomem *ioaddr = tp->mmio_addr;
Hayes Wang5a5e4442011-02-22 17:26:21 +08005337 static const struct ephy_info e_info_8105e_1[] = {
5338 { 0x07, 0, 0x4000 },
5339 { 0x19, 0, 0x0200 },
5340 { 0x19, 0, 0x0020 },
5341 { 0x1e, 0, 0x2000 },
5342 { 0x03, 0, 0x0001 },
5343 { 0x19, 0, 0x0100 },
5344 { 0x19, 0, 0x0004 },
5345 { 0x0a, 0, 0x0020 }
5346 };
5347
Francois Romieucecb5fd2011-04-01 10:21:07 +02005348 /* Force LAN exit from ASPM if Rx/Tx are not idle */
Hayes Wang5a5e4442011-02-22 17:26:21 +08005349 RTL_W32(FuncEvent, RTL_R32(FuncEvent) | 0x002800);
5350
Francois Romieucecb5fd2011-04-01 10:21:07 +02005351 /* Disable Early Tally Counter */
Hayes Wang5a5e4442011-02-22 17:26:21 +08005352 RTL_W32(FuncEvent, RTL_R32(FuncEvent) & ~0x010000);
5353
5354 RTL_W8(MCU, RTL_R8(MCU) | EN_NDP | EN_OOB_RESET);
Hayes Wang4f6b00e52011-07-06 15:58:02 +08005355 RTL_W8(DLLPR, RTL_R8(DLLPR) | PFM_EN);
Hayes Wang5a5e4442011-02-22 17:26:21 +08005356
Francois Romieufdf6fc02012-07-06 22:40:38 +02005357 rtl_ephy_init(tp, e_info_8105e_1, ARRAY_SIZE(e_info_8105e_1));
Hayes Wang5a5e4442011-02-22 17:26:21 +08005358}
5359
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005360static void rtl_hw_start_8105e_2(struct rtl8169_private *tp)
Hayes Wang5a5e4442011-02-22 17:26:21 +08005361{
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005362 rtl_hw_start_8105e_1(tp);
Francois Romieufdf6fc02012-07-06 22:40:38 +02005363 rtl_ephy_write(tp, 0x1e, rtl_ephy_read(tp, 0x1e) | 0x8000);
Hayes Wang5a5e4442011-02-22 17:26:21 +08005364}
5365
Hayes Wang7e18dca2012-03-30 14:33:02 +08005366static void rtl_hw_start_8402(struct rtl8169_private *tp)
5367{
5368 void __iomem *ioaddr = tp->mmio_addr;
5369 static const struct ephy_info e_info_8402[] = {
5370 { 0x19, 0xffff, 0xff64 },
5371 { 0x1e, 0, 0x4000 }
5372 };
5373
5374 rtl_csi_access_enable_2(tp);
5375
5376 /* Force LAN exit from ASPM if Rx/Tx are not idle */
5377 RTL_W32(FuncEvent, RTL_R32(FuncEvent) | 0x002800);
5378
5379 RTL_W32(TxConfig, RTL_R32(TxConfig) | TXCFG_AUTO_FIFO);
5380 RTL_W8(MCU, RTL_R8(MCU) & ~NOW_IS_OOB);
5381
Francois Romieufdf6fc02012-07-06 22:40:38 +02005382 rtl_ephy_init(tp, e_info_8402, ARRAY_SIZE(e_info_8402));
Hayes Wang7e18dca2012-03-30 14:33:02 +08005383
5384 rtl_tx_performance_tweak(tp->pci_dev, 0x5 << MAX_READ_REQUEST_SHIFT);
5385
Francois Romieufdf6fc02012-07-06 22:40:38 +02005386 rtl_eri_write(tp, 0xc8, ERIAR_MASK_1111, 0x00000002, ERIAR_EXGMAC);
5387 rtl_eri_write(tp, 0xe8, ERIAR_MASK_1111, 0x00000006, ERIAR_EXGMAC);
5388 rtl_w1w0_eri(tp, 0xdc, ERIAR_MASK_0001, 0x00, 0x01, ERIAR_EXGMAC);
5389 rtl_w1w0_eri(tp, 0xdc, ERIAR_MASK_0001, 0x01, 0x00, ERIAR_EXGMAC);
5390 rtl_eri_write(tp, 0xc0, ERIAR_MASK_0011, 0x0000, ERIAR_EXGMAC);
5391 rtl_eri_write(tp, 0xb8, ERIAR_MASK_0011, 0x0000, ERIAR_EXGMAC);
5392 rtl_w1w0_eri(tp, 0x0d4, ERIAR_MASK_0011, 0x0e00, 0xff00, ERIAR_EXGMAC);
Hayes Wang7e18dca2012-03-30 14:33:02 +08005393}
5394
Hayes Wang5598bfe2012-07-02 17:23:21 +08005395static void rtl_hw_start_8106(struct rtl8169_private *tp)
5396{
5397 void __iomem *ioaddr = tp->mmio_addr;
5398
5399 /* Force LAN exit from ASPM if Rx/Tx are not idle */
5400 RTL_W32(FuncEvent, RTL_R32(FuncEvent) | 0x002800);
5401
5402 RTL_W32(MISC, (RTL_R32(MISC) | DISABLE_LAN_EN) & ~EARLY_TALLY_EN);
5403 RTL_W8(MCU, RTL_R8(MCU) | EN_NDP | EN_OOB_RESET);
5404 RTL_W8(DLLPR, RTL_R8(DLLPR) & ~PFM_EN);
5405}
5406
Francois Romieu07ce4062007-02-23 23:36:39 +01005407static void rtl_hw_start_8101(struct net_device *dev)
5408{
Francois Romieucdf1a602007-06-11 23:29:50 +02005409 struct rtl8169_private *tp = netdev_priv(dev);
5410 void __iomem *ioaddr = tp->mmio_addr;
5411 struct pci_dev *pdev = tp->pci_dev;
5412
Francois Romieuda78dbf2012-01-26 14:18:23 +01005413 if (tp->mac_version >= RTL_GIGA_MAC_VER_30)
5414 tp->event_slow &= ~RxFIFOOver;
françois romieu811fd302011-12-04 20:30:45 +00005415
Francois Romieucecb5fd2011-04-01 10:21:07 +02005416 if (tp->mac_version == RTL_GIGA_MAC_VER_13 ||
5417 tp->mac_version == RTL_GIGA_MAC_VER_16) {
Jon Masone44daad2011-06-27 07:46:31 +00005418 int cap = pci_pcie_cap(pdev);
Francois Romieu9c14cea2008-07-05 00:21:15 +02005419
5420 if (cap) {
5421 pci_write_config_word(pdev, cap + PCI_EXP_DEVCTL,
5422 PCI_EXP_DEVCTL_NOSNOOP_EN);
5423 }
Francois Romieucdf1a602007-06-11 23:29:50 +02005424 }
5425
Hayes Wangd24e9aa2011-02-22 17:26:19 +08005426 RTL_W8(Cfg9346, Cfg9346_Unlock);
5427
Francois Romieu2857ffb2008-08-02 21:08:49 +02005428 switch (tp->mac_version) {
5429 case RTL_GIGA_MAC_VER_07:
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005430 rtl_hw_start_8102e_1(tp);
Francois Romieu2857ffb2008-08-02 21:08:49 +02005431 break;
5432
5433 case RTL_GIGA_MAC_VER_08:
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005434 rtl_hw_start_8102e_3(tp);
Francois Romieu2857ffb2008-08-02 21:08:49 +02005435 break;
5436
5437 case RTL_GIGA_MAC_VER_09:
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005438 rtl_hw_start_8102e_2(tp);
Francois Romieu2857ffb2008-08-02 21:08:49 +02005439 break;
Hayes Wang5a5e4442011-02-22 17:26:21 +08005440
5441 case RTL_GIGA_MAC_VER_29:
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005442 rtl_hw_start_8105e_1(tp);
Hayes Wang5a5e4442011-02-22 17:26:21 +08005443 break;
5444 case RTL_GIGA_MAC_VER_30:
Hayes Wangbeb1fe12012-03-30 14:33:01 +08005445 rtl_hw_start_8105e_2(tp);
Hayes Wang5a5e4442011-02-22 17:26:21 +08005446 break;
Hayes Wang7e18dca2012-03-30 14:33:02 +08005447
5448 case RTL_GIGA_MAC_VER_37:
5449 rtl_hw_start_8402(tp);
5450 break;
Hayes Wang5598bfe2012-07-02 17:23:21 +08005451
5452 case RTL_GIGA_MAC_VER_39:
5453 rtl_hw_start_8106(tp);
5454 break;
Francois Romieucdf1a602007-06-11 23:29:50 +02005455 }
5456
Hayes Wangd24e9aa2011-02-22 17:26:19 +08005457 RTL_W8(Cfg9346, Cfg9346_Lock);
Francois Romieucdf1a602007-06-11 23:29:50 +02005458
françois romieuf0298f82011-01-03 15:07:42 +00005459 RTL_W8(MaxTxPacketSize, TxPacketMax);
Francois Romieucdf1a602007-06-11 23:29:50 +02005460
Eric Dumazet6f0333b2010-10-11 11:17:47 +00005461 rtl_set_rx_max_size(ioaddr, rx_buf_sz);
Francois Romieucdf1a602007-06-11 23:29:50 +02005462
Hayes Wangd24e9aa2011-02-22 17:26:19 +08005463 tp->cp_cmd &= ~R810X_CPCMD_QUIRK_MASK;
Francois Romieucdf1a602007-06-11 23:29:50 +02005464 RTL_W16(CPlusCmd, tp->cp_cmd);
5465
5466 RTL_W16(IntrMitigate, 0x0000);
5467
5468 rtl_set_rx_tx_desc_registers(tp, ioaddr);
5469
5470 RTL_W8(ChipCmd, CmdTxEnb | CmdRxEnb);
5471 rtl_set_rx_tx_config_registers(tp);
5472
Francois Romieucdf1a602007-06-11 23:29:50 +02005473 RTL_R8(IntrMask);
5474
Francois Romieucdf1a602007-06-11 23:29:50 +02005475 rtl_set_rx_mode(dev);
5476
5477 RTL_W16(MultiIntr, RTL_R16(MultiIntr) & 0xf000);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005478}
5479
5480static int rtl8169_change_mtu(struct net_device *dev, int new_mtu)
5481{
Francois Romieud58d46b2011-05-03 16:38:29 +02005482 struct rtl8169_private *tp = netdev_priv(dev);
5483
5484 if (new_mtu < ETH_ZLEN ||
5485 new_mtu > rtl_chip_infos[tp->mac_version].jumbo_max)
Linus Torvalds1da177e2005-04-16 15:20:36 -07005486 return -EINVAL;
5487
Francois Romieud58d46b2011-05-03 16:38:29 +02005488 if (new_mtu > ETH_DATA_LEN)
5489 rtl_hw_jumbo_enable(tp);
5490 else
5491 rtl_hw_jumbo_disable(tp);
5492
Linus Torvalds1da177e2005-04-16 15:20:36 -07005493 dev->mtu = new_mtu;
Michał Mirosław350fb322011-04-08 06:35:56 +00005494 netdev_update_features(dev);
5495
Stanislaw Gruszka323bb682010-10-20 22:25:41 +00005496 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005497}
5498
5499static inline void rtl8169_make_unusable_by_asic(struct RxDesc *desc)
5500{
Al Viro95e09182007-12-22 18:55:39 +00005501 desc->addr = cpu_to_le64(0x0badbadbadbadbadull);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005502 desc->opts1 &= ~cpu_to_le32(DescOwn | RsvdMask);
5503}
5504
Eric Dumazet6f0333b2010-10-11 11:17:47 +00005505static void rtl8169_free_rx_databuff(struct rtl8169_private *tp,
5506 void **data_buff, struct RxDesc *desc)
Linus Torvalds1da177e2005-04-16 15:20:36 -07005507{
Stanislaw Gruszka48addcc2010-10-20 22:25:39 +00005508 dma_unmap_single(&tp->pci_dev->dev, le64_to_cpu(desc->addr), rx_buf_sz,
Stanislaw Gruszka231aee62010-10-20 22:25:38 +00005509 DMA_FROM_DEVICE);
Stanislaw Gruszka48addcc2010-10-20 22:25:39 +00005510
Eric Dumazet6f0333b2010-10-11 11:17:47 +00005511 kfree(*data_buff);
5512 *data_buff = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005513 rtl8169_make_unusable_by_asic(desc);
5514}
5515
5516static inline void rtl8169_mark_to_asic(struct RxDesc *desc, u32 rx_buf_sz)
5517{
5518 u32 eor = le32_to_cpu(desc->opts1) & RingEnd;
5519
5520 desc->opts1 = cpu_to_le32(DescOwn | eor | rx_buf_sz);
5521}
5522
5523static inline void rtl8169_map_to_asic(struct RxDesc *desc, dma_addr_t mapping,
5524 u32 rx_buf_sz)
5525{
5526 desc->addr = cpu_to_le64(mapping);
5527 wmb();
5528 rtl8169_mark_to_asic(desc, rx_buf_sz);
5529}
5530
Eric Dumazet6f0333b2010-10-11 11:17:47 +00005531static inline void *rtl8169_align(void *data)
Linus Torvalds1da177e2005-04-16 15:20:36 -07005532{
Eric Dumazet6f0333b2010-10-11 11:17:47 +00005533 return (void *)ALIGN((long)data, 16);
5534}
5535
Stanislaw Gruszka0ecbe1c2010-10-20 22:25:37 +00005536static struct sk_buff *rtl8169_alloc_rx_data(struct rtl8169_private *tp,
5537 struct RxDesc *desc)
Eric Dumazet6f0333b2010-10-11 11:17:47 +00005538{
5539 void *data;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005540 dma_addr_t mapping;
Stanislaw Gruszka48addcc2010-10-20 22:25:39 +00005541 struct device *d = &tp->pci_dev->dev;
Stanislaw Gruszka0ecbe1c2010-10-20 22:25:37 +00005542 struct net_device *dev = tp->dev;
Eric Dumazet6f0333b2010-10-11 11:17:47 +00005543 int node = dev->dev.parent ? dev_to_node(dev->dev.parent) : -1;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005544
Eric Dumazet6f0333b2010-10-11 11:17:47 +00005545 data = kmalloc_node(rx_buf_sz, GFP_KERNEL, node);
5546 if (!data)
5547 return NULL;
Francois Romieue9f63f32007-02-28 23:16:57 +01005548
Eric Dumazet6f0333b2010-10-11 11:17:47 +00005549 if (rtl8169_align(data) != data) {
5550 kfree(data);
5551 data = kmalloc_node(rx_buf_sz + 15, GFP_KERNEL, node);
5552 if (!data)
5553 return NULL;
5554 }
Stanislaw Gruszka3eafe502010-10-20 22:25:36 +00005555
Stanislaw Gruszka48addcc2010-10-20 22:25:39 +00005556 mapping = dma_map_single(d, rtl8169_align(data), rx_buf_sz,
Stanislaw Gruszka231aee62010-10-20 22:25:38 +00005557 DMA_FROM_DEVICE);
Stanislaw Gruszkad827d862010-10-20 22:25:43 +00005558 if (unlikely(dma_mapping_error(d, mapping))) {
5559 if (net_ratelimit())
5560 netif_err(tp, drv, tp->dev, "Failed to map RX DMA!\n");
Stanislaw Gruszka3eafe502010-10-20 22:25:36 +00005561 goto err_out;
Stanislaw Gruszkad827d862010-10-20 22:25:43 +00005562 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07005563
5564 rtl8169_map_to_asic(desc, mapping, rx_buf_sz);
Eric Dumazet6f0333b2010-10-11 11:17:47 +00005565 return data;
Stanislaw Gruszka3eafe502010-10-20 22:25:36 +00005566
5567err_out:
5568 kfree(data);
5569 return NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005570}
5571
5572static void rtl8169_rx_clear(struct rtl8169_private *tp)
5573{
Francois Romieu07d3f512007-02-21 22:40:46 +01005574 unsigned int i;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005575
5576 for (i = 0; i < NUM_RX_DESC; i++) {
Eric Dumazet6f0333b2010-10-11 11:17:47 +00005577 if (tp->Rx_databuff[i]) {
5578 rtl8169_free_rx_databuff(tp, tp->Rx_databuff + i,
Linus Torvalds1da177e2005-04-16 15:20:36 -07005579 tp->RxDescArray + i);
5580 }
5581 }
5582}
5583
Stanislaw Gruszka0ecbe1c2010-10-20 22:25:37 +00005584static inline void rtl8169_mark_as_last_descriptor(struct RxDesc *desc)
Linus Torvalds1da177e2005-04-16 15:20:36 -07005585{
Stanislaw Gruszka0ecbe1c2010-10-20 22:25:37 +00005586 desc->opts1 |= cpu_to_le32(RingEnd);
5587}
Francois Romieu5b0384f2006-08-16 16:00:01 +02005588
Stanislaw Gruszka0ecbe1c2010-10-20 22:25:37 +00005589static int rtl8169_rx_fill(struct rtl8169_private *tp)
5590{
5591 unsigned int i;
5592
5593 for (i = 0; i < NUM_RX_DESC; i++) {
Eric Dumazet6f0333b2010-10-11 11:17:47 +00005594 void *data;
Francois Romieu4ae47c22007-06-16 23:28:45 +02005595
Eric Dumazet6f0333b2010-10-11 11:17:47 +00005596 if (tp->Rx_databuff[i])
Linus Torvalds1da177e2005-04-16 15:20:36 -07005597 continue;
Francois Romieubcf0bf92006-07-26 23:14:13 +02005598
Stanislaw Gruszka0ecbe1c2010-10-20 22:25:37 +00005599 data = rtl8169_alloc_rx_data(tp, tp->RxDescArray + i);
Eric Dumazet6f0333b2010-10-11 11:17:47 +00005600 if (!data) {
5601 rtl8169_make_unusable_by_asic(tp->RxDescArray + i);
Stanislaw Gruszka0ecbe1c2010-10-20 22:25:37 +00005602 goto err_out;
Eric Dumazet6f0333b2010-10-11 11:17:47 +00005603 }
5604 tp->Rx_databuff[i] = data;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005605 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07005606
Stanislaw Gruszka0ecbe1c2010-10-20 22:25:37 +00005607 rtl8169_mark_as_last_descriptor(tp->RxDescArray + NUM_RX_DESC - 1);
5608 return 0;
5609
5610err_out:
5611 rtl8169_rx_clear(tp);
5612 return -ENOMEM;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005613}
5614
Linus Torvalds1da177e2005-04-16 15:20:36 -07005615static int rtl8169_init_ring(struct net_device *dev)
5616{
5617 struct rtl8169_private *tp = netdev_priv(dev);
5618
5619 rtl8169_init_ring_indexes(tp);
5620
5621 memset(tp->tx_skb, 0x0, NUM_TX_DESC * sizeof(struct ring_info));
Eric Dumazet6f0333b2010-10-11 11:17:47 +00005622 memset(tp->Rx_databuff, 0x0, NUM_RX_DESC * sizeof(void *));
Linus Torvalds1da177e2005-04-16 15:20:36 -07005623
Stanislaw Gruszka0ecbe1c2010-10-20 22:25:37 +00005624 return rtl8169_rx_fill(tp);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005625}
5626
Stanislaw Gruszka48addcc2010-10-20 22:25:39 +00005627static void rtl8169_unmap_tx_skb(struct device *d, struct ring_info *tx_skb,
Linus Torvalds1da177e2005-04-16 15:20:36 -07005628 struct TxDesc *desc)
5629{
5630 unsigned int len = tx_skb->len;
5631
Stanislaw Gruszka48addcc2010-10-20 22:25:39 +00005632 dma_unmap_single(d, le64_to_cpu(desc->addr), len, DMA_TO_DEVICE);
5633
Linus Torvalds1da177e2005-04-16 15:20:36 -07005634 desc->opts1 = 0x00;
5635 desc->opts2 = 0x00;
5636 desc->addr = 0x00;
5637 tx_skb->len = 0;
5638}
5639
Stanislaw Gruszka3eafe502010-10-20 22:25:36 +00005640static void rtl8169_tx_clear_range(struct rtl8169_private *tp, u32 start,
5641 unsigned int n)
Linus Torvalds1da177e2005-04-16 15:20:36 -07005642{
5643 unsigned int i;
5644
Stanislaw Gruszka3eafe502010-10-20 22:25:36 +00005645 for (i = 0; i < n; i++) {
5646 unsigned int entry = (start + i) % NUM_TX_DESC;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005647 struct ring_info *tx_skb = tp->tx_skb + entry;
5648 unsigned int len = tx_skb->len;
5649
5650 if (len) {
5651 struct sk_buff *skb = tx_skb->skb;
5652
Stanislaw Gruszka48addcc2010-10-20 22:25:39 +00005653 rtl8169_unmap_tx_skb(&tp->pci_dev->dev, tx_skb,
Linus Torvalds1da177e2005-04-16 15:20:36 -07005654 tp->TxDescArray + entry);
5655 if (skb) {
Stanislaw Gruszkacac4b222010-10-20 22:25:40 +00005656 tp->dev->stats.tx_dropped++;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005657 dev_kfree_skb(skb);
5658 tx_skb->skb = NULL;
5659 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07005660 }
5661 }
Stanislaw Gruszka3eafe502010-10-20 22:25:36 +00005662}
5663
5664static void rtl8169_tx_clear(struct rtl8169_private *tp)
5665{
5666 rtl8169_tx_clear_range(tp, tp->dirty_tx, NUM_TX_DESC);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005667 tp->cur_tx = tp->dirty_tx = 0;
Igor Maravic036dafa2012-03-05 00:01:25 +01005668 netdev_reset_queue(tp->dev);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005669}
5670
Francois Romieu4422bcd2012-01-26 11:23:32 +01005671static void rtl_reset_work(struct rtl8169_private *tp)
Linus Torvalds1da177e2005-04-16 15:20:36 -07005672{
David Howellsc4028952006-11-22 14:57:56 +00005673 struct net_device *dev = tp->dev;
Francois Romieu56de4142011-03-15 17:29:31 +01005674 int i;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005675
Francois Romieuda78dbf2012-01-26 14:18:23 +01005676 napi_disable(&tp->napi);
5677 netif_stop_queue(dev);
5678 synchronize_sched();
Linus Torvalds1da177e2005-04-16 15:20:36 -07005679
françois romieuc7c2c392011-12-04 20:30:52 +00005680 rtl8169_hw_reset(tp);
5681
Francois Romieu56de4142011-03-15 17:29:31 +01005682 for (i = 0; i < NUM_RX_DESC; i++)
5683 rtl8169_mark_to_asic(tp->RxDescArray + i, rx_buf_sz);
5684
Linus Torvalds1da177e2005-04-16 15:20:36 -07005685 rtl8169_tx_clear(tp);
françois romieuc7c2c392011-12-04 20:30:52 +00005686 rtl8169_init_ring_indexes(tp);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005687
Francois Romieuda78dbf2012-01-26 14:18:23 +01005688 napi_enable(&tp->napi);
Francois Romieu56de4142011-03-15 17:29:31 +01005689 rtl_hw_start(dev);
5690 netif_wake_queue(dev);
5691 rtl8169_check_link_status(dev, tp, tp->mmio_addr);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005692}
5693
5694static void rtl8169_tx_timeout(struct net_device *dev)
5695{
Francois Romieuda78dbf2012-01-26 14:18:23 +01005696 struct rtl8169_private *tp = netdev_priv(dev);
5697
5698 rtl_schedule_task(tp, RTL_FLAG_TASK_RESET_PENDING);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005699}
5700
5701static int rtl8169_xmit_frags(struct rtl8169_private *tp, struct sk_buff *skb,
Francois Romieu2b7b4312011-04-18 22:53:24 -07005702 u32 *opts)
Linus Torvalds1da177e2005-04-16 15:20:36 -07005703{
5704 struct skb_shared_info *info = skb_shinfo(skb);
5705 unsigned int cur_frag, entry;
Jeff Garzika6343af2007-07-17 05:39:58 -04005706 struct TxDesc * uninitialized_var(txd);
Stanislaw Gruszka48addcc2010-10-20 22:25:39 +00005707 struct device *d = &tp->pci_dev->dev;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005708
5709 entry = tp->cur_tx;
5710 for (cur_frag = 0; cur_frag < info->nr_frags; cur_frag++) {
Eric Dumazet9e903e02011-10-18 21:00:24 +00005711 const skb_frag_t *frag = info->frags + cur_frag;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005712 dma_addr_t mapping;
5713 u32 status, len;
5714 void *addr;
5715
5716 entry = (entry + 1) % NUM_TX_DESC;
5717
5718 txd = tp->TxDescArray + entry;
Eric Dumazet9e903e02011-10-18 21:00:24 +00005719 len = skb_frag_size(frag);
Ian Campbell929f6182011-08-31 00:47:06 +00005720 addr = skb_frag_address(frag);
Stanislaw Gruszka48addcc2010-10-20 22:25:39 +00005721 mapping = dma_map_single(d, addr, len, DMA_TO_DEVICE);
Stanislaw Gruszkad827d862010-10-20 22:25:43 +00005722 if (unlikely(dma_mapping_error(d, mapping))) {
5723 if (net_ratelimit())
5724 netif_err(tp, drv, tp->dev,
5725 "Failed to map TX fragments DMA!\n");
Stanislaw Gruszka3eafe502010-10-20 22:25:36 +00005726 goto err_out;
Stanislaw Gruszkad827d862010-10-20 22:25:43 +00005727 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07005728
Francois Romieucecb5fd2011-04-01 10:21:07 +02005729 /* Anti gcc 2.95.3 bugware (sic) */
Francois Romieu2b7b4312011-04-18 22:53:24 -07005730 status = opts[0] | len |
5731 (RingEnd * !((entry + 1) % NUM_TX_DESC));
Linus Torvalds1da177e2005-04-16 15:20:36 -07005732
5733 txd->opts1 = cpu_to_le32(status);
Francois Romieu2b7b4312011-04-18 22:53:24 -07005734 txd->opts2 = cpu_to_le32(opts[1]);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005735 txd->addr = cpu_to_le64(mapping);
5736
5737 tp->tx_skb[entry].len = len;
5738 }
5739
5740 if (cur_frag) {
5741 tp->tx_skb[entry].skb = skb;
5742 txd->opts1 |= cpu_to_le32(LastFrag);
5743 }
5744
5745 return cur_frag;
Stanislaw Gruszka3eafe502010-10-20 22:25:36 +00005746
5747err_out:
5748 rtl8169_tx_clear_range(tp, tp->cur_tx + 1, cur_frag);
5749 return -EIO;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005750}
5751
Francois Romieu2b7b4312011-04-18 22:53:24 -07005752static inline void rtl8169_tso_csum(struct rtl8169_private *tp,
5753 struct sk_buff *skb, u32 *opts)
Linus Torvalds1da177e2005-04-16 15:20:36 -07005754{
Francois Romieu2b7b4312011-04-18 22:53:24 -07005755 const struct rtl_tx_desc_info *info = tx_desc_info + tp->txd_version;
Michał Mirosław350fb322011-04-08 06:35:56 +00005756 u32 mss = skb_shinfo(skb)->gso_size;
Francois Romieu2b7b4312011-04-18 22:53:24 -07005757 int offset = info->opts_offset;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005758
Francois Romieu2b7b4312011-04-18 22:53:24 -07005759 if (mss) {
5760 opts[0] |= TD_LSO;
5761 opts[offset] |= min(mss, TD_MSS_MAX) << info->mss_shift;
5762 } else if (skb->ip_summed == CHECKSUM_PARTIAL) {
Arnaldo Carvalho de Meloeddc9ec2007-04-20 22:47:35 -07005763 const struct iphdr *ip = ip_hdr(skb);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005764
5765 if (ip->protocol == IPPROTO_TCP)
Francois Romieu2b7b4312011-04-18 22:53:24 -07005766 opts[offset] |= info->checksum.tcp;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005767 else if (ip->protocol == IPPROTO_UDP)
Francois Romieu2b7b4312011-04-18 22:53:24 -07005768 opts[offset] |= info->checksum.udp;
5769 else
5770 WARN_ON_ONCE(1);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005771 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07005772}
5773
Stephen Hemminger613573252009-08-31 19:50:58 +00005774static netdev_tx_t rtl8169_start_xmit(struct sk_buff *skb,
5775 struct net_device *dev)
Linus Torvalds1da177e2005-04-16 15:20:36 -07005776{
5777 struct rtl8169_private *tp = netdev_priv(dev);
Stanislaw Gruszka3eafe502010-10-20 22:25:36 +00005778 unsigned int entry = tp->cur_tx % NUM_TX_DESC;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005779 struct TxDesc *txd = tp->TxDescArray + entry;
5780 void __iomem *ioaddr = tp->mmio_addr;
Stanislaw Gruszka48addcc2010-10-20 22:25:39 +00005781 struct device *d = &tp->pci_dev->dev;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005782 dma_addr_t mapping;
5783 u32 status, len;
Francois Romieu2b7b4312011-04-18 22:53:24 -07005784 u32 opts[2];
Stanislaw Gruszka3eafe502010-10-20 22:25:36 +00005785 int frags;
Francois Romieu5b0384f2006-08-16 16:00:01 +02005786
Julien Ducourthial477206a2012-05-09 00:00:06 +02005787 if (unlikely(!TX_FRAGS_READY_FOR(tp, skb_shinfo(skb)->nr_frags))) {
Joe Perchesbf82c182010-02-09 11:49:50 +00005788 netif_err(tp, drv, dev, "BUG! Tx Ring full when queue awake!\n");
Stanislaw Gruszka3eafe502010-10-20 22:25:36 +00005789 goto err_stop_0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005790 }
5791
5792 if (unlikely(le32_to_cpu(txd->opts1) & DescOwn))
Stanislaw Gruszka3eafe502010-10-20 22:25:36 +00005793 goto err_stop_0;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005794
Stanislaw Gruszka3eafe502010-10-20 22:25:36 +00005795 len = skb_headlen(skb);
Stanislaw Gruszka48addcc2010-10-20 22:25:39 +00005796 mapping = dma_map_single(d, skb->data, len, DMA_TO_DEVICE);
Stanislaw Gruszkad827d862010-10-20 22:25:43 +00005797 if (unlikely(dma_mapping_error(d, mapping))) {
5798 if (net_ratelimit())
5799 netif_err(tp, drv, dev, "Failed to map TX DMA!\n");
Stanislaw Gruszka3eafe502010-10-20 22:25:36 +00005800 goto err_dma_0;
Stanislaw Gruszkad827d862010-10-20 22:25:43 +00005801 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07005802
5803 tp->tx_skb[entry].len = len;
5804 txd->addr = cpu_to_le64(mapping);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005805
Francois Romieu2b7b4312011-04-18 22:53:24 -07005806 opts[1] = cpu_to_le32(rtl8169_tx_vlan_tag(tp, skb));
5807 opts[0] = DescOwn;
Stanislaw Gruszka3eafe502010-10-20 22:25:36 +00005808
Francois Romieu2b7b4312011-04-18 22:53:24 -07005809 rtl8169_tso_csum(tp, skb, opts);
5810
5811 frags = rtl8169_xmit_frags(tp, skb, opts);
Stanislaw Gruszka3eafe502010-10-20 22:25:36 +00005812 if (frags < 0)
5813 goto err_dma_1;
5814 else if (frags)
Francois Romieu2b7b4312011-04-18 22:53:24 -07005815 opts[0] |= FirstFrag;
Stanislaw Gruszka3eafe502010-10-20 22:25:36 +00005816 else {
Francois Romieu2b7b4312011-04-18 22:53:24 -07005817 opts[0] |= FirstFrag | LastFrag;
Stanislaw Gruszka3eafe502010-10-20 22:25:36 +00005818 tp->tx_skb[entry].skb = skb;
5819 }
5820
Francois Romieu2b7b4312011-04-18 22:53:24 -07005821 txd->opts2 = cpu_to_le32(opts[1]);
5822
Igor Maravic036dafa2012-03-05 00:01:25 +01005823 netdev_sent_queue(dev, skb->len);
5824
Richard Cochran5047fb52012-03-10 07:29:42 +00005825 skb_tx_timestamp(skb);
5826
Linus Torvalds1da177e2005-04-16 15:20:36 -07005827 wmb();
5828
Francois Romieucecb5fd2011-04-01 10:21:07 +02005829 /* Anti gcc 2.95.3 bugware (sic) */
Francois Romieu2b7b4312011-04-18 22:53:24 -07005830 status = opts[0] | len | (RingEnd * !((entry + 1) % NUM_TX_DESC));
Linus Torvalds1da177e2005-04-16 15:20:36 -07005831 txd->opts1 = cpu_to_le32(status);
5832
Linus Torvalds1da177e2005-04-16 15:20:36 -07005833 tp->cur_tx += frags + 1;
5834
David Dillow4c020a92010-03-03 16:33:10 +00005835 wmb();
Linus Torvalds1da177e2005-04-16 15:20:36 -07005836
Francois Romieucecb5fd2011-04-01 10:21:07 +02005837 RTL_W8(TxPoll, NPQ);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005838
Francois Romieuda78dbf2012-01-26 14:18:23 +01005839 mmiowb();
5840
Julien Ducourthial477206a2012-05-09 00:00:06 +02005841 if (!TX_FRAGS_READY_FOR(tp, MAX_SKB_FRAGS)) {
Francois Romieuae1f23f2012-01-31 00:00:19 +01005842 /* Avoid wrongly optimistic queue wake-up: rtl_tx thread must
5843 * not miss a ring update when it notices a stopped queue.
5844 */
5845 smp_wmb();
Linus Torvalds1da177e2005-04-16 15:20:36 -07005846 netif_stop_queue(dev);
Francois Romieuae1f23f2012-01-31 00:00:19 +01005847 /* Sync with rtl_tx:
5848 * - publish queue status and cur_tx ring index (write barrier)
5849 * - refresh dirty_tx ring index (read barrier).
5850 * May the current thread have a pessimistic view of the ring
5851 * status and forget to wake up queue, a racing rtl_tx thread
5852 * can't.
5853 */
Francois Romieu1e874e02012-01-27 15:05:38 +01005854 smp_mb();
Julien Ducourthial477206a2012-05-09 00:00:06 +02005855 if (TX_FRAGS_READY_FOR(tp, MAX_SKB_FRAGS))
Linus Torvalds1da177e2005-04-16 15:20:36 -07005856 netif_wake_queue(dev);
5857 }
5858
Stephen Hemminger613573252009-08-31 19:50:58 +00005859 return NETDEV_TX_OK;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005860
Stanislaw Gruszka3eafe502010-10-20 22:25:36 +00005861err_dma_1:
Stanislaw Gruszka48addcc2010-10-20 22:25:39 +00005862 rtl8169_unmap_tx_skb(d, tp->tx_skb + entry, txd);
Stanislaw Gruszka3eafe502010-10-20 22:25:36 +00005863err_dma_0:
5864 dev_kfree_skb(skb);
5865 dev->stats.tx_dropped++;
5866 return NETDEV_TX_OK;
5867
5868err_stop_0:
Linus Torvalds1da177e2005-04-16 15:20:36 -07005869 netif_stop_queue(dev);
Francois Romieucebf8cc2007-10-18 12:06:54 +02005870 dev->stats.tx_dropped++;
Stephen Hemminger613573252009-08-31 19:50:58 +00005871 return NETDEV_TX_BUSY;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005872}
5873
5874static void rtl8169_pcierr_interrupt(struct net_device *dev)
5875{
5876 struct rtl8169_private *tp = netdev_priv(dev);
5877 struct pci_dev *pdev = tp->pci_dev;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005878 u16 pci_status, pci_cmd;
5879
5880 pci_read_config_word(pdev, PCI_COMMAND, &pci_cmd);
5881 pci_read_config_word(pdev, PCI_STATUS, &pci_status);
5882
Joe Perchesbf82c182010-02-09 11:49:50 +00005883 netif_err(tp, intr, dev, "PCI error (cmd = 0x%04x, status = 0x%04x)\n",
5884 pci_cmd, pci_status);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005885
5886 /*
5887 * The recovery sequence below admits a very elaborated explanation:
5888 * - it seems to work;
Francois Romieud03902b2006-11-23 00:00:42 +01005889 * - I did not see what else could be done;
5890 * - it makes iop3xx happy.
Linus Torvalds1da177e2005-04-16 15:20:36 -07005891 *
5892 * Feel free to adjust to your needs.
5893 */
Francois Romieua27993f2006-12-18 00:04:19 +01005894 if (pdev->broken_parity_status)
Francois Romieud03902b2006-11-23 00:00:42 +01005895 pci_cmd &= ~PCI_COMMAND_PARITY;
5896 else
5897 pci_cmd |= PCI_COMMAND_SERR | PCI_COMMAND_PARITY;
5898
5899 pci_write_config_word(pdev, PCI_COMMAND, pci_cmd);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005900
5901 pci_write_config_word(pdev, PCI_STATUS,
5902 pci_status & (PCI_STATUS_DETECTED_PARITY |
5903 PCI_STATUS_SIG_SYSTEM_ERROR | PCI_STATUS_REC_MASTER_ABORT |
5904 PCI_STATUS_REC_TARGET_ABORT | PCI_STATUS_SIG_TARGET_ABORT));
5905
5906 /* The infamous DAC f*ckup only happens at boot time */
5907 if ((tp->cp_cmd & PCIDAC) && !tp->dirty_rx && !tp->cur_rx) {
françois romieue6de30d2011-01-03 15:08:37 +00005908 void __iomem *ioaddr = tp->mmio_addr;
5909
Joe Perchesbf82c182010-02-09 11:49:50 +00005910 netif_info(tp, intr, dev, "disabling PCI DAC\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -07005911 tp->cp_cmd &= ~PCIDAC;
5912 RTL_W16(CPlusCmd, tp->cp_cmd);
5913 dev->features &= ~NETIF_F_HIGHDMA;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005914 }
5915
françois romieue6de30d2011-01-03 15:08:37 +00005916 rtl8169_hw_reset(tp);
Francois Romieud03902b2006-11-23 00:00:42 +01005917
Francois Romieu98ddf982012-01-31 10:47:34 +01005918 rtl_schedule_task(tp, RTL_FLAG_TASK_RESET_PENDING);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005919}
5920
Igor Maravic036dafa2012-03-05 00:01:25 +01005921struct rtl_txc {
5922 int packets;
5923 int bytes;
5924};
5925
Francois Romieuda78dbf2012-01-26 14:18:23 +01005926static void rtl_tx(struct net_device *dev, struct rtl8169_private *tp)
Linus Torvalds1da177e2005-04-16 15:20:36 -07005927{
Igor Maravic036dafa2012-03-05 00:01:25 +01005928 struct rtl8169_stats *tx_stats = &tp->tx_stats;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005929 unsigned int dirty_tx, tx_left;
Igor Maravic036dafa2012-03-05 00:01:25 +01005930 struct rtl_txc txc = { 0, 0 };
Linus Torvalds1da177e2005-04-16 15:20:36 -07005931
Linus Torvalds1da177e2005-04-16 15:20:36 -07005932 dirty_tx = tp->dirty_tx;
5933 smp_rmb();
5934 tx_left = tp->cur_tx - dirty_tx;
5935
5936 while (tx_left > 0) {
5937 unsigned int entry = dirty_tx % NUM_TX_DESC;
5938 struct ring_info *tx_skb = tp->tx_skb + entry;
Linus Torvalds1da177e2005-04-16 15:20:36 -07005939 u32 status;
5940
5941 rmb();
5942 status = le32_to_cpu(tp->TxDescArray[entry].opts1);
5943 if (status & DescOwn)
5944 break;
5945
Stanislaw Gruszka48addcc2010-10-20 22:25:39 +00005946 rtl8169_unmap_tx_skb(&tp->pci_dev->dev, tx_skb,
5947 tp->TxDescArray + entry);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005948 if (status & LastFrag) {
Igor Maravic036dafa2012-03-05 00:01:25 +01005949 struct sk_buff *skb = tx_skb->skb;
5950
5951 txc.packets++;
5952 txc.bytes += skb->len;
5953 dev_kfree_skb(skb);
Linus Torvalds1da177e2005-04-16 15:20:36 -07005954 tx_skb->skb = NULL;
5955 }
5956 dirty_tx++;
5957 tx_left--;
5958 }
5959
Igor Maravic036dafa2012-03-05 00:01:25 +01005960 u64_stats_update_begin(&tx_stats->syncp);
5961 tx_stats->packets += txc.packets;
5962 tx_stats->bytes += txc.bytes;
5963 u64_stats_update_end(&tx_stats->syncp);
5964
5965 netdev_completed_queue(dev, txc.packets, txc.bytes);
5966
Linus Torvalds1da177e2005-04-16 15:20:36 -07005967 if (tp->dirty_tx != dirty_tx) {
5968 tp->dirty_tx = dirty_tx;
Francois Romieuae1f23f2012-01-31 00:00:19 +01005969 /* Sync with rtl8169_start_xmit:
5970 * - publish dirty_tx ring index (write barrier)
5971 * - refresh cur_tx ring index and queue status (read barrier)
5972 * May the current thread miss the stopped queue condition,
5973 * a racing xmit thread can only have a right view of the
5974 * ring status.
5975 */
Francois Romieu1e874e02012-01-27 15:05:38 +01005976 smp_mb();
Linus Torvalds1da177e2005-04-16 15:20:36 -07005977 if (netif_queue_stopped(dev) &&
Julien Ducourthial477206a2012-05-09 00:00:06 +02005978 TX_FRAGS_READY_FOR(tp, MAX_SKB_FRAGS)) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07005979 netif_wake_queue(dev);
5980 }
Francois Romieud78ae2d2007-08-26 20:08:19 +02005981 /*
5982 * 8168 hack: TxPoll requests are lost when the Tx packets are
5983 * too close. Let's kick an extra TxPoll request when a burst
5984 * of start_xmit activity is detected (if it is not detected,
5985 * it is slow enough). -- FR
5986 */
Francois Romieuda78dbf2012-01-26 14:18:23 +01005987 if (tp->cur_tx != dirty_tx) {
5988 void __iomem *ioaddr = tp->mmio_addr;
5989
Francois Romieud78ae2d2007-08-26 20:08:19 +02005990 RTL_W8(TxPoll, NPQ);
Francois Romieuda78dbf2012-01-26 14:18:23 +01005991 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07005992 }
5993}
5994
Francois Romieu126fa4b2005-05-12 20:09:17 -04005995static inline int rtl8169_fragmented_frame(u32 status)
5996{
5997 return (status & (FirstFrag | LastFrag)) != (FirstFrag | LastFrag);
5998}
5999
Eric Dumazetadea1ac72010-09-05 20:04:05 -07006000static inline void rtl8169_rx_csum(struct sk_buff *skb, u32 opts1)
Linus Torvalds1da177e2005-04-16 15:20:36 -07006001{
Linus Torvalds1da177e2005-04-16 15:20:36 -07006002 u32 status = opts1 & RxProtoMask;
6003
6004 if (((status == RxProtoTCP) && !(opts1 & TCPFail)) ||
Shan Weid5d3ebe2010-11-12 00:15:25 +00006005 ((status == RxProtoUDP) && !(opts1 & UDPFail)))
Linus Torvalds1da177e2005-04-16 15:20:36 -07006006 skb->ip_summed = CHECKSUM_UNNECESSARY;
6007 else
Eric Dumazetbc8acf22010-09-02 13:07:41 -07006008 skb_checksum_none_assert(skb);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006009}
6010
Eric Dumazet6f0333b2010-10-11 11:17:47 +00006011static struct sk_buff *rtl8169_try_rx_copy(void *data,
6012 struct rtl8169_private *tp,
6013 int pkt_size,
6014 dma_addr_t addr)
Linus Torvalds1da177e2005-04-16 15:20:36 -07006015{
Stephen Hemmingerb4496552007-06-17 01:06:49 +02006016 struct sk_buff *skb;
Stanislaw Gruszka48addcc2010-10-20 22:25:39 +00006017 struct device *d = &tp->pci_dev->dev;
Linus Torvalds1da177e2005-04-16 15:20:36 -07006018
Eric Dumazet6f0333b2010-10-11 11:17:47 +00006019 data = rtl8169_align(data);
Stanislaw Gruszka48addcc2010-10-20 22:25:39 +00006020 dma_sync_single_for_cpu(d, addr, pkt_size, DMA_FROM_DEVICE);
Eric Dumazet6f0333b2010-10-11 11:17:47 +00006021 prefetch(data);
6022 skb = netdev_alloc_skb_ip_align(tp->dev, pkt_size);
6023 if (skb)
6024 memcpy(skb->data, data, pkt_size);
Stanislaw Gruszka48addcc2010-10-20 22:25:39 +00006025 dma_sync_single_for_device(d, addr, pkt_size, DMA_FROM_DEVICE);
6026
Eric Dumazet6f0333b2010-10-11 11:17:47 +00006027 return skb;
Linus Torvalds1da177e2005-04-16 15:20:36 -07006028}
6029
Francois Romieuda78dbf2012-01-26 14:18:23 +01006030static int rtl_rx(struct net_device *dev, struct rtl8169_private *tp, u32 budget)
Linus Torvalds1da177e2005-04-16 15:20:36 -07006031{
6032 unsigned int cur_rx, rx_left;
Eric Dumazet6f0333b2010-10-11 11:17:47 +00006033 unsigned int count;
Linus Torvalds1da177e2005-04-16 15:20:36 -07006034
Linus Torvalds1da177e2005-04-16 15:20:36 -07006035 cur_rx = tp->cur_rx;
6036 rx_left = NUM_RX_DESC + tp->dirty_rx - cur_rx;
Francois Romieu865c6522008-05-11 14:51:00 +02006037 rx_left = min(rx_left, budget);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006038
Richard Dawe4dcb7d32005-05-27 21:12:00 +02006039 for (; rx_left > 0; rx_left--, cur_rx++) {
Linus Torvalds1da177e2005-04-16 15:20:36 -07006040 unsigned int entry = cur_rx % NUM_RX_DESC;
Francois Romieu126fa4b2005-05-12 20:09:17 -04006041 struct RxDesc *desc = tp->RxDescArray + entry;
Linus Torvalds1da177e2005-04-16 15:20:36 -07006042 u32 status;
6043
6044 rmb();
David S. Miller8decf862011-09-22 03:23:13 -04006045 status = le32_to_cpu(desc->opts1) & tp->opts1_mask;
Linus Torvalds1da177e2005-04-16 15:20:36 -07006046
6047 if (status & DescOwn)
6048 break;
Richard Dawe4dcb7d32005-05-27 21:12:00 +02006049 if (unlikely(status & RxRES)) {
Joe Perchesbf82c182010-02-09 11:49:50 +00006050 netif_info(tp, rx_err, dev, "Rx ERROR. status = %08x\n",
6051 status);
Francois Romieucebf8cc2007-10-18 12:06:54 +02006052 dev->stats.rx_errors++;
Linus Torvalds1da177e2005-04-16 15:20:36 -07006053 if (status & (RxRWT | RxRUNT))
Francois Romieucebf8cc2007-10-18 12:06:54 +02006054 dev->stats.rx_length_errors++;
Linus Torvalds1da177e2005-04-16 15:20:36 -07006055 if (status & RxCRC)
Francois Romieucebf8cc2007-10-18 12:06:54 +02006056 dev->stats.rx_crc_errors++;
Francois Romieu9dccf612006-05-14 12:31:17 +02006057 if (status & RxFOVF) {
Francois Romieuda78dbf2012-01-26 14:18:23 +01006058 rtl_schedule_task(tp, RTL_FLAG_TASK_RESET_PENDING);
Francois Romieucebf8cc2007-10-18 12:06:54 +02006059 dev->stats.rx_fifo_errors++;
Francois Romieu9dccf612006-05-14 12:31:17 +02006060 }
Ben Greear6bbe0212012-02-10 15:04:33 +00006061 if ((status & (RxRUNT | RxCRC)) &&
6062 !(status & (RxRWT | RxFOVF)) &&
6063 (dev->features & NETIF_F_RXALL))
6064 goto process_pkt;
6065
Eric Dumazet6f0333b2010-10-11 11:17:47 +00006066 rtl8169_mark_to_asic(desc, rx_buf_sz);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006067 } else {
Eric Dumazet6f0333b2010-10-11 11:17:47 +00006068 struct sk_buff *skb;
Ben Greear6bbe0212012-02-10 15:04:33 +00006069 dma_addr_t addr;
6070 int pkt_size;
6071
6072process_pkt:
6073 addr = le64_to_cpu(desc->addr);
Ben Greear79d0c1d2012-02-10 15:04:34 +00006074 if (likely(!(dev->features & NETIF_F_RXFCS)))
6075 pkt_size = (status & 0x00003fff) - 4;
6076 else
6077 pkt_size = status & 0x00003fff;
Linus Torvalds1da177e2005-04-16 15:20:36 -07006078
Francois Romieu126fa4b2005-05-12 20:09:17 -04006079 /*
6080 * The driver does not support incoming fragmented
6081 * frames. They are seen as a symptom of over-mtu
6082 * sized frames.
6083 */
6084 if (unlikely(rtl8169_fragmented_frame(status))) {
Francois Romieucebf8cc2007-10-18 12:06:54 +02006085 dev->stats.rx_dropped++;
6086 dev->stats.rx_length_errors++;
Eric Dumazet6f0333b2010-10-11 11:17:47 +00006087 rtl8169_mark_to_asic(desc, rx_buf_sz);
Richard Dawe4dcb7d32005-05-27 21:12:00 +02006088 continue;
Francois Romieu126fa4b2005-05-12 20:09:17 -04006089 }
6090
Eric Dumazet6f0333b2010-10-11 11:17:47 +00006091 skb = rtl8169_try_rx_copy(tp->Rx_databuff[entry],
6092 tp, pkt_size, addr);
6093 rtl8169_mark_to_asic(desc, rx_buf_sz);
6094 if (!skb) {
6095 dev->stats.rx_dropped++;
6096 continue;
Linus Torvalds1da177e2005-04-16 15:20:36 -07006097 }
6098
Eric Dumazetadea1ac72010-09-05 20:04:05 -07006099 rtl8169_rx_csum(skb, status);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006100 skb_put(skb, pkt_size);
6101 skb->protocol = eth_type_trans(skb, dev);
6102
Francois Romieu7a8fc772011-03-01 17:18:33 +01006103 rtl8169_rx_vlan_tag(desc, skb);
6104
Francois Romieu56de4142011-03-15 17:29:31 +01006105 napi_gro_receive(&tp->napi, skb);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006106
Junchang Wang8027aa22012-03-04 23:30:32 +01006107 u64_stats_update_begin(&tp->rx_stats.syncp);
6108 tp->rx_stats.packets++;
6109 tp->rx_stats.bytes += pkt_size;
6110 u64_stats_update_end(&tp->rx_stats.syncp);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006111 }
Francois Romieu6dccd162007-02-13 23:38:05 +01006112
6113 /* Work around for AMD plateform. */
Al Viro95e09182007-12-22 18:55:39 +00006114 if ((desc->opts2 & cpu_to_le32(0xfffe000)) &&
Francois Romieu6dccd162007-02-13 23:38:05 +01006115 (tp->mac_version == RTL_GIGA_MAC_VER_05)) {
6116 desc->opts2 = 0;
6117 cur_rx++;
6118 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07006119 }
6120
6121 count = cur_rx - tp->cur_rx;
6122 tp->cur_rx = cur_rx;
6123
Eric Dumazet6f0333b2010-10-11 11:17:47 +00006124 tp->dirty_rx += count;
Linus Torvalds1da177e2005-04-16 15:20:36 -07006125
6126 return count;
6127}
6128
Francois Romieu07d3f512007-02-21 22:40:46 +01006129static irqreturn_t rtl8169_interrupt(int irq, void *dev_instance)
Linus Torvalds1da177e2005-04-16 15:20:36 -07006130{
Francois Romieu07d3f512007-02-21 22:40:46 +01006131 struct net_device *dev = dev_instance;
Linus Torvalds1da177e2005-04-16 15:20:36 -07006132 struct rtl8169_private *tp = netdev_priv(dev);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006133 int handled = 0;
Francois Romieu9085cdfa2012-01-26 12:59:08 +01006134 u16 status;
Linus Torvalds1da177e2005-04-16 15:20:36 -07006135
Francois Romieu9085cdfa2012-01-26 12:59:08 +01006136 status = rtl_get_events(tp);
Francois Romieuda78dbf2012-01-26 14:18:23 +01006137 if (status && status != 0xffff) {
6138 status &= RTL_EVENT_NAPI | tp->event_slow;
6139 if (status) {
6140 handled = 1;
françois romieu811fd302011-12-04 20:30:45 +00006141
Francois Romieuda78dbf2012-01-26 14:18:23 +01006142 rtl_irq_disable(tp);
6143 napi_schedule(&tp->napi);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006144 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07006145 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07006146 return IRQ_RETVAL(handled);
6147}
6148
Francois Romieuda78dbf2012-01-26 14:18:23 +01006149/*
6150 * Workqueue context.
6151 */
6152static void rtl_slow_event_work(struct rtl8169_private *tp)
6153{
6154 struct net_device *dev = tp->dev;
6155 u16 status;
6156
6157 status = rtl_get_events(tp) & tp->event_slow;
6158 rtl_ack_events(tp, status);
6159
6160 if (unlikely(status & RxFIFOOver)) {
6161 switch (tp->mac_version) {
6162 /* Work around for rx fifo overflow */
6163 case RTL_GIGA_MAC_VER_11:
6164 netif_stop_queue(dev);
Francois Romieu934714d2012-01-31 11:09:21 +01006165 /* XXX - Hack alert. See rtl_task(). */
6166 set_bit(RTL_FLAG_TASK_RESET_PENDING, tp->wk.flags);
Francois Romieuda78dbf2012-01-26 14:18:23 +01006167 default:
6168 break;
6169 }
6170 }
6171
6172 if (unlikely(status & SYSErr))
6173 rtl8169_pcierr_interrupt(dev);
6174
6175 if (status & LinkChg)
6176 __rtl8169_check_link_status(dev, tp, tp->mmio_addr, true);
6177
françois romieu7dbb4912012-06-09 10:53:16 +00006178 rtl_irq_enable_all(tp);
Francois Romieuda78dbf2012-01-26 14:18:23 +01006179}
6180
Francois Romieu4422bcd2012-01-26 11:23:32 +01006181static void rtl_task(struct work_struct *work)
6182{
Francois Romieuda78dbf2012-01-26 14:18:23 +01006183 static const struct {
6184 int bitnr;
6185 void (*action)(struct rtl8169_private *);
6186 } rtl_work[] = {
Francois Romieu934714d2012-01-31 11:09:21 +01006187 /* XXX - keep rtl_slow_event_work() as first element. */
Francois Romieuda78dbf2012-01-26 14:18:23 +01006188 { RTL_FLAG_TASK_SLOW_PENDING, rtl_slow_event_work },
6189 { RTL_FLAG_TASK_RESET_PENDING, rtl_reset_work },
6190 { RTL_FLAG_TASK_PHY_PENDING, rtl_phy_work }
6191 };
Francois Romieu4422bcd2012-01-26 11:23:32 +01006192 struct rtl8169_private *tp =
6193 container_of(work, struct rtl8169_private, wk.work);
Francois Romieuda78dbf2012-01-26 14:18:23 +01006194 struct net_device *dev = tp->dev;
6195 int i;
Francois Romieu4422bcd2012-01-26 11:23:32 +01006196
Francois Romieuda78dbf2012-01-26 14:18:23 +01006197 rtl_lock_work(tp);
6198
Francois Romieu6c4a70c2012-01-31 10:56:44 +01006199 if (!netif_running(dev) ||
6200 !test_bit(RTL_FLAG_TASK_ENABLED, tp->wk.flags))
Francois Romieuda78dbf2012-01-26 14:18:23 +01006201 goto out_unlock;
6202
6203 for (i = 0; i < ARRAY_SIZE(rtl_work); i++) {
6204 bool pending;
6205
Francois Romieuda78dbf2012-01-26 14:18:23 +01006206 pending = test_and_clear_bit(rtl_work[i].bitnr, tp->wk.flags);
Francois Romieuda78dbf2012-01-26 14:18:23 +01006207 if (pending)
6208 rtl_work[i].action(tp);
6209 }
6210
6211out_unlock:
6212 rtl_unlock_work(tp);
Francois Romieu4422bcd2012-01-26 11:23:32 +01006213}
6214
Stephen Hemmingerbea33482007-10-03 16:41:36 -07006215static int rtl8169_poll(struct napi_struct *napi, int budget)
Linus Torvalds1da177e2005-04-16 15:20:36 -07006216{
Stephen Hemmingerbea33482007-10-03 16:41:36 -07006217 struct rtl8169_private *tp = container_of(napi, struct rtl8169_private, napi);
6218 struct net_device *dev = tp->dev;
Francois Romieuda78dbf2012-01-26 14:18:23 +01006219 u16 enable_mask = RTL_EVENT_NAPI | tp->event_slow;
6220 int work_done= 0;
6221 u16 status;
Linus Torvalds1da177e2005-04-16 15:20:36 -07006222
Francois Romieuda78dbf2012-01-26 14:18:23 +01006223 status = rtl_get_events(tp);
6224 rtl_ack_events(tp, status & ~tp->event_slow);
6225
6226 if (status & RTL_EVENT_NAPI_RX)
6227 work_done = rtl_rx(dev, tp, (u32) budget);
6228
6229 if (status & RTL_EVENT_NAPI_TX)
6230 rtl_tx(dev, tp);
6231
6232 if (status & tp->event_slow) {
6233 enable_mask &= ~tp->event_slow;
6234
6235 rtl_schedule_task(tp, RTL_FLAG_TASK_SLOW_PENDING);
6236 }
Linus Torvalds1da177e2005-04-16 15:20:36 -07006237
Stephen Hemmingerbea33482007-10-03 16:41:36 -07006238 if (work_done < budget) {
Ben Hutchings288379f2009-01-19 16:43:59 -08006239 napi_complete(napi);
David Dillowf11a3772009-05-22 15:29:34 +00006240
Francois Romieuda78dbf2012-01-26 14:18:23 +01006241 rtl_irq_enable(tp, enable_mask);
6242 mmiowb();
Linus Torvalds1da177e2005-04-16 15:20:36 -07006243 }
6244
Stephen Hemmingerbea33482007-10-03 16:41:36 -07006245 return work_done;
Linus Torvalds1da177e2005-04-16 15:20:36 -07006246}
Linus Torvalds1da177e2005-04-16 15:20:36 -07006247
Francois Romieu523a6092008-09-10 22:28:56 +02006248static void rtl8169_rx_missed(struct net_device *dev, void __iomem *ioaddr)
6249{
6250 struct rtl8169_private *tp = netdev_priv(dev);
6251
6252 if (tp->mac_version > RTL_GIGA_MAC_VER_06)
6253 return;
6254
6255 dev->stats.rx_missed_errors += (RTL_R32(RxMissed) & 0xffffff);
6256 RTL_W32(RxMissed, 0);
6257}
6258
Linus Torvalds1da177e2005-04-16 15:20:36 -07006259static void rtl8169_down(struct net_device *dev)
6260{
6261 struct rtl8169_private *tp = netdev_priv(dev);
6262 void __iomem *ioaddr = tp->mmio_addr;
Linus Torvalds1da177e2005-04-16 15:20:36 -07006263
Francois Romieu4876cc12011-03-11 21:07:11 +01006264 del_timer_sync(&tp->timer);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006265
Stephen Hemminger93dd79e2007-10-28 17:14:06 +01006266 napi_disable(&tp->napi);
Francois Romieuda78dbf2012-01-26 14:18:23 +01006267 netif_stop_queue(dev);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006268
Hayes Wang92fc43b2011-07-06 15:58:03 +08006269 rtl8169_hw_reset(tp);
Stanislaw Gruszka323bb682010-10-20 22:25:41 +00006270 /*
6271 * At this point device interrupts can not be enabled in any function,
Francois Romieu209e5ac2012-01-26 09:59:50 +01006272 * as netif_running is not true (rtl8169_interrupt, rtl8169_reset_task)
6273 * and napi is disabled (rtl8169_poll).
Stanislaw Gruszka323bb682010-10-20 22:25:41 +00006274 */
Francois Romieu523a6092008-09-10 22:28:56 +02006275 rtl8169_rx_missed(dev, ioaddr);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006276
Linus Torvalds1da177e2005-04-16 15:20:36 -07006277 /* Give a racing hard_start_xmit a few cycles to complete. */
Francois Romieuda78dbf2012-01-26 14:18:23 +01006278 synchronize_sched();
Linus Torvalds1da177e2005-04-16 15:20:36 -07006279
Linus Torvalds1da177e2005-04-16 15:20:36 -07006280 rtl8169_tx_clear(tp);
6281
6282 rtl8169_rx_clear(tp);
françois romieu065c27c2011-01-03 15:08:12 +00006283
6284 rtl_pll_power_down(tp);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006285}
6286
6287static int rtl8169_close(struct net_device *dev)
6288{
6289 struct rtl8169_private *tp = netdev_priv(dev);
6290 struct pci_dev *pdev = tp->pci_dev;
6291
Rafael J. Wysockie1759442010-03-14 14:33:51 +00006292 pm_runtime_get_sync(&pdev->dev);
6293
Francois Romieucecb5fd2011-04-01 10:21:07 +02006294 /* Update counters before going down */
Ivan Vecera355423d2009-02-06 21:49:57 -08006295 rtl8169_update_counters(dev);
6296
Francois Romieuda78dbf2012-01-26 14:18:23 +01006297 rtl_lock_work(tp);
Francois Romieu6c4a70c2012-01-31 10:56:44 +01006298 clear_bit(RTL_FLAG_TASK_ENABLED, tp->wk.flags);
Francois Romieuda78dbf2012-01-26 14:18:23 +01006299
Linus Torvalds1da177e2005-04-16 15:20:36 -07006300 rtl8169_down(dev);
Francois Romieuda78dbf2012-01-26 14:18:23 +01006301 rtl_unlock_work(tp);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006302
Francois Romieu92a7c4e2012-03-10 10:42:12 +01006303 free_irq(pdev->irq, dev);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006304
Stanislaw Gruszka82553bb2010-10-08 04:25:01 +00006305 dma_free_coherent(&pdev->dev, R8169_RX_RING_BYTES, tp->RxDescArray,
6306 tp->RxPhyAddr);
6307 dma_free_coherent(&pdev->dev, R8169_TX_RING_BYTES, tp->TxDescArray,
6308 tp->TxPhyAddr);
Linus Torvalds1da177e2005-04-16 15:20:36 -07006309 tp->TxDescArray = NULL;
6310 tp->RxDescArray = NULL;
6311
Rafael J. Wysockie1759442010-03-14 14:33:51 +00006312 pm_runtime_put_sync(&pdev->dev);
6313
Linus Torvalds1da177e2005-04-16 15:20:36 -07006314 return 0;
6315}
6316
Francois Romieudc1c00c2012-03-08 10:06:18 +01006317#ifdef CONFIG_NET_POLL_CONTROLLER
6318static void rtl8169_netpoll(struct net_device *dev)
6319{
6320 struct rtl8169_private *tp = netdev_priv(dev);
6321
6322 rtl8169_interrupt(tp->pci_dev->irq, dev);
6323}
6324#endif
6325
Francois Romieudf43ac72012-03-08 09:48:40 +01006326static int rtl_open(struct net_device *dev)
6327{
6328 struct rtl8169_private *tp = netdev_priv(dev);
6329 void __iomem *ioaddr = tp->mmio_addr;
6330 struct pci_dev *pdev = tp->pci_dev;
6331 int retval = -ENOMEM;
6332
6333 pm_runtime_get_sync(&pdev->dev);
6334
6335 /*
Jiri Kosinae75d6602012-04-08 21:48:52 +02006336 * Rx and Tx descriptors needs 256 bytes alignment.
Francois Romieudf43ac72012-03-08 09:48:40 +01006337 * dma_alloc_coherent provides more.
6338 */
6339 tp->TxDescArray = dma_alloc_coherent(&pdev->dev, R8169_TX_RING_BYTES,
6340 &tp->TxPhyAddr, GFP_KERNEL);
6341 if (!tp->TxDescArray)
6342 goto err_pm_runtime_put;
6343
6344 tp->RxDescArray = dma_alloc_coherent(&pdev->dev, R8169_RX_RING_BYTES,
6345 &tp->RxPhyAddr, GFP_KERNEL);
6346 if (!tp->RxDescArray)
6347 goto err_free_tx_0;
6348
6349 retval = rtl8169_init_ring(dev);
6350 if (retval < 0)
6351 goto err_free_rx_1;
6352
6353 INIT_WORK(&tp->wk.work, rtl_task);
6354
6355 smp_mb();
6356
6357 rtl_request_firmware(tp);
6358
Francois Romieu92a7c4e2012-03-10 10:42:12 +01006359 retval = request_irq(pdev->irq, rtl8169_interrupt,
Francois Romieudf43ac72012-03-08 09:48:40 +01006360 (tp->features & RTL_FEATURE_MSI) ? 0 : IRQF_SHARED,
6361 dev->name, dev);
6362 if (retval < 0)
6363 goto err_release_fw_2;
6364
6365 rtl_lock_work(tp);
6366
6367 set_bit(RTL_FLAG_TASK_ENABLED, tp->wk.flags);
6368
6369 napi_enable(&tp->napi);
6370
6371 rtl8169_init_phy(dev, tp);
6372
6373 __rtl8169_set_features(dev, dev->features);
6374
6375 rtl_pll_power_up(tp);
6376
6377 rtl_hw_start(dev);
6378
6379 netif_start_queue(dev);
6380
6381 rtl_unlock_work(tp);
6382
6383 tp->saved_wolopts = 0;
6384 pm_runtime_put_noidle(&pdev->dev);
6385
6386 rtl8169_check_link_status(dev, tp, ioaddr);
6387out:
6388 return retval;
6389
6390err_release_fw_2:
6391 rtl_release_firmware(tp);
6392 rtl8169_rx_clear(tp);
6393err_free_rx_1:
6394 dma_free_coherent(&pdev->dev, R8169_RX_RING_BYTES, tp->RxDescArray,
6395 tp->RxPhyAddr);
6396 tp->RxDescArray = NULL;
6397err_free_tx_0:
6398 dma_free_coherent(&pdev->dev, R8169_TX_RING_BYTES, tp->TxDescArray,
6399 tp->TxPhyAddr);
6400 tp->TxDescArray = NULL;
6401err_pm_runtime_put:
6402 pm_runtime_put_noidle(&pdev->dev);
6403 goto out;
6404}
6405
Junchang Wang8027aa22012-03-04 23:30:32 +01006406static struct rtnl_link_stats64 *
6407rtl8169_get_stats64(struct net_device *dev, struct rtnl_link_stats64 *stats)
Linus Torvalds1da177e2005-04-16 15:20:36 -07006408{
6409 struct rtl8169_private *tp = netdev_priv(dev);
6410 void __iomem *ioaddr = tp->mmio_addr;
Junchang Wang8027aa22012-03-04 23:30:32 +01006411 unsigned int start;
Linus Torvalds1da177e2005-04-16 15:20:36 -07006412
Francois Romieuda78dbf2012-01-26 14:18:23 +01006413 if (netif_running(dev))
Francois Romieu523a6092008-09-10 22:28:56 +02006414 rtl8169_rx_missed(dev, ioaddr);
Francois Romieu5b0384f2006-08-16 16:00:01 +02006415
Junchang Wang8027aa22012-03-04 23:30:32 +01006416 do {
6417 start = u64_stats_fetch_begin_bh(&tp->rx_stats.syncp);
6418 stats->rx_packets = tp->rx_stats.packets;
6419 stats->rx_bytes = tp->rx_stats.bytes;
6420 } while (u64_stats_fetch_retry_bh(&tp->rx_stats.syncp, start));
6421
6422
6423 do {
6424 start = u64_stats_fetch_begin_bh(&tp->tx_stats.syncp);
6425 stats->tx_packets = tp->tx_stats.packets;
6426 stats->tx_bytes = tp->tx_stats.bytes;
6427 } while (u64_stats_fetch_retry_bh(&tp->tx_stats.syncp, start));
6428
6429 stats->rx_dropped = dev->stats.rx_dropped;
6430 stats->tx_dropped = dev->stats.tx_dropped;
6431 stats->rx_length_errors = dev->stats.rx_length_errors;
6432 stats->rx_errors = dev->stats.rx_errors;
6433 stats->rx_crc_errors = dev->stats.rx_crc_errors;
6434 stats->rx_fifo_errors = dev->stats.rx_fifo_errors;
6435 stats->rx_missed_errors = dev->stats.rx_missed_errors;
6436
6437 return stats;
Linus Torvalds1da177e2005-04-16 15:20:36 -07006438}
6439
Rafael J. Wysocki861ab442009-04-05 08:40:04 +00006440static void rtl8169_net_suspend(struct net_device *dev)
Francois Romieu5d06a992006-02-23 00:47:58 +01006441{
françois romieu065c27c2011-01-03 15:08:12 +00006442 struct rtl8169_private *tp = netdev_priv(dev);
6443
Francois Romieu5d06a992006-02-23 00:47:58 +01006444 if (!netif_running(dev))
Rafael J. Wysocki861ab442009-04-05 08:40:04 +00006445 return;
Francois Romieu5d06a992006-02-23 00:47:58 +01006446
6447 netif_device_detach(dev);
6448 netif_stop_queue(dev);
Francois Romieuda78dbf2012-01-26 14:18:23 +01006449
6450 rtl_lock_work(tp);
6451 napi_disable(&tp->napi);
Francois Romieu6c4a70c2012-01-31 10:56:44 +01006452 clear_bit(RTL_FLAG_TASK_ENABLED, tp->wk.flags);
Francois Romieuda78dbf2012-01-26 14:18:23 +01006453 rtl_unlock_work(tp);
6454
6455 rtl_pll_power_down(tp);
Rafael J. Wysocki861ab442009-04-05 08:40:04 +00006456}
Francois Romieu5d06a992006-02-23 00:47:58 +01006457
Rafael J. Wysocki861ab442009-04-05 08:40:04 +00006458#ifdef CONFIG_PM
6459
6460static int rtl8169_suspend(struct device *device)
6461{
6462 struct pci_dev *pdev = to_pci_dev(device);
6463 struct net_device *dev = pci_get_drvdata(pdev);
6464
6465 rtl8169_net_suspend(dev);
Francois Romieu1371fa62007-04-02 23:01:11 +02006466
Francois Romieu5d06a992006-02-23 00:47:58 +01006467 return 0;
6468}
6469
Rafael J. Wysockie1759442010-03-14 14:33:51 +00006470static void __rtl8169_resume(struct net_device *dev)
6471{
françois romieu065c27c2011-01-03 15:08:12 +00006472 struct rtl8169_private *tp = netdev_priv(dev);
6473
Rafael J. Wysockie1759442010-03-14 14:33:51 +00006474 netif_device_attach(dev);
françois romieu065c27c2011-01-03 15:08:12 +00006475
6476 rtl_pll_power_up(tp);
6477
Artem Savkovcff4c162012-04-03 10:29:11 +00006478 rtl_lock_work(tp);
6479 napi_enable(&tp->napi);
Francois Romieu6c4a70c2012-01-31 10:56:44 +01006480 set_bit(RTL_FLAG_TASK_ENABLED, tp->wk.flags);
Artem Savkovcff4c162012-04-03 10:29:11 +00006481 rtl_unlock_work(tp);
Francois Romieuda78dbf2012-01-26 14:18:23 +01006482
Francois Romieu98ddf982012-01-31 10:47:34 +01006483 rtl_schedule_task(tp, RTL_FLAG_TASK_RESET_PENDING);
Rafael J. Wysockie1759442010-03-14 14:33:51 +00006484}
6485
Rafael J. Wysocki861ab442009-04-05 08:40:04 +00006486static int rtl8169_resume(struct device *device)
Francois Romieu5d06a992006-02-23 00:47:58 +01006487{
Rafael J. Wysocki861ab442009-04-05 08:40:04 +00006488 struct pci_dev *pdev = to_pci_dev(device);
Francois Romieu5d06a992006-02-23 00:47:58 +01006489 struct net_device *dev = pci_get_drvdata(pdev);
Stanislaw Gruszkafccec102010-10-20 22:25:42 +00006490 struct rtl8169_private *tp = netdev_priv(dev);
6491
6492 rtl8169_init_phy(dev, tp);
Francois Romieu5d06a992006-02-23 00:47:58 +01006493
Rafael J. Wysockie1759442010-03-14 14:33:51 +00006494 if (netif_running(dev))
6495 __rtl8169_resume(dev);
Francois Romieu5d06a992006-02-23 00:47:58 +01006496
Francois Romieu5d06a992006-02-23 00:47:58 +01006497 return 0;
6498}
6499
Rafael J. Wysockie1759442010-03-14 14:33:51 +00006500static int rtl8169_runtime_suspend(struct device *device)
6501{
6502 struct pci_dev *pdev = to_pci_dev(device);
6503 struct net_device *dev = pci_get_drvdata(pdev);
6504 struct rtl8169_private *tp = netdev_priv(dev);
6505
6506 if (!tp->TxDescArray)
6507 return 0;
6508
Francois Romieuda78dbf2012-01-26 14:18:23 +01006509 rtl_lock_work(tp);
Rafael J. Wysockie1759442010-03-14 14:33:51 +00006510 tp->saved_wolopts = __rtl8169_get_wol(tp);
6511 __rtl8169_set_wol(tp, WAKE_ANY);
Francois Romieuda78dbf2012-01-26 14:18:23 +01006512 rtl_unlock_work(tp);
Rafael J. Wysockie1759442010-03-14 14:33:51 +00006513
6514 rtl8169_net_suspend(dev);
6515
6516 return 0;
6517}
6518
6519static int rtl8169_runtime_resume(struct device *device)
6520{
6521 struct pci_dev *pdev = to_pci_dev(device);
6522 struct net_device *dev = pci_get_drvdata(pdev);
6523 struct rtl8169_private *tp = netdev_priv(dev);
6524
6525 if (!tp->TxDescArray)
6526 return 0;
6527
Francois Romieuda78dbf2012-01-26 14:18:23 +01006528 rtl_lock_work(tp);
Rafael J. Wysockie1759442010-03-14 14:33:51 +00006529 __rtl8169_set_wol(tp, tp->saved_wolopts);
6530 tp->saved_wolopts = 0;
Francois Romieuda78dbf2012-01-26 14:18:23 +01006531 rtl_unlock_work(tp);
Rafael J. Wysockie1759442010-03-14 14:33:51 +00006532
Stanislaw Gruszkafccec102010-10-20 22:25:42 +00006533 rtl8169_init_phy(dev, tp);
6534
Rafael J. Wysockie1759442010-03-14 14:33:51 +00006535 __rtl8169_resume(dev);
6536
6537 return 0;
6538}
6539
6540static int rtl8169_runtime_idle(struct device *device)
6541{
6542 struct pci_dev *pdev = to_pci_dev(device);
6543 struct net_device *dev = pci_get_drvdata(pdev);
6544 struct rtl8169_private *tp = netdev_priv(dev);
6545
Rafael J. Wysockie4fbce72010-12-08 15:32:14 +00006546 return tp->TxDescArray ? -EBUSY : 0;
Rafael J. Wysockie1759442010-03-14 14:33:51 +00006547}
6548
Alexey Dobriyan47145212009-12-14 18:00:08 -08006549static const struct dev_pm_ops rtl8169_pm_ops = {
Francois Romieucecb5fd2011-04-01 10:21:07 +02006550 .suspend = rtl8169_suspend,
6551 .resume = rtl8169_resume,
6552 .freeze = rtl8169_suspend,
6553 .thaw = rtl8169_resume,
6554 .poweroff = rtl8169_suspend,
6555 .restore = rtl8169_resume,
6556 .runtime_suspend = rtl8169_runtime_suspend,
6557 .runtime_resume = rtl8169_runtime_resume,
6558 .runtime_idle = rtl8169_runtime_idle,
Rafael J. Wysocki861ab442009-04-05 08:40:04 +00006559};
6560
6561#define RTL8169_PM_OPS (&rtl8169_pm_ops)
6562
6563#else /* !CONFIG_PM */
6564
6565#define RTL8169_PM_OPS NULL
6566
6567#endif /* !CONFIG_PM */
6568
David S. Miller1805b2f2011-10-24 18:18:09 -04006569static void rtl_wol_shutdown_quirk(struct rtl8169_private *tp)
6570{
6571 void __iomem *ioaddr = tp->mmio_addr;
6572
6573 /* WoL fails with 8168b when the receiver is disabled. */
6574 switch (tp->mac_version) {
6575 case RTL_GIGA_MAC_VER_11:
6576 case RTL_GIGA_MAC_VER_12:
6577 case RTL_GIGA_MAC_VER_17:
6578 pci_clear_master(tp->pci_dev);
6579
6580 RTL_W8(ChipCmd, CmdRxEnb);
6581 /* PCI commit */
6582 RTL_R8(ChipCmd);
6583 break;
6584 default:
6585 break;
6586 }
6587}
6588
Francois Romieu1765f952008-09-13 17:21:40 +02006589static void rtl_shutdown(struct pci_dev *pdev)
6590{
Rafael J. Wysocki861ab442009-04-05 08:40:04 +00006591 struct net_device *dev = pci_get_drvdata(pdev);
françois romieu4bb3f522009-06-17 11:41:45 +00006592 struct rtl8169_private *tp = netdev_priv(dev);
françois romieu2a15cd22012-03-06 01:14:12 +00006593 struct device *d = &pdev->dev;
6594
6595 pm_runtime_get_sync(d);
Francois Romieu1765f952008-09-13 17:21:40 +02006596
Rafael J. Wysocki861ab442009-04-05 08:40:04 +00006597 rtl8169_net_suspend(dev);
6598
Francois Romieucecb5fd2011-04-01 10:21:07 +02006599 /* Restore original MAC address */
Ivan Veceracc098dc2009-11-29 23:12:52 -08006600 rtl_rar_set(tp, dev->perm_addr);
6601
Hayes Wang92fc43b2011-07-06 15:58:03 +08006602 rtl8169_hw_reset(tp);
françois romieu4bb3f522009-06-17 11:41:45 +00006603
Rafael J. Wysocki861ab442009-04-05 08:40:04 +00006604 if (system_state == SYSTEM_POWER_OFF) {
David S. Miller1805b2f2011-10-24 18:18:09 -04006605 if (__rtl8169_get_wol(tp) & WAKE_ANY) {
6606 rtl_wol_suspend_quirk(tp);
6607 rtl_wol_shutdown_quirk(tp);
françois romieuca52efd2009-07-24 12:34:19 +00006608 }
6609
Rafael J. Wysocki861ab442009-04-05 08:40:04 +00006610 pci_wake_from_d3(pdev, true);
6611 pci_set_power_state(pdev, PCI_D3hot);
6612 }
françois romieu2a15cd22012-03-06 01:14:12 +00006613
6614 pm_runtime_put_noidle(d);
Rafael J. Wysocki861ab442009-04-05 08:40:04 +00006615}
Francois Romieu5d06a992006-02-23 00:47:58 +01006616
Francois Romieue27566e2012-03-08 09:54:01 +01006617static void __devexit rtl_remove_one(struct pci_dev *pdev)
6618{
6619 struct net_device *dev = pci_get_drvdata(pdev);
6620 struct rtl8169_private *tp = netdev_priv(dev);
6621
6622 if (tp->mac_version == RTL_GIGA_MAC_VER_27 ||
6623 tp->mac_version == RTL_GIGA_MAC_VER_28 ||
6624 tp->mac_version == RTL_GIGA_MAC_VER_31) {
6625 rtl8168_driver_stop(tp);
6626 }
6627
6628 cancel_work_sync(&tp->wk.work);
6629
Devendra Nagaad1be8d2012-05-31 01:51:20 +00006630 netif_napi_del(&tp->napi);
6631
Francois Romieue27566e2012-03-08 09:54:01 +01006632 unregister_netdev(dev);
6633
6634 rtl_release_firmware(tp);
6635
6636 if (pci_dev_run_wake(pdev))
6637 pm_runtime_get_noresume(&pdev->dev);
6638
6639 /* restore original MAC address */
6640 rtl_rar_set(tp, dev->perm_addr);
6641
6642 rtl_disable_msi(pdev, tp);
6643 rtl8169_release_board(pdev, dev, tp->mmio_addr);
6644 pci_set_drvdata(pdev, NULL);
6645}
6646
Francois Romieufa9c3852012-03-08 10:01:50 +01006647static const struct net_device_ops rtl_netdev_ops = {
Francois Romieudf43ac72012-03-08 09:48:40 +01006648 .ndo_open = rtl_open,
Francois Romieufa9c3852012-03-08 10:01:50 +01006649 .ndo_stop = rtl8169_close,
6650 .ndo_get_stats64 = rtl8169_get_stats64,
6651 .ndo_start_xmit = rtl8169_start_xmit,
6652 .ndo_tx_timeout = rtl8169_tx_timeout,
6653 .ndo_validate_addr = eth_validate_addr,
6654 .ndo_change_mtu = rtl8169_change_mtu,
6655 .ndo_fix_features = rtl8169_fix_features,
6656 .ndo_set_features = rtl8169_set_features,
6657 .ndo_set_mac_address = rtl_set_mac_address,
6658 .ndo_do_ioctl = rtl8169_ioctl,
6659 .ndo_set_rx_mode = rtl_set_rx_mode,
6660#ifdef CONFIG_NET_POLL_CONTROLLER
6661 .ndo_poll_controller = rtl8169_netpoll,
6662#endif
6663
6664};
6665
Francois Romieu31fa8b12012-03-08 10:09:40 +01006666static const struct rtl_cfg_info {
6667 void (*hw_start)(struct net_device *);
6668 unsigned int region;
6669 unsigned int align;
6670 u16 event_slow;
6671 unsigned features;
6672 u8 default_ver;
6673} rtl_cfg_infos [] = {
6674 [RTL_CFG_0] = {
6675 .hw_start = rtl_hw_start_8169,
6676 .region = 1,
6677 .align = 0,
6678 .event_slow = SYSErr | LinkChg | RxOverflow | RxFIFOOver,
6679 .features = RTL_FEATURE_GMII,
6680 .default_ver = RTL_GIGA_MAC_VER_01,
6681 },
6682 [RTL_CFG_1] = {
6683 .hw_start = rtl_hw_start_8168,
6684 .region = 2,
6685 .align = 8,
6686 .event_slow = SYSErr | LinkChg | RxOverflow,
6687 .features = RTL_FEATURE_GMII | RTL_FEATURE_MSI,
6688 .default_ver = RTL_GIGA_MAC_VER_11,
6689 },
6690 [RTL_CFG_2] = {
6691 .hw_start = rtl_hw_start_8101,
6692 .region = 2,
6693 .align = 8,
6694 .event_slow = SYSErr | LinkChg | RxOverflow | RxFIFOOver |
6695 PCSTimeout,
6696 .features = RTL_FEATURE_MSI,
6697 .default_ver = RTL_GIGA_MAC_VER_13,
6698 }
6699};
6700
6701/* Cfg9346_Unlock assumed. */
6702static unsigned rtl_try_msi(struct rtl8169_private *tp,
6703 const struct rtl_cfg_info *cfg)
6704{
6705 void __iomem *ioaddr = tp->mmio_addr;
6706 unsigned msi = 0;
6707 u8 cfg2;
6708
6709 cfg2 = RTL_R8(Config2) & ~MSIEnable;
6710 if (cfg->features & RTL_FEATURE_MSI) {
6711 if (pci_enable_msi(tp->pci_dev)) {
6712 netif_info(tp, hw, tp->dev, "no MSI. Back to INTx.\n");
6713 } else {
6714 cfg2 |= MSIEnable;
6715 msi = RTL_FEATURE_MSI;
6716 }
6717 }
6718 if (tp->mac_version <= RTL_GIGA_MAC_VER_06)
6719 RTL_W8(Config2, cfg2);
6720 return msi;
6721}
6722
Hayes Wangc5583862012-07-02 17:23:22 +08006723DECLARE_RTL_COND(rtl_link_list_ready_cond)
6724{
6725 void __iomem *ioaddr = tp->mmio_addr;
6726
6727 return RTL_R8(MCU) & LINK_LIST_RDY;
6728}
6729
6730DECLARE_RTL_COND(rtl_rxtx_empty_cond)
6731{
6732 void __iomem *ioaddr = tp->mmio_addr;
6733
6734 return (RTL_R8(MCU) & RXTX_EMPTY) == RXTX_EMPTY;
6735}
6736
6737static void __devinit rtl_hw_init_8168g(struct rtl8169_private *tp)
6738{
6739 void __iomem *ioaddr = tp->mmio_addr;
6740 u32 data;
6741
6742 tp->ocp_base = OCP_STD_PHY_BASE;
6743
6744 RTL_W32(MISC, RTL_R32(MISC) | RXDV_GATED_EN);
6745
6746 if (!rtl_udelay_loop_wait_high(tp, &rtl_txcfg_empty_cond, 100, 42))
6747 return;
6748
6749 if (!rtl_udelay_loop_wait_high(tp, &rtl_rxtx_empty_cond, 100, 42))
6750 return;
6751
6752 RTL_W8(ChipCmd, RTL_R8(ChipCmd) & ~(CmdTxEnb | CmdRxEnb));
6753 msleep(1);
6754 RTL_W8(MCU, RTL_R8(MCU) & ~NOW_IS_OOB);
6755
Hayes Wang5f8bcce2012-07-10 08:47:05 +02006756 data = r8168_mac_ocp_read(tp, 0xe8de);
Hayes Wangc5583862012-07-02 17:23:22 +08006757 data &= ~(1 << 14);
6758 r8168_mac_ocp_write(tp, 0xe8de, data);
6759
6760 if (!rtl_udelay_loop_wait_high(tp, &rtl_link_list_ready_cond, 100, 42))
6761 return;
6762
Hayes Wang5f8bcce2012-07-10 08:47:05 +02006763 data = r8168_mac_ocp_read(tp, 0xe8de);
Hayes Wangc5583862012-07-02 17:23:22 +08006764 data |= (1 << 15);
6765 r8168_mac_ocp_write(tp, 0xe8de, data);
6766
6767 if (!rtl_udelay_loop_wait_high(tp, &rtl_link_list_ready_cond, 100, 42))
6768 return;
6769}
6770
6771static void __devinit rtl_hw_initialize(struct rtl8169_private *tp)
6772{
6773 switch (tp->mac_version) {
6774 case RTL_GIGA_MAC_VER_40:
6775 case RTL_GIGA_MAC_VER_41:
6776 rtl_hw_init_8168g(tp);
6777 break;
6778
6779 default:
6780 break;
6781 }
6782}
6783
Francois Romieu3b6cf252012-03-08 09:59:04 +01006784static int __devinit
6785rtl_init_one(struct pci_dev *pdev, const struct pci_device_id *ent)
6786{
6787 const struct rtl_cfg_info *cfg = rtl_cfg_infos + ent->driver_data;
6788 const unsigned int region = cfg->region;
6789 struct rtl8169_private *tp;
6790 struct mii_if_info *mii;
6791 struct net_device *dev;
6792 void __iomem *ioaddr;
6793 int chipset, i;
6794 int rc;
6795
6796 if (netif_msg_drv(&debug)) {
6797 printk(KERN_INFO "%s Gigabit Ethernet driver %s loaded\n",
6798 MODULENAME, RTL8169_VERSION);
6799 }
6800
6801 dev = alloc_etherdev(sizeof (*tp));
6802 if (!dev) {
6803 rc = -ENOMEM;
6804 goto out;
6805 }
6806
6807 SET_NETDEV_DEV(dev, &pdev->dev);
Francois Romieufa9c3852012-03-08 10:01:50 +01006808 dev->netdev_ops = &rtl_netdev_ops;
Francois Romieu3b6cf252012-03-08 09:59:04 +01006809 tp = netdev_priv(dev);
6810 tp->dev = dev;
6811 tp->pci_dev = pdev;
6812 tp->msg_enable = netif_msg_init(debug.msg_enable, R8169_MSG_DEFAULT);
6813
6814 mii = &tp->mii;
6815 mii->dev = dev;
6816 mii->mdio_read = rtl_mdio_read;
6817 mii->mdio_write = rtl_mdio_write;
6818 mii->phy_id_mask = 0x1f;
6819 mii->reg_num_mask = 0x1f;
6820 mii->supports_gmii = !!(cfg->features & RTL_FEATURE_GMII);
6821
6822 /* disable ASPM completely as that cause random device stop working
6823 * problems as well as full system hangs for some PCIe devices users */
6824 pci_disable_link_state(pdev, PCIE_LINK_STATE_L0S | PCIE_LINK_STATE_L1 |
6825 PCIE_LINK_STATE_CLKPM);
6826
6827 /* enable device (incl. PCI PM wakeup and hotplug setup) */
6828 rc = pci_enable_device(pdev);
6829 if (rc < 0) {
6830 netif_err(tp, probe, dev, "enable failure\n");
6831 goto err_out_free_dev_1;
6832 }
6833
6834 if (pci_set_mwi(pdev) < 0)
6835 netif_info(tp, probe, dev, "Mem-Wr-Inval unavailable\n");
6836
6837 /* make sure PCI base addr 1 is MMIO */
6838 if (!(pci_resource_flags(pdev, region) & IORESOURCE_MEM)) {
6839 netif_err(tp, probe, dev,
6840 "region #%d not an MMIO resource, aborting\n",
6841 region);
6842 rc = -ENODEV;
6843 goto err_out_mwi_2;
6844 }
6845
6846 /* check for weird/broken PCI region reporting */
6847 if (pci_resource_len(pdev, region) < R8169_REGS_SIZE) {
6848 netif_err(tp, probe, dev,
6849 "Invalid PCI region size(s), aborting\n");
6850 rc = -ENODEV;
6851 goto err_out_mwi_2;
6852 }
6853
6854 rc = pci_request_regions(pdev, MODULENAME);
6855 if (rc < 0) {
6856 netif_err(tp, probe, dev, "could not request regions\n");
6857 goto err_out_mwi_2;
6858 }
6859
6860 tp->cp_cmd = RxChkSum;
6861
6862 if ((sizeof(dma_addr_t) > 4) &&
6863 !pci_set_dma_mask(pdev, DMA_BIT_MASK(64)) && use_dac) {
6864 tp->cp_cmd |= PCIDAC;
6865 dev->features |= NETIF_F_HIGHDMA;
6866 } else {
6867 rc = pci_set_dma_mask(pdev, DMA_BIT_MASK(32));
6868 if (rc < 0) {
6869 netif_err(tp, probe, dev, "DMA configuration failed\n");
6870 goto err_out_free_res_3;
6871 }
6872 }
6873
6874 /* ioremap MMIO region */
6875 ioaddr = ioremap(pci_resource_start(pdev, region), R8169_REGS_SIZE);
6876 if (!ioaddr) {
6877 netif_err(tp, probe, dev, "cannot remap MMIO, aborting\n");
6878 rc = -EIO;
6879 goto err_out_free_res_3;
6880 }
6881 tp->mmio_addr = ioaddr;
6882
6883 if (!pci_is_pcie(pdev))
6884 netif_info(tp, probe, dev, "not PCI Express\n");
6885
6886 /* Identify chip attached to board */
6887 rtl8169_get_mac_version(tp, dev, cfg->default_ver);
6888
6889 rtl_init_rxcfg(tp);
6890
6891 rtl_irq_disable(tp);
6892
Hayes Wangc5583862012-07-02 17:23:22 +08006893 rtl_hw_initialize(tp);
6894
Francois Romieu3b6cf252012-03-08 09:59:04 +01006895 rtl_hw_reset(tp);
6896
6897 rtl_ack_events(tp, 0xffff);
6898
6899 pci_set_master(pdev);
6900
6901 /*
6902 * Pretend we are using VLANs; This bypasses a nasty bug where
6903 * Interrupts stop flowing on high load on 8110SCd controllers.
6904 */
6905 if (tp->mac_version == RTL_GIGA_MAC_VER_05)
6906 tp->cp_cmd |= RxVlan;
6907
6908 rtl_init_mdio_ops(tp);
6909 rtl_init_pll_power_ops(tp);
6910 rtl_init_jumbo_ops(tp);
Hayes Wangbeb1fe12012-03-30 14:33:01 +08006911 rtl_init_csi_ops(tp);
Francois Romieu3b6cf252012-03-08 09:59:04 +01006912
6913 rtl8169_print_mac_version(tp);
6914
6915 chipset = tp->mac_version;
6916 tp->txd_version = rtl_chip_infos[chipset].txd_version;
6917
6918 RTL_W8(Cfg9346, Cfg9346_Unlock);
6919 RTL_W8(Config1, RTL_R8(Config1) | PMEnable);
6920 RTL_W8(Config5, RTL_R8(Config5) & PMEStatus);
6921 if ((RTL_R8(Config3) & (LinkUp | MagicPacket)) != 0)
6922 tp->features |= RTL_FEATURE_WOL;
6923 if ((RTL_R8(Config5) & (UWF | BWF | MWF)) != 0)
6924 tp->features |= RTL_FEATURE_WOL;
6925 tp->features |= rtl_try_msi(tp, cfg);
6926 RTL_W8(Cfg9346, Cfg9346_Lock);
6927
6928 if (rtl_tbi_enabled(tp)) {
6929 tp->set_speed = rtl8169_set_speed_tbi;
6930 tp->get_settings = rtl8169_gset_tbi;
6931 tp->phy_reset_enable = rtl8169_tbi_reset_enable;
6932 tp->phy_reset_pending = rtl8169_tbi_reset_pending;
6933 tp->link_ok = rtl8169_tbi_link_ok;
6934 tp->do_ioctl = rtl_tbi_ioctl;
6935 } else {
6936 tp->set_speed = rtl8169_set_speed_xmii;
6937 tp->get_settings = rtl8169_gset_xmii;
6938 tp->phy_reset_enable = rtl8169_xmii_reset_enable;
6939 tp->phy_reset_pending = rtl8169_xmii_reset_pending;
6940 tp->link_ok = rtl8169_xmii_link_ok;
6941 tp->do_ioctl = rtl_xmii_ioctl;
6942 }
6943
6944 mutex_init(&tp->wk.mutex);
6945
6946 /* Get MAC address */
6947 for (i = 0; i < ETH_ALEN; i++)
6948 dev->dev_addr[i] = RTL_R8(MAC0 + i);
6949 memcpy(dev->perm_addr, dev->dev_addr, dev->addr_len);
6950
6951 SET_ETHTOOL_OPS(dev, &rtl8169_ethtool_ops);
6952 dev->watchdog_timeo = RTL8169_TX_TIMEOUT;
Francois Romieu3b6cf252012-03-08 09:59:04 +01006953
6954 netif_napi_add(dev, &tp->napi, rtl8169_poll, R8169_NAPI_WEIGHT);
6955
6956 /* don't enable SG, IP_CSUM and TSO by default - it might not work
6957 * properly for all devices */
6958 dev->features |= NETIF_F_RXCSUM |
6959 NETIF_F_HW_VLAN_TX | NETIF_F_HW_VLAN_RX;
6960
6961 dev->hw_features = NETIF_F_SG | NETIF_F_IP_CSUM | NETIF_F_TSO |
6962 NETIF_F_RXCSUM | NETIF_F_HW_VLAN_TX | NETIF_F_HW_VLAN_RX;
6963 dev->vlan_features = NETIF_F_SG | NETIF_F_IP_CSUM | NETIF_F_TSO |
6964 NETIF_F_HIGHDMA;
6965
6966 if (tp->mac_version == RTL_GIGA_MAC_VER_05)
6967 /* 8110SCd requires hardware Rx VLAN - disallow toggling */
6968 dev->hw_features &= ~NETIF_F_HW_VLAN_RX;
6969
6970 dev->hw_features |= NETIF_F_RXALL;
6971 dev->hw_features |= NETIF_F_RXFCS;
6972
6973 tp->hw_start = cfg->hw_start;
6974 tp->event_slow = cfg->event_slow;
6975
6976 tp->opts1_mask = (tp->mac_version != RTL_GIGA_MAC_VER_01) ?
6977 ~(RxBOVF | RxFOVF) : ~0;
6978
6979 init_timer(&tp->timer);
6980 tp->timer.data = (unsigned long) dev;
6981 tp->timer.function = rtl8169_phy_timer;
6982
6983 tp->rtl_fw = RTL_FIRMWARE_UNKNOWN;
6984
6985 rc = register_netdev(dev);
6986 if (rc < 0)
6987 goto err_out_msi_4;
6988
6989 pci_set_drvdata(pdev, dev);
6990
Francois Romieu92a7c4e2012-03-10 10:42:12 +01006991 netif_info(tp, probe, dev, "%s at 0x%p, %pM, XID %08x IRQ %d\n",
6992 rtl_chip_infos[chipset].name, ioaddr, dev->dev_addr,
6993 (u32)(RTL_R32(TxConfig) & 0x9cf0f8ff), pdev->irq);
Francois Romieu3b6cf252012-03-08 09:59:04 +01006994 if (rtl_chip_infos[chipset].jumbo_max != JUMBO_1K) {
6995 netif_info(tp, probe, dev, "jumbo features [frames: %d bytes, "
6996 "tx checksumming: %s]\n",
6997 rtl_chip_infos[chipset].jumbo_max,
6998 rtl_chip_infos[chipset].jumbo_tx_csum ? "ok" : "ko");
6999 }
7000
7001 if (tp->mac_version == RTL_GIGA_MAC_VER_27 ||
7002 tp->mac_version == RTL_GIGA_MAC_VER_28 ||
7003 tp->mac_version == RTL_GIGA_MAC_VER_31) {
7004 rtl8168_driver_start(tp);
7005 }
7006
7007 device_set_wakeup_enable(&pdev->dev, tp->features & RTL_FEATURE_WOL);
7008
7009 if (pci_dev_run_wake(pdev))
7010 pm_runtime_put_noidle(&pdev->dev);
7011
7012 netif_carrier_off(dev);
7013
7014out:
7015 return rc;
7016
7017err_out_msi_4:
Devendra Nagaad1be8d2012-05-31 01:51:20 +00007018 netif_napi_del(&tp->napi);
Francois Romieu3b6cf252012-03-08 09:59:04 +01007019 rtl_disable_msi(pdev, tp);
7020 iounmap(ioaddr);
7021err_out_free_res_3:
7022 pci_release_regions(pdev);
7023err_out_mwi_2:
7024 pci_clear_mwi(pdev);
7025 pci_disable_device(pdev);
7026err_out_free_dev_1:
7027 free_netdev(dev);
7028 goto out;
7029}
7030
Linus Torvalds1da177e2005-04-16 15:20:36 -07007031static struct pci_driver rtl8169_pci_driver = {
7032 .name = MODULENAME,
7033 .id_table = rtl8169_pci_tbl,
Francois Romieu3b6cf252012-03-08 09:59:04 +01007034 .probe = rtl_init_one,
Francois Romieue27566e2012-03-08 09:54:01 +01007035 .remove = __devexit_p(rtl_remove_one),
Francois Romieu1765f952008-09-13 17:21:40 +02007036 .shutdown = rtl_shutdown,
Rafael J. Wysocki861ab442009-04-05 08:40:04 +00007037 .driver.pm = RTL8169_PM_OPS,
Linus Torvalds1da177e2005-04-16 15:20:36 -07007038};
7039
Francois Romieu07d3f512007-02-21 22:40:46 +01007040static int __init rtl8169_init_module(void)
Linus Torvalds1da177e2005-04-16 15:20:36 -07007041{
Jeff Garzik29917622006-08-19 17:48:59 -04007042 return pci_register_driver(&rtl8169_pci_driver);
Linus Torvalds1da177e2005-04-16 15:20:36 -07007043}
7044
Francois Romieu07d3f512007-02-21 22:40:46 +01007045static void __exit rtl8169_cleanup_module(void)
Linus Torvalds1da177e2005-04-16 15:20:36 -07007046{
7047 pci_unregister_driver(&rtl8169_pci_driver);
7048}
7049
7050module_init(rtl8169_init_module);
7051module_exit(rtl8169_cleanup_module);