diff options
author | Steven Toth <stoth@hauppauge.com> | 2008-05-01 06:04:09 -0400 |
---|---|---|
committer | Mauro Carvalho Chehab <mchehab@infradead.org> | 2008-05-14 01:56:39 -0400 |
commit | 5c1b20514f592af19974166f130b85346c1fbf3a (patch) | |
tree | 3d2a55ccbe6e84555b5a4931cd9d5eef7b9edac1 /drivers | |
parent | 7f5c3affef2883f49e820db62413e1dff1d4cebb (diff) |
V4L/DVB (7870): mxl5005s: Basic digital support.
ATSC and QAM should be working but basic testing is required.
Signed-off-by: Steven Toth <stoth@hauppauge.com>
Signed-off-by: Mauro Carvalho Chehab <mchehab@infradead.org>
Diffstat (limited to 'drivers')
-rw-r--r-- | drivers/media/common/tuners/mxl5005s.c | 616 | ||||
-rw-r--r-- | drivers/media/common/tuners/mxl5005s.h | 145 |
2 files changed, 389 insertions, 372 deletions
diff --git a/drivers/media/common/tuners/mxl5005s.c b/drivers/media/common/tuners/mxl5005s.c index f7ed9a72db4a..64aa864c5dbf 100644 --- a/drivers/media/common/tuners/mxl5005s.c +++ b/drivers/media/common/tuners/mxl5005s.c | |||
@@ -1,69 +1,40 @@ | |||
1 | /* | 1 | /* |
2 | MaxLinear MXL5005S VSB/QAM/DVBT tuner driver | 2 | * For the Realtek RTL chip RTL2831U |
3 | 3 | * Realtek Release Date: 2008-03-14, ver 080314 | |
4 | Copyright (C) 2008 MaxLinear | 4 | * Realtek version RTL2831 Linux driver version 080314 |
5 | Copyright (C) 2006 Steven Toth <stoth@hauppauge.com> | 5 | * ver 080314 |
6 | Functions: | 6 | * |
7 | mxl5005s_reset() | 7 | * for linux kernel version 2.6.21.4 - 2.6.22-14 |
8 | mxl5005s_writereg() | 8 | * support MXL5005s and MT2060 tuners (support tuner auto-detecting) |
9 | mxl5005s_writeregs() | 9 | * support two IR types -- RC5 and NEC |
10 | mxl5005s_init() | 10 | * |
11 | mxl5005s_reconfigure() | 11 | * Known boards with Realtek RTL chip RTL2821U |
12 | mxl5005s_AssignTunerMode() | 12 | * Freecom USB stick 14aa:0160 (version 4) |
13 | mxl5005s_set_params() | 13 | * Conceptronic CTVDIGRCU |
14 | mxl5005s_get_frequency() | 14 | * |
15 | mxl5005s_get_bandwidth() | 15 | * Copyright (c) 2008 Realtek |
16 | mxl5005s_release() | 16 | * Copyright (c) 2008 Jan Hoogenraad, Barnaby Shearer, Andy Hasper |
17 | mxl5005s_attach() | 17 | * This code is placed under the terms of the GNU General Public License |
18 | 18 | * | |
19 | Copyright (c) 2008 Realtek | 19 | * Released by Realtek under GPLv2. |
20 | Copyright (c) 2008 Jan Hoogenraad, Barnaby Shearer, Andy Hasper | 20 | * Thanks to Realtek for a lot of support we received ! |
21 | Functions: | 21 | * |
22 | mxl5005s_SetRfFreqHz() | 22 | * Revision: 080314 - original version |
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 | */ | ||
60 | 24 | ||
25 | #include <linux/kernel.h> | ||
26 | #include <linux/init.h> | ||
27 | #include <linux/module.h> | ||
28 | #include <linux/string.h> | ||
29 | #include <linux/slab.h> | ||
30 | #include <linux/delay.h> | ||
31 | #include "dvb_frontend.h" | ||
61 | #include "mxl5005s.h" | 32 | #include "mxl5005s.h" |
62 | 33 | ||
63 | static int debug = 2; | 34 | static int debug; |
64 | 35 | ||
65 | #define dprintk(level, arg...) do { \ | 36 | #define dprintk(level, arg...) do { \ |
66 | if (level <= debug) \ | 37 | if (debug >= level) \ |
67 | printk(arg); \ | 38 | printk(arg); \ |
68 | } while (0) | 39 | } while (0) |
69 | 40 | ||
@@ -79,6 +50,13 @@ static int debug = 2; | |||
79 | #define MXLCTRL_NUM 189 | 50 | #define MXLCTRL_NUM 189 |
80 | #define MASTER_CONTROL_ADDR 9 | 51 | #define MASTER_CONTROL_ADDR 9 |
81 | 52 | ||
53 | /* Enumeration of AGC Mode */ | ||
54 | typedef enum | ||
55 | { | ||
56 | MXL_DUAL_AGC = 0, | ||
57 | MXL_SINGLE_AGC | ||
58 | } AGC_Mode; | ||
59 | |||
82 | /* Enumeration of Master Control Register State */ | 60 | /* Enumeration of Master Control Register State */ |
83 | typedef enum | 61 | typedef enum |
84 | { | 62 | { |
@@ -88,6 +66,51 @@ typedef enum | |||
88 | MC_SEQ_OFF | 66 | MC_SEQ_OFF |
89 | } Master_Control_State; | 67 | } Master_Control_State; |
90 | 68 | ||
69 | /* Enumeration of MXL5005 Tuner Mode */ | ||
70 | typedef enum | ||
71 | { | ||
72 | MXL_ANALOG_MODE = 0, | ||
73 | MXL_DIGITAL_MODE | ||
74 | } Tuner_Mode; | ||
75 | |||
76 | /* Enumeration of MXL5005 Tuner IF Mode */ | ||
77 | typedef enum | ||
78 | { | ||
79 | MXL_ZERO_IF = 0, | ||
80 | MXL_LOW_IF | ||
81 | } Tuner_IF_Mode; | ||
82 | |||
83 | /* Enumeration of MXL5005 Tuner Clock Out Mode */ | ||
84 | typedef enum | ||
85 | { | ||
86 | MXL_CLOCK_OUT_DISABLE = 0, | ||
87 | MXL_CLOCK_OUT_ENABLE | ||
88 | } Tuner_Clock_Out; | ||
89 | |||
90 | /* Enumeration of MXL5005 Tuner Div Out Mode */ | ||
91 | typedef enum | ||
92 | { | ||
93 | MXL_DIV_OUT_1 = 0, | ||
94 | MXL_DIV_OUT_4 | ||
95 | |||
96 | } Tuner_Div_Out; | ||
97 | |||
98 | /* Enumeration of MXL5005 Tuner Pull-up Cap Select Mode */ | ||
99 | typedef enum | ||
100 | { | ||
101 | MXL_CAP_SEL_DISABLE = 0, | ||
102 | MXL_CAP_SEL_ENABLE | ||
103 | |||
104 | } Tuner_Cap_Select; | ||
105 | |||
106 | /* Enumeration of MXL5005 Tuner RSSI Mode */ | ||
107 | typedef enum | ||
108 | { | ||
109 | MXL_RSSI_DISABLE = 0, | ||
110 | MXL_RSSI_ENABLE | ||
111 | |||
112 | } Tuner_RSSI; | ||
113 | |||
91 | /* Enumeration of MXL5005 Tuner Modulation Type */ | 114 | /* Enumeration of MXL5005 Tuner Modulation Type */ |
92 | typedef enum | 115 | typedef enum |
93 | { | 116 | { |
@@ -99,6 +122,22 @@ typedef enum | |||
99 | MXL_ANALOG_OTA | 122 | MXL_ANALOG_OTA |
100 | } Tuner_Modu_Type; | 123 | } Tuner_Modu_Type; |
101 | 124 | ||
125 | /* Enumeration of MXL5005 Tuner Tracking Filter Type */ | ||
126 | typedef enum | ||
127 | { | ||
128 | MXL_TF_DEFAULT = 0, | ||
129 | MXL_TF_OFF, | ||
130 | MXL_TF_C, | ||
131 | MXL_TF_C_H, | ||
132 | MXL_TF_D, | ||
133 | MXL_TF_D_L, | ||
134 | MXL_TF_E, | ||
135 | MXL_TF_F, | ||
136 | MXL_TF_E_2, | ||
137 | MXL_TF_E_NA, | ||
138 | MXL_TF_G | ||
139 | } Tuner_TF_Type; | ||
140 | |||
102 | /* MXL5005 Tuner Register Struct */ | 141 | /* MXL5005 Tuner Register Struct */ |
103 | typedef struct _TunerReg_struct | 142 | typedef struct _TunerReg_struct |
104 | { | 143 | { |
@@ -229,6 +268,33 @@ enum | |||
229 | }; | 268 | }; |
230 | #define MXL5005S_BANDWIDTH_MODE_NUM 3 | 269 | #define MXL5005S_BANDWIDTH_MODE_NUM 3 |
231 | 270 | ||
271 | /* Top modes */ | ||
272 | enum | ||
273 | { | ||
274 | MXL5005S_TOP_5P5 = 55, | ||
275 | MXL5005S_TOP_7P2 = 72, | ||
276 | MXL5005S_TOP_9P2 = 92, | ||
277 | MXL5005S_TOP_11P0 = 110, | ||
278 | MXL5005S_TOP_12P9 = 129, | ||
279 | MXL5005S_TOP_14P7 = 147, | ||
280 | MXL5005S_TOP_16P8 = 168, | ||
281 | MXL5005S_TOP_19P4 = 194, | ||
282 | MXL5005S_TOP_21P2 = 212, | ||
283 | MXL5005S_TOP_23P2 = 232, | ||
284 | MXL5005S_TOP_25P2 = 252, | ||
285 | MXL5005S_TOP_27P1 = 271, | ||
286 | MXL5005S_TOP_29P2 = 292, | ||
287 | MXL5005S_TOP_31P7 = 317, | ||
288 | MXL5005S_TOP_34P9 = 349, | ||
289 | }; | ||
290 | |||
291 | /* IF output load */ | ||
292 | enum | ||
293 | { | ||
294 | MXL5005S_IF_OUTPUT_LOAD_200_OHM = 200, | ||
295 | MXL5005S_IF_OUTPUT_LOAD_300_OHM = 300, | ||
296 | }; | ||
297 | |||
232 | /* MXL5005 Tuner Control Struct */ | 298 | /* MXL5005 Tuner Control Struct */ |
233 | typedef struct _TunerControl_struct { | 299 | typedef struct _TunerControl_struct { |
234 | u16 Ctrl_Num; /* Control Number */ | 300 | u16 Ctrl_Num; /* Control Number */ |
@@ -283,15 +349,13 @@ struct mxl5005s_state | |||
283 | TunerRegs[TUNER_REGS_NUM]; /* Tuner Register Array Pointer */ | 349 | TunerRegs[TUNER_REGS_NUM]; /* Tuner Register Array Pointer */ |
284 | 350 | ||
285 | /* Linux driver framework specific */ | 351 | /* Linux driver framework specific */ |
286 | struct mxl5005s_config *config; | 352 | const struct mxl5005s_config *config; |
353 | |||
287 | struct dvb_frontend *frontend; | 354 | struct dvb_frontend *frontend; |
288 | struct i2c_adapter *i2c; | 355 | struct i2c_adapter *i2c; |
289 | |||
290 | /* Cache values */ | ||
291 | u32 current_mode; | ||
292 | |||
293 | }; | 356 | }; |
294 | 357 | ||
358 | // funcs | ||
295 | u16 MXL_ControlWrite(struct dvb_frontend *fe, u16 ControlNum, u32 value); | 359 | u16 MXL_ControlWrite(struct dvb_frontend *fe, u16 ControlNum, u32 value); |
296 | u16 MXL_ControlRead(struct dvb_frontend *fe, u16 controlNum, u32 *value); | 360 | u16 MXL_ControlRead(struct dvb_frontend *fe, u16 controlNum, u32 *value); |
297 | u16 MXL_GetMasterControl(u8 *MasterReg, int state); | 361 | u16 MXL_GetMasterControl(u8 *MasterReg, int state); |
@@ -308,26 +372,14 @@ u16 MXL_TuneRF(struct dvb_frontend *fe, u32 RF_Freq); | |||
308 | void MXL_SynthIFLO_Calc(struct dvb_frontend *fe); | 372 | void MXL_SynthIFLO_Calc(struct dvb_frontend *fe); |
309 | void MXL_SynthRFTGLO_Calc(struct dvb_frontend *fe); | 373 | void MXL_SynthRFTGLO_Calc(struct dvb_frontend *fe); |
310 | u16 MXL_GetCHRegister_ZeroIF(struct dvb_frontend *fe, u8 *RegNum, u8 *RegVal, int *count); | 374 | u16 MXL_GetCHRegister_ZeroIF(struct dvb_frontend *fe, u8 *RegNum, u8 *RegVal, int *count); |
311 | int mxl5005s_writeregs(struct dvb_frontend *fe, u8 *addrtable, u8 *datatable, u8 len); | 375 | int mxl5005s_SetRegsWithTable(struct dvb_frontend *fe, u8 *pAddrTable, u8 *pByteTable, int TableLen); |
312 | u16 MXL_IFSynthInit(struct dvb_frontend *fe); | 376 | u16 MXL_IFSynthInit(struct dvb_frontend *fe); |
313 | int mxl5005s_AssignTunerMode(struct dvb_frontend *fe, u32 mod_type, u32 bandwidth); | 377 | static int mxl5005s_init2(struct dvb_frontend *fe); |
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 | */ | ||
327 | 378 | ||
328 | int mxl5005s_SetRfFreqHz(struct dvb_frontend *fe, unsigned long RfFreqHz) | 379 | int mxl5005s_SetRfFreqHz(struct dvb_frontend *fe, unsigned long RfFreqHz) |
329 | { | 380 | { |
330 | struct mxl5005s_state *state = fe->tuner_priv; | 381 | struct mxl5005s_state *state = fe->tuner_priv; |
382 | u8 AgcMasterByte = state->config->AgcMasterByte; | ||
331 | unsigned char AddrTable[MXL5005S_REG_WRITING_TABLE_LEN_MAX]; | 383 | unsigned char AddrTable[MXL5005S_REG_WRITING_TABLE_LEN_MAX]; |
332 | unsigned char ByteTable[MXL5005S_REG_WRITING_TABLE_LEN_MAX]; | 384 | unsigned char ByteTable[MXL5005S_REG_WRITING_TABLE_LEN_MAX]; |
333 | int TableLen; | 385 | int TableLen; |
@@ -344,7 +396,7 @@ int mxl5005s_SetRfFreqHz(struct dvb_frontend *fe, unsigned long RfFreqHz) | |||
344 | AddrTable[0] = MASTER_CONTROL_ADDR; | 396 | AddrTable[0] = MASTER_CONTROL_ADDR; |
345 | ByteTable[0] |= state->config->AgcMasterByte; | 397 | ByteTable[0] |= state->config->AgcMasterByte; |
346 | 398 | ||
347 | mxl5005s_writeregs(fe, AddrTable, ByteTable, 1); | 399 | mxl5005s_SetRegsWithTable(fe, AddrTable, ByteTable, 1); |
348 | 400 | ||
349 | // Tuner RF frequency setting stage 1 | 401 | // Tuner RF frequency setting stage 1 |
350 | MXL_TuneRF(fe, RfFreqHz); | 402 | MXL_TuneRF(fe, RfFreqHz); |
@@ -358,13 +410,13 @@ int mxl5005s_SetRfFreqHz(struct dvb_frontend *fe, unsigned long RfFreqHz) | |||
358 | 410 | ||
359 | MXL_GetMasterControl(&MasterControlByte, MC_LOAD_START) ; | 411 | MXL_GetMasterControl(&MasterControlByte, MC_LOAD_START) ; |
360 | AddrTable[TableLen] = MASTER_CONTROL_ADDR ; | 412 | AddrTable[TableLen] = MASTER_CONTROL_ADDR ; |
361 | ByteTable[TableLen] = MasterControlByte | state->config->AgcMasterByte; | 413 | ByteTable[TableLen] = MasterControlByte | AgcMasterByte; |
362 | TableLen += 1; | 414 | TableLen += 1; |
363 | 415 | ||
364 | mxl5005s_writeregs(fe, AddrTable, ByteTable, TableLen); | 416 | mxl5005s_SetRegsWithTable(fe, AddrTable, ByteTable, TableLen); |
365 | 417 | ||
366 | // Wait 30 ms. | 418 | // Wait 30 ms. |
367 | msleep(150); | 419 | msleep(30); |
368 | 420 | ||
369 | // Tuner RF frequency setting stage 2 | 421 | // Tuner RF frequency setting stage 2 |
370 | MXL_ControlWrite(fe, SEQ_FSM_PULSE, 1) ; | 422 | MXL_ControlWrite(fe, SEQ_FSM_PULSE, 1) ; |
@@ -373,21 +425,101 @@ int mxl5005s_SetRfFreqHz(struct dvb_frontend *fe, unsigned long RfFreqHz) | |||
373 | 425 | ||
374 | MXL_GetMasterControl(&MasterControlByte, MC_LOAD_START) ; | 426 | MXL_GetMasterControl(&MasterControlByte, MC_LOAD_START) ; |
375 | AddrTable[TableLen] = MASTER_CONTROL_ADDR ; | 427 | AddrTable[TableLen] = MASTER_CONTROL_ADDR ; |
376 | ByteTable[TableLen] = MasterControlByte | state->config->AgcMasterByte ; | 428 | ByteTable[TableLen] = MasterControlByte | AgcMasterByte ; |
377 | TableLen += 1; | 429 | TableLen += 1; |
378 | 430 | ||
379 | mxl5005s_writeregs(fe, AddrTable, ByteTable, TableLen); | 431 | mxl5005s_SetRegsWithTable(fe, AddrTable, ByteTable, TableLen); |
380 | 432 | ||
381 | msleep(100); | 433 | return 0; |
434 | } | ||
382 | 435 | ||
436 | /* Write a single byte to a single reg */ | ||
437 | static int mxl5005s_writereg(struct dvb_frontend *fe, u8 reg, u8 val) | ||
438 | { | ||
439 | struct mxl5005s_state *state = fe->tuner_priv; | ||
440 | u8 buf[2] = { reg, val }; | ||
441 | struct i2c_msg msg = { .addr = state->config->i2c_address, .flags = 0, | ||
442 | .buf = buf, .len = 2 }; | ||
443 | |||
444 | if (i2c_transfer(state->i2c, &msg, 1) != 1) { | ||
445 | printk(KERN_WARNING "mxl5005s I2C write failed\n"); | ||
446 | return -EREMOTEIO; | ||
447 | } | ||
383 | return 0; | 448 | return 0; |
384 | } | 449 | } |
385 | /* End: Custom code taken from the Realtek driver */ | ||
386 | 450 | ||
387 | /* ---------------------------------------------------------------- | 451 | /* Write a word to a single reg */ |
388 | * Begin: Reference driver code found in the Realtek driver. | 452 | static int mxl5005s_writereg16(struct dvb_frontend *fe, u8 reg, u16 val) |
389 | * Copyright (c) 2008 MaxLinear | 453 | { |
390 | */ | 454 | struct mxl5005s_state *state = fe->tuner_priv; |
455 | u8 buf[3] = { reg, val >> 8 , val & 0xff }; | ||
456 | struct i2c_msg msg = { .addr = state->config->i2c_address, .flags = 0, | ||
457 | .buf = buf, .len = 3 }; | ||
458 | |||
459 | if (i2c_transfer(state->i2c, &msg, 1) != 1) { | ||
460 | printk(KERN_WARNING "mxl5005s I2C write16 failed\n"); | ||
461 | return -EREMOTEIO; | ||
462 | } | ||
463 | return 0; | ||
464 | } | ||
465 | |||
466 | int mxl5005s_SetRegsWithTable(struct dvb_frontend *fe, u8 *pAddrTable, u8 *pByteTable, int TableLen) | ||
467 | { | ||
468 | int i, ret; | ||
469 | u8 end_two_bytes_buf[]={ 0 , 0 }; | ||
470 | |||
471 | for( i = 0 ; i < TableLen - 1 ; i++) | ||
472 | { | ||
473 | ret = mxl5005s_writereg(fe, pAddrTable[i], pByteTable[i]); | ||
474 | if (!ret) | ||
475 | return ret; | ||
476 | } | ||
477 | |||
478 | end_two_bytes_buf[0] = pByteTable[i]; | ||
479 | end_two_bytes_buf[1] = MXL5005S_LATCH_BYTE; | ||
480 | |||
481 | ret = mxl5005s_writereg16(fe, pAddrTable[i], (end_two_bytes_buf[0] << 8) | end_two_bytes_buf[1]); | ||
482 | |||
483 | return ret; | ||
484 | } | ||
485 | |||
486 | int mxl5005s_SetRegMaskBits(struct dvb_frontend *fe, | ||
487 | unsigned char RegAddr, | ||
488 | unsigned char Msb, | ||
489 | unsigned char Lsb, | ||
490 | const unsigned char WritingValue | ||
491 | ) | ||
492 | { | ||
493 | int i; | ||
494 | |||
495 | unsigned char Mask; | ||
496 | unsigned char Shift; | ||
497 | unsigned char RegByte; | ||
498 | |||
499 | /* Generate mask and shift according to MSB and LSB. */ | ||
500 | Mask = 0; | ||
501 | for(i = Lsb; i < (unsigned char)(Msb + 1); i++) | ||
502 | Mask |= 0x1 << i; | ||
503 | |||
504 | Shift = Lsb; | ||
505 | |||
506 | /* Get tuner register byte according to register adddress. */ | ||
507 | MXL_RegRead(fe, RegAddr, &RegByte); | ||
508 | |||
509 | /* Reserve register byte unmask bit with mask and inlay writing value into it. */ | ||
510 | RegByte &= ~Mask; | ||
511 | RegByte |= (WritingValue << Shift) & Mask; | ||
512 | |||
513 | /* Update tuner register byte table. */ | ||
514 | MXL_RegWrite(fe, RegAddr, RegByte); | ||
515 | |||
516 | /* Write tuner register byte with writing byte. */ | ||
517 | return mxl5005s_SetRegsWithTable(fe, &RegAddr, &RegByte, 1); | ||
518 | } | ||
519 | |||
520 | // The following context is source code provided by MaxLinear. | ||
521 | // MaxLinear source code - MXL5005_Initialize.cpp | ||
522 | // DONE | ||
391 | u16 MXL5005_RegisterInit(struct dvb_frontend *fe) | 523 | u16 MXL5005_RegisterInit(struct dvb_frontend *fe) |
392 | { | 524 | { |
393 | struct mxl5005s_state *state = fe->tuner_priv; | 525 | struct mxl5005s_state *state = fe->tuner_priv; |
@@ -709,6 +841,7 @@ u16 MXL5005_RegisterInit(struct dvb_frontend *fe) | |||
709 | return 0 ; | 841 | return 0 ; |
710 | } | 842 | } |
711 | 843 | ||
844 | // DONE | ||
712 | u16 MXL5005_ControlInit(struct dvb_frontend *fe) | 845 | u16 MXL5005_ControlInit(struct dvb_frontend *fe) |
713 | { | 846 | { |
714 | struct mxl5005s_state *state = fe->tuner_priv; | 847 | struct mxl5005s_state *state = fe->tuner_priv; |
@@ -1652,6 +1785,7 @@ u16 MXL5005_ControlInit(struct dvb_frontend *fe) | |||
1652 | // MaxLinear source code - MXL5005_c.cpp | 1785 | // MaxLinear source code - MXL5005_c.cpp |
1653 | // MXL5005.cpp : Defines the initialization routines for the DLL. | 1786 | // MXL5005.cpp : Defines the initialization routines for the DLL. |
1654 | // 2.6.12 | 1787 | // 2.6.12 |
1788 | // DONE | ||
1655 | void InitTunerControls(struct dvb_frontend *fe) | 1789 | void InitTunerControls(struct dvb_frontend *fe) |
1656 | { | 1790 | { |
1657 | MXL5005_RegisterInit(fe); | 1791 | MXL5005_RegisterInit(fe); |
@@ -1694,6 +1828,7 @@ void InitTunerControls(struct dvb_frontend *fe) | |||
1694 | // > 0 : Failed // | 1828 | // > 0 : Failed // |
1695 | // // | 1829 | // // |
1696 | /////////////////////////////////////////////////////////////////////////////// | 1830 | /////////////////////////////////////////////////////////////////////////////// |
1831 | // DONE | ||
1697 | u16 MXL5005_TunerConfig(struct dvb_frontend *fe, | 1832 | u16 MXL5005_TunerConfig(struct dvb_frontend *fe, |
1698 | u8 Mode, /* 0: Analog Mode ; 1: Digital Mode */ | 1833 | u8 Mode, /* 0: Analog Mode ; 1: Digital Mode */ |
1699 | u8 IF_mode, /* for Analog Mode, 0: zero IF; 1: low IF */ | 1834 | u8 IF_mode, /* for Analog Mode, 0: zero IF; 1: low IF */ |
@@ -1763,6 +1898,7 @@ u16 MXL5005_TunerConfig(struct dvb_frontend *fe, | |||
1763 | // > 0 : Failed // | 1898 | // > 0 : Failed // |
1764 | // // | 1899 | // // |
1765 | /////////////////////////////////////////////////////////////////////////////// | 1900 | /////////////////////////////////////////////////////////////////////////////// |
1901 | // DONE | ||
1766 | void MXL_SynthIFLO_Calc(struct dvb_frontend *fe) | 1902 | void MXL_SynthIFLO_Calc(struct dvb_frontend *fe) |
1767 | { | 1903 | { |
1768 | struct mxl5005s_state *state = fe->tuner_priv; | 1904 | struct mxl5005s_state *state = fe->tuner_priv; |
@@ -1801,6 +1937,7 @@ void MXL_SynthIFLO_Calc(struct dvb_frontend *fe) | |||
1801 | // > 0 : Failed // | 1937 | // > 0 : Failed // |
1802 | // // | 1938 | // // |
1803 | /////////////////////////////////////////////////////////////////////////////// | 1939 | /////////////////////////////////////////////////////////////////////////////// |
1940 | // DONE | ||
1804 | void MXL_SynthRFTGLO_Calc(struct dvb_frontend *fe) | 1941 | void MXL_SynthRFTGLO_Calc(struct dvb_frontend *fe) |
1805 | { | 1942 | { |
1806 | struct mxl5005s_state *state = fe->tuner_priv; | 1943 | struct mxl5005s_state *state = fe->tuner_priv; |
@@ -1839,6 +1976,7 @@ void MXL_SynthRFTGLO_Calc(struct dvb_frontend *fe) | |||
1839 | // > 0 : Failed // | 1976 | // > 0 : Failed // |
1840 | // // | 1977 | // // |
1841 | /////////////////////////////////////////////////////////////////////////////// | 1978 | /////////////////////////////////////////////////////////////////////////////// |
1979 | // DONE | ||
1842 | u16 MXL_OverwriteICDefault(struct dvb_frontend *fe) | 1980 | u16 MXL_OverwriteICDefault(struct dvb_frontend *fe) |
1843 | { | 1981 | { |
1844 | u16 status = 0; | 1982 | u16 status = 0; |
@@ -1876,6 +2014,7 @@ u16 MXL_OverwriteICDefault(struct dvb_frontend *fe) | |||
1876 | // > 0 : Failed // | 2014 | // > 0 : Failed // |
1877 | // // | 2015 | // // |
1878 | /////////////////////////////////////////////////////////////////////////////// | 2016 | /////////////////////////////////////////////////////////////////////////////// |
2017 | // DONE | ||
1879 | u16 MXL_BlockInit(struct dvb_frontend *fe) | 2018 | u16 MXL_BlockInit(struct dvb_frontend *fe) |
1880 | { | 2019 | { |
1881 | struct mxl5005s_state *state = fe->tuner_priv; | 2020 | struct mxl5005s_state *state = fe->tuner_priv; |
@@ -1903,7 +2042,6 @@ u16 MXL_BlockInit(struct dvb_frontend *fe) | |||
1903 | status += MXL_ControlWrite(fe, BB_DLPF_BANDSEL, 2); | 2042 | status += MXL_ControlWrite(fe, BB_DLPF_BANDSEL, 2); |
1904 | break; | 2043 | break; |
1905 | case 6000000: | 2044 | case 6000000: |
1906 | printk("%s() doing 6MHz digital\n", __func__); | ||
1907 | status += MXL_ControlWrite(fe, BB_DLPF_BANDSEL, 3); | 2045 | status += MXL_ControlWrite(fe, BB_DLPF_BANDSEL, 3); |
1908 | break; | 2046 | break; |
1909 | } | 2047 | } |
@@ -1934,6 +2072,7 @@ u16 MXL_BlockInit(struct dvb_frontend *fe) | |||
1934 | else /* Single AGC Mode Dig Ana */ | 2072 | else /* Single AGC Mode Dig Ana */ |
1935 | status += MXL_ControlWrite(fe, AGC_RF, state->Mode ? 15 : 12); | 2073 | status += MXL_ControlWrite(fe, AGC_RF, state->Mode ? 15 : 12); |
1936 | 2074 | ||
2075 | |||
1937 | if (state->TOP == 55) /* TOP == 5.5 */ | 2076 | if (state->TOP == 55) /* TOP == 5.5 */ |
1938 | status += MXL_ControlWrite(fe, AGC_IF, 0x0); | 2077 | status += MXL_ControlWrite(fe, AGC_IF, 0x0); |
1939 | 2078 | ||
@@ -2163,8 +2302,6 @@ u16 MXL_BlockInit(struct dvb_frontend *fe) | |||
2163 | status += MXL_ControlWrite(fe, BB_IQSWAP, 0); | 2302 | status += MXL_ControlWrite(fe, BB_IQSWAP, 0); |
2164 | else /* High IF */ | 2303 | else /* High IF */ |
2165 | status += MXL_ControlWrite(fe, BB_IQSWAP, 1); | 2304 | status += MXL_ControlWrite(fe, BB_IQSWAP, 1); |
2166 | status += MXL_ControlWrite(fe, RFSYN_CHP_GAIN, 2); | ||
2167 | |||
2168 | } | 2305 | } |
2169 | if (state->Mod_Type == MXL_ANALOG_CABLE) { | 2306 | if (state->Mod_Type == MXL_ANALOG_CABLE) { |
2170 | /* Analog Cable Mode */ | 2307 | /* Analog Cable Mode */ |
@@ -2201,7 +2338,7 @@ u16 MXL_BlockInit(struct dvb_frontend *fe) | |||
2201 | } | 2338 | } |
2202 | 2339 | ||
2203 | /* RSSI disable */ | 2340 | /* RSSI disable */ |
2204 | if(state->EN_RSSI == 0) { | 2341 | if(state->EN_RSSI==0) { |
2205 | status += MXL_ControlWrite(fe, SEQ_EXTSYNTHCALIF, 1); | 2342 | status += MXL_ControlWrite(fe, SEQ_EXTSYNTHCALIF, 1); |
2206 | status += MXL_ControlWrite(fe, SEQ_EXTDCCAL, 1); | 2343 | status += MXL_ControlWrite(fe, SEQ_EXTDCCAL, 1); |
2207 | status += MXL_ControlWrite(fe, AGC_EN_RSSI, 0); | 2344 | status += MXL_ControlWrite(fe, AGC_EN_RSSI, 0); |
@@ -2410,7 +2547,6 @@ u16 MXL_IFSynthInit(struct dvb_frontend *fe) | |||
2410 | Fref = 324000000UL ; | 2547 | Fref = 324000000UL ; |
2411 | } | 2548 | } |
2412 | if (state->IF_LO == 5380000UL) { | 2549 | if (state->IF_LO == 5380000UL) { |
2413 | printk("%s() doing 5.38\n", __func__); | ||
2414 | status += MXL_ControlWrite(fe, IF_DIVVAL, 0x07) ; | 2550 | status += MXL_ControlWrite(fe, IF_DIVVAL, 0x07) ; |
2415 | status += MXL_ControlWrite(fe, IF_VCO_BIAS, 0x0C) ; | 2551 | status += MXL_ControlWrite(fe, IF_VCO_BIAS, 0x0C) ; |
2416 | Fref = 322800000UL ; | 2552 | Fref = 322800000UL ; |
@@ -3093,7 +3229,6 @@ u16 MXL_TuneRF(struct dvb_frontend *fe, u32 RF_Freq) | |||
3093 | 3229 | ||
3094 | if (state->TF_Type == MXL_TF_C_H) // Tracking Filter type C-H for Hauppauge only | 3230 | if (state->TF_Type == MXL_TF_C_H) // Tracking Filter type C-H for Hauppauge only |
3095 | { | 3231 | { |
3096 | printk("%s() CH filter\n", __func__); | ||
3097 | status += MXL_ControlWrite(fe, DAC_DIN_A, 0) ; | 3232 | status += MXL_ControlWrite(fe, DAC_DIN_A, 0) ; |
3098 | 3233 | ||
3099 | if (state->RF_IN >= 43000000 && state->RF_IN < 150000000) | 3234 | if (state->RF_IN >= 43000000 && state->RF_IN < 150000000) |
@@ -3632,6 +3767,7 @@ u16 MXL_TuneRF(struct dvb_frontend *fe, u32 RF_Freq) | |||
3632 | return status ; | 3767 | return status ; |
3633 | } | 3768 | } |
3634 | 3769 | ||
3770 | // DONE | ||
3635 | u16 MXL_SetGPIO(struct dvb_frontend *fe, u8 GPIO_Num, u8 GPIO_Val) | 3771 | u16 MXL_SetGPIO(struct dvb_frontend *fe, u8 GPIO_Num, u8 GPIO_Val) |
3636 | { | 3772 | { |
3637 | u16 status = 0; | 3773 | u16 status = 0; |
@@ -3698,6 +3834,7 @@ u16 MXL_SetGPIO(struct dvb_frontend *fe, u8 GPIO_Num, u8 GPIO_Val) | |||
3698 | // >0 : Value exceed maximum allowed for control number // | 3834 | // >0 : Value exceed maximum allowed for control number // |
3699 | // // | 3835 | // // |
3700 | /////////////////////////////////////////////////////////////////////////////// | 3836 | /////////////////////////////////////////////////////////////////////////////// |
3837 | // DONE | ||
3701 | u16 MXL_ControlWrite(struct dvb_frontend *fe, u16 ControlNum, u32 value) | 3838 | u16 MXL_ControlWrite(struct dvb_frontend *fe, u16 ControlNum, u32 value) |
3702 | { | 3839 | { |
3703 | u16 status = 0; | 3840 | u16 status = 0; |
@@ -3738,6 +3875,7 @@ u16 MXL_ControlWrite(struct dvb_frontend *fe, u16 ControlNum, u32 value) | |||
3738 | // 2 : Control name not found // | 3875 | // 2 : Control name not found // |
3739 | // // | 3876 | // // |
3740 | /////////////////////////////////////////////////////////////////////////////// | 3877 | /////////////////////////////////////////////////////////////////////////////// |
3878 | // DONE | ||
3741 | u16 MXL_ControlWrite_Group(struct dvb_frontend *fe, u16 controlNum, u32 value, u16 controlGroup) | 3879 | u16 MXL_ControlWrite_Group(struct dvb_frontend *fe, u16 controlNum, u32 value, u16 controlGroup) |
3742 | { | 3880 | { |
3743 | struct mxl5005s_state *state = fe->tuner_priv; | 3881 | struct mxl5005s_state *state = fe->tuner_priv; |
@@ -3844,6 +3982,7 @@ u16 MXL_ControlWrite_Group(struct dvb_frontend *fe, u16 controlNum, u32 value, u | |||
3844 | // -1 : Invalid Register Address // | 3982 | // -1 : Invalid Register Address // |
3845 | // // | 3983 | // // |
3846 | /////////////////////////////////////////////////////////////////////////////// | 3984 | /////////////////////////////////////////////////////////////////////////////// |
3985 | // DONE | ||
3847 | u16 MXL_RegWrite(struct dvb_frontend *fe, u8 RegNum, u8 RegVal) | 3986 | u16 MXL_RegWrite(struct dvb_frontend *fe, u8 RegNum, u8 RegVal) |
3848 | { | 3987 | { |
3849 | struct mxl5005s_state *state = fe->tuner_priv; | 3988 | struct mxl5005s_state *state = fe->tuner_priv; |
@@ -3883,6 +4022,7 @@ u16 MXL_RegWrite(struct dvb_frontend *fe, u8 RegNum, u8 RegVal) | |||
3883 | // -1 : Invalid Register Address // | 4022 | // -1 : Invalid Register Address // |
3884 | // // | 4023 | // // |
3885 | /////////////////////////////////////////////////////////////////////////////// | 4024 | /////////////////////////////////////////////////////////////////////////////// |
4025 | // DONE | ||
3886 | u16 MXL_RegRead(struct dvb_frontend *fe, u8 RegNum, u8 *RegVal) | 4026 | u16 MXL_RegRead(struct dvb_frontend *fe, u8 RegNum, u8 *RegVal) |
3887 | { | 4027 | { |
3888 | struct mxl5005s_state *state = fe->tuner_priv; | 4028 | struct mxl5005s_state *state = fe->tuner_priv; |
@@ -3919,6 +4059,7 @@ u16 MXL_RegRead(struct dvb_frontend *fe, u8 RegNum, u8 *RegVal) | |||
3919 | // -1 : Invalid control name // | 4059 | // -1 : Invalid control name // |
3920 | // // | 4060 | // // |
3921 | /////////////////////////////////////////////////////////////////////////////// | 4061 | /////////////////////////////////////////////////////////////////////////////// |
4062 | // DONE | ||
3922 | u16 MXL_ControlRead(struct dvb_frontend *fe, u16 controlNum, u32 *value) | 4063 | u16 MXL_ControlRead(struct dvb_frontend *fe, u16 controlNum, u32 *value) |
3923 | { | 4064 | { |
3924 | struct mxl5005s_state *state = fe->tuner_priv; | 4065 | struct mxl5005s_state *state = fe->tuner_priv; |
@@ -3990,6 +4131,7 @@ u16 MXL_ControlRead(struct dvb_frontend *fe, u16 controlNum, u32 *value) | |||
3990 | // -1 : Invalid control name // | 4131 | // -1 : Invalid control name // |
3991 | // // | 4132 | // // |
3992 | /////////////////////////////////////////////////////////////////////////////// | 4133 | /////////////////////////////////////////////////////////////////////////////// |
4134 | // DONE | ||
3993 | u16 MXL_ControlRegRead(struct dvb_frontend *fe, u16 controlNum, u8 *RegNum, int * count) | 4135 | u16 MXL_ControlRegRead(struct dvb_frontend *fe, u16 controlNum, u8 *RegNum, int * count) |
3994 | { | 4136 | { |
3995 | struct mxl5005s_state *state = fe->tuner_priv; | 4137 | struct mxl5005s_state *state = fe->tuner_priv; |
@@ -4095,6 +4237,7 @@ u16 MXL_ControlRegRead(struct dvb_frontend *fe, u16 controlNum, u8 *RegNum, int | |||
4095 | // NONE // | 4237 | // NONE // |
4096 | // // | 4238 | // // |
4097 | /////////////////////////////////////////////////////////////////////////////// | 4239 | /////////////////////////////////////////////////////////////////////////////// |
4240 | // DONE | ||
4098 | void MXL_RegWriteBit(struct dvb_frontend *fe, u8 address, u8 bit, u8 bitVal) | 4241 | void MXL_RegWriteBit(struct dvb_frontend *fe, u8 address, u8 bit, u8 bitVal) |
4099 | { | 4242 | { |
4100 | struct mxl5005s_state *state = fe->tuner_priv; | 4243 | struct mxl5005s_state *state = fe->tuner_priv; |
@@ -4142,6 +4285,7 @@ void MXL_RegWriteBit(struct dvb_frontend *fe, u8 address, u8 bit, u8 bitVal) | |||
4142 | // Computed value // | 4285 | // Computed value // |
4143 | // // | 4286 | // // |
4144 | /////////////////////////////////////////////////////////////////////////////// | 4287 | /////////////////////////////////////////////////////////////////////////////// |
4288 | // DONE | ||
4145 | u32 MXL_Ceiling(u32 value, u32 resolution) | 4289 | u32 MXL_Ceiling(u32 value, u32 resolution) |
4146 | { | 4290 | { |
4147 | return (value/resolution + (value % resolution > 0 ? 1 : 0)); | 4291 | return (value/resolution + (value % resolution > 0 ? 1 : 0)); |
@@ -4150,6 +4294,7 @@ u32 MXL_Ceiling(u32 value, u32 resolution) | |||
4150 | // | 4294 | // |
4151 | // Retrieve the Initialzation Registers | 4295 | // Retrieve the Initialzation Registers |
4152 | // | 4296 | // |
4297 | // DONE | ||
4153 | u16 MXL_GetInitRegister(struct dvb_frontend *fe, u8 * RegNum, u8 *RegVal, int *count) | 4298 | u16 MXL_GetInitRegister(struct dvb_frontend *fe, u8 * RegNum, u8 *RegVal, int *count) |
4154 | { | 4299 | { |
4155 | u16 status = 0; | 4300 | u16 status = 0; |
@@ -4172,6 +4317,7 @@ u16 MXL_GetInitRegister(struct dvb_frontend *fe, u8 * RegNum, u8 *RegVal, int *c | |||
4172 | return status; | 4317 | return status; |
4173 | } | 4318 | } |
4174 | 4319 | ||
4320 | // DONE | ||
4175 | u16 MXL_GetCHRegister(struct dvb_frontend *fe, u8 * RegNum, u8 *RegVal, int *count) | 4321 | u16 MXL_GetCHRegister(struct dvb_frontend *fe, u8 * RegNum, u8 *RegVal, int *count) |
4176 | { | 4322 | { |
4177 | u16 status = 0; | 4323 | u16 status = 0; |
@@ -4199,6 +4345,7 @@ u16 MXL_GetCHRegister(struct dvb_frontend *fe, u8 * RegNum, u8 *RegVal, int *cou | |||
4199 | return status; | 4345 | return status; |
4200 | } | 4346 | } |
4201 | 4347 | ||
4348 | // DONE | ||
4202 | u16 MXL_GetCHRegister_ZeroIF(struct dvb_frontend *fe, u8 * RegNum, u8 *RegVal, int *count) | 4349 | u16 MXL_GetCHRegister_ZeroIF(struct dvb_frontend *fe, u8 * RegNum, u8 *RegVal, int *count) |
4203 | { | 4350 | { |
4204 | u16 status = 0; | 4351 | u16 status = 0; |
@@ -4216,6 +4363,7 @@ u16 MXL_GetCHRegister_ZeroIF(struct dvb_frontend *fe, u8 * RegNum, u8 *RegVal, i | |||
4216 | return status; | 4363 | return status; |
4217 | } | 4364 | } |
4218 | 4365 | ||
4366 | // DONE | ||
4219 | u16 MXL_GetCHRegister_LowIF(struct dvb_frontend *fe, u8 * RegNum, u8 *RegVal, int *count) | 4367 | u16 MXL_GetCHRegister_LowIF(struct dvb_frontend *fe, u8 * RegNum, u8 *RegVal, int *count) |
4220 | { | 4368 | { |
4221 | u16 status = 0; | 4369 | u16 status = 0; |
@@ -4233,6 +4381,7 @@ u16 MXL_GetCHRegister_LowIF(struct dvb_frontend *fe, u8 * RegNum, u8 *RegVal, in | |||
4233 | return status; | 4381 | return status; |
4234 | } | 4382 | } |
4235 | 4383 | ||
4384 | // DONE | ||
4236 | u16 MXL_GetMasterControl(u8 *MasterReg, int state) | 4385 | u16 MXL_GetMasterControl(u8 *MasterReg, int state) |
4237 | { | 4386 | { |
4238 | if (state == 1) /* Load_Start */ | 4387 | if (state == 1) /* Load_Start */ |
@@ -4377,6 +4526,7 @@ u16 MXL_VCORange_Test(struct dvb_frontend *fe, int VCO_Range) | |||
4377 | return status; | 4526 | return status; |
4378 | } | 4527 | } |
4379 | 4528 | ||
4529 | // DONE | ||
4380 | u16 MXL_Hystersis_Test(struct dvb_frontend *fe, int Hystersis) | 4530 | u16 MXL_Hystersis_Test(struct dvb_frontend *fe, int Hystersis) |
4381 | { | 4531 | { |
4382 | struct mxl5005s_state *state = fe->tuner_priv; | 4532 | struct mxl5005s_state *state = fe->tuner_priv; |
@@ -4387,224 +4537,141 @@ u16 MXL_Hystersis_Test(struct dvb_frontend *fe, int Hystersis) | |||
4387 | 4537 | ||
4388 | return status; | 4538 | return status; |
4389 | } | 4539 | } |
4540 | |||
4390 | #endif | 4541 | #endif |
4391 | /* End: Reference driver code found in the Realtek driver that | ||
4392 | * is copyright MaxLinear */ | ||
4393 | 4542 | ||
4394 | /* ---------------------------------------------------------------- | 4543 | /* Linux driver related functions */ |
4395 | * Begin: Everything after here is new code to adapt the | 4544 | |
4396 | * proprietary Realtek driver into a Linux API tuner. | 4545 | |
4397 | * Copyright (C) 2008 Steven Toth <stoth@hauppauge.com> | 4546 | int mxl5005s_init(struct dvb_frontend *fe) |
4398 | */ | ||
4399 | static int mxl5005s_reset(struct dvb_frontend *fe) | ||
4400 | { | 4547 | { |
4401 | struct mxl5005s_state *state = fe->tuner_priv; | 4548 | int MxlModMode; |
4402 | int ret = 0; | 4549 | int MxlIfMode; |
4550 | unsigned long MxlBandwitdh; | ||
4551 | unsigned long MxlIfFreqHz; | ||
4552 | unsigned long MxlCrystalFreqHz; | ||
4553 | int MxlAgcMode; | ||
4554 | unsigned short MxlTop; | ||
4555 | unsigned short MxlIfOutputLoad; | ||
4556 | int MxlClockOut; | ||
4557 | int MxlDivOut; | ||
4558 | int MxlCapSel; | ||
4559 | int MxlRssiOnOff; | ||
4560 | unsigned char MxlStandard; | ||
4561 | unsigned char MxlTfType; | ||
4403 | 4562 | ||
4404 | u8 buf[2] = { 0xff, 0x00 }; | 4563 | /* Set MxL5005S parameters. */ |
4405 | struct i2c_msg msg = { .addr = state->config->i2c_address, .flags = 0, | 4564 | MxlModMode = MXL_DIGITAL_MODE; |
4406 | .buf = buf, .len = 2 }; | 4565 | MxlIfMode = MXL_ZERO_IF; |
4566 | // steve | ||
4567 | //MxlBandwitdh = MXL5005S_BANDWIDTH_8MHZ; | ||
4568 | //MxlIfFreqHz = IF_FREQ_4570000HZ; | ||
4569 | MxlBandwitdh = MXL5005S_BANDWIDTH_6MHZ; // config | ||
4570 | MxlIfFreqHz = IF_FREQ_5380000HZ; // config | ||
4571 | MxlCrystalFreqHz = CRYSTAL_FREQ_16000000HZ; // config | ||
4572 | MxlAgcMode = MXL_SINGLE_AGC; | ||
4573 | MxlTop = MXL5005S_TOP_25P2; | ||
4574 | MxlIfOutputLoad = MXL5005S_IF_OUTPUT_LOAD_200_OHM; | ||
4575 | MxlClockOut = MXL_CLOCK_OUT_DISABLE; | ||
4576 | MxlDivOut = MXL_DIV_OUT_4; | ||
4577 | MxlCapSel = MXL_CAP_SEL_ENABLE; | ||
4578 | MxlRssiOnOff = MXL_RSSI_ENABLE; // config | ||
4579 | MxlTfType = MXL_TF_C_H; // config | ||
4580 | |||
4581 | MxlStandard = MXL_ATSC; // config | ||
4582 | |||
4583 | // TODO: this is bad, it trashes other configs | ||
4584 | // Set MxL5005S extra module. | ||
4585 | //pExtra->AgcMasterByte = (MxlAgcMode == MXL_DUAL_AGC) ? 0x4 : 0x0; | ||
4407 | 4586 | ||
4408 | dprintk(2, "%s()\n", __func__); | 4587 | MXL5005_TunerConfig( |
4588 | fe, | ||
4589 | (unsigned char)MxlModMode, | ||
4590 | (unsigned char)MxlIfMode, | ||
4591 | MxlBandwitdh, | ||
4592 | MxlIfFreqHz, | ||
4593 | MxlCrystalFreqHz, | ||
4594 | (unsigned char)MxlAgcMode, | ||
4595 | MxlTop, | ||
4596 | MxlIfOutputLoad, | ||
4597 | (unsigned char)MxlClockOut, | ||
4598 | (unsigned char)MxlDivOut, | ||
4599 | (unsigned char)MxlCapSel, | ||
4600 | (unsigned char)MxlRssiOnOff, | ||
4601 | MxlStandard, MxlTfType); | ||
4602 | |||
4603 | return mxl5005s_init2(fe); | ||
4604 | } | ||
4409 | 4605 | ||
4410 | if (fe->ops.i2c_gate_ctrl) | 4606 | static int mxl5005s_set_params(struct dvb_frontend *fe, |
4411 | fe->ops.i2c_gate_ctrl(fe, 1); | 4607 | struct dvb_frontend_parameters *params) |
4608 | { | ||
4609 | u32 freq; | ||
4610 | u32 bw; | ||
4412 | 4611 | ||
4413 | if (i2c_transfer(state->i2c, &msg, 1) != 1) { | 4612 | if (fe->ops.info.type == FE_OFDM) |
4414 | printk(KERN_WARNING "mxl5005s I2C reset failed\n"); | 4613 | bw = params->u.ofdm.bandwidth; |
4415 | ret = -EREMOTEIO; | 4614 | else |
4416 | } | 4615 | bw = MXL5005S_BANDWIDTH_6MHZ; |
4417 | 4616 | ||
4418 | if (fe->ops.i2c_gate_ctrl) | 4617 | freq = params->frequency; /* Hz */ |
4419 | fe->ops.i2c_gate_ctrl(fe, 0); | 4618 | dprintk(1, "%s() freq=%d bw=%d\n", __func__, freq, bw); |
4420 | 4619 | ||
4421 | return ret; | 4620 | return mxl5005s_SetRfFreqHz(fe, freq); |
4422 | } | 4621 | } |
4423 | 4622 | ||
4424 | /* Write a single byte to a single reg, latch the value if required by | 4623 | static int mxl5005s_get_frequency(struct dvb_frontend *fe, u32 *frequency) |
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 | { | 4624 | { |
4429 | struct mxl5005s_state *state = fe->tuner_priv; | 4625 | struct mxl5005s_state *state = fe->tuner_priv; |
4430 | u8 buf[3] = { reg, val, MXL5005S_LATCH_BYTE }; | 4626 | dprintk(1, "%s()\n", __func__); |
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 | 4627 | ||
4437 | dprintk(2, "%s(reg = 0x%x val = 0x%x addr = 0x%x)\n", __func__, reg, val, msg.addr); | 4628 | *frequency = state->RF_IN; |
4438 | 4629 | ||
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; | 4630 | return 0; |
4444 | } | 4631 | } |
4445 | 4632 | ||
4446 | int mxl5005s_writeregs(struct dvb_frontend *fe, u8 *addrtable, u8 *datatable, u8 len) | 4633 | static int mxl5005s_get_bandwidth(struct dvb_frontend *fe, u32 *bandwidth) |
4447 | { | 4634 | { |
4448 | int ret = 0, i; | 4635 | struct mxl5005s_state *state = fe->tuner_priv; |
4449 | 4636 | dprintk(1, "%s()\n", __func__); | |
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 | 4637 | ||
4461 | if (fe->ops.i2c_gate_ctrl) | 4638 | *bandwidth = state->Chan_Bandwidth; |
4462 | fe->ops.i2c_gate_ctrl(fe, 0); | ||
4463 | 4639 | ||
4464 | return ret; | 4640 | return 0; |
4465 | } | 4641 | } |
4466 | 4642 | ||
4467 | 4643 | static int mxl5005s_get_status(struct dvb_frontend *fe, u32 *status) | |
4468 | int mxl5005s_init(struct dvb_frontend *fe) | ||
4469 | { | 4644 | { |
4470 | dprintk(1, "%s()\n", __func__); | 4645 | dprintk(1, "%s()\n", __func__); |
4471 | return mxl5005s_reconfigure(fe, MXL_QAM, MXL5005S_BANDWIDTH_6MHZ); | 4646 | |
4647 | *status = 0; | ||
4648 | // *status = TUNER_STATUS_LOCKED; | ||
4649 | |||
4650 | return 0; | ||
4472 | } | 4651 | } |
4473 | 4652 | ||
4474 | int mxl5005s_reconfigure(struct dvb_frontend *fe, u32 mod_type, u32 bandwidth) | 4653 | static int mxl5005s_init2(struct dvb_frontend *fe) |
4475 | { | 4654 | { |
4476 | struct mxl5005s_state *state = fe->tuner_priv; | 4655 | struct mxl5005s_state *state = fe->tuner_priv; |
4477 | |||
4478 | u8 AddrTable[MXL5005S_REG_WRITING_TABLE_LEN_MAX]; | 4656 | u8 AddrTable[MXL5005S_REG_WRITING_TABLE_LEN_MAX]; |
4479 | u8 ByteTable[MXL5005S_REG_WRITING_TABLE_LEN_MAX]; | 4657 | u8 ByteTable[MXL5005S_REG_WRITING_TABLE_LEN_MAX]; |
4480 | int TableLen; | 4658 | int TableLen; |
4481 | 4659 | ||
4482 | dprintk(1, "%s(type=%d, bw=%d)\n", __func__, mod_type, bandwidth); | 4660 | dprintk(1, "%s()\n", __func__); |
4483 | 4661 | ||
4484 | mxl5005s_reset(fe); | 4662 | /* Initialize MxL5005S tuner according to MxL5005S tuner example code. */ |
4485 | 4663 | ||
4486 | /* Tuner initialization stage 0 */ | 4664 | /* Tuner initialization stage 0 */ |
4487 | MXL_GetMasterControl(ByteTable, MC_SYNTH_RESET); | 4665 | MXL_GetMasterControl(ByteTable, MC_SYNTH_RESET); |
4488 | AddrTable[0] = MASTER_CONTROL_ADDR; | 4666 | AddrTable[0] = MASTER_CONTROL_ADDR; |
4489 | ByteTable[0] |= state->config->AgcMasterByte; | 4667 | ByteTable[0] |= state->config->AgcMasterByte; |
4490 | 4668 | ||
4491 | mxl5005s_writeregs(fe, AddrTable, ByteTable, 1); | 4669 | mxl5005s_SetRegsWithTable(fe, AddrTable, ByteTable, 1); |
4492 | |||
4493 | mxl5005s_AssignTunerMode(fe, mod_type, bandwidth); | ||
4494 | 4670 | ||
4495 | /* Tuner initialization stage 1 */ | 4671 | /* Tuner initialization stage 1 */ |
4496 | MXL_GetInitRegister(fe, AddrTable, ByteTable, &TableLen); | 4672 | MXL_GetInitRegister(fe, AddrTable, ByteTable, &TableLen); |
4497 | 4673 | ||
4498 | mxl5005s_writeregs(fe, AddrTable, ByteTable, TableLen); | 4674 | mxl5005s_SetRegsWithTable(fe, AddrTable, ByteTable, TableLen); |
4499 | |||
4500 | return 0; | ||
4501 | } | ||
4502 | |||
4503 | int mxl5005s_AssignTunerMode(struct dvb_frontend *fe, u32 mod_type, u32 bandwidth) | ||
4504 | { | ||
4505 | struct mxl5005s_state *state = fe->tuner_priv; | ||
4506 | struct mxl5005s_config *c = state->config; | ||
4507 | |||
4508 | InitTunerControls(fe); | ||
4509 | |||
4510 | /* Set MxL5005S parameters. */ | ||
4511 | MXL5005_TunerConfig( | ||
4512 | fe, | ||
4513 | c->mod_mode, | ||
4514 | c->if_mode, | ||
4515 | bandwidth, | ||
4516 | c->if_freq, | ||
4517 | c->xtal_freq, | ||
4518 | c->agc_mode, | ||
4519 | c->top, | ||
4520 | c->output_load, | ||
4521 | c->clock_out, | ||
4522 | c->div_out, | ||
4523 | c->cap_select, | ||
4524 | c->rssi_enable, | ||
4525 | mod_type, | ||
4526 | c->tracking_filter); | ||
4527 | |||
4528 | return 0; | ||
4529 | } | ||
4530 | |||
4531 | static int mxl5005s_set_params(struct dvb_frontend *fe, | ||
4532 | struct dvb_frontend_parameters *params) | ||
4533 | { | ||
4534 | struct mxl5005s_state *state = fe->tuner_priv; | ||
4535 | u32 req_mode, req_bw = 0; | ||
4536 | int ret; | ||
4537 | |||
4538 | dprintk(1, "%s()\n", __func__); | ||
4539 | |||
4540 | if (fe->ops.info.type == FE_ATSC) { | ||
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 | } | ||
4577 | |||
4578 | state->current_mode = req_mode; | ||
4579 | ret = mxl5005s_reconfigure(fe, req_mode, req_bw); | ||
4580 | |||
4581 | } else | ||
4582 | ret = 0; | ||
4583 | |||
4584 | if (ret == 0) { | ||
4585 | dprintk(1, "%s() freq=%d\n", __func__, params->frequency); | ||
4586 | ret = mxl5005s_SetRfFreqHz(fe, params->frequency); | ||
4587 | } | ||
4588 | |||
4589 | return ret; | ||
4590 | } | ||
4591 | |||
4592 | static int mxl5005s_get_frequency(struct dvb_frontend *fe, u32 *frequency) | ||
4593 | { | ||
4594 | struct mxl5005s_state *state = fe->tuner_priv; | ||
4595 | dprintk(1, "%s()\n", __func__); | ||
4596 | |||
4597 | *frequency = state->RF_IN; | ||
4598 | |||
4599 | return 0; | ||
4600 | } | ||
4601 | |||
4602 | static int mxl5005s_get_bandwidth(struct dvb_frontend *fe, u32 *bandwidth) | ||
4603 | { | ||
4604 | struct mxl5005s_state *state = fe->tuner_priv; | ||
4605 | dprintk(1, "%s()\n", __func__); | ||
4606 | |||
4607 | *bandwidth = state->Chan_Bandwidth; | ||
4608 | 4675 | ||
4609 | return 0; | 4676 | return 0; |
4610 | } | 4677 | } |
@@ -4631,6 +4698,7 @@ static const struct dvb_tuner_ops mxl5005s_tuner_ops = { | |||
4631 | .set_params = mxl5005s_set_params, | 4698 | .set_params = mxl5005s_set_params, |
4632 | .get_frequency = mxl5005s_get_frequency, | 4699 | .get_frequency = mxl5005s_get_frequency, |
4633 | .get_bandwidth = mxl5005s_get_bandwidth, | 4700 | .get_bandwidth = mxl5005s_get_bandwidth, |
4701 | .get_status = mxl5005s_get_status | ||
4634 | }; | 4702 | }; |
4635 | 4703 | ||
4636 | struct dvb_frontend *mxl5005s_attach(struct dvb_frontend *fe, | 4704 | struct dvb_frontend *mxl5005s_attach(struct dvb_frontend *fe, |
@@ -4647,7 +4715,6 @@ struct dvb_frontend *mxl5005s_attach(struct dvb_frontend *fe, | |||
4647 | state->frontend = fe; | 4715 | state->frontend = fe; |
4648 | state->config = config; | 4716 | state->config = config; |
4649 | state->i2c = i2c; | 4717 | state->i2c = i2c; |
4650 | state->current_mode = MXL_QAM; | ||
4651 | 4718 | ||
4652 | printk(KERN_INFO "MXL5005S: Attached at address 0x%02x\n", config->i2c_address); | 4719 | printk(KERN_INFO "MXL5005S: Attached at address 0x%02x\n", config->i2c_address); |
4653 | 4720 | ||
@@ -4659,5 +4726,8 @@ struct dvb_frontend *mxl5005s_attach(struct dvb_frontend *fe, | |||
4659 | EXPORT_SYMBOL(mxl5005s_attach); | 4726 | EXPORT_SYMBOL(mxl5005s_attach); |
4660 | 4727 | ||
4661 | MODULE_DESCRIPTION("MaxLinear MXL5005S silicon tuner driver"); | 4728 | MODULE_DESCRIPTION("MaxLinear MXL5005S silicon tuner driver"); |
4729 | MODULE_AUTHOR("Jan Hoogenraad"); | ||
4730 | MODULE_AUTHOR("Barnaby Shearer"); | ||
4731 | MODULE_AUTHOR("Andy Hasper"); | ||
4662 | MODULE_AUTHOR("Steven Toth"); | 4732 | MODULE_AUTHOR("Steven Toth"); |
4663 | MODULE_LICENSE("GPL"); | 4733 | MODULE_LICENSE("GPL"); |
diff --git a/drivers/media/common/tuners/mxl5005s.h b/drivers/media/common/tuners/mxl5005s.h index 7658401f3cdd..7d0727d44536 100644 --- a/drivers/media/common/tuners/mxl5005s.h +++ b/drivers/media/common/tuners/mxl5005s.h | |||
@@ -1,119 +1,66 @@ | |||
1 | /* | 1 | /* |
2 | MaxLinear MXL5005S VSB/QAM/DVBT tuner driver | 2 | * For the Realtek RTL chip RTL2831U |
3 | * Realtek Release Date: 2008-03-14, ver 080314 | ||
4 | * Realtek version RTL2831 Linux driver version 080314 | ||
5 | * ver 080314 | ||
6 | * | ||
7 | * for linux kernel version 2.6.21.4 - 2.6.22-14 | ||
8 | * support MXL5005s and MT2060 tuners (support tuner auto-detecting) | ||
9 | * support two IR types -- RC5 and NEC | ||
10 | * | ||
11 | * Known boards with Realtek RTL chip RTL2821U | ||
12 | * Freecom USB stick 14aa:0160 (version 4) | ||
13 | * Conceptronic CTVDIGRCU | ||
14 | * | ||
15 | * Copyright (c) 2008 Realtek | ||
16 | * Copyright (c) 2008 Jan Hoogenraad, Barnaby Shearer, Andy Hasper | ||
17 | * This code is placed under the terms of the GNU General Public License | ||
18 | * | ||
19 | * Released by Realtek under GPLv2. | ||
20 | * Thanks to Realtek for a lot of support we received ! | ||
21 | * | ||
22 | * Revision: 080314 - original version | ||
23 | */ | ||
3 | 24 | ||
4 | Copyright (C) 2008 MaxLinear | ||
5 | Copyright (C) 2008 Steven Toth <stoth@hauppauge.com> | ||
6 | 25 | ||
7 | This program is free software; you can redistribute it and/or modify | 26 | #ifndef __MXL5005S_H |
8 | it under the terms of the GNU General Public License as published by | 27 | #define __MXL5005S_H |
9 | the Free Software Foundation; either version 2 of the License, or | ||
10 | (at your option) any later version. | ||
11 | |||
12 | This program is distributed in the hope that it will be useful, | ||
13 | but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
14 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
15 | GNU General Public License for more details. | ||
16 | 28 | ||
17 | You should have received a copy of the GNU General Public License | 29 | #include <linux/dvb/frontend.h> |
18 | along with this program; if not, write to the Free Software | ||
19 | Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. | ||
20 | 30 | ||
21 | */ | 31 | /* IF frequency */ |
32 | enum IF_FREQ_HZ | ||
33 | { | ||
34 | IF_FREQ_4570000HZ = 4570000, ///< IF frequency = 4.57 MHz | ||
35 | IF_FREQ_4571429HZ = 4571429, ///< IF frequency = 4.571 MHz | ||
36 | IF_FREQ_5380000HZ = 5380000, ///< IF frequency = 5.38 MHz | ||
37 | IF_FREQ_36000000HZ = 36000000, ///< IF frequency = 36.000 MHz | ||
38 | IF_FREQ_36125000HZ = 36125000, ///< IF frequency = 36.125 MHz | ||
39 | IF_FREQ_36166667HZ = 36166667, ///< IF frequency = 36.167 MHz | ||
40 | IF_FREQ_44000000HZ = 44000000, ///< IF frequency = 44.000 MHz | ||
41 | }; | ||
22 | 42 | ||
23 | #ifndef __MXL5005S_H | 43 | /* Crystal frequency */ |
24 | #define __MXL5005S_H | 44 | enum CRYSTAL_FREQ_HZ |
45 | { | ||
46 | CRYSTAL_FREQ_4000000HZ = 4000000, ///< Crystal frequency = 4.0 MHz | ||
47 | CRYSTAL_FREQ_16000000HZ = 16000000, ///< Crystal frequency = 16.0 MHz | ||
48 | CRYSTAL_FREQ_25000000HZ = 25000000, ///< Crystal frequency = 25.0 MHz | ||
49 | CRYSTAL_FREQ_28800000HZ = 28800000, ///< Crystal frequency = 28.8 MHz | ||
50 | }; | ||
25 | 51 | ||
26 | struct mxl5005s_config | 52 | struct mxl5005s_config |
27 | { | 53 | { |
28 | /* 7 bit i2c address */ | ||
29 | u8 i2c_address; | 54 | u8 i2c_address; |
30 | 55 | ||
31 | #define IF_FREQ_4570000HZ 4570000 | ||
32 | #define IF_FREQ_4571429HZ 4571429 | ||
33 | #define IF_FREQ_5380000HZ 5380000 | ||
34 | #define IF_FREQ_36000000HZ 36000000 | ||
35 | #define IF_FREQ_36125000HZ 36125000 | ||
36 | #define IF_FREQ_36166667HZ 36166667 | ||
37 | #define IF_FREQ_44000000HZ 44000000 | ||
38 | u32 if_freq; | ||
39 | |||
40 | #define CRYSTAL_FREQ_4000000HZ 4000000 | ||
41 | #define CRYSTAL_FREQ_16000000HZ 16000000 | ||
42 | #define CRYSTAL_FREQ_25000000HZ 25000000 | ||
43 | #define CRYSTAL_FREQ_28800000HZ 28800000 | ||
44 | u32 xtal_freq; | ||
45 | |||
46 | #define MXL_DUAL_AGC 0 | ||
47 | #define MXL_SINGLE_AGC 1 | ||
48 | u8 agc_mode; | ||
49 | |||
50 | #define MXL_TF_DEFAULT 0 | ||
51 | #define MXL_TF_OFF 1 | ||
52 | #define MXL_TF_C 2 | ||
53 | #define MXL_TF_C_H 3 | ||
54 | #define MXL_TF_D 4 | ||
55 | #define MXL_TF_D_L 5 | ||
56 | #define MXL_TF_E 6 | ||
57 | #define MXL_TF_F 7 | ||
58 | #define MXL_TF_E_2 8 | ||
59 | #define MXL_TF_E_NA 9 | ||
60 | #define MXL_TF_G 10 | ||
61 | u8 tracking_filter; | ||
62 | |||
63 | #define MXL_RSSI_DISABLE 0 | ||
64 | #define MXL_RSSI_ENABLE 1 | ||
65 | u8 rssi_enable; | ||
66 | |||
67 | #define MXL_CAP_SEL_DISABLE 0 | ||
68 | #define MXL_CAP_SEL_ENABLE 1 | ||
69 | u8 cap_select; | ||
70 | |||
71 | #define MXL_DIV_OUT_1 0 | ||
72 | #define MXL_DIV_OUT_4 1 | ||
73 | u8 div_out; | ||
74 | |||
75 | #define MXL_CLOCK_OUT_DISABLE 0 | ||
76 | #define MXL_CLOCK_OUT_ENABLE 1 | ||
77 | u8 clock_out; | ||
78 | |||
79 | #define MXL5005S_IF_OUTPUT_LOAD_200_OHM 200 | ||
80 | #define MXL5005S_IF_OUTPUT_LOAD_300_OHM 300 | ||
81 | u32 output_load; | ||
82 | |||
83 | #define MXL5005S_TOP_5P5 55 | ||
84 | #define MXL5005S_TOP_7P2 72 | ||
85 | #define MXL5005S_TOP_9P2 92 | ||
86 | #define MXL5005S_TOP_11P0 110 | ||
87 | #define MXL5005S_TOP_12P9 129 | ||
88 | #define MXL5005S_TOP_14P7 147 | ||
89 | #define MXL5005S_TOP_16P8 168 | ||
90 | #define MXL5005S_TOP_19P4 194 | ||
91 | #define MXL5005S_TOP_21P2 212 | ||
92 | #define MXL5005S_TOP_23P2 232 | ||
93 | #define MXL5005S_TOP_25P2 252 | ||
94 | #define MXL5005S_TOP_27P1 271 | ||
95 | #define MXL5005S_TOP_29P2 292 | ||
96 | #define MXL5005S_TOP_31P7 317 | ||
97 | #define MXL5005S_TOP_34P9 349 | ||
98 | u32 top; | ||
99 | |||
100 | #define MXL_ANALOG_MODE 0 | ||
101 | #define MXL_DIGITAL_MODE 1 | ||
102 | u8 mod_mode; | ||
103 | |||
104 | #define MXL_ZERO_IF 0 | ||
105 | #define MXL_LOW_IF 1 | ||
106 | u8 if_mode; | ||
107 | |||
108 | /* Stuff I don't know what to do with */ | 56 | /* Stuff I don't know what to do with */ |
109 | u8 AgcMasterByte; | 57 | u8 AgcMasterByte; |
110 | }; | 58 | }; |
111 | 59 | ||
112 | #if defined(CONFIG_DVB_TUNER_MXL5005S) || \ | 60 | #if defined(CONFIG_MEDIA_TUNER_MXL5005S) || (defined(CONFIG_MEDIA_TUNER_MXL5005S_MODULE) && defined(MODULE)) |
113 | (defined(CONFIG_DVB_TUNER_MXL5005S_MODULE) && defined(MODULE)) | ||
114 | extern struct dvb_frontend *mxl5005s_attach(struct dvb_frontend *fe, | 61 | extern struct dvb_frontend *mxl5005s_attach(struct dvb_frontend *fe, |
115 | struct i2c_adapter *i2c, | 62 | struct i2c_adapter *i2c, |
116 | struct mxl5005s_config *config) | 63 | struct mxl5005s_config *config); |
117 | #else | 64 | #else |
118 | static inline struct dvb_frontend *mxl5005s_attach(struct dvb_frontend *fe, | 65 | static inline struct dvb_frontend *mxl5005s_attach(struct dvb_frontend *fe, |
119 | struct i2c_adapter *i2c, | 66 | struct i2c_adapter *i2c, |