tda8290.c 22.9 KB
Newer Older
L
Linus Torvalds 已提交
1
/*
2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20

   i2c tv tuner chip device driver
   controls the philips tda8290+75 tuner chip combo.

   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., 675 Mass Ave, Cambridge, MA 02139, USA.
*/

L
Linus Torvalds 已提交
21 22
#include <linux/i2c.h>
#include <linux/delay.h>
23
#include <linux/videodev.h>
24
#include "tuner-driver.h"
L
Linus Torvalds 已提交
25 26 27

/* ---------------------------------------------------------------------- */

28
struct tda8290_priv {
29 30
	struct tuner_i2c_props i2c_props;

31 32 33 34 35 36 37 38 39
	unsigned char tda8290_easy_mode;
	unsigned char tda827x_lpsel;
	unsigned char tda827x_addr;
	unsigned char tda827x_ver;
	unsigned int sgIF;
};

/* ---------------------------------------------------------------------- */

40 41 42 43 44 45 46 47
struct tda827x_data {
	u32 lomax;
	u8  spd;
	u8  bs;
	u8  bp;
	u8  cp;
	u8  gc3;
	u8 div1p5;
L
Linus Torvalds 已提交
48 49
};

50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81
     /* Note lomax entry is lo / 62500 */

static struct tda827x_data tda827x_analog[] = {
	{ .lomax =   992, .spd = 3, .bs = 2, .bp = 0, .cp = 0, .gc3 = 3, .div1p5 = 1}, /*  62 MHz */
	{ .lomax =  1056, .spd = 3, .bs = 3, .bp = 0, .cp = 0, .gc3 = 3, .div1p5 = 1}, /*  66 MHz */
	{ .lomax =  1216, .spd = 3, .bs = 1, .bp = 0, .cp = 0, .gc3 = 3, .div1p5 = 0}, /*  76 MHz */
	{ .lomax =  1344, .spd = 3, .bs = 2, .bp = 0, .cp = 0, .gc3 = 3, .div1p5 = 0}, /*  84 MHz */
	{ .lomax =  1488, .spd = 3, .bs = 2, .bp = 0, .cp = 0, .gc3 = 1, .div1p5 = 0}, /*  93 MHz */
	{ .lomax =  1568, .spd = 3, .bs = 3, .bp = 0, .cp = 0, .gc3 = 1, .div1p5 = 0}, /*  98 MHz */
	{ .lomax =  1744, .spd = 3, .bs = 3, .bp = 1, .cp = 0, .gc3 = 1, .div1p5 = 0}, /* 109 MHz */
	{ .lomax =  1968, .spd = 2, .bs = 2, .bp = 1, .cp = 0, .gc3 = 1, .div1p5 = 1}, /* 123 MHz */
	{ .lomax =  2128, .spd = 2, .bs = 3, .bp = 1, .cp = 0, .gc3 = 1, .div1p5 = 1}, /* 133 MHz */
	{ .lomax =  2416, .spd = 2, .bs = 1, .bp = 1, .cp = 0, .gc3 = 1, .div1p5 = 0}, /* 151 MHz */
	{ .lomax =  2464, .spd = 2, .bs = 2, .bp = 1, .cp = 0, .gc3 = 1, .div1p5 = 0}, /* 154 MHz */
	{ .lomax =  2896, .spd = 2, .bs = 2, .bp = 1, .cp = 0, .gc3 = 0, .div1p5 = 0}, /* 181 MHz */
	{ .lomax =  2960, .spd = 2, .bs = 2, .bp = 2, .cp = 0, .gc3 = 1, .div1p5 = 0}, /* 185 MHz */
	{ .lomax =  3472, .spd = 2, .bs = 3, .bp = 2, .cp = 0, .gc3 = 1, .div1p5 = 0}, /* 217 MHz */
	{ .lomax =  3904, .spd = 1, .bs = 2, .bp = 2, .cp = 0, .gc3 = 1, .div1p5 = 1}, /* 244 MHz */
	{ .lomax =  4240, .spd = 1, .bs = 3, .bp = 2, .cp = 0, .gc3 = 1, .div1p5 = 1}, /* 265 MHz */
	{ .lomax =  4832, .spd = 1, .bs = 1, .bp = 2, .cp = 0, .gc3 = 1, .div1p5 = 0}, /* 302 MHz */
	{ .lomax =  5184, .spd = 1, .bs = 2, .bp = 2, .cp = 0, .gc3 = 1, .div1p5 = 0}, /* 324 MHz */
	{ .lomax =  5920, .spd = 1, .bs = 2, .bp = 3, .cp = 0, .gc3 = 1, .div1p5 = 0}, /* 370 MHz */
	{ .lomax =  7264, .spd = 1, .bs = 3, .bp = 3, .cp = 0, .gc3 = 1, .div1p5 = 0}, /* 454 MHz */
	{ .lomax =  7888, .spd = 0, .bs = 2, .bp = 3, .cp = 0, .gc3 = 1, .div1p5 = 1}, /* 493 MHz */
	{ .lomax =  8480, .spd = 0, .bs = 3, .bp = 3, .cp = 0, .gc3 = 1, .div1p5 = 1}, /* 530 MHz */
	{ .lomax =  8864, .spd = 0, .bs = 1, .bp = 3, .cp = 0, .gc3 = 1, .div1p5 = 0}, /* 554 MHz */
	{ .lomax =  9664, .spd = 0, .bs = 1, .bp = 4, .cp = 0, .gc3 = 0, .div1p5 = 0}, /* 604 MHz */
	{ .lomax = 11088, .spd = 0, .bs = 2, .bp = 4, .cp = 0, .gc3 = 0, .div1p5 = 0}, /* 696 MHz */
	{ .lomax = 11840, .spd = 0, .bs = 2, .bp = 4, .cp = 1, .gc3 = 0, .div1p5 = 0}, /* 740 MHz */
	{ .lomax = 13120, .spd = 0, .bs = 3, .bp = 4, .cp = 0, .gc3 = 0, .div1p5 = 0}, /* 820 MHz */
	{ .lomax = 13840, .spd = 0, .bs = 3, .bp = 4, .cp = 1, .gc3 = 0, .div1p5 = 0}, /* 865 MHz */
	{ .lomax =     0, .spd = 0, .bs = 0, .bp = 0, .cp = 0, .gc3 = 0, .div1p5 = 0}  /* End      */
L
Linus Torvalds 已提交
82 83
};

