xref: /openbmc/linux/drivers/ssb/pci.c (revision a8fe58ce)
1 /*
2  * Sonics Silicon Backplane PCI-Hostbus related functions.
3  *
4  * Copyright (C) 2005-2006 Michael Buesch <m@bues.ch>
5  * Copyright (C) 2005 Martin Langer <martin-langer@gmx.de>
6  * Copyright (C) 2005 Stefano Brivio <st3@riseup.net>
7  * Copyright (C) 2005 Danny van Dyk <kugelfang@gentoo.org>
8  * Copyright (C) 2005 Andreas Jaggi <andreas.jaggi@waterwave.ch>
9  *
10  * Derived from the Broadcom 4400 device driver.
11  * Copyright (C) 2002 David S. Miller (davem@redhat.com)
12  * Fixed by Pekka Pietikainen (pp@ee.oulu.fi)
13  * Copyright (C) 2006 Broadcom Corporation.
14  *
15  * Licensed under the GNU/GPL. See COPYING for details.
16  */
17 
18 #include <linux/ssb/ssb.h>
19 #include <linux/ssb/ssb_regs.h>
20 #include <linux/slab.h>
21 #include <linux/pci.h>
22 #include <linux/delay.h>
23 
24 #include "ssb_private.h"
25 
26 
27 /* Define the following to 1 to enable a printk on each coreswitch. */
28 #define SSB_VERBOSE_PCICORESWITCH_DEBUG		0
29 
30 
31 /* Lowlevel coreswitching */
32 int ssb_pci_switch_coreidx(struct ssb_bus *bus, u8 coreidx)
33 {
34 	int err;
35 	int attempts = 0;
36 	u32 cur_core;
37 
38 	while (1) {
39 		err = pci_write_config_dword(bus->host_pci, SSB_BAR0_WIN,
40 					     (coreidx * SSB_CORE_SIZE)
41 					     + SSB_ENUM_BASE);
42 		if (err)
43 			goto error;
44 		err = pci_read_config_dword(bus->host_pci, SSB_BAR0_WIN,
45 					    &cur_core);
46 		if (err)
47 			goto error;
48 		cur_core = (cur_core - SSB_ENUM_BASE)
49 			   / SSB_CORE_SIZE;
50 		if (cur_core == coreidx)
51 			break;
52 
53 		if (attempts++ > SSB_BAR0_MAX_RETRIES)
54 			goto error;
55 		udelay(10);
56 	}
57 	return 0;
58 error:
59 	ssb_err("Failed to switch to core %u\n", coreidx);
60 	return -ENODEV;
61 }
62 
63 int ssb_pci_switch_core(struct ssb_bus *bus,
64 			struct ssb_device *dev)
65 {
66 	int err;
67 	unsigned long flags;
68 
69 #if SSB_VERBOSE_PCICORESWITCH_DEBUG
70 	ssb_info("Switching to %s core, index %d\n",
71 		 ssb_core_name(dev->id.coreid),
72 		 dev->core_index);
73 #endif
74 
75 	spin_lock_irqsave(&bus->bar_lock, flags);
76 	err = ssb_pci_switch_coreidx(bus, dev->core_index);
77 	if (!err)
78 		bus->mapped_device = dev;
79 	spin_unlock_irqrestore(&bus->bar_lock, flags);
80 
81 	return err;
82 }
83 
84 /* Enable/disable the on board crystal oscillator and/or PLL. */
85 int ssb_pci_xtal(struct ssb_bus *bus, u32 what, int turn_on)
86 {
87 	int err;
88 	u32 in, out, outenable;
89 	u16 pci_status;
90 
91 	if (bus->bustype != SSB_BUSTYPE_PCI)
92 		return 0;
93 
94 	err = pci_read_config_dword(bus->host_pci, SSB_GPIO_IN, &in);
95 	if (err)
96 		goto err_pci;
97 	err = pci_read_config_dword(bus->host_pci, SSB_GPIO_OUT, &out);
98 	if (err)
99 		goto err_pci;
100 	err = pci_read_config_dword(bus->host_pci, SSB_GPIO_OUT_ENABLE, &outenable);
101 	if (err)
102 		goto err_pci;
103 
104 	outenable |= what;
105 
106 	if (turn_on) {
107 		/* Avoid glitching the clock if GPRS is already using it.
108 		 * We can't actually read the state of the PLLPD so we infer it
109 		 * by the value of XTAL_PU which *is* readable via gpioin.
110 		 */
111 		if (!(in & SSB_GPIO_XTAL)) {
112 			if (what & SSB_GPIO_XTAL) {
113 				/* Turn the crystal on */
114 				out |= SSB_GPIO_XTAL;
115 				if (what & SSB_GPIO_PLL)
116 					out |= SSB_GPIO_PLL;
117 				err = pci_write_config_dword(bus->host_pci, SSB_GPIO_OUT, out);
118 				if (err)
119 					goto err_pci;
120 				err = pci_write_config_dword(bus->host_pci, SSB_GPIO_OUT_ENABLE,
121 							     outenable);
122 				if (err)
123 					goto err_pci;
124 				msleep(1);
125 			}
126 			if (what & SSB_GPIO_PLL) {
127 				/* Turn the PLL on */
128 				out &= ~SSB_GPIO_PLL;
129 				err = pci_write_config_dword(bus->host_pci, SSB_GPIO_OUT, out);
130 				if (err)
131 					goto err_pci;
132 				msleep(5);
133 			}
134 		}
135 
136 		err = pci_read_config_word(bus->host_pci, PCI_STATUS, &pci_status);
137 		if (err)
138 			goto err_pci;
139 		pci_status &= ~PCI_STATUS_SIG_TARGET_ABORT;
140 		err = pci_write_config_word(bus->host_pci, PCI_STATUS, pci_status);
141 		if (err)
142 			goto err_pci;
143 	} else {
144 		if (what & SSB_GPIO_XTAL) {
145 			/* Turn the crystal off */
146 			out &= ~SSB_GPIO_XTAL;
147 		}
148 		if (what & SSB_GPIO_PLL) {
149 			/* Turn the PLL off */
150 			out |= SSB_GPIO_PLL;
151 		}
152 		err = pci_write_config_dword(bus->host_pci, SSB_GPIO_OUT, out);
153 		if (err)
154 			goto err_pci;
155 		err = pci_write_config_dword(bus->host_pci, SSB_GPIO_OUT_ENABLE, outenable);
156 		if (err)
157 			goto err_pci;
158 	}
159 
160 out:
161 	return err;
162 
163 err_pci:
164 	printk(KERN_ERR PFX "Error: ssb_pci_xtal() could not access PCI config space!\n");
165 	err = -EBUSY;
166 	goto out;
167 }
168 
169 /* Get the word-offset for a SSB_SPROM_XXX define. */
170 #define SPOFF(offset)	((offset) / sizeof(u16))
171 /* Helper to extract some _offset, which is one of the SSB_SPROM_XXX defines. */
172 #define SPEX16(_outvar, _offset, _mask, _shift)	\
173 	out->_outvar = ((in[SPOFF(_offset)] & (_mask)) >> (_shift))
174 #define SPEX32(_outvar, _offset, _mask, _shift)	\
175 	out->_outvar = ((((u32)in[SPOFF((_offset)+2)] << 16 | \
176 			   in[SPOFF(_offset)]) & (_mask)) >> (_shift))
177 #define SPEX(_outvar, _offset, _mask, _shift) \
178 	SPEX16(_outvar, _offset, _mask, _shift)
179 
180 #define SPEX_ARRAY8(_field, _offset, _mask, _shift)	\
181 	do {	\
182 		SPEX(_field[0], _offset +  0, _mask, _shift);	\
183 		SPEX(_field[1], _offset +  2, _mask, _shift);	\
184 		SPEX(_field[2], _offset +  4, _mask, _shift);	\
185 		SPEX(_field[3], _offset +  6, _mask, _shift);	\
186 		SPEX(_field[4], _offset +  8, _mask, _shift);	\
187 		SPEX(_field[5], _offset + 10, _mask, _shift);	\
188 		SPEX(_field[6], _offset + 12, _mask, _shift);	\
189 		SPEX(_field[7], _offset + 14, _mask, _shift);	\
190 	} while (0)
191 
192 
193 static inline u8 ssb_crc8(u8 crc, u8 data)
194 {
195 	/* Polynomial:   x^8 + x^7 + x^6 + x^4 + x^2 + 1   */
196 	static const u8 t[] = {
197 		0x00, 0xF7, 0xB9, 0x4E, 0x25, 0xD2, 0x9C, 0x6B,
198 		0x4A, 0xBD, 0xF3, 0x04, 0x6F, 0x98, 0xD6, 0x21,
199 		0x94, 0x63, 0x2D, 0xDA, 0xB1, 0x46, 0x08, 0xFF,
200 		0xDE, 0x29, 0x67, 0x90, 0xFB, 0x0C, 0x42, 0xB5,
201 		0x7F, 0x88, 0xC6, 0x31, 0x5A, 0xAD, 0xE3, 0x14,
202 		0x35, 0xC2, 0x8C, 0x7B, 0x10, 0xE7, 0xA9, 0x5E,
203 		0xEB, 0x1C, 0x52, 0xA5, 0xCE, 0x39, 0x77, 0x80,
204 		0xA1, 0x56, 0x18, 0xEF, 0x84, 0x73, 0x3D, 0xCA,
205 		0xFE, 0x09, 0x47, 0xB0, 0xDB, 0x2C, 0x62, 0x95,
206 		0xB4, 0x43, 0x0D, 0xFA, 0x91, 0x66, 0x28, 0xDF,
207 		0x6A, 0x9D, 0xD3, 0x24, 0x4F, 0xB8, 0xF6, 0x01,
208 		0x20, 0xD7, 0x99, 0x6E, 0x05, 0xF2, 0xBC, 0x4B,
209 		0x81, 0x76, 0x38, 0xCF, 0xA4, 0x53, 0x1D, 0xEA,
210 		0xCB, 0x3C, 0x72, 0x85, 0xEE, 0x19, 0x57, 0xA0,
211 		0x15, 0xE2, 0xAC, 0x5B, 0x30, 0xC7, 0x89, 0x7E,
212 		0x5F, 0xA8, 0xE6, 0x11, 0x7A, 0x8D, 0xC3, 0x34,
213 		0xAB, 0x5C, 0x12, 0xE5, 0x8E, 0x79, 0x37, 0xC0,
214 		0xE1, 0x16, 0x58, 0xAF, 0xC4, 0x33, 0x7D, 0x8A,
215 		0x3F, 0xC8, 0x86, 0x71, 0x1A, 0xED, 0xA3, 0x54,
216 		0x75, 0x82, 0xCC, 0x3B, 0x50, 0xA7, 0xE9, 0x1E,
217 		0xD4, 0x23, 0x6D, 0x9A, 0xF1, 0x06, 0x48, 0xBF,
218 		0x9E, 0x69, 0x27, 0xD0, 0xBB, 0x4C, 0x02, 0xF5,
219 		0x40, 0xB7, 0xF9, 0x0E, 0x65, 0x92, 0xDC, 0x2B,
220 		0x0A, 0xFD, 0xB3, 0x44, 0x2F, 0xD8, 0x96, 0x61,
221 		0x55, 0xA2, 0xEC, 0x1B, 0x70, 0x87, 0xC9, 0x3E,
222 		0x1F, 0xE8, 0xA6, 0x51, 0x3A, 0xCD, 0x83, 0x74,
223 		0xC1, 0x36, 0x78, 0x8F, 0xE4, 0x13, 0x5D, 0xAA,
224 		0x8B, 0x7C, 0x32, 0xC5, 0xAE, 0x59, 0x17, 0xE0,
225 		0x2A, 0xDD, 0x93, 0x64, 0x0F, 0xF8, 0xB6, 0x41,
226 		0x60, 0x97, 0xD9, 0x2E, 0x45, 0xB2, 0xFC, 0x0B,
227 		0xBE, 0x49, 0x07, 0xF0, 0x9B, 0x6C, 0x22, 0xD5,
228 		0xF4, 0x03, 0x4D, 0xBA, 0xD1, 0x26, 0x68, 0x9F,
229 	};
230 	return t[crc ^ data];
231 }
232 
233 static void sprom_get_mac(char *mac, const u16 *in)
234 {
235 	int i;
236 	for (i = 0; i < 3; i++) {
237 		*mac++ = in[i] >> 8;
238 		*mac++ = in[i];
239 	}
240 }
241 
242 static u8 ssb_sprom_crc(const u16 *sprom, u16 size)
243 {
244 	int word;
245 	u8 crc = 0xFF;
246 
247 	for (word = 0; word < size - 1; word++) {
248 		crc = ssb_crc8(crc, sprom[word] & 0x00FF);
249 		crc = ssb_crc8(crc, (sprom[word] & 0xFF00) >> 8);
250 	}
251 	crc = ssb_crc8(crc, sprom[size - 1] & 0x00FF);
252 	crc ^= 0xFF;
253 
254 	return crc;
255 }
256 
257 static int sprom_check_crc(const u16 *sprom, size_t size)
258 {
259 	u8 crc;
260 	u8 expected_crc;
261 	u16 tmp;
262 
263 	crc = ssb_sprom_crc(sprom, size);
264 	tmp = sprom[size - 1] & SSB_SPROM_REVISION_CRC;
265 	expected_crc = tmp >> SSB_SPROM_REVISION_CRC_SHIFT;
266 	if (crc != expected_crc)
267 		return -EPROTO;
268 
269 	return 0;
270 }
271 
272 static int sprom_do_read(struct ssb_bus *bus, u16 *sprom)
273 {
274 	int i;
275 
276 	for (i = 0; i < bus->sprom_size; i++)
277 		sprom[i] = ioread16(bus->mmio + bus->sprom_offset + (i * 2));
278 
279 	return 0;
280 }
281 
282 static int sprom_do_write(struct ssb_bus *bus, const u16 *sprom)
283 {
284 	struct pci_dev *pdev = bus->host_pci;
285 	int i, err;
286 	u32 spromctl;
287 	u16 size = bus->sprom_size;
288 
289 	ssb_notice("Writing SPROM. Do NOT turn off the power! Please stand by...\n");
290 	err = pci_read_config_dword(pdev, SSB_SPROMCTL, &spromctl);
291 	if (err)
292 		goto err_ctlreg;
293 	spromctl |= SSB_SPROMCTL_WE;
294 	err = pci_write_config_dword(pdev, SSB_SPROMCTL, spromctl);
295 	if (err)
296 		goto err_ctlreg;
297 	ssb_notice("[ 0%%");
298 	msleep(500);
299 	for (i = 0; i < size; i++) {
300 		if (i == size / 4)
301 			ssb_cont("25%%");
302 		else if (i == size / 2)
303 			ssb_cont("50%%");
304 		else if (i == (size * 3) / 4)
305 			ssb_cont("75%%");
306 		else if (i % 2)
307 			ssb_cont(".");
308 		writew(sprom[i], bus->mmio + bus->sprom_offset + (i * 2));
309 		mmiowb();
310 		msleep(20);
311 	}
312 	err = pci_read_config_dword(pdev, SSB_SPROMCTL, &spromctl);
313 	if (err)
314 		goto err_ctlreg;
315 	spromctl &= ~SSB_SPROMCTL_WE;
316 	err = pci_write_config_dword(pdev, SSB_SPROMCTL, spromctl);
317 	if (err)
318 		goto err_ctlreg;
319 	msleep(500);
320 	ssb_cont("100%% ]\n");
321 	ssb_notice("SPROM written\n");
322 
323 	return 0;
324 err_ctlreg:
325 	ssb_err("Could not access SPROM control register.\n");
326 	return err;
327 }
328 
329 static s8 sprom_extract_antgain(u8 sprom_revision, const u16 *in, u16 offset,
330 				u16 mask, u16 shift)
331 {
332 	u16 v;
333 	u8 gain;
334 
335 	v = in[SPOFF(offset)];
336 	gain = (v & mask) >> shift;
337 	if (gain == 0xFF)
338 		gain = 2; /* If unset use 2dBm */
339 	if (sprom_revision == 1) {
340 		/* Convert to Q5.2 */
341 		gain <<= 2;
342 	} else {
343 		/* Q5.2 Fractional part is stored in 0xC0 */
344 		gain = ((gain & 0xC0) >> 6) | ((gain & 0x3F) << 2);
345 	}
346 
347 	return (s8)gain;
348 }
349 
350 static void sprom_extract_r23(struct ssb_sprom *out, const u16 *in)
351 {
352 	SPEX(boardflags_hi, SSB_SPROM2_BFLHI, 0xFFFF, 0);
353 	SPEX(opo, SSB_SPROM2_OPO, SSB_SPROM2_OPO_VALUE, 0);
354 	SPEX(pa1lob0, SSB_SPROM2_PA1LOB0, 0xFFFF, 0);
355 	SPEX(pa1lob1, SSB_SPROM2_PA1LOB1, 0xFFFF, 0);
356 	SPEX(pa1lob2, SSB_SPROM2_PA1LOB2, 0xFFFF, 0);
357 	SPEX(pa1hib0, SSB_SPROM2_PA1HIB0, 0xFFFF, 0);
358 	SPEX(pa1hib1, SSB_SPROM2_PA1HIB1, 0xFFFF, 0);
359 	SPEX(pa1hib2, SSB_SPROM2_PA1HIB2, 0xFFFF, 0);
360 	SPEX(maxpwr_ah, SSB_SPROM2_MAXP_A, SSB_SPROM2_MAXP_A_HI, 0);
361 	SPEX(maxpwr_al, SSB_SPROM2_MAXP_A, SSB_SPROM2_MAXP_A_LO,
362 	     SSB_SPROM2_MAXP_A_LO_SHIFT);
363 }
364 
365 static void sprom_extract_r123(struct ssb_sprom *out, const u16 *in)
366 {
367 	u16 loc[3];
368 
369 	if (out->revision == 3)			/* rev 3 moved MAC */
370 		loc[0] = SSB_SPROM3_IL0MAC;
371 	else {
372 		loc[0] = SSB_SPROM1_IL0MAC;
373 		loc[1] = SSB_SPROM1_ET0MAC;
374 		loc[2] = SSB_SPROM1_ET1MAC;
375 	}
376 	sprom_get_mac(out->il0mac, &in[SPOFF(loc[0])]);
377 	if (out->revision < 3) { 	/* only rev 1-2 have et0, et1 */
378 		sprom_get_mac(out->et0mac, &in[SPOFF(loc[1])]);
379 		sprom_get_mac(out->et1mac, &in[SPOFF(loc[2])]);
380 	}
381 	SPEX(et0phyaddr, SSB_SPROM1_ETHPHY, SSB_SPROM1_ETHPHY_ET0A, 0);
382 	SPEX(et1phyaddr, SSB_SPROM1_ETHPHY, SSB_SPROM1_ETHPHY_ET1A,
383 	     SSB_SPROM1_ETHPHY_ET1A_SHIFT);
384 	SPEX(et0mdcport, SSB_SPROM1_ETHPHY, SSB_SPROM1_ETHPHY_ET0M, 14);
385 	SPEX(et1mdcport, SSB_SPROM1_ETHPHY, SSB_SPROM1_ETHPHY_ET1M, 15);
386 	SPEX(board_rev, SSB_SPROM1_BINF, SSB_SPROM1_BINF_BREV, 0);
387 	SPEX(board_type, SSB_SPROM1_SPID, 0xFFFF, 0);
388 	if (out->revision == 1)
389 		SPEX(country_code, SSB_SPROM1_BINF, SSB_SPROM1_BINF_CCODE,
390 		     SSB_SPROM1_BINF_CCODE_SHIFT);
391 	SPEX(ant_available_a, SSB_SPROM1_BINF, SSB_SPROM1_BINF_ANTA,
392 	     SSB_SPROM1_BINF_ANTA_SHIFT);
393 	SPEX(ant_available_bg, SSB_SPROM1_BINF, SSB_SPROM1_BINF_ANTBG,
394 	     SSB_SPROM1_BINF_ANTBG_SHIFT);
395 	SPEX(pa0b0, SSB_SPROM1_PA0B0, 0xFFFF, 0);
396 	SPEX(pa0b1, SSB_SPROM1_PA0B1, 0xFFFF, 0);
397 	SPEX(pa0b2, SSB_SPROM1_PA0B2, 0xFFFF, 0);
398 	SPEX(pa1b0, SSB_SPROM1_PA1B0, 0xFFFF, 0);
399 	SPEX(pa1b1, SSB_SPROM1_PA1B1, 0xFFFF, 0);
400 	SPEX(pa1b2, SSB_SPROM1_PA1B2, 0xFFFF, 0);
401 	SPEX(gpio0, SSB_SPROM1_GPIOA, SSB_SPROM1_GPIOA_P0, 0);
402 	SPEX(gpio1, SSB_SPROM1_GPIOA, SSB_SPROM1_GPIOA_P1,
403 	     SSB_SPROM1_GPIOA_P1_SHIFT);
404 	SPEX(gpio2, SSB_SPROM1_GPIOB, SSB_SPROM1_GPIOB_P2, 0);
405 	SPEX(gpio3, SSB_SPROM1_GPIOB, SSB_SPROM1_GPIOB_P3,
406 	     SSB_SPROM1_GPIOB_P3_SHIFT);
407 	SPEX(maxpwr_a, SSB_SPROM1_MAXPWR, SSB_SPROM1_MAXPWR_A,
408 	     SSB_SPROM1_MAXPWR_A_SHIFT);
409 	SPEX(maxpwr_bg, SSB_SPROM1_MAXPWR, SSB_SPROM1_MAXPWR_BG, 0);
410 	SPEX(itssi_a, SSB_SPROM1_ITSSI, SSB_SPROM1_ITSSI_A,
411 	     SSB_SPROM1_ITSSI_A_SHIFT);
412 	SPEX(itssi_bg, SSB_SPROM1_ITSSI, SSB_SPROM1_ITSSI_BG, 0);
413 	SPEX(boardflags_lo, SSB_SPROM1_BFLLO, 0xFFFF, 0);
414 
415 	SPEX(alpha2[0], SSB_SPROM1_CCODE, 0xff00, 8);
416 	SPEX(alpha2[1], SSB_SPROM1_CCODE, 0x00ff, 0);
417 
418 	/* Extract the antenna gain values. */
419 	out->antenna_gain.a0 = sprom_extract_antgain(out->revision, in,
420 						     SSB_SPROM1_AGAIN,
421 						     SSB_SPROM1_AGAIN_BG,
422 						     SSB_SPROM1_AGAIN_BG_SHIFT);
423 	out->antenna_gain.a1 = sprom_extract_antgain(out->revision, in,
424 						     SSB_SPROM1_AGAIN,
425 						     SSB_SPROM1_AGAIN_A,
426 						     SSB_SPROM1_AGAIN_A_SHIFT);
427 	if (out->revision >= 2)
428 		sprom_extract_r23(out, in);
429 }
430 
431 /* Revs 4 5 and 8 have partially shared layout */
432 static void sprom_extract_r458(struct ssb_sprom *out, const u16 *in)
433 {
434 	SPEX(txpid2g[0], SSB_SPROM4_TXPID2G01,
435 	     SSB_SPROM4_TXPID2G0, SSB_SPROM4_TXPID2G0_SHIFT);
436 	SPEX(txpid2g[1], SSB_SPROM4_TXPID2G01,
437 	     SSB_SPROM4_TXPID2G1, SSB_SPROM4_TXPID2G1_SHIFT);
438 	SPEX(txpid2g[2], SSB_SPROM4_TXPID2G23,
439 	     SSB_SPROM4_TXPID2G2, SSB_SPROM4_TXPID2G2_SHIFT);
440 	SPEX(txpid2g[3], SSB_SPROM4_TXPID2G23,
441 	     SSB_SPROM4_TXPID2G3, SSB_SPROM4_TXPID2G3_SHIFT);
442 
443 	SPEX(txpid5gl[0], SSB_SPROM4_TXPID5GL01,
444 	     SSB_SPROM4_TXPID5GL0, SSB_SPROM4_TXPID5GL0_SHIFT);
445 	SPEX(txpid5gl[1], SSB_SPROM4_TXPID5GL01,
446 	     SSB_SPROM4_TXPID5GL1, SSB_SPROM4_TXPID5GL1_SHIFT);
447 	SPEX(txpid5gl[2], SSB_SPROM4_TXPID5GL23,
448 	     SSB_SPROM4_TXPID5GL2, SSB_SPROM4_TXPID5GL2_SHIFT);
449 	SPEX(txpid5gl[3], SSB_SPROM4_TXPID5GL23,
450 	     SSB_SPROM4_TXPID5GL3, SSB_SPROM4_TXPID5GL3_SHIFT);
451 
452 	SPEX(txpid5g[0], SSB_SPROM4_TXPID5G01,
453 	     SSB_SPROM4_TXPID5G0, SSB_SPROM4_TXPID5G0_SHIFT);
454 	SPEX(txpid5g[1], SSB_SPROM4_TXPID5G01,
455 	     SSB_SPROM4_TXPID5G1, SSB_SPROM4_TXPID5G1_SHIFT);
456 	SPEX(txpid5g[2], SSB_SPROM4_TXPID5G23,
457 	     SSB_SPROM4_TXPID5G2, SSB_SPROM4_TXPID5G2_SHIFT);
458 	SPEX(txpid5g[3], SSB_SPROM4_TXPID5G23,
459 	     SSB_SPROM4_TXPID5G3, SSB_SPROM4_TXPID5G3_SHIFT);
460 
461 	SPEX(txpid5gh[0], SSB_SPROM4_TXPID5GH01,
462 	     SSB_SPROM4_TXPID5GH0, SSB_SPROM4_TXPID5GH0_SHIFT);
463 	SPEX(txpid5gh[1], SSB_SPROM4_TXPID5GH01,
464 	     SSB_SPROM4_TXPID5GH1, SSB_SPROM4_TXPID5GH1_SHIFT);
465 	SPEX(txpid5gh[2], SSB_SPROM4_TXPID5GH23,
466 	     SSB_SPROM4_TXPID5GH2, SSB_SPROM4_TXPID5GH2_SHIFT);
467 	SPEX(txpid5gh[3], SSB_SPROM4_TXPID5GH23,
468 	     SSB_SPROM4_TXPID5GH3, SSB_SPROM4_TXPID5GH3_SHIFT);
469 }
470 
471 static void sprom_extract_r45(struct ssb_sprom *out, const u16 *in)
472 {
473 	static const u16 pwr_info_offset[] = {
474 		SSB_SPROM4_PWR_INFO_CORE0, SSB_SPROM4_PWR_INFO_CORE1,
475 		SSB_SPROM4_PWR_INFO_CORE2, SSB_SPROM4_PWR_INFO_CORE3
476 	};
477 	u16 il0mac_offset;
478 	int i;
479 
480 	BUILD_BUG_ON(ARRAY_SIZE(pwr_info_offset) !=
481 		     ARRAY_SIZE(out->core_pwr_info));
482 
483 	if (out->revision == 4)
484 		il0mac_offset = SSB_SPROM4_IL0MAC;
485 	else
486 		il0mac_offset = SSB_SPROM5_IL0MAC;
487 
488 	sprom_get_mac(out->il0mac, &in[SPOFF(il0mac_offset)]);
489 
490 	SPEX(et0phyaddr, SSB_SPROM4_ETHPHY, SSB_SPROM4_ETHPHY_ET0A, 0);
491 	SPEX(et1phyaddr, SSB_SPROM4_ETHPHY, SSB_SPROM4_ETHPHY_ET1A,
492 	     SSB_SPROM4_ETHPHY_ET1A_SHIFT);
493 	SPEX(board_rev, SSB_SPROM4_BOARDREV, 0xFFFF, 0);
494 	SPEX(board_type, SSB_SPROM1_SPID, 0xFFFF, 0);
495 	if (out->revision == 4) {
496 		SPEX(alpha2[0], SSB_SPROM4_CCODE, 0xff00, 8);
497 		SPEX(alpha2[1], SSB_SPROM4_CCODE, 0x00ff, 0);
498 		SPEX(boardflags_lo, SSB_SPROM4_BFLLO, 0xFFFF, 0);
499 		SPEX(boardflags_hi, SSB_SPROM4_BFLHI, 0xFFFF, 0);
500 		SPEX(boardflags2_lo, SSB_SPROM4_BFL2LO, 0xFFFF, 0);
501 		SPEX(boardflags2_hi, SSB_SPROM4_BFL2HI, 0xFFFF, 0);
502 	} else {
503 		SPEX(alpha2[0], SSB_SPROM5_CCODE, 0xff00, 8);
504 		SPEX(alpha2[1], SSB_SPROM5_CCODE, 0x00ff, 0);
505 		SPEX(boardflags_lo, SSB_SPROM5_BFLLO, 0xFFFF, 0);
506 		SPEX(boardflags_hi, SSB_SPROM5_BFLHI, 0xFFFF, 0);
507 		SPEX(boardflags2_lo, SSB_SPROM5_BFL2LO, 0xFFFF, 0);
508 		SPEX(boardflags2_hi, SSB_SPROM5_BFL2HI, 0xFFFF, 0);
509 	}
510 	SPEX(ant_available_a, SSB_SPROM4_ANTAVAIL, SSB_SPROM4_ANTAVAIL_A,
511 	     SSB_SPROM4_ANTAVAIL_A_SHIFT);
512 	SPEX(ant_available_bg, SSB_SPROM4_ANTAVAIL, SSB_SPROM4_ANTAVAIL_BG,
513 	     SSB_SPROM4_ANTAVAIL_BG_SHIFT);
514 	SPEX(maxpwr_bg, SSB_SPROM4_MAXP_BG, SSB_SPROM4_MAXP_BG_MASK, 0);
515 	SPEX(itssi_bg, SSB_SPROM4_MAXP_BG, SSB_SPROM4_ITSSI_BG,
516 	     SSB_SPROM4_ITSSI_BG_SHIFT);
517 	SPEX(maxpwr_a, SSB_SPROM4_MAXP_A, SSB_SPROM4_MAXP_A_MASK, 0);
518 	SPEX(itssi_a, SSB_SPROM4_MAXP_A, SSB_SPROM4_ITSSI_A,
519 	     SSB_SPROM4_ITSSI_A_SHIFT);
520 	if (out->revision == 4) {
521 		SPEX(gpio0, SSB_SPROM4_GPIOA, SSB_SPROM4_GPIOA_P0, 0);
522 		SPEX(gpio1, SSB_SPROM4_GPIOA, SSB_SPROM4_GPIOA_P1,
523 		     SSB_SPROM4_GPIOA_P1_SHIFT);
524 		SPEX(gpio2, SSB_SPROM4_GPIOB, SSB_SPROM4_GPIOB_P2, 0);
525 		SPEX(gpio3, SSB_SPROM4_GPIOB, SSB_SPROM4_GPIOB_P3,
526 		     SSB_SPROM4_GPIOB_P3_SHIFT);
527 	} else {
528 		SPEX(gpio0, SSB_SPROM5_GPIOA, SSB_SPROM5_GPIOA_P0, 0);
529 		SPEX(gpio1, SSB_SPROM5_GPIOA, SSB_SPROM5_GPIOA_P1,
530 		     SSB_SPROM5_GPIOA_P1_SHIFT);
531 		SPEX(gpio2, SSB_SPROM5_GPIOB, SSB_SPROM5_GPIOB_P2, 0);
532 		SPEX(gpio3, SSB_SPROM5_GPIOB, SSB_SPROM5_GPIOB_P3,
533 		     SSB_SPROM5_GPIOB_P3_SHIFT);
534 	}
535 
536 	/* Extract the antenna gain values. */
537 	out->antenna_gain.a0 = sprom_extract_antgain(out->revision, in,
538 						     SSB_SPROM4_AGAIN01,
539 						     SSB_SPROM4_AGAIN0,
540 						     SSB_SPROM4_AGAIN0_SHIFT);
541 	out->antenna_gain.a1 = sprom_extract_antgain(out->revision, in,
542 						     SSB_SPROM4_AGAIN01,
543 						     SSB_SPROM4_AGAIN1,
544 						     SSB_SPROM4_AGAIN1_SHIFT);
545 	out->antenna_gain.a2 = sprom_extract_antgain(out->revision, in,
546 						     SSB_SPROM4_AGAIN23,
547 						     SSB_SPROM4_AGAIN2,
548 						     SSB_SPROM4_AGAIN2_SHIFT);
549 	out->antenna_gain.a3 = sprom_extract_antgain(out->revision, in,
550 						     SSB_SPROM4_AGAIN23,
551 						     SSB_SPROM4_AGAIN3,
552 						     SSB_SPROM4_AGAIN3_SHIFT);
553 
554 	/* Extract cores power info info */
555 	for (i = 0; i < ARRAY_SIZE(pwr_info_offset); i++) {
556 		u16 o = pwr_info_offset[i];
557 
558 		SPEX(core_pwr_info[i].itssi_2g, o + SSB_SPROM4_2G_MAXP_ITSSI,
559 			SSB_SPROM4_2G_ITSSI, SSB_SPROM4_2G_ITSSI_SHIFT);
560 		SPEX(core_pwr_info[i].maxpwr_2g, o + SSB_SPROM4_2G_MAXP_ITSSI,
561 			SSB_SPROM4_2G_MAXP, 0);
562 
563 		SPEX(core_pwr_info[i].pa_2g[0], o + SSB_SPROM4_2G_PA_0, ~0, 0);
564 		SPEX(core_pwr_info[i].pa_2g[1], o + SSB_SPROM4_2G_PA_1, ~0, 0);
565 		SPEX(core_pwr_info[i].pa_2g[2], o + SSB_SPROM4_2G_PA_2, ~0, 0);
566 		SPEX(core_pwr_info[i].pa_2g[3], o + SSB_SPROM4_2G_PA_3, ~0, 0);
567 
568 		SPEX(core_pwr_info[i].itssi_5g, o + SSB_SPROM4_5G_MAXP_ITSSI,
569 			SSB_SPROM4_5G_ITSSI, SSB_SPROM4_5G_ITSSI_SHIFT);
570 		SPEX(core_pwr_info[i].maxpwr_5g, o + SSB_SPROM4_5G_MAXP_ITSSI,
571 			SSB_SPROM4_5G_MAXP, 0);
572 		SPEX(core_pwr_info[i].maxpwr_5gh, o + SSB_SPROM4_5GHL_MAXP,
573 			SSB_SPROM4_5GH_MAXP, 0);
574 		SPEX(core_pwr_info[i].maxpwr_5gl, o + SSB_SPROM4_5GHL_MAXP,
575 			SSB_SPROM4_5GL_MAXP, SSB_SPROM4_5GL_MAXP_SHIFT);
576 
577 		SPEX(core_pwr_info[i].pa_5gl[0], o + SSB_SPROM4_5GL_PA_0, ~0, 0);
578 		SPEX(core_pwr_info[i].pa_5gl[1], o + SSB_SPROM4_5GL_PA_1, ~0, 0);
579 		SPEX(core_pwr_info[i].pa_5gl[2], o + SSB_SPROM4_5GL_PA_2, ~0, 0);
580 		SPEX(core_pwr_info[i].pa_5gl[3], o + SSB_SPROM4_5GL_PA_3, ~0, 0);
581 		SPEX(core_pwr_info[i].pa_5g[0], o + SSB_SPROM4_5G_PA_0, ~0, 0);
582 		SPEX(core_pwr_info[i].pa_5g[1], o + SSB_SPROM4_5G_PA_1, ~0, 0);
583 		SPEX(core_pwr_info[i].pa_5g[2], o + SSB_SPROM4_5G_PA_2, ~0, 0);
584 		SPEX(core_pwr_info[i].pa_5g[3], o + SSB_SPROM4_5G_PA_3, ~0, 0);
585 		SPEX(core_pwr_info[i].pa_5gh[0], o + SSB_SPROM4_5GH_PA_0, ~0, 0);
586 		SPEX(core_pwr_info[i].pa_5gh[1], o + SSB_SPROM4_5GH_PA_1, ~0, 0);
587 		SPEX(core_pwr_info[i].pa_5gh[2], o + SSB_SPROM4_5GH_PA_2, ~0, 0);
588 		SPEX(core_pwr_info[i].pa_5gh[3], o + SSB_SPROM4_5GH_PA_3, ~0, 0);
589 	}
590 
591 	sprom_extract_r458(out, in);
592 
593 	/* TODO - get remaining rev 4 stuff needed */
594 }
595 
596 static void sprom_extract_r8(struct ssb_sprom *out, const u16 *in)
597 {
598 	int i;
599 	u16 o;
600 	u16 pwr_info_offset[] = {
601 		SSB_SROM8_PWR_INFO_CORE0, SSB_SROM8_PWR_INFO_CORE1,
602 		SSB_SROM8_PWR_INFO_CORE2, SSB_SROM8_PWR_INFO_CORE3
603 	};
604 	BUILD_BUG_ON(ARRAY_SIZE(pwr_info_offset) !=
605 			ARRAY_SIZE(out->core_pwr_info));
606 
607 	/* extract the MAC address */
608 	sprom_get_mac(out->il0mac, &in[SPOFF(SSB_SPROM8_IL0MAC)]);
609 
610 	SPEX(board_rev, SSB_SPROM8_BOARDREV, 0xFFFF, 0);
611 	SPEX(board_type, SSB_SPROM1_SPID, 0xFFFF, 0);
612 	SPEX(alpha2[0], SSB_SPROM8_CCODE, 0xff00, 8);
613 	SPEX(alpha2[1], SSB_SPROM8_CCODE, 0x00ff, 0);
614 	SPEX(boardflags_lo, SSB_SPROM8_BFLLO, 0xFFFF, 0);
615 	SPEX(boardflags_hi, SSB_SPROM8_BFLHI, 0xFFFF, 0);
616 	SPEX(boardflags2_lo, SSB_SPROM8_BFL2LO, 0xFFFF, 0);
617 	SPEX(boardflags2_hi, SSB_SPROM8_BFL2HI, 0xFFFF, 0);
618 	SPEX(ant_available_a, SSB_SPROM8_ANTAVAIL, SSB_SPROM8_ANTAVAIL_A,
619 	     SSB_SPROM8_ANTAVAIL_A_SHIFT);
620 	SPEX(ant_available_bg, SSB_SPROM8_ANTAVAIL, SSB_SPROM8_ANTAVAIL_BG,
621 	     SSB_SPROM8_ANTAVAIL_BG_SHIFT);
622 	SPEX(maxpwr_bg, SSB_SPROM8_MAXP_BG, SSB_SPROM8_MAXP_BG_MASK, 0);
623 	SPEX(itssi_bg, SSB_SPROM8_MAXP_BG, SSB_SPROM8_ITSSI_BG,
624 	     SSB_SPROM8_ITSSI_BG_SHIFT);
625 	SPEX(maxpwr_a, SSB_SPROM8_MAXP_A, SSB_SPROM8_MAXP_A_MASK, 0);
626 	SPEX(itssi_a, SSB_SPROM8_MAXP_A, SSB_SPROM8_ITSSI_A,
627 	     SSB_SPROM8_ITSSI_A_SHIFT);
628 	SPEX(maxpwr_ah, SSB_SPROM8_MAXP_AHL, SSB_SPROM8_MAXP_AH_MASK, 0);
629 	SPEX(maxpwr_al, SSB_SPROM8_MAXP_AHL, SSB_SPROM8_MAXP_AL_MASK,
630 	     SSB_SPROM8_MAXP_AL_SHIFT);
631 	SPEX(gpio0, SSB_SPROM8_GPIOA, SSB_SPROM8_GPIOA_P0, 0);
632 	SPEX(gpio1, SSB_SPROM8_GPIOA, SSB_SPROM8_GPIOA_P1,
633 	     SSB_SPROM8_GPIOA_P1_SHIFT);
634 	SPEX(gpio2, SSB_SPROM8_GPIOB, SSB_SPROM8_GPIOB_P2, 0);
635 	SPEX(gpio3, SSB_SPROM8_GPIOB, SSB_SPROM8_GPIOB_P3,
636 	     SSB_SPROM8_GPIOB_P3_SHIFT);
637 	SPEX(tri2g, SSB_SPROM8_TRI25G, SSB_SPROM8_TRI2G, 0);
638 	SPEX(tri5g, SSB_SPROM8_TRI25G, SSB_SPROM8_TRI5G,
639 	     SSB_SPROM8_TRI5G_SHIFT);
640 	SPEX(tri5gl, SSB_SPROM8_TRI5GHL, SSB_SPROM8_TRI5GL, 0);
641 	SPEX(tri5gh, SSB_SPROM8_TRI5GHL, SSB_SPROM8_TRI5GH,
642 	     SSB_SPROM8_TRI5GH_SHIFT);
643 	SPEX(rxpo2g, SSB_SPROM8_RXPO, SSB_SPROM8_RXPO2G, 0);
644 	SPEX(rxpo5g, SSB_SPROM8_RXPO, SSB_SPROM8_RXPO5G,
645 	     SSB_SPROM8_RXPO5G_SHIFT);
646 	SPEX(rssismf2g, SSB_SPROM8_RSSIPARM2G, SSB_SPROM8_RSSISMF2G, 0);
647 	SPEX(rssismc2g, SSB_SPROM8_RSSIPARM2G, SSB_SPROM8_RSSISMC2G,
648 	     SSB_SPROM8_RSSISMC2G_SHIFT);
649 	SPEX(rssisav2g, SSB_SPROM8_RSSIPARM2G, SSB_SPROM8_RSSISAV2G,
650 	     SSB_SPROM8_RSSISAV2G_SHIFT);
651 	SPEX(bxa2g, SSB_SPROM8_RSSIPARM2G, SSB_SPROM8_BXA2G,
652 	     SSB_SPROM8_BXA2G_SHIFT);
653 	SPEX(rssismf5g, SSB_SPROM8_RSSIPARM5G, SSB_SPROM8_RSSISMF5G, 0);
654 	SPEX(rssismc5g, SSB_SPROM8_RSSIPARM5G, SSB_SPROM8_RSSISMC5G,
655 	     SSB_SPROM8_RSSISMC5G_SHIFT);
656 	SPEX(rssisav5g, SSB_SPROM8_RSSIPARM5G, SSB_SPROM8_RSSISAV5G,
657 	     SSB_SPROM8_RSSISAV5G_SHIFT);
658 	SPEX(bxa5g, SSB_SPROM8_RSSIPARM5G, SSB_SPROM8_BXA5G,
659 	     SSB_SPROM8_BXA5G_SHIFT);
660 	SPEX(pa0b0, SSB_SPROM8_PA0B0, 0xFFFF, 0);
661 	SPEX(pa0b1, SSB_SPROM8_PA0B1, 0xFFFF, 0);
662 	SPEX(pa0b2, SSB_SPROM8_PA0B2, 0xFFFF, 0);
663 	SPEX(pa1b0, SSB_SPROM8_PA1B0, 0xFFFF, 0);
664 	SPEX(pa1b1, SSB_SPROM8_PA1B1, 0xFFFF, 0);
665 	SPEX(pa1b2, SSB_SPROM8_PA1B2, 0xFFFF, 0);
666 	SPEX(pa1lob0, SSB_SPROM8_PA1LOB0, 0xFFFF, 0);
667 	SPEX(pa1lob1, SSB_SPROM8_PA1LOB1, 0xFFFF, 0);
668 	SPEX(pa1lob2, SSB_SPROM8_PA1LOB2, 0xFFFF, 0);
669 	SPEX(pa1hib0, SSB_SPROM8_PA1HIB0, 0xFFFF, 0);
670 	SPEX(pa1hib1, SSB_SPROM8_PA1HIB1, 0xFFFF, 0);
671 	SPEX(pa1hib2, SSB_SPROM8_PA1HIB2, 0xFFFF, 0);
672 	SPEX(cck2gpo, SSB_SPROM8_CCK2GPO, 0xFFFF, 0);
673 	SPEX32(ofdm2gpo, SSB_SPROM8_OFDM2GPO, 0xFFFFFFFF, 0);
674 	SPEX32(ofdm5glpo, SSB_SPROM8_OFDM5GLPO, 0xFFFFFFFF, 0);
675 	SPEX32(ofdm5gpo, SSB_SPROM8_OFDM5GPO, 0xFFFFFFFF, 0);
676 	SPEX32(ofdm5ghpo, SSB_SPROM8_OFDM5GHPO, 0xFFFFFFFF, 0);
677 
678 	/* Extract the antenna gain values. */
679 	out->antenna_gain.a0 = sprom_extract_antgain(out->revision, in,
680 						     SSB_SPROM8_AGAIN01,
681 						     SSB_SPROM8_AGAIN0,
682 						     SSB_SPROM8_AGAIN0_SHIFT);
683 	out->antenna_gain.a1 = sprom_extract_antgain(out->revision, in,
684 						     SSB_SPROM8_AGAIN01,
685 						     SSB_SPROM8_AGAIN1,
686 						     SSB_SPROM8_AGAIN1_SHIFT);
687 	out->antenna_gain.a2 = sprom_extract_antgain(out->revision, in,
688 						     SSB_SPROM8_AGAIN23,
689 						     SSB_SPROM8_AGAIN2,
690 						     SSB_SPROM8_AGAIN2_SHIFT);
691 	out->antenna_gain.a3 = sprom_extract_antgain(out->revision, in,
692 						     SSB_SPROM8_AGAIN23,
693 						     SSB_SPROM8_AGAIN3,
694 						     SSB_SPROM8_AGAIN3_SHIFT);
695 
696 	/* Extract cores power info info */
697 	for (i = 0; i < ARRAY_SIZE(pwr_info_offset); i++) {
698 		o = pwr_info_offset[i];
699 		SPEX(core_pwr_info[i].itssi_2g, o + SSB_SROM8_2G_MAXP_ITSSI,
700 			SSB_SPROM8_2G_ITSSI, SSB_SPROM8_2G_ITSSI_SHIFT);
701 		SPEX(core_pwr_info[i].maxpwr_2g, o + SSB_SROM8_2G_MAXP_ITSSI,
702 			SSB_SPROM8_2G_MAXP, 0);
703 
704 		SPEX(core_pwr_info[i].pa_2g[0], o + SSB_SROM8_2G_PA_0, ~0, 0);
705 		SPEX(core_pwr_info[i].pa_2g[1], o + SSB_SROM8_2G_PA_1, ~0, 0);
706 		SPEX(core_pwr_info[i].pa_2g[2], o + SSB_SROM8_2G_PA_2, ~0, 0);
707 
708 		SPEX(core_pwr_info[i].itssi_5g, o + SSB_SROM8_5G_MAXP_ITSSI,
709 			SSB_SPROM8_5G_ITSSI, SSB_SPROM8_5G_ITSSI_SHIFT);
710 		SPEX(core_pwr_info[i].maxpwr_5g, o + SSB_SROM8_5G_MAXP_ITSSI,
711 			SSB_SPROM8_5G_MAXP, 0);
712 		SPEX(core_pwr_info[i].maxpwr_5gh, o + SSB_SPROM8_5GHL_MAXP,
713 			SSB_SPROM8_5GH_MAXP, 0);
714 		SPEX(core_pwr_info[i].maxpwr_5gl, o + SSB_SPROM8_5GHL_MAXP,
715 			SSB_SPROM8_5GL_MAXP, SSB_SPROM8_5GL_MAXP_SHIFT);
716 
717 		SPEX(core_pwr_info[i].pa_5gl[0], o + SSB_SROM8_5GL_PA_0, ~0, 0);
718 		SPEX(core_pwr_info[i].pa_5gl[1], o + SSB_SROM8_5GL_PA_1, ~0, 0);
719 		SPEX(core_pwr_info[i].pa_5gl[2], o + SSB_SROM8_5GL_PA_2, ~0, 0);
720 		SPEX(core_pwr_info[i].pa_5g[0], o + SSB_SROM8_5G_PA_0, ~0, 0);
721 		SPEX(core_pwr_info[i].pa_5g[1], o + SSB_SROM8_5G_PA_1, ~0, 0);
722 		SPEX(core_pwr_info[i].pa_5g[2], o + SSB_SROM8_5G_PA_2, ~0, 0);
723 		SPEX(core_pwr_info[i].pa_5gh[0], o + SSB_SROM8_5GH_PA_0, ~0, 0);
724 		SPEX(core_pwr_info[i].pa_5gh[1], o + SSB_SROM8_5GH_PA_1, ~0, 0);
725 		SPEX(core_pwr_info[i].pa_5gh[2], o + SSB_SROM8_5GH_PA_2, ~0, 0);
726 	}
727 
728 	/* Extract FEM info */
729 	SPEX(fem.ghz2.tssipos, SSB_SPROM8_FEM2G,
730 		SSB_SROM8_FEM_TSSIPOS, SSB_SROM8_FEM_TSSIPOS_SHIFT);
731 	SPEX(fem.ghz2.extpa_gain, SSB_SPROM8_FEM2G,
732 		SSB_SROM8_FEM_EXTPA_GAIN, SSB_SROM8_FEM_EXTPA_GAIN_SHIFT);
733 	SPEX(fem.ghz2.pdet_range, SSB_SPROM8_FEM2G,
734 		SSB_SROM8_FEM_PDET_RANGE, SSB_SROM8_FEM_PDET_RANGE_SHIFT);
735 	SPEX(fem.ghz2.tr_iso, SSB_SPROM8_FEM2G,
736 		SSB_SROM8_FEM_TR_ISO, SSB_SROM8_FEM_TR_ISO_SHIFT);
737 	SPEX(fem.ghz2.antswlut, SSB_SPROM8_FEM2G,
738 		SSB_SROM8_FEM_ANTSWLUT, SSB_SROM8_FEM_ANTSWLUT_SHIFT);
739 
740 	SPEX(fem.ghz5.tssipos, SSB_SPROM8_FEM5G,
741 		SSB_SROM8_FEM_TSSIPOS, SSB_SROM8_FEM_TSSIPOS_SHIFT);
742 	SPEX(fem.ghz5.extpa_gain, SSB_SPROM8_FEM5G,
743 		SSB_SROM8_FEM_EXTPA_GAIN, SSB_SROM8_FEM_EXTPA_GAIN_SHIFT);
744 	SPEX(fem.ghz5.pdet_range, SSB_SPROM8_FEM5G,
745 		SSB_SROM8_FEM_PDET_RANGE, SSB_SROM8_FEM_PDET_RANGE_SHIFT);
746 	SPEX(fem.ghz5.tr_iso, SSB_SPROM8_FEM5G,
747 		SSB_SROM8_FEM_TR_ISO, SSB_SROM8_FEM_TR_ISO_SHIFT);
748 	SPEX(fem.ghz5.antswlut, SSB_SPROM8_FEM5G,
749 		SSB_SROM8_FEM_ANTSWLUT, SSB_SROM8_FEM_ANTSWLUT_SHIFT);
750 
751 	SPEX(leddc_on_time, SSB_SPROM8_LEDDC, SSB_SPROM8_LEDDC_ON,
752 	     SSB_SPROM8_LEDDC_ON_SHIFT);
753 	SPEX(leddc_off_time, SSB_SPROM8_LEDDC, SSB_SPROM8_LEDDC_OFF,
754 	     SSB_SPROM8_LEDDC_OFF_SHIFT);
755 
756 	SPEX(txchain, SSB_SPROM8_TXRXC, SSB_SPROM8_TXRXC_TXCHAIN,
757 	     SSB_SPROM8_TXRXC_TXCHAIN_SHIFT);
758 	SPEX(rxchain, SSB_SPROM8_TXRXC, SSB_SPROM8_TXRXC_RXCHAIN,
759 	     SSB_SPROM8_TXRXC_RXCHAIN_SHIFT);
760 	SPEX(antswitch, SSB_SPROM8_TXRXC, SSB_SPROM8_TXRXC_SWITCH,
761 	     SSB_SPROM8_TXRXC_SWITCH_SHIFT);
762 
763 	SPEX(opo, SSB_SPROM8_OFDM2GPO, 0x00ff, 0);
764 
765 	SPEX_ARRAY8(mcs2gpo, SSB_SPROM8_2G_MCSPO, ~0, 0);
766 	SPEX_ARRAY8(mcs5gpo, SSB_SPROM8_5G_MCSPO, ~0, 0);
767 	SPEX_ARRAY8(mcs5glpo, SSB_SPROM8_5GL_MCSPO, ~0, 0);
768 	SPEX_ARRAY8(mcs5ghpo, SSB_SPROM8_5GH_MCSPO, ~0, 0);
769 
770 	SPEX(rawtempsense, SSB_SPROM8_RAWTS, SSB_SPROM8_RAWTS_RAWTEMP,
771 	     SSB_SPROM8_RAWTS_RAWTEMP_SHIFT);
772 	SPEX(measpower, SSB_SPROM8_RAWTS, SSB_SPROM8_RAWTS_MEASPOWER,
773 	     SSB_SPROM8_RAWTS_MEASPOWER_SHIFT);
774 	SPEX(tempsense_slope, SSB_SPROM8_OPT_CORRX,
775 	     SSB_SPROM8_OPT_CORRX_TEMP_SLOPE,
776 	     SSB_SPROM8_OPT_CORRX_TEMP_SLOPE_SHIFT);
777 	SPEX(tempcorrx, SSB_SPROM8_OPT_CORRX, SSB_SPROM8_OPT_CORRX_TEMPCORRX,
778 	     SSB_SPROM8_OPT_CORRX_TEMPCORRX_SHIFT);
779 	SPEX(tempsense_option, SSB_SPROM8_OPT_CORRX,
780 	     SSB_SPROM8_OPT_CORRX_TEMP_OPTION,
781 	     SSB_SPROM8_OPT_CORRX_TEMP_OPTION_SHIFT);
782 	SPEX(freqoffset_corr, SSB_SPROM8_HWIQ_IQSWP,
783 	     SSB_SPROM8_HWIQ_IQSWP_FREQ_CORR,
784 	     SSB_SPROM8_HWIQ_IQSWP_FREQ_CORR_SHIFT);
785 	SPEX(iqcal_swp_dis, SSB_SPROM8_HWIQ_IQSWP,
786 	     SSB_SPROM8_HWIQ_IQSWP_IQCAL_SWP,
787 	     SSB_SPROM8_HWIQ_IQSWP_IQCAL_SWP_SHIFT);
788 	SPEX(hw_iqcal_en, SSB_SPROM8_HWIQ_IQSWP, SSB_SPROM8_HWIQ_IQSWP_HW_IQCAL,
789 	     SSB_SPROM8_HWIQ_IQSWP_HW_IQCAL_SHIFT);
790 
791 	SPEX(bw40po, SSB_SPROM8_BW40PO, ~0, 0);
792 	SPEX(cddpo, SSB_SPROM8_CDDPO, ~0, 0);
793 	SPEX(stbcpo, SSB_SPROM8_STBCPO, ~0, 0);
794 	SPEX(bwduppo, SSB_SPROM8_BWDUPPO, ~0, 0);
795 
796 	SPEX(tempthresh, SSB_SPROM8_THERMAL, SSB_SPROM8_THERMAL_TRESH,
797 	     SSB_SPROM8_THERMAL_TRESH_SHIFT);
798 	SPEX(tempoffset, SSB_SPROM8_THERMAL, SSB_SPROM8_THERMAL_OFFSET,
799 	     SSB_SPROM8_THERMAL_OFFSET_SHIFT);
800 	SPEX(phycal_tempdelta, SSB_SPROM8_TEMPDELTA,
801 	     SSB_SPROM8_TEMPDELTA_PHYCAL,
802 	     SSB_SPROM8_TEMPDELTA_PHYCAL_SHIFT);
803 	SPEX(temps_period, SSB_SPROM8_TEMPDELTA, SSB_SPROM8_TEMPDELTA_PERIOD,
804 	     SSB_SPROM8_TEMPDELTA_PERIOD_SHIFT);
805 	SPEX(temps_hysteresis, SSB_SPROM8_TEMPDELTA,
806 	     SSB_SPROM8_TEMPDELTA_HYSTERESIS,
807 	     SSB_SPROM8_TEMPDELTA_HYSTERESIS_SHIFT);
808 	sprom_extract_r458(out, in);
809 
810 	/* TODO - get remaining rev 8 stuff needed */
811 }
812 
813 static int sprom_extract(struct ssb_bus *bus, struct ssb_sprom *out,
814 			 const u16 *in, u16 size)
815 {
816 	memset(out, 0, sizeof(*out));
817 
818 	out->revision = in[size - 1] & 0x00FF;
819 	ssb_dbg("SPROM revision %d detected\n", out->revision);
820 	memset(out->et0mac, 0xFF, 6);		/* preset et0 and et1 mac */
821 	memset(out->et1mac, 0xFF, 6);
822 
823 	if ((bus->chip_id & 0xFF00) == 0x4400) {
824 		/* Workaround: The BCM44XX chip has a stupid revision
825 		 * number stored in the SPROM.
826 		 * Always extract r1. */
827 		out->revision = 1;
828 		ssb_dbg("SPROM treated as revision %d\n", out->revision);
829 	}
830 
831 	switch (out->revision) {
832 	case 1:
833 	case 2:
834 	case 3:
835 		sprom_extract_r123(out, in);
836 		break;
837 	case 4:
838 	case 5:
839 		sprom_extract_r45(out, in);
840 		break;
841 	case 8:
842 		sprom_extract_r8(out, in);
843 		break;
844 	default:
845 		ssb_warn("Unsupported SPROM revision %d detected. Will extract v1\n",
846 			 out->revision);
847 		out->revision = 1;
848 		sprom_extract_r123(out, in);
849 	}
850 
851 	if (out->boardflags_lo == 0xFFFF)
852 		out->boardflags_lo = 0;  /* per specs */
853 	if (out->boardflags_hi == 0xFFFF)
854 		out->boardflags_hi = 0;  /* per specs */
855 
856 	return 0;
857 }
858 
859 static int ssb_pci_sprom_get(struct ssb_bus *bus,
860 			     struct ssb_sprom *sprom)
861 {
862 	int err;
863 	u16 *buf;
864 
865 	if (!ssb_is_sprom_available(bus)) {
866 		ssb_err("No SPROM available!\n");
867 		return -ENODEV;
868 	}
869 	if (bus->chipco.dev) {	/* can be unavailable! */
870 		/*
871 		 * get SPROM offset: SSB_SPROM_BASE1 except for
872 		 * chipcommon rev >= 31 or chip ID is 0x4312 and
873 		 * chipcommon status & 3 == 2
874 		 */
875 		if (bus->chipco.dev->id.revision >= 31)
876 			bus->sprom_offset = SSB_SPROM_BASE31;
877 		else if (bus->chip_id == 0x4312 &&
878 			 (bus->chipco.status & 0x03) == 2)
879 			bus->sprom_offset = SSB_SPROM_BASE31;
880 		else
881 			bus->sprom_offset = SSB_SPROM_BASE1;
882 	} else {
883 		bus->sprom_offset = SSB_SPROM_BASE1;
884 	}
885 	ssb_dbg("SPROM offset is 0x%x\n", bus->sprom_offset);
886 
887 	buf = kcalloc(SSB_SPROMSIZE_WORDS_R123, sizeof(u16), GFP_KERNEL);
888 	if (!buf)
889 		return -ENOMEM;
890 	bus->sprom_size = SSB_SPROMSIZE_WORDS_R123;
891 	sprom_do_read(bus, buf);
892 	err = sprom_check_crc(buf, bus->sprom_size);
893 	if (err) {
894 		/* try for a 440 byte SPROM - revision 4 and higher */
895 		kfree(buf);
896 		buf = kcalloc(SSB_SPROMSIZE_WORDS_R4, sizeof(u16),
897 			      GFP_KERNEL);
898 		if (!buf)
899 			return -ENOMEM;
900 		bus->sprom_size = SSB_SPROMSIZE_WORDS_R4;
901 		sprom_do_read(bus, buf);
902 		err = sprom_check_crc(buf, bus->sprom_size);
903 		if (err) {
904 			/* All CRC attempts failed.
905 			 * Maybe there is no SPROM on the device?
906 			 * Now we ask the arch code if there is some sprom
907 			 * available for this device in some other storage */
908 			err = ssb_fill_sprom_with_fallback(bus, sprom);
909 			if (err) {
910 				ssb_warn("WARNING: Using fallback SPROM failed (err %d)\n",
911 					 err);
912 			} else {
913 				ssb_dbg("Using SPROM revision %d provided by platform\n",
914 					sprom->revision);
915 				err = 0;
916 				goto out_free;
917 			}
918 			ssb_warn("WARNING: Invalid SPROM CRC (corrupt SPROM)\n");
919 		}
920 	}
921 	err = sprom_extract(bus, sprom, buf, bus->sprom_size);
922 
923 out_free:
924 	kfree(buf);
925 	return err;
926 }
927 
928 static void ssb_pci_get_boardinfo(struct ssb_bus *bus,
929 				  struct ssb_boardinfo *bi)
930 {
931 	bi->vendor = bus->host_pci->subsystem_vendor;
932 	bi->type = bus->host_pci->subsystem_device;
933 }
934 
935 int ssb_pci_get_invariants(struct ssb_bus *bus,
936 			   struct ssb_init_invariants *iv)
937 {
938 	int err;
939 
940 	err = ssb_pci_sprom_get(bus, &iv->sprom);
941 	if (err)
942 		goto out;
943 	ssb_pci_get_boardinfo(bus, &iv->boardinfo);
944 
945 out:
946 	return err;
947 }
948 
949 #ifdef CONFIG_SSB_DEBUG
950 static int ssb_pci_assert_buspower(struct ssb_bus *bus)
951 {
952 	if (likely(bus->powered_up))
953 		return 0;
954 
955 	printk(KERN_ERR PFX "FATAL ERROR: Bus powered down "
956 	       "while accessing PCI MMIO space\n");
957 	if (bus->power_warn_count <= 10) {
958 		bus->power_warn_count++;
959 		dump_stack();
960 	}
961 
962 	return -ENODEV;
963 }
964 #else /* DEBUG */
965 static inline int ssb_pci_assert_buspower(struct ssb_bus *bus)
966 {
967 	return 0;
968 }
969 #endif /* DEBUG */
970 
971 static u8 ssb_pci_read8(struct ssb_device *dev, u16 offset)
972 {
973 	struct ssb_bus *bus = dev->bus;
974 
975 	if (unlikely(ssb_pci_assert_buspower(bus)))
976 		return 0xFF;
977 	if (unlikely(bus->mapped_device != dev)) {
978 		if (unlikely(ssb_pci_switch_core(bus, dev)))
979 			return 0xFF;
980 	}
981 	return ioread8(bus->mmio + offset);
982 }
983 
984 static u16 ssb_pci_read16(struct ssb_device *dev, u16 offset)
985 {
986 	struct ssb_bus *bus = dev->bus;
987 
988 	if (unlikely(ssb_pci_assert_buspower(bus)))
989 		return 0xFFFF;
990 	if (unlikely(bus->mapped_device != dev)) {
991 		if (unlikely(ssb_pci_switch_core(bus, dev)))
992 			return 0xFFFF;
993 	}
994 	return ioread16(bus->mmio + offset);
995 }
996 
997 static u32 ssb_pci_read32(struct ssb_device *dev, u16 offset)
998 {
999 	struct ssb_bus *bus = dev->bus;
1000 
1001 	if (unlikely(ssb_pci_assert_buspower(bus)))
1002 		return 0xFFFFFFFF;
1003 	if (unlikely(bus->mapped_device != dev)) {
1004 		if (unlikely(ssb_pci_switch_core(bus, dev)))
1005 			return 0xFFFFFFFF;
1006 	}
1007 	return ioread32(bus->mmio + offset);
1008 }
1009 
1010 #ifdef CONFIG_SSB_BLOCKIO
1011 static void ssb_pci_block_read(struct ssb_device *dev, void *buffer,
1012 			       size_t count, u16 offset, u8 reg_width)
1013 {
1014 	struct ssb_bus *bus = dev->bus;
1015 	void __iomem *addr = bus->mmio + offset;
1016 
1017 	if (unlikely(ssb_pci_assert_buspower(bus)))
1018 		goto error;
1019 	if (unlikely(bus->mapped_device != dev)) {
1020 		if (unlikely(ssb_pci_switch_core(bus, dev)))
1021 			goto error;
1022 	}
1023 	switch (reg_width) {
1024 	case sizeof(u8):
1025 		ioread8_rep(addr, buffer, count);
1026 		break;
1027 	case sizeof(u16):
1028 		SSB_WARN_ON(count & 1);
1029 		ioread16_rep(addr, buffer, count >> 1);
1030 		break;
1031 	case sizeof(u32):
1032 		SSB_WARN_ON(count & 3);
1033 		ioread32_rep(addr, buffer, count >> 2);
1034 		break;
1035 	default:
1036 		SSB_WARN_ON(1);
1037 	}
1038 
1039 	return;
1040 error:
1041 	memset(buffer, 0xFF, count);
1042 }
1043 #endif /* CONFIG_SSB_BLOCKIO */
1044 
1045 static void ssb_pci_write8(struct ssb_device *dev, u16 offset, u8 value)
1046 {
1047 	struct ssb_bus *bus = dev->bus;
1048 
1049 	if (unlikely(ssb_pci_assert_buspower(bus)))
1050 		return;
1051 	if (unlikely(bus->mapped_device != dev)) {
1052 		if (unlikely(ssb_pci_switch_core(bus, dev)))
1053 			return;
1054 	}
1055 	iowrite8(value, bus->mmio + offset);
1056 }
1057 
1058 static void ssb_pci_write16(struct ssb_device *dev, u16 offset, u16 value)
1059 {
1060 	struct ssb_bus *bus = dev->bus;
1061 
1062 	if (unlikely(ssb_pci_assert_buspower(bus)))
1063 		return;
1064 	if (unlikely(bus->mapped_device != dev)) {
1065 		if (unlikely(ssb_pci_switch_core(bus, dev)))
1066 			return;
1067 	}
1068 	iowrite16(value, bus->mmio + offset);
1069 }
1070 
1071 static void ssb_pci_write32(struct ssb_device *dev, u16 offset, u32 value)
1072 {
1073 	struct ssb_bus *bus = dev->bus;
1074 
1075 	if (unlikely(ssb_pci_assert_buspower(bus)))
1076 		return;
1077 	if (unlikely(bus->mapped_device != dev)) {
1078 		if (unlikely(ssb_pci_switch_core(bus, dev)))
1079 			return;
1080 	}
1081 	iowrite32(value, bus->mmio + offset);
1082 }
1083 
1084 #ifdef CONFIG_SSB_BLOCKIO
1085 static void ssb_pci_block_write(struct ssb_device *dev, const void *buffer,
1086 				size_t count, u16 offset, u8 reg_width)
1087 {
1088 	struct ssb_bus *bus = dev->bus;
1089 	void __iomem *addr = bus->mmio + offset;
1090 
1091 	if (unlikely(ssb_pci_assert_buspower(bus)))
1092 		return;
1093 	if (unlikely(bus->mapped_device != dev)) {
1094 		if (unlikely(ssb_pci_switch_core(bus, dev)))
1095 			return;
1096 	}
1097 	switch (reg_width) {
1098 	case sizeof(u8):
1099 		iowrite8_rep(addr, buffer, count);
1100 		break;
1101 	case sizeof(u16):
1102 		SSB_WARN_ON(count & 1);
1103 		iowrite16_rep(addr, buffer, count >> 1);
1104 		break;
1105 	case sizeof(u32):
1106 		SSB_WARN_ON(count & 3);
1107 		iowrite32_rep(addr, buffer, count >> 2);
1108 		break;
1109 	default:
1110 		SSB_WARN_ON(1);
1111 	}
1112 }
1113 #endif /* CONFIG_SSB_BLOCKIO */
1114 
1115 /* Not "static", as it's used in main.c */
1116 const struct ssb_bus_ops ssb_pci_ops = {
1117 	.read8		= ssb_pci_read8,
1118 	.read16		= ssb_pci_read16,
1119 	.read32		= ssb_pci_read32,
1120 	.write8		= ssb_pci_write8,
1121 	.write16	= ssb_pci_write16,
1122 	.write32	= ssb_pci_write32,
1123 #ifdef CONFIG_SSB_BLOCKIO
1124 	.block_read	= ssb_pci_block_read,
1125 	.block_write	= ssb_pci_block_write,
1126 #endif
1127 };
1128 
1129 static ssize_t ssb_pci_attr_sprom_show(struct device *pcidev,
1130 				       struct device_attribute *attr,
1131 				       char *buf)
1132 {
1133 	struct pci_dev *pdev = container_of(pcidev, struct pci_dev, dev);
1134 	struct ssb_bus *bus;
1135 
1136 	bus = ssb_pci_dev_to_bus(pdev);
1137 	if (!bus)
1138 		return -ENODEV;
1139 
1140 	return ssb_attr_sprom_show(bus, buf, sprom_do_read);
1141 }
1142 
1143 static ssize_t ssb_pci_attr_sprom_store(struct device *pcidev,
1144 					struct device_attribute *attr,
1145 					const char *buf, size_t count)
1146 {
1147 	struct pci_dev *pdev = container_of(pcidev, struct pci_dev, dev);
1148 	struct ssb_bus *bus;
1149 
1150 	bus = ssb_pci_dev_to_bus(pdev);
1151 	if (!bus)
1152 		return -ENODEV;
1153 
1154 	return ssb_attr_sprom_store(bus, buf, count,
1155 				    sprom_check_crc, sprom_do_write);
1156 }
1157 
1158 static DEVICE_ATTR(ssb_sprom, 0600,
1159 		   ssb_pci_attr_sprom_show,
1160 		   ssb_pci_attr_sprom_store);
1161 
1162 void ssb_pci_exit(struct ssb_bus *bus)
1163 {
1164 	struct pci_dev *pdev;
1165 
1166 	if (bus->bustype != SSB_BUSTYPE_PCI)
1167 		return;
1168 
1169 	pdev = bus->host_pci;
1170 	device_remove_file(&pdev->dev, &dev_attr_ssb_sprom);
1171 }
1172 
1173 int ssb_pci_init(struct ssb_bus *bus)
1174 {
1175 	struct pci_dev *pdev;
1176 	int err;
1177 
1178 	if (bus->bustype != SSB_BUSTYPE_PCI)
1179 		return 0;
1180 
1181 	pdev = bus->host_pci;
1182 	mutex_init(&bus->sprom_mutex);
1183 	err = device_create_file(&pdev->dev, &dev_attr_ssb_sprom);
1184 	if (err)
1185 		goto out;
1186 
1187 out:
1188 	return err;
1189 }
1190