1 // SPDX-License-Identifier: GPL-2.0-or-later 2 /* 3 * Linux-DVB Driver for DiBcom's DiB0090 base-band RF Tuner. 4 * 5 * Copyright (C) 2005-9 DiBcom (http://www.dibcom.fr/) 6 * 7 * This code is more or less generated from another driver, please 8 * excuse some codingstyle oddities. 9 */ 10 11 #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt 12 13 #include <linux/kernel.h> 14 #include <linux/slab.h> 15 #include <linux/i2c.h> 16 #include <linux/mutex.h> 17 18 #include <media/dvb_frontend.h> 19 20 #include "dib0090.h" 21 #include "dibx000_common.h" 22 23 static int debug; 24 module_param(debug, int, 0644); 25 MODULE_PARM_DESC(debug, "turn on debugging (default: 0)"); 26 27 #define dprintk(fmt, arg...) do { \ 28 if (debug) \ 29 printk(KERN_DEBUG pr_fmt("%s: " fmt), \ 30 __func__, ##arg); \ 31 } while (0) 32 33 #define CONFIG_SYS_DVBT 34 #define CONFIG_SYS_ISDBT 35 #define CONFIG_BAND_CBAND 36 #define CONFIG_BAND_VHF 37 #define CONFIG_BAND_UHF 38 #define CONFIG_DIB0090_USE_PWM_AGC 39 40 #define EN_LNA0 0x8000 41 #define EN_LNA1 0x4000 42 #define EN_LNA2 0x2000 43 #define EN_LNA3 0x1000 44 #define EN_MIX0 0x0800 45 #define EN_MIX1 0x0400 46 #define EN_MIX2 0x0200 47 #define EN_MIX3 0x0100 48 #define EN_IQADC 0x0040 49 #define EN_PLL 0x0020 50 #define EN_TX 0x0010 51 #define EN_BB 0x0008 52 #define EN_LO 0x0004 53 #define EN_BIAS 0x0001 54 55 #define EN_IQANA 0x0002 56 #define EN_DIGCLK 0x0080 /* not in the 0x24 reg, only in 0x1b */ 57 #define EN_CRYSTAL 0x0002 58 59 #define EN_UHF 0x22E9 60 #define EN_VHF 0x44E9 61 #define EN_LBD 0x11E9 62 #define EN_SBD 0x44E9 63 #define EN_CAB 0x88E9 64 65 /* Calibration defines */ 66 #define DC_CAL 0x1 67 #define WBD_CAL 0x2 68 #define TEMP_CAL 0x4 69 #define CAPTRIM_CAL 0x8 70 71 #define KROSUS_PLL_LOCKED 0x800 72 #define KROSUS 0x2 73 74 /* Use those defines to identify SOC version */ 75 #define SOC 0x02 76 #define SOC_7090_P1G_11R1 0x82 77 #define SOC_7090_P1G_21R1 0x8a 78 #define SOC_8090_P1G_11R1 0x86 79 #define SOC_8090_P1G_21R1 0x8e 80 81 /* else use thos ones to check */ 82 #define P1A_B 0x0 83 #define P1C 0x1 84 #define P1D_E_F 0x3 85 #define P1G 0x7 86 #define P1G_21R2 0xf 87 88 #define MP001 0x1 /* Single 9090/8096 */ 89 #define MP005 0x4 /* Single Sband */ 90 #define MP008 0x6 /* Dual diversity VHF-UHF-LBAND */ 91 #define MP009 0x7 /* Dual diversity 29098 CBAND-UHF-LBAND-SBAND */ 92 93 #define pgm_read_word(w) (*w) 94 95 struct dc_calibration; 96 97 struct dib0090_tuning { 98 u32 max_freq; /* for every frequency less than or equal to that field: this information is correct */ 99 u8 switch_trim; 100 u8 lna_tune; 101 u16 lna_bias; 102 u16 v2i; 103 u16 mix; 104 u16 load; 105 u16 tuner_enable; 106 }; 107 108 struct dib0090_pll { 109 u32 max_freq; /* for every frequency less than or equal to that field: this information is correct */ 110 u8 vco_band; 111 u8 hfdiv_code; 112 u8 hfdiv; 113 u8 topresc; 114 }; 115 116 struct dib0090_identity { 117 u8 version; 118 u8 product; 119 u8 p1g; 120 u8 in_soc; 121 }; 122 123 struct dib0090_state { 124 struct i2c_adapter *i2c; 125 struct dvb_frontend *fe; 126 const struct dib0090_config *config; 127 128 u8 current_band; 129 enum frontend_tune_state tune_state; 130 u32 current_rf; 131 132 u16 wbd_offset; 133 s16 wbd_target; /* in dB */ 134 135 s16 rf_gain_limit; /* take-over-point: where to split between bb and rf gain */ 136 s16 current_gain; /* keeps the currently programmed gain */ 137 u8 agc_step; /* new binary search */ 138 139 u16 gain[2]; /* for channel monitoring */ 140 141 const u16 *rf_ramp; 142 const u16 *bb_ramp; 143 144 /* for the software AGC ramps */ 145 u16 bb_1_def; 146 u16 rf_lt_def; 147 u16 gain_reg[4]; 148 149 /* for the captrim/dc-offset search */ 150 s8 step; 151 s16 adc_diff; 152 s16 min_adc_diff; 153 154 s8 captrim; 155 s8 fcaptrim; 156 157 const struct dc_calibration *dc; 158 u16 bb6, bb7; 159 160 const struct dib0090_tuning *current_tune_table_index; 161 const struct dib0090_pll *current_pll_table_index; 162 163 u8 tuner_is_tuned; 164 u8 agc_freeze; 165 166 struct dib0090_identity identity; 167 168 u32 rf_request; 169 u8 current_standard; 170 171 u8 calibrate; 172 u32 rest; 173 u16 bias; 174 s16 temperature; 175 176 u8 wbd_calibration_gain; 177 const struct dib0090_wbd_slope *current_wbd_table; 178 u16 wbdmux; 179 180 /* for the I2C transfer */ 181 struct i2c_msg msg[2]; 182 u8 i2c_write_buffer[3]; 183 u8 i2c_read_buffer[2]; 184 struct mutex i2c_buffer_lock; 185 }; 186 187 struct dib0090_fw_state { 188 struct i2c_adapter *i2c; 189 struct dvb_frontend *fe; 190 struct dib0090_identity identity; 191 const struct dib0090_config *config; 192 193 /* for the I2C transfer */ 194 struct i2c_msg msg; 195 u8 i2c_write_buffer[2]; 196 u8 i2c_read_buffer[2]; 197 struct mutex i2c_buffer_lock; 198 }; 199 200 static u16 dib0090_read_reg(struct dib0090_state *state, u8 reg) 201 { 202 u16 ret; 203 204 if (mutex_lock_interruptible(&state->i2c_buffer_lock) < 0) { 205 dprintk("could not acquire lock\n"); 206 return 0; 207 } 208 209 state->i2c_write_buffer[0] = reg; 210 211 memset(state->msg, 0, 2 * sizeof(struct i2c_msg)); 212 state->msg[0].addr = state->config->i2c_address; 213 state->msg[0].flags = 0; 214 state->msg[0].buf = state->i2c_write_buffer; 215 state->msg[0].len = 1; 216 state->msg[1].addr = state->config->i2c_address; 217 state->msg[1].flags = I2C_M_RD; 218 state->msg[1].buf = state->i2c_read_buffer; 219 state->msg[1].len = 2; 220 221 if (i2c_transfer(state->i2c, state->msg, 2) != 2) { 222 pr_warn("DiB0090 I2C read failed\n"); 223 ret = 0; 224 } else 225 ret = (state->i2c_read_buffer[0] << 8) 226 | state->i2c_read_buffer[1]; 227 228 mutex_unlock(&state->i2c_buffer_lock); 229 return ret; 230 } 231 232 static int dib0090_write_reg(struct dib0090_state *state, u32 reg, u16 val) 233 { 234 int ret; 235 236 if (mutex_lock_interruptible(&state->i2c_buffer_lock) < 0) { 237 dprintk("could not acquire lock\n"); 238 return -EINVAL; 239 } 240 241 state->i2c_write_buffer[0] = reg & 0xff; 242 state->i2c_write_buffer[1] = val >> 8; 243 state->i2c_write_buffer[2] = val & 0xff; 244 245 memset(state->msg, 0, sizeof(struct i2c_msg)); 246 state->msg[0].addr = state->config->i2c_address; 247 state->msg[0].flags = 0; 248 state->msg[0].buf = state->i2c_write_buffer; 249 state->msg[0].len = 3; 250 251 if (i2c_transfer(state->i2c, state->msg, 1) != 1) { 252 pr_warn("DiB0090 I2C write failed\n"); 253 ret = -EREMOTEIO; 254 } else 255 ret = 0; 256 257 mutex_unlock(&state->i2c_buffer_lock); 258 return ret; 259 } 260 261 static u16 dib0090_fw_read_reg(struct dib0090_fw_state *state, u8 reg) 262 { 263 u16 ret; 264 265 if (mutex_lock_interruptible(&state->i2c_buffer_lock) < 0) { 266 dprintk("could not acquire lock\n"); 267 return 0; 268 } 269 270 state->i2c_write_buffer[0] = reg; 271 272 memset(&state->msg, 0, sizeof(struct i2c_msg)); 273 state->msg.addr = reg; 274 state->msg.flags = I2C_M_RD; 275 state->msg.buf = state->i2c_read_buffer; 276 state->msg.len = 2; 277 if (i2c_transfer(state->i2c, &state->msg, 1) != 1) { 278 pr_warn("DiB0090 I2C read failed\n"); 279 ret = 0; 280 } else 281 ret = (state->i2c_read_buffer[0] << 8) 282 | state->i2c_read_buffer[1]; 283 284 mutex_unlock(&state->i2c_buffer_lock); 285 return ret; 286 } 287 288 static int dib0090_fw_write_reg(struct dib0090_fw_state *state, u8 reg, u16 val) 289 { 290 int ret; 291 292 if (mutex_lock_interruptible(&state->i2c_buffer_lock) < 0) { 293 dprintk("could not acquire lock\n"); 294 return -EINVAL; 295 } 296 297 state->i2c_write_buffer[0] = val >> 8; 298 state->i2c_write_buffer[1] = val & 0xff; 299 300 memset(&state->msg, 0, sizeof(struct i2c_msg)); 301 state->msg.addr = reg; 302 state->msg.flags = 0; 303 state->msg.buf = state->i2c_write_buffer; 304 state->msg.len = 2; 305 if (i2c_transfer(state->i2c, &state->msg, 1) != 1) { 306 pr_warn("DiB0090 I2C write failed\n"); 307 ret = -EREMOTEIO; 308 } else 309 ret = 0; 310 311 mutex_unlock(&state->i2c_buffer_lock); 312 return ret; 313 } 314 315 #define HARD_RESET(state) do { if (cfg->reset) { if (cfg->sleep) cfg->sleep(fe, 0); msleep(10); cfg->reset(fe, 1); msleep(10); cfg->reset(fe, 0); msleep(10); } } while (0) 316 #define ADC_TARGET -220 317 #define GAIN_ALPHA 5 318 #define WBD_ALPHA 6 319 #define LPF 100 320 static void dib0090_write_regs(struct dib0090_state *state, u8 r, const u16 * b, u8 c) 321 { 322 do { 323 dib0090_write_reg(state, r++, *b++); 324 } while (--c); 325 } 326 327 static int dib0090_identify(struct dvb_frontend *fe) 328 { 329 struct dib0090_state *state = fe->tuner_priv; 330 u16 v; 331 struct dib0090_identity *identity = &state->identity; 332 333 v = dib0090_read_reg(state, 0x1a); 334 335 identity->p1g = 0; 336 identity->in_soc = 0; 337 338 dprintk("Tuner identification (Version = 0x%04x)\n", v); 339 340 /* without PLL lock info */ 341 v &= ~KROSUS_PLL_LOCKED; 342 343 identity->version = v & 0xff; 344 identity->product = (v >> 8) & 0xf; 345 346 if (identity->product != KROSUS) 347 goto identification_error; 348 349 if ((identity->version & 0x3) == SOC) { 350 identity->in_soc = 1; 351 switch (identity->version) { 352 case SOC_8090_P1G_11R1: 353 dprintk("SOC 8090 P1-G11R1 Has been detected\n"); 354 identity->p1g = 1; 355 break; 356 case SOC_8090_P1G_21R1: 357 dprintk("SOC 8090 P1-G21R1 Has been detected\n"); 358 identity->p1g = 1; 359 break; 360 case SOC_7090_P1G_11R1: 361 dprintk("SOC 7090 P1-G11R1 Has been detected\n"); 362 identity->p1g = 1; 363 break; 364 case SOC_7090_P1G_21R1: 365 dprintk("SOC 7090 P1-G21R1 Has been detected\n"); 366 identity->p1g = 1; 367 break; 368 default: 369 goto identification_error; 370 } 371 } else { 372 switch ((identity->version >> 5) & 0x7) { 373 case MP001: 374 dprintk("MP001 : 9090/8096\n"); 375 break; 376 case MP005: 377 dprintk("MP005 : Single Sband\n"); 378 break; 379 case MP008: 380 dprintk("MP008 : diversity VHF-UHF-LBAND\n"); 381 break; 382 case MP009: 383 dprintk("MP009 : diversity 29098 CBAND-UHF-LBAND-SBAND\n"); 384 break; 385 default: 386 goto identification_error; 387 } 388 389 switch (identity->version & 0x1f) { 390 case P1G_21R2: 391 dprintk("P1G_21R2 detected\n"); 392 identity->p1g = 1; 393 break; 394 case P1G: 395 dprintk("P1G detected\n"); 396 identity->p1g = 1; 397 break; 398 case P1D_E_F: 399 dprintk("P1D/E/F detected\n"); 400 break; 401 case P1C: 402 dprintk("P1C detected\n"); 403 break; 404 case P1A_B: 405 dprintk("P1-A/B detected: driver is deactivated - not available\n"); 406 goto identification_error; 407 break; 408 default: 409 goto identification_error; 410 } 411 } 412 413 return 0; 414 415 identification_error: 416 return -EIO; 417 } 418 419 static int dib0090_fw_identify(struct dvb_frontend *fe) 420 { 421 struct dib0090_fw_state *state = fe->tuner_priv; 422 struct dib0090_identity *identity = &state->identity; 423 424 u16 v = dib0090_fw_read_reg(state, 0x1a); 425 identity->p1g = 0; 426 identity->in_soc = 0; 427 428 dprintk("FE: Tuner identification (Version = 0x%04x)\n", v); 429 430 /* without PLL lock info */ 431 v &= ~KROSUS_PLL_LOCKED; 432 433 identity->version = v & 0xff; 434 identity->product = (v >> 8) & 0xf; 435 436 if (identity->product != KROSUS) 437 goto identification_error; 438 439 if ((identity->version & 0x3) == SOC) { 440 identity->in_soc = 1; 441 switch (identity->version) { 442 case SOC_8090_P1G_11R1: 443 dprintk("SOC 8090 P1-G11R1 Has been detected\n"); 444 identity->p1g = 1; 445 break; 446 case SOC_8090_P1G_21R1: 447 dprintk("SOC 8090 P1-G21R1 Has been detected\n"); 448 identity->p1g = 1; 449 break; 450 case SOC_7090_P1G_11R1: 451 dprintk("SOC 7090 P1-G11R1 Has been detected\n"); 452 identity->p1g = 1; 453 break; 454 case SOC_7090_P1G_21R1: 455 dprintk("SOC 7090 P1-G21R1 Has been detected\n"); 456 identity->p1g = 1; 457 break; 458 default: 459 goto identification_error; 460 } 461 } else { 462 switch ((identity->version >> 5) & 0x7) { 463 case MP001: 464 dprintk("MP001 : 9090/8096\n"); 465 break; 466 case MP005: 467 dprintk("MP005 : Single Sband\n"); 468 break; 469 case MP008: 470 dprintk("MP008 : diversity VHF-UHF-LBAND\n"); 471 break; 472 case MP009: 473 dprintk("MP009 : diversity 29098 CBAND-UHF-LBAND-SBAND\n"); 474 break; 475 default: 476 goto identification_error; 477 } 478 479 switch (identity->version & 0x1f) { 480 case P1G_21R2: 481 dprintk("P1G_21R2 detected\n"); 482 identity->p1g = 1; 483 break; 484 case P1G: 485 dprintk("P1G detected\n"); 486 identity->p1g = 1; 487 break; 488 case P1D_E_F: 489 dprintk("P1D/E/F detected\n"); 490 break; 491 case P1C: 492 dprintk("P1C detected\n"); 493 break; 494 case P1A_B: 495 dprintk("P1-A/B detected: driver is deactivated - not available\n"); 496 goto identification_error; 497 break; 498 default: 499 goto identification_error; 500 } 501 } 502 503 return 0; 504 505 identification_error: 506 return -EIO; 507 } 508 509 static void dib0090_reset_digital(struct dvb_frontend *fe, const struct dib0090_config *cfg) 510 { 511 struct dib0090_state *state = fe->tuner_priv; 512 u16 PllCfg, i, v; 513 514 HARD_RESET(state); 515 dib0090_write_reg(state, 0x24, EN_PLL | EN_CRYSTAL); 516 if (cfg->in_soc) 517 return; 518 519 dib0090_write_reg(state, 0x1b, EN_DIGCLK | EN_PLL | EN_CRYSTAL); /* PLL, DIG_CLK and CRYSTAL remain */ 520 /* adcClkOutRatio=8->7, release reset */ 521 dib0090_write_reg(state, 0x20, ((cfg->io.adc_clock_ratio - 1) << 11) | (0 << 10) | (1 << 9) | (1 << 8) | (0 << 4) | 0); 522 if (cfg->clkoutdrive != 0) 523 dib0090_write_reg(state, 0x23, (0 << 15) | ((!cfg->analog_output) << 14) | (2 << 10) | (1 << 9) | (0 << 8) 524 | (cfg->clkoutdrive << 5) | (cfg->clkouttobamse << 4) | (0 << 2) | (0)); 525 else 526 dib0090_write_reg(state, 0x23, (0 << 15) | ((!cfg->analog_output) << 14) | (2 << 10) | (1 << 9) | (0 << 8) 527 | (7 << 5) | (cfg->clkouttobamse << 4) | (0 << 2) | (0)); 528 529 /* Read Pll current config * */ 530 PllCfg = dib0090_read_reg(state, 0x21); 531 532 /** Reconfigure PLL if current setting is different from default setting **/ 533 if ((PllCfg & 0x1FFF) != ((cfg->io.pll_range << 12) | (cfg->io.pll_loopdiv << 6) | (cfg->io.pll_prediv)) && (!cfg->in_soc) 534 && !cfg->io.pll_bypass) { 535 536 /* Set Bypass mode */ 537 PllCfg |= (1 << 15); 538 dib0090_write_reg(state, 0x21, PllCfg); 539 540 /* Set Reset Pll */ 541 PllCfg &= ~(1 << 13); 542 dib0090_write_reg(state, 0x21, PllCfg); 543 544 /*** Set new Pll configuration in bypass and reset state ***/ 545 PllCfg = (1 << 15) | (0 << 13) | (cfg->io.pll_range << 12) | (cfg->io.pll_loopdiv << 6) | (cfg->io.pll_prediv); 546 dib0090_write_reg(state, 0x21, PllCfg); 547 548 /* Remove Reset Pll */ 549 PllCfg |= (1 << 13); 550 dib0090_write_reg(state, 0x21, PllCfg); 551 552 /*** Wait for PLL lock ***/ 553 i = 100; 554 do { 555 v = !!(dib0090_read_reg(state, 0x1a) & 0x800); 556 if (v) 557 break; 558 } while (--i); 559 560 if (i == 0) { 561 dprintk("Pll: Unable to lock Pll\n"); 562 return; 563 } 564 565 /* Finally Remove Bypass mode */ 566 PllCfg &= ~(1 << 15); 567 dib0090_write_reg(state, 0x21, PllCfg); 568 } 569 570 if (cfg->io.pll_bypass) { 571 PllCfg |= (cfg->io.pll_bypass << 15); 572 dib0090_write_reg(state, 0x21, PllCfg); 573 } 574 } 575 576 static int dib0090_fw_reset_digital(struct dvb_frontend *fe, const struct dib0090_config *cfg) 577 { 578 struct dib0090_fw_state *state = fe->tuner_priv; 579 u16 PllCfg; 580 u16 v; 581 int i; 582 583 dprintk("fw reset digital\n"); 584 HARD_RESET(state); 585 586 dib0090_fw_write_reg(state, 0x24, EN_PLL | EN_CRYSTAL); 587 dib0090_fw_write_reg(state, 0x1b, EN_DIGCLK | EN_PLL | EN_CRYSTAL); /* PLL, DIG_CLK and CRYSTAL remain */ 588 589 dib0090_fw_write_reg(state, 0x20, 590 ((cfg->io.adc_clock_ratio - 1) << 11) | (0 << 10) | (1 << 9) | (1 << 8) | (cfg->data_tx_drv << 4) | cfg->ls_cfg_pad_drv); 591 592 v = (0 << 15) | ((!cfg->analog_output) << 14) | (1 << 9) | (0 << 8) | (cfg->clkouttobamse << 4) | (0 << 2) | (0); 593 if (cfg->clkoutdrive != 0) 594 v |= cfg->clkoutdrive << 5; 595 else 596 v |= 7 << 5; 597 598 v |= 2 << 10; 599 dib0090_fw_write_reg(state, 0x23, v); 600 601 /* Read Pll current config * */ 602 PllCfg = dib0090_fw_read_reg(state, 0x21); 603 604 /** Reconfigure PLL if current setting is different from default setting **/ 605 if ((PllCfg & 0x1FFF) != ((cfg->io.pll_range << 12) | (cfg->io.pll_loopdiv << 6) | (cfg->io.pll_prediv)) && !cfg->io.pll_bypass) { 606 607 /* Set Bypass mode */ 608 PllCfg |= (1 << 15); 609 dib0090_fw_write_reg(state, 0x21, PllCfg); 610 611 /* Set Reset Pll */ 612 PllCfg &= ~(1 << 13); 613 dib0090_fw_write_reg(state, 0x21, PllCfg); 614 615 /*** Set new Pll configuration in bypass and reset state ***/ 616 PllCfg = (1 << 15) | (0 << 13) | (cfg->io.pll_range << 12) | (cfg->io.pll_loopdiv << 6) | (cfg->io.pll_prediv); 617 dib0090_fw_write_reg(state, 0x21, PllCfg); 618 619 /* Remove Reset Pll */ 620 PllCfg |= (1 << 13); 621 dib0090_fw_write_reg(state, 0x21, PllCfg); 622 623 /*** Wait for PLL lock ***/ 624 i = 100; 625 do { 626 v = !!(dib0090_fw_read_reg(state, 0x1a) & 0x800); 627 if (v) 628 break; 629 } while (--i); 630 631 if (i == 0) { 632 dprintk("Pll: Unable to lock Pll\n"); 633 return -EIO; 634 } 635 636 /* Finally Remove Bypass mode */ 637 PllCfg &= ~(1 << 15); 638 dib0090_fw_write_reg(state, 0x21, PllCfg); 639 } 640 641 if (cfg->io.pll_bypass) { 642 PllCfg |= (cfg->io.pll_bypass << 15); 643 dib0090_fw_write_reg(state, 0x21, PllCfg); 644 } 645 646 return dib0090_fw_identify(fe); 647 } 648 649 static int dib0090_wakeup(struct dvb_frontend *fe) 650 { 651 struct dib0090_state *state = fe->tuner_priv; 652 if (state->config->sleep) 653 state->config->sleep(fe, 0); 654 655 /* enable dataTX in case we have been restarted in the wrong moment */ 656 dib0090_write_reg(state, 0x23, dib0090_read_reg(state, 0x23) | (1 << 14)); 657 return 0; 658 } 659 660 static int dib0090_sleep(struct dvb_frontend *fe) 661 { 662 struct dib0090_state *state = fe->tuner_priv; 663 if (state->config->sleep) 664 state->config->sleep(fe, 1); 665 return 0; 666 } 667 668 void dib0090_dcc_freq(struct dvb_frontend *fe, u8 fast) 669 { 670 struct dib0090_state *state = fe->tuner_priv; 671 if (fast) 672 dib0090_write_reg(state, 0x04, 0); 673 else 674 dib0090_write_reg(state, 0x04, 1); 675 } 676 677 EXPORT_SYMBOL(dib0090_dcc_freq); 678 679 static const u16 bb_ramp_pwm_normal_socs[] = { 680 550, /* max BB gain in 10th of dB */ 681 (1<<9) | 8, /* ramp_slope = 1dB of gain -> clock_ticks_per_db = clk_khz / ramp_slope -> BB_RAMP2 */ 682 440, 683 (4 << 9) | 0, /* BB_RAMP3 = 26dB */ 684 (0 << 9) | 208, /* BB_RAMP4 */ 685 (4 << 9) | 208, /* BB_RAMP5 = 29dB */ 686 (0 << 9) | 440, /* BB_RAMP6 */ 687 }; 688 689 static const u16 rf_ramp_pwm_cband_7090p[] = { 690 280, /* max RF gain in 10th of dB */ 691 18, /* ramp_slope = 1dB of gain -> clock_ticks_per_db = clk_khz / ramp_slope -> RF_RAMP2 */ 692 504, /* ramp_max = maximum X used on the ramp */ 693 (29 << 10) | 364, /* RF_RAMP5, LNA 1 = 8dB */ 694 (0 << 10) | 504, /* RF_RAMP6, LNA 1 */ 695 (60 << 10) | 228, /* RF_RAMP7, LNA 2 = 7.7dB */ 696 (0 << 10) | 364, /* RF_RAMP8, LNA 2 */ 697 (34 << 10) | 109, /* GAIN_4_1, LNA 3 = 6.8dB */ 698 (0 << 10) | 228, /* GAIN_4_2, LNA 3 */ 699 (37 << 10) | 0, /* RF_RAMP3, LNA 4 = 6.2dB */ 700 (0 << 10) | 109, /* RF_RAMP4, LNA 4 */ 701 }; 702 703 static const u16 rf_ramp_pwm_cband_7090e_sensitivity[] = { 704 186, /* max RF gain in 10th of dB */ 705 40, /* ramp_slope = 1dB of gain -> clock_ticks_per_db = clk_khz / ramp_slope -> RF_RAMP2 */ 706 746, /* ramp_max = maximum X used on the ramp */ 707 (10 << 10) | 345, /* RF_RAMP5, LNA 1 = 10dB */ 708 (0 << 10) | 746, /* RF_RAMP6, LNA 1 */ 709 (0 << 10) | 0, /* RF_RAMP7, LNA 2 = 0 dB */ 710 (0 << 10) | 0, /* RF_RAMP8, LNA 2 */ 711 (28 << 10) | 200, /* GAIN_4_1, LNA 3 = 6.8dB */ /* 3.61 dB */ 712 (0 << 10) | 345, /* GAIN_4_2, LNA 3 */ 713 (20 << 10) | 0, /* RF_RAMP3, LNA 4 = 6.2dB */ /* 4.96 dB */ 714 (0 << 10) | 200, /* RF_RAMP4, LNA 4 */ 715 }; 716 717 static const u16 rf_ramp_pwm_cband_7090e_aci[] = { 718 86, /* max RF gain in 10th of dB */ 719 40, /* ramp_slope = 1dB of gain -> clock_ticks_per_db = clk_khz / ramp_slope -> RF_RAMP2 */ 720 345, /* ramp_max = maximum X used on the ramp */ 721 (0 << 10) | 0, /* RF_RAMP5, LNA 1 = 8dB */ /* 7.47 dB */ 722 (0 << 10) | 0, /* RF_RAMP6, LNA 1 */ 723 (0 << 10) | 0, /* RF_RAMP7, LNA 2 = 0 dB */ 724 (0 << 10) | 0, /* RF_RAMP8, LNA 2 */ 725 (28 << 10) | 200, /* GAIN_4_1, LNA 3 = 6.8dB */ /* 3.61 dB */ 726 (0 << 10) | 345, /* GAIN_4_2, LNA 3 */ 727 (20 << 10) | 0, /* RF_RAMP3, LNA 4 = 6.2dB */ /* 4.96 dB */ 728 (0 << 10) | 200, /* RF_RAMP4, LNA 4 */ 729 }; 730 731 static const u16 rf_ramp_pwm_cband_8090[] = { 732 345, /* max RF gain in 10th of dB */ 733 29, /* ramp_slope = 1dB of gain -> clock_ticks_per_db = clk_khz / ramp_slope -> RF_RAMP2 */ 734 1000, /* ramp_max = maximum X used on the ramp */ 735 (35 << 10) | 772, /* RF_RAMP3, LNA 1 = 8dB */ 736 (0 << 10) | 1000, /* RF_RAMP4, LNA 1 */ 737 (58 << 10) | 496, /* RF_RAMP5, LNA 2 = 9.5dB */ 738 (0 << 10) | 772, /* RF_RAMP6, LNA 2 */ 739 (27 << 10) | 200, /* RF_RAMP7, LNA 3 = 10.5dB */ 740 (0 << 10) | 496, /* RF_RAMP8, LNA 3 */ 741 (40 << 10) | 0, /* GAIN_4_1, LNA 4 = 7dB */ 742 (0 << 10) | 200, /* GAIN_4_2, LNA 4 */ 743 }; 744 745 static const u16 rf_ramp_pwm_uhf_7090[] = { 746 407, /* max RF gain in 10th of dB */ 747 13, /* ramp_slope = 1dB of gain -> clock_ticks_per_db = clk_khz / ramp_slope -> RF_RAMP2 */ 748 529, /* ramp_max = maximum X used on the ramp */ 749 (23 << 10) | 0, /* RF_RAMP3, LNA 1 = 14.7dB */ 750 (0 << 10) | 176, /* RF_RAMP4, LNA 1 */ 751 (63 << 10) | 400, /* RF_RAMP5, LNA 2 = 8dB */ 752 (0 << 10) | 529, /* RF_RAMP6, LNA 2 */ 753 (48 << 10) | 316, /* RF_RAMP7, LNA 3 = 6.8dB */ 754 (0 << 10) | 400, /* RF_RAMP8, LNA 3 */ 755 (29 << 10) | 176, /* GAIN_4_1, LNA 4 = 11.5dB */ 756 (0 << 10) | 316, /* GAIN_4_2, LNA 4 */ 757 }; 758 759 static const u16 rf_ramp_pwm_uhf_8090[] = { 760 388, /* max RF gain in 10th of dB */ 761 26, /* ramp_slope = 1dB of gain -> clock_ticks_per_db = clk_khz / ramp_slope -> RF_RAMP2 */ 762 1008, /* ramp_max = maximum X used on the ramp */ 763 (11 << 10) | 0, /* RF_RAMP3, LNA 1 = 14.7dB */ 764 (0 << 10) | 369, /* RF_RAMP4, LNA 1 */ 765 (41 << 10) | 809, /* RF_RAMP5, LNA 2 = 8dB */ 766 (0 << 10) | 1008, /* RF_RAMP6, LNA 2 */ 767 (27 << 10) | 659, /* RF_RAMP7, LNA 3 = 6dB */ 768 (0 << 10) | 809, /* RF_RAMP8, LNA 3 */ 769 (14 << 10) | 369, /* GAIN_4_1, LNA 4 = 11.5dB */ 770 (0 << 10) | 659, /* GAIN_4_2, LNA 4 */ 771 }; 772 773 /* GENERAL PWM ramp definition for all other Krosus */ 774 static const u16 bb_ramp_pwm_normal[] = { 775 500, /* max BB gain in 10th of dB */ 776 8, /* ramp_slope = 1dB of gain -> clock_ticks_per_db = clk_khz / ramp_slope -> BB_RAMP2 */ 777 400, 778 (2 << 9) | 0, /* BB_RAMP3 = 21dB */ 779 (0 << 9) | 168, /* BB_RAMP4 */ 780 (2 << 9) | 168, /* BB_RAMP5 = 29dB */ 781 (0 << 9) | 400, /* BB_RAMP6 */ 782 }; 783 784 #if 0 785 /* Currently unused */ 786 static const u16 bb_ramp_pwm_boost[] = { 787 550, /* max BB gain in 10th of dB */ 788 8, /* ramp_slope = 1dB of gain -> clock_ticks_per_db = clk_khz / ramp_slope -> BB_RAMP2 */ 789 440, 790 (2 << 9) | 0, /* BB_RAMP3 = 26dB */ 791 (0 << 9) | 208, /* BB_RAMP4 */ 792 (2 << 9) | 208, /* BB_RAMP5 = 29dB */ 793 (0 << 9) | 440, /* BB_RAMP6 */ 794 }; 795 #endif 796 797 static const u16 rf_ramp_pwm_cband[] = { 798 314, /* max RF gain in 10th of dB */ 799 33, /* ramp_slope = 1dB of gain -> clock_ticks_per_db = clk_khz / ramp_slope -> RF_RAMP2 */ 800 1023, /* ramp_max = maximum X used on the ramp */ 801 (8 << 10) | 743, /* RF_RAMP3, LNA 1 = 0dB */ 802 (0 << 10) | 1023, /* RF_RAMP4, LNA 1 */ 803 (15 << 10) | 469, /* RF_RAMP5, LNA 2 = 0dB */ 804 (0 << 10) | 742, /* RF_RAMP6, LNA 2 */ 805 (9 << 10) | 234, /* RF_RAMP7, LNA 3 = 0dB */ 806 (0 << 10) | 468, /* RF_RAMP8, LNA 3 */ 807 (9 << 10) | 0, /* GAIN_4_1, LNA 4 = 0dB */ 808 (0 << 10) | 233, /* GAIN_4_2, LNA 4 */ 809 }; 810 811 static const u16 rf_ramp_pwm_vhf[] = { 812 398, /* max RF gain in 10th of dB */ 813 24, /* ramp_slope = 1dB of gain -> clock_ticks_per_db = clk_khz / ramp_slope -> RF_RAMP2 */ 814 954, /* ramp_max = maximum X used on the ramp */ 815 (7 << 10) | 0, /* RF_RAMP3, LNA 1 = 13.2dB */ 816 (0 << 10) | 290, /* RF_RAMP4, LNA 1 */ 817 (16 << 10) | 699, /* RF_RAMP5, LNA 2 = 10.5dB */ 818 (0 << 10) | 954, /* RF_RAMP6, LNA 2 */ 819 (17 << 10) | 580, /* RF_RAMP7, LNA 3 = 5dB */ 820 (0 << 10) | 699, /* RF_RAMP8, LNA 3 */ 821 (7 << 10) | 290, /* GAIN_4_1, LNA 4 = 12.5dB */ 822 (0 << 10) | 580, /* GAIN_4_2, LNA 4 */ 823 }; 824 825 static const u16 rf_ramp_pwm_uhf[] = { 826 398, /* max RF gain in 10th of dB */ 827 24, /* ramp_slope = 1dB of gain -> clock_ticks_per_db = clk_khz / ramp_slope -> RF_RAMP2 */ 828 954, /* ramp_max = maximum X used on the ramp */ 829 (7 << 10) | 0, /* RF_RAMP3, LNA 1 = 13.2dB */ 830 (0 << 10) | 290, /* RF_RAMP4, LNA 1 */ 831 (16 << 10) | 699, /* RF_RAMP5, LNA 2 = 10.5dB */ 832 (0 << 10) | 954, /* RF_RAMP6, LNA 2 */ 833 (17 << 10) | 580, /* RF_RAMP7, LNA 3 = 5dB */ 834 (0 << 10) | 699, /* RF_RAMP8, LNA 3 */ 835 (7 << 10) | 290, /* GAIN_4_1, LNA 4 = 12.5dB */ 836 (0 << 10) | 580, /* GAIN_4_2, LNA 4 */ 837 }; 838 839 #if 0 840 /* Currently unused */ 841 static const u16 rf_ramp_pwm_sband[] = { 842 253, /* max RF gain in 10th of dB */ 843 38, /* ramp_slope = 1dB of gain -> clock_ticks_per_db = clk_khz / ramp_slope -> RF_RAMP2 */ 844 961, 845 (4 << 10) | 0, /* RF_RAMP3, LNA 1 = 14.1dB */ 846 (0 << 10) | 508, /* RF_RAMP4, LNA 1 */ 847 (9 << 10) | 508, /* RF_RAMP5, LNA 2 = 11.2dB */ 848 (0 << 10) | 961, /* RF_RAMP6, LNA 2 */ 849 (0 << 10) | 0, /* RF_RAMP7, LNA 3 = 0dB */ 850 (0 << 10) | 0, /* RF_RAMP8, LNA 3 */ 851 (0 << 10) | 0, /* GAIN_4_1, LNA 4 = 0dB */ 852 (0 << 10) | 0, /* GAIN_4_2, LNA 4 */ 853 }; 854 #endif 855 856 struct slope { 857 s16 range; 858 s16 slope; 859 }; 860 static u16 slopes_to_scale(const struct slope *slopes, u8 num, s16 val) 861 { 862 u8 i; 863 u16 rest; 864 u16 ret = 0; 865 for (i = 0; i < num; i++) { 866 if (val > slopes[i].range) 867 rest = slopes[i].range; 868 else 869 rest = val; 870 ret += (rest * slopes[i].slope) / slopes[i].range; 871 val -= rest; 872 } 873 return ret; 874 } 875 876 static const struct slope dib0090_wbd_slopes[3] = { 877 {66, 120}, /* -64,-52: offset - 65 */ 878 {600, 170}, /* -52,-35: 65 - 665 */ 879 {170, 250}, /* -45,-10: 665 - 835 */ 880 }; 881 882 static s16 dib0090_wbd_to_db(struct dib0090_state *state, u16 wbd) 883 { 884 wbd &= 0x3ff; 885 if (wbd < state->wbd_offset) 886 wbd = 0; 887 else 888 wbd -= state->wbd_offset; 889 /* -64dB is the floor */ 890 return -640 + (s16) slopes_to_scale(dib0090_wbd_slopes, ARRAY_SIZE(dib0090_wbd_slopes), wbd); 891 } 892 893 static void dib0090_wbd_target(struct dib0090_state *state, u32 rf) 894 { 895 u16 offset = 250; 896 897 /* TODO : DAB digital N+/-1 interferer perfs : offset = 10 */ 898 899 if (state->current_band == BAND_VHF) 900 offset = 650; 901 #ifndef FIRMWARE_FIREFLY 902 if (state->current_band == BAND_VHF) 903 offset = state->config->wbd_vhf_offset; 904 if (state->current_band == BAND_CBAND) 905 offset = state->config->wbd_cband_offset; 906 #endif 907 908 state->wbd_target = dib0090_wbd_to_db(state, state->wbd_offset + offset); 909 dprintk("wbd-target: %d dB\n", (u32) state->wbd_target); 910 } 911 912 static const int gain_reg_addr[4] = { 913 0x08, 0x0a, 0x0f, 0x01 914 }; 915 916 static void dib0090_gain_apply(struct dib0090_state *state, s16 gain_delta, s16 top_delta, u8 force) 917 { 918 u16 rf, bb, ref; 919 u16 i, v, gain_reg[4] = { 0 }, gain; 920 const u16 *g; 921 922 if (top_delta < -511) 923 top_delta = -511; 924 if (top_delta > 511) 925 top_delta = 511; 926 927 if (force) { 928 top_delta *= (1 << WBD_ALPHA); 929 gain_delta *= (1 << GAIN_ALPHA); 930 } 931 932 if (top_delta >= ((s16) (state->rf_ramp[0] << WBD_ALPHA) - state->rf_gain_limit)) /* overflow */ 933 state->rf_gain_limit = state->rf_ramp[0] << WBD_ALPHA; 934 else 935 state->rf_gain_limit += top_delta; 936 937 if (state->rf_gain_limit < 0) /*underflow */ 938 state->rf_gain_limit = 0; 939 940 /* use gain as a temporary variable and correct current_gain */ 941 gain = ((state->rf_gain_limit >> WBD_ALPHA) + state->bb_ramp[0]) << GAIN_ALPHA; 942 if (gain_delta >= ((s16) gain - state->current_gain)) /* overflow */ 943 state->current_gain = gain; 944 else 945 state->current_gain += gain_delta; 946 /* cannot be less than 0 (only if gain_delta is less than 0 we can have current_gain < 0) */ 947 if (state->current_gain < 0) 948 state->current_gain = 0; 949 950 /* now split total gain to rf and bb gain */ 951 gain = state->current_gain >> GAIN_ALPHA; 952 953 /* requested gain is bigger than rf gain limit - ACI/WBD adjustment */ 954 if (gain > (state->rf_gain_limit >> WBD_ALPHA)) { 955 rf = state->rf_gain_limit >> WBD_ALPHA; 956 bb = gain - rf; 957 if (bb > state->bb_ramp[0]) 958 bb = state->bb_ramp[0]; 959 } else { /* high signal level -> all gains put on RF */ 960 rf = gain; 961 bb = 0; 962 } 963 964 state->gain[0] = rf; 965 state->gain[1] = bb; 966 967 /* software ramp */ 968 /* Start with RF gains */ 969 g = state->rf_ramp + 1; /* point on RF LNA1 max gain */ 970 ref = rf; 971 for (i = 0; i < 7; i++) { /* Go over all amplifiers => 5RF amps + 2 BB amps = 7 amps */ 972 if (g[0] == 0 || ref < (g[1] - g[0])) /* if total gain of the current amp is null or this amp is not concerned because it starts to work from an higher gain value */ 973 v = 0; /* force the gain to write for the current amp to be null */ 974 else if (ref >= g[1]) /* Gain to set is higher than the high working point of this amp */ 975 v = g[2]; /* force this amp to be full gain */ 976 else /* compute the value to set to this amp because we are somewhere in his range */ 977 v = ((ref - (g[1] - g[0])) * g[2]) / g[0]; 978 979 if (i == 0) /* LNA 1 reg mapping */ 980 gain_reg[0] = v; 981 else if (i == 1) /* LNA 2 reg mapping */ 982 gain_reg[0] |= v << 7; 983 else if (i == 2) /* LNA 3 reg mapping */ 984 gain_reg[1] = v; 985 else if (i == 3) /* LNA 4 reg mapping */ 986 gain_reg[1] |= v << 7; 987 else if (i == 4) /* CBAND LNA reg mapping */ 988 gain_reg[2] = v | state->rf_lt_def; 989 else if (i == 5) /* BB gain 1 reg mapping */ 990 gain_reg[3] = v << 3; 991 else if (i == 6) /* BB gain 2 reg mapping */ 992 gain_reg[3] |= v << 8; 993 994 g += 3; /* go to next gain bloc */ 995 996 /* When RF is finished, start with BB */ 997 if (i == 4) { 998 g = state->bb_ramp + 1; /* point on BB gain 1 max gain */ 999 ref = bb; 1000 } 1001 } 1002 gain_reg[3] |= state->bb_1_def; 1003 gain_reg[3] |= ((bb % 10) * 100) / 125; 1004 1005 #ifdef DEBUG_AGC 1006 dprintk("GA CALC: DB: %3d(rf) + %3d(bb) = %3d gain_reg[0]=%04x gain_reg[1]=%04x gain_reg[2]=%04x gain_reg[0]=%04x\n", rf, bb, rf + bb, 1007 gain_reg[0], gain_reg[1], gain_reg[2], gain_reg[3]); 1008 #endif 1009 1010 /* Write the amplifier regs */ 1011 for (i = 0; i < 4; i++) { 1012 v = gain_reg[i]; 1013 if (force || state->gain_reg[i] != v) { 1014 state->gain_reg[i] = v; 1015 dib0090_write_reg(state, gain_reg_addr[i], v); 1016 } 1017 } 1018 } 1019 1020 static void dib0090_set_boost(struct dib0090_state *state, int onoff) 1021 { 1022 state->bb_1_def &= 0xdfff; 1023 state->bb_1_def |= onoff << 13; 1024 } 1025 1026 static void dib0090_set_rframp(struct dib0090_state *state, const u16 * cfg) 1027 { 1028 state->rf_ramp = cfg; 1029 } 1030 1031 static void dib0090_set_rframp_pwm(struct dib0090_state *state, const u16 * cfg) 1032 { 1033 state->rf_ramp = cfg; 1034 1035 dib0090_write_reg(state, 0x2a, 0xffff); 1036 1037 dprintk("total RF gain: %ddB, step: %d\n", (u32) cfg[0], dib0090_read_reg(state, 0x2a)); 1038 1039 dib0090_write_regs(state, 0x2c, cfg + 3, 6); 1040 dib0090_write_regs(state, 0x3e, cfg + 9, 2); 1041 } 1042 1043 static void dib0090_set_bbramp(struct dib0090_state *state, const u16 * cfg) 1044 { 1045 state->bb_ramp = cfg; 1046 dib0090_set_boost(state, cfg[0] > 500); /* we want the boost if the gain is higher that 50dB */ 1047 } 1048 1049 static void dib0090_set_bbramp_pwm(struct dib0090_state *state, const u16 * cfg) 1050 { 1051 state->bb_ramp = cfg; 1052 1053 dib0090_set_boost(state, cfg[0] > 500); /* we want the boost if the gain is higher that 50dB */ 1054 1055 dib0090_write_reg(state, 0x33, 0xffff); 1056 dprintk("total BB gain: %ddB, step: %d\n", (u32) cfg[0], dib0090_read_reg(state, 0x33)); 1057 dib0090_write_regs(state, 0x35, cfg + 3, 4); 1058 } 1059 1060 void dib0090_pwm_gain_reset(struct dvb_frontend *fe) 1061 { 1062 struct dib0090_state *state = fe->tuner_priv; 1063 const u16 *bb_ramp = bb_ramp_pwm_normal; /* default baseband config */ 1064 const u16 *rf_ramp = NULL; 1065 u8 en_pwm_rf_mux = 1; 1066 1067 /* reset the AGC */ 1068 if (state->config->use_pwm_agc) { 1069 if (state->current_band == BAND_CBAND) { 1070 if (state->identity.in_soc) { 1071 bb_ramp = bb_ramp_pwm_normal_socs; 1072 if (state->identity.version == SOC_8090_P1G_11R1 || state->identity.version == SOC_8090_P1G_21R1) 1073 rf_ramp = rf_ramp_pwm_cband_8090; 1074 else if (state->identity.version == SOC_7090_P1G_11R1 || state->identity.version == SOC_7090_P1G_21R1) { 1075 if (state->config->is_dib7090e) { 1076 if (state->rf_ramp == NULL) 1077 rf_ramp = rf_ramp_pwm_cband_7090e_sensitivity; 1078 else 1079 rf_ramp = state->rf_ramp; 1080 } else 1081 rf_ramp = rf_ramp_pwm_cband_7090p; 1082 } 1083 } else 1084 rf_ramp = rf_ramp_pwm_cband; 1085 } else 1086 1087 if (state->current_band == BAND_VHF) { 1088 if (state->identity.in_soc) { 1089 bb_ramp = bb_ramp_pwm_normal_socs; 1090 /* rf_ramp = &rf_ramp_pwm_vhf_socs; */ /* TODO */ 1091 } else 1092 rf_ramp = rf_ramp_pwm_vhf; 1093 } else if (state->current_band == BAND_UHF) { 1094 if (state->identity.in_soc) { 1095 bb_ramp = bb_ramp_pwm_normal_socs; 1096 if (state->identity.version == SOC_8090_P1G_11R1 || state->identity.version == SOC_8090_P1G_21R1) 1097 rf_ramp = rf_ramp_pwm_uhf_8090; 1098 else if (state->identity.version == SOC_7090_P1G_11R1 || state->identity.version == SOC_7090_P1G_21R1) 1099 rf_ramp = rf_ramp_pwm_uhf_7090; 1100 } else 1101 rf_ramp = rf_ramp_pwm_uhf; 1102 } 1103 if (rf_ramp) 1104 dib0090_set_rframp_pwm(state, rf_ramp); 1105 dib0090_set_bbramp_pwm(state, bb_ramp); 1106 1107 /* activate the ramp generator using PWM control */ 1108 if (state->rf_ramp) 1109 dprintk("ramp RF gain = %d BAND = %s version = %d\n", 1110 state->rf_ramp[0], 1111 (state->current_band == BAND_CBAND) ? "CBAND" : "NOT CBAND", 1112 state->identity.version & 0x1f); 1113 1114 if (rf_ramp && ((state->rf_ramp && state->rf_ramp[0] == 0) || 1115 (state->current_band == BAND_CBAND && 1116 (state->identity.version & 0x1f) <= P1D_E_F))) { 1117 dprintk("DE-Engage mux for direct gain reg control\n"); 1118 en_pwm_rf_mux = 0; 1119 } else 1120 dprintk("Engage mux for PWM control\n"); 1121 1122 dib0090_write_reg(state, 0x32, (en_pwm_rf_mux << 12) | (en_pwm_rf_mux << 11)); 1123 1124 /* Set fast servo cutoff to start AGC; 0 = 1KHz ; 1 = 50Hz ; 2 = 150Hz ; 3 = 50KHz ; 4 = servo fast*/ 1125 if (state->identity.version == SOC_7090_P1G_11R1 || state->identity.version == SOC_7090_P1G_21R1) 1126 dib0090_write_reg(state, 0x04, 3); 1127 else 1128 dib0090_write_reg(state, 0x04, 1); 1129 dib0090_write_reg(state, 0x39, (1 << 10)); /* 0 gain by default */ 1130 } 1131 } 1132 EXPORT_SYMBOL(dib0090_pwm_gain_reset); 1133 1134 void dib0090_set_dc_servo(struct dvb_frontend *fe, u8 DC_servo_cutoff) 1135 { 1136 struct dib0090_state *state = fe->tuner_priv; 1137 if (DC_servo_cutoff < 4) 1138 dib0090_write_reg(state, 0x04, DC_servo_cutoff); 1139 } 1140 EXPORT_SYMBOL(dib0090_set_dc_servo); 1141 1142 static u32 dib0090_get_slow_adc_val(struct dib0090_state *state) 1143 { 1144 u16 adc_val = dib0090_read_reg(state, 0x1d); 1145 if (state->identity.in_soc) 1146 adc_val >>= 2; 1147 return adc_val; 1148 } 1149 1150 int dib0090_gain_control(struct dvb_frontend *fe) 1151 { 1152 struct dib0090_state *state = fe->tuner_priv; 1153 enum frontend_tune_state *tune_state = &state->tune_state; 1154 int ret = 10; 1155 1156 u16 wbd_val = 0; 1157 u8 apply_gain_immediatly = 1; 1158 s16 wbd_error = 0, adc_error = 0; 1159 1160 if (*tune_state == CT_AGC_START) { 1161 state->agc_freeze = 0; 1162 dib0090_write_reg(state, 0x04, 0x0); 1163 1164 #ifdef CONFIG_BAND_SBAND 1165 if (state->current_band == BAND_SBAND) { 1166 dib0090_set_rframp(state, rf_ramp_sband); 1167 dib0090_set_bbramp(state, bb_ramp_boost); 1168 } else 1169 #endif 1170 #ifdef CONFIG_BAND_VHF 1171 if (state->current_band == BAND_VHF && !state->identity.p1g) { 1172 dib0090_set_rframp(state, rf_ramp_pwm_vhf); 1173 dib0090_set_bbramp(state, bb_ramp_pwm_normal); 1174 } else 1175 #endif 1176 #ifdef CONFIG_BAND_CBAND 1177 if (state->current_band == BAND_CBAND && !state->identity.p1g) { 1178 dib0090_set_rframp(state, rf_ramp_pwm_cband); 1179 dib0090_set_bbramp(state, bb_ramp_pwm_normal); 1180 } else 1181 #endif 1182 if ((state->current_band == BAND_CBAND || state->current_band == BAND_VHF) && state->identity.p1g) { 1183 dib0090_set_rframp(state, rf_ramp_pwm_cband_7090p); 1184 dib0090_set_bbramp(state, bb_ramp_pwm_normal_socs); 1185 } else { 1186 dib0090_set_rframp(state, rf_ramp_pwm_uhf); 1187 dib0090_set_bbramp(state, bb_ramp_pwm_normal); 1188 } 1189 1190 dib0090_write_reg(state, 0x32, 0); 1191 dib0090_write_reg(state, 0x39, 0); 1192 1193 dib0090_wbd_target(state, state->current_rf); 1194 1195 state->rf_gain_limit = state->rf_ramp[0] << WBD_ALPHA; 1196 state->current_gain = ((state->rf_ramp[0] + state->bb_ramp[0]) / 2) << GAIN_ALPHA; 1197 1198 *tune_state = CT_AGC_STEP_0; 1199 } else if (!state->agc_freeze) { 1200 s16 wbd = 0, i, cnt; 1201 1202 int adc; 1203 wbd_val = dib0090_get_slow_adc_val(state); 1204 1205 if (*tune_state == CT_AGC_STEP_0) 1206 cnt = 5; 1207 else 1208 cnt = 1; 1209 1210 for (i = 0; i < cnt; i++) { 1211 wbd_val = dib0090_get_slow_adc_val(state); 1212 wbd += dib0090_wbd_to_db(state, wbd_val); 1213 } 1214 wbd /= cnt; 1215 wbd_error = state->wbd_target - wbd; 1216 1217 if (*tune_state == CT_AGC_STEP_0) { 1218 if (wbd_error < 0 && state->rf_gain_limit > 0 && !state->identity.p1g) { 1219 #ifdef CONFIG_BAND_CBAND 1220 /* in case of CBAND tune reduce first the lt_gain2 before adjusting the RF gain */ 1221 u8 ltg2 = (state->rf_lt_def >> 10) & 0x7; 1222 if (state->current_band == BAND_CBAND && ltg2) { 1223 ltg2 >>= 1; 1224 state->rf_lt_def &= ltg2 << 10; /* reduce in 3 steps from 7 to 0 */ 1225 } 1226 #endif 1227 } else { 1228 state->agc_step = 0; 1229 *tune_state = CT_AGC_STEP_1; 1230 } 1231 } else { 1232 /* calc the adc power */ 1233 adc = state->config->get_adc_power(fe); 1234 adc = (adc * ((s32) 355774) + (((s32) 1) << 20)) >> 21; /* included in [0:-700] */ 1235 1236 adc_error = (s16) (((s32) ADC_TARGET) - adc); 1237 #ifdef CONFIG_STANDARD_DAB 1238 if (state->fe->dtv_property_cache.delivery_system == STANDARD_DAB) 1239 adc_error -= 10; 1240 #endif 1241 #ifdef CONFIG_STANDARD_DVBT 1242 if (state->fe->dtv_property_cache.delivery_system == STANDARD_DVBT && 1243 (state->fe->dtv_property_cache.modulation == QAM_64 || state->fe->dtv_property_cache.modulation == QAM_16)) 1244 adc_error += 60; 1245 #endif 1246 #ifdef CONFIG_SYS_ISDBT 1247 if ((state->fe->dtv_property_cache.delivery_system == SYS_ISDBT) && (((state->fe->dtv_property_cache.layer[0].segment_count > 1248 0) 1249 && 1250 ((state->fe->dtv_property_cache.layer[0].modulation == 1251 QAM_64) 1252 || (state->fe->dtv_property_cache. 1253 layer[0].modulation == QAM_16))) 1254 || 1255 ((state->fe->dtv_property_cache.layer[1].segment_count > 1256 0) 1257 && 1258 ((state->fe->dtv_property_cache.layer[1].modulation == 1259 QAM_64) 1260 || (state->fe->dtv_property_cache. 1261 layer[1].modulation == QAM_16))) 1262 || 1263 ((state->fe->dtv_property_cache.layer[2].segment_count > 1264 0) 1265 && 1266 ((state->fe->dtv_property_cache.layer[2].modulation == 1267 QAM_64) 1268 || (state->fe->dtv_property_cache. 1269 layer[2].modulation == QAM_16))) 1270 ) 1271 ) 1272 adc_error += 60; 1273 #endif 1274 1275 if (*tune_state == CT_AGC_STEP_1) { /* quickly go to the correct range of the ADC power */ 1276 if (abs(adc_error) < 50 || state->agc_step++ > 5) { 1277 1278 #ifdef CONFIG_STANDARD_DAB 1279 if (state->fe->dtv_property_cache.delivery_system == STANDARD_DAB) { 1280 dib0090_write_reg(state, 0x02, (1 << 15) | (15 << 11) | (31 << 6) | (63)); /* cap value = 63 : narrow BB filter : Fc = 1.8MHz */ 1281 dib0090_write_reg(state, 0x04, 0x0); 1282 } else 1283 #endif 1284 { 1285 dib0090_write_reg(state, 0x02, (1 << 15) | (3 << 11) | (6 << 6) | (32)); 1286 dib0090_write_reg(state, 0x04, 0x01); /*0 = 1KHz ; 1 = 150Hz ; 2 = 50Hz ; 3 = 50KHz ; 4 = servo fast */ 1287 } 1288 1289 *tune_state = CT_AGC_STOP; 1290 } 1291 } else { 1292 /* everything higher than or equal to CT_AGC_STOP means tracking */ 1293 ret = 100; /* 10ms interval */ 1294 apply_gain_immediatly = 0; 1295 } 1296 } 1297 #ifdef DEBUG_AGC 1298 dprintk 1299 ("tune state %d, ADC = %3ddB (ADC err %3d) WBD %3ddB (WBD err %3d, WBD val SADC: %4d), RFGainLimit (TOP): %3d, signal: %3ddBm", 1300 (u32) *tune_state, (u32) adc, (u32) adc_error, (u32) wbd, (u32) wbd_error, (u32) wbd_val, 1301 (u32) state->rf_gain_limit >> WBD_ALPHA, (s32) 200 + adc - (state->current_gain >> GAIN_ALPHA)); 1302 #endif 1303 } 1304 1305 /* apply gain */ 1306 if (!state->agc_freeze) 1307 dib0090_gain_apply(state, adc_error, wbd_error, apply_gain_immediatly); 1308 return ret; 1309 } 1310 1311 EXPORT_SYMBOL(dib0090_gain_control); 1312 1313 void dib0090_get_current_gain(struct dvb_frontend *fe, u16 * rf, u16 * bb, u16 * rf_gain_limit, u16 * rflt) 1314 { 1315 struct dib0090_state *state = fe->tuner_priv; 1316 if (rf) 1317 *rf = state->gain[0]; 1318 if (bb) 1319 *bb = state->gain[1]; 1320 if (rf_gain_limit) 1321 *rf_gain_limit = state->rf_gain_limit; 1322 if (rflt) 1323 *rflt = (state->rf_lt_def >> 10) & 0x7; 1324 } 1325 1326 EXPORT_SYMBOL(dib0090_get_current_gain); 1327 1328 u16 dib0090_get_wbd_target(struct dvb_frontend *fe) 1329 { 1330 struct dib0090_state *state = fe->tuner_priv; 1331 u32 f_MHz = state->fe->dtv_property_cache.frequency / 1000000; 1332 s32 current_temp = state->temperature; 1333 s32 wbd_thot, wbd_tcold; 1334 const struct dib0090_wbd_slope *wbd = state->current_wbd_table; 1335 1336 while (f_MHz > wbd->max_freq) 1337 wbd++; 1338 1339 dprintk("using wbd-table-entry with max freq %d\n", wbd->max_freq); 1340 1341 if (current_temp < 0) 1342 current_temp = 0; 1343 if (current_temp > 128) 1344 current_temp = 128; 1345 1346 state->wbdmux &= ~(7 << 13); 1347 if (wbd->wbd_gain != 0) 1348 state->wbdmux |= (wbd->wbd_gain << 13); 1349 else 1350 state->wbdmux |= (4 << 13); 1351 1352 dib0090_write_reg(state, 0x10, state->wbdmux); 1353 1354 wbd_thot = wbd->offset_hot - (((u32) wbd->slope_hot * f_MHz) >> 6); 1355 wbd_tcold = wbd->offset_cold - (((u32) wbd->slope_cold * f_MHz) >> 6); 1356 1357 wbd_tcold += ((wbd_thot - wbd_tcold) * current_temp) >> 7; 1358 1359 state->wbd_target = dib0090_wbd_to_db(state, state->wbd_offset + wbd_tcold); 1360 dprintk("wbd-target: %d dB\n", (u32) state->wbd_target); 1361 dprintk("wbd offset applied is %d\n", wbd_tcold); 1362 1363 return state->wbd_offset + wbd_tcold; 1364 } 1365 EXPORT_SYMBOL(dib0090_get_wbd_target); 1366 1367 u16 dib0090_get_wbd_offset(struct dvb_frontend *fe) 1368 { 1369 struct dib0090_state *state = fe->tuner_priv; 1370 return state->wbd_offset; 1371 } 1372 EXPORT_SYMBOL(dib0090_get_wbd_offset); 1373 1374 int dib0090_set_switch(struct dvb_frontend *fe, u8 sw1, u8 sw2, u8 sw3) 1375 { 1376 struct dib0090_state *state = fe->tuner_priv; 1377 1378 dib0090_write_reg(state, 0x0b, (dib0090_read_reg(state, 0x0b) & 0xfff8) 1379 | ((sw3 & 1) << 2) | ((sw2 & 1) << 1) | (sw1 & 1)); 1380 1381 return 0; 1382 } 1383 EXPORT_SYMBOL(dib0090_set_switch); 1384 1385 int dib0090_set_vga(struct dvb_frontend *fe, u8 onoff) 1386 { 1387 struct dib0090_state *state = fe->tuner_priv; 1388 1389 dib0090_write_reg(state, 0x09, (dib0090_read_reg(state, 0x09) & 0x7fff) 1390 | ((onoff & 1) << 15)); 1391 return 0; 1392 } 1393 EXPORT_SYMBOL(dib0090_set_vga); 1394 1395 int dib0090_update_rframp_7090(struct dvb_frontend *fe, u8 cfg_sensitivity) 1396 { 1397 struct dib0090_state *state = fe->tuner_priv; 1398 1399 if ((!state->identity.p1g) || (!state->identity.in_soc) 1400 || ((state->identity.version != SOC_7090_P1G_21R1) 1401 && (state->identity.version != SOC_7090_P1G_11R1))) { 1402 dprintk("%s() function can only be used for dib7090P\n", __func__); 1403 return -ENODEV; 1404 } 1405 1406 if (cfg_sensitivity) 1407 state->rf_ramp = rf_ramp_pwm_cband_7090e_sensitivity; 1408 else 1409 state->rf_ramp = rf_ramp_pwm_cband_7090e_aci; 1410 dib0090_pwm_gain_reset(fe); 1411 1412 return 0; 1413 } 1414 EXPORT_SYMBOL(dib0090_update_rframp_7090); 1415 1416 static const u16 dib0090_defaults[] = { 1417 1418 25, 0x01, 1419 0x0000, 1420 0x99a0, 1421 0x6008, 1422 0x0000, 1423 0x8bcb, 1424 0x0000, 1425 0x0405, 1426 0x0000, 1427 0x0000, 1428 0x0000, 1429 0xb802, 1430 0x0300, 1431 0x2d12, 1432 0xbac0, 1433 0x7c00, 1434 0xdbb9, 1435 0x0954, 1436 0x0743, 1437 0x8000, 1438 0x0001, 1439 0x0040, 1440 0x0100, 1441 0x0000, 1442 0xe910, 1443 0x149e, 1444 1445 1, 0x1c, 1446 0xff2d, 1447 1448 1, 0x39, 1449 0x0000, 1450 1451 2, 0x1e, 1452 0x07FF, 1453 0x0007, 1454 1455 1, 0x24, 1456 EN_UHF | EN_CRYSTAL, 1457 1458 2, 0x3c, 1459 0x3ff, 1460 0x111, 1461 0 1462 }; 1463 1464 static const u16 dib0090_p1g_additionnal_defaults[] = { 1465 1, 0x05, 1466 0xabcd, 1467 1468 1, 0x11, 1469 0x00b4, 1470 1471 1, 0x1c, 1472 0xfffd, 1473 1474 1, 0x40, 1475 0x108, 1476 0 1477 }; 1478 1479 static void dib0090_set_default_config(struct dib0090_state *state, const u16 * n) 1480 { 1481 u16 l, r; 1482 1483 l = pgm_read_word(n++); 1484 while (l) { 1485 r = pgm_read_word(n++); 1486 do { 1487 dib0090_write_reg(state, r, pgm_read_word(n++)); 1488 r++; 1489 } while (--l); 1490 l = pgm_read_word(n++); 1491 } 1492 } 1493 1494 #define CAP_VALUE_MIN (u8) 9 1495 #define CAP_VALUE_MAX (u8) 40 1496 #define HR_MIN (u8) 25 1497 #define HR_MAX (u8) 40 1498 #define POLY_MIN (u8) 0 1499 #define POLY_MAX (u8) 8 1500 1501 static void dib0090_set_EFUSE(struct dib0090_state *state) 1502 { 1503 u8 c, h, n; 1504 u16 e2, e4; 1505 u16 cal; 1506 1507 e2 = dib0090_read_reg(state, 0x26); 1508 e4 = dib0090_read_reg(state, 0x28); 1509 1510 if ((state->identity.version == P1D_E_F) || 1511 (state->identity.version == P1G) || (e2 == 0xffff)) { 1512 1513 dib0090_write_reg(state, 0x22, 0x10); 1514 cal = (dib0090_read_reg(state, 0x22) >> 6) & 0x3ff; 1515 1516 if ((cal < 670) || (cal == 1023)) 1517 cal = 850; 1518 n = 165 - ((cal * 10)>>6) ; 1519 e2 = e4 = (3<<12) | (34<<6) | (n); 1520 } 1521 1522 if (e2 != e4) 1523 e2 &= e4; /* Remove the redundancy */ 1524 1525 if (e2 != 0xffff) { 1526 c = e2 & 0x3f; 1527 n = (e2 >> 12) & 0xf; 1528 h = (e2 >> 6) & 0x3f; 1529 1530 if ((c >= CAP_VALUE_MAX) || (c <= CAP_VALUE_MIN)) 1531 c = 32; 1532 else 1533 c += 14; 1534 if ((h >= HR_MAX) || (h <= HR_MIN)) 1535 h = 34; 1536 if ((n >= POLY_MAX) || (n <= POLY_MIN)) 1537 n = 3; 1538 1539 dib0090_write_reg(state, 0x13, (h << 10)); 1540 e2 = (n << 11) | ((h >> 2)<<6) | c; 1541 dib0090_write_reg(state, 0x2, e2); /* Load the BB_2 */ 1542 } 1543 } 1544 1545 static int dib0090_reset(struct dvb_frontend *fe) 1546 { 1547 struct dib0090_state *state = fe->tuner_priv; 1548 1549 dib0090_reset_digital(fe, state->config); 1550 if (dib0090_identify(fe) < 0) 1551 return -EIO; 1552 1553 #ifdef CONFIG_TUNER_DIB0090_P1B_SUPPORT 1554 if (!(state->identity.version & 0x1)) /* it is P1B - reset is already done */ 1555 return 0; 1556 #endif 1557 1558 if (!state->identity.in_soc) { 1559 if ((dib0090_read_reg(state, 0x1a) >> 5) & 0x2) 1560 dib0090_write_reg(state, 0x1b, (EN_IQADC | EN_BB | EN_BIAS | EN_DIGCLK | EN_PLL | EN_CRYSTAL)); 1561 else 1562 dib0090_write_reg(state, 0x1b, (EN_DIGCLK | EN_PLL | EN_CRYSTAL)); 1563 } 1564 1565 dib0090_set_default_config(state, dib0090_defaults); 1566 1567 if (state->identity.in_soc) 1568 dib0090_write_reg(state, 0x18, 0x2910); /* charge pump current = 0 */ 1569 1570 if (state->identity.p1g) 1571 dib0090_set_default_config(state, dib0090_p1g_additionnal_defaults); 1572 1573 /* Update the efuse : Only available for KROSUS > P1C and SOC as well*/ 1574 if (((state->identity.version & 0x1f) >= P1D_E_F) || (state->identity.in_soc)) 1575 dib0090_set_EFUSE(state); 1576 1577 /* Congigure in function of the crystal */ 1578 if (state->config->force_crystal_mode != 0) 1579 dib0090_write_reg(state, 0x14, 1580 state->config->force_crystal_mode & 3); 1581 else if (state->config->io.clock_khz >= 24000) 1582 dib0090_write_reg(state, 0x14, 1); 1583 else 1584 dib0090_write_reg(state, 0x14, 2); 1585 dprintk("Pll lock : %d\n", (dib0090_read_reg(state, 0x1a) >> 11) & 0x1); 1586 1587 state->calibrate = DC_CAL | WBD_CAL | TEMP_CAL; /* enable iq-offset-calibration and wbd-calibration when tuning next time */ 1588 1589 return 0; 1590 } 1591 1592 #define steps(u) (((u) > 15) ? ((u)-16) : (u)) 1593 #define INTERN_WAIT 10 1594 static int dib0090_get_offset(struct dib0090_state *state, enum frontend_tune_state *tune_state) 1595 { 1596 int ret = INTERN_WAIT * 10; 1597 1598 switch (*tune_state) { 1599 case CT_TUNER_STEP_2: 1600 /* Turns to positive */ 1601 dib0090_write_reg(state, 0x1f, 0x7); 1602 *tune_state = CT_TUNER_STEP_3; 1603 break; 1604 1605 case CT_TUNER_STEP_3: 1606 state->adc_diff = dib0090_read_reg(state, 0x1d); 1607 1608 /* Turns to negative */ 1609 dib0090_write_reg(state, 0x1f, 0x4); 1610 *tune_state = CT_TUNER_STEP_4; 1611 break; 1612 1613 case CT_TUNER_STEP_4: 1614 state->adc_diff -= dib0090_read_reg(state, 0x1d); 1615 *tune_state = CT_TUNER_STEP_5; 1616 ret = 0; 1617 break; 1618 1619 default: 1620 break; 1621 } 1622 1623 return ret; 1624 } 1625 1626 struct dc_calibration { 1627 u8 addr; 1628 u8 offset; 1629 u8 pga:1; 1630 u16 bb1; 1631 u8 i:1; 1632 }; 1633 1634 static const struct dc_calibration dc_table[] = { 1635 /* Step1 BB gain1= 26 with boost 1, gain 2 = 0 */ 1636 {0x06, 5, 1, (1 << 13) | (0 << 8) | (26 << 3), 1}, 1637 {0x07, 11, 1, (1 << 13) | (0 << 8) | (26 << 3), 0}, 1638 /* Step 2 BB gain 1 = 26 with boost = 1 & gain 2 = 29 */ 1639 {0x06, 0, 0, (1 << 13) | (29 << 8) | (26 << 3), 1}, 1640 {0x06, 10, 0, (1 << 13) | (29 << 8) | (26 << 3), 0}, 1641 {0}, 1642 }; 1643 1644 static const struct dc_calibration dc_p1g_table[] = { 1645 /* Step1 BB gain1= 26 with boost 1, gain 2 = 0 */ 1646 /* addr ; trim reg offset ; pga ; CTRL_BB1 value ; i or q */ 1647 {0x06, 5, 1, (1 << 13) | (0 << 8) | (15 << 3), 1}, 1648 {0x07, 11, 1, (1 << 13) | (0 << 8) | (15 << 3), 0}, 1649 /* Step 2 BB gain 1 = 26 with boost = 1 & gain 2 = 29 */ 1650 {0x06, 0, 0, (1 << 13) | (29 << 8) | (15 << 3), 1}, 1651 {0x06, 10, 0, (1 << 13) | (29 << 8) | (15 << 3), 0}, 1652 {0}, 1653 }; 1654 1655 static void dib0090_set_trim(struct dib0090_state *state) 1656 { 1657 u16 *val; 1658 1659 if (state->dc->addr == 0x07) 1660 val = &state->bb7; 1661 else 1662 val = &state->bb6; 1663 1664 *val &= ~(0x1f << state->dc->offset); 1665 *val |= state->step << state->dc->offset; 1666 1667 dib0090_write_reg(state, state->dc->addr, *val); 1668 } 1669 1670 static int dib0090_dc_offset_calibration(struct dib0090_state *state, enum frontend_tune_state *tune_state) 1671 { 1672 int ret = 0; 1673 u16 reg; 1674 1675 switch (*tune_state) { 1676 case CT_TUNER_START: 1677 dprintk("Start DC offset calibration"); 1678 1679 /* force vcm2 = 0.8V */ 1680 state->bb6 = 0; 1681 state->bb7 = 0x040d; 1682 1683 /* the LNA AND LO are off */ 1684 reg = dib0090_read_reg(state, 0x24) & 0x0ffb; /* shutdown lna and lo */ 1685 dib0090_write_reg(state, 0x24, reg); 1686 1687 state->wbdmux = dib0090_read_reg(state, 0x10); 1688 dib0090_write_reg(state, 0x10, (state->wbdmux & ~(0xff << 3)) | (0x7 << 3) | 0x3); 1689 dib0090_write_reg(state, 0x23, dib0090_read_reg(state, 0x23) & ~(1 << 14)); 1690 1691 state->dc = dc_table; 1692 1693 if (state->identity.p1g) 1694 state->dc = dc_p1g_table; 1695 1696 /* fall through */ 1697 case CT_TUNER_STEP_0: 1698 dprintk("Start/continue DC calibration for %s path\n", 1699 (state->dc->i == 1) ? "I" : "Q"); 1700 dib0090_write_reg(state, 0x01, state->dc->bb1); 1701 dib0090_write_reg(state, 0x07, state->bb7 | (state->dc->i << 7)); 1702 1703 state->step = 0; 1704 state->min_adc_diff = 1023; 1705 *tune_state = CT_TUNER_STEP_1; 1706 ret = 50; 1707 break; 1708 1709 case CT_TUNER_STEP_1: 1710 dib0090_set_trim(state); 1711 *tune_state = CT_TUNER_STEP_2; 1712 break; 1713 1714 case CT_TUNER_STEP_2: 1715 case CT_TUNER_STEP_3: 1716 case CT_TUNER_STEP_4: 1717 ret = dib0090_get_offset(state, tune_state); 1718 break; 1719 1720 case CT_TUNER_STEP_5: /* found an offset */ 1721 dprintk("adc_diff = %d, current step= %d\n", (u32) state->adc_diff, state->step); 1722 if (state->step == 0 && state->adc_diff < 0) { 1723 state->min_adc_diff = -1023; 1724 dprintk("Change of sign of the minimum adc diff\n"); 1725 } 1726 1727 dprintk("adc_diff = %d, min_adc_diff = %d current_step = %d\n", state->adc_diff, state->min_adc_diff, state->step); 1728 1729 /* first turn for this frequency */ 1730 if (state->step == 0) { 1731 if (state->dc->pga && state->adc_diff < 0) 1732 state->step = 0x10; 1733 if (state->dc->pga == 0 && state->adc_diff > 0) 1734 state->step = 0x10; 1735 } 1736 1737 /* Look for a change of Sign in the Adc_diff.min_adc_diff is used to STORE the setp N-1 */ 1738 if ((state->adc_diff & 0x8000) == (state->min_adc_diff & 0x8000) && steps(state->step) < 15) { 1739 /* stop search when the delta the sign is changing and Steps =15 and Step=0 is force for continuance */ 1740 state->step++; 1741 state->min_adc_diff = state->adc_diff; 1742 *tune_state = CT_TUNER_STEP_1; 1743 } else { 1744 /* the minimum was what we have seen in the step before */ 1745 if (abs(state->adc_diff) > abs(state->min_adc_diff)) { 1746 dprintk("Since adc_diff N = %d > adc_diff step N-1 = %d, Come back one step\n", state->adc_diff, state->min_adc_diff); 1747 state->step--; 1748 } 1749 1750 dib0090_set_trim(state); 1751 dprintk("BB Offset Cal, BBreg=%u,Offset=%d,Value Set=%d\n", 1752 state->dc->addr, state->adc_diff, state->step); 1753 1754 state->dc++; 1755 if (state->dc->addr == 0) /* done */ 1756 *tune_state = CT_TUNER_STEP_6; 1757 else 1758 *tune_state = CT_TUNER_STEP_0; 1759 1760 } 1761 break; 1762 1763 case CT_TUNER_STEP_6: 1764 dib0090_write_reg(state, 0x07, state->bb7 & ~0x0008); 1765 dib0090_write_reg(state, 0x1f, 0x7); 1766 *tune_state = CT_TUNER_START; /* reset done -> real tuning can now begin */ 1767 state->calibrate &= ~DC_CAL; 1768 default: 1769 break; 1770 } 1771 return ret; 1772 } 1773 1774 static int dib0090_wbd_calibration(struct dib0090_state *state, enum frontend_tune_state *tune_state) 1775 { 1776 u8 wbd_gain; 1777 const struct dib0090_wbd_slope *wbd = state->current_wbd_table; 1778 1779 switch (*tune_state) { 1780 case CT_TUNER_START: 1781 while (state->current_rf / 1000 > wbd->max_freq) 1782 wbd++; 1783 if (wbd->wbd_gain != 0) 1784 wbd_gain = wbd->wbd_gain; 1785 else { 1786 wbd_gain = 4; 1787 #if defined(CONFIG_BAND_LBAND) || defined(CONFIG_BAND_SBAND) 1788 if ((state->current_band == BAND_LBAND) || (state->current_band == BAND_SBAND)) 1789 wbd_gain = 2; 1790 #endif 1791 } 1792 1793 if (wbd_gain == state->wbd_calibration_gain) { /* the WBD calibration has already been done */ 1794 *tune_state = CT_TUNER_START; 1795 state->calibrate &= ~WBD_CAL; 1796 return 0; 1797 } 1798 1799 dib0090_write_reg(state, 0x10, 0x1b81 | (1 << 10) | (wbd_gain << 13) | (1 << 3)); 1800 1801 dib0090_write_reg(state, 0x24, ((EN_UHF & 0x0fff) | (1 << 1))); 1802 *tune_state = CT_TUNER_STEP_0; 1803 state->wbd_calibration_gain = wbd_gain; 1804 return 90; /* wait for the WBDMUX to switch and for the ADC to sample */ 1805 1806 case CT_TUNER_STEP_0: 1807 state->wbd_offset = dib0090_get_slow_adc_val(state); 1808 dprintk("WBD calibration offset = %d\n", state->wbd_offset); 1809 *tune_state = CT_TUNER_START; /* reset done -> real tuning can now begin */ 1810 state->calibrate &= ~WBD_CAL; 1811 break; 1812 1813 default: 1814 break; 1815 } 1816 return 0; 1817 } 1818 1819 static void dib0090_set_bandwidth(struct dib0090_state *state) 1820 { 1821 u16 tmp; 1822 1823 if (state->fe->dtv_property_cache.bandwidth_hz / 1000 <= 5000) 1824 tmp = (3 << 14); 1825 else if (state->fe->dtv_property_cache.bandwidth_hz / 1000 <= 6000) 1826 tmp = (2 << 14); 1827 else if (state->fe->dtv_property_cache.bandwidth_hz / 1000 <= 7000) 1828 tmp = (1 << 14); 1829 else 1830 tmp = (0 << 14); 1831 1832 state->bb_1_def &= 0x3fff; 1833 state->bb_1_def |= tmp; 1834 1835 dib0090_write_reg(state, 0x01, state->bb_1_def); /* be sure that we have the right bb-filter */ 1836 1837 dib0090_write_reg(state, 0x03, 0x6008); /* = 0x6008 : vcm3_trim = 1 ; filter2_gm1_trim = 8 ; filter2_cutoff_freq = 0 */ 1838 dib0090_write_reg(state, 0x04, 0x1); /* 0 = 1KHz ; 1 = 50Hz ; 2 = 150Hz ; 3 = 50KHz ; 4 = servo fast */ 1839 if (state->identity.in_soc) { 1840 dib0090_write_reg(state, 0x05, 0x9bcf); /* attenuator_ibias_tri = 2 ; input_stage_ibias_tr = 1 ; nc = 11 ; ext_gm_trim = 1 ; obuf_ibias_trim = 4 ; filter13_gm2_ibias_t = 15 */ 1841 } else { 1842 dib0090_write_reg(state, 0x02, (5 << 11) | (8 << 6) | (22 & 0x3f)); /* 22 = cap_value */ 1843 dib0090_write_reg(state, 0x05, 0xabcd); /* = 0xabcd : attenuator_ibias_tri = 2 ; input_stage_ibias_tr = 2 ; nc = 11 ; ext_gm_trim = 1 ; obuf_ibias_trim = 4 ; filter13_gm2_ibias_t = 13 */ 1844 } 1845 } 1846 1847 static const struct dib0090_pll dib0090_pll_table[] = { 1848 #ifdef CONFIG_BAND_CBAND 1849 {56000, 0, 9, 48, 6}, 1850 {70000, 1, 9, 48, 6}, 1851 {87000, 0, 8, 32, 4}, 1852 {105000, 1, 8, 32, 4}, 1853 {115000, 0, 7, 24, 6}, 1854 {140000, 1, 7, 24, 6}, 1855 {170000, 0, 6, 16, 4}, 1856 #endif 1857 #ifdef CONFIG_BAND_VHF 1858 {200000, 1, 6, 16, 4}, 1859 {230000, 0, 5, 12, 6}, 1860 {280000, 1, 5, 12, 6}, 1861 {340000, 0, 4, 8, 4}, 1862 {380000, 1, 4, 8, 4}, 1863 {450000, 0, 3, 6, 6}, 1864 #endif 1865 #ifdef CONFIG_BAND_UHF 1866 {580000, 1, 3, 6, 6}, 1867 {700000, 0, 2, 4, 4}, 1868 {860000, 1, 2, 4, 4}, 1869 #endif 1870 #ifdef CONFIG_BAND_LBAND 1871 {1800000, 1, 0, 2, 4}, 1872 #endif 1873 #ifdef CONFIG_BAND_SBAND 1874 {2900000, 0, 14, 1, 4}, 1875 #endif 1876 }; 1877 1878 static const struct dib0090_tuning dib0090_tuning_table_fm_vhf_on_cband[] = { 1879 1880 #ifdef CONFIG_BAND_CBAND 1881 {184000, 4, 1, 15, 0x280, 0x2912, 0xb94e, EN_CAB}, 1882 {227000, 4, 3, 15, 0x280, 0x2912, 0xb94e, EN_CAB}, 1883 {380000, 4, 7, 15, 0x280, 0x2912, 0xb94e, EN_CAB}, 1884 #endif 1885 #ifdef CONFIG_BAND_UHF 1886 {520000, 2, 0, 15, 0x300, 0x1d12, 0xb9ce, EN_UHF}, 1887 {550000, 2, 2, 15, 0x300, 0x1d12, 0xb9ce, EN_UHF}, 1888 {650000, 2, 3, 15, 0x300, 0x1d12, 0xb9ce, EN_UHF}, 1889 {750000, 2, 5, 15, 0x300, 0x1d12, 0xb9ce, EN_UHF}, 1890 {850000, 2, 6, 15, 0x300, 0x1d12, 0xb9ce, EN_UHF}, 1891 {900000, 2, 7, 15, 0x300, 0x1d12, 0xb9ce, EN_UHF}, 1892 #endif 1893 #ifdef CONFIG_BAND_LBAND 1894 {1500000, 4, 0, 20, 0x300, 0x1912, 0x82c9, EN_LBD}, 1895 {1600000, 4, 1, 20, 0x300, 0x1912, 0x82c9, EN_LBD}, 1896 {1800000, 4, 3, 20, 0x300, 0x1912, 0x82c9, EN_LBD}, 1897 #endif 1898 #ifdef CONFIG_BAND_SBAND 1899 {2300000, 1, 4, 20, 0x300, 0x2d2A, 0x82c7, EN_SBD}, 1900 {2900000, 1, 7, 20, 0x280, 0x2deb, 0x8347, EN_SBD}, 1901 #endif 1902 }; 1903 1904 static const struct dib0090_tuning dib0090_tuning_table[] = { 1905 1906 #ifdef CONFIG_BAND_CBAND 1907 {170000, 4, 1, 15, 0x280, 0x2912, 0xb94e, EN_CAB}, 1908 #endif 1909 #ifdef CONFIG_BAND_VHF 1910 {184000, 1, 1, 15, 0x300, 0x4d12, 0xb94e, EN_VHF}, 1911 {227000, 1, 3, 15, 0x300, 0x4d12, 0xb94e, EN_VHF}, 1912 {380000, 1, 7, 15, 0x300, 0x4d12, 0xb94e, EN_VHF}, 1913 #endif 1914 #ifdef CONFIG_BAND_UHF 1915 {520000, 2, 0, 15, 0x300, 0x1d12, 0xb9ce, EN_UHF}, 1916 {550000, 2, 2, 15, 0x300, 0x1d12, 0xb9ce, EN_UHF}, 1917 {650000, 2, 3, 15, 0x300, 0x1d12, 0xb9ce, EN_UHF}, 1918 {750000, 2, 5, 15, 0x300, 0x1d12, 0xb9ce, EN_UHF}, 1919 {850000, 2, 6, 15, 0x300, 0x1d12, 0xb9ce, EN_UHF}, 1920 {900000, 2, 7, 15, 0x300, 0x1d12, 0xb9ce, EN_UHF}, 1921 #endif 1922 #ifdef CONFIG_BAND_LBAND 1923 {1500000, 4, 0, 20, 0x300, 0x1912, 0x82c9, EN_LBD}, 1924 {1600000, 4, 1, 20, 0x300, 0x1912, 0x82c9, EN_LBD}, 1925 {1800000, 4, 3, 20, 0x300, 0x1912, 0x82c9, EN_LBD}, 1926 #endif 1927 #ifdef CONFIG_BAND_SBAND 1928 {2300000, 1, 4, 20, 0x300, 0x2d2A, 0x82c7, EN_SBD}, 1929 {2900000, 1, 7, 20, 0x280, 0x2deb, 0x8347, EN_SBD}, 1930 #endif 1931 }; 1932 1933 static const struct dib0090_tuning dib0090_p1g_tuning_table[] = { 1934 #ifdef CONFIG_BAND_CBAND 1935 {170000, 4, 1, 0x820f, 0x300, 0x2d22, 0x82cb, EN_CAB}, 1936 #endif 1937 #ifdef CONFIG_BAND_VHF 1938 {184000, 1, 1, 15, 0x300, 0x4d12, 0xb94e, EN_VHF}, 1939 {227000, 1, 3, 15, 0x300, 0x4d12, 0xb94e, EN_VHF}, 1940 {380000, 1, 7, 15, 0x300, 0x4d12, 0xb94e, EN_VHF}, 1941 #endif 1942 #ifdef CONFIG_BAND_UHF 1943 {510000, 2, 0, 15, 0x300, 0x1d12, 0xb9ce, EN_UHF}, 1944 {540000, 2, 1, 15, 0x300, 0x1d12, 0xb9ce, EN_UHF}, 1945 {600000, 2, 3, 15, 0x300, 0x1d12, 0xb9ce, EN_UHF}, 1946 {630000, 2, 4, 15, 0x300, 0x1d12, 0xb9ce, EN_UHF}, 1947 {680000, 2, 5, 15, 0x300, 0x1d12, 0xb9ce, EN_UHF}, 1948 {720000, 2, 6, 15, 0x300, 0x1d12, 0xb9ce, EN_UHF}, 1949 {900000, 2, 7, 15, 0x300, 0x1d12, 0xb9ce, EN_UHF}, 1950 #endif 1951 #ifdef CONFIG_BAND_LBAND 1952 {1500000, 4, 0, 20, 0x300, 0x1912, 0x82c9, EN_LBD}, 1953 {1600000, 4, 1, 20, 0x300, 0x1912, 0x82c9, EN_LBD}, 1954 {1800000, 4, 3, 20, 0x300, 0x1912, 0x82c9, EN_LBD}, 1955 #endif 1956 #ifdef CONFIG_BAND_SBAND 1957 {2300000, 1, 4, 20, 0x300, 0x2d2A, 0x82c7, EN_SBD}, 1958 {2900000, 1, 7, 20, 0x280, 0x2deb, 0x8347, EN_SBD}, 1959 #endif 1960 }; 1961 1962 static const struct dib0090_pll dib0090_p1g_pll_table[] = { 1963 #ifdef CONFIG_BAND_CBAND 1964 {57000, 0, 11, 48, 6}, 1965 {70000, 1, 11, 48, 6}, 1966 {86000, 0, 10, 32, 4}, 1967 {105000, 1, 10, 32, 4}, 1968 {115000, 0, 9, 24, 6}, 1969 {140000, 1, 9, 24, 6}, 1970 {170000, 0, 8, 16, 4}, 1971 #endif 1972 #ifdef CONFIG_BAND_VHF 1973 {200000, 1, 8, 16, 4}, 1974 {230000, 0, 7, 12, 6}, 1975 {280000, 1, 7, 12, 6}, 1976 {340000, 0, 6, 8, 4}, 1977 {380000, 1, 6, 8, 4}, 1978 {455000, 0, 5, 6, 6}, 1979 #endif 1980 #ifdef CONFIG_BAND_UHF 1981 {580000, 1, 5, 6, 6}, 1982 {680000, 0, 4, 4, 4}, 1983 {860000, 1, 4, 4, 4}, 1984 #endif 1985 #ifdef CONFIG_BAND_LBAND 1986 {1800000, 1, 2, 2, 4}, 1987 #endif 1988 #ifdef CONFIG_BAND_SBAND 1989 {2900000, 0, 1, 1, 6}, 1990 #endif 1991 }; 1992 1993 static const struct dib0090_tuning dib0090_p1g_tuning_table_fm_vhf_on_cband[] = { 1994 #ifdef CONFIG_BAND_CBAND 1995 {184000, 4, 3, 0x4187, 0x2c0, 0x2d22, 0x81cb, EN_CAB}, 1996 {227000, 4, 3, 0x4187, 0x2c0, 0x2d22, 0x81cb, EN_CAB}, 1997 {380000, 4, 3, 0x4187, 0x2c0, 0x2d22, 0x81cb, EN_CAB}, 1998 #endif 1999 #ifdef CONFIG_BAND_UHF 2000 {520000, 2, 0, 15, 0x300, 0x1d12, 0xb9ce, EN_UHF}, 2001 {550000, 2, 2, 15, 0x300, 0x1d12, 0xb9ce, EN_UHF}, 2002 {650000, 2, 3, 15, 0x300, 0x1d12, 0xb9ce, EN_UHF}, 2003 {750000, 2, 5, 15, 0x300, 0x1d12, 0xb9ce, EN_UHF}, 2004 {850000, 2, 6, 15, 0x300, 0x1d12, 0xb9ce, EN_UHF}, 2005 {900000, 2, 7, 15, 0x300, 0x1d12, 0xb9ce, EN_UHF}, 2006 #endif 2007 #ifdef CONFIG_BAND_LBAND 2008 {1500000, 4, 0, 20, 0x300, 0x1912, 0x82c9, EN_LBD}, 2009 {1600000, 4, 1, 20, 0x300, 0x1912, 0x82c9, EN_LBD}, 2010 {1800000, 4, 3, 20, 0x300, 0x1912, 0x82c9, EN_LBD}, 2011 #endif 2012 #ifdef CONFIG_BAND_SBAND 2013 {2300000, 1, 4, 20, 0x300, 0x2d2A, 0x82c7, EN_SBD}, 2014 {2900000, 1, 7, 20, 0x280, 0x2deb, 0x8347, EN_SBD}, 2015 #endif 2016 }; 2017 2018 static const struct dib0090_tuning dib0090_tuning_table_cband_7090[] = { 2019 #ifdef CONFIG_BAND_CBAND 2020 {300000, 4, 3, 0x018F, 0x2c0, 0x2d22, 0xb9ce, EN_CAB}, 2021 {380000, 4, 10, 0x018F, 0x2c0, 0x2d22, 0xb9ce, EN_CAB}, 2022 {570000, 4, 10, 0x8190, 0x2c0, 0x2d22, 0xb9ce, EN_CAB}, 2023 {858000, 4, 5, 0x8190, 0x2c0, 0x2d22, 0xb9ce, EN_CAB}, 2024 #endif 2025 }; 2026 2027 static const struct dib0090_tuning dib0090_tuning_table_cband_7090e_sensitivity[] = { 2028 #ifdef CONFIG_BAND_CBAND 2029 { 300000, 0 , 3, 0x8105, 0x2c0, 0x2d12, 0xb84e, EN_CAB }, 2030 { 380000, 0 , 10, 0x810F, 0x2c0, 0x2d12, 0xb84e, EN_CAB }, 2031 { 600000, 0 , 10, 0x815E, 0x280, 0x2d12, 0xb84e, EN_CAB }, 2032 { 660000, 0 , 5, 0x85E3, 0x280, 0x2d12, 0xb84e, EN_CAB }, 2033 { 720000, 0 , 5, 0x852E, 0x280, 0x2d12, 0xb84e, EN_CAB }, 2034 { 860000, 0 , 4, 0x85E5, 0x280, 0x2d12, 0xb84e, EN_CAB }, 2035 #endif 2036 }; 2037 2038 int dib0090_update_tuning_table_7090(struct dvb_frontend *fe, 2039 u8 cfg_sensitivity) 2040 { 2041 struct dib0090_state *state = fe->tuner_priv; 2042 const struct dib0090_tuning *tune = 2043 dib0090_tuning_table_cband_7090e_sensitivity; 2044 static const struct dib0090_tuning dib0090_tuning_table_cband_7090e_aci[] = { 2045 { 300000, 0 , 3, 0x8165, 0x2c0, 0x2d12, 0xb84e, EN_CAB }, 2046 { 650000, 0 , 4, 0x815B, 0x280, 0x2d12, 0xb84e, EN_CAB }, 2047 { 860000, 0 , 5, 0x84EF, 0x280, 0x2d12, 0xb84e, EN_CAB }, 2048 }; 2049 2050 if ((!state->identity.p1g) || (!state->identity.in_soc) 2051 || ((state->identity.version != SOC_7090_P1G_21R1) 2052 && (state->identity.version != SOC_7090_P1G_11R1))) { 2053 dprintk("%s() function can only be used for dib7090\n", __func__); 2054 return -ENODEV; 2055 } 2056 2057 if (cfg_sensitivity) 2058 tune = dib0090_tuning_table_cband_7090e_sensitivity; 2059 else 2060 tune = dib0090_tuning_table_cband_7090e_aci; 2061 2062 while (state->rf_request > tune->max_freq) 2063 tune++; 2064 2065 dib0090_write_reg(state, 0x09, (dib0090_read_reg(state, 0x09) & 0x8000) 2066 | (tune->lna_bias & 0x7fff)); 2067 dib0090_write_reg(state, 0x0b, (dib0090_read_reg(state, 0x0b) & 0xf83f) 2068 | ((tune->lna_tune << 6) & 0x07c0)); 2069 return 0; 2070 } 2071 EXPORT_SYMBOL(dib0090_update_tuning_table_7090); 2072 2073 static int dib0090_captrim_search(struct dib0090_state *state, enum frontend_tune_state *tune_state) 2074 { 2075 int ret = 0; 2076 u16 lo4 = 0xe900; 2077 2078 s16 adc_target; 2079 u16 adc; 2080 s8 step_sign; 2081 u8 force_soft_search = 0; 2082 2083 if (state->identity.version == SOC_8090_P1G_11R1 || state->identity.version == SOC_8090_P1G_21R1) 2084 force_soft_search = 1; 2085 2086 if (*tune_state == CT_TUNER_START) { 2087 dprintk("Start Captrim search : %s\n", 2088 (force_soft_search == 1) ? "FORCE SOFT SEARCH" : "AUTO"); 2089 dib0090_write_reg(state, 0x10, 0x2B1); 2090 dib0090_write_reg(state, 0x1e, 0x0032); 2091 2092 if (!state->tuner_is_tuned) { 2093 /* prepare a complete captrim */ 2094 if (!state->identity.p1g || force_soft_search) 2095 state->step = state->captrim = state->fcaptrim = 64; 2096 2097 state->current_rf = state->rf_request; 2098 } else { /* we are already tuned to this frequency - the configuration is correct */ 2099 if (!state->identity.p1g || force_soft_search) { 2100 /* do a minimal captrim even if the frequency has not changed */ 2101 state->step = 4; 2102 state->captrim = state->fcaptrim = dib0090_read_reg(state, 0x18) & 0x7f; 2103 } 2104 } 2105 state->adc_diff = 3000; 2106 *tune_state = CT_TUNER_STEP_0; 2107 2108 } else if (*tune_state == CT_TUNER_STEP_0) { 2109 if (state->identity.p1g && !force_soft_search) { 2110 u8 ratio = 31; 2111 2112 dib0090_write_reg(state, 0x40, (3 << 7) | (ratio << 2) | (1 << 1) | 1); 2113 dib0090_read_reg(state, 0x40); 2114 ret = 50; 2115 } else { 2116 state->step /= 2; 2117 dib0090_write_reg(state, 0x18, lo4 | state->captrim); 2118 2119 if (state->identity.in_soc) 2120 ret = 25; 2121 } 2122 *tune_state = CT_TUNER_STEP_1; 2123 2124 } else if (*tune_state == CT_TUNER_STEP_1) { 2125 if (state->identity.p1g && !force_soft_search) { 2126 dib0090_write_reg(state, 0x40, 0x18c | (0 << 1) | 0); 2127 dib0090_read_reg(state, 0x40); 2128 2129 state->fcaptrim = dib0090_read_reg(state, 0x18) & 0x7F; 2130 dprintk("***Final Captrim= 0x%x\n", state->fcaptrim); 2131 *tune_state = CT_TUNER_STEP_3; 2132 2133 } else { 2134 /* MERGE for all krosus before P1G */ 2135 adc = dib0090_get_slow_adc_val(state); 2136 dprintk("CAPTRIM=%d; ADC = %d (ADC) & %dmV\n", (u32) state->captrim, (u32) adc, (u32) (adc) * (u32) 1800 / (u32) 1024); 2137 2138 if (state->rest == 0 || state->identity.in_soc) { /* Just for 8090P SOCS where auto captrim HW bug : TO CHECK IN ACI for SOCS !!! if 400 for 8090p SOC => tune issue !!! */ 2139 adc_target = 200; 2140 } else 2141 adc_target = 400; 2142 2143 if (adc >= adc_target) { 2144 adc -= adc_target; 2145 step_sign = -1; 2146 } else { 2147 adc = adc_target - adc; 2148 step_sign = 1; 2149 } 2150 2151 if (adc < state->adc_diff) { 2152 dprintk("CAPTRIM=%d is closer to target (%d/%d)\n", (u32) state->captrim, (u32) adc, (u32) state->adc_diff); 2153 state->adc_diff = adc; 2154 state->fcaptrim = state->captrim; 2155 } 2156 2157 state->captrim += step_sign * state->step; 2158 if (state->step >= 1) 2159 *tune_state = CT_TUNER_STEP_0; 2160 else 2161 *tune_state = CT_TUNER_STEP_2; 2162 2163 ret = 25; 2164 } 2165 } else if (*tune_state == CT_TUNER_STEP_2) { /* this step is only used by krosus < P1G */ 2166 /*write the final cptrim config */ 2167 dib0090_write_reg(state, 0x18, lo4 | state->fcaptrim); 2168 2169 *tune_state = CT_TUNER_STEP_3; 2170 2171 } else if (*tune_state == CT_TUNER_STEP_3) { 2172 state->calibrate &= ~CAPTRIM_CAL; 2173 *tune_state = CT_TUNER_STEP_0; 2174 } 2175 2176 return ret; 2177 } 2178 2179 static int dib0090_get_temperature(struct dib0090_state *state, enum frontend_tune_state *tune_state) 2180 { 2181 int ret = 15; 2182 s16 val; 2183 2184 switch (*tune_state) { 2185 case CT_TUNER_START: 2186 state->wbdmux = dib0090_read_reg(state, 0x10); 2187 dib0090_write_reg(state, 0x10, (state->wbdmux & ~(0xff << 3)) | (0x8 << 3)); 2188 2189 state->bias = dib0090_read_reg(state, 0x13); 2190 dib0090_write_reg(state, 0x13, state->bias | (0x3 << 8)); 2191 2192 *tune_state = CT_TUNER_STEP_0; 2193 /* wait for the WBDMUX to switch and for the ADC to sample */ 2194 break; 2195 2196 case CT_TUNER_STEP_0: 2197 state->adc_diff = dib0090_get_slow_adc_val(state); 2198 dib0090_write_reg(state, 0x13, (state->bias & ~(0x3 << 8)) | (0x2 << 8)); 2199 *tune_state = CT_TUNER_STEP_1; 2200 break; 2201 2202 case CT_TUNER_STEP_1: 2203 val = dib0090_get_slow_adc_val(state); 2204 state->temperature = ((s16) ((val - state->adc_diff) * 180) >> 8) + 55; 2205 2206 dprintk("temperature: %d C\n", state->temperature - 30); 2207 2208 *tune_state = CT_TUNER_STEP_2; 2209 break; 2210 2211 case CT_TUNER_STEP_2: 2212 dib0090_write_reg(state, 0x13, state->bias); 2213 dib0090_write_reg(state, 0x10, state->wbdmux); /* write back original WBDMUX */ 2214 2215 *tune_state = CT_TUNER_START; 2216 state->calibrate &= ~TEMP_CAL; 2217 if (state->config->analog_output == 0) 2218 dib0090_write_reg(state, 0x23, dib0090_read_reg(state, 0x23) | (1 << 14)); 2219 2220 break; 2221 2222 default: 2223 ret = 0; 2224 break; 2225 } 2226 return ret; 2227 } 2228 2229 #define WBD 0x781 /* 1 1 1 1 0000 0 0 1 */ 2230 static int dib0090_tune(struct dvb_frontend *fe) 2231 { 2232 struct dib0090_state *state = fe->tuner_priv; 2233 const struct dib0090_tuning *tune = state->current_tune_table_index; 2234 const struct dib0090_pll *pll = state->current_pll_table_index; 2235 enum frontend_tune_state *tune_state = &state->tune_state; 2236 2237 u16 lo5, lo6, Den, tmp; 2238 u32 FBDiv, Rest, FREF, VCOF_kHz = 0; 2239 int ret = 10; /* 1ms is the default delay most of the time */ 2240 u8 c, i; 2241 2242 /************************* VCO ***************************/ 2243 /* Default values for FG */ 2244 /* from these are needed : */ 2245 /* Cp,HFdiv,VCOband,SD,Num,Den,FB and REFDiv */ 2246 2247 /* in any case we first need to do a calibration if needed */ 2248 if (*tune_state == CT_TUNER_START) { 2249 /* deactivate DataTX before some calibrations */ 2250 if (state->calibrate & (DC_CAL | TEMP_CAL | WBD_CAL)) 2251 dib0090_write_reg(state, 0x23, dib0090_read_reg(state, 0x23) & ~(1 << 14)); 2252 else 2253 /* Activate DataTX in case a calibration has been done before */ 2254 if (state->config->analog_output == 0) 2255 dib0090_write_reg(state, 0x23, dib0090_read_reg(state, 0x23) | (1 << 14)); 2256 } 2257 2258 if (state->calibrate & DC_CAL) 2259 return dib0090_dc_offset_calibration(state, tune_state); 2260 else if (state->calibrate & WBD_CAL) { 2261 if (state->current_rf == 0) 2262 state->current_rf = state->fe->dtv_property_cache.frequency / 1000; 2263 return dib0090_wbd_calibration(state, tune_state); 2264 } else if (state->calibrate & TEMP_CAL) 2265 return dib0090_get_temperature(state, tune_state); 2266 else if (state->calibrate & CAPTRIM_CAL) 2267 return dib0090_captrim_search(state, tune_state); 2268 2269 if (*tune_state == CT_TUNER_START) { 2270 /* if soc and AGC pwm control, disengage mux to be able to R/W access to 0x01 register to set the right filter (cutoff_freq_select) during the tune sequence, otherwise, SOC SERPAR error when accessing to 0x01 */ 2271 if (state->config->use_pwm_agc && state->identity.in_soc) { 2272 tmp = dib0090_read_reg(state, 0x39); 2273 if ((tmp >> 10) & 0x1) 2274 dib0090_write_reg(state, 0x39, tmp & ~(1 << 10)); 2275 } 2276 2277 state->current_band = (u8) BAND_OF_FREQUENCY(state->fe->dtv_property_cache.frequency / 1000); 2278 state->rf_request = 2279 state->fe->dtv_property_cache.frequency / 1000 + (state->current_band == 2280 BAND_UHF ? state->config->freq_offset_khz_uhf : state->config-> 2281 freq_offset_khz_vhf); 2282 2283 /* in ISDB-T 1seg we shift tuning frequency */ 2284 if ((state->fe->dtv_property_cache.delivery_system == SYS_ISDBT && state->fe->dtv_property_cache.isdbt_sb_mode == 1 2285 && state->fe->dtv_property_cache.isdbt_partial_reception == 0)) { 2286 const struct dib0090_low_if_offset_table *LUT_offset = state->config->low_if; 2287 u8 found_offset = 0; 2288 u32 margin_khz = 100; 2289 2290 if (LUT_offset != NULL) { 2291 while (LUT_offset->RF_freq != 0xffff) { 2292 if (((state->rf_request > (LUT_offset->RF_freq - margin_khz)) 2293 && (state->rf_request < (LUT_offset->RF_freq + margin_khz))) 2294 && LUT_offset->std == state->fe->dtv_property_cache.delivery_system) { 2295 state->rf_request += LUT_offset->offset_khz; 2296 found_offset = 1; 2297 break; 2298 } 2299 LUT_offset++; 2300 } 2301 } 2302 2303 if (found_offset == 0) 2304 state->rf_request += 400; 2305 } 2306 if (state->current_rf != state->rf_request || (state->current_standard != state->fe->dtv_property_cache.delivery_system)) { 2307 state->tuner_is_tuned = 0; 2308 state->current_rf = 0; 2309 state->current_standard = 0; 2310 2311 tune = dib0090_tuning_table; 2312 if (state->identity.p1g) 2313 tune = dib0090_p1g_tuning_table; 2314 2315 tmp = (state->identity.version >> 5) & 0x7; 2316 2317 if (state->identity.in_soc) { 2318 if (state->config->force_cband_input) { /* Use the CBAND input for all band */ 2319 if (state->current_band & BAND_CBAND || state->current_band & BAND_FM || state->current_band & BAND_VHF 2320 || state->current_band & BAND_UHF) { 2321 state->current_band = BAND_CBAND; 2322 if (state->config->is_dib7090e) 2323 tune = dib0090_tuning_table_cband_7090e_sensitivity; 2324 else 2325 tune = dib0090_tuning_table_cband_7090; 2326 } 2327 } else { /* Use the CBAND input for all band under UHF */ 2328 if (state->current_band & BAND_CBAND || state->current_band & BAND_FM || state->current_band & BAND_VHF) { 2329 state->current_band = BAND_CBAND; 2330 if (state->config->is_dib7090e) 2331 tune = dib0090_tuning_table_cband_7090e_sensitivity; 2332 else 2333 tune = dib0090_tuning_table_cband_7090; 2334 } 2335 } 2336 } else 2337 if (tmp == 0x4 || tmp == 0x7) { 2338 /* CBAND tuner version for VHF */ 2339 if (state->current_band == BAND_FM || state->current_band == BAND_CBAND || state->current_band == BAND_VHF) { 2340 state->current_band = BAND_CBAND; /* Force CBAND */ 2341 2342 tune = dib0090_tuning_table_fm_vhf_on_cband; 2343 if (state->identity.p1g) 2344 tune = dib0090_p1g_tuning_table_fm_vhf_on_cband; 2345 } 2346 } 2347 2348 pll = dib0090_pll_table; 2349 if (state->identity.p1g) 2350 pll = dib0090_p1g_pll_table; 2351 2352 /* Look for the interval */ 2353 while (state->rf_request > tune->max_freq) 2354 tune++; 2355 while (state->rf_request > pll->max_freq) 2356 pll++; 2357 2358 state->current_tune_table_index = tune; 2359 state->current_pll_table_index = pll; 2360 2361 dib0090_write_reg(state, 0x0b, 0xb800 | (tune->switch_trim)); 2362 2363 VCOF_kHz = (pll->hfdiv * state->rf_request) * 2; 2364 2365 FREF = state->config->io.clock_khz; 2366 if (state->config->fref_clock_ratio != 0) 2367 FREF /= state->config->fref_clock_ratio; 2368 2369 FBDiv = (VCOF_kHz / pll->topresc / FREF); 2370 Rest = (VCOF_kHz / pll->topresc) - FBDiv * FREF; 2371 2372 if (Rest < LPF) 2373 Rest = 0; 2374 else if (Rest < 2 * LPF) 2375 Rest = 2 * LPF; 2376 else if (Rest > (FREF - LPF)) { 2377 Rest = 0; 2378 FBDiv += 1; 2379 } else if (Rest > (FREF - 2 * LPF)) 2380 Rest = FREF - 2 * LPF; 2381 Rest = (Rest * 6528) / (FREF / 10); 2382 state->rest = Rest; 2383 2384 /* external loop filter, otherwise: 2385 * lo5 = (0 << 15) | (0 << 12) | (0 << 11) | (3 << 9) | (4 << 6) | (3 << 4) | 4; 2386 * lo6 = 0x0e34 */ 2387 2388 if (Rest == 0) { 2389 if (pll->vco_band) 2390 lo5 = 0x049f; 2391 else 2392 lo5 = 0x041f; 2393 } else { 2394 if (pll->vco_band) 2395 lo5 = 0x049e; 2396 else if (state->config->analog_output) 2397 lo5 = 0x041d; 2398 else 2399 lo5 = 0x041c; 2400 } 2401 2402 if (state->identity.p1g) { /* Bias is done automatically in P1G */ 2403 if (state->identity.in_soc) { 2404 if (state->identity.version == SOC_8090_P1G_11R1) 2405 lo5 = 0x46f; 2406 else 2407 lo5 = 0x42f; 2408 } else 2409 lo5 = 0x42c; 2410 } 2411 2412 lo5 |= (pll->hfdiv_code << 11) | (pll->vco_band << 7); /* bit 15 is the split to the slave, we do not do it here */ 2413 2414 if (!state->config->io.pll_int_loop_filt) { 2415 if (state->identity.in_soc) 2416 lo6 = 0xff98; 2417 else if (state->identity.p1g || (Rest == 0)) 2418 lo6 = 0xfff8; 2419 else 2420 lo6 = 0xff28; 2421 } else 2422 lo6 = (state->config->io.pll_int_loop_filt << 3); 2423 2424 Den = 1; 2425 2426 if (Rest > 0) { 2427 lo6 |= (1 << 2) | 2; 2428 Den = 255; 2429 } 2430 dib0090_write_reg(state, 0x15, (u16) FBDiv); 2431 if (state->config->fref_clock_ratio != 0) 2432 dib0090_write_reg(state, 0x16, (Den << 8) | state->config->fref_clock_ratio); 2433 else 2434 dib0090_write_reg(state, 0x16, (Den << 8) | 1); 2435 dib0090_write_reg(state, 0x17, (u16) Rest); 2436 dib0090_write_reg(state, 0x19, lo5); 2437 dib0090_write_reg(state, 0x1c, lo6); 2438 2439 lo6 = tune->tuner_enable; 2440 if (state->config->analog_output) 2441 lo6 = (lo6 & 0xff9f) | 0x2; 2442 2443 dib0090_write_reg(state, 0x24, lo6 | EN_LO | state->config->use_pwm_agc * EN_CRYSTAL); 2444 2445 } 2446 2447 state->current_rf = state->rf_request; 2448 state->current_standard = state->fe->dtv_property_cache.delivery_system; 2449 2450 ret = 20; 2451 state->calibrate = CAPTRIM_CAL; /* captrim search now */ 2452 } 2453 2454 else if (*tune_state == CT_TUNER_STEP_0) { /* Warning : because of captrim cal, if you change this step, change it also in _cal.c file because it is the step following captrim cal state machine */ 2455 const struct dib0090_wbd_slope *wbd = state->current_wbd_table; 2456 2457 while (state->current_rf / 1000 > wbd->max_freq) 2458 wbd++; 2459 2460 dib0090_write_reg(state, 0x1e, 0x07ff); 2461 dprintk("Final Captrim: %d\n", (u32) state->fcaptrim); 2462 dprintk("HFDIV code: %d\n", (u32) pll->hfdiv_code); 2463 dprintk("VCO = %d\n", (u32) pll->vco_band); 2464 dprintk("VCOF in kHz: %d ((%d*%d) << 1))\n", (u32) ((pll->hfdiv * state->rf_request) * 2), (u32) pll->hfdiv, (u32) state->rf_request); 2465 dprintk("REFDIV: %d, FREF: %d\n", (u32) 1, (u32) state->config->io.clock_khz); 2466 dprintk("FBDIV: %d, Rest: %d\n", (u32) dib0090_read_reg(state, 0x15), (u32) dib0090_read_reg(state, 0x17)); 2467 dprintk("Num: %d, Den: %d, SD: %d\n", (u32) dib0090_read_reg(state, 0x17), (u32) (dib0090_read_reg(state, 0x16) >> 8), 2468 (u32) dib0090_read_reg(state, 0x1c) & 0x3); 2469 2470 #define WBD 0x781 /* 1 1 1 1 0000 0 0 1 */ 2471 c = 4; 2472 i = 3; 2473 2474 if (wbd->wbd_gain != 0) 2475 c = wbd->wbd_gain; 2476 2477 state->wbdmux = (c << 13) | (i << 11) | (WBD | (state->config->use_pwm_agc << 1)); 2478 dib0090_write_reg(state, 0x10, state->wbdmux); 2479 2480 if ((tune->tuner_enable == EN_CAB) && state->identity.p1g) { 2481 dprintk("P1G : The cable band is selected and lna_tune = %d\n", tune->lna_tune); 2482 dib0090_write_reg(state, 0x09, tune->lna_bias); 2483 dib0090_write_reg(state, 0x0b, 0xb800 | (tune->lna_tune << 6) | (tune->switch_trim)); 2484 } else 2485 dib0090_write_reg(state, 0x09, (tune->lna_tune << 5) | tune->lna_bias); 2486 2487 dib0090_write_reg(state, 0x0c, tune->v2i); 2488 dib0090_write_reg(state, 0x0d, tune->mix); 2489 dib0090_write_reg(state, 0x0e, tune->load); 2490 *tune_state = CT_TUNER_STEP_1; 2491 2492 } else if (*tune_state == CT_TUNER_STEP_1) { 2493 /* initialize the lt gain register */ 2494 state->rf_lt_def = 0x7c00; 2495 2496 dib0090_set_bandwidth(state); 2497 state->tuner_is_tuned = 1; 2498 2499 state->calibrate |= WBD_CAL; 2500 state->calibrate |= TEMP_CAL; 2501 *tune_state = CT_TUNER_STOP; 2502 } else 2503 ret = FE_CALLBACK_TIME_NEVER; 2504 return ret; 2505 } 2506 2507 static void dib0090_release(struct dvb_frontend *fe) 2508 { 2509 kfree(fe->tuner_priv); 2510 fe->tuner_priv = NULL; 2511 } 2512 2513 enum frontend_tune_state dib0090_get_tune_state(struct dvb_frontend *fe) 2514 { 2515 struct dib0090_state *state = fe->tuner_priv; 2516 2517 return state->tune_state; 2518 } 2519 2520 EXPORT_SYMBOL(dib0090_get_tune_state); 2521 2522 int dib0090_set_tune_state(struct dvb_frontend *fe, enum frontend_tune_state tune_state) 2523 { 2524 struct dib0090_state *state = fe->tuner_priv; 2525 2526 state->tune_state = tune_state; 2527 return 0; 2528 } 2529 2530 EXPORT_SYMBOL(dib0090_set_tune_state); 2531 2532 static int dib0090_get_frequency(struct dvb_frontend *fe, u32 * frequency) 2533 { 2534 struct dib0090_state *state = fe->tuner_priv; 2535 2536 *frequency = 1000 * state->current_rf; 2537 return 0; 2538 } 2539 2540 static int dib0090_set_params(struct dvb_frontend *fe) 2541 { 2542 struct dib0090_state *state = fe->tuner_priv; 2543 u32 ret; 2544 2545 state->tune_state = CT_TUNER_START; 2546 2547 do { 2548 ret = dib0090_tune(fe); 2549 if (ret == FE_CALLBACK_TIME_NEVER) 2550 break; 2551 2552 /* 2553 * Despite dib0090_tune returns time at a 0.1 ms range, 2554 * the actual sleep time depends on CONFIG_HZ. The worse case 2555 * is when CONFIG_HZ=100. In such case, the minimum granularity 2556 * is 10ms. On some real field tests, the tuner sometimes don't 2557 * lock when this timer is lower than 10ms. So, enforce a 10ms 2558 * granularity and use usleep_range() instead of msleep(). 2559 */ 2560 ret = 10 * (ret + 99)/100; 2561 usleep_range(ret * 1000, (ret + 1) * 1000); 2562 } while (state->tune_state != CT_TUNER_STOP); 2563 2564 return 0; 2565 } 2566 2567 static const struct dvb_tuner_ops dib0090_ops = { 2568 .info = { 2569 .name = "DiBcom DiB0090", 2570 .frequency_min_hz = 45 * MHz, 2571 .frequency_max_hz = 860 * MHz, 2572 .frequency_step_hz = 1 * kHz, 2573 }, 2574 .release = dib0090_release, 2575 2576 .init = dib0090_wakeup, 2577 .sleep = dib0090_sleep, 2578 .set_params = dib0090_set_params, 2579 .get_frequency = dib0090_get_frequency, 2580 }; 2581 2582 static const struct dvb_tuner_ops dib0090_fw_ops = { 2583 .info = { 2584 .name = "DiBcom DiB0090", 2585 .frequency_min_hz = 45 * MHz, 2586 .frequency_max_hz = 860 * MHz, 2587 .frequency_step_hz = 1 * kHz, 2588 }, 2589 .release = dib0090_release, 2590 2591 .init = NULL, 2592 .sleep = NULL, 2593 .set_params = NULL, 2594 .get_frequency = NULL, 2595 }; 2596 2597 static const struct dib0090_wbd_slope dib0090_wbd_table_default[] = { 2598 {470, 0, 250, 0, 100, 4}, 2599 {860, 51, 866, 21, 375, 4}, 2600 {1700, 0, 800, 0, 850, 4}, 2601 {2900, 0, 250, 0, 100, 6}, 2602 {0xFFFF, 0, 0, 0, 0, 0}, 2603 }; 2604 2605 struct dvb_frontend *dib0090_register(struct dvb_frontend *fe, struct i2c_adapter *i2c, const struct dib0090_config *config) 2606 { 2607 struct dib0090_state *st = kzalloc(sizeof(struct dib0090_state), GFP_KERNEL); 2608 if (st == NULL) 2609 return NULL; 2610 2611 st->config = config; 2612 st->i2c = i2c; 2613 st->fe = fe; 2614 mutex_init(&st->i2c_buffer_lock); 2615 fe->tuner_priv = st; 2616 2617 if (config->wbd == NULL) 2618 st->current_wbd_table = dib0090_wbd_table_default; 2619 else 2620 st->current_wbd_table = config->wbd; 2621 2622 if (dib0090_reset(fe) != 0) 2623 goto free_mem; 2624 2625 pr_info("DiB0090: successfully identified\n"); 2626 memcpy(&fe->ops.tuner_ops, &dib0090_ops, sizeof(struct dvb_tuner_ops)); 2627 2628 return fe; 2629 free_mem: 2630 kfree(st); 2631 fe->tuner_priv = NULL; 2632 return NULL; 2633 } 2634 2635 EXPORT_SYMBOL(dib0090_register); 2636 2637 struct dvb_frontend *dib0090_fw_register(struct dvb_frontend *fe, struct i2c_adapter *i2c, const struct dib0090_config *config) 2638 { 2639 struct dib0090_fw_state *st = kzalloc(sizeof(struct dib0090_fw_state), GFP_KERNEL); 2640 if (st == NULL) 2641 return NULL; 2642 2643 st->config = config; 2644 st->i2c = i2c; 2645 st->fe = fe; 2646 mutex_init(&st->i2c_buffer_lock); 2647 fe->tuner_priv = st; 2648 2649 if (dib0090_fw_reset_digital(fe, st->config) != 0) 2650 goto free_mem; 2651 2652 dprintk("DiB0090 FW: successfully identified\n"); 2653 memcpy(&fe->ops.tuner_ops, &dib0090_fw_ops, sizeof(struct dvb_tuner_ops)); 2654 2655 return fe; 2656 free_mem: 2657 kfree(st); 2658 fe->tuner_priv = NULL; 2659 return NULL; 2660 } 2661 EXPORT_SYMBOL(dib0090_fw_register); 2662 2663 MODULE_AUTHOR("Patrick Boettcher <patrick.boettcher@posteo.de>"); 2664 MODULE_AUTHOR("Olivier Grenie <olivier.grenie@parrot.com>"); 2665 MODULE_DESCRIPTION("Driver for the DiBcom 0090 base-band RF Tuner"); 2666 MODULE_LICENSE("GPL"); 2667