]> git.sesse.net Git - vlc/commitdiff
vdpau: basic deinterlacing filter
authorRémi Denis-Courmont <remi@remlab.net>
Thu, 27 Jun 2013 19:16:32 +0000 (22:16 +0300)
committerRémi Denis-Courmont <remi@remlab.net>
Sun, 7 Jul 2013 18:12:01 +0000 (21:12 +0300)
modules/LIST
modules/hw/vdpau/Makefile.am
modules/hw/vdpau/chroma.c
modules/hw/vdpau/deinterlace.c [new file with mode: 0644]
modules/hw/vdpau/picture.c
modules/hw/vdpau/vlc_vdpau.h

index 18a2b0ef1f936ab4324536d21cd52d22d01f2b45..a0506548553b9f7756c392e2b4b7cf2c6796e9a9 100644 (file)
@@ -364,6 +364,7 @@ $Id$
  * vda: VDADecoder hardware-accelerated decoding
  * vdpau_avcodec: VDPAU hardware-accelerated decoding
  * vdpau_chroma: VDPAU hardware surfaces conversion and rendering
+ * vdpau_deinterlace: VDPAU deinterlacing video filter
  * vdpau_display: VDPAU video display
  * vdummy: dummy video display
  * visual: visualisation system
index f81370ccbbe40a39bb117c7da87ec989fb00dcd6..ecb2c2112516ef754f5060cd6b103e5bdb5a7896 100644 (file)
@@ -26,6 +26,11 @@ if HAVE_AVCODEC_VDPAU
 libvlc_LTLIBRARIES += libvdpau_avcodec_plugin.la
 endif
 
+libvdpau_deinterlace_plugin_la_SOURCES = deinterlace.c picture.c
+libvdpau_deinterlace_plugin_la_CFLAGS = $(AM_CFLAGS) # dummy
+libvdpau_deinterlace_plugin_la_LIBADD = $(AM_LIBADD)
+libvlc_LTLIBRARIES += libvdpau_deinterlace_plugin.la
+
 libvdpau_chroma_plugin_la_SOURCES = chroma.c picture.c
 libvdpau_chroma_plugin_la_CFLAGS = $(AM_CFLAGS) # dummy
 libvdpau_chroma_plugin_la_LIBADD = $(AM_LIBADD)
index 99e9b61973484ef027b33903b6c47292269fbe0c..7bc68609aeeb8890da3290504e70b28cb7ff1796 100644 (file)
@@ -288,6 +288,8 @@ static picture_t *MixerRender(filter_t *filter, picture_t *src)
     picture_CopyProperties(dst, src);
 
     /* Render video into output */
+    VdpVideoMixerPictureStructure structure =
+        ((vlc_vdp_video_field_t *)(src->context))->structure;
     VdpVideoSurface past[MAX_PAST];
     VdpVideoSurface surface = picture_GetVideoSurface(src);
     VdpVideoSurface future[MAX_FUTURE];
@@ -322,7 +324,7 @@ static picture_t *MixerRender(filter_t *filter, picture_t *src)
     }
 
     err = vdp_video_mixer_render(sys->vdp, sys->mixer, VDP_INVALID_HANDLE,
-                                 NULL, VDP_VIDEO_MIXER_PICTURE_STRUCTURE_FRAME,
+                                 NULL, structure,
                                  MAX_PAST, past, surface, MAX_FUTURE, future,
                                  &src_rect, output, &dst_rect, NULL, 0, NULL);
     if (err != VDP_STATUS_OK)
