1
mirror of https://code.videolan.org/videolan/vlc synced 2024-09-04 09:11:33 +02:00
vlc/modules/access/v4l2.c

442 lines
14 KiB
C
Raw Normal View History

/*****************************************************************************
* v4l2.c : Video4Linux2 input module for vlc
*****************************************************************************
* Copyright (C) 2002-2004 the VideoLAN team
* $Id: v4l.c 16084 2006-07-19 09:45:02Z zorglub $
*
* Author: Benjamin Pracht <bigben at videolan dot org>
*
* 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, or
* (at your option) any later version.
*
* 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., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA. *****************************************************************************/
/*****************************************************************************
* Preamble
*****************************************************************************/
#include <vlc/vlc.h>
2006-11-26 15:59:36 +01:00
#include <vlc_access.h>
#include <vlc_demux.h>
2006-11-26 15:59:36 +01:00
#include <vlc_input.h>
#include <vlc_vout.h>
#include <fcntl.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/time.h>
#include <sys/ioctl.h>
#include <sys/errno.h>
#include <asm/types.h> /* for videodev2.h */
#include <linux/videodev2.h>
/*****************************************************************************
* Module descriptior
*****************************************************************************/
static int Open ( vlc_object_t * );
static void Close( vlc_object_t * );
#define DEV_TEXT N_("Device name")
#define DEV_LONGTEXT N_( \
"Name of the device to use. " \
"If you don't specify anything, /dev/video0 will be used.")
#define INPUT_TEXT N_( "Input" )
#define INPUT_LONGTEXT N_( \
"Input of the card to use (Usually, 0 = tuner, " \
"1 = composite, 2 = svideo)." )
vlc_module_begin();
set_shortname( _("Video4Linux2") );
set_description( _("Video4Linux2 input") );
set_category( CAT_INPUT );
set_subcategory( SUBCAT_INPUT_ACCESS );
add_string( "v4l2-dev", "/dev/video0", 0, DEV_TEXT, DEV_LONGTEXT,
VLC_FALSE );
add_integer( "v4l2-input", 0, NULL, INPUT_TEXT, INPUT_LONGTEXT,
VLC_TRUE );
add_shortcut( "v4l2" );
set_capability( "access_demux", 10 );
set_callbacks( Open, Close );
vlc_module_end();
/*****************************************************************************
* Access: local prototypes
*****************************************************************************/
static int DemuxMMAP( demux_t * );
static int Control( demux_t *, int, va_list );
static int ProbeDev( demux_t * );
static int OpenVideoDev( demux_t * );
struct demux_sys_t
{
char *psz_device;
int i_fd_video;
struct v4l2_capability dev_cap;
int i_input;
struct v4l2_input *p_inputs;
int i_selected_input;
int i_audio;
/* V4L2 devices cannot have more than 32 audio inputs */
struct v4l2_audio p_audios[32];
int i_tuner;
struct v4l2_tuner *p_tuners;
int i_codec;
struct v4l2_fmtdesc *p_codecs;
};
/*****************************************************************************
* Open: opens v4l device
*****************************************************************************
*
* url: <video device>::::
*
*****************************************************************************/
static int Open( vlc_object_t *p_this )
{
demux_t *p_demux = (demux_t*)p_this;
demux_sys_t *p_sys;
/* Only when selected */
if( *p_demux->psz_access == '\0' )
return VLC_EGENERIC;
/* Set up p_demux */
p_demux->pf_control = Control;
p_demux->info.i_update = 0;
p_demux->info.i_title = 0;
p_demux->info.i_seekpoint = 0;
p_demux->p_sys = p_sys = malloc( sizeof( demux_sys_t ) );
if( p_sys == NULL ) return VLC_ENOMEM;
memset( p_sys, 0, sizeof( demux_sys_t ) );
p_sys->psz_device = var_CreateGetString( p_demux, "v4l2-dev" );
if( ProbeDev( p_demux ) < 0 ) return VLC_EGENERIC;
if( OpenVideoDev( p_demux ) < 0 ) return VLC_EGENERIC;
return VLC_SUCCESS;
}
/*****************************************************************************
* Close: close device, free resources
*****************************************************************************/
static void Close( vlc_object_t *p_this )
{
demux_t *p_demux = (demux_t *)p_this;
demux_sys_t *p_sys = p_demux->p_sys;
if( p_sys->i_fd_video >= 0 ) close( p_sys->i_fd_video );
if( p_sys->psz_device ) free( p_sys->psz_device );
if( p_sys->p_inputs ) free( p_sys->p_inputs );
if( p_sys->p_tuners ) free( p_sys->p_tuners );
if( p_sys->p_codecs ) free( p_sys->p_codecs );
free( p_sys );
}
/*****************************************************************************
* Control:
*****************************************************************************/
static int Control( demux_t *p_demux, int i_query, va_list args )
{
2007-05-27 18:50:56 +02:00
return VLC_EGENERIC;
}
/*****************************************************************************
* Demux:
*****************************************************************************/
static int DemuxMMAP( demux_t *p_demux )
{
2007-05-27 18:50:56 +02:00
msleep( 40000 );
return 1;
}
/*****************************************************************************
* OpenVideoDev: open and set up the video device and probe for capabilities
*****************************************************************************/
int OpenVideoDev( demux_t *p_demux )
{
int i_fd;
demux_sys_t *p_sys = p_demux->p_sys;
if( ( i_fd = open( p_sys->psz_device, O_RDWR ) ) < 0 )
{
msg_Err( p_demux, "cannot open device (%m)" );
goto open_failed;
}
p_sys->i_fd_video = i_fd;
2007-05-27 18:50:56 +02:00
p_sys->i_selected_input = var_CreateGetInteger( p_demux, "v4l2-input" );
2007-05-27 18:50:56 +02:00
if( p_sys->i_selected_input > p_sys->i_input )
{
msg_Warn( p_demux, "invalid input. Using the default one" );
p_sys->i_selected_input = 0;
}
if( ioctl( i_fd, VIDIOC_S_INPUT, &p_sys->i_selected_input ) < 0 )
{
msg_Err( p_demux, "cannot set input (%m)" );
goto open_failed;
}
if( p_sys->dev_cap.capabilities & V4L2_CAP_STREAMING )
{
struct v4l2_requestbuffers reqbuf;
reqbuf.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
reqbuf.memory = V4L2_MEMORY_MMAP;
reqbuf.count = 0;
if( ioctl( i_fd, VIDIOC_REQBUFS, &reqbuf ) < 0 )
{
msg_Err( p_demux, "cannot initiate I/O operation (%m). "
"Only MMAP is supported at the moment" );
goto open_failed;
}
p_demux->pf_demux = DemuxMMAP;
}
else
{
msg_Warn( p_demux, "I/O method not supported at the moment" );
goto open_failed;
}
return VLC_SUCCESS;
open_failed:
if( i_fd ) close( i_fd );
p_sys->i_fd_video = 0;
return VLC_EGENERIC;
}
/*****************************************************************************
* ProbeDev: probe for capabilities
*****************************************************************************/
int ProbeDev( demux_t *p_demux )
{
int i_index;
int i_fd;
demux_sys_t *p_sys = p_demux->p_sys;
if( ( i_fd = open( p_sys->psz_device, O_RDWR ) ) < 0 )
{
msg_Err( p_demux, "cannot open device (%m)" );
goto open_failed;
}
/* Get device capabilites */
if( ioctl( i_fd, VIDIOC_QUERYCAP, &p_sys->dev_cap ) < 0 )
{
msg_Err( p_demux, "cannot get capabilities (%m)" );
goto open_failed;
}
msg_Dbg( p_demux, "V4L2 device: %s using driver: %s (version: %u.%u.%u) on %s",
p_sys->dev_cap.card,
p_sys->dev_cap.driver,
(p_sys->dev_cap.version >> 16) & 0xFF,
(p_sys->dev_cap.version >> 8) & 0xFF,
p_sys->dev_cap.version & 0xFF,
p_sys->dev_cap.bus_info );
msg_Dbg( p_demux, "the device has the capabilities: (%c) Video Capure, "
"(%c) Audio, "
"(%c) Tuner",
( p_sys->dev_cap.capabilities & V4L2_CAP_VIDEO_CAPTURE ? 'X':' '),
( p_sys->dev_cap.capabilities & V4L2_CAP_AUDIO ? 'X':' '),
( p_sys->dev_cap.capabilities & V4L2_CAP_TUNER ? 'X':' ') );
2006-08-24 08:39:20 +02:00
msg_Dbg( p_demux, "supported I/O methods are: (%c) Read/Write, "
2006-08-24 08:39:20 +02:00
"(%c) Streaming, "
"(%c) Asynchronous",
( p_sys->dev_cap.capabilities & V4L2_CAP_READWRITE ? 'X':' ' ),
( p_sys->dev_cap.capabilities & V4L2_CAP_STREAMING ? 'X':' ' ),
( p_sys->dev_cap.capabilities & V4L2_CAP_ASYNCIO ? 'X':' ' ) );
/* Now, enumerate all the video inputs. This is useless at the moment
since we have no way to present that info to the user except with
debug messages */
if( p_sys->dev_cap.capabilities & V4L2_CAP_VIDEO_CAPTURE )
{
while( ioctl( i_fd, VIDIOC_S_INPUT, &p_sys->i_input ) >= 0 )
{
p_sys->i_input++;
}
p_sys->p_inputs = malloc( p_sys->i_input * sizeof( struct v4l2_input ) );
if( !p_sys->p_inputs ) goto open_failed;
memset( p_sys->p_inputs, 0, sizeof( struct v4l2_input ) );
for( i_index = 0; i_index < p_sys->i_input; i_index++ )
{
p_sys->p_inputs[i_index].index = i_index;
if( ioctl( i_fd, VIDIOC_ENUMINPUT, &p_sys->p_inputs[i_index] ) )
{
msg_Err( p_demux,
"cannot get video input characteristics (%m)" );
goto open_failed;
}
msg_Dbg( p_demux, "video input %i (%s) has type: %s",
i_index,
p_sys->p_inputs[i_index].name,
p_sys->p_inputs[i_index].type
== V4L2_INPUT_TYPE_TUNER ?
"Tuner adapter" :
"External analog input" );
}
}
/* initialize the structures for the ioctls */
for( i_index = 0; i_index < 32; i_index++ )
{
p_sys->p_audios[i_index].index = i_index;
}
/* Probe audio inputs */
if( p_sys->dev_cap.capabilities & V4L2_CAP_AUDIO )
{
while( p_sys->i_audio < 32 &&
ioctl( i_fd, VIDIOC_S_AUDIO, &p_sys->p_audios[p_sys->i_audio] ) >= 0 )
{
if( ioctl( i_fd, VIDIOC_G_AUDIO, &p_sys->p_audios[ p_sys->i_audio] ) < 0 )
{
msg_Err( p_demux,
"cannot get video input characteristics (%m)" );
goto open_failed;
}
msg_Dbg( p_demux, "audio device %i (%s) is %s",
p_sys->i_audio,
p_sys->p_audios[p_sys->i_audio].name,
p_sys->p_audios[p_sys->i_audio].capability &
V4L2_AUDCAP_STEREO ?
"Stereo" : "Mono" );
p_sys->i_audio++;
}
}
if( p_sys->dev_cap.capabilities & V4L2_CAP_TUNER )
{
2007-05-27 18:50:56 +02:00
struct v4l2_tuner tuner;
2007-05-27 18:50:56 +02:00
memset( &tuner, 0, sizeof(tuner) );
while( ioctl( i_fd, VIDIOC_S_TUNER, &tuner ) >= 0 )
{
p_sys->i_tuner++;
tuner.index = p_sys->i_tuner;
}
p_sys->p_tuners = malloc( p_sys->i_tuner * sizeof( struct v4l2_tuner ) );
if( !p_sys->p_tuners ) goto open_failed;
memset( p_sys->p_tuners, 0, sizeof( struct v4l2_tuner ) );
for( i_index = 0; i_index < p_sys->i_tuner; i_index++ )
{
p_sys->p_tuners[i_index].index = i_index;
if( ioctl( i_fd, VIDIOC_G_TUNER, &p_sys->p_tuners[i_index] ) )
{
msg_Err( p_demux, "cannot get tuner characteristics (%m)" );
goto open_failed;
}
msg_Dbg( p_demux, "tuner %i (%s) has type: %s, "
"frequency range: %.1f %s -> %.1f %s",
i_index,
p_sys->p_tuners[i_index].name,
p_sys->p_tuners[i_index].type
== V4L2_TUNER_RADIO ?
"Radio" : "Analog TV",
p_sys->p_tuners[i_index].rangelow * 62.5,
p_sys->p_tuners[i_index].capability &
V4L2_TUNER_CAP_LOW ?
"Hz" : "kHz",
p_sys->p_tuners[i_index].rangehigh * 62.5,
p_sys->p_tuners[i_index].capability &
V4L2_TUNER_CAP_LOW ?
"Hz" : "kHz" );
}
}
/* Probe for available chromas */
if( p_sys->dev_cap.capabilities & V4L2_CAP_VIDEO_CAPTURE )
{
2007-05-27 18:50:56 +02:00
struct v4l2_fmtdesc codec;
i_index = 0;
2007-05-27 18:50:56 +02:00
memset( &codec, 0, sizeof(codec) );
codec.index = i_index;
codec.type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
while( ioctl( i_fd, VIDIOC_ENUM_FMT, &codec ) >= 0 )
{
i_index++;
codec.index = i_index;
}
p_sys->i_codec = i_index;
p_sys->p_codecs = malloc( p_sys->i_codec * sizeof( struct v4l2_fmtdesc ) );
2007-09-10 20:56:52 +02:00
memset( p_sys->p_codecs, 0, p_sys->i_codec * sizeof( struct v4l2_fmtdesc ) );
for( i_index = 0; i_index < p_sys->i_codec; i_index++ )
{
p_sys->p_codecs[i_index].index = i_index;
p_sys->p_codecs[i_index].type = V4L2_BUF_TYPE_VIDEO_CAPTURE;
if( ioctl( i_fd, VIDIOC_ENUM_FMT, &p_sys->p_codecs[i_index] ) < 0 )
{
msg_Err( p_demux, "cannot get codec description (%m)" );
goto open_failed;
}
msg_Dbg( p_demux, "device supports Codec %s",
p_sys->p_codecs[i_index].description );
}
}
if( i_fd >= 0 ) close( i_fd );
return VLC_SUCCESS;
open_failed:
2007-09-10 20:56:52 +02:00
if( i_fd >= 0 ) close( i_fd );
return VLC_EGENERIC;
}