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

📄 mxl5007t.c

📁 trident tm5600的linux驱动
💻 C
📖 第 1 页 / 共 2 页
字号:
	unsigned int i = 0;	int ret = 0;	while ((ret == 0) && (reg_pair[i].reg || reg_pair[i].val)) {		ret = mxl5007t_write_reg(state,					 reg_pair[i].reg, reg_pair[i].val);		i++;	}	return ret;}static int mxl5007t_read_reg(struct mxl5007t_state *state, u8 reg, u8 *val){	struct i2c_msg msg[] = {		{ .addr = state->i2c_props.addr, .flags = 0,		  .buf = &reg, .len = 1 },		{ .addr = state->i2c_props.addr, .flags = I2C_M_RD,		  .buf = val, .len = 1 },	};	int ret;	ret = i2c_transfer(state->i2c_props.adap, msg, 2);	if (ret != 2) {		mxl_err("failed!");		return -EREMOTEIO;	}	return 0;}static int mxl5007t_soft_reset(struct mxl5007t_state *state){	u8 d = 0xff;	struct i2c_msg msg = { .addr = state->i2c_props.addr, .flags = 0,			       .buf = &d, .len = 1 };	int ret = i2c_transfer(state->i2c_props.adap, &msg, 1);	if (ret != 1) {		mxl_err("failed!");		return -EREMOTEIO;	}	return 0;}static int mxl5007t_tuner_init(struct mxl5007t_state *state,			       enum mxl5007t_mode mode){	struct reg_pair_t *init_regs;	int ret;	ret = mxl5007t_soft_reset(state);	if (mxl_fail(ret))		goto fail;	/* calculate initialization reg array */	init_regs = mxl5007t_calc_init_regs(state, mode);	ret = mxl5007t_write_regs(state, init_regs);	if (mxl_fail(ret))		goto fail;	mdelay(1);	ret = mxl5007t_write_reg(state, 0x2c, 0x35);	mxl_fail(ret);fail:	return ret;}static int mxl5007t_tuner_rf_tune(struct mxl5007t_state *state, u32 rf_freq_hz,				  enum mxl5007t_bw_mhz bw){	struct reg_pair_t *rf_tune_regs;	int ret;	/* calculate channel change reg array */	rf_tune_regs = mxl5007t_calc_rf_tune_regs(state, rf_freq_hz, bw);	ret = mxl5007t_write_regs(state, rf_tune_regs);	if (mxl_fail(ret))		goto fail;	msleep(3);fail:	return ret;}/* ------------------------------------------------------------------------- */static int mxl5007t_synth_lock_status(struct mxl5007t_state *state,				      int *rf_locked, int *ref_locked){	u8 d;	int ret;	*rf_locked = 0;	*ref_locked = 0;	ret = mxl5007t_read_reg(state, 0xcf, &d);	if (mxl_fail(ret))		goto fail;	if ((d & 0x0c) == 0x0c)		*rf_locked = 1;	if ((d & 0x03) == 0x03)		*ref_locked = 1;fail:	return ret;}static int mxl5007t_check_rf_input_power(struct mxl5007t_state *state,					 s32 *rf_input_level){	u8 d1, d2;	int ret;	ret = mxl5007t_read_reg(state, 0xb7, &d1);	if (mxl_fail(ret))		goto fail;	ret = mxl5007t_read_reg(state, 0xbf, &d2);	if (mxl_fail(ret))		goto fail;	d2 = d2 >> 4;	if (d2 > 7)		d2 += 0xf0;	*rf_input_level = (s32)(d1 + d2 - 113);fail:	return ret;}/* ------------------------------------------------------------------------- */static int mxl5007t_get_status(struct dvb_frontend *fe, u32 *status){	struct mxl5007t_state *state = fe->tuner_priv;	int rf_locked, ref_locked;	s32 rf_input_level;	int ret;	if (fe->ops.i2c_gate_ctrl)		fe->ops.i2c_gate_ctrl(fe, 1);	ret = mxl5007t_synth_lock_status(state, &rf_locked, &ref_locked);	if (mxl_fail(ret))		goto fail;	mxl_debug("%s%s", rf_locked ? "rf locked " : "",		  ref_locked ? "ref locked" : "");	ret = mxl5007t_check_rf_input_power(state, &rf_input_level);	if (mxl_fail(ret))		goto fail;	mxl_debug("rf input power: %d", rf_input_level);fail:	if (fe->ops.i2c_gate_ctrl)		fe->ops.i2c_gate_ctrl(fe, 0);	return ret;}/* ------------------------------------------------------------------------- */static int mxl5007t_set_params(struct dvb_frontend *fe,			       struct dvb_frontend_parameters *params){	struct mxl5007t_state *state = fe->tuner_priv;	enum mxl5007t_bw_mhz bw;	enum mxl5007t_mode mode;	int ret;	u32 freq = params->frequency;	if (fe->ops.info.type == FE_ATSC) {		switch (params->u.vsb.modulation) {		case VSB_8:		case VSB_16:			mode = MxL_MODE_OTA_DVBT_ATSC;			break;		case QAM_64:		case QAM_256:			mode = MxL_MODE_CABLE_DIGITAL;			break;		default:			mxl_err("modulation not set!");			return -EINVAL;		}		bw = MxL_BW_6MHz;	} else if (fe->ops.info.type == FE_OFDM) {		switch (params->u.ofdm.bandwidth) {		case BANDWIDTH_6_MHZ:			bw = MxL_BW_6MHz;			break;		case BANDWIDTH_7_MHZ:			bw = MxL_BW_7MHz;			break;		case BANDWIDTH_8_MHZ:			bw = MxL_BW_8MHz;			break;		default:			mxl_err("bandwidth not set!");			return -EINVAL;		}		mode = MxL_MODE_OTA_DVBT_ATSC;	} else {		mxl_err("modulation type not supported!");		return -EINVAL;	}	if (fe->ops.i2c_gate_ctrl)		fe->ops.i2c_gate_ctrl(fe, 1);	mutex_lock(&state->lock);	ret = mxl5007t_tuner_init(state, mode);	if (mxl_fail(ret))		goto fail;	ret = mxl5007t_tuner_rf_tune(state, freq, bw);	if (mxl_fail(ret))		goto fail;	state->frequency = freq;	state->bandwidth = (fe->ops.info.type == FE_OFDM) ?		params->u.ofdm.bandwidth : 0;fail:	mutex_unlock(&state->lock);	if (fe->ops.i2c_gate_ctrl)		fe->ops.i2c_gate_ctrl(fe, 0);	return ret;}static int mxl5007t_set_analog_params(struct dvb_frontend *fe,				      struct analog_parameters *params){	struct mxl5007t_state *state = fe->tuner_priv;	enum mxl5007t_bw_mhz bw = 0; /* FIXME */	enum mxl5007t_mode cbl_mode;	enum mxl5007t_mode ota_mode;	char *mode_name;	int ret;	u32 freq = params->frequency * 62500;#if 0	if (params->mode == V4L2_TUNER_RADIO) {		freq = freq / 1000;		mode_name = "fm";	} else#else#define cable 1#endif	if (params->std & V4L2_STD_MN) {		cbl_mode = MxL_MODE_CABLE_NTSC_PAL_GH;		ota_mode = MxL_MODE_OTA_NTSC_PAL_GH;		mode_name = "MN";	} else if (params->std & V4L2_STD_B) {		cbl_mode = MxL_MODE_CABLE_PAL_IB;		ota_mode = MxL_MODE_OTA_PAL_IB;		mode_name = "B";	} else if (params->std & V4L2_STD_GH) {		cbl_mode = MxL_MODE_CABLE_NTSC_PAL_GH;		ota_mode = MxL_MODE_OTA_NTSC_PAL_GH;		mode_name = "GH";	} else if (params->std & V4L2_STD_PAL_I) {		cbl_mode = MxL_MODE_CABLE_PAL_IB;		ota_mode = MxL_MODE_OTA_PAL_IB;		mode_name = "I";	} else if (params->std & V4L2_STD_DK) {		cbl_mode = MxL_MODE_CABLE_PAL_D_SECAM_KL;		ota_mode = MxL_MODE_OTA_PAL_D_SECAM_KL;		mode_name = "DK";	} else if (params->std & V4L2_STD_SECAM_L) {		cbl_mode = MxL_MODE_CABLE_PAL_D_SECAM_KL;		ota_mode = MxL_MODE_OTA_PAL_D_SECAM_KL;		mode_name = "L";	} else if (params->std & V4L2_STD_SECAM_LC) {		cbl_mode = MxL_MODE_CABLE_PAL_D_SECAM_KL;		ota_mode = MxL_MODE_OTA_PAL_D_SECAM_KL;		mode_name = "L'";	} else {		mode_name = "xx";		/* FIXME */		cbl_mode = MxL_MODE_CABLE_NTSC_PAL_GH;		ota_mode = MxL_MODE_OTA_NTSC_PAL_GH;	}	mxl_debug("setting mxl5007 to system %s", mode_name);	if (fe->ops.i2c_gate_ctrl)		fe->ops.i2c_gate_ctrl(fe, 1);	mutex_lock(&state->lock);	ret = mxl5007t_tuner_init(state, cable ? cbl_mode : ota_mode);	if (mxl_fail(ret))		goto fail;	ret = mxl5007t_tuner_rf_tune(state, freq, bw);	if (mxl_fail(ret))		goto fail;	state->frequency = freq;	state->bandwidth = 0;fail:	mutex_unlock(&state->lock);	if (fe->ops.i2c_gate_ctrl)		fe->ops.i2c_gate_ctrl(fe, 0);	return ret;}/* ------------------------------------------------------------------------- */static int mxl5007t_init(struct dvb_frontend *fe){	struct mxl5007t_state *state = fe->tuner_priv;	int ret;	u8 d;	if (fe->ops.i2c_gate_ctrl)		fe->ops.i2c_gate_ctrl(fe, 1);	ret = mxl5007t_read_reg(state, 0x05, &d);	if (mxl_fail(ret))		goto fail;	ret = mxl5007t_write_reg(state, 0x05, d | 0x01);	mxl_fail(ret);fail:	if (fe->ops.i2c_gate_ctrl)		fe->ops.i2c_gate_ctrl(fe, 0);	return ret;}static int mxl5007t_sleep(struct dvb_frontend *fe){	struct mxl5007t_state *state = fe->tuner_priv;	int ret;	u8 d;	if (fe->ops.i2c_gate_ctrl)		fe->ops.i2c_gate_ctrl(fe, 1);	ret = mxl5007t_read_reg(state, 0x05, &d);	if (mxl_fail(ret))		goto fail;	ret = mxl5007t_write_reg(state, 0x05, d & ~0x01);	mxl_fail(ret);fail:	if (fe->ops.i2c_gate_ctrl)		fe->ops.i2c_gate_ctrl(fe, 0);	return ret;}/* ------------------------------------------------------------------------- */static int mxl5007t_get_frequency(struct dvb_frontend *fe, u32 *frequency){	struct mxl5007t_state *state = fe->tuner_priv;	*frequency = state->frequency;	return 0;}static int mxl5007t_get_bandwidth(struct dvb_frontend *fe, u32 *bandwidth){	struct mxl5007t_state *state = fe->tuner_priv;	*bandwidth = state->bandwidth;	return 0;}static int mxl5007t_release(struct dvb_frontend *fe){	struct mxl5007t_state *state = fe->tuner_priv;	mutex_lock(&mxl5007t_list_mutex);	if (state)		hybrid_tuner_release_state(state);	mutex_unlock(&mxl5007t_list_mutex);	fe->tuner_priv = NULL;	return 0;}/* ------------------------------------------------------------------------- */static struct dvb_tuner_ops mxl5007t_tuner_ops = {	.info = {		.name = "MaxLinear MxL5007T",#if 0		.frequency_min  = ,		.frequency_max  = ,		.frequency_step = ,#endif	},	.init              = mxl5007t_init,	.sleep             = mxl5007t_sleep,	.set_params        = mxl5007t_set_params,	.set_analog_params = mxl5007t_set_analog_params,	.get_status        = mxl5007t_get_status,	.get_frequency     = mxl5007t_get_frequency,	.get_bandwidth     = mxl5007t_get_bandwidth,	.release           = mxl5007t_release,};static int mxl5007t_get_chip_id(struct mxl5007t_state *state){	char *name;	int ret;	u8 id;	ret = mxl5007t_read_reg(state, 0xd3, &id);	if (mxl_fail(ret))		goto fail;	switch (id) {	case MxL_5007_V1_F1:		name = "MxL5007.v1.f1";		break;	case MxL_5007_V1_F2:		name = "MxL5007.v1.f2";		break;	case MxL_5007_V2_100_F1:		name = "MxL5007.v2.100.f1";		break;	case MxL_5007_V2_100_F2:		name = "MxL5007.v2.100.f2";		break;	case MxL_5007_V2_200_F1:		name = "MxL5007.v2.200.f1";		break;	case MxL_5007_V2_200_F2:		name = "MxL5007.v2.200.f2";		break;	default:#if 0		ret = -EINVAL;		goto fail;#else		name = "MxL5007T";		id = MxL_UNKNOWN_ID;#endif	}	state->chip_id = id;	mxl_info("%s detected @ %d-%04x", name,		 i2c_adapter_id(state->i2c_props.adap),		 state->i2c_props.addr);	return 0;fail:	mxl_warn("unable to identify device @ %d-%04x",		 i2c_adapter_id(state->i2c_props.adap),		 state->i2c_props.addr);	state->chip_id = MxL_UNKNOWN_ID;	return ret;}struct dvb_frontend *mxl5007t_attach(struct dvb_frontend *fe,				     struct i2c_adapter *i2c, u8 addr,				     struct mxl5007t_config *cfg){	struct mxl5007t_state *state = NULL;	int instance, ret;	mutex_lock(&mxl5007t_list_mutex);	instance = hybrid_tuner_request_state(struct mxl5007t_state, state,					      hybrid_tuner_instance_list,					      i2c, addr, "mxl5007");	switch (instance) {	case 0:		goto fail;	case 1:		/* new tuner instance */		state->config = cfg;		mutex_init(&state->lock);		if (fe->ops.i2c_gate_ctrl)			fe->ops.i2c_gate_ctrl(fe, 1);		ret = mxl5007t_get_chip_id(state);		if (fe->ops.i2c_gate_ctrl)			fe->ops.i2c_gate_ctrl(fe, 0);		/* check return value of mxl5007t_get_chip_id */		if (mxl_fail(ret))			goto fail;		break;	default:		/* existing tuner instance */		break;	}	fe->tuner_priv = state;	mutex_unlock(&mxl5007t_list_mutex);	memcpy(&fe->ops.tuner_ops, &mxl5007t_tuner_ops,	       sizeof(struct dvb_tuner_ops));	return fe;fail:	mutex_unlock(&mxl5007t_list_mutex);	mxl5007t_release(fe);	return NULL;}EXPORT_SYMBOL_GPL(mxl5007t_attach);MODULE_DESCRIPTION("MaxLinear MxL5007T Silicon IC tuner driver");MODULE_AUTHOR("Michael Krufky <mkrufky@linuxtv.org>");MODULE_LICENSE("GPL");MODULE_VERSION("0.1");/* * Overrides for Emacs so that we follow Linus's tabbing style. * --------------------------------------------------------------------------- * Local variables: * c-basic-offset: 8 * End: */

⌨️ 快捷键说明

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