diff --git a/modules/hw/vdpau/deinterlace.c b/modules/hw/vdpau/deinterlace.c
new file mode 100644 (file)
index 0000000..b0eea6b
--- /dev/null
@@ -0,0 +1,131 @@
+/*****************************************************************************
+ * deinterlace.c: VDPAU deinterlacing filter
+ *****************************************************************************
+ * Copyright (C) 2013 Rémi Denis-Courmont
+ *
+ * This program is free software; you can redistribute it and/or modify it
+ * under the terms of the GNU Lesser General Public License as published by
+ * the Free Software Foundation; either version 2.1 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 Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser 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.
+ *****************************************************************************/
+
+#ifdef HAVE_CONFIG_H
+# include "config.h"
+#endif
+
+#include <stdlib.h>
+#include <assert.h>
+
+#include <vlc_common.h>
+#include <vlc_plugin.h>
+#include <vlc_filter.h>
+#include "vlc_vdpau.h"
+
+struct filter_sys_t
+{
+    mtime_t last_pts;
+};
+
+static picture_t *Deinterlace(filter_t *filter, picture_t *src)
+{
+    filter_sys_t *sys = filter->p_sys;
+    mtime_t last_pts = sys->last_pts;
+
+    sys->last_pts = src->date;
+
+    vlc_vdp_video_field_t *f1 = src->context;
+    if (f1->structure != VDP_VIDEO_MIXER_PICTURE_STRUCTURE_FRAME)
+        return src; /* cannot deinterlace twice */
+#ifdef VOUT_CORE_GETS_A_CLUE
+    picture_t *dst = filter_NewPicture(filter);
+#else
+    picture_t *dst = picture_NewFromFormat(&src->format);
+#endif
+    if (dst == NULL)
+        return src; /* cannot deinterlace without copying fields */
+    if (vlc_vdp_video_copy(dst, src) != VDP_STATUS_OK) // shallow copy
+    {
+        picture_Release(dst);
+        return src;
+    }
+    picture_CopyProperties(dst, src);
+
+    if (last_pts != VLC_TS_INVALID)
+        dst->date = (3 * src->date - last_pts) / 2;
+    else
+    if (filter->fmt_in.video.i_frame_rate != 0)
+        dst->date = src->date + ((filter->fmt_in.video.i_frame_rate_base
+                            * CLOCK_FREQ) / filter->fmt_in.video.i_frame_rate);
+    dst->b_top_field_first = !src->b_top_field_first;
+    dst->i_nb_fields = 1;
+    src->i_nb_fields = 1;
+
+    assert(src->p_next == NULL);
+    src->p_next = dst;
+
+    vlc_vdp_video_field_t *f2 = dst->context;
+    if (src->b_progressive || src->b_top_field_first)
+    {
+        f1->structure = VDP_VIDEO_MIXER_PICTURE_STRUCTURE_TOP_FIELD;
+        f2->structure = VDP_VIDEO_MIXER_PICTURE_STRUCTURE_BOTTOM_FIELD;
+    }
+    else
+    {
+        f1->structure = VDP_VIDEO_MIXER_PICTURE_STRUCTURE_BOTTOM_FIELD;
+        f2->structure = VDP_VIDEO_MIXER_PICTURE_STRUCTURE_TOP_FIELD;
+    }
+
+    src->b_progressive = true;
+    return src;
+}
+
+static int Open(vlc_object_t *obj)
+{
+    filter_t *filter = (filter_t *)obj;
+
+    if (filter->fmt_in.video.i_chroma != VLC_CODEC_VDPAU_VIDEO_422
+     && filter->fmt_in.video.i_chroma != VLC_CODEC_VDPAU_VIDEO_420)
+        return VLC_EGENERIC;
+    if (!video_format_IsSimilar(&filter->fmt_in.video, &filter->fmt_out.video))
+        return VLC_EGENERIC;
+
+    filter_sys_t *sys = malloc(sizeof (*sys));
+    if (unlikely(sys == NULL))
+        return VLC_ENOMEM;
+
+    /* NOTE: Only weave and bob are mandatory for the hardware to implement.
+     * The other modes and IVTC should be checked. */
+
+    sys->last_pts = VLC_TS_INVALID;
+
+    filter->pf_video_filter = Deinterlace;
+    filter->p_sys = sys;
+    filter->fmt_out.video.i_frame_rate *= 2;
+    return VLC_SUCCESS;
+}
+
+static void Close(vlc_object_t *obj)
+{
+    filter_t *filter = (filter_t *)obj;
+    filter_sys_t *sys = filter->p_sys;
+
+    free(sys);
+}
+
+vlc_module_begin()
+    set_description(N_("VDPAU deinterlacing filter"))
+    set_capability("video filter2", 0)
+    set_category(CAT_VIDEO)
+    set_subcategory(SUBCAT_VIDEO_VFILTER)
+    set_callbacks(Open, Close)
+    add_shortcut ("deinterlace")
+vlc_module_end()
index 37dce53bde3e4d796b1ef303ca67c0954d3c8f8b..ce21f7935f21d75073db5b6e115c77a26f779976 100644 (file)
@@ -71,6 +71,7 @@ VdpStatus vlc_vdp_video_attach(vdp_t *vdp, VdpVideoSurface surface,
 
     field->destroy = SurfaceDestroy;
     field->frame = frame;
+    field->structure = VDP_VIDEO_MIXER_PICTURE_STRUCTURE_FRAME;
 
     atomic_init(&frame->refs, 1);
     frame->surface = surface;
@@ -91,6 +92,7 @@ VdpStatus vlc_vdp_video_copy(picture_t *restrict dst, picture_t *restrict src)
 
     fnew->destroy = SurfaceDestroy;
     fnew->frame = frame;
+    fnew->structure = fold->structure;
 
     atomic_fetch_add(&frame->refs, 1);
     return VDP_STATUS_OK;
index f792298249cc7a02351fe1574880b2ce92087347..08f0e9e092a2f3e25cc3e584b81ef5dc1f364737 100644 (file)
@@ -258,6 +258,7 @@ typedef struct vlc_vdp_video_field
 {
     void (*destroy)(void *); /* must be first @ref picture_Release() */
     vlc_vdp_video_frame_t *frame;
+    VdpVideoMixerPictureStructure structure;
 } vlc_vdp_video_field_t;
 
 /**