606cf9caeb
Audio mode changes are not private to the audio chip - other I2C modules need to see this as well. And since the command in question is VIDIOC_S_TUNER which is a standard v4l2 command, we really should be broadcasting it out. This change sets up a broadcast pathway for VIDIOC_S_TUNER and also eliminates the now redundant code from the audio chip handler. This fix enables stereo reception for the FM radio Signed-off-by: Mike Isely <isely@pobox.com> Signed-off-by: Mauro Carvalho Chehab <mchehab@infradead.org>
280 lines
6.4 KiB
C
280 lines
6.4 KiB
C
/*
|
|
*
|
|
* $Id$
|
|
*
|
|
* Copyright (C) 2005 Mike Isely <isely@pobox.com>
|
|
* Copyright (C) 2004 Aurelien Alleaume <slts@free.fr>
|
|
*
|
|
* 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; either version 2 of the License
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*
|
|
*/
|
|
|
|
#include "pvrusb2-i2c-cmd-v4l2.h"
|
|
#include "pvrusb2-hdw-internal.h"
|
|
#include "pvrusb2-debug.h"
|
|
#include <linux/videodev2.h>
|
|
#include <media/v4l2-common.h>
|
|
|
|
static void set_standard(struct pvr2_hdw *hdw)
|
|
{
|
|
pvr2_trace(PVR2_TRACE_CHIPS,"i2c v4l2 set_standard");
|
|
|
|
if (hdw->input_val == PVR2_CVAL_INPUT_RADIO) {
|
|
pvr2_i2c_core_cmd(hdw,AUDC_SET_RADIO,NULL);
|
|
} else {
|
|
v4l2_std_id vs;
|
|
vs = hdw->std_mask_cur;
|
|
pvr2_i2c_core_cmd(hdw,VIDIOC_S_STD,&vs);
|
|
}
|
|
hdw->tuner_signal_stale = !0;
|
|
}
|
|
|
|
|
|
static int check_standard(struct pvr2_hdw *hdw)
|
|
{
|
|
return (hdw->input_dirty != 0) || (hdw->std_dirty != 0);
|
|
}
|
|
|
|
|
|
const struct pvr2_i2c_op pvr2_i2c_op_v4l2_standard = {
|
|
.check = check_standard,
|
|
.update = set_standard,
|
|
.name = "v4l2_standard",
|
|
};
|
|
|
|
|
|
static void set_bcsh(struct pvr2_hdw *hdw)
|
|
{
|
|
struct v4l2_control ctrl;
|
|
pvr2_trace(PVR2_TRACE_CHIPS,"i2c v4l2 set_bcsh"
|
|
" b=%d c=%d s=%d h=%d",
|
|
hdw->brightness_val,hdw->contrast_val,
|
|
hdw->saturation_val,hdw->hue_val);
|
|
memset(&ctrl,0,sizeof(ctrl));
|
|
ctrl.id = V4L2_CID_BRIGHTNESS;
|
|
ctrl.value = hdw->brightness_val;
|
|
pvr2_i2c_core_cmd(hdw,VIDIOC_S_CTRL,&ctrl);
|
|
ctrl.id = V4L2_CID_CONTRAST;
|
|
ctrl.value = hdw->contrast_val;
|
|
pvr2_i2c_core_cmd(hdw,VIDIOC_S_CTRL,&ctrl);
|
|
ctrl.id = V4L2_CID_SATURATION;
|
|
ctrl.value = hdw->saturation_val;
|
|
pvr2_i2c_core_cmd(hdw,VIDIOC_S_CTRL,&ctrl);
|
|
ctrl.id = V4L2_CID_HUE;
|
|
ctrl.value = hdw->hue_val;
|
|
pvr2_i2c_core_cmd(hdw,VIDIOC_S_CTRL,&ctrl);
|
|
}
|
|
|
|
|
|
static int check_bcsh(struct pvr2_hdw *hdw)
|
|
{
|
|
return (hdw->brightness_dirty ||
|
|
hdw->contrast_dirty ||
|
|
hdw->saturation_dirty ||
|
|
hdw->hue_dirty);
|
|
}
|
|
|
|
|
|
const struct pvr2_i2c_op pvr2_i2c_op_v4l2_bcsh = {
|
|
.check = check_bcsh,
|
|
.update = set_bcsh,
|
|
.name = "v4l2_bcsh",
|
|
};
|
|
|
|
|
|
static void set_volume(struct pvr2_hdw *hdw)
|
|
{
|
|
struct v4l2_control ctrl;
|
|
pvr2_trace(PVR2_TRACE_CHIPS,
|
|
"i2c v4l2 set_volume"
|
|
"(vol=%d bal=%d bas=%d treb=%d mute=%d)",
|
|
hdw->volume_val,
|
|
hdw->balance_val,
|
|
hdw->bass_val,
|
|
hdw->treble_val,
|
|
hdw->mute_val);
|
|
memset(&ctrl,0,sizeof(ctrl));
|
|
ctrl.id = V4L2_CID_AUDIO_MUTE;
|
|
ctrl.value = hdw->mute_val ? 1 : 0;
|
|
pvr2_i2c_core_cmd(hdw,VIDIOC_S_CTRL,&ctrl);
|
|
ctrl.id = V4L2_CID_AUDIO_VOLUME;
|
|
ctrl.value = hdw->volume_val;
|
|
pvr2_i2c_core_cmd(hdw,VIDIOC_S_CTRL,&ctrl);
|
|
ctrl.id = V4L2_CID_AUDIO_BALANCE;
|
|
ctrl.value = hdw->balance_val;
|
|
pvr2_i2c_core_cmd(hdw,VIDIOC_S_CTRL,&ctrl);
|
|
ctrl.id = V4L2_CID_AUDIO_BASS;
|
|
ctrl.value = hdw->bass_val;
|
|
pvr2_i2c_core_cmd(hdw,VIDIOC_S_CTRL,&ctrl);
|
|
ctrl.id = V4L2_CID_AUDIO_TREBLE;
|
|
ctrl.value = hdw->treble_val;
|
|
pvr2_i2c_core_cmd(hdw,VIDIOC_S_CTRL,&ctrl);
|
|
}
|
|
|
|
|
|
static int check_volume(struct pvr2_hdw *hdw)
|
|
{
|
|
return (hdw->volume_dirty ||
|
|
hdw->balance_dirty ||
|
|
hdw->bass_dirty ||
|
|
hdw->treble_dirty ||
|
|
hdw->mute_dirty);
|
|
}
|
|
|
|
|
|
const struct pvr2_i2c_op pvr2_i2c_op_v4l2_volume = {
|
|
.check = check_volume,
|
|
.update = set_volume,
|
|
.name = "v4l2_volume",
|
|
};
|
|
|
|
|
|
static void set_audiomode(struct pvr2_hdw *hdw)
|
|
{
|
|
struct v4l2_tuner vt;
|
|
memset(&vt,0,sizeof(vt));
|
|
vt.audmode = hdw->audiomode_val;
|
|
pvr2_i2c_core_cmd(hdw,VIDIOC_S_TUNER,&vt);
|
|
}
|
|
|
|
|
|
static int check_audiomode(struct pvr2_hdw *hdw)
|
|
{
|
|
return (hdw->input_dirty ||
|
|
hdw->audiomode_dirty);
|
|
}
|
|
|
|
|
|
const struct pvr2_i2c_op pvr2_i2c_op_v4l2_audiomode = {
|
|
.check = check_audiomode,
|
|
.update = set_audiomode,
|
|
.name = "v4l2_audiomode",
|
|
};
|
|
|
|
|
|
static void set_frequency(struct pvr2_hdw *hdw)
|
|
{
|
|
unsigned long fv;
|
|
struct v4l2_frequency freq;
|
|
fv = pvr2_hdw_get_cur_freq(hdw);
|
|
pvr2_trace(PVR2_TRACE_CHIPS,"i2c v4l2 set_freq(%lu)",fv);
|
|
if (hdw->tuner_signal_stale) {
|
|
pvr2_i2c_core_status_poll(hdw);
|
|
}
|
|
memset(&freq,0,sizeof(freq));
|
|
if (hdw->tuner_signal_info.capability & V4L2_TUNER_CAP_LOW) {
|
|
// ((fv * 1000) / 62500)
|
|
freq.frequency = (fv * 2) / 125;
|
|
} else {
|
|
freq.frequency = fv / 62500;
|
|
}
|
|
/* tuner-core currently doesn't seem to care about this, but
|
|
let's set it anyway for completeness. */
|
|
if (hdw->input_val == PVR2_CVAL_INPUT_RADIO) {
|
|
freq.type = V4L2_TUNER_RADIO;
|
|
} else {
|
|
freq.type = V4L2_TUNER_ANALOG_TV;
|
|
}
|
|
freq.tuner = 0;
|
|
pvr2_i2c_core_cmd(hdw,VIDIOC_S_FREQUENCY,&freq);
|
|
}
|
|
|
|
|
|
static int check_frequency(struct pvr2_hdw *hdw)
|
|
{
|
|
return hdw->freqDirty != 0;
|
|
}
|
|
|
|
|
|
const struct pvr2_i2c_op pvr2_i2c_op_v4l2_frequency = {
|
|
.check = check_frequency,
|
|
.update = set_frequency,
|
|
.name = "v4l2_freq",
|
|
};
|
|
|
|
|
|
static void set_size(struct pvr2_hdw *hdw)
|
|
{
|
|
struct v4l2_format fmt;
|
|
|
|
memset(&fmt,0,sizeof(fmt));
|
|
|
|
fmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
|
|
fmt.fmt.pix.width = hdw->res_hor_val;
|
|
fmt.fmt.pix.height = hdw->res_ver_val;
|
|
|
|
pvr2_trace(PVR2_TRACE_CHIPS,"i2c v4l2 set_size(%dx%d)",
|
|
fmt.fmt.pix.width,fmt.fmt.pix.height);
|
|
|
|
pvr2_i2c_core_cmd(hdw,VIDIOC_S_FMT,&fmt);
|
|
}
|
|
|
|
|
|
static int check_size(struct pvr2_hdw *hdw)
|
|
{
|
|
return (hdw->res_hor_dirty || hdw->res_ver_dirty);
|
|
}
|
|
|
|
|
|
const struct pvr2_i2c_op pvr2_i2c_op_v4l2_size = {
|
|
.check = check_size,
|
|
.update = set_size,
|
|
.name = "v4l2_size",
|
|
};
|
|
|
|
|
|
static void do_log(struct pvr2_hdw *hdw)
|
|
{
|
|
pvr2_trace(PVR2_TRACE_CHIPS,"i2c v4l2 do_log()");
|
|
pvr2_i2c_core_cmd(hdw,VIDIOC_LOG_STATUS,NULL);
|
|
|
|
}
|
|
|
|
|
|
static int check_log(struct pvr2_hdw *hdw)
|
|
{
|
|
return hdw->log_requested != 0;
|
|
}
|
|
|
|
|
|
const struct pvr2_i2c_op pvr2_i2c_op_v4l2_log = {
|
|
.check = check_log,
|
|
.update = do_log,
|
|
.name = "v4l2_log",
|
|
};
|
|
|
|
|
|
void pvr2_v4l2_cmd_stream(struct pvr2_i2c_client *cp,int fl)
|
|
{
|
|
pvr2_i2c_client_cmd(cp,
|
|
(fl ? VIDIOC_STREAMON : VIDIOC_STREAMOFF),NULL);
|
|
}
|
|
|
|
|
|
void pvr2_v4l2_cmd_status_poll(struct pvr2_i2c_client *cp)
|
|
{
|
|
pvr2_i2c_client_cmd(cp,VIDIOC_G_TUNER,&cp->hdw->tuner_signal_info);
|
|
}
|
|
|
|
|
|
/*
|
|
Stuff for Emacs to see, in order to encourage consistent editing style:
|
|
*** Local Variables: ***
|
|
*** mode: c ***
|
|
*** fill-column: 70 ***
|
|
*** tab-width: 8 ***
|
|
*** c-basic-offset: 8 ***
|
|
*** End: ***
|
|
*/
|