changeset 23905:fb63124c7920

v4l2 audio/video outputs for linux 2.6.22+ kernels (outputs formerly known as ivtv)
author ben
date Sun, 29 Jul 2007 19:20:55 +0000
parents 545aef4edc84
children 4a50c3a91689
files configure etc/codecs.conf libao2/ao_v4l2.c libao2/audio_out.c libvo/video_out.c libvo/vo_v4l2.c
diffstat 6 files changed, 472 insertions(+), 1 deletions(-) [+]
line wrap: on
line diff
--- a/configure	Sun Jul 29 17:58:20 2007 +0000
+++ b/configure	Sun Jul 29 19:20:55 2007 +0000
@@ -358,6 +358,7 @@
   --enable-dxr2            enable DXR2 video output [autodetect]
   --enable-dxr3            enable DXR3/H+ video output [autodetect]
   --enable-ivtv            enable IVTV TV-Out video output [autodetect]
+  --enable-v4l2            enable V4L2 Decoder audio/video output [autodetect]
   --enable-dvb             enable DVB video output [autodetect]
   --enable-dvbhead         enable DVB video output (HEAD version) [autodetect]
   --enable-mga             enable mga_vid video output [disable]
@@ -539,6 +540,7 @@
 _dxr2=auto
 _dxr3=auto
 _ivtv=auto
+_v4l2=auto
 _iconv=auto
 _langinfo=auto
 _rtc=auto
@@ -846,6 +848,8 @@
   --disable-dxr3)	_dxr3=no	;;
   --enable-ivtv)        _ivtv=yes       ;;
   --disable-ivtv)       _ivtv=no        ;;
+  --enable-v4l2)        _v4l2=yes       ;;
+  --disable-v4l2)       _v4l2=no        ;;
   --enable-iconv)	_iconv=yes	;;
   --disable-iconv)	_iconv=no	;;
   --enable-langinfo)	_langinfo=yes	;;
@@ -4823,6 +4827,39 @@
 echores "$_ivtv"
 
 
+echocheck "V4L2 MPEG Decoder"
+if test "$_v4l2" = auto ; then
+  cat > $TMPC << EOF
+#include <stdlib.h>
+#include <inttypes.h>
+#include <linux/types.h>
+#include <linux/videodev2.h>
+#include <linux/version.h>
+int main(void) {
+#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,6,22)
+  return 0;
+#else
+  return -1;
+#endif
+}
+EOF
+  _v4l2=no
+  cc_check && _v4l2=yes
+fi
+if test "$_v4l2" = yes ; then
+  _def_v4l2='#define HAVE_V4L2_DECODER 1'
+  _vosrc="$_vosrc vo_v4l2.c"
+  _vomodules="v4l2 $_vomodules"
+  _aosrc="$_aosrc ao_v4l2.c"
+  _aomodules="v4l2 $_aomodules"
+else
+  _def_v4l2='#undef HAVE_V4L2_DECODER'
+  _novomodules="v4l2 $_novomodules"
+  _noaomodules="v4l2 $_noaomodules"
+fi
+echores "$_v4l2"
+
+
 
 #########
 # AUDIO #
@@ -8241,6 +8278,7 @@
 $_def_dxr2
 $_def_dxr3
 $_def_ivtv
+$_def_v4l2
 $_def_dvb
 $_def_dvb_in
 $_def_svga
--- a/etc/codecs.conf	Sun Jul 29 17:58:20 2007 +0000
+++ b/etc/codecs.conf	Sun Jul 29 19:20:55 2007 +0000
@@ -35,7 +35,7 @@
 ; Note: mpegpes is preferred for hw decoders:
 
 videocodec mpegpes
