| /* ZD1211 USB-WLAN driver for Linux | 
 |  * | 
 |  * Copyright (C) 2005-2007 Ulrich Kunitz <kune@deine-taler.de> | 
 |  * Copyright (C) 2006-2007 Daniel Drake <dsd@gentoo.org> | 
 |  * | 
 |  * This program is free software; you can redistribute it and/or modify | 
 |  * it under the terms of the GNU General Public License as published by | 
 |  * the Free Software Foundation; either version 2 of the License, or | 
 |  * (at your option) any later version. | 
 |  * | 
 |  * This program is distributed in the hope that it will be useful, | 
 |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
 |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
 |  * GNU General Public License for more details. | 
 |  * | 
 |  * You should have received a copy of the GNU General Public License | 
 |  * along with this program; if not, write to the Free Software | 
 |  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA | 
 |  */ | 
 |  | 
 | #include <linux/kernel.h> | 
 |  | 
 | #include "zd_rf.h" | 
 | #include "zd_usb.h" | 
 | #include "zd_chip.h" | 
 |  | 
 | static const u32 rf2959_table[][2] = { | 
 | 	RF_CHANNEL( 1) = { 0x181979, 0x1e6666 }, | 
 | 	RF_CHANNEL( 2) = { 0x181989, 0x1e6666 }, | 
 | 	RF_CHANNEL( 3) = { 0x181999, 0x1e6666 }, | 
 | 	RF_CHANNEL( 4) = { 0x1819a9, 0x1e6666 }, | 
 | 	RF_CHANNEL( 5) = { 0x1819b9, 0x1e6666 }, | 
 | 	RF_CHANNEL( 6) = { 0x1819c9, 0x1e6666 }, | 
 | 	RF_CHANNEL( 7) = { 0x1819d9, 0x1e6666 }, | 
 | 	RF_CHANNEL( 8) = { 0x1819e9, 0x1e6666 }, | 
 | 	RF_CHANNEL( 9) = { 0x1819f9, 0x1e6666 }, | 
 | 	RF_CHANNEL(10) = { 0x181a09, 0x1e6666 }, | 
 | 	RF_CHANNEL(11) = { 0x181a19, 0x1e6666 }, | 
 | 	RF_CHANNEL(12) = { 0x181a29, 0x1e6666 }, | 
 | 	RF_CHANNEL(13) = { 0x181a39, 0x1e6666 }, | 
 | 	RF_CHANNEL(14) = { 0x181a60, 0x1c0000 }, | 
 | }; | 
 |  | 
 | #if 0 | 
 | static int bits(u32 rw, int from, int to) | 
 | { | 
 | 	rw &= ~(0xffffffffU << (to+1)); | 
 | 	rw >>= from; | 
 | 	return rw; | 
 | } | 
 |  | 
 | static int bit(u32 rw, int bit) | 
 | { | 
 | 	return bits(rw, bit, bit); | 
 | } | 
 |  | 
 | static void dump_regwrite(u32 rw) | 
 | { | 
 | 	int reg = bits(rw, 18, 22); | 
 | 	int rw_flag = bits(rw, 23, 23); | 
 | 	PDEBUG("rf2959 %#010x reg %d rw %d", rw, reg, rw_flag); | 
 |  | 
 | 	switch (reg) { | 
 | 	case 0: | 
 | 		PDEBUG("reg0 CFG1 ref_sel %d hybernate %d rf_vco_reg_en %d" | 
 | 		       " if_vco_reg_en %d if_vga_en %d", | 
 | 		       bits(rw, 14, 15), bit(rw, 3), bit(rw, 2), bit(rw, 1), | 
 | 		       bit(rw, 0)); | 
 | 		break; | 
 | 	case 1: | 
 | 		PDEBUG("reg1 IFPLL1 pll_en1 %d kv_en1 %d vtc_en1 %d lpf1 %d" | 
 | 		       " cpl1 %d pdp1 %d autocal_en1 %d ld_en1 %d ifloopr %d" | 
 | 		       " ifloopc %d dac1 %d", | 
 | 		       bit(rw, 17), bit(rw, 16), bit(rw, 15), bit(rw, 14), | 
 | 		       bit(rw, 13), bit(rw, 12), bit(rw, 11), bit(rw, 10), | 
 | 		       bits(rw, 7, 9), bits(rw, 4, 6), bits(rw, 0, 3)); | 
 | 		break; | 
 | 	case 2: | 
 | 		PDEBUG("reg2 IFPLL2 n1 %d num1 %d", | 
 | 		       bits(rw, 6, 17), bits(rw, 0, 5)); | 
 | 		break; | 
 | 	case 3: | 
 | 		PDEBUG("reg3 IFPLL3 num %d", bits(rw, 0, 17)); | 
 | 		break; | 
 | 	case 4: | 
 | 		PDEBUG("reg4 IFPLL4 dn1 %#04x ct_def1 %d kv_def1 %d", | 
 | 		       bits(rw, 8, 16), bits(rw, 4, 7), bits(rw, 0, 3)); | 
 | 		break; | 
 | 	case 5: | 
 | 		PDEBUG("reg5 RFPLL1 pll_en %d kv_en %d vtc_en %d lpf %d cpl %d" | 
 | 		       " pdp %d autocal_en %d ld_en %d rfloopr %d rfloopc %d" | 
 | 		       " dac %d", | 
 | 		       bit(rw, 17), bit(rw, 16), bit(rw, 15), bit(rw, 14), | 
 | 		       bit(rw, 13), bit(rw, 12), bit(rw, 11), bit(rw, 10), | 
 | 		       bits(rw, 7, 9), bits(rw, 4, 6), bits(rw, 0,3)); | 
 | 		break; | 
 | 	case 6: | 
 | 		PDEBUG("reg6 RFPLL2 n %d num %d", | 
 | 		       bits(rw, 6, 17), bits(rw, 0, 5)); | 
 | 		break; | 
 | 	case 7: | 
 | 		PDEBUG("reg7 RFPLL3 num2 %d", bits(rw, 0, 17)); | 
 | 		break; | 
 | 	case 8: | 
 | 		PDEBUG("reg8 RFPLL4 dn %#06x ct_def %d kv_def %d", | 
 | 		       bits(rw, 8, 16), bits(rw, 4, 7), bits(rw, 0, 3)); | 
 | 		break; | 
 | 	case 9: | 
 | 		PDEBUG("reg9 CAL1 tvco %d tlock %d m_ct_value %d ld_window %d", | 
 | 		       bits(rw, 13, 17), bits(rw, 8, 12), bits(rw, 3, 7), | 
 | 		       bits(rw, 0, 2)); | 
 | 		break; | 
 | 	case 10: | 
 | 		PDEBUG("reg10 TXRX1 rxdcfbbyps %d pcontrol %d txvgc %d" | 
 | 		       " rxlpfbw %d txlpfbw %d txdiffmode %d txenmode %d" | 
 | 		       " intbiasen %d tybypass %d", | 
 | 		       bit(rw, 17), bits(rw, 15, 16), bits(rw, 10, 14), | 
 | 		       bits(rw, 7, 9), bits(rw, 4, 6), bit(rw, 3), bit(rw, 2), | 
 | 		       bit(rw, 1), bit(rw, 0)); | 
 | 		break; | 
 | 	case 11: | 
 | 		PDEBUG("reg11 PCNT1 mid_bias %d p_desired %d pc_offset %d" | 
 | 			" tx_delay %d", | 
 | 			bits(rw, 15, 17), bits(rw, 9, 14), bits(rw, 3, 8), | 
 | 			bits(rw, 0, 2)); | 
 | 		break; | 
 | 	case 12: | 
 | 		PDEBUG("reg12 PCNT2 max_power %d mid_power %d min_power %d", | 
 | 		       bits(rw, 12, 17), bits(rw, 6, 11), bits(rw, 0, 5)); | 
 | 		break; | 
 | 	case 13: | 
 | 		PDEBUG("reg13 VCOT1 rfpll vco comp %d ifpll vco comp %d" | 
 | 		       " lobias %d if_biasbuf %d if_biasvco %d rf_biasbuf %d" | 
 | 		       " rf_biasvco %d", | 
 | 		       bit(rw, 17), bit(rw, 16), bit(rw, 15), | 
 | 		       bits(rw, 8, 9), bits(rw, 5, 7), bits(rw, 3, 4), | 
 | 		       bits(rw, 0, 2)); | 
 | 		break; | 
 | 	case 14: | 
 | 		PDEBUG("reg14 IQCAL rx_acal %d rx_pcal %d" | 
 | 		       " tx_acal %d tx_pcal %d", | 
 | 		       bits(rw, 13, 17), bits(rw, 9, 12), bits(rw, 4, 8), | 
 | 		       bits(rw, 0, 3)); | 
 | 		break; | 
 | 	} | 
 | } | 
 | #endif /* 0 */ | 
 |  | 
 | static int rf2959_init_hw(struct zd_rf *rf) | 
 | { | 
 | 	int r; | 
 | 	struct zd_chip *chip = zd_rf_to_chip(rf); | 
 |  | 
 | 	static const struct zd_ioreq16 ioreqs[] = { | 
 | 		{ CR2,   0x1E }, { CR9,   0x20 }, { CR10,  0x89 }, | 
 | 		{ CR11,  0x00 }, { CR15,  0xD0 }, { CR17,  0x68 }, | 
 | 		{ CR19,  0x4a }, { CR20,  0x0c }, { CR21,  0x0E }, | 
 | 		{ CR23,  0x48 }, | 
 | 		/* normal size for cca threshold */ | 
 | 		{ CR24,  0x14 }, | 
 | 		/* { CR24,  0x20 }, */ | 
 | 		{ CR26,  0x90 }, { CR27,  0x30 }, { CR29,  0x20 }, | 
 | 		{ CR31,  0xb2 }, { CR32,  0x43 }, { CR33,  0x28 }, | 
 | 		{ CR38,  0x30 }, { CR34,  0x0f }, { CR35,  0xF0 }, | 
 | 		{ CR41,  0x2a }, { CR46,  0x7F }, { CR47,  0x1E }, | 
 | 		{ CR51,  0xc5 }, { CR52,  0xc5 }, { CR53,  0xc5 }, | 
 | 		{ CR79,  0x58 }, { CR80,  0x30 }, { CR81,  0x30 }, | 
 | 		{ CR82,  0x00 }, { CR83,  0x24 }, { CR84,  0x04 }, | 
 | 		{ CR85,  0x00 }, { CR86,  0x10 }, { CR87,  0x2A }, | 
 | 		{ CR88,  0x10 }, { CR89,  0x24 }, { CR90,  0x18 }, | 
 | 		/* { CR91,  0x18 }, */ | 
 | 		/* should solve continous CTS frame problems */ | 
 | 		{ CR91,  0x00 }, | 
 | 		{ CR92,  0x0a }, { CR93,  0x00 }, { CR94,  0x01 }, | 
 | 		{ CR95,  0x00 }, { CR96,  0x40 }, { CR97,  0x37 }, | 
 | 		{ CR98,  0x05 }, { CR99,  0x28 }, { CR100, 0x00 }, | 
 | 		{ CR101, 0x13 }, { CR102, 0x27 }, { CR103, 0x27 }, | 
 | 		{ CR104, 0x18 }, { CR105, 0x12 }, | 
 | 		/* normal size */ | 
 | 		{ CR106, 0x1a }, | 
 | 		/* { CR106, 0x22 }, */ | 
 | 		{ CR107, 0x24 }, { CR108, 0x0a }, { CR109, 0x13 }, | 
 | 		{ CR110, 0x2F }, { CR111, 0x27 }, { CR112, 0x27 }, | 
 | 		{ CR113, 0x27 }, { CR114, 0x27 }, { CR115, 0x40 }, | 
 | 		{ CR116, 0x40 }, { CR117, 0xF0 }, { CR118, 0xF0 }, | 
 | 		{ CR119, 0x16 }, | 
 | 		/* no TX continuation */ | 
 | 		{ CR122, 0x00 }, | 
 | 		/* { CR122, 0xff }, */ | 
 | 		{ CR127, 0x03 }, { CR131, 0x08 }, { CR138, 0x28 }, | 
 | 		{ CR148, 0x44 }, { CR150, 0x10 }, { CR169, 0xBB }, | 
 | 		{ CR170, 0xBB }, | 
 | 	}; | 
 |  | 
 | 	static const u32 rv[] = { | 
 | 		0x000007,  /* REG0(CFG1) */ | 
 | 		0x07dd43,  /* REG1(IFPLL1) */ | 
 | 		0x080959,  /* REG2(IFPLL2) */ | 
 | 		0x0e6666, | 
 | 		0x116a57,  /* REG4 */ | 
 | 		0x17dd43,  /* REG5 */ | 
 | 		0x1819f9,  /* REG6 */ | 
 | 		0x1e6666, | 
 | 		0x214554, | 
 | 		0x25e7fa, | 
 | 		0x27fffa, | 
 | 		/* The Zydas driver somehow forgets to set this value. It's | 
 | 		 * only set for Japan. We are using internal power control | 
 | 		 * for now. | 
 | 		 */ | 
 | 		0x294128, /* internal power */ | 
 | 		/* 0x28252c, */ /* External control TX power */ | 
 | 		/* CR31_CCK, CR51_6-36M, CR52_48M, CR53_54M */ | 
 | 		0x2c0000, | 
 | 		0x300000, | 
 | 		0x340000,  /* REG13(0xD) */ | 
 | 		0x381e0f,  /* REG14(0xE) */ | 
 | 		/* Bogus, RF2959's data sheet doesn't know register 27, which is | 
 | 		 * actually referenced here. The commented 0x11 is 17. | 
 | 		 */ | 
 | 		0x6c180f,  /* REG27(0x11) */ | 
 | 	}; | 
 |  | 
 | 	r = zd_iowrite16a_locked(chip, ioreqs, ARRAY_SIZE(ioreqs)); | 
 | 	if (r) | 
 | 		return r; | 
 |  | 
 | 	return zd_rfwritev_locked(chip, rv, ARRAY_SIZE(rv), RF_RV_BITS); | 
 | } | 
 |  | 
 | static int rf2959_set_channel(struct zd_rf *rf, u8 channel) | 
 | { | 
 | 	int i, r; | 
 | 	const u32 *rv = rf2959_table[channel-1]; | 
 | 	struct zd_chip *chip = zd_rf_to_chip(rf); | 
 |  | 
 | 	for (i = 0; i < 2; i++) { | 
 | 		r = zd_rfwrite_locked(chip, rv[i], RF_RV_BITS); | 
 | 		if (r) | 
 | 			return r; | 
 | 	} | 
 | 	return 0; | 
 | } | 
 |  | 
 | static int rf2959_switch_radio_on(struct zd_rf *rf) | 
 | { | 
 | 	static const struct zd_ioreq16 ioreqs[] = { | 
 | 		{ CR10, 0x89 }, | 
 | 		{ CR11, 0x00 }, | 
 | 	}; | 
 | 	struct zd_chip *chip = zd_rf_to_chip(rf); | 
 |  | 
 | 	return zd_iowrite16a_locked(chip, ioreqs, ARRAY_SIZE(ioreqs)); | 
 | } | 
 |  | 
 | static int rf2959_switch_radio_off(struct zd_rf *rf) | 
 | { | 
 | 	static const struct zd_ioreq16 ioreqs[] = { | 
 | 		{ CR10, 0x15 }, | 
 | 		{ CR11, 0x81 }, | 
 | 	}; | 
 | 	struct zd_chip *chip = zd_rf_to_chip(rf); | 
 |  | 
 | 	return zd_iowrite16a_locked(chip, ioreqs, ARRAY_SIZE(ioreqs)); | 
 | } | 
 |  | 
 | int zd_rf_init_rf2959(struct zd_rf *rf) | 
 | { | 
 | 	struct zd_chip *chip = zd_rf_to_chip(rf); | 
 |  | 
 | 	if (zd_chip_is_zd1211b(chip)) { | 
 | 		dev_err(zd_chip_dev(chip), | 
 | 		       "RF2959 is currently not supported for ZD1211B" | 
 | 		       " devices\n"); | 
 | 		return -ENODEV; | 
 | 	} | 
 | 	rf->init_hw = rf2959_init_hw; | 
 | 	rf->set_channel = rf2959_set_channel; | 
 | 	rf->switch_radio_on = rf2959_switch_radio_on; | 
 | 	rf->switch_radio_off = rf2959_switch_radio_off; | 
 | 	return 0; | 
 | } |