84
static void tda827x_tune(struct tuner *t, u16 ifc, unsigned int freq)
85 86 87 88 89
{
	unsigned char tuner_reg[8];
	unsigned char reg2[2];
	u32 N;
	int i;
90 91
	struct tda8290_priv *priv = t->priv;
	struct i2c_msg msg = {.addr = priv->tda827x_addr, .flags = 0};
L
Linus Torvalds 已提交
92

93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109
	if (t->mode == V4L2_TUNER_RADIO)
		freq = freq / 1000;

	N = freq + ifc;
	i = 0;
	while (tda827x_analog[i].lomax < N) {
		if(tda827x_analog[i + 1].lomax == 0)
			break;
		i++;
	}

	N = N << tda827x_analog[i].spd;

	tuner_reg[0] = 0;
	tuner_reg[1] = (unsigned char)(N>>8);
	tuner_reg[2] = (unsigned char) N;
	tuner_reg[3] = 0x40;
110
	tuner_reg[4] = 0x52 + (priv->tda827x_lpsel << 5);
111
	tuner_reg[5] = (tda827x_analog[i].spd   << 6) + (tda827x_analog[i].div1p5 <<5) +
112
		       (tda827x_analog[i].bs     <<3) +  tda827x_analog[i].bp;
113 114 115 116 117
	tuner_reg[6] = 0x8f + (tda827x_analog[i].gc3 << 4);
	tuner_reg[7] = 0x8f;

	msg.buf = tuner_reg;
	msg.len = 8;
118
	i2c_transfer(priv->i2c_props.adap, &msg, 1);
119 120 121 122 123

	msg.buf= reg2;
	msg.len = 2;
	reg2[0] = 0x80;
	reg2[1] = 0;
124
	i2c_transfer(priv->i2c_props.adap, &msg, 1);
125 126 127

	reg2[0] = 0x60;
	reg2[1] = 0xbf;
128
	i2c_transfer(priv->i2c_props.adap, &msg, 1);
129 130 131

	reg2[0] = 0x30;
	reg2[1] = tuner_reg[4] + 0x80;
132
	i2c_transfer(priv->i2c_props.adap, &msg, 1);
133 134 135 136

	msleep(1);
	reg2[0] = 0x30;
	reg2[1] = tuner_reg[4] + 4;
137
	i2c_transfer(priv->i2c_props.adap, &msg, 1);
138 139 140 141

	msleep(1);
	reg2[0] = 0x30;
	reg2[1] = tuner_reg[4];
142
	i2c_transfer(priv->i2c_props.adap, &msg, 1);
143 144 145 146

	msleep(550);
	reg2[0] = 0x30;
	reg2[1] = (tuner_reg[4] & 0xfc) + tda827x_analog[i].cp ;
147
	i2c_transfer(priv->i2c_props.adap, &msg, 1);
L
Linus Torvalds 已提交
148

149
	reg2[0] = 0x60;
150
	reg2[1] = 0x3f;
151
	i2c_transfer(priv->i2c_props.adap, &msg, 1);
152 153 154

	reg2[0] = 0x80;
	reg2[1] = 0x08;   // Vsync en
155
	i2c_transfer(priv->i2c_props.adap, &msg, 1);
156 157
}

158
static void tda827x_agcf(struct tuner *t)
L
Linus Torvalds 已提交
159
{
160
	struct tda8290_priv *priv = t->priv;
161
	unsigned char data[] = {0x80, 0x0c};
162
	struct i2c_msg msg = {.addr = priv->tda827x_addr, .buf = data,
163
			      .flags = 0, .len = 2};
164
	i2c_transfer(priv->i2c_props.adap, &msg, 1);
L
Linus Torvalds 已提交
165 166 167 168
}