-  info "MPEG-PES output (.mpg or DXR3/IVTV/DVB card)"
+  info "MPEG-PES output (.mpg or DXR3/IVTV/DVB/V4L2 card)"
   comment "for hardware decoding"
   status working
   format 0x10000001  ; mpeg 1
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/libao2/ao_v4l2.c	Sun Jul 29 19:20:55 2007 +0000
@@ -0,0 +1,159 @@
+/*
+ *  Copyright (C) 2007 Benjamin Zores
+ *   Audio output for V4L2 hardware MPEG decoders.
+ *
+ *   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
+ *
+ *  WARNING: you need to force -ac hwmpa for audio output to work.
+ */
+
+#include <inttypes.h>
+
+#include "config.h"
+
+#include "mp_msg.h"
+#include "help_mp.h"
+
+#include "audio_out.h"
+#include "audio_out_internal.h"
+#include "libaf/af_format.h"
+#include "libmpdemux/mpeg_packetizer.h"
+
+#define MPEG_AUDIO_ID 0x1C0
+
+static int freq = 0;
+
+static ao_info_t info = 
+{
+  "V4L2 MPEG Audio Decoder output",
+  "v4l2",
+  "Benjamin Zores",
+  ""
+};
+
+LIBAO_EXTERN (v4l2)
+
+/* to set/get/query special features/parameters */
+static int
+control (int cmd,void *arg)
+{
+  return CONTROL_UNKNOWN;
+}
+
+/* open & setup audio device */
+static int
+init (int rate, int channels, int format, int flags)
+{
+  extern int v4l2_fd;
+
+  if (v4l2_fd < 0)
+    return 0;
+
+  if (format != AF_FORMAT_MPEG2)
+  {
+    mp_msg (MSGT_AO, MSGL_FATAL,
+            "AO: [v4l2] can only handle MPEG audio streams.\n");
+    return 0;
+  }
+  
+  ao_data.outburst = 2048;
+  ao_data.samplerate = rate;
+  ao_data.channels = channels;
+  ao_data.format = AF_FORMAT_MPEG2;
+  ao_data.buffersize = 2048;
+  ao_data.bps = rate * 2 * 2;
+  ao_data.pts = 0;
+  freq = rate;
+
+  /* check for supported audio rate */
+  if (rate != 32000 || rate != 41000 || rate != 48000)
+  {
+    mp_msg (MSGT_AO, MSGL_ERR, MSGTR_AO_MPEGPES_UnsupSamplerate, rate);
+    rate = 48000;
+  }
+
+  return 1;
+}
+
+/* close audio device */
+static void
+uninit (int immed)
+{
+  /* nothing to do */
+}
+
+/* stop playing and empty buffers (for seeking/pause) */
+static void
+reset (void)
+{
+  /* nothing to do */
+}
+
+/* stop playing, keep buffers (for pause) */
+static void
+audio_pause (void)
+{
+  reset ();
+}
+
+/* resume playing, after audio_pause() */
+static void
+audio_resume (void)
+{
+  /* nothing to do */
+}
+
+/* how many bytes can be played without blocking */
+static int
+get_space (void)
+{
+  extern int vo_pts;
+  float x;
+  int y;
+
+  x = (float) (vo_pts - ao_data.pts) / 90000.0;
+  if (x <= 0)
+    return 0;
+  
+  y  = freq * 4 * x;
+  y /= ao_data.outburst;
+  y *= ao_data.outburst;
+  
+  if (y > 32000)
+    y = 32000;
+
+  return y;
+}
+
+/* number of bytes played */
+static int
+play (void *data, int len, int flags)
+{
+  extern int v4l2_write (unsigned char *data, int len);
+  
+  if (ao_data.format != AF_FORMAT_MPEG2)
+    return 0;
+
+  send_mpeg_pes_packet (data, len, MPEG_AUDIO_ID, ao_data.pts, 2, v4l2_write);
+
+  return len;
+}
+
+/* delay in seconds between first and last sample in buffer */
+static float
+get_delay (void)
+{
+  return 0.0;
+}
--- a/libao2/audio_out.c	Sun Jul 29 17:58:20 2007 +0000
+++ b/libao2/audio_out.c	Sun Jul 29 19:20:55 2007 +0000
@@ -68,6 +68,9 @@
 #ifdef HAVE_IVTV
 extern ao_functions_t audio_out_ivtv;
 #endif
+#ifdef HAVE_V4L2_DECODER
+extern ao_functions_t audio_out_v4l2;
+#endif
 extern ao_functions_t audio_out_mpegpes;
 extern ao_functions_t audio_out_pcm;
 extern ao_functions_t audio_out_pss;
@@ -131,6 +134,9 @@
 #ifdef HAVE_IVTV
         &audio_out_ivtv,
 #endif
