blob: 5bf9049c6c00dc3baba0b0081f79e171022e48db [file] [log] [blame]
Johannes Stezenbach2add87a2005-05-16 21:54:10 -07001/*
Patrick Boettcherd66b94b2009-05-20 05:08:26 -03002 * Linux driver for digital TV devices equipped with B2C2 FlexcopII(b)/III
3 * flexcop-fe-tuner.c - methods for frontend attachment and DiSEqC controlling
4 * see flexcop.c for copyright information
Johannes Stezenbach2add87a2005-05-16 21:54:10 -07005 */
Michael Krufky827855d2008-04-22 14:46:16 -03006#include <media/tuner.h>
Johannes Stezenbach2add87a2005-05-16 21:54:10 -07007#include "flexcop.h"
Johannes Stezenbach2add87a2005-05-16 21:54:10 -07008#include "mt312.h"
Patrick Boettcherd66b94b2009-05-20 05:08:26 -03009#include "stv0299.h"
Patrick Boettcherca19aaa2008-04-13 15:49:22 -030010#include "s5h1420.h"
11#include "itd1000.h"
Patrick Boettcherc9dd82c2008-03-29 21:28:07 -030012#include "cx24113.h"
Patrick Boettcherd66b94b2009-05-20 05:08:26 -030013#include "cx24123.h"
Patrick Boettcherc9dd82c2008-03-29 21:28:07 -030014#include "isl6421.h"
Patrick Boettcherd66b94b2009-05-20 05:08:26 -030015#include "mt352.h"
16#include "bcm3510.h"
17#include "nxt200x.h"
18#include "dvb-pll.h"
19#include "lgdt330x.h"
20#include "tuner-simple.h"
21#include "stv0297.h"
Patrick Boettcherc9dd82c2008-03-29 21:28:07 -030022
Trent Piepho68b7f762009-06-11 19:31:22 -030023
24/* Can we use the specified front-end? Remember that if we are compiled
25 * into the kernel we can't call code that's in modules. */
26#define FE_SUPPORTED(fe) (defined(CONFIG_DVB_##fe) || \
27 (defined(CONFIG_DVB_##fe##_MODULE) && defined(MODULE)))
28
Johannes Stezenbach2add87a2005-05-16 21:54:10 -070029/* lnb control */
Trent Piepho68b7f762009-06-11 19:31:22 -030030#if FE_SUPPORTED(MT312) || FE_SUPPORTED(STV0299)
Johannes Stezenbach2add87a2005-05-16 21:54:10 -070031static int flexcop_set_voltage(struct dvb_frontend *fe, fe_sec_voltage_t voltage)
32{
33 struct flexcop_device *fc = fe->dvb->priv;
34 flexcop_ibi_value v;
35 deb_tuner("polarity/voltage = %u\n", voltage);
36
37 v = fc->read_ibi_reg(fc, misc_204);
38 switch (voltage) {
Patrick Boettcherd66b94b2009-05-20 05:08:26 -030039 case SEC_VOLTAGE_OFF:
40 v.misc_204.ACPI1_sig = 1;
41 break;
42 case SEC_VOLTAGE_13:
43 v.misc_204.ACPI1_sig = 0;
44 v.misc_204.LNB_L_H_sig = 0;
45 break;
46 case SEC_VOLTAGE_18:
47 v.misc_204.ACPI1_sig = 0;
48 v.misc_204.LNB_L_H_sig = 1;
49 break;
50 default:
51 err("unknown SEC_VOLTAGE value");
52 return -EINVAL;
Johannes Stezenbach2add87a2005-05-16 21:54:10 -070053 }
54 return fc->write_ibi_reg(fc, misc_204, v);
55}
Patrick Boettcherd66b94b2009-05-20 05:08:26 -030056#endif
Johannes Stezenbach2add87a2005-05-16 21:54:10 -070057
Trent Piepho68b7f762009-06-11 19:31:22 -030058#if FE_SUPPORTED(S5H1420) || FE_SUPPORTED(STV0299) || FE_SUPPORTED(MT312)
Johannes Stezenbach2add87a2005-05-16 21:54:10 -070059static int flexcop_sleep(struct dvb_frontend* fe)
60{
61 struct flexcop_device *fc = fe->dvb->priv;
Johannes Stezenbach2add87a2005-05-16 21:54:10 -070062 if (fc->fe_sleep)
63 return fc->fe_sleep(fe);
Johannes Stezenbach2add87a2005-05-16 21:54:10 -070064 return 0;
65}
Patrick Boettcherd66b94b2009-05-20 05:08:26 -030066#endif
Johannes Stezenbach2add87a2005-05-16 21:54:10 -070067
Patrick Boettcherd66b94b2009-05-20 05:08:26 -030068/* SkyStar2 DVB-S rev 2.3 */
Trent Piepho016e53d2009-06-11 19:31:22 -030069#if FE_SUPPORTED(MT312) && FE_SUPPORTED(PLL)
Johannes Stezenbach2add87a2005-05-16 21:54:10 -070070static int flexcop_set_tone(struct dvb_frontend *fe, fe_sec_tone_mode_t tone)
71{
Patrick Boettcherd66b94b2009-05-20 05:08:26 -030072/* u16 wz_half_period_for_45_mhz[] = { 0x01ff, 0x0154, 0x00ff, 0x00cc }; */
Johannes Stezenbach2add87a2005-05-16 21:54:10 -070073 struct flexcop_device *fc = fe->dvb->priv;
74 flexcop_ibi_value v;
75 u16 ax;
76 v.raw = 0;
Johannes Stezenbach2add87a2005-05-16 21:54:10 -070077 deb_tuner("tone = %u\n",tone);
78
79 switch (tone) {
Patrick Boettcherd66b94b2009-05-20 05:08:26 -030080 case SEC_TONE_ON:
81 ax = 0x01ff;
82 break;
83 case SEC_TONE_OFF:
84 ax = 0;
85 break;
86 default:
87 err("unknown SEC_TONE value");
88 return -EINVAL;
Johannes Stezenbach2add87a2005-05-16 21:54:10 -070089 }
90
91 v.lnb_switch_freq_200.LNB_CTLPrescaler_sig = 1; /* divide by 2 */
Johannes Stezenbach958706c2005-05-16 21:54:19 -070092 v.lnb_switch_freq_200.LNB_CTLHighCount_sig = ax;
93 v.lnb_switch_freq_200.LNB_CTLLowCount_sig = ax == 0 ? 0x1ff : ax;
Johannes Stezenbach2add87a2005-05-16 21:54:10 -070094 return fc->write_ibi_reg(fc,lnb_switch_freq_200,v);
95}
96
97static void flexcop_diseqc_send_bit(struct dvb_frontend* fe, int data)
98{
99 flexcop_set_tone(fe, SEC_TONE_ON);
100 udelay(data ? 500 : 1000);
101 flexcop_set_tone(fe, SEC_TONE_OFF);
102 udelay(data ? 1000 : 500);
103}
104
105static void flexcop_diseqc_send_byte(struct dvb_frontend* fe, int data)
106{
107 int i, par = 1, d;
Johannes Stezenbach2add87a2005-05-16 21:54:10 -0700108 for (i = 7; i >= 0; i--) {
109 d = (data >> i) & 1;
110 par ^= d;
111 flexcop_diseqc_send_bit(fe, d);
112 }
Johannes Stezenbach2add87a2005-05-16 21:54:10 -0700113 flexcop_diseqc_send_bit(fe, par);
114}
115
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300116static int flexcop_send_diseqc_msg(struct dvb_frontend *fe,
117 int len, u8 *msg, unsigned long burst)
Johannes Stezenbach2add87a2005-05-16 21:54:10 -0700118{
119 int i;
120
121 flexcop_set_tone(fe, SEC_TONE_OFF);
122 mdelay(16);
123
124 for (i = 0; i < len; i++)
125 flexcop_diseqc_send_byte(fe,msg[i]);
Johannes Stezenbach2add87a2005-05-16 21:54:10 -0700126 mdelay(16);
127
128 if (burst != -1) {
129 if (burst)
130 flexcop_diseqc_send_byte(fe, 0xff);
131 else {
132 flexcop_set_tone(fe, SEC_TONE_ON);
Thierry MERLEc4e3fd92008-09-01 17:32:10 -0300133 mdelay(12);
134 udelay(500);
Johannes Stezenbach2add87a2005-05-16 21:54:10 -0700135 flexcop_set_tone(fe, SEC_TONE_OFF);
136 }
137 msleep(20);
138 }
139 return 0;
140}
141
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300142static int flexcop_diseqc_send_master_cmd(struct dvb_frontend *fe,
143 struct dvb_diseqc_master_cmd *cmd)
Johannes Stezenbach2add87a2005-05-16 21:54:10 -0700144{
145 return flexcop_send_diseqc_msg(fe, cmd->msg_len, cmd->msg, 0);
146}
147
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300148static int flexcop_diseqc_send_burst(struct dvb_frontend *fe,
149 fe_sec_mini_cmd_t minicmd)
Johannes Stezenbach2add87a2005-05-16 21:54:10 -0700150{
151 return flexcop_send_diseqc_msg(fe, 0, NULL, minicmd);
152}
153
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300154static struct mt312_config skystar23_samsung_tbdu18132_config = {
155 .demod_address = 0x0e,
156};
Johannes Stezenbach2add87a2005-05-16 21:54:10 -0700157
Trent Piephoeccd15a2009-06-11 05:33:00 -0300158static int skystar2_rev23_attach(struct flexcop_device *fc,
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300159 struct i2c_adapter *i2c)
160{
Trent Piepho016e53d2009-06-11 19:31:22 -0300161 struct dvb_frontend_ops *ops;
162
Trent Piephoeccd15a2009-06-11 05:33:00 -0300163 fc->fe = dvb_attach(mt312_attach, &skystar23_samsung_tbdu18132_config, i2c);
Trent Piepho016e53d2009-06-11 19:31:22 -0300164 if (!fc->fe)
165 return 0;
166
167 if (!dvb_attach(dvb_pll_attach, fc->fe, 0x61, i2c,
168 DVB_PLL_SAMSUNG_TBDU18132))
169 return 0;
170
171 ops = &fc->fe->ops;
172 ops->diseqc_send_master_cmd = flexcop_diseqc_send_master_cmd;
173 ops->diseqc_send_burst = flexcop_diseqc_send_burst;
174 ops->set_tone = flexcop_set_tone;
175 ops->set_voltage = flexcop_set_voltage;
176 fc->fe_sleep = ops->sleep;
177 ops->sleep = flexcop_sleep;
178 return 1;
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300179}
Trent Piepho68b7f762009-06-11 19:31:22 -0300180#else
181#define skystar2_rev23_attach NULL
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300182#endif
Johannes Stezenbach2add87a2005-05-16 21:54:10 -0700183
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300184/* SkyStar2 DVB-S rev 2.6 */
Trent Piepho016e53d2009-06-11 19:31:22 -0300185#if FE_SUPPORTED(STV0299) && FE_SUPPORTED(PLL)
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300186static int samsung_tbmu24112_set_symbol_rate(struct dvb_frontend *fe,
187 u32 srate, u32 ratio)
188{
189 u8 aclk = 0;
190 u8 bclk = 0;
191
192 if (srate < 1500000) {
193 aclk = 0xb7; bclk = 0x47;
194 } else if (srate < 3000000) {
195 aclk = 0xb7; bclk = 0x4b;
196 } else if (srate < 7000000) {
197 aclk = 0xb7; bclk = 0x4f;
198 } else if (srate < 14000000) {
199 aclk = 0xb7; bclk = 0x53;
200 } else if (srate < 30000000) {
201 aclk = 0xb6; bclk = 0x53;
202 } else if (srate < 45000000) {
203 aclk = 0xb4; bclk = 0x51;
204 }
205
206 stv0299_writereg(fe, 0x13, aclk);
207 stv0299_writereg(fe, 0x14, bclk);
208 stv0299_writereg(fe, 0x1f, (ratio >> 16) & 0xff);
209 stv0299_writereg(fe, 0x20, (ratio >> 8) & 0xff);
210 stv0299_writereg(fe, 0x21, ratio & 0xf0);
211 return 0;
212}
213
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300214static u8 samsung_tbmu24112_inittab[] = {
215 0x01, 0x15,
216 0x02, 0x30,
217 0x03, 0x00,
218 0x04, 0x7D,
219 0x05, 0x35,
220 0x06, 0x02,
221 0x07, 0x00,
222 0x08, 0xC3,
223 0x0C, 0x00,
224 0x0D, 0x81,
225 0x0E, 0x23,
226 0x0F, 0x12,
227 0x10, 0x7E,
228 0x11, 0x84,
229 0x12, 0xB9,
230 0x13, 0x88,
231 0x14, 0x89,
232 0x15, 0xC9,
233 0x16, 0x00,
234 0x17, 0x5C,
235 0x18, 0x00,
236 0x19, 0x00,
237 0x1A, 0x00,
238 0x1C, 0x00,
239 0x1D, 0x00,
240 0x1E, 0x00,
241 0x1F, 0x3A,
242 0x20, 0x2E,
243 0x21, 0x80,
244 0x22, 0xFF,
245 0x23, 0xC1,
246 0x28, 0x00,
247 0x29, 0x1E,
248 0x2A, 0x14,
249 0x2B, 0x0F,
250 0x2C, 0x09,
251 0x2D, 0x05,
252 0x31, 0x1F,
253 0x32, 0x19,
254 0x33, 0xFE,
255 0x34, 0x93,
256 0xff, 0xff,
Johannes Stezenbach2add87a2005-05-16 21:54:10 -0700257};
258
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300259static struct stv0299_config samsung_tbmu24112_config = {
260 .demod_address = 0x68,
261 .inittab = samsung_tbmu24112_inittab,
262 .mclk = 88000000UL,
263 .invert = 0,
264 .skip_reinit = 0,
265 .lock_output = STV0299_LOCKOUTPUT_LK,
266 .volt13_op0_op1 = STV0299_VOLT13_OP1,
267 .min_delay_ms = 100,
268 .set_symbol_rate = samsung_tbmu24112_set_symbol_rate,
269};
270
Trent Piephoeccd15a2009-06-11 05:33:00 -0300271static int skystar2_rev26_attach(struct flexcop_device *fc,
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300272 struct i2c_adapter *i2c)
273{
274 fc->fe = dvb_attach(stv0299_attach, &samsung_tbmu24112_config, i2c);
Trent Piepho016e53d2009-06-11 19:31:22 -0300275 if (!fc->fe)
276 return 0;
277
278 if (!dvb_attach(dvb_pll_attach, fc->fe, 0x61, i2c,
279 DVB_PLL_SAMSUNG_TBMU24112))
280 return 0;
281
282 fc->fe->ops.set_voltage = flexcop_set_voltage;
283 fc->fe_sleep = fc->fe->ops.sleep;
284 fc->fe->ops.sleep = flexcop_sleep;
285 return 1;
286
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300287}
Trent Piepho68b7f762009-06-11 19:31:22 -0300288#else
289#define skystar2_rev26_attach NULL
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300290#endif
291
292/* SkyStar2 DVB-S rev 2.7 */
Trent Piepho68b7f762009-06-11 19:31:22 -0300293#if FE_SUPPORTED(S5H1420) && FE_SUPPORTED(ISL6421) && FE_SUPPORTED(TUNER_ITD1000)
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300294static struct s5h1420_config skystar2_rev2_7_s5h1420_config = {
295 .demod_address = 0x53,
296 .invert = 1,
297 .repeated_start_workaround = 1,
298 .serial_mpeg = 1,
299};
300
301static struct itd1000_config skystar2_rev2_7_itd1000_config = {
302 .i2c_address = 0x61,
303};
304
Trent Piephoeccd15a2009-06-11 05:33:00 -0300305static int skystar2_rev27_attach(struct flexcop_device *fc,
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300306 struct i2c_adapter *i2c)
307{
Trent Piephoeccd15a2009-06-11 05:33:00 -0300308 flexcop_ibi_value r108;
309 struct i2c_adapter *i2c_tuner;
310
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300311 /* enable no_base_addr - no repeated start when reading */
312 fc->fc_i2c_adap[0].no_base_addr = 1;
Trent Piephoeccd15a2009-06-11 05:33:00 -0300313 fc->fe = dvb_attach(s5h1420_attach, &skystar2_rev2_7_s5h1420_config,
314 i2c);
315 if (!fc->fe)
316 goto fail;
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300317
Trent Piephoeccd15a2009-06-11 05:33:00 -0300318 i2c_tuner = s5h1420_get_tuner_i2c_adapter(fc->fe);
319 if (!i2c_tuner)
320 goto fail;
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300321
Trent Piephoeccd15a2009-06-11 05:33:00 -0300322 fc->fe_sleep = fc->fe->ops.sleep;
323 fc->fe->ops.sleep = flexcop_sleep;
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300324
Trent Piephoeccd15a2009-06-11 05:33:00 -0300325 /* enable no_base_addr - no repeated start when reading */
326 fc->fc_i2c_adap[2].no_base_addr = 1;
327 if (!dvb_attach(isl6421_attach, fc->fe, &fc->fc_i2c_adap[2].i2c_adap,
328 0x08, 1, 1)) {
329 err("ISL6421 could NOT be attached");
330 goto fail_isl;
331 }
332 info("ISL6421 successfully attached");
333
334 /* the ITD1000 requires a lower i2c clock - is it a problem ? */
335 r108.raw = 0x00000506;
336 fc->write_ibi_reg(fc, tw_sm_c_108, r108);
337 if (!dvb_attach(itd1000_attach, fc->fe, i2c_tuner,
338 &skystar2_rev2_7_itd1000_config)) {
339 err("ITD1000 could NOT be attached");
340 /* Should i2c clock be restored? */
341 goto fail_isl;
342 }
343 info("ITD1000 successfully attached");
344
345 return 1;
346
347fail_isl:
348 fc->fc_i2c_adap[2].no_base_addr = 0;
349fail:
350 /* for the next devices we need it again */
351 fc->fc_i2c_adap[0].no_base_addr = 0;
352 return 0;
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300353}
Trent Piepho68b7f762009-06-11 19:31:22 -0300354#else
355#define skystar2_rev27_attach NULL
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300356#endif
357
358/* SkyStar2 rev 2.8 */
Trent Piepho68b7f762009-06-11 19:31:22 -0300359#if FE_SUPPORTED(CX24123) && FE_SUPPORTED(ISL6421) && FE_SUPPORTED(TUNER_CX24113)
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300360static struct cx24123_config skystar2_rev2_8_cx24123_config = {
361 .demod_address = 0x55,
362 .dont_use_pll = 1,
363 .agc_callback = cx24113_agc_callback,
364};
365
366static const struct cx24113_config skystar2_rev2_8_cx24113_config = {
367 .i2c_addr = 0x54,
368 .xtal_khz = 10111,
369};
370
Trent Piephoeccd15a2009-06-11 05:33:00 -0300371static int skystar2_rev28_attach(struct flexcop_device *fc,
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300372 struct i2c_adapter *i2c)
373{
Trent Piephoeccd15a2009-06-11 05:33:00 -0300374 struct i2c_adapter *i2c_tuner;
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300375
Trent Piephoeccd15a2009-06-11 05:33:00 -0300376 fc->fe = dvb_attach(cx24123_attach, &skystar2_rev2_8_cx24123_config,
377 i2c);
378 if (!fc->fe)
379 return 0;
380
Joe Perches1ebcad72009-07-02 15:57:09 -0300381 i2c_tuner = cx24123_get_tuner_i2c_adapter(fc->fe);
Trent Piephoeccd15a2009-06-11 05:33:00 -0300382 if (!i2c_tuner)
383 return 0;
384
385 if (!dvb_attach(cx24113_attach, fc->fe, &skystar2_rev2_8_cx24113_config,
386 i2c_tuner)) {
387 err("CX24113 could NOT be attached");
388 return 0;
389 }
390 info("CX24113 successfully attached");
391
392 fc->fc_i2c_adap[2].no_base_addr = 1;
393 if (!dvb_attach(isl6421_attach, fc->fe, &fc->fc_i2c_adap[2].i2c_adap,
394 0x08, 0, 0)) {
395 err("ISL6421 could NOT be attached");
396 fc->fc_i2c_adap[2].no_base_addr = 0;
397 return 0;
398 }
399 info("ISL6421 successfully attached");
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300400 /* TODO on i2c_adap[1] addr 0x11 (EEPROM) there seems to be an
401 * IR-receiver (PIC16F818) - but the card has no input for that ??? */
Trent Piephoeccd15a2009-06-11 05:33:00 -0300402 return 1;
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300403}
Trent Piepho68b7f762009-06-11 19:31:22 -0300404#else
405#define skystar2_rev28_attach NULL
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300406#endif
407
408/* AirStar DVB-T */
Trent Piepho6d67c972009-06-11 19:31:22 -0300409#if FE_SUPPORTED(MT352) && FE_SUPPORTED(PLL)
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300410static int samsung_tdtc9251dh0_demod_init(struct dvb_frontend *fe)
411{
412 static u8 mt352_clock_config[] = { 0x89, 0x18, 0x2d };
413 static u8 mt352_reset[] = { 0x50, 0x80 };
414 static u8 mt352_adc_ctl_1_cfg[] = { 0x8E, 0x40 };
415 static u8 mt352_agc_cfg[] = { 0x67, 0x28, 0xa1 };
416 static u8 mt352_capt_range_cfg[] = { 0x75, 0x32 };
417
418 mt352_write(fe, mt352_clock_config, sizeof(mt352_clock_config));
419 udelay(2000);
420 mt352_write(fe, mt352_reset, sizeof(mt352_reset));
421 mt352_write(fe, mt352_adc_ctl_1_cfg, sizeof(mt352_adc_ctl_1_cfg));
422 mt352_write(fe, mt352_agc_cfg, sizeof(mt352_agc_cfg));
423 mt352_write(fe, mt352_capt_range_cfg, sizeof(mt352_capt_range_cfg));
424 return 0;
425}
426
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300427static struct mt352_config samsung_tdtc9251dh0_config = {
428 .demod_address = 0x0f,
429 .demod_init = samsung_tdtc9251dh0_demod_init,
430};
431
Trent Piephoeccd15a2009-06-11 05:33:00 -0300432static int airstar_dvbt_attach(struct flexcop_device *fc,
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300433 struct i2c_adapter *i2c)
434{
435 fc->fe = dvb_attach(mt352_attach, &samsung_tdtc9251dh0_config, i2c);
Trent Piepho6d67c972009-06-11 19:31:22 -0300436 if (!fc->fe)
437 return 0;
438
439 return !!dvb_attach(dvb_pll_attach, fc->fe, 0x61, NULL,
440 DVB_PLL_SAMSUNG_TDTC9251DH0);
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300441}
Trent Piepho68b7f762009-06-11 19:31:22 -0300442#else
443#define airstar_dvbt_attach NULL
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300444#endif
445
446/* AirStar ATSC 1st generation */
Trent Piepho68b7f762009-06-11 19:31:22 -0300447#if FE_SUPPORTED(BCM3510)
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300448static int flexcop_fe_request_firmware(struct dvb_frontend *fe,
449 const struct firmware **fw, char* name)
450{
451 struct flexcop_device *fc = fe->dvb->priv;
452 return request_firmware(fw, name, fc->dev);
453}
454
455static struct bcm3510_config air2pc_atsc_first_gen_config = {
456 .demod_address = 0x0f,
457 .request_firmware = flexcop_fe_request_firmware,
458};
459
Trent Piephoeccd15a2009-06-11 05:33:00 -0300460static int airstar_atsc1_attach(struct flexcop_device *fc,
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300461 struct i2c_adapter *i2c)
462{
463 fc->fe = dvb_attach(bcm3510_attach, &air2pc_atsc_first_gen_config, i2c);
Trent Piephoeccd15a2009-06-11 05:33:00 -0300464 return fc->fe != NULL;
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300465}
Trent Piepho68b7f762009-06-11 19:31:22 -0300466#else
467#define airstar_atsc1_attach NULL
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300468#endif
469
470/* AirStar ATSC 2nd generation */
Trent Piepho68b7f762009-06-11 19:31:22 -0300471#if FE_SUPPORTED(NXT200X) && FE_SUPPORTED(PLL)
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300472static struct nxt200x_config samsung_tbmv_config = {
473 .demod_address = 0x0a,
474};
475
Trent Piephoeccd15a2009-06-11 05:33:00 -0300476static int airstar_atsc2_attach(struct flexcop_device *fc,
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300477 struct i2c_adapter *i2c)
478{
479 fc->fe = dvb_attach(nxt200x_attach, &samsung_tbmv_config, i2c);
Trent Piephoeccd15a2009-06-11 05:33:00 -0300480 if (!fc->fe)
481 return 0;
482
483 return !!dvb_attach(dvb_pll_attach, fc->fe, 0x61, NULL,
484 DVB_PLL_SAMSUNG_TBMV);
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300485}
Trent Piepho68b7f762009-06-11 19:31:22 -0300486#else
487#define airstar_atsc2_attach NULL
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300488#endif
489
490/* AirStar ATSC 3rd generation */
Trent Piepho68b7f762009-06-11 19:31:22 -0300491#if FE_SUPPORTED(LGDT330X)
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300492static struct lgdt330x_config air2pc_atsc_hd5000_config = {
493 .demod_address = 0x59,
494 .demod_chip = LGDT3303,
495 .serial_mpeg = 0x04,
496 .clock_polarity_flip = 1,
497};
498
Trent Piephoeccd15a2009-06-11 05:33:00 -0300499static int airstar_atsc3_attach(struct flexcop_device *fc,
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300500 struct i2c_adapter *i2c)
501{
502 fc->fe = dvb_attach(lgdt330x_attach, &air2pc_atsc_hd5000_config, i2c);
Trent Piephoeccd15a2009-06-11 05:33:00 -0300503 if (!fc->fe)
504 return 0;
505
506 return !!dvb_attach(simple_tuner_attach, fc->fe, i2c, 0x61,
507 TUNER_LG_TDVS_H06XF);
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300508}
Trent Piepho68b7f762009-06-11 19:31:22 -0300509#else
510#define airstar_atsc3_attach NULL
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300511#endif
512
513/* CableStar2 DVB-C */
Trent Piepho68b7f762009-06-11 19:31:22 -0300514#if FE_SUPPORTED(STV0297)
Adrian Bunkdd00b1e2006-05-29 12:31:44 -0300515static int alps_tdee4_stv0297_tuner_set_params(struct dvb_frontend* fe,
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300516 struct dvb_frontend_parameters *fep)
Andrew de Quincey56e03142006-04-18 17:47:12 -0300517{
518 struct flexcop_device *fc = fe->dvb->priv;
519 u8 buf[4];
520 u16 div;
521 int ret;
522
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300523/* 62.5 kHz * 10 */
Andrew de Quincey56e03142006-04-18 17:47:12 -0300524#define REF_FREQ 625
525#define FREQ_OFFSET 36125
526
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300527 div = ((fep->frequency/1000 + FREQ_OFFSET) * 10) / REF_FREQ;
528/* 4 MHz = 4000 KHz */
Andrew de Quincey56e03142006-04-18 17:47:12 -0300529
530 buf[0] = (u8)( div >> 8) & 0x7f;
531 buf[1] = (u8) div & 0xff;
532
533/* F(osc) = N * Reference Freq. (62.5 kHz)
534 * byte 2 : 0 N14 N13 N12 N11 N10 N9 N8
535 * byte 3 : N7 N6 N5 N4 N3 N2 N1 N0
536 * byte 4 : 1 * * AGD R3 R2 R1 R0
537 * byte 5 : C1 * RE RTS BS4 BS3 BS2 BS1
538 * AGD = 1, R3 R2 R1 R0 = 0 1 0 1 => byte 4 = 1**10101 = 0x95 */
539 buf[2] = 0x95;
540
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300541/* Range(MHz) C1 * RE RTS BS4 BS3 BS2 BS1 Byte 5
542 * 47 - 153 0 * 0 0 0 0 0 1 0x01
543 * 153 - 430 0 * 0 0 0 0 1 0 0x02
544 * 430 - 822 0 * 0 0 1 0 0 0 0x08
545 * 822 - 862 1 * 0 0 1 0 0 0 0x88 */
Andrew de Quincey56e03142006-04-18 17:47:12 -0300546
547 if (fep->frequency <= 153000000) buf[3] = 0x01;
548 else if (fep->frequency <= 430000000) buf[3] = 0x02;
549 else if (fep->frequency <= 822000000) buf[3] = 0x08;
550 else buf[3] = 0x88;
551
Patrick Boettcherdea74862006-05-14 05:01:31 -0300552 if (fe->ops.i2c_gate_ctrl)
Antti Seppälä9d85d772006-12-20 11:10:35 -0300553 fe->ops.i2c_gate_ctrl(fe, 0);
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300554 deb_tuner("tuner buffer for %d Hz: %x %x %x %x\n", fep->frequency,
555 buf[0], buf[1], buf[2], buf[3]);
Patrick Boettcher6394cf52008-03-29 20:49:57 -0300556 ret = fc->i2c_request(&fc->fc_i2c_adap[2],
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300557 FC_WRITE, 0x61, buf[0], &buf[1], 3);
Andrew de Quincey56e03142006-04-18 17:47:12 -0300558 deb_tuner("tuner write returned: %d\n",ret);
Patrick Boettcher6394cf52008-03-29 20:49:57 -0300559 return ret;
Andrew de Quincey56e03142006-04-18 17:47:12 -0300560}
Andrew de Quinceydc27a162005-09-09 13:03:07 -0700561
562static u8 alps_tdee4_stv0297_inittab[] = {
563 0x80, 0x01,
564 0x80, 0x00,
565 0x81, 0x01,
566 0x81, 0x00,
Antti Seppälä9d85d772006-12-20 11:10:35 -0300567 0x00, 0x48,
568 0x01, 0x58,
Andrew de Quinceydc27a162005-09-09 13:03:07 -0700569 0x03, 0x00,
570 0x04, 0x00,
571 0x07, 0x00,
572 0x08, 0x00,
Andrew de Quinceydc27a162005-09-09 13:03:07 -0700573 0x30, 0xff,
Antti Seppälä9d85d772006-12-20 11:10:35 -0300574 0x31, 0x9d,
Andrew de Quinceydc27a162005-09-09 13:03:07 -0700575 0x32, 0xff,
576 0x33, 0x00,
Antti Seppälä9d85d772006-12-20 11:10:35 -0300577 0x34, 0x29,
578 0x35, 0x55,
579 0x36, 0x80,
580 0x37, 0x6e,
581 0x38, 0x9c,
582 0x40, 0x1a,
583 0x41, 0xfe,
584 0x42, 0x33,
Andrew de Quinceydc27a162005-09-09 13:03:07 -0700585 0x43, 0x00,
586 0x44, 0xff,
587 0x45, 0x00,
588 0x46, 0x00,
589 0x49, 0x04,
Antti Seppälä9d85d772006-12-20 11:10:35 -0300590 0x4a, 0x51,
Andrew de Quinceydc27a162005-09-09 13:03:07 -0700591 0x4b, 0xf8,
592 0x52, 0x30,
Antti Seppälä9d85d772006-12-20 11:10:35 -0300593 0x53, 0x06,
594 0x59, 0x06,
595 0x5a, 0x5e,
596 0x5b, 0x04,
597 0x61, 0x49,
598 0x62, 0x0a,
Andrew de Quinceydc27a162005-09-09 13:03:07 -0700599 0x70, 0xff,
Antti Seppälä9d85d772006-12-20 11:10:35 -0300600 0x71, 0x04,
Andrew de Quinceydc27a162005-09-09 13:03:07 -0700601 0x72, 0x00,
602 0x73, 0x00,
603 0x74, 0x0c,
Antti Seppälä9d85d772006-12-20 11:10:35 -0300604 0x80, 0x20,
Andrew de Quinceydc27a162005-09-09 13:03:07 -0700605 0x81, 0x00,
Antti Seppälä9d85d772006-12-20 11:10:35 -0300606 0x82, 0x30,
Andrew de Quinceydc27a162005-09-09 13:03:07 -0700607 0x83, 0x00,
608 0x84, 0x04,
Antti Seppälä9d85d772006-12-20 11:10:35 -0300609 0x85, 0x22,
610 0x86, 0x08,
611 0x87, 0x1b,
612 0x88, 0x00,
Andrew de Quinceydc27a162005-09-09 13:03:07 -0700613 0x89, 0x00,
Antti Seppälä9d85d772006-12-20 11:10:35 -0300614 0x90, 0x00,
615 0x91, 0x04,
616 0xa0, 0x86,
Andrew de Quinceydc27a162005-09-09 13:03:07 -0700617 0xa1, 0x00,
618 0xa2, 0x00,
619 0xb0, 0x91,
620 0xb1, 0x0b,
Antti Seppälä9d85d772006-12-20 11:10:35 -0300621 0xc0, 0x5b,
622 0xc1, 0x10,
Andrew de Quinceydc27a162005-09-09 13:03:07 -0700623 0xc2, 0x12,
Antti Seppälä9d85d772006-12-20 11:10:35 -0300624 0xd0, 0x02,
Andrew de Quinceydc27a162005-09-09 13:03:07 -0700625 0xd1, 0x00,
626 0xd2, 0x00,
627 0xd3, 0x00,
Antti Seppälä9d85d772006-12-20 11:10:35 -0300628 0xd4, 0x02,
Andrew de Quinceydc27a162005-09-09 13:03:07 -0700629 0xd5, 0x00,
630 0xde, 0x00,
Antti Seppälä9d85d772006-12-20 11:10:35 -0300631 0xdf, 0x01,
Andrew de Quinceydc27a162005-09-09 13:03:07 -0700632 0xff, 0xff,
633};
634
Johannes Stezenbach2add87a2005-05-16 21:54:10 -0700635static struct stv0297_config alps_tdee4_stv0297_config = {
636 .demod_address = 0x1c,
Andrew de Quinceydc27a162005-09-09 13:03:07 -0700637 .inittab = alps_tdee4_stv0297_inittab,
Johannes Stezenbach2add87a2005-05-16 21:54:10 -0700638};
639
Trent Piephoeccd15a2009-06-11 05:33:00 -0300640static int cablestar2_attach(struct flexcop_device *fc,
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300641 struct i2c_adapter *i2c)
Johannes Stezenbach2add87a2005-05-16 21:54:10 -0700642{
Antti Seppälä11c6c7f2008-12-01 06:59:37 -0300643 fc->fc_i2c_adap[0].no_base_addr = 1;
Patrick Boettcher6394cf52008-03-29 20:49:57 -0300644 fc->fe = dvb_attach(stv0297_attach, &alps_tdee4_stv0297_config, i2c);
Trent Piephoeccd15a2009-06-11 05:33:00 -0300645 if (!fc->fe) {
646 /* Reset for next frontend to try */
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300647 fc->fc_i2c_adap[0].no_base_addr = 0;
Trent Piephoeccd15a2009-06-11 05:33:00 -0300648 return 0;
649 }
650 fc->fe->ops.tuner_ops.set_params = alps_tdee4_stv0297_tuner_set_params;
651 return 1;
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300652}
Trent Piepho68b7f762009-06-11 19:31:22 -0300653#else
654#define cablestar2_attach NULL
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300655#endif
656
657static struct {
658 flexcop_device_type_t type;
Trent Piephoeccd15a2009-06-11 05:33:00 -0300659 int (*attach)(struct flexcop_device *, struct i2c_adapter *);
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300660} flexcop_frontends[] = {
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300661 { FC_SKY_REV27, skystar2_rev27_attach },
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300662 { FC_SKY_REV28, skystar2_rev28_attach },
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300663 { FC_SKY_REV26, skystar2_rev26_attach },
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300664 { FC_AIR_DVBT, airstar_dvbt_attach },
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300665 { FC_AIR_ATSC2, airstar_atsc2_attach },
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300666 { FC_AIR_ATSC3, airstar_atsc3_attach },
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300667 { FC_AIR_ATSC1, airstar_atsc1_attach },
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300668 { FC_CABLE, cablestar2_attach },
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300669 { FC_SKY_REV23, skystar2_rev23_attach },
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300670};
671
672/* try to figure out the frontend */
673int flexcop_frontend_init(struct flexcop_device *fc)
674{
675 int i;
676 for (i = 0; i < ARRAY_SIZE(flexcop_frontends); i++) {
Trent Piepho68b7f762009-06-11 19:31:22 -0300677 if (!flexcop_frontends[i].attach)
678 continue;
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300679 /* type needs to be set before, because of some workarounds
680 * done based on the probed card type */
681 fc->dev_type = flexcop_frontends[i].type;
Trent Piephoeccd15a2009-06-11 05:33:00 -0300682 if (flexcop_frontends[i].attach(fc, &fc->fc_i2c_adap[0].i2c_adap))
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300683 goto fe_found;
Trent Piephoeccd15a2009-06-11 05:33:00 -0300684 /* Clean up partially attached frontend */
685 if (fc->fe) {
686 dvb_frontend_detach(fc->fe);
687 fc->fe = NULL;
688 }
Patrick Boettcher6394cf52008-03-29 20:49:57 -0300689 }
Patrick Boettcherd66b94b2009-05-20 05:08:26 -0300690 fc->dev_type = FC_UNK;
Patrick Boettcher6394cf52008-03-29 20:49:57 -0300691 err("no frontend driver found for this B2C2/FlexCop adapter");
692 return -ENODEV;
693
694fe_found:
695 info("found '%s' .", fc->fe->ops.info.name);
696 if (dvb_register_frontend(&fc->dvb_adapter, fc->fe)) {
697 err("frontend registration failed!");
Trent Piephoeccd15a2009-06-11 05:33:00 -0300698 dvb_frontend_detach(fc->fe);
Patrick Boettcher6394cf52008-03-29 20:49:57 -0300699 fc->fe = NULL;
700 return -EINVAL;
Johannes Stezenbach2add87a2005-05-16 21:54:10 -0700701 }
702 fc->init_state |= FC_STATE_FE_INIT;
703 return 0;
704}
705
706void flexcop_frontend_exit(struct flexcop_device *fc)
707{
Andrew de Quincey2bfe0312006-08-08 09:10:08 -0300708 if (fc->init_state & FC_STATE_FE_INIT) {
Johannes Stezenbach2add87a2005-05-16 21:54:10 -0700709 dvb_unregister_frontend(fc->fe);
Andrew de Quinceyf52a8382006-08-08 09:10:09 -0300710 dvb_frontend_detach(fc->fe);
Andrew de Quincey2bfe0312006-08-08 09:10:08 -0300711 }
Johannes Stezenbach2add87a2005-05-16 21:54:10 -0700712 fc->init_state &= ~FC_STATE_FE_INIT;
713}