diff options
author | Steven Toth <stoth@hauppauge.com> | 2008-05-01 05:51:36 -0400 |
---|---|---|
committer | Mauro Carvalho Chehab <mchehab@infradead.org> | 2008-05-14 01:56:38 -0400 |
commit | 7f5c3affef2883f49e820db62413e1dff1d4cebb (patch) | |
tree | 3d01284b2e4dfbdbf2988aaf673b97e3b1eb7577 /drivers/media/common/tuners/mxl5005s.c | |
parent | 8c66a19d45fb5b88abf10678db353f35179a2cde (diff) |
V4L/DVB(7869): mxl5005s: Cleanup #6
Cleanup #6
Signed-off-by: Steven Toth <stoth@hauppauge.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@infradead.org>
Diffstat (limited to 'drivers/media/common/tuners/mxl5005s.c')
-rw-r--r-- | drivers/media/common/tuners/mxl5005s.c | 414 |
1 files changed, 225 insertions, 189 deletions
diff --git a/drivers/media/common/tuners/mxl5005s.c b/drivers/media/common/tuners/mxl5005s.c index aad88d5c0dc2..f7ed9a72db4a 100644 --- a/drivers/media/common/tuners/mxl5005s.c +++ b/drivers/media/common/tuners/mxl5005s.c | |||
@@ -1,26 +1,62 @@ | |||
1 | /* | 1 | /* |
2 | * For the Realtek RTL chip RTL2831U | 2 | MaxLinear MXL5005S VSB/QAM/DVBT tuner driver |
3 | * Realtek Release Date: 2008-03-14, ver 080314 | 3 | |
4 | * Realtek version RTL2831 Linux driver version 080314 | 4 | Copyright (C) 2008 MaxLinear |
5 | * ver 080314 | 5 | Copyright (C) 2006 Steven Toth <stoth@hauppauge.com> |
6 | * | 6 | Functions: |
7 | * for linux kernel version 2.6.21.4 - 2.6.22-14 | 7 | mxl5005s_reset() |
8 | * support MXL5005s and MT2060 tuners (support tuner auto-detecting) | 8 | mxl5005s_writereg() |
9 | * support two IR types -- RC5 and NEC | 9 | mxl5005s_writeregs() |
10 | * | 10 | mxl5005s_init() |
11 | * Known boards with Realtek RTL chip RTL2821U | 11 | mxl5005s_reconfigure() |
12 | * Freecom USB stick 14aa:0160 (version 4) | 12 | mxl5005s_AssignTunerMode() |
13 | * Conceptronic CTVDIGRCU | 13 | mxl5005s_set_params() |
14 | * | 14 | mxl5005s_get_frequency() |
15 | * Copyright (c) 2008 Realtek | 15 | mxl5005s_get_bandwidth() |
16 | * Copyright (c) 2008 Jan Hoogenraad, Barnaby Shearer, Andy Hasper | 16 | mxl5005s_release() |
17 | * This code is placed under the terms of the GNU General Public License | 17 | mxl5005s_attach() |
18 | * | 18 | |
19 | * Released by Realtek under GPLv2. | 19 | Copyright (c) 2008 Realtek |
20 | * Thanks to Realtek for a lot of support we received ! | 20 | Copyright (c) 2008 Jan Hoogenraad, Barnaby Shearer, Andy Hasper |
21 | * | 21 | Functions: |
22 | * Revision: 080314 - original version | 22 | mxl5005s_SetRfFreqHz() |
23 | */ | 23 | |
24 | This program is free software; you can redistribute it and/or modify | ||
25 | it under the terms of the GNU General Public License as published by | ||
26 | the Free Software Foundation; either version 2 of the License, or | ||
27 | (at your option) any later version. | ||
28 | |||
29 | This program is distributed in the hope that it will be useful, | ||
30 | but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
31 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
32 | GNU General Public License for more details. | ||
33 | |||
34 | You should have received a copy of the GNU General Public License | ||
35 | along with this program; if not, write to the Free Software | ||
36 | Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. | ||
37 | |||
38 | */ | ||
39 | |||
40 | /* | ||
41 | History of this driver (Steven Toth): | ||
42 | I was given a public release of a linux driver that included | ||
43 | support for the MaxLinear MXL5005S silicon tuner. Analysis of | ||
44 | the tuner driver showed clearly three things. | ||
45 | |||
46 | 1. The tuner driver didn't support the LinuxTV tuner API | ||
47 | so the code Realtek added had to be removed. | ||
48 | |||
49 | 2. A significant amount of the driver is reference driver code | ||
50 | from MaxLinear, I felt it was important to identify and | ||
51 | preserve this. | ||
52 | |||
53 | 3. New code has to be added to interface correctly with the | ||
54 | LinuxTV API, as a regular kernel module. | ||
55 | |||
56 | Other than the reference driver enum's, I've clearly marked | ||
57 | sections of the code and retained the copyright of the | ||
58 | respective owners. | ||
59 | */ | ||
24 | 60 | ||
25 | #include "mxl5005s.h" | 61 | #include "mxl5005s.h" |
26 | 62 | ||
@@ -250,9 +286,12 @@ struct mxl5005s_state | |||
250 | struct mxl5005s_config *config; | 286 | struct mxl5005s_config *config; |
251 | struct dvb_frontend *frontend; | 287 | struct dvb_frontend *frontend; |
252 | struct i2c_adapter *i2c; | 288 | struct i2c_adapter *i2c; |
289 | |||
290 | /* Cache values */ | ||
291 | u32 current_mode; | ||
292 | |||
253 | }; | 293 | }; |
254 | 294 | ||
255 | // funcs | ||
256 | u16 MXL_ControlWrite(struct dvb_frontend *fe, u16 ControlNum, u32 value); | 295 | u16 MXL_ControlWrite(struct dvb_frontend *fe, u16 ControlNum, u32 value); |
257 | u16 MXL_ControlRead(struct dvb_frontend *fe, u16 controlNum, u32 *value); | 296 | u16 MXL_ControlRead(struct dvb_frontend *fe, u16 controlNum, u32 *value); |
258 | u16 MXL_GetMasterControl(u8 *MasterReg, int state); | 297 | u16 MXL_GetMasterControl(u8 *MasterReg, int state); |
@@ -269,9 +308,22 @@ u16 MXL_TuneRF(struct dvb_frontend *fe, u32 RF_Freq); | |||
269 | void MXL_SynthIFLO_Calc(struct dvb_frontend *fe); | 308 | void MXL_SynthIFLO_Calc(struct dvb_frontend *fe); |
270 | void MXL_SynthRFTGLO_Calc(struct dvb_frontend *fe); | 309 | void MXL_SynthRFTGLO_Calc(struct dvb_frontend *fe); |
271 | u16 MXL_GetCHRegister_ZeroIF(struct dvb_frontend *fe, u8 *RegNum, u8 *RegVal, int *count); | 310 | u16 MXL_GetCHRegister_ZeroIF(struct dvb_frontend *fe, u8 *RegNum, u8 *RegVal, int *count); |
272 | int mxl5005s_SetRegsWithTable(struct dvb_frontend *fe, u8 *pAddrTable, u8 *pByteTable, int TableLen); | 311 | int mxl5005s_writeregs(struct dvb_frontend *fe, u8 *addrtable, u8 *datatable, u8 len); |
273 | u16 MXL_IFSynthInit(struct dvb_frontend *fe); | 312 | u16 MXL_IFSynthInit(struct dvb_frontend *fe); |
274 | int mxl5005s_AssignTunerMode(struct dvb_frontend *fe); | 313 | int mxl5005s_AssignTunerMode(struct dvb_frontend *fe, u32 mod_type, u32 bandwidth); |
314 | int mxl5005s_reconfigure(struct dvb_frontend *fe, u32 mod_type, u32 bandwidth); | ||
315 | |||
316 | /* ---------------------------------------------------------------- | ||
317 | * Begin: Custom code salvaged from the Realtek driver. | ||
318 | * Copyright (c) 2008 Realtek | ||
319 | * Copyright (c) 2008 Jan Hoogenraad, Barnaby Shearer, Andy Hasper | ||
320 | * This code is placed under the terms of the GNU General Public License | ||
321 | * | ||
322 | * Released by Realtek under GPLv2. | ||
323 | * Thanks to Realtek for a lot of support we received ! | ||
324 | * | ||
325 | * Revision: 080314 - original version | ||
326 | */ | ||
275 | 327 | ||
276 | int mxl5005s_SetRfFreqHz(struct dvb_frontend *fe, unsigned long RfFreqHz) | 328 | int mxl5005s_SetRfFreqHz(struct dvb_frontend *fe, unsigned long RfFreqHz) |
277 | { | 329 | { |
@@ -292,7 +344,7 @@ int mxl5005s_SetRfFreqHz(struct dvb_frontend *fe, unsigned long RfFreqHz) | |||
292 | AddrTable[0] = MASTER_CONTROL_ADDR; | 344 | AddrTable[0] = MASTER_CONTROL_ADDR; |
293 | ByteTable[0] |= state->config->AgcMasterByte; | 345 | ByteTable[0] |= state->config->AgcMasterByte; |
294 | 346 | ||
295 | mxl5005s_SetRegsWithTable(fe, AddrTable, ByteTable, 1); | 347 | mxl5005s_writeregs(fe, AddrTable, ByteTable, 1); |
296 | 348 | ||
297 | // Tuner RF frequency setting stage 1 | 349 | // Tuner RF frequency setting stage 1 |
298 | MXL_TuneRF(fe, RfFreqHz); | 350 | MXL_TuneRF(fe, RfFreqHz); |
@@ -309,7 +361,7 @@ int mxl5005s_SetRfFreqHz(struct dvb_frontend *fe, unsigned long RfFreqHz) | |||
309 | ByteTable[TableLen] = MasterControlByte | state->config->AgcMasterByte; | 361 | ByteTable[TableLen] = MasterControlByte | state->config->AgcMasterByte; |
310 | TableLen += 1; | 362 | TableLen += 1; |
311 | 363 | ||
312 | mxl5005s_SetRegsWithTable(fe, AddrTable, ByteTable, TableLen); | 364 | mxl5005s_writeregs(fe, AddrTable, ByteTable, TableLen); |
313 | 365 | ||
314 | // Wait 30 ms. | 366 | // Wait 30 ms. |
315 | msleep(150); | 367 | msleep(150); |
@@ -324,118 +376,18 @@ int mxl5005s_SetRfFreqHz(struct dvb_frontend *fe, unsigned long RfFreqHz) | |||
324 | ByteTable[TableLen] = MasterControlByte | state->config->AgcMasterByte ; | 376 | ByteTable[TableLen] = MasterControlByte | state->config->AgcMasterByte ; |
325 | TableLen += 1; | 377 | TableLen += 1; |
326 | 378 | ||
327 | mxl5005s_SetRegsWithTable(fe, AddrTable, ByteTable, TableLen); | 379 | mxl5005s_writeregs(fe, AddrTable, ByteTable, TableLen); |
328 | 380 | ||
329 | msleep(100); | 381 | msleep(100); |
330 | 382 | ||
331 | return 0; | 383 | return 0; |
332 | } | 384 | } |
385 | /* End: Custom code taken from the Realtek driver */ | ||
333 | 386 | ||
334 | static int mxl5005s_reset(struct dvb_frontend *fe) | 387 | /* ---------------------------------------------------------------- |
335 | { | 388 | * Begin: Reference driver code found in the Realtek driver. |
336 | struct mxl5005s_state *state = fe->tuner_priv; | 389 | * Copyright (c) 2008 MaxLinear |
337 | int ret = 0; | 390 | */ |
338 | |||
339 | u8 buf[2] = { 0xff, 0x00 }; | ||
340 | struct i2c_msg msg = { .addr = state->config->i2c_address, .flags = 0, | ||
341 | .buf = buf, .len = 2 }; | ||
342 | |||
343 | dprintk(2, "%s()\n", __func__); | ||
344 | |||
345 | if (fe->ops.i2c_gate_ctrl) | ||
346 | fe->ops.i2c_gate_ctrl(fe, 1); | ||
347 | |||
348 | if (i2c_transfer(state->i2c, &msg, 1) != 1) { | ||
349 | printk(KERN_WARNING "mxl5005s I2C reset failed\n"); | ||
350 | ret = -EREMOTEIO; | ||
351 | } | ||
352 | |||
353 | if (fe->ops.i2c_gate_ctrl) | ||
354 | fe->ops.i2c_gate_ctrl(fe, 0); | ||
355 | |||
356 | return ret; | ||
357 | } | ||
358 | |||
359 | /* Write a single byte to a single reg */ | ||
360 | static int mxl5005s_writereg(struct dvb_frontend *fe, u8 reg, u8 val, int latch) | ||
361 | { | ||
362 | struct mxl5005s_state *state = fe->tuner_priv; | ||
363 | u8 buf[3] = { reg, val, MXL5005S_LATCH_BYTE }; | ||
364 | struct i2c_msg msg = { .addr = state->config->i2c_address, .flags = 0, | ||
365 | .buf = buf, .len = 3 }; | ||
366 | |||
367 | if(latch == 0) | ||
368 | msg.len = 2; | ||
369 | |||
370 | dprintk(2, "%s(reg = 0x%x val = 0x%x addr = 0x%x)\n", __func__, reg, val, msg.addr); | ||
371 | |||
372 | if (i2c_transfer(state->i2c, &msg, 1) != 1) { | ||
373 | printk(KERN_WARNING "mxl5005s I2C write failed\n"); | ||
374 | return -EREMOTEIO; | ||
375 | } | ||
376 | return 0; | ||
377 | } | ||
378 | |||
379 | int mxl5005s_SetRegsWithTable(struct dvb_frontend *fe, u8 *pAddrTable, u8 *pByteTable, int TableLen) | ||
380 | { | ||
381 | int i, ret = 0; | ||
382 | |||
383 | if (fe->ops.i2c_gate_ctrl) | ||
384 | fe->ops.i2c_gate_ctrl(fe, 1); | ||
385 | |||
386 | for( i = 0 ; i < TableLen - 1 ; i++) | ||
387 | { | ||
388 | ret = mxl5005s_writereg(fe, pAddrTable[i], pByteTable[i], 0); | ||
389 | if (ret < 0) | ||
390 | break; | ||
391 | } | ||
392 | |||
393 | ret = mxl5005s_writereg(fe, pAddrTable[i], pByteTable[i], 1); | ||
394 | |||
395 | if (fe->ops.i2c_gate_ctrl) | ||
396 | fe->ops.i2c_gate_ctrl(fe, 0); | ||
397 | |||
398 | return ret; | ||
399 | } | ||
400 | |||
401 | int mxl5005s_SetRegMaskBits(struct dvb_frontend *fe, | ||
402 | unsigned char RegAddr, | ||
403 | unsigned char Msb, | ||
404 | unsigned char Lsb, | ||
405 | const unsigned char WritingValue | ||
406 | ) | ||
407 | { | ||
408 | int i; | ||
409 | |||
410 | unsigned char Mask; | ||
411 | unsigned char Shift; | ||
412 | unsigned char RegByte; | ||
413 | |||
414 | /* Generate mask and shift according to MSB and LSB. */ | ||
415 | Mask = 0; | ||
416 | for(i = Lsb; i < (unsigned char)(Msb + 1); i++) | ||
417 | Mask |= 0x1 << i; | ||
418 | |||
419 | Shift = Lsb; | ||
420 | |||
421 | /* Get tuner register byte according to register adddress. */ | ||
422 | MXL_RegRead(fe, RegAddr, &RegByte); | ||
423 | |||
424 | /* Reserve register byte unmask bit with mask and inlay writing value into it. */ | ||
425 | RegByte &= ~Mask; | ||
426 | RegByte |= (WritingValue << Shift) & Mask; | ||
427 | |||
428 | /* Update tuner register byte table. */ | ||
429 | MXL_RegWrite(fe, RegAddr, RegByte); | ||
430 | |||
431 | /* Write tuner register byte with writing byte. */ | ||
432 | return mxl5005s_SetRegsWithTable(fe, &RegAddr, &RegByte, 1); | ||
433 | } | ||
434 | |||
435 | |||
436 | // The following context is source code provided by MaxLinear. | ||
437 | // MaxLinear source code - MXL5005_Initialize.cpp | ||
438 | // DONE | ||
439 | u16 MXL5005_RegisterInit(struct dvb_frontend *fe) | 391 | u16 MXL5005_RegisterInit(struct dvb_frontend *fe) |
440 | { | 392 | { |
441 | struct mxl5005s_state *state = fe->tuner_priv; | 393 | struct mxl5005s_state *state = fe->tuner_priv; |
@@ -757,7 +709,6 @@ u16 MXL5005_RegisterInit(struct dvb_frontend *fe) | |||
757 | return 0 ; | 709 | return 0 ; |
758 | } | 710 | } |
759 | 711 | ||
760 | // DONE | ||
761 | u16 MXL5005_ControlInit(struct dvb_frontend *fe) | 712 | u16 MXL5005_ControlInit(struct dvb_frontend *fe) |
762 | { | 713 | { |
763 | struct mxl5005s_state *state = fe->tuner_priv; | 714 | struct mxl5005s_state *state = fe->tuner_priv; |
@@ -1701,7 +1652,6 @@ u16 MXL5005_ControlInit(struct dvb_frontend *fe) | |||
1701 | // MaxLinear source code - MXL5005_c.cpp | 1652 | // MaxLinear source code - MXL5005_c.cpp |
1702 | // MXL5005.cpp : Defines the initialization routines for the DLL. | 1653 | // MXL5005.cpp : Defines the initialization routines for the DLL. |
1703 | // 2.6.12 | 1654 | // 2.6.12 |
1704 | // DONE | ||
1705 | void InitTunerControls(struct dvb_frontend *fe) | 1655 | void InitTunerControls(struct dvb_frontend *fe) |
1706 | { | 1656 | { |
1707 | MXL5005_RegisterInit(fe); | 1657 | MXL5005_RegisterInit(fe); |
@@ -1744,7 +1694,6 @@ void InitTunerControls(struct dvb_frontend *fe) | |||
1744 | // > 0 : Failed // | 1694 | // > 0 : Failed // |
1745 | // // | 1695 | // // |
1746 | /////////////////////////////////////////////////////////////////////////////// | 1696 | /////////////////////////////////////////////////////////////////////////////// |
1747 | // DONE | ||
1748 | u16 MXL5005_TunerConfig(struct dvb_frontend *fe, | 1697 | u16 MXL5005_TunerConfig(struct dvb_frontend *fe, |
1749 | u8 Mode, /* 0: Analog Mode ; 1: Digital Mode */ | 1698 | u8 Mode, /* 0: Analog Mode ; 1: Digital Mode */ |
1750 | u8 IF_mode, /* for Analog Mode, 0: zero IF; 1: low IF */ | 1699 | u8 IF_mode, /* for Analog Mode, 0: zero IF; 1: low IF */ |
@@ -1814,7 +1763,6 @@ u16 MXL5005_TunerConfig(struct dvb_frontend *fe, | |||
1814 | // > 0 : Failed // | 1763 | // > 0 : Failed // |
1815 | // // | 1764 | // // |
1816 | /////////////////////////////////////////////////////////////////////////////// | 1765 | /////////////////////////////////////////////////////////////////////////////// |
1817 | // DONE | ||
1818 | void MXL_SynthIFLO_Calc(struct dvb_frontend *fe) | 1766 | void MXL_SynthIFLO_Calc(struct dvb_frontend *fe) |
1819 | { | 1767 | { |
1820 | struct mxl5005s_state *state = fe->tuner_priv; | 1768 | struct mxl5005s_state *state = fe->tuner_priv; |
@@ -1853,7 +1801,6 @@ void MXL_SynthIFLO_Calc(struct dvb_frontend *fe) | |||
1853 | // > 0 : Failed // | 1801 | // > 0 : Failed // |
1854 | // // | 1802 | // // |
1855 | /////////////////////////////////////////////////////////////////////////////// | 1803 | /////////////////////////////////////////////////////////////////////////////// |
1856 | // DONE | ||
1857 | void MXL_SynthRFTGLO_Calc(struct dvb_frontend *fe) | 1804 | void MXL_SynthRFTGLO_Calc(struct dvb_frontend *fe) |
1858 | { | 1805 | { |
1859 | struct mxl5005s_state *state = fe->tuner_priv; | 1806 | struct mxl5005s_state *state = fe->tuner_priv; |
@@ -1892,7 +1839,6 @@ void MXL_SynthRFTGLO_Calc(struct dvb_frontend *fe) | |||
1892 | // > 0 : Failed // | 1839 | // > 0 : Failed // |
1893 | // // | 1840 | // // |
1894 | /////////////////////////////////////////////////////////////////////////////// | 1841 | /////////////////////////////////////////////////////////////////////////////// |
1895 | // DONE | ||
1896 | u16 MXL_OverwriteICDefault(struct dvb_frontend *fe) | 1842 | u16 MXL_OverwriteICDefault(struct dvb_frontend *fe) |
1897 | { | 1843 | { |
1898 | u16 status = 0; | 1844 | u16 status = 0; |
@@ -1930,7 +1876,6 @@ u16 MXL_OverwriteICDefault(struct dvb_frontend *fe) | |||
1930 | // > 0 : Failed // | 1876 | // > 0 : Failed // |
1931 | // // | 1877 | // // |
1932 | /////////////////////////////////////////////////////////////////////////////// | 1878 | /////////////////////////////////////////////////////////////////////////////// |
1933 | // DONE | ||
1934 | u16 MXL_BlockInit(struct dvb_frontend *fe) | 1879 | u16 MXL_BlockInit(struct dvb_frontend *fe) |
1935 | { | 1880 | { |
1936 | struct mxl5005s_state *state = fe->tuner_priv; | 1881 | struct mxl5005s_state *state = fe->tuner_priv; |
@@ -3687,7 +3632,6 @@ u16 MXL_TuneRF(struct dvb_frontend *fe, u32 RF_Freq) | |||
3687 | return status ; | 3632 | return status ; |
3688 | } | 3633 | } |
3689 | 3634 | ||
3690 | // DONE | ||
3691 | u16 MXL_SetGPIO(struct dvb_frontend *fe, u8 GPIO_Num, u8 GPIO_Val) | 3635 | u16 MXL_SetGPIO(struct dvb_frontend *fe, u8 GPIO_Num, u8 GPIO_Val) |
3692 | { | 3636 | { |
3693 | u16 status = 0; | 3637 | u16 status = 0; |
@@ -3754,7 +3698,6 @@ u16 MXL_SetGPIO(struct dvb_frontend *fe, u8 GPIO_Num, u8 GPIO_Val) | |||
3754 | // >0 : Value exceed maximum allowed for control number // | 3698 | // >0 : Value exceed maximum allowed for control number // |
3755 | // // | 3699 | // // |
3756 | /////////////////////////////////////////////////////////////////////////////// | 3700 | /////////////////////////////////////////////////////////////////////////////// |
3757 | // DONE | ||
3758 | u16 MXL_ControlWrite(struct dvb_frontend *fe, u16 ControlNum, u32 value) | 3701 | u16 MXL_ControlWrite(struct dvb_frontend *fe, u16 ControlNum, u32 value) |
3759 | { | 3702 | { |
3760 | u16 status = 0; | 3703 | u16 status = 0; |
@@ -3795,7 +3738,6 @@ u16 MXL_ControlWrite(struct dvb_frontend *fe, u16 ControlNum, u32 value) | |||
3795 | // 2 : Control name not found // | 3738 | // 2 : Control name not found // |
3796 | // // | 3739 | // // |
3797 | /////////////////////////////////////////////////////////////////////////////// | 3740 | /////////////////////////////////////////////////////////////////////////////// |
3798 | // DONE | ||
3799 | u16 MXL_ControlWrite_Group(struct dvb_frontend *fe, u16 controlNum, u32 value, u16 controlGroup) | 3741 | u16 MXL_ControlWrite_Group(struct dvb_frontend *fe, u16 controlNum, u32 value, u16 controlGroup) |
3800 | { | 3742 | { |
3801 | struct mxl5005s_state *state = fe->tuner_priv; | 3743 | struct mxl5005s_state *state = fe->tuner_priv; |
@@ -3902,7 +3844,6 @@ u16 MXL_ControlWrite_Group(struct dvb_frontend *fe, u16 controlNum, u32 value, u | |||
3902 | // -1 : Invalid Register Address // | 3844 | // -1 : Invalid Register Address // |
3903 | // // | 3845 | // // |
3904 | /////////////////////////////////////////////////////////////////////////////// | 3846 | /////////////////////////////////////////////////////////////////////////////// |
3905 | // DONE | ||
3906 | u16 MXL_RegWrite(struct dvb_frontend *fe, u8 RegNum, u8 RegVal) | 3847 | u16 MXL_RegWrite(struct dvb_frontend *fe, u8 RegNum, u8 RegVal) |
3907 | { | 3848 | { |
3908 | struct mxl5005s_state *state = fe->tuner_priv; | 3849 | struct mxl5005s_state *state = fe->tuner_priv; |
@@ -3942,7 +3883,6 @@ u16 MXL_RegWrite(struct dvb_frontend *fe, u8 RegNum, u8 RegVal) | |||
3942 | // -1 : Invalid Register Address // | 3883 | // -1 : Invalid Register Address // |
3943 | // // | 3884 | // // |
3944 | /////////////////////////////////////////////////////////////////////////////// | 3885 | /////////////////////////////////////////////////////////////////////////////// |
3945 | // DONE | ||
3946 | u16 MXL_RegRead(struct dvb_frontend *fe, u8 RegNum, u8 *RegVal) | 3886 | u16 MXL_RegRead(struct dvb_frontend *fe, u8 RegNum, u8 *RegVal) |
3947 | { | 3887 | { |
3948 | struct mxl5005s_state *state = fe->tuner_priv; | 3888 | struct mxl5005s_state *state = fe->tuner_priv; |
@@ -3979,7 +3919,6 @@ u16 MXL_RegRead(struct dvb_frontend *fe, u8 RegNum, u8 *RegVal) | |||
3979 | // -1 : Invalid control name // | 3919 | // -1 : Invalid control name // |
3980 | // // | 3920 | // // |
3981 | /////////////////////////////////////////////////////////////////////////////// | 3921 | /////////////////////////////////////////////////////////////////////////////// |
3982 | // DONE | ||
3983 | u16 MXL_ControlRead(struct dvb_frontend *fe, u16 controlNum, u32 *value) | 3922 | u16 MXL_ControlRead(struct dvb_frontend *fe, u16 controlNum, u32 *value) |
3984 | { | 3923 | { |
3985 | struct mxl5005s_state *state = fe->tuner_priv; | 3924 | struct mxl5005s_state *state = fe->tuner_priv; |
@@ -4051,7 +3990,6 @@ u16 MXL_ControlRead(struct dvb_frontend *fe, u16 controlNum, u32 *value) | |||
4051 | // -1 : Invalid control name // | 3990 | // -1 : Invalid control name // |
4052 | // // | 3991 | // // |
4053 | /////////////////////////////////////////////////////////////////////////////// | 3992 | /////////////////////////////////////////////////////////////////////////////// |
4054 | // DONE | ||
4055 | u16 MXL_ControlRegRead(struct dvb_frontend *fe, u16 controlNum, u8 *RegNum, int * count) | 3993 | u16 MXL_ControlRegRead(struct dvb_frontend *fe, u16 controlNum, u8 *RegNum, int * count) |
4056 | { | 3994 | { |
4057 | struct mxl5005s_state *state = fe->tuner_priv; | 3995 | struct mxl5005s_state *state = fe->tuner_priv; |
@@ -4157,7 +4095,6 @@ u16 MXL_ControlRegRead(struct dvb_frontend *fe, u16 controlNum, u8 *RegNum, int | |||
4157 | // NONE // | 4095 | // NONE // |
4158 | // // | 4096 | // // |
4159 | /////////////////////////////////////////////////////////////////////////////// | 4097 | /////////////////////////////////////////////////////////////////////////////// |
4160 | // DONE | ||
4161 | void MXL_RegWriteBit(struct dvb_frontend *fe, u8 address, u8 bit, u8 bitVal) | 4098 | void MXL_RegWriteBit(struct dvb_frontend *fe, u8 address, u8 bit, u8 bitVal) |
4162 | { | 4099 | { |
4163 | struct mxl5005s_state *state = fe->tuner_priv; | 4100 | struct mxl5005s_state *state = fe->tuner_priv; |
@@ -4205,7 +4142,6 @@ void MXL_RegWriteBit(struct dvb_frontend *fe, u8 address, u8 bit, u8 bitVal) | |||
4205 | // Computed value // | 4142 | // Computed value // |
4206 | // // | 4143 | // // |
4207 | /////////////////////////////////////////////////////////////////////////////// | 4144 | /////////////////////////////////////////////////////////////////////////////// |
4208 | // DONE | ||
4209 | u32 MXL_Ceiling(u32 value, u32 resolution) | 4145 | u32 MXL_Ceiling(u32 value, u32 resolution) |
4210 | { | 4146 | { |
4211 | return (value/resolution + (value % resolution > 0 ? 1 : 0)); | 4147 | return (value/resolution + (value % resolution > 0 ? 1 : 0)); |
@@ -4214,7 +4150,6 @@ u32 MXL_Ceiling(u32 value, u32 resolution) | |||
4214 | // | 4150 | // |
4215 | // Retrieve the Initialzation Registers | 4151 | // Retrieve the Initialzation Registers |
4216 | // | 4152 | // |
4217 | // DONE | ||
4218 | u16 MXL_GetInitRegister(struct dvb_frontend *fe, u8 * RegNum, u8 *RegVal, int *count) | 4153 | u16 MXL_GetInitRegister(struct dvb_frontend *fe, u8 * RegNum, u8 *RegVal, int *count) |
4219 | { | 4154 | { |
4220 | u16 status = 0; | 4155 | u16 status = 0; |
@@ -4237,7 +4172,6 @@ u16 MXL_GetInitRegister(struct dvb_frontend *fe, u8 * RegNum, u8 *RegVal, int *c | |||
4237 | return status; | 4172 | return status; |
4238 | } | 4173 | } |
4239 | 4174 | ||
4240 | // DONE | ||
4241 | u16 MXL_GetCHRegister(struct dvb_frontend *fe, u8 * RegNum, u8 *RegVal, int *count) | 4175 | u16 MXL_GetCHRegister(struct dvb_frontend *fe, u8 * RegNum, u8 *RegVal, int *count) |
4242 | { | 4176 | { |
4243 | u16 status = 0; | 4177 | u16 status = 0; |
@@ -4265,7 +4199,6 @@ u16 MXL_GetCHRegister(struct dvb_frontend *fe, u8 * RegNum, u8 *RegVal, int *cou | |||
4265 | return status; | 4199 | return status; |
4266 | } | 4200 | } |
4267 | 4201 | ||
4268 | // DONE | ||
4269 | u16 MXL_GetCHRegister_ZeroIF(struct dvb_frontend *fe, u8 * RegNum, u8 *RegVal, int *count) | 4202 | u16 MXL_GetCHRegister_ZeroIF(struct dvb_frontend *fe, u8 * RegNum, u8 *RegVal, int *count) |
4270 | { | 4203 | { |
4271 | u16 status = 0; | 4204 | u16 status = 0; |
@@ -4283,7 +4216,6 @@ u16 MXL_GetCHRegister_ZeroIF(struct dvb_frontend *fe, u8 * RegNum, u8 *RegVal, i | |||
4283 | return status; | 4216 | return status; |
4284 | } | 4217 | } |
4285 | 4218 | ||
4286 | // DONE | ||
4287 | u16 MXL_GetCHRegister_LowIF(struct dvb_frontend *fe, u8 * RegNum, u8 *RegVal, int *count) | 4219 | u16 MXL_GetCHRegister_LowIF(struct dvb_frontend *fe, u8 * RegNum, u8 *RegVal, int *count) |
4288 | { | 4220 | { |
4289 | u16 status = 0; | 4221 | u16 status = 0; |
@@ -4301,7 +4233,6 @@ u16 MXL_GetCHRegister_LowIF(struct dvb_frontend *fe, u8 * RegNum, u8 *RegVal, in | |||
4301 | return status; | 4233 | return status; |
4302 | } | 4234 | } |
4303 | 4235 | ||
4304 | // DONE | ||
4305 | u16 MXL_GetMasterControl(u8 *MasterReg, int state) | 4236 | u16 MXL_GetMasterControl(u8 *MasterReg, int state) |
4306 | { | 4237 | { |
4307 | if (state == 1) /* Load_Start */ | 4238 | if (state == 1) /* Load_Start */ |
@@ -4446,7 +4377,6 @@ u16 MXL_VCORange_Test(struct dvb_frontend *fe, int VCO_Range) | |||
4446 | return status; | 4377 | return status; |
4447 | } | 4378 | } |
4448 | 4379 | ||
4449 | // DONE | ||
4450 | u16 MXL_Hystersis_Test(struct dvb_frontend *fe, int Hystersis) | 4380 | u16 MXL_Hystersis_Test(struct dvb_frontend *fe, int Hystersis) |
4451 | { | 4381 | { |
4452 | struct mxl5005s_state *state = fe->tuner_priv; | 4382 | struct mxl5005s_state *state = fe->tuner_priv; |
@@ -4457,20 +4387,99 @@ u16 MXL_Hystersis_Test(struct dvb_frontend *fe, int Hystersis) | |||
4457 | 4387 | ||
4458 | return status; | 4388 | return status; |
4459 | } | 4389 | } |
4460 | |||
4461 | #endif | 4390 | #endif |
4391 | /* End: Reference driver code found in the Realtek driver that | ||
4392 | * is copyright MaxLinear */ | ||
4393 | |||
4394 | /* ---------------------------------------------------------------- | ||
4395 | * Begin: Everything after here is new code to adapt the | ||
4396 | * proprietary Realtek driver into a Linux API tuner. | ||
4397 | * Copyright (C) 2008 Steven Toth <stoth@hauppauge.com> | ||
4398 | */ | ||
4399 | static int mxl5005s_reset(struct dvb_frontend *fe) | ||
4400 | { | ||
4401 | struct mxl5005s_state *state = fe->tuner_priv; | ||
4402 | int ret = 0; | ||
4403 | |||
4404 | u8 buf[2] = { 0xff, 0x00 }; | ||
4405 | struct i2c_msg msg = { .addr = state->config->i2c_address, .flags = 0, | ||
4406 | .buf = buf, .len = 2 }; | ||
4407 | |||
4408 | dprintk(2, "%s()\n", __func__); | ||
4409 | |||
4410 | if (fe->ops.i2c_gate_ctrl) | ||
4411 | fe->ops.i2c_gate_ctrl(fe, 1); | ||
4412 | |||
4413 | if (i2c_transfer(state->i2c, &msg, 1) != 1) { | ||
4414 | printk(KERN_WARNING "mxl5005s I2C reset failed\n"); | ||
4415 | ret = -EREMOTEIO; | ||
4416 | } | ||
4417 | |||
4418 | if (fe->ops.i2c_gate_ctrl) | ||
4419 | fe->ops.i2c_gate_ctrl(fe, 0); | ||
4420 | |||
4421 | return ret; | ||
4422 | } | ||
4423 | |||
4424 | /* Write a single byte to a single reg, latch the value if required by | ||
4425 | * following the transaction with the latch byte. | ||
4426 | */ | ||
4427 | static int mxl5005s_writereg(struct dvb_frontend *fe, u8 reg, u8 val, int latch) | ||
4428 | { | ||
4429 | struct mxl5005s_state *state = fe->tuner_priv; | ||
4430 | u8 buf[3] = { reg, val, MXL5005S_LATCH_BYTE }; | ||
4431 | struct i2c_msg msg = { .addr = state->config->i2c_address, .flags = 0, | ||
4432 | .buf = buf, .len = 3 }; | ||
4433 | |||
4434 | if (latch == 0) | ||
4435 | msg.len = 2; | ||
4436 | |||
4437 | dprintk(2, "%s(reg = 0x%x val = 0x%x addr = 0x%x)\n", __func__, reg, val, msg.addr); | ||
4438 | |||
4439 | if (i2c_transfer(state->i2c, &msg, 1) != 1) { | ||
4440 | printk(KERN_WARNING "mxl5005s I2C write failed\n"); | ||
4441 | return -EREMOTEIO; | ||
4442 | } | ||
4443 | return 0; | ||
4444 | } | ||
4445 | |||
4446 | int mxl5005s_writeregs(struct dvb_frontend *fe, u8 *addrtable, u8 *datatable, u8 len) | ||
4447 | { | ||
4448 | int ret = 0, i; | ||
4449 | |||
4450 | if (fe->ops.i2c_gate_ctrl) | ||
4451 | fe->ops.i2c_gate_ctrl(fe, 1); | ||
4452 | |||
4453 | for (i = 0 ; i < len-1; i++) { | ||
4454 | ret = mxl5005s_writereg(fe, addrtable[i], datatable[i], 0); | ||
4455 | if (ret < 0) | ||
4456 | break; | ||
4457 | } | ||
4458 | |||
4459 | ret = mxl5005s_writereg(fe, addrtable[i], datatable[i], 1); | ||
4460 | |||
4461 | if (fe->ops.i2c_gate_ctrl) | ||
4462 | fe->ops.i2c_gate_ctrl(fe, 0); | ||
4463 | |||
4464 | return ret; | ||
4465 | } | ||
4462 | 4466 | ||
4463 | /* Linux driver related functions */ | ||
4464 | 4467 | ||
4465 | int mxl5005s_init(struct dvb_frontend *fe) | 4468 | int mxl5005s_init(struct dvb_frontend *fe) |
4466 | { | 4469 | { |
4470 | dprintk(1, "%s()\n", __func__); | ||
4471 | return mxl5005s_reconfigure(fe, MXL_QAM, MXL5005S_BANDWIDTH_6MHZ); | ||
4472 | } | ||
4473 | |||
4474 | int mxl5005s_reconfigure(struct dvb_frontend *fe, u32 mod_type, u32 bandwidth) | ||
4475 | { | ||
4467 | struct mxl5005s_state *state = fe->tuner_priv; | 4476 | struct mxl5005s_state *state = fe->tuner_priv; |
4468 | 4477 | ||
4469 | u8 AddrTable[MXL5005S_REG_WRITING_TABLE_LEN_MAX]; | 4478 | u8 AddrTable[MXL5005S_REG_WRITING_TABLE_LEN_MAX]; |
4470 | u8 ByteTable[MXL5005S_REG_WRITING_TABLE_LEN_MAX]; | 4479 | u8 ByteTable[MXL5005S_REG_WRITING_TABLE_LEN_MAX]; |
4471 | int TableLen; | 4480 | int TableLen; |
4472 | 4481 | ||
4473 | dprintk(1, "%s()\n", __func__); | 4482 | dprintk(1, "%s(type=%d, bw=%d)\n", __func__, mod_type, bandwidth); |
4474 | 4483 | ||
4475 | mxl5005s_reset(fe); | 4484 | mxl5005s_reset(fe); |
4476 | 4485 | ||
@@ -4479,19 +4488,19 @@ int mxl5005s_init(struct dvb_frontend *fe) | |||
4479 | AddrTable[0] = MASTER_CONTROL_ADDR; | 4488 | AddrTable[0] = MASTER_CONTROL_ADDR; |
4480 | ByteTable[0] |= state->config->AgcMasterByte; | 4489 | ByteTable[0] |= state->config->AgcMasterByte; |
4481 | 4490 | ||
4482 | mxl5005s_SetRegsWithTable(fe, AddrTable, ByteTable, 1); | 4491 | mxl5005s_writeregs(fe, AddrTable, ByteTable, 1); |
4483 | 4492 | ||
4484 | mxl5005s_AssignTunerMode(fe); // tunre_config | 4493 | mxl5005s_AssignTunerMode(fe, mod_type, bandwidth); |
4485 | 4494 | ||
4486 | /* Tuner initialization stage 1 */ | 4495 | /* Tuner initialization stage 1 */ |
4487 | MXL_GetInitRegister(fe, AddrTable, ByteTable, &TableLen); | 4496 | MXL_GetInitRegister(fe, AddrTable, ByteTable, &TableLen); |
4488 | 4497 | ||
4489 | mxl5005s_SetRegsWithTable(fe, AddrTable, ByteTable, TableLen); | 4498 | mxl5005s_writeregs(fe, AddrTable, ByteTable, TableLen); |
4490 | 4499 | ||
4491 | return 0; | 4500 | return 0; |
4492 | } | 4501 | } |
4493 | 4502 | ||
4494 | int mxl5005s_AssignTunerMode(struct dvb_frontend *fe) | 4503 | int mxl5005s_AssignTunerMode(struct dvb_frontend *fe, u32 mod_type, u32 bandwidth) |
4495 | { | 4504 | { |
4496 | struct mxl5005s_state *state = fe->tuner_priv; | 4505 | struct mxl5005s_state *state = fe->tuner_priv; |
4497 | struct mxl5005s_config *c = state->config; | 4506 | struct mxl5005s_config *c = state->config; |
@@ -4503,7 +4512,7 @@ int mxl5005s_AssignTunerMode(struct dvb_frontend *fe) | |||
4503 | fe, | 4512 | fe, |
4504 | c->mod_mode, | 4513 | c->mod_mode, |
4505 | c->if_mode, | 4514 | c->if_mode, |
4506 | MXL5005S_BANDWIDTH_6MHZ, | 4515 | bandwidth, |
4507 | c->if_freq, | 4516 | c->if_freq, |
4508 | c->xtal_freq, | 4517 | c->xtal_freq, |
4509 | c->agc_mode, | 4518 | c->agc_mode, |
@@ -4513,7 +4522,7 @@ int mxl5005s_AssignTunerMode(struct dvb_frontend *fe) | |||
4513 | c->div_out, | 4522 | c->div_out, |
4514 | c->cap_select, | 4523 | c->cap_select, |
4515 | c->rssi_enable, | 4524 | c->rssi_enable, |
4516 | MXL_QAM, | 4525 | mod_type, |
4517 | c->tracking_filter); | 4526 | c->tracking_filter); |
4518 | 4527 | ||
4519 | return 0; | 4528 | return 0; |
@@ -4522,22 +4531,62 @@ int mxl5005s_AssignTunerMode(struct dvb_frontend *fe) | |||
4522 | static int mxl5005s_set_params(struct dvb_frontend *fe, | 4531 | static int mxl5005s_set_params(struct dvb_frontend *fe, |
4523 | struct dvb_frontend_parameters *params) | 4532 | struct dvb_frontend_parameters *params) |
4524 | { | 4533 | { |
4525 | u32 freq; | 4534 | struct mxl5005s_state *state = fe->tuner_priv; |
4526 | u32 bw; | 4535 | u32 req_mode, req_bw = 0; |
4536 | int ret; | ||
4527 | 4537 | ||
4528 | if (fe->ops.info.type == FE_OFDM) | 4538 | dprintk(1, "%s()\n", __func__); |
4529 | bw = params->u.ofdm.bandwidth; | 4539 | |
4530 | else | 4540 | if (fe->ops.info.type == FE_ATSC) { |
4531 | bw = MXL5005S_BANDWIDTH_6MHZ; | 4541 | switch (params->u.vsb.modulation) { |
4542 | case VSB_8: | ||
4543 | req_mode = MXL_ATSC; break; | ||
4544 | default: | ||
4545 | case QAM_64: | ||
4546 | case QAM_256: | ||
4547 | case QAM_AUTO: | ||
4548 | req_mode = MXL_QAM; break; | ||
4549 | } | ||
4550 | } | ||
4551 | else req_mode = MXL_DVBT; | ||
4552 | |||
4553 | /* Change tuner for new modulation type if reqd */ | ||
4554 | if (req_mode != state->current_mode) { | ||
4555 | switch (req_mode) { | ||
4556 | case VSB_8: | ||
4557 | case QAM_64: | ||
4558 | case QAM_256: | ||
4559 | case QAM_AUTO: | ||
4560 | req_bw = MXL5005S_BANDWIDTH_6MHZ; | ||
4561 | break; | ||
4562 | default: | ||
4563 | /* Assume DVB-T */ | ||
4564 | switch (params->u.ofdm.bandwidth) { | ||
4565 | case BANDWIDTH_6_MHZ: | ||
4566 | req_bw = MXL5005S_BANDWIDTH_6MHZ; | ||
4567 | break; | ||
4568 | case BANDWIDTH_7_MHZ: | ||
4569 | req_bw = MXL5005S_BANDWIDTH_7MHZ; | ||
4570 | break; | ||
4571 | case BANDWIDTH_AUTO: | ||
4572 | case BANDWIDTH_8_MHZ: | ||
4573 | req_bw = MXL5005S_BANDWIDTH_8MHZ; | ||
4574 | break; | ||
4575 | } | ||
4576 | } | ||
4532 | 4577 | ||
4533 | freq = params->frequency; /* Hz */ | 4578 | state->current_mode = req_mode; |
4534 | dprintk(1, "%s() freq=%d bw=%d\n", __func__, freq, bw); | 4579 | ret = mxl5005s_reconfigure(fe, req_mode, req_bw); |
4535 | 4580 | ||
4536 | mxl5005s_SetRfFreqHz(fe, freq); | 4581 | } else |
4582 | ret = 0; | ||
4537 | 4583 | ||
4538 | msleep(350); | 4584 | if (ret == 0) { |
4585 | dprintk(1, "%s() freq=%d\n", __func__, params->frequency); | ||
4586 | ret = mxl5005s_SetRfFreqHz(fe, params->frequency); | ||
4587 | } | ||
4539 | 4588 | ||
4540 | return 0; | 4589 | return ret; |
4541 | } | 4590 | } |
4542 | 4591 | ||
4543 | static int mxl5005s_get_frequency(struct dvb_frontend *fe, u32 *frequency) | 4592 | static int mxl5005s_get_frequency(struct dvb_frontend *fe, u32 *frequency) |
@@ -4560,16 +4609,6 @@ static int mxl5005s_get_bandwidth(struct dvb_frontend *fe, u32 *bandwidth) | |||
4560 | return 0; | 4609 | return 0; |
4561 | } | 4610 | } |
4562 | 4611 | ||
4563 | static int mxl5005s_get_status(struct dvb_frontend *fe, u32 *status) | ||
4564 | { | ||
4565 | dprintk(1, "%s()\n", __func__); | ||
4566 | |||
4567 | *status = 0; | ||
4568 | // *status = TUNER_STATUS_LOCKED; | ||
4569 | |||
4570 | return 0; | ||
4571 | } | ||
4572 | |||
4573 | static int mxl5005s_release(struct dvb_frontend *fe) | 4612 | static int mxl5005s_release(struct dvb_frontend *fe) |
4574 | { | 4613 | { |
4575 | dprintk(1, "%s()\n", __func__); | 4614 | dprintk(1, "%s()\n", __func__); |
@@ -4592,7 +4631,6 @@ static const struct dvb_tuner_ops mxl5005s_tuner_ops = { | |||
4592 | .set_params = mxl5005s_set_params, | 4631 | .set_params = mxl5005s_set_params, |
4593 | .get_frequency = mxl5005s_get_frequency, | 4632 | .get_frequency = mxl5005s_get_frequency, |
4594 | .get_bandwidth = mxl5005s_get_bandwidth, | 4633 | .get_bandwidth = mxl5005s_get_bandwidth, |
4595 | .get_status = mxl5005s_get_status | ||
4596 | }; | 4634 | }; |
4597 | 4635 | ||
4598 | struct dvb_frontend *mxl5005s_attach(struct dvb_frontend *fe, | 4636 | struct dvb_frontend *mxl5005s_attach(struct dvb_frontend *fe, |
@@ -4609,6 +4647,7 @@ struct dvb_frontend *mxl5005s_attach(struct dvb_frontend *fe, | |||
4609 | state->frontend = fe; | 4647 | state->frontend = fe; |
4610 | state->config = config; | 4648 | state->config = config; |
4611 | state->i2c = i2c; | 4649 | state->i2c = i2c; |
4650 | state->current_mode = MXL_QAM; | ||
4612 | 4651 | ||
4613 | printk(KERN_INFO "MXL5005S: Attached at address 0x%02x\n", config->i2c_address); | 4652 | printk(KERN_INFO "MXL5005S: Attached at address 0x%02x\n", config->i2c_address); |
4614 | 4653 | ||
@@ -4620,8 +4659,5 @@ struct dvb_frontend *mxl5005s_attach(struct dvb_frontend *fe, | |||
4620 | EXPORT_SYMBOL(mxl5005s_attach); | 4659 | EXPORT_SYMBOL(mxl5005s_attach); |
4621 | 4660 | ||
4622 | MODULE_DESCRIPTION("MaxLinear MXL5005S silicon tuner driver"); | 4661 | MODULE_DESCRIPTION("MaxLinear MXL5005S silicon tuner driver"); |
4623 | MODULE_AUTHOR("Jan Hoogenraad"); | ||
4624 | MODULE_AUTHOR("Barnaby Shearer"); | ||
4625 | MODULE_AUTHOR("Andy Hasper"); | ||
4626 | MODULE_AUTHOR("Steven Toth"); | 4662 | MODULE_AUTHOR("Steven Toth"); |
4627 | MODULE_LICENSE("GPL"); | 4663 | MODULE_LICENSE("GPL"); |