gstreamer/subprojects/gstreamer-vaapi/tests/internal/test-subpicture.c

172 lines
5 KiB
C

/*
* test-subpicture.c - Test GstVaapiSubpicture
*
* Copyright (C) <2011-2013> Intel Corporation
* Copyright (C) <2011> Collabora Ltd.
* Copyright (C) <2011> Thibault Saunier <thibault.saunier@collabora.com>
*
* 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 "gst/vaapi/sysdeps.h"
#include <gst/vaapi/gstvaapisurface.h>
#include "decoder.h"
#include "output.h"
#include "test-subpicture-data.h"
static inline void
pause (void)
{
g_print ("Press any key to continue...\n");
getchar ();
}
static gchar *g_codec_str;
static gdouble g_global_alpha = 1.0;
static GOptionEntry g_options[] = {
{"codec", 'c',
0,
G_OPTION_ARG_STRING, &g_codec_str,
"codec to test", NULL},
{"global-alpha", 'g',
0,
G_OPTION_ARG_DOUBLE, &g_global_alpha,
"global-alpha value", NULL},
{NULL,}
};
static void
upload_subpicture (GstBuffer * buffer, const VideoSubpictureInfo * subinfo)
{
const guint32 *const src = subinfo->data;
guint i, len = subinfo->data_size / 4;
GstMapInfo map_info;
guint32 *dst;
if (!gst_buffer_map (buffer, &map_info, GST_MAP_WRITE))
return;
dst = (guint32 *) map_info.data;
/* Convert from RGBA source to ARGB */
for (i = 0; i < len; i++) {
const guint32 rgba = src[i];
dst[i] = (rgba >> 8) | (rgba << 24);
}
gst_buffer_unmap (buffer, &map_info);
}
int
main (int argc, char *argv[])
{
GstVaapiDisplay *display;
GstVaapiWindow *window;
GstVaapiDecoder *decoder;
GstVaapiSurfaceProxy *proxy;
GstVaapiSurface *surface;
GstBuffer *buffer;
VideoSubpictureInfo subinfo;
GstVaapiRectangle subrect;
GstVideoOverlayRectangle *overlay;
GstVideoOverlayComposition *compo;
guint flags = 0;
static const guint win_width = 640;
static const guint win_height = 480;
if (!video_output_init (&argc, argv, g_options))
g_error ("failed to initialize video output subsystem");
if (g_global_alpha != 1.0)
flags |= GST_VIDEO_OVERLAY_FORMAT_FLAG_GLOBAL_ALPHA;
g_print ("Test subpicture\n");
display = video_output_create_display (NULL);
if (!display)
g_error ("could not create VA display");
window = video_output_create_window (display, win_width, win_height);
if (!window)
g_error ("could not create window");
decoder = decoder_new (display, g_codec_str);
if (!decoder)
g_error ("could not create decoder");
if (!decoder_put_buffers (decoder))
g_error ("could not fill decoder with sample data");
proxy = decoder_get_surface (decoder);
if (!proxy)
g_error ("could not get decoded surface");
surface = gst_vaapi_surface_proxy_get_surface (proxy);
subpicture_get_info (&subinfo);
buffer = gst_buffer_new_and_alloc (subinfo.data_size);
upload_subpicture (buffer, &subinfo);
/* We position the subpicture at the bottom center */
subrect.x = (gst_vaapi_surface_get_width (surface) - subinfo.width) / 2;
subrect.y = gst_vaapi_surface_get_height (surface) - subinfo.height - 10;
subrect.height = subinfo.height;
subrect.width = subinfo.width;
{
GstVideoMeta *const vmeta =
gst_buffer_add_video_meta (buffer, GST_VIDEO_FRAME_FLAG_NONE,
GST_VIDEO_OVERLAY_COMPOSITION_FORMAT_RGB,
subinfo.width, subinfo.height);
if (!vmeta)
g_error ("could not create video meta");
overlay = gst_video_overlay_rectangle_new_raw (buffer,
subrect.x, subrect.y, subrect.width, subrect.height, flags);
}
if (!overlay)
g_error ("could not create video overlay");
gst_buffer_unref (buffer);
if (flags & GST_VIDEO_OVERLAY_FORMAT_FLAG_GLOBAL_ALPHA)
gst_video_overlay_rectangle_set_global_alpha (overlay, g_global_alpha);
compo = gst_video_overlay_composition_new (overlay);
if (!compo)
g_error ("could not create video overlay composition");
gst_video_overlay_rectangle_unref (overlay);
if (!gst_vaapi_surface_set_subpictures_from_composition (surface, compo))
g_error ("could not create subpictures from video overlay compoition");
gst_vaapi_window_show (window);
if (!gst_vaapi_window_put_surface (window, surface, NULL, NULL,
GST_VAAPI_PICTURE_STRUCTURE_FRAME))
g_error ("could not render surface");
pause ();
gst_video_overlay_composition_unref (compo);
gst_vaapi_surface_proxy_unref (proxy);
gst_object_unref (decoder);
gst_object_unref (window);
gst_object_unref (display);
g_free (g_codec_str);
video_output_exit ();
return 0;
}