blob: e2cfdda20a0ffb3d991272ba65c1c846741fd338 [file] [log] [blame]
Daniel Drakee85d0912006-06-02 17:11:32 +01001/* zd_chip.c
2 *
3 * This program is free software; you can redistribute it and/or modify
4 * it under the terms of the GNU General Public License as published by
5 * the Free Software Foundation; either version 2 of the License, or
6 * (at your option) any later version.
7 *
8 * This program is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 * GNU General Public License for more details.
12 *
13 * You should have received a copy of the GNU General Public License
14 * along with this program; if not, write to the Free Software
15 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
16 */
17
18/* This file implements all the hardware specific functions for the ZD1211
19 * and ZD1211B chips. Support for the ZD1211B was possible after Timothy
20 * Legge sent me a ZD1211B device. Thank you Tim. -- Uli
21 */
22
23#include <linux/kernel.h>
24#include <linux/errno.h>
25
26#include "zd_def.h"
27#include "zd_chip.h"
28#include "zd_ieee80211.h"
29#include "zd_mac.h"
30#include "zd_rf.h"
31#include "zd_util.h"
32
33void zd_chip_init(struct zd_chip *chip,
34 struct net_device *netdev,
35 struct usb_interface *intf)
36{
37 memset(chip, 0, sizeof(*chip));
38 mutex_init(&chip->mutex);
39 zd_usb_init(&chip->usb, netdev, intf);
40 zd_rf_init(&chip->rf);
41}
42
43void zd_chip_clear(struct zd_chip *chip)
44{
Ulrich Kunitzc48cf122006-08-12 18:00:17 +010045 ZD_ASSERT(!mutex_is_locked(&chip->mutex));
Daniel Drakee85d0912006-06-02 17:11:32 +010046 zd_usb_clear(&chip->usb);
47 zd_rf_clear(&chip->rf);
Daniel Drakee85d0912006-06-02 17:11:32 +010048 mutex_destroy(&chip->mutex);
Ulrich Kunitzc48cf122006-08-12 18:00:17 +010049 ZD_MEMCLEAR(chip, sizeof(*chip));
Daniel Drakee85d0912006-06-02 17:11:32 +010050}
51
52static int scnprint_mac_oui(const u8 *addr, char *buffer, size_t size)
53{
54 return scnprintf(buffer, size, "%02x-%02x-%02x",
55 addr[0], addr[1], addr[2]);
56}
57
58/* Prints an identifier line, which will support debugging. */
59static int scnprint_id(struct zd_chip *chip, char *buffer, size_t size)
60{
61 int i = 0;
62
63 i = scnprintf(buffer, size, "zd1211%s chip ",
64 chip->is_zd1211b ? "b" : "");
65 i += zd_usb_scnprint_id(&chip->usb, buffer+i, size-i);
66 i += scnprintf(buffer+i, size-i, " ");
67 i += scnprint_mac_oui(chip->e2p_mac, buffer+i, size-i);
68 i += scnprintf(buffer+i, size-i, " ");
69 i += zd_rf_scnprint_id(&chip->rf, buffer+i, size-i);
Daniel Drakef2a81a12007-03-11 19:54:28 +000070 i += scnprintf(buffer+i, size-i, " pa%1x %c%c%c%c%c", chip->pa_type,
Daniel Drakee85d0912006-06-02 17:11:32 +010071 chip->patch_cck_gain ? 'g' : '-',
72 chip->patch_cr157 ? '7' : '-',
Daniel Drake20fe2172006-08-12 17:59:42 +010073 chip->patch_6m_band_edge ? '6' : '-',
Daniel Drakef2a81a12007-03-11 19:54:28 +000074 chip->new_phy_layout ? 'N' : '-',
75 chip->al2230s_bit ? 'S' : '-');
Daniel Drakee85d0912006-06-02 17:11:32 +010076 return i;
77}
78
79static void print_id(struct zd_chip *chip)
80{
81 char buffer[80];
82
83 scnprint_id(chip, buffer, sizeof(buffer));
84 buffer[sizeof(buffer)-1] = 0;
85 dev_info(zd_chip_dev(chip), "%s\n", buffer);
86}
87
Daniel Drake0ce34bc2006-12-12 01:26:11 +000088static zd_addr_t inc_addr(zd_addr_t addr)
89{
90 u16 a = (u16)addr;
91 /* Control registers use byte addressing, but everything else uses word
92 * addressing. */
93 if ((a & 0xf000) == CR_START)
94 a += 2;
95 else
96 a += 1;
97 return (zd_addr_t)a;
98}
99
Daniel Drakee85d0912006-06-02 17:11:32 +0100100/* Read a variable number of 32-bit values. Parameter count is not allowed to
101 * exceed USB_MAX_IOREAD32_COUNT.
102 */
103int zd_ioread32v_locked(struct zd_chip *chip, u32 *values, const zd_addr_t *addr,
104 unsigned int count)
105{
106 int r;
107 int i;
108 zd_addr_t *a16 = (zd_addr_t *)NULL;
109 u16 *v16;
110 unsigned int count16;
111
112 if (count > USB_MAX_IOREAD32_COUNT)
113 return -EINVAL;
114
115 /* Allocate a single memory block for values and addresses. */
116 count16 = 2*count;
Daniel Drake44956852007-02-10 01:27:18 +0000117 a16 = (zd_addr_t *) kmalloc(count16 * (sizeof(zd_addr_t) + sizeof(u16)),
Ulrich Kunitz35c34042007-02-18 20:28:23 +0000118 GFP_KERNEL);
Daniel Drakee85d0912006-06-02 17:11:32 +0100119 if (!a16) {
120 dev_dbg_f(zd_chip_dev(chip),
121 "error ENOMEM in allocation of a16\n");
122 r = -ENOMEM;
123 goto out;
124 }
125 v16 = (u16 *)(a16 + count16);
126
127 for (i = 0; i < count; i++) {
128 int j = 2*i;
129 /* We read the high word always first. */
Daniel Drake0ce34bc2006-12-12 01:26:11 +0000130 a16[j] = inc_addr(addr[i]);
Daniel Drakee85d0912006-06-02 17:11:32 +0100131 a16[j+1] = addr[i];
132 }
133
134 r = zd_ioread16v_locked(chip, v16, a16, count16);
135 if (r) {
136 dev_dbg_f(zd_chip_dev(chip),
137 "error: zd_ioread16v_locked. Error number %d\n", r);
138 goto out;
139 }
140
141 for (i = 0; i < count; i++) {
142 int j = 2*i;
143 values[i] = (v16[j] << 16) | v16[j+1];
144 }
145
146out:
147 kfree((void *)a16);
148 return r;
149}
150
151int _zd_iowrite32v_locked(struct zd_chip *chip, const struct zd_ioreq32 *ioreqs,
152 unsigned int count)
153{
154 int i, j, r;
155 struct zd_ioreq16 *ioreqs16;
156 unsigned int count16;
157
158 ZD_ASSERT(mutex_is_locked(&chip->mutex));
159
160 if (count == 0)
161 return 0;
162 if (count > USB_MAX_IOWRITE32_COUNT)
163 return -EINVAL;
164
165 /* Allocate a single memory block for values and addresses. */
166 count16 = 2*count;
Ulrich Kunitz35c34042007-02-18 20:28:23 +0000167 ioreqs16 = kmalloc(count16 * sizeof(struct zd_ioreq16), GFP_KERNEL);
Daniel Drakee85d0912006-06-02 17:11:32 +0100168 if (!ioreqs16) {
169 r = -ENOMEM;
170 dev_dbg_f(zd_chip_dev(chip),
171 "error %d in ioreqs16 allocation\n", r);
172 goto out;
173 }
174
175 for (i = 0; i < count; i++) {
176 j = 2*i;
177 /* We write the high word always first. */
178 ioreqs16[j].value = ioreqs[i].value >> 16;
Daniel Drake0ce34bc2006-12-12 01:26:11 +0000179 ioreqs16[j].addr = inc_addr(ioreqs[i].addr);
Daniel Drakee85d0912006-06-02 17:11:32 +0100180 ioreqs16[j+1].value = ioreqs[i].value;
181 ioreqs16[j+1].addr = ioreqs[i].addr;
182 }
183
184 r = zd_usb_iowrite16v(&chip->usb, ioreqs16, count16);
185#ifdef DEBUG
186 if (r) {
187 dev_dbg_f(zd_chip_dev(chip),
188 "error %d in zd_usb_write16v\n", r);
189 }
190#endif /* DEBUG */
191out:
192 kfree(ioreqs16);
193 return r;
194}
195
196int zd_iowrite16a_locked(struct zd_chip *chip,
197 const struct zd_ioreq16 *ioreqs, unsigned int count)
198{
199 int r;
200 unsigned int i, j, t, max;
201
202 ZD_ASSERT(mutex_is_locked(&chip->mutex));
203 for (i = 0; i < count; i += j + t) {
204 t = 0;
205 max = count-i;
206 if (max > USB_MAX_IOWRITE16_COUNT)
207 max = USB_MAX_IOWRITE16_COUNT;
208 for (j = 0; j < max; j++) {
209 if (!ioreqs[i+j].addr) {
210 t = 1;
211 break;
212 }
213 }
214
215 r = zd_usb_iowrite16v(&chip->usb, &ioreqs[i], j);
216 if (r) {
217 dev_dbg_f(zd_chip_dev(chip),
218 "error zd_usb_iowrite16v. Error number %d\n",
219 r);
220 return r;
221 }
222 }
223
224 return 0;
225}
226
227/* Writes a variable number of 32 bit registers. The functions will split
228 * that in several USB requests. A split can be forced by inserting an IO
229 * request with an zero address field.
230 */
231int zd_iowrite32a_locked(struct zd_chip *chip,
232 const struct zd_ioreq32 *ioreqs, unsigned int count)
233{
234 int r;
235 unsigned int i, j, t, max;
236
237 for (i = 0; i < count; i += j + t) {
238 t = 0;
239 max = count-i;
240 if (max > USB_MAX_IOWRITE32_COUNT)
241 max = USB_MAX_IOWRITE32_COUNT;
242 for (j = 0; j < max; j++) {
243 if (!ioreqs[i+j].addr) {
244 t = 1;
245 break;
246 }
247 }
248
249 r = _zd_iowrite32v_locked(chip, &ioreqs[i], j);
250 if (r) {
251 dev_dbg_f(zd_chip_dev(chip),
252 "error _zd_iowrite32v_locked."
253 " Error number %d\n", r);
254 return r;
255 }
256 }
257
258 return 0;
259}
260
261int zd_ioread16(struct zd_chip *chip, zd_addr_t addr, u16 *value)
262{
263 int r;
264
Daniel Drakee85d0912006-06-02 17:11:32 +0100265 mutex_lock(&chip->mutex);
266 r = zd_ioread16_locked(chip, value, addr);
267 mutex_unlock(&chip->mutex);
268 return r;
269}
270
271int zd_ioread32(struct zd_chip *chip, zd_addr_t addr, u32 *value)
272{
273 int r;
274
Daniel Drakee85d0912006-06-02 17:11:32 +0100275 mutex_lock(&chip->mutex);
276 r = zd_ioread32_locked(chip, value, addr);
277 mutex_unlock(&chip->mutex);
278 return r;
279}
280
281int zd_iowrite16(struct zd_chip *chip, zd_addr_t addr, u16 value)
282{
283 int r;
284
Daniel Drakee85d0912006-06-02 17:11:32 +0100285 mutex_lock(&chip->mutex);
286 r = zd_iowrite16_locked(chip, value, addr);
287 mutex_unlock(&chip->mutex);
288 return r;
289}
290
291int zd_iowrite32(struct zd_chip *chip, zd_addr_t addr, u32 value)
292{
293 int r;
294
Daniel Drakee85d0912006-06-02 17:11:32 +0100295 mutex_lock(&chip->mutex);
296 r = zd_iowrite32_locked(chip, value, addr);
297 mutex_unlock(&chip->mutex);
298 return r;
299}
300
301int zd_ioread32v(struct zd_chip *chip, const zd_addr_t *addresses,
302 u32 *values, unsigned int count)
303{
304 int r;
305
Daniel Drakee85d0912006-06-02 17:11:32 +0100306 mutex_lock(&chip->mutex);
307 r = zd_ioread32v_locked(chip, values, addresses, count);
308 mutex_unlock(&chip->mutex);
309 return r;
310}
311
312int zd_iowrite32a(struct zd_chip *chip, const struct zd_ioreq32 *ioreqs,
313 unsigned int count)
314{
315 int r;
316
Daniel Drakee85d0912006-06-02 17:11:32 +0100317 mutex_lock(&chip->mutex);
318 r = zd_iowrite32a_locked(chip, ioreqs, count);
319 mutex_unlock(&chip->mutex);
320 return r;
321}
322
323static int read_pod(struct zd_chip *chip, u8 *rf_type)
324{
325 int r;
326 u32 value;
327
328 ZD_ASSERT(mutex_is_locked(&chip->mutex));
329 r = zd_ioread32_locked(chip, &value, E2P_POD);
330 if (r)
331 goto error;
332 dev_dbg_f(zd_chip_dev(chip), "E2P_POD %#010x\n", value);
333
334 /* FIXME: AL2230 handling (Bit 7 in POD) */
335 *rf_type = value & 0x0f;
336 chip->pa_type = (value >> 16) & 0x0f;
337 chip->patch_cck_gain = (value >> 8) & 0x1;
338 chip->patch_cr157 = (value >> 13) & 0x1;
339 chip->patch_6m_band_edge = (value >> 21) & 0x1;
Daniel Drake20fe2172006-08-12 17:59:42 +0100340 chip->new_phy_layout = (value >> 31) & 0x1;
Daniel Drakeae6ead42007-03-11 19:54:11 +0000341 chip->al2230s_bit = (value >> 7) & 0x1;
Ulrich Kunitz583afd12006-09-13 02:42:38 +0100342 chip->link_led = ((value >> 4) & 1) ? LED1 : LED2;
343 chip->supports_tx_led = 1;
344 if (value & (1 << 24)) { /* LED scenario */
345 if (value & (1 << 29))
346 chip->supports_tx_led = 0;
347 }
Daniel Drakee85d0912006-06-02 17:11:32 +0100348
349 dev_dbg_f(zd_chip_dev(chip),
350 "RF %s %#01x PA type %#01x patch CCK %d patch CR157 %d "
Ulrich Kunitz583afd12006-09-13 02:42:38 +0100351 "patch 6M %d new PHY %d link LED%d tx led %d\n",
Daniel Drakee85d0912006-06-02 17:11:32 +0100352 zd_rf_name(*rf_type), *rf_type,
353 chip->pa_type, chip->patch_cck_gain,
Ulrich Kunitz583afd12006-09-13 02:42:38 +0100354 chip->patch_cr157, chip->patch_6m_band_edge,
355 chip->new_phy_layout,
356 chip->link_led == LED1 ? 1 : 2,
357 chip->supports_tx_led);
Daniel Drakee85d0912006-06-02 17:11:32 +0100358 return 0;
359error:
360 *rf_type = 0;
361 chip->pa_type = 0;
362 chip->patch_cck_gain = 0;
363 chip->patch_cr157 = 0;
364 chip->patch_6m_band_edge = 0;
Daniel Drake20fe2172006-08-12 17:59:42 +0100365 chip->new_phy_layout = 0;
Daniel Drakee85d0912006-06-02 17:11:32 +0100366 return r;
367}
368
369static int _read_mac_addr(struct zd_chip *chip, u8 *mac_addr,
370 const zd_addr_t *addr)
371{
372 int r;
373 u32 parts[2];
374
375 r = zd_ioread32v_locked(chip, parts, (const zd_addr_t *)addr, 2);
376 if (r) {
377 dev_dbg_f(zd_chip_dev(chip),
378 "error: couldn't read e2p macs. Error number %d\n", r);
379 return r;
380 }
381
382 mac_addr[0] = parts[0];
383 mac_addr[1] = parts[0] >> 8;
384 mac_addr[2] = parts[0] >> 16;
385 mac_addr[3] = parts[0] >> 24;
386 mac_addr[4] = parts[1];
387 mac_addr[5] = parts[1] >> 8;
388
389 return 0;
390}
391
392static int read_e2p_mac_addr(struct zd_chip *chip)
393{
394 static const zd_addr_t addr[2] = { E2P_MAC_ADDR_P1, E2P_MAC_ADDR_P2 };
395
396 ZD_ASSERT(mutex_is_locked(&chip->mutex));
397 return _read_mac_addr(chip, chip->e2p_mac, (const zd_addr_t *)addr);
398}
399
400/* MAC address: if custom mac addresses are to to be used CR_MAC_ADDR_P1 and
401 * CR_MAC_ADDR_P2 must be overwritten
402 */
403void zd_get_e2p_mac_addr(struct zd_chip *chip, u8 *mac_addr)
404{
405 mutex_lock(&chip->mutex);
406 memcpy(mac_addr, chip->e2p_mac, ETH_ALEN);
407 mutex_unlock(&chip->mutex);
408}
409
410static int read_mac_addr(struct zd_chip *chip, u8 *mac_addr)
411{
412 static const zd_addr_t addr[2] = { CR_MAC_ADDR_P1, CR_MAC_ADDR_P2 };
413 return _read_mac_addr(chip, mac_addr, (const zd_addr_t *)addr);
414}
415
416int zd_read_mac_addr(struct zd_chip *chip, u8 *mac_addr)
417{
418 int r;
419
420 dev_dbg_f(zd_chip_dev(chip), "\n");
421 mutex_lock(&chip->mutex);
422 r = read_mac_addr(chip, mac_addr);
423 mutex_unlock(&chip->mutex);
424 return r;
425}
426
427int zd_write_mac_addr(struct zd_chip *chip, const u8 *mac_addr)
428{
429 int r;
430 struct zd_ioreq32 reqs[2] = {
431 [0] = { .addr = CR_MAC_ADDR_P1 },
432 [1] = { .addr = CR_MAC_ADDR_P2 },
433 };
434
435 reqs[0].value = (mac_addr[3] << 24)
436 | (mac_addr[2] << 16)
437 | (mac_addr[1] << 8)
438 | mac_addr[0];
439 reqs[1].value = (mac_addr[5] << 8)
440 | mac_addr[4];
441
442 dev_dbg_f(zd_chip_dev(chip),
443 "mac addr " MAC_FMT "\n", MAC_ARG(mac_addr));
444
445 mutex_lock(&chip->mutex);
446 r = zd_iowrite32a_locked(chip, reqs, ARRAY_SIZE(reqs));
447#ifdef DEBUG
448 {
449 u8 tmp[ETH_ALEN];
450 read_mac_addr(chip, tmp);
451 }
452#endif /* DEBUG */
453 mutex_unlock(&chip->mutex);
454 return r;
455}
456
457int zd_read_regdomain(struct zd_chip *chip, u8 *regdomain)
458{
459 int r;
460 u32 value;
461
462 mutex_lock(&chip->mutex);
463 r = zd_ioread32_locked(chip, &value, E2P_SUBID);
464 mutex_unlock(&chip->mutex);
465 if (r)
466 return r;
467
468 *regdomain = value >> 16;
469 dev_dbg_f(zd_chip_dev(chip), "regdomain: %#04x\n", *regdomain);
470
471 return 0;
472}
473
474static int read_values(struct zd_chip *chip, u8 *values, size_t count,
475 zd_addr_t e2p_addr, u32 guard)
476{
477 int r;
478 int i;
479 u32 v;
480
481 ZD_ASSERT(mutex_is_locked(&chip->mutex));
482 for (i = 0;;) {
Daniel Drake0ce34bc2006-12-12 01:26:11 +0000483 r = zd_ioread32_locked(chip, &v,
484 (zd_addr_t)((u16)e2p_addr+i/2));
Daniel Drakee85d0912006-06-02 17:11:32 +0100485 if (r)
486 return r;
487 v -= guard;
488 if (i+4 < count) {
489 values[i++] = v;
490 values[i++] = v >> 8;
491 values[i++] = v >> 16;
492 values[i++] = v >> 24;
493 continue;
494 }
495 for (;i < count; i++)
496 values[i] = v >> (8*(i%3));
497 return 0;
498 }
499}
500
501static int read_pwr_cal_values(struct zd_chip *chip)
502{
503 return read_values(chip, chip->pwr_cal_values,
504 E2P_CHANNEL_COUNT, E2P_PWR_CAL_VALUE1,
505 0);
506}
507
508static int read_pwr_int_values(struct zd_chip *chip)
509{
510 return read_values(chip, chip->pwr_int_values,
511 E2P_CHANNEL_COUNT, E2P_PWR_INT_VALUE1,
512 E2P_PWR_INT_GUARD);
513}
514
515static int read_ofdm_cal_values(struct zd_chip *chip)
516{
517 int r;
518 int i;
519 static const zd_addr_t addresses[] = {
520 E2P_36M_CAL_VALUE1,
521 E2P_48M_CAL_VALUE1,
522 E2P_54M_CAL_VALUE1,
523 };
524
525 for (i = 0; i < 3; i++) {
526 r = read_values(chip, chip->ofdm_cal_values[i],
527 E2P_CHANNEL_COUNT, addresses[i], 0);
528 if (r)
529 return r;
530 }
531 return 0;
532}
533
534static int read_cal_int_tables(struct zd_chip *chip)
535{
536 int r;
537
538 r = read_pwr_cal_values(chip);
539 if (r)
540 return r;
541 r = read_pwr_int_values(chip);
542 if (r)
543 return r;
544 r = read_ofdm_cal_values(chip);
545 if (r)
546 return r;
547 return 0;
548}
549
550/* phy means physical registers */
551int zd_chip_lock_phy_regs(struct zd_chip *chip)
552{
553 int r;
554 u32 tmp;
555
556 ZD_ASSERT(mutex_is_locked(&chip->mutex));
557 r = zd_ioread32_locked(chip, &tmp, CR_REG1);
558 if (r) {
559 dev_err(zd_chip_dev(chip), "error ioread32(CR_REG1): %d\n", r);
560 return r;
561 }
562
563 dev_dbg_f(zd_chip_dev(chip),
564 "CR_REG1: 0x%02x -> 0x%02x\n", tmp, tmp & ~UNLOCK_PHY_REGS);
565 tmp &= ~UNLOCK_PHY_REGS;
566
567 r = zd_iowrite32_locked(chip, tmp, CR_REG1);
568 if (r)
569 dev_err(zd_chip_dev(chip), "error iowrite32(CR_REG1): %d\n", r);
570 return r;
571}
572
573int zd_chip_unlock_phy_regs(struct zd_chip *chip)
574{
575 int r;
576 u32 tmp;
577
578 ZD_ASSERT(mutex_is_locked(&chip->mutex));
579 r = zd_ioread32_locked(chip, &tmp, CR_REG1);
580 if (r) {
581 dev_err(zd_chip_dev(chip),
582 "error ioread32(CR_REG1): %d\n", r);
583 return r;
584 }
585
586 dev_dbg_f(zd_chip_dev(chip),
587 "CR_REG1: 0x%02x -> 0x%02x\n", tmp, tmp | UNLOCK_PHY_REGS);
588 tmp |= UNLOCK_PHY_REGS;
589
590 r = zd_iowrite32_locked(chip, tmp, CR_REG1);
591 if (r)
592 dev_err(zd_chip_dev(chip), "error iowrite32(CR_REG1): %d\n", r);
593 return r;
594}
595
Daniel Drake92b3e2e2007-04-03 23:17:37 +0100596/* CR157 can be optionally patched by the EEPROM for original ZD1211 */
Daniel Drakee85d0912006-06-02 17:11:32 +0100597static int patch_cr157(struct zd_chip *chip)
598{
599 int r;
Daniel Drake92b3e2e2007-04-03 23:17:37 +0100600 u16 value;
Daniel Drakee85d0912006-06-02 17:11:32 +0100601
602 if (!chip->patch_cr157)
603 return 0;
604
Daniel Drake92b3e2e2007-04-03 23:17:37 +0100605 r = zd_ioread16_locked(chip, &value, E2P_PHY_REG);
Daniel Drakee85d0912006-06-02 17:11:32 +0100606 if (r)
607 return r;
608
609 dev_dbg_f(zd_chip_dev(chip), "patching value %x\n", value >> 8);
610 return zd_iowrite32_locked(chip, value >> 8, CR157);
611}
612
613/*
614 * 6M band edge can be optionally overwritten for certain RF's
615 * Vendor driver says: for FCC regulation, enabled per HWFeature 6M band edge
616 * bit (for AL2230, AL2230S)
617 */
618static int patch_6m_band_edge(struct zd_chip *chip, int channel)
619{
620 struct zd_ioreq16 ioreqs[] = {
621 { CR128, 0x14 }, { CR129, 0x12 }, { CR130, 0x10 },
622 { CR47, 0x1e },
623 };
624
625 if (!chip->patch_6m_band_edge || !chip->rf.patch_6m_band_edge)
626 return 0;
627
628 /* FIXME: Channel 11 is not the edge for all regulatory domains. */
629 if (channel == 1 || channel == 11)
630 ioreqs[0].value = 0x12;
631
632 dev_dbg_f(zd_chip_dev(chip), "patching for channel %d\n", channel);
633 return zd_iowrite16a_locked(chip, ioreqs, ARRAY_SIZE(ioreqs));
634}
635
636static int zd1211_hw_reset_phy(struct zd_chip *chip)
637{
638 static const struct zd_ioreq16 ioreqs[] = {
639 { CR0, 0x0a }, { CR1, 0x06 }, { CR2, 0x26 },
640 { CR3, 0x38 }, { CR4, 0x80 }, { CR9, 0xa0 },
641 { CR10, 0x81 }, { CR11, 0x00 }, { CR12, 0x7f },
642 { CR13, 0x8c }, { CR14, 0x80 }, { CR15, 0x3d },
643 { CR16, 0x20 }, { CR17, 0x1e }, { CR18, 0x0a },
644 { CR19, 0x48 }, { CR20, 0x0c }, { CR21, 0x0c },
645 { CR22, 0x23 }, { CR23, 0x90 }, { CR24, 0x14 },
646 { CR25, 0x40 }, { CR26, 0x10 }, { CR27, 0x19 },
647 { CR28, 0x7f }, { CR29, 0x80 }, { CR30, 0x4b },
648 { CR31, 0x60 }, { CR32, 0x43 }, { CR33, 0x08 },
649 { CR34, 0x06 }, { CR35, 0x0a }, { CR36, 0x00 },
650 { CR37, 0x00 }, { CR38, 0x38 }, { CR39, 0x0c },
651 { CR40, 0x84 }, { CR41, 0x2a }, { CR42, 0x80 },
652 { CR43, 0x10 }, { CR44, 0x12 }, { CR46, 0xff },
653 { CR47, 0x1E }, { CR48, 0x26 }, { CR49, 0x5b },
654 { CR64, 0xd0 }, { CR65, 0x04 }, { CR66, 0x58 },
655 { CR67, 0xc9 }, { CR68, 0x88 }, { CR69, 0x41 },
656 { CR70, 0x23 }, { CR71, 0x10 }, { CR72, 0xff },
657 { CR73, 0x32 }, { CR74, 0x30 }, { CR75, 0x65 },
658 { CR76, 0x41 }, { CR77, 0x1b }, { CR78, 0x30 },
659 { CR79, 0x68 }, { CR80, 0x64 }, { CR81, 0x64 },
660 { CR82, 0x00 }, { CR83, 0x00 }, { CR84, 0x00 },
661 { CR85, 0x02 }, { CR86, 0x00 }, { CR87, 0x00 },
662 { CR88, 0xff }, { CR89, 0xfc }, { CR90, 0x00 },
663 { CR91, 0x00 }, { CR92, 0x00 }, { CR93, 0x08 },
664 { CR94, 0x00 }, { CR95, 0x00 }, { CR96, 0xff },
665 { CR97, 0xe7 }, { CR98, 0x00 }, { CR99, 0x00 },
666 { CR100, 0x00 }, { CR101, 0xae }, { CR102, 0x02 },
667 { CR103, 0x00 }, { CR104, 0x03 }, { CR105, 0x65 },
668 { CR106, 0x04 }, { CR107, 0x00 }, { CR108, 0x0a },
669 { CR109, 0xaa }, { CR110, 0xaa }, { CR111, 0x25 },
670 { CR112, 0x25 }, { CR113, 0x00 }, { CR119, 0x1e },
671 { CR125, 0x90 }, { CR126, 0x00 }, { CR127, 0x00 },
672 { },
673 { CR5, 0x00 }, { CR6, 0x00 }, { CR7, 0x00 },
674 { CR8, 0x00 }, { CR9, 0x20 }, { CR12, 0xf0 },
675 { CR20, 0x0e }, { CR21, 0x0e }, { CR27, 0x10 },
676 { CR44, 0x33 }, { CR47, 0x1E }, { CR83, 0x24 },
677 { CR84, 0x04 }, { CR85, 0x00 }, { CR86, 0x0C },
678 { CR87, 0x12 }, { CR88, 0x0C }, { CR89, 0x00 },
679 { CR90, 0x10 }, { CR91, 0x08 }, { CR93, 0x00 },
680 { CR94, 0x01 }, { CR95, 0x00 }, { CR96, 0x50 },
681 { CR97, 0x37 }, { CR98, 0x35 }, { CR101, 0x13 },
682 { CR102, 0x27 }, { CR103, 0x27 }, { CR104, 0x18 },
683 { CR105, 0x12 }, { CR109, 0x27 }, { CR110, 0x27 },
684 { CR111, 0x27 }, { CR112, 0x27 }, { CR113, 0x27 },
685 { CR114, 0x27 }, { CR115, 0x26 }, { CR116, 0x24 },
686 { CR117, 0xfc }, { CR118, 0xfa }, { CR120, 0x4f },
Daniel Drakedc536a72007-04-03 23:17:10 +0100687 { CR125, 0xaa }, { CR127, 0x03 }, { CR128, 0x14 },
688 { CR129, 0x12 }, { CR130, 0x10 }, { CR131, 0x0C },
689 { CR136, 0xdf }, { CR137, 0x40 }, { CR138, 0xa0 },
690 { CR139, 0xb0 }, { CR140, 0x99 }, { CR141, 0x82 },
691 { CR142, 0x54 }, { CR143, 0x1c }, { CR144, 0x6c },
692 { CR147, 0x07 }, { CR148, 0x4c }, { CR149, 0x50 },
693 { CR150, 0x0e }, { CR151, 0x18 }, { CR160, 0xfe },
694 { CR161, 0xee }, { CR162, 0xaa }, { CR163, 0xfa },
695 { CR164, 0xfa }, { CR165, 0xea }, { CR166, 0xbe },
696 { CR167, 0xbe }, { CR168, 0x6a }, { CR169, 0xba },
697 { CR170, 0xba }, { CR171, 0xba },
Daniel Drakee85d0912006-06-02 17:11:32 +0100698 /* Note: CR204 must lead the CR203 */
699 { CR204, 0x7d },
700 { },
701 { CR203, 0x30 },
702 };
703
704 int r, t;
705
706 dev_dbg_f(zd_chip_dev(chip), "\n");
707
708 r = zd_chip_lock_phy_regs(chip);
709 if (r)
710 goto out;
711
712 r = zd_iowrite16a_locked(chip, ioreqs, ARRAY_SIZE(ioreqs));
713 if (r)
714 goto unlock;
715
716 r = patch_cr157(chip);
717unlock:
718 t = zd_chip_unlock_phy_regs(chip);
719 if (t && !r)
720 r = t;
721out:
722 return r;
723}
724
725static int zd1211b_hw_reset_phy(struct zd_chip *chip)
726{
727 static const struct zd_ioreq16 ioreqs[] = {
728 { CR0, 0x14 }, { CR1, 0x06 }, { CR2, 0x26 },
729 { CR3, 0x38 }, { CR4, 0x80 }, { CR9, 0xe0 },
730 { CR10, 0x81 },
731 /* power control { { CR11, 1 << 6 }, */
732 { CR11, 0x00 },
733 { CR12, 0xf0 }, { CR13, 0x8c }, { CR14, 0x80 },
734 { CR15, 0x3d }, { CR16, 0x20 }, { CR17, 0x1e },
735 { CR18, 0x0a }, { CR19, 0x48 },
736 { CR20, 0x10 }, /* Org:0x0E, ComTrend:RalLink AP */
737 { CR21, 0x0e }, { CR22, 0x23 }, { CR23, 0x90 },
738 { CR24, 0x14 }, { CR25, 0x40 }, { CR26, 0x10 },
739 { CR27, 0x10 }, { CR28, 0x7f }, { CR29, 0x80 },
Daniel Drakefe7215c2006-08-12 17:59:12 +0100740 { CR30, 0x4b }, /* ASIC/FWT, no jointly decoder */
Daniel Drakee85d0912006-06-02 17:11:32 +0100741 { CR31, 0x60 }, { CR32, 0x43 }, { CR33, 0x08 },
742 { CR34, 0x06 }, { CR35, 0x0a }, { CR36, 0x00 },
743 { CR37, 0x00 }, { CR38, 0x38 }, { CR39, 0x0c },
744 { CR40, 0x84 }, { CR41, 0x2a }, { CR42, 0x80 },
745 { CR43, 0x10 }, { CR44, 0x33 }, { CR46, 0xff },
746 { CR47, 0x1E }, { CR48, 0x26 }, { CR49, 0x5b },
747 { CR64, 0xd0 }, { CR65, 0x04 }, { CR66, 0x58 },
748 { CR67, 0xc9 }, { CR68, 0x88 }, { CR69, 0x41 },
749 { CR70, 0x23 }, { CR71, 0x10 }, { CR72, 0xff },
750 { CR73, 0x32 }, { CR74, 0x30 }, { CR75, 0x65 },
751 { CR76, 0x41 }, { CR77, 0x1b }, { CR78, 0x30 },
752 { CR79, 0xf0 }, { CR80, 0x64 }, { CR81, 0x64 },
753 { CR82, 0x00 }, { CR83, 0x24 }, { CR84, 0x04 },
754 { CR85, 0x00 }, { CR86, 0x0c }, { CR87, 0x12 },
755 { CR88, 0x0c }, { CR89, 0x00 }, { CR90, 0x58 },
756 { CR91, 0x04 }, { CR92, 0x00 }, { CR93, 0x00 },
757 { CR94, 0x01 },
758 { CR95, 0x20 }, /* ZD1211B */
759 { CR96, 0x50 }, { CR97, 0x37 }, { CR98, 0x35 },
760 { CR99, 0x00 }, { CR100, 0x01 }, { CR101, 0x13 },
761 { CR102, 0x27 }, { CR103, 0x27 }, { CR104, 0x18 },
762 { CR105, 0x12 }, { CR106, 0x04 }, { CR107, 0x00 },
763 { CR108, 0x0a }, { CR109, 0x27 }, { CR110, 0x27 },
764 { CR111, 0x27 }, { CR112, 0x27 }, { CR113, 0x27 },
765 { CR114, 0x27 }, { CR115, 0x26 }, { CR116, 0x24 },
766 { CR117, 0xfc }, { CR118, 0xfa }, { CR119, 0x1e },
767 { CR125, 0x90 }, { CR126, 0x00 }, { CR127, 0x00 },
768 { CR128, 0x14 }, { CR129, 0x12 }, { CR130, 0x10 },
769 { CR131, 0x0c }, { CR136, 0xdf }, { CR137, 0xa0 },
770 { CR138, 0xa8 }, { CR139, 0xb4 }, { CR140, 0x98 },
771 { CR141, 0x82 }, { CR142, 0x53 }, { CR143, 0x1c },
772 { CR144, 0x6c }, { CR147, 0x07 }, { CR148, 0x40 },
773 { CR149, 0x40 }, /* Org:0x50 ComTrend:RalLink AP */
774 { CR150, 0x14 }, /* Org:0x0E ComTrend:RalLink AP */
775 { CR151, 0x18 }, { CR159, 0x70 }, { CR160, 0xfe },
776 { CR161, 0xee }, { CR162, 0xaa }, { CR163, 0xfa },
777 { CR164, 0xfa }, { CR165, 0xea }, { CR166, 0xbe },
778 { CR167, 0xbe }, { CR168, 0x6a }, { CR169, 0xba },
779 { CR170, 0xba }, { CR171, 0xba },
780 /* Note: CR204 must lead the CR203 */
781 { CR204, 0x7d },
782 {},
783 { CR203, 0x30 },
784 };
785
786 int r, t;
787
788 dev_dbg_f(zd_chip_dev(chip), "\n");
789
790 r = zd_chip_lock_phy_regs(chip);
791 if (r)
792 goto out;
793
794 r = zd_iowrite16a_locked(chip, ioreqs, ARRAY_SIZE(ioreqs));
Daniel Drakee85d0912006-06-02 17:11:32 +0100795 t = zd_chip_unlock_phy_regs(chip);
796 if (t && !r)
797 r = t;
798out:
799 return r;
800}
801
802static int hw_reset_phy(struct zd_chip *chip)
803{
804 return chip->is_zd1211b ? zd1211b_hw_reset_phy(chip) :
805 zd1211_hw_reset_phy(chip);
806}
807
808static int zd1211_hw_init_hmac(struct zd_chip *chip)
809{
810 static const struct zd_ioreq32 ioreqs[] = {
Daniel Drakee85d0912006-06-02 17:11:32 +0100811 { CR_ZD1211_RETRY_MAX, 0x2 },
Daniel Drakee85d0912006-06-02 17:11:32 +0100812 { CR_RX_THRESHOLD, 0x000c0640 },
Daniel Drakee85d0912006-06-02 17:11:32 +0100813 };
814
Daniel Drakee85d0912006-06-02 17:11:32 +0100815 dev_dbg_f(zd_chip_dev(chip), "\n");
816 ZD_ASSERT(mutex_is_locked(&chip->mutex));
Daniel Drake34c4491262006-12-12 01:25:13 +0000817 return zd_iowrite32a_locked(chip, ioreqs, ARRAY_SIZE(ioreqs));
Daniel Drakee85d0912006-06-02 17:11:32 +0100818}
819
820static int zd1211b_hw_init_hmac(struct zd_chip *chip)
821{
822 static const struct zd_ioreq32 ioreqs[] = {
Daniel Drakee85d0912006-06-02 17:11:32 +0100823 { CR_ZD1211B_RETRY_MAX, 0x02020202 },
824 { CR_ZD1211B_TX_PWR_CTL4, 0x007f003f },
825 { CR_ZD1211B_TX_PWR_CTL3, 0x007f003f },
826 { CR_ZD1211B_TX_PWR_CTL2, 0x003f001f },
827 { CR_ZD1211B_TX_PWR_CTL1, 0x001f000f },
828 { CR_ZD1211B_AIFS_CTL1, 0x00280028 },
829 { CR_ZD1211B_AIFS_CTL2, 0x008C003C },
830 { CR_ZD1211B_TXOP, 0x01800824 },
Daniel Drake34c4491262006-12-12 01:25:13 +0000831 { CR_RX_THRESHOLD, 0x000c0eff, },
832 };
833
834 dev_dbg_f(zd_chip_dev(chip), "\n");
835 ZD_ASSERT(mutex_is_locked(&chip->mutex));
836 return zd_iowrite32a_locked(chip, ioreqs, ARRAY_SIZE(ioreqs));
837}
838
839static int hw_init_hmac(struct zd_chip *chip)
840{
841 int r;
842 static const struct zd_ioreq32 ioreqs[] = {
843 { CR_ACK_TIMEOUT_EXT, 0x20 },
844 { CR_ADDA_MBIAS_WARMTIME, 0x30000808 },
Daniel Drakee85d0912006-06-02 17:11:32 +0100845 { CR_SNIFFER_ON, 0 },
Ulrich Kunitzfde627b2006-08-01 23:43:35 +0200846 { CR_RX_FILTER, STA_RX_FILTER },
Daniel Drakee85d0912006-06-02 17:11:32 +0100847 { CR_GROUP_HASH_P1, 0x00 },
848 { CR_GROUP_HASH_P2, 0x80000000 },
849 { CR_REG1, 0xa4 },
850 { CR_ADDA_PWR_DWN, 0x7f },
851 { CR_BCN_PLCP_CFG, 0x00f00401 },
852 { CR_PHY_DELAY, 0x00 },
853 { CR_ACK_TIMEOUT_EXT, 0x80 },
854 { CR_ADDA_PWR_DWN, 0x00 },
855 { CR_ACK_TIME_80211, 0x100 },
Daniel Drakee85d0912006-06-02 17:11:32 +0100856 { CR_RX_PE_DELAY, 0x70 },
857 { CR_PS_CTRL, 0x10000000 },
858 { CR_RTS_CTS_RATE, 0x02030203 },
Daniel Drakee85d0912006-06-02 17:11:32 +0100859 { CR_AFTER_PNP, 0x1 },
860 { CR_WEP_PROTECT, 0x114 },
Daniel Drake34c4491262006-12-12 01:25:13 +0000861 { CR_IFS_VALUE, IFS_VALUE_DEFAULT },
Daniel Drakee85d0912006-06-02 17:11:32 +0100862 };
863
Daniel Drakee85d0912006-06-02 17:11:32 +0100864 ZD_ASSERT(mutex_is_locked(&chip->mutex));
865 r = zd_iowrite32a_locked(chip, ioreqs, ARRAY_SIZE(ioreqs));
Daniel Drake34c4491262006-12-12 01:25:13 +0000866 if (r)
867 return r;
Daniel Drakee85d0912006-06-02 17:11:32 +0100868
Daniel Drakee85d0912006-06-02 17:11:32 +0100869 return chip->is_zd1211b ?
870 zd1211b_hw_init_hmac(chip) : zd1211_hw_init_hmac(chip);
871}
872
873struct aw_pt_bi {
874 u32 atim_wnd_period;
875 u32 pre_tbtt;
876 u32 beacon_interval;
877};
878
879static int get_aw_pt_bi(struct zd_chip *chip, struct aw_pt_bi *s)
880{
881 int r;
882 static const zd_addr_t aw_pt_bi_addr[] =
883 { CR_ATIM_WND_PERIOD, CR_PRE_TBTT, CR_BCN_INTERVAL };
884 u32 values[3];
885
886 r = zd_ioread32v_locked(chip, values, (const zd_addr_t *)aw_pt_bi_addr,
887 ARRAY_SIZE(aw_pt_bi_addr));
888 if (r) {
889 memset(s, 0, sizeof(*s));
890 return r;
891 }
892
893 s->atim_wnd_period = values[0];
894 s->pre_tbtt = values[1];
895 s->beacon_interval = values[2];
896 dev_dbg_f(zd_chip_dev(chip), "aw %u pt %u bi %u\n",
897 s->atim_wnd_period, s->pre_tbtt, s->beacon_interval);
898 return 0;
899}
900
901static int set_aw_pt_bi(struct zd_chip *chip, struct aw_pt_bi *s)
902{
903 struct zd_ioreq32 reqs[3];
904
905 if (s->beacon_interval <= 5)
906 s->beacon_interval = 5;
907 if (s->pre_tbtt < 4 || s->pre_tbtt >= s->beacon_interval)
908 s->pre_tbtt = s->beacon_interval - 1;
909 if (s->atim_wnd_period >= s->pre_tbtt)
910 s->atim_wnd_period = s->pre_tbtt - 1;
911
912 reqs[0].addr = CR_ATIM_WND_PERIOD;
913 reqs[0].value = s->atim_wnd_period;
914 reqs[1].addr = CR_PRE_TBTT;
915 reqs[1].value = s->pre_tbtt;
916 reqs[2].addr = CR_BCN_INTERVAL;
917 reqs[2].value = s->beacon_interval;
918
919 dev_dbg_f(zd_chip_dev(chip),
920 "aw %u pt %u bi %u\n", s->atim_wnd_period, s->pre_tbtt,
921 s->beacon_interval);
922 return zd_iowrite32a_locked(chip, reqs, ARRAY_SIZE(reqs));
923}
924
925
926static int set_beacon_interval(struct zd_chip *chip, u32 interval)
927{
928 int r;
929 struct aw_pt_bi s;
930
931 ZD_ASSERT(mutex_is_locked(&chip->mutex));
932 r = get_aw_pt_bi(chip, &s);
933 if (r)
934 return r;
935 s.beacon_interval = interval;
936 return set_aw_pt_bi(chip, &s);
937}
938
939int zd_set_beacon_interval(struct zd_chip *chip, u32 interval)
940{
941 int r;
942
943 mutex_lock(&chip->mutex);
944 r = set_beacon_interval(chip, interval);
945 mutex_unlock(&chip->mutex);
946 return r;
947}
948
949static int hw_init(struct zd_chip *chip)
950{
951 int r;
952
953 dev_dbg_f(zd_chip_dev(chip), "\n");
954 ZD_ASSERT(mutex_is_locked(&chip->mutex));
955 r = hw_reset_phy(chip);
956 if (r)
957 return r;
958
959 r = hw_init_hmac(chip);
960 if (r)
961 return r;
Daniel Drake98227a92006-08-12 17:59:22 +0100962
Daniel Drake98227a92006-08-12 17:59:22 +0100963 return set_beacon_interval(chip, 100);
Daniel Drakee85d0912006-06-02 17:11:32 +0100964}
965
Daniel Drake0ce34bc2006-12-12 01:26:11 +0000966static zd_addr_t fw_reg_addr(struct zd_chip *chip, u16 offset)
967{
968 return (zd_addr_t)((u16)chip->fw_regs_base + offset);
969}
970
Daniel Drakee85d0912006-06-02 17:11:32 +0100971#ifdef DEBUG
972static int dump_cr(struct zd_chip *chip, const zd_addr_t addr,
973 const char *addr_string)
974{
975 int r;
976 u32 value;
977
978 r = zd_ioread32_locked(chip, &value, addr);
979 if (r) {
980 dev_dbg_f(zd_chip_dev(chip),
981 "error reading %s. Error number %d\n", addr_string, r);
982 return r;
983 }
984
985 dev_dbg_f(zd_chip_dev(chip), "%s %#010x\n",
986 addr_string, (unsigned int)value);
987 return 0;
988}
989
990static int test_init(struct zd_chip *chip)
991{
992 int r;
993
994 r = dump_cr(chip, CR_AFTER_PNP, "CR_AFTER_PNP");
995 if (r)
996 return r;
997 r = dump_cr(chip, CR_GPI_EN, "CR_GPI_EN");
998 if (r)
999 return r;
1000 return dump_cr(chip, CR_INTERRUPT, "CR_INTERRUPT");
1001}
1002
1003static void dump_fw_registers(struct zd_chip *chip)
1004{
Daniel Drake0ce34bc2006-12-12 01:26:11 +00001005 const zd_addr_t addr[4] = {
1006 fw_reg_addr(chip, FW_REG_FIRMWARE_VER),
1007 fw_reg_addr(chip, FW_REG_USB_SPEED),
1008 fw_reg_addr(chip, FW_REG_FIX_TX_RATE),
1009 fw_reg_addr(chip, FW_REG_LED_LINK_STATUS),
Daniel Drakee85d0912006-06-02 17:11:32 +01001010 };
1011
1012 int r;
1013 u16 values[4];
1014
1015 r = zd_ioread16v_locked(chip, values, (const zd_addr_t*)addr,
1016 ARRAY_SIZE(addr));
1017 if (r) {
1018 dev_dbg_f(zd_chip_dev(chip), "error %d zd_ioread16v_locked\n",
1019 r);
1020 return;
1021 }
1022
1023 dev_dbg_f(zd_chip_dev(chip), "FW_FIRMWARE_VER %#06hx\n", values[0]);
1024 dev_dbg_f(zd_chip_dev(chip), "FW_USB_SPEED %#06hx\n", values[1]);
1025 dev_dbg_f(zd_chip_dev(chip), "FW_FIX_TX_RATE %#06hx\n", values[2]);
1026 dev_dbg_f(zd_chip_dev(chip), "FW_LINK_STATUS %#06hx\n", values[3]);
1027}
1028#endif /* DEBUG */
1029
1030static int print_fw_version(struct zd_chip *chip)
1031{
1032 int r;
1033 u16 version;
1034
Daniel Drake0ce34bc2006-12-12 01:26:11 +00001035 r = zd_ioread16_locked(chip, &version,
1036 fw_reg_addr(chip, FW_REG_FIRMWARE_VER));
Daniel Drakee85d0912006-06-02 17:11:32 +01001037 if (r)
1038 return r;
1039
1040 dev_info(zd_chip_dev(chip),"firmware version %04hx\n", version);
1041 return 0;
1042}
1043
1044static int set_mandatory_rates(struct zd_chip *chip, enum ieee80211_std std)
1045{
1046 u32 rates;
1047 ZD_ASSERT(mutex_is_locked(&chip->mutex));
1048 /* This sets the mandatory rates, which only depend from the standard
1049 * that the device is supporting. Until further notice we should try
1050 * to support 802.11g also for full speed USB.
1051 */
1052 switch (std) {
1053 case IEEE80211B:
1054 rates = CR_RATE_1M|CR_RATE_2M|CR_RATE_5_5M|CR_RATE_11M;
1055 break;
1056 case IEEE80211G:
1057 rates = CR_RATE_1M|CR_RATE_2M|CR_RATE_5_5M|CR_RATE_11M|
1058 CR_RATE_6M|CR_RATE_12M|CR_RATE_24M;
1059 break;
1060 default:
1061 return -EINVAL;
1062 }
1063 return zd_iowrite32_locked(chip, rates, CR_MANDATORY_RATE_TBL);
1064}
1065
Daniel Drakeb1382ed2006-11-22 00:06:48 +00001066int zd_chip_set_rts_cts_rate_locked(struct zd_chip *chip,
1067 u8 rts_rate, int preamble)
1068{
1069 int rts_mod = ZD_RX_CCK;
1070 u32 value = 0;
1071
1072 /* Modulation bit */
1073 if (ZD_CS_TYPE(rts_rate) == ZD_CS_OFDM)
1074 rts_mod = ZD_RX_OFDM;
1075
1076 dev_dbg_f(zd_chip_dev(chip), "rts_rate=%x preamble=%x\n",
1077 rts_rate, preamble);
1078
1079 value |= rts_rate << RTSCTS_SH_RTS_RATE;
1080 value |= rts_mod << RTSCTS_SH_RTS_MOD_TYPE;
1081 value |= preamble << RTSCTS_SH_RTS_PMB_TYPE;
1082 value |= preamble << RTSCTS_SH_CTS_PMB_TYPE;
1083
1084 /* We always send 11M self-CTS messages, like the vendor driver. */
1085 value |= ZD_CCK_RATE_11M << RTSCTS_SH_CTS_RATE;
1086 value |= ZD_RX_CCK << RTSCTS_SH_CTS_MOD_TYPE;
1087
1088 return zd_iowrite32_locked(chip, value, CR_RTS_CTS_RATE);
1089}
1090
Daniel Drakee85d0912006-06-02 17:11:32 +01001091int zd_chip_enable_hwint(struct zd_chip *chip)
1092{
1093 int r;
1094
1095 mutex_lock(&chip->mutex);
1096 r = zd_iowrite32_locked(chip, HWINT_ENABLED, CR_INTERRUPT);
1097 mutex_unlock(&chip->mutex);
1098 return r;
1099}
1100
1101static int disable_hwint(struct zd_chip *chip)
1102{
1103 return zd_iowrite32_locked(chip, HWINT_DISABLED, CR_INTERRUPT);
1104}
1105
1106int zd_chip_disable_hwint(struct zd_chip *chip)
1107{
1108 int r;
1109
1110 mutex_lock(&chip->mutex);
1111 r = disable_hwint(chip);
1112 mutex_unlock(&chip->mutex);
1113 return r;
1114}
1115
Daniel Drake0ce34bc2006-12-12 01:26:11 +00001116static int read_fw_regs_offset(struct zd_chip *chip)
1117{
1118 int r;
1119
1120 ZD_ASSERT(mutex_is_locked(&chip->mutex));
1121 r = zd_ioread16_locked(chip, (u16*)&chip->fw_regs_base,
1122 FWRAW_REGS_ADDR);
1123 if (r)
1124 return r;
1125 dev_dbg_f(zd_chip_dev(chip), "fw_regs_base: %#06hx\n",
1126 (u16)chip->fw_regs_base);
1127
1128 return 0;
1129}
1130
1131
Daniel Drakee85d0912006-06-02 17:11:32 +01001132int zd_chip_init_hw(struct zd_chip *chip, u8 device_type)
1133{
1134 int r;
1135 u8 rf_type;
1136
1137 dev_dbg_f(zd_chip_dev(chip), "\n");
1138
1139 mutex_lock(&chip->mutex);
1140 chip->is_zd1211b = (device_type == DEVICE_ZD1211B) != 0;
1141
1142#ifdef DEBUG
1143 r = test_init(chip);
1144 if (r)
1145 goto out;
1146#endif
1147 r = zd_iowrite32_locked(chip, 1, CR_AFTER_PNP);
1148 if (r)
1149 goto out;
1150
Daniel Drake0ce34bc2006-12-12 01:26:11 +00001151 r = read_fw_regs_offset(chip);
Daniel Drakee85d0912006-06-02 17:11:32 +01001152 if (r)
1153 goto out;
1154
1155 /* GPI is always disabled, also in the other driver.
1156 */
1157 r = zd_iowrite32_locked(chip, 0, CR_GPI_EN);
1158 if (r)
1159 goto out;
1160 r = zd_iowrite32_locked(chip, CWIN_SIZE, CR_CWMIN_CWMAX);
1161 if (r)
1162 goto out;
1163 /* Currently we support IEEE 802.11g for full and high speed USB.
1164 * It might be discussed, whether we should suppport pure b mode for
1165 * full speed USB.
1166 */
1167 r = set_mandatory_rates(chip, IEEE80211G);
1168 if (r)
1169 goto out;
1170 /* Disabling interrupts is certainly a smart thing here.
1171 */
1172 r = disable_hwint(chip);
1173 if (r)
1174 goto out;
1175 r = read_pod(chip, &rf_type);
1176 if (r)
1177 goto out;
1178 r = hw_init(chip);
1179 if (r)
1180 goto out;
1181 r = zd_rf_init_hw(&chip->rf, rf_type);
1182 if (r)
1183 goto out;
1184
1185 r = print_fw_version(chip);
1186 if (r)
1187 goto out;
1188
1189#ifdef DEBUG
1190 dump_fw_registers(chip);
1191 r = test_init(chip);
1192 if (r)
1193 goto out;
1194#endif /* DEBUG */
1195
1196 r = read_e2p_mac_addr(chip);
1197 if (r)
1198 goto out;
1199
1200 r = read_cal_int_tables(chip);
1201 if (r)
1202 goto out;
1203
1204 print_id(chip);
1205out:
1206 mutex_unlock(&chip->mutex);
1207 return r;
1208}
1209
1210static int update_pwr_int(struct zd_chip *chip, u8 channel)
1211{
1212 u8 value = chip->pwr_int_values[channel - 1];
1213 dev_dbg_f(zd_chip_dev(chip), "channel %d pwr_int %#04x\n",
1214 channel, value);
Ulrich Kunitzcbb5e6b2006-09-13 02:41:02 +01001215 return zd_iowrite16_locked(chip, value, CR31);
Daniel Drakee85d0912006-06-02 17:11:32 +01001216}
1217
1218static int update_pwr_cal(struct zd_chip *chip, u8 channel)
1219{
1220 u8 value = chip->pwr_cal_values[channel-1];
1221 dev_dbg_f(zd_chip_dev(chip), "channel %d pwr_cal %#04x\n",
1222 channel, value);
Ulrich Kunitzcbb5e6b2006-09-13 02:41:02 +01001223 return zd_iowrite16_locked(chip, value, CR68);
Daniel Drakee85d0912006-06-02 17:11:32 +01001224}
1225
1226static int update_ofdm_cal(struct zd_chip *chip, u8 channel)
1227{
Ulrich Kunitzcbb5e6b2006-09-13 02:41:02 +01001228 struct zd_ioreq16 ioreqs[3];
Daniel Drakee85d0912006-06-02 17:11:32 +01001229
1230 ioreqs[0].addr = CR67;
1231 ioreqs[0].value = chip->ofdm_cal_values[OFDM_36M_INDEX][channel-1];
1232 ioreqs[1].addr = CR66;
1233 ioreqs[1].value = chip->ofdm_cal_values[OFDM_48M_INDEX][channel-1];
1234 ioreqs[2].addr = CR65;
1235 ioreqs[2].value = chip->ofdm_cal_values[OFDM_54M_INDEX][channel-1];
1236
1237 dev_dbg_f(zd_chip_dev(chip),
1238 "channel %d ofdm_cal 36M %#04x 48M %#04x 54M %#04x\n",
1239 channel, ioreqs[0].value, ioreqs[1].value, ioreqs[2].value);
Ulrich Kunitzcbb5e6b2006-09-13 02:41:02 +01001240 return zd_iowrite16a_locked(chip, ioreqs, ARRAY_SIZE(ioreqs));
Daniel Drakee85d0912006-06-02 17:11:32 +01001241}
1242
1243static int update_channel_integration_and_calibration(struct zd_chip *chip,
1244 u8 channel)
1245{
1246 int r;
1247
1248 r = update_pwr_int(chip, channel);
1249 if (r)
1250 return r;
1251 if (chip->is_zd1211b) {
Ulrich Kunitzcbb5e6b2006-09-13 02:41:02 +01001252 static const struct zd_ioreq16 ioreqs[] = {
Daniel Drakee85d0912006-06-02 17:11:32 +01001253 { CR69, 0x28 },
1254 {},
1255 { CR69, 0x2a },
1256 };
1257
1258 r = update_ofdm_cal(chip, channel);
1259 if (r)
1260 return r;
1261 r = update_pwr_cal(chip, channel);
1262 if (r)
1263 return r;
Ulrich Kunitzcbb5e6b2006-09-13 02:41:02 +01001264 r = zd_iowrite16a_locked(chip, ioreqs, ARRAY_SIZE(ioreqs));
Daniel Drakee85d0912006-06-02 17:11:32 +01001265 if (r)
1266 return r;
1267 }
1268
1269 return 0;
1270}
1271
1272/* The CCK baseband gain can be optionally patched by the EEPROM */
1273static int patch_cck_gain(struct zd_chip *chip)
1274{
1275 int r;
1276 u32 value;
1277
1278 if (!chip->patch_cck_gain)
1279 return 0;
1280
1281 ZD_ASSERT(mutex_is_locked(&chip->mutex));
1282 r = zd_ioread32_locked(chip, &value, E2P_PHY_REG);
1283 if (r)
1284 return r;
1285 dev_dbg_f(zd_chip_dev(chip), "patching value %x\n", value & 0xff);
Ulrich Kunitzcbb5e6b2006-09-13 02:41:02 +01001286 return zd_iowrite16_locked(chip, value & 0xff, CR47);
Daniel Drakee85d0912006-06-02 17:11:32 +01001287}
1288
1289int zd_chip_set_channel(struct zd_chip *chip, u8 channel)
1290{
1291 int r, t;
1292
1293 mutex_lock(&chip->mutex);
1294 r = zd_chip_lock_phy_regs(chip);
1295 if (r)
1296 goto out;
1297 r = zd_rf_set_channel(&chip->rf, channel);
1298 if (r)
1299 goto unlock;
1300 r = update_channel_integration_and_calibration(chip, channel);
1301 if (r)
1302 goto unlock;
1303 r = patch_cck_gain(chip);
1304 if (r)
1305 goto unlock;
1306 r = patch_6m_band_edge(chip, channel);
1307 if (r)
1308 goto unlock;
1309 r = zd_iowrite32_locked(chip, 0, CR_CONFIG_PHILIPS);
1310unlock:
1311 t = zd_chip_unlock_phy_regs(chip);
1312 if (t && !r)
1313 r = t;
1314out:
1315 mutex_unlock(&chip->mutex);
1316 return r;
1317}
1318
1319u8 zd_chip_get_channel(struct zd_chip *chip)
1320{
1321 u8 channel;
1322
1323 mutex_lock(&chip->mutex);
1324 channel = chip->rf.channel;
1325 mutex_unlock(&chip->mutex);
1326 return channel;
1327}
1328
Ulrich Kunitz583afd12006-09-13 02:42:38 +01001329int zd_chip_control_leds(struct zd_chip *chip, enum led_status status)
Daniel Drakee85d0912006-06-02 17:11:32 +01001330{
Daniel Drake0ce34bc2006-12-12 01:26:11 +00001331 const zd_addr_t a[] = {
1332 fw_reg_addr(chip, FW_REG_LED_LINK_STATUS),
Ulrich Kunitz583afd12006-09-13 02:42:38 +01001333 CR_LED,
1334 };
Daniel Drakee85d0912006-06-02 17:11:32 +01001335
Ulrich Kunitz583afd12006-09-13 02:42:38 +01001336 int r;
1337 u16 v[ARRAY_SIZE(a)];
1338 struct zd_ioreq16 ioreqs[ARRAY_SIZE(a)] = {
Daniel Drake0ce34bc2006-12-12 01:26:11 +00001339 [0] = { fw_reg_addr(chip, FW_REG_LED_LINK_STATUS) },
Ulrich Kunitz583afd12006-09-13 02:42:38 +01001340 [1] = { CR_LED },
1341 };
1342 u16 other_led;
Daniel Drakee85d0912006-06-02 17:11:32 +01001343
Daniel Drakee85d0912006-06-02 17:11:32 +01001344 mutex_lock(&chip->mutex);
Ulrich Kunitz583afd12006-09-13 02:42:38 +01001345 r = zd_ioread16v_locked(chip, v, (const zd_addr_t *)a, ARRAY_SIZE(a));
Daniel Drakee85d0912006-06-02 17:11:32 +01001346 if (r)
Ulrich Kunitz583afd12006-09-13 02:42:38 +01001347 goto out;
1348
1349 other_led = chip->link_led == LED1 ? LED2 : LED1;
1350
Daniel Drakee85d0912006-06-02 17:11:32 +01001351 switch (status) {
Daniel Drakee85d0912006-06-02 17:11:32 +01001352 case LED_OFF:
Ulrich Kunitz583afd12006-09-13 02:42:38 +01001353 ioreqs[0].value = FW_LINK_OFF;
1354 ioreqs[1].value = v[1] & ~(LED1|LED2);
Daniel Drakee85d0912006-06-02 17:11:32 +01001355 break;
Ulrich Kunitz583afd12006-09-13 02:42:38 +01001356 case LED_SCANNING:
1357 ioreqs[0].value = FW_LINK_OFF;
1358 ioreqs[1].value = v[1] & ~other_led;
1359 if (get_seconds() % 3 == 0) {
1360 ioreqs[1].value &= ~chip->link_led;
1361 } else {
1362 ioreqs[1].value |= chip->link_led;
1363 }
Daniel Drakee85d0912006-06-02 17:11:32 +01001364 break;
Ulrich Kunitz583afd12006-09-13 02:42:38 +01001365 case LED_ASSOCIATED:
1366 ioreqs[0].value = FW_LINK_TX;
1367 ioreqs[1].value = v[1] & ~other_led;
1368 ioreqs[1].value |= chip->link_led;
Daniel Drakee85d0912006-06-02 17:11:32 +01001369 break;
1370 default:
Ulrich Kunitz583afd12006-09-13 02:42:38 +01001371 r = -EINVAL;
Daniel Drakee85d0912006-06-02 17:11:32 +01001372 goto out;
1373 }
Ulrich Kunitz583afd12006-09-13 02:42:38 +01001374
1375 if (v[0] != ioreqs[0].value || v[1] != ioreqs[1].value) {
1376 r = zd_iowrite16a_locked(chip, ioreqs, ARRAY_SIZE(ioreqs));
1377 if (r)
1378 goto out;
1379 }
1380 r = 0;
Daniel Drakee85d0912006-06-02 17:11:32 +01001381out:
1382 mutex_unlock(&chip->mutex);
1383 return r;
1384}
1385
Daniel Drakeb1382ed2006-11-22 00:06:48 +00001386int zd_chip_set_basic_rates_locked(struct zd_chip *chip, u16 cr_rates)
Daniel Drakee85d0912006-06-02 17:11:32 +01001387{
Daniel Drakeb1382ed2006-11-22 00:06:48 +00001388 ZD_ASSERT((cr_rates & ~(CR_RATES_80211B | CR_RATES_80211G)) == 0);
1389 dev_dbg_f(zd_chip_dev(chip), "%x\n", cr_rates);
Daniel Drakee85d0912006-06-02 17:11:32 +01001390
Daniel Drakeb1382ed2006-11-22 00:06:48 +00001391 return zd_iowrite32_locked(chip, cr_rates, CR_BASIC_RATE_TBL);
Daniel Drakee85d0912006-06-02 17:11:32 +01001392}
1393
1394static int ofdm_qual_db(u8 status_quality, u8 rate, unsigned int size)
1395{
1396 static const u16 constants[] = {
1397 715, 655, 585, 540, 470, 410, 360, 315,
1398 270, 235, 205, 175, 150, 125, 105, 85,
1399 65, 50, 40, 25, 15
1400 };
1401
1402 int i;
1403 u32 x;
1404
1405 /* It seems that their quality parameter is somehow per signal
1406 * and is now transferred per bit.
1407 */
1408 switch (rate) {
1409 case ZD_OFDM_RATE_6M:
1410 case ZD_OFDM_RATE_12M:
1411 case ZD_OFDM_RATE_24M:
1412 size *= 2;
1413 break;
1414 case ZD_OFDM_RATE_9M:
1415 case ZD_OFDM_RATE_18M:
1416 case ZD_OFDM_RATE_36M:
1417 case ZD_OFDM_RATE_54M:
1418 size *= 4;
1419 size /= 3;
1420 break;
1421 case ZD_OFDM_RATE_48M:
1422 size *= 3;
1423 size /= 2;
1424 break;
1425 default:
1426 return -EINVAL;
1427 }
1428
1429 x = (10000 * status_quality)/size;
1430 for (i = 0; i < ARRAY_SIZE(constants); i++) {
1431 if (x > constants[i])
1432 break;
1433 }
1434
Ulrich Kunitzdb888ae2006-08-29 23:50:29 +01001435 switch (rate) {
1436 case ZD_OFDM_RATE_6M:
1437 case ZD_OFDM_RATE_9M:
1438 i += 3;
1439 break;
1440 case ZD_OFDM_RATE_12M:
1441 case ZD_OFDM_RATE_18M:
1442 i += 5;
1443 break;
1444 case ZD_OFDM_RATE_24M:
1445 case ZD_OFDM_RATE_36M:
1446 i += 9;
1447 break;
1448 case ZD_OFDM_RATE_48M:
1449 case ZD_OFDM_RATE_54M:
1450 i += 15;
1451 break;
1452 default:
1453 return -EINVAL;
1454 }
1455
Daniel Drakee85d0912006-06-02 17:11:32 +01001456 return i;
1457}
1458
Ulrich Kunitzdb888ae2006-08-29 23:50:29 +01001459static int ofdm_qual_percent(u8 status_quality, u8 rate, unsigned int size)
1460{
1461 int r;
1462
1463 r = ofdm_qual_db(status_quality, rate, size);
1464 ZD_ASSERT(r >= 0);
1465 if (r < 0)
1466 r = 0;
1467
1468 r = (r * 100)/29;
1469 return r <= 100 ? r : 100;
1470}
1471
Daniel Drakee85d0912006-06-02 17:11:32 +01001472static unsigned int log10times100(unsigned int x)
1473{
1474 static const u8 log10[] = {
1475 0,
1476 0, 30, 47, 60, 69, 77, 84, 90, 95, 100,
1477 104, 107, 111, 114, 117, 120, 123, 125, 127, 130,
1478 132, 134, 136, 138, 139, 141, 143, 144, 146, 147,
1479 149, 150, 151, 153, 154, 155, 156, 157, 159, 160,
1480 161, 162, 163, 164, 165, 166, 167, 168, 169, 169,
1481 170, 171, 172, 173, 174, 174, 175, 176, 177, 177,
1482 178, 179, 179, 180, 181, 181, 182, 183, 183, 184,
1483 185, 185, 186, 186, 187, 188, 188, 189, 189, 190,
1484 190, 191, 191, 192, 192, 193, 193, 194, 194, 195,
1485 195, 196, 196, 197, 197, 198, 198, 199, 199, 200,
1486 200, 200, 201, 201, 202, 202, 202, 203, 203, 204,
1487 204, 204, 205, 205, 206, 206, 206, 207, 207, 207,
1488 208, 208, 208, 209, 209, 210, 210, 210, 211, 211,
1489 211, 212, 212, 212, 213, 213, 213, 213, 214, 214,
1490 214, 215, 215, 215, 216, 216, 216, 217, 217, 217,
1491 217, 218, 218, 218, 219, 219, 219, 219, 220, 220,
1492 220, 220, 221, 221, 221, 222, 222, 222, 222, 223,
1493 223, 223, 223, 224, 224, 224, 224,
1494 };
1495
1496 return x < ARRAY_SIZE(log10) ? log10[x] : 225;
1497}
1498
1499enum {
1500 MAX_CCK_EVM_DB = 45,
1501};
1502
1503static int cck_evm_db(u8 status_quality)
1504{
1505 return (20 * log10times100(status_quality)) / 100;
1506}
1507
1508static int cck_snr_db(u8 status_quality)
1509{
1510 int r = MAX_CCK_EVM_DB - cck_evm_db(status_quality);
1511 ZD_ASSERT(r >= 0);
1512 return r;
1513}
1514
Ulrich Kunitzdb888ae2006-08-29 23:50:29 +01001515static int cck_qual_percent(u8 status_quality)
Daniel Drakee85d0912006-06-02 17:11:32 +01001516{
Ulrich Kunitzdb888ae2006-08-29 23:50:29 +01001517 int r;
1518
1519 r = cck_snr_db(status_quality);
1520 r = (100*r)/17;
1521 return r <= 100 ? r : 100;
Daniel Drakee85d0912006-06-02 17:11:32 +01001522}
1523
1524u8 zd_rx_qual_percent(const void *rx_frame, unsigned int size,
1525 const struct rx_status *status)
1526{
Ulrich Kunitzdb888ae2006-08-29 23:50:29 +01001527 return (status->frame_status&ZD_RX_OFDM) ?
1528 ofdm_qual_percent(status->signal_quality_ofdm,
1529 zd_ofdm_plcp_header_rate(rx_frame),
1530 size) :
1531 cck_qual_percent(status->signal_quality_cck);
Daniel Drakee85d0912006-06-02 17:11:32 +01001532}
1533
1534u8 zd_rx_strength_percent(u8 rssi)
1535{
Ulrich Kunitzdb888ae2006-08-29 23:50:29 +01001536 int r = (rssi*100) / 41;
Daniel Drakee85d0912006-06-02 17:11:32 +01001537 if (r > 100)
1538 r = 100;
1539 return (u8) r;
1540}
1541
1542u16 zd_rx_rate(const void *rx_frame, const struct rx_status *status)
1543{
1544 static const u16 ofdm_rates[] = {
1545 [ZD_OFDM_RATE_6M] = 60,
1546 [ZD_OFDM_RATE_9M] = 90,
1547 [ZD_OFDM_RATE_12M] = 120,
1548 [ZD_OFDM_RATE_18M] = 180,
1549 [ZD_OFDM_RATE_24M] = 240,
1550 [ZD_OFDM_RATE_36M] = 360,
1551 [ZD_OFDM_RATE_48M] = 480,
1552 [ZD_OFDM_RATE_54M] = 540,
1553 };
1554 u16 rate;
1555 if (status->frame_status & ZD_RX_OFDM) {
1556 u8 ofdm_rate = zd_ofdm_plcp_header_rate(rx_frame);
1557 rate = ofdm_rates[ofdm_rate & 0xf];
1558 } else {
1559 u8 cck_rate = zd_cck_plcp_header_rate(rx_frame);
1560 switch (cck_rate) {
1561 case ZD_CCK_SIGNAL_1M:
1562 rate = 10;
1563 break;
1564 case ZD_CCK_SIGNAL_2M:
1565 rate = 20;
1566 break;
1567 case ZD_CCK_SIGNAL_5M5:
1568 rate = 55;
1569 break;
1570 case ZD_CCK_SIGNAL_11M:
1571 rate = 110;
1572 break;
1573 default:
1574 rate = 0;
1575 }
1576 }
1577
1578 return rate;
1579}
1580
1581int zd_chip_switch_radio_on(struct zd_chip *chip)
1582{
1583 int r;
1584
1585 mutex_lock(&chip->mutex);
1586 r = zd_switch_radio_on(&chip->rf);
1587 mutex_unlock(&chip->mutex);
1588 return r;
1589}
1590
1591int zd_chip_switch_radio_off(struct zd_chip *chip)
1592{
1593 int r;
1594
1595 mutex_lock(&chip->mutex);
1596 r = zd_switch_radio_off(&chip->rf);
1597 mutex_unlock(&chip->mutex);
1598 return r;
1599}
1600
1601int zd_chip_enable_int(struct zd_chip *chip)
1602{
1603 int r;
1604
1605 mutex_lock(&chip->mutex);
1606 r = zd_usb_enable_int(&chip->usb);
1607 mutex_unlock(&chip->mutex);
1608 return r;
1609}
1610
1611void zd_chip_disable_int(struct zd_chip *chip)
1612{
1613 mutex_lock(&chip->mutex);
1614 zd_usb_disable_int(&chip->usb);
1615 mutex_unlock(&chip->mutex);
1616}
1617
1618int zd_chip_enable_rx(struct zd_chip *chip)
1619{
1620 int r;
1621
1622 mutex_lock(&chip->mutex);
1623 r = zd_usb_enable_rx(&chip->usb);
1624 mutex_unlock(&chip->mutex);
1625 return r;
1626}
1627
1628void zd_chip_disable_rx(struct zd_chip *chip)
1629{
1630 mutex_lock(&chip->mutex);
1631 zd_usb_disable_rx(&chip->usb);
1632 mutex_unlock(&chip->mutex);
1633}
1634
1635int zd_rfwritev_locked(struct zd_chip *chip,
1636 const u32* values, unsigned int count, u8 bits)
1637{
1638 int r;
1639 unsigned int i;
1640
1641 for (i = 0; i < count; i++) {
1642 r = zd_rfwrite_locked(chip, values[i], bits);
1643 if (r)
1644 return r;
1645 }
1646
1647 return 0;
1648}
Daniel Drake20fe2172006-08-12 17:59:42 +01001649
1650/*
1651 * We can optionally program the RF directly through CR regs, if supported by
1652 * the hardware. This is much faster than the older method.
1653 */
Daniel Drakeec62bd92006-08-12 17:59:46 +01001654int zd_rfwrite_cr_locked(struct zd_chip *chip, u32 value)
Daniel Drake20fe2172006-08-12 17:59:42 +01001655{
1656 struct zd_ioreq16 ioreqs[] = {
1657 { CR244, (value >> 16) & 0xff },
1658 { CR243, (value >> 8) & 0xff },
1659 { CR242, value & 0xff },
1660 };
1661 ZD_ASSERT(mutex_is_locked(&chip->mutex));
1662 return zd_iowrite16a_locked(chip, ioreqs, ARRAY_SIZE(ioreqs));
1663}
1664
1665int zd_rfwritev_cr_locked(struct zd_chip *chip,
1666 const u32 *values, unsigned int count)
1667{
1668 int r;
1669 unsigned int i;
1670
1671 for (i = 0; i < count; i++) {
1672 r = zd_rfwrite_cr_locked(chip, values[i]);
1673 if (r)
1674 return r;
1675 }
1676
1677 return 0;
1678}
Ulrich Kunitz9cdac962006-12-01 00:58:07 +00001679
1680int zd_chip_set_multicast_hash(struct zd_chip *chip,
1681 struct zd_mc_hash *hash)
1682{
1683 struct zd_ioreq32 ioreqs[] = {
1684 { CR_GROUP_HASH_P1, hash->low },
1685 { CR_GROUP_HASH_P2, hash->high },
1686 };
1687
1688 dev_dbg_f(zd_chip_dev(chip), "hash l 0x%08x h 0x%08x\n",
1689 ioreqs[0].value, ioreqs[1].value);
1690 return zd_iowrite32a(chip, ioreqs, ARRAY_SIZE(ioreqs));
1691}