+#ifdef HAVE_V4L2_DECODER
+        &audio_out_v4l2,
+#endif
         &audio_out_null,
 // should not be auto-selected:
 	&audio_out_pcm,
--- a/libvo/video_out.c	Sun Jul 29 17:58:20 2007 +0000
+++ b/libvo/video_out.c	Sun Jul 29 19:20:55 2007 +0000
@@ -103,6 +103,9 @@
 #ifdef HAVE_IVTV
 extern vo_functions_t video_out_ivtv;
 #endif
+#ifdef HAVE_V4L2_DECODER
+extern vo_functions_t video_out_v4l2;
+#endif
 #ifdef HAVE_JPEG
 extern vo_functions_t video_out_jpeg;
 #endif
@@ -214,6 +217,9 @@
 #ifdef HAVE_IVTV
         &video_out_ivtv,
 #endif
+#ifdef HAVE_V4L2_DECODER
+        &video_out_v4l2,
+#endif
 #ifdef HAVE_ZR
 	&video_out_zr,
 	&video_out_zr2,
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/libvo/vo_v4l2.c	Sun Jul 29 19:20:55 2007 +0000
@@ -0,0 +1,262 @@
+/*
+ *  Copyright (C) 2007 Benjamin Zores
+ *   Video output for V4L2 hardware MPEG decoders.
+ *
+ *   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
+ */
+
+#include "config.h"
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <errno.h>
+#include <sys/types.h>
+#include <sys/stat.h>
+#include <sys/ioctl.h>
+#include <fcntl.h>
+#include <unistd.h>
+#include <stdlib.h>
+#include <inttypes.h>
+#include <linux/types.h>
+#include <linux/videodev2.h>
+#include <linux/ioctl.h>
+
+#include "mp_msg.h"
+#include "subopt-helper.h"
+#include "video_out.h"
+#include "video_out_internal.h"
+#include "libmpdemux/mpeg_packetizer.h"
+
+#define DEFAULT_MPEG_DECODER "/dev/video16"
+#define V4L2_VO_HDR "VO: [v4l2]"
+
+int v4l2_fd = -1;
+static vo_mpegpes_t *pes;
+
+/* suboptions */
+static int output = -1;
+static char *device = NULL;
+
+static opt_t subopts[] = {
+  {"output",   OPT_ARG_INT,       &output,       (opt_test_f)int_non_neg},
+  {"device",   OPT_ARG_MSTRZ,     &device,       NULL},
+  {NULL}
+};
+
+static vo_info_t info = 
+{
+  "V4L2 MPEG Video Decoder Output",
+  "v4l2",
+  "Benjamin Zores",
+  ""
+};
+LIBVO_EXTERN (v4l2)
+
+int
+v4l2_write (unsigned char *data, int len)
+{
+  if (v4l2_fd < 0)
+    return 0;
+  
+  return write (v4l2_fd, data, len);
+}
+
+/* video out functions */
+
+static int
+config (uint32_t width, uint32_t height,
+        uint32_t d_width, uint32_t d_height,
+        uint32_t fullscreen, char *title, uint32_t format)
+{
+  return 0;
+}
+
+static int
+preinit (const char *arg)
+{
+  struct v4l2_output vout;
+  struct v4l2_ext_controls ctrls;
+  int err;
+
+  if (subopt_parse (arg, subopts) != 0)
+  {
+    mp_msg (MSGT_VO, MSGL_FATAL,
+            "\n-vo v4l2 command line help:\n"
+            "Example: mplayer -vo v4l2:device=/dev/video16:output=2\n"
+            "\nOptions:\n"
+            "  device=/dev/videoX\n"
+            "    Name of the MPEG decoder device file.\n"
+            "  output=<0-...>\n"
+            "    V4L2 id of the TV output.\n"
+            "\n" );
+    return -1;
+  }
+  
+  if (!device)
+    device = strdup (DEFAULT_MPEG_DECODER);    
+  
+  v4l2_fd = open (device, O_RDWR);
+  if (v4l2_fd < 0)
+  {  
+    free (device);
+    mp_msg (MSGT_VO, MSGL_FATAL, "%s %s\n", V4L2_VO_HDR, strerror (errno));
+    return -1;
+  }
+
+  /* check for device hardware MPEG decoding capability */
+  ctrls.ctrl_class = V4L2_CTRL_CLASS_MPEG; 
+  ctrls.count = 0; 
+  ctrls.controls = NULL;
+  
+  if (ioctl (v4l2_fd, VIDIOC_G_EXT_CTRLS, &ctrls) < 0)
+  {
+    free (device);
+    mp_msg (MSGT_OPEN, MSGL_FATAL, "%s %s\n", V4L2_VO_HDR, strerror (errno));
+    return -1;
+  }
+  
+  /* list available outputs */
+  vout.index = 0;
+  err = 1;
+  mp_msg (MSGT_VO, MSGL_INFO, "%s Available video outputs: ", V4L2_VO_HDR);
+  while (ioctl (v4l2_fd, VIDIOC_ENUMOUTPUT, &vout) >= 0)
+  {
+    err = 0;
+    mp_msg (MSGT_VO, MSGL_INFO, "'#%d, %s' ", vout.index, vout.name);
+    vout.index++;
+  }
+  if (err)
+  {
+    mp_msg (MSGT_VO, MSGL_INFO, "none\n");
+    free (device);
+    return -1;
+  }
+  else
+    mp_msg (MSGT_VO, MSGL_INFO, "\n");
+
+  /* set user specified output */
+  if (output != -1)
+  {
+    if (ioctl (v4l2_fd, VIDIOC_S_OUTPUT, &output) < 0)
+    {
+      mp_msg (MSGT_VO, MSGL_ERR,
+              "%s can't set output (%s)\n", V4L2_VO_HDR, strerror (errno));
+      free (device);
+      return -1;
+    }
+  }
+
+  /* display device name */
+  mp_msg (MSGT_VO, MSGL_INFO, "%s using %s\n", V4L2_VO_HDR, device);
+  free (device);
+
+  /* display current video output */
+  if (ioctl (v4l2_fd, VIDIOC_G_OUTPUT, &output) == 0)
+  {
+    vout.index = output;
+    if (ioctl (v4l2_fd, VIDIOC_ENUMOUTPUT, &vout) < 0)
+    {
+      mp_msg (MSGT_VO, MSGL_ERR,
+              "%s can't get output (%s).\n", V4L2_VO_HDR, strerror (errno));
+      return -1;
+    }
+    else
+      mp_msg (MSGT_VO, MSGL_INFO,
+              "%s video output: %s\n", V4L2_VO_HDR, vout.name);
+  }
+  else
+  {
+    mp_msg (MSGT_VO, MSGL_ERR,
+            "%s can't get output (%s).\n", V4L2_VO_HDR, strerror (errno));
+    return -1;
+  }
+  
+  return 0;
+}
+
+static void
+draw_osd (void)
+{
+  /* do nothing */
+}
+
+static int
+draw_frame (uint8_t * src[])
+{
+  pes = (vo_mpegpes_t *) src[0];
+  return 0;
+}
+
+static void
+flip_page (void)
+{
+  if (v4l2_fd < 0)
+    return;
+
+  if (!pes)
+    return;
+
+  send_mpeg_pes_packet (pes->data, pes->size, pes->id,
+                        pes->timestamp ? pes->timestamp : vo_pts, 2,
+                        v4l2_write);
+
+  /* ensure flip_page() won't be called twice */
+  pes = NULL;
+}
+
+static int
+draw_slice (uint8_t *image[], int stride[], int w, int h, int x, int y)
+{
+  return 0;
+}
+
+static void
+uninit (void)
+{
+  if (v4l2_fd < 0)
+    return;
+
+  /* close device */
+  close (v4l2_fd);
+  v4l2_fd = -1;
+}
+
+static void
+check_events (void)
+{
+  /* do nothing */
+}
+
+static int
+query_format (uint32_t format)
+{
+  if (format != IMGFMT_MPEGPES)
+    return 0;
+    
+  return VFCAP_CSP_SUPPORTED | VFCAP_CSP_SUPPORTED_BY_HW | VFCAP_TIMER;
+}
+
+static int
+control (uint32_t request, void *data, ...)
+{
+  switch (request)
+  {
+  case VOCTRL_QUERY_FORMAT:
+    return query_format (*((uint32_t*) data));
+  }
+  
+  return VO_NOTIMPL;
+}