blob: 544cdbe88a6c9220eef89326ee363739cb0f2434 [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001#include <linux/module.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -07002#include <linux/kernel.h>
3#include <linux/i2c.h>
4#include <linux/types.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -07005#include <linux/init.h>
6#include <linux/errno.h>
7#include <linux/slab.h>
8#include <linux/delay.h>
Hans Verkuilf894dfd2008-07-25 07:39:54 -03009#include <linux/videodev2.h>
Michael Krufky5e453dc2006-01-09 15:32:31 -020010#include <media/v4l2-common.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070011#include <media/tuner.h>
Michael Krufkyab166052007-12-09 02:26:48 -030012#include "tuner-i2c.h"
Michael Krufky31c95842007-10-21 20:48:48 -030013#include "tda9887.h"
Linus Torvalds1da177e2005-04-16 15:20:36 -070014
Mauro Carvalho Chehab674434c2005-12-12 00:37:28 -080015
Linus Torvalds1da177e2005-04-16 15:20:36 -070016/* Chips:
17 TDA9885 (PAL, NTSC)
18 TDA9886 (PAL, SECAM, NTSC)
19 TDA9887 (PAL, SECAM, NTSC, FM Radio)
20
Mauro Carvalho Chehab15396232006-06-23 16:13:56 -030021 Used as part of several tuners
Linus Torvalds1da177e2005-04-16 15:20:36 -070022*/
23
Michael Krufky790ba182007-12-16 21:20:21 -030024static int debug;
25module_param(debug, int, 0644);
26MODULE_PARM_DESC(debug, "enable verbose debug messages");
Michael Krufky31c95842007-10-21 20:48:48 -030027
Michael Krufkyac8b63b2008-04-22 14:45:51 -030028static DEFINE_MUTEX(tda9887_list_mutex);
29static LIST_HEAD(hybrid_tuner_instance_list);
30
Michael Krufkyb2083192007-05-29 22:54:06 -030031struct tda9887_priv {
Michael Krufkydb8a6952007-08-21 01:24:42 -030032 struct tuner_i2c_props i2c_props;
Michael Krufkyac8b63b2008-04-22 14:45:51 -030033 struct list_head hybrid_tuner_instance_list;
Michael Krufkydb8a6952007-08-21 01:24:42 -030034
Michael Krufkyb2083192007-05-29 22:54:06 -030035 unsigned char data[4];
Michael Krufky710401b2007-12-16 19:53:32 -030036 unsigned int config;
Michael Krufky91c9d4a2007-12-16 20:05:00 -030037 unsigned int mode;
38 unsigned int audmode;
39 v4l2_std_id std;
Michael Krufkyb2083192007-05-29 22:54:06 -030040};
Linus Torvalds1da177e2005-04-16 15:20:36 -070041
42/* ---------------------------------------------------------------------- */
43
44#define UNSET (-1U)
Linus Torvalds1da177e2005-04-16 15:20:36 -070045
46struct tvnorm {
47 v4l2_std_id std;
48 char *name;
49 unsigned char b;
50 unsigned char c;
51 unsigned char e;
52};
53
Linus Torvalds1da177e2005-04-16 15:20:36 -070054/* ---------------------------------------------------------------------- */
55
56//
57// TDA defines
58//
59
60//// first reg (b)
61#define cVideoTrapBypassOFF 0x00 // bit b0
62#define cVideoTrapBypassON 0x01 // bit b0
63
64#define cAutoMuteFmInactive 0x00 // bit b1
65#define cAutoMuteFmActive 0x02 // bit b1
66
67#define cIntercarrier 0x00 // bit b2
68#define cQSS 0x04 // bit b2
69
70#define cPositiveAmTV 0x00 // bit b3:4
71#define cFmRadio 0x08 // bit b3:4
72#define cNegativeFmTV 0x10 // bit b3:4
73
74
75#define cForcedMuteAudioON 0x20 // bit b5
76#define cForcedMuteAudioOFF 0x00 // bit b5
77
78#define cOutputPort1Active 0x00 // bit b6
79#define cOutputPort1Inactive 0x40 // bit b6
80
81#define cOutputPort2Active 0x00 // bit b7
82#define cOutputPort2Inactive 0x80 // bit b7
83
84
85//// second reg (c)
86#define cDeemphasisOFF 0x00 // bit c5
87#define cDeemphasisON 0x20 // bit c5
88
89#define cDeemphasis75 0x00 // bit c6
90#define cDeemphasis50 0x40 // bit c6
91
92#define cAudioGain0 0x00 // bit c7
93#define cAudioGain6 0x80 // bit c7
94
Hans Verkuilf98c55e2006-01-09 15:25:18 -020095#define cTopMask 0x1f // bit c0:4
Hans Verkuilf5b01422006-06-25 15:37:29 -030096#define cTopDefault 0x10 // bit c0:4
Linus Torvalds1da177e2005-04-16 15:20:36 -070097
98//// third reg (e)
99#define cAudioIF_4_5 0x00 // bit e0:1
100#define cAudioIF_5_5 0x01 // bit e0:1
101#define cAudioIF_6_0 0x02 // bit e0:1
102#define cAudioIF_6_5 0x03 // bit e0:1
103
104
Trent Piepho5e082f12007-08-03 18:32:38 -0300105#define cVideoIFMask 0x1c // bit e2:4
106/* Video IF selection in TV Mode (bit B3=0) */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700107#define cVideoIF_58_75 0x00 // bit e2:4
108#define cVideoIF_45_75 0x04 // bit e2:4
109#define cVideoIF_38_90 0x08 // bit e2:4
110#define cVideoIF_38_00 0x0C // bit e2:4
111#define cVideoIF_33_90 0x10 // bit e2:4
112#define cVideoIF_33_40 0x14 // bit e2:4
113#define cRadioIF_45_75 0x18 // bit e2:4
114#define cRadioIF_38_90 0x1C // bit e2:4
115
Trent Piepho5e082f12007-08-03 18:32:38 -0300116/* IF1 selection in Radio Mode (bit B3=1) */
117#define cRadioIF_33_30 0x00 // bit e2,4 (also 0x10,0x14)
118#define cRadioIF_41_30 0x04 // bit e2,4
119
120/* Output of AFC pin in radio mode when bit E7=1 */
121#define cRadioAGC_SIF 0x00 // bit e3
122#define cRadioAGC_FM 0x08 // bit e3
Linus Torvalds1da177e2005-04-16 15:20:36 -0700123
124#define cTunerGainNormal 0x00 // bit e5
125#define cTunerGainLow 0x20 // bit e5
126
127#define cGating_18 0x00 // bit e6
128#define cGating_36 0x40 // bit e6
129
130#define cAgcOutON 0x80 // bit e7
131#define cAgcOutOFF 0x00 // bit e7
132
133/* ---------------------------------------------------------------------- */
134
135static struct tvnorm tvnorms[] = {
136 {
Hans Verkuilf98c55e2006-01-09 15:25:18 -0200137 .std = V4L2_STD_PAL_BG | V4L2_STD_PAL_H | V4L2_STD_PAL_N,
138 .name = "PAL-BGHN",
Linus Torvalds1da177e2005-04-16 15:20:36 -0700139 .b = ( cNegativeFmTV |
140 cQSS ),
141 .c = ( cDeemphasisON |
Hans Verkuilf98c55e2006-01-09 15:25:18 -0200142 cDeemphasis50 |
Hans Verkuilf5b01422006-06-25 15:37:29 -0300143 cTopDefault),
Hans Verkuilf98c55e2006-01-09 15:25:18 -0200144 .e = ( cGating_36 |
145 cAudioIF_5_5 |
Linus Torvalds1da177e2005-04-16 15:20:36 -0700146 cVideoIF_38_90 ),
147 },{
148 .std = V4L2_STD_PAL_I,
149 .name = "PAL-I",
150 .b = ( cNegativeFmTV |
151 cQSS ),
152 .c = ( cDeemphasisON |
Hans Verkuilf98c55e2006-01-09 15:25:18 -0200153 cDeemphasis50 |
Hans Verkuilf5b01422006-06-25 15:37:29 -0300154 cTopDefault),
Hans Verkuilf98c55e2006-01-09 15:25:18 -0200155 .e = ( cGating_36 |
156 cAudioIF_6_0 |
Linus Torvalds1da177e2005-04-16 15:20:36 -0700157 cVideoIF_38_90 ),
158 },{
159 .std = V4L2_STD_PAL_DK,
160 .name = "PAL-DK",
161 .b = ( cNegativeFmTV |
162 cQSS ),
163 .c = ( cDeemphasisON |
Hans Verkuilf98c55e2006-01-09 15:25:18 -0200164 cDeemphasis50 |
Hans Verkuilf5b01422006-06-25 15:37:29 -0300165 cTopDefault),
Hans Verkuilf98c55e2006-01-09 15:25:18 -0200166 .e = ( cGating_36 |
167 cAudioIF_6_5 |
168 cVideoIF_38_90 ),
Linus Torvalds1da177e2005-04-16 15:20:36 -0700169 },{
Hans Verkuilf98c55e2006-01-09 15:25:18 -0200170 .std = V4L2_STD_PAL_M | V4L2_STD_PAL_Nc,
171 .name = "PAL-M/Nc",
Linus Torvalds1da177e2005-04-16 15:20:36 -0700172 .b = ( cNegativeFmTV |
173 cQSS ),
174 .c = ( cDeemphasisON |
Hans Verkuilf98c55e2006-01-09 15:25:18 -0200175 cDeemphasis75 |
Hans Verkuilf5b01422006-06-25 15:37:29 -0300176 cTopDefault),
Hans Verkuilf98c55e2006-01-09 15:25:18 -0200177 .e = ( cGating_36 |
178 cAudioIF_4_5 |
Linus Torvalds1da177e2005-04-16 15:20:36 -0700179 cVideoIF_45_75 ),
180 },{
Hans Verkuilf98c55e2006-01-09 15:25:18 -0200181 .std = V4L2_STD_SECAM_B | V4L2_STD_SECAM_G | V4L2_STD_SECAM_H,
182 .name = "SECAM-BGH",
Frederic CANDb84ca9f2008-10-30 04:53:07 -0300183 .b = ( cNegativeFmTV |
Hans Verkuilf98c55e2006-01-09 15:25:18 -0200184 cQSS ),
Hans Verkuilf5b01422006-06-25 15:37:29 -0300185 .c = ( cTopDefault),
Frederic CANDb84ca9f2008-10-30 04:53:07 -0300186 .e = ( cAudioIF_5_5 |
Hans Verkuilf98c55e2006-01-09 15:25:18 -0200187 cVideoIF_38_90 ),
188 },{
Linus Torvalds1da177e2005-04-16 15:20:36 -0700189 .std = V4L2_STD_SECAM_L,
190 .name = "SECAM-L",
191 .b = ( cPositiveAmTV |
192 cQSS ),
Hans Verkuilf5b01422006-06-25 15:37:29 -0300193 .c = ( cTopDefault),
Nickolay V. Shmyrev3375c392005-11-08 21:37:05 -0800194 .e = ( cGating_36 |
Nickolay V. Shmyrev5f7591c2005-11-08 21:37:04 -0800195 cAudioIF_6_5 |
Linus Torvalds1da177e2005-04-16 15:20:36 -0700196 cVideoIF_38_90 ),
197 },{
Mauro Carvalho Chehabf3c59872006-01-09 15:25:00 -0200198 .std = V4L2_STD_SECAM_LC,
199 .name = "SECAM-L'",
200 .b = ( cOutputPort2Inactive |
201 cPositiveAmTV |
202 cQSS ),
Hans Verkuilf5b01422006-06-25 15:37:29 -0300203 .c = ( cTopDefault),
Mauro Carvalho Chehabf3c59872006-01-09 15:25:00 -0200204 .e = ( cGating_36 |
205 cAudioIF_6_5 |
206 cVideoIF_33_90 ),
207 },{
Linus Torvalds1da177e2005-04-16 15:20:36 -0700208 .std = V4L2_STD_SECAM_DK,
209 .name = "SECAM-DK",
210 .b = ( cNegativeFmTV |
211 cQSS ),
212 .c = ( cDeemphasisON |
Hans Verkuilf98c55e2006-01-09 15:25:18 -0200213 cDeemphasis50 |
Hans Verkuilf5b01422006-06-25 15:37:29 -0300214 cTopDefault),
Hans Verkuilf98c55e2006-01-09 15:25:18 -0200215 .e = ( cGating_36 |
216 cAudioIF_6_5 |
217 cVideoIF_38_90 ),
Linus Torvalds1da177e2005-04-16 15:20:36 -0700218 },{
Hans Verkuil0dfd8122006-02-07 06:45:34 -0200219 .std = V4L2_STD_NTSC_M | V4L2_STD_NTSC_M_KR,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700220 .name = "NTSC-M",
221 .b = ( cNegativeFmTV |
222 cQSS ),
223 .c = ( cDeemphasisON |
Hans Verkuilf98c55e2006-01-09 15:25:18 -0200224 cDeemphasis75 |
Hans Verkuilf5b01422006-06-25 15:37:29 -0300225 cTopDefault),
Linus Torvalds1da177e2005-04-16 15:20:36 -0700226 .e = ( cGating_36 |
227 cAudioIF_4_5 |
228 cVideoIF_45_75 ),
229 },{
230 .std = V4L2_STD_NTSC_M_JP,
Hans Verkuilf98c55e2006-01-09 15:25:18 -0200231 .name = "NTSC-M-JP",
Linus Torvalds1da177e2005-04-16 15:20:36 -0700232 .b = ( cNegativeFmTV |
233 cQSS ),
234 .c = ( cDeemphasisON |
Hans Verkuilf98c55e2006-01-09 15:25:18 -0200235 cDeemphasis50 |
Hans Verkuilf5b01422006-06-25 15:37:29 -0300236 cTopDefault),
Linus Torvalds1da177e2005-04-16 15:20:36 -0700237 .e = ( cGating_36 |
238 cAudioIF_4_5 |
239 cVideoIF_58_75 ),
240 }
241};
242
Mauro Carvalho Chehab56fc08c2005-06-23 22:05:07 -0700243static struct tvnorm radio_stereo = {
244 .name = "Radio Stereo",
245 .b = ( cFmRadio |
246 cQSS ),
247 .c = ( cDeemphasisOFF |
Hans Verkuilf98c55e2006-01-09 15:25:18 -0200248 cAudioGain6 |
Hans Verkuilf5b01422006-06-25 15:37:29 -0300249 cTopDefault),
Hans Verkuilf98c55e2006-01-09 15:25:18 -0200250 .e = ( cTunerGainLow |
251 cAudioIF_5_5 |
Mauro Carvalho Chehab56fc08c2005-06-23 22:05:07 -0700252 cRadioIF_38_90 ),
253};
254
255static struct tvnorm radio_mono = {
256 .name = "Radio Mono",
Linus Torvalds1da177e2005-04-16 15:20:36 -0700257 .b = ( cFmRadio |
258 cQSS ),
259 .c = ( cDeemphasisON |
Hans Verkuilf98c55e2006-01-09 15:25:18 -0200260 cDeemphasis75 |
Hans Verkuilf5b01422006-06-25 15:37:29 -0300261 cTopDefault),
Hans Verkuilf98c55e2006-01-09 15:25:18 -0200262 .e = ( cTunerGainLow |
263 cAudioIF_5_5 |
Linus Torvalds1da177e2005-04-16 15:20:36 -0700264 cRadioIF_38_90 ),
265};
266
267/* ---------------------------------------------------------------------- */
268
Michael Krufky4e9154b2007-10-21 19:39:50 -0300269static void dump_read_message(struct dvb_frontend *fe, unsigned char *buf)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700270{
Michael Krufky4e9154b2007-10-21 19:39:50 -0300271 struct tda9887_priv *priv = fe->analog_demod_priv;
272
Linus Torvalds1da177e2005-04-16 15:20:36 -0700273 static char *afc[16] = {
274 "- 12.5 kHz",
275 "- 37.5 kHz",
276 "- 62.5 kHz",
277 "- 87.5 kHz",
278 "-112.5 kHz",
279 "-137.5 kHz",
280 "-162.5 kHz",
281 "-187.5 kHz [min]",
282 "+187.5 kHz [max]",
283 "+162.5 kHz",
284 "+137.5 kHz",
285 "+112.5 kHz",
286 "+ 87.5 kHz",
287 "+ 62.5 kHz",
288 "+ 37.5 kHz",
289 "+ 12.5 kHz",
290 };
Michael Krufky790ba182007-12-16 21:20:21 -0300291 tuner_info("read: 0x%2x\n", buf[0]);
292 tuner_info(" after power on : %s\n", (buf[0] & 0x01) ? "yes" : "no");
293 tuner_info(" afc : %s\n", afc[(buf[0] >> 1) & 0x0f]);
294 tuner_info(" fmif level : %s\n", (buf[0] & 0x20) ? "high" : "low");
295 tuner_info(" afc window : %s\n", (buf[0] & 0x40) ? "in" : "out");
296 tuner_info(" vfi level : %s\n", (buf[0] & 0x80) ? "high" : "low");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700297}
298
Michael Krufky4e9154b2007-10-21 19:39:50 -0300299static void dump_write_message(struct dvb_frontend *fe, unsigned char *buf)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700300{
Michael Krufky4e9154b2007-10-21 19:39:50 -0300301 struct tda9887_priv *priv = fe->analog_demod_priv;
302
Linus Torvalds1da177e2005-04-16 15:20:36 -0700303 static char *sound[4] = {
304 "AM/TV",
305 "FM/radio",
306 "FM/TV",
307 "FM/radio"
308 };
309 static char *adjust[32] = {
310 "-16", "-15", "-14", "-13", "-12", "-11", "-10", "-9",
311 "-8", "-7", "-6", "-5", "-4", "-3", "-2", "-1",
312 "0", "+1", "+2", "+3", "+4", "+5", "+6", "+7",
313 "+8", "+9", "+10", "+11", "+12", "+13", "+14", "+15"
314 };
315 static char *deemph[4] = {
316 "no", "no", "75", "50"
317 };
318 static char *carrier[4] = {
319 "4.5 MHz",
320 "5.5 MHz",
321 "6.0 MHz",
322 "6.5 MHz / AM"
323 };
324 static char *vif[8] = {
325 "58.75 MHz",
326 "45.75 MHz",
327 "38.9 MHz",
328 "38.0 MHz",
329 "33.9 MHz",
330 "33.4 MHz",
331 "45.75 MHz + pin13",
332 "38.9 MHz + pin13",
333 };
334 static char *rif[4] = {
335 "44 MHz",
336 "52 MHz",
337 "52 MHz",
338 "44 MHz",
339 };
340
Michael Krufky790ba182007-12-16 21:20:21 -0300341 tuner_info("write: byte B 0x%02x\n", buf[1]);
342 tuner_info(" B0 video mode : %s\n",
343 (buf[1] & 0x01) ? "video trap" : "sound trap");
344 tuner_info(" B1 auto mute fm : %s\n",
345 (buf[1] & 0x02) ? "yes" : "no");
346 tuner_info(" B2 carrier mode : %s\n",
347 (buf[1] & 0x04) ? "QSS" : "Intercarrier");
348 tuner_info(" B3-4 tv sound/radio : %s\n",
349 sound[(buf[1] & 0x18) >> 3]);
350 tuner_info(" B5 force mute audio: %s\n",
351 (buf[1] & 0x20) ? "yes" : "no");
352 tuner_info(" B6 output port 1 : %s\n",
353 (buf[1] & 0x40) ? "high (inactive)" : "low (active)");
354 tuner_info(" B7 output port 2 : %s\n",
355 (buf[1] & 0x80) ? "high (inactive)" : "low (active)");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700356
Michael Krufky790ba182007-12-16 21:20:21 -0300357 tuner_info("write: byte C 0x%02x\n", buf[2]);
358 tuner_info(" C0-4 top adjustment : %s dB\n",
359 adjust[buf[2] & 0x1f]);
360 tuner_info(" C5-6 de-emphasis : %s\n",
361 deemph[(buf[2] & 0x60) >> 5]);
362 tuner_info(" C7 audio gain : %s\n",
363 (buf[2] & 0x80) ? "-6" : "0");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700364
Michael Krufky790ba182007-12-16 21:20:21 -0300365 tuner_info("write: byte E 0x%02x\n", buf[3]);
366 tuner_info(" E0-1 sound carrier : %s\n",
367 carrier[(buf[3] & 0x03)]);
368 tuner_info(" E6 l pll gating : %s\n",
369 (buf[3] & 0x40) ? "36" : "13");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700370
371 if (buf[1] & 0x08) {
372 /* radio */
Michael Krufky790ba182007-12-16 21:20:21 -0300373 tuner_info(" E2-4 video if : %s\n",
374 rif[(buf[3] & 0x0c) >> 2]);
375 tuner_info(" E7 vif agc output : %s\n",
376 (buf[3] & 0x80)
377 ? ((buf[3] & 0x10) ? "fm-agc radio" :
378 "sif-agc radio")
379 : "fm radio carrier afc");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700380 } else {
381 /* video */
Michael Krufky790ba182007-12-16 21:20:21 -0300382 tuner_info(" E2-4 video if : %s\n",
383 vif[(buf[3] & 0x1c) >> 2]);
384 tuner_info(" E5 tuner gain : %s\n",
385 (buf[3] & 0x80)
386 ? ((buf[3] & 0x20) ? "external" : "normal")
387 : ((buf[3] & 0x20) ? "minimum" : "normal"));
388 tuner_info(" E7 vif agc output : %s\n",
389 (buf[3] & 0x80) ? ((buf[3] & 0x20)
390 ? "pin3 port, pin22 vif agc out"
391 : "pin22 port, pin3 vif acg ext in")
392 : "pin3+pin22 port");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700393 }
Michael Krufky790ba182007-12-16 21:20:21 -0300394 tuner_info("--\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700395}
396
397/* ---------------------------------------------------------------------- */
398
Michael Krufky4e9154b2007-10-21 19:39:50 -0300399static int tda9887_set_tvnorm(struct dvb_frontend *fe)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700400{
Michael Krufky4e9154b2007-10-21 19:39:50 -0300401 struct tda9887_priv *priv = fe->analog_demod_priv;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700402 struct tvnorm *norm = NULL;
Michael Krufky4e9154b2007-10-21 19:39:50 -0300403 char *buf = priv->data;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700404 int i;
405
Michael Krufky91c9d4a2007-12-16 20:05:00 -0300406 if (priv->mode == V4L2_TUNER_RADIO) {
407 if (priv->audmode == V4L2_TUNER_MODE_MONO)
Mauro Carvalho Chehab56fc08c2005-06-23 22:05:07 -0700408 norm = &radio_mono;
409 else
Mauro Carvalho Chehab586b0ca2005-06-28 20:45:21 -0700410 norm = &radio_stereo;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700411 } else {
412 for (i = 0; i < ARRAY_SIZE(tvnorms); i++) {
Michael Krufky91c9d4a2007-12-16 20:05:00 -0300413 if (tvnorms[i].std & priv->std) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700414 norm = tvnorms+i;
415 break;
416 }
417 }
418 }
419 if (NULL == norm) {
Michael Krufky790ba182007-12-16 21:20:21 -0300420 tuner_dbg("Unsupported tvnorm entry - audio muted\n");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700421 return -1;
422 }
423
Michael Krufky790ba182007-12-16 21:20:21 -0300424 tuner_dbg("configure for: %s\n", norm->name);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700425 buf[1] = norm->b;
426 buf[2] = norm->c;
427 buf[3] = norm->e;
428 return 0;
429}
430
431static unsigned int port1 = UNSET;
432static unsigned int port2 = UNSET;
433static unsigned int qss = UNSET;
Hans Verkuilf98c55e2006-01-09 15:25:18 -0200434static unsigned int adjust = UNSET;
435
Linus Torvalds1da177e2005-04-16 15:20:36 -0700436module_param(port1, int, 0644);
437module_param(port2, int, 0644);
438module_param(qss, int, 0644);
439module_param(adjust, int, 0644);
440
Michael Krufky4e9154b2007-10-21 19:39:50 -0300441static int tda9887_set_insmod(struct dvb_frontend *fe)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700442{
Michael Krufky4e9154b2007-10-21 19:39:50 -0300443 struct tda9887_priv *priv = fe->analog_demod_priv;
444 char *buf = priv->data;
445
Linus Torvalds1da177e2005-04-16 15:20:36 -0700446 if (UNSET != port1) {
447 if (port1)
448 buf[1] |= cOutputPort1Inactive;
449 else
450 buf[1] &= ~cOutputPort1Inactive;
451 }
452 if (UNSET != port2) {
453 if (port2)
454 buf[1] |= cOutputPort2Inactive;
455 else
456 buf[1] &= ~cOutputPort2Inactive;
457 }
458
459 if (UNSET != qss) {
460 if (qss)
461 buf[1] |= cQSS;
462 else
463 buf[1] &= ~cQSS;
464 }
465
Hans Verkuilf98c55e2006-01-09 15:25:18 -0200466 if (adjust >= 0x00 && adjust < 0x20) {
467 buf[2] &= ~cTopMask;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700468 buf[2] |= adjust;
Hans Verkuilf98c55e2006-01-09 15:25:18 -0200469 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700470 return 0;
471}
472
Michael Krufky710401b2007-12-16 19:53:32 -0300473static int tda9887_do_config(struct dvb_frontend *fe)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700474{
Michael Krufky4e9154b2007-10-21 19:39:50 -0300475 struct tda9887_priv *priv = fe->analog_demod_priv;
Michael Krufky4e9154b2007-10-21 19:39:50 -0300476 char *buf = priv->data;
477
Michael Krufky710401b2007-12-16 19:53:32 -0300478 if (priv->config & TDA9887_PORT1_ACTIVE)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700479 buf[1] &= ~cOutputPort1Inactive;
Michael Krufky710401b2007-12-16 19:53:32 -0300480 if (priv->config & TDA9887_PORT1_INACTIVE)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700481 buf[1] |= cOutputPort1Inactive;
Michael Krufky710401b2007-12-16 19:53:32 -0300482 if (priv->config & TDA9887_PORT2_ACTIVE)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700483 buf[1] &= ~cOutputPort2Inactive;
Michael Krufky710401b2007-12-16 19:53:32 -0300484 if (priv->config & TDA9887_PORT2_INACTIVE)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700485 buf[1] |= cOutputPort2Inactive;
486
Michael Krufky710401b2007-12-16 19:53:32 -0300487 if (priv->config & TDA9887_QSS)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700488 buf[1] |= cQSS;
Michael Krufky710401b2007-12-16 19:53:32 -0300489 if (priv->config & TDA9887_INTERCARRIER)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700490 buf[1] &= ~cQSS;
491
Michael Krufky710401b2007-12-16 19:53:32 -0300492 if (priv->config & TDA9887_AUTOMUTE)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700493 buf[1] |= cAutoMuteFmActive;
Michael Krufky710401b2007-12-16 19:53:32 -0300494 if (priv->config & TDA9887_DEEMPHASIS_MASK) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700495 buf[2] &= ~0x60;
Michael Krufky710401b2007-12-16 19:53:32 -0300496 switch (priv->config & TDA9887_DEEMPHASIS_MASK) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700497 case TDA9887_DEEMPHASIS_NONE:
498 buf[2] |= cDeemphasisOFF;
499 break;
500 case TDA9887_DEEMPHASIS_50:
501 buf[2] |= cDeemphasisON | cDeemphasis50;
502 break;
503 case TDA9887_DEEMPHASIS_75:
504 buf[2] |= cDeemphasisON | cDeemphasis75;
505 break;
506 }
507 }
Michael Krufky710401b2007-12-16 19:53:32 -0300508 if (priv->config & TDA9887_TOP_SET) {
Hans Verkuilf98c55e2006-01-09 15:25:18 -0200509 buf[2] &= ~cTopMask;
Michael Krufky710401b2007-12-16 19:53:32 -0300510 buf[2] |= (priv->config >> 8) & cTopMask;
Hans Verkuilf98c55e2006-01-09 15:25:18 -0200511 }
Michael Krufky710401b2007-12-16 19:53:32 -0300512 if ((priv->config & TDA9887_INTERCARRIER_NTSC) &&
Michael Krufky91c9d4a2007-12-16 20:05:00 -0300513 (priv->std & V4L2_STD_NTSC))
Nickolay V. Shmyrev3ae1adc2005-11-08 21:37:39 -0800514 buf[1] &= ~cQSS;
Michael Krufky710401b2007-12-16 19:53:32 -0300515 if (priv->config & TDA9887_GATING_18)
Trent Piephod7304de2006-08-24 22:43:45 -0300516 buf[3] &= ~cGating_36;
Mauro Carvalho Chehabcefccc82006-12-04 08:31:35 -0300517
Michael Krufky91c9d4a2007-12-16 20:05:00 -0300518 if (priv->mode == V4L2_TUNER_RADIO) {
Michael Krufky710401b2007-12-16 19:53:32 -0300519 if (priv->config & TDA9887_RIF_41_3) {
Trent Piepho5e082f12007-08-03 18:32:38 -0300520 buf[3] &= ~cVideoIFMask;
521 buf[3] |= cRadioIF_41_30;
522 }
Michael Krufky710401b2007-12-16 19:53:32 -0300523 if (priv->config & TDA9887_GAIN_NORMAL)
Trent Piepho5e082f12007-08-03 18:32:38 -0300524 buf[3] &= ~cTunerGainLow;
Mauro Carvalho Chehabcefccc82006-12-04 08:31:35 -0300525 }
526
Linus Torvalds1da177e2005-04-16 15:20:36 -0700527 return 0;
528}
529
530/* ---------------------------------------------------------------------- */
531
Michael Krufky4e9154b2007-10-21 19:39:50 -0300532static int tda9887_status(struct dvb_frontend *fe)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700533{
Michael Krufky4e9154b2007-10-21 19:39:50 -0300534 struct tda9887_priv *priv = fe->analog_demod_priv;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700535 unsigned char buf[1];
536 int rc;
537
538 memset(buf,0,sizeof(buf));
Michael Krufkydb8a6952007-08-21 01:24:42 -0300539 if (1 != (rc = tuner_i2c_xfer_recv(&priv->i2c_props,buf,1)))
Michael Krufky790ba182007-12-16 21:20:21 -0300540 tuner_info("i2c i/o error: rc == %d (should be 1)\n", rc);
Michael Krufky4e9154b2007-10-21 19:39:50 -0300541 dump_read_message(fe, buf);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700542 return 0;
543}
544
Michael Krufky4e9154b2007-10-21 19:39:50 -0300545static void tda9887_configure(struct dvb_frontend *fe)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700546{
Michael Krufky4e9154b2007-10-21 19:39:50 -0300547 struct tda9887_priv *priv = fe->analog_demod_priv;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700548 int rc;
549
Michael Krufkyb2083192007-05-29 22:54:06 -0300550 memset(priv->data,0,sizeof(priv->data));
Michael Krufky4e9154b2007-10-21 19:39:50 -0300551 tda9887_set_tvnorm(fe);
Mauro Carvalho Chehab56fc08c2005-06-23 22:05:07 -0700552
Hans Verkuilf98c55e2006-01-09 15:25:18 -0200553 /* A note on the port settings:
554 These settings tend to depend on the specifics of the board.
555 By default they are set to inactive (bit value 1) by this driver,
556 overwriting any changes made by the tvnorm. This means that it
557 is the responsibility of the module using the tda9887 to set
558 these values in case of changes in the tvnorm.
559 In many cases port 2 should be made active (0) when selecting
560 SECAM-L, and port 2 should remain inactive (1) for SECAM-L'.
561
562 For the other standards the tda9887 application note says that
563 the ports should be set to active (0), but, again, that may
564 differ depending on the precise hardware configuration.
565 */
Michael Krufkyb2083192007-05-29 22:54:06 -0300566 priv->data[1] |= cOutputPort1Inactive;
567 priv->data[1] |= cOutputPort2Inactive;
Mauro Carvalho Chehab56fc08c2005-06-23 22:05:07 -0700568
Michael Krufky710401b2007-12-16 19:53:32 -0300569 tda9887_do_config(fe);
Michael Krufky4e9154b2007-10-21 19:39:50 -0300570 tda9887_set_insmod(fe);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700571
Michael Krufky91c9d4a2007-12-16 20:05:00 -0300572 if (priv->mode == T_STANDBY)
Michael Krufkyb2083192007-05-29 22:54:06 -0300573 priv->data[1] |= cForcedMuteAudioON;
Mauro Carvalho Chehab793cf9e2005-09-09 13:03:37 -0700574
Michael Krufky790ba182007-12-16 21:20:21 -0300575 tuner_dbg("writing: b=0x%02x c=0x%02x e=0x%02x\n",
576 priv->data[1], priv->data[2], priv->data[3]);
577 if (debug > 1)
Michael Krufky4e9154b2007-10-21 19:39:50 -0300578 dump_write_message(fe, priv->data);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700579
Michael Krufkydb8a6952007-08-21 01:24:42 -0300580 if (4 != (rc = tuner_i2c_xfer_send(&priv->i2c_props,priv->data,4)))
Michael Krufky790ba182007-12-16 21:20:21 -0300581 tuner_info("i2c i/o error: rc == %d (should be 4)\n", rc);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700582
Michael Krufky790ba182007-12-16 21:20:21 -0300583 if (debug > 2) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700584 msleep_interruptible(1000);
Michael Krufky4e9154b2007-10-21 19:39:50 -0300585 tda9887_status(fe);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700586 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700587}
588
589/* ---------------------------------------------------------------------- */
590
Michael Krufky4e9154b2007-10-21 19:39:50 -0300591static void tda9887_tuner_status(struct dvb_frontend *fe)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700592{
Michael Krufky4e9154b2007-10-21 19:39:50 -0300593 struct tda9887_priv *priv = fe->analog_demod_priv;
Michael Krufky790ba182007-12-16 21:20:21 -0300594 tuner_info("Data bytes: b=0x%02x c=0x%02x e=0x%02x\n",
595 priv->data[1], priv->data[2], priv->data[3]);
Mauro Carvalho Chehab15396232006-06-23 16:13:56 -0300596}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700597
Michael Krufky4e9154b2007-10-21 19:39:50 -0300598static int tda9887_get_afc(struct dvb_frontend *fe)
Mauro Carvalho Chehab15396232006-06-23 16:13:56 -0300599{
Michael Krufky4e9154b2007-10-21 19:39:50 -0300600 struct tda9887_priv *priv = fe->analog_demod_priv;
Mauro Carvalho Chehab15396232006-06-23 16:13:56 -0300601 static int AFC_BITS_2_kHz[] = {
602 -12500, -37500, -62500, -97500,
603 -112500, -137500, -162500, -187500,
604 187500, 162500, 137500, 112500,
605 97500 , 62500, 37500 , 12500
606 };
607 int afc=0;
608 __u8 reg = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700609
Michael Krufkydb8a6952007-08-21 01:24:42 -0300610 if (1 == tuner_i2c_xfer_recv(&priv->i2c_props,&reg,1))
Mauro Carvalho Chehab15396232006-06-23 16:13:56 -0300611 afc = AFC_BITS_2_kHz[(reg>>1)&0x0f];
Mauro Carvalho Chehab56fc08c2005-06-23 22:05:07 -0700612
Mauro Carvalho Chehab15396232006-06-23 16:13:56 -0300613 return afc;
614}
Mauro Carvalho Chehab56fc08c2005-06-23 22:05:07 -0700615
Michael Krufky4e9154b2007-10-21 19:39:50 -0300616static void tda9887_standby(struct dvb_frontend *fe)
Mauro Carvalho Chehab15396232006-06-23 16:13:56 -0300617{
Michael Krufky91c9d4a2007-12-16 20:05:00 -0300618 struct tda9887_priv *priv = fe->analog_demod_priv;
619
620 priv->mode = T_STANDBY;
621
Michael Krufky4e9154b2007-10-21 19:39:50 -0300622 tda9887_configure(fe);
Mauro Carvalho Chehab15396232006-06-23 16:13:56 -0300623}
Hans Verkuil4eb0c142005-11-08 21:37:18 -0800624
Michael Krufkyc7919d52007-12-08 17:06:30 -0300625static void tda9887_set_params(struct dvb_frontend *fe,
626 struct analog_parameters *params)
Mauro Carvalho Chehab15396232006-06-23 16:13:56 -0300627{
Michael Krufky91c9d4a2007-12-16 20:05:00 -0300628 struct tda9887_priv *priv = fe->analog_demod_priv;
629
630 priv->mode = params->mode;
631 priv->audmode = params->audmode;
632 priv->std = params->std;
Michael Krufky4e9154b2007-10-21 19:39:50 -0300633 tda9887_configure(fe);
Mauro Carvalho Chehab15396232006-06-23 16:13:56 -0300634}
635
Michael Krufky710401b2007-12-16 19:53:32 -0300636static int tda9887_set_config(struct dvb_frontend *fe, void *priv_cfg)
637{
638 struct tda9887_priv *priv = fe->analog_demod_priv;
639
640 priv->config = *(unsigned int *)priv_cfg;
641 tda9887_configure(fe);
642
643 return 0;
644}
645
Michael Krufky4e9154b2007-10-21 19:39:50 -0300646static void tda9887_release(struct dvb_frontend *fe)
Michael Krufky024cf532007-06-04 15:20:11 -0300647{
Michael Krufkyac8b63b2008-04-22 14:45:51 -0300648 struct tda9887_priv *priv = fe->analog_demod_priv;
649
650 mutex_lock(&tda9887_list_mutex);
651
652 if (priv)
653 hybrid_tuner_release_state(priv);
654
655 mutex_unlock(&tda9887_list_mutex);
656
Michael Krufky4e9154b2007-10-21 19:39:50 -0300657 fe->analog_demod_priv = NULL;
Michael Krufky024cf532007-06-04 15:20:11 -0300658}
659
Michael Krufkybc3e5c72007-12-21 11:18:32 -0300660static struct analog_demod_ops tda9887_ops = {
Michael Krufkya55db8c2007-12-09 13:52:51 -0300661 .info = {
Michael Krufky0f2ce982008-01-21 10:55:37 -0300662 .name = "tda9887",
Michael Krufkya55db8c2007-12-09 13:52:51 -0300663 },
Michael Krufkyc7919d52007-12-08 17:06:30 -0300664 .set_params = tda9887_set_params,
Michael Krufky9af596e2007-06-06 16:15:48 -0300665 .standby = tda9887_standby,
666 .tuner_status = tda9887_tuner_status,
667 .get_afc = tda9887_get_afc,
668 .release = tda9887_release,
Michael Krufky710401b2007-12-16 19:53:32 -0300669 .set_config = tda9887_set_config,
Michael Krufky9af596e2007-06-06 16:15:48 -0300670};
671
Michael Krufky8ca40832007-12-16 20:11:46 -0300672struct dvb_frontend *tda9887_attach(struct dvb_frontend *fe,
673 struct i2c_adapter *i2c_adap,
674 u8 i2c_addr)
Mauro Carvalho Chehab15396232006-06-23 16:13:56 -0300675{
Michael Krufkyb2083192007-05-29 22:54:06 -0300676 struct tda9887_priv *priv = NULL;
Michael Krufkyac8b63b2008-04-22 14:45:51 -0300677 int instance;
Mauro Carvalho Chehab15396232006-06-23 16:13:56 -0300678
Michael Krufkyac8b63b2008-04-22 14:45:51 -0300679 mutex_lock(&tda9887_list_mutex);
680
681 instance = hybrid_tuner_request_state(struct tda9887_priv, priv,
682 hybrid_tuner_instance_list,
683 i2c_adap, i2c_addr, "tda9887");
684 switch (instance) {
685 case 0:
686 mutex_unlock(&tda9887_list_mutex);
Michael Krufky8ca40832007-12-16 20:11:46 -0300687 return NULL;
Michael Krufkyac8b63b2008-04-22 14:45:51 -0300688 case 1:
689 fe->analog_demod_priv = priv;
690 priv->mode = T_STANDBY;
691 tuner_info("tda988[5/6/7] found\n");
692 break;
693 default:
694 fe->analog_demod_priv = priv;
695 break;
696 }
Michael Krufkyb2083192007-05-29 22:54:06 -0300697
Michael Krufkyac8b63b2008-04-22 14:45:51 -0300698 mutex_unlock(&tda9887_list_mutex);
Mauro Carvalho Chehab15396232006-06-23 16:13:56 -0300699
Michael Krufkybc3e5c72007-12-21 11:18:32 -0300700 memcpy(&fe->ops.analog_ops, &tda9887_ops,
701 sizeof(struct analog_demod_ops));
Mauro Carvalho Chehab15396232006-06-23 16:13:56 -0300702
Michael Krufky8ca40832007-12-16 20:11:46 -0300703 return fe;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700704}
Michael Krufky31c95842007-10-21 20:48:48 -0300705EXPORT_SYMBOL_GPL(tda9887_attach);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700706
Michael Krufky5ef47302007-10-27 02:17:19 -0300707MODULE_LICENSE("GPL");
708
Linus Torvalds1da177e2005-04-16 15:20:36 -0700709/*
710 * Overrides for Emacs so that we follow Linus's tabbing style.
711 * ---------------------------------------------------------------------------
712 * Local variables:
713 * c-basic-offset: 8
714 * End:
715 */