⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 ves1820.c

📁 linux dvb的文件
💻 C
字号:
/*     VES1820  - Single Chip Cable Channel Receiver driver module               used on the the Siemens DVB-C cards    Copyright (C) 1999 Convergence Integrated Media GmbH <ralph@convergence.de>    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 <linux/init.h>#include <linux/module.h>#include <linux/delay.h>#include "compat.h"#include "dvb_frontend.h"#if 0#define dprintk(x...) printk(x)#else#define dprintk(x...)#endif/** *  since we need only a few bits to store internal state we don't allocate *  extra memory but use frontend->data as bitfield */#define SET_PWM(data,pwm) do { 		\	(int) data &= ~0xff; 		\	(int) data |= pwm; 		\} while (0)#define SET_REG0(data,reg0) do {	\	(int) data &= ~(0xff << 8); 	\	(int) data |= reg0 << 8; 	\} while (0)#define SET_TUNER(data,type) do {	\	(int) data &= ~(0xff << 16); 	\	(int) data |= type << 16;	\} while (0)#define SET_DEMOD_ADDR(data,type) do {	\	(int) data &= ~(0xff << 24); 	\	(int) data |= type << 24;	\} while (0)#define GET_PWM(data) ((u8) ((int) data & 0xff))#define GET_REG0(data) ((u8) (((int) data >> 8) & 0xff))#define GET_TUNER(data) ((u8) (((int) data >> 16) & 0xff))#define GET_DEMOD_ADDR(data) ((u8) (((int) data >> 24) & 0xff))static inlinevoid ddelay (int ms){	current->state=TASK_INTERRUPTIBLE;	schedule_timeout((HZ*ms)/1000);}staticstruct dvb_frontend_info ves1820_info = {	.name = "VES1820 based DVB-C frontend",	.type = FE_QAM,	.frequency_stepsize = 62500,	.frequency_min = 51000000,	.frequency_max = 858000000,	.symbol_rate_min = (57840000/2)/64,     /* SACLK/64 == (XIN/2)/64 */	.symbol_rate_max = (57840000/2)/4,      /* SACLK/4 */#if 0	frequency_tolerance: ???,	symbol_rate_tolerance: ???,  /* ppm */  /* == 8% (spec p. 5) */	notifier_delay: ?,#endif	.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 | FE_CAN_INVERSION_AUTO |		FE_CAN_CLEAN_SETUP};staticu8 ves1820_inittab [] ={	0x69, 0x6A, 0x9B, 0x0A, 0x52, 0x46, 0x26, 0x1A,	0x43, 0x6A, 0xAA, 0xAA, 0x1E, 0x85, 0x43, 0x28,	0xE0, 0x00, 0xA1, 0x00, 0x00, 0x00, 0x00, 0x00,	0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00,	0x02, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,	0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,	0x00, 0x00, 0x00, 0x00, 0x40};staticint ves1820_writereg (struct dvb_frontend *fe, u8 reg, u8 data){	u8 addr = GET_DEMOD_ADDR(fe->data);        u8 buf[] = { 0x00, reg, data };	struct i2c_msg msg = { addr: addr, flags: 0, buf: buf, len: 3 };	struct dvb_i2c_bus *i2c = fe->i2c;        int ret;	ret = i2c->xfer (i2c, &msg, 1);	if (ret != 1)		dprintk("%s: writereg error "			"(reg == 0x%02x, val == 0x%02x, ret == %i)\n",			__FUNCTION__, reg, data, ret);	mdelay(10);	return (ret != 1) ? -EREMOTEIO : 0;}staticu8 ves1820_readreg (struct dvb_frontend *fe, u8 reg){	u8 b0 [] = { 0x00, reg };	u8 b1 [] = { 0 };	u8 addr = GET_DEMOD_ADDR(fe->data);	struct i2c_msg msg [] = { { addr: addr, flags: 0, buf: b0, len: 2 },	                   { addr: addr, flags: I2C_M_RD, buf: b1, len: 1 } };	struct dvb_i2c_bus *i2c = fe->i2c;	int ret;	ret = i2c->xfer (i2c, msg, 2);	if (ret != 2)		dprintk("%s: readreg error (ret == %i)\n", __FUNCTION__, ret);	return b1[0];}staticint tuner_write (struct dvb_i2c_bus *i2c, u8 addr, u8 data [4]){        int ret;        struct i2c_msg msg = { addr: addr, flags: 0, buf: data, len: 4 };        ret = i2c->xfer (i2c, &msg, 1);        if (ret != 1)                printk("%s: i/o error (ret == %i)\n", __FUNCTION__, ret);        return (ret != 1) ? -EREMOTEIO : 0;}/** *   set up the downconverter frequency divisor for a *   reference clock comparision frequency of 62.5 kHz. */staticint tuner_set_tv_freq (struct dvb_frontend *fe, u32 freq){        u32 div;	static u8 addr [] = { 0x61, 0x62 };	static u8 byte3 [] = { 0x8e, 0x85 };	int tuner_type = GET_TUNER(fe->data);        u8 buf [4];	if (tuner_type == 0xff)     /*  PLL not reachable over i2c ...  */		return 0;	div = (freq + 36250000 + 31250) / 62500;	buf[0] = (div >> 8) & 0x7f;	buf[1] = div & 0xff;	buf[2] = byte3[tuner_type];	if (tuner_type == 1) {		buf[2] |= (div >> 10) & 0x60;		buf[3] = (freq < 174000000 ? 0x88 :			  freq < 470000000 ? 0x84 : 0x81);	} else {		buf[3] = (freq < 174000000 ? 0xa1 :			  freq < 454000000 ? 0x92 : 0x34);	}        return tuner_write (fe->i2c, addr[tuner_type], buf);}staticint ves1820_setup_reg0 (struct dvb_frontend *fe, u8 reg0){	reg0 |= GET_REG0(fe->data) & 0x62;		ves1820_writereg (fe, 0x00, reg0 & 0xfe);        ves1820_writereg (fe, 0x00, reg0 | 0x01);	/**	 *  check lock and toggle inversion bit if required...	 */	if (!(ves1820_readreg (fe, 0x11) & 0x08)) {		ddelay(1);		if (!(ves1820_readreg (fe, 0x11) & 0x08)) {			reg0 ^= 0x20;			ves1820_writereg (fe, 0x00, reg0 & 0xfe);        		ves1820_writereg (fe, 0x00, reg0 | 0x01);		}	}	SET_REG0(fe->data, reg0);	return 0;}staticint ves1820_init (struct dvb_frontend *fe){	int i;                dprintk("VES1820: init chip\n");        ves1820_writereg (fe, 0, 0);	for (i=0; i<53; i++)                ves1820_writereg (fe, i, ves1820_inittab[i]);	ves1820_writereg (fe, 0x34, GET_PWM(fe->data)); 	return 0;}staticint ves1820_set_symbolrate (struct dvb_frontend *fe, u32 symbolrate){        s32 BDR;         s32 BDRI;        s16 SFIL=0;        u16 NDEC = 0;        u32 tmp, ratio;#define XIN 57840000UL#define FIN (XIN >> 4)        if (symbolrate > XIN/2)                 symbolrate = XIN/2;	if (symbolrate < 500000)                symbolrate = 500000;        if (symbolrate < XIN/16) NDEC = 1;        if (symbolrate < XIN/32) NDEC = 2;        if (symbolrate < XIN/64) NDEC = 3;        if (symbolrate < (u32)(XIN/12.3)) SFIL = 1;        if (symbolrate < (u32)(XIN/16))	 SFIL = 0;        if (symbolrate < (u32)(XIN/24.6)) SFIL = 1;        if (symbolrate < (u32)(XIN/32))	 SFIL = 0;        if (symbolrate < (u32)(XIN/49.2)) SFIL = 1;        if (symbolrate < (u32)(XIN/64))	 SFIL = 0;        if (symbolrate < (u32)(XIN/98.4)) SFIL = 1;                symbolrate <<= NDEC;        ratio = (symbolrate << 4) / FIN;        tmp =  ((symbolrate << 4) % FIN) << 8;        ratio = (ratio << 8) + tmp / FIN;        tmp = (tmp % FIN) << 8;        ratio = (ratio << 8) + (tmp + FIN/2) / FIN;                BDR = ratio;        BDRI = (((XIN << 5) / symbolrate) + 1) / 2;                if (BDRI > 0xFF)                 BDRI = 0xFF;                SFIL = (SFIL << 4) | ves1820_inittab[0x0E];                NDEC = (NDEC << 6) | ves1820_inittab[0x03];        ves1820_writereg (fe, 0x03, NDEC);        ves1820_writereg (fe, 0x0a, BDR&0xff);        ves1820_writereg (fe, 0x0b, (BDR>> 8)&0xff);        ves1820_writereg (fe, 0x0c, (BDR>>16)&0x3f);        ves1820_writereg (fe, 0x0d, BDRI);        ves1820_writereg (fe, 0x0e, SFIL);        return 0;}staticint ves1820_set_parameters (struct dvb_frontend *fe,			    struct dvb_frontend_parameters *p){	static const u8 reg0x00 [] = { 0x00, 0x04, 0x08, 0x0c, 0x10 };	static const u8 reg0x01 [] = {  140,  140,  106,  120,   92 };	static const u8 reg0x05 [] = {  135,  100,   70,   54,   38 };	static const u8 reg0x08 [] = {  162,  116,   67,   52,   35 };	static const u8 reg0x09 [] = {  145,  150,  106,  126,  107 };	int real_qam = p->u.qam.modulation - QAM_16;	if (real_qam < 0 || real_qam > 4)		return -EINVAL;	tuner_set_tv_freq (fe, p->frequency);	ves1820_set_symbolrate (fe, p->u.qam.symbol_rate);	ves1820_writereg (fe, 0x34, GET_PWM(fe->data));        ves1820_writereg (fe, 0x01, reg0x01[real_qam]);        ves1820_writereg (fe, 0x05, reg0x05[real_qam]);        ves1820_writereg (fe, 0x08, reg0x08[real_qam]);        ves1820_writereg (fe, 0x09, reg0x09[real_qam]);	ves1820_setup_reg0 (fe, reg0x00[real_qam]);	return 0;}staticint ves1820_ioctl (struct dvb_frontend *fe, unsigned int cmd, void *arg){        switch (cmd) {	case FE_GET_INFO:		memcpy (arg, &ves1820_info, sizeof(struct dvb_frontend_info));		break;        case FE_READ_STATUS:	{		fe_status_t *status = (fe_status_t *) arg;		int sync;		*status = 0;                sync = ves1820_readreg (fe, 0x11);		if (sync & 2)			*status |= FE_HAS_SIGNAL;		if (sync & 2)			*status |= FE_HAS_CARRIER;		if (sync & 2)           /* XXX FIXME! */			*status |= FE_HAS_VITERBI;				if (sync & 4)			*status |= FE_HAS_SYNC;		if (sync & 8)			*status |= FE_HAS_LOCK;		break;	}	case FE_READ_BER:	{		u32 ber = ves1820_readreg(fe, 0x14) |			 (ves1820_readreg(fe, 0x15) << 8) |			 ((ves1820_readreg(fe, 0x16) & 0x0f) << 16);		*((u32*) arg) = 10 * ber;		break;	}	case FE_READ_SIGNAL_STRENGTH:	{		u8 gain = ves1820_readreg(fe, 0x17);		*((u16*) arg) = (gain << 8) | gain;		break;	}	case FE_READ_SNR:	{		u8 quality = ~ves1820_readreg(fe, 0x18);		*((u16*) arg) = (quality << 8) | quality;		break;	}	case FE_READ_UNCORRECTED_BLOCKS:		*((u32*) arg) = ves1820_readreg (fe, 0x13) & 0x7f;		if (*((u32*) arg) == 0x7f)			*((u32*) arg) = 0xffffffff;		/* reset uncorrected block counter */		ves1820_writereg (fe, 0x10, ves1820_inittab[0x10] & 0xdf);	        ves1820_writereg (fe, 0x10, ves1820_inittab[0x10]);		break;        case FE_SET_FRONTEND:		return ves1820_set_parameters (fe, arg);	case FE_GET_FRONTEND:		/*  XXX FIXME: implement! *//*                struct frontend *front = (struct frontend *)arg;                                front->afc=(int)((char)(readreg(client,0x19)));                front->afc=(front->afc*(int)(front->param.u.qam.SymbolRate/8))/128;*/		break;	case FE_SLEEP:		ves1820_writereg (fe, 0x1b, 0x02);  /* pdown ADC */		ves1820_writereg (fe, 0x00, 0x80);  /* standby */		break;        case FE_INIT:                return ves1820_init (fe);        default:                return -EINVAL;        }        return 0;} staticint probe_tuner (struct dvb_i2c_bus *i2c){	static const	struct i2c_msg msg1 = { addr: 0x61, flags: 0, buf: NULL, len: 0 };	static const	struct i2c_msg msg2 = { addr: 0x62, flags: 0, buf: NULL, len: 0 };	int type;	if (i2c->xfer(i2c, &msg1, 1) == 1) {		type = 0;		printk ("%s: setup for tuner spXXXX\n", __FILE__);	} else if (i2c->xfer(i2c, &msg2, 1) == 1) {		type = 1;		printk ("%s: setup for tuner sp5659c\n", __FILE__);	} else {		type = -1;		printk ("%s: unknown PLL, "			"please report to <linuxdvb@linuxtv.org>!!\n",			__FILE__);	}	return type;}staticu8 read_pwm (struct dvb_i2c_bus *i2c){	u8 b = 0xff;	u8 pwm;	struct i2c_msg msg [] = { { addr: 0x50, flags: 0, buf: &b, len: 1 },			 { addr: 0x50, flags: I2C_M_RD, buf: &pwm, len: 1 } };	i2c->xfer (i2c, msg, 2);	dprintk("VES1820: pwm=%02x\n", pwm);	if (pwm == 0xff)		pwm = 0x48;	return pwm;}staticint probe_demod_addr (struct dvb_i2c_bus *i2c){	u8 b [] = { 0x00, 0x1a };	u8 id;	struct i2c_msg msg [] = { { addr: 0x08, flags: 0, buf: b, len: 2 },	                   { addr: 0x08, flags: I2C_M_RD, buf: &id, len: 1 } };	if (i2c->xfer(i2c, msg, 2) == 2 && (id & 0xf0) == 0x70)		return msg[0].addr;	msg[0].addr = msg[1].addr = 0x09;	if (i2c->xfer(i2c, msg, 2) == 2 && (id & 0xf0) == 0x70)		return msg[0].addr;	return -1;}staticint ves1820_attach (struct dvb_i2c_bus *i2c){	void *data = NULL;	int demod_addr;	int tuner_type;	if ((demod_addr = probe_demod_addr(i2c)) < 0)		return -ENODEV;	if ((tuner_type = probe_tuner(i2c)) < 0)		return -ENODEV;	SET_PWM(data, read_pwm(i2c));	SET_REG0(data, ves1820_inittab[0]);	SET_TUNER(data, tuner_type);	SET_DEMOD_ADDR(data, demod_addr);	dvb_register_frontend (ves1820_ioctl, i2c, data, &ves1820_info);        return 0;}staticvoid ves1820_detach (struct dvb_i2c_bus *i2c){	dvb_unregister_frontend (ves1820_ioctl, i2c);}staticint __init init_ves1820 (void){	return dvb_register_i2c_device (THIS_MODULE,					ves1820_attach, ves1820_detach);}staticvoid __exit exit_ves1820 (void){	dvb_unregister_i2c_device (ves1820_attach);}module_init(init_ves1820);module_exit(exit_ves1820);MODULE_DESCRIPTION("VES1820 DVB-C frontend driver");MODULE_AUTHOR("Ralph Metzler, Holger Waechtler");MODULE_LICENSE("GPL");

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -