dib3000mc.c

来自「trident tm5600的linux驱动」· C语言 代码 · 共 935 行 · 第 1/2 页

C
935
字号
/* * Driver for DiBcom DiB3000MC/P-demodulator. * * Copyright (C) 2004-7 DiBcom (http://www.dibcom.fr/) * Copyright (C) 2004-5 Patrick Boettcher (patrick.boettcher@desy.de) * * This code is partially based on the previous dib3000mc.c . * * 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, version 2. */#include <linux/kernel.h>#include <linux/i2c.h>#include "compat.h"#include "dvb_frontend.h"#include "dib3000mc.h"static int debug;module_param(debug, int, 0644);MODULE_PARM_DESC(debug, "turn on debugging (default: 0)");static int buggy_sfn_workaround;module_param(buggy_sfn_workaround, int, 0644);MODULE_PARM_DESC(buggy_sfn_workaround, "Enable work-around for buggy SFNs (default: 0)");#define dprintk(args...) do { if (debug) { printk(KERN_DEBUG "DiB3000MC/P:"); printk(args); printk("\n"); } } while (0)struct dib3000mc_state {	struct dvb_frontend demod;	struct dib3000mc_config *cfg;	u8 i2c_addr;	struct i2c_adapter *i2c_adap;	struct dibx000_i2c_master i2c_master;	u32 timf;	fe_bandwidth_t current_bandwidth;	u16 dev_id;	u8 sfn_workaround_active :1;};static u16 dib3000mc_read_word(struct dib3000mc_state *state, u16 reg){	u8 wb[2] = { (reg >> 8) | 0x80, reg & 0xff };	u8 rb[2];	struct i2c_msg msg[2] = {		{ .addr = state->i2c_addr >> 1, .flags = 0,        .buf = wb, .len = 2 },		{ .addr = state->i2c_addr >> 1, .flags = I2C_M_RD, .buf = rb, .len = 2 },	};	if (i2c_transfer(state->i2c_adap, msg, 2) != 2)		dprintk("i2c read error on %d\n",reg);	return (rb[0] << 8) | rb[1];}static int dib3000mc_write_word(struct dib3000mc_state *state, u16 reg, u16 val){	u8 b[4] = {		(reg >> 8) & 0xff, reg & 0xff,		(val >> 8) & 0xff, val & 0xff,	};	struct i2c_msg msg = {		.addr = state->i2c_addr >> 1, .flags = 0, .buf = b, .len = 4	};	return i2c_transfer(state->i2c_adap, &msg, 1) != 1 ? -EREMOTEIO : 0;}static int dib3000mc_identify(struct dib3000mc_state *state){	u16 value;	if ((value = dib3000mc_read_word(state, 1025)) != 0x01b3) {		dprintk("-E-  DiB3000MC/P: wrong Vendor ID (read=0x%x)\n",value);		return -EREMOTEIO;	}	value = dib3000mc_read_word(state, 1026);	if (value != 0x3001 && value != 0x3002) {		dprintk("-E-  DiB3000MC/P: wrong Device ID (%x)\n",value);		return -EREMOTEIO;	}	state->dev_id = value;	dprintk("-I-  found DiB3000MC/P: %x\n",state->dev_id);	return 0;}static int dib3000mc_set_timing(struct dib3000mc_state *state, s16 nfft, u32 bw, u8 update_offset){	u32 timf;	if (state->timf == 0) {		timf = 1384402; // default value for 8MHz		if (update_offset)			msleep(200); // first time we do an update	} else		timf = state->timf;	timf *= (bw / 1000);	if (update_offset) {		s16 tim_offs = dib3000mc_read_word(state, 416);		if (tim_offs &  0x2000)			tim_offs -= 0x4000;		if (nfft == TRANSMISSION_MODE_2K)			tim_offs *= 4;		timf += tim_offs;		state->timf = timf / (bw / 1000);	}	dprintk("timf: %d\n", timf);	dib3000mc_write_word(state, 23, (u16) (timf >> 16));	dib3000mc_write_word(state, 24, (u16) (timf      ) & 0xffff);	return 0;}static int dib3000mc_setup_pwm_state(struct dib3000mc_state *state){	u16 reg_51, reg_52 = state->cfg->agc->setup & 0xfefb;    if (state->cfg->pwm3_inversion) {		reg_51 =  (2 << 14) | (0 << 10) | (7 << 6) | (2 << 2) | (2 << 0);		reg_52 |= (1 << 2);	} else {		reg_51 = (2 << 14) | (4 << 10) | (7 << 6) | (2 << 2) | (2 << 0);		reg_52 |= (1 << 8);	}	dib3000mc_write_word(state, 51, reg_51);	dib3000mc_write_word(state, 52, reg_52);    if (state->cfg->use_pwm3)		dib3000mc_write_word(state, 245, (1 << 3) | (1 << 0));	else		dib3000mc_write_word(state, 245, 0);    dib3000mc_write_word(state, 1040, 0x3);	return 0;}static int dib3000mc_set_output_mode(struct dib3000mc_state *state, int mode){	int    ret = 0;	u16 fifo_threshold = 1792;	u16 outreg = 0;	u16 outmode = 0;	u16 elecout = 1;	u16 smo_reg = dib3000mc_read_word(state, 206) & 0x0010; /* keep the pid_parse bit */	dprintk("-I-  Setting output mode for demod %p to %d\n",			&state->demod, mode);	switch (mode) {		case OUTMODE_HIGH_Z:  // disable			elecout = 0;			break;		case OUTMODE_MPEG2_PAR_GATED_CLK:   // STBs with parallel gated clock			outmode = 0;			break;		case OUTMODE_MPEG2_PAR_CONT_CLK:    // STBs with parallel continues clock			outmode = 1;			break;		case OUTMODE_MPEG2_SERIAL:          // STBs with serial input			outmode = 2;			break;		case OUTMODE_MPEG2_FIFO:            // e.g. USB feeding			elecout = 3;			/*ADDR @ 206 :			P_smo_error_discard  [1;6:6] = 0			P_smo_rs_discard     [1;5:5] = 0			P_smo_pid_parse      [1;4:4] = 0			P_smo_fifo_flush     [1;3:3] = 0			P_smo_mode           [2;2:1] = 11			P_smo_ovf_prot       [1;0:0] = 0			*/			smo_reg |= 3 << 1;			fifo_threshold = 512;			outmode = 5;			break;		case OUTMODE_DIVERSITY:			outmode = 4;			elecout = 1;			break;		default:			dprintk("Unhandled output_mode passed to be set for demod %p\n",&state->demod);			outmode = 0;			break;	}	if ((state->cfg->output_mpeg2_in_188_bytes))		smo_reg |= (1 << 5); // P_smo_rs_discard     [1;5:5] = 1	outreg = dib3000mc_read_word(state, 244) & 0x07FF;	outreg |= (outmode << 11);	ret |= dib3000mc_write_word(state,  244, outreg);	ret |= dib3000mc_write_word(state,  206, smo_reg);   /*smo_ mode*/	ret |= dib3000mc_write_word(state,  207, fifo_threshold); /* synchronous fread */	ret |= dib3000mc_write_word(state, 1040, elecout);         /* P_out_cfg */	return ret;}static int dib3000mc_set_bandwidth(struct dib3000mc_state *state, u32 bw){	u16 bw_cfg[6] = { 0 };	u16 imp_bw_cfg[3] = { 0 };	u16 reg;/* settings here are for 27.7MHz */	switch (bw) {		case 8000:			bw_cfg[0] = 0x0019; bw_cfg[1] = 0x5c30; bw_cfg[2] = 0x0054; bw_cfg[3] = 0x88a0; bw_cfg[4] = 0x01a6; bw_cfg[5] = 0xab20;			imp_bw_cfg[0] = 0x04db; imp_bw_cfg[1] = 0x00db; imp_bw_cfg[2] = 0x00b7;			break;		case 7000:			bw_cfg[0] = 0x001c; bw_cfg[1] = 0xfba5; bw_cfg[2] = 0x0060; bw_cfg[3] = 0x9c25; bw_cfg[4] = 0x01e3; bw_cfg[5] = 0x0cb7;			imp_bw_cfg[0] = 0x04c0; imp_bw_cfg[1] = 0x00c0; imp_bw_cfg[2] = 0x00a0;			break;		case 6000:			bw_cfg[0] = 0x0021; bw_cfg[1] = 0xd040; bw_cfg[2] = 0x0070; bw_cfg[3] = 0xb62b; bw_cfg[4] = 0x0233; bw_cfg[5] = 0x8ed5;			imp_bw_cfg[0] = 0x04a5; imp_bw_cfg[1] = 0x00a5; imp_bw_cfg[2] = 0x0089;			break;		case 5000:			bw_cfg[0] = 0x0028; bw_cfg[1] = 0x9380; bw_cfg[2] = 0x0087; bw_cfg[3] = 0x4100; bw_cfg[4] = 0x02a4; bw_cfg[5] = 0x4500;			imp_bw_cfg[0] = 0x0489; imp_bw_cfg[1] = 0x0089; imp_bw_cfg[2] = 0x0072;			break;		default: return -EINVAL;	}	for (reg = 6; reg < 12; reg++)		dib3000mc_write_word(state, reg, bw_cfg[reg - 6]);	dib3000mc_write_word(state, 12, 0x0000);	dib3000mc_write_word(state, 13, 0x03e8);	dib3000mc_write_word(state, 14, 0x0000);	dib3000mc_write_word(state, 15, 0x03f2);	dib3000mc_write_word(state, 16, 0x0001);	dib3000mc_write_word(state, 17, 0xb0d0);	// P_sec_len	dib3000mc_write_word(state, 18, 0x0393);	dib3000mc_write_word(state, 19, 0x8700);	for (reg = 55; reg < 58; reg++)		dib3000mc_write_word(state, reg, imp_bw_cfg[reg - 55]);	// Timing configuration	dib3000mc_set_timing(state, TRANSMISSION_MODE_2K, bw, 0);	return 0;}static u16 impulse_noise_val[29] ={	0x38, 0x6d9, 0x3f28, 0x7a7, 0x3a74, 0x196, 0x32a, 0x48c, 0x3ffe, 0x7f3,	0x2d94, 0x76, 0x53d, 0x3ff8, 0x7e3, 0x3320, 0x76, 0x5b3, 0x3feb, 0x7d2,	0x365e, 0x76, 0x48c, 0x3ffe, 0x5b3, 0x3feb, 0x76, 0x0000, 0xd};static void dib3000mc_set_impulse_noise(struct dib3000mc_state *state, u8 mode, s16 nfft){	u16 i;	for (i = 58; i < 87; i++)		dib3000mc_write_word(state, i, impulse_noise_val[i-58]);	if (nfft == TRANSMISSION_MODE_8K) {		dib3000mc_write_word(state, 58, 0x3b);		dib3000mc_write_word(state, 84, 0x00);		dib3000mc_write_word(state, 85, 0x8200);	}	dib3000mc_write_word(state, 34, 0x1294);	dib3000mc_write_word(state, 35, 0x1ff8);	if (mode == 1)		dib3000mc_write_word(state, 55, dib3000mc_read_word(state, 55) | (1 << 10));}static int dib3000mc_init(struct dvb_frontend *demod){	struct dib3000mc_state *state = demod->demodulator_priv;	struct dibx000_agc_config *agc = state->cfg->agc;	// Restart Configuration	dib3000mc_write_word(state, 1027, 0x8000);	dib3000mc_write_word(state, 1027, 0x0000);	// power up the demod + mobility configuration	dib3000mc_write_word(state, 140, 0x0000);	dib3000mc_write_word(state, 1031, 0);	if (state->cfg->mobile_mode) {		dib3000mc_write_word(state, 139,  0x0000);		dib3000mc_write_word(state, 141,  0x0000);		dib3000mc_write_word(state, 175,  0x0002);		dib3000mc_write_word(state, 1032, 0x0000);	} else {		dib3000mc_write_word(state, 139,  0x0001);		dib3000mc_write_word(state, 141,  0x0000);		dib3000mc_write_word(state, 175,  0x0000);		dib3000mc_write_word(state, 1032, 0x012C);	}	dib3000mc_write_word(state, 1033, 0x0000);	// P_clk_cfg	dib3000mc_write_word(state, 1037, 0x3130);	// other configurations	// P_ctrl_sfreq	dib3000mc_write_word(state, 33, (5 << 0));	dib3000mc_write_word(state, 88, (1 << 10) | (0x10 << 0));	// Phase noise control	// P_fft_phacor_inh, P_fft_phacor_cpe, P_fft_powrange	dib3000mc_write_word(state, 99, (1 << 9) | (0x20 << 0));	if (state->cfg->phase_noise_mode == 0)		dib3000mc_write_word(state, 111, 0x00);	else		dib3000mc_write_word(state, 111, 0x02);	// P_agc_global	dib3000mc_write_word(state, 50, 0x8000);	// agc setup misc	dib3000mc_setup_pwm_state(state);	// P_agc_counter_lock	dib3000mc_write_word(state, 53, 0x87);	// P_agc_counter_unlock	dib3000mc_write_word(state, 54, 0x87);	/* agc */	dib3000mc_write_word(state, 36, state->cfg->max_time);	dib3000mc_write_word(state, 37, (state->cfg->agc_command1 << 13) | (state->cfg->agc_command2 << 12) | (0x1d << 0));	dib3000mc_write_word(state, 38, state->cfg->pwm3_value);	dib3000mc_write_word(state, 39, state->cfg->ln_adc_level);	// set_agc_loop_Bw	dib3000mc_write_word(state, 40, 0x0179);	dib3000mc_write_word(state, 41, 0x03f0);	dib3000mc_write_word(state, 42, agc->agc1_max);	dib3000mc_write_word(state, 43, agc->agc1_min);	dib3000mc_write_word(state, 44, agc->agc2_max);	dib3000mc_write_word(state, 45, agc->agc2_min);	dib3000mc_write_word(state, 46, (agc->agc1_pt1 << 8) | agc->agc1_pt2);	dib3000mc_write_word(state, 47, (agc->agc1_slope1 << 8) | agc->agc1_slope2);	dib3000mc_write_word(state, 48, (agc->agc2_pt1 << 8) | agc->agc2_pt2);	dib3000mc_write_word(state, 49, (agc->agc2_slope1 << 8) | agc->agc2_slope2);// Begin: TimeOut registers	// P_pha3_thres	dib3000mc_write_word(state, 110, 3277);	// P_timf_alpha = 6, P_corm_alpha = 6, P_corm_thres = 0x80	dib3000mc_write_word(state,  26, 0x6680);	// lock_mask0	dib3000mc_write_word(state, 1, 4);	// lock_mask1	dib3000mc_write_word(state, 2, 4);	// lock_mask2	dib3000mc_write_word(state, 3, 0x1000);	// P_search_maxtrial=1	dib3000mc_write_word(state, 5, 1);	dib3000mc_set_bandwidth(state, 8000);	// div_lock_mask	dib3000mc_write_word(state,  4, 0x814);	dib3000mc_write_word(state, 21, (1 << 9) | 0x164);	dib3000mc_write_word(state, 22, 0x463d);	// Spurious rm cfg	// P_cspu_regul, P_cspu_win_cut	dib3000mc_write_word(state, 120, 0x200f);	// P_adp_selec_monit	dib3000mc_write_word(state, 134, 0);	// Fec cfg	dib3000mc_write_word(state, 195, 0x10);	// diversity register: P_dvsy_sync_wait..	dib3000mc_write_word(state, 180, 0x2FF0);	// Impulse noise configuration	dib3000mc_set_impulse_noise(state, 0, TRANSMISSION_MODE_8K);	// output mode set-up	dib3000mc_set_output_mode(state, OUTMODE_HIGH_Z);	/* close the i2c-gate */	dib3000mc_write_word(state, 769, (1 << 7) );	return 0;}static int dib3000mc_sleep(struct dvb_frontend *demod){	struct dib3000mc_state *state = demod->demodulator_priv;	dib3000mc_write_word(state, 1031, 0xFFFF);	dib3000mc_write_word(state, 1032, 0xFFFF);	dib3000mc_write_word(state, 1033, 0xFFF0);    return 0;}static void dib3000mc_set_adp_cfg(struct dib3000mc_state *state, s16 qam){	u16 cfg[4] = { 0 },reg;	switch (qam) {		case QPSK:			cfg[0] = 0x099a; cfg[1] = 0x7fae; cfg[2] = 0x0333; cfg[3] = 0x7ff0;			break;		case QAM_16:			cfg[0] = 0x023d; cfg[1] = 0x7fdf; cfg[2] = 0x00a4; cfg[3] = 0x7ff0;			break;		case QAM_64:			cfg[0] = 0x0148; cfg[1] = 0x7ff0; cfg[2] = 0x00a4; cfg[3] = 0x7ff8;			break;	}	for (reg = 129; reg < 133; reg++)		dib3000mc_write_word(state, reg, cfg[reg - 129]);}static void dib3000mc_set_channel_cfg(struct dib3000mc_state *state, struct dvb_frontend_parameters *ch, u16 seq){	u16 value;    dib3000mc_set_bandwidth(state, BANDWIDTH_TO_KHZ(ch->u.ofdm.bandwidth));	dib3000mc_set_timing(state, ch->u.ofdm.transmission_mode, BANDWIDTH_TO_KHZ(ch->u.ofdm.bandwidth), 0);//	if (boost)//		dib3000mc_write_word(state, 100, (11 << 6) + 6);//	else		dib3000mc_write_word(state, 100, (16 << 6) + 9);	dib3000mc_write_word(state, 1027, 0x0800);	dib3000mc_write_word(state, 1027, 0x0000);	//Default cfg isi offset adp	dib3000mc_write_word(state, 26,  0x6680);	dib3000mc_write_word(state, 29,  0x1273);	dib3000mc_write_word(state, 33,       5);	dib3000mc_set_adp_cfg(state, QAM_16);	dib3000mc_write_word(state, 133,  15564);	dib3000mc_write_word(state, 12 , 0x0);	dib3000mc_write_word(state, 13 , 0x3e8);	dib3000mc_write_word(state, 14 , 0x0);	dib3000mc_write_word(state, 15 , 0x3f2);	dib3000mc_write_word(state, 93,0);	dib3000mc_write_word(state, 94,0);

⌨️ 快捷键说明

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