/* ---------------------------------------------------------------------- */

169 170 171 172 173 174 175
struct tda827xa_data {
	u32 lomax;
	u8  svco;
	u8  spd;
	u8  scr;
	u8  sbs;
	u8  gc3;
L
Linus Torvalds 已提交
176 177
};

178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204
static struct tda827xa_data tda827xa_analog[] = {
	{ .lomax =   910, .svco = 3, .spd = 4, .scr = 0, .sbs = 0, .gc3 = 3},  /*  56.875 MHz */
	{ .lomax =  1076, .svco = 0, .spd = 3, .scr = 0, .sbs = 0, .gc3 = 3},  /*  67.25 MHz */
	{ .lomax =  1300, .svco = 1, .spd = 3, .scr = 0, .sbs = 0, .gc3 = 3},  /*  81.25 MHz */
	{ .lomax =  1560, .svco = 2, .spd = 3, .scr = 0, .sbs = 0, .gc3 = 3},  /*  97.5  MHz */
	{ .lomax =  1820, .svco = 3, .spd = 3, .scr = 0, .sbs = 1, .gc3 = 1},  /* 113.75 MHz */
	{ .lomax =  2152, .svco = 0, .spd = 2, .scr = 0, .sbs = 1, .gc3 = 1},  /* 134.5 MHz */
	{ .lomax =  2464, .svco = 1, .spd = 2, .scr = 0, .sbs = 1, .gc3 = 1},  /* 154   MHz */
	{ .lomax =  2600, .svco = 1, .spd = 2, .scr = 0, .sbs = 1, .gc3 = 1},  /* 162.5 MHz */
	{ .lomax =  2928, .svco = 2, .spd = 2, .scr = 0, .sbs = 1, .gc3 = 1},  /* 183   MHz */
	{ .lomax =  3120, .svco = 2, .spd = 2, .scr = 0, .sbs = 2, .gc3 = 1},  /* 195   MHz */
	{ .lomax =  3640, .svco = 3, .spd = 2, .scr = 0, .sbs = 2, .gc3 = 3},  /* 227.5 MHz */
	{ .lomax =  4304, .svco = 0, .spd = 1, .scr = 0, .sbs = 2, .gc3 = 3},  /* 269   MHz */
	{ .lomax =  5200, .svco = 1, .spd = 1, .scr = 0, .sbs = 2, .gc3 = 1},  /* 325   MHz */
	{ .lomax =  6240, .svco = 2, .spd = 1, .scr = 0, .sbs = 3, .gc3 = 3},  /* 390   MHz */
	{ .lomax =  7280, .svco = 3, .spd = 1, .scr = 0, .sbs = 3, .gc3 = 3},  /* 455   MHz */
	{ .lomax =  8320, .svco = 0, .spd = 0, .scr = 0, .sbs = 3, .gc3 = 1},  /* 520   MHz */
	{ .lomax =  8608, .svco = 0, .spd = 0, .scr = 1, .sbs = 3, .gc3 = 1},  /* 538   MHz */
	{ .lomax =  8864, .svco = 1, .spd = 0, .scr = 0, .sbs = 3, .gc3 = 1},  /* 554   MHz */
	{ .lomax =  9920, .svco = 1, .spd = 0, .scr = 0, .sbs = 4, .gc3 = 0},  /* 620   MHz */
	{ .lomax = 10400, .svco = 1, .spd = 0, .scr = 1, .sbs = 4, .gc3 = 0},  /* 650   MHz */
	{ .lomax = 11200, .svco = 2, .spd = 0, .scr = 0, .sbs = 4, .gc3 = 0},  /* 700   MHz */
	{ .lomax = 12480, .svco = 2, .spd = 0, .scr = 1, .sbs = 4, .gc3 = 0},  /* 780   MHz */
	{ .lomax = 13120, .svco = 3, .spd = 0, .scr = 0, .sbs = 4, .gc3 = 0},  /* 820   MHz */
	{ .lomax = 13920, .svco = 3, .spd = 0, .scr = 1, .sbs = 4, .gc3 = 0},  /* 870   MHz */
	{ .lomax = 14576, .svco = 3, .spd = 0, .scr = 2, .sbs = 4, .gc3 = 0},  /* 911   MHz */
	{ .lomax =     0, .svco = 0, .spd = 0, .scr = 0, .sbs = 0, .gc3 = 0}   /* End */
L
Linus Torvalds 已提交
205 206
};

207
static void tda827xa_lna_gain(struct tuner *t, int high)
208
{
209
	struct tda8290_priv *priv = t->priv;
210 211
	unsigned char buf[] = {0x22, 0x01};
	int arg;
212
	struct i2c_msg msg = {.addr = priv->i2c_props.addr, .flags = 0, .buf = buf, .len = sizeof(buf)};
213 214 215 216 217 218 219 220 221 222 223 224 225
	if (t->config) {
		if (high)
			tuner_dbg("setting LNA to high gain\n");
		else
			tuner_dbg("setting LNA to low gain\n");
	}
	switch (t->config) {
	case 0: /* no LNA */
		break;
	case 1: /* switch is GPIO 0 of tda8290 */
	case 2:
		/* turn Vsync on */
		if (t->std & V4L2_STD_MN)
226
			arg = 1;
227
		else
228 229
			arg = 0;
		if (t->tuner_callback)
230
			t->tuner_callback(priv->i2c_props.adap->algo_data, 1, arg);
231 232 233
		buf[1] = high ? 0 : 1;
		if (t->config == 2)
			buf[1] = high ? 1 : 0;
234
		i2c_transfer(priv->i2c_props.adap, &msg, 1);
235 236
		break;
	case 3: /* switch with GPIO of saa713x */
237
		if (t->tuner_callback)
238
			t->tuner_callback(priv->i2c_props.adap->algo_data, 0, high);
239 240 241 242
		break;
	}
}

243
static void tda827xa_tune(struct tuner *t, u16 ifc, unsigned int freq)
244
{
245
	unsigned char tuner_reg[11];
246 247
	u32 N;
	int i;
248 249
	struct tda8290_priv *priv = t->priv;
	struct i2c_msg msg = {.addr = priv->tda827x_addr, .flags = 0, .buf = tuner_reg};
250

251
	tda827xa_lna_gain(t, 1);
252
	msleep(10);
L
Linus Torvalds 已提交
253

254 255
	if (t->mode == V4L2_TUNER_RADIO)
		freq = freq / 1000;
L
Linus Torvalds 已提交
256

257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272
	N = freq + ifc;
	i = 0;
	while (tda827xa_analog[i].lomax < N) {
		if(tda827xa_analog[i + 1].lomax == 0)
			break;
		i++;
	}

	N = N << tda827xa_analog[i].spd;

	tuner_reg[0] = 0;
	tuner_reg[1] = (unsigned char)(N>>8);
	tuner_reg[2] = (unsigned char) N;
	tuner_reg[3] = 0;
	tuner_reg[4] = 0x16;
	tuner_reg[5] = (tda827xa_analog[i].spd << 5) + (tda827xa_analog[i].svco << 3) +
273
			tda827xa_analog[i].sbs;
274
	tuner_reg[6] = 0x8b + (tda827xa_analog[i].gc3 << 4);
275
	tuner_reg[7] = 0x1c;
276 277
	tuner_reg[8] = 4;
	tuner_reg[9] = 0x20;
278 279
	tuner_reg[10] = 0x00;
	msg.len = 11;
280
	i2c_transfer(priv->i2c_props.adap, &msg, 1);
281

282 283 284 285
	tuner_reg[0] = 0x90;
	tuner_reg[1] = 0xff;
	tuner_reg[2] = 0xe0;
	tuner_reg[3] = 0;
286
	tuner_reg[4] = 0x99 + (priv->tda827x_lpsel << 1);
287
	msg.len = 5;
288
	i2c_transfer(priv->i2c_props.adap, &msg, 1);
289

290 291
	tuner_reg[0] = 0xa0;
	tuner_reg[1] = 0xc0;
292
	msg.len = 2;
293
	i2c_transfer(priv->i2c_props.adap, &msg, 1);
294

295 296
	tuner_reg[0] = 0x30;
	tuner_reg[1] = 0x10 + tda827xa_analog[i].scr;
297
	i2c_transfer(priv->i2c_props.adap, &msg, 1);
298

299
	msg.flags = I2C_M_RD;
300
	i2c_transfer(priv->i2c_props.adap, &msg, 1);
301 302 303 304
	msg.flags = 0;
	tuner_reg[1] >>= 4;
	tuner_dbg("AGC2 gain is: %d\n", tuner_reg[1]);
	if (tuner_reg[1] < 1)
305
		tda827xa_lna_gain(t, 0);
306 307 308 309

	msleep(100);
	tuner_reg[0] = 0x60;
	tuner_reg[1] = 0x3c;
310
	i2c_transfer(priv->i2c_props.adap, &msg, 1);
311

312 313 314
	msleep(163);
	tuner_reg[0] = 0x50;
	tuner_reg[1] = 0x8f + (tda827xa_analog[i].gc3 << 4);
315
	i2c_transfer(priv->i2c_props.adap, &msg, 1);
316

317 318
	tuner_reg[0] = 0x80;
	tuner_reg[1] = 0x28;
319
	i2c_transfer(priv->i2c_props.adap, &msg, 1);
320

321 322
	tuner_reg[0] = 0xb0;
	tuner_reg[1] = 0x01;
323
	i2c_transfer(priv->i2c_props.adap, &msg, 1);
324

325
	tuner_reg[0] = 0xc0;
326
	tuner_reg[1] = 0x19 + (priv->tda827x_lpsel << 1);
327
	i2c_transfer(priv->i2c_props.adap, &msg, 1);
328 329
}

330
static void tda827xa_agcf(struct tuner *t)
L
Linus Torvalds 已提交
331
{
332
	struct tda8290_priv *priv = t->priv;
333
	unsigned char data[] = {0x80, 0x2c};
334
	struct i2c_msg msg = {.addr = priv->tda827x_addr, .buf = data,
335
			      .flags = 0, .len = 2};
336
	i2c_transfer(priv->i2c_props.adap, &msg, 1);
337
}
L
Linus Torvalds 已提交
338

339
/*---------------------------------------------------------------------*/
L
Linus Torvalds 已提交
340

341
static void tda8290_i2c_bridge(struct tuner *t, int close)
342
{
343 344
	struct tda8290_priv *priv = t->priv;

345
	unsigned char  enable[2] = { 0x21, 0xC0 };
346
	unsigned char disable[2] = { 0x21, 0x00 };
347 348 349
	unsigned char *msg;
	if(close) {
		msg = enable;
350
		tuner_i2c_xfer_send(&priv->i2c_props, msg, 2);
351 352 353 354
		/* let the bridge stabilize */
		msleep(20);
	} else {
		msg = disable;
355
		tuner_i2c_xfer_send(&priv->i2c_props, msg, 2);
356
	}
L
Linus Torvalds 已提交
357 358
}

359 360
/*---------------------------------------------------------------------*/

361
static int tda8290_tune(struct tuner *t, u16 ifc, unsigned int freq)
L
Linus Torvalds 已提交
362
{
363
	struct tda8290_priv *priv = t->priv;
364
	unsigned char soft_reset[]  = { 0x00, 0x00 };
365
	unsigned char easy_mode[]   = { 0x01, priv->tda8290_easy_mode };
366
	unsigned char expert_mode[] = { 0x01, 0x80 };
367
	unsigned char agc_out_on[]  = { 0x02, 0x00 };
368 369 370 371 372 373 374 375 376 377 378 379 380 381 382
	unsigned char gainset_off[] = { 0x28, 0x14 };
	unsigned char if_agc_spd[]  = { 0x0f, 0x88 };
	unsigned char adc_head_6[]  = { 0x05, 0x04 };
	unsigned char adc_head_9[]  = { 0x05, 0x02 };
	unsigned char adc_head_12[] = { 0x05, 0x01 };
	unsigned char pll_bw_nom[]  = { 0x0d, 0x47 };
	unsigned char pll_bw_low[]  = { 0x0d, 0x27 };
	unsigned char gainset_2[]   = { 0x28, 0x64 };
	unsigned char agc_rst_on[]  = { 0x0e, 0x0b };
	unsigned char agc_rst_off[] = { 0x0e, 0x09 };
	unsigned char if_agc_set[]  = { 0x0f, 0x81 };
	unsigned char addr_adc_sat  = 0x1a;
	unsigned char addr_agc_stat = 0x1d;
	unsigned char addr_pll_stat = 0x1b;
	unsigned char adc_sat, agc_stat,
383
		      pll_stat;
384
	int i;
385

386
	tuner_dbg("tda827xa config is 0x%02x\n", t->config);
387 388 389
	tuner_i2c_xfer_send(&priv->i2c_props, easy_mode, 2);
	tuner_i2c_xfer_send(&priv->i2c_props, agc_out_on, 2);
	tuner_i2c_xfer_send(&priv->i2c_props, soft_reset, 2);
390 391
	msleep(1);

392
	expert_mode[1] = priv->tda8290_easy_mode + 0x80;
393 394 395
	tuner_i2c_xfer_send(&priv->i2c_props, expert_mode, 2);
	tuner_i2c_xfer_send(&priv->i2c_props, gainset_off, 2);
	tuner_i2c_xfer_send(&priv->i2c_props, if_agc_spd, 2);
396
	if (priv->tda8290_easy_mode & 0x60)
397
		tuner_i2c_xfer_send(&priv->i2c_props, adc_head_9, 2);
398
	else
399 400
		tuner_i2c_xfer_send(&priv->i2c_props, adc_head_6, 2);
	tuner_i2c_xfer_send(&priv->i2c_props, pll_bw_nom, 2);
401

402
	tda8290_i2c_bridge(t, 1);
403
	if (priv->tda827x_ver != 0)
404
		tda827xa_tune(t, ifc, freq);
405
	else
406
		tda827x_tune(t, ifc, freq);
407
	for (i = 0; i < 3; i++) {
408 409
		tuner_i2c_xfer_send(&priv->i2c_props, &addr_pll_stat, 1);
		tuner_i2c_xfer_recv(&priv->i2c_props, &pll_stat, 1);
410
		if (pll_stat & 0x80) {
411 412 413 414
			tuner_i2c_xfer_send(&priv->i2c_props, &addr_adc_sat, 1);
			tuner_i2c_xfer_recv(&priv->i2c_props, &adc_sat, 1);
			tuner_i2c_xfer_send(&priv->i2c_props, &addr_agc_stat, 1);
			tuner_i2c_xfer_recv(&priv->i2c_props, &agc_stat, 1);
415 416 417 418 419 420 421
			tuner_dbg("tda8290 is locked, AGC: %d\n", agc_stat);
			break;
		} else {
			tuner_dbg("tda8290 not locked, no signal?\n");
			msleep(100);
		}
	}
422
	/* adjust headroom resp. gain */
423 424 425
	if ((agc_stat > 115) || (!(pll_stat & 0x80) && (adc_sat < 20))) {
		tuner_dbg("adjust gain, step 1. Agc: %d, ADC stat: %d, lock: %d\n",
			   agc_stat, adc_sat, pll_stat & 0x80);
426
		tuner_i2c_xfer_send(&priv->i2c_props, gainset_2, 2);
427
		msleep(100);
428 429 430 431
		tuner_i2c_xfer_send(&priv->i2c_props, &addr_agc_stat, 1);
		tuner_i2c_xfer_recv(&priv->i2c_props, &agc_stat, 1);
		tuner_i2c_xfer_send(&priv->i2c_props, &addr_pll_stat, 1);
		tuner_i2c_xfer_recv(&priv->i2c_props, &pll_stat, 1);
432
		if ((agc_stat > 115) || !(pll_stat & 0x80)) {
433 434
			tuner_dbg("adjust gain, step 2. Agc: %d, lock: %d\n",
				   agc_stat, pll_stat & 0x80);
435
			if (priv->tda827x_ver != 0)
436
				tda827xa_agcf(t);
437
			else
438
				tda827x_agcf(t);
439
			msleep(100);
440 441 442 443
			tuner_i2c_xfer_send(&priv->i2c_props, &addr_agc_stat, 1);
			tuner_i2c_xfer_recv(&priv->i2c_props, &agc_stat, 1);
			tuner_i2c_xfer_send(&priv->i2c_props, &addr_pll_stat, 1);
			tuner_i2c_xfer_recv(&priv->i2c_props, &pll_stat, 1);
444 445
			if((agc_stat > 115) || !(pll_stat & 0x80)) {
				tuner_dbg("adjust gain, step 3. Agc: %d\n", agc_stat);
446 447
				tuner_i2c_xfer_send(&priv->i2c_props, adc_head_12, 2);
				tuner_i2c_xfer_send(&priv->i2c_props, pll_bw_low, 2);
448 449 450 451
				msleep(100);
			}
		}
	}
L
Linus Torvalds 已提交
452

453
	/* l/ l' deadlock? */
454
	if(priv->tda8290_easy_mode & 0x60) {
455 456 457 458
		tuner_i2c_xfer_send(&priv->i2c_props, &addr_adc_sat, 1);
		tuner_i2c_xfer_recv(&priv->i2c_props, &adc_sat, 1);
		tuner_i2c_xfer_send(&priv->i2c_props, &addr_pll_stat, 1);
		tuner_i2c_xfer_recv(&priv->i2c_props, &pll_stat, 1);
459
		if ((adc_sat > 20) || !(pll_stat & 0x80)) {
460
			tuner_dbg("trying to resolve SECAM L deadlock\n");
461
			tuner_i2c_xfer_send(&priv->i2c_props, agc_rst_on, 2);
462
			msleep(40);
463
			tuner_i2c_xfer_send(&priv->i2c_props, agc_rst_off, 2);
464 465
		}
	}
466

467 468
	tda8290_i2c_bridge(t, 0);
	tuner_i2c_xfer_send(&priv->i2c_props, if_agc_set, 2);
469
	return 0;
L
Linus Torvalds 已提交
470 471
}

472 473
/*---------------------------------------------------------------------*/

L
Linus Torvalds 已提交
474 475
static void set_audio(struct tuner *t)
{
476
	struct tda8290_priv *priv = t->priv;
477
	char* mode;
L
Linus Torvalds 已提交
478

479
	priv->tda827x_lpsel = 0;
480
	if (t->std & V4L2_STD_MN) {
481 482 483
		priv->sgIF = 92;
		priv->tda8290_easy_mode = 0x01;
		priv->tda827x_lpsel = 1;
484
		mode = "MN";
485
	} else if (t->std & V4L2_STD_B) {
486 487
		priv->sgIF = 108;
		priv->tda8290_easy_mode = 0x02;
488
		mode = "B";
489
	} else if (t->std & V4L2_STD_GH) {
490 491
		priv->sgIF = 124;
		priv->tda8290_easy_mode = 0x04;
492
		mode = "GH";
493
	} else if (t->std & V4L2_STD_PAL_I) {
494 495
		priv->sgIF = 124;
		priv->tda8290_easy_mode = 0x08;
496
		mode = "I";
497
	} else if (t->std & V4L2_STD_DK) {
498 499
		priv->sgIF = 124;
		priv->tda8290_easy_mode = 0x10;
500
		mode = "DK";
501
	} else if (t->std & V4L2_STD_SECAM_L) {
502 503
		priv->sgIF = 124;
		priv->tda8290_easy_mode = 0x20;
504
		mode = "L";
505
	} else if (t->std & V4L2_STD_SECAM_LC) {
506 507
		priv->sgIF = 20;
		priv->tda8290_easy_mode = 0x40;
508
		mode = "LC";
509
	} else {
510 511
		priv->sgIF = 124;
		priv->tda8290_easy_mode = 0x10;
512
		mode = "xx";
513
	}
514
	tuner_dbg("setting tda8290 to system %s\n", mode);
L
Linus Torvalds 已提交
515 516
}

