提交 25d8527a 编写于 作者: P Pantelis Koukousoulas 提交者: Mauro Carvalho Chehab

V4L/DVB (5035): Pvrusb2: Enable radio mode round #2

This is the logic that:
  a) Ensures /sys/class/pvrusb2/sn-*/ctl_frequency/{max,min}_val are
     "automagically" reset to sane values on each mode change.

  b) Allows tuning to a radio frequency by something like:
     echo `perl -e "print int(94.9*16000 + 0.5)"` \
       > /sys/class/pvrusb2/sn-*/ctl_input/cur_val


The trick was to take advantage of the already existing .get_{min,max}_value
function pointers in pvr2_ctrl, to "dynamically override" the hardcoded values
for min/max frequency at runtime.

For a moment I thought to dispose of the hardcoded MIN/MAX_FREQ and use the
hirange/lowrange fields of the v4l2_tuner struct instead, but then I see that
tuner-core.c kinda hardcodes these as well, so I decided to not bother.
Signed-off-by: NPantelis Koukousoulas <pakt223@freemail.gr>
Signed-off-by: NMike Isely <isely@pobox.com>
Signed-off-by: NMauro Carvalho Chehab <mchehab@infradead.org>
上级 275b2e28
...@@ -37,6 +37,12 @@ ...@@ -37,6 +37,12 @@
#include "pvrusb2-encoder.h" #include "pvrusb2-encoder.h"
#include "pvrusb2-debug.h" #include "pvrusb2-debug.h"
#define TV_MIN_FREQ 55250000L
#define TV_MAX_FREQ 850000000L
#define RADIO_MIN_FREQ 1392000L //87MHz
#define RADIO_MAX_FREQ 1728000L //108MHz
struct usb_device_id pvr2_device_table[] = { struct usb_device_id pvr2_device_table[] = {
[PVR2_HDW_TYPE_29XXX] = { USB_DEVICE(0x2040, 0x2900) }, [PVR2_HDW_TYPE_29XXX] = { USB_DEVICE(0x2040, 0x2900) },
[PVR2_HDW_TYPE_24XXX] = { USB_DEVICE(0x2040, 0x2400) }, [PVR2_HDW_TYPE_24XXX] = { USB_DEVICE(0x2040, 0x2400) },
...@@ -375,6 +381,28 @@ static int ctrl_vres_min_get(struct pvr2_ctrl *cptr,int *vp) ...@@ -375,6 +381,28 @@ static int ctrl_vres_min_get(struct pvr2_ctrl *cptr,int *vp)
return 0; return 0;
} }
static int ctrl_freq_max_get(struct pvr2_ctrl *cptr, int *vp)
{
/* Actual maximum depends on radio/tv mode */
if (cptr->hdw->input_val == PVR2_CVAL_INPUT_RADIO) {
*vp = RADIO_MAX_FREQ;
} else {
*vp = TV_MAX_FREQ;
}
return 0;
}
static int ctrl_freq_min_get(struct pvr2_ctrl *cptr, int *vp)
{
/* Actual minimum depends on radio/tv mode */
if (cptr->hdw->input_val == PVR2_CVAL_INPUT_RADIO) {
*vp = RADIO_MIN_FREQ;
} else {
*vp = TV_MIN_FREQ;
}
return 0;
}
static int ctrl_cx2341x_is_dirty(struct pvr2_ctrl *cptr) static int ctrl_cx2341x_is_dirty(struct pvr2_ctrl *cptr)
{ {
return cptr->hdw->enc_stale != 0; return cptr->hdw->enc_stale != 0;
...@@ -644,9 +672,6 @@ VCREATE_FUNCS(res_hor) ...@@ -644,9 +672,6 @@ VCREATE_FUNCS(res_hor)
VCREATE_FUNCS(res_ver) VCREATE_FUNCS(res_ver)
VCREATE_FUNCS(srate) VCREATE_FUNCS(srate)
#define MIN_FREQ 55250000L
#define MAX_FREQ 850000000L
/* Table definition of all controls which can be manipulated */ /* Table definition of all controls which can be manipulated */
static const struct pvr2_ctl_info control_defs[] = { static const struct pvr2_ctl_info control_defs[] = {
{ {
...@@ -760,7 +785,11 @@ static const struct pvr2_ctl_info control_defs[] = { ...@@ -760,7 +785,11 @@ static const struct pvr2_ctl_info control_defs[] = {
.get_value = ctrl_freq_get, .get_value = ctrl_freq_get,
.is_dirty = ctrl_freq_is_dirty, .is_dirty = ctrl_freq_is_dirty,
.clear_dirty = ctrl_freq_clear_dirty, .clear_dirty = ctrl_freq_clear_dirty,
DEFINT(MIN_FREQ,MAX_FREQ), DEFINT(TV_MIN_FREQ,TV_MAX_FREQ),
/* Hook in check for input value (tv/radio) and adjust
max/min values accordingly */
.get_max_value = ctrl_freq_max_get,
.get_min_value = ctrl_freq_min_get,
},{ },{
.desc = "Channel", .desc = "Channel",
.name = "channel", .name = "channel",
...@@ -772,7 +801,7 @@ static const struct pvr2_ctl_info control_defs[] = { ...@@ -772,7 +801,7 @@ static const struct pvr2_ctl_info control_defs[] = {
.name = "freq_table_value", .name = "freq_table_value",
.set_value = ctrl_channelfreq_set, .set_value = ctrl_channelfreq_set,
.get_value = ctrl_channelfreq_get, .get_value = ctrl_channelfreq_get,
DEFINT(MIN_FREQ,MAX_FREQ), DEFINT(TV_MIN_FREQ,TV_MAX_FREQ),
},{ },{
.desc = "Channel Program ID", .desc = "Channel Program ID",
.name = "freq_table_channel", .name = "freq_table_channel",
......
...@@ -169,7 +169,9 @@ static void set_frequency(struct pvr2_hdw *hdw) ...@@ -169,7 +169,9 @@ static void set_frequency(struct pvr2_hdw *hdw)
fv = hdw->freqVal; fv = hdw->freqVal;
pvr2_trace(PVR2_TRACE_CHIPS,"i2c v4l2 set_freq(%lu)",fv); pvr2_trace(PVR2_TRACE_CHIPS,"i2c v4l2 set_freq(%lu)",fv);
memset(&freq,0,sizeof(freq)); memset(&freq,0,sizeof(freq));
freq.frequency = fv / 62500; if (hdw->input_val == PVR2_CVAL_INPUT_TV)
fv /= 62500;
freq.frequency = fv;
freq.tuner = 0; freq.tuner = 0;
freq.type = (hdw->input_val == PVR2_CVAL_INPUT_RADIO) ? freq.type = (hdw->input_val == PVR2_CVAL_INPUT_RADIO) ?
V4L2_TUNER_RADIO : V4L2_TUNER_ANALOG_TV; V4L2_TUNER_RADIO : V4L2_TUNER_ANALOG_TV;
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册