Commit bac196d5 authored by Christophe Massiot's avatar Christophe Massiot

* modules/access/pvr/pvr.c: Added support for radio on PVR150/350/500 and

   setting audio volume.
parent f436e5f7
...@@ -51,6 +51,9 @@ static void Close( vlc_object_t * ); ...@@ -51,6 +51,9 @@ static void Close( vlc_object_t * );
#define DEVICE_TEXT N_( "Device" ) #define DEVICE_TEXT N_( "Device" )
#define DEVICE_LONGTEXT N_( "PVR video device" ) #define DEVICE_LONGTEXT N_( "PVR video device" )
#define RADIO_DEVICE_TEXT N_( "Radio device" )
#define RADIO_DEVICE_LONGTEXT N_( "PVR radio device" )
#define NORM_TEXT N_( "Norm" ) #define NORM_TEXT N_( "Norm" )
#define NORM_LONGTEXT N_( "Defines the norm of the stream " \ #define NORM_LONGTEXT N_( "Defines the norm of the stream " \
"(Automatic, SECAM, PAL, or NTSC)" ) "(Automatic, SECAM, PAL, or NTSC)" )
...@@ -90,15 +93,14 @@ static void Close( vlc_object_t * ); ...@@ -90,15 +93,14 @@ static void Close( vlc_object_t * );
#define BITMASK_LONGTEXT N_("This option allows setting of bitmask that will "\ #define BITMASK_LONGTEXT N_("This option allows setting of bitmask that will "\
"get used by the audio part of the card." ) "get used by the audio part of the card." )
#define VOLUME_TEXT N_( "Volume" )
#define VOLUME_LONGTEXT N_("This option allows setting of the audio volume " \
"(0-65535)." )
#define CHAN_TEXT N_( "Channel" ) #define CHAN_TEXT N_( "Channel" )
#define CHAN_LONGTEXT N_( "Channel of the card to use (Usually, 0 = tuner, " \ #define CHAN_LONGTEXT N_( "Channel of the card to use (Usually, 0 = tuner, " \
"1 = composite, 2 = svideo)" ) "1 = composite, 2 = svideo)" )
#define CACHING_TEXT N_("Caching value in ms")
#define CACHING_LONGTEXT N_( \
"Allows you to modify the default caching value for pvr streams. This " \
"value should be set in millisecond units." )
static int i_norm_list[] = static int i_norm_list[] =
{ V4L2_STD_UNKNOWN, V4L2_STD_SECAM, V4L2_STD_PAL, V4L2_STD_NTSC }; { V4L2_STD_UNKNOWN, V4L2_STD_SECAM, V4L2_STD_PAL, V4L2_STD_NTSC };
static char *psz_norm_list_text[] = static char *psz_norm_list_text[] =
...@@ -107,6 +109,8 @@ static char *psz_norm_list_text[] = ...@@ -107,6 +109,8 @@ static char *psz_norm_list_text[] =
static int i_bitrates[] = { 0, 1 }; static int i_bitrates[] = { 0, 1 };
static char *psz_bitrates_list_text[] = { N_("vbr"), N_("cbr") }; static char *psz_bitrates_list_text[] = { N_("vbr"), N_("cbr") };
static int pi_radio_range[2] = { 65000, 108000 };
vlc_module_begin(); vlc_module_begin();
set_shortname( _("PVR") ); set_shortname( _("PVR") );
set_description( _("MPEG Encoding cards input (with ivtv drivers)") ); set_description( _("MPEG Encoding cards input (with ivtv drivers)") );
...@@ -118,6 +122,8 @@ vlc_module_begin(); ...@@ -118,6 +122,8 @@ vlc_module_begin();
add_integer( "pvr-caching", DEFAULT_PTS_DELAY / 1000, NULL, CACHING_TEXT, CACHING_LONGTEXT, VLC_TRUE ); add_integer( "pvr-caching", DEFAULT_PTS_DELAY / 1000, NULL, CACHING_TEXT, CACHING_LONGTEXT, VLC_TRUE );
add_string( "pvr-device", "/dev/video0", NULL, DEVICE_TEXT, add_string( "pvr-device", "/dev/video0", NULL, DEVICE_TEXT,
DEVICE_LONGTEXT, VLC_FALSE ); DEVICE_LONGTEXT, VLC_FALSE );
add_string( "pvr-radio-device", "/dev/radio0", NULL, RADIO_DEVICE_TEXT,
RADIO_DEVICE_LONGTEXT, VLC_FALSE );
add_integer( "pvr-norm", V4L2_STD_UNKNOWN , NULL, NORM_TEXT, add_integer( "pvr-norm", V4L2_STD_UNKNOWN , NULL, NORM_TEXT,
NORM_LONGTEXT, VLC_FALSE ); NORM_LONGTEXT, VLC_FALSE );
change_integer_list( i_norm_list, psz_norm_list_text, 0 ); change_integer_list( i_norm_list, psz_norm_list_text, 0 );
...@@ -141,9 +147,9 @@ vlc_module_begin(); ...@@ -141,9 +147,9 @@ vlc_module_begin();
change_integer_list( i_bitrates, psz_bitrates_list_text, 0 ); change_integer_list( i_bitrates, psz_bitrates_list_text, 0 );
add_integer( "pvr-audio-bitmask", -1, NULL, BITMASK_TEXT, add_integer( "pvr-audio-bitmask", -1, NULL, BITMASK_TEXT,
BITMASK_LONGTEXT, VLC_TRUE ); BITMASK_LONGTEXT, VLC_TRUE );
add_integer( "pvr-audio-volume", -1, NULL, VOLUME_TEXT,
VOLUME_LONGTEXT, VLC_TRUE );
add_integer( "pvr-channel", -1, NULL, CHAN_TEXT, CHAN_LONGTEXT, VLC_TRUE ); add_integer( "pvr-channel", -1, NULL, CHAN_TEXT, CHAN_LONGTEXT, VLC_TRUE );
add_integer( "pvr-caching", DEFAULT_PTS_DELAY / 1000, NULL,
CACHING_TEXT, CACHING_LONGTEXT, VLC_TRUE );
set_callbacks( Open, Close ); set_callbacks( Open, Close );
vlc_module_end(); vlc_module_end();
...@@ -182,6 +188,7 @@ struct access_sys_t ...@@ -182,6 +188,7 @@ struct access_sys_t
{ {
/* file descriptor */ /* file descriptor */
int i_fd; int i_fd;
int i_radio_fd;
/* options */ /* options */
int i_standard; int i_standard;
...@@ -196,6 +203,7 @@ struct access_sys_t ...@@ -196,6 +203,7 @@ struct access_sys_t
int i_bitrate_mode; int i_bitrate_mode;
int i_audio_bitmask; int i_audio_bitmask;
int i_input; int i_input;
int i_volume;
}; };
/***************************************************************************** /*****************************************************************************
...@@ -205,13 +213,9 @@ static int Open( vlc_object_t * p_this ) ...@@ -205,13 +213,9 @@ static int Open( vlc_object_t * p_this )
{ {
access_t *p_access = (access_t*) p_this; access_t *p_access = (access_t*) p_this;
access_sys_t * p_sys; access_sys_t * p_sys;
char * psz_tofree, * psz_parser, * psz_device; char * psz_tofree, * psz_parser, * psz_device, * psz_radio_device;
vlc_value_t val; vlc_value_t val;
struct v4l2_format vfmt;
struct v4l2_frequency vf;
struct ivtv_ioctl_codec codec;
//psz_device = calloc( strlen( "/dev/videox" ) + 1, 1 ); //psz_device = calloc( strlen( "/dev/videox" ) + 1, 1 );
p_access->pf_read = Read; p_access->pf_read = Read;
...@@ -236,6 +240,11 @@ static int Open( vlc_object_t * p_this ) ...@@ -236,6 +240,11 @@ static int Open( vlc_object_t * p_this )
var_Get( p_access, "pvr-device" , &val); var_Get( p_access, "pvr-device" , &val);
psz_device = val.psz_string; psz_device = val.psz_string;
var_Create( p_access, "pvr-radio-device", VLC_VAR_STRING |
VLC_VAR_DOINHERIT );
var_Get( p_access, "pvr-radio-device" , &val);
psz_radio_device = val.psz_string;
var_Create( p_access, "pvr-norm", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT ); var_Create( p_access, "pvr-norm", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT );
var_Get( p_access, "pvr-norm" , &val); var_Get( p_access, "pvr-norm" , &val);
p_sys->i_standard = val.i_int; p_sys->i_standard = val.i_int;
...@@ -282,6 +291,12 @@ static int Open( vlc_object_t * p_this ) ...@@ -282,6 +291,12 @@ static int Open( vlc_object_t * p_this )
VLC_VAR_DOINHERIT ); VLC_VAR_DOINHERIT );
var_Get( p_access, "pvr-audio-bitmask" , &val); var_Get( p_access, "pvr-audio-bitmask" , &val);
p_sys->i_audio_bitmask = val.i_int; p_sys->i_audio_bitmask = val.i_int;
var_Create( p_access, "pvr-audio-volume", VLC_VAR_INTEGER |
VLC_VAR_DOINHERIT );
var_Get( p_access, "pvr-audio-volume" , &val);
p_sys->i_volume = val.i_int;
var_Create( p_access, "pvr-channel", VLC_VAR_INTEGER | var_Create( p_access, "pvr-channel", VLC_VAR_INTEGER |
VLC_VAR_DOINHERIT ); VLC_VAR_DOINHERIT );
var_Get( p_access, "pvr-channel" , &val); var_Get( p_access, "pvr-channel" , &val);
...@@ -331,7 +346,7 @@ static int Open( vlc_object_t * p_this ) ...@@ -331,7 +346,7 @@ static int Open( vlc_object_t * p_this )
strlen( "channel=" ) ) ) strlen( "channel=" ) ) )
{ {
p_sys->i_input = p_sys->i_input =
strtol( psz_parser + strlen( "channel=" ), strtol( psz_parser + strlen( "channel=" ),
&psz_parser, 0 ); &psz_parser, 0 );
} }
else if( !strncmp( psz_parser, "device=", strlen( "device=" ) ) ) else if( !strncmp( psz_parser, "device=", strlen( "device=" ) ) )
...@@ -345,7 +360,7 @@ static int Open( vlc_object_t * p_this ) ...@@ -345,7 +360,7 @@ static int Open( vlc_object_t * p_this )
strlen( "frequency=" ) ) ) strlen( "frequency=" ) ) )
{ {
p_sys->i_frequency = p_sys->i_frequency =
strtol( psz_parser + strlen( "frequency=" ), strtol( psz_parser + strlen( "frequency=" ),
&psz_parser, 0 ); &psz_parser, 0 );
} }
else if( !strncmp( psz_parser, "framerate=", else if( !strncmp( psz_parser, "framerate=",
...@@ -457,9 +472,9 @@ static int Open( vlc_object_t * p_this ) ...@@ -457,9 +472,9 @@ static int Open( vlc_object_t * p_this )
} }
} }
//give a default value to psz_device if none has bee specified //give a default value to psz_device if none has been specified
if (!psz_device) if ( psz_device == NULL )
{ {
psz_device = calloc( strlen( "/dev/videox" ) + 1, 1 ); psz_device = calloc( strlen( "/dev/videox" ) + 1, 1 );
strcpy( psz_device, "/dev/video0" ); strcpy( psz_device, "/dev/video0" );
...@@ -471,6 +486,7 @@ static int Open( vlc_object_t * p_this ) ...@@ -471,6 +486,7 @@ static int Open( vlc_object_t * p_this )
if( ( p_sys->i_fd = open( psz_device, O_RDWR ) ) < 0 ) if( ( p_sys->i_fd = open( psz_device, O_RDWR ) ) < 0 )
{ {
msg_Err( p_access, "cannot open device (%s)", strerror( errno ) ); msg_Err( p_access, "cannot open device (%s)", strerror( errno ) );
free( p_sys );
return VLC_EGENERIC; return VLC_EGENERIC;
} }
else else
...@@ -489,7 +505,7 @@ static int Open( vlc_object_t * p_this ) ...@@ -489,7 +505,7 @@ static int Open( vlc_object_t * p_this )
} }
else else
{ {
msg_Dbg( p_access, "input set to:%d", p_sys->i_input); msg_Dbg( p_access, "input set to :%d", p_sys->i_input);
} }
} }
...@@ -502,13 +518,15 @@ static int Open( vlc_object_t * p_this ) ...@@ -502,13 +518,15 @@ static int Open( vlc_object_t * p_this )
} }
else else
{ {
msg_Dbg( p_access, "video standard set to:%x", p_sys->i_standard); msg_Dbg( p_access, "video standard set to :%x", p_sys->i_standard);
} }
} }
/* set the picture size */ /* set the picture size */
if ( p_sys->i_width != -1 || p_sys->i_height != -1 ) if ( p_sys->i_width != -1 || p_sys->i_height != -1 )
{ {
struct v4l2_format vfmt;
vfmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE; vfmt.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
if ( ioctl( p_sys->i_fd, VIDIOC_G_FMT, &vfmt ) < 0 ) if ( ioctl( p_sys->i_fd, VIDIOC_G_FMT, &vfmt ) < 0 )
{ {
...@@ -532,7 +550,7 @@ static int Open( vlc_object_t * p_this ) ...@@ -532,7 +550,7 @@ static int Open( vlc_object_t * p_this )
} }
else else
{ {
msg_Dbg( p_access, "picture size set to:%dx%d", msg_Dbg( p_access, "picture size set to :%dx%d",
vfmt.fmt.pix.width, vfmt.fmt.pix.height ); vfmt.fmt.pix.width, vfmt.fmt.pix.height );
} }
} }
...@@ -541,28 +559,68 @@ static int Open( vlc_object_t * p_this ) ...@@ -541,28 +559,68 @@ static int Open( vlc_object_t * p_this )
/* set the frequency */ /* set the frequency */
if ( p_sys->i_frequency != -1 ) if ( p_sys->i_frequency != -1 )
{ {
int i_fd;
struct v4l2_frequency vf;
vf.tuner = 0; /* TODO: let the user choose the tuner */ vf.tuner = 0; /* TODO: let the user choose the tuner */
if ( ioctl( p_sys->i_fd, VIDIOC_G_FREQUENCY, &vf ) < 0 )
if ( p_sys->i_frequency >= pi_radio_range[0]
&& p_sys->i_frequency <= pi_radio_range[1] )
{
if( ( p_sys->i_radio_fd = open( psz_radio_device, O_RDWR ) ) < 0 )
{
msg_Err( p_access, "cannot open radio device (%s)",
strerror( errno ) );
close( p_sys->i_fd );
free( p_sys );
return VLC_EGENERIC;
}
else
{
msg_Dbg( p_access, "using radio device: %s", psz_radio_device );
}
i_fd = p_sys->i_radio_fd;
}
else
{
i_fd = p_sys->i_fd;
p_sys->i_radio_fd = -1;
}
if ( ioctl( i_fd, VIDIOC_G_FREQUENCY, &vf ) < 0 )
{ {
msg_Warn( p_access, "VIDIOC_G_FREQUENCY failed (%s)", msg_Warn( p_access, "VIDIOC_G_FREQUENCY failed (%s)",
strerror( errno ) ); strerror( errno ) );
} }
else else
{ {
vf.frequency = p_sys->i_frequency * 16 / 1000; vf.frequency = (p_sys->i_frequency * 16 + 500) / 1000;
if( ioctl( p_sys->i_fd, VIDIOC_S_FREQUENCY, &vf ) < 0 ) if( ioctl( i_fd, VIDIOC_S_FREQUENCY, &vf ) < 0 )
{ {
msg_Warn( p_access, "VIDIOC_S_FREQUENCY failed (%s)", msg_Warn( p_access, "VIDIOC_S_FREQUENCY failed (%s)",
strerror( errno ) ); strerror( errno ) );
} }
else else
{ {
msg_Dbg( p_access, "Tuner frequency set to:%d", msg_Dbg( p_access, "tuner frequency set to :%d",
p_sys->i_frequency); p_sys->i_frequency );
} }
} }
} }
/* control parameters */
if ( p_sys->i_volume != -1 )
{
struct v4l2_control ctrl;
ctrl.id = V4L2_CID_AUDIO_VOLUME;
ctrl.value = p_sys->i_volume;
if ( ioctl( p_sys->i_fd, VIDIOC_S_CTRL, &ctrl ) < 0 )
{
msg_Warn( p_access, "VIDIOC_S_CTRL failed" );
}
}
/* codec parameters */ /* codec parameters */
if ( p_sys->i_framerate != -1 if ( p_sys->i_framerate != -1
|| p_sys->i_bitrate_mode != -1 || p_sys->i_bitrate_mode != -1
...@@ -572,6 +630,8 @@ static int Open( vlc_object_t * p_this ) ...@@ -572,6 +630,8 @@ static int Open( vlc_object_t * p_this )
|| p_sys->i_bitrate != -1 || p_sys->i_bitrate != -1
|| p_sys->i_audio_bitmask != -1 ) || p_sys->i_audio_bitmask != -1 )
{ {
struct ivtv_ioctl_codec codec;
if ( ioctl( p_sys->i_fd, IVTV_IOC_G_CODEC, &codec ) < 0 ) if ( ioctl( p_sys->i_fd, IVTV_IOC_G_CODEC, &codec ) < 0 )
{ {
msg_Warn( p_access, "IVTV_IOC_G_CODEC failed" ); msg_Warn( p_access, "IVTV_IOC_G_CODEC failed" );
...@@ -664,6 +724,8 @@ static void Close( vlc_object_t * p_this ) ...@@ -664,6 +724,8 @@ static void Close( vlc_object_t * p_this )
access_sys_t * p_sys = p_access->p_sys; access_sys_t * p_sys = p_access->p_sys;
close( p_sys->i_fd ); close( p_sys->i_fd );
if ( p_sys->i_radio_fd != -1 )
close( p_sys->i_radio_fd );
free( p_sys ); free( p_sys );
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment