/* * Driver for the DVB-C STV0297 demodulator * * a special, temporary version for the CableStar2 which can hopefully be * rewritten to be used with the other stv0297-based cards * * Copyright (C) 2005 Patrick Boettcher * * Copyright (C) 2004 Andrew de Quincey * * Copyright (C) 2003-2004 Dennis Noermann * * 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. * * Changelog 2005-03-24 - revamp in order to get the CableStar2 working * (thanks to John Jurrius, BBTI Inc.) */ #include #include #include #include #include "dvb_frontend.h" #include "stv0297_cs2.h" #include "stv0297_priv.h" struct stv0297_state { struct i2c_adapter *i2c; const struct stv0297_config *config; struct dvb_frontend frontend; int invert; u32 lastber; unsigned long base_freq; }; static int debug; module_param(debug, int, 0644); MODULE_PARM_DESC(debug, "set debugging level (1=info,2=i2c,4=calc (|-able))."); #define dprintk(level,x...) if (level & debug) printk(x) #define dbufout(b,l) if (debug & 0x02) {\ int i; \ for (i = 0; i < l; i++) \ deb_i2c("%02x ",b[i]); \ } #define deb_info(args...) dprintk(0x01,args) #define deb_i2c(args...) dprintk(0x02,args) #define deb_calc(args...) dprintk(0x04,args) #define abs64(x) (x) < 0 ? -(x) : (x); static s64 div64(s64 dividend,s64 divisor) { s64 rdiv,remainder,result=0,i; s64 divbits,valbits; s64 neg = 0; if (divisor == 0) return 0; if (dividend < 0) neg = !neg; if (divisor < 0) neg = !neg; dividend = abs64(dividend); divisor = abs64(divisor); deb_calc("%Ld / %Ld\n",dividend,divisor); for (i = 1,divbits = 0; i < divisor; i *= 2, divbits++); for (i = 1,valbits = 0; i < dividend; i *= 2, valbits++); rdiv = divisor << (valbits-divbits); for (i = valbits; i >= divbits; i--) { result <<= 1; if (dividend >= rdiv) { result |= 1; dividend -= rdiv; } rdiv >>= 1; } remainder = dividend; if (neg) return -result; else return result; } static int stv0297_writeregsI(struct stv0297_state *state, u8 reg1, u8 *b, u8 len) { int ret; u8 buf[len+1]; struct i2c_msg msg = { .addr = state->config->demod_address, .flags = 0, .buf = buf, .len = len+1 }; deb_i2c("wr: %02x ",reg1); dbufout(b,len); deb_i2c("\n"); buf[0] = reg1; memcpy(&buf[1],b,len); if ((ret = i2c_transfer(state->i2c, &msg, 1)) != 1) deb_info("%s: writereg error (reg == 0x%02x, datalen == %d, " "ret == %i)\n", __FUNCTION__, reg1, len, ret); return (ret != 1) ? -EREMOTEIO : 0; } static int stv0297_writeregI(struct stv0297_state *state, u8 reg, u8 data) { return stv0297_writeregsI(state,reg,&data,1); } static int stv0297_readregsI(struct stv0297_state *state, u8 reg1, u8 *b, u8 len) { int ret; struct i2c_msg msg[] = { { .addr = state->config->demod_address, .flags = 0, .buf = ®1,.len = 1}, { .addr = state->config->demod_address, .flags = I2C_M_RD, .buf = b, .len = len} }; /* the Nexus-CA needs a STOP between the register and data, but this breaks * the common i2c-routine, where you have 2 i2c-message in a package * when doing a read operation TODO */ if ((ret = i2c_transfer(state->i2c, msg, 2)) != 2) { deb_info("%s: readreg error (reg == 0x%02x, ret == %i)\n", __FUNCTION__, reg1, ret); return -EREMOTEIO; } deb_i2c("rd: %02x ",reg1); dbufout(b,len); deb_i2c("\n"); /* if ((ret = i2c_transfer(state->i2c, &msg[1], 1)) != 1) { deb_info("%s: readreg error (reg == 0x%02x, ret == %i)\n", __FUNCTION__, reg1, ret); return -1; }*/ return 0; } static int stv0297_readregI(struct stv0297_state *state, u8 reg) { int ret; u8 b[] = { 0 }; if ((ret = stv0297_readregsI(state,reg,b,1)) < 0) return ret; return b[0]; } static int stv0297_writereg_maskI(struct stv0297_state *state, u8 reg, u8 mask, u8 data) { int val; if ((val = stv0297_readregI(state, reg)) < 0) return val; val &= ~mask; val |= (data & mask); return stv0297_writeregI(state, reg, val); } static void stv0297_set_symbolrate(struct stv0297_state *state, u32 srate) { u64 tmp = div64(((u64) srate) << 32,state->config->fclk); deb_info("symbolrate: %u, regval: %016Lx\n",srate,tmp ); stv0297_writeregI(state, ST_LOOP5, tmp & 0xFF); stv0297_writeregI(state, ST_LOOP6, tmp >> 8); stv0297_writeregI(state, ST_LOOP7, tmp >> 16); stv0297_writeregI(state, ST_LOOP8, tmp >> 24); } static u32 stv0297_get_symbolrate(struct stv0297_state *state) { u64 tmp; tmp = stv0297_readregI(state, ST_LOOP5); tmp |= stv0297_readregI(state, ST_LOOP6) << 8; tmp |= stv0297_readregI(state, ST_LOOP7) << 16; tmp |= stv0297_readregI(state, ST_LOOP8) << 24; tmp *= state->config->fclk; tmp >>= 32; return (u32) tmp; } static void stv0297_set_crl(struct stv0297_state *state, struct dvb_frontend_parameters *p, s32 car_offset) { u8 crlbuf[10] = { 0 }; u32 sweeprate,iphase; u64 tmp; u16 sweepvalue; if (p->u.qam.symbol_rate < 2000000UL) sweeprate = 80000UL; else if (p->u.qam.symbol_rate < 3500000UL) sweeprate = 200000UL; else if (p->u.qam.symbol_rate < 4500000UL) sweeprate = 800000UL; else sweeprate = 1000000UL; tmp = div64(((u64) sweeprate) << 29, p->u.qam.symbol_rate); sweepvalue = div64(tmp,1000000UL); if (sweepvalue & 1) sweepvalue += 2; /* round up, adjust */ sweepvalue >>=1; deb_calc("symbol_rate: %u, sweeprate: %u, sweepvalue: %u (%x)\n",p->u.qam.symbol_rate,sweeprate,sweepvalue,sweepvalue); crlbuf[0] = sweepvalue & 0xff; crlbuf[9] = (sweepvalue >> 4) & 0xf0; if (p->u.qam.modulation == QAM_256) crlbuf[1] = 0x58; else crlbuf[1] = 0x49; crlbuf[2] = 0x0a; crlbuf[3] = crlbuf[4] = crlbuf[5] = 0x00; /* carrier offset */ if (p->u.qam.modulation == QAM_16) iphase = 0; else iphase = div64(((u64) car_offset) << 28, p->u.qam.symbol_rate); deb_calc("symbol_rate: %u, car_offset %d, iphase: %u (%x)\n",p->u.qam.symbol_rate,car_offset,iphase,iphase); crlbuf[6] = iphase & 0xff; crlbuf[7] = (iphase >> 8) & 0xff; crlbuf[8] = (iphase >> 16) & 0xff; crlbuf[9] |= (iphase >> 24) & 0x0f; stv0297_writeregsI(state,CRL_0,crlbuf,sizeof(crlbuf)); } /* static long stv0297_get_carrieroffset(struct stv0297_state *state) { s64 tmp; stv0297_writeregI(state, CRL_11, 0x00); tmp = stv0297_readregI(state, CRL_6); tmp |= (stv0297_readregI(state, CRL_7) << 8); tmp |= (stv0297_readregI(state, CRL_8) << 16); tmp |= (stv0297_readregI(state, CRL_9) & 0x0F) << 24; tmp *= stv0297_get_symbolrate(state); tmp >>= 28; return (s32) tmp; } */ static int stv0297_set_modulation(struct stv0297_state *state, fe_modulation_t modulation) { int val = stv0297_readregI(state,EQU_0) & STV0297_CLEARQAM; switch (modulation) { case QAM_16: val = STV0297_QAM16; break; case QAM_32: val = STV0297_QAM32; break; case QAM_64: val = STV0297_QAM64; break; case QAM_128: val = STV0297_QAM128; break; case QAM_256: val = STV0297_QAM256; break; default: return -EINVAL; } stv0297_writeregI(state, EQU_0, val); return 0; } static int stv0297_set_inversion(struct stv0297_state *state, fe_spectral_inversion_t inversion) { int val = 0; switch (inversion) { case INVERSION_OFF: val = 0; break; case INVERSION_ON: val = 1; break; default: return -EINVAL; } stv0297_writereg_maskI(state, CTRL_3, 0x08, val << 3); return 0; } int stv0297_cs2_enable_plli2c(struct dvb_frontend *fe) { struct stv0297_state *state = fe->demodulator_priv; stv0297_writeregI(state, CTRL_7, 0x78); stv0297_writeregI(state, CTRL_6, 0xc8); return 0; } static int stv0297_init(struct dvb_frontend *fe) { struct stv0297_state *state = fe->demodulator_priv; state->invert = 0; return 0; } static int stv0297_reset(struct dvb_frontend *fe) { struct stv0297_state *state = fe->demodulator_priv; /* soft reset */ stv0297_writeregI(state, CTRL_0, 0x01); stv0297_writeregI(state, CTRL_0, 0x00); /* reset deinterleaver */ stv0297_writeregI(state, CTRL_1, 0x01); stv0297_writeregI(state, CTRL_1, 0x00); /* device specific initialization here */ /* state->config_struct->demod_init(fe); */ /* this is the initialization for the CableStar, maybe it works for other too */ { u8 equ0[] = { 0x48, 0x58 }; /* QAM64, for QAM256: { 0x3a, 0x8a } */ u8 equ3[] = { 0x00, 0x00 }; u8 equ7[] = { 0x00, 0x00 }; u8 delagc[] = { 0xff, /* gold, original: 0x6b or 0x54 */ 0x9d, 0xff, /* ALPS, MT203x: 0x54 (Gold) 0xb5 */ 0x00, /* gold, original: 0x44 or 0xff, 0xb0, 0x8b */ 0x29, /* original, or 0x00, 0x07, 0xb3 */ 0x55, /* gold */ 0x80, /* original */ 0x6e, 0x9c }; u8 wbagc0[] = { 0x1a, 0xfe, 0x33, 0x00 /* original */, 0xff, 0x00, 0x00 }; u8 wbagc9[] = { 0x04, 0x51, 0xf8 }; #define GAIN_PATH0 0x01 #define GAIN_PATH1 0x04 #define INTEGRAL_GAIN_HI 0X0 #define INTEGRAL_GAIN_LO 0X06 #define DIRECT_GAIN_HI 0x0 #define DIRECT_GAIN_LO 0x06 #define STLOOP_ALGORITHM 0x00 u8 st_loop2[] = { (GAIN_PATH0 << 5) | (GAIN_PATH1 << 2) | INTEGRAL_GAIN_HI, DIRECT_GAIN_LO }; u8 st_loop9[] = { DIRECT_GAIN_HI | INTEGRAL_GAIN_LO, 0x5e /* original */, 0x04 | (STLOOP_ALGORITHM << 4) }; u8 pmfagc0[] = { 0xff, 0x04, 0x00, 0x00, 0x0c }; u8 ctrl[] = { 0x20, 0x00, 0x30, stv0297_readregI(state,CTRL_3) & SPECTRAL_INVERSION_TOGGLE, /* Clear, except the Spectral Inversion Bit. */ 0x04,0x22,0x08,0x1b,0x00 /* Gold */,0x00 }; u8 deint_sync[] = { 0x00, 0x04 }; u8 bert[] = { 0x86, 0x00, 0x00 }; /* bit4 of byte0 toggles byte-wise-counting */ u8 deint[] = { 0x91, 0x0b }; u8 outformat[] = { 0x5b, 0x10, 0x12 }; u8 rs_desc0[] = { 0x02, 0x00, 0x00, 0x00, 0x02, 0x00 }; u8 rs_desc14[] = { 0x00, 0x01 }; stv0297_writeregsI(state,EQU_0,equ0,sizeof(equ0)); stv0297_writeregsI(state,EQU_3,equ3,sizeof(equ3)); stv0297_writeregsI(state,EQU_7,equ7,sizeof(equ7)); stv0297_writeregsI(state,DELAGC_0,delagc,sizeof(delagc)); stv0297_writeregsI(state,WBAGC_0,wbagc0,sizeof(wbagc0)); stv0297_writeregsI(state,WBAGC_9,wbagc9,sizeof(wbagc9)); stv0297_writeregsI(state,ST_LOOP2,st_loop2,sizeof(st_loop2)); stv0297_writeregsI(state,ST_LOOP9,st_loop9,sizeof(st_loop9)); stv0297_writeregsI(state,PMFAGC_0,pmfagc0,sizeof(pmfagc0)); stv0297_writeregsI(state,CTRL_0,ctrl,sizeof(ctrl)); stv0297_writeregsI(state,DEINT_SYNC_0,deint_sync,sizeof(deint_sync)); stv0297_writeregsI(state,BERT_0,bert,sizeof(bert)); stv0297_writeregsI(state,DEINT_0,deint,sizeof(deint)); stv0297_writeregsI(state,OUTFORMAT_0,outformat,sizeof(outformat)); stv0297_writeregsI(state,RS_DESC_0,rs_desc0,sizeof(rs_desc0)); stv0297_writeregsI(state,RS_DESC_14,rs_desc14,sizeof(rs_desc14)); } state->lastber = 0; if (state->config->pll_init) state->config->pll_init(fe); return 0; } static void stv0297_set_initdemod(struct stv0297_state *state, s32 offset) { u16 initdemod = div64( (s64) (state->config->demodfreq - offset) << 16 , state->config->fclk); u8 initdem[6]; deb_calc("demodfreq: %d, offset: %d, fclk: %d, initdemod: %d (%x)\n", state->config->demodfreq, offset, state->config->fclk, initdemod, initdemod); initdem[0] = initdemod & 0xff; initdem[1] = (initdemod >> 8) & 0xff; initdem[2] = 0x00; initdem[3] = 0x00; initdem[4] = 0x40; initdem[5] = 0x88; stv0297_writeregsI(state,INITDEM_0, initdem,6); } #define try(expr,num,sleep) \ for (i = 0; i < (num) && !(expr); i++) msleep(sleep); static int stv0297_set_frontend(struct dvb_frontend *fe, struct dvb_frontend_parameters *p) { struct stv0297_state *state = fe->demodulator_priv; int i; u8 equ_save[2]; fe_spectral_inversion_t inversion; switch (p->inversion) { case INVERSION_AUTO: /* fall through wanted */ case INVERSION_OFF: inversion = state->invert ? INVERSION_ON : INVERSION_OFF; break; case INVERSION_ON: inversion = state->invert ? INVERSION_OFF : INVERSION_ON; break; default: return -EINVAL; } deb_info("spectrum inversion: %s\n",inversion == INVERSION_ON ? "on" : "off"); stv0297_reset(fe); state->config->pll_set(fe, p); /* clear software interrupts */ stv0297_writeregI(state, CTRL_2, 0x00); /* set initial demodulation frequency */ stv0297_set_initdemod(state, 0); /* setup AGC */ stv0297_writeregI(state, WBAGC_3, 0x00); /* Wide Band AGC agc2sd initialisation: mid-range */ stv0297_writeregI(state, WBAGC_1, 0x00); // stv0297_writereg_maskI(state, WBAGC_2, 0x03, 0x01); /* Wide Band AGC1&AGC2 nofreeze */ stv0297_writereg_maskI(state, DELAGC_6, 0x7f, 0x00); /* PMF AGC accumulator reset */ stv0297_writereg_maskI(state, PMFAGC_1, 0x80, 0x80); stv0297_writeregI(state, PMFAGC_2, 0x00); stv0297_writeregI(state, PMFAGC_3, 0x00); stv0297_writereg_maskI(state, PMFAGC_4, 0x7f, 0x00); /* Force AGC ACQ low */ stv0297_writereg_maskI(state, WBAGC_3, 0x08, 0x00); /* Disable unlock forcing. */ stv0297_writereg_maskI(state, PMFAGC_1, 0x80, 0x00); /* setup STL * Phase clear */ stv0297_writereg_maskI(state, ST_LOOP10, 0x20, 0x20); /* STL integral path clear */ stv0297_writereg_maskI(state, ST_LOOP11, 0x02, 0x02); /* STL integral path clear release */ stv0297_writereg_maskI(state, ST_LOOP11, 0x02, 0x00); /* integral path enabled only */ stv0297_writereg_maskI(state, ST_LOOP11, 0x01, 0x00); /* direct path immediatly enabled */ stv0297_writereg_maskI(state, ST_LOOP10, 0x40, 0x40); /* disable frequency sweep */ stv0297_writereg_maskI(state, CRL_10, 0x01, 0x00); /* reset deinterleaver */ stv0297_writereg_maskI(state, CTRL_1, 0x01, 0x01); stv0297_writereg_maskI(state, CTRL_1, 0x01, 0x00); /* ??? */ stv0297_writereg_maskI(state, CTRL_3, 0x20, 0x20); stv0297_writereg_maskI(state, CTRL_3, 0x20, 0x00); /* Reed-Salomon clear */ stv0297_writereg_maskI(state, CTRL_3, 0x10, 0x10); stv0297_writereg_maskI(state, CTRL_3, 0x10, 0x00); /* Equalizer values capture */ stv0297_readregsI(state, EQU_0, equ_save, 2); /* reset equalizer */ stv0297_writereg_maskI(state, CTRL_4, 0x01, 0x01); stv0297_writereg_maskI(state, CTRL_4, 0x01, 0x00); /* Equalizer values restore */ stv0297_writeregsI(state, EQU_0, equ_save, 2); /* data comes from internal A/D */ stv0297_writereg_maskI(state, CTRL_7, 0x80, 0x00); /* set parameters */ stv0297_set_modulation(state, p->u.qam.modulation); stv0297_set_symbolrate(state, p->u.qam.symbol_rate); stv0297_set_crl(state, p, -130000); stv0297_set_inversion(state, inversion); stv0297_writereg_maskI(state, EQU_0, 0x0f, 0x09); stv0297_writeregI(state, EQU_1, 0x69); /* only disable corner detection for QAM256 and QAM128, otherwise, enable it */ if (p->u.qam.modulation == QAM_256 || p->u.qam.modulation == QAM_128) stv0297_writereg_maskI(state, CTRL_8, 0x08, 0x00); else stv0297_writereg_maskI(state, CTRL_8, 0x08, 0x08); /* Phase clear release */ stv0297_writereg_maskI(state, ST_LOOP10, 0x20, 0x00); /* Sweep Enable */ stv0297_writereg_maskI(state, CRL_10, 0x01, 0x01); msleep(10); /* Clear wide band AGC */ stv0297_writereg_maskI(state, WBAGC_3, 0x40, 0x40); /* enable wide band AGC */ stv0297_writereg_maskI(state, WBAGC_3, 0x10, WAGC_EN); deb_info("initialized - waiting for the locks now\n"); /* wait for WBAGC lock */ deb_info("waiting for WBAGC lock\n"); try(stv0297_readregI(state, WBAGC_3) & 0x08, 200, 10); if (i == 200) goto timeout; deb_info("WBAGC has lock\n"); msleep(20); /* wait for equalizer 1 lock */ deb_info("waiting for equalizer 1 lock\n"); try(stv0297_readregI(state, CTRL_2) & 0x04, 400, 10); if (i == 400) goto timeout; deb_info("equalizer 1 has lock\n"); /* wait for equalizer 2 lock and if it's stable */ deb_info("waiting for equalizer 2 lock\n"); try(stv0297_readregI(state, CTRL_2) & 0x08, 200, 10); if (i == 200) goto timeout; deb_info("equalizer 2 has lock\n"); msleep(20); if (!(stv0297_readregI(state, CTRL_2) & 0x08)) goto timeout; deb_info("equalizer 2 is stable\n"); /* we have modulation, do we have data */ deb_info("waiting for data lock\n"); try(stv0297_readregI(state,RS_DESC_15) & 0x80,5,10); if (i == 5) { /* try to invert the inversion during the next run */ deb_info("no data lock achieved, trying to invert the spectrum in the next run.\n"); state->invert = !state->invert; goto timeout; } deb_info("we have data lock\n"); /* Turn off corner detection */ stv0297_writereg_maskI(state, CTRL_8,0x08, 0x00); /* success!! */ state->base_freq = p->frequency; return 0; timeout: deb_info("timed out\n"); stv0297_writereg_maskI(state, CRL_10, 0x01, 0x00); return 0; } static int stv0297_get_frontend(struct dvb_frontend *fe, struct dvb_frontend_parameters *p) { struct stv0297_state *state = fe->demodulator_priv; int reg_00, reg_83; reg_00 = stv0297_readregI(state, EQU_0); reg_83 = stv0297_readregI(state, CTRL_3); p->frequency = state->base_freq; if (reg_83 & 0x08) p->inversion = INVERSION_ON; else p->inversion = INVERSION_OFF; p->u.qam.symbol_rate = stv0297_get_symbolrate(state); p->u.qam.fec_inner = FEC_NONE; switch (reg_00 & 0x70) { case STV0297_QAM16: p->u.qam.modulation = QAM_16; break; case STV0297_QAM32: p->u.qam.modulation = QAM_32; break; case STV0297_QAM128: p->u.qam.modulation = QAM_128; break; case STV0297_QAM256: p->u.qam.modulation = QAM_256; break; case STV0297_QAM64: p->u.qam.modulation = QAM_64; break; } return 0; } static int stv0297_sleep(struct dvb_frontend *fe) { struct stv0297_state *state = fe->demodulator_priv; deb_info("stv0297 is going to bed.\n"); stv0297_writereg_maskI(state, CTRL_0, 1, 1); return 0; } static int stv0297_read_status(struct dvb_frontend *fe, fe_status_t * status) { struct stv0297_state *state = fe->demodulator_priv; u8 ctrl_2 = stv0297_readregI(state, CTRL_2); *status = 0; /* The following status assignments are only guesses, but we wanted to have a * kind of grade here */ if (stv0297_readregI(state, WBAGC_3) & 0x08) *status |= FE_HAS_SIGNAL; if (ctrl_2 & 0x04) *status |= FE_HAS_CARRIER; if (ctrl_2 & 0x08) *status |= FE_HAS_VITERBI; if (stv0297_readregI(state, RS_DESC_15) & 0x80) *status |= FE_HAS_SYNC | FE_HAS_LOCK; return 0; } static int stv0297_read_signal_strength(struct dvb_frontend *fe, u16 * strength) { struct stv0297_state *state = (struct stv0297_state *) fe->demodulator_priv; u8 STRENGTH[2]; stv0297_readregsI(state, WBAGC_1, STRENGTH, 2); *strength = (STRENGTH[1] & 0x03) << 8 | STRENGTH[0]; return 0; } static int stv0297_read_snr(struct dvb_frontend *fe, u16 * snr) { struct stv0297_state *state = (struct stv0297_state *) fe->demodulator_priv; u8 SNR[2]; stv0297_readregsI(state, EQU_7, SNR, 2); *snr = SNR[1] << 8 | SNR[0]; return 0; } static int stv0297_read_ber(struct dvb_frontend *fe, u32 * ber) { struct stv0297_state *state = (struct stv0297_state *) fe->demodulator_priv; u8 BER[3]; *ber = 0; stv0297_readregsI(state, BERT_0, BER, 3); if (!(BER[0] & 0x80)) { state->lastber = (BER[0] & 0x07 << 16) | (BER[2] << 8) | BER[1]; /* reset the BER counter */ BER[0] |= 0x80; BER[1] = BER[2] = 0x00; stv0297_writeregsI(state, BERT_0, BER, 3); } *ber = state->lastber; return 0; } static int stv0297_read_ucblocks(struct dvb_frontend *fe, u32 * ucblocks) { struct stv0297_state *state = fe->demodulator_priv; u8 block_count[6]; stv0297_writeregI(state,RS_DESC_15,0x03); /* freeze the counters */ stv0297_readregsI(state,RS_DESC_0,block_count,6); stv0297_writeregI(state,RS_DESC_15,0x02); /* clear the counters */ stv0297_writeregI(state,RS_DESC_15,0x01); /* re-enable the counters */ /* LastBlockCount = (block_count[1] << 8) | block_count[0]; */ /* LastCorrectedBlockCount = (block_count[3] << 8) | block_count[2]; */ *ucblocks = (block_count[5] << 8) | block_count[4]; return 0; } static void stv0297_release(struct dvb_frontend *fe) { struct stv0297_state *state = fe->demodulator_priv; kfree(state); } static struct dvb_frontend_ops stv0297_ops; struct dvb_frontend *stv0297_cs2_attach(const struct stv0297_config *config, struct i2c_adapter *i2c) { struct stv0297_state *state = NULL; /* allocate memory for the internal state */ state = kmalloc(sizeof(struct stv0297_state), GFP_KERNEL); if (state == NULL) goto error; /* setup the state */ state->config = config; state->i2c = i2c; state->base_freq = 0; /* check if the demod is there */ if ((stv0297_readregI(state, CTRL_0) & 0x70) != 0x20) goto error; /* create dvb_frontend */ memcpy(&state->frontend.ops, &stv0297_ops, sizeof(struct dvb_frontend_ops)); state->frontend.demodulator_priv = state; return &state->frontend; error: kfree(state); return NULL; } static struct dvb_frontend_ops stv0297_ops = { .info = { .name = "ST STV0297 DVB-C", .type = FE_QAM, .frequency_min = 64000000, .frequency_max = 1300000000, .frequency_stepsize = 62500, .symbol_rate_min = 870000, .symbol_rate_max = 11700000, .caps = FE_CAN_QAM_16 | FE_CAN_QAM_32 | FE_CAN_QAM_64 | FE_CAN_QAM_128 | FE_CAN_QAM_256 | FE_CAN_FEC_AUTO}, .release = stv0297_release, .init = stv0297_init, .sleep = stv0297_sleep, .set_frontend = stv0297_set_frontend, .get_frontend = stv0297_get_frontend, .read_status = stv0297_read_status, .read_ber = stv0297_read_ber, .read_signal_strength = stv0297_read_signal_strength, .read_snr = stv0297_read_snr, .read_ucblocks = stv0297_read_ucblocks, }; MODULE_DESCRIPTION("ST STV0297 DVB-C Demodulator driver"); MODULE_AUTHOR("Dennis Noermann, Andrew de Quincey and Patrick Boettcher"); MODULE_LICENSE("GPL"); MODULE_VERSION("1.0"); EXPORT_SYMBOL(stv0297_cs2_attach); EXPORT_SYMBOL(stv0297_cs2_enable_plli2c);