/* * stv0900_sw.c * * Driver for ST STV0900 satellite demodulator IC. * * Copyright (C) ST Microelectronics. * Copyright (C) 2009 NetUP Inc. * Copyright (C) 2009 Igor M. Liplianin * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ #include "stv0900.h" #include "stv0900_reg.h" #include "stv0900_priv.h" s32 shiftx(s32 x, int demod, s32 shift) { if (demod == 1) return x - shift; return x; } int stv0900_check_signal_presence(struct stv0900_internal *intp, enum fe_stv0900_demod_num demod) { s32 carr_offset, agc2_integr, max_carrier; int no_signal = FALSE; carr_offset = (stv0900_read_reg(intp, CFR2) << 8) | stv0900_read_reg(intp, CFR1); carr_offset = ge2comp(carr_offset, 16); agc2_integr = (stv0900_read_reg(intp, AGC2I1) << 8) | stv0900_read_reg(intp, AGC2I0); max_carrier = intp->srch_range[demod] / 1000; max_carrier += (max_carrier / 10); max_carrier = 65536 * (max_carrier / 2); max_carrier /= intp->mclk / 1000; if (max_carrier > 0x4000) max_carrier = 0x4000; if ((agc2_integr > 0x2000) || (carr_offset > (2 * max_carrier)) || (carr_offset < (-2 * max_carrier))) no_signal = TRUE; return no_signal; } static void stv0900_get_sw_loop_params(struct stv0900_internal *intp, s32 *frequency_inc, s32 *sw_timeout, s32 *steps, enum fe_stv0900_demod_num demod) { s32 timeout, freq_inc, max_steps, srate, max_carrier; enum fe_stv0900_search_standard standard; srate = intp->symbol_rate[demod]; max_carrier = intp->srch_range[demod] / 1000; max_carrier += max_carrier / 10; standard = intp->srch_standard[demod]; max_carrier = 65536 * (max_carrier / 2); max_carrier /= intp->mclk / 1000; if (max_carrier > 0x4000) max_carrier = 0x4000; freq_inc = srate; freq_inc /= intp->mclk >> 10; freq_inc = freq_inc << 6; switch (standard) { case STV0900_SEARCH_DVBS1: case STV0900_SEARCH_DSS: freq_inc *= 3; timeout = 20; break; case STV0900_SEARCH_DVBS2: freq_inc *= 4; timeout = 25; break; case STV0900_AUTO_SEARCH: default: freq_inc *= 3; timeout = 25; break; } freq_inc /= 100; if ((freq_inc > max_carrier) || (freq_inc < 0)) freq_inc = max_carrier / 2; timeout *= 27500; if (srate > 0) timeout /= srate / 1000; if ((timeout > 100) || (timeout < 0)) timeout = 100; max_steps = (max_carrier / freq_inc) + 1; if ((max_steps > 100) || (max_steps < 0)) { max_steps = 100; freq_inc = max_carrier / max_steps; } *frequency_inc = freq_inc; *sw_timeout = timeout; *steps = max_steps; } static int stv0900_search_carr_sw_loop(struct stv0900_internal *intp, s32 FreqIncr, s32 Timeout, int zigzag, s32 MaxStep, enum fe_stv0900_demod_num demod) { int no_signal, lock = FALSE; s32 stepCpt, freqOffset, max_carrier; max_carrier = intp->srch_range[demod] / 1000; max_carrier += (max_carrier / 10); max_carrier = 65536 * (max_carrier / 2); max_carrier /= intp->mclk / 1000; if (max_carrier > 0x4000) max_carrier = 0x4000; if (zigzag == TRUE) freqOffset = 0; else freqOffset = -max_carrier + FreqIncr; stepCpt = 0; do { stv0900_write_reg(intp, DMDISTATE, 0x1c); stv0900_write_reg(intp, CFRINIT1, (freqOffset / 256) & 0xff); stv0900_write_reg(intp, CFRINIT0, freqOffset & 0xff); stv0900_write_reg(intp, DMDISTATE, 0x18); stv0900_write_bits(intp, ALGOSWRST, 1); if (intp->chip_id == 0x12) { stv0900_write_bits(intp, RST_HWARE, 1); stv0900_write_bits(intp, RST_HWARE, 0); } if (zigzag == TRUE) { if (freqOffset >= 0) freqOffset = -freqOffset - 2 * FreqIncr; else freqOffset = -freqOffset; } else freqOffset += + 2 * FreqIncr; stepCpt++; lock = stv0900_get_demod_lock(intp, demod, Timeout); no_signal = stv0900_check_signal_presence(intp, demod); } while ((lock == FALSE) && (no_signal == FALSE) && ((freqOffset - FreqIncr) < max_carrier) && ((freqOffset + FreqIncr) > -max_carrier) && (stepCpt < MaxStep)); stv0900_write_bits(intp, ALGOSWRST, 0); return lock; } static int stv0900_sw_algo(struct stv0900_internal *intp, enum fe_stv0900_demod_num demod) { int lock = FALSE, no_signal, zigzag; s32 s2fw, fqc_inc, sft_stp_tout, trial_cntr, max_steps; stv0900_get_sw_loop_params(intp, &fqc_inc, &sft_stp_tout, &max_steps, demod); switch (intp->srch_standard[demod]) { case STV0900_SEARCH_DVBS1: case STV0900_SEARCH_DSS: if (intp->chip_id >= 0x20) stv0900_write_reg(intp, CARFREQ, 0x3b); else stv0900_write_reg(intp, CARFREQ, 0xef); stv0900_write_reg(intp, DMDCFGMD, 0x49); zigzag = FALSE; break; case STV0900_SEARCH_DVBS2: if (intp->chip_id >= 0x20) stv0900_write_reg(intp, CORRELABS, 0x79); else stv0900_write_reg(intp, CORRELABS, 0x68); stv0900_write_reg(intp, DMDCFGMD, 0x89); zigzag = TRUE; break; case STV0900_AUTO_SEARCH: default: if (intp->chip_id >= 0x20) { stv0900_write_reg(intp, CARFREQ, 0x3b); stv0900_write_reg(intp, CORRELABS, 0x79); } else { stv0900_write_reg(intp, CARFREQ, 0xef); stv0900_write_reg(intp, CORRELABS, 0x68); } stv0900_write_reg(intp, DMDCFGMD, 0xc9); zigzag = FALSE; break; } trial_cntr = 0; do { lock = stv0900_search_carr_sw_loop(intp, fqc_inc, sft_stp_tout, zigzag, max_steps, demod); no_signal = stv0900_check_signal_presence(intp, demod); trial_cntr++; if ((lock == TRUE) || (no_signal == TRUE) || (trial_cntr == 2)) { if (intp->chip_id >= 0x20) { stv0900_write_reg(intp, CARFREQ, 0x49); stv0900_write_reg(intp, CORRELABS, 0x9e); } else { stv0900_write_reg(intp, CARFREQ, 0xed); stv0900_write_reg(intp, CORRELABS, 0x88); } if ((stv0900_get_bits(intp, HEADER_MODE) == STV0900_DVBS2_FOUND) && (lock == TRUE)) { msleep(sft_stp_tout); s2fw = stv0900_get_bits(intp, FLYWHEEL_CPT); if (s2fw < 0xd) { msleep(sft_stp_tout); s2fw = stv0900_get_bits(intp, FLYWHEEL_CPT); } if (s2fw < 0xd) { lock = FALSE; if (trial_cntr < 2) { if (intp->chip_id >= 0x20) stv0900_write_reg(intp, CORRELABS, 0x79); else stv0900_write_reg(intp, CORRELABS, 0x68); stv0900_write_reg(intp, DMDCFGMD, 0x89); } } } } } while ((lock == FALSE) && (trial_cntr < 2) && (no_signal == FALSE)); return lock; } static u32 stv0900_get_symbol_rate(struct stv0900_internal *intp, u32 mclk, enum fe_stv0900_demod_num demod) { s32 rem1, rem2, intval1, intval2, srate; srate = (stv0900_get_bits(intp, SYMB_FREQ3) << 24) + (stv0900_get_bits(intp, SYMB_FREQ2) << 16) + (stv0900_get_bits(intp, SYMB_FREQ1) << 8) + (stv0900_get_bits(intp, SYMB_FREQ0)); dprintk("lock: srate=%d r0=0x%x r1=0x%x r2=0x%x r3=0x%x \n", srate, stv0900_get_bits(intp, SYMB_FREQ0), stv0900_get_bits(intp, SYMB_FREQ1), stv0900_get_bits(intp, SYMB_FREQ2), stv0900_get_bits(intp, SYMB_FREQ3)); intval1 = (mclk) >> 16; intval2 = (srate) >> 16; rem1 = (mclk) % 0x10000; rem2 = (srate) % 0x10000; srate = (intval1 * intval2) + ((intval1 * rem2) >> 16) + ((intval2 * rem1) >> 16); return srate; } static void stv0900_set_symbol_rate(struct stv0900_internal *intp, u32 mclk, u32 srate, enum fe_stv0900_demod_num demod) { u32 symb; dprintk("%s: Mclk %d, SR %d, Dmd %d\n", __func__, mclk, srate, demod); if (srate > 60000000) { symb = srate << 4; symb /= (mclk >> 12); } else if (srate > 6000000) { symb = srate << 6; symb /= (mclk >> 10); } else { symb = srate << 9; symb /= (mclk >> 7); } stv0900_write_reg(intp, SFRINIT1, (symb >> 8) & 0x7f); stv0900_write_reg(intp, SFRINIT1 + 1, (symb & 0xff)); } static void stv0900_set_max_symbol_rate(struct stv0900_internal *intp, u32 mclk, u32 srate, enum fe_stv0900_demod_num demod) { u32 symb; srate = 105 * (srate / 100); if (srate > 60000000) { symb = srate << 4; symb /= (mclk >> 12); } else if (srate > 6000000) { symb = srate << 6; symb /= (mclk >> 10); } else { symb = srate << 9; symb /= (mclk >> 7); } if (symb < 0x7fff) { stv0900_write_reg(intp, SFRUP1, (symb >> 8) & 0x7f); stv0900_write_reg(intp, SFRUP1 + 1, (symb & 0xff)); } else { stv0900_write_reg(intp, SFRUP1, 0x7f); stv0900_write_reg(intp, SFRUP1 + 1, 0xff); } } static void stv0900_set_min_symbol_rate(struct stv0900_internal *intp, u32 mclk, u32 srate, enum fe_stv0900_demod_num demod) { u32 symb; srate = 95 * (srate / 100); if (srate > 60000000) { symb = srate << 4; symb /= (mclk >> 12); } else if (srate > 6000000) { symb = srate << 6; symb /= (mclk >> 10); } else { symb = srate << 9; symb /= (mclk >> 7); } stv0900_write_reg(intp, SFRLOW1, (symb >> 8) & 0xff); stv0900_write_reg(intp, SFRLOW1 + 1, (symb & 0xff)); } static s32 stv0900_get_timing_offst(struct stv0900_internal *intp, u32 srate, enum fe_stv0900_demod_num demod) { s32 timingoffset; timingoffset = (stv0900_read_reg(intp, TMGREG2) << 16) + (stv0900_read_reg(intp, TMGREG2 + 1) << 8) + (stv0900_read_reg(intp, TMGREG2 + 2)); timingoffset = ge2comp(timingoffset, 24); if (timingoffset == 0) timingoffset = 1; timingoffset = ((s32)srate * 10) / ((s32)0x1000000 / timingoffset); timingoffset /= 320; return timingoffset; } static void stv0900_set_dvbs2_rolloff(struct stv0900_internal *intp, enum fe_stv0900_demod_num demod) { s32 rolloff; if (intp->chip_id == 0x10) { stv0900_write_bits(intp, MANUALSX_ROLLOFF, 1); rolloff = stv0900_read_reg(intp, MATSTR1) & 0x03; stv0900_write_bits(intp, ROLLOFF_CONTROL, rolloff); } else if (intp->chip_id <= 0x20) stv0900_write_bits(intp, MANUALSX_ROLLOFF, 0); else /* cut 3.0 */ stv0900_write_bits(intp, MANUALS2_ROLLOFF, 0); } static u32 stv0900_carrier_width(u32 srate, enum fe_stv0900_rolloff ro) { u32 rolloff; switch (ro) { case STV0900_20: rolloff = 20; break; case STV0900_25: rolloff = 25; break; case STV0900_35: default: rolloff = 35; break; } return srate + (srate * rolloff) / 100; } static int stv0900_check_timing_lock(struct stv0900_internal *intp, enum fe_stv0900_demod_num demod) { int timingLock = FALSE; s32 i, timingcpt = 0; u8 car_freq, tmg_th_high, tmg_th_low; car_freq = stv0900_read_reg(intp, CARFREQ); tmg_th_high = stv0900_read_reg(intp, TMGTHRISE); tmg_th_low = stv0900_read_reg(intp, TMGTHFALL); stv0900_write_reg(intp, TMGTHRISE, 0x20); stv0900_write_reg(intp, TMGTHFALL, 0x0); stv0900_write_bits(intp, CFR_AUTOSCAN, 0); stv0900_write_reg(intp, RTC, 0x80); stv0900_write_reg(intp, RTCS2, 0x40); stv0900_write_reg(intp, CARFREQ, 0x0); stv0900_write_reg(intp, CFRINIT1, 0x0); stv0900_write_reg(intp, CFRINIT0, 0x0); stv0900_write_reg(intp, AGC2REF, 0x65); stv0900_write_reg(intp, DMDISTATE, 0x18); msleep(7); for (i = 0; i < 10; i++) { if (stv0900_get_bits(intp, TMGLOCK_QUALITY) >= 2) timingcpt++; msleep(1); } if (timingcpt >= 3) timingLock = TRUE; stv0900_write_reg(intp, AGC2REF, 0x38); stv0900_write_reg(intp, RTC, 0x88); stv0900_write_reg(intp, RTCS2, 0x68); stv0900_write_reg(intp, CARFREQ, car_freq); stv0900_write_reg(intp, TMGTHRISE, tmg_th_high); stv0900_write_reg(intp, TMGTHFALL, tmg_th_low); return timingLock; } static int stv0900_get_demod_cold_lock(struct dvb_frontend *fe, s32 demod_timeout) { struct stv0900_state *state = fe->demodulator_priv; struct stv0900_internal *intp = state->internal; enum fe_stv0900_demod_num demod = state->demod; int lock = FALSE, d = demod; s32 srate, search_range, locktimeout, currier_step, nb_steps, current_step, direction, tuner_freq, timeout, freq; srate = intp->symbol_rate[d]; search_range = intp->srch_range[d]; if (srate >= 10000000) locktimeout = demod_timeout / 3; else locktimeout = demod_timeout / 2; lock = stv0900_get_demod_lock(intp, d, locktimeout); if (lock != FALSE) return lock; if (srate >= 10000000) { if (stv0900_check_timing_lock(intp, d) == TRUE) { stv0900_write_reg(intp, DMDISTATE, 0x1f); stv0900_write_reg(intp, DMDISTATE, 0x15); lock = stv0900_get_demod_lock(intp, d, demod_timeout); } else lock = FALSE; return lock; } if (intp->chip_id <= 0x20) { if (srate <= 1000000) currier_step = 500; else if (srate <= 4000000) currier_step = 1000; else if (srate <= 7000000) currier_step = 2000; else if (srate <= 10000000) currier_step = 3000; else currier_step = 5000; if (srate >= 2000000) { timeout = (demod_timeout / 3); if (timeout > 1000) timeout = 1000; } else timeout = (demod_timeout / 2); } else { /*cut 3.0 */ currier_step = srate / 4000; timeout = (demod_timeout * 3) / 4; } nb_steps = ((search_range / 1000) / currier_step); if ((nb_steps % 2) != 0) nb_steps += 1; if (nb_steps <= 0) nb_steps = 2; else if (nb_steps > 12) nb_steps = 12; current_step = 1; direction = 1; if (intp->chip_id <= 0x20) { tuner_freq = intp->freq[d]; intp->bw[d] = stv0900_carrier_width(intp->symbol_rate[d], intp->rolloff) + intp->symbol_rate[d]; } else tuner_freq = 0; while ((current_step <= nb_steps) && (lock == FALSE)) { if (direction > 0) tuner_freq += (current_step * currier_step); else tuner_freq -= (current_step * currier_step); if (intp->chip_id <= 0x20) { if (intp->tuner_type[d] == 3) stv0900_set_tuner_auto(intp, tuner_freq, intp->bw[d], demod); else stv0900_set_tuner(fe, tuner_freq, intp->bw[d]); stv0900_write_reg(intp, DMDISTATE, 0x1c); stv0900_write_reg(intp, CFRINIT1, 0); stv0900_write_reg(intp, CFRINIT0, 0); stv0900_write_reg(intp, DMDISTATE, 0x1f); stv0900_write_reg(intp, DMDISTATE, 0x15); } else { stv0900_write_reg(intp, DMDISTATE, 0x1c); freq = (tuner_freq * 65536) / (intp->mclk / 1000); stv0900_write_bits(intp, CFR_INIT1, MSB(freq)); stv0900_write_bits(intp, CFR_INIT0, LSB(freq)); stv0900_write_reg(intp, DMDISTATE, 0x1f); stv0900_write_reg(intp, DMDISTATE, 0x05); } lock = stv0900_get_demod_lock(intp, d, timeout); direction *= -1; current_step++; } return lock; } static void stv0900_get_lock_timeout(s32 *demod_timeout, s32 *fec_timeout, s32 srate, enum fe_stv0900_search_algo algo) { switch (algo) { case STV0900_BLIND_SEARCH: if (srate <= 1500000) { (*demod_timeout) = 1500; (*fec_timeout) = 400; } else if (srate <= 5000000) { (*demod_timeout) = 1000; (*fec_timeout) = 300; } else { (*demod_timeout) = 700; (*fec_timeout) = 100; } break; case STV0900_COLD_START: case STV0900_WARM_START: default: if (srate <= 1000000) { (*demod_timeout) = 3000; (*fec_timeout) = 1700; } else if (srate <= 2000000) { (*demod_timeout) = 2500; (*fec_timeout) = 1100; } else if (srate <= 5000000) { (*demod_timeout) = 1000; (*fec_timeout) = 550; } else if (srate <= 10000000) { (*demod_timeout) = 700; (*fec_timeout) = 250; } else if (srate <= 20000000) { (*demod_timeout) = 400; (*fec_timeout) = 130; } else { (*demod_timeout) = 300; (*fec_timeout) = 100; } break; } if (algo == STV0900_WARM_START) (*demod_timeout) /= 2; } static void stv0900_set_viterbi_tracq(struct stv0900_internal *intp, enum fe_stv0900_demod_num demod) { s32 vth_reg = VTH12; dprintk("%s\n", __func__); stv0900_write_reg(intp, vth_reg++, 0xd0); stv0900_write_reg(intp, vth_reg++, 0x7d); stv0900_write_reg(intp, vth_reg++, 0x53); stv0900_write_reg(intp, vth_reg++, 0x2f); stv0900_write_reg(intp, vth_reg++, 0x24); stv0900_write_reg(intp, vth_reg++, 0x1f); } static void stv0900_set_viterbi_standard(struct stv0900_internal *intp, enum fe_stv0900_search_standard standard, enum fe_stv0900_fec fec, enum fe_stv0900_demod_num demod) { dprintk("%s: ViterbiStandard = ", __func__); switch (standard) { case STV0900_AUTO_SEARCH: dprintk("Auto\n"); stv0900_write_reg(intp, FECM, 0x10); stv0900_write_reg(intp, PRVIT, 0x3f); break; case STV0900_SEARCH_DVBS1: dprintk("DVBS1\n"); stv0900_write_reg(intp, FECM, 0x00); switch (fec) { case STV0900_FEC_UNKNOWN: default: stv0900_write_reg(intp, PRVIT, 0x2f); break; case STV0900_FEC_1_2: stv0900_write_reg(intp, PRVIT, 0x01); break; case STV0900_FEC_2_3: stv0900_write_reg(intp, PRVIT, 0x02); break; case STV0900_FEC_3_4: stv0900_write_reg(intp, PRVIT, 0x04); break; case STV0900_FEC_5_6: stv0900_write_reg(intp, PRVIT, 0x08); break; case STV0900_FEC_7_8: stv0900_write_reg(intp, PRVIT, 0x20); break; } break; case STV0900_SEARCH_DSS: dprintk("DSS\n"); stv0900_write_reg(intp, FECM, 0x80); switch (fec) { case STV0900_FEC_UNKNOWN: default: stv0900_write_reg(intp, PRVIT, 0x13); break; case STV0900_FEC_1_2: stv0900_write_reg(intp, PRVIT, 0x01); break; case STV0900_FEC_2_3: stv0900_write_reg(intp, PRVIT, 0x02); break; case STV0900_FEC_6_7: stv0900_write_reg(intp, PRVIT, 0x10); break; } break; default: break; } } static enum fe_stv0900_fec stv0900_get_vit_fec(struct stv0900_internal *intp, enum fe_stv0900_demod_num demod) { enum fe_stv0900_fec prate; s32 rate_fld = stv0900_get_bits(intp, VIT_CURPUN); switch (rate_fld) { case 13: prate = STV0900_FEC_1_2; break; case 18: prate = STV0900_FEC_2_3; break; case 21: prate = STV0900_FEC_3_4; break; case 24: prate = STV0900_FEC_5_6; break; case 25: prate = STV0900_FEC_6_7; break; case 26: prate = STV0900_FEC_7_8; break; default: prate = STV0900_FEC_UNKNOWN; break; } return prate; } static void stv0900_set_dvbs1_track_car_loop(struct stv0900_internal *intp, enum fe_stv0900_demod_num demod, u32 srate) { if (intp->chip_id >= 0x30) { if (srate >= 15000000) { stv0900_write_reg(intp, ACLC, 0x2b); stv0900_write_reg(intp, BCLC, 0x1a); } else if ((srate >= 7000000) && (15000000 > srate)) { stv0900_write_reg(intp, ACLC, 0x0c); stv0900_write_reg(intp, BCLC, 0x1b); } else if (srate < 7000000) { stv0900_write_reg(intp, ACLC, 0x2c); stv0900_write_reg(intp, BCLC, 0x1c); } } else { /*cut 2.0 and 1.x*/ stv0900_write_reg(intp, ACLC, 0x1a); stv0900_write_reg(intp, BCLC, 0x09); } } static void stv0900_track_optimization(struct dvb_frontend *fe) { struct stv0900_state *state = fe->demodulator_priv; struct stv0900_internal *intp = state->internal; enum fe_stv0900_demod_num demod = state->demod; s32 srate, pilots, aclc, freq1, freq0, i = 0, timed, timef, blind_tun_sw = 0, modulation; enum fe_stv0900_rolloff rolloff; enum fe_stv0900_modcode foundModcod; dprintk("%s\n", __func__); srate = stv0900_get_symbol_rate(intp, intp->mclk, demod); srate += stv0900_get_timing_offst(intp, srate, demod); switch (intp->result[demod].standard) { case STV0900_DVBS1_STANDARD: case STV0900_DSS_STANDARD: dprintk("%s: found DVB-S or DSS\n", __func__); if (intp->srch_standard[demod] == STV0900_AUTO_SEARCH) { stv0900_write_bits(intp, DVBS1_ENABLE, 1); stv0900_write_bits(intp, DVBS2_ENABLE, 0); } stv0900_write_bits(intp, ROLLOFF_CONTROL, intp->rolloff); stv0900_write_bits(intp, MANUALSX_ROLLOFF, 1); if (intp->chip_id < 0x30) { stv0900_write_reg(intp, ERRCTRL1, 0x75); break; } if (stv0900_get_vit_fec(intp, demod) == STV0900_FEC_1_2) { stv0900_write_reg(intp, GAUSSR0, 0x98); stv0900_write_reg(intp, CCIR0, 0x18); } else { stv0900_write_reg(intp, GAUSSR0, 0x18); stv0900_write_reg(intp, CCIR0, 0x18); } stv0900_write_reg(intp, ERRCTRL1, 0x75); break; case STV0900_DVBS2_STANDARD: dprintk("%s: found DVB-S2\n", __func__); stv0900_write_bits(intp, DVBS1_ENABLE, 0); stv0900_write_bits(intp, DVBS2_ENABLE, 1); stv0900_write_reg(intp, ACLC, 0); stv0900_write_reg(intp, BCLC, 0); if (intp->result[demod].frame_len == STV0900_LONG_FRAME) { foundModcod = stv0900_get_bits(intp, DEMOD_MODCOD); pilots = stv0900_get_bits(intp, DEMOD_TYPE) & 0x01; aclc = stv0900_get_optim_carr_loop(srate, foundModcod, pilots, intp->chip_id); if (foundModcod <= STV0900_QPSK_910) stv0900_write_reg(intp, ACLC2S2Q, aclc); else if (foundModcod <= STV0900_8PSK_910) { stv0900_write_reg(intp, ACLC2S2Q, 0x2a); stv0900_write_reg(intp, ACLC2S28, aclc); } if ((intp->demod_mode == STV0900_SINGLE) && (foundModcod > STV0900_8PSK_910)) { if (foundModcod <= STV0900_16APSK_910) { stv0900_write_reg(intp, ACLC2S2Q, 0x2a); stv0900_write_reg(intp, ACLC2S216A, aclc); } else if (foundModcod <= STV0900_32APSK_910) { stv0900_write_reg(intp, ACLC2S2Q, 0x2a); stv0900_write_reg(intp, ACLC2S232A, aclc); } } } else { modulation = intp->result[demod].modulation; aclc = stv0900_get_optim_short_carr_loop(srate, modulation, intp->chip_id); if (modulation == STV0900_QPSK) stv0900_write_reg(intp, ACLC2S2Q, aclc); else if (modulation == STV0900_8PSK) { stv0900_write_reg(intp, ACLC2S2Q, 0x2a); stv0900_write_reg(intp, ACLC2S28, aclc); } else if (modulation == STV0900_16APSK) { stv0900_write_reg(intp, ACLC2S2Q, 0x2a); stv0900_write_reg(intp, ACLC2S216A, aclc); } else if (modulation == STV0900_32APSK) { stv0900_write_reg(intp, ACLC2S2Q, 0x2a); stv0900_write_reg(intp, ACLC2S232A, aclc); } } if (intp->chip_id <= 0x11) { if (intp->demod_mode != STV0900_SINGLE) stv0900_activate_s2_modcod(intp, demod); } stv0900_write_reg(intp, ERRCTRL1, 0x67); break; case STV0900_UNKNOWN_STANDARD: default: dprintk("%s: found unknown standard\n", __func__); stv0900_write_bits(intp, DVBS1_ENABLE, 1); stv0900_write_bits(intp, DVBS2_ENABLE, 1); break; } freq1 = stv0900_read_reg(intp, CFR2); freq0 = stv0900_read_reg(intp, CFR1); rolloff = stv0900_get_bits(intp, ROLLOFF_STATUS); if (intp->srch_algo[demod] == STV0900_BLIND_SEARCH) { stv0900_write_reg(intp, SFRSTEP, 0x00); stv0900_write_bits(intp, SCAN_ENABLE, 0); stv0900_write_bits(intp, CFR_AUTOSCAN, 0); stv0900_write_reg(intp, TMGCFG2, 0xc1); stv0900_set_symbol_rate(intp, intp->mclk, srate, demod); blind_tun_sw = 1; if (intp->result[demod].standard != STV0900_DVBS2_STANDARD) stv0900_set_dvbs1_track_car_loop(intp, demod, srate); } if (intp->chip_id >= 0x20) { if ((intp->srch_standard[demod] == STV0900_SEARCH_DVBS1) || (intp->srch_standard[demod] == STV0900_SEARCH_DSS) || (intp->srch_standard[demod] == STV0900_AUTO_SEARCH)) { stv0900_write_reg(intp, VAVSRVIT, 0x0a); stv0900_write_reg(intp, VITSCALE, 0x0); } } if (intp->chip_id < 0x20) stv0900_write_reg(intp, CARHDR, 0x08); if (intp->chip_id == 0x10) stv0900_write_reg(intp, CORRELEXP, 0x0a); stv0900_write_reg(intp, AGC2REF, 0x38); if ((intp->chip_id >= 0x20) || (blind_tun_sw == 1) || (intp->symbol_rate[demod] < 10000000)) { stv0900_write_reg(intp, CFRINIT1, freq1); stv0900_write_reg(intp, CFRINIT0, freq0); intp->bw[demod] = stv0900_carrier_width(srate, intp->rolloff) + 10000000; if ((intp->chip_id >= 0x20) || (blind_tun_sw == 1)) { if (intp->srch_algo[demod] != STV0900_WARM_START) { if (intp->tuner_type[demod] == 3) stv0900_set_tuner_auto(intp, intp->freq[demod], intp->bw[demod], demod); else stv0900_set_bandwidth(fe, intp->bw[demod]); } } if ((intp->srch_algo[demod] == STV0900_BLIND_SEARCH) || (intp->symbol_rate[demod] < 10000000)) msleep(50); else msleep(5); stv0900_get_lock_timeout(&timed, &timef, srate, STV0900_WARM_START); if (stv0900_get_demod_lock(intp, demod, timed / 2) == FALSE) { stv0900_write_reg(intp, DMDISTATE, 0x1f); stv0900_write_reg(intp, CFRINIT1, freq1); stv0900_write_reg(intp, CFRINIT0, freq0); stv0900_write_reg(intp, DMDISTATE, 0x18); i = 0; while ((stv0900_get_demod_lock(intp, demod, timed / 2) == FALSE) && (i <= 2)) { stv0900_write_reg(intp, DMDISTATE, 0x1f); stv0900_write_reg(intp, CFRINIT1, freq1); stv0900_write_reg(intp, CFRINIT0, freq0); stv0900_write_reg(intp, DMDISTATE, 0x18); i++; } } } if (intp->chip_id >= 0x20) stv0900_write_reg(intp, CARFREQ, 0x49); if ((intp->result[demod].standard == STV0900_DVBS1_STANDARD) || (intp->result[demod].standard == STV0900_DSS_STANDARD)) stv0900_set_viterbi_tracq(intp, demod); } static int stv0900_get_fec_lock(struct stv0900_internal *intp, enum fe_stv0900_demod_num demod, s32 time_out) { s32 timer = 0, lock = 0; enum fe_stv0900_search_state dmd_state; dprintk("%s\n", __func__); dmd_state = stv0900_get_bits(intp, HEADER_MODE); while ((timer < time_out) && (lock == 0)) { switch (dmd_state) { case STV0900_SEARCH: case STV0900_PLH_DETECTED: default: lock = 0; break; case STV0900_DVBS2_FOUND: lock = stv0900_get_bits(intp, PKTDELIN_LOCK); break; case STV0900_DVBS_FOUND: lock = stv0900_get_bits(intp, LOCKEDVIT); break; } if (lock == 0) { msleep(10); timer += 10; } } if (lock) dprintk("%s: DEMOD FEC LOCK OK\n", __func__); else dprintk("%s: DEMOD FEC LOCK FAIL\n", __func__); return lock; } static int stv0900_wait_for_lock(struct stv0900_internal *intp, enum fe_stv0900_demod_num demod, s32 dmd_timeout, s32 fec_timeout) { s32 timer = 0, lock = 0; dprintk("%s\n", __func__); lock = stv0900_get_demod_lock(intp, demod, dmd_timeout); if (lock) lock = lock && stv0900_get_fec_lock(intp, demod, fec_timeout); if (lock) { lock = 0; dprintk("%s: Timer = %d, time_out = %d\n", __func__, timer, fec_timeout); while ((timer < fec_timeout) && (lock == 0)) { lock = stv0900_get_bits(intp, TSFIFO_LINEOK); msleep(1); timer++; } } if (lock) dprintk("%s: DEMOD LOCK OK\n", __func__); else dprintk("%s: DEMOD LOCK FAIL\n", __func__); if (lock) return TRUE; else return FALSE; } enum fe_stv0900_tracking_standard stv0900_get_standard(struct dvb_frontend *fe, enum fe_stv0900_demod_num demod) { struct stv0900_state *state = fe->demodulator_priv; struct stv0900_internal *intp = state->internal; enum fe_stv0900_tracking_standard fnd_standard; int hdr_mode = stv0900_get_bits(intp, HEADER_MODE); switch (hdr_mode) { case 2: fnd_standard = STV0900_DVBS2_STANDARD; break; case 3: if (stv0900_get_bits(intp, DSS_DVB) == 1) fnd_standard = STV0900_DSS_STANDARD; else fnd_standard = STV0900_DVBS1_STANDARD; break; default: fnd_standard = STV0900_UNKNOWN_STANDARD; } dprintk("%s: standard %d\n", __func__, fnd_standard); return fnd_standard; } static s32 stv0900_get_carr_freq(struct stv0900_internal *intp, u32 mclk, enum fe_stv0900_demod_num demod) { s32 derot, rem1, rem2, intval1, intval2; derot = (stv0900_get_bits(intp, CAR_FREQ2) << 16) + (stv0900_get_bits(intp, CAR_FREQ1) << 8) + (stv0900_get_bits(intp, CAR_FREQ0)); derot = ge2comp(derot, 24); intval1 = mclk >> 12; intval2 = derot >> 12; rem1 = mclk % 0x1000; rem2 = derot % 0x1000; derot = (intval1 * intval2) + ((intval1 * rem2) >> 12) + ((intval2 * rem1) >> 12); return derot; } static u32 stv0900_get_tuner_freq(struct dvb_frontend *fe) { struct dvb_frontend_ops *frontend_ops = NULL; struct dvb_tuner_ops *tuner_ops = NULL; u32 freq = 0; if (&fe->ops) frontend_ops = &fe->ops; if (&frontend_ops->tuner_ops) tuner_ops = &frontend_ops->tuner_ops; if (tuner_ops->get_frequency) { if ((tuner_ops->get_frequency(fe, &freq)) < 0) dprintk("%s: Invalid parameter\n", __func__); else dprintk("%s: Frequency=%d\n", __func__, freq); } return freq; } static enum fe_stv0900_signal_type stv0900_get_signal_params(struct dvb_frontend *fe) { struct stv0900_state *state = fe->demodulator_priv; struct stv0900_internal *intp = state->internal; enum fe_stv0900_demod_num demod = state->demod; enum fe_stv0900_signal_type range = STV0900_OUTOFRANGE; struct stv0900_signal_info *result = &intp->result[demod]; s32 offsetFreq, srate_offset; int i = 0, d = demod; u8 timing; msleep(5); if (intp->srch_algo[d] == STV0900_BLIND_SEARCH) { timing = stv0900_read_reg(intp, TMGREG2); i = 0; stv0900_write_reg(intp, SFRSTEP, 0x5c); while ((i <= 50) && (timing != 0) && (timing != 0xff)) { timing = stv0900_read_reg(intp, TMGREG2); msleep(5); i += 5; } } result->standard = stv0900_get_standard(fe, d); if (intp->tuner_type[demod] == 3) result->frequency = stv0900_get_freq_auto(intp, d); else result->frequency = stv0900_get_tuner_freq(fe); offsetFreq = stv0900_get_carr_freq(intp, intp->mclk, d) / 1000; result->frequency += offsetFreq; result->symbol_rate = stv0900_get_symbol_rate(intp, intp->mclk, d); srate_offset = stv0900_get_timing_offst(intp, result->symbol_rate, d); result->symbol_rate += srate_offset; result->fec = stv0900_get_vit_fec(intp, d); result->modcode = stv0900_get_bits(intp, DEMOD_MODCOD); result->pilot = stv0900_get_bits(intp, DEMOD_TYPE) & 0x01; result->frame_len = ((u32)stv0900_get_bits(intp, DEMOD_TYPE)) >> 1; result->rolloff = stv0900_get_bits(intp, ROLLOFF_STATUS); switch (result->standard) { case STV0900_DVBS2_STANDARD: result->spectrum = stv0900_get_bits(intp, SPECINV_DEMOD); if (result->modcode <= STV0900_QPSK_910) result->modulation = STV0900_QPSK; else if (result->modcode <= STV0900_8PSK_910) result->modulation = STV0900_8PSK; else if (result->modcode <= STV0900_16APSK_910) result->modulation = STV0900_16APSK; else if (result->modcode <= STV0900_32APSK_910) result->modulation = STV0900_32APSK; else result->modulation = STV0900_UNKNOWN; break; case STV0900_DVBS1_STANDARD: case STV0900_DSS_STANDARD: result->spectrum = stv0900_get_bits(intp, IQINV); result->modulation = STV0900_QPSK; break; default: break; } if ((intp->srch_algo[d] == STV0900_BLIND_SEARCH) || (intp->symbol_rate[d] < 10000000)) { offsetFreq = result->frequency - intp->freq[d]; if (intp->tuner_type[demod] == 3) intp->freq[d] = stv0900_get_freq_auto(intp, d); else intp->freq[d] = stv0900_get_tuner_freq(fe); if (ABS(offsetFreq) <= ((intp->srch_range[d] / 2000) + 500)) range = STV0900_RANGEOK; else if (ABS(offsetFreq) <= (stv0900_carrier_width(result->symbol_rate, result->rolloff) / 2000)) range = STV0900_RANGEOK; } else if (ABS(offsetFreq) <= ((intp->srch_range[d] / 2000) + 500)) range = STV0900_RANGEOK; dprintk("%s: range %d\n", __func__, range); return range; } static enum fe_stv0900_signal_type stv0900_dvbs1_acq_workaround(struct dvb_frontend *fe) { struct stv0900_state *state = fe->demodulator_priv; struct stv0900_internal *intp = state->internal; enum fe_stv0900_demod_num demod = state->demod; enum fe_stv0900_signal_type signal_type = STV0900_NODATA; s32 srate, demod_timeout, fec_timeout, freq1, freq0; intp->result[demod].locked = FALSE; if (stv0900_get_bits(intp, HEADER_MODE) == STV0900_DVBS_FOUND) { srate = stv0900_get_symbol_rate(intp, intp->mclk, demod); srate += stv0900_get_timing_offst(intp, srate, demod); if (intp->srch_algo[demod] == STV0900_BLIND_SEARCH) stv0900_set_symbol_rate(intp, intp->mclk, srate, demod); stv0900_get_lock_timeout(&demod_timeout, &fec_timeout, srate, STV0900_WARM_START); freq1 = stv0900_read_reg(intp, CFR2); freq0 = stv0900_read_reg(intp, CFR1); stv0900_write_bits(intp, CFR_AUTOSCAN, 0); stv0900_write_bits(intp, SPECINV_CONTROL, STV0900_IQ_FORCE_SWAPPED); stv0900_write_reg(intp, DMDISTATE, 0x1c); stv0900_write_reg(intp, CFRINIT1, freq1); stv0900_write_reg(intp, CFRINIT0, freq0); stv0900_write_reg(intp, DMDISTATE, 0x18); if (stv0900_wait_for_lock(intp, demod, demod_timeout, fec_timeout) == TRUE) { intp->result[demod].locked = TRUE; signal_type = stv0900_get_signal_params(fe); stv0900_track_optimization(fe); } else { stv0900_write_bits(intp, SPECINV_CONTROL, STV0900_IQ_FORCE_NORMAL); stv0900_write_reg(intp, DMDISTATE, 0x1c); stv0900_write_reg(intp, CFRINIT1, freq1); stv0900_write_reg(intp, CFRINIT0, freq0); stv0900_write_reg(intp, DMDISTATE, 0x18); if (stv0900_wait_for_lock(intp, demod, demod_timeout, fec_timeout) == TRUE) { intp->result[demod].locked = TRUE; signal_type = stv0900_get_signal_params(fe); stv0900_track_optimization(fe); } } } else intp->result[demod].locked = FALSE; return signal_type; } static u16 stv0900_blind_check_agc2_min_level(struct stv0900_internal *intp, enum fe_stv0900_demod_num demod) { u32 minagc2level = 0xffff, agc2level, init_freq, freq_step; s32 i, j, nb_steps, direction; dprintk("%s\n", __func__); stv0900_write_reg(intp, AGC2REF, 0x38); stv0900_write_bits(intp, SCAN_ENABLE, 0); stv0900_write_bits(intp, CFR_AUTOSCAN, 0); stv0900_write_bits(intp, AUTO_GUP, 1); stv0900_write_bits(intp, AUTO_GLOW, 1); stv0900_write_reg(intp, DMDT0M, 0x0); stv0900_set_symbol_rate(intp, intp->mclk, 1000000, demod); nb_steps = -1 + (intp->srch_range[demod] / 1000000); nb_steps /= 2; nb_steps = (2 * nb_steps) + 1; if (nb_steps < 0) nb_steps = 1; direction = 1; freq_step = (1000000 << 8) / (intp->mclk >> 8); init_freq = 0; for (i = 0; i < nb_steps; i++) { if (direction > 0) init_freq = init_freq + (freq_step * i); else init_freq = init_freq - (freq_step * i); direction *= -1; stv0900_write_reg(intp, DMDISTATE, 0x5C); stv0900_write_reg(intp, CFRINIT1, (init_freq >> 8) & 0xff); stv0900_write_reg(intp, CFRINIT0, init_freq & 0xff); stv0900_write_reg(intp, DMDISTATE, 0x58); msleep(10); agc2level = 0; for (j = 0; j < 10; j++) agc2level += (stv0900_read_reg(intp, AGC2I1) << 8) | stv0900_read_reg(intp, AGC2I0); agc2level /= 10; if (agc2level < minagc2level) minagc2level = agc2level; } return (u16)minagc2level; } static u32 stv0900_search_srate_coarse(struct dvb_frontend *fe) { struct stv0900_state *state = fe->demodulator_priv; struct stv0900_internal *intp = state->internal; enum fe_stv0900_demod_num demod = state->demod; int timing_lck = FALSE; s32 i, timingcpt = 0, direction = 1, nb_steps, current_step = 0, tuner_freq; u32 agc2_th, coarse_srate = 0, agc2_integr = 0, currier_step = 1200; if (intp->chip_id >= 0x30) agc2_th = 0x2e00; else agc2_th = 0x1f00; stv0900_write_bits(intp, DEMOD_MODE, 0x1f); stv0900_write_reg(intp, TMGCFG, 0x12); stv0900_write_reg(intp, TMGTHRISE, 0xf0); stv0900_write_reg(intp, TMGTHFALL, 0xe0); stv0900_write_bits(intp, SCAN_ENABLE, 1); stv0900_write_bits(intp, CFR_AUTOSCAN, 1); stv0900_write_reg(intp, SFRUP1, 0x83); stv0900_write_reg(intp, SFRUP0, 0xc0); stv0900_write_reg(intp, SFRLOW1, 0x82); stv0900_write_reg(intp, SFRLOW0, 0xa0); stv0900_write_reg(intp, DMDT0M, 0x0); stv0900_write_reg(intp, AGC2REF, 0x50); if (intp->chip_id >= 0x30) { stv0900_write_reg(intp, CARFREQ, 0x99); stv0900_write_reg(intp, SFRSTEP, 0x98); } else if (intp->chip_id >= 0x20) { stv0900_write_reg(intp, CARFREQ, 0x6a); stv0900_write_reg(intp, SFRSTEP, 0x95); } else { stv0900_write_reg(intp, CARFREQ, 0xed); stv0900_write_reg(intp, SFRSTEP, 0x73); } if (intp->symbol_rate[demod] <= 2000000) currier_step = 1000; else if (intp->symbol_rate[demod] <= 5000000) currier_step = 2000; else if (intp->symbol_rate[demod] <= 12000000) currier_step = 3000; else currier_step = 5000; nb_steps = -1 + ((intp->srch_range[demod] / 1000) / currier_step); nb_steps /= 2; nb_steps = (2 * nb_steps) + 1; if (nb_steps < 0) nb_steps = 1; else if (nb_steps > 10) { nb_steps = 11; currier_step = (intp->srch_range[demod] / 1000) / 10; } current_step = 0; direction = 1; tuner_freq = intp->freq[demod]; while ((timing_lck == FALSE) && (current_step < nb_steps)) { stv0900_write_reg(intp, DMDISTATE, 0x5f); stv0900_write_bits(intp, DEMOD_MODE, 0); msleep(50); for (i = 0; i < 10; i++) { if (stv0900_get_bits(intp, TMGLOCK_QUALITY) >= 2) timingcpt++; agc2_integr += (stv0900_read_reg(intp, AGC2I1) << 8) | stv0900_read_reg(intp, AGC2I0); } agc2_integr /= 10; coarse_srate = stv0900_get_symbol_rate(intp, intp->mclk, demod); current_step++; direction *= -1; dprintk("lock: I2C_DEMOD_MODE_FIELD =0. Search started." " tuner freq=%d agc2=0x%x srate_coarse=%d tmg_cpt=%d\n", tuner_freq, agc2_integr, coarse_srate, timingcpt); if ((timingcpt >= 5) && (agc2_integr < agc2_th) && (coarse_srate < 55000000) && (coarse_srate > 850000)) timing_lck = TRUE; else if (current_step < nb_steps) { if (direction > 0) tuner_freq += (current_step * currier_step); else tuner_freq -= (current_step * currier_step); if (intp->tuner_type[demod] == 3) stv0900_set_tuner_auto(intp, tuner_freq, intp->bw[demod], demod); else stv0900_set_tuner(fe, tuner_freq, intp->bw[demod]); } } if (timing_lck == FALSE) coarse_srate = 0; else coarse_srate = stv0900_get_symbol_rate(intp, intp->mclk, demod); return coarse_srate; } static u32 stv0900_search_srate_fine(struct dvb_frontend *fe) { struct stv0900_state *state = fe->demodulator_priv; struct stv0900_internal *intp = state->internal; enum fe_stv0900_demod_num demod = state->demod; u32 coarse_srate, coarse_freq, symb, symbmax, symbmin, symbcomp; coarse_srate = stv0900_get_symbol_rate(intp, intp->mclk, demod); if (coarse_srate > 3000000) { symbmax = 13 * (coarse_srate / 10); symbmax = (symbmax / 1000) * 65536; symbmax /= (intp->mclk / 1000); symbmin = 10 * (coarse_srate / 13); symbmin = (symbmin / 1000)*65536; symbmin /= (intp->mclk / 1000); symb = (coarse_srate / 1000) * 65536; symb /= (intp->mclk / 1000); } else { symbmax = 13 * (coarse_srate / 10); symbmax = (symbmax / 100) * 65536; symbmax /= (intp->mclk / 100); symbmin = 10 * (coarse_srate / 14); symbmin = (symbmin / 100) * 65536; symbmin /= (intp->mclk / 100); symb = (coarse_srate / 100) * 65536; symb /= (intp->mclk / 100); } symbcomp = 13 * (coarse_srate / 10); coarse_freq = (stv0900_read_reg(intp, CFR2) << 8) | stv0900_read_reg(intp, CFR1); if (symbcomp < intp->symbol_rate[demod]) coarse_srate = 0; else { stv0900_write_reg(intp, DMDISTATE, 0x1f); stv0900_write_reg(intp, TMGCFG2, 0xc1); stv0900_write_reg(intp, TMGTHRISE, 0x20); stv0900_write_reg(intp, TMGTHFALL, 0x00); stv0900_write_reg(intp, TMGCFG, 0xd2); stv0900_write_bits(intp, CFR_AUTOSCAN, 0); stv0900_write_reg(intp, AGC2REF, 0x38); if (intp->chip_id >= 0x30) stv0900_write_reg(intp, CARFREQ, 0x79); else if (intp->chip_id >= 0x20) stv0900_write_reg(intp, CARFREQ, 0x49); else stv0900_write_reg(intp, CARFREQ, 0xed); stv0900_write_reg(intp, SFRUP1, (symbmax >> 8) & 0x7f); stv0900_write_reg(intp, SFRUP0, (symbmax & 0xff)); stv0900_write_reg(intp, SFRLOW1, (symbmin >> 8) & 0x7f); stv0900_write_reg(intp, SFRLOW0, (symbmin & 0xff)); stv0900_write_reg(intp, SFRINIT1, (symb >> 8) & 0xff); stv0900_write_reg(intp, SFRINIT0, (symb & 0xff)); stv0900_write_reg(intp, DMDT0M, 0x20); stv0900_write_reg(intp, CFRINIT1, (coarse_freq >> 8) & 0xff); stv0900_write_reg(intp, CFRINIT0, coarse_freq & 0xff); stv0900_write_reg(intp, DMDISTATE, 0x15); } return coarse_srate; } static int stv0900_blind_search_algo(struct dvb_frontend *fe) { struct stv0900_state *state = fe->demodulator_priv; struct stv0900_internal *intp = state->internal; enum fe_stv0900_demod_num demod = state->demod; u8 k_ref_tmg, k_ref_tmg_max, k_ref_tmg_min; u32 coarse_srate, agc2_th; int lock = FALSE, coarse_fail = FALSE; s32 demod_timeout = 500, fec_timeout = 50, fail_cpt, i, agc2_overflow; u16 agc2_int; u8 dstatus2; dprintk("%s\n", __func__); if (intp->chip_id < 0x20) { k_ref_tmg_max = 233; k_ref_tmg_min = 143; } else { k_ref_tmg_max = 110; k_ref_tmg_min = 10; } if (intp->chip_id <= 0x20) agc2_th = STV0900_BLIND_SEARCH_AGC2_TH; else agc2_th = STV0900_BLIND_SEARCH_AGC2_TH_CUT30; agc2_int = stv0900_blind_check_agc2_min_level(intp, demod); if (agc2_int > STV0900_BLIND_SEARCH_AGC2_TH) return FALSE; if (intp->chip_id == 0x10) stv0900_write_reg(intp, CORRELEXP, 0xaa); if (intp->chip_id < 0x20) stv0900_write_reg(intp, CARHDR, 0x55); else stv0900_write_reg(intp, CARHDR, 0x20); if (intp->chip_id <= 0x20) stv0900_write_reg(intp, CARCFG, 0xc4); else stv0900_write_reg(intp, CARCFG, 0x6); stv0900_write_reg(intp, RTCS2, 0x44); if (intp->chip_id >= 0x20) { stv0900_write_reg(intp, EQUALCFG, 0x41); stv0900_write_reg(intp, FFECFG, 0x41); stv0900_write_reg(intp, VITSCALE, 0x82); stv0900_write_reg(intp, VAVSRVIT, 0x0); } k_ref_tmg = k_ref_tmg_max; do { stv0900_write_reg(intp, KREFTMG, k_ref_tmg); if (stv0900_search_srate_coarse(fe) != 0) { coarse_srate = stv0900_search_srate_fine(fe); if (coarse_srate != 0) { stv0900_get_lock_timeout(&demod_timeout, &fec_timeout, coarse_srate, STV0900_BLIND_SEARCH); lock = stv0900_get_demod_lock(intp, demod, demod_timeout); } else lock = FALSE; } else { fail_cpt = 0; agc2_overflow = 0; for (i = 0; i < 10; i++) { agc2_int = (stv0900_read_reg(intp, AGC2I1) << 8) | stv0900_read_reg(intp, AGC2I0); if (agc2_int >= 0xff00) agc2_overflow++; dstatus2 = stv0900_read_reg(intp, DSTATUS2); if (((dstatus2 & 0x1) == 0x1) && ((dstatus2 >> 7) == 1)) fail_cpt++; } if ((fail_cpt > 7) || (agc2_overflow > 7)) coarse_fail = TRUE; lock = FALSE; } k_ref_tmg -= 30; } while ((k_ref_tmg >= k_ref_tmg_min) && (lock == FALSE) && (coarse_fail == FALSE)); return lock; } static void stv0900_set_viterbi_acq(struct stv0900_internal *intp, enum fe_stv0900_demod_num demod) { s32 vth_reg = VTH12; dprintk("%s\n", __func__); stv0900_write_reg(intp, vth_reg++, 0x96); stv0900_write_reg(intp, vth_reg++, 0x64); stv0900_write_reg(intp, vth_reg++, 0x36); stv0900_write_reg(intp, vth_reg++, 0x23); stv0900_write_reg(intp, vth_reg++, 0x1e); stv0900_write_reg(intp, vth_reg++, 0x19); } static void stv0900_set_search_standard(struct stv0900_internal *intp, enum fe_stv0900_demod_num demod) { dprintk("%s\n", __func__); switch (intp->srch_standard[demod]) { case STV0900_SEARCH_DVBS1: dprintk("Search Standard = DVBS1\n"); break; case STV0900_SEARCH_DSS: dprintk("Search Standard = DSS\n"); case STV0900_SEARCH_DVBS2: break; dprintk("Search Standard = DVBS2\n"); case STV0900_AUTO_SEARCH: default: dprintk("Search Standard = AUTO\n"); break; } switch (intp->srch_standard[demod]) { case STV0900_SEARCH_DVBS1: case STV0900_SEARCH_DSS: stv0900_write_bits(intp, DVBS1_ENABLE, 1); stv0900_write_bits(intp, DVBS2_ENABLE, 0); stv0900_write_bits(intp, STOP_CLKVIT, 0); stv0900_set_dvbs1_track_car_loop(intp, demod, intp->symbol_rate[demod]); stv0900_write_reg(intp, CAR2CFG, 0x22); stv0900_set_viterbi_acq(intp, demod); stv0900_set_viterbi_standard(intp, intp->srch_standard[demod], intp->fec[demod], demod); break; case STV0900_SEARCH_DVBS2: stv0900_write_bits(intp, DVBS1_ENABLE, 0); stv0900_write_bits(intp, DVBS2_ENABLE, 1); stv0900_write_bits(intp, STOP_CLKVIT, 1); stv0900_write_reg(intp, ACLC, 0x1a); stv0900_write_reg(intp, BCLC, 0x09); if (intp->chip_id <= 0x20) /*cut 1.x and 2.0*/ stv0900_write_reg(intp, CAR2CFG, 0x26); else stv0900_write_reg(intp, CAR2CFG, 0x66); if (intp->demod_mode != STV0900_SINGLE) { if (intp->chip_id <= 0x11) stv0900_stop_all_s2_modcod(intp, demod); else stv0900_activate_s2_modcod(intp, demod); } else stv0900_activate_s2_modcod_single(intp, demod); stv0900_set_viterbi_tracq(intp, demod); break; case STV0900_AUTO_SEARCH: default: stv0900_write_bits(intp, DVBS1_ENABLE, 1); stv0900_write_bits(intp, DVBS2_ENABLE, 1); stv0900_write_bits(intp, STOP_CLKVIT, 0); stv0900_write_reg(intp, ACLC, 0x1a); stv0900_write_reg(intp, BCLC, 0x09); stv0900_set_dvbs1_track_car_loop(intp, demod, intp->symbol_rate[demod]); if (intp->chip_id <= 0x20) /*cut 1.x and 2.0*/ stv0900_write_reg(intp, CAR2CFG, 0x26); else stv0900_write_reg(intp, CAR2CFG, 0x66); if (intp->demod_mode != STV0900_SINGLE) { if (intp->chip_id <= 0x11) stv0900_stop_all_s2_modcod(intp, demod); else stv0900_activate_s2_modcod(intp, demod); } else stv0900_activate_s2_modcod_single(intp, demod); stv0900_set_viterbi_tracq(intp, demod); stv0900_set_viterbi_standard(intp, intp->srch_standard[demod], intp->fec[demod], demod); break; } } enum fe_stv0900_signal_type stv0900_algo(struct dvb_frontend *fe) { struct stv0900_state *state = fe->demodulator_priv; struct stv0900_internal *intp = state->internal; enum fe_stv0900_demod_num demod = state->demod; s32 demod_timeout = 500, fec_timeout = 50; s32 aq_power, agc1_power, i; int lock = FALSE, low_sr = FALSE; enum fe_stv0900_signal_type signal_type = STV0900_NOCARRIER; enum fe_stv0900_search_algo algo; int no_signal = FALSE; dprintk("%s\n", __func__); algo = intp->srch_algo[demod]; stv0900_write_bits(intp, RST_HWARE, 1); stv0900_write_reg(intp, DMDISTATE, 0x5c); if (intp->chip_id >= 0x20) { if (intp->symbol_rate[demod] > 5000000) stv0900_write_reg(intp, CORRELABS, 0x9e); else stv0900_write_reg(intp, CORRELABS, 0x82); } else stv0900_write_reg(intp, CORRELABS, 0x88); stv0900_get_lock_timeout(&demod_timeout, &fec_timeout, intp->symbol_rate[demod], intp->srch_algo[demod]); if (intp->srch_algo[demod] == STV0900_BLIND_SEARCH) { intp->bw[demod] = 2 * 36000000; stv0900_write_reg(intp, TMGCFG2, 0xc0); stv0900_write_reg(intp, CORRELMANT, 0x70); stv0900_set_symbol_rate(intp, intp->mclk, 1000000, demod); } else { stv0900_write_reg(intp, DMDT0M, 0x20); stv0900_write_reg(intp, TMGCFG, 0xd2); if (intp->symbol_rate[demod] < 2000000) stv0900_write_reg(intp, CORRELMANT, 0x63); else stv0900_write_reg(intp, CORRELMANT, 0x70); stv0900_write_reg(intp, AGC2REF, 0x38); intp->bw[demod] = stv0900_carrier_width(intp->symbol_rate[demod], intp->rolloff); if (intp->chip_id >= 0x20) { stv0900_write_reg(intp, KREFTMG, 0x5a); if (intp->srch_algo[demod] == STV0900_COLD_START) { intp->bw[demod] += 10000000; intp->bw[demod] *= 15; intp->bw[demod] /= 10; } else if (intp->srch_algo[demod] == STV0900_WARM_START) intp->bw[demod] += 10000000; } else { stv0900_write_reg(intp, KREFTMG, 0xc1); intp->bw[demod] += 10000000; intp->bw[demod] *= 15; intp->bw[demod] /= 10; } stv0900_write_reg(intp, TMGCFG2, 0xc1); stv0900_set_symbol_rate(intp, intp->mclk, intp->symbol_rate[demod], demod); stv0900_set_max_symbol_rate(intp, intp->mclk, intp->symbol_rate[demod], demod); stv0900_set_min_symbol_rate(intp, intp->mclk, intp->symbol_rate[demod], demod); if (intp->symbol_rate[demod] >= 10000000) low_sr = FALSE; else low_sr = TRUE; } if (intp->tuner_type[demod] == 3) stv0900_set_tuner_auto(intp, intp->freq[demod], intp->bw[demod], demod); else stv0900_set_tuner(fe, intp->freq[demod], intp->bw[demod]); agc1_power = MAKEWORD(stv0900_get_bits(intp, AGCIQ_VALUE1), stv0900_get_bits(intp, AGCIQ_VALUE0)); aq_power = 0; if (agc1_power == 0) { for (i = 0; i < 5; i++) aq_power += (stv0900_get_bits(intp, POWER_I) + stv0900_get_bits(intp, POWER_Q)) / 2; aq_power /= 5; } if ((agc1_power == 0) && (aq_power < IQPOWER_THRESHOLD)) { intp->result[demod].locked = FALSE; signal_type = STV0900_NOAGC1; dprintk("%s: NO AGC1, POWERI, POWERQ\n", __func__); } else { stv0900_write_bits(intp, SPECINV_CONTROL, intp->srch_iq_inv[demod]); if (intp->chip_id <= 0x20) /*cut 2.0*/ stv0900_write_bits(intp, MANUALSX_ROLLOFF, 1); else /*cut 3.0*/ stv0900_write_bits(intp, MANUALS2_ROLLOFF, 1); stv0900_set_search_standard(intp, demod); if (intp->srch_algo[demod] != STV0900_BLIND_SEARCH) stv0900_start_search(intp, demod); } if (signal_type == STV0900_NOAGC1) return signal_type; if (intp->chip_id == 0x12) { stv0900_write_bits(intp, RST_HWARE, 0); msleep(3); stv0900_write_bits(intp, RST_HWARE, 1); stv0900_write_bits(intp, RST_HWARE, 0); } if (algo == STV0900_BLIND_SEARCH) lock = stv0900_blind_search_algo(fe); else if (algo == STV0900_COLD_START) lock = stv0900_get_demod_cold_lock(fe, demod_timeout); else if (algo == STV0900_WARM_START) lock = stv0900_get_demod_lock(intp, demod, demod_timeout); if ((lock == FALSE) && (algo == STV0900_COLD_START)) { if (low_sr == FALSE) { if (stv0900_check_timing_lock(intp, demod) == TRUE) lock = stv0900_sw_algo(intp, demod); } } if (lock == TRUE) signal_type = stv0900_get_signal_params(fe); if ((lock == TRUE) && (signal_type == STV0900_RANGEOK)) { stv0900_track_optimization(fe); if (intp->chip_id <= 0x11) { if ((stv0900_get_standard(fe, 0) == STV0900_DVBS1_STANDARD) && (stv0900_get_standard(fe, 1) == STV0900_DVBS1_STANDARD)) { msleep(20); stv0900_write_bits(intp, RST_HWARE, 0); } else { stv0900_write_bits(intp, RST_HWARE, 0); msleep(3); stv0900_write_bits(intp, RST_HWARE, 1); stv0900_write_bits(intp, RST_HWARE, 0); } } else if (intp->chip_id >= 0x20) { stv0900_write_bits(intp, RST_HWARE, 0); msleep(3); stv0900_write_bits(intp, RST_HWARE, 1); stv0900_write_bits(intp, RST_HWARE, 0); } if (stv0900_wait_for_lock(intp, demod, fec_timeout, fec_timeout) == TRUE) { lock = TRUE; intp->result[demod].locked = TRUE; if (intp->result[demod].standard == STV0900_DVBS2_STANDARD) { stv0900_set_dvbs2_rolloff(intp, demod); stv0900_write_bits(intp, RESET_UPKO_COUNT, 1); stv0900_write_bits(intp, RESET_UPKO_COUNT, 0); stv0900_write_reg(intp, ERRCTRL1, 0x67); } else { stv0900_write_reg(intp, ERRCTRL1, 0x75); } stv0900_write_reg(intp, FBERCPT4, 0); stv0900_write_reg(intp, ERRCTRL2, 0xc1); } else { lock = FALSE; signal_type = STV0900_NODATA; no_signal = stv0900_check_signal_presence(intp, demod); intp->result[demod].locked = FALSE; } } if ((signal_type != STV0900_NODATA) || (no_signal != FALSE)) return signal_type; if (intp->chip_id > 0x11) { intp->result[demod].locked = FALSE; return signal_type; } if ((stv0900_get_bits(intp, HEADER_MODE) == STV0900_DVBS_FOUND) && (intp->srch_iq_inv[demod] <= STV0900_IQ_AUTO_NORMAL_FIRST)) signal_type = stv0900_dvbs1_acq_workaround(fe); return signal_type; }