18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later 28c2ecf20Sopenharmony_ci/* 38c2ecf20Sopenharmony_ci * mxl111sf-tuner.c - driver for the MaxLinear MXL111SF CMOS tuner 48c2ecf20Sopenharmony_ci * 58c2ecf20Sopenharmony_ci * Copyright (C) 2010-2014 Michael Krufky <mkrufky@linuxtv.org> 68c2ecf20Sopenharmony_ci */ 78c2ecf20Sopenharmony_ci 88c2ecf20Sopenharmony_ci#include "mxl111sf-tuner.h" 98c2ecf20Sopenharmony_ci#include "mxl111sf-phy.h" 108c2ecf20Sopenharmony_ci#include "mxl111sf-reg.h" 118c2ecf20Sopenharmony_ci 128c2ecf20Sopenharmony_ci/* debug */ 138c2ecf20Sopenharmony_cistatic int mxl111sf_tuner_debug; 148c2ecf20Sopenharmony_cimodule_param_named(debug, mxl111sf_tuner_debug, int, 0644); 158c2ecf20Sopenharmony_ciMODULE_PARM_DESC(debug, "set debugging level (1=info (or-able))."); 168c2ecf20Sopenharmony_ci 178c2ecf20Sopenharmony_ci#define mxl_dbg(fmt, arg...) \ 188c2ecf20Sopenharmony_ci if (mxl111sf_tuner_debug) \ 198c2ecf20Sopenharmony_ci mxl_printk(KERN_DEBUG, fmt, ##arg) 208c2ecf20Sopenharmony_ci 218c2ecf20Sopenharmony_ci/* ------------------------------------------------------------------------ */ 228c2ecf20Sopenharmony_ci 238c2ecf20Sopenharmony_cistruct mxl111sf_tuner_state { 248c2ecf20Sopenharmony_ci struct mxl111sf_state *mxl_state; 258c2ecf20Sopenharmony_ci 268c2ecf20Sopenharmony_ci const struct mxl111sf_tuner_config *cfg; 278c2ecf20Sopenharmony_ci 288c2ecf20Sopenharmony_ci enum mxl_if_freq if_freq; 298c2ecf20Sopenharmony_ci 308c2ecf20Sopenharmony_ci u32 frequency; 318c2ecf20Sopenharmony_ci u32 bandwidth; 328c2ecf20Sopenharmony_ci}; 338c2ecf20Sopenharmony_ci 348c2ecf20Sopenharmony_cistatic int mxl111sf_tuner_read_reg(struct mxl111sf_tuner_state *state, 358c2ecf20Sopenharmony_ci u8 addr, u8 *data) 368c2ecf20Sopenharmony_ci{ 378c2ecf20Sopenharmony_ci return (state->cfg->read_reg) ? 388c2ecf20Sopenharmony_ci state->cfg->read_reg(state->mxl_state, addr, data) : 398c2ecf20Sopenharmony_ci -EINVAL; 408c2ecf20Sopenharmony_ci} 418c2ecf20Sopenharmony_ci 428c2ecf20Sopenharmony_cistatic int mxl111sf_tuner_write_reg(struct mxl111sf_tuner_state *state, 438c2ecf20Sopenharmony_ci u8 addr, u8 data) 448c2ecf20Sopenharmony_ci{ 458c2ecf20Sopenharmony_ci return (state->cfg->write_reg) ? 468c2ecf20Sopenharmony_ci state->cfg->write_reg(state->mxl_state, addr, data) : 478c2ecf20Sopenharmony_ci -EINVAL; 488c2ecf20Sopenharmony_ci} 498c2ecf20Sopenharmony_ci 508c2ecf20Sopenharmony_cistatic int mxl111sf_tuner_program_regs(struct mxl111sf_tuner_state *state, 518c2ecf20Sopenharmony_ci struct mxl111sf_reg_ctrl_info *ctrl_reg_info) 528c2ecf20Sopenharmony_ci{ 538c2ecf20Sopenharmony_ci return (state->cfg->program_regs) ? 548c2ecf20Sopenharmony_ci state->cfg->program_regs(state->mxl_state, ctrl_reg_info) : 558c2ecf20Sopenharmony_ci -EINVAL; 568c2ecf20Sopenharmony_ci} 578c2ecf20Sopenharmony_ci 588c2ecf20Sopenharmony_cistatic int mxl1x1sf_tuner_top_master_ctrl(struct mxl111sf_tuner_state *state, 598c2ecf20Sopenharmony_ci int onoff) 608c2ecf20Sopenharmony_ci{ 618c2ecf20Sopenharmony_ci return (state->cfg->top_master_ctrl) ? 628c2ecf20Sopenharmony_ci state->cfg->top_master_ctrl(state->mxl_state, onoff) : 638c2ecf20Sopenharmony_ci -EINVAL; 648c2ecf20Sopenharmony_ci} 658c2ecf20Sopenharmony_ci 668c2ecf20Sopenharmony_ci/* ------------------------------------------------------------------------ */ 678c2ecf20Sopenharmony_ci 688c2ecf20Sopenharmony_cistatic struct mxl111sf_reg_ctrl_info mxl_phy_tune_rf[] = { 698c2ecf20Sopenharmony_ci {0x1d, 0x7f, 0x00}, /* channel bandwidth section 1/2/3, 708c2ecf20Sopenharmony_ci DIG_MODEINDEX, _A, _CSF, */ 718c2ecf20Sopenharmony_ci {0x1e, 0xff, 0x00}, /* channel frequency (lo and fractional) */ 728c2ecf20Sopenharmony_ci {0x1f, 0xff, 0x00}, /* channel frequency (hi for integer portion) */ 738c2ecf20Sopenharmony_ci {0, 0, 0} 748c2ecf20Sopenharmony_ci}; 758c2ecf20Sopenharmony_ci 768c2ecf20Sopenharmony_ci/* ------------------------------------------------------------------------ */ 778c2ecf20Sopenharmony_ci 788c2ecf20Sopenharmony_cistatic struct mxl111sf_reg_ctrl_info *mxl111sf_calc_phy_tune_regs(u32 freq, 798c2ecf20Sopenharmony_ci u8 bw) 808c2ecf20Sopenharmony_ci{ 818c2ecf20Sopenharmony_ci u8 filt_bw; 828c2ecf20Sopenharmony_ci 838c2ecf20Sopenharmony_ci /* set channel bandwidth */ 848c2ecf20Sopenharmony_ci switch (bw) { 858c2ecf20Sopenharmony_ci case 0: /* ATSC */ 868c2ecf20Sopenharmony_ci filt_bw = 25; 878c2ecf20Sopenharmony_ci break; 888c2ecf20Sopenharmony_ci case 1: /* QAM */ 898c2ecf20Sopenharmony_ci filt_bw = 69; 908c2ecf20Sopenharmony_ci break; 918c2ecf20Sopenharmony_ci case 6: 928c2ecf20Sopenharmony_ci filt_bw = 21; 938c2ecf20Sopenharmony_ci break; 948c2ecf20Sopenharmony_ci case 7: 958c2ecf20Sopenharmony_ci filt_bw = 42; 968c2ecf20Sopenharmony_ci break; 978c2ecf20Sopenharmony_ci case 8: 988c2ecf20Sopenharmony_ci filt_bw = 63; 998c2ecf20Sopenharmony_ci break; 1008c2ecf20Sopenharmony_ci default: 1018c2ecf20Sopenharmony_ci pr_err("%s: invalid bandwidth setting!", __func__); 1028c2ecf20Sopenharmony_ci return NULL; 1038c2ecf20Sopenharmony_ci } 1048c2ecf20Sopenharmony_ci 1058c2ecf20Sopenharmony_ci /* calculate RF channel */ 1068c2ecf20Sopenharmony_ci freq /= 1000000; 1078c2ecf20Sopenharmony_ci 1088c2ecf20Sopenharmony_ci freq *= 64; 1098c2ecf20Sopenharmony_ci#if 0 1108c2ecf20Sopenharmony_ci /* do round */ 1118c2ecf20Sopenharmony_ci freq += 0.5; 1128c2ecf20Sopenharmony_ci#endif 1138c2ecf20Sopenharmony_ci /* set bandwidth */ 1148c2ecf20Sopenharmony_ci mxl_phy_tune_rf[0].data = filt_bw; 1158c2ecf20Sopenharmony_ci 1168c2ecf20Sopenharmony_ci /* set RF */ 1178c2ecf20Sopenharmony_ci mxl_phy_tune_rf[1].data = (freq & 0xff); 1188c2ecf20Sopenharmony_ci mxl_phy_tune_rf[2].data = (freq >> 8) & 0xff; 1198c2ecf20Sopenharmony_ci 1208c2ecf20Sopenharmony_ci /* start tune */ 1218c2ecf20Sopenharmony_ci return mxl_phy_tune_rf; 1228c2ecf20Sopenharmony_ci} 1238c2ecf20Sopenharmony_ci 1248c2ecf20Sopenharmony_cistatic int mxl1x1sf_tuner_set_if_output_freq(struct mxl111sf_tuner_state *state) 1258c2ecf20Sopenharmony_ci{ 1268c2ecf20Sopenharmony_ci int ret; 1278c2ecf20Sopenharmony_ci u8 ctrl; 1288c2ecf20Sopenharmony_ci#if 0 1298c2ecf20Sopenharmony_ci u16 iffcw; 1308c2ecf20Sopenharmony_ci u32 if_freq; 1318c2ecf20Sopenharmony_ci#endif 1328c2ecf20Sopenharmony_ci mxl_dbg("(IF polarity = %d, IF freq = 0x%02x)", 1338c2ecf20Sopenharmony_ci state->cfg->invert_spectrum, state->cfg->if_freq); 1348c2ecf20Sopenharmony_ci 1358c2ecf20Sopenharmony_ci /* set IF polarity */ 1368c2ecf20Sopenharmony_ci ctrl = state->cfg->invert_spectrum; 1378c2ecf20Sopenharmony_ci 1388c2ecf20Sopenharmony_ci ctrl |= state->cfg->if_freq; 1398c2ecf20Sopenharmony_ci 1408c2ecf20Sopenharmony_ci ret = mxl111sf_tuner_write_reg(state, V6_TUNER_IF_SEL_REG, ctrl); 1418c2ecf20Sopenharmony_ci if (mxl_fail(ret)) 1428c2ecf20Sopenharmony_ci goto fail; 1438c2ecf20Sopenharmony_ci 1448c2ecf20Sopenharmony_ci#if 0 1458c2ecf20Sopenharmony_ci if_freq /= 1000000; 1468c2ecf20Sopenharmony_ci 1478c2ecf20Sopenharmony_ci /* do round */ 1488c2ecf20Sopenharmony_ci if_freq += 0.5; 1498c2ecf20Sopenharmony_ci 1508c2ecf20Sopenharmony_ci if (MXL_IF_LO == state->cfg->if_freq) { 1518c2ecf20Sopenharmony_ci ctrl = 0x08; 1528c2ecf20Sopenharmony_ci iffcw = (u16)(if_freq / (108 * 4096)); 1538c2ecf20Sopenharmony_ci } else if (MXL_IF_HI == state->cfg->if_freq) { 1548c2ecf20Sopenharmony_ci ctrl = 0x08; 1558c2ecf20Sopenharmony_ci iffcw = (u16)(if_freq / (216 * 4096)); 1568c2ecf20Sopenharmony_ci } else { 1578c2ecf20Sopenharmony_ci ctrl = 0; 1588c2ecf20Sopenharmony_ci iffcw = 0; 1598c2ecf20Sopenharmony_ci } 1608c2ecf20Sopenharmony_ci 1618c2ecf20Sopenharmony_ci ctrl |= (iffcw >> 8); 1628c2ecf20Sopenharmony_ci#endif 1638c2ecf20Sopenharmony_ci ret = mxl111sf_tuner_read_reg(state, V6_TUNER_IF_FCW_BYP_REG, &ctrl); 1648c2ecf20Sopenharmony_ci if (mxl_fail(ret)) 1658c2ecf20Sopenharmony_ci goto fail; 1668c2ecf20Sopenharmony_ci 1678c2ecf20Sopenharmony_ci ctrl &= 0xf0; 1688c2ecf20Sopenharmony_ci ctrl |= 0x90; 1698c2ecf20Sopenharmony_ci 1708c2ecf20Sopenharmony_ci ret = mxl111sf_tuner_write_reg(state, V6_TUNER_IF_FCW_BYP_REG, ctrl); 1718c2ecf20Sopenharmony_ci if (mxl_fail(ret)) 1728c2ecf20Sopenharmony_ci goto fail; 1738c2ecf20Sopenharmony_ci 1748c2ecf20Sopenharmony_ci#if 0 1758c2ecf20Sopenharmony_ci ctrl = iffcw & 0x00ff; 1768c2ecf20Sopenharmony_ci#endif 1778c2ecf20Sopenharmony_ci ret = mxl111sf_tuner_write_reg(state, V6_TUNER_IF_FCW_REG, ctrl); 1788c2ecf20Sopenharmony_ci if (mxl_fail(ret)) 1798c2ecf20Sopenharmony_ci goto fail; 1808c2ecf20Sopenharmony_ci 1818c2ecf20Sopenharmony_ci state->if_freq = state->cfg->if_freq; 1828c2ecf20Sopenharmony_cifail: 1838c2ecf20Sopenharmony_ci return ret; 1848c2ecf20Sopenharmony_ci} 1858c2ecf20Sopenharmony_ci 1868c2ecf20Sopenharmony_cistatic int mxl1x1sf_tune_rf(struct dvb_frontend *fe, u32 freq, u8 bw) 1878c2ecf20Sopenharmony_ci{ 1888c2ecf20Sopenharmony_ci struct mxl111sf_tuner_state *state = fe->tuner_priv; 1898c2ecf20Sopenharmony_ci static struct mxl111sf_reg_ctrl_info *reg_ctrl_array; 1908c2ecf20Sopenharmony_ci int ret; 1918c2ecf20Sopenharmony_ci u8 mxl_mode; 1928c2ecf20Sopenharmony_ci 1938c2ecf20Sopenharmony_ci mxl_dbg("(freq = %d, bw = 0x%x)", freq, bw); 1948c2ecf20Sopenharmony_ci 1958c2ecf20Sopenharmony_ci /* stop tune */ 1968c2ecf20Sopenharmony_ci ret = mxl111sf_tuner_write_reg(state, START_TUNE_REG, 0); 1978c2ecf20Sopenharmony_ci if (mxl_fail(ret)) 1988c2ecf20Sopenharmony_ci goto fail; 1998c2ecf20Sopenharmony_ci 2008c2ecf20Sopenharmony_ci /* check device mode */ 2018c2ecf20Sopenharmony_ci ret = mxl111sf_tuner_read_reg(state, MXL_MODE_REG, &mxl_mode); 2028c2ecf20Sopenharmony_ci if (mxl_fail(ret)) 2038c2ecf20Sopenharmony_ci goto fail; 2048c2ecf20Sopenharmony_ci 2058c2ecf20Sopenharmony_ci /* Fill out registers for channel tune */ 2068c2ecf20Sopenharmony_ci reg_ctrl_array = mxl111sf_calc_phy_tune_regs(freq, bw); 2078c2ecf20Sopenharmony_ci if (!reg_ctrl_array) 2088c2ecf20Sopenharmony_ci return -EINVAL; 2098c2ecf20Sopenharmony_ci 2108c2ecf20Sopenharmony_ci ret = mxl111sf_tuner_program_regs(state, reg_ctrl_array); 2118c2ecf20Sopenharmony_ci if (mxl_fail(ret)) 2128c2ecf20Sopenharmony_ci goto fail; 2138c2ecf20Sopenharmony_ci 2148c2ecf20Sopenharmony_ci if ((mxl_mode & MXL_DEV_MODE_MASK) == MXL_TUNER_MODE) { 2158c2ecf20Sopenharmony_ci /* IF tuner mode only */ 2168c2ecf20Sopenharmony_ci mxl1x1sf_tuner_top_master_ctrl(state, 0); 2178c2ecf20Sopenharmony_ci mxl1x1sf_tuner_top_master_ctrl(state, 1); 2188c2ecf20Sopenharmony_ci mxl1x1sf_tuner_set_if_output_freq(state); 2198c2ecf20Sopenharmony_ci } 2208c2ecf20Sopenharmony_ci 2218c2ecf20Sopenharmony_ci ret = mxl111sf_tuner_write_reg(state, START_TUNE_REG, 1); 2228c2ecf20Sopenharmony_ci if (mxl_fail(ret)) 2238c2ecf20Sopenharmony_ci goto fail; 2248c2ecf20Sopenharmony_ci 2258c2ecf20Sopenharmony_ci if (state->cfg->ant_hunt) 2268c2ecf20Sopenharmony_ci state->cfg->ant_hunt(fe); 2278c2ecf20Sopenharmony_cifail: 2288c2ecf20Sopenharmony_ci return ret; 2298c2ecf20Sopenharmony_ci} 2308c2ecf20Sopenharmony_ci 2318c2ecf20Sopenharmony_cistatic int mxl1x1sf_tuner_get_lock_status(struct mxl111sf_tuner_state *state, 2328c2ecf20Sopenharmony_ci int *rf_synth_lock, 2338c2ecf20Sopenharmony_ci int *ref_synth_lock) 2348c2ecf20Sopenharmony_ci{ 2358c2ecf20Sopenharmony_ci int ret; 2368c2ecf20Sopenharmony_ci u8 data; 2378c2ecf20Sopenharmony_ci 2388c2ecf20Sopenharmony_ci *rf_synth_lock = 0; 2398c2ecf20Sopenharmony_ci *ref_synth_lock = 0; 2408c2ecf20Sopenharmony_ci 2418c2ecf20Sopenharmony_ci ret = mxl111sf_tuner_read_reg(state, V6_RF_LOCK_STATUS_REG, &data); 2428c2ecf20Sopenharmony_ci if (mxl_fail(ret)) 2438c2ecf20Sopenharmony_ci goto fail; 2448c2ecf20Sopenharmony_ci 2458c2ecf20Sopenharmony_ci *ref_synth_lock = ((data & 0x03) == 0x03) ? 1 : 0; 2468c2ecf20Sopenharmony_ci *rf_synth_lock = ((data & 0x0c) == 0x0c) ? 1 : 0; 2478c2ecf20Sopenharmony_cifail: 2488c2ecf20Sopenharmony_ci return ret; 2498c2ecf20Sopenharmony_ci} 2508c2ecf20Sopenharmony_ci 2518c2ecf20Sopenharmony_ci#if 0 2528c2ecf20Sopenharmony_cistatic int mxl1x1sf_tuner_loop_thru_ctrl(struct mxl111sf_tuner_state *state, 2538c2ecf20Sopenharmony_ci int onoff) 2548c2ecf20Sopenharmony_ci{ 2558c2ecf20Sopenharmony_ci return mxl111sf_tuner_write_reg(state, V6_TUNER_LOOP_THRU_CTRL_REG, 2568c2ecf20Sopenharmony_ci onoff ? 1 : 0); 2578c2ecf20Sopenharmony_ci} 2588c2ecf20Sopenharmony_ci#endif 2598c2ecf20Sopenharmony_ci 2608c2ecf20Sopenharmony_ci/* ------------------------------------------------------------------------ */ 2618c2ecf20Sopenharmony_ci 2628c2ecf20Sopenharmony_cistatic int mxl111sf_tuner_set_params(struct dvb_frontend *fe) 2638c2ecf20Sopenharmony_ci{ 2648c2ecf20Sopenharmony_ci struct dtv_frontend_properties *c = &fe->dtv_property_cache; 2658c2ecf20Sopenharmony_ci u32 delsys = c->delivery_system; 2668c2ecf20Sopenharmony_ci struct mxl111sf_tuner_state *state = fe->tuner_priv; 2678c2ecf20Sopenharmony_ci int ret; 2688c2ecf20Sopenharmony_ci u8 bw; 2698c2ecf20Sopenharmony_ci 2708c2ecf20Sopenharmony_ci mxl_dbg("()"); 2718c2ecf20Sopenharmony_ci 2728c2ecf20Sopenharmony_ci switch (delsys) { 2738c2ecf20Sopenharmony_ci case SYS_ATSC: 2748c2ecf20Sopenharmony_ci case SYS_ATSCMH: 2758c2ecf20Sopenharmony_ci bw = 0; /* ATSC */ 2768c2ecf20Sopenharmony_ci break; 2778c2ecf20Sopenharmony_ci case SYS_DVBC_ANNEX_B: 2788c2ecf20Sopenharmony_ci bw = 1; /* US CABLE */ 2798c2ecf20Sopenharmony_ci break; 2808c2ecf20Sopenharmony_ci case SYS_DVBT: 2818c2ecf20Sopenharmony_ci switch (c->bandwidth_hz) { 2828c2ecf20Sopenharmony_ci case 6000000: 2838c2ecf20Sopenharmony_ci bw = 6; 2848c2ecf20Sopenharmony_ci break; 2858c2ecf20Sopenharmony_ci case 7000000: 2868c2ecf20Sopenharmony_ci bw = 7; 2878c2ecf20Sopenharmony_ci break; 2888c2ecf20Sopenharmony_ci case 8000000: 2898c2ecf20Sopenharmony_ci bw = 8; 2908c2ecf20Sopenharmony_ci break; 2918c2ecf20Sopenharmony_ci default: 2928c2ecf20Sopenharmony_ci pr_err("%s: bandwidth not set!", __func__); 2938c2ecf20Sopenharmony_ci return -EINVAL; 2948c2ecf20Sopenharmony_ci } 2958c2ecf20Sopenharmony_ci break; 2968c2ecf20Sopenharmony_ci default: 2978c2ecf20Sopenharmony_ci pr_err("%s: modulation type not supported!", __func__); 2988c2ecf20Sopenharmony_ci return -EINVAL; 2998c2ecf20Sopenharmony_ci } 3008c2ecf20Sopenharmony_ci ret = mxl1x1sf_tune_rf(fe, c->frequency, bw); 3018c2ecf20Sopenharmony_ci if (mxl_fail(ret)) 3028c2ecf20Sopenharmony_ci goto fail; 3038c2ecf20Sopenharmony_ci 3048c2ecf20Sopenharmony_ci state->frequency = c->frequency; 3058c2ecf20Sopenharmony_ci state->bandwidth = c->bandwidth_hz; 3068c2ecf20Sopenharmony_cifail: 3078c2ecf20Sopenharmony_ci return ret; 3088c2ecf20Sopenharmony_ci} 3098c2ecf20Sopenharmony_ci 3108c2ecf20Sopenharmony_ci/* ------------------------------------------------------------------------ */ 3118c2ecf20Sopenharmony_ci 3128c2ecf20Sopenharmony_ci#if 0 3138c2ecf20Sopenharmony_cistatic int mxl111sf_tuner_init(struct dvb_frontend *fe) 3148c2ecf20Sopenharmony_ci{ 3158c2ecf20Sopenharmony_ci struct mxl111sf_tuner_state *state = fe->tuner_priv; 3168c2ecf20Sopenharmony_ci int ret; 3178c2ecf20Sopenharmony_ci 3188c2ecf20Sopenharmony_ci /* wake from standby handled by usb driver */ 3198c2ecf20Sopenharmony_ci 3208c2ecf20Sopenharmony_ci return ret; 3218c2ecf20Sopenharmony_ci} 3228c2ecf20Sopenharmony_ci 3238c2ecf20Sopenharmony_cistatic int mxl111sf_tuner_sleep(struct dvb_frontend *fe) 3248c2ecf20Sopenharmony_ci{ 3258c2ecf20Sopenharmony_ci struct mxl111sf_tuner_state *state = fe->tuner_priv; 3268c2ecf20Sopenharmony_ci int ret; 3278c2ecf20Sopenharmony_ci 3288c2ecf20Sopenharmony_ci /* enter standby mode handled by usb driver */ 3298c2ecf20Sopenharmony_ci 3308c2ecf20Sopenharmony_ci return ret; 3318c2ecf20Sopenharmony_ci} 3328c2ecf20Sopenharmony_ci#endif 3338c2ecf20Sopenharmony_ci 3348c2ecf20Sopenharmony_ci/* ------------------------------------------------------------------------ */ 3358c2ecf20Sopenharmony_ci 3368c2ecf20Sopenharmony_cistatic int mxl111sf_tuner_get_status(struct dvb_frontend *fe, u32 *status) 3378c2ecf20Sopenharmony_ci{ 3388c2ecf20Sopenharmony_ci struct mxl111sf_tuner_state *state = fe->tuner_priv; 3398c2ecf20Sopenharmony_ci int rf_locked, ref_locked, ret; 3408c2ecf20Sopenharmony_ci 3418c2ecf20Sopenharmony_ci *status = 0; 3428c2ecf20Sopenharmony_ci 3438c2ecf20Sopenharmony_ci ret = mxl1x1sf_tuner_get_lock_status(state, &rf_locked, &ref_locked); 3448c2ecf20Sopenharmony_ci if (mxl_fail(ret)) 3458c2ecf20Sopenharmony_ci goto fail; 3468c2ecf20Sopenharmony_ci mxl_info("%s%s", rf_locked ? "rf locked " : "", 3478c2ecf20Sopenharmony_ci ref_locked ? "ref locked" : ""); 3488c2ecf20Sopenharmony_ci 3498c2ecf20Sopenharmony_ci if ((rf_locked) || (ref_locked)) 3508c2ecf20Sopenharmony_ci *status |= TUNER_STATUS_LOCKED; 3518c2ecf20Sopenharmony_cifail: 3528c2ecf20Sopenharmony_ci return ret; 3538c2ecf20Sopenharmony_ci} 3548c2ecf20Sopenharmony_ci 3558c2ecf20Sopenharmony_cistatic int mxl111sf_get_rf_strength(struct dvb_frontend *fe, u16 *strength) 3568c2ecf20Sopenharmony_ci{ 3578c2ecf20Sopenharmony_ci struct mxl111sf_tuner_state *state = fe->tuner_priv; 3588c2ecf20Sopenharmony_ci u8 val1, val2; 3598c2ecf20Sopenharmony_ci int ret; 3608c2ecf20Sopenharmony_ci 3618c2ecf20Sopenharmony_ci *strength = 0; 3628c2ecf20Sopenharmony_ci 3638c2ecf20Sopenharmony_ci ret = mxl111sf_tuner_write_reg(state, 0x00, 0x02); 3648c2ecf20Sopenharmony_ci if (mxl_fail(ret)) 3658c2ecf20Sopenharmony_ci goto fail; 3668c2ecf20Sopenharmony_ci ret = mxl111sf_tuner_read_reg(state, V6_DIG_RF_PWR_LSB_REG, &val1); 3678c2ecf20Sopenharmony_ci if (mxl_fail(ret)) 3688c2ecf20Sopenharmony_ci goto fail; 3698c2ecf20Sopenharmony_ci ret = mxl111sf_tuner_read_reg(state, V6_DIG_RF_PWR_MSB_REG, &val2); 3708c2ecf20Sopenharmony_ci if (mxl_fail(ret)) 3718c2ecf20Sopenharmony_ci goto fail; 3728c2ecf20Sopenharmony_ci 3738c2ecf20Sopenharmony_ci *strength = val1 | ((val2 & 0x07) << 8); 3748c2ecf20Sopenharmony_cifail: 3758c2ecf20Sopenharmony_ci ret = mxl111sf_tuner_write_reg(state, 0x00, 0x00); 3768c2ecf20Sopenharmony_ci mxl_fail(ret); 3778c2ecf20Sopenharmony_ci 3788c2ecf20Sopenharmony_ci return ret; 3798c2ecf20Sopenharmony_ci} 3808c2ecf20Sopenharmony_ci 3818c2ecf20Sopenharmony_ci/* ------------------------------------------------------------------------ */ 3828c2ecf20Sopenharmony_ci 3838c2ecf20Sopenharmony_cistatic int mxl111sf_tuner_get_frequency(struct dvb_frontend *fe, u32 *frequency) 3848c2ecf20Sopenharmony_ci{ 3858c2ecf20Sopenharmony_ci struct mxl111sf_tuner_state *state = fe->tuner_priv; 3868c2ecf20Sopenharmony_ci *frequency = state->frequency; 3878c2ecf20Sopenharmony_ci return 0; 3888c2ecf20Sopenharmony_ci} 3898c2ecf20Sopenharmony_ci 3908c2ecf20Sopenharmony_cistatic int mxl111sf_tuner_get_bandwidth(struct dvb_frontend *fe, u32 *bandwidth) 3918c2ecf20Sopenharmony_ci{ 3928c2ecf20Sopenharmony_ci struct mxl111sf_tuner_state *state = fe->tuner_priv; 3938c2ecf20Sopenharmony_ci *bandwidth = state->bandwidth; 3948c2ecf20Sopenharmony_ci return 0; 3958c2ecf20Sopenharmony_ci} 3968c2ecf20Sopenharmony_ci 3978c2ecf20Sopenharmony_cistatic int mxl111sf_tuner_get_if_frequency(struct dvb_frontend *fe, 3988c2ecf20Sopenharmony_ci u32 *frequency) 3998c2ecf20Sopenharmony_ci{ 4008c2ecf20Sopenharmony_ci struct mxl111sf_tuner_state *state = fe->tuner_priv; 4018c2ecf20Sopenharmony_ci 4028c2ecf20Sopenharmony_ci *frequency = 0; 4038c2ecf20Sopenharmony_ci 4048c2ecf20Sopenharmony_ci switch (state->if_freq) { 4058c2ecf20Sopenharmony_ci case MXL_IF_4_0: /* 4.0 MHz */ 4068c2ecf20Sopenharmony_ci *frequency = 4000000; 4078c2ecf20Sopenharmony_ci break; 4088c2ecf20Sopenharmony_ci case MXL_IF_4_5: /* 4.5 MHz */ 4098c2ecf20Sopenharmony_ci *frequency = 4500000; 4108c2ecf20Sopenharmony_ci break; 4118c2ecf20Sopenharmony_ci case MXL_IF_4_57: /* 4.57 MHz */ 4128c2ecf20Sopenharmony_ci *frequency = 4570000; 4138c2ecf20Sopenharmony_ci break; 4148c2ecf20Sopenharmony_ci case MXL_IF_5_0: /* 5.0 MHz */ 4158c2ecf20Sopenharmony_ci *frequency = 5000000; 4168c2ecf20Sopenharmony_ci break; 4178c2ecf20Sopenharmony_ci case MXL_IF_5_38: /* 5.38 MHz */ 4188c2ecf20Sopenharmony_ci *frequency = 5380000; 4198c2ecf20Sopenharmony_ci break; 4208c2ecf20Sopenharmony_ci case MXL_IF_6_0: /* 6.0 MHz */ 4218c2ecf20Sopenharmony_ci *frequency = 6000000; 4228c2ecf20Sopenharmony_ci break; 4238c2ecf20Sopenharmony_ci case MXL_IF_6_28: /* 6.28 MHz */ 4248c2ecf20Sopenharmony_ci *frequency = 6280000; 4258c2ecf20Sopenharmony_ci break; 4268c2ecf20Sopenharmony_ci case MXL_IF_7_2: /* 7.2 MHz */ 4278c2ecf20Sopenharmony_ci *frequency = 7200000; 4288c2ecf20Sopenharmony_ci break; 4298c2ecf20Sopenharmony_ci case MXL_IF_35_25: /* 35.25 MHz */ 4308c2ecf20Sopenharmony_ci *frequency = 35250000; 4318c2ecf20Sopenharmony_ci break; 4328c2ecf20Sopenharmony_ci case MXL_IF_36: /* 36 MHz */ 4338c2ecf20Sopenharmony_ci *frequency = 36000000; 4348c2ecf20Sopenharmony_ci break; 4358c2ecf20Sopenharmony_ci case MXL_IF_36_15: /* 36.15 MHz */ 4368c2ecf20Sopenharmony_ci *frequency = 36150000; 4378c2ecf20Sopenharmony_ci break; 4388c2ecf20Sopenharmony_ci case MXL_IF_44: /* 44 MHz */ 4398c2ecf20Sopenharmony_ci *frequency = 44000000; 4408c2ecf20Sopenharmony_ci break; 4418c2ecf20Sopenharmony_ci } 4428c2ecf20Sopenharmony_ci return 0; 4438c2ecf20Sopenharmony_ci} 4448c2ecf20Sopenharmony_ci 4458c2ecf20Sopenharmony_cistatic void mxl111sf_tuner_release(struct dvb_frontend *fe) 4468c2ecf20Sopenharmony_ci{ 4478c2ecf20Sopenharmony_ci struct mxl111sf_tuner_state *state = fe->tuner_priv; 4488c2ecf20Sopenharmony_ci mxl_dbg("()"); 4498c2ecf20Sopenharmony_ci kfree(state); 4508c2ecf20Sopenharmony_ci fe->tuner_priv = NULL; 4518c2ecf20Sopenharmony_ci} 4528c2ecf20Sopenharmony_ci 4538c2ecf20Sopenharmony_ci/* ------------------------------------------------------------------------- */ 4548c2ecf20Sopenharmony_ci 4558c2ecf20Sopenharmony_cistatic const struct dvb_tuner_ops mxl111sf_tuner_tuner_ops = { 4568c2ecf20Sopenharmony_ci .info = { 4578c2ecf20Sopenharmony_ci .name = "MaxLinear MxL111SF", 4588c2ecf20Sopenharmony_ci#if 0 4598c2ecf20Sopenharmony_ci .frequency_min_hz = , 4608c2ecf20Sopenharmony_ci .frequency_max_hz = , 4618c2ecf20Sopenharmony_ci .frequency_step_hz = , 4628c2ecf20Sopenharmony_ci#endif 4638c2ecf20Sopenharmony_ci }, 4648c2ecf20Sopenharmony_ci#if 0 4658c2ecf20Sopenharmony_ci .init = mxl111sf_tuner_init, 4668c2ecf20Sopenharmony_ci .sleep = mxl111sf_tuner_sleep, 4678c2ecf20Sopenharmony_ci#endif 4688c2ecf20Sopenharmony_ci .set_params = mxl111sf_tuner_set_params, 4698c2ecf20Sopenharmony_ci .get_status = mxl111sf_tuner_get_status, 4708c2ecf20Sopenharmony_ci .get_rf_strength = mxl111sf_get_rf_strength, 4718c2ecf20Sopenharmony_ci .get_frequency = mxl111sf_tuner_get_frequency, 4728c2ecf20Sopenharmony_ci .get_bandwidth = mxl111sf_tuner_get_bandwidth, 4738c2ecf20Sopenharmony_ci .get_if_frequency = mxl111sf_tuner_get_if_frequency, 4748c2ecf20Sopenharmony_ci .release = mxl111sf_tuner_release, 4758c2ecf20Sopenharmony_ci}; 4768c2ecf20Sopenharmony_ci 4778c2ecf20Sopenharmony_cistruct dvb_frontend *mxl111sf_tuner_attach(struct dvb_frontend *fe, 4788c2ecf20Sopenharmony_ci struct mxl111sf_state *mxl_state, 4798c2ecf20Sopenharmony_ci const struct mxl111sf_tuner_config *cfg) 4808c2ecf20Sopenharmony_ci{ 4818c2ecf20Sopenharmony_ci struct mxl111sf_tuner_state *state = NULL; 4828c2ecf20Sopenharmony_ci 4838c2ecf20Sopenharmony_ci mxl_dbg("()"); 4848c2ecf20Sopenharmony_ci 4858c2ecf20Sopenharmony_ci state = kzalloc(sizeof(struct mxl111sf_tuner_state), GFP_KERNEL); 4868c2ecf20Sopenharmony_ci if (state == NULL) 4878c2ecf20Sopenharmony_ci return NULL; 4888c2ecf20Sopenharmony_ci 4898c2ecf20Sopenharmony_ci state->mxl_state = mxl_state; 4908c2ecf20Sopenharmony_ci state->cfg = cfg; 4918c2ecf20Sopenharmony_ci 4928c2ecf20Sopenharmony_ci memcpy(&fe->ops.tuner_ops, &mxl111sf_tuner_tuner_ops, 4938c2ecf20Sopenharmony_ci sizeof(struct dvb_tuner_ops)); 4948c2ecf20Sopenharmony_ci 4958c2ecf20Sopenharmony_ci fe->tuner_priv = state; 4968c2ecf20Sopenharmony_ci return fe; 4978c2ecf20Sopenharmony_ci} 4988c2ecf20Sopenharmony_ciEXPORT_SYMBOL_GPL(mxl111sf_tuner_attach); 4998c2ecf20Sopenharmony_ci 5008c2ecf20Sopenharmony_ciMODULE_DESCRIPTION("MaxLinear MxL111SF CMOS tuner driver"); 5018c2ecf20Sopenharmony_ciMODULE_AUTHOR("Michael Krufky <mkrufky@linuxtv.org>"); 5028c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL"); 5038c2ecf20Sopenharmony_ciMODULE_VERSION("0.1"); 504