diff options
| author | Michael Krufky <mkrufky@linuxtv.org> | 2005-08-08 12:22:43 -0400 |
|---|---|---|
| committer | Linus Torvalds <torvalds@g5.osdl.org> | 2005-08-08 14:49:57 -0400 |
| commit | 1963c907b21e140082d081b1c8f8c2154593c7d7 (patch) | |
| tree | 71410cd2ad9f4bdab2f2466a6621ba2930f16b5f /drivers | |
| parent | 66aea23ff84ca81bfaeaf7d63e248b873f5c2616 (diff) | |
[PATCH] dvb: lgdt330x frontend: some bug fixes & add lgdt3303 support
- Structural changes within lgdt330x driver, framework now supports
both chips... tested OK on lgdt3302 and lgdt3303.
- Add LG/TUA6034 dvb_pll_desc for ATSC with LG TDVS-H062F & DViCO FusionHDTV5.
- Fixed LGDT330X signal strength: For now, always set it to 0.
- Corrected LGDT330X boundary condition error in read_snr: dB calculation.
Signed-off-by: Mac Michaels <wmichaels1@earthlink.net>
Signed-off-by: Michael Krufky <mkrufky@m1k.net>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
Diffstat (limited to 'drivers')
| -rw-r--r-- | drivers/media/dvb/frontends/dvb-pll.c | 16 | ||||
| -rw-r--r-- | drivers/media/dvb/frontends/dvb-pll.h | 1 | ||||
| -rw-r--r-- | drivers/media/dvb/frontends/lgdt330x.c | 549 | ||||
| -rw-r--r-- | drivers/media/dvb/frontends/lgdt330x.h | 16 | ||||
| -rw-r--r-- | drivers/media/dvb/frontends/lgdt330x_priv.h | 8 | ||||
| -rw-r--r-- | drivers/media/video/cx88/cx88-dvb.c | 26 |
6 files changed, 443 insertions, 173 deletions
diff --git a/drivers/media/dvb/frontends/dvb-pll.c b/drivers/media/dvb/frontends/dvb-pll.c index 5264310c070e..536c35d969b7 100644 --- a/drivers/media/dvb/frontends/dvb-pll.c +++ b/drivers/media/dvb/frontends/dvb-pll.c | |||
| @@ -225,6 +225,22 @@ struct dvb_pll_desc dvb_pll_tua6034 = { | |||
| 225 | }; | 225 | }; |
| 226 | EXPORT_SYMBOL(dvb_pll_tua6034); | 226 | EXPORT_SYMBOL(dvb_pll_tua6034); |
| 227 | 227 | ||
| 228 | /* Infineon TUA6034 | ||
| 229 | * used in LG Innotek TDVS-H062F | ||
| 230 | */ | ||
| 231 | struct dvb_pll_desc dvb_pll_tdvs_tua6034 = { | ||
| 232 | .name = "LG/Infineon TUA6034", | ||
| 233 | .min = 54000000, | ||
| 234 | .max = 863000000, | ||
| 235 | .count = 3, | ||
| 236 | .entries = { | ||
| 237 | { 160000000, 44000000, 62500, 0xce, 0x01 }, | ||
| 238 | { 455000000, 44000000, 62500, 0xce, 0x02 }, | ||
| 239 | { 999999999, 44000000, 62500, 0xce, 0x04 }, | ||
| 240 | }, | ||
| 241 | }; | ||
| 242 | EXPORT_SYMBOL(dvb_pll_tdvs_tua6034); | ||
| 243 | |||
| 228 | /* Philips FMD1216ME | 244 | /* Philips FMD1216ME |
| 229 | * used in Medion Hybrid PCMCIA card and USB Box | 245 | * used in Medion Hybrid PCMCIA card and USB Box |
| 230 | */ | 246 | */ |
diff --git a/drivers/media/dvb/frontends/dvb-pll.h b/drivers/media/dvb/frontends/dvb-pll.h index cb794759d89e..205b2d1a8852 100644 --- a/drivers/media/dvb/frontends/dvb-pll.h +++ b/drivers/media/dvb/frontends/dvb-pll.h | |||
| @@ -31,6 +31,7 @@ extern struct dvb_pll_desc dvb_pll_unknown_1; | |||
| 31 | extern struct dvb_pll_desc dvb_pll_tua6010xs; | 31 | extern struct dvb_pll_desc dvb_pll_tua6010xs; |
| 32 | extern struct dvb_pll_desc dvb_pll_env57h1xd5; | 32 | extern struct dvb_pll_desc dvb_pll_env57h1xd5; |
| 33 | extern struct dvb_pll_desc dvb_pll_tua6034; | 33 | extern struct dvb_pll_desc dvb_pll_tua6034; |
| 34 | extern struct dvb_pll_desc dvb_pll_tdvs_tua6034; | ||
| 34 | extern struct dvb_pll_desc dvb_pll_tda665x; | 35 | extern struct dvb_pll_desc dvb_pll_tda665x; |
| 35 | extern struct dvb_pll_desc dvb_pll_fmd1216me; | 36 | extern struct dvb_pll_desc dvb_pll_fmd1216me; |
| 36 | extern struct dvb_pll_desc dvb_pll_tded4; | 37 | extern struct dvb_pll_desc dvb_pll_tded4; |
diff --git a/drivers/media/dvb/frontends/lgdt330x.c b/drivers/media/dvb/frontends/lgdt330x.c index e94dee50eecd..c48e7c11d708 100644 --- a/drivers/media/dvb/frontends/lgdt330x.c +++ b/drivers/media/dvb/frontends/lgdt330x.c | |||
| @@ -1,11 +1,8 @@ | |||
| 1 | /* | 1 | /* |
| 2 | * Support for LGDT3302 & LGDT3303 (DViCO FusionHDTV Gold) - VSB/QAM | 2 | * Support for LGDT3302 and LGDT3303 - VSB/QAM |
| 3 | * | 3 | * |
| 4 | * Copyright (C) 2005 Wilson Michaels <wilsonmichaels@earthlink.net> | 4 | * Copyright (C) 2005 Wilson Michaels <wilsonmichaels@earthlink.net> |
| 5 | * | 5 | * |
| 6 | * Based on code from Kirk Lapray <kirk_lapray@bigfoot.com> | ||
| 7 | * Copyright (C) 2005 | ||
| 8 | * | ||
| 9 | * This program is free software; you can redistribute it and/or modify | 6 | * This program is free software; you can redistribute it and/or modify |
| 10 | * it under the terms of the GNU General Public License as published by | 7 | * it under the terms of the GNU General Public License as published by |
| 11 | * the Free Software Foundation; either version 2 of the License, or | 8 | * the Free Software Foundation; either version 2 of the License, or |
| @@ -25,11 +22,13 @@ | |||
| 25 | /* | 22 | /* |
| 26 | * NOTES ABOUT THIS DRIVER | 23 | * NOTES ABOUT THIS DRIVER |
| 27 | * | 24 | * |
| 28 | * This driver supports DViCO FusionHDTV Gold under Linux. | 25 | * This Linux driver supports: |
| 26 | * DViCO FusionHDTV 3 Gold-Q | ||
| 27 | * DViCO FusionHDTV 3 Gold-T | ||
| 28 | * DViCO FusionHDTV 5 Gold | ||
| 29 | * | 29 | * |
| 30 | * TODO: | 30 | * TODO: |
| 31 | * BER and signal strength always return 0. | 31 | * signal strength always returns 0. |
| 32 | * Include support for LGDT3303 | ||
| 33 | * | 32 | * |
| 34 | */ | 33 | */ |
| 35 | 34 | ||
| @@ -41,7 +40,6 @@ | |||
| 41 | #include <asm/byteorder.h> | 40 | #include <asm/byteorder.h> |
| 42 | 41 | ||
| 43 | #include "dvb_frontend.h" | 42 | #include "dvb_frontend.h" |
| 44 | #include "dvb-pll.h" | ||
| 45 | #include "lgdt330x_priv.h" | 43 | #include "lgdt330x_priv.h" |
| 46 | #include "lgdt330x.h" | 44 | #include "lgdt330x.h" |
| 47 | 45 | ||
| @@ -70,55 +68,37 @@ struct lgdt330x_state | |||
| 70 | u32 current_frequency; | 68 | u32 current_frequency; |
| 71 | }; | 69 | }; |
| 72 | 70 | ||
| 73 | static int i2c_writebytes (struct lgdt330x_state* state, | 71 | static int i2c_write_demod_bytes (struct lgdt330x_state* state, |
| 74 | u8 addr, /* demod_address or pll_address */ | ||
| 75 | u8 *buf, /* data bytes to send */ | 72 | u8 *buf, /* data bytes to send */ |
| 76 | int len /* number of bytes to send */ ) | 73 | int len /* number of bytes to send */ ) |
| 77 | { | 74 | { |
| 78 | u8 tmp[] = { buf[0], buf[1] }; | ||
| 79 | struct i2c_msg msg = | 75 | struct i2c_msg msg = |
| 80 | { .addr = addr, .flags = 0, .buf = tmp, .len = 2 }; | 76 | { .addr = state->config->demod_address, |
| 81 | int err; | 77 | .flags = 0, |
| 78 | .buf = buf, | ||
| 79 | .len = 2 }; | ||
| 82 | int i; | 80 | int i; |
| 81 | int err; | ||
| 83 | 82 | ||
| 84 | for (i=1; i<len; i++) { | 83 | for (i=0; i<len-1; i+=2){ |
| 85 | tmp[1] = buf[i]; | ||
| 86 | if ((err = i2c_transfer(state->i2c, &msg, 1)) != 1) { | 84 | if ((err = i2c_transfer(state->i2c, &msg, 1)) != 1) { |
| 87 | printk(KERN_WARNING "lgdt330x: %s error (addr %02x <- %02x, err == %i)\n", __FUNCTION__, addr, buf[0], err); | 85 | printk(KERN_WARNING "lgdt330x: %s error (addr %02x <- %02x, err = %i)\n", __FUNCTION__, msg.buf[0], msg.buf[1], err); |
| 88 | if (err < 0) | 86 | if (err < 0) |
| 89 | return err; | 87 | return err; |
| 90 | else | 88 | else |
| 91 | return -EREMOTEIO; | 89 | return -EREMOTEIO; |
| 92 | } | 90 | } |
| 93 | tmp[0]++; | 91 | msg.buf += 2; |
| 94 | } | ||
| 95 | return 0; | ||
| 96 | } | ||
| 97 | |||
| 98 | #if 0 | ||
| 99 | static int i2c_readbytes (struct lgdt330x_state* state, | ||
| 100 | u8 addr, /* demod_address or pll_address */ | ||
| 101 | u8 *buf, /* holds data bytes read */ | ||
| 102 | int len /* number of bytes to read */ ) | ||
| 103 | { | ||
| 104 | struct i2c_msg msg = | ||
| 105 | { .addr = addr, .flags = I2C_M_RD, .buf = buf, .len = len }; | ||
| 106 | int err; | ||
| 107 | |||
| 108 | if ((err = i2c_transfer(state->i2c, &msg, 1)) != 1) { | ||
| 109 | printk(KERN_WARNING "lgdt330x: %s error (addr %02x, err == %i)\n", __FUNCTION__, addr, err); | ||
| 110 | return -EREMOTEIO; | ||
| 111 | } | 92 | } |
| 112 | return 0; | 93 | return 0; |
| 113 | } | 94 | } |
| 114 | #endif | ||
| 115 | 95 | ||
| 116 | /* | 96 | /* |
| 117 | * This routine writes the register (reg) to the demod bus | 97 | * This routine writes the register (reg) to the demod bus |
| 118 | * then reads the data returned for (len) bytes. | 98 | * then reads the data returned for (len) bytes. |
| 119 | */ | 99 | */ |
| 120 | 100 | ||
| 121 | static u8 i2c_selectreadbytes (struct lgdt330x_state* state, | 101 | static u8 i2c_read_demod_bytes (struct lgdt330x_state* state, |
| 122 | enum I2C_REG reg, u8* buf, int len) | 102 | enum I2C_REG reg, u8* buf, int len) |
| 123 | { | 103 | { |
| 124 | u8 wr [] = { reg }; | 104 | u8 wr [] = { reg }; |
| @@ -139,7 +119,7 @@ static u8 i2c_selectreadbytes (struct lgdt330x_state* state, | |||
| 139 | } | 119 | } |
| 140 | 120 | ||
| 141 | /* Software reset */ | 121 | /* Software reset */ |
| 142 | int lgdt330x_SwReset(struct lgdt330x_state* state) | 122 | static int lgdt3302_SwReset(struct lgdt330x_state* state) |
| 143 | { | 123 | { |
| 144 | u8 ret; | 124 | u8 ret; |
| 145 | u8 reset[] = { | 125 | u8 reset[] = { |
| @@ -148,23 +128,83 @@ int lgdt330x_SwReset(struct lgdt330x_state* state) | |||
| 148 | * bits 5-0 are 1 to mask interrupts */ | 128 | * bits 5-0 are 1 to mask interrupts */ |
| 149 | }; | 129 | }; |
| 150 | 130 | ||
| 151 | ret = i2c_writebytes(state, | 131 | ret = i2c_write_demod_bytes(state, |
| 152 | state->config->demod_address, | ||
| 153 | reset, sizeof(reset)); | 132 | reset, sizeof(reset)); |
| 154 | if (ret == 0) { | 133 | if (ret == 0) { |
| 155 | /* spec says reset takes 100 ns why wait */ | 134 | |
| 156 | /* mdelay(100); */ /* keep low for 100mS */ | 135 | /* force reset high (inactive) and unmask interrupts */ |
| 157 | reset[1] = 0x7f; /* force reset high (inactive) | 136 | reset[1] = 0x7f; |
| 158 | * and unmask interrupts */ | 137 | ret = i2c_write_demod_bytes(state, |
| 159 | ret = i2c_writebytes(state, | ||
| 160 | state->config->demod_address, | ||
| 161 | reset, sizeof(reset)); | 138 | reset, sizeof(reset)); |
| 162 | } | 139 | } |
| 163 | /* Spec does not indicate a need for this either */ | ||
| 164 | /*mdelay(5); */ /* wait 5 msec before doing more */ | ||
| 165 | return ret; | 140 | return ret; |
| 166 | } | 141 | } |
| 167 | 142 | ||
| 143 | static int lgdt3303_SwReset(struct lgdt330x_state* state) | ||
| 144 | { | ||
| 145 | u8 ret; | ||
| 146 | u8 reset[] = { | ||
| 147 | 0x02, | ||
| 148 | 0x00 /* bit 0 is active low software reset */ | ||
| 149 | }; | ||
| 150 | |||
| 151 | ret = i2c_write_demod_bytes(state, | ||
| 152 | reset, sizeof(reset)); | ||
| 153 | if (ret == 0) { | ||
| 154 | |||
| 155 | /* force reset high (inactive) */ | ||
| 156 | reset[1] = 0x01; | ||
| 157 | ret = i2c_write_demod_bytes(state, | ||
| 158 | reset, sizeof(reset)); | ||
| 159 | } | ||
| 160 | return ret; | ||
| 161 | } | ||
| 162 | |||
| 163 | static int lgdt330x_SwReset(struct lgdt330x_state* state) | ||
| 164 | { | ||
| 165 | switch (state->config->demod_chip) { | ||
| 166 | case LGDT3302: | ||
| 167 | return lgdt3302_SwReset(state); | ||
| 168 | case LGDT3303: | ||
| 169 | return lgdt3303_SwReset(state); | ||
| 170 | default: | ||
| 171 | return -ENODEV; | ||
| 172 | } | ||
| 173 | } | ||
| 174 | |||
| 175 | #ifdef MUTE_TDA9887 | ||
| 176 | static int i2c_write_ntsc_demod (struct lgdt330x_state* state, u8 buf[2]) | ||
| 177 | { | ||
| 178 | struct i2c_msg msg = | ||
| 179 | { .addr = 0x43, | ||
| 180 | .flags = 0, | ||
| 181 | .buf = buf, | ||
| 182 | .len = 2 }; | ||
| 183 | int err; | ||
| 184 | |||
| 185 | if ((err = i2c_transfer(state->i2c, &msg, 1)) != 1) { | ||
| 186 | printk(KERN_WARNING "lgdt330x: %s error (addr %02x <- %02x, err = %i)\n", __FUNCTION__, msg.buf[0], msg.buf[1], err); | ||
| 187 | if (err < 0) | ||
| 188 | return err; | ||
| 189 | else | ||
| 190 | return -EREMOTEIO; | ||
| 191 | } | ||
| 192 | return 0; | ||
| 193 | } | ||
| 194 | |||
| 195 | static void fiddle_with_ntsc_if_demod(struct lgdt330x_state* state) | ||
| 196 | { | ||
| 197 | // Experimental code | ||
| 198 | u8 buf0[] = {0x00, 0x20}; | ||
| 199 | u8 buf1[] = {0x01, 0x00}; | ||
| 200 | u8 buf2[] = {0x02, 0x00}; | ||
| 201 | |||
| 202 | i2c_write_ntsc_demod(state, buf0); | ||
| 203 | i2c_write_ntsc_demod(state, buf1); | ||
| 204 | i2c_write_ntsc_demod(state, buf2); | ||
| 205 | } | ||
| 206 | #endif | ||
| 207 | |||
| 168 | static int lgdt330x_init(struct dvb_frontend* fe) | 208 | static int lgdt330x_init(struct dvb_frontend* fe) |
| 169 | { | 209 | { |
| 170 | /* Hardware reset is done using gpio[0] of cx23880x chip. | 210 | /* Hardware reset is done using gpio[0] of cx23880x chip. |
| @@ -173,22 +213,101 @@ static int lgdt330x_init(struct dvb_frontend* fe) | |||
| 173 | * Maybe there needs to be a callable function in cx88-core or | 213 | * Maybe there needs to be a callable function in cx88-core or |
| 174 | * the caller of this function needs to do it. */ | 214 | * the caller of this function needs to do it. */ |
| 175 | 215 | ||
| 176 | dprintk("%s entered\n", __FUNCTION__); | 216 | /* |
| 177 | return lgdt330x_SwReset((struct lgdt330x_state*) fe->demodulator_priv); | 217 | * Array of byte pairs <address, value> |
| 218 | * to initialize each different chip | ||
| 219 | */ | ||
| 220 | static u8 lgdt3302_init_data[] = { | ||
| 221 | /* Use 50MHz parameter values from spec sheet since xtal is 50 */ | ||
| 222 | /* Change the value of NCOCTFV[25:0] of carrier | ||
| 223 | recovery center frequency register */ | ||
| 224 | VSB_CARRIER_FREQ0, 0x00, | ||
| 225 | VSB_CARRIER_FREQ1, 0x87, | ||
| 226 | VSB_CARRIER_FREQ2, 0x8e, | ||
| 227 | VSB_CARRIER_FREQ3, 0x01, | ||
| 228 | /* Change the TPCLK pin polarity | ||
| 229 | data is valid on falling clock */ | ||
| 230 | DEMUX_CONTROL, 0xfb, | ||
| 231 | /* Change the value of IFBW[11:0] of | ||
| 232 | AGC IF/RF loop filter bandwidth register */ | ||
| 233 | AGC_RF_BANDWIDTH0, 0x40, | ||
| 234 | AGC_RF_BANDWIDTH1, 0x93, | ||
| 235 | AGC_RF_BANDWIDTH2, 0x00, | ||
| 236 | /* Change the value of bit 6, 'nINAGCBY' and | ||
| 237 | 'NSSEL[1:0] of ACG function control register 2 */ | ||
| 238 | AGC_FUNC_CTRL2, 0xc6, | ||
| 239 | /* Change the value of bit 6 'RFFIX' | ||
| 240 | of AGC function control register 3 */ | ||
| 241 | AGC_FUNC_CTRL3, 0x40, | ||
| 242 | /* Set the value of 'INLVTHD' register 0x2a/0x2c | ||
| 243 | to 0x7fe */ | ||
| 244 | AGC_DELAY0, 0x07, | ||
| 245 | AGC_DELAY2, 0xfe, | ||
| 246 | /* Change the value of IAGCBW[15:8] | ||
| 247 | of inner AGC loop filter bandwith */ | ||
| 248 | AGC_LOOP_BANDWIDTH0, 0x08, | ||
| 249 | AGC_LOOP_BANDWIDTH1, 0x9a | ||
| 250 | }; | ||
| 251 | |||
| 252 | static u8 lgdt3303_init_data[] = { | ||
| 253 | 0x4c, 0x14 | ||
| 254 | }; | ||
| 255 | |||
| 256 | struct lgdt330x_state* state = fe->demodulator_priv; | ||
| 257 | char *chip_name; | ||
| 258 | int err; | ||
| 259 | |||
| 260 | switch (state->config->demod_chip) { | ||
| 261 | case LGDT3302: | ||
| 262 | chip_name = "LGDT3302"; | ||
| 263 | err = i2c_write_demod_bytes(state, lgdt3302_init_data, | ||
| 264 | sizeof(lgdt3302_init_data)); | ||
| 265 | break; | ||
| 266 | case LGDT3303: | ||
| 267 | chip_name = "LGDT3303"; | ||
| 268 | err = i2c_write_demod_bytes(state, lgdt3303_init_data, | ||
| 269 | sizeof(lgdt3303_init_data)); | ||
| 270 | #ifdef MUTE_TDA9887 | ||
| 271 | fiddle_with_ntsc_if_demod(state); | ||
| 272 | #endif | ||
| 273 | break; | ||
| 274 | default: | ||
| 275 | chip_name = "undefined"; | ||
| 276 | printk (KERN_WARNING "Only LGDT3302 and LGDT3303 are supported chips.\n"); | ||
| 277 | err = -ENODEV; | ||
| 278 | } | ||
| 279 | dprintk("%s entered as %s\n", __FUNCTION__, chip_name); | ||
| 280 | if (err < 0) | ||
| 281 | return err; | ||
| 282 | return lgdt330x_SwReset(state); | ||
| 178 | } | 283 | } |
| 179 | 284 | ||
| 180 | static int lgdt330x_read_ber(struct dvb_frontend* fe, u32* ber) | 285 | static int lgdt330x_read_ber(struct dvb_frontend* fe, u32* ber) |
| 181 | { | 286 | { |
| 182 | *ber = 0; /* Dummy out for now */ | 287 | *ber = 0; /* Not supplied by the demod chips */ |
| 183 | return 0; | 288 | return 0; |
| 184 | } | 289 | } |
| 185 | 290 | ||
| 186 | static int lgdt330x_read_ucblocks(struct dvb_frontend* fe, u32* ucblocks) | 291 | static int lgdt330x_read_ucblocks(struct dvb_frontend* fe, u32* ucblocks) |
| 187 | { | 292 | { |
| 188 | struct lgdt330x_state* state = (struct lgdt330x_state*) fe->demodulator_priv; | 293 | struct lgdt330x_state* state = fe->demodulator_priv; |
| 294 | int err; | ||
| 189 | u8 buf[2]; | 295 | u8 buf[2]; |
| 190 | 296 | ||
| 191 | i2c_selectreadbytes(state, PACKET_ERR_COUNTER1, buf, sizeof(buf)); | 297 | switch (state->config->demod_chip) { |
| 298 | case LGDT3302: | ||
| 299 | err = i2c_read_demod_bytes(state, LGDT3302_PACKET_ERR_COUNTER1, | ||
| 300 | buf, sizeof(buf)); | ||
| 301 | break; | ||
| 302 | case LGDT3303: | ||
| 303 | err = i2c_read_demod_bytes(state, LGDT3303_PACKET_ERR_COUNTER1, | ||
| 304 | buf, sizeof(buf)); | ||
| 305 | break; | ||
| 306 | default: | ||
| 307 | printk(KERN_WARNING | ||
| 308 | "Only LGDT3302 and LGDT3303 are supported chips.\n"); | ||
| 309 | err = -ENODEV; | ||
| 310 | } | ||
| 192 | 311 | ||
| 193 | *ucblocks = (buf[0] << 8) | buf[1]; | 312 | *ucblocks = (buf[0] << 8) | buf[1]; |
| 194 | return 0; | 313 | return 0; |
| @@ -197,123 +316,113 @@ static int lgdt330x_read_ucblocks(struct dvb_frontend* fe, u32* ucblocks) | |||
| 197 | static int lgdt330x_set_parameters(struct dvb_frontend* fe, | 316 | static int lgdt330x_set_parameters(struct dvb_frontend* fe, |
| 198 | struct dvb_frontend_parameters *param) | 317 | struct dvb_frontend_parameters *param) |
| 199 | { | 318 | { |
| 200 | struct lgdt330x_state* state = | 319 | /* |
| 201 | (struct lgdt330x_state*) fe->demodulator_priv; | 320 | * Array of byte pairs <address, value> |
| 321 | * to initialize 8VSB for lgdt3303 chip 50 MHz IF | ||
| 322 | */ | ||
| 323 | static u8 lgdt3303_8vsb_44_data[] = { | ||
| 324 | 0x04, 0x00, | ||
| 325 | 0x0d, 0x40, | ||
| 326 | 0x0e, 0x87, | ||
| 327 | 0x0f, 0x8e, | ||
| 328 | 0x10, 0x01, | ||
| 329 | 0x47, 0x8b }; | ||
| 330 | |||
| 331 | /* | ||
| 332 | * Array of byte pairs <address, value> | ||
| 333 | * to initialize QAM for lgdt3303 chip | ||
| 334 | */ | ||
| 335 | static u8 lgdt3303_qam_data[] = { | ||
| 336 | 0x04, 0x00, | ||
| 337 | 0x0d, 0x00, | ||
| 338 | 0x0e, 0x00, | ||
| 339 | 0x0f, 0x00, | ||
| 340 | 0x10, 0x00, | ||
| 341 | 0x51, 0x63, | ||
| 342 | 0x47, 0x66, | ||
| 343 | 0x48, 0x66, | ||
| 344 | 0x4d, 0x1a, | ||
| 345 | 0x49, 0x08, | ||
| 346 | 0x4a, 0x9b }; | ||
| 347 | |||
| 348 | struct lgdt330x_state* state = fe->demodulator_priv; | ||
| 202 | 349 | ||
| 203 | /* Use 50MHz parameter values from spec sheet since xtal is 50 */ | ||
| 204 | static u8 top_ctrl_cfg[] = { TOP_CONTROL, 0x03 }; | 350 | static u8 top_ctrl_cfg[] = { TOP_CONTROL, 0x03 }; |
| 205 | static u8 vsb_freq_cfg[] = { VSB_CARRIER_FREQ0, 0x00, 0x87, 0x8e, 0x01 }; | ||
| 206 | static u8 demux_ctrl_cfg[] = { DEMUX_CONTROL, 0xfb }; | ||
| 207 | static u8 agc_rf_cfg[] = { AGC_RF_BANDWIDTH0, 0x40, 0x93, 0x00 }; | ||
| 208 | static u8 agc_ctrl_cfg[] = { AGC_FUNC_CTRL2, 0xc6, 0x40 }; | ||
| 209 | static u8 agc_delay_cfg[] = { AGC_DELAY0, 0x07, 0x00, 0xfe }; | ||
| 210 | static u8 agc_loop_cfg[] = { AGC_LOOP_BANDWIDTH0, 0x08, 0x9a }; | ||
| 211 | 351 | ||
| 352 | int err; | ||
| 212 | /* Change only if we are actually changing the modulation */ | 353 | /* Change only if we are actually changing the modulation */ |
| 213 | if (state->current_modulation != param->u.vsb.modulation) { | 354 | if (state->current_modulation != param->u.vsb.modulation) { |
| 214 | switch(param->u.vsb.modulation) { | 355 | switch(param->u.vsb.modulation) { |
| 215 | case VSB_8: | 356 | case VSB_8: |
| 216 | dprintk("%s: VSB_8 MODE\n", __FUNCTION__); | 357 | dprintk("%s: VSB_8 MODE\n", __FUNCTION__); |
| 217 | 358 | ||
| 218 | /* Select VSB mode and serial MPEG interface */ | 359 | /* Select VSB mode */ |
| 219 | top_ctrl_cfg[1] = 0x07; | 360 | top_ctrl_cfg[1] = 0x03; |
| 220 | 361 | ||
| 221 | /* Select ANT connector if supported by card */ | 362 | /* Select ANT connector if supported by card */ |
| 222 | if (state->config->pll_rf_set) | 363 | if (state->config->pll_rf_set) |
| 223 | state->config->pll_rf_set(fe, 1); | 364 | state->config->pll_rf_set(fe, 1); |
| 365 | |||
| 366 | if (state->config->demod_chip == LGDT3303) { | ||
| 367 | err = i2c_write_demod_bytes(state, lgdt3303_8vsb_44_data, | ||
| 368 | sizeof(lgdt3303_8vsb_44_data)); | ||
| 369 | } | ||
| 224 | break; | 370 | break; |
| 225 | 371 | ||
| 226 | case QAM_64: | 372 | case QAM_64: |
| 227 | dprintk("%s: QAM_64 MODE\n", __FUNCTION__); | 373 | dprintk("%s: QAM_64 MODE\n", __FUNCTION__); |
| 228 | 374 | ||
| 229 | /* Select QAM_64 mode and serial MPEG interface */ | 375 | /* Select QAM_64 mode */ |
| 230 | top_ctrl_cfg[1] = 0x04; | 376 | top_ctrl_cfg[1] = 0x00; |
| 231 | 377 | ||
| 232 | /* Select CABLE connector if supported by card */ | 378 | /* Select CABLE connector if supported by card */ |
| 233 | if (state->config->pll_rf_set) | 379 | if (state->config->pll_rf_set) |
| 234 | state->config->pll_rf_set(fe, 0); | 380 | state->config->pll_rf_set(fe, 0); |
| 381 | |||
| 382 | if (state->config->demod_chip == LGDT3303) { | ||
| 383 | err = i2c_write_demod_bytes(state, lgdt3303_qam_data, | ||
| 384 | sizeof(lgdt3303_qam_data)); | ||
| 385 | } | ||
| 235 | break; | 386 | break; |
| 236 | 387 | ||
| 237 | case QAM_256: | 388 | case QAM_256: |
| 238 | dprintk("%s: QAM_256 MODE\n", __FUNCTION__); | 389 | dprintk("%s: QAM_256 MODE\n", __FUNCTION__); |
| 239 | 390 | ||
| 240 | /* Select QAM_256 mode and serial MPEG interface */ | 391 | /* Select QAM_256 mode */ |
| 241 | top_ctrl_cfg[1] = 0x05; | 392 | top_ctrl_cfg[1] = 0x01; |
| 242 | 393 | ||
| 243 | /* Select CABLE connector if supported by card */ | 394 | /* Select CABLE connector if supported by card */ |
| 244 | if (state->config->pll_rf_set) | 395 | if (state->config->pll_rf_set) |
| 245 | state->config->pll_rf_set(fe, 0); | 396 | state->config->pll_rf_set(fe, 0); |
| 397 | |||
| 398 | if (state->config->demod_chip == LGDT3303) { | ||
| 399 | err = i2c_write_demod_bytes(state, lgdt3303_qam_data, | ||
| 400 | sizeof(lgdt3303_qam_data)); | ||
| 401 | } | ||
| 246 | break; | 402 | break; |
| 247 | default: | 403 | default: |
| 248 | printk(KERN_WARNING "lgdt330x: %s: Modulation type(%d) UNSUPPORTED\n", __FUNCTION__, param->u.vsb.modulation); | 404 | printk(KERN_WARNING "lgdt330x: %s: Modulation type(%d) UNSUPPORTED\n", __FUNCTION__, param->u.vsb.modulation); |
| 249 | return -1; | 405 | return -1; |
| 250 | } | 406 | } |
| 251 | /* Initializations common to all modes */ | 407 | /* |
| 408 | * select serial or parallel MPEG harware interface | ||
| 409 | * Serial: 0x04 for LGDT3302 or 0x40 for LGDT3303 | ||
| 410 | * Parallel: 0x00 | ||
| 411 | */ | ||
| 412 | top_ctrl_cfg[1] |= state->config->serial_mpeg; | ||
| 252 | 413 | ||
| 253 | /* Select the requested mode */ | 414 | /* Select the requested mode */ |
| 254 | i2c_writebytes(state, state->config->demod_address, | 415 | i2c_write_demod_bytes(state, top_ctrl_cfg, |
| 255 | top_ctrl_cfg, sizeof(top_ctrl_cfg)); | 416 | sizeof(top_ctrl_cfg)); |
| 256 | |||
| 257 | /* Change the value of IFBW[11:0] | ||
| 258 | of AGC IF/RF loop filter bandwidth register */ | ||
| 259 | i2c_writebytes(state, state->config->demod_address, | ||
| 260 | agc_rf_cfg, sizeof(agc_rf_cfg)); | ||
| 261 | |||
| 262 | /* Change the value of bit 6, 'nINAGCBY' and | ||
| 263 | 'NSSEL[1:0] of ACG function control register 2 */ | ||
| 264 | /* Change the value of bit 6 'RFFIX' | ||
| 265 | of AGC function control register 3 */ | ||
| 266 | i2c_writebytes(state, state->config->demod_address, | ||
| 267 | agc_ctrl_cfg, sizeof(agc_ctrl_cfg)); | ||
| 268 | |||
| 269 | /* Change the TPCLK pin polarity | ||
| 270 | data is valid on falling clock */ | ||
| 271 | i2c_writebytes(state, state->config->demod_address, | ||
| 272 | demux_ctrl_cfg, sizeof(demux_ctrl_cfg)); | ||
| 273 | |||
| 274 | /* Change the value of NCOCTFV[25:0] of carrier | ||
| 275 | recovery center frequency register */ | ||
| 276 | i2c_writebytes(state, state->config->demod_address, | ||
| 277 | vsb_freq_cfg, sizeof(vsb_freq_cfg)); | ||
| 278 | |||
| 279 | /* Set the value of 'INLVTHD' register 0x2a/0x2c to 0x7fe */ | ||
| 280 | i2c_writebytes(state, state->config->demod_address, | ||
| 281 | agc_delay_cfg, sizeof(agc_delay_cfg)); | ||
| 282 | |||
| 283 | /* Change the value of IAGCBW[15:8] | ||
| 284 | of inner AGC loop filter bandwith */ | ||
| 285 | i2c_writebytes(state, state->config->demod_address, | ||
| 286 | agc_loop_cfg, sizeof(agc_loop_cfg)); | ||
| 287 | |||
| 288 | state->config->set_ts_params(fe, 0); | 417 | state->config->set_ts_params(fe, 0); |
| 289 | state->current_modulation = param->u.vsb.modulation; | 418 | state->current_modulation = param->u.vsb.modulation; |
| 290 | } | 419 | } |
| 291 | 420 | ||
| 292 | /* Change only if we are actually changing the channel */ | 421 | /* Change only if we are actually changing the channel */ |
| 293 | if (state->current_frequency != param->frequency) { | 422 | if (state->current_frequency != param->frequency) { |
| 294 | u8 buf[5]; | 423 | /* Tune to the new frequency */ |
| 295 | struct i2c_msg msg = { .flags = 0, .buf = &buf[1], .len = 4 }; | 424 | state->config->pll_set(fe, param); |
| 296 | int err; | 425 | /* Keep track of the new frequency */ |
| 297 | |||
| 298 | state->config->pll_set(fe, param, buf); | ||
| 299 | msg.addr = buf[0]; | ||
| 300 | |||
| 301 | dprintk("%s: tuner at 0x%02x bytes: 0x%02x 0x%02x " | ||
| 302 | "0x%02x 0x%02x\n", __FUNCTION__, | ||
| 303 | buf[0],buf[1],buf[2],buf[3],buf[4]); | ||
| 304 | if ((err = i2c_transfer(state->i2c, &msg, 1)) != 1) { | ||
| 305 | printk(KERN_WARNING "lgdt330x: %s error (addr %02x <- %02x, err = %i)\n", __FUNCTION__, buf[0], buf[1], err); | ||
| 306 | if (err < 0) | ||
| 307 | return err; | ||
| 308 | else | ||
| 309 | return -EREMOTEIO; | ||
| 310 | } | ||
| 311 | #if 0 | ||
| 312 | /* Check the status of the tuner pll */ | ||
| 313 | i2c_readbytes(state, buf[0], &buf[1], 1); | ||
| 314 | dprintk("%s: tuner status byte = 0x%02x\n", __FUNCTION__, buf[1]); | ||
| 315 | #endif | ||
| 316 | /* Update current frequency */ | ||
| 317 | state->current_frequency = param->frequency; | 426 | state->current_frequency = param->frequency; |
| 318 | } | 427 | } |
| 319 | lgdt330x_SwReset(state); | 428 | lgdt330x_SwReset(state); |
| @@ -328,21 +437,15 @@ static int lgdt330x_get_frontend(struct dvb_frontend* fe, | |||
| 328 | return 0; | 437 | return 0; |
| 329 | } | 438 | } |
| 330 | 439 | ||
| 331 | static int lgdt330x_read_status(struct dvb_frontend* fe, fe_status_t* status) | 440 | static int lgdt3302_read_status(struct dvb_frontend* fe, fe_status_t* status) |
| 332 | { | 441 | { |
| 333 | struct lgdt330x_state* state = (struct lgdt330x_state*) fe->demodulator_priv; | 442 | struct lgdt330x_state* state = fe->demodulator_priv; |
| 334 | u8 buf[3]; | 443 | u8 buf[3]; |
| 335 | 444 | ||
| 336 | *status = 0; /* Reset status result */ | 445 | *status = 0; /* Reset status result */ |
| 337 | 446 | ||
| 338 | /* | ||
| 339 | * You must set the Mask bits to 1 in the IRQ_MASK in order | ||
| 340 | * to see that status bit in the IRQ_STATUS register. | ||
| 341 | * This is done in SwReset(); | ||
| 342 | */ | ||
| 343 | |||
| 344 | /* AGC status register */ | 447 | /* AGC status register */ |
| 345 | i2c_selectreadbytes(state, AGC_STATUS, buf, 1); | 448 | i2c_read_demod_bytes(state, AGC_STATUS, buf, 1); |
| 346 | dprintk("%s: AGC_STATUS = 0x%02x\n", __FUNCTION__, buf[0]); | 449 | dprintk("%s: AGC_STATUS = 0x%02x\n", __FUNCTION__, buf[0]); |
| 347 | if ((buf[0] & 0x0c) == 0x8){ | 450 | if ((buf[0] & 0x0c) == 0x8){ |
| 348 | /* Test signal does not exist flag */ | 451 | /* Test signal does not exist flag */ |
| @@ -353,16 +456,15 @@ static int lgdt330x_read_status(struct dvb_frontend* fe, fe_status_t* status) | |||
| 353 | return 0; | 456 | return 0; |
| 354 | } | 457 | } |
| 355 | 458 | ||
| 459 | /* | ||
| 460 | * You must set the Mask bits to 1 in the IRQ_MASK in order | ||
| 461 | * to see that status bit in the IRQ_STATUS register. | ||
| 462 | * This is done in SwReset(); | ||
| 463 | */ | ||
| 356 | /* signal status */ | 464 | /* signal status */ |
| 357 | i2c_selectreadbytes(state, TOP_CONTROL, buf, sizeof(buf)); | 465 | i2c_read_demod_bytes(state, TOP_CONTROL, buf, sizeof(buf)); |
| 358 | dprintk("%s: TOP_CONTROL = 0x%02x, IRO_MASK = 0x%02x, IRQ_STATUS = 0x%02x\n", __FUNCTION__, buf[0], buf[1], buf[2]); | 466 | dprintk("%s: TOP_CONTROL = 0x%02x, IRO_MASK = 0x%02x, IRQ_STATUS = 0x%02x\n", __FUNCTION__, buf[0], buf[1], buf[2]); |
| 359 | 467 | ||
| 360 | #if 0 | ||
| 361 | /* Alternative method to check for a signal */ | ||
| 362 | /* using the SNR good/bad interrupts. */ | ||
| 363 | if ((buf[2] & 0x30) == 0x10) | ||
| 364 | *status |= FE_HAS_SIGNAL; | ||
| 365 | #endif | ||
| 366 | 468 | ||
| 367 | /* sync status */ | 469 | /* sync status */ |
| 368 | if ((buf[2] & 0x03) == 0x01) { | 470 | if ((buf[2] & 0x03) == 0x01) { |
| @@ -376,7 +478,7 @@ static int lgdt330x_read_status(struct dvb_frontend* fe, fe_status_t* status) | |||
| 376 | } | 478 | } |
| 377 | 479 | ||
| 378 | /* Carrier Recovery Lock Status Register */ | 480 | /* Carrier Recovery Lock Status Register */ |
| 379 | i2c_selectreadbytes(state, CARRIER_LOCK, buf, 1); | 481 | i2c_read_demod_bytes(state, CARRIER_LOCK, buf, 1); |
| 380 | dprintk("%s: CARRIER_LOCK = 0x%02x\n", __FUNCTION__, buf[0]); | 482 | dprintk("%s: CARRIER_LOCK = 0x%02x\n", __FUNCTION__, buf[0]); |
| 381 | switch (state->current_modulation) { | 483 | switch (state->current_modulation) { |
| 382 | case QAM_256: | 484 | case QAM_256: |
| @@ -396,13 +498,75 @@ static int lgdt330x_read_status(struct dvb_frontend* fe, fe_status_t* status) | |||
| 396 | return 0; | 498 | return 0; |
| 397 | } | 499 | } |
| 398 | 500 | ||
| 501 | static int lgdt3303_read_status(struct dvb_frontend* fe, fe_status_t* status) | ||
| 502 | { | ||
| 503 | struct lgdt330x_state* state = fe->demodulator_priv; | ||
| 504 | int err; | ||
| 505 | u8 buf[3]; | ||
| 506 | |||
| 507 | *status = 0; /* Reset status result */ | ||
| 508 | |||
| 509 | /* lgdt3303 AGC status register */ | ||
| 510 | err = i2c_read_demod_bytes(state, 0x58, buf, 1); | ||
| 511 | if (err < 0) | ||
| 512 | return err; | ||
| 513 | |||
| 514 | dprintk("%s: AGC_STATUS = 0x%02x\n", __FUNCTION__, buf[0]); | ||
| 515 | if ((buf[0] & 0x21) == 0x01){ | ||
| 516 | /* Test input signal does not exist flag */ | ||
| 517 | /* as well as the AGC lock flag. */ | ||
| 518 | *status |= FE_HAS_SIGNAL; | ||
| 519 | } else { | ||
| 520 | /* Without a signal all other status bits are meaningless */ | ||
| 521 | return 0; | ||
| 522 | } | ||
| 523 | |||
| 524 | /* Carrier Recovery Lock Status Register */ | ||
| 525 | i2c_read_demod_bytes(state, CARRIER_LOCK, buf, 1); | ||
| 526 | dprintk("%s: CARRIER_LOCK = 0x%02x\n", __FUNCTION__, buf[0]); | ||
| 527 | switch (state->current_modulation) { | ||
| 528 | case QAM_256: | ||
| 529 | case QAM_64: | ||
| 530 | /* Need to undestand why there are 3 lock levels here */ | ||
| 531 | if ((buf[0] & 0x07) == 0x07) | ||
| 532 | *status |= FE_HAS_CARRIER; | ||
| 533 | else | ||
| 534 | break; | ||
| 535 | i2c_read_demod_bytes(state, 0x8a, buf, 1); | ||
| 536 | if ((buf[0] & 0x04) == 0x04) | ||
| 537 | *status |= FE_HAS_SYNC; | ||
| 538 | if ((buf[0] & 0x01) == 0x01) | ||
| 539 | *status |= FE_HAS_LOCK; | ||
| 540 | if ((buf[0] & 0x08) == 0x08) | ||
| 541 | *status |= FE_HAS_VITERBI; | ||
| 542 | break; | ||
| 543 | case VSB_8: | ||
| 544 | if ((buf[0] & 0x80) == 0x80) | ||
| 545 | *status |= FE_HAS_CARRIER; | ||
| 546 | else | ||
| 547 | break; | ||
| 548 | i2c_read_demod_bytes(state, 0x38, buf, 1); | ||
| 549 | if ((buf[0] & 0x02) == 0x00) | ||
| 550 | *status |= FE_HAS_SYNC; | ||
| 551 | if ((buf[0] & 0x01) == 0x01) { | ||
| 552 | *status |= FE_HAS_LOCK; | ||
| 553 | *status |= FE_HAS_VITERBI; | ||
| 554 | } | ||
| 555 | break; | ||
| 556 | default: | ||
| 557 | printk("KERN_WARNING lgdt330x: %s: Modulation set to unsupported value\n", __FUNCTION__); | ||
| 558 | } | ||
| 559 | return 0; | ||
| 560 | } | ||
| 561 | |||
| 399 | static int lgdt330x_read_signal_strength(struct dvb_frontend* fe, u16* strength) | 562 | static int lgdt330x_read_signal_strength(struct dvb_frontend* fe, u16* strength) |
| 400 | { | 563 | { |
| 401 | /* not directly available. */ | 564 | /* not directly available. */ |
| 565 | *strength = 0; | ||
| 402 | return 0; | 566 | return 0; |
| 403 | } | 567 | } |
| 404 | 568 | ||
| 405 | static int lgdt330x_read_snr(struct dvb_frontend* fe, u16* snr) | 569 | static int lgdt3302_read_snr(struct dvb_frontend* fe, u16* snr) |
| 406 | { | 570 | { |
| 407 | #ifdef SNR_IN_DB | 571 | #ifdef SNR_IN_DB |
| 408 | /* | 572 | /* |
| @@ -451,7 +615,7 @@ static int lgdt330x_read_snr(struct dvb_frontend* fe, u16* snr) | |||
| 451 | 91, 115, 144, 182, 229, 288, 362, 456, 574, 722, | 615 | 91, 115, 144, 182, 229, 288, 362, 456, 574, 722, |
| 452 | 909, 1144, 1440, 1813, 2282, 2873, 3617, 4553, 5732, 7216, | 616 | 909, 1144, 1440, 1813, 2282, 2873, 3617, 4553, 5732, 7216, |
| 453 | 9084, 11436, 14396, 18124, 22817, 28724, 36161, 45524, 57312, 72151, | 617 | 9084, 11436, 14396, 18124, 22817, 28724, 36161, 45524, 57312, 72151, |
| 454 | 90833, 114351, 143960, 181235, 228161, 0x040000 | 618 | 90833, 114351, 143960, 181235, 228161, 0x080000 |
| 455 | }; | 619 | }; |
| 456 | 620 | ||
| 457 | static u8 buf[5];/* read data buffer */ | 621 | static u8 buf[5];/* read data buffer */ |
| @@ -459,8 +623,8 @@ static int lgdt330x_read_snr(struct dvb_frontend* fe, u16* snr) | |||
| 459 | static u32 snr_db; /* index into SNR_EQ[] */ | 623 | static u32 snr_db; /* index into SNR_EQ[] */ |
| 460 | struct lgdt330x_state* state = (struct lgdt330x_state*) fe->demodulator_priv; | 624 | struct lgdt330x_state* state = (struct lgdt330x_state*) fe->demodulator_priv; |
| 461 | 625 | ||
| 462 | /* read both equalizer and pase tracker noise data */ | 626 | /* read both equalizer and phase tracker noise data */ |
| 463 | i2c_selectreadbytes(state, EQPH_ERR0, buf, sizeof(buf)); | 627 | i2c_read_demod_bytes(state, EQPH_ERR0, buf, sizeof(buf)); |
| 464 | 628 | ||
| 465 | if (state->current_modulation == VSB_8) { | 629 | if (state->current_modulation == VSB_8) { |
| 466 | /* Equalizer Mean-Square Error Register for VSB */ | 630 | /* Equalizer Mean-Square Error Register for VSB */ |
| @@ -496,19 +660,20 @@ static int lgdt330x_read_snr(struct dvb_frontend* fe, u16* snr) | |||
| 496 | struct lgdt330x_state* state = (struct lgdt330x_state*) fe->demodulator_priv; | 660 | struct lgdt330x_state* state = (struct lgdt330x_state*) fe->demodulator_priv; |
| 497 | 661 | ||
| 498 | /* read both equalizer and pase tracker noise data */ | 662 | /* read both equalizer and pase tracker noise data */ |
| 499 | i2c_selectreadbytes(state, EQPH_ERR0, buf, sizeof(buf)); | 663 | i2c_read_demod_bytes(state, EQPH_ERR0, buf, sizeof(buf)); |
| 500 | 664 | ||
| 501 | if (state->current_modulation == VSB_8) { | 665 | if (state->current_modulation == VSB_8) { |
| 502 | /* Equalizer Mean-Square Error Register for VSB */ | 666 | /* Phase Tracker Mean-Square Error Register for VSB */ |
| 503 | noise = ((buf[0] & 7) << 16) | (buf[1] << 8) | buf[2]; | ||
| 504 | } else { | ||
| 505 | /* Phase Tracker Mean-Square Error Register for QAM */ | ||
| 506 | noise = ((buf[0] & 7<<3) << 13) | (buf[3] << 8) | buf[4]; | 667 | noise = ((buf[0] & 7<<3) << 13) | (buf[3] << 8) | buf[4]; |
| 668 | } else { | ||
| 669 | |||
| 670 | /* Carrier Recovery Mean-Square Error for QAM */ | ||
| 671 | i2c_read_demod_bytes(state, 0x1a, buf, 2); | ||
| 672 | noise = ((buf[0] & 3) << 8) | buf[1]; | ||
| 507 | } | 673 | } |
| 508 | 674 | ||
| 509 | /* Small values for noise mean signal is better so invert noise */ | 675 | /* Small values for noise mean signal is better so invert noise */ |
| 510 | /* Noise is 19 bit value so discard 3 LSB*/ | 676 | *snr = ~noise; |
| 511 | *snr = ~noise>>3; | ||
| 512 | #endif | 677 | #endif |
| 513 | 678 | ||
| 514 | dprintk("%s: noise = 0x%05x, snr = %idb\n",__FUNCTION__, noise, *snr); | 679 | dprintk("%s: noise = 0x%05x, snr = %idb\n",__FUNCTION__, noise, *snr); |
| @@ -516,6 +681,32 @@ static int lgdt330x_read_snr(struct dvb_frontend* fe, u16* snr) | |||
| 516 | return 0; | 681 | return 0; |
| 517 | } | 682 | } |
| 518 | 683 | ||
| 684 | static int lgdt3303_read_snr(struct dvb_frontend* fe, u16* snr) | ||
| 685 | { | ||
| 686 | /* Return the raw noise value */ | ||
| 687 | static u8 buf[5];/* read data buffer */ | ||
| 688 | static u32 noise; /* noise value */ | ||
| 689 | struct lgdt330x_state* state = (struct lgdt330x_state*) fe->demodulator_priv; | ||
| 690 | |||
| 691 | if (state->current_modulation == VSB_8) { | ||
| 692 | |||
| 693 | /* Phase Tracker Mean-Square Error Register for VSB */ | ||
| 694 | noise = ((buf[0] & 7) << 16) | (buf[3] << 8) | buf[4]; | ||
| 695 | } else { | ||
| 696 | |||
| 697 | /* Carrier Recovery Mean-Square Error for QAM */ | ||
| 698 | i2c_read_demod_bytes(state, 0x1a, buf, 2); | ||
| 699 | noise = (buf[0] << 8) | buf[1]; | ||
| 700 | } | ||
| 701 | |||
| 702 | /* Small values for noise mean signal is better so invert noise */ | ||
| 703 | *snr = ~noise; | ||
| 704 | |||
| 705 | dprintk("%s: noise = 0x%05x, snr = %idb\n",__FUNCTION__, noise, *snr); | ||
| 706 | |||
| 707 | return 0; | ||
| 708 | } | ||
| 709 | |||
| 519 | static int lgdt330x_get_tune_settings(struct dvb_frontend* fe, struct dvb_frontend_tune_settings* fe_tune_settings) | 710 | static int lgdt330x_get_tune_settings(struct dvb_frontend* fe, struct dvb_frontend_tune_settings* fe_tune_settings) |
| 520 | { | 711 | { |
| 521 | /* I have no idea about this - it may not be needed */ | 712 | /* I have no idea about this - it may not be needed */ |
| @@ -531,7 +722,8 @@ static void lgdt330x_release(struct dvb_frontend* fe) | |||
| 531 | kfree(state); | 722 | kfree(state); |
| 532 | } | 723 | } |
| 533 | 724 | ||
| 534 | static struct dvb_frontend_ops lgdt330x_ops; | 725 | static struct dvb_frontend_ops lgdt3302_ops; |
| 726 | static struct dvb_frontend_ops lgdt3303_ops; | ||
| 535 | 727 | ||
| 536 | struct dvb_frontend* lgdt330x_attach(const struct lgdt330x_config* config, | 728 | struct dvb_frontend* lgdt330x_attach(const struct lgdt330x_config* config, |
| 537 | struct i2c_adapter* i2c) | 729 | struct i2c_adapter* i2c) |
| @@ -548,9 +740,19 @@ struct dvb_frontend* lgdt330x_attach(const struct lgdt330x_config* config, | |||
| 548 | /* Setup the state */ | 740 | /* Setup the state */ |
| 549 | state->config = config; | 741 | state->config = config; |
| 550 | state->i2c = i2c; | 742 | state->i2c = i2c; |
| 551 | memcpy(&state->ops, &lgdt330x_ops, sizeof(struct dvb_frontend_ops)); | 743 | switch (config->demod_chip) { |
| 744 | case LGDT3302: | ||
| 745 | memcpy(&state->ops, &lgdt3302_ops, sizeof(struct dvb_frontend_ops)); | ||
| 746 | break; | ||
| 747 | case LGDT3303: | ||
| 748 | memcpy(&state->ops, &lgdt3303_ops, sizeof(struct dvb_frontend_ops)); | ||
| 749 | break; | ||
| 750 | default: | ||
| 751 | goto error; | ||
| 752 | } | ||
| 753 | |||
| 552 | /* Verify communication with demod chip */ | 754 | /* Verify communication with demod chip */ |
| 553 | if (i2c_selectreadbytes(state, 2, buf, 1)) | 755 | if (i2c_read_demod_bytes(state, 2, buf, 1)) |
| 554 | goto error; | 756 | goto error; |
| 555 | 757 | ||
| 556 | state->current_frequency = -1; | 758 | state->current_frequency = -1; |
| @@ -568,9 +770,33 @@ error: | |||
| 568 | return NULL; | 770 | return NULL; |
| 569 | } | 771 | } |
| 570 | 772 | ||
| 571 | static struct dvb_frontend_ops lgdt330x_ops = { | 773 | static struct dvb_frontend_ops lgdt3302_ops = { |
| 774 | .info = { | ||
| 775 | .name= "LG Electronics LGDT3302/LGDT3303 VSB/QAM Frontend", | ||
| 776 | .type = FE_ATSC, | ||
| 777 | .frequency_min= 54000000, | ||
| 778 | .frequency_max= 858000000, | ||
| 779 | .frequency_stepsize= 62500, | ||
| 780 | /* Symbol rate is for all VSB modes need to check QAM */ | ||
| 781 | .symbol_rate_min = 10762000, | ||
| 782 | .symbol_rate_max = 10762000, | ||
| 783 | .caps = FE_CAN_QAM_64 | FE_CAN_QAM_256 | FE_CAN_8VSB | ||
| 784 | }, | ||
| 785 | .init = lgdt330x_init, | ||
| 786 | .set_frontend = lgdt330x_set_parameters, | ||
| 787 | .get_frontend = lgdt330x_get_frontend, | ||
| 788 | .get_tune_settings = lgdt330x_get_tune_settings, | ||
| 789 | .read_status = lgdt3302_read_status, | ||
| 790 | .read_ber = lgdt330x_read_ber, | ||
| 791 | .read_signal_strength = lgdt330x_read_signal_strength, | ||
| 792 | .read_snr = lgdt3302_read_snr, | ||
| 793 | .read_ucblocks = lgdt330x_read_ucblocks, | ||
| 794 | .release = lgdt330x_release, | ||
| 795 | }; | ||
| 796 | |||
| 797 | static struct dvb_frontend_ops lgdt3303_ops = { | ||
| 572 | .info = { | 798 | .info = { |
| 573 | .name= "LG Electronics lgdt330x VSB/QAM Frontend", | 799 | .name= "LG Electronics LGDT3303 VSB/QAM Frontend", |
| 574 | .type = FE_ATSC, | 800 | .type = FE_ATSC, |
| 575 | .frequency_min= 54000000, | 801 | .frequency_min= 54000000, |
| 576 | .frequency_max= 858000000, | 802 | .frequency_max= 858000000, |
| @@ -584,15 +810,15 @@ static struct dvb_frontend_ops lgdt330x_ops = { | |||
| 584 | .set_frontend = lgdt330x_set_parameters, | 810 | .set_frontend = lgdt330x_set_parameters, |
| 585 | .get_frontend = lgdt330x_get_frontend, | 811 | .get_frontend = lgdt330x_get_frontend, |
| 586 | .get_tune_settings = lgdt330x_get_tune_settings, | 812 | .get_tune_settings = lgdt330x_get_tune_settings, |
| 587 | .read_status = lgdt330x_read_status, | 813 | .read_status = lgdt3303_read_status, |
| 588 | .read_ber = lgdt330x_read_ber, | 814 | .read_ber = lgdt330x_read_ber, |
| 589 | .read_signal_strength = lgdt330x_read_signal_strength, | 815 | .read_signal_strength = lgdt330x_read_signal_strength, |
| 590 | .read_snr = lgdt330x_read_snr, | 816 | .read_snr = lgdt3303_read_snr, |
| 591 | .read_ucblocks = lgdt330x_read_ucblocks, | 817 | .read_ucblocks = lgdt330x_read_ucblocks, |
| 592 | .release = lgdt330x_release, | 818 | .release = lgdt330x_release, |
| 593 | }; | 819 | }; |
| 594 | 820 | ||
| 595 | MODULE_DESCRIPTION("lgdt330x [DViCO FusionHDTV 3 Gold] (ATSC 8VSB & ITU-T J.83 AnnexB 64/256 QAM) Demodulator Driver"); | 821 | MODULE_DESCRIPTION("LGDT330X (ATSC 8VSB & ITU-T J.83 AnnexB 64/256 QAM) Demodulator Driver"); |
| 596 | MODULE_AUTHOR("Wilson Michaels"); | 822 | MODULE_AUTHOR("Wilson Michaels"); |
| 597 | MODULE_LICENSE("GPL"); | 823 | MODULE_LICENSE("GPL"); |
| 598 | 824 | ||
| @@ -601,6 +827,5 @@ EXPORT_SYMBOL(lgdt330x_attach); | |||
| 601 | /* | 827 | /* |
| 602 | * Local variables: | 828 | * Local variables: |
| 603 | * c-basic-offset: 8 | 829 | * c-basic-offset: 8 |
| 604 | * compile-command: "make DVB=1" | ||
| 605 | * End: | 830 | * End: |
| 606 | */ | 831 | */ |
diff --git a/drivers/media/dvb/frontends/lgdt330x.h b/drivers/media/dvb/frontends/lgdt330x.h index 04986f8e7565..e209ba1e47c5 100644 --- a/drivers/media/dvb/frontends/lgdt330x.h +++ b/drivers/media/dvb/frontends/lgdt330x.h | |||
| @@ -1,5 +1,5 @@ | |||
| 1 | /* | 1 | /* |
| 2 | * Support for LGDT3302 & LGDT3303 (DViCO FustionHDTV Gold) - VSB/QAM | 2 | * Support for LGDT3302 and LGDT3303 - VSB/QAM |
| 3 | * | 3 | * |
| 4 | * Copyright (C) 2005 Wilson Michaels <wilsonmichaels@earthlink.net> | 4 | * Copyright (C) 2005 Wilson Michaels <wilsonmichaels@earthlink.net> |
| 5 | * | 5 | * |
| @@ -24,14 +24,26 @@ | |||
| 24 | 24 | ||
| 25 | #include <linux/dvb/frontend.h> | 25 | #include <linux/dvb/frontend.h> |
| 26 | 26 | ||
| 27 | typedef enum lg_chip_t { | ||
| 28 | UNDEFINED, | ||
| 29 | LGDT3302, | ||
| 30 | LGDT3303 | ||
| 31 | }lg_chip_type; | ||
| 32 | |||
| 27 | struct lgdt330x_config | 33 | struct lgdt330x_config |
| 28 | { | 34 | { |
| 29 | /* The demodulator's i2c address */ | 35 | /* The demodulator's i2c address */ |
| 30 | u8 demod_address; | 36 | u8 demod_address; |
| 31 | 37 | ||
| 38 | /* LG demodulator chip LGDT3302 or LGDT3303 */ | ||
| 39 | lg_chip_type demod_chip; | ||
| 40 | |||
| 41 | /* MPEG hardware interface - 0:parallel 1:serial */ | ||
| 42 | int serial_mpeg; | ||
| 43 | |||
| 32 | /* PLL interface */ | 44 | /* PLL interface */ |
| 33 | int (*pll_rf_set) (struct dvb_frontend* fe, int index); | 45 | int (*pll_rf_set) (struct dvb_frontend* fe, int index); |
| 34 | int (*pll_set)(struct dvb_frontend* fe, struct dvb_frontend_parameters* params, u8* pll_address); | 46 | int (*pll_set)(struct dvb_frontend* fe, struct dvb_frontend_parameters* params); |
| 35 | 47 | ||
| 36 | /* Need to set device param for start_dma */ | 48 | /* Need to set device param for start_dma */ |
| 37 | int (*set_ts_params)(struct dvb_frontend* fe, int is_punctured); | 49 | int (*set_ts_params)(struct dvb_frontend* fe, int is_punctured); |
diff --git a/drivers/media/dvb/frontends/lgdt330x_priv.h b/drivers/media/dvb/frontends/lgdt330x_priv.h index 4143ce8f1a95..59b7c5b9012d 100644 --- a/drivers/media/dvb/frontends/lgdt330x_priv.h +++ b/drivers/media/dvb/frontends/lgdt330x_priv.h | |||
| @@ -1,5 +1,5 @@ | |||
| 1 | /* | 1 | /* |
| 2 | * Support for LGDT3302 & LGDT3303 (DViCO FustionHDTV Gold) - VSB/QAM | 2 | * Support for LGDT3302 and LGDT3303 - VSB/QAM |
| 3 | * | 3 | * |
| 4 | * Copyright (C) 2005 Wilson Michaels <wilsonmichaels@earthlink.net> | 4 | * Copyright (C) 2005 Wilson Michaels <wilsonmichaels@earthlink.net> |
| 5 | * | 5 | * |
| @@ -57,8 +57,10 @@ enum I2C_REG { | |||
| 57 | PH_ERR1= 0x4a, | 57 | PH_ERR1= 0x4a, |
| 58 | PH_ERR2= 0x4b, | 58 | PH_ERR2= 0x4b, |
| 59 | DEMUX_CONTROL= 0x66, | 59 | DEMUX_CONTROL= 0x66, |
| 60 | PACKET_ERR_COUNTER1= 0x6a, | 60 | LGDT3302_PACKET_ERR_COUNTER1= 0x6a, |
| 61 | PACKET_ERR_COUNTER2= 0x6b, | 61 | LGDT3302_PACKET_ERR_COUNTER2= 0x6b, |
| 62 | LGDT3303_PACKET_ERR_COUNTER1= 0x8b, | ||
| 63 | LGDT3303_PACKET_ERR_COUNTER2= 0x8c, | ||
| 62 | }; | 64 | }; |
| 63 | 65 | ||
| 64 | #endif /* _LGDT330X_PRIV_ */ | 66 | #endif /* _LGDT330X_PRIV_ */ |
diff --git a/drivers/media/video/cx88/cx88-dvb.c b/drivers/media/video/cx88/cx88-dvb.c index ef0e9a85c359..78d223257a68 100644 --- a/drivers/media/video/cx88/cx88-dvb.c +++ b/drivers/media/video/cx88/cx88-dvb.c | |||
| @@ -1,5 +1,5 @@ | |||
| 1 | /* | 1 | /* |
| 2 | * $Id: cx88-dvb.c,v 1.54 2005/07/25 05:13:50 mkrufky Exp $ | 2 | * $Id: cx88-dvb.c,v 1.58 2005/08/07 09:24:08 mkrufky Exp $ |
| 3 | * | 3 | * |
| 4 | * device driver for Conexant 2388x based TV cards | 4 | * device driver for Conexant 2388x based TV cards |
| 5 | * MPEG Transport Stream (DVB) routines | 5 | * MPEG Transport Stream (DVB) routines |
| @@ -208,14 +208,26 @@ static struct or51132_config pchdtv_hd3000 = { | |||
| 208 | 208 | ||
| 209 | #ifdef HAVE_LGDT330X | 209 | #ifdef HAVE_LGDT330X |
| 210 | static int lgdt330x_pll_set(struct dvb_frontend* fe, | 210 | static int lgdt330x_pll_set(struct dvb_frontend* fe, |
| 211 | struct dvb_frontend_parameters* params, | 211 | struct dvb_frontend_parameters* params) |
| 212 | u8* pllbuf) | ||
| 213 | { | 212 | { |
| 214 | struct cx8802_dev *dev= fe->dvb->priv; | 213 | struct cx8802_dev *dev= fe->dvb->priv; |
| 214 | u8 buf[4]; | ||
| 215 | struct i2c_msg msg = | ||
| 216 | { .addr = dev->core->pll_addr, .flags = 0, .buf = buf, .len = 4 }; | ||
| 217 | int err; | ||
| 215 | 218 | ||
| 216 | pllbuf[0] = dev->core->pll_addr; | 219 | dvb_pll_configure(dev->core->pll_desc, buf, params->frequency, 0); |
| 217 | dvb_pll_configure(dev->core->pll_desc, &pllbuf[1], | 220 | dprintk(1, "%s: tuner at 0x%02x bytes: 0x%02x 0x%02x 0x%02x 0x%02x\n", |
| 218 | params->frequency, 0); | 221 | __FUNCTION__, msg.addr, buf[0],buf[1],buf[2],buf[3]); |
| 222 | if ((err = i2c_transfer(&dev->core->i2c_adap, &msg, 1)) != 1) { | ||
| 223 | printk(KERN_WARNING "cx88-dvb: %s error " | ||
| 224 | "(addr %02x <- %02x, err = %i)\n", | ||
| 225 | __FUNCTION__, buf[0], buf[1], err); | ||
| 226 | if (err < 0) | ||
| 227 | return err; | ||
| 228 | else | ||
| 229 | return -EREMOTEIO; | ||
| 230 | } | ||
| 219 | return 0; | 231 | return 0; |
| 220 | } | 232 | } |
| 221 | 233 | ||
| @@ -244,6 +256,8 @@ static int lgdt330x_set_ts_param(struct dvb_frontend* fe, int is_punctured) | |||
| 244 | 256 | ||
| 245 | static struct lgdt330x_config fusionhdtv_3_gold = { | 257 | static struct lgdt330x_config fusionhdtv_3_gold = { |
| 246 | .demod_address = 0x0e, | 258 | .demod_address = 0x0e, |
| 259 | .demod_chip = LGDT3302, | ||
| 260 | .serial_mpeg = 0x04, /* TPSERIAL for 3302 in TOP_CONTROL */ | ||
| 247 | .pll_set = lgdt330x_pll_set, | 261 | .pll_set = lgdt330x_pll_set, |
| 248 | .set_ts_params = lgdt330x_set_ts_param, | 262 | .set_ts_params = lgdt330x_set_ts_param, |
| 249 | }; | 263 | }; |
