📄 mxl5007t.c
字号:
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 = ®, .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 + -