517
static void set_tv_freq(struct tuner *t, unsigned int freq)
L
Linus Torvalds 已提交
518
{
519
	struct tda8290_priv *priv = t->priv;
L
Linus Torvalds 已提交
520 521

	set_audio(t);
522
	tda8290_tune(t, priv->sgIF, freq);
L
Linus Torvalds 已提交
523 524
}

525
static void set_radio_freq(struct tuner *t, unsigned int freq)
L
Linus Torvalds 已提交
526
{
527
	/* if frequency is 5.5 MHz */
528
	tda8290_tune(t, 88, freq);
L
Linus Torvalds 已提交
529 530
}

531
static int has_signal(struct tuner *t)
L
Linus Torvalds 已提交
532
{
533 534
	struct tda8290_priv *priv = t->priv;

L
Linus Torvalds 已提交
535 536 537
	unsigned char i2c_get_afc[1] = { 0x1B };
	unsigned char afc = 0;

538 539
	tuner_i2c_xfer_send(&priv->i2c_props, i2c_get_afc, ARRAY_SIZE(i2c_get_afc));
	tuner_i2c_xfer_recv(&priv->i2c_props, &afc, 1);
L
Linus Torvalds 已提交
540 541 542
	return (afc & 0x80)? 65535:0;
}

543 544
/*---------------------------------------------------------------------*/

545
static void standby(struct tuner *t)
546
{
547
	struct tda8290_priv *priv = t->priv;
548 549
	unsigned char cb1[] = { 0x30, 0xD0 };
	unsigned char tda8290_standby[] = { 0x00, 0x02 };
550
	unsigned char tda8290_agc_tri[] = { 0x02, 0x20 };
551
	struct i2c_msg msg = {.addr = priv->tda827x_addr, .flags=0, .buf=cb1, .len = 2};
552

553
	tda8290_i2c_bridge(t, 1);
554
	if (priv->tda827x_ver != 0)
555
		cb1[1] = 0x90;
556 557 558 559
	i2c_transfer(priv->i2c_props.adap, &msg, 1);
	tda8290_i2c_bridge(t, 0);
	tuner_i2c_xfer_send(&priv->i2c_props, tda8290_agc_tri, 2);
	tuner_i2c_xfer_send(&priv->i2c_props, tda8290_standby, 2);
560 561
}

562

563
static void tda8290_init_if(struct tuner *t)
564
{
565 566
	struct tda8290_priv *priv = t->priv;

567
	unsigned char set_VS[] = { 0x30, 0x6F };
568
	unsigned char set_GP00_CF[] = { 0x20, 0x01 };
569 570
	unsigned char set_GP01_CF[] = { 0x20, 0x0B };

571
	if ((t->config == 1) || (t->config == 2))
572
		tuner_i2c_xfer_send(&priv->i2c_props, set_GP00_CF, 2);
573
	else
574 575
		tuner_i2c_xfer_send(&priv->i2c_props, set_GP01_CF, 2);
	tuner_i2c_xfer_send(&priv->i2c_props, set_VS, 2);
576 577
}

578
static void tda8290_init_tuner(struct tuner *t)
L
Linus Torvalds 已提交
579
{
580
	struct tda8290_priv *priv = t->priv;
581
	unsigned char tda8275_init[]  = { 0x00, 0x00, 0x00, 0x40, 0xdC, 0x04, 0xAf,
582
					  0x3F, 0x2A, 0x04, 0xFF, 0x00, 0x00, 0x40 };
583
	unsigned char tda8275a_init[] = { 0x00, 0x00, 0x00, 0x00, 0xdC, 0x05, 0x8b,
584
					  0x0c, 0x04, 0x20, 0xFF, 0x00, 0x00, 0x4b };
585
	struct i2c_msg msg = {.addr = priv->tda827x_addr, .flags=0,
586
			      .buf=tda8275_init, .len = 14};
587
	if (priv->tda827x_ver != 0)
588 589
		msg.buf = tda8275a_init;

590 591 592
	tda8290_i2c_bridge(t, 1);
	i2c_transfer(priv->i2c_props.adap, &msg, 1);
	tda8290_i2c_bridge(t, 0);
593 594 595
}

/*---------------------------------------------------------------------*/
L
Linus Torvalds 已提交
596

597
static void tda8290_release(struct tuner *t)
598 599 600 601 602
{
	kfree(t->priv);
	t->priv = NULL;
}

603 604 605 606 607 608 609 610
static struct tuner_operations tda8290_tuner_ops = {
	.set_tv_freq    = set_tv_freq,
	.set_radio_freq = set_radio_freq,
	.has_signal     = has_signal,
	.standby        = standby,
	.release        = tda8290_release,
};

611
int tda8290_init(struct tuner *t)
612
{
613
	struct tda8290_priv *priv = NULL;
614 615 616 617 618
	u8 data;
	int i, ret, tuners_found;
	u32 tuner_addrs;
	struct i2c_msg msg = {.flags=I2C_M_RD, .buf=&data, .len = 1};

619 620 621 622 623
	priv = kzalloc(sizeof(struct tda8290_priv), GFP_KERNEL);
	if (priv == NULL)
		return -ENOMEM;
	t->priv = priv;

624 625 626 627
	priv->i2c_props.addr = t->i2c.addr;
	priv->i2c_props.adap = t->i2c.adapter;

	tda8290_i2c_bridge(t, 1);
628 629 630 631 632
	/* probe for tuner chip */
	tuners_found = 0;
	tuner_addrs = 0;
	for (i=0x60; i<= 0x63; i++) {
		msg.addr = i;
633
		ret = i2c_transfer(priv->i2c_props.adap, &msg, 1);
634 635 636 637 638 639 640 641 642
		if (ret == 1) {
			tuners_found++;
			tuner_addrs = (tuner_addrs << 8) + i;
		}
	}
	/* if there is more than one tuner, we expect the right one is
	   behind the bridge and we choose the highest address that doesn't
	   give a response now
	 */
643
	tda8290_i2c_bridge(t, 0);
644 645 646
	if(tuners_found > 1)
		for (i = 0; i < tuners_found; i++) {
			msg.addr = tuner_addrs  & 0xff;
647
			ret = i2c_transfer(priv->i2c_props.adap, &msg, 1);
648 649 650 651 652 653 654
			if(ret == 1)
				tuner_addrs = tuner_addrs >> 8;
			else
				break;
		}
	if (tuner_addrs == 0) {
		tuner_addrs = 0x61;
655
		tuner_info ("could not clearly identify tuner address, defaulting to %x\n",
656
			     tuner_addrs);
657 658 659 660
	} else {
		tuner_addrs = tuner_addrs & 0xff;
		tuner_info ("setting tuner address to %x\n", tuner_addrs);
	}
661
	priv->tda827x_addr = tuner_addrs;
662 663
	msg.addr = tuner_addrs;

664 665
	tda8290_i2c_bridge(t, 1);
	ret = i2c_transfer(priv->i2c_props.adap, &msg, 1);
666 667 668
	if( ret != 1)
		tuner_warn ("TDA827x access failed!\n");
	if ((data & 0x3c) == 0) {
669
		strlcpy(t->i2c.name, "tda8290+75", sizeof(t->i2c.name));
670
		priv->tda827x_ver = 0;
671
	} else {
672
		strlcpy(t->i2c.name, "tda8290+75a", sizeof(t->i2c.name));
673
		priv->tda827x_ver = 2;
674
	}
675
	tuner_info("type set to %s\n", t->i2c.name);
676

677 678
	memcpy(&t->ops, &tda8290_tuner_ops, sizeof(struct tuner_operations));

679
	priv->tda827x_lpsel = 0;
680
	t->mode = V4L2_TUNER_ANALOG_TV;
L
Linus Torvalds 已提交
681

682 683
	tda8290_init_tuner(t);
	tda8290_init_if(t);
L
Linus Torvalds 已提交
684 685 686
	return 0;
}

687
int tda8290_probe(struct tuner *t)
688
{
689 690
	struct i2c_client *c = &t->i2c;

691 692 693 694
	unsigned char soft_reset[]   = { 0x00, 0x00 };
	unsigned char easy_mode_b[]  = { 0x01, 0x02 };
	unsigned char easy_mode_g[]  = { 0x01, 0x04 };
	unsigned char restore_9886[] = { 0x00, 0xd6, 0x30 };
695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710
	unsigned char addr_dto_lsb = 0x07;
	unsigned char data;

	i2c_master_send(c, easy_mode_b, 2);
	i2c_master_send(c, soft_reset, 2);
	i2c_master_send(c, &addr_dto_lsb, 1);
	i2c_master_recv(c, &data, 1);
	if (data == 0) {
		i2c_master_send(c, easy_mode_g, 2);
		i2c_master_send(c, soft_reset, 2);
		i2c_master_send(c, &addr_dto_lsb, 1);
		i2c_master_recv(c, &data, 1);
		if (data == 0x7b) {
			return 0;
		}
	}
711
	i2c_master_send(c, restore_9886, 3);
712 713 714
	return -1;
}

L
Linus Torvalds 已提交
715 716 717 718 719 720 721
/*
 * Overrides for Emacs so that we follow Linus's tabbing style.
 * ---------------------------------------------------------------------------
 * Local variables:
 * c-basic-offset: 8
 * End:
 */