xref: /openbmc/linux/drivers/net/ethernet/dec/tulip/eeprom.c (revision 4f2c0a4acffbec01079c28f839422e64ddeff004)
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