1a88394cfSJeff Kirsher /*
23396c782SPaul Gortmaker drivers/net/ethernet/dec/tulip/eeprom.c
3a88394cfSJeff Kirsher
4a88394cfSJeff Kirsher Copyright 2000,2001 The Linux Kernel Team
5a88394cfSJeff Kirsher Written/copyright 1994-2001 by Donald Becker.
6a88394cfSJeff Kirsher
7a88394cfSJeff Kirsher This software may be used and distributed according to the terms
8a88394cfSJeff Kirsher of the GNU General Public License, incorporated herein by reference.
9a88394cfSJeff Kirsher
10a88394cfSJeff Kirsher Please submit bug reports to http://bugzilla.kernel.org/.
11a88394cfSJeff Kirsher */
12a88394cfSJeff Kirsher
13a88394cfSJeff Kirsher #include <linux/pci.h>
14a88394cfSJeff Kirsher #include <linux/slab.h>
15a88394cfSJeff Kirsher #include "tulip.h"
16a88394cfSJeff Kirsher #include <asm/unaligned.h>
17a88394cfSJeff Kirsher
18a88394cfSJeff Kirsher
19a88394cfSJeff Kirsher
20a88394cfSJeff Kirsher /* Serial EEPROM section. */
21a88394cfSJeff Kirsher /* The main routine to parse the very complicated SROM structure.
22a88394cfSJeff Kirsher Search www.digital.com for "21X4 SROM" to get details.
23a88394cfSJeff Kirsher This code is very complex, and will require changes to support
24a88394cfSJeff Kirsher additional cards, so I'll be verbose about what is going on.
25a88394cfSJeff Kirsher */
26a88394cfSJeff Kirsher
27a88394cfSJeff Kirsher /* Known cards that have old-style EEPROMs. */
28779c1a85SBill Pemberton static struct eeprom_fixup eeprom_fixups[] = {
29a88394cfSJeff Kirsher {"Asante", 0, 0, 0x94, {0x1e00, 0x0000, 0x0800, 0x0100, 0x018c,
30a88394cfSJeff Kirsher 0x0000, 0x0000, 0xe078, 0x0001, 0x0050, 0x0018 }},
31a88394cfSJeff Kirsher {"SMC9332DST", 0, 0, 0xC0, { 0x1e00, 0x0000, 0x0800, 0x041f,
32a88394cfSJeff Kirsher 0x0000, 0x009E, /* 10baseT */
33a88394cfSJeff Kirsher 0x0004, 0x009E, /* 10baseT-FD */
34a88394cfSJeff Kirsher 0x0903, 0x006D, /* 100baseTx */
35a88394cfSJeff Kirsher 0x0905, 0x006D, /* 100baseTx-FD */ }},
36a88394cfSJeff Kirsher {"Cogent EM100", 0, 0, 0x92, { 0x1e00, 0x0000, 0x0800, 0x063f,
37a88394cfSJeff Kirsher 0x0107, 0x8021, /* 100baseFx */
38a88394cfSJeff Kirsher 0x0108, 0x8021, /* 100baseFx-FD */
39a88394cfSJeff Kirsher 0x0100, 0x009E, /* 10baseT */
40a88394cfSJeff Kirsher 0x0104, 0x009E, /* 10baseT-FD */
41a88394cfSJeff Kirsher 0x0103, 0x006D, /* 100baseTx */
42a88394cfSJeff Kirsher 0x0105, 0x006D, /* 100baseTx-FD */ }},
43a88394cfSJeff Kirsher {"Maxtech NX-110", 0, 0, 0xE8, { 0x1e00, 0x0000, 0x0800, 0x0513,
44a88394cfSJeff Kirsher 0x1001, 0x009E, /* 10base2, CSR12 0x10*/
45a88394cfSJeff Kirsher 0x0000, 0x009E, /* 10baseT */
46a88394cfSJeff Kirsher 0x0004, 0x009E, /* 10baseT-FD */
47a88394cfSJeff Kirsher 0x0303, 0x006D, /* 100baseTx, CSR12 0x03 */
48a88394cfSJeff Kirsher 0x0305, 0x006D, /* 100baseTx-FD CSR12 0x03 */}},
49a88394cfSJeff Kirsher {"Accton EN1207", 0, 0, 0xE8, { 0x1e00, 0x0000, 0x0800, 0x051F,
50a88394cfSJeff Kirsher 0x1B01, 0x0000, /* 10base2, CSR12 0x1B */
51a88394cfSJeff Kirsher 0x0B00, 0x009E, /* 10baseT, CSR12 0x0B */
52a88394cfSJeff Kirsher 0x0B04, 0x009E, /* 10baseT-FD,CSR12 0x0B */
53a88394cfSJeff Kirsher 0x1B03, 0x006D, /* 100baseTx, CSR12 0x1B */
54a88394cfSJeff Kirsher 0x1B05, 0x006D, /* 100baseTx-FD CSR12 0x1B */
55a88394cfSJeff Kirsher }},
56a88394cfSJeff Kirsher {"NetWinder", 0x00, 0x10, 0x57,
57a88394cfSJeff Kirsher /* Default media = MII
58a88394cfSJeff Kirsher * MII block, reset sequence (3) = 0x0821 0x0000 0x0001, capabilities 0x01e1
59a88394cfSJeff Kirsher */
60a88394cfSJeff Kirsher { 0x1e00, 0x0000, 0x000b, 0x8f01, 0x0103, 0x0300, 0x0821, 0x000, 0x0001, 0x0000, 0x01e1 }
61a88394cfSJeff Kirsher },
62a88394cfSJeff Kirsher {"Cobalt Microserver", 0, 0x10, 0xE0, {0x1e00, /* 0 == controller #, 1e == offset */
63a88394cfSJeff Kirsher 0x0000, /* 0 == high offset, 0 == gap */
64a88394cfSJeff Kirsher 0x0800, /* Default Autoselect */
65a88394cfSJeff Kirsher 0x8001, /* 1 leaf, extended type, bogus len */
66a88394cfSJeff Kirsher 0x0003, /* Type 3 (MII), PHY #0 */
67a88394cfSJeff Kirsher 0x0400, /* 0 init instr, 4 reset instr */
68a88394cfSJeff Kirsher 0x0801, /* Set control mode, GP0 output */
69a88394cfSJeff Kirsher 0x0000, /* Drive GP0 Low (RST is active low) */
70a88394cfSJeff Kirsher 0x0800, /* control mode, GP0 input (undriven) */
71a88394cfSJeff Kirsher 0x0000, /* clear control mode */
72a88394cfSJeff Kirsher 0x7800, /* 100TX FDX + HDX, 10bT FDX + HDX */
73a88394cfSJeff Kirsher 0x01e0, /* Advertise all above */
74a88394cfSJeff Kirsher 0x5000, /* FDX all above */
75a88394cfSJeff Kirsher 0x1800, /* Set fast TTM in 100bt modes */
76a88394cfSJeff Kirsher 0x0000, /* PHY cannot be unplugged */
77a88394cfSJeff Kirsher }},
78a88394cfSJeff Kirsher {NULL}};
79a88394cfSJeff Kirsher
80a88394cfSJeff Kirsher
81779c1a85SBill Pemberton static const char *const block_name[] = {
82a88394cfSJeff Kirsher "21140 non-MII",
83a88394cfSJeff Kirsher "21140 MII PHY",
84a88394cfSJeff Kirsher "21142 Serial PHY",
85a88394cfSJeff Kirsher "21142 MII PHY",
86a88394cfSJeff Kirsher "21143 SYM PHY",
87a88394cfSJeff Kirsher "21143 reset method"
88a88394cfSJeff Kirsher };
89a88394cfSJeff Kirsher
90a88394cfSJeff Kirsher
91a88394cfSJeff Kirsher /**
92a88394cfSJeff Kirsher * tulip_build_fake_mediatable - Build a fake mediatable entry.
93a88394cfSJeff Kirsher * @tp: Ptr to the tulip private data.
94a88394cfSJeff Kirsher *
95a88394cfSJeff Kirsher * Some cards like the 3x5 HSC cards (J3514A) do not have a standard
96a88394cfSJeff Kirsher * srom and can not be handled under the fixup routine. These cards
97a88394cfSJeff Kirsher * still need a valid mediatable entry for correct csr12 setup and
98a88394cfSJeff Kirsher * mii handling.
99a88394cfSJeff Kirsher *
100a88394cfSJeff Kirsher * Since this is currently a parisc-linux specific function, the
101a88394cfSJeff Kirsher * #ifdef __hppa__ should completely optimize this function away for
102a88394cfSJeff Kirsher * non-parisc hardware.
103a88394cfSJeff Kirsher */
tulip_build_fake_mediatable(struct tulip_private * tp)104779c1a85SBill Pemberton static void tulip_build_fake_mediatable(struct tulip_private *tp)
105a88394cfSJeff Kirsher {
106a88394cfSJeff Kirsher #ifdef CONFIG_GSC
107a88394cfSJeff Kirsher if (tp->flags & NEEDS_FAKE_MEDIA_TABLE) {
108a88394cfSJeff Kirsher static unsigned char leafdata[] =
109a88394cfSJeff Kirsher { 0x01, /* phy number */
110a88394cfSJeff Kirsher 0x02, /* gpr setup sequence length */
111a88394cfSJeff Kirsher 0x02, 0x00, /* gpr setup sequence */
112a88394cfSJeff Kirsher 0x02, /* phy reset sequence length */
113a88394cfSJeff Kirsher 0x01, 0x00, /* phy reset sequence */
114a88394cfSJeff Kirsher 0x00, 0x78, /* media capabilities */
115a88394cfSJeff Kirsher 0x00, 0xe0, /* nway advertisement */
116a88394cfSJeff Kirsher 0x00, 0x05, /* fdx bit map */
117a88394cfSJeff Kirsher 0x00, 0x06 /* ttm bit map */
118a88394cfSJeff Kirsher };
119a88394cfSJeff Kirsher
120*dc2df00aSRolf Eike Beer tp->mtable = devm_kmalloc(&tp->pdev->dev, sizeof(struct mediatable) +
121a88394cfSJeff Kirsher sizeof(struct medialeaf), GFP_KERNEL);
122a88394cfSJeff Kirsher
123a88394cfSJeff Kirsher if (tp->mtable == NULL)
124a88394cfSJeff Kirsher return; /* Horrible, impossible failure. */
125a88394cfSJeff Kirsher
126a88394cfSJeff Kirsher tp->mtable->defaultmedia = 0x800;
127a88394cfSJeff Kirsher tp->mtable->leafcount = 1;
128a88394cfSJeff Kirsher tp->mtable->csr12dir = 0x3f; /* inputs on bit7 for hsc-pci, bit6 for pci-fx */
129a88394cfSJeff Kirsher tp->mtable->has_nonmii = 0;
130a88394cfSJeff Kirsher tp->mtable->has_reset = 0;
131a88394cfSJeff Kirsher tp->mtable->has_mii = 1;
132a88394cfSJeff Kirsher tp->mtable->csr15dir = tp->mtable->csr15val = 0;
133a88394cfSJeff Kirsher tp->mtable->mleaf[0].type = 1;
134a88394cfSJeff Kirsher tp->mtable->mleaf[0].media = 11;
135a88394cfSJeff Kirsher tp->mtable->mleaf[0].leafdata = &leafdata[0];
136a88394cfSJeff Kirsher tp->flags |= HAS_PHY_IRQ;
137a88394cfSJeff Kirsher tp->csr12_shadow = -1;
138a88394cfSJeff Kirsher }
139a88394cfSJeff Kirsher #endif
140a88394cfSJeff Kirsher }
141a88394cfSJeff Kirsher
tulip_parse_eeprom(struct net_device * dev)142779c1a85SBill Pemberton void tulip_parse_eeprom(struct net_device *dev)
143a88394cfSJeff Kirsher {
144a88394cfSJeff Kirsher /*
145a88394cfSJeff Kirsher dev is not registered at this point, so logging messages can't
146a88394cfSJeff Kirsher use dev_<level> or netdev_<level> but dev->name is good via a
147a88394cfSJeff Kirsher hack in the caller
148a88394cfSJeff Kirsher */
149a88394cfSJeff Kirsher
150a88394cfSJeff Kirsher /* The last media info list parsed, for multiport boards. */
151a88394cfSJeff Kirsher static struct mediatable *last_mediatable;
152a88394cfSJeff Kirsher static unsigned char *last_ee_data;
153a88394cfSJeff Kirsher static int controller_index;
154a88394cfSJeff Kirsher struct tulip_private *tp = netdev_priv(dev);
155a88394cfSJeff Kirsher unsigned char *ee_data = tp->eeprom;
156a88394cfSJeff Kirsher int i;
157a88394cfSJeff Kirsher
158a88394cfSJeff Kirsher tp->mtable = NULL;
159a88394cfSJeff Kirsher /* Detect an old-style (SA only) EEPROM layout:
160a88394cfSJeff Kirsher memcmp(eedata, eedata+16, 8). */
161a88394cfSJeff Kirsher for (i = 0; i < 8; i ++)
162a88394cfSJeff Kirsher if (ee_data[i] != ee_data[16+i])
163a88394cfSJeff Kirsher break;
164a88394cfSJeff Kirsher if (i >= 8) {
165a88394cfSJeff Kirsher if (ee_data[0] == 0xff) {
166a88394cfSJeff Kirsher if (last_mediatable) {
167a88394cfSJeff Kirsher controller_index++;
168a88394cfSJeff Kirsher pr_info("%s: Controller %d of multiport board\n",
169a88394cfSJeff Kirsher dev->name, controller_index);
170a88394cfSJeff Kirsher tp->mtable = last_mediatable;
171a88394cfSJeff Kirsher ee_data = last_ee_data;
172a88394cfSJeff Kirsher goto subsequent_board;
173a88394cfSJeff Kirsher } else
174a88394cfSJeff Kirsher pr_info("%s: Missing EEPROM, this interface may not work correctly!\n",
175a88394cfSJeff Kirsher dev->name);
176a88394cfSJeff Kirsher return;
177a88394cfSJeff Kirsher }
178a88394cfSJeff Kirsher /* Do a fix-up based on the vendor half of the station address prefix. */
179a88394cfSJeff Kirsher for (i = 0; eeprom_fixups[i].name; i++) {
180a88394cfSJeff Kirsher if (dev->dev_addr[0] == eeprom_fixups[i].addr0 &&
181a88394cfSJeff Kirsher dev->dev_addr[1] == eeprom_fixups[i].addr1 &&
182a88394cfSJeff Kirsher dev->dev_addr[2] == eeprom_fixups[i].addr2) {
183a88394cfSJeff Kirsher if (dev->dev_addr[2] == 0xE8 && ee_data[0x1a] == 0x55)
184a88394cfSJeff Kirsher i++; /* An Accton EN1207, not an outlaw Maxtech. */
185a88394cfSJeff Kirsher memcpy(ee_data + 26, eeprom_fixups[i].newtable,
186a88394cfSJeff Kirsher sizeof(eeprom_fixups[i].newtable));
187a88394cfSJeff Kirsher pr_info("%s: Old format EEPROM on '%s' board. Using substitute media control info\n",
188a88394cfSJeff Kirsher dev->name, eeprom_fixups[i].name);
189a88394cfSJeff Kirsher break;
190a88394cfSJeff Kirsher }
191a88394cfSJeff Kirsher }
192a88394cfSJeff Kirsher if (eeprom_fixups[i].name == NULL) { /* No fixup found. */
193a88394cfSJeff Kirsher pr_info("%s: Old style EEPROM with no media selection information\n",
194a88394cfSJeff Kirsher dev->name);
195a88394cfSJeff Kirsher return;
196a88394cfSJeff Kirsher }
197a88394cfSJeff Kirsher }
198a88394cfSJeff Kirsher
199a88394cfSJeff Kirsher controller_index = 0;
200a88394cfSJeff Kirsher if (ee_data[19] > 1) { /* Multiport board. */
201a88394cfSJeff Kirsher last_ee_data = ee_data;
202a88394cfSJeff Kirsher }
203a88394cfSJeff Kirsher subsequent_board:
204a88394cfSJeff Kirsher
205a88394cfSJeff Kirsher if (ee_data[27] == 0) { /* No valid media table. */
206a88394cfSJeff Kirsher tulip_build_fake_mediatable(tp);
207a88394cfSJeff Kirsher } else {
208a88394cfSJeff Kirsher unsigned char *p = (void *)ee_data + ee_data[27];
209a88394cfSJeff Kirsher unsigned char csr12dir = 0;
210a88394cfSJeff Kirsher int count, new_advertise = 0;
211a88394cfSJeff Kirsher struct mediatable *mtable;
212a88394cfSJeff Kirsher u16 media = get_u16(p);
213a88394cfSJeff Kirsher
214a88394cfSJeff Kirsher p += 2;
215a88394cfSJeff Kirsher if (tp->flags & CSR12_IN_SROM)
216a88394cfSJeff Kirsher csr12dir = *p++;
217a88394cfSJeff Kirsher count = *p++;
218a88394cfSJeff Kirsher
219a88394cfSJeff Kirsher /* there is no phy information, don't even try to build mtable */
220a88394cfSJeff Kirsher if (count == 0) {
221a88394cfSJeff Kirsher if (tulip_debug > 0)
222a88394cfSJeff Kirsher pr_warn("%s: no phy info, aborting mtable build\n",
223a88394cfSJeff Kirsher dev->name);
224a88394cfSJeff Kirsher return;
225a88394cfSJeff Kirsher }
226a88394cfSJeff Kirsher
2273daebfbeSRolf Eike Beer mtable = devm_kmalloc(&tp->pdev->dev, struct_size(mtable, mleaf, count),
2283daebfbeSRolf Eike Beer GFP_KERNEL);
229a88394cfSJeff Kirsher if (mtable == NULL)
230a88394cfSJeff Kirsher return; /* Horrible, impossible failure. */
231a88394cfSJeff Kirsher last_mediatable = tp->mtable = mtable;
232a88394cfSJeff Kirsher mtable->defaultmedia = media;
233a88394cfSJeff Kirsher mtable->leafcount = count;
234a88394cfSJeff Kirsher mtable->csr12dir = csr12dir;
235a88394cfSJeff Kirsher mtable->has_nonmii = mtable->has_mii = mtable->has_reset = 0;
236a88394cfSJeff Kirsher mtable->csr15dir = mtable->csr15val = 0;
237a88394cfSJeff Kirsher
238a88394cfSJeff Kirsher pr_info("%s: EEPROM default media type %s\n",
239a88394cfSJeff Kirsher dev->name,
240a88394cfSJeff Kirsher media & 0x0800 ? "Autosense"
241a88394cfSJeff Kirsher : medianame[media & MEDIA_MASK]);
242a88394cfSJeff Kirsher for (i = 0; i < count; i++) {
243a88394cfSJeff Kirsher struct medialeaf *leaf = &mtable->mleaf[i];
244a88394cfSJeff Kirsher
245a88394cfSJeff Kirsher if ((p[0] & 0x80) == 0) { /* 21140 Compact block. */
246a88394cfSJeff Kirsher leaf->type = 0;
247a88394cfSJeff Kirsher leaf->media = p[0] & 0x3f;
248a88394cfSJeff Kirsher leaf->leafdata = p;
249a88394cfSJeff Kirsher if ((p[2] & 0x61) == 0x01) /* Bogus, but Znyx boards do it. */
250a88394cfSJeff Kirsher mtable->has_mii = 1;
251a88394cfSJeff Kirsher p += 4;
252a88394cfSJeff Kirsher } else {
253a88394cfSJeff Kirsher leaf->type = p[1];
254a88394cfSJeff Kirsher if (p[1] == 0x05) {
255a88394cfSJeff Kirsher mtable->has_reset = i;
256a88394cfSJeff Kirsher leaf->media = p[2] & 0x0f;
257a88394cfSJeff Kirsher } else if (tp->chip_id == DM910X && p[1] == 0x80) {
258a88394cfSJeff Kirsher /* Hack to ignore Davicom delay period block */
259a88394cfSJeff Kirsher mtable->leafcount--;
260a88394cfSJeff Kirsher count--;
261a88394cfSJeff Kirsher i--;
262a88394cfSJeff Kirsher leaf->leafdata = p + 2;
263a88394cfSJeff Kirsher p += (p[0] & 0x3f) + 1;
264a88394cfSJeff Kirsher continue;
265a88394cfSJeff Kirsher } else if (p[1] & 1) {
266a88394cfSJeff Kirsher int gpr_len, reset_len;
267a88394cfSJeff Kirsher
268a88394cfSJeff Kirsher mtable->has_mii = 1;
269a88394cfSJeff Kirsher leaf->media = 11;
270a88394cfSJeff Kirsher gpr_len=p[3]*2;
271a88394cfSJeff Kirsher reset_len=p[4+gpr_len]*2;
272a88394cfSJeff Kirsher new_advertise |= get_u16(&p[7+gpr_len+reset_len]);
273a88394cfSJeff Kirsher } else {
274a88394cfSJeff Kirsher mtable->has_nonmii = 1;
275a88394cfSJeff Kirsher leaf->media = p[2] & MEDIA_MASK;
276a88394cfSJeff Kirsher /* Davicom's media number for 100BaseTX is strange */
277a88394cfSJeff Kirsher if (tp->chip_id == DM910X && leaf->media == 1)
278a88394cfSJeff Kirsher leaf->media = 3;
279a88394cfSJeff Kirsher switch (leaf->media) {
280a88394cfSJeff Kirsher case 0: new_advertise |= 0x0020; break;
281a88394cfSJeff Kirsher case 4: new_advertise |= 0x0040; break;
282a88394cfSJeff Kirsher case 3: new_advertise |= 0x0080; break;
283a88394cfSJeff Kirsher case 5: new_advertise |= 0x0100; break;
284a88394cfSJeff Kirsher case 6: new_advertise |= 0x0200; break;
285a88394cfSJeff Kirsher }
286a88394cfSJeff Kirsher if (p[1] == 2 && leaf->media == 0) {
287a88394cfSJeff Kirsher if (p[2] & 0x40) {
288a88394cfSJeff Kirsher u32 base15 = get_unaligned((u16*)&p[7]);
289a88394cfSJeff Kirsher mtable->csr15dir =
290a88394cfSJeff Kirsher (get_unaligned((u16*)&p[9])<<16) + base15;
291a88394cfSJeff Kirsher mtable->csr15val =
292a88394cfSJeff Kirsher (get_unaligned((u16*)&p[11])<<16) + base15;
293a88394cfSJeff Kirsher } else {
294a88394cfSJeff Kirsher mtable->csr15dir = get_unaligned((u16*)&p[3])<<16;
295a88394cfSJeff Kirsher mtable->csr15val = get_unaligned((u16*)&p[5])<<16;
296a88394cfSJeff Kirsher }
297a88394cfSJeff Kirsher }
298a88394cfSJeff Kirsher }
299a88394cfSJeff Kirsher leaf->leafdata = p + 2;
300a88394cfSJeff Kirsher p += (p[0] & 0x3f) + 1;
301a88394cfSJeff Kirsher }
302a88394cfSJeff Kirsher if (tulip_debug > 1 && leaf->media == 11) {
303a88394cfSJeff Kirsher unsigned char *bp = leaf->leafdata;
304a88394cfSJeff Kirsher pr_info("%s: MII interface PHY %d, setup/reset sequences %d/%d long, capabilities %02x %02x\n",
305a88394cfSJeff Kirsher dev->name,
306a88394cfSJeff Kirsher bp[0], bp[1], bp[2 + bp[1]*2],
307a88394cfSJeff Kirsher bp[5 + bp[2 + bp[1]*2]*2],
308a88394cfSJeff Kirsher bp[4 + bp[2 + bp[1]*2]*2]);
309a88394cfSJeff Kirsher }
310a88394cfSJeff Kirsher pr_info("%s: Index #%d - Media %s (#%d) described by a %s (%d) block\n",
311a88394cfSJeff Kirsher dev->name,
312a88394cfSJeff Kirsher i, medianame[leaf->media & 15], leaf->media,
313a88394cfSJeff Kirsher leaf->type < ARRAY_SIZE(block_name) ? block_name[leaf->type] : "<unknown>",
314a88394cfSJeff Kirsher leaf->type);
315a88394cfSJeff Kirsher }
316a88394cfSJeff Kirsher if (new_advertise)
317a88394cfSJeff Kirsher tp->sym_advertise = new_advertise;
318a88394cfSJeff Kirsher }
319a88394cfSJeff Kirsher }
320a88394cfSJeff Kirsher /* Reading a serial EEPROM is a "bit" grungy, but we work our way through:->.*/
321a88394cfSJeff Kirsher
322a88394cfSJeff Kirsher /* EEPROM_Ctrl bits. */
323a88394cfSJeff Kirsher #define EE_SHIFT_CLK 0x02 /* EEPROM shift clock. */
324a88394cfSJeff Kirsher #define EE_CS 0x01 /* EEPROM chip select. */
325a88394cfSJeff Kirsher #define EE_DATA_WRITE 0x04 /* Data from the Tulip to EEPROM. */
326a88394cfSJeff Kirsher #define EE_WRITE_0 0x01
327a88394cfSJeff Kirsher #define EE_WRITE_1 0x05
328a88394cfSJeff Kirsher #define EE_DATA_READ 0x08 /* Data from the EEPROM chip. */
329a88394cfSJeff Kirsher #define EE_ENB (0x4800 | EE_CS)
330a88394cfSJeff Kirsher
331a88394cfSJeff Kirsher /* Delay between EEPROM clock transitions.
332a88394cfSJeff Kirsher Even at 33Mhz current PCI implementations don't overrun the EEPROM clock.
333a88394cfSJeff Kirsher We add a bus turn-around to insure that this remains true. */
334a88394cfSJeff Kirsher #define eeprom_delay() ioread32(ee_addr)
335a88394cfSJeff Kirsher
336a88394cfSJeff Kirsher /* The EEPROM commands include the alway-set leading bit. */
337a88394cfSJeff Kirsher #define EE_READ_CMD (6)
338a88394cfSJeff Kirsher
339a88394cfSJeff Kirsher /* Note: this routine returns extra data bits for size detection. */
tulip_read_eeprom(struct net_device * dev,int location,int addr_len)340779c1a85SBill Pemberton int tulip_read_eeprom(struct net_device *dev, int location, int addr_len)
341a88394cfSJeff Kirsher {
342a88394cfSJeff Kirsher int i;
343a88394cfSJeff Kirsher unsigned retval = 0;
344a88394cfSJeff Kirsher struct tulip_private *tp = netdev_priv(dev);
345a88394cfSJeff Kirsher void __iomem *ee_addr = tp->base_addr + CSR9;
346a88394cfSJeff Kirsher int read_cmd = location | (EE_READ_CMD << addr_len);
347a88394cfSJeff Kirsher
348a88394cfSJeff Kirsher /* If location is past the end of what we can address, don't
349a88394cfSJeff Kirsher * read some other location (ie truncate). Just return zero.
350a88394cfSJeff Kirsher */
351a88394cfSJeff Kirsher if (location > (1 << addr_len) - 1)
352a88394cfSJeff Kirsher return 0;
353a88394cfSJeff Kirsher
354a88394cfSJeff Kirsher iowrite32(EE_ENB & ~EE_CS, ee_addr);
355a88394cfSJeff Kirsher iowrite32(EE_ENB, ee_addr);
356a88394cfSJeff Kirsher
357a88394cfSJeff Kirsher /* Shift the read command bits out. */
358a88394cfSJeff Kirsher for (i = 4 + addr_len; i >= 0; i--) {
359a88394cfSJeff Kirsher short dataval = (read_cmd & (1 << i)) ? EE_DATA_WRITE : 0;
360a88394cfSJeff Kirsher iowrite32(EE_ENB | dataval, ee_addr);
361a88394cfSJeff Kirsher eeprom_delay();
362a88394cfSJeff Kirsher iowrite32(EE_ENB | dataval | EE_SHIFT_CLK, ee_addr);
363a88394cfSJeff Kirsher eeprom_delay();
364a88394cfSJeff Kirsher retval = (retval << 1) | ((ioread32(ee_addr) & EE_DATA_READ) ? 1 : 0);
365a88394cfSJeff Kirsher }
366a88394cfSJeff Kirsher iowrite32(EE_ENB, ee_addr);
367a88394cfSJeff Kirsher eeprom_delay();
368a88394cfSJeff Kirsher
369a88394cfSJeff Kirsher for (i = 16; i > 0; i--) {
370a88394cfSJeff Kirsher iowrite32(EE_ENB | EE_SHIFT_CLK, ee_addr);
371a88394cfSJeff Kirsher eeprom_delay();
372a88394cfSJeff Kirsher retval = (retval << 1) | ((ioread32(ee_addr) & EE_DATA_READ) ? 1 : 0);
373a88394cfSJeff Kirsher iowrite32(EE_ENB, ee_addr);
374a88394cfSJeff Kirsher eeprom_delay();
375a88394cfSJeff Kirsher }
376a88394cfSJeff Kirsher
377a88394cfSJeff Kirsher /* Terminate the EEPROM access. */
378a88394cfSJeff Kirsher iowrite32(EE_ENB & ~EE_CS, ee_addr);
379a88394cfSJeff Kirsher return (tp->flags & HAS_SWAPPED_SEEPROM) ? swab16(retval) : retval;
380a88394cfSJeff Kirsher }
381a88394cfSJeff Kirsher
382